3642
- 收藏
- 点赞
- 分享
- 举报
用VHDL制作可变频可加减可复位计数器
用VHDL制作可变频可加减可复位计数器
1. 系统功能及要求:
该计数器应能实现加减计数,还应该有异步复位功能;与一般计数器不同的是该计数器
应可以变频;如果你想用不同频率实现计数,只要通过EDA实验板上按键进行切换即
可。如果你想实现加减计数,也只要预设一下方向状态即可.如果你想用某一频率进行
某一方向计数,只要预设方向和频率即可。
2.制作思路:
该系统核心是加减计数器,计数方向受方向按键控制,计数器采用串行进位。
要想实现不同频率计数,需做一系列方波信号,然后将这些不同频率的脉冲通过数据选
择器选择一个作为计数脉冲。
复位信号与方向控制信号需做成单脉冲行式,故需制作单脉冲发生器。
计数频率应适当,人眼必须能明显分辨出计数频率的变化,在这里共作了6档频率,最低
的约为0.5hz,然后每档以二倍向上增长。
要将单脉冲通过触发器,触发器输出用于判断该切换频率还是该切换计数方向。
数码管用于显示计数情况,LED指使灯最右边三个用于指示现在处于哪个计数频率,
最左边一个用于指示现在是加法计数还是减法计数。
LED右三个指使灯为000时处于0号计数频率;为001时处于1号计数频率;为010时处于2号计数频率;为011时处于3号计数频率;为100时处于4号计数频率;为101时处于5号计数频率。
LED最左边一个为0(不亮)时实行加法计数,为1(亮)时实行减法计数。
注意:LED灯亮表示1,但此时给LED送的电平为低。
3. 源程序注释:
library IEEE;
use IEEE.std_logic_1164.all;
use IEEE.std_logic_unsigned.all;
entity muli_fre_light is
port (
change: in STD_LOGIC; -------频率切换按键
reset: in STD_LOGIC; -------复位与开始计数按键
direct:in std_logic; -------方向切换按键
dig: out std_logic_vector(3 downto 0); ------数码管位选
led: out STD_LOGIC_VECTOR (6 downto 0); ------数码管位选
dp:out std_logic; -------小数点点亮
dire:out std_logic; -------计数方向指使灯
light: out STD_LOGIC_VECTOR (2 downto 0) -------频率状态指使灯
);
end muli_fre_light;
architecture muli_fre_light_arch of muli_fre_light is
component clk_25hz ---------元件clk_25用来产生6种计数频率和LED扫描
port(clks,clks1,clks2,clks3,clks4,clks5,wer:inout STD_LOGIC); 频率
end component;
signal refresh:std_logic_vector(1 downto 0);
signal zzm:std_logic_vector(2 downto 0);
signal bcd1,bcd2,bcd3,bcd4,bcd:std_logic_vector(3 downto 0);
signal carry1,carry2,carry3,clock,clks,clks1,clks2,clks3,clks4,clks5,wer,sel:std_logic;
signal chan1,chan0,direct0,direct1,dir,a_s:std_logic;
begin
u2:clk_25hz port map(clks,clks1,clks2,clks3,clks4,clks5,wer);-----调用clk_25产生
各种频率
process(clks1)
begin
if(clks1'event and clks1='0')then -----单脉冲信号sel dir产生
chan1<=chan0;
chan0<=change;
direct1<=direct0;
direct0<=direct;
end if;
end process;
process(chan0,direct0) dir为切换方向按键按下后
begin 产生的单脉冲
sel<=clks1 and chan0 and (not chan1); sel为切换频率按键按下后
dir<=clks1 and direct0 and (not direct1); 产生的单脉冲
end process;
process(sel) ------每次频率切换通过zzm记录所处
begin 频率状态
if(sel'event and sel='1')then
if(zzm="101")then
zzm<="000";
else
zzm<=zzm+1;
end if;
end if;
end process;
light<=not zzm; ------当前频率状态送LED右三位显示
process(dir) -------每次方向切换通过a_s记录所处
begin 方向状态
if(dir'event and dir='1')then
a_s<=not a_s;
end if;
end process;
dire<=not a_s; -----将方向状态送LED最左边显示
clock<=clks when zzm="000" else -----6选1 mux选择计数脉冲
clks1 when zzm="001" else
clks2 when zzm="010" else
clks3 when zzm="011" else
clks4 when zzm="100" else
clks5 ;
process(clock) ------clock为mux输出信号
begin
if(reset='0')then ------如果reset=0则允许加减计数
if(clock'event and clock='1')then 此时如a_s=0则做减法
if(a_s='0')then --add count 此时如a_s=1则做加法
if(bcd1="1001")then
bcd1<="0000";
carry1<='1'; 个位计数
else
bcd1<=bcd1+1;
carry1<='0';
end if;
else --sub count
if(bcd1="0000")then
bcd1<="1001";
carry1<='1';
else
bcd1<=bcd1-1;
carry1<='0';
end if;
end if;
end if;
else
bcd1<="0000";
end if;
end process;
process(carry1,clock)
begin
if(carry2=’1’)then *************这个if 很重要,如没有则会出现错误
carry2<=’0’;**************
else
if(reset='0')then
if(carry1'event and carry1='1')then 十位计数
if(a_s='0')then --add count
if(bcd2="1001")then
bcd2<="0000";
carry2<='1';
else
bcd2<=bcd2+1;
carry2<='0';
end if;
else --sub count
if(bcd2="0000")then
bcd2<="1001";
carry2<='1';
else
bcd2<=bcd2-1;
carry2<='0';
end if;
end if;
end if;
else
bcd2<="0000";
end if;
end if;
end process;
process(carry2,clock)
begin
if(carry3=’1’)then **********************************
carry3<=’0’;
else
if(reset='0')then
if(carry2'event and carry2='1')then
if(a_s='0')then --add count 百位计数
if(bcd3="1001")then
bcd3<="0000";
carry3<='1';
else
bcd3<=bcd3+1;
carry3<='0';
end if;
else --sub count
if(bcd3="0000")then
bcd3<="1001";
carry3<='1';
else
bcd3<=bcd3-1;
carry3<='0';
end if;
end if;
end if;
else
bcd3<="0000";
end if;
end if;
end process;
process(carry3)
begin
if(reset='0')then
if(carry3'event and carry3='1')then 千位计数
if(a_s='0')then --add count
if(bcd4="1001")then
bcd4<="0000";
else
bcd4<=bcd4+1;
end if;
else --sub count
if(bcd4="0000")then
bcd4<="1001";
else
bcd4<=bcd4-1;
end if;
end if;
end if;
else
bcd4<="0000";
end if;
end process;
--display ------使百位小数点一直点亮
dp<='1' when refresh="10" else
'0';
LED<= "0111111" when(BCD="0000") else -----数码管7段译码
"0000110" when(BCD="0001") else
"1011011" when(BCD="0010") else
"1001111" when(BCD=" 0011") else
"1100110" when(BCD="0100") else
"1101101" when(BCD="0101") else
"1111101" when(BCD="0110") else
"0000111" when(BCD="0111") else
"1111111" when(BCD="1000") else
"1101111" when(BCD="1001") else
"XXXXXXX";
--bcd for display
BCD<=BCD1 when(refresh="00" ) else ----refresh
BCD2 when(refresh="01" ) else
BCD3 when(refresh="10" ) else
BCD4 when(refresh="11" ) else
"XXXX";
process(wer) -----wer为扫描脉冲
begin
if(wer'event and wer='1')then
refresh<=refresh+1; -----通过refresh四种状态选择点亮
end if; 数码管
end process;
--dig signal generate
with REFRESH select -------数码管位选译码
DIG <= "1110" when "00",
"1101" when "01",
"1011" when "10",
"0111" when "11",
"XXXX" when others;
end muli_fre_light_arch;
4.管脚锁定文件说明:
net reset loc=p33; 复位和启动管脚,在EDA板上为跳线最左端脚
使用时把P33接地则开始计数,把P33不接地则清零
net change loc=p7; 频率选择按键,每按一下换一档频率
net direct loc=p6; 方向选择按键,每按一下换一次方向
net dp loc=p39; 百位小数点,常亮
net led<6> loc=p24; 数码管段选
net led<5> loc=p22;
net led<4> loc=p20;
net led<3> loc=p19;
net led<2> loc=p18;
net led<1> loc=p14;
net led<0> loc=p13;
net dig<3> loc=p8; 数码管位选
net dig<2> loc=p9;
net dig<1> loc=p11;
net dig<0> loc=p12;
net light<2> loc=p1; 频率状态指使灯
net light<1> loc=p40;
net light<0> loc=p42;
net dire loc=p29; 方向状态指使灯
5.元件clk_25说明:
library IEEE; ——clk_25用来产生计数频率和
use IEEE.std_logic_1164.all; 扫描频率
use IEEE.std_logic_unsigned.all;
entity clk_25hz is
port (
clks,clks1,clks2,clks3,clks4,clks5,wer: inout STD_LOGIC);
end clk_25hz;
architecture clk_25hz_arch of clk_25hz is
signal clkin:std_logic;
signal counter:std_logic_vector(12 downto 0);
signal aq:std_logic_vector(1 downto 0);
signal fp:std_logic_vector(4 downto 0);
component os -----只要熟悉分频,这边很简单
port (osout:out std_logic);
end component;
begin
u1:os port map(clkin);
wer<=fp(4);
process(clkin)
begin
if(clkin'event and clkin='1')then
fp<=fp+1;
end if;
end process;
process(wer)
begin
if wer='1' and wer'event then
if(counter="1111000000000")then
clks<=not clks;
counter<="0000000000000";
else
COUNTER<=COUNTER+1;
end if;
end if;
end process;
process(clks)
begin
if(clks'event and clks='1')then
aq<=aq+1;
end if;
end process;
clks1<=aq(1);
process(clks1)
begin
if(clks1'event and clks1='1')then
clks2<=not clks2;
end if;
end process;
process(clks2)
begin
if(clks2'event and clks2='1')then
clks3<=not clks3;
end if;
end process;
process(clks3)
begin
if(clks3'event and clks3='1')then
clks4<=not clks4;
end if;
end process;
process(clks4)
begin
if(clks4'event and clks4='1')then
clks5<=not clks5;
end if;
end process;
end clk_25hz_arch;
6. muli_fre_light具体实现方法:
#1.新建一个文件夹,将muli_fre_light.vhd, muli_hz_vhd, os.xnf, muli_fre_light.ucf
copy 到该文件夹下。
#2.运行xilinx软件,并将管脚锁定。
#3.跳线用于启动和清零,S2用于切换频率,S1用于切换方向。
1. 系统功能及要求:
该计数器应能实现加减计数,还应该有异步复位功能;与一般计数器不同的是该计数器
应可以变频;如果你想用不同频率实现计数,只要通过EDA实验板上按键进行切换即
可。如果你想实现加减计数,也只要预设一下方向状态即可.如果你想用某一频率进行
某一方向计数,只要预设方向和频率即可。
2.制作思路:
该系统核心是加减计数器,计数方向受方向按键控制,计数器采用串行进位。
要想实现不同频率计数,需做一系列方波信号,然后将这些不同频率的脉冲通过数据选
择器选择一个作为计数脉冲。
复位信号与方向控制信号需做成单脉冲行式,故需制作单脉冲发生器。
计数频率应适当,人眼必须能明显分辨出计数频率的变化,在这里共作了6档频率,最低
的约为0.5hz,然后每档以二倍向上增长。
要将单脉冲通过触发器,触发器输出用于判断该切换频率还是该切换计数方向。
数码管用于显示计数情况,LED指使灯最右边三个用于指示现在处于哪个计数频率,
最左边一个用于指示现在是加法计数还是减法计数。
LED右三个指使灯为000时处于0号计数频率;为001时处于1号计数频率;为010时处于2号计数频率;为011时处于3号计数频率;为100时处于4号计数频率;为101时处于5号计数频率。
LED最左边一个为0(不亮)时实行加法计数,为1(亮)时实行减法计数。
注意:LED灯亮表示1,但此时给LED送的电平为低。
3. 源程序注释:
library IEEE;
use IEEE.std_logic_1164.all;
use IEEE.std_logic_unsigned.all;
entity muli_fre_light is
port (
change: in STD_LOGIC; -------频率切换按键
reset: in STD_LOGIC; -------复位与开始计数按键
direct:in std_logic; -------方向切换按键
dig: out std_logic_vector(3 downto 0); ------数码管位选
led: out STD_LOGIC_VECTOR (6 downto 0); ------数码管位选
dp:out std_logic; -------小数点点亮
dire:out std_logic; -------计数方向指使灯
light: out STD_LOGIC_VECTOR (2 downto 0) -------频率状态指使灯
);
end muli_fre_light;
architecture muli_fre_light_arch of muli_fre_light is
component clk_25hz ---------元件clk_25用来产生6种计数频率和LED扫描
port(clks,clks1,clks2,clks3,clks4,clks5,wer:inout STD_LOGIC); 频率
end component;
signal refresh:std_logic_vector(1 downto 0);
signal zzm:std_logic_vector(2 downto 0);
signal bcd1,bcd2,bcd3,bcd4,bcd:std_logic_vector(3 downto 0);
signal carry1,carry2,carry3,clock,clks,clks1,clks2,clks3,clks4,clks5,wer,sel:std_logic;
signal chan1,chan0,direct0,direct1,dir,a_s:std_logic;
begin
u2:clk_25hz port map(clks,clks1,clks2,clks3,clks4,clks5,wer);-----调用clk_25产生
各种频率
process(clks1)
begin
if(clks1'event and clks1='0')then -----单脉冲信号sel dir产生
chan1<=chan0;
chan0<=change;
direct1<=direct0;
direct0<=direct;
end if;
end process;
process(chan0,direct0) dir为切换方向按键按下后
begin 产生的单脉冲
sel<=clks1 and chan0 and (not chan1); sel为切换频率按键按下后
dir<=clks1 and direct0 and (not direct1); 产生的单脉冲
end process;
process(sel) ------每次频率切换通过zzm记录所处
begin 频率状态
if(sel'event and sel='1')then
if(zzm="101")then
zzm<="000";
else
zzm<=zzm+1;
end if;
end if;
end process;
light<=not zzm; ------当前频率状态送LED右三位显示
process(dir) -------每次方向切换通过a_s记录所处
begin 方向状态
if(dir'event and dir='1')then
a_s<=not a_s;
end if;
end process;
dire<=not a_s; -----将方向状态送LED最左边显示
clock<=clks when zzm="000" else -----6选1 mux选择计数脉冲
clks1 when zzm="001" else
clks2 when zzm="010" else
clks3 when zzm="011" else
clks4 when zzm="100" else
clks5 ;
process(clock) ------clock为mux输出信号
begin
if(reset='0')then ------如果reset=0则允许加减计数
if(clock'event and clock='1')then 此时如a_s=0则做减法
if(a_s='0')then --add count 此时如a_s=1则做加法
if(bcd1="1001")then
bcd1<="0000";
carry1<='1'; 个位计数
else
bcd1<=bcd1+1;
carry1<='0';
end if;
else --sub count
if(bcd1="0000")then
bcd1<="1001";
carry1<='1';
else
bcd1<=bcd1-1;
carry1<='0';
end if;
end if;
end if;
else
bcd1<="0000";
end if;
end process;
process(carry1,clock)
begin
if(carry2=’1’)then *************这个if 很重要,如没有则会出现错误
carry2<=’0’;**************
else
if(reset='0')then
if(carry1'event and carry1='1')then 十位计数
if(a_s='0')then --add count
if(bcd2="1001")then
bcd2<="0000";
carry2<='1';
else
bcd2<=bcd2+1;
carry2<='0';
end if;
else --sub count
if(bcd2="0000")then
bcd2<="1001";
carry2<='1';
else
bcd2<=bcd2-1;
carry2<='0';
end if;
end if;
end if;
else
bcd2<="0000";
end if;
end if;
end process;
process(carry2,clock)
begin
if(carry3=’1’)then **********************************
carry3<=’0’;
else
if(reset='0')then
if(carry2'event and carry2='1')then
if(a_s='0')then --add count 百位计数
if(bcd3="1001")then
bcd3<="0000";
carry3<='1';
else
bcd3<=bcd3+1;
carry3<='0';
end if;
else --sub count
if(bcd3="0000")then
bcd3<="1001";
carry3<='1';
else
bcd3<=bcd3-1;
carry3<='0';
end if;
end if;
end if;
else
bcd3<="0000";
end if;
end if;
end process;
process(carry3)
begin
if(reset='0')then
if(carry3'event and carry3='1')then 千位计数
if(a_s='0')then --add count
if(bcd4="1001")then
bcd4<="0000";
else
bcd4<=bcd4+1;
end if;
else --sub count
if(bcd4="0000")then
bcd4<="1001";
else
bcd4<=bcd4-1;
end if;
end if;
end if;
else
bcd4<="0000";
end if;
end process;
--display ------使百位小数点一直点亮
dp<='1' when refresh="10" else
'0';
LED<= "0111111" when(BCD="0000") else -----数码管7段译码
"0000110" when(BCD="0001") else
"1011011" when(BCD="0010") else
"1001111" when(BCD=" 0011") else
"1100110" when(BCD="0100") else
"1101101" when(BCD="0101") else
"1111101" when(BCD="0110") else
"0000111" when(BCD="0111") else
"1111111" when(BCD="1000") else
"1101111" when(BCD="1001") else
"XXXXXXX";
--bcd for display
BCD<=BCD1 when(refresh="00" ) else ----refresh
BCD2 when(refresh="01" ) else
BCD3 when(refresh="10" ) else
BCD4 when(refresh="11" ) else
"XXXX";
process(wer) -----wer为扫描脉冲
begin
if(wer'event and wer='1')then
refresh<=refresh+1; -----通过refresh四种状态选择点亮
end if; 数码管
end process;
--dig signal generate
with REFRESH select -------数码管位选译码
DIG <= "1110" when "00",
"1101" when "01",
"1011" when "10",
"0111" when "11",
"XXXX" when others;
end muli_fre_light_arch;
4.管脚锁定文件说明:
net reset loc=p33; 复位和启动管脚,在EDA板上为跳线最左端脚
使用时把P33接地则开始计数,把P33不接地则清零
net change loc=p7; 频率选择按键,每按一下换一档频率
net direct loc=p6; 方向选择按键,每按一下换一次方向
net dp loc=p39; 百位小数点,常亮
net led<6> loc=p24; 数码管段选
net led<5> loc=p22;
net led<4> loc=p20;
net led<3> loc=p19;
net led<2> loc=p18;
net led<1> loc=p14;
net led<0> loc=p13;
net dig<3> loc=p8; 数码管位选
net dig<2> loc=p9;
net dig<1> loc=p11;
net dig<0> loc=p12;
net light<2> loc=p1; 频率状态指使灯
net light<1> loc=p40;
net light<0> loc=p42;
net dire loc=p29; 方向状态指使灯
5.元件clk_25说明:
library IEEE; ——clk_25用来产生计数频率和
use IEEE.std_logic_1164.all; 扫描频率
use IEEE.std_logic_unsigned.all;
entity clk_25hz is
port (
clks,clks1,clks2,clks3,clks4,clks5,wer: inout STD_LOGIC);
end clk_25hz;
architecture clk_25hz_arch of clk_25hz is
signal clkin:std_logic;
signal counter:std_logic_vector(12 downto 0);
signal aq:std_logic_vector(1 downto 0);
signal fp:std_logic_vector(4 downto 0);
component os -----只要熟悉分频,这边很简单
port (osout:out std_logic);
end component;
begin
u1:os port map(clkin);
wer<=fp(4);
process(clkin)
begin
if(clkin'event and clkin='1')then
fp<=fp+1;
end if;
end process;
process(wer)
begin
if wer='1' and wer'event then
if(counter="1111000000000")then
clks<=not clks;
counter<="0000000000000";
else
COUNTER<=COUNTER+1;
end if;
end if;
end process;
process(clks)
begin
if(clks'event and clks='1')then
aq<=aq+1;
end if;
end process;
clks1<=aq(1);
process(clks1)
begin
if(clks1'event and clks1='1')then
clks2<=not clks2;
end if;
end process;
process(clks2)
begin
if(clks2'event and clks2='1')then
clks3<=not clks3;
end if;
end process;
process(clks3)
begin
if(clks3'event and clks3='1')then
clks4<=not clks4;
end if;
end process;
process(clks4)
begin
if(clks4'event and clks4='1')then
clks5<=not clks5;
end if;
end process;
end clk_25hz_arch;
6. muli_fre_light具体实现方法:
#1.新建一个文件夹,将muli_fre_light.vhd, muli_hz_vhd, os.xnf, muli_fre_light.ucf
copy 到该文件夹下。
#2.运行xilinx软件,并将管脚锁定。
#3.跳线用于启动和清零,S2用于切换频率,S1用于切换方向。
我来回答
回答0个
时间排序
认可量排序
暂无数据
或将文件直接拖到这里
悬赏:
E币
网盘
* 网盘链接:
* 提取码:
悬赏:
E币
Markdown 语法
- 加粗**内容**
- 斜体*内容*
- 删除线~~内容~~
- 引用> 引用内容
- 代码`代码`
- 代码块```编程语言↵代码```
- 链接[链接标题](url)
- 无序列表- 内容
- 有序列表1. 内容
- 缩进内容
- 图片![alt](url)
相关问答
-
2012-12-24 14:52:36
-
2015-11-04 12:05:37
-
2012-12-24 14:56:39
-
2013-08-24 22:32:51
-
2018-10-29 11:56:13
-
2020-11-22 18:03:55
-
2023-01-29 08:59:34
-
2008-10-02 20:18:52
-
2008-10-02 20:25:24
-
2008-10-02 20:51:40
-
2008-10-02 20:19:41
-
2008-10-02 20:27:32
-
2018-11-21 13:51:14
-
2008-10-02 20:40:04
-
2019-01-11 10:37:04
-
2018-12-11 15:22:54
-
2013-06-24 10:17:35
-
2020-10-10 17:53:41
-
2013-12-10 20:07:17
无更多相似问答 去提问
点击登录
-- 积分
-- E币
提问
—
收益
—
被采纳
—
我要提问
切换马甲
上一页
下一页
悬赏问答
-
5SS928的emmc有32GB,bootargs设置使用16GB,但是为啥能用的只有rootfs的大小
-
33SS928怎样烧写ubuntu系统
-
10ToolPlatform下载rootfs提示网络失败
-
10谁有GK7205V500的SDK
-
5Hi3516CV610 烧录不进去
-
10Hi3559AV100 芯片硬解码h265编码格式的视频时出现视频播放错误,解码错误信息 s32PackErr:码流有错
-
5海思SS928 / SD3403的sample_venc.c摄像头编码Demo中,采集到的摄像头的YUV数据在哪个相关的函数中?
-
5海鸥派openEuler无法启动网卡,连接WIFI存在问题
-
66有没有ISP相关的巨佬帮忙看看SS928对接IMX347的图像问题
-
50求助hi3559与FPGA通过SLVS-EC接口对接问题
举报反馈
举报类型
- 内容涉黄/赌/毒
- 内容侵权/抄袭
- 政治相关
- 涉嫌广告
- 侮辱谩骂
- 其他
详细说明
提醒
你的问题还没有最佳答案,是否结题,结题后将扣除20%的悬赏金
取消
确认
提醒
你的问题还没有最佳答案,是否结题,结题后将根据回答情况扣除相应悬赏金(1回答=1E币)
取消
确认