SunWayEfemSimulator.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531
  1. using System;
  2. using System.Text.RegularExpressions;
  3. using System.Threading;
  4. using System.Threading.Tasks;
  5. using Aitex.Core.UI.Control;
  6. using MECF.Framework.Simulator.Core.Driver;
  7. using Aitex.Core.Util;
  8. using PunkHPX8_Core;
  9. using Aitex.Core.RT.DataCenter;
  10. using System.Linq;
  11. using MECF.Framework.Common.DataCenter;
  12. using Aitex.Core.RT.ConfigCenter;
  13. using PunkHPX8_Simulator.Instances;
  14. using System.Collections.Generic;
  15. namespace PunkHPX8_Simulator.Devices
  16. {
  17. public class SunWayEfemSimulator : SocketDeviceSimulator
  18. {
  19. private const string SCMD = @"(?<=\:)(.*?)(?=\/)";
  20. private const string SFOUP = @"(?<=P)(\d{1})(?=[\;\/])";
  21. private const string ACK = "ACK";
  22. private readonly char[] delimiters = { ':', '/', '>', '|', ';' };
  23. private string[] _slotMap = new string[25];
  24. private string[] _slotDummy = new string[25];
  25. private PeriodicJob _HwThread;
  26. private bool _bCommReady;
  27. private bool _isDoorOpen;
  28. private bool _isLp1Placed;
  29. private bool _isLp2Placed;
  30. private bool _isLp3Placed;
  31. private bool _isLp1Clamped;
  32. private bool _isLp2Clamped;
  33. private bool _isLp3Clamped;
  34. private bool _isLP1Docked;
  35. private bool _isLP2Docked;
  36. private bool _isLP3Docked;
  37. public int WaferSize = 200;
  38. private int _speed = 20;
  39. private List<int> _lpStationNumber;
  40. private List<int> _dummyStationNumber;
  41. private readonly Queue<(string ack, TaskCompletionSource<bool> tcs)> _pendingAcks = new Queue<(string, TaskCompletionSource<bool>)>();
  42. private readonly object _syncRoot = new object();
  43. private bool _isPaused;
  44. public string SlotMap
  45. {
  46. get { return string.Join("", _slotMap); }
  47. }
  48. public bool _isVacuumError { get; set; }
  49. public bool _isAirError { get; set; }
  50. public bool _isFlowError { get; set; }
  51. public bool _isLeak { get; set; }
  52. public bool _isMaintain { get; set; }
  53. public bool _isWaferPresent { get; set; }
  54. public bool _isMaintainDoorOpen { get; private set; }
  55. public bool _isProtrude1 { get; set; }
  56. public bool _isProtrude2 { get; set; }
  57. public bool _isVAC { get; set; }
  58. public bool _isAIR { get; set; }
  59. private bool _isDoorOpen1;
  60. public bool IsDoorOpen1 { get { return _isDoorOpen1; } set { _isDoorOpen1 = value; SendLP1Data(); } }
  61. private bool _isDoorOpen2;
  62. public bool IsDoorOpen2 { get { return _isDoorOpen2; } set { _isDoorOpen2 = value;SendLP2Data(); } }
  63. private bool _isDoorOpen3;
  64. public bool IsDoorOpen3
  65. {
  66. get { return _isDoorOpen3; }
  67. set { _isDoorOpen3 = value; SendLP3Data(); }
  68. }
  69. public SunWayEfemSimulator() : base(1102, -1, "\r", ' ')
  70. {
  71. for (int i = 0; i < _slotMap.Length; i++)
  72. _slotMap[i] = "0";
  73. for(int i=0; i < _slotDummy.Length; i++)
  74. {
  75. _slotDummy[i]="0";
  76. }
  77. _HwThread = new PeriodicJob(5000, OnSendEvent, "EfemHardware", true);
  78. GetStationNumber();
  79. }
  80. private void Pause()
  81. {
  82. lock (_syncRoot)
  83. {
  84. _isPaused = true;
  85. }
  86. }
  87. private void Resume()
  88. {
  89. lock (_syncRoot)
  90. {
  91. _isPaused = false;
  92. // 发送所有积压的消息
  93. while (_pendingAcks.Count > 0)
  94. {
  95. var (ack, tcs) = _pendingAcks.Dequeue();
  96. OnWriteMessage(ack);
  97. tcs.TrySetResult(true);
  98. }
  99. }
  100. }
  101. private bool OnSendEvent()
  102. {
  103. if (IsConnected)
  104. {
  105. }
  106. return true;
  107. }
  108. private void GetStationNumber()
  109. {
  110. _lpStationNumber = new List<int>();
  111. _lpStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.LP1.Cassete200Station_Blade1_Upper"));
  112. _lpStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.LP1.Cassete150Station_Blade1_Upper"));
  113. _lpStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.LP1.Cassete100Station_Blade1_Upper"));
  114. _lpStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.LP2.Cassete200Station_Blade1_Upper"));
  115. _lpStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.LP2.Cassete150Station_Blade1_Upper"));
  116. _lpStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.LP2.Cassete100Station_Blade1_Upper"));
  117. _lpStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.LP3.Cassete200Station_Blade1_Upper"));
  118. _lpStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.LP3.Cassete150Station_Blade1_Upper"));
  119. _lpStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.LP3.Cassete100Station_Blade1_Upper"));
  120. _dummyStationNumber = new List<int>();
  121. _dummyStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.Dummy1.Cassete200Station_Blade1_Upper"));
  122. _dummyStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.Dummy1.Cassete150Station_Blade1_Upper"));
  123. _dummyStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.Dummy1.Cassete100Station_Blade1_Upper"));
  124. _dummyStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.Dummy2.Cassete200Station_Blade1_Upper"));
  125. _dummyStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.Dummy2.Cassete150Station_Blade1_Upper"));
  126. _dummyStationNumber.Add(SystemConfig.Instance.GetValue<int>("EFEM.Dummy2.Cassete100Station_Blade1_Upper"));
  127. }
  128. protected override void ProcessUnsplitMessage(string str)
  129. {
  130. //OnWriteMessage(ack);
  131. // 处理INF
  132. OnWork(str);
  133. }
  134. internal void SetCassetteDoor(bool doorOpen)
  135. {
  136. _isMaintainDoorOpen = doorOpen;
  137. uint data = doorOpen ? (uint)0b0111111111000001110 : (uint)0b0111111111000001111;
  138. string msg = $"EVT:SIGSTAT/System/{data:X8}/00000000;";
  139. OnWriteMessage(msg);
  140. }
  141. internal void SetMaintain(bool maintain)
  142. {
  143. _isMaintain = maintain;
  144. uint data = maintain ? (uint)0b0111111111000011111 : (uint)0b0111111111000001111;
  145. string msg = $"EVT:SIGSTAT/System/{data:X8}/00000000;";
  146. OnWriteMessage(msg);
  147. }
  148. internal void SetProtrude1(bool protrude)
  149. {
  150. _isProtrude1 = protrude;
  151. uint data = protrude? (uint)0b0111111111000101111: (uint)0b0111111111000001111;
  152. string msg = $"EVT:SIGSTAT/System/{data:X8}/00000000;";
  153. OnWriteMessage(msg);
  154. }
  155. internal void SetProtrude2(bool protrude)
  156. {
  157. _isProtrude2 = protrude;
  158. uint data = protrude ? (uint)0b0111111111001001111 : (uint)0b0111111111000001111;
  159. string msg = $"EVT:SIGSTAT/System/{data:X8}/00000000;";
  160. OnWriteMessage(msg);
  161. }
  162. internal void SetVAC(bool VAC)
  163. {
  164. _isVAC = VAC;
  165. uint data = VAC ? (uint)0b0111111111000001101 : (uint)0b0111111111000001111;
  166. string msg = $"EVT:SIGSTAT/System/{data:X8}/00000000;";
  167. OnWriteMessage(msg);
  168. }
  169. internal void SetAIR(bool AIR)
  170. {
  171. _isAIR = AIR;
  172. uint data = AIR ? (uint)0b0111111111000001011 : (uint)0b0111111111000001111;
  173. string msg = $"EVT:SIGSTAT/System/{data:X8}/00000000;";
  174. OnWriteMessage(msg);
  175. }
  176. private async void OnWork(string str)
  177. {
  178. string ack = "";
  179. if (str.StartsWith("RQ LOAD")) //查询手指是否带片
  180. {
  181. ack = "LOAD A OFF\n_RDY";
  182. OnWriteMessage(ack);
  183. }
  184. else if (str.StartsWith("RSR")) //返回扫片结果
  185. {
  186. string[] strings = str.Split(' ').ToArray();
  187. int stationNumber = int.Parse(strings[1]);
  188. if (_lpStationNumber.Contains(stationNumber)) //LP
  189. {
  190. string lpwafer = "";
  191. foreach (var item in _slotMap)
  192. {
  193. lpwafer += $" {item}";
  194. }
  195. ack = "MAP" + lpwafer;
  196. }
  197. else if (_dummyStationNumber.Contains(stationNumber)) //dummy
  198. {
  199. string dummywafer = "";
  200. foreach (var item in _slotDummy)
  201. {
  202. dummywafer += $" {item}";
  203. }
  204. ack = "MAP" + dummywafer;
  205. }
  206. OnWriteMessage(ack);
  207. }
  208. else if (str.StartsWith("PAUSE"))
  209. {
  210. Pause();
  211. ack = "PAUSE _RDY";
  212. OnWriteMessage(ack);
  213. }
  214. else if(str.StartsWith("RESUME"))
  215. {
  216. ack = "RESUME _RDY";
  217. OnWriteMessage(ack);
  218. _ = Task.Run(DelayAnswerResume); // 启动后台任务
  219. Resume();
  220. }
  221. else //默认回复
  222. {
  223. ack = "_RDY";
  224. if (str.StartsWith("PICK") || str.StartsWith("PLACE") || str.StartsWith("MAP")
  225. || str.StartsWith("ALIGNER ALGN") || str.StartsWith("ALIGNER HOME"))
  226. {
  227. var tcs = new TaskCompletionSource<bool>();
  228. lock (_syncRoot)
  229. {
  230. // 如果已经暂停,直接加入队列
  231. if (_isPaused)
  232. {
  233. _pendingAcks.Enqueue((ack, tcs));
  234. return;
  235. }
  236. }
  237. // 启动延迟任务
  238. var delayTask = Task.Delay(2200);
  239. // 创建监控任务
  240. var monitorTask = Task.Run(async () =>
  241. {
  242. while (!delayTask.IsCompleted)
  243. {
  244. await Task.Delay(100);
  245. lock (_syncRoot)
  246. {
  247. if (_isPaused)
  248. {
  249. _pendingAcks.Enqueue((ack, tcs));
  250. return;
  251. }
  252. }
  253. }
  254. // 延迟完成且未被暂停
  255. OnWriteMessage(ack);
  256. tcs.SetResult(true);
  257. });
  258. await monitorTask;
  259. }
  260. else
  261. {
  262. OnWriteMessage(ack);
  263. }
  264. }
  265. }
  266. private void DelayAnswerResume()
  267. {
  268. Thread.Sleep((ushort)2200);
  269. if (_pendingAcks.Count > 0)
  270. {
  271. OnWriteMessage("ACTION _RDY");
  272. }
  273. }
  274. private void UpdateLocked(string lpNumber,bool locked)
  275. {
  276. if (lpNumber== "1")
  277. {
  278. _isLP1Docked = locked;
  279. SendLP1Data();
  280. }
  281. else if (lpNumber == "2")
  282. {
  283. _isLP2Docked = locked;
  284. SendLP2Data();
  285. }
  286. else if (lpNumber == "3")
  287. {
  288. _isLP3Docked = locked;
  289. SendLP3Data();
  290. }
  291. }
  292. private void UpdateClamped(string lpNumber, bool clamped)
  293. {
  294. if (lpNumber == "1")
  295. {
  296. _isLp1Clamped = clamped;
  297. SendLP1Data();
  298. }
  299. else if (lpNumber == "2")
  300. {
  301. _isLp2Clamped = clamped;
  302. SendLP2Data();
  303. }
  304. else if (lpNumber == "3")
  305. {
  306. _isLp3Clamped = clamped;
  307. SendLP3Data();
  308. }
  309. }
  310. private void SendLP1Data()
  311. {
  312. uint data = GetLP1Data1();
  313. string msg = $"EVT:SIGSTAT/LP1/{data:X8}/0000;";
  314. OnWriteMessage(msg);
  315. }
  316. private void SendLP2Data()
  317. {
  318. uint data = GetLP2Data1();
  319. string msg = $"EVT:SIGSTAT/LP2/{data:X8}/0000;";
  320. OnWriteMessage(msg);
  321. }
  322. private void SendLP3Data()
  323. {
  324. uint data = GetLP3Data1();
  325. string msg = $"EVT:SIGSTAT/LP3/{data:X8}/0000;";
  326. OnWriteMessage(msg);
  327. }
  328. private void SendAlignData()
  329. {
  330. string msg = $"EVT:ALIGN/ALN1/0;";
  331. OnWriteMessage(msg);
  332. }
  333. private void SendDummyMapData(string dummy)
  334. {
  335. string s2 = $"EVT:MAPDT/{dummy}/{_slotDummy};\r";
  336. OnWriteMessage(s2);
  337. }
  338. private void SendSystemData()
  339. {
  340. uint data = 0b0111111111000001111;
  341. string msg = $"EVT:SIGSTAT/System/{data:X8}/00000000;";
  342. OnWriteMessage(msg);
  343. }
  344. private ushort SimuOperationTime(EfemOperation op)
  345. {
  346. ushort sec = 0;
  347. switch (op)
  348. {
  349. case EfemOperation.Map:
  350. case EfemOperation.GetWaferInfo:
  351. case EfemOperation.Align:
  352. sec = 2200;
  353. break;
  354. case EfemOperation.Pick:
  355. case EfemOperation.Place:
  356. case EfemOperation.Orgsh:
  357. sec = 2200;
  358. break;
  359. case EfemOperation.Light:
  360. case EfemOperation.SigStatus:
  361. sec = 0;
  362. break;
  363. default:
  364. sec = 100;
  365. break;
  366. }
  367. return sec;
  368. }
  369. public void PlaceCarrier1()
  370. {
  371. _isLp1Placed = true;
  372. SendLP1Data();
  373. }
  374. public void RemoveCarrier1()
  375. {
  376. _isLp1Placed = false;
  377. SendLP1Data();
  378. }
  379. public void PlaceCarrier2()
  380. {
  381. _isLp2Placed = true;
  382. SendLP2Data();
  383. }
  384. public void RemoveCarrier2()
  385. {
  386. _isLp2Placed = false;
  387. SendLP2Data();
  388. }
  389. public void PlaceCarrier3()
  390. {
  391. _isLp3Placed = true;
  392. SendLP3Data();
  393. }
  394. public void RemoveCarrier3()
  395. {
  396. _isLp3Placed = false;
  397. SendLP3Data();
  398. }
  399. public void ClearWafer()
  400. {
  401. for (int i = 0; i < _slotMap.Length; i++)
  402. {
  403. _slotMap[i] = "0";
  404. }
  405. for (int i = 0; i < _slotDummy.Length; i++)
  406. {
  407. _slotDummy[i] = "0";
  408. }
  409. }
  410. public void SetAllWafer()
  411. {
  412. for (int i = 0; i < _slotMap.Length; i++)
  413. {
  414. _slotMap[i] = "1";
  415. }
  416. }
  417. public void RandomWafer()
  418. {
  419. Random _rd = new Random();
  420. for (int i = 0; i < _slotMap.Length; i++)
  421. {
  422. //_slotMap[i] = (i % 9).ToString();
  423. _slotMap[i]= _rd.Next(0, 10) < 6 ? "0" : "1";
  424. }
  425. }
  426. public void RandomDummyWafer()
  427. {
  428. Random _rdDummy = new Random();
  429. for (int i = 0; i < _slotDummy.Length; i++)
  430. {
  431. _slotDummy[i] = _rdDummy.Next(0, 10) < 6 ? "0" : "1";
  432. }
  433. }
  434. private uint GetLP1Data1()
  435. {
  436. uint data1 = 0x0u;
  437. data1 |= (_isLp1Placed ? 0x00000001u : 0x0);
  438. data1 |= (_isLp1Clamped ? 0x00000002u : 0x0);
  439. data1 |= (_isLP1Docked ? 0x00000004u : 0x0);
  440. data1 |= (_isDoorOpen1 ? 0x00000020u : 0x0);
  441. return data1;
  442. }
  443. private uint GetLP2Data1()
  444. {
  445. uint data1 = 0x0u;
  446. data1 |= (_isLp2Placed ? 0x00000001u : 0x0);
  447. data1 |= (_isLp2Clamped ? 0x00000002u : 0x0);
  448. data1 |= (_isLP2Docked ? 0x00000004u : 0x0);
  449. data1 |= (_isDoorOpen2 ? 0x00000020u : 0x0);
  450. return data1;
  451. }
  452. private uint GetLP3Data1()
  453. {
  454. uint data1 = 0x0u;
  455. data1 |= (_isLp3Placed ? 0x00000001u : 0x0);
  456. data1 |= (_isLp3Clamped ? 0x00000002u : 0x0);
  457. data1 |= (_isLP3Docked ? 0x00000004u : 0x0);
  458. data1 |= (_isDoorOpen3 ? 0x00000020u : 0x0);
  459. return data1;
  460. }
  461. }
  462. }