2024-04-15 07:14:50 -05:00
|
|
|
#include "kernel/yosys.h"
|
|
|
|
#include "kernel/drivertools.h"
|
|
|
|
#include "kernel/topo_scc.h"
|
2024-07-25 06:10:59 -05:00
|
|
|
#include "kernel/compute_graph.h"
|
2024-04-15 07:14:50 -05:00
|
|
|
|
|
|
|
USING_YOSYS_NAMESPACE
|
|
|
|
PRIVATE_NAMESPACE_BEGIN
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
struct ExampleWorker
|
|
|
|
{
|
|
|
|
DriverMap dm;
|
|
|
|
Module *module;
|
|
|
|
|
|
|
|
ExampleWorker(Module *module) : module(module) {
|
|
|
|
dm.celltypes.setup();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ExampleDtPass : public Pass
|
|
|
|
{
|
|
|
|
ExampleDtPass() : Pass("example_dt", "drivertools example") {}
|
|
|
|
|
|
|
|
void help() override
|
|
|
|
{
|
|
|
|
// |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|
|
|
|
|
log("\n");
|
2024-10-08 01:47:51 -05:00
|
|
|
log("TODO: add help message\n");
|
|
|
|
log("\n");
|
2024-04-15 07:14:50 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void execute(std::vector<std::string> args, RTLIL::Design *design) override
|
|
|
|
{
|
|
|
|
size_t argidx = 1;
|
|
|
|
extra_args(args, argidx, design);
|
|
|
|
|
|
|
|
for (auto module : design->selected_modules()) {
|
|
|
|
ExampleWorker worker(module);
|
|
|
|
DriverMap dm;
|
|
|
|
|
2024-04-11 06:48:25 -05:00
|
|
|
struct ExampleFn {
|
|
|
|
IdString name;
|
|
|
|
dict<IdString, Const> parameters;
|
|
|
|
|
|
|
|
ExampleFn(IdString name) : name(name) {}
|
|
|
|
ExampleFn(IdString name, dict<IdString, Const> parameters) : name(name), parameters(parameters) {}
|
|
|
|
|
|
|
|
bool operator==(ExampleFn const &other) const {
|
|
|
|
return name == other.name && parameters == other.parameters;
|
|
|
|
}
|
|
|
|
|
2024-11-11 08:45:11 -06:00
|
|
|
Hasher hash_eat(Hasher h) const {
|
|
|
|
h.eat(name);
|
|
|
|
h.eat(parameters);
|
2024-10-01 08:12:03 -05:00
|
|
|
return h;
|
2024-04-11 06:48:25 -05:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef ComputeGraph<ExampleFn, int, IdString, IdString> ExampleGraph;
|
|
|
|
|
|
|
|
ExampleGraph compute_graph;
|
|
|
|
|
|
|
|
|
2024-04-15 07:14:50 -05:00
|
|
|
dm.add(module);
|
|
|
|
|
|
|
|
idict<DriveSpec> queue;
|
|
|
|
idict<Cell *> cells;
|
|
|
|
|
|
|
|
IntGraph edges;
|
2024-04-11 06:48:25 -05:00
|
|
|
std::vector<int> graph_nodes;
|
2024-04-15 07:14:50 -05:00
|
|
|
|
2024-04-11 06:48:25 -05:00
|
|
|
auto enqueue = [&](DriveSpec const &spec) {
|
|
|
|
int index = queue(spec);
|
|
|
|
if (index == GetSize(graph_nodes))
|
|
|
|
graph_nodes.emplace_back(compute_graph.add(ID($pending), index).index());
|
|
|
|
//if (index >= GetSize(graph_nodes))
|
|
|
|
return compute_graph[graph_nodes[index]];
|
|
|
|
};
|
2024-04-15 07:14:50 -05:00
|
|
|
|
|
|
|
for (auto cell : module->cells()) {
|
|
|
|
if (cell->type.in(ID($assert), ID($assume), ID($cover), ID($check)))
|
2024-04-11 06:48:25 -05:00
|
|
|
enqueue(DriveBitMarker(cells(cell), 0));
|
2024-04-15 07:14:50 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
for (auto wire : module->wires()) {
|
|
|
|
if (!wire->port_output)
|
|
|
|
continue;
|
2024-04-11 06:48:25 -05:00
|
|
|
enqueue(DriveChunk(DriveChunkWire(wire, 0, wire->width))).assign_key(wire->name);
|
2024-04-15 07:14:50 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i != GetSize(queue); ++i)
|
|
|
|
{
|
|
|
|
DriveSpec spec = queue[i];
|
2024-04-11 06:48:25 -05:00
|
|
|
ExampleGraph::Ref node = compute_graph[i];
|
|
|
|
|
2024-04-15 07:14:50 -05:00
|
|
|
if (spec.chunks().size() > 1) {
|
2024-04-11 06:48:25 -05:00
|
|
|
node.set_function(ID($$concat));
|
|
|
|
|
2024-04-15 07:14:50 -05:00
|
|
|
for (auto const &chunk : spec.chunks()) {
|
2024-04-11 06:48:25 -05:00
|
|
|
node.append_arg(enqueue(chunk));
|
2024-04-15 07:14:50 -05:00
|
|
|
}
|
|
|
|
} else if (spec.chunks().size() == 1) {
|
|
|
|
DriveChunk chunk = spec.chunks()[0];
|
|
|
|
if (chunk.is_wire()) {
|
|
|
|
DriveChunkWire wire_chunk = chunk.wire();
|
|
|
|
if (wire_chunk.is_whole()) {
|
2024-04-11 06:48:25 -05:00
|
|
|
node.sparse_attr() = wire_chunk.wire->name;
|
2024-04-15 07:14:50 -05:00
|
|
|
if (wire_chunk.wire->port_input) {
|
2024-04-11 06:48:25 -05:00
|
|
|
node.set_function(ExampleFn(ID($$input), {{wire_chunk.wire->name, {}}}));
|
2024-04-15 07:14:50 -05:00
|
|
|
} else {
|
|
|
|
DriveSpec driver = dm(DriveSpec(wire_chunk));
|
2024-04-11 06:48:25 -05:00
|
|
|
node.set_function(ID($$buf));
|
|
|
|
|
|
|
|
node.append_arg(enqueue(driver));
|
2024-04-15 07:14:50 -05:00
|
|
|
}
|
|
|
|
} else {
|
2024-05-01 05:47:16 -05:00
|
|
|
DriveChunkWire whole_wire(wire_chunk.wire, 0, wire_chunk.wire->width);
|
2024-04-11 06:48:25 -05:00
|
|
|
node.set_function(ExampleFn(ID($$slice), {{ID(offset), wire_chunk.offset}, {ID(width), wire_chunk.width}}));
|
|
|
|
node.append_arg(enqueue(whole_wire));
|
2024-04-15 07:14:50 -05:00
|
|
|
}
|
|
|
|
} else if (chunk.is_port()) {
|
|
|
|
DriveChunkPort port_chunk = chunk.port();
|
|
|
|
if (port_chunk.is_whole()) {
|
|
|
|
if (dm.celltypes.cell_output(port_chunk.cell->type, port_chunk.port)) {
|
2024-04-11 06:48:25 -05:00
|
|
|
if (port_chunk.cell->type.in(ID($dff), ID($ff)))
|
|
|
|
{
|
|
|
|
Cell *cell = port_chunk.cell;
|
|
|
|
node.set_function(ExampleFn(ID($$state), {{cell->name, {}}}));
|
|
|
|
for (auto const &conn : cell->connections()) {
|
|
|
|
if (!dm.celltypes.cell_input(cell->type, conn.first))
|
|
|
|
continue;
|
|
|
|
enqueue(DriveChunkPort(cell, conn)).assign_key(cell->name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
node.set_function(ExampleFn(ID($$cell_output), {{port_chunk.port, {}}}));
|
|
|
|
node.append_arg(enqueue(DriveBitMarker(cells(port_chunk.cell), 0)));
|
|
|
|
}
|
2024-04-15 07:14:50 -05:00
|
|
|
} else {
|
2024-04-11 06:48:25 -05:00
|
|
|
node.set_function(ID($$buf));
|
|
|
|
|
2024-04-15 07:14:50 -05:00
|
|
|
DriveSpec driver = dm(DriveSpec(port_chunk));
|
2024-04-11 06:48:25 -05:00
|
|
|
node.append_arg(enqueue(driver));
|
2024-04-15 07:14:50 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
DriveChunkPort whole_port(port_chunk.cell, port_chunk.port, 0, GetSize(port_chunk.cell->connections().at(port_chunk.port)));
|
2024-05-01 05:47:16 -05:00
|
|
|
node.set_function(ExampleFn(ID($$slice), {{ID(offset), port_chunk.offset}}));
|
2024-04-11 06:48:25 -05:00
|
|
|
node.append_arg(enqueue(whole_port));
|
2024-04-15 07:14:50 -05:00
|
|
|
}
|
|
|
|
} else if (chunk.is_constant()) {
|
2024-04-11 06:48:25 -05:00
|
|
|
node.set_function(ExampleFn(ID($$const), {{ID(value), chunk.constant()}}));
|
|
|
|
|
|
|
|
} else if (chunk.is_multiple()) {
|
|
|
|
node.set_function(ID($$multi));
|
|
|
|
for (auto const &driver : chunk.multiple().multiple())
|
|
|
|
node.append_arg(enqueue(driver));
|
2024-04-15 07:14:50 -05:00
|
|
|
} else if (chunk.is_marker()) {
|
|
|
|
Cell *cell = cells[chunk.marker().marker];
|
2024-04-11 06:48:25 -05:00
|
|
|
|
|
|
|
node.set_function(ExampleFn(cell->type, cell->parameters));
|
2024-04-15 07:14:50 -05:00
|
|
|
for (auto const &conn : cell->connections()) {
|
|
|
|
if (!dm.celltypes.cell_input(cell->type, conn.first))
|
|
|
|
continue;
|
2024-04-11 06:48:25 -05:00
|
|
|
|
|
|
|
node.append_arg(enqueue(DriveChunkPort(cell, conn)));
|
2024-04-15 07:14:50 -05:00
|
|
|
}
|
2024-04-11 06:48:25 -05:00
|
|
|
} else if (chunk.is_none()) {
|
|
|
|
node.set_function(ID($$undriven));
|
|
|
|
|
2024-04-15 07:14:50 -05:00
|
|
|
} else {
|
2024-04-11 06:48:25 -05:00
|
|
|
log_error("unhandled drivespec: %s\n", log_signal(chunk));
|
2024-04-15 07:14:50 -05:00
|
|
|
log_abort();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log_abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-04-11 06:48:25 -05:00
|
|
|
// Perform topo sort and detect SCCs
|
|
|
|
ExampleGraph::SccAdaptor compute_graph_scc(compute_graph);
|
|
|
|
|
|
|
|
|
|
|
|
std::vector<int> perm;
|
2024-07-25 04:35:32 -05:00
|
|
|
TopoSortedSccs(compute_graph_scc, [&](int *begin, int *end) {
|
2024-04-11 06:48:25 -05:00
|
|
|
perm.insert(perm.end(), begin, end);
|
|
|
|
if (end > begin + 1)
|
|
|
|
{
|
|
|
|
log_warning("SCC:");
|
|
|
|
for (int *i = begin; i != end; ++i)
|
|
|
|
log(" %d", *i);
|
|
|
|
log("\n");
|
|
|
|
}
|
2024-07-25 04:35:32 -05:00
|
|
|
}).process_sources().process_all();
|
2024-04-11 06:48:25 -05:00
|
|
|
compute_graph.permute(perm);
|
|
|
|
|
|
|
|
|
|
|
|
// Forward $$buf unless we have a name in the sparse attribute
|
|
|
|
std::vector<int> alias;
|
|
|
|
perm.clear();
|
|
|
|
|
|
|
|
for (int i = 0; i < compute_graph.size(); ++i)
|
|
|
|
{
|
|
|
|
if (compute_graph[i].function().name == ID($$buf) && !compute_graph[i].has_sparse_attr() && compute_graph[i].arg(0).index() < i)
|
|
|
|
{
|
|
|
|
|
|
|
|
alias.push_back(alias[compute_graph[i].arg(0).index()]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
alias.push_back(GetSize(perm));
|
|
|
|
perm.push_back(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
compute_graph.permute(perm, alias);
|
|
|
|
|
|
|
|
// Dump the compute graph
|
|
|
|
for (int i = 0; i < compute_graph.size(); ++i)
|
|
|
|
{
|
|
|
|
auto ref = compute_graph[i];
|
|
|
|
log("n%d ", i);
|
|
|
|
log("%s", log_id(ref.function().name));
|
|
|
|
for (auto const ¶m : ref.function().parameters)
|
|
|
|
{
|
|
|
|
if (param.second.empty())
|
|
|
|
log("[%s]", log_id(param.first));
|
|
|
|
else
|
|
|
|
log("[%s=%s]", log_id(param.first), log_const(param.second));
|
|
|
|
}
|
|
|
|
log("(");
|
|
|
|
|
|
|
|
for (int i = 0, end = ref.size(); i != end; ++i)
|
|
|
|
{
|
|
|
|
if (i > 0)
|
|
|
|
log(", ");
|
|
|
|
log("n%d", ref.arg(i).index());
|
|
|
|
}
|
|
|
|
log(")\n");
|
|
|
|
if (ref.has_sparse_attr())
|
|
|
|
log("// wire %s\n", log_id(ref.sparse_attr()));
|
|
|
|
log("// was #%d %s\n", ref.attr(), log_signal(queue[ref.attr()]));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto const &key : compute_graph.keys())
|
|
|
|
{
|
|
|
|
log("return %d as %s \n", key.second, log_id(key.first));
|
|
|
|
}
|
2024-04-15 07:14:50 -05:00
|
|
|
}
|
|
|
|
log("Plugin test passed!\n");
|
|
|
|
}
|
|
|
|
} ExampleDtPass;
|
|
|
|
|
|
|
|
PRIVATE_NAMESPACE_END
|