SchedulerRinse.cs 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  1. using Aitex.Common.Util;
  2. using Aitex.Core.RT.Fsm;
  3. using Aitex.Core.RT.Log;
  4. using Aitex.Core.Util;
  5. using CyberX8_Core;
  6. using CyberX8_RT.Dispatch;
  7. using CyberX8_RT.Modules;
  8. using CyberX8_RT.Modules.Dryer;
  9. using CyberX8_RT.Modules.Loader;
  10. using CyberX8_RT.Modules.Prewet;
  11. using CyberX8_RT.Modules.Rinse;
  12. using CyberX8_RT.Schedulers.Transporter;
  13. using MECF.Framework.Common.CommonData;
  14. using MECF.Framework.Common.Equipment;
  15. using MECF.Framework.Common.RecipeCenter;
  16. using MECF.Framework.Common.WaferHolder;
  17. using System;
  18. using System.Collections.Generic;
  19. using System.Linq;
  20. using System.Text;
  21. using System.Threading.Tasks;
  22. namespace CyberX8_RT.Schedulers.Rinse
  23. {
  24. public class SchedulerRinse : SchedulerModule
  25. {
  26. #region 内部变量
  27. private RinseEntity _rinseEntity;
  28. private bool _isStartRunRecipe = false;
  29. #endregion
  30. #region 属性
  31. /// <summary>
  32. /// 是否空闲
  33. /// </summary>
  34. public override bool IsIdle
  35. {
  36. get { return _state == RState.End; }
  37. }
  38. /// <summary>
  39. /// 是否错误
  40. /// </summary>
  41. public override bool IsError
  42. {
  43. get { return _state == RState.Failed || _state == RState.Timeout; }
  44. }
  45. #endregion
  46. /// <summary>
  47. /// 构造函数
  48. /// </summary>
  49. /// <param name="module"></param>
  50. public SchedulerRinse(ModuleName moduleName) : base(moduleName.ToString())
  51. {
  52. _rinseEntity = Singleton<RouteManager>.Instance.GetModule<RinseEntity>(moduleName.ToString());
  53. }
  54. /// <summary>
  55. /// 执行
  56. /// </summary>
  57. /// <param name="parameter"></param>
  58. /// <returns></returns>
  59. public override bool RunProcess(object recipe, object parameter, List<SchedulerSyncModuleMessage> syncModuleMessages)
  60. {
  61. if (!(recipe is QdrRecipe))
  62. {
  63. _state = RState.Failed;
  64. LOG.WriteLog(eEvent.ERR_RINSE, Module.ToString(), "recipe is invalid");
  65. return false;
  66. }
  67. _isStartRunRecipe = false;
  68. QdrRecipe qdrRecipe = (QdrRecipe)recipe;
  69. bool result = _rinseEntity.CheckToPostMessage<RinseState, RinseMsg>(eEvent.ERR_RINSE, Module.ToString(), (int)RinseMsg.RunRecipe, qdrRecipe,1);
  70. if (result)
  71. {
  72. _state = RState.Running;
  73. }
  74. return result;
  75. }
  76. /// <summary>
  77. /// 监控执行
  78. /// </summary>
  79. /// <returns></returns>
  80. public override bool MonitorProcess(SchedulerSequence schedulerSequence, bool hasMatchWafer)
  81. {
  82. if (!_isStartRunRecipe)
  83. {
  84. _isStartRunRecipe = _rinseEntity.State == (int)RinseState.RunReciping;
  85. }
  86. if (_rinseEntity.IsError)
  87. {
  88. _state = RState.Failed;
  89. _isStartRunRecipe = false;
  90. }
  91. if (_isStartRunRecipe && _rinseEntity.IsIdle)
  92. {
  93. _state = RState.End;
  94. _isStartRunRecipe = false;
  95. }
  96. if(_rinseEntity.State==(int)RinseState.RunRecipeComplete||_rinseEntity.State==(int)RinseState.KeepWeting)
  97. {
  98. if (schedulerSequence.NextModuleType != ModuleType.Metal)
  99. {
  100. NotifyRinseComplete();
  101. }
  102. else
  103. {
  104. if (SchedulerManager.Instance.IsQdrCheckConflict)
  105. {
  106. WaferHolderInfo waferHolderInfo = WaferHolderManager.Instance.GetWaferHolder(Module.ToString());
  107. if (waferHolderInfo == null)
  108. {
  109. return true;
  110. }
  111. SchedulerSequence nextTransporterSequence = WaferHolderTaskManager.Instance.GetNextTransporterlSequence(schedulerSequence.SequenceIndex, waferHolderInfo.Id);
  112. if (nextTransporterSequence == null)
  113. {
  114. return true;
  115. }
  116. if(!(nextTransporterSequence.Parameters is TransporterAction))
  117. {
  118. return true;
  119. }
  120. TransporterAction transporterAction=nextTransporterSequence.Parameters as TransporterAction;
  121. WaferHolderMoveItem waferHolderMoveItem = transporterAction.Parameter as WaferHolderMoveItem;
  122. if (waferHolderMoveItem.DestModule == ModuleName.Unknown)
  123. {
  124. WaferHolderTask waferHolderTask = WaferHolderTaskManager.Instance.GetWaferHolderTaskByWaferHolderId(waferHolderInfo.Id);
  125. if (waferHolderTask == null)
  126. {
  127. return true;
  128. }
  129. bool result = SchedulerModuleTimeManager.Instance.ConfirmAllMetalQdrAndDryer(waferHolderTask.SchedulerSequences, waferHolderInfo.Id, waferHolderTask.GetCurrentSchedulerIndex(), true, DateTime.Now);
  130. if (result)
  131. {
  132. NotifyRinseComplete();
  133. }
  134. return true;
  135. }
  136. if (WaferHolderManager.Instance.HasWaferHolder(waferHolderMoveItem.DestModule.ToString()))
  137. {
  138. NotifyRinseKeepWet();
  139. }
  140. else
  141. {
  142. NotifyRinseComplete();
  143. }
  144. }
  145. else
  146. {
  147. bool exsitEnableCell = false;
  148. DepRecipe depRecipe = schedulerSequence.NextRecipe as DepRecipe;
  149. ModuleName moduleName = SchedulerSequenceManager.Instance.CalculateAvaibleMetalCellByChemistry(depRecipe.Chemistry,
  150. Module.ToString(), schedulerSequence.SequenceType, schedulerSequence.WaferSize, ref exsitEnableCell);
  151. if (moduleName != ModuleName.Unknown)
  152. {
  153. IModuleEntity moduleEntity = Singleton<RouteManager>.Instance.GetModule<IModuleEntity>(moduleName.ToString());
  154. if (SchedulerSequenceManager.Instance.CheckMetalCellRecipeTimeAvaible(moduleEntity, depRecipe))
  155. {
  156. NotifyRinseComplete();
  157. }
  158. else
  159. {
  160. NotifyRinseKeepWet();
  161. }
  162. }
  163. else
  164. {
  165. if (exsitEnableCell)
  166. {
  167. NotifyRinseKeepWet();
  168. }
  169. else
  170. {
  171. NotifyRinseComplete();
  172. }
  173. }
  174. }
  175. }
  176. }
  177. return true;
  178. }
  179. /// <summary>
  180. /// 通知Rinse完成
  181. /// </summary>
  182. private void NotifyRinseComplete()
  183. {
  184. _rinseEntity.CheckToPostMessage<RinseState, RinseMsg>(eEvent.INFO_RINSE, Module.ToString(), (int)RinseMsg.RecipeComplete);
  185. }
  186. /// <summary>
  187. /// 通知Rinse准备Keep Wet
  188. /// </summary>
  189. private void NotifyRinseKeepWet()
  190. {
  191. if (_rinseEntity.State == (int)RinseState.KeepWeting)
  192. {
  193. return;
  194. }
  195. _rinseEntity.CheckToPostMessage<RinseState, RinseMsg>(eEvent.WARN_PREWET, Module.ToString(),
  196. (int)RinseMsg.Keepwet);
  197. }
  198. /// <summary>
  199. /// 检验前置条件
  200. /// </summary>
  201. /// <param name="sequenceIndex"></param>
  202. /// <param name="parameter"></param>
  203. /// <returns></returns>
  204. public override bool CheckPrecondition(List<SchedulerSequence> schedulerSequences, int sequenceIndex, object parameter, string materialId,ref string reason)
  205. {
  206. if (_state == RState.Running)
  207. {
  208. reason = "scheduler module is already running";
  209. return false;
  210. }
  211. if (_rinseEntity.IsBusy)
  212. {
  213. reason = $"{_rinseEntity.Module} is busy";
  214. return false;
  215. }
  216. if(_rinseEntity.WaferHolderInfo==null)
  217. {
  218. reason = $"{_rinseEntity.Module} has no wafer shuttle";
  219. return false;
  220. }
  221. if (_rinseEntity.WaferHolderInfo.Id != materialId)
  222. {
  223. reason = $"{_rinseEntity.Module} wafer shuttle {_rinseEntity.WaferHolderInfo.Id} is not matched with {materialId}";
  224. return false;
  225. }
  226. return true;
  227. }
  228. }
  229. }