# 组合逻辑电路设计

北京科技大学 计算机系 齐悦

#### 目录

Verilog 建模方式

2 组合逻辑电路简介

3 常用组合逻辑模块的Verilog描述

4 组合逻辑电路可综合描述的常见问题

5 以加法器为例讲述关键路径的时延问题

### Verilog 建模方式

- Verilog 对电路建模方式有三类:
  - 结构化描述方式
  - □数据流描述方式
  - □ 行为描述方式

## 全加器



| $C_{in}$ | A | В | S | $C_{out}$ |
|----------|---|---|---|-----------|
| 0        | 0 | 0 | 0 | 0         |
| 0        | 0 | 1 | 1 | 0         |
| 0        | 1 | 0 | 1 | 0         |
| 0        | 1 | 1 | 0 | 1         |
| 1        | 0 | 0 | 1 | 0         |
| 1        | 0 | 1 | 0 | 1         |
| 1        | 1 | 0 | 0 | 1         |
| 1        | 1 | 1 | 1 | 1         |

S=A⊕B⊕Cin

Cout=ACin+BCin+AB

#### 结构化建模方式

通过对电路的层次和组成结构进行描述,并使用线网来连接各器件来描述一个模块的结构。



```
module FA_struct (A, B, Cin,
Sum, out);
input A;
input B;
input Cin;
output Sum;
output Cout;
wire S1, T1, T2, T3;
// -- statements -- //
xor x1 (S1, A, B);
xor x2 (Sum, S1, Cin);
and A1 (T1, A, B);
and A2 (T2, B, Cin);
and A3 (T3, A, Cin);
or O1 (Cout, T1, T2, T3);
endmodule
```

### 数据流建模方式

- 对数据流的具体行为的描述
- 连续赋值语句
- 比较复杂的连续赋值语句,描述一 些较复杂的线网变量的行为



```
module
FA_flow(A,B,Cin,Sum,Cout)
input
        A,B,Cin;
output Sum, Cout;
wire
       S1,T1,T2,T3;
         S1 = A \wedge B;
assign
assign
         Sum = S1 ^ Cin;
assign T1 = A \& B;
assign T2 = B \& Cin;
         T3 = A \& Cin;
assign
assign
         Cout = T1|T2|T3;
endmodule
```

#### 行为描述方式建模

- 只关心电路具有什么样的功能,不关心电路结构(组成和连接)
- 一般,用initial或always块语句描述,可以使用如if else、case、for、while等等,描述复杂的电路。



Cout=ACin+BCin+AB

module FA\_behav1(A, B, Cin, Sum, Cout);
input A,B,Cin;
output Sum,Cout;
reg Sum, Cout;
reg T1,T2,T3;
always@ ( A or B or Cin )
begin



#### 目录

Verilog 建模方式

2 组合逻辑电路简介

3 常用组合逻辑模块的Verilog描述

4 组合逻辑电路可综合描述的常见问题

5 以加法器为例讲述关键路径的时延问题

#### 组合逻辑电路概述

- 数字电路按其完成逻辑功能的不同特点,划分 为组合逻辑电路和时序逻辑电路两大类
- 复杂数字逻辑由组合逻辑电路和时序逻辑电路 构成
- 常用的组合逻辑器件包括编码器、译码器、多 路选择器、比较器、加法器等

组合逻辑电路和时序逻辑电路的主要区别是什么?

#### 组合逻辑电路: 电路无记忆功能

- 从功能上,组合逻辑电路在任一时刻的输出仅和电路当前的输入有关
- 从内部结构上,组合电路都是单纯由逻辑单元组成(不 含存储单元)



### 时序电路: 有存储器

- 还和电路的内部状态有关
- 还有寄存器等存储单元,电路内容有反馈



#### 逻辑关系:

# Verilog如何描述组合逻辑

- module中描述组合逻辑的语句
- 简单逻辑函数

- □ 使用assign描述
- □ 使用always描述

复杂组合逻辑

```
assign q = (al==1?) d : 0;

always @(al or d)

begin

if (al==1) q = d;

else q = 0;

end
```

#### 目录

1 Verilog 建模方式

2 组合逻辑电路简介

- 3 常用组合逻辑模块的Verilog描述
- 4 组合逻辑电路可综合描述的常见问题
- 5 以加法器为例讲述关键路径的时延问题

### 多路选择器

多路选择器是一个 多输入、单输出的组 合逻辑电路。它根据 选择码,从多个输入 数据流中选取一个。 输出到公共输出端。



其典型Verilog描述语句有以下三种:

- assign语句及条件操作符;
- if...else及其嵌套语句;
- case多分支语句.

#### 多路选择器

output f;

■ assign语句及条件运算符

module mux4to1 (w0, w1, w2, w3, S, f); input w0, w1, w2, w3; input [1:0] S;



(a) Graphic symbol (b) Truth table

assign f = S[1] ? (S[0] ? w3 : w2) : (S[0] ? w1 : w0); endmodule

#### ■ 采用 if else 分支语句

```
module mux4to1 (w0, w1, w2, w3, S, f);
    input w0, w1, w2, w3;
    input [1:0] S;
    output reg f;
    always @(*)
        if (S == 2'b00)
          f = w0;
        else if (S == 2'b01)
          f = w1;
        else if (S == 2'b10)
          f = w2;
        else if (S == 2'b11)
          f = w3:
endmodule
```

```
module mux4to1 (W, S, f);
    input [0:3] W;
    input [1:0] S;
    output reg f;
    always @(W, S)
       if (S == 0)
         f = W[0];
       else if (S == 1)
         f = W[1];
       else if (S == 2)
         f = W[2];
       else if (S == 3)
          f = W[3];
endmodule
```

(b) Truth table

#### ■ 采用 case 分支语句

```
module mux4to1 (W, S, f);
    input [0:3] W;
    input [1:0] S;
    output reg f;
    always @(W, S)
        case (S)
            0: f = W[0];
            1: f = W[1];
            2: f = W[2];
            3: f = W[3];
            default: f = 1'b0;
        endcase
endmodule
```

| s <sub>1</sub> | <i>s</i> <sub>0</sub> | f                     |
|----------------|-----------------------|-----------------------|
| 0              | 0                     | w <sub>0</sub>        |
| 0              | 1                     | <i>W</i> <sub>1</sub> |
| 1              | 0                     | w <sub>2</sub>        |
| 1              | 1                     | w <sub>3</sub>        |

(b) Truth table

#### ■ 结构化描述实现16选1选通器

```
module mux16to1 (W, S, f);
    input [0:15] W;
    input [3:0] S;
    output f;
    wire [0:3] M;
    mux4to1 Mux1 (W[0:3], S[1:0], M[0]);
    mux4to1 Mux2 (W[4:7], S[1:0], M[1]);
    mux4to1 Mux3 (W[8:11], S[1:0], M[2]);
    mux4to1 Mux4 (W[12:15], S[1:0], M[3]);
    mux4to1 Mux5 (M[0:3], S[3:2], f);
endmodule
```



# 多路选择器的Verilog编码风格

- 二选一建议使用if else
- 四选一以上建议使用case
- 多选一超过8输入时,建议拆分成多个小规模选通器

### 译码器/编码器

- 将一组形式的二进制数据转化为另一种形式的 二进制数据
- 编码器用来将多位输入数据流编码成更短的码流,使得编码器的输出端口减少,具有2<sup>n</sup>(或小于)输入位的编码器可提供n位编码输出线。
- 译码器将先前编码过的数据解码; 是编码器的 逆过程。N位的输入可代表2°种不同的信息

### 描述译码器/编码器的语句

- 通常,采用下列语句描述:
- If...else及其嵌套结构
- case/casex/casez结构
- for循环结构(初学者不建议)

#### 2-4译码器

```
module dec2to4 (W, En, Y);
    input [1:0] W;
    input En;
    output reg [0:3] Y;
    always @(W, En)
       case ({En, W})
          3'b100: Y = 4'b1000;
          3'b101: Y = 4'b0100;
          3'b110: Y = 4'b0010;
          3'b111: Y = 4'b0001;
          default: Y = 4'b0000;
       endcase
endmodule
```

| En | $w_1$ | $w_0$ | $y_0$ | $y_1$ | $y_2$ | $y_3$ |
|----|-------|-------|-------|-------|-------|-------|
| 1  | 0     | 0     | 1     | 0     | 0     | 0     |
| 1  | 0     | 1     | 0     | 1     | 0     | 0     |
| 1  | 1     | 0     | 0     | 0     | 1     | 0     |
| 1  | 1     | 1     | 0     | 0     | 0     | 1     |
| 0  | X     | х     | 0     | 0     | 0     | 0     |

(a) Truth table



(b) Graphical symbol

#### ■ 2-4译码器的另一种实现方式

```
module dec2to4 (W, En, Y);
    input [1:0] W;
    input En;
    output reg [0:3] Y;
    always @(W, En)
    begin
      if (En == 0)
        Y = 4'b0000;
      else
        case (W)
          0: Y = 4'b1000;
          1: Y = 4'b0100;
          2: Y = 4'b0010;
          default: Y = 4'b0001;
        endcase
    end
endmodule
```

| En | $w_1$ | $w_0$ | $y_0$ | $y_1$ | $y_2$ | $y_3$ |
|----|-------|-------|-------|-------|-------|-------|
| 1  | 0     | 0     | 1     | 0     | 0     | 0     |
| 1  | 0     | 1     | 0     | 1     | 0     | 0     |
| 1  | 1     | 0     | 0     | 0     | 1     | 0     |
| 1  | 1     | 1     | 0     | 0     | 0     | 1     |
| 0  | X     | х     | 0     | 0     | 0     | 0     |

(a) Truth table



(b) Graphical symbol

#### ■ 结构化描述实现4-16译码器



#### 代码转化器

|     |   | $w_3$ | $w_2$ | $w_1$ | $w_0$ | а | b | c | d | е | f | g |
|-----|---|-------|-------|-------|-------|---|---|---|---|---|---|---|
|     |   | 0     | 0     | 0     | 0     | 1 | 1 | 1 | 1 | 1 | 1 | 0 |
|     |   | 0     | 0     | 0     | 1     | 0 | 1 | 1 | 0 | 0 | 0 | 0 |
|     |   | 0     | 0     | 1     | 0     | 1 | 1 | 0 | 1 | 1 | 0 | 1 |
|     |   | 0     | 0     | 1     | 1     | 1 | 1 | 1 | 1 | 0 | 0 | 1 |
| а   |   | 0     | 1     | 0     | 0     | 0 | 1 | 1 | 0 | 0 | 1 | 1 |
| f   | b | 0     | 1     | 0     | 1     | 1 | 0 | 1 | 1 | 0 | 1 | 1 |
| -   |   | 0     | 1     | 1     | 0     | 1 | 0 | 1 | 1 | 1 | 1 | 1 |
| e g | c | 0     | 1     | 1     | 1     | 1 | 1 | 1 | 0 | 0 | 0 | 0 |
|     | Ŭ | 1     | 0     | 0     | 0     | 1 | 1 | 1 | 1 | 1 | 1 | 1 |
| d   | d | 1     | 0     | 0     | 1     | 1 | 1 | 1 | 1 | 0 | 1 | 1 |
|     |   | 1     | 0     | 1     | 0     | 1 | 1 | 1 | 0 | 1 | 1 | 1 |
|     |   | 1     | 0     | 1     | 1     | 0 | 0 | 1 | 1 | 1 | 1 | 1 |
|     |   | 1     | 1     | 0     | 0     | 1 | 0 | 0 | 1 | 1 | 1 | 0 |
|     |   | 1     | 1     | 0     | 1     | 0 | 1 | 1 | 1 | 1 | 0 | 1 |
|     |   | 1     | 1     | 1     | 0     | 1 | 0 | 0 | 1 | 1 | 1 | 1 |
|     |   | 1     | 1     | 1     | 1     | 1 | 0 | 0 | 0 | 1 | 1 | 1 |

```
module seg7 (hex, leds);
  input [3:0] hex;
  output reg [1:7] leds;
  always @(hex)
     case (hex) //abcdefg
         0: leds = 7'b11111110;
         1: leds = 7'b0110000;
         2: leds = 7'b1101101;
         3: leds = 7'b1111001;
         4: leds = 7'b0110011;
         5: leds = 7'b1011011;
         6: leds = 7'b10111111;
         7: leds = 7'b1110000;
         8: leds = 7'b11111111;
         9: leds = 7'b1111011;
        10: leds = 7'b1110111;
        11: leds = 7'b00111111;
        12: leds = 7'b1001110;
        13: leds = 7'b0111101;
        14: leds = 7'b1001111;
        15: leds = 7'b1000111;
     endcase
```

endmodule

#### 优先级编码器

| $w_3$ | $w_2$ | <i>w</i> <sub>1</sub> | $w_0$ | <i>y</i> <sub>1</sub> | <i>y</i> <sub>0</sub> |
|-------|-------|-----------------------|-------|-----------------------|-----------------------|
| 0     | 0     | 0                     | 1     | 0                     | 0                     |
| 0     | 0     | 1                     | 0     | 0                     | 1                     |
| 0     | 1     | 0                     | 0     | 1                     | 0                     |
| 1     | 0     | 0                     | 0     | 1                     | 1                     |

| $w_3$ | <i>w</i> <sub>2</sub> | <i>w</i> <sub>1</sub> | $w_0$ | <i>y</i> <sub>1</sub> | <i>y</i> <sub>0</sub> | Z |
|-------|-----------------------|-----------------------|-------|-----------------------|-----------------------|---|
| 0     | 0                     | 0                     | 0     | d                     | d                     | 0 |
| 0     | 0                     | 0                     | 1     | 0                     | 0                     | 1 |
| 0     | 0                     | 1                     | Χ     | 0                     | 1                     | 1 |
| 0     | 1                     | Χ                     | Χ     | 1                     | 0                     | 1 |
| 1     | Χ                     | X                     | Χ     | 1                     | 1                     | 1 |

```
always @ (W)
begin

if (W[3]) Y=2'b11;
else if (W[2]) Y=2'b10;
else if (W[1]) Y=2'b01;
else Y=2'b00;
end
```

# 编码/译码电路的Verilog编码风格

- 通常使用case语句实现编解码模块
- 优先级编码器可以采用if语句实现
- 有时也可以用for循环结构实现,初学者不建议!!!

#### 设计举例: 简单ALU

| Operation | Inputs $s_2 s_1 s_0$ | Outputs<br>F    |
|-----------|----------------------|-----------------|
| Clear     | 000                  | 0000            |
| B-A       | 001                  | B-A             |
| A-B       | 010                  | A - B           |
| ADD       | 0 1 1                | A + B           |
| XOR       | 100                  | $A \times OR B$ |
| OR        | 101                  | A  OR  B        |
| AND       | 110                  | A  AND  B       |
| Preset    | 1 1 1                | 1111            |

#### 确定输入信号和输出信号

指令译码、操作数A、B;运算结果确定输入和输出的逻辑状态关系用Verilog正确描述电路功能

```
module alu(s, A, B, F); // 74381 ALU
    input [2:0] S;
    input [3:0] A, B;
    output reg [3:0]/
    always @(S, A, B)
                        子电路模块
       case (S)
         0: F = 4'b0000'
         1: F = B - A;
         2: F = A - B
         3: F = A + B;
         4: F = A \wedge B;
         5: F = A | B;
         6: F = A \& B;
         default: F = 4'b1111;
       endcase
endmodule
```

#### 设计举例:裁判表决器

■ 设计一个举重裁判表决器,举重比赛有3个裁判,1个主裁判和2个副裁判,只有当两个或两个以上裁判判明成功,并且其中有主裁判时,表决器判决举重成功

#### 确定输入信号和输出信号

input a,b,c; output y;

确定输入和输出的逻辑状态关系

用Verilog正确描述电路功能

| 车 | 俞入 | 输出 |   |
|---|----|----|---|
| A | B  | C  | Y |
| 0 | 0  | 0  | 0 |
| 0 | 0  | 1  | 0 |
| 0 | 1  | 0  | 0 |
| 0 | 1  | 1  | 0 |
| 1 | 0  | 0  | 0 |
| 1 | 0  | 1  | 1 |
| 1 | 1  | 0  | 1 |
| 1 | 1  | 1  | 1 |

assign y=a&b+a&c;

#### 目录

1 Verilog 建模方式

2 组合逻辑电路简介

3 常用组合逻辑模块的Verilog描述

4 组合逻辑电路可综合描述的常见问题

5 以加法器为例讲述关键路径的时延问题

#### 阻塞赋值实现组合逻辑电路

- 在描述组合逻辑电路的always块里用阻塞赋值
- 组合逻辑里的信号是单向按顺序经过一系列处理的 过程
- 综合不支持延迟语句#

# always块的敏感表

- 实现组合逻辑电路的always块敏感表必须写全,否则仿 真结果和综合结果会不一致
- 所有赋值号右边出现的信号都要放到敏感表里

```
always @(a \text{ or } b \text{ or } c)

f = a\&\&b||c;
```

```
always @(a or b)
f = a&&b||c ;
```

```
always @(*)
f = a&&b||c ;
```



### 避免Latch的产生

实现组合逻辑电路的always块中if和case语句的分支必须写全。

```
always @(bcd) begin
 case (bcd)
    4' d0 : out = 3' b001;
    4' d1 : out = 3' b010;
    4' d2 : out = 3' b011;
  endcase
end
always @(a or b or gate)
begin
  if (gate)
     out = a \mid b;
end
```



### 避免Latch的产生

```
always @(bcd) begin
case (bcd)

2' b00 : begin out1 = 3' b001; out2=2' b00; end
2' b01 : begin out1 = 3' b011; out2=2' b10; end
2' b10 : out1 = 3' b101;
2' b11 : begin out1 = 3' b111; out2=2' b11; end
endcase
end
```

没有改变out2 导致产生Latch

#### 目录

1 Verilog 建模方式

2 组合逻辑电路简介

3 常用组合逻辑模块的Verilog描述

4 组合逻辑电路可综合描述的常见问题

5 以加法器为例讲述关键路径的时延问题

### 组合逻辑应用一加法器设计

加法器常常是限制速度的部件。加法器的优化可在逻辑级和电路级进行

全加器 (Full-Adder)



| A | В | $C_{\boldsymbol{i}}$ | S | $C_{o}$ | Carry<br>status |
|---|---|----------------------|---|---------|-----------------|
| 0 | 0 | 0                    | 0 | 0       | delete          |
| 0 | 0 | 1                    | 1 | 0       | delete          |
| 0 | 1 | 0                    | 1 | 0       | propagate       |
| 0 | 1 | 1                    | 0 | 1       | propagate       |
| 1 | 0 | 0                    | 1 | 0       | propagate       |
| 1 | 0 | 1                    | 0 | 1       | propagate       |
| 1 | 1 | 0                    | 0 | 1       | generate        |
| 1 | 1 | 1                    | 1 | 1       | generate        |

### 设计示例 用一位全加器组成四位全加器

```
module FullAdder (A, B, Ci, S, Co);
input A, B, Ci;
output S, Co;

assign S = A ^ B ^ Ci; 本位和
assign Co = (A & B) | (A & Ci) | (B & Ci); 世位
endmodule
```

### 设计示例(续) 用一位全加器组成四位全加器

```
module ADDER4BIT (Ai, Bi, S, OVF);
  input [3:0] Ai, Bi;
  output [3:0] S;
  wire[2:0] CY;
  output OVF:
  FullAdder U0 (Ai[0], Bi[0], 0, S [0], CY[0]);
  FullAdder U1 (Ai[1], Bi[1], CY[0], S[1], CY[1]);
  FullAdder U2 (Ai[2], Bi[2], CY[1], S[2], CY[2]);
  FullAdder U3 (Ai[3], Bi[3], CY[2], S[3], OVF);
endmodule
```

### 简单加法器:二进制加法运算



$$S = A \oplus B \oplus C_{i}$$

$$= A\overline{B}\overline{C}_{i} + \overline{A}B\overline{C}_{i} + \overline{A}\overline{B}C_{i} + ABC_{i}$$

$$C_{0} = AB + BC_{i} + AC_{i}$$

### 逐位进位(Ripple-Carry)加法器

(1) 结构:由N个一位加法器串联而成,第i级的Carry-out用来产生第i+1级的SUM和Carry

(2) 特点:结构直观简单,但因高位运算必须等低位进位来到后才能进行,故运行速度慢



最坏情况下关键路径的延时:

$$t_{adder} = (N-1)t_{carry} + t_{sum}$$

可见运算的延迟是由于进位的延迟。进位的解决是核心问题。

### 进位选择(Carry-Select)加法器

两个逐位进位加法器的进位链构成:

一个最低位进位Cin=0,另一个最低位进位Cin=1。这两个加法器的低位进位Cin=1。这两个加法器的进位链分别计算出对应不同Cin值的两个"进位输出"。

一个MUX选择这两个"进位输出"中的一个作为最终的"进位输出",这个MUX的控制信号是前级的进位输出Carry-out



## 进位选择(Carry-Select)加法器



#### 进位选择加法器的关键路径与求和时间

线性进位选择加法器的总进位传播时间仍与位数N成正比,但比逐位进位加法器快。硬件开销为一个额外的进位路径和一个MUX,大约比一个逐位进位加法器多30%



#### 平方根进位选择加法器的求和时间

考虑到前级进位输出要经过一个MUX才到达本级的进位输入,因此在两条信号路径之间相差一个延时时间,故本级的位数可以比前一级多一位



#### 进一步优化方法(平方根进位选择加法器)

平方根进位选择加法器的延时:

假设N位的加法器含有P个级,且第一级加是M位,后续级逐级增加一位,于是:

$$N = M + (M+1) + (M+2) + \dots + (M+P-1)$$
$$= MP + \frac{P(P-1)}{2} = \frac{P^2}{2} + P(M - \frac{1}{2})$$

若 M<<N, (如M=2, N=64), 则

$$N = \frac{P^2}{2} + P(M - \frac{1}{2}) \approx \frac{P^2}{2}$$

于是

$$P = \sqrt{2N}$$

此时延时正比于 $\sqrt{2N}$ (亚线性关系),而不是N(线性关系)当N很大时,延时几乎变为常数。

# 加法器延时比较



#### 进位产生、进位传播信号

为了利于具体实现,常常定义一些中间信号(注意它们与 $C_{in}$ 无关)

进位产生(Generate)信号: G = AB

进位传播(Propagate)信号: P=A⊕B

这样可以建立起 S和  $C_o$ 与 G、P之间的关系:



### 超前进位的基本原理

- 每一位的进位输出只与最初 的进位输入有关
- 各位进位信号同时形成
- 但与门的扇入=**n+1**,或门 的扇入=**n+1**
- 适合n较小的时候(n<=4)

$$C1 = G1 + P1C0$$
  
 $C2 = G2 + P2C1$   
 $= G2 + P2G1 + P2P1C0$ 

$$Cn = Gn + PnCn-1$$
  
=  $Gn + PnGn-1 + ... + PnPn-1...P2P1C0$ 



# 4位超前进位加法器

■ 进位产生逻辑:

 $C[1]=G0+P0\cdot C[0]$ 

C[2]=G1+P1·G0+P1·P0·C[0]

 $C[3]=G2+P2\cdot G1-P2\cdot P1\cdot G0+P2\cdot P1\cdot P0\cdot C[0]$ 

C[4]=G3+P3•G2+P3·P2·G1+P3·P2·P1·G0+ P3·P2·P1·P0·C[0]

■ 各位的进位相互独立,进一步减小了延迟

思考: 16位超前进位加法器如何实现