323 lines
7.6 KiB
C
323 lines
7.6 KiB
C
#include "board.h"
|
||
#include "motor.h"
|
||
|
||
void motor_process(frame_msg_t *rx, frame_msg_t *bk);
|
||
void sensor_procss(frame_msg_t *rx, frame_msg_t *bk);
|
||
|
||
/************************************* 串口通讯 *************************************/
|
||
uart_t *uarts[UART_NUM_MAX];
|
||
__IO static BOOL execute_state = 0; // 数据处理标志
|
||
static frame_msg_t rx_msg; // 接收数据句柄
|
||
static frame_msg_t bk_msg; // 回复数据句柄
|
||
static uint8_t host_send_buffer[UART_TXSIZE] = {0}; // 数据发送缓存
|
||
|
||
// 串口发送
|
||
static void host_send_msg(uint8_t *data, uint16_t len)
|
||
{
|
||
uart_send_data(uarts[UART_NUM_1], data, len);
|
||
}
|
||
|
||
// 数据有效性检查
|
||
static BOOL host_data_verify(uint8_t *data, uint16_t len)
|
||
{
|
||
BOOL ret = TRUE;
|
||
uint8_t ver_len = 0;
|
||
uint8_t msg_crc = 0, ver_crc = 0;
|
||
|
||
// 包头包尾检查
|
||
if (data[0] != PACKET_STX || data[1] != ST_DEV_NORMAL || data[len - 1] != PACKET_ETX)
|
||
{
|
||
ret = FALSE;
|
||
return ret;
|
||
}
|
||
// 长度检查
|
||
ver_len = PACKET_MIN_LEN + data[DATA_LEN_INDEX];
|
||
if (ver_len != len)
|
||
{
|
||
ret = FALSE;
|
||
return ret;
|
||
}
|
||
// CRC校验
|
||
msg_crc = data[len - 2];
|
||
ver_crc = xor_compute((uint8_t *)&data[1], (len - 3)); // 减去包头、包尾、校验
|
||
if (msg_crc != ver_crc)
|
||
{
|
||
ret = FALSE;
|
||
return ret;
|
||
}
|
||
|
||
return ret;
|
||
}
|
||
|
||
// 数据解码
|
||
static void host_data_decode(uint8_t *data, uint16_t len)
|
||
{
|
||
DBG_ASSERT(data != NULL __DBG_LINE);
|
||
|
||
// 获取数据长度和消息长度
|
||
uint8_t data_len = data[DATA_LEN_INDEX];
|
||
uint8_t msg_len = DATA_INDEX + data_len;
|
||
// 获取消息数据
|
||
osel_memcpy((uint8_t *)&rx_msg, data, msg_len);
|
||
}
|
||
|
||
// 指令执行
|
||
static void host_cmd_execute(frame_msg_t *rx_msg, frame_msg_t *bk_msg)
|
||
{
|
||
DBG_ASSERT(rx_msg != NULL __DBG_LINE);
|
||
DBG_ASSERT(bk_msg != NULL __DBG_LINE);
|
||
|
||
switch (rx_msg->dev_no)
|
||
{
|
||
case ADC_SENSOR:
|
||
sensor_procss(rx_msg, bk_msg);
|
||
break;
|
||
case MOTOR:
|
||
motor_process(rx_msg, bk_msg);
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
}
|
||
|
||
// 数据编码
|
||
static void host_data_encode(frame_msg_t *msg)
|
||
{
|
||
DBG_ASSERT(msg != NULL __DBG_LINE);
|
||
|
||
uint8_t *pbuf = (uint8_t *)host_send_buffer;
|
||
uint8_t index = 0, len = 0;
|
||
|
||
// 消息数据
|
||
uint8_t msg_len = DATA_INDEX + msg->len;
|
||
osel_memcpy(pbuf, (uint8_t *)msg, msg_len);
|
||
// 校验码
|
||
index += msg_len;
|
||
pbuf[index] = xor_compute((uint8_t *)&pbuf[1], (msg_len - 1)); // 减去包头
|
||
// 包尾
|
||
index += 1;
|
||
pbuf[index] = PACKET_ETX;
|
||
|
||
// 计算报文长度
|
||
len = msg_len + 2; // 加上校验、包尾
|
||
// 发送
|
||
host_send_msg(pbuf, len);
|
||
}
|
||
|
||
// 数据处理
|
||
void host_rx_msg_deal(void)
|
||
{
|
||
if (execute_state == TRUE)
|
||
{
|
||
return;
|
||
}
|
||
|
||
frame_msg_t *p_rx_msg = (frame_msg_t *)&rx_msg;
|
||
frame_msg_t *p_bk_msg = (frame_msg_t *)&bk_msg;
|
||
|
||
// 执行
|
||
host_cmd_execute(p_rx_msg, p_bk_msg);
|
||
// 回复(组包 + 发送)
|
||
host_data_encode(p_bk_msg);
|
||
// 复位
|
||
osel_memset((uint8_t *)host_send_buffer, 0, UART_TXSIZE);
|
||
osel_memset((uint8_t *)&rx_msg, 0, sizeof(frame_msg_t));
|
||
osel_memset((uint8_t *)&bk_msg, 0, sizeof(frame_msg_t));
|
||
execute_state = IDLE;
|
||
}
|
||
|
||
// 串口接收回调
|
||
static void host_rx_cb(uint8_t uart_index, uint8_t *data, uint16_t len)
|
||
{
|
||
BOOL ret = FALSE;
|
||
|
||
// 数据有效性检查
|
||
ret = host_data_verify(data, len);
|
||
// 数据解码
|
||
if (ret == TRUE && execute_state == IDLE)
|
||
{
|
||
host_data_decode(data, len);
|
||
execute_state = BUSY;
|
||
}
|
||
}
|
||
|
||
// 串口初始化
|
||
static void host_uart_init(void)
|
||
{
|
||
if (uarts[UART_NUM_1] == NULL)
|
||
{
|
||
uarts[UART_NUM_1] = uart_create(USART1, TRUE, UART_RXSIZE, host_rx_cb, TRUE, UART_TXSIZE, NULL);
|
||
uarts[UART_NUM_1]->uart_index = UART_NUM_1;
|
||
uarts[UART_NUM_1]->dma = DMA1;
|
||
uarts[UART_NUM_1]->dma_rx_channel = LL_DMA_CHANNEL_3;
|
||
uarts[UART_NUM_1]->dma_tx_channel = LL_DMA_CHANNEL_2;
|
||
uart_recv_en(uarts[UART_NUM_1]);
|
||
}
|
||
}
|
||
|
||
/************************************* 电机 *************************************/
|
||
// 电机对象
|
||
motor_t *motor;
|
||
motor_ctrl_t *motor_param = NULL;
|
||
|
||
// 电机初始化
|
||
static void motor_init(void)
|
||
{
|
||
motor = motor_create(STEP_MOTOR);
|
||
gpio_t dir = {
|
||
.port = DIR_GPIO_Port,
|
||
.pin = DIR_Pin,
|
||
};
|
||
gpio_t en = {
|
||
.port = ENA_GPIO_Port,
|
||
.pin = ENA_Pin,
|
||
};
|
||
motor->handle.step_motor.interface.init(motor, dir, en, MIN_STEP_ANGLE, TIM21, LL_TIM_CHANNEL_CH2);
|
||
}
|
||
|
||
void motor_process_init(void)
|
||
{
|
||
motor_param = (motor_ctrl_t *)osel_mem_alloc(sizeof(motor_ctrl_t));
|
||
DBG_ASSERT(motor_param != NULL __DBG_LINE);
|
||
osel_memset((uint8_t *)motor_param, 0, sizeof(motor_ctrl_t));
|
||
}
|
||
|
||
// 电机任务
|
||
void motor_process(frame_msg_t *rx, frame_msg_t *bk)
|
||
{
|
||
DBG_ASSERT(rx != NULL __DBG_LINE);
|
||
|
||
// step_motor_t *step_motor = &motor->handle.step_motor;
|
||
// uint8_t state = ST_DEV_NORMAL;
|
||
|
||
switch (rx->cmd_no)
|
||
{
|
||
case SET_MOTOR_SPEED: // 设置电机转速
|
||
break;
|
||
case GET_MIN_STEP: // 获取最小步距
|
||
break;
|
||
case MOTOR_MOVE: // 运行电机
|
||
break;
|
||
case MOTOR_STOP: // 停止电机
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
}
|
||
|
||
/************************************* ADc传感器 *************************************/
|
||
// AD传感器
|
||
adc_t adc1 = {
|
||
.adc = ADC1,
|
||
.dma = DMA1,
|
||
.dma_channel = LL_DMA_CHANNEL_1,
|
||
};
|
||
|
||
// 扭力,单位:N·m
|
||
uint16_t get_torsion_adc(void)
|
||
{
|
||
uint16_t adc = adc_get_result_average(IN13);
|
||
return adc;
|
||
}
|
||
|
||
float32 torsion_detect(void)
|
||
{
|
||
float32 adc = 0.0, val = 0.0;
|
||
|
||
adc = get_torsion_adc();
|
||
val = adc / 4096 * 3000;
|
||
val = (val - 1500) / 101 / (1.934 * 3) * 10;
|
||
|
||
return val;
|
||
}
|
||
|
||
// 压力,单位:kPa
|
||
uint16_t get_pressure_adc(void)
|
||
{
|
||
uint16_t adc = adc_get_result_average(IN7);
|
||
return adc;
|
||
}
|
||
|
||
float32 pressure_detect(void)
|
||
{
|
||
float32 adc = 0.0, val = 0.0;
|
||
|
||
adc = get_pressure_adc();
|
||
val = adc / 4096 * 3000;
|
||
val = (val - 600) / (10 * 15) / 16 * 1000;
|
||
|
||
return val;
|
||
}
|
||
|
||
// 流量,单位:SLPM
|
||
uint16_t get_flow_adc(void)
|
||
{
|
||
uint16_t adc = adc_get_result_average(IN8);
|
||
return adc;
|
||
}
|
||
|
||
float32 flow_detect(void)
|
||
{
|
||
float32 adc = 0.0, val = 0.0;
|
||
|
||
adc = get_flow_adc();
|
||
val = adc / 4096 * 3000;
|
||
val = (val - 600) / (10 * 15) / 16 * 300;
|
||
|
||
return val;
|
||
}
|
||
|
||
// 获取扭力值
|
||
static void cmd_torsion_value(frame_msg_t *bk)
|
||
{
|
||
|
||
uint8_t state = ST_DEV_NORMAL;
|
||
|
||
// 计算数据长度
|
||
uint8_t data_len = sizeof(uint16_t);
|
||
// 采集
|
||
uint16_t value = get_torsion_adc();
|
||
// TODO 检查数据合法性
|
||
|
||
// 回复数据
|
||
bk->head = PACKET_STX;
|
||
bk->state = state;
|
||
bk->dev_no = ADC_SENSOR;
|
||
bk->cmd_no = GET_TORSION_VALUE;
|
||
bk->len = data_len;
|
||
value = S2B_UINT16(value);
|
||
osel_memcpy((uint8_t *)bk->data, (uint8_t *)&value, data_len);
|
||
}
|
||
|
||
// 传感器任务
|
||
void sensor_procss(frame_msg_t *rx, frame_msg_t *bk)
|
||
{
|
||
switch (rx->cmd_no)
|
||
{
|
||
case GET_TORSION_VALUE: // 获取扭力值
|
||
cmd_torsion_value(bk);
|
||
break;
|
||
case GET_PRESSURE_VALUE:
|
||
break;
|
||
case GET_FLOW_VALUE:
|
||
break;
|
||
case GET_ALL_VALUE:
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
}
|
||
|
||
/************************************* 板卡 *************************************/
|
||
// 板卡初始化
|
||
void board_init(void)
|
||
{
|
||
motor = NULL;
|
||
my_mem_init(SRAMIN);
|
||
|
||
motor_init(); // 电机初始化
|
||
adc_init(adc1); // 初始化ADC1通道,默认采集AD
|
||
host_uart_init(); // 串口初始化
|
||
ENABLE_TIM(TIM6); // 任务流程定时器使能
|
||
}
|