Browse Source

Add SRDRunRecipe;

niuyx 2 weeks ago
parent
commit
98951c81ff

+ 14 - 0
CyberX8_RT/Modules/SRD/SRDEntity.cs

@@ -71,6 +71,10 @@ namespace CyberX8_RT.Modules.SRD
         ///  run recipe complete time
         /// </summary>
         private DateTime _runRecipeCompleteTime;
+        /// <summary>
+        /// Wafer已Loader完
+        /// </summary>
+        private bool _isLoaded = false;
         #endregion
 
         #region Routine
@@ -269,6 +273,14 @@ namespace CyberX8_RT.Modules.SRD
         /// 是否为产品模式
         /// </summary>
         public bool IsProduction { get { return _persistentValue != null && _persistentValue.RecipeOperatingMode == PRODUCTION; } }
+        /// <summary>
+        /// Wafer已Loader完
+        /// </summary>
+        public bool IsLoaded
+        {
+            get { return _isLoaded; }
+        }
+        
         #endregion
 
         /// <summary>
@@ -968,6 +980,7 @@ namespace CyberX8_RT.Modules.SRD
             RState state = _loaderRoutine.Monitor();
             if (state == RState.End || state == RState.Failed || state == RState.Timeout)
             {
+                _isLoaded = state == RState.End;
                 return true;
             }
             return false;
@@ -994,6 +1007,7 @@ namespace CyberX8_RT.Modules.SRD
             RState state = _unloaderRoutine.Monitor();
             if (state == RState.End || state == RState.Failed || state == RState.Timeout)
             {
+                _isLoaded = !(state == RState.End);
                 return true;
             }
             return false;

+ 15 - 15
CyberX8_RT/Modules/SRD/SRDLoaderRoutine.cs

@@ -15,14 +15,14 @@ namespace CyberX8_RT.Modules.SRD
     public class SRDLoaderRoutine : RoutineBase, IRoutine
     {
         private enum SRDLoaderStep 
-        { 
-            LiftUpOff,
-            FlippersIn,
-            N2On,
-            Delay,
-            N2Off,
-            ChuckVacuumOn,
-            FlippersOut,
+        {
+            Loader_LiftUpOff,
+            Loader_FlippersIn,
+            Loader_N2On,
+            Loader_Delay,
+            Loader_N2Off,
+            Loader_ChuckVacuumOn,
+            Loader_FlippersOut,
             End
         }
         #region 常量 
@@ -74,13 +74,13 @@ namespace CyberX8_RT.Modules.SRD
         /// <returns></returns>
         public RState Monitor()
         {
-            Runner.Run(SRDLoaderStep.LiftUpOff, LiftUpOff, CheckLiftUpOffEndStatus, CheckLiftUpOffStopStatus)
-                .Run(SRDLoaderStep.FlippersIn, FlippersIn, CheckFlippersInEndStatus, CheckFlippersInStopStatus)
-                .Run(SRDLoaderStep.N2On, N2On, _delay_1ms)
-                .Delay(SRDLoaderStep.Delay, 500)
-                .Run(SRDLoaderStep.N2Off, N2Off, _delay_1ms)
-                .Run(SRDLoaderStep.ChuckVacuumOn, ChuckVacuumOn, CheckChuckVacuumOnEndStatus, CheckChuckVacuumOnStopStatus)
-                .Run(SRDLoaderStep.FlippersOut, FlippersOut, CheckFlippersOutEndStatus, CheckFlippersOutStopStatus)
+            Runner.Run(SRDLoaderStep.Loader_LiftUpOff, LiftUpOff, CheckLiftUpOffEndStatus, CheckLiftUpOffStopStatus)
+                .Run(SRDLoaderStep.Loader_FlippersIn, FlippersIn, CheckFlippersInEndStatus, CheckFlippersInStopStatus)
+                .Run(SRDLoaderStep.Loader_N2On, N2On, _delay_1ms)
+                .Delay(SRDLoaderStep.Loader_Delay, 500)
+                .Run(SRDLoaderStep.Loader_N2Off, N2Off, _delay_1ms)
+                .Run(SRDLoaderStep.Loader_ChuckVacuumOn, ChuckVacuumOn, CheckChuckVacuumOnEndStatus, CheckChuckVacuumOnStopStatus)
+                .Run(SRDLoaderStep.Loader_FlippersOut, FlippersOut, CheckFlippersOutEndStatus, CheckFlippersOutStopStatus)
                 .End(SRDLoaderStep.End, NullFun, _delay_1ms);
             return Runner.Status;
         }

+ 117 - 74
CyberX8_RT/Modules/SRD/SRDProcessRecipeRoutine.cs

@@ -16,6 +16,7 @@ using CyberX8_RT.Devices.Facilities;
 using Aitex.Core.Util;
 using CyberX8_RT.Modules.Transporter;
 using MECF.Framework.Common.WaferHolder;
+using MECF.Framework.Common.Persistent.SRD;
 
 namespace CyberX8_RT.Modules.SRD
 {
@@ -27,18 +28,15 @@ namespace CyberX8_RT.Modules.SRD
 
         private enum SRDProcessState
         {
-            Ready,
-            RunWafer,
+            Ready,    
+            StartLoader,
+            Loading,
+            StartRunRecipe,
+            RunReciping,
+            StartUnloader,
             Unloading,
-            End,
-            InitialRunWaferStateMachine,
-            InitialUnloadingStateMachine,
-            CheckRunWaferStateMachineInitialStatus,
-            CheckUnloadingStateMachineInitialStatus,
-            StartRunWaferStateMachine,
-            StartUnloadingStateMachine,
-            CycleRun,
-            CycleEnd
+            CycleEnd,
+            End
         }
 
         #region 内部变量
@@ -63,13 +61,17 @@ namespace CyberX8_RT.Modules.SRD
         /// </summary>
         private SrdCommonDevice _srdCommonDevice;
         /// <summary>
-        /// Run wafer Recipe Routine
+        /// Run Recipe Routine
         /// </summary>
-        private SRDRunWaferRecipeRoutine _runWaferRecipRoutine;
+        private SRDRunRecipeRoutine _runRecipeRoutine;
         /// <summary>
-        /// Unload Routine
+        /// Unloader Routine
         /// </summary>
-        private SRDUnloadRoutine _unloadRoutine;
+        private SRDUnloaderRoutine _unloaderRoutine;
+        /// <summary>
+        /// Loader Routine
+        /// </summary>
+        private SRDLoaderRoutine _loaderRoutine;
         /// <summary>
         /// lock track time
         /// </summary>
@@ -86,6 +88,10 @@ namespace CyberX8_RT.Modules.SRD
         /// Facilities
         /// </summary>
         private SystemFacilities _facilities;
+        /// <summary>
+        /// Manual模式
+        /// </summary>
+        private bool _isManual = false;
         #endregion
 
         #region 属性
@@ -103,7 +109,7 @@ namespace CyberX8_RT.Modules.SRD
         /// <summary>
         /// 是否正在用水
         /// </summary>
-        public bool IsUsingWater { get { return _runWaferRecipRoutine.IsUsingWater; } }
+        public bool IsUsingWater { get { return _runRecipeRoutine.IsUsingWater; } }
         /// <summary>
         /// LotTrack数据
         /// </summary>
@@ -123,22 +129,27 @@ namespace CyberX8_RT.Modules.SRD
             _rotationAxis = rotationAxis;
             _armAxis = armAxis;
             _srdCommonDevice = srdCommon;
-            _runWaferRecipRoutine = new SRDRunWaferRecipeRoutine(module);
-            _unloadRoutine= new SRDUnloadRoutine(module);
+            _runRecipeRoutine = new SRDRunRecipeRoutine(module);
+            _unloaderRoutine= new SRDUnloaderRoutine(module);
+            _loaderRoutine = new SRDLoaderRoutine(module);
         }
         /// <summary>
         /// 取消
         /// </summary>
         public void Abort()
         {
-            if(_unloadRoutine.Monitor() == RState.Running)
+            if(_unloaderRoutine.Monitor() == RState.Running)
             {
-                _unloadRoutine.Abort();
+                _unloaderRoutine.Abort();
             }
-            if (_runWaferRecipRoutine.Monitor() == RState.Running)
+            if (_runRecipeRoutine.Monitor() == RState.Running)
             {
-                _runWaferRecipRoutine.Abort();
-            }          
+                _runRecipeRoutine.Abort();
+            }
+            if (_loaderRoutine.Monitor() == RState.Running)
+            {
+                _loaderRoutine.Abort();
+            }
             Runner.Stop("Manual Abort");
             if (_srdCommonDevice != null)
             {
@@ -154,11 +165,15 @@ namespace CyberX8_RT.Modules.SRD
         {
             LottrackRecord();
             Runner.LoopStart(SRDProcessState.Ready, "Process Recipe Start", _cycleCount, NullFun, _delay_1ms)
-                .LoopRun(SRDProcessState.StartRunWaferStateMachine, () => { return StartRunWaferRecipeStateMachine(); }, _delay_1ms)
-                .LoopRunWithStopStatus(SRDProcessState.RunWafer, CheckRunWaferRecipeStateCompleteState, CheckRunWaferRecipeStateErrorState)
 
-                .LoopRun(SRDProcessState.StartUnloadingStateMachine, () => { return StartUnloadingStateMachine(); }, _delay_1ms)
-                .LoopRunWithStopStatus(SRDProcessState.Unloading, CheckUnloadingStateCompleteState, CheckUnloadingStateErrorState)
+                .LoopRunIf(SRDProcessState.StartLoader, !_isManual,() => { return StartLoader(); }, _delay_1ms)
+                .LoopRunIfWithStopStatus(SRDProcessState.Loading, !_isManual, CheckLoaderEndStatus, CheckLoaderErrorStatus)
+
+                .LoopRun(SRDProcessState.StartRunRecipe, () => { return StartRunRecipe(); }, _delay_1ms)
+                .LoopRunWithStopStatus(SRDProcessState.RunReciping, CheckRunRecipeEndStatus, CheckRunRecipeErrorStatus)
+
+                .LoopRunIf(SRDProcessState.StartUnloader, !_isManual, () => { return StartUnloader(); }, _delay_1ms)
+                .LoopRunIfWithStopStatus(SRDProcessState.Unloading, !_isManual, CheckUnloaderEndStatus, CheckUnloaderErrorStatus)
 
                 .LoopEnd(SRDProcessState.CycleEnd, NullFun, _delay_1ms)             
                 .End(SRDProcessState.End, NullFun, _delay_1ms);
@@ -183,7 +198,11 @@ namespace CyberX8_RT.Modules.SRD
             {
                 return RState.End;
             }
-            
+            _isManual = false;
+            SRDPersistentValue srdPersistentValue = SRDPersistentManager.Instance.GetModulePersistentValue(Module);
+            if(srdPersistentValue.OperatingMode.Equals("Manual")) _isManual = true;
+
+
             _header.SoftWareVersion = System.Reflection.Assembly.GetExecutingAssembly().GetName().Version.ToString();
             _header.Recipe = $"{_srdRecipe.Ppid}.srd.rcp";
             if (SC.ContainsItem("System.ToolID")) _header.ToolID = SC.GetStringValue("System.ToolID");
@@ -197,40 +216,81 @@ namespace CyberX8_RT.Modules.SRD
             _lotTackTime = DateTime.Now;
             return Runner.Start(Module, "Process Recipe");
         }
-        #region RunWaferStateMatcine      
+
+        #region Loader
+        /// <summary>
+        /// 启动LoaderRoutine
+        /// </summary>
+        /// <param name="recipe"></param>
+        /// <returns></returns>
+        private bool StartLoader()
+        {
+            bool result = _loaderRoutine.Start() == RState.Running;
+            if (!result)
+            {
+                NotifyError(eEvent.ERR_SRD, _loaderRoutine.ErrorMsg, 0);
+            }
+            return result;
+        }
+        /// <summary>
+        /// 检验LoaderRoutine完成情况 
+        /// </summary>
+        /// <returns></returns>
+        private bool CheckLoaderEndStatus()
+        {
+            return CommonFunction.CheckRoutineEndState(_loaderRoutine);
+        }
+        /// <summary>
+        /// 检验LoaderRoutine错误情况 
+        /// </summary>
+        /// <returns></returns>
+        private bool CheckLoaderErrorStatus()
+        {
+            bool result = CommonFunction.CheckRoutineStopState(_loaderRoutine);
+            if (result)
+            {
+                AddLotTrackData();
+                NotifyError(eEvent.ERR_SRD, _loaderRoutine.ErrorMsg, 0);
+            }
+            return result;
+        }
+
+        #endregion
+
+        #region RunRecipe
         /// <summary>
         /// 启动状态机
         /// </summary>
         /// <param name="recipe"></param>
         /// <returns></returns>
-        private bool StartRunWaferRecipeStateMachine()
+        private bool StartRunRecipe()
         {
-            bool result = _runWaferRecipRoutine.Start(_srdRecipe)==RState.Running;           
+            bool result = _runRecipeRoutine.Start(_srdRecipe)==RState.Running;           
             if (result)
             {
-                LOG.WriteLog(eEvent.INFO_SRD, Module.ToString(), $"Start Run Wafer State Machine, recipe[{_srdRecipe.Ppid}] times[{_cycleCount + 1}]");
+                LOG.WriteLog(eEvent.INFO_SRD, Module.ToString(), $"Start Run Recipe, recipe[{_srdRecipe.Ppid}] times[{_cycleCount + 1}]");
             }
             else
             {
-                NotifyError(eEvent.ERR_SRD, _runWaferRecipRoutine.ErrorMsg, 0);
+                NotifyError(eEvent.ERR_SRD, _runRecipeRoutine.ErrorMsg, 0);
             }
             return result;
         }
         /// <summary>
-        /// 检验RunWafer状态机完成情况 
+        /// 检验RunRecipeRoutine完成情况 
         /// </summary>
         /// <returns></returns>
-        private bool CheckRunWaferRecipeStateCompleteState()
+        private bool CheckRunRecipeEndStatus()
         {
-            return CommonFunction.CheckRoutineEndState(_runWaferRecipRoutine);
+            return CommonFunction.CheckRoutineEndState(_runRecipeRoutine);
         }
         /// <summary>
-        /// 检验RunWafer状态机错误情况 
+        /// 检验RunRecipeRoutine错误情况 
         /// </summary>
         /// <returns></returns>
-        private bool CheckRunWaferRecipeStateErrorState()
+        private bool CheckRunRecipeErrorStatus()
         {
-            bool result = CommonFunction.CheckRoutineStopState(_runWaferRecipRoutine);
+            bool result = CommonFunction.CheckRoutineStopState(_runRecipeRoutine);
             if (result)
             {
                 AddLotTrackData(); 
@@ -238,68 +298,51 @@ namespace CyberX8_RT.Modules.SRD
                 {
                     _srdCommonDevice.EnterErrorOperation();
                 }
-                NotifyError(eEvent.ERR_SRD, _runWaferRecipRoutine.ErrorMsg, 0);
+                NotifyError(eEvent.ERR_SRD, _runRecipeRoutine.ErrorMsg, 0);
             }
             return result;           
         }
-        
+
         #endregion
 
-        #region UnloadingStateMatcine        
+        #region Unloader
         /// <summary>
-        /// 启动状态机
+        /// 启动UnloaderRoutine
         /// </summary>
         /// <param name="recipe"></param>
         /// <returns></returns>
-        private bool StartUnloadingStateMachine()
+        private bool StartUnloader()
         {
-            bool result= _unloadRoutine.Start() == RState.Running;
+            bool result= _unloaderRoutine.Start() == RState.Running;
             if (!result)
             {
-                NotifyError(eEvent.ERR_SRD,_unloadRoutine.ErrorMsg, 0);
+                NotifyError(eEvent.ERR_SRD,_unloaderRoutine.ErrorMsg, 0);
             }
             return result;
         }
         /// <summary>
-        /// 检验Unloading状态机完成情况 
+        /// 检验UnloaderRoutine完成情况 
         /// </summary>
         /// <returns></returns>
-        private bool CheckUnloadingStateCompleteState()
+        private bool CheckUnloaderEndStatus()
         {
-            return CommonFunction.CheckRoutineEndState(_unloadRoutine);
+            return CommonFunction.CheckRoutineEndState(_unloaderRoutine);
         }
         /// <summary>
-        /// 检验Unloading状态机错误情况 
+        /// 检验UnloaderRoutine错误情况 
         /// </summary>
         /// <returns></returns>
-        private bool CheckUnloadingStateErrorState()
+        private bool CheckUnloaderErrorStatus()
         {
-            bool result= CommonFunction.CheckRoutineStopState(_unloadRoutine);
+            bool result= CommonFunction.CheckRoutineStopState(_unloaderRoutine);
             if (result)
             {
                 AddLotTrackData();
-                NotifyError(eEvent.ERR_SRD, _unloadRoutine.ErrorMsg, 0);
+                NotifyError(eEvent.ERR_SRD, _unloaderRoutine.ErrorMsg, 0);
             }
             return result;
         }
-        /// <summary>
-        /// 停止unloading状态机
-        /// </summary>
-        /// <returns></returns>
-        private bool StopUnloadingStateMachine()
-        {
-            //_unloadingStateMachine.Stop();
-            if (_rotationAxis.IsRun)
-            {
-                _rotationAxis.StopPositionOperation();
-            }
-            if (_armAxis.IsRun)
-            {
-                _armAxis.StopPositionOperation();
-            }
-            return true;
-        }
-
+        
         #endregion
               
         /// <summary>
@@ -312,13 +355,13 @@ namespace CyberX8_RT.Modules.SRD
             {
                 result = "Ready";
             }
-            else if(Runner.CurrentStep.ToString().Contains("RunWafer"))
+            else if(Runner.CurrentStep.ToString().Contains("RunReciping"))
             {
-                result = _runWaferRecipRoutine.CurrentStep;
+                result = _runRecipeRoutine.CurrentStep;
             }
             else if(Runner.CurrentStep.ToString().Contains("Unloading"))
             {
-                result = _unloadRoutine.CurrentStep;
+                result = _unloaderRoutine.CurrentStep;
             }
             else
             {

+ 411 - 12
CyberX8_RT/Modules/SRD/SRDRunRecipeRoutine.cs

@@ -1,6 +1,7 @@
 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;
@@ -9,6 +10,8 @@ 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
 {
@@ -16,12 +19,30 @@ namespace CyberX8_RT.Modules.SRD
     {
         private enum SRDRunRecipeStep
         {
-            CloseDoor,
-            CheckDoorClosed,
-
+            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 常量 
+        /// <summary>
+        /// 旋转增加时长
+        /// </summary>
+        private const int ROTATION_PLUS_TIME = 10;
+        /// <summary>
+        /// ROTATION电机转速比例
+        /// </summary>
+        private const int SPEED_RATIO = 10;
         #endregion
 
         #region 内部变量
@@ -57,6 +78,18 @@ namespace CyberX8_RT.Modules.SRD
         /// SRD rotation Provider对象
         /// </summary>
         private BeckhoffProviderAxis _rotationProviderAxis;
+        /// <summary>
+        /// enterTime
+        /// </summary>
+        private int _enterTime;
+        /// <summary>
+        /// 转换增加的时间
+        /// </summary>
+        private int _rotationPlusSecond = ROTATION_PLUS_TIME;
+        /// <summary>
+        /// Dry速度
+        /// </summary>
+        private int _drySpeed;
         #endregion
 
         #region 属性
@@ -87,7 +120,19 @@ namespace CyberX8_RT.Modules.SRD
         /// <returns></returns>
         public RState Monitor()
         {
-            Runner//.Run(SRDRunRecipeStep.CloseDoor, !_presenceTestFlag, CheckFacilities, _delay_1ms)
+            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;
         }
@@ -104,16 +149,9 @@ namespace CyberX8_RT.Modules.SRD
                 NotifyError(eEvent.ERR_SRD, "srd recipe is null", 0);
                 return RState.Failed;
             }
-            _rotationAxis = DEVICE.GetDevice<JetAxisBase>($"{Module}.Rotation");
-            if (!_rotationAxis.IsHomed)
-            {
-                NotifyError(eEvent.ERR_SRD, "Rotation is not homed", 0);
-                return RState.Failed;
-            }
+            _rotationAxis = DEVICE.GetDevice<JetAxisBase>($"{Module}.Rotation");           
             _srdCommon = DEVICE.GetDevice<SrdCommonDevice>($"{Module}.Common");
-
             _totalSRDDevice = DEVICE.GetDevice<TotalSRDDevice>("SRD");
-
             _systemFacilities = DEVICE.GetDevice<SystemFacilities>("System.Facilities");
             string otherSRD = Module == "SRD1" ? "SRD2" : "SRD1";
             _otherSrdEntity = Singleton<RouteManager>.Instance.GetModule<SRDEntity>(otherSRD);
@@ -124,9 +162,370 @@ namespace CyberX8_RT.Modules.SRD
                 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");
         }
+        /// <summary>
+        /// Check Pre Condition
+        /// </summary>
+        /// <returns></returns>
+        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<int>("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;
+        }
+        /// <summary>
+        /// Close Door
+        /// </summary>
+        /// <param name="param"></param>
+        /// <returns></returns>
+        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;
+        }
+        /// <summary>
+        /// 检验DoorClosed结束状态
+        /// </summary>
+        /// <param name="param"></param>
+        /// <returns></returns>
+        private bool CheckDoorClosedEndStatus()
+        {
+            return _srdCommon.Status == RState.End && _srdCommon.CommonData.DoorClosed;
+        }
+        /// <summary>
+        /// 检验DoorClosed结束状态
+        /// </summary>
+        /// <param name="param"></param>
+        /// <returns></returns>
+        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;
+        }
+        /// <summary>
+        /// 检验另外一个SRD是否在用水
+        /// </summary>
+        /// <param name="param"></param>
+        /// <returns></returns>
+        private bool CheckOtherSRD()
+        {
+            return (_otherSrdEntity == null || !_otherSrdEntity.IsUsingWater);
+        }
+        /// <summary>
+        /// CheckWaterPressure
+        /// </summary>
+        /// <returns></returns>
+        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;
+        }
+        /// <summary>
+        /// Water On
+        /// </summary>
+        /// <param name="param"></param>
+        /// <returns></returns>
+        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;
+        }
+        /// <summary>
+        /// Check Flow End
+        /// </summary>
+        /// <returns></returns>
+        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;
+        }
+        /// <summary>
+        /// Check Flow Stop
+        /// </summary>
+        /// <param name="param"></param>
+        /// <returns></returns>
+        private bool CheckFlowStopStatus()
+        {
+            int ticks = Environment.TickCount - _enterTime;
+            if (ticks >= _srdRecipe.FlowCheckDelay * 1000)
+            {
+                if (!WaterFlowCheck()) return true;
+            }
+            return false;
+        }
+        /// <summary>
+        /// WaterFlow检测
+        /// </summary>
+        /// <returns></returns>
+        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;
+        }
+        /// <summary>
+        /// 开始旋转
+        /// </summary>
+        /// <param name="param"></param>
+        /// <returns></returns>
+        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;
+        }
+        /// <summary>
+        /// WaterFlow End Monitor
+        /// </summary>
+        /// <returns></returns>
+        private bool MonitorWaterFlowEndStatus()
+        {
+            int ticks = Environment.TickCount - _enterTime;
+            if (ticks >= _srdRecipe.RinseTime * 1000)
+            {
+                return true;
+            }
+            return false;
+        }
+        /// <summary>
+        /// WaterFlow Stop Monitor
+        /// </summary>
+        /// <returns></returns>
+        private bool MonitorWaterFlowStopStatus()
+        {
+            int ticks = Environment.TickCount - _enterTime;
+            if (ticks < _srdRecipe.RinseTime * 1000)
+            {
+                if (!WaterFlowCheck()) return true;
+            }
+            return false;
+        }
+        /// <summary>
+        /// Washing Finished
+        /// </summary>
+        /// <returns></returns>
+        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;
+        }
+        /// <summary>
+        /// change Dry speed
+        /// </summary>
+        /// <returns></returns>
+        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;
+        }
+        /// <summary>
+        /// 停止旋转
+        /// </summary>
+        /// <returns></returns>
+        private bool StopRotation()
+        {
+            bool result = _rotationAxis.StopPositionOperation();
+            LOG.WriteLog(eEvent.INFO_SRD, Module, "Stop Rotation is done");
+            return result;
+        }
+        /// <summary>
+        /// 检验Rotation是否停止
+        /// </summary>
+        /// <returns></returns>
+        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;
+        }
+        /// <summary>
+        /// 检验RotationHome是否完成
+        /// </summary>
+        /// <returns></returns>
+        private bool CheckRotationHomeEndStatus()
+        {
+            if (_rotationAxis.IsHomed && _rotationAxis.Status == RState.End)
+            {                
+                return true;
+            }
+            return false;
+        }
+        /// <summary>
+        /// 检验Rotation Home 停止状态
+        /// </summary>
+        /// <returns></returns>
+        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;
+        }
     }
+    
 }

+ 12 - 12
CyberX8_RT/Modules/SRD/SRDUnloaderRoutine.cs

@@ -14,12 +14,12 @@ namespace CyberX8_RT.Modules.SRD
     {         
         private enum SRDUnloaderStep
         {
-            ChuckVacuumOff,
-            ChuckATMOn,
-            CheckVacuum,
-            LiftUpOn,
-            CheckWaferPresent,
-            OpenDoor,
+            Unloader_ChuckVacuumOff,
+            Unloader_ChuckATMOn,
+            Unloader_CheckVacuum,
+            Unloader_LiftUpOn,
+            Unloader_CheckWaferPresent,
+            Unloader_OpenDoor,
             End
         }
         #region 常量 
@@ -80,12 +80,12 @@ namespace CyberX8_RT.Modules.SRD
         /// <returns></returns>
         public RState Monitor()
         {
-            Runner.Run(SRDUnloaderStep.ChuckVacuumOff, ChuckVacuumOff, CheckChuckVacuumOffEndStatus, CheckChuckVacuumOffStopStatus)
-                .Run(SRDUnloaderStep.ChuckATMOn, ChuckATMOn, CheckChuckATMEndStatus, CheckChuckATMStopStatus)
-                .WaitWithStopCondition(SRDUnloaderStep.CheckVacuum, CheckVacuumEndStatus, CheckVacuumStopStatus)
-                .Run(SRDUnloaderStep.LiftUpOn, LiftUpOn, CheckLiftUpOnEndStatus, CheckLiftUpOnStopStatus)
-                .Run(SRDUnloaderStep.CheckWaferPresent, CheckWaferPresent, _delay_1ms)
-                .Run(SRDUnloaderStep.OpenDoor, OpenDoor, CheckDoorOpenedEndStatus, CheckDoorOpenedStopStatus)
+            Runner.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;
         }

+ 1 - 0
CyberX8_Simulator/Devices/WagoSocketSimulator.cs

@@ -431,6 +431,7 @@ namespace CyberX8_Simulator.Devices
             if (DONameIndexDic.ContainsKey("c_SRD2_LIFT_UP")) DOBytes[DONameIndexDic["c_SRD2_LIFT_UP"]] = 1;
             if (DINameIndexDic.ContainsKey("r_SRD1_WAFER_PRESENT")) DIBytes[DINameIndexDic["r_SRD1_WAFER_PRESENT"]] = 1;
             if (DINameIndexDic.ContainsKey("r_SRD2_WAFER_PRESENT")) DIBytes[DINameIndexDic["r_SRD2_WAFER_PRESENT"]] = 1;
+            if (AINameIndexDic.ContainsKey("r_SRD_SUPPLY_WATER_PRESS")) AIShorts[AINameIndexDic["r_SRD_SUPPLY_WATER_PRESS"]] = 20000;
             //Metal
             if (AINameIndexDic.ContainsKey("r_PUMP4_FLOW")) AIShorts[AINameIndexDic["r_PUMP4_FLOW"]] = 3277;
             if (AINameIndexDic.ContainsKey("r_PUMP3_FLOW")) AIShorts[AINameIndexDic["r_PUMP3_FLOW"]] = 3277;