几种常见加法器的verilog实现

                                              **几种常见加法器的verilog实现**

本人刚开始研一的学习,想着开个博客把研究生学习的内容记录一下,也养成一个好的习惯。研究生的方向是数字IC,把最近做的几个加法器作业记录一下。
一:16位进位跳跃加法器。
利用门级实现的加法器电路,通过进位传递函数Pi进行进位选择,可以提前进位的计算,使得后一级计算可以提前,加快计算的过程。这个电路由一位全加器,级联成四位加法器,再级联成16位加法器,具体电路如下图:

几种常见加法器的verilog实现_第1张图片
其中Pi = ai ⊕ bi,Pi:j = Pi ·Pi-1…Pj,代码会附上。

二:16位进位选择加法器
利用门级实现,实际上是加法器进行两路计算,一路是进位为1的计算,一路是进位为0的计算,提前计算好,再根据进位去选择用哪一路的值,相当于并行执行计算过程,缩短了等待计算的时间,代码见后几种常见加法器的verilog实现_第2张图片

三:16位并行前缀加法器
利用门级实现,并行的意思相当于在知道加数的前提下经过相关运算,可以提前计算出每一位的进位是多少,这样便可以很大程度节省计算等待的过程,相当于每一位的计算都是并行执行的,但这个电路的缺点是不能级联,写起来相对复杂。几种常见加法器的verilog实现_第3张图片
几种常见加法器的verilog实现_第4张图片
本人写这个加法器时先定义好黑块,灰色块和缓冲器块,在一行一行书写代码,代码看起来虽多,其实都是一些模块的调用。

总结
这三个模块经过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 

你可能感兴趣的:(数字ic)