ESPHome 2026.5.0
Loading...
Searching...
No Matches
qmc5883l.cpp
Go to the documentation of this file.
1#include "qmc5883l.h"
3#include "esphome/core/log.h"
4#include "esphome/core/hal.h"
5#include <cmath>
6
8
9static const char *const TAG = "qmc5883l";
10
11static const uint8_t QMC5883L_ADDRESS = 0x0D;
12
13static const uint8_t QMC5883L_REGISTER_DATA_X_LSB = 0x00;
14static const uint8_t QMC5883L_REGISTER_DATA_X_MSB = 0x01;
15static const uint8_t QMC5883L_REGISTER_DATA_Y_LSB = 0x02;
16static const uint8_t QMC5883L_REGISTER_DATA_Y_MSB = 0x03;
17static const uint8_t QMC5883L_REGISTER_DATA_Z_LSB = 0x04;
18static const uint8_t QMC5883L_REGISTER_DATA_Z_MSB = 0x05;
19static const uint8_t QMC5883L_REGISTER_STATUS = 0x06;
20static const uint8_t QMC5883L_REGISTER_TEMPERATURE_LSB = 0x07;
21static const uint8_t QMC5883L_REGISTER_TEMPERATURE_MSB = 0x08;
22static const uint8_t QMC5883L_REGISTER_CONTROL_1 = 0x09;
23static const uint8_t QMC5883L_REGISTER_CONTROL_2 = 0x0A;
24static const uint8_t QMC5883L_REGISTER_PERIOD = 0x0B;
25
27
29 // Soft Reset
30 if (!this->write_byte(QMC5883L_REGISTER_CONTROL_2, 1 << 7)) {
32 this->mark_failed();
33 return;
34 }
35 delay(10);
36
37 if (this->drdy_pin_) {
38 this->drdy_pin_->setup();
39 if (this->drdy_pin_->is_internal()) {
40 static_cast<InternalGPIOPin *>(this->drdy_pin_)
42 this->drdy_use_isr_ = true;
43 this->stop_poller();
44 }
45 }
46
47 uint8_t control_1 = 0;
48 control_1 |= 0b01 << 0; // MODE (Mode) -> 0b00=standby, 0b01=continuous
49 control_1 |= this->datarate_ << 2;
50 control_1 |= this->range_ << 4;
51 control_1 |= this->oversampling_ << 6;
52 if (!this->write_byte(QMC5883L_REGISTER_CONTROL_1, control_1)) {
54 this->mark_failed();
55 return;
56 }
57
58 uint8_t control_2 = 0;
59 control_2 |= 0b0 << 7; // SOFT_RST (Soft Reset) -> 0b00=disabled, 0b01=enabled
60 control_2 |= 0b0 << 6; // ROL_PNT (Pointer Roll Over) -> 0b00=disabled, 0b01=enabled
61 control_2 |= 0b0 << 0; // INT_ENB (Interrupt) -> 0b00=disabled, 0b01=enabled
62 if (!this->write_byte(QMC5883L_REGISTER_CONTROL_2, control_2)) {
64 this->mark_failed();
65 return;
66 }
67
68 uint8_t period = 0x01; // recommended value
69 if (!this->write_byte(QMC5883L_REGISTER_PERIOD, period)) {
71 this->mark_failed();
72 return;
73 }
74
75 if (!this->drdy_use_isr_ && this->get_update_interval() < App.get_loop_interval()) {
76 this->high_freq_.start();
77 }
78}
79
81 ESP_LOGCONFIG(TAG, "QMC5883L:");
82 LOG_I2C_DEVICE(this);
83 if (this->error_code_ == COMMUNICATION_FAILED) {
84 ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
85 }
86 LOG_UPDATE_INTERVAL(this);
87
88 LOG_SENSOR(" ", "X Axis", this->x_sensor_);
89 LOG_SENSOR(" ", "Y Axis", this->y_sensor_);
90 LOG_SENSOR(" ", "Z Axis", this->z_sensor_);
91 LOG_SENSOR(" ", "Heading", this->heading_sensor_);
92 LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
93 LOG_PIN(" DRDY Pin: ", this->drdy_pin_);
94 if (this->drdy_pin_ != nullptr) {
95 ESP_LOGCONFIG(TAG, " DRDY mode: %s",
96 this->drdy_use_isr_ ? LOG_STR_LITERAL("interrupt") : LOG_STR_LITERAL("polling"));
97 }
98}
99
101 // If DRDY is on an external expander we keep the polling path and early-return
102 // if data is not ready yet. Internal DRDY pins take the ISR path via loop().
103 if (this->drdy_pin_ && !this->drdy_pin_->digital_read()) {
104 return;
105 }
106 this->read_sensor_();
107}
108
110 this->disable_loop();
111 if (!this->drdy_use_isr_ || !this->drdy_pin_->digital_read()) {
112 return;
113 }
114 this->read_sensor_();
115}
116
118 i2c::ErrorCode err;
119 uint8_t status = false;
120
121 // Status byte gets cleared when data is read, so we have to read this first.
122 // If status and two axes are desired, it's possible to save one byte of traffic by enabling
123 // ROL_PNT in setup and reading 7 bytes starting at the status register.
124 // If status and all three axes are desired, using ROL_PNT saves you 3 bytes.
125 // But simply not reading status saves you 4 bytes always and is much simpler.
126 if (ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE) {
127 err = this->read_register(QMC5883L_REGISTER_STATUS, &status, 1);
128 if (err != i2c::ERROR_OK) {
129 char buf[32];
130 snprintf(buf, sizeof(buf), "status read failed (%d)", err);
131 this->status_set_warning(buf);
132 return;
133 }
134 }
135
136 uint16_t raw[3] = {0};
137 // Z must always be requested, otherwise the data registers will remain locked against updates.
138 // Skipping the Y axis if X and Z are needed actually requires an additional byte of comms.
139 // Starting partway through the axes does save you traffic.
140 uint8_t start, dest;
141 if (this->heading_sensor_ != nullptr || this->x_sensor_ != nullptr) {
142 start = QMC5883L_REGISTER_DATA_X_LSB;
143 dest = 0;
144 } else if (this->y_sensor_ != nullptr) {
145 start = QMC5883L_REGISTER_DATA_Y_LSB;
146 dest = 1;
147 } else {
148 start = QMC5883L_REGISTER_DATA_Z_LSB;
149 dest = 2;
150 }
151 err = this->read_bytes_16_le_(start, &raw[dest], 3 - dest);
152 if (err != i2c::ERROR_OK) {
153 char buf[32];
154 snprintf(buf, sizeof(buf), "mag read failed (%d)", err);
155 this->status_set_warning(buf);
156 return;
157 }
158
159 float mg_per_bit;
160 switch (this->range_) {
162 mg_per_bit = 0.0833f;
163 break;
165 mg_per_bit = 0.333f;
166 break;
167 default:
168 mg_per_bit = NAN;
169 }
170
171 // in µT
172 const float x = int16_t(raw[0]) * mg_per_bit * 0.1f;
173 const float y = int16_t(raw[1]) * mg_per_bit * 0.1f;
174 const float z = int16_t(raw[2]) * mg_per_bit * 0.1f;
175
176 float heading = atan2f(0.0f - x, y) * 180.0f / M_PI;
177
178 float temp = NAN;
179 if (this->temperature_sensor_ != nullptr) {
180 uint16_t raw_temp;
181 err = this->read_bytes_16_le_(QMC5883L_REGISTER_TEMPERATURE_LSB, &raw_temp);
182 if (err != i2c::ERROR_OK) {
183 char buf[32];
184 snprintf(buf, sizeof(buf), "temp read failed (%d)", err);
185 this->status_set_warning(buf);
186 return;
187 }
188 temp = int16_t(raw_temp) * 0.01f;
189 }
190
191 ESP_LOGV(TAG, "Got x=%0.02fµT y=%0.02fµT z=%0.02fµT heading=%0.01f° temperature=%0.01f°C status=%u", x, y, z, heading,
192 temp, status);
193
194 if (this->x_sensor_ != nullptr)
195 this->x_sensor_->publish_state(x);
196 if (this->y_sensor_ != nullptr)
197 this->y_sensor_->publish_state(y);
198 if (this->z_sensor_ != nullptr)
199 this->z_sensor_->publish_state(z);
200 if (this->heading_sensor_ != nullptr)
201 this->heading_sensor_->publish_state(heading);
202 if (this->temperature_sensor_ != nullptr)
204}
205
206i2c::ErrorCode QMC5883LComponent::read_bytes_16_le_(uint8_t a_register, uint16_t *data, uint8_t len) {
207 i2c::ErrorCode err = this->read_register(a_register, reinterpret_cast<uint8_t *>(data), len * 2);
208 if (err != i2c::ERROR_OK)
209 return err;
210 for (size_t i = 0; i < len; i++)
211 data[i] = convert_little_endian(data[i]);
212 return err;
213}
214
215} // namespace esphome::qmc5883l
uint8_t raw[35]
Definition bl0939.h:0
uint8_t status
Definition bl0942.h:8
uint32_t get_loop_interval() const
void mark_failed()
Mark this component as failed.
void enable_loop_soon_any_context()
Thread and ISR-safe version of enable_loop() that can be called from any context.
void disable_loop()
Disable this component's loop.
virtual void setup()=0
virtual ESPDEPRECATED("Override dump_summary(char*, size_t) instead. Will be removed in 2026.7.0.", "2026.1.0") virtual std boo is_internal)()
Get a summary of this pin as a string.
Definition gpio.h:88
virtual bool digital_read()=0
void start()
Start running the loop continuously.
Definition helpers.cpp:730
virtual uint32_t get_update_interval() const
Get the update interval in ms of this sensor.
bool write_byte(uint8_t a_register, uint8_t data) const
Definition i2c.h:265
ErrorCode read_register(uint8_t a_register, uint8_t *data, size_t len)
reads an array of bytes from a specific register in the I²C device
Definition i2c.cpp:25
sensor::Sensor * temperature_sensor_
Definition qmc5883l.h:57
enum esphome::qmc5883l::QMC5883LComponent::ErrorCode error_code_
i2c::ErrorCode read_bytes_16_le_(uint8_t a_register, uint16_t *data, uint8_t len=1)
Definition qmc5883l.cpp:206
QMC5883LOversampling oversampling_
Definition qmc5883l.h:52
static void IRAM_ATTR gpio_intr(QMC5883LComponent *arg)
Definition qmc5883l.cpp:26
HighFrequencyLoopRequester high_freq_
Definition qmc5883l.h:65
void publish_state(float state)
Publish a new state to the front-end.
Definition sensor.cpp:68
u_int8_t raw_temp
bool z
Definition msa3xx.h:1
@ INTERRUPT_RISING_EDGE
Definition gpio.h:50
ErrorCode
Error codes returned by I2CBus and I2CDevice methods.
Definition i2c_bus.h:12
@ ERROR_OK
No error found during execution of method.
Definition i2c_bus.h:14
constexpr T convert_little_endian(T val)
Convert a value between host byte order and little endian (least significant byte first) order.
Definition helpers.h:924
std::string size_t len
void HOT delay(uint32_t ms)
Definition hal.cpp:82
Application App
Global storage of Application pointer - only one Application can exist.
uint16_t x
Definition tt21100.cpp:5
uint16_t y
Definition tt21100.cpp:6