2022-10-06 19:08:50 -05:00
|
|
|
#include "bus_group.h"
|
|
|
|
|
2022-02-17 17:02:37 -06:00
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
#include "vtr_assert.h"
|
|
|
|
#include "vtr_log.h"
|
|
|
|
|
2022-10-06 19:08:50 -05:00
|
|
|
namespace openfpga { // Begin namespace openfpga
|
2022-02-17 19:09:03 -06:00
|
|
|
|
2022-02-17 17:02:37 -06:00
|
|
|
/************************************************************************
|
|
|
|
* Member functions for class BusGroup
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
/************************************************************************
|
|
|
|
* Constructors
|
|
|
|
***********************************************************************/
|
2022-10-06 19:08:50 -05:00
|
|
|
BusGroup::BusGroup() { return; }
|
2022-02-17 17:02:37 -06:00
|
|
|
|
|
|
|
/************************************************************************
|
|
|
|
* Public Accessors : aggregates
|
|
|
|
***********************************************************************/
|
|
|
|
BusGroup::bus_group_range BusGroup::buses() const {
|
|
|
|
return vtr::make_range(bus_ids_.begin(), bus_ids_.end());
|
|
|
|
}
|
|
|
|
|
|
|
|
/************************************************************************
|
2022-10-06 19:08:50 -05:00
|
|
|
* Public Accessors : Basic data query
|
2022-02-17 17:02:37 -06:00
|
|
|
***********************************************************************/
|
|
|
|
openfpga::BasicPort BusGroup::bus_port(const BusGroupId& bus_id) const {
|
|
|
|
VTR_ASSERT(valid_bus_id(bus_id));
|
2022-10-06 19:08:50 -05:00
|
|
|
return bus_ports_[bus_id];
|
2022-02-17 17:02:37 -06:00
|
|
|
}
|
|
|
|
|
2022-02-19 01:05:03 -06:00
|
|
|
bool BusGroup::is_big_endian(const BusGroupId& bus_id) const {
|
|
|
|
VTR_ASSERT(valid_bus_id(bus_id));
|
2022-10-06 19:08:50 -05:00
|
|
|
return bus_big_endians_[bus_id];
|
2022-02-19 01:05:03 -06:00
|
|
|
}
|
|
|
|
|
2022-02-17 18:17:37 -06:00
|
|
|
std::vector<BusPinId> BusGroup::bus_pins(const BusGroupId& bus_id) const {
|
|
|
|
VTR_ASSERT(valid_bus_id(bus_id));
|
2022-10-06 19:08:50 -05:00
|
|
|
return bus_pin_ids_[bus_id];
|
2022-02-17 18:17:37 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
int BusGroup::pin_index(const BusPinId& pin_id) const {
|
|
|
|
VTR_ASSERT(valid_pin_id(pin_id));
|
2022-10-06 19:08:50 -05:00
|
|
|
return pin_indices_[pin_id];
|
2022-02-17 18:17:37 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string BusGroup::pin_name(const BusPinId& pin_id) const {
|
|
|
|
VTR_ASSERT(valid_pin_id(pin_id));
|
2022-10-06 19:08:50 -05:00
|
|
|
return pin_names_[pin_id];
|
2022-02-17 18:17:37 -06:00
|
|
|
}
|
|
|
|
|
2022-02-18 01:14:28 -06:00
|
|
|
BusGroupId BusGroup::find_pin_bus(const std::string& pin_name) const {
|
2022-10-06 19:08:50 -05:00
|
|
|
std::map<std::string, BusPinId>::const_iterator result =
|
|
|
|
pin_name2id_map_.find(pin_name);
|
2022-02-18 01:14:28 -06:00
|
|
|
if (result == pin_name2id_map_.end()) {
|
|
|
|
/* Not found, return an invalid id */
|
|
|
|
return BusGroupId::INVALID();
|
|
|
|
}
|
|
|
|
/* Found, we should get the parent bus */
|
2022-10-06 19:08:50 -05:00
|
|
|
BusPinId pin_id = result->second;
|
2022-02-18 01:14:28 -06:00
|
|
|
return pin_parent_bus_ids_[pin_id];
|
|
|
|
}
|
|
|
|
|
2022-02-18 14:03:26 -06:00
|
|
|
BusGroupId BusGroup::find_bus(const std::string& bus_name) const {
|
2022-10-06 19:08:50 -05:00
|
|
|
std::map<std::string, BusGroupId>::const_iterator result =
|
|
|
|
bus_name2id_map_.find(bus_name);
|
2022-02-18 14:03:26 -06:00
|
|
|
if (result == bus_name2id_map_.end()) {
|
|
|
|
/* Not found, return an invalid id */
|
|
|
|
return BusGroupId::INVALID();
|
|
|
|
}
|
|
|
|
/* Found, we should get the parent bus */
|
2022-10-06 19:08:50 -05:00
|
|
|
return result->second;
|
2022-02-18 14:03:26 -06:00
|
|
|
}
|
|
|
|
|
2022-02-18 01:14:28 -06:00
|
|
|
BusPinId BusGroup::find_pin(const std::string& pin_name) const {
|
2022-10-06 19:08:50 -05:00
|
|
|
std::map<std::string, BusPinId>::const_iterator result =
|
|
|
|
pin_name2id_map_.find(pin_name);
|
2022-02-18 01:14:28 -06:00
|
|
|
if (result == pin_name2id_map_.end()) {
|
|
|
|
/* Not found, return an invalid id */
|
|
|
|
return BusPinId::INVALID();
|
|
|
|
}
|
|
|
|
/* Found, we should get the parent bus */
|
2022-10-06 19:08:50 -05:00
|
|
|
return result->second;
|
2022-02-18 01:14:28 -06:00
|
|
|
}
|
|
|
|
|
2022-10-06 19:08:50 -05:00
|
|
|
bool BusGroup::empty() const { return 0 == bus_ids_.size(); }
|
2022-02-17 17:02:37 -06:00
|
|
|
|
|
|
|
/************************************************************************
|
|
|
|
* Public Mutators
|
|
|
|
***********************************************************************/
|
|
|
|
void BusGroup::reserve_buses(const size_t& num_buses) {
|
|
|
|
bus_ids_.reserve(num_buses);
|
|
|
|
bus_ports_.reserve(num_buses);
|
2022-02-19 01:05:03 -06:00
|
|
|
bus_big_endians_.reserve(num_buses);
|
2022-02-17 19:09:03 -06:00
|
|
|
bus_pin_ids_.reserve(num_buses);
|
2022-02-17 17:02:37 -06:00
|
|
|
}
|
|
|
|
|
2022-02-17 17:50:44 -06:00
|
|
|
void BusGroup::reserve_pins(const size_t& num_pins) {
|
|
|
|
pin_ids_.reserve(num_pins);
|
|
|
|
pin_indices_.reserve(num_pins);
|
|
|
|
pin_names_.reserve(num_pins);
|
2022-02-18 01:14:28 -06:00
|
|
|
pin_parent_bus_ids_.reserve(num_pins);
|
2022-02-17 17:50:44 -06:00
|
|
|
}
|
|
|
|
|
2022-02-17 17:02:37 -06:00
|
|
|
BusGroupId BusGroup::create_bus(const openfpga::BasicPort& bus_port) {
|
|
|
|
/* Create a new id */
|
|
|
|
BusGroupId bus_id = BusGroupId(bus_ids_.size());
|
2022-10-06 19:08:50 -05:00
|
|
|
|
2022-02-17 17:02:37 -06:00
|
|
|
bus_ids_.push_back(bus_id);
|
2022-02-17 19:09:03 -06:00
|
|
|
bus_ports_.push_back(bus_port);
|
2022-02-19 01:05:03 -06:00
|
|
|
bus_big_endians_.push_back(true);
|
2022-02-17 19:09:03 -06:00
|
|
|
bus_pin_ids_.emplace_back();
|
2022-02-18 14:03:26 -06:00
|
|
|
|
|
|
|
/* Register to fast look-up */
|
|
|
|
auto result = bus_name2id_map_.find(bus_port.get_name());
|
|
|
|
if (result == bus_name2id_map_.end()) {
|
|
|
|
bus_name2id_map_[bus_port.get_name()] = bus_id;
|
|
|
|
} else {
|
2022-10-06 19:08:50 -05:00
|
|
|
VTR_LOG_ERROR("Duplicated bus name '%s' in bus group",
|
|
|
|
bus_port.get_name().c_str());
|
2022-02-18 14:03:26 -06:00
|
|
|
exit(1);
|
|
|
|
}
|
2022-10-06 19:08:50 -05:00
|
|
|
|
2022-02-17 17:02:37 -06:00
|
|
|
return bus_id;
|
|
|
|
}
|
|
|
|
|
2022-10-06 19:08:50 -05:00
|
|
|
void BusGroup::set_bus_big_endian(const BusGroupId& bus_id,
|
|
|
|
const bool& big_endian) {
|
2022-02-19 01:05:03 -06:00
|
|
|
VTR_ASSERT(valid_bus_id(bus_id));
|
|
|
|
bus_big_endians_[bus_id] = big_endian;
|
|
|
|
}
|
|
|
|
|
2022-02-18 14:03:26 -06:00
|
|
|
BusPinId BusGroup::create_pin(const BusGroupId& bus_id, const int& index) {
|
2022-02-17 17:50:44 -06:00
|
|
|
/* Create a new id */
|
|
|
|
BusPinId pin_id = BusPinId(pin_ids_.size());
|
2022-10-06 19:08:50 -05:00
|
|
|
|
2022-02-17 17:50:44 -06:00
|
|
|
pin_ids_.push_back(pin_id);
|
|
|
|
|
2022-02-18 14:03:26 -06:00
|
|
|
pin_indices_.push_back(index);
|
2022-02-17 17:50:44 -06:00
|
|
|
pin_names_.emplace_back();
|
|
|
|
|
|
|
|
/* Register the pin to the bus */
|
|
|
|
VTR_ASSERT(valid_bus_id(bus_id));
|
2022-02-18 01:14:28 -06:00
|
|
|
pin_parent_bus_ids_.push_back(bus_id);
|
2022-10-06 19:08:50 -05:00
|
|
|
|
2022-02-18 14:03:26 -06:00
|
|
|
/* If the pin index is beyond the range of the bus_pin_ids, resize it */
|
2022-02-18 14:36:03 -06:00
|
|
|
if (size_t(index) >= bus_pin_ids_[bus_id].size()) {
|
2022-02-18 14:03:26 -06:00
|
|
|
bus_pin_ids_[bus_id].resize(index + 1);
|
|
|
|
}
|
|
|
|
bus_pin_ids_[bus_id][index] = pin_id;
|
2022-02-18 01:14:28 -06:00
|
|
|
|
2022-02-17 19:09:03 -06:00
|
|
|
return pin_id;
|
2022-02-17 17:50:44 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
void BusGroup::set_pin_name(const BusPinId& pin_id, const std::string& name) {
|
|
|
|
VTR_ASSERT(valid_pin_id(pin_id));
|
|
|
|
pin_names_[pin_id] = name;
|
2022-02-18 01:14:28 -06:00
|
|
|
|
|
|
|
/* Register to fast look-up */
|
|
|
|
auto result = pin_name2id_map_.find(name);
|
|
|
|
if (result == pin_name2id_map_.end()) {
|
|
|
|
pin_name2id_map_[name] = pin_id;
|
|
|
|
} else {
|
|
|
|
VTR_LOG_ERROR("Duplicated pin name '%s' in bus group", name.c_str());
|
|
|
|
exit(1);
|
|
|
|
}
|
2022-02-17 17:50:44 -06:00
|
|
|
}
|
|
|
|
|
2022-02-17 17:02:37 -06:00
|
|
|
/************************************************************************
|
2022-10-06 19:08:50 -05:00
|
|
|
* Internal invalidators/validators
|
2022-02-17 17:02:37 -06:00
|
|
|
***********************************************************************/
|
|
|
|
bool BusGroup::valid_bus_id(const BusGroupId& bus_id) const {
|
2022-10-06 19:08:50 -05:00
|
|
|
return (size_t(bus_id) < bus_ids_.size()) && (bus_id == bus_ids_[bus_id]);
|
2022-02-17 17:02:37 -06:00
|
|
|
}
|
|
|
|
|
2022-02-17 17:50:44 -06:00
|
|
|
bool BusGroup::valid_pin_id(const BusPinId& pin_id) const {
|
2022-10-06 19:08:50 -05:00
|
|
|
return (size_t(pin_id) < pin_ids_.size()) && (pin_id == pin_ids_[pin_id]);
|
2022-02-17 17:50:44 -06:00
|
|
|
}
|
|
|
|
|
2022-10-06 19:08:50 -05:00
|
|
|
} // End of namespace openfpga
|