小米电机驱动(STM32F4 HAL库)使用教程 - LJW0401/MotorDrive GitHub Wiki
在使用小米电机的时候我们通常会选择一种模式进行控制,可以参考示例代码中的各种模式控制代码。
除了发送控制指令对电机进行控制外,我们通常还需要获取电机的状态,这时候就需要用到CAN中断回调函数来接收电机返回的信息,同样可以参考示例代码中的中断回调函数代码。
小米电机结构体
- 我们定义了小米电机结构体来标识电机的数据。
MI_Motor_t
- 使用对应的初始化函数进行初始化。
MI_motor_Init()
依据CyberGear官方文档我们写了对应的几种通信类型
- 通信类型0:获取设备 ID,获取设备的ID和64位MCU唯一标识符。
MI_motor_GetID()
- 通信类型1: 运控模式电机控制指令,用来向电机发送控制指令 。
MI_motor_Control()
- 通信类型2:电机反馈数据,在CAN中断回调函数中使用,用来向主机反馈电机运行状态。
MI_motor_RxDecode()
- 通信类型3:电机使能运行,启动电机。
MI_motor_Enable()
- 通信类型4:电机停止运行。
MI_motor_Stop()
- 通信类型6:设置电机机械零位,会把当前电机位置设为机械零位(掉电丢失) 。
MI_motor_SetMechPositionToZero()
- 通信类型7: 设置电机 CAN_ID,更改当前电机 CAN_ID , 立即生效。
MI_motor_ChangeID()
- 通信类型17:单个参数读取。
MI_motor_ReadParam()
- 通信类型18:单个参数写入。
MI_motor_ModeSwitch() MI_motor_WritePram()
同时我们也封装了不同控制模式的函数,直接使用即可。
因为在2023.10.11的测试中发现之前按照小米官方文档写的函数驱动多台电机时出现问题,参考小米电机CyberGear STM32HAL 使用指南后修改了位置模式和速度模式的控制方式,至于依据小米官方文档的控制方式继续研究中。
- 力矩模式
MI_motor_TorqueControl()
- 位置模式
MI_motor_LocationControl()
- 速度模式
MI_motor_SpeedControl()
为了接收电机的数据,我们还提供了CAN中断回调函数的示例代码
#include "MI_motor_drive.h"
extern CAN_HandleTypeDef hcan1;
MI_Motor_t MI_motor;
void control_task(void const *pvParameters)
{
MI_motor_Init(&MI_Motor,&MI_CAN_1,1);//小米电机结构体初始化
MI_motor_Enable(&MI_Motor);//电机使能
while (1)
{
float torque = 1.0f;//设定输出力矩为1N*m
MI_motor_TorqueControl(&MI_Motor, torque);//发送控制信号
vTaskDelay(4);//系统延时(os delay)
}
}
#include "MI_motor_drive.h"
extern CAN_HandleTypeDef hcan1;
MI_Motor_t MI_motor;
void control_task(void const *pvParameters)
{
MI_motor_Init(&MI_Motor,&MI_CAN_1,1);//小米电机结构体初始化
MI_motor_Enable(&MI_Motor);//电机使能
while (1)
{
float location = 1;//设定停止在 1rad 的位置
MI_motor_LocationControl(&MI_Motor,location,5,0.5);
vTaskDelay(4);//系统延时(os delay)
}
}
#include "MI_motor_drive.h"
extern CAN_HandleTypeDef hcan1;
MI_Motor_t MI_motor;
void control_task(void const *pvParameters)
{
MI_motor_Init(&MI_Motor,&MI_CAN_1,1);//小米电机结构体初始化
MI_motor_Enable(&MI_Motor);//电机使能
while (1)
{
float speed = 2;//设定运行速度为 2rad/s
MI_motor_SpeedControl(&MI_Motor,speed,1);
vTaskDelay(4);//系统延时(os delay)
}
}
/**
* @brief hal库CAN回调函数,接收电机数据
* @param[in] hcan:CAN句柄指针
* @retval none
*/
void HAL_CAN_RxFifo0MsgPendingCallback(CAN_HandleTypeDef *hcan)
{
CAN_RxHeaderTypeDef rx_header;
RxCAN_info_s RxCAN_info;//用于存储小米电机反馈的数据
uint8_t rx_data[8];
HAL_CAN_GetRxMessage(hcan, CAN_RX_FIFO0, &rx_header, rx_data);
memcpy(&RxCAN_info,&rx_header.ExtId,4);//将扩展标识符的内容解码到缓存区,获取通信类型
if(RxCAN_info.communication_type == 0){//通信类型0的反馈帧解码
RxCAN_info_type_0_s RxCAN_info_type_0;
memcpy(&RxCAN_info_type_0,&rx_header.ExtId,4);//将扩展标识符的内容解码成通信类型0的对应内容
memcpy(&RxCAN_info_type_0.MCU_id,rx_data,8);//获取MCU标识符
OutputData.data_3 = RxCAN_info_type_0.motor_id;
}else if(RxCAN_info.communication_type == 2){//通信类型2的反馈帧解码
RxCAN_info_type_2_s RxCAN_info_type_2;
memcpy(&RxCAN_info_type_2,&rx_header.ExtId,4);//将扩展标识符的内容解码成通信类型2的对应内容
MI_motor_RxDecode(&RxCAN_info_type_2,rx_data);//通信类型2的数据解码
if (RxCAN_info_type_2.motor_id == 1){
MI_Motor.RxCAN_info = RxCAN_info_type_2;
MI_Motor.motor_mode_state = RxCAN_info_type_2.mode_state;
}
}else if(RxCAN_info.communication_type == 17){//通信类型17的反馈帧解码
RxCAN_info_type_17_s RxCAN_info_type_17;
memcpy(&RxCAN_info_type_17,&rx_header.ExtId,4);//将扩展标识符的内容解码成通信类型17的对应内容
memcpy(&RxCAN_info_type_17.index,&rx_data[0],2);//获取查找的参数索引码
memcpy(&RxCAN_info_type_17.param,&rx_data[4],4);//获取查找的参数信息
}
}
CyberGear微电机使用说明书.pdf
小米微电机STM32 HAL库驱动教程
小米电机CyberGear STM32HAL 使用指南
通信类型 0:获取设备 ID,获取设备的ID和64位MCU唯一标识符
数据域 | ExtID(29 bit) | 数据区(8 Byte) | ||
---|---|---|---|---|
数据域 | 0~7 | 8~23 | 24~28 | 0~7 |
描述 | 目标电机CAN_ID | 8~15:用来标识主机CAN_ID 15~23:0 |
0 | 0 |
数据域 | ExtID(29 bit) | 数据区(8 Byte) | ||
---|---|---|---|---|
数据域 | 0~7 | 8~23 | 24~28 | 0~7 |
描述 | 0xFE | 目标电机 CAN_ID | 0 | 64 位 MCU 唯一标识符 |
通信类型 1:运控模式电机控制指令,用来向电机发送控制指令
数据域 | ExtID(29 bit) | 数据区(8 Byte) | ||
---|---|---|---|---|
数据域 | 0~7 | 8~23 | 24~28 | 0~7 |
描述 | 目标电机CAN_ID | 力矩(0~65535) 对应(-12Nm~12Nm) |
1 | 0~1: 目标角度[0~65535]对应(-4π~4π) 2~3: 目标角速度[0~65535]对应(-30rad/s~30rad/s) 4~5:Kp [0~65535]对应(0.0~500.0) 6~7:Kd [0~65535]对应(0.0~5.0) |
通信类型 2:电机反馈数据,用来向主机反馈电机运行状态
数据域 | ExtID(29 bit) | 数据区(8 Byte) | ||
---|---|---|---|---|
数据域 | 0~7 | 8~23 | 24~28 | 0~7 |
描述 | 目标电机CAN_ID | 8~15:当前电机 CAN ID bit21~16:故障信息(0 无 1 有) bit21: 未标定 bit20: HALL 编码故障 bit19: 磁编码故障 bit18: 过温 bit17: 过流 bit16: 欠压故障 bit22~23:模式状态 0 : Reset 模式[复位] 1 : Cali 模式[标定] 2 : Motor 模式[运行] |
2 | 0~1: 当前角度[0~65535]对应(-4π~4π) 2~3: 当前角速度[0~65535]对应(-30rad/s~30rad/s) 4~5:当前力矩[0~65535]对应(-12Nm~12Nm) 6~7:当前温度:Temp(摄氏度)*10 |
通信类型 3:电机使能运行
数据域 | ExtID(29 bit) | 数据区(8 Byte) | ||
---|---|---|---|---|
数据域 | 0~7 | 8~23 | 24~28 | 0~7 |
描述 | 目标电机CAN_ID | 用来标识主CAN_ID | 3 | 0 |
通信类型 4:电机停止运行
数据域 | ExtID(29 bit) | 数据区(8 Byte) | ||
---|---|---|---|---|
数据域 | 0~7 | 8~23 | 24~28 | 0~7 |
描述 | 目标电机CAN_ID | 用来标识主CAN_ID | 4 | 正常运行时,data 区需清 0; Byte[0]=1 时:清故障; |
通信类型 6:设置电机机械零位,会把当前电机位置设为机械零位(掉电丢失)
数据域 | ExtID(29 bit) | 数据区(8 Byte) | ||
---|---|---|---|---|
数据域 | 0~7 | 8~23 | 24~28 | 0~7 |
描述 | 目标电机CAN_ID | 用来标识主CAN_ID | 6 | Byte[0]=1 |
通信类型 7:设置电机 CAN_ID,更改当前电机 CAN_ID , 立即生效。
数据域 | ExtID(29 bit) | 数据区(8 Byte) | ||
---|---|---|---|---|
数据域 | 0~7 | 8~23 | 24~28 | 0~7 |
描述 | 目标电机CAN_ID | 8~15:用来标识主CAN_ID 16~23: 预设置 CAN_ID |
7 | 0 |
通信类型 17:单个参数读取
数据域 | ExtID(29 bit) | 数据区(8 Byte) | ||
---|---|---|---|---|
数据域 | 0~7 | 8~23 | 24~28 | 0~7 |
描述 | 目标电机CAN_ID | 8~15:用来标识主CAN_ID | 17 | 0~1: index 2~3: 00 4~7: 00 |
数据域 | ExtID(29 bit) | 数据区(8 Byte) | ||
---|---|---|---|---|
数据域 | 0~7 | 8~23 | 24~28 | 0~7 |
描述 | 电机CAN_ID | 8~15:用来标识主CAN_ID | 17 | 0~1: index,参数列表详见 4.1.11 2~3: 00 4~7: 参数数据,1 字节数据在 Byte4 |
通信类型 18:单个参数写入(掉电丢失)
数据域 | ExtID(29 bit) | 数据区(8 Byte) | ||
---|---|---|---|---|
数据域 | 0~7 | 8~23 | 24~28 | 0~7 |
描述 | 目标电机CAN_ID | 8~15:用来标识主CAN_ID | 18 | 0~1: index 2~3: 00 4~7: 参数数据 |
通信类型 21: 故障反馈帧
数据域 | ExtID(29 bit) | 数据区(8 Byte) | ||
---|---|---|---|---|
数据域 | 0~7 | 8~23 | 24~28 | 0~7 |
描述 | 目标电机CAN_ID | 8~15:用来标识主CAN_ID | 21 | Byte0~3: fault 值(非 0:有故障,0:正常) bit0:电机过温故障,默认 80 度 bit1:驱动芯片故障 bit2:欠压故障 bit3:过压故障 bit4:B 相电流采样过流 bit5:C 相电流采样过流 bit7:编码器未标定 bit15~bit8:过载故障 bit16:A 相电流采样过流 Byte4~7: warning 值 bit0:电机过温预警,默认 75 度 |