rm_balance/User/task/imu.c
2025-10-19 13:37:10 +08:00

220 lines
9.5 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
imu Task
*/
/* Includes ----------------------------------------------------------------- */
#include "task/user_task.h"
/* USER INCLUDE BEGIN */
#include "component/user_math.h"
#include "bsp/can.h"
#include "bsp/time.h"
#include "component/ahrs.h"
#include "component/filter.h"
#include <string.h>
#include <stdint.h>
/* USER INCLUDE END */
/* Private typedef ---------------------------------------------------------- */
/* Private define ----------------------------------------------------------- */
/* AHRS数据CAN ID定义 */
#define CAN_ID_AHRS_ACCL 0x301 /* 加速度计数据 */
#define CAN_ID_AHRS_GYRO 0x302 /* 陀螺仪数据 */
#define CAN_ID_AHRS_EULR 0x303 /* 欧拉角数据 */
#define CAN_ID_AHRS_QUAT 0x304 /* 四元数数据 */
/* 数据范围定义 */
#define ACCEL_CAN_MAX (58.8f)
#define ACCEL_CAN_MIN (-58.8f)
#define GYRO_CAN_MAX (34.88f)
#define GYRO_CAN_MIN (-34.88f)
#define PITCH_CAN_MAX (M_PI_2) /* π/2 弧度 ≈ 90° */
#define PITCH_CAN_MIN (-M_PI_2) /* -π/2 弧度 ≈ -90° */
#define ROLL_CAN_MAX (M_PI) /* π 弧度 ≈ 180° */
#define ROLL_CAN_MIN (-M_PI) /* -π 弧度 ≈ -180° */
#define YAW_CAN_MAX (M_PI) /* π 弧度 ≈ 180° */
#define YAW_CAN_MIN (-M_PI) /* -π 弧度 ≈ -180° */
#define QUATERNION_MIN (-1.0f)
#define QUATERNION_MAX (1.0f)
/* Private macro ------------------------------------------------------------ */
/* 数据映射宏将浮点值映射到16位整数范围 */
#define MAP_TO_INT16(val, min, max) \
((int16_t)(((val) - (min)) / ((max) - (min)) * 65535.0f - 32768.0f))
/* 限制值在指定范围内 */
#define CLAMP(val, min, max) \
(((val) < (min)) ? (min) : (((val) > (max)) ? (max) : (val)))
/* Private variables -------------------------------------------------------- */
/* USER STRUCT BEGIN */
AHRS_Accl_t accl;
AHRS_Gyro_t gyro;
AHRS_Eulr_t eulr;
AHRS_Quaternion_t quat;
/* 低通滤波器定义 */
/* 加速度计3轴滤波器 */
LowPassFilter2p_t accl_filter_x;
LowPassFilter2p_t accl_filter_y;
LowPassFilter2p_t accl_filter_z;
/* 陀螺仪3轴滤波器 */
LowPassFilter2p_t gyro_filter_x;
LowPassFilter2p_t gyro_filter_y;
LowPassFilter2p_t gyro_filter_z;
/* 欧拉角3轴滤波器 */
LowPassFilter2p_t eulr_filter_yaw;
LowPassFilter2p_t eulr_filter_pitch;
LowPassFilter2p_t eulr_filter_roll;
/* 滤波器参数定义 */
#define FILTER_CUTOFF_FREQ (30.0f) /* 截止频率50Hz */
/* USER STRUCT END */
/* Private function --------------------------------------------------------- */
/* Exported functions ------------------------------------------------------- */
void Task_imu(void *argument) {
(void)argument; /* 未使用argument消除警告 */
/* 计算任务运行到指定频率需要等待的tick数 */
const uint32_t delay_tick = osKernelGetTickFreq() / IMU_FREQ;
osDelay(IMU_INIT_DELAY); /* 延时一段时间再开启任务 */
uint32_t tick = osKernelGetTickCount(); /* 控制任务运行频率的计时 */
/* USER CODE INIT BEGIN */
BSP_CAN_Init();
/* 初始化低通滤波器 */
/* 加速度计滤波器初始化 */
LowPassFilter2p_Init(&accl_filter_x, IMU_FREQ, FILTER_CUTOFF_FREQ);
LowPassFilter2p_Init(&accl_filter_y, IMU_FREQ, FILTER_CUTOFF_FREQ);
LowPassFilter2p_Init(&accl_filter_z, IMU_FREQ, FILTER_CUTOFF_FREQ);
/* 陀螺仪滤波器初始化 */
LowPassFilter2p_Init(&gyro_filter_x, IMU_FREQ, FILTER_CUTOFF_FREQ);
LowPassFilter2p_Init(&gyro_filter_y, IMU_FREQ, FILTER_CUTOFF_FREQ);
LowPassFilter2p_Init(&gyro_filter_z, IMU_FREQ, FILTER_CUTOFF_FREQ);
/* 欧拉角滤波器初始化 */
LowPassFilter2p_Init(&eulr_filter_yaw, IMU_FREQ, FILTER_CUTOFF_FREQ);
LowPassFilter2p_Init(&eulr_filter_pitch, IMU_FREQ, FILTER_CUTOFF_FREQ);
LowPassFilter2p_Init(&eulr_filter_roll, IMU_FREQ, FILTER_CUTOFF_FREQ);
/* USER CODE INIT END */
while (1) {
tick += delay_tick; /* 计算下一个唤醒时刻 */
/* USER CODE BEGIN */
/* 获取加速度计数据应用滤波器后通过CAN发送 - 每轴使用16位编码 */
if (osMessageQueueGet(task_runtime.msgq.imu.accl, &accl, NULL, 0) == osOK) {
BSP_CAN_StdDataFrame_t accl_frame;
accl_frame.id = CAN_ID_AHRS_ACCL;
accl_frame.dlc = 6; /* 3轴 × 2字节 = 6字节 */
/* 对加速度数据应用低通滤波器 */
float ax_filtered = LowPassFilter2p_Apply(&accl_filter_x, accl.x);
float ay_filtered = LowPassFilter2p_Apply(&accl_filter_y, accl.y);
float az_filtered = LowPassFilter2p_Apply(&accl_filter_z, accl.z);
/* 限制并映射滤波后的加速度数据到16位整数 */
float ax = CLAMP(ax_filtered, ACCEL_CAN_MIN, ACCEL_CAN_MAX);
float ay = CLAMP(ay_filtered, ACCEL_CAN_MIN, ACCEL_CAN_MAX);
float az = CLAMP(az_filtered, ACCEL_CAN_MIN, ACCEL_CAN_MAX);
int16_t ax_int = MAP_TO_INT16(ax, ACCEL_CAN_MIN, ACCEL_CAN_MAX);
int16_t ay_int = MAP_TO_INT16(ay, ACCEL_CAN_MIN, ACCEL_CAN_MAX);
int16_t az_int = MAP_TO_INT16(az, ACCEL_CAN_MIN, ACCEL_CAN_MAX);
memcpy(&accl_frame.data[0], &ax_int, sizeof(int16_t));
memcpy(&accl_frame.data[2], &ay_int, sizeof(int16_t));
memcpy(&accl_frame.data[4], &az_int, sizeof(int16_t));
BSP_CAN_TransmitStdDataFrame(BSP_CAN_1, &accl_frame);
}
/* 获取陀螺仪数据应用滤波器后通过CAN发送 - 每轴使用16位编码 */
if (osMessageQueueGet(task_runtime.msgq.imu.gyro, &gyro, NULL, 0) == osOK) {
BSP_CAN_StdDataFrame_t gyro_frame;
gyro_frame.id = CAN_ID_AHRS_GYRO;
gyro_frame.dlc = 6; /* 3轴 × 2字节 = 6字节 */
/* 对陀螺仪数据应用低通滤波器 */
float gx_filtered = LowPassFilter2p_Apply(&gyro_filter_x, gyro.x);
float gy_filtered = LowPassFilter2p_Apply(&gyro_filter_y, gyro.y);
float gz_filtered = LowPassFilter2p_Apply(&gyro_filter_z, gyro.z);
/* 限制并映射滤波后的陀螺仪数据到16位整数 */
float gx = CLAMP(gx_filtered, GYRO_CAN_MIN, GYRO_CAN_MAX);
float gy = CLAMP(gy_filtered, GYRO_CAN_MIN, GYRO_CAN_MAX);
float gz = CLAMP(gz_filtered, GYRO_CAN_MIN, GYRO_CAN_MAX);
int16_t gx_int = MAP_TO_INT16(gx, GYRO_CAN_MIN, GYRO_CAN_MAX);
int16_t gy_int = MAP_TO_INT16(gy, GYRO_CAN_MIN, GYRO_CAN_MAX);
int16_t gz_int = MAP_TO_INT16(gz, GYRO_CAN_MIN, GYRO_CAN_MAX);
memcpy(&gyro_frame.data[0], &gx_int, sizeof(int16_t));
memcpy(&gyro_frame.data[2], &gy_int, sizeof(int16_t));
memcpy(&gyro_frame.data[4], &gz_int, sizeof(int16_t));
BSP_CAN_TransmitStdDataFrame(BSP_CAN_1, &gyro_frame);
}
/* 获取欧拉角数据应用滤波器后通过CAN发送 - 每角使用16位编码 */
if (osMessageQueueGet(task_runtime.msgq.imu.eulr, &eulr, NULL, 0) == osOK) {
BSP_CAN_StdDataFrame_t eulr_frame;
eulr_frame.id = CAN_ID_AHRS_EULR;
eulr_frame.dlc = 6; /* 3个角度 × 2字节 = 6字节 */
/* 对欧拉角数据应用低通滤波器 */
float yaw_filtered = LowPassFilter2p_Apply(&eulr_filter_yaw, eulr.yaw);
float pitch_filtered = LowPassFilter2p_Apply(&eulr_filter_pitch, eulr.pit);
float roll_filtered = LowPassFilter2p_Apply(&eulr_filter_roll, eulr.rol);
/* 限制并映射滤波后的欧拉角数据到16位整数 */
float yaw = CLAMP(yaw_filtered, YAW_CAN_MIN, YAW_CAN_MAX);
float pitch = CLAMP(pitch_filtered, PITCH_CAN_MIN, PITCH_CAN_MAX);
float roll = CLAMP(roll_filtered, ROLL_CAN_MIN, ROLL_CAN_MAX);
int16_t yaw_int = MAP_TO_INT16(yaw, YAW_CAN_MIN, YAW_CAN_MAX);
int16_t pitch_int = MAP_TO_INT16(pitch, PITCH_CAN_MIN, PITCH_CAN_MAX);
int16_t roll_int = MAP_TO_INT16(roll, ROLL_CAN_MIN, ROLL_CAN_MAX);
memcpy(&eulr_frame.data[0], &yaw_int, sizeof(int16_t));
memcpy(&eulr_frame.data[2], &pitch_int, sizeof(int16_t));
memcpy(&eulr_frame.data[4], &roll_int, sizeof(int16_t));
BSP_CAN_TransmitStdDataFrame(BSP_CAN_1, &eulr_frame);
}
/* 获取四元数数据并通过CAN发送 - 每个分量使用16位编码 */
if (osMessageQueueGet(task_runtime.msgq.imu.quat, &quat, NULL, 0) == osOK) {
BSP_CAN_StdDataFrame_t quat_frame;
quat_frame.id = CAN_ID_AHRS_QUAT;
quat_frame.dlc = 8; /* 4个四元数分量 × 2字节 = 8字节 */
/* 限制并映射四元数到16位整数 */
float q0 = CLAMP(quat.q0, QUATERNION_MIN, QUATERNION_MAX);
float q1 = CLAMP(quat.q1, QUATERNION_MIN, QUATERNION_MAX);
float q2 = CLAMP(quat.q2, QUATERNION_MIN, QUATERNION_MAX);
float q3 = CLAMP(quat.q3, QUATERNION_MIN, QUATERNION_MAX);
int16_t q0_int = MAP_TO_INT16(q0, QUATERNION_MIN, QUATERNION_MAX);
int16_t q1_int = MAP_TO_INT16(q1, QUATERNION_MIN, QUATERNION_MAX);
int16_t q2_int = MAP_TO_INT16(q2, QUATERNION_MIN, QUATERNION_MAX);
int16_t q3_int = MAP_TO_INT16(q3, QUATERNION_MIN, QUATERNION_MAX);
memcpy(&quat_frame.data[0], &q0_int, sizeof(int16_t));
memcpy(&quat_frame.data[2], &q1_int, sizeof(int16_t));
memcpy(&quat_frame.data[4], &q2_int, sizeof(int16_t));
memcpy(&quat_frame.data[6], &q3_int, sizeof(int16_t));
BSP_CAN_TransmitStdDataFrame(BSP_CAN_1, &quat_frame);
}
/* USER CODE END */
osDelayUntil(tick); /* 运行结束,等待下一次唤醒 */
}
}