mirror of
https://github.com/YosysHQ/yosys
synced 2025-04-24 01:25:33 +00:00
Merge pull request #4946 from povik/cost-cc-enhance
cost: Add `$mem_v2`, `$macc_v2` estimates
This commit is contained in:
commit
38beae1e06
9 changed files with 143 additions and 123 deletions
|
@ -315,7 +315,7 @@ struct ConstEval
|
|||
Macc macc;
|
||||
macc.from_cell(cell);
|
||||
|
||||
for (auto &port : macc.ports) {
|
||||
for (auto &port : macc.terms) {
|
||||
if (!eval(port.in_a, undef, cell))
|
||||
return false;
|
||||
if (!eval(port.in_b, undef, cell))
|
||||
|
|
|
@ -1,4 +1,5 @@
|
|||
#include "kernel/cost.h"
|
||||
#include "kernel/macc.h"
|
||||
|
||||
USING_YOSYS_NAMESPACE
|
||||
|
||||
|
@ -148,6 +149,9 @@ unsigned int CellCosts::get(RTLIL::Cell *cell)
|
|||
log_assert(cell->hasPort(ID::Q) && "Weird flip flop");
|
||||
log_debug("%s is ff\n", cell->name.c_str());
|
||||
return cell->getParam(ID::WIDTH).as_int();
|
||||
} else if (cell->type.in(ID($mem), ID($mem_v2))) {
|
||||
log_debug("%s is mem\n", cell->name.c_str());
|
||||
return cell->getParam(ID::WIDTH).as_int() * cell->getParam(ID::SIZE).as_int();
|
||||
} else if (y_coef(cell->type)) {
|
||||
// linear with Y_WIDTH or WIDTH
|
||||
log_assert((cell->hasParam(ID::Y_WIDTH) || cell->hasParam(ID::WIDTH)) && "Unknown width");
|
||||
|
@ -173,6 +177,22 @@ unsigned int CellCosts::get(RTLIL::Cell *cell)
|
|||
unsigned int coef = cell->type == ID($mul) ? 3 : 5;
|
||||
log_debug("%s coef*(sum**2) %d * %d\n", cell->name.c_str(), coef, sum * sum);
|
||||
return coef * sum * sum;
|
||||
} else if (cell->type.in(ID($macc), ID($macc_v2))) {
|
||||
// quadratic per term
|
||||
unsigned int cost_sum = 0;
|
||||
Macc macc;
|
||||
macc.from_cell(cell);
|
||||
unsigned int y_width = cell->getParam(ID::Y_WIDTH).as_int();
|
||||
for (auto &term: macc.terms) {
|
||||
if (term.in_b.empty()) {
|
||||
// neglect addends
|
||||
continue;
|
||||
}
|
||||
unsigned a_width = term.in_a.size(), b_width = term.in_b.size();
|
||||
unsigned int sum = a_width + b_width + std::min(y_width, a_width + b_width);
|
||||
cost_sum += 3 * sum * sum;
|
||||
}
|
||||
return cost_sum;
|
||||
} else if (cell->type == ID($lut)) {
|
||||
int width = cell->getParam(ID::WIDTH).as_int();
|
||||
unsigned int cost = 1U << (unsigned int)width;
|
||||
|
@ -187,8 +207,8 @@ unsigned int CellCosts::get(RTLIL::Cell *cell)
|
|||
log_debug("%s is free\n", cell->name.c_str());
|
||||
return 0;
|
||||
}
|
||||
// TODO: $fsm $mem.* $macc
|
||||
// ignored: $pow
|
||||
// TODO: $fsm
|
||||
// ignored: $pow $memrd $memwr $meminit (and v2 counterparts)
|
||||
|
||||
log_warning("Can't determine cost of %s cell (%d parameters).\n", log_id(cell->type), GetSize(cell->parameters));
|
||||
return 1;
|
||||
|
|
|
@ -26,18 +26,18 @@ YOSYS_NAMESPACE_BEGIN
|
|||
|
||||
struct Macc
|
||||
{
|
||||
struct port_t {
|
||||
struct term_t {
|
||||
RTLIL::SigSpec in_a, in_b;
|
||||
bool is_signed, do_subtract;
|
||||
};
|
||||
std::vector<port_t> ports;
|
||||
std::vector<term_t> terms;
|
||||
|
||||
void optimize(int width)
|
||||
{
|
||||
std::vector<port_t> new_ports;
|
||||
std::vector<term_t> new_terms;
|
||||
RTLIL::Const off(0, width);
|
||||
|
||||
for (auto &port : ports)
|
||||
for (auto &port : terms)
|
||||
{
|
||||
if (GetSize(port.in_a) == 0 && GetSize(port.in_b) == 0)
|
||||
continue;
|
||||
|
@ -68,25 +68,25 @@ struct Macc
|
|||
port.in_b.remove(GetSize(port.in_b)-1);
|
||||
}
|
||||
|
||||
new_ports.push_back(port);
|
||||
new_terms.push_back(port);
|
||||
}
|
||||
|
||||
if (off.as_bool()) {
|
||||
port_t port;
|
||||
term_t port;
|
||||
port.in_a = off;
|
||||
port.is_signed = false;
|
||||
port.do_subtract = false;
|
||||
new_ports.push_back(port);
|
||||
new_terms.push_back(port);
|
||||
}
|
||||
|
||||
new_ports.swap(ports);
|
||||
new_terms.swap(terms);
|
||||
}
|
||||
|
||||
void from_cell_v1(RTLIL::Cell *cell)
|
||||
{
|
||||
RTLIL::SigSpec port_a = cell->getPort(ID::A);
|
||||
|
||||
ports.clear();
|
||||
terms.clear();
|
||||
|
||||
auto config_bits = cell->getParam(ID::CONFIG);
|
||||
int config_cursor = 0;
|
||||
|
@ -105,7 +105,7 @@ struct Macc
|
|||
{
|
||||
log_assert(config_cursor + 2 + 2*num_bits <= config_width);
|
||||
|
||||
port_t this_port;
|
||||
term_t this_port;
|
||||
this_port.is_signed = config_bits[config_cursor++] == State::S1;
|
||||
this_port.do_subtract = config_bits[config_cursor++] == State::S1;
|
||||
|
||||
|
@ -126,11 +126,11 @@ struct Macc
|
|||
port_a_cursor += size_b;
|
||||
|
||||
if (size_a || size_b)
|
||||
ports.push_back(this_port);
|
||||
terms.push_back(this_port);
|
||||
}
|
||||
|
||||
for (auto bit : cell->getPort(ID::B))
|
||||
ports.push_back(port_t{{bit}, {}, false, false});
|
||||
terms.push_back(term_t{{bit}, {}, false, false});
|
||||
|
||||
log_assert(config_cursor == config_width);
|
||||
log_assert(port_a_cursor == GetSize(port_a));
|
||||
|
@ -148,7 +148,7 @@ struct Macc
|
|||
RTLIL::SigSpec port_b = cell->getPort(ID::B);
|
||||
RTLIL::SigSpec port_c = cell->getPort(ID::C);
|
||||
|
||||
ports.clear();
|
||||
terms.clear();
|
||||
|
||||
int nproducts = cell->getParam(ID::NPRODUCTS).as_int();
|
||||
const Const &product_neg = cell->getParam(ID::PRODUCT_NEGATED);
|
||||
|
@ -158,7 +158,7 @@ struct Macc
|
|||
const Const &b_signed = cell->getParam(ID::B_SIGNED);
|
||||
int ai = 0, bi = 0;
|
||||
for (int i = 0; i < nproducts; i++) {
|
||||
port_t term;
|
||||
term_t term;
|
||||
|
||||
log_assert(a_signed[i] == b_signed[i]);
|
||||
term.is_signed = (a_signed[i] == State::S1);
|
||||
|
@ -171,7 +171,7 @@ struct Macc
|
|||
bi += b_width;
|
||||
term.do_subtract = (product_neg[i] == State::S1);
|
||||
|
||||
ports.push_back(term);
|
||||
terms.push_back(term);
|
||||
}
|
||||
log_assert(port_a.size() == ai);
|
||||
log_assert(port_b.size() == bi);
|
||||
|
@ -182,7 +182,7 @@ struct Macc
|
|||
const Const &c_signed = cell->getParam(ID::C_SIGNED);
|
||||
int ci = 0;
|
||||
for (int i = 0; i < naddends; i++) {
|
||||
port_t term;
|
||||
term_t term;
|
||||
|
||||
term.is_signed = (c_signed[i] == State::S1);
|
||||
int c_width = c_widths.extract(16 * i, 16).as_int(false);
|
||||
|
@ -191,7 +191,7 @@ struct Macc
|
|||
ci += c_width;
|
||||
term.do_subtract = (addend_neg[i] == State::S1);
|
||||
|
||||
ports.push_back(term);
|
||||
terms.push_back(term);
|
||||
}
|
||||
log_assert(port_c.size() == ci);
|
||||
}
|
||||
|
@ -205,23 +205,23 @@ struct Macc
|
|||
Const c_signed, c_widths, addend_negated;
|
||||
SigSpec a, b, c;
|
||||
|
||||
for (int i = 0; i < (int) ports.size(); i++) {
|
||||
SigSpec term_a = ports[i].in_a, term_b = ports[i].in_b;
|
||||
for (int i = 0; i < (int) terms.size(); i++) {
|
||||
SigSpec term_a = terms[i].in_a, term_b = terms[i].in_b;
|
||||
|
||||
if (term_b.empty()) {
|
||||
// addend
|
||||
c_widths.append(Const(term_a.size(), 16));
|
||||
c_signed.append(ports[i].is_signed ? RTLIL::S1 : RTLIL::S0);
|
||||
addend_negated.append(ports[i].do_subtract ? RTLIL::S1 : RTLIL::S0);
|
||||
c_signed.append(terms[i].is_signed ? RTLIL::S1 : RTLIL::S0);
|
||||
addend_negated.append(terms[i].do_subtract ? RTLIL::S1 : RTLIL::S0);
|
||||
c.append(term_a);
|
||||
naddends++;
|
||||
} else {
|
||||
// product
|
||||
a_widths.append(Const(term_a.size(), 16));
|
||||
b_widths.append(Const(term_b.size(), 16));
|
||||
a_signed.append(ports[i].is_signed ? RTLIL::S1 : RTLIL::S0);
|
||||
b_signed.append(ports[i].is_signed ? RTLIL::S1 : RTLIL::S0);
|
||||
product_negated.append(ports[i].do_subtract ? RTLIL::S1 : RTLIL::S0);
|
||||
a_signed.append(terms[i].is_signed ? RTLIL::S1 : RTLIL::S0);
|
||||
b_signed.append(terms[i].is_signed ? RTLIL::S1 : RTLIL::S0);
|
||||
product_negated.append(terms[i].do_subtract ? RTLIL::S1 : RTLIL::S0);
|
||||
a.append(term_a);
|
||||
b.append(term_b);
|
||||
nproducts++;
|
||||
|
@ -265,7 +265,7 @@ struct Macc
|
|||
for (auto &bit : result.bits())
|
||||
bit = State::S0;
|
||||
|
||||
for (auto &port : ports)
|
||||
for (auto &port : terms)
|
||||
{
|
||||
if (!port.in_a.is_fully_const() || !port.in_b.is_fully_const())
|
||||
return false;
|
||||
|
@ -287,9 +287,9 @@ struct Macc
|
|||
|
||||
bool is_simple_product()
|
||||
{
|
||||
return ports.size() == 1 &&
|
||||
!ports[0].in_b.empty() &&
|
||||
!ports[0].do_subtract;
|
||||
return terms.size() == 1 &&
|
||||
!terms[0].in_b.empty() &&
|
||||
!terms[0].do_subtract;
|
||||
}
|
||||
|
||||
Macc(RTLIL::Cell *cell = nullptr)
|
||||
|
|
|
@ -750,7 +750,7 @@ bool SatGen::importCell(RTLIL::Cell *cell, int timestep)
|
|||
|
||||
std::vector<int> tmp(GetSize(y), ez->CONST_FALSE);
|
||||
|
||||
for (auto &port : macc.ports)
|
||||
for (auto &port : macc.terms)
|
||||
{
|
||||
std::vector<int> in_a = importDefSigSpec(port.in_a, timestep);
|
||||
std::vector<int> in_b = importDefSigSpec(port.in_b, timestep);
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue