ESPHome 2025.9.0-dev
Loading...
Searching...
No Matches
pca9685_output.cpp
Go to the documentation of this file.
1#include "pca9685_output.h"
2#include "esphome/core/hal.h"
4#include "esphome/core/log.h"
5
6namespace esphome {
7namespace pca9685 {
8
9static const char *const TAG = "pca9685";
10
11const uint8_t PCA9685_MODE_INVERTED = 0x10;
12const uint8_t PCA9685_MODE_OUTPUT_ONACK = 0x08;
13const uint8_t PCA9685_MODE_OUTPUT_TOTEM_POLE = 0x04;
14const uint8_t PCA9685_MODE_OUTNE_HIGHZ = 0x02;
15const uint8_t PCA9685_MODE_OUTNE_LOW = 0x01;
16
17static const uint8_t PCA9685_REGISTER_SOFTWARE_RESET = 0x06;
18static const uint8_t PCA9685_REGISTER_MODE1 = 0x00;
19static const uint8_t PCA9685_REGISTER_MODE2 = 0x01;
20static const uint8_t PCA9685_REGISTER_LED0 = 0x06;
21static const uint8_t PCA9685_REGISTER_PRE_SCALE = 0xFE;
22
23static const uint8_t PCA9685_MODE1_RESTART = 0b10000000;
24static const uint8_t PCA9685_MODE1_EXTCLK = 0b01000000;
25static const uint8_t PCA9685_MODE1_AUTOINC = 0b00100000;
26static const uint8_t PCA9685_MODE1_SLEEP = 0b00010000;
27
29 ESP_LOGV(TAG, " Resetting devices");
30 if (!this->write_bytes(PCA9685_REGISTER_SOFTWARE_RESET, nullptr, 0)) {
31 this->mark_failed();
32 return;
33 }
34
35 if (!this->write_byte(PCA9685_REGISTER_MODE1, PCA9685_MODE1_RESTART | PCA9685_MODE1_AUTOINC)) {
36 this->mark_failed();
37 return;
38 }
39 if (!this->write_byte(PCA9685_REGISTER_MODE2, this->mode_)) {
40 this->mark_failed();
41 return;
42 }
43
44 uint8_t mode1;
45 if (!this->read_byte(PCA9685_REGISTER_MODE1, &mode1)) {
46 this->mark_failed();
47 return;
48 }
49 mode1 = (mode1 & ~PCA9685_MODE1_RESTART) | PCA9685_MODE1_SLEEP;
50 if (!this->write_byte(PCA9685_REGISTER_MODE1, mode1)) {
51 this->mark_failed();
52 return;
53 }
54
55 int pre_scaler = 3;
56 if (this->extclk_) {
57 mode1 = mode1 | PCA9685_MODE1_EXTCLK;
58 if (!this->write_byte(PCA9685_REGISTER_MODE1, mode1)) {
59 this->mark_failed();
60 return;
61 }
62 } else {
63 pre_scaler = static_cast<int>((25000000 / (4096 * this->frequency_)) - 1);
64 pre_scaler = clamp(pre_scaler, 3, 255);
65
66 ESP_LOGV(TAG, " -> Prescaler: %d", pre_scaler);
67 }
68 if (!this->write_byte(PCA9685_REGISTER_PRE_SCALE, pre_scaler)) {
69 this->mark_failed();
70 return;
71 }
72
73 mode1 = (mode1 & ~PCA9685_MODE1_SLEEP) | PCA9685_MODE1_RESTART;
74 if (!this->write_byte(PCA9685_REGISTER_MODE1, mode1)) {
75 this->mark_failed();
76 return;
77 }
79
80 this->loop();
81}
82
84 ESP_LOGCONFIG(TAG,
85 "PCA9685:\n"
86 " Mode: 0x%02X",
87 this->mode_);
88 if (this->extclk_) {
89 ESP_LOGCONFIG(TAG, " EXTCLK: enabled");
90 } else {
91 ESP_LOGCONFIG(TAG,
92 " EXTCLK: disabled\n"
93 " Frequency: %.0f Hz",
94 this->frequency_);
95 }
96 if (this->is_failed()) {
97 ESP_LOGE(TAG, "Setting up PCA9685 failed!");
98 }
99}
100
102 if (this->min_channel_ == 0xFF || !this->update_)
103 return;
104
105 const uint16_t num_channels = this->max_channel_ - this->min_channel_ + 1;
106 const uint16_t phase_delta_begin = 4096 / num_channels;
107 for (uint8_t channel = this->min_channel_; channel <= this->max_channel_; channel++) {
108 uint16_t phase_begin = (channel - this->min_channel_) * phase_delta_begin;
109 uint16_t phase_end;
110 uint16_t amount = this->pwm_amounts_[channel];
111 if (amount == 0) {
112 phase_end = 4096;
113 } else if (amount >= 4096) {
114 phase_begin = 4096;
115 phase_end = 0;
116 } else {
117 phase_end = phase_begin + amount;
118 if (phase_end >= 4096)
119 phase_end -= 4096;
120 }
121
122 ESP_LOGVV(TAG, "Channel %02u: amount=%04u phase_begin=%04u phase_end=%04u", channel, amount, phase_begin,
123 phase_end);
124
125 uint8_t data[4];
126 data[0] = phase_begin & 0xFF;
127 data[1] = (phase_begin >> 8) & 0xFF;
128 data[2] = phase_end & 0xFF;
129 data[3] = (phase_end >> 8) & 0xFF;
130
131 uint8_t reg = PCA9685_REGISTER_LED0 + 4 * channel;
132 if (!this->write_bytes(reg, data, 4)) {
133 this->status_set_warning();
134 return;
135 }
136 }
137
138 this->status_clear_warning();
139 this->update_ = false;
140}
141
143 auto c = channel->channel_;
144 this->min_channel_ = std::min(this->min_channel_, c);
145 this->max_channel_ = std::max(this->max_channel_, c);
146 channel->set_parent(this);
147}
148
150 const uint16_t max_duty = 4096;
151 const float duty_rounded = roundf(state * max_duty);
152 auto duty = static_cast<uint16_t>(duty_rounded);
153 this->parent_->set_channel_value_(this->channel_, duty);
154}
155
156} // namespace pca9685
157} // namespace esphome
virtual void mark_failed()
Mark this component as failed.
bool is_failed() const
void status_set_warning(const char *message=nullptr)
void status_clear_warning()
bool write_bytes(uint8_t a_register, const uint8_t *data, uint8_t len, bool stop=true)
Definition i2c.h:252
bool write_byte(uint8_t a_register, uint8_t data, bool stop=true)
Definition i2c.h:266
I2CRegister reg(uint8_t a_register)
calls the I2CRegister constructor
Definition i2c.h:153
bool read_byte(uint8_t a_register, uint8_t *data, bool stop=true)
Definition i2c.h:239
void write_state(float state) override
void set_parent(PCA9685Output *parent)
void set_channel_value_(uint8_t channel, uint16_t value)
void register_channel(PCA9685Channel *channel)
bool state
Definition fan.h:0
const uint8_t PCA9685_MODE_OUTPUT_ONACK
Channel update happens upon ACK (post-set) rather than on STOP (endTransmission)
const uint8_t PCA9685_MODE_OUTPUT_TOTEM_POLE
Use a totem-pole (push-pull) style output rather than an open-drain structure.
const uint8_t PCA9685_MODE_OUTNE_HIGHZ
For active low output enable, sets channel output to high-impedance state.
const uint8_t PCA9685_MODE_OUTNE_LOW
Similarly, sets channel output to high if in totem-pole mode, otherwise.
const uint8_t PCA9685_MODE_INVERTED
Inverts polarity of channel output signal.
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
void IRAM_ATTR HOT delayMicroseconds(uint32_t us)
Definition core.cpp:31