RouteManager.cs 41 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075
  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Text;
  6. using System.Threading.Tasks;
  7. using Aitex.Core.RT.Fsm;
  8. using Aitex.Core.Common;
  9. using Aitex.Core.RT.DataCenter;
  10. using Aitex.Core.RT.Event;
  11. using Aitex.Core.RT.OperationCenter;
  12. using Aitex.Core.RT.SCCore;
  13. using Aitex.Core.Util;
  14. using MECF.Framework.Common.Jobs;
  15. using MECF.Framework.Common.Equipment;
  16. using MECF.Framework.Common.SubstrateTrackings;
  17. using CyberX8_Core;
  18. using Aitex.Core.RT.Log;
  19. using SecsGem.Core.Application;
  20. using CyberX8_RT.Modules.SRD;
  21. using CyberX8_RT.Modules.PUF;
  22. using CyberX8_RT.Modules.Dummy;
  23. using CyberX8_RT.Modules.Loader;
  24. using Aitex.Core.RT.Device;
  25. using CyberX8_RT.Modules.Prewet;
  26. using CyberX8_RT.Modules.Dryer;
  27. using MECF.Framework.Common.ToolLayout;
  28. using CyberX8_RT.Modules.Metal;
  29. using CyberX8_RT.Modules.Rinse;
  30. using MECF.Framework.Common.Routine;
  31. using CyberX8_RT.Dispatch;
  32. using CyberX8_RT.Modules.Reservoir;
  33. using System.IO;
  34. using CyberX8_RT.Modules.Buffer;
  35. using CyberX8_RT.Modules.Transporter;
  36. using CyberX8_RT.Devices.Loader;
  37. using CyberX8_RT.Devices.PUF;
  38. using CyberX8_RT.Devices.SRD;
  39. using MECF.Framework.Common.Alarm;
  40. using System.Diagnostics;
  41. using MECF.Framework.Common.WaferHolder;
  42. using CyberX8_RT.Schedulers;
  43. namespace CyberX8_RT.Modules
  44. {
  45. class RouteManager : Entity, IEntity
  46. {
  47. public enum MSG
  48. {
  49. MoveWafer,
  50. ReturnWafer,
  51. HomeUnit,
  52. PauseAuto,
  53. ResumeAuto,
  54. Stop,
  55. StartCycle,
  56. StopCycle,
  57. HOME,
  58. RESET,
  59. ABORT,
  60. ERROR,
  61. SetAutoMode,
  62. SetManualMode,
  63. ResetIdleCleanTime,
  64. ResetIdlePurgeTime,
  65. CreateJob,
  66. PauseJob,
  67. PauseAllJobs,
  68. ResumeJob,
  69. ResumeAllJobs,
  70. StartJob,
  71. StopJob,
  72. AbortJob,
  73. JobDone,
  74. CassetteLeave, //For unload light control off afer job done
  75. Map,
  76. ReturnAllWafer,
  77. TMCycle,
  78. Retry,
  79. ConfirmComplete,
  80. ResetWafers,
  81. FaStartJob,
  82. FaStopJob,
  83. }
  84. #region 属性
  85. public string Name { get; set; }
  86. public bool IsAutoMode
  87. {
  88. get
  89. {
  90. return fsm.State == (int)RtState.AutoRunning || fsm.State == (int)RtState.AutoIdle||fsm.State==(int)RtState.AutoAborting;
  91. }
  92. }
  93. public bool IsFaAutoMode
  94. {
  95. get
  96. {
  97. return fsm.State == (int)RtState.AutoRunning || fsm.State == (int)RtState.AutoIdle;
  98. }
  99. }
  100. public bool IsInit
  101. {
  102. get { return fsm.State == (int)RtState.Init; }
  103. }
  104. public bool IsIdle
  105. {
  106. get { return fsm.State == (int)RtState.Idle || fsm.State == (int)RtState.AutoIdle; }
  107. }
  108. public bool IsAlarm
  109. {
  110. get { return fsm.State == (int)RtState.Error; }
  111. }
  112. public bool IsPaused
  113. {
  114. get { return _jobCycle.CycleState == RState.Paused; }
  115. }
  116. public bool IsRunning
  117. {
  118. get
  119. {
  120. return !IsInit && !IsAlarm && !IsIdle;
  121. }
  122. }
  123. /// <summary>
  124. /// 是否AutoRunning
  125. /// </summary>
  126. public bool IsAutoRunning
  127. {
  128. get { return fsm.State == (int)RtState.AutoRunning||fsm.State==(int)RtState.AutoAborting; }
  129. }
  130. /// <summary>
  131. /// 是否
  132. /// </summary>
  133. public bool IsAutoIdle
  134. {
  135. get { return fsm.State == (int)RtState.AutoIdle; }
  136. }
  137. /// <summary>
  138. /// EFEM实体
  139. /// </summary>
  140. public EfemEntity EFEM { get; }
  141. public WaferSize WaferSize { get; private set; }
  142. #endregion
  143. #region 内部变量
  144. private string _systemControlIp;
  145. private ICycle _jobCycle;
  146. private ModuleHomeAllRoutine _homeAllRoutine;
  147. private List<ReservoirMetalHomeRoutine> _reservoirMetalRoutines=new List<ReservoirMetalHomeRoutine>();
  148. /// <summary>
  149. /// 模块实体字典
  150. /// </summary>
  151. private Dictionary<string, IModuleEntity> _moduleEntitiesDic = new Dictionary<string, IModuleEntity>();
  152. /// <summary>
  153. /// 模块类型实体字典
  154. /// </summary>
  155. private Dictionary<ModuleType, List<IModuleEntity>> _modultTypeEntitiesDic = new Dictionary<ModuleType, List<IModuleEntity>>();
  156. /// <summary>
  157. /// Home stopwatch
  158. /// </summary>
  159. private Stopwatch _homeStopWatch = new Stopwatch();
  160. #endregion
  161. /// <summary>
  162. /// 构造函数
  163. /// </summary>
  164. public RouteManager()
  165. {
  166. Name = "System";
  167. WaferSize = (WaferSize)SC.GetValue<int>("System.WaferSize");
  168. if (ModuleHelper.IsInstalled(ModuleName.EFEM))
  169. {
  170. EFEM = new EfemEntity();
  171. EFEM.Initialize();
  172. _moduleEntitiesDic[ModuleName.EFEM.ToString()] = EFEM;
  173. }
  174. InitialModuleList(PufItemManager.Instance.InstalledModules, typeof(PUFEntity),ModuleType.PUF);
  175. InitialModuleList(DummyCasseteItemManager.Instance.InstalledModules, typeof(DummyEntity),ModuleType.Dummy);
  176. InitialModuleList(LoaderItemManager.Instance.InstalledModules, typeof(LoaderEntity),ModuleType.Loader);
  177. InitialModuleList(TransporterItemManager.Instance.InstalledModules, typeof(TransporterEntity),ModuleType.Transporter);
  178. InitialModuleList(BufferItemManager.Instance.InstalledModules, typeof(BufferEntity), ModuleType.Buffer);
  179. InitialModuleList(PrewetItemManager.Instance.InstalledModules, typeof(PrewetEntity),ModuleType.Prewet);
  180. InitialModuleList(DryerItemManager.Instance.InstalledModules, typeof(DryerEntity),ModuleType.Dryer);
  181. InitialModuleList(MetalItemManager.Instance.InstalledModules, typeof(MetalEntity),ModuleType.Metal);
  182. InitialModuleList(RinseItemManager.Instance.InstalledModules, typeof(RinseEntity),ModuleType.Rinse);
  183. InitialModuleList(ReservoirItemManager.Instance.InstalledModules,typeof(ReservoirEntity),ModuleType.Reservoir);
  184. InitialModuleList(SrdItemManager.Instance.InstalledModules, typeof(SRDEntity), ModuleType.SRD);
  185. fsm = new StateMachine<RouteManager>(Name, (int)RtState.Init, 200);
  186. SubscribeOperation();
  187. SubscribeDataVariable();
  188. SchedulerSequenceManager.Instance.Initialize();
  189. }
  190. /// <summary>
  191. /// 初始化模块集合
  192. /// </summary>
  193. /// <param name="lst"></param>
  194. /// <param name="entityType"></param>
  195. private void InitialModuleList(List<string> lst,Type entityType,ModuleType moduleType)
  196. {
  197. foreach(string item in lst)
  198. {
  199. InitialModule(item, entityType,moduleType);
  200. }
  201. }
  202. /// <summary>
  203. /// 初始化模块对象
  204. /// </summary>
  205. /// <param name="item"></param>
  206. /// <param name="moduleType"></param>
  207. private void InitialModule(string item,Type entityType,ModuleType moduleType)
  208. {
  209. ModuleName moduleName = (ModuleName)Enum.Parse(typeof(ModuleName), item);
  210. IModuleEntity moduleEntity= (IModuleEntity)System.Activator.CreateInstance(entityType,moduleName);
  211. moduleEntity.Initialize();
  212. _moduleEntitiesDic[item] = moduleEntity;
  213. List<IModuleEntity> lst = new List<IModuleEntity>();
  214. if(_modultTypeEntitiesDic.ContainsKey(moduleType))
  215. {
  216. lst= _modultTypeEntitiesDic[moduleType];
  217. }
  218. else
  219. {
  220. _modultTypeEntitiesDic[moduleType] = lst;
  221. }
  222. lst.Add(moduleEntity);
  223. }
  224. public bool Check(int msg, out string reason, params object[] args)
  225. {
  226. if (!fsm.FindTransition(fsm.State, msg))
  227. {
  228. reason = String.Format("{0} is in {1} state,can not do {2}", Name, 0, (MSG)msg);
  229. return false;
  230. }
  231. if (msg == (int)MSG.StartCycle)
  232. {
  233. if (!IsAutoMode)
  234. {
  235. reason = String.Format("can not do {0}, isn't auto mode.", msg.ToString());
  236. return false;
  237. }
  238. }
  239. reason = "";
  240. return true;
  241. }
  242. /// <summary>
  243. /// 订阅数据
  244. /// </summary>
  245. void SubscribeDataVariable()
  246. {
  247. DATA.Subscribe("System.SystemControlIp", ()=>_systemControlIp, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  248. DATA.Subscribe("Rt.Status", () => ((RtState)fsm.State).ToString(), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  249. DATA.Subscribe("System.State", () => ((RtState)fsm.State).ToString(), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  250. DATA.Subscribe("System.IsAutoMode", () => IsAutoMode, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  251. DATA.Subscribe("System.IsAutoRunning", () => IsAutoRunning, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  252. DATA.Subscribe("System.IsIdle", () => IsIdle || IsInit, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  253. DATA.Subscribe("System.IsAlarm", () => IsAlarm, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  254. DATA.Subscribe("System.IsBusy", () => IsRunning, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  255. DATA.Subscribe("System.IsConnectedWithHost", () => CheckSecsGemOnline(Singleton<SecGemApplication>.Instance.ControlState),SubscriptionAttribute.FLAG.IgnoreSaveDB);
  256. DATA.Subscribe("System.IsDisconnectWithHost", () => CheckSecsGemOnline(Singleton<SecGemApplication>.Instance.ControlState),SubscriptionAttribute.FLAG.IgnoreSaveDB);
  257. DATA.Subscribe("System.EquipmentMode", () => IsAutoMode ? "Auto":"Manual", SubscriptionAttribute.FLAG.IgnoreSaveDB);
  258. DATA.Subscribe("EquipmentStatus", () =>
  259. {
  260. if (IsInit) return 0;
  261. if (IsIdle) return 1;
  262. if (IsAlarm) return 3;
  263. if (IsPaused) return 4;
  264. return 2;
  265. }, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  266. }
  267. /// <summary>
  268. /// 订阅操作
  269. /// </summary>
  270. void SubscribeOperation()
  271. {
  272. OP.Subscribe("ApplySystemControl", ApplySystemControl);
  273. OP.Subscribe("ReleaseSystemControl", ReleaseSystemControl);
  274. OP.Subscribe("CreateWafer", InvokeCreateWafer);
  275. OP.Subscribe("DeleteWafer", InvokeDeleteWafer);
  276. OP.Subscribe("System.Home", (cmd, args) => CheckToPostMessage((int)MSG.HOME, args));
  277. DATA.Subscribe("SYSTEM.FsmState", () => (((RtState)fsm.State).ToString()), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  278. OP.Subscribe("System.ReturnAllWafer", (string cmd, object[] args) =>
  279. {
  280. return CheckToPostMessage((int)MSG.ReturnAllWafer, args[0], args[1], args[2], args[3]);
  281. });
  282. OP.Subscribe("System.MoveWafer", (string cmd, object[] args) =>
  283. {
  284. if (!Enum.TryParse((string)args[0], out ModuleName source))
  285. {
  286. EV.PostWarningLog(Name, $"Parameter source {(string)args[0]} not valid");
  287. return false;
  288. }
  289. if (!Enum.TryParse((string)args[2], out ModuleName destination))
  290. {
  291. EV.PostWarningLog(Name, $"Parameter destination {(string)args[1]} not valid");
  292. return false;
  293. }
  294. return CheckToPostMessage((int)MSG.MoveWafer,
  295. source, (int)args[1],
  296. destination, (int)args[3],
  297. args[4], args[5],
  298. args[6], args[7], (string)args[8]);
  299. });
  300. OP.Subscribe("System.HomeAll", (string cmd, object[] args) =>
  301. {
  302. return CheckToPostMessage((int)MSG.HOME);
  303. });
  304. OP.Subscribe("System.Abort", (string cmd, object[] args) =>
  305. {
  306. return CheckToPostMessage((int)MSG.ABORT);
  307. });
  308. OP.Subscribe("System.Reset", (string cmd, object[] args) =>
  309. {
  310. return CheckToPostMessage((int)MSG.RESET);
  311. });
  312. OP.Subscribe("System.SetAutoMode", (string cmd, object[] args) =>
  313. {
  314. return CheckToPostMessage((int)MSG.SetAutoMode);
  315. });
  316. OP.Subscribe("System.SetManualMode", (string cmd, object[] args) =>
  317. {
  318. return CheckToPostMessage((int)MSG.SetManualMode);
  319. });
  320. OP.Subscribe("System.CreateJob", (string cmd, object[] args) =>
  321. {
  322. return CheckToPostMessage((int)MSG.CreateJob, args[0]);
  323. });
  324. OP.Subscribe("System.StartJob", (string cmd, object[] args) =>
  325. {
  326. return CheckToPostMessage((int)MSG.StartJob, args[0]);
  327. });
  328. OP.Subscribe("System.PauseJob", (string cmd, object[] args) =>
  329. {
  330. return CheckToPostMessage((int)MSG.PauseJob, args[0]);
  331. });
  332. OP.Subscribe("System.ResumeJob", (string cmd, object[] args) =>
  333. {
  334. return CheckToPostMessage((int)MSG.ResumeJob, args[0]);
  335. });
  336. OP.Subscribe("System.PauseAllJob", (cmd, args) => { return CheckToPostMessage((int)MSG.PauseAllJobs); }) ;
  337. OP.Subscribe("System.ResumeAllJob", (cmd, args) => { return CheckToPostMessage((int)MSG.ResumeAllJobs); });
  338. OP.Subscribe("System.StopJob", (string cmd, object[] args) =>
  339. {
  340. return CheckToPostMessage((int)MSG.StopJob, args[0]);
  341. });
  342. OP.Subscribe("System.AbortJob", (string cmd, object[] args) =>
  343. {
  344. return CheckToPostMessage((int)MSG.AbortJob, args[0]);
  345. });
  346. OP.Subscribe(RtOperation.SetConfig.ToString(), (name, args) =>
  347. {
  348. string sc_key = args[0] as string;
  349. if (!string.IsNullOrWhiteSpace(sc_key) && args.Length > 1)
  350. {
  351. SC.SetItemValue(sc_key, args[1]);
  352. }
  353. return true;
  354. });
  355. OP.Subscribe("System.ResetIdleCleanTime", (string cmd, object[] args) =>
  356. {
  357. return CheckToPostMessage((int)MSG.ResetIdleCleanTime, args[0]);
  358. });
  359. OP.Subscribe("System.ResetIdlePurgeTime", (string cmd, object[] args) =>
  360. {
  361. return CheckToPostMessage((int)MSG.ResetIdlePurgeTime, args[0]);
  362. });
  363. OP.Subscribe("System.SetWaferSize", (string cmd, object[] args) =>
  364. {
  365. string module = (string)args[0];
  366. string size = (string)args[1];
  367. switch (size)
  368. {
  369. case "3":
  370. WaferManager.Instance.UpdateWaferSize(ModuleHelper.Converter(module), 0, WaferSize.WS3);
  371. break;
  372. case "4":
  373. WaferManager.Instance.UpdateWaferSize(ModuleHelper.Converter(module), 0, WaferSize.WS4);
  374. break;
  375. case "6":
  376. WaferManager.Instance.UpdateWaferSize(ModuleHelper.Converter(module), 0, WaferSize.WS6);
  377. break;
  378. default:
  379. EV.PostWarningLog("System", $"wafer size {size} not valid");
  380. break;
  381. }
  382. return true;
  383. });
  384. //OP.Subscribe("System.CassetteLeave", (string cmd, object[] args) =>
  385. //{
  386. // return CheckToPostMessage((int)MSG.CassetteLeave);
  387. //});
  388. OP.Subscribe("System.Retry", (string cmd, object[] args) =>
  389. {
  390. return CheckToPostMessage((int)MSG.Retry, args[0]);
  391. });
  392. OP.Subscribe("System.ConfirmComplete", (string cmd, object[] args) =>
  393. {
  394. return CheckToPostMessage((int)MSG.ConfirmComplete, args[0]);
  395. });
  396. OP.Subscribe("WaferHolder.ResetLayoutWafers", (string cmd, object[] args) =>
  397. {
  398. return CheckToPostMessage((int)MSG.ResetWafers, args);
  399. });
  400. }
  401. /// <summary>
  402. /// 申请系统控制权
  403. /// </summary>
  404. /// <param name="cmd"></param>
  405. /// <param name="args"></param>
  406. /// <returns></returns>
  407. private bool ApplySystemControl(string cmd, object[] args)
  408. {
  409. if (args[0].ToString() == "")
  410. {
  411. LOG.WriteLog(eEvent.WARN_ROUTER, "System", $"Current IP does not allow the application of UI control permissions");
  412. return false;
  413. }
  414. string applyIp = args[0].ToString();
  415. if (string.IsNullOrEmpty(_systemControlIp))
  416. {
  417. _systemControlIp = applyIp;
  418. LOG.WriteLog(eEvent.EV_ROUTER, "System", $"{_systemControlIp} apply system control success");
  419. return true;
  420. }
  421. else if(_systemControlIp!=applyIp)
  422. {
  423. LOG.WriteLog(eEvent.WARN_ROUTER, "System", $"{_systemControlIp} already has system control permission");
  424. return false;
  425. }
  426. return false;
  427. }
  428. /// <summary>
  429. /// 释放系统控制权
  430. /// </summary>
  431. /// <param name="cmd"></param>
  432. /// <param name="args"></param>
  433. /// <returns></returns>
  434. private bool ReleaseSystemControl(string cmd, object[] args)
  435. {
  436. string releaseIp = args[0].ToString();
  437. if (_systemControlIp == releaseIp)
  438. {
  439. LOG.WriteLog(eEvent.EV_ROUTER, "System", $"{releaseIp} release system control success");
  440. _systemControlIp = "";
  441. }
  442. else
  443. {
  444. LOG.WriteLog(eEvent.ERR_ROUTER, "System", $"{releaseIp} is not SystemControlIp");
  445. }
  446. return false;
  447. }
  448. /// <summary>
  449. /// 检验Secsgem在线情况
  450. /// </summary>
  451. /// <param name="controlState"></param>
  452. /// <returns></returns>
  453. private bool CheckSecsGemOnline(SecsGem.Core.EnumData.ControlState controlState)
  454. {
  455. if (controlState == SecsGem.Core.EnumData.ControlState.OnlineLocal || controlState == SecsGem.Core.EnumData.ControlState.OnlineRemote)
  456. {
  457. return true;
  458. }
  459. return false;
  460. }
  461. public bool CheckToPostMessage(int msg, params object[] args)
  462. {
  463. if (!fsm.FindTransition(fsm.State, msg))
  464. {
  465. LOG.Write(eEvent.WARN_ROUTER, ModuleName.System, $"System is in {(RtState)fsm.State} state,can not do {(MSG)msg}");
  466. return false;
  467. }
  468. Running = true;
  469. fsm.PostMsg(msg, args);
  470. return true;
  471. }
  472. private bool InvokeCreateWafer(string arg1, object[] args)
  473. {
  474. ModuleName chamber = ModuleHelper.Converter(args[0].ToString());
  475. int slot = (int)args[1];
  476. WaferStatus state = WaferStatus.Normal;
  477. if (WaferManager.Instance.IsWaferSlotLocationValid(chamber, slot))
  478. {
  479. if (WaferManager.Instance.CheckHasWafer(chamber, slot))
  480. {
  481. LOG.Write(eEvent.EV_ROUTER, "System", string.Format("{0} slot {1} already has wafer.create wafer is not valid", chamber, slot));
  482. }
  483. else if (WaferManager.Instance.CreateWafer(chamber, slot, state) != null)
  484. {
  485. LOG.Write(eEvent.EV_WAFER_CREATE, ModuleName.System, chamber.ToString(), (slot + 1).ToString(), state.ToString());
  486. }
  487. }
  488. else
  489. {
  490. LOG.Write(eEvent.WARN_ROUTER, "System", string.Format("Invalid position,{0},{1}", chamber.ToString(), slot.ToString()));
  491. return false;
  492. }
  493. return true;
  494. }
  495. private bool InvokeDeleteWafer(string arg1, object[] args)
  496. {
  497. ModuleName chamber = ModuleHelper.Converter(args[0].ToString());
  498. int slot = (int)args[1];
  499. if (WaferManager.Instance.IsWaferSlotLocationValid(chamber, slot))
  500. {
  501. if (WaferManager.Instance.CheckHasWafer(chamber, slot))
  502. {
  503. WaferManager.Instance.DeleteWafer(chamber, slot);
  504. EV.PostMessage(ModuleName.System.ToString(), EventEnum.WaferDelete, chamber.ToString(), slot + 1);
  505. }
  506. else
  507. {
  508. LOG.Write(eEvent.EV_ROUTER, "System", string.Format("No wafer at {0} {1}, delete not valid", chamber.ToString(), slot + 1));
  509. }
  510. }
  511. else
  512. {
  513. LOG.Write(eEvent.WARN_ROUTER, "System", string.Format("Invalid position,{0},{1}", chamber.ToString(), slot.ToString()));
  514. return false;
  515. }
  516. return true;
  517. }
  518. private bool InvokeReturnWafer(string arg1, object[] args)
  519. {
  520. ModuleName target = ModuleHelper.Converter(args[0].ToString());
  521. int slot = (int)args[1];
  522. if (ModuleHelper.IsLoadPort(target))
  523. {
  524. LOG.Write(eEvent.WARN_ROUTER, "System", string.Format("Wafer already at LoadPort {0} {1}, return operation is not valid", target.ToString(), slot + 1));
  525. return false;
  526. }
  527. if (!WaferManager.Instance.IsWaferSlotLocationValid(target, slot))
  528. {
  529. LOG.Write(eEvent.WARN_ROUTER, "System", string.Format("Invalid position,{0},{1}", target.ToString(), slot.ToString()));
  530. return false;
  531. }
  532. WaferInfo wafer = WaferManager.Instance.GetWafer(target, slot);
  533. if (wafer.IsEmpty)
  534. {
  535. LOG.Write(eEvent.WARN_ROUTER, "System", string.Format("No wafer at {0} {1}, return operation is not valid", target.ToString(), slot + 1));
  536. return false;
  537. }
  538. return CheckToPostMessage((int)MSG.MoveWafer,
  539. target, slot,
  540. (ModuleName)wafer.OriginStation, wafer.OriginSlot,
  541. false, 0, false, 0, "Blade1");
  542. }
  543. protected override bool Init()
  544. {
  545. _jobCycle = new JobDispatcher();
  546. Singleton<FaRemoteManager>.Instance.JobCycle = _jobCycle;
  547. _homeAllRoutine = new ModuleHomeAllRoutine("System");
  548. BuildTransitionTable();
  549. return true;
  550. }
  551. private void BuildTransitionTable()
  552. {
  553. //Init sequence
  554. Transition(RtState.Init, MSG.HOME, FsmStartHome, RtState.Initializing);
  555. Transition(RtState.Idle, MSG.HOME, FsmStartHome, RtState.Initializing);
  556. Transition(RtState.Error, MSG.HOME, FsmStartHome, RtState.Initializing);
  557. EnterExitTransition<RtState, FSM_MSG>(RtState.AutoRunning, FsmEnterAutoRunning, FSM_MSG.NONE, FsmExitAutoTransfer);
  558. EnterExitTransition<RtState, FSM_MSG>(RtState.Transfer, null, FSM_MSG.NONE, FsmExitTransfer);
  559. EnterExitTransition<RtState, FSM_MSG>(RtState.ReturnWafer, null, FSM_MSG.NONE, FsmExitReturnWafer);
  560. AnyStateTransition(MSG.ERROR, FsmError, RtState.Error);
  561. Transition(RtState.Idle, FSM_MSG.TIMER, FsmMonitor, RtState.Idle);
  562. Transition(RtState.Init, FSM_MSG.TIMER, FsmMonitor, RtState.Init);
  563. Transition(RtState.Init, MSG.ABORT, FsmAbort, RtState.Init);
  564. Transition(RtState.Error, MSG.ABORT, FsmAbort, RtState.Init);
  565. Transition(RtState.Idle, MSG.ABORT, FsmAbort, RtState.Idle);
  566. Transition(RtState.Initializing, MSG.ABORT, FsmAbort, RtState.Init);
  567. Transition(RtState.Initializing, FSM_MSG.TIMER, FsmMonitorHome, RtState.Idle);
  568. Transition(RtState.Initializing, MSG.ERROR, FsmError, RtState.Error);
  569. //Auto/manual
  570. Transition(RtState.Idle, MSG.SetAutoMode, FsmStartAutoTransfer, RtState.AutoIdle);
  571. Transition(RtState.AutoAborting, FSM_MSG.TIMER, FsmAutoTransfer, RtState.AutoIdle);
  572. Transition(RtState.AutoRunning, FSM_MSG.TIMER, FsmAutoTransfer, RtState.AutoIdle);
  573. Transition(RtState.AutoRunning, MSG.ABORT, FsmAbort, RtState.AutoAborting);
  574. Transition(RtState.AutoAborting, MSG.SetManualMode, FsmStartAbortingSetManualMode, RtState.Idle);
  575. //Transition(RtState.AutoRunning, MSG.SetManualMode, FsmStartSetManualMode, RtState.Idle);
  576. Transition(RtState.AutoRunning, MSG.JobDone, FsmJobDone, RtState.AutoIdle);
  577. //Transition(RtState.AutoRunning, MSG.CassetteLeave, fCassetteLeave, RtState.AutoRunning); //For unload light control off afer job done
  578. Transition(RtState.AutoRunning, MSG.CreateJob, FsmCreateJob, RtState.AutoRunning);
  579. Transition(RtState.AutoRunning, MSG.StartJob, FsmStartJob, RtState.AutoRunning);
  580. Transition(RtState.AutoRunning, MSG.PauseJob, FsmPauseJob, RtState.AutoRunning);
  581. Transition(RtState.AutoRunning, MSG.PauseAllJobs, FsmPauseAllJobs, RtState.AutoRunning);
  582. Transition(RtState.AutoRunning, MSG.ResumeJob, FsmResumeJob, RtState.AutoRunning);
  583. Transition(RtState.AutoRunning, MSG.ResumeAllJobs, FsmResumeAllJobs, RtState.AutoRunning);
  584. Transition(RtState.AutoRunning, MSG.StopJob, FsmStopJob, RtState.AutoRunning);
  585. Transition(RtState.AutoRunning, MSG.Retry, FsmRetry, RtState.AutoRunning);
  586. Transition(RtState.AutoRunning, MSG.ConfirmComplete, FsmConfirmComplete, RtState.AutoRunning);
  587. Transition(RtState.AutoIdle, FSM_MSG.TIMER, FsmMonitorAutoIdle, RtState.AutoIdle);
  588. Transition(RtState.AutoIdle, MSG.SetManualMode, FsmStartSetManualMode, RtState.Idle);
  589. Transition(RtState.AutoIdle, MSG.CreateJob, FsmCreateJob, RtState.AutoIdle);
  590. Transition(RtState.AutoIdle, MSG.StartJob, FsmStartJob, RtState.AutoRunning);
  591. Transition(RtState.AutoIdle, MSG.PauseJob, FsmPauseJob, RtState.AutoIdle);
  592. Transition(RtState.AutoIdle, MSG.PauseAllJobs, FsmPauseAllJobs, RtState.AutoIdle);
  593. Transition(RtState.AutoIdle, MSG.ResumeJob, FsmResumeJob, RtState.AutoIdle);
  594. Transition(RtState.AutoIdle, MSG.ResumeAllJobs, FsmResumeAllJobs, RtState.AutoIdle);
  595. Transition(RtState.AutoIdle, MSG.StopJob, FsmStopJob, RtState.AutoIdle);
  596. Transition(RtState.AutoIdle, MSG.FaStartJob, FsmEnterAutoRunning, RtState.AutoRunning);
  597. Transition(RtState.AutoRunning, MSG.FaStartJob, FsmEnterAutoRunning, RtState.AutoRunning);
  598. Transition(RtState.AutoIdle, MSG.ResetWafers, ResetWafers, RtState.AutoIdle);
  599. Transition(RtState.Idle, MSG.ResetWafers, ResetWafers, RtState.Idle);
  600. Transition(RtState.Init, MSG.ResetWafers, ResetWafers, RtState.Init);
  601. }
  602. private bool FsmMonitor(object[] objs)
  603. {
  604. return true;
  605. }
  606. /// <summary>
  607. /// 检验前置条件
  608. /// </summary>
  609. /// <returns></returns>
  610. private bool CheckPreCondition()
  611. {
  612. if (ModuleHelper.IsInstalled(ModuleName.PUF1))
  613. {
  614. PufVacuum pufVacuum = DEVICE.GetDevice<PufVacuum>($"{ModuleName.PUF1}.Vacuum");
  615. if (!pufVacuum.IsChuckAReleased || !pufVacuum.ISChuckBReleased)
  616. {
  617. LOG.WriteLog(eEvent.ERR_AXIS, ModuleName.PUF1.ToString(), "Vacuum is on");
  618. return false;
  619. }
  620. }
  621. if (ModuleHelper.IsInstalled(ModuleName.Loader1))
  622. {
  623. LoaderSideDevice loaderSideADevice = DEVICE.GetDevice<LoaderSideDevice>($"{ModuleName.Loader1}.SideA");
  624. LoaderSideDevice loaderSideBDevice = DEVICE.GetDevice<LoaderSideDevice>($"{ModuleName.Loader1}.SideB");
  625. if (loaderSideADevice.SideData.CRSVacuum || loaderSideBDevice.SideData.CRSVacuum)
  626. {
  627. LOG.WriteLog(eEvent.ERR_AXIS, ModuleName.Loader1.ToString(), "Vacuum is on");
  628. return false;
  629. }
  630. }
  631. if (ModuleHelper.IsInstalled(ModuleName.SRD1))
  632. {
  633. SrdCommonDevice srdCommonDevice = DEVICE.GetDevice<SrdCommonDevice>($"{ModuleName.SRD1}.Common");
  634. if (!srdCommonDevice.CommonData.ChuckVacuum)
  635. {
  636. LOG.WriteLog(eEvent.ERR_AXIS, ModuleName.SRD1.ToString(), "Vacuum is on");
  637. return false;
  638. }
  639. }
  640. if (ModuleHelper.IsInstalled(ModuleName.SRD2))
  641. {
  642. SrdCommonDevice srdCommonDevice = DEVICE.GetDevice<SrdCommonDevice>($"{ModuleName.SRD2}.Common");
  643. if (!srdCommonDevice.CommonData.ChuckVacuum)
  644. {
  645. LOG.WriteLog(eEvent.ERR_AXIS, ModuleName.SRD2.ToString(), "Vacuum is on");
  646. return false;
  647. }
  648. }
  649. return true;
  650. }
  651. private bool FsmStartHome(object[] objs)
  652. {
  653. if (!CheckPreCondition())
  654. {
  655. return false;
  656. }
  657. List<string> keys = _moduleEntitiesDic.Keys.ToList();
  658. foreach (string item in keys)
  659. {
  660. if (!NeedSequenceHome(item))
  661. {
  662. IModuleEntity entity = _moduleEntitiesDic[item];
  663. if (!entity.IsDisable)
  664. {
  665. entity.Invoke("HomeAll");
  666. }
  667. }
  668. }
  669. _homeAllRoutine.Start();
  670. _reservoirMetalRoutines.Clear();
  671. foreach (string item in keys)
  672. {
  673. if (Enum.TryParse(item, out ModuleName moduleName))
  674. {
  675. if (ModuleHelper.IsReservoir(moduleName))
  676. {
  677. IModuleEntity moduleEntity = _moduleEntitiesDic[item];
  678. if (!moduleEntity.IsDisable)
  679. {
  680. ReservoirMetalHomeRoutine reservoirMetalHomeRoutine = new ReservoirMetalHomeRoutine(item);
  681. _reservoirMetalRoutines.Add(reservoirMetalHomeRoutine);
  682. }
  683. }
  684. }
  685. }
  686. foreach (ReservoirMetalHomeRoutine item in _reservoirMetalRoutines)
  687. {
  688. item.Start();
  689. }
  690. _homeStopWatch.Restart();
  691. _jobCycle.RemoveAllJob();
  692. return true;
  693. }
  694. /// <summary>
  695. /// 需要按照顺序Home
  696. /// </summary>
  697. /// <returns></returns>
  698. private bool NeedSequenceHome(string item)
  699. {
  700. bool result = item == ModuleName.PUF1.ToString() || item == ModuleName.Loader1.ToString() ||
  701. item == ModuleName.Transporter2.ToString() || item == ModuleName.Transporter1.ToString();
  702. if (result)
  703. {
  704. return true;
  705. }
  706. else
  707. {
  708. if(Enum.TryParse(item,out ModuleName moduleName))
  709. {
  710. if (!ModuleHelper.IsMetal(moduleName)&&!ModuleHelper.IsReservoir(moduleName))
  711. {
  712. return false;
  713. }
  714. }
  715. return true;
  716. }
  717. }
  718. private bool FsmMonitorHome(object[] objs)
  719. {
  720. RState rstate = _homeAllRoutine.Monitor();
  721. if (_homeStopWatch.ElapsedMilliseconds <= 500)
  722. {
  723. return false;
  724. }
  725. if (rstate == RState.Failed || rstate == RState.Timeout)
  726. {
  727. PostMsg(MSG.ERROR);
  728. return false;
  729. }
  730. List<string> keys = LoadInitializeCheckModuleList();
  731. int idleCount = 0;
  732. foreach (string item in keys)
  733. {
  734. IModuleEntity ModuleEntity = _moduleEntitiesDic[item];
  735. if (ModuleEntity.IsError && !NeedSequenceHome(item))
  736. {
  737. LOG.Write(eEvent.ERR_ROUTER, ModuleName.System, $"{ModuleEntity.Module} home error");
  738. PostMsg(MSG.ERROR);
  739. return true;
  740. }
  741. else if (ModuleEntity.IsIdle)
  742. {
  743. idleCount++;
  744. }
  745. }
  746. int homeCount = 0;
  747. foreach (ReservoirMetalHomeRoutine item in _reservoirMetalRoutines)
  748. {
  749. RState ret = item.Monitor();
  750. if (ret != RState.Running)
  751. {
  752. homeCount++;
  753. }
  754. }
  755. if (homeCount != _reservoirMetalRoutines.Count)
  756. {
  757. return false;
  758. }
  759. if (idleCount == keys.Count)
  760. {
  761. return true;
  762. }
  763. else
  764. {
  765. return false;
  766. }
  767. }
  768. /// <summary>
  769. /// 加载初始化检验模块集合
  770. /// </summary>
  771. /// <returns></returns>
  772. private List<string> LoadInitializeCheckModuleList()
  773. {
  774. string checkModuleList = SC.GetStringValue("System.InitializeCheckModuleList");
  775. List<string> moduleList= checkModuleList.Split(',').ToList();
  776. List<string> modules = new List<string>();
  777. foreach (var item in moduleList)
  778. {
  779. if(Enum.TryParse(item, out ModuleName moduleName))
  780. {
  781. if (ModuleHelper.IsInstalled(moduleName))
  782. {
  783. modules.Add(item);
  784. }
  785. }
  786. }
  787. return modules;
  788. }
  789. private bool FsmEnterAutoRunning(object[] objs)
  790. {
  791. return true;
  792. }
  793. private bool FsmExitAutoTransfer(object[] objs)
  794. {
  795. _jobCycle.Clear();
  796. return true;
  797. }
  798. private bool FsmExitTransfer(object[] objs)
  799. {
  800. return true;
  801. }
  802. private bool FsmExitReturnWafer(object[] objs)
  803. {
  804. return true;
  805. }
  806. private bool FsmError(object[] objs)
  807. {
  808. return true;
  809. }
  810. private bool FsmAbort(object[] objs)
  811. {
  812. //_manualTransfer.Clear();
  813. //_returnWafer.Clear();
  814. //_jobCycle.Clear();
  815. AlarmListManager.Instance.ClearAllAlarm();
  816. return _jobCycle.SystemAbort();
  817. }
  818. private bool FsmStartAutoTransfer(object[] objs)
  819. {
  820. return _jobCycle.Start(objs) == RState.Running;
  821. }
  822. private bool FsmAutoTransfer(object[] objs)
  823. {
  824. if (_jobCycle.CheckJobJustDone(out string jobInfo))
  825. {
  826. //EV.PostPopDialogMessage(EventLevel.InformationNoDelay, "Job complete", jobInfo);
  827. LOG.Write(eEvent.EV_ROUTER, ModuleName.System, $"Job complete,{jobInfo}");
  828. }
  829. if (_jobCycle.CheckAllJobDone())
  830. {
  831. if (!CheckToPostMessage((int)MSG.JobDone))
  832. return false;
  833. }
  834. RState ret = _jobCycle.Monitor();
  835. if (ret == RState.Failed)
  836. {
  837. if (!CheckToPostMessage((int)MSG.ERROR))
  838. return false;
  839. }
  840. return ret == RState.End;
  841. }
  842. private bool FsmAbortAutoTransfer(object[] objs)
  843. {
  844. _jobCycle.Clear();
  845. return true;
  846. }
  847. private bool FsmJobDone(object[] objs)
  848. {
  849. return true;
  850. }
  851. private bool FsmCreateJob(object[] objs)
  852. {
  853. return _jobCycle.CreateJob((Dictionary<string, object>)objs[0],out string reason);
  854. }
  855. private bool FsmStartJob(object[] objs)
  856. {
  857. return _jobCycle.StartJob((string)objs[0],out string reason);
  858. }
  859. private bool FsmPauseJob(object[] objs)
  860. {
  861. return _jobCycle.PauseJob((string)objs[0],out string reason);
  862. }
  863. private bool FsmResumeJob(object[] objs)
  864. {
  865. return _jobCycle.ResumeJob((string)objs[0],out string reason);
  866. }
  867. private bool FsmPauseAllJobs(object[] objs)
  868. {
  869. return _jobCycle.PauseAllJobs();
  870. }
  871. private bool FsmResumeAllJobs(object[] objs)
  872. {
  873. return _jobCycle.ResumeAllJobs();
  874. }
  875. private bool FsmStopJob(object[] objs)
  876. {
  877. return _jobCycle.StopJob((string)objs[0], out string reason);
  878. }
  879. /// <summary>
  880. /// 重试
  881. /// </summary>
  882. /// <param name="objs"></param>
  883. /// <returns></returns>
  884. private bool FsmRetry(object[] objs)
  885. {
  886. string moduleName = objs[0].ToString();
  887. IModuleEntity moduleEntity = GetModule<IModuleEntity>(moduleName);
  888. moduleEntity.Invoke("Retry");
  889. return true;
  890. }
  891. /// <summary>
  892. /// 重试
  893. /// </summary>
  894. /// <param name="objs"></param>
  895. /// <returns></returns>
  896. private bool FsmConfirmComplete(object[] objs)
  897. {
  898. string moduleName = objs[0].ToString();
  899. IModuleEntity moduleEntity = GetModule<IModuleEntity>(moduleName);
  900. moduleEntity.Invoke("ConfirmComplete");
  901. return true;
  902. }
  903. private bool FsmMonitorAutoIdle(object[] objs)
  904. {
  905. RState ret = _jobCycle.Monitor();
  906. return ret == RState.End;
  907. }
  908. private bool FsmStartAbortingSetManualMode(object[] objs)
  909. {
  910. return _jobCycle.RemoveAllJob();
  911. }
  912. private bool FsmStartSetManualMode(object[] objs)
  913. {
  914. if (_jobCycle.HasJobRunning)
  915. {
  916. LOG.Write(eEvent.WARN_ROUTER, "System", "Can not change to manual mode, abort running job first");
  917. return false;
  918. }
  919. return true;
  920. }
  921. /// <summary>
  922. /// 重置Wafers
  923. /// </summary>
  924. /// <param name="objs"></param>
  925. /// <returns></returns>
  926. private bool ResetWafers(object[] objs)
  927. {
  928. return WaferHolderManager.Instance.ResetLayoutWafers();
  929. }
  930. /// <summary>
  931. /// 获取模块对象
  932. /// </summary>
  933. /// <typeparam name="T"></typeparam>
  934. /// <param name="name"></param>
  935. /// <returns></returns>
  936. public T GetModule<T>(string name) where T : class, IModuleEntity
  937. {
  938. return _moduleEntitiesDic.ContainsKey(name)?_moduleEntitiesDic[name] as T:default(T);
  939. }
  940. /// <summary>
  941. /// 根据模块类型获取模块集合
  942. /// </summary>
  943. /// <param name="type"></param>
  944. /// <returns></returns>
  945. public List<IModuleEntity> GetModulesByModuleType(ModuleType type)
  946. {
  947. return _modultTypeEntitiesDic.ContainsKey(type) ? _modultTypeEntitiesDic[type] : new List<IModuleEntity>();
  948. }
  949. }
  950. }