⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 or1200_mult_mac.v

📁 一个开放的risc
💻 V
字号:
//////////////////////////////////////////////////////////////////////////                                                              ////////  OR1200's Top level multiplier and MAC                       ////////                                                              ////////  This file is part of the OpenRISC 1200 project              ////////  http://www.opencores.org/cores/or1k/                        ////////                                                              ////////  Description                                                 ////////  Multiplier is 32x32 however multiply instructions only      ////////  use lower 32 bits of the result. MAC is 32x32=64+64.        ////////                                                              ////////  To Do:                                                      ////////   - make signed division better, w/o negating the operands   ////////                                                              ////////  Author(s):                                                  ////////      - Damjan Lampret, lampret@opencores.org                 ////////                                                              //////////////////////////////////////////////////////////////////////////////                                                              //////// Copyright (C) 2000 Authors and OPENCORES.ORG                 ////////                                                              //////// This source file may be used and distributed without         //////// restriction provided that this copyright statement is not    //////// removed from the file and that any derivative work contains  //////// the original copyright notice and the associated disclaimer. ////////                                                              //////// This source file is free software; you can redistribute it   //////// and/or modify it under the terms of the GNU Lesser General   //////// Public License as published by the Free Software Foundation; //////// either version 2.1 of the License, or (at your option) any   //////// later version.                                               ////////                                                              //////// This source is distributed in the hope that it will be       //////// useful, but WITHOUT ANY WARRANTY; without even the implied   //////// warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR      //////// PURPOSE.  See the GNU Lesser General Public License for more //////// details.                                                     ////////                                                              //////// You should have received a copy of the GNU Lesser General    //////// Public License along with this source; if not, download it   //////// from http://www.opencores.org/lgpl.shtml                     ////////                                                              ////////////////////////////////////////////////////////////////////////////// CVS Revision History//// $Log: or1200_mult_mac.v,v $// Revision 1.3  2003/04/24 00:16:07  lampret// No functional changes. Added defines to disable implementation of multiplier/MAC//// Revision 1.2  2002/09/08 05:52:16  lampret// Added optional l.div/l.divu insns. By default they are disabled.//// Revision 1.1  2002/01/03 08:16:15  lampret// New prefixes for RTL files, prefixed module names. Updated cache controllers and MMUs.//// Revision 1.3  2001/10/21 17:57:16  lampret// Removed params from generic_XX.v. Added translate_off/on in sprs.v and id.v. Removed spr_addr from dc.v and ic.v. Fixed CR+LF.//// Revision 1.2  2001/10/14 13:12:09  lampret// MP3 version.//// Revision 1.1.1.1  2001/10/06 10:18:38  igorm// no message////// synopsys translate_off`include "timescale.v"// synopsys translate_on`include "or1200_defines.v"module or1200_mult_mac(	// Clock and reset	clk, rst,	// Multiplier/MAC interface	ex_freeze, id_macrc_op, macrc_op, a, b, mac_op, alu_op, result, mac_stall_r,	// SPR interface	spr_cs, spr_write, spr_addr, spr_dat_i, spr_dat_o);parameter width = `OR1200_OPERAND_WIDTH;//// I/O////// Clock and reset//input				clk;input				rst;//// Multiplier/MAC interface//input				ex_freeze;input				id_macrc_op;input				macrc_op;input	[width-1:0]		a;input	[width-1:0]		b;input	[`OR1200_MACOP_WIDTH-1:0]	mac_op;input	[`OR1200_ALUOP_WIDTH-1:0]	alu_op;output	[width-1:0]		result;output				mac_stall_r;//// SPR interface//input				spr_cs;input				spr_write;input	[31:0]			spr_addr;input	[31:0]			spr_dat_i;output	[31:0]			spr_dat_o;//// Internal wires and regs//`ifdef OR1200_MULT_IMPLEMENTEDreg	[width-1:0]		result;reg	[2*width-1:0]		mul_prod_r;`elsewire	[width-1:0]		result;wire	[2*width-1:0]		mul_prod_r;`endifwire	[2*width-1:0]		mul_prod;wire	[`OR1200_MACOP_WIDTH-1:0]	mac_op;`ifdef OR1200_MAC_IMPLEMENTEDreg	[`OR1200_MACOP_WIDTH-1:0]	mac_op_r1;reg	[`OR1200_MACOP_WIDTH-1:0]	mac_op_r2;reg	[`OR1200_MACOP_WIDTH-1:0]	mac_op_r3;reg				mac_stall_r;reg	[2*width-1:0]		mac_r;`elsewire	[`OR1200_MACOP_WIDTH-1:0]	mac_op_r1;wire	[`OR1200_MACOP_WIDTH-1:0]	mac_op_r2;wire	[`OR1200_MACOP_WIDTH-1:0]	mac_op_r3;wire				mac_stall_r;wire	[2*width-1:0]		mac_r;`endifwire	[width-1:0]		x;wire	[width-1:0]		y;wire				spr_maclo_we;wire				spr_machi_we;wire				alu_op_div_divu;wire				alu_op_div;reg				div_free;`ifdef OR1200_IMPL_DIVwire	[width-1:0]		div_tmp;reg	[5:0]			div_cntr;`endif//// Combinatorial logic//`ifdef OR1200_MAC_IMPLEMENTEDassign spr_maclo_we = spr_cs & spr_write & spr_addr[`OR1200_MAC_ADDR];assign spr_machi_we = spr_cs & spr_write & !spr_addr[`OR1200_MAC_ADDR];assign spr_dat_o = spr_addr[`OR1200_MAC_ADDR] ? mac_r[31:0] : mac_r[63:32];`elseassign spr_maclo_we = 1'b0;assign spr_machi_we = 1'b0;assign spr_dat_o = 32'h0000_0000;`endif`ifdef OR1200_LOWPWR_MULTassign x = (alu_op_div & a[31]) ? ~a + 1'b1 : alu_op_div_divu | (alu_op == `OR1200_ALUOP_MUL) | (|mac_op) ? a : 32'h0000_0000;assign y = (alu_op_div & b[31]) ? ~b + 1'b1 : alu_op_div_divu | (alu_op == `OR1200_ALUOP_MUL) | (|mac_op) ? b : 32'h0000_0000;`elseassign x = alu_op_div & a[31] ? ~a + 1'b1 : a;assign y = alu_op_div & b[31] ? ~b + 1'b1 : b;`endif`ifdef OR1200_IMPL_DIVassign alu_op_div = (alu_op == `OR1200_ALUOP_DIV);assign alu_op_div_divu = alu_op_div | (alu_op == `OR1200_ALUOP_DIVU);assign div_tmp = mul_prod_r[63:32] - y;`elseassign alu_op_div = 1'b0;assign alu_op_div_divu = 1'b0;`endif`ifdef OR1200_MULT_IMPLEMENTED//// Select result of current ALU operation to be forwarded// to next instruction and to WB stage//always @(alu_op or mul_prod_r or mac_r or a or b)	casex(alu_op)	// synopsys parallel_case`ifdef OR1200_IMPL_DIV		`OR1200_ALUOP_DIV:			result = a[31] ^ b[31] ? ~mul_prod_r[31:0] + 1'b1 : mul_prod_r[31:0];		`OR1200_ALUOP_DIVU,`endif		`OR1200_ALUOP_MUL: begin			result = mul_prod_r[31:0];		end		default:			result = mac_r[59:28];	endcase//// Instantiation of the multiplier//`ifdef OR1200_ASIC_MULTP2_32X32or1200_amultp2_32x32 or1200_amultp2_32x32(	.X(x),	.Y(y),	.RST(rst),	.CLK(clk),	.P(mul_prod));`else // OR1200_ASIC_MULTP2_32X32or1200_gmultp2_32x32 or1200_gmultp2_32x32(	.X(x),	.Y(y),	.RST(rst),	.CLK(clk),	.P(mul_prod));`endif // OR1200_ASIC_MULTP2_32X32//// Registered output from the multiplier and// an optional divider//always @(posedge rst or posedge clk)	if (rst) begin		mul_prod_r <= #1 64'h0000_0000_0000_0000;		div_free <= #1 1'b1;`ifdef OR1200_IMPL_DIV		div_cntr <= #1 6'b00_0000;`endif	end`ifdef OR1200_IMPL_DIV	else if (|div_cntr) begin		if (div_tmp[31])			mul_prod_r <= #1 {mul_prod_r[62:0], 1'b0};		else			mul_prod_r <= #1 {div_tmp[30:0], mul_prod_r[31:0], 1'b1};		div_cntr <= #1 div_cntr - 1'b1;	end	else if (alu_op_div_divu && div_free) begin		mul_prod_r <= #1 {31'b0, x[31:0], 1'b0};		div_cntr <= #1 6'b10_0000;		div_free <= #1 1'b0;	end`endif // OR1200_IMPL_DIV	else if (div_free | !ex_freeze) begin		mul_prod_r <= #1 mul_prod[63:0];		div_free <= #1 1'b1;	end`else // OR1200_MULT_IMPLEMENTEDassign result = {width{1'b0}};assign mul_prod = {2*width{1'b0}};assign mul_prod_r = {2*width{1'b0}};`endif // OR1200_MULT_IMPLEMENTED`ifdef OR1200_MAC_IMPLEMENTED//// Propagation of l.mac opcode//always @(posedge clk or posedge rst)	if (rst)		mac_op_r1 <= #1 `OR1200_MACOP_WIDTH'b0;	else		mac_op_r1 <= #1 mac_op;//// Propagation of l.mac opcode//always @(posedge clk or posedge rst)	if (rst)		mac_op_r2 <= #1 `OR1200_MACOP_WIDTH'b0;	else		mac_op_r2 <= #1 mac_op_r1;//// Propagation of l.mac opcode//always @(posedge clk or posedge rst)	if (rst)		mac_op_r3 <= #1 `OR1200_MACOP_WIDTH'b0;	else		mac_op_r3 <= #1 mac_op_r2;//// Implementation of MAC//always @(posedge rst or posedge clk)	if (rst)		mac_r <= #1 64'h0000_0000_0000_0000;`ifdef OR1200_MAC_SPR_WE	else if (spr_maclo_we)		mac_r[31:0] <= #1 spr_dat_i;	else if (spr_machi_we)		mac_r[63:32] <= #1 spr_dat_i;`endif	else if (mac_op_r3 == `OR1200_MACOP_MAC)		mac_r <= #1 mac_r + mul_prod_r;	else if (mac_op_r3 == `OR1200_MACOP_MSB)		mac_r <= #1 mac_r - mul_prod_r;	else if (macrc_op & !ex_freeze)		mac_r <= #1 64'h0000_0000_0000_0000;//// Stall CPU if l.macrc is in ID and MAC still has to process l.mac instructions// in EX stage (e.g. inside multiplier)// This stall signal is also used by the divider.//always @(posedge rst or posedge clk)	if (rst)		mac_stall_r <= #1 1'b0;	else		mac_stall_r <= #1 (|mac_op | (|mac_op_r1) | (|mac_op_r2)) & id_macrc_op`ifdef OR1200_IMPL_DIV				| (|div_cntr)`endif				;`else // OR1200_MAC_IMPLEMENTEDassign mac_stall_r = 1'b0;assign mac_r = {2*width{1'b0}};assign mac_op_r1 = `OR1200_MACOP_WIDTH'b0;assign mac_op_r2 = `OR1200_MACOP_WIDTH'b0;assign mac_op_r3 = `OR1200_MACOP_WIDTH'b0;`endif // OR1200_MAC_IMPLEMENTEDendmodule

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -