FestoSocketSimulator.cs 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. using Aitex.Common.Util;
  2. using Aitex.Core.RT.Device;
  3. using Aitex.Core.RT.IOCore;
  4. using Aitex.Core.Util;
  5. using MECF.Framework.Common.Device.Festo;
  6. using MECF.Framework.Common.Net;
  7. using MECF.Framework.Simulator.Core.Driver;
  8. using System;
  9. using System.Collections.Generic;
  10. using System.IO;
  11. using System.Net;
  12. using Xceed.Wpf.Toolkit.PropertyGrid.Attributes;
  13. namespace CyberX8_Simulator.Devices
  14. {
  15. public class FestoSocketSimulator : SocketDeviceSimulator
  16. {
  17. //Write Address
  18. private const ushort FESTO_DO_START_ADDRESS = 0x9C43;//FestoDO起始地址(40003)
  19. //Read Address
  20. private const ushort FESTO_DI_START_ADDRESS = 0xB153;//FestoDI起始地址(45395)
  21. //Register Count
  22. private const int FESTO_REGISTER_COUNT = 100;
  23. private IByteTransform byteTransform = new BigEndianByteTransformBase();
  24. /// <summary>
  25. /// 数据(index - data)
  26. /// </summary>
  27. private short[] _festoOutputDataDic = new short[FESTO_REGISTER_COUNT];
  28. /// <summary>
  29. /// 数据(DOName - index)
  30. /// </summary>
  31. private Dictionary<string, FestoDO> _festoNameIndexDic = new Dictionary<string, FestoDO>();
  32. /// <summary>
  33. /// do 索引对象字典(key-地址索引-bit,value--do名称)
  34. /// </summary>
  35. private Dictionary<string, string> _doNameDictionary = new Dictionary<string, string>();
  36. /// <summary>
  37. /// 构造函数
  38. /// </summary>
  39. /// <param name="port"></param>
  40. public FestoSocketSimulator(int port) : base(port)
  41. {
  42. InitData(port);
  43. }
  44. /// <summary>
  45. /// 解析信息
  46. /// </summary>
  47. /// <param name="data"></param>
  48. protected override void ProcessUnsplitMessage(byte[] data)
  49. {
  50. short flag = byteTransform.TransInt16(data, 0);//事务标识符
  51. byte channel = data[6];//单元标识符
  52. byte command = data[7];//功能码
  53. if (command == 0x03)//读取
  54. {
  55. ushort startAddress = byteTransform.TransUInt16(data, 8);//起始寄存器地址
  56. short registerCount = byteTransform.TransInt16(data, 10);//寄存器数量
  57. byte[] bytes = new byte[2 * registerCount];//读取2*registerCount Byte数据
  58. if (startAddress >= FESTO_DI_START_ADDRESS)
  59. {
  60. for (int i = 0; i < registerCount; i++)
  61. {
  62. Array.Copy(byteTransform.GetBytes(_festoOutputDataDic[startAddress - FESTO_DI_START_ADDRESS + i]), 0, bytes, i * 2, 2);
  63. }
  64. OnWriteMessage(CreateReadResponse(flag, channel, command, registerCount, bytes));
  65. }
  66. else
  67. {
  68. OnWriteMessage(CreateError(flag, channel, command, 0x8A));
  69. }
  70. }
  71. else if(command == 0x06)//写入
  72. {
  73. ushort startAddress = byteTransform.TransUInt16(data, 8);//起始寄存器地址
  74. short value = byteTransform.TransInt16(data, 10);//写入的值(2 Byte)
  75. if(startAddress >= FESTO_DO_START_ADDRESS)
  76. {
  77. //通知相关数据变化
  78. bool changeFlag = true;
  79. var result = DecodeDOData(startAddress, value, out changeFlag);
  80. if (changeFlag)
  81. {
  82. SimulatorCommManager.Instance.CheckDataChanged(result.Item1, result.Item2);
  83. }
  84. //modbus起始地址n为数据,n+1为诊断数据,取地址n下的数据
  85. _festoOutputDataDic[(startAddress - FESTO_DO_START_ADDRESS) * 2] = value;
  86. OnWriteMessage(CreateWriteResponse(flag, channel, command, startAddress, value));
  87. }
  88. else
  89. {
  90. OnWriteMessage(CreateError(flag, channel, command, 0x8A));
  91. }
  92. }
  93. else
  94. {
  95. OnWriteMessage(CreateError(flag, channel, command, 0x84));
  96. }
  97. }
  98. /// <summary>
  99. /// 读回复
  100. /// </summary>
  101. /// <param name="flag"></param>
  102. /// <param name="channel"></param>
  103. /// <param name="command"></param>
  104. /// <param name="registerCount"></param>
  105. /// <param name="values"></param>
  106. /// <returns></returns>
  107. private byte[] CreateReadResponse(short flag, byte channel, byte command, short registerCount, byte[] values)
  108. {
  109. byte[] bytes = new byte[6 + 3 + values.Length];
  110. Array.Copy(byteTransform.GetBytes(flag), 0, bytes, 0, 2);
  111. bytes[2] = 0x00;
  112. bytes[3] = 0x00;
  113. short dataLength = (short)(3 + values.Length);
  114. Array.Copy(byteTransform.GetBytes(dataLength), 0, bytes, 4, 2);
  115. bytes[6] = channel;
  116. bytes[7] = command;
  117. bytes[8] = (byte)(2 * registerCount);
  118. Array.Copy(values, 0, bytes, 9, values.Length);
  119. return bytes;
  120. }
  121. /// <summary>
  122. /// 写回复
  123. /// </summary>
  124. /// <param name="flag"></param>
  125. /// <param name="channel"></param>
  126. /// <param name="command"></param>
  127. /// <param name="startAddress"></param>
  128. /// <param name="value"></param>
  129. /// <returns></returns>
  130. private byte[] CreateWriteResponse(short flag, byte channel, byte command, ushort startAddress, short value)
  131. {
  132. byte[] bytes = new byte[12];
  133. Array.Copy(byteTransform.GetBytes(flag), 0, bytes, 0, 2);
  134. bytes[2] = 0x00;
  135. bytes[3] = 0x00;
  136. bytes[4] = 0x00;
  137. bytes[5] = 0x06;
  138. bytes[6] = channel;
  139. bytes[7] = command;
  140. byte[] addressByt = byteTransform.GetBytes(startAddress);
  141. Array.Copy(addressByt, 0, bytes, 8, 2);
  142. byte[] valueByt = byteTransform.GetBytes(value);
  143. Array.Copy(valueByt, 0, bytes, 10, 2);
  144. return bytes;
  145. }
  146. /// <summary>
  147. /// 错误回复
  148. /// </summary>
  149. /// <param name="flag"></param>
  150. /// <param name="channel"></param>
  151. /// <param name="command"></param>
  152. /// <param name="error"></param>
  153. /// <returns></returns>
  154. private byte[] CreateError(short flag, byte channel, byte command, byte error)
  155. {
  156. byte[] bytes = new byte[9];
  157. Array.Copy(byteTransform.GetBytes(flag), 0, bytes, 0, 2);
  158. bytes[2] = 0x00;
  159. bytes[3] = 0x00;
  160. bytes[4] = 0x00;
  161. bytes[5] = 0x03;
  162. bytes[6] = channel;
  163. bytes[7] = (byte)(command | 0x80);
  164. bytes[8] = error;
  165. return bytes;
  166. }
  167. /// <summary>
  168. /// 更新DI数据
  169. /// </summary>
  170. /// <param name="name"></param>
  171. /// <param name="value"></param>
  172. public void UpdataDOBytes(string name, int value)
  173. {
  174. if (_festoNameIndexDic.ContainsKey(name))
  175. {
  176. FestoDO festoDO = _festoNameIndexDic[name];
  177. short byteValue = (short) (_festoOutputDataDic[(festoDO.Address - FESTO_DO_START_ADDRESS) * 2] & ~(1 << festoDO.Bit));
  178. short tmp = (short)(value << festoDO.Bit);
  179. _festoOutputDataDic[(festoDO.Address - FESTO_DO_START_ADDRESS) * 2] = (short)(byteValue | tmp);
  180. }
  181. }
  182. /// <summary>
  183. /// 解析数据
  184. /// </summary>
  185. /// <param name="address"></param>
  186. /// <param name="value"></param>
  187. /// <returns></returns>
  188. private (string, bool) DecodeDOData(ushort address, short value, out bool flag)
  189. {
  190. flag = true;
  191. int index = (address - FESTO_DO_START_ADDRESS) * 2;
  192. if (_festoOutputDataDic[index] == value)
  193. {
  194. flag = false;
  195. return ("", false);
  196. }
  197. int bitNum = (int)Math.Log(_festoOutputDataDic[index] ^ value, 2);
  198. bool valueBool = (value & (1 << bitNum)) != 0 ? true : false;
  199. string str = $"{address}-{bitNum}";
  200. return (_doNameDictionary[str], valueBool);
  201. }
  202. /// <summary>
  203. /// 初始化
  204. /// </summary>
  205. private void InitData(int port)
  206. {
  207. //初始化数据
  208. for (int i = 0; i < FESTO_REGISTER_COUNT; i++)
  209. {
  210. _festoOutputDataDic[i] = 0x00;
  211. }
  212. //加载对应配置文件 FestoControllerCfg-Simulator.xml
  213. string oldXmlPath = PathManager.GetCfgDir();
  214. string newXmlPath = oldXmlPath.Replace("CyberX8_Simulator", "CyberX8_RT") + "Devices\\FestoControllerCfg-Simulator.xml";
  215. FestoControllerCfg cfg = CustomXmlSerializer.Deserialize<FestoControllerCfg>(new FileInfo(newXmlPath));
  216. foreach (FestoDeviceConfig config in cfg.FestoDeviceConfigs)
  217. {
  218. if (port == config.Port)
  219. {
  220. foreach (FestoDO item in config.FestoDoes)
  221. {
  222. _festoNameIndexDic[item.Name] = item;
  223. string str = $"{item.Address}-{item.Bit}";
  224. _doNameDictionary[str] = item.Name;
  225. }
  226. }
  227. }
  228. }
  229. }
  230. }