Hefei University 在系统编程课程设计
课题名称4位十进制频率计的设计姓名钟文俊
学号1105012012
院系电子信息与电气工程系专业电子信息工程
指导教师黄慧
2013年 12月28日
4位十进制频率计的设计
一、设计目的
1、学习掌握频率计的设计方法。
2、掌握动态扫描输出电路的实现方法。
3、学习较复杂的数字系统设计方法。
二、设计的主要内容和要求
4位十进制频率计外部接口设计,顶层文件设计,包含4个模块,Tctl,Reg16,scan_led和一个x4cnt10(4个十进制计数器)。
三、整体设计方案
1、设计方案
根据频率计的定义和频率测量的基本原理:频率计即是指单位时间1秒内输入脉冲个数并显示出来的电路。因为要显示被测信号的,只要限制计数器的计数过程为一秒则计数器的结果即为被测信号的频率。频率值为:Fx=N,N为一秒内计数器所计脉冲个数。因为是4位十进制频率计计数器是模为9999的十进制加法计数器,可以由4个模为10的十进制计数器级联而成,所以可以显示的频率范围是1-9999HZ。因此,频率计的功能分割成四个模块:计数器,测频控制信号发生器和输出锁存器,。各个模块均用VHDL语言描述。待测信号的转速必须有一个脉宽为1秒的输入信号脉冲计数允许的信号;1秒计数结束后,计数值锁入锁存器的锁存信号,并有为下一测频计数周期作准备的计数器清零信号。这三个信号可以由一个测频控制信号发生器产生。当系统正常工作时,标准信号提供频率为1 Hz的输入信号,经过测频控制信号发生器进行信号变换,产生计数信号,将被测信号当做脉冲信号送入计数器模块,计数模块对输入的脉冲个数进行计数数结束后,将计数结果送入锁存器中,保证系统可以稳定显示数据,计数结果能够显示在七段数码显示管上。
根据系统设计要求,要实现一个4位十进制数字频率计,则要设计测频控制信号发生器模块tct1、四位十进制加法计数器模块x4cnt10、reg16锁存器模块, 七段译码显示模块scan_led其原理框图如下图所示。
2、四位十进制频率计顶层文件原理图如下:
图3.2 顶层文件生成4位十进制频率计外部接口图
3、本设计中重要端口说明:
Clk_1hz : 给Tctl 模块提供1hz 的频率输入。 Fin ;被测频率输入。
scan_led :给scan_led 模块提供扫描输入频率输入。 bt[1..0]:片选信号输出。 Sg[6..0]:译码信号输出。 Cout :进位输出
四、各个模块的设计说明及设计步骤
1、顶层文件原理图设计:
测 频 控 制 信 号 发 生 器
锁存器
七 段 译 码 显 示
计 数 器
图 4.1 顶层文件原理图
2、x4Cnt10 模块说明
x4Cnt10为含异步清零和同步时钟使能的十进制计数器,采用级联的方法进行计数,计数范围与所用x4cnt10级联个数有关,本次设计采用的是4个cnt10的级联,所有测频范围是0—9999hz。
仿真时序图如下:
图 4.2 x4Cnt10仿真波形图
3、 Tctl 模块说明
根据频率的定义和测量的基本原理,测定信号的频率必须有一个脉宽为1
秒的对输入信号脉冲计数允许的信号;1秒计数结束后,计数值锁入锁存器信号和为下一测频计数周期做准备的计数器清零信号。这三个信号可以由一个测频控制信号发生器Tctl产生,其设计要求是Tctl的计数使能信号en能产生一个一秒脉宽的周期信号,并且对频率计的每一计数器cnt10的en使能进行同步控制,当en高定平时,允许继续:低电平时停止计数,并保持其所计数的脉冲数,在停止计数期间,首先需要一个锁存信号load的上升沿将计数器前一秒钟的计数值锁存进各锁存器reg16中,并由外部的译码器译出并显示计数值,锁存信号之
后,必须有一个清零信号rst对计数器进行清零,为下一秒的计数操作做准备。
控制模块Tctl时序仿真:
图4.3 频率计测频控制器Tctl侧可控时序图
图中clk的频率为1hz,en的高电平程序时间为1秒,此时计数器开始工作,当en低电平时停止计数,并将计数值经过load锁存的reg16中最后通过
scan_led译码器显示输入,en的计时长度为一秒,即en的周期为2秒,计数器在en高电平计数完成即en从高电平跳跃到低电平时将计数结果锁存在reg16中,rst在满足clk和en均为低电平的时候对已锁存入锁存器的计数信号进行清零工作,为下一个计数周期作准备。
4、锁存器reg16模块的说明
设置锁存器的目的是使显示的数据稳定,不会由于周期性的清零信号而不断闪烁,每次显示数据位上一计数周期锁存的计数值。
Reg16锁存器时序仿真图:
图4.4 reg16锁存器仿真波形图
5、scan_led 模块说明
动态显示是把所有的led管的输入信号连在一起,每次向LED写数据时,通过片选信号选通其中一个Led管并把数据写入,因此每一个时刻只有一个Led
是亮的。为了能够程序看到led上面的显示内容,必须对led管进行扫描,即一次循环点亮各个led管,利用人眼的视觉暂留,加上发光器件的余晖效益,在一定的扫描频率下,人眼就会看到多个led管一起点亮。扫描的大小必须合适才能
达到很好的效果。如果扫描频率太低,就会产生闪烁;而扫描频率太高会造成Led的频率开启和关断,增加led的功耗。通常扫描频率选择50Hz比较合适。
本次设计为4为led动态扫描输出,clkwie扫描时钟,考虑到cnt8的分频作用,此时设计采用的频率为200hz:sg[6..0]为当前正在显示的led地址的已译码数据;bt[1..0]为输出的片选信号,决定某led在某时某刻显示数据:
din[15..0]为4为一组的bcd码,共4位供显示输出的数据输入,scan_led的内部以下三个进程:计数模块扫描时钟发生器,多路选通器模块和译码模块。
led动态扫描输出scan_led时序仿真图;
图4.5 动态扫描仿真波形图
Clk_1hz输入周期为1hz;
Fin为测量频率,频率范围是100-9999hz,此次波形仿真设计时间为1khz;
Clk为扫描时钟,周期为200hz;
Sg为输出led七段显示译码结果,bt为输出片选信号,由波形仿真图可以看出片选0(00)时led显示为0,片选1(01)时led显示为0,片选2(10)时led显示为0,片选3(11)时led显示为1;即此时频率计计数频率为1000hz,与实际输入频率相同。
Clk_1hz的1hz输入频率通过DE2内部clock_50的50M晶振分频来实现。输入频率信号和扫描频率也由50M晶振分频得到,又由于仿真设计中采用的是一个led现实动态数据,所以下载到DE2板上是还要进行动态显示电路的设计。
6、分频器:
(可以通过改变不同的cout值来进行不同的分频)
分频器时序仿真图:
图4.6 分频仿真波形图
7、动态显示时序仿真图
图4.7 动态显示仿真波形8、调用各子模块编译仿真,时序仿真图如下
图4.8 调用各子模块时序仿真图9、编译仿真完成后,定义管脚,如下图所示:
图4.9 引脚定义
10、定义完引脚后,开始下载设计文件,在试验箱进行硬件测试并完善设计。
11、预测结果:
试验箱选择模式6,1HZ测频控制信号由clock0输入控制,待测信号由clock5输入,可选择相应的频率值,在数码管显示器上显示相应的频率值。五、总结
通过这次的eda设计使我更加的了解了vhdl语言的应用,由于书上设计的是由数码管进行动态显示,但是这种情况下不利于读取数据,所以我们后来设计成由四个七段数码管静态输出结果的方式,但书上没有源程序可以参考,所以只能通过自己在网上查找资料并尝试,在这其过程中出现了很多错误,我们一步步更改,要求我们修改编写程序时要细心,谨防出现细小的错误。尤其是实用语句的时候要考虑全面,不能前面写过忘记结尾,例如在使用if语句时要注意end if 的个数与前相对应,不能少也不能多。在设计时还要考虑到设计引脚是否能够在试验箱上找到。最后在老师的指导下我们总算是顺利地完成了这次的课程设计。参考文献
[1] 潘松,黄继业.EDA技术实用教程(第二版).科学出版社,1998。
[2] 宋万杰,罗丰,吴顺军.CPLD技术及其应用.西安:西安电子科技大学出版社,1999。
[3] 徐志军王金明.EDA技术与PLD设计.人民邮电出版社。
[4] 康华光.电子技术基础(数字部分).高等教育出版社。
[5]百度文科4位十进制频率计设计。
附录:
四位十进制频率计顶层文件源程序如下:
library ieee;
use ieee.std_logic_1164.all;
use ieee.std_logic_unsigned.all;
entity dc is
port(fin,clk_1hz:in std_logic;
ledout:out std_logic_vector(27 downto 0));
end dc;
architecture one of dc is
component Tct1 ---Tct1元件声明
port( clk: in std_logic;
en,rst,load :out std_logic);
end component;
component x4cnt10 ----x4cnt10元件声明
port(clk,rst,en: in std_logic;
q0,q1,q2,q3 :out std_logic_vector(3 downto 0);
cout:out std_logic);
end component;
component reg16 ---reg16元件声明
port(load: in std_logic;
di : in std_logic_vector(15 downto 0);
dout : out std_logic_vector(15 downto 0) );
end component;
component scan_led ---scan_led元件声明
port(din: in std_logic_vector(15 downto 0);
sg :out std_logic_vector(6 downto 0);
bt: out std_logic_vector(1 downto 0));
end component;
signal x,z,f:std_logic;
signal h:std_logic_vector(3 downto 0);
signal g0,g1,g2,g3:std_logic_vector(3 downto 0);
signal h0,h1,h2,h3:std_logic_vector(3 downto 0);
signal leds:std_logic_vector(27 downto 0);
begin ---元件例化产生电路,完成设计
u1: Tct1 port map(clk=>clk_1hz, en=>x,rst=>z,load=>f);
u2: x4cnt10 port map(clk=>fin,rst=>z,en=>x,q0=>g0,q1=>g1,q2=>g2,q3=>g3);
u3:reg16 port map(load=>f,di(3 downto 0)=>g0,di(7 downto 4)=>g1,di(11 downto 8)=>g2,di(15 downto 12)=>g3,dout(3 downto 0)=>h0,dout(7 downto
4)=>h1,dout(11 downto 8)=>h2,dout(15 downto 12)=>h3);
u4: scan_led port map(din(3 downto 0)=>h0(3 downto 0),sg(6 downto
0)=>leds(6 downto 0));
u5: scan_led port map(din(7 downto 4)=>h1(3 downto 0),sg(6 downto
0)=>leds(13 downto 7));
u6: scan_led port map(din(11 downto 8)=>h2(3 downto 0),sg(6 downto
0)=>leds(20 downto 14));
u7: scan_led port map(din(15 downto 12)=>h3(3 downto 0),sg(6 downto
0)=>leds(27 downto 21));
ledout<=leds;
end;
计数模块x4cnt10的源代码程序:
library ieee;
use ieee.std_logic_1164.all;
use ieee.std_logic_unsigned.all;
entity x4cnt10 is
port(clk,rst,ena: in std_logic; --端口定义q0,q1,q2,q3 : buffer std_logic_vector(3 downto 0); --分别定义四个计cout:out std_logic); 数端口,用于4
end x4cnt10; 个cnt10的设计architecture three of x4cnt10 is
signal en1,en2,en3 : std_logic; ---使能信号声明
begin
process(clk,rst,ena,q0) ---控制q0端的十进制计数begin
if rst='1' then q0 <= "0000";
elsif clk'event and clk='1' then
if ena='1' then
if q0="1001" then q0<="0000";else q0<=q0+1;end if;
end if;
end if;
if q0="1001" then en1 <='1'; else en1 <='0';end if;
end process;
process(clk,rst,en1,q1) ---控制q1端的十进制计数
begin
if rst='1' then q1 <= "0000";
elsif clk'event and clk='1' then
if en1='1' then
if q1="1001" then q1<="0000";else q1<=q1+1;end if;
end if;
end if;
if q1="1001" then en2 <='1'; else en2 <='0';end if;
end process;
process(clk,rst,en2,q2) --控制q2端的十进制计数begin
if rst='1' then q2 <= "0000";
elsif clk'event and clk='1' then
if en2='1' and en1='1'then
if q2="1001" then q2<="0000";else q2<=q2+1;end if;
end if;
end if;
if q2="1001" then en3 <='1'; else en3 <='0';end if;
end process;
process(clk,rst,en3,q3) ---控制q3端的十进制计数
begin
if rst='1' then q3 <= "0000";
elsif clk'event and clk='1' then
if en3='1' and en2='1' and en1='1'then
if q3="1001" then q3<="0000";else q3<=q3+1;end if;
end if;
end if;
if q3="1001" then cout<='1'; else cout<='0';end if;
end process;
end three;
library ieee; ---将设计元x4cnt10的声明装入my_pkg程序包中use ieee.std_logic_1164.all;
package my_pkg1 is
component x4cnt10
port(clk,rst,ena: in std_logic;
q0,q1,q2,q3 : buffer std_logic_vector(3 downto 0);
cout:out std_logic);
end component;
end;
控制模块Tctl源代码程序;
Library ieee;
Use ieee.std_logic_1164.all;
Use ieee.std_logic_unsigned.all;
Entity Tct1 is
Port(clk:in std_logic; ---端口定义
ena,rst,load:out std_logic);
End Tct1;
Architecture one of Tct1 is
Signal divclk :std_logic; ---divclk信号声明
Begin
Process(clk)
Begin
if clk'event and clk='1' then divclk<=not divclk; ---clk上升沿时,divclk取反end if ;
End process;
Process(clk,divclk)
Begin
If clk='0' and divclk='0' then ---clk和divclk同时为0时,rst置1,否则rst清0 rst<='1';
Else rst<='0';
End if;
End process;
Load<=not divclk; ----将divclk取反赋值到load端,将divclk的值赋给ena ena<=divclk;
end one;
Reg16锁存器源代码程序:
Library ieee;
Use ieee.std_logic_1164.all;
Entity reg16 is
Port(load:in std_logic; ----端口定义
din:in std_logic_vector(15 downto 0);
dout:out std_logic_vector(15 downto 0));
End reg16;
Architecture one of reg16 is
Begin
Process(load,din)
Begin
if(load'event and load='1') then ---load为上升沿时,把din值锁存到dout 中
dout<=din;
end if;