using Aitex.Core.RT.Device;
using Aitex.Core.RT.Log;
using Aitex.Core.RT.Routine;
using Aitex.Core.RT.SCCore;
using Aitex.Core.Util;
using CyberX8_Core;
using CyberX8_RT.Devices.AXIS;
using CyberX8_RT.Devices.Facilities;
using CyberX8_RT.Devices.SRD;
using MECF.Framework.Common.Beckhoff.AxisProvider;
using MECF.Framework.Common.RecipeCenter;
using MECF.Framework.Common.Routine;
using MECF.Framework.Common.Utilities;
using System;
namespace CyberX8_RT.Modules.SRD
{
public class SRDRunRecipeRoutine : RoutineBase, IRoutine
{
private enum SRDRunRecipeStep
{
RunRecipe_CloseDoor,
RunRecipe_CheckOtherSRD,
RunRecipe_CheckWaterPressure,
RunRecipe_WaterOn,
RunRecipe_CheckWaterFlow,
RunRecipe_StartRotation,
RunRecipe_MonitorWaterFlow,
RunRecipe_WashingFinished,
RunRecipe_ChangeDrySpeed,
RunRecipe_Drying,
RunRecipe_StopRotation,
RunRecipe_CheckRotationStopped,
RunRecipe_CheckRotationHomed,
End
}
#region 常量
///
/// 旋转增加时长
///
private const int ROTATION_PLUS_TIME = 10;
///
/// ROTATION电机转速比例
///
private const int SPEED_RATIO = 10;
#endregion
#region 内部变量
///
/// Rotation Axis
///
private JetAxisBase _rotationAxis;
///
/// SRD Common
///
private SrdCommonDevice _srdCommon;
///
/// Total SRD
///
private TotalSRDDevice _totalSRDDevice;
///
/// 另外SRD实例
///
private SRDEntity _otherSrdEntity;
///
/// Loader Common
///
private SystemFacilities _systemFacilities;
///
/// 是否正在用水
///
private bool _isUsingWater;
///
/// Recipe
///
private SrdRecipe _srdRecipe;
///
/// SRD rotation Provider对象
///
private BeckhoffProviderAxis _rotationProviderAxis;
///
/// enterTime
///
private int _enterTime;
///
/// 转换增加的时间
///
private int _rotationPlusSecond = ROTATION_PLUS_TIME;
///
/// Dry速度
///
private int _drySpeed;
#endregion
#region 属性
///
/// 是否正在用水
///
public bool IsUsingWater { get { return _isUsingWater; } }
#endregion
///
/// 构造函数
///
///
public SRDRunRecipeRoutine(string module) : base(module)
{
}
///
/// 中止
///
public void Abort()
{
Runner.Stop("SRD Run Recipe Abort");
}
///
/// 监控
///
///
public RState Monitor()
{
Runner.Run(SRDRunRecipeStep.RunRecipe_CloseDoor, CloseDoor, CheckDoorClosedEndStatus, CheckDoorClosedStopStatus)
.Wait(SRDRunRecipeStep.RunRecipe_CheckOtherSRD, CheckOtherSRD, _delay_60s)
.Run(SRDRunRecipeStep.RunRecipe_CheckWaterPressure, CheckWaterPressure, _delay_1ms)
.Run(SRDRunRecipeStep.RunRecipe_WaterOn, WaterOn, _delay_1ms)
.WaitWithStopCondition(SRDRunRecipeStep.RunRecipe_CheckWaterFlow, CheckFlowEndStatus, CheckFlowStopStatus)
.Run(SRDRunRecipeStep.RunRecipe_StartRotation, StartRotation, _delay_1ms)
.WaitWithStopCondition(SRDRunRecipeStep.RunRecipe_MonitorWaterFlow, MonitorWaterFlowEndStatus, MonitorWaterFlowStopStatus)
.Run(SRDRunRecipeStep.RunRecipe_WashingFinished, WashingFinished, _delay_1ms)
.Run(SRDRunRecipeStep.RunRecipe_ChangeDrySpeed, ChangeSpeed, _delay_1ms)
.Delay(SRDRunRecipeStep.RunRecipe_Drying, _srdRecipe.DryTime * 1000)
.Run(SRDRunRecipeStep.RunRecipe_StopRotation, StopRotation, _delay_1ms)
.Wait(SRDRunRecipeStep.RunRecipe_CheckRotationStopped, CheckRotationStopEndStatus, _delay_5s)
.WaitWithStopCondition(SRDRunRecipeStep.RunRecipe_CheckRotationHomed, CheckRotationHomeEndStatus, CheckRotationHomedStopStatus)
.End(SRDRunRecipeStep.End, NullFun, _delay_1ms);
return Runner.Status;
}
///
/// 启动
///
///
///
public RState Start(params object[] objs)
{
_srdRecipe = (SrdRecipe)objs[0];
if (_srdRecipe == null)
{
NotifyError(eEvent.ERR_SRD, "srd recipe is null", 0);
return RState.Failed;
}
_rotationAxis = DEVICE.GetDevice($"{Module}.Rotation");
_srdCommon = DEVICE.GetDevice($"{Module}.Common");
_totalSRDDevice = DEVICE.GetDevice("SRD");
_systemFacilities = DEVICE.GetDevice("System.Facilities");
string otherSRD = Module == "SRD1" ? "SRD2" : "SRD1";
_otherSrdEntity = Singleton.Instance.GetModule(otherSRD);
_rotationProviderAxis = BeckhoffAxisProviderManager.Instance.GetAxisProvider($"{Module}.Rotation");
if (_rotationProviderAxis == null)
{
NotifyError(eEvent.ERR_SRD, $"{Module}.Rotation Provider is not exist", 0);
return RState.Failed;
}
if (!CheckPreCondition())
{
return RState.Failed;
}
return Runner.Start(Module, "SRD Run Recipe 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;
}
///
/// Close Door
///
///
///
private bool CloseDoor()
{
if (_srdCommon.CommonData.DoorOpened)
{
bool result = _srdCommon.DoorCloseAction("", null);
if (!result)
{
NotifyError(eEvent.ERR_SRD, "Door Close Action is failed", 0);
return result;
}
}
return true;
}
///
/// 检验DoorClosed结束状态
///
///
///
private bool CheckDoorClosedEndStatus()
{
return _srdCommon.Status == RState.End && _srdCommon.CommonData.DoorClosed;
}
///
/// 检验DoorClosed结束状态
///
///
///
private bool CheckDoorClosedStopStatus()
{
if (_srdCommon.Status == RState.Failed || _srdCommon.Status == RState.Timeout)
{
NotifyError(eEvent.ERR_SRD, "Check Door Closed is failed", 0);
return true;
}
return false;
}
///
/// 检验另外一个SRD是否在用水
///
///
///
private bool CheckOtherSRD()
{
return (_otherSrdEntity == null || !_otherSrdEntity.IsUsingWater);
}
///
/// CheckWaterPressure
///
///
private bool CheckWaterPressure()
{
if (_srdCommon.CommonData.WaterPressure < _srdRecipe.MinWaterPressure)
{
NotifyError(eEvent.ERR_SRD, $"Water Pressure {_srdCommon.CommonData.WaterPressure} is less than recipe's MinWaterPressure {_srdRecipe.MinWaterPressure}", 0);
return false;
}
else if(_srdCommon.CommonData.WaterPressure > _srdRecipe.MaxWaterPressure)
{
NotifyError(eEvent.ERR_SRD, $"Water Pressure {_srdCommon.CommonData.WaterPressure} is over recipe's MaxWaterPressure {_srdRecipe.MaxWaterPressure}", 0);
return false;
}
return true;
}
///
/// Water On
///
///
///
private bool WaterOn()
{
if (_srdRecipe.RinseTime > 0)
{
if (!_srdCommon.CommonData.WaterOn)
{
bool result = _srdCommon.WaterOn();
if (result)
{
LOG.WriteLog(eEvent.INFO_SRD, Module, "Water On");
_isUsingWater = true;
_enterTime = Environment.TickCount;
}
else
{
NotifyError(eEvent.ERR_SRD, "Water On is failed", 0);
}
return result;
}
return true;
}
return true;
}
///
/// Check Flow End
///
///
private bool CheckFlowEndStatus()
{
if (_srdRecipe.FlowCheckDelay == 0)
{
return true;
}
int ticks = Environment.TickCount - _enterTime;
if (ticks >= _srdRecipe.FlowCheckDelay * 1000)
{
if (WaterFlowCheck()) return true;
}
return false;
}
///
/// Check Flow Stop
///
///
///
private bool CheckFlowStopStatus()
{
int ticks = Environment.TickCount - _enterTime;
if (ticks >= _srdRecipe.FlowCheckDelay * 1000)
{
if (!WaterFlowCheck()) return true;
}
return false;
}
///
/// WaterFlow检测
///
///
private bool WaterFlowCheck()
{
if (_srdCommon.CommonData.WaterFlow > _srdRecipe.MaxSRDWaterFlow)
{
NotifyError(eEvent.ERR_SRD, $"{Module} Water Flow:{_srdCommon.CommonData.WaterFlow} is over recipe's MaxSRDWaterFlow:{_srdRecipe.MaxSRDWaterFlow}", 0);
return false;
}
else if (_srdCommon.CommonData.WaterFlow < _srdRecipe.MinSRDWaterFlow)
{
NotifyError(eEvent.ERR_SRD, $"{Module} Water Flow:{_srdCommon.CommonData.WaterFlow} is less than recipe's MinSRDWaterFlow:{_srdRecipe.MinSRDWaterFlow}", 0);
return false;
}
return true;
}
///
/// 开始旋转
///
///
///
private bool StartRotation()
{
if (_srdCommon.CommonData.DoorOpened)
{
NotifyError(eEvent.ERR_SRD, "Door is not closed. Can't do rotation", 0);
return false;
}
double _scale = _rotationProviderAxis.ScaleFactor;
//rinse 目标位置
double rinsePosition = _srdRecipe.RinseTime * BeckhoffVelocityUtil.ConvertVelocityToDegPerSecondByRPM(_srdRecipe.RinseSpeed);
//dry目标位置
double dryPosition = BeckhoffVelocityUtil.ConvertVelocityToDegPerSecondByRPM(_srdRecipe.DrySpeed) * _srdRecipe.DryTime;
//为了让 rotation 不停止,增加了旋转时间(覆盖Arm运动时间)
double plusPosition = BeckhoffVelocityUtil.ConvertVelocityToDegPerSecondByRPM(_srdRecipe.DrySpeed) * _rotationPlusSecond;
int targetPosition = (int)Math.Round((rinsePosition + dryPosition + plusPosition) * _scale, 0);
int rotationSpeed = (int)Math.Round(_scale * BeckhoffVelocityUtil.ConvertVelocityToDegPerSecondByRPM(_srdRecipe.RinseSpeed), 0);
_drySpeed = (int)Math.Round(_scale * BeckhoffVelocityUtil.ConvertVelocityToDegPerSecondByRPM(_srdRecipe.DrySpeed * SPEED_RATIO), 0);
bool result = _rotationAxis.ProfilePosition(targetPosition, rotationSpeed * SPEED_RATIO, 0, 0);
if (!result)
{
NotifyError(eEvent.ERR_SRD, "Start Rotation is failed", 0);
return false;
}
LOG.WriteLog(eEvent.INFO_SRD, Module, "Start Rotation");
//Rinse开始时间
_enterTime = Environment.TickCount;
return true;
}
///
/// WaterFlow End Monitor
///
///
private bool MonitorWaterFlowEndStatus()
{
int ticks = Environment.TickCount - _enterTime;
if (ticks >= _srdRecipe.RinseTime * 1000)
{
return true;
}
return false;
}
///
/// WaterFlow Stop Monitor
///
///
private bool MonitorWaterFlowStopStatus()
{
int ticks = Environment.TickCount - _enterTime;
if (ticks < _srdRecipe.RinseTime * 1000)
{
if (!WaterFlowCheck()) return true;
}
return false;
}
///
/// Washing Finished
///
///
private bool WashingFinished()
{
if (_srdCommon.CommonData.WaterOn)
{
if (!_srdCommon.WaterOff())
{
NotifyError(eEvent.ERR_SRD, "Water Off is failed", 0);
return false;
}
}
_isUsingWater = false;
return true;
}
///
/// change Dry speed
///
///
private bool ChangeSpeed()
{
//调整速度为 drySpeed
bool result = _rotationAxis.ChangeSpeed(_drySpeed);
if (!result)
{
NotifyError(eEvent.ERR_SRD, "Change Speed to Dry Speed is failed", 0);
return false;
}
return true;
}
///
/// 停止旋转
///
///
private bool StopRotation()
{
bool result = _rotationAxis.StopPositionOperation();
LOG.WriteLog(eEvent.INFO_SRD, Module, "Stop Rotation is done");
return result;
}
///
/// 检验Rotation是否停止
///
///
private bool CheckRotationStopEndStatus()
{
if (!_rotationAxis.IsRun && _rotationAxis.Status == RState.End)
{
bool homeResult = _rotationAxis.Home(false);
if (!homeResult)
{
NotifyError(eEvent.ERR_SRD, $"{Module}.Rotation home action is failed", 0);
return false;
}
return true;
}
return false;
}
///
/// 检验RotationHome是否完成
///
///
private bool CheckRotationHomeEndStatus()
{
if (_rotationAxis.IsHomed && _rotationAxis.Status == RState.End)
{
return true;
}
return false;
}
///
/// 检验Rotation Home 停止状态
///
///
private bool CheckRotationHomedStopStatus()
{
if(_rotationAxis.Status == RState.Failed || _rotationAxis.Status == RState.Timeout)
{
NotifyError(eEvent.ERR_SRD, $"{Module}.Rotation home is failed", 0);
return true;
}
return false;
}
}
}