🎉欢迎来到FPGA专栏~基于FPGA的循迹小车

🎉 基于FPGA的循迹小车

基于FPGA的循迹小车【Spirit_V2】
由于场地的问题,打滑现象严重,所以用手轻推辅助小车前进。
对于效果演示的分析,见调试及结果分析。
该系列文章只是作为学习记录,并无其余用途。所发文章内容是经过自己本身操作和记录整理得来。
本篇文章记录基于小精灵V2(Spirit_V2)开发板的循迹小车。
❤️特别鸣谢:小月电子工作室
🔸【小月电子】大佬博客链接:Moon_3181961725
🔸【FPGA】Altera Cyclone IV EP4CE6入门系统板购买链接:EP4CE6
🔸【FPGA】智能小车驱动板购买链接:FPGA小车驱动版
🔸【FPGA】配套电池与充电器购买链接:电池与充电器
❤️有不清楚的地方可以咨询客服:小月电子工作室淘宝店铺
开发板详细介绍:【FPGA-Spirit_V2】小精灵V2开发板初使用
一套完整的循迹小车包括:核心控制板、小车驱动板、电源和循迹模块。
红外循迹模块基本介绍和使用:红外循迹模块使用介绍。
按照店铺提供的资料,组装好小车驱动板,加上循迹模块,效果如下:

小车烧录程序时的连线图:

本次项目的编程思路见下图的思维导图:

对于循迹过程,使用红外传感器来判断是否检测到黑线:如果检测到黑线,模块输出低电平;如果没有检测到黑线,模块输出高电平。
使用最简单的电机差速来控制小车的循迹过程:如果左侧的传感器检测到黑线,那么左侧电机停止转动,右侧电机继续转动;如果右侧的传感器检测到黑线,那么右侧电机停止转动,左侧电机继续转动;如果两个传感器都没有检测到黑线,那么两个电机以相同的速度同时运行。
该项目主要需要使用到PWM模块和LCD模块。
PWM是用来控制电机转速的。在使用电机差速来控制循迹时,也可以让一侧的电机转速低于另一侧,而不是使一侧停止转动,使用该方法可以更加丝滑快速地完成转向。
小月电子提供的PWM模块:
//脉冲生成模块,通过控制输出脉冲频率及占空比来控制小车的速度
module ctrl_moto_pwm(
input clk ,//时钟50M
input rst_n ,//复位,低电平有效
input [7:0] spd_high_time,
input [7:0] spd_low_time ,
output period_fini,
output reg pwm //脉冲信号
);
//状态机
parameter idle = 8'h0,//空闲状态
step_high = 8'h1,//脉冲高电平状态,当为该状态时,pwm为高电平
step_low = 8'h2;//脉冲低电平状态,当为该状态时,pwm为低电平
reg [7:0] curr_st ;
reg [7:0] curr_st_ff1 ;
reg [10:0] step_high_time ;
reg [10:0] step_low_time ;
reg [10:0] step_high_cnt ;
reg [10:0] step_low_cnt ;
// wire[10:0] spd_high_time;
// wire[10:0] spd_low_time ;
wire[10:0] hspd_high_time;
wire[10:0] hspd_low_time ;
// assign spd_high_time=20;
// assign spd_low_time =10;
assign period_fini=(step_low_cnt==step_low_time)?1'b1:1'b0;
always@(posedge clk or negedge rst_n)
begin
if(!rst_n)
begin
step_high_time<=0;
step_low_time<=0;
end
else
begin
step_high_time<=spd_high_time;
step_low_time<=spd_low_time;
end
end
always@(posedge clk)curr_st_ff1<=curr_st ;
always@(posedge clk or negedge rst_n)
begin
if(!rst_n)
curr_st<=idle;
else case(curr_st)
idle:curr_st<=step_high ;
step_high:
begin
if(step_high_cnt==step_high_time)
curr_st<=step_low;
else
;
end
step_low:
begin
if(step_low_cnt==step_low_time)
curr_st<=step_high;
else
;
end
default:;
endcase
end
always@(posedge clk or negedge rst_n)
begin
if(!rst_n)
step_high_cnt<=0;
else if(curr_st==idle)
step_high_cnt<=0;
else if(curr_st==step_high)
step_high_cnt<=step_high_cnt+1;
else
step_high_cnt<=0;
end
always@(posedge clk or negedge rst_n)
begin
if(!rst_n)
step_low_cnt<=0;
else if(curr_st==idle)
step_low_cnt<=0;
else if(curr_st==step_low)
step_low_cnt<=step_low_cnt+1;
else
step_low_cnt<=0;
end
always@(posedge clk or negedge rst_n)
begin
if(!rst_n)
begin
pwm<=0;
end
else if(curr_st==idle)
begin
pwm<=0;
end
else if(curr_st==step_high)
pwm<=1;
else if(curr_st==step_low)
pwm<=0;
else
pwm<=1;
end
endmodule
LCD液晶屏模块主要是用来显示数据的,通过显示的数据,可以帮助我们进行更高效的调试。在该项目中,LCD只用来显示字符。
LCD1602模块:
//模块介绍:LCD1602显示驱动
module lcd (
input clk ,//系统时钟输入50M
input rst_n ,//复位,低电平有效
output reg [7:0] dat ,//LCD的8位数据口
output reg rs ,//数据命令选择信号,高电平表示数据,低电平表示命令
output rw ,//读写标志,高电平表示读,低电平表示写,该程序我们只对液晶屏进行写操作
output en //LCD的控制脚
);
reg [15:0] counter ;
reg [ 5:0] current ;
reg clkr ;
reg e ;
//定义了LCD状态机需要的状态。
parameter set0 =6'd0;
parameter set1 =6'd1;
parameter set2 =6'd2;
parameter set3 =6'd3;
parameter set4 =6'd4;
parameter dat0 =6'd5;
parameter dat1 =6'd6;
parameter dat2 =6'd7;
parameter dat3 =6'd8;
parameter dat4 =6'd9;
parameter dat5 =6'd10;
parameter dat6 =6'd11;
parameter dat7 =6'd12;
parameter dat8 =6'd13;
parameter dat9 =6'd14;
parameter dat10=6'd15;
parameter dat11=6'd16;
parameter dat12=6'd17;
parameter dat13=6'd18;
parameter dat14=6'd19;
parameter dat15=6'd20;
parameter fini=6'hF1;
always @(posedge clk or negedge rst_n) //da de data_w1 zhong pinlv
begin
if(!rst_n)
begin
counter<=0;
clkr<=0;
end
else
begin
counter<=counter+1;
if(counter==16'h000f)
clkr=~clkr;
else
;
end
end
always @(posedge clkr or negedge rst_n)
begin
if(!rst_n)
begin
current<=set0;
dat<=0;
rs<=0;
e<=1;
end
else
begin
case(current)
set0: begin e<=0;rs<=0; dat<=8'h38; current<=set1; end //*设置8位格式,2行,5*7*
set1: begin e<=0;rs<=0; dat<=8'h0C; current<=set2; end //*整体显示,关光标,不闪烁*/
set2: begin e<=0;rs<=0; dat<=8'h06; current<=set3; end //*设定输入方式,增量不移位*/
set3: begin e<=0;rs<=0; dat<=8'h01; current<=set4; end //*清除显示*/
set4: begin e<=0;rs<=0; dat<=8'h00; current<=dat0; end //设置显示第一行
dat0: begin e<=0;rs<=1; dat<="H"; current<=dat1; end
dat1: begin e<=0;rs<=1; dat<="E"; current<=dat2; end
dat2: begin e<=0;rs<=1; dat<="L"; current<=dat3; end
dat3: begin e<=0;rs<=1; dat<="L"; current<=dat4; end
dat4: begin e<=0;rs<=1; dat<="O"; current<=dat5; end
dat5: begin e<=0;rs<=1; dat<=" "; current<=dat6; end
dat6: begin e<=0;rs<=1; dat<="S"; current<=dat7; end
dat7: begin e<=0;rs<=1; dat<="p"; current<=dat8; end
dat8: begin e<=0;rs<=1; dat<="i"; current<=dat9; end
dat9: begin e<=0;rs<=1; dat<="r"; current<=dat10 ; end
dat10: begin e<=0;rs<=1; dat<="i"; current<=dat11; end
dat11: begin e<=0;rs<=1; dat<="t"; current<=dat12; end
dat12: begin e<=0;rs<=1; dat<="_"; current<=dat13; end
dat13: begin e<=0;rs<=1; dat<="V"; current<=dat14; end
dat14: begin e<=0;rs<=1; dat<="2"; current<=dat15; end
dat15: begin e<=0;rs<=1; dat<="!"; current<=fini; end
fini: begin e<=1;rs<=0; dat<=8'h00; end
default: current<=set0;
endcase
end
end
assign en=clkr|e;
assign rw=0;
endmodule
上述代码的LCD显示效果如下:

该项目中的顶层模块主要用于控制循迹过程。
同时,通过板载的按键模块来控制小车的启动与停止:

板载按键模块低电平时有效,代码:
//通过标志位flag判断小车是否开启运行
always@(posedge clk or negedge rst_n)begin
if(!rst_n)
flag <= 0;
else if(!key && !flag)
flag <= 1;
else
flag <= flag;
end
最后,加入蜂鸣器来更好地判断小车是否检测到黑线。板载蜂鸣器为有源蜂鸣器,低电平时发出声音,控制代码:
always@(posedge clk or negedge rst_n)begin
if(!rst_n)
beep_key <= 0;
else if(track1 == 1 && track2 == 0)
beep_key <= 1;
else if(track1 == 0 && track2 == 1)
beep_key <= 1;
else if(track3 == 1 && track4 == 0)
beep_key <= 1;
else if(track3 == 0 && track4 == 1)
beep_key <= 1;
else
beep_key <= 0;
end
//加入蜂鸣器来查看小车的转向情况
assign beep = beep_key?1'b0:1'b1;
完整的顶层模块代码:
module car_top(
input clk,//50MHZ
input rst_n,//全局复位,低电平有效
input key,//控制小车是否开始运行
input track1,
input track2,
input track3,
input track4,
output reg dir_l_1=0,//控制左电机正转或者反转
output reg dir_l_2=0,
output reg dir_r_1=0,//控制右电机正转或者反转
output reg dir_r_2=0,
output reg f_pwm_l=0,//左电机pwm值
output reg f_pwm_r=0,//右电机pwm值
output [7:0]udat,//LCD的8位数据口
output urs,//数据命令选择信号,高电平表示数据,低电平表示命令
output urw,//读写标志,高电平表示读,低电平表示写,该程序我们只对液晶屏进行写操作
output uen,//LCD的控制脚
output beep
);
reg [1:0] flag = 0;
reg [1:0] beep_key = 0;
ctrl_moto_pwm uctrl_moto_pwm(
.clk(clk),//时钟50M
.rst_n(rst_n),//复位,低电平有效
.spd_high_time(10),
.spd_low_time(75),
.period_fini(),
.pwm(pwm)//脉冲信号
);
lcd Ulcd(
.clk(clk),
.rst_n(rst_n),
.dat(udat),
.rs(urs),
.rw(urw),
.en(uen)
);
//通过标志位flag判断小车是否开启运行
always@(posedge clk or negedge rst_n)begin
if(!rst_n)
flag <= 0;
else if(!key && !flag)
flag <= 1;
else
flag <= flag;
end
//循迹与差速转向
always@(posedge clk)begin
//当标志位flag为0时,小车静止
if(flag == 0)begin
dir_l_1<=0;
dir_l_2<=0;
f_pwm_l<=0;
dir_r_1<=0;
dir_r_2<=0;
f_pwm_r<=0;
end
//当标志位flag为1时,小车开始循迹
else if(flag==1)begin
//当两侧传感器都没有检测到黑线时,两轮同时行进
if(track1 == 0 && track2 == 0)begin
dir_l_1<=0;
dir_l_2<=1;
f_pwm_l<=pwm;
dir_r_1<=0;
dir_r_2<=1;
f_pwm_r<=pwm;
end
//内-向右调整
else if(track1 == 0 && track2 == 1)begin
dir_l_1<=0;
dir_l_2<=1;
f_pwm_l<=pwm;
dir_r_1<=0;
dir_r_2<=0;
f_pwm_r<=0;
end
//内-向左调整
else if(track1 == 1 && track2 == 0)begin
dir_l_1<=0;
dir_l_2<=0;
f_pwm_l<=0;
dir_r_1<=0;
dir_r_2<=1;
f_pwm_r<=pwm;
end
//外-向右调整
else if(track3 == 1 && track4 == 0 && track1 == 1 && track2 == 1)begin
dir_l_1<=0;
dir_l_2<=0;
f_pwm_l<=0;
dir_r_1<=0;
dir_r_2<=1;
f_pwm_r<=pwm;
end
//外-向左调整
else if(track3 == 0 && track4 == 1 && track1 == 1 && track2 == 1)begin
dir_l_1<=0;
dir_l_2<=1;
f_pwm_l<=pwm;
dir_r_1<=0;
dir_r_2<=0;
f_pwm_r<=0;
end
//当内部两个红外传感器都检测不到黑线时,停止运动
else if(track1 == 1 && track2 == 1)begin
dir_l_1<=0;
dir_l_2<=0;
f_pwm_l<=0;
dir_r_1<=0;
dir_r_2<=0;
f_pwm_r<=0;
end
end
else;
end
always@(posedge clk or negedge rst_n)begin
if(!rst_n)
beep_key <= 0;
else if(track1 == 1 && track2 == 0)
beep_key <= 1;
else if(track1 == 0 && track2 == 1)
beep_key <= 1;
else if(track3 == 1 && track4 == 0)
beep_key <= 1;
else if(track3 == 0 && track4 == 1)
beep_key <= 1;
else
beep_key <= 0;
end
//加入蜂鸣器来查看小车的转向情况
assign beep = beep_key?1'b0:1'b1;
endmodule
RTL视图:

本项目使用了四个循迹模块(四路循迹),在代码编写的过程中也可以把循迹部分和蜂鸣器部分单独封装成模块的形式,方便在顶层模块的调用。
🔸需要根据黑线的宽度调整好红外模块的位置和方向;
🔸调整好合适的红外模块灵敏度(电阻值);
🔸选择一个合适的循迹场地,地面平整,不会打滑;
🔸调整好合适的PWM。
🔸本次测试场地会导致打滑现象的出现;
🔸没有调整好合适的PWM,导致小车前轮有时几乎不转;
🔸小车在行驶过程中遇到阻碍使轮子停止转动时,需要立刻将小车调入待机状态或者关闭电源,否则会导致电机驱动模块发烫。

🧸结尾
导读:随着叮咚买菜业务的发展,不同的业务场景对数据分析提出了不同的需求,他们希望引入一款实时OLAP数据库,构建一个灵活的多维实时查询和分析的平台,统一数据的接入和查询方案,解决各业务线对数据高效实时查询和精细化运营的需求。经过调研选型,最终引入ApacheDoris作为最终的OLAP分析引擎,Doris作为核心的OLAP引擎支持复杂地分析操作、提供多维的数据视图,在叮咚买菜数十个业务场景中广泛应用。作者|叮咚买菜资深数据工程师韩青叮咚买菜创立于2017年5月,是一家专注美好食物的创业公司。叮咚买菜专注吃的事业,为满足更多人“想吃什么”而努力,通过美好食材的供应、美好滋味的开发以及美食品牌的孵
C#实现简易绘图工具一.引言实验目的:通过制作窗体应用程序(C#画图软件),熟悉基本的窗体设计过程以及控件设计,事件处理等,熟悉使用C#的winform窗体进行绘图的基本步骤,对于面向对象编程有更加深刻的体会.Tutorial任务设计一个具有基本功能的画图软件**·包括简单的新建文件,保存,重新绘图等功能**·实现一些基本图形的绘制,包括铅笔和基本形状等,学习橡皮工具的创建**·设计一个合理舒适的UI界面**注明:你可能需要先了解一些关于winform窗体应用程序绘图的基本知识,以及关于GDI+类和结构的知识二.实验环境Windows系统下的visualstudio2017C#窗体应用程序三.
1.错误信息:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:requestcanceledwhilewaitingforconnection(Client.Timeoutexceededwhileawaitingheaders)或者:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:TLShandshaketimeout2.报错原因:docker使用的镜像网址默认为国外,下载容易超时,需要修改成国内镜像地址(首先阿里
需求:要创建虚拟机,就需要给他提供一个虚拟的磁盘,我们就在/opt目录下创建一个10G大小的raw格式的虚拟磁盘CentOS-7-x86_64.raw命令格式:qemu-imgcreate-f磁盘格式磁盘名称磁盘大小qemu-imgcreate-f磁盘格式-o?1.创建磁盘qemu-imgcreate-fraw/opt/CentOS-7-x86_64.raw10G执行效果#ls/opt/CentOS-7-x86_64.raw2.安装虚拟机使用virt-install命令,基于我们提供的系统镜像和虚拟磁盘来创建一个虚拟机,另外在创建虚拟机之前,提前打开vnc客户端,在创建虚拟机的时候,通过vnc
我正在寻找用于Rails的优质管理插件。似乎大多数现有的插件/gem(例如“restful_authentication”、“acts_as_authenticated”)都围绕着self注册等展开。但是,我正在寻找一种功能齐全的基于管理/管理角色的解决方案——但不是简单地附加到另一个非基于角色的解决方案。如果我找不到,我想我会自己动手......只是不想重新发明轮子。 最佳答案 RyanBates最近做了两个关于授权的railscast(注意身份验证和授权之间的区别;身份验证检查用户是否如她所说的那样,授权检查用户是否有权访问资源
我正在根据Rakefile中的现有测试文件动态生成测试任务。假设您有各种以模式命名的单元测试文件test_.rb.所以我正在做的是创建一个以“测试”命名空间内的文件名命名的任务。使用下面的代码,我可以用raketest:调用所有测试require'rake/testtask'task:default=>'test:all'namespace:testdodesc"Runalltests"Rake::TestTask.new(:all)do|t|t.test_files=FileList['test_*.rb']endFileList['test_*.rb'].eachdo|task|n
我想要像“嘿那里”这样的东西变成,例如,#316583。我希望将任意长度的字符串“归结”为十六进制颜色。我不知道从哪里开始。我在想,每个字符串的MD5散列都是不同的-但如何将该散列转换为十六进制颜色数字? 最佳答案 你可以只取几位前几位:require'digest/md5'color=Digest::MD5.hexdigest('Mytext')[0..5] 关于ruby-如何使用Ruby基于字母数字字符串生成颜色?,我们在StackOverflow上找到一个类似的问题:
文章目录1.自动驾驶实战:基于Paddle3D的点云障碍物检测1.1环境信息1.2准备点云数据1.3安装Paddle3D1.4模型训练1.5模型评估1.6模型导出1.7模型部署效果附录show_lidar_pred_on_image.py1.自动驾驶实战:基于Paddle3D的点云障碍物检测项目地址——自动驾驶实战:基于Paddle3D的点云障碍物检测课程地址——自动驾驶感知系统揭秘1.1环境信息硬件信息CPU:2核AI加速卡:v100总显存:16GB总内存:16GB总硬盘:100GB环境配置Python:3.7.4框架信息框架版本:PaddlePaddle2.4.0(项目默认框架版本为2.3
SCIM是Google、Salesforce、PingIdentity等提出的用户配置的新标准。是否有现有的ruby实现来支持它?SimilarbutforJava 最佳答案 您可以获得一个面向Okta的开始fromhere或从头开始here但我认为您只能靠自己。耶!您有机会为Ruby社区贡献一个开源项目:) 关于SCIMv2的Ruby实现,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/ques
我正在尝试整个BDD方法并想测试AMQP基于Vanilla的方面Ruby我正在写的应用程序。选择Minitest后作为与其他名副其实的蔬菜框架不同的平衡功能和表现力的测试框架,我着手编写此规范:#File./test/specs/services/my_service_spec.rb#Requirementsfortestrunningandconfigurationrequire"minitest/autorun"require"./test/specs/spec_helper"#Externalrequires#MinitestSpecsforEventMachinerequire