在verilog编程中,常数与寄存器变量的乘法综合出来的电路不同于寄存器变量乘以寄存器变量的综合电路。知乎里的解释非常好https://www.zhihu.com/question/45554104,总结乘法器模块的实现https://blog.csdn.net/yf210yf/article/details/70156855

乘法的实现是移位求和的过程

乘法器模块的实现主要有以下三种方法

1.串行实现方法

占用资源最多,需要的时钟频率高些,但是数据吞吐量却不大

两个N位二进制数x、y的乘积用简单的方法计算就是利用移位操作来实现。

其框图如下:

其状态图如下:

代码:

module multi_CX(clk, x, y, result);

    input clk;

    input [:] x, y;

    output [:] result;

    reg [:] result;

    parameter s0 = , s1 = , s2 = ;

    reg [:] count = ;

    reg [:] state = ;

    reg [:] P, T;

    reg [:] y_reg;

    always @(posedge clk) begin

        case (state)

            s0: begin

                count <= ;

                P <= ;

                y_reg <= y;

                T <= {{{'b0}}, x};

                state <= s1;

            end

            s1: begin

                if(count == 'b111)

                    state <= s2;

                else begin

                    if(y_reg[] == 'b1)

                        P <= P + T;

                    else

                        P <= P;

                    y_reg <= y_reg >> ;

                    T <= T << ;

                    count <= count + ;

                    state <= s1;

                end

            end

            s2: begin

                result <= P;

                state <= s0;

            end

            default: ;

        endcase

    end

endmodule

慢速信号处理中常用到的。

2.并行流水线实现方法

将操作数的N位并行提交给乘法器,这种方法并不是最优的实现架构,在FPGA中进位的速度远大于加法的速度,因此将相临的寄存器相加,相当于一个二叉树的结构,实际上对于n位的乘法处理,需要logn级流水来实现。

一个8位乘法器,其原理图如下图所示:

其实现的代码如下:

module multi_4bits_pipelining(mul_a, mul_b, clk, rst_n, mul_out);
input [:] mul_a, mul_b;
input clk;
input rst_n;
output [:] mul_out; reg [:] mul_out;
reg [:] stored0;
reg [:] stored1;
reg [:] stored2;
reg [:] stored3;
reg [:] stored4;
reg [:] stored5;
reg [:] stored6;
reg [:] stored7; reg [:] mul_out01;
reg [:] mul_out23; reg [:] add01;
reg [:] add23;
reg [:] add45;
reg [:] add67; always @(posedge clk or negedge rst_n) begin
if(!rst_n) begin
mul_out <= ;
stored0 <= ;
stored1 <= ;
stored2 <= ;
stored3 <= ;
stored4 <= ;
stored5 <= ;
stored6 <= ;
stored7 <= ; add01 <= ;
add23 <= ;
add45 <= ;
add67 <= ;
end
else begin
stored0 <= mul_b[]? {'b0, mul_a} : 16'b0;
stored1 <= mul_b[]? {'b0, mul_a, 1'b0} : 'b0;
stored2 <= mul_b[]? {'b0, mul_a, 2'b0} : 'b0;
stored3 <= mul_b[]? {'b0, mul_a, 3'b0} : 'b0;
stored4 <= mul_b[]? {'b0, mul_a, 4'b0} : 'b0;
stored5 <= mul_b[]? {'b0, mul_a, 5'b0} : 'b0;
stored6 <= mul_b[]? {'b0, mul_a, 6'b0} : 'b0;
stored7 <= mul_b[]? {'b0, mul_a, 7'b0} : 'b0;
add01 <= stored1 + stored0;
add23 <= stored3 + stored2;
add45 <= stored5 + stored4;
add67 <= stored7 + stored6; mul_out01 <= add01 + add23;
mul_out23 <= add45 + add67; mul_out <= mul_out01 + mul_out23; end
end
endmodule

流水线乘法器比串行乘法器的速度快很多很多,在非高速的信号处理中有广泛的应用。至于高速信号的乘法一般需要利用FPGA芯片中内嵌的硬核DSP单元来实现。

3.booth算法

看了原文献,有基2和基4两种实现

最常用的主要还是基2实现也就是用被除数的每两位做编码,Booth算法对乘数从低位开始判断,根据两个数据位的情况决定进行加法减法还是仅仅移位操作。判断的两个数据位为当前位及其右边的位(初始时需要增加一个辅助位0),移位操作是向右移动。

在Booth算法中,操作的方式取决于表达式(y【i+1】-y【i】)的值,这个表达式的值所代表的操作为:
0 无操作
+1 加x
-1 减x
Booth算法操作表示
yi yi+1 操作 说明
0 0 无 处于0串中,不需要操作
0 1 加x 1串的结尾
1 0 减x 1串的开始
1 1 无 处于1串中,不需要操作
乘法过程中,被乘数相对于乘积的左移操作可表示为乘以2,每次循环中的运算可表示为对于x(y(i+1)-yi)2^31-i项的加法运算(i=3l,30,…,1,0)。这样,Booth算法所计算的结果 可表示为:
x×(0-y31)×2^0
+x×(y31-y30)×2^1
+x×(y30-y29)×2^2
+x×(y1-y0)×2^31 [1] 
=x×(-y0×2^31 +y1×2^30 +y2×2^29+.......+y31×2^0)
=x×y

代码

module booth( start_sig, a, b, done_sig , product)
wire [:] start_sig;
wire [:] a;
wire [:] b;
wire [:] product;
/********************************/
reg[:] i;
reg[:] ra;
reg[:] rs;
reg[:] rp;
reg[:] x;
reg isdone;
always @(posedge clk or negedge rst_n)
if(!rst_n)
begin
i<='d0;
ra<='d0;
rs<='d0;
rp<='d0;
x<='d0;
isdone<='b0;
end
else if(start_sig)
case(i)
:
begin
ra<=a;
rs<=(~a+);
rp<={'d0,b,1'b0};
i<=i+;
end
:
if(x==)
begin
x<='d0;
i<=i+;
end
else if(rp[:]=='b01)
begin
rp<={rp[:]+ra,rp[:]};
i<=i+;
end
else if(rp[:]=='b10)
begin
rp<={rp[:]+rs,rp[:]};
i<=i+;
end
else
i<=i+;
:
begin
rp<={rp[],rp[:]};
x<=x+;//返回去检测对应寄存器值的方法
i<=i-;
end
:
begin
isdone<='b1;
i<=i+;
end
:
begin
isdone<='b0;
i<='d0;
end
endcase
assign product=rp[:];
assign done_sig=isdone;
endmodule
例1.38 设被乘数M=0111(7),乘数Q=1101(-3),相乘过程如下:A为开始补加的零,
A         Q Q-1
0000 1101 0 初始值
1001 1101 0 A=A-M=0000-0111=1001
1100 11 1 右移(第1次循环)/2
0011 1110 1 A=A + M=1100+0111=0011
0001 1111 0 右移(第2次循环)/2
1010 1111 0 A=A-M=0001-0111=1010
1101 0111 1 右移(第3次循环)/2
1110 1011 1 右移(第4次循环)/2
乘积=11101011=(-21)(十进制)
其中的移位是算数移位.
被乘数n位要移位3次.

verilog乘法器的设计的更多相关文章

  1. Verilog学习笔记设计和验证篇(三)...............同步有限状态机的指导原则

    因为大多数的FPGA内部的触发器数目相当多,又加上独热码状态机(one hot code machine)的译码逻辑最为简单,所以在FPGA实现状态机时,往往采用独热码状态机(即每个状态只有一个寄存器 ...

  2. Verilog乘法器

    乘法器,不能用乘号直接表示,略坑呀 坑归坑,做还是要做的 思路:首先乘法分为有符号乘与无符号乘,所以建立两个module分别运算有符号与无符号.然后在总module中用case语句判断输出应赋的值. ...

  3. verilog分频模块设计

    verilog设计: 分频器的设计: 分频器就是将一个时钟源的频率降低的过程(可以通过观察分频之后周期中包含几个原时钟周期来看是几分频),分频分为基数分频也分为偶数分频, 偶数分频的代码如下:(其中就 ...

  4. 8421BCD转余3码Verilog HDL的设计(1)

    近期阅读Verilog HDL高级数字设计(第二版)中,遇到了串行比特流BCD码转余3码转换器的设计,比较独特的是: (1)该转换器的输入为1位串行比特流,输出也为1位串行比特流. BCD码与余三码的 ...

  5. Verilog学习笔记设计和验证篇(五)...............层次化事件队列

    详细的了解层次化事件队列有助于理解Verilog的阻塞赋值和非阻塞赋值功能.所谓层次化事件队列指的是用于调度仿真时间的不同Verilog事件队列.在IEEE的5.3节中定义了层次化事件队列在逻辑上分为 ...

  6. Verilog学习笔记设计和验证篇(一)...............总线和流水线

    总线 总线是运算部件之间数据流通的公共通道.在硬线逻辑构成的运算电路中只要电路的规模允许可以比较自由的确定总线的位宽,从而大大的提高数据流通的速度.各个运算部件和数据寄存器组可以通过带有控制端的三态门 ...

  7. Verilog学习笔记设计和验证篇(二)...............同步有限状态机

    上图表示的就是数字电路设计中常用的时钟同步状态机的结构.其中共有四个部分产生下一状态的组合逻辑F.状态寄存器组.输出组合逻辑G.流水线输出寄存器组.如果状态寄存器组由n个寄存器组成,就可以记忆2^n个 ...

  8. 转载Verilog乘法器

    1. 串行乘法器 两个N位二进制数x.y的乘积用简单的方法计算就是利用移位操作来实现. module multi_CX(clk, x, y, result); input clk; input [7: ...

  9. Verilog分频器的设计

    大三都要结束了,才发现自己太多东西没深入学习. 对于偶分频:(计数到分频数的一半就翻转) 注: 图中只用了一个计数器,当然也可以用多个: 图中只计数到需要分频的一半,当然也可计数到更多: 图中从第一个 ...

随机推荐

  1. tkinter对话框和窗体

    1.标准对话框(tkinter.messagebox) 常用标准对话框 tkinter.messagebox.askokcancel tkinter.messagebox.askquestiontki ...

  2. OpenMP笔记(六)

    OpenMP有三种常见的加锁操作: critical是OpenMP的指令,它规定其后的代码为临界块,任何时候只允许一个线程访问: omp_set_lock是OpenMP的库函数,要跟omp_unset ...

  3. 3分钟搞定高逼格PPT封底——简约型

    封底想要高逼格又简约? 发现了这五类,看完不会制作算我输.   一.纯文字 白色背景下,一段结束语,或提问或感谢.       重叠文字,看上去非常有创意. 没有操作难度,END放大字号,颜色设置为浅 ...

  4. php IP地址转换

    <?php $enip = ip2long('210.110.11.49); echo $enip."<br />";//-764540111 echo long ...

  5. 时间复杂度T(n)

    1:概念 T(n)被称为时间复杂度,一般为在某个算法中操作步骤的重复次数与问题规模n的关系,下面一一举例说明 2:具体说明 2.1:常数阶o(1) 无论代码有多少行,只要没有循环等复杂的结构,其算法时 ...

  6. [原]调试实战——程序CPU占用率飙升,你知道如何快速定位吗?

    原调试debugwindbghangprocess explorer 前言 如果我们自己的程序的CPU Usage(CPU占用率)飙升,并且居高不下,很有可能陷入了死循环.你知道怎么快速定位并解决吗? ...

  7. 基于redis实现锁控制

    多数据源 数据源1为锁控制,数据源2自定义,可用于存储. 锁:当出现并发的时候为了保证数据的一致性,不会出现并发问题,假设,用户1修改一条信息,用户2也同时修改,会按照顺序覆盖自修改的值,为了避免这种

  8. 使用PHANTOMJS对网页截屏

    PhantomJS 是一个基于 WebKit 的服务器端 JavaScript API.它全面支持web而不需浏览器支持,其快速,原生支持各种Web标准: DOM 处理, CSS 选择器, JSON, ...

  9. PTC【Creo OR Proe】添加参数的方法

    Dim model As IpfcModel Try model = CoCreoAsyncConnection.Session.CurrentModel If model Is Nothing Th ...

  10. MACOSX下查看某个端口被哪个程序占用及杀进程方法

    sudo lsof -i :9000 COMMAND PID USER FD TYPE DEVICE SIZE/OFF NODE NAME java 61342 a 313u IPv6 0x11111 ...