本科毕业于自动化,期间学习了各种电机运动控制原理,自动控制原理,但是只会考试而未究其理。最近接触到simplefoc这个基于arduino的开源无刷电机驱动库,想正好借此机会将本科学到的内容用于实际,于是就有了这个小项目,基于无刷电机FOC的平衡小车。
无刷电机相比于普通小车用的有刷电机,相同体积下扭矩更大,无刷电机驱动小车时可以直驱而不像有刷电机需要带减速器才能满足扭矩要求,且响应速度更快。采用无刷电机可以把小车做的更紧凑美观。
无刷电机的驱动电路和算法要更加复杂,无刷电机在低速工作时,算法层面上采用磁场定向控制算法FOC(Field-oriented control),可以精确控制电机的位置,速度乃至电流也就是转矩,由此可以做出很多好玩的东西。基于foc的无刷电机在机器人关节,相机云台电机控制等领域得到广泛应用。
博客文档链接
https://blog.csdn.net/weixin_42487906/article/details/124898392
视频链接
https://www.bilibili.com/video/BV1Sv4y137FG/
github仓库,内有代码,3D打印文件,pcb制板文件:
https://github.com/FranHawk/simplefoc_balance_car
立创eda工程 可在线查看原理图和pcb:
总共有三个板子
主板(包括电机控制和驱动部分):https://oshwhub.com/FranHawk/balance_car_esp32_simplefoc
编码器板:https://oshwhub.com/FranHawk/as5600-16mm
RGB灯带板:https://oshwhub.com/FranHawk/rgb_strip
所有的电子元件材料都可以对照BOM表或者工程里的原理图买,在淘宝或者立创上买均可,下面列出一些关键材料的链接
AS5600 L6234PD INA240A1:【淘宝】https://m.tb.cn/h.ft947n2?tk=gn3u2P8IocI「L6234PD L6234D L6234 HSOP 原装汽车IC 进口芯片热卖 质量超好」点击链接直接打开
电机部分的芯片 这个店家买到的货还算便宜
其他的电子元器件都可以在优信电子之类的找到,财力充足的可以直接立创,省事。
2210云台电机(至少两个)【淘宝】https://m.tb.cn/h.fslSDE8?tk=G8Pk2P8HluO「清仓HJ2210-75T无刷云台电机狗3云台专用 航拍电机 云台空心轴马」点击链接直接打开
轮胎胎皮(至少两个) 【淘宝】https://m.tb.cn/h.ft9UFV4?tk=L8Ez2P8u1Ma「AXIAL 1比24车SCX24 90081小米吉姆尼通用橡胶攀爬轮胎皮直径52mm」点击链接直接打开
电机支架(至少两个) 【淘宝】https://m.tb.cn/h.fsl86Kp?tk=i9Oo2P8FvxY「390电机支架 385固定座 365安装底座 380马达固定架 送螺丝螺母 点击链接直接打开
黑色亚克力板(至少两个) 【淘宝】https://m.tb.cn/h.fH9ybkd?tk=2OCV2P8vqAf「平衡小车DIY专用亚克力支架板扩展板」点击链接直接打开
电池 900mah【淘宝】https://m.tb.cn/h.fGk2f0O?tk=33dF2PjbvTN「航模电池5200mAh 11.1V足容高C7.4V车模船模 RC攀爬车质量保证」点击链接直接打开
其他机械小零件
打开开源资料链接上的三个pcb工程,在嘉立创投板生产,把元器件都买到并焊上去
主板
编码器板
编码器板没什么元器件,记得上面这个焊点要焊上,背面还有一个as5600芯片,总共要做两个
RGB灯板
把灯焊上去,没什么好说的,总共要做两个
sh1.0 3P和4P的线
分别做两条,用作连接RGB板和编码器板到主板,注意接线前一定要检查编码器板和主板的线顺序是不是对应的,我买到的反的,需要自己调换顺序
制作两个电机部分
先把径向磁铁粘在电机轴心
把编码器板,电机支架和电机用m2.5螺钉按上图组装,中间加些垫片,防止相互间摩擦,如下图
3D打印轮毂,套在电机上,再套上胎皮,3D打印文件在github里
3D打印固定板,用魔术贴和扎带固定电池,用m3*5 铜柱和螺钉将3D打印固定板和主板组装
将两个电机部分和主板部分用m3螺钉组装,电机线通过3.8mm 插头连接到主板,顺序随意。编码器板用sh1.0 mm 4P线连接,第一层亚克力板上装四个m3 20mm的铜柱,如下图
RGB灯板和第二层亚克力板粘在一起
连接RGB的线到主板,固定第一层和第二层亚克力板,完成组装
通过vscode+platformio来烧写程序
第一次烧写程序前,先找到hw_bldc.cpp里的hw_bldc_init()函数里
有这样几句
motor_0.init();
motor_1.init();
motor_0.initFOC();
motor_1.initFOC();
第一次烧写时,initFOC里不需要有参数,上电后simplefoc会运行自检程序,检测电机旋转方向和电角度,会在串口中输出,通过串口得到参数后填入initFOC
motor_0.init();
motor_1.init();
motor_0.initFOC(4.53, Direction::CW);
motor_1.initFOC(0.64, Direction::CCW);
如上,每个人的这两个参数都是不一样的,然后第二次烧写代码。之后每次启动就不会消耗时间自检了。
手机安装点灯APP,https://diandeng.tech/dev
创建一个蓝牙设备,添加一个手柄控件,起名叫JoyKey
对于灯光效果的遥控可自行开发
电路部分分为一个主板,两个编码器小板,两个RGB灯板
驱动是小车的核心部分,是小车动起来的关键。无刷电机转动时需要分别控制电机的三相电压。采用FOC算法控制时,还需要位置传感器提供转子和定子的相对位置,如果要控制转矩,还需要电流传感器。初次学习FOC,我采用灯哥开源FOC驱动的方案,灯哥github仓库地址。
电机驱动采用L6234P三相H桥集成驱动芯片
电流检测采用INA240A2芯片,放大增益为50,这个芯片也比较贵,可以用INA240A1代替,放大增益为20,我焊的时候用的就是这个,一样能用。
电机位置检测采用AS5600绝对值磁编码器芯片,IIC通讯,我为这个芯片画了个小板子附在电机轴处,与主板相连,同时还需要在电机轴线处固定一个径向磁铁。
由主要功能可知,需要MCU具有蓝牙和WIFI功能,还需要足够的性能用于FOC算法,同时控制RGB。最近看到的esp32可以说正合适,esp32有320K RAM,4M ROM,两个core,一个核专门用来跑foc算法,另外一个核用来跑其他程序包括RGB控制,simplefoc官网上说esp32单次foc运算能有1ms,引脚也足够多,满足平衡控制的需求。
平衡控制需要通过传感器检测到小车自身的姿态,我采用最常见的MPU6050 6轴加速度计陀螺仪芯片
WIFI图传由主板通过2.54排母插接ESP-CAM模块实现
RGB采用WS1802B LED灯,MCU一个IO口就可以驱动一串RGB
其他一些接口和转串口芯片什么就不多介绍了,可以看我开源的电路工程了解
无刷电机平衡小车的理论部分需要了解
1.无刷电机磁场定向控制FOC算法
2.平衡车建模和控制算法
理论部分网上有很多资料,讲解也非常清楚易懂,珠玉在前,我引出一些比较好的文章,供想要深究的读者查看
FOC算法的内容比较多,初学者可以看下稚晖君这篇文章,比较通俗易懂
simplefoc官方文档的介绍也比较详细,还有源码讲解,代码中也用到了simplefoc的库,所以最好去官网文档了解一些simplefoc的使用方法
https://simplefoc.com/
其他比较好的文章
ESP32这款MCU有两个核心,我采用freertos来对两个核进行调度,整体软件框架基于arduino+freertos
Adafruit_NeoPixel 用于驱动RGB灯带
blinker-library点灯科技的蓝牙库,可以通过手机app和平衡车蓝牙通讯
MPU6050_lightMPU6050驱动库
Arduino-FOCsimplefoc的arduino支持库
0 core:
task 0:simplefoc的电流环的运算、俯仰角数据读取的和电机电压的输出
1 core:
task 1:平衡环和速度环pid运算
task 2:RGB闪烁和通过蓝牙接收手机的控制命令
这个任务的代码比较简单,loopFOC和move负责simplefoc电流环的运算和电压输出
mpu.update()读出MPU6050数据并返回俯仰角,俯仰角通过互补滤波结算得到就可以了
shaftVelocity返回电机速度,用于task 1速度环pid的运算
void task_motor_move(void *pvParameters)
{
while (1)
{
// iterative setting FOC phase voltage
motor_1.loopFOC();
motor_0.loopFOC();
// Serial.printf("d:%f,%f\n",motor_1.shaftVelocity(),motor_0.shaftVelocity());
motor_1.move(motor_output_left);
motor_0.move(motor_output_right);
// // iterative function setting the outter loop target
mpu.update();
speed_right = motor_0.shaftVelocity();
speed_left = motor_1.shaftVelocity();
}
}
该任务主要是平衡环和速度环pid运算,用状态机的思维将平衡车状态分为STOP,RUNNING,PICKUP三个状态,这样做主要是为了避免平衡车被拿起时电机疯转。
开机时状态为STOP,把车扶正到平衡位置附近则进入RUNNING状态,当拿起倾斜角后过大,则进入PICKUP状态,逆时针旋转车体后进入STOP状态。
RUNNING状态运行平衡环,速度环和角度环。平衡环采用pd,速度环用pi,角度环仅用单p。
void task_control(void *pvParameters)
{
while (1)
{
angle_pitch = mpu.getAngleY();
// Serial.printf("d:%f\n",angle_pitch);
acc_Z = mpu.getAccZ();
if (running_state == STATE_STOP)
{
if (abs(angle_pitch - angle_pitch_offset) < 0.5)
{
running_state = STATE_RUNNING;
}
}
else if (running_state == STATE_RUNNING)
{
speed_average = (speed_right + speed_left) / 2;
if (abs(angle_pitch - angle_pitch_offset) > 50 )
{
speed_I_sum=0;//清空积分系数
running_state = STATE_PICKUP;
}
else
{
speed_target = translate_speed;
speed_error = speed_target - speed_average;
speed_I_sum+=(speed_error*speed_I);
if(speed_I_sum>15){
speed_I_sum = 15;
}
if(speed_I_sum<-15){
speed_I_sum = -15;
}
angle_target_pitch = speed_P * speed_error + speed_I_sum;
motor_output_left = angle_P * ((angle_pitch - angle_pitch_offset) - angle_target_pitch) + angle_D * mpu.getGyroY()+rotate_speed; //+ angle_D * mpu.getGyroY()
motor_output_right = angle_P * ((angle_pitch - angle_pitch_offset) - angle_target_pitch) + angle_D * mpu.getGyroY()-rotate_speed;
angle_last_pitch = angle_pitch;
}
}
else if (running_state == STATE_PICKUP)
{
motor_output_left = 0;
motor_output_right = 0;
if (speed_right > 3.14 && speed_left < -3.14)
{
running_state = STATE_STOP;
}
}
vTaskDelay(1);
}
}
task 2用于RGB效果显示,和接收手机blinker app通过蓝牙发送的数据。一般的RGB等效都有一个延时函数,普通的延时函数都是在原地死等,这样大大降低CPU效率,影响其他程序运行,这里我编写了一系列非阻塞式RGB灯效:在显示完灯效后不再原地延时,而是直接进行调度,切换到其他task。详细可看我代码。
void task_neopixel(void *pvParameters)
{
uint32_t neopixel_pattern_previous = 0;
uint32_t neopixel_pattern_interval = 500;
uint32_t neopixel_pattern_number = 0;
while (1)
{
// 效果切换
if((neopixel_count-neopixel_pattern_previous)>neopixel_pattern_interval){
neopixel_pattern_previous = neopixel_count;
neopixel_pattern_number++;
neopixel_state_clear(strip_1_handler);
neopixel_state_clear(strip_2_handler);
if(neopixel_pattern_number>4){
neopixel_pattern_number = 0;
}
}
switch(neopixel_pattern_number){
case 0:
neopixel_police_style_2(strip_2_handler,10,5);
neopixel_police_style_2(strip_1_handler,10,5);
break;
case 1:
neopixel_police_style_1(strip_2_handler,10,5);
neopixel_police_style_1(strip_1_handler,10,5);
break;
case 2:
neopixel_loop(strip_2_handler, 255, 255, 255, 4, 4);
neopixel_rainbow(strip_1_handler);
break;
case 3:
neopixel_bounce(strip_1_handler, 0, 255, 0, 2, 3, 0);
neopixel_fade_in_out(strip_2_handler, 255, 0, 255, 2, 1,25);
break;
case 4:
neopixel_spread_out(strip_2_handler, strip_2_handler.strip.Color(0,255,0), strip_2_handler.strip.Color(0,0,255), 4, 10);
neopixel_spread_out(strip_1_handler, strip_1_handler.strip.Color(255,0,0), strip_1_handler.strip.Color(255,255,0), 4, 10);
break;
default:
break;
}
neopixel_count++;
#ifdef USE_BLINKER
Blinker.run();
#endif
vTaskDelay(10);
}
}