FestoSocketSimulator.cs 10 KB

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