Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Rename math operations #107

Merged
merged 7 commits into from
Dec 12, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 5 additions & 5 deletions src/AdaptiveCurveIrm.sol
Original file line number Diff line number Diff line change
Expand Up @@ -114,7 +114,7 @@ contract AdaptiveCurveIrm is IAdaptiveCurveIrm {
int256(market.totalSupplyAssets > 0 ? market.totalBorrowAssets.wDivDown(market.totalSupplyAssets) : 0);

int256 errNormFactor = utilization > TARGET_UTILIZATION ? WAD - TARGET_UTILIZATION : TARGET_UTILIZATION;
int256 err = (utilization - TARGET_UTILIZATION).wDivDown(errNormFactor);
int256 err = (utilization - TARGET_UTILIZATION).wDiv(errNormFactor);

int256 startRateAtTarget = rateAtTarget[id];

Expand All @@ -128,7 +128,7 @@ contract AdaptiveCurveIrm is IAdaptiveCurveIrm {
} else {
// Note that the speed is assumed constant between two interactions, but in theory it increases because of
// interests. So the rate will be slightly underestimated.
int256 speed = ADJUSTMENT_SPEED.wMulDown(err);
int256 speed = ADJUSTMENT_SPEED.wMul(err);
// market.lastUpdate != 0 because it is not the first interaction with this market.
// Safe "unchecked" cast because block.timestamp - market.lastUpdate <= block.timestamp <= type(int256).max.
int256 elapsed = int256(block.timestamp - market.lastUpdate);
Expand Down Expand Up @@ -168,16 +168,16 @@ contract AdaptiveCurveIrm is IAdaptiveCurveIrm {
/// ((C-1)*err + 1) * rateAtTarget else.
function _curve(int256 _rateAtTarget, int256 err) private view returns (int256) {
// Non negative because 1 - 1/C >= 0, C - 1 >= 0.
int256 coeff = err < 0 ? WAD - WAD.wDivDown(CURVE_STEEPNESS) : CURVE_STEEPNESS - WAD;
int256 coeff = err < 0 ? WAD - WAD.wDiv(CURVE_STEEPNESS) : CURVE_STEEPNESS - WAD;
// Non negative if _rateAtTarget >= 0 because if err < 0, coeff <= 1.
return (coeff.wMulDown(err) + WAD).wMulDown(int256(_rateAtTarget));
return (coeff.wMul(err) + WAD).wMul(int256(_rateAtTarget));
}

/// @dev Returns the new rate at target, for a given `startRateAtTarget` and a given `linearAdaptation`.
/// The formula is: max(min(startRateAtTarget * exp(linearAdaptation), maxRateAtTarget), minRateAtTarget).
function _newRateAtTarget(int256 startRateAtTarget, int256 linearAdaptation) private pure returns (int256) {
// Non negative because MIN_RATE_AT_TARGET > 0.
return startRateAtTarget.wMulDown(ExpLib.wExp(linearAdaptation)).bound(
return startRateAtTarget.wMul(ExpLib.wExp(linearAdaptation)).bound(
ConstantsLib.MIN_RATE_AT_TARGET, ConstantsLib.MAX_RATE_AT_TARGET
);
}
Expand Down
6 changes: 4 additions & 2 deletions src/libraries/MathLib.sol
Original file line number Diff line number Diff line change
Expand Up @@ -10,11 +10,13 @@ int256 constant WAD_INT = int256(WAD);
/// @custom:contact security@morpho.org
/// @notice Library to manage fixed-point arithmetic on signed integers.
library MathLib {
function wMulDown(int256 a, int256 b) internal pure returns (int256) {
/// @dev Returns the multiplication of `a` by `b` (in WAD) rounded towards 0.
function wMul(int256 a, int256 b) internal pure returns (int256) {
MathisGD marked this conversation as resolved.
Show resolved Hide resolved
return a * b / WAD_INT;
}

function wDivDown(int256 a, int256 b) internal pure returns (int256) {
/// @dev Returns the division of `a` by `b` (in WAD) rounded towards 0.
function wDiv(int256 a, int256 b) internal pure returns (int256) {
return a * WAD_INT / b;
}
}
41 changes: 16 additions & 25 deletions test/AdaptiveCurveIrmTest.sol
Original file line number Diff line number Diff line change
Expand Up @@ -72,13 +72,13 @@ contract AdaptiveCurveIrmTest is Test {
// (exp((50/365)*5) ~= 1.9836.
assertApproxEqRel(
irm.borrowRateView(marketParams, market),
uint256((INITIAL_RATE_AT_TARGET * 4).wMulDown((1.9836 ether - 1 ether) * WAD / (ADJUSTMENT_SPEED * 5 days))),
uint256((INITIAL_RATE_AT_TARGET * 4).wMul((1.9836 ether - 1 ether) * WAD / (ADJUSTMENT_SPEED * 5 days))),
0.1 ether
);
// The average value of exp((50/365)*x) between 0 and 5 is approx. 1.4361.
assertApproxEqRel(
irm.borrowRateView(marketParams, market),
uint256((INITIAL_RATE_AT_TARGET * 4).wMulDown(1.4361 ether)),
uint256((INITIAL_RATE_AT_TARGET * 4).wMul(1.4361 ether)),
0.1 ether
);
// Expected rate: 5.744%.
Expand All @@ -97,16 +97,12 @@ contract AdaptiveCurveIrmTest is Test {
// (exp((-50/365)*5) ~= 0.5041.
assertApproxEqRel(
irm.borrowRateView(marketParams, market),
uint256(
(INITIAL_RATE_AT_TARGET / 4).wMulDown((0.5041 ether - 1 ether) * WAD / (-ADJUSTMENT_SPEED * 5 days))
),
uint256((INITIAL_RATE_AT_TARGET / 4).wMul((0.5041 ether - 1 ether) * WAD / (-ADJUSTMENT_SPEED * 5 days))),
0.1 ether
);
// The average value of exp((-50/365*x)) between 0 and 5 is approx. 0.7240.
assertApproxEqRel(
irm.borrowRateView(marketParams, market),
uint256((INITIAL_RATE_AT_TARGET / 4).wMulDown(0.724 ether)),
0.1 ether
irm.borrowRateView(marketParams, market), uint256((INITIAL_RATE_AT_TARGET / 4).wMul(0.724 ether)), 0.1 ether
);
// Expected rate: 0.181%.
assertApproxEqRel(irm.borrowRateView(marketParams, market), uint256(0.00181 ether) / 365 days, 0.1 ether);
Expand Down Expand Up @@ -337,11 +333,9 @@ contract AdaptiveCurveIrmTest is Test {
market.totalSupplyAssets = 10 ether;

assertGe(
irm.borrowRateView(marketParams, market), uint256(ConstantsLib.MIN_RATE_AT_TARGET.wDivDown(CURVE_STEEPNESS))
);
assertGe(
irm.borrowRate(marketParams, market), uint256(ConstantsLib.MIN_RATE_AT_TARGET.wDivDown(CURVE_STEEPNESS))
irm.borrowRateView(marketParams, market), uint256(ConstantsLib.MIN_RATE_AT_TARGET.wDiv(CURVE_STEEPNESS))
);
assertGe(irm.borrowRate(marketParams, market), uint256(ConstantsLib.MIN_RATE_AT_TARGET.wDiv(CURVE_STEEPNESS)));
}

function invariantLeMaxRateAtTarget() public {
Expand All @@ -350,23 +344,21 @@ contract AdaptiveCurveIrmTest is Test {
market.totalSupplyAssets = 10 ether;

assertLe(
irm.borrowRateView(marketParams, market), uint256(ConstantsLib.MAX_RATE_AT_TARGET.wMulDown(CURVE_STEEPNESS))
);
assertLe(
irm.borrowRate(marketParams, market), uint256(ConstantsLib.MAX_RATE_AT_TARGET.wMulDown(CURVE_STEEPNESS))
irm.borrowRateView(marketParams, market), uint256(ConstantsLib.MAX_RATE_AT_TARGET.wMul(CURVE_STEEPNESS))
);
assertLe(irm.borrowRate(marketParams, market), uint256(ConstantsLib.MAX_RATE_AT_TARGET.wMul(CURVE_STEEPNESS)));
}

/* HELPERS */

function _expectedRateAtTarget(Id id, Market memory market) internal view returns (int256) {
int256 rateAtTarget = irm.rateAtTarget(id);
int256 speed = ADJUSTMENT_SPEED.wMulDown(_err(market));
int256 speed = ADJUSTMENT_SPEED.wMul(_err(market));
uint256 elapsed = (rateAtTarget > 0) ? block.timestamp - market.lastUpdate : 0;
int256 linearAdaptation = speed * int256(elapsed);
int256 adaptationMultiplier = ExpLib.wExp(linearAdaptation);
return (rateAtTarget > 0)
? rateAtTarget.wMulDown(adaptationMultiplier).bound(
? rateAtTarget.wMul(adaptationMultiplier).bound(
ConstantsLib.MIN_RATE_AT_TARGET, ConstantsLib.MAX_RATE_AT_TARGET
)
: INITIAL_RATE_AT_TARGET;
Expand All @@ -375,7 +367,7 @@ contract AdaptiveCurveIrmTest is Test {
function _expectedAvgRate(Id id, Market memory market) internal view returns (uint256) {
int256 rateAtTarget = irm.rateAtTarget(id);
int256 err = _err(market);
int256 speed = ADJUSTMENT_SPEED.wMulDown(err);
int256 speed = ADJUSTMENT_SPEED.wMul(err);
uint256 elapsed = (rateAtTarget > 0) ? block.timestamp - market.lastUpdate : 0;
int256 linearAdaptation = speed * int256(elapsed);
int256 endRateAtTarget = int256(_expectedRateAtTarget(id, market));
Expand All @@ -386,18 +378,17 @@ contract AdaptiveCurveIrmTest is Test {
avgBorrowRate = newBorrowRate;
} else {
// Safe "unchecked" cast to uint256 because linearAdaptation < 0 <=> newBorrowRate <= borrowRateAfterJump.
avgBorrowRate =
uint256((int256(newBorrowRate) - int256(_curve(rateAtTarget, err))).wDivDown(linearAdaptation));
avgBorrowRate = uint256((int256(newBorrowRate) - int256(_curve(rateAtTarget, err))).wDiv(linearAdaptation));
}
return avgBorrowRate;
}

function _curve(int256 rateAtTarget, int256 err) internal pure returns (uint256) {
// Safe "unchecked" cast because err >= -1 (in WAD).
if (err < 0) {
return uint256(((WAD - WAD.wDivDown(CURVE_STEEPNESS)).wMulDown(err) + WAD).wMulDown(rateAtTarget));
return uint256(((WAD - WAD.wDiv(CURVE_STEEPNESS)).wMul(err) + WAD).wMul(rateAtTarget));
} else {
return uint256(((CURVE_STEEPNESS - WAD).wMulDown(err) + WAD).wMulDown(rateAtTarget));
return uint256(((CURVE_STEEPNESS - WAD).wMul(err) + WAD).wMul(rateAtTarget));
}
}

Expand All @@ -407,9 +398,9 @@ contract AdaptiveCurveIrmTest is Test {
int256 utilization = int256(market.totalBorrowAssets.wDivDown(market.totalSupplyAssets));

if (utilization > TARGET_UTILIZATION) {
err = (utilization - TARGET_UTILIZATION).wDivDown(WAD - TARGET_UTILIZATION);
err = (utilization - TARGET_UTILIZATION).wDiv(WAD - TARGET_UTILIZATION);
} else {
err = (utilization - TARGET_UTILIZATION).wDivDown(TARGET_UTILIZATION);
err = (utilization - TARGET_UTILIZATION).wDiv(TARGET_UTILIZATION);
}
}
}
4 changes: 2 additions & 2 deletions test/ExpLibTest.sol
Original file line number Diff line number Diff line change
Expand Up @@ -62,8 +62,8 @@ contract ExpLibTest is Test {
assertLe(ExpLib.wExp(x), 1e18);
}

function testWExpWMulDownMaxRate() public pure {
ExpLib.wExp(ExpLib.WEXP_UPPER_BOUND).wMulDown(ConstantsLib.MAX_RATE_AT_TARGET);
function testWExpWMulMaxRate() public pure {
ExpLib.wExp(ExpLib.WEXP_UPPER_BOUND).wMul(ConstantsLib.MAX_RATE_AT_TARGET);
}

function _wExpUnbounded(int256 x) internal pure returns (int256) {
Expand Down