1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147 |
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Net.Sockets;
- using System.Runtime.InteropServices;
- using System.Text;
- using System.Threading.Tasks;
- using System.Xml;
- using Aitex.Core.RT.DataCenter;
- using Aitex.Core.RT.Event;
- using Aitex.Core.RT.IOCore;
- using Aitex.Core.RT.Log;
- using Aitex.Core.RT.PLC;
- using Aitex.Core.RT.SCCore;
- using Aitex.Core.Util;
- using MECF.Framework.Common.IOCore;
- using MECF.Framework.RT.Core.IoProviders;
- namespace VirgoRT.Devices.IODevices
- {
- public enum BlockType
- {
- CIO = 0x30,
- WR = 0x31,
- D = 0x82,
- }
-
- public class FinsPlc :IoProvider
- {
-
- TcpClient msender;
- Socket msock;
- byte[] Client, Server;
- private bool _isOpened = false;
- private string _ip = "192.168.10.10";
- private int _port = 9600;
- private int _aoBlockStartPosition = 1000;
- private int _aiBlockStartPosition = 2000;
- private int _doBlockStartPosition = 0;
- private int _diBlockStartPosition = 20;
- private BlockType _diBlockType = BlockType.WR;
- private BlockType _doBlockType = BlockType.WR;
- private BlockType _aiBlockType = BlockType.D;
- private BlockType _aoBlockType = BlockType.D;
- R_TRIG _failedTrigger = new R_TRIG();
- DeviceTimer _timerWrite = new DeviceTimer();
- DeviceTimer _timerRead = new DeviceTimer();
- private double _averageWriteTime = 0;
- private double _averageReadTime = 0;
-
- public FinsPlc()
- {
- }
-
- protected override void SetParameter(XmlElement nodeParameter)
- {
- string strIp = nodeParameter.GetAttribute("ip");
- string strPort = nodeParameter.GetAttribute("port");
- string diBlockType = nodeParameter.GetAttribute("diBlockType");
- string doBlockType = nodeParameter.GetAttribute("doBlockType");
- string aiBlockType = nodeParameter.GetAttribute("aiBlockType");
- string aoBlockType = nodeParameter.GetAttribute("aoBlockType");
- string diStartPosition = nodeParameter.GetAttribute("diStartPosition");
- string doStartPosition = nodeParameter.GetAttribute("doStartPosition");
- string aiStartPosition = nodeParameter.GetAttribute("aiStartPosition");
- string aoStartPosition = nodeParameter.GetAttribute("aoStartPosition");
- _port = int.Parse(strPort);
- _ip = strIp;
- if (!Enum.TryParse(diBlockType, out _diBlockType))
- {
- LOG.Error($"plc config error, block type {diBlockType} not valid");
- }
- if (!Enum.TryParse(doBlockType, out _doBlockType))
- {
- LOG.Error($"plc config error, block type {doBlockType} not valid");
- }
- if (!Enum.TryParse(aiBlockType, out _aiBlockType))
- {
- LOG.Error($"plc config error, block type {aiBlockType} not valid");
- }
- if (!Enum.TryParse(aoBlockType, out _aoBlockType))
- {
- LOG.Error($"plc config error, block type {aoBlockType} not valid");
- }
- if (!int.TryParse(diStartPosition, out _diBlockStartPosition))
- {
- LOG.Error($"plc config error, start position {diStartPosition} not valid");
- }
- if (!int.TryParse(doStartPosition, out _doBlockStartPosition))
- {
- LOG.Error($"plc config error, start position {doStartPosition} not valid");
- }
- if (!int.TryParse(aiStartPosition, out _aiBlockStartPosition))
- {
- LOG.Error($"plc config error, start position {aiStartPosition} not valid");
- }
- if (!int.TryParse(aoStartPosition, out _aoBlockStartPosition))
- {
- LOG.Error($"plc config error, start position {aoStartPosition} not valid");
- }
- }
- public override void Initialize(string module, string name, List<IoBlockItem> lstBuffers, IIoBuffer buffer, XmlElement nodeParameter, string ioMappingPathFile, string ioModule)
- {
- Module = module;
- Name = name;
- _source = module + "." + name;
- _buffer = buffer;
- _nodeParameter = nodeParameter;
- _blockSections = lstBuffers;
- buffer.SetBufferBlock(_source, lstBuffers);
- SetParameter(nodeParameter);
- SetIoMap(_source, 0, ioMappingPathFile, ioModule);
- State = IoProviderStateEnum.Uninitialized;
- _thread = new PeriodicJob(50, OnTimer, name);
- }
- private int GetIndex(string addr, int first)
- {
- if (String.IsNullOrEmpty(addr))
- {
- LOG.Write("GetIndex addr is empty");
- return -1;
- }
- string[] parts = addr.Trim().ToUpper().Split('.');
- int len = parts.Length;
- if (len == 1)
- {
- string ch = parts[0].TrimStart('D');
- int index = Convert.ToUInt16(ch);
- return (index - first) ;
- }
- if (len == 2)
- {
- char[] trim = { 'W', 'C', 'I', 'O', ' ' };
- string ch = parts[0].TrimStart(trim);
- int index = Convert.ToUInt16(ch);
- int bit = Convert.ToUInt16(parts[1]);
- return (index - first) * 16 + bit;
- }
- LOG.Info("IOManager GetIndex error");
- return -1;
- }
- private void SetIoMap(string provider, int blockOffset, string xmlPathFile, string module = "")
- {
-
- XmlDocument xml = new XmlDocument();
- xml.Load(xmlPathFile);
- XmlNodeList lstDi = xml.SelectNodes("IO_DEFINE/Dig_In/DI_ITEM");
- var scConfig = SC.GetConfigItem("System.IsIgnoreSaveDB");
- var isIgnoreSaveDB = scConfig == null ? false : scConfig.BoolValue;
- //<DI_ITEM Index="0" Name="" BufferOffset="0" Addr="0" Description=""/>
- var dibuffer= _buffer.GetDiBuffer(provider);
- List<DIAccessor> diList = new List<DIAccessor>();
- foreach (var diItem in lstDi)
- {
- XmlElement element = diItem as XmlElement;
- if (element == null)
- continue;
- string index = element.GetAttribute("Index");
- string name = element.GetAttribute("Name");
- string address = element.GetAttribute("Addr");
- string description = element.GetAttribute("Description");
- if (string.IsNullOrEmpty(name) || string.IsNullOrEmpty(index) || string.IsNullOrEmpty(address))
- continue;
- name = name.Trim();
- index = index.Trim();
- string moduleName = string.IsNullOrEmpty(module) ? name : $"{module}.{name}";
- int intIndex;
- if (!int.TryParse(index, out intIndex))
- continue;
- int intBufferOffset = GetIndex(address, _diBlockStartPosition);
- if (intBufferOffset == -1)
- {
- throw new Exception($"address not valid, " + provider);
- }
- if (!dibuffer.ContainsKey(blockOffset))
- {
- throw new Exception("Not defined DI buffer from IO provider, " + provider);
- }
- DIAccessor diAccessor = new DIAccessor(moduleName, intBufferOffset, dibuffer[blockOffset], dibuffer[blockOffset]);
- diAccessor.IoTableIndex = intIndex;
- diAccessor.Addr = address;
- diAccessor.Provider = provider;
- diAccessor.BlockOffset = blockOffset;
- diAccessor.Description = description;
- diList.Add(diAccessor);
- }
- _buffer.SetIoMap(provider, blockOffset, diList);
- XmlNodeList lstDo = xml.SelectNodes("IO_DEFINE/Dig_Out/DO_ITEM");
- var dobuffer = _buffer.GetDoBuffer(provider);
- List<DOAccessor> doList = new List<DOAccessor>();
- foreach (var doItem in lstDo)
- {
- XmlElement element = doItem as XmlElement;
- if (element == null)
- continue;
- string index = element.GetAttribute("Index");
-
- string name = element.GetAttribute("Name");
- string address = element.GetAttribute("Addr");
- string description = element.GetAttribute("Description");
- if (string.IsNullOrEmpty(name) || string.IsNullOrEmpty(index) || string.IsNullOrEmpty(address))
- continue;
- name = name.Trim();
- index = index.Trim();
-
- string moduleName = string.IsNullOrEmpty(module) ? name : $"{module}.{name}";
- int intIndex;
- if (!int.TryParse(index, out intIndex))
- continue;
- int intBufferOffset = GetIndex(address, _doBlockStartPosition);
- if (intBufferOffset == -1)
- {
- throw new Exception($"address not valid, " + provider);
- }
- if (!dobuffer.ContainsKey(blockOffset))
- {
- throw new Exception("Not defined DO buffer from IO provider, " + provider);
- }
- DOAccessor doAccessor = new DOAccessor(moduleName, intBufferOffset, dobuffer[blockOffset]);
-
- doAccessor.IoTableIndex = intIndex;
- doAccessor.Addr = address;
- doAccessor.Provider = provider;
- doAccessor.BlockOffset = blockOffset;
- doAccessor.Description = description;
- doList.Add(doAccessor);
- }
- _buffer.SetIoMap(provider, blockOffset, doList);
- XmlNodeList lstAo = xml.SelectNodes("IO_DEFINE/Ana_Out/AO_ITEM");
- var aobuffer = _buffer.GetAoBuffer(provider);
- List<AOAccessor> aoList = new List<AOAccessor>();
- foreach (var aoItem in lstAo)
- {
- XmlElement element = aoItem as XmlElement;
- if (element == null)
- continue;
- string index = element.GetAttribute("Index");
-
- string name = element.GetAttribute("Name");
- string address = element.GetAttribute("Addr");
- string description = element.GetAttribute("Description");
- if (string.IsNullOrEmpty(name) || string.IsNullOrEmpty(index) || string.IsNullOrEmpty(address))
- continue;
- name = name.Trim();
- index = index.Trim();
-
- string moduleName = string.IsNullOrEmpty(module) ? name : $"{module}.{name}";
- int intIndex;
- if (!int.TryParse(index, out intIndex))
- continue;
- int intBufferOffset = GetIndex(address, _aoBlockStartPosition);
- if (intBufferOffset == -1)
- {
- throw new Exception($"address not valid, " + provider);
- }
- if (!aobuffer.ContainsKey(blockOffset))
- {
- throw new Exception("Not defined AO buffer from IO provider, " + provider);
- }
- AOAccessor aoAccessor = new AOAccessor(moduleName, intBufferOffset, aobuffer[blockOffset]);
- aoAccessor.IoTableIndex = intIndex;
- aoAccessor.Addr = address;
- aoAccessor.Provider = provider;
- aoAccessor.BlockOffset = blockOffset;
- aoAccessor.Description = description;
- aoList.Add(aoAccessor);
- }
- _buffer.SetIoMap(provider, blockOffset, aoList);
- XmlNodeList lstAi = xml.SelectNodes("IO_DEFINE/Ana_In/AI_ITEM");
- var aibuffer = _buffer.GetAiBuffer(provider);
- List<AIAccessor> aiList = new List<AIAccessor>();
- foreach (var aiItem in lstAi)
- {
- XmlElement element = aiItem as XmlElement;
- if (element == null)
- continue;
- string index = element.GetAttribute("Index");
-
- string name = element.GetAttribute("Name");
- string address = element.GetAttribute("Addr");
- string description = element.GetAttribute("Description");
- if (string.IsNullOrEmpty(name) || string.IsNullOrEmpty(index) || string.IsNullOrEmpty(address))
- continue;
- name = name.Trim();
- index = index.Trim();
-
- string moduleName = string.IsNullOrEmpty(module) ? name : $"{module}.{name}";
- int intIndex;
- if (!int.TryParse(index, out intIndex))
- continue;
- int intBufferOffset = GetIndex(address, _aiBlockStartPosition);
- if (intBufferOffset == -1)
- {
- throw new Exception($"address not valid, " + provider);
- }
- if (!aibuffer.ContainsKey(blockOffset))
- {
- throw new Exception("Not defined AI buffer from IO provider, " + provider);
- }
- AIAccessor aiAccessor = new AIAccessor(moduleName, intBufferOffset, aibuffer[blockOffset]);
-
- aiAccessor.IoTableIndex = intIndex;
- aiAccessor.Addr = address;
- aiAccessor.Provider = provider;
- aiAccessor.BlockOffset = blockOffset;
- aiAccessor.Description = description;
- aiList.Add(aiAccessor);
- }
- _buffer.SetIoMap(provider, blockOffset, aiList);
- }
- protected override void Open()
- {
- try
- {
- LOG.Write(String.Format("试图连接PLC {0}:{1}", _ip, _port));
- Close();
- _isOpened = false;
- msender = new TcpClient(_ip, _port);
- msock = msender.Client;
- msock.Send(HandShake());
- byte[] buffer = new byte[24];
- msock.Receive(buffer, SocketFlags.None);
- {
- Client = new byte[4];
- Client[0] = buffer[16];
- Client[1] = buffer[17];
- Client[2] = buffer[18];
- Client[3] = buffer[19];
- Server = new byte[4];
- Server[0] = buffer[20];
- Server[1] = buffer[21];
- Server[2] = buffer[22];
- Server[3] = buffer[23];
- }
- _isOpened = true;
- SetState(IoProviderStateEnum.Opened);
- }
- catch (Exception ex)
- {
- _failedTrigger.CLK = true;
- if (_failedTrigger.Q)
- {
- LOG.Write(ex, String.Format("Communication failed with PLC {0}:{1}", _ip, _port));
- EV.PostMessage("System", EventEnum.DefaultWarning, String.Format("Communication failed with PLC {0}:{1}", _ip, _port));
- }
- return ;
- }
- _failedTrigger.RST = true;
- _isOpened = true;
- return ;
- }
- protected override void Close()
- {
- try
- {
- if (_isOpened)
- {
- _isOpened = false;
- msender.Close();
- }
- }
- catch (Exception ex)
- {
- LOG.Write(ex);
- }
- }
- protected override bool[] ReadDi(int offset, int size)
- {
- try
- {
- byte[] data = new byte[200];
- double interval = _timerRead.GetElapseTime();
- if (interval > _averageWriteTime)
- {
- //LOG.Write(_ip + ":Max read PLC interval : " + interval);
- if (_averageReadTime < 0.1)
- _averageReadTime = interval;
- _averageReadTime = (interval + _averageReadTime) / 2;
- }
- ReadBlock<byte>(_diBlockType, (ushort)_diBlockStartPosition, (ushort)data.Length, ref data);
-
- _timerRead.Start(0);
- return Array.ConvertAll(data, x => x == 1);
- }
- catch (Exception ex)
- {
- LOG.Error(String.Format("PLC ({0}) Read exception.", _ip), ex);
- EV.PostMessage("System", EventEnum.DefaultWarning, "PLC Read exception");
- Close();
- }
- return null;
- }
- protected override short[] ReadAi(int offset, int size)
- {
- try
- {
- short[] data = new short[200];
- double interval = _timerRead.GetElapseTime();
- if (interval > _averageWriteTime)
- {
- //LOG.Write(_ip + ":Max read PLC interval : " + interval);
- if (_averageReadTime < 0.1)
- _averageReadTime = interval;
- _averageReadTime = (interval + _averageReadTime) / 2;
- }
- ReadBlock_short(_aiBlockType, (ushort)_aiBlockStartPosition, (ushort)data.Length, ref data);
- _timerRead.Start(0);
- return data;
- }
- catch (Exception ex)
- {
- LOG.Error(String.Format("PLC ({0}) Read exception.", _ip), ex);
- EV.PostMessage("System", EventEnum.DefaultWarning, "PLC Read exception");
- Close();
- }
- return null;
- }
- protected override void WriteDo(int offset, bool[] buffer)
- {
- try
- {
- _timerWrite.Start(0);
-
- WriteBlock_Byte(_doBlockType, (ushort)_doBlockStartPosition, (ushort)buffer.Length, Array.ConvertAll(buffer, x=>x?(byte)1: (byte)0));
-
- double interval = _timerWrite.GetElapseTime();
- if (interval > _averageWriteTime)
- {
- LOG.Write(_ip + ":Max write PLC interval : " + interval);
- if (_averageWriteTime < 0.1)
- _averageWriteTime = interval;
- _averageWriteTime = (_averageWriteTime + interval) / 2;
- }
- return ;
- }
- catch (Exception ex)
- {
- LOG.Error(String.Format("PLC ({0})Write exception.", _ip), ex);
- EV.PostMessage("System", EventEnum.DefaultWarning, "PLC Write exception");
- Close();
- return ;
- }
- }
- protected override void WriteAo(int offset, short[] buffer)
- {
- try
- {
- _timerWrite.Start(0);
-
- WriteBlock_short(_aoBlockType, (ushort)_aoBlockStartPosition, (ushort)buffer.Length, buffer);
- double interval = _timerWrite.GetElapseTime();
- if (interval > _averageWriteTime)
- {
- LOG.Write(_ip + ":Max write PLC interval : " + interval);
- if (_averageWriteTime < 0.1)
- _averageWriteTime = interval;
- _averageWriteTime = (_averageWriteTime + interval) / 2;
- }
- return ;
- }
- catch (Exception ex)
- {
- LOG.Error(String.Format("PLC ({0})Write exception.", _ip), ex);
- EV.PostMessage("System", EventEnum.DefaultWarning, "PLC Write exception");
- Close();
- return ;
- }
- }
-
- private void ReadBlock<T>(BlockType type, ushort addr, ushort len, ref T[] data)
- {
- byte[] FullCmd = new byte[34];
- //TCP FINS header
- FullCmd[0] = 0x46;//F
- FullCmd[1] = 0x49;//I
- FullCmd[2] = 0x4e;//N
- FullCmd[3] = 0x53;//S
- FullCmd[4] = 0;//cmd length
- FullCmd[5] = 0;
- FullCmd[6] = 0;
- FullCmd[7] = 0x1A;
- FullCmd[8] = 0;//frame command
- FullCmd[9] = 0;
- FullCmd[10] = 0;
- FullCmd[11] = 0x02;
- FullCmd[12] = 0;//err
- FullCmd[13] = 0;
- FullCmd[14] = 0;
- FullCmd[15] = 0;
- //command frame header
- FullCmd[16] = 0x80;//ICF
- FullCmd[17] = 0x00;//RSV
- FullCmd[18] = 0x02;//GCT, less than 8 network layers
- FullCmd[19] = 0x00;//DNA, local network
- FullCmd[20] = Server[3];//DA1
- FullCmd[21] = 0x00;//DA2, CPU unit
- FullCmd[22] = 0x00;//SNA, local network
- FullCmd[23] = Client[3];//SA1
- FullCmd[24] = 0x00;//SA2, CPU unit
- FullCmd[25] = Convert.ToByte(21);//SID
- FullCmd[26] = 0x01; //FINS CMD Main Request Code 0101 Read IO Area
- FullCmd[27] = 0x01; //Sub Request Code
- //Parameter
- FullCmd[28] = (byte)type;
- FullCmd[29] = (byte)((addr >> 8) & 0xFF); //Block Add
- FullCmd[30] = (byte)(addr & 0xFF);
- FullCmd[31] = 0; //Bit Add
- FullCmd[32] = (byte)((len >> 8) & 0xFF); ; //Read Count
- FullCmd[33] = (byte)(len & 0xFF);
- msock.Send(FullCmd, SocketFlags.None);
- byte[] buffer = new byte[len + 30];
- msock.Receive(buffer);
- bool Succeed = true;
- if (buffer[11] == 3)
- Succeed = CheckHeadError(buffer[15]);
- if (Succeed)//no header error
- {
- T[] buf = new T[len];
- Array.Copy(buffer, 30, buf, 0, len);
- if (CheckEndCode(buffer[28], buffer[29]))
- {
- for (int i = 0; i < len; i++)
- {
- data[i] = buf[i];
- }
- }
- }
- }
- private void ReadBlock_short(BlockType type, ushort addr, ushort len, ref short[] data)
- {
- byte[] FullCmd = new byte[34];
- //TCP FINS header
- FullCmd[0] = 0x46;//F
- FullCmd[1] = 0x49;//I
- FullCmd[2] = 0x4e;//N
- FullCmd[3] = 0x53;//S
- FullCmd[4] = 0;//cmd length
- FullCmd[5] = 0;
- FullCmd[6] = 0;
- FullCmd[7] = 0x1A;
- FullCmd[8] = 0;//frame command
- FullCmd[9] = 0;
- FullCmd[10] = 0;
- FullCmd[11] = 0x02;
- FullCmd[12] = 0;//err
- FullCmd[13] = 0;
- FullCmd[14] = 0;
- FullCmd[15] = 0;
- //command frame header
- FullCmd[16] = 0x80;//ICF
- FullCmd[17] = 0x00;//RSV
- FullCmd[18] = 0x02;//GCT, less than 8 network layers
- FullCmd[19] = 0x00;//DNA, local network
- FullCmd[20] = Server[3];//DA1
- FullCmd[21] = 0x00;//DA2, CPU unit
- FullCmd[22] = 0x00;//SNA, local network
- FullCmd[23] = Client[3];//SA1
- FullCmd[24] = 0x00;//SA2, CPU unit
- FullCmd[25] = Convert.ToByte(21);//SID
- FullCmd[26] = 0x01; //FINS CMD Main Request Code 0101 Read IO Area
- FullCmd[27] = 0x01; //Sub Request Code
- //Parameter
- FullCmd[28] = (byte)type;
- FullCmd[29] = (byte)((addr >> 8) & 0xFF); //Block Add
- FullCmd[30] = (byte)(addr & 0xFF);
- FullCmd[31] = 0; //Bit Add
- int count = len ;
- FullCmd[32] = (byte)((count >> 8) & 0xFF); ; //Read Count
- FullCmd[33] = (byte)(count & 0xFF);
- msock.Send(FullCmd, SocketFlags.None);
- byte[] buffer = new byte[31 + len * 2];
- msock.Receive(buffer);
- bool Succeed = true;
- if (buffer[11] == 3)
- Succeed = CheckHeadError(buffer[15]);
- if (Succeed)//no header error
- {
- if (CheckEndCode(buffer[28], buffer[29]))
- {
- byte[] buf = new byte[2];
- for (int i = 0; i < len; i++)
- {
- buf[0] = buffer[30 + i * 2 + 1];
- buf[1] = buffer[30 + i * 2 + 0];
-
- data[i] = BitConverter.ToInt16(buf, 0);
- }
- }
- }
- }
-
- private void WriteBlock_Byte(BlockType type, ushort addr, ushort len, byte[] data)
- {
- byte[] FullCmd = new byte[34 + len];
- //TCP FINS header
- FullCmd[0] = 0x46;//F
- FullCmd[1] = 0x49;//I
- FullCmd[2] = 0x4e;//N
- FullCmd[3] = 0x53;//S
- int cmdLen = 26 + len;
- FullCmd[4] = 0;//cmd length
- FullCmd[5] = 0;
- FullCmd[6] = (byte)((cmdLen >> 8) & 0xFF);
- FullCmd[7] = (byte)(cmdLen & 0xFF);
- FullCmd[8] = 0;//frame command
- FullCmd[9] = 0;
- FullCmd[10] = 0;
- FullCmd[11] = 0x02;
- FullCmd[12] = 0;//err
- FullCmd[13] = 0;
- FullCmd[14] = 0;
- FullCmd[15] = 0;
- //command frame header
- FullCmd[16] = 0x80;//ICF
- FullCmd[17] = 0x00;//RSV
- FullCmd[18] = 0x02;//GCT, less than 8 network layers
- FullCmd[19] = 0x00;//DNA, local network
- FullCmd[20] = Server[3];//DA1
- FullCmd[21] = 0x00;//DA2, CPU unit
- FullCmd[22] = 0x00;//SNA, local network
- FullCmd[23] = Client[3];//SA1
- FullCmd[24] = 0x00;//SA2, CPU unit
- FullCmd[25] = Convert.ToByte(21);//SID
- FullCmd[26] = 0x01; //FINS CMD Main Request Code 0101 Read IO Area
- FullCmd[27] = 0x02; //Sub Request Code
- //Parameter
- FullCmd[28] = (byte)type;
- FullCmd[29] = (byte)((addr >> 8) & 0xFF); //Block Add
- FullCmd[30] = (byte)(addr & 0xFF);
- FullCmd[31] = 0; //Bit Add
- FullCmd[32] = (byte)((len >> 8) & 0xFF); ; //Write Count
- FullCmd[33] = (byte)(len & 0xFF);
- for (int i = 0; i < len; i++)
- {
- FullCmd[34 + i] = data[i];
- }
- msock.Send(FullCmd, SocketFlags.None);
- byte[] buffer = new byte[len + 34];
- msock.Receive(buffer);
- bool Succeed = true;
- if (buffer[11] == 3)
- Succeed = CheckHeadError(buffer[15]);
- if (Succeed)//no header error
- {
- if (CheckEndCode(buffer[28], buffer[29]))
- {
- //do nothing
- }
- }
- }
- private void WriteBlock_short(BlockType type, ushort addr, ushort len, short[] data)
- {
- byte[] FullCmd = new byte[34 + len * 2];
- //TCP FINS header
- FullCmd[0] = 0x46;//F
- FullCmd[1] = 0x49;//I
- FullCmd[2] = 0x4e;//N
- FullCmd[3] = 0x53;//S
- int cmdLen = 26 + len * 2;
- FullCmd[4] = 0;//cmd length
- FullCmd[5] = 0;
- FullCmd[6] = (byte)((cmdLen >> 8) & 0xFF);
- FullCmd[7] = (byte)(cmdLen & 0xFF);
- FullCmd[8] = 0;//frame command
- FullCmd[9] = 0;
- FullCmd[10] = 0;
- FullCmd[11] = 0x02;
- FullCmd[12] = 0;//err
- FullCmd[13] = 0;
- FullCmd[14] = 0;
- FullCmd[15] = 0;
- //command frame header
- FullCmd[16] = 0x80;//ICF
- FullCmd[17] = 0x00;//RSV
- FullCmd[18] = 0x02;//GCT, less than 8 network layers
- FullCmd[19] = 0x00;//DNA, local network
- FullCmd[20] = Server[3];//DA1
- FullCmd[21] = 0x00;//DA2, CPU unit
- FullCmd[22] = 0x00;//SNA, local network
- FullCmd[23] = Client[3];//SA1
- FullCmd[24] = 0x00;//SA2, CPU unit
- FullCmd[25] = Convert.ToByte(21);//SID
- FullCmd[26] = 0x01; //FINS CMD Main Request Code 0101 Read IO Area
- FullCmd[27] = 0x02; //Sub Request Code
- //Parameter
- FullCmd[28] = (byte)type;
- FullCmd[29] = (byte)((addr >> 8) & 0xFF); //Block Add
- FullCmd[30] = (byte)(addr & 0xFF);
- FullCmd[31] = 0; //Bit Add
- int count = len ;
- FullCmd[32] = (byte)((count >> 8) & 0xFF); ; //Write Count
- FullCmd[33] = (byte)(count & 0xFF);
- try
- {
- for (int i = 0; i < len; i++)
- {
- byte[] buf = BitConverter.GetBytes(data[i]);
- FullCmd[34 + i * 2 + 0] = buf[1];
- FullCmd[34 + i * 2 + 1] = buf[0];
- }
- }
- catch (Exception ex)
- {
- LOG.Write(ex);
- }
- msock.Send(FullCmd, SocketFlags.None);
- byte[] buffer = new byte[31 + len * 2];
- msock.Receive(buffer);
- bool Succeed = true;
- if (buffer[11] == 3)
- Succeed = CheckHeadError(buffer[15]);
- if (Succeed)//no header error
- {
- if (CheckEndCode(buffer[28], buffer[29]))
- {
- //do nothing
- }
- }
- }
-
- private byte[] HandShake()
- {
- //handshake
- byte[] Handshake = new byte[20];
- Handshake[0] = 0x46;
- Handshake[1] = 0x49;
- Handshake[2] = 0x4e;
- Handshake[3] = 0x53;
- Handshake[4] = 0;
- Handshake[5] = 0;
- Handshake[6] = 0;
- Handshake[7] = 0x0c;
- Handshake[8] = 0;
- Handshake[9] = 0;
- Handshake[10] = 0;
- Handshake[11] = 0;
- Handshake[12] = 0;
- Handshake[13] = 0;
- Handshake[14] = 0;
- Handshake[15] = 0;
- Handshake[16] = 0;
- Handshake[17] = 0;
- Handshake[18] = 0;
- Handshake[19] = 0;//ask for client and server node number, the client node will allocated automatically
- return Handshake;
- }
- /// <summary>
- /// (若返回的头指令为3)检查命令头中的错误代码
- /// </summary>
- /// <param name="Code">错误代码</param>
- /// <returns>指示程序是否可以继续进行</returns>
- bool CheckHeadError(byte Code)
- {
- switch (Code)
- {
- case 0x00: return true;
- case 0x01: RaiseException("the head is not 'FINS'"); return false;
- case 0x02: RaiseException("the data length is too long"); return false;
- case 0x03: RaiseException("the command is not supported"); return false;
- }
- //no hit
- RaiseException("unknown exception"); return false;
- }
- private void RaiseException(string p)
- {
- _isOpened = false;
- }
- /// <summary>
- /// 检查命令帧中的EndCode
- /// </summary>
- /// <param name="Main">主码</param>
- /// <param name="Sub">副码</param>
- /// <returns>指示程序是否可以继续进行</returns>
- bool CheckEndCode(byte Main, byte Sub)
- {
- switch (Main)
- {
- case 0x00:
- switch (Sub)
- {
- case 0x00: return true;//the only situation of success
- case 0x01: RaiseException("service canceled"); return false;
- }
- break;
- case 0x01:
- switch (Sub)
- {
- case 0x01: RaiseException("local node not in network"); return false;
- case 0x02: RaiseException("token timeout"); return false;
- case 0x03: RaiseException("retries failed"); return false;
- case 0x04: RaiseException("too many send frames"); return false;
- case 0x05: RaiseException("node address range error"); return false;
- case 0x06: RaiseException("node address duplication"); return false;
- }
- break;
- case 0x02:
- switch (Sub)
- {
- case 0x01: RaiseException("destination node not in network"); return false;
- case 0x02: RaiseException("unit missing"); return false;
- case 0x03: RaiseException("third node missing"); return false;
- case 0x04: RaiseException("destination node busy"); return false;
- case 0x05: RaiseException("response timeout"); return false;
- }
- break;
- case 0x03:
- switch (Sub)
- {
- case 0x01: RaiseException("communications controller error"); return false;
- case 0x02: RaiseException("CPU unit error"); return false;
- case 0x03: RaiseException("controller error"); return false;
- case 0x04: RaiseException("unit number error"); return false;
- }
- break;
- case 0x04:
- switch (Sub)
- {
- case 0x01: RaiseException("undefined command"); return false;
- case 0x02: RaiseException("not supported by model/version"); return false;
- }
- break;
- case 0x05:
- switch (Sub)
- {
- case 0x01: RaiseException("destination address setting error"); return false;
- case 0x02: RaiseException("no routing tables"); return false;
- case 0x03: RaiseException("routing table error"); return false;
- case 0x04: RaiseException("too many relays"); return false;
- }
- break;
- case 0x10:
- switch (Sub)
- {
- case 0x01: RaiseException("command too long"); return false;
- case 0x02: RaiseException("command too short"); return false;
- case 0x03: RaiseException("elements/data don't match"); return false;
- case 0x04: RaiseException("command format error"); return false;
- case 0x05: RaiseException("header error"); return false;
- }
- break;
- case 0x11:
- switch (Sub)
- {
- case 0x01: RaiseException("area classification missing"); return false;
- case 0x02: RaiseException("access size error"); return false;
- case 0x03: RaiseException("address range error"); return false;
- case 0x04: RaiseException("address range exceeded"); return false;
- case 0x06: RaiseException("program missing"); return false;
- case 0x09: RaiseException("relational error"); return false;
- case 0x0a: RaiseException("duplicate data access"); return false;
- case 0x0b: RaiseException("response too long"); return false;
- case 0x0c: RaiseException("parameter error"); return false;
- }
- break;
- case 0x20:
- switch (Sub)
- {
- case 0x02: RaiseException("protected"); return false;
- case 0x03: RaiseException("table missing"); return false;
- case 0x04: RaiseException("data missing"); return false;
- case 0x05: RaiseException("program missing"); return false;
- case 0x06: RaiseException("file missing"); return false;
- case 0x07: RaiseException("data mismatch"); return false;
- }
- break;
- case 0x21:
- switch (Sub)
- {
- case 0x01: RaiseException("read-only"); return false;
- case 0x02: RaiseException("protected , cannot write data link table"); return false;
- case 0x03: RaiseException("cannot register"); return false;
- case 0x05: RaiseException("program missing"); return false;
- case 0x06: RaiseException("file missing"); return false;
- case 0x07: RaiseException("file name already exists"); return false;
- case 0x08: RaiseException("cannot change"); return false;
- }
- break;
- case 0x22:
- switch (Sub)
- {
- case 0x01: RaiseException("not possible during execution"); return false;
- case 0x02: RaiseException("not possible while running"); return false;
- case 0x03: RaiseException("wrong PLC mode"); return false;
- case 0x04: RaiseException("wrong PLC mode"); return false;
- case 0x05: RaiseException("wrong PLC mode"); return false;
- case 0x06: RaiseException("wrong PLC mode"); return false;
- case 0x07: RaiseException("specified node not polling node"); return false;
- case 0x08: RaiseException("step cannot be executed"); return false;
- }
- break;
- case 0x23:
- switch (Sub)
- {
- case 0x01: RaiseException("file device missing"); return false;
- case 0x02: RaiseException("memory missing"); return false;
- case 0x03: RaiseException("clock missing"); return false;
- }
- break;
- case 0x24:
- switch (Sub)
- { case 0x01: RaiseException("table missing"); return false; }
- break;
- case 0x25:
- switch (Sub)
- {
- case 0x02: RaiseException("memory error"); return false;
- case 0x03: RaiseException("I/O setting error"); return false;
- case 0x04: RaiseException("too many I/O points"); return false;
- case 0x05: RaiseException("CPU bus error"); return false;
- case 0x06: RaiseException("I/O duplication"); return false;
- case 0x07: RaiseException("CPU bus error"); return false;
- case 0x09: RaiseException("SYSMAC BUS/2 error"); return false;
- case 0x0a: RaiseException("CPU bus unit error"); return false;
- case 0x0d: RaiseException("SYSMAC BUS No. duplication"); return false;
- case 0x0f: RaiseException("memory error"); return false;
- case 0x10: RaiseException("SYSMAC BUS terminator missing"); return false;
- }
- break;
- case 0x26:
- switch (Sub)
- {
- case 0x01: RaiseException("no protection"); return false;
- case 0x02: RaiseException("incorrect password"); return false;
- case 0x04: RaiseException("protected"); return false;
- case 0x05: RaiseException("service already executing"); return false;
- case 0x06: RaiseException("service stopped"); return false;
- case 0x07: RaiseException("no execution right"); return false;
- case 0x08: RaiseException("settings required before execution"); return false;
- case 0x09: RaiseException("necessary items not set"); return false;
- case 0x0a: RaiseException("number already defined"); return false;
- case 0x0b: RaiseException("error will not clear"); return false;
- }
- break;
- case 0x30:
- switch (Sub)
- { case 0x01: RaiseException("no access right"); return false; }
- break;
- case 0x40:
- switch (Sub)
- { case 0x01: RaiseException("service aborted"); return false; }
- break;
- }
- //no hit
- RaiseException("unknown exception"); return false;
- }
-
- }
- //public class Converter
- //{
- // public static byte[] StructToBytes(object obj)
- // {
- // int rawsize = Marshal.SizeOf(obj);
- // IntPtr buffer = Marshal.AllocHGlobal(rawsize);
- // Marshal.StructureToPtr(obj, buffer, false);
- // byte[] rawdatas = new byte[rawsize];
- // Marshal.Copy(buffer, rawdatas, 0, rawsize);
- // Marshal.FreeHGlobal(buffer);
- // return rawdatas;
- // }
- // public static object BytesToStruct(byte[] buf, int len, Type type)
- // {
- // object rtn;
- // IntPtr buffer = Marshal.AllocHGlobal(len);
- // Marshal.Copy(buf, 0, buffer, len);
- // rtn = Marshal.PtrToStructure(buffer, type);
- // Marshal.FreeHGlobal(buffer);
- // return rtn;
- // }
- // public static void BytesToStruct(byte[] buf, int len, object rtn)
- // {
- // IntPtr buffer = Marshal.AllocHGlobal(len);
- // Marshal.Copy(buf, 0, buffer, len);
- // Marshal.PtrToStructure(buffer, rtn);
- // Marshal.FreeHGlobal(buffer);
- // }
- // public static void BytesToStruct(byte[] buf, object rtn)
- // {
- // BytesToStruct(buf, buf.Length, rtn);
- // }
- // public static object BytesToStruct(byte[] buf, Type type)
- // {
- // return BytesToStruct(buf, buf.Length, type);
- // }
- //}
- }
|