From 9042a072cea4ede57c39eff02950e3e668a6add9 Mon Sep 17 00:00:00 2001 From: Marc Doerflinger Date: Fri, 30 Aug 2024 07:00:34 +0000 Subject: [PATCH] convert Fixed to 160-bit (closes #652) --- contracts/type/UFixed.sol | 37 ++++-- test/staking/RewardCalculation.t.sol | 2 +- test/type/UFixed.t.sol | 170 ++++++++++++++------------- 3 files changed, 119 insertions(+), 90 deletions(-) diff --git a/contracts/type/UFixed.sol b/contracts/type/UFixed.sol index 952299287..47de66546 100644 --- a/contracts/type/UFixed.sol +++ b/contracts/type/UFixed.sol @@ -3,8 +3,8 @@ pragma solidity ^0.8.20; import {Math} from "@openzeppelin/contracts/utils/math/Math.sol"; -/// @dev UFixed is a fixed point number with 18 decimals precision. -type UFixed is uint256; // TODO: change to 128-bit for storage optimization +/// @dev UFixed is a 160-bit fixed point number with 15 decimals precision. +type UFixed is uint160; using { addUFixed as +, @@ -41,7 +41,7 @@ function subUFixed(UFixed a, UFixed b) pure returns (UFixed) { function mulUFixed(UFixed a, UFixed b) pure returns (UFixed) { return - UFixed.wrap(Math.mulDiv(UFixed.unwrap(a), UFixed.unwrap(b), 10 ** 18)); + UFixed.wrap(uint160(Math.mulDiv(UFixed.unwrap(a), UFixed.unwrap(b), 10 ** 15))); } function divUFixed(UFixed a, UFixed b) pure returns (UFixed) { @@ -50,7 +50,7 @@ function divUFixed(UFixed a, UFixed b) pure returns (UFixed) { } return - UFixed.wrap(Math.mulDiv(UFixed.unwrap(a), 10 ** 18, UFixed.unwrap(b))); + UFixed.wrap(uint160(Math.mulDiv(UFixed.unwrap(a), 10 ** 15, UFixed.unwrap(b)))); } function gtUFixed(UFixed a, UFixed b) pure returns (bool isGreaterThan) { @@ -100,7 +100,9 @@ library UFixedLib { error UFixedLibExponentTooSmall(int8 exp); error UFixedLibExponentTooLarge(int8 exp); - int8 public constant EXP = 18; + error UFixedLibNumberTooLarge(uint256 number); + + int8 public constant EXP = 15; uint256 public constant MULTIPLIER = 10 ** uint256(int256(EXP)); uint256 public constant MULTIPLIER_HALF = MULTIPLIER / 2; @@ -119,9 +121,14 @@ library UFixedLib { return uint8(2); } - /// @dev Converts the uint256 to a UFixed. + /// @dev Converts the uint256 to a uint160 based UFixed. + /// This method reverts if the number is too large to fit in a uint160. function toUFixed(uint256 a) public pure returns (UFixed) { - return UFixed.wrap(a * MULTIPLIER); + uint256 n = a * MULTIPLIER; + if (n > type(uint160).max) { + revert UFixedLibNumberTooLarge(a); + } + return UFixed.wrap(uint160(n)); } /// @dev Converts the uint256 to a UFixed with given exponent. @@ -129,11 +136,17 @@ library UFixedLib { if (EXP + exp < 0) { revert UFixedLibExponentTooSmall(exp); } - if (EXP + exp > 64) { + if (EXP + exp > 48) { revert UFixedLibExponentTooLarge(exp); } - return UFixed.wrap(a * 10 ** uint8(EXP + exp)); + uint256 n = a * 10 ** uint8(EXP + exp); + + if (n > type(uint160).max) { + revert UFixedLibNumberTooLarge(n); + } + + return UFixed.wrap(uint160(n)); } /// @dev returns the decimals precision of the UFixed type @@ -235,7 +248,11 @@ library UFixedLib { } function one() public pure returns (UFixed) { - return UFixed.wrap(MULTIPLIER); + return UFixed.wrap(uint160(MULTIPLIER)); + } + + function max() public pure returns (UFixed) { + return UFixed.wrap(type(uint160).max); } /// @dev return the absolute delta between two UFixed numbers diff --git a/test/staking/RewardCalculation.t.sol b/test/staking/RewardCalculation.t.sol index 23e3a5a38..78000704f 100644 --- a/test/staking/RewardCalculation.t.sol +++ b/test/staking/RewardCalculation.t.sol @@ -59,7 +59,7 @@ contract RewardCalculation is GifTest { Seconds oneHourDuration = SecondsLib.toSeconds(3600); // 1 / (365 * 24) = 0.00011415525114155251 - UFixed oneHFraction = UFixedLib.toUFixed(114155251141, -15); + UFixed oneHFraction = UFixedLib.toUFixed(114155251, -12); UFixed fractionOneH = StakingLib.getYearFraction(oneHourDuration); assertEq(_times1e9(fractionOneH), 114155, "unexpected 1h fraction (x1e9)"); assertTrue(UFixedLib.delta(fractionOneH, oneHFraction) < epsilon, "unexpected 1h fraction (equals)"); diff --git a/test/type/UFixed.t.sol b/test/type/UFixed.t.sol index 1e07b5623..a763a2c25 100644 --- a/test/type/UFixed.t.sol +++ b/test/type/UFixed.t.sol @@ -8,32 +8,40 @@ contract UFixedTest is Test { using UFixedLib for UFixed; function testTestDecimals() public { - assertEq(UFixedLib.decimals(), 18); + assertEq(UFixedLib.decimals(), 15); } function testOpEqual() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(1 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(1 * 10 ** 15); assertTrue(a == b); - UFixed c = UFixed.wrap(2 * 10 ** 18); + UFixed c = UFixed.wrap(2 * 10 ** 15); assertFalse(a == c); } function testUFixedMathLib() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(1 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(1 * 10 ** 15); assertTrue(UFixedLib.eq(a, b)); - UFixed c = UFixed.wrap(2 * 10 ** 18); + UFixed c = UFixed.wrap(2 * 10 ** 15); assertFalse(UFixedLib.eq(a, c)); } function testItof() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); assertTrue(a == UFixedLib.toUFixed(1)); } + function testItof_max() public { + uint256 n = UFixedLib.max().toInt() + 1; + vm.expectRevert(abi.encodeWithSelector( + UFixedLib.UFixedLibNumberTooLarge.selector, + n)); + UFixedLib.toUFixed(n); + } + function testItofExp() public { UFixed a = UFixedLib.toUFixed(1, 2); assertTrue(a.toInt() == 100); @@ -43,47 +51,51 @@ contract UFixedTest is Test { assertTrue(b.toInt() == 1); // smalltest possible value - UFixedLib.toUFixed(1, -18); + UFixedLib.toUFixed(1, -15); // one order of magnitude smaller reverts - vm.expectRevert(abi.encodeWithSelector(UFixedLib.UFixedLibExponentTooSmall.selector, -19)); - UFixedLib.toUFixed(1, -19); + vm.expectRevert(abi.encodeWithSelector(UFixedLib.UFixedLibExponentTooSmall.selector, -16)); + UFixedLib.toUFixed(1, -16); - // largest possible value -- 10 ** 46 (64 - EXP(18)) + // largest possible value -- 10 ** 33 (48 - EXP(15)) assertTrue( - UFixedLib.toUFixed(1, 46) == UFixedLib.toUFixed(1 * 10 ** 46) + UFixedLib.toUFixed(1, 33) == UFixedLib.toUFixed(1 * 10 ** 33) ); // one order of magnitude larger reverts - vm.expectRevert(abi.encodeWithSelector(UFixedLib.UFixedLibExponentTooLarge.selector, 47)); - UFixedLib.toUFixed(1, 64 - 18 + 1); + vm.expectRevert(abi.encodeWithSelector(UFixedLib.UFixedLibExponentTooLarge.selector, 34)); + UFixedLib.toUFixed(1, 48 - 15 + 1); + + // resulting number is too large + vm.expectRevert(abi.encodeWithSelector(UFixedLib.UFixedLibNumberTooLarge.selector, (1 * 10 ** 30 + 1) * 10 ** (15 + 4))); + UFixedLib.toUFixed(1 * 10 ** 30 + 1, 4); } function testFtoi() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); assertTrue(a.toInt() == 1); } function testFtoiRounding() public { - UFixed a = UFixed.wrap(4 * 10 ** 17); + UFixed a = UFixed.wrap(4 * 10 ** 14); assertTrue(UFixedLib.toIntWithRounding(a, UFixedLib.ROUNDING_UP()) == 1); assertTrue(UFixedLib.toIntWithRounding(a, UFixedLib.ROUNDING_DOWN()) == 0); assertTrue(UFixedLib.toIntWithRounding(a, UFixedLib.ROUNDING_HALF_UP()) == 0); - UFixed b = UFixed.wrap(5 * 10 ** 17); + UFixed b = UFixed.wrap(5 * 10 ** 14); assertTrue(UFixedLib.toIntWithRounding(b, UFixedLib.ROUNDING_UP()) == 1); assertTrue(UFixedLib.toIntWithRounding(b, UFixedLib.ROUNDING_DOWN()) == 0); assertTrue(UFixedLib.toIntWithRounding(b, UFixedLib.ROUNDING_HALF_UP()) == 1); - UFixed c = UFixed.wrap(6 * 10 ** 17); + UFixed c = UFixed.wrap(6 * 10 ** 14); assertTrue(UFixedLib.toIntWithRounding(c, UFixedLib.ROUNDING_UP()) == 1); assertTrue(UFixedLib.toIntWithRounding(c, UFixedLib.ROUNDING_DOWN()) == 0); assertTrue(UFixedLib.toIntWithRounding(c, UFixedLib.ROUNDING_HALF_UP()) == 1); } function testOpAdd() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(1 * 10 ** 18); - UFixed c = UFixed.wrap(2 * 10 ** 18); - UFixed d = UFixed.wrap(3 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(1 * 10 ** 15); + UFixed c = UFixed.wrap(2 * 10 ** 15); + UFixed d = UFixed.wrap(3 * 10 ** 15); assertTrue((a + b) == c); assertTrue(UFixedLib.add(a, b) == c); assertFalse((a + b) == d); @@ -92,7 +104,7 @@ contract UFixedTest is Test { assertTrue((a + c) == d); assertTrue(UFixedLib.add(a, c) == d); - UFixed e = UFixed.wrap(0 * 10 ** 18); + UFixed e = UFixed.wrap(0 * 10 ** 15); assertTrue((a + e) == a); assertTrue(UFixedLib.add(a, e) == a); assertTrue((e + e) == e); @@ -100,10 +112,10 @@ contract UFixedTest is Test { } function testOpSub() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(1 * 10 ** 18); - UFixed c = UFixed.wrap(2 * 10 ** 18); - UFixed d = UFixed.wrap(3 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(1 * 10 ** 15); + UFixed c = UFixed.wrap(2 * 10 ** 15); + UFixed d = UFixed.wrap(3 * 10 ** 15); assertTrue((c - b) == a); assertTrue(UFixedLib.sub(c, b) == a); @@ -112,7 +124,7 @@ contract UFixedTest is Test { assertFalse((d - b) == b); assertFalse(UFixedLib.sub(d, b) == b); - UFixed e = UFixed.wrap(0 * 10 ** 18); + UFixed e = UFixed.wrap(0 * 10 ** 15); assertTrue((a - a) == e); assertTrue(UFixedLib.sub(a, a) == e); assertTrue((a - e) == a); @@ -128,35 +140,35 @@ contract UFixedTest is Test { function testOpMul() public { // 1 * 1 = 1 - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(1 * 10 ** 18); - UFixed c = UFixed.wrap(1 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(1 * 10 ** 15); + UFixed c = UFixed.wrap(1 * 10 ** 15); assertTrue((a * b) == c); assertTrue(a.mul(b).eq(c)); // 1 * 2 = 2 - UFixed d = UFixed.wrap(2 * 10 ** 18); + UFixed d = UFixed.wrap(2 * 10 ** 15); assertTrue((a * d) == d); assertTrue(a.mul(d).eq(d)); // 2 * 2 = 4 - UFixed e = UFixed.wrap(4 * 10 ** 18); + UFixed e = UFixed.wrap(4 * 10 ** 15); assertTrue((d * d) == e); assertTrue(d.mul(d).eq(e)); assertFalse((a * d) == e); assertFalse(a.mul(d).eq(e)); // 2 * 21 = 42 - UFixed f = UFixed.wrap(21 * 10 ** 18); - UFixed g = UFixed.wrap(42 * 10 ** 18); + UFixed f = UFixed.wrap(21 * 10 ** 15); + UFixed g = UFixed.wrap(42 * 10 ** 15); assertTrue((d * f) == g); assertTrue(d.mul(f).eq(g)); } function testOpMulFrac() public { // 1 * 0.5 = 0.5 - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(5 * 10 ** 17); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(5 * 10 ** 14); assertTrue((a * b) == b); assertTrue((a.mul(b)).eq(b)); @@ -164,7 +176,7 @@ contract UFixedTest is Test { assertTrue((b.mul(a)).eq(b)); // 0.5 * 0.5 = 0.25 - UFixed c = UFixed.wrap(25 * 10 ** 16); + UFixed c = UFixed.wrap(25 * 10 ** 13); assertTrue((b * b) == c); assertTrue((b.mul(b)).eq(c)); } @@ -185,11 +197,11 @@ contract UFixedTest is Test { } function testOpMulZero() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(1 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(1 * 10 ** 15); // 1 * 0 = 0 - UFixed z = UFixed.wrap(0 * 10 ** 18); + UFixed z = UFixed.wrap(0 * 10 ** 15); assertTrue((a * z) == z); assertTrue((a.mul(z)).eq(z)); @@ -206,14 +218,14 @@ contract UFixedTest is Test { function testOpDiv() public { // 1 / 1 = 1 - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(1 * 10 ** 18); - UFixed c = UFixed.wrap(1 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(1 * 10 ** 15); + UFixed c = UFixed.wrap(1 * 10 ** 15); assertTrue((a / b) == c); assertTrue(a.div(b).eq(c)); // 2 / 1 = 2 - UFixed d = UFixed.wrap(2 * 10 ** 18); + UFixed d = UFixed.wrap(2 * 10 ** 15); assertTrue((d / a) == d); assertTrue(d.div(a).eq(d)); @@ -222,46 +234,46 @@ contract UFixedTest is Test { assertTrue(d.div(d).eq(b)); // 4 / 2 = 2 - UFixed e = UFixed.wrap(4 * 10 ** 18); + UFixed e = UFixed.wrap(4 * 10 ** 15); assertTrue((e / d) == d); assertTrue(e.div(d).eq(d)); // 42 / 2 = 21 - UFixed f = UFixed.wrap(21 * 10 ** 18); - UFixed g = UFixed.wrap(42 * 10 ** 18); + UFixed f = UFixed.wrap(21 * 10 ** 15); + UFixed g = UFixed.wrap(42 * 10 ** 15); assertTrue((g / d) == f); assertTrue(g.div(d).eq(f)); } function testOpDivFrac() public { - UFixed d = UFixed.wrap(2 * 10 ** 18); + UFixed d = UFixed.wrap(2 * 10 ** 15); // 5 / 2 = 2.5 - UFixed f1 = UFixed.wrap(5 * 10 ** 18); - UFixed ex1 = UFixed.wrap(2.5 * 10 ** 18); + UFixed f1 = UFixed.wrap(5 * 10 ** 15); + UFixed ex1 = UFixed.wrap(2.5 * 10 ** 15); assertTrue((f1 / d) == ex1); assertTrue(f1.div(d).eq(ex1)); // 2 / 5 = 0.4 - UFixed ex2 = UFixed.wrap(0.4 * 10 ** 18); + UFixed ex2 = UFixed.wrap(0.4 * 10 ** 15); assertTrue((d / f1) == ex2); assertTrue(d.div(f1).eq(ex2)); // 2 / 0.5 = 4 - UFixed f2 = UFixed.wrap(5 * 10 ** 17); - UFixed ex3 = UFixed.wrap(4 * 10 ** 18); + UFixed f2 = UFixed.wrap(5 * 10 ** 14); + UFixed ex3 = UFixed.wrap(4 * 10 ** 15); assertTrue((d / f2) == ex3); assertTrue(d.div(f2).eq(ex3)); // 0.5 / 2 = 0.25 - UFixed ex4 = UFixed.wrap(25 * 10 ** 16); + UFixed ex4 = UFixed.wrap(25 * 10 ** 13); assertTrue((f2 / d) == ex4); assertTrue(f2.div(d).eq(ex4)); } function testOpDivBig() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed d = UFixed.wrap(2 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed d = UFixed.wrap(2 * 10 ** 15); // bigUFixed / 1 = bigUFixed // bigUFixed = 1 * 10 ** 31 @@ -277,10 +289,10 @@ contract UFixedTest is Test { } function testOpDivZero() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); // 0 / 1 = 0 - UFixed z = UFixed.wrap(0 * 10 ** 18); + UFixed z = UFixed.wrap(0 * 10 ** 15); assertTrue((z / a) == z); assertTrue(z.div(a).eq(z)); @@ -298,9 +310,9 @@ contract UFixedTest is Test { } function testOpGt() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(1 * 10 ** 18); - UFixed c = UFixed.wrap(2 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(1 * 10 ** 15); + UFixed c = UFixed.wrap(2 * 10 ** 15); assertTrue(c > b); assertTrue(c.gt(b)); assertFalse(a > b); @@ -308,9 +320,9 @@ contract UFixedTest is Test { } function testOpGte() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(1 * 10 ** 18); - UFixed c = UFixed.wrap(2 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(1 * 10 ** 15); + UFixed c = UFixed.wrap(2 * 10 ** 15); assertTrue(c >= b); assertTrue(c.gte(b)); assertFalse(a >= c); @@ -322,9 +334,9 @@ contract UFixedTest is Test { } function testOpLt() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(1 * 10 ** 18); - UFixed c = UFixed.wrap(2 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(1 * 10 ** 15); + UFixed c = UFixed.wrap(2 * 10 ** 15); assertTrue(a < c); assertTrue(a.lt(c)); assertFalse(b < a); @@ -332,9 +344,9 @@ contract UFixedTest is Test { } function testOpLte() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(1 * 10 ** 18); - UFixed c = UFixed.wrap(2 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(1 * 10 ** 15); + UFixed c = UFixed.wrap(2 * 10 ** 15); assertTrue(a <= c); assertTrue(a.lte(c)); assertTrue(b <= a); @@ -344,26 +356,26 @@ contract UFixedTest is Test { } function testGtz() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(0 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(0 * 10 ** 15); assertTrue(a.gtz()); assertFalse(b.gtz()); } function testEqz() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(0 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(0 * 10 ** 15); assertFalse(a.eqz()); assertTrue(b.eqz()); } function testDelta() public { - UFixed a = UFixed.wrap(1 * 10 ** 18); - UFixed b = UFixed.wrap(0 * 10 ** 18); + UFixed a = UFixed.wrap(1 * 10 ** 15); + UFixed b = UFixed.wrap(0 * 10 ** 15); assertTrue(a.delta(b).eq(UFixedLib.toUFixed(1))); assertTrue(b.delta(a).eq(UFixedLib.toUFixed(1))); - UFixed c = UFixed.wrap(2 * 10 ** 18); + UFixed c = UFixed.wrap(2 * 10 ** 15); assertTrue(c.delta(a).eq(UFixedLib.toUFixed(1))); assertTrue(a.delta(c).eq(UFixedLib.toUFixed(1)));