mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-03 06:28:27 -04:00
AP_PiccoloCAN: Framework for CAN servo outputs
This commit is contained in:
parent
43b7b4eb3b
commit
7b0da02a18
@ -69,6 +69,21 @@ const AP_Param::GroupInfo AP_PiccoloCAN::var_info[] = {
|
|||||||
// @Range: 1 500
|
// @Range: 1 500
|
||||||
AP_GROUPINFO("ESC_RT", 2, AP_PiccoloCAN, _esc_hz, PICCOLO_MSG_RATE_HZ_DEFAULT),
|
AP_GROUPINFO("ESC_RT", 2, AP_PiccoloCAN, _esc_hz, PICCOLO_MSG_RATE_HZ_DEFAULT),
|
||||||
|
|
||||||
|
// @Param: SRV_BM
|
||||||
|
// @DisplayName: Servo channels
|
||||||
|
// @Description: Bitmask defining which servo channels are to be transmitted over Piccolo CAN
|
||||||
|
// @Bitmask: 0: Servo 1, 1: Servo 2, 2: Servo 3, 3: Servo 4, 4: Servo 5, 5: Servo 6, 6: Servo 7, 7: Servo 8, 8: Servo 9, 9: Servo 10, 10: Servo 11, 11: Servo 12, 12: Servo 13, 13: Servo 14, 14: Servo 15, 15: Servo 16
|
||||||
|
// @User: Advanced
|
||||||
|
AP_GROUPINFO("SRV_BM", 3, AP_PiccoloCAN, _srv_bm, 0xFFFF),
|
||||||
|
|
||||||
|
// @Param: SRV_RT
|
||||||
|
// @DisplayName: Servo command output rate
|
||||||
|
// @Description: Output rate of servo command messages
|
||||||
|
// @Units: Hz
|
||||||
|
// @User: Advanced
|
||||||
|
// @Range: 1 500
|
||||||
|
AP_GROUPINFO("SRV_RT", 4, AP_PiccoloCAN, _srv_hz, PICCOLO_MSG_RATE_HZ_DEFAULT),
|
||||||
|
|
||||||
AP_GROUPEND
|
AP_GROUPEND
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -145,6 +160,7 @@ void AP_PiccoloCAN::loop()
|
|||||||
AP_HAL::CANFrame rxFrame;
|
AP_HAL::CANFrame rxFrame;
|
||||||
|
|
||||||
uint16_t esc_tx_counter = 0;
|
uint16_t esc_tx_counter = 0;
|
||||||
|
uint16_t servo_tx_counter = 0;
|
||||||
|
|
||||||
// CAN Frame ID components
|
// CAN Frame ID components
|
||||||
uint8_t frame_id_group; // Piccolo message group
|
uint8_t frame_id_group; // Piccolo message group
|
||||||
@ -152,16 +168,22 @@ void AP_PiccoloCAN::loop()
|
|||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
|
|
||||||
_esc_hz = constrain_int16(_esc_hz, PICCOLO_MSG_RATE_HZ_MIN, PICCOLO_MSG_RATE_HZ_MAX);
|
|
||||||
|
|
||||||
uint16_t escCmdRateMs = (uint16_t) ((float) 1000 / _esc_hz);
|
|
||||||
|
|
||||||
if (!_initialized) {
|
if (!_initialized) {
|
||||||
debug_can(AP_CANManager::LOG_ERROR, "PiccoloCAN: not initialized\n\r");
|
debug_can(AP_CANManager::LOG_ERROR, "PiccoloCAN: not initialized\n\r");
|
||||||
hal.scheduler->delay_microseconds(10000);
|
hal.scheduler->delay_microseconds(10000);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Calculate the output rate for ESC commands
|
||||||
|
_esc_hz = constrain_int16(_esc_hz, PICCOLO_MSG_RATE_HZ_MIN, PICCOLO_MSG_RATE_HZ_MAX);
|
||||||
|
|
||||||
|
uint16_t escCmdRateMs = (uint16_t) ((float) 1000 / _esc_hz);
|
||||||
|
|
||||||
|
// Calculate the output rate for servo commands
|
||||||
|
_srv_hz = constrain_int16(_srv_hz, PICCOLO_MSG_RATE_HZ_MIN, PICCOLO_MSG_RATE_HZ_MAX);
|
||||||
|
|
||||||
|
uint16_t servoCmdRateMs = (uint16_t) ((float) 1000 / _srv_hz);
|
||||||
|
|
||||||
uint64_t timeout = AP_HAL::micros64() + 250ULL;
|
uint64_t timeout = AP_HAL::micros64() + 250ULL;
|
||||||
|
|
||||||
// 1ms loop delay
|
// 1ms loop delay
|
||||||
@ -170,10 +192,15 @@ void AP_PiccoloCAN::loop()
|
|||||||
// Transmit ESC commands at regular intervals
|
// Transmit ESC commands at regular intervals
|
||||||
if (esc_tx_counter++ > escCmdRateMs) {
|
if (esc_tx_counter++ > escCmdRateMs) {
|
||||||
esc_tx_counter = 0;
|
esc_tx_counter = 0;
|
||||||
|
|
||||||
send_esc_messages();
|
send_esc_messages();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Transmit servo commands at regular intervals
|
||||||
|
if (servo_tx_counter++ > servoCmdRateMs) {
|
||||||
|
servo_tx_counter = 0;
|
||||||
|
send_servo_messages();
|
||||||
|
}
|
||||||
|
|
||||||
// Look for any message responses on the CAN bus
|
// Look for any message responses on the CAN bus
|
||||||
while (read_frame(rxFrame, timeout)) {
|
while (read_frame(rxFrame, timeout)) {
|
||||||
|
|
||||||
@ -191,6 +218,11 @@ void AP_PiccoloCAN::loop()
|
|||||||
case MessageGroup::ACTUATOR:
|
case MessageGroup::ACTUATOR:
|
||||||
|
|
||||||
switch (ActuatorType(frame_id_device)) {
|
switch (ActuatorType(frame_id_device)) {
|
||||||
|
case ActuatorType::SERVO:
|
||||||
|
if (handle_servo_message(rxFrame)) {
|
||||||
|
// Returns true if the message was successfully decoded
|
||||||
|
}
|
||||||
|
break;
|
||||||
case ActuatorType::ESC:
|
case ActuatorType::ESC:
|
||||||
if (handle_esc_message(rxFrame)) {
|
if (handle_esc_message(rxFrame)) {
|
||||||
// Returns true if the message was successfully decoded
|
// Returns true if the message was successfully decoded
|
||||||
@ -254,6 +286,16 @@ bool AP_PiccoloCAN::read_frame(AP_HAL::CANFrame &recv_frame, uint64_t timeout)
|
|||||||
// called from SRV_Channels
|
// called from SRV_Channels
|
||||||
void AP_PiccoloCAN::update()
|
void AP_PiccoloCAN::update()
|
||||||
{
|
{
|
||||||
|
uint64_t timestamp = AP_HAL::micros64();
|
||||||
|
|
||||||
|
/* Read out the servo commands from the channel mixer */
|
||||||
|
for (uint8_t ii = 0; ii < PICCOLO_CAN_MAX_NUM_SERVO; ii++) {
|
||||||
|
|
||||||
|
if (is_servo_channel_active(ii)) {
|
||||||
|
// TODO - Extract output command here
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Read out the ESC commands from the channel mixer */
|
/* Read out the ESC commands from the channel mixer */
|
||||||
for (uint8_t i = 0; i < PICCOLO_CAN_MAX_NUM_ESC; i++) {
|
for (uint8_t i = 0; i < PICCOLO_CAN_MAX_NUM_ESC; i++) {
|
||||||
|
|
||||||
@ -278,6 +320,8 @@ void AP_PiccoloCAN::update()
|
|||||||
|
|
||||||
WITH_SEMAPHORE(_telem_sem);
|
WITH_SEMAPHORE(_telem_sem);
|
||||||
|
|
||||||
|
// TODO - Add servo log data here
|
||||||
|
|
||||||
for (uint8_t i = 0; i < PICCOLO_CAN_MAX_NUM_ESC; i++) {
|
for (uint8_t i = 0; i < PICCOLO_CAN_MAX_NUM_ESC; i++) {
|
||||||
|
|
||||||
PiccoloESC_Info_t &esc = _esc_info[i];
|
PiccoloESC_Info_t &esc = _esc_info[i];
|
||||||
@ -370,6 +414,75 @@ void AP_PiccoloCAN::send_esc_telemetry_mavlink(uint8_t mav_chan)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// send servo messages over CAN
|
||||||
|
void AP_PiccoloCAN::send_servo_messages(void)
|
||||||
|
{
|
||||||
|
AP_HAL::CANFrame txFrame;
|
||||||
|
|
||||||
|
uint64_t timeout = AP_HAL::micros64() + 1000ULL;
|
||||||
|
|
||||||
|
// No servos are selected? Don't send anything!
|
||||||
|
if (_srv_bm == 0x00) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool send_cmd = false;
|
||||||
|
int16_t cmd[4] {};
|
||||||
|
uint8_t idx;
|
||||||
|
|
||||||
|
// Transmit bulk command packets to 4x servos simultaneously
|
||||||
|
for (uint8_t ii = 0; ii < PICCOLO_CAN_MAX_GROUP_SERVO; ii++) {
|
||||||
|
|
||||||
|
send_cmd = false;
|
||||||
|
|
||||||
|
for (uint8_t jj = 0; jj < 4; jj++) {
|
||||||
|
|
||||||
|
idx = (ii * 4) + jj;
|
||||||
|
|
||||||
|
// Set default command value if an output field is unused
|
||||||
|
cmd[jj] = 0x7FFF;
|
||||||
|
|
||||||
|
// Skip servo if the output is not enabled
|
||||||
|
if (!is_servo_channel_active(idx)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check if the servo is enabled.
|
||||||
|
* If it is not enabled, send an enable message.
|
||||||
|
*/
|
||||||
|
|
||||||
|
if (!is_servo_present(idx) || !is_servo_enabled(idx)) {
|
||||||
|
// Servo is not enabled
|
||||||
|
encodeServo_EnablePacket(&txFrame);
|
||||||
|
txFrame.id |= (idx + 1);
|
||||||
|
write_frame(txFrame, timeout);
|
||||||
|
} else if (_servo_info[idx].newCommand) {
|
||||||
|
// A new command is provided
|
||||||
|
send_cmd = true;
|
||||||
|
cmd[jj] = _servo_info[idx].command;
|
||||||
|
_servo_info[idx].newCommand = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (send_cmd) {
|
||||||
|
encodeServo_MultiPositionCommandPacket(
|
||||||
|
&txFrame,
|
||||||
|
cmd[0],
|
||||||
|
cmd[1],
|
||||||
|
cmd[2],
|
||||||
|
cmd[3],
|
||||||
|
(PKT_SERVO_MULTI_COMMAND_1 + ii)
|
||||||
|
);
|
||||||
|
|
||||||
|
// Broadcast the command to all servos
|
||||||
|
txFrame.id |= 0xFF;
|
||||||
|
|
||||||
|
write_frame(txFrame, timeout);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// send ESC messages over CAN
|
// send ESC messages over CAN
|
||||||
void AP_PiccoloCAN::send_esc_messages(void)
|
void AP_PiccoloCAN::send_esc_messages(void)
|
||||||
{
|
{
|
||||||
@ -398,6 +511,9 @@ void AP_PiccoloCAN::send_esc_messages(void)
|
|||||||
|
|
||||||
idx = (ii * 4) + jj;
|
idx = (ii * 4) + jj;
|
||||||
|
|
||||||
|
// Set default command value if an output field is unused
|
||||||
|
cmd[jj] = 0x7FFF;
|
||||||
|
|
||||||
// Skip an ESC if the motor channel is not enabled
|
// Skip an ESC if the motor channel is not enabled
|
||||||
if (!is_esc_channel_active(idx)) {
|
if (!is_esc_channel_active(idx)) {
|
||||||
continue;
|
continue;
|
||||||
@ -416,8 +532,8 @@ void AP_PiccoloCAN::send_esc_messages(void)
|
|||||||
cmd[jj] = _esc_info[idx].command;
|
cmd[jj] = _esc_info[idx].command;
|
||||||
_esc_info[idx].newCommand = false;
|
_esc_info[idx].newCommand = false;
|
||||||
} else {
|
} else {
|
||||||
// A command of 0xFFFF is 'out of range' and will be ignored by the corresponding ESC
|
// A command of 0x7FFF is 'out of range' and will be ignored by the corresponding ESC
|
||||||
cmd[jj] = 0xFFFF;
|
cmd[jj] = 0x7FFF;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -452,6 +568,57 @@ void AP_PiccoloCAN::send_esc_messages(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// interpret a servo message received over CAN
|
||||||
|
bool AP_PiccoloCAN::handle_servo_message(AP_HAL::CANFrame &frame)
|
||||||
|
{
|
||||||
|
uint64_t timestamp = AP_HAL::micros64();
|
||||||
|
|
||||||
|
// The servo address is the lower byte of the frame ID
|
||||||
|
uint8_t addr = frame.id & 0xFF;
|
||||||
|
|
||||||
|
// Ignore servo with an invalid node ID
|
||||||
|
if (addr == 0x00) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Subtract to get the address in memory
|
||||||
|
addr -= 1;
|
||||||
|
|
||||||
|
// Maximum number of servos allowed
|
||||||
|
if (addr >= PICCOLO_CAN_MAX_NUM_SERVO) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
CBSServo_Info_t &servo = _servo_info[addr];
|
||||||
|
|
||||||
|
bool result = true;
|
||||||
|
|
||||||
|
// Throw the incoming packet against each decoding routine
|
||||||
|
if (decodeServo_StatusAPacketStructure(&frame, &servo.statusA)) {
|
||||||
|
servo.newTelemetry = true;
|
||||||
|
} else if (decodeServo_StatusBPacketStructure(&frame, &servo.statusB)) {
|
||||||
|
servo.newTelemetry = true;
|
||||||
|
} else if (decodeServo_FirmwarePacketStructure(&frame, &servo.firmware)) {
|
||||||
|
// TODO
|
||||||
|
} else if (decodeServo_AddressPacketStructure(&frame, &servo.address)) {
|
||||||
|
// TODO
|
||||||
|
} else if (decodeServo_SettingsInfoPacketStructure(&frame, &servo.settings)) {
|
||||||
|
// TODO
|
||||||
|
} else if (decodeServo_TelemetryConfigPacketStructure(&frame, &servo.telemetry)) {
|
||||||
|
} else {
|
||||||
|
// Incoming frame did not match any of the packet decoding routines
|
||||||
|
result = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result) {
|
||||||
|
// Reset the rx timestamp
|
||||||
|
servo.last_rx_msg_timestamp = timestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// interpret an ESC message received over CAN
|
// interpret an ESC message received over CAN
|
||||||
bool AP_PiccoloCAN::handle_esc_message(AP_HAL::CANFrame &frame)
|
bool AP_PiccoloCAN::handle_esc_message(AP_HAL::CANFrame &frame)
|
||||||
{
|
{
|
||||||
@ -460,7 +627,7 @@ bool AP_PiccoloCAN::handle_esc_message(AP_HAL::CANFrame &frame)
|
|||||||
#if HAL_WITH_ESC_TELEM
|
#if HAL_WITH_ESC_TELEM
|
||||||
uint64_t timestamp = AP_HAL::micros64();
|
uint64_t timestamp = AP_HAL::micros64();
|
||||||
|
|
||||||
// The ESC address is the lower byte of the address
|
// The ESC address is the lower byte of the frame ID
|
||||||
uint8_t addr = frame.id & 0xFF;
|
uint8_t addr = frame.id & 0xFF;
|
||||||
|
|
||||||
// Ignore any ESC with node ID of zero
|
// Ignore any ESC with node ID of zero
|
||||||
@ -555,7 +722,27 @@ bool AP_PiccoloCAN::handle_esc_message(AP_HAL::CANFrame &frame)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if a given ESC channel is "active" (has been configured correctly)
|
* Check if a given servo channel is "active" (has been configured for Piccolo control output)
|
||||||
|
*/
|
||||||
|
bool AP_PiccoloCAN::is_servo_channel_active(uint8_t chan)
|
||||||
|
{
|
||||||
|
// First check if the particular servo channel is enabled in the channel mask
|
||||||
|
if (((_srv_bm >> chan) & 0x01) == 0x00) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if a servo function is assigned for this motor channel
|
||||||
|
// TODO - ?
|
||||||
|
if (1) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if a given ESC channel is "active" (has been configured for Piccolo control output)
|
||||||
*/
|
*/
|
||||||
bool AP_PiccoloCAN::is_esc_channel_active(uint8_t chan)
|
bool AP_PiccoloCAN::is_esc_channel_active(uint8_t chan)
|
||||||
{
|
{
|
||||||
@ -575,6 +762,34 @@ bool AP_PiccoloCAN::is_esc_channel_active(uint8_t chan)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine if a servo is present on the CAN bus (has telemetry data been received)
|
||||||
|
*/
|
||||||
|
bool AP_PiccoloCAN::is_servo_present(uint8_t chan, uint64_t timeout_ms)
|
||||||
|
{
|
||||||
|
if (chan >= PICCOLO_CAN_MAX_NUM_SERVO) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
CBSServo_Info_t &servo = _servo_info[chan];
|
||||||
|
|
||||||
|
// No messages received from this servo
|
||||||
|
if (servo.last_rx_msg_timestamp == 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t now = AP_HAL::micros64();
|
||||||
|
|
||||||
|
uint64_t timeout_us = timeout_ms * 1000ULL;
|
||||||
|
|
||||||
|
if (now > (servo.last_rx_msg_timestamp + timeout_us)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Determine if an ESC is present on the CAN bus (has telemetry data been received)
|
* Determine if an ESC is present on the CAN bus (has telemetry data been received)
|
||||||
*/
|
*/
|
||||||
@ -603,6 +818,26 @@ bool AP_PiccoloCAN::is_esc_present(uint8_t chan, uint64_t timeout_ms) const
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if a given servo is enabled
|
||||||
|
*/
|
||||||
|
bool AP_PiccoloCAN::is_servo_enabled(uint8_t chan)
|
||||||
|
{
|
||||||
|
if (chan >= PICCOLO_CAN_MAX_NUM_SERVO) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the servo is not present, we cannot determine if it is enabled or not
|
||||||
|
if (!is_servo_present(chan)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
CBSServo_Info_t &servo = _servo_info[chan];
|
||||||
|
|
||||||
|
return servo.statusA.status.enabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if a given ESC is enabled (both hardware and software enable flags)
|
* Check if a given ESC is enabled (both hardware and software enable flags)
|
||||||
*/
|
*/
|
||||||
@ -631,6 +866,18 @@ bool AP_PiccoloCAN::is_esc_enabled(uint8_t chan)
|
|||||||
|
|
||||||
bool AP_PiccoloCAN::pre_arm_check(char* reason, uint8_t reason_len)
|
bool AP_PiccoloCAN::pre_arm_check(char* reason, uint8_t reason_len)
|
||||||
{
|
{
|
||||||
|
// Check that each required servo is present on the bus
|
||||||
|
for (uint8_t ii = 0; ii < PICCOLO_CAN_MAX_NUM_SERVO; ii++) {
|
||||||
|
|
||||||
|
if (is_servo_channel_active(ii)) {
|
||||||
|
|
||||||
|
if (!is_servo_present(ii)) {
|
||||||
|
snprintf(reason, reason_len, "Servo %u not detected", ii + 1);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Check that each required ESC is present on the bus
|
// Check that each required ESC is present on the bus
|
||||||
for (uint8_t ii = 0; ii < PICCOLO_CAN_MAX_NUM_ESC; ii++) {
|
for (uint8_t ii = 0; ii < PICCOLO_CAN_MAX_NUM_ESC; ii++) {
|
||||||
|
|
||||||
|
@ -84,11 +84,23 @@ public:
|
|||||||
// called from SRV_Channels
|
// called from SRV_Channels
|
||||||
void update();
|
void update();
|
||||||
|
|
||||||
|
// send ESC telemetry messages over MAVLink
|
||||||
|
void send_esc_telemetry_mavlink(uint8_t mav_chan);
|
||||||
|
|
||||||
|
// return true if a particular servo is 'active' on the Piccolo interface
|
||||||
|
bool is_servo_channel_active(uint8_t chan);
|
||||||
|
|
||||||
// return true if a particular ESC is 'active' on the Piccolo interface
|
// return true if a particular ESC is 'active' on the Piccolo interface
|
||||||
bool is_esc_channel_active(uint8_t chan);
|
bool is_esc_channel_active(uint8_t chan);
|
||||||
|
|
||||||
// return true if a particular ESC has been detected
|
// return true if a particular servo has been detected on the CAN interface
|
||||||
bool is_esc_present(uint8_t chan, uint64_t timeout_ms = 2000) const;
|
bool is_servo_present(uint8_t chan, uint64_t timeout_ms = 2000);
|
||||||
|
|
||||||
|
// return true if a particular ESC has been detected on the CAN interface
|
||||||
|
bool is_esc_present(uint8_t chan, uint64_t timeout_ms = 2000);
|
||||||
|
|
||||||
|
// return true if a particular servo is enabled
|
||||||
|
bool is_servo_enabled(uint8_t chan);
|
||||||
|
|
||||||
// return true if a particular ESC is enabled
|
// return true if a particular ESC is enabled
|
||||||
bool is_esc_enabled(uint8_t chan);
|
bool is_esc_enabled(uint8_t chan);
|
||||||
@ -113,6 +125,12 @@ private:
|
|||||||
// interpret an ESC message received over CAN
|
// interpret an ESC message received over CAN
|
||||||
bool handle_esc_message(AP_HAL::CANFrame &frame);
|
bool handle_esc_message(AP_HAL::CANFrame &frame);
|
||||||
|
|
||||||
|
// send servo commands over CAN
|
||||||
|
void send_servo_messages(void);
|
||||||
|
|
||||||
|
// interpret a servo message received over CAN
|
||||||
|
bool handle_servo_message(AP_HAL::CANFrame &frame);
|
||||||
|
|
||||||
bool _initialized;
|
bool _initialized;
|
||||||
char _thread_name[16];
|
char _thread_name[16];
|
||||||
uint8_t _driver_index;
|
uint8_t _driver_index;
|
||||||
@ -131,7 +149,15 @@ private:
|
|||||||
Servo_Address_t address;
|
Servo_Address_t address;
|
||||||
Servo_SettingsInfo_t settings;
|
Servo_SettingsInfo_t settings;
|
||||||
Servo_SystemInfo_t systemInfo;
|
Servo_SystemInfo_t systemInfo;
|
||||||
Servo_TelemetrySettings_t telemetry;
|
Servo_TelemetryConfig_t telemetry;
|
||||||
|
|
||||||
|
/* Internal state information */
|
||||||
|
|
||||||
|
int16_t command; //! Raw command to send to each servo
|
||||||
|
bool newCommand; //! Is the command "new"?
|
||||||
|
bool newTelemetry; //! Is there new telemetry data available?
|
||||||
|
|
||||||
|
uint64_t last_rx_msg_timestamp = 0; //! Time of most recently received message
|
||||||
|
|
||||||
} _servo_info[PICCOLO_CAN_MAX_NUM_SERVO];
|
} _servo_info[PICCOLO_CAN_MAX_NUM_SERVO];
|
||||||
|
|
||||||
@ -178,6 +204,8 @@ private:
|
|||||||
AP_Int32 _esc_bm; //! ESC selection bitmask
|
AP_Int32 _esc_bm; //! ESC selection bitmask
|
||||||
AP_Int16 _esc_hz; //! ESC update rate (Hz)
|
AP_Int16 _esc_hz; //! ESC update rate (Hz)
|
||||||
|
|
||||||
|
AP_Int32 _srv_bm; //! Servo selection bitmask
|
||||||
|
AP_Int16 _srv_hz; //! Servo update rate (Hz)
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // HAL_PICCOLO_CAN_ENABLE
|
#endif // HAL_PICCOLO_CAN_ENABLE
|
||||||
|
@ -261,19 +261,19 @@ int decodeServo_Config_t(const uint8_t* _pg_data, int* _pg_bytecount, Servo_Conf
|
|||||||
/*!
|
/*!
|
||||||
* \brief Create the Servo_MultiPositionCommand packet
|
* \brief Create the Servo_MultiPositionCommand packet
|
||||||
*
|
*
|
||||||
* This command can be used to command multiple servos (1 to 4) with a single
|
* This packet can be used to simultaneously command multiple servos which have
|
||||||
* CAN frame). The addresses of the targetted servos must be sequential, with
|
* sequential CAN ID values. This packet must be sent as a broadcast packet
|
||||||
* the initial address based on the packet ID. The example packet shown below
|
* (address = 0xFF) such that all servos can receive it. These commands can be
|
||||||
* corresponds to servos with IDs {1, 2, 3, 4}. For other addresses refer to
|
* sent to groups of servos with ID values up to 64, using different
|
||||||
* the *ServoMultiCommandPackets* enumeration. Note: The CAN frame must be sent
|
* PKT_SERVO_MULTI_COMMAND_x packet ID values.
|
||||||
* to the broadcast ID (0xFF) for all servos to receive this message.
|
|
||||||
* \param _pg_pkt points to the packet which will be created by this function
|
* \param _pg_pkt points to the packet which will be created by this function
|
||||||
* \param commandA is Servo command for servo with address offset 0
|
* \param commandA is Servo command for servo with address offset 0
|
||||||
* \param commandB is Servo command for servo with address offset 1
|
* \param commandB is Servo command for servo with address offset 1
|
||||||
* \param commandC is Servo command for servo with address offset 3
|
* \param commandC is Servo command for servo with address offset 3
|
||||||
* \param commandD is Servo command for servo with address offset 3
|
* \param commandD is Servo command for servo with address offset 3
|
||||||
|
* \param id is the packet identifier for _pg_pkt
|
||||||
*/
|
*/
|
||||||
void encodeServo_MultiPositionCommandPacket(void* _pg_pkt, int16_t commandA, int16_t commandB, int16_t commandC, int16_t commandD)
|
void encodeServo_MultiPositionCommandPacket(void* _pg_pkt, int16_t commandA, int16_t commandB, int16_t commandC, int16_t commandD, uint32_t _pg_id)
|
||||||
{
|
{
|
||||||
uint8_t* _pg_data = getServoPacketData(_pg_pkt);
|
uint8_t* _pg_data = getServoPacketData(_pg_pkt);
|
||||||
int _pg_byteindex = 0;
|
int _pg_byteindex = 0;
|
||||||
@ -295,19 +295,18 @@ void encodeServo_MultiPositionCommandPacket(void* _pg_pkt, int16_t commandA, int
|
|||||||
int16ToBeBytes(commandD, _pg_data, &_pg_byteindex);
|
int16ToBeBytes(commandD, _pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
// complete the process of creating the packet
|
// complete the process of creating the packet
|
||||||
finishServoPacket(_pg_pkt, _pg_byteindex, getServo_MultiPositionCommandPacketID());
|
finishServoPacket(_pg_pkt, _pg_byteindex, _pg_id);
|
||||||
|
|
||||||
}// encodeServo_MultiPositionCommandPacket
|
}// encodeServo_MultiPositionCommandPacket
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \brief Decode the Servo_MultiPositionCommand packet
|
* \brief Decode the Servo_MultiPositionCommand packet
|
||||||
*
|
*
|
||||||
* This command can be used to command multiple servos (1 to 4) with a single
|
* This packet can be used to simultaneously command multiple servos which have
|
||||||
* CAN frame). The addresses of the targetted servos must be sequential, with
|
* sequential CAN ID values. This packet must be sent as a broadcast packet
|
||||||
* the initial address based on the packet ID. The example packet shown below
|
* (address = 0xFF) such that all servos can receive it. These commands can be
|
||||||
* corresponds to servos with IDs {1, 2, 3, 4}. For other addresses refer to
|
* sent to groups of servos with ID values up to 64, using different
|
||||||
* the *ServoMultiCommandPackets* enumeration. Note: The CAN frame must be sent
|
* PKT_SERVO_MULTI_COMMAND_x packet ID values.
|
||||||
* to the broadcast ID (0xFF) for all servos to receive this message.
|
|
||||||
* \param _pg_pkt points to the packet being decoded by this function
|
* \param _pg_pkt points to the packet being decoded by this function
|
||||||
* \param commandA receives Servo command for servo with address offset 0
|
* \param commandA receives Servo command for servo with address offset 0
|
||||||
* \param commandB receives Servo command for servo with address offset 1
|
* \param commandB receives Servo command for servo with address offset 1
|
||||||
@ -321,8 +320,24 @@ int decodeServo_MultiPositionCommandPacket(const void* _pg_pkt, int16_t* command
|
|||||||
const uint8_t* _pg_data = getServoPacketDataConst(_pg_pkt);
|
const uint8_t* _pg_data = getServoPacketDataConst(_pg_pkt);
|
||||||
int _pg_numbytes = getServoPacketSize(_pg_pkt);
|
int _pg_numbytes = getServoPacketSize(_pg_pkt);
|
||||||
|
|
||||||
// Verify the packet identifier
|
// Verify the packet identifier, multiple options exist
|
||||||
if(getServoPacketID(_pg_pkt) != getServo_MultiPositionCommandPacketID())
|
uint32_t packetid = getServoPacketID(_pg_pkt);
|
||||||
|
if( packetid != PKT_SERVO_MULTI_COMMAND_1 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_2 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_3 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_4 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_5 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_6 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_7 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_8 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_9 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_10 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_11 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_12 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_13 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_14 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_15 &&
|
||||||
|
packetid != PKT_SERVO_MULTI_COMMAND_16 )
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if(_pg_numbytes < getServo_MultiPositionCommandMinDataLength())
|
if(_pg_numbytes < getServo_MultiPositionCommandMinDataLength())
|
||||||
@ -555,6 +570,92 @@ int decodeServo_SetTitlePacket(const void* _pg_pkt, uint8_t title[8])
|
|||||||
|
|
||||||
}// decodeServo_SetTitlePacket
|
}// decodeServo_SetTitlePacket
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* \brief Create the Servo_StatusA packet
|
||||||
|
*
|
||||||
|
* The *SERVO_STATUS_A* packet contains status, warning and error information,
|
||||||
|
* in addition to the servo position
|
||||||
|
* \param _pg_pkt points to the packet which will be created by this function
|
||||||
|
* \param _pg_user points to the user data that will be encoded in _pg_pkt
|
||||||
|
*/
|
||||||
|
void encodeServo_StatusAPacketStructure(void* _pg_pkt, const Servo_StatusA_t* _pg_user)
|
||||||
|
{
|
||||||
|
uint8_t* _pg_data = getServoPacketData(_pg_pkt);
|
||||||
|
int _pg_byteindex = 0;
|
||||||
|
|
||||||
|
// Status bits contain information on servo operation
|
||||||
|
encodeServo_StatusBits_t(_pg_data, &_pg_byteindex, &_pg_user->status);
|
||||||
|
|
||||||
|
// Warning bits indicate servo is operation outside of desired range
|
||||||
|
encodeServo_WarningBits_t(_pg_data, &_pg_byteindex, &_pg_user->warnings);
|
||||||
|
|
||||||
|
// These bits indicate critical system error information
|
||||||
|
encodeServo_ErrorBits_t(_pg_data, &_pg_byteindex, &_pg_user->errors);
|
||||||
|
|
||||||
|
// Servo position, mapped to input units
|
||||||
|
// Range of position is -32768 to 32767.
|
||||||
|
int16ToBeBytes(_pg_user->position, _pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
// Servo commanded position
|
||||||
|
// Range of command is -32768 to 32767.
|
||||||
|
int16ToBeBytes(_pg_user->command, _pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
// complete the process of creating the packet
|
||||||
|
finishServoPacket(_pg_pkt, _pg_byteindex, getServo_StatusAPacketID());
|
||||||
|
|
||||||
|
}// encodeServo_StatusAPacketStructure
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* \brief Decode the Servo_StatusA packet
|
||||||
|
*
|
||||||
|
* The *SERVO_STATUS_A* packet contains status, warning and error information,
|
||||||
|
* in addition to the servo position
|
||||||
|
* \param _pg_pkt points to the packet being decoded by this function
|
||||||
|
* \param _pg_user receives the data decoded from the packet
|
||||||
|
* \return 0 is returned if the packet ID or size is wrong, else 1
|
||||||
|
*/
|
||||||
|
int decodeServo_StatusAPacketStructure(const void* _pg_pkt, Servo_StatusA_t* _pg_user)
|
||||||
|
{
|
||||||
|
int _pg_numbytes;
|
||||||
|
int _pg_byteindex = 0;
|
||||||
|
const uint8_t* _pg_data;
|
||||||
|
|
||||||
|
// Verify the packet identifier
|
||||||
|
if(getServoPacketID(_pg_pkt) != getServo_StatusAPacketID())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// Verify the packet size
|
||||||
|
_pg_numbytes = getServoPacketSize(_pg_pkt);
|
||||||
|
if(_pg_numbytes < getServo_StatusAMinDataLength())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// The raw data from the packet
|
||||||
|
_pg_data = getServoPacketDataConst(_pg_pkt);
|
||||||
|
|
||||||
|
// Status bits contain information on servo operation
|
||||||
|
if(decodeServo_StatusBits_t(_pg_data, &_pg_byteindex, &_pg_user->status) == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// Warning bits indicate servo is operation outside of desired range
|
||||||
|
if(decodeServo_WarningBits_t(_pg_data, &_pg_byteindex, &_pg_user->warnings) == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// These bits indicate critical system error information
|
||||||
|
if(decodeServo_ErrorBits_t(_pg_data, &_pg_byteindex, &_pg_user->errors) == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// Servo position, mapped to input units
|
||||||
|
// Range of position is -32768 to 32767.
|
||||||
|
_pg_user->position = int16FromBeBytes(_pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
// Servo commanded position
|
||||||
|
// Range of command is -32768 to 32767.
|
||||||
|
_pg_user->command = int16FromBeBytes(_pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
}// decodeServo_StatusAPacketStructure
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \brief Create the Servo_StatusA packet
|
* \brief Create the Servo_StatusA packet
|
||||||
*
|
*
|
||||||
@ -644,6 +745,77 @@ int decodeServo_StatusAPacket(const void* _pg_pkt, Servo_StatusBits_t* status, S
|
|||||||
|
|
||||||
}// decodeServo_StatusAPacket
|
}// decodeServo_StatusAPacket
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* \brief Create the Servo_StatusB packet
|
||||||
|
*
|
||||||
|
* The *SERVO_STATUS_B* packet contains various servo feedback data
|
||||||
|
* \param _pg_pkt points to the packet which will be created by this function
|
||||||
|
* \param _pg_user points to the user data that will be encoded in _pg_pkt
|
||||||
|
*/
|
||||||
|
void encodeServo_StatusBPacketStructure(void* _pg_pkt, const Servo_StatusB_t* _pg_user)
|
||||||
|
{
|
||||||
|
uint8_t* _pg_data = getServoPacketData(_pg_pkt);
|
||||||
|
int _pg_byteindex = 0;
|
||||||
|
|
||||||
|
// Servo current
|
||||||
|
// Range of current is 0 to 65535.
|
||||||
|
uint16ToBeBytes(_pg_user->current, _pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
// Servo supply voltage
|
||||||
|
// Range of voltage is 0 to 65535.
|
||||||
|
uint16ToBeBytes(_pg_user->voltage, _pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
// Servo temperature
|
||||||
|
// Range of temperature is -128 to 127.
|
||||||
|
int8ToBytes(_pg_user->temperature, _pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
// complete the process of creating the packet
|
||||||
|
finishServoPacket(_pg_pkt, _pg_byteindex, getServo_StatusBPacketID());
|
||||||
|
|
||||||
|
}// encodeServo_StatusBPacketStructure
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* \brief Decode the Servo_StatusB packet
|
||||||
|
*
|
||||||
|
* The *SERVO_STATUS_B* packet contains various servo feedback data
|
||||||
|
* \param _pg_pkt points to the packet being decoded by this function
|
||||||
|
* \param _pg_user receives the data decoded from the packet
|
||||||
|
* \return 0 is returned if the packet ID or size is wrong, else 1
|
||||||
|
*/
|
||||||
|
int decodeServo_StatusBPacketStructure(const void* _pg_pkt, Servo_StatusB_t* _pg_user)
|
||||||
|
{
|
||||||
|
int _pg_numbytes;
|
||||||
|
int _pg_byteindex = 0;
|
||||||
|
const uint8_t* _pg_data;
|
||||||
|
|
||||||
|
// Verify the packet identifier
|
||||||
|
if(getServoPacketID(_pg_pkt) != getServo_StatusBPacketID())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// Verify the packet size
|
||||||
|
_pg_numbytes = getServoPacketSize(_pg_pkt);
|
||||||
|
if(_pg_numbytes < getServo_StatusBMinDataLength())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// The raw data from the packet
|
||||||
|
_pg_data = getServoPacketDataConst(_pg_pkt);
|
||||||
|
|
||||||
|
// Servo current
|
||||||
|
// Range of current is 0 to 65535.
|
||||||
|
_pg_user->current = uint16FromBeBytes(_pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
// Servo supply voltage
|
||||||
|
// Range of voltage is 0 to 65535.
|
||||||
|
_pg_user->voltage = uint16FromBeBytes(_pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
// Servo temperature
|
||||||
|
// Range of temperature is -128 to 127.
|
||||||
|
_pg_user->temperature = int8FromBytes(_pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
}// decodeServo_StatusBPacketStructure
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \brief Create the Servo_StatusB packet
|
* \brief Create the Servo_StatusB packet
|
||||||
*
|
*
|
||||||
@ -714,6 +886,63 @@ int decodeServo_StatusBPacket(const void* _pg_pkt, uint16_t* current, uint16_t*
|
|||||||
|
|
||||||
}// decodeServo_StatusBPacket
|
}// decodeServo_StatusBPacket
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* \brief Create the Servo_StatusC packet
|
||||||
|
*
|
||||||
|
* The *SERVO_STATUS_C* packet contains servo position data. It is a cut-down
|
||||||
|
* packet to allow high-speed feedback on servo position
|
||||||
|
* \param _pg_pkt points to the packet which will be created by this function
|
||||||
|
* \param _pg_user points to the user data that will be encoded in _pg_pkt
|
||||||
|
*/
|
||||||
|
void encodeServo_StatusCPacketStructure(void* _pg_pkt, const Servo_StatusC_t* _pg_user)
|
||||||
|
{
|
||||||
|
uint8_t* _pg_data = getServoPacketData(_pg_pkt);
|
||||||
|
int _pg_byteindex = 0;
|
||||||
|
|
||||||
|
// Servo position, mapped to input units
|
||||||
|
// Range of position is -32768 to 32767.
|
||||||
|
int16ToBeBytes(_pg_user->position, _pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
// complete the process of creating the packet
|
||||||
|
finishServoPacket(_pg_pkt, _pg_byteindex, getServo_StatusCPacketID());
|
||||||
|
|
||||||
|
}// encodeServo_StatusCPacketStructure
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* \brief Decode the Servo_StatusC packet
|
||||||
|
*
|
||||||
|
* The *SERVO_STATUS_C* packet contains servo position data. It is a cut-down
|
||||||
|
* packet to allow high-speed feedback on servo position
|
||||||
|
* \param _pg_pkt points to the packet being decoded by this function
|
||||||
|
* \param _pg_user receives the data decoded from the packet
|
||||||
|
* \return 0 is returned if the packet ID or size is wrong, else 1
|
||||||
|
*/
|
||||||
|
int decodeServo_StatusCPacketStructure(const void* _pg_pkt, Servo_StatusC_t* _pg_user)
|
||||||
|
{
|
||||||
|
int _pg_numbytes;
|
||||||
|
int _pg_byteindex = 0;
|
||||||
|
const uint8_t* _pg_data;
|
||||||
|
|
||||||
|
// Verify the packet identifier
|
||||||
|
if(getServoPacketID(_pg_pkt) != getServo_StatusCPacketID())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// Verify the packet size
|
||||||
|
_pg_numbytes = getServoPacketSize(_pg_pkt);
|
||||||
|
if(_pg_numbytes < getServo_StatusCMinDataLength())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// The raw data from the packet
|
||||||
|
_pg_data = getServoPacketDataConst(_pg_pkt);
|
||||||
|
|
||||||
|
// Servo position, mapped to input units
|
||||||
|
// Range of position is -32768 to 32767.
|
||||||
|
_pg_user->position = int16FromBeBytes(_pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
}// decodeServo_StatusCPacketStructure
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \brief Create the Servo_StatusC packet
|
* \brief Create the Servo_StatusC packet
|
||||||
*
|
*
|
||||||
@ -1421,6 +1650,75 @@ int decodeServo_SystemInfoPacket(const void* _pg_pkt, uint32_t* msSinceReset, ui
|
|||||||
|
|
||||||
}// decodeServo_SystemInfoPacket
|
}// decodeServo_SystemInfoPacket
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* \brief Create the Servo_TelemetryConfig packet
|
||||||
|
*
|
||||||
|
* Telemetry settings configuration packet
|
||||||
|
* \param _pg_pkt points to the packet which will be created by this function
|
||||||
|
* \param _pg_user points to the user data that will be encoded in _pg_pkt
|
||||||
|
*/
|
||||||
|
void encodeServo_TelemetryConfigPacketStructure(void* _pg_pkt, const Servo_TelemetryConfig_t* _pg_user)
|
||||||
|
{
|
||||||
|
uint8_t* _pg_data = getServoPacketData(_pg_pkt);
|
||||||
|
int _pg_byteindex = 0;
|
||||||
|
unsigned _pg_i = 0;
|
||||||
|
|
||||||
|
// Servo telemetry settings
|
||||||
|
encodeServo_TelemetrySettings_t(_pg_data, &_pg_byteindex, &_pg_user->settings);
|
||||||
|
|
||||||
|
// Reserved for future use
|
||||||
|
for(_pg_i = 0; _pg_i < 3; _pg_i++)
|
||||||
|
uint8ToBytes((uint8_t)(0), _pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
// Servo ICD revision
|
||||||
|
uint8ToBytes((uint8_t)(getServoApi()), _pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
// complete the process of creating the packet
|
||||||
|
finishServoPacket(_pg_pkt, _pg_byteindex, getServo_TelemetryConfigPacketID());
|
||||||
|
|
||||||
|
}// encodeServo_TelemetryConfigPacketStructure
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* \brief Decode the Servo_TelemetryConfig packet
|
||||||
|
*
|
||||||
|
* Telemetry settings configuration packet
|
||||||
|
* \param _pg_pkt points to the packet being decoded by this function
|
||||||
|
* \param _pg_user receives the data decoded from the packet
|
||||||
|
* \return 0 is returned if the packet ID or size is wrong, else 1
|
||||||
|
*/
|
||||||
|
int decodeServo_TelemetryConfigPacketStructure(const void* _pg_pkt, Servo_TelemetryConfig_t* _pg_user)
|
||||||
|
{
|
||||||
|
int _pg_numbytes;
|
||||||
|
int _pg_byteindex = 0;
|
||||||
|
const uint8_t* _pg_data;
|
||||||
|
|
||||||
|
// Verify the packet identifier
|
||||||
|
if(getServoPacketID(_pg_pkt) != getServo_TelemetryConfigPacketID())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// Verify the packet size
|
||||||
|
_pg_numbytes = getServoPacketSize(_pg_pkt);
|
||||||
|
if(_pg_numbytes < getServo_TelemetryConfigMinDataLength())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// The raw data from the packet
|
||||||
|
_pg_data = getServoPacketDataConst(_pg_pkt);
|
||||||
|
|
||||||
|
// Servo telemetry settings
|
||||||
|
if(decodeServo_TelemetrySettings_t(_pg_data, &_pg_byteindex, &_pg_user->settings) == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
// Reserved for future use
|
||||||
|
_pg_byteindex += 1*3;
|
||||||
|
|
||||||
|
// Servo ICD revision
|
||||||
|
// Range of icdVersion is 0 to 255.
|
||||||
|
_pg_user->icdVersion = uint8FromBytes(_pg_data, &_pg_byteindex);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
}// decodeServo_TelemetryConfigPacketStructure
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \brief Create the Servo_TelemetryConfig packet
|
* \brief Create the Servo_TelemetryConfig packet
|
||||||
*
|
*
|
||||||
|
@ -72,12 +72,11 @@ int decodeServo_Config_t(const uint8_t* data, int* bytecount, Servo_Config_t* us
|
|||||||
#define getServo_ConfigMaxDataLength() (8)
|
#define getServo_ConfigMaxDataLength() (8)
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* This command can be used to command multiple servos (1 to 4) with a single
|
* This packet can be used to simultaneously command multiple servos which have
|
||||||
* CAN frame). The addresses of the targetted servos must be sequential, with
|
* sequential CAN ID values. This packet must be sent as a broadcast packet
|
||||||
* the initial address based on the packet ID. The example packet shown below
|
* (address = 0xFF) such that all servos can receive it. These commands can be
|
||||||
* corresponds to servos with IDs {1, 2, 3, 4}. For other addresses refer to
|
* sent to groups of servos with ID values up to 64, using different
|
||||||
* the *ServoMultiCommandPackets* enumeration. Note: The CAN frame must be sent
|
* PKT_SERVO_MULTI_COMMAND_x packet ID values.
|
||||||
* to the broadcast ID (0xFF) for all servos to receive this message.
|
|
||||||
*/
|
*/
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
@ -88,14 +87,11 @@ typedef struct
|
|||||||
}Servo_MultiPositionCommand_t;
|
}Servo_MultiPositionCommand_t;
|
||||||
|
|
||||||
//! Create the Servo_MultiPositionCommand packet from parameters
|
//! Create the Servo_MultiPositionCommand packet from parameters
|
||||||
void encodeServo_MultiPositionCommandPacket(void* pkt, int16_t commandA, int16_t commandB, int16_t commandC, int16_t commandD);
|
void encodeServo_MultiPositionCommandPacket(void* pkt, int16_t commandA, int16_t commandB, int16_t commandC, int16_t commandD, uint32_t id);
|
||||||
|
|
||||||
//! Decode the Servo_MultiPositionCommand packet to parameters
|
//! Decode the Servo_MultiPositionCommand packet to parameters
|
||||||
int decodeServo_MultiPositionCommandPacket(const void* pkt, int16_t* commandA, int16_t* commandB, int16_t* commandC, int16_t* commandD);
|
int decodeServo_MultiPositionCommandPacket(const void* pkt, int16_t* commandA, int16_t* commandB, int16_t* commandC, int16_t* commandD);
|
||||||
|
|
||||||
//! return the packet ID for the Servo_MultiPositionCommand packet
|
|
||||||
#define getServo_MultiPositionCommandPacketID() (PKT_SERVO_MULTI_COMMAND_1)
|
|
||||||
|
|
||||||
//! return the minimum encoded length for the Servo_MultiPositionCommand packet
|
//! return the minimum encoded length for the Servo_MultiPositionCommand packet
|
||||||
#define getServo_MultiPositionCommandMinDataLength() (8)
|
#define getServo_MultiPositionCommandMinDataLength() (8)
|
||||||
|
|
||||||
@ -208,6 +204,12 @@ typedef struct
|
|||||||
int16_t command; //!< Servo commanded position
|
int16_t command; //!< Servo commanded position
|
||||||
}Servo_StatusA_t;
|
}Servo_StatusA_t;
|
||||||
|
|
||||||
|
//! Create the Servo_StatusA packet
|
||||||
|
void encodeServo_StatusAPacketStructure(void* pkt, const Servo_StatusA_t* user);
|
||||||
|
|
||||||
|
//! Decode the Servo_StatusA packet
|
||||||
|
int decodeServo_StatusAPacketStructure(const void* pkt, Servo_StatusA_t* user);
|
||||||
|
|
||||||
//! Create the Servo_StatusA packet from parameters
|
//! Create the Servo_StatusA packet from parameters
|
||||||
void encodeServo_StatusAPacket(void* pkt, const Servo_StatusBits_t* status, const Servo_WarningBits_t* warnings, const Servo_ErrorBits_t* errors, int16_t position, int16_t command);
|
void encodeServo_StatusAPacket(void* pkt, const Servo_StatusBits_t* status, const Servo_WarningBits_t* warnings, const Servo_ErrorBits_t* errors, int16_t position, int16_t command);
|
||||||
|
|
||||||
@ -233,6 +235,12 @@ typedef struct
|
|||||||
int8_t temperature; //!< Servo temperature
|
int8_t temperature; //!< Servo temperature
|
||||||
}Servo_StatusB_t;
|
}Servo_StatusB_t;
|
||||||
|
|
||||||
|
//! Create the Servo_StatusB packet
|
||||||
|
void encodeServo_StatusBPacketStructure(void* pkt, const Servo_StatusB_t* user);
|
||||||
|
|
||||||
|
//! Decode the Servo_StatusB packet
|
||||||
|
int decodeServo_StatusBPacketStructure(const void* pkt, Servo_StatusB_t* user);
|
||||||
|
|
||||||
//! Create the Servo_StatusB packet from parameters
|
//! Create the Servo_StatusB packet from parameters
|
||||||
void encodeServo_StatusBPacket(void* pkt, uint16_t current, uint16_t voltage, int8_t temperature);
|
void encodeServo_StatusBPacket(void* pkt, uint16_t current, uint16_t voltage, int8_t temperature);
|
||||||
|
|
||||||
@ -257,6 +265,12 @@ typedef struct
|
|||||||
int16_t position; //!< Servo position, mapped to input units
|
int16_t position; //!< Servo position, mapped to input units
|
||||||
}Servo_StatusC_t;
|
}Servo_StatusC_t;
|
||||||
|
|
||||||
|
//! Create the Servo_StatusC packet
|
||||||
|
void encodeServo_StatusCPacketStructure(void* pkt, const Servo_StatusC_t* user);
|
||||||
|
|
||||||
|
//! Decode the Servo_StatusC packet
|
||||||
|
int decodeServo_StatusCPacketStructure(const void* pkt, Servo_StatusC_t* user);
|
||||||
|
|
||||||
//! Create the Servo_StatusC packet from parameters
|
//! Create the Servo_StatusC packet from parameters
|
||||||
void encodeServo_StatusCPacket(void* pkt, int16_t position);
|
void encodeServo_StatusCPacket(void* pkt, int16_t position);
|
||||||
|
|
||||||
@ -430,6 +444,12 @@ typedef struct
|
|||||||
uint8_t icdVersion; //!< Servo ICD revision. Field is encoded constant.
|
uint8_t icdVersion; //!< Servo ICD revision. Field is encoded constant.
|
||||||
}Servo_TelemetryConfig_t;
|
}Servo_TelemetryConfig_t;
|
||||||
|
|
||||||
|
//! Create the Servo_TelemetryConfig packet
|
||||||
|
void encodeServo_TelemetryConfigPacketStructure(void* pkt, const Servo_TelemetryConfig_t* user);
|
||||||
|
|
||||||
|
//! Decode the Servo_TelemetryConfig packet
|
||||||
|
int decodeServo_TelemetryConfigPacketStructure(const void* pkt, Servo_TelemetryConfig_t* user);
|
||||||
|
|
||||||
//! Create the Servo_TelemetryConfig packet from parameters
|
//! Create the Servo_TelemetryConfig packet from parameters
|
||||||
void encodeServo_TelemetryConfigPacket(void* pkt, const Servo_TelemetrySettings_t* settings);
|
void encodeServo_TelemetryConfigPacket(void* pkt, const Servo_TelemetrySettings_t* settings);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user