diff --git a/Source/Meadow.Foundation.Peripherals/ICs.ADC.Ads1x15/Driver/Ads1x15Base.Enums.cs b/Source/Meadow.Foundation.Peripherals/ICs.ADC.Ads1x15/Driver/Ads1x15Base.Enums.cs
index 9daf174bc..9b3ab69d0 100644
--- a/Source/Meadow.Foundation.Peripherals/ICs.ADC.Ads1x15/Driver/Ads1x15Base.Enums.cs
+++ b/Source/Meadow.Foundation.Peripherals/ICs.ADC.Ads1x15/Driver/Ads1x15Base.Enums.cs
@@ -35,27 +35,27 @@ public enum Addresses : byte
public enum FsrGain
{
///
- /// 2/3rds
+ /// Gain = 2/3rds. Range: ±6.144 V
///
TwoThirds = 0x00,
///
- /// 1
+ /// Gain = 1. Range: ±4.096 V
///
One = 0x01,
///
- /// 2
+ /// Gain = 2. Range: ±2.048 V
///
Two = 0x02,
///
- /// 4
+ /// Gain = 4. Range: ±1.024 V
///
Four = 0x03,
///
- /// 8
+ /// Gain = 8. Range: ±0.512 V
///
Eight = 0x04,
///
- /// 16
+ /// Gain = 16. Range: ±0.256 V
///
Sixteen = 0x05
}
diff --git a/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/FifoBuffer.cs b/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/FifoBuffer.cs
index 74aeee7fe..e07424db6 100644
--- a/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/FifoBuffer.cs
+++ b/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/FifoBuffer.cs
@@ -1,5 +1,6 @@
using System;
using System.Text;
+using System.Threading;
namespace Meadow.Foundation.ICs.IOExpanders;
@@ -7,6 +8,10 @@ namespace Meadow.Foundation.ICs.IOExpanders;
/// This FIFO buffer is implemented as a circular buffer. (Standard computer science stuff)
/// It operates on bytes. The only special method is WriteString,
/// which writes a string to the buffer after converting it with Encoding.ASCII.GetBytes().
+///
+/// 2024.04.20, KW: I've recently discovered the CircularBuffer class in the Meadow.Contracts library.
+/// We should perhaps use that instead of this custom implementation.
+/// But this method is specific to bytes, so it might still be more efficient.
///
public class FifoBuffer
{
@@ -39,8 +44,8 @@ public FifoBuffer(int capacity)
throw new ArgumentException("Capacity must be greater than 0.", nameof(capacity));
_buffer = new byte[capacity];
- _nextRead = 0; // The next read position.
- _nextWrite = 0; // The next write position.
+ _nextRead = 0; // The next read position. (Tail)
+ _nextWrite = 0; // The next write position. (Head)
_count = 0; // The number of items in the buffer.
}
@@ -54,6 +59,8 @@ public void Clear()
_count = 0; // The number of items in the buffer.
}
+ private int skippedBytes = 0;
+ private DateTime skippedBytesLastLogged = DateTime.MinValue;
///
/// Add new item to the _nestWrite position in the buffer.
/// Then move the _nextWrite position. If it goes beyond the capacity, it will wrap around to 0.
@@ -64,7 +71,18 @@ public void Clear()
public void Write(byte item)
{
if (_count == Capacity)
- throw new InvalidOperationException("The buffer is full.");
+ {
+ //throw new InvalidOperationException($"The software FIFO buffer is full. Capacity is {Capacity} bytes.");
+ // 21.04.2024, KW: These exception must be handled by the caller. We count the occurences and throw an error every 5 seconds.
+ skippedBytes++;
+ if (DateTime.Now - skippedBytesLastLogged > TimeSpan.FromSeconds(5))
+ {
+ skippedBytesLastLogged = DateTime.Now;
+ //Resolver.Log.Info($"The software FIFO buffer is full. Capacity is {Capacity} bytes. Skipped bytes: {skippedBytes} (This message will be repeated every 5 seconds)");
+ throw new Exception($"The software FIFO buffer is full. Capacity is {Capacity} bytes. Skipped bytes: {skippedBytes} (This message will be repeated every 5 seconds)");
+ }
+ return;
+ }
_buffer[_nextWrite] = item;
_nextWrite = (_nextWrite + 1) % Capacity; // Move _nextWrite forward and possibly wrap.
@@ -134,4 +152,53 @@ public void WriteString(string text)
Write((byte)c);
}
}
+
+ ///
+ /// Copied from the CircularBuffer class in the Meadow.Contracts library.
+ /// Removes items from the buffer and places them in a target array
+ ///
+ /// The destination array for the move
+ /// The beginning index of the destination
+ /// The desired number of items to move
+ /// The actual number of items moved
+ public int MoveItemsTo(byte[] destination, int index, int count)
+ {
+ if (count <= 0) { return 0; }
+
+ // How many are we moving?
+ // Move from current toward the tail
+ var actual = (count > Count) ? Count : count;
+ var fromReadToEnd = _buffer.Length - _nextRead;
+
+ //Resolver.Log.Info($"MoveItemsTo() _nextRead={_nextRead} _nextWrite={_nextWrite} actual={actual} fromReadToEnd={fromReadToEnd}");
+
+ if ((_nextRead < _nextWrite)
+ || (_nextRead == 0 && _count == Capacity)
+ || (fromReadToEnd >= actual))
+ {
+ // the data is linear, just copy
+ Array.Copy(_buffer, _nextRead, destination, index, actual);
+
+ // move the tail pointer
+ _nextRead += actual;
+ _count -= actual;
+ }
+ else
+ {
+ // there's a data wrap
+ // copy from here to the end
+ Array.Copy(_buffer, _nextRead, destination, index, fromReadToEnd);
+ // now copy from the start (tail == 0) the remaining data
+ _nextRead = 0;
+ _count -= fromReadToEnd;
+ var remaining = actual - fromReadToEnd;
+ Array.Copy(_buffer, _nextRead, destination, index + fromReadToEnd, remaining);
+
+ // move the tail pointer
+ _nextRead = remaining;
+ _count -= remaining;
+ }
+
+ return actual;
+ }
}
diff --git a/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.Enums.cs b/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.Enums.cs
index 5bfe9545c..80b10d753 100644
--- a/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.Enums.cs
+++ b/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.Enums.cs
@@ -75,6 +75,75 @@ public enum Addresses : byte
/// The default bus address (0x48)
///
Default = Address_0x48,
+
+ // A new set of constants that show the direct relationship between the address pins and the I2C address
+ // Please refer to the datasheet for the SC16IS752/SC16IS762 page 39.
+ // Note that the address in the datasheet multiplied by 2, because they include a left shit for the R/W bit.
+
+ ///
+ /// A1 pin connected to Vdd, A0 pin connected to Vdd
+ ///
+ A1Vdd_A0Vdd = 0x48,
+ ///
+ /// A1 pin connected to Vdd, A0 pin connected to Gnd
+ ///
+ A1Vdd_A0Gnd = 0x49,
+ ///
+ /// A1 pin connected to Vdd, A0 pin connected to SCL
+ ///
+ A1Vdd_A0Scl = 0x4A,
+ ///
+ /// A1 pin connected to Vdd, A0 pin connected to SDA
+ ///
+ A1Vdd_A0Sda = 0x4B,
+ ///
+ /// A1 pin connected to Gnd, A0 pin connected to Vdd
+ ///
+ A1Gnd_A0Vdd = 0x4C,
+ ///
+ /// A1 pin connected to Gnd, A0 pin connected to Gnd
+ ///
+ A1Gnd_A0Gnd = 0x4D,
+ ///
+ /// A1 pin connected to Gnd, A0 pin connected to SCL
+ ///
+ A1Gnd_A0Scl = 0x4E,
+ ///
+ /// A1 pin connected to Gnd, A0 pin connected to SDA
+ ///
+ A1Gnd_A0Sda = 0x4F,
+ ///
+ /// A1 pin connected to SCL, A0 pin connected to Vdd
+ ///
+ A1Scl_A0Vdd = 0x50,
+ ///
+ /// A1 pin connected to SCL, A0 pin connected to Gnd
+ ///
+ A1Scl_A0Gnd = 0x51,
+ ///
+ /// A1 pin connected to SCL, A0 pin connected to SCL
+ ///
+ A1Scl_A0Scl = 0x52,
+ ///
+ /// A1 pin connected to SCL, A0 pin connected to SDA
+ ///
+ A1Scl_A0Sda = 0x53,
+ ///
+ /// A1 pin connected to SDA, A0 pin connected to Vdd
+ ///
+ A1Sda_A0Vdd = 0x54,
+ ///
+ /// A1 pin connected to SDA, A0 pin connected to Gnd
+ ///
+ A1Sda_A0Gnd = 0x55,
+ ///
+ /// A1 pin connected to SDA, A0 pin connected to SCL
+ ///
+ A1Sda_A0Scl = 0x56,
+ ///
+ /// A1 pin connected to SDA, A0 pin connected to SDA
+ ///
+ A1Sda_A0Sda = 0x57,
}
internal enum Registers : byte
diff --git a/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.I2c.cs b/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.I2c.cs
index 0137598dc..d2451d263 100644
--- a/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.I2c.cs
+++ b/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.I2c.cs
@@ -29,6 +29,7 @@ public Sc16is7x2(II2cBus i2cBus, Frequency oscillatorFrequency, Addresses addres
: this(oscillatorFrequency, irq, latchGpioInterrupt)
{
_i2cComms = new I2cCommunications(i2cBus, (byte)address);
+ Reset(); // We need to call this _after_ the I2C communications object is created
}
}
}
\ No newline at end of file
diff --git a/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.Sc16is7x2Channel.cs b/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.Sc16is7x2Channel.cs
index 0323ee3d7..e60f7fa7d 100644
--- a/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.Sc16is7x2Channel.cs
+++ b/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.Sc16is7x2Channel.cs
@@ -24,8 +24,8 @@ public class Sc16is7x2Channel : ISerialPort
///
public int BytesToRead => (_irqReadBuffer == null)
- ? _controller.GetReadFifoCount(_channel)
- : _controller.GetReadFifoCount(_channel) + _irqReadBuffer.Count;
+ ? GetReadHwFifoCount()
+ : GetReadHwFifoCount() + _irqReadBuffer.Count;
///
public bool IsOpen { get; private set; }
@@ -51,6 +51,15 @@ public class Sc16is7x2Channel : ISerialPort
private readonly FifoBuffer? _irqReadBuffer;
+ public bool IsIrqDriven => _irq != null;
+
+ // Optimizations for fast read. Precalculated values for this channel.
+ private byte _rhrAddress;
+ private byte _lsrAddress;
+ private byte _rxlvlAddress;
+ private byte _txlvlAddress;
+ private IByteCommunications _comms;
+
///
/// This method is never called directly from user code.
///
@@ -71,10 +80,17 @@ internal Sc16is7x2Channel(Sc16is7x2 controller, string portName, Channels channe
_controller = controller;
_channel = channel;
- Initialize(baudRate, dataBits, parity, stopBits);
+ _rhrAddress = CalculateChannelAddress((byte)Registers.RHR);
+ _lsrAddress = CalculateChannelAddress((byte)Registers.LSR);
+ _rxlvlAddress = CalculateChannelAddress((byte)Registers.RXLVL);
+ _txlvlAddress = CalculateChannelAddress((byte)Registers.TXLVL);
+ _comms = controller.Comms;
+
+ InitChannel(baudRate, dataBits, parity, stopBits);
+
if (isRS485)
{
- InitializeRS485(invertDE);
+ EnableRS485(invertDE);
}
if (irq != null)
@@ -82,8 +98,9 @@ internal Sc16is7x2Channel(Sc16is7x2 controller, string portName, Channels channe
// Setting up IRQ read with a large software FIFO buffer to offload the hardware FIFO of only 64 bytes.
_irq = irq;
_irqReadBuffer = new FifoBuffer(readBufferSize);
- _controller.EnableReceiveInterrupts(_channel);
- _irq.Changed += OnInterruptLineChanged;
+ // 21.04.2024, KW: We'll try to move this to Open(). And we'll remove this handler in Close().
+ //EnableReceiveInterrupts();
+ //_irq.Changed += UartChannelInterruptHandler;
}
// https://github.com/WildernessLabs/Meadow_Issues/issues/74
@@ -98,17 +115,19 @@ internal Sc16is7x2Channel(Sc16is7x2 controller, string portName, Channels channe
// a.RaiseInterrupt(this, new DigitalPortResult());
}
- internal void OnInterruptLineChanged(object sender, DigitalPortResult e)
+ internal void UartChannelInterruptHandler(object sender, DigitalPortResult e)
{
// If the first message after reboot is longer than the FIFO buffer, we already have a buffer overrun at this point.
// For any consecutive messages, it works fine.
// Ref: https://github.com/WildernessLabs/Meadow_Issues/issues/74
- //int count = _controller.GetReadFifoCount(_channel);
- //Resolver.Log.Info($"Channel {_channel} port FIFO: {count} bytes");
+ int count = GetReadHwFifoCount();
+ //Resolver.Log.Info($"->UART interrupt. Channel={_channel} port HW FIFO: {count} bytes");
+
+ if (count == 0) return; // The IRQ was not for us.
if (_irqReadBuffer == null)
{
- if (_controller.ReceiveInterruptPending(_channel))
+ if (ReceiveInterruptPending())
this.DataReceived?.Invoke(this, new SerialDataReceivedEventArgs(SerialDataType.Chars));
}
else
@@ -124,7 +143,7 @@ internal void OnInterruptLineChanged(object sender, DigitalPortResult e)
public int BaudRate
{
get => _baudRate;
- set => _baudRate = _controller.SetBaudRate(_channel, value);
+ set => _baudRate = SetBaudRate(value);
}
///
@@ -133,7 +152,7 @@ public int DataBits
get => _dataBits;
set
{
- _controller.SetLineSettings(_channel, value, Parity, StopBits);
+ SetLineSettings(value, Parity, StopBits);
_dataBits = value;
}
}
@@ -144,7 +163,7 @@ public Parity Parity
get => _parity;
set
{
- _controller.SetLineSettings(_channel, DataBits, value, StopBits);
+ SetLineSettings(DataBits, value, StopBits);
_parity = value;
}
}
@@ -155,24 +174,17 @@ public StopBits StopBits
get => _stopBits;
set
{
- _controller.SetLineSettings(_channel, DataBits, Parity, value);
+ SetLineSettings(DataBits, Parity, value);
_stopBits = value;
}
}
- private void Initialize(int baudRate, int dataBits, Parity parity, StopBits stopBits)
+ private void InitChannel(int baudRate, int dataBits, Parity parity, StopBits stopBits)
{
- _dataBits = dataBits;
- _stopBits = stopBits;
- _controller.Reset();
- _controller.EnableFifo(_channel);
- _baudRate = _controller.SetBaudRate(_channel, baudRate);
- _controller.SetLineSettings(_channel, dataBits, parity, stopBits);
- }
-
- private void InitializeRS485(bool invertDE)
- {
- _controller.EnableRS485(_channel, invertDE);
+ //_controller.Reset();
+ EnableFifo();
+ _baudRate = SetBaudRate(baudRate);
+ SetLineSettings(dataBits, parity, stopBits);
}
///
@@ -181,25 +193,42 @@ private void InitializeRS485(bool invertDE)
internal void ReadAllIrqFifo()
{
if (_irqReadBuffer == null) return;
+
int totalRead = 0;
- int count = _controller.GetReadFifoCount(_channel); // How may bytes to read.
+ int count = GetReadHwFifoCount(); // How may bytes to read.
+ //int count = _comms.ReadRegister(_rxlvlAddress); // How may bytes to read.
while (count > 0)
- {
+ {
for (int i = 0; i < count; i++)
- _irqReadBuffer.Write((byte)_controller.ReadByte(_channel));
+ {
+ byte b = _comms.ReadRegister(_rhrAddress);
+ try
+ {
+ _irqReadBuffer.Write(b);
+ }
+ catch (Exception ex)
+ {
+ Resolver.Log.Error($"!!!!!---> ReadAllIrqFifo: Channel={_channel} Exception: {ex.Message}");
+ BufferOverrun?.Invoke(this, new ThreadExceptionEventArgs(ex));
+ }
+ }
totalRead += count;
- count = _controller.GetReadFifoCount(_channel); // Check that we're all done. To make sure IRQ is reset.
+ //Resolver.Log.Info($"---> ReadAllIrqFifo: Channel={_channel} Read {count}/{totalRead}");
- byte lsr = _controller.ReadChannelRegister(Registers.LSR, _channel);
+ //byte lsr = _controller.ReadChannelRegister(Registers.LSR, _channel);
+ byte lsr = _comms.ReadRegister(_lsrAddress);
if ((lsr & RegisterBits.LSR_OVERRUN_ERROR) > 0)
{
- _irqReadBuffer.WriteString("[BUFFER OVERRUN]"); // Not sure to keep this, but nice when debugging.
+ _irqReadBuffer.WriteString("[OVERRUN]"); // Not sure to keep this, but nice when debugging.
BufferOverrun?.Invoke(this, EventArgs.Empty);
}
+
+ count = GetReadHwFifoCount(); // Check that we're all done. To make sure IRQ is reset.
+ //count = _comms.ReadRegister(_rxlvlAddress); // Check that we're all done. To make sure IRQ is reset.
}
if (totalRead > 0)
{
- Resolver.Log.Info($"---> ReadAllIrqFifo: Channel {_channel} port read {totalRead} bytes");
+ //Resolver.Log.Info($"---> ReadAllIrqFifo: Channel={_channel} Done {count}/{totalRead}");
}
}
@@ -210,17 +239,17 @@ public int ReadByte()
{
// The normal way...
// check if data is available
- if (!_controller.IsFifoDataAvailable(_channel))
+ if (!IsHwFifoDataAvailable())
{
return -1;
}
// read the data
- return _controller.ReadByte(_channel);
+ return ReadHwFifoByte();
}
else
{
- // IRQ fast read....
+ // IRQ fast read from software FIFO....
ReadAllIrqFifo();
if (_irqReadBuffer.Count == 0)
return -1;
@@ -241,13 +270,13 @@ public int Read(byte[] buffer, int offset, int count)
if (_irqReadBuffer == null)
{
// The normal way...
- var available = _controller.GetReadFifoCount(_channel);
+ var available = GetReadHwFifoCount();
// read either the available or count, whichever is less, unless available is 0, in which case we wait until timeout
while (available == 0)
{
Thread.Sleep(10);
- available = _controller.GetReadFifoCount(_channel);
+ available = GetReadHwFifoCount();
if (timeout > 0)
{
@@ -262,7 +291,7 @@ public int Read(byte[] buffer, int offset, int count)
for (var i = 0; i < toRead; i++)
{
- buffer[i + offset] = _controller.ReadByte(_channel);
+ buffer[i + offset] = ReadHwFifoByte();
}
return toRead;
@@ -306,7 +335,7 @@ public byte[] ReadAll()
if (_irqReadBuffer == null)
{
// The normal way...
- var available = _controller.GetReadFifoCount(_channel);
+ var available = GetReadHwFifoCount();
var buffer = new byte[available];
Read(buffer, 0, available);
return buffer;
@@ -317,10 +346,11 @@ public byte[] ReadAll()
ReadAllIrqFifo(); // Always read whatever is in the FIFO.
var available = _irqReadBuffer.Count;
var buffer = new byte[available];
- for (int i = 0; i < available; i++)
- {
- buffer[i] = _irqReadBuffer.Read();
- }
+ //for (int i = 0; i < available; i++)
+ //{
+ // buffer[i] = _irqReadBuffer.Read();
+ //}
+ _irqReadBuffer.MoveItemsTo(buffer, 0, available);
return buffer;
}
}
@@ -328,29 +358,41 @@ public byte[] ReadAll()
///
public void ClearReceiveBuffer()
{
- _controller.ResetReadFifo(_channel);
- if (_irqReadBuffer != null)
+ ResetReadHwFifo();
+ if (IsIrqDriven)
_irqReadBuffer.Clear();
}
///
- public void Close()
+ public void Dispose()
{
- IsOpen = false;
- if (_irqReadBuffer != null)
- _irqReadBuffer.Clear();
+ // nop
}
///
- public void Dispose()
+ public void Open()
{
- // nop
+ if (!IsOpen)
+ {
+ IsOpen = true;
+ ResetWriteHwFifo();
+ ClearReceiveBuffer();
+ if (IsIrqDriven)
+ _irq.Changed += UartChannelInterruptHandler;
+ EnableReceiveInterrupts();
+ }
}
///
- public void Open()
+ public void Close()
{
- IsOpen = true;
+ if (IsOpen)
+ {
+ IsOpen = false;
+ DisableReceiveInterrupts();
+ if (IsIrqDriven)
+ _irq.Changed -= UartChannelInterruptHandler;
+ }
}
///
@@ -378,13 +420,13 @@ public int Write(byte[] buffer, int offset, int count)
timeout = Environment.TickCount + (int)ReadTimeout.TotalMilliseconds;
}
// wait for THR to be empty
- while (!_controller.IsTransmitHoldingRegisterEmpty(_channel))
+ while (!IsTransmitHoldingRegisterEmpty())
{
Thread.Sleep(10);
}
// write until we're either written all or the THR is full
- var available = _controller.GetWriteFifoSpace(_channel);
+ var available = GetWriteHwFifoSpace();
while (remaining > 0)
{
@@ -392,7 +434,7 @@ public int Write(byte[] buffer, int offset, int count)
while (available <= 0)
{
Thread.Sleep(10);
- available = _controller.GetWriteFifoSpace(_channel);
+ available = GetWriteHwFifoSpace();
if (timeout > 0)
{
@@ -403,18 +445,266 @@ public int Write(byte[] buffer, int offset, int count)
}
}
- _controller.WriteByte(_channel, buffer[index]);
+ WriteHwFifoByte(buffer[index]);
index++;
available--;
remaining--;
if (available == 0)
{
- available = _controller.GetWriteFifoSpace(_channel);
+ available = GetWriteHwFifoSpace();
}
}
return count;
}
+
+
+ // ******************* Hardware communication methods *******************
+ // Moved here from Sc16is7x2.cs so we can ommit the channel parameter
+ // and unnecessary calls to parent class. (We keep calls local)
+
+ internal void EnableFifo()
+ {
+ var fcr = ReadChannelRegister(Registers.FCR);
+ fcr |= RegisterBits.FCR_FIFO_ENABLE;
+ WriteChannelRegister(Registers.FCR, fcr);
+ }
+
+ internal void EnableReceiveInterrupts()
+ {
+ //var ier = ReadChannelRegister(Registers.IER);
+ //ier |= RegisterBits.IER_RHR_ENABLE;
+ //WriteChannelRegister(Registers.IER, ier);
+ SetChannelRegisterBits(Registers.IER, RegisterBits.IER_RHR_ENABLE);
+ }
+
+ internal void DisableReceiveInterrupts()
+ {
+ //var ier = ReadChannelRegister(Registers.IER);
+ //ier &= (byte)~(RegisterBits.IER_RHR_ENABLE);
+ //WriteChannelRegister(Registers.IER, ier);
+ ClearChannelRegisterBits(Registers.IER, RegisterBits.IER_RHR_ENABLE);
+ }
+
+ internal bool ReceiveInterruptPending()
+ {
+ // IIR[0] is 0 for any pending interrupt
+ // RHR will be IIR[2] *exclusively*
+ var iir = ReadChannelRegister(Registers.IIR);
+ return (iir & RegisterBits.IIR_RHR_INTERRUPT) == RegisterBits.IIR_RHR_INTERRUPT;
+ }
+
+ internal void SetLineSettings(int dataBits, Parity parity, StopBits stopBits)
+ {
+ var lcr = ReadChannelRegister(Registers.LCR);
+ lcr &= unchecked((byte)~0x3f); // clear all of the line setting bits for simplicity
+
+ switch (dataBits)
+ {
+ case 5:
+ lcr |= RegisterBits.LCR_5_DATA_BITS;
+ break;
+ case 6:
+ lcr |= RegisterBits.LCR_6_DATA_BITS;
+ break;
+ case 7:
+ lcr |= RegisterBits.LCR_7_DATA_BITS;
+ break;
+ case 8:
+ lcr |= RegisterBits.LCR_8_DATA_BITS;
+ break;
+ default:
+ throw new ArgumentOutOfRangeException(nameof(dataBits));
+
+ }
+
+ if (stopBits == StopBits.Two)
+ {
+ lcr |= RegisterBits.LCR_2_STOP_BITS;
+ }
+
+ switch (parity)
+ {
+ case Parity.None:
+ lcr |= RegisterBits.LCR_PARITY_NONE;
+ break;
+ case Parity.Odd:
+ lcr |= RegisterBits.LCR_PARITY_ODD;
+ break;
+ case Parity.Even:
+ lcr |= RegisterBits.LCR_PARITY_EVEN;
+ break;
+ // device supports mark and space, but Meadow doesn't have values for it
+ }
+
+ WriteChannelRegister(Registers.LCR, lcr);
+ }
+
+ internal int SetBaudRate(int baudRate)
+ {
+ // the part baud rate is a division of the oscillator frequency, not necessarily the value requested
+ var mcr = ReadChannelRegister(Registers.MCR);
+ var prescaler = ((mcr & RegisterBits.MCR_CLOCK_DIVISOR) == 0) ? 1 : 4;
+ var divisor1 = _controller.OscillatorFrequency.Hertz / prescaler;
+ var divisor2 = baudRate * 16;
+
+ if (divisor2 > divisor1) throw new ArgumentOutOfRangeException(nameof(baudRate), "Oscillator does not allow requested baud rate");
+
+ var divisor = (ushort)Math.Ceiling(divisor1 / divisor2);
+
+ // enable the divisor latch
+ var lcr = ReadChannelRegister(Registers.LCR);
+ lcr |= RegisterBits.LCR_DIVISOR_LATCH_ENABLE;
+ WriteChannelRegister(Registers.LCR, lcr);
+
+ // set the baud rate
+ WriteChannelRegister(Registers.DLL, (byte)(divisor & 0xff));
+ WriteChannelRegister(Registers.DLH, (byte)(divisor >> 8));
+
+ // disable the divisor latch
+ lcr &= unchecked((byte)~RegisterBits.LCR_DIVISOR_LATCH_ENABLE);
+ WriteChannelRegister(Registers.LCR, lcr);
+
+ // return the actual baud rate achieved
+ return (int)(divisor1 / divisor / 16);
+ }
+
+ internal void EnableRS485(bool invertDE)
+ {
+ var efcr = ReadChannelRegister(Registers.EFCR);
+ efcr |= RegisterBits.EFCR_9BITMODE | RegisterBits.EFCR_RTSCON;
+
+ if (invertDE)
+ {
+ efcr |= RegisterBits.EFCR_RTSINVER;
+ }
+ else
+ {
+ efcr &= unchecked((byte)~RegisterBits.EFCR_RTSINVER);
+ }
+
+ WriteChannelRegister(Registers.EFCR, efcr);
+ }
+
+
+ // ******************* UART read methods *******************
+
+ ///
+ /// Returns the number of bytes in the receive FIFO.
+ ///
+ ///
+ internal int GetReadHwFifoCount()
+ {
+ return _comms.ReadRegister(_rxlvlAddress);
+ }
+
+ ///
+ /// Returns true if there is data available in the receive FIFO.
+ ///
+ ///
+ internal bool IsHwFifoDataAvailable()
+ {
+ return GetReadHwFifoCount() > 0;
+ }
+
+ internal void ResetReadHwFifo()
+ {
+ var fcr = ReadChannelRegister(Registers.FCR);
+ fcr |= RegisterBits.FCR_RX_FIFO_RESET;
+ WriteChannelRegister(Registers.FCR, fcr);
+ }
+
+ internal byte ReadHwFifoByte()
+ {
+ return ReadChannelRegister(Registers.RHR);
+ }
+
+
+ // ******************* UART write methods *******************
+
+ ///
+ /// Returns the empty space in the transmit FIFO.
+ ///
+ ///
+ internal int GetWriteHwFifoSpace()
+ {
+ return _comms.ReadRegister(_txlvlAddress);
+ }
+
+ ///
+ /// Reading status from the THR bit in the LSR register.
+ ///
+ ///
+ internal bool IsTransmitHoldingRegisterEmpty()
+ {
+ var lsr = ReadChannelRegister(Registers.LSR);
+ return (lsr & RegisterBits.LSR_THR_EMPTY) == RegisterBits.LSR_THR_EMPTY;
+ }
+
+ internal void ResetWriteHwFifo()
+ {
+ var fcr = ReadChannelRegister(Registers.FCR);
+ fcr |= RegisterBits.FCR_TX_FIFO_RESET;
+ WriteChannelRegister(Registers.FCR, fcr);
+ }
+
+ internal void WriteHwFifoByte(byte data)
+ {
+ WriteChannelRegister(Registers.THR, data);
+ }
+
+
+ // ******************* Channel communication methods *******************
+
+ private byte CalculateChannelAddress(byte register)
+ {
+ // see page 40 of the data sheet for explanation of this
+ var subaddress = (byte)(((byte)register << 3) | ((byte)_channel << 1));
+ return subaddress;
+ }
+
+ private byte ReadChannelRegister(Registers register)
+ {
+ // see page 40 of the data sheet for explanation of this
+ var subaddress = (byte)(((byte)register << 3) | ((byte)_channel << 1));
+ byte v = _comms.ReadRegister(subaddress);
+ return v;
+ }
+
+ private void WriteChannelRegister(Registers register, byte value)
+ {
+ // see page 40 of the data sheet for explanation of this
+ var subaddress = (byte)(((byte)register << 3) | ((byte)_channel << 1));
+ _comms.WriteRegister(subaddress, value);
+
+ //int b = GetWriteHwFifoSpace();
+ //int c = GetReadHwFifoCount();
+ //Resolver.Log.Info($"->UART write. Channel={_channel} port HW FIFO: {b} bytes Read FIFO: {c}");
+ }
+
+ ///
+ /// Sets bits in the register if the corresponding bit in the value parameter is set.
+ ///
+ ///
+ ///
+ private void SetChannelRegisterBits(Registers register, byte value)
+ {
+ byte currentValue = ReadChannelRegister(register);
+ currentValue |= value; // Set the bits we're going to change
+ WriteChannelRegister(register, currentValue);
+ }
+
+ ///
+ /// Clears bits in the register if the corresponding bit in the mask parameter is set.
+ ///
+ ///
+ ///
+ private void ClearChannelRegisterBits(Registers register, byte mask)
+ {
+ byte currentValue = ReadChannelRegister(register);
+ currentValue &= (byte)~mask; // Flip all bits in mask, then AND with currentValue
+ WriteChannelRegister(register, currentValue);
+ }
}
}
diff --git a/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.Spi.cs b/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.Spi.cs
index 01daa1b50..edc51a7b0 100644
--- a/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.Spi.cs
+++ b/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.Spi.cs
@@ -39,6 +39,7 @@ public Sc16is7x2(ISpiBus spiBus, Frequency oscillatorFrequency, IDigitalOutputPo
: this(oscillatorFrequency, irq, latchGpioInterrupt)
{
_spiComms = new SpiCommunications(spiBus, chipSelect, DefaultSpiBusSpeed);
+ Reset(); // We need to call this _after_ the SPI communications object is created
}
}
}
\ No newline at end of file
diff --git a/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.cs b/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.cs
index 593ab0e18..47abf097d 100644
--- a/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.cs
+++ b/Source/Meadow.Foundation.Peripherals/ICs.IOExpanders.Sc16is7x2/Driver/Sc16is7x2.cs
@@ -22,7 +22,7 @@ public abstract partial class Sc16is7x2 : ISerialController, IDigitalInputOutput
private Sc16is7x2Channel? _channelA;
private Sc16is7x2Channel? _channelB;
- private Frequency _oscillatorFrequency;
+ public Frequency OscillatorFrequency { get; private set; }
private IDigitalInterruptPort? _irq;
private bool _latchGpioInterrupt;
@@ -33,16 +33,17 @@ public abstract partial class Sc16is7x2 : ISerialController, IDigitalInputOutput
internal Sc16is7x2(Frequency oscillatorFrequency, IDigitalInterruptPort? irq, bool latchGpioInterrupt = false)
{
- _oscillatorFrequency = oscillatorFrequency;
+ OscillatorFrequency = oscillatorFrequency;
_irq = irq;
_latchGpioInterrupt = latchGpioInterrupt;
Pins = new PinDefinitions(this);
- if (irq != null)
- {
- irq.Changed += GpioInterruptHandler;
- }
+ // This has to move to GPIO init. We don't know yet if comms init will succeed.
+ //if (irq != null)
+ //{
+ // irq.Changed += GpioInterruptHandler;
+ //}
}
private IByteCommunications Comms
@@ -66,10 +67,10 @@ private IByteCommunications Comms
/// The software FIFO read buffer size. (Not the 64 bytes on chip FIFO)
public ISerialPort CreateSerialPort(SerialPortName portName, int baudRate = 9600, int dataBits = 8, Parity parity = Parity.None, StopBits stopBits = StopBits.One, int readBufferSize = 1024)
{
- //if (_irq != null && _irq.InterruptMode != InterruptMode.EdgeRising)
- //{
- // throw new ArgumentException("If an interrupt port is provided, it must be a rising edge interrupt");
- //}
+ if (_irq != null && _irq.InterruptMode != InterruptMode.EdgeFalling)
+ {
+ Resolver.Log.Warn($"Warning: You have specified InterruptMode={_irq.InterruptMode}. SC16IS7x2 have a falling edge IRQ signal.");
+ }
switch (portName.SystemName)
{
@@ -129,305 +130,101 @@ public ISerialPort CreateRs485SerialPort(Sc16SerialPortName portName, int baudRa
throw new Exception("Unknown port");
}
- internal bool ReceiveInterruptPending(Channels channel)
- {
- // IIR[0] is 0 for any pending interrupt
- // RHR will be IIR[2] *exclusively*
- var iir = ReadChannelRegister(Registers.IIR, channel);
- return (iir & RegisterBits.IIR_RHR_INTERRUPT) == RegisterBits.IIR_RHR_INTERRUPT;
- }
-
- internal void EnableReceiveInterrupts(Channels channel)
- {
- var ier = ReadChannelRegister(Registers.IER, channel);
- ier |= RegisterBits.IER_RHR_ENABLE;
- WriteChannelRegister(Registers.IER, channel, ier);
- }
-
- internal void EnableRS485(Channels channel, bool invertDE)
- {
- var efcr = ReadChannelRegister(Registers.EFCR, channel);
- efcr |= RegisterBits.EFCR_9BITMODE | RegisterBits.EFCR_RTSCON;
-
- if (invertDE)
- {
- efcr |= RegisterBits.EFCR_RTSINVER;
- }
- else
- {
- efcr &= unchecked((byte)~RegisterBits.EFCR_RTSINVER);
- }
-
- WriteChannelRegister(Registers.EFCR, channel, efcr);
- }
-
- internal void WriteByte(Channels channel, byte data)
- {
- WriteChannelRegister(Registers.THR, channel, data);
- }
-
///
- /// Reads the empty space in the transmit fifo
+ /// Reset the device
///
- ///
- ///
- internal int GetWriteFifoSpace(Channels channel)
- {
- return ReadChannelRegister(Registers.TXLVL, channel);
- }
-
- internal int GetReadFifoCount(Channels channel)
- {
- return ReadChannelRegister(Registers.RXLVL, channel);
- }
-
- internal void ResetReadFifo(Channels channel)
- {
- var fcr = ReadChannelRegister(Registers.FCR, channel);
- fcr |= RegisterBits.FCR_RX_FIFO_RESET;
- WriteChannelRegister(Registers.FCR, channel, fcr);
- }
-
- internal bool IsTransmitHoldingRegisterEmpty(Channels channel)
- {
- var thr = ReadChannelRegister(Registers.LSR, channel);
- return (thr & RegisterBits.LSR_THR_EMPTY) == RegisterBits.LSR_THR_EMPTY;
- }
-
- internal bool IsFifoDataAvailable(Channels channel)
- {
- return GetReadFifoCount(channel) > 0;
- }
-
- internal byte ReadByte(Channels channel)
- {
- return ReadChannelRegister(Registers.RHR, channel);
- }
-
- internal void SetLineSettings(Channels channel, int dataBits, Parity parity, StopBits stopBits)
- {
- var lcr = ReadChannelRegister(Registers.LCR, channel);
- lcr &= unchecked((byte)~0x3f); // clear all of the line setting bits for simplicity
-
- switch (dataBits)
- {
- case 5:
- lcr |= RegisterBits.LCR_5_DATA_BITS;
- break;
- case 6:
- lcr |= RegisterBits.LCR_6_DATA_BITS;
- break;
- case 7:
- lcr |= RegisterBits.LCR_7_DATA_BITS;
- break;
- case 8:
- lcr |= RegisterBits.LCR_8_DATA_BITS;
- break;
- default:
- throw new ArgumentOutOfRangeException(nameof(dataBits));
-
- }
-
- if (stopBits == StopBits.Two)
- {
- lcr |= RegisterBits.LCR_2_STOP_BITS;
- }
-
- switch (parity)
- {
- case Parity.None:
- lcr |= RegisterBits.LCR_PARITY_NONE;
- break;
- case Parity.Odd:
- lcr |= RegisterBits.LCR_PARITY_ODD;
- break;
- case Parity.Even:
- lcr |= RegisterBits.LCR_PARITY_EVEN;
- break;
- // device supports mark and space, but Meadow doesn't have values for it
- }
-
- WriteChannelRegister(Registers.LCR, channel, lcr);
- }
-
- internal int SetBaudRate(Channels channel, int baudRate)
- {
- // the part baud rate is a division of the oscillator frequency, not necessarily the value requested
- var mcr = ReadChannelRegister(Registers.MCR, channel);
- var prescaler = ((mcr & RegisterBits.MCR_CLOCK_DIVISOR) == 0) ? 1 : 4;
- var divisor1 = _oscillatorFrequency.Hertz / prescaler;
- var divisor2 = baudRate * 16;
-
- if (divisor2 > divisor1) throw new ArgumentOutOfRangeException(nameof(baudRate), "Oscillator does not allow requested baud rate");
-
- var divisor = (ushort)Math.Ceiling(divisor1 / divisor2);
-
- // enable the divisor latch
- var lcr = ReadChannelRegister(Registers.LCR, channel);
- lcr |= RegisterBits.LCR_DIVISOR_LATCH_ENABLE;
- WriteChannelRegister(Registers.LCR, channel, lcr);
-
- // set the baud rate
- WriteChannelRegister(Registers.DLL, channel, (byte)(divisor & 0xff));
- WriteChannelRegister(Registers.DLH, channel, (byte)(divisor >> 8));
-
- // disable the divisor latch
- lcr &= unchecked((byte)~RegisterBits.LCR_DIVISOR_LATCH_ENABLE);
- WriteChannelRegister(Registers.LCR, channel, lcr);
-
- // return the actual baud rate achieved
- return (int)(divisor1 / divisor / 16);
- }
-
internal void Reset()
{
- var value = ReadChannelRegister(Registers.IOControl, Channels.Both);
+ var value = ReadRegister(Registers.IOControl);
value |= RegisterBits.IOCTL_RESET;
try
{
- WriteChannelRegister(Registers.IOControl, Channels.Both, value);
+ WriteRegister(Registers.IOControl, value);
}
- catch
+ catch (Exception ex)
{
// we expect to get a NACK on this. Very confusing
+ // 06.10.2024, KW: Ignore this error. We get it even when the device works fine. ("Error code 70")
+ Resolver.Log.Trace($"Ignoring SC16IS7x2 error on reset: {ex.Message}");
}
}
- internal void EnableFifo(Channels channel)
- {
- var fcr = ReadChannelRegister(Registers.FCR, channel);
- fcr |= RegisterBits.FCR_FIFO_ENABLE;
- WriteChannelRegister(Registers.FCR, channel, fcr);
- }
-
- private byte ReadChannelRegister(Registers register, Channels channel)
- {
- // see page 40 of the data sheet for explanation of this
- var subaddress = (byte)(((byte)register << 3) | ((byte)channel << 1));
- byte v = Comms.ReadRegister(subaddress);
- if (register == Registers.IOState)
- {
- //Resolver.Log.Info($"ReadChannelRegister: {register} {channel} {v:X2}");
- if (v == 0)
- {
- byte v2 = Comms.ReadRegister(subaddress);
- if (v2 != v)
- {
- v = v2; // Fix buggy value!
- Resolver.Log.Info($"ReadChannelRegister: {register} {channel} {v2:X2} (Corrected value!)");
- }
- }
- }
- return v;
- }
-
- private void WriteChannelRegister(Registers register, Channels channel, byte value)
- {
- // see page 40 of the data sheet for explanation of this
- var subaddress = (byte)(((byte)register << 3) | ((byte)channel << 1));
- //if (register == Registers.IOState)
- //{
- // Resolver.Log.Info($"WriteChannelRegister: {register} {channel} {value:X2}");
- //}
- Comms.WriteRegister(subaddress, value);
- }
-
- private void SetChannelRegisterBits(Registers register, Channels channel, byte value)
- {
- byte currentValue = ReadChannelRegister(register, channel);
- currentValue |= value; // Set the bits we're going to change
- WriteChannelRegister(register, channel, currentValue);
- }
-
- private void ClearChannelRegisterBits(Registers register, Channels channel, byte mask)
- {
- byte currentValue = ReadChannelRegister(register, channel);
- currentValue &= (byte)~mask; // Flip all bits in value, then AND with currentValue
- WriteChannelRegister(register, channel, currentValue);
- }
+ /********************* GPIO **********************/
+ private IDigitalPort[] gpioPorts = new IDigitalPort[8];
- /********************* GPIO **********************/
+ ///
+ /// Is the pin valid for this device instance
+ ///
+ /// The IPin to validate
+ /// True if pin is valid
+ protected bool IsValidPin(IPin pin) => Pins.AllPins.Contains(pin);
- private bool initDone = false;
+ private bool initGpioDone = false;
private void InitGpio()
{
- if (initDone) return;
+ if (initGpioDone) return;
- var a = ReadChannelRegister(Registers.IER, Channels.A);
- a &= unchecked((byte)~RegisterBits.IER_SLEEP_MODE_ENABLE);
- WriteChannelRegister(Registers.IER, Channels.A, a);
- var b = ReadChannelRegister(Registers.IER, Channels.B);
- b &= unchecked((byte)~RegisterBits.IER_SLEEP_MODE_ENABLE);
- WriteChannelRegister(Registers.IER, Channels.B, b);
-
- byte ioControlBefore = ReadGpioRegister(Registers.IOControl);
if (_latchGpioInterrupt)
- SetGpioRegisterBit(Registers.IOControl, RegisterBits.IOCTL_IO_LATCH);
+ SetRegisterBit(Registers.IOControl, RegisterBits.IOCTL_IO_LATCH);
else
- ClearGpioRegisterBits(Registers.IOControl, RegisterBits.IOCTL_IO_LATCH);
- ClearGpioRegisterBits(Registers.IOControl, RegisterBits.IOCTL_GPIO_7to4);
- ClearGpioRegisterBits(Registers.IOControl, RegisterBits.IOCTL_GPIO_3to0);
- byte ioControlAfter = ReadGpioRegister(Registers.IOControl);
- //Resolver.Log.Info($"ioControl: {ioControlBefore} -> {ioControlAfter}");
+ ClearRegisterBit(Registers.IOControl, RegisterBits.IOCTL_IO_LATCH);
+ ClearRegisterBit(Registers.IOControl, RegisterBits.IOCTL_GPIO_7to4);
+ ClearRegisterBit(Registers.IOControl, RegisterBits.IOCTL_GPIO_3to0);
// Set direction of all GPIO's to input
- WriteGpioRegister(Registers.IODir, ioDir);
+ WriteRegister(Registers.IODir, ioDir);
- initDone = true;
+ // 22.04.2024, KW: This has been moved here from the constructor. Add the handler _after_ comms init is OK.
+ if (_irq != null)
+ {
+ _irq.Changed += GpioInterruptHandler;
+ }
+
+ initGpioDone = true;
+ }
+
+ ///
+ /// Get the Interrupt Identification Register (IIR) value.
+ ///
+ ///
+ public byte GetInterruptSource()
+ {
+ byte iir = ReadRegister(Registers.IIR);
+ iir &= RegisterBits.IIR_Id0 + RegisterBits.IIR_Id1 + RegisterBits.IIR_Id2 +
+ RegisterBits.IIR_Id3 + RegisterBits.IIR_Id4 + RegisterBits.IIR_Id5;
+ return iir;
}
- int debugId = 0;
byte lastInputState = 0;
private void GpioInterruptHandler(object sender, DigitalPortResult e)
{
try
{
- //lock(this)
+ //byte iir = GetInterruptSource();
+
+ //Resolver.Log.Info($"HandleGpioInterrupt. Interrupt pin state: {e.Old?.State} {e.New.State} {e.New.Time.ToString("hh:mm:ss.fffffff")} {_irq?.State}");
+ byte state = ReadIoState();
+ byte dirMask = (byte)~ioDir; // Only look at the input pins
+ byte masked = (byte)(state & dirMask);
+ //Resolver.Log.Info($"State: {state:X2} {dirMask:X2} {masked:X2}");
+ //Resolver.Log.Info($"LastState: {lastState} NewState: {state}");
+ if (masked == lastInputState) return;
+
+ byte diff = (byte)(masked ^ lastInputState);
+ //Resolver.Log.Info($"GPIO state: {lastInputState} -> {state}");
+ for (byte i = 0; i < 8; i++)
{
- //int id = debugId++;
- //Resolver.Log.Info($"GpioInterruptHandler... {id}");
-
- //// Prioritize reading the IRQ FIFO's
- //if (_channelA?.IsOpen ?? false)
- // _channelA.OnInterruptLineChanged(sender, e);
- //if (_channelB?.IsOpen ?? false)
- // _channelB.OnInterruptLineChanged(sender, e);
-
- byte iir = GetInterruptSource();
-
- //Resolver.Log.Info($"HandleGpioInterrupt. Interrupt pin state: {e.Old?.State} {e.New.State} {e.New.Time.ToString("hh:mm:ss.fffffff")} {_irq?.State}");
- byte state = ReadGpioRegister(Registers.IOState);
- byte dirMask = (byte)~ioDir; // Only look at the input pins
- byte masked = (byte)(state & dirMask);
- //Resolver.Log.Info($"State: {state:X2} {dirMask:X2} {masked:X2}");
- //Resolver.Log.Info($"LastState: {lastState} NewState: {state}");
- if (masked == lastInputState) return;
-
- byte diff = (byte)(masked ^ lastInputState);
- //Resolver.Log.Info($"GPIO state: {lastState} -> {state}");
- for (byte i = 0; i < 8; i++)
+ if ((diff & (1 << i)) == 0) continue; // No change on this pin
+ if (gpioPorts[i] == null) continue; // No port defined for this pin
+ if (gpioPorts[i] is DigitalInputPort port)
{
- if ((diff & (1 << i)) == 0) continue; // No change on this pin
- if (gpioPorts[i] == null) continue; // No input port defined for this pin
- if (gpioPorts[i] is DigitalInputPort port)
- {
- var newState = BitHelpers.GetBitValue(state, i);
- port.Update(newState);
- }
+ var newState = BitHelpers.GetBitValue(state, i);
+ //Resolver.Log.Info($"Pin {i} new state: {newState}");
+ port.Update(newState);
}
- lastInputState = masked;
-
- //PrintAddressContent();
- //byte irqTest = ReadGpioRegister(Registers.IOIntEna);
- //if (irqTest != irqEna)
- //{
- // Resolver.Log.Info($"irqTest: {irqTest} irqEna: {irqEna}");
- // WriteGpioRegister(Registers.IOIntEna, irqEna);
- //}
-
- //Resolver.Log.Info($"DONE. {id}");
}
+ lastInputState = masked;
}
catch (Exception ex)
{
@@ -435,9 +232,8 @@ private void GpioInterruptHandler(object sender, DigitalPortResult e)
}
}
- /****************** INPUT PORTS ******************/
- private IDigitalPort[] gpioPorts = new IDigitalPort[8];
+ /****************** INPUT PORTS ******************/
///
/// Create a digital input port on a SC16IS7x2 IO expander.
@@ -464,7 +260,7 @@ public IDigitalInputPort CreateDigitalInputPort(
throw new Exception("Only ResistorMode.Disabled is supported. The GPIO ports of SC16IS7x2 does not need/support external pull-up or pull-down resistors.");
}
- var state = ReadGpioRegister(Registers.IOState);
+ var state = ReadIoState();
var initState = BitHelpers.GetBitValue(state, pinIndex);
var port = new DigitalInputPort(pin, initState);
gpioPorts[pinIndex] = port;
@@ -478,8 +274,6 @@ public IDigitalInputPort CreateDigitalInputPort(
throw new Exception("Pin is out of range");
}
- byte irqEna = 0;
-
///
/// Configure the hardware port settings on the SC16IS7x2
///
@@ -496,12 +290,11 @@ private void ConfigureInputPort(IPin pin)
if (_irq != null)
{
- irqEna = BitHelpers.SetBit(irqEna, bitIndex, true);
- SetGpioRegisterBit(Registers.IOIntEna, bitIndex);
+ SetRegisterBit(Registers.IOIntEna, bitIndex);
}
else
{
- ClearGpioRegisterBit(Registers.IOIntEna, bitIndex);
+ ClearRegisterBit(Registers.IOIntEna, bitIndex);
}
}
else
@@ -524,7 +317,7 @@ private void PreValidatedSetPortDirection(IPin pin, PortDirectionType direction)
if (newIoDir == ioDir) return;
//Resolver.Log.Info($"newIoDir: {newIoDir}");
- WriteGpioRegister(Registers.IODir, newIoDir);
+ WriteRegister(Registers.IODir, newIoDir);
ioDir = newIoDir;
}
@@ -560,7 +353,7 @@ public bool ReadPort(IPin pin)
if (portDir == true)
throw new Exception($"Cant read from port {pin.Name}. It is not configured as input");
- var gpio = ReadGpioRegister(Registers.IOState);
+ var gpio = ReadIoState();
// Return the value on that port
return BitHelpers.GetBitValue(gpio, (byte)pin.Key);
@@ -627,111 +420,105 @@ private void WriteToGpioPort(IPin pin, bool value)
{
byte bitIndex = (byte)pin.Key;
if (value)
- SetGpioRegisterBit(Registers.IOState, bitIndex);
+ SetIoStateBit(bitIndex);
else
- ClearGpioRegisterBit(Registers.IOState, bitIndex);
+ ClearIoStateBit(bitIndex);
}
+
/*********** INPUT/OUTPUT PORT HELPERS ***********/
- ///
- /// Get the Interrupt Identification Register (IIR) value.
- ///
- ///
- public byte GetInterruptSource()
+ private void SetRegisterBit(Registers register, int bitIndex)
{
- byte iir = ReadGpioRegister(Registers.IIR);
- iir &= RegisterBits.IIR_Id0 +
- RegisterBits.IIR_Id1 +
- RegisterBits.IIR_Id2 +
- RegisterBits.IIR_Id3 +
- RegisterBits.IIR_Id4 +
- RegisterBits.IIR_Id5;
- return iir;
+ if (bitIndex > 7 || bitIndex < 0) { throw new ArgumentOutOfRangeException(); }
+
+ var oldValue = ReadRegister(register);
+ byte newValue = (byte)(oldValue | ((byte)(1 << bitIndex)));
+ WriteRegister(register, newValue);
}
- ///
- /// Get the interrupt source type from the IIR register value. (Interrupt Identification Register)
- /// Ref. page 14 and 24 in datasheet. Mostly for debugging.
- /// Not sure if multiple sources can be active at the same time. If so, this will need to be modified.
- ///
- /// The IIR register value. (Interrupt Identification Register)
- ///
- public InterruptSourceType GetInterruptSourceType(byte iir)
+ private void ClearRegisterBit(Registers register, int bitIndex)
{
- iir &= (RegisterBits.IIR_Id0 +
- RegisterBits.IIR_Id1 +
- RegisterBits.IIR_Id2 +
- RegisterBits.IIR_Id3 +
- RegisterBits.IIR_Id4 +
- RegisterBits.IIR_Id5);
- if (iir == RegisterBits.IIR_IdNone) return InterruptSourceType.None;
- if (iir == RegisterBits.IIR_IdReceiverLineStatus) return InterruptSourceType.ReceiverLineStatus;
- if (iir == RegisterBits.IIR_IdRxTimeout) return InterruptSourceType.RxTimeout;
- if (iir == RegisterBits.IIR_IdRHR) return InterruptSourceType.RHR;
- if (iir == RegisterBits.IIR_IdTHR) return InterruptSourceType.THR;
- if (iir == RegisterBits.IIR_IdModemStatus) return InterruptSourceType.ModemStatus;
- if (iir == RegisterBits.IIR_IdGpioPins) return InterruptSourceType.GpioPins;
- if (iir == RegisterBits.IIR_IdXoff) return InterruptSourceType.Xoff;
- if (iir == RegisterBits.IIR_IdCtsRts) return InterruptSourceType.CtsRts;
- Resolver.Log.Info($"UNKNOWN INTERRUPT SOURCE (or combination): {iir}");
- return InterruptSourceType.Unknown;
+ if (bitIndex > 7 || bitIndex < 0) { throw new ArgumentOutOfRangeException(); }
+ var oldValue = ReadRegister(register);
+ var newValue = (byte)(oldValue & ((byte)~(1 << bitIndex)));
+ WriteRegister(register, newValue);
}
- ///
- /// Is the pin valid for this device instance
- ///
- /// The IPin to validate
- /// True if pin is valid
- protected bool IsValidPin(IPin pin) => Pins.AllPins.Contains(pin);
-
- private byte ReadGpioRegister(Registers register)
+ // Read a register that is not channel specific.
+ private byte ReadRegister(Registers register)
{
- return ReadChannelRegister(register, Channels.Both);
+ byte v = Comms.ReadRegister((byte)((byte)register << 3));
+ return v;
}
- private void WriteGpioRegister(Registers register, byte value)
+ // Write to a register that is not channel specific.
+ private void WriteRegister(Registers register, byte value)
{
- WriteChannelRegister(register, Channels.Both, value);
+ Comms.WriteRegister((byte)((byte)register << 3), value);
}
- private void SetGpioRegisterBits(Registers register, byte value)
+
+ /********************* OPTIMIZATIONS **********************/
+
+ ///
+ /// Specific method to read the IOState register. This is an optimization.
+ /// We don't want to slow down all other register reads with the validation
+ /// for this specific register.
+ ///
+ ///
+ private byte ReadIoState()
{
- SetChannelRegisterBits(register, Channels.Both, value);
+ // See page 40 of the data sheet for explanation of this
+ byte v = Comms.ReadRegister((byte)Registers.IOState << 3);
+ if (v == 0)
+ {
+ byte v2 = Comms.ReadRegister((byte)Registers.IOState << 3);
+ if (v2 != v)
+ {
+ v = v2; // Fix buggy value!
+ Resolver.Log.Info($"ReadIoState: {v2:X2} (Corrected value!)");
+ }
+ }
+ return v;
}
- private void ClearGpioRegisterBits(Registers register, byte mask)
+ private void WriteIoState(byte value)
{
- ClearChannelRegisterBits(register, Channels.Both, mask);
+ // see page 40 of the data sheet for explanation of this
+ Comms.WriteRegister((byte)Registers.IOState << 3, value);
}
- private void SetGpioRegisterBit(Registers register, int bitIndex)
+ // This is an optimization for the GPIO ports.
+ private void ClearIoStateBit(int bitIndex)
{
- SetChannelRegisterBit(register, Channels.Both, bitIndex);
+ if (bitIndex > 7 || bitIndex < 0) { throw new ArgumentOutOfRangeException(); }
+
+ var oldValue = ReadIoState();
+ byte newValue = (byte)(oldValue & ((byte)~(1 << bitIndex)));
+ if (oldValue != newValue)
+ WriteIoState(newValue);
}
- private void SetChannelRegisterBit(Registers register, Channels channel, int bitIndex)
+ // This is an optimization for the IOState register.
+ private void SetIoStateBit(int bitIndex)
{
if (bitIndex > 7 || bitIndex < 0) { throw new ArgumentOutOfRangeException(); }
- var oldValue = ReadChannelRegister(register, channel);
+ var oldValue = ReadIoState();
byte newValue = (byte)(oldValue | ((byte)(1 << bitIndex)));
- WriteChannelRegister(register, channel, newValue);
+ if (oldValue != newValue)
+ WriteIoState(newValue);
}
- private void ClearGpioRegisterBit(Registers register, int bitIndex)
- {
- ClearChannelRegisterBit(register, Channels.Both, bitIndex);
- }
+ /*********************** NICE TO HAVE METHODS ***********************/
- private void ClearChannelRegisterBit(Registers register, Channels channel, int bitIndex)
+ private byte DebugReadChannelRegister(Registers register, Channels channel)
{
- if (bitIndex > 7 || bitIndex < 0) { throw new ArgumentOutOfRangeException(); }
- var oldValue = ReadChannelRegister(register, channel);
- var newValue = (byte)(oldValue & ((byte)~(1 << bitIndex)));
- WriteChannelRegister(register, channel, newValue);
- //var check = ReadChannelRegister(register, channel);
- //Resolver.Log.Info($"ClearChannelRegisterBit: Old={oldValue} New={newValue} Check={check}");
+ // see page 40 of the data sheet for explanation of this
+ var subaddress = (byte)(((byte)register << 3) | ((byte)channel << 1));
+ byte v = Comms.ReadRegister(subaddress);
+ return v;
}
///
@@ -742,19 +529,52 @@ public void PrintAddressContent()
Resolver.Log.Info($"Register: 00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F");
string a = $"Channel A: ";
for (int r = 0; r < 16; r++)
- a += $"{ReadChannelRegister((Registers)r, Channels.A):X2} ";
+ a += $"{DebugReadChannelRegister((Registers)r, Channels.A):X2} ";
Resolver.Log.Info(a);
string b = $"Channel B: ";
for (int r = 0; r < 16; r++)
- b += $"{ReadChannelRegister((Registers)r, Channels.B):X2} ";
+ b += $"{DebugReadChannelRegister((Registers)r, Channels.B):X2} ";
Resolver.Log.Info(b);
}
+ ///
+ /// Nice-to-have conversion from byte to binary string.
+ ///
+ ///
+ ///
public string ByteToBinaryString(byte b)
{
// Format the byte as a binary string and pad it with zeroes
string binaryString = $"0b{Convert.ToString(b, 2).PadLeft(8, '0')}";
return binaryString;
}
+
+ ///
+ /// Get the interrupt source type from the IIR register value. (Interrupt Identification Register)
+ /// Ref. page 14 and 24 in datasheet. Mostly for debugging.
+ /// Not sure if multiple sources can be active at the same time. If so, this will need to be modified.
+ ///
+ /// The IIR register value. (Interrupt Identification Register)
+ ///
+ public InterruptSourceType GetInterruptSourceType(byte iir)
+ {
+ iir &= (RegisterBits.IIR_Id0 +
+ RegisterBits.IIR_Id1 +
+ RegisterBits.IIR_Id2 +
+ RegisterBits.IIR_Id3 +
+ RegisterBits.IIR_Id4 +
+ RegisterBits.IIR_Id5);
+ if (iir == RegisterBits.IIR_IdNone) return InterruptSourceType.None;
+ if (iir == RegisterBits.IIR_IdReceiverLineStatus) return InterruptSourceType.ReceiverLineStatus;
+ if (iir == RegisterBits.IIR_IdRxTimeout) return InterruptSourceType.RxTimeout;
+ if (iir == RegisterBits.IIR_IdRHR) return InterruptSourceType.RHR;
+ if (iir == RegisterBits.IIR_IdTHR) return InterruptSourceType.THR;
+ if (iir == RegisterBits.IIR_IdModemStatus) return InterruptSourceType.ModemStatus;
+ if (iir == RegisterBits.IIR_IdGpioPins) return InterruptSourceType.GpioPins;
+ if (iir == RegisterBits.IIR_IdXoff) return InterruptSourceType.Xoff;
+ if (iir == RegisterBits.IIR_IdCtsRts) return InterruptSourceType.CtsRts;
+ Resolver.Log.Info($"UNKNOWN INTERRUPT SOURCE (or combination): {iir}");
+ return InterruptSourceType.Unknown;
+ }
}
-}
\ No newline at end of file
+}