| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040 | using System;using System.Collections.Generic;using System.Linq;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.Log;using Aitex.Core.RT.OperationCenter;using Aitex.Core.RT.SCCore;using Aitex.Core.Util;using MECF.Framework.RT.Core.IoProviders;namespace MECF.Framework.Common.IOCore{    public class IoManager : Singleton<IoManager>, IIoBuffer    {        private Dictionary<string, DIAccessor> _diMap = new Dictionary<string, DIAccessor>();        private Dictionary<string, DOAccessor> _doMap = new Dictionary<string, DOAccessor>();        private Dictionary<string, AIAccessor> _aiMap = new Dictionary<string, AIAccessor>();        private Dictionary<string, AOAccessor> _aoMap = new Dictionary<string, AOAccessor>();        private Dictionary<string, Dictionary<int, bool[]>> _diBuffer = new Dictionary<string, Dictionary<int, bool[]>>();        private Dictionary<string, Dictionary<int, bool[]>> _doBuffer = new Dictionary<string, Dictionary<int, bool[]>>();        private Dictionary<string, Dictionary<int, short[]>> _aiBuffer = new Dictionary<string, Dictionary<int, short[]>>();        private Dictionary<string, Dictionary<int, short[]>> _aoBuffer = new Dictionary<string, Dictionary<int, short[]>>();        private Dictionary<string, List<DIAccessor>> _diList = new Dictionary<string, List<DIAccessor>>();        private Dictionary<string, List<DOAccessor>> _doList = new Dictionary<string, List<DOAccessor>>();        private Dictionary<string, List<AIAccessor>> _aiList = new Dictionary<string, List<AIAccessor>>();        private Dictionary<string, List<AOAccessor>> _aoList = new Dictionary<string, List<AOAccessor>>();        private Dictionary<string, List<NotifiableIoItem>> _ioItemList = new Dictionary<string, List<NotifiableIoItem>>();         private PeriodicJob _monitorThread;        public void Initialize(string interlockConfigFile)        {            string reason = string.Empty;            if (!InterlockManager.Instance.Initialize(interlockConfigFile, _doMap, _diMap, out reason))            {                throw new Exception(string.Format("interlock define file found error: \r\n {0}", reason));            }            _monitorThread = new PeriodicJob(200, OnTimer, "IO Monitor Thread", true);        }        private bool OnTimer()        {            try            {                InterlockManager.Instance.Monitor();            }            catch (Exception ex)            {                LOG.Write(ex);            }            return true;        }        internal List<Tuple<int, int, string>> GetIONameList(string group, IOType ioType)        {            return null;        }                public bool CanSetDo(string doName, bool onOff, out string reason)        {            return InterlockManager.Instance.CanSetDo(doName, onOff, out reason);        }        public List<DIAccessor> GetDIList(string source)        {            return _diList.ContainsKey(source) ? _diList[source] : null;        }        public List<DOAccessor> GetDOList(string source)        {            return _doList.ContainsKey(source) ? _doList[source] : null;        }        public List<AIAccessor> GetAIList(string source)        {            return _aiList.ContainsKey(source) ? _aiList[source] : null;        }        public List<AOAccessor> GetAOList(string source)        {            return _aoList.ContainsKey(source) ? _aoList[source] : null;        }               public IoManager()        {            OP.Subscribe("System.SetDoValue", InvokeSetDo);            OP.Subscribe("System.SetAoValue", InvokeSetAo);            OP.Subscribe("System.SetAoValue32", InvokeSetAo32);            OP.Subscribe("System.SetDoValueWithPrivoder", InvokeSetDoWithPrivoder);            OP.Subscribe("System.SetAoValueWithPrivoder", InvokeSetAoWithPrivoder);            OP.Subscribe("System.SetAiBuffer", InvokeSetAiBuffer);            OP.Subscribe("System.SetDiBuffer", InvokeSetDiBuffer);        }        private bool InvokeSetDo(string arg1, object[] args)        {            string name = (string) args[0];            bool setpoint = (bool) args[1];            string reason;            if (!CanSetDo(name, setpoint, out reason))            {                EV.PostWarningLog("System", $"Can not set DO {name} to {setpoint}, {reason}");                return false;            }            DOAccessor do1 = GetIO<DOAccessor>(name);            if (do1 == null)            {                EV.PostWarningLog("System", $"Can not set DO {name} to {setpoint}, not defined do");                return false;            }            if (!do1.SetValue(setpoint, out reason))            {                EV.PostWarningLog("System", $"Can not set DO {name} to {setpoint}, {reason}");                return false;            }            EV.PostInfoLog("System", $"Change DO {name} to {setpoint}");            return true;        }        private bool InvokeSetAo(string arg1, object[] args)        {            string name = (string)args[0];            short setpoint = (short)args[1];             AOAccessor io = GetIO<AOAccessor>(name);            if (io == null)            {                EV.PostWarningLog("System", $"Can not set AO {name} to {setpoint}, not defined do");                return false;            }            io.Value = setpoint;            EV.PostInfoLog("System", $"Change AO {name} to {setpoint}");            return true;        }        private bool InvokeSetAo32(string arg1, object[] args)        {            string name = (string)args[0];            float setpoint = (float)args[1];            AOAccessor io = GetIO<AOAccessor>(name);            if (io == null)            {                EV.PostWarningLog("System", $"Can not set AO {name} to {setpoint}, not defined do");                return false;            }            byte[] flow = BitConverter.GetBytes(setpoint);            io.Buffer[io.Index] = BitConverter.ToInt16(flow, 0);            if (io.Index < io.Buffer.Length - 1)            {                io.Buffer[io.Index + 1] = BitConverter.ToInt16(flow, 2);            }             EV.PostInfoLog("System", $"Change AO {name} to {setpoint}");            return true;        }        private bool InvokeSetDoWithPrivoder(string arg1, object[] args)        {            string provider = (string)args[0];            int offset = (int)args[1];            string name = (string)args[2];            bool setpoint = (bool)args[3];            string reason;            if (!CanSetDo(name, setpoint, out reason))            {                EV.PostWarningLog("System", $"Can not set DO {provider}.{name} to {setpoint}, {reason}");                return false;            }            var doList = GetDOList(provider);            if(doList == null)            {                EV.PostWarningLog("System", $"Can not set DO {provider}.{name} to {setpoint}, {reason}");                return false;            }            var doAccessor = doList.FirstOrDefault(x => x.Name == name);            if(doAccessor == default(DOAccessor))            {                EV.PostWarningLog("System", $"Can not set DO {provider}.{name} to {setpoint}, {reason}");                return false;            }            if (!doAccessor.SetValue(setpoint, out reason))            {                EV.PostWarningLog("System", $"Can not set DO {provider}.{name} to {setpoint}, {reason}");                return false;            }            EV.PostInfoLog("System", $"Change DO {provider}.{name} to {setpoint}");            return true;        }        private bool InvokeSetAiBuffer(string arg1, object[] args)        {            string provider = (string)args[0];            int offset = (int)args[1];            short[] buffer = (short [])args[2];            SetAiBuffer(provider, offset, buffer);            return true;        }        private bool InvokeSetDiBuffer(string arg1, object[] args)        {            string provider = (string)args[0];            int offset = (int)args[1];            bool[] buffer = (bool[])args[2];            SetDiBuffer(provider, offset, buffer);            return true;        }        private bool InvokeSetAoWithPrivoder(string arg1, object[] args)        {            string provider = (string)args[0];            int offset = (int)args[1];            string name = (string)args[2];            float setpoint = (float)args[3];            string reason = "";            var aoList = GetAOList(provider);            if (aoList == null)            {                EV.PostWarningLog("System", $"Can not set AO {provider}.{name} to {setpoint}, {reason}");                return false;            }            var aoAccessor = aoList.FirstOrDefault(x => x.Name == name);            if (aoAccessor == default(AOAccessor))            {                EV.PostWarningLog("System", $"Can not set AO {provider}.{name} to {setpoint}, {reason}");                return false;            }            aoAccessor.Value = (short)setpoint;            EV.PostInfoLog("System", $"Change DO {provider}.{name} to {setpoint}");            return true;        }         public T GetIO<T>(string name) where T : class        {            if (typeof(T) == typeof(DIAccessor) && _diMap.ContainsKey(name))            {                return _diMap[name] as T;            }            if (typeof(T) == typeof(DOAccessor) && _doMap.ContainsKey(name))            {                return _doMap[name] as T;            }            if (typeof(T) == typeof(AIAccessor) && _aiMap.ContainsKey(name))            {                return _aiMap[name] as T;            }            if (typeof(T) == typeof(AOAccessor) && _aoMap.ContainsKey(name))            {                return _aoMap[name] as T;            }            return null;        }        public Dictionary<int, bool[]> GetDiBuffer(string source)        {            if (_diBuffer.ContainsKey(source))            {                return _diBuffer[source];            }            return null;        }        public Dictionary<int, bool[]> GetDoBuffer(string source)        {            if (_doBuffer.ContainsKey(source))            {                return _doBuffer[source];            }            return null;        }        public Dictionary<int, short[]> GetAiBuffer(string source)        {            if (_aiBuffer.ContainsKey(source))            {                return _aiBuffer[source];            }            return null;        }        public Dictionary<int, short[]> GetAoBuffer(string source)        {            if (_aoBuffer.ContainsKey(source))            {                return _aoBuffer[source];            }            return null;        }        public void SetDiBuffer(string source, int offset, bool[] buffer)        {            if (_diBuffer.ContainsKey(source) && _diBuffer[source].ContainsKey(offset))            {                for (int i = 0; i < buffer.Length && i < _diBuffer[source][offset].Length; i++)                {                    _diBuffer[source][offset][i] = buffer[i];                }            }        }        public void SetAiBuffer(string source, int offset, short[] buffer, int skipSize = 0)        {            if (_aiBuffer.ContainsKey(source) && _aiBuffer[source].ContainsKey(offset))            {                for (int i = 0; i < buffer.Length && i < _aiBuffer[source][offset].Length; i++)                {                    _aiBuffer[source][offset][i + skipSize] = buffer[i];                }            }        }        public void SetDoBuffer(string source, int offset, bool[] buffer)        {            if (_doBuffer.ContainsKey(source) && _doBuffer[source].ContainsKey(offset))            {                for (int i = 0; i < buffer.Length && i < _doBuffer[source][offset].Length; i++)                {                    _doBuffer[source][offset][i] = buffer[i];                }            }        }        public void SetAoBuffer(string source, int offset, short[] buffer)        {            if (_aoBuffer.ContainsKey(source) && _aoBuffer[source].ContainsKey(offset))            {                for (int i = 0; i < buffer.Length && i < _aoBuffer[source][offset].Length; i++)                {                    _aoBuffer[source][offset][i] = buffer[i];                }            }        }        //spin recipe set        public void SetAoBuffer(string source, int offset, short[] buffer, int bufferStartIndex)        {            if (_aoBuffer.ContainsKey(source) && _aoBuffer[source].ContainsKey(offset) && _aoBuffer[source][offset].Length > bufferStartIndex)            {                for (int i = 0; i < buffer.Length && bufferStartIndex + i < _aoBuffer[source][offset].Length; i++)                {                    _aoBuffer[source][offset][bufferStartIndex + i] = buffer[i];                }            }        }        public void SetBufferBlock(string provider, List<IoBlockItem> lstBlocks)        {            foreach (var ioBlockItem in lstBlocks)            {                switch (ioBlockItem.Type)                {                    case IoType.AI:                        if (!_aiBuffer.ContainsKey(provider))                        {                            _aiBuffer[provider] = new Dictionary<int, short[]>();                        }                        if (!_aiBuffer[provider].ContainsKey(ioBlockItem.Offset))                        {                            _aiBuffer[provider][ioBlockItem.Offset] = new short[ioBlockItem.Size];                        }                        break;                    case IoType.AO:                        if (!_aoBuffer.ContainsKey(provider))                        {                            _aoBuffer[provider] = new Dictionary<int, short[]>();                        }                        if (!_aoBuffer[provider].ContainsKey(ioBlockItem.Offset))                        {                            _aoBuffer[provider][ioBlockItem.Offset] = new short[ioBlockItem.Size];                        }                        break;                    case IoType.DI:                        if (!_diBuffer.ContainsKey(provider))                        {                            _diBuffer[provider] = new Dictionary<int, bool[]>();                        }                        if (!_diBuffer[provider].ContainsKey(ioBlockItem.Offset))                        {                            _diBuffer[provider][ioBlockItem.Offset] = new bool[ioBlockItem.Size];                        }                        break;                    case IoType.DO:                        if (!_doBuffer.ContainsKey(provider))                        {                            _doBuffer[provider] = new Dictionary<int, bool[]>();                        }                        if (!_doBuffer[provider].ContainsKey(ioBlockItem.Offset))                        {                            _doBuffer[provider][ioBlockItem.Offset] = new bool[ioBlockItem.Size];                        }                        break;                }            }        }        List<NotifiableIoItem> SubscribeDiData()        {            List<NotifiableIoItem> result = new List<NotifiableIoItem>();            foreach (var ioDefine in _diMap)            {                NotifiableIoItem di = new NotifiableIoItem()                {                    Address = ioDefine.Value.Addr,                    Name = ioDefine.Value.Name,                    Description = ioDefine.Value.Description,                    Index = ioDefine.Value.Index,                    BoolValue = ioDefine.Value.Value,                    Provider = ioDefine.Value.Provider,                    BlockOffset = ioDefine.Value.BlockOffset,                    BlockIndex = ioDefine.Value.Index,                };                result.Add(di);            }            return result;        }        List<NotifiableIoItem> SubscribeDoData()        {            List<NotifiableIoItem> result = new List<NotifiableIoItem>();            foreach (var ioDefine in _doMap)            {                NotifiableIoItem io = new NotifiableIoItem()                {                    Address = ioDefine.Value.Addr,                    Name = ioDefine.Value.Name,                    Description = ioDefine.Value.Description,                    Index = ioDefine.Value.Index,                    BoolValue = ioDefine.Value.Value,                    Provider = ioDefine.Value.Provider,                    BlockOffset = ioDefine.Value.BlockOffset,                    BlockIndex = ioDefine.Value.Index,                };                result.Add(io);            }            return result;        }        List<NotifiableIoItem> SubscribeAiData()        {            List<NotifiableIoItem> result = new List<NotifiableIoItem>();            foreach (var ioDefine in _aiMap)            {                NotifiableIoItem di = new NotifiableIoItem()                {                    Address = ioDefine.Value.Addr,                    Name = ioDefine.Value.Name,                    Description = ioDefine.Value.Description,                    Index = ioDefine.Value.Index,                    ShortValue = ioDefine.Value.Value,                    Provider = ioDefine.Value.Provider,                    BlockOffset = ioDefine.Value.BlockOffset,                    BlockIndex = ioDefine.Value.Index,                };                result.Add(di);            }            return result;        }        List<NotifiableIoItem> SubscribeAoData()        {            List<NotifiableIoItem> result = new List<NotifiableIoItem>();            foreach (var ioDefine in _aoMap)            {                NotifiableIoItem ao = new NotifiableIoItem()                {                    Address = ioDefine.Value.Addr,                    Name = ioDefine.Value.Name,                    Description = ioDefine.Value.Description,                    Index = ioDefine.Value.Index,                    ShortValue = ioDefine.Value.Value,                    Provider = ioDefine.Value.Provider,                    BlockOffset = ioDefine.Value.BlockOffset,                    BlockIndex = ioDefine.Value.Index,                };                result.Add(ao);            }            return result;        }         public void SetIoMap(string provider, int blockOffset, List<DIAccessor> ioList)        {            SubscribeIoItemList(provider);            var scConfig = SC.GetConfigItem("System.IsIgnoreSaveDB");            var isIgnoreSaveDB = scConfig != null && scConfig.BoolValue;            foreach (var accessor in ioList)            {                accessor.Provider = provider;                accessor.BlockOffset = blockOffset;                _diMap[accessor.Name] = accessor;                if (!_diList.ContainsKey(provider))                    _diList[provider] = new List<DIAccessor>();                _diList[provider].Add(accessor);                _ioItemList[$"{provider}.DIItemList"].Add(new NotifiableIoItem()                {                    Address = accessor.Addr,                    Name = accessor.Name,                    Description = accessor.Description,                    Index = accessor.Index,                    Provider = provider,                    BlockOffset = blockOffset,                    BlockIndex = accessor.BlockOffset,                });                if (!isIgnoreSaveDB)                    DATA.Subscribe($"IO.{accessor.Name}", () => accessor.Value);            }        }        public void SetIoMap(string provider, int blockOffset, List<DOAccessor> ioList)        {            SubscribeIoItemList(provider);            var scConfig = SC.GetConfigItem("System.IsIgnoreSaveDB");            var isIgnoreSaveDB = scConfig != null && scConfig.BoolValue;            foreach (var accessor in ioList)            {                accessor.Provider = provider;                accessor.BlockOffset = blockOffset;                _doMap[accessor.Name] = accessor;                if (!_doList.ContainsKey(provider))                    _doList[provider] = new List<DOAccessor>();                _doList[provider].Add(accessor);                _ioItemList[$"{provider}.DOItemList"].Add(new NotifiableIoItem()                {                    Address = accessor.Addr,                    Name = accessor.Name,                    Description = accessor.Description,                    Index = accessor.Index,                    Provider = provider,                    BlockOffset = blockOffset,                    BlockIndex = accessor.BlockOffset,                });                if (!isIgnoreSaveDB)                    DATA.Subscribe($"IO.{accessor.Name}", () => accessor.Value);            }        }        public void SetIoMap(string provider, int blockOffset, List<AIAccessor> ioList)        {            SubscribeIoItemList(provider);            var scConfig = SC.GetConfigItem("System.IsIgnoreSaveDB");            var isIgnoreSaveDB = scConfig != null && scConfig.BoolValue;            foreach (var accessor in ioList)            {                accessor.Provider = provider;                accessor.BlockOffset = blockOffset;                _aiMap[accessor.Name] = accessor;                if (!_aiList.ContainsKey(provider))                    _aiList[provider] = new List<AIAccessor>();                _aiList[provider].Add(accessor);                _ioItemList[$"{provider}.AIItemList"].Add(new NotifiableIoItem()                {                    Address = accessor.Addr,                    Name = accessor.Name,                    Description = accessor.Description,                    Index = accessor.Index,                    Provider = provider,                    BlockOffset = blockOffset,                    BlockIndex = accessor.BlockOffset,                });                if (!isIgnoreSaveDB)                {                    //DATA.Subscribe($"IO.{accessor.Name}", () => accessor.Value);                    DATA.Subscribe($"IO32.{accessor.Name}", () =>                    {                        if (accessor.Index < accessor.Buffer.Length - 1)                        {                            byte[] high = BitConverter.GetBytes(accessor.Buffer[accessor.Index]);                            byte[] low = BitConverter.GetBytes(accessor.Buffer[accessor.Index + 1]);                            return BitConverter.ToSingle(new[] { high[0], high[1], low[0], low[1] }, 0);                        }                        else                        {                            return accessor.Value;                        }                    });                }            }        }        public void SetIoMap(string provider, int blockOffset, List<AOAccessor> ioList)        {            SubscribeIoItemList(provider);            var scConfig = SC.GetConfigItem("System.IsIgnoreSaveDB");            var isIgnoreSaveDB = scConfig != null && scConfig.BoolValue;            foreach (var accessor in ioList)            {                accessor.Provider = provider;                accessor.BlockOffset = blockOffset;                _aoMap[accessor.Name] = accessor;                if (!_aoList.ContainsKey(provider))                    _aoList[provider] = new List<AOAccessor>();                _aoList[provider].Add(accessor);                _ioItemList[$"{provider}.AOItemList"].Add(new NotifiableIoItem()                {                    Address = accessor.Addr,                    Name = accessor.Name,                    Description = accessor.Description,                    Index = accessor.Index,                    Provider = provider,                    BlockOffset = blockOffset,                    BlockIndex = accessor.BlockOffset,                });                if (!isIgnoreSaveDB)                {                    //DATA.Subscribe($"IO.{accessor.Name}", () => accessor.Value);                    DATA.Subscribe($"IO32.{accessor.Name}", () =>                    {                        if (accessor.Index < accessor.Buffer.Length - 1)                        {                            byte[] high = BitConverter.GetBytes(accessor.Buffer[accessor.Index]);                            byte[] low = BitConverter.GetBytes(accessor.Buffer[accessor.Index + 1]);                            return BitConverter.ToSingle(new[] { high[0], high[1], low[0], low[1] }, 0);                        }                        else                        {                            return accessor.Value;                        }                    });                }            }        }        public void SetIoMap(string provider, int blockOffset, string xmlPathFile, string module="")        {            SubscribeIoItemList(provider);            XmlDocument xml = new XmlDocument();            xml.Load(xmlPathFile);             XmlNodeList lstDi = xml.SelectNodes("IO_DEFINE/Dig_In/DI_ITEM");            var scConfig = SC.GetConfigItem("System.IsIgnoreSaveDB");            var isIgnoreSaveDB = scConfig == null ? false : scConfig.BoolValue;            //<DI_ITEM Index="0" Name="" BufferOffset="0" Addr="0" Description=""/>            List<DIAccessor> diList = new List<DIAccessor>();            foreach (var diItem in lstDi)            {                XmlElement element = diItem as XmlElement;                if (element == null)                    continue;                string index = element.GetAttribute("Index");                string bufferOffset = element.GetAttribute("BufferOffset");                if (string.IsNullOrEmpty(bufferOffset))                    bufferOffset = index;                string name = element.GetAttribute("Name");                string address = element.GetAttribute("Addr");                string description = element.GetAttribute("Description");                if (string.IsNullOrEmpty(name) || string.IsNullOrEmpty(index) || string.IsNullOrEmpty(bufferOffset))                    continue;                name = name.Trim();                index = index.Trim();                bufferOffset = bufferOffset.Trim();                string moduleName = string.IsNullOrEmpty(module) ? name : $"{module}.{name}";                int intIndex;                if (!int.TryParse(index, out intIndex))                    continue;                int intBufferOffset;                if (!int.TryParse(bufferOffset, out intBufferOffset))                    continue;                if (!_diBuffer.ContainsKey(provider) || !_diBuffer[provider].ContainsKey(blockOffset))                {                    throw new Exception("Not defined DI buffer from IO provider, " + provider);                }                DIAccessor diAccessor = new DIAccessor(moduleName, intBufferOffset, _diBuffer[provider][blockOffset], _diBuffer[provider][blockOffset]);                diAccessor.IoTableIndex = intIndex;                diAccessor.Addr = address;                diAccessor.Provider = provider;                diAccessor.BlockOffset = blockOffset;                diAccessor.Description = description;                diList.Add(diAccessor);                _diMap[moduleName] = diAccessor;                if (!_diList.ContainsKey(provider))                    _diList[provider] = new List<DIAccessor>();                _diList[provider].Add(diAccessor);                _ioItemList[$"{provider}.DIItemList"].Add(new NotifiableIoItem()                {                    Address = address,                    Name = moduleName,                    Description = description,                    Index = intIndex,                    Provider = provider,                    BlockOffset = blockOffset,                    BlockIndex = intIndex,                });                if (!isIgnoreSaveDB)                    DATA.Subscribe($"IO.{moduleName}", () => diAccessor.Value);            }            XmlNodeList lstDo = xml.SelectNodes("IO_DEFINE/Dig_Out/DO_ITEM");            //    < DO_ITEM Index = "0" BufferOffset="0" Name = "" Addr = "0" Description = "" />            foreach (var doItem in lstDo)            {                XmlElement element = doItem as XmlElement;                if (element == null)                    continue;                string index = element.GetAttribute("Index");                string bufferOffset = element.GetAttribute("BufferOffset");                if (string.IsNullOrEmpty(bufferOffset))                    bufferOffset = index;                string name = element.GetAttribute("Name");                string address = element.GetAttribute("Addr");                string description = element.GetAttribute("Description");                if (string.IsNullOrEmpty(name) || string.IsNullOrEmpty(index) || string.IsNullOrEmpty(bufferOffset))                    continue;                name = name.Trim();                index = index.Trim();                bufferOffset = bufferOffset.Trim();                string moduleName = string.IsNullOrEmpty(module) ? name : $"{module}.{name}";                int intIndex;                if (!int.TryParse(index, out intIndex))                    continue;                int intBufferOffset;                if (!int.TryParse(bufferOffset, out intBufferOffset))                    continue;                if (!_doBuffer.ContainsKey(provider) || !_doBuffer[provider].ContainsKey(blockOffset))                {                    throw new Exception("Not defined DO buffer from IO provider, " + provider);                }                DOAccessor doAccessor = new DOAccessor(moduleName, intBufferOffset, _doBuffer[provider][blockOffset]);                _doMap[moduleName] = doAccessor;                doAccessor.IoTableIndex = intIndex;                doAccessor.Addr = address;                doAccessor.Provider = provider;                doAccessor.BlockOffset = blockOffset;                doAccessor.Description = description;                if (!_doList.ContainsKey(provider))                    _doList[provider] = new List<DOAccessor>();                _doList[provider].Add(doAccessor);                _ioItemList[$"{provider}.DOItemList"].Add(new NotifiableIoItem()                {                    Address = address,                    Name = moduleName,                    Description = description,                    Index = intIndex,                    Provider = provider,                    BlockOffset = blockOffset,                    BlockIndex = intIndex,                });                if (!isIgnoreSaveDB)                    DATA.Subscribe($"IO.{moduleName}", () => doAccessor.Value);            }            XmlNodeList lstAo = xml.SelectNodes("IO_DEFINE/Ana_Out/AO_ITEM");            //    < AO_ITEM Index = "0" BufferOffset="0" Name = "" Addr = "0" Description = "" />            foreach (var aoItem in lstAo)            {                XmlElement element = aoItem as XmlElement;                if (element == null)                    continue;                string index = element.GetAttribute("Index");                string bufferOffset = element.GetAttribute("BufferOffset");                if (string.IsNullOrEmpty(bufferOffset))                    bufferOffset = index;                string name = element.GetAttribute("Name");                string address = element.GetAttribute("Addr");                string description = element.GetAttribute("Description");                if (string.IsNullOrEmpty(name) || string.IsNullOrEmpty(index) || string.IsNullOrEmpty(bufferOffset))                    continue;                name = name.Trim();                index = index.Trim();                bufferOffset = bufferOffset.Trim();                string moduleName = string.IsNullOrEmpty(module) ? name : $"{module}.{name}";                int intIndex;                if (!int.TryParse(index, out intIndex))                    continue;                int intBufferOffset;                if (!int.TryParse(bufferOffset, out intBufferOffset))                    continue;                if (!_aoBuffer.ContainsKey(provider) || !_aoBuffer[provider].ContainsKey(blockOffset))                {                    throw new Exception("Not defined AO buffer from IO provider, " + provider);                }                AOAccessor aoAccessor = new AOAccessor(moduleName, intBufferOffset, _aoBuffer[provider][blockOffset]);                _aoMap[moduleName] = aoAccessor;                aoAccessor.IoTableIndex = intIndex;                aoAccessor.Addr = address;                aoAccessor.Provider = provider;                aoAccessor.BlockOffset = blockOffset;                aoAccessor.Description = description;                if (!_aoList.ContainsKey(provider))                    _aoList[provider] = new List<AOAccessor>();                _aoList[provider].Add(aoAccessor);                _ioItemList[$"{provider}.AOItemList"].Add(new NotifiableIoItem()                {                    Address = address,                    Name = moduleName,                    Description = description,                    Index = intIndex,                    Provider = provider,                    BlockOffset = blockOffset,                    BlockIndex = intIndex,                });                if (!isIgnoreSaveDB)                {                    //DATA.Subscribe($"IO.{moduleName}", () => aoAccessor.Value);                    DATA.Subscribe($"IO32.{moduleName}", () =>                    {                        if (aoAccessor.Index < aoAccessor.Buffer.Length-1)                        {                            byte[] high = BitConverter.GetBytes(aoAccessor.Buffer[aoAccessor.Index]);                            byte[] low = BitConverter.GetBytes(aoAccessor.Buffer[aoAccessor.Index + 1]);                            return BitConverter.ToSingle(new[] { high[0], high[1], low[0], low[1] }, 0);                        }                        else                        {                            return aoAccessor.Value;                        }                    });                }            }            XmlNodeList lstAi = xml.SelectNodes("IO_DEFINE/Ana_In/AI_ITEM");            //    < AO_ITEM Index = "0" BufferOffset="0" Name = "" Addr = "0" Description = "" />            foreach (var aiItem in lstAi)            {                XmlElement element = aiItem as XmlElement;                if (element == null)                    continue;                string index = element.GetAttribute("Index");                string bufferOffset = element.GetAttribute("BufferOffset");                if (string.IsNullOrEmpty(bufferOffset))                    bufferOffset = index;                string name = element.GetAttribute("Name");                string address = element.GetAttribute("Addr");                string description = element.GetAttribute("Description");                if (string.IsNullOrEmpty(name) || string.IsNullOrEmpty(index) || string.IsNullOrEmpty(bufferOffset))                    continue;                name = name.Trim();                index = index.Trim();                bufferOffset = bufferOffset.Trim();                string moduleName = string.IsNullOrEmpty(module) ? name : $"{module}.{name}";                int intIndex;                if (!int.TryParse(index, out intIndex))                    continue;                int intBufferOffset;                if (!int.TryParse(bufferOffset, out intBufferOffset))                    continue;                if (!_aiBuffer.ContainsKey(provider) || !_aiBuffer[provider].ContainsKey(blockOffset))                {                    throw new Exception("Not defined AI buffer from IO provider, " + provider);                }                AIAccessor aiAccessor = new AIAccessor(moduleName, intBufferOffset, _aiBuffer[provider][blockOffset]);                _aiMap[moduleName] = aiAccessor;                aiAccessor.IoTableIndex = intIndex;                aiAccessor.Addr = address;                aiAccessor.Provider = provider;                aiAccessor.BlockOffset = blockOffset;                aiAccessor.Description = description;                if (!_aiList.ContainsKey(provider))                    _aiList[provider] = new List<AIAccessor>();                _aiList[provider].Add(aiAccessor);                _ioItemList[$"{provider}.AIItemList"].Add(new NotifiableIoItem()                {                    Address = address,                    Name = moduleName,                    Description = description,                    Index = intIndex,                    Provider = provider,                    BlockOffset = blockOffset,                    BlockIndex = intIndex,                });                if (!isIgnoreSaveDB)                {                    //DATA.Subscribe($"IO.{moduleName}", () => aiAccessor.Value);                    DATA.Subscribe($"IO32.{moduleName}", () =>                    {                        if (aiAccessor.Index < aiAccessor.Buffer.Length-1)                        {                            byte[] high = BitConverter.GetBytes(aiAccessor.Buffer[aiAccessor.Index]);                            byte[] low = BitConverter.GetBytes(aiAccessor.Buffer[aiAccessor.Index + 1]);                            return BitConverter.ToSingle(new[] { high[0], high[1], low[0], low[1] }, 0);                        }                        else                        {                            return aiAccessor.Value;                        }                    });                }            }        }        public void SetIoMap(string provider, Dictionary<int, string> ioMappingPathFile)        {            foreach (var map in ioMappingPathFile)            {                SetIoMap(provider, map.Key, map.Value);            }            DATA.Subscribe(provider, "DIList", SubscribeDiData);            DATA.Subscribe(provider, "DOList", SubscribeDoData);            DATA.Subscribe(provider, "AIList", SubscribeAiData);            DATA.Subscribe(provider, "AOList", SubscribeAoData);        }        public void SetIoMapByModule(string provider, int offset, string ioMappingPathFile, string module)        {            SetIoMap(provider, offset, ioMappingPathFile, module);            DATA.Subscribe(provider, "DIList", SubscribeDiData);            DATA.Subscribe(provider, "DOList", SubscribeDoData);            DATA.Subscribe(provider, "AIList", SubscribeAiData);            DATA.Subscribe(provider, "AOList", SubscribeAoData);        }        private void SubscribeIoItemList(string provider)        {            string diKey = $"{provider}.DIItemList";            if (!_ioItemList.ContainsKey(diKey))            {                _ioItemList[diKey] = new List<NotifiableIoItem>();                DATA.Subscribe(diKey, () => _ioItemList[diKey]);            }            string doKey = $"{provider}.DOItemList";            if (!_ioItemList.ContainsKey(doKey))            {                _ioItemList[doKey] = new List<NotifiableIoItem>();                DATA.Subscribe(doKey, () => _ioItemList[doKey]);            }            string aiKey = $"{provider}.AIItemList";            if (!_ioItemList.ContainsKey(aiKey))            {                _ioItemList[aiKey] = new List<NotifiableIoItem>();                DATA.Subscribe(aiKey, () => _ioItemList[aiKey]);            }            string aoKey = $"{provider}.AOItemList";            if (!_ioItemList.ContainsKey(aoKey))            {                _ioItemList[aoKey] = new List<NotifiableIoItem>();                DATA.Subscribe(aoKey, () => _ioItemList[aoKey]);            }        }    }}
 |