TemperatureControllerSerialPortDevice.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534
  1. using Aitex.Common.Util;
  2. using Aitex.Core.RT.Log;
  3. using Aitex.Core.Util;
  4. using MECF.Framework.Common.Device.TemperatureController;
  5. using MECF.Framework.Simulator.Core.Driver;
  6. using System;
  7. using System.Collections.Generic;
  8. using System.Diagnostics;
  9. using System.IO;
  10. using System.Linq;
  11. using System.Text;
  12. using System.Threading.Tasks;
  13. namespace MECF.Framework.Simulator.Core.Commons
  14. {
  15. public class TemperatureControllerSerialPortDevice : BaseSerialSimulator
  16. {
  17. #region 常量
  18. private const string PORT_NAME = "com12";
  19. private const byte ADD_FLAG = 0x30;
  20. #endregion
  21. #region 内部变量
  22. /// <summary>
  23. /// 控制字字典
  24. /// </summary>
  25. private Dictionary<byte, string> dic = new Dictionary<byte, string>()
  26. {
  27. { 0x05,"ENQ" },
  28. { 0x02,"STX" },
  29. { 0x03,"ETX" },
  30. { 0x06,"ACK" },
  31. { 0x0D,"CR" },
  32. { 0x01,"SOH" }
  33. };
  34. /// <summary>
  35. /// 命令字典
  36. /// </summary>
  37. private Dictionary<byte, string> comdic = new Dictionary<byte, string>()
  38. {
  39. { 0x31,"Target Temp" },
  40. { 0x32,"Internal" },
  41. { 0x33,"External" },
  42. { 0x34,"Alarm" },
  43. { 0x35,"Average" },
  44. { 0x36,"offset" },
  45. { 0x39,"control" },
  46. { 0x41,"PB" },
  47. { 0x42,"ARW" },
  48. { 0x43,"IConstant" },
  49. { 0x44,"DConstant" },
  50. { 0x45,"OutPutRatio" },
  51. { 0x46,"HeatingLimit" },
  52. { 0x47,"CoolingLimit" },
  53. { 0x48,"Saved" }
  54. };
  55. private double[] _reserviorValues = new double[10];
  56. private double[] _heatValues = new double[10];
  57. private double[] _targetValues = new double[10];
  58. private int _controlOperation = 0;
  59. //private byte[] alarm = new byte[12] { 0,0,0,0,1,0,0,0,0,0,0,0};
  60. private byte[] alarm = new byte[3] {0x30,0x33,0x30};
  61. System.Timers.Timer[] _timers;
  62. private int _tcCount = 1;
  63. #endregion
  64. /// <summary>
  65. /// 构造函数
  66. /// </summary>
  67. /// <param name="portName"></param>
  68. /// <param name="type"></param>
  69. public TemperatureControllerSerialPortDevice(string port) : base(port, SerialType.BUFFER)
  70. {
  71. Init(port);
  72. }
  73. /// <summary>
  74. /// 初始化
  75. /// </summary>
  76. /// <param name="port"></param>
  77. private void Init(string port)
  78. {
  79. //加载TC配置
  80. try
  81. {
  82. string oldXmlPath = PathManager.GetCfgDir();
  83. string newXmlPath = oldXmlPath.Replace("CyberX8_Simulator", "CyberX8_RT") + "Devices\\SMCCfg.xml";
  84. TemperatureConfig cfg = CustomXmlSerializer.Deserialize<TemperatureConfig>(new FileInfo(newXmlPath));
  85. if (cfg != null)
  86. {
  87. foreach (TemperatureDeviceConfig config in cfg.TemperatureDeviceConfigs)
  88. {
  89. if (int.TryParse(config.Port.Substring(3,2), out int result1) && int.TryParse(port.Substring(3, 2), out int result2))
  90. {
  91. if(result1 + 1 == result2) _tcCount = config.TemperatureDevices.Count;
  92. }
  93. }
  94. }
  95. }
  96. catch (Exception ex)
  97. {
  98. }
  99. //
  100. _timers = new System.Timers.Timer[_tcCount];
  101. for(int i = 0; i < _tcCount; i++)
  102. {
  103. _heatValues[i] = 16.5;
  104. _targetValues[i] = 0;
  105. _reserviorValues[i] = 16.6;
  106. _timers[i] = new System.Timers.Timer(50);
  107. switch (i)
  108. {
  109. case 0:
  110. _timers[0].Elapsed += Timer_Elapsed0;
  111. break;
  112. case 1:
  113. _timers[1].Elapsed += Timer_Elapsed1;
  114. break;
  115. case 2:
  116. _timers[2].Elapsed += Timer_Elapsed2;
  117. break;
  118. case 3:
  119. _timers[3].Elapsed += Timer_Elapsed3;
  120. break;
  121. case 4:
  122. _timers[4].Elapsed += Timer_Elapsed4;
  123. break;
  124. default:
  125. break;
  126. }
  127. }
  128. }
  129. /// <summary>
  130. /// TC1-1的调温
  131. /// </summary>
  132. /// <param name="sender"></param>
  133. /// <param name="e"></param>
  134. private void Timer_Elapsed0(object sender, System.Timers.ElapsedEventArgs e)
  135. {
  136. if (Math.Abs(_heatValues[0] - _targetValues[0]) >= 0.2)
  137. {
  138. //内部调温
  139. if (_heatValues[0] < _targetValues[0])
  140. {
  141. _heatValues[0] += 0.1;
  142. }
  143. else
  144. {
  145. _heatValues[0] -= 0.1;
  146. }
  147. //外部调温
  148. if (_reserviorValues[0] < _targetValues[0])
  149. {
  150. _reserviorValues[0] += 0.1;
  151. }
  152. else
  153. {
  154. _reserviorValues[0] -= 0.1;
  155. }
  156. }
  157. else
  158. {
  159. _timers[0].Stop();
  160. }
  161. }
  162. /// <summary>
  163. /// TC1-2的调温
  164. /// </summary>
  165. /// <param name="sender"></param>
  166. /// <param name="e"></param>
  167. private void Timer_Elapsed1(object sender, System.Timers.ElapsedEventArgs e)
  168. {
  169. if (Math.Abs(_heatValues[1] - _targetValues[1]) >= 0.2)
  170. {
  171. //内部调温
  172. if (_heatValues[1] < _targetValues[1])
  173. {
  174. _heatValues[1] += 0.1;
  175. }
  176. else
  177. {
  178. _heatValues[1] -= 0.1;
  179. }
  180. //外部调温
  181. if (_reserviorValues[1] < _targetValues[1])
  182. {
  183. _reserviorValues[1] += 0.1;
  184. }
  185. else
  186. {
  187. _reserviorValues[1] -= 0.1;
  188. }
  189. }
  190. else
  191. {
  192. _timers[1].Stop();
  193. }
  194. }
  195. /// <summary>
  196. /// TC1-3的调温
  197. /// </summary>
  198. /// <param name="sender"></param>
  199. /// <param name="e"></param>
  200. private void Timer_Elapsed2(object sender, System.Timers.ElapsedEventArgs e)
  201. {
  202. if (Math.Abs(_heatValues[2] - _targetValues[2]) >= 0.2)
  203. {
  204. //内部调温
  205. if (_heatValues[2] < _targetValues[2])
  206. {
  207. _heatValues[2] += 0.1;
  208. }
  209. else
  210. {
  211. _heatValues[2] -= 0.1;
  212. }
  213. //外部调温
  214. if (_reserviorValues[2] < _targetValues[2])
  215. {
  216. _reserviorValues[2] += 0.1;
  217. }
  218. else
  219. {
  220. _reserviorValues[2] -= 0.1;
  221. }
  222. }
  223. else
  224. {
  225. _timers[2].Stop();
  226. }
  227. }
  228. /// <summary>
  229. /// TC1-4的调温
  230. /// </summary>
  231. /// <param name="sender"></param>
  232. /// <param name="e"></param>
  233. private void Timer_Elapsed3(object sender, System.Timers.ElapsedEventArgs e)
  234. {
  235. if (Math.Abs(_heatValues[3] - _targetValues[3]) >= 0.2)
  236. {
  237. //内部调温
  238. if (_heatValues[3] < _targetValues[3])
  239. {
  240. _heatValues[3] += 0.1;
  241. }
  242. else
  243. {
  244. _heatValues[3] -= 0.1;
  245. }
  246. //外部调温
  247. if (_reserviorValues[3] < _targetValues[3])
  248. {
  249. _reserviorValues[3] += 0.1;
  250. }
  251. else
  252. {
  253. _reserviorValues[3] -= 0.1;
  254. }
  255. }
  256. else
  257. {
  258. _timers[3].Stop();
  259. }
  260. }
  261. /// <summary>
  262. /// TC1-5的调温
  263. /// </summary>
  264. /// <param name="sender"></param>
  265. /// <param name="e"></param>
  266. private void Timer_Elapsed4(object sender, System.Timers.ElapsedEventArgs e)
  267. {
  268. if (Math.Abs(_heatValues[4] - _targetValues[4]) >= 0.2)
  269. {
  270. //内部调温
  271. if (_heatValues[4] < _targetValues[4])
  272. {
  273. _heatValues[4] += 0.1;
  274. }
  275. else
  276. {
  277. _heatValues[4] -= 0.1;
  278. }
  279. //外部调温
  280. if (_reserviorValues[4] < _targetValues[4])
  281. {
  282. _reserviorValues[4] += 0.1;
  283. }
  284. else
  285. {
  286. _reserviorValues[4] -= 0.1;
  287. }
  288. }
  289. else
  290. {
  291. _timers[4].Stop();
  292. }
  293. }
  294. protected override string MessageConvert(byte[] byt)
  295. {
  296. string str = "";
  297. for(int i=0;i<byt.Length;i++)
  298. {
  299. byte item = byt[i];
  300. if(dic.ContainsKey(item))
  301. {
  302. str += dic[item];
  303. }
  304. else if(i==1)
  305. {
  306. str += (item - 0x30).ToString("D2");
  307. }
  308. else if(i==3&&comdic.ContainsKey(item))
  309. {
  310. str += comdic[item];
  311. }
  312. else
  313. {
  314. str += item.ToString("X2");
  315. }
  316. str += " ";
  317. }
  318. return str;
  319. }
  320. protected override void ProcessMessageBuffer(byte[] data)
  321. {
  322. if (data[0] == 0x06 && data[2]==0x0D&&data.Length>7)
  323. {
  324. byte[] byt = new byte[data.Length - 3];
  325. Array.Copy(data, 3, byt, 0, byt.Length);
  326. ProcessNormalCommand(byt);
  327. }
  328. else if (data[0] == 0x01)
  329. {
  330. ProcessNormalCommand(data);
  331. }
  332. }
  333. private void ProcessNormalCommand(byte[] data)
  334. {
  335. if (data[0] == 0x01 && data[data.Length-1]==0x0D)
  336. {
  337. if (data.Length == 7)
  338. {
  339. ProcessReadCommand(data);
  340. }
  341. else if (data.Length > 7)
  342. {
  343. ProcessSetCommand(data);
  344. }
  345. }
  346. }
  347. private void ProcessReadCommand(byte[] data)
  348. {
  349. switch(data[3])
  350. {
  351. case 0x31:
  352. ReadTargetValue(data);
  353. break;
  354. case 0x32:
  355. ReadReserviorValue(data);
  356. break;
  357. case 0x33:
  358. ReadHeatValue(data);
  359. break;
  360. case 0x34:
  361. ReadAlarmValue(data);
  362. break;
  363. case 0x39:
  364. ReadControlOperationValue(data);
  365. break;
  366. }
  367. }
  368. private void ProcessSetCommand(byte[] data)
  369. {
  370. switch(data[3])
  371. {
  372. case 0x31:
  373. _targetValues[data[1] - ADD_FLAG - 1] = (data[4] - ADD_FLAG) * 10 + (data[5] - ADD_FLAG) + (data[6]-ADD_FLAG)*0.1;
  374. _timers[data[1] - ADD_FLAG -1].Start();
  375. break;
  376. case 0x39:
  377. _controlOperation = data[7] - ADD_FLAG;
  378. if(_controlOperation == 0)
  379. {
  380. //停止调温
  381. _timers[data[1] - ADD_FLAG - 1].Stop();
  382. }
  383. else
  384. {
  385. //继续调温
  386. _timers[data[1] - ADD_FLAG - 1].Start();
  387. }
  388. break;
  389. }
  390. WriteConfirmData(data[1]);
  391. }
  392. /// <summary>
  393. /// 回复确认
  394. /// </summary>
  395. /// <param name="id"></param>
  396. private void WriteConfirmData(byte id)
  397. {
  398. byte[] confirm = new byte[3];
  399. confirm[0] = 0x06;
  400. confirm[1] = id;
  401. confirm[2] = 0x0d;
  402. WriteBuffer(confirm);
  403. }
  404. private void ReadTargetValue(byte[] data)
  405. {
  406. byte[] byt = new byte[12];
  407. byt[0] = data[0];
  408. byt[1] = data[1];
  409. byt[2] = 0x02;
  410. byt[3] = data[3];
  411. byte[] decadeByte = GetDecadeBytes(_targetValues[data[1] - ADD_FLAG - 1]);
  412. Array.Copy(decadeByte, 0, byt, 4, decadeByte.Length);
  413. byt[5 + decadeByte.Length] = 0;
  414. byt[6 + decadeByte.Length] = 0;
  415. byt[11] = 0x0D;
  416. WriteBuffer(byt);
  417. }
  418. private void ReadReserviorValue(byte[] data)
  419. {
  420. byte[] byt = new byte[12];
  421. byt[0] = data[0];
  422. byt[1] = data[1];
  423. byt[2] = 0x02;
  424. byt[3] = data[3];
  425. byte[] decadeByte = GetDecadeBytes(_reserviorValues[data[1] - ADD_FLAG - 1]);
  426. Array.Copy(decadeByte, 0, byt, 4, decadeByte.Length);
  427. byt[5 + decadeByte.Length] = 0;
  428. byt[6 + decadeByte.Length] = 0;
  429. byt[11] = 0x0D;
  430. WriteBuffer(byt);
  431. }
  432. private void ReadHeatValue(byte[] data)
  433. {
  434. byte[] byt = new byte[12];
  435. byt[0] = data[0];
  436. byt[1] = data[1];
  437. byt[2] = 0x02;
  438. byt[3] = data[3];
  439. byte[] decadeByte = GetDecadeBytes(_heatValues[data[1] - ADD_FLAG - 1]);
  440. Array.Copy(decadeByte, 0, byt, 4, decadeByte.Length);
  441. byt[5 + decadeByte.Length] = 0;
  442. byt[6 + decadeByte.Length] = 0;
  443. byt[11] = 0x0D;
  444. WriteBuffer(byt);
  445. }
  446. private void ReadControlOperationValue(byte[] data)
  447. {
  448. byte[] byt = new byte[12];
  449. byt[0] = data[0];
  450. byt[1] = data[1];
  451. byt[2] = 0x02;
  452. byt[3] = data[3];
  453. byte[] decadeByte = GetKilloBytes(_controlOperation);
  454. Array.Copy(decadeByte, 0, byt, 4, decadeByte.Length);
  455. byt[5 + decadeByte.Length] = 0;
  456. byt[6 + decadeByte.Length] = 0;
  457. byt[11] = 0x0D;
  458. WriteBuffer(byt);
  459. }
  460. private void ReadAlarmValue(byte[] data)
  461. {
  462. byte[] byt = new byte[8+alarm.Length];
  463. byt[0] = data[0];
  464. byt[1] = data[1];
  465. byt[2] = 0x02;
  466. byt[3] = data[3];
  467. Array.Copy(alarm, 0, byt, 4, alarm.Length);
  468. byt[5 + alarm.Length] = 0;
  469. byt[6 + alarm.Length] = 0;
  470. byt[7+alarm.Length] = 0x0D;
  471. WriteBuffer(byt);
  472. }
  473. /// <summary>
  474. /// 获取温度数组(10|1|0.1|0.01)
  475. /// </summary>
  476. /// <param name="temperature"></param>
  477. /// <returns></returns>
  478. private byte[] GetDecadeBytes(double temperature)
  479. {
  480. byte decade = GetSendByteData((byte)Math.Floor(temperature / 10));
  481. byte unit = GetSendByteData((byte)Math.Floor(temperature % 10));
  482. byte digital = GetSendByteData((byte)Math.Floor(temperature * 10 % 10));
  483. return new byte[4] { decade, unit, digital, ADD_FLAG };
  484. }
  485. /// <summary>
  486. /// 获取千级数组(1000|100|10|1)
  487. /// </summary>
  488. /// <param name="temperature"></param>
  489. /// <returns></returns>
  490. private byte[] GetKilloBytes(double temperature)
  491. {
  492. byte kilo = GetSendByteData((byte)Math.Floor(temperature / 1000));
  493. byte hundred = GetSendByteData((byte)Math.Floor(temperature % 1000/100));
  494. byte decade = GetSendByteData((byte)Math.Floor(temperature % 100 / 10));
  495. byte unit = GetSendByteData((byte)Math.Floor(temperature % 10));
  496. return new byte[4] { kilo, hundred, decade, unit };
  497. }
  498. /// <summary>
  499. /// 获取发送位数据
  500. /// </summary>
  501. /// <param name="originalData"></param>
  502. /// <returns></returns>
  503. private byte GetSendByteData(byte originalData)
  504. {
  505. return (byte)(ADD_FLAG + originalData);
  506. }
  507. }
  508. }