1 引言
可编程序控制器(PLC)都有一个编程口。以日本三菱公司生产的PLC为例(包括FX系列和A系列),其编程口为RS-422格式,根据PLC型号不同又分为8针座编程口和25针座编程口。对于后者,可直接将SC—08编程电缆将PLC的编程口和微型计算机的RS—232口连接起来;对于后者,则还需要一根转换电缆将PLC 的8针座编程口和25针编程电缆相连。无论何种情况,一旦将PLC用户程序由微型计算机编程环境传到PLC 用户程序区,其编程口大多就没有被再利用。其实,这是一种浪费。也就是说,可利用此编程口实现微型计算机和PLC 的数据通讯,将PLC的工作状态纳入微型计算机管理之下。
2 编程口操作命令类型与通讯端口初始化
串行通讯是计算机与其它机器之间进行通讯的一种常用方法,在WINDOWS操作系统中提供了实现各种串行通讯的API函数。通过SC—08编程电缆或FX232AW模块,可将微型计算机的串行通讯口RS—232和PLC 的编程口连接起来,这样微型计算机就可对PLC的RAM区数据进行读、写操作。由PLC本身所具有的特性,可对PLC进行以下四种类型的操作:
(1)位元件或字元件状态读操作(CMD0);
(2)位元件或字元件状态写操作(CMD1);
(3)位元件强制ON操作(CMD7);
(4)位元件强制OFF操作(CMD8)。
另外,在进行上述四类操作以前,首先要对端口进行初始化操作,即设定通讯协议(包括设置通讯波特率、数据位数、数据停止位及奇偶校验)。在WINDOWS的SDK中定义了一个结构DCB,该结构详细地说明了如何对通讯端口进行控制,所以通讯端口的初始化也是围绕着对这个结构的正确设置为中心进行的。用VC++语言实现端口初始化如下:
BOOL CSerial::Open(int nPort)
{
//nPort 为微型计算机串行通讯口端口号。nport=1为端口1;nPort=2为端口2。
char szPort[15];
DCB dob;
m_hIDComDev=CreateFile(szPort GENERIC_READ│GENERIC_WRITEONULLOPEN_EXISTINGFILE_ATTRIBUTE_NORMAL│FILE_FLAG_OVERLAPPEDNULL);
dcb.DCB1ength=sizeof(DCB);
GetCommState(m_hIDComDev&dcb);//取得通讯资源当前设置
dcb.BaudRate=9600;//设定波特率为9600
dcb.ByteSize=7;//7数据位
dcb.Parity=2;//偶校验
dcb.StopBits=0;//设定1个停止位
if(SetCommState(m_hIDComDev&dcb)return(TRUE);
else return(FALSE);//设置端口,若设置则返回TRUE,否则返回FALSE
}
需要说明的是CSerial是一个用于串行通讯的类,它包含了进行串行通讯的所需的函数。除上述端口初始化成员函数Open外,还包括另两个重要成员函数:一个是endData把数据从一个缓冲区发送到串行端口。另一个是ReadData从端口的接收缓冲区中读入数据。
其次,在每进行一次上述四类操作中的一种操作以前,还要进行握手联络。对PLC发请求讯号ENQ(代码为OX05),然后读PLC 的响应讯号。如果读到的响应讯号为ACK(代码为OX06),则表示PLC已准备就绪,等待接收通讯数据。握手联络VC++语言示PLC已准备就绪,等待接收通讯数据。握手联络VC++语言实现为:
BOOL CNTJD1g::ReadFromPLC(char *Read_char char *Read_addressint Read_bytes)
{
CSerial Serial;//用于串行通讯的类
char read_BUFFER;
if(Serial.Open(2)//初始化串行口通讯口COM2
{ Serial.SendData(&ENQ_request1);//发送联络讯号
Sleep(1000);//等待1秒钟
Serial.ReadData(&read_BUFFER1);//读取PLC响应讯号
if(read_BUFFER==ACK)
{ 如果PLC响应讯号等于ACK,则进行上述四种操作:}}
Serial.Close()://操作完毕后,关闭通讯口
}
3 编程口命令操作
(1)位元件或字元件状态读操作
操作对象元件:PLC内部的X、Y、M、S、T、C、D元件;命令格式:
说明:①为读命令起始标志STX,代码为OX02;
②为位元件或字元件状态读命令CMDO,命令代码为OX30;
③为读位元件或字元件的4位起始地址,高位先发,低位后发,且是以ASCII码的形式发送;
④为一次读取位元件或字元件的个数,多一次可读取OXff个字节的元件,以ASCII码的形式发送;
⑤为停止位标志ETX,代码为OX03;
⑥为2位和校验,和累计为②、③、④项代码,取其和低两位转化成ASCII码,高位先发,低位后发。
在发送完上述命令格式代码后,就可直接读取PLC响应的信息。响应信息格式如下
VC++语言实现:
BOOL CNTJDlg::ReadFromPLC(char *Read_char char
*Read_address int Read_bytes)
{
char senddatasum_CHECK[2];char readdatasum_CHECK[2]; char total_DATABYTES[2];
char readdatasum_check[2];int readdata_sum;
int datasum_check=0; int i;
Serial.SendData(&STX_start1);/向PLC发送“开始”标志代码
Serial.SendData(&CMDO_read1);//发送“读”命令代码datasum_check+=CMDO_read;
for(i=0;i<4;i++){Serial.SendData(&Read_address[i]1);//发送起始元件地址的ASCII代码datasum_check+=Read_address[i];}
Change to ASCII(total DATABYTESRead_bytes);//将字节数转化成ASCII代码
for (i=0;i<2;i++){Serial.SendData(&total_DATABYTES[i]1);//发送元件字节数的ASCII代码)datasum_check+total_DATABYTES[i];}
Serial.SendData(&ETX_end1);//发送“结束”标志代码senddatasum_CHECK+ETX_end;
Change_to_ASCII(senddatasum_CHECKsenddatasum_CHECK);//将“和”转化成ASCII码
for (i=0;i<2;i++) Serial.SendData(&senddatasum_CHECK[i]1);
Sleep(1000);//等待PLC响应
Serial.ReadData(&read_BUFFER1);
if(read_BUFFER==STX_start){
readdata_sum=0;
for(i=0;i<2*Read_bytes;i++){Serial.ReadData(&Read_char[i]1);//读Read_bytes个字节readdata_sum+Read_char[i];}
Serial.ReadData(&read_BUFFER1);
if(read_BUFFER==ETX_end){Serial.ReadData(readdatasum_CHECK2);//读入的“和”的低2位ASCII码Readdata_sum+=ETX_end;}
Change_to_ASCII(readdatasum_checkreaddata_sum);//将计算得到的“和”转化成ASCII码
if(*readdatasum_CHECK==*readdatasum_check)//“和”校验
{ AfxMessageBox(“数据读出!”)return TRUE;}
else { AfxMessageBox(“校验错误”)return FALSE.}
}
(2)位元件或字元件状态写操作
操作对象元件:同3(1);命令格式:
说明:①为写命令起始标志STX,代码为OX02;
②为位元件或字元件状态写命令CMD1,命令代码为OX31;
③为写位元件或字元件的4位起始地址,高位先发,低位后发,且是以ASCII码的形式发送;
④为一次写入位元件或字元件的个数,以ASCII码的形式发送;
⑤为待写到PLC RAM区的数据DATA,以ASCII码形式发送;
⑥为停止位标志ETX,代码为OX03;
⑦为2位和校验,和累计为②、③、④项代码,取其和低两位转化成ASCII码,高位先发,低位后发。
VC++语言实现:
BOOL CNTJDlg::WritePLC(char *data_ADDRESSchar *Write_ASCint bytesnumber)
{
char total_BYTES[2];char senddatasum_CHECK[2];
char read_BUFFER;char read_finishBUFFER;
int datasum_check=0; int i=0;
Serial.SendData(&STX_start1);//向PLC发送“开始”标志代码
datasum_check=0;Serial.SendData(&CMD1_write1);//发送“写”命令代码
datasum_check+CMD1_write;
for(i=0;i<4;i++) {Serial.SendData(&data_ADDRESS[i]1);//发送起始元件地址的ASCII码
datasum_check+=data_ADDRESS[i];
Change_to_ASCII(total_DATABYTESbytesnumber);//将字节数转化成ASCII码
for(i=0;i<2;i++)
{
Serial.SendData(&total_BYTES1);//发送元件字节数的ASCII代码
datasum_check+=total_BYTES[i];}
for {i=0;i
Serial.SendData(&Write_ASC[i]1);//发送要写入的数据的ASCII码
datasum_check+=Write_ASC[i];}
Serial.SendData(&ETX_end1);//发送“结束”标志代码
datasum_check+=ETX_end;
Change_to_ASCII(senddatasum_CHECKdatasum_check);//将“和”转化成ASCII码
Serial.SendData(&senddatasum_CHECK2);
Sleep(1000); Serial.ReadData(&read_finishBUFFER1);
if (read_finishBUFFER==ACK_reply)
{AfxMessageBox(“数据写入 OK”)return TRUE;}
else {AfxMessageBox(“数据写入失败”)return FALSE。}
(3)位元件强制ON操作
操作对象:X、Y、M、S、T、C元件;
命令格式:
说明:①为强制OFF命令起始标志STX,代码为OX02;
②为强制OFF命令CMD8,命令代码为OX38H;
③为强制OFF位元件4位起始地址,高位先发,低位后发,以ASCII码形式发送;
④为停止位标志ETX,代码为OX03;
⑤为2位和校验,和累计为②、③、④项代码,取其和低两位转化成ASCII码,高位先发,低位后发。
VC++语言实现:
void NTJDlg::ForceOffOperation (char *OFF_Address)
{
int i;
char Sum_Check[2];
char read_buffer;
int Sum=0;
Serial.SendData(&STX_start1);//向PLC发送“开始”标志代码
Serial.SendData(&CMD8_ForceOFF1);//发送“OFF”命令代码
Sum=CMD8_ForceOFF;
for (i=0;i<4;i++) {
Serial.SendData(&OFF_Address[i]1);//发送起始元件地址的ASCII码
Sum+=OFF_Address[i];}
Serial.SendData(&ETX_end1);//发送“结束”标志代码
Sum+=ETX_end;
Change_to_ASCII(Sum_CheckSum);//将“和”转化成ASCII码
Serial.SendData(&Sum_Check2);
Skeeo(1000);
Serial.ReadData(&read_buffer1);
if(read_fininhBUFFER==ACK_reply) AfxMessageBox(“OFF 操作 OK ”);
else AfxMessageBox(“OFF 操作失败”)。
}
注意:必须严格按照上述四种操作命令格式进行发送,在发送前,起始地址、数据、数据个数、校验和都必须按位转换成ASCII码。从PLC读到的数据亦是ASCII码形式,需要经过适当转换才能利用。另外,要注意强制命令地址与读写地址的顺序不是一样,且一次多只能传送64个字节数据。