using System; using System.Threading; using System.Threading.Tasks; using Aitex.Core.RT.DataCenter; using Aitex.Core.RT.Device; using Aitex.Core.RT.Event; using Aitex.Core.RT.Fsm; using Aitex.Core.RT.OperationCenter; using Aitex.Core.RT.Routine; using Aitex.Core.Utilities; using Aitex.Sorter.Common; using MECF.Framework.Common.Equipment; using MECF.Framework.Common.Schedulers; using MECF.Framework.RT.EquipmentLibrary.HardwareUnits.Aligners.HiWinAligner; using SicModules.Aligners.Routines; namespace SicModules.Aligners { public class AlignerModule : AlignerModuleBase { public enum STATE { NotInstall, Init, Idle, Homing, Aligning, Error, } public enum MSG { Home, Reset, Abort, Error, Aligner, SetOnline, SetOffline, ToInit, }; public bool IsAlarm { get { return FsmState == (int)STATE.Error; } } public override bool IsReady { get { if(_alignerDevice != null) { return FsmState == (int)STATE.Idle && CheckAllMessageProcessed() && !_alignerDevice.IsBusy; } else { return FsmState == (int)STATE.Idle && CheckAllMessageProcessed(); } } } public override bool IsError { get { return FsmState == (int)STATE.Error; } } public override bool IsInit { get { return FsmState == (int)STATE.Init; } } public bool IsBusy { get { return !IsInit && !IsError && !IsReady; } } public override bool IsIdle { get { return FsmState == (int)STATE.Idle && CheckAllMessageProcessed(); } } private bool _isInit; public HwAlignerGuide _alignerDevice { get; set; } private AlignerAlignRoutine _alignerAlignRoutine; private AlignerHomeRoutine _alignerHomeRoutine; public AlignerModule(ModuleName module) : base(1) { Module = module.ToString(); Name = module.ToString(); IsOnline = false; EnumLoop.ForEach((item) => { MapState((int)item, item.ToString()); }); EnumLoop.ForEach((item) => { MapMessage((int)item, item.ToString()); }); EnableFsm(50, IsInstalled ? STATE.Init : STATE.NotInstall); } public override bool Initialize() { InitRoutine(); InitDevice(); InitFsm(); InitOp(); InitData(); return base.Initialize(); } private void InitRoutine() { ModuleName module = ModuleHelper.Converter(Module); _alignerAlignRoutine = new AlignerAlignRoutine(); _alignerHomeRoutine = new AlignerHomeRoutine(); } private void InitDevice() { _alignerDevice = DEVICE.GetDevice($"Aligner.HiWinAligner"); } private void InitFsm() { //Error AnyStateTransition(MSG.Error, FsmOnError, STATE.Error); AnyStateTransition(FSM_MSG.ALARM, FsmOnError, STATE.Error); Transition(STATE.Error, MSG.Reset, FsmReset, STATE.Idle); EnterExitTransition(STATE.Error, FsmEnterError, FSM_MSG.NONE, FsmExitError); //Home Transition(STATE.Init, MSG.Home, FsmStartHome, STATE.Homing); Transition(STATE.Error, MSG.Home, FsmStartHome, STATE.Homing); Transition(STATE.Idle, MSG.Home, FsmStartHome, STATE.Homing); Transition(STATE.Homing, FSM_MSG.TIMER, FsmMonitorHomeTask, STATE.Idle); Transition(STATE.Homing, MSG.Error, null, STATE.Init); Transition(STATE.Homing, MSG.Abort, FsmAbortTask, STATE.Init); EnterExitTransition((int)STATE.Homing, FsmEnterIdle, (int)FSM_MSG.NONE, FsmExitIdle); AnyStateTransition(MSG.ToInit, FsmToInit, STATE.Init); //Online Transition(STATE.Idle, MSG.SetOnline, FsmStartSetOnline, STATE.Idle); Transition(STATE.Idle, MSG.SetOffline, FsmStartSetOffline, STATE.Idle); //Aligner Transition(STATE.Idle, MSG.Aligner, FsmStartAligner, STATE.Aligning); Transition(STATE.Aligning, FSM_MSG.TIMER, FsmMonitorTask, STATE.Idle); Transition(STATE.Aligning, MSG.Abort, FsmAbortTask, STATE.Idle); } private void InitOp() { OP.Subscribe($"{Name}.Home", (string cmd, object[] args) => CheckToPostMessage((int)MSG.Home)); OP.Subscribe($"{Name}.Reset", (string cmd, object[] args) => CheckToPostMessage((int)MSG.Reset)); OP.Subscribe($"{Name}.Abort", (string cmd, object[] args) => CheckToPostMessage((int)MSG.Abort)); OP.Subscribe($"{Module}.SetOnline", (string cmd, object[] args) => PutOnline()); OP.Subscribe($"{Module}.SetOffline", (string cmd, object[] args) => PutOffline()); OP.Subscribe($"{Name}.Aligner", (string cmd, object[] args) => CheckToPostMessage((int)MSG.Aligner)); } private void InitData() { DATA.Subscribe($"{Name}.Status", () => StringFsmStatus); DATA.Subscribe($"{Name}.IsOnline", () => IsOnline); DATA.Subscribe($"{Name}.IsBusy", () => IsBusy); DATA.Subscribe($"{Module}.IsAlarm", () => IsAlarm); } private bool FsmOnError(object[] param) { IsOnline = false; if (FsmState == (int)STATE.Error) { return false; } if (FsmState == (int)STATE.Init) return false; return true; } private bool FsmReset(object[] param) { if (FsmState == (int)STATE.Error) { _alignerDevice?.Reset(); EV.ClearAlarmEvent(); } if (!_isInit) { PostMsg(MSG.ToInit); return false; } return true; } private bool FsmExitError(object[] param) { return true; } private bool FsmEnterError(object[] param) { InvokeOnEnterError(Module); return true; } private bool FsmStartHome(object[] param) { Result ret = StartRoutine(_alignerHomeRoutine); if (ret == Result.FAIL || ret == Result.DONE) return false; return ret == Result.RUN; } private bool FsmMonitorHomeTask(object[] param) { Result ret = MonitorRoutine(); if (ret == Result.FAIL) { PostMsg(MSG.Error); return false; } if (ret == Result.DONE) { _isInit = true; return true; } return false; } private bool FsmStartAligner(object[] param) { Result ret = StartRoutine(_alignerAlignRoutine); if (ret == Result.FAIL || ret == Result.DONE) return false; return ret == Result.RUN; } private bool FsmAbortTask(object[] param) { AbortRoutine(); return true; } private bool FsmExitIdle(object[] param) { return true; } private bool FsmEnterIdle(object[] param) { return true; } private bool FsmToInit(object[] param) { return true; } private bool FsmStartSetOffline(object[] param) { IsOnline = false; return true; } private bool FsmStartSetOnline(object[] param) { IsOnline = true; return true; } private bool FsmMonitorTask(object[] param) { Result ret = MonitorRoutine(); if (ret == Result.FAIL) { PostMsg(MSG.Error); return false; } return ret == Result.DONE; } public override bool Home(out string reason) { CheckToPostMessage((int)MSG.Home); reason = string.Empty; return true; } public override bool CheckAcked(int entityTaskToken) { return FsmState == (int)STATE.Idle && CheckAllMessageProcessed(); //return true; } public override void NoteTransferStart(ModuleName robot, Hand blade, int targetSlot, EnumTransferType transferType) { //CheckToPostMessage(MSG.InTransfer); } public override void NoteTransferStop(ModuleName robot, Hand blade, int targetSlot, EnumTransferType transferType) { //if (FsmState == (int)STATE.InTransfer) // CheckToPostMessage(MSG.TransferComplete); } public override bool PrepareTransfer(ModuleName robot, Hand blade, int targetSlot, EnumTransferType transferType, out string reason) { reason = string.Empty; return true; } public override bool TransferHandoff(ModuleName robot, Hand blade, int targetSlot, EnumTransferType transferType, out string reason) { reason = string.Empty; return true; } public override bool PostTransfer(ModuleName robot, Hand blade, int targetSlot, EnumTransferType transferType, out string reason) { reason = string.Empty; return true; } public override bool CheckReadyForTransfer(ModuleName robot, Hand blade, int targetSlot, EnumTransferType transferType, out string reason) { reason = string.Empty; return true; } public override bool CheckReadyForMap(ModuleName robot, Hand blade, out string reason) { reason = string.Empty; return true; } private bool PutOnline() { IsOnline = true; NotifyModuleOfflineCancellation.Cancel(); return true; } private bool PutOffline() { IsOnline = false; NotifyModuleOfflineCancellation?.Dispose(); NotifyModuleOfflineCancellation = new CancellationTokenSource(); NotifyModuleOfflineTask(5 * 60 * 1000);//5min return true; } public void InvokeOffline() { PutOffline(); //PostMsg((int)MSG.SetOffline); } public void InvokeOnline() { PutOnline(); //PostMsg((int)MSG.SetOnline); } private CancellationTokenSource NotifyModuleOfflineCancellation = new CancellationTokenSource(); private Task NotifyModuleOfflineTask(int delayTime) { var t1 = Task.Run(async () => { await Task.Delay(delayTime, NotifyModuleOfflineCancellation.Token).ContinueWith(x => EV.PostWarningLog(Module, $"{Module} is still in offline"), NotifyModuleOfflineCancellation.Token); }); return t1; } public override int InvokeAligner() { if (CheckToPostMessage((int)MSG.Aligner)) return (int)MSG.Aligner; return (int)FSM_MSG.NONE; } public int InvokeError() { if (CheckToPostMessage((int)MSG.Error)) return (int)MSG.Error; return (int)FSM_MSG.NONE; } } }