FPGA学习笔记(七)verilog的深入学习之任务与函数(语法篇3)
创始人
2024-03-31 03:45:26
0

目录

  • 任务与函数
    • 任务
      • 任务的基本概况
      • 自动(可重入)任务
    • 函数
      • 函数概述
      • 自动(递归)函数
      • 常量函数
      • 带符号函数
  • 条件编译

在之前学习的基础上,继续加深对Verilog HDl的学习

前两个见:
FPGA学习笔记(二)Verilog语法初步学习( 语法篇1)

FPGA数字电子技术复习笔记(一)verilog语法规则补充(语法篇2)

任务与函数

在这里插入图片描述
任务更像替代一块verilog代码,其中可以包括延时、时序、事件等语法结构,并且可以具有多个输出变量。
函数更像c语言中有返回值的函数类型,只有一个输出。

任务

任务的基本概况

在这里插入图片描述
语法图看不懂,先记下。

例子:ANSI C风格(c语言的函数风格)

// 定义任务
task bitwise_oper (output [15:0] ab_and,ab_or,ab_xor,input [15:0] a,b);
begin#delay ab_and = a & b;ab_or = a \ b;ab_xor = a ^b;
end
endtask

普通写法:

//定义bitwise_oper任务
task bitwise_oper;
output [15:0] ab_and,ab_or,ab_xor; //任务的输出变量
input [15:0] a, b; //输入到任务中的变量
begin#delay ab_and = a & b;ab_or = a | b;ab_xor = a ^ b;
end
endtask

调用:

reg [15:0]A,B;
reg [15:0]AB_AND,AB_OR,AB_XOR;//变量的指定必须按照任务定义时的声明次序
bitwise_oper(AB_AND,AB_OR,AB_XOR,A,B);

自动(可重入)任务

任务在本质上是静态的,任务中的所有声明项的地址空间是静态分配的,同时并发执行的多个任务共享这些存储区。
如果这个任务在模块中的两个地方被同时调用,则这两个任务调用将对同一块地址空间进行操作。

Verilog通过在task关键字前面添加automatic关键字,使任务中声明的所有模块项的存储空间都是动态分配的,每次调用都对各自独立的地址空间进行操作。所以,最好使用自动任务。
例子:
在这里插入图片描述

函数

在这里插入图片描述

函数概述

在这里插入图片描述
当函数声明的时候,在Verilog 的内部隐含地声明了一个名为function_identifier(函数标识符)的寄存器类型变量,函数的输出结果将通过这个寄存器类型变量被传递回来,所以不需要自己定义输出变量了。

函数的调用通过指明函数名和输人变量来进行。选项range_or_type(类型或范围)说明了内部寄存器的位宽。如果没有指定返回值的类型或位宽,则默认位宽为1。

举例:

reg [31:0] addr;
reg parity;
parity = calc_parity(addr); //定义偶校验位计算函数
function calc_parity;
input [31:0] address;
begin
//适当地设置输出值,使用隐含的内部寄存器calc_parity 
calc_parity = ^address; //返回所有地址位的异或值
end
endfunction

ANSIC的定义风格:

//定义偶校验位计算函数,该函数采用ANSI风格的变量声明
function calc_parity (input [31:0]address) ;
begin
//适当地设置输出值,使用隐含的内部寄存器calc_parity
calc_parity = ^address; //返回所有地址位的异或值
end
endfunction

自动(递归)函数

若某函数在两个不同的地方被同时并发调用,由于这两个调用同时对同一块地址空间进行操作,那么计算结果将是不确定的。
若在函数声明时使用了关键字 automatic,那么该函数将成为自动的或可递归的,即仿真器为每一次函数调用动态地分配新的地址空间,每个函数调用对各自的地址空间进行操作。同自动任务。

//定义自动(递归)函数
function automatic integer factorial;
input [31:0] oper;
integer i;
begin
if (operand >= 2)factorial = factorial (oper -1) *oper; //递归调用
elsefactorial = l ;
end
endfunction

常量函数

之后再加深学习

//定义一个RAM类型
module ram (...) ;
parameter RAM_DEPTH = 256;
input [clogb2(RAM_DEPTH)-1:0] addr_bus; //通过调用下面定义的函数得到clogb2=8
//相当于input [7: 0]addr_bus;
....
//常量函数
function integer clogb2 ( input integer depth) ;
begin
for (clogb2=0 ; depth >0; clogb2=clogb2+1)depth = depth >> 1;
end
endfunction
...
endmodule

带符号函数

之后再加深学习

module top;
...
//带符号的函数声明
//返回一个64位带符号数
function signed [63:0] compute_signed(input [63:0] vector) ;
...
...
endfunction
...
//从上层模块调用带符号函数
if (compute_signed(vector) < -3)
begin
...
end
...
endmodule

条件编译

很想c语言

`ifdef 宏名(标识符)程序段1
`elsif 宏名(标识符)程序段2
`else程序段3
`endif

‘ifndef语句,与’ifdef功能相反,即当宏名没被定义过,就编译程序段1。

如何定义:


`define 宏名 

注意:` ifdef 语句中不允许使用布尔表带式,例如使用 A&& B来表示编译条件是不允许的。

同时可以把这些宏名另外写到txt里面:

`include "xxx.txt"

然后在xxx.txt中写入`define xxxxxxx,该头文件存放位置只要在工程文件中任何位置就行,不能放到工程文件之外。

相关内容

热门资讯

监控摄像头接入GB28181平... 流程简介将监控摄像头的视频在网站和APP中直播,要解决的几个问题是:1&...
Windows10添加群晖磁盘... 在使用群晖NAS时,我们需要通过本地映射的方式把NAS映射成本地的一块磁盘使用。 通过...
protocol buffer... 目录 目录 什么是protocol buffer 1.protobuf 1.1安装  1.2使用...
在Word、WPS中插入AxM... 引言 我最近需要写一些文章,在排版时发现AxMath插入的公式竟然会导致行间距异常&#...
Fluent中创建监测点 1 概述某些仿真问题,需要创建监测点,用于获取空间定点的数据࿰...
educoder数据结构与算法...                                                   ...
MySQL下载和安装(Wind... 前言:刚换了一台电脑,里面所有东西都需要重新配置,习惯了所...
MFC文件操作  MFC提供了一个文件操作的基类CFile,这个类提供了一个没有缓存的二进制格式的磁盘...
有效的括号 一、题目 给定一个只包括 '(',')','{','}'...
【PdgCntEditor】解... 一、问题背景 大部分的图书对应的PDF,目录中的页码并非PDF中直接索引的页码...