Files
iot-objets-connectes/T-Display-S3-main/lib/PCA95x5/PCA95x5.h

188 lines
4.6 KiB
C
Raw Normal View History

2026-03-31 13:17:21 +02:00
/*
* @Description: None
* @version: V1.0.0
* @Author: None
* @Date: 2023-07-14 09:26:36
* @LastEditors: LILYGO_L
* @LastEditTime: 2023-07-14 09:45:19
* @License: GPL 3.0
*/
#include <Arduino.h>
#include <Wire.h>
namespace PCA95x5 {
namespace Reg {
enum : uint8_t {
INPUT_PORT_0,
INPUT_PORT_1,
OUTPUT_PORT_0,
OUTPUT_PORT_1,
POLARITY_INVERSION_PORT_0,
POLARITY_INVERSION_PORT_1,
CONFIGURATION_PORT_0,
CONFIGURATION_PORT_1,
};
}
namespace Port {
enum Port : uint8_t {
P00,
P01,
P02,
P03,
P04,
P05,
P06,
P07,
P10,
P11,
P12,
P13,
P14,
P15,
P16,
P17,
};
} // namespace Port
namespace Level {
enum Level : uint8_t { L, H };
enum LevelAll : uint16_t { L_ALL = 0x0000, H_ALL = 0xFFFF };
} // namespace Level
namespace Polarity {
enum Polarity : uint8_t { ORIGINAL, INVERTED };
enum PolarityAll : uint16_t { ORIGINAL_ALL = 0x0000, INVERTED_ALL = 0xFFFF };
} // namespace Polarity
namespace Direction {
enum Direction : uint8_t { OUT, IN };
enum DirectionAll : uint16_t { OUT_ALL = 0x0000, IN_ALL = 0xFFFF };
} // namespace Direction
template <typename WireType = TwoWire>
class PCA95x5 {
union Ports {
uint16_t w;
uint8_t b[2];
};
static constexpr uint8_t BASE_I2C_ADDR = 0x20;
WireType* wire {nullptr};
uint8_t addr {BASE_I2C_ADDR};
Ports input {0x0000};
Ports output {0xFFFF};
Ports pol {0x0000};
Ports dir {0xFFFF};
uint8_t status {0x00};
public:
void attach(WireType& wire, uint8_t i2c_addr = BASE_I2C_ADDR) {
this->wire = &wire;
this->addr = i2c_addr;
}
uint16_t read() {
read_bytes(this->addr, Reg::INPUT_PORT_0, this->input.b, 2);
return this->input.w;
}
Level::Level read(const Port::Port port) {
uint16_t v = read();
return (v & (1 << port)) ? Level::H : Level::L;
}
Level::Level read(uint8_t port) {
uint16_t v = read();
return (v & (1 << port)) ? Level::H : Level::L;
}
bool write(const uint16_t value) {
this->output.w = value;
return write_impl();
}
bool write(const Port::Port port, const Level::Level level) {
if (level == Level::H) {
this->output.w |= (1 << port);
} else {
this->output.w &= ~(1 << port);
}
return write_impl();
}
bool write(uint8_t port, uint8_t level) {
if (level == Level::H) {
this->output.w |= (1 << port);
} else {
this->output.w &= ~(1 << port);
}
return write_impl();
}
bool polarity(const uint16_t value) {
this->pol.w = value;
return polarity_impl();
}
bool polarity(const Port::Port port, const Polarity::Polarity pol) {
if (pol == Polarity::INVERTED) {
this->pol.w |= (1 << port);
} else {
this->pol.w &= ~(1 << port);
}
return polarity_impl();
}
bool direction(const uint16_t value) {
this->dir.w = value;
return direction_impl();
}
bool direction(const Port::Port port, const Direction::Direction dir) {
if (dir == Direction::IN) {
this->dir.w |= (1 << port);
} else {
this->dir.w &= ~(1 << port);
}
return direction_impl();
}
uint8_t i2c_error() const {
return status;
}
private:
bool write_impl() {
return write_bytes(this->addr, Reg::OUTPUT_PORT_0, this->output.b, 2);
}
bool polarity_impl() {
return write_bytes(this->addr, Reg::POLARITY_INVERSION_PORT_0, this->pol.b, 2);
}
bool direction_impl() {
return write_bytes(this->addr, Reg::CONFIGURATION_PORT_0, this->dir.b, 2);
}
int8_t read_bytes(const uint8_t dev, const uint8_t reg, uint8_t* data, const uint8_t size) {
wire->beginTransmission(dev);
wire->write(reg);
wire->endTransmission();
wire->requestFrom(dev, size);
int8_t count = 0;
while (wire->available()) data[count++] = wire->read();
return count;
}
bool write_bytes(const uint8_t dev, const uint8_t reg, const uint8_t* data, const uint8_t size) {
wire->beginTransmission(dev);
wire->write(reg);
for (uint8_t i = 0; i < size; i++) wire->write(data[i]);
status = wire->endTransmission();
return (status == 0);
}
};
} // namespace PCA95x5
using PCA9535 = PCA95x5::PCA95x5<>;
using PCA9555 = PCA95x5::PCA95x5<>;