| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467 | using Aitex.Core.RT.Device;using System;using System.Collections.Generic;using System.Linq;using System.Net.Configuration;using System.Text;using System.Text.RegularExpressions;using System.Threading.Tasks;using Aitex.Core.Common;using Aitex.Core.RT.DataCenter;using Aitex.Core.RT.Event;using Aitex.Core.RT.OperationCenter;using Aitex.Core.RT.SCCore;using Aitex.Core.Util;using Aitex.Sorter.Common;using MECF.Framework.Common.Equipment;using MECF.Framework.RT.Core.Equipments;using MECF.Framework.Common.SubstrateTrackings;using MECF.Framework.RT.EquipmentLibrary.HardwareUnits.LoadPorts.TDK;namespace MECF.Framework.RT.EquipmentLibrary.HardwareUnits.LoadPorts{    public enum IndicatorType    {        Load,        Unload,        Presence,        Placement,        Alarm,        Access,        Status1,        Status2,    }    public interface IE87CallBack    {        void CarrierArrive();        void CarrerRemove(string carrierID);        void CarrierIDReadSuccess(string carrierID);        void CarrierIDReadFail();        void MappingComplete(string carrierID, string slotmap);        void LoadStart();        void LoadportError(string errorcode);        void LoadComplete();        void UnLoadStart();        void UnloadComplete();        void OnLPHomed();        void OnE84HandoffStart(bool isload);        void OnE84HandoffComplete(bool isload);    }    public abstract class LoadPort : BaseDevice, IDevice    {        public IndicatorState IndicatiorLoad { get; set; }        public IndicatorState IndicatiorUnload { get; set; }        public IndicatorState IndicatiorPresence { get; set; }        public IndicatorState IndicatorAlarm { get; set; }        public IndicatorState IndicatiorPlacement { get; set; }        public IndicatorState IndicatiorOpAccess { get; set; }        public IndicatorState IndicatiorStatus1 { get; set; }        public IndicatorState IndicatiorStatus2 { get; set; }        public virtual FoupClampState ClampState { get; set; }        public virtual FoupDoorState DoorState { get; set; }        public virtual CasstleType  CasstleType { get; set; }        public string SlotMap        {            get { return GetSlotMap(); }        }        public virtual bool IsBusy { get; set; }        public virtual bool IsIdle { get; set; }        public virtual bool IsMoving { get; set; }        public virtual LoadportCassetteState CassetteState        {            get;            set;        }        public bool IsMapWaferByLoadPort { get; set; }        public EnumLoadPortType PortType { get; set; }        public bool Initalized { get; set; }        public bool IsPresent        {            get { return _isPresent; }        }        public bool IsPlacement        {            get { return _isPlaced; }        }        public bool IsMapped        {            get { return _isMapped; }        }        public bool Error { get; set; }        public string ErrorCode { get; set; }                /// <summary>        /// 是否处于FOSB模式        /// </summary>        public bool IsFOSBMode { get; set; }                    public DeviceState State        {            get            {                if (!Initalized)                {                    return DeviceState.Unknown;                }                if (Error)                {                    return DeviceState.Error;                }                if (IsBusy)                    return DeviceState.Busy;                return DeviceState.Idle;            }        }        public string CarrierId        {            get { return _carrierId; }        }        public string RfId        {            get { return _rfid; }        }        protected bool _isPresent;        protected bool _isPlaced;        protected string _carrierId;        protected string _rfid;        protected bool _isMapped;        private ModuleName _module;        private ModuleStateEnum _state;        private List<List<string>> _waferId = new List<List<string>>();        private static bool _isIndependentCEID = SC.ContainsItem("Fa.IndependentCEIDForLp")?SC.GetValue<bool>("Fa.IndependentCEIDForLp"):false;        public string PortId        {            get { return (_lpIndex+1).ToString(); }        }        public string PortCategory        {            get            {                if (SC.GetConfigItem(PortLabelScNames[_lpIndex]) == null)                    return "";                return SC.GetStringValue(PortLabelScNames[_lpIndex]) ;            }        }        string[] PortStateVariableNames = new[]        {            "CARRIER_PORT1_STATE", "CARRIER_PORT2_STATE", "CARRIER_PORT3_STATE",            "CARRIER_PORT4_STATE", "CARRIER_PORT5_STATE", "CARRIER_PORT6_STATE", "CARRIER_PORT7_STATE",            "CARRIER_PORT8_STATE", "CARRIER_PORT9_STATE", "CARRIER_PORT10_STATE"        };        ModuleName[] PortModuleNames = new[]        {            ModuleName.LP1,ModuleName.LP2,ModuleName.LP3,ModuleName.LP4,ModuleName.LP5,            ModuleName.LP6,ModuleName.LP7,ModuleName.LP8,ModuleName.LP9,ModuleName.LP10,        };        string[] PortSlotMapVariableNames = new[]        {            "CARRIER_PORT1_CURRENT_SLOTMAP", "CARRIER_PORT2_CURRENT_SLOTMAP", "CARRIER_PORT3_CURRENT_SLOTMAP",            "CARRIER_PORT4_CURRENT_SLOTMAP", "CARRIER_PORT5_CURRENT_SLOTMAP", "CARRIER_PORT6_CURRENT_SLOTMAP", "CARRIER_PORT7_CURRENT_SLOTMAP",            "CARRIER_PORT8_CURRENT_SLOTMAP", "CARRIER_PORT9_CURRENT_SLOTMAP", "CARRIER_PORT10_CURRENT_SLOTMAP"        };        string[] PortWaferIdVariableNames = new[]        {            "CARRIER_PORT1_CURRENT_WAFERID", "CARRIER_PORT2_CURRENT_WAFERID", "CARRIER_PORT3_CURRENT_WAFERID",            "CARRIER_PORT4_CURRENT_WAFERID", "CARRIER_PORT5_CURRENT_WAFERID", "CARRIER_PORT6_CURRENT_WAFERID", "CARRIER_PORT7_CURRENT_WAFERID",            "CARRIER_PORT8_CURRENT_WAFERID", "CARRIER_PORT9_CURRENT_WAFERID", "CARRIER_PORT10_CURRENT_WAFERID"        };        private string[] PortLabelScNames = new[]        {            "LoadPort.LoadPort1CarrierLabel","LoadPort.LoadPort2CarrierLabel","LoadPort.LoadPort3CarrierLabel","LoadPort.LoadPort4CarrierLabel",            "LoadPort.LoadPort5CarrierLabel","LoadPort.LoadPort6CarrierLabel","LoadPort.LoadPort7CarrierLabel","LoadPort.LoadPort8CarrierLabel",            "LoadPort.LoadPort9CarrierLabel","LoadPort.LoadPort10CarrierLabel",        };        private int _lpIndex = -1;        private string EventCarrierArrived= "CARRIER_ARRIVED";        private string EventCarrierIdRead= "CARRIER_ID_READ";        private string EventCarrierIdReadFailed = "CARRIER_ID_READ_FAILED";        private string EventCarrierIdWrite = "CARRIER_ID_WRITE";        private string EventCarrierIdWriteFailed = "CARRIER_ID_WRITE_FAILED";        private string EventSlotMapAvailable= "SLOT_MAP_AVAILABLE";        private string EventCarrierRemoved = "CARRIER_REMOVED";        private string EventCarrierUnloaded = "CARRIER_UNLOADED";        private string EventCarrierloaded = "CARRIR_DOCK_COMPLETE";        private string PORT_ID = "PORT_ID";        private string CAR_ID = "CAR_ID";        private string SLOT_MAP = "SLOT_MAP";        private string PORT_CTGRY = "PORT_CTGRY";        private string RF_ID = "RF_ID";        private string EventRfIdRead = "RF_ID_READ";        private string EventRfIdReadFailed = "RF_ID_READ_FAILED";        private string EventRfIdWrite = "RF_ID_WRITE";#pragma warning disable 414        private string EventRfIdWriteFailed = "RF_ID_WRITE_FAILED";#pragma warning restore 414        private string AlarmCarrierPortError = "CarrierPortError";        public LoadPort(string module, string name):base(module, name, name, "")        {            for (int i = 0; i < 25; i++)            {                _waferId.Add(new List<string>()                {                    i.ToString("D2"),"","",""                });            }                 if (!Enum.TryParse(name, out ModuleName m))                    Enum.TryParse(module, out m);                _module = m;             _lpIndex = Array.IndexOf(PortModuleNames, _module);            DoorState = FoupDoorState.Unknown;            System.Diagnostics.Trace.Assert(_lpIndex != -1);        }        public virtual bool Initialize()        {            WaferManager.Instance.SubscribeLocation(_module, 25);            CarrierManager.Instance.SubscribeLocation(_module.ToString());             DATA.Subscribe(Name, "IsPresent", ()=>_isPresent);            DATA.Subscribe(Name, "IsPlaced", () => _isPlaced);            DATA.Subscribe(Name, "ModuleState", () => _state.ToString());            DATA.Subscribe(Name, "CarrierId", () => _carrierId);            DATA.Subscribe(Name, "IsMapped", () => _isMapped);            DATA.Subscribe($"{Name}.LoadportState", () => State);            DATA.Subscribe($"{Name}.LoadportBusy", () => IsBusy);            DATA.Subscribe($"{Name}.LoadportError", () => ErrorCode);            DATA.Subscribe($"{Name}.CassetteState", () => CassetteState);            DATA.Subscribe($"{Name}.FoupClampState", () => ClampState);            DATA.Subscribe($"{Name}.FoupDoorState", () => DoorState);            DATA.Subscribe($"{Name}.SlotMap", () => SlotMap);            DATA.Subscribe($"{Name}.IndicatiorLoad", () => IndicatiorLoad);            DATA.Subscribe($"{Name}.IndicatiorUnload", () => IndicatiorUnload);            DATA.Subscribe($"{Name}.IndicatiorPresence", () => IndicatiorPresence);            DATA.Subscribe($"{Name}.IndicatiorPlacement", () => IndicatiorPlacement);            DATA.Subscribe($"{Name}.IndicatiorAlarm", () => IndicatorAlarm);            DATA.Subscribe($"{Name}.IndicatiorOpAccess", () => IndicatiorOpAccess);            DATA.Subscribe($"{Name}.IndicatiorStatus1", () => IndicatiorStatus1);            DATA.Subscribe($"{Name}.IndicatiorStatus2", () => IndicatiorStatus2);            DATA.Subscribe($"{Name}.CasstleType", () => CasstleType);            DATA.Subscribe(PortStateVariableNames[_lpIndex], () => (_isPlaced && _isPresent) ? "1" : "0");            DATA.Subscribe(PortSlotMapVariableNames[_lpIndex], () => SlotMap);            DATA.Subscribe(PortWaferIdVariableNames[_lpIndex], UpdatedWaferIdList);            if (_isIndependentCEID)            {                                EV.Subscribe(new EventItem("Event", $"{_module}_{EventCarrierArrived}", "Carrier arrived"));                EV.Subscribe(new EventItem("Event", $"{_module}_{EventCarrierRemoved}", "Carrier removed"));                EV.Subscribe(new EventItem("Event", $"{_module}_{EventCarrierIdRead}", "Carrier ID read"));                EV.Subscribe(new EventItem("Event", $"{_module}_{EventCarrierIdReadFailed}", "Carrier ID read failed"));                EV.Subscribe(new EventItem("Event", $"{_module}_CAR_ID_WRITE_SUCCESSFULLY", "Carrier ID write"));                EV.Subscribe(new EventItem("Event", $"{_module}_CAR_ID_WRITE_FAIL", "Carrier ID write failed"));                EV.Subscribe(new EventItem("Event", $"{_module}_{EventSlotMapAvailable}", "Slot map available"));                EV.Subscribe(new EventItem("Event", $"{_module}_{EventCarrierUnloaded}", "Carrier unloaded"));                EV.Subscribe(new EventItem("Event", $"{_module}_CARRIR_LOAD_COMPLETE", "Carrier loaded"));                EV.Subscribe(new EventItem("Event", $"{_module}_{EventRfIdRead}", "Carrier RFID read"));                EV.Subscribe(new EventItem("Event", $"{_module}_{EventRfIdReadFailed}", "Carrier RFID read failed"));                EV.Subscribe(new EventItem("Event", $"{_module}_{EventRfIdWrite}", "Carrier RFID write"));                EV.Subscribe(new EventItem("Event", $"{_module}_{AlarmCarrierPortError}", "Carrier Port error", EventLevel.Alarm, EventType.HostNotification));            }            else            {                EV.Subscribe(new EventItem("Event", EventCarrierArrived, "Carrier arrived"));                EV.Subscribe(new EventItem("Event", EventCarrierRemoved, "Carrier removed"));                EV.Subscribe(new EventItem("Event", EventCarrierIdRead, "Carrier ID read"));                EV.Subscribe(new EventItem("Event", EventCarrierIdReadFailed, "Carrier ID read failed"));                EV.Subscribe(new EventItem("Event", EventCarrierIdWrite, "Carrier ID write"));                EV.Subscribe(new EventItem("Event", EventCarrierIdWriteFailed, "Carrier ID write failed"));                EV.Subscribe(new EventItem("Event", EventSlotMapAvailable, "Slot map available"));                EV.Subscribe(new EventItem("Event", EventCarrierUnloaded, "Carrier unloaded"));                EV.Subscribe(new EventItem("Event", EventCarrierloaded, "Carrier loaded"));                EV.Subscribe(new EventItem("Event", EventRfIdRead, "Carrier RFID read"));                EV.Subscribe(new EventItem("Event", EventRfIdReadFailed, "Carrier RFID read failed"));                EV.Subscribe(new EventItem("Event", EventRfIdWrite, "Carrier RFID write"));                EV.Subscribe(new EventItem("Event", AlarmCarrierPortError, "Carrier Port error", EventLevel.Alarm, EventType.HostNotification));            }            IsIdle = true;            _state = ModuleStateEnum.Idle;            RegisterOperation();            return true;        }        public virtual bool Connect()        {            return true;        }        private void RegisterOperation()        {            OP.Subscribe($"{Name}.LoadportHome", (string cmd, object[] param) =>            {                if (!Home(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not start home, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start home");                return true;            });            OP.Subscribe($"{Name}.LoadportReset", (string cmd, object[] param) =>            {                if (!ClearError(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not reset, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start reset");                return true;            });            OP.Subscribe($"{Name}.LoadportStop", (string cmd, object[] param) =>            {                if (!Stop(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not stop, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} stop");                return true;            });            OP.Subscribe($"{Name}.LoadportLoad", (string cmd, object[] param) =>            {                if (!Load(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not load, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start load");                return true;            });            OP.Subscribe($"{Name}.LoadportLoadWithoutMap", (string cmd, object[] param) =>            {                if (!LoadWithoutMap(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not load without map, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start load without map");                return true;            });            OP.Subscribe($"{Name}.LoadportLoadWithMap", (string cmd, object[] param) =>            {                if (!Load(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not load with map, {reason}");                    return false;                }                if (!QueryWaferMap(out reason))                {                    EV.PostWarningLog(Module, $"{Name} can not map, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start load with map");                return true;            });            OP.Subscribe($"{Name}.LoadportUnload", (string cmd, object[] param) =>            {                if (!Unload(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not unload, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start unload");                return true;            });            OP.Subscribe($"{Name}.LoadportClamp", (string cmd, object[] param) =>            {                if (!Clamp(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not clamp, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start clamp");                return true;            });            OP.Subscribe($"{Name}.LoadportUnclamp", (string cmd, object[] param) =>            {                if (!Unclamp(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not unclamp, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start unclamp");                return true;            });            OP.Subscribe($"{Name}.LoadportOpenDoor", (string cmd, object[] param) =>            {                if (!OpenDoor(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not open door, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start open door");                return true;            });            OP.Subscribe($"{Name}.LoadportOpenDoorNoMap", (string cmd, object[] param) =>            {                if (!OpenDoorNoMap(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not open door, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start open door");                return true;            });            OP.Subscribe($"{Name}.LoadportCloseDoor", (string cmd, object[] param) =>            {                if (!CloseDoor(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not close door, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start close door");                return true;            });            OP.Subscribe($"{Name}.LoadportDock", (string cmd, object[] param) =>            {                if (!Dock(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not dock, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start dock");                return true;            });            OP.Subscribe($"{Name}.LoadportUndock", (string cmd, object[] param) =>            {                if (!Undock(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not undock, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start undock");                return true;            });            OP.Subscribe($"{Name}.LoadportQueryState", (string cmd, object[] param) =>            {                if (!QueryState(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not query state, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start query state");                return true;            });            OP.Subscribe($"{Name}.LoadportQueryLED", (string cmd, object[] param) =>            {                if (!QueryIndicator(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not query led state, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start query led state");                return true;            });            OP.Subscribe($"{Name}.LoadportSetLED", (string cmd, object[] param) =>            {                int light = (int) param[0];                int state = (int) param[1];                if (!SetIndicator((Indicator) light, (IndicatorState) state, out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not set led state, {reason}");                    return true;                }                EV.PostInfoLog(Module, $"{Name} start set led state");                return true;            });            OP.Subscribe($"{Name}.LoadportMap", (string cmd, object[] param) =>            {                if (!QueryWaferMap(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not map, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start map");                return true;            });                        OP.Subscribe($"{Name}.SetCassetteType", (string cmd, object[] param) =>            {                if (!SetCassetteType(param,out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not set type, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} cassette type have set to {CasstleType}");                return true;            });            OP.Subscribe($"{Name}.LoadportForceHome", (string cmd, object[] param) =>            {                if (!ForceHome(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not start force home, {reason}");                    return false;                }                EV.PostInfoLog(Module, $"{Name} start force home");                return true;            });            OP.Subscribe($"{Name}.LoadportFOSBMode", (string cmd, object[] param) =>            {                if (!FOSBMode(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not change to FOSB mode, {reason}");                    return false;                }                IsFOSBMode = true;                EV.PostInfoLog(Module, $"{Name} changed to FOSB mode");                return true;            });            OP.Subscribe($"{Name}.LoadportFOUPMode", (string cmd, object[] param) =>            {                if (!FOUPMode(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not change to FOSB mode, {reason}");                    return false;                }                IsFOSBMode = true;                EV.PostInfoLog(Module, $"{Name} changed to FOSB mode");                return true;            });            OP.Subscribe($"{Name}.LoadportQueryFOSBMode", (string cmd, object[] param) =>            {                if (!QueryFOSBMode(out string reason))                {                    EV.PostWarningLog(Module, $"{Name} can not change to FOUP mode, {reason}");                    return false;                }                IsFOSBMode = false;                EV.PostInfoLog(Module, $"{Name} changed to FOSB mode");                return true;            });                    }        private bool SetCassetteType(object[] param,out string reason)        {            reason = "";            if (param.Length != 1)            {                reason = "Invalid setting parameter.";                return false;            }            CasstleType = (CasstleType)int.Parse(param[0].ToString());            return true;        }        public virtual bool Load(out string reason)        {            reason = "";            return true;        }        public virtual bool LoadWithoutMap(out string reason)        {            reason = "";            return true;        }        public virtual bool QueryWaferMap(out string reason)        {            reason = "";            return true;        }        public virtual bool QueryFOSBMode(out string reason)        {            reason = "";            return true;        }        /// <summary>        /// FOSB模式下的Dock指令        /// </summary>        /// <param name="reason"></param>        /// <returns></returns>        public virtual bool FOSBDock(out string reason)        {            reason = "";            return true;        }                /// <summary>        /// FOSB模式下的FOSBUnDock指令        /// </summary>        /// <param name="reason"></param>        /// <returns></returns>        public virtual bool FOSBUnDock(out string reason)        {            reason = "";            return true;        }                /// <summary>        /// FOSB模式下的开门指令        /// </summary>        /// <param name="reason"></param>        /// <returns></returns>        public virtual bool FOSBDoorOpen(out string reason)        {            reason = "";            return true;        }                /// <summary>        /// FOSB模式下的关门指令        /// </summary>        /// <param name="reason"></param>        /// <returns></returns>        public virtual bool FOSBDoorClose(out string reason)        {            reason = "";            return true;        }        /// <summary>        /// FOSB模式下的门下移指令        /// </summary>        /// <param name="reason"></param>        /// <returns></returns>        public virtual bool FOSBDoorDown(out string reason)        {            reason = "";            return true;        }        /// <summary>        /// FOSB模式下的门上移指令        /// </summary>        /// <param name="reason"></param>        /// <returns></returns>        public virtual bool FOSBDoorUp(out string reason)        {            reason = "";            return true;        }                public bool SetIndicator(IndicatorType light, IndicatorState state)        {            switch (light)            {                case IndicatorType.Load:                    SetIndicator(Indicator.LOAD, state, out string _);                    break;                case IndicatorType.Unload:                    SetIndicator(Indicator.UNLOAD, state, out string _);                    break;                case IndicatorType.Access:                    SetIndicator(Indicator.OPACCESS, state, out string _);                    break;                case IndicatorType.Alarm:                    SetIndicator(Indicator.ALARM, state, out string _);                    break;                case IndicatorType.Presence:                    SetIndicator(Indicator.PRESENCE, state, out string _);                    break;                case IndicatorType.Placement:                    SetIndicator(Indicator.PLACEMENT, state, out string _);                    break;                case IndicatorType.Status1:                    SetIndicator(Indicator.STATUS1, state, out string _);                    break;                case IndicatorType.Status2:                    SetIndicator(Indicator.STATUS2, state, out string _);                    break;                default:                    EV.PostWarningLog(Module, $"Not supported indicator {light}");                    return false;            }            return true;        }        public virtual bool SetIndicator(Indicator light, IndicatorState state, out string reason)        {            reason = "";            return true;        }        public virtual bool QueryIndicator(out string reason)        {            reason = "";            return true;        }        public virtual bool QueryState(out string reason)        {            reason = "";            return true;        }        public virtual bool Undock(out string reason)        {            reason = "";            return true;        }        public virtual bool Dock(out string reason)        {            reason = "";            return true;        }        public virtual bool CloseDoor(out string reason)        {            reason = "";            return true;        }        public virtual bool OpenDoor(out string reason)        {            reason = "";            return true;        }        public virtual bool OpenDoorNoMap(out string reason)        {            reason = "";            return true;        }        public virtual bool OpenDoorAndMap(out string reason)        {            reason = "";            return true;        }        public virtual bool Unclamp(out string reason)        {            reason = "";            return true;        }        public virtual bool Clamp(out string reason)        {            reason = "";            return true;        }        public virtual bool Unload(out string reason)        {            reason = "";            return true;        }         public virtual bool Stop(out string reason)        {            reason = "";            return true;        }        public virtual bool ClearError(out string reason)        {            reason = "";            return true;        }        public virtual bool Home(out string reason )        {            reason = "";            return true;        }         public virtual bool ForceHome(out string reason )        {            reason = "";            return true;        }        public virtual bool FOSBMode(out string reason)        {            reason = "";            return true;        }                public virtual bool FOUPMode(out string reason)        {            reason = "";            return true;        }        public virtual  void Monitor()        {                     }        public virtual void Reset()        {                    }        public virtual void Terminate()        {                     }        private List<List<string>> UpdatedWaferIdList()        {            WaferInfo[] wafers = WaferManager.Instance.GetWafers(_module);            for (int i = 0; i < wafers.Length; i++)            {                _waferId[i][1] = wafers[i].LaserMarker;                _waferId[i][2] = wafers[i].T7Code;                _waferId[i][3] = wafers[i].WaferID;            }            return _waferId;        }        public abstract bool IsEnableMapWafer();        public abstract bool IsEnableTransferWafer();        public abstract bool IsEnableTransferWafer(out string reason);        public virtual bool IsEnableLoad()        {            return _isPresent && _isPlaced;        }        protected void ConfirmAddCarrier()        {            if (_isPresent && _isPlaced)            {                CarrierManager.Instance.CreateCarrier(Name);                if (_isIndependentCEID)                {                    var dvid1 = new SerializableDictionary<string, string>                    {                        [$"{PORT_ID}_{_module}"] = PortId,                        [$"{PORT_CTGRY}_{_module}"] = PortCategory                    };                    EV.Notify($"{_module}_{EventCarrierArrived}", dvid1);                }                else                {                    SerializableDictionary<string, string> dvid = new SerializableDictionary<string, string>();                    dvid[PORT_ID] = PortId;                    dvid[PORT_CTGRY] = PortCategory;                    EV.Notify(EventCarrierArrived, dvid );                }            }        }        protected void ConfirmRemoveCarrier()        {            if (!_isPresent && !_isPlaced)            {                for (int i = 0; i < 25; i++)                {                    WaferManager.Instance.DeleteWafer(ModuleHelper.Converter(Name), 0,25);                }                CarrierManager.Instance.DeleteCarrier(Name);                if (_isIndependentCEID)                {                    var dvid1 = new SerializableDictionary<string, string>                    {                        [$"{PORT_ID}_{_module}"] = PortId,                        [$"{CAR_ID}_{_module}"] = _carrierId ?? ""                    };                    EV.Notify($"{_module}_{EventCarrierRemoved}", dvid1);                }                else                {                    SerializableDictionary<string, string> dvid = new SerializableDictionary<string, string>();                    dvid[PORT_ID] = PortId;                    dvid[PORT_CTGRY] = PortCategory;                    dvid[CAR_ID] = _carrierId ?? "";                    EV.Notify(EventCarrierRemoved, dvid);                }                                                                _isMapped = false;                _carrierId = "";            }        }        public void OnSlotMapRead(string slotMap)        {            for (int i = 0; i < 25; i++)            {                // No wafer: "0", Wafer: "1", Crossed:"2", Undefined: "?", Overlapping wafers: "W"                WaferInfo wafer = null;                switch (slotMap[i])                {                    case '0':                        WaferManager.Instance.DeleteWafer(_module, i);                        CarrierManager.Instance.UnregisterCarrierWafer(Name, i);                        break;                    case '1':                        wafer = WaferManager.Instance.CreateWafer(_module, i, WaferStatus.Normal);                        CarrierManager.Instance.RegisterCarrierWafer(Name, i, wafer);                        break;                    case '2':                        wafer = WaferManager.Instance.CreateWafer(_module, i, WaferStatus.Crossed);                        CarrierManager.Instance.RegisterCarrierWafer(Name, i, wafer);                        //NotifyWaferError(Name, i, WaferStatus.Crossed);                        break;                    case 'W':                        wafer = WaferManager.Instance.CreateWafer(_module, i, WaferStatus.Double);                        CarrierManager.Instance.RegisterCarrierWafer(Name, i, wafer);                        //NotifyWaferError(Name, i, WaferStatus.Double);                        break;                    case '?':                        wafer = WaferManager.Instance.CreateWafer(_module, i, WaferStatus.Unknown);                        CarrierManager.Instance.RegisterCarrierWafer(Name, i, wafer);                        //NotifyWaferError(Name, i, WaferStatus.Unknown);                        break;                }            }            if (_isIndependentCEID)            {                var dvid1 = new SerializableDictionary<string, string>                {                    [$"{SLOT_MAP}_{_module}"] = SlotMap,                    [$"{PORT_ID}_{_module}"] = PortId,                    [$"{PORT_CTGRY}_{_module}"] = PortCategory,                    [$"{CAR_ID}_{_module}"] = _carrierId ?? ""                };                EV.Notify($"{_module}_{EventSlotMapAvailable}", dvid1);            }            else            {                SerializableDictionary<string, string> dvid = new SerializableDictionary<string, string>();                 dvid[SLOT_MAP] = SlotMap;                dvid[PORT_ID] = PortId;                dvid[PORT_CTGRY] = PortCategory;                dvid[CAR_ID] =  CarrierId == null ? "" : CarrierId;                EV.Notify(EventSlotMapAvailable, dvid);            }            _isMapped = true;        }         private string GetSlotMap()        {            WaferInfo[] wafers = WaferManager.Instance.GetWafers(ModuleHelper.Converter(Name) );            string slot = "";            for (int i = 0; i < 25; i++)            {                slot += wafers[i].IsEmpty? "0" : "1";            }            return slot;        }        /// <summary>        /// 获取LP中空缺Slot        /// </summary>        /// <returns>返回一个list, 顺序为从下到上.(0-25)</returns>        public List<int> GetEmptySlot()        {            List<int> slot = new List<int>();            if (IsMapped)            {                WaferInfo[] wafers = WaferManager.Instance.GetWafers(ModuleHelper.Converter(Name));                for (int i = 0; i < 25 ; i++)                {                    if(wafers[i].IsEmpty)                        slot.Add(i);                }                return slot;            }            else            {                return null;            }                    }        public void OnCarrierIdRead(ModuleName module, string name, string carrierId)        {            if (_isPlaced && _isPresent)            {                _carrierId = carrierId;                if (_isIndependentCEID)                {                    var dvid1 = new SerializableDictionary<string, string>                    {                        [$"{CAR_ID}_{_module}"] = _carrierId ?? "",                        [$"{PORT_ID}_{_module}"] = PortId,                        [$"{PORT_CTGRY}_{_module}"] = PortCategory                    };                    EV.Notify($"{_module}_{EventCarrierIdRead}", dvid1);                }                else                {                    SerializableDictionary<string, string> dvid = new SerializableDictionary<string, string>();                    dvid[CAR_ID] = carrierId ?? "";                    dvid[PORT_ID] = PortId;                    dvid[PORT_CTGRY] = PortCategory;                    EV.Notify(EventCarrierIdRead, dvid);                 }                CarrierManager.Instance.UpdateCarrierId(Name, carrierId);            }            else            {                EV.PostWarningLog(Module, $"No FOUP found, carrier id {carrierId} not saved");            }        }        public void OnCarrierIdWrite(ModuleName module, string name, string carrierId)        {            if (_isPlaced && _isPresent)            {                //_carrierId = carrierId;                if (_isIndependentCEID)                {                    var dvid1 = new SerializableDictionary<string, string>                    {                        [$"{CAR_ID}_{_module}"] = carrierId ?? "",                        [$"{PORT_ID}_{_module}"] = PortId                    };                    EV.Notify($"{_module}_CAR_ID_WRITE_SUCCESSFULLY", dvid1);                }                else                {                    SerializableDictionary<string, string> dvid = new SerializableDictionary<string, string>();                    dvid[CAR_ID] = carrierId ?? "";                    dvid[PORT_ID] = PortId;                    dvid[PORT_CTGRY] = PortCategory;                    EV.Notify(EventCarrierIdWrite, dvid);                }            }            else            {                EV.PostWarningLog(Module, $"No FOUP found, carrier id {carrierId} not saved");            }        }        public void OnCarrierIdReadFailed(ModuleName module, string name)        {            if (_isPlaced && _isPresent)            {                _carrierId = "";                if (_isIndependentCEID)                {                    var dvid1 = new SerializableDictionary<string, string>                    {                        [$"{PORT_ID}_{_module}"] = PortId,                         [$"{PORT_CTGRY}_{_module}"] = PortCategory                    };                    EV.Notify($"{_module}_{EventCarrierIdReadFailed}", dvid1);                }                else                {                    SerializableDictionary<string, string> dvid = new SerializableDictionary<string, string>();                    dvid[PORT_ID] = PortId;                    dvid[PORT_CTGRY] = PortCategory;                    EV.Notify(EventCarrierIdReadFailed, dvid);                }                                            }            else            {                EV.PostWarningLog(Module, "No FOUP found, carrier id read is not valid");            }        }        public void OnCarrierIdWriteFailed(ModuleName module, string name)        {            if (_isPlaced && _isPresent)            {                //_carrierId = "";                if (_isIndependentCEID)                {                    var dvid1 = new SerializableDictionary<string, string>                    {                        [$"{PORT_ID}_{_module}"] = PortId,                        [$"{PORT_CTGRY}_{_module}"] = PortCategory,                        [$"{CAR_ID}_{_module}"] = _carrierId ?? ""                                        };                    EV.Notify($"{_module}_CAR_ID_WRITE_FAIL", dvid1);                }                else                {                    SerializableDictionary<string, string> dvid = new SerializableDictionary<string, string>();                                    dvid[PORT_ID] = PortId;                    dvid[PORT_CTGRY] = PortCategory;                    EV.Notify(EventCarrierIdWriteFailed, dvid);                }            }            else            {                EV.PostWarningLog(Module, "No FOUP found, carrier id not valid");            }        }        public void OnCarrierIdRead(ModuleName module, string carrierId)        {            OnCarrierIdRead(module, "", carrierId);        }        public void OnCarrierIdReadFailed(ModuleName module)        {            OnCarrierIdReadFailed(module, "");        }        public void OnCarrierIdWrite(ModuleName module, string carrierId)        {            OnCarrierIdWrite(module, "", carrierId);        }                public void OnCarrierIdWriteFailed(ModuleName module)        {            OnCarrierIdWriteFailed(module, "");        }        public void OnRfIdRead(ModuleName module, string rfid)        {            if (_isPlaced && _isPresent)            {                _rfid = rfid;                if (_isIndependentCEID)                {                    var dvid1 = new SerializableDictionary<string, string>                    {                        [$"{CAR_ID}_{_module}"] = _carrierId ?? "",                        [$"{PORT_ID}_{_module}"] = PortId,                        [$"{PORT_CTGRY}_{_module}"] = PortCategory,                        [$"{RF_ID}_{_module}"] = rfid ?? ""                    };                    EV.Notify($"{_module}_{EventRfIdRead}", dvid1);                }                else                {                    SerializableDictionary<string, string> dvid = new SerializableDictionary<string, string>();                    dvid[CAR_ID] = _carrierId ?? "";                    dvid[PORT_ID] = PortId;                    dvid[PORT_CTGRY] = PortCategory;                    dvid[RF_ID] = rfid ?? "";                    EV.Notify(EventRfIdRead, dvid);                }                                CarrierManager.Instance.UpdateRfId(Name, rfid);            }            else            {                EV.PostWarningLog(Module, "No FOUP found, rf id read not valid");            }        }        public void OnRfIdReadFailed(ModuleName module)        {            if (_isPlaced && _isPresent)            {                _rfid = "";                if (_isIndependentCEID)                {                    var dvid1 = new SerializableDictionary<string, string>                    {                        [$"{CAR_ID}_{_module}"] = _carrierId ?? "",                        [$"{PORT_ID}_{_module}"] = PortId,                        [$"{PORT_CTGRY}_{_module}"] = PortCategory                    };                    EV.Notify($"{_module}_{EventRfIdReadFailed}", dvid1);                }                else                {                    SerializableDictionary<string, string> dvid = new SerializableDictionary<string, string>();                    dvid[CAR_ID] = _carrierId ?? "";                    dvid[PORT_ID] = PortId;                    dvid[PORT_CTGRY] = PortCategory;                    EV.Notify(EventRfIdReadFailed, dvid);                }            }            else            {                EV.PostWarningLog(Module, "No FOUP found, rf id read is not valid");            }        }        public void OnRfIdWrite(ModuleName module, string rfid)        {            if (_isPlaced && _isPresent)            {                _rfid = rfid;                if (_isIndependentCEID)                {                    var dvid1 = new SerializableDictionary<string, string>                    {                        [$"{CAR_ID}_{_module}"] = _carrierId ?? "",                        [$"{PORT_ID}_{_module}"] = PortId,                        [$"{PORT_CTGRY}_{_module}"] = PortCategory,                        [$"{RF_ID}_{_module}"] = rfid ?? ""                    };                    EV.Notify($"{_module}_{EventRfIdWrite}", dvid1);                }                else                {                    SerializableDictionary<string, string> dvid = new SerializableDictionary<string, string>();                    dvid[CAR_ID] = _carrierId ?? "";                    dvid[PORT_ID] = PortId;                    dvid[PORT_CTGRY] = PortCategory;                    dvid[RF_ID] = rfid ?? "";                    EV.Notify(EventRfIdWrite, dvid);                }                CarrierManager.Instance.UpdateRfId(Name, rfid);            }            else            {                EV.PostWarningLog(Module, "No FOUP found, rf id write not valid");            }        }        public void OnRfIdWriteFailed(ModuleName module )        {            if (_isPlaced && _isPresent)            {                _rfid = "";                if (_isIndependentCEID)                {                    var dvid1 = new SerializableDictionary<string, string>                    {                        [$"{CAR_ID}_{_module}"] = _carrierId ?? "",                        [$"{PORT_ID}_{_module}"] = PortId,                        [$"{PORT_CTGRY}_{_module}"] = PortCategory,                        [$"{RF_ID}_{_module}"] = ""                    };                    EV.Notify($"{_module}_{EventRfIdWrite}", dvid1);                }                else                {                    SerializableDictionary<string, string> dvid = new SerializableDictionary<string, string>();                    dvid[CAR_ID] = _carrierId ?? "";                    dvid[PORT_ID] = PortId;                    dvid[PORT_CTGRY] = PortCategory;                    dvid[RF_ID] = "";                    //EV.PostWarningLog(Module, "Write RFID failed.");                    EV.Notify(EventRfIdWrite, dvid);                }            }            else            {                EV.PostWarningLog(Module, "No FOUP found, rf id write not valid");            }        }        public void OnLoaded()        {            if (_isIndependentCEID)            {                var dvid1 = new SerializableDictionary<string, string>                {                    [$"{CAR_ID}_{_module}"] = _carrierId ?? "",                    [$"{PORT_ID}_{_module}"] = PortId                };                EV.Notify($"{_module}_CARRIR_LOAD_COMPLETE", dvid1);            }            else            {                var dvid = new SerializableDictionary<string, string>                {                    [CAR_ID] = _carrierId ?? "",                     [PORT_ID] = PortId                };                EV.Notify(EventCarrierloaded, dvid);            }        }        public void OnUnloaded()        {            if (_isIndependentCEID)            {                var dvid1 = new SerializableDictionary<string, string>                {                    [$"{PORT_CTGRY}_{_module}"] = PortCategory,                    [$"{PORT_ID}_{_module}"] = PortId,                    [$"{CAR_ID}_{_module}"] = _carrierId ?? "",                    [$"{SLOT_MAP}_{_module}"] = SlotMap,                    [$"LOT_ID_{_module}"] = CarrierManager.Instance.GetLotIdByLoadPort(_module.ToString())                                };                EV.Notify($"{_module}_{EventCarrierUnloaded}", dvid1);            }            else            {                var dvid = new SerializableDictionary<string, string>();                dvid[PORT_CTGRY] = PortCategory;                dvid[PORT_ID] = PortId;                dvid[CAR_ID] = _carrierId ?? "";                EV.Notify(EventCarrierUnloaded, dvid);            }                                    for (int i = 0; i < 25; i++)            {                WaferManager.Instance.DeleteWafer(ModuleHelper.Converter(Name), 0, 25);            }                        _isMapped = false;        }        public void OnHomed()        {            for (int i = 0; i < 25; i++)            {                WaferManager.Instance.DeleteWafer(ModuleHelper.Converter(Name), 0, 25);            }            _isMapped = false;        }        public void OnCloseDoor()        {            for (int i = 0; i < 25; i++)            {                WaferManager.Instance.DeleteWafer(ModuleHelper.Converter(Name), 0, 25);            }            _isMapped = false;        }        public void OnError()        {            EV.Notify(AlarmCarrierPortError);            EV.Notify($"{_module}_{AlarmCarrierPortError}");        }         protected void SetPresent(bool isPresent)        {            _isPresent = isPresent;            if (_isPresent)            {                ConfirmAddCarrier();            }            else            {                ConfirmRemoveCarrier();            }        }        protected void SetPlaced(bool isPlaced)        {            _isPlaced = isPlaced;            if (_isPlaced)            {                ConfirmAddCarrier();            }            else            {                ConfirmRemoveCarrier();            }        }     }}
 |