libxr  1.0
Want to be the best embedded framework
Loading...
Searching...
No Matches
stm32_canfd.cpp
1#include "stm32_canfd.hpp"
2
3#ifdef HAL_FDCAN_MODULE_ENABLED
4
5using namespace LibXR;
6
7STM32CANFD* STM32CANFD::map[STM32_FDCAN_NUMBER] = {nullptr};
8
15stm32_fdcan_id_t STM32_FDCAN_GetID(FDCAN_GlobalTypeDef* addr)
16{
17 if (addr == nullptr)
18 { // NOLINT
19 return stm32_fdcan_id_t::STM32_FDCAN_ID_ERROR;
20 }
21#ifdef FDCAN1
22 else if (addr == FDCAN1)
23 { // NOLINT
24 return stm32_fdcan_id_t::STM32_FDCAN1;
25 }
26#endif
27#ifdef FDCAN2
28 else if (addr == FDCAN2)
29 { // NOLINT
30 return stm32_fdcan_id_t::STM32_FDCAN2;
31 }
32#endif
33#ifdef FDCAN3
34 else if (addr == FDCAN3)
35 { // NOLINT
36 return stm32_fdcan_id_t::STM32_FDCAN3;
37 }
38#endif
39 else
40 {
41 return stm32_fdcan_id_t::STM32_FDCAN_ID_ERROR;
42 }
43}
44
45static inline uint32_t BytesToDlc(uint32_t n)
46{
47 if (n <= 8U)
48 {
49 return n;
50 } // FDCAN_DLC_BYTES_0..8 == 0..8
51 if (n <= 12U)
52 {
53 return FDCAN_DLC_BYTES_12;
54 }
55 if (n <= 16U)
56 {
57 return FDCAN_DLC_BYTES_16;
58 }
59 if (n <= 20U)
60 {
61 return FDCAN_DLC_BYTES_20;
62 }
63 if (n <= 24U)
64 {
65 return FDCAN_DLC_BYTES_24;
66 }
67 if (n <= 32U)
68 {
69 return FDCAN_DLC_BYTES_32;
70 }
71 if (n <= 48U)
72 {
73 return FDCAN_DLC_BYTES_48;
74 }
75 return FDCAN_DLC_BYTES_64; // n >= 49 → 64
76}
77
78static inline uint32_t DlcToBytes(uint32_t dlc)
79{
80 if (dlc <= FDCAN_DLC_BYTES_8)
81 { // 0..8 → 0..8
82 return dlc;
83 }
84 else if (dlc == FDCAN_DLC_BYTES_12)
85 {
86 return 12U;
87 }
88 else if (dlc == FDCAN_DLC_BYTES_16)
89 {
90 return 16U;
91 }
92 else if (dlc == FDCAN_DLC_BYTES_20)
93 {
94 return 20U;
95 }
96 else if (dlc == FDCAN_DLC_BYTES_24)
97 {
98 return 24U;
99 }
100 else if (dlc == FDCAN_DLC_BYTES_32)
101 {
102 return 32U;
103 }
104 else if (dlc == FDCAN_DLC_BYTES_48)
105 {
106 return 48U;
107 }
108 else
109 { // FDCAN_DLC_BYTES_64 或其它非法值
110 return 64U;
111 }
112}
113
114inline void STM32CANFD::BuildTxHeader(const ClassicPack& p, FDCAN_TxHeaderTypeDef& h)
115{
116 const bool is_ext = (p.type == Type::EXTENDED) || (p.type == Type::REMOTE_EXTENDED);
117 const bool is_rtr =
118 (p.type == Type::REMOTE_STANDARD) || (p.type == Type::REMOTE_EXTENDED);
119
120 h.Identifier = p.id;
121 h.IdType = is_ext ? FDCAN_EXTENDED_ID : FDCAN_STANDARD_ID;
122 h.TxFrameType = is_rtr ? FDCAN_REMOTE_FRAME : FDCAN_DATA_FRAME;
123
124 uint32_t bytes = (p.dlc <= 8u) ? p.dlc : 8u;
125 h.DataLength = BytesToDlc(bytes);
126
127 h.ErrorStateIndicator = FDCAN_ESI_PASSIVE;
128 h.BitRateSwitch = FDCAN_BRS_OFF;
129 h.FDFormat = FDCAN_CLASSIC_CAN;
130
131 h.TxEventFifoControl = FDCAN_NO_TX_EVENTS;
132 h.MessageMarker = 0x01;
133}
134
135inline void STM32CANFD::BuildTxHeader(const FDPack& p, FDCAN_TxHeaderTypeDef& h)
136{
137 h.Identifier = p.id;
138
139 switch (p.type)
140 {
141 case Type::STANDARD:
142 h.IdType = FDCAN_STANDARD_ID;
143 h.TxFrameType = FDCAN_DATA_FRAME;
144 break;
145
146 case Type::EXTENDED:
147 h.IdType = FDCAN_EXTENDED_ID;
148 h.TxFrameType = FDCAN_DATA_FRAME;
149 break;
150
151 default:
152 ASSERT(false);
153 return;
154 }
155
156 ASSERT(p.len <= 64u);
157 h.DataLength = BytesToDlc(p.len);
158
159 h.ErrorStateIndicator = FDCAN_ESI_PASSIVE;
160 h.BitRateSwitch = FDCAN_BRS_ON;
161 h.FDFormat = FDCAN_FD_CAN;
162
163 h.TxEventFifoControl = FDCAN_NO_TX_EVENTS;
164 h.MessageMarker = 0x00;
165}
166
167STM32CANFD::STM32CANFD(FDCAN_HandleTypeDef* hcan, uint32_t queue_size)
168 : FDCAN(),
169 hcan_(hcan),
170 id_(STM32_FDCAN_GetID(hcan->Instance)),
171 tx_pool_(queue_size),
172 tx_pool_fd_(queue_size)
173{
174 CheckMessageRAMOffset(hcan);
175 map[id_] = this;
176 Init();
177}
178
179ErrorCode STM32CANFD::Init(void)
180{
181 FDCAN_FilterTypeDef can_filter = {};
182 can_filter.IdType = FDCAN_STANDARD_ID;
183 can_filter.FilterType = FDCAN_FILTER_MASK;
184 can_filter.FilterID1 = 0x0000;
185 can_filter.FilterID2 = 0x0000;
186 can_filter.FilterIndex = 0;
187
188#ifdef FDCAN3
189 if (id_ == STM32_FDCAN1)
190 {
191 can_filter.FilterConfig = FDCAN_FILTER_TO_RXFIFO0;
192 }
193 else if (id_ == STM32_FDCAN2)
194 {
195 can_filter.FilterConfig = FDCAN_FILTER_TO_RXFIFO1;
196 }
197 else if (id_ == STM32_FDCAN3)
198 {
199 can_filter.FilterConfig = FDCAN_FILTER_TO_RXFIFO1;
200 }
201#else
202#ifdef FDCAN2
203 if (id_ == STM32_FDCAN1)
204 {
205 can_filter.FilterConfig = FDCAN_FILTER_TO_RXFIFO0;
206 }
207 else if (id_ == STM32_FDCAN2)
208 {
209 can_filter.FilterConfig = FDCAN_FILTER_TO_RXFIFO1;
210 }
211#else
212 can_filter.FilterConfig = FDCAN_FILTER_TO_RXFIFO0;
213#endif
214#endif
215
216 if (HAL_FDCAN_ConfigFilter(hcan_, &can_filter) != HAL_OK)
217 {
218 return ErrorCode::FAILED;
219 }
220
221 can_filter.IdType = FDCAN_EXTENDED_ID;
222
223 if (HAL_FDCAN_ConfigFilter(hcan_, &can_filter) != HAL_OK)
224 {
225 return ErrorCode::FAILED;
226 }
227
228 if (HAL_FDCAN_Start(hcan_) != HAL_OK)
229 {
230 return ErrorCode::FAILED;
231 }
232
233 if (can_filter.FilterConfig == FDCAN_FILTER_TO_RXFIFO0)
234 {
235 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_RX_FIFO0_NEW_MESSAGE, 0);
236 }
237 else
238 {
239 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_RX_FIFO1_NEW_MESSAGE, 0);
240 }
241
242 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_ERROR_PASSIVE, 0);
243 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_ERROR_WARNING, 0);
244 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_BUS_OFF, 0);
245 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_TX_FIFO_EMPTY, 0);
246 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_TX_COMPLETE, 0xFFFFFFFF);
247
248 return ErrorCode::OK;
249}
250
252{
253 if (pack.type == Type::ERROR)
254 {
255 return ErrorCode::ARG_ERR;
256 }
257
258 // 先入池;满则先服务一次再 Put 一次
259 if (tx_pool_.Put(pack) != ErrorCode::OK)
260 {
261 TxService();
262 if (tx_pool_.Put(pack) != ErrorCode::OK)
263 {
264 return ErrorCode::FULL;
265 }
266 }
267
268 TxService();
269 return ErrorCode::OK;
270}
271
273{
274 // 兼容接口:只用仲裁相位参数,FD 数据相位保持不动
275 FDCAN::Configuration fd_cfg{};
276 fd_cfg.bitrate = cfg.bitrate;
277 fd_cfg.sample_point = cfg.sample_point;
278 fd_cfg.bit_timing = cfg.bit_timing;
279 fd_cfg.mode = cfg.mode;
280 // data_timing / fd_mode 全部置 0 → “保持原值”
281 return SetConfig(fd_cfg);
282}
283
285{
286 if (hcan_ == nullptr || hcan_->Instance == nullptr)
287 {
288 ASSERT(false);
289 return ErrorCode::ARG_ERR;
290 }
291
292 FDCAN_GlobalTypeDef* can = hcan_->Instance;
293
294 // 先关通知:只关本驱动用到的这些
295 uint32_t it_mask = 0u;
296
297#ifdef FDCAN_IT_RX_FIFO0_NEW_MESSAGE
298 it_mask |= FDCAN_IT_RX_FIFO0_NEW_MESSAGE;
299#endif
300#ifdef FDCAN_IT_RX_FIFO1_NEW_MESSAGE
301 it_mask |= FDCAN_IT_RX_FIFO1_NEW_MESSAGE;
302#endif
303#ifdef FDCAN_IT_TX_FIFO_EMPTY
304 it_mask |= FDCAN_IT_TX_FIFO_EMPTY;
305#endif
306
307 if (it_mask != 0u)
308 {
309 HAL_FDCAN_DeactivateNotification(hcan_, it_mask);
310 }
311
312 // 停止 FDCAN,进入 INIT / CCE 配置状态
313 if (HAL_FDCAN_Stop(hcan_) != HAL_OK)
314 {
315 return ErrorCode::FAILED;
316 }
317
318#ifdef FDCAN_CCCR_INIT
319 SET_BIT(can->CCCR, FDCAN_CCCR_INIT);
320#endif
321#ifdef FDCAN_CCCR_CCE
322 SET_BIT(can->CCCR, FDCAN_CCCR_CCE);
323#endif
324
325 // ===== 模式相关:one-shot / loopback / listen-only =====
326#ifdef FDCAN_CCCR_DAR
327 if (cfg.mode.one_shot)
328 {
329 // 禁用自动重发
330 SET_BIT(can->CCCR, FDCAN_CCCR_DAR);
331 }
332 else
333 {
334 CLEAR_BIT(can->CCCR, FDCAN_CCCR_DAR);
335 }
336#endif
337
338 // triple_sampling 对 FDCAN 没意义,这里直接忽略
339 (void)cfg.mode.triple_sampling;
340
341#ifdef FDCAN_CCCR_TEST
342#ifdef FDCAN_TEST_LBCK
343 if (cfg.mode.loopback)
344 {
345 // 内部回环
346 SET_BIT(can->CCCR, FDCAN_CCCR_TEST);
347 SET_BIT(can->TEST, FDCAN_TEST_LBCK);
348 }
349 else
350 {
351 CLEAR_BIT(can->TEST, FDCAN_TEST_LBCK);
352 CLEAR_BIT(can->CCCR, FDCAN_CCCR_TEST);
353 }
354#endif
355#endif
356
357#ifdef FDCAN_CCCR_MON
358 if (cfg.mode.listen_only)
359 {
360 // 总线监控(只听)
361 SET_BIT(can->CCCR, FDCAN_CCCR_MON);
362 }
363 else
364 {
365 CLEAR_BIT(can->CCCR, FDCAN_CCCR_MON);
366 }
367#endif
368
369 // ===== 仲裁相位 bit timing 范围校验 + 写 NBTP =====
370 const auto& bt = cfg.bit_timing;
371
372 // 用掩码算出字段最大值,避免硬编码
373#ifdef FDCAN_NBTP_NBRP_Msk
374 constexpr uint32_t NBRP_FIELD_MAX = (FDCAN_NBTP_NBRP_Msk >> FDCAN_NBTP_NBRP_Pos);
375 constexpr uint32_t NTSEG1_FIELD_MAX = (FDCAN_NBTP_NTSEG1_Msk >> FDCAN_NBTP_NTSEG1_Pos);
376 constexpr uint32_t NTSEG2_FIELD_MAX = (FDCAN_NBTP_NTSEG2_Msk >> FDCAN_NBTP_NTSEG2_Pos);
377 constexpr uint32_t NSJW_FIELD_MAX = (FDCAN_NBTP_NSJW_Msk >> FDCAN_NBTP_NSJW_Pos);
378
379 constexpr uint32_t NBRP_MAX = NBRP_FIELD_MAX + 1u;
380 constexpr uint32_t NTSEG1_MAX = NTSEG1_FIELD_MAX + 1u;
381 constexpr uint32_t NTSEG2_MAX = NTSEG2_FIELD_MAX + 1u;
382 constexpr uint32_t NSJW_MAX = NSJW_FIELD_MAX + 1u;
383
384 // 0 = 保持原值,非 0 做范围检查
385 if (bt.brp != 0u)
386 {
387 if (bt.brp < 1u || bt.brp > NBRP_MAX)
388 {
389 ASSERT(false);
390 return ErrorCode::ARG_ERR;
391 }
392 }
393
394 uint32_t tseg1 = bt.prop_seg + bt.phase_seg1;
395 if (bt.prop_seg != 0u || bt.phase_seg1 != 0u)
396 {
397 if (tseg1 < 1u || tseg1 > NTSEG1_MAX)
398 {
399 ASSERT(false);
400 return ErrorCode::ARG_ERR;
401 }
402 }
403
404 if (bt.phase_seg2 != 0u)
405 {
406 if (bt.phase_seg2 < 1u || bt.phase_seg2 > NTSEG2_MAX)
407 {
408 ASSERT(false);
409 return ErrorCode::ARG_ERR;
410 }
411 }
412
413 if (bt.sjw != 0u)
414 {
415 if (bt.sjw < 1u || bt.sjw > NSJW_MAX)
416 {
417 ASSERT(false);
418 return ErrorCode::ARG_ERR;
419 }
420 if (bt.phase_seg2 != 0u && bt.sjw > bt.phase_seg2)
421 {
422 ASSERT(false);
423 return ErrorCode::ARG_ERR;
424 }
425 }
426
427 uint32_t nbtp_old = can->NBTP;
428 uint32_t nbtp_new = nbtp_old;
429 uint32_t nbtp_mask = 0u;
430
431 // NBRP
432 if (bt.brp != 0u)
433 {
434 uint32_t nbrp = (bt.brp - 1u) & NBRP_FIELD_MAX;
435 uint32_t mask = FDCAN_NBTP_NBRP_Msk;
436 nbtp_mask |= mask;
437 nbtp_new &= ~mask;
438 nbtp_new |= (nbrp << FDCAN_NBTP_NBRP_Pos);
439 }
440
441 // NTSEG1 = PROP_SEG + PHASE_SEG1
442 if (bt.prop_seg != 0u || bt.phase_seg1 != 0u)
443 {
444 uint32_t ntseg1 = (tseg1 - 1u) & NTSEG1_FIELD_MAX;
445 uint32_t mask = FDCAN_NBTP_NTSEG1_Msk;
446 nbtp_mask |= mask;
447 nbtp_new &= ~mask;
448 nbtp_new |= (ntseg1 << FDCAN_NBTP_NTSEG1_Pos);
449 }
450
451 // NTSEG2 = PHASE_SEG2
452 if (bt.phase_seg2 != 0u)
453 {
454 uint32_t ntseg2 = (bt.phase_seg2 - 1u) & NTSEG2_FIELD_MAX;
455 uint32_t mask = FDCAN_NBTP_NTSEG2_Msk;
456 nbtp_mask |= mask;
457 nbtp_new &= ~mask;
458 nbtp_new |= (ntseg2 << FDCAN_NBTP_NTSEG2_Pos);
459 }
460
461 // NSJW
462 if (bt.sjw != 0u)
463 {
464 uint32_t nsjw = (bt.sjw - 1u) & NSJW_FIELD_MAX;
465 uint32_t mask = FDCAN_NBTP_NSJW_Msk;
466 nbtp_mask |= mask;
467 nbtp_new &= ~mask;
468 nbtp_new |= (nsjw << FDCAN_NBTP_NSJW_Pos);
469 }
470
471 if (nbtp_mask != 0u)
472 {
473 nbtp_old &= ~nbtp_mask;
474 nbtp_old |= (nbtp_new & nbtp_mask);
475 can->NBTP = nbtp_old;
476 }
477#endif // FDCAN_NBTP_NBRP_Msk
478
479 // ===== 数据相位 bit timing(CAN FD 才用),写 DBTP =====
480 const auto& dbt = cfg.data_timing;
481
482#ifdef FDCAN_DBTP_DBRP_Msk
483 constexpr uint32_t DBRP_FIELD_MAX = (FDCAN_DBTP_DBRP_Msk >> FDCAN_DBTP_DBRP_Pos);
484 constexpr uint32_t DTSEG1_FIELD_MAX = (FDCAN_DBTP_DTSEG1_Msk >> FDCAN_DBTP_DTSEG1_Pos);
485 constexpr uint32_t DTSEG2_FIELD_MAX = (FDCAN_DBTP_DTSEG2_Msk >> FDCAN_DBTP_DTSEG2_Pos);
486 constexpr uint32_t DSJW_FIELD_MAX = (FDCAN_DBTP_DSJW_Msk >> FDCAN_DBTP_DSJW_Pos);
487
488 constexpr uint32_t DBRP_MAX = DBRP_FIELD_MAX + 1u;
489 constexpr uint32_t DTSEG1_MAX = DTSEG1_FIELD_MAX + 1u;
490 constexpr uint32_t DTSEG2_MAX = DTSEG2_FIELD_MAX + 1u;
491 constexpr uint32_t DSJW_MAX = DSJW_FIELD_MAX + 1u;
492
493 if (dbt.brp != 0u)
494 {
495 if (dbt.brp < 1u || dbt.brp > DBRP_MAX)
496 {
497 ASSERT(false);
498 return ErrorCode::ARG_ERR;
499 }
500 }
501
502 uint32_t dtseg1 = dbt.prop_seg + dbt.phase_seg1;
503 if (dbt.prop_seg != 0u || dbt.phase_seg1 != 0u)
504 {
505 if (dtseg1 < 1u || dtseg1 > DTSEG1_MAX)
506 {
507 ASSERT(false);
508 return ErrorCode::ARG_ERR;
509 }
510 }
511
512 if (dbt.phase_seg2 != 0u)
513 {
514 if (dbt.phase_seg2 < 1u || dbt.phase_seg2 > DTSEG2_MAX)
515 {
516 ASSERT(false);
517 return ErrorCode::ARG_ERR;
518 }
519 }
520
521 if (dbt.sjw != 0u)
522 {
523 if (dbt.sjw < 1u || dbt.sjw > DSJW_MAX)
524 {
525 ASSERT(false);
526 return ErrorCode::ARG_ERR;
527 }
528 if (dbt.phase_seg2 != 0u && dbt.sjw > dbt.phase_seg2)
529 {
530 ASSERT(false);
531 return ErrorCode::ARG_ERR;
532 }
533 }
534
535 uint32_t dbtp_old = can->DBTP;
536 uint32_t dbtp_new = dbtp_old;
537 uint32_t dbtp_mask = 0u;
538
539 if (dbt.brp != 0u)
540 {
541 uint32_t dbrp = (dbt.brp - 1u) & DBRP_FIELD_MAX;
542 uint32_t mask = FDCAN_DBTP_DBRP_Msk;
543 dbtp_mask |= mask;
544 dbtp_new &= ~mask;
545 dbtp_new |= (dbrp << FDCAN_DBTP_DBRP_Pos);
546 }
547
548 if (dbt.prop_seg != 0u || dbt.phase_seg1 != 0u)
549 {
550 uint32_t dt1 = (dtseg1 - 1u) & DTSEG1_FIELD_MAX;
551 uint32_t mask = FDCAN_DBTP_DTSEG1_Msk;
552 dbtp_mask |= mask;
553 dbtp_new &= ~mask;
554 dbtp_new |= (dt1 << FDCAN_DBTP_DTSEG1_Pos);
555 }
556
557 if (dbt.phase_seg2 != 0u)
558 {
559 uint32_t dt2 = (dbt.phase_seg2 - 1u) & DTSEG2_FIELD_MAX;
560 uint32_t mask = FDCAN_DBTP_DTSEG2_Msk;
561 dbtp_mask |= mask;
562 dbtp_new &= ~mask;
563 dbtp_new |= (dt2 << FDCAN_DBTP_DTSEG2_Pos);
564 }
565
566 if (dbt.sjw != 0u)
567 {
568 uint32_t dsjw = (dbt.sjw - 1u) & DSJW_FIELD_MAX;
569 uint32_t mask = FDCAN_DBTP_DSJW_Msk;
570 dbtp_mask |= mask;
571 dbtp_new &= ~mask;
572 dbtp_new |= (dsjw << FDCAN_DBTP_DSJW_Pos);
573 }
574
575 if (dbtp_mask != 0u)
576 {
577 dbtp_old &= ~dbtp_mask;
578 dbtp_old |= (dbtp_new & dbtp_mask);
579 can->DBTP = dbtp_old;
580 }
581#else
582 (void)dbt;
583#endif // FDCAN_DBTP_DBRP_Msk
584
585 // 数据相位 FDMode:这里不动寄存器,只保留在上层语义中使用
586 (void)cfg.fd_mode;
587
588 // 重新启动 FDCAN
589 if (HAL_FDCAN_Start(hcan_) != HAL_OK)
590 {
591 return ErrorCode::FAILED;
592 }
593
594 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_ERROR_PASSIVE, 0);
595 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_ERROR_WARNING, 0);
596 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_BUS_OFF, 0);
597 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_RX_FIFO0_NEW_MESSAGE, 0);
598 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_RX_FIFO1_NEW_MESSAGE, 0);
599 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_TX_FIFO_EMPTY, 0);
600 HAL_FDCAN_ActivateNotification(hcan_, FDCAN_IT_TX_COMPLETE, 0xFFFFFFFF);
601
602 return ErrorCode::OK;
603}
604
606{
607 // 所有带 FDCAN 的 STM32 都通过 RCCEx 提供核时钟查询
608#if defined(RCC_PERIPHCLK_FDCAN)
609 return HAL_RCCEx_GetPeriphCLKFreq(RCC_PERIPHCLK_FDCAN);
610#elif defined(RCC_PERIPHCLK_FDCAN1)
611 return HAL_RCCEx_GetPeriphCLKFreq(RCC_PERIPHCLK_FDCAN1);
612#else
613 // 理论上不会走到这里,有就说明 HAL/RCC 宏不一致
614 ASSERT(false);
615 return 0u;
616#endif
617}
618
619ErrorCode STM32CANFD::AddMessage(const FDPack& pack)
620{
621 if (pack.type == Type::ERROR)
622 {
623 return ErrorCode::ARG_ERR;
624 }
625
626 ASSERT(pack.len <= 64u);
627 if (pack.type != Type::STANDARD && pack.type != Type::EXTENDED)
628 {
629 ASSERT(false);
630 return ErrorCode::FAILED;
631 }
632
633 // 先入池;满则先服务一次再 Put 一次
634 if (tx_pool_fd_.Put(pack) != ErrorCode::OK)
635 {
636 TxService();
637 if (tx_pool_fd_.Put(pack) != ErrorCode::OK)
638 {
639 return ErrorCode::FULL;
640 }
641 }
642
643 TxService();
644 return ErrorCode::OK;
645}
646
648{
649 if (HAL_FDCAN_GetRxMessage(hcan_, fifo, &rx_buff_.header, rx_buff_.pack_fd.data) ==
650 HAL_OK)
651 {
652 if (rx_buff_.header.FDFormat == FDCAN_FD_CAN)
653 {
654 rx_buff_.pack_fd.id = rx_buff_.header.Identifier;
655 rx_buff_.pack_fd.type =
656 (rx_buff_.header.IdType == FDCAN_EXTENDED_ID) ? Type::EXTENDED : Type::STANDARD;
657
658 if (rx_buff_.header.RxFrameType != FDCAN_DATA_FRAME)
659 {
660 if (rx_buff_.pack_fd.type == Type::STANDARD)
661 {
662 rx_buff_.pack_fd.type = Type::REMOTE_STANDARD;
663 }
664 else
665 {
666 rx_buff_.pack_fd.type = Type::REMOTE_EXTENDED;
667 }
668 }
669
670 rx_buff_.pack_fd.len = DlcToBytes(rx_buff_.header.DataLength);
671
672 OnMessage(rx_buff_.pack_fd, true);
673 }
674 else
675 {
676 rx_buff_.pack.id = rx_buff_.header.Identifier;
677 rx_buff_.pack.type =
678 (rx_buff_.header.IdType == FDCAN_EXTENDED_ID) ? Type::EXTENDED : Type::STANDARD;
679
680 uint32_t bytes = DlcToBytes(rx_buff_.header.DataLength);
681 if (bytes > 8u)
682 {
683 bytes = 8u;
684 }
685
686 if (rx_buff_.header.RxFrameType != FDCAN_DATA_FRAME)
687 {
688 if (rx_buff_.pack.type == Type::STANDARD)
689 {
690 rx_buff_.pack.type = Type::REMOTE_STANDARD;
691 }
692 else
693 {
694 rx_buff_.pack.type = Type::REMOTE_EXTENDED;
695 }
696 rx_buff_.pack.dlc = static_cast<uint8_t>(bytes);
697 }
698 else
699 {
700 rx_buff_.pack.dlc = static_cast<uint8_t>(bytes);
701 if (bytes > 0u)
702 {
703 Memory::FastCopy(rx_buff_.pack.data, rx_buff_.pack_fd.data, bytes);
704 }
705 }
706
707 OnMessage(rx_buff_.pack, true);
708 }
709 }
710}
711
712void STM32CANFD::TxService()
713{
714 if (hcan_ == nullptr || hcan_->Instance == nullptr)
715 {
716 return;
717 }
718
719 tx_pend_.store(1u, std::memory_order_release);
720
721 uint32_t expected = 0u;
722 if (!tx_lock_.compare_exchange_strong(expected, 1u, std::memory_order_acquire,
723 std::memory_order_relaxed))
724 {
725 return;
726 }
727
728 for (;;)
729 {
730 tx_pend_.store(0u, std::memory_order_release);
731
732 while (HardwareTxQueueEmptySize() != 0u)
733 {
734 // FD 优先
735 FDPack pfd{};
736 if (tx_pool_fd_.Get(pfd) == ErrorCode::OK)
737 {
738 FDCAN_TxHeaderTypeDef hdr{};
739 BuildTxHeader(pfd, hdr);
740
741 if (HAL_FDCAN_AddMessageToTxFifoQ(hcan_, &hdr, pfd.data) != HAL_OK)
742 {
743 // 发送失败:回队列,必须兜底
744 if (tx_pool_fd_.Put(pfd) != ErrorCode::OK)
745 {
746 ASSERT(false); // 丢包属于异常:池满/实现问题
747 }
748 break; // 不做立即 retry
749 }
750 continue;
751 }
752
753 // Classic
754 ClassicPack pc{};
755 if (tx_pool_.Get(pc) == ErrorCode::OK)
756 {
757 FDCAN_TxHeaderTypeDef hdr{};
758 BuildTxHeader(pc, hdr);
759
760 if (HAL_FDCAN_AddMessageToTxFifoQ(hcan_, &hdr, pc.data) != HAL_OK)
761 {
762 if (tx_pool_.Put(pc) != ErrorCode::OK)
763 {
764 ASSERT(false);
765 }
766 break;
767 }
768 continue;
769 }
770
771 break; // 两个池都空
772 }
773
774 tx_lock_.store(0u, std::memory_order_release);
775
776 if (tx_pend_.load(std::memory_order_acquire) == 0u)
777 {
778 return;
779 }
780
781 expected = 0u;
782 if (!tx_lock_.compare_exchange_strong(expected, 1u, std::memory_order_acquire,
783 std::memory_order_relaxed))
784 {
785 return;
786 }
787 }
788}
789
790void STM32CANFD::ProcessErrorStatusInterrupt(uint32_t error_status_its)
791{
792 FDCAN_ProtocolStatusTypeDef protocol_status = {};
793 HAL_FDCAN_GetProtocolStatus(hcan_, &protocol_status);
794
795#ifdef FDCAN_IT_BUS_OFF
796#ifdef FDCAN_CCCR_INIT
797 if ((error_status_its & FDCAN_IT_BUS_OFF) != 0u && protocol_status.BusOff != 0u)
798 {
799 CLEAR_BIT(hcan_->Instance->CCCR, FDCAN_CCCR_INIT);
800 }
801#endif
802#endif
803
804 CAN::ClassicPack pack{};
805 pack.type = CAN::Type::ERROR;
806 pack.dlc = 0;
807
808 CAN::ErrorID eid = CAN::ErrorID::CAN_ERROR_ID_GENERIC;
809
810 if (protocol_status.BusOff != 0u)
811 {
812 eid = CAN::ErrorID::CAN_ERROR_ID_BUS_OFF;
813 }
814 else if (protocol_status.ErrorPassive != 0u)
815 {
816 eid = CAN::ErrorID::CAN_ERROR_ID_ERROR_PASSIVE;
817 }
818 else if (protocol_status.Warning != 0u)
819 {
820 eid = CAN::ErrorID::CAN_ERROR_ID_ERROR_WARNING;
821 }
822 else
823 {
824 uint32_t lec = protocol_status.LastErrorCode & 0x7u;
825 if (lec == 0u)
826 {
827 lec = protocol_status.DataLastErrorCode & 0x7u;
828 }
829
830 switch (lec)
831 {
832 case 0x01u:
833 eid = CAN::ErrorID::CAN_ERROR_ID_STUFF;
834 break;
835 case 0x02u:
836 eid = CAN::ErrorID::CAN_ERROR_ID_FORM;
837 break;
838 case 0x03u:
839 eid = CAN::ErrorID::CAN_ERROR_ID_ACK;
840 break;
841 case 0x04u:
842 eid = CAN::ErrorID::CAN_ERROR_ID_BIT1;
843 break;
844 case 0x05u:
845 eid = CAN::ErrorID::CAN_ERROR_ID_BIT0;
846 break;
847 case 0x06u:
848 eid = CAN::ErrorID::CAN_ERROR_ID_CRC;
849 break;
850 default:
851 eid = CAN::ErrorID::CAN_ERROR_ID_OTHER;
852 break;
853 }
854 }
855
856 pack.id = static_cast<uint32_t>(eid);
857
858 OnMessage(pack, true);
859}
860
862{
863 if (hcan_ == nullptr || hcan_->Instance == nullptr)
864 {
865 return ErrorCode::ARG_ERR;
866 }
867
868 FDCAN_ErrorCountersTypeDef counters{};
869 if (HAL_FDCAN_GetErrorCounters(hcan_, &counters) != HAL_OK)
870 {
871 return ErrorCode::FAILED;
872 }
873
874 FDCAN_ProtocolStatusTypeDef proto{};
875 if (HAL_FDCAN_GetProtocolStatus(hcan_, &proto) != HAL_OK)
876 {
877 return ErrorCode::FAILED;
878 }
879
880 state.tx_error_counter = counters.TxErrorCnt;
881 state.rx_error_counter = counters.RxErrorCnt;
882
883 state.bus_off = (proto.BusOff != 0u);
884 state.error_passive = (proto.ErrorPassive != 0u);
885 state.error_warning = (proto.Warning != 0u);
886
887 return ErrorCode::OK;
888}
889
890extern "C" void HAL_FDCAN_ErrorCallback(FDCAN_HandleTypeDef* hcan)
891{
892 hcan->ErrorCode = HAL_FDCAN_ERROR_NONE;
893 auto can = STM32CANFD::map[STM32_FDCAN_GetID(hcan->Instance)];
894 if (can)
895 {
896 can->TxService();
897 }
898}
899
900extern "C" void HAL_FDCAN_ErrorStatusCallback(FDCAN_HandleTypeDef* hfdcan,
901 uint32_t ErrorStatusITs)
902{
903 auto can = STM32CANFD::map[STM32_FDCAN_GetID(hfdcan->Instance)];
904 if (can)
905 {
906 can->ProcessErrorStatusInterrupt(ErrorStatusITs);
907 can->TxService();
908 }
909}
910
911extern "C" void HAL_FDCAN_TxBufferCompleteCallback(FDCAN_HandleTypeDef* hcan,
912 uint32_t BufferIndexes)
913{
914 UNUSED(BufferIndexes);
915 auto can = STM32CANFD::map[STM32_FDCAN_GetID(hcan->Instance)];
916 if (can)
917 {
918 can->TxService();
919 }
920}
921
922extern "C" void HAL_FDCAN_TxFifoEmptyCallback(FDCAN_HandleTypeDef* hcan)
923{
924 auto can = STM32CANFD::map[STM32_FDCAN_GetID(hcan->Instance)];
925 if (can)
926 {
927 can->TxService();
928 }
929}
930
931extern "C" void HAL_FDCAN_RxFifo0Callback(FDCAN_HandleTypeDef* hcan, uint32_t RxFifo0ITs)
932{
933 UNUSED(RxFifo0ITs);
934 auto can = STM32CANFD::map[STM32_FDCAN_GetID(hcan->Instance)];
935 if (can)
936 {
937 can->ProcessRxInterrupt(FDCAN_RX_FIFO0);
938 }
939}
940
941extern "C" void HAL_FDCAN_RxFifo1Callback(FDCAN_HandleTypeDef* hcan, uint32_t RxFifo1ITs)
942{
943 UNUSED(RxFifo1ITs);
944 auto can = STM32CANFD::map[STM32_FDCAN_GetID(hcan->Instance)];
945 if (can)
946 {
947 can->ProcessRxInterrupt(FDCAN_RX_FIFO1);
948 }
949}
950
951#endif
@ 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).
ErrorID
ClassicPack::type == Type::ERROR 时使用的虚拟 ID。 Virtual IDs used when ClassicPack::type == Type::ERROR.
Definition can.hpp:146
FDCAN 通信抽象类,扩展支持 CAN FD 帧。 Abstract class for FDCAN communication with CAN FD frame support.
Definition can.hpp:247
void OnMessage(const FDPack &pack, bool in_isr)
分发接收到的 FD CAN 帧。 Dispatch a received FD CAN frame.
Definition can.cpp:54
ErrorCode Put(const Data &data)
向池中放入一个元素 / Put an element into the pool
ErrorCode Get(Data &data)
从池中取出一个元素 / Retrieve an element from the pool
static void FastCopy(void *dst, const void *src, size_t size)
快速内存拷贝 / Fast memory copy
Definition libxr_mem.cpp:3
STM32CANFD 类,用于处理 STM32 系统的 CANFD 通道。 Provides handling for STM32 CANFD.
void ProcessErrorStatusInterrupt(uint32_t error_status_its)
处理错误状态中断
uint32_t GetClockFreq() const override
获取 FDCAN 外设时钟(Hz)。Get FDCAN kernel clock (Hz).
ErrorCode Init(void)
初始化
void ProcessRxInterrupt(uint32_t fifo)
处理接收中断
STM32CANFD(FDCAN_HandleTypeDef *hcan, uint32_t queue_size)
STM32CANFD 类,用于处理 STM32 系统的 CANFD 通道。 Provides handling for STM32 CANFD.
ErrorCode GetErrorState(CAN::ErrorState &state) const override
查询当前错误状态(快照)。 Query current FDCAN error state (snapshot).
ErrorCode AddMessage(const ClassicPack &pack) override
添加经典 CAN 消息。Add classic CAN message.
size_t HardwareTxQueueEmptySize()
获取硬件发送队列空闲大小
ErrorCode SetConfig(const CAN::Configuration &cfg) override
设置 CAN/FDCAN 配置。Set CAN/FDCAN configuration.
LibXR 命名空间
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
float sample_point
仲裁相位采样点(0~1)。Nominal sample point (0–1).
Definition can.hpp:64
Mode mode
工作模式。Operating mode.
Definition can.hpp:66
uint32_t bitrate
仲裁相位目标波特率。Target nominal bitrate.
Definition can.hpp:63
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
FDCAN 配置参数,扩展 CAN::Configuration。 FDCAN configuration, extending CAN::Configuration.
Definition can.hpp:323
FDMode fd_mode
FD 模式配置。FD mode configuration.
Definition can.hpp:327
DataBitTiming data_timing
数据相位位时序。Data-phase bit timing.
Definition can.hpp:326
uint32_t prop_seg
传播段。Propagation segment.
Definition can.hpp:300
CAN FD 帧数据结构。CAN FD frame structure.
Definition can.hpp:265
Type type
帧类型。Frame type.
Definition can.hpp:267
uint8_t len
数据长度(0~64)。Data length (0–64 bytes).
Definition can.hpp:268