ESPHome 2025.5.0
Loading...
Searching...
No Matches
midea_protocol.cpp
Go to the documentation of this file.
1#include "midea_protocol.h"
2#include "esphome/core/log.h"
3
4namespace esphome {
5namespace remote_base {
6
7static const char *const TAG = "remote.midea";
8
9static const int32_t TICK_US = 560;
10static const int32_t HEADER_MARK_US = 8 * TICK_US;
11static const int32_t HEADER_SPACE_US = 8 * TICK_US;
12static const int32_t BIT_MARK_US = 1 * TICK_US;
13static const int32_t BIT_ONE_SPACE_US = 3 * TICK_US;
14static const int32_t BIT_ZERO_SPACE_US = 1 * TICK_US;
15static const int32_t FOOTER_MARK_US = 1 * TICK_US;
16static const int32_t FOOTER_SPACE_US = 10 * TICK_US;
17
18uint8_t MideaData::calc_cs_() const {
19 uint8_t cs = 0;
20 for (uint8_t idx = 0; idx < OFFSET_CS; idx++)
21 cs -= reverse_bits(this->data_[idx]);
22 return reverse_bits(cs);
23}
24
25bool MideaData::is_compliment(const MideaData &rhs) const {
26 return std::equal(this->data_.begin(), this->data_.end(), rhs.data_.begin(),
27 [](const uint8_t &a, const uint8_t &b) { return a + b == 255; });
28}
29
31 dst->set_carrier_frequency(38000);
32 dst->reserve(2 + 48 * 2 + 2 + 2 + 48 * 2 + 1);
33 dst->item(HEADER_MARK_US, HEADER_SPACE_US);
34 for (unsigned idx = 0; idx < 6; idx++) {
35 for (uint8_t mask = 1 << 7; mask; mask >>= 1)
36 dst->item(BIT_MARK_US, (src[idx] & mask) ? BIT_ONE_SPACE_US : BIT_ZERO_SPACE_US);
37 }
38 dst->item(FOOTER_MARK_US, FOOTER_SPACE_US);
39 dst->item(HEADER_MARK_US, HEADER_SPACE_US);
40 for (unsigned idx = 0; idx < 6; idx++) {
41 for (uint8_t mask = 1 << 7; mask; mask >>= 1)
42 dst->item(BIT_MARK_US, (src[idx] & mask) ? BIT_ZERO_SPACE_US : BIT_ONE_SPACE_US);
43 }
44 dst->mark(FOOTER_MARK_US);
45}
46
47static bool decode_data(RemoteReceiveData &src, MideaData &dst) {
48 for (unsigned idx = 0; idx < 6; idx++) {
49 uint8_t data = 0;
50 for (uint8_t mask = 1 << 7; mask; mask >>= 1) {
51 if (!src.expect_mark(BIT_MARK_US))
52 return false;
53 if (src.expect_space(BIT_ONE_SPACE_US)) {
54 data |= mask;
55 } else if (!src.expect_space(BIT_ZERO_SPACE_US)) {
56 return false;
57 }
58 }
59 dst[idx] = data;
60 }
61 return true;
62}
63
65 MideaData out, inv;
66 if (src.expect_item(HEADER_MARK_US, HEADER_SPACE_US) && decode_data(src, out) && out.is_valid() &&
67 src.expect_item(FOOTER_MARK_US, FOOTER_SPACE_US) && src.expect_item(HEADER_MARK_US, HEADER_SPACE_US) &&
68 decode_data(src, inv) && src.expect_mark(FOOTER_MARK_US) && out.is_compliment(inv))
69 return out;
70 return {};
71}
72
73void MideaProtocol::dump(const MideaData &data) { ESP_LOGI(TAG, "Received Midea: %s", data.to_string().c_str()); }
74
75} // namespace remote_base
76} // namespace esphome
std::array< uint8_t, 6 > data_
static const uint8_t OFFSET_CS
bool is_compliment(const MideaData &rhs) const
void dump(const MideaData &data) override
void encode(RemoteTransmitData *dst, const MideaData &src) override
optional< MideaData > decode(RemoteReceiveData src) override
bool expect_item(uint32_t mark, uint32_t space)
void set_carrier_frequency(uint32_t carrier_frequency)
Definition remote_base.h:34
void item(uint32_t mark, uint32_t space)
Definition remote_base.h:29
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
uint8_t reverse_bits(uint8_t x)
Reverse the order of 8 bits.
Definition helpers.h:231