Skip to content

Latest commit

 

History

History
373 lines (286 loc) · 14.8 KB

API.md

File metadata and controls

373 lines (286 loc) · 14.8 KB

This document describes how one may use can2040 in an application.

Note that can2040 uses the GNU GPLv3 license. See the COPYING file for more information.

Compiling

The can2040 code is intended to be compiled using the "gcc" C compiler. The can2040 implementation is contained in the can2040.c and can2040.h C files. The code is intended to be compiled at -O2 (or higher) optimization.

The code depends on a few files from the pico sdk (version 1.3.0 or later) that must be in the include path when compiling can2040. For example: arm-none-eabi-gcc -O2 -I/path/to/sdk/src/rp2040/ -I/path/to/sdk/src/rp2_common/cmsis/stub/CMSIS/Device/RaspberryPi/RP2040/Include/ ...

If compiling for the rp2350 then pico_sdk version 2.0.0 or later is required and the compiler flags must include -DPICO_RP2350 (the pico-sdk build rules typically provide this definition).

Startup

The following provides example startup C code for can2040:

static struct can2040 cbus;

static void
can2040_cb(struct can2040 *cd, uint32_t notify, struct can2040_msg *msg)
{
    // Add message processing code here...
}

static void
PIOx_IRQHandler(void)
{
    can2040_pio_irq_handler(&cbus);
}

void
canbus_setup(void)
{
    uint32_t pio_num = 0;
    uint32_t sys_clock = 125000000, bitrate = 500000;
    uint32_t gpio_rx = 4, gpio_tx = 5;

    // Setup canbus
    can2040_setup(&cbus, pio_num);
    can2040_callback_config(&cbus, can2040_cb);

    // Enable irqs
    irq_set_exclusive_handler(PIO0_IRQ_0_IRQn, PIOx_IRQHandler);
    NVIC_SetPriority(PIO0_IRQ_0_IRQn, 1);
    NVIC_EnableIRQ(PIO0_IRQ_0_IRQn);

    // Start canbus
    can2040_start(&cbus, sys_clock, bitrate, gpio_rx, gpio_tx);
}

The can2040.h header file provides the definition of struct can2040. This definition is exported so that one may statically allocate an instance of it. (It is also valid to dynamically allocate it if desired.) The content of the struct can2040 is considered "private" - callers should not inspect or modify its content for any reason. All interaction with can2040 is done via can2040 API functions.

The can2040 code does not dynamically allocate memory; all storage is contained in the struct can2040 that the caller allocates.

can2040_msg

The can2040.h header file provides a definition for struct can2040_msg. This struct is used for transmitting and receiving CAN bus messages.

The id field contains the CAN bus message id. It may also have the CAN2040_ID_RTR bit and/or CAN2040_ID_EFF bits set. The CAN2040_ID_RTR bit indicates that a "remote" frame should be used. The CAN2040_ID_EFF bit indicates that a 29-bit extended header should be used.

The dlc field specifies the number of data bytes the CAN bus message contains. In accordance with the CAN bus specification, the value may be between 0 and 15, however there are at most 8 data bytes. (A dlc value between 8 to 15 will have 8 data bytes.)

The data field contains the data bytes of the CAN bus message.

Function reference

can2040_setup

void can2040_setup(struct can2040 *cd, uint32_t pio_num)

This function initializes the struct can2040 passed in the cd parameter. The caller must either statically or dynamically allocate it prior to calling this function.

The pio_num should be either 0 or 1 to use either the PIO0 or PIO1 hardware block. On the rp2350 the pio_num may be 2 for PIO2.

can2040_callback_config

void can2040_callback_config(struct can2040 *cd, can2040_rx_cb rx_cb)

This function specifies the main can2040 callback (as specified in the rx_cb parameter).

The can2040_rx_cb callback function will be invoked with each successfully received and transmitted message. It must be provided by the user code.

The callback uses the function prototype: void can2040_rx_cb(struct can2040 *cd, uint32_t notify, struct can2040_msg *msg)

The cd parameter of the callback contains the same pointer passed to can2040_callback_config().

The notify parameter of the callback contains one of CAN2040_NOTIFY_RX, CAN2040_NOTIFY_TX, CAN2040_NOTIFY_ERROR.

  • A CAN2040_NOTIFY_RX event indicates a message has been successfully read. The message contents will be in the msg parameter.
  • A CAN2040_NOTIFY_TX event indicates a message has been successfully transmitted on the CAN bus. The transmitted message contents will be in the msg parameter.
  • A CAN2040_NOTIFY_ERROR event indicates that the internal receive buffers have overflowed and that some number of CAN bus messages may have been lost. The msg parameter will point to an allocated but otherwise empty struct can2040_msg in this event.

The callback is invoked in IRQ context (it is called from can2040_pio_irq_handler()). To keep irq latency low, it is recommended that the callback copy the given message to memory to be processed during normal (non-irq) context.

The msg pointer is only valid during the duration of the callback. The callback code should copy any desired content from msg to its own storage during the callback.

The callback is invoked for all valid received messages on the CAN bus. The can2040 code does not implement receive message filtering. It is expected that the caller will implement any desired filtering in their callback function.

can2040_start

void can2040_start(struct can2040 *cd, uint32_t sys_clock, uint32_t bitrate, uint32_t gpio_rx, uint32_t gpio_tx)

This function starts the main can2040 CAN bus implementation. The provided GPIO pins will be configured and associated with the rp2040/rp2350 PIO hardware block.

The sys_clock parameter specifies the system clock rate (for example 125000000 for an rp2040/rp2350 ARM core running at 125Mhz).

The bitrate parameter specifies the CAN bus speed (for example 500000 for a 500Kbit/s CAN bus).

The gpio_rx parameter specifies the gpio number that is routed to the "CAN RX" pin of the CAN bus transceiver. On rp2040 it should be between 0 and 29 (for GPIO0 to GPIO29). On the rp2350 chips it may be between 0 and 31 (for GPIO0 to GPIO31), or alternatively between 16 and 47 (for GPIO16 to GPIO47) if both gpio_rx and gpio_tx are between 16 and 47.

The gpio_tx parameter specifies the gpio number that is routed to the "CAN TX" pin of the CAN bus transceiver. On rp2040 chips it should be between 0 and 29 (for GPIO0 to GPIO29). On the rp2350 chips it may be between 0 and 31 (for GPIO0 to GPIO31), or alternatively between 16 and 47 (for GPIO16 to GPIO47) if both gpio_rx and gpio_tx are between 16 and 47.

After calling this function, activity on the CAN bus may result in the user specified can2040_rx_cb callback being invoked.

can2040_pio_irq_handler

void can2040_pio_irq_handler(struct can2040 *cd)

This function should be invoked on each PIO hardware IRQ event. The caller should define the underlying IRQ handler function, enable it, and arrange to call can2040_pio_irq_handler() with each event. The can2040 code uses the first IRQ of each PIO block (PIO0_IRQ_0_IRQn for PIO0, or PIO1_IRQ_0_IRQn for PIO1).

This function is intended to be invoked from IRQ context. It is reentrant safe with respect to other can2040 function calls made on the same ARM core.

can2040_transmit

int can2040_transmit(struct can2040 *cd, struct can2040_msg *msg)

This function schedules a message for transmission on the CAN bus. The msg parameter should contain a pointer to the struct can2040_msg that should be sent. The function returns 0 if the message is successfully queued, or a negative number if there is no space for the message in the internal queue. If scheduled, the contents of the msg pointer are copied to internal storage during the can2040_transmit() call.

When a scheduled message is successfully transmitted on the CAN bus the user supplied can2040_rx_cb callback will be invoked with a CAN2040_NOTIFY_TX event.

The can2040 code may buffer up to four messages for transmission. If multiple messages are buffered then they are transmitted in "first in first out" order. (The buffered transmit messages are not reordered by message id priority.)

It is valid to invoke can2040_transmit() from the user supplied can2040_rx_cb callback function, however doing so is not recommended. The can2040_transmit() function has processing overhead (it performs CRC calculation and bitstuffing on the message) and calling it from IRQ context may increase irq latency.

It is valid to invoke can2040_transmit() on one ARM core while the other ARM core may be running can2040_pio_irq_handler().

can2040_check_transmit

int can2040_check_transmit(struct can2040 *cd)

This function may be called to determine if there is space available to schedule a message transmission. That is, it may be used to determine if a call to can2040_transmit() will succeed.

It is valid to invoke can2040_check_transmit() from the user supplied can2040_rx_cb callback function.

It is valid to invoke can2040_check_transmit() on one ARM core while the other ARM core may be running can2040_pio_irq_handler().

can2040_stop

void can2040_stop(struct can2040 *cd)

This function disables processing of can2040 CAN bus messages. Upon completion of this function the user supplied can2040_rx_cb() callback function will no longer be invoked.

If this function is called while a message is queued for transmit then it is possible that the message may be successfully transmitted without it being removed from the local transmit queue and without a CAN2040_NOTIFY_TX event sent to the user supplied can2040_rx_cb() callback function.

The can2040 CAN bus processing may be restarted by calling can2040_start().

To clear the transmit queue before restarting, call can2040_setup(), can2040_callback_config(), and then can2040_start().

can2040_get_statistics

void can2040_get_statistics(struct can2040 *cd, struct can2040_stats *stats)

This function may be called to obtain can2040 receive and transmit statistics. This may be useful for insight on how well the CAN bus network hardware is performing.

The caller must allocate a struct can2040_stats and pass a pointer to it in the stats parameter. The can2040_get_statistics() function will copy its internal can2040 statistics to the provided struct. The caller may then inspect its local copy of struct can2040_stats after the function completes.

The can2040.h header file provides the definition for struct can2040_stats. It has the following fields:

  • rx_total: The total number of successfully received messages. This is the number of times that can2040_rx_cb() is invoked with CAN2040_NOTIFY_RX.
  • tx_total: The total number of successfully transmitted messages. This is the number of times that can2040_rx_cb() is invoked with CAN2040_NOTIFY_TX.
  • tx_attempt: The total number of transmit attempts. If this is more than one greater than tx_total it indicates some transmits were retried. A transmit may be retried due to line arbitration (a transmit attempt was interrupted by a higher priority transmission from another node on the CAN bus), due to the lack of an acknowledgment from another node on the CAN bus, or due to some other parse error during a transmit attempt.
  • parse_error: The total number of data errors observed during content parsing on the CAN bus. This may increment due to hardware noise on the CAN bus, due to error frames generated from other nodes on the CAN bus, due to lack of transmit acknowledgments on the CAN bus, or due to some other error in read data.

The above counters are only set to zero during the initial call to can2040_setup(). One may call can2040_get_statistics() periodically and subtract each counter from the value found at the previous call to obtain the statistics over that discrete period. When subtracting, it is recommended to store the difference in a uint32_t for improved handling of 32bit counter rollovers.

It is valid to invoke can2040_get_statistics() at any time after can2040_setup() is called (including from another ARM core and including from the user supplied can2040_rx_cb callback function).

Not reentrant safe

Unless explicitly stated otherwise, the can2040 code is not reentrant safe. For example, unless stated otherwise, one may not invoke can2040 functions from IRQ context (where they might preempt a can2040 function running in another context), one may not invoke can2040 functions from multiple ARM cores (such that two can2040 functions could be running simultaneously), nor invoke can2040 functions from the user supplied can2040_rx_cb callback function.

Multiple can2040 instances

Each instance of can2040 uses one of the PIO hardware blocks on the rp2040/rp2350. If there are separate CAN transceivers (and separate sets of CAN rx and CAN tx gpio pins) then one may create multiple simultaneous instances of can2040.

To use this functionality, the startup code should be run multiple times, each with their own separate instance of a struct can2040.

In this case, the multiple instances of can2040 do not share state. Therefore, no particular synchronization is needed between instances. That is, one must ensure each instance is not reentrant with respect to itself, but it is not required to synchronize between instances. One may run multiple can2040 instances on the same ARM core or different ARM cores.

Low interrupt latency

The can2040 implementation requires low interrupt response time for proper operation. See the Features Document for more information on the impact of latency to can2040.

To ensure low-latency it is recommended to limit the amount of code that runs at a higher interrupt priority than can2040. Higher priority code would be any code that disables interrupts and any interrupt handling code that is registered with a higher priority (a lower or equal value passed to NVIC_SetPriority()).

Also consider loading the can2040 code (and any code that runs at a higher priority) into memory on the rp2040/rp2350 chip. An rp2040 running at 125Mhz will take a minimum of 320ns for each 32bit load from flash. Thus, even a handful of flash accesses (to load instructions or data) may cause a delay sufficient to impact can2040. Be sure to also load the can2040 callback function into ram. Consider also loading the ARM core interrupt vector table into ram (if it is not already in ram).

The specifics of loading code into ram is beyond the scope of this document. At a high-level, it typically involves locating the build "linker script" and adding can2040.o(.text*) and can2040.o(.rodata*) to the ram segment.

Using can2040 from C++

The can2040 code is intended to be compiled with gcc. If can2040.c is compiled with a C compiler and linked with a C++ application then be careful to always import the can2040.h header file into C++ code in "C" mode - for example:

extern "C" {
  #include "can2040.h"
}