我是靠谱客的博主 小巧小松鼠,这篇文章主要介绍HDLBits/Counters/12-hour clock 记录一下个人思路/自用,现在分享给大家,希望可以做个参考。

要求:

Create a set of counters suitable for use as a 12-hour clock (with am/pm indicator). Your counters are clocked by a fast-running clk, with a pulse on ena whenever your clock should increment (i.e., once per second).

reset resets the clock to 12:00 AM. pm is 0 for AM and 1 for PM. hh, mm, and ss are two BCD (Binary-Coded Decimal) digits each for hours (01-12), minutes (00-59), and seconds (00-59). Reset has higher priority than enable, and can occur even when not enabled.

思路:

1. ss、mm、hh都是由两个bcd码组成的,所以要分成十位和个位,否则会显示成16进制;

2. 分成十位和分位后,就不能使用模60和模12计数器,也可以不用计数器,但是这样代码中重复太多,还是选用计数器并引用模块更简洁明了。ss和mm个位改用模10计数器(0-9),十位改用模6计数器(0-5)。因为时十位只有1、0两个状态,所以hh采用if语句直接置数比较方便;

3. 首先写两个计数器并引用模块,注意每位的使能端都是与上一位的输出有关,这是这部分的重点,搞清这个基本就能解决;

4. pm注意都是在11时59分59秒状态发生转换,注意不能直接令pm为!pm,会报错,要先赋成另一个值。

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
module top_module( input clk, input reset, input ena, output pm, output [7:0] hh, output [7:0] mm, output [7:0] ss); reg [3:0] ss_one; reg [3:0] ss_ten; reg [3:0] mm_one; reg [3:0] mm_ten; reg [3:0] hh_one; reg [3:0] hh_ten; //秒 wire ena_ss_ten; //秒十位的使能端 assign ena_ss_ten=ena&&(ss_one==4'd9); //9秒且ena高电平时秒十位开始计数 counter10 counter_ss_one(clk, ena, reset, ss_one); counter6 counter_ss_ten(clk, ena_ss_ten, reset, ss_ten); assign ss={ss_ten, ss_one};//输出秒 //分 wire ena_mm_one; wire ena_mm_ten; assign ena_mm_one=(ss_ten==4'd5)&&(ss_one==4'd9); //59秒时分个位开始计数 assign ena_mm_ten=ena_mm_one&&(mm_one==4'd9); //9分59秒分十位开始计数 counter10 counter_mm_one(clk, ena_mm_one, reset, mm_one); counter6 counter_mm_ten(clk, ena_mm_ten, reset, mm_ten); assign mm={mm_ten, mm_one};//输出分 //时 wire ena_hh; assign ena_hh=ena_mm_ten&&(mm_ten==4'd5);//59分59秒时时个位开始计数 always@(posedge clk)begin if(reset)begin hh_one<=4'd2;//重置时为12 hh_ten<=4'd1; end else if(ena_hh)begin if(hh_one==4'd9&&(hh_ten==4'd0))begin hh_one<=4'd0; hh_ten<=4'd1;//时为09时,置数为10 end else if(hh_one==4'd2&&(hh_ten==4'd1))begin hh_one<=4'd1; hh_ten<=4'd0;//时为12时,置数为01 end else begin hh_one<=hh_one+1'b1; end end else if(!ena_hh)begin hh_one<=hh_one; hh_ten<=hh_ten; end end assign hh={hh_ten, hh_one};//输出时 //pm的设置 wire pm_temp; wire ena_pm; assign pm_temp=pm; assign ena_pm=ena_hh&&(hh_ten==4'd1)&&(hh_one==4'd1);//11时59分59秒时翻转 always@(posedge clk)begin if(reset)begin pm<=1'b0; end else if(ena_pm)begin pm<=~pm_temp;//转置 end else if(!ena_pm)begin pm<=pm; end end endmodule module counter10( input clk, input enable, input reset, output [3:0] q);//模10计数器 always@(posedge clk)begin if(reset)begin q<=4'd0; end else if(enable)begin if(q>=4'd9)begin q<=4'd0; end else begin q<=q+1'b1; end end else if(!enable)begin q<=q; end end endmodule module counter6( input clk, input enable, input reset, output [3:0] q);//模6计数器 always@(posedge clk)begin if(reset)begin q<=4'd0; end else if(enable)begin if(q>=4'd5)begin q<=4'd0; end else begin q<=q+1'b1; end end else if(!enable)begin q<=q; end end endmodule

一些基础语法复习:

assign后面只能用阻塞性语句;

if语句都要带begin end,之前写代码习惯不好,要改掉。

最后

以上就是小巧小松鼠最近收集整理的关于HDLBits/Counters/12-hour clock 记录一下个人思路/自用的全部内容,更多相关HDLBits/Counters/12-hour内容请搜索靠谱客的其他文章。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(62)

评论列表共有 0 条评论

立即
投稿
返回
顶部