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

Subversion Repositories openarty

[/] [openarty/] [trunk/] [rtl/] [wbucompress.v] - Rev 3

Compare with Previous | Blame | View Log

////////////////////////////////////////////////////////////////////////////////
//
// Filename:	wbucompress.v
//
// Project:	FPGA library
//
// Purpose:	When reading many words that are identical, it makes no sense
//		to spend the time transmitting the same thing over and over
//	again, especially on a slow channel.  Hence this routine uses a table
//	lookup to see if the word to be transmitted was one from the recent
//	past.  If so, the word is replaced with an address of the recently
//	transmitted word.  Mind you, the table lookup takes one clock per table
//	entry, so even if a word is in the table it might not be found in time.
//	If the word is not in the table, or if it isn't found due to a lack of
//	time, the word is placed into the table while incrementing every other
//	table address.
//
//	Oh, and on a new address--the table is reset and starts over.  This way,
//	any time the host software changes, the host software will always start
//	by issuing a new address--hence the table is reset for every new piece
//	of software that may wish to communicate.
//
//
// Creator:	Dan Gisselquist, Ph.D.
//		Gisselquist Technology, LLC
//
////////////////////////////////////////////////////////////////////////////////
//
// Copyright (C) 2015-2016, Gisselquist Technology, LLC
//
// This program is free software (firmware): you can redistribute it and/or
// modify it under the terms of  the GNU General Public License as published
// by the Free Software Foundation, either version 3 of the License, or (at
// your option) any later version.
//
// This program is distributed in the hope that it will be useful, but WITHOUT
// ANY WARRANTY; without even the implied warranty of MERCHANTIBILITY or
// FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
// for more details.
//
// License:	GPL, v3, as defined and found on www.gnu.org,
//		http://www.gnu.org/licenses/gpl.html
//
//
////////////////////////////////////////////////////////////////////////////////
//
//
// All input words are valid codewords.  If we can, we make them
// better here.
module	wbucompress(i_clk, i_stb, i_codword, o_stb, o_cword, i_busy);
	parameter	DW=32, CW=36, TBITS=10;
	input				i_clk, i_stb;
	input		[(CW-1):0]	i_codword;
	output	wire			o_stb;
	output	wire	[(CW-1):0]	o_cword;
	input				i_busy;
 
	//
	//
	// First stage is to compress the address.
	// This stage requires one clock.
	//
	//	ISTB,ICODWORD
	//	ISTB2,IWRD2	ASTB,AWORD
	//	ISTB3,IWRD3	ASTB2,AWRD2	I_BUSY(1)
	//	ISTB3,IWRD3	ASTB2,AWRD2	I_BUSY(1)
	//	ISTB3,IWRD3	ASTB2,AWRD2	I_BUSY(1)
	//	ISTB3,IWRD3	ASTB2,AWRD2	
	//	ISTB4,IWRD4	ASTB3,AWRD3	I_BUSY(2)
	//	ISTB4,IWRD4	ASTB3,AWRD3	I_BUSY(2)
	//	ISTB4,IWRD4	ASTB3,AWRD3	I_BUSY(2)
	reg		a_stb;
	reg	[35:0]	a_addrword;
	wire	[31:0]	w_addr;
	assign	w_addr = i_codword[31:0];
	always @(posedge i_clk)
		if ((i_stb)&&(~a_stb))
		begin
			if (i_codword[35:32] != 4'h2)
			begin
				a_addrword <= i_codword;
			end else if (w_addr[31:6] == 26'h00)
				a_addrword <= { 6'hc, w_addr[ 5:0], 24'h00 };
			else if (w_addr[31:12] == 20'h00)
				a_addrword <= { 6'hd, w_addr[11:0], 18'h00 };
			else if (w_addr[31:18] == 14'h00)
				a_addrword <= { 6'he, w_addr[17:0], 12'h00 };
			else if (w_addr[31:24] == 8'h00)
				a_addrword <= { 6'hf, w_addr[23:0],  6'h00 };
			else begin
				a_addrword <= i_codword;
			end
		end
	initial	a_stb = 1'b0;
	always @(posedge i_clk)
		if ((i_stb)&&(~a_stb))
			a_stb <= i_stb;
		else if (~i_busy)
			a_stb <= 1'b0;
 
 
	//
	//
	// The next stage attempts to replace data codewords with previous
	// codewords that may have been sent.  The memory is only allowed
	// to be as old as the last new address command.  In this fashion,
	// any program that wishes to talk to the device can start with a
	// known compression table by simply setting the address and then
	// reading from the device.
	//
 
	// We start over any time a new value shows up, and 
	// the follow-on isn't busy and can take it.  Likewise,
	// we reset the writer on the compression any time a
	// i_clr value comes through (i.e., ~i_cyc or new
	// address)
 
	wire	w_accepted;
	assign	w_accepted = (a_stb)&&(~i_busy);
 
	reg		r_stb;
	always @(posedge i_clk)
		r_stb <= a_stb;
 
	wire	[35:0]	r_word;
	assign	r_word = a_addrword;
 
 
	//
	// First step of the compression is keeping track of a compression
	// table.  And the first part of that is keeping track of what address
	// to write into the compression table, and whether or not the entire
	// table is full or not.  This logic follows:
	//
	reg	[(TBITS-1):0]	tbl_addr;
	reg			tbl_filled;
	// First part, write the compression table
	always @(posedge i_clk)
		// If we send a new address, then reset the table to empty
		if (w_accepted)
		begin
			// Reset on new address (0010xx) and on new compressed
			// addresses (0011ll).
			if (o_cword[35:33]==3'h1)
				tbl_addr <= 0;
			// Otherwise, on any valid return result that wasn't
			// from our table, for whatever reason (such as didn't
			// have the clocks to find it, etc.), increment the
			// address to add another value into our table
			else if (o_cword[35:33] == 3'b111)
				tbl_addr <= tbl_addr + {{(TBITS-1){1'b0}},1'b1};
		end
	always @(posedge i_clk)
		if ((w_accepted)&&(o_cword[35:33]==3'h1)) // on new address
			tbl_filled <= 1'b0;
		else if (tbl_addr == 10'h3ff)
			tbl_filled <= 1'b1;
 
	// Now that we know where we are writing into the table, and what
	// values of the table are valid, we need to actually write into
	// the table.
	//
	// We can keep this logic really simple by writing on every clock
	// and writing junk on many of those clocks, but we'll need to remember
	// that the value of the table at tbl_addr is unreliable until tbl_addr
	// changes.
	//
	reg	[31:0]	compression_tbl	[0:((1<<TBITS)-1)];
	// Write new values into the table
	always @(posedge i_clk)
		compression_tbl[tbl_addr] <= { r_word[32:31], r_word[29:0] };
 
	// Now that we have a working table, can we use it?
	// On any new word, we'll start looking through our codewords.
	// If we find any that matches, we're there.  We might (or might not)
	// make it through the table first.  That's irrelevant.  We just look
	// while we can.
	reg			tbl_match, nxt_match; // <= (nxt_rd_addr == tbl_addr);
	reg	[(TBITS-1):0]	rd_addr;
	reg	[(TBITS-1):0]	nxt_rd_addr;
	initial	rd_addr = 0;
	initial	tbl_match = 0;
	always @(posedge i_clk)
	begin
		nxt_match <= ((nxt_rd_addr-tbl_addr)=={{(TBITS-1){1'b0}},1'b1});
		if ((w_accepted)||(~a_stb))
		begin
			// Keep in mind, if a write was just accepted, then
			// rd_addr will need to be reset on the next clock
			// when (~a_stb).  Hence this must be a two clock
			// update
			rd_addr <= tbl_addr + {(TBITS){1'b1}};
			nxt_rd_addr = tbl_addr + { {(TBITS-1){1'b1}}, 1'b0 };
			tbl_match <= 1'b0;
		end else if ((~tbl_match)&&(~match)
				&&((~nxt_rd_addr[TBITS-1])||(tbl_filled)))
		begin
			rd_addr <= nxt_rd_addr;
			nxt_rd_addr = nxt_rd_addr - { {(TBITS-1){1'b0}}, 1'b1 };
			tbl_match <= nxt_match;
		end
	end
 
	reg	[1:0]		pmatch;
	reg			dmatch, // Match, on clock 'd'
				vaddr;	// Was the address valid then?
	reg	[(DW-1):0]	cword;
	reg	[(TBITS-1):0]	caddr, daddr, maddr;
	always @(posedge i_clk)
	begin
		cword <= compression_tbl[rd_addr];
		caddr <= rd_addr;
 
		dmatch <= (cword == { r_word[32:31], r_word[29:0] });
		daddr  <= caddr;
		maddr  <= tbl_addr - caddr;
 
		vaddr <= ( {1'b0, caddr} < {tbl_filled, tbl_addr} )
			&&(caddr != tbl_addr);
	end
 
	always @(posedge i_clk)
		if ((w_accepted)||(~a_stb))
			pmatch <= 0; // rd_addr is set on this clock
		else
			// cword is set on the next clock, pmatch = 3'b001
			// dmatch is set on the next clock, pmatch = 3'b011
			pmatch <= { pmatch[0], 1'b1 };
 
	reg		match;
	reg	[(TBITS-1):0]	matchaddr;
	always @(posedge i_clk)
		if((w_accepted)||(~a_stb)||(~r_stb))// Reset upon any write
			match <= 1'b0;
		else if (~match)
		begin
			// To be a match, the table must not be empty,
			match <= (vaddr)&&(dmatch)&&(r_word[35:33]==3'b111)
					&&(pmatch == 2'b11);
		end
 
	reg	zmatch, hmatch, fmatch;
	always @(posedge i_clk)
		if (~match)
		begin
			matchaddr <= maddr;
			fmatch    <= (maddr < 10'h521);
			zmatch    <= (maddr == 10'h1);
			hmatch    <= (maddr < 10'd10);
		end
 
	// Did we find something?
	wire	[(TBITS-1):0]	adr_diff;
	wire	[9:0]		adr_dbld;
	wire	[2:0]		adr_hlfd;
	assign	adr_diff = matchaddr;
	assign	adr_hlfd = matchaddr[2:0]- 3'd2;
	assign	adr_dbld = matchaddr- 10'd10;
	reg	[(CW-1):0]	r_cword; // Record our result
	always @(posedge i_clk)
	begin
		if ((~a_stb)||(~r_stb)||(w_accepted))//Reset whenever word gets written
		begin
			r_cword <= r_word;
		end else if ((match)&&(fmatch)) // &&(r_word == a_addrword))
		begin
			r_cword <= r_word;
			if (zmatch) // matchaddr == 1
				r_cword[35:30] <= { 5'h3, r_word[30] };
			else if (hmatch) // 2 <= matchaddr <= 9
				r_cword[35:30] <= { 2'b10, adr_hlfd, r_word[30] };
			else // if (adr_diff < 10'd521)
				r_cword[35:24] <= { 2'b01, adr_dbld[8:6],
						r_word[30], adr_dbld[5:0] };
		end else
			r_cword <= r_word;
	end
 
	// Can we do this without a clock delay?
	assign	o_stb = a_stb;
	assign	o_cword = (r_stb)?(r_cword):(a_addrword);
endmodule
 
 

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.