**几种常见加法器的verilog实现**
本人刚开始研一的学习,想着开个博客把研究生学习的内容记录一下,也养成一个好的习惯。研究生的方向是数字IC,把最近做的几个加法器作业记录一下。
一:16位进位跳跃加法器。
利用门级实现的加法器电路,通过进位传递函数Pi进行进位选择,可以提前进位的计算,使得后一级计算可以提前,加快计算的过程。这个电路由一位全加器,级联成四位加法器,再级联成16位加法器,具体电路如下图:
其中Pi = ai ⊕ bi,Pi:j = Pi ·Pi-1…Pj,代码会附上。
二:16位进位选择加法器
利用门级实现,实际上是加法器进行两路计算,一路是进位为1的计算,一路是进位为0的计算,提前计算好,再根据进位去选择用哪一路的值,相当于并行执行计算过程,缩短了等待计算的时间,代码见后
三:16位并行前缀加法器
利用门级实现,并行的意思相当于在知道加数的前提下经过相关运算,可以提前计算出每一位的进位是多少,这样便可以很大程度节省计算等待的过程,相当于每一位的计算都是并行执行的,但这个电路的缺点是不能级联,写起来相对复杂。
本人写这个加法器时先定义好黑块,灰色块和缓冲器块,在一行一行书写代码,代码看起来虽多,其实都是一些模块的调用。
总结
这三个模块经过modelsim仿真,结果正确。
`//进位跳跃加法器设计
module carry_skip_adder16(a,b,Cin,sum,Cout);
input [16:1] a,b;
input Cin;
output [16:1] sum;
output Cout;
wire Cout1,Cout2,Cout3,c1,c2,c3,c4,p1,p2,p3,p4;
muxtwo m1(Cout1,Cin,c1,p1); //c1和cin选择一个作为下一级的进位
full_adder4 a1(a[4:1],b[4:1],Cin,sum[4:1],c1); //产生低四位和数和进位C1,
pi P1(a[4:1],b[4:1],p1); //产生进位选择信号p1,用来选择c1还是cin作为下一级的进位。
muxtwo m2(Cout2,Cout1,c2,p2);
full_adder4 a2(a[8:5],b[8:5],Cout1,sum[8:5],c2);
pi P2(a[8:5],b[8:5],p2);
muxtwo m3(Cout3,Cout2,c3,p3);
full_adder4 a3(a[12:9],b[12:9],Cout2,sum[12:9],c3);
pi P3(a[12:9],b[12:9],p3);
muxtwo m4(Cout,Cout3,c4,p4);
full_adder4 a4(a[16:13],b[16:13],Cout3,sum[16:13],c4);
pi P4(a[16:13],b[16:13],p4);
endmodule
//一位全加器
module full_adder(a,b,Cin,sum,Cout);
input a,b;
input Cin;
output sum;
output Cout;
wire t1,t2,t3;
wire s1;
xor (s1,a,b);
xor (sum,s1,Cin);
and (t3,a,b);
and (t2,b,Cin);
and (t1,a,Cin);
or (Cout,t1,t2,t3);
endmodule
//四位全加器设计
module full_adder4(a,b,Cin,sum,Cout);
input [4:1] a,b;
input Cin;
output [4:1]sum;
output Cout;
wire Cout1,Cout2,Cout3;
full_adder f1(a[1],b[1],Cin,sum[1],Cout1);
full_adder f2(a[2],b[2],Cout1,sum[2],Cout2);
full_adder f3(a[3],b[3],Cout2,sum[3],Cout3);
full_adder f4(a[4],b[4],Cout3,sum[4],Cout);
endmodule
//二选一多路选择器
module muxtwo(Cout,c0,c1,p);
input c0,c1,p;
output Cout;
wire np,sela,selb;
not (np,p);
and (sela,c1,np);
and (selb,c0,p);
or (Cout,sela,selb);
endmodule
//进位传递函数生成模块
module pi(a,b,p);
input [4:1] a,b;
output p;
wire p1,p2,p3,p4;
xor (p1,a[1],b[1]);
xor (p2,a[2],b[2]);
xor (p3,a[3],b[3]);
xor (p4,a[4],b[4]);
and (p,p1,p2,p3,p4);
endmodule
测试文件
`timescale 1ns/1ns
module carry_skip_adder16_tb;
reg [16:1] a,b;
reg Cin;
wire [16:1] sum;
wire Cout;
carry_skip_adder16 c1(
.a(a),
.b(b),
.Cin(Cin),
.sum(sum),
.Cout(Cout)
);
initial begin
Cin = 1;
a = 231;
b= 331;
#20;
a = 500;
b = 600;
#30;
a = 10;
b = 5;
#200;
$stop;
end
endmodule `
//16位进位选择加法器
module carry_select(a,b,Cin,sum,Cout);
input [16:1] a,b;
input Cin;
output [16:1] sum;
output Cout;
wire c4,c8,c12;
wire c8_0,c8_1,c12_0,c12_1,c16_0,c16_1;
wire [16:1]sum1,sum2;
full_adder4 f1(a[4:1],b[4:1],Cin,sum[4:1],c4);//第一级4位加法器
full_adder4 f2(a[8:5],b[8:5],1'b0,sum1[8:5],c8_0);//第二级进位为零加法器
full_adder4 f3(a[8:5],b[8:5],1'b1,sum2[8:5],c8_1);//第二级进位为1加法器
muxtwo m1(sum[8:5],sum1[8:5],sum2[8:5],c4);
carry_gene u1(c4,c8_0,c8_1,c8); //产生第二级进位
full_adder4 f4(a[12:9],b[12:9],1'b0,sum1[12:9],c12_0);//第三级进位为零加法器
full_adder4 f5(a[12:9],b[12:9],1'b1,sum2[12:9],c12_1);//第三级进位为1加法器
muxtwo m2(sum[12:9],sum1[12:9],sum2[12:9],c8);
carry_gene u2(c8,c12_0,c12_1,c12); //产生第三级进位
full_adder4 f6(a[16:13],b[16:13],1'b0,sum1[16:13],c16_0);//第三级进位为零加法器
full_adder4 f7(a[16:13],b[16:13],1'b1,sum2[16:13],c16_1);//第三级进位为1加法器
muxtwo m3(sum[16:13],sum1[16:13],sum2[16:13],c12);
carry_gene u3(c12,c16_0,c16_1,Cout); //产生第三级进位
endmodule
//二选一多路选择器
module muxtwo(sum,sum1,sum2,c);
input [4:1]sum1,sum2;
input c;
output [4:1] sum;
wire nc;
wire [4:1]sela,selb;
not (nc,c);
and (sela[1],sum1[1],nc);
and (sela[2],sum1[2],nc);
and (sela[3],sum1[3],nc);
and (sela[4],sum1[4],nc);
and (selb[1],sum2[1],c);
and (selb[2],sum2[2],c);
and (selb[3],sum2[3],c);
and (selb[4],sum2[4],c);
or (sum[1],sela[1],selb[1]);
or (sum[2],sela[2],selb[2]);
or (sum[3],sela[3],selb[3]);
or (sum[4],sela[4],selb[4]);
endmodule
//进位产生模块
module carry_gene(c0,c1,c2,Cout);
input c0,c1,c2;
output Cout;
wire m1;
and (m1,c2,c0);
or (Cout,c1,m1);
endmodule
//四位全加器设计
module full_adder4(a,b,Cin,sum,Cout);
input [4:1] a,b;
input Cin;
output [4:1]sum;
output Cout;
wire Cout1,Cout2,Cout3;
full_adder f1(a[1],b[1],Cin,sum[1],Cout1);
full_adder f2(a[2],b[2],Cout1,sum[2],Cout2);
full_adder f3(a[3],b[3],Cout2,sum[3],Cout3);
full_adder f4(a[4],b[4],Cout3,sum[4],Cout);
endmodule
//一位全加器
module full_adder(a,b,Cin,sum,Cout);
input a,b;
input Cin;
output sum;
output Cout;
wire t1,t2,t3;
wire s1;
xor (s1,a,b);
xor (sum,s1,Cin);
and (t3,a,b);
and (t2,b,Cin);
and (t1,a,Cin);
or (Cout,t1,t2,t3);
endmodule
//测试文件
`timescale 1ns/1ns
module carry_select_tb;
reg [16:1] a,b;
reg Cin;
wire [16:1] sum;
wire Cout;
carry_select c1(
.a(a),
.b(b),
.Cin(Cin),
.sum(sum),
.Cout(Cout)
);
initial begin
Cin = 1;
a = 2131;
b= 3351;
#20;
a = 50640;
b = 600463;
#30;
a = 3436310;
b = 523252;
#200;
$stop;
end
endmodule
//并行前缀加法器设计
module prefix_adder(a,b,sum,Cout);
input [15:0]a,b;
output [15:0]sum;
output Cout;
wire G0,G0F,G1,G2,G3,G4,G5,G6,G7,G8,G9,G10,G11,G12,G13,G14,G15,
P0,P1,P2,P3,P4,P5,P6,P7,P8,P9,P10,P11,P12,P13,P14,P15,
G1_0,G1_0F,G2_1,G3_2,G4_3,G5_4,G6_5,G7_6,G8_7,G9_8,G10_9,G11_10,G12_11,G13_12,G14_13,G15_14,
P0_0,P1_0,P2_1,P3_2,P4_3,P5_4,P6_5,P7_6,P8_7,P9_8,P10_9,P11_10,P12_11,P13_12,P14_13,P15_14,
G2_0,G2_0F,G3_0,G3_0F,G4_0,G4_0F,G5_0,G5_0F,G6_0,G6_0F,G7_0,G7_0F,
G4_1,G5_2,G6_3,G7_4,G8_5,G9_6,G10_7,G11_8,G12_9,G13_10,G14_11,G15_12,
P4_1,P5_2,P6_3,P7_4,P8_5,P9_6,P10_7,P11_8,P12_9,P13_10,P14_11,P15_12,
G8_1,G9_2,G10_3,G11_4,G12_5,G13_6,G14_7,G15_8,
P8_1,P9_2,P10_3,P11_4,P12_5,P13_6,P14_7,P15_8,
G8_0,G9_0,G10_0,G11_0,G12_0,G13_0,G14_0;
GiPi u0(a[0],b[0],G0F,P0);
GiPi u1(a[1],b[1],G1,P1);
GiPi u2(a[2],b[2],G2,P2);
GiPi u3(a[3],b[3],G3,P3);
GiPi u4(a[4],b[4],G4,P4);
GiPi u5(a[5],b[5],G5,P5);
GiPi u6(a[6],b[6],G6,P6);
GiPi u7(a[7],b[7],G7,P7);
GiPi u8(a[8],b[8],G8,P8);
GiPi u9(a[9],b[9],G9,P9);
GiPi u10(a[10],b[10],G10,P10);
GiPi u11(a[11],b[11],G11,P11);
GiPi u12(a[12],b[12],G12,P12);
GiPi u13(a[13],b[13],G13,P13);
GiPi u14(a[14],b[14],G14,P14);
GiPi u15(a[15],b[15],G15,P15);
BUFFER B1(G0F,G0);
gray_cell g1(G1,P1,G0F,G1_0F); //第一行
black_cell b1(G2,P2,G1,P1,G2_1,P2_1);
black_cell b2(G3,P3,G2,P2,G3_2,P3_2);
black_cell b3(G4,P4,G3,P3,G4_3,P4_3);
black_cell b4(G5,P5,G4,P4,G5_4,P5_4);
black_cell b5(G6,P6,G5,P5,G6_5,P6_5);
black_cell b6(G7,P7,G6,P6,G7_6,P7_6);
black_cell b7(G8,P8,G7,P7,G8_7,P8_7);
black_cell b8(G9,P9,G8,P8,G9_8,P9_8);
black_cell b9(G10,P10,G9,P9,G10_9,P10_9);
black_cell b10(G11,P11,G10,P10,G11_10,P11_10);
black_cell b11(G12,P12,G11,P11,G12_11,P12_11);
black_cell b12(G13,P13,G12,P12,G13_12,P13_12);
black_cell b13(G14,P14,G13,P13,G14_13,P14_13);
black_cell b14(G15,P15,G14,P14,G15_14,P15_14);
BUFFER B2(G1_0F,G1_0);
gray_cell g2(G2_1,P2_1,G0,G2_0F); //第二行
gray_cell g3(G3_2,P3_2,G1_0F,G3_0F);
black_cell b15(G4_3,P4_3,G2_1,P2_1,G4_1,P4_1);
black_cell b17(G5_4,P5_4,G3_2,P3_2,G5_2,P5_2);
black_cell b18(G6_5,P6_5,G4_3,P4_3,G6_3,P6_3);
black_cell b19(G7_6,P7_6,G5_4,P5_4,G7_4,P7_4);
black_cell b20(G8_7,P8_7,G6_5,P6_5,G8_5,P8_5);
black_cell b21(G9_8,P9_8,G7_6,P7_6,G9_6,P9_6);
black_cell b22(G10_9,P10_9,G8_7,P8_7,G10_7,P10_7);
black_cell b23(G11_10,P11_10,G9_8,P9_8,G11_8,P11_8);
black_cell b24(G12_11,P12_11,G10_9,P10_9,G12_9,P12_9);
black_cell b25(G13_12,P13_12,G11_10,P11_10,G13_10,P13_10);
black_cell b26(G14_13,P14_13,G12_11,P12_11,G14_11,P14_11);
black_cell b27(G15_14,P15_14,G13_12,P13_12,G15_12,P15_12);
BUFFER B3(G2_0F,G2_0); //第三行
BUFFER B4(G3_0F,G3_0);
gray_cell g4(G4_1,P4_1,G0,G4_0F);
gray_cell g5(G5_2,P5_2,G1_0,G5_0F);
gray_cell g6(G6_3,P6_3,G2_0,G6_0F);
gray_cell g7(G7_4,P7_4,G3_0,G7_0F);
black_cell b28(G8_5,P8_5,G4_1,P4_1,G8_1,P8_1);
black_cell b29(G9_6,P9_6,G5_2,P5_2,G9_2,P9_2);
black_cell b30(G10_7,P10_7,G6_3,P6_3,G10_3,P10_3);
black_cell b31(G11_8,P11_8,G7_4,P7_4,G11_4,P11_4);
black_cell b32(G12_9,P12_9,G8_5,P8_5,G12_5,P12_5);
black_cell b33(G13_10,P13_10,G9_6,P9_6,G13_6,P13_6);
black_cell b34(G14_11,P14_11,G10_7,P10_7,G14_7,P14_7);
black_cell b35(G15_12,P15_12,G11_8,P11_8,G15_8,P15_8);
BUFFER B5(G4_0F,G4_0);
BUFFER B6(G5_0F,G5_0);
BUFFER B7(G6_0F,G6_0);
BUFFER B8(G7_0F,G7_0);
gray_cell g(G8_1,P8_1,G0,G8_0);
gray_cell g8(G9_2,P9_2,G1_0,G9_0);
gray_cell g9(G10_3,P10_3,G2_0,G10_0);
gray_cell g10(G11_4,P11_4,G3_0,G11_0);
gray_cell g11(G12_5,P12_5,G4_0,G12_0);
gray_cell g12(G13_6,P13_6,G5_0,G13_0);
gray_cell g13(G14_7,P14_7,G6_0,G14_0);
gray_cell g14(G15_8,P15_8,G7_0,Cout);
xor (sum[0],P0,1'b0);
xor (sum[1],P1,G0);
xor (sum[2],P2,G1_0);
xor (sum[3],P3,G2_0);
xor (sum[4],P4,G3_0);
xor (sum[5],P5,G4_0);
xor (sum[6],P6,G5_0);
xor (sum[7],P7,G6_0);
xor (sum[8],P8,G7_0);
xor (sum[9],P9,G8_0);
xor (sum[10],P10,G9_0);
xor (sum[11],P11,G10_0);
xor (sum[12],P12,G11_0);
xor (sum[13],P13,G12_0);
xor (sum[14],P14,G13_0);
xor (sum[15],P15,G14_0);
endmodule
//black_cell运算块
module black_cell(Gik,Pik,Gk_1j,Pk_1j,Gij,Pij);
input Gik,Pik,Gk_1j,Pk_1j;
output Gij,Pij;
wire h;
and (h,Pik,Gk_1j);
and (Pij,Pik,Pk_1j);
or (Gij,Gik,h);
endmodule
//gray_cell运算块
module gray_cell(Gik,Pik,Gk_1j,Gij);
input Gik,Pik,Gk_1j;
output Gij;
wire h;
and (h,Pik,Gk_1j);
or (Gij,h,Gik);
endmodule
//buffer模块
module BUFFER(a,b);
input a;
output b;
buf (b,a);
endmodule
//gipi函数生成模块
module GiPi(ai,bi,Gi,Pi);
input ai,bi;
output Gi,Pi;
and (Gi,ai,bi);
xor (Pi,ai,bi);
endmodule
//测试文件代码
`timescale 1ns/1ns
module prefix_adder_tb;
reg [15:0] a,b;
wire [15:0] sum;
wire Cout;
prefix_adder c1(
.a(a),
.b(b),
.sum(sum),
.Cout(Cout)
);
initial begin
a = 231;
b= 331;
#20;
a = 500;
b = 600;
#30;
a = 10;
b = 5;
#200;
#30;
a = 16'b1111000011110000;
b = 16'b0000111100001111;
#200;
$stop;
end
endmodule