diff --git a/vesc_driver/include/vesc_driver/datatypes.h b/vesc_driver/include/vesc_driver/datatypes.h index d71df93..142d283 100644 --- a/vesc_driver/include/vesc_driver/datatypes.h +++ b/vesc_driver/include/vesc_driver/datatypes.h @@ -1,7 +1,7 @@ /* - Copyright 2012-2014 Benjamin Vedder benjamin@vedder.se + Copyright 2012-2014 Benjamin Vedder benjamin@vedder.se - This program is free software: you can redistribute it and/or modify + This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. @@ -28,7 +28,7 @@ #include #include //#include "ch.h" -typedef uint32_t systime_t; // defined in ch.h +typedef uint32_t systime_t; // Data types typedef enum { @@ -39,333 +39,486 @@ typedef enum { } mc_state; typedef enum { - PWM_MODE_NONSYNCHRONOUS_HISW = 0, // This mode is not recommended - PWM_MODE_SYNCHRONOUS, // The recommended and most tested mode - PWM_MODE_BIPOLAR // Some glitches occasionally, can kill MOSFETs + PWM_MODE_NONSYNCHRONOUS_HISW = 0, // This mode is not recommended + PWM_MODE_SYNCHRONOUS, // The recommended and most tested mode + PWM_MODE_BIPOLAR // Some glitches occasionally, can kill MOSFETs } mc_pwm_mode; typedef enum { - COMM_MODE_INTEGRATE = 0, - COMM_MODE_DELAY + COMM_MODE_INTEGRATE = 0, + COMM_MODE_DELAY } mc_comm_mode; typedef enum { - SENSOR_MODE_SENSORLESS = 0, - SENSOR_MODE_SENSORED, - SENSOR_MODE_HYBRID + SENSOR_MODE_SENSORLESS = 0, + SENSOR_MODE_SENSORED, + SENSOR_MODE_HYBRID } mc_sensor_mode; typedef enum { - MOTOR_TYPE_BLDC = 0, - MOTOR_TYPE_DC, + FOC_SENSOR_MODE_SENSORLESS = 0, + FOC_SENSOR_MODE_ENCODER, + FOC_SENSOR_MODE_HALL +} mc_foc_sensor_mode; + +typedef enum { + MOTOR_TYPE_BLDC = 0, + MOTOR_TYPE_DC, + MOTOR_TYPE_FOC } mc_motor_type; typedef enum { - FAULT_CODE_NONE = 0, - FAULT_CODE_OVER_VOLTAGE, - FAULT_CODE_UNDER_VOLTAGE, - FAULT_CODE_DRV8302, - FAULT_CODE_ABS_OVER_CURRENT, - FAULT_CODE_OVER_TEMP_FET, - FAULT_CODE_OVER_TEMP_MOTOR + FAULT_CODE_NONE = 0, + FAULT_CODE_OVER_VOLTAGE, + FAULT_CODE_UNDER_VOLTAGE, + FAULT_CODE_DRV8302, + FAULT_CODE_ABS_OVER_CURRENT, + FAULT_CODE_OVER_TEMP_FET, + FAULT_CODE_OVER_TEMP_MOTOR } mc_fault_code; typedef enum { - CONTROL_MODE_DUTY = 0, - CONTROL_MODE_SPEED, - CONTROL_MODE_CURRENT, - CONTROL_MODE_CURRENT_BRAKE, - CONTROL_MODE_POS, - CONTROL_MODE_NONE + CONTROL_MODE_DUTY = 0, + CONTROL_MODE_SPEED, + CONTROL_MODE_CURRENT, + CONTROL_MODE_CURRENT_BRAKE, + CONTROL_MODE_POS, + CONTROL_MODE_NONE } mc_control_mode; +typedef enum { + DISP_POS_MODE_NONE = 0, + DISP_POS_MODE_INDUCTANCE, + DISP_POS_MODE_OBSERVER, + DISP_POS_MODE_ENCODER, + DISP_POS_MODE_PID_POS, + DISP_POS_MODE_PID_POS_ERROR, + DISP_POS_MODE_ENCODER_OBSERVER_ERROR +} disp_pos_mode; + +typedef enum { + SENSOR_PORT_MODE_HALL = 0, + SENSOR_PORT_MODE_ABI, + SENSOR_PORT_MODE_AS5047_SPI +} sensor_port_mode; + typedef struct { - float cycle_int_limit; - float cycle_int_limit_running; - float cycle_int_limit_max; - float comm_time_sum; - float comm_time_sum_min_rpm; - int32_t comms; - uint32_t time_at_comm; + float cycle_int_limit; + float cycle_int_limit_running; + float cycle_int_limit_max; + float comm_time_sum; + float comm_time_sum_min_rpm; + int32_t comms; + uint32_t time_at_comm; } mc_rpm_dep_struct; typedef struct { - // Switching and drive - mc_pwm_mode pwm_mode; - mc_comm_mode comm_mode; - mc_motor_type motor_type; - mc_sensor_mode sensor_mode; - // Limits - float l_current_max; - float l_current_min; - float l_in_current_max; - float l_in_current_min; - float l_abs_current_max; - float l_min_erpm; - float l_max_erpm; - float l_max_erpm_fbrake; - float l_max_erpm_fbrake_cc; - float l_min_vin; - float l_max_vin; - bool l_slow_abs_current; - bool l_rpm_lim_neg_torque; - float l_temp_fet_start; - float l_temp_fet_end; - float l_temp_motor_start; - float l_temp_motor_end; - float l_min_duty; - float l_max_duty; - // Overridden limits (Computed during runtime) - float lo_current_max; - float lo_current_min; - float lo_in_current_max; - float lo_in_current_min; - // Sensorless - float sl_min_erpm; - float sl_min_erpm_cycle_int_limit; - float sl_max_fullbreak_current_dir_change; - float sl_cycle_int_limit; - float sl_phase_advance_at_br; - float sl_cycle_int_rpm_br; - float sl_bemf_coupling_k; - // Hall sensor - int8_t hall_table[8]; - float hall_sl_erpm; - // Speed PID - float s_pid_kp; - float s_pid_ki; - float s_pid_kd; - float s_pid_min_rpm; - // Pos PID - float p_pid_kp; - float p_pid_ki; - float p_pid_kd; - // Current controller - float cc_startup_boost_duty; - float cc_min_current; - float cc_gain; - float cc_ramp_step_max; - // Misc - int32_t m_fault_stop_time_ms; + // Switching and drive + mc_pwm_mode pwm_mode; + mc_comm_mode comm_mode; + mc_motor_type motor_type; + mc_sensor_mode sensor_mode; + // Limits + float l_current_max; + float l_current_min; + float l_in_current_max; + float l_in_current_min; + float l_abs_current_max; + float l_min_erpm; + float l_max_erpm; + float l_max_erpm_fbrake; + float l_max_erpm_fbrake_cc; + float l_min_vin; + float l_max_vin; + float l_battery_cut_start; + float l_battery_cut_end; + bool l_slow_abs_current; + bool l_rpm_lim_neg_torque; + float l_temp_fet_start; + float l_temp_fet_end; + float l_temp_motor_start; + float l_temp_motor_end; + float l_min_duty; + float l_max_duty; + // Overridden limits (Computed during runtime) + float lo_current_max; + float lo_current_min; + float lo_in_current_max; + float lo_in_current_min; + // Sensorless + float sl_min_erpm; + float sl_min_erpm_cycle_int_limit; + float sl_max_fullbreak_current_dir_change; + float sl_cycle_int_limit; + float sl_phase_advance_at_br; + float sl_cycle_int_rpm_br; + float sl_bemf_coupling_k; + // Hall sensor + int8_t hall_table[8]; + float hall_sl_erpm; + // FOC + float foc_current_kp; + float foc_current_ki; + float foc_f_sw; + float foc_dt_us; + float foc_encoder_offset; + bool foc_encoder_inverted; + float foc_encoder_ratio; + float foc_motor_l; + float foc_motor_r; + float foc_motor_flux_linkage; + float foc_observer_gain; + float foc_pll_kp; + float foc_pll_ki; + float foc_duty_dowmramp_kp; + float foc_duty_dowmramp_ki; + float foc_openloop_rpm; + float foc_sl_openloop_hyst; + float foc_sl_openloop_time; + float foc_sl_d_current_duty; + float foc_sl_d_current_factor; + mc_foc_sensor_mode foc_sensor_mode; + uint8_t foc_hall_table[8]; + float foc_sl_erpm; + // Speed PID + float s_pid_kp; + float s_pid_ki; + float s_pid_kd; + float s_pid_min_erpm; + // Pos PID + float p_pid_kp; + float p_pid_ki; + float p_pid_kd; + float p_pid_ang_div; + // Current controller + float cc_startup_boost_duty; + float cc_min_current; + float cc_gain; + float cc_ramp_step_max; + // Misc + int32_t m_fault_stop_time_ms; + float m_duty_ramp_step; + float m_duty_ramp_step_rpm_lim; + float m_current_backoff_gain; + uint32_t m_encoder_counts; + sensor_port_mode m_sensor_port_mode; } mc_configuration; // Applications to use typedef enum { - APP_NONE = 0, - APP_PPM, - APP_ADC, - APP_UART, - APP_PPM_UART, - APP_ADC_UART, - APP_NUNCHUK, - APP_NRF, - APP_CUSTOM + APP_NONE = 0, + APP_PPM, + APP_ADC, + APP_UART, + APP_PPM_UART, + APP_ADC_UART, + APP_NUNCHUK, + APP_NRF, + APP_CUSTOM } app_use; // PPM control types typedef enum { - PPM_CTRL_TYPE_NONE = 0, - PPM_CTRL_TYPE_CURRENT, - PPM_CTRL_TYPE_CURRENT_NOREV, - PPM_CTRL_TYPE_CURRENT_NOREV_BRAKE, - PPM_CTRL_TYPE_DUTY, - PPM_CTRL_TYPE_DUTY_NOREV, - PPM_CTRL_TYPE_PID, - PPM_CTRL_TYPE_PID_NOREV + PPM_CTRL_TYPE_NONE = 0, + PPM_CTRL_TYPE_CURRENT, + PPM_CTRL_TYPE_CURRENT_NOREV, + PPM_CTRL_TYPE_CURRENT_NOREV_BRAKE, + PPM_CTRL_TYPE_DUTY, + PPM_CTRL_TYPE_DUTY_NOREV, + PPM_CTRL_TYPE_PID, + PPM_CTRL_TYPE_PID_NOREV } ppm_control_type; typedef struct { - ppm_control_type ctrl_type; - float pid_max_erpm; - float hyst; - float pulse_start; - float pulse_end; - bool median_filter; - bool safe_start; - float rpm_lim_start; - float rpm_lim_end; - bool multi_esc; - bool tc; - float tc_max_diff; + ppm_control_type ctrl_type; + float pid_max_erpm; + float hyst; + float pulse_start; + float pulse_end; + bool median_filter; + bool safe_start; + float rpm_lim_start; + float rpm_lim_end; + bool multi_esc; + bool tc; + float tc_max_diff; } ppm_config; // ADC control types typedef enum { - ADC_CTRL_TYPE_NONE = 0, - ADC_CTRL_TYPE_CURRENT, - ADC_CTRL_TYPE_CURRENT_REV_CENTER, - ADC_CTRL_TYPE_CURRENT_REV_BUTTON, - ADC_CTRL_TYPE_CURRENT_NOREV_BRAKE_CENTER, - ADC_CTRL_TYPE_CURRENT_NOREV_BRAKE_BUTTON, - ADC_CTRL_TYPE_DUTY, - ADC_CTRL_TYPE_DUTY_REV_CENTER, - ADC_CTRL_TYPE_DUTY_REV_BUTTON + ADC_CTRL_TYPE_NONE = 0, + ADC_CTRL_TYPE_CURRENT, + ADC_CTRL_TYPE_CURRENT_REV_CENTER, + ADC_CTRL_TYPE_CURRENT_REV_BUTTON, + ADC_CTRL_TYPE_CURRENT_NOREV_BRAKE_CENTER, + ADC_CTRL_TYPE_CURRENT_NOREV_BRAKE_BUTTON, + ADC_CTRL_TYPE_CURRENT_NOREV_BRAKE_ADC, + ADC_CTRL_TYPE_DUTY, + ADC_CTRL_TYPE_DUTY_REV_CENTER, + ADC_CTRL_TYPE_DUTY_REV_BUTTON } adc_control_type; typedef struct { - adc_control_type ctrl_type; - float hyst; - float voltage_start; - float voltage_end; - bool use_filter; - bool safe_start; - bool button_inverted; - bool voltage_inverted; - float rpm_lim_start; - float rpm_lim_end; - bool multi_esc; - bool tc; - float tc_max_diff; - uint32_t update_rate_hz; + adc_control_type ctrl_type; + float hyst; + float voltage_start; + float voltage_end; + bool use_filter; + bool safe_start; + bool cc_button_inverted; + bool rev_button_inverted; + bool voltage_inverted; + float rpm_lim_start; + float rpm_lim_end; + bool multi_esc; + bool tc; + float tc_max_diff; + uint32_t update_rate_hz; } adc_config; // Nunchuk control types typedef enum { - CHUK_CTRL_TYPE_NONE = 0, - CHUK_CTRL_TYPE_CURRENT, - CHUK_CTRL_TYPE_CURRENT_NOREV + CHUK_CTRL_TYPE_NONE = 0, + CHUK_CTRL_TYPE_CURRENT, + CHUK_CTRL_TYPE_CURRENT_NOREV } chuk_control_type; typedef struct { - chuk_control_type ctrl_type; - float hyst; - float rpm_lim_start; - float rpm_lim_end; - float ramp_time_pos; - float ramp_time_neg; - bool multi_esc; - bool tc; - float tc_max_diff; + chuk_control_type ctrl_type; + float hyst; + float rpm_lim_start; + float rpm_lim_end; + float ramp_time_pos; + float ramp_time_neg; + float stick_erpm_per_s_in_cc; + bool multi_esc; + bool tc; + float tc_max_diff; } chuk_config; +// NRF Datatypes +typedef enum { + NRF_SPEED_250K = 0, + NRF_SPEED_1M, + NRF_SPEED_2M +} NRF_SPEED; + +typedef enum { + NRF_POWER_M18DBM = 0, + NRF_POWER_M12DBM, + NRF_POWER_M6DBM, + NRF_POWER_0DBM +} NRF_POWER; + +typedef enum { + NRF_AW_3 = 0, + NRF_AW_4, + NRF_AW_5 +} NRF_AW; + +typedef enum { + NRF_CRC_DISABLED = 0, + NRF_CRC_1B, + NRF_CRC_2B +} NRF_CRC; + +typedef enum { + NRF_RETR_DELAY_250US = 0, + NRF_RETR_DELAY_500US, + NRF_RETR_DELAY_750US, + NRF_RETR_DELAY_1000US, + NRF_RETR_DELAY_1250US, + NRF_RETR_DELAY_1500US, + NRF_RETR_DELAY_1750US, + NRF_RETR_DELAY_2000US, + NRF_RETR_DELAY_2250US, + NRF_RETR_DELAY_2500US, + NRF_RETR_DELAY_2750US, + NRF_RETR_DELAY_3000US, + NRF_RETR_DELAY_3250US, + NRF_RETR_DELAY_3500US, + NRF_RETR_DELAY_3750US, + NRF_RETR_DELAY_4000US +} NRF_RETR_DELAY; + typedef struct { - // Settings - uint8_t controller_id; - uint32_t timeout_msec; - float timeout_brake_current; - bool send_can_status; - uint32_t send_can_status_rate_hz; + NRF_SPEED speed; + NRF_POWER power; + NRF_CRC crc_type; + NRF_RETR_DELAY retry_delay; + unsigned char retries; + unsigned char channel; + unsigned char address[3]; + bool send_crc_ack; +} nrf_config; - // Application to use - app_use app_to_use; +typedef struct { + // Settings + uint8_t controller_id; + uint32_t timeout_msec; + float timeout_brake_current; + bool send_can_status; + uint32_t send_can_status_rate_hz; + + // Application to use + app_use app_to_use; + + // PPM application settings + ppm_config app_ppm_conf; - // PPM application settings - ppm_config app_ppm_conf; + // ADC application settings + adc_config app_adc_conf; - // ADC application settings - adc_config app_adc_conf; + // UART application settings + uint32_t app_uart_baudrate; - // UART application settings - uint32_t app_uart_baudrate; + // Nunchuk application settings + chuk_config app_chuk_conf; - // Nunchuk application settings - chuk_config app_chuk_conf; + // NRF application settings + nrf_config app_nrf_conf; } app_configuration; // Communication commands typedef enum { - COMM_FW_VERSION = 0, - COMM_JUMP_TO_BOOTLOADER, - COMM_ERASE_NEW_APP, - COMM_WRITE_NEW_APP_DATA, - COMM_GET_VALUES, - COMM_SET_DUTY, - COMM_SET_CURRENT, - COMM_SET_CURRENT_BRAKE, - COMM_SET_RPM, - COMM_SET_POS, - COMM_SET_DETECT, - COMM_SET_SERVO_POS, - COMM_SET_MCCONF, - COMM_GET_MCCONF, - COMM_SET_APPCONF, - COMM_GET_APPCONF, - COMM_SAMPLE_PRINT, - COMM_TERMINAL_CMD, - COMM_PRINT, - COMM_ROTOR_POSITION, - COMM_EXPERIMENT_SAMPLE, - COMM_DETECT_MOTOR_PARAM, - COMM_REBOOT, - COMM_ALIVE, - COMM_GET_DECODED_PPM, - COMM_GET_DECODED_ADC, - COMM_GET_DECODED_CHUK, - COMM_FORWARD_CAN + COMM_FW_VERSION = 0, + COMM_JUMP_TO_BOOTLOADER, + COMM_ERASE_NEW_APP, + COMM_WRITE_NEW_APP_DATA, + COMM_GET_VALUES, + COMM_SET_DUTY, + COMM_SET_CURRENT, + COMM_SET_CURRENT_BRAKE, + COMM_SET_RPM, + COMM_SET_POS, + COMM_SET_DETECT, + COMM_SET_SERVO_POS, + COMM_SET_MCCONF, + COMM_GET_MCCONF, + COMM_GET_MCCONF_DEFAULT, + COMM_SET_APPCONF, + COMM_GET_APPCONF, + COMM_GET_APPCONF_DEFAULT, + COMM_SAMPLE_PRINT, + COMM_TERMINAL_CMD, + COMM_PRINT, + COMM_ROTOR_POSITION, + COMM_EXPERIMENT_SAMPLE, + COMM_DETECT_MOTOR_PARAM, + COMM_DETECT_MOTOR_R_L, + COMM_DETECT_MOTOR_FLUX_LINKAGE, + COMM_DETECT_ENCODER, + COMM_DETECT_HALL_FOC, + COMM_REBOOT, + COMM_ALIVE, + COMM_GET_DECODED_PPM, + COMM_GET_DECODED_ADC, + COMM_GET_DECODED_CHUK, + COMM_FORWARD_CAN, + COMM_SET_CHUCK_DATA, + COMM_CUSTOM_APP_DATA } COMM_PACKET_ID; // CAN commands typedef enum { - CAN_PACKET_SET_DUTY = 0, - CAN_PACKET_SET_CURRENT, - CAN_PACKET_SET_CURRENT_BRAKE, - CAN_PACKET_SET_RPM, - CAN_PACKET_SET_POS, - CAN_PACKET_FILL_RX_BUFFER, - CAN_PACKET_FILL_RX_BUFFER_LONG, - CAN_PACKET_PROCESS_RX_BUFFER, - CAN_PACKET_PROCESS_SHORT_BUFFER, - CAN_PACKET_STATUS + CAN_PACKET_SET_DUTY = 0, + CAN_PACKET_SET_CURRENT, + CAN_PACKET_SET_CURRENT_BRAKE, + CAN_PACKET_SET_RPM, + CAN_PACKET_SET_POS, + CAN_PACKET_FILL_RX_BUFFER, + CAN_PACKET_FILL_RX_BUFFER_LONG, + CAN_PACKET_PROCESS_RX_BUFFER, + CAN_PACKET_PROCESS_SHORT_BUFFER, + CAN_PACKET_STATUS } CAN_PACKET_ID; // Logged fault data typedef struct { - mc_fault_code fault; - float current; - float current_filtered; - float voltage; - float duty; - float rpm; - int tacho; - int tim_pwm_cnt; - int tim_samp_cnt; - int comm_step; - float temperature; + mc_fault_code fault; + float current; + float current_filtered; + float voltage; + float duty; + float rpm; + int tacho; + int cycles_running; + int tim_val_samp; + int tim_current_samp; + int tim_top; + int comm_step; + float temperature; } fault_data; // External LED state typedef enum { - LED_EXT_OFF = 0, - LED_EXT_NORMAL, - LED_EXT_BRAKE, - LED_EXT_TURN_LEFT, - LED_EXT_TURN_RIGHT, - LED_EXT_BRAKE_TURN_LEFT, - LED_EXT_BRAKE_TURN_RIGHT, - LED_EXT_BATT + LED_EXT_OFF = 0, + LED_EXT_NORMAL, + LED_EXT_BRAKE, + LED_EXT_TURN_LEFT, + LED_EXT_TURN_RIGHT, + LED_EXT_BRAKE_TURN_LEFT, + LED_EXT_BRAKE_TURN_RIGHT, + LED_EXT_BATT } LED_EXT_STATE; typedef struct { - int js_x; - int js_y; - int acc_x; - int acc_y; - int acc_z; - bool bt_c; - bool bt_z; + int js_x; + int js_y; + int acc_x; + int acc_y; + int acc_z; + bool bt_c; + bool bt_z; } chuck_data; typedef struct { - int id; - systime_t rx_time; - float rpm; - float current; - float duty; + int id; + systime_t rx_time; + float rpm; + float current; + float duty; } can_status_msg; typedef struct { - uint8_t js_x; - uint8_t js_y; - bool bt_c; - bool bt_z; - bool bt_push; - float vbat; + uint8_t js_x; + uint8_t js_y; + bool bt_c; + bool bt_z; + bool bt_push; + float vbat; } mote_state; typedef enum { - MOTE_PACKET_BATT_LEVEL = 0, - MOTE_PACKET_BUTTONS, - MOTE_PACKET_ALIVE + MOTE_PACKET_BATT_LEVEL = 0, + MOTE_PACKET_BUTTONS, + MOTE_PACKET_ALIVE, + MOTE_PACKET_FILL_RX_BUFFER, + MOTE_PACKET_FILL_RX_BUFFER_LONG, + MOTE_PACKET_PROCESS_RX_BUFFER, + MOTE_PACKET_PROCESS_SHORT_BUFFER, } MOTE_PACKET; +typedef struct { + float v_in; + float temp_mos1; + float temp_mos2; + float temp_mos3; + float temp_mos4; + float temp_mos5; + float temp_mos6; + float temp_pcb; + float current_motor; + float current_in; + float rpm; + float duty_now; + float amp_hours; + float amp_hours_charged; + float watt_hours; + float watt_hours_charged; + int tachometer; + int tachometer_abs; + mc_fault_code fault_code; +} mc_values; + #endif /* DATATYPES_H_ */ diff --git a/vesc_driver/include/vesc_driver/vesc_driver.h b/vesc_driver/include/vesc_driver/vesc_driver.h index 005c0be..98e7bcf 100644 --- a/vesc_driver/include/vesc_driver/vesc_driver.h +++ b/vesc_driver/include/vesc_driver/vesc_driver.h @@ -55,6 +55,7 @@ class VescDriver ros::Subscriber speed_sub_; ros::Subscriber position_sub_; ros::Subscriber servo_sub_; + ros::Publisher rotor_position_pub_; ros::Timer timer_; // driver modes (possible states) diff --git a/vesc_driver/include/vesc_driver/vesc_interface.h b/vesc_driver/include/vesc_driver/vesc_interface.h index 3bc50d0..0887fea 100644 --- a/vesc_driver/include/vesc_driver/vesc_interface.h +++ b/vesc_driver/include/vesc_driver/vesc_interface.h @@ -14,6 +14,7 @@ #include #include "vesc_driver/vesc_packet.h" +#include "vesc_driver/datatypes.h" namespace vesc_driver { @@ -91,6 +92,7 @@ class VescInterface : private boost::noncopyable void setSpeed(double speed); void setPosition(double position); void setServo(double servo); + void setDetect(disp_pos_mode mode); private: // Pimpl - hide serial port members from class users diff --git a/vesc_driver/include/vesc_driver/vesc_packet.h b/vesc_driver/include/vesc_driver/vesc_packet.h index 630634e..381331b 100644 --- a/vesc_driver/include/vesc_driver/vesc_packet.h +++ b/vesc_driver/include/vesc_driver/vesc_packet.h @@ -75,6 +75,18 @@ class VescPacket : public VescFrame typedef boost::shared_ptr VescPacketPtr; typedef boost::shared_ptr VescPacketConstPtr; +/*------------------------------------------------------------------------------------------------*/ + +class VescPacketRotorPosition : public VescPacket +{ +public: + VescPacketRotorPosition(boost::shared_ptr raw); + + float position() const; + +}; + + /*------------------------------------------------------------------------------------------------*/ class VescPacketFWVersion : public VescPacket @@ -189,6 +201,14 @@ class VescPacketSetServoPos : public VescPacket // double servo_pos() const; }; +/*------------------------------------------------------------------------------------------------*/ + +class VescPacketSetDetect : public VescPacket +{ +public: + VescPacketSetDetect(uint8_t mode); +}; + } // namespace vesc_driver #endif // VESC_DRIVER_VESC_PACKET_H_ diff --git a/vesc_driver/launch/vesc_driver_node.launch b/vesc_driver/launch/vesc_driver_node.launch index f7d4765..3cac041 100644 --- a/vesc_driver/launch/vesc_driver_node.launch +++ b/vesc_driver/launch/vesc_driver_node.launch @@ -11,10 +11,15 @@ + + + + diff --git a/vesc_driver/launch/vesc_driver_nodelet.launch b/vesc_driver/launch/vesc_driver_nodelet.launch index c9f4508..184ef09 100644 --- a/vesc_driver/launch/vesc_driver_nodelet.launch +++ b/vesc_driver/launch/vesc_driver_nodelet.launch @@ -14,7 +14,11 @@ - + + + + @@ -29,6 +33,7 @@ - + + diff --git a/vesc_driver/src/vesc_driver.cpp b/vesc_driver/src/vesc_driver.cpp index 3c4788a..831c23e 100644 --- a/vesc_driver/src/vesc_driver.cpp +++ b/vesc_driver/src/vesc_driver.cpp @@ -1,6 +1,6 @@ // -*- mode:c++; fill-column: 100; -*- - #include "vesc_driver/vesc_driver.h" +#include "vesc_driver/datatypes.h" #include #include @@ -8,6 +8,7 @@ #include #include +#include namespace vesc_driver { @@ -58,6 +59,36 @@ VescDriver::VescDriver(ros::NodeHandle nh, // create a 50Hz timer, used for state machine & polling VESC telemetry timer_ = nh.createTimer(ros::Duration(1.0/50.0), &VescDriver::timerCallback, this); + + // decide whether or not we want to publish rotor positions + // this is enabled with by sending a "set detect" packet with a mode + // "encoder" is probably the most useful + std::string rotor_position_source; + if (private_nh.getParam("rotor_position_source", rotor_position_source)) { + disp_pos_mode mode = DISP_POS_MODE_NONE; + if (rotor_position_source.compare("inductance") == 0) { + mode = DISP_POS_MODE_INDUCTANCE; + } else if (rotor_position_source.compare("observer") == 0) { + mode = DISP_POS_MODE_OBSERVER; + } else if (rotor_position_source.compare("encoder") == 0) { + mode = DISP_POS_MODE_ENCODER; + } else if (rotor_position_source.compare("pid_pos") == 0) { + mode = DISP_POS_MODE_PID_POS; + } else if (rotor_position_source.compare("pid_pos_error") == 0) { + mode = DISP_POS_MODE_PID_POS_ERROR; + } else if (rotor_position_source.compare("encoder_observer_error") == 0) { + mode = DISP_POS_MODE_ENCODER_OBSERVER_ERROR; + } else if (rotor_position_source.compare("none") != 0) { + ROS_WARN("Invalid display mode parameter, defaulting to none"); + } + + if (mode != DISP_POS_MODE_NONE) { + // create rotor position publisher + rotor_position_pub_ = nh.advertise("sensors/rotor_position", 10); + ROS_INFO("Enabling rotor position publisher from source: %s", rotor_position_source.c_str()); + } + vesc_.setDetect(mode); + } } /* TODO or TO-THINKABOUT LIST @@ -109,7 +140,7 @@ void VescDriver::timerCallback(const ros::TimerEvent& event) void VescDriver::vescPacketCallback(const boost::shared_ptr& packet) { - if (packet->name() == "Values") { + if (packet->name() == "Values" && state_pub_) { boost::shared_ptr values = boost::dynamic_pointer_cast(packet); @@ -138,6 +169,16 @@ void VescDriver::vescPacketCallback(const boost::shared_ptr& p fw_version_major_ = fw_version->fwMajor(); fw_version_minor_ = fw_version->fwMinor(); } + else if (packet->name() == "RotorPosition" && rotor_position_pub_) { + //pubish the value of the rotor position + boost::shared_ptr position = + boost::dynamic_pointer_cast(packet); + + std_msgs::Float32::Ptr rotor_position_msg(new std_msgs::Float32); + // we get degrees from the VESC, but radians are the standard in ROS + rotor_position_msg->data = position->position() * M_PI / 180.0; + rotor_position_pub_.publish(rotor_position_msg); + } } void VescDriver::vescErrorCallback(const std::string& error) diff --git a/vesc_driver/src/vesc_interface.cpp b/vesc_driver/src/vesc_interface.cpp index 84a2cf2..bd50918 100644 --- a/vesc_driver/src/vesc_interface.cpp +++ b/vesc_driver/src/vesc_interface.cpp @@ -13,6 +13,7 @@ #include #include "vesc_driver/vesc_packet_factory.h" +#include "vesc_driver/datatypes.h" namespace vesc_driver { @@ -241,4 +242,9 @@ void VescInterface::setServo(double servo) send(VescPacketSetServoPos(servo)); } +void VescInterface::setDetect(disp_pos_mode mode) +{ + send(VescPacketSetDetect(mode)); +} + } // namespace vesc_driver diff --git a/vesc_driver/src/vesc_packet.cpp b/vesc_driver/src/vesc_packet.cpp index 80328a7..b911270 100644 --- a/vesc_driver/src/vesc_packet.cpp +++ b/vesc_driver/src/vesc_packet.cpp @@ -69,6 +69,26 @@ VescPacket::VescPacket(const std::string& name, boost::shared_ptr raw /*------------------------------------------------------------------------------------------------*/ +VescPacketRotorPosition::VescPacketRotorPosition(boost::shared_ptr raw) : + VescPacket("RotorPosition", raw) +{ +} + +float VescPacketRotorPosition::position() const +{ + int32_t value = 0; + value |= *(payload_.first + 1) << 24; + value |= *(payload_.first + 2) << 16; + value |= *(payload_.first + 3) << 8; + value |= *(payload_.first + 4); + + return value / 100000.0; +} + +REGISTER_PACKET_TYPE(COMM_ROTOR_POSITION, VescPacketRotorPosition) + +/*------------------------------------------------------------------------------------------------*/ + VescPacketFWVersion::VescPacketFWVersion(boost::shared_ptr raw) : VescPacket("FWVersion", raw) { @@ -365,4 +385,18 @@ VescPacketSetServoPos::VescPacketSetServoPos(double servo_pos) : *(frame_->end() - 2) = static_cast(crc & 0xFF); } +/*------------------------------------------------------------------------------------------------*/ + +VescPacketSetDetect::VescPacketSetDetect(uint8_t mode) : + VescPacket("SetDetect", 3, COMM_SET_DETECT) +{ + *(payload_.first + 1) = mode; + + VescFrame::CRC crc_calc; + crc_calc.process_bytes(&(*payload_.first), boost::distance(payload_)); + uint16_t crc = crc_calc.checksum(); + *(frame_->end() - 3) = static_cast(crc >> 8); + *(frame_->end() - 2) = static_cast(crc & 0xFF); +} + } // namespace vesc_driver