hdlbits: 在线学习 systemverilog(三)-problem 10-14 hdlbits 是一组小型电路设计习题集,使用 verilog/systemverilog 硬件描述语言 (hdl) 练习数字硬件设计~
网址如下:
https://hdlbits.01xz.net/
关于hdlbits的verilog实现可以查看下面专栏:
https://www.zhihu.com/column/c_1131528588117385216
缩略词索引:
sv:systemverilog 向量 从这个题目到后面八道题目左右是关于向量的,所以我们先回顾一下向量。
systemverilog有两种类型的数组:压缩数组和非压缩数组。压缩数组是连续存储的位的集合,通常称为向量。非压缩数组是网络或变量的集合。
集合中的每个网络或变量称为数组元素。未压缩数组的每个元素的类型、数据类型和向量大小都完全相同。每个未压缩的数组元素可以独立于其他元素存储;这些元素不需要连续存储。软件工具,如仿真器和综合编译器,可以以工具认为最佳的任何形式组织未压缩数组的存储。
未压缩数组的基本声明语法为:
数组的维度定义了数组可以存储的元素总数。未压缩的数组可以用任意数量的维度声明,每个维度存储指定数量的元素。声明数组维度有两种编码样式:显式地址和数组大小。
显式地址样式指定方括号之间数组维度的起始地址和结束地址,格式为:
start_address 和 end_address可以是任何整数值,数组可以以地址0、地址512或被建模硬件所需的任何地址开始。起始地址和结束地址之间的范围表示数组维度的大小(元素数)。
数组大小样式定义要存储在方括号中的元素数(类似于c语言数组声明样式)。
[size]
使用array_size样式,起始地址始终为0,结束地址始终为size-1
以下是一些未压缩的数组声明示例:
前面的mem声明是16位logic变量的一维数组。一维阵列有时被称为内存阵列,因为它通常用于仿真硬件内存设备(如ram和rom)的存储。
访问数组元素 可以使用数组索引引用未压缩数组的每个元素,索引紧跟在数组名称之后,并且位于方括号中,多维数组需要多组方括号才能从数组中选择单个元素:
数组索引也可以是网络或变量的值,如下一个示例所示:
复制数组(阵列) 如果两个数组(阵列)具有相同的布局,则可以使用赋值语句将一个未压缩的数组(阵列)复制到另一个未压缩的数组(阵列)。也就是说,这两个数组(阵列)必须存储相同向量大小的相同数据类型,必须具有相同的维度数,并且每个维度的大小都相同-
数组(阵列)复制会将源数组(赋值的右侧)的每个元素复制到目标数组(阵列)(赋值的左侧)中相应的元素。两个数组(阵列)的索引编号不需要相同。数组(阵列)的布局和类型必须完全匹配。
与复制数组(阵列)的方式类似,如果两个切片的布局相同,则可以将数组(阵列)的一部分(称为数组(阵列)切片)复制到另一个数组(阵列)的切片。切片是数组(阵列)一维内的一个或多个连续编号的元素。
在成为systemverilog之前,最初的verilog语言将对数组(阵列)的访问限制为一次只能访问数组中的一个元素。不允许对数组(阵列)的多个元素进行数组(阵列)复制和读/写操作。
数组列表赋值 可以为未压缩的数组或数组的一个片段分配一个值列表,这些值包含在每个数组维度的’{and}大括号之间。
列表语法类似于在c中为数组指定值列表,但在大括号前添加了撇号使用’-“作为开头分隔符”表明,所包含的值是表达式列表,而不是systemverilog连接运算符(后面会详细介绍)。
还可以使用嵌套列表为多维数组分配值列表。嵌套的列表集必须与数组的维度完全匹配。
此数组分配相当于以下各项的单独分配:
通过指定默认值,可以为未压缩数组的所有元素指定相同的值。默认值是使用
指定的,如以下代码段所示:
数组元素的位选择和部分选择 可以从数组元素向量中选择一位或一组位。必须首先选择数组的单个元素,然后进行位选择或部分选择。
通过端口将数组传递给任务和函数。任何类型和任意数量的未压缩数组都可以通过模块端口传递,也可以传递到任务和函数参数。端口或任务/函数形式参数也必须声明为数组,端口或参数数组必须与要传递的数组具有相同的布局(与数组复制的规则相同)。
最初的verilog语言只允许简单的向量通过模块端口,或传递到任务或函数参数。要传递上述示例中表数组的值,需要256个端口,数组的每个元素一个端口。
problem 10-vector0 题目说明 构造一个电路,拥有 1 个 3 bit 位宽的输入端口,4 个输出端口。其中一个输出端口直接输出输入的向量,剩下 3 个输出端口分别各自输出 3 bit 中的 1 bit。
图片来自 hdlbits 这个题目的核心就是上面的图片,模块和端口已经被定义好了,黑色的框图以及箭头代表模块和端口。我们需要做的工作是完成图中绿色的部分,其中箭头上的小斜杠旁边的数字代表该向量(总线)的位宽,用于将向量同 wire 信号区别开来。
模块端口声明 module top_module ( input wire [2:0] vec, output wire [2:0] outv, output wire o2, output wire o1, output wire o0 ); 题目解析 这个题目重点是向量的选取和赋值,在文章最前面已经介绍过了,大家仔细看过应该很好理解。
module top_module ( input logic [2:0] vec, output logic [2:0] outv, output logic o2, output logic o1, output logic o0 ); assign outv = vec; assign o0 = vec[0]; assign o1 = vec[1]; assign o2 = vec[2]; endmodule 点击submit,等待一会就能看到下图结果:
注意图中的ref是参考波形,yours是你的代码生成的波形,网站会对比这两个波形,一旦这两者不匹配,仿真结果会变红(后面会展示)。
这一题就结束了。
problem 11-vector1 题目说明 16 位输入信号分别输出 高 8 位 和低 8 位。
模块端口声明 `default_nettype none // disable implicit nets. reduces some types of bugs.module top_module( input wire [15:0] in, output wire [7:0] out_hi, output wire [7:0] out_lo ); 通过添加 `default_nettype none 宏定义会关闭隐式声明功能,那么这样一来,使用未声明的变量就会变成一个 error 而不再只是 warning。 题目解析 这道题难度不大主要是向量的选择,但是要掌握的知识很多,这题前面由很多这方面的知识点,包括压缩数组和非压缩数组,隐式转换等,建议大家在做题前仔细阅读题目前的知识点。
也可以查看下面的文章:
https://zhuanlan.zhihu.com/p/57452966
`default_nettype none // disable implicit nets. reduces some types of bugs.module top_module( input wire logic [15:0] in, output var logic [7:0] out_hi, output var logic [7:0] out_lo ); assign out_hi = in[15:8] ; assign out_lo = in[7:0] ;endmodule 点击submit,等待一会就能看到下图结果:
注意图中的ref是参考波形,yours是你的代码生成的波形,网站会对比这两个波形,一旦这两者不匹配,仿真结果会变红(后面会展示)。
这一题就结束了。
problem 12-vector2 题目说明 一个 32 位的向量可以看做由 4 个字节组成(bits[31:24],[23:16],等等)。构建一个电路,将输入向量的字节顺序颠倒,也就是字节序大小端转换。
如:
aaaaaaaabbbbbbbbccccccccdddddddd => ddddddddccccccccbbbbbbbbaaaaaaaa
模块端口声明 module top_module( input [31:0] in, output [31:0] out ); 题目解析 还是在联系向量的提取和赋值。
module top_module( input logic [31:0] in, output logic[31:0] out ); assign out[31:24] = in[7:0] ; assign out[23:16] = in[15:8] ; assign out[15:8] = in[23:16] ; assign out[7:0] = in[31:24] ;endmodule 点击submit,等待一会就能看到下图结果:
注意图中的ref是参考波形,yours是你的代码生成的波形,网站会对比这两个波形,一旦这两者不匹配,仿真结果会变红(后面会展示)。
这一题就结束了。
ps:下面的答案也是对的
assign out = {in[7-:8],in[15-:8],in[23-:8],in[31-:8]}; 可以自行查找知识学习上面的写法。
problem 13-vectorgates 题目说明 a,b是两个 3bit 宽的输入变量,要求输出 a,b 逐位或,a,b 逻辑或以及 a,b 按位取反的结果,其中 b 在高位。
图片来自 hdlbits 问题的核心就是上面的图片,模块和端口已经被定义好了,黑色的框图以及箭头代表模块和端口。我们需要做的工作是完成图中绿色的部分。
模块端口声明 module top_module( input [2:0] a, input [2:0] b, output [2:0] out_or_bitwise, output out_or_logical, output [5:0] out_not); 题目解析 这个题目还是挺绕的,难度不是很大,理解题目就能写出来,这里除了上一个题目的向量提取和赋值以外,还由一个知识点-逻辑求反运算符(!)和按位反转运算符(~)之间的区别~
之前简单介绍了以下,这次就详细介绍,后续就不展开说了。
应注意不要混淆逻辑求反运算符(!)以及按位反转运算符(~)。求反运算符对其操作数执行真/假求值,并返回表示真、假或未知结果的1位值。按位反转运算符对操作数的每一位(补码)执行逻辑反转,并返回与操作数相同位宽的值。
在某些操作中,这些操作的结果恰好相同,但在其他操作中,它们返回的值非常不同。当运算符与决策语句一起被错误使用时,这种差异可能导致错误代码。考虑下面的例子:
前面代码片段的最后两行之所以不同,是因为这两个运算符的工作方式不同——逻辑求反运算符(!)通过将两位相加或相减,对2位选择执行真/假计算,然后反转1位结果,按位反转运算符(~)只反转2位选择向量的每一位的值,并返回2位结果。if语句然后对2位向量进行真/假测试,该向量的计算结果为真,因为反转后的值仍有一位设置为1。
最佳实践指南5-1
使用按位反转运算符反转值的位,不要使用按位反转运算符对逻辑求反运算符求反。相反,使用逻辑求反运算符来否定真/假测试的结果。不要使用逻辑求反运算符反转值。
最佳实践指南5-2
仅使用逻辑求反运算符求反来测试标量(1位)值,而不是 对向量执行真/假测试。
如果向量的任何位为1,逻辑运算将返回true,这可能会导致在测试特定位时出现设计错误。计算向量值时,使用等式或关系运算符测试可接受的值。
module top_module( input logic [2:0] a, input logic [2:0] b, output logic [2:0] out_or_bitwise, output logic out_or_logical, output logic [5:0] out_not); assign out_or_bitwise = a | b ; assign out_or_logical = a || b; assign out_not = {~b , ~a}; endmodule 点击submit,等待一会就能看到下图结果:
注意图中的ref是参考波形,yours是你的代码生成的波形,网站会对比这两个波形,一旦这两者不匹配,仿真结果会变红(后面会展示)。
这一题就结束了。
problem 14-gates4 题目说明 分别构建一个 4 输入与门,或门以及异或门。
模块端口声明 module top_module( input [3:0] in, output out_and, output out_or, output out_xor); 题目解析 这一题的关键是多输入与、或以及异或门的实现,学过数电的化难度就不是很大了。
module top_module( input logic [3:0] in, output logic out_and, output logic out_or, output logic out_xor); assign out_and = &in ; assign out_or = |in ; assign out_xor = ^in ;endmodule 或者
module top_module( input logic [3:0] in, output logic out_and, output logic out_or, output logic out_xor); assign out_and = in[0] & in[1] & in[2] & in[3] ; assign out_or = in[0] | in[1] | in[2] | in[3] ; assign out_xor = in[0] ^ in[1] ^ in[2] ^ in[3] ; ;endmodule 第二种方式更易于理解,第一种方式就是上面的按位操作符,可以翻看上面的说明。
点击submit,等待一会就能看到下图结果:
注意图中的ref是参考波形,yours是你的代码生成的波形,网站会对比这两个波形,一旦这两者不匹配,仿真结果会变红。
总结 今天的5道题就结束了,整体难度不大,后面的题目难度会越来越大~
最后我这边做题的代码也是个人理解使用,有错误欢迎大家批评指正,祝大家学习愉快~
代码链接:
https://github.com/suisuisi/systemverilog/tree/main/systemveriloghdlbits
华为P10迫于压力一夜跌至最低价,网友大呼:值得入手
RFID在服装业中的应用
CV公司纷纷向科创板发起IPO冲锋号角
红外物理国家重点实验室在非平衡电子态操控领域取得突破
超声波液位计工作原理及常见误差校准
HDLBits: 在线学习SystemVerilog(三)-Problem 10-14
MAX3205E, MAX3207E, MAX3208E 高
大型充氮回流焊的性能优势
汽车可以上天了?广汽首款分离式飞行汽车顺利首飞,接连发布三大全新动力技术
PICOG2小怪兽VR一体机评测 显示效果还是比较满意的运行也很流畅
74hc164驱动数码管电路,8位LED显示器电路
英飞凌推出新一代多模HSPA+射频收发器SMARTiTM U
联发科向美国申请给华为供货
通过利用工业以太网实现袋式除尘器电气控制系统的设计
汽车仪表盘、数据图传、内网穿透 如何快速上手国产车规级开发板?
Altera Arria10 FPGA的约束性规格及会对电源产生什么影响
150多家企业“上缴”数据后,美商务部结论:未发现“炒芯”,无法解决供应瓶颈...
VR应用会是酒店经营业绩的强心剂吗
蓝牙技术与WIFI技术有什么异同点
中国正打造全球电力供应商的角色