ESPHome 2025.12.0-dev
Loading...
Searching...
No Matches
canbus.h
Go to the documentation of this file.
1#pragma once
2
6
7#include <cinttypes>
8#include <vector>
9
10namespace esphome {
11namespace canbus {
12
21
45
46class CanbusTrigger;
47template<typename... Ts> class CanbusSendAction;
48
49/* CAN payload length definitions according to ISO 11898-1 */
50static const uint8_t CAN_MAX_DATA_LENGTH = 8;
51
52/*
53Can Frame describes a normative CAN Frame
54The RTR = Remote Transmission Request is implemented in every CAN controller but rarely used
55So currently the flag is passed to and from the hardware but currently ignored to the user application.
56*/
57struct CanFrame {
58 bool use_extended_id = false;
60 uint32_t can_id; /* 29 or 11 bit CAN_ID */
61 uint8_t can_data_length_code; /* frame payload length in byte (0 .. CAN_MAX_DATA_LENGTH) */
62 uint8_t data[CAN_MAX_DATA_LENGTH] __attribute__((aligned(8)));
63};
64
65class Canbus : public Component {
66 public:
67 Canbus(){};
68 void setup() override;
69 void dump_config() override;
70 float get_setup_priority() const override { return setup_priority::HARDWARE; }
71 void loop() override;
72
73 canbus::Error send_data(uint32_t can_id, bool use_extended_id, bool remote_transmission_request,
74 const std::vector<uint8_t> &data);
75 canbus::Error send_data(uint32_t can_id, bool use_extended_id, const std::vector<uint8_t> &data) {
76 // for backwards compatibility only
77 return this->send_data(can_id, use_extended_id, false, data);
78 }
79 void set_can_id(uint32_t can_id) { this->can_id_ = can_id; }
80 void set_use_extended_id(bool use_extended_id) { this->use_extended_id_ = use_extended_id; }
81 void set_bitrate(CanSpeed bit_rate) { this->bit_rate_ = bit_rate; }
82
83 void add_trigger(CanbusTrigger *trigger);
95 std::function<void(uint32_t can_id, bool extended_id, bool rtr, const std::vector<uint8_t> &data)> callback) {
96 this->callback_manager_.add(std::move(callback));
97 }
98
99 protected:
100 template<typename... Ts> friend class CanbusSendAction;
101 std::vector<CanbusTrigger *> triggers_{};
102 uint32_t can_id_;
105 CallbackManager<void(uint32_t can_id, bool extended_id, bool rtr, const std::vector<uint8_t> &data)>
107
108 virtual bool setup_internal() = 0;
109 virtual Error send_message(struct CanFrame *frame) = 0;
110 virtual Error read_message(struct CanFrame *frame) = 0;
111};
112
113template<typename... Ts> class CanbusSendAction : public Action<Ts...>, public Parented<Canbus> {
114 public:
115 void set_data_template(std::vector<uint8_t> (*func)(Ts...)) {
116 // Stateless lambdas (generated by ESPHome) implicitly convert to function pointers
117 this->data_.func = func;
118 this->len_ = -1; // Sentinel value indicates template mode
119 }
120
121 // Store pointer to static data in flash (no RAM copy)
122 void set_data_static(const uint8_t *data, size_t len) {
123 this->data_.data = data;
124 this->len_ = len; // Length >= 0 indicates static mode
125 }
126
127 void set_can_id(uint32_t can_id) { this->can_id_ = can_id; }
128
129 void set_use_extended_id(bool use_extended_id) { this->use_extended_id_ = use_extended_id; }
130
131 void set_remote_transmission_request(bool remote_transmission_request) {
132 this->remote_transmission_request_ = remote_transmission_request;
133 }
134
135 void play(const Ts &...x) override {
136 auto can_id = this->can_id_.has_value() ? *this->can_id_ : this->parent_->can_id_;
137 auto use_extended_id =
138 this->use_extended_id_.has_value() ? *this->use_extended_id_ : this->parent_->use_extended_id_;
139 std::vector<uint8_t> data;
140 if (this->len_ >= 0) {
141 // Static mode: copy from flash to vector
142 data.assign(this->data_.data, this->data_.data + this->len_);
143 } else {
144 // Template mode: call function
145 data = this->data_.func(x...);
146 }
147 this->parent_->send_data(can_id, use_extended_id, this->remote_transmission_request_, data);
148 }
149
150 protected:
154 ssize_t len_{-1}; // -1 = template mode, >=0 = static mode with length
155 union Data {
156 std::vector<uint8_t> (*func)(Ts...); // Function pointer (stateless lambdas)
157 const uint8_t *data; // Pointer to static data in flash
159};
160
161class CanbusTrigger : public Trigger<std::vector<uint8_t>, uint32_t, bool>, public Component {
162 friend class Canbus;
163
164 public:
165 explicit CanbusTrigger(Canbus *parent, const std::uint32_t can_id, const std::uint32_t can_id_mask,
166 const bool use_extended_id)
167 : parent_(parent), can_id_(can_id), can_id_mask_(can_id_mask), use_extended_id_(use_extended_id){};
168
169 void set_remote_transmission_request(bool remote_transmission_request) {
170 this->remote_transmission_request_ = remote_transmission_request;
171 }
172
173 void setup() override { this->parent_->add_trigger(this); }
174
175 protected:
177 uint32_t can_id_;
178 uint32_t can_id_mask_;
181};
182
183} // namespace canbus
184} // namespace esphome
Helper class to easily give an object a parent of type T.
Definition helpers.h:918
CallbackManager< void(uint32_t can_id, bool extended_id, bool rtr, const std::vector< uint8_t > &data)> callback_manager_
Definition canbus.h:106
void loop() override
Definition canbus.cpp:68
void set_use_extended_id(bool use_extended_id)
Definition canbus.h:80
canbus::Error send_data(uint32_t can_id, bool use_extended_id, bool remote_transmission_request, const std::vector< uint8_t > &data)
Definition canbus.cpp:24
void setup() override
Definition canbus.cpp:9
void add_trigger(CanbusTrigger *trigger)
Definition canbus.cpp:59
canbus::Error send_data(uint32_t can_id, bool use_extended_id, const std::vector< uint8_t > &data)
Definition canbus.h:75
void add_callback(std::function< void(uint32_t can_id, bool extended_id, bool rtr, const std::vector< uint8_t > &data)> callback)
Add a callback to be called when a CAN message is received.
Definition canbus.h:94
void set_bitrate(CanSpeed bit_rate)
Definition canbus.h:81
void set_can_id(uint32_t can_id)
Definition canbus.h:79
float get_setup_priority() const override
Definition canbus.h:70
void dump_config() override
Definition canbus.cpp:16
virtual Error send_message(struct CanFrame *frame)=0
virtual bool setup_internal()=0
std::vector< CanbusTrigger * > triggers_
Definition canbus.h:101
virtual Error read_message(struct CanFrame *frame)=0
void set_data_template(std::vector< uint8_t >(*func)(Ts...))
Definition canbus.h:115
void set_remote_transmission_request(bool remote_transmission_request)
Definition canbus.h:131
void set_data_static(const uint8_t *data, size_t len)
Definition canbus.h:122
optional< uint32_t > can_id_
Definition canbus.h:151
void play(const Ts &...x) override
Definition canbus.h:135
void set_use_extended_id(bool use_extended_id)
Definition canbus.h:129
optional< bool > use_extended_id_
Definition canbus.h:152
void set_can_id(uint32_t can_id)
Definition canbus.h:127
union esphome::canbus::CanbusSendAction::Data data_
optional< bool > remote_transmission_request_
Definition canbus.h:180
CanbusTrigger(Canbus *parent, const std::uint32_t can_id, const std::uint32_t can_id_mask, const bool use_extended_id)
Definition canbus.h:165
void set_remote_transmission_request(bool remote_transmission_request)
Definition canbus.h:169
bool has_value() const
Definition optional.h:92
__int64 ssize_t
Definition httplib.h:178
const float HARDWARE
For components that deal with hardware and are very important like GPIO switch.
Definition component.cpp:58
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
std::string size_t len
Definition helpers.h:500
uint8_t can_data_length_code
Definition canbus.h:61
uint8_t data[CAN_MAX_DATA_LENGTH] __attribute__((aligned(8)))
uint16_t x
Definition tt21100.cpp:5
std::vector< uint8_t >(* func)(Ts...)
Definition canbus.h:156