S7_SCL_Gen 为SCL生成器,用户只要配置好配置文件,运行生成器会产生 S7 CPU 对应功能所需要的 SCL 源码。
- npm:
npm install s7-scl-gen -G
- yarn:
yarn global add s7-scl-gen
- pnpm:
pnpm add s7-scl-gen -G
我们将配置文件称为GCL文件,将配置文件所在的文件夹称为GCL文件夹。
SCL生成器能够将GCL文件翻译成对应的SCL文件,后者用于西门子S7项目中。输出文件夹在GCL文件(准确说是CPU文档)中定义。
安装 S7_SCL_Gen 后,便可以使用命令 s7scl
来进行相关操作了。
命令 s7scl
有一系列子命令,比如 s7scl help
子命令即可显示该命令的帮助。
下面简要说明一些子命令的用法:
s7scl gcl [GCL文件夹路径]
执行后会在指定GCL文件夹中生成GCL样板文件,该文件夹内含样本配置文件和本说明文件README.md
,然后可以自行修改以符合业务。
如果省略GCL文件夹路径
时,则会在当前文件夹中生成一个名为GCL的文件夹用于生成GCL样板文件。
s7scl convert [GCL文件夹路径]
省略GCL文件夹路径
时,默认为当前文件夹 .
,这时需要先进入到对应的GCL配置文件夹下。
事实上,如果GCL文件夹为当前文件夹的话,convert
子命令也可以省略,用下方命令即可完成当前文件夹的配置文件转换。
s7scl
执行 s7scl help
查看生成器命令帮助
s7scl watch [GCL文件夹路径]
用于持续转换,在GCL文件夹中对GCL的任何修改,都会触发转换器生成新的SCL文件。
配置文档采用YAML语法,包含配置文档的YAML文件在这里称为GCL文件。
一个GCL文件可以包括多个文档,下面的示例有2个配置文档:
--- # 文档分界指令
#CPU: AS1 # 指示属于哪一个CPU
#feature: CPU # 指示本配置的功能
name: AS1-CPU
platform: step7 # 运行平台
device: CPU410-5H
symbols:
- [Clock_Byte, MB10] # 设置时钟存储器为MB10
options:
output_dir : SCL # 设置输出文件夹
--- # 文档分界指令
# CPU: AS1 # 指示属于哪一个CPU
# feature: AI # 指示本配置的功能
# name 指令相当于 CPU 和 feature 指令的组合
name: AS1-AI
list: # 功能项列表
- comment: 气温
DB: [TIT002, DB+]
input: [AI01-03, PIW516]
$zero: "-40.0"
$span: "80.0"
- comment: 液位
DB: [LIT010, DB+]
input: '"RecvDB".Tank1'
... # 文档结束指令
---
YAML语法,它指示一个配置文档开始;...
YAML语法,它指示一个配置文档结束;- 一个配置文档为一个基本配置单位,不可分割;
- 可以在一个文件里书写多个配置文档,当然也可以将配置文档分散在多个文件中;
- 每个文档的根属性称为指令,比如上方的
feature
CPU
list
options
注意指令兼容性,运行 s7scl -v
查看生成器的当前指令版本。
配置文档必须有 name
指令,或 CPU
,feature
组合指令,两种方法用一种。
这2个指令值类型都是字符串,name 相当于后2者组合而成 <CPU>-<feature>
,作用一样,推荐 name 便于理解文档唯一性。
每个文档的 name 必须唯一,即相同 CPU 和 feature 组合的配置文档只能有一个。
目前只实现了9种类型配置文档——CPU文档和8种功能文档,由 feature
指令指示,不区分大小写。分别是:
CPU
CPU文档 指示这是一个CPU功能,使用该CPU的其它配置文档都共享的资源、信息和指令。 比如指示 CPU 所属的平台、输出文件夹、共用的符号和包含等。 所有属于同一CPU的配置文件,会在生成代码时,统一检查资源冲突情况、统一资源分配、自动合并符号表,避免占用同一DB块、同一连接等。AI
AI通道的模拟量(含超限报警)功能文档Alarm
过程量限值与报警功能文档 用于非AI通道过来的过程值报警,比如485接收的过程值PI
脉冲量转换功能文档SC
串行轮询功能文档 用在RS232 RS422 RS485通信功能中。 该功能包括了 modbus RTU 轮询,故指令也可以写为MB
modbusRTU
ModbusTCP
modbusTCP 轮询功能文档 指令名称也可以缩写为MT
Valve
阀门控制功能文档Motor
电机控制功能文档Interlock
报警连锁功能文档(实现最简单的输入_或运算_后上升沿输出)Timer
计时功能文档
具体功能文档的配置和说明可参看 example 目录下的YAML文件
- 类型: 键值对
一些额外设定,比如 options.output_file 设定输出文件名
- 类型: 数组
- 数组元素: S7符号定义
每类配置文档都内置了一些符号,内置符号都有默认地址。
内置符号通常不必书写,但如果存在地址冲突,可以在 symbols 列表中重写内置符号以更改地址和注释,但内置符号的名称不可更改。
- 类型: 字符串数组|字符串
indludes指示要在当前功能输出文件中附加包含的SCL代码,并将代码内容合并在输出文件开始处。
includes指令值分2种:
I. 字符串 表示直接将字符串作为SCL代码合并
例:
includes: |
DATA_BLOCK "RecvDB"
STRUCT
ID: INT;
Tank1 : WORD;
END_STRUCT;
BEGIN
Tank1 := W#16#02D0;
END_DATA_BLOCK
II. 数组 表示文件列表,列表项为同目录下的相对路径名称,或是包含名称和编码格式的对象,转换程序会提取每一个文件内容作为合并来源
例:
includes:
- JSFlow.scl # 只有文件名称的写法,默认该文件编码格式为 UTF-8
- filename: FXGasFlow.scl # 文件对象的写法
encoding: utf8 # 指明文件编码
- {filename: JS_Flow.scl, encoding: gbk} # 写在一行里的文件对象,指明文件编码
includes只能采用上述2种之一,由于在YAML中书写SCL代码有很多局限,推荐用外部文件的方式。
注意: includes 所指向的额外SCL代码需要用户自己编写,生成程序不检查其语法错误。
[!tip] 高级应用1 外部SCL文件可以使用
{{ expression }}
替换符,在主 yaml 文件attributes
部分定义要替换要字符串,实现更好的复用。
DATA_BLOCK "PE{{ tag_postfix_number }}"
{{if platform == "step7"}}_
{ S7_m_c := 'true'}
{{endif}}_
{{if platform == "portal"}}_
{ S7_Optimized_Access := 'FALSE' }
{{endif}}_
AUTHOR:Goosy
FAMILY:GooLib
STRUCT
U_VFD {S7_m_c := 'true'}: INT ; // 506 变频器电压
I_VFD {S7_m_c := 'true'}: INT ; // 507 变频器电流
Ua {S7_m_c := 'true'}: INT ; // 511 AA_电压
Ub {S7_m_c := 'true'}: INT ; // 512 BB_电压
Uc {S7_m_c := 'true'}: INT ; // 513 CC_电压
error {S7_m_c := 'true'}: INT ; // 515 故障代码
END_STRUCT;
BEGIN
END_DATA_BLOCK
[!tip] 高级应用2 可以在外部SCL文件中用
(**
和**)
两行包裹注释,这些注释不会输出在最终的SCL中:
(**
description: |
Syntax_ID: BYTE; Always 10 Hex
DataType: BYTE; Code for data type
1 BOOL 2 BYTE 3 CHAR 4 WORD 5 INT 6 DWORD 7 DINT 8 REAL
count: WORD; Number of Byte
DB_Number: WORD; Numbet of DB
Byte_Pointer: DWORD; Pointer to bit- and byte address
**)
TYPE UDT_ANY_Pointer
STRUCT
Syntax_ID: BYTE;
DataType: BYTE;
count: WORD;
DB_Number: WORD;
Byte_Pointer: DWORD;
END_STRUCT
END_TYPE
- 类型: 字符串数组
指示要额外复制的文件或文件夹,会在输出文件夹中生成同样名称文件。
files数组的每一项是相对于当前配置文件目录的相对路径。必须使用 /
符为路径分隔符。
注意:
- 与includes不同,files 只能采用外部文件的方式,并且可以是任何类型的文件(建议SCL或AWL代码文件)
- 只有文件名的方式,会只复制,不进行编码转换
- 指明编码格式的文件,会转换成 GBK 保存的目标文件夹中。(GBK格式是为了方便西门子软件导入文件)
- 默认不会将路径复制到目标目录中,在输出文件夹(这里假设是
output_dir
)中只有路径最后的文件或文件夹。
大多数情况下,都不用指明编码,只要书写文件名就可以了。
假设目标文件夹是 output_dir ,则:
files:
- a/b/c.scl # 复制后的文件为 `output_dir/c.scl` 仅仅复制文件
- ../readme.docx # 复制后的文件为 `output_dir/readme.docx` 非文本文件一定只写名称
- foo/a_folder # 文件夹复制,产生目标文件夹 `output_dir/a_folder`,并包含该文件夹下的文件
[!tip] 高级应用1 如果需要复制相对路径,可用"//"放置在需保留的路径范围之前。
files:
- `- os//ab/c.scl` # 会将 `os/ab/c.scl` 复制为 `output_dir/ab/c.scl`
- `- ../..//lib/c.scl` # 会将 `../lib/c.scl` 复制为 `output_dir/lib/c.scl`
[!tip] 高级应用2 可以指明源文件的编码格式,这样可以转换成西门子认识的 GBK 编码格式
files:
- filename: myfunc.scl # 复制后的文件为 `output_dir/myfunc.scl`
encoding: utf8 # 指明文件编码是 UTF-8 ,复制后的文件编码强制为GBK
- {filename: folder, encoding: gbk} # 在一行里附加编码格式,这里复制 `folder` 文件夹,里面的源文件编码都是GBK
[!tip] 高级应用3 可以使用 glob 匹配符。
files:
- `- lib/*` # 复制 lib 目录下的所有文件,不包括子目录
- filename: - lib//**.scl # 复制 lib 目录及其子目录下的所有 word 文件
encoding: utf8 # 所有匹配的文件,其编码都是 UTF-8
注意:生成器不检查文件内容的错误,也不解析文件。
- list 对应功能的列表 类型: 对象列表
- loop_begin 附加代码 类型: 字符串 指令值为SCL代码,合并在当前循环函数体开始处
- loop_end 附加代码 类型: 字符串 指令值为SCL代码,合并在当前循环函数体末尾处
对具体指令下的某个配置项,它的值通常为以下几类之一。对于具体某个配置项的类型,可参看样本配置文件。
可用的字面量有2个: true
false
不区分大小写。
配置项值为数字,其字面量可以是十进制数字,也可以是16进制的数字。16进制字面量的例子: 0x4A98
数字值的例子有 AI 配置中的 zero
span
等。
大多数情况下字符值串可以不使用引号,具体参看在YAML语法。
字面量形式同字符串,但要求其内容是一个标准的SCL表达式,具体要求以SCL语法为准。
如果字面量中有双引号,要依照YAML语法要求,用单引号包裹表达式,比如 'NOT "TIT001".AL_Flag'
S7符号对应指西门子软件中的符号,通常有名称、地址、类型。在配置文件有很多配置项要求是一个S7符号。
该配置项的值通常用一个YAML方括号语法的数组来指定,形式为 [名称, 地址, 类型, 注释]
,数组后二项可省略,类型必须是一个有效的S7类型。
比如 [recvDB, DB100, FB512, 接收块]
就定义一个名为“recvDB”的DB块符号,该DB为FB512的背景块。[length, M100, INT, 长度]
也是一个有效的M区域符号。
每个符号只能定义一次,即名称和地址不得重复,否则转换器会提示出错。
所有FB、FC、UDT符号的类型一定是它自身,所以这三种符号的type可以省略,DB符号省略类型时默认类型为自身。
在symbols指令的每一个配置项,其值必须是符号定义。
如果已配置过一个符号定义,在另一处要使用相同的S7符号,可以简单地用符号名称来引用。
比如可以用 recvDB
来指示上面 [recvDB, DB100, FB512, 接收块]
定义的S7符号。
这2种类型为以上配置项类型的组合。
有些配置项可以是多种配置值类型之一。
比如AI配置文档中的 DB
和 input
配置项,可以是符号定义,也可以是符号引用。
再比如interlock配置文档中的 input_list
中的每个元素,配置项类型可以是对象,也可以是符号定义,也可以是符号引用,也可以是SCL表达式。