ReservoirEntity.cs 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458
  1. using Aitex.Core.RT.DataCenter;
  2. using Aitex.Core.RT.Device;
  3. using Aitex.Core.RT.Fsm;
  4. using Aitex.Core.RT.Log;
  5. using Aitex.Core.RT.OperationCenter;
  6. using Aitex.Core.RT.RecipeCenter;
  7. using Aitex.Core.Util;
  8. using MECF.Framework.Common.Equipment;
  9. using MECF.Framework.Common.Persistent.Reservoirs;
  10. using MECF.Framework.Common.RecipeCenter;
  11. using MECF.Framework.Common.ToolLayout;
  12. using CyberX8_Core;
  13. using CyberX8_RT.Devices.Metal;
  14. using CyberX8_RT.Devices.PowerSupplier;
  15. using CyberX8_RT.Devices.Temperature;
  16. using CyberX8_RT.Modules.Prewet;
  17. using System;
  18. using System.Collections.Generic;
  19. using System.Linq;
  20. using System.Text;
  21. using System.Threading.Tasks;
  22. using Aitex.Core.Utilities;
  23. using MECF.Framework.Common.ProcessCell;
  24. using CyberX8_RT.Modules.Metal;
  25. using CyberX8_RT.Modules.SRD;
  26. using System.Windows;
  27. using Aitex.Core.RT.SCCore;
  28. using SecsGem.Core.ItemModel;
  29. namespace CyberX8_RT.Modules.Reservoir
  30. {
  31. public class ReservoirEntity : Entity, IEntity, IModuleEntity
  32. {
  33. #region 常量
  34. private const string STRATUS = "Stratus";
  35. private const string AUTO = "Auto";
  36. private const string MANUAL = "Manual";
  37. private const string DISABLED = "Disabled";
  38. private const string ENGINEERING = "Engineering";
  39. private const string PRODUCTION = "Production";
  40. #endregion
  41. #region 内部变量
  42. /// <summary>
  43. /// 持久化数值
  44. /// </summary>
  45. private ReservoirsPersistentValue _persistentValue;
  46. /// <summary>
  47. /// S$H Initialize Routine
  48. /// </summary>
  49. private StandardHotReservoirInitializeRoutine _shInitializeRoutine;
  50. /// <summary>
  51. /// C&HInitialize Routine
  52. /// </summary>
  53. private CompactEmbranceInitializeRoutine _cmInitializeRoutine;
  54. /// <summary>
  55. /// DosingSystem Initialize Routine
  56. /// </summary>
  57. private DosingSystemInitializeRoutine _dosingSystemInitializeRoutine;
  58. /// <summary>
  59. /// Metal 集合
  60. /// </summary>
  61. private List<MetalCellDevice> _metalDevices = new List<MetalCellDevice>();
  62. /// <summary>
  63. /// 电源
  64. /// </summary>
  65. private CellPowerSupplier _powerSupplier;
  66. /// <summary>
  67. /// 温度控制设备
  68. /// </summary>
  69. private TemperatureController _temperatureController;
  70. /// <summary>
  71. /// Reservoir项
  72. /// </summary>
  73. private ReservoirItem _reservoirItem;
  74. #endregion
  75. #region 属性
  76. /// <summary>
  77. /// 模块名称
  78. /// </summary>
  79. public ModuleName Module { get; private set; }
  80. /// <summary>
  81. /// 是否Init
  82. /// </summary>
  83. public bool IsInit
  84. {
  85. get { return fsm.State == (int)ReservoirState.Init; }
  86. }
  87. /// <summary>
  88. /// 是否Idle
  89. /// </summary>
  90. public bool IsIdle
  91. {
  92. get
  93. {
  94. return fsm.State == (int)ReservoirState.Idle;
  95. }
  96. }
  97. /// <summary>
  98. /// 是否错误
  99. /// </summary>
  100. public bool IsError
  101. {
  102. get { return fsm.State == (int)ReservoirState.Error; }
  103. }
  104. /// <summary>
  105. /// 正在忙碌
  106. /// </summary>
  107. public bool IsBusy
  108. {
  109. get { return fsm.State == (int)ReservoirState.Initializing; }
  110. }
  111. /// <summary>
  112. /// 化学液
  113. /// </summary>
  114. public string Chemistry
  115. {
  116. get { return CurrentRecipe != null ? CurrentRecipe.Metal : ""; }
  117. }
  118. /// <summary>
  119. /// 是否禁用
  120. /// </summary>
  121. public bool IsDisable { get { return _persistentValue == null || _persistentValue.OperatingMode == DISABLED; } }
  122. /// <summary>
  123. /// 自动模式
  124. /// </summary>
  125. public bool IsAuto { get { return _persistentValue != null && _persistentValue.OperatingMode == AUTO; } }
  126. /// <summary>
  127. /// 自动模式
  128. /// </summary>
  129. public bool IsManual { get { return _persistentValue != null && _persistentValue.OperatingMode == MANUAL; } }
  130. /// <summary>
  131. /// 是否为工程模式
  132. /// </summary>
  133. public bool IsEngineering { get { return _persistentValue != null && _persistentValue.RecipeOperatingMode == ENGINEERING; } }
  134. /// <summary>
  135. /// 是否为产品模式
  136. /// </summary>
  137. public bool IsProduction { get { return _persistentValue != null && _persistentValue.RecipeOperatingMode == PRODUCTION; } }
  138. /// <summary>
  139. /// 状态机状态
  140. /// </summary>
  141. public ReservoirState State { get { return (ReservoirState)fsm.State; } }
  142. /// <summary>
  143. /// 是否初始化完成
  144. /// </summary>
  145. public bool IsInitialized { get { return fsm.State >= (int)ReservoirState.Initialized; } }
  146. /// <summary>
  147. /// 当前Recipe
  148. /// </summary>
  149. public ResRecipe CurrentRecipe { get { return LoadCurrentRecipe(); } }
  150. /// <summary>
  151. /// 用量
  152. /// </summary>
  153. public ReservoirUsage ReservoirUsage { get { return ReservoirUsageManager.Instance.GetReservoirUsage(Module.ToString()); } }
  154. /// <summary>
  155. /// Metals是否有Busy
  156. /// </summary>
  157. public bool IsMetalBusy
  158. {
  159. get {
  160. if(_metalDevices == null) return false;
  161. foreach(var item in _metalDevices)
  162. {
  163. MetalEntity metalEntity = Singleton<RouteManager>.Instance.GetModule<MetalEntity>(item.Module.ToString());
  164. if (metalEntity != null && metalEntity.IsBusy) return true;
  165. };
  166. return false;
  167. }
  168. }
  169. #endregion
  170. /// <summary>
  171. /// 构造函数
  172. /// </summary>
  173. /// <param name="module"></param>
  174. public ReservoirEntity(ModuleName module)
  175. {
  176. this.Module = module;
  177. InitializeParameter();
  178. InitialFsm();
  179. }
  180. /// <summary>
  181. /// 初始化
  182. /// </summary>
  183. /// <returns></returns>
  184. protected override bool Init()
  185. {
  186. InitializeRoutine();
  187. InitialDATA();
  188. InitializeOperation();
  189. return true;
  190. }
  191. /// <summary>
  192. /// 初始化参数
  193. /// </summary>
  194. private void InitializeParameter()
  195. {
  196. _persistentValue = ReservoirsPersistentManager.Instance.GetReservoirsPersistentValue(Module.ToString());
  197. if (_persistentValue == null)
  198. {
  199. LOG.WriteLog(eEvent.ERR_RESERVOIR, Module.ToString(), "Persistent Value Object is not exist");
  200. }
  201. LoadAllComponents();
  202. }
  203. /// <summary>
  204. /// 加载当前recipe
  205. /// </summary>
  206. /// <returns></returns>
  207. private ResRecipe LoadCurrentRecipe()
  208. {
  209. if(_persistentValue!=null&&!string.IsNullOrEmpty(_persistentValue.Recipe))
  210. {
  211. return RecipeFileManager.Instance.LoadGenericityRecipe<ResRecipe>(_persistentValue.Recipe);
  212. }
  213. return null;
  214. }
  215. /// <summary>
  216. /// 加载Metals
  217. /// </summary>
  218. private void LoadAllComponents()
  219. {
  220. _reservoirItem = ReservoirItemManager.Instance.GetReservoirItem(Module.ToString());
  221. if (_reservoirItem != null)
  222. {
  223. List<MetalItem> metalItems = _reservoirItem.MetalCells;
  224. if (metalItems != null && metalItems.Count > 0)
  225. {
  226. foreach (MetalItem metalItem in metalItems)
  227. {
  228. MetalCellDevice metalDevice = DEVICE.GetDevice<MetalCellDevice>(metalItem.ModuleName);
  229. if (metalDevice != null)
  230. {
  231. _metalDevices.Add(metalDevice);
  232. }
  233. }
  234. }
  235. if (!string.IsNullOrEmpty(_reservoirItem.CMMSupplyID))
  236. {
  237. _powerSupplier = DEVICE.GetDevice<CellPowerSupplier>(_reservoirItem.CMMSupplyID);
  238. }
  239. if (!string.IsNullOrEmpty(_reservoirItem.TCID))
  240. {
  241. _temperatureController = DEVICE.GetDevice<TemperatureController>(_reservoirItem.TCID);
  242. }
  243. }
  244. }
  245. /// 初始化状态机
  246. /// </summary>
  247. private void InitialFsm()
  248. {
  249. fsm = new StateMachine<ReservoirEntity>(Module.ToString(), (int)ReservoirState.Init, 100);
  250. fsm.EnableRepeatedMsg(true);
  251. AnyStateTransition(ReservoirMsg.Error, NullFunc, ReservoirState.Error);
  252. //Initialized
  253. Transition(ReservoirState.Error, ReservoirMsg.Initialize, InitializeAll, ReservoirState.Initializing);
  254. Transition(ReservoirState.Init, ReservoirMsg.Initialize, InitializeAll, ReservoirState.Initializing);
  255. Transition(ReservoirState.Idle, ReservoirMsg.Initialize, InitializeAll, ReservoirState.Initializing);
  256. Transition(ReservoirState.Initializing, FSM_MSG.TIMER, InitializeAllMonitor, ReservoirState.Idle);
  257. //直接进入Idle
  258. Transition(ReservoirState.Initialized, FSM_MSG.TIMER, NullFunc, ReservoirState.Idle);
  259. //Enter Init
  260. Transition(ReservoirState.Idle, ReservoirMsg.Init, NullFunc, ReservoirState.Init);
  261. EnumLoop<ReservoirState>.ForEach((item) => { fsm.MapState((int)item, item.ToString()); });
  262. EnumLoop<ReservoirMsg>.ForEach((item) => { fsm.MapMessage((int)item, item.ToString()); });
  263. }
  264. /// <summary>
  265. /// 初始化数据
  266. /// </summary>
  267. private void InitialDATA()
  268. {
  269. InitializeSVID();
  270. DATA.Subscribe($"{Module}.FsmState", () => ((ReservoirState)fsm.State).ToString(), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  271. DATA.Subscribe($"{Module}.IsIdle", () => IsIdle,SubscriptionAttribute.FLAG.IgnoreSaveDB);
  272. DATA.Subscribe($"{Module}.IsInit", () => IsInit, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  273. DATA.Subscribe($"{Module}.IsDisable", () => IsDisable,SubscriptionAttribute.FLAG.IgnoreSaveDB);
  274. DATA.Subscribe($"{Module}.IsBusy", () => IsBusy, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  275. DATA.Subscribe($"{Module}.IsError", () => IsError, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  276. DATA.Subscribe($"{Module}.Metals",()=>_metalDevices.Select(O=>O.Name).ToList(),SubscriptionAttribute.FLAG.IgnoreSaveDB);
  277. DATA.Subscribe($"{Module}.TemperatureControllerData", () => _temperatureController == null ? null : _temperatureController.TemperatureData, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  278. DATA.Subscribe($"{Module}.CmmPowerSupplierData", () => _powerSupplier == null ? null : _powerSupplier.PowerSupplierData, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  279. DATA.Subscribe($"{Module}.ReservoirUsage", () => ReservoirUsage, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  280. DATA.Subscribe($"{Module}.IsCMMInstalled", () => ReservoirItemManager.Instance.GetReservoirItem(Module.ToString()).CMMType == "Standard"? true : false, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  281. }
  282. /// <summary>
  283. /// 初始化SVID
  284. /// </summary>
  285. private void InitializeSVID()
  286. {
  287. DATA.Subscribe($"{Module}.State", () => ((ReservoirState)fsm.State).ToString(), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  288. DATA.Subscribe($"{Module}.OperatingMode", () => _persistentValue != null ? _persistentValue.OperatingMode : "None", SubscriptionAttribute.FLAG.IgnoreSaveDB);
  289. DATA.Subscribe($"{Module}.ReservoirTotalAmpHours",()=> ReservoirUsage!=null?ReservoirUsage.TotalUsage:0, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  290. DATA.Subscribe($"{Module}.ReservoirTotalAmpHours.WarningLimit", () => SC.GetValue<double>("Reservoir.ReservoirTotalAmpHoursWarningLimit"), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  291. DATA.Subscribe($"{Module}.ReservoirTotalAmpHours.FaultLimit", () => SC.GetValue<double>("Reservoir.ReservoirTotalAmpHoursFaultLimit"), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  292. DATA.Subscribe($"{Module}.BathTotalAmpHours",()=> ReservoirUsage != null ? ReservoirUsage.BathUsage:0, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  293. DATA.Subscribe($"{Module}.BathTotalAmpHours.WarningLimit", () => SC.GetValue<double>("Reservoir.BathTotalAmpHoursWarningLimit"), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  294. DATA.Subscribe($"{Module}.BathTotalAmpHours.FaultLimit", () => SC.GetValue<double>("Reservoir.BathTotalAmpHoursFaultLimit"), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  295. DATA.Subscribe($"{Module}.ANBathTotalAmpHours",()=> ReservoirUsage != null ? ReservoirUsage.AnodeUsage:0, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  296. DATA.Subscribe($"{Module}.ANBathTotalAmpHours.WarningLimit", () => SC.GetValue<double>("Reservoir.ANBathTotalAmpHoursWarningLimit"), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  297. DATA.Subscribe($"{Module}.ANBathTotalAmpHours.FaultLimit", () => SC.GetValue<double>("Reservoir.ANBathTotalAmpHoursFaultLimit"), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  298. DATA.Subscribe($"{Module}.MembraneTotalAmpHours",()=> ReservoirUsage != null ? ReservoirUsage.MembranceUsage:0, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  299. DATA.Subscribe($"{Module}.MembraneTotalAmpHours.WarningLimit", () => SC.GetValue<double>("Reservoir.MembraneTotalAmpHoursWarningLimit"), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  300. DATA.Subscribe($"{Module}.MembraneTotalAmpHours.FaultLimit", () => SC.GetValue<double>("Reservoir.MembraneTotalAmpHoursFaultLimit"), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  301. DATA.Subscribe($"{Module}.BathTotalDays", () => ReservoirUsage != null ? ReservoirUsage.BathUsageDays:0, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  302. DATA.Subscribe($"{Module}.BathTotalDays.WarningLimit", () => SC.GetValue<int>("Reservoir.BathTotalDaysWarningLimit"), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  303. DATA.Subscribe($"{Module}.BathTotalDays.FaultLimit", () => SC.GetValue<int>("Reservoir.BathTotalDaysFaultLimit"), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  304. DATA.Subscribe($"{Module}.ReservoirTotalWafers",()=> ReservoirUsage != null ? ReservoirUsage.TotalWafers:0, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  305. DATA.Subscribe($"{Module}.ReservoirTotalWafers.WarningLimit", () => SC.GetValue<int>("Reservoir.ReservoirTotalWafersWarningLimit"), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  306. DATA.Subscribe($"{Module}.ReservoirTotalWafers.FaultLimit", () => SC.GetValue<int>("Reservoir.ReservoirTotalWafersFaultLimit"), SubscriptionAttribute.FLAG.IgnoreSaveDB);
  307. DATA.Subscribe($"{Module}.Temperature", () => _temperatureController!=null? _temperatureController.TemperatureData.ReserviorTemperature:0, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  308. DATA.Subscribe($"{Module}.Temperature.ID", () => _temperatureController != null ? _temperatureController.Module : "", SubscriptionAttribute.FLAG.IgnoreSaveDB);
  309. DATA.Subscribe($"{Module}.Temperature.Status", () => _temperatureController != null ? _temperatureController.TemperatureData.Status : "", SubscriptionAttribute.FLAG.IgnoreSaveDB);
  310. DATA.Subscribe($"{Module}.Temperature.TargetTemperature", () => _temperatureController != null ? _temperatureController.TemperatureData.TargetTemperature : 0, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  311. DATA.Subscribe($"{Module}.Temperature.Alarm", () => _temperatureController != null ? _temperatureController.TemperatureData.Alarm : "", SubscriptionAttribute.FLAG.IgnoreSaveDB);
  312. DATA.Subscribe($"{Module}.Temperature.HeatExchangerTemperature", () => _temperatureController != null ? _temperatureController.TemperatureData.HeatExchangerTemperature : 0, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  313. }
  314. /// <summary>
  315. /// 初始化Routine
  316. /// </summary>
  317. private void InitializeRoutine()
  318. {
  319. _shInitializeRoutine = new StandardHotReservoirInitializeRoutine(Module.ToString());
  320. _cmInitializeRoutine=new CompactEmbranceInitializeRoutine(Module.ToString());
  321. _dosingSystemInitializeRoutine = new DosingSystemInitializeRoutine(Module.ToString());
  322. }
  323. /// <summary>
  324. /// 初始化Operation
  325. /// </summary>
  326. private void InitializeOperation()
  327. {
  328. OP.Subscribe($"{Module}.UpdateReservoirUsage", UpdateReservoirUsageAction);
  329. OP.Subscribe($"{Module}.InitializeAll", (cmd, args) => { return CheckToPostMessage<ReservoirState, ReservoirMsg>(eEvent.ERR_RESERVOIR, Module.ToString(), (int)ReservoirMsg.Initialize); });
  330. }
  331. /// <summary>
  332. /// EnterInit
  333. /// </summary>
  334. public void EnterInit()
  335. {
  336. if ((ReservoirState)fsm.State != ReservoirState.Idle) return;
  337. else
  338. {
  339. CheckToPostMessage<ReservoirState, ReservoirMsg>(eEvent.ERR_RESERVOIR, Module.ToString(), (int)ReservoirMsg.Init);
  340. }
  341. }
  342. /// <summary>
  343. /// UpdateReservoirUsageAction
  344. /// </summary>
  345. /// <param name="cmd"></param>
  346. /// <param name="param"></param>
  347. /// <returns></returns>
  348. private bool UpdateReservoirUsageAction(string cmd, object[] args)
  349. {
  350. if (args.Length > 2)
  351. {
  352. ReservoirUsageManager.Instance.UpdateReservoirUsageByManualOperation(args[0].ToString(), args[1].ToString(), args[2].ToString());
  353. LOG.WriteLog(eEvent.INFO_RESERVOIR, Module.ToString(), $"New {args[1].ToString()} value {args[2].ToString()} was input");
  354. }
  355. return true;
  356. }
  357. #region InitializeAll
  358. /// <summary>
  359. /// Initialize
  360. /// </summary>
  361. /// <param name="param"></param>
  362. /// <returns></returns>
  363. private bool InitializeAll(object[] param)
  364. {
  365. if (_metalDevices==null||_metalDevices.Count==0)
  366. {
  367. LOG.WriteLog(eEvent.ERR_RESERVOIR, Module.ToString(), "metal device is empty");
  368. return false;
  369. }
  370. if(_persistentValue==null)
  371. {
  372. LOG.WriteLog(eEvent.ERR_RESERVOIR, Module.ToString(), "persistent is null");
  373. return false;
  374. }
  375. if (_reservoirItem.SubType == STRATUS)
  376. {
  377. return _shInitializeRoutine.Start(_metalDevices, _powerSupplier, _temperatureController) == RState.Running;
  378. }
  379. else
  380. {
  381. return _cmInitializeRoutine.Start(_metalDevices, _temperatureController) ==RState.Running;
  382. }
  383. }
  384. /// <summary>
  385. /// Initialize 监控
  386. /// </summary>
  387. /// <param name="param"></param>
  388. /// <returns></returns>
  389. private bool InitializeAllMonitor(object[] param)
  390. {
  391. RState ret = RState.Init;
  392. if (_reservoirItem.SubType == STRATUS)
  393. {
  394. ret= _shInitializeRoutine.Monitor();
  395. }
  396. else
  397. {
  398. ret = _cmInitializeRoutine.Monitor();
  399. }
  400. if (ret == RState.Failed || ret == RState.Timeout)
  401. {
  402. PostMsg(ReservoirMsg.Error);
  403. return false;
  404. }
  405. return ret == RState.End;
  406. }
  407. #endregion
  408. public bool Check(int msg, out string reason, params object[] args)
  409. {
  410. reason = "";
  411. return true;
  412. }
  413. public bool CheckAcked(int msg)
  414. {
  415. return true;
  416. }
  417. public int Invoke(string function, params object[] args)
  418. {
  419. switch (function)
  420. {
  421. case "HomeAll":
  422. if (IsIdle)
  423. {
  424. return (int)ReservoirMsg.Initialize;
  425. }
  426. if (CheckToPostMessage<ReservoirState, ReservoirMsg>(eEvent.ERR_RESERVOIR, Module.ToString(), (int)ReservoirMsg.Initialize))
  427. {
  428. return (int)ReservoirMsg.Initialize;
  429. }
  430. else
  431. {
  432. return (int)FSM_MSG.NONE;
  433. }
  434. }
  435. return (int)FSM_MSG.NONE;
  436. }
  437. }
  438. }