always@(a or b) case(a) 1'b0:f=b;1'b1:g=b;endcase 为什么不可综合
时间: 2023-06-15 18:05:43 浏览: 142
这段代码不可综合的原因是它使用了非阻塞赋值语句(<=)和敏感列表(always@(a or b))。在 Verilog 中,敏感列表必须是一个确定的、静态的信号列表(如 always@(a)),而不能是一个表达式(如 always@(a or b))。因此,这段代码不符合综合的要求。
此外,这段代码的行为也不够明确。在 a 的值为 0 时,它将 f 赋值为 b,但在 a 的值为 1 时,它将 g 赋值为 b。这种行为可能会导致数据冲突和不确定性,因此不应该在硬件设计中使用。如果想要实现类似的功能,应该使用条件语句或选择语句来明确地指定赋值的条件和结果。
相关问题
下列代码输出pwm,是否有错: parameter cnt_end = 5000; parameter cnt_25 = cnt_end / 4; parameter cnt_50 = cnt_end / 2; parameter cnt_75 = cnt_end * 3 / 4; reg [24:0] count; always @(posedge clk, posedge clr) begin if(clr) begin count <= 0; end else begin case(count) cnt_end: count <= 0; default: count <= count + 1; endcase end end // 只需要进行一次比较,并根据结果进行赋值 assign pwm1 = (count<=cnt_25)?1'b1:1'b0; assign pwm2 = (count<=cnt_50)?1'b1:1'b0; assign pwm3 = (count<=cnt_75)?1'b1:1'b0; assign pw = (duty == 2'b01)? pwm1:(duty == 2'b10)? pwm2:(duty == 2'b11)? pwm3:1'b0; assign pw1 = (duty1 == 2'b01)? pwm1:(duty1 == 2'b10)? pwm2:(duty1 == 2'b11)? pwm3:1'b0; assign pw2 = (duty2 == 2'b01)? pwm1:(duty2 == 2'b10)? pwm2:(duty2 == 2'b11)? pwm3:1'b0; assign pw3 = (duty3 == 2'b01)? pwm1:(duty3 == 2'b10)? pwm2:(duty3 == 2'b11)? pwm3:1'b0;
这段代码存在一些问题。
首先,在always块中,使用了非阻塞赋值语句“<=”来更新count寄存器的值。这是正确的。但是,在case语句中,将count赋值为0时,使用了阻塞赋值语句“=”,这是不正确的。应该改为非阻塞赋值语句“<=”。
其次,在assign语句中,根据输入的duty值计算PWM输出的逻辑存在问题。对于duty的值为2'b00的情况,没有进行处理,应该添加一个条件分支来处理这种情况。
最后,在assign语句中,将pwm1、pwm2和pwm3分别赋值给了pw1、pw2和pw3。这是不正确的,应该将它们分别赋值给pw1、pw2和pw3。
以下是修改后的代码:
```verilog
parameter cnt_end = 5000;
parameter cnt_25 = cnt_end / 4;
parameter cnt_50 = cnt_end / 2;
parameter cnt_75 = cnt_end * 3 / 4;
reg [24:0] count;
always @(posedge clk, posedge clr) begin
if(clr) begin
count <= 0;
end else begin
case(count)
cnt_end:
count <= 0;
default:
count <= count + 1;
endcase
end
end
assign pwm1 = (count <= cnt_25) ? 1'b1 : 1'b0;
assign pwm2 = (count <= cnt_50) ? 1'b1 : 1'b0;
assign pwm3 = (count <= cnt_75) ? 1'b1 : 1'b0;
assign pw = (duty == 2'b00) ? 1'b0 :
(duty == 2'b01) ? pwm1 :
(duty == 2'b10) ? pwm2 :
(duty == 2'b11) ? pwm3 :
1'b0;
assign pw1 = (duty1 == 2'b00) ? 1'b0 :
(duty1 == 2'b01) ? pwm1 :
(duty1 == 2'b10) ? pwm2 :
(duty1 == 2'b11) ? pwm3 :
1'b0;
assign pw2 = (duty2 == 2'b00) ? 1'b0 :
(duty2 == 2'b01) ? pwm1 :
(duty2 == 2'b10) ? pwm2 :
(duty2 == 2'b11) ? pwm3 :
1'b0;
assign pw3 = (duty3 == 2'b00) ? 1'b0 :
(duty3 == 2'b01) ? pwm1 :
(duty3 == 2'b10) ? pwm2 :
(duty3 == 2'b11) ? pwm3 :
1'b0;
```
在修改后的代码中,修正了always块中赋值语句的问题,并添加了对duty值为2'b00的情况的处理。同时,将pwm1、pwm2和pwm3正确地赋值给了pw1、pw2和pw3。
module cout(clk,clk1,clk2,clk3,clk4,clk5,start,pause,msh,msl,sh,sl,rst,kin,kout,clk,wei,shi_h,shi_l,fen_h,fen_l,duan, a,led7s); input clk,clk3,clk4,clk5,start,pause,rst,kin; output clk1; output clk2; reg [15:0]k2; reg[7:0] k1; reg clk2; reg clk1; output [3:0]msh,msl,sh,sl; reg[3:0] msh,msl,sh,sl; reg cn1; reg start1=1,pause1=1,rst1=0; output kout; reg kout; reg [3:0]kh,kl; input [3:0]shi_h,shi_l,fen_h,fen_l; output [3:0]duan; output [3:0]wei; reg [3:0]duan; reg [3:0]wei; parameter s0=0,s1=1,s2=2,s3=3; reg [3:0]c_st,n_st; input[3:0]a; output[6:0]led7s; reg[6:0]led7s; //分频模块 always@(posedge clk2) begin if(k2<16'd12499) k2=k2+8'd1; else k2=0; if(k2==16'd12499) clk2=clk2+1;//clk2=2000hz end always @(posedge clk2) begin if(k1<8'd9) k1=k1+8'd1; else k1=0; if(k1==8'd9) clk1=clk1+1;//clk1=100hz end //计数模块 always @(posedge start) start1=~start1; always @(posedge pause) pause1=~pause1; always @(posedge rst) rst1=rst1+1'b1; always @(posedge clk3 or negedge rst1 ) begin if(!rst1) begin{msh,msl}<=8'h00; cn1<=0; end else if(pause1^start1) begin if(msl==9) begin msl<=0; if(msh==9) begin msh<=0; cn1<=1; end else msh<=msh+1'h1; end else begin msl<=msl+1'h1; cn1<=0; end end end always @(posedge cn1 or negedge rst1 ) begin if(!rst1) begin{sh,sl}<=8'h00; end else if(start1^pause1) begin if(sl==9) begin sl<=0; if(sh==5) sh<=0; else sh<=sh+1'h1; end else begin sl<=sl+1'h1; end end end //按键消抖模块 always@(posedge clk4) begin if(!kin) kl<=kl+1'b1; else kl<=4'b0000; end always@(posedge clk4) begin if(kin) kh<=kh+1'b1; else kh<=4'b0000; end always@(posedge clk4) begin if(kh>4'b1100) kout<=1'b1; else if(kl>4'b0111) kout<=1'b0; end //数码管位选模块 always@(posedge clk5) begin c_st<=n_st; end always@* begin case(c_st) s0:begin n_st=s1;wei<=4'b0111;duan<=shi_h; end s1:begin n_st=s2;wei<=4'b1011;duan<=shi_l; end s2:begin n_st=s3;wei<=4'b1101;duan<=fen_h; end s3:begin n_st=s0;wei<=4'b1110;duan<=fen_l; end default:begin n_st=s1;wei<=4'b0111;duan<=shi_h; end endcase end //数码管显示模块 always@(a) case(a) 4'b0000 : led7s<=~7'b0111111; 4'b0001 : led7s<=~7'b0000110; 4'b0010 : led7s<=~7'b1011011; 4'b0011 : led7s<=~7'b1001111; 4'b0100 : led7s<=~7'b1100110; 4'b0101 : led7s<=~7'b1101101; 4'b0110 : led7s<=~7'b1111101; 4'b0111 : led7s<=~7'b0000111; 4'b1000 : led7s<=~7'b1111111; 4'b1001 : led7s<=~7'b1101111; 4'b1010 : led7s<=~7'b1110111; 4'b1011 : led7s<=~7'b1111100; 4'b1100 : led7s<=~7'b0111001; 4'b1101 : led7s<=~7'b1011110; 4'b1110 : led7s<=~7'b1111001; 4'b1111 : led7s<=~7'b1110001; default : led7s<=~7'b0111111; endcase endmodule
这段代码是一个 Verilog HDL 代码,实现了一个数字钟的功能,包括分频模块、计数模块、按键消抖模块、数码管位选模块和数码管显示模块。其中,分频模块将输入时钟分频得到一个 2 kHz 的时钟信号,计数模块使用该时钟信号实现秒的计数和分的计数,按键消抖模块实现了按键输入的消抖,数码管位选模块实现了数码管的位选功能,数码管显示模块通过接收输入的数字信号 a 实现了数码管的数码显示。
阅读全文