基于RT-Thread的RoboMaster电控框架(四)

发布时间:2024-01-12  

背景


使用的开发板为大疆的 RoboMaster-C 型开发板,基础工程为 rt-thread>bsp>stm32f407-robomaster-c



IMU姿态解算


使用 BMI088 robomaster-c 开发板上集成的6轴 imu 进行姿态解算,使用的方案其实是RM中比较普遍成熟的一套,主要使用了四元数和卡尔曼滤波进行融合解算,解算频率为 1Khz。


dt = dwt_get_delta(&ins_dwt);

    imu_ops.gyro_read(ins.gyro);

    imu_ops.accel_read(ins.accel);

    // 核心函数,EKF更新四元数

    IMU_QuaternionEKF_Update(ins.gyro[X], ins.gyro[Y], ins.gyro[Z], ins.accel[X], ins.accel[Y], ins.accel[Z], dt);

    memcpy(ins.q, QEKF_INS.q, sizeof(QEKF_INS.q));

    // 机体系基向量转换到导航坐标系,本例选取惯性系为导航系

    BodyFrameToEarthFrame(xb, ins.xn, ins.q);

    BodyFrameToEarthFrame(yb, ins.yn, ins.q);

    BodyFrameToEarthFrame(zb, ins.zn, ins.q);

    // 将重力从导航坐标系n转换到机体系b,随后根据加速度计数据计算运动加速度

    float gravity_b[3];

    EarthFrameToBodyFrame(gravity, gravity_b, ins.q);

    for (uint8_t i = 0; i < 3; ++i) // 同样过一个低通滤波

    {

        ins.motion_accel_b[i] = (ins.accel[i] - gravity_b[i]) * dt / (ins.accel_lpf + dt) + ins.motion_accel_b[i] * ins.accel_lpf / (ins.accel_lpf + dt);

    }

    BodyFrameToEarthFrame(ins.motion_accel_b, ins.motion_accel_b, ins.q); // 转换回导航系n

    ins.yaw = QEKF_INS.Yaw;

    ins.pitch = QEKF_INS.Pitch;

    ins.roll = QEKF_INS.Roll;

    ins.yaw_total_angle = QEKF_INS.YawTotalAngle;

恒温控制


C型开发板上的 BMI088 周围是又一圈加热电阻的,可以通过 PWM 控制加热功率,从而实现恒温控制,有助于抑制陀螺仪漂移,根据手册提示恒温控制在40摄氏度较好,


static pid_obj_t *imu_temp_pid;

static pid_config_t imu_temp_config = {

.Kp = 50000,

.Ki = 8000,

.Kd = 0,

.IntegralLimit = 50000,

.Improve = PID_Integral_Limit,

.MaxOut = 250000,

};

static rt_err_t temp_pwm_init(rt_uint32_t period, rt_uint32_t pulse)

{

temp_pwm_dev = (struct rt_device_pwm )rt_device_find(TEMP_PWM_DEV_NAME);

if (temp_pwm_dev == RT_NULL)

{

LOG_E("Can't find %s device!", TEMP_PWM_DEV_NAME);

return -RT_ERROR;

}

/ 设置PWM周期和脉冲宽度默认值 /

rt_pwm_set(temp_pwm_dev, TEMP_PWM_DEV_CHANNEL, period, pulse);

/ 使能设备 */

rt_pwm_enable(temp_pwm_dev, TEMP_PWM_DEV_CHANNEL);

}


在 ins_task 中以 500hz 的频率进行恒温控制,需要注意,使用加热电阻外围电路需要给C板额外供电。


void ins_thread_entry(void argument)

{

static uint32_t count;

temp_pwm_init(period, pulse);

/ 注册 PID 实例 */

imu_temp_pid = pid_register(&imu_temp_config);

imu_ops.imu_init();

LOG_I("Example Task Start");

for (;;)

{

example_start = dwt_get_time_ms();

imu_ops.gyro_read(gyro);

imu_ops.accel_read(acc);

if(count % 2 == 0){

temp = imu_ops.temp_read();

pulse = pid_calculate(imu_temp_pid, temp, IMU_TARGET_TEMP);

rt_pwm_set_pulse(temp_pwm_dev, TEMP_PWM_DEV_CHANNEL, pulse);

}

count++;

rt_thread_delay(1);

}

}


实际效果测试如下:


不加恒温控制,室温下(23摄氏度左右)十分钟内,yaw 轴偏移近20度;

加入恒温控制后(40摄氏度),十分钟内,yaw 轴偏移10度

arm_math库使用

有一个小插曲就是arm_math库的移植使用,需要通过修改 Scons 文件,将 arm_math 库链接到工程中,并添加需要的宏定义:


使用 arm_math.h 需要添加相关内核定义


CPPDEFINES = ['ARM_MATH_CM4']

LIBPATH = [cwd + '/arm_math']

LIBS = ['libarm_cortexM4lf_math.a']

path += [cwd + '/arm_math']

group = DefineGroup('RM_Algorithms', src, depend = [''], CPPPATH = path, CPPDEFINES = CPPDEFINES, LIBS = LIBS, LIBPATH=LIBPATH)


到此就可以使用解算得到的欧拉角等数据去对云台等进行闭环控制啦。


存在问题及优化方向


虽然通过陀螺仪校准和恒温控制等有效抑制了零漂,但yaw的零飘依然存在;

之后考虑通过融合磁力计数据,解决零飘问题;


文章来源于:电子工程世界    原文链接
本站所有转载文章系出于传递更多信息之目的,且明确注明来源,不希望被转载的媒体或个人可与我们联系,我们将立即进行删除处理。

我们与500+贴片厂合作,完美满足客户的定制需求。为品牌提供定制化的推广方案、专属产品特色页,多渠道推广,SEM/SEO精准营销以及与公众号的联合推广...详细>>

利用葫芦芯平台的卓越技术服务和新产品推广能力,原厂代理能轻松打入消费物联网(IOT)、信息与通信(ICT)、汽车及新能源汽车、工业自动化及工业物联网、装备及功率电子...详细>>

充分利用其强大的电子元器件采购流量,创新性地为这些物料提供了一个全新的窗口。我们的高效数字营销技术,不仅可以助你轻松识别与连接到需求方,更能够极大地提高“闲置物料”的处理能力,通过葫芦芯平台...详细>>

我们的目标很明确:构建一个全方位的半导体产业生态系统。成为一家全球领先的半导体互联网生态公司。目前,我们已成功打造了智能汽车、智能家居、大健康医疗、机器人和材料等五大生态领域。更为重要的是...详细>>

我们深知加工与定制类服务商的价值和重要性,因此,我们倾力为您提供最顶尖的营销资源。在我们的平台上,您可以直接接触到100万的研发工程师和采购工程师,以及10万的活跃客户群体...详细>>

凭借我们强大的专业流量和尖端的互联网数字营销技术,我们承诺为原厂提供免费的产品资料推广服务。无论是最新的资讯、技术动态还是创新产品,都可以通过我们的平台迅速传达给目标客户...详细>>

我们不止于将线索转化为潜在客户。葫芦芯平台致力于形成业务闭环,从引流、宣传到最终销售,全程跟进,确保每一个potential lead都得到妥善处理,从而大幅提高转化率。不仅如此...详细>>