CAN Protocols for Bootloader

Overview

For the Bootloader, the following operations will be carried through datagrams. For some operations like querying, update ID/name, and flash application code, the data will first be serialized through protobufs. To generate protobuf files, run make bootloader_protos.

 

Datagram Format

Please check over the format for a datagram (Bootloader (Flash over CAN) | Transport layer (Datagrams)). Below, the id and type of data will be specified. Status return functions will share the same datagram ID of 0. The number of node IDs addressed through the client is arbitrary, and the special value 0 means every controller board is addressed. Every response datagram will have a value of 0 for the list of node IDs parameter, as 0 is the ID of the client.

 

Querying

In the querying function, the client sends out a datagram containing the pattern-matching information: controller board IDs, name, current project name, project info, and/or git version information. The client message can target one, several, or zero projects from the information. An example of this would be id=5, project=bms_carrier, name=curiosity to match controller board 5 running BMS carrier named curiosity.

The client → boards datagram should have the following format:

uint8 id = 1 // Use 1 here as the datagram ID to avoid conflicts uint8 ids_addressed = 0 // The special value 0 means every controller board is addressed uint16 data_size // Size of the protobuf (in n bytes) uint8[n] data // The protobuf (located in the protos folder in the bootloader project: querying.proto)

The response datagram will have the following format:

uint8 id = 2 // Use 2 here as the datagram ID to avoid conflicts uint16 data_size // Size of the protobuf (in n bytes) uint8[n] data // The protobuf (located in the protos folder in the bootloader project: querying_response.proto)

 

Ping

The ping function sends out a list of node IDs or none to ping all boards. Each board will send back a message with its ID if it is in the bootloader and ready to receive commands.

The client → boards datagram should have the following format:

uint8 id = 3 // Use 3 here as the datagram ID to avoid conflicts uint8 ids_addressed // The number(m) of nodes on the network that should receive the datagram. The special value 0 means every controller board is addressed uint8[m] node_id // The different node IDs. Keep this blank if ids_addressed = 0

Note: This function has no data

The response datagram will have the following format:

 

Jump to application

Direct the matched boards to jump to the application code. The client sends out a list of IDs it wants to jump; each matched controller board computes the CRC of the application code, checks it, responds with a status code, and jumps to the application. (Actually doing this is super tricky: info here From Zero to main(): How to Write a Bootloader from Scratch .)

Depending on how the design works out, we might be able to skip the CRC.

The client → boards datagram should have the following format:

Note: This function has no data

The response datagram will have the following format:

 

Update ID / Update name

This function will be split into two different commands, update ID and update name. These functions update the name or ID of a single controller board. The client sends out an ID to update and the new ID or name. The controller board overwrites the config (being careful to write one page, check it, and then write the other), and responds back with a status code. The client should not allow this to be run when more than one controller board is matched, or when the ID/name matches the ID/name of another controller board on the network; however, it can’t detect if the ID/name is used in a controller board not on the network, so be careful. The protobuf used will change based on which function is used (name vs ID).

The client → boards datagram should have the following format:

Update ID:

Update name:

The response datagram will have the following format:

 

Flash application code

The client sends out IDs of the boards to flash to, metadata like project name, git version (+branch?) info, and application CRC and size, then the application code itself. The controller boards write the application code to flash (making sure not to keep it all in memory at once to avoid overflows) and compute their own CRC of the application code. If it matches, they overwrite the project name/git version info/application CRC/application size in the config and clear the project info (being careful to write one page, check it, and then write the other), then respond back with an “OK” status code message. If it doesn’t match, they mark their config as “no project” (again being careful) and respond back with an appropriate status code.

The first client → boards metadata datagram should have the following format:

The response datagram will have the following format:

The client script will wait for this response from every controller board (or a timeout), then begin sending the application code in 2048-byte chunks. A client → boards datagram containing 2048 bytes of application code looks like this:

If there are less than 2048 bytes of application code remaining, use a smaller data_size. After each such datagram is sent, the client script will wait for another response datagram from every controller board (or a timeout; this gives the controller board time to process the datagram and write it to flash):

Once it has received a response datagram from every controller board, it sends the next chunk of application code data in another datagram until all the application code has been transferred.