TransporterPickUpFromRoutine.cs 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705
  1. using Aitex.Core.RT.Device;
  2. using Aitex.Core.RT.Log;
  3. using Aitex.Core.RT.Routine;
  4. using Aitex.Core.RT.SCCore;
  5. using Aitex.Core.Util;
  6. using MECF.Framework.Common.Beckhoff.Station;
  7. using MECF.Framework.Common.Equipment;
  8. using MECF.Framework.Common.Layout;
  9. using MECF.Framework.Common.Routine;
  10. using MECF.Framework.Common.Utilities;
  11. using MECF.Framework.Common.WaferHolder;
  12. using CyberX8_Core;
  13. using CyberX8_RT.Devices.AXIS;
  14. using CyberX8_RT.Devices.AXIS.CANOpen;
  15. using CyberX8_RT.Devices.Facilities;
  16. using CyberX8_RT.Devices.Loader;
  17. using CyberX8_RT.Devices.TransPorter;
  18. using CyberX8_RT.Modules.Loader;
  19. using System;
  20. using System.Collections.Generic;
  21. using System.Linq;
  22. using System.Text;
  23. using System.Threading.Tasks;
  24. using CyberX8_RT.Devices.Rinse;
  25. using CyberX8_RT.Devices.Metal;
  26. using Aitex.Core.Common;
  27. namespace CyberX8_RT.Modules.Transporter
  28. {
  29. public class TransporterPickUpFromRoutine : RoutineBase, IRoutine
  30. {
  31. private enum PickUpStep
  32. {
  33. CheckPreCondition,
  34. TargetCellUnclamp,
  35. TargetCellUnclampWait,
  36. DropBlockLockoff,
  37. ElevatorGotoUp,
  38. ElevatorGotoUpWait,
  39. SafeMoveTo,
  40. CheckMoveToStatus,
  41. GantryPosition,
  42. GantryPoisitionWait,
  43. ElevatorPositionToCell,
  44. ElevatorPositionToCellWait,
  45. Delay,
  46. DropBlockLockOn,
  47. CalculateLiftupSpeed,
  48. PickupDelay,
  49. ElevatorPositionToUp,
  50. ElevatorPositionToUpWait,
  51. UpdateWaferShuttle,
  52. CheckWSPresent,
  53. //ReadBarcodeConfirm,
  54. End
  55. }
  56. #region 内部变量
  57. private string _cellName;
  58. private JetAxisBase _gantryAxis;
  59. private JetAxisBase _elevatorAxis;
  60. private JetAxisBase _loaderRotationAxis;
  61. private SystemFacilities _facilities;
  62. private TransporterConflictRoutine _conflictRoutine;
  63. private LoaderPreTransferUnclampRoutine _preTransferUnclampRoutine;
  64. private TransporterCommon _transporterCommon;
  65. private LoaderCommonDevice _loaderCommonDevice;
  66. ProcessLayoutCellItem _cellItem;
  67. private int _pickupTime;
  68. private int _pickupDelayTime;
  69. private int _velocity;
  70. private int _acceleration;
  71. private bool _bypassWaferHolderPresent;
  72. private int _pickMaxRetries = 2;
  73. #endregion
  74. /// <summary>
  75. /// 构造函数
  76. /// </summary>
  77. /// <param name="module"></param>
  78. public TransporterPickUpFromRoutine(string module) : base(module)
  79. {
  80. }
  81. /// <summary>
  82. /// 中止
  83. /// </summary>
  84. public void Abort()
  85. {
  86. Runner.Stop("Manual Abort");
  87. }
  88. /// <summary>
  89. /// 监控
  90. /// </summary>
  91. /// <returns></returns>
  92. public RState Monitor()
  93. {
  94. Runner.Run(PickUpStep.CheckPreCondition, CheckStartPreConfition, _delay_1ms)
  95. .Run(PickUpStep.TargetCellUnclamp, TargetCellUnclamp, _delay_1ms)
  96. .WaitWithStopCondition(PickUpStep.TargetCellUnclampWait, TargetCellUnclampEndStatus, TargetCellUnclampStopStatus)
  97. //1. 确认DropBlockLock是否off
  98. .Run(PickUpStep.DropBlockLockoff, DropBlockLockoff, _delay_1ms)
  99. //2. Elevator go to up
  100. .Run(PickUpStep.ElevatorGotoUp, ElevatorGotoUP, _delay_1ms)
  101. .WaitWithStopCondition(PickUpStep.ElevatorGotoUpWait, CheckElevatorPositionEndStatus, CheckElevatorPositionEndStatus)
  102. //3. other Transporter Safe Move
  103. .Run(PickUpStep.SafeMoveTo, SafeMoveTo, _delay_1ms)
  104. .WaitWithStopCondition(PickUpStep.CheckMoveToStatus, () => CommonFunction.CheckRoutineEndState(_conflictRoutine),
  105. CheckSafeMoveToStopStatus)
  106. //4. Gantry 移动到目标位置
  107. .Run(PickUpStep.GantryPosition, GantryPositionToCell, _delay_1ms)
  108. .WaitWithStopCondition(PickUpStep.GantryPoisitionWait, CheckGantryPositionStatus, CheckGantryPositionRunStop)
  109. //5. Elevator 移动至对应cell位
  110. .Run(PickUpStep.ElevatorPositionToCell, ElevatorPositionToCell, _delay_1ms)
  111. .WaitWithStopCondition(PickUpStep.ElevatorPositionToCellWait, CheckElevatorPositionEndStatus, CheckElevatorPositionStopStatus)
  112. .Delay(PickUpStep.Delay,500)
  113. //6. Drop Block Lock on,抓取WS
  114. .Run(PickUpStep.DropBlockLockOn, DropBlockLockon, _delay_1ms)
  115. //7. 确认Elevator lift speed
  116. .Run(PickUpStep.CalculateLiftupSpeed,CalculateLiftupSpeed,_delay_1ms)
  117. //8. Pickup delay
  118. .Delay(PickUpStep.PickupDelay,_pickupDelayTime * 1000)
  119. //9. Elevator goto Up
  120. .Run(PickUpStep.ElevatorPositionToUp, ElevatorGotoUP, 100)
  121. .WaitWithStopCondition(PickUpStep.ElevatorPositionToUpWait, CheckElevatorPositionEndStatus, CheckElevatorPositionStopStatus)
  122. //10. 确认Transporter的Wafer shuttle hold present信号为on
  123. .Wait(PickUpStep.CheckWSPresent, CheckWSPresent, _delay_2s)
  124. //11. Material Tracking Update
  125. .Run(PickUpStep.UpdateWaferShuttle, WaferShuttleTransfer,_delay_1ms)
  126. //12. Read Barcode 确认与Material Tracking的转移是否一致
  127. //.Run(PickUpStep.ReadBarcodeConfirm,ReadBarcode,_delay_1ms)
  128. .End(PickUpStep.End,NullFun,100);
  129. return Runner.Status;
  130. }
  131. /// <summary>
  132. /// 目标cell unclamp
  133. /// </summary>
  134. /// <returns></returns>
  135. private bool TargetCellUnclamp()
  136. {
  137. if (_cellName == "Loader")
  138. {
  139. bool result = _preTransferUnclampRoutine.Start() == RState.Running;
  140. if (!result)
  141. {
  142. NotifyError(eEvent.ERR_TRANSPORTER, "Loader Unclamp failed", 0);
  143. }
  144. return result;
  145. }
  146. else
  147. {
  148. _cellItem = ProcessLayoutManager.Instance.GetProcessLayoutCellItemByModuleName(_cellName);
  149. if (_cellItem != null)
  150. {
  151. bool result = false;
  152. if (Enum.TryParse(_cellItem.ModuleName, out ModuleName cellModuleName))
  153. {
  154. if (ModuleHelper.IsRinse(cellModuleName))
  155. {
  156. RinseDevice rinseDevice = DEVICE.GetDevice<RinseDevice>(_cellItem.ModuleName);
  157. if (rinseDevice != null)
  158. {
  159. result = rinseDevice.WaferHolderClampValveOff();
  160. if (!result)
  161. {
  162. NotifyError(eEvent.ERR_TRANSPORTER, $"{_cellItem.ModuleName} Clamp off failed", 0);
  163. }
  164. return result;
  165. }
  166. }
  167. else if (ModuleHelper.IsMetal(cellModuleName))
  168. {
  169. MetalCellDevice metalCellDevice = DEVICE.GetDevice<MetalCellDevice>(_cellItem.ModuleName);
  170. if (metalCellDevice != null)
  171. {
  172. result = metalCellDevice.WaferHolderClampOff();
  173. if (!result)
  174. {
  175. NotifyError(eEvent.ERR_TRANSPORTER, $"{_cellItem.ModuleName} Clamp off failed", 0);
  176. }
  177. return result;
  178. }
  179. }
  180. }
  181. }
  182. }
  183. return true;
  184. }
  185. /// <summary>
  186. /// 目标Cell Unclamp状态
  187. /// </summary>
  188. /// <returns></returns>
  189. private bool TargetCellUnclampEndStatus()
  190. {
  191. if (_cellName == "Loader")
  192. {
  193. return CommonFunction.CheckRoutineEndState(_preTransferUnclampRoutine);
  194. }
  195. else
  196. {
  197. _cellItem = ProcessLayoutManager.Instance.GetProcessLayoutCellItemByModuleName(_cellName);
  198. if (_cellItem != null)
  199. {
  200. if (Enum.TryParse(_cellItem.ModuleName, out ModuleName cellModuleName))
  201. {
  202. if (ModuleHelper.IsRinse(cellModuleName))
  203. {
  204. RinseDevice rinseDevice = DEVICE.GetDevice<RinseDevice>(_cellItem.ModuleName);
  205. if (rinseDevice != null)
  206. {
  207. return !rinseDevice.RinseData.WaferHolderClamp;
  208. }
  209. }
  210. else if (ModuleHelper.IsMetal(cellModuleName))
  211. {
  212. MetalCellDevice metalCellDevice = DEVICE.GetDevice<MetalCellDevice>(_cellItem.ModuleName);
  213. if (metalCellDevice != null)
  214. {
  215. return metalCellDevice.ClampOff;
  216. }
  217. }
  218. }
  219. }
  220. }
  221. return true;
  222. }
  223. /// <summary>
  224. /// 目标Cell Unclamp运行中止状态
  225. /// </summary>
  226. /// <returns></returns>
  227. private bool TargetCellUnclampStopStatus()
  228. {
  229. if (_cellName == "Loader")
  230. {
  231. bool result = CommonFunction.CheckRoutineStopState(_preTransferUnclampRoutine);
  232. if (result)
  233. {
  234. NotifyError(eEvent.ERR_TRANSPORTER, "Loader Unclamp failed", 0);
  235. }
  236. return result;
  237. }
  238. return false;
  239. }
  240. /// <summary>
  241. /// 关闭DropBlockLock
  242. /// </summary>
  243. /// <returns></returns>
  244. private bool DropBlockLockoff()
  245. {
  246. if (_transporterCommon.TransporterData.Lock)
  247. {
  248. return _transporterCommon.UnlockOperation("", null);
  249. }
  250. return true;
  251. }
  252. /// <summary>
  253. /// 打开DropBlockLock
  254. /// </summary>
  255. /// <returns></returns>
  256. private bool DropBlockLockon()
  257. {
  258. if (!_transporterCommon.TransporterData.Lock)
  259. {
  260. return _transporterCommon.LockOperation("", null);
  261. }
  262. return true;
  263. }
  264. /// <summary>
  265. /// 安全避障移动
  266. /// </summary>
  267. /// <returns></returns>
  268. private bool SafeMoveTo()
  269. {
  270. _cellItem = ProcessLayoutManager.Instance.GetProcessLayoutCellItemByModuleName(_cellName);
  271. string stationName = _cellName;
  272. if (_cellItem != null)
  273. {
  274. if (_cellName.ToLower() != "loader" && _cellName.ToLower() != "park")
  275. {
  276. stationName = $"Cell{_cellItem.CellId}";
  277. }
  278. }
  279. else
  280. {
  281. NotifyError(eEvent.ERR_TRANSPORTER, $"{_cellName} not in layout", 0);
  282. return false;
  283. }
  284. var result = _gantryAxis.GetPositionByStation(stationName);
  285. if(result.success)
  286. {
  287. bool isPositive = false;
  288. if(_gantryAxis.MotionData.MotorPosition<result.position)
  289. {
  290. isPositive = true;
  291. }
  292. return _conflictRoutine.Start(result.position, isPositive) == RState.Running;
  293. }
  294. else
  295. {
  296. NotifyError(eEvent.ERR_TRANSPORTER, $"{_cellName} not in station list",0);
  297. return false;
  298. }
  299. }
  300. /// <summary>
  301. /// 检验安全避让异常结束状态
  302. /// </summary>
  303. /// <returns></returns>
  304. private bool CheckSafeMoveToStopStatus()
  305. {
  306. bool result = CommonFunction.CheckRoutineStopState(_conflictRoutine);
  307. if (result)
  308. {
  309. NotifyError(eEvent.ERR_TRANSPORTER, "Safe Move failed", 0);
  310. }
  311. return result;
  312. }
  313. /// <summary>
  314. /// Gantry Position To cell
  315. /// </summary>
  316. /// <returns></returns>
  317. private bool GantryPositionToCell()
  318. {
  319. _cellItem= ProcessLayoutManager.Instance.GetProcessLayoutCellItemByModuleName(_cellName);
  320. bool result = false;
  321. if (_cellItem != null)
  322. {
  323. if (_cellName.ToLower() != "loader"&&_cellName.ToLower()!="park")
  324. {
  325. result= _gantryAxis.PositionStation($"Cell{_cellItem.CellId}", false);
  326. }
  327. else
  328. {
  329. result= _gantryAxis.PositionStation(_cellName,false);
  330. }
  331. if (!result)
  332. {
  333. NotifyError(eEvent.ERR_TRANSPORTER, "gantry axis motion failed", 0);
  334. }
  335. return result;
  336. }
  337. else
  338. {
  339. NotifyError(eEvent.ERR_TRANSPORTER, $"{_cellName} not in layout", 0);
  340. return false;
  341. }
  342. }
  343. /// <summary>
  344. /// Elevator Position to cell
  345. /// </summary>
  346. /// <returns></returns>
  347. private bool ElevatorPositionToCell()
  348. {
  349. bool result = false;
  350. if( _cellItem != null )
  351. {
  352. if (_cellName.ToLower()!="loader")
  353. {
  354. result= _elevatorAxis.PositionStation($"Cell{_cellItem.CellId}", false);
  355. }
  356. else
  357. {
  358. result= _elevatorAxis.PositionStation(_cellName, false);
  359. }
  360. if (!result)
  361. {
  362. NotifyError(eEvent.ERR_TRANSPORTER, "elevator axis motion failed", 0);
  363. }
  364. return result;
  365. }
  366. else
  367. {
  368. NotifyError(eEvent.ERR_TRANSPORTER, $"{_cellName} not in layout",0);
  369. return false;
  370. }
  371. }
  372. /// <summary>
  373. /// 检验Vertical移动状态
  374. /// </summary>
  375. /// <returns></returns>
  376. private bool CheckElevatorPositionEndStatus()
  377. {
  378. return _elevatorAxis.Status == RState.End;
  379. }
  380. /// <summary>
  381. /// 检验Vertical是否还在运动
  382. /// </summary>
  383. /// <returns></returns>
  384. private bool CheckElevatorPositionStopStatus()
  385. {
  386. bool result = _elevatorAxis.Status == RState.Failed || _elevatorAxis.Status == RState.Timeout ;
  387. if (result)
  388. {
  389. NotifyError(eEvent.ERR_TRANSPORTER, "elevator axis motion failed",0);
  390. }
  391. return result;
  392. }
  393. /// <summary>
  394. /// 检验Gantry移动状态
  395. /// </summary>
  396. /// <returns></returns>
  397. private bool CheckGantryPositionStatus()
  398. {
  399. return _gantryAxis.Status == RState.End;
  400. }
  401. /// <summary>
  402. /// 检验Gantry是否还在运动
  403. /// </summary>
  404. /// <returns></returns>
  405. private bool CheckGantryPositionRunStop()
  406. {
  407. bool result= _gantryAxis.Status == RState.Failed||_gantryAxis.Status==RState.Timeout;
  408. if(result)
  409. {
  410. NotifyError(eEvent.ERR_TRANSPORTER, "Gantry Motion failed", 0);
  411. }
  412. return result;
  413. }
  414. /// <summary>
  415. /// 计算提升速度
  416. /// </summary>
  417. /// <returns></returns>
  418. private bool CalculateLiftupSpeed()
  419. {
  420. if(_pickupTime==0)
  421. {
  422. return true;
  423. }
  424. BeckhoffStationAxis stationAxis = BeckhoffStationLocationManager.Instance.GetStationAxis(Module, "Elevator", 0);
  425. Station upStation = stationAxis.Stations.Find(O => O.Name.EndsWith("UP"));
  426. Station cellStation = null;
  427. if (_cellName == "Loader")
  428. {
  429. cellStation = stationAxis.Stations.Find(O => O.Name.EndsWith("Loader"));
  430. }
  431. else
  432. {
  433. cellStation = stationAxis.Stations.Find(O => O.Name.EndsWith($"Cell{_cellItem.CellId}"));
  434. }
  435. if(upStation==null)
  436. {
  437. NotifyError(eEvent.ERR_TRANSPORTER, "UP is not in Station List",0);
  438. return false;
  439. }
  440. if(cellStation==null)
  441. {
  442. NotifyError(eEvent.ERR_TRANSPORTER, $"Cell{_cellItem.CellId} is not in Station List",0);
  443. return false;
  444. }
  445. if(!double.TryParse(cellStation.Position,out double cellStationPosition))
  446. {
  447. NotifyError(eEvent.ERR_TRANSPORTER, $"Cell{_cellItem.CellId} Station Position is invalid",0);
  448. return false;
  449. }
  450. if (!double.TryParse(upStation.Position, out double upStationPosition))
  451. {
  452. NotifyError(eEvent.ERR_TRANSPORTER, $"Cell{_cellItem.CellId} Station Position is invalid", 0);
  453. return false;
  454. }
  455. int distance =(int)Math.Round(Math.Abs(cellStationPosition - upStationPosition));
  456. double tmpVelocity = (double) distance/ _pickupTime;
  457. _velocity = _elevatorAxis.CalculateMultiplySpeedRatio(_elevatorAxis.CalculateValueMultiplyScale(tmpVelocity));
  458. double tmpAccelaration = (double)(2 * distance) / Math.Pow(_pickupTime, 2);
  459. _acceleration = _elevatorAxis.CalculateMultiplyAccelerationRatio(_elevatorAxis.CalculateValueMultiplyScale(tmpAccelaration));
  460. LOG.WriteBackgroundLog(eEvent.INFO_TRANSPORTER, Module, "adjust profile speed");
  461. return true;
  462. }
  463. /// <summary>
  464. /// Elevator运动至UP
  465. /// </summary>
  466. /// <returns></returns>
  467. private bool ElevatorGotoUP()
  468. {
  469. bool result= _elevatorAxis.PositionStation("UP", false, _velocity, _acceleration, _acceleration);
  470. if(!result)
  471. {
  472. NotifyError(eEvent.ERR_TRANSPORTER, "elevator goto up failed",0);
  473. }
  474. return result;
  475. }
  476. /// <summary>
  477. /// 更新WaferHolder移动信息
  478. /// </summary>
  479. /// <returns></returns>
  480. private bool WaferShuttleTransfer()
  481. {
  482. bool isMetal = false;
  483. DateTime lastMetalRecipeTime = DateTime.Now;
  484. if(Enum.TryParse(_cellName,out ModuleName moduleName)&&ModuleHelper.IsMetal(moduleName))
  485. {
  486. isMetal = true;
  487. WaferHolderInfo info = WaferHolderManager.Instance.GetWaferHolder(_cellName);
  488. if (info != null)
  489. {
  490. lastMetalRecipeTime = info.LastMetalRecipeCompleteTime;
  491. }
  492. }
  493. bool result= WaferHolderManager.Instance.TransferWaferHolder(Module,_cellName, Module);
  494. if (!result)
  495. {
  496. NotifyError(eEvent.ERR_TRANSPORTER, "tranfer waferHolder message failed", 0);
  497. }
  498. else
  499. {
  500. if (!Singleton<RouteManager>.Instance.IsAutoRunning)
  501. {
  502. return true;
  503. }
  504. WaferHolderInfo waferHolderInfo = WaferHolderManager.Instance.GetWaferHolder(Module);
  505. if (waferHolderInfo == null)
  506. {
  507. return true;
  508. }
  509. string strTime = DateTime.Now.ToString("yyyy/MM/dd HH:mm:ss");
  510. waferHolderInfo.SchedulerModules.Add($"{strTime} {_cellName} => {Module}");
  511. if (isMetal)
  512. {
  513. int waferShuttleSoakMaxTime = SC.GetValue<int>("Metal.WaferShuttleSoakMaxTime");
  514. double soakTime = Math.Round(DateTime.Now.Subtract(lastMetalRecipeTime).TotalMinutes,2);
  515. if (soakTime > waferShuttleSoakMaxTime)
  516. {
  517. LOG.WriteLog(eEvent.WARN_METAL, _cellName, $"time of {waferHolderInfo.Id} soak in metal is {soakTime} minute over {waferShuttleSoakMaxTime}");
  518. }
  519. else
  520. {
  521. LOG.WriteLog(eEvent.INFO_METAL, _cellName, $"time of {waferHolderInfo.Id} soak in metal is {soakTime} minute");
  522. }
  523. }
  524. }
  525. return result;
  526. }
  527. /// <summary>
  528. /// 检验WHPresent与Lock状态
  529. /// </summary>
  530. /// <returns></returns>
  531. private bool CheckWSPresent()
  532. {
  533. bool locked = _transporterCommon.TransporterData.Lock;
  534. bool whPresent= _transporterCommon.TransporterData.WSHoldPresent;
  535. bool result= locked && (whPresent||_bypassWaferHolderPresent);
  536. if (!result)
  537. {
  538. LOG.WriteLog(eEvent.INFO_TRANSPORTER, Module, "check wafer Shuttle present and locked failed");
  539. }
  540. return result;
  541. }
  542. /// <summary>
  543. /// 读取条码
  544. /// </summary>
  545. /// <returns></returns>
  546. private bool ReadBarcode()
  547. {
  548. string str=_transporterCommon.ReaderBarcode();
  549. str = str.Trim();
  550. if(!string.IsNullOrEmpty(str))
  551. {
  552. TransporterEntity transporterEntity = Singleton<RouteManager>.Instance.GetModule<TransporterEntity>(Module.ToString());
  553. if (transporterEntity == null)
  554. {
  555. return true;
  556. }
  557. WaferHolderInfo waferHolderInfo = transporterEntity.WaferHolderInfo;
  558. if (waferHolderInfo == null)
  559. {
  560. return true;
  561. }
  562. if (waferHolderInfo.Id == str)
  563. {
  564. return true;
  565. }
  566. WaferHolderManager.Instance.SwitchWaferHolderId(waferHolderInfo,Module,str);
  567. return true;
  568. }
  569. return true;
  570. }
  571. /// <summary>
  572. /// 启动
  573. /// </summary>
  574. /// <param name="objs"></param>
  575. /// <returns></returns>
  576. public RState Start(params object[] objs)
  577. {
  578. _cellName = objs[0].ToString();
  579. _elevatorAxis = DEVICE.GetDevice<JetAxisBase>($"{Module}.Elevator");;
  580. _gantryAxis = DEVICE.GetDevice<JetAxisBase>($"{Module}.Gantry");
  581. _loaderRotationAxis = DEVICE.GetDevice<JetAxisBase>($"{ModuleName.Loader1}.Rotation");
  582. _conflictRoutine = new TransporterConflictRoutine(Module);
  583. _transporterCommon = DEVICE.GetDevice<TransporterCommon>($"{Module}.Common");
  584. _loaderCommonDevice = DEVICE.GetDevice<LoaderCommonDevice>($"{ModuleName.Loader1}.Common");
  585. _bypassWaferHolderPresent = SC.GetValue<bool>("Transporter.BypassWaferHolderPresent");
  586. _preTransferUnclampRoutine = new LoaderPreTransferUnclampRoutine(ModuleName.Loader1.ToString());
  587. _velocity = 0;
  588. _acceleration = 0;
  589. _pickMaxRetries = SC.GetValue<int>("Transporter.MaxPickTries");
  590. string preConfig = SC.GetConfigPreContent(_cellName);
  591. if (SC.ContainsItem($"{preConfig}.PickTimeSeconds"))
  592. {
  593. _pickupTime = SC.GetValue<int>($"{preConfig}.PickTimeSeconds");
  594. }
  595. if (SC.ContainsItem($"{preConfig}.PickDelaySeconds"))
  596. {
  597. _pickupDelayTime = SC.GetValue<int>($"{preConfig}.PickDelaySeconds");
  598. }
  599. return Runner.Start(Module, $"PickUpFrom {_cellName}");
  600. }
  601. /// <summary>
  602. /// 启动校验条件
  603. /// </summary>
  604. /// <returns></returns>
  605. private bool CheckStartPreConfition()
  606. {
  607. //所有轴上电并Homed
  608. bool result = CheckPreCondition();
  609. if(!result)
  610. {
  611. return false;
  612. }
  613. //Transporter的Wafer shuttle hold present信号为off
  614. if (!_bypassWaferHolderPresent && _transporterCommon.TransporterData.WSHoldPresent)
  615. {
  616. NotifyError(eEvent.ERR_TRANSPORTER, $"{Module} WaferShuttleHoldPresent is on", -1);
  617. return false;
  618. }
  619. //Loader is Home
  620. if (ModuleHelper.IsInstalled(ModuleName.Loader1))
  621. {
  622. //Loader is Home
  623. LoaderEntity loaderEntity = Singleton<RouteManager>.Instance.GetModule<LoaderEntity>(ModuleName.Loader1.ToString());
  624. if (loaderEntity != null && !loaderEntity.IsHomed)
  625. {
  626. NotifyError(eEvent.ERR_TRANSPORTER, $"{ModuleName.Loader1} is not homed", -1);
  627. return false;
  628. }
  629. }
  630. //若目标Cell为Loader, 则Loader需在TRNA或TRANB位置且WaferShuttlePresent信号on
  631. if (_cellName == "Loader")
  632. {
  633. double loaderRotationPosition = _loaderRotationAxis.MotionData.MotorPosition;
  634. if (!_loaderRotationAxis.CheckPositionIsInStation(loaderRotationPosition, "TRNPA") &&
  635. !_loaderRotationAxis.CheckPositionIsInStation(loaderRotationPosition, "TRNPB"))
  636. {
  637. NotifyError(eEvent.ERR_TRANSPORTER, $"{ModuleName.Loader1} rotation axis {loaderRotationPosition} is not int TRNPA or TRNPB station",-1);
  638. return false;
  639. }
  640. if(!_loaderCommonDevice.CommonData.WaferHolderPresent)
  641. {
  642. NotifyError(eEvent.ERR_TRANSPORTER, $"{ModuleName.Loader1} WaferShuttlePresent is off", -1);
  643. return false;
  644. }
  645. }
  646. //Transporter没有WS信息
  647. if (WaferHolderManager.Instance.HasWaferHolder(Module))
  648. {
  649. NotifyError(eEvent.ERR_TRANSPORTER, $"{Module} already has wafer Shuttle", -1);
  650. return false;
  651. }
  652. //目标Cell应有WS信息
  653. if (!WaferHolderManager.Instance.HasWaferHolder(_cellName))
  654. {
  655. NotifyError(eEvent.ERR_TRANSPORTER, $"{_cellName} does not has wafer shuttle", -1);
  656. return false;
  657. }
  658. //检验Facilities
  659. //var faciltiesResult = _facilities.CheckCDA();
  660. //if (!faciltiesResult.result)
  661. //{
  662. // NotifyError(eEvent.ERR_TRANSPORTER, faciltiesResult.reason, -1);
  663. // return false;
  664. //}
  665. return true;
  666. }
  667. /// <summary>
  668. /// 检验前置条件
  669. /// </summary>
  670. /// <returns></returns>
  671. private bool CheckPreCondition()
  672. {
  673. if (!_gantryAxis.IsSwitchOn)
  674. {
  675. NotifyError(eEvent.ERR_TRANSPORTER, $"{Module} gantry axis is not switch on ", -1);
  676. return false;
  677. }
  678. if (!_gantryAxis.IsHomed)
  679. {
  680. NotifyError(eEvent.ERR_TRANSPORTER, $"{Module} gantry axis is not homed ",-1);
  681. return false;
  682. }
  683. if (!_elevatorAxis.IsSwitchOn)
  684. {
  685. NotifyError(eEvent.ERR_TRANSPORTER, $"{Module} elevator axis is not switch on ", -1);
  686. return false;
  687. }
  688. if (!_elevatorAxis.IsHomed)
  689. {
  690. NotifyError(eEvent.ERR_TRANSPORTER, $"{Module} elevator axis is not homed ",-1);
  691. return false;
  692. }
  693. return true;
  694. }
  695. }
  696. }