Math Operations
Last updated
Last updated
All integer division rounds down to the nearest integer.
So we cannot natively rely on Solidity for rounding to the nearest integer, or similar use cases. Hence, the use of WadRayLibrary in Aave.
Provides mul and div function for wads (decimal numbers with 18 digits of precision) and rays (decimal numbers with 27 digits of precision)
Operations are rounded. If a value is >= 0.5, will be rounded up, otherwise rounded down
Rounding deals with fractional values.
Since solidity always rounds down, we can allow for rounding to the nearest integer by adding half
if remainder < 0.5 -> remainder +0.5 < 1 => rounded down to 0
However, we cannot simply apply this as:
As the rounding would have already occurred as part of the division operation. We need to repackage this such that the division occurs last:
General Form: To allow for rounding to nearest integer, add half of the divisor to the dividend
Let's connect this understanding with the implementation of wadDiv in the library.
divides two wads, rounding half wad and above up to the nearest wad.
The division occurs at:
Multiply a WAD to negate the WAD removal from division. This ensures that WAD reprsentation is adhered to.
Add half of the divisor (b / 2
) to the dividend, a
to allow for rounding to the nearest integer .
The addition of half b was explained earlier.
Overflow check
To avoid overflow, a <= (type(uint256).max - HALF_WAD) / b
The check is executed first, so that if it fails, transaction can revert early saving on gas.
The if or()
condition is required to account for the possibility that the divisor, b, is 0.
div by zero in Yul just returns 0 -> a/0 = 0. This should revert, but does not.
This condition prevents the result of the multiplication from exceeding the maximum value that can be represented by a uint256
.
Both a
and b
are passed into the function as uint256 parameters; so we do not check them.
However, some manipulations are applied to the dividend, as part of our rounding effort and WAD representation:
dividend, A
: [(a * WAD ) + (b/2)]
, for A/b
we need to ensure that A does not overflow
when using assembly, we must apply {over,under}flow checks ourselves.
Since Solidity v0.8, the compiler has checks for {over,under}flow by default for all arithmetic operations. These checks do not apply to assembly (yul) arithmetic operations.
The bitwise negation operation, denoted as not
, flips all the bits in the binary representation of the given value.
not(0)
flips all the bits of 0
, resulting in a value where all bits are set to 1
not(0)
in YUL would be equivalent to 2^256 - 1
not(0) => type(uint256).max
The multiplication occurs at:
WAD / 2
: Add half of the divisor to the dividend, to allow for rounding to the nearest integer.
Divide by WAD to negate the additional WAD from multiplication. This ensures that WAD representation is adhered to.
The above applies to rays just the same, as such we will not go through them in detail.
if remainder 0.5 -> remainder + 0.5 1 => rounded up to 1