# Verilog语言(自学材料)

ga 北京航空航天大学

82

# 第二部分: 组合逻辑

- 一. 逻辑门电路
- 二. 布尔代数
- 三. Verilog HDL介绍
  - 1. Verilog HDL概述
  - 2. Verilog HDL的词法
  - 3. Verilog HDL常用语句
  - 4. 不同抽象级别的Verilog HDL模型
- 四. 基本组合逻辑部件设计
  - 1. 运算单元电路
  - 2. 编码器/译码器
  - 3. 多路选择器

3. 多附处针

gg 北京航空航天大學

### 3.1 Verilog HDL概述 —— 硬件描述语言

❖硬件描述语言 (Hardware Description Language) 是一种用形式化方法 (即文本形式)来描述和设计数字电路和数字系统的高级模块化语言。 它是设计人员和EDA工具之间的一个桥梁,主要用于编写设计文件,在 EDA工具中建立电路模型;也用来编写测试文件进行仿真。

### 用HDL描 述设计

EDA工具 经A 公首

综合、仿真 → 目标文件

→ 编程下载

- ❖ HDL发展至今已有近三十年的历史,到20世纪80年代,已出现了数十种 硬件描述语言。80年代后期, HDL向着标准化、集成化的方向发展, 最终VHDL、Verilog HDL先后成为IEEE标准。
  - ➤ VHDL: VHSIC Hardware Description Language (VHSIC—— Very High Speed Integrated Circuits), 甚高速集成电路的硬件描述语言,来源于美国军方,1987年成为IEEE标准。目前标准化程度最高的一种硬件描述语言。
  - ➤ Verilog HDL: Verilog Hardware Description Language,则是目前应用最广泛的一种硬件描述语言。

**66** 北京航空航天大學

84

# 3.1 Verilog HDL概述 —— Verilog HDL的功能和特点

- ❖ Verilog HDL可用于数字电路的建模、仿真验证、时序分析、逻辑综合
- ❖ Verilog HDL抽象级别:系统级,算法级,RTL级,门级,开关级
- ❖ Verilog HDL具有行为描述和结构描述功能。
  - ▶行为描述包括:系统级、算法级和RTL级3种抽象级别:
  - >结构描述包括:门级和开关级2种抽象级别。
- ❖ Verilog HDL语法结构上的主要特点
  - ▶形式化地表示电路的行为和结构:
  - ▶借用C语言的结构和语句;
  - ▶可在多个层次上对所设计的系统加以描述,语言对设计规模不加任何 限制:
  - ▶具有混合建模能力:一个设计中的各子模块可用不同级别的抽象模型 来描述:
  - ▶基本逻辑门、开关级结构模型均内置于Verilog HDL语言库中,可直接调用,如:与门and、或门or、三态门bufif1(或bufif0)等

**《** 北京航空航天大學

# 3.1 Verilog HDL概述 —— Verilog HDL与C语言比较

# ❖ 语句及函数的比较

| 语句及函数  | C语言            | Verilog HDL    |
|--------|----------------|----------------|
| 函数     | 无参函数,有参函数      | function块语句    |
| 赋值语句   | 赋值变量 = 表达式;    | 阻塞赋值=,非阻塞赋值<=  |
| 条件语句   | if-else        | if-else        |
| 条件语句   | switch         | case           |
| 循环语句   | for            | for            |
| 循环语句   | while          | while          |
| 中止语句   | break          | break          |
| 宏定义语句  | define(以符号#开头) | define(以符号'开头) |
| 格式输出函数 | printf         | printf         |

g. 北京航空航天大学

86

# 3.1 Verilog HDL概述 —— Verilog HDL与C语言比较

# **❖** 运算符的比较

| C语言 | Verilog HDL | 功能   | C语言 | Verilog HDL | 功能         |
|-----|-------------|------|-----|-------------|------------|
| +   | +           | 加    | <=  | <=          | 小于等于       |
| _   | _           | 减    | ==  | ==          | 等于         |
| *   | *           | 乘    | !=  | !=          | 不等于        |
| 1   | 1           | 除    | ~   | ~           | 按位取反       |
| %   | %           | 取模   | &   | &           | 按位与        |
| !   | !           | 逻辑非  | 1   |             | 按位或        |
| &&  | &&          | 逻辑与  | ٨   | ^           | 按位异或       |
| II  |             | 逻辑或  | <<  | <<          | 左移         |
| >   | >           | 大于   | >>  | >>          | 右移         |
| <   | <           | 小于   | ?:  | ?:          | 等同于if-else |
| >=  | >=          | 大于等于 |     |             |            |

Verilog HDL与C语言的运算符几乎完全相同!

**企** 北京航空航天大學

```
3.1 Verilog HDL概述 —— Verilog HDL模块的结构
❖ Verilog的基本设计单元是"模块(module)",可实现一个特定功
❖一个"与门"、"加法器"、ALU都可以是一个模块
❖ Verilog模块的结构由在module和endmodule关键词之间的4个主要
 部分组成:
                module block1(a,b,c,d);
     端口定义
                   input a, b, c; /* I/O变量缺省为wire变量*/
     I/O说明
                   output d;
                   wire x;
     信号类型声明
                   assign d = a \mid x; //组合逻辑功能描述
     功能描述
                   assign x = (b \& \sim c);
                 endmodule
  结构描述
```

**建** 北京航空航天大學

```
3.1 Verilog HDL概述 —— Verilog HDL模块的结构
 一、简单的Verilog HDL例子
                  模块名(文件名)
  【例1】8位全加器
                                         端口定义
      module adder8 (cout,sum,a,b,cin);
                           // 输出端口声明
        output cout;
                                         I/O说明
        output [7:0] sum;
                          // 输入端口声明
        input [7:0] a,b;
        input cin;
                                         功能描述
        assign {cout,sum}=a+b+cin;
      endmodule
     ❖整个程序嵌套在module和endmodule声明语句中。
     ❖每条语句相对module和endmodule最好缩进2格或4格!
     ❖// ...... 表示注释部分,一般只占据一行。对编译不起作用!
     ❖□: 数组/总线定义
                                 单行注释符
     ❖{}: 位拼接运算符
公 北京航空航天大学
```

### 【例2】8位与门

module and8 (input [7:0] a, output y); assign y &a; // &a is much easier to write than // assign y= a[7] & a[6] & a[5] & a[4] & a[3] & a[2] & a[1] & a[0]; endmodule

### ❖ &:缩位操作符



2 北京航空航天大学

an

# 3.1 Verilog HDL概述 —— Verilog HDL模块的结构

#### 【例3】 多路选择器

module mux2 (input [3:0] d0, d1, input s, output [3:0] y); assign y = s? d1 : d0; endmodule

### ❖?: 条件运算符



**(4)** 北京航空航天大學

#### 【例4】 多路选择器

```
module mux4 (input [3:0] d0, d1, d2, d3, input [1:0] s, output [3:0] y); wire [3:0] low, high; mux2 lowmux (d0, d1, s[0], low); mux2 highmux (d2, d3, s[0], high); mux2 finalmux (low, high, s[1], y); endmodule
```

 $module\ mux2\ (input\ [3:0]\ d0,\ d1,\ input\ s,\ output\ [3:0]\ y);$ 

assign y = s ? d1 : d0;

endmodule



2 北京航空航天大學

92

# 3.1 Verilog HDL概述 —— Verilog HDL模块的结构

# 1、模块端口定义

- ▶ 模块端口定义用来声明设计电路模块的输入输出端口,端口定义格式: module 模块名(端口1,端口2,端口3,...);
- 在端口定义的圆括弧中,是设计电路模块与外界联系的全部输入输出端口信号或引脚,它是设计实体对外的一个通信界面,是外界可以看到的部分(不包含电源和接地端),多个端口名之间用","分隔。

### 2、I/O说明

▶ 模块的I/O说明用来声明模块端口定义中各端口数据流动方向,包括输入(input)、输出(output)和双向(inout)。

```
input 端口1,端口2,端口3,...;
output 端口1,端口2,端口3,...;
inout 端口1,端口2,端口3,...;
```

端口定义、I/O说明和程序语句中的标点符号及 圆括弧均要求用半角符号书写!

**②** 北京航空航天大学

### 3、信号类型声明

- ▶ 信号类型声明用来说明设计电路的功能描述中,所用的信号的数据 类型以及函数声明。
- ➤ 信号的数据类型主要有 连线 (wire)、寄存器 (reg)、整型 (integer)、实型 (real) 和时间 (time) 等类型。

### 4、功能描述

- ➤ 功能描述是Verilog HDL程序设计中最主要的部分,用来描述设计模块的内部结构和模块端口间的逻辑关系,在电路上相当于器件的内部电路结构。
- ➤ 功能描述可以用assign语句、元件例化(instantiate)、always 块语句等方法来实现, 通常把确定这些设计模块描述的方法称为 建模。

**企** 北京航空航天大學

Q/

# 3.1 Verilog HDL概述 —— Verilog HDL模块的结构

- ❖在Verilog 模块中有3种方法可以描述电路的逻辑功能:
  - ① 用assign 语句

数据流描述

assign  $x = (b \& \sim c)$ ;

常用于描述 组合逻辑

②用元件例化(instantiate)

and myand3(f,a,b,c);

结构描述

#### 门元件关键字

#### 例化元件名

- → 注1: 元件例化即是调用Verilog HDL提供的元件 或由某子模块定义的实例元件;
  - ❖ 注2: 元件例化包括门元件例化和模块元件例化:
  - ❖ 注3: 例化元件名也可以省略!若不省略,则每个 实例元件的名字必须唯一!以避免与其它调用该 元件的实例相混淆。

**企** 北京航空航天太

③ 用 "always" 块语句 <mark>行为描述</mark>

always @(posedge clk) // 每当时钟上升沿到来时执行一遍块内语句begin

if(load) out = data; // 同步预置数据 else out = data + 1 + cin; // 加1计数 end

- ◆注1: "always"块语句常用于描述时序逻辑,也可描述组合逻辑。
  - ❖注2: "always"块可用多种手段来表达逻辑关系,如用if-else语句或case语句。
  - ❖注3: "always"块语句与assign语句是并发执行的, assign语句一定要放在"always"块语句之外!

**《** 北京航空航天大學

96

# 3.1 Verilog HDL概述 —— Verilog HDL模块的结构

❖ Verilog HDL程序的三种描述方式

结构 (Structur al) 描述

- ▶ 对设计电路的结构进行描述,即描述设计电路使用的元件及这些元件之间的连接关系
- ▶ 属于低层次的描述方法,包括门级和开关级2种抽象级别

行为 (Behaviou ral) 描述

- ▶ 对设计电路的逻辑功能的描述,并不用关心设计电路 使用哪些元件以及这些元件之间的连接关系
- ▶属于高层次的描述方法,包括系统级、算法级和寄存器传输级等3种抽象级别

数据流 (Data Flow) 描述

> 采用持续赋值语句

gg 北京航空航天大学

❖Verilog HDL模块的模板(仅考虑用于逻辑综合的程序)

```
module <项层模块名> (< 输入输出端口列表>);
output 输出端口列表;
input 输入端口列表;
//(1)使用assign语句定义逻辑功能
wire <结果信号名>;
assign <结果信号名>=表达式;
//(2)使用always块定义逻辑功能
always @(<敏感信号表达式>)
begin
//过程赋值语句
//if语句
//case语句
//while,repeat,for循环语句
//task,function调用
end
```

端口定义

I/O说明

信号类型声明

功能描述

**3** 北京航空航天大学

Q.

# 3.1 Verilog HDL概述 —— Verilog HDL模块的结构

❖ Verilog HDL模块的模板(续)

```
// (3) 元件例化
```

例化元件名也可以省略!

gg 北京航空航天大学

# 第二部分:组合逻辑

- 一. 逻辑门电路
- 二. 布尔代数
- 三. Verilog HDL介绍
  - 1. Verilog HDL概述
  - 2. Verilog HDL的词法
  - 3. Verilog HDL常用语句
  - 4. 不同抽象级别的Verilog HDL模型
- 四. 基本组合逻辑部件设计
  - 1. 运算单元电路
  - 2. 编码器/译码器
  - 3. 多路选择器

2 北京航空航天大學

100

### 3.2 Verilog HDL的词法 —— 空白符和注释

- ❖ Verilog HDL源程序由空白符分隔的词法符号流组成。
- ❖ 词法符号包括空白符、注释、操作符(运算符)、常数、 字符串、标识符及关键字。

### 一、空白符和注释

- ❖ Verilog HDL的空白符包括空格、Tab、换行和换页符号。空白符如果不是出现在字符串中,编译源程序时将被忽略。
- ❖注释用来帮助读者理解程序,编译源程序时将被忽略。注释 分为行注释和块注释两种方式。
  - ▶ 行注释用符号// (两个斜杠)开始,注释到本行结束。
  - ▶<mark>块注释</mark>用/\*开始,用\*/结束。块注释可以跨越多行,但它们不能 嵌套。

### 3.2 Verilog HDL的词法 —— 常数

### 二、常数

- ❖ 常数包括整数、x(未知)和z(高阻)值、负数、实数
- ❖ 整数的4种进制表示形式:
  - ▶二进制整数(b或B):
  - ▶十进制整数 (d或D):
  - ▶十六进制整数(h或H);
  - ▶八进制整数(o或O)。



| 整数的3种表达方式       | 说 明                       | 举例                |
|-----------------|---------------------------|-------------------|
| <位宽>'<进制符号><数字> | 完整的表达方式                   | 8'b11000101或8'hc5 |
| <进制符号><数字>      | 缺省位宽,则位宽由机器<br>系统决定,至少32位 | hc5               |
| <数字>            | 缺省进制为十进制,位宽<br>默认为32位     | 197               |

- ❖这里位宽指对应二进制数的宽度。
- ❖整数型常量是可以综合的,而实数型和 字符串型常量都是不可综合的

**3.** 北京航空航天大學

102

### 3.2 Verilog HDL的词法 —— 常数 —— x和z值

### ❖x和z值

- ▶x表示不定值,z表示高阻值;
- >x和z代表的二进制位数取决于所用的进制 ——
  - x(或z)在二进制中代表1位的x(或z)
  - 在八进制中代表3位的x(或z)
  - 在十六进制中代表4位的x(或z)
- "?"是z的另一种表示符号,建议在case语句中使用?表示高阻态z 【例2.17】casez (select)

4'b???1: out = a;

4'b??1?: out = b;

4'b?1??: out = c;

4'b1???: out = d;

endcase

#### ❖ 负数

- ➤ 在位宽前加一个负号,即表示负数,负数通常表示为该负数的二进制补码
- > 如: -8'd5 //-5的补码, = 8'b11111011

**3.** 北京航空航天大学

### 3.2 Verilog HDL的词法 —— 常数 —— 实数 (Real)

- ❖实数的两种表示法
  - > 十进制表示法

**2.0, 5.678, 0.1** //合法

**2**. //非法,小数点两侧都必须有数字

科学计数法

43\_5.1e2 // 等于435.1×10<sup>2</sup>=43510

• 5E-4 //等于5×10<sup>-4</sup>=0.0005, e与E相同

❖实数通过四舍五入被转换为最相近的整数

【例】42.446, 42.45 //若转换为整数都是42

92.5, 92.699 //若转换为整数都是93

-15.62, -25.26 //若转换为整数分别为-16, -25

下划线""可随意用在整数或实数的数字中间,以提高可读性; 但数字的第1个字符不能是下划线,也不能用在位宽和进制处。

**对 北京航空航天大學** 

### 3.2 Verilog HDL的词法 —— 字符串

# 三、字符串

- ❖字符串是用双引号括起来的可打印字符序列,不能多行书写。
- ❖作用:在仿真时显示一些相关信息,或者指定显示的格式 >例: "INTERNAL ERROR", "this is an example for Verilog HDL"
- ❖字符串能够用在系统任务(如\$display、\$monitor)中作为变量, 字符串的值可像数值一样存储在寄存器中,也可以像对数字一样 对字符串进行赋值、比较和拼接操作

【例】\$display(\$time,,,"a=%h b=%h c=%h",a,b,c); // 显示当前仿真时间, 空3格后显示a=xx b=xx c=xx

宽度为字符串中字符 的个数乘以8。

❖ 字符串属于reg型变量, 【例】reg[8\*12:1] stringvar; initial begin stringvar = "Hello world!"; end

**公 北京航空航天大学** 

### 3.2 Verilog HDL的词法 —— 标识符

#### 四、标识符

- ❖任何用Verilog HDL语言描述的对象都通过其名字来识别,这个名字被称为标识符。标识符可由字母、数字、下划线和\$符号构成。
- ❖如源文件名、模块名、端口名、变量名、常量名、实例名等。
- ❖ 定义标识符时应遵循如下规则
  - ① 首字符必须是字母或下划线,不能是数字或\$符号!
  - ② 字符数不能多于1024个。
  - ③ 大小写字母是不同的。
  - ④ 不要与关键字同名!
  - ❖ 合法的名字:
    - > A 99 Z
    - Reset
    - > 54MHz Clock\$
    - Module

#### ❖ 不合法的名字:

- > 123a
- > \$data
- module
- > 7seg.v
- > out\*

//不允许包含字符\*

**力** 北京航空航天大學

106

### 3.2 Verilog HDL的词法 —— 关键字

### 五、关键字

- ❖关键字(保留字)——Verilog HDL事先定义好的确认符, 用来组织语言结构;或者用于定义Verilog HDL提供的门元件(如and, not, or, buf)。
- ❖每个关键字全部用小写字母定义!
  - 如: always, assign, begin, case, casex, else, end, endmodule, for, function, if, input, module, output, repeat, table, time, while, wire
- ❖ Verilog -1995的关键字有97个(见教材表), Verilog -2001 增加了5个共102个。

### 3.2 Verilog HDL的词法 —— 运算符及表达式

### 六、运算符及表达式

❖ 运算符也称为操作符,是Verilog HDL预定义的函数符号, 这些函数对被操作的对象(即操作数)进行规定的运算, 得到一个结果。

#### ❖运算符按功能分为9类:

- > 算术运算符
- > 逻辑运算符
- > 关系运算符
- > 等值运算符
- > 缩减运算符
- > 条件运算符
- > 位运算符
- > 移位运算符
- > 位拼接运算符

### ❖ 运算符按操作数的个数分为3类:

- ▶ 単目运算符——带一个操作数 逻辑非!,按位取反~,缩减 运算符,移位运算符
- ▶ 双目运算符——带两个操作数 算术、关系、等值运算符,逻 辑运算符(除逻辑非外)、位 运算符(除按位取反外)
- ➢ 三目运算符──带三个操作数 条件运算符

**① 北京航空航天大學** 

108

# 3.2 Verilog HDL的词法 — 运算符及表达式 — 算术运算符

❖双目运算符

| 算术运算符 | 功能     |
|-------|--------|
| +     | 加      |
| _     | 减      |
| *     | 减<br>乘 |
| 1     | 除      |
| %     | 求模     |

- ❖ 进行整数除法运算时,结果值略去小数部分,只取整数部分!
- ❖ 求模即是求一个数被另一个数相除后所得的余数。%称为求模 (或求余)运算符,要求%两侧均为整型数据;
- ❖ 求模运算结果值的符号位取第一个操作数的符号位! 【例】-11%3 结果为-2
- ❖ 进行算术运算时,若某操作数为不定值x,则整个结果也为x。

### 3.2 Verilog HDL的词法 - 运算符及表达式 - 逻辑运算符

- ❖逻辑运算符把它的操作数当作布尔变量(逻辑1、逻辑0或不定值):
  - ▶非零的操作数被认为是真(1'b1);
  - ▶零被认为是假(1'b0);
  - ▶ 不确定的操作数如4'bxx00,被认为是不确定的(可能为零,也可能为非零)(记为1'bx);但4'bxx11被认为是真(记为1'b1,因为它肯定是非零的)。
- 【例】若A=4'b0000,B=4'b0101 则 !A=1'b1,A&&B=1'b0,A||B=1'b1。
- 逻辑运算符
   功能

   &&(双目)
   逻辑与

   ||(双目)
   逻辑或

   !(单目)
   逻辑非
- ❖ 如果操作数不止一位,应将操作数作为一个整体来对待!
- ❖ 进行逻辑运算后的结果为布尔值(为1或0或x)!

**企** 北京航空航天大學

110

# 3.2 Verilog HDL的词法 - 运算符及表达式 - 位运算符



- ❖ 位运算符中的双目运算符要求对两个操作数的相应位<mark>逐位</mark> 进行逻辑运算。位运算其结果与操作数位数相同。
- ❖ 两个不同长度的操作数进行位运算时,将自动按右端对齐, 位数少的操作数会在高位用0补齐。

【例】若A = 5'b11001,B = 3'b101, 则A & B = (5'b11001) & (5'b00101) = 5'b00001

gg 北京航空航天大学

### 3.2 Verilog HDL的词法 - 运算符及表达式 - 关系运算符

❖ 双目运算符

❖ 用来对两个操作数进行比较。

也是非阻塞赋值 运算的赋值符号

| 关系运算符        | 功能    |
|--------------|-------|
| <b>'</b>     | 小于    |
| <b>(=</b>    | 小于或等于 |
| >            | 大于    |
| <b>&gt;=</b> | 大于或等于 |

- ❖ 运算结果为1位的逻辑值1或0或x。关系运算时,若声明的关系为真, 则返回值为1: 若关系为假,则返回值为0: 若某操作数为不定值x, 则返回值为x,表示结果是模糊的。
- ❖ 所有的关系运算符优先级别相同。
- ❖ 关系运算符的优先级低于算术运算符。

【例】 a<size - 1

等同于: a<(size - 1)

size - (1<a) 不等同于: size-1<a

括号内先运算

算术运算先运算

2 北京航空航天大學

3.2 Verilog HDL的词法 - 运算符及表达式 - 等值运算符

❖双目运算符

| 等值运算符 | 功能  |  |
|-------|-----|--|
| ==    | 等于  |  |
| !=    | 不等于 |  |
| ===   | 全等  |  |
| !==   | 不全等 |  |

#### Quartus II不支持!

- ❖ 运算结果为1位的逻辑值1或0或x。
- ❖ 所有的等值运算符优先级别相同。
- ❖ ===和!==运算符常用于case表达式的判别,又称为 "case等式 运算符"。
- ❖等于运算符(==)和全等运算符(===)的区别:
  - ▶ 使用等于运算符时,两个操作数必须逐位相等,结果才为1; 若某些位 为x或z,则结果为x。
  - 使用全等运算符时,若两个操作数的相应位形式上完全一致(如同 是1,或同是0,或同是x,或同是z),则结果为1;否则为0。

**公 北京航空航天大学** 

# 3.2 Verilog HDL的词法 — 运算符及表达式 — 缩减运算符

- ❖ 单目运算符
- ❖运算法则与位运算符类似,但 运算对象只有一个,运算过程 不同!

注意缩减运算符和 位运算符的区别!

| 缩减运算符  | 功能 |
|--------|----|
| &      | 与  |
| ~&     | 与非 |
| I      | 或  |
| ~      | 或非 |
| ^      | 异或 |
| ^~, ~^ | 同或 |

位运 算符 没有

- ❖ 对单个操作数进行递推运算,即先将操作数的最低位与第二位进行 与、或、与非、或非等运算,再将运算结果与第三位进行相同的运 算,依次类推,直至最高位。
- ❖ 运算结果缩减为1位二进制数。

【例 】 reg[3:0] a;

b=|a; //等效于 b =( (a[0] | a[1]) | a(2)) | a[3]

【例2.18】设A = 8'b11010001,则&A = 0(在与缩减运算中,只有A中的数字全为1时,结果才为1);A = 1(在或缩减运算中,只有A中的数字全为0时,结果才为0)。

**公** 北京航空航天大學

114

# 3.2 Verilog HDL的词法 - 运算符及表达式 - 移位运算符

- ❖单目运算符
- ❖常用于移位寄存器的设计

| 移位运算符 | 功能 |
|-------|----|
| >>    | 右移 |
| <<    | 左移 |

❖ 用法: A>>n 或 A<<n 将操作数右移或左移n位,同时用n个0填补移出的空位。注意 操作数的位数不变!

【例】 4'b1001>>3 的结果= 4'b0001; 4'b1001>>4的结果= 4'b0000 4'b1001<<1的结果= 4'b0010; 4'b1001<<2的结果= 4'b0100; 1<<6=32'b00...01000000

左移的数据 会丢失!

将操作数右移或左移n位,相 当于将操作数除以或乘以2<sup>n</sup>。 右移的数据 会丢失!

# 3.2 Verilog HDL的词法 — 运算符及表达式 — 条件运算符

- ❖三目运算符
- ❖常用于数据选择器的设计
- ❖条件运算符为?:

当条件为真, 信号取表 达式1的值;为假,则 取表达式2的值。

❖用法: 信号 = 条件?表达式1: 表达式2;

【例】数据选择器assign out = sel? in1:in0;



2 北京航空航天大學

- 3.2 Verilog HDL的词法 运算符及表达式 位拼接运算符
- ❖位拼接运算符为{}
- ❖用于将两个或多个信号的某些位拼接起来,表示一个整体信 号。
- ❖用法: {信号1的某几位,信号2的某几位,.....,信号n的某几位}
- >例如在进行全加运算时,可将进位输出与算术和拼接在一起使用。

【例2.19】output [3:0] sum;

//算术和

output cout;

//进位输出

input[3:0] ina,inb;

input cin;

assign {cout,sum} = ina + inb +cin; //进位与算术和拼接在一起

>位拼接可以嵌套使用,或用重复法简化书写

【例2.20】{3{a,b[3:0]}}

 $=\{ \{a,b[3],b[2],b[1],b[0]\}, \{a,b[3],b[2],b[1],b[0]\}, \{a,b[3],b[2],b[1],b[0]\} \}$ 

**公 北京航空航天大学** 

# 3.2 Verilog HDL的词法 — 运算符及表达式—运算符的优先级

| 类 别     | 运算符        | 优先级 |
|---------|------------|-----|
| 逻辑非、按位取 | ! ~        | 高   |
| 反       |            |     |
| 算术运算符   | * / %      |     |
|         | + -        |     |
| 移位运算符   | << >>      |     |
| 关系运算符   | < <= > >=  |     |
| 等式运算符   | ==!====!== |     |
| 缩减运算符   | & ~&       |     |
|         | ۸ ۸~       |     |
|         | ~          |     |
| 逻辑运算符   | &&         | 1   |
|         |            | 低   |
| 条件运算符   | ?:         |     |

\*为避免错误,提高程序的可读性,建议使用括号来控制运算的优先级!

【例】 (a>b)&&(b>c) (a==b)||(x==y) (!a)||(a>b)

**3.** 北京航空航天大學

118

# 3.2 Verilog HDL的词法 — Verilog HDL数据对象 — 常量

# 七、 Verilog HDL数据对象

- ❖ Verilog HDL数据对象是指用来存放各种类型数据的容器,包括常量和变量 (1) 常量
  - ▶常量用来存放一个恒定不变的数据,一般在程序前部定义。用parameter 来定义一个标识符,代表一个常量,称为符号常量或parameter常量。

### parameter 常量名1 = 表达式, 常量名2 = 表达式, ..., 常量名n = 表达式;

- ▶parameter是常量定义关键字,常量名是用户定义的标识符,表达式是为常量赋的值。
- ▶每个赋值语句的右边必须为常数表达式,即只能包含数字或先前定义过的符号常量!
- ▶parameter常量常用来定义延迟时间和变量宽度。当程序中有多处地方用到相同的常量时,建议用parameter常量来定义—便于修改,有意义
- >parameter常量是本地的,其定义只在本模块内有效。

【例】 parameter addrwidth = 16;

gg 北京航空航天大学

### (2) 变量

- ❖在程序运行过程中,其值可以改变的量,称为变量。
- ❖其数据类型有19种,常用的有3种:
  - ▶网络型(nets type)
  - ▶寄存器型(register type)
  - ▶数组 (memory type)
- ❖其它数据类型: large型、medium型、 scalared型、small型、time型、tri 型、tri0型、tri1型、triand型、trior 型、trireg型、vectored型、wand型、 wor型等

┏ 北京航空航天大學

120

# 3.2 Verilog HDL的词法 — Verilog HDL数据对象 — 变量

# 1. nets型变量

- ❖ 网络型变量 (nets型变量) 是输出值始终随输入的变化 而变化的变量。
- ❖ 一般用来定义电路中的各种物理连线。
- ❖ 有两种驱动方式:在结构描述中将其连接到一个门元件 或模块的输出端;或用assign语句对其赋值
- ❖常用的nets型变量
  - ▶ wire, tri: 连线类型(两者功能一致), 可综合
  - > wor, trior: 具有线或特性的连线(两者功能一致)
  - > wand, triand: 具有线与特性的连线(两者功能一致)
  - ➤ tri1, tri0: 上拉电阻和下拉电阻
  - > supply1, supply0: 电源(逻辑1)和地(逻辑0),可综合

### ❖wire型变量

- ▶ 最常用的nets型变量,常用来表示以assign语句赋值的组合逻辑信号
- ▶ 模块中的输入/输出信号类型<mark>缺省为wire型——当对输入/输出信号不加以信号类型声明时,则输入/输出信号为wire型</mark>
- > 可用做任何方程式的输入,或 "assign"语句和实例元件的输出

### 

wire 变量名1.变量名2. ....变量名n:

【例】 将输入a赋值给wire型变量b input a:

wire b; /\* 中间节点。若为output信号,则默认为wire型变量,不必单独声明 \*/ assign b=a; //当a变化时,b立即随之变化

**6.** 北京航空航天大學

122

# 3.2 Verilog HDL的词法 — Verilog HDL数据对象 — 变量

### wire型向量(总线)

位宽为1位的变量称为标量,

位宽超过1位的变量称为向量。向量的宽度定义:

[MSB: LSB] /\* MSB(Most Significant Bit, 最高有效位), LSB (Least Significant Bit, 最低有效位) \*/

EEB

wire[n-1:0] 变量名1,变量名2,...,变量名m; 或 wire[n;1] 变量名1,变量名2,...,变量名m;

> 每条总线 位宽为n

共有m 条总线

# 【例】 wire型向量

wire[7:0] in,out;

assign out=in; //将等号右边的值赋给等号左边的变量。

**企** 北京航空航天大學

# 2. register型变量

- ❖寄存器型变量(register型变量)对应具有状态保持作用的 电路元件(如触发器、寄存器等),常用来表示过程块语句 (如 initial, always, task, function)内的指定信号。
- ❖常用的register型变量
  - ▶reg: 常代表触发器、寄存器,可综合
  - ▶integer: 32位带符号整数型变量,可综合
  - ┌≥real: 64位带符号实数型变量,表示实数寄存器,用于仿真
  - ≥time: 无符号时间变量,用于对仿真时间的存储与处理

纯数学的 抽象描述

┏ 北京航空航天大學

124

# 3.2 Verilog HDL的词法 — Verilog HDL数据对象 — 变量

register型变量与nets型变量的区别



- ❖ register型变量必须通过过程赋值语句赋值!不能通过assign语句赋值!
- ❖ nets型变量必须通过assign语句赋值!不能通过 过程赋值语句赋值!
- ❖ 在always、initial、task、function等过程块内被 赋值的每个信号必须定义成register型!

**企** 北京航空航天大学

### **❖reg型变量**

- ▶ reg型变量是数字系统中存储设备的抽象,常用于具体的硬件描述, 是最常用的寄存器型变量。
- ▶它是在过程块中被赋值的信号,<mark>往往</mark>代表触发器(沿触发时),但 不一定就是触发器(也可以是组合逻辑信号,电平触发时)!

**信**式 reg 变量名1,变量名2, ......,变量名n;

❖reg型向量(总线)

reg[n-1:0] 变量名1,变量名2, ...,变量名m; 或 reg[n:1] 变量名1,变量名2, ...,变量名m;

【例】 reg[4:1] regc,regd; //regc,regd为4位宽的reg型向量 reg[0:7] data; //8位寄存器型变量,最高有效位是0,最低有效位是7 >向量定义后可以采用多种使用形式(即赋值) data=8'b00000000; data[5:3]=3'B111; data[7]=1;

**加** 北京航空航天大學

126

# 3.2 Verilog HDL的词法 — Verilog HDL数据对象 — 变量

reg型变量牛成触发器和组合逻辑举例

【例】在时钟沿触发的always块中,用 reg型变量生成<mark>触发器</mark> module rw1( clk, d, out1);

module rw1( clk, d, out input clk, d; output out1;

reg out1;

always @(posedge clk) //沿触发 out1 <= d;

endmodule



【例】使用电平触发,用reg型 变量生成组合逻辑

module rw2( clk, d, out1); input clk, d; output out1;

reg out1:

always @(d) //电平触发 out1 <= d; endmodule



gg 北京航空航天大学

# 第二部分:组合逻辑

- 一. 逻辑门电路
- 二. 布尔代数
- 三. Verilog HDL介绍
  - 1. Verilog HDL概述
  - 2. Verilog HDL的词法
  - 3. Verilog HDL常用语句
  - 4. 不同抽象级别的Verilog HDL模型
- 四. 基本组合逻辑部件设计
  - 1. 运算单元电路
  - 2. 编码器/译码器
  - 3. 多路选择器

O 北京航空航天大学

128

### 3.3 Verilog HDL常用语句

- 一、结构声明语句
- 二、赋值语句
- 三、条件语句
- 四、循环语句
- 五、语句的顺序执行与并行执行
- ❖ 语句是构成Verilog HDL程序不可缺少的部分。
- ❖ Verilog HDL的语句包括赋值语句、条件语句、循环语句、 结构声明语句和编译预处理语句(本课程不学习)等类型, 每一类语句又包括几种不同的语句。
- ❖ 有些语句属于顺序执行语句,有些语句属于并行执行语句。

# 3.3 Verilog HDL常用语句

|         | 门基元赋值语句      |               |
|---------|--------------|---------------|
| 赋值语句    | 连续赋值语句       |               |
|         | 过程赋值语句       |               |
| 块语句     | begin_end语句  |               |
|         | fork_join语句  | Quartus II不支持 |
| 条件语句    | if_else语句    |               |
|         | case语句       |               |
|         | forever语句    |               |
| 循环语句    | repeat语句     |               |
|         | while语句      |               |
|         | for语句        |               |
|         | initial语句    | Quartus II不支持 |
| 结构声明语句  | always语句     |               |
|         | task语句       |               |
|         | function语句   |               |
|         | 'define语句    |               |
| 编译预处理语句 | 'include语句   | Quartus II不支持 |
|         | 'timescale语句 | Quartus II不支持 |

**6** 北京航空航天大学

130

# 3.3 Verilog HDL常用语句—结构声明语句—always块语句

# 一、结构声明语句

- ❖具有某种独立功能的电路,均在过程块中描述,Verilog HDL的任何过程模块都是放在结构声明语句中,结构声明语句分为4种:
  - ▶initial语句——常用于仿真中各变量的初始化,沿时间轴只执行一次
  - ▶always语句——不断重复执行,直到仿真结束
  - ▶task语句——用于定义任务,可在程序模块中的一处或多处调用
  - ▶function语句——用于定义函数,可在程序模块中的一处或多处调用

### 1、always块语句

- ❖ 包含一个或一个以上的声明语句 (如: 过程赋值语句、任务调用、 条件语句和循环语句等),在仿真运行的全过程中,在定时控制 下被反复执行。
  - ❖ always块通常带触发条件;



- ❖ 在always块中被赋值的只能是register型变量。
- ❖ 每个在仿真一开始便开始执行,当执行完块中最后一个语句,继续从always块的开头执行。

ca 北京航空航天大学

#### 3.3 Verilog HDL常用语句—结构声明语句—always块语句 always @(敏感信号表达式) begin #过程赋值语句; 经部 #if语句, case语句; // for语句,while语句,repeat语句; // tast语句、function语句: end 个变量不 能在多个 always块中 被赋值! 正确的写法: always @ (posedge clk or 错误的写法: negedge reset) always @ (posedge clk) begin q=q+1; if(!reset) q=0; //异步清零 always @ (negedge reset) else q=q+1; q=0; end **3.** 北京航空航天大學

### 3.3 Verilog HDL常用语句—结构声明语句—initial语句

#### 2、initial语句

❖ initial语句是面向模拟仿真的过程语句,通常不能被逻辑综合工 具支持。initial块内的语句仅执行一次。

### 【例2.26】对各变量进行初始化。

超到

```
initial
begin
语句1;
语句2;
.....
语句n;
end
```

```
parameter size=16;
reg[3:0] addr;
reg reg1;
reg[7:0] memory[0:15];
initial
  begin
    reg1 = 0;
    for(addr=0;addr<size;addr=addr+1);
    memory[addr]=0;
end</pre>
```

# 用途

- ❖ 在仿真的初始状态对各变量进行初始化;
- ❖ 在测试文件中生成激励波形(如时钟信号)作为电路的仿真信号。

gg 北京航空航天大学

# 3.3 Verilog HDL常用语句—结构声明语句 — task语句

#### 3、task语句

- ❖ task语句用来由用户定义任务,任务类似高级语言中的子程序,用来单独完成某项具体任务,并可以被模块或其他任务调用。
- ❖ 当希望能够对多个信号进行一些运算并输出多个结果(即有多个输出变量)时,宜采用任务结构。
- ❖ 常常利用任务来帮助实现结构化的模块设计,将批量的操作以任务的形式独立出来,使设计简单明了,而且便于调试。

假务定义

task <任务名>; 端口声明语句; 数据类型声明语句; 实现逻辑功能的语句; endtask

端口名列表与任 务定义中的I/O变 量一一对应!

注意无端口列表!

任务调用

<任务名>(端口1,端口2,.....);\*

2 北京航空航天大學

134

# 3.3 Verilog HDL常用语句—结构声明语句— task语句

### task语句使用注意事项

注1: 任务的定义与调用必须在一个module模块内!

注2: 任务被调用时,需列出端口名列表,端口名的排

序与类型必须与任务定义中的I/O变量相一致!

注3: 一个任务可以调用其他任务和函数。

### 【例】任务的定义与调用。

任务定义

# task my\_task; input a,b;

inout c; output d,e;

------<语句> //

//执行任务工作相应的语句

c = foo1;

d = foo2; //对任务的输出变量赋值

e = foo3;

endtask

任务调用

my\_task (v,w,x,y,z);

- ➤ 当任务启动时,由v、w和 x传入的变量赋给了a、b 和c:
- ▶ 当任务完成后,输出通过 c、d和e赋给了x、y和z。

### 3.3 Verilog HDL常用语句—结构声明语句 — function语句

#### 4、function语句

- ❖ function语句用来定义函数,函数的目的是通过返回一个用于某表达式的值,来响应输入信号。适于对不同变量采取同一运算的操作。
- ❖ 函数在模块内部定义,通常在本模块中调用,也能根据按模块层次分级命名的函数名从其他模块调用。而任务只能在同一模块内定义与调用!

國際定义

function <返回值位宽或类型说明>函数名; 少端口声明;

只能是input

局部变量定义; 其他语句: 缺省则返回1位 reg型数据

endfunction

与函数定义中的 输入变量对应!

**國數頒用** 

<函数名>(<表达式1>, <表达式2>, .....)

2 北京航空航天大学

136

# 3.3 Verilog HDL常用语句—结构声明语句 — function语句

### function语句举例

函数在综合时被理解成具有独立运算功能的电路,每调用一次 函数,相当于改变此电路的输入,以得到相应的计算结果。

【例2.29】 利用函数对一个8位二进制数中为0的位进行计数。 function[3:0] get0: //函数的定义,计算x中0的个数

reg[3:0] count; integer i;

begin count=0:

for(i=0;i<=7;i=i+1) //循环核对x中的每一位

if(x[i]==1'b0) count=count+1;

- get0 = count; //将运算结果赋给与函数同名的内部寄存器

内部寄存器 endfunction

对应函数的输入变量

assign number = get0(réga); //对函数的调用

函数的调用是通过将函数作为调用函数的表达式中的操作数来实现的!

**企** 北京航空航天大學

# 3.3 Verilog HDL常用语句—赋值语句—门基元赋值语句

### 二、赋值语句

- ❖赋值语句分为3类:
  - 1、门基元赋值语句(门元件例化)

### 基本逻辑门关键字(门输出,门输入1,门输入2,...,门输入n);

- ➤ 基本逻辑门关键字是Verilog HDL预定义的逻辑门,包括 and、or、not、xor、nand、nor等; 圆括弧中内容是被描述门的输出和输入信号。
- ▶ 例如,具有a、b、c、d 这4个输入和y为输出的与非门的门基元赋值语句为nand (y,a,b,c,d);

该语句与assign y = ! (a && b && c && d);等效

┏ 北京航空航天大學

138

# 3.3 Verilog HDL常用语句--赋值语句--连续赋值语句

2、连续赋值语句(assign语句)

用于对wire型变量赋值,是描述组合逻辑最常用的方法之一。

assign 赋值变量 = 表达式;

- ▶【例】4输入与非门 assign y = ! (a && b && c && d);
- ▶ 连续赋值语句的 "="号两边的变量都应该是wire型变量。
- ➤ 在执行中,输出y的变化跟随输入a、b、c、d的变化而变化, 反映了信息传送的连续性。

### 【例】 2选1多路选择器

module mux2\_1(out,a,b,sel);

input a,b,sel; output out; //输入、输出信号默认为wire型变量 assign out =( sel==0) ? a:b; //若sel为0,则out=a; 否则out=b endmodule

**60 北京航空航车表** 

# 3.3 Verilog HDL常用语句 — 赋值语句 — 过程赋值语句

### 3、过程赋值语句

用于对**reg**型变量赋值,过程赋值语句出现在initial和always块语句中,有两种赋值方式:

▶阻塞(blocking)赋值方式: 赋值符号为=,如 b = a;

#### 赋值变量 = 表达式;

▶非阻塞(non-blocking)赋值方式;赋值符号为<=,如 b <= a;

赋值变量 <= 表达式;

gg 北京航空航天大學

140

# 非阻塞赋值与阻塞赋值的区别 (1) 非阻塞赋值方式 always @(posedge clk) begin clk b <= a; c <= b; c end **DFF DFF** 非阻塞赋值在 块结束时才完 成赋值操作! > c的值比b的值落后一个时钟周期! > 若块内有多条赋值语句,则在块结束时同时赋值。 多条非阻塞赋值语句并行执行! **分** 北京航空航天大 141



# 3.3 Verilog HDL常用语句 — 条件语句 — if-else语句

条件语句分为两种: if-else语句和case语句; 它们都是顺序语句,应放在 "always"块内!

> 对于每个判定 只有两个分支

### 1、if-else语句

- ❖ 判定所给条件是否满足,根据判定的结果(真或假)决定执行给出的两种操作之一。
- ❖ if-else语句有3种形式,格式与C语言中的if-else语句类似
  - ▶其中"表达式"为逻辑表达式或关系表达式,或一位的变量。
  - ▶若表达式的值为0、或x、或z,则判定的结果为"假";若为1,则结果为"真"。
  - ▶执行的语句可为单句,也可为多句;多句时一定要用"begin\_end" 语句括起来,形成一个复合块语句。

# 3.3 Verilog HDL常用语句 — 条件语句 — if-else语句

if-else语句的表示方式

方式1(非完整性IF语句): 方式3(多重选择

方式3(多重选择 多重选择 条件,执行不 同的语句

if (表达式) 语句1;

方式2 (二重选择的IF语句):

if (表达式1) 语句1; else 语句2; if (表达式1) 语句1; else if (表达式2) 语句2; else if (表达式3) 语句3;

else 语句n:

- ❖ else子句不能作为语句单独使用,它是if 语句的一部分,必须与if配对使用!
- ❖允许一定形式的表达式简写方式,如:
  - ➤if(expression) 等同于if(expression = = 1)
  - ▶if(! expression) 等同于if(expression!=1)

**《** 北京航空航天大學

144

# 3.3 Verilog HDL常用语句 — 条件语句 — case语句

2、case语句

多分支选择语句

适于对同一组控制 信号取不同的值时, 输出取不同的值!

- ❖ 当敏感表达式取不同的值时,执行不同的语句。
- ❖ 功能: 当某个或某组(控制)信号取不同的值时,给 另一个(输出)信号赋不同的值。常用于多条件译码 电路(如译码器、数据选择器、状态机、微处理器的 指令译码)!

case语句的3种形式

?°°

case

casez

casex

case语句与if-else语 句有什么区别呢?

**企** 北京航空航天大學

# 3.3 Verilog HDL常用语句 — 条件语句 — case语句

#### case语句的语法格式

case (敏感表达式)

值1: 语句1; 值2: 语句2:

...

值n: 语句n; default: 语句n+1;

endcase

#### ❖说明:

- ▶其中"敏感表达式"又称为"控制表达式",通常表示为控制信号的某些位。当有多个信号时,可用位拼接符将它们连接起来:
- 【例】 case({D3,D2,D1,D0})
- ▶值1~值n称为分支表达式,用控制信号的具体状态值表示,因此又称为常量表达式。
- ▶default项可有可无,一个case语句里只能有一个default项!
- ▶值1~值n必须互不相同,否则矛盾。
- ▶值1~值n的位宽必须相等,且与控制表达式的位宽相同。

**3.** 北京航空航天大學

146

# 3.3 Verilog HDL常用语句 — 循环语句

- ❖循环语句用来控制语句的执行次数。分为4种:
  - ▶ for语句——有条件的循环语句。通过3个步骤来决定语句的循环 执行:
    - (1) 给控制循环次数的变量赋初值。
    - (2) 判定循环执行条件,若为假则跳出循环;若为真,则执行指定的语句后,转到第(3)步。
    - (3)修改循环变量的值,返回第(2)步。
  - > repeat语句——连续执行一条语句n次
  - ➤ while语句——执行一条语句直到某个条件不满足。首先判断循环执行条件表达式是否为真,若为真,则执行后面的语句或语句块,直到条件表达式不为真;若不为真,则其后的语句一次也不被执行!
  - ➤ forever语句——无限连续地执行语句,可用disable语句中断! 多用在initial块中,以生成时钟等周期性波形

### 3.3 Verilog HDL常用语句 — 循环语句 — for语句

#### 1、for语句

功能:一般用途的循环语句,允许一条或更多的语句被重复地执行。

一般形式

for (表达式1:表达式2:表达式3)语句

简单应用形式

for (循环指针 = 初值; 循环指针 < 终值; 循环指针 = 循环指针 + 步长值)
begin 执行语句; end

for语句比while语句简洁!

两条语句

规则: 当for循环开始执行时,循环变量已赋予初值。在每一次循环执行之前(包括第一次),都必须检查表达式2(循环执行条件),若它为假(0、x或z),则立刻退出循环。而在每一次循环执行之后,都要使循环变量增值。

**分,北京航空航天大学** 

148

# 3.3 Verilog HDL常用语句 — 循环语句 — for语句

### for语句举例

【例2.34】用for语句描述的7人投票表决器: 若超过4人(含4人)投赞成票,则表决通过。

```
module vote7 ( pass, vote );
  output pass;
  input [6:0] vote;
  reg[2:0] sum; //sum为reg型变量,用于统计赞成的人数
  integer i;
              #循环变量
  reg pass;
  always @(vote)
                      将循环变量定义为整型
   begin
                        __//sum初值为0
     sum = 3'b000;
     for(i = 0; i < = 6; i = i+1)
                          //for语句
       if(vote[i]) sum = sum+1; //只要有人投赞成票,则 sum加1
     pass =1'b 0;
     else
   end
endmodule
```

# 3.3 Verilog HDL常用语句 — 循环语句 — repeat语句

### 2、repeat语句

只有部分综合工具可以 综合此语句!

- ❖功能: 把一条或多条语句连续执行指定的次数。
- ❖规则: 重复执行的次数由循环次数表达式的值决定,若该值为0、x或z,则不会重复执行。

repeat (循环次数表达式) 语句

或

repeat (循环次数表达式) begin

end

执行语句为多条语句

2 北京航空航天大學

150

# 3.3 Verilog HDL常用语句 — 循环语句 — while语句

- 3、while语句
- ❖ 功能:有条件地执行一条或多条语句。只要循环执行条件表达式为真,则循环语句就重复执行!
- ❖ 规则: 首先判断循环执行条件表达式是否为真。若为真,则执行后面的语句或语句块; 然后再回头判断循环执行条件表达式是否为真, 若为真, 再执行一次后面的语句; 如此不断, 直到条件表达式不为真。

(EE)

while (循环执行条件表达式) begin ......

end

while语句 通常用在测 试文件中!

- 1.首先判断循环执行条件表达式是否为真,若不为真,则其后的语句 一次也不被执行!
- 2.在执行语句中,必须有一条改变循环执行条件表达式的值的语句!
- 3.while语句只有当循环块有事件控制(即@(posedge clock))时 才可综合!

**《** 北京航空航天大學

# 3.3 Verilog HDL常用语句 — 循环语句 — while语句

【例2.37】用while语句对一个8位二进制数中值为1的位进行计数

```
module count1s_while ( count,rega,clk );
      output[3:0] count;
                                         此何用for语句
      input [7:0] rega;
      input clk;
                                         改写此程序呢?
                count:
      rea[3:0]
      always @(posedge clk)
        begin:count1
          reg[7:0] tempreg;
                              //用作循环执行条件表达式
                             // count 初值为0
          count = 0:
          tempreg = rega;
                             // tempreg 初值为rega
          while(tempreg)
                             // 若tempreg 非 0, 则执行以下语句
            begin
              if(tempreg[0])
                            count = count+1;
                         //只要tempreg最低位为1,则 count 加1
              tempreg = tempreg >>1; //右移1位
            end
        end
                           改变循环执行条件表达式的值
    endmodule
外北京航空航天大学
```

# 3.3 Verilog HDL常用语句 — 循环语句 — forever语句

- 4、forever语句
- ❖功能: 无条件连续执行forever后面的语句或语句块。

(EE)

forever begin end



- ▶forever循环应包括定时控制或能够使其自身停止循环,否则循环将无限进行下去!
- ▶常用在测试文件中产生周期性的波形,作为估真激励信号。
- ▶可综合性: 尽管Quartus II支持该语句,但一般情况下是不可综合的! 如果forever循环被@(posedge clock)形式的时间控制打断,则是可综合的。

**67** 北京航空航天大学

153

# 3.3 Verilog HDL常用语句 — 语句的顺序执行与并行执行

### 五、语句的顺序执行与并行执行

- ❖Verilog HDL中有顺序执行语句和并行执行语句之分。
- 1、语句的顺序执行
  - ➤在 "always"模块内,对于阻塞赋值语句,逻辑按书写的顺序执行,若随意颠倒赋值语句的书写顺序,可能导致不同的结果!
    (见下页例子)。
  - ▶而对于非阻塞赋值语句,是并发执行的。
  - >注意阻塞赋值语句当本语句结束时即完成赋值操作!

**3.** 北京航空航天大学

154

# 3.3 Verilog HDL常用语句 — 语句的顺序执行与并行执行

### 语句的顺序执行举例

```
【例2.41】顺序执行模块2。
【例2.40】顺序执行模块1。
                              module serial2(q,a,clk);
module serial1(q,a,clk);
                                output q,a;
  output q,a;
                                input clk;
  input clk;
                                reg q,a;
  reg q,a;
  always @(pesadge alk)
对前一时刻的q值取反
                                always 分析一时刻的q值取反
                                  begin
   begin
     q=~q;//阻塞赋值语句
                                    a=~q;
                                    q=~q;
                                  end对前一时刻的q值取反
   end
           对当前时刻的q值反
endmodule
                              endmoduie
    a和q的波形反相!
                                a和q的波形完全相同!
```

**企** 北京航空航天大學

# 3.3 Verilog HDL常用语句 — 语句的顺序执行与并行执行

### 2、语句的并行执行

- ▶多个 "always"模块、 "assign"语句、实例元件调 用、"always"模块内的非阻 塞赋值语句都是并行执行的!
- ▶它们在程序中的先后顺序对结果并没有影响。
- ▶【例2.42】将两条赋值语句分别放在两个 "always"模块中,若颠倒两个 "always"模块的顺序,对仿真结果没有影响,同【例2.41】——q和a的波形完全一样。

```
【例2.42】并行执行模块。
module parall1(q,a,clk);
output q,a;
input clk;
reg q,a;
always @(posedge clk)
begin
q=~q;
end
always @(posedge clk)
begin
a=~q;
end
endmodule
```

gg 北京航空航天大學

156

# 第二部分:组合逻辑

- 一. 逻辑门电路
- 二. 布尔代数
- 三. Verilog HDL介绍
  - 1. Verilog HDL概述
  - 2. Verilog HDL的词法
    - Verilog HDL常用语句
  - 4. 不同抽象级别的Verilog HDL模型
- 四. 基本组合逻辑部件设计
  - 1. 运算单元电路
  - 2. 编码器/译码器
  - 3. 多路选择器

gg 北京航空航天大學

### 3.4 不同抽象级别的Verilog HDL模型

- ❖ 用Verilog HDL描述的电路称为该设计电路的Verilog HDL模型。
- ❖ 一个复杂电路的完整Verilog HDL模型由若干个Verilog HDL模块构成,每个模块由若干的子模块构成——可分别用不同抽象级别的Verilog HDL描述。
- ❖ 在同一个Verilog HDL模块中可有多种级别的描述。
  - ▶<mark>系统级(system level)</mark>: 用高级语言结构(如case语句)实现的设计模块外部性能的模型:
  - ▶<mark>算法级(algorithmic level)</mark>: 用高级语言结构实现的设计算法模型(写出逻辑表达式):
  - ▶ RTL级(register transfer level): 描述数据在寄存器之间流动和如何处理这些数据的模型:
  - ▶ <mark>门级(gate level)</mark>: 描述逻辑门(如与门、非门、或门、与非门、三态门等) 以及逻辑门之间连接的模型:
  - ▶开关级(switch level): 描述器件中三极管和储存节点及其之间连接的模型。

☑ 北京航空航天大學

158

# 3.4 不同抽象级别的Verilog HDL模型

# Verilog HDL的行为描述和结构描述

- ❖ Verilog HDL具有行为描述和结构描述功能。
  - ▶<mark>行为描述</mark>是对设计电路的<mark>逻辑功能</mark>的描述,并不用关心设计电路使用哪些元件以及这些元件之间的连接关系。行为描述通过描述电路的行为特性来设计电路,属于高层次的描述方法,包括系统级、算法级和寄存器传输级等3种抽象级别。
  - ▶结构描述是对设计电路的结构进行描述,即描述设计电路使用的元件及这些元件之间的连接关系。结构描述通过调用电路元件(如逻辑门,甚至晶体管)来构建电路,属于低层次的描述方法,包括门级和开关级2种抽象级别。
- ❖ 应重点掌握高层次描述方法,但门级描述在一些电路设计中也有一定的实际意义(系统速度快)。

# 3.4 不同抽象级别的Verilog HDL模型 —— 结构描述

### 1、Verilog HDL的门级描述

- ❖ 门级描述即直接调用门原语进行逻辑的结构描述。
  - ◆ 以门级为基础的结构描述所建立的硬件模型不仅是可仿 真的,也是可综合的;
  - 一个逻辑网络由许多逻辑门和开关组成,用逻辑门的模型来描述逻辑网络最直观!
- ❖ Verilog HDL内置了26个基本元件,其中14个门级元件,12 个开关级元件。常用的门级元件有9个: not, and, nand, or, nor, xor, xnor, buf, bufif1, bufif0, notif1, notif0(各种三态门)
- ❖ 调用门原语的句法:

可省略!

### 门类型关键字 <例化门的名称> ( <端口列表>);

注1: "例化门的名称"是用户定义的标识符,属于可选项;端口列表按(输出,输入,使能控制端)的顺序列出。

注2: 门级描述不适于描述复杂的系统!

**3.** 北京航空航天大學

160

# 3.4 不同抽象级别的Verilog HDL模型 —— 结构描述

### 门原语调用举例(1/2)

- ❖【例2.43】调用门原语实现4选1数据选择器
- ❖注: 首先必须根据逻辑功能列出真值表!



### 真值表

| 输入            |   | 输出  |
|---------------|---|-----|
| cntrl1 cntrl2 |   | out |
| 0             | 0 | in1 |
| 0             | 1 | in2 |
| 1             | 0 | in3 |
| 1             | 1 | in4 |



gg 北京航空航天大学

### 3.4 不同抽象级别的Verilog HDL模型 —— 结构描述

### 门原语调用举例(2/2)

```
//4_1 multiplexer using gate primitive.
module mymux (out, in1, in2, in3, in4, cntrl1, cntrl2);
    output out;
    input in1, in2, in3, in4, cntrl1, cntrl2;
    wire notctrl1,notcntrl2,w,x,y,z;
    not(notcntrl1,cntrl1);
    not(notcntrl2,cntrl2);
    and(w,in1,notcntrl1,notcntrl2);
    and(x,in2,notcntrl1,cntrl2);
    and(y,in3,cntrl1,notcntrl2);
    and(z,in4,cntrl1,cntrl2);
    or(out,w,x,y,z);
endmodule
```

注: 这里省略了所有的例化门元件名称!

gg 北京航空航天大學

16

# 3.4 不同抽象级别的Verilog HDL模型 —— 行为描述

# 2、Verilog HDL的行为描述

- ❖行为描述是对设计实体的数学模型的描述,其抽象程度远高于结构描述。
- ❖类似于高级编程语言,描述一个设计实体的行为时,无须知道具体电路的结构,只需描述清楚输入与输出信号的行为。
- ❖包括系统级、算法级和RTL级描述等3种抽象级别。
  - ❖ 可综合的行为描述方式多采用always过程语句实现,既 适于描述时序逻辑电路,也适于描述组合逻辑电路。
  - ❖ 设计者只需写出源程序,挑选电路方案的工作由EDA软件自动完成;电路的优化程度取决于综合工具的技术水平和器件的支持能力;最终选取的电路方案所耗用器件资源可能不是最少的。

# 3.4 不同抽象级别的Verilog HDL模型 —— 行为描述

### (1) 算法级抽象—逻辑表达式

算法级抽象主要使用assign语句写出输出的逻辑表述

方法二

注: 首先必须根据逻辑功能写出逻辑表达式! 优点是耗用器件逻辑资源比较少。

164

# 3.4 不同抽象级别的Verilog HDL模型 —— 行为描述

# (2) 系统级抽象—case语句

采用case语句只需知道输入与输出间的真值表! 比调用门原语和 采用逻辑表达式描述都简单和简洁!

```
【例2.45】用case语句描述4选1数据选择器
module mux4_1(out,in1,in2,in3,in4,cntrl1,cntrl2);
output out;
input in1,in2,in3,in4,cntrl1,cntrl2;
reg out;
always @(in1 or in2 or in3 or in4 or cntrl1 or cntrl2)
case ({cntrl1,cntrl2})
    2'b00:out=in1;
    2'b01:out=in2;
    2'b11:out=in3;
    2'b11:out=in4;
    default :out=1'bx;
endcase
endmodule
```

**企** 北京航空航天大學

### 3.4 不同抽象级别的Verilog HDL模型 —— 行为描述

### (3) 算法级抽象—条件运算符

只需知道输入与输出间的真值表!

方法四

【例2.46】 用条件运算符描述4选1数据选择器

module mux4\_1(out,in1,in2,in3,in4,cntrl1,cntrl2);

output out;

input in1,in2,in3,in4,cntrl1,cntrl2;

assign out= cntrl1? (cntrl2? in4 :in3) : (cntrl2? in2 :in1) ;

endmodule

当cntrl1=1时执 行 当cntrl1=0时执 行

注:比调用门原语,采用逻辑表达式或case语句描述代码更简单!但也更抽象!且耗用器件资源更多!

gg 北京航空航天大學

166

# 3.4 不同抽象级别的Verilog HDL模型 —— 小结

# 不同抽象级别的Verilog HDL模型小结

- ① 采用的抽象级别越高,设计越容易,程序代码越简单;但耗用器件资源更多。对特定综合器,可能无法将某些抽象级别高的描述转化为电路!
- ② 基于门级描述的硬件模型不仅可以仿真,而且可 综合,且系统速度快。
- ③ 一般用系统级、算法级(写出逻辑表达式)或RTL 级来描述逻辑功能,尽量避免用门级描述,除非 对系统速度要求比较高的场合才采用门级描述。

gg . 改京,航空航天大學

### 3.4 不同抽象级别的Verilog HDL模型 —— 思考

### 思考:

### ① 采用什么抽象级别更合适?

- ❖ 门级描述要求根据逻辑功能画出逻辑电路图,对于复杂的数字系统很难做到,不建议采用;
- ❖ 系统级描述太抽象,有时无法综合成具体的物理电路, 但代码简单,对于复杂系统建议采用系统级描述;
- ❖ 算法级和RTL级抽象级别适中,代码不是很复杂,且一般容易综合成具体的物理电路,当电路功能不是很复杂时建议采用算法级和RTL级来描述。

### ② 怎样减少器件逻辑资源的耗用?

- 当器件容量有限时,为减少器件逻辑资源的耗用,建议 少用if-else语句和case语句,尽量直接使用逻辑表达式 来描述系统的逻辑功能;
- 🤄 或者用 case语句取代if-else语句。

gg 北京航空航天大学