// UNCLASSIFIED /*------------------------------------------------------------------------- RAYTHEON PROPRIETARY: THIS DOCUMENT CONTAINS DATA OR INFORMATION PROPRIETARY TO RAYTHEON COMPANY AND IS RESTRICTED TO USE ONLY BY PERSONS AUTHORIZED BY RAYTHEON COMPANY IN WRITING TO USE IT. DISCLOSURE TO UNAUTHORIZED PERSONS WOULD LIKELY CAUSE SUBSTANTIAL COMPETITIVE HARM TO RAYTHEON COMPANY'S BUSINESS POSITION. NEITHER SAID DOCUMENT NOR ITS CONTENTS SHALL BE FURNISHED OR DISCLOSED TO OR COPIED OR USED BY PERSONS OUTSIDE RAYTHEON COMPANY WITHOUT THE EXPRESS WRITTEN APPROVAL OF RAYTHEON COMPANY. THIS PROPRIETARY NOTICE IS NOT APPLICABLE IF DELIVERED TO THE U.S. GOVERNMENT. UNPUBLISHED WORK - COPYRIGHT RAYTHEON COMPANY. -------------------------------------------------------------------------*/ using System; using System.Collections.Generic; using System.IO; using System.Reflection; using Automation.BDaq; using NLog; using Raytheon.Common; using Raytheon.Instruments.GeneralIO; namespace Raytheon.Instruments { /// /// A class that implements a Context DIO card /// public class DIOAdvantech : IGeneralIO, IDisposable { #region PrivateClassMembers private InstantDiCtrl _inputCtrl; private InstantDoCtrl _outputCtrl; private string _name; private readonly int _deviceNum; private readonly SelfTestResult _selfTestResult; private State _state; private object _syncObj = new Object(); private ErrorCode errorCode = ErrorCode.Success; private int _numChannelPerPort = 8; private int _channelStartIndex = 0; private int _numInputChannels; private int _numOutputChannels; private bool _shallWeInitializeOutput = false; private Dictionary _signalNameToChannelInfoMap = new Dictionary(); private readonly ILogger _logger; private readonly IConfigurationManager _configurationManager; private readonly IConfiguration _configuration; #endregion #region PrivateClassFunctions ~DIOAdvantech() { Dispose(false); } private static bool BioFailed(ErrorCode err) { return err < ErrorCode.Success && err >= ErrorCode.ErrorHandleNotValid; } /// /// Dispose the object's resources /// /// protected virtual void Dispose(bool disposing) { if (disposing) { if (_state == State.Ready) { Shutdown(); _state = State.Uninitialized; } } } /// /// Because the user has to define each channel number in the config file, some people will use 0 or 1 start their starting channel number /// Whether the user prefers 0 or 1 as their starting channel number, this function will then calculate the port index and bit index that the /// driver API needs to be able to drive/read the signal /// /// portIndex will range from 0..N /// bitIndex will range from 0..M /// private void GetPortIndexAndBitIndex(string signalName, out int portIndex, out int bitIndex) { portIndex = (int)(Math.Ceiling((double)((int)_signalNameToChannelInfoMap[signalName].channelNumber + Math.Abs(_channelStartIndex - 1)) / (double)_numChannelPerPort) - 1.0); int multiplier = ((int)_signalNameToChannelInfoMap[signalName].channelNumber + _numChannelPerPort) / _numChannelPerPort; bitIndex = (((int)_signalNameToChannelInfoMap[signalName].channelNumber + _numChannelPerPort) - (_numChannelPerPort * multiplier)) - _channelStartIndex; if (bitIndex < 0) { bitIndex = _numChannelPerPort - 1; } } /// /// Invert Contec DIO bits. /// private byte InvertBit(byte bitValue) { if (bitValue == 0) { return 1; } else if (bitValue == 1) { return 0; } else { throw new Exception("Unexpected input value: " + bitValue.ToString()); } } private void ReadBit(int port, int bitNum, out byte data) { _inputCtrl.ReadBit(port, bitNum, out data); } private void WriteBit(int port, int bitNum, byte bitState) { errorCode = _outputCtrl.WriteBit(port, bitNum, bitState); if (BioFailed(errorCode)) { throw new Exception("call to WriteBit returned error: " + errorCode.ToString() + " on card: " + _name); ; } } #endregion #region PublicClassFunctions /// /// DIOAdvantech factory constructor /// /// /// public DIOAdvantech(string deviceName, IConfigurationManager configurationManager) { Name = deviceName; _logger = LogManager.GetLogger($"{this.GetType().Name} - {deviceName}"); _configurationManager = configurationManager; _configuration = _configurationManager.GetConfiguration(Name); string assemblyFolder = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location); string dioModuleDefPath = _configuration.GetConfigurationValue(deviceName, ConfigXml.DIO_MODULE_DEF_FILEPATH.ToString()); if (!Path.IsPathRooted(dioModuleDefPath)) dioModuleDefPath = Path.GetFullPath(Path.Combine(assemblyFolder, dioModuleDefPath)); IConfigurationFile dioModuleConfig = new ConfigurationFile(dioModuleDefPath); Boolean.TryParse(dioModuleConfig.ReadValue(Name, ConfigIni.SHALL_WE_DRIVE_OUTPUT_UPON_INITIALIZATION.ToString()), out _shallWeInitializeOutput); Int32.TryParse(dioModuleConfig.ReadValue(Name, ConfigIni.DEVICE_NUMBER.ToString()), out _deviceNum); Int32.TryParse(dioModuleConfig.ReadValue(Name, ConfigIni.NUM_INPUT_CHANNELS.ToString()), out _numInputChannels); Int32.TryParse(dioModuleConfig.ReadValue(Name, ConfigIni.NUM_OUTPUT_CHANNELS.ToString()), out _numOutputChannels); Int32.TryParse(dioModuleConfig.ReadValue(Name, ConfigIni.NUM_CHANNELS_PER_PORT.ToString()), out _numChannelPerPort); Int32.TryParse(dioModuleConfig.ReadValue(Name, ConfigIni.CHANNEL_START_INDEX.ToString()), out _channelStartIndex); if (!(_channelStartIndex == 0 || _channelStartIndex == 1)) { throw new Exception($"The value for key {ConfigIni.CHANNEL_START_INDEX.ToString()} in section {Name} must be 0 or 1 in {dioModuleDefPath}"); } List outputSignalNames = dioModuleConfig.ReadAllKeys($"{Name}.{ConfigIni.OUTPUT_SIGNALS}"); List intputSignalNames = dioModuleConfig.ReadAllKeys($"{Name}.{ConfigIni.INPUT_SIGNALS}"); IODatatypes.DIOChannelInfo info; foreach (string signalName in outputSignalNames) { if (_signalNameToChannelInfoMap.ContainsKey(signalName)) throw new Exception($"Key {signalName} in section {Name}.{ConfigIni.OUTPUT_SIGNALS} conflicts with the same key defined in another section."); string iniLine = dioModuleConfig.ReadValue($"{Name}.{ConfigIni.OUTPUT_SIGNALS}", signalName); string[] infoTokens = iniLine.Split('|'); if (infoTokens.Length != 2) { throw new Exception($"Key {signalName} in section {Name}.{ConfigIni.OUTPUT_SIGNALS} does not contain 2 tokens"); } info.channelNumber = Convert.ToUInt32(infoTokens[0]); info.ioType = IODatatypes.IOType.DigitalOutput; info.initialValue = Convert.ToInt32(infoTokens[1]); _signalNameToChannelInfoMap[signalName] = info; } foreach (string signalName in intputSignalNames) { if (_signalNameToChannelInfoMap.ContainsKey(signalName)) throw new Exception($"Key {signalName} in section {Name}.{ConfigIni.INPUT_SIGNALS} conflicts with the same key defined in another section."); string iniLine = dioModuleConfig.ReadValue($"{Name}.{ConfigIni.INPUT_SIGNALS}", signalName); info.channelNumber = Convert.ToUInt32(iniLine); info.ioType = IODatatypes.IOType.DigitalInput; info.initialValue = -1; _signalNameToChannelInfoMap[signalName] = info; } _state = State.Uninitialized; _selfTestResult = SelfTestResult.Unknown; } /// /// /// public DIOAdvantech(string deviceName, int deviceNum) { _deviceNum = deviceNum; _name = deviceName; _logger = LogManager.GetLogger($"{this.GetType().Name} - {deviceName}"); _state = State.Uninitialized; _selfTestResult = SelfTestResult.Unknown; } public bool ClearErrors() { throw new NotImplementedException(); } /// /// /// public string DetailedStatus { get { //_outputCtrl.SelectedDevice; Need to see what the card returns for this return "This is a Advantech DIO Card " + _name; } } /// /// /// public bool DisplayEnabled { get { throw new NotImplementedException(); } set { throw new NotImplementedException(); } } /// /// Dispose of this object. /// public void Dispose() { lock (_syncObj) { Dispose(true); GC.SuppressFinalize(this); } } /// /// /// public bool FrontPanelEnabled { get { throw new NotImplementedException(); } set { throw new NotImplementedException(); } } /// /// /// /// /// public IODatatypes.BitState GetBitState(string signalName) { lock (_syncObj) { if (!_signalNameToChannelInfoMap.ContainsKey(signalName)) throw new Exception($"Signal name {signalName} doesn't exist for card: " + _name); if (_signalNameToChannelInfoMap[signalName].channelNumber >= _numInputChannels || _signalNameToChannelInfoMap[signalName].channelNumber < _channelStartIndex) { throw new Exception($"The input channel number {_signalNameToChannelInfoMap[signalName].channelNumber} specified must be >= {_channelStartIndex} and < {_numInputChannels + _channelStartIndex} on card " + _name); } GetPortIndexAndBitIndex(signalName, out int portIndex, out int bitIndex); ReadBit(portIndex, bitIndex, out byte data); return (IODatatypes.BitState)(data); } } /// /// /// public InstrumentMetadata Info { get { throw new NotImplementedException(); } } public void Initialize() { lock (_syncObj) { if (_state == State.Uninitialized) { //@@@ do we need both of these? DO can read and write. Read may be for internal read back _inputCtrl = new InstantDiCtrl(); _outputCtrl = new InstantDoCtrl(); _inputCtrl.SelectedDevice = new DeviceInformation(_deviceNum); _outputCtrl.SelectedDevice = new DeviceInformation(_deviceNum); if (_shallWeInitializeOutput) { foreach (KeyValuePair item in _signalNameToChannelInfoMap) { if (item.Value.initialValue != -1) { SetBit(item.Key, (IODatatypes.BitState)item.Value.initialValue); } } } _state = State.Ready; } else { throw new Exception("expected the state to be Uninitialized, state was: " + _state.ToString() + " on card " + _name); } } } /// /// Return map of all signals /// public Dictionary GetAllSignals() { return _signalNameToChannelInfoMap; } /// /// /// public string Name { get { return _name; } set { _name = value; } } /// /// /// public uint NumberOfInputBits { get { return (uint)_numInputChannels; } } /// /// /// public uint NumberOfOutputBits { get { return (uint)_numOutputChannels; } } /// /// /// /// public SelfTestResult PerformSelfTest() { lock (_syncObj) { // card does not support self test //throw new NotImplementedException("card does not support self test" + " on card " + _name); throw new NotImplementedException(); } } public void Reset() { lock (_syncObj) { Shutdown(); Initialize(); } } /// /// /// /// /// high(open) or low(closed) /// public void SetBit(string signalName, IODatatypes.BitState state) { lock (_syncObj) { if (!_signalNameToChannelInfoMap.ContainsKey(signalName)) throw new Exception($"Signal name {signalName} doesn't exist for card: " + _name); if (_signalNameToChannelInfoMap[signalName].channelNumber >= _numOutputChannels || _signalNameToChannelInfoMap[signalName].channelNumber < _channelStartIndex) { throw new Exception($"The output channel number {_signalNameToChannelInfoMap[signalName].channelNumber} specified must be >= {_channelStartIndex} and < {_numOutputChannels + _channelStartIndex} on card " + _name); } GetPortIndexAndBitIndex(signalName, out int portIndex, out int bitIndex); WriteBit(portIndex, bitIndex, (byte)state); } } public void SetTristate(string signalName) { throw new NotImplementedException(); } /// /// /// public SelfTestResult SelfTestResult { get { return _selfTestResult; } } public void Shutdown() { lock (_syncObj) { if (_state == State.Ready) { _outputCtrl.Dispose(); _inputCtrl.Dispose(); _state = State.Uninitialized; } } } /// /// /// public State Status { get { return _state; } } #endregion } }