VehicleIO Plugins Overview
The VehicleIO Plugin back-end enables writing fully customized drive-by-wire abstraction that will be represented as a VehicleIO interface for the rest of the software stack.
Implementing Custom VehicleIO Plugin
A VehicleIO plugin in DriveWorks requires implementing a set of pre-defined functions. The declarations of these functions are located in dw/control/vehicleio/plugins/VehicleIODriver.h
.
In this tutorial, we are going to implement a simple drive-by-wire system that is responsible for steering wheel angle. See VehicleIO Plugin Sample for the corresponding sample code.
Initialization
In essence, our plugin is a dynamically loaded shared library, and the core VehicleIO module will first call _dwVehicleIODriver_initialize()
function of this library, which should prepare all objects and state that is needed for the plugin to function:
{
checkError(result, status, "cannot create DBC-based CAN message interpreter");
return result;
}
dwStatus
Status definition.
DW_API_PUBLIC dwStatus dwInitialize(dwContextHandle_t *const context, dwVersion const headerVersion, dwContextParameters const *const params)
Creates and initializes an SDK context.
A set of parameters that is passed to the SDK to create the context.
DW_API_PUBLIC dwStatus dwCANInterpreter_buildFromDBC(dwCANInterpreterHandle_t *interpreter, const char8_t *inputDBC, dwContextHandle_t ctx)
Creates a CAN data interpreter based on DBC file format.
dwStatus _dwVehicleIODriver_initialize()
Initializes the VehicleIO Driver.
Above, dwCANInterpreter_buildFromDBC()
function initializes an interpreter object that will be useful to work with our DBC file. See CAN Message Interpreter Sample for more information about CAN interpreter usage.
When using the plugin with, for example, VehicleIO Sample, in order for VehicleIO core to load the plugin, Rig file should specify "custom" VehicleIO node:
{
"rig": {
"vehicleio": [
{
"type": "custom",
"parent-sensor": "can:vehicle:custom",
"custom-lib": "../../../bin/libsample_vehicleio_plugin.so"
}
],
"sensors": [
{
"name": "can:vehicle:custom",
"parameter": "file=can-vio-generic.bin",
"protocol": "can.virtual",
"properties": null,
"nominalSensor2Rig": { "quaternion": [ 0.0, 0.0, 0.0, 1.0 ], "t": [ 0.0, 0.0, 0.0 ] },
"sensor2Rig": { "quaternion": [ 0.0, 0.0, 0.0, 1.0 ], "t": [ 0.0, 0.0, 0.0 ] }
}
]
}
}
Full example of Rig file is available in data/samples/vehicleio/rig-plugin.json
, and exact commands how to run the VehicleIO sample with the plugin is available at VehicleIO Plugin Sample.
Releasing
When plugin is no longer needed, VehicleIO core will call _dwVehicleIODriver_release()
method, which is an opportunity for the module to cleanup any created objects and state:
{
}
DW_API_PUBLIC dwStatus dwRelease(dwContextHandle_t const context)
Releases the context.
DW_API_PUBLIC dwStatus dwCANInterpreter_release(dwCANInterpreterHandle_t interpreter)
Closes previously opened interpreter.
dwStatus _dwVehicleIODriver_release()
Releases the VehicleIO Driver.
CAN messages consumption
CAN messages represent the input to the module, which plugin must parse and update its state, if needed. VehicleIO core will call plugin's _dwVehicleIODriver_consume()
function when there is a new message to consume:
{
checkError(result ,status, "consume message failed");
parseSteeringReport(state);
return result;
}
The vehicle IO state data. Fields only set if supported by VehicleIO driver.
DW_API_PUBLIC dwStatus dwCANInterpreter_consume(const dwCANMessage *msg, dwCANInterpreterHandle_t interpreter)
Pushes a new message to the interpreter.
dwStatus _dwVehicleIODriver_consume(const dwCANMessage *msg, dwVehicleIOState *state)
Consume a received CAN message and update the vehicle state.
In the example above, we simply parse current steering angle report using our CAN/DBC interpreter:
{
uint32_t num;
{
const char* name;
for (uint32_t i = 0; i < num; ++i)
{
{
{
if (0 == strcmp(name, GENERIC_ID_STEERING_WHEEL_ANGLE_REPORT))
else if (0 == strcmp(name, GENERIC_ID_STEERING_WHEEL_TORQUE_REPORT))
else if (0 == strcmp(name, GENERIC_ID_STEERING_WHEEL_SPEED_REPORT))
}
}
}
}
}
float32_t steeringWheelTorque
Steering wheel torque (0 to 10.0 +- 0.01 Nm)
float32_t speed
Signed norm of velocity vector.
float32_t steeringWheelAngle
Steering wheel angle (-10.0 to 10.0 +- 0.01rad)
float float32_t
Specifies POD types.
int64_t dwTime_t
Specifies a timestamp unit, in microseconds.
DW_API_PUBLIC dwStatus dwCANInterpreter_getSignalName(const char8_t **name, uint32_t idx, dwCANInterpreterHandle_t interpreter)
Gets the name of a signal available for consumption by the application as a string.
DW_API_PUBLIC dwStatus dwCANInterpreter_getNumberSignals(uint32_t *num, dwCANInterpreterHandle_t interpreter)
Gets the number of signals decoded and available for consumption by the application side.
DW_API_PUBLIC dwStatus dwCANInterpreter_getf32(float32_t *value, dwTime_t *timestamp_us, uint32_t idx, dwCANInterpreterHandle_t interpreter)
Gets a 'float32_t' value from the available values.
CAN messages generation
CAN messages are also the output of the module, thus when VehicleIO is requested to send a command, the core module will relay the request to the plugin, issuing two calls: _dwVehicleIODriver_sendCommand()
and _dwVehicleIODriver_sendMiscCommand()
. Each of the function should encode command arguments into CAN messages and send via a provided sensor handle:
{
checkError(result, status, "create steering message failed");
status = encodeSteering(msgCAN, cmd);
checkError(result, status, "encode steering failed");
checkError(result, status, "send steering failed");
return result;
}
{
}
@ DW_NOT_IMPLEMENTED
Requested feature/method is not yet implemented.
DW_API_PUBLIC dwStatus dwSensorCAN_sendMessage(const dwCANMessage *const msg, dwTime_t const timeoutUs, dwSensorHandle_t const sensor)
Sends a message over the CAN bus within a specified timeout.
DW_API_PUBLIC dwStatus dwCANInterpreter_createMessageFromName(dwCANMessage *msg, const char8_t *msgName, dwCANInterpreterHandle_t interpreter)
Initializes an empty can message an interpreter can encoded signals into.
struct dwSensorObject * dwSensorHandle_t
Handle representing a sensor.
dwStatus _dwVehicleIODriver_sendCommand(const dwVehicleIOCommand *cmd, dwSensorHandle_t sensor)
Send a vehicle command to the given CAN sensor.
dwStatus _dwVehicleIODriver_sendMiscCommand(const dwVehicleIOMiscCommand *cmd, dwSensorHandle_t sensor)
Send misc vehicle command to the given CAN sensor.
If certain type of commands are not relevant, the plugin is permitted to return DW_NOT_IMPLEMENTED
status, as was shown in _dwVehicleIODriver_sendMiscCommand()
above.
And as in previous section, we use CAN interpreter to encode commands into CAN messages:
{
GENERIC_ID_STEERING_WHEEL_ANGLE_COMMAND,
&msgCAN, gIntp);
checkError(result, status, "encode steering angle failed");
GENERIC_ID_STEERING_WHEEL_ANGLE_COMMAND_VALID,
&msgCAN, gIntp);
checkError(result, status, "encode steering valid failed");
GENERIC_ID_STEERING_WHEEL_STEER_SPEED,
&msgCAN, gIntp);
checkError(result, status, "encode steering speed failed");
GENERIC_ID_STEERING_WHEEL_STEER_CLEAR_FAULT,
&msgCAN, gIntp);
checkError(result, status, "encode steering clear fault failed");
return result;
}
float32_t steeringWheelAngle
Desired steering wheel angle (rad)
bool clearFaults
Setting > 0 clears any canbus faults/errors.
DW_API_PUBLIC dwStatus dwCANInterpreter_encodei32(int32_t value, const char8_t *signal, dwCANMessage *msg, dwCANInterpreterHandle_t interpreter)
Same as dwCANInterpreter_encodef32, but for int32 types.
DW_API_PUBLIC dwStatus dwCANInterpreter_encodef32(float32_t value, const char8_t *signal, dwCANMessage *msg, dwCANInterpreterHandle_t interpreter)
Encodes a value for a signal into a given message.
Building plugin as a dynamic library
The plugin code must be compiled into a dynamic library.
Firstly, we should create a folder under samples
called vehicleio_plugin
and create a CMakeLists.txt
under this folder:
project(sample_vehicleio_plugin C CXX)
set(SOURCES
driverConf.hpp
driver.cpp
)
set(LIBRARIES
samples_framework
${Driveworks_LIBRARIES}
)
add_library(${PROJECT_NAME} SHARED ${SOURCES})
target_link_libraries(${PROJECT_NAME} PRIVATE ${LIBRARIES})
set_property(TARGET ${PROJECT_NAME} PROPERTY FOLDER "Samples")
get_property(TARGET_PACKAGE GLOBAL PROPERTY TARGET_PACKAGE)
install(TARGETS ${PROJECT_NAME}
COMPONENT dw-samples-${TARGET_PACKAGE} # targeted package (dw / dav / ..)
DESTINATION ${SDK_SAMPLE_DESTINATION}
)
sdk_add_sample(${PROJECT_NAME})
Run cmake
and the preferred build system to compile the library. This will generate a libsample_vehicleio_plugin.so
.
See VehicleIO Plugin Sample for more details.