Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
CanPipe Library
Introduction
CanPipe is a library to make it simple to implement more complex CAN protocols, such as CANOpen. You can easily create message filters and attach callbacks associated with those filters. Callbacks for a given filter can be chained together to create complex responses to any given message.
Message Handling
The CanPipe
class stores incoming and outgoing messages in buffers so that the handling of complex and timing sensitive routines and be executed outside of the interrupt context. The user must call CanPipe::HandleMessages
on a regular basis to handle all incoming messages as well as actually write posted messages to the CAN bus. This call should not be made from inside of an interrupt routine.
Received Messages
When a message is received, the CanPipe
class automatically adds the message to a buffer. User callbacks will be executed the next time HandleMessages
is called.
Outgoing Messages
Users can use CanPipe::PostMessage
to add new messages to a buffer. They will be written to the bus the next time HandleMessages
is called.
Because messages are handled outside of the interrupt, it is safe to handle and post messages in response to others however you wish without threat of blocking additional messages.
Filters
To create a filter use the CanPipe::RegisterFilter
method. This method gets passes an id
, which is the 11- or 29-bit message id to filter for and a 32-bit mask
. The mask
is applied to both the incoming message ID and the filter id
, which are then compared for a match. If no handle
is specified, or is zero, then the next available filter handle will be used. If handle
is specified as a value valid for your device, then it will override any existing filter at that handle. Filters are triggered in the order in which they are registered (In the future, it is desirable to sort software filters so that filters with the lowest handle are always triggered first). The first filter that passes is the one that is used.
An optional FilterMode
can be passed during CanPipe
construction.
CanPipe::kFilterAuto
: Hardware filter will be attempted if possible, and software filter will be created (Default).CanPipe::kFilterHardwareOnly
: Hardware filter will be attempted, and do nothing if it fails.CanPipe::kFilterSoftwareOnly
: Software filter will be created.
If a hardware filter succeeds, it will prevent any unmatched messages from triggering an interrupt. A software copy of the filter is saved in order to sort messages when they are handled. Specifying kFilterHardwareOnly
will turn off software filter sorting, and all callback messages should be attached to handle 0
. If kFilterSoftwareOnly
is specified, then all messages will be let through and cause an interrupt. The software filter will then sort the message when handled.
NOTE: Hardware filters are currently available only to LPC11Cxx (handles 1 to 31), LPC15xx (handles 1 to 31), and mbed Renesas (handle 1 only) devices.
NOTE: The filter "handle" specified here is similar to "mailbox" or "message object" used in some device user manuals.
Callbacks
Users can attach a callback of the form int(CANMessage&)
to a filter in order to handle received messages. For each pending received message, during CanPipe::HandleMessages
call:
- A filter handle is obtained, either from hardware when message was received, or by executing software filters
- A list callbacks is obtained based on the message's filter handle.
- Each callback in the list is called in the order in which it is attached.
If a callback returns anything other than 0 (CanPipe::kOkay
), the chain of callbacks will be broken. The user can use this feature to end execution in the event of an error or simply claim that the message is fully handled (i.e. returned CanPipe::kDone
).
Callbacks can be attached to handle 0 in order to handle messages that failed to match a filter or in a case where no filters were created.
Known Issues
The mbed CAN api implementation for LPC15xx does not currently execute attached interrupts correctly. This library is being developed with LPC15xx hardware, so several lines of LPC15xx specific code have been added to the CanPipe::ReadIrq
method to get it to work for development hardware.
Example
https://developer.mbed.org/users/ptpaterson/code/CanPipe_Example/
Example demonstrates how to create a message filter and use the returned handle to associate callback functions with messages that pass that filter.
Diff: CanPipe.cpp
- Revision:
- 0:26f78923d093
- Child:
- 1:ebd382c0a2b8
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/CanPipe.cpp Thu Nov 03 20:48:12 2016 +0000 @@ -0,0 +1,147 @@ +#include "CanPipe.h" + +/* Public Methods ************************************************************/ +CanPipe::CanPipe(CAN *p_can, FilterMode filter_mode): + p_can_(p_can), + filter_mode_(filter_mode) +{ + filter_auto_result_ = kFilterHardwareOnly; + num_software_filters_ = 0; + p_can_->attach<CanPipe>(this, &CanPipe::ReadIrq); +} + +int CanPipe::RegisterFilter(unsigned int id, + unsigned int mask, + CANFormat format, + int handle) +{ + if (!p_can_->filter(id, mask, format, handle)) { + + } + + return 0; //TODO: IMPLEMENT +} + +int CanPipe::RegisterCallback(CanMessageCallback callback, int handle) { + /* get new node from the pool */ + CallbackNode *new_node = AllocateNode(callback); + if (!new_node) + return kErrorCbNodeMemory; + + /* get the appropriate list */ + CallbackList *list = FindListWithHandle(handle); + if (!list) { + /* no list for handle yet. Allocate it now */ + list = AllocateList(handle); + } + if (!list) + return kErrorCbListMemory; + + /* add new node to list */ + if (list->begin) { + /* append to the end */ + CallbackNode *node_iterator = list->begin; + while (node_iterator->next_node) { + node_iterator = node_iterator->next_node; + } + node_iterator->next_node = new_node; + } else { + /* this is the first node */ + list->begin = new_node; + } + return kOkay; +} + + +void CanPipe::PostMessage(CANMessage msg) { + write_buffer.push(msg); +} + +void CanPipe::HandleMessages(void) { + /* Perform callbacks for all received messages */ + FilteredMessage filtered_msg; + while (read_buffer.pop(filtered_msg)) { + CANMessage msg = filtered_msg.msg; + int handle = filtered_msg.handle; + + // TODO: implement software filters + + CallbackList *callback_list = FindListWithHandle(handle); + if (!callback_list) + return; + + CallbackNode *node_iterator = callback_list->begin; + while (node_iterator) { + int result = node_iterator->callback(p_can_, msg); + + /* if an error occurred or callback claims to have completed message + * handling, then leave the message chain. + */ + if (result) { + break; + } else { + node_iterator = node_iterator->next_node; + } + } + } + + /* Write all posted messages to the bus */ + CANMessage write_msg; + while (write_buffer.pop(write_msg)) { + p_can_->write(write_msg); + } +} + +/* Private Methods ************************************************************/ +void CanPipe::ReadIrq(void) { + CANMessage msg; + int handle = p_can_->read(msg, 0); + + // TODO: raise flag if buffer is full + + FilteredMessage filtered_msg = {msg, handle}; + read_buffer.push(filtered_msg); +} + +CanPipe::CallbackNode* CanPipe::AllocateNode(CanMessageCallback callback) { + CallbackNode* result = 0; + if (num_nodes_ < kMaxCallbacks) { + result = &callback_node_pool_[num_nodes_++]; + result->callback = callback; + } + return result; +} + +CanPipe::CallbackList* CanPipe::AllocateList(int handle) { + CallbackList* result = 0; + if (num_lists_ < kMaxHandles) { + result = &callback_list_map_[num_lists_++]; + result->handle = handle; + } + return result; +} + +CanPipe::CallbackList* CanPipe::FindListWithHandle(int handle) { + CallbackList* result = 0; + /* iterate over all of the lists */ + for (int i = 0; i < num_lists_; ++i) { + if (callback_list_map_[i].handle == handle) { + result = &callback_list_map_[i]; + break; + } + } + return result; +} + + + + + + + + + + + + +