TMCycle.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327
  1. using Aitex.Core.RT.Routine;
  2. using Aitex.Core.RT.SCCore;
  3. using Aitex.Sorter.Common;
  4. using Venus_RT.Devices;
  5. using MECF.Framework.Common.Routine;
  6. using MECF.Framework.Common.Equipment;
  7. using MECF.Framework.Common.SubstrateTrackings;
  8. using Venus_Core;
  9. using Aitex.Core.RT.Log;
  10. using Aitex.Core.Util;
  11. using System.Collections.Generic;
  12. using System.Linq;
  13. using Venus_RT.Modules.Schedulers;
  14. using Venus_RT.Scheduler;
  15. using System;
  16. using MECF.Framework.Common.Schedulers;
  17. using Aitex.Core.RT.Fsm;
  18. namespace Venus_RT.Modules
  19. {
  20. class TMCycle : ModuleRoutineBase, IRoutine
  21. {
  22. enum TMCycleStep
  23. {
  24. Start,
  25. ReturnBack,
  26. Cycling,
  27. End,
  28. }
  29. private bool IsModuleAvailable(ModuleName module) => dictSchedulers.Keys.Contains(module) && dictSchedulers[module].IsAvailable;
  30. List<ModuleName> tmCycleRoutine = new List<ModuleName>() { ModuleName.LLA, ModuleName.PMA, ModuleName.PMB, ModuleName.LLB };
  31. int cycleCount = 1000;
  32. ModuleName _sourceModule = ModuleName.LLA;
  33. ModuleName _destinationModule = ModuleName.LLB;
  34. int _sourceSlotNumber = 4;
  35. int _destinationSlotNumber = 4;
  36. SchedulerTMRobot _TMRobot = (SchedulerTMRobot)Singleton<TransferModule>.Instance.GetScheduler(ModuleName.TMRobot);
  37. private readonly int INVALID_SLOT = -1;
  38. Dictionary<ModuleName, SchedulerModule> dictSchedulers = new Dictionary<ModuleName, SchedulerModule>();
  39. Queue<MoveItem> _ReturnWafers = new Queue<MoveItem>();
  40. Queue<MoveItem> _runningItems = new Queue<MoveItem>();
  41. Queue<MoveItem> _CycleWafers = new Queue<MoveItem>();
  42. public int? CycleIndex;
  43. public TMCycle() : base(ModuleName.System)
  44. {
  45. Name = "TM Cycle";
  46. void _initMoudle(ModuleName name, SchedulerModule sche)
  47. {
  48. if (ModuleHelper.IsInstalled(name))
  49. {
  50. dictSchedulers[name] = sche;
  51. }
  52. }
  53. _initMoudle(ModuleName.LLA, new SchedulerLoadLock(ModuleName.LLA));
  54. _initMoudle(ModuleName.LLB, new SchedulerLoadLock(ModuleName.LLB));
  55. _initMoudle(ModuleName.PMA, new SchedulerPM(ModuleName.PMA));
  56. _initMoudle(ModuleName.PMB, new SchedulerPM(ModuleName.PMB));
  57. _initMoudle(ModuleName.PMC, new SchedulerPM(ModuleName.PMC));
  58. _initMoudle(ModuleName.PMD, new SchedulerPM(ModuleName.PMD));
  59. }
  60. public RState Start(params object[] objs)
  61. {
  62. CycleIndex = 0;
  63. if (objs.Length == 2)
  64. {
  65. var modules = ((string[])objs[0]).ToList();
  66. if (modules.Count >= 2)
  67. tmCycleRoutine.Clear();
  68. foreach(var mod in modules)
  69. {
  70. try
  71. {
  72. ModuleName module = ModuleHelper.Converter(mod);
  73. tmCycleRoutine.Add(module);
  74. }
  75. catch
  76. {
  77. LOG.Write(eEvent.ERR_ROUTER, "TMCycle", $"Invalid module string: {mod}");
  78. return RState.Failed;
  79. }
  80. }
  81. cycleCount = (int)objs[1];
  82. }
  83. return Runner.Start(Module, Name);
  84. }
  85. public RState Monitor()
  86. {
  87. Runner.Run(TMCycleStep.Start, NullFun)
  88. .LoopStart(TMCycleStep.ReturnBack, "Cycle", cycleCount, StartReturn, ReturnBack)
  89. .LoopEnd(TMCycleStep.Cycling, StartCycle, Cycling)
  90. .End(TMCycleStep.End, NullFun, _delay_50ms);
  91. return Runner.Status;
  92. }
  93. private bool StartReturn()
  94. {
  95. CycleIndex += 1;
  96. _destinationModule = tmCycleRoutine.Last();
  97. _destinationSlotNumber = SC.GetValue<int>($"{_destinationModule}.SlotNumber");
  98. _sourceModule = tmCycleRoutine.First();
  99. _sourceSlotNumber = SC.GetValue<int>($"{_sourceModule}.SlotNumber");
  100. _ReturnWafers.Clear();
  101. for(int i = 0; i < _destinationSlotNumber; i++)
  102. {
  103. if(WaferManager.Instance.CheckHasWafer(_destinationModule, i))
  104. {
  105. _ReturnWafers.Enqueue(new MoveItem(_destinationModule, i, _sourceModule, INVALID_SLOT, Hand.None));
  106. }
  107. }
  108. return true;
  109. }
  110. List<int> GetReadyInSlot(ModuleName module, int slotCount)
  111. {
  112. List<int> slots = new List<int>();
  113. for (int i = 0; i < slotCount; i++)
  114. {
  115. if (WaferManager.Instance.CheckNoWafer(module, i))
  116. slots.Add(i);
  117. if (slots.Count >= 2)
  118. return slots;
  119. }
  120. return slots;
  121. }
  122. List<int> GetReadyOutSlot(ModuleName module, int slotCount)
  123. {
  124. List<int> slots = new List<int>();
  125. for (int i = 0; i < slotCount; i++)
  126. {
  127. if (WaferManager.Instance.CheckHasWafer(module, i))
  128. slots.Add(i);
  129. if (slots.Count >= 2)
  130. return slots;
  131. }
  132. return slots;
  133. }
  134. List<ModuleName> GetReadyOutPMs()
  135. {
  136. List<ModuleName> outpm = new List<ModuleName>();
  137. foreach(var module in tmCycleRoutine)
  138. {
  139. if(ModuleHelper.IsPm(module))
  140. {
  141. if(IsModuleAvailable(module) && WaferManager.Instance.CheckHasWafer(module, 0))
  142. {
  143. outpm.Add(module);
  144. if (outpm.Count >= 2)
  145. break;
  146. }
  147. }
  148. }
  149. return outpm;
  150. }
  151. List<ModuleName> GetReadyInPMs()
  152. {
  153. List<ModuleName> inpm = new List<ModuleName>();
  154. foreach (var module in tmCycleRoutine)
  155. {
  156. if (ModuleHelper.IsPm(module))
  157. {
  158. if (IsModuleAvailable(module) && WaferManager.Instance.CheckNoWafer(module, 0))
  159. {
  160. inpm.Add(module);
  161. if (inpm.Count >= 3)
  162. break;
  163. }
  164. }
  165. }
  166. return inpm;
  167. }
  168. private bool ReturnBack()
  169. {
  170. var t1 = IsModuleAvailable(_sourceModule);
  171. var t2 = IsModuleAvailable(_destinationModule);
  172. var t3 = _TMRobot.IsAvailable;
  173. if (IsModuleAvailable(_sourceModule) && IsModuleAvailable(_destinationModule) && _TMRobot.IsAvailable)
  174. {
  175. var InSlots = GetReadyInSlot(_sourceModule, _sourceSlotNumber);
  176. var OutSlots = GetReadyOutSlot(_destinationModule, _destinationSlotNumber);
  177. if (InSlots.Count == 0 || OutSlots.Count == 0)
  178. return true;
  179. Queue<MoveItem> items = new Queue<MoveItem>();
  180. for(int i = 0; i < Math.Min(InSlots.Count, OutSlots.Count); i++)
  181. {
  182. items.Enqueue(new MoveItem(_destinationModule, OutSlots[i], _sourceModule, InSlots[i], (Hand)i));
  183. }
  184. if(items.Count > 0)
  185. {
  186. _TMRobot.PostMoveItems(items.ToArray());
  187. }
  188. }
  189. return false;
  190. }
  191. private bool ModuleHasWafer(ModuleName mod, int nSlots)
  192. {
  193. for(int i= 0; i< nSlots; i++)
  194. {
  195. if (WaferManager.Instance.CheckHasWafer(mod, i))
  196. return true;
  197. }
  198. return false;
  199. }
  200. private bool PMsHasWafers()
  201. {
  202. foreach (var module in tmCycleRoutine)
  203. {
  204. if (ModuleHelper.IsPm(module) && ModuleHelper.IsInstalled(module))
  205. {
  206. if (WaferManager.Instance.CheckHasWafer(module, 0))
  207. return true;
  208. }
  209. }
  210. return false;
  211. }
  212. private bool IsPMsAvailabe()
  213. {
  214. foreach (var module in tmCycleRoutine)
  215. {
  216. if (ModuleHelper.IsPm(module)&&ModuleHelper.IsInstalled(module))
  217. {
  218. if (!IsModuleAvailable(module))
  219. return false;
  220. }
  221. }
  222. return true;
  223. }
  224. private bool Cycling()
  225. {
  226. if(IsPMsAvailabe() && IsModuleAvailable(_destinationModule) && IsModuleAvailable(_sourceModule) && _TMRobot.IsAvailable)
  227. {
  228. if (!PMsHasWafers() && _TMRobot.IsAvailable && _CycleWafers.Count == 0)
  229. return true;
  230. if(PMsHasWafers())
  231. {
  232. var pmSlots = GetReadyOutPMs();
  233. var inSlots = GetReadyInSlot(_destinationModule, _destinationSlotNumber);
  234. for(int i = 0; i < Math.Min(pmSlots.Count, inSlots.Count); i++)
  235. {
  236. _runningItems.Enqueue(new MoveItem(pmSlots[i], 0, _destinationModule, inSlots[i], Hand.Both));
  237. }
  238. }
  239. else
  240. {
  241. var InPMs = GetReadyInPMs();
  242. if(_CycleWafers.Count > 0 && InPMs.Count >= 1)
  243. {
  244. var item = _CycleWafers.Dequeue();
  245. _runningItems.Enqueue(new MoveItem(item.SourceModule, item.SourceSlot, InPMs[0], 0, Hand.Both));
  246. }
  247. if(_CycleWafers.Count > 0 && InPMs.Count >= 2)
  248. {
  249. var item = _CycleWafers.Dequeue();
  250. _runningItems.Enqueue(new MoveItem(item.SourceModule, item.SourceSlot, InPMs[1], 0, Hand.Both));
  251. }
  252. }
  253. if (_runningItems.Count > 0)
  254. {
  255. if (_TMRobot.PostMoveItems(_runningItems.ToArray()))
  256. _runningItems.Clear();
  257. }
  258. }
  259. return false;
  260. }
  261. private bool StartCycle()
  262. {
  263. _CycleWafers.Clear();
  264. for(int i = 0; i< _sourceSlotNumber; i++)
  265. {
  266. if (WaferManager.Instance.CheckHasWafer(_sourceModule, i))
  267. _CycleWafers.Enqueue(new MoveItem(_sourceModule, i, _destinationModule, INVALID_SLOT, Hand.None));
  268. }
  269. return true;
  270. }
  271. public void Abort()
  272. {
  273. CycleIndex = null;
  274. _TMRobot._entityTaskToken = (int)FSM_MSG.NONE;
  275. }
  276. }
  277. }