|
EDA365欢迎您登录!
您需要 登录 才可以下载或查看,没有帐号?注册
x
/ q4 `' Z) v3 f/ W5 T! s6 q1 N7 {# u, d. n; U/ N7 b1 o) R
这是特权同学在《深入浅出玩转FPGA》里面的“基于FPGA的跨时钟域信号处理”一章里面的一个例子,以此来强调同步设计思想。该程序因为脉冲信号和由CPU控制总线产生的选通信号是来至两个异步时钟域的信号,它们作为内部时钟信号时,如果同一时刻出现一个时钟在写寄存器counter,另一个时钟在读寄存器counter,那么很明显存在发生冲突的可能。: u4 k* f# p: }. X: y( Y I: U
module fp(4 A3 {, ^( Y: M0 G+ r# U
clk,
d" _/ h5 M) E1 r" U% _ rst,* F' H. ~% ~, n2 h
pulse, //脉冲信号,高电平有效' {4 u$ m c% y
cs_n, //低电平有效
- ^& I6 A+ m( o" P rd_n, //读使能信号,低电平有效
( o7 `" X' p: j+ R addr_bus, //地址总线
; p9 t6 ~7 U$ U( _+ o4 w5 m( l# k data_bus //数据总线
7 c8 t: d1 x" B. d) Z+ M n( d );9 @8 o) a' F; ]
input clk;+ @ h/ s* k+ R
input rst;% e# p, n! [& T2 N5 Q
input plush;9 o5 {. N$ y7 \* J9 ^- N: Y
input cs_n;
& L/ B. j( U5 _' f- G input rd_n;
/ R8 U& d3 @1 ?( N6 Q9 I input [3:0] addr_bus;
! ?; v' j0 a/ P. O
. S- W- r5 m; y s: r# o9 w7 Z* A/ O output [15:0] data_bus;, A+ R+ y# c5 @8 ^5 H
2 I! e+ `6 T7 \& h
reg [15:0] counter;
' b* K: T3 Y+ s3 l$ z& q G' ] reg [15:0] data_bus;" ^/ ^& s1 ~7 ~1 k, n' o9 R
) O' w" f8 M5 l" B( i% _' J) x always @(posedge plush or negedge rst) K4 C6 w) {& b, r
if(!rst) 2 i" |2 ?# P& ]& K
counter <= 16'b0;
! ^' ~9 B7 ^# i9 O; a4 N/ \ else if(plush)
# m) M% J0 \; ~* B- c6 [ counter <= counter + 1'b1;, B4 K6 Z4 y* }- Q3 F1 k" K# d8 k
3 S! z1 u* `1 F( `" `
wire DSP_cs = cs_en || rd_en; ! N1 U3 o; ^( A4 t
always @(dsp_cs or addr_bus)
- T2 w5 A$ L( g+ Q, p9 o if(dsp_cs) . h4 {% |% @ Q' u8 l/ Q, H
data_bus <= 16'hzzzz;; T6 w8 t! z4 } L
else begin
2 x5 i* Y) F+ ~9 y, C case (addr_bus)4 c8 A7 z9 v: U% y1 T1 b5 Z
4'h0: data_bus <= counter;* Y9 J2 a8 [/ K* O
..." u' z |& v8 }& ~+ o
endcase
7 ~/ e3 k$ v2 t! ^/ I end
1 g$ \; f/ Z7 }7 I1 Q0 A: B endmodule
) D7 N8 u) L5 B4 d
. F, L7 r# |+ v# @; ~& { 该例同步设计方法:先使用脉冲同步法把脉冲信号与系统时钟信号clk同步,然后使用脉冲同步法得到一个系统时钟宽度的使能信号作为数据锁存信号,同时也将CPU的控制信号与系统时钟信号同步。本人根据这个思想写了如下程序。我的疑问是同一个时刻clk出现写counter和读counter,怎么不会发生数据冲突。6 S" q* _4 N5 o( y
module fp(6 }8 c2 J0 d0 W7 D9 v4 O
clk,
8 T( q5 s: L z# T; o rst,( J5 _) P1 V+ i% @. c6 _; y
pulse, //脉冲信号,高电平有效" X \0 {6 R" {: v E" E3 _
cs_n, //低电平有效
' l7 B# f- _2 C i rd_n, //读使能信号,低电平有效
. _$ M2 o2 l; Y. W# o addr_bus, //地址总线; Z5 N) d7 N2 v* R! R* O( H' y
data_bus //数据总线
; y* U a& H6 S. v8 D );
. H" p) V& x, h: K. Iinput clk;" L6 |7 v, Z+ x9 L+ \ T
input rst;
+ z. [- ?0 o p2 v; _* yinput plush;2 y' g6 P- g) O/ P2 g2 {
input cs_n;5 J; g# W# L4 q' W; e
input rd_n;5 g J1 w, h7 z/ p' K( L
input [3:0] addr_bus;
8 {- }# \+ V t7 }+ q. Z7 i0 z z1 p$ M1 {' Q
output [15:0] data_bus;
D( X) W H2 {. F+ h7 B6 H9 D Y; O
reg [15:0] counter;
3 ?, Q1 r6 U }- c& Nreg [15:0] data_bus;
: U. A" c! z, c1 v9 M0 C
! Y: ^# f7 f# y1 B+ D2 e; @$ xreg preg1,preg2;
6 J1 @' V% R( \" Yalways @(posedge clk or negedge rst)6 f; J5 y _4 m7 @3 x( C8 q9 ~1 d
if(!rst) begin3 B0 G% E) w* Y0 x) q8 Z$ U
preg1 <= 1'b1;
0 H. l! _4 p( I preg2 <= 1'b1;
; o* F& j' Y' H end
0 J) w' @6 k) s$ e2 d0 r else begin5 g. E( n6 l& r6 ^2 c$ q! h/ a. [
preg1 <= plush;
6 L( W7 z0 B1 X5 h% {7 j# e preg2 <= preg1;7 G9 t. c! u! t: w/ a) ?+ \+ q
end3 x# W, h0 Z4 O
wire pos_en;3 `% n7 q. {, w! M
assign pos_en = preg1 & (~preg2); //检测脉冲信号,一个时钟周期有效的高电平
. I% y4 \! a5 S2 i- V& i& p
, R( r* [9 [( L" O( n) B) _always @(posedge clk or negedge rst)
% [9 T! I' O: p- a2 ?; H; K6 Y# W if(!rst) . f+ r. e& R0 M* v( a% l- O
counter <= 16'b0;8 c7 n1 V. ]4 @! }0 w5 O
else if(pos_en) 3 d" M1 @1 d+ m* a" v8 |; B5 B, O
counter <= counter + 1'b1;+ |* t" J: v; f- A3 w( P% X* R5 q
1 G! y8 Y4 K1 _2 ~( H* ^
3 }5 ]4 f/ @6 }5 }* ~9 }; M0 @5 Q+ Kwire pos_en = cs_en || rd_en; //片选信号和读信号同时为低时,对地址总线进行译码,把采样脉冲值送给数据总线
9 a6 c! z9 a( e, wreg dreg1,dreg2;
/ H9 r8 }8 F2 p, ]9 _; salways @(posedge clk or negedge rst)1 h! f3 z, ^/ G
if(!rst) begin7 A' ?2 a' G6 f$ r
dreg1 <= 1'b1;
- q4 c; D1 Q: [5 \& q) k dreg2 <= 1'b1;
5 E1 I- O& i. p1 ^0 K6 _ end. R5 i& E9 Q" [) O
else begin
3 o2 u P( w& J/ @ dreg1 <= ce_en;" Z/ A" U' F# E1 S; v- {) l- b% N
dreg2 <= preg1;
4 e, t' l9 a6 E0 U3 I( P( Z end9 R/ r6 @- J' M- _" K- P% D
wire dsp_cs;: E) O8 W4 i$ E+ O' e' G
assign dsp_cs = dreg2 & (~dreg1); //使能信号
$ v( u7 O+ ?4 z6 t8 Z& o, S& Hs
' I* u) X) Q f5 j! t, palways @(posedge clk or negedge rst)6 P& D. C; V8 e. u5 v( }
if(!rst)
$ I/ O/ s& f! c* n& J data_bus <= 16'hzzzz;0 ? x# t/ F/ j" T! Y
else if(dsp_cs) begin8 p' _- I+ A3 f6 L' b t
case (addr_bus)- |* u9 j& ~! O; R' C
4'h0: data_bus <= counter;
: n9 t- K& f! X* Q S ...% q0 N4 `- `. e' S
endcase
4 ~% ?7 ^" p1 D3 b) C* w$ Q" F; Pend4 d; Q( X) t/ z
endmodule/ P0 e# Z1 f* k9 x
|
|