OpenCores
URL https://opencores.org/ocsvn/hive/hive/trunk

Subversion Repositories hive

[/] [hive/] [trunk/] [v04.05/] [hive_core.v] - Rev 8

Go to most recent revision | Compare with Previous | Blame | View Log

/*
--------------------------------------------------------------------------------
 
Module : hive_core.v
 
--------------------------------------------------------------------------------
 
Function:
- General purpose barrel processor FPGA core with:
  - 8 threads & 8 stage pipeline
  - 8 simple stacks per thread
  - 32 bit data
  - 16 bit opcode
  - 16 bit address
 
Instantiates (at this level):
- control_ring.v
- data_ring.v
- reg_set.v
- reg_mem_shim.v
- dp_ram_infer.v
 
--------------------
- Revision History -
--------------------
 
v04.05 - 2014-01-02
- Note main version jump.
- Branched design back into main line.
- OP_CODE_W is now MEM_DATA_W.
 
v01.04 - 2014-01-01
- Moved register set and main memory data port one pipeline stage later.
- op_pow is back and is now sign neutral.
- Op renaming: psh_iu => psu_i.
- `_0, `_1, etc. is now `s* in all boot code.
- Added header blocking statements to some *.h files (didn't work for all).
- Use of real rather than integer types in UART calculations for clarity.
- EP3C5E144C: 2662 LEs, 198MHz (w/o DSE, synthesis optimized for speed).
- Passes all boot code verification & functional tests.
 
v01.03 - 2013-12-23
- Short 4 bit (+7/-8) immediate (A?B) jumps have replaced all following jumps.
- Removed unconditional immediate jump (use op_jmp_ie).
- Immediate (A?0) jumps, data, and add IM value reduced to 6 bits.
- Removed op_pow_i, op_shl_iu is op_psh_iu: combo pow2 and right shift, 
  op_shl_u is strictly unsigned shift.
- UART added to register set.
- Op renaming:
    dat_f* => lit_*
    shl_iu => psh_iu
- Small changes to register set base component parameters & I/O.
- Opcode encode/decode now over full opcode width.
- EP3C5E144C: 2650 LEs, 189MHz (w/o DSE).
- Passes all boot code verification & functional tests.
 
v01.02 - 2013-12-06
- Following jumps (jmp_f) have replaced all skips.
- Added op_pow & op_pow_i opcodes.
- A odd testing removed (lack of opcode space).
- op_pop now covers all stacks at once via {pb, sb, pa, sa} binary field.
- op_reg_r now signed, added read and write high register ops (_wh is ~free).
- Added op_lit_u to accomodate 16 bit addresses & such.
- Op renaming:
    lit => dat_f
    byt => dat_i
    *_sx => *_xs
    *_ux => *_xu
- Moved PC interrupt & jmp_f address loads to stage 4 of PC pipe.
- op_dat_fh now uses A as source of low data rather than B,
  which is more consistent and allows unrelated pop.
- Register set addresses now defined as 8 bits wide.
- EP3C5E144C: ~2500 LEs, 185MHz (w/o DSE).
- Passes all boot code verification tests.
- Passes boot code functional tests: divide, sqrt, log2, exp2.
 
v01.01 - 2013-11-19
- Born.  Based on Hive v3.10.  Has 8 stacks per thread.
- Skips are back for A odd and (A?B) testing.
- Removed op_cls as it seems too dangerous.  May put back in register set.
- Lots of op renaming: 
    dat_f => lit
    dat_i => byt
    or_br => bro, etc.
    or => orr
    pc => pgc (to make all op bases 3 letters)
- Reg access now unsigned low with no immediate offset.
- Added register to flag decode output, moved all PC changes to stage 3.
- EP3C5E144C: ~2400 LEs, 178MHz (w/o DSE).
- BROKEN: reg_mem_shim.v has bad decoding for dmem addr.
 
--------------------------------------------------------------------------------
*/
 
module hive_core
	#(
	parameter	integer							CLK_HZ	 		= 160000000,	// master clk_i rate (Hz)
	parameter	integer							DATA_W			= 32,		// data width
	parameter	integer							THREADS			= 8,		// threads (don't change!)
	parameter	[DATA_W/4-1:0]					VER_MAJ			= 'h04,	// core version
	parameter	[DATA_W/4-1:0]					VER_MIN			= 'h05	// core version
	)
	(
	// clocks & resets
	input			wire								clk_i,						// clock
	input			wire								rst_i,						// async. reset, active high
	//
	input			wire	[THREADS-1:0]			intr_req_i,					// event request, active high
	//
	input			wire	[DATA_W-1:0]			io_i,							// gpio
	output		wire	[DATA_W-1:0]			io_o,
	//
	input			wire								uart_rx_i,					// serial data
	output		wire								uart_tx_o					// serial data
	);
 
 
	/*
	----------------------
	-- internal signals --
	----------------------
	*/
	`include "functions.h"  // for clog2()
	//
	localparam	integer							ADDR_W			= DATA_W/2;	// address width
	localparam	integer							PNTR_W			= 5;		// stack pointer width
	localparam	integer							MEM_ADDR_W		= 13;		// main memory address width
	localparam	integer							MEM_DATA_W		= DATA_W/2;		// main memory data width
	localparam	[ADDR_W-1:0]					CLR_BASE			= 'h0;	// clear address base (concat)
	localparam	integer							CLR_SPAN			= 2;		// clear address span (2^n)
	localparam	[ADDR_W-1:0]					INTR_BASE		= 'h20;	// interrupt address base (concat)
	localparam	integer							INTR_SPAN		= 2;		// interrupt address span (2^n)
	localparam	integer							THRD_W			= clog2( THREADS );
	localparam	integer							STACKS			= 8;		// number of stacks
	localparam	integer							STK_W				= clog2( STACKS );
	localparam	integer							IM_W				= 6;		// immediate width
	localparam	integer							LG_SEL_W			= 4;
	localparam	integer							REG_ADDR_W		= 4;
	localparam	integer							DM_OFFS_W		= 4;
	localparam	integer							PROT_POP			= 1;		// 1=error protection, 0=none
	localparam	integer							PROT_PUSH		= 1;		// 1=error protection, 0=none
	localparam	integer							UART_DATA_W		= 8;		// uart data width (bits)
	localparam	integer							UART_BAUD_RATE	= 115200;	// uart baud rate (Hz)
	//
	wire					[THREADS-1:0]			clr_req;
	wire					[THREADS-1:0]			intr_en;
	wire					[MEM_DATA_W-1:0]		op_code;
	wire												op_code_er;
	wire					[STK_W-1:0]				data_sel_a, data_sel_b, addr_sel_b;
	wire												imda;
	wire												imad;
	wire												sgn, ext, hgh;
	wire												lg;
	wire					[LG_SEL_W-1:0]			lg_sel;
	wire												add, sub, mul, shl, pow, rtn;
	wire												stk_clr;
	wire					[STACKS-1:0]			pop, push, pop_er_2, push_er_3;
	wire					[DATA_W-1:0]			a_data;
	wire					[ADDR_W-1:0]			b_addr;
	wire					[IM_W-1:0]				im;
	wire												flg_od_2, flg_nz_2, flg_lz_2, flg_ne_2, flg_lt_2;
	wire					[THRD_W-1:0]			thrd_0, thrd_2, thrd_3, thrd_6;
	wire					[ADDR_W-1:0]			pc_1, pc_3, pc_4;
	wire					[MEM_DATA_W-1:0]		dm_rd_data_4, rg_rd_data_4;
	wire					[MEM_DATA_W-1:0]		wr_data_2;
	wire					[ADDR_W-1:0]			dm_addr_2, rg_addr_2;
	wire												dm_rd, dm_wr, rg_rd, rg_wr;
	wire												dm_wr_2, rg_rd_2, rg_wr_2;
	wire												lit;
 
 
	/*
	================
	== code start ==
	================
	*/
 
 
	// the control ring
	control_ring
	#(
	.DATA_W				( DATA_W ),
	.ADDR_W				( ADDR_W ),
	.THREADS				( THREADS ),
	.THRD_W				( THRD_W ),
	.STACKS				( STACKS ),
	.STK_W				( STK_W ),
	.IM_W					( IM_W ),
	.MEM_DATA_W			( MEM_DATA_W ),
	.LG_SEL_W			( LG_SEL_W ),
	.CLR_BASE			( CLR_BASE ),
	.CLR_SPAN			( CLR_SPAN ),
	.INTR_BASE			( INTR_BASE ),
	.INTR_SPAN			( INTR_SPAN )
	)
	control_ring
	(
	.clk_i				( clk_i ),
	.rst_i				( rst_i ),
	.clr_req_i			( clr_req ),
	.clr_ack_o			(  ),  // unused
	.intr_en_i			( intr_en ),
	.intr_req_i			( intr_req_i ),
	.intr_ack_o			(  ),  // unused
	.op_code_i			( op_code ),
	.op_code_er_o		( op_code_er ),
	.b_addr_i			( b_addr ),
	.im_o					( im ),
	.data_sel_a_o		( data_sel_a ),
	.data_sel_b_o		( data_sel_b ),
	.addr_sel_b_o		( addr_sel_b ),
	.imda_o				( imda ),
	.imad_o				( imad ),
	.sgn_o				( sgn ),
	.hgh_o				( hgh ),
	.ext_o				( ext ),
	.lg_sel_o			( lg_sel ),
	.add_o				( add ),
	.sub_o				( sub ),
	.mul_o				( mul ),
	.shl_o				( shl ),
	.pow_o				( pow ),
	.rtn_o				( rtn ),
	.lit_o				( lit ),
	.dm_rd_o				( dm_rd ),
	.dm_wr_o				( dm_wr ),
	.rg_rd_o				( rg_rd ),
	.rg_wr_o				( rg_wr ),
	.stk_clr_o			( stk_clr ),
	.pop_o				( pop ),
	.push_o				( push ),
	.flg_nz_2_i			( flg_nz_2 ),
	.flg_lz_2_i			( flg_lz_2 ),
	.flg_ne_2_i			( flg_ne_2 ),
	.flg_lt_2_i			( flg_lt_2 ),
	.thrd_0_o			( thrd_0 ),
	.thrd_2_o			( thrd_2 ),
	.thrd_3_o			( thrd_3 ),
	.thrd_6_o			( thrd_6 ),
	.pc_1_o				( pc_1 ),
	.pc_3_o				( pc_3 ),
	.pc_4_o				( pc_4 )
	);
 
 
	// the data ring
	data_ring
	#(
	.DATA_W				( DATA_W ),
	.ADDR_W				( ADDR_W ),
	.THREADS				( THREADS ),
	.THRD_W				( THRD_W ),
	.STACKS				( STACKS ),
	.STK_W				( STK_W ),
	.PNTR_W				( PNTR_W ),
	.IM_W					( IM_W ),
	.LG_SEL_W			( LG_SEL_W ),
	.PROT_POP			( PROT_POP ),
	.PROT_PUSH			( PROT_PUSH )
	)
	data_ring
	(
	.clk_i				( clk_i ),
	.rst_i				( rst_i ),
	.data_sel_a_i		( data_sel_a ),
	.data_sel_b_i		( data_sel_b ),
	.addr_sel_b_i		( addr_sel_b ),
	.imda_i				( imda ),
	.imad_i				( imad ),
	.sgn_i				( sgn ),
	.ext_i				( ext ),
	.hgh_i				( hgh ),
	.lg_sel_i			( lg_sel ),
	.add_i				( add ),
	.sub_i				( sub ),
	.mul_i				( mul ),
	.shl_i				( shl ),
	.pow_i				( pow ),
	.rtn_i				( rtn ),
	.dm_rd_i				( dm_rd ),
	.rg_rd_i				( rg_rd ),
	.stk_clr_i			( stk_clr ),
	.pop_i				( pop ),
	.push_i				( push ),
	.thrd_6_i			( thrd_6 ),
	.im_i					( im ),
	.dm_rd_data_4_i	( dm_rd_data_4 ),
	.rg_rd_data_4_i	( rg_rd_data_4 ),
	.pc_3_i				( pc_3 ),
	.a_data_o			( a_data ),
	.b_addr_o			( b_addr ),
	.flg_nz_2_o			( flg_nz_2 ),
	.flg_lz_2_o			( flg_lz_2 ),
	.flg_ne_2_o			( flg_ne_2 ),
	.flg_lt_2_o			( flg_lt_2 ),
	.pop_er_2_o			( pop_er_2 ),
	.push_er_3_o		( push_er_3 )
	);
 
 
	// shim for memory and register set access
	reg_mem_shim
	#(
	.DATA_W				( DATA_W ),
	.ADDR_W				( ADDR_W ),
	.IM_W					( DM_OFFS_W )
	)
	reg_mem_shim
	(
	.clk_i				( clk_i ),
	.rst_i				( rst_i ),
	.hgh_i				( hgh ),
	.lit_i				( lit ),
	.dm_wr_i				( dm_wr ),
	.rg_rd_i				( rg_rd ),
	.rg_wr_i				( rg_wr ),
	.dm_wr_o				( dm_wr_2 ),
	.rg_rd_o				( rg_rd_2 ),
	.rg_wr_o				( rg_wr_2 ),
	.a_data_i			( a_data ),
	.wr_data_o			( wr_data_2 ),
	.b_addr_i			( b_addr ),
	.im_i					( im[DM_OFFS_W-1:0] ),
	.pc_1_i				( pc_1 ),
	.rg_addr_o			( rg_addr_2 ),
	.dm_addr_o			( dm_addr_2 )
	);
 
 
	// internal register set
	reg_set
	#(
	.REGS_IN				( 1 ),
	.REGS_OUT			( 1 ),
	.DATA_W				( DATA_W/2 ),
	.ADDR_W				( REG_ADDR_W ),
	.THREADS				( THREADS ),
	.THRD_W				( THRD_W ),
	.STACKS				( STACKS ),
	.STK_W				( STK_W ),
	.VER_MAJ				( VER_MAJ ),
	.VER_MIN				( VER_MIN ),
	.UART_DATA_W		( UART_DATA_W ),
	.CLK_HZ	 			( CLK_HZ ),
	.UART_BAUD_RATE	( UART_BAUD_RATE )
	)
	reg_set
	(
	.clk_i				( clk_i ),
	.rst_i				( rst_i ),
	.addr_i				( rg_addr_2[REG_ADDR_W-1:0] ),
	.wr_i					( rg_wr_2 ),
	.rd_i					( rg_rd_2 ),
	.data_i				( wr_data_2 ),
	.data_o				( rg_rd_data_4 ),
	.clr_req_o			( clr_req ),
	.intr_en_o			( intr_en ),
	.thrd_0_i			( thrd_0 ),
	.op_code_er_i		( op_code_er ),
	.thrd_2_i			( thrd_2 ),
	.pop_er_2_i			( pop_er_2 ),
	.thrd_3_i			( thrd_3 ),
	.push_er_3_i		( push_er_3 ),
	.io_lo_i				( io_i[DATA_W/2-1:0] ),
	.io_hi_i				( io_i[DATA_W-1:DATA_W/2] ),
	.io_lo_o				( io_o[DATA_W/2-1:0] ),
	.io_hi_o				( io_o[DATA_W-1:DATA_W/2] ),
	.uart_rx_i			( uart_rx_i ),
	.uart_tx_o			( uart_tx_o )
	);
 
 
	// instruction and data memory
	dp_ram_infer
	#(
	.REG_A_OUT			( 1 ),
	.REG_B_OUT			( 1 ),
	.DATA_W				( MEM_DATA_W ),
	.ADDR_W				( MEM_ADDR_W ),
	.MODE 				( "RAW" )  // functional don't care
	)
	main_mem
	(
	.a_clk_i				( clk_i ),
	.a_addr_i			( dm_addr_2 ),
	.a_wr_i				( dm_wr_2 ),
	.a_data_i			( wr_data_2 ),
	.a_data_o			( dm_rd_data_4 ),
	.b_clk_i				( clk_i ),
	.b_addr_i			( pc_4 ),
	.b_wr_i				( 1'b0 ),  // unused
	.b_data_i			(  ),  // unused
	.b_data_o			( op_code )
	);
 
 
endmodule
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.