using Aitex.Core.Common;
using Aitex.Core.RT.Device;
using Aitex.Core.RT.Log;
using Aitex.Core.RT.Routine;
using Aitex.Core.RT.SCCore;
using CyberX8_Core;
using CyberX8_RT.Devices.AXIS;
using CyberX8_RT.Devices.Facilities;
using CyberX8_RT.Devices.SRD;
using MECF.Framework.Common.Equipment;
using MECF.Framework.Common.RecipeCenter;
using MECF.Framework.Common.Routine;
using MECF.Framework.Common.SubstrateTrackings;
using System;
using System.Windows.Input;
namespace CyberX8_RT.Modules.SRD
{
public class SRDUnloaderRoutine : RoutineBase, IRoutine
{
private enum SRDUnloaderStep
{
Unloader_FlippersOut,
Unloader_ChuckVacuumOff,
Unloader_ChuckATMOn,
Unloader_CheckVacuum,
Unloader_LiftUpOn,
Unloader_CheckWaferPresent,
Unloader_OpenDoor,
End
}
#region 常量
private const int RETRY_TIMES = 3;
#endregion
#region 内部变量
///
/// Rotation Axis
///
private JetAxisBase _rotationAxis;
///
/// SRD Common
///
private SrdCommonDevice _srdCommon;
///
/// Total SRD
///
private TotalSRDDevice _totalSRDDevice;
///
/// System Facility
///
private SystemFacilities _systemFacilities;
///
/// 当前WaferSize
///
private int _waferSize = 200;
///
/// 当前Retry次数
///
private int _currentRetryTimes = 0;
///
/// 真空值
///
private int _vacuumOffLimit = 0;
///
/// Process Error状态
///
private bool _isProcessError = false;
#endregion
#region 属性
#endregion
///
/// 构造函数
///
///
public SRDUnloaderRoutine(string module) : base(module)
{
}
///
/// 中止
///
public void Abort()
{
Runner.Stop("SRD Loader Abort");
}
///
/// 监控
///
///
public RState Monitor()
{
Runner.RunIf(SRDUnloaderStep.Unloader_FlippersOut, _isProcessError, FlippersOut, CheckFlippersOutEndStatus, CheckFlippersOutStopStatus)
.Run(SRDUnloaderStep.Unloader_ChuckVacuumOff, ChuckVacuumOff, CheckChuckVacuumOffEndStatus, CheckChuckVacuumOffStopStatus)
.Run(SRDUnloaderStep.Unloader_ChuckATMOn, ChuckATMOn, CheckChuckATMEndStatus, CheckChuckATMStopStatus)
.WaitWithStopCondition(SRDUnloaderStep.Unloader_CheckVacuum, CheckVacuumEndStatus, CheckVacuumStopStatus)
.Run(SRDUnloaderStep.Unloader_LiftUpOn, LiftUpOn, CheckLiftUpOnEndStatus, CheckLiftUpOnStopStatus)
.Run(SRDUnloaderStep.Unloader_CheckWaferPresent, CheckWaferPresent, _delay_1ms)
.Run(SRDUnloaderStep.Unloader_OpenDoor, OpenDoor, CheckDoorOpenedEndStatus, CheckDoorOpenedStopStatus)
.End(SRDUnloaderStep.End, NullFun, _delay_1ms);
return Runner.Status;
}
///
/// 启动
///
///
///
public RState Start(params object[] objs)
{
_srdCommon = DEVICE.GetDevice($"{Module}.Common");
_totalSRDDevice = DEVICE.GetDevice("SRD");
_rotationAxis = DEVICE.GetDevice($"{Module}.Rotation");
_systemFacilities = DEVICE.GetDevice("System.Facilities");
_vacuumOffLimit = SC.GetValue("SRD.ChuckVacuumOffLimit");
if (objs.Length >= 1)
{
_isProcessError = (bool)objs[0];
}
if (!_isProcessError && !CheckPreCondition())
{
return RState.Failed;
}
if (!GetWaferSize())
{
NotifyError(eEvent.ERR_SRD, "Wafer Size is invalid", 0);
return RState.Failed;
}
return Runner.Start(Module, "SRD Unloader Start");
}
///
/// Check Pre Condition
///
///
private bool CheckPreCondition()
{
//Check Rotation Home
if (!_rotationAxis.IsHomed)
{
NotifyError(eEvent.ERR_SRD, "Rotation is not homed", 0);
return false;
}
//Check LiftUp
if (_srdCommon.CommonData.LiftUp)
{
NotifyError(eEvent.ERR_SRD, "LiftUp is on", 0);
return false;
}
//Check LiftUp Status
if (_srdCommon.CommonData.LiftUpStatus)
{
NotifyError(eEvent.ERR_SRD, "LiftUp sensor is on", 0);
return false;
}
//Check Wafer Present
if (!_srdCommon.CommonData.WaferPresent)
{
NotifyError(eEvent.ERR_SRD, "WaferPresent sensor is off", 0);
return false;
}
//Check LoaderDI
if (!_systemFacilities.LoaderDiEnable)
{
NotifyError(eEvent.ERR_SRD, "Load DI Is Disable", 0);
return false;
}
//Check Vacuum
int vacuumOnLimit = SC.GetValue("SRD.ChuckVacuumOnLimit");
if (!_srdCommon.CommonData.ChuckVacuum)
{
if(_srdCommon.CommonData.VacuumValue >= vacuumOnLimit)
{
LOG.WriteLog(eEvent.ERR_SRD, Module, $"VacuumValue:{_srdCommon.CommonData.VacuumValue}, VacuumOn Limit:{vacuumOnLimit}");
return false;
}
}
else
{
LOG.WriteLog(eEvent.ERR_SRD, Module, $"Chuck Vacuum is off");
return false;
}
//Check Flippers
if (_srdCommon.CommonData.FlippersIn150 || _srdCommon.CommonData.FlippersIn200) //|| _srdCommon.CommonData.FlippersIn100
{
NotifyError(eEvent.ERR_SRD, "FlippersIn is on", 0);
return false;
}
if (!_srdCommon.CommonData.Flipper1Out150Status || !_srdCommon.CommonData.Flipper2Out150Status || !_srdCommon.CommonData.Flipper3Out150Status
|| !_srdCommon.CommonData.Flipper1Out200Status || !_srdCommon.CommonData.Flipper2Out200Status || !_srdCommon.CommonData.Flipper3Out200Status)
//|| !_srdCommon.CommonData.Flipper1Out100Status || !_srdCommon.CommonData.Flipper2Out100Status || !_srdCommon.CommonData.Flipper3Out100Status
{
NotifyError(eEvent.ERR_SRD, "Flippers are at In position", 0);
return false;
}
return true;
}
///
/// ChuckATMOn
///
///
private bool ChuckATMOn()
{
bool result = _srdCommon.ChuckATMAction("", null);
if (!result)
{
NotifyError(eEvent.ERR_SRD, "Chuck ATM Action is failed", 0);
return result;
}
return true;
}
///
/// Chuck ATM On End
///
///
private bool CheckChuckATMEndStatus()
{
return _srdCommon.Status == RState.End && !_srdCommon.CommonData.ChuckATMOn;
}
///
/// Chuck ATM On Stop
///
///
private bool CheckChuckATMStopStatus()
{
if (_srdCommon.Status == RState.Failed || _srdCommon.Status == RState.Timeout)
{
NotifyError(eEvent.ERR_SRD, "Check ChuckATM is failed", 0);
return true;
}
return false;
}
///
/// Check Vacuum End
///
///
private bool CheckVacuumEndStatus()
{
if(_srdCommon.Status == RState.End && CheckVacuumValue())
{
return true;
}
return false;
}
///
/// Check Vacuum Stop
///
///
private bool CheckVacuumStopStatus()
{
if (_srdCommon.Status == RState.Failed || _srdCommon.Status == RState.Timeout)
{
NotifyError(eEvent.ERR_SRD, "Check ChuckATM is failed", 0);
return true;
}
else if (_srdCommon.Status == RState.End)
{
if (_currentRetryTimes < RETRY_TIMES)
{
_currentRetryTimes++;
NotifyError(eEvent.WARN_SRD, $"Current Chuck ATM Retry Times is {_currentRetryTimes}", 0);
_srdCommon.ChuckATMAction("", null);
}
else
{
NotifyError(eEvent.ERR_SRD, $"Chuck ATM Retry Times is over {RETRY_TIMES}. ChuckATM is failed!", 0);
return true;
}
}
return false;
}
///
/// Check Vacuum Value
///
///
private bool CheckVacuumValue()
{
if (_srdCommon.CommonData.VacuumValue >= _vacuumOffLimit)
{
//LOG.WriteLog(eEvent.INFO_SRD, Module, $"VacuumValue:{_srdCommon.CommonData.VacuumValue}, VacuumOn Limit:{_vacuumOffLimit}");
return true;
}
else
{
LOG.WriteLog(eEvent.WARN_SRD, Module, $"VacuumValue:{_srdCommon.CommonData.VacuumValue}, VacuumOn Limit:{_vacuumOffLimit}");
return false;
}
}
///
/// LiftUpOff
///
///
///
private bool LiftUpOn()
{
bool result = _srdCommon.LiftUpOnAction("", null);
if (!result)
{
NotifyError(eEvent.ERR_SRD, "Lift Up On Action is failed", 0);
return result;
}
return true;
}
///
/// 检验LiftUpOff结束状态
///
///
///
private bool CheckLiftUpOnEndStatus()
{
return _srdCommon.Status == RState.End && _srdCommon.CommonData.LiftUpStatus;
}
///
/// 检验LiftUpOff结束状态
///
///
///
private bool CheckLiftUpOnStopStatus()
{
if (_srdCommon.Status == RState.Failed || _srdCommon.Status == RState.Timeout)
{
NotifyError(eEvent.ERR_SRD, "Check LiftUpOn is failed", 0);
return true;
}
return false;
}
///
/// ChuckVacuumOn
///
///
///
private bool ChuckVacuumOff()
{
bool result = _srdCommon.ChuckVacuumOffAction("", null);
if (!result)
{
NotifyError(eEvent.ERR_SRD, "ChuckVacuumOff Action is failed", 0);
return result;
}
return true;
}
///
/// 检验ChuckVacuumOn结束状态
///
///
///
private bool CheckChuckVacuumOffEndStatus()
{
return _srdCommon.Status == RState.End && _srdCommon.CommonData.ChuckVacuum;
}
///
/// 检验ChuckVacuumOn结束状态
///
///
///
private bool CheckChuckVacuumOffStopStatus()
{
if (_srdCommon.Status == RState.Failed || _srdCommon.Status == RState.Timeout)
{
NotifyError(eEvent.ERR_SRD, "Check ChuckVacuumOff is failed", 0);
return true;
}
return false;
}
///
/// Check WaferPresent
///
///
private bool CheckWaferPresent()
{
if (!_srdCommon.CommonData.WaferPresent)
{
NotifyError(eEvent.ERR_SRD, "WaferPresent sensor is off", 0);
return false;
}
return true;
}
///
/// Open Door
///
///
///
private bool OpenDoor()
{
return _srdCommon.DoorOpenAction("", null);
}
///
/// 检验DoorOpened
///
///
///
private bool CheckDoorOpenedEndStatus()
{
bool result = _srdCommon.Status == RState.End;
if (result)
{
if (_srdCommon.CommonData.DoorOpened && !_srdCommon.CommonData.DoorClosed)
{
return true;
}
else
{
NotifyError(eEvent.ERR_SRD, $"Opened {_srdCommon.CommonData.DoorOpened}&&Closed {_srdCommon.CommonData.DoorClosed}", 0);
return false;
}
}
return false;
}
///
/// 检验Door
///
///
private bool CheckDoorOpenedStopStatus()
{
return _srdCommon.Status == RState.Failed || _srdCommon.Status == RState.Timeout;
}
///
/// Flippers Out
///
///
///
private bool FlippersOut()
{
bool result = false;
object[] objects = new object[1];
objects[0] = _waferSize;
result = _srdCommon.FlipperOutAction("", objects);
if (!result)
{
NotifyError(eEvent.ERR_SRD, $"FlipperOut{_waferSize} Action is failed", 0);
return result;
}
return true;
}
///
/// 检验FlippersOut结束状态
///
///
///
private bool CheckFlippersOutEndStatus()
{
return _srdCommon.Status == RState.End;
}
///
/// 检验FlippersOut结束状态
///
///
///
private bool CheckFlippersOutStopStatus()
{
if (_srdCommon.Status == RState.Failed || _srdCommon.Status == RState.Timeout)
{
NotifyError(eEvent.ERR_SRD, $"Check FlipperOut{_waferSize} Action is failed", 0);
return true;
}
return false;
}
///
/// Get current WaferSize
///
///
private bool GetWaferSize()
{
WaferInfo waferInfo = WaferManager.Instance.GetWafer(ModuleNameString.ToEnum(Module), 0);
if (waferInfo == null)
{
return false;
}
switch (waferInfo.Size)
{
case WaferSize.WS4:
_waferSize = 100;
break;
case WaferSize.WS6:
case WaferSize.WS150:
case WaferSize.WS159:
_waferSize = 150;
break;
case WaferSize.WS0:
case WaferSize.WS8:
_waferSize = 200;
break;
default:
return false;
}
return true;
}
///
/// 关闭 Wafer N2
///
///
private bool N2Off()
{
if(!_srdCommon.CommonData.N2On) return true;
bool result = _srdCommon.N2OffAction("", null);
if (!result)
{
NotifyError(eEvent.ERR_SRD, $"N2 Off Action is failed", 0);
}
return result;
}
///
/// Water Off
///
///
///
private bool WaterOff()
{
if (_srdCommon.CommonData.WaterOn)
{
bool result = _srdCommon.WaterOff();
if (!result)
{
NotifyError(eEvent.ERR_SRD, "Water On is failed", 0);
}
return result;
}
return true;
}
}
}