blob: 33a048e305674f87fe200aaf2e7f99a1439986bc [file] [log] [blame] [edit]
//
//------------------------------------------------------------------------------
// Copyright 2007-2018 Mentor Graphics Corporation
// Copyright 2014 Semifore
// Copyright 2017 Intel Corporation
// Copyright 2010-2014 Synopsys, Inc.
// Copyright 2007-2018 Cadence Design Systems, Inc.
// Copyright 2013 Verilab
// Copyright 2012 AMD
// Copyright 2013-2018 NVIDIA Corporation
// Copyright 2014-2018 Cisco Systems, Inc.
// All Rights Reserved Worldwide
//
// Licensed under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in
// compliance with the License. You may obtain a copy of
// the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in
// writing, software 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.
//------------------------------------------------------------------------------
// File -- NODOCS -- Miscellaneous Structures
//------------------------------------------------------------------------------
//
// Class -- NODOCS -- uvm_void
//
// The ~uvm_void~ class is the base class for all UVM classes. It is an abstract
// class with no data members or functions. It allows for generic containers of
// objects to be created, similar to a void pointer in the C programming
// language. User classes derived directly from ~uvm_void~ inherit none of the
// UVM functionality, but such classes may be placed in ~uvm_void~-typed
// containers along with other UVM objects.
//
//------------------------------------------------------------------------------
// @uvm-ieee 1800.2-2017 auto 5.2
virtual class uvm_void;
endclass
// Append/prepend symbolic values for order-dependent APIs
typedef enum {UVM_APPEND, UVM_PREPEND} uvm_apprepend;
// Forward declaration since scope stack uses uvm_objects now
typedef class uvm_object;
typedef class uvm_coreservice_t;
typedef class uvm_factory;
typedef class uvm_config_db;
// m_uvm_config_obj_misc is an internal typedef for the uvm_misc.svh file
// to use. UVM users should use the uvm_config_object typedef
typedef uvm_config_db#(uvm_object) m_uvm_config_obj_misc;
typedef class uvm_comparer ;
typedef class uvm_packer ;
typedef class uvm_recorder ;
typedef class uvm_printer ;
// Variable- uvm_global_random_seed
//
// Create a seed which is based off of the global seed which can be used to seed
// srandom processes but will change if the command line seed setting is
// changed.
//
int unsigned uvm_global_random_seed = $urandom;
// Class- uvm_seed_map
//
// This map is a seed map that can be used to update seeds. The update
// is done automatically by the seed hashing routine. The seed_table_lookup
// uses an instance name lookup and the seed_table inside a given map
// uses a type name for the lookup.
//
class uvm_seed_map;
int unsigned seed_table [string];
int unsigned count [string];
endclass
uvm_seed_map uvm_random_seed_table_lookup [string];
//------------------------------------------------------------------------------
// Internal utility functions
//------------------------------------------------------------------------------
// Function- uvm_instance_scope
//
// A function that returns the scope that the UVM library lives in, either
// an instance, a module, or a package.
//
function string uvm_instance_scope();
byte c;
int pos;
//first time through the scope is ~null~ and we need to calculate, afterwards it
//is correctly set.
if(uvm_instance_scope != "")
return uvm_instance_scope;
$swrite(uvm_instance_scope, "%m");
//remove the extraneous .uvm_instance_scope piece or ::uvm_instance_scope
pos = uvm_instance_scope.len()-1;
c = uvm_instance_scope[pos];
while(pos && (c != ".") && (c != ":"))
c = uvm_instance_scope[--pos];
if(pos == 0)
uvm_report_error("SCPSTR", $sformatf("Illegal name %s in scope string",uvm_instance_scope));
uvm_instance_scope = uvm_instance_scope.substr(0,pos);
endfunction
// Function- uvm_oneway_hash
//
// A one-way hash function that is useful for creating srandom seeds. An
// unsigned int value is generated from the string input. An initial seed can
// be used to seed the hash, if not supplied the uvm_global_random_seed
// value is used. Uses a CRC like functionality to minimize collisions.
//
parameter UVM_STR_CRC_POLYNOMIAL = 32'h04c11db6;
function int unsigned uvm_oneway_hash ( string string_in, int unsigned seed=0 );
bit msb;
bit [7:0] current_byte;
bit [31:0] crc1;
if(!seed) seed = uvm_global_random_seed;
uvm_oneway_hash = seed;
crc1 = 32'hffffffff;
for (int _byte=0; _byte < string_in.len(); _byte++) begin
current_byte = string_in[_byte];
if (current_byte == 0) break;
for (int _bit=0; _bit < 8; _bit++) begin
msb = crc1[31];
crc1 <<= 1;
if (msb ^ current_byte[_bit]) begin
crc1 ^= UVM_STR_CRC_POLYNOMIAL;
crc1[0] = 1;
end
end
end
uvm_oneway_hash += ~{crc1[7:0], crc1[15:8], crc1[23:16], crc1[31:24]};
endfunction
// Function- uvm_create_random_seed
//
// Creates a random seed and updates the seed map so that if the same string
// is used again, a new value will be generated. The inst_id is used to hash
// by instance name and get a map of type name hashes which the type_id uses
// for its lookup.
function int unsigned uvm_create_random_seed ( string type_id, string inst_id="" );
uvm_seed_map seed_map;
if(inst_id == "")
inst_id = "__global__";
if(!uvm_random_seed_table_lookup.exists(inst_id))
uvm_random_seed_table_lookup[inst_id] = new;
seed_map = uvm_random_seed_table_lookup[inst_id];
type_id = {uvm_instance_scope(),type_id};
if(!seed_map.seed_table.exists(type_id)) begin
seed_map.seed_table[type_id] = uvm_oneway_hash ({type_id,"::",inst_id}, uvm_global_random_seed);
end
if (!seed_map.count.exists(type_id)) begin
seed_map.count[type_id] = 0;
end
//can't just increment, otherwise too much chance for collision, so
//randomize the seed using the last seed as the seed value. Check if
//the seed has been used before and if so increment it.
seed_map.seed_table[type_id] = seed_map.seed_table[type_id]+seed_map.count[type_id];
seed_map.count[type_id]++;
return seed_map.seed_table[type_id];
endfunction
// Function- uvm_object_value_str
//
//
function string uvm_object_value_str(uvm_object v);
if (v == null)
return "<null>";
uvm_object_value_str.itoa(v.get_inst_id());
uvm_object_value_str = {"@",uvm_object_value_str};
endfunction
// Function- uvm_leaf_scope
//
//
function string uvm_leaf_scope (string full_name, byte scope_separator = ".");
byte bracket_match;
int pos;
int bmatches;
bmatches = 0;
case(scope_separator)
"[": bracket_match = "]";
"(": bracket_match = ")";
"<": bracket_match = ">";
"{": bracket_match = "}";
default: bracket_match = "";
endcase
//Only use bracket matching if the input string has the end match
if(bracket_match != "" && bracket_match != full_name[full_name.len()-1])
bracket_match = "";
for(pos=full_name.len()-1; pos>0; --pos) begin
if(full_name[pos] == bracket_match) bmatches++;
else if(full_name[pos] == scope_separator) begin
bmatches--;
if(!bmatches || (bracket_match == "")) break;
end
end
if(pos) begin
if(scope_separator != ".") pos--;
uvm_leaf_scope = full_name.substr(pos+1,full_name.len()-1);
end
else begin
uvm_leaf_scope = full_name;
end
endfunction
// Function- uvm_bitstream_to_string
//
//
function string uvm_bitstream_to_string (uvm_bitstream_t value, int size,
uvm_radix_enum radix=UVM_NORADIX,
string radix_str="");
// sign extend & don't show radix for negative values
if (radix == UVM_DEC && value[size-1] === 1)
return $sformatf("%0d", value);
// TODO $countbits(value,'z) would be even better
if($isunknown(value)) begin
uvm_bitstream_t _t;
_t=0;
for(int idx=0;idx<size;idx++)
_t[idx]=value[idx];
value=_t;
end
else
value &= (1 << size)-1;
case(radix)
UVM_BIN: return $sformatf("%0s%0b", radix_str, value);
UVM_OCT: return $sformatf("%0s%0o", radix_str, value);
UVM_UNSIGNED: return $sformatf("%0s%0d", radix_str, value);
UVM_STRING: return $sformatf("%0s%0s", radix_str, value);
UVM_TIME: return $sformatf("%0s%0t", radix_str, value);
UVM_DEC: return $sformatf("%0s%0d", radix_str, value);
default: return $sformatf("%0s%0x", radix_str, value);
endcase
endfunction
// Function- uvm_integral_to_string
//
//
function string uvm_integral_to_string (uvm_integral_t value, int size,
uvm_radix_enum radix=UVM_NORADIX,
string radix_str="");
// sign extend & don't show radix for negative values
if (radix == UVM_DEC && value[size-1] === 1)
return $sformatf("%0d", value);
// TODO $countbits(value,'z) would be even better
if($isunknown(value)) begin
uvm_integral_t _t;
_t=0;
for(int idx=0;idx<size;idx++)
_t[idx]=value[idx];
value=_t;
end
else
value &= (1 << size)-1;
case(radix)
UVM_BIN: return $sformatf("%0s%0b", radix_str, value);
UVM_OCT: return $sformatf("%0s%0o", radix_str, value);
UVM_UNSIGNED: return $sformatf("%0s%0d", radix_str, value);
UVM_STRING: return $sformatf("%0s%0s", radix_str, value);
UVM_TIME: return $sformatf("%0s%0t", radix_str, value);
UVM_DEC: return $sformatf("%0s%0d", radix_str, value);
default: return $sformatf("%0s%0x", radix_str, value);
endcase
endfunction
// Function- uvm_get_array_index_int
//
// The following functions check to see if a string is representing an array
// index, and if so, what the index is.
function int uvm_get_array_index_int(string arg, output bit is_wildcard);
int i;
uvm_get_array_index_int = 0;
is_wildcard = 1;
i = arg.len() - 1;
if(arg[i] == "]")
while(i > 0 && (arg[i] != "[")) begin
--i;
if((arg[i] == "*") || (arg[i] == "?")) i=0;
else if((arg[i] < "0") || (arg[i] > "9") && (arg[i] != "[")) begin
uvm_get_array_index_int = -1; //illegal integral index
i=0;
end
end
else begin
is_wildcard = 0;
return 0;
end
if(i>0) begin
arg = arg.substr(i+1, arg.len()-2);
uvm_get_array_index_int = arg.atoi();
is_wildcard = 0;
end
endfunction
// Function- uvm_get_array_index_string
//
//
function string uvm_get_array_index_string(string arg, output bit is_wildcard);
int i;
uvm_get_array_index_string = "";
is_wildcard = 1;
i = arg.len() - 1;
if(arg[i] == "]")
while(i > 0 && (arg[i] != "[")) begin
if((arg[i] == "*") || (arg[i] == "?")) i=0;
--i;
end
if(i>0) begin
uvm_get_array_index_string = arg.substr(i+1, arg.len()-2);
is_wildcard = 0;
end
endfunction
// Function- uvm_is_array
//
//
function bit uvm_is_array(string arg);
return arg[arg.len()-1] == "]";
endfunction
// Function- uvm_has_wildcard
//
//
function automatic bit uvm_has_wildcard (string arg);
uvm_has_wildcard = 0;
//if it is a regex then return true
if( (arg.len() > 1) && (arg[0] == "/") && (arg[arg.len()-1] == "/") )
return 1;
//check if it has globs
foreach(arg[i])
if( (arg[i] == "*") || (arg[i] == "+") || (arg[i] == "?") )
uvm_has_wildcard = 1;
endfunction
typedef class uvm_component;
typedef class uvm_root;
typedef class uvm_report_object;
`ifdef UVM_ENABLE_DEPRECATED_API
//------------------------------------------------------------------------------
// CLASS -- NODOCS -- uvm_utils #(TYPE,FIELD)
//
// This class contains useful template functions.
//
//------------------------------------------------------------------------------
class uvm_utils #(type TYPE=int, string FIELD="config");
typedef TYPE types_t[$];
// Function -- NODOCS -- find_all
//
// Recursively finds all component instances of the parameter type ~TYPE~,
// starting with the component given by ~start~. Uses <uvm_root::find_all>.
static function types_t find_all(uvm_component start);
uvm_component list[$];
types_t types;
uvm_root top;
uvm_coreservice_t cs;
cs = uvm_coreservice_t::get();
top = cs.get_root();
top.find_all("*",list,start);
foreach (list[i]) begin
TYPE typ;
if ($cast(typ,list[i]))
types.push_back(typ);
end
if (types.size() == 0) begin
`uvm_warning("find_type-no match",{"Instance of type '",TYPE::type_name,
" not found in component hierarchy beginning at ",start.get_full_name()})
end
return types;
endfunction
static function TYPE find(uvm_component start);
types_t types = find_all(start);
if (types.size() == 0)
return null;
if (types.size() > 1) begin
`uvm_warning("find_type-multi match",{"More than one instance of type '",TYPE::type_name,
" found in component hierarchy beginning at ",start.get_full_name()})
return null;
end
return types[0];
endfunction
static function TYPE create_type_by_name(string type_name, string contxt);
uvm_object obj;
TYPE typ;
uvm_coreservice_t cs = uvm_coreservice_t::get();
uvm_factory factory=cs.get_factory();
obj = factory.create_object_by_name(type_name,contxt,type_name);
if (!$cast(typ,obj))
uvm_report_error("WRONG_TYPE",{"The type_name given '",type_name,
"' with context '",contxt,"' did not produce the expected type."});
return typ;
endfunction
// Function -- NODOCS -- get_config
//
// This method gets the object config of type ~TYPE~
// associated with component ~comp~.
// We check for the two kinds of error which may occur with this kind of
// operation.
static function TYPE get_config(uvm_component comp, bit is_fatal);
uvm_object obj;
TYPE cfg;
if (!m_uvm_config_obj_misc::get(comp,"",FIELD, obj)) begin
if (is_fatal)
comp.uvm_report_fatal("NO_SET_CFG", {"no set_config to field '", FIELD,
"' for component '",comp.get_full_name(),"'"},
UVM_MEDIUM, `uvm_file , `uvm_line );
else
comp.uvm_report_warning("NO_SET_CFG", {"no set_config to field '", FIELD,
"' for component '",comp.get_full_name(),"'"},
UVM_MEDIUM, `uvm_file , `uvm_line );
return null;
end
if (!$cast(cfg, obj)) begin
if (is_fatal)
comp.uvm_report_fatal( "GET_CFG_TYPE_FAIL",
{"set_config_object with field name ",FIELD,
" is not of type '",TYPE::type_name,"'"},
UVM_NONE , `uvm_file , `uvm_line );
else
comp.uvm_report_warning( "GET_CFG_TYPE_FAIL",
{"set_config_object with field name ",FIELD,
" is not of type '",TYPE::type_name,"'"},
UVM_NONE , `uvm_file , `uvm_line );
end
return cfg;
endfunction
endclass
`endif
`ifdef UVM_USE_PROCESS_CONTAINER
class process_container_c;
process p;
function new(process p_);
p=p_;
endfunction
endclass
`endif
// this is an internal function and provides a string join independent of a streaming pack
function automatic string m_uvm_string_queue_join(ref string i[$]);
`ifndef QUESTA
m_uvm_string_queue_join = {>>{i}};
`else
foreach(i[idx])
m_uvm_string_queue_join = {m_uvm_string_queue_join,i[idx]};
`endif
endfunction