# ALU设计作业

# 作业内容

★ 参考以下4bit ALU电路图和真值表,对32位ALU组合逻辑电路进行设计,并进行仿真验证。 将程序代码和有关文档打包提交。



|                                                                                                               | 选择控制信号 |    |    |    |    |   | 逻辑                | The Alt |
|---------------------------------------------------------------------------------------------------------------|--------|----|----|----|----|---|-------------------|---------|
|                                                                                                               | S3     | S2 | S1 | S0 | Ci | М | A=a[i]; B=b[i];   | 功能      |
|                                                                                                               | 0      | 0  | 0  | 0  | 1  | 0 | 0                 | 置全0     |
|                                                                                                               | 0      | 0  | 0  | 1  | 1  | 0 | !A & !B           | nor     |
| $S = S_3 \cdot a \cdot b +$                                                                                   | 0      | 0  | 1  | 0  | 1  | 0 | !A & B            | notand  |
| C                                                                                                             | 0      | 0  | 1  | 1  | 1  | 0 | !A                | not A   |
| $S_2 \cdot a \cdot \overline{b} + S_1 \cdot \overline{a} \cdot b + S_0 \cdot \overline{a} \cdot \overline{b}$ | 0      | 1  | 0  | 0  | 1  | 0 | A & !B            | andnot  |
| $S_1 \cdot \overline{a} \cdot b +$                                                                            | 0      | 1  | 0  | 1  | 1  | 0 | !B                | not B   |
| 51 4 5 1                                                                                                      | 0      | 1  | 1  | 0  | 1  | 0 | A&!B !A&B         | xor     |
| $S_0 \cdot \overline{a} \cdot \overline{b}$                                                                   | 0      | 1  | 1  | 1  | 1  | 0 | !A   !B           | nand    |
|                                                                                                               | 1      | 0  | 0  | 0  | 1  | 0 | A & B             | and     |
|                                                                                                               | 1      | 0  | 0  | 1  | 1  | 0 | A&B   !A & !B     | xnor    |
|                                                                                                               | 1      | 0  | 1  | 0  | 1  | 0 | В                 | 传送B     |
|                                                                                                               | 1      | 0  | 1  | 1  | 1  | 0 | A&B   !A&B  !A&!B | notor   |
|                                                                                                               | 1      | 1  | 0  | 0  | 1  | 0 | A                 | 传送A     |
|                                                                                                               | 1      | 1  | 0  | 1  | 1  | 0 | A&B A&!B !A&!B    | or not  |
|                                                                                                               | 1      | 1  | 1  | 0  | 1  | 0 | A&B A&!B !A&B     | or      |
|                                                                                                               | 1      | 1  | 1  | 1  | 1  | 0 | 1                 | 置全1     |
|                                                                                                               | 1      | 0  | 0  | 1  | 0  | 1 | A^B^C             | 加法      |
|                                                                                                               | 0      | 1  | 1  | 0  | 1  | 1 | ( A ~^ B) ^ C     | 减法      |

# 文件夹内容说明

- alu
  - Testbench
    - tb\_32b\_alu\_top.v
    - tb\_4b\_alu\_top.v
  - Vsrc
    - alu\_core.v
    - alu\_core\_unit.v
  - Makefile
  - srclist
  - tb\_alu\_core.fsdb

Makefile 包含 make versim 命令,该命令启动 vcs 对 srclist 的所有 \*.v 文件进行综合和 Tb仿真,用verdi打开Tb生成的 \*.fsdb 波形。

# 32位ALU电路设计

# 电路架构图

本次作业提交的ALU模型如下:







## 设计注意点

## 进位信号C产生电路的设计

因为加减法的进位输出信号C的含义不同,减法的C表达的应该是借位的含义,因此在C输出产生之前增加 MUX,以通过判断是否是减法,而选择不同的两条数据通路,产生正确的输出进位信号。相关代码实现:

```
1 assign C = (M==1)\&\&(S[2]\&S[1])? (opA <opB): ci [n]; //the C of add and substrac
```

### 溢出信号V产生电路的设计

本小节在课程所介绍的加法判断溢出的三种方法基础上,对有符号数加、减法的三种溢出判断方法进行理论分析,并给出了示例代码。

设两个有符号操作数分别为signed A[31:0]、B[31:0],结果为signed S[31:0],符号位分别是A[31]、B[31]、S[31]。

#### 方法1:

利用输入信号和输出信号的符号位是否产生变化来判断。

#### • 加法

加法正溢: {A[31], B[31], S[31]} = 3' b001

加法负溢: {A[31], B[31], S[31]} = 3' b110

```
1 V = (A[31] \sim B[31]) & (A[31]! == S[31])
```

#### • 减法

减法正溢: {A[31], B[31], S[31]} = 3' b011

减法负溢: {A[31], B[31], S[31]} = 3' b100

```
1 V = (A[31]^B[31])&&(A[31]!==S[31])
```

#### 方法2:

第 30 位(从0开始计数)向第 31 位的进位C[31] ,和第 31 位向第 32 位 的进位C[32],C[31]与C[32]不同。

加法和减法同时适用该方法。

 $1 V = ci[31] ^ ci[32]$ 

#### 方法3:

将 ALU 扩展为 33 位计算,最高两位符号位不同。

加法和减法同时适用该方法。

### 对算术运算更深入的理解

经过课下与同学关于电路做加减法的输入数据范围以及输出信号有效性的讨论以及自己思考,先将自己的这方面理解总结如下。

电路支持输入为无符号数的加法,输入为有符号数的加法和减法。由于操作数opB在做减法时首先需要转变为补码表示,要求opB的最高位必须是符号位,因此不存在无符号数减法,或者说无符号数减法本质上和有符号数减法应归为一类。

无符号数加法的进位信号有效,溢出信号无效。有符号数加法的进位信号无效,溢出信号有效。有符号数减法的进位信号和溢出信号同时有效。

它们各自输入数据范围以及输出信号有效性如下表所示:

| 算术操作类型 | 输入操作数数值范围(设ALU位宽为N<br>位)                      | 进位输出信号有效性 | 溢出输出信号有效性 |
|--------|-----------------------------------------------|-----------|-----------|
| 无符号数加法 | $0\sim 2^N-1$                                 | 有效        | 无效        |
| 有符号数加法 | $-2^{\textstyle N} \sim 2^{\textstyle N} - 1$ | 无效        | 有效        |
| 有符号数减法 | $-2^N \sim 2^N - 1$                           | 有效        | 有效        |

# 32位ALU仿真验证

## Testbench测试思路

因为ALU的位宽较大,达到了32位,普通计算机无法完成所有测试用例的完整覆盖率的测试。

因此本次作业采取了如下方法进行了测试:

- 1. 编写了每种情况正确输出的仿真计算公式,实现自动化测试,可以自动检查电路输出信号的错误。
- 2. 32位ALU的测试:每种逻辑运算情况用1个测试用例进行测试,对算术运算情况每种用2个测试用例进行测试。

3. N位ALU的完整覆盖率自动化测试:编写了完整覆盖率测试Tb,ALU的位宽N需要为一个较小的数字,例如4位。已实现TODO:

TODO: 后续可以进一步对电路进行结构化、更多测试用例的测试,让测试达到更高覆盖率。

### 自动化测试代码

对于每种情况,列写出正确(output\_golden)输出结果的公式,通过output和output\_golden的对比,实现自动化检错功能。

```
1 always @(*) begin
                    case ({S,Cin,M})
  2
                                6'b0000 1 0: begin//all bits are 0
  3
                                            DO_golden = 'b0;
  4
  5
                                            \{C_{golden}, V_{golden}, N_{golden}, Z_{golden}\} = \{1'b1, 1'b0, 1'b0, 1'b1\};
  6
                                end
  7
                                6'b0001 1 0: begin//nor
                                            DO_golden = ~opA & ~opB;
  8
  9
                                            \{C_{golden}, V_{golden}, V_{golden}, Z_{golden}\} = \{1'b1,1'b0,1'b0,!(|DO_{golden}, V_{golden}, V_{go
10
                                 end
11 .....
                                 6'b1110 1 0: begin//or
12
                                            DO_golden = opA | opB;
13
                                            \{C_{golden}, V_{golden}, V_{golden}\} = \{1'b1,1'b0,1'b0,!(|DO_{golden}\}\}
14
15
                                 end
                                6'b1111 1 0: begin//all bits are 1
16
                                            DO_golden = 32'hffffffff;
17
                                            \{C_{golden}, V_{golden}, N_{golden}, Z_{golden}\} = \{1'b1, 1'b0, 1'b0, 1'b0\};
18
19
                                 end
                                6'b1001 0 1: begin//add
20
21
                                            {C_golden,DO_golden} = opA + opB;
                                            \{V_{golden}, N_{golden}, Z_{golden}\} = \{(opA[31] \sim opB[31]) \& (opA[31]! == DO_{golden}\} \}
22
23
                                end
                                6'b0110_1_1: begin//sub
24
                                            DO_golden = opA - opB;
25
26
                                            C_golden = opA < opB;</pre>
27
                                            \{V_{golden,N_{golden,Z_{golden}} = \{(opA[31]^{opB[31]})\&\&(opA[31]!==DO_{golden,D_{golden,D_{golden}}\}\}
28
                                 end
                                 default: $display($time,"*** There is one illegal input! -- INPUT: {S,Ci
29
                     endcase
30
31 end
32 initial begin
33
                     #2
                     for (integer i=0; i<19; i=i+1) begin
34
35
                                if ({C_golden,V_golden,N_golden,Z_golden}!=={C,V,N,Z}) begin
36
```

```
37
               error_count = error_count + 1;
               if (error_count <= 10) begin</pre>
38
                   //$display("***ERROR at time = %0d ***", $time);
39
                   $display("ERROR at time = %0d -- INPUT:S = %b, Cin = %b, M = %b;
40
               end
41
               if (error_count == 10) begin
42
                   $display("\n\nError count reached 10, subsequent error messages
43
44
               end
45
           end
46
       end
       if (error_count == 0)
47
           $display("*** Testbench Successfully Completed! There is no error!***");
48
       else begin
49
           $display("\n*****************************;
50
           $display("*** Testbench completed with %0d errors ***",error_count);
51
           $display("***********************************\n\n");
52
       end
53
54
       $finish;
55 end
```

# 仿真结果

自动检错结果

32位ALU的测试

```
IC@IC:~/homework/alu
                                                                            _ 🗆 X
File Edit View Search Terminal Help
*Verdi3* : Create FSDB file 'tb alu top.fsdb'
*Verdi3* : Begin traversing the scopes, layer (0).
*Verdi3* : End of traversing.
                   0 -- INPUT: {S,Cin,M}=xxxxxx;opA=xxxxxxxx,opB=xxxxxxxxx,
OUTPUT: D0=xxxxxxxxx,C=x,V=x,N=x,Z=x
                  10 -- INPUT: {S,Cin,M}=000010;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=000000000, C=1, V=0, N=0, Z=1
                  20 -- INPUT: {S,Cin,M}=000110;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=00ff0000,C=1,V=0,N=0,Z=0
                  30 -- INPUT: {S,Cin,M}=001010;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=ff000000, C=1, V=0, N=0, Z=0
                  40 -- INPUT: {S,Cin,M}=001110;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=ffff0000,C=1,V=0,N=0,Z=0
                  50 -- INPUT: {S,Cin,M}=010010;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=000000ff, C=1, V=0, N=0, Z=0
                  60 -- INPUT: {S,Cin,M}=010110;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=00ff00ff, C=1, V=0, N=0, Z=0
                  70 -- INPUT: {S,Cin,M}=011010;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=ff0000ff, C=1, V=0, N=0, Z=0
                  80 -- INPUT: {S,Cin,M}=011110;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=fffff00ff,C=1,V=0,N=0,Z=0
                  90 -- INPUT: {S,Cin,M}=100010;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=0000ff00,C=1,V=0,N=0,Z=0
                 100 -- INPUT: {S,Cin,M}=100110;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=00ffff00,C=1,V=0,N=0,Z=0
                 110 -- INPUT: {S,Cin,M}=101010;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=ff00ff00,C=1,V=0,N=0,Z=0
                 120 -- INPUT: {S,Cin,M}=101110;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=fffffff00,C=1,V=0,N=0,Z=0
                 130 -- INPUT: {S,Cin,M}=110010;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=0000ffff,C=1,V=0,N=0,Z=0
                 140 -- INPUT: {S,Cin,M}=110110;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=00fffffff.C=1.V=0.N=0.Z=0
                 150 -- INPUT: {S,Cin,M}=111010;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=ff00ffff,C=1,V=0,N=0,Z=0
                 160 -- INPUT: {S,Cin,M}=111110;opA=0000ffff,opB=ff00ff00,
OUTPUT: D0=ffffffffff,C=1,V=0,N=0,Z=0
                 170 -- INPUT: {S,Cin,M}=100101; opA=fffffffff,opB=abcd4321,
OUTPUT: D0=abcd4320,C=1,V=0,N=1,Z=0
                 180 -- INPUT: {S,Cin,M}=011011; opA=fffffffff, opB=0000ffff,
OUTPUT: D0=fffff0000,C=1,V=0,N=1,Z=0
                 190 -- INPUT: {S,Cin,M}=011011;opA=00000001,opB=00000002,
OUTPUT: D0=fffffffff, C=1, V=0, N=1, Z=0
*** Testbench Successfully Completed! There is no error!***
```

Tb成功通过所有的20个测试用例。

#### N位ALU的完整覆盖率自动化测试

\*\*\* Testbench Successfully Completed! There is no error!\*\*\*
在N=4时,电路通过所有测试用例的测试。

## 波形截图

# 32位ALU的测试



# N位ALU的完整覆盖率自动化测试

