mirror of
				https://github.com/openhwgroup/cvw
				synced 2025-02-11 06:05:49 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			183 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			Systemverilog
		
	
	
	
	
	
			
		
		
	
	
			183 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			Systemverilog
		
	
	
	
	
	
| ///////////////////////////////////////////
 | |
| // functionName.sv
 | |
| //
 | |
| // Written: Rose Thompson rose@rosethompson.net
 | |
| // 
 | |
| // Purpose: decode name of function
 | |
| // 
 | |
| // A component of the Wally configurable RISC-V project.
 | |
| // 
 | |
| // Copyright (C) 2021 Harvey Mudd College & Oklahoma State University
 | |
| //
 | |
| // SPDX-License-Identifier: Apache-2.0 WITH SHL-2.1
 | |
| //
 | |
| // Licensed under the Solderpad Hardware License v 2.1 (the “License”); you may not use this file 
 | |
| // except in compliance with the License, or, at your option, the Apache License version 2.0. You 
 | |
| // may obtain a copy of the License at
 | |
| //
 | |
| // https://solderpad.org/licenses/SHL-2.1/
 | |
| //
 | |
| // Unless required by applicable law or agreed to in writing, any work distributed under the 
 | |
| // License is distributed on an “AS IS” BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, 
 | |
| // either express or implied. See the License for the specific language governing permissions 
 | |
| // and limitations under the License.
 | |
| ////////////////////////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| module FunctionName import cvw::*; #(parameter cvw_t P) (
 | |
|   input logic reset,
 | |
|   input logic clk,
 | |
|   input string ProgramAddrMapFile,
 | |
|   input string ProgramLabelMapFile
 | |
|   );
 | |
|   
 | |
|   logic [P.XLEN-1:0] ProgramAddrMapMemory [logic [P.XLEN-1:0]];
 | |
|   string 	    ProgramLabelMapMemory [logic [P.XLEN-1:0]];
 | |
|   string 	    FunctionName;
 | |
|   
 | |
| 
 | |
|   logic [P.XLEN-1:0] PCF, PCD, PCE, PCM, FunctionAddr, PCM_temp, PCMOld;
 | |
|   logic 	    StallD, StallE, StallM, FlushD, FlushE, FlushM;
 | |
|   logic 		InstrValidM;
 | |
|   logic [P.XLEN-1:0] 	    ProgramAddrIndex, ProgramAddrIndexQ;
 | |
| 
 | |
|   assign PCF = testbench.dut.core.ifu.PCF;
 | |
|   assign StallD = testbench.dut.core.StallD;
 | |
|   assign StallE = testbench.dut.core.StallE;  
 | |
|   assign StallM = testbench.dut.core.StallM;  
 | |
|   assign FlushD = testbench.dut.core.FlushD;
 | |
|   assign FlushE = testbench.dut.core.FlushE;
 | |
|   assign FlushM = testbench.dut.core.FlushM;
 | |
|   assign InstrValidM = testbench.dut.core.InstrValidM;
 | |
| 
 | |
|   // copy from ifu
 | |
|   // when the F and D stages are flushed we need to ensure the PCE is held so that the function name does not
 | |
|   // erroneously change.
 | |
|   // also need to hold the old value not an erroneously fetched PC.
 | |
|   flopenr #(P.XLEN) PCDReg(clk, reset, ~StallD, FlushD ? PCE : PCF, PCD);
 | |
|   flopenr #(P.XLEN) PCEReg(clk, reset, ~StallE, FlushD & FlushE ? PCF : FlushE ? PCE : PCD, PCE);
 | |
|   flopenr #(P.XLEN) PCMReg(clk, reset, ~StallM, FlushD & FlushE & FlushM ? PCF : FlushE & FlushM ? PCE : FlushM ? PCM : PCE, PCM_temp);
 | |
|   flopenr #(P.XLEN) PCMOldReg(clk, reset, InstrValidM, PCM_temp, PCMOld);
 | |
|   assign PCM = InstrValidM ? PCM_temp : PCMOld;
 | |
| 
 | |
|   task automatic bin_search_min;
 | |
|     input logic [P.XLEN-1:0] pc;
 | |
|     input logic [P.XLEN-1:0] length;
 | |
|     ref logic [P.XLEN-1:0]   array [logic [P.XLEN-1:0]];
 | |
|     output logic [P.XLEN-1:0] minval;
 | |
|     output     logic [P.XLEN-1:0] mid;
 | |
| 
 | |
|     logic [P.XLEN-1:0] 	     left, right;
 | |
| 
 | |
|     begin
 | |
|       if ( pc == 0 ) begin
 | |
| 	// *** want to  keep the old value for mid and minval
 | |
| 	mid = 0;
 | |
| 	return;
 | |
|       end
 | |
|       left = 0;
 | |
|       right = length;
 | |
|       while (left <= right) begin
 | |
| 	mid = left + ((right - left) / 2);
 | |
| 	if (array[mid] == pc) begin
 | |
| 	  minval = array[mid];
 | |
| 	  return;
 | |
|         end
 | |
| 	if (array[mid] < pc) begin
 | |
| 	  left = mid + 1;
 | |
| 	end else if( array[mid] > pc) begin
 | |
| 	  right = mid -1;
 | |
| 	end else begin
 | |
| 	  //$display("Critical Error in FunctionName. PC, %x not found.", pc);
 | |
| 	  return;
 | |
| 	  //$stop();
 | |
| 	end	  
 | |
|       end // while (left <= right)
 | |
|       // if the element pc is now found, right and left will be equal at this point.
 | |
|       // we need to check if pc is less than the array at left or greather.
 | |
|       // if it is less than pc, then we select left as the index.
 | |
|       // if it is greather we want 1 less than left.
 | |
|       if (array[left] < pc) begin
 | |
| 	minval = array[left];
 | |
| 	mid = left;      
 | |
| 	return;	    
 | |
|       end else begin
 | |
| 	minval = array[left-1];
 | |
| 	mid = left - 1;	
 | |
| 	return;
 | |
|       end
 | |
|     end
 | |
|   endtask // bin_search_min
 | |
| 
 | |
|   integer ProgramAddrMapFP, ProgramLabelMapFP;
 | |
|   logic [P.XLEN-1:0] ProgramAddrMapLineCount;
 | |
|   logic [P.XLEN-1:0] ProgramLabelMapLineCount;
 | |
|   logic [P.XLEN-1:0] ProgramAddrMapLine;
 | |
|   string  ProgramLabelMapLine;
 | |
|   integer status;
 | |
|   
 | |
| 
 | |
|   // preload
 | |
| //  initial begin
 | |
|   always @ (negedge reset) begin
 | |
| 
 | |
|     // cannot readmemh directoy to a dynmaic array. Sad times :(
 | |
|     // Let's initialize a static array with FFFF_FFFF for all addresses.
 | |
|     // Then we can readmemh and finally copy to the dynamic array.
 | |
|     
 | |
| 	// clear out the old mapping between programs.
 | |
|     ProgramAddrMapMemory.delete();
 | |
|     ProgramLabelMapMemory.delete();
 | |
| 
 | |
|     // Unfortunately verilator version 5.011 readmemh does not support dynamic arrays
 | |
|     //$readmemh(ProgramAddrMapFile, ProgramAddrMapMemory);
 | |
|     // we need to count the number of lines in the file so we can set FunctionRadixLineCount.
 | |
| 
 | |
|     ProgramAddrMapLineCount = 0;
 | |
|     ProgramAddrMapFP = $fopen(ProgramAddrMapFile, "r");
 | |
| 
 | |
|     // read line by line to count lines
 | |
|     if (ProgramAddrMapFP != 0) begin
 | |
|       while (! $feof(ProgramAddrMapFP)) begin
 | |
| 	    status = $fscanf(ProgramAddrMapFP, "%h\n", ProgramAddrMapLine);
 | |
|         ProgramAddrMapMemory[ProgramAddrMapLineCount] = ProgramAddrMapLine;
 | |
| 	    ProgramAddrMapLineCount = ProgramAddrMapLineCount + 1;
 | |
|       end
 | |
|     end else begin
 | |
|       $display("Cannot open file %s for reading.", ProgramAddrMapFile);
 | |
|     end
 | |
|     $fclose(ProgramAddrMapFP);
 | |
| 
 | |
|     // ProgramIndexFile maps the program name to the compile index.
 | |
|     // The compile index is then used to inditify the application
 | |
|     // in the custom radix.
 | |
|     // Build an associative array to convert the name to an index.
 | |
|     ProgramLabelMapLineCount = 0;
 | |
|     ProgramLabelMapFP = $fopen(ProgramLabelMapFile, "r");
 | |
|     
 | |
|     if (ProgramLabelMapFP != 0) begin
 | |
|       while (! $feof(ProgramLabelMapFP)) begin
 | |
| 	status = $fscanf(ProgramLabelMapFP, "%s\n", ProgramLabelMapLine);
 | |
| 	ProgramLabelMapMemory[ProgramLabelMapLineCount] = ProgramLabelMapLine;
 | |
| 	ProgramLabelMapLineCount = ProgramLabelMapLineCount + 1;
 | |
|       end
 | |
|     end else begin
 | |
|       $display("Cannot open file %s for reading.", ProgramLabelMapFile);
 | |
|     end
 | |
|     $fclose(ProgramLabelMapFP);
 | |
| 
 | |
|   end
 | |
| 
 | |
|   always @(PCM) begin
 | |
|     bin_search_min(PCM, ProgramAddrMapLineCount, ProgramAddrMapMemory, FunctionAddr, ProgramAddrIndex);
 | |
|   end
 | |
| 
 | |
|   logic OrReducedAdr, AnyUnknown;
 | |
|   assign OrReducedAdr = |ProgramAddrIndex;
 | |
|   assign AnyUnknown = (OrReducedAdr === 1'bx) ? 1'b1 : 1'b0;
 | |
|   initial ProgramAddrIndex = 0;
 | |
| 
 | |
|   always @(*) FunctionName = AnyUnknown ? "Unknown!" : ProgramLabelMapMemory[ProgramAddrIndex];
 | |
| 
 | |
| endmodule // function_radix
 | |
| 
 |