2022-01-07 20:23:20 +01:00
|
|
|
//
|
|
|
|
// FILE: MCP23S17.cpp
|
|
|
|
// AUTHOR: Rob Tillaart
|
2023-11-13 17:05:27 +01:00
|
|
|
// VERSION: 0.2.7
|
2022-01-07 20:23:20 +01:00
|
|
|
// PURPOSE: Arduino library for SPI MCP23S17 16 channel port expander
|
|
|
|
// DATE: 2021-12-30
|
2022-01-10 12:58:20 +01:00
|
|
|
// URL: https://github.com/RobTillaart/MCP23S17
|
2022-07-01 12:10:38 +02:00
|
|
|
|
2022-01-07 20:23:20 +01:00
|
|
|
|
|
|
|
#include "MCP23S17.h"
|
|
|
|
|
|
|
|
|
2022-10-24 11:13:46 +02:00
|
|
|
// SOFTWARE SPI
|
2022-01-07 20:23:20 +01:00
|
|
|
MCP23S17::MCP23S17(uint8_t select, uint8_t dataIn, uint8_t dataOut, uint8_t clock, uint8_t address)
|
|
|
|
{
|
2022-07-01 12:10:38 +02:00
|
|
|
_address = (address << 1);
|
2022-01-07 20:23:20 +01:00
|
|
|
_select = select;
|
|
|
|
_dataIn = dataIn;
|
|
|
|
_dataOut = dataOut;
|
|
|
|
_clock = clock;
|
|
|
|
_error = MCP23S17_OK;
|
|
|
|
_hwSPI = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-10-24 11:13:46 +02:00
|
|
|
// HARDWARE SPI
|
2022-07-01 12:10:38 +02:00
|
|
|
MCP23S17::MCP23S17(uint8_t select, SPIClass* spi)
|
|
|
|
{
|
|
|
|
MCP23S17(select, 0x00, spi);
|
|
|
|
}
|
|
|
|
|
2022-10-24 11:13:46 +02:00
|
|
|
|
2022-07-01 12:10:38 +02:00
|
|
|
MCP23S17::MCP23S17(uint8_t select, uint8_t address, SPIClass* spi)
|
2022-01-07 20:23:20 +01:00
|
|
|
{
|
2022-07-01 12:10:38 +02:00
|
|
|
_address = (address << 1);
|
2022-01-07 20:23:20 +01:00
|
|
|
_select = select;
|
|
|
|
_error = MCP23S17_OK;
|
2022-07-01 12:10:38 +02:00
|
|
|
_mySPI = spi;
|
2022-01-07 20:23:20 +01:00
|
|
|
_hwSPI = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool MCP23S17::begin()
|
|
|
|
{
|
|
|
|
::pinMode(_select, OUTPUT);
|
|
|
|
::digitalWrite(_select, HIGH);
|
|
|
|
|
2023-02-04 16:29:14 +01:00
|
|
|
// 8 MHz - datasheet page 8
|
|
|
|
_spi_settings = SPISettings(_SPIspeed, MSBFIRST, SPI_MODE0);
|
2022-01-07 20:23:20 +01:00
|
|
|
|
|
|
|
if (_hwSPI)
|
|
|
|
{
|
2023-08-14 12:56:15 +02:00
|
|
|
#if defined(ESP32)
|
|
|
|
if (_useHSPI) // HSPI
|
|
|
|
{
|
|
|
|
_mySPI = new SPIClass(HSPI);
|
|
|
|
_mySPI->end();
|
|
|
|
_mySPI->begin(14, 12, 13, _select); // CLK=14 MISO=12 MOSI=13
|
|
|
|
}
|
|
|
|
else // VSPI
|
|
|
|
{
|
|
|
|
_mySPI = new SPIClass(VSPI);
|
|
|
|
_mySPI->end();
|
|
|
|
_mySPI->begin(18, 19, 23, _select); // CLK=18 MISO=19 MOSI=23
|
|
|
|
}
|
|
|
|
#else // generic hardware SPI
|
|
|
|
_mySPI = &SPI;
|
2022-07-01 12:10:38 +02:00
|
|
|
_mySPI->end();
|
|
|
|
_mySPI->begin();
|
2023-08-14 12:56:15 +02:00
|
|
|
#endif
|
2022-01-07 20:23:20 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
::pinMode(_dataIn, INPUT);
|
|
|
|
::pinMode(_dataOut, OUTPUT);
|
|
|
|
::pinMode(_clock, OUTPUT);
|
|
|
|
::digitalWrite(_dataOut, LOW);
|
|
|
|
::digitalWrite(_clock, LOW);
|
|
|
|
}
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// check connected
|
2022-01-07 20:23:20 +01:00
|
|
|
if (! isConnected()) return false;
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// disable address increment (datasheet P20
|
2023-02-04 16:29:14 +01:00
|
|
|
// SEQOP: Sequential Operation mode bit
|
|
|
|
// 1 = Sequential operation disabled, address pointer does not increment.
|
|
|
|
// 0 = Sequential operation enabled, address pointer increments.
|
2022-04-14 10:53:15 +02:00
|
|
|
if (! writeReg(MCP23S17_IOCR, MCP23S17_IOCR_SEQOP)) return false;
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// Force INPUT_PULLUP
|
2022-04-14 10:53:15 +02:00
|
|
|
if (! writeReg(MCP23S17_PUR_A, 0xFF)) return false; // 0xFF == all UP
|
|
|
|
if (! writeReg(MCP23S17_PUR_B, 0xFF)) return false; // 0xFF == all UP
|
2022-01-07 20:23:20 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-10-24 11:13:46 +02:00
|
|
|
// just to keep interface in sync with I2C MCP23017 library.
|
2022-01-07 20:23:20 +01:00
|
|
|
bool MCP23S17::isConnected()
|
|
|
|
{
|
|
|
|
_error = MCP23S17_OK;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-07-01 12:10:38 +02:00
|
|
|
uint8_t MCP23S17::getAddress()
|
|
|
|
{
|
|
|
|
return (_address >> 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-10-24 11:13:46 +02:00
|
|
|
///////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2022-09-28 12:56:28 +02:00
|
|
|
// single pin interface
|
2022-10-24 11:13:46 +02:00
|
|
|
//
|
2022-09-28 12:56:28 +02:00
|
|
|
// pin = 0..15
|
|
|
|
// mode = INPUT, OUTPUT, INPUT_PULLUP (= same as INPUT)
|
2022-01-07 20:23:20 +01:00
|
|
|
bool MCP23S17::pinMode(uint8_t pin, uint8_t mode)
|
|
|
|
{
|
|
|
|
if (pin > 15)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PIN_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if ((mode != INPUT) && (mode != INPUT_PULLUP) && (mode != OUTPUT))
|
|
|
|
{
|
|
|
|
_error = MCP23S17_VALUE_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t dataDirectionRegister = MCP23S17_DDR_A;
|
|
|
|
if (pin > 7)
|
|
|
|
{
|
|
|
|
dataDirectionRegister = MCP23S17_DDR_B;
|
|
|
|
pin -= 8;
|
|
|
|
}
|
|
|
|
uint8_t val = readReg(dataDirectionRegister);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint8_t mask = 1 << pin;
|
2022-09-28 12:56:28 +02:00
|
|
|
// only work with valid
|
2022-01-07 20:23:20 +01:00
|
|
|
if ((mode == INPUT) || (mode == INPUT_PULLUP))
|
|
|
|
{
|
|
|
|
val |= mask;
|
|
|
|
}
|
|
|
|
else if (mode == OUTPUT)
|
|
|
|
{
|
|
|
|
val &= ~mask;
|
|
|
|
}
|
2022-09-28 12:56:28 +02:00
|
|
|
// other values won't change val ....
|
2022-01-07 20:23:20 +01:00
|
|
|
writeReg(dataDirectionRegister, val);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// pin = 0..15
|
|
|
|
// value = LOW, HIGH
|
2022-01-07 20:23:20 +01:00
|
|
|
bool MCP23S17::digitalWrite(uint8_t pin, uint8_t value)
|
|
|
|
{
|
|
|
|
if (pin > 15)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PIN_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint8_t IOR = MCP23S17_GPIO_A;
|
|
|
|
if (pin > 7)
|
|
|
|
{
|
|
|
|
IOR = MCP23S17_GPIO_B;
|
|
|
|
pin -= 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t val = readReg(IOR);
|
2022-09-28 12:56:28 +02:00
|
|
|
uint8_t pre = val;
|
2022-01-07 20:23:20 +01:00
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t mask = 1 << pin;
|
|
|
|
if (value)
|
|
|
|
{
|
|
|
|
val |= mask;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
val &= ~mask;
|
|
|
|
}
|
2022-10-24 11:13:46 +02:00
|
|
|
// only write when changed.
|
2022-09-28 12:56:28 +02:00
|
|
|
if (pre != val)
|
2022-01-07 20:23:20 +01:00
|
|
|
{
|
2022-09-28 12:56:28 +02:00
|
|
|
writeReg(IOR, val);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2022-01-07 20:23:20 +01:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t MCP23S17::digitalRead(uint8_t pin)
|
|
|
|
{
|
|
|
|
if (pin > 15)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PIN_ERROR;
|
|
|
|
return MCP23S17_INVALID_READ;
|
|
|
|
}
|
|
|
|
uint8_t IOR = MCP23S17_GPIO_A;
|
|
|
|
if (pin > 7)
|
|
|
|
{
|
|
|
|
IOR = MCP23S17_GPIO_B;
|
|
|
|
pin -= 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t val = readReg(IOR);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return MCP23S17_INVALID_READ;
|
|
|
|
}
|
|
|
|
uint8_t mask = 1 << pin;
|
|
|
|
if (val & mask) return HIGH;
|
|
|
|
return LOW;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// pin = 0..15
|
|
|
|
// reversed = true or false
|
2022-01-07 20:23:20 +01:00
|
|
|
bool MCP23S17::setPolarity(uint8_t pin, bool reversed)
|
|
|
|
{
|
|
|
|
if (pin > 15)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PIN_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint8_t inputPolarityRegister = MCP23S17_POL_A;
|
|
|
|
if (pin > 7)
|
|
|
|
{
|
|
|
|
inputPolarityRegister = MCP23S17_POL_B;
|
|
|
|
pin -= 8;
|
|
|
|
}
|
|
|
|
uint8_t val = readReg(inputPolarityRegister);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint8_t mask = 1 << pin;
|
|
|
|
if (reversed)
|
|
|
|
{
|
|
|
|
val |= mask;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
val &= ~mask;
|
|
|
|
}
|
|
|
|
writeReg(inputPolarityRegister, val);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool MCP23S17::getPolarity(uint8_t pin, bool &reversed)
|
|
|
|
{
|
|
|
|
if (pin > 15)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PIN_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint8_t inputPolarityRegister = MCP23S17_POL_A;
|
|
|
|
if (pin > 7)
|
|
|
|
{
|
|
|
|
inputPolarityRegister = MCP23S17_POL_B;
|
|
|
|
pin -= 8;
|
|
|
|
}
|
|
|
|
uint8_t val = readReg(inputPolarityRegister);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint8_t mask = 1 << pin;
|
|
|
|
reversed = (val & mask) > 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// pin = 0..15
|
|
|
|
// pullup = true or false
|
2022-01-07 20:23:20 +01:00
|
|
|
bool MCP23S17::setPullup(uint8_t pin, bool pullup)
|
|
|
|
{
|
|
|
|
if (pin > 15)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PIN_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint8_t inputPullupRegister = MCP23S17_PUR_A;
|
|
|
|
if (pin > 7)
|
|
|
|
{
|
|
|
|
inputPullupRegister = MCP23S17_PUR_B;
|
|
|
|
pin -= 8;
|
|
|
|
}
|
|
|
|
uint8_t val = readReg(inputPullupRegister);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint8_t mask = 1 << pin;
|
|
|
|
if (pullup)
|
|
|
|
{
|
|
|
|
val |= mask;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
val &= ~mask;
|
|
|
|
}
|
|
|
|
writeReg(inputPullupRegister, val);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool MCP23S17::getPullup(uint8_t pin, bool &pullup)
|
|
|
|
{
|
|
|
|
if (pin > 15)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PIN_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint8_t inputPullupRegister = MCP23S17_PUR_A;
|
|
|
|
if (pin > 7)
|
|
|
|
{
|
|
|
|
inputPullupRegister = MCP23S17_PUR_B;
|
|
|
|
pin -= 8;
|
|
|
|
}
|
|
|
|
uint8_t val = readReg(inputPullupRegister);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint8_t mask = 1 << pin;
|
|
|
|
pullup = (val & mask) > 0;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void MCP23S17::setSPIspeed(uint32_t speed)
|
|
|
|
{
|
|
|
|
_SPIspeed = speed;
|
|
|
|
_spi_settings = SPISettings(_SPIspeed, MSBFIRST, SPI_MODE0);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2022-10-24 11:13:46 +02:00
|
|
|
///////////////////////////////////////////////////////////////////
|
2022-09-28 12:56:28 +02:00
|
|
|
//
|
|
|
|
// 8 pins interface
|
2022-10-24 11:13:46 +02:00
|
|
|
//
|
2022-09-28 12:56:28 +02:00
|
|
|
// whole register at once
|
|
|
|
// port = 0..1
|
|
|
|
// value = 0..0xFF bit pattern
|
2022-01-07 20:23:20 +01:00
|
|
|
bool MCP23S17::pinMode8(uint8_t port, uint8_t value)
|
|
|
|
{
|
|
|
|
if (port > 1)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PORT_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (port == 0) writeReg(MCP23S17_DDR_A, value);
|
|
|
|
if (port == 1) writeReg(MCP23S17_DDR_B, value);
|
|
|
|
_error = MCP23S17_OK;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// port = 0..1
|
|
|
|
bool MCP23S17::write8(uint8_t port, uint8_t value)
|
2022-01-07 20:23:20 +01:00
|
|
|
{
|
|
|
|
if (port > 1)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PORT_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (port == 0) writeReg(MCP23S17_GPIO_A, value);
|
|
|
|
if (port == 1) writeReg(MCP23S17_GPIO_B, value);
|
|
|
|
_error = MCP23S17_OK;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int MCP23S17::read8(uint8_t port)
|
|
|
|
{
|
|
|
|
if (port > 1)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PORT_ERROR;
|
|
|
|
return MCP23S17_INVALID_READ;
|
|
|
|
}
|
|
|
|
_error = MCP23S17_OK;
|
|
|
|
if (port == 0) return readReg(MCP23S17_GPIO_A);
|
2023-02-04 16:29:14 +01:00
|
|
|
return readReg(MCP23S17_GPIO_B); // port == 1
|
2022-01-07 20:23:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// port = 0..1
|
|
|
|
// mask = 0..0xFF bit pattern
|
2022-01-07 20:23:20 +01:00
|
|
|
bool MCP23S17::setPolarity8(uint8_t port, uint8_t mask)
|
|
|
|
{
|
|
|
|
if (port > 1)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PORT_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (port == 0) writeReg(MCP23S17_POL_A, mask);
|
|
|
|
if (port == 1) writeReg(MCP23S17_POL_B, mask);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool MCP23S17::getPolarity8(uint8_t port, uint8_t &mask)
|
|
|
|
{
|
|
|
|
if (port > 1)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PORT_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (port == 0) mask = readReg(MCP23S17_POL_A);
|
|
|
|
if (port == 1) mask = readReg(MCP23S17_POL_B);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// port = 0..1
|
|
|
|
// mask = 0..0xFF bit pattern
|
2022-01-07 20:23:20 +01:00
|
|
|
bool MCP23S17::setPullup8(uint8_t port, uint8_t mask)
|
|
|
|
{
|
|
|
|
if (port > 1)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PORT_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (port == 0) writeReg(MCP23S17_PUR_A, mask);
|
|
|
|
if (port == 1) writeReg(MCP23S17_PUR_B, mask);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool MCP23S17::getPullup8(uint8_t port, uint8_t &mask)
|
|
|
|
{
|
|
|
|
if (port > 1)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_PORT_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (port == 0) mask = readReg(MCP23S17_PUR_A);
|
|
|
|
if (port == 1) mask = readReg(MCP23S17_PUR_B);
|
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-10-24 11:13:46 +02:00
|
|
|
///////////////////////////////////////////////////////////////////
|
2022-09-28 12:56:28 +02:00
|
|
|
//
|
|
|
|
// 16 pins interface
|
2022-10-24 11:13:46 +02:00
|
|
|
//
|
2022-09-28 12:56:28 +02:00
|
|
|
// two register at once
|
|
|
|
// value = 0x0000..0xFFFF bit pattern
|
2022-01-10 12:58:20 +01:00
|
|
|
bool MCP23S17::pinMode16(uint16_t value)
|
|
|
|
{
|
2023-08-17 14:24:58 +02:00
|
|
|
writeReg16(MCP23S17_DDR_A, value);
|
2022-01-10 12:58:20 +01:00
|
|
|
_error = MCP23S17_OK;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// value = 0x0000..0xFFFF bit pattern
|
2022-01-10 12:58:20 +01:00
|
|
|
bool MCP23S17::write16(uint16_t value)
|
|
|
|
{
|
2023-08-17 14:24:58 +02:00
|
|
|
writeReg16(MCP23S17_GPIO_A, value);
|
2022-01-10 12:58:20 +01:00
|
|
|
_error = MCP23S17_OK;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// return = 0x0000..0xFFFF bit pattern
|
2022-01-10 12:58:20 +01:00
|
|
|
uint16_t MCP23S17::read16()
|
|
|
|
{
|
|
|
|
_error = MCP23S17_OK;
|
2023-08-17 14:24:58 +02:00
|
|
|
uint16_t value = readReg16(MCP23S17_GPIO_A);
|
2022-01-10 12:58:20 +01:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// mask = 0x0000..0xFFFF bit pattern
|
2022-01-10 12:58:20 +01:00
|
|
|
bool MCP23S17::setPolarity16(uint16_t mask)
|
|
|
|
{
|
2023-08-17 14:24:58 +02:00
|
|
|
writeReg16(MCP23S17_POL_A, mask);
|
2022-01-10 12:58:20 +01:00
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// mask = 0x0000..0xFFFF bit pattern
|
2022-01-10 12:58:20 +01:00
|
|
|
bool MCP23S17::getPolarity16(uint16_t &mask)
|
|
|
|
{
|
2023-08-17 14:24:58 +02:00
|
|
|
mask = readReg16(MCP23S17_POL_A);
|
2022-01-10 12:58:20 +01:00
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// mask = 0x0000..0xFFFF bit pattern
|
2022-01-10 12:58:20 +01:00
|
|
|
bool MCP23S17::setPullup16(uint16_t mask)
|
|
|
|
{
|
2023-08-17 14:24:58 +02:00
|
|
|
writeReg16(MCP23S17_PUR_A, mask);
|
2022-01-10 12:58:20 +01:00
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-28 12:56:28 +02:00
|
|
|
// mask = 0x0000..0xFFFF bit pattern
|
2022-01-10 12:58:20 +01:00
|
|
|
bool MCP23S17::getPullup16(uint16_t &mask)
|
|
|
|
{
|
2023-08-17 14:24:58 +02:00
|
|
|
mask = readReg16(MCP23S17_PUR_A);
|
2022-01-10 12:58:20 +01:00
|
|
|
if (_error != MCP23S17_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-07 20:23:20 +01:00
|
|
|
int MCP23S17::lastError()
|
|
|
|
{
|
|
|
|
int e = _error;
|
2022-09-28 12:56:28 +02:00
|
|
|
_error = MCP23S17_OK; // reset error after read.
|
2022-01-07 20:23:20 +01:00
|
|
|
return e;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-10-24 11:13:46 +02:00
|
|
|
void MCP23S17::enableControlRegister(uint8_t mask)
|
|
|
|
{
|
|
|
|
uint8_t reg = readReg(MCP23S17_IOCR);
|
|
|
|
reg |= mask;
|
|
|
|
writeReg(MCP23S17_IOCR, reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void MCP23S17::disableControlRegister(uint8_t mask)
|
|
|
|
{
|
|
|
|
uint8_t reg = readReg(MCP23S17_IOCR);
|
|
|
|
reg &= ~mask;
|
|
|
|
writeReg(MCP23S17_IOCR, reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-08-14 12:56:15 +02:00
|
|
|
void MCP23S17::enableHardwareAddress()
|
|
|
|
{
|
|
|
|
enableControlRegister(MCP23S17_IOCR_HAEN);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void MCP23S17::disableHardwareAddress()
|
|
|
|
{
|
|
|
|
disableControlRegister(MCP23S17_IOCR_HAEN);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if defined(ESP32)
|
|
|
|
|
|
|
|
void MCP23S17::selectHSPI()
|
|
|
|
{
|
|
|
|
_useHSPI = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void MCP23S17::selectVSPI()
|
|
|
|
{
|
|
|
|
_useHSPI = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool MCP23S17::usesHSPI()
|
|
|
|
{
|
|
|
|
return _useHSPI;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool MCP23S17::usesVSPI()
|
|
|
|
{
|
|
|
|
return !_useHSPI;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void MCP23S17::setGPIOpins(uint8_t clk, uint8_t miso, uint8_t mosi, uint8_t select)
|
|
|
|
{
|
|
|
|
_clock = clk;
|
|
|
|
_dataOut = mosi;
|
|
|
|
_dataIn = miso;
|
|
|
|
_select = select;
|
|
|
|
pinMode(_select, OUTPUT);
|
|
|
|
digitalWrite(_select, HIGH);
|
|
|
|
|
|
|
|
_mySPI->end(); // disable old SPI
|
|
|
|
|
|
|
|
_mySPI->begin(clk, miso, mosi, select); // enable new pins
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
2022-01-07 20:23:20 +01:00
|
|
|
////////////////////////////////////////////////////
|
|
|
|
//
|
2022-09-28 12:56:28 +02:00
|
|
|
// PRIVATE
|
2022-01-10 12:58:20 +01:00
|
|
|
//
|
2022-01-07 20:23:20 +01:00
|
|
|
|
2022-10-24 11:13:46 +02:00
|
|
|
|
|
|
|
// low level read / write masks
|
|
|
|
#define MCP23S17_WRITE_REG 0x40
|
|
|
|
#define MCP23S17_READ_REG 0x41
|
|
|
|
|
|
|
|
|
2022-01-07 20:23:20 +01:00
|
|
|
bool MCP23S17::writeReg(uint8_t reg, uint8_t value)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_OK;
|
|
|
|
|
2022-01-10 12:58:20 +01:00
|
|
|
if (reg > MCP23S17_OLAT_B)
|
2022-01-07 20:23:20 +01:00
|
|
|
{
|
2022-07-01 12:10:38 +02:00
|
|
|
_error = MCP23S17_REGISTER_ERROR;
|
2022-01-07 20:23:20 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
::digitalWrite(_select, LOW);
|
|
|
|
if (_hwSPI)
|
|
|
|
{
|
2022-07-01 12:10:38 +02:00
|
|
|
_mySPI->beginTransaction(_spi_settings);
|
|
|
|
// _address already shifted
|
2023-02-04 16:29:14 +01:00
|
|
|
_mySPI->transfer(MCP23S17_WRITE_REG | _address );
|
2022-07-01 12:10:38 +02:00
|
|
|
_mySPI->transfer(reg);
|
|
|
|
_mySPI->transfer(value);
|
|
|
|
_mySPI->endTransaction();
|
2022-01-07 20:23:20 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-07-01 12:10:38 +02:00
|
|
|
// _address already shifted
|
|
|
|
swSPI_transfer(MCP23S17_WRITE_REG | _address );
|
2022-01-07 20:23:20 +01:00
|
|
|
swSPI_transfer(reg);
|
|
|
|
swSPI_transfer(value);
|
|
|
|
}
|
|
|
|
::digitalWrite(_select, HIGH);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t MCP23S17::readReg(uint8_t reg)
|
|
|
|
{
|
|
|
|
uint8_t rv = 0;
|
|
|
|
|
|
|
|
_error = MCP23S17_OK;
|
|
|
|
|
2022-01-10 12:58:20 +01:00
|
|
|
if (reg > MCP23S17_OLAT_B)
|
2022-01-07 20:23:20 +01:00
|
|
|
{
|
2022-07-01 12:10:38 +02:00
|
|
|
_error = MCP23S17_REGISTER_ERROR;
|
2022-01-07 20:23:20 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
::digitalWrite(_select, LOW);
|
|
|
|
if (_hwSPI)
|
|
|
|
{
|
2022-07-01 12:10:38 +02:00
|
|
|
_mySPI->beginTransaction(_spi_settings);
|
|
|
|
// _address already shifted
|
|
|
|
_mySPI->transfer(MCP23S17_READ_REG | _address );
|
|
|
|
_mySPI->transfer(reg);
|
|
|
|
rv = _mySPI->transfer(0xFF);
|
|
|
|
_mySPI->endTransaction();
|
2022-01-07 20:23:20 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-07-01 12:10:38 +02:00
|
|
|
// _address already shifted
|
|
|
|
swSPI_transfer(MCP23S17_READ_REG | _address );
|
2022-01-07 20:23:20 +01:00
|
|
|
swSPI_transfer(reg);
|
|
|
|
rv = swSPI_transfer(0xFF);
|
|
|
|
}
|
|
|
|
::digitalWrite(_select, HIGH);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-08-17 14:24:58 +02:00
|
|
|
// writes HIGH byte first, LOW byte last
|
|
|
|
bool MCP23S17::writeReg16(uint8_t reg, uint16_t value)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_OK;
|
|
|
|
|
|
|
|
if (reg > MCP23S17_OLAT_B)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_REGISTER_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
::digitalWrite(_select, LOW);
|
|
|
|
if (_hwSPI)
|
|
|
|
{
|
|
|
|
_mySPI->beginTransaction(_spi_settings);
|
|
|
|
// _address already shifted
|
|
|
|
_mySPI->transfer(MCP23S17_WRITE_REG | _address );
|
|
|
|
_mySPI->transfer(reg);
|
|
|
|
_mySPI->transfer(value >> 8);
|
|
|
|
::digitalWrite(_select, HIGH);
|
|
|
|
|
|
|
|
::digitalWrite(_select, LOW);
|
|
|
|
_mySPI->transfer(MCP23S17_WRITE_REG | _address );
|
|
|
|
_mySPI->transfer(reg + 1);
|
|
|
|
_mySPI->transfer(value & 0xFF);
|
|
|
|
_mySPI->endTransaction();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// _address already shifted
|
|
|
|
swSPI_transfer(MCP23S17_WRITE_REG | _address );
|
|
|
|
swSPI_transfer(reg);
|
|
|
|
swSPI_transfer(value >> 8);
|
|
|
|
::digitalWrite(_select, HIGH);
|
|
|
|
|
|
|
|
::digitalWrite(_select, LOW);
|
|
|
|
swSPI_transfer(MCP23S17_WRITE_REG | _address );
|
|
|
|
swSPI_transfer(reg + 1);
|
|
|
|
swSPI_transfer(value & 0xFF);
|
|
|
|
}
|
|
|
|
::digitalWrite(_select, HIGH);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint16_t MCP23S17::readReg16(uint8_t reg)
|
|
|
|
{
|
|
|
|
uint16_t rv = 0;
|
|
|
|
|
|
|
|
_error = MCP23S17_OK;
|
|
|
|
|
|
|
|
if (reg > MCP23S17_OLAT_B)
|
|
|
|
{
|
|
|
|
_error = MCP23S17_REGISTER_ERROR;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
::digitalWrite(_select, LOW);
|
|
|
|
if (_hwSPI)
|
|
|
|
{
|
|
|
|
_mySPI->beginTransaction(_spi_settings);
|
|
|
|
// _address already shifted
|
|
|
|
_mySPI->transfer(MCP23S17_READ_REG | _address );
|
|
|
|
_mySPI->transfer(reg);
|
|
|
|
rv = _mySPI->transfer(0xFF) << 8;
|
|
|
|
::digitalWrite(_select, HIGH);
|
|
|
|
|
|
|
|
::digitalWrite(_select, LOW);
|
|
|
|
_mySPI->transfer(MCP23S17_READ_REG | _address );
|
|
|
|
_mySPI->transfer(reg + 1);
|
|
|
|
rv += _mySPI->transfer(0xFF);
|
|
|
|
_mySPI->endTransaction();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// _address already shifted
|
|
|
|
swSPI_transfer(MCP23S17_READ_REG | _address );
|
|
|
|
swSPI_transfer(reg);
|
|
|
|
rv = swSPI_transfer(0xFF) << 8;
|
|
|
|
::digitalWrite(_select, HIGH);
|
|
|
|
|
|
|
|
::digitalWrite(_select, LOW);
|
|
|
|
swSPI_transfer(MCP23S17_READ_REG | _address );
|
|
|
|
swSPI_transfer(reg + 1);
|
|
|
|
rv += swSPI_transfer(0xFF);
|
|
|
|
}
|
|
|
|
::digitalWrite(_select, HIGH);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-07 20:23:20 +01:00
|
|
|
uint8_t MCP23S17::swSPI_transfer(uint8_t val)
|
|
|
|
{
|
|
|
|
uint8_t clk = _clock;
|
|
|
|
uint8_t dao = _dataOut;
|
|
|
|
uint8_t dai = _dataIn;
|
|
|
|
|
|
|
|
uint8_t rv = 0;
|
|
|
|
for (uint8_t mask = 0x80; mask > 0; mask >>= 1)
|
|
|
|
{
|
2022-04-14 10:53:15 +02:00
|
|
|
::digitalWrite(dao, (val & mask) ? HIGH : LOW);
|
2022-01-07 20:23:20 +01:00
|
|
|
::digitalWrite(clk, HIGH);
|
|
|
|
if (::digitalRead(dai) == HIGH) rv |= mask;
|
|
|
|
::digitalWrite(clk, LOW);
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-02-04 16:29:14 +01:00
|
|
|
// -- END OF FILE --
|
2022-01-07 20:23:20 +01:00
|
|
|
|