LoaderUnloadSideRoutine.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. using Aitex.Core.RT.Device;
  2. using Aitex.Core.RT.Routine;
  3. using MECF.Framework.Common.Equipment;
  4. using MECF.Framework.Common.Routine;
  5. using CyberX8_Core;
  6. using CyberX8_RT.Devices.AXIS;
  7. using CyberX8_RT.Devices.Loader;
  8. using System;
  9. using System.Collections.Generic;
  10. using Aitex.Core.RT.Log;
  11. using MECF.Framework.Common.Utilities;
  12. using MECF.Framework.Common.CommonData.Loader;
  13. using MECF.Framework.Common.CommonData;
  14. using Aitex.Core.RT.DataCenter;
  15. using Aitex.Core.RT.SCCore;
  16. namespace CyberX8_RT.Modules.Loader
  17. {
  18. public class LoaderUnloadSideRoutine : RoutineBase, IRoutine
  19. {
  20. private enum UnloadStep
  21. {
  22. RotationGoToLOADA,
  23. RotationGoToLOADAWait,
  24. SideUnload,
  25. UnloadAllWait,
  26. End
  27. }
  28. #region 常量
  29. private const string SIDE_A = "SideA";
  30. private const string SIDE_B = "SideB";
  31. private const int LOTTRACK_TIME = 1000;
  32. #endregion
  33. #region 内部变量
  34. private JetAxisBase _rotationAxis;
  35. private string _side;
  36. private LoaderUnloadRoutine _sideUnloadRoutine;
  37. private bool _isSideUnloaded = false;
  38. private bool _isSideStop = false;
  39. /// <summary>
  40. /// lotTrack time
  41. /// </summary>
  42. private DateTime _lotTackTime = DateTime.Now;
  43. /// <summary>
  44. /// Loader Common
  45. /// </summary>
  46. private LoaderCommonDevice _loaderCommon;
  47. /// <summary>
  48. /// LoaderSide
  49. /// </summary>
  50. private LoaderSideDevice _loaderSide;
  51. /// <summary>
  52. /// Loader LotTrackData
  53. /// </summary>
  54. private List<LoaderLotTrackData> _datas = new List<LoaderLotTrackData>();
  55. /// <summary>
  56. /// WaferSize
  57. /// </summary>
  58. private int _waferSize;
  59. #endregion
  60. #region 属性
  61. /// <summary>
  62. /// UnLoad LotTrackData
  63. /// </summary>
  64. public List<LoaderLotTrackData> UnloadLotTrackDatas { get { return _datas; } }
  65. #endregion
  66. /// <summary>
  67. /// 构造函数
  68. /// </summary>
  69. /// <param name="module"></param>
  70. public LoaderUnloadSideRoutine(string module) : base(module)
  71. {
  72. }
  73. /// <summary>
  74. /// 中止
  75. /// </summary>
  76. public void Abort()
  77. {
  78. }
  79. /// <summary>
  80. /// 监控
  81. /// </summary>
  82. /// <returns></returns>
  83. public RState Monitor()
  84. {
  85. LottrackRecord();
  86. Runner.Run(UnloadStep.RotationGoToLOADA,RotationGotoLOAD,_delay_1ms)
  87. .WaitWithStopCondition(UnloadStep.RotationGoToLOADAWait,CheckRotationPositionStatus,CheckRotationPositionRunStop)
  88. .Run(UnloadStep.SideUnload, () => StartUnloadRoutine(_sideUnloadRoutine,_isSideUnloaded), _delay_1ms)
  89. .WaitWithStopCondition(UnloadStep.UnloadAllWait, CheckUnloadAllRoutineEndStatus,CheckUnloadAllRoutineStopStatus)
  90. .End(UnloadStep.End, NullFun, _delay_1ms);
  91. return Runner.Status;
  92. }
  93. /// <summary>
  94. /// Rotation Goto LOADA
  95. /// </summary>
  96. /// <returns></returns>
  97. private bool RotationGotoLOAD()
  98. {
  99. string side = _side == SIDE_A ? "A" : "B";
  100. bool result = _rotationAxis.PositionStation($"LOAD{side}{_waferSize}", false);
  101. if (!result)
  102. {
  103. NotifyError(eEvent.ERR_LOADER, "rotation start goto LOAD failed", 0);
  104. }
  105. return result;
  106. }
  107. /// <summary>
  108. /// 检验Rotation移动状态
  109. /// </summary>
  110. /// <returns></returns>
  111. private bool CheckRotationPositionStatus()
  112. {
  113. return _rotationAxis.Status == RState.End;
  114. }
  115. /// <summary>
  116. /// 检验Rotation是否还在运动
  117. /// </summary>
  118. /// <returns></returns>
  119. private bool CheckRotationPositionRunStop()
  120. {
  121. bool result = _rotationAxis.Status == RState.Failed || _rotationAxis.Status == RState.Timeout ;
  122. if (result)
  123. {
  124. NotifyError(eEvent.ERR_LOADER, "rotation goto position failed",0);
  125. }
  126. return result;
  127. }
  128. /// <summary>
  129. /// 启动Unload routine
  130. /// </summary>
  131. /// <param name="unloadRoutine"></param>
  132. /// <returns></returns>
  133. private bool StartUnloadRoutine(LoaderUnloadRoutine unloadRoutine,bool isSideUnloaded)
  134. {
  135. if (isSideUnloaded)
  136. {
  137. return true;
  138. }
  139. bool result= unloadRoutine.Start()==RState.Running;
  140. if(!result)
  141. {
  142. NotifyError(eEvent.ERR_LOADER, unloadRoutine.ErrorMsg, 0);
  143. }
  144. return result;
  145. }
  146. /// <summary>
  147. /// 检验UnloadAll完成状态
  148. /// </summary>
  149. /// <returns></returns>
  150. private bool CheckUnloadAllRoutineEndStatus()
  151. {
  152. bool sideResult = true;
  153. if (!_isSideUnloaded)
  154. {
  155. sideResult = CheckUnloadRoutineEndStatus(_sideUnloadRoutine);
  156. }
  157. return sideResult;
  158. }
  159. /// <summary>
  160. /// 检查UnloadAll停止状态
  161. /// </summary>
  162. /// <returns></returns>
  163. private bool CheckUnloadAllRoutineStopStatus()
  164. {
  165. bool sideComplete = false;
  166. if (!_isSideUnloaded&&!_isSideStop)
  167. {
  168. RState ret = _sideUnloadRoutine.Monitor();
  169. _isSideStop = ret == RState.Failed || ret == RState.Timeout;
  170. sideComplete = (ret != RState.Running);
  171. if (_isSideStop)
  172. {
  173. NotifyError(eEvent.ERR_LOADER, $"unload A failed\r\n{_sideUnloadRoutine.ErrorMsg}", 1);
  174. }
  175. }
  176. return _isSideStop;
  177. }
  178. /// <summary>
  179. /// 检验routine完成状态
  180. /// </summary>
  181. /// <param name="unloadRoutine"></param>
  182. /// <returns></returns>
  183. private bool CheckUnloadRoutineEndStatus(LoaderUnloadRoutine unloadRoutine)
  184. {
  185. return unloadRoutine.Monitor() == RState.End;
  186. }
  187. /// <summary>
  188. /// 检验Routine结束状态
  189. /// </summary>
  190. /// <param name="unloadRoutine"></param>
  191. /// <returns></returns>
  192. private bool CheckUnloadRoutineStopStatus(LoaderUnloadRoutine unloadRoutine,string side)
  193. {
  194. RState state=unloadRoutine.Monitor();
  195. if (state == RState.Failed || state == RState.Timeout)
  196. {
  197. NotifyError(eEvent.ERR_LOADER, $"{side} Unload failed", 0);
  198. return true;
  199. }
  200. return false;
  201. }
  202. /// <summary>
  203. /// 启动
  204. /// </summary>
  205. /// <param name="objs"></param>
  206. /// <returns></returns>
  207. public RState Start(params object[] objs)
  208. {
  209. _side = objs[0].ToString();
  210. _waferSize = SC.GetValue<int>($"Loader1.{_side}WaferSize");
  211. InitializeParameters();
  212. _loaderCommon = DEVICE.GetDevice<LoaderCommonDevice>($"{Module}.Common");
  213. _loaderSide = DEVICE.GetDevice<LoaderSideDevice>($"{Module}.{_side}");
  214. return Runner.Start(Module, "Start Unload side");
  215. }
  216. /// <summary>
  217. /// 初始化参数
  218. /// </summary>
  219. private void InitializeParameters()
  220. {
  221. _rotationAxis = DEVICE.GetDevice<JetAxisBase>($"{Module}.Rotation");
  222. _sideUnloadRoutine = new LoaderUnloadRoutine(ModuleName.Loader1.ToString(), _side);
  223. _isSideUnloaded = false;
  224. _isSideStop = false;
  225. }
  226. /// <summary>
  227. /// 重试
  228. /// </summary>
  229. /// <param name="step"></param>
  230. public RState Retry(int step)
  231. {
  232. InitializeParameters();
  233. List<Enum> preStepIds = new List<Enum>();
  234. return Runner.Retry(UnloadStep.RotationGoToLOADA,preStepIds,Module,"UnloadAll Retry");
  235. }
  236. /// <summary>
  237. /// 检验前面Unload完成状态
  238. /// </summary>
  239. /// <returns></returns>
  240. public bool CheckCompleteCondition(int index)
  241. {
  242. string side = _side == SIDE_A ? "A" : "B";
  243. if (!CheckSideUnloadCondition(side, index,true))
  244. {
  245. return false;
  246. }
  247. return true;
  248. }
  249. /// <summary>
  250. /// 检验Side Unload情况
  251. /// </summary>
  252. /// <param name="side"></param>
  253. /// <param name="index"></param>
  254. /// <returns></returns>
  255. private bool CheckSideUnloadCondition(string side, int index,bool showError)
  256. {
  257. JetAxisBase shuttleAxis = DEVICE.GetDevice<JetAxisBase>($"{ModuleName.Loader1}.Shuttle{side}");
  258. double shuttlePosition = shuttleAxis.MotionData.MotorPosition;
  259. if (!shuttleAxis.CheckPositionInStationIgnoreWaferSize(shuttlePosition, "OUT"))
  260. {
  261. if (showError)
  262. {
  263. NotifyError(eEvent.ERR_LOADER, $"shuttle{side} {shuttlePosition} is not in Out", index);
  264. }
  265. return false;
  266. }
  267. JetAxisBase tiltAxis = DEVICE.GetDevice<JetAxisBase>($"{ModuleName.Loader1}.Tilt{side}");
  268. double tiltPosition = tiltAxis.MotionData.MotorPosition;
  269. if (!tiltAxis.CheckPositionInStationIgnoreWaferSize(tiltPosition, "HORI"))
  270. {
  271. if (showError)
  272. {
  273. NotifyError(eEvent.ERR_LOADER, $"tilt{side} {tiltPosition} is not in HORI", index);
  274. return false;
  275. }
  276. }
  277. JetAxisBase crsAxis = DEVICE.GetDevice<JetAxisBase>($"{ModuleName.Loader1}.LS{side}");
  278. double crsPosition = crsAxis.MotionData.MotorPosition;
  279. if (!crsAxis.CheckPositionInStationIgnoreWaferSize(crsPosition, "Unlock"))
  280. {
  281. if (showError)
  282. {
  283. NotifyError(eEvent.ERR_LOADER, $"LS{side} {crsPosition} is not in Unlock", index);
  284. }
  285. return false;
  286. }
  287. return true;
  288. }
  289. /// <summary>
  290. /// 记录Lottrack
  291. /// </summary>
  292. private void LottrackRecord()
  293. {
  294. //记录Lottrack
  295. if (DateTime.Now.Subtract(_lotTackTime).TotalMilliseconds >= LOTTRACK_TIME)
  296. {
  297. AddLotTrackData();
  298. _lotTackTime = DateTime.Now;
  299. }
  300. }
  301. /// <summary>
  302. /// 获取Lot Track数据
  303. /// </summary>
  304. /// <returns></returns>
  305. private void AddLotTrackData()
  306. {
  307. LoaderLotTrackData data = new LoaderLotTrackData();
  308. data.TimeStamp = DateTime.Now;
  309. data.LoaderABernoulliBladderEnable = _loaderSide.SideData.BernoulliBladder;
  310. data.LoaderABernoulliExtended = _loaderSide.SideData.BernoulliExtended;
  311. data.LoaderABernoulliBladderPressure = _loaderSide.SideData.BernoulliBladderPressure;
  312. data.LoaderABernoulliN2Pressure = _loaderSide.SideData.BernoulliPressure;
  313. data.LoaderACRSVacuum = _loaderSide.SideData.CRSVacuum;
  314. data.LoaderACRSVacuumAnlg = _loaderSide.SideData.CRSVacuumValue;
  315. data.LoaderAWHPressure = _loaderSide.SideData.WHBladderPressure;
  316. data.LoaderATranslatePressure = _loaderSide.SideData.TransPressure;
  317. data.LoaderWHClamped = _loaderCommon.CommonData.WaferHolderClamp;
  318. _datas.Add(data);
  319. }
  320. }
  321. }