實驗現(xiàn)象
將MAX-10小腳丫FPGA和超聲波模塊HC_SR04插在面包板上,用杜邦線將對應(yīng)的引腳連接好,燒錄程序,小腳丫自帶的數(shù)碼管顯示距離數(shù)據(jù)(單位是厘米)。這張圖拍花了,數(shù)碼管顯示的數(shù)據(jù)是18CM
簡單介紹超聲波測距模塊HC_SR04
HC-SR04是一種基于超聲波的測距模塊。該模塊向前15度內(nèi)發(fā)送超聲波并接收回響,通過發(fā)出超聲波到收到回響的這個時間間隔計算前方的障礙物距離,可以用來給智能小車做障礙物監(jiān)測??商峁?cm- 400cm的非接觸式距離感測功能,測距精度可達高到3mm;模塊包括超聲波發(fā)射器、接收器與控制電路。
該模塊的時序圖如下:
該模塊的引腳圖如下:
我們在編寫代碼的時候,想要發(fā)出測距命令,需要先保持觸發(fā)信號輸入(trig引腳)為低電平,然后保持大于10us的高電平,再變成低電平即可(時序圖第一行所示)。
發(fā)出測距命令后,回響信號輸出(echo引腳)會保持一段時間的高電平,這個高電平的持續(xù)時間與距離有關(guān)。我們在FPGA編寫代碼測量高電平持續(xù)時間,然后將時間轉(zhuǎn)化為距離即可。
經(jīng)過測試,得到以下結(jié)論:
1、發(fā)送測距的時候,每個脈沖之間的間隔不要過近,我用的間隔是300ms,即每300ms測一次距離,也可以根據(jù)需要修改。
2、接收端高電平時間與距離的關(guān)系式是:距離(cm)= 高電平持續(xù)時間(us)x 0.034cm / 2 (除以2是因為持續(xù)時間是往返時間,除以2才是單程時間)
模塊框圖
我們需要設(shè)計以下模塊:
1、測距信號源模塊,輸入時鐘與復(fù)位,每隔300ms輸出15us高電平。
2、距離計算模塊,輸入時鐘、復(fù)位與回響信號echo,輸出距離(cm)
3、數(shù)據(jù)顯示模塊,將得到的距離可視化,我用的是2個1位數(shù)碼管模塊(小腳丫自帶兩個1位共陰極數(shù)碼管)
4、頂層模塊
框圖如下:
模塊編寫
測距信號源
在此模塊,寫一個計時周期為300us的循環(huán)計數(shù)器,當(dāng)計數(shù)器值小于15時輸出高電平,其他時候輸出低電平即可。
代碼如下:
module trigger_send #(
parameter TIME_1S = 12_000_000
) (
input clk ,
input rst_n ,
output trigger
);
//1us生成
wire clk_1us;
PLL UPLL(
.inclk0 (clk),
.c0 (clk_1us)
);
reg [25:0] cnt_1us;
always @(posedge clk_1us or negedge rst_n) begin
if (!rst_n) begin
cnt_1us <= 20'd0;
end
else if (cnt_1us == 20'd300_000 - 1)begin
cnt_1us <= 20'd0;
end
else begin
cnt_1us <= cnt_1us + 1'b1;
end
end
assign trigger = cnt_1us < 15 ? 1'b1 : 1'b0;
endmodule
我懶得自己寫1us的計數(shù)器,用鎖相環(huán)生成了一個1MHZ(1us)的時鐘,讓它驅(qū)動計時,你可以不用鎖相環(huán)自己寫1us比特計數(shù),不知道怎么用鎖相環(huán)可以搜一下,很簡單。
距離計算
這個模塊輸入了HC_SR04的回響引腳,要做的就是計算它的高電平持續(xù)時間,并轉(zhuǎn)化為距離數(shù)據(jù)輸出。
這里持續(xù)時間的單位用us最合適,因此同樣生成一個1us時鐘,用這個時鐘監(jiān)測回響信號的上升沿與下降沿。檢測到上升沿,就每過一個時鐘周期就把計時器數(shù)值加1(可以自己設(shè)置上限,表示最大檢測距離);等檢測到下降沿,就把計時器的數(shù)據(jù)保存起來,然后把計時器清空準備下次計時。
拿到保存好的計時后,使用上文提到的公式計算出距離輸出。
代碼如下:
module data_rec #(
parameter TIME_1S = 12_000_000
) (
input clk ,//系統(tǒng)時鐘
input rst_n ,//復(fù)位
input rec_data ,//回響,早知道取名echo了
output [11:0] distance //計算好的距離
);
//鎖相環(huán)生成1us周期時鐘,因為后面的計時單位全是1us,這樣方便
wire clk_1us;
PLL UPLL(
.inclk0 (clk),
.c0 (clk_1us)
);
//給回響信號打拍,檢測上升沿下降沿
reg rec_data2;
reg rec_data3;
wire rec_negedge;
wire rec_posedge;
assign rec_negedge = (!rec_data2) && rec_data3;
assign rec_posedge = rec_data2 && (!rec_data3);
always @(posedge clk_1us or negedge rst_n) begin
if (!rst_n) begin
rec_data2 <= 1'b0;
end
else begin
rec_data2 <= rec_data;
end
end
always @(posedge clk_1us or negedge rst_n) begin
if (!rst_n) begin
rec_data3 <= 1'b0;
end
else begin
rec_data3 <= rec_data2;
end
end
//計時啟動標(biāo)志,上升沿啟動,下降沿結(jié)束
reg flag;
always @(posedge clk_1us or negedge rst_n) begin
if (!rst_n) begin
flag <= 1'b0;
end
else if (rec_posedge)begin
flag <= 1'b1;
end
else if (rec_negedge)begin
flag <= 1'b0;
end
else begin
flag <= flag;
end
end
//計時器數(shù)值,flag期間計數(shù),有下降沿就清0,急了多少數(shù)就是保持了多少us,因為是用1us時鐘驅(qū)動的
reg [14:0] cnt_1us;
always @(posedge clk_1us or negedge rst_n) begin
if (!rst_n) begin
cnt_1us <= 15'd0;
end
else if (rec_negedge)begin
cnt_1us <= 15'd0;
end
else if (flag && cnt_1us < 15'd15_000)begin
cnt_1us <= cnt_1us + 1'b1;
end
else begin
cnt_1us <= cnt_1us;
end
end
//因為計時器結(jié)束計時會變0,因此要用另外的變量,在它清0的時候把值保存下來
reg [14:0] high_time;
always @(posedge clk or negedge rst_n) begin
if (!rst_n) begin
high_time <= 15'd0;
end
else if (rec_negedge)begin
high_time <= cnt_1us ;
end
else begin
high_time <= high_time;
end
end
//計算距離,Verilog不能直接用浮點數(shù),就這樣實現(xiàn)乘以0.017
//為什么不是0.034看前面
reg [11:0] distance_buf;
always @(posedge clk or negedge rst_n) begin
if (!rst_n) begin
distance_buf <= 12'd0;
end
else begin
distance_buf <= high_time * 17 / 1000;
end
end
assign distance = distance_buf;
endmodule
數(shù)碼管模塊
距離模塊的distance就是測到的距離,你拿去數(shù)碼管輸出,串口打印都可以。我的小腳丫自帶兩個1位的共陰極數(shù)碼管,就用它們來顯示,畢竟自己在面包板上給數(shù)碼管插線插電阻還是挺麻煩的。
用的時候取出測的距離的十位個位分別給兩個模塊就行。
1位七段式共陰極數(shù)碼管十進制顯示模塊如下:
module nixietube_1 (
input clk,
input rst_n,
input [3:0] din, //輸入0-9
output drive_out, //使能
output [6:0] _dig, //輸出數(shù)碼管
output dot_out //小數(shù)點要亮嗎
);
//dot小數(shù)點輸出
assign dot_out = 1'b0;
//使能,陰極為0
assign drive_out = 1'b0;
//dig段選輸出
parameter ZER = 7'b0111111,
ONE = 7'b0000110,
TWO = 7'b1011011,
THR = 7'b1001111,
FOU = 7'b1100110,
FIV = 7'b1101101,
SIX = 7'b1111101,
SEV = 7'b0000111,
EIG = 7'b1111111,
NIN = 7'b1101111;
reg [6:0] dig;
always @(posedge clk or negedge rst_n) begin
if(!rst_n) begin
dig <= ZER;
end
else begin
case (din)
0 : dig <= ZER;
1 : dig <= ONE;
2 : dig <= TWO;
3 : dig <= THR;
4 : dig <= FOU;
5 : dig <= FIV;
6 : dig <= SIX;
7 : dig <= SEV;
8 : dig <= EIG;
9 : dig <= NIN;
default : dig <= ZER;
endcase
end
end
assign _dig = dig;
endmodule
頂層模塊
連起來就可以了,沒什么好說的。
module Ultrasound (
input clk,
input rst_n,
input rec_data ,
output trigger ,
output [6:0] dig1,
output [6:0] dig2,
output dot1,
output dot2,
output drive1,
output drive2
);
trigger_send u_trigger_send(
.clk (clk),
.rst_n (rst_n),
.trigger (trigger)
);
wire [11:0] distance;
data_rec u_data_rec(
.clk (clk),
.rst_n (rst_n),
.rec_data (rec_data),
.distance (distance)
);
nixietube_1 u_nixietube_1(
.clk (clk),
.rst_n (rst_n),
.din ((distance/10)%10),
._dig (dig1),
.dot_out (dot1),
.drive_out (drive1)
);
nixietube_1 u_nixietube_2(
.clk (clk),
.rst_n (rst_n),
.din ((distance/1)%10),
._dig (dig2),
.dot_out (dot2),
.drive_out (drive2)
);
endmodule
引腳設(shè)置的時候,兩個數(shù)碼管,時鐘按照小腳丫的原理圖來設(shè)置,trig與echo自己設(shè)置,復(fù)位鍵可以用它自帶的按鈕。文章來源:http://www.zghlxwxcb.cn/news/detail-633324.html
總結(jié)
本次實驗實現(xiàn)了使用FPGA驅(qū)動超聲波模塊HC_SR04,以前在小車比賽見過,但當(dāng)時用的官方提供好的示例代碼?,F(xiàn)在自己也知道如何使用了,可以拿去做很多東西,收獲很大。文章來源地址http://www.zghlxwxcb.cn/news/detail-633324.html
到了這里,關(guān)于基于MAX-10 FPGA 讀取超聲波模塊HC_SR04距離數(shù)據(jù)到數(shù)碼管上的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!