| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367 | using System;using System.Xml;using Aitex.Core.Common.DeviceData;using Aitex.Core.RT.DataCenter;using Aitex.Core.RT.Event;using Aitex.Core.RT.IOCore;using Aitex.Core.RT.OperationCenter;using Aitex.Core.RT.SCCore;using Aitex.Core.RT.Tolerance;using Aitex.Core.Util;using MECF.Framework.RT.EquipmentLibrary.HardwareUnits.MFCs;namespace Aitex.Core.RT.Device.Unit{    public class IoMfc : BaseDevice, IDevice, IMfc    {        public string Unit        {            get; set;        }        [Subscription(AITMfcDataPropertyName.Scale)]        public double Scale        {            get            {                if (_scN2Scale == null || _scScaleFactor == null)                    return 0;                return _scN2Scale.DoubleValue * _scScaleFactor.DoubleValue;            }        }        [Subscription(AITMfcDataPropertyName.SetPoint)]        public double SetPoint        {            get            {                if (_aoFlow != null)                {                    return _aoFlow.Value;                }                return 0;            }            set            {                if (_aoFlow != null)                {                    _aoFlow.Value = (short)value;                }            }        }        [Subscription(AITMfcDataPropertyName.DefaultSetPoint)]        public double DefaultSetPoint        {            get            {                if (_scDefaultSetPoint != null)                    return _scDefaultSetPoint.DoubleValue;                return 0;            }        }        [Subscription(AITMfcDataPropertyName.FeedBack)]        public double FeedBack        {            get            {                if (_aiFlow != null)                    return (_scRegulationFactor!=null && _scRegulationFactor.DoubleValue > 0.001) ?  _aiFlow.Value / _scRegulationFactor.DoubleValue : _aiFlow.Value;                return 0;            }        }        [Subscription(AITMfcDataPropertyName.IsOutOfTolerance)]        public bool IsOutOfTolerance        {            get            {                return _toleranceChecker.Result;             }        }        [Subscription(AITMfcDataPropertyName.IsEnableAlarm)]        public bool EnableAlarm        {            get            {                if (_scEnableAlarm != null)                    return _scEnableAlarm.BoolValue;                return false;            }        }        [Subscription(AITMfcDataPropertyName.AlarmRange)]        public double AlarmRange        {            get            {                if (_scAlarmRange != null)                    return _scAlarmRange.DoubleValue;                return 0;            }        }        [Subscription(AITMfcDataPropertyName.AlarmTime)]        public double AlarmTime        {            get            {                if (_scAlarmTime != null)                    return _scAlarmTime.DoubleValue;                return 0;            }        }         [Subscription(AITMfcDataPropertyName.IsOffline)]        public bool IsOffline        {            get            {                if (_diOffline != null)                    return _diOffline.Value;                return false;            }        }        public string DisplayName        {            get            {                if (_scGasName != null)                    return _scGasName.StringValue;                return Display;            }        }        private AITMfcData DeviceData        {            get            {                AITMfcData data = new AITMfcData()                {                    UniqueName = _uniqueName,                    Type = "MFC",                    DeviceName = Name,                    DeviceSchematicId = DeviceID,                    DisplayName = DisplayName,                    FeedBack = FeedBack,                    SetPoint = SetPoint,                    Scale = Scale,                    IsOffline = IsOffline,                };                return data;            }        }        private DeviceTimer rampTimer = new DeviceTimer();        private double rampTarget;        private double rampInitValue;          private int rampTime;        private ToleranceChecker _toleranceChecker = new ToleranceChecker();        private AIAccessor _aiFlow;        private AOAccessor _aoFlow;        private AOAccessor _aoRange;        private DIAccessor _diOffline;        private SCConfigItem _scGasName;        private SCConfigItem _scEnable;        private SCConfigItem _scN2Scale;        private SCConfigItem _scScaleFactor;        private SCConfigItem _scAlarmRange;        private SCConfigItem _scEnableAlarm;        private SCConfigItem _scAlarmTime;        private SCConfigItem _scDefaultSetPoint;        private SCConfigItem _scRegulationFactor;         private R_TRIG _trigOffline = new R_TRIG();        private string _uniqueName;        public IoMfc(string module, XmlElement node, string ioModule="")         {              Unit = node.GetAttribute("unit");            base.Module = string.IsNullOrEmpty(node.GetAttribute("module")) ?  module :  node.GetAttribute("module");            base.Name = node.GetAttribute("id");            base.Display = node.GetAttribute("display");            base.DeviceID = node.GetAttribute("schematicId");                      _aoRange = ParseAoNode("aoRange", node, ioModule);            _diOffline = ParseDiNode("diOffline", node, ioModule);            _aiFlow = ParseAiNode("aiFlow", node, ioModule);            _aoFlow = ParseAoNode("aoFlow", node, ioModule);            _scGasName = ParseScNode("scGasName",node);            _scEnable = ParseScNode("scEnable",node);            _scN2Scale = ParseScNode("scN2Scale",node, ioModule, $"{Module}.{Name}.N2Scale");             _scScaleFactor = ParseScNode("scScaleFactor",node, ioModule, $"{Module}.{Name}.ScaleFactor");            _scAlarmRange = ParseScNode("scAlarmRange",node, ioModule, $"{Module}.{Name}.AlarmRange");            _scEnableAlarm = ParseScNode("scEnableAlarm",node, ioModule, $"{Module}.{Name}.EnableAlarm");            _scAlarmTime = ParseScNode("scAlarmTime",node, ioModule, $"{Module}.{Name}.AlarmTime");            _scDefaultSetPoint = ParseScNode("scDefaultSetPoint",node);             _scRegulationFactor = ParseScNode("scFlowRegulationFactor",node, ioModule, $"{Module}.{Name}.RegulationFactor");            _uniqueName = $"{Module}.{Name}";        }        public bool Initialize()        {            DATA.Subscribe($"{_uniqueName}.DeviceData", () => DeviceData);            DATA.Subscribe($"Device.{Module}.{Name}", () => DeviceData);            OP.Subscribe($"{_uniqueName}.{AITMfcOperation.Ramp}", InvokeRamp);            DEVICE.Register(String.Format("{0}.{1}", Name, AITMfcOperation.Ramp), (out string reason, int time, object[] param) =>            {                double target = Convert.ToDouble((string)param[0]);                target = Math.Min(target, Scale);                target = Math.Max(target, 0);                Ramp(target, time);                reason = string.Format("{0} ramp to {1}{2}", Display, target, Unit);                return true;            });            //@AAA use recipe            DEVICE.Register(String.Format("{0}", Name), (out string reason, int time, object[] param) =>            {                double target = Convert.ToDouble((string)param[0]);                target = Math.Min(target, Scale);                target = Math.Max(target, 0);                Ramp(target, time);                reason = string.Format("{0} ramp to {1}{2}", Display, target, Unit);                return true;            });            return true;        }        private bool InvokeRamp(string method, object[] args)        {            double target = Convert.ToDouble((string)(args[0].ToString()));            target = Math.Min(target, Scale);            target = Math.Max(target, 0);            int time = 0;            if (args.Length >= 2)                time  = Convert.ToInt32((string)(args[1].ToString()));            Ramp(target, time);              EV.PostInfoLog(Module, $"{_uniqueName} ramp to {target}{Unit} in {time} seconds");            return true;        }        public void Monitor()        {             Ramping();            CheckTolerance();            if (_aoRange != null)                _aoRange.Value = (short)Scale;            _trigOffline.CLK = IsOffline;            if (_trigOffline.Q)            {                EV.PostMessage(Module, EventEnum.DefaultAlarm, string.Format("{0} is offline", DisplayName));            }        }        public void Reset()        {            _toleranceChecker.Reset(AlarmTime);            _trigOffline.RST = true;        }         public void Terminate()        {            Ramp(DefaultSetPoint, 0);        }        public void Ramp(int time)        {            Ramp(DefaultSetPoint, time);        }        public bool Ramp(double flowSetPoint, int time, out string reason)        {            if (HasAlarm)            {                reason = $"{DisplayName} in error status, can not flow";                return false;            }            if (flowSetPoint < 0 || flowSetPoint > Scale)            {                reason = $"{DisplayName} range is [0, {Scale}], can not flow {flowSetPoint}";                return false;            }            EV.PostInfoLog(Module, $"{DisplayName} flow {flowSetPoint} {Unit} in {time} seconds");            Ramp(flowSetPoint, time);            reason = string.Empty;            return true;        }        public void Ramp(double target, int time)        {            target = Math.Max(0, target);            target = Math.Min(Scale, target);            rampInitValue = SetPoint;    //ramp 初始值取当前设定值,而非实际读取值.零漂问题            rampTime = time;            rampTarget = target;            rampTimer.Start(rampTime);        }        public void StopRamp()        {            Ramp(SetPoint, 0);        }        private void Ramping()        {            if (rampTimer.IsTimeout() || rampTime == 0)            {                SetPoint = rampTarget;            }            else            {                SetPoint = rampInitValue + (rampTarget - rampInitValue) * rampTimer.GetElapseTime() / rampTime;            }        }        private void CheckTolerance()        {            if (!EnableAlarm)                return;            _toleranceChecker.Monitor(FeedBack,  (SetPoint-Math.Abs(AlarmRange)), (SetPoint+Math.Abs(AlarmRange)), AlarmTime);            if (_toleranceChecker.Trig)            {                EV.PostMessage(Module, EventEnum.ToleranceAlarm, Module, Display,                                       String.Format("Out of range in {0} seconds", AlarmTime.ToString("0")));            }        }     }}
 |