MetalCellDevice.cs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461
  1. using Aitex.Core.RT.DataCenter;
  2. using Aitex.Core.RT.Device;
  3. using Aitex.Core.RT.Log;
  4. using Aitex.Core.Util;
  5. using MECF.Framework.Common.Persistent.Reservoirs;
  6. using MECF.Framework.Common.ToolLayout;
  7. using CyberX8_Core;
  8. using CyberX8_RT.Devices.LinMot;
  9. using CyberX8_RT.Devices.PowerSupplier;
  10. using Aitex.Core.RT.OperationCenter;
  11. using CyberX8_RT.Modules;
  12. using CyberX8_RT.Modules.Metal;
  13. namespace CyberX8_RT.Devices.Metal
  14. {
  15. public class MetalCellDevice : BaseDevice, IDevice
  16. {
  17. #region 常量
  18. private const string PERSISTENT_VALUE = "PersistentValue";
  19. private const string AUTO = "Auto";
  20. private const string MANUAL = "Manual";
  21. private const string STRATUS = "Stratus";
  22. private const string DISABLED = "Disabled";
  23. #endregion
  24. #region 内部变量
  25. /// <summary>
  26. /// 操作当前状态
  27. /// </summary>
  28. protected RState _status;
  29. /// <summary>
  30. /// 持久化数据
  31. /// </summary>
  32. protected MetalPersistentValue _persistentValue;
  33. /// <summary>
  34. /// A面PowerSupplier
  35. /// </summary>
  36. protected CellPowerSupplier _sideAPowerSupplier;
  37. /// <summary>
  38. /// B面PowerSupplier
  39. /// </summary>
  40. protected CellPowerSupplier _sideBPowerSupplier;
  41. /// <summary>
  42. /// Linmot
  43. /// </summary>
  44. protected LinMotAxis _linmotAxis;
  45. /// <summary>
  46. /// Metal项
  47. /// </summary>
  48. private MetalItem _metalItem;
  49. #endregion
  50. #region 属性
  51. /// <summary>
  52. /// 状态
  53. /// </summary>
  54. public RState Status { get { return _status; } }
  55. /// <summary>
  56. /// 是否禁用
  57. /// </summary>
  58. public bool IsDisable { get { return _persistentValue == null || _persistentValue.OperatingMode == DISABLED; } }
  59. /// <summary>
  60. /// clamp off状态
  61. /// </summary>
  62. public bool ClampOff
  63. {
  64. get
  65. {
  66. if (_metalItem.SubType == STRATUS)
  67. {
  68. StandardHotMetalDevice metalDevice = DEVICE.GetDevice<StandardHotMetalDevice>(Module.ToString());
  69. return !metalDevice.MetalDeviceData.WaferHolderClamp;
  70. }
  71. else
  72. {
  73. CompactMembranMetalDevice metalDevice = DEVICE.GetDevice<CompactMembranMetalDevice>(Module.ToString());
  74. return !metalDevice.MetalDeviceData.WHClamp && metalDevice.MetalDeviceData.WHUnclamp;
  75. }
  76. }
  77. }
  78. /// <summary>
  79. /// 操作模式
  80. /// </summary>
  81. public string OperationMode { get { return _persistentValue.OperatingMode; } }
  82. /// <summary>
  83. /// 工程模式
  84. /// </summary>
  85. public string EngineerMode { get { return _persistentValue.RecipeOperatingMode; } }
  86. /// <summary>
  87. /// A面PowerSupplier
  88. /// </summary>
  89. public CellPowerSupplier SideAPowerSupplier { get { return _sideAPowerSupplier; } }
  90. /// <summary>
  91. /// B面PowerSupplier
  92. /// </summary>
  93. public CellPowerSupplier SideBPowerSupplier { get { return _sideBPowerSupplier; } }
  94. /// <summary>
  95. /// 是否为Auto
  96. /// </summary>
  97. public bool IsAuto { get { return _persistentValue != null ? _persistentValue.OperatingMode == AUTO : false; } }
  98. /// <summary>
  99. /// 是否为Auto
  100. /// </summary>
  101. public bool IsManual { get { return _persistentValue != null ? _persistentValue.OperatingMode == MANUAL : false; } }
  102. /// <summary>
  103. /// linmot motor on
  104. /// </summary>
  105. public bool IsLinmotMotorOn { get { return _linmotAxis != null ? _linmotAxis.IsMotorOn : false; } }
  106. #endregion
  107. /// <summary>
  108. /// 构造函数
  109. /// </summary>
  110. /// <param name="moduleName"></param>
  111. public MetalCellDevice(string moduleName) : base(moduleName, moduleName, moduleName, moduleName)
  112. {
  113. }
  114. /// <summary>
  115. /// 初始化
  116. /// </summary>
  117. /// <returns></returns>
  118. public virtual bool Initialize()
  119. {
  120. InitializeParameter();
  121. SubscribeData();
  122. InitializeOperation();
  123. return true;
  124. }
  125. /// <summary>
  126. /// 定时器执行
  127. /// </summary>
  128. public virtual bool OnTimer(int interval)
  129. {
  130. return true;
  131. }
  132. /// <summary>
  133. /// 初始化参数
  134. /// </summary>
  135. private void InitializeParameter()
  136. {
  137. _persistentValue = MetalPersistentManager.Instance.GetMetalPersistentValue(Module);
  138. if(_persistentValue == null)
  139. {
  140. LOG.WriteLog(eEvent.ERR_RESERVOIR, Module, "Persistent Value Object is not exist");
  141. }
  142. _metalItem = MetalItemManager.Instance.GetMetalItem(Module);
  143. if (_metalItem != null)
  144. {
  145. _sideAPowerSupplier = DEVICE.GetDevice<CellPowerSupplier>(_metalItem.PlatingPowerSupplyAID);
  146. _sideBPowerSupplier = DEVICE.GetDevice<CellPowerSupplier>(_metalItem.PlatingPowerSupplyBID);
  147. _linmotAxis = DEVICE.GetDevice<LinMotAxis>(_metalItem.LinmotID);
  148. }
  149. }
  150. /// <summary>
  151. /// 订阅数据
  152. /// </summary>
  153. private void SubscribeData()
  154. {
  155. DATA.Subscribe($"{Module}.{PERSISTENT_VALUE}", () => _persistentValue, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  156. DATA.Subscribe($"{Module}.SideAPowerSupplierData", () => _sideAPowerSupplier.PowerSupplierData, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  157. DATA.Subscribe($"{Module}.SideBPowerSupplierData", () => _sideBPowerSupplier.PowerSupplierData, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  158. DATA.Subscribe($"{Module}.SideAPowerSupplier.ID", () => _sideAPowerSupplier.Module, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  159. DATA.Subscribe($"{Module}.SideAPowerSupplier.IsConnected", () => _sideAPowerSupplier.IsConnected, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  160. DATA.Subscribe($"{Module}.SideAPowerSupplier.Voltage", () => _sideAPowerSupplier.PowerSupplierData.Voltage, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  161. DATA.Subscribe($"{Module}.SideAPowerSupplier.Current", () => _sideAPowerSupplier.PowerSupplierData.Current, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  162. DATA.Subscribe($"{Module}.SideAPowerSupplier.SetPoint", () => _sideAPowerSupplier.PowerSupplierData.SetPoint, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  163. DATA.Subscribe($"{Module}.SideAPowerSupplier.RunModel", () => _sideAPowerSupplier.PowerSupplierData.PowerRunModelContent, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  164. DATA.Subscribe($"{Module}.SideAPowerSupplier.PowerControl", () => _sideAPowerSupplier.PowerSupplierData.PowerControlContent, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  165. DATA.Subscribe($"{Module}.SideAPowerSupplier.PowerStatus", () => _sideAPowerSupplier.PowerSupplierData.PowerStatusContent, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  166. DATA.Subscribe($"{Module}.SideAPowerSupplier.Enable", () => _sideBPowerSupplier.PowerSupplierData.Enabled, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  167. DATA.Subscribe($"{Module}.SideBPowerSupplier.ID", () => _sideBPowerSupplier.Module, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  168. DATA.Subscribe($"{Module}.SideBPowerSupplier.IsConnected", () => _sideAPowerSupplier.IsConnected, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  169. DATA.Subscribe($"{Module}.SideBPowerSupplier.Voltage", () => _sideBPowerSupplier.PowerSupplierData.Voltage, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  170. DATA.Subscribe($"{Module}.SideBPowerSupplier.Current", () => _sideBPowerSupplier.PowerSupplierData.Current, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  171. DATA.Subscribe($"{Module}.SideBPowerSupplier.SetPoint", () => _sideBPowerSupplier.PowerSupplierData.SetPoint, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  172. DATA.Subscribe($"{Module}.SideBPowerSupplier.RunModel", () => _sideBPowerSupplier.PowerSupplierData.PowerRunModelContent, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  173. DATA.Subscribe($"{Module}.SideBPowerSupplier.PowerControl", () => _sideBPowerSupplier.PowerSupplierData.PowerControlContent, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  174. DATA.Subscribe($"{Module}.SideBPowerSupplier.PowerStatus", () => _sideBPowerSupplier.PowerSupplierData.PowerStatusContent, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  175. DATA.Subscribe($"{Module}.SideBPowerSupplier.Enable", () => _sideBPowerSupplier.PowerSupplierData.Enabled, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  176. DATA.Subscribe($"{Module}.Linmot.ID", () => _linmotAxis.Module, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  177. DATA.Subscribe($"{Module}.Linmot.IsMotorOn", () => _linmotAxis.IsMotorOn, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  178. DATA.Subscribe($"{Module}.Linmot.IsError", () => _linmotAxis.IsError, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  179. DATA.Subscribe($"{Module}.Linmot.IsSwitchOn", () => _linmotAxis.IsSwitchOn, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  180. DATA.Subscribe($"{Module}.Linmot.CurveSpeed", () => _linmotAxis.CurveSpeed, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  181. DATA.Subscribe($"{Module}.Linmot.ErrorCode", () => _linmotAxis.ErrorCode, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  182. DATA.Subscribe($"{Module}.Linmot.CurrentPosition", () => _linmotAxis.CurrentPosition, SubscriptionAttribute.FLAG.IgnoreSaveDB);
  183. }
  184. /// <summary>
  185. /// 初始化操作
  186. /// </summary>
  187. protected virtual void InitializeOperation()
  188. {
  189. OP.Subscribe($"{Module}.DisabledAction", DisabledOperation);
  190. OP.Subscribe($"{Module}.ManualAction", ManualOperation);
  191. OP.Subscribe($"{Module}.AutoAction", AutoOperation);
  192. OP.Subscribe($"{Module}.EngineeringModeAction", EngineeringModeOperation);
  193. OP.Subscribe($"{Module}.ProductionModeAction", ProductionModeOperation);
  194. OP.Subscribe($"{Module}.SetMetalWaferSize", (cmd, args) => { return SetMetalWaferSize(cmd, args); });
  195. }
  196. /// <summary>
  197. /// 开始Curve
  198. /// </summary>
  199. /// <param name="speed"></param>
  200. /// <returns></returns>
  201. public bool StartCurveMotion(int speed)
  202. {
  203. if (_linmotAxis != null)
  204. {
  205. return _linmotAxis.StartCurve(speed);
  206. }
  207. else
  208. {
  209. LOG.WriteLog(eEvent.ERR_METAL, Module, "linmot is null");
  210. return false;
  211. }
  212. }
  213. /// <summary>
  214. /// 开始Curve
  215. /// </summary>
  216. /// <param name="speed"></param>
  217. /// <returns></returns>
  218. public bool ChangeCurveSpeedMotion(int speed)
  219. {
  220. if (_linmotAxis != null)
  221. {
  222. return _linmotAxis.ChangeCurveSpeed(speed);
  223. }
  224. else
  225. {
  226. LOG.WriteLog(eEvent.ERR_METAL, Module, "linmot is null");
  227. return false;
  228. }
  229. }
  230. /// <summary>
  231. /// ResetLinmot
  232. /// </summary>
  233. /// <returns></returns>
  234. public bool ResetLinmot()
  235. {
  236. if (_linmotAxis != null)
  237. {
  238. return _linmotAxis.ResetOperation("", false);
  239. }
  240. else
  241. {
  242. LOG.WriteLog(eEvent.ERR_METAL, Module, "linmot is null");
  243. return false;
  244. }
  245. }
  246. /// <summary>
  247. /// 检验Linmot Routine状态是否为结束状态
  248. /// </summary>
  249. /// <returns></returns>
  250. public bool CheckLinmotRoutineEnd()
  251. {
  252. if (_linmotAxis != null)
  253. {
  254. return _linmotAxis.Status == RState.End;
  255. }
  256. else
  257. {
  258. LOG.WriteLog(eEvent.ERR_METAL, Module, "linmot is null");
  259. return false;
  260. }
  261. }
  262. /// <summary>
  263. /// 检验Linmot Routine状态是否为错误状态
  264. /// </summary>
  265. /// <returns></returns>
  266. public bool CheckLinmotRoutineError()
  267. {
  268. if (_linmotAxis != null)
  269. {
  270. return _linmotAxis.Status == RState.Failed || _linmotAxis.Status == RState.Timeout;
  271. }
  272. else
  273. {
  274. return false;
  275. }
  276. }
  277. /// <summary>
  278. /// 停止Linmot
  279. /// </summary>
  280. /// <returns></returns>
  281. public bool StopLinmot()
  282. {
  283. if (_linmotAxis != null)
  284. {
  285. return _linmotAxis.StopOperation("", null);
  286. }
  287. else
  288. {
  289. LOG.WriteLog(eEvent.ERR_METAL, Module, "linmot is null");
  290. return false;
  291. }
  292. }
  293. #region Operation
  294. /// <summary>
  295. /// DisabledAction
  296. /// </summary>
  297. /// <param name="cmd"></param>
  298. /// <param name="param"></param>
  299. /// <returns></returns>
  300. public bool DisabledOperation(string cmd, object[] args)
  301. {
  302. string currentOperation = "Disabled";
  303. MetalEntity metalEntity = Singleton<RouteManager>.Instance.GetModule<MetalEntity>(Module);
  304. if (metalEntity != null && _persistentValue != null && _persistentValue.OperatingMode != currentOperation)
  305. {
  306. string preOperation = _persistentValue.OperatingMode;
  307. if (metalEntity.IsBusy)
  308. {
  309. LOG.WriteLog(eEvent.ERR_METAL, Module, $"{Module} is Busy, can't switch to Disabled mode");
  310. return false;
  311. }
  312. metalEntity.EnterInit();
  313. _persistentValue.OperatingMode = currentOperation;
  314. LOG.WriteLog(eEvent.INFO_METAL, Module, $"Operating mode is switched from {preOperation} to {currentOperation}");
  315. }
  316. MetalPersistentManager.Instance.UpdatePersistentValue(Module);
  317. return true;
  318. }
  319. /// <summary>
  320. /// ManualAction
  321. /// </summary>
  322. /// <param name="cmd"></param>
  323. /// <param name="param"></param>
  324. /// <returns></returns>
  325. public bool ManualOperation(string cmd, object[] args)
  326. {
  327. string currentOperation = "Manual";
  328. MetalEntity metalEntity = Singleton<RouteManager>.Instance.GetModule<MetalEntity>(Module);
  329. if (metalEntity != null && _persistentValue != null && _persistentValue.OperatingMode != currentOperation)
  330. {
  331. string preOperation = _persistentValue.OperatingMode;
  332. if (metalEntity.IsBusy)
  333. {
  334. LOG.WriteLog(eEvent.ERR_METAL, Module, $"{Module} is Busy, can't switch to Manual mode");
  335. return false;
  336. }
  337. metalEntity.EnterInit();
  338. _persistentValue.OperatingMode = currentOperation;
  339. LOG.WriteLog(eEvent.INFO_METAL, Module, $"Operating mode is switched from {preOperation} to {currentOperation}");
  340. }
  341. MetalPersistentManager.Instance.UpdatePersistentValue(Module);
  342. return true;
  343. }
  344. /// <summary>
  345. /// AutoAction
  346. /// </summary>
  347. /// <param name="cmd"></param>
  348. /// <param name="param"></param>
  349. /// <returns></returns>
  350. public bool AutoOperation(string cmd, object[] args)
  351. {
  352. string currentOperation = "Auto";
  353. MetalEntity metalEntity = Singleton<RouteManager>.Instance.GetModule<MetalEntity>(Module);
  354. if (metalEntity != null && _persistentValue != null && _persistentValue.OperatingMode != currentOperation)
  355. {
  356. string preOperation = _persistentValue.OperatingMode;
  357. if (metalEntity.IsBusy)
  358. {
  359. LOG.WriteLog(eEvent.ERR_METAL, Module, $"{Module} is Busy, can't switch to Auto mode");
  360. return false;
  361. }
  362. metalEntity.EnterInit();
  363. _persistentValue.OperatingMode = currentOperation;
  364. LOG.WriteLog(eEvent.INFO_METAL, Module, $"Operating mode is switched from {preOperation} to {currentOperation}");
  365. }
  366. MetalPersistentManager.Instance.UpdatePersistentValue(Module);
  367. return true;
  368. }
  369. /// <summary>
  370. /// EngineeringModeAction
  371. /// </summary>
  372. /// <param name="cmd"></param>
  373. /// <param name="param"></param>
  374. /// <returns></returns>
  375. private bool EngineeringModeOperation(string cmd, object[] args)
  376. {
  377. string currentRecipeOperation = "Engineering";
  378. if (_persistentValue != null)
  379. {
  380. _persistentValue.RecipeOperatingMode = currentRecipeOperation;
  381. }
  382. MetalPersistentManager.Instance.UpdatePersistentValue(Module);
  383. return true;
  384. }
  385. /// <summary>
  386. /// ProductionAction
  387. /// </summary>
  388. /// <param name="cmd"></param>
  389. /// <param name="param"></param>
  390. /// <returns></returns>
  391. private bool ProductionModeOperation(string cmd, object[] args)
  392. {
  393. string currentRecipeOperation = "Production";
  394. if (_persistentValue != null)
  395. {
  396. _persistentValue.RecipeOperatingMode = currentRecipeOperation;
  397. }
  398. MetalPersistentManager.Instance.UpdatePersistentValue(Module);
  399. return true;
  400. }
  401. private bool SetMetalWaferSize(string cmd, object[] args)
  402. {
  403. string metalWaferSize = args[0] as string;
  404. if (_persistentValue != null)
  405. {
  406. _persistentValue.MetalWaferSize = int.Parse(metalWaferSize);
  407. }
  408. MetalPersistentManager.Instance.UpdatePersistentValue(Module);
  409. return true;
  410. }
  411. #endregion
  412. #region Clamp
  413. /// <summary>
  414. /// Clamp Off
  415. /// </summary>
  416. /// <returns></returns>
  417. public bool WaferHolderClampOff()
  418. {
  419. if (_metalItem.SubType == STRATUS)
  420. {
  421. StandardHotMetalDevice metalDevice = DEVICE.GetDevice<StandardHotMetalDevice>(Module.ToString());
  422. return metalDevice.WaferHolderClampOff("", null);
  423. }
  424. else
  425. {
  426. CompactMembranMetalDevice metalDevice = DEVICE.GetDevice<CompactMembranMetalDevice>(Module.ToString());
  427. return metalDevice.WaferHolderUnclampOn("", null);
  428. }
  429. }
  430. #endregion
  431. public virtual void Monitor()
  432. {
  433. }
  434. public virtual void Reset()
  435. {
  436. }
  437. public virtual void Terminate()
  438. {
  439. }
  440. }
  441. }