Lane
数字逻辑设计对Verilog使用的记录

数字逻辑设计对Verilog使用的记录

分享网站

一个非常优质的训练verilog代码能力的网站

introduction

进行大作业手撕verilog时会出现很多奇奇怪怪的报错喵,所以我就在这里简简单单整理一下叭。

记录

  • 在Verilog中,wire数据类型不能在always块中被赋值,wire类型只能通过连续赋值或模块端口赋值来改变。如果你需要在always块中改变变量的值,你应该使用reg类型。但是当你需要在always块中要对变量p0至p8做清零操作,并且你在setqi a1模块端口中也需要进行赋值操作,这导致了命名空间的冲突。
    解决这个问题的一个可行的方案是你可以使用额外的reg类型的变量在always块中操作。这意味着你需要对每个p都定义一个wire和reg配对。在always块中,你可以继续对reg类型的变量进行赋值操作。然后,你可以通过连续赋值语句将wire类型的变量设置为对应的reg类型的变量的值。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
wire [2:0]w_p0, w_p1, w_p2, w_p3, w_p4, w_p5, w_p6, w_p7, w_p8;
reg [2:0]r_p0 = 0, r_p1 = 0, r_p2 = 0, r_p3 = 0, r_p4 = 0, r_p5 = 0, r_p6 = 0, r_p7 = 0, r_p8 = 0;
reg count = 1;

//连续赋值将wire变量设置为对应的reg变量的值
assign w_p0 = r_p0;
assign w_p1 = r_p1;
assign w_p2 = r_p2;
assign w_p3 = r_p3;
assign w_p4 = r_p4;
assign w_p5 = r_p5;
assign w_p6 = r_p6;
assign w_p7 = r_p7;
assign w_p8 = r_p8;

setqi a1(.count(count), .sw(sw), .p0(w_p0), .p1(w_p1), .p2(w_p2), .p3(w_p3), .p4(w_p4), .p5(w_p5), .p6(w_p6), .p7(w_p7), .p8(w_p8), .stop(stop));

always @(posedge stop)
begin
r_p0 <= 0; r_p1 <= 0; r_p2 <= 0; r_p3 <= 0; r_p4 <= 0; r_p5 <= 0; r_p6 <= 0; r_p7 <= 0; r_p8 <= 0; //进行清零操作
//引入vga图片显示game over
end
  • 仿真中需要改变的变量一定是reg类型

verilog基础知识

矢量操作

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
取反操作
module top_module(
input [7:0] in,
output [7:0] out
);
assign out={in[0],in[1],in[2],in[3],in[4],in[5],in[6],in[7]};
endmodule
//如果为 assign out[7:0]=in[0:7] 会报错

矢量亦可以复制
{5{1'b1}} // 5'b11111 (or 5'd31 or 5'h1f)
{2{a,b,c}} // The same as {a,b,c,a,b,c}
{3'd5, {2{3'd6}}} // 9'b101_110_110. It's a concatenation of 101 with
// the second vector, which is two copies of 3'b110.
module top_module (
input [7:0] in,
output [31:0] out );//

// assign out = { replicate-sign-bit , the-input };
assign out={ {24{in[7]}},in};
endmodule//括号一定不能少打

多个复制问题
module top_module (
input a, b, c, d, e,
output [24:0] out );//
// The output is XNOR of two vectors created by
// concatenating and replicating the five inputs.
// assign out = ~{ ... } ^ { ... };
assign out=~{{5{a}},{5{b}},{5{c}},{5{d}},{5{e}}}^{5{a,b,c,d,e}};
endmodule

多模块以及选择器:(模块一定要注意对应位置不要出错,太关键了)
module top_module (
input clk,
input [7:0] d,
input [1:0] sel,
output [7:0] q
);
wire [7:0] q1,q2,q3;
my_dff8 a1(clk,d,q1);
my_dff8 a2(clk,q1,q2);
my_dff8 a3(clk,q2,q3);
always @(*) begin
case(sel)
2'b00:q=d;
2'b01:q=q1;
2'b10:q=q2;
default:q=q3;
endcase
end
endmodule

分线与加法器
module top_module(
input [31:0] a,
input [31:0] b,
output [31:0] sum
);
wire [15:0] sum1,sum2;
wire cout1,cout2;
add16 a1(a[15:0],b[15:0],0,sum1,cout1);
add16 a2(a[31:16],b[31:16],cout1,sum2,cout2);
assign sum={sum2,sum1};
endmodule

简单加法器
module add1 ( input a, input b, input cin, output sum, output cout );
assign sum=a^b^cin;
assign cout=(a & b) | (b & cin) | (cin & a);
// Full adder module here

endmodule

加减法 需要注意的是按位与必须两者都要相同位数,则需要进行复制操作
module top_module(
input [31:0] a,
input [31:0] b,
input sub,
output [31:0] sum
);
wire [31:0] b1;
wire [15:0] sum1,sum2;
wire cout1,cout2;
assign b1=b^{32{sub}};
add16 a1(a[15:0],b1[15:0],sub,sum1,cout1);
add16 a2(a[31:16],b1[31:16],cout1,sum2,cout2);
assign sum={sum2,sum1};
endmodule

if语句运用 需要注意的是寄存器必须改变状态
// synthesis verilog_input_version verilog_2001
module top_module (
input cpu_overheated,
output reg shut_off_computer,
input arrived,
input gas_tank_empty,
output reg keep_driving ); //

always @(*) begin
if (cpu_overheated)
shut_off_computer = 1;
else
shut_off_computer=0;
end

always @(*) begin
if (~arrived&&~gas_tank_empty)
keep_driving = 1;
else
keep_driving=0;
end

endmodule

计组lab记录

lab1

任务一 根据结构图写ALU代码

  • 变量前后不一致很关键,如果发现仿真时有不确定变量一定要注意变量
    (大小写或者0和O很容易混淆)

lab2

  • 进行cpu测试模块的书写
  • 但是最后生成比特流时报了离谱的错误
  • 但是连线后面变量仔细琢磨了没有大的问题

lab3

lab3-1 手搓乘法器

  • 很重要的一点是要弄明白Verilog语言中always语句的并发执行,这和我们以前接触过的高级语言不同,Verilog代码中的语句可以不按顺序执行,这个有点像多线程,也就是说多个任务同时进行。
    上面的代码中共有5个always语句,每个always语句都是时钟信号clk的上跳沿触发,也就是说当clk从0变为1的时候,会触发always语句的执行。

lab4

lab4-2 数据通路的实现

  • 这个lab其实最主要的还是连线,但是在第一次仿真的时候我发现Rs1_data和Rs2_data均为0,我还要仔细研究一下。
  • 经过debug首先是仿真时对立即数的选择出现问题
  • 紧接着发现o2_0的值出现未定义的情况(此时是因为大小写的问题alusrc_b)
  • 草,别相信ai了写verilog是真不行

lab4-3 扩展指令的实现

  • 首先按照stone佬的建议把各个寄存器的值全部输入到vga上面方便查看
  • 但是在下板验证时发现连lui都不对,于是乖乖仿真喵
  • 仿真1:出现了alu_out?怎么会经过alu?
  • 没改immgen啊 废了–immgen和alu都不仅要在接口上进行扩展,还需要在alu内部增加单元,也需要在immgen的控制信号进行修改。
  • 在数据通路中检测无误后,但是下板验证时发现load操作有误,也就是ram存在问题。首先我发现我忘记对ram进行初始化,此后观察warning命令发现端口似乎不匹配?wea貌似不是32位?但是我的顶层模块的定义是32位
  • 经过再次分析发现,竟然是control模块的bug,我没有定义jalr

lab5

  • 流水线操作,当我在最初仿真的时候,我发现output reg貌似不兼容?
  • 第一次仿真,全部都没有被赋值,是哪里有误呢?
  • 开始从pipline_if开始,怎么好像add_32有误
  • 开始仿真ID,发现alu_control输出为Z,数据的写入没问题,说明regs寄存器没有问题(regs之前少了一位,要注意)再次使用指令但是还是发现当前的alu_control输出为Z(未被驱动)草!符号大小写的问题
  • 直接开始进行冒险操作,但是仿真时发现存在Z,怀疑是端口的大小有误
  • 但是现在发现单独对Ex进行仿真也出现了一点错误,ALU有误!后来直接用逻辑算符实现ALU
  • Immsel的端口实现出现错误,ALU实现时的激励端口出现错误
  • 现在基本功能仿真没有太大问题,主要观察冒险时可能出现的错误。(stall的Rs1的addr_id始终为零,把out接成addr导致的!!)
  • 继续找错误,怀疑是寄存器的逻辑有误?(但是有一说一,助教哥哥教的在vivado仿真时把各个模块的值直接接进去简直不要太方便 orzorz)接下来就是思考冒险到底是怎样实现的!!valid_out_Exmem并没有变成0
  • 还有就是流水线的驱动程序,以及最后的wb是否需要reg变量 ,又是激发信号的错误,这和ALU的错误是一样的(不靠谱的github(bushi))
  • 最终进入比特流生成阶段,但是现在突然引脚的led信号显示出现错误。
    set_property -dict { PACKAGE_PIN H17 IOSTANDARD LVCMOS33 } [get_ports { LED_out[0] }];
    set_property -dict { PACKAGE_PIN K15 IOSTANDARD LVCMOS33 } [get_ports { LED_out[1] }];
    set_property -dict { PACKAGE_PIN J13 IOSTANDARD LVCMOS33 } [get_ports { LED_out[2] }];
    set_property -dict { PACKAGE_PIN N14 IOSTANDARD LVCMOS33 } [get_ports { LED_out[3] }];
    set_property -dict { PACKAGE_PIN R18 IOSTANDARD LVCMOS33 } [get_ports { LED_out[4] }];
    set_property -dict { PACKAGE_PIN V17 IOSTANDARD LVCMOS33 } [get_ports { LED_out[5] }];
    set_property -dict { PACKAGE_PIN U17 IOSTANDARD LVCMOS33 } [get_ports { LED_out[6] }];
    set_property -dict { PACKAGE_PIN U16 IOSTANDARD LVCMOS33 } [get_ports { LED_out[7] }];
    set_property -dict { PACKAGE_PIN V16 IOSTANDARD LVCMOS33 } [get_ports { LED_out[8] }];
    set_property -dict { PACKAGE_PIN T15 IOSTANDARD LVCMOS33 } [get_ports { LED_out[9] }];
    set_property -dict { PACKAGE_PIN U14 IOSTANDARD LVCMOS33 } [get_ports { LED_out[10] }];
    set_property -dict { PACKAGE_PIN T16 IOSTANDARD LVCMOS33 } [get_ports { LED_out[11] }];
    set_property -dict { PACKAGE_PIN V15 IOSTANDARD LVCMOS33 } [get_ports { LED_out[12] }];
    set_property -dict { PACKAGE_PIN V14 IOSTANDARD LVCMOS33 } [get_ports { LED_out[13] }];
    set_property -dict { PACKAGE_PIN V12 IOSTANDARD LVCMOS33 } [get_ports { LED_out[14] }];
    set_property -dict { PACKAGE_PIN V11 IOSTANDARD LVCMOS33 } [get_ports { LED_out[15] }];我先删掉这段。
    后来发现是程序的板子型号选择出现错误
  • 今天去实验室上机看了看效果,但是发现控制信号的冲突出现一点问题。仿真发现pc_imm的计算是没有问题的,原来是if的pc_in的接线此时应该接imm_out的pc
本文作者:Lane
本文链接:https://lakerswillwin.github.io/2024/06/15/computerdesign/
版权声明:本文采用 CC BY-NC-SA 3.0 CN 协议进行许可