2019-04-29 06:38:56 -05:00
|
|
|
pattern shiftmul
|
2019-05-28 08:33:47 -05:00
|
|
|
//
|
|
|
|
// Optimize mul+shift pairs that result from expressions such as foo[s*W+:W]
|
|
|
|
//
|
2019-04-29 06:38:56 -05:00
|
|
|
|
|
|
|
state <SigSpec> shamt
|
|
|
|
|
|
|
|
match shift
|
|
|
|
select shift->type.in($shift, $shiftx, $shr)
|
|
|
|
endmatch
|
|
|
|
|
|
|
|
code shamt
|
|
|
|
shamt = port(shift, \B);
|
2019-05-06 08:34:19 -05:00
|
|
|
if (shamt.empty())
|
|
|
|
reject;
|
2019-04-29 06:38:56 -05:00
|
|
|
if (shamt[GetSize(shamt)-1] == State::S0) {
|
|
|
|
do {
|
|
|
|
shamt.remove(GetSize(shamt)-1);
|
2019-05-06 08:34:19 -05:00
|
|
|
if (shamt.empty())
|
|
|
|
reject;
|
2019-04-29 06:38:56 -05:00
|
|
|
} while (shamt[GetSize(shamt)-1] == State::S0);
|
|
|
|
} else
|
2019-04-30 00:59:39 -05:00
|
|
|
if (shift->type.in($shift, $shiftx) && param(shift, \B_SIGNED).as_bool()) {
|
2019-04-29 06:38:56 -05:00
|
|
|
reject;
|
|
|
|
}
|
2019-04-30 00:59:39 -05:00
|
|
|
if (GetSize(shamt) > 20)
|
|
|
|
reject;
|
2019-04-29 06:38:56 -05:00
|
|
|
endcode
|
|
|
|
|
|
|
|
match mul
|
|
|
|
select mul->type.in($mul)
|
|
|
|
select port(mul, \A).is_fully_const() || port(mul, \B).is_fully_const()
|
|
|
|
index <SigSpec> port(mul, \Y) === shamt
|
2020-08-05 14:01:20 -05:00
|
|
|
filter !param(mul, \A_SIGNED).as_bool()
|
2019-04-29 06:38:56 -05:00
|
|
|
endmatch
|
|
|
|
|
|
|
|
code
|
2019-08-15 11:34:36 -05:00
|
|
|
{
|
2019-04-29 06:38:56 -05:00
|
|
|
IdString const_factor_port = port(mul, \A).is_fully_const() ? \A : \B;
|
2019-04-30 00:59:39 -05:00
|
|
|
Const const_factor_cnst = port(mul, const_factor_port).as_const();
|
|
|
|
int const_factor = const_factor_cnst.as_int();
|
|
|
|
|
|
|
|
if (GetSize(const_factor_cnst) == 0)
|
|
|
|
reject;
|
|
|
|
|
|
|
|
if (GetSize(const_factor_cnst) > 20)
|
|
|
|
reject;
|
|
|
|
|
2019-04-29 06:38:56 -05:00
|
|
|
if (GetSize(port(shift, \Y)) > const_factor)
|
|
|
|
reject;
|
2019-04-30 00:59:39 -05:00
|
|
|
|
2019-05-28 08:33:47 -05:00
|
|
|
int factor_bits = ceil_log2(const_factor);
|
|
|
|
SigSpec mul_din = port(mul, const_factor_port == \A ? \B : \A);
|
|
|
|
|
|
|
|
if (GetSize(shamt) < factor_bits+GetSize(mul_din))
|
|
|
|
reject;
|
|
|
|
|
2019-04-30 03:51:51 -05:00
|
|
|
did_something = true;
|
2019-04-30 00:59:39 -05:00
|
|
|
log("shiftmul pattern in %s: shift=%s, mul=%s\n", log_id(module), log_id(shift), log_id(mul));
|
|
|
|
|
2019-05-28 08:33:47 -05:00
|
|
|
int new_const_factor = 1 << factor_bits;
|
2019-04-30 00:59:39 -05:00
|
|
|
SigSpec padding(State::Sx, new_const_factor-const_factor);
|
|
|
|
SigSpec old_a = port(shift, \A), new_a;
|
|
|
|
int trunc = 0;
|
|
|
|
|
|
|
|
if (GetSize(old_a) % const_factor != 0) {
|
|
|
|
trunc = const_factor - GetSize(old_a) % const_factor;
|
|
|
|
old_a.append(SigSpec(State::Sx, trunc));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i*const_factor < GetSize(old_a); i++) {
|
|
|
|
SigSpec slice = old_a.extract(i*const_factor, const_factor);
|
|
|
|
new_a.append(slice);
|
|
|
|
new_a.append(padding);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (trunc > 0)
|
|
|
|
new_a.remove(GetSize(new_a)-trunc, trunc);
|
|
|
|
|
2019-05-28 08:33:47 -05:00
|
|
|
SigSpec new_b = {mul_din, SigSpec(State::S0, factor_bits)};
|
2019-04-30 00:59:39 -05:00
|
|
|
if (param(shift, \B_SIGNED).as_bool())
|
|
|
|
new_b.append(State::S0);
|
|
|
|
|
|
|
|
shift->setPort(\A, new_a);
|
|
|
|
shift->setParam(\A_WIDTH, GetSize(new_a));
|
|
|
|
shift->setPort(\B, new_b);
|
|
|
|
shift->setParam(\B_WIDTH, GetSize(new_b));
|
|
|
|
|
|
|
|
blacklist(shift);
|
2019-08-15 15:47:59 -05:00
|
|
|
accept;
|
2019-08-15 11:34:36 -05:00
|
|
|
}
|
2019-04-29 06:38:56 -05:00
|
|
|
endcode
|