2014-07-19 08:33:00 -05:00
|
|
|
/*
|
|
|
|
* yosys -- Yosys Open SYnthesis Suite
|
|
|
|
*
|
|
|
|
* Copyright (C) 2012 Clifford Wolf <clifford@clifford.at>
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2014-07-31 16:30:18 -05:00
|
|
|
#ifndef MODTOOLS_H
|
|
|
|
#define MODTOOLS_H
|
2014-07-19 08:33:00 -05:00
|
|
|
|
2014-08-01 09:53:15 -05:00
|
|
|
#include "kernel/yosys.h"
|
2014-07-19 08:33:00 -05:00
|
|
|
#include "kernel/sigtools.h"
|
|
|
|
#include "kernel/celltypes.h"
|
|
|
|
|
2014-08-01 09:53:15 -05:00
|
|
|
YOSYS_NAMESPACE_BEGIN
|
|
|
|
|
|
|
|
struct ModIndex : public RTLIL::Monitor
|
|
|
|
{
|
|
|
|
struct PortInfo {
|
2014-08-03 08:00:38 -05:00
|
|
|
RTLIL::Cell* cell;
|
|
|
|
RTLIL::IdString port;
|
|
|
|
int offset;
|
2014-08-01 09:53:15 -05:00
|
|
|
|
2014-08-03 08:00:38 -05:00
|
|
|
PortInfo(RTLIL::Cell* _c, RTLIL::IdString _p, int _o) : cell(_c), port(_p), offset(_o) { }
|
2014-08-01 09:53:15 -05:00
|
|
|
|
|
|
|
bool operator<(const PortInfo &other) const {
|
|
|
|
if (cell != other.cell)
|
|
|
|
return cell < other.cell;
|
|
|
|
if (offset != other.offset)
|
|
|
|
return offset < other.offset;
|
|
|
|
return port < other.port;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct SigBitInfo
|
|
|
|
{
|
|
|
|
bool is_input, is_output;
|
|
|
|
std::set<PortInfo> ports;
|
|
|
|
|
|
|
|
SigBitInfo() : is_input(false), is_output(false) { }
|
|
|
|
};
|
|
|
|
|
|
|
|
SigMap sigmap;
|
|
|
|
RTLIL::Module *module;
|
|
|
|
std::map<RTLIL::SigBit, SigBitInfo> database;
|
|
|
|
bool auto_reload_module;
|
|
|
|
|
2014-08-03 08:00:38 -05:00
|
|
|
void port_add(RTLIL::Cell *cell, RTLIL::IdString port, const RTLIL::SigSpec &sig)
|
2014-08-01 09:53:15 -05:00
|
|
|
{
|
2014-10-10 09:59:44 -05:00
|
|
|
for (int i = 0; i < GetSize(sig); i++) {
|
2014-08-05 07:47:03 -05:00
|
|
|
RTLIL::SigBit bit = sigmap(sig[i]);
|
|
|
|
if (bit.wire)
|
|
|
|
database[bit].ports.insert(PortInfo(cell, port, i));
|
|
|
|
}
|
2014-08-01 09:53:15 -05:00
|
|
|
}
|
|
|
|
|
2014-08-03 08:00:38 -05:00
|
|
|
void port_del(RTLIL::Cell *cell, RTLIL::IdString port, const RTLIL::SigSpec &sig)
|
2014-08-01 09:53:15 -05:00
|
|
|
{
|
2014-10-10 09:59:44 -05:00
|
|
|
for (int i = 0; i < GetSize(sig); i++) {
|
2014-08-05 07:47:03 -05:00
|
|
|
RTLIL::SigBit bit = sigmap(sig[i]);
|
|
|
|
if (bit.wire)
|
|
|
|
database[bit].ports.erase(PortInfo(cell, port, i));
|
|
|
|
}
|
2014-08-01 09:53:15 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
const SigBitInfo &info(RTLIL::SigBit bit)
|
|
|
|
{
|
|
|
|
return database[sigmap(bit)];
|
|
|
|
}
|
|
|
|
|
|
|
|
void reload_module()
|
|
|
|
{
|
|
|
|
sigmap.clear();
|
|
|
|
sigmap.set(module);
|
|
|
|
|
|
|
|
database.clear();
|
|
|
|
for (auto wire : module->wires())
|
|
|
|
if (wire->port_input || wire->port_output)
|
2014-10-10 09:59:44 -05:00
|
|
|
for (int i = 0; i < GetSize(wire); i++) {
|
2014-08-05 07:47:03 -05:00
|
|
|
RTLIL::SigBit bit = sigmap(RTLIL::SigBit(wire, i));
|
|
|
|
if (bit.wire && wire->port_input)
|
|
|
|
database[bit].is_input = true;
|
|
|
|
if (bit.wire && wire->port_output)
|
|
|
|
database[bit].is_output = true;
|
2014-08-01 09:53:15 -05:00
|
|
|
}
|
|
|
|
for (auto cell : module->cells())
|
|
|
|
for (auto &conn : cell->connections())
|
|
|
|
port_add(cell, conn.first, conn.second);
|
|
|
|
|
|
|
|
auto_reload_module = false;
|
|
|
|
}
|
|
|
|
|
2014-11-09 03:55:04 -06:00
|
|
|
virtual void notify_connect(RTLIL::Cell *cell, const RTLIL::IdString &port, const RTLIL::SigSpec &old_sig, RTLIL::SigSpec &sig) YS_OVERRIDE
|
2014-08-01 09:53:15 -05:00
|
|
|
{
|
|
|
|
if (auto_reload_module)
|
|
|
|
reload_module();
|
|
|
|
|
|
|
|
port_del(cell, port, old_sig);
|
|
|
|
port_add(cell, port, sig);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void notify_connect(RTLIL::Module *mod, const RTLIL::SigSig&)
|
|
|
|
{
|
|
|
|
log_assert(module == mod);
|
|
|
|
auto_reload_module = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void notify_connect(RTLIL::Module *mod, const std::vector<RTLIL::SigSig>&)
|
|
|
|
{
|
|
|
|
log_assert(module == mod);
|
|
|
|
auto_reload_module = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void notify_blackout(RTLIL::Module *mod)
|
|
|
|
{
|
|
|
|
log_assert(module == mod);
|
|
|
|
auto_reload_module = true;
|
|
|
|
}
|
|
|
|
|
2014-08-03 08:00:38 -05:00
|
|
|
ModIndex(RTLIL::Module *_m) : module(_m)
|
|
|
|
{
|
2014-08-01 09:53:15 -05:00
|
|
|
auto_reload_module = true;
|
|
|
|
module->monitors.insert(this);
|
|
|
|
}
|
|
|
|
|
2014-08-03 08:00:38 -05:00
|
|
|
~ModIndex()
|
|
|
|
{
|
2014-08-01 09:53:15 -05:00
|
|
|
module->monitors.erase(this);
|
|
|
|
}
|
2014-08-03 08:00:38 -05:00
|
|
|
|
|
|
|
SigBitInfo *query(RTLIL::SigBit bit)
|
|
|
|
{
|
|
|
|
if (auto_reload_module)
|
|
|
|
reload_module();
|
2014-08-05 07:47:03 -05:00
|
|
|
|
2014-08-03 08:00:38 -05:00
|
|
|
auto it = database.find(sigmap(bit));
|
|
|
|
if (it == database.end())
|
|
|
|
return nullptr;
|
|
|
|
else
|
|
|
|
return &it->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool query_is_input(RTLIL::SigBit bit)
|
|
|
|
{
|
|
|
|
const SigBitInfo *info = query(bit);
|
|
|
|
if (info == nullptr)
|
|
|
|
return false;
|
|
|
|
return info->is_input;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool query_is_output(RTLIL::SigBit bit)
|
|
|
|
{
|
|
|
|
const SigBitInfo *info = query(bit);
|
|
|
|
if (info == nullptr)
|
|
|
|
return false;
|
|
|
|
return info->is_output;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::set<PortInfo> &query_ports(RTLIL::SigBit bit)
|
|
|
|
{
|
|
|
|
static std::set<PortInfo> empty_result_set;
|
|
|
|
SigBitInfo *info = query(bit);
|
|
|
|
if (info == nullptr)
|
|
|
|
return empty_result_set;
|
|
|
|
return info->ports;
|
|
|
|
}
|
2014-08-01 09:53:15 -05:00
|
|
|
};
|
|
|
|
|
2014-07-19 08:33:00 -05:00
|
|
|
struct ModWalker
|
|
|
|
{
|
|
|
|
struct PortBit
|
|
|
|
{
|
|
|
|
RTLIL::Cell *cell;
|
|
|
|
RTLIL::IdString port;
|
|
|
|
int offset;
|
|
|
|
|
|
|
|
bool operator<(const PortBit &other) const {
|
|
|
|
if (cell != other.cell)
|
|
|
|
return cell < other.cell;
|
|
|
|
if (port != other.port)
|
|
|
|
return port < other.port;
|
|
|
|
return offset < other.offset;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
RTLIL::Design *design;
|
|
|
|
RTLIL::Module *module;
|
|
|
|
|
|
|
|
CellTypes ct;
|
|
|
|
SigMap sigmap;
|
|
|
|
|
|
|
|
std::map<RTLIL::SigBit, std::set<PortBit>> signal_drivers;
|
|
|
|
std::map<RTLIL::SigBit, std::set<PortBit>> signal_consumers;
|
|
|
|
std::set<RTLIL::SigBit> signal_inputs, signal_outputs;
|
|
|
|
|
|
|
|
std::map<RTLIL::Cell*, std::set<RTLIL::SigBit>> cell_outputs, cell_inputs;
|
|
|
|
|
|
|
|
void add_wire(RTLIL::Wire *wire)
|
|
|
|
{
|
|
|
|
if (wire->port_input) {
|
|
|
|
std::vector<RTLIL::SigBit> bits = sigmap(wire);
|
|
|
|
for (auto bit : bits)
|
|
|
|
if (bit.wire != NULL)
|
|
|
|
signal_inputs.insert(bit);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wire->port_output) {
|
|
|
|
std::vector<RTLIL::SigBit> bits = sigmap(wire);
|
|
|
|
for (auto bit : bits)
|
|
|
|
if (bit.wire != NULL)
|
|
|
|
signal_outputs.insert(bit);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void add_cell_port(RTLIL::Cell *cell, RTLIL::IdString port, std::vector<RTLIL::SigBit> bits, bool is_output, bool is_input)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < int(bits.size()); i++)
|
|
|
|
if (bits[i].wire != NULL) {
|
|
|
|
PortBit pbit = { cell, port, i };
|
|
|
|
if (is_output) {
|
|
|
|
signal_drivers[bits[i]].insert(pbit);
|
|
|
|
cell_outputs[cell].insert(bits[i]);
|
|
|
|
}
|
|
|
|
if (is_input) {
|
|
|
|
signal_consumers[bits[i]].insert(pbit);
|
|
|
|
cell_inputs[cell].insert(bits[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void add_cell(RTLIL::Cell *cell)
|
|
|
|
{
|
|
|
|
if (ct.cell_known(cell->type)) {
|
2014-07-26 07:32:50 -05:00
|
|
|
for (auto &conn : cell->connections())
|
2014-07-19 08:33:00 -05:00
|
|
|
add_cell_port(cell, conn.first, sigmap(conn.second),
|
|
|
|
ct.cell_output(cell->type, conn.first),
|
|
|
|
ct.cell_input(cell->type, conn.first));
|
|
|
|
} else {
|
2014-07-26 07:32:50 -05:00
|
|
|
for (auto &conn : cell->connections())
|
2014-07-19 08:33:00 -05:00
|
|
|
add_cell_port(cell, conn.first, sigmap(conn.second), true, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ModWalker() : design(NULL), module(NULL)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
ModWalker(RTLIL::Design *design, RTLIL::Module *module, CellTypes *filter_ct = NULL)
|
|
|
|
{
|
|
|
|
setup(design, module, filter_ct);
|
|
|
|
}
|
|
|
|
|
|
|
|
void setup(RTLIL::Design *design, RTLIL::Module *module, CellTypes *filter_ct = NULL)
|
|
|
|
{
|
|
|
|
this->design = design;
|
|
|
|
this->module = module;
|
|
|
|
|
|
|
|
ct.clear();
|
|
|
|
ct.setup(design);
|
|
|
|
sigmap.set(module);
|
|
|
|
|
|
|
|
signal_drivers.clear();
|
|
|
|
signal_consumers.clear();
|
|
|
|
signal_inputs.clear();
|
|
|
|
signal_outputs.clear();
|
|
|
|
|
2014-07-26 18:49:51 -05:00
|
|
|
for (auto &it : module->wires_)
|
2014-07-19 08:33:00 -05:00
|
|
|
add_wire(it.second);
|
2014-07-26 18:51:45 -05:00
|
|
|
for (auto &it : module->cells_)
|
2014-07-19 08:33:00 -05:00
|
|
|
if (filter_ct == NULL || filter_ct->cell_known(it.second->type))
|
|
|
|
add_cell(it.second);
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_* methods -- single RTLIL::SigBit
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline bool get_drivers(std::set<PortBit> &result, RTLIL::SigBit bit) const
|
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
if (signal_drivers.count(bit)) {
|
|
|
|
const std::set<PortBit> &r = signal_drivers.at(bit);
|
|
|
|
result.insert(r.begin(), r.end());
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline bool get_consumers(std::set<PortBit> &result, RTLIL::SigBit bit) const
|
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
if (signal_consumers.count(bit)) {
|
|
|
|
const std::set<PortBit> &r = signal_consumers.at(bit);
|
|
|
|
result.insert(r.begin(), r.end());
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline bool get_inputs(std::set<RTLIL::SigBit> &result, RTLIL::SigBit bit) const
|
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
if (signal_inputs.count(bit))
|
|
|
|
result.insert(bit), found = true;
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline bool get_outputs(std::set<RTLIL::SigBit> &result, RTLIL::SigBit bit) const
|
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
if (signal_outputs.count(bit))
|
|
|
|
result.insert(bit), found = true;
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_* methods -- container of RTLIL::SigBit's (always by reference)
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline bool get_drivers(std::set<PortBit> &result, const T &bits) const
|
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
for (RTLIL::SigBit bit : bits)
|
|
|
|
if (signal_drivers.count(bit)) {
|
|
|
|
const std::set<PortBit> &r = signal_drivers.at(bit);
|
|
|
|
result.insert(r.begin(), r.end());
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline bool get_consumers(std::set<PortBit> &result, const T &bits) const
|
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
for (RTLIL::SigBit bit : bits)
|
|
|
|
if (signal_consumers.count(bit)) {
|
|
|
|
const std::set<PortBit> &r = signal_consumers.at(bit);
|
|
|
|
result.insert(r.begin(), r.end());
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline bool get_inputs(std::set<RTLIL::SigBit> &result, const T &bits) const
|
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
for (RTLIL::SigBit bit : bits)
|
|
|
|
if (signal_inputs.count(bit))
|
|
|
|
result.insert(bit), found = true;
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline bool get_outputs(std::set<RTLIL::SigBit> &result, const T &bits) const
|
|
|
|
{
|
|
|
|
bool found = false;
|
|
|
|
for (RTLIL::SigBit bit : bits)
|
|
|
|
if (signal_outputs.count(bit))
|
|
|
|
result.insert(bit), found = true;
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get_* methods -- call by RTLIL::SigSpec (always by value)
|
|
|
|
|
|
|
|
bool get_drivers(std::set<PortBit> &result, RTLIL::SigSpec signal) const
|
|
|
|
{
|
|
|
|
std::vector<RTLIL::SigBit> bits = sigmap(signal);
|
|
|
|
return get_drivers(result, bits);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool get_consumers(std::set<PortBit> &result, RTLIL::SigSpec signal) const
|
|
|
|
{
|
|
|
|
std::vector<RTLIL::SigBit> bits = sigmap(signal);
|
|
|
|
return get_consumers(result, bits);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool get_inputs(std::set<RTLIL::SigBit> &result, RTLIL::SigSpec signal) const
|
|
|
|
{
|
|
|
|
std::vector<RTLIL::SigBit> bits = sigmap(signal);
|
|
|
|
return get_inputs(result, bits);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool get_outputs(std::set<RTLIL::SigBit> &result, RTLIL::SigSpec signal) const
|
|
|
|
{
|
|
|
|
std::vector<RTLIL::SigBit> bits = sigmap(signal);
|
|
|
|
return get_outputs(result, bits);
|
|
|
|
}
|
|
|
|
|
|
|
|
// has_* methods -- call by reference
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline bool has_drivers(const T &sig) const {
|
|
|
|
std::set<PortBit> result;
|
|
|
|
return get_drivers(result, sig);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline bool has_consumers(const T &sig) const {
|
|
|
|
std::set<PortBit> result;
|
|
|
|
return get_consumers(result, sig);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline bool has_inputs(const T &sig) const {
|
|
|
|
std::set<RTLIL::SigBit> result;
|
|
|
|
return get_inputs(result, sig);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline bool has_outputs(const T &sig) const {
|
|
|
|
std::set<RTLIL::SigBit> result;
|
|
|
|
return get_outputs(result, sig);
|
|
|
|
}
|
|
|
|
|
|
|
|
// has_* methods -- call by value
|
|
|
|
|
|
|
|
inline bool has_drivers(RTLIL::SigSpec sig) const {
|
|
|
|
std::set<PortBit> result;
|
|
|
|
return get_drivers(result, sig);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool has_consumers(RTLIL::SigSpec sig) const {
|
|
|
|
std::set<PortBit> result;
|
|
|
|
return get_consumers(result, sig);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool has_inputs(RTLIL::SigSpec sig) const {
|
|
|
|
std::set<RTLIL::SigBit> result;
|
|
|
|
return get_inputs(result, sig);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool has_outputs(RTLIL::SigSpec sig) const {
|
|
|
|
std::set<RTLIL::SigBit> result;
|
|
|
|
return get_outputs(result, sig);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-08-01 09:53:15 -05:00
|
|
|
YOSYS_NAMESPACE_END
|
|
|
|
|
2014-07-19 08:33:00 -05:00
|
|
|
#endif
|