libxr  1.0
Want to be the best embedded framework
Loading...
Searching...
No Matches
stm32_can.cpp
1#include "stm32_can.hpp"
2
3#ifdef HAL_CAN_MODULE_ENABLED
4
5using namespace LibXR;
6
7STM32CAN* STM32CAN::map[STM32_CAN_NUMBER] = {nullptr};
8
15stm32_can_id_t STM32_CAN_GetID(CAN_TypeDef* addr)
16{
17 // NOLINTBEGIN
18 if (addr == nullptr)
19 { // NOLINT
20 return stm32_can_id_t::STM32_CAN_ID_ERROR;
21 }
22#ifdef CAN1
23 else if (addr == CAN1)
24 { // NOLINT
25 return stm32_can_id_t::STM32_CAN1;
26 }
27#endif
28#ifdef CAN2
29 else if (addr == CAN2)
30 { // NOLINT
31 return stm32_can_id_t::STM32_CAN2;
32 }
33#endif
34#ifdef CAN3
35 else if (addr == CAN3)
36 { // NOLINT
37 return stm32_can_id_t::STM32_CAN3;
38 }
39#endif
40 else
41 {
42 return stm32_can_id_t::STM32_CAN_ID_ERROR;
43 }
44 // NOLINTEND
45}
46
47STM32CAN::STM32CAN(CAN_HandleTypeDef* hcan, uint32_t pool_size)
48 : CAN(), hcan_(hcan), id_(STM32_CAN_GetID(hcan->Instance)), tx_pool_(pool_size)
49{
50 map[id_] = this;
51 Init();
52}
53
54ErrorCode STM32CAN::Init(void)
55{
56 CAN_FilterTypeDef can_filter = {};
57
58 can_filter.FilterIdHigh = 0;
59 can_filter.FilterIdLow = 0;
60 can_filter.FilterMode = CAN_FILTERMODE_IDMASK;
61 can_filter.FilterScale = CAN_FILTERSCALE_32BIT;
62 can_filter.FilterMaskIdHigh = 0;
63 can_filter.FilterMaskIdLow = 0;
64 can_filter.FilterActivation = ENABLE;
65
66#ifdef CAN3
67 if (id_ == STM32_CAN1)
68 {
69 can_filter.FilterBank = 0;
70 can_filter.SlaveStartFilterBank = 14;
71 fifo_ = CAN_RX_FIFO0;
72 }
73 else if (id_ == STM32_CAN2)
74 {
75 can_filter.FilterBank = 14;
76 can_filter.SlaveStartFilterBank = 14;
77 fifo_ = CAN_RX_FIFO0;
78 }
79 else if (id_ == STM32_CAN3)
80 {
81 can_filter.FilterBank = 0;
82 fifo_ = CAN_RX_FIFO1;
83 }
84#else
85#ifdef CAN2
86 if (id_ == STM32_CAN1)
87 {
88 can_filter.FilterBank = 0;
89 can_filter.SlaveStartFilterBank = 14;
90 fifo_ = CAN_RX_FIFO0;
91 }
92 else if (id_ == STM32_CAN2)
93 {
94 can_filter.FilterBank = 14;
95 can_filter.SlaveStartFilterBank = 14;
96 fifo_ = CAN_RX_FIFO1;
97 }
98#else
99 if (id_ == STM32_CAN1)
100 {
101 can_filter.FilterBank = 0;
102 can_filter.SlaveStartFilterBank = 14;
103 fifo_ = CAN_RX_FIFO0;
104 }
105#endif
106#endif
107 else
108 {
109 ASSERT(false);
110 return ErrorCode::FAILED;
111 }
112
113 can_filter.FilterFIFOAssignment = fifo_;
114
115 if (HAL_CAN_ConfigFilter(hcan_, &can_filter) != HAL_OK)
116 {
117 return ErrorCode::FAILED;
118 }
119
120 if (HAL_CAN_Start(hcan_) != HAL_OK)
121 {
122 return ErrorCode::FAILED;
123 }
124
125 if (fifo_ == CAN_RX_FIFO0)
126 {
127 HAL_CAN_ActivateNotification(hcan_, CAN_IT_RX_FIFO0_MSG_PENDING);
128 }
129 else
130 {
131 HAL_CAN_ActivateNotification(hcan_, CAN_IT_RX_FIFO1_MSG_PENDING);
132 }
133
134 HAL_CAN_ActivateNotification(hcan_, CAN_IT_ERROR);
135 HAL_CAN_ActivateNotification(hcan_, CAN_IT_BUSOFF);
136 HAL_CAN_ActivateNotification(hcan_, CAN_IT_ERROR_PASSIVE);
137 HAL_CAN_ActivateNotification(hcan_, CAN_IT_ERROR_WARNING);
138 HAL_CAN_ActivateNotification(hcan_, CAN_IT_TX_MAILBOX_EMPTY);
139
140 return ErrorCode::OK;
141}
142
144{
145 if (hcan_ == nullptr || hcan_->Instance == nullptr)
146 {
147 ASSERT(false);
148 return ErrorCode::ARG_ERR;
149 }
150
151 CAN_TypeDef* can = hcan_->Instance;
152
153 // 先关掉与 Init 对应的中断
154 uint32_t it_mask = 0u;
155
156#ifdef CAN_IT_RX_FIFO0_MSG_PENDING
157 if (fifo_ == CAN_RX_FIFO0)
158 {
159 it_mask |= CAN_IT_RX_FIFO0_MSG_PENDING;
160 }
161#endif
162
163#ifdef CAN_IT_RX_FIFO1_MSG_PENDING
164 if (fifo_ == CAN_RX_FIFO1)
165 {
166 it_mask |= CAN_IT_RX_FIFO1_MSG_PENDING;
167 }
168#endif
169
170#ifdef CAN_IT_ERROR
171 it_mask |= CAN_IT_ERROR;
172#endif
173
174#ifdef CAN_IT_TX_MAILBOX_EMPTY
175 it_mask |= CAN_IT_TX_MAILBOX_EMPTY;
176#endif
177
178 if (it_mask != 0u)
179 {
180 HAL_CAN_DeactivateNotification(hcan_, it_mask);
181 }
182
183 // 停止 CAN,进入配置状态
184 if (HAL_CAN_Stop(hcan_) != HAL_OK)
185 {
186 return ErrorCode::FAILED;
187 }
188
189 // 一次发送模式(不自动重发)→ NART
190#ifdef CAN_MCR_NART
191 if (cfg.mode.one_shot)
192 {
193 SET_BIT(can->MCR, CAN_MCR_NART);
194 }
195 else
196 {
197 CLEAR_BIT(can->MCR, CAN_MCR_NART);
198 }
199#endif
200
201 const auto& bt = cfg.bit_timing;
202
203 // 参数范围校验(0 表示保留原值)。
204 // Derive field maxima from masks instead of hard-coded constants.
205 constexpr uint32_t BRP_FIELD_MAX =
206 (CAN_BTR_BRP_Msk >> CAN_BTR_BRP_Pos); // 存的是 brp-1
207 constexpr uint32_t TS1_FIELD_MAX =
208 (CAN_BTR_TS1_Msk >> CAN_BTR_TS1_Pos); // 存的是 ts1-1
209 constexpr uint32_t TS2_FIELD_MAX =
210 (CAN_BTR_TS2_Msk >> CAN_BTR_TS2_Pos); // 存的是 ts2-1
211 constexpr uint32_t SJW_FIELD_MAX =
212 (CAN_BTR_SJW_Msk >> CAN_BTR_SJW_Pos); // 存的是 sjw-1
213
214 // 实际可配置的量(寄存器值 + 1)
215 constexpr uint32_t BRP_MAX = BRP_FIELD_MAX + 1u; // 1..1024
216 constexpr uint32_t TS1_MAX = TS1_FIELD_MAX + 1u; // 1..16
217 constexpr uint32_t TS2_MAX = TS2_FIELD_MAX + 1u; // 1..8
218 constexpr uint32_t SJW_MAX = SJW_FIELD_MAX + 1u; // 1..4
219
220 // brp 范围检查
221 if (bt.brp != 0u)
222 {
223 if (bt.brp < 1u || bt.brp > BRP_MAX)
224 {
225 ASSERT(false);
226 return ErrorCode::ARG_ERR;
227 }
228 }
229
230 // tseg1 = prop_seg + phase_seg1
231 uint32_t tseg1 = bt.prop_seg + bt.phase_seg1;
232 if (bt.prop_seg != 0u || bt.phase_seg1 != 0u)
233 {
234 if (tseg1 < 1u || tseg1 > TS1_MAX)
235 {
236 ASSERT(false);
237 return ErrorCode::ARG_ERR;
238 }
239 }
240
241 if (bt.phase_seg2 != 0u)
242 {
243 if (bt.phase_seg2 < 1u || bt.phase_seg2 > TS2_MAX)
244 {
245 ASSERT(false);
246 return ErrorCode::ARG_ERR;
247 }
248 }
249
250 if (bt.sjw != 0u)
251 {
252 if (bt.sjw < 1u || bt.sjw > SJW_MAX)
253 {
254 ASSERT(false);
255 return ErrorCode::ARG_ERR;
256 }
257
258 // 规范上 SJW ≤ TSEG2(只在二者都要更新时检查)
259 if (bt.phase_seg2 != 0u && bt.sjw > bt.phase_seg2)
260 {
261 ASSERT(false);
262 return ErrorCode::ARG_ERR;
263 }
264 }
265
266 uint32_t btr_old = can->BTR;
267 uint32_t btr_new = btr_old;
268 uint32_t btr_mask = 0u;
269
270 // BRP:bt.brp == 0 → 保持原值
271 if (bt.brp != 0u)
272 {
273 uint32_t brp = (bt.brp - 1u) & BRP_FIELD_MAX;
274 uint32_t mask = CAN_BTR_BRP_Msk;
275 btr_mask |= mask;
276 btr_new &= ~mask;
277 btr_new |= (brp << CAN_BTR_BRP_Pos);
278 }
279
280 // TSEG1 = PROP_SEG + PHASE_SEG1,两者都为 0 → 保持原值
281 if (bt.prop_seg != 0u || bt.phase_seg1 != 0u)
282 {
283 uint32_t ts1 = (tseg1 - 1u) & TS1_FIELD_MAX;
284 uint32_t mask = CAN_BTR_TS1_Msk;
285 btr_mask |= mask;
286 btr_new &= ~mask;
287 btr_new |= (ts1 << CAN_BTR_TS1_Pos);
288 }
289
290 // TSEG2 = PHASE_SEG2,phase_seg2 == 0 → 保持原值
291 if (bt.phase_seg2 != 0u)
292 {
293 uint32_t ts2 = (bt.phase_seg2 - 1u) & TS2_FIELD_MAX;
294 uint32_t mask = CAN_BTR_TS2_Msk;
295 btr_mask |= mask;
296 btr_new &= ~mask;
297 btr_new |= (ts2 << CAN_BTR_TS2_Pos);
298 }
299
300 // SJW,sjw == 0 → 保持原值
301 if (bt.sjw != 0u)
302 {
303 uint32_t sjw = (bt.sjw - 1u) & SJW_FIELD_MAX;
304 uint32_t mask = CAN_BTR_SJW_Msk;
305 btr_mask |= mask;
306 btr_new &= ~mask;
307 btr_new |= (sjw << CAN_BTR_SJW_Pos);
308 }
309
310 // 三采样:只有 HAL 定义了 CAN_BTR_SAM 才动;否则忽略 triple_sampling
311#ifdef CAN_BTR_SAM
312 {
313 uint32_t mask = CAN_BTR_SAM;
314 btr_mask |= mask;
315 btr_new &= ~mask;
316 if (cfg.mode.triple_sampling)
317 {
318 btr_new |= mask;
319 }
320 }
321#else
322 (void)cfg.mode.triple_sampling;
323#endif
324
325 // Loopback:bool 两态,强制覆盖
326#ifdef CAN_BTR_LBKM
327 {
328 uint32_t mask = CAN_BTR_LBKM;
329 btr_mask |= mask;
330 btr_new &= ~mask;
331 if (cfg.mode.loopback)
332 {
333 btr_new |= mask;
334 }
335 }
336#endif
337
338 // Listen-only:bool 两态,强制覆盖
339#ifdef CAN_BTR_SILM
340 {
341 uint32_t mask = CAN_BTR_SILM;
342 btr_mask |= mask;
343 btr_new &= ~mask;
344 if (cfg.mode.listen_only)
345 {
346 btr_new |= mask;
347 }
348 }
349#endif
350
351 // 只改被 btr_mask 覆盖到的位,其余保持原值
352 if (btr_mask != 0u)
353 {
354 btr_old &= ~btr_mask;
355 btr_old |= (btr_new & btr_mask);
356 can->BTR = btr_old;
357 }
358
359 // 重新启动 CAN
360 if (HAL_CAN_Start(hcan_) != HAL_OK)
361 {
362 return ErrorCode::FAILED;
363 }
364
365 // 按 Init 的方式恢复中断
366 uint32_t it_rx = 0u;
367
368#ifdef CAN_IT_RX_FIFO0_MSG_PENDING
369 if (fifo_ == CAN_RX_FIFO0)
370 {
371 it_rx = CAN_IT_RX_FIFO0_MSG_PENDING;
372 }
373#endif
374
375#ifdef CAN_IT_RX_FIFO1_MSG_PENDING
376 if (fifo_ == CAN_RX_FIFO1)
377 {
378 it_rx = CAN_IT_RX_FIFO1_MSG_PENDING;
379 }
380#endif
381
382 if (it_rx != 0u)
383 {
384 HAL_CAN_ActivateNotification(hcan_, it_rx);
385 }
386
387 HAL_CAN_ActivateNotification(hcan_, CAN_IT_ERROR);
388 HAL_CAN_ActivateNotification(hcan_, CAN_IT_BUSOFF);
389 HAL_CAN_ActivateNotification(hcan_, CAN_IT_ERROR_PASSIVE);
390 HAL_CAN_ActivateNotification(hcan_, CAN_IT_ERROR_WARNING);
391 HAL_CAN_ActivateNotification(hcan_, CAN_IT_TX_MAILBOX_EMPTY);
392
393 return ErrorCode::OK;
394}
395
397{
398 // 经典 bxCAN 始终挂在 APB1 上
399 return HAL_RCC_GetPCLK1Freq();
400}
401
402inline void STM32CAN::BuildTxHeader(const ClassicPack& p, CAN_TxHeaderTypeDef& h)
403{
404 const bool IS_EXT = (p.type == Type::EXTENDED) || (p.type == Type::REMOTE_EXTENDED);
405 const bool IS_RTR =
406 (p.type == Type::REMOTE_STANDARD) || (p.type == Type::REMOTE_EXTENDED);
407
408 h.DLC = (p.dlc <= 8u) ? p.dlc : 8u;
409 h.IDE = IS_EXT ? CAN_ID_EXT : CAN_ID_STD;
410 h.RTR = IS_RTR ? CAN_RTR_REMOTE : CAN_RTR_DATA;
411
412 h.StdId = IS_EXT ? 0u : (p.id & 0x7FFu);
413 h.ExtId = IS_EXT ? (p.id & 0x1FFFFFFFu) : 0u;
414
415 h.TransmitGlobalTime = DISABLE;
416}
417
418void STM32CAN::TxService()
419{
420 if (hcan_ == nullptr || hcan_->Instance == nullptr)
421 {
422 return;
423 }
424
425 // 标记:需要一次 TX 服务(无论本次是否抢到锁)
426 tx_pend_.store(1u, std::memory_order_release);
427
428 // 尝试抢占服务锁:0 -> 1
429 uint32_t expected = 0u;
430 if (!tx_lock_.compare_exchange_strong(expected, 1u, std::memory_order_acquire,
431 std::memory_order_relaxed))
432 {
433 // 有别的上下文在服务;PEND 已置位,它会在结束时看到并再服务
434 return;
435 }
436
437 constexpr uint32_t TME_MASK = (CAN_TSR_TME0 | CAN_TSR_TME1 | CAN_TSR_TME2);
438
439 for (;;)
440 {
441 // 本轮服务开始:消费掉 pend(期间如有新 kick,会再次置 1)
442 tx_pend_.store(0u, std::memory_order_release);
443
444 // 尽可能把空 mailbox 填满:一直到队列空或者 AddTxMessage 失败(邮箱满/忙)
445 while ((hcan_->Instance->TSR & TME_MASK) != 0u)
446 {
447 ClassicPack p{};
448 if (tx_pool_.Get(p) != ErrorCode::OK)
449 {
450 break; // 队列空
451 }
452
453 CAN_TxHeaderTypeDef hdr{};
454 BuildTxHeader(p, hdr);
455
456 uint32_t mailbox = 0u;
457 if (HAL_CAN_AddTxMessage(hcan_, &hdr, p.data, &mailbox) != HAL_OK)
458 {
459 // 发送失败:回队列,不做任何兜底/处理
460 (void)tx_pool_.Put(p);
461 break;
462 }
463
464 txMailbox = mailbox;
465 }
466
467 // 先解锁,再检查是否仍有待处理发送请求。
468 // Release lock first, then check whether pending TX work remains.
469 tx_lock_.store(0u, std::memory_order_release);
470
471 // 再看是否还有 pend:若没有,直接退出
472 if (tx_pend_.load(std::memory_order_acquire) == 0u)
473 {
474 return;
475 }
476
477 // 允许出现 PEND=1, LOCK=0 的“无人服务”状态:尝试重新加锁再服务
478 expected = 0u;
479 if (!tx_lock_.compare_exchange_strong(expected, 1u, std::memory_order_acquire,
480 std::memory_order_relaxed))
481 {
482 return;
483 }
484 }
485}
486
487ErrorCode STM32CAN::AddMessage(const ClassicPack& pack)
488{
489 if (pack.type == Type::ERROR)
490 {
491 return ErrorCode::ARG_ERR;
492 }
493
494 // 池满直接返回 FULL,不做补救
495 if (tx_pool_.Put(pack) != ErrorCode::OK)
496 {
497 return ErrorCode::FULL;
498 }
499
500 // kick
501 TxService();
502 return ErrorCode::OK;
503}
504
506{
507 while (HAL_CAN_GetRxMessage(hcan_, fifo_, &rx_buff_.header, rx_buff_.pack.data) ==
508 HAL_OK)
509 {
510 if (rx_buff_.header.IDE == CAN_ID_STD)
511 {
512 rx_buff_.pack.id = rx_buff_.header.StdId;
513 rx_buff_.pack.type = Type::STANDARD;
514 }
515 else
516 {
517 rx_buff_.pack.id = rx_buff_.header.ExtId;
518 rx_buff_.pack.type = Type::EXTENDED;
519 }
520
521 if (rx_buff_.header.RTR == CAN_RTR_REMOTE)
522 {
523 if (rx_buff_.pack.type == Type::STANDARD)
524 {
525 rx_buff_.pack.type = Type::REMOTE_STANDARD;
526 }
527 else
528 {
529 rx_buff_.pack.type = Type::REMOTE_EXTENDED;
530 }
531 }
532
533 uint8_t dlc = (rx_buff_.header.DLC <= 8u) ? rx_buff_.header.DLC : 8u;
534 rx_buff_.pack.dlc = dlc;
535
536 OnMessage(rx_buff_.pack, true);
537 }
538}
539
541{
542 if (hcan_ == nullptr || hcan_->Instance == nullptr)
543 {
544 return;
545 }
546
547 uint32_t esr = hcan_->Instance->ESR;
548
549 CAN::ClassicPack pack{};
550 pack.type = CAN::Type::ERROR;
551 pack.dlc = 0;
552
553 CAN::ErrorID eid = CAN::ErrorID::CAN_ERROR_ID_GENERIC;
554
555 if (esr & CAN_ESR_BOFF)
556 {
557 eid = CAN::ErrorID::CAN_ERROR_ID_BUS_OFF;
558 }
559 else if (esr & CAN_ESR_EPVF)
560 {
561 eid = CAN::ErrorID::CAN_ERROR_ID_ERROR_PASSIVE;
562 }
563 else if (esr & CAN_ESR_EWGF)
564 {
565 eid = CAN::ErrorID::CAN_ERROR_ID_ERROR_WARNING;
566 }
567 else
568 {
569 uint32_t lec = (esr >> 4) & 0x7u;
570
571 switch (lec)
572 {
573 case 0x01:
574 eid = CAN::ErrorID::CAN_ERROR_ID_STUFF;
575 break;
576 case 0x02:
577 eid = CAN::ErrorID::CAN_ERROR_ID_FORM;
578 break;
579 case 0x03:
580 eid = CAN::ErrorID::CAN_ERROR_ID_ACK;
581 break;
582 case 0x04:
583 eid = CAN::ErrorID::CAN_ERROR_ID_BIT1;
584 break;
585 case 0x05:
586 eid = CAN::ErrorID::CAN_ERROR_ID_BIT0;
587 break;
588 case 0x06:
589 eid = CAN::ErrorID::CAN_ERROR_ID_CRC;
590 break;
591 default:
592 eid = CAN::ErrorID::CAN_ERROR_ID_OTHER;
593 break;
594 }
595 }
596
597 pack.id = static_cast<uint32_t>(eid);
598
599 // 在中断上下文中分发错误帧
600 OnMessage(pack, true);
601}
602
604{
605 if (hcan_ == nullptr || hcan_->Instance == nullptr)
606 {
607 return ErrorCode::ARG_ERR;
608 }
609
610 // 直接读取 bxCAN 的错误状态寄存器 ESR
611 uint32_t esr = hcan_->Instance->ESR;
612
613 // TEC: bits 23:16, REC: bits 31:24(参考 RM & CAN_ESR 描述)
614 state.tx_error_counter = static_cast<uint8_t>((esr >> 16) & 0xFFu);
615 state.rx_error_counter = static_cast<uint8_t>((esr >> 24) & 0xFFu);
616
617 // 状态位:BOFF / EPVF / EWGF
618 state.bus_off = (esr & CAN_ESR_BOFF) != 0u;
619 state.error_passive = (esr & CAN_ESR_EPVF) != 0u;
620 state.error_warning = (esr & CAN_ESR_EWGF) != 0u;
621
622 return ErrorCode::OK;
623}
624
625extern "C" void HAL_CAN_RxFifo0MsgPendingCallback(CAN_HandleTypeDef* hcan)
626{
627 STM32CAN* can = STM32CAN::map[STM32_CAN_GetID(hcan->Instance)];
628 if (can)
629 {
630 can->ProcessRxInterrupt();
631 }
632}
633
634extern "C" void HAL_CAN_RxFifo1MsgPendingCallback(CAN_HandleTypeDef* hcan)
635{
636 STM32CAN* can = STM32CAN::map[STM32_CAN_GetID(hcan->Instance)];
637 if (can)
638 {
639 can->ProcessRxInterrupt();
640 }
641}
642
643extern "C" void HAL_CAN_ErrorCallback(CAN_HandleTypeDef* hcan)
644{
645 STM32CAN* can = STM32CAN::map[STM32_CAN_GetID(hcan->Instance)];
646 if (can)
647 {
649 }
650
651 HAL_CAN_ResetError(hcan);
652}
653
654extern "C" void HAL_CAN_TxMailbox0CompleteCallback(CAN_HandleTypeDef* hcan)
655{
656 STM32CAN* can = STM32CAN::map[STM32_CAN_GetID(hcan->Instance)];
657 if (can)
658 {
659 can->TxService();
660 }
661}
662
663extern "C" void HAL_CAN_TxMailbox1CompleteCallback(CAN_HandleTypeDef* hcan)
664{
665 STM32CAN* can = STM32CAN::map[STM32_CAN_GetID(hcan->Instance)];
666 if (can)
667 {
668 can->TxService();
669 }
670}
671
672extern "C" void HAL_CAN_TxMailbox2CompleteCallback(CAN_HandleTypeDef* hcan)
673{
674 STM32CAN* can = STM32CAN::map[STM32_CAN_GetID(hcan->Instance)];
675 if (can)
676 {
677 can->TxService();
678 }
679}
680
681#endif
CAN 通信抽象类,定义经典 CAN 帧与订阅接口。 Abstract class for CAN communication with classic CAN frames and subscript...
Definition can.hpp:16
@ EXTENDED
扩展数据帧(29-bit ID)。Extended data frame (29-bit ID).
@ REMOTE_EXTENDED
扩展远程帧。Extended remote frame.
@ STANDARD
标准数据帧(11-bit ID)。Standard data frame (11-bit ID).
@ REMOTE_STANDARD
标准远程帧。Standard remote frame.
@ ERROR
错误帧(虚拟事件)。Error frame (virtual event).
void OnMessage(const ClassicPack &pack, bool in_isr)
分发接收到的经典 CAN 帧。 Dispatch a received classic CAN frame.
Definition can.cpp:18
ErrorID
ClassicPack::type == Type::ERROR 时使用的虚拟 ID。 Virtual IDs used when ClassicPack::type == Type::ERROR.
Definition can.hpp:146
ErrorCode Put(const Data &data)
向池中放入一个元素 / Put an element into the pool
ErrorCode Get(Data &data)
从池中取出一个元素 / Retrieve an element from the pool
STM32 CAN 驱动实现 / STM32 CAN driver implementation.
Definition stm32_can.hpp:42
ErrorCode GetErrorState(CAN::ErrorState &state) const override
查询当前错误状态(快照)。 Query current CAN controller error state (snapshot).
ErrorCode SetConfig(const CAN::Configuration &cfg) override
设置 CAN 配置 / Set CAN configuration
STM32CAN(CAN_HandleTypeDef *hcan, uint32_t pool_size)
构造 CAN 驱动对象 / Construct CAN driver object
Definition stm32_can.cpp:47
uint32_t GetClockFreq() const override
获取 CAN 外设时钟频率(Hz)。 Get CAN peripheral clock frequency in Hz.
ErrorCode Init(void)
初始化驱动 / Initialize driver
Definition stm32_can.cpp:54
ErrorCode AddMessage(const ClassicPack &pack) override
添加经典 CAN 消息。Add classic CAN message.
void ProcessErrorInterrupt()
处理错误中断 / Handle error interrupt
void ProcessRxInterrupt()
处理接收中断 / Handle RX interrupt
LibXR 命名空间
Definition ch32_can.hpp:14
uint32_t prop_seg
传播段。Propagation segment.
Definition can.hpp:39
经典 CAN 帧数据结构。Classic CAN frame structure.
Definition can.hpp:129
Type type
帧类型。Frame type.
Definition can.hpp:131
uint32_t id
CAN ID(11/29 bit 或 ErrorID)。CAN ID (11/29 bits or ErrorID).
Definition can.hpp:130
uint8_t dlc
有效数据长度(0~8)。Data length code (0–8).
Definition can.hpp:132
CAN 配置参数。CAN configuration parameters.
Definition can.hpp:62
BitTiming bit_timing
位时序配置。Bit timing configuration.
Definition can.hpp:65
Mode mode
工作模式。Operating mode.
Definition can.hpp:66
CAN 当前错误状态快照(来自硬件计数器/状态机)。 Snapshot of current CAN controller error state (from HW counters/state).
Definition can.hpp:88
bool bus_off
是否处于 BUS-OFF。True if controller is bus-off.
Definition can.hpp:92
bool error_passive
是否处于 Error Passive。True if error-passive.
Definition can.hpp:93
uint8_t tx_error_counter
发送错误计数 TEC。Transmit error counter (TEC).
Definition can.hpp:89
bool error_warning
是否处于 Error Warning。True if error-warning.
Definition can.hpp:94
uint8_t rx_error_counter
接收错误计数 REC。Receive error counter (REC).
Definition can.hpp:90
bool triple_sampling
三采样。Triple sampling.
Definition can.hpp:53
bool loopback
回环模式。Loopback mode.
Definition can.hpp:51
bool one_shot
单次发送模式。One-shot transmission.
Definition can.hpp:54
bool listen_only
只听(静默)模式。Listen-only (silent) mode.
Definition can.hpp:52