| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803 | using System;using System.Xml;using Aitex.Core.RT.DataCenter;using Aitex.Core.RT.Event;using Aitex.Core.RT.IOCore;using Aitex.Core.RT.Log;using Aitex.Core.RT.OperationCenter;using Aitex.Core.RT.SCCore;using Aitex.Core.Util;using MECF.Framework.Common.Equipment;using MECF.Framework.Common.SubstrateTrackings;using MECF.Framework.RT.EquipmentLibrary.HardwareUnits.Flipper.FlipperBase;namespace Aitex.Core.RT.Device.Unit{    /// <summary>    ///     /// </summary>    public class IoTurnOver : FlipperBaseDevice    {        private readonly DIAccessor _di0Degree;        private readonly DIAccessor _di180Degree;        private readonly DIAccessor _diAlarm;        private readonly DIAccessor _diBusy;        private readonly DIAccessor _diGrip;        private readonly DIAccessor _diOrigin;        private readonly DIAccessor _diPlacement;        private readonly DIAccessor _diPressureError;        private readonly DIAccessor _diUngrip;        private readonly DOAccessor _doGrip;        private readonly DOAccessor _doM0;        private readonly DOAccessor _doM1;        private readonly DOAccessor _doOrigin;        private readonly DOAccessor _doResetError;        private readonly DOAccessor _doStop;        private readonly DOAccessor _doUngrip;        private DateTime _dtActionStart;                //private DeviceTimer _loopTimeout = new DeviceTimer();        //private readonly DeviceTimer _loopTimer = new DeviceTimer();        //private readonly SCConfigItem _scLoopInterval;                //private TurnOverState _state = TurnOverState.Idle;        //public TurnOverState State => _state;                //private R_TRIG _trigCloseError = new R_TRIG();        //private R_TRIG _trigOpenError = new R_TRIG();        //private R_TRIG _trigReset = new R_TRIG();        //private R_TRIG _trigSetPointDone = new R_TRIG();                        // DI_TurnOverPlacement        // DI_TurnOverGrip        // DI_TurnOverUngrip        // DI_TurnOverBusy        // DI_TurnOverPressureError        //        // DI_TurnOverAlarm        // DI_TurnOverOrigin        // DI_TurnOver0Degree        // DI_TurnOver180Degree        //        // DO_TurnOverGrip        // DO_TurnOverUngrip        // DO_TurnOverM0        // DO_TurnOverM1        // DO_TurnOverStop        //        // DO_TurnOverOrigin        // DO_TurnOverResetError        public string Display, DeviceID;        private PeriodicJob _thread;        public IoTurnOver(string module, XmlElement node, string ioModule = ""):base(node.GetAttribute("module"), node.GetAttribute("id"))        {            Module = node.GetAttribute("module");            Name = node.GetAttribute("id");            Display = node.GetAttribute("display");            DeviceID = node.GetAttribute("schematicId");            _diPlacement = ParseDiNode("DI_TurnOverPlacement", node, ioModule);            _diGrip = ParseDiNode("DI_TurnOverGrip", node, ioModule);            _diUngrip = ParseDiNode("DI_TurnOverUngrip", node, ioModule);            _diBusy = ParseDiNode("DI_TurnOverBusy", node, ioModule);            _diPressureError = ParseDiNode("DI_TurnOverPressureError", node, ioModule);            _diAlarm = ParseDiNode("DI_TurnOverAlarm", node, ioModule);            _diOrigin = ParseDiNode("DI_TurnOverOrigin", node, ioModule);            _di0Degree = ParseDiNode("DI_TurnOver0Degree", node, ioModule);            _di180Degree = ParseDiNode("DI_TurnOver180Degree", node, ioModule);            _doGrip = ParseDoNode("DO_TurnOverGrip", node, ioModule);            _doUngrip = ParseDoNode("DO_TurnOverUngrip", node, ioModule);            _doM0 = ParseDoNode("DO_TurnOverM0", node, ioModule);            _doM1 = ParseDoNode("DO_TurnOverM1", node, ioModule);            _doStop = ParseDoNode("DO_TurnOverStop", node, ioModule);            _doOrigin = ParseDoNode("DO_TurnOverOrigin", node, ioModule);            _doResetError = ParseDoNode("DO_TurnOverResetError", node, ioModule);            //_scLoopInterval = SC.GetConfigItem("Turnover.IntervalTimeLimit");                        InitializeIoTurnover();        }                       public override bool IsPlacement        {            get            {                if(_diPlacement !=null) return !_diPlacement.Value;                return WaferManager.Instance.CheckHasWafer(ModuleName.TurnOverStation, 0);            }        }                public bool IsAlarm => !_diAlarm.Value;        public bool IsPressureError => _diPressureError ==null? false: !_diPressureError.Value;        public override FlipperPosEnum CurrentFlipperPosition         {            get            {                if (_di0Degree.Value && !_di180Degree.Value) return FlipperPosEnum.FrontSide;                if (!_di0Degree.Value && _di180Degree.Value) return FlipperPosEnum.BackSide;                return FlipperPosEnum.Unknow;            }                      }        public override GripPosEnum CurrentGripperPositon         {             get             {                if (_diGrip.Value && !_diUngrip.Value) return GripPosEnum.Close;                if (!_diGrip.Value && _diUngrip.Value) return GripPosEnum.Open;                return GripPosEnum.Unknow;            }        }        //public bool IsBusy => _diBusy.Value || _state != TurnOverState.Idle;        //public bool IsIdle => !_diBusy.Value && _state == TurnOverState.Idle;        //public bool IsGrip => _diGrip.Value && !_diUngrip.Value;        //public bool IsUnGrip => _diUngrip.Value && !_diGrip.Value;        //public bool Is0Degree => _di0Degree.Value &&!_di180Degree.Value;        //public bool Is180Degree => _di180Degree.Value&&!_di0Degree.Value;        //public bool IsEnableWaferTransfer => Is0Degree && !IsBusy && IsUnGrip && !IsAlarm &&        //                                     IsPlacement;        public bool InitializeIoTurnover()        {            //DATA.Subscribe($"{Module}.{Name}.State", () => _state.ToString());            DATA.Subscribe($"{Module}.{Name}.IsHomed", () => _diOrigin.Value);            DATA.Subscribe($"{Module}.{Name}.IsBusy", () => _diBusy.Value);            DATA.Subscribe($"{Module}.{Name}.IsGrip", () => _diGrip.Value);            DATA.Subscribe($"{Module}.{Name}.IsUnGrip", () => _diUngrip.Value);            DATA.Subscribe($"{Module}.{Name}.Is0Degree", () => _di0Degree.Value);            DATA.Subscribe($"{Module}.{Name}.Is180Degree", () => _di180Degree.Value);            DATA.Subscribe($"{Module}.{Name}.IsPlacement", () => IsPlacement);            DATA.Subscribe($"{Module}.{Name}.IsAlarm", () => _diAlarm.Value);            DATA.Subscribe($"{Module}.{Name}.IsPressureError", () => _diPressureError==null?false: _diPressureError.Value);            DATA.Subscribe($"{Module}.{Name}.GripCmd", () => _doGrip.Value);            DATA.Subscribe($"{Module}.{Name}.TurnTo0Cmd", () => _doM0.Value);            DATA.Subscribe($"{Module}.{Name}.TurnTo180Cmd", () => _doM1.Value);            DATA.Subscribe($"{Module}.{Name}.HomeCmd", () => _doOrigin.Value);            DATA.Subscribe($"{Module}.{Name}.ResetCmd", () => _doResetError.Value);            DATA.Subscribe($"{Module}.{Name}.StopCmd", () => _doStop==null?false: _doStop.Value);            DATA.Subscribe($"{Module}.{Name}.UnGripCmd", () => _doUngrip.Value);                        _trigError = new R_TRIG();            _thread = new PeriodicJob(50, OnTimerMonitor, $"{Module}.{Name} MonitorHandler", true);            return true;        }        private R_TRIG _trigError;        private bool OnTimerMonitor()        {            _trigError.CLK = (_diAlarm != null && !_diAlarm.Value) || (_diPressureError != null && !_diPressureError.Value);            if(_trigError.Q)            {                OnError($"{FlipperModuleName}Error");            }            return true;        }        public override void Terminate()        {        }        //public void Monitor()        //{        //    if (IsAlarm || IsPressureError)        //    {        //        _state = TurnOverState.Error;        //        _doGrip.SetValue(false, out _);        //        _doM0.SetValue(false, out _);        //        _doM1.SetValue(false, out _);        //        _doOrigin.SetValue(false, out _);        //        if(_doStop !=null)        //            _doStop.SetValue(false, out _);        //        _doUngrip.SetValue(false, out _);        //        return;        //    }        //    switch (_state)        //    {        //        case TurnOverState.OnHoming:        //            _doM0.SetValue(false, out _);        //            _doM1.SetValue(false, out _);        //            _doOrigin.SetValue(true, out _);        //            if (IsHomed)// && _di0Degree.Value)        //            {        //                if (!_doOrigin.SetValue(false, out var reason))        //                    LOG.Error($"{Module} reset DO failed, {reason}");        //                _state = TurnOverState.Idle;        //            }        //            else if (_loopTimer.IsTimeout())        //            {        //                if (!_doOrigin.SetValue(false, out var reason))        //                    LOG.Error($"{Module} reset DO failed, {reason}");        //                EV.PostAlarmLog(Module, $"{Module} {Name} Can not Home in {_scLoopInterval.IntValue} seconds");        //                _state = TurnOverState.Error;        //            }        //            break;        //        case TurnOverState.OnGripping:        //            if (IsGrip)        //            {        //                //if (!_doGrip.SetValue(false, out var reason)) LOG.Error($"{Module} reset DO failed, {reason}");        //                _state = TurnOverState.Idle;        //            }        //            else if (_loopTimer.IsTimeout())        //            {        //                //if (!_doOrigin.SetValue(false, out var reason))        //                //    LOG.Error($"{Module} reset DO failed, {reason}");        //                EV.PostAlarmLog(Module, $"{Module} {Name} Can not Grip in {_scLoopInterval.IntValue} seconds");        //                _state = TurnOverState.Error;        //            }        //            break;        //        case TurnOverState.OnUnGripping:        //            if (IsUnGrip)        //            {        //                //if (!_doUngrip.SetValue(false, out var reason))        //                //    LOG.Error($"{Module} reset DO failed, {reason}");        //                _state = TurnOverState.Idle;        //            }        //            else if (_loopTimer.IsTimeout())        //            {        //                //if (!_doUngrip.SetValue(false, out var reason))        //                //    LOG.Error($"{Module} reset DO failed, {reason}");        //                EV.PostAlarmLog(Module,        //                    $"{Module} {Name} Can not UnGrip in {_scLoopInterval.IntValue} seconds");        //                _state = TurnOverState.Error;        //            }        //            break;        //        case TurnOverState.OnTurningTo0:        //            if (_doM1.Value == true) _doM1.SetValue(false, out _);        //            if (_doM0.Value == false) _doM0.SetValue(true, out _);        //            if (Is0Degree && !_diBusy.Value)        //            {    if (!_doM0.SetValue(false, out var reason)) LOG.Error($"{Module} reset DO failed, {reason}");        //                _state = TurnOverState.Idle;        //            }        //            else if (_loopTimer.IsTimeout())        //            {        //                ////if (!_doM0.SetValue(false, out var reason))        //                //    LOG.Error($"{Module} reset DO failed, {reason}");        //                EV.PostAlarmLog(Module,        //                    $"{Module} {Name} Can not Turn to 0 in {_scLoopInterval.IntValue} seconds");        //                _state = TurnOverState.Error;        //            }        //            break;        //        case TurnOverState.OnTurningTo180:        //            if(_doM1.Value == false) _doM1.SetValue(true, out _);        //            if (_doM0.Value == true) _doM0.SetValue(false, out _);        //            if (Is180Degree &&!_diBusy.Value)        //            {        //                if (!_doM1.SetValue(false, out var reason)) LOG.Error($"{Module} reset DO failed, {reason}");        //                _state = TurnOverState.Idle;        //                var wafer = WaferManager.Instance.GetWafer(ModuleName.TurnOverStation, 0);        //                if (!wafer.IsEmpty)        //                {        //                    var dvid = new SerializableDictionary<string, string>()        //                    {        //                        {"LOT_ID", wafer.LotId},        //                        {"WAFER_ID", wafer.WaferID},        //                        {"ARRIVE_POS_NAME", "TRN1"},        //                    };        //                    EV.Notify(EventWaferTurnOverEnd, dvid);        //                }                                                        //            }        //            else if (_loopTimer.IsTimeout())        //            {        //                //if (!_doM1.SetValue(false, out var reason)) LOG.Error($"{Module} reset DO failed, {reason}");        //                EV.PostAlarmLog(Module,        //                    $"{Module} {Name} Can not Turn to 180 in {_scLoopInterval.IntValue} seconds");        //                _state = TurnOverState.Error;        //            }        //            break;        //        case TurnOverState.OnErrorCleaning:        //            if (!IsAlarm)        //            {        //                if (!_doResetError.SetValue(false, out var reason))        //                    LOG.Error($"{Module} reset DO failed, {reason}");        //                _state = TurnOverState.Idle;        //            }        //            break;        //        case TurnOverState.Stopping:        //            if (!IsBusy)        //            {        //                if (_doStop != null)        //                {        //                    if (!_doStop.SetValue(false, out var reason))        //                        LOG.Error($"{Module} reset DO failed, {reason}");        //                }        //                _state = TurnOverState.Stop;        //            }        //            break;        //        default:        //            //if (!_diBusy.Value && !_doStop.Value) _state = DeviceState.Idle;        //            break;        //    }                                               //}        //public bool Home(out string reason)        //{        //    reason = string.Empty;        //    EV.PostInfoLog(Module, $"{Module}.{Name} Home");        //    //if (IsPressureError)        //    //{        //    //    reason = "Turn over station pressure error";        //    //    return false;        //    //}        //    _doM0.SetValue(false, out _);        //    _doM1.SetValue(false, out _);        //    _doOrigin.SetValue(true, out _);        //    _state = TurnOverState.OnHoming;        //    _loopTimer.Start(_scLoopInterval.IntValue * 1000);        //    return true;        //}        //public bool Grip(out string reason)        //{        //    reason = string.Empty;        //    EV.PostInfoLog(Module, $"{Module}.{Name} Grip");        //    if (IsPressureError)        //    {        //        reason = "Turn over station pressure error";        //        EV.PostAlarmLog(Module, $"{Module}.{Name} pressure error.");        //        return false;                        //    }        //    //if (_doGrip.Value)        //    //{        //    //    reason = "Gripping, can't do again";        //    //    EV.PostAlarmLog(Module, $"{Module}.{Name} {reason}.");        //    //    return false;        //    //}        //    _doUngrip.SetValue(false, out _);        //    _doGrip.SetValue(true, out _);        //    _state = TurnOverState.OnGripping;        //    _loopTimer.Start(_scLoopInterval.IntValue * 1000);        //    return true;        //}        //public bool UnGrip(out string reason)        //{        //    reason = string.Empty;        //    EV.PostInfoLog(Module, $"{Module}.{Name} UnGrip");        //    if (IsPressureError)        //    {        //        reason = "Turn over station pressure error";        //        return false;        //    }        //    //if (_doUngrip.Value)        //    //{        //    //    reason = "UnGripping, can't do again";        //    //    return false;        //    //}        //    _doGrip.SetValue(false, out _);        //    _doUngrip.SetValue(true, out _);        //    _state = TurnOverState.OnUnGripping;        //    _loopTimer.Start(_scLoopInterval.IntValue * 1000);        //    return true;        //}        //public bool TurnTo0(out string reason)        //{        //    reason = string.Empty;        //    EV.PostInfoLog(Module, $"{Module}.{Name} Turn to 0");        //    //if (IsPressureError)        //    //{        //    //    reason = "Turn over station pressure error";        //    //    return false;        //    //}        //    //if (_doM0.Value)        //    //{        //    //    reason = "Turning, can't do again";        //    //    return false;        //    //}        //    _state = TurnOverState.OnTurningTo0;        //    //if (_doM1.Value == true) _doM1.SetValue(false, out _);        //    //if (_doM0.Value == false) _doM0.SetValue(true, out _);        //    _loopTimer.Start(_scLoopInterval.IntValue * 1000);        //    return true;        //}        //public bool TurnTo180(out string reason)        //{        //    reason = string.Empty;        //    EV.PostInfoLog(Module, $"{Module}.{Name} Turn to 180");        //    //if (IsPressureError)        //    //{        //    //    reason = "Turn over station pressure error";        //    //    return false;        //    //}        //    //if (_doM1.Value)        //    //{        //    //    reason = "Turning, can't do again";        //    //    return false;        //    //}        //    _state = TurnOverState.OnTurningTo180;        //    //if (_doM0.Value == true) _doM0.SetValue(false, out _);        //    //if (_doM1.Value == false) _doM1.SetValue(true, out _);        //    _loopTimer.Start(_scLoopInterval.IntValue * 1000);        //    var wafer = WaferManager.Instance.GetWafer(ModuleName.TurnOverStation,0);        //    if (!wafer.IsEmpty)        //    {        //        var dvid = new SerializableDictionary<string, string>()        //        {        //            {"LOT_ID", wafer.LotId},        //            {"WAFER_ID", wafer.WaferID},        //            {"ARRIVE_POS_NAME", "TRN1"}        //        };        //        EV.Notify(EventWaferTurnOverStart, dvid);        //    }                                //    return true;        //}        //public bool Stop(out string reason)        //{        //    reason = string.Empty;        //    EV.PostInfoLog(Module, $"{Module}.{Name} Stop");        //    if (_doStop != null)        //    {        //        if (_doStop.Value)        //        {        //            reason = "Stopping, can't do again";        //            return false;        //        }        //        _doStop.SetValue(true, out _);        //    }        //    _state = TurnOverState.Stopping;        //    _loopTimer.Start(_scLoopInterval.IntValue * 1000);        //    return true;        //}        //private bool ResetError(out string reason)        //{        //    reason = string.Empty;        //    //EV.PostInfoLog(Module, $"{Module}.{Name} Reset Error");        //    if (_doResetError.Value)        //        return true;        //    _doResetError.SetValue(true, out _);        //    _state = TurnOverState.OnErrorCleaning;        //    _loopTimer.Start(_scLoopInterval.IntValue * 1000);        //    return true;        //}        public DOAccessor ParseDoNode(string name, XmlElement node, string ioModule = "")        {            if (!string.IsNullOrEmpty(node.GetAttribute(name).Trim()))                return IO.DO[string.IsNullOrEmpty(ioModule) ? node.GetAttribute(name).Trim() : $"{ioModule}.{node.GetAttribute(name).Trim()}"];            return null;        }        public DIAccessor ParseDiNode(string name, XmlElement node, string ioModule = "")        {            if (!string.IsNullOrEmpty(node.GetAttribute(name).Trim()))                return IO.DI[string.IsNullOrEmpty(ioModule) ? node.GetAttribute(name).Trim() : $"{ioModule}.{node.GetAttribute(name).Trim()}"];            return null;        }        protected override bool fStartGrip(object[] param)        {            _doGrip.SetValue(true, out _);            _doUngrip.SetValue(false, out _);            _dtActionStart = DateTime.Now;            return true;        }        protected override bool fMonitorGrip(object[] param)        {            IsBusy = false;            if (DateTime.Now - _dtActionStart > TimeSpan.FromSeconds(TimelimitAction))            {                OnError("Grip timeout");                return false;            }            if (_diGrip.Value && !_diUngrip.Value &&  !_diBusy.Value)            {                EV.PostInfoLog("Flipper", $"{FlipperModuleName} grip completed");                return true;            }            return false;        }        protected override bool fStartUnGrip(object[] param)        {            _doGrip.SetValue(false, out _);            _doUngrip.SetValue(true, out _);            _dtActionStart = DateTime.Now;            return true;        }        protected override bool fMonitorUnGrip(object[] param)        {            IsBusy = false;            if (DateTime.Now - _dtActionStart > TimeSpan.FromSeconds(TimelimitAction))            {                OnError("UnGrip timeout");                return false;            }            if (!_diGrip.Value && _diUngrip.Value && !_diBusy.Value)            {                EV.PostInfoLog("Flipper", $"{FlipperModuleName} ungrip completed");                return true;            }            return false;        }        protected override bool fStartHome(object[] param)        {            _doM0.SetValue(false, out _);            _doM1.SetValue(false, out _);            _doOrigin.SetValue(true, out _);            _dtActionStart = DateTime.Now;            _dtActionStart = DateTime.Now;            return true;        }        protected override bool fMonitorHome(object[] param)        {            IsBusy = false;            if (DateTime.Now - _dtActionStart > TimeSpan.FromSeconds(TimelimitAction))            {                OnError("Home timeout");                 _doM1.SetValue(false, out _);                _doM0.SetValue(false, out _);                _doOrigin.SetValue(false, out _);                return false;            }            if (_diOrigin.Value && !_diBusy.Value)            {                _doOrigin.SetValue(false, out _);                EV.PostInfoLog("Flipper", $"{FlipperModuleName} home completed");                _doM1.SetValue(false, out _);                _doM0.SetValue(false, out _);                _doOrigin.SetValue(false, out _);                return true;            }            return false;        }        protected override bool fStartTurnTo0(object[] param)        {            _doM0.SetValue(true, out _);            _doM1.SetValue(false, out _);            _dtActionStart = DateTime.Now;            var wafer = WaferManager.Instance.GetWafer(ModuleName.TurnOverStation, 0);            if (!wafer.IsEmpty)            {                var dvid = new SerializableDictionary<string, string>()                    {                        {"LOT_ID", wafer.LotId},                        {"WAFER_ID", wafer.WaferID},                        {"ARRIVE_POS_NAME", FlipperModuleName.ToString()}                    };                EV.Notify(EventWaferTurnOverStart, dvid);            }            return true;        }        protected override bool fMonitorTurnTo0(object[] param)        {            IsBusy = false;            if (DateTime.Now - _dtActionStart > TimeSpan.FromSeconds(TimelimitAction))            {                OnError("Turn to 0 timeout");                _doM1.SetValue(false, out _);                _doM0.SetValue(false, out _);                _doOrigin.SetValue(false, out _);                return false;            }            if (_di0Degree.Value && !_di180Degree.Value && !_diBusy.Value)            {                EV.PostInfoLog("Flipper", $"{FlipperModuleName} turn to 0 degree completed");                var wafer = WaferManager.Instance.GetWafer(ModuleName.TurnOverStation, 0);                if (!wafer.IsEmpty)                {                    var dvid = new SerializableDictionary<string, string>()                    {                        {"LOT_ID", wafer.LotId},                        {"WAFER_ID", wafer.WaferID},                        {"ARRIVE_POS_NAME", FlipperModuleName.ToString()}                    };                    EV.Notify(EventWaferTurnOverStart, dvid);                }                _doM1.SetValue(false, out _);                _doM0.SetValue(false, out _);                _doOrigin.SetValue(false, out _);                return true;            }            return false;        }        protected override bool fStartTurnTo180(object[] param)        {            _doM0.SetValue(false, out _);            _doM1.SetValue(true, out _);            _dtActionStart = DateTime.Now;            var wafer = WaferManager.Instance.GetWafer(ModuleName.TurnOverStation, 0);            if (!wafer.IsEmpty)            {                var dvid = new SerializableDictionary<string, string>()                    {                        {"LOT_ID", wafer.LotId},                        {"WAFER_ID", wafer.WaferID},                        {"ARRIVE_POS_NAME", "TRN1"}                    };                EV.Notify(EventWaferTurnOverStart, dvid);            }            return true;        }               protected override bool fMonitorTurnTo180(object[] param)        {            IsBusy = false;            if (DateTime.Now - _dtActionStart > TimeSpan.FromSeconds(TimelimitAction))            {                OnError("Turn to 180 timeout");                _doM1.SetValue(false, out _);                _doM0.SetValue(false, out _);                _doOrigin.SetValue(false, out _);                return false;            }            if (!_di0Degree.Value && _di180Degree.Value && !_diBusy.Value)            {                EV.PostInfoLog("Flipper", $"{FlipperModuleName} turn to 180 degree completed");                var wafer = WaferManager.Instance.GetWafer(ModuleName.TurnOverStation, 0);                if (!wafer.IsEmpty)                {                    var dvid = new SerializableDictionary<string, string>()                    {                        {"LOT_ID", wafer.LotId},                        {"WAFER_ID", wafer.WaferID},                        {"ARRIVE_POS_NAME", FlipperModuleName.ToString()}                    };                    EV.Notify(EventWaferTurnOverStart, dvid);                }                _doM1.SetValue(false, out _);                _doM0.SetValue(false, out _);                _doOrigin.SetValue(false, out _);                return true;            }            return false;        }        protected override bool fStartAbort(object[] param)        {            _doStop.SetValue(true, out _);            _doResetError.SetValue(false, out _);            _doM0.SetValue(false, out _);            _doM1.SetValue(false, out _);            IsBusy = false;            return true;        }        protected override bool fStartReset(object[] param)        {            _doM0.SetValue(false, out _);            _doM1.SetValue(false, out _);            _doOrigin.SetValue(false, out _);            _doStop.SetValue(false, out _);            _doResetError.SetValue(true, out _);            _dtActionStart = DateTime.Now;            return true;        }        protected override bool fMonitorReset(object[] param)        {            IsBusy = false;            if (DateTime.Now - _dtActionStart > TimeSpan.FromSeconds(TimelimitAction))            {                OnError("Reset timeout");                _doResetError.SetValue(false, out _);                return false;            }            if (_diAlarm != null && !_diAlarm.Value)                 return false;                                    EV.PostInfoLog("Flipper", $"{FlipperModuleName} reset completed");            _doResetError.SetValue(false, out _);            return true;                   }    }    public enum TurnOverState    {        Idle,        OnHoming,        OnTurningTo0,        OnTurningTo180,        OnGripping,        OnUnGripping,        Error,        OnErrorCleaning,        Stopping,        Stop    }}
 |