rusefi/firmware/hw_layer/drivers/can/can_msg_tx.h

118 lines
2.7 KiB
C
Raw Normal View History

/**
* @file can_msg_tx.h
*
* CAN message transmission
*
* @date Mar 13, 2020
* @author Matthew Kennedy, (c) 2012-2020
*/
#pragma once
#include <cstdint>
#include <cstddef>
#include "os_access.h"
2021-12-08 09:01:28 -08:00
#include "can.h"
/**
* Represent a message to be transmitted over CAN.
*
* Usage:
* * Create an instance of CanTxMessage
* * Set any data you'd like to transmit either using the subscript operator to directly access bytes, or any of the helper functions.
* * Upon destruction, the message is transmitted.
*/
class CanTxMessage
{
public:
/**
* Create a new CAN message, with the specified extended ID.
*/
explicit CanTxMessage(uint32_t eid, uint8_t dlc = 8, bool isExtended = false);
/**
* Destruction of an instance of CanTxMessage will transmit the message over the wire.
*/
~CanTxMessage();
2021-02-05 20:57:28 -08:00
#if EFI_CAN_SUPPORT
/**
* Configures the device for all messages to transmit from.
*/
2021-12-20 21:48:13 -08:00
static void setDevice(CANDriver* device1, CANDriver* device2);
2021-02-05 20:57:28 -08:00
#endif // EFI_CAN_SUPPORT
/**
* @brief Read & write the raw underlying 8-byte buffer.
*/
uint8_t& operator[](size_t);
/**
* @brief Write a 16-bit short value to the buffer. Note: this writes in big endian byte order.
*/
void setShortValue(uint16_t value, size_t offset);
/**
* @brief Set a single bit in the transmit buffer. Useful for single-bit flags.
*/
void setBit(size_t byteIdx, size_t bitIdx);
void setDlc(uint8_t dlc);
#if HAL_USE_CAN || EFI_UNIT_TEST
2021-12-08 09:01:28 -08:00
const CANTxFrame *getFrame() const {
return &m_frame;
}
#endif // HAL_USE_CAN || EFI_UNIT_TEST
2021-12-08 09:01:28 -08:00
protected:
#if HAL_USE_CAN || EFI_UNIT_TEST
CANTxFrame m_frame;
#endif // HAL_USE_CAN || EFI_UNIT_TEST
private:
2021-02-05 20:57:28 -08:00
#if EFI_CAN_SUPPORT
2021-12-20 21:48:13 -08:00
static CANDriver* s_devices[2];
2021-02-05 20:57:28 -08:00
#endif // EFI_CAN_SUPPORT
};
/**
* A CAN message based on a type, removing the need for manually flipping bits/bytes.
*/
template <typename TData>
class CanTxTyped final : public CanTxMessage
{
2021-02-05 20:57:28 -08:00
#if EFI_CAN_SUPPORT
static_assert(sizeof(TData) <= sizeof(CANTxFrame::data8));
2021-02-05 20:57:28 -08:00
#endif // EFI_CAN_SUPPORT
public:
explicit CanTxTyped(uint32_t id, bool isExtended) : CanTxMessage(id, sizeof(TData), isExtended) { }
2021-02-05 20:57:28 -08:00
#if EFI_CAN_SUPPORT
/**
* Access members of the templated type.
*
* So you can do:
* CanTxTyped<MyType> d;
* d->memberOfMyType = 23;
*/
TData* operator->() {
return reinterpret_cast<TData*>(&m_frame.data8);
}
TData& get() {
return *reinterpret_cast<TData*>(&m_frame.data8);
}
2021-02-05 20:57:28 -08:00
#endif // EFI_CAN_SUPPORT
};
template <typename TData>
void transmitStruct(uint32_t id, bool isExtended)
{
CanTxTyped<TData> frame(id, isExtended);
2020-03-31 17:57:52 -07:00
// Destruction of an instance of CanTxMessage will transmit the message over the wire.
// see CanTxMessage::~CanTxMessage()
populateFrame(frame.get());
}