概述 ----本例说明如何以自由协议实现计算机与S7-200的通信,计算机作为主站,可以实现对PLC从站各寄存器的读/写操作。 ----计算机通过COM口发送指令到PLC的PORT0(或PORT1)口,PLC通过RCV接收指令,然后对指令进行译码,译码后调用相应的读/写子程序实现指令要求的操作,并返回指令执行的状态信息。
通信协议 ----在自由口模式下,通信协议是由用户定义的。用户可以用梯形图程序调用接收中断、发送中断、发送指令(XMT)、接受指令(RCV)来控制通信操作。在自由口模式下,通信协议完全由梯形图程序控制。
指令格式定义
- 计算机每次发送一个33字节长的指令来实现一次读/写操作,指令格式见表1 说明:
- 起始字符
----起始字符标志着指令的开始,在本例中被定义为ASCII码的"g",不同的PLC从站可以定义不同的起始字符以接收真对该PLC的指令。
- 指令类型
----该字节用来标志指令的类型,在本例中05H代表读操作,06H代表写操作。
- 目标PLC站地址
----目标PLC站地址占用指令的B2、B3两个字节,以十六进制ASCII码的格式表示目标PLC的站地址。
- 目标寄存器地址
----在PLC内部可以用4个字节来表示一个寄存器的地址(但不能表示一个位地址)。前两个字节表示寄存器类型,后两个字节表示寄存器号。
| 00 00(H): |
|
I寄存器区 |
| 01 00(H): |
|
Q寄存器区 |
| 02 00(H): |
|
M寄存器区 |
| 08 00(H): |
|
V寄存器区 | 例如:
| IB000的地址可表示为 |
|
00 00 00 00(H) |
| VB100的地址可表示为 |
|
08 00 00 64(H) |
- 读/写字节数M
----当读命令时,始终读回从目标寄存器开始的连续8个字节的数据(转换为十六进制ASCII码后占用16个字节),可以根据自己的需要取用,M可以任意写入。 ----当写命令时,M表示的是要写入数据的十六进制ASCII码所占用的字节数。例如要写入1个字节的数据,数据在指令中以十六进制ASCII码表示,它将占用2个字节,此时应向M中写入"02"。同理,如果要写入5个字节的数据,M中应写入"0A"。
- 要写入的数据
----要写入的数据在指令中以十六进制ASCII码的格式表示,占用指令的B14-B29共16个字节。数据区必须填满,但只有前M个字节的数据会被写入目标寄存器。一条指令最多可以写入8个字节的数据(此时M中应写入"10",代表十进制的16)
- BCC校验码
----在传输过程中,指令有可能受到任何的干扰而使原来的数据信号发生扭曲,此时的指令当然是错误的,为了侦测指令在传输过程中发生的错误,接收方必须对指令作进一步的确认工作,以防止错误的指令被执行,最简单的方法就是使用校验码。BCC校验码的方法就是将要传送的字符串的ASCII码以字节为单位作异或和,并将此异或和作为指令的一部分传送出去;同样地,接收方在接到指令后,以相同的方式对接收到的字符串作异或和,并与传送方所送过来的值作对比,若其值相等,则代表接收到的指令是正确的,反之则是错误的。 ----在本例中,bcc为指令B1到B29的异或和,BCC为bcc的十六进制ASCII码。 ----bcc=B1 xor B2 xor B3 xor B4 xor …… xor B29
- 结束字符
----结束字符标志着指令的结束,在本例中被定义为ASCII码的"G",不同的PLC从站可以定义不同的结束字符以接收真对该PLC的指令。
- PLC在接到上位机指令后,将发送一个21字节长反馈信息,格式见表2
说明:
- 起始字符
----起始字符标志着反馈信息的开始,在本例中被定义为ASCII码的"g",不同的PLC从站可以定义不同的起始字符,这样上位机可以根据信息的起始字符来判断反馈信息的来源。
- 状态信息
----该字节包含指令执行的状态信息,在本例中
| 01H |
|
代表 |
|
读取正确 |
| 02H |
|
代表 |
|
写入正确 |
| 03H |
|
代表 |
|
BCC校验码错误 |
| 04H |
|
代表 |
|
指令不合法 |
- 数据区
----反馈信息的B3到B18为读指令所要读取的数据,以十六进制ASCII码表示。
- BCC校验码
----与上位机指令中的BCC校验码类似,它是反馈信息B3到B18的异或和。
- 结束字符
----结束字符标志着反馈信息的结束,在本例中被定义为26H。
指令中为何要使用ASCII码 ----一条指令除包含数据外,还包含必要的控制字(起始字符、结束字符、指令类型等)。如果指令中的数据直接以其原本的形式传输,则不可避免的会与指令中的控制字发生混淆。 ----例如本例中,指令的起始字符为"g",其ASCII码值为67H,结束字符为"G",其ASCII码值为47H。假设要写入的数据中也有47H,并且数据直接以其原本的形式传输,则PLC会因为接收到了数据中的47H而停止接收,这样PLC接收到的指令将是一个不完整的非法指令,很可能造成PLC的误动作。 ----为了避免这种情况的发生,可以用文本来传送二进制数据。通过以16进制ASCII码的格式描述数据,每个二进制的字节都可以表示成一对ASCII编码,这对编码表示这个字节的两个16进制字符。这种格式可以表示任何的数值,仅仅使用ASCII代码的30H到39H(表示0到9)和41H到46H(表示A到F)。ASCII码的其余部分可以用作控制字(起始标志、结束标志、指令类型等)。这样,数据中的47H以ASCII码的形式进行传送就变成了34H 37H 两个字节,从而避免了PLC因接收到数据中的47H而停止接收的错误。
表1 上位机指令格式
| Byte0 |
起始字符 |
| Byte1 |
指令类型(读/写) |
| Byte2 |
目标PLC站地址(十六进制ASCII码) |
| Byte3 |
| Byte4 |
目标寄存器地址(十六进制ASCII码) |
| Byte5 |
| Byte6 |
| Byte7 |
| Byte8 |
| Byte9 |
| Byte10 |
| Byte11 |
| Byte12 |
读/写字节数M(十六进制ASCII码) |
| Byte13 |
| Byte14 |
要写入的数据(十六进制ASCII码) |
| Byte15 |
| Byte16 |
| Byte17 |
| Byte18 |
| Byte19 |
| Byte20 |
| Byte21 |
| Byte22 |
| Byte23 |
| Byte24 |
| Byte25 |
| Byte26 |
| Byte27 |
| Byte28 |
| Byte29 |
| Byte30 |
BCC校验码(十六进制ASCII码) |
| Byte31 |
| Byte32 |
结束字符 |
表2 反馈信息格式
| Byte0 |
起始字符 |
| Byte1 |
状态信息 |
| Byte2 |
数据区(十六进制ASCII码) |
| Byte3 |
| Byte4 |
| Byte5 |
| Byte6 |
| Byte7 |
| Byte8 |
| Byte9 |
| Byte10 |
| Byte11 |
| Byte12 |
| Byte13 |
| Byte14 |
| Byte15 |
| Byte16 |
| Byte17 |
| Byte18 |
BCC校验码(十六进制ASCII码) |
| Byte19 |
| Byte20 |
结束字符 |
PLC程序执行过程 ----PLC在第一次扫描时执行初始化子程序,对端口及RCV指令进行初始化。初始化完成后,运行RCV指令使端口处于接受状态。 ----RCV会将以"g"开头"G"结尾的指令保存到接收缓冲区,并同时产生接收完成中断。 ----RCVcomplete中断服务程序用来处理接收完成中断事件,它会将接收缓冲区中的十六进制ASCII码还原成数据并保存,同时置位Verify子程序的触发条件(M0.1)。 ----Verify子程序首先复位本身的触发条件以防止子程序被重复调用,然后求出接收缓冲区中指令的BCC校验码并与指令中的BCC校验码进行比对。如果相等则置BCC码校验正确的标志位(M0.0)为1;如果指令格式正确(指令的结束标志在接收缓冲区中特定的位置VB133)而BCC码不相等,则发送代表BCC校验码错误的反馈信息;如果指令格式不正确(VB133中不是指令的结束标志),则返回代表指令格式错误的反馈信息。 ----Read子程序的触发条件为:指令中的站地址与本机站地址相符、指令类型为读指令、BCC检验码正确。当条件满足时,Read子程序被执行。Read子程序首先禁止RCV,然后将指令所要读取的数据转换成十六进制ASCII码并写入发送缓冲区、计算BCC检验码、最后发送反馈信息。 ----Write子程序的触发条件为:指令中的站地址与本机站地址相符、指令类型为写指令、BCC检验码正确。当条件满足时,Write子程序被执行。Write子程序首先禁止RCV,然后将指令中的数据写入目标寄存器,最后发送代表写入正确的反馈信息。 ----PLC每接到一条指令后都会发送一条反馈信息,当反馈信息发送完成时,会产生发送完成中断,XMTcomplete中断服务程序用来处理发送完成中断事件。在XMTcomplete中断服务程序中所要执行的操作包括:复位BCC校验码正确的标志位(M0.0);允许RCV;bcc码寄存器清零;重新装入用于计算BCC校验码的地址指针;接收缓冲区中存放指令结束字符的字节VB133清零(用来判断下一条指令格式是否正确)。
PLC寄存器地址分配 ----此程序占用PLC寄存器的VB100-VB199,内部继电器占用M0.0和M0.1。寄存器地址分配见表3、表4、表5、表6。
表3 接收缓冲区
| VB100 |
字符数 |
| VB101 |
起始字符 |
Byte0 |
| VB102 |
指令类型(读/写) |
Byte1 |
| VB103 |
目标PLC站地址(十六进制ASCII码) |
Byte2 |
| VB104 |
Byte3 |
| VB105 |
目标寄存器地址(十六进制ASCII码) |
Byte4 |
| VB106 |
Byte5 |
| VB107 |
Byte6 |
| VB108 |
Byte7 |
| VB109 |
Byte8 |
| VB110 |
Byte9 |
| VB111 |
Byte10 |
| VB112 |
Byte11 |
| VB113 |
读/写字节数M(十六进制ASCII码) |
Byte12 |
| VB114 |
Byte13 |
| VB115 |
要写入的数据(十六进制ASCII码) |
Byte14 |
| VB116 |
Byte15 |
| VB117 |
Byte16 |
| VB118 |
Byte17 |
| VB119 |
Byte18 |
| VB120 |
Byte19 |
| VB121 |
Byte20 |
| VB122 |
Byte21 |
| VB123 |
Byte22 |
| VB124 |
Byte23 |
| VB125 |
Byte24 |
| VB126 |
Byte25 |
| VB127 |
Byte26 |
| VB128 |
Byte27 |
| VB129 |
Byte28 |
| VB130 |
Byte29 |
| VB131 |
BCC校验码(十六进制ASCII码) |
Byte30 |
| VB132 |
Byte31 |
| VB133 |
结束字符 |
Byte32 |
表4 译码区
| VB134 |
PLC站号(ATH from VB103-VB104) |
| VB135 |
合成为VD135作为目标寄存器的地址指针 (ATH from VB105-VB112) |
| VB136 |
| VB137 |
| VB138 |
| VB139 |
读/写字节数(ATH from VB113-VB114) |
| VB140 |
bcc码(ATH from VB131-VB132) |
| VB141 |
未使用 |
| VB142 |
| VB143 |
| VB144 |
| VB145 |
| VB146 |
| VB147 |
| VB148 |
| VB149 |
和成为VD149作为VB102的地址指针 用以计算BCC校验码 |
| VB150 |
| VB151 |
| VB152 |
表5 发送缓冲区
| VB153 |
字符数 |
| VB154 |
起始字符 |
Byte0 |
| VB155 |
状态信息 |
Byte1 |
| VB156 |
数据区(十六进制ASCII码) |
Byte2 |
| VB157 |
Byte3 |
| VB158 |
Byte4 |
| VB159 |
Byte5 |
| VB160 |
Byte6 |
| VB161 |
Byte7 |
| VB162 |
Byte8 |
| VB163 |
Byte9 |
| VB164 |
Byte10 |
| VB165 |
Byte11 |
| VB166 |
Byte12 |
| VB167 |
Byte13 |
| VB168 |
Byte14 |
| VB169 |
Byte15 |
| VB170 |
Byte16 |
| VB171 |
Byte17 |
| VB172 |
BCC校验码(十六进制ASCII码) |
Byte18 |
| VB173 |
Byte19 |
| VB174 |
结束字符 |
Byte20 |
表6 其它
| VB175 |
合成为VW175 作为接收时计算bcc码循环的INDX |
| VB176 |
| VB177 |
合成为VW177 作为发送时计算bcc码循环的INDX |
| VB178 |
| VB179 |
接收数据的bcc码 |
| VB180 |
发送数据的bcc码 |
| VB181 |
合成为VD181作为VB156的地址指针 (计算发送反馈信息的bcc码时使用) |
| VB182 |
| VB183 |
| VB184 |
| VB185至VB198 |
未使用 |
| VB199 |
本机站号 |
程序清单
主程序:
| NETWORK 1 |
| LD |
SM0.1 |
//第一次扫描调用初始化子程序 |
| CALL |
initialize |
|
| |
| NETWORK 2 |
| LDB= |
VB134, VB199 |
//指令中的站地址与本机站地址相符 |
| AB= |
VB102, 5 |
//指令类型为读指令 |
| A |
M0.0 |
//BCC码校验正确 |
| CALL |
Read |
//调用读子程序 |
| |
| NETWORK 3 |
| LDB= |
VB134, VB199 |
//指令中的站地址与本机站地址相符 |
| AB= |
VB102, 6 |
//指令类型为写指令 |
| A |
M0.0 |
//BCC码校验正确 |
| CALL |
Write |
//调用写子程序 |
| |
| NETWORK 4 |
| LD |
M0.1 |
//指令接收完成后调用BCC码校验子程序 |
| CALL |
Verify |
|
| |
| NETWORK 5 |
| LD |
SM4.5 |
//当端口空闲时启动RCV |
| RCV |
VB100, 0 |
|
Read子程序:
| NETWORK 1 |
| LD |
SM0.0 |
//停止端口0的接收 |
| R |
SM87.7, 1 |
|
| R |
M0.0, 1 |
|
| RCV |
VB100, 0 |
|
| |
| NETWORK 2 |
| LD |
SM0.0 |
//将数据写入发送缓冲区 |
| MOVB |
103, VB154 |
|
| MOVB |
1, VB155 |
|
| HTA |
*VD135, VB156, 16 |
|
| MOVB |
26, VB174 |
|
| MOVB |
21, VB153 |
|
| |
| NETWORK 3 |
| LD |
SM0.0 |
//计算BCC校验码 |
| FOR |
VW177, +1, +16 |
|
| |
| NETWORK 4 |
| LD |
SM0.0 |
|
| XORB |
*VD181, VB180 |
|
| |
| NETWORK 5 |
| LD |
SM0.0 |
|
| INCD |
VD181 |
|
| |
| NETWORK 6 |
| NEXT |
|
|
| |
| NETWORK 7 |
| LD |
SM0.0 |
|
| HTA |
VB180, VB172, 2 |
//BCC校验码写入发送缓冲区 |
| |
| NETWORK 8 |
| LD |
SM4.5 |
//发送反馈信息 |
| XMT |
VB153, 0 |
|
Write子程序:
| NETWORK 1 |
| LD |
SM0.0 |
//停止端口0的接收 |
| R |
SM87.7, 1 |
|
| R |
M0.0, 1 |
|
| RCV |
VB100, 0 |
|
| |
| NETWORK 2 |
| LD |
SM0.0 |
//装入要写如数据源的地址指针 |
| MOVD |
&VB115, VD145 |
|
| |
| NETWORK 3 |
| LD |
SM0.0 |
//写入数据 |
| ATH |
*VD145, *VD135, VB139 |
|
| |
| NETWORK 4 |
| LD |
SM0.0 |
//指令执行的反馈信息写入发送缓冲区 |
| MOVB |
21, VB153 |
|
| MOVB |
103, VB154 |
|
| MOVB |
2, VB155 |
|
| MOVB |
26, VB174 |
|
| |
| NETWORK 5 |
| LD |
SM4.5 |
//发送指令执行的反馈信息 |
| XMT |
VB153, 0 |
|
Verify子程序:
| NETWORK 1 |
| LD |
SM0.0 |
|
| R |
M0.1, 1 |
//复位verify子程序的执行条件 |
| |
| NETWORK 2 |
| LD |
SM0.0 |
//计算BCC码 |
| FOR |
VW175, +1, +29 |
|
| |
| NETWORK 3 |
| LD |
SM0.0 |
|
| XORB |
*VD149, VB179 |
|
| |
| NETWORK 4 |
| LD |
SM0.0 |
|
| INCD |
VD149 |
|
| |
| NETWORK 5 |
| NEXT |
|
|
| |
| NETWORK 6 |
| LDB= |
VB179, VB140 |
//当BCC码校验正确时,M0.0置1 |
| AB= |
VB133, 71 |
|
| S |
M0.0, 1 |
|
| |
| NETWORK 7 |
| LDB= |
VB133, 71 |
//BCC码错误时发送反馈信息 |
| AB<> |
VB179, VB140 |
|
| MOVB |
21, VB153 |
|
| MOVB |
103, VB154 |
|
| MOVB |
3, VB155 |
|
| MOVB |
26, VB174 |
|
| R |
SM87.7, 1 |
|
| RCV |
VB100, 0 |
|
| XMT |
VB153, 0 |
|
| |
| NETWORK 8 |
| LDB<> |
VB133, 71 |
//指令格式错误或RCV超时时发送反馈信息 |
| MOVB |
21, VB153 |
|
| MOVB |
103, VB154 |
|
| MOVB |
4, VB155 |
|
| MOVB |
26, VB174 |
|
| R |
SM87.7, 1 |
|
| RCV |
VB100, 0 |
|
| XMT |
VB153, 0 |
|
Initialize子程序:
| NETWORK 1 |
| LD |
SM0.0 |
|
| MOVB |
9, SMB30 |
//0口"9600,N,8,1" |
| |
| NETWORK 2 |
| LD |
SM0.0 |
//RCV指令初始化 |
| MOVB |
16#EC, SMB87 |
|
| MOVB |
103, SMB88 |
|
| MOVB |
71, SMB89 |
|
| MOVB |
+1000, SMW92 |
|
| MOVB |
35, SMB94 |
|
| R |
SM87.2, 1 |
|
| |
| NETWORK 3 |
| LD |
SM0.0 |
|
| ATCH |
RCVcomplete, 23 |
//连接口0接收完成的中断 |
| |
| NETWORK 4 |
| LD |
SM0.0 |
|
| ATCH |
XMTcomplete, 9 |
//连接口0发送完成的中断 |
| |
| NETWORK 5 |
| LD |
SM0.0 |
|
| ENI |
|
//中断允许 |
| |
| NETWORK 6 |
| LD |
SM0.0 |
|
| MOVB |
2, VB199 |
//将本机站地址装入寄存器 |
| |
| NETWORK 7 |
| LD |
SM0.0 |
|
| MOVB |
&VB102, VD149 |
//装入地址指针 |
| MOVB |
0, VB179 |
//BCC码寄存器清零 |
| MOVB |
&VB156, VD181 |
//装入地址指针 |
| MOVB |
0, VB180 |
//BCC码寄存器清零 |
RCVcomplete中断程序
| NETWORK 1 |
| LD |
SM0.0 |
|
| ATH |
VB103, VB134, 2 |
//指令译码(ASCII码到十六进制) |
| ATH |
VB105, VB135, 8 |
|
| ATH |
VB113, VB139, 2 |
|
| ATH |
VB131, VB140, 2 |
|
| S |
M0.1, 1 |
//置位Verify子程序的触发条件 |
| MOVB |
0, VB179 |
//BCC码寄存器清零 |
| MOVD |
&VB102, VD149 |
//装入地址指针 |
XMTcomplete中断程序
| NETWORK 1 |
| LD |
SM0.0 |
|
| R |
M0.0, 1 |
//复位BCC校验码正确的标志位 |
| S |
SM87.7, 1 |
//允许口0进行接收 |
| MOVB |
0, VB179 |
//BCC校验码寄存器清零 |
| MOVB |
0, VB180 |
//BCC校验码寄存器清零 |
| MOVD |
&VB102, VD149 |
//重新装入地址指针 |
| MOVD |
&VB156, VD181 |
|
| MOVB |
0, VB133 |
//接收缓冲区中存放指令结束字符的字节 |
|