ESPHome 2026.5.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);
94 template<typename F> void add_callback(F &&callback) { this->callback_manager_.add(std::forward<F>(callback)); }
95
96 protected:
97 template<typename... Ts> friend class CanbusSendAction;
98 std::vector<CanbusTrigger *> triggers_{};
102 CallbackManager<void(uint32_t can_id, bool extended_id, bool rtr, const std::vector<uint8_t> &data)>
104
105 virtual bool setup_internal() = 0;
106 virtual Error send_message(struct CanFrame *frame) = 0;
107 virtual Error read_message(struct CanFrame *frame) = 0;
108};
109
110template<typename... Ts> class CanbusSendAction : public Action<Ts...>, public Parented<Canbus> {
111 public:
112 void set_data_template(std::vector<uint8_t> (*func)(Ts...)) {
113 // Stateless lambdas (generated by ESPHome) implicitly convert to function pointers
114 this->data_.func = func;
115 this->len_ = -1; // Sentinel value indicates template mode
116 }
117
118 // Store pointer to static data in flash (no RAM copy)
119 void set_data_static(const uint8_t *data, size_t len) {
120 this->data_.data = data;
121 this->len_ = len; // Length >= 0 indicates static mode
122 }
123
124 void set_can_id(uint32_t can_id) { this->can_id_ = can_id; }
125
126 void set_use_extended_id(bool use_extended_id) { this->use_extended_id_ = use_extended_id; }
127
128 void set_remote_transmission_request(bool remote_transmission_request) {
129 this->remote_transmission_request_ = remote_transmission_request;
130 }
131
132 void play(const Ts &...x) override {
133 auto can_id = this->can_id_.has_value() ? *this->can_id_ : this->parent_->can_id_;
134 auto use_extended_id =
135 this->use_extended_id_.has_value() ? *this->use_extended_id_ : this->parent_->use_extended_id_;
136 std::vector<uint8_t> data;
137 if (this->len_ >= 0) {
138 // Static mode: copy from flash to vector
139 data.assign(this->data_.data, this->data_.data + this->len_);
140 } else {
141 // Template mode: call function
142 data = this->data_.func(x...);
143 }
144 this->parent_->send_data(can_id, use_extended_id, this->remote_transmission_request_, data);
145 }
146
147 protected:
148 optional<uint32_t> can_id_{};
149 optional<bool> use_extended_id_{};
151 ssize_t len_{-1}; // -1 = template mode, >=0 = static mode with length
152 union Data {
153 std::vector<uint8_t> (*func)(Ts...); // Function pointer (stateless lambdas)
154 const uint8_t *data; // Pointer to static data in flash
156};
157
158class CanbusTrigger : public Trigger<std::vector<uint8_t>, uint32_t, bool>, public Component {
159 friend class Canbus;
160
161 public:
162 explicit CanbusTrigger(Canbus *parent, const std::uint32_t can_id, const std::uint32_t can_id_mask,
163 const bool use_extended_id)
164 : parent_(parent), can_id_(can_id), can_id_mask_(can_id_mask), use_extended_id_(use_extended_id){};
165
166 void set_remote_transmission_request(bool remote_transmission_request) {
167 this->remote_transmission_request_ = remote_transmission_request;
168 }
169
170 void setup() override { this->parent_->add_trigger(this); }
171
172 protected:
178};
179
180} // namespace canbus
181} // namespace esphome
Helper class to easily give an object a parent of type T.
Definition helpers.h:2013
CallbackManager< void(uint32_t can_id, bool extended_id, bool rtr, const std::vector< uint8_t > &data)> callback_manager_
Definition canbus.h:103
void loop() override
Definition canbus.cpp:69
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:25
void add_callback(F &&callback)
Add a callback to be called when a CAN message is received.
Definition canbus.h:94
void setup() override
Definition canbus.cpp:10
void add_trigger(CanbusTrigger *trigger)
Definition canbus.cpp:60
canbus::Error send_data(uint32_t can_id, bool use_extended_id, const std::vector< uint8_t > &data)
Definition canbus.h:75
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:17
virtual Error send_message(struct CanFrame *frame)=0
virtual bool setup_internal()=0
std::vector< CanbusTrigger * > triggers_
Definition canbus.h:98
virtual Error read_message(struct CanFrame *frame)=0
void set_data_template(std::vector< uint8_t >(*func)(Ts...))
Definition canbus.h:112
void set_remote_transmission_request(bool remote_transmission_request)
Definition canbus.h:128
void set_data_static(const uint8_t *data, size_t len)
Definition canbus.h:119
optional< uint32_t > can_id_
Definition canbus.h:148
void play(const Ts &...x) override
Definition canbus.h:132
void set_use_extended_id(bool use_extended_id)
Definition canbus.h:126
optional< bool > use_extended_id_
Definition canbus.h:149
void set_can_id(uint32_t can_id)
Definition canbus.h:124
union esphome::canbus::CanbusSendAction::Data data_
optional< bool > remote_transmission_request_
Definition canbus.h:177
CanbusTrigger(Canbus *parent, const std::uint32_t can_id, const std::uint32_t can_id_mask, const bool use_extended_id)
Definition canbus.h:162
void set_remote_transmission_request(bool remote_transmission_request)
Definition canbus.h:166
__int64 ssize_t
Definition httplib.h:178
constexpr float HARDWARE
For components that deal with hardware and are very important like GPIO switch.
Definition component.h:40
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
std::string size_t len
Definition helpers.h:1045
static void uint32_t
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:153