can.c 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. #include <stdio.h>
  2. #include "os/queue.h"
  3. #include "bsp/bsp.h"
  4. #include "bsp/can.h"
  5. #include "libs/utils.h"
  6. #include "libs/circle_buffer.h"
  7. #define CAN_RX_MESSAGE_RX_ID 1
  8. #define CAN_SEND_QUEUE_SIZE 32
  9. #define RX_ID_OFFSET 16
  10. #define CAN_SEND_OK 0
  11. #define CAN_SEND_ERROR -1
  12. #define CAN_SEND_WAIT_TIMEOUT -2
  13. #define TX_NUM 64
  14. #define RX_NUM 64
  15. static c_buffer_t g_tx_circle;
  16. static c_buffer_t g_rx_circle;
  17. static uint8_t _g_tx_buffer[sizeof(can_trasnmit_message_struct) * TX_NUM + 1];
  18. static uint8_t _g_rx_buffer[sizeof(can_receive_message_struct) * RX_NUM + 1];
  19. static int shark_send_can0_data(can_trasnmit_message_struct *P_message);
  20. static uint8_t can_get_mailbox(uint32_t can_periph);
  21. /* this function can be overide by app, which need recv the can frame */
  22. __weak void handle_can_frame(can_id_t id, uint8_t *data, int len){
  23. }
  24. void can_rx_poll(void){
  25. can_receive_message_struct message;
  26. if (circle_get_data(&g_rx_circle, (uint8_t *)&message, sizeof(message)) != sizeof(message)) {
  27. return;
  28. }
  29. can_id_t can_id;
  30. can_id.id = message.rx_efid;
  31. handle_can_frame(can_id, message.rx_data, message.rx_dlen);
  32. return ;
  33. }
  34. void can_tx_poll(void){
  35. can_trasnmit_message_struct can_tr_m;
  36. if (CAN_ERR(CAN0) & CAN_ERR_BOERR){
  37. shark_can0_reset();
  38. }
  39. while (can_get_mailbox(CAN0) != CAN_NOMAILBOX) {
  40. if (circle_get_data(&g_tx_circle, (uint8_t * )&can_tr_m, sizeof(can_tr_m)) != sizeof(can_tr_m)) {
  41. break;
  42. }
  43. can_message_transmit(CAN0,&can_tr_m);
  44. }
  45. }
  46. static u32 _can_poll_task(void *args) {
  47. can_rx_poll();
  48. can_tx_poll();
  49. return 0;
  50. }
  51. static __inline__ void can_fifo_recv(int fifo){
  52. can_receive_message_struct Rxmessage;
  53. can_message_receive(CAN0, fifo, &Rxmessage);
  54. circle_put_data(&g_rx_circle, (uint8_t *)&Rxmessage, sizeof(Rxmessage));
  55. }
  56. void USBD_LP_CAN0_RX0_IRQHandler(void)
  57. {
  58. can_fifo_recv(CAN_FIFO0);
  59. }
  60. void CAN0_RX1_IRQHandler(void)
  61. {
  62. can_fifo_recv(CAN_FIFO1);
  63. }
  64. static void shark_can0_txrx_pin_config(void){
  65. /* enable can clock */
  66. rcu_periph_clock_enable(RCU_CAN0);
  67. rcu_periph_clock_enable(RCU_GPIOA);
  68. rcu_periph_clock_enable(RCU_AF);
  69. //gpio_pin_remap_config(GPIO_CAN_FULL_REMAP,DISABLE);
  70. //gpio_pin_remap_config(GPIO_CAN_PARTIAL_REMAP,ENABLE);
  71. gpio_init(GPIOA, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_12);
  72. gpio_init(GPIOA, GPIO_MODE_IPU, GPIO_OSPEED_50MHZ, GPIO_PIN_11);
  73. }
  74. static void shark_can0_config(void)
  75. {
  76. can_parameter_struct can_parameter;
  77. can_struct_para_init(CAN_INIT_STRUCT, &can_parameter);
  78. /* initialize CAN register */
  79. can_deinit(CAN0);
  80. //can_deinit(CAN1);
  81. /* initialize CAN parameters */
  82. can_parameter.time_triggered = DISABLE;
  83. can_parameter.auto_bus_off_recovery = ENABLE;
  84. can_parameter.auto_wake_up = DISABLE;
  85. can_parameter.rec_fifo_overwrite = DISABLE;
  86. can_parameter.trans_fifo_order = ENABLE;
  87. can_parameter.no_auto_retrans = ENABLE;
  88. #if CAN0_SPEED==250 //250k bps
  89. if (SystemCoreClock == 120000000) {
  90. can_parameter.resync_jump_width = CAN_BT_SJW_1TQ;
  91. can_parameter.time_segment_1 = CAN_BT_BS1_5TQ;
  92. can_parameter.time_segment_2 = CAN_BT_BS2_4TQ;
  93. } else {
  94. can_parameter.resync_jump_width = CAN_BT_SJW_1TQ;
  95. can_parameter.time_segment_1 = CAN_BT_BS1_5TQ;
  96. can_parameter.time_segment_2 = CAN_BT_BS2_3TQ;
  97. }
  98. #endif
  99. can_parameter.working_mode = CAN_NORMAL_MODE;
  100. can_parameter.prescaler = 24;
  101. /* initialize CAN */
  102. can_init(CAN0, &can_parameter);
  103. /* recv my can ID, use fifo0 */
  104. can_filter_mask_mode_init(CAN_MY_ADDRESS, CAN_FILTER_DEST_MASK, CAN_EXTENDED_FIFO0, 0);
  105. nvic_priority_group_set(NVIC_PRIGROUP_PRE4_SUB0);
  106. nvic_irq_enable(USBD_LP_CAN0_RX0_IRQn,CAN_IRQ_PRIORITY,0);
  107. /* enable can receive FIFO0 not empty interrupt */
  108. can_interrupt_enable(CAN0, CAN_INTEN_RFNEIE0);
  109. //can_filter_mask_mode_init(0x1A0, CAN_FILTER_DEST_MASK, CAN_EXTENDED_FIFO1, 1);
  110. //nvic_irq_enable(CAN0_RX1_IRQn,CAN_IRQ_PRIORITY,0);
  111. /* enable can receive FIFO1 not empty interrupt */
  112. //can_interrupt_enable(CAN0, CAN_INTEN_RFNEIE1);
  113. }
  114. static int shark_send_can0_data(can_trasnmit_message_struct *P_message){
  115. can_tx_poll();
  116. if (circle_put_data(&g_tx_circle, (u8 *)P_message, sizeof(can_trasnmit_message_struct))){
  117. return CAN_SEND_OK;
  118. }
  119. return CAN_SEND_ERROR;
  120. }
  121. static uint8_t can_get_mailbox(uint32_t can_periph)
  122. {
  123. uint8_t mailbox_number = CAN_MAILBOX0;
  124. /* select one empty mailbox */
  125. if(CAN_TSTAT_TME0 == (CAN_TSTAT(can_periph)&CAN_TSTAT_TME0)){
  126. mailbox_number = CAN_MAILBOX0;
  127. }else if(CAN_TSTAT_TME1 == (CAN_TSTAT(can_periph)&CAN_TSTAT_TME1)){
  128. mailbox_number = CAN_MAILBOX1;
  129. }else if(CAN_TSTAT_TME2 == (CAN_TSTAT(can_periph)&CAN_TSTAT_TME2)){
  130. mailbox_number = CAN_MAILBOX2;
  131. }else{
  132. mailbox_number = CAN_NOMAILBOX;
  133. }
  134. /* return no mailbox empty */
  135. if(CAN_NOMAILBOX == mailbox_number){
  136. return CAN_NOMAILBOX;
  137. }
  138. return mailbox_number;
  139. }
  140. int shark_can0_send_message(uint32_t can_id, const void*buff, int len){
  141. can_trasnmit_message_struct trasnmit_msg;
  142. can_id_t can_frame_id;
  143. u32 total_frames = ((len + 7) >> 3);
  144. int send_len = len;
  145. u32 frame_id = 1;
  146. can_frame_id.id = can_id;
  147. can_frame_id.total = total_frames;
  148. while(send_len > 0){
  149. can_frame_id.idx = frame_id;
  150. trasnmit_msg.tx_sfid = 0;
  151. trasnmit_msg.tx_efid = can_frame_id.id;
  152. trasnmit_msg.tx_ft = CAN_FT_DATA;
  153. trasnmit_msg.tx_ff = CAN_FF_EXTENDED;
  154. trasnmit_msg.tx_dlen = min(CAN_DATA_SIZE,send_len);
  155. memcpy((char *)trasnmit_msg.tx_data, (char *)buff + (len - send_len), trasnmit_msg.tx_dlen);
  156. send_len -= trasnmit_msg.tx_dlen;
  157. frame_id ++;
  158. if (shark_send_can0_data(&trasnmit_msg) != CAN_SEND_OK){
  159. return CAN_SEND_ERROR;
  160. }
  161. }
  162. return CAN_SEND_OK;
  163. }
  164. int shark_can0_send_ext_message(uint32_t can_id, const void*buff, int len){
  165. can_trasnmit_message_struct trasnmit_msg;
  166. trasnmit_msg.tx_sfid = 0;
  167. trasnmit_msg.tx_efid = can_id;
  168. trasnmit_msg.tx_ft = CAN_FT_DATA;
  169. trasnmit_msg.tx_ff = CAN_FF_EXTENDED;
  170. trasnmit_msg.tx_dlen = min(CAN_DLC_LENGTH,len);
  171. memcpy((char *)trasnmit_msg.tx_data, (char *)buff, trasnmit_msg.tx_dlen);
  172. if (shark_send_can0_data(&trasnmit_msg) != CAN_SEND_OK){
  173. return CAN_SEND_ERROR;
  174. }
  175. return CAN_SEND_OK;
  176. }
  177. void shark_can0_reset(void){
  178. shark_can0_txrx_pin_config();
  179. shark_can0_config();
  180. }
  181. void shark_can0_deinit(void){
  182. can_deinit(CAN0);
  183. rcu_periph_clock_disable(RCU_CAN0);
  184. }
  185. void shark_can0_init(void){
  186. circle_buffer_init(&g_tx_circle, _g_tx_buffer, sizeof(_g_tx_buffer));
  187. circle_buffer_init(&g_rx_circle, _g_rx_buffer, sizeof(_g_rx_buffer));
  188. shark_task_create(_can_poll_task, NULL);
  189. shark_can0_txrx_pin_config();
  190. shark_can0_config();
  191. }