#include "vtr_assert.h" #include "technology_library.h" /************************************************************************ * Member functions for class TechnologyLibrary ***********************************************************************/ /************************************************************************ * Constructors ***********************************************************************/ TechnologyLibrary::TechnologyLibrary() { return; } /************************************************************************ * Public Accessors : aggregates ***********************************************************************/ TechnologyLibrary::technology_device_range TechnologyLibrary::devices() const { return vtr::make_range(device_ids_.begin(), device_ids_.end()); } TechnologyLibrary::technology_variation_range TechnologyLibrary::variations() const { return vtr::make_range(variation_ids_.begin(), variation_ids_.end()); } /* Find technology devices in the same type (defined by users) and return a list of ids */ std::vector TechnologyLibrary::devices_by_type(const enum e_tech_lib_device_type& type) const { std::vector type_ids; for (auto id : devices()) { /* Skip unmatched types */ if (type != device_type(id)) { continue; } /* Matched type, update the vector */ type_ids.push_back(id); } return type_ids; } /************************************************************************ * Public Accessors : Basic data query on technology devices ***********************************************************************/ /* Access the name of a technology device */ std::string TechnologyLibrary::device_name(const TechnologyDeviceId& device_id) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); return device_names_[device_id]; } /* Access the id of a technology device by name, * If the name is valid, we return a valid id * Otherwise, return an invalid id */ TechnologyDeviceId TechnologyLibrary::device(const std::string& name) const { std::map::const_iterator it = device_name2ids_.find(name); if (it != device_name2ids_.end()) { return TechnologyDeviceId::INVALID(); } return device_name2ids_.at(name); } /* Access the type of a technology device */ enum e_tech_lib_device_type TechnologyLibrary::device_type(const TechnologyDeviceId& device_id) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); return device_types_[device_id]; } /* Access the model type of a technology device */ enum e_tech_lib_model_type TechnologyLibrary::device_model_type(const TechnologyDeviceId& device_id) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); return device_model_types_[device_id]; } /* Access the process corner name of a technology device */ std::string TechnologyLibrary::device_corner(const TechnologyDeviceId& device_id) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); return device_corners_[device_id]; } /* Access the model reference name of a technology device */ std::string TechnologyLibrary::device_model_ref(const TechnologyDeviceId& device_id) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); return device_model_refs_[device_id]; } /* Access the path of library for a technology device */ std::string TechnologyLibrary::device_lib_path(const TechnologyDeviceId& device_id) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); return device_lib_paths_[device_id]; } /* Access the VDD of a technology device * Note: This is ONLY applicable to transistor device */ float TechnologyLibrary::device_vdd(const TechnologyDeviceId& device_id) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); /* This is only applicable to transistor device */ VTR_ASSERT(TECH_LIB_DEVICE_TRANSISTOR == device_type(device_id)); return device_vdds_[device_id]; } /* Access the width ratio between PMOS and NMOS for a technology device * Note: This is ONLY applicable to transistor device */ float TechnologyLibrary::device_pn_ratio(const TechnologyDeviceId& device_id) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); /* This is only applicable to transistor device */ VTR_ASSERT(TECH_LIB_DEVICE_TRANSISTOR == device_type(device_id)); return device_pn_ratios_[device_id]; } /************************************************************************ * Public Accessors : Basic data query on transistors ***********************************************************************/ /* Access the model name of a transistor (either PMOS or NMOS) for a technology device * Note: This is ONLY applicable to transistor device */ std::string TechnologyLibrary::transistor_model_name(const TechnologyDeviceId& device_id, const e_tech_lib_trans_type& transistor_type) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); /* This is only applicable to transistor device */ VTR_ASSERT(TECH_LIB_DEVICE_TRANSISTOR == device_type(device_id)); return transistor_model_names_[device_id][transistor_type]; } /* Access the channel length of a transistor (either PMOS or NMOS) for a technology device * Note: This is ONLY applicable to transistor device */ float TechnologyLibrary::transistor_model_chan_length(const TechnologyDeviceId& device_id, const e_tech_lib_trans_type& transistor_type) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); /* This is only applicable to transistor device */ VTR_ASSERT(TECH_LIB_DEVICE_TRANSISTOR == device_type(device_id)); return transistor_model_chan_lengths_[device_id][transistor_type]; } /* Access the minimum width of a transistor (either PMOS or NMOS) for a technology device * Note: This is ONLY applicable to transistor device */ float TechnologyLibrary::transistor_model_min_width(const TechnologyDeviceId& device_id, const e_tech_lib_trans_type& transistor_type) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); /* This is only applicable to transistor device */ VTR_ASSERT(TECH_LIB_DEVICE_TRANSISTOR == device_type(device_id)); return transistor_model_min_widths_[device_id][transistor_type]; } /* Access the minimum width of a transistor (either PMOS or NMOS) for a technology device * Note: This is ONLY applicable to transistor device */ TechnologyVariationId TechnologyLibrary::transistor_model_variation(const TechnologyDeviceId& device_id, const e_tech_lib_trans_type& transistor_type) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); /* This is only applicable to transistor device */ VTR_ASSERT(TECH_LIB_DEVICE_TRANSISTOR == device_type(device_id)); return transistor_model_variation_ids_[device_id][transistor_type]; } /************************************************************************ * Public Accessors : Basic data query on RRAM devices ***********************************************************************/ /* Access the Low Resistence of a RRAM for a technology device * Note: This is ONLY applicable to RRAM device */ float TechnologyLibrary::rram_rlrs(const TechnologyDeviceId& device_id) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); /* This is only applicable to transistor device */ VTR_ASSERT(TECH_LIB_DEVICE_RRAM == device_type(device_id)); return rram_resistances_[device_id].x(); } /* Access the High Resistence of a RRAM for a technology device * Note: This is ONLY applicable to RRAM device */ float TechnologyLibrary::rram_rhrs(const TechnologyDeviceId& device_id) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); /* This is only applicable to transistor device */ VTR_ASSERT(TECH_LIB_DEVICE_RRAM == device_type(device_id)); return rram_resistances_[device_id].y(); } /* Access the Variation id of a RRAM for a technology device * Note: This is ONLY applicable to RRAM device */ TechnologyVariationId TechnologyLibrary::rram_variation(const TechnologyDeviceId& device_id) const { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); /* This is only applicable to RRAM device */ VTR_ASSERT(TECH_LIB_DEVICE_RRAM == device_type(device_id)); return rram_variation_ids_[device_id]; } /************************************************************************ * Public Accessors : Basic data query on technology variations ***********************************************************************/ /* Access the name of a technology variation */ std::string TechnologyLibrary::variation_name(const TechnologyVariationId& variation_id) const { /* validate the variation_id */ VTR_ASSERT(valid_variation_id(variation_id)); return variation_names_[variation_id]; } /* Access the id of a technology variation by name * If the name is valid, we return a valid id * Otherwise, return an invalid id */ TechnologyVariationId TechnologyLibrary::variation(const std::string& name) const { std::map::const_iterator it = variation_name2ids_.find(name); if (it != variation_name2ids_.end()) { return TechnologyVariationId::INVALID(); } return variation_name2ids_.at(name); } /* Access the abs value of a technology variation */ float TechnologyLibrary::variation_abs_value(const TechnologyVariationId& variation_id) const { /* validate the variation_id */ VTR_ASSERT(valid_variation_id(variation_id)); return variation_abs_values_[variation_id]; } /* Access the abs value of a technology variation */ size_t TechnologyLibrary::variation_num_sigma(const TechnologyVariationId& variation_id) const { /* validate the variation_id */ VTR_ASSERT(valid_variation_id(variation_id)); return variation_num_sigmas_[variation_id]; } /************************************************************************ * Public Mutators: device-related ***********************************************************************/ /* Add a new device to the library, return an id * This function will check if the name has already been used inside the data structure. * If used, it will return an invalid id */ TechnologyDeviceId TechnologyLibrary::add_device(const std::string& name) { std::map::iterator it = device_name2ids_.find(name); if (it != device_name2ids_.end()) { return TechnologyDeviceId::INVALID(); } /* This is a legal name. we can create a new id */ TechnologyDeviceId device = TechnologyDeviceId(device_ids_.size()); device_ids_.push_back(device); device_names_.push_back(name); device_types_.emplace_back(NUM_TECH_LIB_DEVICE_TYPES); device_model_types_.emplace_back(NUM_TECH_LIB_MODEL_TYPES); device_corners_.emplace_back(); device_model_refs_.emplace_back(); device_lib_paths_.emplace_back(); device_vdds_.push_back(-1); device_pn_ratios_.push_back(-1); transistor_model_names_.emplace_back(); transistor_model_chan_lengths_.emplace_back(); transistor_model_min_widths_.emplace_back(); transistor_model_variation_names_.emplace_back(); transistor_model_variation_ids_.emplace_back(); rram_resistances_.emplace_back(); rram_variation_names_.emplace_back(); rram_variation_ids_.emplace_back(); /* Register in the name-to-id map */ device_name2ids_[name] = device; return device; } /* Set the device type of a device in the library */ void TechnologyLibrary::set_device_type(const TechnologyDeviceId& device_id, const e_tech_lib_device_type& type) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); device_types_[device_id] = type; return; } /* Set the model type of a device in the library */ void TechnologyLibrary::set_device_model_type(const TechnologyDeviceId& device_id, const e_tech_lib_model_type& model_type) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); device_model_types_[device_id] = model_type; return; } /* Set the process corner of a device in the library */ void TechnologyLibrary::set_device_corner(const TechnologyDeviceId& device_id, const std::string& corner) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); device_corners_[device_id] = corner; return; } /* Set the string used to model reference of a device in the library */ void TechnologyLibrary::set_device_model_ref(const TechnologyDeviceId& device_id, const std::string& model_ref) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); device_model_refs_[device_id] = model_ref; return; } /* Set the library file path of a device in the library */ void TechnologyLibrary::set_device_lib_path(const TechnologyDeviceId& device_id, const std::string& lib_path) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); device_lib_paths_[device_id] = lib_path; return; } /* Set the operating voltage of a device in the library * This is ONLY applicable to transistors */ void TechnologyLibrary::set_device_vdd(const TechnologyDeviceId& device_id, const float& vdd) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); VTR_ASSERT(TECH_LIB_DEVICE_TRANSISTOR == device_type(device_id)); device_vdds_[device_id] = vdd; return; } /* Set the width ratio between PMOS and NMOS of a device in the library * This is ONLY applicable to transistors */ void TechnologyLibrary::set_device_pn_ratio(const TechnologyDeviceId& device_id, const float& pn_ratio) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); VTR_ASSERT(TECH_LIB_DEVICE_TRANSISTOR == device_type(device_id)); device_pn_ratios_[device_id] = pn_ratio; return; } /************************************************************************ * Public Mutators: transistor-related ***********************************************************************/ /* Set the model name for either PMOS or NMOS of a device in the library * This is ONLY applicable to transistors */ void TechnologyLibrary::set_transistor_model_name(const TechnologyDeviceId& device_id, const e_tech_lib_trans_type& transistor_type, const std::string& model_name) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); VTR_ASSERT(TECH_LIB_DEVICE_TRANSISTOR == device_type(device_id)); transistor_model_names_[device_id][transistor_type] = model_name; return; } /* Set the channel length for either PMOS or NMOS of a device in the library * This is ONLY applicable to transistors */ void TechnologyLibrary::set_transistor_model_chan_length(const TechnologyDeviceId& device_id, const e_tech_lib_trans_type& transistor_type, const float& chan_length) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); VTR_ASSERT(TECH_LIB_DEVICE_TRANSISTOR == device_type(device_id)); transistor_model_chan_lengths_[device_id][transistor_type] = chan_length; return; } /* Set the minimum width for either PMOS or NMOS of a device in the library * This is ONLY applicable to transistors */ void TechnologyLibrary::set_transistor_model_min_width(const TechnologyDeviceId& device_id, const e_tech_lib_trans_type& transistor_type, const float& min_width) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); VTR_ASSERT(TECH_LIB_DEVICE_TRANSISTOR == device_type(device_id)); transistor_model_min_widths_[device_id][transistor_type] = min_width; return; } /* Set the variation name for either PMOS or NMOS of a device in the library * This is ONLY applicable to transistors */ void TechnologyLibrary::set_transistor_model_variation_name(const TechnologyDeviceId& device_id, const e_tech_lib_trans_type& transistor_type, const std::string& variation_name) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); VTR_ASSERT(TECH_LIB_DEVICE_TRANSISTOR == device_type(device_id)); transistor_model_variation_names_[device_id][transistor_type] = variation_name; return; } /************************************************************************ * Public Mutators: RRAM-related ***********************************************************************/ /* Set the Low Resistance State (LRS) resistance for a RRAM device in the library * This is ONLY applicable to RRAM devices */ void TechnologyLibrary::set_rram_rlrs(const TechnologyDeviceId& device_id, const float& rlrs) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); VTR_ASSERT(TECH_LIB_DEVICE_RRAM == device_type(device_id)); rram_resistances_[device_id].set_x(rlrs); return; } /* Set the High Resistance State (HRS) resistance for a RRAM device in the library * This is ONLY applicable to RRAM devices */ void TechnologyLibrary::set_rram_rhrs(const TechnologyDeviceId& device_id, const float& rhrs) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); VTR_ASSERT(TECH_LIB_DEVICE_RRAM == device_type(device_id)); rram_resistances_[device_id].set_y(rhrs); return; } /* Set the variation name for a RRAM device in the library * This is ONLY applicable to RRAM devices */ void TechnologyLibrary::set_rram_variation_name(const TechnologyDeviceId& device_id, const std::string& variation_name) { /* validate the device_id */ VTR_ASSERT(valid_device_id(device_id)); VTR_ASSERT(TECH_LIB_DEVICE_RRAM == device_type(device_id)); rram_variation_names_[device_id] = variation_name; return; } /************************************************************************ * Public Mutators: variation-related ***********************************************************************/ /* Add a new variation to the library, return an id * This function will check if the name has already been used inside the data structure. * If used, it will return an invalid id */ TechnologyVariationId TechnologyLibrary::add_variation(const std::string& name) { std::map::iterator it = variation_name2ids_.find(name); if (it != variation_name2ids_.end()) { return TechnologyVariationId::INVALID(); } /* This is a legal name. we can create a new id */ TechnologyVariationId variation = TechnologyVariationId(variation_ids_.size()); variation_names_.push_back(name); variation_abs_values_.push_back(0.); variation_num_sigmas_.push_back(size_t(-1)); /* Register in the name-to-id map */ variation_name2ids_[name] = variation; return variation; } /* Set the absolute value of a variation */ void TechnologyLibrary::set_variation_abs_value(const TechnologyVariationId& variation_id, const float& abs_value) { /* validate the variation_id */ VTR_ASSERT(valid_variation_id(variation_id)); variation_abs_values_[variation_id] = abs_value; return; } /* Set the number of sigma of a variation */ void TechnologyLibrary::set_variation_num_sigma(const TechnologyVariationId& variation_id, const size_t& num_sigma) { /* validate the variation_id */ VTR_ASSERT(valid_variation_id(variation_id)); variation_num_sigmas_[variation_id] = num_sigma; return; } /************************************************************************ * Public mutators: linkers ***********************************************************************/ /* This function builds the links between devices and variations, * which have been defined in the technology library */ void TechnologyLibrary::link_devices_to_variations() { for (const TechnologyDeviceId& device : devices()) { /* For transistors, find the variation name for each model and build a link */ if (TECH_LIB_DEVICE_TRANSISTOR == device_type(device)) { /* PMOS transistor, if a variation name is specified, we try to build a link * Otherwise, we assign any invalid id */ const std::string& pmos_var_name = transistor_model_variation_names_[device][TECH_LIB_TRANSISTOR_PMOS]; transistor_model_variation_ids_[device][TECH_LIB_TRANSISTOR_PMOS] = variation(pmos_var_name); /* NMOS transistor, if a variation name is specified, we try to build a link * Otherwise, we assign any invalid id */ const std::string& nmos_var_name = transistor_model_variation_names_[device][TECH_LIB_TRANSISTOR_NMOS]; transistor_model_variation_ids_[device][TECH_LIB_TRANSISTOR_NMOS] = variation(nmos_var_name); /* Finish for transistors, go to the next */ continue; } /* Reach here it means an RRAM device, we find the variation name and try to build a link */ VTR_ASSERT(TECH_LIB_DEVICE_RRAM == device_type(device)); const std::string& rram_var_name = rram_variation_names_[device]; rram_variation_ids_[device] = variation(rram_var_name); /* Finish for RRAMs, go to the next */ } } /************************************************************************ * Internal invalidators/validators ***********************************************************************/ /* Validators */ bool TechnologyLibrary::valid_device_id(const TechnologyDeviceId& device_id) const { return ( size_t(device_id) < device_ids_.size() ) && ( device_id == device_ids_[device_id] ); } bool TechnologyLibrary::valid_variation_id(const TechnologyVariationId& variation_id) const { return ( size_t(variation_id) < variation_ids_.size() ) && ( variation_id == variation_ids_[variation_id] ); }