lib: Update to latest can2040 code

Disable tx state machine before resetting it
Check for rx message with same id as tx but different payload
Do not use canrx line to alter bit timing of cantx line
Improve bit time synchronization with slow transmitters
Implement tx synchronization to faster transmitters

Signed-off-by: Kevin O'Connor <kevin@koconnor.net>
This commit is contained in:
Kevin O'Connor 2023-05-03 10:45:41 -04:00
parent 1a24e7c5b6
commit 3bd0be40d5
1 changed files with 39 additions and 26 deletions

View File

@ -76,21 +76,21 @@ rp2040_gpio_peripheral(uint32_t gpio, int func, int pull_up)
#define can2040_offset_shared_rx_read 13u #define can2040_offset_shared_rx_read 13u
#define can2040_offset_shared_rx_end 15u #define can2040_offset_shared_rx_end 15u
#define can2040_offset_match_load_next 18u #define can2040_offset_match_load_next 18u
#define can2040_offset_tx_conflict 24u
#define can2040_offset_match_end 25u #define can2040_offset_match_end 25u
#define can2040_offset_tx_got_recessive 25u #define can2040_offset_tx_got_recessive 25u
#define can2040_offset_tx_start 26u #define can2040_offset_tx_write_pin 27u
#define can2040_offset_tx_conflict 31u
static const uint16_t can2040_program_instructions[] = { static const uint16_t can2040_program_instructions[] = {
0x0085, // 0: jmp y--, 5 0x0085, // 0: jmp y--, 5
0x0048, // 1: jmp x--, 8 0x0048, // 1: jmp x--, 8
0xe12a, // 2: set x, 10 [1] 0xe029, // 2: set x, 9
0x00cc, // 3: jmp pin, 12 0x00cc, // 3: jmp pin, 12
0xc000, // 4: irq nowait 0 0xc000, // 4: irq nowait 0
0x00c0, // 5: jmp pin, 0 0x00c0, // 5: jmp pin, 0
0xc040, // 6: irq clear 0 0xc040, // 6: irq clear 0
0xe229, // 7: set x, 9 [2] 0xe429, // 7: set x, 9 [4]
0xf242, // 8: set y, 2 [18] 0xf043, // 8: set y, 3 [16]
0xc104, // 9: irq nowait 4 [1] 0xc104, // 9: irq nowait 4 [1]
0x03c5, // 10: jmp pin, 5 [3] 0x03c5, // 10: jmp pin, 5 [3]
0x0307, // 11: jmp 7 [3] 0x0307, // 11: jmp 7 [3]
@ -98,7 +98,7 @@ static const uint16_t can2040_program_instructions[] = {
0x20c4, // 13: wait 1 irq, 4 0x20c4, // 13: wait 1 irq, 4
0x4001, // 14: in pins, 1 0x4001, // 14: in pins, 1
0xa046, // 15: mov y, isr 0xa046, // 15: mov y, isr
0x00b2, // 16: jmp x != y, 18 0x01b2, // 16: jmp x != y, 18 [1]
0xc002, // 17: irq nowait 2 0xc002, // 17: irq nowait 2
0x40eb, // 18: in osr, 11 0x40eb, // 18: in osr, 11
0x4054, // 19: in y, 20 0x4054, // 19: in y, 20
@ -107,13 +107,13 @@ static const uint16_t can2040_program_instructions[] = {
0xa027, // 22: mov x, osr 0xa027, // 22: mov x, osr
0x0098, // 23: jmp y--, 24 0x0098, // 23: jmp y--, 24
0xa0e2, // 24: mov osr, y 0xa0e2, // 24: mov osr, y
0xa242, // 25: nop [2] 0x6021, // 25: out x, 1
0x6021, // 26: out x, 1 0x00df, // 26: jmp pin, 31
0xa001, // 27: mov pins, x 0xb801, // 27: mov pins, x [24]
0x20c4, // 28: wait 1 irq, 4 0x02d9, // 28: jmp pin, 25 [2]
0x00d9, // 29: jmp pin, 25 0x0058, // 29: jmp x--, 24
0x023a, // 30: jmp !x, 26 [2] 0x6021, // 30: out x, 1
0x001f, // 31: jmp 31 0x011b, // 31: jmp 27 [1]
}; };
// Local names for PIO state machine IRQs // Local names for PIO state machine IRQs
@ -138,7 +138,7 @@ pio_sync_setup(struct can2040 *cd)
| cd->gpio_rx << PIO_SM0_PINCTRL_SET_BASE_LSB); | cd->gpio_rx << PIO_SM0_PINCTRL_SET_BASE_LSB);
sm->instr = 0xe080; // set pindirs, 0 sm->instr = 0xe080; // set pindirs, 0
sm->pinctrl = 0; sm->pinctrl = 0;
pio_hw->txf[0] = PIO_CLOCK_PER_BIT / 2 * 7 - 5 - 1; pio_hw->txf[0] = 9 + 6 * PIO_CLOCK_PER_BIT / 2;
sm->instr = 0x80a0; // pull block sm->instr = 0x80a0; // pull block
sm->instr = can2040_offset_sync_entry; // jmp sync_entry sm->instr = can2040_offset_sync_entry; // jmp sync_entry
} }
@ -183,7 +183,10 @@ pio_tx_setup(struct can2040 *cd)
{ {
pio_hw_t *pio_hw = cd->pio_hw; pio_hw_t *pio_hw = cd->pio_hw;
struct pio_sm_hw *sm = &pio_hw->sm[3]; struct pio_sm_hw *sm = &pio_hw->sm[3];
sm->execctrl = cd->gpio_rx << PIO_SM0_EXECCTRL_JMP_PIN_LSB; sm->execctrl = (
cd->gpio_rx << PIO_SM0_EXECCTRL_JMP_PIN_LSB
| can2040_offset_tx_conflict << PIO_SM0_EXECCTRL_WRAP_TOP_LSB
| can2040_offset_tx_conflict << PIO_SM0_EXECCTRL_WRAP_BOTTOM_LSB);
sm->shiftctrl = (PIO_SM0_SHIFTCTRL_FJOIN_TX_BITS sm->shiftctrl = (PIO_SM0_SHIFTCTRL_FJOIN_TX_BITS
| PIO_SM0_SHIFTCTRL_AUTOPULL_BITS); | PIO_SM0_SHIFTCTRL_AUTOPULL_BITS);
sm->pinctrl = (1 << PIO_SM0_PINCTRL_SET_COUNT_LSB sm->pinctrl = (1 << PIO_SM0_PINCTRL_SET_COUNT_LSB
@ -247,6 +250,7 @@ static void
pio_tx_reset(struct can2040 *cd) pio_tx_reset(struct can2040 *cd)
{ {
pio_hw_t *pio_hw = cd->pio_hw; pio_hw_t *pio_hw = cd->pio_hw;
pio_hw->ctrl = 0x07 << PIO_CTRL_SM_ENABLE_LSB;
pio_hw->ctrl = ((0x07 << PIO_CTRL_SM_ENABLE_LSB) pio_hw->ctrl = ((0x07 << PIO_CTRL_SM_ENABLE_LSB)
| (0x08 << PIO_CTRL_SM_RESTART_LSB)); | (0x08 << PIO_CTRL_SM_RESTART_LSB));
pio_hw->irq = (SI_MATCHED | SI_ACKDONE) >> 8; // clear PIO irq flags pio_hw->irq = (SI_MATCHED | SI_ACKDONE) >> 8; // clear PIO irq flags
@ -255,9 +259,6 @@ pio_tx_reset(struct can2040 *cd)
sm->shiftctrl = 0; sm->shiftctrl = 0;
sm->shiftctrl = (PIO_SM0_SHIFTCTRL_FJOIN_TX_BITS sm->shiftctrl = (PIO_SM0_SHIFTCTRL_FJOIN_TX_BITS
| PIO_SM0_SHIFTCTRL_AUTOPULL_BITS); | PIO_SM0_SHIFTCTRL_AUTOPULL_BITS);
// Must reset again after clearing fifo
pio_hw->ctrl = ((0x07 << PIO_CTRL_SM_ENABLE_LSB)
| (0x08 << PIO_CTRL_SM_RESTART_LSB));
} }
// Queue a message for transmission on PIO "tx" state machine // Queue a message for transmission on PIO "tx" state machine
@ -266,13 +267,14 @@ pio_tx_send(struct can2040 *cd, uint32_t *data, uint32_t count)
{ {
pio_hw_t *pio_hw = cd->pio_hw; pio_hw_t *pio_hw = cd->pio_hw;
pio_tx_reset(cd); pio_tx_reset(cd);
pio_hw->instr_mem[can2040_offset_tx_got_recessive] = 0xa242; // nop [2] pio_hw->instr_mem[can2040_offset_tx_got_recessive] = 0x6021; // out x, 1
uint32_t i; uint32_t i;
for (i=0; i<count; i++) for (i=0; i<count; i++)
pio_hw->txf[3] = data[i]; pio_hw->txf[3] = data[i];
struct pio_sm_hw *sm = &pio_hw->sm[3]; struct pio_sm_hw *sm = &pio_hw->sm[3];
sm->instr = 0xe001; // set pins, 1 sm->instr = 0xe001; // set pins, 1
sm->instr = can2040_offset_tx_start; // jmp tx_start sm->instr = 0x6021; // out x, 1
sm->instr = can2040_offset_tx_write_pin; // jmp tx_write_pin
sm->instr = 0x20c0; // wait 1 irq, 0 sm->instr = 0x20c0; // wait 1 irq, 0
pio_hw->ctrl = 0x0f << PIO_CTRL_SM_ENABLE_LSB; pio_hw->ctrl = 0x0f << PIO_CTRL_SM_ENABLE_LSB;
} }
@ -287,7 +289,8 @@ pio_tx_inject_ack(struct can2040 *cd, uint32_t match_key)
pio_hw->txf[3] = 0x7fffffff; pio_hw->txf[3] = 0x7fffffff;
struct pio_sm_hw *sm = &pio_hw->sm[3]; struct pio_sm_hw *sm = &pio_hw->sm[3];
sm->instr = 0xe001; // set pins, 1 sm->instr = 0xe001; // set pins, 1
sm->instr = can2040_offset_tx_start; // jmp tx_start sm->instr = 0x6021; // out x, 1
sm->instr = can2040_offset_tx_write_pin; // jmp tx_write_pin
sm->instr = 0x20c2; // wait 1 irq, 2 sm->instr = 0x20c2; // wait 1 irq, 2
pio_hw->ctrl = 0x0f << PIO_CTRL_SM_ENABLE_LSB; pio_hw->ctrl = 0x0f << PIO_CTRL_SM_ENABLE_LSB;
@ -680,8 +683,11 @@ tx_check_local_message(struct can2040 *cd)
return 0; return 0;
struct can2040_transmit *qt = &cd->tx_queue[tx_qpos(cd, cd->tx_pull_pos)]; struct can2040_transmit *qt = &cd->tx_queue[tx_qpos(cd, cd->tx_pull_pos)];
struct can2040_msg *pm = &cd->parse_msg, *tm = &qt->msg; struct can2040_msg *pm = &cd->parse_msg, *tm = &qt->msg;
if (qt->crc == cd->parse_crc && tm->id == pm->id && tm->dlc == pm->dlc if (tm->id == pm->id) {
&& tm->data32[0] == pm->data32[0] && tm->data32[1] == pm->data32[1]) { if (qt->crc != cd->parse_crc || tm->dlc != pm->dlc
|| tm->data32[0] != pm->data32[0] || tm->data32[1] != pm->data32[1])
// Message with same id that differs in content - an error
return -1;
// This is a self transmit // This is a self transmit
cd->tx_state = TS_CONFIRM_TX; cd->tx_state = TS_CONFIRM_TX;
return 1; return 1;
@ -757,22 +763,25 @@ report_note_message_start(struct can2040 *cd)
} }
// Setup for ack injection (if receiving) or ack confirmation (if transmit) // Setup for ack injection (if receiving) or ack confirmation (if transmit)
static void static int
report_note_crc_start(struct can2040 *cd) report_note_crc_start(struct can2040 *cd)
{ {
int ret = tx_check_local_message(cd); int ret = tx_check_local_message(cd);
if (ret) { if (ret) {
if (ret < 0)
return -1;
// This is a self transmit - setup tx eof "matched" signal // This is a self transmit - setup tx eof "matched" signal
cd->report_state = RS_NEED_TX_ACK; cd->report_state = RS_NEED_TX_ACK;
uint32_t bits = (cd->parse_crc_bits << 9) | 0x0ff; uint32_t bits = (cd->parse_crc_bits << 9) | 0x0ff;
pio_match_check(cd, pio_match_calc_key(bits, cd->parse_crc_pos + 9)); pio_match_check(cd, pio_match_calc_key(bits, cd->parse_crc_pos + 9));
return; return 0;
} }
// Setup for ack inject (after rx fifos fully drained) // Setup for ack inject (after rx fifos fully drained)
cd->report_state = RS_NEED_RX_ACK; cd->report_state = RS_NEED_RX_ACK;
pio_signal_set_txpending(cd); pio_signal_set_txpending(cd);
pio_irq_set(cd, SI_MAYTX | SI_TXPENDING); pio_irq_set(cd, SI_MAYTX | SI_TXPENDING);
return 0;
} }
// Parser successfully found matching crc // Parser successfully found matching crc
@ -961,7 +970,11 @@ data_state_go_crc(struct can2040 *cd)
cd->parse_crc_bits = (crc_bits << 1) | 0x01; // Add crc delimiter cd->parse_crc_bits = (crc_bits << 1) | 0x01; // Add crc delimiter
cd->parse_crc_pos = crcstart_bitpos + crc_bitcount + 1; cd->parse_crc_pos = crcstart_bitpos + crc_bitcount + 1;
report_note_crc_start(cd); int ret = report_note_crc_start(cd);
if (ret) {
data_state_go_discard(cd);
return;
}
data_state_go_next(cd, MS_CRC, 16); data_state_go_next(cd, MS_CRC, 16);
} }