ESPHome 2026.4.3
Loading...
Searching...
No Matches
mcp23016.cpp
Go to the documentation of this file.
1#include "mcp23016.h"
2#include "esphome/core/log.h"
3#include <cstdio>
4
5namespace esphome {
6namespace mcp23016 {
7
8static const char *const TAG = "mcp23016";
9
11 uint16_t iocon;
12 // MCP23016 registers operate as paired 16-bit registers. Addressing the
13 // odd register (e.g. IOCON1) reads/writes that register first, then wraps
14 // to the even register (IOCON0) in the same pair. Starting from the odd
15 // address gives the correct byte order for 1 << pin mapping:
16 // high byte = port 1 (pins 8-15), low byte = port 0 (pins 0-7).
17 if (!this->read_reg_(MCP23016_IOCON1, &iocon)) {
18 this->mark_failed();
19 return;
20 }
21
22 // Read current output register state
23 this->read_reg_(MCP23016_OLAT1, &this->olat_);
24
25 // all pins input
26 this->write_reg_(MCP23016_IODIR1, 0xFFFF);
27
28 if (this->interrupt_pin_ != nullptr) {
29 this->interrupt_pin_->setup();
31 this->set_invalidate_on_read_(false);
32 }
33 this->disable_loop();
34}
35
38 // Invalidate cache at the start of each loop
39 this->reset_pin_cache_();
40 // Only disable the loop once INT has actually gone HIGH. Input transitions that straddle the
41 // I2C read leave INT asserted without re-firing a falling edge, which would strand us with
42 // stale state forever; keep looping until the line is released so we self-heal.
43 if (this->interrupt_pin_ != nullptr && this->interrupt_pin_->digital_read()) {
44 this->disable_loop();
45 }
46}
47bool MCP23016::digital_read_hw(uint8_t pin) { return this->read_reg_(MCP23016_GP1, &this->input_mask_); }
48
49bool MCP23016::digital_read_cache(uint8_t pin) { return this->input_mask_ & (1 << pin); }
50void MCP23016::digital_write_hw(uint8_t pin, bool value) { this->update_reg_(pin, value, MCP23016_OLAT1); }
52 if (flags == gpio::FLAG_INPUT) {
53 this->update_reg_(pin, true, MCP23016_IODIR1);
54 if (this->interrupt_pin_ == nullptr) {
55 this->enable_loop();
56 }
57 } else if (flags == gpio::FLAG_OUTPUT) {
58 this->update_reg_(pin, false, MCP23016_IODIR1);
59 }
60}
62bool MCP23016::read_reg_(uint8_t reg, uint16_t *value) {
63 if (this->is_failed())
64 return false;
65
66 return this->read_byte_16(reg, value);
67}
68bool MCP23016::write_reg_(uint8_t reg, uint16_t value) {
69 if (this->is_failed())
70 return false;
71
72 return this->write_byte_16(reg, value);
73}
74void MCP23016::update_reg_(uint8_t pin, bool pin_value, uint8_t reg_addr) {
75 uint16_t reg_value = 0;
76
77 if (reg_addr == MCP23016_OLAT1) {
78 reg_value = this->olat_;
79 } else {
80 this->read_reg_(reg_addr, &reg_value);
81 }
82
83 if (pin_value) {
84 reg_value |= 1 << pin;
85 } else {
86 reg_value &= ~(1 << pin);
87 }
88
89 this->write_reg_(reg_addr, reg_value);
90
91 if (reg_addr == MCP23016_OLAT1) {
92 this->olat_ = reg_value;
93 }
94}
95
98bool MCP23016GPIOPin::digital_read() { return this->parent_->digital_read(this->pin_) != this->inverted_; }
99void MCP23016GPIOPin::digital_write(bool value) { this->parent_->digital_write(this->pin_, value != this->inverted_); }
100size_t MCP23016GPIOPin::dump_summary(char *buffer, size_t len) const {
101 return buf_append_printf(buffer, len, 0, "%u via MCP23016", this->pin_);
102}
103
104} // namespace mcp23016
105} // namespace esphome
void mark_failed()
Mark this component as failed.
bool is_failed() const
Definition component.h:284
void enable_loop_soon_any_context()
Thread and ISR-safe version of enable_loop() that can be called from any context.
void enable_loop()
Enable this component's loop.
Definition component.h:258
void disable_loop()
Disable this component's loop.
virtual void setup()=0
virtual bool digital_read()=0
void attach_interrupt(void(*func)(T *), T *arg, gpio::InterruptType type) const
Definition gpio.h:107
bool digital_read(P pin)
Read the state of the given pin.
Definition cached_gpio.h:37
bool read_byte_16(uint8_t a_register, uint16_t *data)
Definition i2c.h:249
bool write_byte_16(uint8_t a_register, uint16_t data) const
Definition i2c.h:267
void pin_mode(gpio::Flags flags) override
Definition mcp23016.cpp:97
size_t dump_summary(char *buffer, size_t len) const override
Definition mcp23016.cpp:100
void digital_write(bool value) override
Definition mcp23016.cpp:99
void pin_mode(uint8_t pin, gpio::Flags flags)
Definition mcp23016.cpp:51
bool digital_read_cache(uint8_t pin) override
Definition mcp23016.cpp:49
InternalGPIOPin * interrupt_pin_
Definition mcp23016.h:57
void digital_write_hw(uint8_t pin, bool value) override
Definition mcp23016.cpp:50
static void IRAM_ATTR gpio_intr(MCP23016 *arg)
Definition mcp23016.cpp:36
bool digital_read_hw(uint8_t pin) override
Definition mcp23016.cpp:47
void update_reg_(uint8_t pin, bool pin_value, uint8_t reg_a)
Definition mcp23016.cpp:74
float get_setup_priority() const override
Definition mcp23016.cpp:61
bool read_reg_(uint8_t reg, uint16_t *value)
Definition mcp23016.cpp:62
bool write_reg_(uint8_t reg, uint16_t value)
Definition mcp23016.cpp:68
uint16_t flags
@ INTERRUPT_FALLING_EDGE
Definition gpio.h:51
@ FLAG_OUTPUT
Definition gpio.h:28
@ FLAG_INPUT
Definition gpio.h:27
constexpr float IO
For components that represent GPIO pins like PCF8573.
Definition component.h:38
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
std::string size_t len
Definition helpers.h:1045