| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545 | using Aitex.Core.RT.Device;using Aitex.Core.RT.Log;using Aitex.Core.RT.Routine;using Aitex.Core.Util;using MECF.Framework.Common.Equipment;using MECF.Framework.Common.Routine;using MECF.Framework.Common.Utilities;using CyberX8_Core;using CyberX8_RT.Devices.AXIS;using CyberX8_RT.Devices.AXIS.CANOpen;using CyberX8_RT.Modules.PUF;using CyberX8_RT.Modules;using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Threading.Tasks;using MECF.Framework.Common.CommonData.Loader;using CyberX8_RT.Modules.Transporter;using Aitex.Core.RT.SCCore;using CyberX8_RT.Modules.Loader;namespace CyberX8_RT.Devices.Loader{    public class LoaderUnloadRoutine : RoutineBase, IRoutine    {        private enum UnloadStep        {            CheckPreCondition,            WSClampOn,            LSGotoSetUp,            LSGotoSetUpWait,            ShuttleGotoIN,            ShuttleGotoINWait,            BernoulliBladderOn,            BernoulliBladderOnCheck,            BernoulliN2On,            WSBladderOn,            WSBladderOnCheck,            LSVacuumOn,            LSVacuumOnCheck,            LSVacuumLevelCheck,            LSVacuumLevelCheckWait,            ShuttleGotoLS,            ShuttleGotoLSWait,            LSGotoSetup,            LSGotoSetupWait,            LSGotoUnlock,            LSGotoUnlockWait,            ReLSVacuumLevelCheck,            ReLSVacuumLevelCheckWait,            ShuttleGotoOUT,            ShuttleGotoOUTWait,            TiltGotoHori,            TiltGotoHoriCheck,            BernoulliN2Off,            WSBladderOff,            WSBladderOffCheck,                        End        }        #region 常量        private const string SIDE_A = "SideA";        private const string SIDE_B = "SideB";        #endregion        #region 内部变量        private string _side = "";        private JetAxisBase _lsAxis;        private LoaderSideBernoulliBladderRoutine _bernoulliBladderRoutine;        private LoaderSideVacuumRoutine _vacuumRoutine;        private LoaderSideVacuumLevelCheckRoutine _vacuumLevelCheckRoutine;        private LoaderSideWhBladderRoutine _whBladderRoutine;        private JetAxisBase _shuttleAxis;        private JetAxisBase _tiltAxis;        private JetAxisBase _rotationAxis;        private LoaderSideDevice _sideDevice;        private LoaderCommonDevice _loaderCommonDevice;        /// <summary>        /// WaferSize        /// </summary>        private int _waferSize;        #endregion        /// <summary>        /// 构造函数        /// </summary>        /// <param name="module"></param>        public LoaderUnloadRoutine(string module, string side) : base(module)        {            _side = side;                   }        /// <summary>        /// 中止        /// </summary>        public void Abort()        {            Runner.Stop("Manual Abort");        }        /// <summary>        /// 监控        /// </summary>        /// <returns></returns>        public RState Monitor()        {            Runner.Run(UnloadStep.CheckPreCondition,CheckPreCondition,_delay_1ms)                //1. WS Clamp on                .Run(UnloadStep.WSClampOn, WSClampOnAction, () => { return _loaderCommonDevice.CommonData.WaferHolderClamp; }, _delay_5s)                //2. LS Goto SetUp位                .Run(UnloadStep.LSGotoSetUp, () => { return AxisPosition(_lsAxis,$"Setup{_waferSize}"); }, NullFun, _delay_1ms)                .WaitWithStopCondition(UnloadStep.LSGotoSetUpWait, () => { return _lsAxis.Status == RState.End; },                     ()=>CheckAxisMotionStopStatus(_lsAxis))                //3. Shuttle Goto IN                .Run(UnloadStep.ShuttleGotoIN, () => { return AxisPosition(_shuttleAxis, "IN"); }, NullFun, _delay_1ms)                .WaitWithStopCondition(UnloadStep.ShuttleGotoINWait, () => { return _shuttleAxis.Status == RState.End; },                    () => CheckAxisMotionStopStatus(_shuttleAxis))                //4. BernoulliBladderOn                .Run(UnloadStep.BernoulliBladderOn, () => { return _bernoulliBladderRoutine.Start(true) == RState.Running; }, _delay_1ms)                .WaitWithStopCondition(UnloadStep.BernoulliBladderOnCheck, () => { return CommonFunction.CheckRoutineEndState(_bernoulliBladderRoutine); },                     () => CheckRoutineStopStatus(_bernoulliBladderRoutine, "BernoulliBladder On failed"))                //5. BernoulliN2 On                .Run(UnloadStep.BernoulliN2On, BernoulliN2On, () => { return _sideDevice.SideData.BernoulliN2; }, _delay_5s)                //6. WS Bladder On                .Run(UnloadStep.WSBladderOn, () => { return _whBladderRoutine.Start(true) == RState.Running; }, _delay_1ms)                .WaitWithStopCondition(UnloadStep.WSBladderOnCheck, () => { return CommonFunction.CheckRoutineEndState(_whBladderRoutine); },                    () => CheckRoutineStopStatus(_whBladderRoutine, "WSBladder On failed"))                //7. LS Vacuum On                .Run(UnloadStep.LSVacuumOn, () => { return _vacuumRoutine.Start(true) == RState.Running; }, _delay_1ms)                .WaitWithStopCondition(UnloadStep.LSVacuumOnCheck, () => { return CommonFunction.CheckRoutineEndState(_vacuumRoutine); },                    () => CheckRoutineStopStatus(_vacuumRoutine, "LS Vaccum On failed"))                //8. LS vacuum level check                .Run(UnloadStep.LSVacuumLevelCheck, () => { return _vacuumLevelCheckRoutine.Start(true) == RState.Running; }, _delay_1ms)                .WaitWithStopCondition(UnloadStep.LSVacuumLevelCheckWait, () => { return CommonFunction.CheckRoutineEndState(_vacuumLevelCheckRoutine); },                    () => CheckRoutineStopStatus(_vacuumLevelCheckRoutine, "LS Vacuum Level Check failed"))                //9. Shuttle Goto LS                .Run(UnloadStep.ShuttleGotoLS, () => { return AxisPosition(_shuttleAxis, "LS"); }, NullFun, _delay_1ms)                .WaitWithStopCondition(UnloadStep.ShuttleGotoLSWait, () => { return _shuttleAxis.Status == RState.End; },                    () => CheckAxisMotionStopStatus(_shuttleAxis))                //10. LS Goto Unlock                .Run(UnloadStep.LSGotoUnlock, () => { return AxisPosition(_lsAxis, $"Unlock{_waferSize}"); }, NullFun, _delay_1ms)                .WaitWithStopCondition(UnloadStep.LSGotoUnlockWait, () => { return _lsAxis.Status == RState.End; },                    () => CheckAxisMotionStopStatus(_lsAxis))                //11. Re LS Vacuum level Check                .Run(UnloadStep.ReLSVacuumLevelCheck, () => { return _vacuumLevelCheckRoutine.Start(true) == RState.Running; }, _delay_1ms)                .WaitWithStopCondition(UnloadStep.ReLSVacuumLevelCheckWait, () => { return CommonFunction.CheckRoutineEndState(_vacuumLevelCheckRoutine); },                     () => CheckRoutineStopStatus(_vacuumLevelCheckRoutine,"LS Vacuum Level check failed"))                //12. Shuttle Goto OUT                .Run(UnloadStep.ShuttleGotoOUT, () => { return AxisPosition(_shuttleAxis, $"OUT{_waferSize}"); }, NullFun, _delay_1ms)                .WaitWithStopCondition(UnloadStep.ShuttleGotoOUTWait, () => { return _shuttleAxis.Status == RState.End; },                    () => CheckAxisMotionStopStatus(_shuttleAxis))                //13. Tilt Goto HORI                .Run(UnloadStep.TiltGotoHori, () => { return AxisPosition(_tiltAxis,"HORI"); }, NullFun, _delay_1ms)                .WaitWithStopCondition(UnloadStep.TiltGotoHoriCheck, () => { return _tiltAxis.Status == RState.End; },                     () => CheckAxisMotionStopStatus(_tiltAxis))                //14. Bernoulli N2 Off                .Run(UnloadStep.BernoulliN2Off, BernoulliN2Off, () => { return !_sideDevice.SideData.BernoulliN2; }, _delay_5s)                //15. WS Bladder Off                .Run(UnloadStep.WSBladderOff, () => { return _whBladderRoutine.Start(false) == RState.Running; }, _delay_1ms)                .WaitWithStopCondition(UnloadStep.WSBladderOffCheck, () => { return CommonFunction.CheckRoutineEndState(_whBladderRoutine); },                    () => CheckRoutineStopStatus(_whBladderRoutine, "WSBladder Off failed"))                //17 Home CRS Axis                //.Run(UnloadStep.HomingCRSAxis, CRSHome, _delay_1ms)                //.WaitWithStopCondition(UnloadStep.HomingCRSAxisWait, () => { return  _lsAxis.Status == RState.End; },                //    () => CheckCRSHomeStopStatus())                //.Run(UnloadStep.CRSHomedGotoSetUp, () => { return AxisPosition(_lsAxis, $"Setup{_waferSize}"); }, _delay_1ms)                //.WaitWithStopCondition(UnloadStep.CRSHomedGotoSetUpCheck, () => { return _lsAxis.Status == RState.End; },                //    () => CheckAxisMotionStopStatus(_lsAxis))                .End(UnloadStep.End, NullFun, 10);            return Runner.Status;        }        /// <summary>        /// 检验Routine异常状态        /// </summary>        /// <returns></returns>        private bool CheckRoutineStopStatus(IRoutine routine,string error)        {            bool result = CommonFunction.CheckRoutineStopState(routine);            if (result)            {                NotifyError(eEvent.ERR_LOADER, $"{error}", 0);            }            return result;        }        /// <summary>        /// Axis goto position        /// </summary>        /// <param name="axis"></param>        /// <param name="station"></param>        /// <returns></returns>        private bool AxisPosition(JetAxisBase axis,string station)        {            bool result = axis.PositionStation(station);            if (!result)            {                NotifyError(eEvent.ERR_LOADER, $"{axis.Module} goto {station} failed", 0);            }            return result;        }        /// <summary>        /// 检验电机运动异常状态        /// </summary>        /// <param name="axis"></param>        /// <returns></returns>        private bool CheckAxisMotionStopStatus(JetAxisBase axis)        {            bool result = axis.Status == RState.Failed || axis.Status == RState.Timeout;            if (result)            {                NotifyError(eEvent.ERR_LOADER, $"{axis.Module} motion failed", 0);            }            return result;        }        /// <summary>        /// BerolliN2 On        /// </summary>        /// <returns></returns>        private bool BernoulliN2On()        {            bool result= _sideDevice.BernoulliN2OnAction("", null);            if (!result)            {                NotifyError(eEvent.ERR_LOADER, $"BernoulliN2 On failed", 0);            }            return result;        }        /// <summary>        /// BernoulliN2Off        /// </summary>        /// <returns></returns>        private bool BernoulliN2Off()        {            bool result = _sideDevice.BernoulliN2OffAction("", null);            if (!result)            {                NotifyError(eEvent.ERR_LOADER, "BernoulliN2 Off failed", 0);            }            return result;        }        /// <summary>        /// Wafer Holder Clamp Off        /// </summary>        /// <returns></returns>        public bool WSClampOffAction()        {            bool result = _loaderCommonDevice.WaferHolderClampOffAction();            if (!result)            {                NotifyError(eEvent.ERR_LOADER, "Wafer Shuttle Clamp Off failed", 2);            }            return result;        }        /// <summary>        /// Wafer Holder Clamp On        /// </summary>        /// <returns></returns>        public bool WSClampOnAction()        {            bool result = _loaderCommonDevice.WaferHolderClampOnAction();            if (!result)            {                NotifyError(eEvent.ERR_LOADER, "Wafer Shuttle Clamp On failed", 2);            }            return result;        }        /// <summary>        /// 启动        /// </summary>        /// <param name="objs"></param>        /// <returns></returns>        /// <exception cref="NotImplementedException"></exception>        public RState Start(params object[] objs)        {            LoaderEntity loaderEntity = Singleton<RouteManager>.Instance.GetModule<LoaderEntity>(ModuleName.Loader1.ToString());            if (_side == SIDE_A)            {                _waferSize = loaderEntity.SideAWaferSize;            }            else            {                _waferSize = loaderEntity.SideBWaferSize;            }            _shuttleAxis = GetShuttleAxis();            _lsAxis = GetCrsAxis();            _tiltAxis = GetTiltAxis();            _loaderCommonDevice = DEVICE.GetDevice<LoaderCommonDevice>($"Loader1.Common");            _rotationAxis = DEVICE.GetDevice<JetAxisBase>($"{ModuleName.Loader1}.Rotation");            _sideDevice = DEVICE.GetDevice<LoaderSideDevice>($"{Module}.{_side}");            _vacuumRoutine = new LoaderSideVacuumRoutine($"{Module}.{_side}");            _vacuumLevelCheckRoutine = new LoaderSideVacuumLevelCheckRoutine($"{Module}.{_side}");            //_unloadVacuumLevelCheckRoutine = new LoaderSideUnloadVacuumLevelCheckRoutine($"{Module}.{_side}");            _whBladderRoutine = new LoaderSideWhBladderRoutine($"{Module}.{_side}");            _bernoulliBladderRoutine = new LoaderSideBernoulliBladderRoutine($"{Module}.{_side}");            Runner.Start(Module, $"Unload {_side}");            return RState.Running;        }        /// <summary>        /// 获取Shuttle轴对象        /// </summary>        /// <returns></returns>        private JetAxisBase GetShuttleAxis()        {            switch (_side)            {                case "SideA":                    return DEVICE.GetDevice<JetAxisBase>($"{Module}.ShuttleA");                default:                    return DEVICE.GetDevice<JetAxisBase>($"{Module}.ShuttleB");            }        }        /// <summary>        /// 获取CRS轴对象        /// </summary>        /// <returns></returns>        private JetAxisBase GetCrsAxis()        {            switch (_side)            {                case "SideA":                    return DEVICE.GetDevice<JetAxisBase>($"{Module}.LSA");                default:                    return DEVICE.GetDevice<JetAxisBase>($"{Module}.LSB");            }        }        /// <summary>        /// 获取Tilt轴对象        /// </summary>        /// <returns></returns>        private JetAxisBase GetTiltAxis()        {            switch (_side)            {                case "SideA":                    return DEVICE.GetDevice<JetAxisBase>($"{Module}.TiltA");                default:                    return DEVICE.GetDevice<JetAxisBase>($"{Module}.TiltB");            }        }        /// <summary>        /// 检验前置条件        /// </summary>        /// <returns></returns>        private bool CheckPreCondition()        {            if (!CheckHomeCondition())            {                return false;            }            if (!CheckUnloadAxisCondition())            {                return false;            }            if (!UnloadStatusCheck())            {                return false;            }            return true;        }        /// <summary>        /// 检验Home条件        /// </summary>        /// <returns></returns>        private bool CheckHomeCondition()        {            //检验PUF、Loader Transporter,Robot均Homed            //Efem Home            if (ModuleHelper.IsInstalled(ModuleName.EFEM))            {                EfemEntity efemEntity = Singleton<RouteManager>.Instance.GetModule<EfemEntity>(ModuleName.EFEM.ToString());                                if (!efemEntity.IsHomed)                {                    LOG.WriteLog(eEvent.ERR_EFEM_ROBOT, Module, $"{ModuleName.EFEM.ToString()} is not home, Cannot execute GotoSavedPosition");                    return false;                }            }            if (ModuleHelper.IsInstalled(ModuleName.PUF1))            {                PUFEntity puf1Entity = Singleton<RouteManager>.Instance.GetModule<PUFEntity>(ModuleName.PUF1.ToString());                if (!puf1Entity.IsHomed)                {                    NotifyError(eEvent.ERR_LOADER, "PUF1 is not homed",-1);                    return false;                }            }                       if (ModuleHelper.IsInstalled(ModuleName.Transporter2))            {                TransporterEntity loaderTransportEntity = Singleton<RouteManager>.Instance.GetModule<TransporterEntity>(ModuleName.Transporter2.ToString());                if (!loaderTransportEntity.IsHomed)                {                    NotifyError(eEvent.ERR_LOADER, "Loader Transporter is not homed",-1);                    return false;                }            }            return true;        }        /// <summary>        /// 检验Axis条件        /// </summary>        /// <param name="side"></param>        /// <returns></returns>        private bool CheckUnloadAxisCondition()        {            if (!_rotationAxis.IsHomed)            {                NotifyError(eEvent.ERR_LOADER, $"Rotation is not homed",-1);                return false;            }            if (!_shuttleAxis.IsHomed)            {                NotifyError(eEvent.ERR_LOADER, $"{_shuttleAxis.Name} is not homed", -1);                return false;            }            if (!_tiltAxis.IsHomed)            {                NotifyError(eEvent.ERR_LOADER, $"{_tiltAxis.Name} is not homed",-1);                return false;            }            if (!_lsAxis.IsHomed)            {                NotifyError(eEvent.ERR_LOADER, $"{_lsAxis.Name} is not homed", -1);                return false;            }            //LS已经运动到 Setup 位            double crsPosition = _lsAxis.MotionData.MotorPosition;            if (!_lsAxis.CheckPositionIsInStation(crsPosition, $"Setup{_waferSize}"))            {                NotifyError(eEvent.ERR_LOADER, $"LS {crsPosition} not in Setup{_waferSize}", -1);                return false;            }            //Rotation已经运动到 loaderA 位(sideA 下片时)或 loaderB 位(sideB 下片时)            double rotationPosition = _rotationAxis.MotionData.MotorPosition;            if(_side == "SideA")            {                                      if (!_rotationAxis.CheckPositionIsInStation(rotationPosition, $"LOADA{_waferSize}"))                {                    NotifyError(eEvent.ERR_LOADER, $"Rotation {rotationPosition} not in LOADA{_waferSize}", -1);                    return false;                }            }            else            {                if (!_rotationAxis.CheckPositionIsInStation(rotationPosition, $"LOADB{_waferSize}"))                {                    NotifyError(eEvent.ERR_LOADER, $"Rotation {rotationPosition} not in LOADB{_waferSize}", -1);                    return false;                }            }                        //Shuttle已经运动到 MID 位            double shuttlePosition =_shuttleAxis.MotionData.MotorPosition;            if (!_shuttleAxis.CheckPositionIsInStation(shuttlePosition, "MID"))            {                NotifyError(eEvent.ERR_LOADER, $"Shuttle {shuttlePosition} not in MID", -1);                return false;            }            //Tilt已经运动到 VERT 位            double tiltPosition = _tiltAxis.MotionData.MotorPosition;            if (!_tiltAxis.CheckPositionIsInStation(tiltPosition, "VERT"))            {                NotifyError(eEvent.ERR_LOADER, $"Tilt {tiltPosition} not in VERT", -1);                return false;            }                                 return true;        }                /// <summary>        /// 检验状态条件        /// </summary>        /// <param name="side"></param>        /// <returns></returns>        private bool UnloadStatusCheck()        {            LoaderSideData sideData = _sideDevice.SideData;            LoaderCommonData commonData = _loaderCommonDevice.CommonData;            //Bernoulli N2(BernoulliN2 off)            if (sideData.BernoulliN2)            {                NotifyError(eEvent.ERR_LOADER, "Bernoulli N2 is on",-1);                return false;            }            //LS Vacuum检验(LS Vaccum off)            if (sideData.CRSVacuum)            {                NotifyError(eEvent.ERR_LOADER, "LS Vacuum is on", -1);                return false;            }                      //Wafer Shuttle Present(WS present)                     if (!commonData.WaferHolderPresent)            {                NotifyError(eEvent.ERR_LOADER, "Wafer Shuttle is absent", -1);                return false;            }            //Wafer Shuttle Clamp(Clamp on/off均可)                                  //Wafer Shuttle Bladder(Bladder off)                      if (sideData.WHBladder)            {                NotifyError(eEvent.ERR_LOADER, "Wafer Shuttle Bladder is on", -1);                return false;            }            //Drip Tray Fluid(Drip Tray Fluid正常)            if (commonData.DripTrayFluid)            {                NotifyError(eEvent.ERR_LOADER, "Drip Tray Fluid is on", -1);                return false;            }            return true;        }        /// <summary>        /// CRS Home        /// </summary>        /// <returns></returns>        private bool CRSHome()        {            bool result = _lsAxis.Home();            if (!result)            {                NotifyError(eEvent.ERR_LOADER, "LS Axis home failed", 0);            }            return result;        }        /// <summary>        /// 检验CRS home异常状态        /// </summary>        /// <returns></returns>        private bool CheckCRSHomeStopStatus()        {            bool result = _lsAxis.Status == RState.Failed || _lsAxis.Status == RState.Timeout;            if (result)            {                NotifyError(eEvent.ERR_LOADER, "LS Axis home failed", 0);            }            return result;        }    }}
 |