Verilog有符号数处理

内容主要摘自如下两个连接:
如今FPGA编译器都支持verilog有符号运算的综合,而且综合后的有符号数都是以补码形式存在,明白点说,就是编译器能够自动把有
符号数编码成补码形式。具体在有符号数处理过程当中注意那些点,
 
一、有符号二进制加减法原理
有符号数一般以2的补码形式来表示。图1列出了4位二进制表示法所对应正负数。进一步观察,咱们发现两种类型数的加减法是同样的,
作加法和减法就是 在数轮上按正时钟转转或按反时钟转。比方说,1001+0100,意味着从1001按照顺时钟方向移动4个位置,其结果
为1101。在无符号数类型中,它 表明(+9)+(+4)=+13;而在有符号数类型中,它则表明(-7)+(+4)=-3。
从数轮上看,如果加法所得的结果溢出了,那么也就是穿越了数轮的临界点。注意这个临界点对于无符号数和有符号数来讲,是不同
的:无符号数,是介于1111和0000之间;有符号数,则是介于0111和1000之 间。
物理加减法的行为正好和数轮的移动相似。只要全部的运算子和结果具备相同的位宽,那么有符号数或无符号数的形式就可用于相同的
电路。比方说,设a、b和sum都是8位信号,表达式
1 sum = a+ b;
不管这些信号被转译成有符号数或无符号数,它都会引用相同的硬件且使用相同的二进制表示法。这种现象在其余算术运算中也是正确
的(可是它不可用于非算术运算中,比方说有理数运算或溢出标志位的生成)。
 
图1 4位二进制数轮
二、位扩展
此外,当运算子或其结果的位宽不一样时,咱们须要注意区分符号类型。由于不一样的符号类型须要扩展方式是不一样的。
(1)对于无符号数,前置一个0,即所谓的零扩展位;
(2)对于有符号数来讲,须要前置n个所谓的符号扩展位。
 
举个例子,设a和sum为8位信号,b为4位信号即b3b2b1b0。表达式
1 sum = a + b
须要将b扩展为8位。若是是无符号数形式,那么b扩展为0000_b3b2b1b0;若是是有符号数形式,那么b扩展为 b3b3b3b3_b3b2b1b0。
上述表达式所引用的硬件包括位宽扩展电路和加法器。由于对于有符号数和无符号数来讲,扩展电路是不一样的;因此对上面表达式,有
符号数和无符号数形式,要使用不一样的硬件实现。
 
三、Verilog-1995中的有符号数
在Verilog-1995中,只有integer数据类型被转移成有符号数,而reg和wire数据类型则被转移成无符号数。因为integer 类型有固
定的32位宽,所以它不太灵活。咱们一般使用手动加上扩展位来实现有符号数运算。下面的代码片断将描述有符号数和无符号数的运算:
01 reg [7:0] a, b;
02 reg [3:0] c,
03 reg [7:0] sum1, sum2, sum3, sum4;
04 . . .
05 // same width. can be applied to signed and unsigned
06 sum1 = a + b;
07 // 寄存器c高位自动扩展0
08 sum2 = a + c;
09 // 高位手动扩展0
10 sum3 = a + {4{ 1'b0 }, c};
11 // 高位手动扩展符号位
12 sum4 = a + {4{c[3]}, c};
在第一条语句中,a、b和sum1有相同的位宽,所以不管是转译成有符号数仍是无符号数,它都将引用相同的加法器电路。
在第二条语句中,c的位宽仅为4,在加法运算中,它的位宽会被调整。由于reg类型被做为无符号数看待,因此c的前面会被自动置入0扩展位。
在第三条语句中,咱们给c手动前置4个0,以实现和第二个表达式同样的效果。
在第四条语句中,咱们须要把变量转译成有符号数。为了实现所需的行为,c必须扩展符号位到8位。没有其余的办法,只好手动扩展。
在代码中,咱们重复复制c的最高位4次(4{c[3]})来建立具备扩展符号位的8位数。
四、Verilog-2001中的有符号数
在Verilog-2001中,有符号形式也被扩展到reg和wire数据类型中。在定义时加一个关键字signed,能够按照下面的方式定义:
   input  signed [ 7:0] a, b;
   output  signed [15:0] c;
   wire signed [15:0] x;
   reg signed [15:0] y;
使用有符号数据类型, 第2节所述代码能够被改写为:
1 reg signed [7:0] a, b;
2 reg signed [3:0] c;
3 reg signed [7:0] sum1, sum4;
reg signed [10:0] sum5;
4 . . .
5 // same width. can be applied to signed and unsigned
6 sum1 = a + b;
7 // automatic sign extension
8 sum4 = a + c;
sum5 = a + c;
第一条语句将引用一个常规的加法器,由于a、b和sum1具备相同的位宽。
第二条语句,全部的右手边变量都具备signed数据类型,c被自动扩展符号位到8位。所以,无需再手动添加符号位,即便sum5位
宽与a和c都不同也是会自动扩展到11位的。
五、有符号数与无符号数据混合使用
在小型的数字系统中,咱们一般能够选用有符号数或者无符号数。然而,在一些大型的系统中,会包括不一样形式的子系统。Verilog
是一种弱类型语 言,无符合变量和有符号变量能够在同一表达式中混用。根据Verilog的标准,只有当全部右手边的变量具备
signed数据类型属性的时候,扩展符号位才被执行。不然,全部的变量都只扩展0。考虑下面的代码片断:
1 reg signed [7:0] a, sum;
2 reg signed [3:0] b;
3 reg [3:0] c;
4 . . .
5 sum = a + b + c;
因为c不具备signed数据类型属性,所以右手边的变量b和c的扩展位为0。
Verilog有两个系统函数,$signed和$unsigned(),用以将括号内的表达式转换为signed和unsigned数据类型。比方说,
咱们能够转换c的数据类型,
1 sum = a + b + $signed(c);
如今,右手边的全部变量都具备signed数据类型属性,所以b和c将扩展符号位。
在复杂的表达式中,混用signed和unsigned数据类型将引入一些微妙的错误,所以应当避免混用。若是真的颇有必要,那么
表达式须要保持简单,同时通用转换函数,以确保数据类型的一致性。
须要补充的是$signed和$unsigned是能够综合的。
 
五、对于有符号的移位运算“>>”,高位是补0的
 
例子:对输入a,b取平均值,而后赋值给c输出
     always @(posedge clk)
        c<=(a+b)>>1;
1.a,b均为无符号数,结果正确
2.a,b中一个为有符号数(a),另外一个为无符号数(b),编译器会自动将无符号数(b)转换成有符号数,这样就成了2个有符号数
之间的运算了,结果是个有符号数,此时
1>若是a+b结果为正数(最高位为0),那么结果正确
2>若是a+b结果为负数(最高位为1),那么结果错误,由于移位运算新移入的位将用0来填补,此时负数将变为正数,显然错误。
 
因此综上,在进行有符号数运算的时候,最好像以下这样写:
reg signed [3:0] a;
reg signed [3:0] b;
reg signed [3:0] c;
 always @(posedge clk)
c<=(a+b)/2;
这样就能够避免没必要要的错误。
相关文章
相关标签/搜索