2021-01-28 02:29:12 +03:00
|
|
|
// Serial over CAN emulation for STM32 boards.
|
|
|
|
//
|
|
|
|
// Copyright (C) 2019 Eug Krashtan <eug.krashtan@gmail.com>
|
|
|
|
// Copyright (C) 2020 Pontus Borg <glpontus@gmail.com>
|
2021-01-28 08:54:45 +03:00
|
|
|
// Copyright (C) 2021 Kevin O'Connor <kevin@koconnor.net>
|
2021-01-28 02:29:12 +03:00
|
|
|
//
|
|
|
|
// This file may be distributed under the terms of the GNU GPLv3 license.
|
|
|
|
|
|
|
|
#include <string.h> // memcpy
|
|
|
|
#include "autoconf.h" // CONFIG_MACH_STM32F1
|
2021-01-28 08:54:45 +03:00
|
|
|
#include "board/irq.h" // irq_disable
|
2021-01-28 02:29:12 +03:00
|
|
|
#include "can.h" // SHORT_UUID_LEN
|
2020-06-25 01:59:38 +03:00
|
|
|
#include "command.h" // DECL_CONSTANT_STR
|
2021-01-28 02:29:12 +03:00
|
|
|
#include "fasthash.h" // fasthash64
|
|
|
|
#include "generic/armcm_boot.h" // armcm_enable_irq
|
|
|
|
#include "generic/serial_irq.h" // serial_rx_byte
|
2020-06-25 01:59:38 +03:00
|
|
|
#include "internal.h" // enable_pclock
|
|
|
|
#include "sched.h" // DECL_INIT
|
2021-01-28 02:29:12 +03:00
|
|
|
|
|
|
|
#if CONFIG_CAN_PINS_PA11_PA12
|
|
|
|
DECL_CONSTANT_STR("RESERVE_PINS_CAN", "PA11,PA12");
|
|
|
|
#define GPIO_Rx GPIO('A', 11)
|
|
|
|
#define GPIO_Tx GPIO('A', 12)
|
2020-06-25 01:59:38 +03:00
|
|
|
#endif
|
2021-01-28 02:29:12 +03:00
|
|
|
#if CONFIG_CAN_PINS_PB8_PB9
|
|
|
|
DECL_CONSTANT_STR("RESERVE_PINS_CAN", "PB8,PB9");
|
|
|
|
#define GPIO_Rx GPIO('B', 8)
|
|
|
|
#define GPIO_Tx GPIO('B', 9)
|
2020-06-25 01:59:38 +03:00
|
|
|
#endif
|
2021-01-28 02:29:12 +03:00
|
|
|
#if CONFIG_CAN_PINS_PI8_PH13
|
|
|
|
DECL_CONSTANT_STR("RESERVE_PINS_CAN", "PI8,PH13");
|
|
|
|
#define GPIO_Rx GPIO('I', 8)
|
|
|
|
#define GPIO_Tx GPIO('H', 13)
|
2020-06-25 01:59:38 +03:00
|
|
|
#endif
|
2021-01-28 02:29:12 +03:00
|
|
|
#if CONFIG_CAN_PINS_PB5_PB6
|
|
|
|
DECL_CONSTANT_STR("RESERVE_PINS_CAN", "PB5,PB6");
|
|
|
|
#define GPIO_Rx GPIO('B', 5)
|
|
|
|
#define GPIO_Tx GPIO('B', 6)
|
2020-06-25 01:59:38 +03:00
|
|
|
#endif
|
2021-01-28 02:29:12 +03:00
|
|
|
#if CONFIG_CAN_PINS_PB12_PB13
|
|
|
|
DECL_CONSTANT_STR("RESERVE_PINS_CAN", "PB12,PB13");
|
|
|
|
#define GPIO_Rx GPIO('B', 12)
|
|
|
|
#define GPIO_Tx GPIO('B', 13)
|
2020-06-25 01:59:38 +03:00
|
|
|
#endif
|
|
|
|
|
2021-01-28 02:29:12 +03:00
|
|
|
#if CONFIG_MACH_STM32F0
|
|
|
|
#define SOC_CAN CAN
|
|
|
|
#define CAN_RX0_IRQn CEC_CAN_IRQn
|
|
|
|
#define CAN_RX1_IRQn CEC_CAN_IRQn
|
|
|
|
#define CAN_TX_IRQn CEC_CAN_IRQn
|
|
|
|
#define CAN_SCE_IRQn CEC_CAN_IRQn
|
|
|
|
#define CAN_FUNCTION GPIO_FUNCTION(4) // Alternative function mapping number
|
2020-06-25 01:59:38 +03:00
|
|
|
#endif
|
|
|
|
|
2021-01-28 02:29:12 +03:00
|
|
|
#if CONFIG_MACH_STM32F1
|
|
|
|
#define SOC_CAN CAN1
|
|
|
|
#define CAN_RX0_IRQn CAN1_RX0_IRQn
|
|
|
|
#define CAN_RX1_IRQn CAN1_RX1_IRQn
|
|
|
|
#define CAN_TX_IRQn CAN1_TX_IRQn
|
|
|
|
#define CAN_SCE_IRQn CAN1_SCE_IRQn
|
|
|
|
#define CAN_FUNCTION GPIO_FUNCTION(9) // Alternative function mapping number
|
2020-06-25 01:59:38 +03:00
|
|
|
#endif
|
|
|
|
|
2021-01-28 02:29:12 +03:00
|
|
|
#if CONFIG_MACH_STM32F4
|
|
|
|
#warning CAN on STM32F4 is untested
|
|
|
|
#if (CONFIG_CAN_PINS_PA11_PA12 || CONFIG_CAN_PINS_PB8_PB9 \
|
|
|
|
|| CONFIG_CAN_PINS_PI8_PH13)
|
|
|
|
#define SOC_CAN CAN1
|
|
|
|
#define CAN_RX0_IRQn CAN1_RX0_IRQn
|
|
|
|
#define CAN_RX1_IRQn CAN1_RX1_IRQn
|
|
|
|
#define CAN_TX_IRQn CAN1_TX_IRQn
|
|
|
|
#define CAN_SCE_IRQn CAN1_SCE_IRQn
|
|
|
|
#elif CONFIG_CAN_PINS_PB5_PB6 || CONFIG_CAN_PINS_PB12_PB13
|
|
|
|
#define SOC_CAN CAN2
|
|
|
|
#define CAN_RX0_IRQn CAN2_RX0_IRQn
|
|
|
|
#define CAN_RX1_IRQn CAN2_RX1_IRQn
|
|
|
|
#define CAN_TX_IRQn CAN2_TX_IRQn
|
|
|
|
#define CAN_SCE_IRQn CAN2_SCE_IRQn
|
|
|
|
#else
|
|
|
|
#error Uknown pins for STMF32F4 CAN
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define CAN_FUNCTION GPIO_FUNCTION(9) // Alternative function mapping number
|
2020-06-25 01:59:38 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef SOC_CAN
|
2021-01-28 02:29:12 +03:00
|
|
|
#error No known CAN device for configured MCU
|
2020-06-25 01:59:38 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
static uint16_t MyCanId = 0;
|
|
|
|
|
2021-01-28 02:29:12 +03:00
|
|
|
static int
|
|
|
|
can_find_empty_tx_mbox(void)
|
|
|
|
{
|
2020-06-25 01:59:38 +03:00
|
|
|
uint32_t tsr = SOC_CAN->TSR;
|
2021-01-28 05:58:04 +03:00
|
|
|
if (tsr & (CAN_TSR_TME0|CAN_TSR_TME1|CAN_TSR_TME2))
|
|
|
|
return (tsr & CAN_TSR_CODE) >> CAN_TSR_CODE_Pos;
|
2020-06-25 01:59:38 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2021-01-28 02:29:12 +03:00
|
|
|
static void
|
|
|
|
can_transmit_mbox(uint32_t id, int mbox, uint32_t dlc, uint8_t *pkt)
|
2020-06-25 01:59:38 +03:00
|
|
|
{
|
2021-01-28 02:29:12 +03:00
|
|
|
CAN_TxMailBox_TypeDef *mb = &SOC_CAN->sTxMailBox[mbox];
|
2020-06-25 01:59:38 +03:00
|
|
|
|
|
|
|
/* Set up the DLC */
|
2021-01-28 05:15:33 +03:00
|
|
|
mb->TDTR = (mb->TDTR & 0xFFFFFFF0) | (dlc & 0x0F);
|
2020-06-25 01:59:38 +03:00
|
|
|
|
|
|
|
/* Set up the data field */
|
2021-01-28 02:29:12 +03:00
|
|
|
if (pkt) {
|
|
|
|
mb->TDLR = (((uint32_t)pkt[3] << 24)
|
|
|
|
| ((uint32_t)pkt[2] << 16)
|
|
|
|
| ((uint32_t)pkt[1] << 8)
|
|
|
|
| ((uint32_t)pkt[0] << 0));
|
|
|
|
mb->TDHR = (((uint32_t)pkt[7] << 24)
|
|
|
|
| ((uint32_t)pkt[6] << 16)
|
|
|
|
| ((uint32_t)pkt[5] << 8)
|
|
|
|
| ((uint32_t)pkt[4] << 0));
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|
|
|
|
|
2021-01-28 02:29:12 +03:00
|
|
|
/* Request transmission */
|
2021-01-28 05:15:33 +03:00
|
|
|
mb->TIR = (id << CAN_TI0R_STID_Pos) | CAN_TI0R_TXRQ;
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|
|
|
|
|
2021-01-28 08:54:45 +03:00
|
|
|
// Blocking transmit function
|
2021-01-28 02:29:12 +03:00
|
|
|
static void
|
|
|
|
can_transmit(uint32_t id, uint32_t dlc, uint8_t *pkt)
|
2020-06-25 01:59:38 +03:00
|
|
|
{
|
|
|
|
int mbox = -1;
|
|
|
|
|
|
|
|
do {
|
2021-01-28 02:29:12 +03:00
|
|
|
mbox = can_find_empty_tx_mbox();
|
|
|
|
} while (mbox < 0);
|
2020-06-25 01:59:38 +03:00
|
|
|
|
|
|
|
can_transmit_mbox(id, mbox, dlc, pkt);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert Unique 96-bit value into 48 bit representation
|
2021-01-28 02:29:12 +03:00
|
|
|
static void
|
|
|
|
pack_uuid(uint8_t *u)
|
2020-06-25 01:59:38 +03:00
|
|
|
{
|
2020-08-10 04:07:59 +03:00
|
|
|
uint64_t hash = fasthash64((uint8_t*)UID_BASE, 12, 0xA16231A7);
|
|
|
|
memcpy(u, &hash, SHORT_UUID_LEN);
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|
|
|
|
|
2021-01-28 05:02:32 +03:00
|
|
|
#define CAN_FILTER_NUMBER 0
|
|
|
|
|
|
|
|
static void
|
|
|
|
can_set_filter(uint32_t id1, uint32_t id2)
|
|
|
|
{
|
|
|
|
uint32_t filternbrbitpos = 1 << CAN_FILTER_NUMBER;
|
|
|
|
|
|
|
|
/* Select the start slave bank */
|
|
|
|
SOC_CAN->FMR |= CAN_FMR_FINIT;
|
|
|
|
/* Initialisation mode for the filter */
|
|
|
|
SOC_CAN->FA1R = 0;
|
|
|
|
|
|
|
|
SOC_CAN->sFilterRegister[CAN_FILTER_NUMBER].FR1 = id1 << (5 + 16);
|
|
|
|
SOC_CAN->sFilterRegister[CAN_FILTER_NUMBER].FR2 = id2 << (5 + 16);
|
|
|
|
|
|
|
|
/* Identifier list mode for the filter */
|
|
|
|
SOC_CAN->FM1R = filternbrbitpos;
|
|
|
|
/* 32-bit scale for the filter */
|
|
|
|
SOC_CAN->FS1R = filternbrbitpos;
|
|
|
|
|
|
|
|
/* FIFO 0 assigned for the filter */
|
|
|
|
SOC_CAN->FFA1R = 0;
|
|
|
|
|
|
|
|
/* Filter activation */
|
|
|
|
SOC_CAN->FA1R = filternbrbitpos;
|
|
|
|
/* Leave the initialisation mode for the filter */
|
|
|
|
SOC_CAN->FMR &= ~CAN_FMR_FINIT;
|
|
|
|
}
|
|
|
|
|
2021-01-28 02:29:12 +03:00
|
|
|
static void
|
2021-01-28 08:54:45 +03:00
|
|
|
can_process_data(uint32_t id, uint32_t dlc, uint8_t *data)
|
2020-06-25 01:59:38 +03:00
|
|
|
{
|
2021-01-28 08:54:45 +03:00
|
|
|
int i;
|
|
|
|
for (i=0; i < dlc; i++)
|
|
|
|
serial_rx_byte(data[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
can_process_ping(uint32_t id, uint32_t dlc, uint8_t *data)
|
|
|
|
{
|
|
|
|
can_transmit(MyCanId+1, 0, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
can_process_reset(uint32_t id, uint32_t dlc, uint8_t *data)
|
|
|
|
{
|
|
|
|
uint32_t reset_id = data[0] | (data[1] << 8);
|
|
|
|
if (reset_id == MyCanId)
|
|
|
|
NVIC_SystemReset();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
can_process_uuid(uint32_t id, uint32_t dlc, uint8_t *data)
|
|
|
|
{
|
|
|
|
if (MyCanId)
|
|
|
|
return;
|
|
|
|
uint8_t short_uuid[SHORT_UUID_LEN];
|
|
|
|
pack_uuid(short_uuid);
|
|
|
|
can_transmit(PKT_ID_UUID_RESP, SHORT_UUID_LEN, short_uuid);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
can_process_set_id(uint32_t id, uint32_t dlc, uint8_t *data)
|
|
|
|
{
|
|
|
|
uint8_t short_uuid[SHORT_UUID_LEN];
|
|
|
|
pack_uuid(short_uuid);
|
|
|
|
|
|
|
|
// compare my UUID with packet to check if this packet mine
|
|
|
|
if (memcmp(&data[2], short_uuid, SHORT_UUID_LEN) == 0) {
|
|
|
|
MyCanId = data[0] | (data[1] << 8);
|
|
|
|
can_set_filter(MyCanId, PKT_ID_UUID);
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-28 08:54:45 +03:00
|
|
|
static void
|
|
|
|
can_process(uint32_t id, uint32_t dlc, uint8_t *data)
|
|
|
|
{
|
|
|
|
if (id == MyCanId) {
|
|
|
|
if (dlc)
|
|
|
|
can_process_data(id, dlc, data);
|
|
|
|
else
|
|
|
|
can_process_ping(id, dlc, data);
|
|
|
|
} else if (id == PKT_ID_UUID) {
|
|
|
|
if (dlc)
|
|
|
|
can_process_reset(id, dlc, data);
|
|
|
|
else
|
|
|
|
can_process_uuid(id, dlc, data);
|
|
|
|
} else if (id==PKT_ID_SET) {
|
|
|
|
can_process_set_id(id, dlc, data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct task_wake canbus_wake;
|
|
|
|
|
2020-06-25 01:59:38 +03:00
|
|
|
void
|
2021-01-28 08:54:45 +03:00
|
|
|
can_dispatch_task(void)
|
2020-06-25 01:59:38 +03:00
|
|
|
{
|
2021-01-28 08:54:45 +03:00
|
|
|
if (!sched_check_wake(&canbus_wake))
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Check for rx
|
|
|
|
for (;;) {
|
|
|
|
if (!(SOC_CAN->RF0R & CAN_RF0R_FMP0)) {
|
|
|
|
// All rx mboxes empty, enable wake on rx IRQ
|
|
|
|
irq_disable();
|
|
|
|
SOC_CAN->IER |= CAN_IER_FMPIE0;
|
|
|
|
irq_enable();
|
|
|
|
break;
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|
2021-01-28 08:54:45 +03:00
|
|
|
|
|
|
|
// Read and ack packet
|
|
|
|
CAN_FIFOMailBox_TypeDef *mb = &SOC_CAN->sFIFOMailBox[0];
|
|
|
|
uint32_t id = (mb->RIR >> CAN_RI0R_STID_Pos) & 0x7FF;
|
|
|
|
uint32_t dlc = mb->RDTR & CAN_RDT0R_DLC;
|
|
|
|
uint32_t rdlr = mb->RDLR, rdhr = mb->RDHR;
|
|
|
|
SOC_CAN->RF0R = CAN_RF0R_RFOM0;
|
|
|
|
|
|
|
|
// Process packet
|
|
|
|
uint8_t data[8];
|
|
|
|
data[0] = (rdlr >> 0) & 0xff;
|
|
|
|
data[1] = (rdlr >> 8) & 0xff;
|
|
|
|
data[2] = (rdlr >> 16) & 0xff;
|
|
|
|
data[3] = (rdlr >> 24) & 0xff;
|
|
|
|
data[4] = (rdhr >> 0) & 0xff;
|
|
|
|
data[5] = (rdhr >> 8) & 0xff;
|
|
|
|
data[6] = (rdhr >> 16) & 0xff;
|
|
|
|
data[7] = (rdhr >> 24) & 0xff;
|
|
|
|
can_process(id, dlc, data);
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|
2021-01-28 08:54:45 +03:00
|
|
|
|
|
|
|
// Check for tx data
|
|
|
|
for (;;) {
|
|
|
|
int mbox = can_find_empty_tx_mbox();
|
|
|
|
if (mbox < 0) {
|
|
|
|
// All tx mboxes full, enable wake on tx IRQ
|
|
|
|
irq_disable();
|
|
|
|
SOC_CAN->IER |= CAN_IER_TMEIE;
|
|
|
|
irq_enable();
|
|
|
|
break;
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|
2021-01-28 08:54:45 +03:00
|
|
|
int i;
|
|
|
|
uint8_t databuf[8];
|
|
|
|
for (i=0; i<8; i++) {
|
|
|
|
if (serial_get_tx_byte(&(databuf[i])) == -1)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!i)
|
|
|
|
break;
|
|
|
|
can_transmit_mbox(MyCanId+1, mbox, i, databuf);
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|
2021-01-28 08:54:45 +03:00
|
|
|
}
|
|
|
|
DECL_TASK(can_dispatch_task);
|
2020-06-25 01:59:38 +03:00
|
|
|
|
2021-01-28 08:54:45 +03:00
|
|
|
// This function handles CAN global interrupts
|
|
|
|
void
|
|
|
|
CAN_IRQHandler(void)
|
|
|
|
{
|
|
|
|
if (SOC_CAN->RF0R & CAN_RF0R_FMP0) {
|
|
|
|
// Rx
|
|
|
|
SOC_CAN->IER &= ~CAN_IER_FMPIE0;
|
|
|
|
sched_wake_task(&canbus_wake);
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|
2021-01-28 08:54:45 +03:00
|
|
|
uint32_t ier = SOC_CAN->IER;
|
|
|
|
if (ier & CAN_IER_TMEIE
|
|
|
|
&& SOC_CAN->TSR & (CAN_TSR_RQCP0|CAN_TSR_RQCP1|CAN_TSR_RQCP2)) {
|
|
|
|
// Tx
|
|
|
|
SOC_CAN->IER &= ~CAN_IER_TMEIE;
|
|
|
|
sched_wake_task(&canbus_wake);
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline const uint32_t
|
|
|
|
make_btr(uint32_t sjw, // Sync jump width, ... hmm
|
|
|
|
uint32_t time_seg1, // time segment before sample point, 1 .. 16
|
|
|
|
uint32_t time_seg2, // time segment after sample point, 1 .. 8
|
|
|
|
uint32_t brp) // Baud rate prescaler, 1 .. 1024
|
|
|
|
{
|
2021-01-28 02:29:12 +03:00
|
|
|
return (((uint32_t)(sjw-1)) << CAN_BTR_SJW_Pos
|
|
|
|
| ((uint32_t)(time_seg1-1)) << CAN_BTR_TS1_Pos
|
|
|
|
| ((uint32_t)(time_seg2-1)) << CAN_BTR_TS2_Pos
|
|
|
|
| ((uint32_t)(brp - 1)) << CAN_BTR_BRP_Pos);
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline const uint32_t
|
2021-01-28 02:29:12 +03:00
|
|
|
compute_btr(uint32_t pclock, uint32_t bitrate)
|
|
|
|
{
|
2020-06-25 01:59:38 +03:00
|
|
|
/*
|
|
|
|
Some equations:
|
|
|
|
Tpclock = 1 / pclock
|
|
|
|
Tq = brp * Tpclock
|
|
|
|
Tbs1 = Tq * TS1
|
|
|
|
Tbs2 = Tq * TS2
|
|
|
|
NominalBitTime = Tq + Tbs1 + Tbs2
|
|
|
|
BaudRate = 1/NominalBitTime
|
|
|
|
|
|
|
|
Bit value sample point is after Tq+Tbs1. Ideal sample point
|
|
|
|
is at 87.5% of NominalBitTime
|
|
|
|
|
|
|
|
Use the lowest brp where ts1 and ts2 are in valid range
|
|
|
|
*/
|
|
|
|
|
|
|
|
uint32_t bit_clocks = pclock / bitrate; // clock ticks per bit
|
|
|
|
|
|
|
|
uint32_t sjw = 2;
|
|
|
|
uint32_t qs;
|
|
|
|
// Find number of time quantas that gives us the exact wanted bit time
|
2021-01-28 02:29:12 +03:00
|
|
|
for (qs = 18; qs > 9; qs--) {
|
2020-06-25 01:59:38 +03:00
|
|
|
// check that bit_clocks / quantas is an integer
|
|
|
|
uint32_t brp_rem = bit_clocks % qs;
|
2021-01-28 02:29:12 +03:00
|
|
|
if (brp_rem == 0)
|
2020-06-25 01:59:38 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
uint32_t brp = bit_clocks / qs;
|
|
|
|
uint32_t time_seg2 = qs / 8; // sample at ~87.5%
|
|
|
|
uint32_t time_seg1 = qs - (1 + time_seg2);
|
|
|
|
|
|
|
|
return make_btr(sjw, time_seg1, time_seg2, brp);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
can_init(void)
|
|
|
|
{
|
|
|
|
enable_pclock((uint32_t)SOC_CAN);
|
|
|
|
|
|
|
|
gpio_peripheral(GPIO_Rx, CAN_FUNCTION, 1);
|
|
|
|
gpio_peripheral(GPIO_Tx, CAN_FUNCTION, 0);
|
|
|
|
|
|
|
|
uint32_t pclock = get_pclock_frequency((uint32_t)SOC_CAN);
|
|
|
|
|
|
|
|
uint32_t btr = compute_btr(pclock, CONFIG_SERIAL_BAUD);
|
|
|
|
|
|
|
|
/*##-1- Configure the CAN #######################################*/
|
|
|
|
|
|
|
|
/* Request initialisation */
|
2021-01-28 05:15:33 +03:00
|
|
|
SOC_CAN->MCR = CAN_MCR_INRQ;
|
2020-06-25 01:59:38 +03:00
|
|
|
/* Wait the acknowledge */
|
2021-01-28 02:29:12 +03:00
|
|
|
while (!(SOC_CAN->MSR & CAN_MSR_INAK))
|
|
|
|
;
|
2020-06-25 01:59:38 +03:00
|
|
|
|
|
|
|
SOC_CAN->BTR = btr;
|
|
|
|
|
2021-01-28 05:15:33 +03:00
|
|
|
// TXFP makes packets posted to the TX mboxes transmit in chronologcal order
|
|
|
|
// ABOM makes the hardware automatically leave bus-off state
|
|
|
|
SOC_CAN->MCR = CAN_MCR_TXFP | CAN_MCR_ABOM;
|
2020-06-25 01:59:38 +03:00
|
|
|
/* Wait the acknowledge */
|
2021-01-28 02:29:12 +03:00
|
|
|
while (SOC_CAN->MSR & CAN_MSR_INAK)
|
|
|
|
;
|
2020-06-25 01:59:38 +03:00
|
|
|
|
|
|
|
/*##-2- Configure the CAN Filter #######################################*/
|
2021-01-28 05:02:32 +03:00
|
|
|
can_set_filter(PKT_ID_UUID, PKT_ID_SET);
|
2020-06-25 01:59:38 +03:00
|
|
|
|
|
|
|
/*##-3- Configure Interrupts #################################*/
|
|
|
|
|
2021-01-28 08:54:45 +03:00
|
|
|
SOC_CAN->IER = CAN_IER_FMPIE0; // RX mailbox IRQ
|
2020-06-25 01:59:38 +03:00
|
|
|
|
|
|
|
armcm_enable_irq(CAN_IRQHandler, CAN_RX0_IRQn, 0);
|
2021-01-28 02:29:12 +03:00
|
|
|
if (CAN_RX0_IRQn != CAN_RX1_IRQn)
|
|
|
|
armcm_enable_irq(CAN_IRQHandler, CAN_RX1_IRQn, 0);
|
|
|
|
if (CAN_RX0_IRQn != CAN_TX_IRQn)
|
|
|
|
armcm_enable_irq(CAN_IRQHandler, CAN_TX_IRQn, 0);
|
2020-06-25 01:59:38 +03:00
|
|
|
|
|
|
|
/*##-4- Say Hello #################################*/
|
2021-01-28 08:54:45 +03:00
|
|
|
can_process_uuid(0, 0, NULL);
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|
|
|
|
DECL_INIT(can_init);
|
|
|
|
|
|
|
|
void
|
|
|
|
serial_enable_tx_irq(void)
|
|
|
|
{
|
2021-01-28 08:54:45 +03:00
|
|
|
sched_wake_task(&canbus_wake);
|
2020-06-25 01:59:38 +03:00
|
|
|
}
|