FloatingPointAdderSimple constructor
- FloatingPoint a,
- FloatingPoint b, {
- ParallelPrefix ppGen() = KoggeStone.new,
- String name = 'floatingpoint_adder_simple',
Add two floating point numbers a
and b
, returning result in sum
Implementation
FloatingPointAdderSimple(FloatingPoint a, FloatingPoint b,
{ParallelPrefix Function(List<Logic>, Logic Function(Logic, Logic))
ppGen = KoggeStone.new,
super.name = 'floatingpoint_adder_simple'})
: exponentWidth = a.exponent.width,
mantissaWidth = a.mantissa.width {
if (b.exponent.width != exponentWidth ||
b.mantissa.width != mantissaWidth) {
throw RohdHclException('FloatingPoint widths must match');
}
a = a.clone()..gets(addInput('a', a, width: a.width));
b = b.clone()..gets(addInput('b', b, width: b.width));
addOutput('sum', width: _sum.width) <= _sum;
// Ensure that the larger number is wired as 'a'
final doSwap = a.exponent.lt(b.exponent) |
(a.exponent.eq(b.exponent) & a.mantissa.lt(b.mantissa)) |
((a.exponent.eq(b.exponent) & a.mantissa.eq(b.mantissa)) & b.sign);
(a, b) = _swap(doSwap, (a, b));
final aExp =
a.exponent + mux(a.isNormal(), a.zeroExponent(), a.oneExponent());
final bExp =
b.exponent + mux(b.isNormal(), b.zeroExponent(), b.oneExponent());
// Align and add mantissas
final expDiff = aExp - bExp;
final adder = SignMagnitudeAdder(
a.sign,
[a.isNormal(), a.mantissa].swizzle(),
b.sign,
[b.isNormal(), b.mantissa].swizzle() >>> expDiff,
(a, b, {carryIn}) =>
ParallelPrefixAdder(a, b, carryIn: carryIn, ppGen: ppGen));
final sum = adder.sum.slice(adder.sum.width - 2, 0);
final leadOneE =
ParallelPrefixPriorityEncoder(sum.reversed, ppGen: ppGen).out;
final leadOne = leadOneE.zeroExtend(exponentWidth);
// Assemble the output FloatingPoint
_sum.sign <= adder.sign;
Combinational([
If.block([
Iff(adder.sum[-1] & a.sign.eq(b.sign), [
_sum.mantissa < (sum >> 1).slice(mantissaWidth - 1, 0),
_sum.exponent < a.exponent + 1
]),
ElseIf(a.exponent.gt(leadOne) & sum.or(), [
_sum.mantissa < (sum << leadOne).slice(mantissaWidth - 1, 0),
_sum.exponent < a.exponent - leadOne
]),
ElseIf(leadOne.eq(0) & sum.or(), [
_sum.mantissa < (sum << leadOne).slice(mantissaWidth - 1, 0),
_sum.exponent < a.exponent - leadOne + 1
]),
Else([
// subnormal result
_sum.mantissa < sum.slice(mantissaWidth - 1, 0),
_sum.exponent < _sum.zeroExponent()
])
])
]);
}