GalilSocketSimulator.cs 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744
  1. using Aitex.Common.Util;
  2. using Aitex.Core.RT.Log;
  3. using Aitex.Core.Util;
  4. using MECF.Framework.Common.Beckhoff.AxisProvider;
  5. using MECF.Framework.Common.Device.Galil;
  6. using MECF.Framework.Common.Net;
  7. using MECF.Framework.Common.Simulator;
  8. using MECF.Framework.Simulator.Core.Driver;
  9. using System;
  10. using System.Collections.Generic;
  11. using System.IO;
  12. using System.Text;
  13. namespace CyberX8_Simulator.Devices
  14. {
  15. public class GalilSocketSimulator : SocketDeviceSimulator
  16. {
  17. #region 常量
  18. //最大运动轴数量(abcdefg)
  19. private const int MAX_AXIS_NUM = 8;
  20. //GalilControllerData最大数据长度
  21. private const int MAX_CONTROL_DATA_LENGTH_21 = 264;
  22. private const int MAX_CONTROL_DATA_LENGTH_40 = 366;
  23. //GalilAxisData最大数据长度
  24. private const int MAX_AXIS_DATA_LENGTH_21 = 28;
  25. private const int MAX_AXIS_DATA_LENGTH_40 = 36;
  26. //GalilType
  27. private const string GALIL_21 = "Galil21";
  28. private const string GALIL_40 = "Galil40";
  29. private const string TARGET_VELOCITY = "TargetVelocity";
  30. private const string TARGET_ACCEL = "TargetAcceleration";
  31. private const string TARGET_DECEL = "TargetDeceleration";
  32. private const string TARGET_POSITION = "TargetPosition";
  33. private const string SWITCH_SIGNAL = "SwitchSignal";
  34. private const string ACTUAL_POSITION = "ActualPosition";
  35. private const string AUXILIARY_POSITION = "AuxiliaryPosition";
  36. private const string HOMING_SIGNAL = "HomingSignal";
  37. private const string FI_HOMING_SIGNAL = "FIHomingSignal";
  38. private const string MOTION_SIGNAL = "MotionSignal";
  39. private const string STOP_SIGNAL = "StopSignal";
  40. #endregion
  41. #region 内部变量
  42. private IByteTransform byteTransform = new BigEndianByteTransformBase();
  43. /// <summary>
  44. /// Galil数据
  45. /// </summary>
  46. private GalilControllerData _galilControlData = new GalilControllerData();
  47. /// <summary>
  48. /// Axis名称字典(key:axisName, value:index)
  49. /// </summary>
  50. private Dictionary<string, int> _axisNameIndexDic = new Dictionary<string, int>();
  51. /// <summary>
  52. /// Axis名称列表(index - axisName)
  53. /// </summary>
  54. private string[] _nameAxisList = new string[MAX_AXIS_NUM];
  55. /// <summary>
  56. /// ModuleName
  57. /// </summary>
  58. /// <param name="port"></param>
  59. private string _moduleName;
  60. /// <summary>
  61. /// Galil Type
  62. /// </summary>
  63. private string _galilType;
  64. /// <summary>
  65. /// GalilControllerData数据长度
  66. /// </summary>
  67. private int _controlDataLength;
  68. /// <summary>
  69. /// GalilAxisDatas数据长度
  70. /// </summary>
  71. private int _axisDataLength;
  72. /// <summary>
  73. /// MotorPosition比例系数字典(index - motorPositionRate)
  74. /// </summary>
  75. private Dictionary<string, double> _motorPositionRateDic = new Dictionary<string, double>();
  76. /// <summary>
  77. /// Velocity比例系数字典(index - motorPositionRate)
  78. /// </summary>
  79. private Dictionary<string, double> _velocityRateDic = new Dictionary<string, double>();
  80. /// <summary>
  81. /// Key:dataName - Value:GalilDI
  82. /// </summary>
  83. public Dictionary<string, GalilDI> InputDataNameDIDic = new Dictionary<string, GalilDI>();
  84. #endregion
  85. /// <summary>
  86. /// 构造函数
  87. /// </summary>
  88. /// <param name="port"></param>
  89. public GalilSocketSimulator(int port) :base(port)
  90. {
  91. InitData(port);
  92. }
  93. /// <summary>
  94. /// 解析信息
  95. /// </summary>
  96. /// <param name="data"></param>
  97. protected override void ProcessUnsplitMessage(byte[] data)
  98. {
  99. string cmdStr = ASCIIEncoding.ASCII.GetString(data);
  100. if (CheckCmdValid(cmdStr))
  101. {
  102. var cmdOperation = AnalyseCommand(cmdStr);
  103. byte[] response;
  104. if(cmdOperation.Item1 == "QR")
  105. {
  106. //Read
  107. response = CreateReadResponse(_galilControlData);
  108. }
  109. else
  110. {
  111. //Write
  112. SetOperation(cmdOperation.Item1, cmdOperation.Item2, cmdOperation.Item3);
  113. response = CreateWriteResponse();
  114. }
  115. OnWriteMessage(response);
  116. }
  117. else
  118. {
  119. OnWriteMessage(CreateError());
  120. }
  121. }
  122. /// <summary>
  123. /// 读回复
  124. /// </summary>
  125. /// <param name="flag"></param>
  126. /// <param name="channel"></param>
  127. /// <param name="command"></param>
  128. /// <param name="registerCount"></param>
  129. /// <param name="values"></param>
  130. /// <returns></returns>
  131. private byte[] CreateReadResponse(GalilControllerData data)
  132. {
  133. //数据头(4 bytes)
  134. int headLength = 4;
  135. byte[] result = new byte[headLength + _controlDataLength];
  136. result[0] = 0x00;
  137. result[1] = 0x00;
  138. short dataLength = (short)(headLength + _controlDataLength);
  139. Array.Copy(byteTransform.GetBytes(dataLength), 0, result, 2, 2);
  140. //数据体(MAX_CONTROL_DATA_LENGTH bytes)
  141. switch (_galilType)
  142. {
  143. case GALIL_21:
  144. Array.Copy(CodeControlData21(data), 0, result, 4, MAX_CONTROL_DATA_LENGTH_21);
  145. break;
  146. case GALIL_40:
  147. Array.Copy(CodeControlData40(data), 0, result, 4, MAX_CONTROL_DATA_LENGTH_40);
  148. break;
  149. default:
  150. break;
  151. }
  152. return result;
  153. }
  154. /// <summary>
  155. /// 写回复
  156. /// </summary>
  157. /// <param name="flag"></param>
  158. /// <param name="channel"></param>
  159. /// <param name="command"></param>
  160. /// <param name="startAddress"></param>
  161. /// <param name="value"></param>
  162. /// <returns></returns>
  163. private byte[] CreateWriteResponse()
  164. {
  165. //数据头(1 bytes)
  166. int headLength = 1;
  167. byte[] result = new byte[headLength];
  168. result[0] = 0x3A;
  169. return result;
  170. }
  171. /// <summary>
  172. /// 错误回复
  173. /// </summary>
  174. /// <param name="flag"></param>
  175. /// <param name="channel"></param>
  176. /// <param name="command"></param>
  177. /// <param name="error"></param>
  178. /// <returns></returns>
  179. private byte[] CreateError()
  180. {
  181. int headLength = 1;
  182. byte[] result = new byte[headLength];
  183. //Command not valid in program
  184. result[0] = 0x03;
  185. return result;
  186. }
  187. /// <summary>
  188. /// 初始化
  189. /// </summary>
  190. private void InitData(int port)
  191. {
  192. //初始化AxisData(最大MAX_AXIS_NUM个axis)
  193. _galilControlData.GalilAxisDatas = new List<GalilAxisData>();
  194. for (int i = 0; i < MAX_AXIS_NUM; i++)
  195. {
  196. _galilControlData.GalilAxisDatas.Add(new GalilAxisData());
  197. _galilControlData.GalilAxisDatas[i].Status = 0x00;
  198. }
  199. _galilControlData.Inputs = new byte[10];
  200. _galilControlData.Outputs = new byte[10];
  201. //电机数据更新事件
  202. MotorSimulator.Instance.OnUpdateMotionDatasChanged += UpdateRealTimeMotionData;
  203. MotorSimulator.Instance.OnUpdateInputDatasChanged += UpdateInputData;
  204. //加载GalilControllerCfg-Simulator.xml
  205. try
  206. {
  207. string oldXmlPath = PathManager.GetCfgDir();
  208. string newXmlPath = oldXmlPath.Replace("CyberX8_Simulator", "CyberX8_RT") + "Devices\\GalilControllerCfg-Simulator.xml";
  209. GalilControllerCfg cfg = CustomXmlSerializer.Deserialize<GalilControllerCfg>(new FileInfo(newXmlPath));
  210. if(cfg != null)
  211. {
  212. foreach (GalilDeviceConfig config in cfg.GalilDeviceConfigs)
  213. {
  214. if (port == config.Port)
  215. {
  216. _moduleName = config.Module;
  217. _galilType = config.GalilType;
  218. foreach (GalilAxisConfig item in config.GalilAxises)
  219. {
  220. _axisNameIndexDic[item.Name] = item.Index;
  221. _nameAxisList[item.Index] = item.Name;
  222. }
  223. if (config.GalilDigIn != null && config.GalilDigIn.DIs.Count != 0)
  224. {
  225. foreach (var items in config.GalilDigIn.DIs)
  226. {
  227. InputDataNameDIDic[items.Name] = items;
  228. }
  229. }
  230. }
  231. }
  232. }
  233. }
  234. catch
  235. {
  236. LOG.WriteLog(eEvent.ERR_GALIL, "Galil", "Load galil GalilControllerCfg-Simulator.xml failed");
  237. }
  238. //加载AxisProviderCfg.xml
  239. try
  240. {
  241. string oldXmlPath = PathManager.GetCfgDir();
  242. string newXmlPath = oldXmlPath.Replace("CyberX8_Simulator", "CyberX8_RT") + "Devices\\AxisProviderCfg.xml";
  243. BeckhoffAxisProviderCfg axisProviderCfg = CustomXmlSerializer.Deserialize<BeckhoffAxisProviderCfg>(new FileInfo(newXmlPath));
  244. if (axisProviderCfg != null)
  245. {
  246. foreach (BeckhoffProviderAxis item in axisProviderCfg.Axes)
  247. {
  248. for (int i = 0; i < MAX_AXIS_NUM; i++)
  249. {
  250. if (_nameAxisList[i] == item.Name)
  251. {
  252. _motorPositionRateDic[item.Name] = item.MotorPositionRate;
  253. _velocityRateDic[item.Name] = item.VelocityRate;
  254. }
  255. }
  256. }
  257. }
  258. }
  259. catch
  260. {
  261. LOG.WriteLog(eEvent.ERR_AXIS, "axisProvider", "Load AxisProviderCfg.xml failed");
  262. }
  263. //初始化对应Galiltype数据
  264. if (_galilType == GALIL_40)
  265. {
  266. _controlDataLength = MAX_CONTROL_DATA_LENGTH_40;
  267. _axisDataLength = MAX_AXIS_DATA_LENGTH_40;
  268. _galilControlData.SBlocks = new byte[10];
  269. _galilControlData.TBlocks = new byte[10];
  270. }
  271. else
  272. {
  273. _controlDataLength = MAX_CONTROL_DATA_LENGTH_21;
  274. _axisDataLength = MAX_AXIS_DATA_LENGTH_21;
  275. _galilControlData.SBlocks = new byte[8];
  276. _galilControlData.TBlocks = new byte[8];
  277. }
  278. }
  279. #region Galil编码
  280. /// <summary>
  281. /// GalilControllerData编码(Galil21)
  282. /// </summary>
  283. /// <returns></returns>
  284. private byte[] CodeControlData21(GalilControllerData ctrlData)
  285. {
  286. byte[] result = new byte[MAX_CONTROL_DATA_LENGTH_21];
  287. int index = 0;
  288. //Sample(2 bytes)
  289. Array.Copy(BitConverter.GetBytes(ctrlData.Sample), 0, result, index, 2);
  290. index += 2;
  291. //Inputs(1*10 bytes)
  292. Array.Copy(ctrlData.Inputs, 0, result, index, 10);
  293. index += 10;
  294. //Outputs(1*10 bytes)
  295. Array.Copy(ctrlData.Outputs, 0, result, index, 10);
  296. index += 10;
  297. //Error Code(1 bytes)
  298. result[13] = ctrlData.ErrorCode;
  299. index++;
  300. //General Status(1 bytes)
  301. result[14] = ctrlData.Status;
  302. index++;
  303. //S Block(2+2+4=8 bytes)
  304. Array.Copy(ctrlData.SBlocks, 0, result, 15, 8);
  305. index += 8;
  306. //T Block(2+2+4=8 bytes)
  307. Array.Copy(ctrlData.TBlocks, 0, result, 23, 8);
  308. index += 8;
  309. //Axis Datas(28 * 8 bytes)
  310. for (int i = 0; i < MAX_AXIS_NUM; i++)
  311. {
  312. Array.Copy(CodeAxisData21(ctrlData.GalilAxisDatas[i]), 0, result, index, MAX_AXIS_DATA_LENGTH_21);
  313. index += MAX_AXIS_DATA_LENGTH_21;
  314. }
  315. return result;
  316. }
  317. /// <summary>
  318. /// GalilAxisData编码(Galil21)
  319. /// </summary>
  320. /// <param name="axisData"></param>
  321. /// <returns></returns>
  322. private byte[] CodeAxisData21(GalilAxisData axisData)
  323. {
  324. byte[] result = new byte[MAX_AXIS_DATA_LENGTH_21];
  325. int index = 0;
  326. //Status(2 bytes)
  327. Array.Copy(byteTransform.GetBytes(axisData.Status), 0, result, index, 2);
  328. index += 2;
  329. //Switches(1 bytes)
  330. result[index] = axisData.Switches;
  331. index++;
  332. //Stop Code(1 bytes)
  333. result[index] = axisData.StopCode;
  334. index++;
  335. //Reference Position(4 bytes)
  336. Array.Copy(BitConverter.GetBytes(axisData.ReferencePosition), 0, result, index, 4);
  337. index += 4;
  338. //Motor Position(4 bytes)
  339. Array.Copy(BitConverter.GetBytes(axisData.MotorPosition), 0, result, index, 4);
  340. index += 4;
  341. //Position Error
  342. Array.Copy(BitConverter.GetBytes(axisData.PositionError), 0, result, index, 4);
  343. index += 4;
  344. //Auxiliary Position
  345. Array.Copy(BitConverter.GetBytes(axisData.AuxiliaryPosition), 0, result, index, 4);
  346. index += 4;
  347. //Velocity
  348. Array.Copy(BitConverter.GetBytes(axisData.Velocity), 0, result, index, 4);
  349. index += 4;
  350. //Torque
  351. Array.Copy(BitConverter.GetBytes(axisData.Torque), 0, result, index, 2);
  352. index += 2;
  353. //Analog
  354. Array.Copy(BitConverter.GetBytes(axisData.Res), 0, result, index, 2);
  355. return result;
  356. }
  357. /// <summary>
  358. /// GalilControllerData编码(Galil40)
  359. /// </summary>
  360. /// <returns></returns>
  361. private byte[] CodeControlData40(GalilControllerData ctrlData)
  362. {
  363. byte[] result = new byte[MAX_CONTROL_DATA_LENGTH_40];
  364. int index = 0;
  365. //Sample(2 bytes)
  366. Array.Copy(BitConverter.GetBytes(ctrlData.Sample), 0, result, index, 2);
  367. index += 2;
  368. //Inputs(1*10 bytes)
  369. Array.Copy(ctrlData.Inputs, 0, result, index, 10);
  370. index += 10;
  371. //Outputs(1*10 bytes)
  372. Array.Copy(ctrlData.Outputs, 0, result, index, 10);
  373. index += 10;
  374. //reservers
  375. byte[] reservers = new byte[16];
  376. Array.Copy(reservers, 0, result, index, 16);
  377. index += 16;
  378. //ethernet
  379. byte[] ethernet = new byte[8];
  380. Array.Copy(ethernet, 0, result, index, 8);
  381. index += 8;
  382. //ethernet
  383. //Error Code(1 bytes)
  384. result[13] = ctrlData.ErrorCode;
  385. index++;
  386. //General Status(1 bytes)
  387. result[14] = ctrlData.Status;
  388. index++;
  389. //ampliferStatus
  390. byte[] ampliferStatus = new byte[4];
  391. Array.Copy(ampliferStatus, 0, result, index, 4);
  392. index += 4;
  393. //counterModel
  394. byte[] counterModel = new byte[6];
  395. Array.Copy(counterModel, 0, result, index, 6);
  396. index += 6;
  397. //S Block(10 bytes)
  398. Array.Copy(ctrlData.SBlocks, 0, result, index, 10);
  399. index += 10;
  400. //T Block(10 bytes)
  401. Array.Copy(ctrlData.TBlocks, 0, result, index, 10);
  402. index += 10;
  403. //Axis Datas(36 * 8 bytes)
  404. for (int i = 0; i < MAX_AXIS_NUM; i++)
  405. {
  406. Array.Copy(CodeAxisData40(ctrlData.GalilAxisDatas[i]), 0, result, index, MAX_AXIS_DATA_LENGTH_40);
  407. index += MAX_AXIS_DATA_LENGTH_40;
  408. }
  409. return result;
  410. }
  411. /// <summary>
  412. /// GalilAxisData编码(Galil40)
  413. /// </summary>
  414. /// <param name="axisData"></param>
  415. /// <returns></returns>
  416. private byte[] CodeAxisData40(GalilAxisData axisData)
  417. {
  418. byte[] result = new byte[MAX_AXIS_DATA_LENGTH_40];
  419. int index = 0;
  420. //Status(2 bytes)
  421. Array.Copy(byteTransform.GetBytes(axisData.Status), 0, result, index, 2);
  422. index += 2;
  423. //Switches(1 bytes)
  424. result[index] = axisData.Switches;
  425. index++;
  426. //Stop Code(1 bytes)
  427. result[index] = axisData.StopCode;
  428. index++;
  429. //Reference Position(4 bytes)
  430. Array.Copy(BitConverter.GetBytes(axisData.ReferencePosition), 0, result, index, 4);
  431. index += 4;
  432. //Motor Position(4 bytes)
  433. Array.Copy(BitConverter.GetBytes(axisData.MotorPosition), 0, result, index, 4);
  434. index += 4;
  435. //Position Error
  436. Array.Copy(BitConverter.GetBytes(axisData.PositionError), 0, result, index, 4);
  437. index += 4;
  438. //Auxiliary Position
  439. Array.Copy(BitConverter.GetBytes(axisData.AuxiliaryPosition), 0, result, index, 4);
  440. index += 4;
  441. //Velocity
  442. Array.Copy(BitConverter.GetBytes(axisData.Velocity), 0, result, index, 4);
  443. index += 4;
  444. //Torque
  445. Array.Copy(BitConverter.GetBytes(axisData.Torque), 0, result, index, 4);
  446. index += 4;
  447. //Analog
  448. Array.Copy(BitConverter.GetBytes(axisData.Res), 0, result, index, 2);
  449. index += 2;
  450. byte hallInput = 0x00;
  451. result[index] = hallInput;
  452. index ++;
  453. byte reserved = 0x00;
  454. result[index] = reserved;
  455. index ++;
  456. byte[] userVariables = new byte[4];
  457. Array.Copy(userVariables, 0, result, index, 4);
  458. return result;
  459. }
  460. #endregion
  461. /// <summary>
  462. /// 解析指令(操作符, 运动轴, 操作数)
  463. /// </summary>
  464. /// <param name="cmd"></param>
  465. /// <returns></returns>
  466. private (string, char, int) AnalyseCommand(string cmdStr)
  467. {
  468. var result = ("", ' ', -1);
  469. //操作符
  470. result.Item1 = cmdStr.Substring(0, 2);
  471. //运动轴
  472. if (cmdStr.Length >= 4)
  473. {
  474. result.Item2 = Convert.ToChar(cmdStr.Substring(2, 1));
  475. }
  476. //操作数
  477. if (cmdStr.Length >= 5 && int.TryParse(cmdStr.Substring(4, cmdStr.Length - 5), out int tmp))
  478. {
  479. result.Item3 = tmp;
  480. }
  481. return result;
  482. }
  483. /// <summary>
  484. /// CMD校验
  485. /// </summary>
  486. /// <returns></returns>
  487. private bool CheckCmdValid(string cmdStr)
  488. {
  489. //长度
  490. if(cmdStr.Length < 3) return false;
  491. //;结尾
  492. if (!cmdStr.EndsWith(";")) return false;
  493. //第1位为A~Z
  494. if (cmdStr[0] < 'A' || cmdStr[0] > 'Z') return false;
  495. //第2位为A~Z
  496. if (cmdStr[1] < 'A' || cmdStr[1] > 'Z') return false;
  497. if(cmdStr[0] == 'C'&& cmdStr[1] == 'N')
  498. {
  499. return true;
  500. }
  501. if (cmdStr.Length >= 4)
  502. {
  503. //axis名称为A~H
  504. if (cmdStr[2] < 'A' || cmdStr[2] > 'H') return false;
  505. //=
  506. if(cmdStr.Length > 4 && !cmdStr.Contains("=")) return false;
  507. }
  508. return true;
  509. }
  510. /// <summary>
  511. /// 设置操作
  512. /// </summary>
  513. /// <param name="cmd"></param>
  514. /// <param name="axis"></param>
  515. /// <param name="value"></param>
  516. /// <returns></returns>
  517. private bool SetOperation(string cmd, char axis, int value)
  518. {
  519. switch (cmd)
  520. {
  521. case "SH"://上电
  522. SwitchMotor(axis, true);
  523. break;
  524. case "MO"://下电
  525. SwitchMotor(axis, false);
  526. break;
  527. case "PR"://相对位置
  528. SetTargetRelativePosition(axis, value);
  529. break;
  530. case "PA"://绝对位置
  531. SetTargetAbsolutePosition(axis, value);
  532. break;
  533. case "SP"://速度
  534. SetSpeed(axis, value);
  535. break;
  536. case "AC"://加速度
  537. SetAcceleration(axis, value);
  538. break;
  539. case "DC"://减速度
  540. SetDeceleration(axis, value);
  541. break;
  542. case "ST"://停止
  543. ControlMotion(axis, false);
  544. break;
  545. case "BG"://启动
  546. ControlMotion(axis, true);
  547. break;
  548. case "HM"://回零
  549. Home(axis);
  550. break;
  551. case "FI"://SRDRotation回零
  552. Home(axis);
  553. break;
  554. case "DP"://设定motor position
  555. SetMotorPosition(axis, value);
  556. break;
  557. case "DE"://设定auxiliary position
  558. SetAuxiliaryPosition(axis, value);
  559. break;
  560. case "CN":
  561. break;
  562. default:
  563. break;
  564. }
  565. return true;
  566. }
  567. /// <summary>
  568. /// 实时更新电机数据
  569. /// </summary>
  570. /// <param name="datasDic"></param>
  571. private void UpdateRealTimeMotionData(Dictionary<string, SimulatorMotionData> datasDic)
  572. {
  573. foreach(var dataItem in datasDic)
  574. {
  575. if (_axisNameIndexDic.ContainsKey(dataItem.Key))
  576. {
  577. UpdateAxisData(_axisNameIndexDic[dataItem.Key], dataItem.Value);
  578. }
  579. }
  580. }
  581. /// <summary>
  582. /// 更新Galil Input datas
  583. /// </summary>
  584. private void UpdateInputData(string module, string VariableName, bool value)
  585. {
  586. if(_moduleName == module && InputDataNameDIDic.ContainsKey(VariableName))
  587. {
  588. int tmp = InputDataNameDIDic[VariableName].Invert ? (!value ? 1 : 0) : (value ? 1 : 0);
  589. _galilControlData.Inputs[InputDataNameDIDic[VariableName].Address] = (byte)(tmp << InputDataNameDIDic[VariableName].Bit);
  590. }
  591. }
  592. /// <summary>
  593. /// 更新对应Axis数据
  594. /// </summary>
  595. /// <param name="index"></param>
  596. private void UpdateAxisData(int index, SimulatorMotionData data)
  597. {
  598. _galilControlData.GalilAxisDatas[index].Velocity = (int)(data.TargetVelocity * _velocityRateDic[_nameAxisList[index]]);
  599. _galilControlData.GalilAxisDatas[index].MotorPosition = (int)(data.ActualPosition / _motorPositionRateDic[_nameAxisList[index]]);
  600. _galilControlData.GalilAxisDatas[index].ReferencePosition = data.TargetPosition;
  601. _galilControlData.GalilAxisDatas[index].Torque = data.ActualTorque;
  602. _galilControlData.GalilAxisDatas[index].StopCode = data.StopCode;
  603. _galilControlData.GalilAxisDatas[index].Switches = (byte)(_galilControlData.GalilAxisDatas[index].Switches & 0xF9
  604. | ((data.ForwardLimit ? 1 : 0) << 3) | ((data.ReverseLimit ? 1 : 0) << 2));
  605. }
  606. /// <summary>
  607. /// 模拟器UI更新inputs数据
  608. /// </summary>
  609. /// <param name="name"></param>
  610. /// <param name="value"></param>
  611. public void UpdateInputByte(string name, int value)
  612. {
  613. if (InputDataNameDIDic.ContainsKey(name))
  614. {
  615. GalilDI galilDI = InputDataNameDIDic[name];
  616. short byteValue = (short)(_galilControlData.Inputs[galilDI.Address] & ~(1 << galilDI.Bit));
  617. short tmp = (short)(value << galilDI.Bit);
  618. _galilControlData.Inputs[galilDI.Address] = (byte)(byteValue | tmp);
  619. }
  620. }
  621. /// <summary>
  622. /// 模拟器获取inputs数据
  623. /// </summary>
  624. /// <param name="name"></param>
  625. public int GetInputData(string name)
  626. {
  627. int result = 0;
  628. if (!InputDataNameDIDic.ContainsKey(name)) return 0;
  629. GalilDI galilDI = InputDataNameDIDic[name];
  630. int tmp = _galilControlData.Inputs[galilDI.Address] & (1 << galilDI.Bit);
  631. result = (tmp == (1 << galilDI.Bit)) ? 1 : 0;
  632. return result;
  633. }
  634. #region AxisOperation
  635. /// <summary>
  636. /// 开/关电机
  637. /// </summary>
  638. /// <param name="axis"></param>
  639. /// <param name="flag"></param>
  640. private void SwitchMotor(char axis, bool flag)
  641. {
  642. byte[] data = byteTransform.GetBytes(_galilControlData.GalilAxisDatas[axis - 'A'].Status);
  643. data[0] &= 0xFE;
  644. data[0] |= (byte)(flag ? 0x00 : 0x01);
  645. _galilControlData.GalilAxisDatas[axis - 'A'].Status = byteTransform.TransUInt16(data, 0);
  646. MotorSimulator.Instance.SetMotionData(_nameAxisList[axis - 'A'], SWITCH_SIGNAL, flag);
  647. }
  648. /// <summary>
  649. /// 设置速度
  650. /// </summary>
  651. private void SetSpeed(char axis, int value)
  652. {
  653. MotorSimulator.Instance.SetMotionData(_nameAxisList[axis - 'A'], TARGET_VELOCITY, value);
  654. }
  655. /// <summary>
  656. /// 设置加速度
  657. /// </summary>
  658. private void SetAcceleration(char axis, int value)
  659. {
  660. MotorSimulator.Instance.SetMotionData(_nameAxisList[axis - 'A'], TARGET_ACCEL, value);
  661. }
  662. /// <summary>
  663. /// 设置减速度
  664. /// </summary>
  665. private void SetDeceleration(char axis, int value)
  666. {
  667. MotorSimulator.Instance.SetMotionData(_nameAxisList[axis - 'A'], TARGET_DECEL, value);
  668. }
  669. /// <summary>
  670. /// 设置目标绝对位置(设定位置)
  671. /// </summary>
  672. private void SetTargetAbsolutePosition(char axis, int value)
  673. {
  674. _galilControlData.GalilAxisDatas[axis - 'A'].ReferencePosition = value;
  675. MotorSimulator.Instance.SetMotionData(_nameAxisList[axis - 'A'], TARGET_POSITION, value);
  676. }
  677. /// <summary>
  678. /// 设置目标相对位置(设定位置)
  679. /// </summary>
  680. private void SetTargetRelativePosition(char axis, int value)
  681. {
  682. _galilControlData.GalilAxisDatas[axis - 'A'].ReferencePosition = _galilControlData.GalilAxisDatas[axis - 'A'].MotorPosition + value;
  683. MotorSimulator.Instance.SetMotionData(_nameAxisList[axis - 'A'], TARGET_POSITION, _galilControlData.GalilAxisDatas[axis - 'A'].ReferencePosition);
  684. }
  685. /// <summary>
  686. /// 设置 Motor Position(实际位置)
  687. /// </summary>
  688. /// <param name="axis"></param>
  689. /// <param name="value"></param>
  690. private void SetMotorPosition(char axis, int value)
  691. {
  692. MotorSimulator.Instance.SetMotionData(_nameAxisList[axis - 'A'], ACTUAL_POSITION, value);
  693. }
  694. /// <summary>
  695. /// 设置 Auxiliary Position
  696. /// </summary>
  697. /// <param name="axis"></param>
  698. /// <param name="value"></param>
  699. private void SetAuxiliaryPosition(char axis, int value)
  700. {
  701. MotorSimulator.Instance.SetMotionData(_nameAxisList[axis - 'A'], AUXILIARY_POSITION, value);
  702. }
  703. /// <summary>
  704. /// 控制运动
  705. /// </summary>
  706. private void ControlMotion(char axis, bool value)
  707. {
  708. MotorSimulator.Instance.SetMotionData(_nameAxisList[axis - 'A'], value ? MOTION_SIGNAL : STOP_SIGNAL, value);
  709. }
  710. /// <summary>
  711. /// 电机回零
  712. /// </summary>
  713. /// <param name="axis"></param>
  714. private void Home(char axis)
  715. {
  716. MotorSimulator.Instance.SetMotionData(_nameAxisList[axis - 'A'], HOMING_SIGNAL, true);
  717. }
  718. /// <summary>
  719. /// 电机FI回零
  720. /// </summary>
  721. /// <param name="axis"></param>
  722. private void FI(char axis)
  723. {
  724. MotorSimulator.Instance.SetMotionData(_nameAxisList[axis - 'A'], FI_HOMING_SIGNAL, true);
  725. }
  726. public void SetForwardLimit()
  727. {
  728. }
  729. #endregion
  730. }
  731. }