Platform: Code4rena
Start Date: 27/05/2022
Pot Size: $75,000 USDC
Total HM: 20
Participants: 58
Period: 7 days
Judge: GalloDaSballo
Total Solo HM: 15
Id: 131
League: ETH
Rank: 50/58
Findings: 1
Award: $113.83
š Selected for report: 0
š Solo Findings: 0
š Selected for report: IllIllI
Also found by: 0x1f8b, 0x29A, 0xKitsune, 0xNazgul, 0xf15ers, 0xkatana, Chom, Dravee, Fitraldys, Funen, Kaiziron, MiloTruck, Picodes, Randyyy, RoiEvenHaim, SecureZeroX, Sm4rty, SmartSek, StyxRave, Tadashi, Tomio, Waze, asutorufos, berndartmueller, c3phas, catchup, csanuragjain, defsec, delfin454000, djxploit, fatherOfBlocks, gzeon, hake, hansfriese, oyc_109, robee, sach1r0, sashik_eth, scaraven, simon135
113.8256 USDC - $113.83
The following sections detail the gas optimizations found throughout the codebase. Each optimization is documented with the setup, an explainer for the optimization, a gas report and line identifiers for each optimization across the codebase. For each section's gas report, the optimizer was turned on and set to 10000 runs. You can replicate any tests/gas reports by heading to 0xKitsune/gas-lab and cloning the repo. Then, simply copy/paste the contract examples from any section and run forge test --gas-report
. You can also easily update the optimizer runs in the foundry.toml
.
contract GasTest is DSTest { Contract0 c0; Contract1 c1; function setUp() public { c0 = new Contract0(); c1 = new Contract1(); } function testGas() public view { c0.solidityHash(2309349, 2304923409); c1.assemblyHash(2309349, 2304923409); } } contract Contract0 { function solidityHash(uint256 a, uint256 b) public view { //unoptimized keccak256(abi.encodePacked(a, b)); } } contract Contract1 { function assemblyHash(uint256 a, uint256 b) public view { //optimized assembly { mstore(0x00, a) mstore(0x20, b) let hashedVal := keccak256(0x00, 0x40) } } }
āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract0 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 36687 ā 214 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā solidityHash ā 313 ā 313 ā 313 ā 313 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract1 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 31281 ā 186 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā assemblyHash ā 231 ā 231 ā 231 ā 231 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāāāÆ
InflationManager.sol:628
InflationManager.sol:632
InflationManager.sol:636
unchecked{++i}
instead of i++
(or use assembly when applicable)Use ++i
instead of i++
. This is especially useful in for loops but this optimization can be used anywhere in your code. You can also use unchecked{++i;}
for even more gas savings but this will not check to see if i
overflows. For extra safety if you are worried about this, you can add a require statement after the loop checking if i
is equal to the final incremented value. For best gas savings, use inline assembly, however this limits the functionality you can achieve. For example you cant use Solidity syntax to internally call your own contract within an assembly block and external calls must be done with the call()
or delegatecall()
instruction. However when applicable, inline assembly will save much more gas.
contract GasTest is DSTest { Contract0 c0; Contract1 c1; Contract2 c2; Contract3 c3; Contract4 c4; function setUp() public { c0 = new Contract0(); c1 = new Contract1(); c2 = new Contract2(); c3 = new Contract3(); c4 = new Contract4(); } function testGas() public { c0.iPlusPlus(); c1.plusPlusI(); c2.uncheckedPlusPlusI(); c3.safeUncheckedPlusPlusI(); c4.inlineAssemblyLoop(); } } contract Contract0 { //loop with i++ function iPlusPlus() public pure { uint256 j = 0; for (uint256 i; i < 10; i++) { j++; } } } contract Contract1 { //loop with ++i function plusPlusI() public pure { uint256 j = 0; for (uint256 i; i < 10; ++i) { j++; } } } contract Contract2 { //loop with unchecked{++i} function uncheckedPlusPlusI() public pure { uint256 j = 0; for (uint256 i; i < 10; ) { j++; unchecked { ++i; } } } } contract Contract3 { //loop with unchecked{++i} with additional overflow check function safeUncheckedPlusPlusI() public pure { uint256 j = 0; uint256 i = 0; for (i; i < 10; ) { j++; unchecked { ++i; } } //check for overflow assembly { if lt(i, 10) { mstore(0x00, "loop overflow") revert(0x00, 0x20) } } } } contract Contract4 { //loop with inline assembly function inlineAssemblyLoop() public pure { assembly { let j := 0 for { let i := 0 } lt(i, 10) { i := add(i, 0x01) } { j := add(j, 0x01) } } } }
āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāāā¬āāāāāāāāā¬āāāāāāā¬āāāāāāāāāā® ā Contract0 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāāŖāāāāāāāāāŖāāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāāā¼āāāāāāāāā¼āāāāāāā¼āāāāāāāāā⤠ā 37687 ā 219 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāāā¼āāāāāāāāā¼āāāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāāā¼āāāāāāāāā¼āāāāāāā¼āāāāāāāāā⤠ā iPlusPlus ā 2039 ā 2039 ā 2039 ā 2039 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāāā“āāāāāāāāā“āāāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāāā¬āāāāāāāāā¬āāāāāāā¬āāāāāāāāāā® ā Contract1 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāāŖāāāāāāāāāŖāāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāāā¼āāāāāāāāā¼āāāāāāā¼āāāāāāāāā⤠ā 37287 ā 217 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāāā¼āāāāāāāāā¼āāāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāāā¼āāāāāāāāā¼āāāāāāā¼āāāāāāāāā⤠ā plusPlusI ā 1989 ā 1989 ā 1989 ā 1989 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāāā“āāāāāāāāā“āāāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāāā¬āāāāāāāāā¬āāāāāāā¬āāāāāāāāāā® ā Contract3 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāāŖāāāāāāāāāŖāāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāāā¼āāāāāāāāā¼āāāāāāā¼āāāāāāāāā⤠ā 42693 ā 244 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāāā¼āāāāāāāāā¼āāāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāāā¼āāāāāāāāā¼āāāāāāā¼āāāāāāāāā⤠ā safeUncheckedPlusPlusI ā 1355 ā 1355 ā 1355 ā 1355 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāāā“āāāāāāāāā“āāāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāāā¬āāāāāāāāā¬āāāāāāā¬āāāāāāāāāā® ā Contract2 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāāŖāāāāāāāāāŖāāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāāā¼āāāāāāāāā¼āāāāāāā¼āāāāāāāāā⤠ā 35887 ā 210 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāāā¼āāāāāāāāā¼āāāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāāā¼āāāāāāāāā¼āāāāāāā¼āāāāāāāāā⤠ā uncheckedPlusPlusI ā 1329 ā 1329 ā 1329 ā 1329 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāāā“āāāāāāāāā“āāāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract4 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 26881 ā 164 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā inlineAssemblyLoop ā 709 ā 709 ā 709 ā 709 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāāāÆ
KeeperGauge.sol:59
KeeperGauge.sol:98
Use assembly for math instead of Solidity. You can check for overflow/underflow in assembly to ensure safety. If using Solidity versions < 0.8.0 and you are using Safemath, you can gain significant gas savings by using assembly to calculate values and checking for overflow/underflow.
contract GasTest is DSTest { Contract0 c0; Contract1 c1; Contract2 c2; Contract3 c3; Contract4 c4; Contract5 c5; Contract6 c6; Contract7 c7; function setUp() public { c0 = new Contract0(); c1 = new Contract1(); c2 = new Contract2(); c3 = new Contract3(); c4 = new Contract4(); c5 = new Contract5(); c6 = new Contract6(); c7 = new Contract7(); } function testGas() public { c0.addTest(34598345, 100); c1.addAssemblyTest(34598345, 100); c2.subTest(34598345, 100); c3.subAssemblyTest(34598345, 100); c4.mulTest(34598345, 100); c5.mulAssemblyTest(34598345, 100); c6.divTest(34598345, 100); c7.divAssemblyTest(34598345, 100); } } contract Contract0 { //addition in Solidity function addTest(uint256 a, uint256 b) public pure { uint256 c = a + b; } } contract Contract1 { //addition in assembly function addAssemblyTest(uint256 a, uint256 b) public pure { assembly { let c := add(a, b) if lt(c, a) { mstore(0x00, "overflow") revert(0x00, 0x20) } } } } contract Contract2 { //subtraction in Solidity function subTest(uint256 a, uint256 b) public pure { uint256 c = a - b; } } contract Contract3 { //subtraction in assembly function subAssemblyTest(uint256 a, uint256 b) public pure { assembly { let c := sub(a, b) if gt(c, a) { mstore(0x00, "underflow") revert(0x00, 0x20) } } } } contract Contract4 { //multiplication in Solidity function mulTest(uint256 a, uint256 b) public pure { uint256 c = a * b; } } contract Contract5 { //multiplication in assembly function mulAssemblyTest(uint256 a, uint256 b) public pure { assembly { let c := mul(a, b) if lt(c, a) { mstore(0x00, "overflow") revert(0x00, 0x20) } } } } contract Contract6 { //division in Solidity function divTest(uint256 a, uint256 b) public pure { uint256 c = a * b; } } contract Contract7 { //division in assembly function divAssemblyTest(uint256 a, uint256 b) public pure { assembly { let c := div(a, b) if gt(c, a) { mstore(0x00, "underflow") revert(0x00, 0x20) } } } }
āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract0 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 40493 ā 233 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā addTest ā 303 ā 303 ā 303 ā 303 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract1 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 37087 ā 216 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā addAssemblyTest ā 263 ā 263 ā 263 ā 263 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract2 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 40293 ā 232 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā subTest ā 300 ā 300 ā 300 ā 300 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract3 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 37287 ā 217 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā subAssemblyTest ā 263 ā 263 ā 263 ā 263 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract4 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 41893 ā 240 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā mulTest ā 325 ā 325 ā 325 ā 325 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract5 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 37087 ā 216 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā mulAssemblyTest ā 265 ā 265 ā 265 ā 265 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract6 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 41893 ā 240 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā divTest ā 325 ā 325 ā 325 ā 325 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract7 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 37287 ā 217 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā divAssemblyTest ā 265 ā 265 ā 265 ā 265 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāāāÆ
AmmGauge.sol:111
AmmGauge.sol:133
AmmGauge.sol:148
BkdLocker.sol:140
BkdLocker.sol:326
CvxMintAmount.sol:10
CvxMintAmount.sol:12
CvxMintAmount.sol:21
CvxMintAmount.sol:30
CvxMintAmount.sol:33
InflationManager.sol:574
InflationManager.sol:588
InflationManager.sol:601
KeeperGauge.sol:114
LpGauge.sol:115
Minter.sol:188
Minter.sol:190
Minter.sol:200
Minter.sol:204
Minter.sol:207
Minter.sol:219
Minter.sol:86
Minter.sol:87
Minter.sol:88
Minter.sol:90
VestedEscrow.sol:104
VestedEscrow.sol:108
VestedEscrow.sol:109
VestedEscrow.sol:155
VestedEscrow.sol:63
VestedEscrowRevocable.sol:59
You can use selfbalance()
instead of address(this).balance
when getting your contract's balance of ETH to save gas. Additionally, you can use balance(address)
instead of address.balance()
when getting an external contract's balance of ETH.
contract GasTest is DSTest { Contract0 c0; Contract1 c1; Contract2 c2; Contract3 c3; function setUp() public { c0 = new Contract0(); c1 = new Contract1(); c2 = new Contract2(); c3 = new Contract3(); } function testGas() public { c0.addressInternalBalance(); c1.assemblyInternalBalance(); c2.addressExternalBalance(address(this)); c3.assemblyExternalBalance(address(this)); } } contract Contract0 { function addressInternalBalance() public returns (uint256) { return address(this).balance; } } contract Contract1 { function assemblyInternalBalance() public returns (uint256) { assembly { let c := selfbalance() mstore(0x00, c) return(0x00, 0x20) } } } contract Contract2 { function addressExternalBalance(address addr) public { uint256 bal = address(addr).balance; bal++; } } contract Contract3 { function assemblyExternalBalance(address addr) public { uint256 bal; assembly { bal := balance(addr) } bal++; } }
āāāāāāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract0 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 23675 ā 147 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā addressInternalBalance ā 148 ā 148 ā 148 ā 148 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract1 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 27081 ā 165 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā assemblyInternalBalance ā 133 ā 133 ā 133 ā 133 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract2 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 61511 ā 339 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā addressExternalBalance ā 417 ā 417 ā 417 ā 417 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract3 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 57105 ā 317 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā assemblyExternalBalance ā 411 ā 411 ā 411 ā 411 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāāāÆ
FeeBurner.sol:102
RewardHandler.sol:40
contract GasTest is DSTest { Contract0 c0; Contract1 c1; function setUp() public { c0 = new Contract0(); c1 = new Contract1(); } function testGas() public view { c0.ownerNotZero(address(this)); c1.assemblyOwnerNotZero(address(this)); } } contract Contract0 { function ownerNotZero(address _addr) public pure { require(_addr != address(0), "zero address)"); } } contract Contract1 { function assemblyOwnerNotZero(address _addr) public pure { assembly { if iszero(_addr) { mstore(0x00, "zero address") revert(0x00, 0x20) } } } }
āāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract0 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 61311 ā 338 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā ownerNotZero ā 258 ā 258 ā 258 ā 258 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāā⯠āāāāāāāāāāāāāāāāāāāāāāāā¬āāāāāāāāāāāāāāāāāā¬āāāāāā¬āāāāāāāāā¬āāāāāā¬āāāāāāāāāā® ā Contract1 contract ā ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāāŖāāāāāāāāāāāāāāāāāāŖāāāāāāŖāāāāāāāāāŖāāāāāāŖāāāāāāāāāā” ā Deployment Cost ā Deployment Size ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā 44893 ā 255 ā ā ā ā ā āāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā Function Name ā min ā avg ā median ā max ā # calls ā āāāāāāāāāāāāāāāāāāāāāāāā¼āāāāāāāāāāāāāāāāāā¼āāāāāā¼āāāāāāāāā¼āāāāāā¼āāāāāāāāā⤠ā assemblyOwnerNotZero ā 252 ā 252 ā 252 ā 252 ā 1 ā ā°āāāāāāāāāāāāāāāāāāāāāāā“āāāāāāāāāāāāāāāāāā“āāāāāā“āāāāāāāāā“āāāāāā“āāāāāāāāāāÆ
AddressProvider.sol:102
AddressProvider.sol:105
AddressProvider.sol:122
AddressProvider.sol:163
AddressProvider.sol:166
AddressProvider.sol:295
AddressProvider.sol:98
Controller.sol:34
Controller.sol:35
Controller.sol:48
Controller.sol:50
Controller.sol:68
Controller.sol:82
FeeBurner.sol:101
FeeBurner.sol:60
InflationManager.sol:244
InflationManager.sol:270
InflationManager.sol:295
InflationManager.sol:486
InflationManager.sol:496
InflationManager.sol:508
InflationManager.sol:522
InflationManager.sol:59
InflationManager.sol:60
InflationManager.sol:622
LpGauge.sol:31
LpGauge.sol:35
Minter.sol:100
PoolMigrationZap.sol:26
PoolMigrationZap.sol:62
Preparable.sol:100
Preparable.sol:142
RewardHandler.sol:45
StakerVault.sol:119
StakerVault.sol:160
StakerVault.sol:205
StakerVault.sol:226
StakerVault.sol:238
StakerVault.sol:326
StakerVault.sol:372
StakerVault.sol:66
StakerVault.sol:76
VestedEscrow.sol:69
VestedEscrow.sol:75
VestedEscrow.sol:98
#0 - GalloDaSballo
2022-06-15T23:45:33Z
unchecked{++i} instead of i++ (or use assembly when applicable)
I was very incredulous of the gas savings of the assembly vs some other checks, so I forked the code and did my own tests.
It turns out that
function safeUncheckedPlusPlusI() public pure { uint256 j = 0; uint256 i = 0; for (i; i < 10; ) { unchecked { ++j; ++i; } } }
Is equivalent to
function inlineAssemblyLoop() public pure { assembly { let j := 0 for { let i := 0 } lt(i, 10) { i := add(i, 0x01) } { j := add(j, 0x01) } } }
Which is not reported in the document.
That said the savings are correct in that I also get 709 gas for the pure functions
#1 - GalloDaSballo
2022-06-16T20:53:19Z
I believe the gas savings here are caused by the operations in encodePacked
vs MSTORING directly.
That said the finding is valid and would save 100 gas per function
100 * 3 = 300
Because assembly is equivalent to a full unchecked I'll just use that one, effectively saving 5 gas (pre-increment) and 20 gas (unchecked operation) 25 * 2 = 50
My counter argument here is that assembly doesn't actually save gas, see my test here:
<img width="613" alt="Screenshot 2022-06-16 at 22 29 32" src="https://user-images.githubusercontent.com/13383782/174158227-ca0f484d-fb75-4973-8ffe-37157496e7d7.png">// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import "../../lib/test.sol"; import "../../lib/Console.sol"; contract GasTest is DSTest { Contract6 c6; Contract7 c7; function setUp() public { c6 = new Contract6(); c7 = new Contract7(); } function testGas() public { c6.divTest(34598345, 100); c7.divAssemblyTest(34598345, 100); } } contract Contract6 { //division in Solidity function divTest(uint256 a, uint256 b) public pure { uint c; unchecked {c = a * b;} require(c > a, "underflow"); } } contract Contract7 { //division in assembly function divAssemblyTest(uint256 a, uint256 b) public pure { assembly { let c := div(a, b) if gt(c, a) { mstore(0x00, "underflow") revert(0x00, 0x20) } } } }
Surprisingly the assembly check actually costs 3 more gas
Because of this, I find it hard to accept the finding at face value, as the finding is just saying: Use unchecked, with extra steps
I'll evaluate the finding as if it just suggested to use unchecked, which should yield a 20 gas saving per line
20 * 31 = 620
Would save 15 gas in two instances 30
From my checks the gas savings are very hard to quantify concretely as I've seen it save 2 gas on a basic check, and 6 gas on a require, in lack of a detailed comparison will give it 2 gas per finding
2 * 45 = 90
Am conflicted on this submission, on one hand the code playground and the format is well reasoned, on the other hand I feel the warden just dropped a bunch of ideas without spending the extra time to truly optimize the code given.
I feel in the future submissions like this will be held in higher regard, provided that wardens also use real before and after code snippets instead of these "magic trick" examples
Gas Saved: 1045