Verilog实现无源蜂鸣器发声

要求:顺序循环发声(Do、Re、Mi、Fa....),时间间隔为0.5s。

Verilog实现无源蜂鸣器发声_第1张图片

若输入为262Hz的方波,则蜂鸣器发Do的声音,以此类推。

那如何向蜂鸣器输入262Hz的方波?假设系统时钟周期为50MHz。

50MHz对应的一个周期为20ns,若占空比为50%,则每10ns电平反转一次。

则50MHz实现0.5s需要从0开始计数到24_999_999。

由于50MHz计数器是上升沿记一次数,262Hz对应的一个周期为3816794ns,则其计数周期为3816794ns/20ns=190840ns,若占空比为50%,则每95420ns电平反转一次。如下表格所示

Verilog实现无源蜂鸣器发声_第2张图片

关于系统的输入输出如下图所示

Verilog实现无源蜂鸣器发声_第3张图片 

根据输入输出可以判断:

第一,需要一个计数器cnt,实现0.5s的功能,从0开始计数到24_999_999;

第二,因为每0.5s发出的声音需要变换一次,总共7个音,即cnt每次计数满0.5s需要自加一,并且记满7个音从0开始计数。

第三,每个音频(Do、Re、Mi、Fa....)有一个一个确定的值freq_data,还需要freq_cnt,从0开始计数到freq_data。

第四,占空比为50%,则当计数机器freq_cnt计数到(freq_data/2)时,电平需要反转一次。

另外,在每0.5s内,需要不断地发出Do的声音,波形图如下:

Verilog实现无源蜂鸣器发声_第4张图片

所以通过上述整体的分析可以整体波形图如下

Verilog实现无源蜂鸣器发声_第5张图片 

Verilog代码如下:

module beep
#(
	parameter CNT_MAX=25'd24_999_999,
	parameter DO=18'd190839,
	parameter RE=18'd170067,
	parameter MI=18'd151514,
	parameter FA=18'd143265,
	parameter SO=18'd127550,
	parameter LA=18'd113635,
	parameter XI=18'd101213
    )
(
	input wire sys_clk,
	input wire sys_rst_n,
	output reg beep
);

	reg [24:0]cnt;
	reg [2:0]cnt_500ms;
	reg [17:0]freq_cnt;
	reg [17:0]freq_data;
	wire [16:0]duty_data;
	
always@(posedge sys_clk or negedge sys_rst_n)//0.5s的计数器
	if(sys_rst_n==1'b0)
		cnt<=25'd0;
	else if(cnt==CNT_MAX)
		cnt<=25'd0;
	else
		cnt<=cnt+25'd1;

always@(posedge sys_clk or negedge sys_rst_n)//每经过0.5s自加1
	if(sys_rst_n==1'b0)
		cnt_500ms<=3'd0;
	else if((cnt==CNT_MAX)&&(cnt_500ms==3'd6))
		cnt_500ms<=3'd0;
	else if(cnt==CNT_MAX)
		cnt_500ms<=cnt_500ms+3'd1;
	else
		cnt_500ms<=cnt_500ms;
		
always@(posedge sys_clk or negedge sys_rst_n)//每个音的输入频率都不同
	if(sys_rst_n==1'b0)
		freq_data<=DO;
	else case(cnt_500ms)
		3'd0:freq_data<=DO;
		3'd1:freq_data<=RE;
		3'd2:freq_data<=MI;
		3'd3:freq_data<=FA;
		3'd4:freq_data<=SO;
		3'd5:freq_data<=LA;
		3'd6:freq_data<=XI;
	default:freq_data<=DO;
	endcase
	
always@(posedge sys_clk or negedge sys_rst_n)//对每个音的输入频率进行计数
	if(sys_rst_n==1'b0)
		freq_cnt<=18'd0;
	else if((freq_cnt==freq_data)||(cnt==CNT_MAX))
		freq_cnt<=18'd0;
	else
		freq_cnt<=freq_cnt+18'd1;

assign duty_data=freq_data>>1;//右移一位相当于除以2
		
always@(posedge sys_clk or negedge sys_rst_n)//经过一个duty_data电平反转一次
	if(sys_rst_n==1'b0)
		beep<=1'b0;
	else if(freq_cnt

 仿真测试代码如下:

`timescale 1ns / 1ns

module tb_beep();

reg sys_clk;
reg sys_rst_n;
wire beep;

initial
	begin
		sys_clk=1'b1;
		sys_rst_n=1'b0;
		#20
		sys_rst_n=1'b1;
	end
always #10 sys_clk=~sys_clk;

beep
#(
.CNT_MAX(25'd24_999),//仿真数值都缩小了1000倍进行仿真
.DO(18'd190),
.RE(18'd170),
.MI(18'd151),
.FA(18'd143),
.SO(18'd127),
.LA(18'd113),
.XI(18'd101)
    )
beep_inst	
(
.sys_clk(sys_clk),
.sys_rst_n(sys_rst_n),
.beep(beep)
);

endmodule

 vivado仿真波形

第一个音DO,50%占空比。

Verilog实现无源蜂鸣器发声_第6张图片

计数到24999时转换为下一个音,且下一个音占空比为50%

Verilog实现无源蜂鸣器发声_第7张图片

 Verilog实现无源蜂鸣器发声_第8张图片

 

 当第六个音记满,开始循环发声

Verilog实现无源蜂鸣器发声_第9张图片

 

你可能感兴趣的:(fpga开发)