4周PADS强化班 HOT
林超文手把手教你学!
张飞暑期特训班
教你1000种电路设计思路
年度IP:PFC电源
张飞硬件电路之PFC全集
30天AD项目众筹
参与免费送VIP+原创视频
最全模电系列教程 NEW
运放、ADC、电磁兼容

[FPGA作品] 勇敢的芯伴你玩转Altera FPGA连载59:按键消抖与LED开关实例

[复制链接]
发表于 2018-4-3 20:13:34   1061 查看 0 回复 显示全部楼层 倒序浏览
分享
勇敢的芯伴你玩转Altera FPGA连载59:按键消抖与LED开关实例

特权同学,版权所有

配套例程和更多资料下载链接:

http://pan.baidu.com/s/1i5LMUUD

1.jpg

         除了前面所论及的按键消抖处理,该实验还需要用到LED指示灯进行按键状态的指示。该实验要实现一个独立按键控制一个发光二极管亮暗状态翻转。上电初始,发光二极管不亮,当某一个按键被按下后(即键值为0),发光二极管被点亮,当按键再次被按下时,发光二极管则又灭了,按键控制发光二级管如此反复的进行亮暗变化。

         本实例代码如下。

module cy4(

            input ext_clk_25m, //外部输入25MHz时钟信号

            input ext_rst_n,   //外部输入复位信号,低电平有效

            input[3:0] key_v,//4个独立按键输入,未按下为高电平,按下后为低电平

            output reg[7:0] led    //8个LED指示灯接口   

            );

//-------------------------------------

//按键抖动判断逻辑

wire key;   //所有按键值相与的结果,用于按键触发判断

reg[3:0] keyr;  //按键值key的缓存寄存器

assign key = key_v[0] & key_v[1] & key_v[2] & key_v[3];

always @(posedge ext_clk_25m or negedge ext_rst_n)

    if (!ext_rst_n) keyr <=4'b1111;

    else keyr <= {keyr[2:0],key};

wire key_neg = ~keyr[2] & keyr[3]; //有按键被按下  

wire key_pos = keyr[2] & ~keyr[3]; //有按键被释放

//-------------------------------------

//定时计数逻辑,用于对按键的消抖判断

reg[19:0]  cnt;

    //按键消抖定时计数器

always @ (posedge ext_clk_25m or negedge ext_rst_n)

   if (!ext_rst_n) cnt <= 20'd0;   

    else if(key_pos || key_neg) cnt<= 20'd0;

    else if(cnt < 20'd999_999) cnt<= cnt + 1'b1;

    else cnt <= 20'd0;

  

reg[3:0] key_value[1:0];

    //定时采集按键值

always @(posedge ext_clk_25m or negedge ext_rst_n)

    if (!ext_rst_n) begin

        key_value[0] <= 4'b1111;

        key_value[1] <= 4'b1111;

    end

    else begin

        key_value[1] <=key_value[0];      

        if(cnt == 20'd999_999)key_value[0] <= key_v;  //定时键值采集

        else ;  

    end

wire[3:0] key_press = key_value[1] & ~key_value[0];       //消抖后按键值变化标志位

//-------------------------------------

//LED切换控制

always @ (posedge ext_clk_25m or negedge ext_rst_n)

    if (!ext_rst_n) led <= 8'hff;

    else if(key_press[0]) led[0]<= ~led[0];

    else if(key_press[1]) led[1] <=~led[1];

    else if(key_press[2]) led[2] <=~led[2];

    else if(key_press[3]) led[3] <=~led[3];

    else ;

  

endmodule

         这段代码的前提是,所有4个独立按键,在任意一个按键被按下和释放期间,不会有其它按键也被按下或释放。在通常的应用中,一定也是符合这个假设的场景。

         我们处理消抖的逻辑是这样的:首先将所有按键输入信号做“逻辑与”操作,得到信号key。信号key的值锁存4拍分别存储到寄存器keyr[0]、keyr[1]、keyr[2]和keyr[3]中(此时的采样频率和基准时钟一致,为25MHz),通过keyr[2]和keyr[3]这两个寄存器获得key信号的上升沿标志位key_pos和下降沿标志位key_neg。key_pos和key_neg的获得过程分别如图8.13和图8.14所示,这是很典型的“脉冲边沿检测法”,后续很多代码中我们都会用到这个逻辑。

2.jpg

图8.13 上升沿脉冲检测波形

3.jpg

图8.14 下降沿脉冲检测波形

计数器cnt在key_pos和key_neg有效拉高时,都会清零重新开始计数,计数器cnt的最大计数值为40ms,若在某个固定时间内按键有抖动(这个抖动通常不会大于40ms,这是经验值),那么这段时间内计数器cnt会频繁的清0,cnt的计数值就不会计数到最大值。一旦cnt计数到最大值,我们就会对当前所有的按键值做一次锁存,锁存到4位寄存器key_value[0]中(即这个锁存操作的采样率是40ms为周期的,若按键的抖动小于40ms,那么采样的按键值是不会变化的,那么就达到了消除抖动的目的)。随后,以系统时钟节拍下,key_value[1]会锁存key_value[0]的值。当按键按下操作,产生按键值的下降沿变化,那么key_press就会获得一个时钟周期高脉冲的键值指示信号,通过这个键值指示信号,我们就可以对LED的翻转做相应处理。


只有小组成员才能发言,加入小组>>

209个成员聚集在这个小组

加入小组

创建小组步骤

4周PADS强化班 HOT
林超文手把手教你学!
张飞暑期特训班
教你1000种电路设计思路
年度IP:PFC电源
张飞硬件电路之PFC全集
30天AD项目众筹
参与免费送VIP+原创视频
最全模电系列教程 NEW
运放、ADC、电磁兼容
关闭

站长推荐 上一条 /9 下一条

快速回复 返回顶部 返回列表
-

推荐专区

技术干货集中营

专家问答

用户帮助┃咨询与建议┃版主议事

工程师杂谈

项目|工程师创意

招聘|求职}工程师职场

论坛电子赛事

社区活动专版

发烧友活动

-

嵌入式论坛

ARM技术论坛

Android论坛

Linux论坛

单片机/MCU论坛

MSP430技术论坛

FPGA|CPLD|ASIC论坛

STM32/STM8技术论坛

NXP MCU 技术论坛

PIC单片机论坛

DSP论坛

瑞萨单片机论坛

嵌入式系统论坛

-

电源技术论坛

电源技术论坛

无线充电技术

-

硬件设计论坛

PCB设计论坛

电路设计论坛

电子元器件论坛

控制|传感

总线技术|接口技术

-

测试测量论坛

LabVIEW论坛

Matlab论坛

测试测量技术专区

仪器仪表技术专区

-

EDA设计论坛

multisim论坛

PADS技术论坛

Protel|AD|DXP论坛

Allegro论坛

proteus论坛|仿真论坛

EasyEDA-中国人自已的EDA工具

Orcad论坛

-

综合技术与应用

电机控制

智能电网

光电及显示

工程资源中心

汽车电子技术论坛

医疗电子论坛

-

开源硬件

-

无线通信论坛

无线通信技术专区

天线|RF射频|微波|雷达技术

-

IC设计论坛

芯片测试与失效分析

Mixed Signal/SOC[数模混合芯片设计]

Analog/RF IC设计

设计与制造封装测试

-

厂商专区

TI论坛

TI Deyisupport社区

-

检测技术与质量

电磁兼容(EMC)设计与整改

安规知识论坛

检测与认证

-

消费电子论坛

手机技术论坛

平板电脑/mid论坛

音视/视频/机顶盒论坛

-

电子论坛综合区

聚丰众筹官方社区

新人报道区

聚丰供应链

-

论坛服务区

-

供求信息发布

供需广告

电子展览展会专区

芯片求购|供应发布区