diff --git a/Node/.gitignore b/Node/.gitignore index 608f518..42637a7 100644 --- a/Node/.gitignore +++ b/Node/.gitignore @@ -1,4 +1,5 @@ build/ +Build/ Drivers Makefile .mxproject @@ -8,5 +9,13 @@ Core/Inc Core/Src !Core/Src/main.c !Core/Inc/main.h +!Core/Src/can_handler.c +!Core/Inc/can_handler.h +!Core/Inc/utilities.h +!Core/Inc/can_message_defs.h +!Core/Src/state.c +!Core/Inc/state.h +!Core/Src/motor_control.c +!Core/Inc/motor_control.h compile_commands.json .cache/ diff --git a/Node/CMakeLists.txt b/Node/CMakeLists.txt index 3c3f62c..a41d63d 100644 --- a/Node/CMakeLists.txt +++ b/Node/CMakeLists.txt @@ -13,8 +13,6 @@ set(STM32CUBEMX_GENERATED_FILES Core/Src/main.c Core/Src/gpio.c Core/Src/can.c - Core/Src/i2c.c - Core/Src/spi.c Core/Src/tim.c Core/Src/usart.c Core/Src/stm32f3xx_it.c @@ -43,8 +41,22 @@ set(STM32CUBEMX_GENERATED_FILES startup_stm32f334x8.s ) +set(STM32_USER_CREATED_FILES + Core/Src/can_handler.c + Core/Src/state.c + Core/Src/motor_control.c +) + +set(EXTERNAL_FILES + External/Src/bldc_interface_uart.c + External/Src/bldc_interface.c + External/Src/buffer.c + External/Src/crc.c + External/Src/packet.c +) + set(EXECUTABLE ${PROJECT_NAME}.out) -add_executable(${EXECUTABLE} ${STM32CUBEMX_GENERATED_FILES}) +add_executable(${EXECUTABLE} ${STM32CUBEMX_GENERATED_FILES} ${STM32_USER_CREATED_FILES} ${EXTERNAL_FILES}) target_compile_definitions(${EXECUTABLE} PRIVATE -DUSE_HAL_DRIVER @@ -52,6 +64,7 @@ target_compile_definitions(${EXECUTABLE} PRIVATE ) target_include_directories(${EXECUTABLE} PRIVATE + External/Inc Core/Inc Drivers/STM32F3xx_HAL_Driver/Inc Drivers/CMSIS/Device/ST/STM32F3xx/Include @@ -69,7 +82,7 @@ target_compile_options(${EXECUTABLE} PRIVATE -Wall - $<$:-Og> + -Og ) target_link_options(${EXECUTABLE} PRIVATE diff --git a/Node/Core/Inc/can_handler.h b/Node/Core/Inc/can_handler.h new file mode 100644 index 0000000..3fbdeec --- /dev/null +++ b/Node/Core/Inc/can_handler.h @@ -0,0 +1,21 @@ +#include + +typedef struct CanMessage +{ + uint16_t id; + uint8_t len; + uint8_t data[8]; +} CanMessage; + +typedef enum CanStatus +{ + CAN_GOOD, + CAN_RX_FIFO_EMPTY, + CAN_TX_MAILBOXES_FULL, + CAN_DATA_TOO_LONG, +} CanStatus; + +CanStatus receive_can_message(CanMessage* message); +CanStatus send_can_message(CanMessage* message); +CanStatus send_can_message_blocking(CanMessage* message); +void handle_can_messages(uint8_t num_msgs_to_handle); diff --git a/Node/Core/Inc/can_message_defs.h b/Node/Core/Inc/can_message_defs.h new file mode 100644 index 0000000..3f295d6 --- /dev/null +++ b/Node/Core/Inc/can_message_defs.h @@ -0,0 +1,70 @@ +#define HEARTBEAT_ID 0x0001 + +#define STATE_CHANGE_CAN_ID 0x0010 +#define STANDBY_REQUEST_CAN_ID 0x0011 +#define AUTONOMOUS_REQUEST_CAN_ID 0x0012 +#define MANUAL_REQUEST_CAN_ID 0x0013 +#define BOOT_RESPONSE_CAN_ID 0x0016 +#define STANDBY_RESPONSE_CAN_ID 0x0017 +#define AUTONOMOUS_RESPONSE_CAN_ID 0x0018 +#define MANUAL_RESPONSE_CAN_ID 0x0019 + +#define NODE_GOOD_CAN_ID_MASK 0x0014 +#define NODE_BAD_CAN_ID_MASK 0x0015 + +#define MOTOR_CONTROL_RPM_MASK_MANUAL 0x0020 +#define MOTOR_1_RPM_MANUAL 0x0021 +#define MOTOR_2_RPM_MANUAL 0x0022 +#define MOTOR_3_RPM_MANUAL 0x0023 + +#define MOTOR_CONTROL_RPM_MASK_AUTONOMOUS 0x0040 +#define MOTOR_1_RPM_AUTONOMOUS 0x0041 +#define MOTOR_2_RPM_AUTONOMOUS 0x0042 +#define MOTOR_3_RPM_AUTONOMOUS 0x0043 + +#define IS_MOTOR1_RPM_CONTROL(motor) ((motor == MOTOR_1_RPM_AUTONOMOUS) || (motor == MOTOR_1_RPM_MANUAL)) +#define IS_MOTOR2_RPM_CONTROL(motor) ((motor == MOTOR_2_RPM_AUTONOMOUS) || (motor == MOTOR_2_RPM_MANUAL)) +#define IS_MOTOR3_RPM_CONTROL(motor) ((motor == MOTOR_3_RPM_AUTONOMOUS) || (motor == MOTOR_3_RPM_MANUAL)) + +#define MOTOR_CONTROL_CURRENT_MASK_MANUAL 0x0080 +#define MOTOR_1_CURRENT_MANUAL 0x0081 +#define MOTOR_2_CURRENT_MANUAL 0x0082 +#define MOTOR_3_CURRENT_MANUAL 0x0083 + +#define MOTOR_CONTROL_CURRENT_MASK_AUTONOMOUS 0x0100 +#define MOTOR_1_CURRENT_AUTONOMOUS 0x0101 +#define MOTOR_2_CURRENT_AUTONOMOUS 0x0102 +#define MOTOR_3_CURRENT_AUTONOMOUS 0x0103 + +#define IS_MOTOR1_CURRENT_CONTROL(motor) ((motor == MOTOR_1_CURRENT_AUTONOMOUS) || (motor == MOTOR_1_CURRENT_MANUAL)) +#define IS_MOTOR2_CURRENT_CONTROL(motor) ((motor == MOTOR_2_CURRENT_AUTONOMOUS) || (motor == MOTOR_2_CURRENT_MANUAL)) +#define IS_MOTOR3_CURRENT_CONTROL(motor) ((motor == MOTOR_3_CURRENT_AUTONOMOUS) || (motor == MOTOR_3_CURRENT_MANUAL)) + +#define MOTOR_CONTROL_DUTY_CYCLE_MASK_MANUAL 0x0200 +#define MOTOR_1_DUTY_CYCLE_MANUAL 0x0201 +#define MOTOR_2_DUTY_CYCLE_MANUAL 0x0202 +#define MOTOR_3_DUTY_CYCLE_MANUAL 0x0203 + +#define MOTOR_CONTROL_DUTY_CYCLE_MASK_AUTONOMOUS 0x0400 +#define MOTOR_1_DUTY_CYCLE_AUTONOMOUS 0x0401 +#define MOTOR_2_DUTY_CYCLE_AUTONOMOUS 0x0402 +#define MOTOR_3_DUTY_CYCLE_AUTONOMOUS 0x0403 + +#define IS_MOTOR1_DUTY_CYCLE_CONTROL(motor) ((motor == MOTOR_1_DUTY_CYCLE_AUTONOMOUS) || (motor == MOTOR_1_DUTY_CYCLE_MANUAL)) +#define IS_MOTOR2_DUTY_CYCLE_CONTROL(motor) ((motor == MOTOR_2_DUTY_CYCLE_AUTONOMOUS) || (motor == MOTOR_2_DUTY_CYCLE_MANUAL)) +#define IS_MOTOR3_DUTY_CYCLE_CONTROL(motor) ((motor == MOTOR_3_DUTY_CYCLE_AUTONOMOUS) || (motor == MOTOR_3_DUTY_CYCLE_MANUAL)) + +#define WIRELESS_ESTOP 120 + +#define LED_1_TEST_ON 0x0002 +#define LED_2_TEST_ON 0x0003 +#define LAMP_1_TEST_ON 0x0004 +#define LAMP_2_TEST_ON 0x0005 +#define LAMP_3_TEST_ON 0x0006 +#define LAMP_4_TEST_ON 0x0007 +#define LED_1_TEST_OFF 0x0008 +#define LED_2_TEST_OFF 0x0009 +#define LAMP_1_TEST_OFF 0x000a +#define LAMP_2_TEST_OFF 0x000b +#define LAMP_3_TEST_OFF 0x000c +#define LAMP_4_TEST_OFF 0x000d diff --git a/Node/Core/Inc/main.h b/Node/Core/Inc/main.h index d47ef39..f80e9f7 100644 --- a/Node/Core/Inc/main.h +++ b/Node/Core/Inc/main.h @@ -36,6 +36,8 @@ extern "C" { /* Exported types ------------------------------------------------------------*/ /* USER CODE BEGIN ET */ +#include "bldc_interface.h" + /* USER CODE END ET */ @@ -53,6 +55,11 @@ extern "C" { void Error_Handler(void); /* USER CODE BEGIN EFP */ +extern BldcInterface motor1; +extern BldcInterface motor2; +extern BldcInterface motor3; +extern uint8_t vesc_data_valid[3]; +int heartbeat_expired(uint32_t last_heartbeat_received_ms); /* USER CODE END EFP */ diff --git a/Node/Core/Inc/motor_control.h b/Node/Core/Inc/motor_control.h new file mode 100644 index 0000000..55623d4 --- /dev/null +++ b/Node/Core/Inc/motor_control.h @@ -0,0 +1,10 @@ +#ifndef MOTOR_CONTROL_H +#define MOTOR_CONTROL_H + +#include + +void motor_control_rpm(uint32_t id, uint8_t data[]); +void motor_control_current(uint32_t id, uint8_t data[]); +void motor_control_duty_cycle(uint32_t id, uint8_t data[]); + +#endif \ No newline at end of file diff --git a/Node/Core/Inc/state.h b/Node/Core/Inc/state.h new file mode 100644 index 0000000..2242951 --- /dev/null +++ b/Node/Core/Inc/state.h @@ -0,0 +1,20 @@ +#ifndef STATE_H +#define STATE_H + +#include + +typedef enum +{ + STANDBY, + AUTONOMOUS, + MANUAL, + BOOT +} state; + +void get_next_state(uint32_t id); + +extern state curr_state; +extern state next_state; +extern uint32_t last_heartbeat_received; + +#endif \ No newline at end of file diff --git a/Node/Core/Inc/utilities.h b/Node/Core/Inc/utilities.h new file mode 100644 index 0000000..05abd75 --- /dev/null +++ b/Node/Core/Inc/utilities.h @@ -0,0 +1 @@ +#define ARRAY_SIZE(arr) (sizeof(arr)/sizeof(arr[0])) \ No newline at end of file diff --git a/Node/Core/Src/can_handler.c b/Node/Core/Src/can_handler.c new file mode 100644 index 0000000..c023711 --- /dev/null +++ b/Node/Core/Src/can_handler.c @@ -0,0 +1,186 @@ +#include "bldc_interface.h" +#include "can_handler.h" +#include "can_message_defs.h" +#include "main.h" +#include "motor_control.h" +#include "state.h" +#include "stm32f334x8.h" +#include "stm32f3xx_hal.h" +#include "stm32f3xx_hal_can.h" +#include "stm32f3xx_hal_gpio.h" +#include "utilities.h" + +#include + +static CAN_TxHeaderTypeDef txHeader; + +static void led_test(uint32_t id, GPIO_PinState state) +{ + GPIO_TypeDef* port; + uint16_t pin; + + switch(id) + { + case LED_1_TEST_ON: + case LED_1_TEST_OFF: + port = LED1_GPIO_Port; + pin = LED1_Pin; + break; + case LED_2_TEST_ON: + case LED_2_TEST_OFF: + port = LED2_GPIO_Port; + pin = LED2_Pin; + break; + case LAMP_1_TEST_ON: + case LAMP_1_TEST_OFF: + port = LAMP1_ON_GPIO_Port; + pin = LAMP1_ON_Pin; + break; + case LAMP_2_TEST_ON: + case LAMP_2_TEST_OFF: + port = LAMP2_ON_GPIO_Port; + pin = LAMP2_ON_Pin; + break; + case LAMP_3_TEST_ON: + case LAMP_3_TEST_OFF: + port = LAMP3_ON_GPIO_Port; + pin = LAMP3_ON_Pin; + break; + case LAMP_4_TEST_ON: + case LAMP_4_TEST_OFF: + port = LAMP4_ON_GPIO_Port; + pin = LAMP4_ON_Pin; + break; + default: + return; + } + + HAL_GPIO_WritePin(port, pin, state); +} + +CanStatus receive_can_message(CanMessage* message) +{ + CAN_RxHeaderTypeDef header; + + uint32_t rx_fifos[] = { CAN_RX_FIFO0, CAN_RX_FIFO1 }; + + for (int i = 0; i < ARRAY_SIZE(rx_fifos); i++) + { + if (HAL_CAN_GetRxFifoFillLevel(&hcan, rx_fifos[i]) != 0) + { + HAL_CAN_GetRxMessage(&hcan, rx_fifos[i], &header, message->data); + message->id = header.StdId; + message->len = header.DLC; + + return CAN_GOOD; + } + } + + return CAN_RX_FIFO_EMPTY; +} + +// Won't send message if Tx mailbox is full +CanStatus send_can_message(CanMessage* message) +{ + if (message->len > 8) { return CAN_DATA_TOO_LONG; } + + if (HAL_CAN_GetTxMailboxesFreeLevel(&hcan) == 0) + { + return CAN_TX_MAILBOXES_FULL; + } + + txHeader.StdId = message->id; + txHeader.ExtId = message->id; + txHeader.IDE = CAN_ID_STD; + txHeader.RTR = CAN_RTR_DATA; + txHeader.DLC = message->len; + txHeader.TransmitGlobalTime = DISABLE; + + uint32_t mailbox; + + if (HAL_CAN_AddTxMessage(&hcan, &txHeader, message->data, &mailbox) != HAL_OK) + { + return CAN_TX_MAILBOXES_FULL; + } + + return CAN_GOOD; +} + +// Will block and try to send CAN message +CanStatus send_can_message_blocking(CanMessage* message) +{ + CanStatus status; + do { status = send_can_message(message); } + while (status == CAN_TX_MAILBOXES_FULL); + + return status; +} + +void handle_can_messages(uint8_t num_msgs_to_handle) +{ + CanMessage message; + CanStatus status; + + for (int i = 0; i < num_msgs_to_handle; i++) + { + status = receive_can_message(&message); + if (status == CAN_GOOD) + { + if (message.id == HEARTBEAT_ID) + { + last_heartbeat_received = HAL_GetTick(); + } + + // RPM motor control + else if ((message.id & MOTOR_CONTROL_RPM_MASK_MANUAL) && + curr_state == MANUAL) + { + motor_control_rpm(message.id, message.data); + } + else if ((message.id & MOTOR_CONTROL_RPM_MASK_AUTONOMOUS) && + curr_state == AUTONOMOUS) + { + motor_control_rpm(message.id, message.data); + } + + // Current motor control + else if ((message.id & MOTOR_CONTROL_CURRENT_MASK_MANUAL) && + curr_state == MANUAL) + { + motor_control_current(message.id, message.data); + } + else if ((message.id & MOTOR_CONTROL_CURRENT_MASK_AUTONOMOUS) && + curr_state == AUTONOMOUS) + { + motor_control_current(message.id, message.data); + } + + // Duty cycle motor control + else if ((message.id & MOTOR_CONTROL_DUTY_CYCLE_MASK_MANUAL) && + curr_state == MANUAL) + { + motor_control_duty_cycle(message.id, message.data); + } + else if ((message.id & MOTOR_CONTROL_DUTY_CYCLE_MASK_AUTONOMOUS) && + curr_state == AUTONOMOUS) + { + motor_control_duty_cycle(message.id, message.data); + } + + else if (message.id & STATE_CHANGE_CAN_ID) + { + get_next_state(message.id); + } + + else if (message.id < 0x7) + { + led_test(message.id, GPIO_PIN_SET); + } + else if (message.id > 0x7 && message.id < 0xe) + { + led_test(message.id, GPIO_PIN_RESET); + } + } + else if (status == CAN_RX_FIFO_EMPTY) { break; } + } +} diff --git a/Node/Core/Src/main.c b/Node/Core/Src/main.c index 95a595c..2d9d95c 100644 --- a/Node/Core/Src/main.c +++ b/Node/Core/Src/main.c @@ -17,17 +17,24 @@ */ /* USER CODE END Header */ /* Includes ------------------------------------------------------------------*/ -#include "main.h" +#include "bldc_interface.h" #include "can.h" -#include "i2c.h" -#include "spi.h" +#include "gpio.h" +#include "main.h" +#include "stm32f3xx_hal.h" +#include "stm32f3xx_hal_gpio.h" #include "tim.h" #include "usart.h" -#include "gpio.h" /* Private includes ----------------------------------------------------------*/ /* USER CODE BEGIN Includes */ +#include "bldc_interface_uart.h" +#include "can_handler.h" +#include "can_message_defs.h" +#include "state.h" +#include +#include /* USER CODE END Includes */ /* Private typedef -----------------------------------------------------------*/ @@ -47,7 +54,17 @@ /* Private variables ---------------------------------------------------------*/ /* USER CODE BEGIN PV */ +BldcInterface motor1 = {0}; +BldcInterface motor2 = {0}; +BldcInterface motor3 = {0}; + +state curr_state = BOOT; +state next_state = BOOT; + +uint32_t last_heartbeat_received = 0; +static const uint32_t HEARTBEAT_EXPIRED_MS = 1000; +uint8_t vesc_data_valid[3] = {0}; /* USER CODE END PV */ /* Private function prototypes -----------------------------------------------*/ @@ -58,6 +75,311 @@ void SystemClock_Config(void); /* Private user code ---------------------------------------------------------*/ /* USER CODE BEGIN 0 */ +static void write_packet_motor1(unsigned char* data, unsigned int len) +{ + HAL_UART_Transmit(&huart2, data, len, 1000); +} + +static void write_packet_motor2(unsigned char* data, unsigned int len) +{ + HAL_UART_Transmit(&huart3, data, len, 1000); +} + +static void write_packet_motor3(unsigned char* data, unsigned int len) +{ + HAL_UART_Transmit(&huart1, data, len, 1000); +} + +uint8_t rx_data_motor1; +uint8_t rx_data_motor2; +uint8_t rx_data_motor3; + +uint32_t last_time; +uint32_t second_last_time; + +// E-stop interrupts +void HAL_GPIO_EXTI_Callback(uint16_t gpio_pin) +{ + if (gpio_pin == GPIO_PIN_7 || gpio_pin == GPIO_PIN_4) + { + next_state = STANDBY; + } +} + +void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) +{ + BldcInterface* motor; + unsigned char* data; + if (huart == &huart1) + { + motor = &motor3; + data = &rx_data_motor3; + } + else if (huart == &huart2) + { + motor = &motor1; + data = &rx_data_motor1; + } + else // huart == &huart3 + { + motor = &motor2; + data = &rx_data_motor2; + } + + bldc_interface_uart_process_byte(motor, *data); + HAL_UART_Receive_IT(huart, data, 1); +} + +static void bldc_values_received_motor1(mc_values* val) +{ + static const uint32_t motor1_ids[] = { + 0x40, + 0x41, + 0x42, + 0x43, + 0x44 + }; + + vesc_data_valid[0] = 1; + + CanMessage message1; + message1.id = motor1_ids[0]; + message1.len = 8; + memcpy(message1.data, &val->v_in, 4); + memcpy((message1.data + 4), &val->temp_mos, 4); + + CanMessage message2; + message2.id = motor1_ids[1]; + message2.len = 8; + memcpy(message2.data, &val->temp_motor, 4); + memcpy((message2.data + 4), &val->current_motor, 4); + + CanMessage message3; + message3.id = motor1_ids[2]; + message3.len = 8; + memcpy(message3.data, &val->current_in, 4); + memcpy((message3.data + 4), &val->id, 4); + + CanMessage message4; + message4.id = motor1_ids[3]; + message4.len = 8; + memcpy(message4.data, &val->iq, 4); + memcpy((message4.data + 4), &val->rpm, 4); + + CanMessage message5; + message5.id = motor1_ids[4]; + message5.len = 8; + memcpy(message5.data, &val->duty_now, 4); + memcpy((message5.data + 4), &val->amp_hours, 4); + + //send_can_message(&message1); + // send_can_message_blocking(&message2); + // send_can_message_blocking(&message3); + // send_can_message_blocking(&message4); + // send_can_message_blocking(&message5); +} + +static void bldc_values_received_motor2(mc_values* val) +{ + static const uint32_t motor2_ids[] = { + 0x50, + 0x51, + 0x52, + 0x53, + 0x54 + }; + + vesc_data_valid[1] = 1; + + CanMessage message1; + message1.id = motor2_ids[0]; + message1.len = 8; + memcpy(message1.data, &val->v_in, 4); + memcpy((message1.data + 4), &val->temp_mos, 4); + + CanMessage message2; + message2.id = motor2_ids[1]; + message2.len = 8; + memcpy(message2.data, &val->temp_motor, 4); + memcpy((message2.data + 4), &val->current_motor, 4); + + CanMessage message3; + message3.id = motor2_ids[2]; + message3.len = 8; + memcpy(message3.data, &val->current_in, 4); + memcpy((message3.data + 4), &val->id, 4); + + CanMessage message4; + message4.id = motor2_ids[3]; + message4.len = 8; + memcpy(message4.data, &val->iq, 4); + memcpy((message4.data + 4), &val->rpm, 4); + + CanMessage message5; + message5.id = motor2_ids[4]; + message5.len = 8; + memcpy(message5.data, &val->duty_now, 4); + memcpy((message5.data + 4), &val->amp_hours, 4); + + send_can_message_blocking(&message1); + // send_can_message_blocking(&message2); + // send_can_message_blocking(&message3); + // send_can_message_blocking(&message4); + // send_can_message_blocking(&message5); +} +static void bldc_values_received_motor3(mc_values* val) +{ + static const uint32_t motor3_ids[] = { + 0x60, + 0x61, + 0x62, + 0x63, + 0x64 + }; + + vesc_data_valid[2] = 1; + + CanMessage message1; + message1.id = motor3_ids[0]; + message1.len = 8; + memcpy(message1.data, &val->v_in, 4); + memcpy((message1.data + 4), &val->temp_mos, 4); + + CanMessage message2; + message2.id = motor3_ids[1]; + message2.len = 8; + memcpy(message2.data, &val->temp_motor, 4); + memcpy((message2.data + 4), &val->current_motor, 4); + + CanMessage message3; + message3.id = motor3_ids[2]; + message3.len = 8; + memcpy(message3.data, &val->current_in, 4); + memcpy((message3.data + 4), &val->id, 4); + + CanMessage message4; + message4.id = motor3_ids[3]; + message4.len = 8; + memcpy(message4.data, &val->iq, 4); + memcpy((message4.data + 4), &val->rpm, 4); + + CanMessage message5; + message5.id = motor3_ids[4]; + message5.len = 8; + memcpy(message5.data, &val->duty_now, 4); + memcpy((message5.data + 4), &val->amp_hours, 4); + + send_can_message_blocking(&message1); + // send_can_message_blocking(&message2); + // send_can_message_blocking(&message3); + // send_can_message_blocking(&message4); + // send_can_message_blocking(&message5); +} + +static void write_lamps(void) +{ + // Solid LEDs + if (curr_state == STANDBY) + { + HAL_GPIO_WritePin(LAMP1_ON_GPIO_Port, LAMP1_ON_Pin, GPIO_PIN_SET); + HAL_GPIO_WritePin(LAMP2_ON_GPIO_Port, LAMP2_ON_Pin, GPIO_PIN_SET); + HAL_GPIO_WritePin(LAMP3_ON_GPIO_Port, LAMP3_ON_Pin, GPIO_PIN_SET); + HAL_GPIO_WritePin(LAMP4_ON_GPIO_Port, LAMP4_ON_Pin, GPIO_PIN_SET); + } + + // Blink LEDs in Autonomous drive + else if (curr_state == AUTONOMOUS) + { + // Wait 0.5 seconds for flash in Autonomous + if (HAL_GetTick()-last_time > 250) + { + HAL_GPIO_TogglePin(LAMP1_ON_GPIO_Port, LAMP1_ON_Pin); + HAL_GPIO_TogglePin(LAMP2_ON_GPIO_Port, LAMP2_ON_Pin); + HAL_GPIO_TogglePin(LAMP3_ON_GPIO_Port, LAMP3_ON_Pin); + HAL_GPIO_TogglePin(LAMP4_ON_GPIO_Port, LAMP4_ON_Pin); + last_time = HAL_GetTick(); + } + } + + // Blink LEDs in Manual drive + else if (curr_state == MANUAL) + { + // Wait 0.3 seconds for flash in Manual + if (HAL_GetTick()-second_last_time > 2000) + { + HAL_GPIO_TogglePin(LAMP1_ON_GPIO_Port, LAMP1_ON_Pin); + HAL_GPIO_TogglePin(LAMP2_ON_GPIO_Port, LAMP2_ON_Pin); + HAL_GPIO_TogglePin(LAMP3_ON_GPIO_Port, LAMP3_ON_Pin); + HAL_GPIO_TogglePin(LAMP4_ON_GPIO_Port, LAMP4_ON_Pin); + second_last_time = HAL_GetTick(); + } + } + + // Blink LEDs in Manual drive + else if (curr_state == BOOT) + { + // Wait 0.3 seconds for flash in Manual + if (HAL_GetTick()-second_last_time > 25) + { + HAL_GPIO_TogglePin(LAMP1_ON_GPIO_Port, LAMP1_ON_Pin); + HAL_GPIO_TogglePin(LAMP2_ON_GPIO_Port, LAMP2_ON_Pin); + HAL_GPIO_TogglePin(LAMP3_ON_GPIO_Port, LAMP3_ON_Pin); + HAL_GPIO_TogglePin(LAMP4_ON_GPIO_Port, LAMP4_ON_Pin); + second_last_time = HAL_GetTick(); + } + } +} + +int heartbeat_expired(uint32_t last_heartbeat_received_ms) +{ + return (HAL_GetTick() - last_heartbeat_received_ms) > HEARTBEAT_EXPIRED_MS; +} + +static void send_current_state(state curr_state) +{ + CanMessage message; + message.len = 0; + + switch (curr_state) + { + case BOOT: + message.id = BOOT_RESPONSE_CAN_ID; + break; + case STANDBY: + message.id = STANDBY_RESPONSE_CAN_ID; + break; + case MANUAL: + message.id = MANUAL_RESPONSE_CAN_ID; + break; + case AUTONOMOUS: + message.id = AUTONOMOUS_RESPONSE_CAN_ID; + break; + } + + send_can_message(&message); +} + +static const uint32_t GET_VALUES_INTERVAL = 500; + +static void try_get_values_motor(BldcInterface* motor, uint32_t* last_received) +{ + if (HAL_GetTick() - *last_received > GET_VALUES_INTERVAL) + { + bldc_interface_get_values(motor); + *last_received = HAL_GetTick(); + } +} + +static const uint32_t SEND_CURRENT_STATE_INTERVAL = 500; + +static void try_send_current_state(state curr_state, uint32_t* last_received) +{ + if (HAL_GetTick() - *last_received > SEND_CURRENT_STATE_INTERVAL) + { + send_current_state(curr_state); + *last_received = HAL_GetTick(); + } +} /* USER CODE END 0 */ @@ -68,6 +390,14 @@ void SystemClock_Config(void); int main(void) { /* USER CODE BEGIN 1 */ + bldc_interface_uart_init(&motor1, write_packet_motor1); + bldc_interface_set_rx_value_func(&motor1, bldc_values_received_motor1); + + bldc_interface_uart_init(&motor2, write_packet_motor2); + bldc_interface_set_rx_value_func(&motor2, bldc_values_received_motor2); + + bldc_interface_uart_init(&motor3, write_packet_motor3); + bldc_interface_set_rx_value_func(&motor3, bldc_values_received_motor3); /* USER CODE END 1 */ @@ -96,19 +426,72 @@ int main(void) MX_TIM2_Init(); MX_TIM3_Init(); MX_TIM17_Init(); - MX_I2C1_Init(); - MX_SPI1_Init(); /* USER CODE BEGIN 2 */ + last_time = HAL_GetTick(); + second_last_time = HAL_GetTick(); + + CAN_FilterTypeDef sf; + sf.FilterMaskIdHigh = 0x0000; + sf.FilterMaskIdLow = 0x0000; + sf.FilterFIFOAssignment = CAN_FILTER_FIFO0; + sf.FilterBank = 0; + sf.FilterMode = CAN_FILTERMODE_IDMASK; + sf.FilterScale = CAN_FILTERSCALE_32BIT; + sf.FilterActivation = CAN_FILTER_ENABLE; + if (HAL_CAN_ConfigFilter(&hcan, &sf) != HAL_OK) { + Error_Handler(); + } + HAL_CAN_Start(&hcan); + + HAL_UART_Receive_IT(&huart1, &rx_data_motor1, 1); + HAL_UART_Receive_IT(&huart2, &rx_data_motor2, 1); + HAL_UART_Receive_IT(&huart3, &rx_data_motor3, 1); + + uint32_t motor1_values_last_received = HAL_GetTick(); + uint32_t motor2_values_last_received = HAL_GetTick() + 100; + uint32_t motor3_values_last_received = HAL_GetTick() + 200; + + uint32_t current_state_last_sent = HAL_GetTick(); + /* USER CODE END 2 */ /* Infinite loop */ /* USER CODE BEGIN WHILE */ while (1) { + handle_can_messages(5); + + // Actuate GPIOs based on current state + write_lamps(); + // try_send_current_state(curr_state, ¤t_state_last_sent); + + if (heartbeat_expired(last_heartbeat_received)) + { + curr_state = BOOT; + next_state = BOOT; + vesc_data_valid[0] = 0; + vesc_data_valid[1] = 0; + vesc_data_valid[2] = 0; + last_heartbeat_received = 0; + continue; + } + + // try_get_values_motor(&motor1, &motor1_values_last_received); + // try_get_values_motor(&motor2, &motor2_values_last_received); + // try_get_values_motor(&motor3, &motor3_values_last_received); + + if (curr_state == BOOT) + { + next_state = STANDBY; + } + + curr_state = next_state; // next state is set based on CAN Messages + /* USER CODE END WHILE */ /* USER CODE BEGIN 3 */ + } /* USER CODE END 3 */ } @@ -148,9 +531,8 @@ void SystemClock_Config(void) { Error_Handler(); } - PeriphClkInit.PeriphClockSelection = RCC_PERIPHCLK_USART1|RCC_PERIPHCLK_I2C1; + PeriphClkInit.PeriphClockSelection = RCC_PERIPHCLK_USART1; PeriphClkInit.Usart1ClockSelection = RCC_USART1CLKSOURCE_PCLK1; - PeriphClkInit.I2c1ClockSelection = RCC_I2C1CLKSOURCE_HSI; if (HAL_RCCEx_PeriphCLKConfig(&PeriphClkInit) != HAL_OK) { Error_Handler(); diff --git a/Node/Core/Src/motor_control.c b/Node/Core/Src/motor_control.c new file mode 100644 index 0000000..0d05d81 --- /dev/null +++ b/Node/Core/Src/motor_control.c @@ -0,0 +1,103 @@ +#include "bldc_interface.h" +#include "can_message_defs.h" +#include "main.h" +#include "motor_control.h" + +#include + +static const float MAX_RPM = 100000.0; +static const float MAX_CURRENT = 500.0; +static const float MAX_DUTY_CYCLE = 0.05; + +void motor_control_rpm(uint32_t id, uint8_t data[]) +{ + float rpm; + BldcInterface* motor; + memcpy(&rpm, data, sizeof(float)); + + if (rpm > MAX_RPM || rpm < -MAX_RPM) + { + return; + } + + if (IS_MOTOR1_RPM_CONTROL(id)) + { + motor = &motor1; + } + else if (IS_MOTOR2_RPM_CONTROL(id)) + { + motor = &motor2; + } + else if (IS_MOTOR3_RPM_CONTROL(id)) + { + motor = &motor3; + } + else + { + return; + } + + bldc_interface_set_rpm(motor, rpm); +} + +void motor_control_current(uint32_t id, uint8_t data[]) +{ + float current; + BldcInterface* motor; + memcpy(¤t, data, sizeof(float)); + + if (current > MAX_CURRENT || current < -MAX_CURRENT) + { + return; + } + + if (IS_MOTOR1_CURRENT_CONTROL(id)) + { + motor = &motor1; + } + else if (IS_MOTOR2_CURRENT_CONTROL(id)) + { + motor = &motor2; + } + else if (IS_MOTOR3_CURRENT_CONTROL(id)) + { + motor = &motor3; + } + else + { + return; + } + + bldc_interface_set_current(motor, current); +} + +void motor_control_duty_cycle(uint32_t id, uint8_t data[]) +{ + float duty_cycle; + BldcInterface* motor; + memcpy(&duty_cycle, data, sizeof(float)); + + if (duty_cycle > MAX_DUTY_CYCLE || duty_cycle < -MAX_DUTY_CYCLE) + { + return; + } + + if (IS_MOTOR1_DUTY_CYCLE_CONTROL(id)) + { + motor = &motor1; + } + else if (IS_MOTOR2_DUTY_CYCLE_CONTROL(id)) + { + motor = &motor2; + } + else if (IS_MOTOR3_DUTY_CYCLE_CONTROL(id)) + { + motor = &motor3; + } + else + { + return; + } + + bldc_interface_set_duty_cycle(motor, duty_cycle); +} \ No newline at end of file diff --git a/Node/Core/Src/state.c b/Node/Core/Src/state.c new file mode 100644 index 0000000..3e7b23d --- /dev/null +++ b/Node/Core/Src/state.c @@ -0,0 +1,41 @@ +#include "state.h" +#include "can_message_defs.h" +#include "main.h" +#include "stm32f3xx_hal.h" +#include +#include "state.h" + +// Called in handle_can_messages(), LEDs actuated by state in main +void get_next_state(uint32_t id) +{ + switch(curr_state) + { + case STANDBY: + if (id == AUTONOMOUS_REQUEST_CAN_ID) + { + next_state = AUTONOMOUS; + break; + } + if (id == MANUAL_REQUEST_CAN_ID) + { + next_state = MANUAL; + break; + } + break; + case AUTONOMOUS: + if (id == STANDBY_REQUEST_CAN_ID) + { + next_state = STANDBY; + } + break; + case MANUAL: + if (id == STANDBY_REQUEST_CAN_ID) + { + next_state = STANDBY; + } + break; + default: + next_state = BOOT; + break; + } +} diff --git a/Node/External/Inc/bldc_interface.h b/Node/External/Inc/bldc_interface.h new file mode 100644 index 0000000..9210ce9 --- /dev/null +++ b/Node/External/Inc/bldc_interface.h @@ -0,0 +1,122 @@ +/* + Copyright 2016-2017 Benjamin Vedder benjamin@vedder.se + + 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. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ + +#ifndef BLDC_INTERFACE_H_ +#define BLDC_INTERFACE_H_ + +#include "datatypes.h" + +typedef struct BldcInterface +{ + // Private variables + unsigned char send_buffer[1024]; + int packet_handler_num; + + // Private variables for received data + mc_values values; + int fw_major; + int fw_minor; + float rotor_pos; + mc_configuration mcconf; + app_configuration appconf; + float detect_cycle_int_limit; + float detect_coupling_k; + signed char detect_hall_table[8]; + signed char detect_hall_res; + float dec_ppm; + float dec_ppm_len; + float dec_adc; + float dec_adc_voltage; + float dec_chuk; + + // Function pointers + void(*send_func)(struct BldcInterface* interface, unsigned char *data, unsigned int len); + void(*forward_func)(unsigned char *data, unsigned int len); + + // Function pointers for received data + void(*rx_value_func)(mc_values *values); + void(*rx_printf_func)(char *str); + void(*rx_fw_func)(int major, int minor); + void(*rx_rotor_pos_func)(float pos); + void(*rx_mcconf_func)(mc_configuration *conf); + void(*rx_appconf_func)(app_configuration *conf); + void(*rx_detect_func)(float cycle_int_limit, float coupling_k, + const signed char *hall_table, signed char hall_res); + void(*rx_dec_ppm_func)(float val, float ms); + void(*rx_dec_adc_func)(float val, float voltage); + void(*rx_dec_chuk_func)(float val); + void(*rx_mcconf_received_func)(void); + void(*rx_appconf_received_func)(void); + void(*motor_control_set_func)(motor_control_mode mode, float value); + void(*values_requested_func)(void); +} BldcInterface; + +// interface functions +void bldc_interface_init(BldcInterface* interface, void(*func)(BldcInterface* interface, unsigned char *data, unsigned int len)); +void bldc_interface_set_forward_func(BldcInterface* interface, void(*func)(unsigned char *data, unsigned int len)); +void bldc_interface_send_packet(BldcInterface* interface, unsigned char *data, unsigned int len); +void bldc_interface_process_packet(BldcInterface* interface, unsigned char *data, unsigned int len); + +// Function pointer setters +void bldc_interface_set_rx_value_func(BldcInterface* interface, void(*func)(mc_values *values)); +void bldc_interface_set_rx_printf_func(BldcInterface* interface, void(*func)(char *str)); +void bldc_interface_set_rx_fw_func(BldcInterface* interface, void(*func)(int major, int minor)); +void bldc_interface_set_rx_rotor_pos_func(BldcInterface* interface, void(*func)(float pos)); +void bldc_interface_set_rx_mcconf_func(BldcInterface* interface, void(*func)(mc_configuration *conf)); +void bldc_interface_set_rx_appconf_func(BldcInterface* interface, void(*func)(app_configuration *conf)); +void bldc_interface_set_rx_detect_func(BldcInterface* interface, void(*func)(float cycle_int_limit, float coupling_k, + const signed char *hall_table, signed char hall_res)); +void bldc_interface_set_rx_dec_ppm_func(BldcInterface* interface, void(*func)(float val, float ms)); +void bldc_interface_set_rx_dec_adc_func(BldcInterface* interface, void(*func)(float val, float voltage)); +void bldc_interface_set_rx_dec_chuk_func(BldcInterface* interface, void(*func)(float val)); +void bldc_interface_set_rx_mcconf_received_func(BldcInterface* interface, void(*func)(void)); +void bldc_interface_set_rx_appconf_received_func(BldcInterface* interface, void(*func)(void)); + +void bldc_interface_set_sim_control_function(BldcInterface* interface, void(*func)(motor_control_mode mode, float value)); +void bldc_interface_set_sim_values_func(BldcInterface* interface, void(*func)(void)); + +// Setters +void bldc_interface_terminal_cmd(BldcInterface* interface, char* cmd); +void bldc_interface_set_duty_cycle(BldcInterface* interface, float dutyCycle); +void bldc_interface_set_current(BldcInterface* interface, float current); +void bldc_interface_set_current_brake(BldcInterface* interface, float current); +void bldc_interface_set_rpm(BldcInterface* interface, int rpm); +void bldc_interface_set_pos(BldcInterface* interface, float pos); +void bldc_interface_set_handbrake(BldcInterface* interface, float current); +void bldc_interface_set_servo_pos(BldcInterface* interface, float pos); +void bldc_interface_set_mcconf(BldcInterface* interface, const mc_configuration *mcconf); +void bldc_interface_set_appconf(BldcInterface* interface, const app_configuration *appconf); + +// Getters +void bldc_interface_get_fw_version(BldcInterface* interface); +void bldc_interface_get_values(BldcInterface* interface); +void bldc_interface_get_mcconf(BldcInterface* interface); +void bldc_interface_get_appconf(BldcInterface* interface); +void bldc_interface_get_decoded_ppm(BldcInterface* interface); +void bldc_interface_get_decoded_adc(BldcInterface* interface); +void bldc_interface_get_decoded_chuk(BldcInterface* interface); + +// Other functions +void bldc_interface_detect_motor_param(BldcInterface*, float current, float min_rpm, float low_duty); +void bldc_interface_reboot(BldcInterface*); +void bldc_interface_send_alive(BldcInterface*); +void send_values_to_receiver(BldcInterface*, mc_values *values); + +// Helpers +const char* bldc_interface_fault_to_string(mc_fault_code fault); + +#endif /* BLDC_INTERFACE_H_ */ diff --git a/Node/External/Inc/bldc_interface_uart.h b/Node/External/Inc/bldc_interface_uart.h new file mode 100644 index 0000000..acfa058 --- /dev/null +++ b/Node/External/Inc/bldc_interface_uart.h @@ -0,0 +1,37 @@ +/* + Copyright 2015 Benjamin Vedder benjamin@vedder.se + + 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. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ + +/* + * bldc_interface_uart.h + * + * Created on: 9 okt 2015 + * Author: benjamin + */ + +#ifndef BLDC_INTERFACE_UART_H_ +#define BLDC_INTERFACE_UART_H_ + +// Includes +#include "packet.h" // For the MAX_PACKET_LEN define +#include "bldc_interface.h" + +// Functions +void bldc_interface_uart_init(BldcInterface* interface, void(*func)(unsigned char *data, unsigned int len)); +void bldc_interface_uart_process_byte(BldcInterface* interface, unsigned char b); +void bldc_interface_uart_run_timer(BldcInterface* interface); + +#endif /* BLDC_INTERFACE_UART_H_ */ diff --git a/Node/External/Inc/buffer.h b/Node/External/Inc/buffer.h new file mode 100644 index 0000000..75315ff --- /dev/null +++ b/Node/External/Inc/buffer.h @@ -0,0 +1,44 @@ +/* + Copyright 2012-2018 Benjamin Vedder benjamin@vedder.se + + 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. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ + +#ifndef BUFFER_H_ +#define BUFFER_H_ + +#include + +void buffer_append_int16(uint8_t* buffer, int16_t number, int32_t *index); +void buffer_append_uint16(uint8_t* buffer, uint16_t number, int32_t *index); +void buffer_append_int32(uint8_t* buffer, int32_t number, int32_t *index); +void buffer_append_uint32(uint8_t* buffer, uint32_t number, int32_t *index); +void buffer_append_int64(uint8_t* buffer, int64_t number, int32_t *index); +void buffer_append_uint64(uint8_t* buffer, uint64_t number, int32_t *index); +void buffer_append_float16(uint8_t* buffer, float number, float scale, int32_t *index); +void buffer_append_float32(uint8_t* buffer, float number, float scale, int32_t *index); +void buffer_append_double64(uint8_t* buffer, double number, double scale, int32_t *index); +void buffer_append_float32_auto(uint8_t* buffer, float number, int32_t *index); +int16_t buffer_get_int16(const uint8_t *buffer, int32_t *index); +uint16_t buffer_get_uint16(const uint8_t *buffer, int32_t *index); +int32_t buffer_get_int32(const uint8_t *buffer, int32_t *index); +uint32_t buffer_get_uint32(const uint8_t *buffer, int32_t *index); +int64_t buffer_get_int64(const uint8_t *buffer, int32_t *index); +uint64_t buffer_get_uint64(const uint8_t *buffer, int32_t *index); +float buffer_get_float16(const uint8_t *buffer, float scale, int32_t *index); +float buffer_get_float32(const uint8_t *buffer, float scale, int32_t *index); +double buffer_get_double64(const uint8_t *buffer, double scale, int32_t *index); +float buffer_get_float32_auto(const uint8_t *buffer, int32_t *index); + +#endif /* BUFFER_H_ */ diff --git a/Node/External/Inc/crc.h b/Node/External/Inc/crc.h new file mode 100644 index 0000000..7d533fc --- /dev/null +++ b/Node/External/Inc/crc.h @@ -0,0 +1,33 @@ +/* + Copyright 2012-2014 Benjamin Vedder benjamin@vedder.se + + 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. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ + +/* + * crc.h + * + * Created on: 26 feb 2012 + * Author: benjamin + */ + +#ifndef CRC_H_ +#define CRC_H_ + +/* + * Functions + */ +unsigned short crc16(unsigned char *buf, unsigned int len); + +#endif /* CRC_H_ */ diff --git a/Node/External/Inc/datatypes.h b/Node/External/Inc/datatypes.h new file mode 100644 index 0000000..c3046b6 --- /dev/null +++ b/Node/External/Inc/datatypes.h @@ -0,0 +1,485 @@ +/* + Copyright 2012-2018 Benjamin Vedder benjamin@vedder.se + + 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. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ + +#ifndef DATATYPES_H_ +#define DATATYPES_H_ + +#include +#include + +// CAN status sent by VESC +typedef struct { + int id; + uint32_t rx_time; + float rpm; + float current; + float duty; +} can_status_msg; + +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 +} mc_pwm_mode; + +typedef enum { + COMM_MODE_INTEGRATE = 0, + COMM_MODE_DELAY +} mc_comm_mode; + +typedef enum { + SENSOR_MODE_SENSORLESS = 0, + SENSOR_MODE_SENSORED, + SENSOR_MODE_HYBRID +} mc_sensor_mode; + +// Auxiliary output mode +typedef enum { + OUT_AUX_MODE_OFF = 0, + OUT_AUX_MODE_ON_AFTER_2S, + OUT_AUX_MODE_ON_AFTER_5S, + OUT_AUX_MODE_ON_AFTER_10S +} out_aux_mode; + +typedef enum { + 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_DRV, + FAULT_CODE_ABS_OVER_CURRENT, + FAULT_CODE_OVER_TEMP_FET, + FAULT_CODE_OVER_TEMP_MOTOR +} mc_fault_code; + +typedef struct { + float v_in; + float temp_mos; + float temp_motor; + float current_motor; + float current_in; + float id; + float iq; + 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; + float pid_pos; + uint8_t vesc_id; +} mc_values; + +typedef enum { + SENSOR_PORT_MODE_HALL = 0, + SENSOR_PORT_MODE_ABI, + SENSOR_PORT_MODE_AS5047_SPI +} sensor_port_mode; + +typedef enum { + DRV8301_OC_LIMIT = 0, + DRV8301_OC_LATCH_SHUTDOWN, + DRV8301_OC_REPORT_ONLY, + DRV8301_OC_DISABLED +} drv8301_oc_mode; + +typedef enum { + MOTOR_CONTROL_DUTY = 0, + MOTOR_CONTROL_CURRENT, + MOTOR_CONTROL_CURRENT_BRAKE, + MOTOR_CONTROL_RPM, + MOTOR_CONTROL_POS +} motor_control_mode; + +typedef enum { + CAN_BAUD_125K = 0, + CAN_BAUD_250K, + CAN_BAUD_500K, + CAN_BAUD_1M +} CAN_BAUD; + +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_erpm_start; + 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; + float l_temp_fet_start; + float l_temp_fet_end; + float l_temp_motor_start; + float l_temp_motor_end; + float l_temp_accel_dec; + float l_min_duty; + float l_max_duty; + float l_watt_max; + float l_watt_min; + // Overridden limits (Computed during runtime) + float lo_current_max; + float lo_current_min; + float lo_in_current_max; + float lo_in_current_min; + float lo_current_motor_max_now; + float lo_current_motor_min_now; + // Sensorless (bldc) + 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_observer_gain_slow; + 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; + bool foc_sample_v0_v7; + bool foc_sample_high_current; + float foc_sat_comp; + bool foc_temp_comp; + float foc_temp_comp_base_temp; + float foc_current_filter_const; + // Speed PID + float s_pid_kp; + float s_pid_ki; + float s_pid_kd; + float s_pid_kd_filter; + float s_pid_min_erpm; + bool s_pid_allow_braking; + // Pos PID + float p_pid_kp; + float p_pid_ki; + float p_pid_kd; + float p_pid_kd_filter; + 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_current_backoff_gain; + uint32_t m_encoder_counts; + sensor_port_mode m_sensor_port_mode; + bool m_invert_direction; + drv8301_oc_mode m_drv8301_oc_mode; + int m_drv8301_oc_adj; + float m_bldc_f_sw_min; + float m_bldc_f_sw_max; + float m_dc_f_sw; + float m_ntc_motor_beta; + out_aux_mode m_out_aux_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_use; + +// Throttle curve mode +typedef enum { + THR_EXP_EXPO = 0, + THR_EXP_NATURAL, + THR_EXP_POLY +} thr_exp_mode; + +// 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_control_type; + +typedef struct { + ppm_control_type ctrl_type; + float pid_max_erpm; + float hyst; + float pulse_start; + float pulse_end; + float pulse_center; + bool median_filter; + bool safe_start; + float throttle_exp; + float throttle_exp_brake; + thr_exp_mode throttle_exp_mode; + float ramp_time_pos; + float ramp_time_neg; + 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_REV_BUTTON_BRAKE_ADC, + 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_CTRL_TYPE_PID, + ADC_CTRL_TYPE_PID_REV_CENTER, + ADC_CTRL_TYPE_PID_REV_BUTTON +} adc_control_type; + +typedef struct { + adc_control_type ctrl_type; + float hyst; + float voltage_start; + float voltage_end; + float voltage_center; + float voltage2_start; + float voltage2_end; + bool use_filter; + bool safe_start; + bool cc_button_inverted; + bool rev_button_inverted; + bool voltage_inverted; + bool voltage2_inverted; + float throttle_exp; + float throttle_exp_brake; + thr_exp_mode throttle_exp_mode; + float ramp_time_pos; + float ramp_time_neg; + 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_control_type; + +typedef struct { + chuk_control_type ctrl_type; + float hyst; + float ramp_time_pos; + float ramp_time_neg; + float stick_erpm_per_s_in_cc; + float throttle_exp; + float throttle_exp_brake; + thr_exp_mode throttle_exp_mode; + 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_OFF +} 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 { + 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; + +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; + CAN_BAUD can_baud_rate; + + // Application to use + app_use app_to_use; + + // PPM application settings + ppm_config app_ppm_conf; + + // ADC application settings + adc_config app_adc_conf; + + // UART application settings + uint32_t app_uart_baudrate; + + // 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_HANDBRAKE, + 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_NRF_START_PAIRING +} COMM_PACKET_ID; + +#endif /* DATATYPES_H_ */ diff --git a/Node/External/Inc/packet.h b/Node/External/Inc/packet.h new file mode 100644 index 0000000..348325c --- /dev/null +++ b/Node/External/Inc/packet.h @@ -0,0 +1,44 @@ +/* + Copyright 2012-2014 Benjamin Vedder benjamin@vedder.se + + 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. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ + +/* + * packet.h + * + * Created on: 21 mar 2013 + * Author: benjamin + */ + +#ifndef PACKET_H_ +#define PACKET_H_ + +#include + +#include "bldc_interface.h" + +// Settings +#define PACKET_RX_TIMEOUT 2 +#define PACKET_HANDLERS 3 +#define PACKET_MAX_PL_LEN 512 + +// Functions +void packet_init(void (*s_func)(unsigned char *data, unsigned int len), + void (*p_func)(BldcInterface* interface, unsigned char *data, unsigned int len), int handler_num); +void packet_process_byte(BldcInterface* interface, uint8_t rx_data, int handler_num); +void packet_timerfunc(void); +void packet_send_packet(unsigned char *data, unsigned int len, int handler_num); + +#endif /* PACKET_H_ */ diff --git a/Node/External/Src/bldc_interface.c b/Node/External/Src/bldc_interface.c new file mode 100644 index 0000000..60bc07c --- /dev/null +++ b/Node/External/Src/bldc_interface.c @@ -0,0 +1,873 @@ +/* + Copyright 2016-2018 Benjamin Vedder benjamin@vedder.se + + 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. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ + +/* + * bldc_interface.c + * + * Compatible Firmware Versions + * 3.39 + * 3.40 + * + */ + +#include "bldc_interface.h" +#include "buffer.h" +#include + + +// Private functions +void send_packet_no_fwd(BldcInterface* interface, unsigned char *data, unsigned int len); + +void bldc_interface_init(BldcInterface* interface, void(*func)(BldcInterface* interface, unsigned char *data, unsigned int len)) { + interface->send_func = func; +} + +void bldc_interface_set_forward_func(BldcInterface* interface, void(*func)(unsigned char *data, unsigned int len)) { + interface->forward_func = func; +} + +/** + * Send a packet using the set send function. + * + * @param data + * The packet data. + * + * @param len + * The data length. + */ +void bldc_interface_send_packet(BldcInterface* interface, unsigned char *data, unsigned int len) { + if (interface->send_func) { + interface->send_func(interface, data, len); + } +} + +/** + * Process a received buffer with commands and data. + * + * @param data + * The buffer to process. + * + * @param len + * The length of the buffer. + */ +void bldc_interface_process_packet(BldcInterface* interface, unsigned char *data, unsigned int len) { + if (!len) { + return; + } + + if (interface->forward_func) { + interface->forward_func(data, len); + return; + } + + int32_t ind = 0; + int i = 0; + unsigned char id = data[0]; + data++; + len--; + + switch (id) { + case COMM_FW_VERSION: + if (len == 2) { + ind = 0; + interface->fw_major = data[ind++]; + interface->fw_minor = data[ind++]; + } else { + interface->fw_major = -1; + interface->fw_minor = -1; + } + break; + + case COMM_ERASE_NEW_APP: + case COMM_WRITE_NEW_APP_DATA: + // TODO + break; + + case COMM_GET_VALUES: + ind = 0; + interface->values.temp_mos = buffer_get_float16(data, 1e1, &ind); + interface->values.temp_motor = buffer_get_float16(data, 1e1, &ind); + interface->values.current_motor = buffer_get_float32(data, 1e2, &ind); + interface->values.current_in = buffer_get_float32(data, 1e2, &ind); + interface->values.id = buffer_get_float32(data, 1e2, &ind); + interface->values.iq = buffer_get_float32(data, 1e2, &ind); + interface->values.duty_now = buffer_get_float16(data, 1e3, &ind); + interface->values.rpm = buffer_get_float32(data, 1e0, &ind); + interface->values.v_in = buffer_get_float16(data, 1e1, &ind); + interface->values.amp_hours = buffer_get_float32(data, 1e4, &ind); + interface->values.amp_hours_charged = buffer_get_float32(data, 1e4, &ind); + interface->values.watt_hours = buffer_get_float32(data, 1e4, &ind); + interface->values.watt_hours_charged = buffer_get_float32(data, 1e4, &ind); + interface->values.tachometer = buffer_get_int32(data, &ind); + interface->values.tachometer_abs = buffer_get_int32(data, &ind); + interface->values.fault_code = (mc_fault_code)data[ind++]; + + if (ind < (int)len) { + interface->values.pid_pos = buffer_get_float32(data, 1e6, &ind); + } else { + interface->values.pid_pos = 0.0; + } + + if (ind < (int)len) { + interface->values.vesc_id = data[ind++]; + } else { + interface->values.vesc_id = 255; + } + + if (interface->rx_value_func) { + interface->rx_value_func(&interface->values); + } + break; + + case COMM_PRINT: + if (interface->rx_printf_func) { + data[len] = '\0'; + interface->rx_printf_func((char*)data); + } + break; + + case COMM_SAMPLE_PRINT: + // TODO + break; + + case COMM_ROTOR_POSITION: + ind = 0; + interface->rotor_pos = buffer_get_float32(data, 100000.0, &ind); + + if (interface->rx_rotor_pos_func) { + interface->rx_rotor_pos_func(interface->rotor_pos); + } + break; + + case COMM_EXPERIMENT_SAMPLE: + // TODO + break; + + case COMM_GET_MCCONF: + case COMM_GET_MCCONF_DEFAULT: + ind = 0; + interface->mcconf.pwm_mode = data[ind++]; + interface->mcconf.comm_mode = data[ind++]; + interface->mcconf.motor_type = data[ind++]; + interface->mcconf.sensor_mode = data[ind++]; + + interface->mcconf.l_current_max = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_current_min = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_in_current_max = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_in_current_min = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_abs_current_max = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_min_erpm = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_max_erpm = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_erpm_start = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_max_erpm_fbrake = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_max_erpm_fbrake_cc = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_min_vin = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_max_vin = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_battery_cut_start = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_battery_cut_end = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_slow_abs_current = data[ind++]; + interface->mcconf.l_temp_fet_start = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_temp_fet_end = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_temp_motor_start = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_temp_motor_end = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_temp_accel_dec = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_min_duty = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_max_duty = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_watt_max = buffer_get_float32_auto(data, &ind); + interface->mcconf.l_watt_min = buffer_get_float32_auto(data, &ind); + + interface->mcconf.lo_current_max = interface->mcconf.l_current_max; + interface->mcconf.lo_current_min = interface->mcconf.l_current_min; + interface->mcconf.lo_in_current_max = interface->mcconf.l_in_current_max; + interface->mcconf.lo_in_current_min = interface->mcconf.l_in_current_min; + interface->mcconf.lo_current_motor_max_now = interface->mcconf.l_current_max; + interface->mcconf.lo_current_motor_min_now = interface->mcconf.l_current_min; + + interface->mcconf.sl_min_erpm = buffer_get_float32_auto(data, &ind); + interface->mcconf.sl_min_erpm_cycle_int_limit = buffer_get_float32_auto(data, &ind); + interface->mcconf.sl_max_fullbreak_current_dir_change = buffer_get_float32_auto(data, &ind); + interface->mcconf.sl_cycle_int_limit = buffer_get_float32_auto(data, &ind); + interface->mcconf.sl_phase_advance_at_br = buffer_get_float32_auto(data, &ind); + interface->mcconf.sl_cycle_int_rpm_br = buffer_get_float32_auto(data, &ind); + interface->mcconf.sl_bemf_coupling_k = buffer_get_float32_auto(data, &ind); + + memcpy(interface->mcconf.hall_table, data + ind, 8); + ind += 8; + interface->mcconf.hall_sl_erpm = buffer_get_float32_auto(data, &ind); + + interface->mcconf.foc_current_kp = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_current_ki = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_f_sw = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_dt_us = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_encoder_inverted = data[ind++]; + interface->mcconf.foc_encoder_offset = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_encoder_ratio = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_sensor_mode = data[ind++]; + interface->mcconf.foc_pll_kp = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_pll_ki = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_motor_l = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_motor_r = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_motor_flux_linkage = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_observer_gain = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_observer_gain_slow = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_duty_dowmramp_kp = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_duty_dowmramp_ki = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_openloop_rpm = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_sl_openloop_hyst = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_sl_openloop_time = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_sl_d_current_duty = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_sl_d_current_factor = buffer_get_float32_auto(data, &ind); + memcpy(interface->mcconf.foc_hall_table, data + ind, 8); + ind += 8; + interface->mcconf.foc_sl_erpm = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_sample_v0_v7 = data[ind++]; + interface->mcconf.foc_sample_high_current = data[ind++]; + interface->mcconf.foc_sat_comp = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_temp_comp = data[ind++]; + interface->mcconf.foc_temp_comp_base_temp = buffer_get_float32_auto(data, &ind); + interface->mcconf.foc_current_filter_const = buffer_get_float32_auto(data, &ind); + + interface->mcconf.s_pid_kp = buffer_get_float32_auto(data, &ind); + interface->mcconf.s_pid_ki = buffer_get_float32_auto(data, &ind); + interface->mcconf.s_pid_kd = buffer_get_float32_auto(data, &ind); + interface->mcconf.s_pid_kd_filter = buffer_get_float32_auto(data, &ind); + interface->mcconf.s_pid_min_erpm = buffer_get_float32_auto(data, &ind); + interface->mcconf.s_pid_allow_braking = data[ind++]; + + interface->mcconf.p_pid_kp = buffer_get_float32_auto(data, &ind); + interface->mcconf.p_pid_ki = buffer_get_float32_auto(data, &ind); + interface->mcconf.p_pid_kd = buffer_get_float32_auto(data, &ind); + interface->mcconf.p_pid_kd_filter = buffer_get_float32_auto(data, &ind); + interface->mcconf.p_pid_ang_div = buffer_get_float32_auto(data, &ind); + + interface->mcconf.cc_startup_boost_duty = buffer_get_float32_auto(data, &ind); + interface->mcconf.cc_min_current = buffer_get_float32_auto(data, &ind); + interface->mcconf.cc_gain = buffer_get_float32_auto(data, &ind); + interface->mcconf.cc_ramp_step_max = buffer_get_float32_auto(data, &ind); + + interface->mcconf.m_fault_stop_time_ms = buffer_get_int32(data, &ind); + interface->mcconf.m_duty_ramp_step = buffer_get_float32_auto(data, &ind); + interface->mcconf.m_current_backoff_gain = buffer_get_float32_auto(data, &ind); + interface->mcconf.m_encoder_counts = buffer_get_uint32(data, &ind); + interface->mcconf.m_sensor_port_mode = data[ind++]; + interface->mcconf.m_invert_direction = data[ind++]; + interface->mcconf.m_drv8301_oc_mode = data[ind++]; + interface->mcconf.m_drv8301_oc_adj = data[ind++]; + interface->mcconf.m_bldc_f_sw_min = buffer_get_float32_auto(data, &ind); + interface->mcconf.m_bldc_f_sw_max = buffer_get_float32_auto(data, &ind); + interface->mcconf.m_dc_f_sw = buffer_get_float32_auto(data, &ind); + interface->mcconf.m_ntc_motor_beta = buffer_get_float32_auto(data, &ind); + interface->mcconf.m_out_aux_mode = data[ind++]; + + if (interface->rx_mcconf_func) { + interface->rx_mcconf_func(&interface->mcconf); + } + break; + + case COMM_GET_APPCONF: + case COMM_GET_APPCONF_DEFAULT: + ind = 0; + interface->appconf.controller_id = data[ind++]; + interface->appconf.timeout_msec = buffer_get_uint32(data, &ind); + interface->appconf.timeout_brake_current = buffer_get_float32_auto(data, &ind); + interface->appconf.send_can_status = data[ind++]; + interface->appconf.send_can_status_rate_hz = buffer_get_uint16(data, &ind); + interface->appconf.can_baud_rate = data[ind++]; + + interface->appconf.app_to_use = data[ind++]; + + interface->appconf.app_ppm_conf.ctrl_type = data[ind++]; + interface->appconf.app_ppm_conf.pid_max_erpm = buffer_get_float32_auto(data, &ind); + interface->appconf.app_ppm_conf.hyst = buffer_get_float32_auto(data, &ind); + interface->appconf.app_ppm_conf.pulse_start = buffer_get_float32_auto(data, &ind); + interface->appconf.app_ppm_conf.pulse_end = buffer_get_float32_auto(data, &ind); + interface->appconf.app_ppm_conf.pulse_center = buffer_get_float32_auto(data, &ind); + interface->appconf.app_ppm_conf.median_filter = data[ind++]; + interface->appconf.app_ppm_conf.safe_start = data[ind++]; + interface->appconf.app_ppm_conf.throttle_exp = buffer_get_float32_auto(data, &ind); + interface->appconf.app_ppm_conf.throttle_exp_brake = buffer_get_float32_auto(data, &ind); + interface->appconf.app_ppm_conf.throttle_exp_mode = data[ind++]; + interface->appconf.app_ppm_conf.ramp_time_pos = buffer_get_float32_auto(data, &ind); + interface->appconf.app_ppm_conf.ramp_time_neg = buffer_get_float32_auto(data, &ind); + interface->appconf.app_ppm_conf.multi_esc = data[ind++]; + interface->appconf.app_ppm_conf.tc = data[ind++]; + interface->appconf.app_ppm_conf.tc_max_diff = buffer_get_float32_auto(data, &ind); + + interface->appconf.app_adc_conf.ctrl_type = data[ind++]; + interface->appconf.app_adc_conf.hyst = buffer_get_float32_auto(data, &ind); + interface->appconf.app_adc_conf.voltage_start = buffer_get_float32_auto(data, &ind); + interface->appconf.app_adc_conf.voltage_end = buffer_get_float32_auto(data, &ind); + interface->appconf.app_adc_conf.voltage_center = buffer_get_float32_auto(data, &ind); + interface->appconf.app_adc_conf.voltage2_start = buffer_get_float32_auto(data, &ind); + interface->appconf.app_adc_conf.voltage2_end = buffer_get_float32_auto(data, &ind); + interface->appconf.app_adc_conf.use_filter = data[ind++]; + interface->appconf.app_adc_conf.safe_start = data[ind++]; + interface->appconf.app_adc_conf.cc_button_inverted = data[ind++]; + interface->appconf.app_adc_conf.rev_button_inverted = data[ind++]; + interface->appconf.app_adc_conf.voltage_inverted = data[ind++]; + interface->appconf.app_adc_conf.voltage2_inverted = data[ind++]; + interface->appconf.app_adc_conf.throttle_exp = buffer_get_float32_auto(data, &ind); + interface->appconf.app_adc_conf.throttle_exp_brake = buffer_get_float32_auto(data, &ind); + interface->appconf.app_adc_conf.throttle_exp_mode = data[ind++]; + interface->appconf.app_adc_conf.ramp_time_pos = buffer_get_float32_auto(data, &ind); + interface->appconf.app_adc_conf.ramp_time_neg = buffer_get_float32_auto(data, &ind); + interface->appconf.app_adc_conf.multi_esc = data[ind++]; + interface->appconf.app_adc_conf.tc = data[ind++]; + interface->appconf.app_adc_conf.tc_max_diff = buffer_get_float32_auto(data, &ind); + interface->appconf.app_adc_conf.update_rate_hz = buffer_get_uint16(data, &ind); + + interface->appconf.app_uart_baudrate = buffer_get_uint32(data, &ind); + + interface->appconf.app_chuk_conf.ctrl_type = data[ind++]; + interface->appconf.app_chuk_conf.hyst = buffer_get_float32_auto(data, &ind); + interface->appconf.app_chuk_conf.ramp_time_pos = buffer_get_float32_auto(data, &ind); + interface->appconf.app_chuk_conf.ramp_time_neg = buffer_get_float32_auto(data, &ind); + interface->appconf.app_chuk_conf.stick_erpm_per_s_in_cc = buffer_get_float32_auto(data, &ind); + interface->appconf.app_chuk_conf.throttle_exp = buffer_get_float32_auto(data, &ind); + interface->appconf.app_chuk_conf.throttle_exp_brake = buffer_get_float32_auto(data, &ind); + interface->appconf.app_chuk_conf.throttle_exp_mode = data[ind++]; + interface->appconf.app_chuk_conf.multi_esc = data[ind++]; + interface->appconf.app_chuk_conf.tc = data[ind++]; + interface->appconf.app_chuk_conf.tc_max_diff = buffer_get_float32_auto(data, &ind); + + interface->appconf.app_nrf_conf.speed = data[ind++]; + interface->appconf.app_nrf_conf.power = data[ind++]; + interface->appconf.app_nrf_conf.crc_type = data[ind++]; + interface->appconf.app_nrf_conf.retry_delay = data[ind++]; + interface->appconf.app_nrf_conf.retries = data[ind++]; + interface->appconf.app_nrf_conf.channel = data[ind++]; + memcpy(interface->appconf.app_nrf_conf.address, data + ind, 3); + ind += 3; + interface->appconf.app_nrf_conf.send_crc_ack = data[ind++]; + + if (interface->rx_appconf_func) { + interface->rx_appconf_func(&interface->appconf); + } + break; + + case COMM_DETECT_MOTOR_PARAM: + ind = 0; + interface->detect_cycle_int_limit = buffer_get_float32(data, 1000.0, &ind); + interface->detect_coupling_k = buffer_get_float32(data, 1000.0, &ind); + for (i = 0;i < 8;i++) { + interface->detect_hall_table[i] = (const signed char)(data[ind++]); + } + interface->detect_hall_res = (const signed char)(data[ind++]); + + if (interface->rx_detect_func) { + interface->rx_detect_func(interface->detect_cycle_int_limit, + interface->detect_coupling_k, + interface->detect_hall_table, + interface->detect_hall_res); + } + break; + + case COMM_DETECT_MOTOR_R_L: { + // TODO! + } break; + + case COMM_DETECT_MOTOR_FLUX_LINKAGE: { + // TODO! + } break; + + case COMM_DETECT_ENCODER: { + // TODO! + } break; + + case COMM_DETECT_HALL_FOC: { + // TODO! + } break; + + case COMM_GET_DECODED_PPM: + ind = 0; + interface->dec_ppm = buffer_get_float32(data, 1000000.0, &ind); + interface->dec_ppm_len = buffer_get_float32(data, 1000000.0, &ind); + + if (interface->rx_dec_ppm_func) { + interface->rx_dec_ppm_func(interface->dec_ppm, interface->dec_ppm_len); + } + break; + + case COMM_GET_DECODED_ADC: + ind = 0; + interface->dec_adc = buffer_get_float32(data, 1000000.0, &ind); + interface->dec_adc_voltage = buffer_get_float32(data, 1000000.0, &ind); + // TODO for adc2 + + if (interface->rx_dec_adc_func) { + interface->rx_dec_adc_func(interface->dec_adc, interface->dec_adc_voltage); + } + break; + + case COMM_GET_DECODED_CHUK: + ind = 0; + interface->dec_chuk = buffer_get_float32(data, 1000000.0, &ind); + + if (interface->rx_dec_chuk_func) { + interface->rx_dec_chuk_func(interface->dec_chuk); + } + break; + + case COMM_SET_MCCONF: + // This is a confirmation that the new mcconf is received. + if (interface->rx_mcconf_received_func) { + interface->rx_mcconf_received_func(); + } + break; + + case COMM_SET_APPCONF: + // This is a confirmation that the new appconf is received. + if (interface->rx_appconf_received_func) { + interface->rx_appconf_received_func(); + } + break; + + default: + break; + } +} + +/** + * Function pointer setters. When data that is requested with the get functions + * is received, the corresponding function pointer will be called with the + * received data. + * + * @param func + * A function to be called when the corresponding data is received. + */ + +void bldc_interface_set_rx_value_func(BldcInterface* interface, void(*func)(mc_values *values)) { + interface->rx_value_func = func; +} + +void bldc_interface_set_rx_printf_func(BldcInterface* interface, void(*func)(char *str)) { + interface->rx_printf_func = func; +} + +void bldc_interface_set_rx_fw_func(BldcInterface* interface, void(*func)(int major, int minor)) { + interface->rx_fw_func = func; +} + +void bldc_interface_set_rx_rotor_pos_func(BldcInterface* interface, void(*func)(float pos)) { + interface->rx_rotor_pos_func = func; +} + +void bldc_interface_set_rx_mcconf_func(BldcInterface* interface, void(*func)(mc_configuration *conf)) { + interface->rx_mcconf_func = func; +} + +void bldc_interface_set_rx_appconf_func(BldcInterface* interface, void(*func)(app_configuration *conf)) { + interface->rx_appconf_func = func; +} + +void bldc_interface_set_rx_detect_func(BldcInterface* interface, void(*func)(float cycle_int_limit, float coupling_k, + const signed char *hall_table, signed char hall_res)) { + interface->rx_detect_func = func; +} + +void bldc_interface_set_rx_dec_ppm_func(BldcInterface* interface, void(*func)(float val, float ms)) { + interface->rx_dec_ppm_func = func; +} + +void bldc_interface_set_rx_dec_adc_func(BldcInterface* interface, void(*func)(float val, float voltage)) { + interface->rx_dec_adc_func = func; +} + +void bldc_interface_set_rx_dec_chuk_func(BldcInterface* interface, void(*func)(float val)) { + interface->rx_dec_chuk_func = func; +} + +void bldc_interface_set_rx_mcconf_received_func(BldcInterface* interface, void(*func)(void)) { + interface->rx_mcconf_received_func = func; +} + +void bldc_interface_set_rx_appconf_received_func(BldcInterface* interface, void(*func)(void)) { + interface->rx_appconf_received_func = func; +} + +void bldc_interface_set_sim_control_function(BldcInterface* interface, void(*func)(motor_control_mode mode, float value)) { + interface->motor_control_set_func = func; +} + +void bldc_interface_set_sim_values_func(BldcInterface* interface, void(*func)(void)) { + interface->values_requested_func = func; +} + +// Setters +void bldc_interface_terminal_cmd(BldcInterface* interface, char* cmd) { + int len = strlen(cmd); + interface->send_buffer[0] = COMM_TERMINAL_CMD; + memcpy(interface->send_buffer + 1, cmd, len); + send_packet_no_fwd(interface, interface->send_buffer, len + 1); +} + +void bldc_interface_set_duty_cycle(BldcInterface* interface, float dutyCycle) { + if (interface->motor_control_set_func) { + interface->motor_control_set_func(MOTOR_CONTROL_DUTY, dutyCycle); + return; + } + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_SET_DUTY; + buffer_append_float32(interface->send_buffer, dutyCycle, 100000.0, &send_index); + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_set_current(BldcInterface* interface, float current) { + if (interface->motor_control_set_func) { + interface->motor_control_set_func(MOTOR_CONTROL_CURRENT, current); + return; + } + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_SET_CURRENT; + buffer_append_float32(interface->send_buffer, current, 1000.0, &send_index); + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_set_current_brake(BldcInterface* interface, float current) { + if (interface->motor_control_set_func) { + interface->motor_control_set_func(MOTOR_CONTROL_CURRENT_BRAKE, current); + return; + } + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_SET_CURRENT_BRAKE; + buffer_append_float32(interface->send_buffer, current, 1000.0, &send_index); + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_set_rpm(BldcInterface* interface, int rpm) { + if (interface->motor_control_set_func) { + interface->motor_control_set_func(MOTOR_CONTROL_RPM, rpm); + return; + } + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_SET_RPM; + buffer_append_int32(interface->send_buffer, rpm, &send_index); + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_set_pos(BldcInterface* interface, float pos) { + if (interface->motor_control_set_func) { + interface->motor_control_set_func(MOTOR_CONTROL_POS, pos); + return; + } + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_SET_POS; + buffer_append_float32(interface->send_buffer, pos, 1000000.0, &send_index); + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_set_handbrake(BldcInterface* interface, float current) { + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_SET_HANDBRAKE; + buffer_append_float32(interface->send_buffer, current, 1e3, &send_index); + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_set_servo_pos(BldcInterface* interface, float pos) { + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_SET_SERVO_POS; + buffer_append_float16(interface->send_buffer, pos, 1000.0, &send_index); + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_set_mcconf(BldcInterface* interface, const mc_configuration *mcconf) { + int32_t ind = 0; + interface->send_buffer[ind++] = COMM_SET_MCCONF; + + interface->send_buffer[ind++] = mcconf->pwm_mode; + interface->send_buffer[ind++] = mcconf->comm_mode; + interface->send_buffer[ind++] = mcconf->motor_type; + interface->send_buffer[ind++] = mcconf->sensor_mode; + + buffer_append_float32_auto(interface->send_buffer, mcconf->l_current_max, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_current_min, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_in_current_max, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_in_current_min, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_abs_current_max, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_min_erpm, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_max_erpm, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_erpm_start, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_max_erpm_fbrake, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_max_erpm_fbrake_cc, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_min_vin, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_max_vin, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_battery_cut_start, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_battery_cut_end, &ind); + interface->send_buffer[ind++] = mcconf->l_slow_abs_current; + buffer_append_float32_auto(interface->send_buffer, mcconf->l_temp_fet_start, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_temp_fet_end, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_temp_motor_start, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_temp_motor_end, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_temp_accel_dec, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_min_duty, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_max_duty, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_watt_max, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->l_watt_min, &ind); + + buffer_append_float32_auto(interface->send_buffer, mcconf->sl_min_erpm, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->sl_min_erpm_cycle_int_limit, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->sl_max_fullbreak_current_dir_change, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->sl_cycle_int_limit, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->sl_phase_advance_at_br, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->sl_cycle_int_rpm_br, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->sl_bemf_coupling_k, &ind); + + memcpy(interface->send_buffer + ind, mcconf->hall_table, 8); + ind += 8; + buffer_append_float32_auto(interface->send_buffer, mcconf->hall_sl_erpm, &ind); + + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_current_kp, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_current_ki, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_f_sw, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_dt_us, &ind); + interface->send_buffer[ind++] = mcconf->foc_encoder_inverted; + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_encoder_offset, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_encoder_ratio, &ind); + interface->send_buffer[ind++] = mcconf->foc_sensor_mode; + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_pll_kp, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_pll_ki, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_motor_l, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_motor_r, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_motor_flux_linkage, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_observer_gain, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_observer_gain_slow, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_duty_dowmramp_kp, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_duty_dowmramp_ki, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_openloop_rpm, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_sl_openloop_hyst, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_sl_openloop_time, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_sl_d_current_duty, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_sl_d_current_factor, &ind); + memcpy(interface->send_buffer + ind, mcconf->foc_hall_table, 8); + ind += 8; + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_sl_erpm, &ind); + interface->send_buffer[ind++] = mcconf->foc_sample_v0_v7; + interface->send_buffer[ind++] = mcconf->foc_sample_high_current; + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_sat_comp, &ind); + interface->send_buffer[ind++] = mcconf->foc_temp_comp; + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_temp_comp_base_temp, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->foc_current_filter_const, &ind); + + buffer_append_float32_auto(interface->send_buffer, mcconf->s_pid_kp, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->s_pid_ki, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->s_pid_kd, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->s_pid_kd_filter, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->s_pid_min_erpm, &ind); + interface->send_buffer[ind++] = mcconf->s_pid_allow_braking; + + buffer_append_float32_auto(interface->send_buffer, mcconf->p_pid_kp, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->p_pid_ki, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->p_pid_kd, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->p_pid_kd_filter, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->p_pid_ang_div, &ind); + + buffer_append_float32_auto(interface->send_buffer, mcconf->cc_startup_boost_duty, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->cc_min_current, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->cc_gain, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->cc_ramp_step_max, &ind); + + buffer_append_int32(interface->send_buffer, mcconf->m_fault_stop_time_ms, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->m_duty_ramp_step, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->m_current_backoff_gain, &ind); + buffer_append_uint32(interface->send_buffer, mcconf->m_encoder_counts, &ind); + interface->send_buffer[ind++] = mcconf->m_sensor_port_mode; + interface->send_buffer[ind++] = mcconf->m_invert_direction; + interface->send_buffer[ind++] = mcconf->m_drv8301_oc_mode; + interface->send_buffer[ind++] = mcconf->m_drv8301_oc_adj; + buffer_append_float32_auto(interface->send_buffer, mcconf->m_bldc_f_sw_min, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->m_bldc_f_sw_max, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->m_dc_f_sw, &ind); + buffer_append_float32_auto(interface->send_buffer, mcconf->m_ntc_motor_beta, &ind); + interface->send_buffer[ind++] = mcconf->m_out_aux_mode; + + send_packet_no_fwd(interface, interface->send_buffer, ind); +} + +void bldc_interface_set_appconf(BldcInterface* interface, const app_configuration *appconf) { + int32_t ind = 0; + interface->send_buffer[ind++] = COMM_SET_APPCONF; + interface->send_buffer[ind++] = appconf->controller_id; + buffer_append_uint32(interface->send_buffer, appconf->timeout_msec, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->timeout_brake_current, &ind); + interface->send_buffer[ind++] = appconf->send_can_status; + buffer_append_uint16(interface->send_buffer, appconf->send_can_status_rate_hz, &ind); + interface->send_buffer[ind++] = appconf->can_baud_rate; + + interface->send_buffer[ind++] = appconf->app_to_use; + + interface->send_buffer[ind++] = appconf->app_ppm_conf.ctrl_type; + buffer_append_float32_auto(interface->send_buffer, appconf->app_ppm_conf.pid_max_erpm, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_ppm_conf.hyst, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_ppm_conf.pulse_start, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_ppm_conf.pulse_end, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_ppm_conf.pulse_center, &ind); + interface->send_buffer[ind++] = appconf->app_ppm_conf.median_filter; + interface->send_buffer[ind++] = appconf->app_ppm_conf.safe_start; + buffer_append_float32_auto(interface->send_buffer, appconf->app_ppm_conf.throttle_exp, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_ppm_conf.throttle_exp_brake, &ind); + interface->send_buffer[ind++] = appconf->app_ppm_conf.throttle_exp_mode; + buffer_append_float32_auto(interface->send_buffer, appconf->app_ppm_conf.ramp_time_pos, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_ppm_conf.ramp_time_neg, &ind); + interface->send_buffer[ind++] = appconf->app_ppm_conf.multi_esc; + interface->send_buffer[ind++] = appconf->app_ppm_conf.tc; + buffer_append_float32_auto(interface->send_buffer, appconf->app_ppm_conf.tc_max_diff, &ind); + + interface->send_buffer[ind++] = appconf->app_adc_conf.ctrl_type; + buffer_append_float32_auto(interface->send_buffer, appconf->app_adc_conf.hyst, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_adc_conf.voltage_start, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_adc_conf.voltage_end, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_adc_conf.voltage_center, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_adc_conf.voltage2_start, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_adc_conf.voltage2_end, &ind); + interface->send_buffer[ind++] = appconf->app_adc_conf.use_filter; + interface->send_buffer[ind++] = appconf->app_adc_conf.safe_start; + interface->send_buffer[ind++] = appconf->app_adc_conf.cc_button_inverted; + interface->send_buffer[ind++] = appconf->app_adc_conf.rev_button_inverted; + interface->send_buffer[ind++] = appconf->app_adc_conf.voltage_inverted; + interface->send_buffer[ind++] = appconf->app_adc_conf.voltage2_inverted; + buffer_append_float32_auto(interface->send_buffer, appconf->app_adc_conf.throttle_exp, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_adc_conf.throttle_exp_brake, &ind); + interface->send_buffer[ind++] = appconf->app_adc_conf.throttle_exp_mode; + buffer_append_float32_auto(interface->send_buffer, appconf->app_adc_conf.ramp_time_pos, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_adc_conf.ramp_time_neg, &ind); + interface->send_buffer[ind++] = appconf->app_adc_conf.multi_esc; + interface->send_buffer[ind++] = appconf->app_adc_conf.tc; + buffer_append_float32_auto(interface->send_buffer, appconf->app_adc_conf.tc_max_diff, &ind); + buffer_append_uint16(interface->send_buffer, appconf->app_adc_conf.update_rate_hz, &ind); + + buffer_append_uint32(interface->send_buffer, appconf->app_uart_baudrate, &ind); + + interface->send_buffer[ind++] = appconf->app_chuk_conf.ctrl_type; + buffer_append_float32_auto(interface->send_buffer, appconf->app_chuk_conf.hyst, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_chuk_conf.ramp_time_pos, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_chuk_conf.ramp_time_neg, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_chuk_conf.stick_erpm_per_s_in_cc, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_chuk_conf.throttle_exp, &ind); + buffer_append_float32_auto(interface->send_buffer, appconf->app_chuk_conf.throttle_exp_brake, &ind); + interface->send_buffer[ind++] = appconf->app_chuk_conf.throttle_exp_mode; + interface->send_buffer[ind++] = appconf->app_chuk_conf.multi_esc; + interface->send_buffer[ind++] = appconf->app_chuk_conf.tc; + buffer_append_float32_auto(interface->send_buffer, appconf->app_chuk_conf.tc_max_diff, &ind); + + interface->send_buffer[ind++] = appconf->app_nrf_conf.speed; + interface->send_buffer[ind++] = appconf->app_nrf_conf.power; + interface->send_buffer[ind++] = appconf->app_nrf_conf.crc_type; + interface->send_buffer[ind++] = appconf->app_nrf_conf.retry_delay; + interface->send_buffer[ind++] = appconf->app_nrf_conf.retries; + interface->send_buffer[ind++] = appconf->app_nrf_conf.channel; + memcpy(interface->send_buffer + ind, appconf->app_nrf_conf.address, 3); + ind += 3; + interface->send_buffer[ind++] = appconf->app_nrf_conf.send_crc_ack; + + send_packet_no_fwd(interface, interface->send_buffer, ind); +} + +// Getters +void bldc_interface_get_fw_version(BldcInterface* interface) { + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_FW_VERSION; + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_get_values(BldcInterface* interface) { + if (interface->values_requested_func) { + interface->values_requested_func(); + return; + } + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_GET_VALUES; + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_get_mcconf(BldcInterface* interface) { + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_GET_MCCONF; + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_get_appconf(BldcInterface* interface) { + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_GET_APPCONF; + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_get_decoded_ppm(BldcInterface* interface) { + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_GET_DECODED_PPM; + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_get_decoded_adc(BldcInterface* interface) { + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_GET_DECODED_ADC; + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_get_decoded_chuk(BldcInterface* interface) { + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_GET_DECODED_CHUK; + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +// Other functions +void bldc_interface_detect_motor_param(BldcInterface* interface, float current, float min_rpm, float low_duty) { + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_DETECT_MOTOR_PARAM; + buffer_append_float32(interface->send_buffer, current, 1000.0, &send_index); + buffer_append_float32(interface->send_buffer, min_rpm, 1000.0, &send_index); + buffer_append_float32(interface->send_buffer, low_duty, 1000.0, &send_index); + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_reboot(BldcInterface* interface) { + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_REBOOT; + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void bldc_interface_send_alive(BldcInterface* interface) { + int32_t send_index = 0; + interface->send_buffer[send_index++] = COMM_ALIVE; + send_packet_no_fwd(interface, interface->send_buffer, send_index); +} + +void send_values_to_receiver(BldcInterface* interface, mc_values *values) { + if (interface->rx_value_func) { + interface->rx_value_func(values); + } +} + +// Helpers +const char* bldc_interface_fault_to_string(mc_fault_code fault) { + switch (fault) { + case FAULT_CODE_NONE: return "FAULT_CODE_NONE"; + case FAULT_CODE_OVER_VOLTAGE: return "FAULT_CODE_OVER_VOLTAGE"; + case FAULT_CODE_UNDER_VOLTAGE: return "FAULT_CODE_UNDER_VOLTAGE"; + case FAULT_CODE_DRV: return "FAULT_CODE_DRV"; + case FAULT_CODE_ABS_OVER_CURRENT: return "FAULT_CODE_ABS_OVER_CURRENT"; + case FAULT_CODE_OVER_TEMP_FET: return "FAULT_CODE_OVER_TEMP_FET"; + case FAULT_CODE_OVER_TEMP_MOTOR: return "FAULT_CODE_OVER_TEMP_MOTOR"; + default: return "Unknown fault"; + } +} + +// Private functions +void send_packet_no_fwd(BldcInterface* interface, unsigned char *data, unsigned int len) { + if (!interface->forward_func) { + bldc_interface_send_packet(interface, data, len); + } +} diff --git a/Node/External/Src/bldc_interface_uart.c b/Node/External/Src/bldc_interface_uart.c new file mode 100644 index 0000000..6587266 --- /dev/null +++ b/Node/External/Src/bldc_interface_uart.c @@ -0,0 +1,99 @@ +/* + Copyright 2015 Benjamin Vedder benjamin@vedder.se + + 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. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ + +/* + * bldc_interface_uart.c + * + * Created on: 9 okt 2015 + * Author: benjamin + */ + +#include "bldc_interface_uart.h" +#include "bldc_interface.h" + +// Settings +#define PACKET_HANDLER 0 + +// Private functions +static void process_packet(BldcInterface* interface, unsigned char *data, unsigned int len); +static void send_packet_bldc_interface(BldcInterface* interface, unsigned char *data, unsigned int len); + +/** + * Initialize the UART BLDC interface and provide a function to be used for + * sending packets. + * + * @param func + * Function provided for sending packets. + */ +void bldc_interface_uart_init(BldcInterface* interface, void(*func)(unsigned char *data, unsigned int len)) { + // Initialize packet handler + static int packet_handler_num = 0; + + interface->packet_handler_num = packet_handler_num++; + packet_init(func, process_packet, interface->packet_handler_num); + + // Initialize the bldc interface and provide a send function + bldc_interface_init(interface, send_packet_bldc_interface); +} + +/** + * Process one byte received on the UART. Once a full packet is received the + * corresponding callback will be called by bldc_interface. + * + * @param b + * The byte received on the UART to process. + */ +void bldc_interface_uart_process_byte(BldcInterface* interface, unsigned char b) { + packet_process_byte(interface, b, interface->packet_handler_num); +} + +/** + * Call this function at around 1 khz to reset the state of the packet + * interface after a timeout in case data is lost. + */ +void bldc_interface_uart_run_timer(BldcInterface* interface) { + packet_timerfunc(); +} + +/** + * Callback for the packet handled for when a whole packet is received, + * assembled and checked. + * + * @param data + * Data array pointer + * @param len + * Data array length + */ +static void process_packet(BldcInterface* interface, unsigned char *data, unsigned int len) { + // Let bldc_interface process the packet. + bldc_interface_process_packet(interface, data, len); +} + +/** + * Callback that bldc_interface uses to send packets. + * + * @param data + * Data array pointer + * @param len + * Data array length + */ +static void send_packet_bldc_interface(BldcInterface* interface, unsigned char *data, unsigned int len) { + // Pass the packet to the packet handler to add checksum, length, start and stop bytes. + packet_send_packet(data, len, interface->packet_handler_num); +} + + diff --git a/Node/External/Src/buffer.c b/Node/External/Src/buffer.c new file mode 100644 index 0000000..a001925 --- /dev/null +++ b/Node/External/Src/buffer.c @@ -0,0 +1,187 @@ +/* + Copyright 2012-2018 Benjamin Vedder benjamin@vedder.se + + 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. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ + +#include "buffer.h" +#include +#include + +void buffer_append_int16(uint8_t* buffer, int16_t number, int32_t *index) { + buffer[(*index)++] = number >> 8; + buffer[(*index)++] = number; +} + +void buffer_append_uint16(uint8_t* buffer, uint16_t number, int32_t *index) { + buffer[(*index)++] = number >> 8; + buffer[(*index)++] = number; +} + +void buffer_append_int32(uint8_t* buffer, int32_t number, int32_t *index) { + buffer[(*index)++] = number >> 24; + buffer[(*index)++] = number >> 16; + buffer[(*index)++] = number >> 8; + buffer[(*index)++] = number; +} + +void buffer_append_uint32(uint8_t* buffer, uint32_t number, int32_t *index) { + buffer[(*index)++] = number >> 24; + buffer[(*index)++] = number >> 16; + buffer[(*index)++] = number >> 8; + buffer[(*index)++] = number; +} + +void buffer_append_int64(uint8_t* buffer, int64_t number, int32_t *index) { + buffer[(*index)++] = number >> 56; + buffer[(*index)++] = number >> 48; + buffer[(*index)++] = number >> 40; + buffer[(*index)++] = number >> 32; + buffer[(*index)++] = number >> 24; + buffer[(*index)++] = number >> 16; + buffer[(*index)++] = number >> 8; + buffer[(*index)++] = number; +} + +void buffer_append_uint64(uint8_t* buffer, uint64_t number, int32_t *index) { + buffer[(*index)++] = number >> 56; + buffer[(*index)++] = number >> 48; + buffer[(*index)++] = number >> 40; + buffer[(*index)++] = number >> 32; + buffer[(*index)++] = number >> 24; + buffer[(*index)++] = number >> 16; + buffer[(*index)++] = number >> 8; + buffer[(*index)++] = number; +} + +void buffer_append_float16(uint8_t* buffer, float number, float scale, int32_t *index) { + buffer_append_int16(buffer, (int16_t)(number * scale), index); +} + +void buffer_append_float32(uint8_t* buffer, float number, float scale, int32_t *index) { + buffer_append_int32(buffer, (int32_t)(number * scale), index); +} + +void buffer_append_double64(uint8_t* buffer, double number, double scale, int32_t *index) { + buffer_append_int64(buffer, (int64_t)(number * scale), index); +} + +void buffer_append_float32_auto(uint8_t* buffer, float number, int32_t *index) { + int e = 0; + float sig = frexpf(number, &e); + float sig_abs = fabsf(sig); + uint32_t sig_i = 0; + + if (sig_abs >= 0.5) { + sig_i = (uint32_t)((sig_abs - 0.5f) * 2.0f * 8388608.0f); + e += 126; + } + + uint32_t res = ((e & 0xFF) << 23) | (sig_i & 0x7FFFFF); + if (sig < 0) { + res |= 1 << 31; + } + + buffer_append_uint32(buffer, res, index); +} + +int16_t buffer_get_int16(const uint8_t *buffer, int32_t *index) { + int16_t res = ((uint16_t) buffer[*index]) << 8 | + ((uint16_t) buffer[*index + 1]); + *index += 2; + return res; +} + +uint16_t buffer_get_uint16(const uint8_t *buffer, int32_t *index) { + uint16_t res = ((uint16_t) buffer[*index]) << 8 | + ((uint16_t) buffer[*index + 1]); + *index += 2; + return res; +} + +int32_t buffer_get_int32(const uint8_t *buffer, int32_t *index) { + int32_t res = ((uint32_t) buffer[*index]) << 24 | + ((uint32_t) buffer[*index + 1]) << 16 | + ((uint32_t) buffer[*index + 2]) << 8 | + ((uint32_t) buffer[*index + 3]); + *index += 4; + return res; +} + +uint32_t buffer_get_uint32(const uint8_t *buffer, int32_t *index) { + uint32_t res = ((uint32_t) buffer[*index]) << 24 | + ((uint32_t) buffer[*index + 1]) << 16 | + ((uint32_t) buffer[*index + 2]) << 8 | + ((uint32_t) buffer[*index + 3]); + *index += 4; + return res; +} + +int64_t buffer_get_int64(const uint8_t *buffer, int32_t *index) { + int64_t res = ((uint64_t) buffer[*index]) << 56 | + ((uint64_t) buffer[*index + 1]) << 48 | + ((uint64_t) buffer[*index + 2]) << 40 | + ((uint64_t) buffer[*index + 3]) << 32 | + ((uint64_t) buffer[*index + 4]) << 24 | + ((uint64_t) buffer[*index + 5]) << 16 | + ((uint64_t) buffer[*index + 6]) << 8 | + ((uint64_t) buffer[*index + 7]); + *index += 8; + return res; +} + +uint64_t buffer_get_uint64(const uint8_t *buffer, int32_t *index) { + uint64_t res = ((uint64_t) buffer[*index]) << 56 | + ((uint64_t) buffer[*index + 1]) << 48 | + ((uint64_t) buffer[*index + 2]) << 40 | + ((uint64_t) buffer[*index + 3]) << 32 | + ((uint64_t) buffer[*index + 4]) << 24 | + ((uint64_t) buffer[*index + 5]) << 16 | + ((uint64_t) buffer[*index + 6]) << 8 | + ((uint64_t) buffer[*index + 7]); + *index += 8; + return res; +} + +float buffer_get_float16(const uint8_t *buffer, float scale, int32_t *index) { + return (float)buffer_get_int16(buffer, index) / scale; +} + +float buffer_get_float32(const uint8_t *buffer, float scale, int32_t *index) { + return (float)buffer_get_int32(buffer, index) / scale; +} + +double buffer_get_double64(const uint8_t *buffer, double scale, int32_t *index) { + return (double)buffer_get_int64(buffer, index) / scale; +} + +float buffer_get_float32_auto(const uint8_t *buffer, int32_t *index) { + uint32_t res = buffer_get_uint32(buffer, index); + + int e = (res >> 23) & 0xFF; + uint32_t sig_i = res & 0x7FFFFF; + bool neg = res & (1 << 31); + + float sig = 0.0; + if (e != 0 || sig_i != 0) { + sig = (float)sig_i / (8388608.0 * 2.0) + 0.5; + e -= 126; + } + + if (neg) { + sig = -sig; + } + + return ldexpf(sig, e); +} diff --git a/Node/External/Src/crc.c b/Node/External/Src/crc.c new file mode 100644 index 0000000..9768b84 --- /dev/null +++ b/Node/External/Src/crc.c @@ -0,0 +1,64 @@ +/* + Copyright 2012-2014 Benjamin Vedder benjamin@vedder.se + + 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. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ + +/* + * crc.c + * + * Created on: 26 feb 2012 + * Author: benjamin + */ +#include "crc.h" + +// CRC Table +const unsigned short crc16_tab[] = { 0x0000, 0x1021, 0x2042, 0x3063, 0x4084, + 0x50a5, 0x60c6, 0x70e7, 0x8108, 0x9129, 0xa14a, 0xb16b, 0xc18c, 0xd1ad, + 0xe1ce, 0xf1ef, 0x1231, 0x0210, 0x3273, 0x2252, 0x52b5, 0x4294, 0x72f7, + 0x62d6, 0x9339, 0x8318, 0xb37b, 0xa35a, 0xd3bd, 0xc39c, 0xf3ff, 0xe3de, + 0x2462, 0x3443, 0x0420, 0x1401, 0x64e6, 0x74c7, 0x44a4, 0x5485, 0xa56a, + 0xb54b, 0x8528, 0x9509, 0xe5ee, 0xf5cf, 0xc5ac, 0xd58d, 0x3653, 0x2672, + 0x1611, 0x0630, 0x76d7, 0x66f6, 0x5695, 0x46b4, 0xb75b, 0xa77a, 0x9719, + 0x8738, 0xf7df, 0xe7fe, 0xd79d, 0xc7bc, 0x48c4, 0x58e5, 0x6886, 0x78a7, + 0x0840, 0x1861, 0x2802, 0x3823, 0xc9cc, 0xd9ed, 0xe98e, 0xf9af, 0x8948, + 0x9969, 0xa90a, 0xb92b, 0x5af5, 0x4ad4, 0x7ab7, 0x6a96, 0x1a71, 0x0a50, + 0x3a33, 0x2a12, 0xdbfd, 0xcbdc, 0xfbbf, 0xeb9e, 0x9b79, 0x8b58, 0xbb3b, + 0xab1a, 0x6ca6, 0x7c87, 0x4ce4, 0x5cc5, 0x2c22, 0x3c03, 0x0c60, 0x1c41, + 0xedae, 0xfd8f, 0xcdec, 0xddcd, 0xad2a, 0xbd0b, 0x8d68, 0x9d49, 0x7e97, + 0x6eb6, 0x5ed5, 0x4ef4, 0x3e13, 0x2e32, 0x1e51, 0x0e70, 0xff9f, 0xefbe, + 0xdfdd, 0xcffc, 0xbf1b, 0xaf3a, 0x9f59, 0x8f78, 0x9188, 0x81a9, 0xb1ca, + 0xa1eb, 0xd10c, 0xc12d, 0xf14e, 0xe16f, 0x1080, 0x00a1, 0x30c2, 0x20e3, + 0x5004, 0x4025, 0x7046, 0x6067, 0x83b9, 0x9398, 0xa3fb, 0xb3da, 0xc33d, + 0xd31c, 0xe37f, 0xf35e, 0x02b1, 0x1290, 0x22f3, 0x32d2, 0x4235, 0x5214, + 0x6277, 0x7256, 0xb5ea, 0xa5cb, 0x95a8, 0x8589, 0xf56e, 0xe54f, 0xd52c, + 0xc50d, 0x34e2, 0x24c3, 0x14a0, 0x0481, 0x7466, 0x6447, 0x5424, 0x4405, + 0xa7db, 0xb7fa, 0x8799, 0x97b8, 0xe75f, 0xf77e, 0xc71d, 0xd73c, 0x26d3, + 0x36f2, 0x0691, 0x16b0, 0x6657, 0x7676, 0x4615, 0x5634, 0xd94c, 0xc96d, + 0xf90e, 0xe92f, 0x99c8, 0x89e9, 0xb98a, 0xa9ab, 0x5844, 0x4865, 0x7806, + 0x6827, 0x18c0, 0x08e1, 0x3882, 0x28a3, 0xcb7d, 0xdb5c, 0xeb3f, 0xfb1e, + 0x8bf9, 0x9bd8, 0xabbb, 0xbb9a, 0x4a75, 0x5a54, 0x6a37, 0x7a16, 0x0af1, + 0x1ad0, 0x2ab3, 0x3a92, 0xfd2e, 0xed0f, 0xdd6c, 0xcd4d, 0xbdaa, 0xad8b, + 0x9de8, 0x8dc9, 0x7c26, 0x6c07, 0x5c64, 0x4c45, 0x3ca2, 0x2c83, 0x1ce0, + 0x0cc1, 0xef1f, 0xff3e, 0xcf5d, 0xdf7c, 0xaf9b, 0xbfba, 0x8fd9, 0x9ff8, + 0x6e17, 0x7e36, 0x4e55, 0x5e74, 0x2e93, 0x3eb2, 0x0ed1, 0x1ef0 }; + +unsigned short crc16(unsigned char *buf, unsigned int len) { + unsigned int i; + unsigned short cksum = 0; + for (i = 0; i < len; i++) { + cksum = crc16_tab[(((cksum >> 8) ^ *buf++) & 0xFF)] ^ (cksum << 8); + } + return cksum; +} diff --git a/Node/External/Src/packet.c b/Node/External/Src/packet.c new file mode 100644 index 0000000..07aa1f4 --- /dev/null +++ b/Node/External/Src/packet.c @@ -0,0 +1,169 @@ +/* + Copyright 2012-2014 Benjamin Vedder benjamin@vedder.se + + 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. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ + +/* + * packet.c + * + * Created on: 21 mar 2013 + * Author: benjamin + */ + +#include +#include "packet.h" +#include "crc.h" + +typedef struct { + volatile unsigned char rx_state; + volatile unsigned char rx_timeout; + void(*send_func)(unsigned char *data, unsigned int len); + void(*process_func)(BldcInterface* interface, unsigned char *data, unsigned int len); + unsigned int payload_length; + unsigned char rx_buffer[PACKET_MAX_PL_LEN]; + unsigned char tx_buffer[PACKET_MAX_PL_LEN + 6]; + unsigned int rx_data_ptr; + unsigned char crc_low; + unsigned char crc_high; +} PACKET_STATE_t; + +static PACKET_STATE_t handler_states[PACKET_HANDLERS]; + +void packet_init(void (*s_func)(unsigned char *data, unsigned int len), + void (*p_func)(BldcInterface* interface, unsigned char *data, unsigned int len), int handler_num) { + handler_states[handler_num].send_func = s_func; + handler_states[handler_num].process_func = p_func; +} + +void packet_send_packet(unsigned char *data, unsigned int len, int handler_num) { + if (len > PACKET_MAX_PL_LEN) { + return; + } + + int b_ind = 0; + + if (len <= 256) { + handler_states[handler_num].tx_buffer[b_ind++] = 2; + handler_states[handler_num].tx_buffer[b_ind++] = len; + } else { + handler_states[handler_num].tx_buffer[b_ind++] = 3; + handler_states[handler_num].tx_buffer[b_ind++] = len >> 8; + handler_states[handler_num].tx_buffer[b_ind++] = len & 0xFF; + } + + memcpy(handler_states[handler_num].tx_buffer + b_ind, data, len); + b_ind += len; + + unsigned short crc = crc16(data, len); + handler_states[handler_num].tx_buffer[b_ind++] = (uint8_t)(crc >> 8); + handler_states[handler_num].tx_buffer[b_ind++] = (uint8_t)(crc & 0xFF); + handler_states[handler_num].tx_buffer[b_ind++] = 3; + + if (handler_states[handler_num].send_func) { + handler_states[handler_num].send_func(handler_states[handler_num].tx_buffer, b_ind); + } +} + +/** + * Call this function every millisecond. + */ +void packet_timerfunc(void) { + int i = 0; + for (i = 0;i < PACKET_HANDLERS;i++) { + if (handler_states[i].rx_timeout) { + handler_states[i].rx_timeout--; + } else { + handler_states[i].rx_state = 0; + } + } +} + +void packet_process_byte(BldcInterface* interface, uint8_t rx_data, int handler_num) { + switch (handler_states[handler_num].rx_state) { + case 0: + if (rx_data == 2) { + // 1 byte PL len + handler_states[handler_num].rx_state += 2; + handler_states[handler_num].rx_timeout = PACKET_RX_TIMEOUT; + handler_states[handler_num].rx_data_ptr = 0; + handler_states[handler_num].payload_length = 0; + } else if (rx_data == 3) { + // 2 byte PL len + handler_states[handler_num].rx_state++; + handler_states[handler_num].rx_timeout = PACKET_RX_TIMEOUT; + handler_states[handler_num].rx_data_ptr = 0; + handler_states[handler_num].payload_length = 0; + } else { + handler_states[handler_num].rx_state = 0; + } + break; + + case 1: + handler_states[handler_num].payload_length = (unsigned int)rx_data << 8; + handler_states[handler_num].rx_state++; + handler_states[handler_num].rx_timeout = PACKET_RX_TIMEOUT; + break; + + case 2: + handler_states[handler_num].payload_length |= (unsigned int)rx_data; + if (handler_states[handler_num].payload_length > 0 && + handler_states[handler_num].payload_length <= PACKET_MAX_PL_LEN) { + handler_states[handler_num].rx_state++; + handler_states[handler_num].rx_timeout = PACKET_RX_TIMEOUT; + } else { + handler_states[handler_num].rx_state = 0; + } + break; + + case 3: + handler_states[handler_num].rx_buffer[handler_states[handler_num].rx_data_ptr++] = rx_data; + if (handler_states[handler_num].rx_data_ptr == handler_states[handler_num].payload_length) { + handler_states[handler_num].rx_state++; + } + handler_states[handler_num].rx_timeout = PACKET_RX_TIMEOUT; + break; + + case 4: + handler_states[handler_num].crc_high = rx_data; + handler_states[handler_num].rx_state++; + handler_states[handler_num].rx_timeout = PACKET_RX_TIMEOUT; + break; + + case 5: + handler_states[handler_num].crc_low = rx_data; + handler_states[handler_num].rx_state++; + handler_states[handler_num].rx_timeout = PACKET_RX_TIMEOUT; + break; + + case 6: + if (rx_data == 3) { + if (crc16(handler_states[handler_num].rx_buffer, handler_states[handler_num].payload_length) + == ((unsigned short)handler_states[handler_num].crc_high << 8 + | (unsigned short)handler_states[handler_num].crc_low)) { + // Packet received! + if (handler_states[handler_num].process_func) { + handler_states[handler_num].process_func(interface, handler_states[handler_num].rx_buffer, + handler_states[handler_num].payload_length); + } + } + } + handler_states[handler_num].rx_state = 0; + break; + + default: + handler_states[handler_num].rx_state = 0; + break; + } +} diff --git a/Node/Node.ioc b/Node/Node.ioc index 82d5b36..c0500c5 100644 --- a/Node/Node.ioc +++ b/Node/Node.ioc @@ -1,26 +1,27 @@ #MicroXplorer Configuration settings - do not modify -CAN.CalculateBaudRate=166666 -CAN.CalculateTimeBit=6000 -CAN.CalculateTimeQuantum=2000.0 -CAN.IPParameters=CalculateTimeQuantum,CalculateTimeBit,CalculateBaudRate +CAN.BS1=CAN_BS1_3TQ +CAN.BS2=CAN_BS2_4TQ +CAN.CalculateBaudRate=500000 +CAN.CalculateTimeBit=2000 +CAN.CalculateTimeQuantum=250.0 +CAN.IPParameters=CalculateTimeQuantum,CalculateTimeBit,CalculateBaudRate,Prescaler,BS1,BS2 +CAN.Prescaler=2 File.Version=6 -GPIO.groupedBy= +GPIO.groupedBy=Group By Peripherals KeepUserPlacement=false Mcu.CPN=STM32F334R8T6TR Mcu.Family=STM32F3 Mcu.IP0=CAN -Mcu.IP1=I2C1 -Mcu.IP10=USART2 -Mcu.IP11=USART3 -Mcu.IP2=NVIC -Mcu.IP3=RCC -Mcu.IP4=SPI1 -Mcu.IP5=SYS -Mcu.IP6=TIM2 -Mcu.IP7=TIM3 -Mcu.IP8=TIM17 -Mcu.IP9=USART1 -Mcu.IPNb=12 +Mcu.IP1=NVIC +Mcu.IP2=RCC +Mcu.IP3=SYS +Mcu.IP4=TIM2 +Mcu.IP5=TIM3 +Mcu.IP6=TIM17 +Mcu.IP7=USART1 +Mcu.IP8=USART2 +Mcu.IP9=USART3 +Mcu.IPNb=10 Mcu.Name=STM32F334R(6-8)Tx Mcu.Package=LQFP64 Mcu.Pin0=PC13 @@ -43,22 +44,20 @@ Mcu.Pin23=PA11 Mcu.Pin24=PA12 Mcu.Pin25=PA13 Mcu.Pin26=PA14 -Mcu.Pin27=PA15 -Mcu.Pin28=PB3 -Mcu.Pin29=PB4 +Mcu.Pin27=PB4 +Mcu.Pin28=PB5 +Mcu.Pin29=PB7 Mcu.Pin3=PF0 / OSC_IN -Mcu.Pin30=PB5 -Mcu.Pin31=PB7 -Mcu.Pin32=PB9 -Mcu.Pin33=VP_SYS_VS_Systick -Mcu.Pin34=VP_TIM17_VS_ClockSourceINT +Mcu.Pin30=PB9 +Mcu.Pin31=VP_SYS_VS_Systick +Mcu.Pin32=VP_TIM17_VS_ClockSourceINT Mcu.Pin4=PF1 / OSC_OUT Mcu.Pin5=PA0 Mcu.Pin6=PA1 Mcu.Pin7=PA2 Mcu.Pin8=PA3 Mcu.Pin9=PA4 -Mcu.PinsNb=35 +Mcu.PinsNb=33 Mcu.ThirdPartyNb=0 Mcu.UserConstants= Mcu.UserName=STM32F334R8Tx @@ -66,6 +65,8 @@ MxCube.Version=6.5.0 MxDb.Version=DB.6.0.50 NVIC.BusFault_IRQn=true\:0\:0\:false\:false\:true\:false\:false\:true NVIC.DebugMonitor_IRQn=true\:0\:0\:false\:false\:true\:false\:false\:true +NVIC.EXTI4_IRQn=true\:0\:0\:false\:false\:true\:true\:true\:true +NVIC.EXTI9_5_IRQn=true\:0\:0\:false\:false\:true\:true\:true\:true NVIC.ForceEnableDMAVector=true NVIC.HardFault_IRQn=true\:0\:0\:false\:false\:true\:false\:false\:true NVIC.MemoryManagement_IRQn=true\:0\:0\:false\:false\:true\:false\:false\:true @@ -74,6 +75,9 @@ NVIC.PendSV_IRQn=true\:0\:0\:false\:false\:true\:false\:false\:true NVIC.PriorityGroup=NVIC_PRIORITYGROUP_4 NVIC.SVCall_IRQn=true\:0\:0\:false\:false\:true\:false\:false\:true NVIC.SysTick_IRQn=true\:15\:0\:false\:false\:true\:false\:true\:true +NVIC.USART1_IRQn=true\:0\:0\:false\:false\:true\:true\:true\:true +NVIC.USART2_IRQn=true\:0\:0\:false\:false\:true\:true\:true\:true +NVIC.USART3_IRQn=true\:0\:0\:false\:false\:true\:true\:true\:true NVIC.UsageFault_IRQn=true\:0\:0\:false\:false\:true\:false\:false\:true PA0.GPIOParameters=GPIO_Label PA0.GPIO_Label=ESC1_ON @@ -95,8 +99,6 @@ PA13.Mode=Serial_Wire PA13.Signal=SYS_JTMS-SWDIO PA14.Mode=Serial_Wire PA14.Signal=SYS_JTCK-SWCLK -PA15.Mode=I2C -PA15.Signal=I2C1_SCL PA2.GPIOParameters=GPIO_Label PA2.GPIO_Label=ESC1_TX PA2.Mode=Asynchronous @@ -112,7 +114,6 @@ PA5.Signal=GPIO_Output PA6.Locked=true PA6.Signal=S_TIM3_CH1 PA7.Locked=true -PA7.Mode=Full_Duplex_Master PA7.Signal=SPI1_MOSI PB0.Locked=true PB0.Signal=S_TIM3_CH3 @@ -128,14 +129,12 @@ PB14.Locked=true PB14.Signal=S_TIM15_CH1 PB15.Locked=true PB15.Signal=S_TIM15_CH2 -PB3.Mode=Full_Duplex_Master -PB3.Signal=SPI1_SCK -PB4.Mode=Full_Duplex_Master -PB4.Signal=SPI1_MISO +PB4.Locked=true +PB4.Signal=GPXTI4 PB5.Locked=true PB5.Signal=GPIO_Output -PB7.Mode=I2C -PB7.Signal=I2C1_SDA +PB7.Locked=true +PB7.Signal=GPXTI7 PB9.GPIOParameters=GPIO_Label PB9.GPIO_Label=LAMP4_ON PB9.Locked=true @@ -192,7 +191,7 @@ ProjectManager.StackSize=0x400 ProjectManager.TargetToolchain=Makefile ProjectManager.ToolChainLocation= ProjectManager.UnderRoot=false -ProjectManager.functionlistsort=1-SystemClock_Config-RCC-false-HAL-false,2-MX_GPIO_Init-GPIO-false-HAL-true,3-MX_CAN_Init-CAN-false-HAL-true,4-MX_USART1_UART_Init-USART1-false-HAL-true,5-MX_USART2_UART_Init-USART2-false-HAL-true,6-MX_USART3_UART_Init-USART3-false-HAL-true,7-MX_TIM2_Init-TIM2-false-HAL-true,8-MX_TIM3_Init-TIM3-false-HAL-true,9-MX_TIM17_Init-TIM17-false-HAL-true,10-MX_I2C1_Init-I2C1-false-HAL-true,11-MX_SPI1_Init-SPI1-false-HAL-true +ProjectManager.functionlistsort=1-SystemClock_Config-RCC-false-HAL-false,2-MX_GPIO_Init-GPIO-false-HAL-true,3-MX_CAN_Init-CAN-false-HAL-true,4-MX_USART1_UART_Init-USART1-false-HAL-true,5-MX_USART2_UART_Init-USART2-false-HAL-true,6-MX_USART3_UART_Init-USART3-false-HAL-true,7-MX_TIM2_Init-TIM2-false-HAL-true,8-MX_TIM3_Init-TIM3-false-HAL-true,9-MX_TIM17_Init-TIM17-false-HAL-true,10-MX_SPI1_Init-SPI1-false-HAL-true RCC.AHBFreq_Value=8000000 RCC.APB1Freq_Value=8000000 RCC.APB2Freq_Value=8000000 @@ -214,6 +213,10 @@ RCC.SYSCLKFreq_VALUE=8000000 RCC.TIM2Freq_Value=8000000 RCC.USART1Freq_Value=8000000 RCC.VCOOutput2Freq_Value=4000000 +SH.GPXTI4.0=GPIO_EXTI4 +SH.GPXTI4.ConfNb=1 +SH.GPXTI7.0=GPIO_EXTI7 +SH.GPXTI7.ConfNb=1 SH.S_TIM15_CH1.0=TIM15_CH1 SH.S_TIM15_CH1.ConfNb=1 SH.S_TIM15_CH2.0=TIM15_CH2 @@ -226,21 +229,19 @@ SH.S_TIM3_CH2.0=TIM3_CH2,PWM Generation2 CH2 SH.S_TIM3_CH2.ConfNb=1 SH.S_TIM3_CH3.0=TIM3_CH3,PWM Generation3 CH3 SH.S_TIM3_CH3.ConfNb=1 -SPI1.CalculateBaudRate=4.0 MBits/s -SPI1.Direction=SPI_DIRECTION_2LINES -SPI1.IPParameters=VirtualType,Mode,Direction,CalculateBaudRate -SPI1.Mode=SPI_MODE_MASTER -SPI1.VirtualType=VM_MASTER TIM2.Channel-PWM\ Generation2\ CH2=TIM_CHANNEL_2 TIM2.IPParameters=Channel-PWM Generation2 CH2 TIM3.Channel-PWM\ Generation2\ CH2=TIM_CHANNEL_2 TIM3.Channel-PWM\ Generation3\ CH3=TIM_CHANNEL_3 TIM3.IPParameters=Channel-PWM Generation2 CH2,Channel-PWM Generation3 CH3 -USART1.IPParameters=VirtualMode-Asynchronous +USART1.BaudRate=115200 +USART1.IPParameters=VirtualMode-Asynchronous,BaudRate USART1.VirtualMode-Asynchronous=VM_ASYNC -USART2.IPParameters=VirtualMode-Asynchronous +USART2.BaudRate=115200 +USART2.IPParameters=VirtualMode-Asynchronous,BaudRate USART2.VirtualMode-Asynchronous=VM_ASYNC -USART3.IPParameters=VirtualMode-Asynchronous +USART3.BaudRate=115200 +USART3.IPParameters=VirtualMode-Asynchronous,BaudRate USART3.VirtualMode-Asynchronous=VM_ASYNC VP_SYS_VS_Systick.Mode=SysTick VP_SYS_VS_Systick.Signal=SYS_VS_Systick diff --git a/Node/build.sh b/Node/build.sh index 0d1ebe9..49698ae 100755 --- a/Node/build.sh +++ b/Node/build.sh @@ -1,9 +1,9 @@ #!/bin/bash -if [ ! -d "build" ]; then +if [ ! -d "Build" ]; then echo "Making build directory..." - mkdir build + mkdir Build fi -pushd build +pushd Build cmake -DCMAKE_TOOLCHAIN_FILE=../arm-none-eabi-gcc.cmake -DCMAKE_BUILD_TYPE=Debug .. make mv compile_commands.json ..