| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751 | using Aitex.Core.RT.DataCenter;using Aitex.Core.RT.Device;using Aitex.Core.RT.Fsm;using Aitex.Core.RT.Log;using Aitex.Core.RT.OperationCenter;using Aitex.Core.RT.Routine;using Aitex.Core.Utilities;using MECF.Framework.Common.Equipment;using MECF.Framework.Common.WaferHolder;using CyberX8_Core;using CyberX8_RT.Devices.AXIS;using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Threading.Tasks;using Aitex.Core.Util;using MECF.Framework.Common.Layout;using Aitex.Core.RT.SCCore;using MECF.Framework.Common.SubstrateTrackings;using MECF.Framework.Common.CommonData;using MECF.Framework.Common.Alarm;using MECF.Framework.Common.Utilities;using Aitex.Core.Common;namespace CyberX8_RT.Modules.Transporter{    public class TransporterEntity : Entity, IEntity, IModuleEntity    {        #region 常量        private const string TRNPA = "TRNPA";        private const string TRNPB = "TRNPB";        #endregion        #region 内部变量        /// <summary>        /// 是否完成Home        /// </summary>        private bool _isHomed;        /// <summary>        /// Gantry电机        /// </summary>        private JetAxisBase _gantryAxis;        /// <summary>        /// Elevator电机        /// </summary>        private JetAxisBase _elevatorAxis;        /// <summary>        /// 当前Routine        /// </summary>        private IRoutine _currentRoutine;        /// <summary>        /// 目标Cell(Transfer或pickupMoveto或moveto目标Cell)        /// </summary>        private string _targetCell;        /// <summary>        /// 源Cell(Transfer或pickupMoveto)        /// </summary>        private string _sourceCell;        #endregion        #region routine        private TransporterHomeRoutine _homeAllRoutine;        private TransporterSwitchOnRoutine _switchOnRoutine;        private TransporterSwitchOffRoutine _switchOffRoutine;        private TransporterPositionRoutine _positionRoutine;        private TransporterGantryPositionRoutine _gantryPositionRoutine;        private TransporterPickUpFromRoutine _pickUpFromRoutine;        private TransporterMoveToRoutine _moveToRoutine;        private TransporterPickDownToRoutine _placeRoutine;        private TransporterParkRoutine _parkRoutine;        private TransporterElevatorUpRoutine _elevatorUpRoutine;        private TransporterElevatorLowRoutine _elevatorLowRoutine;        private TransporterTransferRoutine _transferRoutine;        private TransporterPickUpMoveToRoutine _pickUpMoveToRoutine;        private TransporterPickUpValidateRoutine _pickUpValidateRoutine;        #endregion        #region 属性        public ModuleName Module { get; private set; }        //初始状态        public bool IsInit        {            get { return fsm.State == (int)TransporterState.Init; }        }        /// <summary>        ///Initialized状态(Safety)        /// </summary>        public bool IsInitialized { get { return fsm.State==(int)TransporterState.Initialized; } }        /// <summary>        /// Busy状态        /// </summary>        public bool IsBusy        {            get { return !IsInit && !IsError && !IsIdle; }        }        /// <summary>        /// Idle状态        /// </summary>        public bool IsIdle        {            get { return fsm.State == (int)TransporterState.Idle; }        }        public bool IsAuto { get; } = true;        /// <summary>        /// 是否为工程模式        /// </summary>        public bool IsEngineering { get; } = false;        /// <summary>        /// 是否为产品模式        /// </summary>        public bool IsProduction { get; } = true;        /// <summary>        /// 错误状态        /// </summary>        public bool IsError        {            get { return fsm.State == (int)TransporterState.Error; }        }        /// <summary>        /// Home状态        /// </summary>        public bool IsHomed        {            get { return _isHomed; }        }        /// <summary>        /// 是否禁用        /// </summary>        public bool IsDisable { get; internal set; }        /// <summary>        /// Grantry是否SwitchOn        /// </summary>        public bool IsGantrySwitchOn        {            get { return _gantryAxis.IsSwitchOn; }        }        /// <summary>        /// Elevator是否SwitchOn        /// </summary>        public bool IsElevatorSwitchOn        {            get { return _elevatorAxis.IsSwitchOn; }        }        /// <summary>        /// 当前状态机状态        /// </summary>        public int State { get { return fsm.State; } }        /// <summary>        /// 目标Cell(Transfer或pickupMoveto或moveto目标Cell)        /// </summary>        public string TargetCell { get { return _targetCell; } }        /// <summary>        /// 源Cell(Transfer或pickupMoveto)        /// </summary>        public string SourceCell { get { return _sourceCell; } }        /// <summary>        /// WaferHolder信息        /// </summary>        public WaferHolderInfo WaferHolderInfo { get { return WaferHolderManager.Instance.GetWaferHolder(Module.ToString()); } }        #endregion        /// <summary>        /// 构造函数        /// </summary>        /// <param name="module"></param>        public TransporterEntity(ModuleName module)        {            this.Module = module;            _elevatorAxis = DEVICE.GetDevice<JetAxisBase>($"{module}.Elevator");            _gantryAxis = DEVICE.GetDevice<JetAxisBase>($"{Module}.Gantry");            WaferManager.Instance.SubscribeLocation(Module, 2);            InitialFsm();        }        /// <summary>        /// 初始化        /// </summary>        /// <returns></returns>        protected override bool Init()        {            InitialOperation();            InitialDATA();            InitialRoutine();            return true;        }        /// <summary>        /// 检验所有电机是否已经完成Home        /// </summary>        /// <returns></returns>        private bool CheckAllAxisIsHomed()        {            bool gantryIsHomed = _gantryAxis.CheckAxisIsAreadyHomed();            if (!gantryIsHomed)            {                return false;            }            bool elevatorIsHomed = _elevatorAxis.CheckAxisIsAreadyHomed();            if (!elevatorIsHomed)            {                return false;            }            bool elevatoeUp = _elevatorAxis.CheckPositionIsInStation(_elevatorAxis.MotionData.MotorPosition, "UP");            if (!elevatorIsHomed)            {                return false;            }            if (WaferHolderInfo != null)            {                return false;            }            return true;        }        /// <summary>        /// 初始化状态机        /// </summary>        private void InitialFsm()        {            if (CheckAllAxisIsHomed())            {                _isHomed = true;                fsm = new StateMachine<TransporterEntity>(Module.ToString(), (int)TransporterState.Idle, 100);            }            else            {                fsm = new StateMachine<TransporterEntity>(Module.ToString(), (int)TransporterState.Init, 100);            }            fsm.EnableRepeatedMsg(true);            AnyStateTransition(TransporterMSG.Error, EnterError, TransporterState.Error);            AnyStateTransition(TransporterMSG.Abort, Abort, TransporterState.Init);            AnyStateTransition(TransporterMSG.ReturnIdle, NullFunc, TransporterState.Idle);            AnyStateTransition(TransporterMSG.HomeAll, HomeAll, TransporterState.Homing);            Transition(TransporterState.Error, TransporterMSG.ResumeError,ResumeError, TransporterState.Init);            //SwitchOn            Transition(TransporterState.Init, TransporterMSG.SwitchOn, SwitchOnAll, TransporterState.SwitchOning);            Transition(TransporterState.Idle, TransporterMSG.SwitchOn, SwitchOnAll, TransporterState.SwitchOning);            Transition(TransporterState.Error, TransporterMSG.SwitchOn, SwitchOnAll, TransporterState.SwitchOning);            Transition(TransporterState.SwitchOning, FSM_MSG.TIMER, SwitchOnTimeout, TransporterState.Init);            //SwitchOff            Transition(TransporterState.Init, TransporterMSG.SwitchOff, SwitchOffAll, TransporterState.SwitchOffing);            Transition(TransporterState.Idle, TransporterMSG.SwitchOff, SwitchOffAll, TransporterState.SwitchOffing);            Transition(TransporterState.Error, TransporterMSG.SwitchOff, SwitchOffAll, TransporterState.SwitchOffing);            Transition(TransporterState.SwitchOffing, FSM_MSG.TIMER, SwitchOffTimeout, TransporterState.Init);            // Home            Transition(TransporterState.Homing, FSM_MSG.TIMER, HomingTimeout, TransporterState.Idle);            //Gantry Save Move            Transition(TransporterState.Idle, TransporterMSG.GantrySafeMove, GantrySafeMove, TransporterState.GantrySafeMoving);            Transition(TransporterState.GantrySafeMoving, FSM_MSG.TIMER, GantrySafeMoveMonitor, TransporterState.Idle);            //GantryGoToSavedPosition            Transition(TransporterState.Error, TransporterMSG.GantryGoToSavedPosition, ManualGantryGotoPosition, TransporterState.ErrorGantryPositioning);            Transition(TransporterState.ErrorGantryPositioning, FSM_MSG.TIMER, ManualGantryGotoPositionTimeout, TransporterState.Error);            Transition(TransporterState.Idle,TransporterMSG.GantryGoToSavedPosition, ManualGantryGotoPosition, TransporterState.GantryPositioning);            Transition(TransporterState.GantryPositioning, FSM_MSG.TIMER, ManualGantryGotoPositionTimeout, TransporterState.Idle);            //Pickup            Transition(TransporterState.Idle, TransporterMSG.PickUpFrom, PickUp, TransporterState.PickUping);            Transition(TransporterState.PickUping, FSM_MSG.TIMER, PickUpFromTimeout, TransporterState.Idle);            //MoveTo            Transition(TransporterState.Idle, TransporterMSG.MoveTo, MoveTo, TransporterState.MovingTo);            Transition(TransporterState.MovingTo, FSM_MSG.TIMER, MoveToTimeout, TransporterState.Idle);            //Place            Transition(TransporterState.Idle, TransporterMSG.Place, Place, TransporterState.Placing);            Transition(TransporterState.Placing, FSM_MSG.TIMER, PlaceTimeout, TransporterState.Idle);            //Park            Transition(TransporterState.Idle, TransporterMSG.Park, Park, TransporterState.Parking);            Transition(TransporterState.Parking, FSM_MSG.TIMER, ParkTimeout, TransporterState.Idle);            //Elevator Up            Transition(TransporterState.Idle, TransporterMSG.ElevatorUp, ElevatorUp, TransporterState.ElevatorUping);            Transition(TransporterState.ElevatorUping, FSM_MSG.TIMER, ElevatorUpTimeout, TransporterState.Idle);            //Elevator Low            Transition(TransporterState.Idle, TransporterMSG.ElevatorLow, ElevatorLow, TransporterState.ElevatorLowing);            Transition(TransporterState.ElevatorLowing, FSM_MSG.TIMER, ElevatorLowTimeout, TransporterState.Idle);            //Transfer            Transition(TransporterState.Idle, TransporterMSG.Transfer, Transfer, TransporterState.Transfering);            Transition(TransporterState.Transfering, FSM_MSG.TIMER, TransferTimeout, TransporterState.Idle);            //PickUpMoveTo            Transition(TransporterState.Idle, TransporterMSG.PickUpMoveTo, PickUpMoveTo, TransporterState.PickUpMoveToing);            Transition(TransporterState.PickUpMoveToing, FSM_MSG.TIMER, PickUpMoveToTimeout, TransporterState.PickUpMoveToComplete);            Transition(TransporterState.PickUpMoveToComplete,TransporterMSG.Place, Place, TransporterState.Placing);            Transition(TransporterState.Placing, FSM_MSG.TIMER, PlaceTimeout, TransporterState.Idle);            //PickUpValidate            Transition(TransporterState.Idle, TransporterMSG.PickUpValidate, PickUpValidate, TransporterState.PickUpValidating);            Transition(TransporterState.PickUpValidating, FSM_MSG.TIMER, PickUpValidateTimeout, TransporterState.PickUpValidateComplete);            Transition(TransporterState.PickUpValidateComplete, TransporterMSG.MoveTo, MoveTo, TransporterState.ValidateMoveTo);            Transition(TransporterState.ValidateMoveTo, FSM_MSG.TIMER, MoveToTimeout, TransporterState.ValidateMoveToComplete);            Transition(TransporterState.ValidateMoveToComplete, TransporterMSG.Place, Place, TransporterState.Placing);            //Retry            Transition(TransporterState.Error, TransporterMSG.Retry, NullFunc, TransporterState.Retrying);            Transition(TransporterState.Retrying, FSM_MSG.TIMER, TransporterRetry, TransporterState.Retrying);            Transition(TransporterState.Retrying, TransporterMSG.Transfer, RetryTransfer, TransporterState.Transfering);            Transition(TransporterState.Retrying, TransporterMSG.PickUpMoveTo, RetryPickUpMoveTo, TransporterState.PickUpMoveToing);            Transition(TransporterState.Retrying, TransporterMSG.PickUpValidate, RetryPickUpValidate, TransporterState.PickUpValidating);            Transition(TransporterState.Retrying, TransporterMSG.Place, RetryPlace, TransporterState.Placing);            Transition(TransporterState.Retrying, TransporterMSG.MoveTo, RetryMoveTo, TransporterState.MovingTo);            //ConfirmComplete            Transition(TransporterState.Init, TransporterMSG.ConfirmComplete, ClearModuleAlarm, TransporterState.Init);            Transition(TransporterState.Idle, TransporterMSG.ConfirmComplete, ClearModuleAlarm, TransporterState.Idle);            Transition(TransporterState.Error, TransporterMSG.ConfirmComplete, NullFunc, TransporterState.ConfirmCompleting);            Transition(TransporterState.ConfirmCompleting, FSM_MSG.TIMER, ConfirmComplete, TransporterState.ConfirmCompleting);            Transition(TransporterState.ConfirmCompleting, TransporterMSG.Transfer, ConfirmTransfer, TransporterState.Idle);            Transition(TransporterState.ConfirmCompleting, TransporterMSG.PickUpValidate, ConfirmPickupValidate, TransporterState.PickUpValidateComplete);            Transition(TransporterState.ConfirmCompleting, TransporterMSG.PickUpMoveTo, ConfirmPickupMoveto, TransporterState.PickUpMoveToComplete);            Transition(TransporterState.ConfirmCompleting, TransporterMSG.Place, ConfirmPlace, TransporterState.Idle);             Transition(TransporterState.ConfirmCompleting, TransporterMSG.MoveTo, ConfirmMoveto, TransporterState.ValidateMoveToComplete);            EnumLoop<TransporterState>.ForEach((item) => { fsm.MapState((int)item, item.ToString()); });            EnumLoop<TransporterMSG>.ForEach((item) => { fsm.MapMessage((int)item, item.ToString()); });        }        /// <summary>        /// 初始化操作        /// </summary>        private void InitialOperation()        {            OP.Subscribe($"{Module}.Abort", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.Abort); });            OP.Subscribe($"{Module}.ClearError", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.ResumeError); });            OP.Subscribe($"{Module}.HomeAll", (cmd, args) => { return CheckToPostMessage<TransporterState,TransporterMSG>(eEvent.ERR_TRANSPORTER,Module.ToString(),(int)TransporterMSG.HomeAll); });            OP.Subscribe($"{Module}.Gantry.GantryGotoSavedPosition", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.GantryGoToSavedPosition, "Gantry", args); });            //OP.Subscribe($"{Module}.Elevator.GotoSavedPosition", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.GoToSavedPosition, "Elevator", args); });             OP.Subscribe($"{Module}.PickUpFrom", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.PickUpFrom,args); });            OP.Subscribe($"{Module}.MoveTo", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.MoveTo, args); });            OP.Subscribe($"{Module}.PutDownTo", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.Place, args); });            OP.Subscribe($"{Module}.Park", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.Park, args); });            OP.Subscribe($"{Module}.ElevatorUp", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.ElevatorUp, args); });            OP.Subscribe($"{Module}.ElevatorLow", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.ElevatorLow, args); });            OP.Subscribe($"{Module}.SwitchOn", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.SwitchOn, args); });            OP.Subscribe($"{Module}.SwitchOff", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.SwitchOff, args); });            OP.Subscribe($"{Module}.Transfer", (cmd, args) => { return CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.Transfer, args); });        }        /// <summary>        /// 初始化数据        /// </summary>        private void InitialDATA()        {            InitializeSvid();            DATA.Subscribe($"{Module}.FsmState", () => ((TransporterState)fsm.State).ToString(), SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.IsHomed", () => _isHomed, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.IsIdle", () => IsIdle, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.IsError", () => IsError, SubscriptionAttribute.FLAG.IgnoreSaveDB);        }        /// <summary>        /// 初始化SVID        /// </summary>        private void InitializeSvid()        {            DATA.Subscribe($"{Module}.State", () => ((TransporterState)fsm.State).ToString(), SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.LotID", () => WaferHolderInfo?.LotId, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.WSID", () => WaferHolderInfo?.Id, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.LSAID", () => WaferHolderInfo?.CrsAId, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.LSBID", () => WaferHolderInfo?.CrsBId, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.SequenceRecipe", () => WaferHolderInfo?.SequenceId, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.WaferAID", () => WaferHolderInfo?.WaferAId, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.WaferBID", () => WaferHolderInfo?.WaferBId, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.Task", () => WaferHolderInfo?.CurrentControlJobId, SubscriptionAttribute.FLAG.IgnoreSaveDB);        }        /// <summary>        /// 初始化Routine        /// </summary>        private void InitialRoutine()        {            _homeAllRoutine = new TransporterHomeRoutine(Module.ToString());            _switchOnRoutine = new TransporterSwitchOnRoutine(Module.ToString());            _switchOffRoutine=new TransporterSwitchOffRoutine(Module.ToString());                        _positionRoutine = new TransporterPositionRoutine(Module.ToString());            _gantryPositionRoutine = new TransporterGantryPositionRoutine(Module.ToString());            _pickUpFromRoutine = new TransporterPickUpFromRoutine(Module.ToString());            _moveToRoutine=new TransporterMoveToRoutine(Module.ToString());            _placeRoutine=new TransporterPickDownToRoutine(Module.ToString());            _parkRoutine=new TransporterParkRoutine(Module.ToString());            _elevatorUpRoutine=new TransporterElevatorUpRoutine(Module.ToString());            _elevatorLowRoutine = new TransporterElevatorLowRoutine(Module.ToString());            _transferRoutine =new TransporterTransferRoutine(Module.ToString());            _pickUpMoveToRoutine = new TransporterPickUpMoveToRoutine(Module.ToString());            _pickUpValidateRoutine=new TransporterPickUpValidateRoutine(Module.ToString());        }        /// <summary>        /// 进入Error状态        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool EnterError(object[] param)        {            return true;        }        /// <summary>        /// 恢复错误        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ResumeError(object[] param)        {            if(_isHomed)            {                PostMsg(TransporterMSG.ReturnIdle);                return false;            }            return true;        }        #region Abort        private bool Abort(object parameter)        {            bool preHomed = IsHomed;            _gantryAxis.StopPositionOperation();            _elevatorAxis.StopPositionOperation();            if (_currentRoutine != null)            {                _currentRoutine.Abort();                _currentRoutine = null;            }            if (preHomed)            {                PostMsg(TransporterMSG.ReturnIdle);                return false;            }            return true;        }        #endregion        #region Switch On        /// <summary>        /// SwitchAll        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool SwitchOnAll(object[] param)        {            return _switchOnRoutine.Start() == RState.Running;        }        private bool SwitchOnTimeout(object[] param)        {            RState ret = _switchOnRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                PostMsg(TransporterMSG.Error);                return false;            }            bool result = ret == RState.End;            if (result)            {                _isHomed = false;            }            return result;        }        #endregion        #region Switch Off        /// <summary>        /// SwitchAll        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool SwitchOffAll(object[] param)        {            return _switchOffRoutine.Start() == RState.Running;        }        private bool SwitchOffTimeout(object[] param)        {            RState ret = _switchOffRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                PostMsg(TransporterMSG.Error);                return false;            }            bool result = ret == RState.End;            if (result)            {                _isHomed = false;            }            return result;        }        #endregion        #region Home        /// <summary>        /// HomeAll        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool HomeAll(object[] param)        {            _isHomed = false;            bool result= _homeAllRoutine.Start() == RState.Running;            if (result)            {                _currentRoutine = _homeAllRoutine;            }            return result;        }        /// <summary>        /// Home超时        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool HomingTimeout(object[] param)        {            RState ret = _homeAllRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                _currentRoutine = null;                PostMsg(TransporterMSG.Error);                _isHomed = false;                return false;            }            bool result = ret == RState.End;            if (result)            {                _currentRoutine = null;                _isHomed = true;            }            return result;        }        #endregion        #region GantrySafeMove        /// <summary>        /// Gantry安全移动        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool GantrySafeMove(object[] param)        {            double targetPosition=(double)param[0];            return _gantryAxis.ProfilePositionOperation(targetPosition);        }        /// <summary>        /// Gantry安全移动监控        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool GantrySafeMoveMonitor(object[] param)        {            if (_gantryAxis.Status == RState.End)            {                return true;            }            if (_gantryAxis.Status == RState.Failed || _gantryAxis.Status == RState.Timeout)            {                PostMsg(TransporterMSG.Error);                return false;            }            return false;        }        #endregion        #region Manual Gantry GoToPosition        /// <summary>        /// Manual Gantry Go to Position        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ManualGantryGotoPosition(object[] param)        {            string axis = param[0].ToString();            object[] objs = (object[])param[1];            string position = objs[1].ToString();            var result = CheckGotoPositionPreCondition(axis, position);            if (result.result)            {                bool posresult = _gantryPositionRoutine.Start(position) == RState.Running;                if (posresult)                {                    _currentRoutine = _positionRoutine;                }                return posresult;            }            else            {                return false;            }        }        /// <summary>        /// Manual gantry Go to Position Time Out        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ManualGantryGotoPositionTimeout(object[] param)        {            RState ret = _gantryPositionRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                PostMsg(TransporterMSG.Error);                _currentRoutine = null;                return false;            }            bool result = ret == RState.End;            if (result)            {                _currentRoutine = null;            }            return result;        }        /// <summary>        /// 检验GotoPosition前置条件        /// </summary>        /// <param name="axis"></param>        /// <param name="position"></param>        /// <returns></returns>        private (bool result, JetAxisBase axis) CheckGotoPositionPreCondition(string axis, string position)        {            switch (axis)            {                case "Gantry":                    return (_gantryAxis.CheckGotoPosition(position), _gantryAxis);                case "Elevator":                    return (_elevatorAxis.CheckGotoPosition(position), _elevatorAxis);                default:                    return (false, null);            }        }        #endregion        #region Pick Up        /// <summary>        /// Pick Up        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool PickUp(object[] param)        {            bool result= _pickUpFromRoutine.Start(param[0]) == RState.Running;            if(result)            {                _currentRoutine = _pickUpFromRoutine;            }            return result;        }        /// <summary>        /// PickUpFrom超时        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool PickUpFromTimeout(object[] param)        {            RState ret = _pickUpFromRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                _currentRoutine = null;                PostMsg(TransporterMSG.Error);                return false;            }            bool result= ret == RState.End;            if(result)            {                _currentRoutine = null;                AlarmListManager.Instance.CheckModuleAlamAndRemove(Module.ToString(), TransporterState.PickUping.ToString());            }            return result;        }        #endregion        #region Move To        /// <summary>        /// Move To        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool MoveTo(object[] param)        {            if (!CheckOtherEntityStatus(param[0].ToString()))            {                LOG.WriteLog(eEvent.WARN_TRANSPORTER, Module.ToString(), $"Module Axis meets conflict of other Axis");                return false;            }            bool result= _moveToRoutine.Start(param[0]) == RState.Running;            if(result)            {                _targetCell = param[0].ToString();                _sourceCell = "";                _currentRoutine = _moveToRoutine;            }            return result;        }        /// <summary>        /// Retry Place        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool RetryMoveTo(object[] param)        {            int stepIndex = (int)param[0];            bool result = _moveToRoutine.Retry(stepIndex) == RState.Running;            if (result)            {                _currentRoutine = _moveToRoutine;                _targetCell = _moveToRoutine.TargetCell;            }            return result;        }        /// <summary>        /// MoveTo超时        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool MoveToTimeout(object[] param)        {            RState ret = _moveToRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                _targetCell = "";                _currentRoutine = null;                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmList alarmList = new AlarmList(Module.ToString(), ((TransporterState)fsm.State).ToString(), (int)TransporterMSG.MoveTo,                    _placeRoutine.ErrorMsg, _placeRoutine.ErrorStep, (int)AlarmType.Error);                    AlarmListManager.Instance.AddAlarm(alarmList);                }                PostMsg(TransporterMSG.Error);                return false;            }            bool result = ret == RState.End;            if (result)            {                _targetCell = "";                _currentRoutine = null;                if (!string.IsNullOrEmpty(_sourceCell))                {                    _sourceCell = "";                }            }            return result;        }        /// <summary>        /// 确认Place是否完成        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ConfirmMoveto(object[] param)        {            int stepIdex = (int)param[0];            bool result = _moveToRoutine.CheckCompleteCondition(stepIdex);            if (!result)            {                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmList alarmList = new AlarmList(Module.ToString(), ((TransporterState)fsm.State).ToString(), (int)TransporterMSG.MoveTo,                    _moveToRoutine.ErrorMsg, _moveToRoutine.ErrorStep, (int)AlarmType.Error);                    AlarmListManager.Instance.AddAlarm(alarmList);                }                PostMsg(TransporterMSG.Error);            }            else            {                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmListManager.Instance.CheckModuleAlamAndRemove(Module.ToString(), TransporterState.MovingTo.ToString());                }            }            return result;        }        #endregion        #region Place        /// <summary>        /// Place        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool Place(object[] param)        {            _targetCell = "";            bool result = _placeRoutine.Start(param[0]) == RState.Running;            if(result)            {                _currentRoutine = _placeRoutine;                _targetCell = param[0].ToString();            }            return result;        }        /// <summary>        /// Retry Place        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool RetryPlace(object[] param)        {                        int stepIndex = (int)param[0];            bool result = _placeRoutine.Retry(stepIndex) == RState.Running;            if (result)            {                _currentRoutine = _placeRoutine;                _targetCell = _placeRoutine.TargetCell;            }            return result;        }        /// <summary>        /// Place超时        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool PlaceTimeout(object[] param)        {            RState ret = _placeRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                _currentRoutine = null;                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmList alarmList = new AlarmList(Module.ToString(), ((TransporterState)fsm.State).ToString(), (int)TransporterMSG.Place,                    _placeRoutine.ErrorMsg, _placeRoutine.ErrorStep, (int)AlarmType.Error);                    AlarmListManager.Instance.AddAlarm(alarmList);                }                PostMsg(TransporterMSG.Error);                return false;            }            bool result = ret == RState.End;            if (result)            {                _currentRoutine = null;                _targetCell = "";                if (!string.IsNullOrEmpty(_sourceCell))                {                    _sourceCell = "";                }                AlarmListManager.Instance.CheckModuleAlamAndRemove(Module.ToString(), TransporterState.Placing.ToString());            }            return result;        }        /// <summary>        /// 确认Place是否完成        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ConfirmPlace(object[] param)        {            int stepIdex = (int)param[0];            bool result = _placeRoutine.CheckCompleteCondition(stepIdex);            if (!result)            {                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmList alarmList = new AlarmList(Module.ToString(), ((TransporterState)fsm.State).ToString(), (int)TransporterMSG.Place,                    _placeRoutine.ErrorMsg, _placeRoutine.ErrorStep, (int)AlarmType.Error);                    AlarmListManager.Instance.AddAlarm(alarmList);                }                PostMsg(TransporterMSG.Error);            }            else            {                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmListManager.Instance.CheckModuleAlamAndRemove(Module.ToString(), TransporterState.Placing.ToString());                }            }            return result;        }        #endregion        #region Park        /// <summary>        /// Park        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool Park(object[] param)        {            bool result = _parkRoutine.Start() == RState.Running;            if (result)            {                _currentRoutine = _parkRoutine;            }            return result;        }        /// <summary>        /// Park超时        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ParkTimeout(object[] param)        {            RState ret = _parkRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                _currentRoutine = null;                PostMsg(TransporterMSG.Error);                return false;            }            bool result = ret == RState.End;            if (result)            {                _currentRoutine = null;            }            return result;        }        #endregion        #region Elevator Up        /// <summary>        /// Elevator Up        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ElevatorUp(object[] param)        {            bool result = _elevatorUpRoutine.Start() == RState.Running;            if (result)            {                _currentRoutine = _elevatorUpRoutine;            }            return result;        }        /// <summary>        /// Elevator Up超时        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ElevatorUpTimeout(object[] param)        {            RState ret = _elevatorUpRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                _currentRoutine = null;                PostMsg(TransporterMSG.Error);                return false;            }            bool result = ret == RState.End;            if (result)            {                _currentRoutine = null;            }            return result;        }        #endregion        #region Elevator Low        /// <summary>        /// Elevator Low        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ElevatorLow(object[] param)        {            bool result= _elevatorLowRoutine.Start() == RState.Running;            if (result)            {                _currentRoutine = _elevatorLowRoutine;            }            return result;        }        /// <summary>        /// Elevator Low超时        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ElevatorLowTimeout(object[] param)        {            RState ret = _elevatorLowRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                _currentRoutine = null;                PostMsg(TransporterMSG.Error);                return false;            }            bool result = ret == RState.End;            if (result)            {                _currentRoutine = null;            }            return result;        }        #endregion        #region Transfer        /// <summary>        /// Transfer        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool Transfer(object[] param)        {            if (!CheckOtherEntityStatus(param[0].ToString()))            {                LOG.WriteLog(eEvent.WARN_TRANSPORTER, Module.ToString(), $"Module Axis meets conflict of other Axis from {param[0]}");                return false;            }            if (!CheckOtherEntityStatus(param[1].ToString()))            {                LOG.WriteLog(eEvent.WARN_TRANSPORTER, Module.ToString(), $"Module Axis meets conflict of other Axis to {param[1]}");                return false;            }            bool result = _transferRoutine.Start(param) == RState.Running;            if (result)            {                _sourceCell=param[0].ToString();                _targetCell = param[1].ToString();                _currentRoutine = _transferRoutine;            }            return result;        }        /// <summary>        /// Retry Transfer        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool RetryTransfer(object[] param)        {            if (!CheckOtherEntityStatus(_sourceCell))            {                LOG.WriteLog(eEvent.WARN_TRANSPORTER, Module.ToString(), $"Module Axis meets conflict of other Axis from {_sourceCell}");                return false;            }            if (!CheckOtherEntityStatus(_targetCell))            {                LOG.WriteLog(eEvent.WARN_TRANSPORTER, Module.ToString(), $"Module Axis meets conflict of other Axis to {_targetCell}");                return false;            }            int stepIndex = (int)param[0];            bool result = _transferRoutine.Retry(stepIndex) == RState.Running;            if (result)            {                _currentRoutine = _transferRoutine;                _sourceCell=_transferRoutine.SourceCell;                _targetCell=_transferRoutine.TargetCell;            }            return result;        }        /// <summary>        /// Transfer超时        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool TransferTimeout(object[] param)        {            RState ret = _transferRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                _targetCell = "";                _sourceCell = "";                _currentRoutine = null;                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmList alarmList = new AlarmList(Module.ToString(), ((TransporterState)fsm.State).ToString(), (int)TransporterMSG.Transfer,                    _transferRoutine.ErrorMsg, _transferRoutine.ErrorStep, (int)AlarmType.Error);                    AlarmListManager.Instance.AddAlarm(alarmList);                }                PostMsg(TransporterMSG.Error);                return false;            }            bool result = ret == RState.End;            if (result)            {                _targetCell = "";                _sourceCell = "";                _currentRoutine = null;                AlarmListManager.Instance.CheckModuleAlamAndRemove(Module.ToString(), TransporterState.Transfering.ToString());            }            return result;        }        /// <summary>        /// 确认Transfer是否完成        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ConfirmTransfer(object[] param)        {            int stepIdex = (int)param[0];            bool result = _transferRoutine.CheckCompleteCondition(stepIdex);            if (!result)            {                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmList alarmList = new AlarmList(Module.ToString(), ((TransporterState)fsm.State).ToString(), (int)TransporterMSG.Transfer,                    _transferRoutine.ErrorMsg, _transferRoutine.ErrorStep, (int)AlarmType.Error);                    AlarmListManager.Instance.AddAlarm(alarmList);                }                PostMsg(TransporterMSG.Error);            }            else            {                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmListManager.Instance.CheckModuleAlamAndRemove(Module.ToString(), TransporterState.Transfering.ToString());                }            }            return result;        }        /// <summary>        /// 检验另一个Axis状态        /// </summary>        /// <returns></returns>        public bool CheckOtherEntityStatus(string targetCell)        {            bool positive = false;            TransporterEntity otherEntity=null;            string otherModule = "";            if (Module == ModuleName.Transporter2)            {                otherModule = ModuleName.Transporter1.ToString();                otherEntity = Singleton<RouteManager>.Instance.GetModule<TransporterEntity>(ModuleName.Transporter1.ToString());                positive = true;            }            else            {                otherModule = ModuleName.Transporter2.ToString();                otherEntity = Singleton<RouteManager>.Instance.GetModule<TransporterEntity>(ModuleName.Transporter2.ToString());            }            if(otherEntity==null)            {                return true;            }            if (otherEntity.IsIdle)            {                return true;            }            if (otherEntity.IsError)            {                return false;            }            if(otherEntity.State==(int)TransporterState.Transfering||otherEntity.State==(int)TransporterState.PickUpMoveToing                ||otherEntity.State==(int)TransporterState.MovingTo||otherEntity.State==(int)TransporterState.PickUpValidating                ||otherEntity.State==(int)TransporterState.PickUpValidateComplete||otherEntity.State==(int)TransporterState.ValidateMoveTo)            {                bool conflict= CheckModuleWithOtherModuleConflict(otherEntity,otherModule,positive,targetCell);                if(conflict)                {                    return false;                }                else                {                    return true;                }            }            else            {                return true;            }        }        /// <summary>        /// 分析目标cell的位置        /// </summary>        /// <param name="otherModule"></param>        /// <param name="targetCell"></param>        /// <returns></returns>        private (bool result,double targetPosition) AnalyseTargetCellPosition(string otherModule,string targetCell)        {            ProcessLayoutCellItem _cellItem = ProcessLayoutManager.Instance.GetProcessLayoutCellItemByModuleName(targetCell);            string stationName = targetCell;            if (_cellItem != null)            {                if (targetCell.ToLower() != "loader" && targetCell.ToLower() != "park")                {                    stationName = $"Cell{_cellItem.CellId}";                }            }            else            {                LOG.WriteLog(eEvent.ERR_TRANSPORTER, Module.ToString(), $"{targetCell} not in layout");                return (false,0);            }            JetAxisBase otherGantryAxis = DEVICE.GetDevice<JetAxisBase>($"{otherModule}.Gantry");            var result = otherGantryAxis.GetPositionByStation(stationName);            if(result.success)            {                return (true,result.position);            }            return (false,0);        }        /// <summary>        /// 检验当前模块与另一个模块是否存在冲突        /// </summary>        /// <returns></returns>        private bool CheckModuleWithOtherModuleConflict(TransporterEntity otherEntity,string otherModule,bool positive,string targetCell)        {            int transporterMinimumDistance = SC.GetValue<int>("Transporter.TransporterMinimumDistance");            double motorPosition = _gantryAxis.MotionData.MotorPosition;            if(targetCell==ModuleName.Loader1.ToString())            {                targetCell = "Loader";            }            var result = AnalyseTargetCellPosition(Module.ToString(), targetCell);            if(!result.result)            {                return false;            }            double targetPosition=result.targetPosition;            JetAxisBase otherGantryAxis = DEVICE.GetDevice<JetAxisBase>($"{otherModule}.Gantry");            if(otherGantryAxis==null)            {                return false;            }            double otherPosition = otherGantryAxis.MotionData.MotorPosition;            if (!string.IsNullOrEmpty(otherEntity.TargetCell))            {                return CheckOtherModuleCellConflict(otherModule, otherEntity.TargetCell,otherPosition, positive, targetPosition);            }            if(!string.IsNullOrEmpty(otherEntity.SourceCell))            {                bool conflict= CheckOtherModuleCellConflict(otherModule, otherEntity.SourceCell,otherPosition, positive, targetPosition);                if (conflict)                {                    //另一个Entity已经到达了目标位置                    if (!string.IsNullOrEmpty(otherEntity.TargetCell)&&CheckOtherEntityAlreadyInTargetCell(otherEntity, otherEntity.TargetCell))                    {                        return false;                    }                }                return conflict;            }            else            {                return false;            }        }        /// <summary>        /// 检验其他TransporterEntity已经到达目标cell        /// </summary>        /// <param name="otherEntity"></param>        /// <param name="targetCell"></param>        /// <returns></returns>        private bool CheckOtherEntityAlreadyInTargetCell(TransporterEntity otherEntity,string targetCell)        {            //另一个Transporter已经取走了WaferHolder,同时gantry已经到达了目标cell            if (otherEntity.WaferHolderInfo != null)            {                JetAxisBase jetAxisBase = DEVICE.GetDevice<JetAxisBase>($"{otherEntity.Module}.Gantry");                if (jetAxisBase != null && jetAxisBase.CheckPositionIsInStation(jetAxisBase.MotionData.MotorPosition, targetCell))                {                    return true;                }            }            return false;        }        /// <summary>        /// 检验另一个模块Cell是否存在冲突        /// </summary>        /// <param name="otherModule"></param>        /// <param name="cell"></param>        /// <param name="positive"></param>        /// <param name="motorPosition"></param>        /// <param name="targetPosition"></param>        /// <returns></returns>        private bool CheckOtherModuleCellConflict(string otherModule, string cell,double otherPosition, bool positive,double targetPosition)        {            int transporterMinimumDistance = SC.GetValue<int>("Transporter.TransporterMinimumDistance");            var result = AnalyseTargetCellPosition(otherModule, cell);            if (!result.result)            {                return false;            }            else            {                if (positive)                {                //    if (result.targetPosition - transporterMinimumDistance <= motorPosition)                //    {                //        return true;                //    }                    if (result.targetPosition - transporterMinimumDistance <= targetPosition)                    {                        return true;                    }                    if(otherPosition - transporterMinimumDistance <= targetPosition)                    {                        return true;                    }                    return false;                }                else                {                    //if (result.targetPosition + transporterMinimumDistance >= motorPosition)                    //{                    //    return true;                    //}                    if (result.targetPosition + transporterMinimumDistance >= targetPosition)                    {                        return true;                    }                    if(otherPosition + transporterMinimumDistance >= targetPosition)                    {                        return true;                    }                    return false;                }            }        }        #endregion        #region PickUpMoveTo        /// <summary>        /// PickUpMoveTo        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool PickUpMoveTo(object[] param)        {            if (!CheckOtherEntityStatus(param[0].ToString()))            {                LOG.WriteLog(eEvent.WARN_TRANSPORTER, Module.ToString(), $"Module Axis meets conflict of other Axis from {param[0]}");                return false;            }            if (!CheckOtherEntityStatus(param[1].ToString()))            {                LOG.WriteLog(eEvent.WARN_TRANSPORTER, Module.ToString(), $"Module Axis meets conflict of other Axis to {param[1]}");                return false;            }            bool result = _pickUpMoveToRoutine.Start(param) == RState.Running;            if (result)            {                _sourceCell = param[0].ToString();                _targetCell = param[1].ToString();                _currentRoutine = _pickUpMoveToRoutine;            }            return result;        }        /// <summary>        /// Retry PickUpMoveTo        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool RetryPickUpMoveTo(object[] param)        {            if (!CheckOtherEntityStatus(_sourceCell))            {                LOG.WriteLog(eEvent.WARN_TRANSPORTER, Module.ToString(), $"Module Axis meets conflict of other Axis from {_sourceCell}");                return false;            }            if (!CheckOtherEntityStatus(_targetCell))            {                LOG.WriteLog(eEvent.WARN_TRANSPORTER, Module.ToString(), $"Module Axis meets conflict of other Axis to {_targetCell}");                return false;            }            int stepIndex = (int)param[0];            bool result = _pickUpMoveToRoutine.Retry(stepIndex) == RState.Running;            if (result)            {                _currentRoutine = _pickUpMoveToRoutine;                _sourceCell = _pickUpMoveToRoutine.SourceCell;                _targetCell= _pickUpMoveToRoutine.TargetCell;            }            return result;        }        /// <summary>        /// PickUpMoveTo超时        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool PickUpMoveToTimeout(object[] param)        {            RState ret = _pickUpMoveToRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                _targetCell = "";                _sourceCell = "";                _currentRoutine = null;                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmList alarmList = new AlarmList(Module.ToString(), ((TransporterState)fsm.State).ToString(), (int)TransporterMSG.PickUpMoveTo,                    _pickUpMoveToRoutine.ErrorMsg, _pickUpMoveToRoutine.ErrorStep, (int)AlarmType.Error);                    AlarmListManager.Instance.AddAlarm(alarmList);                }                PostMsg(TransporterMSG.Error);                return false;            }            bool result = ret == RState.End;            if (result)            {                _targetCell = "";                _sourceCell = "";                _currentRoutine = null;                AlarmListManager.Instance.CheckModuleAlamAndRemove(Module.ToString(), TransporterState.PickUpMoveToing.ToString());            }            return result;        }        /// <summary>        /// 确认PickupMoveto是否完成        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ConfirmPickupMoveto(object[] param)        {            int stepIdex = (int)param[0];            bool result = _pickUpMoveToRoutine.CheckCompleteCondition(stepIdex);            if (!result)            {                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmList alarmList = new AlarmList(Module.ToString(), ((TransporterState)fsm.State).ToString(), (int)TransporterMSG.PickUpMoveTo,                    _pickUpMoveToRoutine.ErrorMsg, _pickUpMoveToRoutine.ErrorStep, (int)AlarmType.Error);                    AlarmListManager.Instance.AddAlarm(alarmList);                }                PostMsg(TransporterMSG.Error);            }            else            {                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmListManager.Instance.CheckModuleAlamAndRemove(Module.ToString(), TransporterState.PickUpMoveToing.ToString());                }            }            return result;        }        #endregion        #region PickUpValidate        /// <summary>        /// PickUpValidate        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool PickUpValidate(object[] param)        {            if (!CheckOtherEntityStatus(param[0].ToString()))            {                LOG.WriteLog(eEvent.WARN_TRANSPORTER, Module.ToString(), $"Module Axis meets conflict of other Axis from {param[0]}");                return false;            }            bool result = _pickUpValidateRoutine.Start(param) == RState.Running;            if (result)            {                _sourceCell = param[0].ToString();                _currentRoutine = _pickUpValidateRoutine;            }            return result;        }        /// <summary>        /// Retry PickUpValidate        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool RetryPickUpValidate(object[] param)        {            if (!CheckOtherEntityStatus(_sourceCell))            {                LOG.WriteLog(eEvent.WARN_TRANSPORTER, Module.ToString(), $"Module Axis meets conflict of other Axis from {_sourceCell}");                return false;            }            int stepIndex = (int)param[0];            bool result = _pickUpValidateRoutine.Retry(stepIndex) == RState.Running;            if (result)            {                _currentRoutine = _pickUpValidateRoutine;                _sourceCell = _pickUpValidateRoutine.SourceCell;            }            return result;        }        /// <summary>        /// PickUpValidate超时        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool PickUpValidateTimeout(object[] param)        {            RState ret = _pickUpValidateRoutine.Monitor();            if (ret == RState.Failed || ret == RState.Timeout)            {                _sourceCell = "";                _currentRoutine = null;                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmList alarmList = new AlarmList(Module.ToString(), ((TransporterState)fsm.State).ToString(), (int)TransporterMSG.PickUpValidate,                    _pickUpValidateRoutine.ErrorMsg, _pickUpValidateRoutine.ErrorStep, (int)AlarmType.Error);                    AlarmListManager.Instance.AddAlarm(alarmList);                }                PostMsg(TransporterMSG.Error);                return false;            }            bool result = ret == RState.End;            if (result)            {                _currentRoutine = null;                AlarmListManager.Instance.CheckModuleAlamAndRemove(Module.ToString(), TransporterState.PickUpValidating.ToString());                //Routine完成,但transporter却没有WaferHolder,则表示此次取WH失败,transporter返回Idle,便于调度重新选择下一个WaferHolder                if (WaferHolderInfo == null)                {                    PostMsg(TransporterMSG.ReturnIdle);                    return false;                }            }            return result;        }        /// <summary>        /// 确认PickupValidate是否完成        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ConfirmPickupValidate(object[] param)        {            int stepIdex = (int)param[0];            bool result = _pickUpValidateRoutine.CheckCompleteCondition(stepIdex);            if (!result)            {                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmList alarmList = new AlarmList(Module.ToString(), ((TransporterState)fsm.State).ToString(), (int)TransporterMSG.PickUpValidate,                    _pickUpValidateRoutine.ErrorMsg, _pickUpValidateRoutine.ErrorStep, (int)AlarmType.Error);                    AlarmListManager.Instance.AddAlarm(alarmList);                }                PostMsg(TransporterMSG.Error);            }            else            {                if (Singleton<RouteManager>.Instance.IsAutoRunning)                {                    AlarmListManager.Instance.CheckModuleAlamAndRemove(Module.ToString(), TransporterState.PickUpValidating.ToString());                }            }            return result;        }        #endregion        #region TransporterRetry        /// <summary>        /// Retry        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool TransporterRetry(object[] param)        {            AlarmList alarmList = AlarmListManager.Instance.GetAlarmListByModule(Module.ToString());            if (alarmList != null)            {                CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), alarmList.ModuleCmd,                    alarmList.ModuleStep);            }            return false;        }        #endregion        #region ConfirmComplete        /// <summary>        /// 确认是否完成        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ConfirmComplete(object[] param)        {            AlarmList alarmList = AlarmListManager.Instance.GetAlarmListByModule(Module.ToString());            if (alarmList != null)            {                if (alarmList.ModuleState == TransporterState.Transfering.ToString())                {                    CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.Transfer, alarmList.ModuleStep);                }                else if (alarmList.ModuleState == TransporterState.PickUpMoveToing.ToString())                {                    CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.PickUpMoveTo, alarmList.ModuleStep);                }                else if (alarmList.ModuleState == TransporterState.Placing.ToString())                {                    CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.Place, alarmList.ModuleStep);                }                else if (alarmList.ModuleState == TransporterState.MovingTo.ToString())                {                    CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.MoveTo, alarmList.ModuleStep);                }                else                {                    PostMsg(TransporterMSG.Error);                }            }            return false;        }        /// <summary>        /// 清除报警        /// </summary>        /// <param name="param"></param>        /// <returns></returns>        private bool ClearModuleAlarm(object[] param)        {            AlarmList alarmList = AlarmListManager.Instance.GetAlarmListByModule(Module.ToString());            if (alarmList != null)            {                AlarmListManager.Instance.CheckModuleAlamAndRemove(Module.ToString(), "");            }            return true;        }        #endregion        #region switchWafer        /// <summary>        /// 交换Wafer        /// </summary>        /// <param name="from"></param>        /// <param name="to"></param>        public void SwitchWafer(string from,string to)        {            ModuleName fromModuleName = ModuleName.Unknown;            if (from == "Loader")            {                fromModuleName = ModuleName.Loader1;            }            else            {                fromModuleName = (ModuleName)Enum.Parse(typeof(ModuleName), from);            }            ModuleName toModuleName = ModuleName.Unknown;            if (to == "Loader")            {                toModuleName = ModuleName.Loader1;            }            else            {                toModuleName = (ModuleName)Enum.Parse(typeof(ModuleName), to);            }            bool reverse = false;            JetAxisBase loaderRotation = DEVICE.GetDevice<JetAxisBase>($"{ModuleName.Loader1}.Rotation");            if (fromModuleName == ModuleName.Loader1)            {                if (loaderRotation.CheckPositionInStationIgnoreWaferSize(loaderRotation.MotionData.MotorPosition, TRNPB))                {                    reverse = true;                }            }            if (toModuleName == ModuleName.Loader1)            {                if (loaderRotation.CheckPositionInStationIgnoreWaferSize(loaderRotation.MotionData.MotorPosition, TRNPB))                {                    reverse = true;                }            }            if (WaferManager.Instance.CheckHasWafer(fromModuleName, 0))            {                if (!reverse)                {                    WaferManager.Instance.WaferMoved(fromModuleName, 0, toModuleName, 0);                }                else                {                    WaferInfo waferInfo = WaferManager.Instance.GetWafer(fromModuleName, 0);                    WaferHolderInfo info = WaferHolderManager.Instance.GetWaferHolder(to);                    info.WaferBId = waferInfo.WaferID;                    info.WaferBType =(int)waferInfo.WaferType;                    WaferManager.Instance.WaferMoved(fromModuleName, 0, toModuleName, 1);                }            }            if (WaferManager.Instance.CheckHasWafer(fromModuleName, 1))            {                if (!reverse)                {                    WaferManager.Instance.WaferMoved(fromModuleName, 1, toModuleName, 1);                }                else                {                    WaferInfo waferInfo = WaferManager.Instance.GetWafer(fromModuleName, 1);                    WaferHolderInfo info = WaferHolderManager.Instance.GetWaferHolder(to);                    info.WaferAId = waferInfo.WaferID;                    info.WaferAType = (int)waferInfo.WaferType;                    WaferManager.Instance.WaferMoved(fromModuleName, 1, toModuleName, 0);                }            }            MaterialTrackerManager.Instance.UpdateModuleMaterial(to);        }        #endregion        public bool Check(int msg, out string reason, params object[] args)        {            reason = "";            return false;        }        public bool CheckAcked(int msg)        {            return false;        }        public int Invoke(string function, params object[] args)        {            switch (function)            {                case "HomeAll":                    if(IsIdle)                    {                        return (int)FSM_MSG.NONE;                    }                    if (CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.HomeAll))                    {                        return (int)FSM_MSG.NONE;                    }                    else                    {                        return (int)FSM_MSG.ALARM;                    }                case "Abort":                    CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.Abort);                    return (int)FSM_MSG.NONE;                case "Retry":                    if (CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.Retry, args))                    {                        return (int)TransporterMSG.Retry;                    }                    else                    {                        return (int)FSM_MSG.NONE;                    }                case "ConfirmComplete":                    if (CheckToPostMessage<TransporterState, TransporterMSG>(eEvent.ERR_TRANSPORTER, Module.ToString(), (int)TransporterMSG.ConfirmComplete, args))                    {                        return (int)TransporterMSG.ConfirmComplete;                    }                    else                    {                        return (int)FSM_MSG.NONE;                    }            }            return (int)FSM_MSG.NONE;        }    }    public enum TransporterMSG    {        HomeAll,            // 0        SwitchOn,        SwitchOff,        Error,        ResumeError,        ReturnIdle,        Abort,        GantryGoToSavedPosition,        GoToSavedPosition,        PickUpFrom,        MoveTo,        Place,        Park,        ElevatorUp,        ElevatorLow,        Transfer,        PickUpMoveTo,        PickUpValidate,        GantrySafeMove,        Retry,        ConfirmComplete,        Flip    }}
 |