431 lines
12 KiB
C#
431 lines
12 KiB
C#
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.HwAligner;
|
|
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<STATE>.ForEach((item) =>
|
|
{
|
|
MapState((int)item, item.ToString());
|
|
});
|
|
|
|
EnumLoop<MSG>.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<HwAlignerGuide>($"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, FSM_MSG>(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;
|
|
}
|
|
}
|
|
}
|