diff --git a/.gitmodules b/.gitmodules index f498e23..66a4cff 100644 --- a/.gitmodules +++ b/.gitmodules @@ -9,4 +9,4 @@ url = https://github.com/transmissions11/solmate [submodule "lib/solady"] path = lib/solady - url = https://github.com/vectorized/solady \ No newline at end of file + url = https://github.com/vectorized/solady diff --git a/lib/solady b/lib/solady new file mode 160000 index 0000000..fad3f67 --- /dev/null +++ b/lib/solady @@ -0,0 +1 @@ +Subproject commit fad3f6703c2cd4cfd185f9921790d117503f54c6 diff --git a/lib/solady/.gas-snapshot b/lib/solady/.gas-snapshot deleted file mode 100644 index 427cbaa..0000000 --- a/lib/solady/.gas-snapshot +++ /dev/null @@ -1,1052 +0,0 @@ -Base64Test:testBase64DecodeSentenceGas() (gas: 3663) -Base64Test:testBase64DecodeShortStringGas() (gas: 919) -Base64Test:testBase64EncodeDecode(bytes) (runs: 256, μ: 13919, ~: 10697) -Base64Test:testBase64EncodeDecodeAltModes(bytes) (runs: 256, μ: 834983, ~: 764290) -Base64Test:testBase64EncodeEmptyString() (gas: 953) -Base64Test:testBase64EncodeFileSafeAndNoPadding(bytes,bool,bool) (runs: 256, μ: 16092, ~: 13634) -Base64Test:testBase64EncodeSentence() (gas: 4757) -Base64Test:testBase64EncodeShortStrings() (gas: 8496) -Base64Test:testBase64EncodeToStringWithDoublePadding() (gas: 1702) -Base64Test:testBase64EncodeToStringWithNoPadding() (gas: 1680) -Base64Test:testBase64EncodeToStringWithSinglePadding() (gas: 1636) -Base64Test:testBase64WordBoundary() (gas: 12511) -Base64Test:test__codesize() (gas: 7802) -CREATE3Test:testDeployERC20() (gas: 761926) -CREATE3Test:testDeployERC20(bytes32,string,string,uint8) (runs: 256, μ: 805598, ~: 808154) -CREATE3Test:testDeployedUpperBitsSafeForPlainSolidity() (gas: 657) -CREATE3Test:testDoubleDeployDifferentBytecodeReverts() (gas: 96899279) -CREATE3Test:testDoubleDeployDifferentBytecodeReverts(bytes32,bytes,bytes) (runs: 256, μ: 96880132, ~: 96880005) -CREATE3Test:testDoubleDeploySameBytecodeReverts() (gas: 96889159) -CREATE3Test:testDoubleDeploySameBytecodeReverts(bytes32,bytes) (runs: 256, μ: 96881054, ~: 96881007) -CREATE3Test:test__codesize() (gas: 15337) -DateTimeLibTest:testAddSubDiffDays(uint256,uint256) (runs: 256, μ: 4033, ~: 4074) -DateTimeLibTest:testAddSubDiffHours(uint256,uint256) (runs: 256, μ: 3976, ~: 3932) -DateTimeLibTest:testAddSubDiffMinutes(uint256,uint256) (runs: 256, μ: 3974, ~: 3939) -DateTimeLibTest:testAddSubDiffMonths(uint256,uint256) (runs: 256, μ: 7084, ~: 7075) -DateTimeLibTest:testAddSubDiffSeconds(uint256,uint256) (runs: 256, μ: 3617, ~: 3586) -DateTimeLibTest:testAddSubDiffYears(uint256,uint256) (runs: 256, μ: 6648, ~: 6640) -DateTimeLibTest:testDateTimeArithmeticReverts() (gas: 4597) -DateTimeLibTest:testDateTimeMaxSupported() (gas: 2502) -DateTimeLibTest:testDateTimeToAndFroTimestamp((uint256,uint256,uint256,uint256,uint256,uint256)) (runs: 256, μ: 3971, ~: 4050) -DateTimeLibTest:testDateToAndFroEpochDay((uint256,uint256,uint256,uint256,uint256,uint256)) (runs: 256, μ: 2466, ~: 2440) -DateTimeLibTest:testDateToAndFroEpochDay() (gas: 865464) -DateTimeLibTest:testDateToAndFroTimestamp() (gas: 909205) -DateTimeLibTest:testDateToEpochDay() (gas: 1603) -DateTimeLibTest:testDateToEpochDayDifferential((uint256,uint256,uint256,uint256,uint256,uint256)) (runs: 256, μ: 2209, ~: 2178) -DateTimeLibTest:testDateToEpochDayDifferential2((uint256,uint256,uint256,uint256,uint256,uint256)) (runs: 256, μ: 2131, ~: 2105) -DateTimeLibTest:testDateToEpochDayGas() (gas: 756479) -DateTimeLibTest:testDateToEpochDayGas2() (gas: 759345) -DateTimeLibTest:testDayOfWeek() (gas: 175297) -DateTimeLibTest:testDaysInMonth() (gas: 1204) -DateTimeLibTest:testDaysInMonth(uint256,uint256) (runs: 256, μ: 1033, ~: 1051) -DateTimeLibTest:testDaysToDate() (gas: 1958) -DateTimeLibTest:testEpochDayToDate(uint256) (runs: 256, μ: 1019, ~: 1019) -DateTimeLibTest:testEpochDayToDateDifferential(uint256) (runs: 256, μ: 1812, ~: 1762) -DateTimeLibTest:testEpochDayToDateDifferential2(uint256) (runs: 256, μ: 1750, ~: 1683) -DateTimeLibTest:testEpochDayToDateGas() (gas: 339723) -DateTimeLibTest:testEpochDayToDateGas2() (gas: 357444) -DateTimeLibTest:testIsLeapYear() (gas: 741) -DateTimeLibTest:testIsLeapYear(uint256) (runs: 256, μ: 521, ~: 495) -DateTimeLibTest:testIsSupportedDateFalse() (gas: 1651) -DateTimeLibTest:testIsSupportedDateTime((uint256,uint256,uint256,uint256,uint256,uint256)) (runs: 256, μ: 2830, ~: 2842) -DateTimeLibTest:testIsSupportedDateTrue() (gas: 670) -DateTimeLibTest:testIsSupportedEpochDayFalse() (gas: 597) -DateTimeLibTest:testIsSupportedEpochDayTrue() (gas: 305) -DateTimeLibTest:testIsSupportedTimestampFalse() (gas: 599) -DateTimeLibTest:testIsSupportedTimestampTrue() (gas: 304) -DateTimeLibTest:testIsWeekEnd(uint256) (runs: 256, μ: 727, ~: 662) -DateTimeLibTest:testMondayTimestamp() (gas: 1106) -DateTimeLibTest:testMondayTimestamp(uint256) (runs: 256, μ: 733, ~: 814) -DateTimeLibTest:testNthWeekdayInMonthOfYearTimestamp() (gas: 12031) -DateTimeLibTest:testNthWeekdayInMonthOfYearTimestamp(uint256,uint256,uint256,uint256) (runs: 256, μ: 3462, ~: 3494) -DateTimeLibTest:testWeekday() (gas: 705) -DateTimeLibTest:test__codesize() (gas: 20051) -DynamicBufferLibTest:testClear(uint256) (runs: 256, μ: 171012, ~: 171193) -DynamicBufferLibTest:testDynamicBuffer(bytes[],uint256) (runs: 256, μ: 1032333, ~: 903881) -DynamicBufferLibTest:testDynamicBuffer(uint256) (runs: 256, μ: 1036837, ~: 737806) -DynamicBufferLibTest:testDynamicBufferChaining() (gas: 22178) -DynamicBufferLibTest:testDynamicBufferReserveFromEmpty() (gas: 2781) -DynamicBufferLibTest:testDynamicBufferReserveFromEmpty2() (gas: 2423) -DynamicBufferLibTest:testDynamicBufferReserveFromEmpty3(bytes,uint256) (runs: 256, μ: 4205, ~: 2068) -DynamicBufferLibTest:testJoinWithConcat() (gas: 31407) -DynamicBufferLibTest:testJoinWithDynamicBuffer() (gas: 10764) -DynamicBufferLibTest:test__codesize() (gas: 11003) -ECDSATest:testBytes32ToEthSignedMessageHash() (gas: 381) -ECDSATest:testBytesToEthSignedMessageHash() (gas: 11582079) -ECDSATest:testBytesToEthSignedMessageHashEmpty() (gas: 556) -ECDSATest:testBytesToEthSignedMessageHashExceedsMaxLengthReverts() (gas: 608392) -ECDSATest:testBytesToEthSignedMessageHashLong() (gas: 681) -ECDSATest:testBytesToEthSignedMessageHashShort() (gas: 620) -ECDSATest:testEmptyCalldataHelpers() (gas: 3253) -ECDSATest:testRecoverAndTryRecover(bytes32) (runs: 256, μ: 2788434, ~: 2507144) -ECDSATest:testRecoverWithInvalidLongSignatureReverts() (gas: 7377) -ECDSATest:testRecoverWithInvalidShortSignatureReturnsZero() (gas: 7168) -ECDSATest:testRecoverWithInvalidSignatureReverts() (gas: 7955) -ECDSATest:testRecoverWithV0SignatureWithShortEIP2098Format() (gas: 4567) -ECDSATest:testRecoverWithV0SignatureWithShortEIP2098FormatAsCalldata() (gas: 8712) -ECDSATest:testRecoverWithV0SignatureWithVersion00Reverts() (gas: 7898) -ECDSATest:testRecoverWithV0SignatureWithVersion27() (gas: 8812) -ECDSATest:testRecoverWithV0SignatureWithWrongVersionReverts() (gas: 7876) -ECDSATest:testRecoverWithV1SignatureWithShortEIP2098Format() (gas: 4611) -ECDSATest:testRecoverWithV1SignatureWithShortEIP2098FormatAsCalldata() (gas: 8644) -ECDSATest:testRecoverWithV1SignatureWithVersion01Reverts() (gas: 7910) -ECDSATest:testRecoverWithV1SignatureWithVersion28() (gas: 8813) -ECDSATest:testRecoverWithV1SignatureWithWrongVersionReverts() (gas: 7877) -ECDSATest:testRecoverWithValidSignature() (gas: 8938) -ECDSATest:testRecoverWithWrongSigner() (gas: 8829) -ECDSATest:testTryRecoverWithInvalidLongSignatureReturnsZero() (gas: 5223) -ECDSATest:testTryRecoverWithInvalidShortSignatureReturnsZero() (gas: 5049) -ECDSATest:testTryRecoverWithInvalidSignature() (gas: 8924) -ECDSATest:testTryRecoverWithV0SignatureWithShortEIP2098Format() (gas: 4556) -ECDSATest:testTryRecoverWithV0SignatureWithShortEIP2098FormatAsCalldata() (gas: 8713) -ECDSATest:testTryRecoverWithV0SignatureWithVersion00ReturnsZero() (gas: 8809) -ECDSATest:testTryRecoverWithV0SignatureWithVersion27() (gas: 8832) -ECDSATest:testTryRecoverWithV0SignatureWithWrongVersionReturnsZero() (gas: 8832) -ECDSATest:testTryRecoverWithV1SignatureWithShortEIP2098Format() (gas: 4610) -ECDSATest:testTryRecoverWithV1SignatureWithShortEIP2098FormatAsCalldata() (gas: 8735) -ECDSATest:testTryRecoverWithV1SignatureWithVersion01ReturnsZero() (gas: 8853) -ECDSATest:testTryRecoverWithV1SignatureWithVersion28() (gas: 8832) -ECDSATest:testTryRecoverWithV1SignatureWithWrongVersionReturnsZero() (gas: 8831) -ECDSATest:testTryRecoverWithValidSignature() (gas: 8901) -ECDSATest:testTryRecoverWithWrongSigner() (gas: 8827) -ECDSATest:test__codesize() (gas: 15480) -EIP712Test:testDomainSeparator() (gas: 6026) -EIP712Test:testDomainSeparatorOnClone() (gas: 8842) -EIP712Test:testDomainSeparatorOnCloneDynamicWithChainIdChange() (gas: 69650) -EIP712Test:testDomainSeparatorOnCloneWithChainIdChange() (gas: 13334) -EIP712Test:testDomainSeparatorOnDynamicWithChainIdChange() (gas: 29288) -EIP712Test:testDomainSeparatorWithChainIdChange() (gas: 10391) -EIP712Test:testEIP5267() (gas: 35389) -EIP712Test:testHashTypedData() (gas: 37210) -EIP712Test:testHashTypedDataOnClone() (gas: 40216) -EIP712Test:testHashTypedDataOnCloneDynamic() (gas: 46957) -EIP712Test:testHashTypedDataOnCloneDynamicWithChaindIdChange() (gas: 58250) -EIP712Test:testHashTypedDataOnCloneWithChaindIdChange() (gas: 49558) -EIP712Test:testHashTypedDataOnDynamic() (gas: 44865) -EIP712Test:testHashTypedDataOnDynamicWithChaindIdChange() (gas: 57357) -EIP712Test:testHashTypedDataWithChaindIdChange() (gas: 45775) -EIP712Test:test__codesize() (gas: 13881) -ERC1155HooksTest:testERC1155Hooks() (gas: 4396277) -ERC1155HooksTest:test__codesize() (gas: 12073) -ERC1155Test:testApproveAll(address,bool) (runs: 256, μ: 50193, ~: 58576) -ERC1155Test:testAuthorizedEquivalence(address,address,bool) (runs: 256, μ: 715, ~: 715) -ERC1155Test:testBalanceOfBatchWithArrayMismatchReverts(uint256) (runs: 256, μ: 32318, ~: 34937) -ERC1155Test:testBatchBalanceOf(uint256) (runs: 256, μ: 114503, ~: 93582) -ERC1155Test:testBatchBurn(uint256) (runs: 256, μ: 175450, ~: 161695) -ERC1155Test:testBatchBurnInsufficientBalanceReverts(uint256) (runs: 256, μ: 168826, ~: 173551) -ERC1155Test:testBatchBurnWithArrayLengthMismatchReverts(uint256) (runs: 256, μ: 43075, ~: 42595) -ERC1155Test:testBatchMintToEOA(uint256) (runs: 256, μ: 126044, ~: 148527) -ERC1155Test:testBatchMintToERC1155Recipient(uint256) (runs: 256, μ: 776209, ~: 776871) -ERC1155Test:testBatchMintToNonERC1155RecipientReverts(uint256) (runs: 256, μ: 172369, ~: 185840) -ERC1155Test:testBatchMintToRevertingERC1155RecipientReverts(uint256) (runs: 256, μ: 321976, ~: 315851) -ERC1155Test:testBatchMintToWrongReturnDataERC1155RecipientReverts(uint256) (runs: 256, μ: 293927, ~: 318698) -ERC1155Test:testBatchMintToZeroReverts(uint256) (runs: 256, μ: 73027, ~: 64999) -ERC1155Test:testBatchMintWithArrayMismatchReverts(uint256) (runs: 256, μ: 33385, ~: 35512) -ERC1155Test:testBurn(uint256) (runs: 256, μ: 91097, ~: 82321) -ERC1155Test:testBurnInsufficientBalanceReverts(uint256) (runs: 256, μ: 97148, ~: 97538) -ERC1155Test:testDirectSetApprovalForAll(address,address,bool) (runs: 256, μ: 24436, ~: 15497) -ERC1155Test:testMintToEOA(uint256) (runs: 256, μ: 72096, ~: 71717) -ERC1155Test:testMintToERC1155Recipient(uint256) (runs: 256, μ: 672296, ~: 653975) -ERC1155Test:testMintToNonERC155RecipientReverts(uint256) (runs: 256, μ: 103261, ~: 103225) -ERC1155Test:testMintToRevertingERC155RecipientReverts(uint256) (runs: 256, μ: 281517, ~: 281563) -ERC1155Test:testMintToWrongReturnDataERC155RecipientReverts(uint256) (runs: 256, μ: 235945, ~: 235934) -ERC1155Test:testMintToZeroReverts(uint256) (runs: 256, μ: 33088, ~: 33061) -ERC1155Test:testSafeBatchTransfer() (gas: 8330001) -ERC1155Test:testSafeBatchTransferFromToEOA(uint256) (runs: 256, μ: 208325, ~: 189416) -ERC1155Test:testSafeBatchTransferFromToERC1155Recipient(uint256) (runs: 256, μ: 883187, ~: 884928) -ERC1155Test:testSafeBatchTransferFromToNonERC1155RecipientReverts(uint256) (runs: 256, μ: 254320, ~: 268256) -ERC1155Test:testSafeBatchTransferFromToRevertingERC1155RecipientReverts(uint256) (runs: 256, μ: 432345, ~: 446565) -ERC1155Test:testSafeBatchTransferFromToWrongReturnDataERC1155RecipientReverts(uint256) (runs: 256, μ: 347280, ~: 330274) -ERC1155Test:testSafeBatchTransferFromToZeroReverts(uint256) (runs: 256, μ: 140880, ~: 124791) -ERC1155Test:testSafeBatchTransferFromWithArrayLengthMismatchReverts(uint256) (runs: 256, μ: 51206, ~: 66621) -ERC1155Test:testSafeBatchTransferInsufficientBalanceReverts(uint256) (runs: 256, μ: 170775, ~: 173686) -ERC1155Test:testSafeTransferFromInsufficientBalanceReverts(uint256) (runs: 256, μ: 99216, ~: 99806) -ERC1155Test:testSafeTransferFromSelf(uint256) (runs: 256, μ: 106266, ~: 105774) -ERC1155Test:testSafeTransferFromSelfInsufficientBalanceReverts(uint256) (runs: 256, μ: 71408, ~: 72157) -ERC1155Test:testSafeTransferFromToEOA(uint256) (runs: 256, μ: 117984, ~: 111719) -ERC1155Test:testSafeTransferFromToERC1155Recipient(uint256) (runs: 256, μ: 757271, ~: 777302) -ERC1155Test:testSafeTransferFromToNonERC155RecipientReverts(uint256) (runs: 256, μ: 140466, ~: 139762) -ERC1155Test:testSafeTransferFromToRevertingERC1155RecipientReverts(uint256) (runs: 256, μ: 317084, ~: 318526) -ERC1155Test:testSafeTransferFromToWrongReturnDataERC1155RecipientReverts(uint256) (runs: 256, μ: 272852, ~: 272482) -ERC1155Test:testSafeTransferFromToZeroReverts(uint256) (runs: 256, μ: 70843, ~: 70216) -ERC1155Test:test__codesize() (gas: 43462) -ERC1967FactoryTest:testChangeAdmin() (gas: 266356) -ERC1967FactoryTest:testChangeAdminUnauthorized() (gas: 257236) -ERC1967FactoryTest:testDeploy() (gas: 257363) -ERC1967FactoryTest:testDeployAndCall(uint256) (runs: 256, μ: 339688, ~: 340911) -ERC1967FactoryTest:testDeployAndCallWithRevert() (gas: 211881) -ERC1967FactoryTest:testDeployBrutalized(uint256) (runs: 256, μ: 116015, ~: 44119) -ERC1967FactoryTest:testDeployDeterministicAndCall(uint256) (runs: 256, μ: 313913, ~: 350129) -ERC1967FactoryTest:testFactoryDeployment() (gas: 856479) -ERC1967FactoryTest:testProxyFails() (gas: 259019) -ERC1967FactoryTest:testProxySucceeds() (gas: 255707) -ERC1967FactoryTest:testUpgrade() (gas: 266792) -ERC1967FactoryTest:testUpgradeAndCall() (gas: 354415) -ERC1967FactoryTest:testUpgradeAndCallWithRevert() (gas: 265645) -ERC1967FactoryTest:testUpgradeUnauthorized() (gas: 270260) -ERC1967FactoryTest:testUpgradeWithCorruptedProxy() (gas: 263163) -ERC1967FactoryTest:test__codesize() (gas: 33920) -ERC20Invariants:invariantBalanceSum() (runs: 256, calls: 3840, reverts: 2351) -ERC20Invariants:test__codesize() (gas: 8050) -ERC20Test:testApprove() (gas: 35730) -ERC20Test:testApprove(address,uint256) (runs: 256, μ: 29937, ~: 31181) -ERC20Test:testBurn() (gas: 61920) -ERC20Test:testBurn(address,uint256,uint256) (runs: 256, μ: 60240, ~: 62469) -ERC20Test:testBurnInsufficientBalanceReverts(address,uint256,uint256) (runs: 256, μ: 55227, ~: 56454) -ERC20Test:testDirectSpendAllowance(uint256) (runs: 256, μ: 61647, ~: 61904) -ERC20Test:testDirectTransfer(uint256) (runs: 256, μ: 100856, ~: 111918) -ERC20Test:testInfiniteApproveTransferFrom() (gas: 89993) -ERC20Test:testMetadata() (gas: 17598) -ERC20Test:testMint() (gas: 58668) -ERC20Test:testMint(address,uint256) (runs: 256, μ: 56402, ~: 58890) -ERC20Test:testMintOverMaxUintReverts() (gas: 55753) -ERC20Test:testPermit() (gas: 89727) -ERC20Test:testPermit(uint256) (runs: 256, μ: 89575, ~: 89659) -ERC20Test:testPermitBadDeadlineReverts(uint256) (runs: 256, μ: 41403, ~: 41421) -ERC20Test:testPermitBadNonceReverts(uint256) (runs: 256, μ: 41837, ~: 41787) -ERC20Test:testPermitPastDeadlineReverts(uint256) (runs: 256, μ: 35995, ~: 35987) -ERC20Test:testPermitReplayReverts(uint256) (runs: 256, μ: 92277, ~: 92351) -ERC20Test:testTransfer() (gas: 65298) -ERC20Test:testTransfer(address,uint256) (runs: 256, μ: 63026, ~: 65514) -ERC20Test:testTransferFrom() (gas: 85633) -ERC20Test:testTransferFrom(address,address,address,uint256,uint256) (runs: 256, μ: 89459, ~: 94758) -ERC20Test:testTransferFromInsufficientAllowanceReverts() (gas: 81443) -ERC20Test:testTransferFromInsufficientAllowanceReverts(address,uint256,uint256) (runs: 256, μ: 81380, ~: 81983) -ERC20Test:testTransferFromInsufficientBalanceReverts() (gas: 61836) -ERC20Test:testTransferFromInsufficientBalanceReverts(address,uint256,uint256) (runs: 256, μ: 64833, ~: 62492) -ERC20Test:testTransferInsufficientBalanceReverts() (gas: 55919) -ERC20Test:testTransferInsufficientBalanceReverts(address,uint256,uint256) (runs: 256, μ: 55201, ~: 56424) -ERC20Test:test__codesize() (gas: 24081) -ERC2981Test:testRoyaltyOverflowCheckDifferential(uint256,uint256) (runs: 256, μ: 456, ~: 458) -ERC2981Test:testSetAndGetRoyaltyInfo(uint256) (runs: 256, μ: 107958, ~: 104834) -ERC2981Test:test__codesize() (gas: 8419) -ERC4337FactoryTest:testCreateAccountRepeatedDeployment() (gas: 149757) -ERC4337FactoryTest:testCreateAccountRepeatedDeployment(uint256) (runs: 256, μ: 171753, ~: 171617) -ERC4337FactoryTest:testDeployDeterministic(uint256) (runs: 256, μ: 134818, ~: 139900) -ERC4337FactoryTest:test__codesize() (gas: 13197) -ERC4337Test:testCdFallback() (gas: 443962) -ERC4337Test:testCdFallback2() (gas: 1140793) -ERC4337Test:testDelegateExecute() (gas: 369570) -ERC4337Test:testDelegateExecute(uint256) (runs: 256, μ: 356052, ~: 344533) -ERC4337Test:testDelegateExecuteRevertsIfOwnerSlotValueChanged() (gas: 319282) -ERC4337Test:testDepositFunctions() (gas: 502928) -ERC4337Test:testDirectStorage() (gas: 70391) -ERC4337Test:testETHReceived() (gas: 16584) -ERC4337Test:testExecute() (gas: 382774) -ERC4337Test:testExecuteBatch() (gas: 692605) -ERC4337Test:testExecuteBatch(uint256) (runs: 256, μ: 528646, ~: 668616) -ERC4337Test:testInitializer() (gas: 285192) -ERC4337Test:testIsValidSignature() (gas: 111663) -ERC4337Test:testIsValidSignaturePersonalSign() (gas: 96270) -ERC4337Test:testIsValidSignatureWrapped() (gas: 406706) -ERC4337Test:testOnERC1155BatchReceived() (gas: 1393788) -ERC4337Test:testOnERC1155Received() (gas: 1391176) -ERC4337Test:testOnERC721Received() (gas: 1311273) -ERC4337Test:testOwnerRecovery() (gas: 486056) -ERC4337Test:testValidateUserOp() (gas: 491506) -ERC4337Test:test__codesize() (gas: 54000) -ERC4626Test:testDepositWithNoApprovalReverts() (gas: 16371) -ERC4626Test:testDepositWithNotEnoughApprovalReverts() (gas: 89884) -ERC4626Test:testDifferentialFullMulDiv(uint256,uint256,uint256) (runs: 256, μ: 3343, ~: 3201) -ERC4626Test:testMetadata() (gas: 15439) -ERC4626Test:testMintWithNoApprovalReverts() (gas: 16345) -ERC4626Test:testMintZero() (gas: 54317) -ERC4626Test:testMultipleMintDepositRedeemWithdraw() (gas: 425667) -ERC4626Test:testRedeemWithNoShareAmountReverts() (gas: 10918) -ERC4626Test:testRedeemWithNotEnoughShareAmountReverts() (gas: 142915) -ERC4626Test:testSingleDepositWithdraw(uint128) (runs: 256, μ: 202683, ~: 202694) -ERC4626Test:testSingleMintRedeem(uint128) (runs: 256, μ: 201604, ~: 201615) -ERC4626Test:testTryGetAssetDecimals() (gas: 30490416) -ERC4626Test:testUseVirtualShares() (gas: 2442168) -ERC4626Test:testVaultInteractionsForSomeoneElse() (gas: 296189) -ERC4626Test:testVirtualSharesMultipleMintDepositRedeemWithdraw() (gas: 1638354) -ERC4626Test:testWithdrawWithNoUnderlyingAmountReverts() (gas: 13102) -ERC4626Test:testWithdrawWithNotEnoughUnderlyingAmountReverts() (gas: 144082) -ERC4626Test:testWithdrawZero() (gas: 52807) -ERC4626Test:test__codesize() (gas: 41081) -ERC6551Test:testCdFallback() (gas: 894557) -ERC6551Test:testDeployERC6551(uint256) (runs: 256, μ: 170988, ~: 168739) -ERC6551Test:testDeployERC6551Proxy() (gas: 102497) -ERC6551Test:testExecute() (gas: 507843) -ERC6551Test:testExecuteBatch() (gas: 816977) -ERC6551Test:testExecuteBatch(uint256) (runs: 256, μ: 606828, ~: 483186) -ERC6551Test:testInitializeERC6551ProxyImplementation() (gas: 189801) -ERC6551Test:testIsValidSignature() (gas: 187612) -ERC6551Test:testOnERC1155BatchReceived() (gas: 1526542) -ERC6551Test:testOnERC1155Received() (gas: 1523898) -ERC6551Test:testOnERC721Received() (gas: 1447973) -ERC6551Test:testOnERC721ReceivedCycles() (gas: 1710837) -ERC6551Test:testOnERC721ReceivedCyclesWithDifferentChainIds(uint256) (runs: 256, μ: 449186, ~: 455552) -ERC6551Test:testSupportsInterface() (gas: 169387) -ERC6551Test:testUpgrade() (gas: 1154845) -ERC6551Test:test__codesize() (gas: 47900) -ERC6909Test:testApprove() (gas: 36738) -ERC6909Test:testApprove(address,uint256,uint256) (runs: 256, μ: 36369, ~: 37380) -ERC6909Test:testBurn() (gas: 40643) -ERC6909Test:testBurn(address,uint256,uint256,uint256) (runs: 256, μ: 39876, ~: 41154) -ERC6909Test:testBurnInsufficientBalanceReverts(address,uint256,uint256,uint256) (runs: 256, μ: 34112, ~: 34972) -ERC6909Test:testDecimals() (gas: 5632) -ERC6909Test:testDirectApprove() (gas: 36700) -ERC6909Test:testDirectFunctions(uint256) (runs: 256, μ: 199858, ~: 201362) -ERC6909Test:testDirectSetOperator() (gas: 35937) -ERC6909Test:testDirectTransfer() (gas: 65833) -ERC6909Test:testInfiniteApproveTransferFrom() (gas: 78062) -ERC6909Test:testMetadata() (gas: 9332) -ERC6909Test:testMint() (gas: 36905) -ERC6909Test:testMint(address,uint256,uint256) (runs: 256, μ: 36208, ~: 37141) -ERC6909Test:testMintOverMaxUintReverts() (gas: 34388) -ERC6909Test:testMintOverMaxUintReverts(address,uint256,uint256,uint256) (runs: 256, μ: 35296, ~: 35348) -ERC6909Test:testOperatorTransferFrom() (gas: 73755) -ERC6909Test:testSetOperator() (gas: 37020) -ERC6909Test:testSetOperator(address,address,bool) (runs: 256, μ: 25706, ~: 16767) -ERC6909Test:testTokenURI() (gas: 33117) -ERC6909Test:testTransfer() (gas: 48028) -ERC6909Test:testTransfer(address,uint256,uint256) (runs: 256, μ: 46126, ~: 47169) -ERC6909Test:testTransferFrom() (gas: 74272) -ERC6909Test:testTransferFrom(address,address,address,uint256,uint256,uint256) (runs: 256, μ: 76831, ~: 80611) -ERC6909Test:testTransferFromCallerIsNotOperator(address,uint256,uint256) (runs: 256, μ: 39318, ~: 39315) -ERC6909Test:testTransferFromInsufficientAllowanceReverts(address,uint256,uint256,uint256) (runs: 256, μ: 66020, ~: 67346) -ERC6909Test:testTransferFromInsufficientBalanceReverts() (gas: 49521) -ERC6909Test:testTransferFromInsufficientBalanceReverts(address,uint256,uint256,uint256) (runs: 256, μ: 51956, ~: 49764) -ERC6909Test:testTransferFromInsufficientPermission() (gas: 66821) -ERC6909Test:testTransferFromOverMaxUintReverts() (gas: 89920) -ERC6909Test:testTransferFromOverMaxUintReverts(address,uint256,uint256,uint256) (runs: 256, μ: 112328, ~: 112849) -ERC6909Test:testTransferInsufficientBalanceReverts() (gas: 34534) -ERC6909Test:testTransferInsufficientBalanceReverts(address,uint256,uint256,uint256) (runs: 256, μ: 34680, ~: 35076) -ERC6909Test:testTransferOverMaxUintReverts() (gas: 63438) -ERC6909Test:testTransferOverMaxUintReverts(address,uint256,uint256,uint256) (runs: 256, μ: 63907, ~: 63923) -ERC6909Test:test__codesize() (gas: 26423) -ERC721HooksTest:testERC721Hooks() (gas: 2877778) -ERC721HooksTest:test__codesize() (gas: 9707) -ERC721Test:testApprove(uint256) (runs: 256, μ: 108103, ~: 108158) -ERC721Test:testApproveAll(uint256) (runs: 256, μ: 47875, ~: 40312) -ERC721Test:testApproveBurn(uint256) (runs: 256, μ: 86754, ~: 86771) -ERC721Test:testApproveNonExistentReverts(uint256,address) (runs: 256, μ: 33662, ~: 33621) -ERC721Test:testApproveUnauthorizedReverts(uint256) (runs: 256, μ: 83185, ~: 82400) -ERC721Test:testAuthorizedEquivalence(address,bool,bool) (runs: 256, μ: 748, ~: 743) -ERC721Test:testAux(uint256) (runs: 256, μ: 191880, ~: 193043) -ERC721Test:testBurn(uint256) (runs: 256, μ: 82530, ~: 93880) -ERC721Test:testBurnNonExistentReverts(uint256) (runs: 256, μ: 10761, ~: 10761) -ERC721Test:testCannotExceedMaxBalance() (gas: 149876) -ERC721Test:testDoubleBurnReverts(uint256) (runs: 256, μ: 63486, ~: 63432) -ERC721Test:testDoubleMintReverts(uint256) (runs: 256, μ: 79131, ~: 79166) -ERC721Test:testEverything(uint256) (runs: 256, μ: 308702, ~: 302430) -ERC721Test:testExtraData(uint256) (runs: 256, μ: 99055, ~: 99093) -ERC721Test:testExtraData2(uint256,uint256) (runs: 256, μ: 54254, ~: 53903) -ERC721Test:testIsApprovedOrOwner(uint256) (runs: 256, μ: 135460, ~: 135446) -ERC721Test:testMint(uint256) (runs: 256, μ: 82886, ~: 82916) -ERC721Test:testMintToZeroReverts(uint256) (runs: 256, μ: 8686, ~: 8686) -ERC721Test:testOwnerOfNonExistent(uint256) (runs: 256, μ: 33389, ~: 33338) -ERC721Test:testSafeMintToEOA(uint256) (runs: 256, μ: 83480, ~: 83519) -ERC721Test:testSafeMintToERC721Recipient(uint256) (runs: 256, μ: 409459, ~: 410571) -ERC721Test:testSafeMintToERC721RecipientWithData(uint256,bytes) (runs: 256, μ: 470871, ~: 460001) -ERC721Test:testSafeMintToERC721RecipientWithWrongReturnData(uint256) (runs: 256, μ: 170008, ~: 170008) -ERC721Test:testSafeMintToERC721RecipientWithWrongReturnDataWithData(uint256,bytes) (runs: 256, μ: 171224, ~: 171171) -ERC721Test:testSafeMintToNonERC721RecipientReverts(uint256) (runs: 256, μ: 100470, ~: 100470) -ERC721Test:testSafeMintToNonERC721RecipientWithDataReverts(uint256,bytes) (runs: 256, μ: 101718, ~: 101665) -ERC721Test:testSafeMintToRevertingERC721RecipientReverts(uint256) (runs: 256, μ: 203127, ~: 203127) -ERC721Test:testSafeMintToRevertingERC721RecipientWithDataReverts(uint256,bytes) (runs: 256, μ: 204364, ~: 204311) -ERC721Test:testSafeTransferFromToEOA(uint256) (runs: 256, μ: 121957, ~: 122057) -ERC721Test:testSafeTransferFromToERC721Recipient(uint256) (runs: 256, μ: 470968, ~: 472076) -ERC721Test:testSafeTransferFromToERC721RecipientWithData(uint256,bytes) (runs: 256, μ: 532415, ~: 521775) -ERC721Test:testSafeTransferFromToERC721RecipientWithWrongReturnDataReverts(uint256) (runs: 256, μ: 200852, ~: 200908) -ERC721Test:testSafeTransferFromToERC721RecipientWithWrongReturnDataWithDataReverts(uint256,bytes) (runs: 256, μ: 202121, ~: 202117) -ERC721Test:testSafeTransferFromToNonERC721RecipientReverts(uint256) (runs: 256, μ: 131260, ~: 131198) -ERC721Test:testSafeTransferFromToNonERC721RecipientWithDataReverts(uint256,bytes) (runs: 256, μ: 132601, ~: 132599) -ERC721Test:testSafeTransferFromToRevertingERC721RecipientReverts(uint256) (runs: 256, μ: 233927, ~: 233985) -ERC721Test:testSafeTransferFromToRevertingERC721RecipientWithDataReverts(uint256,bytes) (runs: 256, μ: 235263, ~: 235260) -ERC721Test:testSafetyOfCustomStorage(uint256,uint256) (runs: 256, μ: 1063, ~: 713) -ERC721Test:testTransferFrom() (gas: 85744) -ERC721Test:testTransferFrom(uint256) (runs: 256, μ: 114125, ~: 112511) -ERC721Test:testTransferFromApproveAll(uint256) (runs: 256, μ: 119340, ~: 119297) -ERC721Test:testTransferFromNotExistentReverts(address,address,uint256) (runs: 256, μ: 34014, ~: 33992) -ERC721Test:testTransferFromNotOwner(uint256) (runs: 256, μ: 84724, ~: 84691) -ERC721Test:testTransferFromSelf(uint256) (runs: 256, μ: 92806, ~: 92767) -ERC721Test:testTransferFromToZeroReverts(uint256) (runs: 256, μ: 79049, ~: 79023) -ERC721Test:testTransferFromWrongFromReverts(address,uint256) (runs: 256, μ: 80413, ~: 80398) -ERC721Test:test__codesize() (gas: 41923) -FixedPointMathLibTest:testAbs() (gas: 578) -FixedPointMathLibTest:testAbs(int256) (runs: 256, μ: 516, ~: 485) -FixedPointMathLibTest:testAbsEdgeCases() (gas: 432) -FixedPointMathLibTest:testAvg() (gas: 442) -FixedPointMathLibTest:testAvgEdgeCase() (gas: 425) -FixedPointMathLibTest:testAvgSigned() (gas: 897) -FixedPointMathLibTest:testCbrt() (gas: 10261) -FixedPointMathLibTest:testCbrt(uint256) (runs: 256, μ: 1424, ~: 1513) -FixedPointMathLibTest:testCbrtBack(uint256) (runs: 256, μ: 31710, ~: 40363) -FixedPointMathLibTest:testCbrtWad() (gas: 12006) -FixedPointMathLibTest:testCbrtWad(uint256) (runs: 256, μ: 1996, ~: 2005) -FixedPointMathLibTest:testClamp(uint256,uint256,uint256) (runs: 256, μ: 546, ~: 546) -FixedPointMathLibTest:testClampSigned(int256,int256,int256) (runs: 256, μ: 611, ~: 611) -FixedPointMathLibTest:testDist() (gas: 634) -FixedPointMathLibTest:testDist(int256,int256) (runs: 256, μ: 541, ~: 546) -FixedPointMathLibTest:testDistEdgeCases() (gas: 505) -FixedPointMathLibTest:testDivWad() (gas: 702) -FixedPointMathLibTest:testDivWad(uint256,uint256) (runs: 256, μ: 725, ~: 812) -FixedPointMathLibTest:testDivWadEdgeCases() (gas: 458) -FixedPointMathLibTest:testDivWadOverflowReverts(uint256,uint256) (runs: 256, μ: 3749, ~: 3749) -FixedPointMathLibTest:testDivWadUp() (gas: 3084) -FixedPointMathLibTest:testDivWadUp(uint256,uint256) (runs: 256, μ: 795, ~: 942) -FixedPointMathLibTest:testDivWadUpEdgeCases() (gas: 483) -FixedPointMathLibTest:testDivWadUpOverflowReverts(uint256,uint256) (runs: 256, μ: 3748, ~: 3748) -FixedPointMathLibTest:testDivWadUpZeroDenominatorReverts() (gas: 3239) -FixedPointMathLibTest:testDivWadUpZeroDenominatorReverts(uint256) (runs: 256, μ: 3295, ~: 3295) -FixedPointMathLibTest:testDivWadZeroDenominatorReverts() (gas: 3240) -FixedPointMathLibTest:testDivWadZeroDenominatorReverts(uint256) (runs: 256, μ: 3315, ~: 3315) -FixedPointMathLibTest:testExpWad() (gas: 7544) -FixedPointMathLibTest:testFactorial() (gas: 98825) -FixedPointMathLibTest:testFactorialOriginal() (gas: 94187) -FixedPointMathLibTest:testFullMulDiv() (gas: 1211) -FixedPointMathLibTest:testFullMulDiv(uint256,uint256,uint256) (runs: 256, μ: 1373, ~: 996) -FixedPointMathLibTest:testFullMulDivUp(uint256,uint256,uint256) (runs: 256, μ: 1733, ~: 1539) -FixedPointMathLibTest:testFullMulDivUpRevertsIfRoundedUpResultOverflowsCase1() (gas: 3688) -FixedPointMathLibTest:testFullMulDivUpRevertsIfRoundedUpResultOverflowsCase2() (gas: 3699) -FixedPointMathLibTest:testGcd() (gas: 4206) -FixedPointMathLibTest:testGcd(uint256,uint256) (runs: 256, μ: 5030, ~: 2223) -FixedPointMathLibTest:testLambertW0WadAccuracy() (gas: 7194) -FixedPointMathLibTest:testLambertW0WadAccuracy(uint184) (runs: 256, μ: 1763, ~: 386) -FixedPointMathLibTest:testLambertW0WadKnownValues() (gas: 1698467) -FixedPointMathLibTest:testLambertW0WadMonoDebug() (gas: 781937) -FixedPointMathLibTest:testLambertW0WadMonotonicallyIncreasing() (gas: 18627294) -FixedPointMathLibTest:testLambertW0WadMonotonicallyIncreasing(int256,int256) (runs: 256, μ: 7024, ~: 6871) -FixedPointMathLibTest:testLambertW0WadMonotonicallyIncreasing2() (gas: 4083980) -FixedPointMathLibTest:testLambertW0WadMonotonicallyIncreasingAround(int256) (runs: 256, μ: 26638, ~: 17576) -FixedPointMathLibTest:testLambertW0WadMonotonicallyIncreasingAround2(uint96) (runs: 256, μ: 41138, ~: 17636) -FixedPointMathLibTest:testLambertW0WadRevertsForOutOfDomain() (gas: 16568) -FixedPointMathLibTest:testLambertW0WadWithinBounds() (gas: 15587) -FixedPointMathLibTest:testLambertW0WadWithinBounds(int256) (runs: 256, μ: 3223, ~: 3692) -FixedPointMathLibTest:testLnWad() (gas: 2167) -FixedPointMathLibTest:testLnWadBig() (gas: 2178) -FixedPointMathLibTest:testLnWadNegativeReverts() (gas: 3198) -FixedPointMathLibTest:testLnWadOverflowReverts() (gas: 3173) -FixedPointMathLibTest:testLnWadSmall() (gas: 2780) -FixedPointMathLibTest:testLog10() (gas: 76189) -FixedPointMathLibTest:testLog10(uint256,uint256) (runs: 256, μ: 2175, ~: 2237) -FixedPointMathLibTest:testLog10Up() (gas: 4413) -FixedPointMathLibTest:testLog2() (gas: 243209) -FixedPointMathLibTest:testLog256() (gas: 22786) -FixedPointMathLibTest:testLog256(uint256,uint256) (runs: 256, μ: 2023, ~: 2082) -FixedPointMathLibTest:testLog256Up() (gas: 1249) -FixedPointMathLibTest:testLog2Differential(uint256) (runs: 256, μ: 871, ~: 864) -FixedPointMathLibTest:testLog2Up() (gas: 297368) -FixedPointMathLibTest:testMax(uint256,uint256) (runs: 256, μ: 518, ~: 518) -FixedPointMathLibTest:testMaxCasted(uint32,uint32,uint256) (runs: 256, μ: 887, ~: 892) -FixedPointMathLibTest:testMaxSigned(int256,int256) (runs: 256, μ: 475, ~: 471) -FixedPointMathLibTest:testMin(uint256,uint256) (runs: 256, μ: 499, ~: 504) -FixedPointMathLibTest:testMinBrutalized(uint256,uint256) (runs: 256, μ: 829, ~: 839) -FixedPointMathLibTest:testMinSigned(int256,int256) (runs: 256, μ: 475, ~: 480) -FixedPointMathLibTest:testMulDiv() (gas: 1890) -FixedPointMathLibTest:testMulDiv(uint256,uint256,uint256) (runs: 256, μ: 717, ~: 816) -FixedPointMathLibTest:testMulDivEdgeCases() (gas: 740) -FixedPointMathLibTest:testMulDivOverflowReverts(uint256,uint256,uint256) (runs: 256, μ: 3854, ~: 3854) -FixedPointMathLibTest:testMulDivUp() (gas: 2103) -FixedPointMathLibTest:testMulDivUp(uint256,uint256,uint256) (runs: 256, μ: 906, ~: 1121) -FixedPointMathLibTest:testMulDivUpEdgeCases() (gas: 861) -FixedPointMathLibTest:testMulDivUpOverflowReverts(uint256,uint256,uint256) (runs: 256, μ: 3877, ~: 3877) -FixedPointMathLibTest:testMulDivUpZeroDenominator() (gas: 3253) -FixedPointMathLibTest:testMulDivUpZeroDenominatorReverts(uint256,uint256) (runs: 256, μ: 3310, ~: 3310) -FixedPointMathLibTest:testMulDivZeroDenominatorReverts() (gas: 3252) -FixedPointMathLibTest:testMulDivZeroDenominatorReverts(uint256,uint256) (runs: 256, μ: 3310, ~: 3310) -FixedPointMathLibTest:testMulWad() (gas: 714) -FixedPointMathLibTest:testMulWad(uint256,uint256) (runs: 256, μ: 659, ~: 756) -FixedPointMathLibTest:testMulWadEdgeCases() (gas: 692) -FixedPointMathLibTest:testMulWadOverflowReverts(uint256,uint256) (runs: 256, μ: 3832, ~: 3832) -FixedPointMathLibTest:testMulWadUp() (gas: 815) -FixedPointMathLibTest:testMulWadUp(uint256,uint256) (runs: 256, μ: 860, ~: 1065) -FixedPointMathLibTest:testMulWadUpEdgeCases() (gas: 793) -FixedPointMathLibTest:testMulWadUpOverflowReverts(uint256,uint256) (runs: 256, μ: 3769, ~: 3769) -FixedPointMathLibTest:testPackUnpackSci() (gas: 129349) -FixedPointMathLibTest:testPackUnpackSci(uint256) (runs: 256, μ: 30147, ~: 30088) -FixedPointMathLibTest:testRPow() (gas: 3276) -FixedPointMathLibTest:testRPowOverflowReverts() (gas: 4973) -FixedPointMathLibTest:testRawAdd(int256,int256) (runs: 256, μ: 484, ~: 484) -FixedPointMathLibTest:testRawAdd(uint256,uint256) (runs: 256, μ: 485, ~: 485) -FixedPointMathLibTest:testRawAddMod(uint256,uint256,uint256) (runs: 256, μ: 577, ~: 577) -FixedPointMathLibTest:testRawDiv(uint256,uint256) (runs: 256, μ: 487, ~: 487) -FixedPointMathLibTest:testRawMod(uint256,uint256) (runs: 256, μ: 422, ~: 422) -FixedPointMathLibTest:testRawMul(int256,int256) (runs: 256, μ: 412, ~: 412) -FixedPointMathLibTest:testRawMul(uint256,uint256) (runs: 256, μ: 442, ~: 442) -FixedPointMathLibTest:testRawMulMod(uint256,uint256,uint256) (runs: 256, μ: 512, ~: 512) -FixedPointMathLibTest:testRawSDiv(int256,int256) (runs: 256, μ: 487, ~: 487) -FixedPointMathLibTest:testRawSMod(int256,int256) (runs: 256, μ: 422, ~: 422) -FixedPointMathLibTest:testRawSub(int256,int256) (runs: 256, μ: 485, ~: 485) -FixedPointMathLibTest:testRawSub(uint256,uint256) (runs: 256, μ: 440, ~: 440) -FixedPointMathLibTest:testSci() (gas: 1838635) -FixedPointMathLibTest:testSci(uint256) (runs: 256, μ: 31826, ~: 37538) -FixedPointMathLibTest:testSci2(uint256) (runs: 256, μ: 901, ~: 911) -FixedPointMathLibTest:testSqrt() (gas: 42576) -FixedPointMathLibTest:testSqrt(uint256) (runs: 256, μ: 1037, ~: 1086) -FixedPointMathLibTest:testSqrtBack(uint256) (runs: 256, μ: 14284, ~: 385) -FixedPointMathLibTest:testSqrtHashed(uint256) (runs: 256, μ: 53185, ~: 53549) -FixedPointMathLibTest:testSqrtHashedSingle() (gas: 53041) -FixedPointMathLibTest:testSqrtWad() (gas: 7470) -FixedPointMathLibTest:testSqrtWad(uint256) (runs: 256, μ: 1493, ~: 1498) -FixedPointMathLibTest:testZeroFloorSub(uint256,uint256) (runs: 256, μ: 503, ~: 503) -FixedPointMathLibTest:testZeroFloorSubCasted(uint32,uint32,uint256) (runs: 256, μ: 926, ~: 886) -FixedPointMathLibTest:test__codesize() (gas: 39696) -GasBurnerLibTest:testBurnGas() (gas: 1700805) -GasBurnerLibTest:test__codesize() (gas: 1435) -JSONParserLibTest:testDecodeEncodedStringDoesNotRevert(string) (runs: 256, μ: 58291, ~: 57099) -JSONParserLibTest:testDecodeInvalidStringReverts() (gas: 174221) -JSONParserLibTest:testDecodeString() (gas: 201120) -JSONParserLibTest:testEmptyItem() (gas: 3693) -JSONParserLibTest:testParseEmptyArrays() (gas: 591760) -JSONParserLibTest:testParseEmptyObjects() (gas: 592699) -JSONParserLibTest:testParseGas() (gas: 151934) -JSONParserLibTest:testParseInt() (gas: 181371) -JSONParserLibTest:testParseInvalidIntReverts() (gas: 211374) -JSONParserLibTest:testParseInvalidNumberReverts() (gas: 4075307) -JSONParserLibTest:testParseInvalidReverts() (gas: 8599742) -JSONParserLibTest:testParseInvalidStringReverts() (gas: 2121353) -JSONParserLibTest:testParseInvalidUintFromHexReverts() (gas: 90700) -JSONParserLibTest:testParseInvalidUintReverts() (gas: 309989) -JSONParserLibTest:testParseJWTGas() (gas: 52735) -JSONParserLibTest:testParseNumber() (gas: 787032) -JSONParserLibTest:testParseObject() (gas: 49089) -JSONParserLibTest:testParseRecursiveObject() (gas: 104550) -JSONParserLibTest:testParseSimpleArray() (gas: 27206) -JSONParserLibTest:testParseSimpleUintArray() (gas: 1227270) -JSONParserLibTest:testParseSpecials() (gas: 295420) -JSONParserLibTest:testParseString() (gas: 1435230) -JSONParserLibTest:testParseUint() (gas: 96207) -JSONParserLibTest:testParseUint(uint256) (runs: 256, μ: 36531, ~: 30330) -JSONParserLibTest:testParseUintFromHex() (gas: 1367554) -JSONParserLibTest:testParseValidObjectDoesNotRevert(string,string) (runs: 256, μ: 49906, ~: 48850) -JSONParserLibTest:test__codesize() (gas: 38289) -LibBitTest:testAnd() (gas: 185677) -LibBitTest:testAnd(bool,bool) (runs: 256, μ: 669, ~: 671) -LibBitTest:testAutoClean(uint256,uint256) (runs: 256, μ: 446, ~: 446) -LibBitTest:testBoolToUint(bool) (runs: 256, μ: 524, ~: 524) -LibBitTest:testCLZ() (gas: 306382) -LibBitTest:testFFS() (gas: 181271) -LibBitTest:testFLS() (gas: 254359) -LibBitTest:testIsPo2() (gas: 65189) -LibBitTest:testIsPo2(uint256) (runs: 256, μ: 4805, ~: 1106) -LibBitTest:testIsPo2(uint8,uint8) (runs: 256, μ: 629, ~: 631) -LibBitTest:testOr() (gas: 188517) -LibBitTest:testOr(bool,bool) (runs: 256, μ: 620, ~: 619) -LibBitTest:testPassInBool() (gas: 802) -LibBitTest:testPopCount() (gas: 63716) -LibBitTest:testPopCount(uint256) (runs: 256, μ: 4925, ~: 1226) -LibBitTest:testReturnsBool() (gas: 735) -LibBitTest:testReverseBitsDifferential(uint256) (runs: 256, μ: 19195, ~: 19195) -LibBitTest:testReverseBytesDifferential(uint256) (runs: 256, μ: 3047, ~: 3047) -LibBitTest:test__codesize() (gas: 6067) -LibBitmapTest:testBitmapClaimWithGetSet() (gas: 27089) -LibBitmapTest:testBitmapClaimWithToggle() (gas: 17479) -LibBitmapTest:testBitmapFindLastSet() (gas: 1300541) -LibBitmapTest:testBitmapFindLastSet(uint256,uint256) (runs: 256, μ: 75758, ~: 76072) -LibBitmapTest:testBitmapGet() (gas: 2565) -LibBitmapTest:testBitmapGet(uint256) (runs: 256, μ: 2593, ~: 2593) -LibBitmapTest:testBitmapPopCount() (gas: 750426) -LibBitmapTest:testBitmapPopCount(uint256,uint256,uint256) (runs: 256, μ: 201971, ~: 171380) -LibBitmapTest:testBitmapPopCountAcrossMultipleBuckets() (gas: 73611) -LibBitmapTest:testBitmapPopCountWithinSingleBucket() (gas: 34032) -LibBitmapTest:testBitmapSet() (gas: 22527) -LibBitmapTest:testBitmapSet(uint256) (runs: 256, μ: 22599, ~: 22599) -LibBitmapTest:testBitmapSetAndGet(uint256) (runs: 256, μ: 22633, ~: 22633) -LibBitmapTest:testBitmapSetBatch() (gas: 3147640) -LibBitmapTest:testBitmapSetBatchAcrossMultipleBuckets() (gas: 452545) -LibBitmapTest:testBitmapSetBatchWithinSingleBucket() (gas: 417119) -LibBitmapTest:testBitmapSetTo() (gas: 14275) -LibBitmapTest:testBitmapSetTo(uint256,bool,uint256) (runs: 256, μ: 13050, ~: 22774) -LibBitmapTest:testBitmapSetTo(uint256,uint256) (runs: 256, μ: 45097, ~: 50098) -LibBitmapTest:testBitmapToggle() (gas: 30810) -LibBitmapTest:testBitmapToggle(uint256,bool) (runs: 256, μ: 17743, ~: 14247) -LibBitmapTest:testBitmapUnset() (gas: 22572) -LibBitmapTest:testBitmapUnset(uint256) (runs: 256, μ: 14342, ~: 14363) -LibBitmapTest:testBitmapUnsetBatch() (gas: 3086142) -LibBitmapTest:testBitmapUnsetBatchAcrossMultipleBuckets() (gas: 453362) -LibBitmapTest:testBitmapUnsetBatchWithinSingleBucket() (gas: 446762) -LibBitmapTest:test__codesize() (gas: 8132) -LibCloneTest:testClone() (gas: 72901) -LibCloneTest:testClone(uint256) (runs: 256, μ: 72952, ~: 72952) -LibCloneTest:testCloneDeteministicWithImmutableArgs() (gas: 192911) -LibCloneTest:testCloneDeteministicWithImmutableArgs(address,uint256,uint256[],bytes,uint64,uint8,uint256) (runs: 256, μ: 1055001, ~: 996152) -LibCloneTest:testCloneDeterministic() (gas: 96699) -LibCloneTest:testCloneDeterministic(uint256,bytes32) (runs: 256, μ: 96769, ~: 96769) -LibCloneTest:testCloneDeterministicRevertsIfAddressAlreadyUsed() (gas: 96882831) -LibCloneTest:testCloneWithImmutableArgs() (gas: 120744) -LibCloneTest:testCloneWithImmutableArgs(uint256,address,uint256,uint256[],uint64,uint8) (runs: 256, μ: 987490, ~: 1012210) -LibCloneTest:testCloneWithImmutableArgsRevertsIfDataTooBig() (gas: 97305788) -LibCloneTest:testDeployDeterministicERC1967() (gas: 123067) -LibCloneTest:testDeployDeterministicERC1967(uint256,bytes32) (runs: 256, μ: 123072, ~: 123072) -LibCloneTest:testDeployERC1967() (gas: 99147) -LibCloneTest:testDeployERC1967(uint256) (runs: 256, μ: 99220, ~: 99220) -LibCloneTest:testInitialDeposit() (gas: 323883) -LibCloneTest:testStartsWith(uint256) (runs: 256, μ: 29015, ~: 29025) -LibCloneTest:test__codesize() (gas: 17578) -LibMapTest:testFoundStatementDifferential(uint256,uint256,uint256) (runs: 256, μ: 499, ~: 499) -LibMapTest:testGeneralMapFunctionsGas() (gas: 3304775) -LibMapTest:testGeneralMapFunctionsWithSmallBitWidths(uint256) (runs: 256, μ: 79345, ~: 85415) -LibMapTest:testGeneralMapFunctionsWithZeroBitWidth() (gas: 12521) -LibMapTest:testGeneralMapSearchSorted(uint256) (runs: 256, μ: 191920, ~: 127091) -LibMapTest:testMapGetFromBigArray() (gas: 3308) -LibMapTest:testMapGetUint8() (gas: 3335) -LibMapTest:testMapSetUint8() (gas: 23321) -LibMapTest:testMapSetUint8FromBigArray() (gas: 23264) -LibMapTest:testUint128MapSearchSorted(uint256) (runs: 256, μ: 188304, ~: 131590) -LibMapTest:testUint128MapSetAndGet() (gas: 1607245) -LibMapTest:testUint128MapSetAndGet(uint256) (runs: 256, μ: 172303, ~: 169095) -LibMapTest:testUint128MapSetAndGet2(uint256) (runs: 256, μ: 68640, ~: 69275) -LibMapTest:testUint16MapSearchSorted(uint256) (runs: 256, μ: 92917, ~: 86851) -LibMapTest:testUint16MapSetAndGet() (gas: 958193) -LibMapTest:testUint16MapSetAndGet(uint256) (runs: 256, μ: 81211, ~: 81195) -LibMapTest:testUint16MapSetAndGet2(uint256) (runs: 256, μ: 66936, ~: 69189) -LibMapTest:testUint32MapSearchSorted(uint256) (runs: 256, μ: 100991, ~: 86096) -LibMapTest:testUint32MapSetAndGet() (gas: 1067093) -LibMapTest:testUint32MapSetAndGet(uint256) (runs: 256, μ: 114350, ~: 124336) -LibMapTest:testUint32MapSetAndGet2(uint256) (runs: 256, μ: 67417, ~: 69139) -LibMapTest:testUint32Maps(uint256) (runs: 256, μ: 64099, ~: 72236) -LibMapTest:testUint40MapSearchSorted(uint256) (runs: 256, μ: 134900, ~: 106702) -LibMapTest:testUint40MapSetAndGet() (gas: 1127344) -LibMapTest:testUint40MapSetAndGet(uint256) (runs: 256, μ: 136972, ~: 145996) -LibMapTest:testUint40MapSetAndGet2(uint256) (runs: 256, μ: 67025, ~: 69222) -LibMapTest:testUint64MapSearchSorted(uint256) (runs: 256, μ: 126533, ~: 108400) -LibMapTest:testUint64MapSetAndGet() (gas: 1220437) -LibMapTest:testUint64MapSetAndGet(uint256) (runs: 256, μ: 141134, ~: 146540) -LibMapTest:testUint64MapSetAndGet2(uint256) (runs: 256, μ: 65001, ~: 69181) -LibMapTest:testUint8MapSearchSorted(uint256) (runs: 256, μ: 98669, ~: 86281) -LibMapTest:testUint8MapSetAndGet() (gas: 883454) -LibMapTest:testUint8MapSetAndGet(uint256) (runs: 256, μ: 59148, ~: 59100) -LibMapTest:testUint8MapSetAndGet2(uint256) (runs: 256, μ: 67769, ~: 69101) -LibMapTest:test__codesize() (gas: 13187) -LibPRNGTest:testLCGGas() (gas: 20736) -LibPRNGTest:testPRNGGas() (gas: 25645) -LibPRNGTest:testPRNGNext() (gas: 16184) -LibPRNGTest:testPRNGShuffle() (gas: 504485) -LibPRNGTest:testPRNGShuffleBytes() (gas: 222095) -LibPRNGTest:testPRNGShuffleBytesGas() (gas: 1322452) -LibPRNGTest:testPRNGShuffleGas() (gas: 1610949) -LibPRNGTest:testPRNGUniform() (gas: 559341) -LibPRNGTest:test__codesize() (gas: 4526) -LibRLPTest:testComputeAddressDifferential(address,uint256) (runs: 256, μ: 1948, ~: 1831) -LibRLPTest:testComputeAddressForLargeNonces() (gas: 1771) -LibRLPTest:testComputeAddressForSmallNonces() (gas: 967) -LibRLPTest:testComputeAddressOriginalForLargeNonces() (gas: 1887) -LibRLPTest:testComputeAddressOriginalForSmallNonces() (gas: 1477) -LibRLPTest:test__codesize() (gas: 8108) -LibSortTest:testInsertionSortAddressesDifferential(uint256) (runs: 256, μ: 35967, ~: 25750) -LibSortTest:testInsertionSortInts() (gas: 112135) -LibSortTest:testInsertionSortPsuedorandom() (gas: 62806) -LibSortTest:testInsertionSortPsuedorandom(uint256) (runs: 256, μ: 61996, ~: 61637) -LibSortTest:testIsSortedAddressesDifferential(uint256) (runs: 256, μ: 37530, ~: 29367) -LibSortTest:testIsSortedAndUniquifiedAddressesDifferential(uint256) (runs: 256, μ: 33725, ~: 29154) -LibSortTest:testIsSortedAndUniquifiedDifferential(uint256) (runs: 256, μ: 33577, ~: 26806) -LibSortTest:testIsSortedAndUniquifiedIntsDifferential(uint256) (runs: 256, μ: 89821, ~: 32035) -LibSortTest:testIsSortedDifferential(uint256) (runs: 256, μ: 34398, ~: 27521) -LibSortTest:testIsSortedIntsDifferential(uint256) (runs: 256, μ: 43804, ~: 30264) -LibSortTest:testReverse() (gas: 114573) -LibSortTest:testSearchSorted() (gas: 118972) -LibSortTest:testSearchSortedAddresses() (gas: 254850) -LibSortTest:testSearchSortedBasicCases() (gas: 2244) -LibSortTest:testSearchSortedDifferential(uint256) (runs: 256, μ: 46869, ~: 29238) -LibSortTest:testSearchSortedEdgeCases() (gas: 1675) -LibSortTest:testSearchSortedElementInArray(uint256) (runs: 256, μ: 52908, ~: 31626) -LibSortTest:testSearchSortedElementNotInArray() (gas: 6338) -LibSortTest:testSearchSortedElementNotInArray(uint256) (runs: 256, μ: 67613, ~: 35422) -LibSortTest:testSearchSortedInts() (gas: 120794) -LibSortTest:testSearchSortedInts(int256[],int256) (runs: 256, μ: 11887, ~: 12059) -LibSortTest:testSearchSortedWithEmptyArray() (gas: 738) -LibSortTest:testSort(uint256) (runs: 256, μ: 46941, ~: 29774) -LibSortTest:testSortAddressesDifferential(uint256) (runs: 256, μ: 106312, ~: 29625) -LibSortTest:testSortAddressesPsuedorandom() (gas: 144831) -LibSortTest:testSortAddressesPsuedorandom(uint256) (runs: 256, μ: 146511, ~: 146653) -LibSortTest:testSortAddressesPsuedorandomBrutalizeUpperBits() (gas: 195855) -LibSortTest:testSortAddressesReversed() (gas: 45400) -LibSortTest:testSortAddressesSorted() (gas: 42376) -LibSortTest:testSortBasicCase() (gas: 1210) -LibSortTest:testSortChecksumed(uint256) (runs: 256, μ: 41765, ~: 28194) -LibSortTest:testSortDifferential(uint256) (runs: 256, μ: 50383, ~: 27194) -LibSortTest:testSortInts() (gas: 612197) -LibSortTest:testSortMostlySame() (gas: 67374) -LibSortTest:testSortOriginalMostlySame() (gas: 207985) -LibSortTest:testSortOriginalPsuedorandom() (gas: 284071) -LibSortTest:testSortOriginalPsuedorandom(uint256) (runs: 256, μ: 285134, ~: 285198) -LibSortTest:testSortOriginalReversed() (gas: 156282) -LibSortTest:testSortOriginalSorted() (gas: 142263) -LibSortTest:testSortPsuedorandom() (gas: 138551) -LibSortTest:testSortPsuedorandom(uint256) (runs: 256, μ: 139957, ~: 140097) -LibSortTest:testSortPsuedorandomNonuniform() (gas: 145689) -LibSortTest:testSortPsuedorandomNonuniform(uint256) (runs: 256, μ: 145242, ~: 145503) -LibSortTest:testSortReversed() (gas: 38827) -LibSortTest:testSortSorted() (gas: 35785) -LibSortTest:testSortTestOverhead() (gas: 102190) -LibSortTest:testSortedDifferenceDifferential() (gas: 174014) -LibSortTest:testSortedDifferenceDifferential(uint256) (runs: 256, μ: 35743, ~: 35595) -LibSortTest:testSortedDifferenceDifferentialInt(uint256) (runs: 256, μ: 35099, ~: 34485) -LibSortTest:testSortedDifferenceUnionIntersection(uint256) (runs: 256, μ: 58456, ~: 53535) -LibSortTest:testSortedIntersectionDifferential() (gas: 208149) -LibSortTest:testSortedIntersectionDifferential(uint256) (runs: 256, μ: 32268, ~: 31481) -LibSortTest:testSortedIntersectionDifferentialInt(uint256) (runs: 256, μ: 35871, ~: 35215) -LibSortTest:testSortedUnionDifferential() (gas: 212603) -LibSortTest:testSortedUnionDifferential(uint256) (runs: 256, μ: 34338, ~: 34163) -LibSortTest:testSortedUnionDifferentialInt(uint256) (runs: 256, μ: 38745, ~: 39202) -LibSortTest:testTwoComplementConversionSort(int256,int256) (runs: 256, μ: 515, ~: 514) -LibSortTest:testUniquifySorted() (gas: 1710) -LibSortTest:testUniquifySorted(uint256) (runs: 256, μ: 42329, ~: 24996) -LibSortTest:testUniquifySortedAddress() (gas: 4039) -LibSortTest:testUniquifySortedAddress(uint256) (runs: 256, μ: 67632, ~: 30265) -LibSortTest:testUniquifySortedDifferential(uint256) (runs: 256, μ: 67605, ~: 33106) -LibSortTest:testUniquifySortedWithEmptyArray() (gas: 506) -LibSortTest:test__codesize() (gas: 23230) -LibStringTest:testAddressToHexStringZeroRightPadded(address) (runs: 256, μ: 3370, ~: 3370) -LibStringTest:testBytesToHexString() (gas: 6446) -LibStringTest:testBytesToHexString(bytes) (runs: 256, μ: 692470, ~: 577893) -LibStringTest:testBytesToHexStringNoPrefix() (gas: 6064) -LibStringTest:testBytesToHexStringNoPrefix(bytes) (runs: 256, μ: 757633, ~: 578238) -LibStringTest:testContains() (gas: 43922) -LibStringTest:testFromAddressToHexString() (gas: 3774) -LibStringTest:testFromAddressToHexStringChecksummed() (gas: 40426) -LibStringTest:testFromAddressToHexStringChecksummedDifferential(uint256) (runs: 256, μ: 738350, ~: 567130) -LibStringTest:testFromAddressToHexStringWithLeadingZeros() (gas: 3795) -LibStringTest:testHexStringNoPrefixVariants(uint256,uint256) (runs: 256, μ: 648819, ~: 577685) -LibStringTest:testStringConcat() (gas: 7385) -LibStringTest:testStringConcat(string,string) (runs: 256, μ: 632721, ~: 558799) -LibStringTest:testStringConcatOriginal() (gas: 8045) -LibStringTest:testStringDirectReturn() (gas: 8213) -LibStringTest:testStringDirectReturn(string) (runs: 256, μ: 3614, ~: 3455) -LibStringTest:testStringEndsWith() (gas: 2852) -LibStringTest:testStringEndsWith(uint256) (runs: 256, μ: 845913, ~: 589643) -LibStringTest:testStringEq(string,string) (runs: 256, μ: 1586, ~: 1587) -LibStringTest:testStringEqs() (gas: 1861) -LibStringTest:testStringEscapeHTML() (gas: 11903) -LibStringTest:testStringEscapeHTML(uint256) (runs: 256, μ: 724158, ~: 606340) -LibStringTest:testStringEscapeJSON() (gas: 53305) -LibStringTest:testStringEscapeJSONHexEncode() (gas: 694952) -LibStringTest:testStringIndexOf() (gas: 17420) -LibStringTest:testStringIndexOf(uint256) (runs: 256, μ: 811014, ~: 602628) -LibStringTest:testStringIndicesOf() (gas: 11658) -LibStringTest:testStringIndicesOf(uint256) (runs: 256, μ: 765072, ~: 604437) -LibStringTest:testStringIs7BitASCII() (gas: 205190) -LibStringTest:testStringIs7BitASCIIDifferential(bytes) (runs: 256, μ: 702666, ~: 557455) -LibStringTest:testStringLastIndexOf() (gas: 24020) -LibStringTest:testStringLastIndexOf(uint256) (runs: 256, μ: 647483, ~: 598426) -LibStringTest:testStringLowerDifferential() (gas: 3947004) -LibStringTest:testStringLowerDifferential(string) (runs: 256, μ: 8929, ~: 8560) -LibStringTest:testStringLowerOriginal() (gas: 1783) -LibStringTest:testStringPackAndUnpackOne() (gas: 705911) -LibStringTest:testStringPackAndUnpackOne(string) (runs: 256, μ: 631676, ~: 557252) -LibStringTest:testStringPackAndUnpackOneDifferential(string) (runs: 256, μ: 671502, ~: 556594) -LibStringTest:testStringPackAndUnpackTwo() (gas: 864247) -LibStringTest:testStringPackAndUnpackTwo(string,string) (runs: 256, μ: 639722, ~: 558976) -LibStringTest:testStringPackAndUnpackTwoDifferential(string,string) (runs: 256, μ: 665171, ~: 557636) -LibStringTest:testStringRepeat() (gas: 8998) -LibStringTest:testStringRepeat(string,uint256) (runs: 256, μ: 648566, ~: 560206) -LibStringTest:testStringRepeatOriginal() (gas: 13671) -LibStringTest:testStringReplace(uint256) (runs: 256, μ: 669171, ~: 608064) -LibStringTest:testStringReplaceLong() (gas: 9799) -LibStringTest:testStringReplaceMedium() (gas: 8541) -LibStringTest:testStringReplaceShort() (gas: 17434) -LibStringTest:testStringRuneCount() (gas: 2972587) -LibStringTest:testStringRuneCountDifferential(string) (runs: 256, μ: 6085, ~: 5862) -LibStringTest:testStringSlice() (gas: 17767) -LibStringTest:testStringSlice(uint256) (runs: 256, μ: 664210, ~: 602350) -LibStringTest:testStringSplit() (gas: 20265) -LibStringTest:testStringSplit(uint256) (runs: 256, μ: 745574, ~: 601378) -LibStringTest:testStringStartsWith() (gas: 2567) -LibStringTest:testStringStartsWith(uint256) (runs: 256, μ: 656349, ~: 589578) -LibStringTest:testStringUpperDifferential() (gas: 3492145) -LibStringTest:testStringUpperDifferential(string) (runs: 256, μ: 8929, ~: 8560) -LibStringTest:testStringUpperOriginal() (gas: 1759) -LibStringTest:testToHexStringFixedLengthInsufficientLength() (gas: 3344) -LibStringTest:testToHexStringFixedLengthPositiveNumberLong() (gas: 4454) -LibStringTest:testToHexStringFixedLengthPositiveNumberShort() (gas: 1459) -LibStringTest:testToHexStringFixedLengthUint256Max() (gas: 4478) -LibStringTest:testToHexStringFixedLengthZeroRightPadded(uint256,uint256) (runs: 256, μ: 8223, ~: 4860) -LibStringTest:testToHexStringPositiveNumber() (gas: 1407) -LibStringTest:testToHexStringUint256Max() (gas: 4219) -LibStringTest:testToHexStringZero() (gas: 1345) -LibStringTest:testToHexStringZeroRightPadded(uint256) (runs: 256, μ: 2025, ~: 1322) -LibStringTest:testToMinimalHexStringNoPrefixPositiveNumber() (gas: 6022) -LibStringTest:testToMinimalHexStringNoPrefixUint256Max() (gas: 4064) -LibStringTest:testToMinimalHexStringNoPrefixZero() (gas: 1350) -LibStringTest:testToMinimalHexStringNoPrefixZeroRightPadded(uint256) (runs: 256, μ: 2039, ~: 1343) -LibStringTest:testToMinimalHexStringPositiveNumber() (gas: 6111) -LibStringTest:testToMinimalHexStringUint256Max() (gas: 4247) -LibStringTest:testToMinimalHexStringZero() (gas: 1393) -LibStringTest:testToMinimalHexStringZeroRightPadded(uint256) (runs: 256, μ: 2072, ~: 1369) -LibStringTest:testToStringPositiveNumber() (gas: 1448) -LibStringTest:testToStringPositiveNumberBrutalized() (gas: 1056792) -LibStringTest:testToStringSignedDifferential(int256) (runs: 256, μ: 644623, ~: 557786) -LibStringTest:testToStringSignedGas() (gas: 7253) -LibStringTest:testToStringSignedMemory(int256) (runs: 256, μ: 666499, ~: 556934) -LibStringTest:testToStringSignedOriginalGas() (gas: 9861) -LibStringTest:testToStringUint256Max() (gas: 7442) -LibStringTest:testToStringUint256MaxBrutalized() (gas: 569890) -LibStringTest:testToStringZero() (gas: 1196) -LibStringTest:testToStringZeroBrutalized() (gas: 557263) -LibStringTest:testToStringZeroRightPadded(uint256) (runs: 256, μ: 682989, ~: 557533) -LibStringTest:test__codesize() (gas: 41335) -LibZipTest:testCdCompress() (gas: 156007) -LibZipTest:testCdCompressDecompress(bytes) (runs: 256, μ: 716189, ~: 645775) -LibZipTest:testCdCompressDecompress(uint256) (runs: 256, μ: 782010, ~: 694845) -LibZipTest:testCdDecompressOnInvalidInput() (gas: 26080) -LibZipTest:testCdFallback() (gas: 5674556) -LibZipTest:testCdFallback(bytes,uint256) (runs: 256, μ: 1199984, ~: 1049321) -LibZipTest:testCdFallbackDecompressor(bytes) (runs: 256, μ: 121353, ~: 117716) -LibZipTest:testCdFallbackDecompressor(uint256) (runs: 256, μ: 167254, ~: 156539) -LibZipTest:testCdFallbackMaskTrick(uint256,uint256) (runs: 256, μ: 689, ~: 663) -LibZipTest:testDecompressWontRevert(bytes) (runs: 256, μ: 725418, ~: 626106) -LibZipTest:testFlzCompressDecompress() (gas: 1920321) -LibZipTest:testFlzCompressDecompress(bytes) (runs: 256, μ: 846262, ~: 678560) -LibZipTest:testFlzCompressDecompress2() (gas: 942068) -LibZipTest:test__codesize() (gas: 21712) -MerkleProofLibTest:testEmptyCalldataHelpers() (gas: 1086) -MerkleProofLibTest:testVerifyMultiProof(bool,bool,bool,bool,bytes32) (runs: 256, μ: 757466, ~: 629117) -MerkleProofLibTest:testVerifyMultiProofForHeightOneTree(bool,bool,bool,bool,bool,bool[]) (runs: 256, μ: 37116, ~: 37668) -MerkleProofLibTest:testVerifyMultiProofForHeightTwoTree(bool,bool,bool,bool,bool,bytes32) (runs: 256, μ: 6515, ~: 6489) -MerkleProofLibTest:testVerifyMultiProofForSingleLeaf(bytes32[],uint256) (runs: 256, μ: 880518, ~: 787948) -MerkleProofLibTest:testVerifyMultiProofIsInvalid() (gas: 627912) -MerkleProofLibTest:testVerifyMultiProofIsValid() (gas: 629253) -MerkleProofLibTest:testVerifyMultiProofMalicious() (gas: 7998) -MerkleProofLibTest:testVerifyProof(bytes32[],uint256) (runs: 256, μ: 885329, ~: 772683) -MerkleProofLibTest:testVerifyProofBasicCase(bool,bool,bool,bytes32) (runs: 256, μ: 4077, ~: 4222) -MerkleProofLibTest:testVerifyProofBasicCaseIsInvalid() (gas: 3581) -MerkleProofLibTest:testVerifyProofBasicCaseIsValid() (gas: 3590) -MerkleProofLibTest:testVerifyProofForHeightOneTree(bool,bool,bool,bool) (runs: 256, μ: 2697, ~: 2971) -MerkleProofLibTest:test__codesize() (gas: 12117) -MetadataReaderLibTest:testBoundsCheckDifferential(uint256) (runs: 256, μ: 25571, ~: 25536) -MetadataReaderLibTest:testReadBytes32String() (gas: 641910) -MetadataReaderLibTest:testReadBytes32StringTruncated() (gas: 752829) -MetadataReaderLibTest:testReadString(uint256) (runs: 256, μ: 857661, ~: 722894) -MetadataReaderLibTest:testReadStringChopped() (gas: 223238) -MetadataReaderLibTest:testReadStringTruncated(uint256) (runs: 256, μ: 794124, ~: 710222) -MetadataReaderLibTest:testReadUint() (gas: 542217) -MetadataReaderLibTest:testReadUint(uint256) (runs: 256, μ: 22765, ~: 23854) -MetadataReaderLibTest:test__codesize() (gas: 8645) -MinHeapLibTest:testHeapEnqueue(uint256) (runs: 256, μ: 184437, ~: 184485) -MinHeapLibTest:testHeapEnqueueGas(uint256) (runs: 256, μ: 293352, ~: 293395) -MinHeapLibTest:testHeapPushAndPop(uint256) (runs: 256, μ: 108526, ~: 99702) -MinHeapLibTest:testHeapPushPop(uint256) (runs: 256, μ: 246549, ~: 254170) -MinHeapLibTest:testHeapReplace(uint256) (runs: 256, μ: 306102, ~: 314615) -MinHeapLibTest:testHeapRoot(uint256) (runs: 256, μ: 5232, ~: 5232) -MinHeapLibTest:test__codesize() (gas: 5404) -MulticallableTest:testMulticallableBenchmark() (gas: 29588) -MulticallableTest:testMulticallableOriginalBenchmark() (gas: 38849) -MulticallableTest:testMulticallablePreservesMsgSender() (gas: 11193) -MulticallableTest:testMulticallableReturnDataIsProperlyEncoded() (gas: 11667) -MulticallableTest:testMulticallableReturnDataIsProperlyEncoded(string,string,uint256) (runs: 256, μ: 9874, ~: 9484) -MulticallableTest:testMulticallableReturnDataIsProperlyEncoded(uint256,uint256,uint256,uint256) (runs: 256, μ: 11813, ~: 11813) -MulticallableTest:testMulticallableRevertWithCustomError() (gas: 11841) -MulticallableTest:testMulticallableRevertWithMessage() (gas: 13513) -MulticallableTest:testMulticallableRevertWithMessage(string) (runs: 256, μ: 14126, ~: 13974) -MulticallableTest:testMulticallableRevertWithNothing() (gas: 11730) -MulticallableTest:testMulticallableWithNoData() (gas: 6322) -MulticallableTest:test__codesize() (gas: 9796) -OwnableRolesTest:testBytecodeSize() (gas: 350635) -OwnableRolesTest:testGrantAndRemoveRolesDirect(address,uint256,uint256) (runs: 256, μ: 38391, ~: 40582) -OwnableRolesTest:testGrantAndRevokeOrRenounceRoles(address,bool,bool,bool,uint256,uint256) (runs: 256, μ: 27601, ~: 28129) -OwnableRolesTest:testGrantRoles() (gas: 36122) -OwnableRolesTest:testHandoverOwnership() (gas: 32342) -OwnableRolesTest:testHandoverOwnership(address) (runs: 256, μ: 32400, ~: 32386) -OwnableRolesTest:testHandoverOwnershipAfterExpiration() (gas: 36930) -OwnableRolesTest:testHandoverOwnershipBeforeExpiration() (gas: 28667) -OwnableRolesTest:testHandoverOwnershipRevertsIfCompleteIsNotOwner() (gas: 35585) -OwnableRolesTest:testHandoverOwnershipWithCancellation() (gas: 30715) -OwnableRolesTest:testHasAllRoles(address,uint256,uint256,uint256,bool) (runs: 256, μ: 33540, ~: 33537) -OwnableRolesTest:testHasAnyRole(address,uint256,uint256) (runs: 256, μ: 32710, ~: 33332) -OwnableRolesTest:testInitializeOwnerDirect() (gas: 16724) -OwnableRolesTest:testOnlyOwnerModifier(address,bool) (runs: 256, μ: 23708, ~: 32884) -OwnableRolesTest:testOnlyOwnerOrRolesModifier() (gas: 36582) -OwnableRolesTest:testOnlyOwnerOrRolesModifier(address,bool,uint256,uint256) (runs: 256, μ: 55705, ~: 57924) -OwnableRolesTest:testOnlyRolesModifier(address,uint256,uint256) (runs: 256, μ: 53721, ~: 57912) -OwnableRolesTest:testOnlyRolesOrOwnerModifier(address,bool,uint256,uint256) (runs: 256, μ: 56866, ~: 58382) -OwnableRolesTest:testOrdinalsFromRoles() (gas: 3610415) -OwnableRolesTest:testOrdinalsFromRoles(uint256) (runs: 256, μ: 69762, ~: 33845) -OwnableRolesTest:testOwnershipHandoverValidForDefaultValue() (gas: 5825) -OwnableRolesTest:testRenounceOwnership() (gas: 12899) -OwnableRolesTest:testRolesFromOrdinals() (gas: 651259) -OwnableRolesTest:testRolesFromOrdinals(uint8[]) (runs: 256, μ: 78088, ~: 74221) -OwnableRolesTest:testSetOwnerDirect() (gas: 17776) -OwnableRolesTest:testSetOwnerDirect(address) (runs: 256, μ: 17901, ~: 17901) -OwnableRolesTest:testSetRolesDirect(uint256) (runs: 256, μ: 99847, ~: 99819) -OwnableRolesTest:testTransferOwnership() (gas: 19657) -OwnableRolesTest:testTransferOwnership(address,bool,bool) (runs: 256, μ: 14470, ~: 12721) -OwnableRolesTest:test__codesize() (gas: 23704) -OwnableTest:testBytecodeSize() (gas: 235158) -OwnableTest:testHandoverOwnership() (gas: 32072) -OwnableTest:testHandoverOwnership(address) (runs: 256, μ: 32094, ~: 32081) -OwnableTest:testHandoverOwnershipAfterExpiration() (gas: 36642) -OwnableTest:testHandoverOwnershipBeforeExpiration() (gas: 28516) -OwnableTest:testHandoverOwnershipRevertsIfCompleteIsNotOwner() (gas: 35341) -OwnableTest:testHandoverOwnershipWithCancellation() (gas: 30318) -OwnableTest:testInitializeOwnerDirect() (gas: 16599) -OwnableTest:testOnlyOwnerModifier(address,bool) (runs: 256, μ: 23624, ~: 32828) -OwnableTest:testOwnershipHandoverValidForDefaultValue() (gas: 5737) -OwnableTest:testRenounceOwnership() (gas: 12755) -OwnableTest:testSetOwnerDirect() (gas: 17686) -OwnableTest:testSetOwnerDirect(address) (runs: 256, μ: 17812, ~: 17812) -OwnableTest:testTransferOwnership() (gas: 19467) -OwnableTest:testTransferOwnership(address,bool,bool) (runs: 256, μ: 14191, ~: 12487) -OwnableTest:test__codesize() (gas: 12233) -ReceiverTest:testETHReceived() (gas: 9621) -ReceiverTest:testOnERC1155BatchReceived() (gas: 48975) -ReceiverTest:testOnERC1155Received() (gas: 46717) -ReceiverTest:testOnERC721Received() (gas: 64108) -ReceiverTest:test__codesize() (gas: 3310) -RedBlackTreeLibTest:testRedBlackTreeBenchUint160() (gas: 3438446) -RedBlackTreeLibTest:testRedBlackTreeBenchUint256() (gas: 5850739) -RedBlackTreeLibTest:testRedBlackTreeClear() (gas: 57543) -RedBlackTreeLibTest:testRedBlackTreeClear(uint256) (runs: 256, μ: 289199, ~: 214826) -RedBlackTreeLibTest:testRedBlackTreeInsertAndRemove(uint256) (runs: 256, μ: 698454, ~: 525248) -RedBlackTreeLibTest:testRedBlackTreeInsertAndRemove2(uint256) (runs: 256, μ: 420135, ~: 388876) -RedBlackTreeLibTest:testRedBlackTreeInsertAndRemove3() (gas: 21537948) -RedBlackTreeLibTest:testRedBlackTreeInsertBenchStep() (gas: 3711319) -RedBlackTreeLibTest:testRedBlackTreeInsertBenchUint160() (gas: 3476937) -RedBlackTreeLibTest:testRedBlackTreeInsertBenchUint256() (gas: 6376001) -RedBlackTreeLibTest:testRedBlackTreeNearest(uint256) (runs: 256, μ: 227576, ~: 222624) -RedBlackTreeLibTest:testRedBlackTreeNearestAfter(uint256) (runs: 256, μ: 254293, ~: 246502) -RedBlackTreeLibTest:testRedBlackTreeNearestBefore(uint256) (runs: 256, μ: 235352, ~: 220313) -RedBlackTreeLibTest:testRedBlackTreePointers() (gas: 91901) -RedBlackTreeLibTest:testRedBlackTreeRejectsEmptyValue() (gas: 3194) -RedBlackTreeLibTest:testRedBlackTreeRemoveViaPointer() (gas: 58199) -RedBlackTreeLibTest:testRedBlackTreeTreeFullReverts() (gas: 50293) -RedBlackTreeLibTest:testRedBlackTreeTryInsertAndRemove() (gas: 56127) -RedBlackTreeLibTest:test__codesize() (gas: 13237) -SSTORE2Test:testReadInvalidPointerCustomBoundsReverts() (gas: 3242) -SSTORE2Test:testReadInvalidPointerCustomBoundsReverts(address,uint256,uint256) (runs: 256, μ: 765666, ~: 632367) -SSTORE2Test:testReadInvalidPointerCustomStartBoundReverts() (gas: 3241) -SSTORE2Test:testReadInvalidPointerCustomStartBoundReverts(address,uint256) (runs: 256, μ: 702843, ~: 632453) -SSTORE2Test:testReadInvalidPointerRevert(address) (runs: 256, μ: 729579, ~: 632130) -SSTORE2Test:testReadInvalidPointerReverts() (gas: 3215) -SSTORE2Test:testWriteRead() (gas: 69590) -SSTORE2Test:testWriteRead(bytes) (runs: 256, μ: 805883, ~: 673902) -SSTORE2Test:testWriteReadCustomBounds() (gas: 34463) -SSTORE2Test:testWriteReadCustomBounds(bytes,uint256,uint256) (runs: 256, μ: 761817, ~: 666288) -SSTORE2Test:testWriteReadCustomBoundsOutOfRangeReverts(bytes,uint256,uint256) (runs: 256, μ: 835843, ~: 669947) -SSTORE2Test:testWriteReadCustomStartBound() (gas: 34651) -SSTORE2Test:testWriteReadCustomStartBound(bytes,uint256) (runs: 256, μ: 776405, ~: 671909) -SSTORE2Test:testWriteReadCustomStartBoundOutOfRangeReverts(bytes,uint256) (runs: 256, μ: 798063, ~: 670044) -SSTORE2Test:testWriteReadDeterministic(bytes) (runs: 256, μ: 875920, ~: 746015) -SSTORE2Test:testWriteReadEmptyBound() (gas: 33825) -SSTORE2Test:testWriteReadEmptyOutOfBoundsReverts() (gas: 36485) -SSTORE2Test:testWriteReadFullBoundedRead() (gas: 69630) -SSTORE2Test:testWriteReadFullStartBound() (gas: 34815) -SSTORE2Test:testWriteReadOutOfBoundsReverts() (gas: 36463) -SSTORE2Test:testWriteReadOutOfStartBoundReverts() (gas: 36467) -SSTORE2Test:testWriteWithTooBigDataReverts() (gas: 97662218) -SSTORE2Test:test__codesize() (gas: 9625) -SafeCastLibTest:testSafeCastToInt(int256) (runs: 256, μ: 4364, ~: 3415) -SafeCastLibTest:testSafeCastToInt256(uint256) (runs: 256, μ: 942, ~: 390) -SafeCastLibTest:testSafeCastToIntBench() (gas: 383456) -SafeCastLibTest:testSafeCastToUint() (gas: 10560) -SafeCastLibTest:testSafeCastToUint(uint256) (runs: 256, μ: 4150, ~: 3351) -SafeCastLibTest:testSafeCastToUint256(int256) (runs: 256, μ: 1297, ~: 432) -SafeCastLibTest:testSafeCastToUintBench() (gas: 326306) -SafeCastLibTest:test__codesize() (gas: 16001) -SafeTransferLibTest:testApproveWithGarbageReverts(address,uint256) (runs: 256, μ: 110649, ~: 125884) -SafeTransferLibTest:testApproveWithMissingReturn() (gas: 31945) -SafeTransferLibTest:testApproveWithMissingReturn(address,uint256) (runs: 256, μ: 30860, ~: 32104) -SafeTransferLibTest:testApproveWithNonContract() (gas: 2989) -SafeTransferLibTest:testApproveWithNonContract(address,address,uint256) (runs: 256, μ: 3513, ~: 3536) -SafeTransferLibTest:testApproveWithNonGarbage(address,uint256) (runs: 256, μ: 77932, ~: 59187) -SafeTransferLibTest:testApproveWithRetry() (gas: 768579) -SafeTransferLibTest:testApproveWithRetry(address,uint256,uint256) (runs: 256, μ: 767124, ~: 768723) -SafeTransferLibTest:testApproveWithRetryWithNonContract() (gas: 2990) -SafeTransferLibTest:testApproveWithRetryWithNonContract(address,address,uint256) (runs: 256, μ: 3510, ~: 3533) -SafeTransferLibTest:testApproveWithReturnsFalseReverts() (gas: 9515) -SafeTransferLibTest:testApproveWithReturnsFalseReverts(address,uint256) (runs: 256, μ: 9652, ~: 9652) -SafeTransferLibTest:testApproveWithReturnsTooLittleReverts() (gas: 9419) -SafeTransferLibTest:testApproveWithReturnsTooLittleReverts(address,uint256) (runs: 256, μ: 9603, ~: 9603) -SafeTransferLibTest:testApproveWithReturnsTooMuch() (gas: 32324) -SafeTransferLibTest:testApproveWithReturnsTooMuch(address,uint256) (runs: 256, μ: 31266, ~: 32510) -SafeTransferLibTest:testApproveWithReturnsTwoReverts(address,uint256) (runs: 256, μ: 9675, ~: 9675) -SafeTransferLibTest:testApproveWithRevertingReverts() (gas: 9376) -SafeTransferLibTest:testApproveWithRevertingReverts(address,uint256) (runs: 256, μ: 9570, ~: 9570) -SafeTransferLibTest:testApproveWithStandardERC20() (gas: 31786) -SafeTransferLibTest:testApproveWithStandardERC20(address,uint256) (runs: 256, μ: 30725, ~: 31969) -SafeTransferLibTest:testBalanceOfStandardERC20() (gas: 7830) -SafeTransferLibTest:testBalanceOfStandardERC20(address,uint256) (runs: 256, μ: 39516, ~: 40437) -SafeTransferLibTest:testForceTransferETHToGriever() (gas: 1511304) -SafeTransferLibTest:testForceTransferETHToGriever(uint256,uint256) (runs: 256, μ: 532316, ~: 570828) -SafeTransferLibTest:testTransferAllETH() (gas: 34598) -SafeTransferLibTest:testTransferAllETH(address) (runs: 256, μ: 34355, ~: 35018) -SafeTransferLibTest:testTransferAllETHToContractWithoutFallbackReverts() (gas: 10788) -SafeTransferLibTest:testTransferAllETHToContractWithoutFallbackReverts(uint256) (runs: 256, μ: 10898, ~: 10898) -SafeTransferLibTest:testTransferAllFromWithStandardERC20() (gas: 33092) -SafeTransferLibTest:testTransferAllFromWithStandardERC20(address,address,uint256) (runs: 256, μ: 56508, ~: 56508) -SafeTransferLibTest:testTransferAllWithStandardERC20() (gas: 30735) -SafeTransferLibTest:testTransferAllWithStandardERC20(address,uint256) (runs: 256, μ: 42114, ~: 42114) -SafeTransferLibTest:testTransferETH() (gas: 34600) -SafeTransferLibTest:testTransferETH(address,uint256) (runs: 256, μ: 32952, ~: 35250) -SafeTransferLibTest:testTransferETHToContractWithoutFallbackReverts() (gas: 10846) -SafeTransferLibTest:testTransferETHToContractWithoutFallbackReverts(uint256) (runs: 256, μ: 10545, ~: 10928) -SafeTransferLibTest:testTransferFromWithGarbageReverts(address,address,uint256) (runs: 256, μ: 984465, ~: 786287) -SafeTransferLibTest:testTransferFromWithMissingReturn() (gas: 670140) -SafeTransferLibTest:testTransferFromWithMissingReturn(address,address,uint256) (runs: 256, μ: 748670, ~: 669734) -SafeTransferLibTest:testTransferFromWithNonContract() (gas: 3022) -SafeTransferLibTest:testTransferFromWithNonContract(address,address,address,uint256) (runs: 256, μ: 3635, ~: 3694) -SafeTransferLibTest:testTransferFromWithNonGarbage(address,address,uint256) (runs: 256, μ: 893726, ~: 719729) -SafeTransferLibTest:testTransferFromWithReturnsFalseReverts() (gas: 635037) -SafeTransferLibTest:testTransferFromWithReturnsFalseReverts(address,address,uint256) (runs: 256, μ: 699133, ~: 635457) -SafeTransferLibTest:testTransferFromWithReturnsTooLittleReverts() (gas: 635204) -SafeTransferLibTest:testTransferFromWithReturnsTooLittleReverts(address,address,uint256) (runs: 256, μ: 724923, ~: 635291) -SafeTransferLibTest:testTransferFromWithReturnsTooMuch() (gas: 669975) -SafeTransferLibTest:testTransferFromWithReturnsTooMuch(address,address,uint256) (runs: 256, μ: 754128, ~: 670469) -SafeTransferLibTest:testTransferFromWithReturnsTwoReverts(address,address,uint256) (runs: 256, μ: 733108, ~: 635396) -SafeTransferLibTest:testTransferFromWithRevertingReverts() (gas: 628748) -SafeTransferLibTest:testTransferFromWithRevertingReverts(address,address,uint256) (runs: 256, μ: 734431, ~: 628831) -SafeTransferLibTest:testTransferFromWithStandardERC20() (gas: 667720) -SafeTransferLibTest:testTransferFromWithStandardERC20(address,address,uint256) (runs: 256, μ: 715461, ~: 667300) -SafeTransferLibTest:testTransferWithGarbageReverts(address,uint256) (runs: 256, μ: 849643, ~: 747655) -SafeTransferLibTest:testTransferWithMissingReturn() (gas: 655554) -SafeTransferLibTest:testTransferWithMissingReturn(address,uint256) (runs: 256, μ: 765497, ~: 656170) -SafeTransferLibTest:testTransferWithNonContract() (gas: 2988) -SafeTransferLibTest:testTransferWithNonContract(address,address,uint256) (runs: 256, μ: 3534, ~: 3557) -SafeTransferLibTest:testTransferWithNonGarbage(address,uint256) (runs: 256, μ: 816012, ~: 723827) -SafeTransferLibTest:testTransferWithReturnsFalseReverts() (gas: 632101) -SafeTransferLibTest:testTransferWithReturnsFalseReverts(address,uint256) (runs: 256, μ: 787756, ~: 632598) -SafeTransferLibTest:testTransferWithReturnsTooLittleReverts() (gas: 632737) -SafeTransferLibTest:testTransferWithReturnsTooLittleReverts(address,uint256) (runs: 256, μ: 766198, ~: 632456) -SafeTransferLibTest:testTransferWithReturnsTooMuch() (gas: 656022) -SafeTransferLibTest:testTransferWithReturnsTooMuch(address,uint256) (runs: 256, μ: 801430, ~: 656583) -SafeTransferLibTest:testTransferWithReturnsTwoReverts(address,uint256) (runs: 256, μ: 743827, ~: 632574) -SafeTransferLibTest:testTransferWithRevertingReverts() (gas: 631994) -SafeTransferLibTest:testTransferWithRevertingReverts(address,uint256) (runs: 256, μ: 792882, ~: 632490) -SafeTransferLibTest:testTransferWithStandardERC20() (gas: 656603) -SafeTransferLibTest:testTransferWithStandardERC20(address,uint256) (runs: 256, μ: 787015, ~: 656344) -SafeTransferLibTest:testTryTransferAllETH() (gas: 148698) -SafeTransferLibTest:testTryTransferETH() (gas: 148745) -SafeTransferLibTest:testTryTransferETHWithNoGrief() (gas: 537083) -SafeTransferLibTest:testTryTransferETHWithNoStorageWrites() (gas: 192518) -SafeTransferLibTest:test__codesize() (gas: 37521) -SignatureCheckerLibTest:testEmptyCalldataHelpers() (gas: 3972) -SignatureCheckerLibTest:testSignatureChecker(bytes32) (runs: 256, μ: 57309, ~: 46802) -SignatureCheckerLibTest:testSignatureCheckerOnEOAWithInvalidSignature() (gas: 21228) -SignatureCheckerLibTest:testSignatureCheckerOnEOAWithInvalidSigner() (gas: 30728) -SignatureCheckerLibTest:testSignatureCheckerOnEOAWithMatchingSignerAndSignature() (gas: 17670) -SignatureCheckerLibTest:testSignatureCheckerOnEOAWithWrongSignedMessageHash() (gas: 21251) -SignatureCheckerLibTest:testSignatureCheckerOnMaliciousWallet() (gas: 31820) -SignatureCheckerLibTest:testSignatureCheckerOnWalletWithInvalidSignature() (gas: 77090) -SignatureCheckerLibTest:testSignatureCheckerOnWalletWithInvalidSigner() (gas: 25608) -SignatureCheckerLibTest:testSignatureCheckerOnWalletWithMatchingSignerAndSignature() (gas: 64034) -SignatureCheckerLibTest:testSignatureCheckerOnWalletWithWrongSignedMessageHash() (gas: 64091) -SignatureCheckerLibTest:testSignatureCheckerOnWalletWithZeroAddressSigner() (gas: 12264) -SignatureCheckerLibTest:testToEthSignedMessageHashDifferential(bytes) (runs: 256, μ: 1304, ~: 1305) -SignatureCheckerLibTest:testToEthSignedMessageHashDifferential(bytes32) (runs: 256, μ: 526, ~: 526) -SignatureCheckerLibTest:test__codesize() (gas: 11030) -SoladyTest:test__codesize() (gas: 1102) -TestPlus:test__codesize() (gas: 406) -UUPSUpgradeableTest:testNotDelegatedGuard() (gas: 15875) -UUPSUpgradeableTest:testOnlyProxyGuard() (gas: 8896) -UUPSUpgradeableTest:testUpgradeTo() (gas: 287374) -UUPSUpgradeableTest:testUpgradeToAndCall() (gas: 309608) -UUPSUpgradeableTest:testUpgradeToAndCallRevertWithCustomError() (gas: 286618) -UUPSUpgradeableTest:testUpgradeToAndCallRevertWithUnauthorized() (gas: 15964) -UUPSUpgradeableTest:testUpgradeToAndCallRevertWithUpgradeFailed() (gas: 18264) -UUPSUpgradeableTest:testUpgradeToRevertWithUnauthorized() (gas: 16143) -UUPSUpgradeableTest:testUpgradeToRevertWithUpgradeFailed() (gas: 18464) -UUPSUpgradeableTest:test__codesize() (gas: 7195) -WETHInvariants:invariantTotalSupplyEqualsBalance() (runs: 256, calls: 3840, reverts: 1737) -WETHInvariants:test__codesize() (gas: 6008) -WETHTest:testDeposit() (gas: 68090) -WETHTest:testDeposit(uint256) (runs: 256, μ: 65899, ~: 68384) -WETHTest:testFallbackDeposit() (gas: 67796) -WETHTest:testFallbackDeposit(uint256) (runs: 256, μ: 65560, ~: 68045) -WETHTest:testMetadata() (gas: 10098) -WETHTest:testPartialWithdraw() (gas: 79566) -WETHTest:testWithdraw() (gas: 59332) -WETHTest:testWithdraw(uint256,uint256) (runs: 256, μ: 76778, ~: 80326) -WETHTest:testWithdrawToContractWithoutReceiveReverts() (gas: 93923) -WETHTest:test__codesize() (gas: 11080) \ No newline at end of file diff --git a/lib/solady/.github/pull_request_template.md b/lib/solady/.github/pull_request_template.md deleted file mode 100644 index 1ddf674..0000000 --- a/lib/solady/.github/pull_request_template.md +++ /dev/null @@ -1,24 +0,0 @@ -## Description - -Describe the changes made in your pull request here. - -## Checklist - -Ensure you completed **all of the steps** below before submitting your pull request: - -- [ ] Ran `forge fmt`? -- [ ] Ran `forge snapshot`? -- [ ] Ran `forge test`? - -_Pull requests with an incomplete checklist will be thrown out._ - - - - - - - - - - - diff --git a/lib/solady/.github/workflows/ci-all-via-ir.yml b/lib/solady/.github/workflows/ci-all-via-ir.yml deleted file mode 100644 index 3c29422..0000000 --- a/lib/solady/.github/workflows/ci-all-via-ir.yml +++ /dev/null @@ -1,54 +0,0 @@ -name: ci-all-via-ir - -on: - workflow_dispatch: - -jobs: - tests: - name: Forge Testing all via-ir - runs-on: ubuntu-latest - - strategy: - matrix: - profile: [via-ir-0,via-ir-1,via-ir-2,via-ir-3] - - steps: - - uses: actions/checkout@v3 - - - name: Install Foundry - uses: onbjerg/foundry-toolchain@v1 - with: - version: nightly - - - name: Install Dependencies - run: forge install - - - name: Run Tests with ${{ matrix.profile }} - run: > - ( [ "${{ matrix.profile }}" = "via-ir-0" ] && - forge test --use 0.8.4 --via-ir && - forge test --use 0.8.5 --via-ir && - forge test --use 0.8.6 --via-ir && - forge test --use 0.8.7 --via-ir && - forge test --use 0.8.8 --via-ir && - forge test --use 0.8.9 --via-ir - ) || - ( [ "${{ matrix.profile }}" = "via-ir-1" ] && - forge test --use 0.8.13 --via-ir && - forge test --use 0.8.12 --via-ir && - forge test --use 0.8.11 --via-ir && - forge test --use 0.8.10 --via-ir - ) || - ( [ "${{ matrix.profile }}" = "via-ir-2" ] && - forge test --use 0.8.14 --via-ir && - forge test --use 0.8.15 --via-ir && - forge test --use 0.8.16 --via-ir && - forge test --use 0.8.17 --via-ir - ) || - ( [ "${{ matrix.profile }}" = "via-ir-3" ] && - forge test --use 0.8.22 --via-ir && - forge test --use 0.8.21 --via-ir && - forge test --use 0.8.20 --via-ir && - forge test --use 0.8.19 --via-ir && - forge test --use 0.8.18 --via-ir - ) diff --git a/lib/solady/.github/workflows/ci-woke.yml b/lib/solady/.github/workflows/ci-woke.yml deleted file mode 100644 index 53a4c94..0000000 --- a/lib/solady/.github/workflows/ci-woke.yml +++ /dev/null @@ -1,40 +0,0 @@ -name: ci - Woke - -on: - workflow_dispatch: - -jobs: - tests: - name: Woke Testing - runs-on: ubuntu-latest - - strategy: - matrix: - profile: [via-ir-off, via-ir-on] - - steps: - - uses: actions/checkout@v3 - - - name: Install Foundry - uses: onbjerg/foundry-toolchain@v1 - with: - version: nightly - - - name: Install Dependencies - run: forge install - - - name: Move Woke test files - run: mv ext/woke tests && mv tests/woke*.toml . - - - name: Install Woke - run: pip3 install woke - - - name: Copy Woke config - if: ${{ matrix.profile }} == 'via-ir-on' - run: cp woke-via-ir.toml woke.toml - - - name: Generate pytypes - run: woke init pytypes - - - name: Run tests - run: woke test diff --git a/lib/solady/.github/workflows/ci.yml b/lib/solady/.github/workflows/ci.yml deleted file mode 100644 index df11c78..0000000 --- a/lib/solady/.github/workflows/ci.yml +++ /dev/null @@ -1,88 +0,0 @@ -name: ci - -on: - push: - paths: - - '**.sol' - - '**.yml' - -jobs: - tests: - name: Forge Testing - runs-on: ubuntu-latest - - strategy: - matrix: - profile: [post-shanghai,post-shanghai-via-ir,solc-past-versions-0,solc-past-versions-1,via-ir,min-solc,min-solc-via-ir,intense] - - steps: - - uses: actions/checkout@v3 - - - name: Install Foundry - uses: onbjerg/foundry-toolchain@v1 - with: - version: nightly - - - name: Install Dependencies - run: forge install - - - name: Run Tests with ${{ matrix.profile }} - run: > - ( [ "${{ matrix.profile }}" = "post-shanghai" ] && - forge test --use 0.8.22 --evm-version "shanghai" - ) || - ( [ "${{ matrix.profile }}" = "post-shanghai-via-ir" ] && - forge test --use 0.8.22 --evm-version "shanghai" --via-ir - ) || - ( [ "${{ matrix.profile }}" = "solc-past-versions-0" ] && - forge test --use 0.8.5 --fuzz-runs 16 && - forge test --use 0.8.6 --fuzz-runs 16 && - forge test --use 0.8.7 --fuzz-runs 16 && - forge test --use 0.8.8 --fuzz-runs 16 && - forge test --use 0.8.9 --fuzz-runs 16 && - forge test --use 0.8.10 --fuzz-runs 16 && - forge test --use 0.8.11 --fuzz-runs 16 && - forge test --use 0.8.12 --fuzz-runs 16 - ) || - ( [ "${{ matrix.profile }}" = "solc-past-versions-1" ] && - forge test --use 0.8.13 --fuzz-runs 16 && - forge test --use 0.8.14 --fuzz-runs 16 && - forge test --use 0.8.15 --fuzz-runs 16 && - forge test --use 0.8.16 --fuzz-runs 16 && - forge test --use 0.8.17 --fuzz-runs 16 && - forge test --use 0.8.18 --fuzz-runs 16 && - forge test --use 0.8.19 --fuzz-runs 16 && - forge test --use 0.8.20 --fuzz-runs 16 && - forge test --use 0.8.21 --fuzz-runs 16 - ) || - ( [ "${{ matrix.profile }}" = "via-ir" ] && - forge test --via-ir - ) || - ( [ "${{ matrix.profile }}" = "min-solc" ] && - forge fmt --check && - forge test --use 0.8.4 - ) || - ( [ "${{ matrix.profile }}" = "min-solc-via-ir" ] && - forge test --use 0.8.4 --via-ir - ) || - ( [ "${{ matrix.profile }}" = "intense" ] && - forge test --fuzz-runs 5000 - ) - - codespell: - runs-on: ${{ matrix.os }} - strategy: - matrix: - os: - - ubuntu-latest - - steps: - - name: Checkout - uses: actions/checkout@v3 - - - name: Run codespell - uses: codespell-project/actions-codespell@v2.0 - with: - check_filenames: true - ignore_words_list: usera - skip: ./.git,package-lock.json,ackee-blockchain-solady-report.pdf,EIP712Mock.sol diff --git a/lib/solady/.gitignore b/lib/solady/.gitignore deleted file mode 100644 index ff0f3d5..0000000 --- a/lib/solady/.gitignore +++ /dev/null @@ -1,36 +0,0 @@ -# NodeJS files -node_modules/ -coverage -coverage.json -typechain - -# Hardhat files -cache -artifacts - -cache/ -out/ - -# Ignore Environment Variables! -.env -.env.prod - -# Ignore all vscode settings -.vscode/ - -# Ignore flattened files -flattened.txt - -broadcast - -# Coverage -lcov.info - -# Woke testing -.woke-build -.woke-logs -pytypes -__pycache__/ -*.py[cod] -.hypothesis/ -woke-coverage.cov diff --git a/lib/solady/.gitmodules b/lib/solady/.gitmodules deleted file mode 100644 index ba5a595..0000000 --- a/lib/solady/.gitmodules +++ /dev/null @@ -1,4 +0,0 @@ -[submodule "lib/ds-test"] - path = lib/ds-test - url = https://github.com/dapphub/ds-test - branch = 9310e879db8ba3ea6d5c6489a579118fd264a3f5 \ No newline at end of file diff --git a/lib/solady/LICENSE.txt b/lib/solady/LICENSE.txt deleted file mode 100644 index c1939a1..0000000 --- a/lib/solady/LICENSE.txt +++ /dev/null @@ -1,21 +0,0 @@ -MIT License - -Copyright (c) 2022 Solady. - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all -copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE -SOFTWARE. \ No newline at end of file diff --git a/lib/solady/README.md b/lib/solady/README.md deleted file mode 100644 index 814704d..0000000 --- a/lib/solady/README.md +++ /dev/null @@ -1,140 +0,0 @@ -# solady - -[![NPM][npm-shield]][npm-url] -[![CI][ci-shield]][ci-url] -[![Solidity][solidity-shield]][solidity-ci-url] - -Gas optimized Solidity snippets. - -I'm sooooooOooooooooOoooOoooooooooooooooo... - -## Installation - -To install with [**Foundry**](https://github.com/gakonst/foundry): - -```sh -forge install vectorized/solady -``` - -To install with [**Hardhat**](https://github.com/nomiclabs/hardhat) or [**Truffle**](https://github.com/trufflesuite/truffle): - -```sh -npm install solady -``` - -## Contracts - -The Solidity smart contracts are located in the `src` directory. - -```ml -accounts -├─ Receiver — "Receiver mixin for ETH and safe-transferred ERC721 and ERC1155 tokens" -├─ ERC1271 — "ERC1271 mixin with nested EIP-712 approach" -├─ ERC4337 — "Simple ERC4337 account implementation" -├─ ERC4337Factory — "Simple ERC4337 account factory implementation" -├─ ERC6551 — "Simple ERC6551 account implementation" -├─ ERC6551Proxy — "Relay proxy for upgradeable ERC6551 accounts" -auth -├─ Ownable — "Simple single owner authorization mixin" -├─ OwnableRoles — "Simple single owner and multiroles authorization mixin" -tokens -├─ WETH — "Simple Wrapped Ether implementation" -├─ ERC20 — "Simple ERC20 + EIP-2612 implementation" -├─ ERC4626 — "Simple ERC4626 tokenized Vault implementation" -├─ ERC721 — "Simple ERC721 implementation with storage hitchhiking" -├─ ERC2981 — "Simple ERC2981 NFT Royalty Standard implementation" -├─ ERC1155 — "Simple ERC1155 implementation" -├─ ERC6909 — "Simple EIP-6909 minimal multi-token implementation" -utils -├─ MerkleProofLib — "Library for verification of Merkle proofs" -├─ SignatureCheckerLib — "Library for verification of ECDSA and ERC1271 signatures" -├─ ECDSA — "Library for verification of ECDSA signatures" -├─ EIP712 — "Contract for EIP-712 typed structured data hashing and signing" -├─ ERC1967Factory — "Factory for deploying and managing ERC1967 proxy contracts" -├─ ERC1967FactoryConstants — "The address and bytecode of the canonical ERC1967Factory" -├─ JSONParserLib — "Library for parsing JSONs" -├─ LibSort — "Library for efficient sorting of memory arrays" -├─ LibPRNG — "Library for generating pseudorandom numbers" -├─ Base64 — "Library for Base64 encoding and decoding" -├─ SSTORE2 — "Library for cheaper reads and writes to persistent storage" -├─ CREATE3 — "Deploy to deterministic addresses without an initcode factor" -├─ LibRLP — "Library for computing contract addresses from their deployer and nonce" -├─ LibBit — "Library for bit twiddling and boolean operations" -├─ LibZip — "Library for compressing and decompressing bytes" -├─ Clone — "Class with helper read functions for clone with immutable args" -├─ LibClone — "Minimal proxy library" -├─ UUPSUpgradeable — "UUPS proxy mixin" -├─ LibString — "Library for converting numbers into strings and other string operations" -├─ LibBitmap — "Library for storage of packed booleans" -├─ LibMap — "Library for storage of packed unsigned integers" -├─ MinHeapLib — "Library for managing a min-heap in storage" -├─ RedBlackTreeLib — "Library for managing a red-black-tree in storage" -├─ Multicallable — "Contract that enables a single call to call multiple methods on itself" -├─ GasBurnerLib — "Library for burning gas without reverting" -├─ SafeTransferLib — "Safe ERC20/ETH transfer lib that handles missing return values" -├─ DynamicBufferLib — "Library for buffers with automatic capacity resizing" -├─ MetadataReaderLib — "Library for reading contract metadata robustly" -├─ FixedPointMathLib — "Arithmetic library with operations for fixed-point numbers" -├─ SafeCastLib — "Library for integer casting that reverts on overflow" -└─ DateTimeLib — "Library for date time operations" -``` - -## Directories - -```ml -src — "Solidity smart contracts" -test — "Foundry Forge tests" -js — "Accompanying JavaScript helper library" -ext — "Extra tests" -audits — "Audit reports" -``` - -## Contributing - -This repository serves as a laboratory for cutting edge snippets that may be merged into [Solmate](https://github.com/transmissions11/solmate). - -Feel free to make a pull request. - -Do refer to the [contribution guidelines](https://github.com/Vectorized/solady/issues/19) for more details. - -## Safety - -This is **experimental software** and is provided on an "as is" and "as available" basis. - -We **do not give any warranties** and **will not be liable for any loss** incurred through any use of this codebase. - -While Solady has been heavily tested, there may be parts that may exhibit unexpected emergent behavior when used with other code, or may break in future Solidity versions. - -Please always include your own thorough tests when using Solady to make sure it works correctly with your code. - -## Upgradability - -Most contracts in Solady are compatible with both upgradeable and non-upgradeable (i.e. regular) contracts. - -Please call any required internal initialization methods accordingly. - -## EVM Compatibility - -Some parts of Solady may not be compatible with chains with partial EVM equivalence. - -Please always check and test for compatibility accordingly. - -## Acknowledgements - -This repository is inspired by or directly modified from many sources, primarily: - -- [Solmate](https://github.com/transmissions11/solmate) -- [OpenZeppelin](https://github.com/OpenZeppelin/openzeppelin-contracts) -- [ERC721A](https://github.com/chiru-labs/ERC721A) -- [Zolidity](https://github.com/z0r0z/zolidity) -- [🐍 Snekmate](https://github.com/pcaversaccio/snekmate) -- [Femplate](https://github.com/abigger87/femplate) - -[npm-shield]: https://img.shields.io/npm/v/solady.svg -[npm-url]: https://www.npmjs.com/package/solady - -[ci-shield]: https://img.shields.io/github/actions/workflow/status/vectorized/solady/ci.yml?branch=main&label=build -[ci-url]: https://github.com/vectorized/solady/actions/workflows/ci.yml - -[solidity-shield]: https://img.shields.io/badge/solidity-%3E=0.8.4%20%3C=0.8.22-aa6746 -[solidity-ci-url]: https://github.com/Vectorized/solady/actions/workflows/ci-all-via-ir.yml diff --git a/lib/solady/audits/ackee-blockchain-solady-report.pdf b/lib/solady/audits/ackee-blockchain-solady-report.pdf deleted file mode 100644 index 04ac76c..0000000 Binary files a/lib/solady/audits/ackee-blockchain-solady-report.pdf and /dev/null differ diff --git a/lib/solady/audits/cantina-solady-report.pdf b/lib/solady/audits/cantina-solady-report.pdf deleted file mode 100644 index 3854df0..0000000 Binary files a/lib/solady/audits/cantina-solady-report.pdf and /dev/null differ diff --git a/lib/solady/audits/shung-solady-erc721-audit.pdf b/lib/solady/audits/shung-solady-erc721-audit.pdf deleted file mode 100644 index 471a9ff..0000000 Binary files a/lib/solady/audits/shung-solady-erc721-audit.pdf and /dev/null differ diff --git a/lib/solady/ext/woke/EIP712Mock.sol b/lib/solady/ext/woke/EIP712Mock.sol deleted file mode 100644 index b0a7916..0000000 --- a/lib/solady/ext/woke/EIP712Mock.sol +++ /dev/null @@ -1,17 +0,0 @@ -// SPDX-License-Identifier: MIT - -import "src/utils/EIP712.sol"; - -contract EIP712Mock is EIP712 { - string constant public NAME = "Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Donec ipsum massa, ullamcorper in, auctor et, scelerisque sed, est. Maecenas libero. Phasellus et lorem id felis nonummy placerat. Vestibulum erat nulla, ullamcorper nec, rutrum non, nonummy ac, erat. Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas. Etiam commodo dui eget wisi. Duis viverra diam non justo. Proin pede metus, vulputate nec, fermentum fringilla, vehicula vitae, justo. Mauris elementum mauris vitae tortor. Ut tempus purus at lorem. Maecenas sollicitudin. Vivamus ac leo pretium faucibus. Vivamus luctus egestas leo. Proin in tellus sit amet nibh dignissim sagittis. Aliquam erat volutpat. Cras elementum. Curabitur ligula sapien, pulvinar a vestibulum quis, facilisis vel sapien. Fusce wisi. Suspendisse sagittis ultrices augue. Aenean id metus id velit ullamcorper pulvinar. Cras pede libero, dapibus nec, pretium sit amet, tempor quis. Cras elementum. Nunc auctor. Quis autem vel eum iure reprehenderit qui in ea voluptate velit esse quam nihil molestiae consequatur, vel illum qui dolorem eum fugiat quo voluptas nulla pariatur? Nulla est. Aliquam ornare wisi eu metus. Et harum quidem rerum facilis est et expedita distinctio. Mauris metus. Phasellus faucibus molestie nisl. Mauris tincidunt sem sed arcu. Aenean placerat. Etiam neque. Nullam feugiat, turpis at pulvinar vulputate, erat libero tristique tellus, nec bibendum odio risus sit amet ante. Nullam dapibus fermentum ipsum. Mauris tincidunt sem sed arcu. Nam quis nulla. Nullam feugiat, turpis at pulvinar vulputate, erat libero tristique tellus, nec bibendum odio risus sit amet ante. Pellentesque sapien. Class aptent taciti sociosqu ad litora torquent per conubia nostra, per inceptos hymenaeos. Vestibulum erat nulla, ullamcorper nec, rutrum non, nonummy ac, erat. Maecenas aliquet accumsan leo. Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas. Class aptent taciti sociosqu ad litora torquent per conubia nostra, per inceptos hymenaeos. Nullam rhoncus aliquam metus. Praesent vitae arcu tempor neque lacinia pretium. Nulla pulvinar eleifend sem. Etiam bibendum elit eget erat. Aenean placerat. Mauris metus. Aenean fermentum risus id tortor. Suspendisse nisl. Fusce wisi. Fusce nibh. Sed vel lectus. Donec odio tempus molestie, porttitor ut, iaculis quis, sem. Curabitur ligula sapien, pulvinar a vestibulum quis, facilisis vel sapien. Mauris elementum mauris vitae tortor. Duis sapien nunc, commodo et, interdum suscipit, sollicitudin et, dolor. Quisque tincidunt scelerisque libero. Vivamus luctus egestas leo. Etiam dui sem, fermentum vitae, sagittis id, malesuada in, quam. Duis risus. Ut enim ad minima veniam, quis nostrum exercitationem ullam corporis suscipit laboriosam, nisi ut aliquid ex ea commodi consequatur? Suspendisse nisl. Cras elementum. Phasellus rhoncus. Nam libero tempore, cum soluta nobis est eligendi optio cumque nihil impedit quo minus id quod maxime placeat facere possimus, omnis voluptas assumenda est, omnis dolor repellendus. In enim a arcu imperdiet malesuada. In dapibus augue non sapien. "; - string constant public VERSION = "1.0.0"; - - function _domainNameAndVersion() internal pure override returns (string memory, string memory) { - return (NAME, VERSION); - } - - function hashTypedData(bytes32 structHash) external view returns(bytes32) { - return _hashTypedData(structHash); - } - -} \ No newline at end of file diff --git a/lib/solady/ext/woke/ERC1155Mock.sol b/lib/solady/ext/woke/ERC1155Mock.sol deleted file mode 100644 index bcc75ee..0000000 --- a/lib/solady/ext/woke/ERC1155Mock.sol +++ /dev/null @@ -1,149 +0,0 @@ -// SPDX-License-Identifier: MIT -import "src/tokens/ERC1155.sol"; - -contract ERC1155Mock is ERC1155 { - event BeforeTokenTransfer(address from, address to, uint256[] ids, uint256[] amounts, bytes data); - - event AfterTokenTransfer(address from, address to, uint256[] ids, uint256[] amounts, bytes data); - - bool immutable private _enableHooks; - - constructor(bool enableHooks_) { - _enableHooks = enableHooks_; - } - - function _useBeforeTokenTransfer() internal view override returns (bool) { - return _enableHooks; - } - - function _useAfterTokenTransfer() internal view override returns (bool) { - return _enableHooks; - } - - function _beforeTokenTransfer( - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) internal override { - emit BeforeTokenTransfer(from, to, ids, amounts, data); - } - - function _afterTokenTransfer( - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) internal override { - emit AfterTokenTransfer(from, to, ids, amounts, data); - } - - function uri(uint256 id) public view override returns (string memory) {} - - function mint(address to, uint256 id, uint256 amount, bytes memory data) external { - _mint(to, id, amount, data); - } - - function batchMint( - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) external { - _batchMint(to, ids, amounts, data); - } - - function burnUnchecked(address by, address from, uint256 id, uint256 amount) external { - _burn(by, from, id, amount); - } - - function burn(address from, uint256 id, uint256 amount) external { - _burn(msg.sender, from, id, amount); - } - - function batchBurnUnchecked(address by, address from, uint256[] memory ids, uint256[] memory amounts) external { - _batchBurn(by, from, ids, amounts); - } - - function batchBurn(address from, uint256[] memory ids, uint256[] memory amounts) external { - _batchBurn(msg.sender, from, ids, amounts); - } - - function setApprovalForAllUnchecked(address by, address operator, bool approved) external { - _setApprovalForAll(by, operator, approved); - } - - function safeTransferUnchecked( - address by, - address from, - address to, - uint256 id, - uint256 amount, - bytes memory data - ) external { - _safeTransfer(by, from, to, id, amount, data); - } - - function safeBatchTransferUnchecked( - address by, - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) external { - _safeBatchTransfer(by, from, to, ids, amounts, data); - } -} - -contract ERC1155ReceiverMock { - function onERC1155Received( - address, - address, - uint256, - uint256, - bytes calldata data - ) external pure returns(bytes4) { - require(keccak256(data) == keccak256(hex"00112233"), "ERC1155ReceiverMock: invalid payload received"); - return this.onERC1155Received.selector; - } - - function onERC1155BatchReceived( - address, - address, - uint256[] calldata, - uint256[] calldata, - bytes calldata data - ) external pure returns(bytes4) { - require(keccak256(data) == keccak256(hex"00112233"), "ERC1155ReceiverMock: invalid payload received"); - return this.onERC1155BatchReceived.selector; - } -} - -contract ERC1155ReentrancyAttacker { - function onERC1155Received( - address, - address, - uint256, - uint256, - bytes calldata data - ) external returns(bytes4) { - if (data.length == 0) - ERC1155Mock(msg.sender).mint(address(this), 1024, 1, hex"00112233"); - return this.onERC1155Received.selector; - } - - function onERC1155BatchReceived( - address, - address, - uint256[] calldata, - uint256[] calldata, - bytes calldata data - ) external returns(bytes4) { - if (data.length == 0) - ERC1155Mock(msg.sender).mint(address(this), 1024, 1, hex"00112233"); - return this.onERC1155BatchReceived.selector; - } -} \ No newline at end of file diff --git a/lib/solady/ext/woke/ERC20Mock.sol b/lib/solady/ext/woke/ERC20Mock.sol deleted file mode 100644 index a4a2fae..0000000 --- a/lib/solady/ext/woke/ERC20Mock.sol +++ /dev/null @@ -1,121 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/* -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢠⣾⣿⣷⣶⣤⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⠿⣿⣿⣿⣿⣿⣿⣶⣤⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣀⣠⣤⣴⣶⣾⣿⣷⣆⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠉⠙⠻⢿⣿⣿⣿⣿⣷⣦⣄⡀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣠⣶⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⠏⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⠻⢿⣿⣿⣿⣿⣦⡀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣠⣾⣿⣿⣿⣿⣿⡿⠿⠟⠛⠉⠉⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⠻⣿⣿⣿⣆⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣴⣿⣿⣿⡿⠟⠋⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠙⣿⣿⣧⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⣠⣿⣿⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⢻⣿⣇⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⣠⣾⣿⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣀⣴⣶⣿⣿⣿⣿⣿⣿⣿⣶⣤⡀⠀⠙⠋⠀⠀⠀ -⠀⠀⠀⠀⠀⣠⣾⣿⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣤⣾⠟⢋⣥⣤⠀⣶⣶⣶⣦⣤⣌⣉⠛⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⣴⣿⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠋⢁⣴⣿⣿⡿⠀⣿⣿⣿⣿⣿⣿⣿⣷⡄⠀⠀⠀⠀⠀ -⠀⠀⠀⣼⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣠⣤⣤⣶⣶⣾⣿⣿⣿⣿⣷⣶⡆⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣠⣿⣿⣿⠁⠀⠀⢹⣿⣿⣿⣿⣿⣿⢻⣿⡄⠀⠀⠀⠀ -⠀⠀⠀⠛⠋⠀⠀⠀⠀⠀⠀⠀⢀⣤⣾⣿⠿⠛⣛⣉⣉⣀⣀⡀⠀⠀⠀⠀⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣸⣿⣿⣿⣿⠀⠀⠀⠀⣿⣿⣿⣿⣿⣿⢸⣿⣿⡄⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢠⣾⡿⢋⣩⣶⣾⣿⣿⣿⣿⣿⣿⣿⣿⣶⣦⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢠⣿⣿⣿⣿⣿⣦⣀⣀⣴⣿⣿⣿⣿⣿⡿⢸⣿⢿⣷⡀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⣡⣄⠀⠋⠁⠀⠈⠹⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⣄⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣸⣿⡟⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⢸⡿⠀⠛⠃⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⢀⣴⣿⣿⣿⣧⡀⠀⠀⠀⠀⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣦⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠛⠛⠃⢹⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠁⠈⠁⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⢀⣴⣿⣿⣿⢿⣿⣿⣿⣷⣦⣤⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣇⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣶⣶⠀⠈⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡟⠀⣿⠇⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⢠⣿⣿⣿⠟⠉⠀⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢿⣿⠀⠀⢹⣿⣿⣿⣿⣿⣿⣿⣿⣿⠁⢸⣿⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⣼⣿⡟⠁⣠⣦⠀⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠉⣿⣿⡇⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢸⣿⡆⠀⠀⢻⣿⣿⣿⣿⣿⣿⣿⠏⠀⣸⡏⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⣿⡏⠀⠀⣿⣿⡀⠀⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠀⢹⣿⣧⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⣿⣇⠀⠀⠀⠙⢿⣿⣿⡿⠟⠁⠀⣸⡿⠁⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⢸⣿⠁⠀⠀⢸⣿⣇⠀⠀⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠁⠀⢀⣀⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⢿⣦⡀⠀⠀⠀⠈⠉⠀⠀⠀⣼⡿⠁⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠈⠁⠀⠀⠀⠀⢿⣿⡄⠀⠀⠈⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠁⠀⠀⣼⣿⡇⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠙⢿⣷⣦⣄⣀⠀⠀⢀⡈⠙⠁⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⢻⣿⣆⠀⠀⠀⠉⠛⠿⢿⣿⣿⠿⠛⠁⠀⠀⠀⣠⣿⣿⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⠿⣿⣿⣷⣿⣯⣤⣶⠄⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠹⣿⣷⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣴⣿⣿⠃⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠉⠉⠙⠛⠋⠁⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⢿⣷⣤⣀⠀⠀⠀⠀⠀⠀⠀⠺⣿⣿⡿⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠛⢻⣿⣶⣤⣤⣤⣶⣷⣤⠈⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⢿⣿⣿⣿⣿⡿⠿⠛⠋⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠉⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⡄⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⠶⢤⣄⣀⣀⣤⠶⠋⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀*/ - -import "src/tokens/ERC20.sol"; -import "src/utils/SafeTransferLib.sol"; - -interface IERC20 { - event Transfer(address indexed from, address indexed to, uint256 value); - event Approval(address indexed owner, address indexed spender, uint256 value); - function totalSupply() external view returns (uint256); - function balanceOf(address account) external view returns (uint256); - function transfer(address to, uint256 amount) external returns (bool); - function allowance(address owner, address spender) external view returns (uint256); - function approve(address spender, uint256 amount) external returns (bool); - function transferFrom(address from, address to, uint256 amount) external returns (bool); -} - -contract ERC20Mock is ERC20 { - string private $name; - string private $symbol; - uint8 private $decimals; - - constructor(string memory name_, string memory symbol_, uint8 decimals_) { - $name = name_; - $symbol = symbol_; - $decimals = decimals_; - } - - function mint(address account, uint256 amount) public { - _mint(account, amount); - } - - function burn(address account, uint256 amount) public { - _burn(account, amount); - } - - function name() public view virtual override returns (string memory) { - return $name; - } - - function symbol() public view virtual override returns (string memory) { - return $symbol; - } - - function decimals() public view virtual override returns (uint8) { - return $decimals; - } - - function safeTransferETH(address to, uint256 value) public { - SafeTransferLib.safeTransferETH(to, value); - } - - function forceSafeTransferETH(address to, uint256 value) public { - SafeTransferLib.forceSafeTransferETH(to, value); - } - - function forceSafeTransferETHGas(address to, uint256 value, uint256 gas) public { - SafeTransferLib.forceSafeTransferETH(to, value, gas); - } - - function trySafeTransferETH(address to, uint256 value, uint256 gasStipend) public returns (bool) { - return SafeTransferLib.trySafeTransferETH(to, value, gasStipend); - } - - function safeTransferFrom(address token, address from, address to, uint256 value) public { - SafeTransferLib.safeTransferFrom(token, from, to, value); - } - - function safeTransferAllFrom(address token, address from, address to) public { - SafeTransferLib.safeTransferAllFrom(token, from, to); - } - - function safeTransfer(address token, address to, uint256 value) public { - SafeTransferLib.safeTransfer(token, to, value); - } - - function safeTransferAll(address token, address to) public { - SafeTransferLib.safeTransferAll(token, to); - } - - function safeApprove(address token, address spender, uint256 value) public { - SafeTransferLib.safeApprove(token, spender, value); - } - - function balanceOfoor(address token, address account) public view returns (uint256) { - return SafeTransferLib.balanceOf(token, account); - } -} diff --git a/lib/solady/ext/woke/ERC721Mock.sol b/lib/solady/ext/woke/ERC721Mock.sol deleted file mode 100644 index 88aa7f1..0000000 --- a/lib/solady/ext/woke/ERC721Mock.sol +++ /dev/null @@ -1,86 +0,0 @@ -// SPDX-License-Identifier: MIT -import "src/tokens/ERC721.sol"; -contract ERC721Mock is ERC721 { - - event BeforeTokenTransfer(address from, address to, uint256 id); - - event AfterTokenTransfer(address from, address to, uint256 id); - - uint256 private constant _ERC721_MASTER_SLOT_SEED = 0x7d8825530a5a2e7a << 192; - /// @dev Returns the token collection name. - function name() public view override returns (string memory) { - return "Mock ERC721"; - } - /// @dev Returns the token collection symbol. - function symbol() public view override returns (string memory) { - return "MERC721"; - } - /// @dev Returns the Uniform Resource Identifier (URI) for token `id`. - function tokenURI(uint256 id) public view override returns (string memory) { - return "aaa"; - } - function getAux(address owner) public view returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - mstore(0x00, owner) - result := shr(32, sload(keccak256(0x0c, 0x1c))) - } - } - function _beforeTokenTransfer( - address from, - address to, - uint256 id - ) internal override { - emit BeforeTokenTransfer(from, to, id); - } - - function _afterTokenTransfer( - address from, - address to, - uint256 id - ) internal override { - emit AfterTokenTransfer(from, to, id); - } - - function mint(address to, uint256 id) public { - _mint(to, id); - } - - function burnZero(uint256 id) public { - _burn(id); - } - - function burn(uint256 id) public { - _burn(msg.sender, id); - } - - function transfer(address from, address to, uint256 id) public { - _transfer(msg.sender, from, to, id); - } - - - function balanceOf(address owner) public view virtual override returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - // Revert if the `owner` is the zero address. - if iszero(owner) { - mstore(0x00, 0x8f4eb604) // `BalanceQueryForZeroAddress()`. - revert(0x1c, 0x04) - } - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - mstore(0x00, owner) - result := and(sload(keccak256(0x0c, 0x1c)), _MAX_ACCOUNT_BALANCE) - } - } - function ownerOf(uint256 id) public view virtual override returns (address result) { - result = _ownerOf(id); - /// @solidity memory-safe-assembly - assembly { - if iszero(result) { - mstore(0x00, 0xceea21b6) // `TokenDoesNotExist()`. - revert(0x1c, 0x04) - } - } - } -} \ No newline at end of file diff --git a/lib/solady/ext/woke/MerkleProofMock.sol b/lib/solady/ext/woke/MerkleProofMock.sol deleted file mode 100644 index 57c28a0..0000000 --- a/lib/solady/ext/woke/MerkleProofMock.sol +++ /dev/null @@ -1,21 +0,0 @@ -// SPDX-License-Identifier: MIT - -import "src/utils/MerkleProofLib.sol"; - -contract MerkleProofMock { - function verify(bytes32[] memory proof, bytes32 root, bytes32 leaf) external pure returns (bool) { - return MerkleProofLib.verify(proof, root, leaf); - } - - function verifyCalldata(bytes32[] calldata proof, bytes32 root, bytes32 leaf) external pure returns (bool) { - return MerkleProofLib.verify(proof, root, leaf); - } - - function verifyMultiProof(bytes32[] memory proof, bytes32 root, bytes32[] memory leaves, bool[] memory flags) external pure returns (bool) { - return MerkleProofLib.verifyMultiProof(proof, root, leaves, flags); - } - - function verifyMultiProofCalldata(bytes32[] calldata proof, bytes32 root, bytes32[] calldata leaves, bool[] calldata flags) external pure returns (bool) { - return MerkleProofLib.verifyMultiProof(proof, root, leaves, flags); - } -} \ No newline at end of file diff --git a/lib/solady/ext/woke/NoETHMock.sol b/lib/solady/ext/woke/NoETHMock.sol deleted file mode 100644 index 7453f38..0000000 --- a/lib/solady/ext/woke/NoETHMock.sol +++ /dev/null @@ -1,44 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/* -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢠⣾⣿⣷⣶⣤⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⠿⣿⣿⣿⣿⣿⣿⣶⣤⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣀⣠⣤⣴⣶⣾⣿⣷⣆⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠉⠙⠻⢿⣿⣿⣿⣿⣷⣦⣄⡀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣠⣶⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⠏⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⠻⢿⣿⣿⣿⣿⣦⡀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣠⣾⣿⣿⣿⣿⣿⡿⠿⠟⠛⠉⠉⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⠻⣿⣿⣿⣆⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣴⣿⣿⣿⡿⠟⠋⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠙⣿⣿⣧⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⣠⣿⣿⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⢻⣿⣇⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⣠⣾⣿⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣀⣴⣶⣿⣿⣿⣿⣿⣿⣿⣶⣤⡀⠀⠙⠋⠀⠀⠀ -⠀⠀⠀⠀⠀⣠⣾⣿⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣤⣾⠟⢋⣥⣤⠀⣶⣶⣶⣦⣤⣌⣉⠛⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⣴⣿⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠋⢁⣴⣿⣿⡿⠀⣿⣿⣿⣿⣿⣿⣿⣷⡄⠀⠀⠀⠀⠀ -⠀⠀⠀⣼⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣠⣤⣤⣶⣶⣾⣿⣿⣿⣿⣷⣶⡆⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣠⣿⣿⣿⠁⠀⠀⢹⣿⣿⣿⣿⣿⣿⢻⣿⡄⠀⠀⠀⠀ -⠀⠀⠀⠛⠋⠀⠀⠀⠀⠀⠀⠀⢀⣤⣾⣿⠿⠛⣛⣉⣉⣀⣀⡀⠀⠀⠀⠀⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣸⣿⣿⣿⣿⠀⠀⠀⠀⣿⣿⣿⣿⣿⣿⢸⣿⣿⡄⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢠⣾⡿⢋⣩⣶⣾⣿⣿⣿⣿⣿⣿⣿⣿⣶⣦⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢠⣿⣿⣿⣿⣿⣦⣀⣀⣴⣿⣿⣿⣿⣿⡿⢸⣿⢿⣷⡀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⣡⣄⠀⠋⠁⠀⠈⠹⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⣄⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣸⣿⡟⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⢸⡿⠀⠛⠃⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⢀⣴⣿⣿⣿⣧⡀⠀⠀⠀⠀⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣦⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠛⠛⠃⢹⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠁⠈⠁⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⢀⣴⣿⣿⣿⢿⣿⣿⣿⣷⣦⣤⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣇⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣶⣶⠀⠈⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡟⠀⣿⠇⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⢠⣿⣿⣿⠟⠉⠀⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢿⣿⠀⠀⢹⣿⣿⣿⣿⣿⣿⣿⣿⣿⠁⢸⣿⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⣼⣿⡟⠁⣠⣦⠀⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠉⣿⣿⡇⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢸⣿⡆⠀⠀⢻⣿⣿⣿⣿⣿⣿⣿⠏⠀⣸⡏⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⣿⡏⠀⠀⣿⣿⡀⠀⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠀⢹⣿⣧⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⣿⣇⠀⠀⠀⠙⢿⣿⣿⡿⠟⠁⠀⣸⡿⠁⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⢸⣿⠁⠀⠀⢸⣿⣇⠀⠀⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠁⠀⢀⣀⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⢿⣦⡀⠀⠀⠀⠈⠉⠀⠀⠀⣼⡿⠁⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠈⠁⠀⠀⠀⠀⢿⣿⡄⠀⠀⠈⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠁⠀⠀⣼⣿⡇⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠙⢿⣷⣦⣄⣀⠀⠀⢀⡈⠙⠁⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⢻⣿⣆⠀⠀⠀⠉⠛⠿⢿⣿⣿⠿⠛⠁⠀⠀⠀⣠⣿⣿⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⠿⣿⣿⣷⣿⣯⣤⣶⠄⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠹⣿⣷⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣴⣿⣿⠃⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠉⠉⠙⠛⠋⠁⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⢿⣷⣤⣀⠀⠀⠀⠀⠀⠀⠀⠺⣿⣿⡿⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠛⢻⣿⣶⣤⣤⣤⣶⣷⣤⠈⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⢿⣿⣿⣿⣿⡿⠿⠛⠋⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠉⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⡄⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⠶⢤⣄⣀⣀⣤⠶⠋⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ -⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀*/ - -contract NoETHMock { - receive() external payable { - revert("No ETH pls"); - } - fallback() external payable { - revert("No ETH pls"); - } -} diff --git a/lib/solady/ext/woke/SignatureCheckerMock.sol b/lib/solady/ext/woke/SignatureCheckerMock.sol deleted file mode 100644 index c28adc3..0000000 --- a/lib/solady/ext/woke/SignatureCheckerMock.sol +++ /dev/null @@ -1,58 +0,0 @@ -// SPDX-License-Identifier: MIT -import "src/utils/SignatureCheckerLib.sol"; - -contract SignatureCheckerMock { - function isValidSignatureNow(address signer, bytes32 hash, bytes memory signature) external view returns (bool) { - return SignatureCheckerLib.isValidSignatureNow(signer, hash, signature); - } - - function isValidSignatureNowCalldata(address signer, bytes32 hash, bytes calldata signature) external view returns (bool) { - return SignatureCheckerLib.isValidSignatureNowCalldata(signer, hash, signature); - } - - function isValidSignatureNow(address signer, bytes32 hash, bytes32 r, bytes32 vs) external view returns(bool) { - return SignatureCheckerLib.isValidSignatureNow(signer, hash, r, vs); - } - - function isValidSignatureNow(address signer, bytes32 hash, uint8 v, bytes32 r, bytes32 s) external view returns(bool) { - return SignatureCheckerLib.isValidSignatureNow(signer, hash, v, r, s); - } - - function isValidERC1271SignatureNow(address signer, bytes32 hash, bytes memory signature) external view returns (bool) { - return SignatureCheckerLib.isValidERC1271SignatureNow(signer, hash, signature); - } - - function isValidERC1271SignatureNowCalldata(address signer, bytes32 hash, bytes calldata signature) external view returns (bool) { - return SignatureCheckerLib.isValidERC1271SignatureNowCalldata(signer, hash, signature); - } - - function isValidERC1271SignatureNow(address signer, bytes32 hash, bytes32 r, bytes32 vs) external view returns(bool) { - return SignatureCheckerLib.isValidERC1271SignatureNow(signer, hash, r, vs); - } - - function isValidERC1271SignatureNow(address signer, bytes32 hash, uint8 v, bytes32 r, bytes32 s) external view returns(bool) { - return SignatureCheckerLib.isValidERC1271SignatureNow(signer, hash, v, r, s); - } -} - -contract ERC1217SignatureChecker { - bytes4 constant internal MAGICVALUE = 0x1626ba7e; - - function isValidSignature(bytes32 _hash, bytes memory _signature) public view returns (bytes4) { - uint8 v; - bytes32 r; - bytes32 s; - - assembly { - r := mload(add(_signature, 0x20)) - s := mload(add(_signature, 0x40)) - v := byte(0, mload(add(_signature, 0x60))) - } - - if (tx.origin == ecrecover(_hash, v, r, s)) { - return MAGICVALUE; - } else { - return 0x0; - } - } -} \ No newline at end of file diff --git a/lib/solady/ext/woke/__init__.py b/lib/solady/ext/woke/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/lib/solady/ext/woke/test_eip712.py b/lib/solady/ext/woke/test_eip712.py deleted file mode 100644 index d56b431..0000000 --- a/lib/solady/ext/woke/test_eip712.py +++ /dev/null @@ -1,84 +0,0 @@ -from dataclasses import dataclass, field - -from eth_account._utils.structured_data.hashing import hash_message -from woke.testing import * -from pytypes.src.utils.ERC1967Factory import ERC1967Factory -from pytypes.tests.EIP712Mock import EIP712Mock - - -@dataclass -class Person: - name: str - wallet: Address - - -@dataclass -class Mail: - from_: Person = field(metadata={"original_name": "from"}) - to: Person - contents: str - - -mail = Mail( - from_=Person("Cow", Address("0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826")), - to=Person("Bob", Address("0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB")), - contents="Hello, Bob!", -) - - -@default_chain.connect() -def test_eip712(): - default_chain.set_default_accounts(default_chain.accounts[0]) - signer = Account.new() - - eip712 = EIP712Mock.deploy() - assert eip712.eip712Domain() == ( - 0b01111.to_bytes(1, "big"), - eip712.NAME(), - eip712.VERSION(), - default_chain.chain_id, - eip712.address, - b"\x00" * 32, - [], - ) - - mail_hash = hash_message(signer._prepare_eip712_dict(mail, Eip712Domain(), False)) - - sign1 = signer.sign_hash(eip712.hashTypedData(mail_hash)) - sign2 = signer.sign_structured(mail, Eip712Domain( - name=eip712.NAME(), - version=eip712.VERSION(), - chainId=default_chain.chain_id, - verifyingContract=eip712.address, - )) - assert sign1 == sign2 - - -@default_chain.connect() -def test_eip712_proxy(): - default_chain.set_default_accounts(default_chain.accounts[0]) - signer = Account.new() - - proxy_factory = ERC1967Factory.deploy() - impl = EIP712Mock.deploy() - eip712 = EIP712Mock(proxy_factory.deploy_(impl, signer).return_value) - assert eip712.eip712Domain() == ( - 0b01111.to_bytes(1, "big"), - eip712.NAME(), - eip712.VERSION(), - default_chain.chain_id, - eip712.address, - b"\x00" * 32, - [], - ) - - mail_hash = hash_message(signer._prepare_eip712_dict(mail, Eip712Domain(), False)) - - sign1 = signer.sign_hash(eip712.hashTypedData(mail_hash)) - sign2 = signer.sign_structured(mail, Eip712Domain( - name=eip712.NAME(), - version=eip712.VERSION(), - chainId=default_chain.chain_id, - verifyingContract=eip712.address, - )) - assert sign1 == sign2 diff --git a/lib/solady/ext/woke/test_eip712_fuzz.py b/lib/solady/ext/woke/test_eip712_fuzz.py deleted file mode 100644 index 85fb8ae..0000000 --- a/lib/solady/ext/woke/test_eip712_fuzz.py +++ /dev/null @@ -1,65 +0,0 @@ -from dataclasses import dataclass, field - -from eth_account._utils.structured_data.hashing import hash_message -from woke.testing import * -from woke.testing.fuzzing import * -from pytypes.src.utils.ERC1967Factory import ERC1967Factory -from pytypes.tests.EIP712Mock import EIP712Mock - - -@dataclass -class Person: - name: str - wallet: Address - - -@dataclass -class Mail: - from_: Person = field(metadata={"original_name": "from"}) - to: Person - contents: str - - -class Eip712FuzzTest(FuzzTest): - _proxy_factory: ERC1967Factory - _eip712: EIP712Mock - _eip712_proxy: EIP712Mock - _signer: Account - - def __init__(self): - self._proxy_factory = ERC1967Factory.deploy() - - def pre_sequence(self) -> None: - self._eip712 = EIP712Mock.deploy() - self._signer = Account.new() - self._eip712_proxy = EIP712Mock( - self._proxy_factory.deploy_(self._eip712, self._signer).return_value - ) - - @flow() - def sign_flow(self, mail: Mail) -> None: - mail_hash = hash_message(self._signer._prepare_eip712_dict(mail, Eip712Domain(), False)) - - sign1 = self._signer.sign_hash(self._eip712.hashTypedData(mail_hash)) - sign2 = self._signer.sign_structured(mail, Eip712Domain( - name=self._eip712.NAME(), - version=self._eip712.VERSION(), - chainId=default_chain.chain_id, - verifyingContract=self._eip712.address, - )) - assert sign1 == sign2 - - sign1 = self._signer.sign_hash(self._eip712_proxy.hashTypedData(mail_hash)) - sign2 = self._signer.sign_structured(mail, Eip712Domain( - name=self._eip712_proxy.NAME(), - version=self._eip712_proxy.VERSION(), - chainId=default_chain.chain_id, - verifyingContract=self._eip712_proxy.address, - )) - assert sign1 == sign2 - - -@default_chain.connect() -def test_eip712_fuzz(): - default_chain.set_default_accounts(default_chain.accounts[0]) - Eip712FuzzTest().run(10, 10) diff --git a/lib/solady/ext/woke/test_erc1155.py b/lib/solady/ext/woke/test_erc1155.py deleted file mode 100644 index 8a2ce7a..0000000 --- a/lib/solady/ext/woke/test_erc1155.py +++ /dev/null @@ -1,443 +0,0 @@ -from woke.testing import * -from pytypes.tests.ERC1155Mock import ERC1155Mock, ERC1155ReceiverMock - - -@default_chain.connect() -def test_erc1155_misc(): - a = default_chain.accounts[0] - - erc1155 = ERC1155Mock.deploy(False, from_=a) - - # ERC-165 - assert erc1155.supportsInterface(bytes.fromhex("01ffc9a7")) - # ERC-1155 - assert erc1155.supportsInterface(bytes.fromhex("d9b67a26")) - # ERC-1155 Metadata URI - assert erc1155.supportsInterface(bytes.fromhex("0e89341c")) - - assert not erc1155.supportsInterface(bytes.fromhex("deadbeef")) - - erc1155.mint(a, 0, 100, b"\x00\x11", from_=a) - erc1155.setApprovalForAll(a, False, from_=a) - assert not erc1155.isApprovedForAll(a, a) - erc1155.safeTransferFrom(a, a, 0, 100, b"\x00\x11", from_=a) - assert erc1155.balanceOf(a, 0) == 100 - - # check overflow when sending to self - erc1155.mint(a, 0, 2 ** 256 - 1 - 100, b"", from_=a) - erc1155.safeTransferFrom(a, a, 0, 2 ** 256 - 1, b"", from_=a) - assert erc1155.balanceOf(a, 0) == 2 ** 256 - 1 - - -@default_chain.connect() -def test_erc1155_events(): - assert keccak256("TransferSingle(address,address,address,uint256,uint256)".encode()) == bytes.fromhex("c3d58168c5ae7397731d063d5bbf3d657854427343f4c083240f7aacaa2d0f62") - assert keccak256("TransferBatch(address,address,address,uint256[],uint256[])".encode()) == bytes.fromhex("4a39dc06d4c0dbc64b70af90fd698a233a518aa5d07e595d983b8c0526c8f7fb") - assert keccak256("ApprovalForAll(address,address,bool)".encode()) == bytes.fromhex("17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31") - - a = default_chain.accounts[0] - b = default_chain.accounts[1] - - erc1155 = ERC1155Mock.deploy(True, from_=a) - - tx = erc1155.mint(a, 0, 100, b"\x00\x11", from_=a) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(Address.ZERO, a.address, [0], [100], bytearray(b"\x00\x11")), - ERC1155Mock.TransferSingle(a.address, Address.ZERO, a.address, 0, 100), - ERC1155Mock.AfterTokenTransfer(Address.ZERO, a.address, [0], [100], bytearray(b"\x00\x11")), - ] - - tx = erc1155.burn(a, 0, 100, from_=a) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(a.address, Address.ZERO, [0], [100], bytearray(b"")), - ERC1155Mock.TransferSingle(a.address, a.address, Address.ZERO, 0, 100), - ERC1155Mock.AfterTokenTransfer(a.address, Address.ZERO, [0], [100], bytearray(b"")), - ] - - tx = erc1155.batchMint(a, [0, 1, 2], [100, 200, 300], b"\x11\x22", from_=a) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(Address.ZERO, a.address, [0, 1, 2], [100, 200, 300], bytearray(b"\x11\x22")), - ERC1155Mock.TransferBatch(a.address, Address.ZERO, a.address, [0, 1, 2], [100, 200, 300]), - ERC1155Mock.AfterTokenTransfer(Address.ZERO, a.address, [0, 1, 2], [100, 200, 300], bytearray(b"\x11\x22")), - ] - - tx = erc1155.batchBurn(a, [0, 1, 2], [100, 200, 300], from_=a) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(a.address, Address.ZERO, [0, 1, 2], [100, 200, 300], bytearray(b"")), - ERC1155Mock.TransferBatch(a.address, a.address, Address.ZERO, [0, 1, 2], [100, 200, 300]), - ERC1155Mock.AfterTokenTransfer(a.address, Address.ZERO, [0, 1, 2], [100, 200, 300], bytearray(b"")), - ] - - erc1155.mint(a, 0, 100, b"", from_=a) - tx = erc1155.safeTransferFrom(a, b, 0, 100, b"", from_=a) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(a.address, b.address, [0], [100], bytearray(b"")), - ERC1155Mock.TransferSingle(a.address, a.address, b.address, 0, 100), - ERC1155Mock.AfterTokenTransfer(a.address, b.address, [0], [100], bytearray(b"")), - ] - - tx = erc1155.safeTransferFrom(b, b, 0, 100, b"\x33", from_=b) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(b.address, b.address, [0], [100], bytearray(b"\x33")), - ERC1155Mock.TransferSingle(b.address, b.address, b.address, 0, 100), - ERC1155Mock.AfterTokenTransfer(b.address, b.address, [0], [100], bytearray(b"\x33")), - ] - - erc1155.setApprovalForAll(a, True, from_=b) - tx = erc1155.safeTransferFrom(b, a, 0, 100, b"", from_=a) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(b.address, a.address, [0], [100], bytearray(b"")), - ERC1155Mock.TransferSingle(a.address, b.address, a.address, 0, 100), - ERC1155Mock.AfterTokenTransfer(b.address, a.address, [0], [100], bytearray(b"")), - ] - erc1155.setApprovalForAll(a, False, from_=b) - - erc1155.setApprovalForAll(b, True, from_=a) - tx = erc1155.safeBatchTransferFrom(a, b, [0, 0, 0], [35, 30, 35], b"", from_=b) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(a.address, b.address, [0, 0, 0], [35, 30, 35], bytearray(b"")), - ERC1155Mock.TransferBatch(b.address, a.address, b.address, [0, 0, 0], [35, 30, 35]), - ERC1155Mock.AfterTokenTransfer(a.address, b.address, [0, 0, 0], [35, 30, 35], bytearray(b"")), - ] - erc1155.setApprovalForAll(b, False, from_=a) - - erc1155.setApprovalForAll(a, True, from_=b) - tx = erc1155.burn(b, 0, 100, from_=a) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(b.address, Address.ZERO, [0], [100], bytearray(b"")), - ERC1155Mock.TransferSingle(a.address, b.address, Address.ZERO, 0, 100), - ERC1155Mock.AfterTokenTransfer(b.address, Address.ZERO, [0], [100], bytearray(b"")), - ] - - assert erc1155.isApprovedForAll(a, b) == False - tx = erc1155.setApprovalForAll(b, False, from_=a) - assert tx.events == [ERC1155Mock.ApprovalForAll(a.address, b.address, False)] - assert erc1155.isApprovedForAll(a, b) == False - - tx = erc1155.setApprovalForAll(b, True, from_=a) - assert tx.events == [ERC1155Mock.ApprovalForAll(a.address, b.address, True)] - assert erc1155.isApprovedForAll(a, b) == True - - tx = erc1155.setApprovalForAll(b, True, from_=a) - assert tx.events == [ERC1155Mock.ApprovalForAll(a.address, b.address, True)] - assert erc1155.isApprovedForAll(a, b) == True - - tx = erc1155.setApprovalForAll(b, False, from_=a) - assert tx.events == [ERC1155Mock.ApprovalForAll(a.address, b.address, False)] - assert erc1155.isApprovedForAll(a, b) == False - - c = default_chain.accounts[2] - - tx = erc1155.setApprovalForAllUnchecked(a, b, True, from_=c) - assert tx.events == [ERC1155Mock.ApprovalForAll(a.address, b.address, True)] - tx = erc1155.setApprovalForAllUnchecked(a, b, True, from_=c) - assert tx.events == [ERC1155Mock.ApprovalForAll(a.address, b.address, True)] - tx = erc1155.setApprovalForAllUnchecked(a, b, False, from_=c) - assert tx.events == [ERC1155Mock.ApprovalForAll(a.address, b.address, False)] - - erc1155.batchMint(a, [0, 1, 2], [100, 200, 300], b"", from_=a) - tx = erc1155.burnUnchecked(Address.ZERO, a, 2, 100, from_=c) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(a.address, Address.ZERO, [2], [100], bytearray(b"")), - ERC1155Mock.TransferSingle(c.address, a.address, Address.ZERO, 2, 100), - ERC1155Mock.AfterTokenTransfer(a.address, Address.ZERO, [2], [100], bytearray(b"")), - ] - - tx = erc1155.batchBurnUnchecked(Address.ZERO, a, [0, 1, 2], [100, 100, 100], from_=c) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(a.address, Address.ZERO, [0, 1, 2], [100, 100, 100], bytearray(b"")), - ERC1155Mock.TransferBatch(c.address, a.address, Address.ZERO, [0, 1, 2], [100, 100, 100]), - ERC1155Mock.AfterTokenTransfer(a.address, Address.ZERO, [0, 1, 2], [100, 100, 100], bytearray(b"")), - ] - - tx = erc1155.safeTransferUnchecked(Address.ZERO, a, b, 1, 50, b"\x11", from_=c) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(a.address, b.address, [1], [50], bytearray(b"\x11")), - ERC1155Mock.TransferSingle(c.address, a.address, b.address, 1, 50), - ERC1155Mock.AfterTokenTransfer(a.address, b.address, [1], [50], bytearray(b"\x11")), - ] - - tx = erc1155.safeBatchTransferUnchecked(Address.ZERO, a, b, [0, 1, 2], [0, 50, 0], b"\x22", from_=c) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(a.address, b.address, [0, 1, 2], [0, 50, 0], bytearray(b"\x22")), - ERC1155Mock.TransferBatch(c.address, a.address, b.address, [0, 1, 2], [0, 50, 0]), - ERC1155Mock.AfterTokenTransfer(a.address, b.address, [0, 1, 2], [0, 50, 0], bytearray(b"\x22")), - ] - - -@default_chain.connect() -def test_erc1155_mint_burn(): - a = default_chain.accounts[0] - b = default_chain.accounts[1] - - erc1155 = ERC1155Mock.deploy(False, from_=a) - - assert erc1155.balanceOfBatch([], []) == [] - - erc1155.mint(b, 0, 100, b"", from_=a) - assert erc1155.balanceOf(b, 0) == 100 - - erc1155.burn(b, 0, 50, from_=b) - assert erc1155.balanceOf(b, 0) == 50 - - # b is not owner nor approved - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.NotOwnerNorApproved.selector)): - erc1155.burn(b, 0, 50, from_=a) - - # insufficient balance - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.InsufficientBalance.selector)): - erc1155.burn(b, 0, 51, from_=b) - - erc1155.burn(b, 0, 50, from_=b) - assert erc1155.balanceOf(b, 0) == 0 - - # mint to zero address - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.TransferToZeroAddress.selector)): - erc1155.mint(Address.ZERO, 0, 100, b"", from_=a) - - # balance overflow - erc1155.mint(b, 0, 2 ** 256 - 1, b"", from_=a) - assert erc1155.balanceOf(b, 0) == 2 ** 256 - 1 - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.AccountBalanceOverflow.selector)): - erc1155.mint(b, 0, 1, b"", from_=a) - erc1155.burn(b, 0, 2 ** 256 - 1, from_=b) - - # ids and amounts length mismatch - with must_revert(ERC1155Mock.ArrayLengthsMismatch): - erc1155.batchMint(b, [0, 1], [100], b"", from_=a) - - # ids and amounts length mismatch - with must_revert(ERC1155Mock.ArrayLengthsMismatch): - erc1155.batchBurn(b, [0, 1], [100], from_=a) - - # mint to zero address - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.TransferToZeroAddress.selector)): - erc1155.batchMint(Address.ZERO, [0, 1], [100, 200], b"", from_=a) - - # balance overflow - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.AccountBalanceOverflow.selector)): - erc1155.batchMint(b, [0, 1, 0], [2 ** 256 - 1, 1, 1], b"", from_=a) - - # not owner nor approved - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.NotOwnerNorApproved.selector)): - erc1155.batchBurn(b, [0, 1, 0], [1, 1, 1], from_=a) - - # insufficient balance - erc1155.mint(b, 0, 100, b"", from_=a) - erc1155.mint(b, 1, 1, b"", from_=a) - assert erc1155.balanceOf(b, 0) == 100 - assert erc1155.balanceOf(b, 1) == 1 - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.InsufficientBalance.selector)): - erc1155.batchBurn(b, [0, 1, 0], [70, 1, 31], from_=b) - erc1155.burn(b, 0, 100, from_=b) - erc1155.burn(b, 1, 1, from_=b) - assert erc1155.balanceOf(b, 0) == 0 - assert erc1155.balanceOf(b, 1) == 0 - - -@default_chain.connect() -def test_erc1155_transfers(): - a = default_chain.accounts[0] - b = default_chain.accounts[1] - - erc1155 = ERC1155Mock.deploy(False, from_=a) - - erc1155.mint(a, 0, 100, b"", from_=a) - erc1155.mint(a, 1, 100, b"", from_=a) - assert erc1155.balanceOf(a, 0) == 100 - assert erc1155.balanceOf(a, 1) == 100 - - erc1155.safeTransferFrom(a, b, 0, 50, b"", from_=a) - assert erc1155.balanceOf(a, 0) == 50 - assert erc1155.balanceOf(a, 1) == 100 - assert erc1155.balanceOf(b, 0) == 50 - assert erc1155.balanceOfBatch([a, a, b, a], [0, 1, 0, 2]) == [50, 100, 50, 0] - - erc1155.safeBatchTransferFrom(a, b, [], [], b"", from_=a) - assert erc1155.balanceOfBatch([a, a, b, a], [0, 1, 0, 2]) == [50, 100, 50, 0] - - # owners and ids length mismatch - with must_revert(ERC1155Mock.ArrayLengthsMismatch): - assert erc1155.balanceOfBatch([a, a, b], [0, 1, 0, 2]) == [50, 100, 50, 0] - - # not owner nor approved - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.NotOwnerNorApproved.selector)): - erc1155.safeTransferFrom(a, b, 0, 50, b"", from_=b) - - # insufficient balance - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.InsufficientBalance.selector)): - erc1155.safeTransferFrom(a, b, 0, 51, b"", from_=a) - - # transfer to zero address - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.TransferToZeroAddress.selector)): - erc1155.safeTransferFrom(a, Address.ZERO, 0, 50, b"", from_=a) - - # transfer to self - erc1155.safeTransferFrom(a, a, 0, 50, b"", from_=a) - assert erc1155.balanceOf(a, 0) == 50 - - # balance overflow - erc1155.mint(a, 0, 2 ** 256 - 1 - 50, b"", from_=a) - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.AccountBalanceOverflow.selector)): - erc1155.safeTransferFrom(a, b, 0, 2 ** 256 - 1 - 49, b"", from_=a) - - # transfer to non-erc1155 receiver - with must_revert(ERC1155Mock.TransferToNonERC1155ReceiverImplementer): - erc1155.safeTransferFrom(a, erc1155, 0, 50, b"", from_=a) - with must_revert(ERC1155Mock.TransferToNonERC1155ReceiverImplementer): - erc1155.safeBatchTransferFrom(a, erc1155, [0], [50], b"", from_=a) - - # clear balances - c = default_chain.accounts[2] - assert erc1155.isApprovedForAll(a, c) is False - erc1155.batchBurnUnchecked(Address.ZERO, a, [0, 1], [2 ** 256 - 1, 100], from_=c) - erc1155.burnUnchecked(Address.ZERO, b, 0, 50, from_=c) - - erc1155.batchMint(a, [0, 1], [100, 100], b"", from_=a) - erc1155.safeBatchTransferFrom(a, b, [0, 1], [70, 30], b"", from_=a) - assert erc1155.balanceOfBatch([a, a, b, b], [0, 1, 0, 1]) == [30, 70, 70, 30] - - with must_revert(ERC1155Mock.ArrayLengthsMismatch()): - erc1155.safeBatchTransferFrom(a, b, [0, 1], [30], b"", from_=a) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.TransferToZeroAddress.selector)): - erc1155.safeBatchTransferFrom(a, Address.ZERO, [0, 1], [30, 30], b"", from_=a) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.NotOwnerNorApproved.selector)): - erc1155.safeBatchTransferFrom(a, b, [0, 1], [30, 30], b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.InsufficientBalance.selector)): - erc1155.safeBatchTransferFrom(a, b, [0, 1], [31, 30], b"", from_=a) - - erc1155.mint(a, 0, 2 ** 256 - 1 - 30, b"", from_=a) - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.AccountBalanceOverflow.selector)): - erc1155.safeBatchTransferFrom(a, b, [0, 1], [2 ** 256 - 1 - 29, 30], b"", from_=a) - - -@default_chain.connect() -def test_erc1155_contract_receiver(): - a = default_chain.accounts[0] - - erc1155 = ERC1155Mock.deploy(False, from_=a) - receiver = ERC1155ReceiverMock.deploy(from_=a) - - erc1155.mint(receiver, 0, 100, b"\x00\x11\x22\x33", from_=a) - assert erc1155.balanceOf(receiver, 0) == 100 - - with must_revert(Error("ERC1155ReceiverMock: invalid payload received")): - erc1155.mint(receiver, 0, 100, b"", from_=a) - - erc1155.mint(a, 1, 100, b"", from_=a) - erc1155.safeTransferFrom(a, receiver, 1, 50, b"\x00\x11\x22\x33", from_=a) - assert erc1155.balanceOfBatch([receiver, receiver], [0, 1]) == [100, 50] - - with must_revert(Error("ERC1155ReceiverMock: invalid payload received")): - erc1155.safeTransferFrom(a, receiver, 1, 50, b"\x00\x11\x22", from_=a) - - erc1155.safeBatchTransferFrom(a, receiver, [0, 1], [0, 50], b"\x00\x11\x22\x33", from_=a) - assert erc1155.balanceOfBatch([receiver, receiver], [0, 1]) == [100, 100] - - with must_revert(Error("ERC1155ReceiverMock: invalid payload received")): - erc1155.safeBatchTransferFrom(a, receiver, [0, 1], [0, 0], b"\x11\x22\x33", from_=a) - - erc1155.batchMint(receiver, [0, 1], [100, 100], b"\x00\x11\x22\x33", from_=a) - assert erc1155.balanceOfBatch([receiver, receiver], [0, 1]) == [200, 200] - - with must_revert(Error("ERC1155ReceiverMock: invalid payload received")): - erc1155.batchMint(receiver, [0, 1], [100, 100], b"", from_=a) - - -@default_chain.connect() -def test_erc1155_unchecked(): - a = default_chain.accounts[0] - b = default_chain.accounts[1] - c = default_chain.accounts[2] - default_chain.set_default_accounts(c) - - erc1155 = ERC1155Mock.deploy(False, from_=a) - - erc1155.mint(a, 0, 100, b"", from_=c) - assert erc1155.balanceOf(a, 0) == 100 - - erc1155.safeBatchTransferUnchecked(Address.ZERO, a, b, [0], [100], b"", from_=c) - assert erc1155.balanceOf(a, 0) == 0 - assert erc1155.balanceOf(b, 0) == 100 - - erc1155.setApprovalForAllUnchecked(b, a, True, from_=c) - erc1155.safeTransferUnchecked(a, b, a, 0, 100, b"", from_=c) - assert erc1155.balanceOf(a, 0) == 100 - assert erc1155.balanceOf(b, 0) == 0 - - erc1155.setApprovalForAllUnchecked(a, b, True, from_=c) - erc1155.burnUnchecked(b, a, 0, 100, from_=c) - assert erc1155.balanceOf(a, 0) == 0 - assert erc1155.balanceOf(b, 0) == 0 - - erc1155.batchMint(a, [0, 1], [100, 100], b"", from_=c) - erc1155.batchBurnUnchecked(b, a, [0, 1], [100, 100], from_=c) - assert erc1155.balanceOf(a, 0) == 0 - assert erc1155.balanceOf(a, 1) == 0 - - erc1155.setApprovalForAllUnchecked(a, b, False, from_=c) - erc1155.setApprovalForAllUnchecked(b, a, False, from_=c) - erc1155.mint(a, 0, 100, b"", from_=c) - assert erc1155.balanceOf(a, 0) == 100 - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.NotOwnerNorApproved.selector)): - erc1155.safeTransferUnchecked(b, a, b, 0, 100, b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.NotOwnerNorApproved.selector)): - erc1155.safeBatchTransferUnchecked(b, a, b, [0], [100], b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.TransferToZeroAddress.selector)): - erc1155.safeTransferUnchecked(Address.ZERO, a, Address.ZERO, 0, 100, b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.TransferToZeroAddress.selector)): - erc1155.safeTransferUnchecked(a, a, Address.ZERO, 0, 100, b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.TransferToZeroAddress.selector)): - erc1155.safeBatchTransferUnchecked(Address.ZERO, a, Address.ZERO, [0], [100], b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.TransferToZeroAddress.selector)): - erc1155.safeBatchTransferUnchecked(a, a, Address.ZERO, [0], [100], b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.InsufficientBalance.selector)): - erc1155.safeTransferUnchecked(Address.ZERO, a, b, 0, 101, b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.InsufficientBalance.selector)): - erc1155.safeTransferUnchecked(a, a, b, 0, 101, b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.InsufficientBalance.selector)): - erc1155.safeBatchTransferUnchecked(Address.ZERO, a, b, [0], [101], b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.InsufficientBalance.selector)): - erc1155.safeBatchTransferUnchecked(a, a, b, [0], [101], b"", from_=c) - - erc1155.mint(b, 0, 2 ** 256 - 10, b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.AccountBalanceOverflow.selector)): - erc1155.safeTransferUnchecked(Address.ZERO, a, b, 0, 100, b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.AccountBalanceOverflow.selector)): - erc1155.safeTransferUnchecked(a, a, b, 0, 100, b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.AccountBalanceOverflow.selector)): - erc1155.safeBatchTransferUnchecked(Address.ZERO, a, b, [0, 0], [9, 21], b"", from_=c) - - with must_revert(UnknownTransactionRevertedError(ERC1155Mock.AccountBalanceOverflow.selector)): - erc1155.safeBatchTransferUnchecked(a, a, b, [0, 0], [9, 21], b"", from_=c) - - with must_revert(ERC1155Mock.ArrayLengthsMismatch): - erc1155.safeBatchTransferUnchecked(Address.ZERO, a, b, [0], [100, 100], b"", from_=c) - - with must_revert(ERC1155Mock.TransferToNonERC1155ReceiverImplementer): - erc1155.safeTransferUnchecked(Address.ZERO, a, erc1155, 0, 100, b"", from_=c) - - with must_revert(ERC1155Mock.TransferToNonERC1155ReceiverImplementer): - erc1155.safeTransferUnchecked(a, a, erc1155, 0, 100, b"", from_=c) - - with must_revert(ERC1155Mock.TransferToNonERC1155ReceiverImplementer): - erc1155.safeBatchTransferUnchecked(Address.ZERO, a, erc1155, [0], [100], b"", from_=c) - - with must_revert(ERC1155Mock.TransferToNonERC1155ReceiverImplementer): - erc1155.safeBatchTransferUnchecked(a, a, erc1155, [0], [100], b"", from_=c) diff --git a/lib/solady/ext/woke/test_erc1155_fuzz.py b/lib/solady/ext/woke/test_erc1155_fuzz.py deleted file mode 100644 index f85fbf8..0000000 --- a/lib/solady/ext/woke/test_erc1155_fuzz.py +++ /dev/null @@ -1,537 +0,0 @@ -import logging -from collections import defaultdict -import random -from typing import DefaultDict, Set - -from woke.testing import * -from woke.testing.fuzzing import * -from pytypes.tests.ERC1155Mock import ERC1155Mock - - -logger = logging.getLogger(__name__) -#logger.setLevel(logging.DEBUG) - - -class ERC1155FuzzTest(FuzzTest): - _erc1155: ERC1155Mock - _balances: DefaultDict[Account, DefaultDict[uint256, uint256]] - _approvals: DefaultDict[Account, Set[Account]] - _token_ids: List[uint256] - - def pre_sequence(self) -> None: - self._erc1155 = ERC1155Mock.deploy(True) - self._balances = defaultdict(lambda: defaultdict(lambda: 0)) - self._approvals = defaultdict(set) - self._token_ids = [random_int(0, 2 ** 256 - 1, edge_values_prob=0.25) for _ in range(10)] - - @flow() - def flow_mint(self, payload: bytearray) -> None: - a = random_account() - id = random.choice(self._token_ids) - amount = random_int(0, 2 ** 256 - 1, edge_values_prob=0.05) - - try: - tx = self._erc1155.mint(a, id, amount, payload) - assert self._balances[a][id] + amount < 2 ** 256 - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(Address.ZERO, a.address, [id], [amount], payload), - ERC1155Mock.TransferSingle(tx.from_.address, Address.ZERO, a.address, id, amount), - ERC1155Mock.AfterTokenTransfer(Address.ZERO, a.address, [id], [amount], payload), - ] - self._balances[a][id] += amount - - logger.debug(f"Minted {amount} of {id} to {a}") - except UnknownTransactionRevertedError as e: - assert e.data == ERC1155Mock.AccountBalanceOverflow.selector - assert self._balances[a][id] + amount >= 2 ** 256 - - logger.debug(f"Failed to mint {amount} of {id} to {a}") - - @flow() - def flow_batch_mint(self, payload: bytearray) -> None: - a = random_account() - ids = [random.choice(self._token_ids) for _ in range(random_int(0, 10, edge_values_prob=0.05))] - amounts = [random_int(0, 2 ** 256 - 1, edge_values_prob=0.05) for _ in range(len(ids))] - - try: - tx = self._erc1155.batchMint(a, ids, amounts, payload) - for id, amount in zip(ids, amounts): - assert self._balances[a][id] + amount < 2 ** 256 - self._balances[a][id] += amount - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(Address.ZERO, a.address, ids, amounts, payload), - ERC1155Mock.TransferBatch(tx.from_.address, Address.ZERO, a.address, ids, amounts), - ERC1155Mock.AfterTokenTransfer(Address.ZERO, a.address, ids, amounts, payload), - ] - - logger.debug(f"Minted {amounts} of {ids} to {a}") - except UnknownTransactionRevertedError as e: - assert e.data == ERC1155Mock.AccountBalanceOverflow.selector - amounts_by_ids = defaultdict(int) - for id, amount in zip(ids, amounts): - amounts_by_ids[id] += amount - assert any(self._balances[a][id] + amount >= 2 ** 256 for id, amount in amounts_by_ids.items()) - - logger.debug(f"Failed to mint {amounts} of {ids} to {a}") - - @flow() - def flow_burn(self) -> None: - owner = random_account() - - if random.random() < 0.8 and sum(self._balances[owner].values()) > 0: - id = random.choice([k for k in self._balances[owner].keys() if self._balances[owner][k] > 0]) - else: - id = random.choice(self._token_ids) - - if self._balances[owner][id] == 0: - amount = random.choice([0, 1]) - else: - amount = random_int(0, min(self._balances[owner][id] + 1, 2 ** 256 - 1), min_prob=0.05, max_prob=0.01) - - operator = random.choices( - default_chain.accounts, - [0.5 if a == owner else 0.5 / (len(default_chain.accounts) - 1) for a in default_chain.accounts] - )[0] - - try: - tx = self._erc1155.burn(owner, id, amount, from_=operator) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(owner.address, Address.ZERO, [id], [amount], bytearray()), - ERC1155Mock.TransferSingle(operator.address, owner.address, Address.ZERO, id, amount), - ERC1155Mock.AfterTokenTransfer(owner.address, Address.ZERO, [id], [amount], bytearray()), - ] - assert self._balances[owner][id] - amount >= 0 - self._balances[owner][id] -= amount - - assert operator == owner or operator in self._approvals[owner] - - logger.debug(f"Burned {amount} of {id} from {owner}") - except UnknownTransactionRevertedError as e: - if e.data == ERC1155Mock.InsufficientBalance.selector: - assert self._balances[owner][id] - amount < 0 - elif e.data == ERC1155Mock.NotOwnerNorApproved.selector: - assert operator != owner and operator not in self._approvals[owner] - else: - raise - - logger.debug(f"Failed to burn {amount} of {id} from {owner}") - - @flow() - def flow_burn_batch(self) -> None: - owner = random_account() - - ids = [] - amounts = [] - for _ in range(random_int(0, 10, edge_values_prob=0.05)): - if random.random() < 0.98 and sum(self._balances[owner].values()) > 0: - id = random.choice([k for k in self._balances[owner].keys() if self._balances[owner][k] > 0]) - ids.append(id) - if self._balances[owner][id] == 0: - amount = random.choice([0, 1]) - else: - amount = random_int(0, min(self._balances[owner][id] + 1, 2 ** 256 - 1), edge_values_prob=0.05) - amounts.append(amount) - else: - id = random.choice(self._token_ids) - ids.append(id) - amount = random_int(0, 2 ** 256 - 1, edge_values_prob=0.05) - amounts.append(amount) - - operator = random.choices( - default_chain.accounts, - [0.5 if a == owner else 0.5 / (len(default_chain.accounts) - 1) for a in default_chain.accounts] - )[0] - - try: - tx = self._erc1155.batchBurn(owner, ids, amounts, from_=operator) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(owner.address, Address.ZERO, ids, amounts, bytearray()), - ERC1155Mock.TransferBatch(operator.address, owner.address, Address.ZERO, ids, amounts), - ERC1155Mock.AfterTokenTransfer(owner.address, Address.ZERO, ids, amounts, bytearray()), - ] - for id, amount in zip(ids, amounts): - assert self._balances[owner][id] - amount >= 0 - self._balances[owner][id] -= amount - - assert operator == owner or operator in self._approvals[owner] - - logger.debug(f"Burned {amounts} of {ids} from {owner}") - except UnknownTransactionRevertedError as e: - if e.data == ERC1155Mock.InsufficientBalance.selector: - amounts_by_ids = defaultdict(int) - for id, amount in zip(ids, amounts): - amounts_by_ids[id] += amount - assert any(self._balances[owner][id] - amount < 0 for id, amount in amounts_by_ids.items()) - elif e.data == ERC1155Mock.NotOwnerNorApproved.selector: - assert operator != owner and operator not in self._approvals[owner] - else: - raise - - logger.debug(f"Failed to burn {amounts} of {ids} from {owner}") - - @flow() - def flow_burn_unchecked(self) -> None: - owner = random_account() - - if random.random() < 0.8 and sum(self._balances[owner].values()) > 0: - id = random.choice([k for k in self._balances[owner].keys() if self._balances[owner][k] > 0]) - else: - id = random.choice(self._token_ids) - - if self._balances[owner][id] == 0: - amount = random.choice([0, 1]) - else: - amount = random_int(0, min(self._balances[owner][id] + 1, 2 ** 256 - 1), min_prob=0.05, max_prob=0.01) - - operator = random.choices( - default_chain.accounts + (Account(0), ), - [0.25 if a == owner else 0.5 / (len(default_chain.accounts) - 1) for a in default_chain.accounts] + [0.25] - )[0] - executor = random_account() - - try: - tx = self._erc1155.burnUnchecked(operator, owner, id, amount, from_=executor) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(owner.address, Address.ZERO, [id], [amount], bytearray()), - ERC1155Mock.TransferSingle(executor.address, owner.address, Address.ZERO, id, amount), - ERC1155Mock.AfterTokenTransfer(owner.address, Address.ZERO, [id], [amount], bytearray()), - ] - assert self._balances[owner][id] - amount >= 0 - self._balances[owner][id] -= amount - - assert operator == owner or operator == Account(0) or operator in self._approvals[owner] - - logger.debug(f"Burned {amount} of {id} from {owner}") - except UnknownTransactionRevertedError as e: - if e.data == ERC1155Mock.InsufficientBalance.selector: - assert self._balances[owner][id] - amount < 0 - elif e.data == ERC1155Mock.NotOwnerNorApproved.selector: - assert operator != owner and operator != Account(0) and operator not in self._approvals[owner] - else: - raise - - logger.debug(f"Failed to burn {amount} of {id} from {owner}") - - @flow() - def flow_burn_batch_unchecked(self): - owner = random_account() - - ids = [] - amounts = [] - for _ in range(random_int(0, 10, edge_values_prob=0.05)): - if random.random() < 0.98 and sum(self._balances[owner].values()) > 0: - id = random.choice([k for k in self._balances[owner].keys() if self._balances[owner][k] > 0]) - ids.append(id) - if self._balances[owner][id] == 0: - amount = random.choice([0, 1]) - else: - amount = random_int(0, min(self._balances[owner][id] + 1, 2 ** 256 - 1), edge_values_prob=0.05) - amounts.append(amount) - else: - id = random.choice(self._token_ids) - ids.append(id) - amount = random_int(0, 2 ** 256 - 1, edge_values_prob=0.05) - amounts.append(amount) - - operator = random.choices( - default_chain.accounts + (Account(0), ), - [0.25 if a == owner else 0.5 / (len(default_chain.accounts) - 1) for a in default_chain.accounts] + [0.25] - )[0] - executor = random_account() - - try: - tx = self._erc1155.batchBurnUnchecked(operator, owner, ids, amounts, from_=executor) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(owner.address, Address.ZERO, ids, amounts, bytearray()), - ERC1155Mock.TransferBatch(executor.address, owner.address, Address.ZERO, ids, amounts), - ERC1155Mock.AfterTokenTransfer(owner.address, Address.ZERO, ids, amounts, bytearray()), - ] - for id, amount in zip(ids, amounts): - assert self._balances[owner][id] - amount >= 0 - self._balances[owner][id] -= amount - - assert operator == owner or operator == Account(0) or operator in self._approvals[owner] - - logger.debug(f"Burned {amounts} of {ids} from {owner}") - except UnknownTransactionRevertedError as e: - if e.data == ERC1155Mock.InsufficientBalance.selector: - amounts_by_ids = defaultdict(int) - for id, amount in zip(ids, amounts): - amounts_by_ids[id] += amount - assert any(self._balances[owner][id] - amount < 0 for id, amount in amounts_by_ids.items()) - elif e.data == ERC1155Mock.NotOwnerNorApproved.selector: - assert operator != owner and operator != Account(0) and operator not in self._approvals[owner] - else: - raise - - logger.debug(f"Failed to burn {amounts} of {ids} from {owner}") - - @flow() - def flow_change_approval(self) -> None: - a = random_account() - operator = random_account() - approval = operator in self._approvals[a] - - tx = self._erc1155.setApprovalForAll(operator, not approval, from_=a) - assert tx.events == [ - ERC1155Mock.ApprovalForAll(a.address, operator.address, not approval), - ] - - if approval: - self._approvals[a].remove(operator) - else: - self._approvals[a].add(operator) - - logger.debug(f"Changed approval of {operator} for {a} to {not approval}") - - @flow() - def flow_change_approval_unchecked(self) -> None: - owner = random_account() - operator = random_account() - executor = random_account() - approval = operator in self._approvals[owner] - - tx = self._erc1155.setApprovalForAllUnchecked(owner, operator, not approval, from_=executor) - assert tx.events == [ - ERC1155Mock.ApprovalForAll(owner.address, operator.address, not approval), - ] - - if approval: - self._approvals[owner].remove(operator) - else: - self._approvals[owner].add(operator) - - logger.debug(f"Changed approval of {operator} for {owner} to {not approval}") - - @flow() - def flow_safe_transfer(self, payload: bytearray): - owner = random_account() - recipient = random_account() - - if random.random() < 0.8 and sum(self._balances[owner].values()) > 0: - id = random.choice([k for k in self._balances[owner].keys() if self._balances[owner][k] > 0]) - else: - id = random.choice(self._token_ids) - - if self._balances[owner][id] == 0: - amount = random.choice([0, 1]) - else: - amount = random_int(0, min(self._balances[owner][id] + 1, 2 ** 256 - 1), min_prob=0.05, max_prob=0.01) - - operator = random.choices( - default_chain.accounts, - [0.5 if a == owner else 0.5 / (len(default_chain.accounts) - 1) for a in default_chain.accounts] - )[0] - - try: - tx = self._erc1155.safeTransferFrom(owner, recipient, id, amount, payload, from_=operator) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(owner.address, recipient.address, [id], [amount], payload), - ERC1155Mock.TransferSingle(operator.address, owner.address, recipient.address, id, amount), - ERC1155Mock.AfterTokenTransfer(owner.address, recipient.address, [id], [amount], payload), - ] - assert self._balances[owner][id] - amount >= 0 - self._balances[owner][id] -= amount - assert self._balances[recipient][id] + amount <= 2 ** 256 - 1 - self._balances[recipient][id] += amount - - assert operator == owner or operator in self._approvals[owner] - - logger.debug(f"Transferred {amount} of {id} from {owner} to {recipient}") - except UnknownTransactionRevertedError as e: - if e.data == ERC1155Mock.InsufficientBalance.selector: - assert self._balances[owner][id] - amount < 0 - elif e.data == ERC1155Mock.AccountBalanceOverflow.selector: - assert self._balances[recipient][id] + amount > 2 ** 256 - 1 - elif e.data == ERC1155Mock.NotOwnerNorApproved.selector: - assert operator != owner and operator not in self._approvals[owner] - else: - raise - - logger.debug(f"Failed to transfer {amount} of {id} from {owner} to {recipient}") - - @flow() - def flow_safe_batch_transfer(self, payload: bytearray) -> None: - owner = random_account() - ids = [] - amounts = [] - for _ in range(random_int(0, 10, edge_values_prob=0.05)): - if random.random() < 0.98 and sum(self._balances[owner].values()) > 0: - id = random.choice([k for k in self._balances[owner].keys() if self._balances[owner][k] > 0]) - ids.append(id) - if self._balances[owner][id] == 0: - amount = random.choice([0, 1]) - else: - amount = random_int(0, min(self._balances[owner][id] + 1, 2 ** 256 - 1), edge_values_prob=0.05) - amounts.append(amount) - else: - id = random.choice(self._token_ids) - ids.append(id) - amount = random_int(0, 2 ** 256 - 1, edge_values_prob=0.05) - amounts.append(amount) - - operator = random.choices( - default_chain.accounts, - [0.5 if a == owner else 0.5 / (len(default_chain.accounts) - 1) for a in default_chain.accounts] - )[0] - - try: - tx = self._erc1155.safeBatchTransferFrom(owner, owner, ids, amounts, payload, from_=operator) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(owner.address, owner.address, ids, amounts, payload), - ERC1155Mock.TransferBatch(operator.address, owner.address, owner.address, ids, amounts), - ERC1155Mock.AfterTokenTransfer(owner.address, owner.address, ids, amounts, payload), - ] - for id, amount in zip(ids, amounts): - assert self._balances[owner][id] - amount >= 0 - self._balances[owner][id] -= amount - assert self._balances[owner][id] + amount <= 2 ** 256 - 1 - self._balances[owner][id] += amount - - assert operator == owner or operator in self._approvals[owner] - - logger.debug(f"Transferred {amounts} of {ids} from {owner} to {owner}") - except UnknownTransactionRevertedError as e: - if e.data == ERC1155Mock.InsufficientBalance.selector: - amounts_by_ids = defaultdict(int) - for id, amount in zip(ids, amounts): - amounts_by_ids[id] += amount - assert any(self._balances[owner][id] - amount < 0 for id, amount in amounts_by_ids.items()) - elif e.data == ERC1155Mock.AccountBalanceOverflow.selector: - amounts_by_ids = defaultdict(int) - for id, amount in zip(ids, amounts): - amounts_by_ids[id] += amount - assert any(self._balances[owner][id] + amount > 2 ** 256 - 1 for id, amount in amounts_by_ids.items()) - elif e.data == ERC1155Mock.NotOwnerNorApproved.selector: - assert operator != owner and operator not in self._approvals[owner] - else: - raise - - logger.debug(f"Failed to transfer {amounts} of {ids} from {owner} to {owner}") - - @flow() - def flow_safe_transfer_unchecked(self, payload: bytearray) -> None: - owner = random_account() - recipient = random_account() - - if random.random() < 0.8 and sum(self._balances[owner].values()) > 0: - id = random.choice([k for k in self._balances[owner].keys() if self._balances[owner][k] > 0]) - else: - id = random.choice(self._token_ids) - - if self._balances[owner][id] == 0: - amount = random.choice([0, 1]) - else: - amount = random_int(0, min(self._balances[owner][id] + 1, 2 ** 256 - 1), min_prob=0.05, max_prob=0.01) - - operator = random.choices( - default_chain.accounts + (Account(0), ), - [0.25 if a == owner else 0.5 / (len(default_chain.accounts) - 1) for a in default_chain.accounts] + [0.25] - )[0] - executor = random_account() - - try: - tx = self._erc1155.safeTransferUnchecked(operator, owner, recipient, id, amount, payload, from_=executor) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(owner.address, recipient.address, [id], [amount], payload), - ERC1155Mock.TransferSingle(executor.address, owner.address, recipient.address, id, amount), - ERC1155Mock.AfterTokenTransfer(owner.address, recipient.address, [id], [amount], payload), - ] - assert self._balances[owner][id] - amount >= 0 - self._balances[owner][id] -= amount - assert self._balances[recipient][id] + amount <= 2 ** 256 - 1 - self._balances[recipient][id] += amount - - assert operator == owner or operator == Account(0) or operator in self._approvals[owner] - - logger.debug(f"Transferred {amount} of {id} from {owner} to {recipient}") - except UnknownTransactionRevertedError as e: - if e.data == ERC1155Mock.InsufficientBalance.selector: - assert self._balances[owner][id] - amount < 0 - elif e.data == ERC1155Mock.AccountBalanceOverflow.selector: - assert self._balances[recipient][id] + amount > 2 ** 256 - 1 - elif e.data == ERC1155Mock.NotOwnerNorApproved.selector: - assert operator != owner and operator != Account(0) and operator not in self._approvals[owner] - else: - raise - - logger.debug(f"Failed to transfer {amount} of {id} from {owner} to {recipient}") - - @flow() - def flow_safe_batch_transfer_unchecked(self, payload: bytearray) -> None: - owner = random_account() - ids = [] - amounts = [] - for _ in range(random_int(0, 10, edge_values_prob=0.05)): - if random.random() < 0.98 and sum(self._balances[owner].values()) > 0: - id = random.choice([k for k in self._balances[owner].keys() if self._balances[owner][k] > 0]) - ids.append(id) - if self._balances[owner][id] == 0: - amount = random.choice([0, 1]) - else: - amount = random_int(0, min(self._balances[owner][id] + 1, 2 ** 256 - 1), edge_values_prob=0.05) - amounts.append(amount) - else: - id = random.choice(self._token_ids) - ids.append(id) - amount = random_int(0, 2 ** 256 - 1, edge_values_prob=0.05) - amounts.append(amount) - - operator = random.choices( - default_chain.accounts + (Account(0), ), - [0.25 if a == owner else 0.5 / (len(default_chain.accounts) - 1) for a in default_chain.accounts] + [0.25] - )[0] - executor = random_account() - - try: - tx = self._erc1155.safeBatchTransferUnchecked(operator, owner, owner, ids, amounts, payload, from_=executor) - assert tx.events == [ - ERC1155Mock.BeforeTokenTransfer(owner.address, owner.address, ids, amounts, payload), - ERC1155Mock.TransferBatch(executor.address, owner.address, owner.address, ids, amounts), - ERC1155Mock.AfterTokenTransfer(owner.address, owner.address, ids, amounts, payload), - ] - for id, amount in zip(ids, amounts): - assert self._balances[owner][id] - amount >= 0 - self._balances[owner][id] -= amount - assert self._balances[owner][id] + amount <= 2 ** 256 - 1 - self._balances[owner][id] += amount - - assert operator == owner or operator == Account(0) or operator in self._approvals[owner] - - logger.debug(f"Transferred {amounts} of {ids} from {owner} to {owner}") - except UnknownTransactionRevertedError as e: - if e.data == ERC1155Mock.InsufficientBalance.selector: - amounts_by_ids = defaultdict(int) - for id, amount in zip(ids, amounts): - amounts_by_ids[id] += amount - assert any(self._balances[owner][id] - amount < 0 for id, amount in amounts_by_ids.items()) - elif e.data == ERC1155Mock.AccountBalanceOverflow.selector: - amounts_by_ids = defaultdict(int) - for id, amount in zip(ids, amounts): - amounts_by_ids[id] += amount - assert any(self._balances[owner][id] + amount > 2 ** 256 - 1 for id, amount in amounts_by_ids.items()) - elif e.data == ERC1155Mock.NotOwnerNorApproved.selector: - assert operator != owner and operator != Account(0) and operator not in self._approvals[owner] - else: - raise - - logger.debug(f"Failed to transfer {amounts} of {ids} from {owner} to {owner}") - - @invariant(period=20) - def invariant_balances(self) -> None: - for a, balances in self._balances.items(): - assert self._erc1155.balanceOfBatch([a] * len(balances), list(balances.keys())) == list(balances.values()) - for id, balance in balances.items(): - assert self._erc1155.balanceOf(a, id) == balance - - @invariant(period=20) - def invariant_approvals(self) -> None: - for a in default_chain.accounts: - for b in default_chain.accounts: - assert self._erc1155.isApprovedForAll(a, b) == (b in self._approvals[a]) - - -@default_chain.connect(accounts=20) -def test_erc1155_fuzz(): - default_chain.set_default_accounts(default_chain.accounts[0]) - ERC1155FuzzTest().run(1, 100) diff --git a/lib/solady/ext/woke/test_erc20.py b/lib/solady/ext/woke/test_erc20.py deleted file mode 100644 index 94fb8cd..0000000 --- a/lib/solady/ext/woke/test_erc20.py +++ /dev/null @@ -1,242 +0,0 @@ -from woke.testing import * - -from pytypes.tests.ERC20Mock import ERC20Mock -from pytypes.tests.NoETHMock import NoETHMock -from pytypes.tests.weird.Approval import ApprovalRaceToken -from pytypes.tests.weird.ApprovalToZero import ApprovalToZeroToken -from pytypes.tests.weird.BlockList import BlockableToken -from pytypes.tests.weird.HighDecimals import HighDecimalToken -from pytypes.tests.weird.Bytes32Metadata import ERC20 as Bytes32MetadataToken -from pytypes.tests.weird.MissingReturns import MissingReturnToken -from pytypes.tests.weird.NoRevert import NoRevertToken -from pytypes.tests.weird.Pausable import PausableToken -from pytypes.tests.weird.Proxied import ProxiedToken, TokenProxy -from pytypes.tests.weird.Reentrant import ReentrantToken -from pytypes.tests.weird.ReturnsFalse import ReturnsFalseToken -from pytypes.tests.weird.TransferFee import TransferFeeToken -from pytypes.tests.weird.Uint96 import Uint96ERC20 -from pytypes.tests.weird.Upgradable import Proxy as UpgradableToken - -from pytypes.src.utils.SafeTransferLib import SafeTransferLib - - -@default_chain.connect() -def test_erc20(): - milady = default_chain.accounts[0] - accountoor = default_chain.accounts[1] - default_chain.set_default_accounts(milady) - - tokenoor = ERC20Mock.deploy("Mockoor", "MOCK", 18) - - tokenoor.mint(milady, 2**30) - assert tokenoor.balanceOf(milady) == 2**30 - - tokenoor.approve(accountoor, 2**30) - assert tokenoor.allowance(milady, accountoor) == 2**30 - - tokenoor.transferFrom(milady, accountoor, 2**30, from_=accountoor) - assert tokenoor.allowance(milady, accountoor) == 0 - - assert tokenoor.balanceOf(milady) == 0 - assert tokenoor.balanceOf(accountoor) == 2**30 - -@default_chain.connect() -def test_safe_transfer_eth(): - milady = default_chain.accounts[0] - accountoor = default_chain.accounts[1] - default_chain.set_default_accounts(milady) - - SafeTransferLib.deploy() - - tokenoor = ERC20Mock.deploy("Mockoor", "MOCK", 18) - tokenoor.balance = 5000 - accountoor.balance = 0 - - # should change balance - tokenoor.safeTransferETH(accountoor, 1000) - assert tokenoor.balance == 4000 - assert accountoor.balance == 1000 - - noeth = NoETHMock.deploy() - - # should revert - with must_revert(): - tokenoor.safeTransferETH(noeth, 1000) - - # should change balance - tokenoor.forceSafeTransferETH(noeth, 1000) - assert tokenoor.balance == 3000 - assert noeth.balance == 1000 - - # should force on bad gas stipend - tokenoor.forceSafeTransferETHGas(accountoor, 1000, 0) - assert tokenoor.balance == 2000 - assert accountoor.balance == 2000 - - # should change balance - tokenoor.trySafeTransferETH(accountoor, 1000, 0) - assert tokenoor.balance == 1000 - assert accountoor.balance == 3000 - - # should not revert - tokenoor.trySafeTransferETH(noeth, 1000, 0) - assert tokenoor.balance == 1000 - assert noeth.balance == 1000 - -@default_chain.connect() -def test_safe_transfer(): - milady = default_chain.accounts[0] - default_chain.set_default_accounts(milady) - - SafeTransferLib.deploy() - - tokenoor = ERC20Mock.deploy("Mockoor", "MOCK", 18) - tokenoor.mint(tokenoor, 2**30) - - tokenoor.safeTransfer(tokenoor, milady, 2**30) - assert tokenoor.balanceOf(milady) == 2**30 - assert tokenoor.balanceOf(tokenoor) == 0 - - tokenoor.approve(tokenoor, 2**30) - assert tokenoor.allowance(milady, tokenoor) == 2**30 - tokenoor.safeTransferFrom(tokenoor, milady, tokenoor, 2**30) - assert tokenoor.balanceOf(milady) == 0 - assert tokenoor.balanceOf(tokenoor) == 2**30 - - tokenoor.mint(tokenoor, 2**30) - tokenoor.safeTransferAll(tokenoor, milady) - assert tokenoor.balanceOf(milady) == 2**30 * 2 - assert tokenoor.balanceOf(tokenoor) == 0 - - # test safe balanceOf - assert tokenoor.balanceOfoor(milady, milady) == 0 - assert tokenoor.balanceOfoor(tokenoor, milady) == 2**30 * 2 - -def safe_transfer_weird(weird: Account): - milady = default_chain.accounts[0] - weird = ERC20Mock(weird) - - SafeTransferLib.deploy() - - tokenoor = ERC20Mock.deploy("Mockoor", "MOCK", 18) - - weird.mint(tokenoor, 2**30) - tokenoor.safeTransfer(weird, milady, 2**30) - assert weird.balanceOf(milady) == 2**30 - assert weird.balanceOf(tokenoor) == 0 - - weird.mint(tokenoor, 2**30) - tokenoor.safeTransferAll(weird, milady) - assert weird.balanceOf(milady) == 2**30 * 2 - assert weird.balanceOf(tokenoor) == 0 - - weird.mint(tokenoor, 2**30) - tokenoor.safeTransferFrom(weird, tokenoor, milady, 2**30) - assert weird.balanceOf(milady) == 2**30 * 3 - assert weird.balanceOf(tokenoor) == 0 - - weird.approve(tokenoor, 2**30) - assert weird.allowance(milady, tokenoor) == 2**30 - tokenoor.safeTransferFrom(weird, milady, tokenoor, 2**30) - assert weird.balanceOf(milady) == 2**30 * 2 - assert weird.balanceOf(tokenoor) == 2**30 - - # test safe balanceOf - assert tokenoor.balanceOfoor(milady, milady) == 0 - assert tokenoor.balanceOfoor(weird, milady) == 2**30 * 2 - -@default_chain.connect() -def test_safe_transfer_weird_1(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = ApprovalRaceToken.deploy(0) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_2(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = ApprovalToZeroToken.deploy(0) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_3(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = BlockableToken.deploy(0) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_4(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = HighDecimalToken.deploy(0) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_5(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = Bytes32MetadataToken.deploy(0) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_6(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = MissingReturnToken.deploy(0) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_7(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = NoRevertToken.deploy(0) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_8(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = PausableToken.deploy(0) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_9(): - default_chain.set_default_accounts(default_chain.accounts[0]) - impl = ProxiedToken.deploy(0) - weird_proxy = TokenProxy.deploy(impl) - weird = ProxiedToken(weird_proxy) - weird.setDelegator(weird_proxy, True) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_10(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = ReturnsFalseToken.deploy(0) - with must_revert(): - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_11(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = ReentrantToken.deploy(0) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_12(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = TransferFeeToken.deploy(0,0) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_13(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = Uint96ERC20.deploy(0) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_safe_transfer_weird_14(): - default_chain.set_default_accounts(default_chain.accounts[0]) - weird = UpgradableToken.deploy(0) - safe_transfer_weird(weird) - -@default_chain.connect() -def test_mint_to_zero_address(): - milady = default_chain.accounts[0] - default_chain.set_default_accounts(milady) - tokenoor = ERC20Mock.deploy("Mockoor", "MOCK", 18) - tokenoor.mint(Address(0), 2**256-1) - assert tokenoor.balanceOf(Address(0)) == 2**256-1 \ No newline at end of file diff --git a/lib/solady/ext/woke/test_erc721_fuzz.py b/lib/solady/ext/woke/test_erc721_fuzz.py deleted file mode 100644 index fb29f77..0000000 --- a/lib/solady/ext/woke/test_erc721_fuzz.py +++ /dev/null @@ -1,365 +0,0 @@ -import random - -from woke.testing.fuzzing import * -from woke.testing import * - -from pytypes.tests.ERC721Mock import ERC721Mock - - -################################################################### -####################### PYTHON ERC721 MODEL ####################### -################################################################### -class ERC721: - # mapping owner -> token id - owners: dict[int, Address] - # mapping owner -> token count - balances: dict[Address, int] - # mapping token id -> approved address - approvals: dict[int, Address] - # mapping operator approval owner -> operator - operators: dict[Address, Address] - def __init__(self): - self.owners = {} - self.balances = {} - self.approvals = {} - self.operators = {} - - def balance_of(self, _owner: Address): - return self.balances[_owner] - - def owner_of(self, _token_id: uint): - return self.owners[_token_id] - - def safe_transfer_from(_from: Address, _to: Address, _token_id: uint, _data: bytes): - return - - def safe_transfer_from(_from: Address, _to: Address, _token_id: uint): - return - - def transfer_from(self,_by: Address, _from: Address, _to: Address, _token_id: uint): - self.transfer(_by, _from, _to, _token_id) - - def transfer(self, _by: Address, _from: Address, _to: Address, _token_id: uint): - if self.owners[_token_id] != _by and self.approvals[_token_id] != _by and self.operators[_from] != _by: - return - self.balances[_from] -= 1 - if _to in self.balances.keys(): - self.balances[_to] += 1 - else: - self.balances[_to] = 1 - - self.owners[_token_id] = _to - if _token_id in self.approvals.keys(): - del self.approvals[_token_id] - - def approve(self, _approved: Address, _token_id: uint): - if _approved == Address(0): - del self.approvals[_token_id] - self.approvals[_token_id] = _approved - - def set_approval_for_all(self, _operator: Address, _owner: Address): - self.operators[_owner] = _operator - - def get_approved(self, _token_id: uint): - return self.approvals[_token_id] - - def is_approve_for_all(self, _owner: Address, _operator: Address): - if self.operators[_owner]: - return True - return False - - def mint(self, _to: Address, _token_id: int): - self.owners[_token_id] = _to - if _to in self.balances.keys(): - self.balances[_to] += 1 - else: - self.balances[_to] = 1 - - def burn(self,_token_id: int): - if _token_id in self.owners.keys(): - self.balances[self.owner_of(_token_id)] -= 1 - del self.owners[_token_id] - if _token_id in self.approvals.keys(): - del self.approvals[_token_id] - - -################################################################### -########################### Fuzz Test ############################# -################################################################### - -class ERC721FuzzTest(FuzzTest): - _erc721: ERC721Mock - _py_erc721: ERC721 - _id_counter: int - _ids: List[int] - # We dont want to use random addresses in flows - # We want more interaction by addresses that are already managing something - _addresses: List[Address] - - def pre_sequence(self) -> None: - self._erc721 = ERC721Mock.deploy() - self._py_erc721 = ERC721() - self._addresses = [] - for i in range(20): - self._addresses.append(random_address()) - - ######################## MINT ######################## - @flow(weight=100) - def mint(self) -> None: - # Random data - to = random.choice(self._addresses) - token_id = random_int(0,(2**256)-1) - # Mint in contract - tx = self._erc721.mint(to, token_id) - # Check events - assert tx.events == [ - ERC721Mock.BeforeTokenTransfer(Address(0), to, token_id), - ERC721Mock.Transfer(Address(0), to, token_id), - ERC721Mock.AfterTokenTransfer(Address(0), to, token_id) - ] - # Mint in Py model - self._py_erc721.mint(to, token_id) - - ######################## BURNS ######################## - @flow(weight=50) - def burn_owner(self) -> None: - if self._py_erc721.owners: - # Random token with owner - token_id, owner = random.choice(list(self._py_erc721.owners.items())) - # Burn in contract, msg.sender == owner - tx = self._erc721.burn(token_id, from_=owner) - # Check events - assert tx.events == [ - ERC721Mock.BeforeTokenTransfer(owner, Address(0), token_id), - ERC721Mock.Transfer(owner, Address(0), token_id), - ERC721Mock.AfterTokenTransfer(owner, Address(0), token_id) - ] - # Burn in Py model - self._py_erc721.burn(token_id) - - @flow(weight=40) - def burn_approved(self) -> None: - if self._py_erc721.approvals: - # Random token with owner - token_id, approved = random.choice(list(self._py_erc721.approvals.items())) - if token_id in self._py_erc721.owners.keys(): - owner = self._py_erc721.owners[token_id] - # Burn in contract, msg.sender == approved - tx = self._erc721.burn(token_id, from_=approved) - # Check events - assert tx.events == [ - ERC721Mock.BeforeTokenTransfer(owner, Address(0), token_id), - ERC721Mock.Transfer(owner, Address(0), token_id), - ERC721Mock.AfterTokenTransfer(owner, Address(0), token_id) - ] - # Burn in Py model - self._py_erc721.burn(token_id) - - @flow(weight=40) - def burn_operator(self) -> None: - if self._py_erc721.operators: - owner, operator = random.choice(list(self._py_erc721.operators.items())) - if owner in self._py_erc721.owners.keys(): - token_id = self._py_erc721.owners[owner] - # Burn in contract, msg.sender == operator - tx = self._erc721.burn(token_id, from_=operator) - # Check events - assert tx.events == [ - ERC721Mock.BeforeTokenTransfer(owner, Address(0), token_id), - ERC721Mock.Transfer(owner, Address(0), token_id), - ERC721Mock.AfterTokenTransfer(owner, Address(0), token_id) - ] - # Burn in Py model - self._py_erc721.burn(token_id) - - ######################## TRANSFER ######################## - @flow(weight=80) - def transfer_owner(self) -> None: - # from == by == owner - if self._py_erc721.owners: - token_id, owner = random.choice(list(self._py_erc721.owners.items())) - to = random.choice(self._addresses) - # Transfer in contract, msg.sender == owner - tx = self._erc721.transfer(owner, to, token_id, from_ = owner) - assert tx.events == [ - ERC721Mock.BeforeTokenTransfer(owner, to, token_id), - ERC721Mock.Transfer(owner, to, token_id), - ERC721Mock.AfterTokenTransfer(owner, to, token_id) - ] - # Transfer in Py model - self._py_erc721.transfer(owner, owner, to, token_id) - - @flow(weight=60) - def transfer_approved(self) -> None: - # by == approved, from == owner - if self._py_erc721.approvals: - token_id, approved = random.choice(list(self._py_erc721.approvals.items())) - if token_id in self._py_erc721.owners.keys(): - owner = self._py_erc721.owners[token_id] - to = random.choice(self._addresses) - # Transfer in contract, msg.sender == approved - tx = self._erc721.transfer(owner, to, token_id, from_ = approved) - assert tx.events == [ - ERC721Mock.BeforeTokenTransfer(owner, to, token_id), - ERC721Mock.Transfer(owner, to, token_id), - ERC721Mock.AfterTokenTransfer(owner, to, token_id) - ] - # Transfer in Py model - self._py_erc721.transfer(approved, owner, to, token_id) - - @flow(weight=60) - def transfer_operator(self) -> None: - # by == operator, from == owner - if self._py_erc721.operators: - owner, operator = random.choice(list(self._py_erc721.operators.items())) - if owner in self._py_erc721.owners.keys(): - token_id = self._py_erc721.owners[owner] - to = random.choice(self._addresses) - # Transfer in contract, msg.sender == operator - tx = self._erc721.transfer(owner, to, token_id, from_ = operator) - assert tx.events == [ - ERC721Mock.BeforeTokenTransfer(owner, to, token_id), - ERC721Mock.Transfer(owner, to, token_id), - ERC721Mock.AfterTokenTransfer(owner, to, token_id) - ] - # Transfer in Py model - self._py_erc721.transfer(operator, owner, to, token_id) - - ###################### TRANSFERS FROM ###################### - @flow(weight=80) - def transfer_from_owner(self) -> None: - # from == by == owner - if self._py_erc721.owners: - token_id, owner = random.choice(list(self._py_erc721.owners.items())) - to = random.choice(self._addresses) - # Transfer in contract, msg.sender == owner - tx = self._erc721.transferFrom(owner, to, token_id, from_ = owner) - assert tx.events == [ - ERC721Mock.BeforeTokenTransfer(owner, to, token_id), - ERC721Mock.Transfer(owner, to, token_id), - ERC721Mock.AfterTokenTransfer(owner, to, token_id) - ] - # Transfer in Py model - self._py_erc721.transfer_from(owner, owner, to, token_id) - - @flow(weight=60) - def transfer_from_approved(self) -> None: - # by == approved, from == owner - if self._py_erc721.approvals: - token_id, approved = random.choice(list(self._py_erc721.approvals.items())) - if token_id in self._py_erc721.owners.keys(): - owner = self._py_erc721.owners[token_id] - to = random.choice(self._addresses) - # Transfer in contract, msg.sender == approved - tx = self._erc721.transferFrom(owner, to, token_id, from_ = approved) - assert tx.events == [ - ERC721Mock.BeforeTokenTransfer(owner, to, token_id), - ERC721Mock.Transfer(owner, to, token_id), - ERC721Mock.AfterTokenTransfer(owner, to, token_id) - ] - # Transfer in Py model - self._py_erc721.transfer_from(approved, owner, to, token_id) - - @flow(weight=60) - def transfer_from_operator(self) -> None: - # by == operator, from == owner - if self._py_erc721.operators: - owner, operator = random.choice(list(self._py_erc721.operators.items())) - if owner in self._py_erc721.owners.keys(): - token_id = self._py_erc721.owners[owner] - to = random.choice(self._addresses) - # Transfer in contract, msg.sender == operator - tx = self._erc721.transferFrom(owner, to, token_id, from_ = operator) - assert tx.events == [ - ERC721Mock.BeforeTokenTransfer(owner, to, token_id), - ERC721Mock.Transfer(owner, to, token_id), - ERC721Mock.AfterTokenTransfer(owner, to, token_id) - ] - # Transfer in Py model - self._py_erc721.transfer_from(operator, owner, to, token_id) - - ######################## APPROVALS ######################## - @flow(weight=50) - def approve_owner(self) -> None: - if self._py_erc721.owners: - token_id, owner = random.choice(list(self._py_erc721.owners.items())) - account = random.choice(self._addresses) - # Approve in contract - tx = self._erc721.approve(account, token_id, from_=owner) - # Check events - assert tx.events == [ - ERC721Mock.Approval(owner, account, token_id), - - ] - # Approve in Py model - self._py_erc721.approve(account, token_id) - - @flow(weight=40) - def dis_approve_owner(self) -> None: - if self._py_erc721.owners: - token_id, owner = random.choice(list(self._py_erc721.owners.items())) - if token_id in self._py_erc721.approvals.keys(): - # Delete approval in contract - tx = self._erc721.approve(Address(0), token_id, from_=owner) - # Check events - assert tx.events == [ - ERC721Mock.Approval(owner, Address(0), token_id), - ] - # Delete approval in Py model - self._py_erc721.approve(Address(0), token_id) - - @flow(weight=40) - def approve_operator(self) -> None: - if self._py_erc721.operators: - owner, operator = random.choice(list(self._py_erc721.operators.items())) - if owner in self._py_erc721.owners.keys(): - token_id = self._py_erc721.owners[owner] - account = random.choice(self._addresses) - # Approve in contract - tx = self._erc721.approve(account, token_id, from_=operator) - # Check events - assert tx.events == [ - ERC721Mock.Approval(owner, account, token_id), - ] - # Approve in Py model - self._py_erc721.approve(account, token_id) - - #################### APPROVE FOR ALL ######################## - @flow(weight=40) - def approve_for_all(self) -> None: - if self._py_erc721.owners: - _, owner = random.choice(list(self._py_erc721.owners.items())) - operator = random.choice(self._addresses) - # Set approve for all in contract - tx = self._erc721.setApprovalForAll(operator, True, from_=owner) - assert tx.events == [ - ERC721Mock.ApprovalForAll(owner, operator, True), - ] - # Set approve for all in Py model - self._py_erc721.set_approval_for_all(operator, owner) - - - @invariant(period=20) - def invariant_owners(self) -> None: - owners = self._py_erc721.owners.items() - for token_id, owner in owners: - assert self._erc721.ownerOf(token_id) == owner - - @invariant(period=20) - def invariant_balances(self) -> None: - balances = self._py_erc721.balances.items() - for owner, count in balances: - assert self._erc721.balanceOf(owner) == count - - @invariant(period=20) - def invariant_approvals(self) -> None: - approvals = self._py_erc721.approvals.items() - for token_id, approved in approvals: - assert self._erc721.getApproved(token_id) == approved - - -@default_chain.connect() -def test_eip712_fuzz(): - default_chain.set_default_accounts(default_chain.accounts[0]) - ERC721FuzzTest().run(30, 600) - diff --git a/lib/solady/ext/woke/test_merkle_proof.py b/lib/solady/ext/woke/test_merkle_proof.py deleted file mode 100644 index 33e1214..0000000 --- a/lib/solady/ext/woke/test_merkle_proof.py +++ /dev/null @@ -1,55 +0,0 @@ -import random - -from woke.testing import * -from woke.testing.fuzzing import random_bytes, random_int -from pytypes.tests.MerkleProofMock import MerkleProofMock - -from .utils import MerkleTree - - -@default_chain.connect() -def test_merkle_proof(): - default_chain.set_default_accounts(default_chain.accounts[0]) - - tree = MerkleTree() - for _ in range(100): - tree.add_leaf(random_bytes(0, 1_000)) - - merkle_proof_mock = MerkleProofMock.deploy() - - for i in range(100): - assert merkle_proof_mock.verify(tree.get_proof(i), tree.root, keccak256(tree.values[i])) - assert merkle_proof_mock.verifyCalldata(tree.get_proof(i), tree.root, keccak256(tree.values[i])) - - -@default_chain.connect() -def test_merkle_multiproof_single(): - default_chain.set_default_accounts(default_chain.accounts[0]) - - tree = MerkleTree() - tree.add_leaf(random_bytes(0, 1_000)) - - merkle_proof_mock = MerkleProofMock.deploy() - assert merkle_proof_mock.verifyMultiProof([], tree.root, [keccak256(tree.values[0])], []) - assert merkle_proof_mock.verifyMultiProofCalldata([], tree.root, [keccak256(tree.values[0])], []) - - assert merkle_proof_mock.verifyMultiProof([keccak256(tree.values[0])], tree.root, [], []) - assert merkle_proof_mock.verifyMultiProofCalldata([keccak256(tree.values[0])], tree.root, [], []) - - -@default_chain.connect() -def test_merkle_multiproof(): - default_chain.set_default_accounts(default_chain.accounts[0]) - - tree = MerkleTree() - for _ in range(1_000): - tree.add_leaf(random_bytes(0, 1_000)) - - merkle_proof_mock = MerkleProofMock.deploy() - - for _ in range(100): - indexes = sorted(random.sample(range(len(tree.values)), random_int(1, 100))) - leaves = [tree.values[i] for i in indexes] - proof, flags = tree.get_multiproof(indexes) - assert merkle_proof_mock.verifyMultiProof(proof, tree.root, [keccak256(leaf) for leaf in leaves], flags) - assert merkle_proof_mock.verifyMultiProofCalldata(proof, tree.root, [keccak256(leaf) for leaf in leaves], flags) diff --git a/lib/solady/ext/woke/test_merkle_proof_fuzz.py b/lib/solady/ext/woke/test_merkle_proof_fuzz.py deleted file mode 100644 index 6006c45..0000000 --- a/lib/solady/ext/woke/test_merkle_proof_fuzz.py +++ /dev/null @@ -1,124 +0,0 @@ -import random - -from woke.testing import * -from woke.testing.fuzzing import * -from pytypes.tests.MerkleProofMock import MerkleProofMock - -from .utils import MerkleTree - - -class MerkleProofFuzzTest(FuzzTest): - _merkle_proof: MerkleProofMock - _tree: MerkleTree - - def __init__(self): - self._merkle_proof = MerkleProofMock.deploy() - - def pre_sequence(self) -> None: - self._tree = MerkleTree() - for _ in range(random_int(1, 1_000)): - self._tree.add_leaf(random_bytes(0, 100)) - - @flow() - def flow_verify(self) -> None: - index = random_int(0, len(self._tree.values) - 1) - proof = self._tree.get_proof(index) - leaf = self._tree.values[index] - - assert self._merkle_proof.verify(proof, self._tree.root, keccak256(leaf)) - assert self._merkle_proof.verifyCalldata(proof, self._tree.root, keccak256(leaf)) - - @flow(weight=40) - def flow_verify_invalid_random(self, proof: List[bytes32], root: bytes32, leaf: bytes) -> None: - try: - index = self._tree.values.index(leaf) - assert self._tree.root == root - assert self._tree.get_proof(index) == proof - return - except Exception: - pass - - leaf_hash = keccak256(leaf) - assert not self._merkle_proof.verify(proof, root, leaf_hash) - assert not self._merkle_proof.verifyCalldata(proof, root, leaf_hash) - - @flow(weight=60) - def flow_verify_invalid_modified(self) -> None: - index = random_int(0, len(self._tree.values) - 1) - leaf = self._tree.values[index] - proof = self._tree.get_proof(index) - root = self._tree.root - - r = random_int(0, 2) - if r == 0: - leaf = random_bytes(32) - elif r == 1: - if len(proof) != 0: - proof[random_int(0, len(proof) - 1)] = random_bytes(32) - else: - proof.append(random_bytes(32)) - else: - root = random_bytes(32) - - assert not self._merkle_proof.verify(proof, root, keccak256(leaf)) - assert not self._merkle_proof.verifyCalldata(proof, root, keccak256(leaf)) - - @flow() - def flow_verify_multiproof(self) -> None: - indexes = sorted(random.sample(range(len(self._tree.values)), random_int(1, len(self._tree.values)))) - leaves = [self._tree.values[i] for i in indexes] - proof, flags = self._tree.get_multiproof(indexes) - - assert self._merkle_proof.verifyMultiProof(proof, self._tree.root, [keccak256(leaf) for leaf in leaves], flags) - assert self._merkle_proof.verifyMultiProofCalldata(proof, self._tree.root, [keccak256(leaf) for leaf in leaves], flags) - - @flow(weight=40) - def flow_verify_multiproof_invalid_random(self, proof: List[bytes32], root: bytes32, leaves: List[bytes], flags: List[bool]) -> None: - try: - indexes = sorted([self._tree.values.index(leaf) for leaf in leaves]) - assert self._tree.root == root - assert self._tree.get_multiproof(indexes) == (proof, flags) - return - except Exception: - pass - - leaf_hashes = [keccak256(leaf) for leaf in leaves] - assert not self._merkle_proof.verifyMultiProof(proof, root, leaf_hashes, flags) - assert not self._merkle_proof.verifyMultiProofCalldata(proof, root, leaf_hashes, flags) - - @flow(weight=60) - def flow_verify_multiproof_invalid_modified(self) -> None: - indexes = sorted(random.sample(range(len(self._tree.values)), random_int(1, len(self._tree.values)))) - leaves = [self._tree.values[i] for i in indexes] - proof, flags = self._tree.get_multiproof(indexes) - root = self._tree.root - - r = random_int(0, 3) - if r == 0: - if len(leaves) != 0: - leaves[random_int(0, len(leaves) - 1)] = random_bytes(32) - else: - leaves.append(random_bytes(32)) - elif r == 1: - if len(proof) != 0: - proof[random_int(0, len(proof) - 1)] = random_bytes(32) - else: - proof.append(random_bytes(32)) - elif r == 2: - if len(flags) != 0: - pos = random_int(0, len(flags) - 1) - flags[pos] = not flags[pos] - else: - flags.append(random.choice([True, False])) - else: - root = random_bytes(32) - - leaf_hashes = [keccak256(leaf) for leaf in leaves] - assert not self._merkle_proof.verifyMultiProof(proof, root, leaf_hashes, flags) - assert not self._merkle_proof.verifyMultiProofCalldata(proof, root, leaf_hashes, flags) - - -@default_chain.connect() -def test_merkle_proof_fuzz(): - default_chain.set_default_accounts(default_chain.accounts[0]) - MerkleProofFuzzTest().run(10, 100) diff --git a/lib/solady/ext/woke/test_signature_checker_fuzz.py b/lib/solady/ext/woke/test_signature_checker_fuzz.py deleted file mode 100644 index c2833b8..0000000 --- a/lib/solady/ext/woke/test_signature_checker_fuzz.py +++ /dev/null @@ -1,178 +0,0 @@ -from woke.testing import * -from woke.testing.fuzzing import * -from pytypes.tests.SignatureCheckerMock import SignatureCheckerMock, ERC1217SignatureChecker - -class SignatureCheckerFuzzTest(FuzzTest): - _signature_checker: SignatureCheckerMock - _erc1271_signature_checker: ERC1217SignatureChecker - _signer: Account - - def pre_sequence(self) -> None: - self._signature_checker = SignatureCheckerMock.deploy() - self._erc1271_signature_checker = ERC1217SignatureChecker.deploy() - self._signer = Account.new() - - @flow() - def flow_check_signature(self) -> None: - data = random_bytes(0, 1000) - hash = keccak256(data) - signature = self._signer.sign_hash(hash) - r = signature[:32] - s = signature[32:64] - v = signature[64] - - assert self._signature_checker.isValidSignatureNow(self._signer, hash, signature) - assert self._signature_checker.isValidSignatureNow_( - self._signer, - hash, - r, - s if v == 27 else (s[0] | 0x80).to_bytes(1, "big") + s[1:], - ) - assert self._signature_checker.isValidSignatureNow__(self._signer, hash, v, r, s) - assert self._signature_checker.isValidSignatureNowCalldata(self._signer, hash, signature) - - # erc1271 - assert self._signature_checker.isValidSignatureNow(self._erc1271_signature_checker, hash, signature, from_=self._signer) - assert self._signature_checker.isValidSignatureNow_( - self._erc1271_signature_checker, - hash, - r, - s if v == 27 else (s[0] | 0x80).to_bytes(1, "big") + s[1:], - from_=self._signer, - ) - assert self._signature_checker.isValidSignatureNow__(self._erc1271_signature_checker, hash, v, r, s, from_=self._signer) - assert self._signature_checker.isValidSignatureNowCalldata(self._erc1271_signature_checker, hash, signature, from_=self._signer) - - assert not self._signature_checker.isValidSignatureNow(self._erc1271_signature_checker, hash, signature) - assert not self._signature_checker.isValidSignatureNow_( - self._erc1271_signature_checker, - hash, - r, - s if v == 27 else (s[0] | 0x80).to_bytes(1, "big") + s[1:], - ) - assert not self._signature_checker.isValidSignatureNow__(self._erc1271_signature_checker, hash, v, r, s) - assert not self._signature_checker.isValidSignatureNowCalldata(self._erc1271_signature_checker, hash, signature) - - @flow(weight=40) - def flow_check_signature_invalid_random(self, signer: Address, hash: bytes32) -> None: - signature = random_bytes(65) - r = signature[:32] - s = signature[32:64] - v = signature[64] - - assert not self._signature_checker.isValidSignatureNow(signer, hash, signature) - assert not self._signature_checker.isValidSignatureNow_( - self._signer, - hash, - r, - s if v == 27 else (s[0] | 0x80).to_bytes(1, "big") + s[1:], - ) - assert not self._signature_checker.isValidSignatureNow__(signer, hash, v, r, s) - assert not self._signature_checker.isValidSignatureNowCalldata(signer, hash, signature) - - assert not self._signature_checker.isValidSignatureNow(self._erc1271_signature_checker, hash, signature, from_=signer) - assert not self._signature_checker.isValidSignatureNow_( - self._erc1271_signature_checker, - hash, - r, - s if v == 27 else (s[0] | 0x80).to_bytes(1, "big") + s[1:], - from_=signer, - ) - assert not self._signature_checker.isValidSignatureNow__(self._erc1271_signature_checker, hash, v, r, s, from_=signer) - assert not self._signature_checker.isValidSignatureNowCalldata(self._erc1271_signature_checker, hash, signature, from_=signer) - - assert not self._signature_checker.isValidERC1271SignatureNow(self._erc1271_signature_checker, hash, signature, from_=signer) - assert not self._signature_checker.isValidERC1271SignatureNow_( - self._erc1271_signature_checker, - hash, - r, - s if v == 27 else (s[0] | 0x80).to_bytes(1, "big") + s[1:], - from_=signer, - ) - assert not self._signature_checker.isValidERC1271SignatureNow__(self._erc1271_signature_checker, hash, v, r, s, from_=signer) - assert not self._signature_checker.isValidERC1271SignatureNowCalldata(self._erc1271_signature_checker, hash, signature, from_=signer) - - @flow(weight=60) - def flow_check_signature_invalid_modified(self) -> None: - signer = self._signer.address - data = random_bytes(0, 1000) - hash = bytearray(keccak256(data)) - signature = bytearray(self._signer.sign_hash(hash)) - original_v = None - - x = random_int(0, 2) - if x == 0: - signer_bytes = bytearray(bytes(signer)) - pos = random_int(0, 19) - new_byte = random_bytes(1)[0] - while signer_bytes[pos] == new_byte: - new_byte = random_bytes(1)[0] - signer_bytes[pos] = new_byte - signer = Address(signer_bytes.hex()) - elif x == 1: - pos = random_int(0, 31) - new_byte = random_bytes(1)[0] - while hash[pos] == new_byte: - new_byte = random_bytes(1)[0] - hash[pos] = new_byte - elif x == 2: - pos = random_int(0, 64) - if pos == 64: - original_v = signature[64] - - new_byte = random_bytes(1)[0] - while signature[pos] == new_byte: - new_byte = random_bytes(1)[0] - signature[pos] = new_byte - else: - assert False - - r = signature[:32] - s = signature[32:64] - v = signature[64] - - if original_v is None: - # v was not modified - vs = s if v == 27 else (s[0] | 0x80).to_bytes(1, "big") + s[1:] - else: - # v was modified - vs = s if original_v == 28 else (s[0] | 0x80).to_bytes(1, "big") + s[1:] - - assert not self._signature_checker.isValidSignatureNow(signer, hash, signature) - assert not self._signature_checker.isValidSignatureNow_( - signer, - hash, - r, - vs, - ) - assert not self._signature_checker.isValidSignatureNow__(signer, hash, v, r, s) - assert not self._signature_checker.isValidSignatureNowCalldata(signer, hash, signature) - - # erc1271 - assert not self._signature_checker.isValidSignatureNow(self._erc1271_signature_checker, hash, signature, from_=signer) - assert not self._signature_checker.isValidSignatureNow_( - self._erc1271_signature_checker, - hash, - r, - vs, - from_=signer, - ) - assert not self._signature_checker.isValidSignatureNow__(self._erc1271_signature_checker, hash, v, r, s, from_=signer) - assert not self._signature_checker.isValidSignatureNowCalldata(self._erc1271_signature_checker, hash, signature, from_=signer) - - assert not self._signature_checker.isValidERC1271SignatureNow(self._erc1271_signature_checker, hash, signature, from_=signer) - assert not self._signature_checker.isValidERC1271SignatureNow_( - self._erc1271_signature_checker, - hash, - r, - vs, - from_=signer, - ) - assert not self._signature_checker.isValidERC1271SignatureNow__(self._erc1271_signature_checker, hash, v, r, s, from_=signer) - assert not self._signature_checker.isValidERC1271SignatureNowCalldata(self._erc1271_signature_checker, hash, signature, from_=signer) - - -@default_chain.connect() -def test_signature_checker(): - default_chain.set_default_accounts(default_chain.accounts[0]) - SignatureCheckerFuzzTest().run(10, 20) diff --git a/lib/solady/ext/woke/utils.py b/lib/solady/ext/woke/utils.py deleted file mode 100644 index dfdab8b..0000000 --- a/lib/solady/ext/woke/utils.py +++ /dev/null @@ -1,89 +0,0 @@ -from typing import List, Tuple - -from woke.testing import keccak256 - - -class MerkleTree: - _is_ready: bool - _leaves: List[bytes] - _levels: List[List[bytes]] - - def __init__(self): - self._is_ready = False - self._leaves = [] - self._levels = [] - - @property - def root(self) -> bytes: - if not self._is_ready: - self._build_tree() - return self._levels[-1][0] - - @property - def values(self) -> Tuple[bytes, ...]: - return tuple(self._leaves) - - def get_proof(self, index: int) -> List[bytes]: - if not self._is_ready: - self._build_tree() - - proof = [] - for level in self._levels[:-1]: - if index % 2 == 0: - proof.append(level[index + 1]) - else: - proof.append(level[index - 1]) - index //= 2 - return proof - - def get_multiproof(self, indexes: List[int]) -> Tuple[List[bytes], List[bool]]: - if not self._is_ready: - self._build_tree() - - proof = [] - flags = [] - known = indexes - assert known == sorted(known), "Leaves must be sorted" - - for level in self._levels[:-1]: - new_known = [] - for i in known: - if i % 2 == 0: - if i + 1 in known: - flags.append(True) - else: - flags.append(False) - if i + 1 < len(level): - proof.append(level[i + 1]) - else: - proof.append(level[i]) - else: - if i - 1 in known: - pass # already processed - else: - flags.append(False) - proof.append(level[i - 1]) - if len(new_known) == 0 or new_known[-1] != i // 2: - new_known.append(i // 2) - known = new_known - - return proof, flags - - def add_leaf(self, leaf: bytes): - self._leaves.append(leaf) - self._is_ready = False - - def _build_tree(self) -> None: - self._levels.append([keccak256(leaf) for leaf in self._leaves]) - while len(self._levels[-1]) > 1: - self._levels.append(self._build_level(self._levels[-1])) - self._is_ready = True - - def _build_level(self, level: List[bytes]) -> List[bytes]: - if len(level) % 2 == 1: - level.append(level[-1]) - return [ - keccak256(level[i] + level[i + 1]) if level[i] < level[i + 1] - else keccak256(level[i + 1] + level[i]) - for i in range(0, len(level), 2) - ] diff --git a/lib/solady/ext/woke/weird/Approval.sol b/lib/solady/ext/woke/weird/Approval.sol deleted file mode 100644 index ff1026f..0000000 --- a/lib/solady/ext/woke/weird/Approval.sol +++ /dev/null @@ -1,17 +0,0 @@ -// Copyright (C) 2020 d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -import {ERC20} from "./ERC20.sol"; - -contract ApprovalRaceToken is ERC20 { - // --- Init --- - constructor(uint _totalSupply) ERC20(_totalSupply) public {} - - // --- Token --- - function approve(address usr, uint wad) override public returns (bool) { - require(allowance[msg.sender][usr] == 0, "unsafe-approve"); - return super.approve(usr, wad); - } -} diff --git a/lib/solady/ext/woke/weird/ApprovalToZero.sol b/lib/solady/ext/woke/weird/ApprovalToZero.sol deleted file mode 100644 index 01ffff3..0000000 --- a/lib/solady/ext/woke/weird/ApprovalToZero.sol +++ /dev/null @@ -1,17 +0,0 @@ -// Copyright (C) 2020 d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -import {ERC20} from "./ERC20.sol"; - -contract ApprovalToZeroToken is ERC20 { - // --- Init --- - constructor(uint _totalSupply) ERC20(_totalSupply) public {} - - // --- Token --- - function approve(address usr, uint wad) override public returns (bool) { - require(usr != address(0), "no approval for the zero address"); - return super.approve(usr, wad); - } -} diff --git a/lib/solady/ext/woke/weird/BlockList.sol b/lib/solady/ext/woke/weird/BlockList.sol deleted file mode 100644 index a4f76f1..0000000 --- a/lib/solady/ext/woke/weird/BlockList.sol +++ /dev/null @@ -1,29 +0,0 @@ -// Copyright (C) 2020 d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -import {ERC20} from "./ERC20.sol"; - -contract BlockableToken is ERC20 { - // --- Access Control --- - address owner; - modifier auth() { require(msg.sender == owner, "unauthorised"); _; } - - // --- BlockList --- - mapping(address => bool) blocked; - function block(address usr) auth public { blocked[usr] = true; } - function allow(address usr) auth public { blocked[usr] = false; } - - // --- Init --- - constructor(uint _totalSupply) ERC20(_totalSupply) public { - owner = msg.sender; - } - - // --- Token --- - function transferFrom(address src, address dst, uint wad) override public returns (bool) { - require(!blocked[src], "blocked"); - require(!blocked[dst], "blocked"); - return super.transferFrom(src, dst, wad); - } -} diff --git a/lib/solady/ext/woke/weird/Bytes32Metadata.sol b/lib/solady/ext/woke/weird/Bytes32Metadata.sol deleted file mode 100644 index 3a03f27..0000000 --- a/lib/solady/ext/woke/weird/Bytes32Metadata.sol +++ /dev/null @@ -1,62 +0,0 @@ -// Copyright (C) 2017, 2018, 2019, 2020 dbrock, rain, mrchico, d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -contract Math { - // --- Math --- - function add(uint x, uint y) internal pure returns (uint z) { - require((z = x + y) >= x); - } - function sub(uint x, uint y) internal pure returns (uint z) { - require((z = x - y) <= x); - } -} - -contract ERC20 is Math { - // --- ERC20 Data --- - bytes32 public constant name = "Token"; - bytes32 public constant symbol = "TKN"; - uint8 public constant decimals = 18; - uint256 public totalSupply; - - mapping (address => uint) public balanceOf; - mapping (address => mapping (address => uint)) public allowance; - - event Approval(address indexed src, address indexed guy, uint wad); - event Transfer(address indexed src, address indexed dst, uint wad); - - // --- Init --- - constructor(uint _totalSupply) public { - totalSupply = _totalSupply; - balanceOf[msg.sender] = _totalSupply; - emit Transfer(address(0), msg.sender, _totalSupply); - } - - // --- Token --- - function transfer(address dst, uint wad) virtual public returns (bool) { - return transferFrom(msg.sender, dst, wad); - } - function transferFrom(address src, address dst, uint wad) virtual public returns (bool) { - require(balanceOf[src] >= wad, "insufficient-balance"); - if (src != msg.sender && allowance[src][msg.sender] != type(uint).max) { - require(allowance[src][msg.sender] >= wad, "insufficient-allowance"); - allowance[src][msg.sender] = sub(allowance[src][msg.sender], wad); - } - balanceOf[src] = sub(balanceOf[src], wad); - balanceOf[dst] = add(balanceOf[dst], wad); - emit Transfer(src, dst, wad); - return true; - } - function approve(address usr, uint wad) virtual public returns (bool) { - allowance[msg.sender][usr] = wad; - emit Approval(msg.sender, usr, wad); - return true; - } - - function mint(address usr, uint wad) virtual public { - balanceOf[usr] = add(balanceOf[usr], wad); - totalSupply = add(totalSupply, wad); - emit Transfer(address(0), usr, wad); - } -} diff --git a/lib/solady/ext/woke/weird/DaiPermit.sol b/lib/solady/ext/woke/weird/DaiPermit.sol deleted file mode 100644 index 80188b8..0000000 --- a/lib/solady/ext/woke/weird/DaiPermit.sol +++ /dev/null @@ -1,89 +0,0 @@ -// Copyright (C) 2017, 2018, 2019, 2020 dbrock, rain, mrchico, d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -contract Math { - // --- Math --- - function add(uint x, uint y) internal pure returns (uint z) { - require((z = x + y) >= x); - } - function sub(uint x, uint y) internal pure returns (uint z) { - require((z = x - y) <= x); - } -} - -contract DaiPermit is Math { - // --- ERC20 Data --- - string public constant name = "Token"; - string public constant symbol = "TKN"; - uint8 public decimals = 18; - uint256 public totalSupply; - bytes32 public constant PERMIT_TYPEHASH = 0xea2aa0a1be11a07ed86d755c93467f4f82362b452371d1ba94d1715123511acb; - bytes32 public immutable DOMAIN_SEPARATOR = keccak256( - abi.encode( - keccak256('EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)'), - keccak256(bytes(name)), - keccak256(bytes('1')), - block.chainid, - address(this) - ) - ); - - mapping (address => uint) public balanceOf; - mapping (address => mapping (address => uint)) public allowance; - mapping (address => uint) public nonces; - - event Approval(address indexed src, address indexed guy, uint wad); - event Transfer(address indexed src, address indexed dst, uint wad); - - // --- Init --- - constructor(uint _totalSupply) public { - totalSupply = _totalSupply; - balanceOf[msg.sender] = _totalSupply; - emit Transfer(address(0), msg.sender, _totalSupply); - } - - // --- Token --- - function transfer(address dst, uint wad) virtual public returns (bool) { - return transferFrom(msg.sender, dst, wad); - } - function transferFrom(address src, address dst, uint wad) virtual public returns (bool) { - require(balanceOf[src] >= wad, "insufficient-balance"); - if (src != msg.sender && allowance[src][msg.sender] != type(uint).max) { - require(allowance[src][msg.sender] >= wad, "insufficient-allowance"); - allowance[src][msg.sender] = sub(allowance[src][msg.sender], wad); - } - balanceOf[src] = sub(balanceOf[src], wad); - balanceOf[dst] = add(balanceOf[dst], wad); - emit Transfer(src, dst, wad); - return true; - } - function approve(address usr, uint wad) virtual public returns (bool) { - allowance[msg.sender][usr] = wad; - emit Approval(msg.sender, usr, wad); - return true; - } - function permit(address holder, address spender, uint256 nonce, uint256 expiry, - bool allowed, uint8 v, bytes32 r, bytes32 s) external - { - bytes32 digest = - keccak256(abi.encodePacked( - "\x19\x01", - DOMAIN_SEPARATOR, - keccak256(abi.encode(PERMIT_TYPEHASH, - holder, - spender, - nonce, - expiry, - allowed)) - )); - - require(holder != address(0), "Dai/invalid-address-0"); - require(holder == ecrecover(digest, v, r, s), "Dai/invalid-permit"); - require(expiry == 0 || block.timestamp <= expiry, "Dai/permit-expired"); - require(nonce == nonces[holder]++, "Dai/invalid-nonce"); - uint wad = allowed ? type(uint256).max : 0; - allowance[holder][spender] = wad; - } -} diff --git a/lib/solady/ext/woke/weird/ERC20.sol b/lib/solady/ext/woke/weird/ERC20.sol deleted file mode 100644 index 15adf96..0000000 --- a/lib/solady/ext/woke/weird/ERC20.sol +++ /dev/null @@ -1,62 +0,0 @@ -// Copyright (C) 2017, 2018, 2019, 2020 dbrock, rain, mrchico, d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -contract Math { - // --- Math --- - function add(uint x, uint y) internal pure returns (uint z) { - require((z = x + y) >= x); - } - function sub(uint x, uint y) internal pure returns (uint z) { - require((z = x - y) <= x); - } -} - -contract ERC20 is Math { - // --- ERC20 Data --- - string public constant name = "Token"; - string public constant symbol = "TKN"; - uint8 public decimals = 18; - uint256 public totalSupply; - - mapping (address => uint) public balanceOf; - mapping (address => mapping (address => uint)) public allowance; - - event Approval(address indexed src, address indexed guy, uint wad); - event Transfer(address indexed src, address indexed dst, uint wad); - - // --- Init --- - constructor(uint _totalSupply) public { - totalSupply = _totalSupply; - balanceOf[msg.sender] = _totalSupply; - emit Transfer(address(0), msg.sender, _totalSupply); - } - - // --- Token --- - function transfer(address dst, uint wad) virtual public returns (bool) { - return transferFrom(msg.sender, dst, wad); - } - function transferFrom(address src, address dst, uint wad) virtual public returns (bool) { - require(balanceOf[src] >= wad, "insufficient-balance"); - if (src != msg.sender && allowance[src][msg.sender] != type(uint).max) { - require(allowance[src][msg.sender] >= wad, "insufficient-allowance"); - allowance[src][msg.sender] = sub(allowance[src][msg.sender], wad); - } - balanceOf[src] = sub(balanceOf[src], wad); - balanceOf[dst] = add(balanceOf[dst], wad); - emit Transfer(src, dst, wad); - return true; - } - function approve(address usr, uint wad) virtual public returns (bool) { - allowance[msg.sender][usr] = wad; - emit Approval(msg.sender, usr, wad); - return true; - } - - function mint(address usr, uint wad) virtual public { - balanceOf[usr] = add(balanceOf[usr], wad); - totalSupply = add(totalSupply, wad); - emit Transfer(address(0), usr, wad); - } -} diff --git a/lib/solady/ext/woke/weird/HighDecimals.sol b/lib/solady/ext/woke/weird/HighDecimals.sol deleted file mode 100644 index 423c2f4..0000000 --- a/lib/solady/ext/woke/weird/HighDecimals.sol +++ /dev/null @@ -1,12 +0,0 @@ -// Copyright (C) 2020 d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -import {ERC20} from "./ERC20.sol"; - -contract HighDecimalToken is ERC20 { - constructor(uint _totalSupply) ERC20(_totalSupply) public { - decimals = 50; - } -} diff --git a/lib/solady/ext/woke/weird/LowDecimals.sol b/lib/solady/ext/woke/weird/LowDecimals.sol deleted file mode 100644 index aa5284d..0000000 --- a/lib/solady/ext/woke/weird/LowDecimals.sol +++ /dev/null @@ -1,12 +0,0 @@ -// Copyright (C) 2020 d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -import {ERC20} from "./ERC20.sol"; - -contract LowDecimalToken is ERC20 { - constructor(uint _totalSupply) ERC20(_totalSupply) public { - decimals = 2; - } -} diff --git a/lib/solady/ext/woke/weird/MissingReturns.sol b/lib/solady/ext/woke/weird/MissingReturns.sol deleted file mode 100644 index 6793b53..0000000 --- a/lib/solady/ext/woke/weird/MissingReturns.sol +++ /dev/null @@ -1,58 +0,0 @@ -// Copyright (C) 2017, 2018, 2019, 2020 dbrock, rain, mrchico, d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -contract MissingReturnToken { - // --- ERC20 Data --- - string public constant name = "Token"; - string public constant symbol = "TKN"; - uint8 public constant decimals = 18; - uint256 public totalSupply; - - mapping (address => uint) public balanceOf; - mapping (address => mapping (address => uint)) public allowance; - - event Approval(address indexed src, address indexed guy, uint wad); - event Transfer(address indexed src, address indexed dst, uint wad); - - // --- Math --- - function add(uint x, uint y) internal pure returns (uint z) { - require((z = x + y) >= x); - } - function sub(uint x, uint y) internal pure returns (uint z) { - require((z = x - y) <= x); - } - - // --- Init --- - constructor(uint _totalSupply) public { - totalSupply = _totalSupply; - balanceOf[msg.sender] = _totalSupply; - emit Transfer(address(0), msg.sender, _totalSupply); - } - - // --- Token --- - function transfer(address dst, uint wad) external { - transferFrom(msg.sender, dst, wad); - } - function transferFrom(address src, address dst, uint wad) public { - require(balanceOf[src] >= wad, "insufficient-balance"); - if (src != msg.sender && allowance[src][msg.sender] != type(uint).max) { - require(allowance[src][msg.sender] >= wad, "insufficient-allowance"); - allowance[src][msg.sender] = sub(allowance[src][msg.sender], wad); - } - balanceOf[src] = sub(balanceOf[src], wad); - balanceOf[dst] = add(balanceOf[dst], wad); - emit Transfer(src, dst, wad); - } - function approve(address usr, uint wad) external { - allowance[msg.sender][usr] = wad; - emit Approval(msg.sender, usr, wad); - } - - function mint(address usr, uint wad) external { - balanceOf[usr] = add(balanceOf[usr], wad); - totalSupply = add(totalSupply, wad); - emit Transfer(address(0), usr, wad); - } -} diff --git a/lib/solady/ext/woke/weird/NoRevert.sol b/lib/solady/ext/woke/weird/NoRevert.sol deleted file mode 100644 index 01e5abc..0000000 --- a/lib/solady/ext/woke/weird/NoRevert.sol +++ /dev/null @@ -1,56 +0,0 @@ -// Copyright (C) 2017, 2018, 2019, 2020 dbrock, rain, mrchico, d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -contract NoRevertToken { - // --- ERC20 Data --- - string public constant name = "Token"; - string public constant symbol = "TKN"; - uint8 public decimals = 18; - uint256 public totalSupply; - - mapping (address => uint) public balanceOf; - mapping (address => mapping (address => uint)) public allowance; - - event Approval(address indexed src, address indexed guy, uint wad); - event Transfer(address indexed src, address indexed dst, uint wad); - - // --- Init --- - constructor(uint _totalSupply) public { - totalSupply = _totalSupply; - balanceOf[msg.sender] = _totalSupply; - emit Transfer(address(0), msg.sender, _totalSupply); - } - - // --- Token --- - function transfer(address dst, uint wad) external returns (bool) { - return transferFrom(msg.sender, dst, wad); - } - function transferFrom(address src, address dst, uint wad) virtual public returns (bool) { - if (balanceOf[src] < wad) return false; // insufficient src bal - if (balanceOf[dst] >= (type(uint256).max - wad)) return false; // dst bal too high - - if (src != msg.sender && allowance[src][msg.sender] != type(uint).max) { - if (allowance[src][msg.sender] < wad) return false; // insufficient allowance - allowance[src][msg.sender] = allowance[src][msg.sender] - wad; - } - - balanceOf[src] = balanceOf[src] - wad; - balanceOf[dst] = balanceOf[dst] + wad; - - emit Transfer(src, dst, wad); - return true; - } - function approve(address usr, uint wad) virtual external returns (bool) { - allowance[msg.sender][usr] = wad; - emit Approval(msg.sender, usr, wad); - return true; - } - - function mint(address usr, uint wad) virtual external { - balanceOf[usr] = balanceOf[usr] + wad; - totalSupply = totalSupply + wad; - emit Transfer(address(0), usr, wad); - } -} diff --git a/lib/solady/ext/woke/weird/Pausable.sol b/lib/solady/ext/woke/weird/Pausable.sol deleted file mode 100644 index cc0518f..0000000 --- a/lib/solady/ext/woke/weird/Pausable.sol +++ /dev/null @@ -1,36 +0,0 @@ -// Copyright (C) 2020 d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -import {ERC20} from "./ERC20.sol"; - -contract PausableToken is ERC20 { - // --- Access Control --- - address owner; - modifier auth() { require(msg.sender == owner, "unauthorised"); _; } - - // --- Pause --- - bool live = true; - function stop() auth external { live = false; } - function start() auth external { live = true; } - - // --- Init --- - constructor(uint _totalSupply) ERC20(_totalSupply) public { - owner = msg.sender; - } - - // --- Token --- - function approve(address usr, uint wad) override public returns (bool) { - require(live, "paused"); - return super.approve(usr, wad); - } - function transfer(address dst, uint wad) override public returns (bool) { - require(live, "paused"); - return super.transfer(dst, wad); - } - function transferFrom(address src, address dst, uint wad) override public returns (bool) { - require(live, "paused"); - return super.transferFrom(src, dst, wad); - } -} diff --git a/lib/solady/ext/woke/weird/Proxied.sol b/lib/solady/ext/woke/weird/Proxied.sol deleted file mode 100644 index 6ae632d..0000000 --- a/lib/solady/ext/woke/weird/Proxied.sol +++ /dev/null @@ -1,140 +0,0 @@ -// Copyright (C) 2017, 2018, 2019, 2020 dbrock, rain, mrchico, d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -/* - Provides two contracts: - - 1. ProxiedToken: The underlying token, state modifications must be made through a proxy - 2. TokenProxy: Proxy contract, appends the original msg.sender to any calldata provided by the user - - The ProxiedToken can have many trusted frontends (TokenProxy's). - The frontends should append the address of their caller to calldata when calling into the backend. - Admin users of the ProxiedToken can add new delegators. -*/ - -contract ProxiedToken { - // --- ERC20 Data --- - string public constant name = "Token"; - string public constant symbol = "TKN"; - uint8 public constant decimals = 18; - uint256 public totalSupply; - - mapping (address => uint) public balanceOf; - mapping (address => mapping (address => uint)) public allowance; - - event Approval(address indexed src, address indexed guy, uint wad); - event Transfer(address indexed src, address indexed dst, uint wad); - - // --- Math --- - function add(uint x, uint y) internal pure returns (uint z) { - require((z = x + y) >= x); - } - function sub(uint x, uint y) internal pure returns (uint z) { - require((z = x - y) <= x); - } - - // --- Init --- - constructor(uint _totalSupply) public { - admin[msg.sender] = true; - totalSupply = _totalSupply; - balanceOf[msg.sender] = _totalSupply; - emit Transfer(address(0), msg.sender, _totalSupply); - } - - // --- Access Control --- - mapping(address => bool) public admin; - function rely(address usr) external auth { admin[usr] = true; } - function deny(address usr) external auth { admin[usr] = false; } - modifier auth() { require(admin[msg.sender], "non-admin-call"); _; } - - mapping(address => bool) public delegators; - modifier delegated() { require(delegators[msg.sender], "non-delegator-call"); _; } - function setDelegator(address delegator, bool status) external { - delegators[delegator] = status; - } - - // --- Token --- - function transfer(address dst, uint wad) delegated external returns (bool) { - return _transferFrom(_getCaller(), _getCaller(), dst, wad); - } - function transferFrom(address src, address dst, uint wad) delegated external returns (bool) { - return _transferFrom(_getCaller(), src, dst, wad); - } - function approve(address usr, uint wad) delegated external returns (bool) { - return _approve(_getCaller(), usr, wad); - } - - // --- Internals --- - function _transferFrom( - address caller, address src, address dst, uint wad - ) internal returns (bool) { - require(balanceOf[src] >= wad, "insufficient-balance"); - if (src != caller && allowance[src][caller] != type(uint).max) { - require(allowance[src][caller] >= wad, "insufficient-allowance"); - allowance[src][caller] = sub(allowance[src][caller], wad); - } - balanceOf[src] = sub(balanceOf[src], wad); - balanceOf[dst] = add(balanceOf[dst], wad); - emit Transfer(src, dst, wad); - return true; - } - function _approve(address caller, address usr, uint wad) internal returns (bool) { - allowance[caller][usr] = wad; - emit Approval(caller, usr, wad); - return true; - } - // grabs the first word after the calldata and masks it with 20bytes of 1's - // to turn it into an address - function _getCaller() internal pure returns (address result) { - bytes memory array = msg.data; - uint256 index = msg.data.length; - assembly { - result := and(mload(add(array, index)), 0xffffffffffffffffffffffffffffffffffffffff) - } - return result; - } - - function mint(address usr, uint wad) external { - balanceOf[usr] = add(balanceOf[usr], wad); - totalSupply = add(totalSupply, wad); - emit Transfer(address(0), usr, wad); - } -} - -contract TokenProxy { - address payable immutable public impl; - constructor(address _impl) public { - impl = payable(_impl); - } - - receive() external payable { revert("don't send me ETH!"); } - - fallback() external payable { - address _impl = impl; // pull impl onto the stack - assembly { - // get free data pointer - let ptr := mload(0x40) - - // write calldata to ptr - calldatacopy(ptr, 0, calldatasize()) - // store msg.sender after the calldata - mstore(add(ptr, calldatasize()), caller()) - - // call impl with the contents of ptr as caldata - let result := call(gas(), _impl, callvalue(), ptr, add(calldatasize(), 32), 0, 0) - - // copy the returndata to ptr - let size := returndatasize() - returndatacopy(ptr, 0, size) - - switch result - // revert if the call failed - case 0 { revert(ptr, size) } - // return ptr otherwise - default { return(ptr, size) } - } - } - -} diff --git a/lib/solady/ext/woke/weird/Reentrant.sol b/lib/solady/ext/woke/weird/Reentrant.sol deleted file mode 100644 index f14dbe9..0000000 --- a/lib/solady/ext/woke/weird/Reentrant.sol +++ /dev/null @@ -1,31 +0,0 @@ -// Copyright (C) 2020 d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -import {ERC20} from "./ERC20.sol"; - -contract ReentrantToken is ERC20 { - // --- Init --- - constructor(uint _totalSupply) ERC20(_totalSupply) public {} - - // --- Call Targets --- - mapping (address => Target) public targets; - struct Target { - bytes data; - address addr; - } - function setTarget(address addr, bytes calldata data) external { - targets[msg.sender] = Target(data, addr); - } - - // --- Token --- - function transferFrom(address src, address dst, uint wad) override public returns (bool res) { - res = super.transferFrom(src, dst, wad); - Target memory target = targets[src]; - if (target.addr != address(0)) { - (bool status,) = target.addr.call{gas: gasleft()}(target.data); - require(status, "call failed"); - } - } -} diff --git a/lib/solady/ext/woke/weird/ReturnsFalse.sol b/lib/solady/ext/woke/weird/ReturnsFalse.sol deleted file mode 100644 index 5800805..0000000 --- a/lib/solady/ext/woke/weird/ReturnsFalse.sol +++ /dev/null @@ -1,61 +0,0 @@ -// Copyright (C) 2017, 2018, 2019, 2020 dbrock, rain, mrchico, d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -contract ReturnsFalseToken { - // --- ERC20 Data --- - string public constant name = "Token"; - string public constant symbol = "TKN"; - uint8 public constant decimals = 18; - uint256 public totalSupply; - - mapping (address => uint) public balanceOf; - mapping (address => mapping (address => uint)) public allowance; - - event Approval(address indexed src, address indexed guy, uint wad); - event Transfer(address indexed src, address indexed dst, uint wad); - - // --- Math --- - function add(uint x, uint y) internal pure returns (uint z) { - require((z = x + y) >= x); - } - function sub(uint x, uint y) internal pure returns (uint z) { - require((z = x - y) <= x); - } - - // --- Init --- - constructor(uint _totalSupply) public { - totalSupply = _totalSupply; - balanceOf[msg.sender] = _totalSupply; - emit Transfer(address(0), msg.sender, _totalSupply); - } - - // --- Token --- - function transfer(address dst, uint wad) external returns (bool) { - return transferFrom(msg.sender, dst, wad); - } - function transferFrom(address src, address dst, uint wad) public returns (bool) { - require(balanceOf[src] >= wad, "insufficient-balance"); - if (src != msg.sender && allowance[src][msg.sender] != type(uint).max) { - require(allowance[src][msg.sender] >= wad, "insufficient-allowance"); - allowance[src][msg.sender] = sub(allowance[src][msg.sender], wad); - } - balanceOf[src] = sub(balanceOf[src], wad); - balanceOf[dst] = add(balanceOf[dst], wad); - emit Transfer(src, dst, wad); - return false; - } - function approve(address usr, uint wad) external returns (bool) { - allowance[msg.sender][usr] = wad; - emit Approval(msg.sender, usr, wad); - return false; - } - - function mint(address usr, uint wad) external returns (bool) { - balanceOf[usr] = add(balanceOf[usr], wad); - totalSupply = add(totalSupply, wad); - emit Transfer(address(0), usr, wad); - return false; - } -} diff --git a/lib/solady/ext/woke/weird/RevertToZero.sol b/lib/solady/ext/woke/weird/RevertToZero.sol deleted file mode 100644 index 6a50c90..0000000 --- a/lib/solady/ext/woke/weird/RevertToZero.sol +++ /dev/null @@ -1,17 +0,0 @@ -// Copyright (C) 2020 d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -import {ERC20} from "./ERC20.sol"; - -contract RevertToZeroToken is ERC20 { - // --- Init --- - constructor(uint _totalSupply) ERC20(_totalSupply) public {} - - // --- Token --- - function transferFrom(address src, address dst, uint wad) override public returns (bool) { - require(dst != address(0), "transfer-to-zero"); - return super.transferFrom(src, dst, wad); - } -} diff --git a/lib/solady/ext/woke/weird/RevertZero.sol b/lib/solady/ext/woke/weird/RevertZero.sol deleted file mode 100644 index 1357578..0000000 --- a/lib/solady/ext/woke/weird/RevertZero.sol +++ /dev/null @@ -1,17 +0,0 @@ -// Copyright (C) 2020 d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -import {ERC20} from "./ERC20.sol"; - -contract RevertZeroToken is ERC20 { - // --- Init --- - constructor(uint _totalSupply) ERC20(_totalSupply) public {} - - // --- Token --- - function transferFrom(address src, address dst, uint wad) override public returns (bool) { - require(wad != 0, "zero-value-transfer"); - return super.transferFrom(src, dst, wad); - } -} diff --git a/lib/solady/ext/woke/weird/TransferFee.sol b/lib/solady/ext/woke/weird/TransferFee.sol deleted file mode 100644 index 0a1eb58..0000000 --- a/lib/solady/ext/woke/weird/TransferFee.sol +++ /dev/null @@ -1,34 +0,0 @@ -// Copyright (C) 2020 d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -import {ERC20} from "./ERC20.sol"; - -contract TransferFeeToken is ERC20 { - - uint immutable fee; - - // --- Init --- - constructor(uint _totalSupply, uint _fee) ERC20(_totalSupply) public { - fee = _fee; - } - - // --- Token --- - function transferFrom(address src, address dst, uint wad) override public returns (bool) { - require(balanceOf[src] >= wad, "insufficient-balance"); - if (src != msg.sender && allowance[src][msg.sender] != type(uint).max) { - require(allowance[src][msg.sender] >= wad, "insufficient-allowance"); - allowance[src][msg.sender] = sub(allowance[src][msg.sender], wad); - } - - balanceOf[src] = sub(balanceOf[src], wad); - balanceOf[dst] = add(balanceOf[dst], sub(wad, fee)); - balanceOf[address(0)] = add(balanceOf[address(0)], fee); - - emit Transfer(src, dst, sub(wad, fee)); - emit Transfer(src, address(0), fee); - - return true; - } -} diff --git a/lib/solady/ext/woke/weird/Uint96.sol b/lib/solady/ext/woke/weird/Uint96.sol deleted file mode 100644 index 26a862e..0000000 --- a/lib/solady/ext/woke/weird/Uint96.sol +++ /dev/null @@ -1,87 +0,0 @@ -// Copyright (C) 2017, 2018, 2019, 2020 dbrock, rain, mrchico, d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -contract Uint96ERC20 { - // --- ERC20 Data --- - string public constant name = "Token"; - string public constant symbol = "TKN"; - uint8 public decimals = 18; - uint96 internal supply; - - mapping (address => uint96) internal balances; - mapping (address => mapping (address => uint96)) internal allowances; - - event Approval(address indexed src, address indexed guy, uint wad); - event Transfer(address indexed src, address indexed dst, uint wad); - - // --- Math --- - function add(uint96 x, uint96 y) internal pure returns (uint96 z) { - require((z = x + y) >= x); - } - function sub(uint96 x, uint96 y) internal pure returns (uint96 z) { - require((z = x - y) <= x); - } - function safe96(uint256 n) internal pure returns (uint96) { - require(n < 2**96); - return uint96(n); - } - - // --- Init --- - constructor(uint96 _supply) public { - supply = _supply; - balances[msg.sender] = _supply; - emit Transfer(address(0), msg.sender, _supply); - } - - // --- Getters --- - function totalSupply() external view returns (uint) { - return supply; - } - function balanceOf(address usr) external view returns (uint) { - return balances[usr]; - } - function allowance(address src, address dst) external view returns (uint) { - return allowances[src][dst]; - } - - // --- Token --- - function transfer(address dst, uint wad) virtual public returns (bool) { - return transferFrom(msg.sender, dst, wad); - } - function transferFrom(address src, address dst, uint wad) virtual public returns (bool) { - uint96 amt = safe96(wad); - - if (src != msg.sender && allowances[src][msg.sender] != type(uint96).max) { - allowances[src][msg.sender] = sub(allowances[src][msg.sender], amt); - } - - balances[src] = sub(balances[src], amt); - balances[dst] = add(balances[dst], amt); - emit Transfer(src, dst, wad); - return true; - } - function approve(address usr, uint wad) virtual public returns (bool) { - uint96 amt; - if (wad == type(uint).max) { - amt = type(uint96).max; - } else { - amt = safe96(wad); - } - - allowances[msg.sender][usr] = amt; - - emit Approval(msg.sender, usr, amt); - return true; - } - - function mint(address usr, uint wad) virtual public { - uint96 amt = safe96(wad); - - balances[usr] = add(balances[usr], amt); - supply = add(supply, amt); - - emit Transfer(address(0), usr, wad); - } -} diff --git a/lib/solady/ext/woke/weird/Upgradable.sol b/lib/solady/ext/woke/weird/Upgradable.sol deleted file mode 100644 index fa04d3a..0000000 --- a/lib/solady/ext/woke/weird/Upgradable.sol +++ /dev/null @@ -1,61 +0,0 @@ -// Copyright (C) 2020 d-xo -// SPDX-License-Identifier: AGPL-3.0-only - -pragma solidity >=0.6.12; - -import {ERC20} from "./ERC20.sol"; - -contract Proxy { - bytes32 constant ADMIN_KEY = bytes32(uint256(keccak256('eip1967.proxy.admin')) - 1); - bytes32 constant IMPLEMENTATION_KEY = bytes32(uint256(keccak256('eip1967.proxy.implementation')) - 1); - - // --- init --- - - constructor(uint totalSupply) public { - - // Manual give() - bytes32 slot = ADMIN_KEY; - address usr = msg.sender; - assembly { sstore(slot, usr) } - - upgrade(address(new ERC20(totalSupply))); - - } - - // --- auth --- - - modifier auth() { require(msg.sender == owner(), "unauthorised"); _; } - - function owner() public view returns (address usr) { - bytes32 slot = ADMIN_KEY; - assembly { usr := sload(slot) } - } - - function give(address usr) public auth { - bytes32 slot = ADMIN_KEY; - assembly { sstore(slot, usr) } - } - - // --- upgrade --- - - function implementation() public view returns (address impl) { - bytes32 slot = IMPLEMENTATION_KEY; - assembly { impl := sload(slot) } - } - - function upgrade(address impl) public auth { - bytes32 slot = IMPLEMENTATION_KEY; - assembly { sstore(slot, impl) } - } - - // --- proxy --- - - fallback() external payable { - address impl = implementation(); - (bool success, bytes memory returndata) = impl.delegatecall{gas: gasleft()}(msg.data); - require(success); - assembly { return(add(returndata, 0x20), mload(returndata)) } - } - - receive() external payable { revert("don't send me ETH!"); } -} diff --git a/lib/solady/ext/woke/woke-via-ir.toml b/lib/solady/ext/woke/woke-via-ir.toml deleted file mode 100644 index e999716..0000000 --- a/lib/solady/ext/woke/woke-via-ir.toml +++ /dev/null @@ -1,28 +0,0 @@ -[compiler.solc] -ignore_paths = ["node_modules", ".woke-build", "venv", "lib", "test"] -include_paths = ["node_modules"] -remappings = [ - "ds-test/=lib/ds-test/src/", - "forge-std/=test/utils/forge-std/", -] -via_IR = true - -[compiler.solc.optimizer] -enabled = true -runs = 1000 - -[detectors] -exclude = ["unused-contract"] -ignore_paths = ["node_modules", ".woke-build", "venv", "lib"] - -[testing] -cmd = "anvil" - -[testing.anvil] -cmd_args = "--prune-history 100 --transaction-block-keeper 10 --steps-tracing --silent" - -[testing.ganache] -cmd_args = "-k istanbul -q" - -[testing.hardhat] -cmd_args = "" \ No newline at end of file diff --git a/lib/solady/ext/woke/woke.toml b/lib/solady/ext/woke/woke.toml deleted file mode 100644 index 476a151..0000000 --- a/lib/solady/ext/woke/woke.toml +++ /dev/null @@ -1,27 +0,0 @@ -[compiler.solc] -ignore_paths = ["node_modules", ".woke-build", "venv", "lib", "test"] -include_paths = ["node_modules"] -remappings = [ - "ds-test/=lib/ds-test/src/", - "forge-std/=test/utils/forge-std/", -] - -[compiler.solc.optimizer] -enabled = true -runs = 1000 - -[detectors] -exclude = ["unused-contract"] -ignore_paths = ["node_modules", ".woke-build", "venv", "lib"] - -[testing] -cmd = "anvil" - -[testing.anvil] -cmd_args = "--prune-history 100 --transaction-block-keeper 10 --steps-tracing --silent" - -[testing.ganache] -cmd_args = "-k istanbul -q" - -[testing.hardhat] -cmd_args = "" diff --git a/lib/solady/foundry.toml b/lib/solady/foundry.toml deleted file mode 100644 index 7f418d8..0000000 --- a/lib/solady/foundry.toml +++ /dev/null @@ -1,21 +0,0 @@ -# Foundry Configuration File -# Default definitions: https://github.com/gakonst/foundry/blob/b7917fa8491aedda4dd6db53fbb206ea233cd531/config/src/lib.rs#L782 -# See more config options at: https://github.com/gakonst/foundry/tree/master/config - -# The Default Profile -[profile.default] -solc_version = "0.8.22" -evm_version = "paris" # Shanghai will be tested in the CI. -auto_detect_solc = false -optimizer = true -optimizer_runs = 1_000 -gas_limit = 100_000_000 # ETH is 30M, but we use a higher value. -remappings = [ - "forge-std=test/utils/forge-std/" -] - -[fmt] -line_length = 100 # While we allow up to 120, we lint at 100 for readability. - -[profile.default.fuzz] -runs = 256 diff --git a/lib/solady/js/solady.d.ts b/lib/solady/js/solady.d.ts deleted file mode 100644 index 318c86b..0000000 --- a/lib/solady/js/solady.d.ts +++ /dev/null @@ -1,66 +0,0 @@ -/** - * Accompanying JavaScript library for Solady. - * - * To install: - * - * ``` - * npm install solady - * ``` - * - * Module exports: - * - * - `LibZip` - * - `flzCompress(data)`: Compresses hex encoded data with FastLZ. - * - `flzDecompress(data)`: Decompresses hex encoded data with FastLZ. - * - `cdCompress(data)`: Compresses hex encoded calldata. - * - `cdDecompress(data)`: Decompresses hex encoded calldata. - * - * - `ERC1967Factory` - * - `address`: Canonical address of Solady's ERC1967Factory. - * - `abi`: ABI of Solady's ERC1967Factory. - */ -declare module "solady" { - /** - * FastLZ and calldata compression / decompression functions. - */ - namespace LibZip { - /** - * Compresses hex encoded data with the FastLZ LZ77 algorithm. - * @param data - A hex encoded string representing the original data. - * @returns The compressed result as a hex encoded string. - */ - function flzCompress(data: string): string; - /** - * Decompresses hex encoded data with the FastLZ LZ77 algorithm. - * @param data - A hex encoded string representing the compressed data. - * @returns The decompressed result as a hex encoded string. - */ - function flzDecompress(data: string): string; - /** - * Compresses hex encoded calldata. - * @param data - A hex encoded string representing the original data. - * @returns The compressed result as a hex encoded string. - */ - function cdCompress(data: string): string; - /** - * Decompresses hex encoded calldata. - * @param data - A hex encoded string representing the compressed data. - * @returns The decompressed result as a hex encoded string. - */ - function cdDecompress(data: string): string; - } - /** - * ERC1967Factory canonical address and ABI. - */ - namespace ERC1967Factory { - /** - * Canonical address of Solady's ERC1967Factory. - */ - var address: string; - /** - * ABI of Solady's ERC1967Factory. - */ - var abi: any; - } -} - diff --git a/lib/solady/js/solady.js b/lib/solady/js/solady.js deleted file mode 100644 index 9b90c80..0000000 --- a/lib/solady/js/solady.js +++ /dev/null @@ -1,255 +0,0 @@ -/** - * Accompanying JavaScript library for Solady. - * - * To install: - * - * ``` - * npm install solady - * ``` - * - * Module exports: - * - * - `LibZip` - * - `flzCompress(data)`: Compresses hex encoded data with FastLZ. - * - `flzDecompress(data)`: Decompresses hex encoded data with FastLZ. - * - `cdCompress(data)`: Compresses hex encoded calldata. - * - `cdDecompress(data)`: Decompresses hex encoded calldata. - * - * - `ERC1967Factory` - * - `address`: Canonical address of Solady's ERC1967Factory. - * - `abi`: ABI of Solady's ERC1967Factory. - * - * @module solady - */ -(function(global, factory) { - - "use strict"; - - if (typeof module === "object" && typeof module.exports === "object") { - module.exports = factory(global, 1); - if (typeof exports === "object") { - exports.LibZip = module.exports.LibZip; - exports.ERC1967Factory = module.exports.ERC1967Factory; - } - } else { - factory(global); - } - -})(typeof window !== "undefined" ? window : this, function(window, noGlobal) { - - "use strict"; - - var solady = {}; - - /*============================================================*/ - /* LibZip Operations */ - /*============================================================*/ - - // See: https://github.com/vectorized/solady/blob/main/src/utils/LibZip.sol - - /** - * FastLZ and calldata compression / decompression functions. - * @namespace - * @alias module:solady.LibZip - */ - var LibZip = {}; - - solady.LibZip = LibZip; - - function hexString(data) { - if (typeof data === "string" || data instanceof String) { - if (data = data.match(/^[\s\uFEFF\xA0]*(0[Xx])?([0-9A-Fa-f]*)[\s\uFEFF\xA0]*$/)) { - if (data[2].length % 2) { - throw new Error("Hex string length must be a multiple of 2."); - } - return data[2]; - } - } - throw new Error("Data must be a hex string."); - } - - function byteToString(b) { - return (b | 0x100).toString(16).slice(1); - } - - function parseByte(data, i) { - return parseInt(data.substr(i, 2), 16); - } - - function hexToBytes(data) { - var a = [], i = 0; - for (; i < data.length; i += 2) a.push(parseByte(data, i)); - return a; - } - - function bytesToHex(a) { - var o = "0x", i = 0; - for (; i < a.length; o += byteToString(a[i++])) ; - return o; - } - - /** - * Compresses hex encoded data with the FastLZ LZ77 algorithm. - * @param {string} data A hex encoded string representing the original data. - * @returns {string} The compressed result as a hex encoded string. - */ - LibZip.flzCompress = function(data) { - var ib = hexToBytes(hexString(data)), b = ib.length - 4; - var ht = [], ob = [], a = 0, i = 2, o = 0, j, s, h, d, c, l, r, p, q, e, m = 0xffffff; - - function u32(i) { - return ib[i] | (ib[++i] << 8) | (ib[++i] << 16) | (ib[++i] << 24); - } - - function hash(x) { - return ((2654435769 * x) >> 19) & 8191; - } - - function literals(r, s) { - while (r >= 32) for (ob[o++] = 31, j = 32; j--; r--) ob[o++] = ib[s++]; - if (r) for (ob[o++] = r - 1; r--; ) ob[o++] = ib[s++]; - } - - while (i < b - 9) { - do { - r = ht[h = hash(s = u32(i) & m)]; - c = (d = (ht[h] = i) - r) < 8192 ? u32(r) & m : m + 1; - } while (i < b - 9 && i++ && s != c); - if (i >= b - 9) break; - if (--i > a) literals(i - a, a); - for (l = 0, p = r + 3, q = i + 3, e = b - q; l < e; l++) e *= ib[p + l] === ib[q + l]; - s = u32(i += l); - for (--d; l > 262; l -= 262) ob[o++] = 224 + (d >> 8), ob[o++] = 253, ob[o++] = d & 255; - if (l < 7) ob[o++] = (l << 5) + (d >> 8), ob[o++] = d & 255; - else ob[o++] = 224 + (d >> 8), ob[o++] = l - 7, ob[o++] = d & 255; - ht[hash(s & m)] = i++, ht[hash(s >> 8)] = i++, a = i; - } - literals(b + 4 - a, a); - return bytesToHex(ob); - } - - /** - * Decompresses hex encoded data with the FastLZ LZ77 algorithm. - * @param {string} data A hex encoded string representing the compressed data. - * @returns {string} The decompressed result as a hex encoded string. - */ - LibZip.flzDecompress = function(data) { - var ib = hexToBytes(hexString(data)), i = 0, o = 0, l, f, t, r, h, ob = []; - while (i < ib.length) { - if (!(t = ib[i] >> 5)) { - for (l = 1 + ib[i++]; l--;) ob[o++] = ib[i++]; - } else { - f = 256 * (ib[i] & 31) + ib[i + 2 - (t = t < 7)]; - l = t ? 2 + (ib[i] >> 5) : 9 + ib[i + 1]; - i = i + 3 - t; - r = o - f - 1; - while (l--) ob[o++] = ob[r++]; - } - } - return bytesToHex(ob); - } - - /** - * Compresses hex encoded calldata. - * @param {string} data A hex encoded string representing the original data. - * @returns {string} The compressed result as a hex encoded string. - */ - LibZip.cdCompress = function(data) { - data = hexString(data); - var o = "0x", z = 0, y = 0, i = 0, c; - - function pushByte(b) { - o += byteToString(((o.length < 4 * 2 + 2) * 0xff) ^ b); - } - - function rle(v, d) { - pushByte(0x00); - pushByte(d - 1 + v * 0x80); - } - - for (; i < data.length; i += 2) { - c = parseByte(data, i); - if (!c) { - if (y) rle(1, y), y = 0; - if (++z === 0x80) rle(0, 0x80), z = 0; - continue; - } - if (c === 0xff) { - if (z) rle(0, z), z = 0; - if (++y === 0x20) rle(1, 0x20), y = 0; - continue; - } - if (y) rle(1, y), y = 0; - if (z) rle(0, z), z = 0; - pushByte(c); - } - if (y) rle(1, y), y = 0; - if (z) rle(0, z), z = 0; - return o; - } - - /** - * Decompresses hex encoded calldata. - * @param {string} data A hex encoded string representing the compressed data. - * @returns {string} The decompressed result as a hex encoded string. - */ - LibZip.cdDecompress = function(data) { - data = hexString(data); - var o = "0x", i = 0, j, c, s; - - while (i < data.length) { - c = ((i < 4 * 2) * 0xff) ^ parseByte(data, i); - i += 2; - if (!c) { - c = ((i < 4 * 2) * 0xff) ^ parseByte(data, i); - s = (c & 0x7f) + 1; - i += 2; - for (j = 0; j < s; ++j) o += byteToString((c >> 7 && j < 32) * 0xff); - continue; - } - o += byteToString(c); - } - return o; - } - - /*============================================================*/ - /* ERC1967Factory */ - /*============================================================*/ - - // See: https://github.com/vectorized/solady/blob/main/src/utils/ERC1967Factory.sol - - /** - * ERC1967Factory canonical address and ABI. - * @namespace - * @alias module:solady.ERC1967Factory - */ - var ERC1967Factory = {}; - - solady.ERC1967Factory = ERC1967Factory; - - /** - * Canonical address of Solady's ERC1967Factory. - * @type {string} - */ - ERC1967Factory.address = "0x0000000000006396FF2a80c067f99B3d2Ab4Df24"; - - /** - * ABI of Solady's ERC1967Factory. - * @type {Object} - */ - ERC1967Factory.abi = JSON.parse('[{0:[],1:"DeploymentFailed"96"SaltDoesNotStartWithCaller"96"Unauthorized"96"UpgradeFailed",2:3959790,9791],1:"AdminChanged",2:10959790,9792,9791],1:"Deployed",2:10959790,9792],1:"Upgraded",2:10},{0:[{90],1:"adminOf",12:[{9199{0:[{90,{91],1:"changeAdmin",12:[],13:"nonpayable",2:15},{0:[{92,{91],1:"deploy",12:[{9098,{0:[{92,{91,{94],1:"deployAndCall",12:[{9098,{0:[{92,{91,{93],1:"deployDeterministic",12:[{9098,{0:[{92,{91,{93,{94],1:"deployDeterministicAndCall",12:[{9098,{0:[],1:"initCodeHash",12:[{6:19,1:"result",2:19}99{0:[{93],1:"predictDeterministicAddress",12:[{6:7,1:"predicted",2:7}99{0:[{90,{92],1:"upgrade",12:[98,{0:[{90,{92,{94],1:"upgradeAndCall",12:[98]'.replace(/9\d/g, function (m) { return ["6:7,1:8,2:7}","6:7,1:9,2:7}","6:7,1:11,2:7}","6:19,1:20,2:19}","6:17,1:18,2:17}","},{4:false,0:[",",2:3},{0:[],1:","{5:true,","],13:16,2:15}","],13:14,2:15},"][m-90] }).replace(/\d+/g, function (m) { return '"' + ("inputs,name,type,error,anonymous,indexed,internalType,address,proxy,admin,event,implementation,outputs,stateMutability,view,function,payable,bytes,data,bytes32,salt".split(",")[m]) + '"' })); - - /*--------------------------- END ----------------------------*/ - - if (typeof define === "function" && define.amd) { - define("solady", [], function() { - return solady - }); - } - - if (!noGlobal) { - window.solady = solady; - } - - return solady; -}); diff --git a/lib/solady/js/solady.test.js b/lib/solady/js/solady.test.js deleted file mode 100644 index 7049b49..0000000 --- a/lib/solady/js/solady.test.js +++ /dev/null @@ -1,123 +0,0 @@ -var solady = require("./solady.js"); - -function test(msg, fn) { - msg = msg.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, "").replace(/([^\.])$/, "$1."); - try { - fn(); - console.log("\x1b[32m[PASS]\x1b[0m", msg); - } catch (e) { - process.exitCode = 1; - console.error("\x1b[31m[FAIL]\x1b[0m", msg); - console.error(e.stack); - } -} - -function assert(cond, msg) { - if (!cond) throw new Error(msg); -} - -function assertEq(a, b) { - assert(a === b, "Assertion failed!\n Expected: " + b + "\n Actual: " + a); -} - -function expectRevert(fn) { - var hasRevert = false; - try { fn() } catch (e) { hasRevert = true } - assert(hasRevert, "Revert expected.\n" + fn); -} - -function randomData() { - var n = ~~(Math.random() * 2000); - var s = Math.random() < 0.5 ? "" : "0x"; - var g = Math.random() < 0.5 ? 0.45 : (Math.random() ? 0.99 : 0.999); - var h = g + 0.5 * (1.0 - g); - for (var i = 0; i < n; ++i) { - var r = Math.random(); - if (r < g) { - s += "00"; - } else if (r < h) { - s += "ff"; - } else { - var b = ((Math.random() * 0x100) & 0xff).toString(16); - s += b.length === 1 ? "0" + b : b; - } - } - return Math.random() < 0.5 ? s.toUpperCase() : s.toLowerCase(); -} - -function padRandomWhitespace(data) { - var before = ""; - var after = ""; - while (Math.random() < 0.5) before += Math.random() ? "\t" : " "; - while (Math.random() < 0.5) after += Math.random() ? "\t" : " "; - return before + data + after; -} - -function testCompressDecompress(compress, decompress) { - var totalDataLength = 0; - var totalCompressedLength = 0; - for (var t = 0; t < 1000; ++t) { - var data = randomData(); - var compressed = compress(padRandomWhitespace(data)); - var decompressed = decompress(padRandomWhitespace(compressed)); - totalDataLength += data.length; - totalCompressedLength += compressed.length; - assertEq(compressed.slice(0, 2), "0x"); - assertEq(decompressed.slice(0, 2), "0x"); - assertEq(decompressed.replace(/^0x/, ""), data.toLowerCase().replace(/^0x/, "")); - } - assert(totalCompressedLength < totalDataLength, "Compress not working as intended."); - - assertEq(compress(""), "0x"); - assertEq(compress("0x"), "0x"); - assertEq(decompress(""), "0x"); - assertEq(decompress("0x"), "0x"); - - function checkRevertOnInvalidInputs(fn) { - expectRevert(function () { fn("hehe") }); - expectRevert(function () { fn("0xa") }); - expectRevert(function () { fn("0xas") }); - expectRevert(function () { fn(123) }); - expectRevert(function () { fn(false) }); - expectRevert(function () { fn(null) }); - expectRevert(function () { fn(undefined) }); - expectRevert(function () { fn([]) }); - expectRevert(function () { fn({}) }); - } - - checkRevertOnInvalidInputs(compress); - checkRevertOnInvalidInputs(decompress); -} - -test("LibZip: FastLZ compress / decompress.", function() { - testCompressDecompress(solady.LibZip.flzCompress, solady.LibZip.flzDecompress); -}); - -test("LibZip: Calldata compress / decompress.", function() { - testCompressDecompress(solady.LibZip.cdCompress, solady.LibZip.cdDecompress); -}); - -test("LibZip: Calldata compress", function() { - var data = "0xac9650d80000000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000000000000000400000000000000000000000000000000000000000000000000000000000000800000000000000000000000000000000000000000000000000000000000000160000000000000000000000000000000000000000000000000000000000000022000000000000000000000000000000000000000000000000000000000000002a000000000000000000000000000000000000000000000000000000000000000a40c49ccbe000000000000000000000000000000000000000000000000000000000005b70e00000000000000000000000000000000000000000000000000000dfc79825feb0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000645c48a7000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000084fc6f7865000000000000000000000000000000000000000000000000000000000005b70e000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffff00000000000000000000000000000000ffffffffffffffffffffffffffffffff00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000004449404b7c00000000000000000000000000000000000000000000000000000000000000000000000000000000000000001f1cdf1a632eaaab40d1c263edf49faf749010a1000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000064df2ab5bb0000000000000000000000007f5c764cbc14f9669b88837ca1490cca17c3160700000000000000000000000000000000000000000000000000000000000000000000000000000000000000001f1cdf1a632eaaab40d1c263edf49faf749010a100000000000000000000000000000000000000000000000000000000"; - var expected = "0x5369af27001e20001e04001e80001d0160001d0220001d02a0001ea40c49ccbe001c05b70e00190dfc79825feb005b645c48a7003a84fc6f7865001c05b70e002f008f000f008f003a4449404b7c002b1f1cdf1a632eaaab40d1c263edf49faf749010a1003a64df2ab5bb000b7f5c764cbc14f9669b88837ca1490cca17c31607002b1f1cdf1a632eaaab40d1c263edf49faf749010a1001b"; - assertEq(solady.LibZip.cdCompress(data), expected); -}); - -test("LibZip: Calldata decompress on invalid input", function() { - var data = "0xffffffff00ff"; - var expected = "0x0000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000"; - assertEq(solady.LibZip.cdDecompress(data), expected); -}); - -test("ERC1967Factory: ABI and address", function() { - function hashFnv32a(s) { - var h = 0x811c9dc5; - for (var i = 0; i < s.length; i++) { - h ^= s.charCodeAt(i); - h += (h << 1) + (h << 4) + (h << 7) + (h << 8) + (h << 24); - } - return h >>> 0; - } - assertEq(hashFnv32a(JSON.stringify(solady.ERC1967Factory.abi)), 1277805820); - assertEq(solady.ERC1967Factory.address, "0x0000000000006396FF2a80c067f99B3d2Ab4Df24"); -}); diff --git a/lib/solady/lib/ds-test/.gitignore b/lib/solady/lib/ds-test/.gitignore deleted file mode 100644 index 63f0b2c..0000000 --- a/lib/solady/lib/ds-test/.gitignore +++ /dev/null @@ -1,3 +0,0 @@ -/.dapple -/build -/out diff --git a/lib/solady/lib/ds-test/LICENSE b/lib/solady/lib/ds-test/LICENSE deleted file mode 100644 index 94a9ed0..0000000 --- a/lib/solady/lib/ds-test/LICENSE +++ /dev/null @@ -1,674 +0,0 @@ - GNU GENERAL PUBLIC LICENSE - Version 3, 29 June 2007 - - Copyright (C) 2007 Free Software Foundation, Inc. - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - - Preamble - - The GNU General Public License is a free, copyleft license for -software and other kinds of works. - - The licenses for most software and other practical works are designed -to take away your freedom to share and change the works. By contrast, -the GNU General Public License is intended to guarantee your freedom to -share and change all versions of a program--to make sure it remains free -software for all its users. We, the Free Software Foundation, use the -GNU General Public License for most of our software; it applies also to -any other work released this way by its authors. You can apply it to -your programs, too. - - When we speak of free software, we are referring to freedom, not -price. Our General Public Licenses are designed to make sure that you -have the freedom to distribute copies of free software (and charge for -them if you wish), that you receive source code or can get it if you -want it, that you can change the software or use pieces of it in new -free programs, and that you know you can do these things. - - To protect your rights, we need to prevent others from denying you -these rights or asking you to surrender the rights. Therefore, you have -certain responsibilities if you distribute copies of the software, or if -you modify it: responsibilities to respect the freedom of others. - - For example, if you distribute copies of such a program, whether -gratis or for a fee, you must pass on to the recipients the same -freedoms that you received. You must make sure that they, too, receive -or can get the source code. And you must show them these terms so they -know their rights. - - Developers that use the GNU GPL protect your rights with two steps: -(1) assert copyright on the software, and (2) offer you this License -giving you legal permission to copy, distribute and/or modify it. - - For the developers' and authors' protection, the GPL clearly explains -that there is no warranty for this free software. For both users' and -authors' sake, the GPL requires that modified versions be marked as -changed, so that their problems will not be attributed erroneously to -authors of previous versions. - - Some devices are designed to deny users access to install or run -modified versions of the software inside them, although the manufacturer -can do so. This is fundamentally incompatible with the aim of -protecting users' freedom to change the software. The systematic -pattern of such abuse occurs in the area of products for individuals to -use, which is precisely where it is most unacceptable. Therefore, we -have designed this version of the GPL to prohibit the practice for those -products. If such problems arise substantially in other domains, we -stand ready to extend this provision to those domains in future versions -of the GPL, as needed to protect the freedom of users. - - Finally, every program is threatened constantly by software patents. -States should not allow patents to restrict development and use of -software on general-purpose computers, but in those that do, we wish to -avoid the special danger that patents applied to a free program could -make it effectively proprietary. To prevent this, the GPL assures that -patents cannot be used to render the program non-free. - - The precise terms and conditions for copying, distribution and -modification follow. - - TERMS AND CONDITIONS - - 0. Definitions. - - "This License" refers to version 3 of the GNU General Public License. - - "Copyright" also means copyright-like laws that apply to other kinds of -works, such as semiconductor masks. - - "The Program" refers to any copyrightable work licensed under this -License. Each licensee is addressed as "you". "Licensees" and -"recipients" may be individuals or organizations. - - To "modify" a work means to copy from or adapt all or part of the work -in a fashion requiring copyright permission, other than the making of an -exact copy. The resulting work is called a "modified version" of the -earlier work or a work "based on" the earlier work. - - A "covered work" means either the unmodified Program or a work based -on the Program. - - To "propagate" a work means to do anything with it that, without -permission, would make you directly or secondarily liable for -infringement under applicable copyright law, except executing it on a -computer or modifying a private copy. Propagation includes copying, -distribution (with or without modification), making available to the -public, and in some countries other activities as well. - - To "convey" a work means any kind of propagation that enables other -parties to make or receive copies. Mere interaction with a user through -a computer network, with no transfer of a copy, is not conveying. - - An interactive user interface displays "Appropriate Legal Notices" -to the extent that it includes a convenient and prominently visible -feature that (1) displays an appropriate copyright notice, and (2) -tells the user that there is no warranty for the work (except to the -extent that warranties are provided), that licensees may convey the -work under this License, and how to view a copy of this License. If -the interface presents a list of user commands or options, such as a -menu, a prominent item in the list meets this criterion. - - 1. Source Code. - - The "source code" for a work means the preferred form of the work -for making modifications to it. "Object code" means any non-source -form of a work. - - A "Standard Interface" means an interface that either is an official -standard defined by a recognized standards body, or, in the case of -interfaces specified for a particular programming language, one that -is widely used among developers working in that language. - - The "System Libraries" of an executable work include anything, other -than the work as a whole, that (a) is included in the normal form of -packaging a Major Component, but which is not part of that Major -Component, and (b) serves only to enable use of the work with that -Major Component, or to implement a Standard Interface for which an -implementation is available to the public in source code form. A -"Major Component", in this context, means a major essential component -(kernel, window system, and so on) of the specific operating system -(if any) on which the executable work runs, or a compiler used to -produce the work, or an object code interpreter used to run it. - - The "Corresponding Source" for a work in object code form means all -the source code needed to generate, install, and (for an executable -work) run the object code and to modify the work, including scripts to -control those activities. However, it does not include the work's -System Libraries, or general-purpose tools or generally available free -programs which are used unmodified in performing those activities but -which are not part of the work. For example, Corresponding Source -includes interface definition files associated with source files for -the work, and the source code for shared libraries and dynamically -linked subprograms that the work is specifically designed to require, -such as by intimate data communication or control flow between those -subprograms and other parts of the work. - - The Corresponding Source need not include anything that users -can regenerate automatically from other parts of the Corresponding -Source. - - The Corresponding Source for a work in source code form is that -same work. - - 2. Basic Permissions. - - All rights granted under this License are granted for the term of -copyright on the Program, and are irrevocable provided the stated -conditions are met. This License explicitly affirms your unlimited -permission to run the unmodified Program. The output from running a -covered work is covered by this License only if the output, given its -content, constitutes a covered work. This License acknowledges your -rights of fair use or other equivalent, as provided by copyright law. - - You may make, run and propagate covered works that you do not -convey, without conditions so long as your license otherwise remains -in force. You may convey covered works to others for the sole purpose -of having them make modifications exclusively for you, or provide you -with facilities for running those works, provided that you comply with -the terms of this License in conveying all material for which you do -not control copyright. Those thus making or running the covered works -for you must do so exclusively on your behalf, under your direction -and control, on terms that prohibit them from making any copies of -your copyrighted material outside their relationship with you. - - Conveying under any other circumstances is permitted solely under -the conditions stated below. Sublicensing is not allowed; section 10 -makes it unnecessary. - - 3. Protecting Users' Legal Rights From Anti-Circumvention Law. - - No covered work shall be deemed part of an effective technological -measure under any applicable law fulfilling obligations under article -11 of the WIPO copyright treaty adopted on 20 December 1996, or -similar laws prohibiting or restricting circumvention of such -measures. - - When you convey a covered work, you waive any legal power to forbid -circumvention of technological measures to the extent such circumvention -is effected by exercising rights under this License with respect to -the covered work, and you disclaim any intention to limit operation or -modification of the work as a means of enforcing, against the work's -users, your or third parties' legal rights to forbid circumvention of -technological measures. - - 4. Conveying Verbatim Copies. - - You may convey verbatim copies of the Program's source code as you -receive it, in any medium, provided that you conspicuously and -appropriately publish on each copy an appropriate copyright notice; -keep intact all notices stating that this License and any -non-permissive terms added in accord with section 7 apply to the code; -keep intact all notices of the absence of any warranty; and give all -recipients a copy of this License along with the Program. - - You may charge any price or no price for each copy that you convey, -and you may offer support or warranty protection for a fee. - - 5. Conveying Modified Source Versions. - - You may convey a work based on the Program, or the modifications to -produce it from the Program, in the form of source code under the -terms of section 4, provided that you also meet all of these conditions: - - a) The work must carry prominent notices stating that you modified - it, and giving a relevant date. - - b) The work must carry prominent notices stating that it is - released under this License and any conditions added under section - 7. This requirement modifies the requirement in section 4 to - "keep intact all notices". - - c) You must license the entire work, as a whole, under this - License to anyone who comes into possession of a copy. This - License will therefore apply, along with any applicable section 7 - additional terms, to the whole of the work, and all its parts, - regardless of how they are packaged. This License gives no - permission to license the work in any other way, but it does not - invalidate such permission if you have separately received it. - - d) If the work has interactive user interfaces, each must display - Appropriate Legal Notices; however, if the Program has interactive - interfaces that do not display Appropriate Legal Notices, your - work need not make them do so. - - A compilation of a covered work with other separate and independent -works, which are not by their nature extensions of the covered work, -and which are not combined with it such as to form a larger program, -in or on a volume of a storage or distribution medium, is called an -"aggregate" if the compilation and its resulting copyright are not -used to limit the access or legal rights of the compilation's users -beyond what the individual works permit. Inclusion of a covered work -in an aggregate does not cause this License to apply to the other -parts of the aggregate. - - 6. Conveying Non-Source Forms. - - You may convey a covered work in object code form under the terms -of sections 4 and 5, provided that you also convey the -machine-readable Corresponding Source under the terms of this License, -in one of these ways: - - a) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by the - Corresponding Source fixed on a durable physical medium - customarily used for software interchange. - - b) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by a - written offer, valid for at least three years and valid for as - long as you offer spare parts or customer support for that product - model, to give anyone who possesses the object code either (1) a - copy of the Corresponding Source for all the software in the - product that is covered by this License, on a durable physical - medium customarily used for software interchange, for a price no - more than your reasonable cost of physically performing this - conveying of source, or (2) access to copy the - Corresponding Source from a network server at no charge. - - c) Convey individual copies of the object code with a copy of the - written offer to provide the Corresponding Source. This - alternative is allowed only occasionally and noncommercially, and - only if you received the object code with such an offer, in accord - with subsection 6b. - - d) Convey the object code by offering access from a designated - place (gratis or for a charge), and offer equivalent access to the - Corresponding Source in the same way through the same place at no - further charge. You need not require recipients to copy the - Corresponding Source along with the object code. If the place to - copy the object code is a network server, the Corresponding Source - may be on a different server (operated by you or a third party) - that supports equivalent copying facilities, provided you maintain - clear directions next to the object code saying where to find the - Corresponding Source. Regardless of what server hosts the - Corresponding Source, you remain obligated to ensure that it is - available for as long as needed to satisfy these requirements. - - e) Convey the object code using peer-to-peer transmission, provided - you inform other peers where the object code and Corresponding - Source of the work are being offered to the general public at no - charge under subsection 6d. - - A separable portion of the object code, whose source code is excluded -from the Corresponding Source as a System Library, need not be -included in conveying the object code work. - - A "User Product" is either (1) a "consumer product", which means any -tangible personal property which is normally used for personal, family, -or household purposes, or (2) anything designed or sold for incorporation -into a dwelling. In determining whether a product is a consumer product, -doubtful cases shall be resolved in favor of coverage. For a particular -product received by a particular user, "normally used" refers to a -typical or common use of that class of product, regardless of the status -of the particular user or of the way in which the particular user -actually uses, or expects or is expected to use, the product. A product -is a consumer product regardless of whether the product has substantial -commercial, industrial or non-consumer uses, unless such uses represent -the only significant mode of use of the product. - - "Installation Information" for a User Product means any methods, -procedures, authorization keys, or other information required to install -and execute modified versions of a covered work in that User Product from -a modified version of its Corresponding Source. The information must -suffice to ensure that the continued functioning of the modified object -code is in no case prevented or interfered with solely because -modification has been made. - - If you convey an object code work under this section in, or with, or -specifically for use in, a User Product, and the conveying occurs as -part of a transaction in which the right of possession and use of the -User Product is transferred to the recipient in perpetuity or for a -fixed term (regardless of how the transaction is characterized), the -Corresponding Source conveyed under this section must be accompanied -by the Installation Information. But this requirement does not apply -if neither you nor any third party retains the ability to install -modified object code on the User Product (for example, the work has -been installed in ROM). - - The requirement to provide Installation Information does not include a -requirement to continue to provide support service, warranty, or updates -for a work that has been modified or installed by the recipient, or for -the User Product in which it has been modified or installed. Access to a -network may be denied when the modification itself materially and -adversely affects the operation of the network or violates the rules and -protocols for communication across the network. - - Corresponding Source conveyed, and Installation Information provided, -in accord with this section must be in a format that is publicly -documented (and with an implementation available to the public in -source code form), and must require no special password or key for -unpacking, reading or copying. - - 7. Additional Terms. - - "Additional permissions" are terms that supplement the terms of this -License by making exceptions from one or more of its conditions. -Additional permissions that are applicable to the entire Program shall -be treated as though they were included in this License, to the extent -that they are valid under applicable law. If additional permissions -apply only to part of the Program, that part may be used separately -under those permissions, but the entire Program remains governed by -this License without regard to the additional permissions. - - When you convey a copy of a covered work, you may at your option -remove any additional permissions from that copy, or from any part of -it. (Additional permissions may be written to require their own -removal in certain cases when you modify the work.) You may place -additional permissions on material, added by you to a covered work, -for which you have or can give appropriate copyright permission. - - Notwithstanding any other provision of this License, for material you -add to a covered work, you may (if authorized by the copyright holders of -that material) supplement the terms of this License with terms: - - a) Disclaiming warranty or limiting liability differently from the - terms of sections 15 and 16 of this License; or - - b) Requiring preservation of specified reasonable legal notices or - author attributions in that material or in the Appropriate Legal - Notices displayed by works containing it; or - - c) Prohibiting misrepresentation of the origin of that material, or - requiring that modified versions of such material be marked in - reasonable ways as different from the original version; or - - d) Limiting the use for publicity purposes of names of licensors or - authors of the material; or - - e) Declining to grant rights under trademark law for use of some - trade names, trademarks, or service marks; or - - f) Requiring indemnification of licensors and authors of that - material by anyone who conveys the material (or modified versions of - it) with contractual assumptions of liability to the recipient, for - any liability that these contractual assumptions directly impose on - those licensors and authors. - - All other non-permissive additional terms are considered "further -restrictions" within the meaning of section 10. If the Program as you -received it, or any part of it, contains a notice stating that it is -governed by this License along with a term that is a further -restriction, you may remove that term. If a license document contains -a further restriction but permits relicensing or conveying under this -License, you may add to a covered work material governed by the terms -of that license document, provided that the further restriction does -not survive such relicensing or conveying. - - If you add terms to a covered work in accord with this section, you -must place, in the relevant source files, a statement of the -additional terms that apply to those files, or a notice indicating -where to find the applicable terms. - - Additional terms, permissive or non-permissive, may be stated in the -form of a separately written license, or stated as exceptions; -the above requirements apply either way. - - 8. Termination. - - You may not propagate or modify a covered work except as expressly -provided under this License. Any attempt otherwise to propagate or -modify it is void, and will automatically terminate your rights under -this License (including any patent licenses granted under the third -paragraph of section 11). - - However, if you cease all violation of this License, then your -license from a particular copyright holder is reinstated (a) -provisionally, unless and until the copyright holder explicitly and -finally terminates your license, and (b) permanently, if the copyright -holder fails to notify you of the violation by some reasonable means -prior to 60 days after the cessation. - - Moreover, your license from a particular copyright holder is -reinstated permanently if the copyright holder notifies you of the -violation by some reasonable means, this is the first time you have -received notice of violation of this License (for any work) from that -copyright holder, and you cure the violation prior to 30 days after -your receipt of the notice. - - Termination of your rights under this section does not terminate the -licenses of parties who have received copies or rights from you under -this License. If your rights have been terminated and not permanently -reinstated, you do not qualify to receive new licenses for the same -material under section 10. - - 9. Acceptance Not Required for Having Copies. - - You are not required to accept this License in order to receive or -run a copy of the Program. Ancillary propagation of a covered work -occurring solely as a consequence of using peer-to-peer transmission -to receive a copy likewise does not require acceptance. However, -nothing other than this License grants you permission to propagate or -modify any covered work. These actions infringe copyright if you do -not accept this License. Therefore, by modifying or propagating a -covered work, you indicate your acceptance of this License to do so. - - 10. Automatic Licensing of Downstream Recipients. - - Each time you convey a covered work, the recipient automatically -receives a license from the original licensors, to run, modify and -propagate that work, subject to this License. You are not responsible -for enforcing compliance by third parties with this License. - - An "entity transaction" is a transaction transferring control of an -organization, or substantially all assets of one, or subdividing an -organization, or merging organizations. If propagation of a covered -work results from an entity transaction, each party to that -transaction who receives a copy of the work also receives whatever -licenses to the work the party's predecessor in interest had or could -give under the previous paragraph, plus a right to possession of the -Corresponding Source of the work from the predecessor in interest, if -the predecessor has it or can get it with reasonable efforts. - - You may not impose any further restrictions on the exercise of the -rights granted or affirmed under this License. For example, you may -not impose a license fee, royalty, or other charge for exercise of -rights granted under this License, and you may not initiate litigation -(including a cross-claim or counterclaim in a lawsuit) alleging that -any patent claim is infringed by making, using, selling, offering for -sale, or importing the Program or any portion of it. - - 11. Patents. - - A "contributor" is a copyright holder who authorizes use under this -License of the Program or a work on which the Program is based. The -work thus licensed is called the contributor's "contributor version". - - A contributor's "essential patent claims" are all patent claims -owned or controlled by the contributor, whether already acquired or -hereafter acquired, that would be infringed by some manner, permitted -by this License, of making, using, or selling its contributor version, -but do not include claims that would be infringed only as a -consequence of further modification of the contributor version. For -purposes of this definition, "control" includes the right to grant -patent sublicenses in a manner consistent with the requirements of -this License. - - Each contributor grants you a non-exclusive, worldwide, royalty-free -patent license under the contributor's essential patent claims, to -make, use, sell, offer for sale, import and otherwise run, modify and -propagate the contents of its contributor version. - - In the following three paragraphs, a "patent license" is any express -agreement or commitment, however denominated, not to enforce a patent -(such as an express permission to practice a patent or covenant not to -sue for patent infringement). To "grant" such a patent license to a -party means to make such an agreement or commitment not to enforce a -patent against the party. - - If you convey a covered work, knowingly relying on a patent license, -and the Corresponding Source of the work is not available for anyone -to copy, free of charge and under the terms of this License, through a -publicly available network server or other readily accessible means, -then you must either (1) cause the Corresponding Source to be so -available, or (2) arrange to deprive yourself of the benefit of the -patent license for this particular work, or (3) arrange, in a manner -consistent with the requirements of this License, to extend the patent -license to downstream recipients. "Knowingly relying" means you have -actual knowledge that, but for the patent license, your conveying the -covered work in a country, or your recipient's use of the covered work -in a country, would infringe one or more identifiable patents in that -country that you have reason to believe are valid. - - If, pursuant to or in connection with a single transaction or -arrangement, you convey, or propagate by procuring conveyance of, a -covered work, and grant a patent license to some of the parties -receiving the covered work authorizing them to use, propagate, modify -or convey a specific copy of the covered work, then the patent license -you grant is automatically extended to all recipients of the covered -work and works based on it. - - A patent license is "discriminatory" if it does not include within -the scope of its coverage, prohibits the exercise of, or is -conditioned on the non-exercise of one or more of the rights that are -specifically granted under this License. You may not convey a covered -work if you are a party to an arrangement with a third party that is -in the business of distributing software, under which you make payment -to the third party based on the extent of your activity of conveying -the work, and under which the third party grants, to any of the -parties who would receive the covered work from you, a discriminatory -patent license (a) in connection with copies of the covered work -conveyed by you (or copies made from those copies), or (b) primarily -for and in connection with specific products or compilations that -contain the covered work, unless you entered into that arrangement, -or that patent license was granted, prior to 28 March 2007. - - Nothing in this License shall be construed as excluding or limiting -any implied license or other defenses to infringement that may -otherwise be available to you under applicable patent law. - - 12. No Surrender of Others' Freedom. - - If conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot convey a -covered work so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you may -not convey it at all. For example, if you agree to terms that obligate you -to collect a royalty for further conveying from those to whom you convey -the Program, the only way you could satisfy both those terms and this -License would be to refrain entirely from conveying the Program. - - 13. Use with the GNU Affero General Public License. - - Notwithstanding any other provision of this License, you have -permission to link or combine any covered work with a work licensed -under version 3 of the GNU Affero General Public License into a single -combined work, and to convey the resulting work. The terms of this -License will continue to apply to the part which is the covered work, -but the special requirements of the GNU Affero General Public License, -section 13, concerning interaction through a network will apply to the -combination as such. - - 14. Revised Versions of this License. - - The Free Software Foundation may publish revised and/or new versions of -the GNU General Public License from time to time. Such new versions will -be similar in spirit to the present version, but may differ in detail to -address new problems or concerns. - - Each version is given a distinguishing version number. If the -Program specifies that a certain numbered version of the GNU General -Public License "or any later version" applies to it, you have the -option of following the terms and conditions either of that numbered -version or of any later version published by the Free Software -Foundation. If the Program does not specify a version number of the -GNU General Public License, you may choose any version ever published -by the Free Software Foundation. - - If the Program specifies that a proxy can decide which future -versions of the GNU General Public License can be used, that proxy's -public statement of acceptance of a version permanently authorizes you -to choose that version for the Program. - - Later license versions may give you additional or different -permissions. However, no additional obligations are imposed on any -author or copyright holder as a result of your choosing to follow a -later version. - - 15. Disclaimer of Warranty. - - THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY -APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT -HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY -OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, -THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM -IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF -ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. Limitation of Liability. - - IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING -WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS -THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY -GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE -USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF -DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD -PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), -EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF -SUCH DAMAGES. - - 17. Interpretation of Sections 15 and 16. - - If the disclaimer of warranty and limitation of liability provided -above cannot be given local legal effect according to their terms, -reviewing courts shall apply local law that most closely approximates -an absolute waiver of all civil liability in connection with the -Program, unless a warranty or assumption of liability accompanies a -copy of the Program in return for a fee. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Programs - - If you develop a new program, and you want it to be of the greatest -possible use to the public, the best way to achieve this is to make it -free software which everyone can redistribute and change under these terms. - - To do so, attach the following notices to the program. It is safest -to attach them to the start of each source file to most effectively -state the exclusion of warranty; and each file should have at least -the "copyright" line and a pointer to where the full notice is found. - - - Copyright (C) - - This program is free software: you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation, either version 3 of the License, or - (at your option) any later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with this program. If not, see . - -Also add information on how to contact you by electronic and paper mail. - - If the program does terminal interaction, make it output a short -notice like this when it starts in an interactive mode: - - Copyright (C) - This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. - This is free software, and you are welcome to redistribute it - under certain conditions; type `show c' for details. - -The hypothetical commands `show w' and `show c' should show the appropriate -parts of the General Public License. Of course, your program's commands -might be different; for a GUI interface, you would use an "about box". - - You should also get your employer (if you work as a programmer) or school, -if any, to sign a "copyright disclaimer" for the program, if necessary. -For more information on this, and how to apply and follow the GNU GPL, see -. - - The GNU General Public License does not permit incorporating your program -into proprietary programs. If your program is a subroutine library, you -may consider it more useful to permit linking proprietary applications with -the library. If this is what you want to do, use the GNU Lesser General -Public License instead of this License. But first, please read -. diff --git a/lib/solady/lib/ds-test/Makefile b/lib/solady/lib/ds-test/Makefile deleted file mode 100644 index 661dac4..0000000 --- a/lib/solady/lib/ds-test/Makefile +++ /dev/null @@ -1,14 +0,0 @@ -all:; dapp build - -test: - -dapp --use solc:0.4.23 build - -dapp --use solc:0.4.26 build - -dapp --use solc:0.5.17 build - -dapp --use solc:0.6.12 build - -dapp --use solc:0.7.5 build - -demo: - DAPP_SRC=demo dapp --use solc:0.7.5 build - -hevm dapp-test --verbose 3 - -.PHONY: test demo diff --git a/lib/solady/lib/ds-test/default.nix b/lib/solady/lib/ds-test/default.nix deleted file mode 100644 index cf65419..0000000 --- a/lib/solady/lib/ds-test/default.nix +++ /dev/null @@ -1,4 +0,0 @@ -{ solidityPackage, dappsys }: solidityPackage { - name = "ds-test"; - src = ./src; -} diff --git a/lib/solady/lib/ds-test/demo/demo.sol b/lib/solady/lib/ds-test/demo/demo.sol deleted file mode 100644 index f3bb48e..0000000 --- a/lib/solady/lib/ds-test/demo/demo.sol +++ /dev/null @@ -1,222 +0,0 @@ -// SPDX-License-Identifier: GPL-3.0-or-later -pragma solidity >=0.5.0; - -import "../src/test.sol"; - -contract DemoTest is DSTest { - function test_this() public pure { - require(true); - } - function test_logs() public { - emit log("-- log(string)"); - emit log("a string"); - - emit log("-- log_named_uint(string, uint)"); - emit log_named_uint("uint", 512); - - emit log("-- log_named_int(string, int)"); - emit log_named_int("int", -512); - - emit log("-- log_named_address(string, address)"); - emit log_named_address("address", address(this)); - - emit log("-- log_named_bytes32(string, bytes32)"); - emit log_named_bytes32("bytes32", "a string"); - - emit log("-- log_named_bytes(string, bytes)"); - emit log_named_bytes("bytes", hex"cafefe"); - - emit log("-- log_named_string(string, string)"); - emit log_named_string("string", "a string"); - - emit log("-- log_named_decimal_uint(string, uint, uint)"); - emit log_named_decimal_uint("decimal uint", 1.0e18, 18); - - emit log("-- log_named_decimal_int(string, int, uint)"); - emit log_named_decimal_int("decimal int", -1.0e18, 18); - } - event log_old_named_uint(bytes32,uint); - function test_old_logs() public { - emit log_old_named_uint("key", 500); - emit log_named_bytes32("bkey", "val"); - } - function test_trace() public view { - this.echo("string 1", "string 2"); - } - function test_multiline() public { - emit log("a multiline\\nstring"); - emit log("a multiline string"); - emit log_bytes("a string"); - emit log_bytes("a multiline\nstring"); - emit log_bytes("a multiline\\nstring"); - emit logs(hex"0000"); - emit log_named_bytes("0x0000", hex"0000"); - emit logs(hex"ff"); - } - function echo(string memory s1, string memory s2) public pure - returns (string memory, string memory) - { - return (s1, s2); - } - - function prove_this(uint x) public { - emit log_named_uint("sym x", x); - assertGt(x + 1, 0); - } - - function test_logn() public { - assembly { - log0(0x01, 0x02) - log1(0x01, 0x02, 0x03) - log2(0x01, 0x02, 0x03, 0x04) - log3(0x01, 0x02, 0x03, 0x04, 0x05) - } - } - - event MyEvent(uint, uint indexed, uint, uint indexed); - function test_events() public { - emit MyEvent(1, 2, 3, 4); - } - - function test_asserts() public { - string memory err = "this test has failed!"; - emit log("## assertTrue(bool)\n"); - assertTrue(false); - emit log("\n"); - assertTrue(false, err); - - emit log("\n## assertEq(address,address)\n"); - assertEq(address(this), msg.sender); - emit log("\n"); - assertEq(address(this), msg.sender, err); - - emit log("\n## assertEq32(bytes32,bytes32)\n"); - assertEq32("bytes 1", "bytes 2"); - emit log("\n"); - assertEq32("bytes 1", "bytes 2", err); - - emit log("\n## assertEq(bytes32,bytes32)\n"); - assertEq32("bytes 1", "bytes 2"); - emit log("\n"); - assertEq32("bytes 1", "bytes 2", err); - - emit log("\n## assertEq(uint,uint)\n"); - assertEq(uint(0), 1); - emit log("\n"); - assertEq(uint(0), 1, err); - - emit log("\n## assertEq(int,int)\n"); - assertEq(-1, -2); - emit log("\n"); - assertEq(-1, -2, err); - - emit log("\n## assertEqDecimal(int,int,uint)\n"); - assertEqDecimal(-1.0e18, -1.1e18, 18); - emit log("\n"); - assertEqDecimal(-1.0e18, -1.1e18, 18, err); - - emit log("\n## assertEqDecimal(uint,uint,uint)\n"); - assertEqDecimal(uint(1.0e18), 1.1e18, 18); - emit log("\n"); - assertEqDecimal(uint(1.0e18), 1.1e18, 18, err); - - emit log("\n## assertGt(uint,uint)\n"); - assertGt(uint(0), 0); - emit log("\n"); - assertGt(uint(0), 0, err); - - emit log("\n## assertGt(int,int)\n"); - assertGt(-1, -1); - emit log("\n"); - assertGt(-1, -1, err); - - emit log("\n## assertGtDecimal(int,int,uint)\n"); - assertGtDecimal(-2.0e18, -1.1e18, 18); - emit log("\n"); - assertGtDecimal(-2.0e18, -1.1e18, 18, err); - - emit log("\n## assertGtDecimal(uint,uint,uint)\n"); - assertGtDecimal(uint(1.0e18), 1.1e18, 18); - emit log("\n"); - assertGtDecimal(uint(1.0e18), 1.1e18, 18, err); - - emit log("\n## assertGe(uint,uint)\n"); - assertGe(uint(0), 1); - emit log("\n"); - assertGe(uint(0), 1, err); - - emit log("\n## assertGe(int,int)\n"); - assertGe(-1, 0); - emit log("\n"); - assertGe(-1, 0, err); - - emit log("\n## assertGeDecimal(int,int,uint)\n"); - assertGeDecimal(-2.0e18, -1.1e18, 18); - emit log("\n"); - assertGeDecimal(-2.0e18, -1.1e18, 18, err); - - emit log("\n## assertGeDecimal(uint,uint,uint)\n"); - assertGeDecimal(uint(1.0e18), 1.1e18, 18); - emit log("\n"); - assertGeDecimal(uint(1.0e18), 1.1e18, 18, err); - - emit log("\n## assertLt(uint,uint)\n"); - assertLt(uint(0), 0); - emit log("\n"); - assertLt(uint(0), 0, err); - - emit log("\n## assertLt(int,int)\n"); - assertLt(-1, -1); - emit log("\n"); - assertLt(-1, -1, err); - - emit log("\n## assertLtDecimal(int,int,uint)\n"); - assertLtDecimal(-1.0e18, -1.1e18, 18); - emit log("\n"); - assertLtDecimal(-1.0e18, -1.1e18, 18, err); - - emit log("\n## assertLtDecimal(uint,uint,uint)\n"); - assertLtDecimal(uint(2.0e18), 1.1e18, 18); - emit log("\n"); - assertLtDecimal(uint(2.0e18), 1.1e18, 18, err); - - emit log("\n## assertLe(uint,uint)\n"); - assertLe(uint(1), 0); - emit log("\n"); - assertLe(uint(1), 0, err); - - emit log("\n## assertLe(int,int)\n"); - assertLe(0, -1); - emit log("\n"); - assertLe(0, -1, err); - - emit log("\n## assertLeDecimal(int,int,uint)\n"); - assertLeDecimal(-1.0e18, -1.1e18, 18); - emit log("\n"); - assertLeDecimal(-1.0e18, -1.1e18, 18, err); - - emit log("\n## assertLeDecimal(uint,uint,uint)\n"); - assertLeDecimal(uint(2.0e18), 1.1e18, 18); - emit log("\n"); - assertLeDecimal(uint(2.0e18), 1.1e18, 18, err); - - emit log("\n## assertEq(string,string)\n"); - string memory s1 = "string 1"; - string memory s2 = "string 2"; - assertEq(s1, s2); - emit log("\n"); - assertEq(s1, s2, err); - - emit log("\n## assertEq0(bytes,bytes)\n"); - assertEq0(hex"abcdef01", hex"abcdef02"); - emit log("\n"); - assertEq0(hex"abcdef01", hex"abcdef02", err); - } -} - -contract DemoTestWithSetUp { - function setUp() public { - } - function test_pass() public pure { - } -} diff --git a/lib/solady/lib/ds-test/src/test.sol b/lib/solady/lib/ds-test/src/test.sol deleted file mode 100644 index 515a3bd..0000000 --- a/lib/solady/lib/ds-test/src/test.sol +++ /dev/null @@ -1,469 +0,0 @@ -// SPDX-License-Identifier: GPL-3.0-or-later - -// This program is free software: you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation, either version 3 of the License, or -// (at your option) any later version. - -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. - -// You should have received a copy of the GNU General Public License -// along with this program. If not, see . - -pragma solidity >=0.5.0; - -contract DSTest { - event log (string); - event logs (bytes); - - event log_address (address); - event log_bytes32 (bytes32); - event log_int (int); - event log_uint (uint); - event log_bytes (bytes); - event log_string (string); - - event log_named_address (string key, address val); - event log_named_bytes32 (string key, bytes32 val); - event log_named_decimal_int (string key, int val, uint decimals); - event log_named_decimal_uint (string key, uint val, uint decimals); - event log_named_int (string key, int val); - event log_named_uint (string key, uint val); - event log_named_bytes (string key, bytes val); - event log_named_string (string key, string val); - - bool public IS_TEST = true; - bool private _failed; - - address constant HEVM_ADDRESS = - address(bytes20(uint160(uint256(keccak256('hevm cheat code'))))); - - modifier mayRevert() { _; } - modifier testopts(string memory) { _; } - - function failed() public returns (bool) { - if (_failed) { - return _failed; - } else { - bool globalFailed = false; - if (hasHEVMContext()) { - (, bytes memory retdata) = HEVM_ADDRESS.call( - abi.encodePacked( - bytes4(keccak256("load(address,bytes32)")), - abi.encode(HEVM_ADDRESS, bytes32("failed")) - ) - ); - globalFailed = abi.decode(retdata, (bool)); - } - return globalFailed; - } - } - - function fail() internal { - if (hasHEVMContext()) { - (bool status, ) = HEVM_ADDRESS.call( - abi.encodePacked( - bytes4(keccak256("store(address,bytes32,bytes32)")), - abi.encode(HEVM_ADDRESS, bytes32("failed"), bytes32(uint256(0x01))) - ) - ); - status; // Silence compiler warnings - } - _failed = true; - } - - function hasHEVMContext() internal view returns (bool) { - uint256 hevmCodeSize = 0; - assembly { - hevmCodeSize := extcodesize(0x7109709ECfa91a80626fF3989D68f67F5b1DD12D) - } - return hevmCodeSize > 0; - } - - modifier logs_gas() { - uint startGas = gasleft(); - _; - uint endGas = gasleft(); - emit log_named_uint("gas", startGas - endGas); - } - - function assertTrue(bool condition) internal { - if (!condition) { - emit log("Error: Assertion Failed"); - fail(); - } - } - - function assertTrue(bool condition, string memory err) internal { - if (!condition) { - emit log_named_string("Error", err); - assertTrue(condition); - } - } - - function assertEq(address a, address b) internal { - if (a != b) { - emit log("Error: a == b not satisfied [address]"); - emit log_named_address(" Expected", b); - emit log_named_address(" Actual", a); - fail(); - } - } - function assertEq(address a, address b, string memory err) internal { - if (a != b) { - emit log_named_string ("Error", err); - assertEq(a, b); - } - } - - function assertEq(bytes32 a, bytes32 b) internal { - if (a != b) { - emit log("Error: a == b not satisfied [bytes32]"); - emit log_named_bytes32(" Expected", b); - emit log_named_bytes32(" Actual", a); - fail(); - } - } - function assertEq(bytes32 a, bytes32 b, string memory err) internal { - if (a != b) { - emit log_named_string ("Error", err); - assertEq(a, b); - } - } - function assertEq32(bytes32 a, bytes32 b) internal { - assertEq(a, b); - } - function assertEq32(bytes32 a, bytes32 b, string memory err) internal { - assertEq(a, b, err); - } - - function assertEq(int a, int b) internal { - if (a != b) { - emit log("Error: a == b not satisfied [int]"); - emit log_named_int(" Expected", b); - emit log_named_int(" Actual", a); - fail(); - } - } - function assertEq(int a, int b, string memory err) internal { - if (a != b) { - emit log_named_string("Error", err); - assertEq(a, b); - } - } - function assertEq(uint a, uint b) internal { - if (a != b) { - emit log("Error: a == b not satisfied [uint]"); - emit log_named_uint(" Expected", b); - emit log_named_uint(" Actual", a); - fail(); - } - } - function assertEq(uint a, uint b, string memory err) internal { - if (a != b) { - emit log_named_string("Error", err); - assertEq(a, b); - } - } - function assertEqDecimal(int a, int b, uint decimals) internal { - if (a != b) { - emit log("Error: a == b not satisfied [decimal int]"); - emit log_named_decimal_int(" Expected", b, decimals); - emit log_named_decimal_int(" Actual", a, decimals); - fail(); - } - } - function assertEqDecimal(int a, int b, uint decimals, string memory err) internal { - if (a != b) { - emit log_named_string("Error", err); - assertEqDecimal(a, b, decimals); - } - } - function assertEqDecimal(uint a, uint b, uint decimals) internal { - if (a != b) { - emit log("Error: a == b not satisfied [decimal uint]"); - emit log_named_decimal_uint(" Expected", b, decimals); - emit log_named_decimal_uint(" Actual", a, decimals); - fail(); - } - } - function assertEqDecimal(uint a, uint b, uint decimals, string memory err) internal { - if (a != b) { - emit log_named_string("Error", err); - assertEqDecimal(a, b, decimals); - } - } - - function assertGt(uint a, uint b) internal { - if (a <= b) { - emit log("Error: a > b not satisfied [uint]"); - emit log_named_uint(" Value a", a); - emit log_named_uint(" Value b", b); - fail(); - } - } - function assertGt(uint a, uint b, string memory err) internal { - if (a <= b) { - emit log_named_string("Error", err); - assertGt(a, b); - } - } - function assertGt(int a, int b) internal { - if (a <= b) { - emit log("Error: a > b not satisfied [int]"); - emit log_named_int(" Value a", a); - emit log_named_int(" Value b", b); - fail(); - } - } - function assertGt(int a, int b, string memory err) internal { - if (a <= b) { - emit log_named_string("Error", err); - assertGt(a, b); - } - } - function assertGtDecimal(int a, int b, uint decimals) internal { - if (a <= b) { - emit log("Error: a > b not satisfied [decimal int]"); - emit log_named_decimal_int(" Value a", a, decimals); - emit log_named_decimal_int(" Value b", b, decimals); - fail(); - } - } - function assertGtDecimal(int a, int b, uint decimals, string memory err) internal { - if (a <= b) { - emit log_named_string("Error", err); - assertGtDecimal(a, b, decimals); - } - } - function assertGtDecimal(uint a, uint b, uint decimals) internal { - if (a <= b) { - emit log("Error: a > b not satisfied [decimal uint]"); - emit log_named_decimal_uint(" Value a", a, decimals); - emit log_named_decimal_uint(" Value b", b, decimals); - fail(); - } - } - function assertGtDecimal(uint a, uint b, uint decimals, string memory err) internal { - if (a <= b) { - emit log_named_string("Error", err); - assertGtDecimal(a, b, decimals); - } - } - - function assertGe(uint a, uint b) internal { - if (a < b) { - emit log("Error: a >= b not satisfied [uint]"); - emit log_named_uint(" Value a", a); - emit log_named_uint(" Value b", b); - fail(); - } - } - function assertGe(uint a, uint b, string memory err) internal { - if (a < b) { - emit log_named_string("Error", err); - assertGe(a, b); - } - } - function assertGe(int a, int b) internal { - if (a < b) { - emit log("Error: a >= b not satisfied [int]"); - emit log_named_int(" Value a", a); - emit log_named_int(" Value b", b); - fail(); - } - } - function assertGe(int a, int b, string memory err) internal { - if (a < b) { - emit log_named_string("Error", err); - assertGe(a, b); - } - } - function assertGeDecimal(int a, int b, uint decimals) internal { - if (a < b) { - emit log("Error: a >= b not satisfied [decimal int]"); - emit log_named_decimal_int(" Value a", a, decimals); - emit log_named_decimal_int(" Value b", b, decimals); - fail(); - } - } - function assertGeDecimal(int a, int b, uint decimals, string memory err) internal { - if (a < b) { - emit log_named_string("Error", err); - assertGeDecimal(a, b, decimals); - } - } - function assertGeDecimal(uint a, uint b, uint decimals) internal { - if (a < b) { - emit log("Error: a >= b not satisfied [decimal uint]"); - emit log_named_decimal_uint(" Value a", a, decimals); - emit log_named_decimal_uint(" Value b", b, decimals); - fail(); - } - } - function assertGeDecimal(uint a, uint b, uint decimals, string memory err) internal { - if (a < b) { - emit log_named_string("Error", err); - assertGeDecimal(a, b, decimals); - } - } - - function assertLt(uint a, uint b) internal { - if (a >= b) { - emit log("Error: a < b not satisfied [uint]"); - emit log_named_uint(" Value a", a); - emit log_named_uint(" Value b", b); - fail(); - } - } - function assertLt(uint a, uint b, string memory err) internal { - if (a >= b) { - emit log_named_string("Error", err); - assertLt(a, b); - } - } - function assertLt(int a, int b) internal { - if (a >= b) { - emit log("Error: a < b not satisfied [int]"); - emit log_named_int(" Value a", a); - emit log_named_int(" Value b", b); - fail(); - } - } - function assertLt(int a, int b, string memory err) internal { - if (a >= b) { - emit log_named_string("Error", err); - assertLt(a, b); - } - } - function assertLtDecimal(int a, int b, uint decimals) internal { - if (a >= b) { - emit log("Error: a < b not satisfied [decimal int]"); - emit log_named_decimal_int(" Value a", a, decimals); - emit log_named_decimal_int(" Value b", b, decimals); - fail(); - } - } - function assertLtDecimal(int a, int b, uint decimals, string memory err) internal { - if (a >= b) { - emit log_named_string("Error", err); - assertLtDecimal(a, b, decimals); - } - } - function assertLtDecimal(uint a, uint b, uint decimals) internal { - if (a >= b) { - emit log("Error: a < b not satisfied [decimal uint]"); - emit log_named_decimal_uint(" Value a", a, decimals); - emit log_named_decimal_uint(" Value b", b, decimals); - fail(); - } - } - function assertLtDecimal(uint a, uint b, uint decimals, string memory err) internal { - if (a >= b) { - emit log_named_string("Error", err); - assertLtDecimal(a, b, decimals); - } - } - - function assertLe(uint a, uint b) internal { - if (a > b) { - emit log("Error: a <= b not satisfied [uint]"); - emit log_named_uint(" Value a", a); - emit log_named_uint(" Value b", b); - fail(); - } - } - function assertLe(uint a, uint b, string memory err) internal { - if (a > b) { - emit log_named_string("Error", err); - assertLe(a, b); - } - } - function assertLe(int a, int b) internal { - if (a > b) { - emit log("Error: a <= b not satisfied [int]"); - emit log_named_int(" Value a", a); - emit log_named_int(" Value b", b); - fail(); - } - } - function assertLe(int a, int b, string memory err) internal { - if (a > b) { - emit log_named_string("Error", err); - assertLe(a, b); - } - } - function assertLeDecimal(int a, int b, uint decimals) internal { - if (a > b) { - emit log("Error: a <= b not satisfied [decimal int]"); - emit log_named_decimal_int(" Value a", a, decimals); - emit log_named_decimal_int(" Value b", b, decimals); - fail(); - } - } - function assertLeDecimal(int a, int b, uint decimals, string memory err) internal { - if (a > b) { - emit log_named_string("Error", err); - assertLeDecimal(a, b, decimals); - } - } - function assertLeDecimal(uint a, uint b, uint decimals) internal { - if (a > b) { - emit log("Error: a <= b not satisfied [decimal uint]"); - emit log_named_decimal_uint(" Value a", a, decimals); - emit log_named_decimal_uint(" Value b", b, decimals); - fail(); - } - } - function assertLeDecimal(uint a, uint b, uint decimals, string memory err) internal { - if (a > b) { - emit log_named_string("Error", err); - assertGeDecimal(a, b, decimals); - } - } - - function assertEq(string memory a, string memory b) internal { - if (keccak256(abi.encodePacked(a)) != keccak256(abi.encodePacked(b))) { - emit log("Error: a == b not satisfied [string]"); - emit log_named_string(" Expected", b); - emit log_named_string(" Actual", a); - fail(); - } - } - function assertEq(string memory a, string memory b, string memory err) internal { - if (keccak256(abi.encodePacked(a)) != keccak256(abi.encodePacked(b))) { - emit log_named_string("Error", err); - assertEq(a, b); - } - } - - function checkEq0(bytes memory a, bytes memory b) internal pure returns (bool ok) { - ok = true; - if (a.length == b.length) { - for (uint i = 0; i < a.length; i++) { - if (a[i] != b[i]) { - ok = false; - } - } - } else { - ok = false; - } - } - function assertEq0(bytes memory a, bytes memory b) internal { - if (!checkEq0(a, b)) { - emit log("Error: a == b not satisfied [bytes]"); - emit log_named_bytes(" Expected", b); - emit log_named_bytes(" Actual", a); - fail(); - } - } - function assertEq0(bytes memory a, bytes memory b, string memory err) internal { - if (!checkEq0(a, b)) { - emit log_named_string("Error", err); - assertEq0(a, b); - } - } -} diff --git a/lib/solady/logo.svg b/lib/solady/logo.svg deleted file mode 100644 index 8589e73..0000000 --- a/lib/solady/logo.svg +++ /dev/null @@ -1 +0,0 @@ -solady diff --git a/lib/solady/package-lock.json b/lib/solady/package-lock.json deleted file mode 100644 index e58c913..0000000 --- a/lib/solady/package-lock.json +++ /dev/null @@ -1,13 +0,0 @@ -{ - "name": "solady", - "version": "0.0.55", - "lockfileVersion": 2, - "requires": true, - "packages": { - "": { - "name": "solady", - "version": "0.0.55", - "license": "MIT" - } - } -} diff --git a/lib/solady/package.json b/lib/solady/package.json deleted file mode 100644 index f471f5d..0000000 --- a/lib/solady/package.json +++ /dev/null @@ -1,17 +0,0 @@ -{ - "name": "solady", - "license": "MIT", - "version": "0.0.145", - "description": "Optimized Solidity snippets.", - "files": [ - "src/**/*.sol", - "js/**/*" - ], - "main": "js/solady.js", - "module": "js/solady.js", - "types": "js/solady.d.ts", - "repository": { - "type": "git", - "url": "git+https://github.com/vectorized/solady.git" - } -} diff --git a/lib/solady/src/Milady.sol b/lib/solady/src/Milady.sol deleted file mode 100644 index 75748c2..0000000 --- a/lib/solady/src/Milady.sol +++ /dev/null @@ -1,55 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./accounts/Receiver.sol"; -import "./accounts/ERC1271.sol"; -import "./accounts/ERC4337.sol"; -import "./accounts/ERC4337Factory.sol"; -import "./accounts/ERC6551.sol"; -import "./accounts/ERC6551Proxy.sol"; -import "./auth/Ownable.sol"; -import "./auth/OwnableRoles.sol"; -import "./tokens/WETH.sol"; -import "./tokens/ERC20.sol"; -import "./tokens/ERC4626.sol"; -import "./tokens/ERC721.sol"; -import "./tokens/ERC2981.sol"; -import "./tokens/ERC1155.sol"; -import "./tokens/ERC6909.sol"; -import "./utils/Base64.sol"; -import "./utils/CREATE3.sol"; -import "./utils/Clone.sol"; -import "./utils/DateTimeLib.sol"; -import "./utils/DynamicBufferLib.sol"; -import "./utils/ECDSA.sol"; -import "./utils/EIP712.sol"; -import "./utils/ERC1967Factory.sol"; -import "./utils/ERC1967FactoryConstants.sol"; -import "./utils/FixedPointMathLib.sol"; -import "./utils/GasBurnerLib.sol"; -import "./utils/JSONParserLib.sol"; -import "./utils/LibBit.sol"; -import "./utils/LibBitmap.sol"; -import "./utils/LibMap.sol"; -import "./utils/LibClone.sol"; -import "./utils/LibPRNG.sol"; -import "./utils/LibRLP.sol"; -import "./utils/LibSort.sol"; -import "./utils/LibString.sol"; -import "./utils/LibZip.sol"; -import "./utils/MerkleProofLib.sol"; -import "./utils/MetadataReaderLib.sol"; -import "./utils/MinHeapLib.sol"; -import "./utils/Multicallable.sol"; -import "./utils/RedBlackTreeLib.sol"; -import "./utils/SSTORE2.sol"; -import "./utils/SafeCastLib.sol"; -import "./utils/SafeTransferLib.sol"; -import "./utils/SignatureCheckerLib.sol"; -import "./utils/UUPSUpgradeable.sol"; - -library Milady { - string internal constant WEBSITE = "https://miladymaker.net"; - - address internal constant CONTRACT = 0x5Af0D9827E0c53E4799BB226655A1de152A425a5; -} diff --git a/lib/solady/src/accounts/ERC1271.sol b/lib/solady/src/accounts/ERC1271.sol deleted file mode 100644 index 61e4aeb..0000000 --- a/lib/solady/src/accounts/ERC1271.sol +++ /dev/null @@ -1,104 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {EIP712} from "../utils/EIP712.sol"; -import {SignatureCheckerLib} from "../utils/SignatureCheckerLib.sol"; - -/// @notice ERC1271 mixin with nested EIP-712 approach. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/accounts/ERC1271.sol) -abstract contract ERC1271 is EIP712 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC1271 OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the ERC1271 signer. - /// Override to return the signer `isValidSignature` checks against. - function _erc1271Signer() internal view virtual returns (address); - - /// @dev Validates the signature with ERC1271 return, - /// so that this account can also be used as a signer. - /// - /// This implementation uses ECDSA recovery. It also uses a nested EIP-712 approach to - /// prevent signature replays when a single EOA owns multiple smart contract accounts, - /// while still enabling wallet UIs (e.g. Metamask) to show the EIP-712 values. - /// - /// For the nested EIP-712 workflow, the final hash will be: - /// ``` - /// keccak256(\x19\x01 || DOMAIN_SEP_A || - /// hashStruct(Parent({ - /// childHash: keccak256(\x19\x01 || DOMAIN_SEP_B || hashStruct(originalStruct)), - /// child: hashStruct(originalStruct) - /// })) - /// ) - /// ``` - /// where `||` denotes the concatenation operator for bytes. - /// The signature will be `r || s || v || PARENT_TYPEHASH || DOMAIN_SEP_B || child`. - /// - /// The `DOMAIN_SEP_B` and `child` will be used to verify if `childHash` is indeed correct. - /// - /// For the `personal_sign` workflow, the final hash will be: - /// ``` - /// keccak256(\x19\x01 || DOMAIN_SEP_A || - /// hashStruct(Parent({ - /// childHash: personalSign(someBytes) - /// })) - /// ) - /// ``` - /// where `||` denotes the concatenation operator for bytes. - /// The signature will be `r || s || v || PARENT_TYPEHASH`. - /// - /// For demo and typescript code, see: - /// - https://github.com/junomonster/nested-eip-712 - /// - https://github.com/frangio/eip712-wrapper-for-eip1271 - /// - /// Of course, if you are a wallet app maker and can update your app's UI at will, - /// you can choose a more minimalistic signature scheme like - /// `keccak256(abi.encode(address(this), hash))` instead of all these acrobatics. - /// All these are just for widespead out-of-the-box compatibility with other wallet apps. - /// - /// The `hash` parameter is the `childHash`. - function isValidSignature(bytes32 hash, bytes calldata signature) - public - view - virtual - returns (bytes4 result) - { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Cache the free memory pointer. - let o := add(signature.offset, sub(signature.length, 0x60)) - calldatacopy(0x00, o, 0x60) // Copy the `DOMAIN_SEP_B` and child's structHash. - mstore(0x00, 0x1901) // Store the "\x19\x01" prefix, overwriting 0x00. - for {} 1 {} { - // Use the nested EIP-712 workflow if the reconstructed childHash matches, - // and the signature is at least 96 bytes long. - if iszero(or(xor(keccak256(0x1e, 0x42), hash), lt(signature.length, 0x60))) { - // Truncate the `signature.length` by 3 words (96 bytes). - signature.length := sub(signature.length, 0x60) - mstore(0x00, calldataload(o)) // Store the `PARENT_TYPEHASH`. - mstore(0x20, hash) // Store the `childHash`. - // The child's structHash is already at 0x40. - hash := keccak256(0x00, 0x60) // Compute the parent's structHash. - break - } - // Else, use the `personal_sign` workflow. - // Truncate the `signature.length` by 1 word (32 bytes), until zero. - signature.length := mul(gt(signature.length, 0x20), sub(signature.length, 0x20)) - // The `PARENT_TYPEHASH` is already at 0x40. - mstore(0x60, hash) // Store the `childHash`. - hash := keccak256(0x40, 0x40) // Compute the parent's structHash. - mstore(0x60, 0) // Restore the zero pointer. - break - } - mstore(0x40, m) // Restore the free memory pointer. - } - bool success = SignatureCheckerLib.isValidSignatureNowCalldata( - _erc1271Signer(), _hashTypedData(hash), signature - ); - /// @solidity memory-safe-assembly - assembly { - // `success ? bytes4(keccak256("isValidSignature(bytes32,bytes)")) : 0xffffffff`. - result := shl(224, or(0x1626ba7e, sub(0, iszero(success)))) - } - } -} diff --git a/lib/solady/src/accounts/ERC4337.sol b/lib/solady/src/accounts/ERC4337.sol deleted file mode 100644 index 093a22e..0000000 --- a/lib/solady/src/accounts/ERC4337.sol +++ /dev/null @@ -1,386 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {Receiver} from "./Receiver.sol"; -import {LibZip} from "../utils/LibZip.sol"; -import {Ownable} from "../auth/Ownable.sol"; -import {UUPSUpgradeable} from "../utils/UUPSUpgradeable.sol"; -import {SignatureCheckerLib, ERC1271} from "../accounts/ERC1271.sol"; - -/// @notice Simple ERC4337 account implementation. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/accounts/ERC4337.sol) -/// @author Infinitism (https://github.com/eth-infinitism/account-abstraction/blob/develop/contracts/samples/SimpleAccount.sol) -/// -/// Recommended usage: -/// 1. Deploy the ERC4337 as an implementation contract, and verify it on Etherscan. -/// 2. Create a factory that uses `LibClone.deployERC1967` or -/// `LibClone.deployDeterministicERC1967` to clone the implementation. -/// See: `ERC4337Factory.sol`. -abstract contract ERC4337 is Ownable, UUPSUpgradeable, Receiver, ERC1271 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STRUCTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The ERC4337 user operation (userOp) struct. - struct UserOperation { - address sender; - uint256 nonce; - bytes initCode; - bytes callData; - uint256 callGasLimit; - uint256 verificationGasLimit; - uint256 preVerificationGas; - uint256 maxFeePerGas; - uint256 maxPriorityFeePerGas; - bytes paymasterAndData; - bytes signature; - } - - /// @dev Call struct for the `executeBatch` function. - struct Call { - address target; - uint256 value; - bytes data; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INITIALIZER */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Initializes the account with the owner. Can only be called once. - function initialize(address newOwner) public payable virtual { - _initializeOwner(newOwner); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ENTRY POINT */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the canonical ERC4337 EntryPoint contract. - /// Override this function to return a different EntryPoint. - function entryPoint() public view virtual returns (address) { - return 0x5FF137D4b0FDCD49DcA30c7CF57E578a026d2789; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* VALIDATION OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Validates the signature and nonce. - /// The EntryPoint will make the call to the recipient only if - /// this validation call returns successfully. - /// - /// Signature failure should be reported by returning 1 (see: `_validateSignature`). - /// This allows making a "simulation call" without a valid signature. - /// Other failures (e.g. nonce mismatch, or invalid signature format) - /// should still revert to signal failure. - function validateUserOp( - UserOperation calldata userOp, - bytes32 userOpHash, - uint256 missingAccountFunds - ) - external - payable - virtual - onlyEntryPoint - payPrefund(missingAccountFunds) - returns (uint256 validationData) - { - validationData = _validateSignature(userOp, userOpHash); - _validateNonce(userOp.nonce); - } - - /// @dev Validate `userOp.signature` for the `userOpHash`. - function _validateSignature(UserOperation calldata userOp, bytes32 userOpHash) - internal - virtual - returns (uint256 validationData) - { - bool success = SignatureCheckerLib.isValidSignatureNowCalldata( - owner(), SignatureCheckerLib.toEthSignedMessageHash(userOpHash), userOp.signature - ); - /// @solidity memory-safe-assembly - assembly { - // Returns 0 if the recovered address matches the owner. - // Else returns 1, which is equivalent to: - // `(success ? 0 : 1) | (uint256(validUntil) << 160) | (uint256(validAfter) << (160 + 48))` - // where `validUntil` is 0 (indefinite) and `validAfter` is 0. - validationData := iszero(success) - } - } - - /// @dev Override to validate the nonce of the userOp. - /// This method may validate the nonce requirement of this account. - /// e.g. - /// To limit the nonce to use sequenced userOps only (no "out of order" userOps): - /// `require(nonce < type(uint64).max)` - /// For a hypothetical account that *requires* the nonce to be out-of-order: - /// `require(nonce & type(uint64).max == 0)` - /// - /// The actual nonce uniqueness is managed by the EntryPoint, and thus no other - /// action is needed by the account itself. - function _validateNonce(uint256 nonce) internal virtual { - nonce = nonce; // Silence unused variable warning. - } - - /// @dev Sends to the EntryPoint (i.e. `msg.sender`) the missing funds for this transaction. - /// Subclass MAY override this modifier for better funds management. - /// (e.g. send to the EntryPoint more than the minimum required, so that in future transactions - /// it will not be required to send again) - /// - /// `missingAccountFunds` is the minimum value this modifier should send the EntryPoint, - /// which MAY be zero, in case there is enough deposit, or the userOp has a paymaster. - modifier payPrefund(uint256 missingAccountFunds) virtual { - _; - /// @solidity memory-safe-assembly - assembly { - if missingAccountFunds { - // Ignore failure (it's EntryPoint's job to verify, not the account's). - pop(call(gas(), caller(), missingAccountFunds, codesize(), 0x00, codesize(), 0x00)) - } - } - } - - /// @dev Requires that the caller is the EntryPoint. - modifier onlyEntryPoint() virtual { - if (msg.sender != entryPoint()) revert Unauthorized(); - _; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EXECUTION OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Execute a call from this account. - function execute(address target, uint256 value, bytes calldata data) - public - payable - virtual - onlyEntryPointOrOwner - returns (bytes memory result) - { - /// @solidity memory-safe-assembly - assembly { - result := mload(0x40) - calldatacopy(result, data.offset, data.length) - if iszero(call(gas(), target, value, result, data.length, codesize(), 0x00)) { - // Bubble up the revert if the call reverts. - returndatacopy(result, 0x00, returndatasize()) - revert(result, returndatasize()) - } - mstore(result, returndatasize()) // Store the length. - let o := add(result, 0x20) - returndatacopy(o, 0x00, returndatasize()) // Copy the returndata. - mstore(0x40, add(o, returndatasize())) // Allocate the memory. - } - } - - /// @dev Execute a sequence of calls from this account. - function executeBatch(Call[] calldata calls) - public - payable - virtual - onlyEntryPointOrOwner - returns (bytes[] memory results) - { - /// @solidity memory-safe-assembly - assembly { - results := mload(0x40) - mstore(results, calls.length) - let r := add(0x20, results) - let m := add(r, shl(5, calls.length)) - calldatacopy(r, calls.offset, shl(5, calls.length)) - for { let end := m } iszero(eq(r, end)) { r := add(r, 0x20) } { - let e := add(calls.offset, mload(r)) - let o := add(e, calldataload(add(e, 0x40))) - calldatacopy(m, add(o, 0x20), calldataload(o)) - // forgefmt: disable-next-item - if iszero(call(gas(), calldataload(e), calldataload(add(e, 0x20)), - m, calldataload(o), codesize(), 0x00)) { - // Bubble up the revert if the call reverts. - returndatacopy(m, 0x00, returndatasize()) - revert(m, returndatasize()) - } - mstore(r, m) // Append `m` into `results`. - mstore(m, returndatasize()) // Store the length, - let p := add(m, 0x20) - returndatacopy(p, 0x00, returndatasize()) // and copy the returndata. - m := add(p, returndatasize()) // Advance `m`. - } - mstore(0x40, m) // Allocate the memory. - } - } - - /// @dev Execute a delegatecall with `delegate` on this account. - function delegateExecute(address delegate, bytes calldata data) - public - payable - virtual - onlyEntryPointOrOwner - delegateExecuteGuard - returns (bytes memory result) - { - /// @solidity memory-safe-assembly - assembly { - result := mload(0x40) - calldatacopy(result, data.offset, data.length) - // Forwards the `data` to `delegate` via delegatecall. - if iszero(delegatecall(gas(), delegate, result, data.length, codesize(), 0x00)) { - // Bubble up the revert if the call reverts. - returndatacopy(result, 0x00, returndatasize()) - revert(result, returndatasize()) - } - mstore(result, returndatasize()) // Store the length. - let o := add(result, 0x20) - returndatacopy(o, 0x00, returndatasize()) // Copy the returndata. - mstore(0x40, add(o, returndatasize())) // Allocate the memory. - } - } - - /// @dev Ensures that the owner and implementation slots' values aren't changed. - /// You can override this modifier to ensure the sanctity of other storage slots too. - modifier delegateExecuteGuard() virtual { - bytes32 ownerSlotValue; - bytes32 implementationSlotValue; - /// @solidity memory-safe-assembly - assembly { - implementationSlotValue := sload(_ERC1967_IMPLEMENTATION_SLOT) - ownerSlotValue := sload(_OWNER_SLOT) - } - _; - /// @solidity memory-safe-assembly - assembly { - if iszero( - and( - eq(implementationSlotValue, sload(_ERC1967_IMPLEMENTATION_SLOT)), - eq(ownerSlotValue, sload(_OWNER_SLOT)) - ) - ) { revert(codesize(), 0x00) } - } - } - - /// @dev Requires that the caller is the EntryPoint, the owner, or the account itself. - modifier onlyEntryPointOrOwner() virtual { - if (msg.sender != entryPoint()) _checkOwner(); - _; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* DIRECT STORAGE OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the raw storage value at `storageSlot`. - function storageLoad(bytes32 storageSlot) public view virtual returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - result := sload(storageSlot) - } - } - - /// @dev Writes the raw storage value at `storageSlot`. - function storageStore(bytes32 storageSlot, bytes32 storageValue) - public - payable - virtual - onlyEntryPointOrOwner - storageStoreGuard(storageSlot) - { - /// @solidity memory-safe-assembly - assembly { - sstore(storageSlot, storageValue) - } - } - - /// @dev Ensures that the `storageSlot` is not prohibited for direct storage writes. - /// You can override this modifier to ensure the sanctity of other storage slots too. - modifier storageStoreGuard(bytes32 storageSlot) virtual { - /// @solidity memory-safe-assembly - assembly { - if or(eq(storageSlot, _OWNER_SLOT), eq(storageSlot, _ERC1967_IMPLEMENTATION_SLOT)) { - revert(codesize(), 0x00) - } - } - _; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* DEPOSIT OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the account's balance on the EntryPoint. - function getDeposit() public view virtual returns (uint256 result) { - address ep = entryPoint(); - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, address()) // Store the `account` argument. - mstore(0x00, 0x70a08231) // `balanceOf(address)`. - result := - mul( // Returns 0 if the EntryPoint does not exist. - mload(0x20), - and( // The arguments of `and` are evaluated from right to left. - gt(returndatasize(), 0x1f), // At least 32 bytes returned. - staticcall(gas(), ep, 0x1c, 0x24, 0x20, 0x20) - ) - ) - } - } - - /// @dev Deposit more funds for this account in the EntryPoint. - function addDeposit() public payable virtual { - address ep = entryPoint(); - /// @solidity memory-safe-assembly - assembly { - // The EntryPoint has balance accounting logic in the `receive()` function. - // forgefmt: disable-next-item - if iszero(mul(extcodesize(ep), call(gas(), ep, callvalue(), codesize(), 0x00, codesize(), 0x00))) { - revert(codesize(), 0x00) // For gas estimation. - } - } - } - - /// @dev Withdraw ETH from the account's deposit on the EntryPoint. - function withdrawDepositTo(address to, uint256 amount) public payable virtual onlyOwner { - address ep = entryPoint(); - /// @solidity memory-safe-assembly - assembly { - mstore(0x14, to) // Store the `to` argument. - mstore(0x34, amount) // Store the `amount` argument. - mstore(0x00, 0x205c2878000000000000000000000000) // `withdrawTo(address,uint256)`. - if iszero(mul(extcodesize(ep), call(gas(), ep, 0, 0x10, 0x44, codesize(), 0x00))) { - returndatacopy(mload(0x40), 0x00, returndatasize()) - revert(mload(0x40), returndatasize()) - } - mstore(0x34, 0) // Restore the part of the free memory pointer that was overwritten. - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* OVERRIDES */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Requires that the caller is the owner or the account itself. - /// This override affects the `onlyOwner` modifier. - function _checkOwner() internal view virtual override(Ownable) { - if (msg.sender != owner()) if (msg.sender != address(this)) revert Unauthorized(); - } - - /// @dev To prevent double-initialization (reuses the owner storage slot for efficiency). - function _guardInitializeOwner() internal pure virtual override(Ownable) returns (bool) { - return true; - } - - /// @dev Uses the `owner` as the ERC1271 signer. - function _erc1271Signer() internal view virtual override(ERC1271) returns (address) { - return owner(); - } - - /// @dev To ensure that only the owner or the account itself can upgrade the implementation. - function _authorizeUpgrade(address) internal virtual override(UUPSUpgradeable) onlyOwner {} - - /// @dev Handle token callbacks. If no token callback is triggered, - /// use `LibZip.cdFallback` for generalized calldata decompression. - /// If you don't need either, re-override this function. - fallback() external payable virtual override(Receiver) receiverFallback { - LibZip.cdFallback(); - } -} diff --git a/lib/solady/src/accounts/ERC4337Factory.sol b/lib/solady/src/accounts/ERC4337Factory.sol deleted file mode 100644 index 8d0e2bc..0000000 --- a/lib/solady/src/accounts/ERC4337Factory.sol +++ /dev/null @@ -1,71 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {LibClone} from "../utils/LibClone.sol"; - -/// @notice Simple ERC4337 account factory implementation. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/accounts/ERC4337Factory.sol) -/// -/// Note: -/// - Unlike the ERC1967Factory, this factory does NOT store any admin info on the factory itself. -/// The deployed ERC4337 accounts are minimal ERC1967 proxies to an ERC4337 implementation. -/// The proxy bytecode does NOT contain any upgrading logic. -/// - This factory does NOT contain any logic for upgrading the ERC4337 accounts. -/// Upgrading must be done via UUPS logic on the accounts themselves. -/// - The ERC4337 standard expects the factory to use deterministic deployment. -/// As such, this factory does not include any non-deterministic deployment methods. -contract ERC4337Factory { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* IMMUTABLES */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Address of the ERC4337 implementation. - address public immutable implementation; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTRUCTOR */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - constructor(address erc4337) payable { - implementation = erc4337; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* DEPLOY FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Deploys an ERC4337 account with `salt` and returns its deterministic address. - /// If the account is already deployed, it will simply return its address. - /// Any `msg.value` will simply be forwarded to the account, regardless. - function createAccount(address owner, bytes32 salt) public payable virtual returns (address) { - // Check that the salt is tied to the owner if required, regardless. - LibClone.checkStartsWith(salt, owner); - // Constructor data is optional, and is omitted for easier Etherscan verification. - (bool alreadyDeployed, address account) = - LibClone.createDeterministicERC1967(msg.value, implementation, salt); - - if (!alreadyDeployed) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x14, owner) // Store the `owner` argument. - mstore(0x00, 0xc4d66de8000000000000000000000000) // `initialize(address)`. - if iszero(call(gas(), account, 0, 0x10, 0x24, codesize(), 0x00)) { - returndatacopy(mload(0x40), 0x00, returndatasize()) - revert(mload(0x40), returndatasize()) - } - } - } - return account; - } - - /// @dev Returns the deterministic address of the account created via `createAccount`. - function getAddress(bytes32 salt) public view virtual returns (address) { - return LibClone.predictDeterministicAddressERC1967(implementation, salt, address(this)); - } - - /// @dev Returns the initialization code hash of the ERC4337 account (a minimal ERC1967 proxy). - /// Used for mining vanity addresses with create2crunch. - function initCodeHash() public view virtual returns (bytes32) { - return LibClone.initCodeHashERC1967(implementation); - } -} diff --git a/lib/solady/src/accounts/ERC6551.sol b/lib/solady/src/accounts/ERC6551.sol deleted file mode 100644 index 718d358..0000000 --- a/lib/solady/src/accounts/ERC6551.sol +++ /dev/null @@ -1,321 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {Receiver} from "./Receiver.sol"; -import {ERC1271} from "./ERC1271.sol"; -import {LibZip} from "../utils/LibZip.sol"; -import {UUPSUpgradeable} from "../utils/UUPSUpgradeable.sol"; - -/// @notice Simple ERC6551 account implementation. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/accounts/ERC6551.sol) -/// @author ERC6551 team (https://github.com/erc6551/reference/blob/main/src/examples/upgradeable/ERC6551AccountUpgradeable.sol) -/// -/// Recommended usage (regular): -/// 1. Deploy the ERC6551 as an implementation contract, and verify it on Etherscan. -/// 2. Use the canonical ERC6551Registry to deploy a clone to the ERC6551 implementation. -/// The UUPSUpgradeable functions will simply become no-ops. -/// -/// Recommended usage (upgradeable): -/// 1. Deploy the ERC6551 as an implementation contract, and verify it on Etherscan. -/// 2. Deploy the ERC6551Proxy pointing to the implementation. -/// This relay proxy is required, but Etherscan verification of it is optional. -/// 3. Use the canonical ERC6551Registry to deploy a clone to the ERC6551Proxy. -/// If you want to reveal the "Read as Proxy" and "Write as Proxy" tabs on Etherscan, -/// send 0 ETH to the clone to initialize its ERC1967 implementation slot, -/// the click on "Is this a proxy?" on the clone's page on Etherscan. -/// -/// Note: -/// - ERC6551 accounts are not compatible with ERC4337 -/// (at least not without crazy hacks) -/// due to storage access limitations during ERC4337 UserOp validation. -/// - Please refer to the official [ERC6551](https://github.com/erc6551/reference) reference -/// for latest updates on the ERC6551 standard, as well as canonical registry information. -abstract contract ERC6551 is UUPSUpgradeable, Receiver, ERC1271 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STRUCTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Call struct for the `executeBatch` function. - struct Call { - address target; - uint256 value; - bytes data; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The caller is not authorized to call the function. - error Unauthorized(); - - /// @dev The operation is not supported. - error OperationNotSupported(); - - /// @dev Self ownership detected. - error SelfOwnDetected(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The ERC6551 state slot is given by: - /// `bytes32(~uint256(uint32(bytes4(keccak256("_ERC6551_STATE_SLOT_NOT")))))`. - /// It is intentionally chosen to be a high value - /// to avoid collision with lower slots. - /// The choice of manual storage layout is to enable compatibility - /// with both regular and upgradeable contracts. - uint256 internal constant _ERC6551_STATE_SLOT = - 0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffb919c7a5; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* TOKEN-BOUND OWNERSHIP OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the token-bound information. - function token() - public - view - virtual - returns (uint256 chainId, address tokenContract, uint256 tokenId) - { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Cache the free memory pointer. - extcodecopy(address(), 0x00, 0x4d, 0x60) - chainId := mload(0x00) - tokenContract := mload(0x20) // Upper 96 bits will be clean. - tokenId := mload(0x40) - mstore(0x40, m) // Restore the free memory pointer. - } - } - - /// @dev Returns the owner of the contract. - function owner() public view virtual returns (address result) { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Cache the free memory pointer. - extcodecopy(address(), 0x00, 0x4d, 0x60) - if eq(mload(0x00), chainid()) { - let tokenContract := mload(0x20) - // `tokenId` is already at 0x40. - mstore(0x20, 0x6352211e) // `ownerOf(uint256)`. - result := - mul( // Returns `address(0)` on failure or if contract does not exist. - mload(0x20), - and( - gt(returndatasize(), 0x1f), - staticcall(gas(), tokenContract, 0x3c, 0x24, 0x20, 0x20) - ) - ) - } - mstore(0x40, m) // Restore the free memory pointer. - } - } - - /// @dev Returns if `signer` is an authorized signer. - function _isValidSigner(address signer) internal view virtual returns (bool) { - return signer == owner(); - } - - /// @dev Requires that the caller is a valid signer (i.e. the owner), or the contract itself. - modifier onlyValidSigner() virtual { - if (!_isValidSigner(msg.sender)) if (msg.sender != address(this)) revert Unauthorized(); - _; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STATE OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the current value of the state counter. - function state() public view virtual returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - result := sload(_ERC6551_STATE_SLOT) - } - } - - /// @dev Increments the state counter. This modifier is required for every - /// public / external function that may modify storage or emit events. - modifier incrementState() virtual { - /// @solidity memory-safe-assembly - assembly { - let s := _ERC6551_STATE_SLOT - sstore(s, add(1, sload(s))) - } - _; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EXECUTION OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Execute a call from this account. - function execute(address target, uint256 value, bytes calldata data, uint8 operation) - public - payable - virtual - onlyValidSigner - onlyValidExecuteOperation(operation) - incrementState - returns (bytes memory result) - { - /// @solidity memory-safe-assembly - assembly { - result := mload(0x40) - calldatacopy(result, data.offset, data.length) - if iszero(call(gas(), target, value, result, data.length, codesize(), 0x00)) { - // Bubble up the revert if the call reverts. - returndatacopy(result, 0x00, returndatasize()) - revert(result, returndatasize()) - } - mstore(result, returndatasize()) // Store the length. - let o := add(result, 0x20) - returndatacopy(o, 0x00, returndatasize()) // Copy the returndata. - mstore(0x40, add(o, returndatasize())) // Allocate the memory. - } - } - - /// @dev Execute a sequence of calls from this account. - function executeBatch(Call[] calldata calls, uint8 operation) - public - payable - virtual - onlyValidSigner - onlyValidExecuteOperation(operation) - incrementState - returns (bytes[] memory results) - { - /// @solidity memory-safe-assembly - assembly { - results := mload(0x40) - mstore(results, calls.length) - let r := add(0x20, results) - let m := add(r, shl(5, calls.length)) - calldatacopy(r, calls.offset, shl(5, calls.length)) - for { let end := m } iszero(eq(r, end)) { r := add(r, 0x20) } { - let e := add(calls.offset, mload(r)) - let o := add(e, calldataload(add(e, 0x40))) - calldatacopy(m, add(o, 0x20), calldataload(o)) - // forgefmt: disable-next-item - if iszero(call(gas(), calldataload(e), calldataload(add(e, 0x20)), - m, calldataload(o), codesize(), 0x00)) { - // Bubble up the revert if the call reverts. - returndatacopy(m, 0x00, returndatasize()) - revert(m, returndatasize()) - } - mstore(r, m) // Append `m` into `results`. - mstore(m, returndatasize()) // Store the length, - let p := add(m, 0x20) - returndatacopy(p, 0x00, returndatasize()) // and copy the returndata. - m := add(p, returndatasize()) // Advance `m`. - } - mstore(0x40, m) // Allocate the memory. - } - } - - /// @dev Requires that the execute `operation` is supported. - modifier onlyValidExecuteOperation(uint8 operation) virtual { - if (operation != 0) revert OperationNotSupported(); - _; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC165 */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns true if this contract implements the interface defined by `interfaceId`. - /// See: https://eips.ethereum.org/EIPS/eip-165 - /// This function call must use less than 30000 gas. - function supportsInterface(bytes4 interfaceId) public view virtual returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - let s := shr(224, interfaceId) - // ERC165: 0x01ffc9a7, ERC6551: 0x6faff5f1, ERC6551Executable: 0x74420f4c. - result := or(or(eq(s, 0x01ffc9a7), eq(s, 0x6faff5f1)), eq(s, 0x74420f4c)) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* OVERRIDES */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev To ensure that only the owner or the account itself can upgrade the implementation. - function _authorizeUpgrade(address) - internal - virtual - override(UUPSUpgradeable) - onlyValidSigner - incrementState - {} - - /// @dev Uses the `owner` as the ERC1271 signer. - function _erc1271Signer() internal view virtual override(ERC1271) returns (address) { - return owner(); - } - - /// @dev For handling token callbacks. - /// Safe-transferred ERC721 tokens will trigger a ownership cycle check. - modifier receiverFallback() override(Receiver) { - /// @solidity memory-safe-assembly - assembly { - let s := shr(224, calldataload(0x00)) - // 0x150b7a02: `onERC721Received(address,address,uint256,bytes)`. - if eq(s, 0x150b7a02) { - extcodecopy(address(), 0x00, 0x4d, 0x60) // `chainId`, `tokenContract`, `tokenId`. - mstore(0x60, 0xfc0c546a) // `token()`. - for {} 1 {} { - let tokenContract := mload(0x20) - // `tokenId` is already at 0x40. - mstore(0x20, 0x6352211e) // `ownerOf(uint256)`. - let chainsEq := eq(mload(0x00), chainid()) - let currentOwner := - mul( - mload(0x20), - and( - and(gt(returndatasize(), 0x1f), chainsEq), - staticcall(gas(), tokenContract, 0x3c, 0x24, 0x20, 0x20) - ) - ) - if iszero( - or( - eq(currentOwner, address()), - and( - and(chainsEq, eq(tokenContract, caller())), - eq(mload(0x40), calldataload(0x44)) - ) - ) - ) { - if iszero( - and( - gt(returndatasize(), 0x5f), - staticcall(gas(), currentOwner, 0x7c, 0x04, 0x00, 0x60) - ) - ) { - mstore(0x40, s) // Store `msg.sig`. - return(0x5c, 0x20) // Return `msg.sig`. - } - continue - } - mstore(0x00, 0xaed146d3) // `SelfOwnDetected()`. - revert(0x1c, 0x04) - } - } - // 0xf23a6e61: `onERC1155Received(address,address,uint256,uint256,bytes)`. - // 0xbc197c81: `onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)`. - if or(eq(s, 0xf23a6e61), eq(s, 0xbc197c81)) { - mstore(0x20, s) // Store `msg.sig`. - return(0x3c, 0x20) // Return `msg.sig`. - } - } - _; - } - - /// @dev Handle token callbacks. If no token callback is triggered, - /// use `LibZip.cdFallback` for generalized calldata decompression. - /// If you don't need either, re-override this function. - fallback() external payable virtual override(Receiver) receiverFallback { - LibZip.cdFallback(); - } -} diff --git a/lib/solady/src/accounts/ERC6551Proxy.sol b/lib/solady/src/accounts/ERC6551Proxy.sol deleted file mode 100644 index eb82b74..0000000 --- a/lib/solady/src/accounts/ERC6551Proxy.sol +++ /dev/null @@ -1,72 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Relay proxy for upgradeable ERC6551 accounts. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/accounts/ERC6551Proxy.sol) -/// @author ERC6551 team (https://github.com/erc6551/reference/blob/main/src/examples/upgradeable/ERC6551AccountProxy.sol) -/// -/// Note: This relay proxy is required for upgradeable ERC6551 accounts. -/// -/// ERC6551 clone -> ERC6551Proxy (relay) -> ERC6551 account implementation. -/// -/// This relay proxy also allows for correctly revealing the -/// "Read as Proxy" and "Write as Proxy" tabs on Etherscan. -/// -/// After using the registry to deploy a ERC6551 clone pointing to this relay proxy, -/// users must send 0 ETH to the clone before clicking on "Is this a proxy?" on Etherscan. -/// Verification of this relay proxy on Etherscan is optional. -contract ERC6551Proxy { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* IMMUTABLES */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The default implementation. - uint256 internal immutable _defaultImplementation; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STORAGE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The ERC-1967 storage slot for the implementation in the proxy. - /// `uint256(keccak256("eip1967.proxy.implementation")) - 1`. - bytes32 internal constant _ERC1967_IMPLEMENTATION_SLOT = - 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTRUCTOR */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - constructor(address defaultImplementation) payable { - _defaultImplementation = uint256(uint160(defaultImplementation)); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* FALLBACK */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - fallback() external payable virtual { - uint256 d = _defaultImplementation; - assembly { - mstore(0x40, returndatasize()) // Some optimization trick. - calldatacopy(returndatasize(), returndatasize(), calldatasize()) - let implementation := sload(_ERC1967_IMPLEMENTATION_SLOT) - // If the implementation is zero, initialize it to the default. - if iszero(implementation) { - implementation := d - // Only initialize if the calldatasize is zero, so that staticcalls to - // functions (which will have 4-byte function selectors) won't revert. - // Some users may be fine without Etherscan proxy detection and thus may - // choose to not initialize the ERC1967 implementation slot. - if iszero(calldatasize()) { sstore(_ERC1967_IMPLEMENTATION_SLOT, d) } - } - // forgefmt: disable-next-item - if iszero(delegatecall(gas(), implementation, - returndatasize(), calldatasize(), codesize(), returndatasize())) { - returndatacopy(0x00, 0x00, returndatasize()) - revert(0x00, returndatasize()) - } - returndatacopy(0x00, 0x00, returndatasize()) - return(0x00, returndatasize()) - } - } -} diff --git a/lib/solady/src/accounts/Receiver.sol b/lib/solady/src/accounts/Receiver.sol deleted file mode 100644 index 372038b..0000000 --- a/lib/solady/src/accounts/Receiver.sol +++ /dev/null @@ -1,33 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Receiver mixin for ETH and safe-transferred ERC721 and ERC1155 tokens. -/// @author Solady (https://github.com/Vectorized/solady/blob/main/src/accounts/Receiver.sol) -/// -/// @dev Note: -/// - Handles all ERC721 and ERC1155 token safety callbacks. -/// - Collapses function table gas overhead and code size. -/// - Utilizes fallback so unknown calldata will pass on. -abstract contract Receiver { - /// @dev For receiving ETH. - receive() external payable virtual {} - - /// @dev Fallback function with the `receiverFallback` modifier. - fallback() external payable virtual receiverFallback {} - - /// @dev Modifier for the fallback function to handle token callbacks. - modifier receiverFallback() virtual { - /// @solidity memory-safe-assembly - assembly { - let s := shr(224, calldataload(0)) - // 0x150b7a02: `onERC721Received(address,address,uint256,bytes)`. - // 0xf23a6e61: `onERC1155Received(address,address,uint256,uint256,bytes)`. - // 0xbc197c81: `onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)`. - if or(eq(s, 0x150b7a02), or(eq(s, 0xf23a6e61), eq(s, 0xbc197c81))) { - mstore(0x20, s) // Store `msg.sig`. - return(0x3c, 0x20) // Return `msg.sig`. - } - } - _; - } -} diff --git a/lib/solady/src/auth/Ownable.sol b/lib/solady/src/auth/Ownable.sol deleted file mode 100644 index a9d3214..0000000 --- a/lib/solady/src/auth/Ownable.sol +++ /dev/null @@ -1,278 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Simple single owner authorization mixin. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/auth/Ownable.sol) -/// -/// @dev Note: -/// This implementation does NOT auto-initialize the owner to `msg.sender`. -/// You MUST call the `_initializeOwner` in the constructor / initializer. -/// -/// While the ownable portion follows -/// [EIP-173](https://eips.ethereum.org/EIPS/eip-173) for compatibility, -/// the nomenclature for the 2-step ownership handover may be unique to this codebase. -abstract contract Ownable { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The caller is not authorized to call the function. - error Unauthorized(); - - /// @dev The `newOwner` cannot be the zero address. - error NewOwnerIsZeroAddress(); - - /// @dev The `pendingOwner` does not have a valid handover request. - error NoHandoverRequest(); - - /// @dev Cannot double-initialize. - error AlreadyInitialized(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EVENTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The ownership is transferred from `oldOwner` to `newOwner`. - /// This event is intentionally kept the same as OpenZeppelin's Ownable to be - /// compatible with indexers and [EIP-173](https://eips.ethereum.org/EIPS/eip-173), - /// despite it not being as lightweight as a single argument event. - event OwnershipTransferred(address indexed oldOwner, address indexed newOwner); - - /// @dev An ownership handover to `pendingOwner` has been requested. - event OwnershipHandoverRequested(address indexed pendingOwner); - - /// @dev The ownership handover to `pendingOwner` has been canceled. - event OwnershipHandoverCanceled(address indexed pendingOwner); - - /// @dev `keccak256(bytes("OwnershipTransferred(address,address)"))`. - uint256 private constant _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE = - 0x8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0; - - /// @dev `keccak256(bytes("OwnershipHandoverRequested(address)"))`. - uint256 private constant _OWNERSHIP_HANDOVER_REQUESTED_EVENT_SIGNATURE = - 0xdbf36a107da19e49527a7176a1babf963b4b0ff8cde35ee35d6cd8f1f9ac7e1d; - - /// @dev `keccak256(bytes("OwnershipHandoverCanceled(address)"))`. - uint256 private constant _OWNERSHIP_HANDOVER_CANCELED_EVENT_SIGNATURE = - 0xfa7b8eab7da67f412cc9575ed43464468f9bfbae89d1675917346ca6d8fe3c92; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STORAGE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The owner slot is given by: - /// `bytes32(~uint256(uint32(bytes4(keccak256("_OWNER_SLOT_NOT")))))`. - /// It is intentionally chosen to be a high value - /// to avoid collision with lower slots. - /// The choice of manual storage layout is to enable compatibility - /// with both regular and upgradeable contracts. - bytes32 internal constant _OWNER_SLOT = - 0xffffffffffffffffffffffffffffffffffffffffffffffffffffffff74873927; - - /// The ownership handover slot of `newOwner` is given by: - /// ``` - /// mstore(0x00, or(shl(96, user), _HANDOVER_SLOT_SEED)) - /// let handoverSlot := keccak256(0x00, 0x20) - /// ``` - /// It stores the expiry timestamp of the two-step ownership handover. - uint256 private constant _HANDOVER_SLOT_SEED = 0x389a75e1; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Override to return true to make `_initializeOwner` prevent double-initialization. - function _guardInitializeOwner() internal pure virtual returns (bool guard) {} - - /// @dev Initializes the owner directly without authorization guard. - /// This function must be called upon initialization, - /// regardless of whether the contract is upgradeable or not. - /// This is to enable generalization to both regular and upgradeable contracts, - /// and to save gas in case the initial owner is not the caller. - /// For performance reasons, this function will not check if there - /// is an existing owner. - function _initializeOwner(address newOwner) internal virtual { - if (_guardInitializeOwner()) { - /// @solidity memory-safe-assembly - assembly { - let ownerSlot := _OWNER_SLOT - if sload(ownerSlot) { - mstore(0x00, 0x0dc149f0) // `AlreadyInitialized()`. - revert(0x1c, 0x04) - } - // Clean the upper 96 bits. - newOwner := shr(96, shl(96, newOwner)) - // Store the new value. - sstore(ownerSlot, or(newOwner, shl(255, iszero(newOwner)))) - // Emit the {OwnershipTransferred} event. - log3(0, 0, _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE, 0, newOwner) - } - } else { - /// @solidity memory-safe-assembly - assembly { - // Clean the upper 96 bits. - newOwner := shr(96, shl(96, newOwner)) - // Store the new value. - sstore(_OWNER_SLOT, newOwner) - // Emit the {OwnershipTransferred} event. - log3(0, 0, _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE, 0, newOwner) - } - } - } - - /// @dev Sets the owner directly without authorization guard. - function _setOwner(address newOwner) internal virtual { - if (_guardInitializeOwner()) { - /// @solidity memory-safe-assembly - assembly { - let ownerSlot := _OWNER_SLOT - // Clean the upper 96 bits. - newOwner := shr(96, shl(96, newOwner)) - // Emit the {OwnershipTransferred} event. - log3(0, 0, _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE, sload(ownerSlot), newOwner) - // Store the new value. - sstore(ownerSlot, or(newOwner, shl(255, iszero(newOwner)))) - } - } else { - /// @solidity memory-safe-assembly - assembly { - let ownerSlot := _OWNER_SLOT - // Clean the upper 96 bits. - newOwner := shr(96, shl(96, newOwner)) - // Emit the {OwnershipTransferred} event. - log3(0, 0, _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE, sload(ownerSlot), newOwner) - // Store the new value. - sstore(ownerSlot, newOwner) - } - } - } - - /// @dev Throws if the sender is not the owner. - function _checkOwner() internal view virtual { - /// @solidity memory-safe-assembly - assembly { - // If the caller is not the stored owner, revert. - if iszero(eq(caller(), sload(_OWNER_SLOT))) { - mstore(0x00, 0x82b42900) // `Unauthorized()`. - revert(0x1c, 0x04) - } - } - } - - /// @dev Returns how long a two-step ownership handover is valid for in seconds. - /// Override to return a different value if needed. - /// Made internal to conserve bytecode. Wrap it in a public function if needed. - function _ownershipHandoverValidFor() internal view virtual returns (uint64) { - return 48 * 3600; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PUBLIC UPDATE FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Allows the owner to transfer the ownership to `newOwner`. - function transferOwnership(address newOwner) public payable virtual onlyOwner { - /// @solidity memory-safe-assembly - assembly { - if iszero(shl(96, newOwner)) { - mstore(0x00, 0x7448fbae) // `NewOwnerIsZeroAddress()`. - revert(0x1c, 0x04) - } - } - _setOwner(newOwner); - } - - /// @dev Allows the owner to renounce their ownership. - function renounceOwnership() public payable virtual onlyOwner { - _setOwner(address(0)); - } - - /// @dev Request a two-step ownership handover to the caller. - /// The request will automatically expire in 48 hours (172800 seconds) by default. - function requestOwnershipHandover() public payable virtual { - unchecked { - uint256 expires = block.timestamp + _ownershipHandoverValidFor(); - /// @solidity memory-safe-assembly - assembly { - // Compute and set the handover slot to `expires`. - mstore(0x0c, _HANDOVER_SLOT_SEED) - mstore(0x00, caller()) - sstore(keccak256(0x0c, 0x20), expires) - // Emit the {OwnershipHandoverRequested} event. - log2(0, 0, _OWNERSHIP_HANDOVER_REQUESTED_EVENT_SIGNATURE, caller()) - } - } - } - - /// @dev Cancels the two-step ownership handover to the caller, if any. - function cancelOwnershipHandover() public payable virtual { - /// @solidity memory-safe-assembly - assembly { - // Compute and set the handover slot to 0. - mstore(0x0c, _HANDOVER_SLOT_SEED) - mstore(0x00, caller()) - sstore(keccak256(0x0c, 0x20), 0) - // Emit the {OwnershipHandoverCanceled} event. - log2(0, 0, _OWNERSHIP_HANDOVER_CANCELED_EVENT_SIGNATURE, caller()) - } - } - - /// @dev Allows the owner to complete the two-step ownership handover to `pendingOwner`. - /// Reverts if there is no existing ownership handover requested by `pendingOwner`. - function completeOwnershipHandover(address pendingOwner) public payable virtual onlyOwner { - /// @solidity memory-safe-assembly - assembly { - // Compute and set the handover slot to 0. - mstore(0x0c, _HANDOVER_SLOT_SEED) - mstore(0x00, pendingOwner) - let handoverSlot := keccak256(0x0c, 0x20) - // If the handover does not exist, or has expired. - if gt(timestamp(), sload(handoverSlot)) { - mstore(0x00, 0x6f5e8818) // `NoHandoverRequest()`. - revert(0x1c, 0x04) - } - // Set the handover slot to 0. - sstore(handoverSlot, 0) - } - _setOwner(pendingOwner); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PUBLIC READ FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the owner of the contract. - function owner() public view virtual returns (address result) { - /// @solidity memory-safe-assembly - assembly { - result := sload(_OWNER_SLOT) - } - } - - /// @dev Returns the expiry timestamp for the two-step ownership handover to `pendingOwner`. - function ownershipHandoverExpiresAt(address pendingOwner) - public - view - virtual - returns (uint256 result) - { - /// @solidity memory-safe-assembly - assembly { - // Compute the handover slot. - mstore(0x0c, _HANDOVER_SLOT_SEED) - mstore(0x00, pendingOwner) - // Load the handover slot. - result := sload(keccak256(0x0c, 0x20)) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* MODIFIERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Marks a function as only callable by the owner. - modifier onlyOwner() virtual { - _checkOwner(); - _; - } -} diff --git a/lib/solady/src/auth/OwnableRoles.sol b/lib/solady/src/auth/OwnableRoles.sol deleted file mode 100644 index cd20f7d..0000000 --- a/lib/solady/src/auth/OwnableRoles.sol +++ /dev/null @@ -1,530 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {Ownable} from "./Ownable.sol"; - -/// @notice Simple single owner and multiroles authorization mixin. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/auth/Ownable.sol) -/// @dev While the ownable portion follows [EIP-173](https://eips.ethereum.org/EIPS/eip-173) -/// for compatibility, the nomenclature for the 2-step ownership handover and roles -/// may be unique to this codebase. -abstract contract OwnableRoles is Ownable { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EVENTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The `user`'s roles is updated to `roles`. - /// Each bit of `roles` represents whether the role is set. - event RolesUpdated(address indexed user, uint256 indexed roles); - - /// @dev `keccak256(bytes("RolesUpdated(address,uint256)"))`. - uint256 private constant _ROLES_UPDATED_EVENT_SIGNATURE = - 0x715ad5ce61fc9595c7b415289d59cf203f23a94fa06f04af7e489a0a76e1fe26; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STORAGE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The role slot of `user` is given by: - /// ``` - /// mstore(0x00, or(shl(96, user), _ROLE_SLOT_SEED)) - /// let roleSlot := keccak256(0x00, 0x20) - /// ``` - /// This automatically ignores the upper bits of the `user` in case - /// they are not clean, as well as keep the `keccak256` under 32-bytes. - /// - /// Note: This is equivalent to `uint32(bytes4(keccak256("_OWNER_SLOT_NOT")))`. - uint256 private constant _ROLE_SLOT_SEED = 0x8b78c6d8; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Overwrite the roles directly without authorization guard. - function _setRoles(address user, uint256 roles) internal virtual { - /// @solidity memory-safe-assembly - assembly { - mstore(0x0c, _ROLE_SLOT_SEED) - mstore(0x00, user) - // Store the new value. - sstore(keccak256(0x0c, 0x20), roles) - // Emit the {RolesUpdated} event. - log3(0, 0, _ROLES_UPDATED_EVENT_SIGNATURE, shr(96, mload(0x0c)), roles) - } - } - - /// @dev Updates the roles directly without authorization guard. - /// If `on` is true, each set bit of `roles` will be turned on, - /// otherwise, each set bit of `roles` will be turned off. - function _updateRoles(address user, uint256 roles, bool on) internal virtual { - /// @solidity memory-safe-assembly - assembly { - mstore(0x0c, _ROLE_SLOT_SEED) - mstore(0x00, user) - let roleSlot := keccak256(0x0c, 0x20) - // Load the current value. - let current := sload(roleSlot) - // Compute the updated roles if `on` is true. - let updated := or(current, roles) - // Compute the updated roles if `on` is false. - // Use `and` to compute the intersection of `current` and `roles`, - // `xor` it with `current` to flip the bits in the intersection. - if iszero(on) { updated := xor(current, and(current, roles)) } - // Then, store the new value. - sstore(roleSlot, updated) - // Emit the {RolesUpdated} event. - log3(0, 0, _ROLES_UPDATED_EVENT_SIGNATURE, shr(96, mload(0x0c)), updated) - } - } - - /// @dev Grants the roles directly without authorization guard. - /// Each bit of `roles` represents the role to turn on. - function _grantRoles(address user, uint256 roles) internal virtual { - _updateRoles(user, roles, true); - } - - /// @dev Removes the roles directly without authorization guard. - /// Each bit of `roles` represents the role to turn off. - function _removeRoles(address user, uint256 roles) internal virtual { - _updateRoles(user, roles, false); - } - - /// @dev Throws if the sender does not have any of the `roles`. - function _checkRoles(uint256 roles) internal view virtual { - /// @solidity memory-safe-assembly - assembly { - // Compute the role slot. - mstore(0x0c, _ROLE_SLOT_SEED) - mstore(0x00, caller()) - // Load the stored value, and if the `and` intersection - // of the value and `roles` is zero, revert. - if iszero(and(sload(keccak256(0x0c, 0x20)), roles)) { - mstore(0x00, 0x82b42900) // `Unauthorized()`. - revert(0x1c, 0x04) - } - } - } - - /// @dev Throws if the sender is not the owner, - /// and does not have any of the `roles`. - /// Checks for ownership first, then lazily checks for roles. - function _checkOwnerOrRoles(uint256 roles) internal view virtual { - /// @solidity memory-safe-assembly - assembly { - // If the caller is not the stored owner. - // Note: `_ROLE_SLOT_SEED` is equal to `_OWNER_SLOT_NOT`. - if iszero(eq(caller(), sload(not(_ROLE_SLOT_SEED)))) { - // Compute the role slot. - mstore(0x0c, _ROLE_SLOT_SEED) - mstore(0x00, caller()) - // Load the stored value, and if the `and` intersection - // of the value and `roles` is zero, revert. - if iszero(and(sload(keccak256(0x0c, 0x20)), roles)) { - mstore(0x00, 0x82b42900) // `Unauthorized()`. - revert(0x1c, 0x04) - } - } - } - } - - /// @dev Throws if the sender does not have any of the `roles`, - /// and is not the owner. - /// Checks for roles first, then lazily checks for ownership. - function _checkRolesOrOwner(uint256 roles) internal view virtual { - /// @solidity memory-safe-assembly - assembly { - // Compute the role slot. - mstore(0x0c, _ROLE_SLOT_SEED) - mstore(0x00, caller()) - // Load the stored value, and if the `and` intersection - // of the value and `roles` is zero, revert. - if iszero(and(sload(keccak256(0x0c, 0x20)), roles)) { - // If the caller is not the stored owner. - // Note: `_ROLE_SLOT_SEED` is equal to `_OWNER_SLOT_NOT`. - if iszero(eq(caller(), sload(not(_ROLE_SLOT_SEED)))) { - mstore(0x00, 0x82b42900) // `Unauthorized()`. - revert(0x1c, 0x04) - } - } - } - } - - /// @dev Convenience function to return a `roles` bitmap from an array of `ordinals`. - /// This is meant for frontends like Etherscan, and is therefore not fully optimized. - /// Not recommended to be called on-chain. - /// Made internal to conserve bytecode. Wrap it in a public function if needed. - function _rolesFromOrdinals(uint8[] memory ordinals) internal pure returns (uint256 roles) { - /// @solidity memory-safe-assembly - assembly { - for { let i := shl(5, mload(ordinals)) } i { i := sub(i, 0x20) } { - // We don't need to mask the values of `ordinals`, as Solidity - // cleans dirty upper bits when storing variables into memory. - roles := or(shl(mload(add(ordinals, i)), 1), roles) - } - } - } - - /// @dev Convenience function to return an array of `ordinals` from the `roles` bitmap. - /// This is meant for frontends like Etherscan, and is therefore not fully optimized. - /// Not recommended to be called on-chain. - /// Made internal to conserve bytecode. Wrap it in a public function if needed. - function _ordinalsFromRoles(uint256 roles) internal pure returns (uint8[] memory ordinals) { - /// @solidity memory-safe-assembly - assembly { - // Grab the pointer to the free memory. - ordinals := mload(0x40) - let ptr := add(ordinals, 0x20) - let o := 0 - // The absence of lookup tables, De Bruijn, etc., here is intentional for - // smaller bytecode, as this function is not meant to be called on-chain. - for { let t := roles } 1 {} { - mstore(ptr, o) - // `shr` 5 is equivalent to multiplying by 0x20. - // Push back into the ordinals array if the bit is set. - ptr := add(ptr, shl(5, and(t, 1))) - o := add(o, 1) - t := shr(o, roles) - if iszero(t) { break } - } - // Store the length of `ordinals`. - mstore(ordinals, shr(5, sub(ptr, add(ordinals, 0x20)))) - // Allocate the memory. - mstore(0x40, ptr) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PUBLIC UPDATE FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Allows the owner to grant `user` `roles`. - /// If the `user` already has a role, then it will be an no-op for the role. - function grantRoles(address user, uint256 roles) public payable virtual onlyOwner { - _grantRoles(user, roles); - } - - /// @dev Allows the owner to remove `user` `roles`. - /// If the `user` does not have a role, then it will be an no-op for the role. - function revokeRoles(address user, uint256 roles) public payable virtual onlyOwner { - _removeRoles(user, roles); - } - - /// @dev Allow the caller to remove their own roles. - /// If the caller does not have a role, then it will be an no-op for the role. - function renounceRoles(uint256 roles) public payable virtual { - _removeRoles(msg.sender, roles); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PUBLIC READ FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the roles of `user`. - function rolesOf(address user) public view virtual returns (uint256 roles) { - /// @solidity memory-safe-assembly - assembly { - // Compute the role slot. - mstore(0x0c, _ROLE_SLOT_SEED) - mstore(0x00, user) - // Load the stored value. - roles := sload(keccak256(0x0c, 0x20)) - } - } - - /// @dev Returns whether `user` has any of `roles`. - function hasAnyRole(address user, uint256 roles) public view virtual returns (bool) { - return rolesOf(user) & roles != 0; - } - - /// @dev Returns whether `user` has all of `roles`. - function hasAllRoles(address user, uint256 roles) public view virtual returns (bool) { - return rolesOf(user) & roles == roles; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* MODIFIERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Marks a function as only callable by an account with `roles`. - modifier onlyRoles(uint256 roles) virtual { - _checkRoles(roles); - _; - } - - /// @dev Marks a function as only callable by the owner or by an account - /// with `roles`. Checks for ownership first, then lazily checks for roles. - modifier onlyOwnerOrRoles(uint256 roles) virtual { - _checkOwnerOrRoles(roles); - _; - } - - /// @dev Marks a function as only callable by an account with `roles` - /// or the owner. Checks for roles first, then lazily checks for ownership. - modifier onlyRolesOrOwner(uint256 roles) virtual { - _checkRolesOrOwner(roles); - _; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ROLE CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // IYKYK - - uint256 internal constant _ROLE_0 = 1 << 0; - uint256 internal constant _ROLE_1 = 1 << 1; - uint256 internal constant _ROLE_2 = 1 << 2; - uint256 internal constant _ROLE_3 = 1 << 3; - uint256 internal constant _ROLE_4 = 1 << 4; - uint256 internal constant _ROLE_5 = 1 << 5; - uint256 internal constant _ROLE_6 = 1 << 6; - uint256 internal constant _ROLE_7 = 1 << 7; - uint256 internal constant _ROLE_8 = 1 << 8; - uint256 internal constant _ROLE_9 = 1 << 9; - uint256 internal constant _ROLE_10 = 1 << 10; - uint256 internal constant _ROLE_11 = 1 << 11; - uint256 internal constant _ROLE_12 = 1 << 12; - uint256 internal constant _ROLE_13 = 1 << 13; - uint256 internal constant _ROLE_14 = 1 << 14; - uint256 internal constant _ROLE_15 = 1 << 15; - uint256 internal constant _ROLE_16 = 1 << 16; - uint256 internal constant _ROLE_17 = 1 << 17; - uint256 internal constant _ROLE_18 = 1 << 18; - uint256 internal constant _ROLE_19 = 1 << 19; - uint256 internal constant _ROLE_20 = 1 << 20; - uint256 internal constant _ROLE_21 = 1 << 21; - uint256 internal constant _ROLE_22 = 1 << 22; - uint256 internal constant _ROLE_23 = 1 << 23; - uint256 internal constant _ROLE_24 = 1 << 24; - uint256 internal constant _ROLE_25 = 1 << 25; - uint256 internal constant _ROLE_26 = 1 << 26; - uint256 internal constant _ROLE_27 = 1 << 27; - uint256 internal constant _ROLE_28 = 1 << 28; - uint256 internal constant _ROLE_29 = 1 << 29; - uint256 internal constant _ROLE_30 = 1 << 30; - uint256 internal constant _ROLE_31 = 1 << 31; - uint256 internal constant _ROLE_32 = 1 << 32; - uint256 internal constant _ROLE_33 = 1 << 33; - uint256 internal constant _ROLE_34 = 1 << 34; - uint256 internal constant _ROLE_35 = 1 << 35; - uint256 internal constant _ROLE_36 = 1 << 36; - uint256 internal constant _ROLE_37 = 1 << 37; - uint256 internal constant _ROLE_38 = 1 << 38; - uint256 internal constant _ROLE_39 = 1 << 39; - uint256 internal constant _ROLE_40 = 1 << 40; - uint256 internal constant _ROLE_41 = 1 << 41; - uint256 internal constant _ROLE_42 = 1 << 42; - uint256 internal constant _ROLE_43 = 1 << 43; - uint256 internal constant _ROLE_44 = 1 << 44; - uint256 internal constant _ROLE_45 = 1 << 45; - uint256 internal constant _ROLE_46 = 1 << 46; - uint256 internal constant _ROLE_47 = 1 << 47; - uint256 internal constant _ROLE_48 = 1 << 48; - uint256 internal constant _ROLE_49 = 1 << 49; - uint256 internal constant _ROLE_50 = 1 << 50; - uint256 internal constant _ROLE_51 = 1 << 51; - uint256 internal constant _ROLE_52 = 1 << 52; - uint256 internal constant _ROLE_53 = 1 << 53; - uint256 internal constant _ROLE_54 = 1 << 54; - uint256 internal constant _ROLE_55 = 1 << 55; - uint256 internal constant _ROLE_56 = 1 << 56; - uint256 internal constant _ROLE_57 = 1 << 57; - uint256 internal constant _ROLE_58 = 1 << 58; - uint256 internal constant _ROLE_59 = 1 << 59; - uint256 internal constant _ROLE_60 = 1 << 60; - uint256 internal constant _ROLE_61 = 1 << 61; - uint256 internal constant _ROLE_62 = 1 << 62; - uint256 internal constant _ROLE_63 = 1 << 63; - uint256 internal constant _ROLE_64 = 1 << 64; - uint256 internal constant _ROLE_65 = 1 << 65; - uint256 internal constant _ROLE_66 = 1 << 66; - uint256 internal constant _ROLE_67 = 1 << 67; - uint256 internal constant _ROLE_68 = 1 << 68; - uint256 internal constant _ROLE_69 = 1 << 69; - uint256 internal constant _ROLE_70 = 1 << 70; - uint256 internal constant _ROLE_71 = 1 << 71; - uint256 internal constant _ROLE_72 = 1 << 72; - uint256 internal constant _ROLE_73 = 1 << 73; - uint256 internal constant _ROLE_74 = 1 << 74; - uint256 internal constant _ROLE_75 = 1 << 75; - uint256 internal constant _ROLE_76 = 1 << 76; - uint256 internal constant _ROLE_77 = 1 << 77; - uint256 internal constant _ROLE_78 = 1 << 78; - uint256 internal constant _ROLE_79 = 1 << 79; - uint256 internal constant _ROLE_80 = 1 << 80; - uint256 internal constant _ROLE_81 = 1 << 81; - uint256 internal constant _ROLE_82 = 1 << 82; - uint256 internal constant _ROLE_83 = 1 << 83; - uint256 internal constant _ROLE_84 = 1 << 84; - uint256 internal constant _ROLE_85 = 1 << 85; - uint256 internal constant _ROLE_86 = 1 << 86; - uint256 internal constant _ROLE_87 = 1 << 87; - uint256 internal constant _ROLE_88 = 1 << 88; - uint256 internal constant _ROLE_89 = 1 << 89; - uint256 internal constant _ROLE_90 = 1 << 90; - uint256 internal constant _ROLE_91 = 1 << 91; - uint256 internal constant _ROLE_92 = 1 << 92; - uint256 internal constant _ROLE_93 = 1 << 93; - uint256 internal constant _ROLE_94 = 1 << 94; - uint256 internal constant _ROLE_95 = 1 << 95; - uint256 internal constant _ROLE_96 = 1 << 96; - uint256 internal constant _ROLE_97 = 1 << 97; - uint256 internal constant _ROLE_98 = 1 << 98; - uint256 internal constant _ROLE_99 = 1 << 99; - uint256 internal constant _ROLE_100 = 1 << 100; - uint256 internal constant _ROLE_101 = 1 << 101; - uint256 internal constant _ROLE_102 = 1 << 102; - uint256 internal constant _ROLE_103 = 1 << 103; - uint256 internal constant _ROLE_104 = 1 << 104; - uint256 internal constant _ROLE_105 = 1 << 105; - uint256 internal constant _ROLE_106 = 1 << 106; - uint256 internal constant _ROLE_107 = 1 << 107; - uint256 internal constant _ROLE_108 = 1 << 108; - uint256 internal constant _ROLE_109 = 1 << 109; - uint256 internal constant _ROLE_110 = 1 << 110; - uint256 internal constant _ROLE_111 = 1 << 111; - uint256 internal constant _ROLE_112 = 1 << 112; - uint256 internal constant _ROLE_113 = 1 << 113; - uint256 internal constant _ROLE_114 = 1 << 114; - uint256 internal constant _ROLE_115 = 1 << 115; - uint256 internal constant _ROLE_116 = 1 << 116; - uint256 internal constant _ROLE_117 = 1 << 117; - uint256 internal constant _ROLE_118 = 1 << 118; - uint256 internal constant _ROLE_119 = 1 << 119; - uint256 internal constant _ROLE_120 = 1 << 120; - uint256 internal constant _ROLE_121 = 1 << 121; - uint256 internal constant _ROLE_122 = 1 << 122; - uint256 internal constant _ROLE_123 = 1 << 123; - uint256 internal constant _ROLE_124 = 1 << 124; - uint256 internal constant _ROLE_125 = 1 << 125; - uint256 internal constant _ROLE_126 = 1 << 126; - uint256 internal constant _ROLE_127 = 1 << 127; - uint256 internal constant _ROLE_128 = 1 << 128; - uint256 internal constant _ROLE_129 = 1 << 129; - uint256 internal constant _ROLE_130 = 1 << 130; - uint256 internal constant _ROLE_131 = 1 << 131; - uint256 internal constant _ROLE_132 = 1 << 132; - uint256 internal constant _ROLE_133 = 1 << 133; - uint256 internal constant _ROLE_134 = 1 << 134; - uint256 internal constant _ROLE_135 = 1 << 135; - uint256 internal constant _ROLE_136 = 1 << 136; - uint256 internal constant _ROLE_137 = 1 << 137; - uint256 internal constant _ROLE_138 = 1 << 138; - uint256 internal constant _ROLE_139 = 1 << 139; - uint256 internal constant _ROLE_140 = 1 << 140; - uint256 internal constant _ROLE_141 = 1 << 141; - uint256 internal constant _ROLE_142 = 1 << 142; - uint256 internal constant _ROLE_143 = 1 << 143; - uint256 internal constant _ROLE_144 = 1 << 144; - uint256 internal constant _ROLE_145 = 1 << 145; - uint256 internal constant _ROLE_146 = 1 << 146; - uint256 internal constant _ROLE_147 = 1 << 147; - uint256 internal constant _ROLE_148 = 1 << 148; - uint256 internal constant _ROLE_149 = 1 << 149; - uint256 internal constant _ROLE_150 = 1 << 150; - uint256 internal constant _ROLE_151 = 1 << 151; - uint256 internal constant _ROLE_152 = 1 << 152; - uint256 internal constant _ROLE_153 = 1 << 153; - uint256 internal constant _ROLE_154 = 1 << 154; - uint256 internal constant _ROLE_155 = 1 << 155; - uint256 internal constant _ROLE_156 = 1 << 156; - uint256 internal constant _ROLE_157 = 1 << 157; - uint256 internal constant _ROLE_158 = 1 << 158; - uint256 internal constant _ROLE_159 = 1 << 159; - uint256 internal constant _ROLE_160 = 1 << 160; - uint256 internal constant _ROLE_161 = 1 << 161; - uint256 internal constant _ROLE_162 = 1 << 162; - uint256 internal constant _ROLE_163 = 1 << 163; - uint256 internal constant _ROLE_164 = 1 << 164; - uint256 internal constant _ROLE_165 = 1 << 165; - uint256 internal constant _ROLE_166 = 1 << 166; - uint256 internal constant _ROLE_167 = 1 << 167; - uint256 internal constant _ROLE_168 = 1 << 168; - uint256 internal constant _ROLE_169 = 1 << 169; - uint256 internal constant _ROLE_170 = 1 << 170; - uint256 internal constant _ROLE_171 = 1 << 171; - uint256 internal constant _ROLE_172 = 1 << 172; - uint256 internal constant _ROLE_173 = 1 << 173; - uint256 internal constant _ROLE_174 = 1 << 174; - uint256 internal constant _ROLE_175 = 1 << 175; - uint256 internal constant _ROLE_176 = 1 << 176; - uint256 internal constant _ROLE_177 = 1 << 177; - uint256 internal constant _ROLE_178 = 1 << 178; - uint256 internal constant _ROLE_179 = 1 << 179; - uint256 internal constant _ROLE_180 = 1 << 180; - uint256 internal constant _ROLE_181 = 1 << 181; - uint256 internal constant _ROLE_182 = 1 << 182; - uint256 internal constant _ROLE_183 = 1 << 183; - uint256 internal constant _ROLE_184 = 1 << 184; - uint256 internal constant _ROLE_185 = 1 << 185; - uint256 internal constant _ROLE_186 = 1 << 186; - uint256 internal constant _ROLE_187 = 1 << 187; - uint256 internal constant _ROLE_188 = 1 << 188; - uint256 internal constant _ROLE_189 = 1 << 189; - uint256 internal constant _ROLE_190 = 1 << 190; - uint256 internal constant _ROLE_191 = 1 << 191; - uint256 internal constant _ROLE_192 = 1 << 192; - uint256 internal constant _ROLE_193 = 1 << 193; - uint256 internal constant _ROLE_194 = 1 << 194; - uint256 internal constant _ROLE_195 = 1 << 195; - uint256 internal constant _ROLE_196 = 1 << 196; - uint256 internal constant _ROLE_197 = 1 << 197; - uint256 internal constant _ROLE_198 = 1 << 198; - uint256 internal constant _ROLE_199 = 1 << 199; - uint256 internal constant _ROLE_200 = 1 << 200; - uint256 internal constant _ROLE_201 = 1 << 201; - uint256 internal constant _ROLE_202 = 1 << 202; - uint256 internal constant _ROLE_203 = 1 << 203; - uint256 internal constant _ROLE_204 = 1 << 204; - uint256 internal constant _ROLE_205 = 1 << 205; - uint256 internal constant _ROLE_206 = 1 << 206; - uint256 internal constant _ROLE_207 = 1 << 207; - uint256 internal constant _ROLE_208 = 1 << 208; - uint256 internal constant _ROLE_209 = 1 << 209; - uint256 internal constant _ROLE_210 = 1 << 210; - uint256 internal constant _ROLE_211 = 1 << 211; - uint256 internal constant _ROLE_212 = 1 << 212; - uint256 internal constant _ROLE_213 = 1 << 213; - uint256 internal constant _ROLE_214 = 1 << 214; - uint256 internal constant _ROLE_215 = 1 << 215; - uint256 internal constant _ROLE_216 = 1 << 216; - uint256 internal constant _ROLE_217 = 1 << 217; - uint256 internal constant _ROLE_218 = 1 << 218; - uint256 internal constant _ROLE_219 = 1 << 219; - uint256 internal constant _ROLE_220 = 1 << 220; - uint256 internal constant _ROLE_221 = 1 << 221; - uint256 internal constant _ROLE_222 = 1 << 222; - uint256 internal constant _ROLE_223 = 1 << 223; - uint256 internal constant _ROLE_224 = 1 << 224; - uint256 internal constant _ROLE_225 = 1 << 225; - uint256 internal constant _ROLE_226 = 1 << 226; - uint256 internal constant _ROLE_227 = 1 << 227; - uint256 internal constant _ROLE_228 = 1 << 228; - uint256 internal constant _ROLE_229 = 1 << 229; - uint256 internal constant _ROLE_230 = 1 << 230; - uint256 internal constant _ROLE_231 = 1 << 231; - uint256 internal constant _ROLE_232 = 1 << 232; - uint256 internal constant _ROLE_233 = 1 << 233; - uint256 internal constant _ROLE_234 = 1 << 234; - uint256 internal constant _ROLE_235 = 1 << 235; - uint256 internal constant _ROLE_236 = 1 << 236; - uint256 internal constant _ROLE_237 = 1 << 237; - uint256 internal constant _ROLE_238 = 1 << 238; - uint256 internal constant _ROLE_239 = 1 << 239; - uint256 internal constant _ROLE_240 = 1 << 240; - uint256 internal constant _ROLE_241 = 1 << 241; - uint256 internal constant _ROLE_242 = 1 << 242; - uint256 internal constant _ROLE_243 = 1 << 243; - uint256 internal constant _ROLE_244 = 1 << 244; - uint256 internal constant _ROLE_245 = 1 << 245; - uint256 internal constant _ROLE_246 = 1 << 246; - uint256 internal constant _ROLE_247 = 1 << 247; - uint256 internal constant _ROLE_248 = 1 << 248; - uint256 internal constant _ROLE_249 = 1 << 249; - uint256 internal constant _ROLE_250 = 1 << 250; - uint256 internal constant _ROLE_251 = 1 << 251; - uint256 internal constant _ROLE_252 = 1 << 252; - uint256 internal constant _ROLE_253 = 1 << 253; - uint256 internal constant _ROLE_254 = 1 << 254; - uint256 internal constant _ROLE_255 = 1 << 255; -} diff --git a/lib/solady/src/tokens/ERC1155.sol b/lib/solady/src/tokens/ERC1155.sol deleted file mode 100644 index a5a173e..0000000 --- a/lib/solady/src/tokens/ERC1155.sol +++ /dev/null @@ -1,1114 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Simple ERC1155 implementation. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/tokens/ERC1155.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC1155.sol) -/// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/tree/master/contracts/token/ERC1155/ERC1155.sol) -/// -/// @dev Note: -/// - The ERC1155 standard allows for self-approvals. -/// For performance, this implementation WILL NOT revert for such actions. -/// Please add any checks with overrides if desired. -/// - The transfer functions use the identity precompile (0x4) -/// to copy memory internally. -/// -/// If you are overriding: -/// - Make sure all variables written to storage are properly cleaned -// (e.g. the bool value for `isApprovedForAll` MUST be either 1 or 0 under the hood). -/// - Check that the overridden function is actually used in the function you want to -/// change the behavior of. Much of the code has been manually inlined for performance. -abstract contract ERC1155 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The lengths of the input arrays are not the same. - error ArrayLengthsMismatch(); - - /// @dev Cannot mint or transfer to the zero address. - error TransferToZeroAddress(); - - /// @dev The recipient's balance has overflowed. - error AccountBalanceOverflow(); - - /// @dev Insufficient balance. - error InsufficientBalance(); - - /// @dev Only the token owner or an approved account can manage the tokens. - error NotOwnerNorApproved(); - - /// @dev Cannot safely transfer to a contract that does not implement - /// the ERC1155Receiver interface. - error TransferToNonERC1155ReceiverImplementer(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EVENTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Emitted when `amount` of token `id` is transferred - /// from `from` to `to` by `operator`. - event TransferSingle( - address indexed operator, - address indexed from, - address indexed to, - uint256 id, - uint256 amount - ); - - /// @dev Emitted when `amounts` of token `ids` are transferred - /// from `from` to `to` by `operator`. - event TransferBatch( - address indexed operator, - address indexed from, - address indexed to, - uint256[] ids, - uint256[] amounts - ); - - /// @dev Emitted when `owner` enables or disables `operator` to manage all of their tokens. - event ApprovalForAll(address indexed owner, address indexed operator, bool isApproved); - - /// @dev Emitted when the Uniform Resource Identifier (URI) for token `id` - /// is updated to `value`. This event is not used in the base contract. - /// You may need to emit this event depending on your URI logic. - /// - /// See: https://eips.ethereum.org/EIPS/eip-1155#metadata - event URI(string value, uint256 indexed id); - - /// @dev `keccak256(bytes("TransferSingle(address,address,address,uint256,uint256)"))`. - uint256 private constant _TRANSFER_SINGLE_EVENT_SIGNATURE = - 0xc3d58168c5ae7397731d063d5bbf3d657854427343f4c083240f7aacaa2d0f62; - - /// @dev `keccak256(bytes("TransferBatch(address,address,address,uint256[],uint256[])"))`. - uint256 private constant _TRANSFER_BATCH_EVENT_SIGNATURE = - 0x4a39dc06d4c0dbc64b70af90fd698a233a518aa5d07e595d983b8c0526c8f7fb; - - /// @dev `keccak256(bytes("ApprovalForAll(address,address,bool)"))`. - uint256 private constant _APPROVAL_FOR_ALL_EVENT_SIGNATURE = - 0x17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STORAGE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The `ownerSlotSeed` of a given owner is given by. - /// ``` - /// let ownerSlotSeed := or(_ERC1155_MASTER_SLOT_SEED, shl(96, owner)) - /// ``` - /// - /// The balance slot of `owner` is given by. - /// ``` - /// mstore(0x20, ownerSlotSeed) - /// mstore(0x00, id) - /// let balanceSlot := keccak256(0x00, 0x40) - /// ``` - /// - /// The operator approval slot of `owner` is given by. - /// ``` - /// mstore(0x20, ownerSlotSeed) - /// mstore(0x00, operator) - /// let operatorApprovalSlot := keccak256(0x0c, 0x34) - /// ``` - uint256 private constant _ERC1155_MASTER_SLOT_SEED = 0x9a31110384e0b0c9; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC1155 METADATA */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the URI for token `id`. - /// - /// You can either return the same templated URI for all token IDs, - /// (e.g. "https://example.com/api/{id}.json"), - /// or return a unique URI for each `id`. - /// - /// See: https://eips.ethereum.org/EIPS/eip-1155#metadata - function uri(uint256 id) public view virtual returns (string memory); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC1155 */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the amount of `id` owned by `owner`. - function balanceOf(address owner, uint256 id) public view virtual returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, _ERC1155_MASTER_SLOT_SEED) - mstore(0x14, owner) - mstore(0x00, id) - result := sload(keccak256(0x00, 0x40)) - } - } - - /// @dev Returns whether `operator` is approved to manage the tokens of `owner`. - function isApprovedForAll(address owner, address operator) - public - view - virtual - returns (bool result) - { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, _ERC1155_MASTER_SLOT_SEED) - mstore(0x14, owner) - mstore(0x00, operator) - result := sload(keccak256(0x0c, 0x34)) - } - } - - /// @dev Sets whether `operator` is approved to manage the tokens of the caller. - /// - /// Emits a {ApprovalForAll} event. - function setApprovalForAll(address operator, bool isApproved) public virtual { - /// @solidity memory-safe-assembly - assembly { - // Convert to 0 or 1. - isApproved := iszero(iszero(isApproved)) - // Update the `isApproved` for (`msg.sender`, `operator`). - mstore(0x20, _ERC1155_MASTER_SLOT_SEED) - mstore(0x14, caller()) - mstore(0x00, operator) - sstore(keccak256(0x0c, 0x34), isApproved) - // Emit the {ApprovalForAll} event. - mstore(0x00, isApproved) - // forgefmt: disable-next-line - log3(0x00, 0x20, _APPROVAL_FOR_ALL_EVENT_SIGNATURE, caller(), shr(96, shl(96, operator))) - } - } - - /// @dev Transfers `amount` of `id` from `from` to `to`. - /// - /// Requirements: - /// - `to` cannot be the zero address. - /// - `from` must have at least `amount` of `id`. - /// - If the caller is not `from`, - /// it must be approved to manage the tokens of `from`. - /// - If `to` refers to a smart contract, it must implement - /// {ERC1155-onERC1155Reveived}, which is called upon a batch transfer. - /// - /// Emits a {Transfer} event. - function safeTransferFrom( - address from, - address to, - uint256 id, - uint256 amount, - bytes calldata data - ) public virtual { - if (_useBeforeTokenTransfer()) { - _beforeTokenTransfer(from, to, _single(id), _single(amount), data); - } - /// @solidity memory-safe-assembly - assembly { - let fromSlotSeed := or(_ERC1155_MASTER_SLOT_SEED, shl(96, from)) - let toSlotSeed := or(_ERC1155_MASTER_SLOT_SEED, shl(96, to)) - mstore(0x20, fromSlotSeed) - // Clear the upper 96 bits. - from := shr(96, fromSlotSeed) - to := shr(96, toSlotSeed) - // Revert if `to` is the zero address. - if iszero(to) { - mstore(0x00, 0xea553b34) // `TransferToZeroAddress()`. - revert(0x1c, 0x04) - } - // If the caller is not `from`, do the authorization check. - if iszero(eq(caller(), from)) { - mstore(0x00, caller()) - if iszero(sload(keccak256(0x0c, 0x34))) { - mstore(0x00, 0x4b6e7f18) // `NotOwnerNorApproved()`. - revert(0x1c, 0x04) - } - } - // Subtract and store the updated balance of `from`. - { - mstore(0x00, id) - let fromBalanceSlot := keccak256(0x00, 0x40) - let fromBalance := sload(fromBalanceSlot) - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - sstore(fromBalanceSlot, sub(fromBalance, amount)) - } - // Increase and store the updated balance of `to`. - { - mstore(0x20, toSlotSeed) - let toBalanceSlot := keccak256(0x00, 0x40) - let toBalanceBefore := sload(toBalanceSlot) - let toBalanceAfter := add(toBalanceBefore, amount) - if lt(toBalanceAfter, toBalanceBefore) { - mstore(0x00, 0x01336cea) // `AccountBalanceOverflow()`. - revert(0x1c, 0x04) - } - sstore(toBalanceSlot, toBalanceAfter) - } - // Emit a {TransferSingle} event. - mstore(0x20, amount) - log4(0x00, 0x40, _TRANSFER_SINGLE_EVENT_SIGNATURE, caller(), from, to) - } - if (_useAfterTokenTransfer()) { - _afterTokenTransfer(from, to, _single(id), _single(amount), data); - } - /// @solidity memory-safe-assembly - assembly { - // Do the {onERC1155Received} check if `to` is a smart contract. - if extcodesize(to) { - // Prepare the calldata. - let m := mload(0x40) - // `onERC1155Received(address,address,uint256,uint256,bytes)`. - mstore(m, 0xf23a6e61) - mstore(add(m, 0x20), caller()) - mstore(add(m, 0x40), from) - mstore(add(m, 0x60), id) - mstore(add(m, 0x80), amount) - mstore(add(m, 0xa0), 0xa0) - calldatacopy(add(m, 0xc0), sub(data.offset, 0x20), add(0x20, data.length)) - // Revert if the call reverts. - if iszero(call(gas(), to, 0, add(m, 0x1c), add(0xc4, data.length), m, 0x20)) { - if returndatasize() { - // Bubble up the revert if the call reverts. - returndatacopy(m, 0x00, returndatasize()) - revert(m, returndatasize()) - } - } - // Load the returndata and compare it with the function selector. - if iszero(eq(mload(m), shl(224, 0xf23a6e61))) { - mstore(0x00, 0x9c05499b) // `TransferToNonERC1155ReceiverImplementer()`. - revert(0x1c, 0x04) - } - } - } - } - - /// @dev Transfers `amounts` of `ids` from `from` to `to`. - /// - /// Requirements: - /// - `to` cannot be the zero address. - /// - `from` must have at least `amount` of `id`. - /// - `ids` and `amounts` must have the same length. - /// - If the caller is not `from`, - /// it must be approved to manage the tokens of `from`. - /// - If `to` refers to a smart contract, it must implement - /// {ERC1155-onERC1155BatchReveived}, which is called upon a batch transfer. - /// - /// Emits a {TransferBatch} event. - function safeBatchTransferFrom( - address from, - address to, - uint256[] calldata ids, - uint256[] calldata amounts, - bytes calldata data - ) public virtual { - if (_useBeforeTokenTransfer()) { - _beforeTokenTransfer(from, to, ids, amounts, data); - } - /// @solidity memory-safe-assembly - assembly { - if iszero(eq(ids.length, amounts.length)) { - mstore(0x00, 0x3b800a46) // `ArrayLengthsMismatch()`. - revert(0x1c, 0x04) - } - let fromSlotSeed := or(_ERC1155_MASTER_SLOT_SEED, shl(96, from)) - let toSlotSeed := or(_ERC1155_MASTER_SLOT_SEED, shl(96, to)) - mstore(0x20, fromSlotSeed) - // Clear the upper 96 bits. - from := shr(96, fromSlotSeed) - to := shr(96, toSlotSeed) - // Revert if `to` is the zero address. - if iszero(to) { - mstore(0x00, 0xea553b34) // `TransferToZeroAddress()`. - revert(0x1c, 0x04) - } - // If the caller is not `from`, do the authorization check. - if iszero(eq(caller(), from)) { - mstore(0x00, caller()) - if iszero(sload(keccak256(0x0c, 0x34))) { - mstore(0x00, 0x4b6e7f18) // `NotOwnerNorApproved()`. - revert(0x1c, 0x04) - } - } - // Loop through all the `ids` and update the balances. - { - for { let i := shl(5, ids.length) } i {} { - i := sub(i, 0x20) - let amount := calldataload(add(amounts.offset, i)) - // Subtract and store the updated balance of `from`. - { - mstore(0x20, fromSlotSeed) - mstore(0x00, calldataload(add(ids.offset, i))) - let fromBalanceSlot := keccak256(0x00, 0x40) - let fromBalance := sload(fromBalanceSlot) - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - sstore(fromBalanceSlot, sub(fromBalance, amount)) - } - // Increase and store the updated balance of `to`. - { - mstore(0x20, toSlotSeed) - let toBalanceSlot := keccak256(0x00, 0x40) - let toBalanceBefore := sload(toBalanceSlot) - let toBalanceAfter := add(toBalanceBefore, amount) - if lt(toBalanceAfter, toBalanceBefore) { - mstore(0x00, 0x01336cea) // `AccountBalanceOverflow()`. - revert(0x1c, 0x04) - } - sstore(toBalanceSlot, toBalanceAfter) - } - } - } - // Emit a {TransferBatch} event. - { - let m := mload(0x40) - // Copy the `ids`. - mstore(m, 0x40) - let n := add(0x20, shl(5, ids.length)) - let o := add(m, 0x40) - calldatacopy(o, sub(ids.offset, 0x20), n) - // Copy the `amounts`. - mstore(add(m, 0x20), add(0x40, n)) - calldatacopy(add(o, n), sub(amounts.offset, 0x20), n) - // Do the emit. - log4(m, add(add(n, n), 0x40), _TRANSFER_BATCH_EVENT_SIGNATURE, caller(), from, to) - } - } - if (_useAfterTokenTransfer()) { - _afterTokenTransferCalldata(from, to, ids, amounts, data); - } - /// @solidity memory-safe-assembly - assembly { - // Do the {onERC1155BatchReceived} check if `to` is a smart contract. - if extcodesize(to) { - mstore(0x00, to) // Cache `to` to prevent stack too deep. - let m := mload(0x40) - // Prepare the calldata. - // `onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)`. - mstore(m, 0xbc197c81) - mstore(add(m, 0x20), caller()) - mstore(add(m, 0x40), from) - // Copy the `ids`. - mstore(add(m, 0x60), 0xa0) - let n := add(0x20, shl(5, ids.length)) - let o := add(m, 0xc0) - calldatacopy(o, sub(ids.offset, 0x20), n) - // Copy the `amounts`. - let s := add(0xa0, n) - mstore(add(m, 0x80), s) - calldatacopy(add(o, n), sub(amounts.offset, 0x20), n) - // Copy the `data`. - mstore(add(m, 0xa0), add(s, n)) - calldatacopy(add(o, add(n, n)), sub(data.offset, 0x20), add(0x20, data.length)) - let nAll := add(0xc4, add(data.length, add(n, n))) - // Revert if the call reverts. - if iszero(call(gas(), mload(0x00), 0, add(m, 0x1c), nAll, m, 0x20)) { - if returndatasize() { - // Bubble up the revert if the call reverts. - returndatacopy(m, 0x00, returndatasize()) - revert(m, returndatasize()) - } - } - // Load the returndata and compare it with the function selector. - if iszero(eq(mload(m), shl(224, 0xbc197c81))) { - mstore(0x00, 0x9c05499b) // `TransferToNonERC1155ReceiverImplementer()`. - revert(0x1c, 0x04) - } - } - } - } - - /// @dev Returns the amounts of `ids` for `owners. - /// - /// Requirements: - /// - `owners` and `ids` must have the same length. - function balanceOfBatch(address[] calldata owners, uint256[] calldata ids) - public - view - virtual - returns (uint256[] memory balances) - { - /// @solidity memory-safe-assembly - assembly { - if iszero(eq(ids.length, owners.length)) { - mstore(0x00, 0x3b800a46) // `ArrayLengthsMismatch()`. - revert(0x1c, 0x04) - } - balances := mload(0x40) - mstore(balances, ids.length) - let o := add(balances, 0x20) - let i := shl(5, ids.length) - mstore(0x40, add(i, o)) - // Loop through all the `ids` and load the balances. - for {} i {} { - i := sub(i, 0x20) - let owner := calldataload(add(owners.offset, i)) - mstore(0x20, or(_ERC1155_MASTER_SLOT_SEED, shl(96, owner))) - mstore(0x00, calldataload(add(ids.offset, i))) - mstore(add(o, i), sload(keccak256(0x00, 0x40))) - } - } - } - - /// @dev Returns true if this contract implements the interface defined by `interfaceId`. - /// See: https://eips.ethereum.org/EIPS/eip-165 - /// This function call must use less than 30000 gas. - function supportsInterface(bytes4 interfaceId) public view virtual returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - let s := shr(224, interfaceId) - // ERC165: 0x01ffc9a7, ERC1155: 0xd9b67a26, ERC1155MetadataURI: 0x0e89341c. - result := or(or(eq(s, 0x01ffc9a7), eq(s, 0xd9b67a26)), eq(s, 0x0e89341c)) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL MINT FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Mints `amount` of `id` to `to`. - /// - /// Requirements: - /// - `to` cannot be the zero address. - /// - If `to` refers to a smart contract, it must implement - /// {ERC1155-onERC1155Reveived}, which is called upon a batch transfer. - /// - /// Emits a {Transfer} event. - function _mint(address to, uint256 id, uint256 amount, bytes memory data) internal virtual { - if (_useBeforeTokenTransfer()) { - _beforeTokenTransfer(address(0), to, _single(id), _single(amount), data); - } - /// @solidity memory-safe-assembly - assembly { - let to_ := shl(96, to) - // Revert if `to` is the zero address. - if iszero(to_) { - mstore(0x00, 0xea553b34) // `TransferToZeroAddress()`. - revert(0x1c, 0x04) - } - // Increase and store the updated balance of `to`. - { - mstore(0x20, _ERC1155_MASTER_SLOT_SEED) - mstore(0x14, to) - mstore(0x00, id) - let toBalanceSlot := keccak256(0x00, 0x40) - let toBalanceBefore := sload(toBalanceSlot) - let toBalanceAfter := add(toBalanceBefore, amount) - if lt(toBalanceAfter, toBalanceBefore) { - mstore(0x00, 0x01336cea) // `AccountBalanceOverflow()`. - revert(0x1c, 0x04) - } - sstore(toBalanceSlot, toBalanceAfter) - } - // Emit a {TransferSingle} event. - mstore(0x20, amount) - log4(0x00, 0x40, _TRANSFER_SINGLE_EVENT_SIGNATURE, caller(), 0, shr(96, to_)) - } - if (_useAfterTokenTransfer()) { - _afterTokenTransfer(address(0), to, _single(id), _single(amount), data); - } - if (_hasCode(to)) _checkOnERC1155Received(address(0), to, id, amount, data); - } - - /// @dev Mints `amounts` of `ids` to `to`. - /// - /// Requirements: - /// - `to` cannot be the zero address. - /// - `ids` and `amounts` must have the same length. - /// - If `to` refers to a smart contract, it must implement - /// {ERC1155-onERC1155BatchReveived}, which is called upon a batch transfer. - /// - /// Emits a {TransferBatch} event. - function _batchMint( - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) internal virtual { - if (_useBeforeTokenTransfer()) { - _beforeTokenTransfer(address(0), to, ids, amounts, data); - } - /// @solidity memory-safe-assembly - assembly { - if iszero(eq(mload(ids), mload(amounts))) { - mstore(0x00, 0x3b800a46) // `ArrayLengthsMismatch()`. - revert(0x1c, 0x04) - } - let to_ := shl(96, to) - // Revert if `to` is the zero address. - if iszero(to_) { - mstore(0x00, 0xea553b34) // `TransferToZeroAddress()`. - revert(0x1c, 0x04) - } - // Loop through all the `ids` and update the balances. - { - mstore(0x20, or(_ERC1155_MASTER_SLOT_SEED, to_)) - for { let i := shl(5, mload(ids)) } i { i := sub(i, 0x20) } { - let amount := mload(add(amounts, i)) - // Increase and store the updated balance of `to`. - { - mstore(0x00, mload(add(ids, i))) - let toBalanceSlot := keccak256(0x00, 0x40) - let toBalanceBefore := sload(toBalanceSlot) - let toBalanceAfter := add(toBalanceBefore, amount) - if lt(toBalanceAfter, toBalanceBefore) { - mstore(0x00, 0x01336cea) // `AccountBalanceOverflow()`. - revert(0x1c, 0x04) - } - sstore(toBalanceSlot, toBalanceAfter) - } - } - } - // Emit a {TransferBatch} event. - { - let m := mload(0x40) - // Copy the `ids`. - mstore(m, 0x40) - let n := add(0x20, shl(5, mload(ids))) - let o := add(m, 0x40) - pop(staticcall(gas(), 4, ids, n, o, n)) - // Copy the `amounts`. - mstore(add(m, 0x20), add(0x40, returndatasize())) - o := add(o, returndatasize()) - n := add(0x20, shl(5, mload(amounts))) - pop(staticcall(gas(), 4, amounts, n, o, n)) - n := sub(add(o, returndatasize()), m) - // Do the emit. - log4(m, n, _TRANSFER_BATCH_EVENT_SIGNATURE, caller(), 0, shr(96, to_)) - } - } - if (_useAfterTokenTransfer()) { - _afterTokenTransfer(address(0), to, ids, amounts, data); - } - if (_hasCode(to)) _checkOnERC1155BatchReceived(address(0), to, ids, amounts, data); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL BURN FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Equivalent to `_burn(address(0), from, id, amount)`. - function _burn(address from, uint256 id, uint256 amount) internal virtual { - _burn(address(0), from, id, amount); - } - - /// @dev Destroys `amount` of `id` from `from`. - /// - /// Requirements: - /// - `from` must have at least `amount` of `id`. - /// - If `by` is not the zero address, it must be either `from`, - /// or approved to manage the tokens of `from`. - /// - /// Emits a {Transfer} event. - function _burn(address by, address from, uint256 id, uint256 amount) internal virtual { - if (_useBeforeTokenTransfer()) { - _beforeTokenTransfer(from, address(0), _single(id), _single(amount), ""); - } - /// @solidity memory-safe-assembly - assembly { - let from_ := shl(96, from) - mstore(0x20, or(_ERC1155_MASTER_SLOT_SEED, from_)) - // If `by` is not the zero address, and not equal to `from`, - // check if it is approved to manage all the tokens of `from`. - if iszero(or(iszero(shl(96, by)), eq(shl(96, by), from_))) { - mstore(0x00, by) - if iszero(sload(keccak256(0x0c, 0x34))) { - mstore(0x00, 0x4b6e7f18) // `NotOwnerNorApproved()`. - revert(0x1c, 0x04) - } - } - // Decrease and store the updated balance of `from`. - { - mstore(0x00, id) - let fromBalanceSlot := keccak256(0x00, 0x40) - let fromBalance := sload(fromBalanceSlot) - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - sstore(fromBalanceSlot, sub(fromBalance, amount)) - } - // Emit a {TransferSingle} event. - mstore(0x20, amount) - log4(0x00, 0x40, _TRANSFER_SINGLE_EVENT_SIGNATURE, caller(), shr(96, from_), 0) - } - if (_useAfterTokenTransfer()) { - _afterTokenTransfer(from, address(0), _single(id), _single(amount), ""); - } - } - - /// @dev Equivalent to `_batchBurn(address(0), from, ids, amounts)`. - function _batchBurn(address from, uint256[] memory ids, uint256[] memory amounts) - internal - virtual - { - _batchBurn(address(0), from, ids, amounts); - } - - /// @dev Destroys `amounts` of `ids` from `from`. - /// - /// Requirements: - /// - `ids` and `amounts` must have the same length. - /// - `from` must have at least `amounts` of `ids`. - /// - If `by` is not the zero address, it must be either `from`, - /// or approved to manage the tokens of `from`. - /// - /// Emits a {TransferBatch} event. - function _batchBurn(address by, address from, uint256[] memory ids, uint256[] memory amounts) - internal - virtual - { - if (_useBeforeTokenTransfer()) { - _beforeTokenTransfer(from, address(0), ids, amounts, ""); - } - /// @solidity memory-safe-assembly - assembly { - if iszero(eq(mload(ids), mload(amounts))) { - mstore(0x00, 0x3b800a46) // `ArrayLengthsMismatch()`. - revert(0x1c, 0x04) - } - let from_ := shl(96, from) - mstore(0x20, or(_ERC1155_MASTER_SLOT_SEED, from_)) - // If `by` is not the zero address, and not equal to `from`, - // check if it is approved to manage all the tokens of `from`. - let by_ := shl(96, by) - if iszero(or(iszero(by_), eq(by_, from_))) { - mstore(0x00, by) - if iszero(sload(keccak256(0x0c, 0x34))) { - mstore(0x00, 0x4b6e7f18) // `NotOwnerNorApproved()`. - revert(0x1c, 0x04) - } - } - // Loop through all the `ids` and update the balances. - { - for { let i := shl(5, mload(ids)) } i { i := sub(i, 0x20) } { - let amount := mload(add(amounts, i)) - // Decrease and store the updated balance of `from`. - { - mstore(0x00, mload(add(ids, i))) - let fromBalanceSlot := keccak256(0x00, 0x40) - let fromBalance := sload(fromBalanceSlot) - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - sstore(fromBalanceSlot, sub(fromBalance, amount)) - } - } - } - // Emit a {TransferBatch} event. - { - let m := mload(0x40) - // Copy the `ids`. - mstore(m, 0x40) - let n := add(0x20, shl(5, mload(ids))) - let o := add(m, 0x40) - pop(staticcall(gas(), 4, ids, n, o, n)) - // Copy the `amounts`. - mstore(add(m, 0x20), add(0x40, returndatasize())) - o := add(o, returndatasize()) - n := add(0x20, shl(5, mload(amounts))) - pop(staticcall(gas(), 4, amounts, n, o, n)) - n := sub(add(o, returndatasize()), m) - // Do the emit. - log4(m, n, _TRANSFER_BATCH_EVENT_SIGNATURE, caller(), shr(96, from_), 0) - } - } - if (_useAfterTokenTransfer()) { - _afterTokenTransfer(from, address(0), ids, amounts, ""); - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL APPROVAL FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Approve or remove the `operator` as an operator for `by`, - /// without authorization checks. - /// - /// Emits a {ApprovalForAll} event. - function _setApprovalForAll(address by, address operator, bool isApproved) internal virtual { - /// @solidity memory-safe-assembly - assembly { - // Convert to 0 or 1. - isApproved := iszero(iszero(isApproved)) - // Update the `isApproved` for (`by`, `operator`). - mstore(0x20, _ERC1155_MASTER_SLOT_SEED) - mstore(0x14, by) - mstore(0x00, operator) - sstore(keccak256(0x0c, 0x34), isApproved) - // Emit the {ApprovalForAll} event. - mstore(0x00, isApproved) - let m := shr(96, not(0)) - log3(0x00, 0x20, _APPROVAL_FOR_ALL_EVENT_SIGNATURE, and(m, by), and(m, operator)) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL TRANSFER FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Equivalent to `_safeTransfer(address(0), from, to, id, amount, data)`. - function _safeTransfer(address from, address to, uint256 id, uint256 amount, bytes memory data) - internal - virtual - { - _safeTransfer(address(0), from, to, id, amount, data); - } - - /// @dev Transfers `amount` of `id` from `from` to `to`. - /// - /// Requirements: - /// - `to` cannot be the zero address. - /// - `from` must have at least `amount` of `id`. - /// - If `by` is not the zero address, it must be either `from`, - /// or approved to manage the tokens of `from`. - /// - If `to` refers to a smart contract, it must implement - /// {ERC1155-onERC1155Reveived}, which is called upon a batch transfer. - /// - /// Emits a {Transfer} event. - function _safeTransfer( - address by, - address from, - address to, - uint256 id, - uint256 amount, - bytes memory data - ) internal virtual { - if (_useBeforeTokenTransfer()) { - _beforeTokenTransfer(from, to, _single(id), _single(amount), data); - } - /// @solidity memory-safe-assembly - assembly { - let from_ := shl(96, from) - let to_ := shl(96, to) - // Revert if `to` is the zero address. - if iszero(to_) { - mstore(0x00, 0xea553b34) // `TransferToZeroAddress()`. - revert(0x1c, 0x04) - } - mstore(0x20, or(_ERC1155_MASTER_SLOT_SEED, from_)) - // If `by` is not the zero address, and not equal to `from`, - // check if it is approved to manage all the tokens of `from`. - let by_ := shl(96, by) - if iszero(or(iszero(by_), eq(by_, from_))) { - mstore(0x00, by) - if iszero(sload(keccak256(0x0c, 0x34))) { - mstore(0x00, 0x4b6e7f18) // `NotOwnerNorApproved()`. - revert(0x1c, 0x04) - } - } - // Subtract and store the updated balance of `from`. - { - mstore(0x00, id) - let fromBalanceSlot := keccak256(0x00, 0x40) - let fromBalance := sload(fromBalanceSlot) - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - sstore(fromBalanceSlot, sub(fromBalance, amount)) - } - // Increase and store the updated balance of `to`. - { - mstore(0x20, or(_ERC1155_MASTER_SLOT_SEED, to_)) - let toBalanceSlot := keccak256(0x00, 0x40) - let toBalanceBefore := sload(toBalanceSlot) - let toBalanceAfter := add(toBalanceBefore, amount) - if lt(toBalanceAfter, toBalanceBefore) { - mstore(0x00, 0x01336cea) // `AccountBalanceOverflow()`. - revert(0x1c, 0x04) - } - sstore(toBalanceSlot, toBalanceAfter) - } - // Emit a {TransferSingle} event. - mstore(0x20, amount) - // forgefmt: disable-next-line - log4(0x00, 0x40, _TRANSFER_SINGLE_EVENT_SIGNATURE, caller(), shr(96, from_), shr(96, to_)) - } - if (_useAfterTokenTransfer()) { - _afterTokenTransfer(from, to, _single(id), _single(amount), data); - } - if (_hasCode(to)) _checkOnERC1155Received(from, to, id, amount, data); - } - - /// @dev Equivalent to `_safeBatchTransfer(address(0), from, to, ids, amounts, data)`. - function _safeBatchTransfer( - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) internal virtual { - _safeBatchTransfer(address(0), from, to, ids, amounts, data); - } - - /// @dev Transfers `amounts` of `ids` from `from` to `to`. - /// - /// Requirements: - /// - `to` cannot be the zero address. - /// - `ids` and `amounts` must have the same length. - /// - `from` must have at least `amounts` of `ids`. - /// - If `by` is not the zero address, it must be either `from`, - /// or approved to manage the tokens of `from`. - /// - If `to` refers to a smart contract, it must implement - /// {ERC1155-onERC1155BatchReveived}, which is called upon a batch transfer. - /// - /// Emits a {TransferBatch} event. - function _safeBatchTransfer( - address by, - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) internal virtual { - if (_useBeforeTokenTransfer()) { - _beforeTokenTransfer(from, to, ids, amounts, data); - } - /// @solidity memory-safe-assembly - assembly { - if iszero(eq(mload(ids), mload(amounts))) { - mstore(0x00, 0x3b800a46) // `ArrayLengthsMismatch()`. - revert(0x1c, 0x04) - } - let from_ := shl(96, from) - let to_ := shl(96, to) - // Revert if `to` is the zero address. - if iszero(to_) { - mstore(0x00, 0xea553b34) // `TransferToZeroAddress()`. - revert(0x1c, 0x04) - } - let fromSlotSeed := or(_ERC1155_MASTER_SLOT_SEED, from_) - let toSlotSeed := or(_ERC1155_MASTER_SLOT_SEED, to_) - mstore(0x20, fromSlotSeed) - // If `by` is not the zero address, and not equal to `from`, - // check if it is approved to manage all the tokens of `from`. - let by_ := shl(96, by) - if iszero(or(iszero(by_), eq(by_, from_))) { - mstore(0x00, by) - if iszero(sload(keccak256(0x0c, 0x34))) { - mstore(0x00, 0x4b6e7f18) // `NotOwnerNorApproved()`. - revert(0x1c, 0x04) - } - } - // Loop through all the `ids` and update the balances. - { - for { let i := shl(5, mload(ids)) } i { i := sub(i, 0x20) } { - let amount := mload(add(amounts, i)) - // Subtract and store the updated balance of `from`. - { - mstore(0x20, fromSlotSeed) - mstore(0x00, mload(add(ids, i))) - let fromBalanceSlot := keccak256(0x00, 0x40) - let fromBalance := sload(fromBalanceSlot) - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - sstore(fromBalanceSlot, sub(fromBalance, amount)) - } - // Increase and store the updated balance of `to`. - { - mstore(0x20, toSlotSeed) - let toBalanceSlot := keccak256(0x00, 0x40) - let toBalanceBefore := sload(toBalanceSlot) - let toBalanceAfter := add(toBalanceBefore, amount) - if lt(toBalanceAfter, toBalanceBefore) { - mstore(0x00, 0x01336cea) // `AccountBalanceOverflow()`. - revert(0x1c, 0x04) - } - sstore(toBalanceSlot, toBalanceAfter) - } - } - } - // Emit a {TransferBatch} event. - { - let m := mload(0x40) - // Copy the `ids`. - mstore(m, 0x40) - let n := add(0x20, shl(5, mload(ids))) - let o := add(m, 0x40) - pop(staticcall(gas(), 4, ids, n, o, n)) - // Copy the `amounts`. - mstore(add(m, 0x20), add(0x40, returndatasize())) - o := add(o, returndatasize()) - n := add(0x20, shl(5, mload(amounts))) - pop(staticcall(gas(), 4, amounts, n, o, n)) - n := sub(add(o, returndatasize()), m) - // Do the emit. - log4(m, n, _TRANSFER_BATCH_EVENT_SIGNATURE, caller(), shr(96, from_), shr(96, to_)) - } - } - if (_useAfterTokenTransfer()) { - _afterTokenTransfer(from, to, ids, amounts, data); - } - if (_hasCode(to)) _checkOnERC1155BatchReceived(from, to, ids, amounts, data); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* HOOKS FOR OVERRIDING */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Override this function to return true if `_beforeTokenTransfer` is used. - /// This is to help the compiler avoid producing dead bytecode. - function _useBeforeTokenTransfer() internal view virtual returns (bool) { - return false; - } - - /// @dev Hook that is called before any token transfer. - /// This includes minting and burning, as well as batched variants. - /// - /// The same hook is called on both single and batched variants. - /// For single transfers, the length of the `id` and `amount` arrays are 1. - function _beforeTokenTransfer( - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) internal virtual {} - - /// @dev Override this function to return true if `_afterTokenTransfer` is used. - /// This is to help the compiler avoid producing dead bytecode. - function _useAfterTokenTransfer() internal view virtual returns (bool) { - return false; - } - - /// @dev Hook that is called after any token transfer. - /// This includes minting and burning, as well as batched variants. - /// - /// The same hook is called on both single and batched variants. - /// For single transfers, the length of the `id` and `amount` arrays are 1. - function _afterTokenTransfer( - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) internal virtual {} - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PRIVATE HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Helper for calling the `_afterTokenTransfer` hook. - /// This is to help the compiler avoid producing dead bytecode. - function _afterTokenTransferCalldata( - address from, - address to, - uint256[] calldata ids, - uint256[] calldata amounts, - bytes calldata data - ) private { - if (_useAfterTokenTransfer()) { - _afterTokenTransfer(from, to, ids, amounts, data); - } - } - - /// @dev Returns if `a` has bytecode of non-zero length. - function _hasCode(address a) private view returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - result := extcodesize(a) // Can handle dirty upper bits. - } - } - - /// @dev Perform a call to invoke {IERC1155Receiver-onERC1155Received} on `to`. - /// Reverts if the target does not support the function correctly. - function _checkOnERC1155Received( - address from, - address to, - uint256 id, - uint256 amount, - bytes memory data - ) private { - /// @solidity memory-safe-assembly - assembly { - // Prepare the calldata. - let m := mload(0x40) - // `onERC1155Received(address,address,uint256,uint256,bytes)`. - mstore(m, 0xf23a6e61) - mstore(add(m, 0x20), caller()) - mstore(add(m, 0x40), shr(96, shl(96, from))) - mstore(add(m, 0x60), id) - mstore(add(m, 0x80), amount) - mstore(add(m, 0xa0), 0xa0) - let n := mload(data) - mstore(add(m, 0xc0), n) - if n { pop(staticcall(gas(), 4, add(data, 0x20), n, add(m, 0xe0), n)) } - // Revert if the call reverts. - if iszero(call(gas(), to, 0, add(m, 0x1c), add(0xc4, n), m, 0x20)) { - if returndatasize() { - // Bubble up the revert if the call reverts. - returndatacopy(m, 0x00, returndatasize()) - revert(m, returndatasize()) - } - } - // Load the returndata and compare it with the function selector. - if iszero(eq(mload(m), shl(224, 0xf23a6e61))) { - mstore(0x00, 0x9c05499b) // `TransferToNonERC1155ReceiverImplementer()`. - revert(0x1c, 0x04) - } - } - } - - /// @dev Perform a call to invoke {IERC1155Receiver-onERC1155BatchReceived} on `to`. - /// Reverts if the target does not support the function correctly. - function _checkOnERC1155BatchReceived( - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) private { - /// @solidity memory-safe-assembly - assembly { - // Prepare the calldata. - let m := mload(0x40) - // `onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)`. - mstore(m, 0xbc197c81) - mstore(add(m, 0x20), caller()) - mstore(add(m, 0x40), shr(96, shl(96, from))) - // Copy the `ids`. - mstore(add(m, 0x60), 0xa0) - let n := add(0x20, shl(5, mload(ids))) - let o := add(m, 0xc0) - pop(staticcall(gas(), 4, ids, n, o, n)) - // Copy the `amounts`. - let s := add(0xa0, returndatasize()) - mstore(add(m, 0x80), s) - o := add(o, returndatasize()) - n := add(0x20, shl(5, mload(amounts))) - pop(staticcall(gas(), 4, amounts, n, o, n)) - // Copy the `data`. - mstore(add(m, 0xa0), add(s, returndatasize())) - o := add(o, returndatasize()) - n := add(0x20, mload(data)) - pop(staticcall(gas(), 4, data, n, o, n)) - n := sub(add(o, returndatasize()), add(m, 0x1c)) - // Revert if the call reverts. - if iszero(call(gas(), to, 0, add(m, 0x1c), n, m, 0x20)) { - if returndatasize() { - // Bubble up the revert if the call reverts. - returndatacopy(m, 0x00, returndatasize()) - revert(m, returndatasize()) - } - } - // Load the returndata and compare it with the function selector. - if iszero(eq(mload(m), shl(224, 0xbc197c81))) { - mstore(0x00, 0x9c05499b) // `TransferToNonERC1155ReceiverImplementer()`. - revert(0x1c, 0x04) - } - } - } - - /// @dev Returns `x` in an array with a single element. - function _single(uint256 x) private pure returns (uint256[] memory result) { - /// @solidity memory-safe-assembly - assembly { - result := mload(0x40) - mstore(0x40, add(result, 0x40)) - mstore(result, 1) - mstore(add(result, 0x20), x) - } - } -} diff --git a/lib/solady/src/tokens/ERC20.sol b/lib/solady/src/tokens/ERC20.sol deleted file mode 100644 index 98b5d67..0000000 --- a/lib/solady/src/tokens/ERC20.sol +++ /dev/null @@ -1,546 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Simple ERC20 + EIP-2612 implementation. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/tokens/ERC20.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC20.sol) -/// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/token/ERC20/ERC20.sol) -/// -/// @dev Note: -/// - The ERC20 standard allows minting and transferring to and from the zero address, -/// minting and transferring zero tokens, as well as self-approvals. -/// For performance, this implementation WILL NOT revert for such actions. -/// Please add any checks with overrides if desired. -/// - The `permit` function uses the ecrecover precompile (0x1). -/// -/// If you are overriding: -/// - NEVER violate the ERC20 invariant: -/// the total sum of all balances must be equal to `totalSupply()`. -/// - Check that the overridden function is actually used in the function you want to -/// change the behavior of. Much of the code has been manually inlined for performance. -abstract contract ERC20 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The total supply has overflowed. - error TotalSupplyOverflow(); - - /// @dev The allowance has overflowed. - error AllowanceOverflow(); - - /// @dev The allowance has underflowed. - error AllowanceUnderflow(); - - /// @dev Insufficient balance. - error InsufficientBalance(); - - /// @dev Insufficient allowance. - error InsufficientAllowance(); - - /// @dev The permit is invalid. - error InvalidPermit(); - - /// @dev The permit has expired. - error PermitExpired(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EVENTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Emitted when `amount` tokens is transferred from `from` to `to`. - event Transfer(address indexed from, address indexed to, uint256 amount); - - /// @dev Emitted when `amount` tokens is approved by `owner` to be used by `spender`. - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /// @dev `keccak256(bytes("Transfer(address,address,uint256)"))`. - uint256 private constant _TRANSFER_EVENT_SIGNATURE = - 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef; - - /// @dev `keccak256(bytes("Approval(address,address,uint256)"))`. - uint256 private constant _APPROVAL_EVENT_SIGNATURE = - 0x8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STORAGE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The storage slot for the total supply. - uint256 private constant _TOTAL_SUPPLY_SLOT = 0x05345cdf77eb68f44c; - - /// @dev The balance slot of `owner` is given by: - /// ``` - /// mstore(0x0c, _BALANCE_SLOT_SEED) - /// mstore(0x00, owner) - /// let balanceSlot := keccak256(0x0c, 0x20) - /// ``` - uint256 private constant _BALANCE_SLOT_SEED = 0x87a211a2; - - /// @dev The allowance slot of (`owner`, `spender`) is given by: - /// ``` - /// mstore(0x20, spender) - /// mstore(0x0c, _ALLOWANCE_SLOT_SEED) - /// mstore(0x00, owner) - /// let allowanceSlot := keccak256(0x0c, 0x34) - /// ``` - uint256 private constant _ALLOWANCE_SLOT_SEED = 0x7f5e9f20; - - /// @dev The nonce slot of `owner` is given by: - /// ``` - /// mstore(0x0c, _NONCES_SLOT_SEED) - /// mstore(0x00, owner) - /// let nonceSlot := keccak256(0x0c, 0x20) - /// ``` - uint256 private constant _NONCES_SLOT_SEED = 0x38377508; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev `(_NONCES_SLOT_SEED << 16) | 0x1901`. - uint256 private constant _NONCES_SLOT_SEED_WITH_SIGNATURE_PREFIX = 0x383775081901; - - /// @dev `keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)")`. - bytes32 private constant _DOMAIN_TYPEHASH = - 0x8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f; - - /// @dev `keccak256("1")`. - bytes32 private constant _VERSION_HASH = - 0xc89efdaa54c0f20c7adf612882df0950f5a951637e0307cdcb4c672f298b8bc6; - - /// @dev `keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)")`. - bytes32 private constant _PERMIT_TYPEHASH = - 0x6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c9; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC20 METADATA */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the name of the token. - function name() public view virtual returns (string memory); - - /// @dev Returns the symbol of the token. - function symbol() public view virtual returns (string memory); - - /// @dev Returns the decimals places of the token. - function decimals() public view virtual returns (uint8) { - return 18; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC20 */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the amount of tokens in existence. - function totalSupply() public view virtual returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - result := sload(_TOTAL_SUPPLY_SLOT) - } - } - - /// @dev Returns the amount of tokens owned by `owner`. - function balanceOf(address owner) public view virtual returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x0c, _BALANCE_SLOT_SEED) - mstore(0x00, owner) - result := sload(keccak256(0x0c, 0x20)) - } - } - - /// @dev Returns the amount of tokens that `spender` can spend on behalf of `owner`. - function allowance(address owner, address spender) - public - view - virtual - returns (uint256 result) - { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, spender) - mstore(0x0c, _ALLOWANCE_SLOT_SEED) - mstore(0x00, owner) - result := sload(keccak256(0x0c, 0x34)) - } - } - - /// @dev Sets `amount` as the allowance of `spender` over the caller's tokens. - /// - /// Emits a {Approval} event. - function approve(address spender, uint256 amount) public virtual returns (bool) { - /// @solidity memory-safe-assembly - assembly { - // Compute the allowance slot and store the amount. - mstore(0x20, spender) - mstore(0x0c, _ALLOWANCE_SLOT_SEED) - mstore(0x00, caller()) - sstore(keccak256(0x0c, 0x34), amount) - // Emit the {Approval} event. - mstore(0x00, amount) - log3(0x00, 0x20, _APPROVAL_EVENT_SIGNATURE, caller(), shr(96, mload(0x2c))) - } - return true; - } - - /// @dev Transfer `amount` tokens from the caller to `to`. - /// - /// Requirements: - /// - `from` must at least have `amount`. - /// - /// Emits a {Transfer} event. - function transfer(address to, uint256 amount) public virtual returns (bool) { - _beforeTokenTransfer(msg.sender, to, amount); - /// @solidity memory-safe-assembly - assembly { - // Compute the balance slot and load its value. - mstore(0x0c, _BALANCE_SLOT_SEED) - mstore(0x00, caller()) - let fromBalanceSlot := keccak256(0x0c, 0x20) - let fromBalance := sload(fromBalanceSlot) - // Revert if insufficient balance. - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - // Subtract and store the updated balance. - sstore(fromBalanceSlot, sub(fromBalance, amount)) - // Compute the balance slot of `to`. - mstore(0x00, to) - let toBalanceSlot := keccak256(0x0c, 0x20) - // Add and store the updated balance of `to`. - // Will not overflow because the sum of all user balances - // cannot exceed the maximum uint256 value. - sstore(toBalanceSlot, add(sload(toBalanceSlot), amount)) - // Emit the {Transfer} event. - mstore(0x20, amount) - log3(0x20, 0x20, _TRANSFER_EVENT_SIGNATURE, caller(), shr(96, mload(0x0c))) - } - _afterTokenTransfer(msg.sender, to, amount); - return true; - } - - /// @dev Transfers `amount` tokens from `from` to `to`. - /// - /// Note: Does not update the allowance if it is the maximum uint256 value. - /// - /// Requirements: - /// - `from` must at least have `amount`. - /// - The caller must have at least `amount` of allowance to transfer the tokens of `from`. - /// - /// Emits a {Transfer} event. - function transferFrom(address from, address to, uint256 amount) public virtual returns (bool) { - _beforeTokenTransfer(from, to, amount); - /// @solidity memory-safe-assembly - assembly { - let from_ := shl(96, from) - // Compute the allowance slot and load its value. - mstore(0x20, caller()) - mstore(0x0c, or(from_, _ALLOWANCE_SLOT_SEED)) - let allowanceSlot := keccak256(0x0c, 0x34) - let allowance_ := sload(allowanceSlot) - // If the allowance is not the maximum uint256 value. - if add(allowance_, 1) { - // Revert if the amount to be transferred exceeds the allowance. - if gt(amount, allowance_) { - mstore(0x00, 0x13be252b) // `InsufficientAllowance()`. - revert(0x1c, 0x04) - } - // Subtract and store the updated allowance. - sstore(allowanceSlot, sub(allowance_, amount)) - } - // Compute the balance slot and load its value. - mstore(0x0c, or(from_, _BALANCE_SLOT_SEED)) - let fromBalanceSlot := keccak256(0x0c, 0x20) - let fromBalance := sload(fromBalanceSlot) - // Revert if insufficient balance. - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - // Subtract and store the updated balance. - sstore(fromBalanceSlot, sub(fromBalance, amount)) - // Compute the balance slot of `to`. - mstore(0x00, to) - let toBalanceSlot := keccak256(0x0c, 0x20) - // Add and store the updated balance of `to`. - // Will not overflow because the sum of all user balances - // cannot exceed the maximum uint256 value. - sstore(toBalanceSlot, add(sload(toBalanceSlot), amount)) - // Emit the {Transfer} event. - mstore(0x20, amount) - log3(0x20, 0x20, _TRANSFER_EVENT_SIGNATURE, shr(96, from_), shr(96, mload(0x0c))) - } - _afterTokenTransfer(from, to, amount); - return true; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EIP-2612 */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev For more performance, override to return the constant value - /// of `keccak256(bytes(name()))` if `name()` will never change. - function _constantNameHash() internal view virtual returns (bytes32 result) {} - - /// @dev Returns the current nonce for `owner`. - /// This value is used to compute the signature for EIP-2612 permit. - function nonces(address owner) public view virtual returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - // Compute the nonce slot and load its value. - mstore(0x0c, _NONCES_SLOT_SEED) - mstore(0x00, owner) - result := sload(keccak256(0x0c, 0x20)) - } - } - - /// @dev Sets `value` as the allowance of `spender` over the tokens of `owner`, - /// authorized by a signed approval by `owner`. - /// - /// Emits a {Approval} event. - function permit( - address owner, - address spender, - uint256 value, - uint256 deadline, - uint8 v, - bytes32 r, - bytes32 s - ) public virtual { - bytes32 nameHash = _constantNameHash(); - // We simply calculate it on-the-fly to allow for cases where the `name` may change. - if (nameHash == bytes32(0)) nameHash = keccak256(bytes(name())); - /// @solidity memory-safe-assembly - assembly { - // Revert if the block timestamp is greater than `deadline`. - if gt(timestamp(), deadline) { - mstore(0x00, 0x1a15a3cc) // `PermitExpired()`. - revert(0x1c, 0x04) - } - let m := mload(0x40) // Grab the free memory pointer. - // Clean the upper 96 bits. - owner := shr(96, shl(96, owner)) - spender := shr(96, shl(96, spender)) - // Compute the nonce slot and load its value. - mstore(0x0e, _NONCES_SLOT_SEED_WITH_SIGNATURE_PREFIX) - mstore(0x00, owner) - let nonceSlot := keccak256(0x0c, 0x20) - let nonceValue := sload(nonceSlot) - // Prepare the domain separator. - mstore(m, _DOMAIN_TYPEHASH) - mstore(add(m, 0x20), nameHash) - mstore(add(m, 0x40), _VERSION_HASH) - mstore(add(m, 0x60), chainid()) - mstore(add(m, 0x80), address()) - mstore(0x2e, keccak256(m, 0xa0)) - // Prepare the struct hash. - mstore(m, _PERMIT_TYPEHASH) - mstore(add(m, 0x20), owner) - mstore(add(m, 0x40), spender) - mstore(add(m, 0x60), value) - mstore(add(m, 0x80), nonceValue) - mstore(add(m, 0xa0), deadline) - mstore(0x4e, keccak256(m, 0xc0)) - // Prepare the ecrecover calldata. - mstore(0x00, keccak256(0x2c, 0x42)) - mstore(0x20, and(0xff, v)) - mstore(0x40, r) - mstore(0x60, s) - let t := staticcall(gas(), 1, 0, 0x80, 0x20, 0x20) - // If the ecrecover fails, the returndatasize will be 0x00, - // `owner` will be checked if it equals the hash at 0x00, - // which evaluates to false (i.e. 0), and we will revert. - // If the ecrecover succeeds, the returndatasize will be 0x20, - // `owner` will be compared against the returned address at 0x20. - if iszero(eq(mload(returndatasize()), owner)) { - mstore(0x00, 0xddafbaef) // `InvalidPermit()`. - revert(0x1c, 0x04) - } - // Increment and store the updated nonce. - sstore(nonceSlot, add(nonceValue, t)) // `t` is 1 if ecrecover succeeds. - // Compute the allowance slot and store the value. - // The `owner` is already at slot 0x20. - mstore(0x40, or(shl(160, _ALLOWANCE_SLOT_SEED), spender)) - sstore(keccak256(0x2c, 0x34), value) - // Emit the {Approval} event. - log3(add(m, 0x60), 0x20, _APPROVAL_EVENT_SIGNATURE, owner, spender) - mstore(0x40, m) // Restore the free memory pointer. - mstore(0x60, 0) // Restore the zero pointer. - } - } - - /// @dev Returns the EIP-712 domain separator for the EIP-2612 permit. - function DOMAIN_SEPARATOR() public view virtual returns (bytes32 result) { - bytes32 nameHash = _constantNameHash(); - // We simply calculate it on-the-fly to allow for cases where the `name` may change. - if (nameHash == bytes32(0)) nameHash = keccak256(bytes(name())); - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Grab the free memory pointer. - mstore(m, _DOMAIN_TYPEHASH) - mstore(add(m, 0x20), nameHash) - mstore(add(m, 0x40), _VERSION_HASH) - mstore(add(m, 0x60), chainid()) - mstore(add(m, 0x80), address()) - result := keccak256(m, 0xa0) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL MINT FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Mints `amount` tokens to `to`, increasing the total supply. - /// - /// Emits a {Transfer} event. - function _mint(address to, uint256 amount) internal virtual { - _beforeTokenTransfer(address(0), to, amount); - /// @solidity memory-safe-assembly - assembly { - let totalSupplyBefore := sload(_TOTAL_SUPPLY_SLOT) - let totalSupplyAfter := add(totalSupplyBefore, amount) - // Revert if the total supply overflows. - if lt(totalSupplyAfter, totalSupplyBefore) { - mstore(0x00, 0xe5cfe957) // `TotalSupplyOverflow()`. - revert(0x1c, 0x04) - } - // Store the updated total supply. - sstore(_TOTAL_SUPPLY_SLOT, totalSupplyAfter) - // Compute the balance slot and load its value. - mstore(0x0c, _BALANCE_SLOT_SEED) - mstore(0x00, to) - let toBalanceSlot := keccak256(0x0c, 0x20) - // Add and store the updated balance. - sstore(toBalanceSlot, add(sload(toBalanceSlot), amount)) - // Emit the {Transfer} event. - mstore(0x20, amount) - log3(0x20, 0x20, _TRANSFER_EVENT_SIGNATURE, 0, shr(96, mload(0x0c))) - } - _afterTokenTransfer(address(0), to, amount); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL BURN FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Burns `amount` tokens from `from`, reducing the total supply. - /// - /// Emits a {Transfer} event. - function _burn(address from, uint256 amount) internal virtual { - _beforeTokenTransfer(from, address(0), amount); - /// @solidity memory-safe-assembly - assembly { - // Compute the balance slot and load its value. - mstore(0x0c, _BALANCE_SLOT_SEED) - mstore(0x00, from) - let fromBalanceSlot := keccak256(0x0c, 0x20) - let fromBalance := sload(fromBalanceSlot) - // Revert if insufficient balance. - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - // Subtract and store the updated balance. - sstore(fromBalanceSlot, sub(fromBalance, amount)) - // Subtract and store the updated total supply. - sstore(_TOTAL_SUPPLY_SLOT, sub(sload(_TOTAL_SUPPLY_SLOT), amount)) - // Emit the {Transfer} event. - mstore(0x00, amount) - log3(0x00, 0x20, _TRANSFER_EVENT_SIGNATURE, shr(96, shl(96, from)), 0) - } - _afterTokenTransfer(from, address(0), amount); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL TRANSFER FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Moves `amount` of tokens from `from` to `to`. - function _transfer(address from, address to, uint256 amount) internal virtual { - _beforeTokenTransfer(from, to, amount); - /// @solidity memory-safe-assembly - assembly { - let from_ := shl(96, from) - // Compute the balance slot and load its value. - mstore(0x0c, or(from_, _BALANCE_SLOT_SEED)) - let fromBalanceSlot := keccak256(0x0c, 0x20) - let fromBalance := sload(fromBalanceSlot) - // Revert if insufficient balance. - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - // Subtract and store the updated balance. - sstore(fromBalanceSlot, sub(fromBalance, amount)) - // Compute the balance slot of `to`. - mstore(0x00, to) - let toBalanceSlot := keccak256(0x0c, 0x20) - // Add and store the updated balance of `to`. - // Will not overflow because the sum of all user balances - // cannot exceed the maximum uint256 value. - sstore(toBalanceSlot, add(sload(toBalanceSlot), amount)) - // Emit the {Transfer} event. - mstore(0x20, amount) - log3(0x20, 0x20, _TRANSFER_EVENT_SIGNATURE, shr(96, from_), shr(96, mload(0x0c))) - } - _afterTokenTransfer(from, to, amount); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL ALLOWANCE FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Updates the allowance of `owner` for `spender` based on spent `amount`. - function _spendAllowance(address owner, address spender, uint256 amount) internal virtual { - /// @solidity memory-safe-assembly - assembly { - // Compute the allowance slot and load its value. - mstore(0x20, spender) - mstore(0x0c, _ALLOWANCE_SLOT_SEED) - mstore(0x00, owner) - let allowanceSlot := keccak256(0x0c, 0x34) - let allowance_ := sload(allowanceSlot) - // If the allowance is not the maximum uint256 value. - if add(allowance_, 1) { - // Revert if the amount to be transferred exceeds the allowance. - if gt(amount, allowance_) { - mstore(0x00, 0x13be252b) // `InsufficientAllowance()`. - revert(0x1c, 0x04) - } - // Subtract and store the updated allowance. - sstore(allowanceSlot, sub(allowance_, amount)) - } - } - } - - /// @dev Sets `amount` as the allowance of `spender` over the tokens of `owner`. - /// - /// Emits a {Approval} event. - function _approve(address owner, address spender, uint256 amount) internal virtual { - /// @solidity memory-safe-assembly - assembly { - let owner_ := shl(96, owner) - // Compute the allowance slot and store the amount. - mstore(0x20, spender) - mstore(0x0c, or(owner_, _ALLOWANCE_SLOT_SEED)) - sstore(keccak256(0x0c, 0x34), amount) - // Emit the {Approval} event. - mstore(0x00, amount) - log3(0x00, 0x20, _APPROVAL_EVENT_SIGNATURE, shr(96, owner_), shr(96, mload(0x2c))) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* HOOKS TO OVERRIDE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Hook that is called before any transfer of tokens. - /// This includes minting and burning. - function _beforeTokenTransfer(address from, address to, uint256 amount) internal virtual {} - - /// @dev Hook that is called after any transfer of tokens. - /// This includes minting and burning. - function _afterTokenTransfer(address from, address to, uint256 amount) internal virtual {} -} diff --git a/lib/solady/src/tokens/ERC2981.sol b/lib/solady/src/tokens/ERC2981.sol deleted file mode 100644 index be1c7a7..0000000 --- a/lib/solady/src/tokens/ERC2981.sol +++ /dev/null @@ -1,162 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Simple ERC2981 NFT Royalty Standard implementation. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/tokens/ERC2981.sol) -/// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/token/common/ERC2981.sol) -abstract contract ERC2981 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The royalty fee numerator exceeds the fee denominator. - error RoyaltyOverflow(); - - /// @dev The royalty receiver cannot be the zero address. - error RoyaltyReceiverIsZeroAddress(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STORAGE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The default royalty info is given by: - /// ``` - /// let packed := sload(_ERC2981_MASTER_SLOT_SEED) - /// let receiver := shr(96, packed) - /// let royaltyFraction := xor(packed, shl(96, receiver)) - /// ``` - /// - /// The per token royalty info is given by. - /// ``` - /// mstore(0x00, tokenId) - /// mstore(0x20, _ERC2981_MASTER_SLOT_SEED) - /// let packed := sload(keccak256(0x00, 0x40)) - /// let receiver := shr(96, packed) - /// let royaltyFraction := xor(packed, shl(96, receiver)) - /// ``` - uint256 private constant _ERC2981_MASTER_SLOT_SEED = 0xaa4ec00224afccfdb7; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC2981 */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Checks that `_feeDenominator` is non-zero. - constructor() { - require(_feeDenominator() != 0, "Fee denominator cannot be zero."); - } - - /// @dev Returns the denominator for the royalty amount. - /// Defaults to 10000, which represents fees in basis points. - /// Override this function to return a custom amount if needed. - function _feeDenominator() internal pure virtual returns (uint96) { - return 10000; - } - - /// @dev Returns true if this contract implements the interface defined by `interfaceId`. - /// See: https://eips.ethereum.org/EIPS/eip-165 - /// This function call must use less than 30000 gas. - function supportsInterface(bytes4 interfaceId) public view virtual returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - let s := shr(224, interfaceId) - // ERC165: 0x01ffc9a7, ERC2981: 0x2a55205a. - result := or(eq(s, 0x01ffc9a7), eq(s, 0x2a55205a)) - } - } - - /// @dev Returns the `receiver` and `royaltyAmount` for `tokenId` sold at `salePrice`. - function royaltyInfo(uint256 tokenId, uint256 salePrice) - public - view - virtual - returns (address receiver, uint256 royaltyAmount) - { - uint256 feeDenominator = _feeDenominator(); - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, tokenId) - mstore(0x20, _ERC2981_MASTER_SLOT_SEED) - let packed := sload(keccak256(0x00, 0x40)) - receiver := shr(96, packed) - if iszero(receiver) { - packed := sload(mload(0x20)) - receiver := shr(96, packed) - } - let x := salePrice - let y := xor(packed, shl(96, receiver)) // `feeNumerator`. - // Overflow check, equivalent to `require(y == 0 || x <= type(uint256).max / y)`. - // Out-of-gas revert. Should not be triggered in practice, but included for safety. - returndatacopy(returndatasize(), returndatasize(), mul(y, gt(x, div(not(0), y)))) - royaltyAmount := div(mul(x, y), feeDenominator) - } - } - - /// @dev Sets the default royalty `receiver` and `feeNumerator`. - /// - /// Requirements: - /// - `receiver` must not be the zero address. - /// - `feeNumerator` must not be greater than the fee denominator. - function _setDefaultRoyalty(address receiver, uint96 feeNumerator) internal virtual { - uint256 feeDenominator = _feeDenominator(); - /// @solidity memory-safe-assembly - assembly { - feeNumerator := shr(160, shl(160, feeNumerator)) - if gt(feeNumerator, feeDenominator) { - mstore(0x00, 0x350a88b3) // `RoyaltyOverflow()`. - revert(0x1c, 0x04) - } - let packed := shl(96, receiver) - if iszero(packed) { - mstore(0x00, 0xb4457eaa) // `RoyaltyReceiverIsZeroAddress()`. - revert(0x1c, 0x04) - } - sstore(_ERC2981_MASTER_SLOT_SEED, or(packed, feeNumerator)) - } - } - - /// @dev Sets the default royalty `receiver` and `feeNumerator` to zero. - function _deleteDefaultRoyalty() internal virtual { - /// @solidity memory-safe-assembly - assembly { - sstore(_ERC2981_MASTER_SLOT_SEED, 0) - } - } - - /// @dev Sets the royalty `receiver` and `feeNumerator` for `tokenId`. - /// - /// Requirements: - /// - `receiver` must not be the zero address. - /// - `feeNumerator` must not be greater than the fee denominator. - function _setTokenRoyalty(uint256 tokenId, address receiver, uint96 feeNumerator) - internal - virtual - { - uint256 feeDenominator = _feeDenominator(); - /// @solidity memory-safe-assembly - assembly { - feeNumerator := shr(160, shl(160, feeNumerator)) - if gt(feeNumerator, feeDenominator) { - mstore(0x00, 0x350a88b3) // `RoyaltyOverflow()`. - revert(0x1c, 0x04) - } - let packed := shl(96, receiver) - if iszero(packed) { - mstore(0x00, 0xb4457eaa) // `RoyaltyReceiverIsZeroAddress()`. - revert(0x1c, 0x04) - } - mstore(0x00, tokenId) - mstore(0x20, _ERC2981_MASTER_SLOT_SEED) - sstore(keccak256(0x00, 0x40), or(packed, feeNumerator)) - } - } - - /// @dev Sets the royalty `receiver` and `feeNumerator` for `tokenId` to zero. - function _resetTokenRoyalty(uint256 tokenId) internal virtual { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, tokenId) - mstore(0x20, _ERC2981_MASTER_SLOT_SEED) - sstore(keccak256(0x00, 0x40), 0) - } - } -} diff --git a/lib/solady/src/tokens/ERC4626.sol b/lib/solady/src/tokens/ERC4626.sol deleted file mode 100644 index 0429e33..0000000 --- a/lib/solady/src/tokens/ERC4626.sol +++ /dev/null @@ -1,524 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {ERC20} from "./ERC20.sol"; -import {FixedPointMathLib} from "../utils/FixedPointMathLib.sol"; -import {SafeTransferLib} from "../utils/SafeTransferLib.sol"; - -/// @notice Simple ERC4626 tokenized Vault implementation. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/tokens/ERC4626.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/mixins/ERC4626.sol) -/// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/token/ERC20/extensions/ERC4626.sol) -abstract contract ERC4626 is ERC20 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The default underlying decimals. - uint8 internal constant _DEFAULT_UNDERLYING_DECIMALS = 18; - - /// @dev The default decimals offset. - uint8 internal constant _DEFAULT_DECIMALS_OFFSET = 0; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Cannot deposit more than the max limit. - error DepositMoreThanMax(); - - /// @dev Cannot mint more than the max limit. - error MintMoreThanMax(); - - /// @dev Cannot withdraw more than the max limit. - error WithdrawMoreThanMax(); - - /// @dev Cannot redeem more than the max limit. - error RedeemMoreThanMax(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EVENTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Emitted during a mint call or deposit call. - event Deposit(address indexed by, address indexed owner, uint256 assets, uint256 shares); - - /// @dev Emitted during a withdraw call or redeem call. - event Withdraw( - address indexed by, - address indexed to, - address indexed owner, - uint256 assets, - uint256 shares - ); - - /// @dev `keccak256(bytes("Deposit(address,address,uint256,uint256)"))`. - uint256 private constant _DEPOSIT_EVENT_SIGNATURE = - 0xdcbc1c05240f31ff3ad067ef1ee35ce4997762752e3a095284754544f4c709d7; - - /// @dev `keccak256(bytes("Withdraw(address,address,address,uint256,uint256)"))`. - uint256 private constant _WITHDRAW_EVENT_SIGNATURE = - 0xfbde797d201c681b91056529119e0b02407c7bb96a4a2c75c01fc9667232c8db; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC4626 CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev To be overridden to return the address of the underlying asset. - /// - /// - MUST be an ERC20 token contract. - /// - MUST NOT revert. - function asset() public view virtual returns (address); - - /// @dev To be overridden to return the number of decimals of the underlying asset. - /// Default: 18. - /// - /// - MUST NOT revert. - function _underlyingDecimals() internal view virtual returns (uint8) { - return _DEFAULT_UNDERLYING_DECIMALS; - } - - /// @dev Override to return a non-zero value to make the inflation attack even more unfeasible. - /// Only used when {_useVirtualShares} returns true. - /// Default: 0. - /// - /// - MUST NOT revert. - function _decimalsOffset() internal view virtual returns (uint8) { - return _DEFAULT_DECIMALS_OFFSET; - } - - /// @dev Returns whether virtual shares will be used to mitigate the inflation attack. - /// See: https://github.com/OpenZeppelin/openzeppelin-contracts/issues/3706 - /// Override to return true or false. - /// Default: true. - /// - /// - MUST NOT revert. - function _useVirtualShares() internal view virtual returns (bool) { - return true; - } - - /// @dev Returns the decimals places of the token. - /// - /// - MUST NOT revert. - function decimals() public view virtual override(ERC20) returns (uint8) { - if (!_useVirtualShares()) return _underlyingDecimals(); - return _underlyingDecimals() + _decimalsOffset(); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ASSET DECIMALS GETTER HELPER */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Helper function to get the decimals of the underlying asset. - /// Useful for setting the return value of `_underlyingDecimals` during initialization. - /// If the retrieval succeeds, `success` will be true, and `result` will hold the result. - /// Otherwise, `success` will be false, and `result` will be zero. - /// - /// Example usage: - /// ``` - /// (bool success, uint8 result) = _tryGetAssetDecimals(underlying); - /// _decimals = success ? result : _DEFAULT_UNDERLYING_DECIMALS; - /// ``` - function _tryGetAssetDecimals(address underlying) - internal - view - returns (bool success, uint8 result) - { - /// @solidity memory-safe-assembly - assembly { - // Store the function selector of `decimals()`. - mstore(0x00, 0x313ce567) - // Arguments are evaluated last to first. - success := - and( - // Returned value is less than 256, at left-padded to 32 bytes. - and(lt(mload(0x00), 0x100), gt(returndatasize(), 0x1f)), - // The staticcall succeeds. - staticcall(gas(), underlying, 0x1c, 0x04, 0x00, 0x20) - ) - result := mul(mload(0x00), success) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ACCOUNTING LOGIC */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the total amount of the underlying asset managed by the Vault. - /// - /// - SHOULD include any compounding that occurs from the yield. - /// - MUST be inclusive of any fees that are charged against assets in the Vault. - /// - MUST NOT revert. - function totalAssets() public view virtual returns (uint256 assets) { - assets = SafeTransferLib.balanceOf(asset(), address(this)); - } - - /// @dev Returns the amount of shares that the Vault will exchange for the amount of - /// assets provided, in an ideal scenario where all conditions are met. - /// - /// - MUST NOT be inclusive of any fees that are charged against assets in the Vault. - /// - MUST NOT show any variations depending on the caller. - /// - MUST NOT reflect slippage or other on-chain conditions, during the actual exchange. - /// - MUST NOT revert. - /// - /// Note: This calculation MAY NOT reflect the "per-user" price-per-share, and instead - /// should reflect the "average-user's" price-per-share, i.e. what the average user should - /// expect to see when exchanging to and from. - function convertToShares(uint256 assets) public view virtual returns (uint256 shares) { - if (!_useVirtualShares()) { - uint256 supply = totalSupply(); - return _eitherIsZero(assets, supply) - ? _initialConvertToShares(assets) - : FixedPointMathLib.fullMulDiv(assets, supply, totalAssets()); - } - uint256 o = _decimalsOffset(); - if (o == 0) { - return FixedPointMathLib.fullMulDiv(assets, totalSupply() + 1, _inc(totalAssets())); - } - return FixedPointMathLib.fullMulDiv(assets, totalSupply() + 10 ** o, _inc(totalAssets())); - } - - /// @dev Returns the amount of assets that the Vault will exchange for the amount of - /// shares provided, in an ideal scenario where all conditions are met. - /// - /// - MUST NOT be inclusive of any fees that are charged against assets in the Vault. - /// - MUST NOT show any variations depending on the caller. - /// - MUST NOT reflect slippage or other on-chain conditions, during the actual exchange. - /// - MUST NOT revert. - /// - /// Note: This calculation MAY NOT reflect the "per-user" price-per-share, and instead - /// should reflect the "average-user's" price-per-share, i.e. what the average user should - /// expect to see when exchanging to and from. - function convertToAssets(uint256 shares) public view virtual returns (uint256 assets) { - if (!_useVirtualShares()) { - uint256 supply = totalSupply(); - return supply == 0 - ? _initialConvertToAssets(shares) - : FixedPointMathLib.fullMulDiv(shares, totalAssets(), supply); - } - uint256 o = _decimalsOffset(); - if (o == 0) { - return FixedPointMathLib.fullMulDiv(shares, totalAssets() + 1, _inc(totalSupply())); - } - return FixedPointMathLib.fullMulDiv(shares, totalAssets() + 1, totalSupply() + 10 ** o); - } - - /// @dev Allows an on-chain or off-chain user to simulate the effects of their deposit - /// at the current block, given current on-chain conditions. - /// - /// - MUST return as close to and no more than the exact amount of Vault shares that - /// will be minted in a deposit call in the same transaction, i.e. deposit should - /// return the same or more shares as `previewDeposit` if call in the same transaction. - /// - MUST NOT account for deposit limits like those returned from `maxDeposit` and should - /// always act as if the deposit will be accepted, regardless of approvals, etc. - /// - MUST be inclusive of deposit fees. Integrators should be aware of this. - /// - MUST not revert. - /// - /// Note: Any unfavorable discrepancy between `convertToShares` and `previewDeposit` SHOULD - /// be considered slippage in share price or some other type of condition, meaning - /// the depositor will lose assets by depositing. - function previewDeposit(uint256 assets) public view virtual returns (uint256 shares) { - shares = convertToShares(assets); - } - - /// @dev Allows an on-chain or off-chain user to simulate the effects of their mint - /// at the current block, given current on-chain conditions. - /// - /// - MUST return as close to and no fewer than the exact amount of assets that - /// will be deposited in a mint call in the same transaction, i.e. mint should - /// return the same or fewer assets as `previewMint` if called in the same transaction. - /// - MUST NOT account for mint limits like those returned from `maxMint` and should - /// always act as if the mint will be accepted, regardless of approvals, etc. - /// - MUST be inclusive of deposit fees. Integrators should be aware of this. - /// - MUST not revert. - /// - /// Note: Any unfavorable discrepancy between `convertToAssets` and `previewMint` SHOULD - /// be considered slippage in share price or some other type of condition, - /// meaning the depositor will lose assets by minting. - function previewMint(uint256 shares) public view virtual returns (uint256 assets) { - if (!_useVirtualShares()) { - uint256 supply = totalSupply(); - return supply == 0 - ? _initialConvertToAssets(shares) - : FixedPointMathLib.fullMulDivUp(shares, totalAssets(), supply); - } - uint256 o = _decimalsOffset(); - if (o == 0) { - return FixedPointMathLib.fullMulDivUp(shares, totalAssets() + 1, _inc(totalSupply())); - } - return FixedPointMathLib.fullMulDivUp(shares, totalAssets() + 1, totalSupply() + 10 ** o); - } - - /// @dev Allows an on-chain or off-chain user to simulate the effects of their withdrawal - /// at the current block, given the current on-chain conditions. - /// - /// - MUST return as close to and no fewer than the exact amount of Vault shares that - /// will be burned in a withdraw call in the same transaction, i.e. withdraw should - /// return the same or fewer shares as `previewWithdraw` if call in the same transaction. - /// - MUST NOT account for withdrawal limits like those returned from `maxWithdraw` and should - /// always act as if the withdrawal will be accepted, regardless of share balance, etc. - /// - MUST be inclusive of withdrawal fees. Integrators should be aware of this. - /// - MUST not revert. - /// - /// Note: Any unfavorable discrepancy between `convertToShares` and `previewWithdraw` SHOULD - /// be considered slippage in share price or some other type of condition, - /// meaning the depositor will lose assets by depositing. - function previewWithdraw(uint256 assets) public view virtual returns (uint256 shares) { - if (!_useVirtualShares()) { - uint256 supply = totalSupply(); - return _eitherIsZero(assets, supply) - ? _initialConvertToShares(assets) - : FixedPointMathLib.fullMulDivUp(assets, supply, totalAssets()); - } - uint256 o = _decimalsOffset(); - if (o == 0) { - return FixedPointMathLib.fullMulDivUp(assets, totalSupply() + 1, _inc(totalAssets())); - } - return FixedPointMathLib.fullMulDivUp(assets, totalSupply() + 10 ** o, _inc(totalAssets())); - } - - /// @dev Allows an on-chain or off-chain user to simulate the effects of their redemption - /// at the current block, given current on-chain conditions. - /// - /// - MUST return as close to and no more than the exact amount of assets that - /// will be withdrawn in a redeem call in the same transaction, i.e. redeem should - /// return the same or more assets as `previewRedeem` if called in the same transaction. - /// - MUST NOT account for redemption limits like those returned from `maxRedeem` and should - /// always act as if the redemption will be accepted, regardless of approvals, etc. - /// - MUST be inclusive of withdrawal fees. Integrators should be aware of this. - /// - MUST NOT revert. - /// - /// Note: Any unfavorable discrepancy between `convertToAssets` and `previewRedeem` SHOULD - /// be considered slippage in share price or some other type of condition, - /// meaning the depositor will lose assets by depositing. - function previewRedeem(uint256 shares) public view virtual returns (uint256 assets) { - assets = convertToAssets(shares); - } - - /// @dev Private helper to return if either value is zero. - function _eitherIsZero(uint256 a, uint256 b) private pure returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - result := or(iszero(a), iszero(b)) - } - } - - /// @dev Private helper to return the value plus one. - function _inc(uint256 x) private pure returns (uint256) { - unchecked { - return x + 1; - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* DEPOSIT / WITHDRAWAL LIMIT LOGIC */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the maximum amount of the underlying asset that can be deposited - /// into the Vault for `to`, via a deposit call. - /// - /// - MUST return a limited value if `to` is subject to some deposit limit. - /// - MUST return `2**256-1` if there is no maximum limit. - /// - MUST NOT revert. - function maxDeposit(address to) public view virtual returns (uint256 maxAssets) { - to = to; // Silence unused variable warning. - maxAssets = type(uint256).max; - } - - /// @dev Returns the maximum amount of the Vault shares that can be minter for `to`, - /// via a mint call. - /// - /// - MUST return a limited value if `to` is subject to some mint limit. - /// - MUST return `2**256-1` if there is no maximum limit. - /// - MUST NOT revert. - function maxMint(address to) public view virtual returns (uint256 maxShares) { - to = to; // Silence unused variable warning. - maxShares = type(uint256).max; - } - - /// @dev Returns the maximum amount of the underlying asset that can be withdrawn - /// from the `owner`'s balance in the Vault, via a withdraw call. - /// - /// - MUST return a limited value if `owner` is subject to some withdrawal limit or timelock. - /// - MUST NOT revert. - function maxWithdraw(address owner) public view virtual returns (uint256 maxAssets) { - maxAssets = convertToAssets(balanceOf(owner)); - } - - /// @dev Returns the maximum amount of Vault shares that can be redeemed - /// from the `owner`'s balance in the Vault, via a redeem call. - /// - /// - MUST return a limited value if `owner` is subject to some withdrawal limit or timelock. - /// - MUST return `balanceOf(owner)` otherwise. - /// - MUST NOT revert. - function maxRedeem(address owner) public view virtual returns (uint256 maxShares) { - maxShares = balanceOf(owner); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* DEPOSIT / WITHDRAWAL LOGIC */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Mints `shares` Vault shares to `to` by depositing exactly `assets` - /// of underlying tokens. - /// - /// - MUST emit the {Deposit} event. - /// - MAY support an additional flow in which the underlying tokens are owned by the Vault - /// contract before the deposit execution, and are accounted for during deposit. - /// - MUST revert if all of `assets` cannot be deposited, such as due to deposit limit, - /// slippage, insufficient approval, etc. - /// - /// Note: Most implementations will require pre-approval of the Vault with the - /// Vault's underlying `asset` token. - function deposit(uint256 assets, address to) public virtual returns (uint256 shares) { - if (assets > maxDeposit(to)) _revert(0xb3c61a83); // `DepositMoreThanMax()`. - shares = previewDeposit(assets); - _deposit(msg.sender, to, assets, shares); - } - - /// @dev Mints exactly `shares` Vault shares to `to` by depositing `assets` - /// of underlying tokens. - /// - /// - MUST emit the {Deposit} event. - /// - MAY support an additional flow in which the underlying tokens are owned by the Vault - /// contract before the mint execution, and are accounted for during mint. - /// - MUST revert if all of `shares` cannot be deposited, such as due to deposit limit, - /// slippage, insufficient approval, etc. - /// - /// Note: Most implementations will require pre-approval of the Vault with the - /// Vault's underlying `asset` token. - function mint(uint256 shares, address to) public virtual returns (uint256 assets) { - if (shares > maxMint(to)) _revert(0x6a695959); // `MintMoreThanMax()`. - assets = previewMint(shares); - _deposit(msg.sender, to, assets, shares); - } - - /// @dev Burns `shares` from `owner` and sends exactly `assets` of underlying tokens to `to`. - /// - /// - MUST emit the {Withdraw} event. - /// - MAY support an additional flow in which the underlying tokens are owned by the Vault - /// contract before the withdraw execution, and are accounted for during withdraw. - /// - MUST revert if all of `assets` cannot be withdrawn, such as due to withdrawal limit, - /// slippage, insufficient balance, etc. - /// - /// Note: Some implementations will require pre-requesting to the Vault before a withdrawal - /// may be performed. Those methods should be performed separately. - function withdraw(uint256 assets, address to, address owner) - public - virtual - returns (uint256 shares) - { - if (assets > maxWithdraw(owner)) _revert(0x936941fc); // `WithdrawMoreThanMax()`. - shares = previewWithdraw(assets); - _withdraw(msg.sender, to, owner, assets, shares); - } - - /// @dev Burns exactly `shares` from `owner` and sends `assets` of underlying tokens to `to`. - /// - /// - MUST emit the {Withdraw} event. - /// - MAY support an additional flow in which the underlying tokens are owned by the Vault - /// contract before the redeem execution, and are accounted for during redeem. - /// - MUST revert if all of shares cannot be redeemed, such as due to withdrawal limit, - /// slippage, insufficient balance, etc. - /// - /// Note: Some implementations will require pre-requesting to the Vault before a redeem - /// may be performed. Those methods should be performed separately. - function redeem(uint256 shares, address to, address owner) - public - virtual - returns (uint256 assets) - { - if (shares > maxRedeem(owner)) _revert(0x4656425a); // `RedeemMoreThanMax()`. - assets = previewRedeem(shares); - _withdraw(msg.sender, to, owner, assets, shares); - } - - /// @dev Internal helper for reverting efficiently. - function _revert(uint256 s) private pure { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, s) - revert(0x1c, 0x04) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev For deposits and mints. - /// - /// Emits a {Deposit} event. - function _deposit(address by, address to, uint256 assets, uint256 shares) internal virtual { - SafeTransferLib.safeTransferFrom(asset(), by, address(this), assets); - _mint(to, shares); - /// @solidity memory-safe-assembly - assembly { - // Emit the {Deposit} event. - mstore(0x00, assets) - mstore(0x20, shares) - let m := shr(96, not(0)) - log3(0x00, 0x40, _DEPOSIT_EVENT_SIGNATURE, and(m, by), and(m, to)) - } - _afterDeposit(assets, shares); - } - - /// @dev For withdrawals and redemptions. - /// - /// Emits a {Withdraw} event. - function _withdraw(address by, address to, address owner, uint256 assets, uint256 shares) - internal - virtual - { - if (by != owner) _spendAllowance(owner, by, shares); - _beforeWithdraw(assets, shares); - _burn(owner, shares); - SafeTransferLib.safeTransfer(asset(), to, assets); - /// @solidity memory-safe-assembly - assembly { - // Emit the {Withdraw} event. - mstore(0x00, assets) - mstore(0x20, shares) - let m := shr(96, not(0)) - log4(0x00, 0x40, _WITHDRAW_EVENT_SIGNATURE, and(m, by), and(m, to), and(m, owner)) - } - } - - /// @dev Internal conversion function (from assets to shares) to apply when the Vault is empty. - /// Only used when {_useVirtualShares} returns false. - /// - /// Note: Make sure to keep this function consistent with {_initialConvertToAssets} - /// when overriding it. - function _initialConvertToShares(uint256 assets) - internal - view - virtual - returns (uint256 shares) - { - shares = assets; - } - - /// @dev Internal conversion function (from shares to assets) to apply when the Vault is empty. - /// Only used when {_useVirtualShares} returns false. - /// - /// Note: Make sure to keep this function consistent with {_initialConvertToShares} - /// when overriding it. - function _initialConvertToAssets(uint256 shares) - internal - view - virtual - returns (uint256 assets) - { - assets = shares; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* HOOKS TO OVERRIDE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Hook that is called before any withdrawal or redemption. - function _beforeWithdraw(uint256 assets, uint256 shares) internal virtual {} - - /// @dev Hook that is called after any deposit or mint. - function _afterDeposit(uint256 assets, uint256 shares) internal virtual {} -} diff --git a/lib/solady/src/tokens/ERC6909.sol b/lib/solady/src/tokens/ERC6909.sol deleted file mode 100644 index 871bba1..0000000 --- a/lib/solady/src/tokens/ERC6909.sol +++ /dev/null @@ -1,541 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Simple EIP-6909 implementation. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/tokens/ERC6909.sol) -/// -/// @dev Note: -/// The ERC6909 standard allows minting and transferring to and from the zero address, -/// minting and transferring zero tokens, as well as self-approvals. -/// For performance, this implementation WILL NOT revert for such actions. -/// Please add any checks with overrides if desired. -/// -/// If you are overriding: -/// - Make sure all variables written to storage are properly cleaned -// (e.g. the bool value for `isOperator` MUST be either 1 or 0 under the hood). -/// - Check that the overridden function is actually used in the function you want to -/// change the behavior of. Much of the code has been manually inlined for performance. -abstract contract ERC6909 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Insufficient balance. - error InsufficientBalance(); - - /// @dev Insufficient permission to perform the action. - error InsufficientPermission(); - - /// @dev The balance has overflowed. - error BalanceOverflow(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EVENTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Emitted when `by` transfers `amount` of token `id` from `from` to `to`. - event Transfer( - address by, address indexed from, address indexed to, uint256 indexed id, uint256 amount - ); - - /// @dev Emitted when `owner` enables or disables `operator` to manage all of their tokens. - event OperatorSet(address indexed owner, address indexed operator, bool approved); - - /// @dev Emitted when `owner` approves `spender` to use `amount` of `id` token. - event Approval( - address indexed owner, address indexed spender, uint256 indexed id, uint256 amount - ); - - /// @dev `keccak256(bytes("Transfer(address,address,address,uint256,uint256)"))`. - uint256 private constant _TRANSFER_EVENT_SIGNATURE = - 0x1b3d7edb2e9c0b0e7c525b20aaaef0f5940d2ed71663c7d39266ecafac728859; - - /// @dev `keccak256(bytes("OperatorSet(address,address,bool)"))`. - uint256 private constant _OPERATOR_SET_EVENT_SIGNATURE = - 0xceb576d9f15e4e200fdb5096d64d5dfd667e16def20c1eefd14256d8e3faa267; - - /// @dev `keccak256(bytes("Approval(address,address,uint256,uint256)"))`. - uint256 private constant _APPROVAL_EVENT_SIGNATURE = - 0xb3fd5071835887567a0671151121894ddccc2842f1d10bedad13e0d17cace9a7; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STORAGE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The `ownerSlotSeed` of a given owner is given by. - /// ``` - /// let ownerSlotSeed := or(_ERC6909_MASTER_SLOT_SEED, shl(96, owner)) - /// ``` - /// - /// The balance slot of `owner` is given by. - /// ``` - /// mstore(0x20, ownerSlotSeed) - /// mstore(0x00, id) - /// let balanceSlot := keccak256(0x00, 0x40) - /// ``` - /// - /// The operator approval slot of `owner` is given by. - /// ``` - /// mstore(0x20, ownerSlotSeed) - /// mstore(0x00, operator) - /// let operatorApprovalSlot := keccak256(0x0c, 0x34) - /// ``` - /// - /// The allowance slot of (`owner`, `spender`, `id`) is given by: - /// ``` - /// mstore(0x34, ownerSlotSeed) - /// mstore(0x14, spender) - /// mstore(0x00, id) - /// let allowanceSlot := keccak256(0x00, 0x54) - /// ``` - uint256 private constant _ERC6909_MASTER_SLOT_SEED = 0xedcaa89a82293940; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC6909 METADATA */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the token collection name. - function name() public view virtual returns (string memory); - - /// @dev Returns the token collection symbol. - function symbol() public view virtual returns (string memory); - - /// @dev Returns the number of decimals for token `id`. - /// Returns 18 by default. - /// Please override this function if you need to return a custom value. - function decimals(uint256 id) public view virtual returns (uint8) { - id = id; // Silence compiler warning. - return 18; - } - - /// @dev Returns the Uniform Resource Identifier (URI) for token `id`. - function tokenURI(uint256 id) public view virtual returns (string memory); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC6909 */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the amount of token `id` owned by `owner`. - function balanceOf(address owner, uint256 id) public view virtual returns (uint256 amount) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, _ERC6909_MASTER_SLOT_SEED) - mstore(0x14, owner) - mstore(0x00, id) - amount := sload(keccak256(0x00, 0x40)) - } - } - - /// @dev Returns the amount of token `id` that `spender` can spend on behalf of `owner`. - function allowance(address owner, address spender, uint256 id) - public - view - virtual - returns (uint256 amount) - { - /// @solidity memory-safe-assembly - assembly { - mstore(0x34, _ERC6909_MASTER_SLOT_SEED) - mstore(0x28, owner) - mstore(0x14, spender) - mstore(0x00, id) - amount := sload(keccak256(0x00, 0x54)) - // Restore the part of the free memory pointer that has been overwritten. - mstore(0x34, 0x00) - } - } - - /// @dev Checks if a `spender` is approved by `owner` to manage all of their tokens. - function isOperator(address owner, address spender) public view virtual returns (bool status) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, _ERC6909_MASTER_SLOT_SEED) - mstore(0x14, owner) - mstore(0x00, spender) - status := sload(keccak256(0x0c, 0x34)) - } - } - - /// @dev Transfers `amount` of token `id` from the caller to `to`. - /// - /// Requirements: - /// - caller must at least have `amount`. - /// - /// Emits a {Transfer} event. - function transfer(address to, uint256 id, uint256 amount) - public - payable - virtual - returns (bool) - { - _beforeTokenTransfer(msg.sender, to, id, amount); - /// @solidity memory-safe-assembly - assembly { - /// Compute the balance slot and load its value. - mstore(0x20, _ERC6909_MASTER_SLOT_SEED) - mstore(0x14, caller()) - mstore(0x00, id) - let fromBalanceSlot := keccak256(0x00, 0x40) - let fromBalance := sload(fromBalanceSlot) - // Revert if insufficient balance. - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - // Subtract and store the updated balance. - sstore(fromBalanceSlot, sub(fromBalance, amount)) - // Compute the balance slot of `to`. - mstore(0x14, to) - mstore(0x00, id) - let toBalanceSlot := keccak256(0x00, 0x40) - let toBalanceBefore := sload(toBalanceSlot) - let toBalanceAfter := add(toBalanceBefore, amount) - // Revert if the balance overflows. - if lt(toBalanceAfter, toBalanceBefore) { - mstore(0x00, 0x89560ca1) // `BalanceOverflow()`. - revert(0x1c, 0x04) - } - // Store the updated balance of `to`. - sstore(toBalanceSlot, toBalanceAfter) - // Emit the {Transfer} event. - mstore(0x00, caller()) - mstore(0x20, amount) - log4(0x00, 0x40, _TRANSFER_EVENT_SIGNATURE, caller(), shr(96, shl(96, to)), id) - } - _afterTokenTransfer(msg.sender, to, id, amount); - return true; - } - - /// @dev Transfers `amount` of token `id` from `from` to `to`. - /// - /// Note: Does not update the allowance if it is the maximum uint256 value. - /// - /// Requirements: - /// - `from` must at least have `amount` of token `id`. - /// - The caller must have at least `amount` of allowance to transfer the - /// tokens of `from` or approved as an operator. - /// - /// Emits a {Transfer} event. - function transferFrom(address from, address to, uint256 id, uint256 amount) - public - payable - virtual - returns (bool) - { - _beforeTokenTransfer(from, to, id, amount); - /// @solidity memory-safe-assembly - assembly { - // Compute the operator slot and load its value. - mstore(0x34, _ERC6909_MASTER_SLOT_SEED) - mstore(0x28, from) - mstore(0x14, caller()) - // Check if the caller is an operator. - if iszero(sload(keccak256(0x20, 0x34))) { - // Compute the allowance slot and load its value. - mstore(0x00, id) - let allowanceSlot := keccak256(0x00, 0x54) - let allowance_ := sload(allowanceSlot) - // If the allowance is not the maximum uint256 value. - if add(allowance_, 1) { - // Revert if the amount to be transferred exceeds the allowance. - if gt(amount, allowance_) { - mstore(0x00, 0xdeda9030) // `InsufficientPermission()`. - revert(0x1c, 0x04) - } - // Subtract and store the updated allowance. - sstore(allowanceSlot, sub(allowance_, amount)) - } - } - // Compute the balance slot and load its value. - mstore(0x14, id) - let fromBalanceSlot := keccak256(0x14, 0x40) - let fromBalance := sload(fromBalanceSlot) - // Revert if insufficient balance. - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - // Subtract and store the updated balance. - sstore(fromBalanceSlot, sub(fromBalance, amount)) - // Compute the balance slot of `to`. - mstore(0x28, to) - mstore(0x14, id) - let toBalanceSlot := keccak256(0x14, 0x40) - let toBalanceBefore := sload(toBalanceSlot) - let toBalanceAfter := add(toBalanceBefore, amount) - // Revert if the balance overflows. - if lt(toBalanceAfter, toBalanceBefore) { - mstore(0x00, 0x89560ca1) // `BalanceOverflow()`. - revert(0x1c, 0x04) - } - // Store the updated balance of `to`. - sstore(toBalanceSlot, toBalanceAfter) - // Emit the {Transfer} event. - mstore(0x00, caller()) - mstore(0x20, amount) - // forgefmt: disable-next-line - log4(0x00, 0x40, _TRANSFER_EVENT_SIGNATURE, shr(96, shl(96, from)), shr(96, shl(96, to)), id) - // Restore the part of the free memory pointer that has been overwritten. - mstore(0x34, 0x00) - } - _afterTokenTransfer(from, to, id, amount); - return true; - } - - /// @dev Sets `amount` as the allowance of `spender` for the caller for token `id`. - /// - /// Emits a {Approval} event. - function approve(address spender, uint256 id, uint256 amount) - public - payable - virtual - returns (bool) - { - /// @solidity memory-safe-assembly - assembly { - // Compute the allowance slot and store the amount. - mstore(0x34, _ERC6909_MASTER_SLOT_SEED) - mstore(0x28, caller()) - mstore(0x14, spender) - mstore(0x00, id) - sstore(keccak256(0x00, 0x54), amount) - // Emit the {Approval} event. - mstore(0x00, amount) - log4(0x00, 0x20, _APPROVAL_EVENT_SIGNATURE, caller(), shr(96, mload(0x20)), id) - // Restore the part of the free memory pointer that has been overwritten. - mstore(0x34, 0x00) - } - return true; - } - - /// @dev Sets whether `operator` is approved to manage the tokens of the caller. - /// - /// Emits {OperatorSet} event. - function setOperator(address operator, bool approved) public payable virtual returns (bool) { - /// @solidity memory-safe-assembly - assembly { - // Convert `approved` to `0` or `1`. - let approvedCleaned := iszero(iszero(approved)) - // Compute the operator slot and store the approved. - mstore(0x20, _ERC6909_MASTER_SLOT_SEED) - mstore(0x14, caller()) - mstore(0x00, operator) - sstore(keccak256(0x0c, 0x34), approvedCleaned) - // Emit the {OperatorSet} event. - mstore(0x20, approvedCleaned) - log3(0x20, 0x20, _OPERATOR_SET_EVENT_SIGNATURE, caller(), shr(96, mload(0x0c))) - } - return true; - } - - /// @dev Returns true if this contract implements the interface defined by `interfaceId`. - function supportsInterface(bytes4 interfaceId) public view virtual returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - let s := shr(224, interfaceId) - // ERC165: 0x01ffc9a7, ERC6909: 0x0f632fb3. - result := or(eq(s, 0x01ffc9a7), eq(s, 0x0f632fb3)) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Mints `amount` of token `id` to `to`. - /// - /// Emits a {Transfer} event. - function _mint(address to, uint256 id, uint256 amount) internal virtual { - _beforeTokenTransfer(address(0), to, id, amount); - /// @solidity memory-safe-assembly - assembly { - // Compute the balance slot. - mstore(0x20, _ERC6909_MASTER_SLOT_SEED) - mstore(0x14, to) - mstore(0x00, id) - let toBalanceSlot := keccak256(0x00, 0x40) - // Add and store the updated balance - let toBalanceBefore := sload(toBalanceSlot) - let toBalanceAfter := add(toBalanceBefore, amount) - // Revert if the balance overflows. - if lt(toBalanceAfter, toBalanceBefore) { - mstore(0x00, 0x89560ca1) // `BalanceOverflow()`. - revert(0x1c, 0x04) - } - sstore(toBalanceSlot, toBalanceAfter) - // Emit the {Transfer} event. - mstore(0x00, caller()) - mstore(0x20, amount) - log4(0x00, 0x40, _TRANSFER_EVENT_SIGNATURE, 0, shr(96, shl(96, to)), id) - } - _afterTokenTransfer(address(0), to, id, amount); - } - - /// @dev Burns `amount` token `id` from `from`. - /// - /// Emits a {Transfer} event. - function _burn(address from, uint256 id, uint256 amount) internal virtual { - _beforeTokenTransfer(from, address(0), id, amount); - /// @solidity memory-safe-assembly - assembly { - // Compute the balance slot. - mstore(0x20, _ERC6909_MASTER_SLOT_SEED) - mstore(0x14, from) - mstore(0x00, id) - let fromBalanceSlot := keccak256(0x00, 0x40) - let fromBalance := sload(fromBalanceSlot) - // Revert if insufficient balance. - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - // Subtract and store the updated balance. - sstore(fromBalanceSlot, sub(fromBalance, amount)) - // Emit the {Transfer} event. - mstore(0x00, caller()) - mstore(0x20, amount) - log4(0x00, 0x40, _TRANSFER_EVENT_SIGNATURE, shr(96, shl(96, from)), 0, id) - } - _afterTokenTransfer(from, address(0), id, amount); - } - - /// @dev Transfers `amount` of token `id` from `from` to `to`. - /// - /// Note: Does not update the allowance if it is the maximum uint256 value. - /// - /// Requirements: - /// - `from` must at least have `amount` of token `id`. - /// - If `by` is not the zero address, - /// it must have at least `amount` of allowance to transfer the - /// tokens of `from` or approved as an operator. - /// - /// Emits a {Transfer} event. - function _transfer(address by, address from, address to, uint256 id, uint256 amount) - internal - virtual - { - _beforeTokenTransfer(from, to, id, amount); - /// @solidity memory-safe-assembly - assembly { - let bitmaskAddress := 0xffffffffffffffffffffffffffffffffffffffff - // Compute the operator slot and load its value. - mstore(0x34, _ERC6909_MASTER_SLOT_SEED) - mstore(0x28, from) - // If `by` is not the zero address. - if and(bitmaskAddress, by) { - mstore(0x14, by) - // Check if the `by` is an operator. - if iszero(sload(keccak256(0x20, 0x34))) { - // Compute the allowance slot and load its value. - mstore(0x00, id) - let allowanceSlot := keccak256(0x00, 0x54) - let allowance_ := sload(allowanceSlot) - // If the allowance is not the maximum uint256 value. - if add(allowance_, 1) { - // Revert if the amount to be transferred exceeds the allowance. - if gt(amount, allowance_) { - mstore(0x00, 0xdeda9030) // `InsufficientPermission()`. - revert(0x1c, 0x04) - } - // Subtract and store the updated allowance. - sstore(allowanceSlot, sub(allowance_, amount)) - } - } - } - // Compute the balance slot and load its value. - mstore(0x14, id) - let fromBalanceSlot := keccak256(0x14, 0x40) - let fromBalance := sload(fromBalanceSlot) - // Revert if insufficient balance. - if gt(amount, fromBalance) { - mstore(0x00, 0xf4d678b8) // `InsufficientBalance()`. - revert(0x1c, 0x04) - } - // Subtract and store the updated balance. - sstore(fromBalanceSlot, sub(fromBalance, amount)) - // Compute the balance slot of `to`. - mstore(0x28, to) - mstore(0x14, id) - let toBalanceSlot := keccak256(0x14, 0x40) - let toBalanceBefore := sload(toBalanceSlot) - let toBalanceAfter := add(toBalanceBefore, amount) - // Revert if the balance overflows. - if lt(toBalanceAfter, toBalanceBefore) { - mstore(0x00, 0x89560ca1) // `BalanceOverflow()`. - revert(0x1c, 0x04) - } - // Store the updated balance of `to`. - sstore(toBalanceSlot, toBalanceAfter) - // Emit the {Transfer} event. - mstore(0x00, and(bitmaskAddress, by)) - mstore(0x20, amount) - // forgefmt: disable-next-line - log4(0x00, 0x40, _TRANSFER_EVENT_SIGNATURE, and(bitmaskAddress, from), and(bitmaskAddress, to), id) - // Restore the part of the free memory pointer that has been overwritten. - mstore(0x34, 0x00) - } - _afterTokenTransfer(from, to, id, amount); - } - - /// @dev Sets `amount` as the allowance of `spender` for `owner` for token `id`. - /// - /// Emits a {Approval} event. - function _approve(address owner, address spender, uint256 id, uint256 amount) - internal - virtual - { - /// @solidity memory-safe-assembly - assembly { - // Compute the allowance slot and store the amount. - mstore(0x34, _ERC6909_MASTER_SLOT_SEED) - mstore(0x28, owner) - mstore(0x14, spender) - mstore(0x00, id) - sstore(keccak256(0x00, 0x54), amount) - // Emit the {Approval} event. - mstore(0x00, amount) - // forgefmt: disable-next-line - log4(0x00, 0x20, _APPROVAL_EVENT_SIGNATURE, shr(96, mload(0x34)), shr(96, mload(0x20)), id) - // Restore the part of the free memory pointer that has been overwritten. - mstore(0x34, 0x00) - } - } - - /// @dev Sets whether `operator` is approved to manage the tokens of `owner`. - /// - /// Emits {OperatorSet} event. - function _setOperator(address owner, address operator, bool approved) internal virtual { - /// @solidity memory-safe-assembly - assembly { - // Convert `approved` to `0` or `1`. - let approvedCleaned := iszero(iszero(approved)) - // Compute the operator slot and store the approved. - mstore(0x20, _ERC6909_MASTER_SLOT_SEED) - mstore(0x14, owner) - mstore(0x00, operator) - sstore(keccak256(0x0c, 0x34), approvedCleaned) - // Emit the {OperatorSet} event. - mstore(0x20, approvedCleaned) - // forgefmt: disable-next-line - log3(0x20, 0x20, _OPERATOR_SET_EVENT_SIGNATURE, shr(96, shl(96, owner)), shr(96, mload(0x0c))) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* HOOKS TO OVERRIDE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Hook that is called before any transfer of tokens. - /// This includes minting and burning. - function _beforeTokenTransfer(address from, address to, uint256 id, uint256 amount) - internal - virtual - {} - - /// @dev Hook that is called after any transfer of tokens. - /// This includes minting and burning. - function _afterTokenTransfer(address from, address to, uint256 id, uint256 amount) - internal - virtual - {} -} diff --git a/lib/solady/src/tokens/ERC721.sol b/lib/solady/src/tokens/ERC721.sol deleted file mode 100644 index d21b424..0000000 --- a/lib/solady/src/tokens/ERC721.sol +++ /dev/null @@ -1,890 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Simple ERC721 implementation with storage hitchhiking. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/tokens/ERC721.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC721.sol) -/// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/tree/master/contracts/token/ERC721/ERC721.sol) -/// -/// @dev Note: -/// - The ERC721 standard allows for self-approvals. -/// For performance, this implementation WILL NOT revert for such actions. -/// Please add any checks with overrides if desired. -/// - For performance, methods are made payable where permitted by the ERC721 standard. -/// - The `safeTransfer` functions use the identity precompile (0x4) -/// to copy memory internally. -/// -/// If you are overriding: -/// - NEVER violate the ERC721 invariant: -/// the balance of an owner MUST always be equal to their number of ownership slots. -/// The transfer functions do not have an underflow guard for user token balances. -/// - Make sure all variables written to storage are properly cleaned -// (e.g. the bool value for `isApprovedForAll` MUST be either 1 or 0 under the hood). -/// - Check that the overridden function is actually used in the function you want to -/// change the behavior of. Much of the code has been manually inlined for performance. -abstract contract ERC721 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev An account can hold up to 4294967295 tokens. - uint256 internal constant _MAX_ACCOUNT_BALANCE = 0xffffffff; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Only the token owner or an approved account can manage the token. - error NotOwnerNorApproved(); - - /// @dev The token does not exist. - error TokenDoesNotExist(); - - /// @dev The token already exists. - error TokenAlreadyExists(); - - /// @dev Cannot query the balance for the zero address. - error BalanceQueryForZeroAddress(); - - /// @dev Cannot mint or transfer to the zero address. - error TransferToZeroAddress(); - - /// @dev The token must be owned by `from`. - error TransferFromIncorrectOwner(); - - /// @dev The recipient's balance has overflowed. - error AccountBalanceOverflow(); - - /// @dev Cannot safely transfer to a contract that does not implement - /// the ERC721Receiver interface. - error TransferToNonERC721ReceiverImplementer(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EVENTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Emitted when token `id` is transferred from `from` to `to`. - event Transfer(address indexed from, address indexed to, uint256 indexed id); - - /// @dev Emitted when `owner` enables `account` to manage the `id` token. - event Approval(address indexed owner, address indexed account, uint256 indexed id); - - /// @dev Emitted when `owner` enables or disables `operator` to manage all of their tokens. - event ApprovalForAll(address indexed owner, address indexed operator, bool isApproved); - - /// @dev `keccak256(bytes("Transfer(address,address,uint256)"))`. - uint256 private constant _TRANSFER_EVENT_SIGNATURE = - 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef; - - /// @dev `keccak256(bytes("Approval(address,address,uint256)"))`. - uint256 private constant _APPROVAL_EVENT_SIGNATURE = - 0x8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925; - - /// @dev `keccak256(bytes("ApprovalForAll(address,address,bool)"))`. - uint256 private constant _APPROVAL_FOR_ALL_EVENT_SIGNATURE = - 0x17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STORAGE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The ownership data slot of `id` is given by: - /// ``` - /// mstore(0x00, id) - /// mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - /// let ownershipSlot := add(id, add(id, keccak256(0x00, 0x20))) - /// ``` - /// Bits Layout: - /// - [0..159] `addr` - /// - [160..255] `extraData` - /// - /// The approved address slot is given by: `add(1, ownershipSlot)`. - /// - /// See: https://notes.ethereum.org/%40vbuterin/verkle_tree_eip - /// - /// The balance slot of `owner` is given by: - /// ``` - /// mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - /// mstore(0x00, owner) - /// let balanceSlot := keccak256(0x0c, 0x1c) - /// ``` - /// Bits Layout: - /// - [0..31] `balance` - /// - [32..255] `aux` - /// - /// The `operator` approval slot of `owner` is given by: - /// ``` - /// mstore(0x1c, or(_ERC721_MASTER_SLOT_SEED, operator)) - /// mstore(0x00, owner) - /// let operatorApprovalSlot := keccak256(0x0c, 0x30) - /// ``` - uint256 private constant _ERC721_MASTER_SLOT_SEED = 0x7d8825530a5a2e7a << 192; - - /// @dev Pre-shifted and pre-masked constant. - uint256 private constant _ERC721_MASTER_SLOT_SEED_MASKED = 0x0a5a2e7a00000000; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC721 METADATA */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the token collection name. - function name() public view virtual returns (string memory); - - /// @dev Returns the token collection symbol. - function symbol() public view virtual returns (string memory); - - /// @dev Returns the Uniform Resource Identifier (URI) for token `id`. - function tokenURI(uint256 id) public view virtual returns (string memory); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC721 */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the owner of token `id`. - /// - /// Requirements: - /// - Token `id` must exist. - function ownerOf(uint256 id) public view virtual returns (address result) { - result = _ownerOf(id); - /// @solidity memory-safe-assembly - assembly { - if iszero(result) { - mstore(0x00, 0xceea21b6) // `TokenDoesNotExist()`. - revert(0x1c, 0x04) - } - } - } - - /// @dev Returns the number of tokens owned by `owner`. - /// - /// Requirements: - /// - `owner` must not be the zero address. - function balanceOf(address owner) public view virtual returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - // Revert if the `owner` is the zero address. - if iszero(owner) { - mstore(0x00, 0x8f4eb604) // `BalanceQueryForZeroAddress()`. - revert(0x1c, 0x04) - } - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - mstore(0x00, owner) - result := and(sload(keccak256(0x0c, 0x1c)), _MAX_ACCOUNT_BALANCE) - } - } - - /// @dev Returns the account approved to manage token `id`. - /// - /// Requirements: - /// - Token `id` must exist. - function getApproved(uint256 id) public view virtual returns (address result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, id) - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - let ownershipSlot := add(id, add(id, keccak256(0x00, 0x20))) - if iszero(shl(96, sload(ownershipSlot))) { - mstore(0x00, 0xceea21b6) // `TokenDoesNotExist()`. - revert(0x1c, 0x04) - } - result := sload(add(1, ownershipSlot)) - } - } - - /// @dev Sets `account` as the approved account to manage token `id`. - /// - /// Requirements: - /// - Token `id` must exist. - /// - The caller must be the owner of the token, - /// or an approved operator for the token owner. - /// - /// Emits an {Approval} event. - function approve(address account, uint256 id) public payable virtual { - _approve(msg.sender, account, id); - } - - /// @dev Returns whether `operator` is approved to manage the tokens of `owner`. - function isApprovedForAll(address owner, address operator) - public - view - virtual - returns (bool result) - { - /// @solidity memory-safe-assembly - assembly { - mstore(0x1c, operator) - mstore(0x08, _ERC721_MASTER_SLOT_SEED_MASKED) - mstore(0x00, owner) - result := sload(keccak256(0x0c, 0x30)) - } - } - - /// @dev Sets whether `operator` is approved to manage the tokens of the caller. - /// - /// Emits an {ApprovalForAll} event. - function setApprovalForAll(address operator, bool isApproved) public virtual { - /// @solidity memory-safe-assembly - assembly { - // Convert to 0 or 1. - isApproved := iszero(iszero(isApproved)) - // Update the `isApproved` for (`msg.sender`, `operator`). - mstore(0x1c, operator) - mstore(0x08, _ERC721_MASTER_SLOT_SEED_MASKED) - mstore(0x00, caller()) - sstore(keccak256(0x0c, 0x30), isApproved) - // Emit the {ApprovalForAll} event. - mstore(0x00, isApproved) - // forgefmt: disable-next-item - log3(0x00, 0x20, _APPROVAL_FOR_ALL_EVENT_SIGNATURE, caller(), shr(96, shl(96, operator))) - } - } - - /// @dev Transfers token `id` from `from` to `to`. - /// - /// Requirements: - /// - /// - Token `id` must exist. - /// - `from` must be the owner of the token. - /// - `to` cannot be the zero address. - /// - The caller must be the owner of the token, or be approved to manage the token. - /// - /// Emits a {Transfer} event. - function transferFrom(address from, address to, uint256 id) public payable virtual { - _beforeTokenTransfer(from, to, id); - /// @solidity memory-safe-assembly - assembly { - // Clear the upper 96 bits. - let bitmaskAddress := shr(96, not(0)) - from := and(bitmaskAddress, from) - to := and(bitmaskAddress, to) - // Load the ownership data. - mstore(0x00, id) - mstore(0x1c, or(_ERC721_MASTER_SLOT_SEED, caller())) - let ownershipSlot := add(id, add(id, keccak256(0x00, 0x20))) - let ownershipPacked := sload(ownershipSlot) - let owner := and(bitmaskAddress, ownershipPacked) - // Revert if `from` is not the owner, or does not exist. - if iszero(mul(owner, eq(owner, from))) { - if iszero(owner) { - mstore(0x00, 0xceea21b6) // `TokenDoesNotExist()`. - revert(0x1c, 0x04) - } - mstore(0x00, 0xa1148100) // `TransferFromIncorrectOwner()`. - revert(0x1c, 0x04) - } - // Revert if `to` is the zero address. - if iszero(to) { - mstore(0x00, 0xea553b34) // `TransferToZeroAddress()`. - revert(0x1c, 0x04) - } - // Load, check, and update the token approval. - { - mstore(0x00, from) - let approvedAddress := sload(add(1, ownershipSlot)) - // Revert if the caller is not the owner, nor approved. - if iszero(or(eq(caller(), from), eq(caller(), approvedAddress))) { - if iszero(sload(keccak256(0x0c, 0x30))) { - mstore(0x00, 0x4b6e7f18) // `NotOwnerNorApproved()`. - revert(0x1c, 0x04) - } - } - // Delete the approved address if any. - if approvedAddress { sstore(add(1, ownershipSlot), 0) } - } - // Update with the new owner. - sstore(ownershipSlot, xor(ownershipPacked, xor(from, to))) - // Decrement the balance of `from`. - { - let fromBalanceSlot := keccak256(0x0c, 0x1c) - sstore(fromBalanceSlot, sub(sload(fromBalanceSlot), 1)) - } - // Increment the balance of `to`. - { - mstore(0x00, to) - let toBalanceSlot := keccak256(0x0c, 0x1c) - let toBalanceSlotPacked := add(sload(toBalanceSlot), 1) - if iszero(and(toBalanceSlotPacked, _MAX_ACCOUNT_BALANCE)) { - mstore(0x00, 0x01336cea) // `AccountBalanceOverflow()`. - revert(0x1c, 0x04) - } - sstore(toBalanceSlot, toBalanceSlotPacked) - } - // Emit the {Transfer} event. - log4(codesize(), 0x00, _TRANSFER_EVENT_SIGNATURE, from, to, id) - } - _afterTokenTransfer(from, to, id); - } - - /// @dev Equivalent to `safeTransferFrom(from, to, id, "")`. - function safeTransferFrom(address from, address to, uint256 id) public payable virtual { - transferFrom(from, to, id); - if (_hasCode(to)) _checkOnERC721Received(from, to, id, ""); - } - - /// @dev Transfers token `id` from `from` to `to`. - /// - /// Requirements: - /// - /// - Token `id` must exist. - /// - `from` must be the owner of the token. - /// - `to` cannot be the zero address. - /// - The caller must be the owner of the token, or be approved to manage the token. - /// - If `to` refers to a smart contract, it must implement - /// {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. - /// - /// Emits a {Transfer} event. - function safeTransferFrom(address from, address to, uint256 id, bytes calldata data) - public - payable - virtual - { - transferFrom(from, to, id); - if (_hasCode(to)) _checkOnERC721Received(from, to, id, data); - } - - /// @dev Returns true if this contract implements the interface defined by `interfaceId`. - /// See: https://eips.ethereum.org/EIPS/eip-165 - /// This function call must use less than 30000 gas. - function supportsInterface(bytes4 interfaceId) public view virtual returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - let s := shr(224, interfaceId) - // ERC165: 0x01ffc9a7, ERC721: 0x80ac58cd, ERC721Metadata: 0x5b5e139f. - result := or(or(eq(s, 0x01ffc9a7), eq(s, 0x80ac58cd)), eq(s, 0x5b5e139f)) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL QUERY FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns if token `id` exists. - function _exists(uint256 id) internal view virtual returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, id) - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - result := iszero(iszero(shl(96, sload(add(id, add(id, keccak256(0x00, 0x20))))))) - } - } - - /// @dev Returns the owner of token `id`. - /// Returns the zero address instead of reverting if the token does not exist. - function _ownerOf(uint256 id) internal view virtual returns (address result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, id) - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - result := shr(96, shl(96, sload(add(id, add(id, keccak256(0x00, 0x20)))))) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL DATA HITCHHIKING FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // For performance, no events are emitted for the hitchhiking setters. - // Please emit your own events if required. - - /// @dev Returns the auxiliary data for `owner`. - /// Minting, transferring, burning the tokens of `owner` will not change the auxiliary data. - /// Auxiliary data can be set for any address, even if it does not have any tokens. - function _getAux(address owner) internal view virtual returns (uint224 result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - mstore(0x00, owner) - result := shr(32, sload(keccak256(0x0c, 0x1c))) - } - } - - /// @dev Set the auxiliary data for `owner` to `value`. - /// Minting, transferring, burning the tokens of `owner` will not change the auxiliary data. - /// Auxiliary data can be set for any address, even if it does not have any tokens. - function _setAux(address owner, uint224 value) internal virtual { - /// @solidity memory-safe-assembly - assembly { - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - mstore(0x00, owner) - let balanceSlot := keccak256(0x0c, 0x1c) - let packed := sload(balanceSlot) - sstore(balanceSlot, xor(packed, shl(32, xor(value, shr(32, packed))))) - } - } - - /// @dev Returns the extra data for token `id`. - /// Minting, transferring, burning a token will not change the extra data. - /// The extra data can be set on a non-existent token. - function _getExtraData(uint256 id) internal view virtual returns (uint96 result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, id) - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - result := shr(160, sload(add(id, add(id, keccak256(0x00, 0x20))))) - } - } - - /// @dev Sets the extra data for token `id` to `value`. - /// Minting, transferring, burning a token will not change the extra data. - /// The extra data can be set on a non-existent token. - function _setExtraData(uint256 id, uint96 value) internal virtual { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, id) - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - let ownershipSlot := add(id, add(id, keccak256(0x00, 0x20))) - let packed := sload(ownershipSlot) - sstore(ownershipSlot, xor(packed, shl(160, xor(value, shr(160, packed))))) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL MINT FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Mints token `id` to `to`. - /// - /// Requirements: - /// - /// - Token `id` must not exist. - /// - `to` cannot be the zero address. - /// - /// Emits a {Transfer} event. - function _mint(address to, uint256 id) internal virtual { - _beforeTokenTransfer(address(0), to, id); - /// @solidity memory-safe-assembly - assembly { - // Clear the upper 96 bits. - to := shr(96, shl(96, to)) - // Revert if `to` is the zero address. - if iszero(to) { - mstore(0x00, 0xea553b34) // `TransferToZeroAddress()`. - revert(0x1c, 0x04) - } - // Load the ownership data. - mstore(0x00, id) - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - let ownershipSlot := add(id, add(id, keccak256(0x00, 0x20))) - let ownershipPacked := sload(ownershipSlot) - // Revert if the token already exists. - if shl(96, ownershipPacked) { - mstore(0x00, 0xc991cbb1) // `TokenAlreadyExists()`. - revert(0x1c, 0x04) - } - // Update with the owner. - sstore(ownershipSlot, or(ownershipPacked, to)) - // Increment the balance of the owner. - { - mstore(0x00, to) - let balanceSlot := keccak256(0x0c, 0x1c) - let balanceSlotPacked := add(sload(balanceSlot), 1) - if iszero(and(balanceSlotPacked, _MAX_ACCOUNT_BALANCE)) { - mstore(0x00, 0x01336cea) // `AccountBalanceOverflow()`. - revert(0x1c, 0x04) - } - sstore(balanceSlot, balanceSlotPacked) - } - // Emit the {Transfer} event. - log4(codesize(), 0x00, _TRANSFER_EVENT_SIGNATURE, 0, to, id) - } - _afterTokenTransfer(address(0), to, id); - } - - /// @dev Equivalent to `_safeMint(to, id, "")`. - function _safeMint(address to, uint256 id) internal virtual { - _safeMint(to, id, ""); - } - - /// @dev Mints token `id` to `to`. - /// - /// Requirements: - /// - /// - Token `id` must not exist. - /// - `to` cannot be the zero address. - /// - If `to` refers to a smart contract, it must implement - /// {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. - /// - /// Emits a {Transfer} event. - function _safeMint(address to, uint256 id, bytes memory data) internal virtual { - _mint(to, id); - if (_hasCode(to)) _checkOnERC721Received(address(0), to, id, data); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL BURN FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Equivalent to `_burn(address(0), id)`. - function _burn(uint256 id) internal virtual { - _burn(address(0), id); - } - - /// @dev Destroys token `id`, using `by`. - /// - /// Requirements: - /// - /// - Token `id` must exist. - /// - If `by` is not the zero address, - /// it must be the owner of the token, or be approved to manage the token. - /// - /// Emits a {Transfer} event. - function _burn(address by, uint256 id) internal virtual { - address owner = ownerOf(id); - _beforeTokenTransfer(owner, address(0), id); - /// @solidity memory-safe-assembly - assembly { - // Clear the upper 96 bits. - by := shr(96, shl(96, by)) - // Load the ownership data. - mstore(0x00, id) - mstore(0x1c, or(_ERC721_MASTER_SLOT_SEED, by)) - let ownershipSlot := add(id, add(id, keccak256(0x00, 0x20))) - let ownershipPacked := sload(ownershipSlot) - // Reload the owner in case it is changed in `_beforeTokenTransfer`. - owner := shr(96, shl(96, ownershipPacked)) - // Revert if the token does not exist. - if iszero(owner) { - mstore(0x00, 0xceea21b6) // `TokenDoesNotExist()`. - revert(0x1c, 0x04) - } - // Load and check the token approval. - { - mstore(0x00, owner) - let approvedAddress := sload(add(1, ownershipSlot)) - // If `by` is not the zero address, do the authorization check. - // Revert if the `by` is not the owner, nor approved. - if iszero(or(iszero(by), or(eq(by, owner), eq(by, approvedAddress)))) { - if iszero(sload(keccak256(0x0c, 0x30))) { - mstore(0x00, 0x4b6e7f18) // `NotOwnerNorApproved()`. - revert(0x1c, 0x04) - } - } - // Delete the approved address if any. - if approvedAddress { sstore(add(1, ownershipSlot), 0) } - } - // Clear the owner. - sstore(ownershipSlot, xor(ownershipPacked, owner)) - // Decrement the balance of `owner`. - { - let balanceSlot := keccak256(0x0c, 0x1c) - sstore(balanceSlot, sub(sload(balanceSlot), 1)) - } - // Emit the {Transfer} event. - log4(codesize(), 0x00, _TRANSFER_EVENT_SIGNATURE, owner, 0, id) - } - _afterTokenTransfer(owner, address(0), id); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL APPROVAL FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns whether `account` is the owner of token `id`, or is approved to manage it. - /// - /// Requirements: - /// - Token `id` must exist. - function _isApprovedOrOwner(address account, uint256 id) - internal - view - virtual - returns (bool result) - { - /// @solidity memory-safe-assembly - assembly { - result := 1 - // Clear the upper 96 bits. - account := shr(96, shl(96, account)) - // Load the ownership data. - mstore(0x00, id) - mstore(0x1c, or(_ERC721_MASTER_SLOT_SEED, account)) - let ownershipSlot := add(id, add(id, keccak256(0x00, 0x20))) - let owner := shr(96, shl(96, sload(ownershipSlot))) - // Revert if the token does not exist. - if iszero(owner) { - mstore(0x00, 0xceea21b6) // `TokenDoesNotExist()`. - revert(0x1c, 0x04) - } - // Check if `account` is the `owner`. - if iszero(eq(account, owner)) { - mstore(0x00, owner) - // Check if `account` is approved to manage the token. - if iszero(sload(keccak256(0x0c, 0x30))) { - result := eq(account, sload(add(1, ownershipSlot))) - } - } - } - } - - /// @dev Returns the account approved to manage token `id`. - /// Returns the zero address instead of reverting if the token does not exist. - function _getApproved(uint256 id) internal view virtual returns (address result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, id) - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - result := sload(add(1, add(id, add(id, keccak256(0x00, 0x20))))) - } - } - - /// @dev Equivalent to `_approve(address(0), account, id)`. - function _approve(address account, uint256 id) internal virtual { - _approve(address(0), account, id); - } - - /// @dev Sets `account` as the approved account to manage token `id`, using `by`. - /// - /// Requirements: - /// - Token `id` must exist. - /// - If `by` is not the zero address, `by` must be the owner - /// or an approved operator for the token owner. - /// - /// Emits a {Transfer} event. - function _approve(address by, address account, uint256 id) internal virtual { - assembly { - // Clear the upper 96 bits. - let bitmaskAddress := shr(96, not(0)) - account := and(bitmaskAddress, account) - by := and(bitmaskAddress, by) - // Load the owner of the token. - mstore(0x00, id) - mstore(0x1c, or(_ERC721_MASTER_SLOT_SEED, by)) - let ownershipSlot := add(id, add(id, keccak256(0x00, 0x20))) - let owner := and(bitmaskAddress, sload(ownershipSlot)) - // Revert if the token does not exist. - if iszero(owner) { - mstore(0x00, 0xceea21b6) // `TokenDoesNotExist()`. - revert(0x1c, 0x04) - } - // If `by` is not the zero address, do the authorization check. - // Revert if `by` is not the owner, nor approved. - if iszero(or(iszero(by), eq(by, owner))) { - mstore(0x00, owner) - if iszero(sload(keccak256(0x0c, 0x30))) { - mstore(0x00, 0x4b6e7f18) // `NotOwnerNorApproved()`. - revert(0x1c, 0x04) - } - } - // Sets `account` as the approved account to manage `id`. - sstore(add(1, ownershipSlot), account) - // Emit the {Approval} event. - log4(codesize(), 0x00, _APPROVAL_EVENT_SIGNATURE, owner, account, id) - } - } - - /// @dev Approve or remove the `operator` as an operator for `by`, - /// without authorization checks. - /// - /// Emits an {ApprovalForAll} event. - function _setApprovalForAll(address by, address operator, bool isApproved) internal virtual { - /// @solidity memory-safe-assembly - assembly { - // Clear the upper 96 bits. - by := shr(96, shl(96, by)) - operator := shr(96, shl(96, operator)) - // Convert to 0 or 1. - isApproved := iszero(iszero(isApproved)) - // Update the `isApproved` for (`by`, `operator`). - mstore(0x1c, or(_ERC721_MASTER_SLOT_SEED, operator)) - mstore(0x00, by) - sstore(keccak256(0x0c, 0x30), isApproved) - // Emit the {ApprovalForAll} event. - mstore(0x00, isApproved) - log3(0x00, 0x20, _APPROVAL_FOR_ALL_EVENT_SIGNATURE, by, operator) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTERNAL TRANSFER FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Equivalent to `_transfer(address(0), from, to, id)`. - function _transfer(address from, address to, uint256 id) internal virtual { - _transfer(address(0), from, to, id); - } - - /// @dev Transfers token `id` from `from` to `to`. - /// - /// Requirements: - /// - /// - Token `id` must exist. - /// - `from` must be the owner of the token. - /// - `to` cannot be the zero address. - /// - If `by` is not the zero address, - /// it must be the owner of the token, or be approved to manage the token. - /// - /// Emits a {Transfer} event. - function _transfer(address by, address from, address to, uint256 id) internal virtual { - _beforeTokenTransfer(from, to, id); - /// @solidity memory-safe-assembly - assembly { - // Clear the upper 96 bits. - let bitmaskAddress := shr(96, not(0)) - from := and(bitmaskAddress, from) - to := and(bitmaskAddress, to) - by := and(bitmaskAddress, by) - // Load the ownership data. - mstore(0x00, id) - mstore(0x1c, or(_ERC721_MASTER_SLOT_SEED, by)) - let ownershipSlot := add(id, add(id, keccak256(0x00, 0x20))) - let ownershipPacked := sload(ownershipSlot) - let owner := and(bitmaskAddress, ownershipPacked) - // Revert if `from` is not the owner, or does not exist. - if iszero(mul(owner, eq(owner, from))) { - if iszero(owner) { - mstore(0x00, 0xceea21b6) // `TokenDoesNotExist()`. - revert(0x1c, 0x04) - } - mstore(0x00, 0xa1148100) // `TransferFromIncorrectOwner()`. - revert(0x1c, 0x04) - } - // Revert if `to` is the zero address. - if iszero(to) { - mstore(0x00, 0xea553b34) // `TransferToZeroAddress()`. - revert(0x1c, 0x04) - } - // Load, check, and update the token approval. - { - mstore(0x00, from) - let approvedAddress := sload(add(1, ownershipSlot)) - // If `by` is not the zero address, do the authorization check. - // Revert if the `by` is not the owner, nor approved. - if iszero(or(iszero(by), or(eq(by, from), eq(by, approvedAddress)))) { - if iszero(sload(keccak256(0x0c, 0x30))) { - mstore(0x00, 0x4b6e7f18) // `NotOwnerNorApproved()`. - revert(0x1c, 0x04) - } - } - // Delete the approved address if any. - if approvedAddress { sstore(add(1, ownershipSlot), 0) } - } - // Update with the new owner. - sstore(ownershipSlot, xor(ownershipPacked, xor(from, to))) - // Decrement the balance of `from`. - { - let fromBalanceSlot := keccak256(0x0c, 0x1c) - sstore(fromBalanceSlot, sub(sload(fromBalanceSlot), 1)) - } - // Increment the balance of `to`. - { - mstore(0x00, to) - let toBalanceSlot := keccak256(0x0c, 0x1c) - let toBalanceSlotPacked := add(sload(toBalanceSlot), 1) - if iszero(and(toBalanceSlotPacked, _MAX_ACCOUNT_BALANCE)) { - mstore(0x00, 0x01336cea) // `AccountBalanceOverflow()`. - revert(0x1c, 0x04) - } - sstore(toBalanceSlot, toBalanceSlotPacked) - } - // Emit the {Transfer} event. - log4(codesize(), 0x00, _TRANSFER_EVENT_SIGNATURE, from, to, id) - } - _afterTokenTransfer(from, to, id); - } - - /// @dev Equivalent to `_safeTransfer(from, to, id, "")`. - function _safeTransfer(address from, address to, uint256 id) internal virtual { - _safeTransfer(from, to, id, ""); - } - - /// @dev Transfers token `id` from `from` to `to`. - /// - /// Requirements: - /// - /// - Token `id` must exist. - /// - `from` must be the owner of the token. - /// - `to` cannot be the zero address. - /// - The caller must be the owner of the token, or be approved to manage the token. - /// - If `to` refers to a smart contract, it must implement - /// {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. - /// - /// Emits a {Transfer} event. - function _safeTransfer(address from, address to, uint256 id, bytes memory data) - internal - virtual - { - _transfer(address(0), from, to, id); - if (_hasCode(to)) _checkOnERC721Received(from, to, id, data); - } - - /// @dev Equivalent to `_safeTransfer(by, from, to, id, "")`. - function _safeTransfer(address by, address from, address to, uint256 id) internal virtual { - _safeTransfer(by, from, to, id, ""); - } - - /// @dev Transfers token `id` from `from` to `to`. - /// - /// Requirements: - /// - /// - Token `id` must exist. - /// - `from` must be the owner of the token. - /// - `to` cannot be the zero address. - /// - If `by` is not the zero address, - /// it must be the owner of the token, or be approved to manage the token. - /// - If `to` refers to a smart contract, it must implement - /// {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. - /// - /// Emits a {Transfer} event. - function _safeTransfer(address by, address from, address to, uint256 id, bytes memory data) - internal - virtual - { - _transfer(by, from, to, id); - if (_hasCode(to)) _checkOnERC721Received(from, to, id, data); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* HOOKS FOR OVERRIDING */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Hook that is called before any token transfers, including minting and burning. - function _beforeTokenTransfer(address from, address to, uint256 id) internal virtual {} - - /// @dev Hook that is called after any token transfers, including minting and burning. - function _afterTokenTransfer(address from, address to, uint256 id) internal virtual {} - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PRIVATE HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns if `a` has bytecode of non-zero length. - function _hasCode(address a) private view returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - result := extcodesize(a) // Can handle dirty upper bits. - } - } - - /// @dev Perform a call to invoke {IERC721Receiver-onERC721Received} on `to`. - /// Reverts if the target does not support the function correctly. - function _checkOnERC721Received(address from, address to, uint256 id, bytes memory data) - private - { - /// @solidity memory-safe-assembly - assembly { - // Prepare the calldata. - let m := mload(0x40) - let onERC721ReceivedSelector := 0x150b7a02 - mstore(m, onERC721ReceivedSelector) - mstore(add(m, 0x20), caller()) // The `operator`, which is always `msg.sender`. - mstore(add(m, 0x40), shr(96, shl(96, from))) - mstore(add(m, 0x60), id) - mstore(add(m, 0x80), 0x80) - let n := mload(data) - mstore(add(m, 0xa0), n) - if n { pop(staticcall(gas(), 4, add(data, 0x20), n, add(m, 0xc0), n)) } - // Revert if the call reverts. - if iszero(call(gas(), to, 0, add(m, 0x1c), add(n, 0xa4), m, 0x20)) { - if returndatasize() { - // Bubble up the revert if the call reverts. - returndatacopy(m, 0x00, returndatasize()) - revert(m, returndatasize()) - } - } - // Load the returndata and compare it. - if iszero(eq(mload(m), shl(224, onERC721ReceivedSelector))) { - mstore(0x00, 0xd1a57ed6) // `TransferToNonERC721ReceiverImplementer()`. - revert(0x1c, 0x04) - } - } - } -} diff --git a/lib/solady/src/tokens/WETH.sol b/lib/solady/src/tokens/WETH.sol deleted file mode 100644 index a0e3841..0000000 --- a/lib/solady/src/tokens/WETH.sol +++ /dev/null @@ -1,89 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {ERC20} from "./ERC20.sol"; - -/// @notice Simple Wrapped Ether implementation. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/tokens/WETH.sol) -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/WETH.sol) -/// @author Inspired by WETH9 (https://github.com/dapphub/ds-weth/blob/master/src/weth9.sol) -contract WETH is ERC20 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The ETH transfer has failed. - error ETHTransferFailed(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EVENTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Emitted when `amount` is deposited from `from`. - event Deposit(address indexed from, uint256 amount); - - /// @dev Emitted when `amount` is withdrawn to `to`. - event Withdrawal(address indexed to, uint256 amount); - - /// @dev `keccak256(bytes("Deposit(address,uint256)"))`. - uint256 private constant _DEPOSIT_EVENT_SIGNATURE = - 0xe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c; - - /// @dev `keccak256(bytes("Withdrawal(address,uint256)"))`. - uint256 private constant _WITHDRAWAL_EVENT_SIGNATURE = - 0x7fcf532c15f0a6db0bd6d0e038bea71d30d808c7d98cb3bf7268a95bf5081b65; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC20 METADATA */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the name of the token. - function name() public view virtual override returns (string memory) { - return "Wrapped Ether"; - } - - /// @dev Returns the symbol of the token. - function symbol() public view virtual override returns (string memory) { - return "WETH"; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* WETH */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Deposits `amount` ETH of the caller and mints `amount` WETH to the caller. - /// - /// Emits a {Deposit} event. - function deposit() public payable virtual { - _mint(msg.sender, msg.value); - /// @solidity memory-safe-assembly - assembly { - // Emit the {Deposit} event. - mstore(0x00, callvalue()) - log2(0x00, 0x20, _DEPOSIT_EVENT_SIGNATURE, caller()) - } - } - - /// @dev Burns `amount` WETH of the caller and sends `amount` ETH to the caller. - /// - /// Emits a {Withdrawal} event. - function withdraw(uint256 amount) public virtual { - _burn(msg.sender, amount); - /// @solidity memory-safe-assembly - assembly { - // Emit the {Withdrawal} event. - mstore(0x00, amount) - log2(0x00, 0x20, _WITHDRAWAL_EVENT_SIGNATURE, caller()) - // Transfer the ETH and check if it succeeded or not. - if iszero(call(gas(), caller(), amount, codesize(), 0x00, codesize(), 0x00)) { - mstore(0x00, 0xb12d13eb) // `ETHTransferFailed()`. - revert(0x1c, 0x04) - } - } - } - - /// @dev Equivalent to `deposit()`. - receive() external payable virtual { - deposit(); - } -} diff --git a/lib/solady/src/utils/Base64.sol b/lib/solady/src/utils/Base64.sol deleted file mode 100644 index 17754bc..0000000 --- a/lib/solady/src/utils/Base64.sol +++ /dev/null @@ -1,166 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library to encode strings in Base64. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/Base64.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/Base64.sol) -/// @author Modified from (https://github.com/Brechtpd/base64/blob/main/base64.sol) by Brecht Devos - . -library Base64 { - /// @dev Encodes `data` using the base64 encoding described in RFC 4648. - /// See: https://datatracker.ietf.org/doc/html/rfc4648 - /// @param fileSafe Whether to replace '+' with '-' and '/' with '_'. - /// @param noPadding Whether to strip away the padding. - function encode(bytes memory data, bool fileSafe, bool noPadding) - internal - pure - returns (string memory result) - { - /// @solidity memory-safe-assembly - assembly { - let dataLength := mload(data) - - if dataLength { - // Multiply by 4/3 rounded up. - // The `shl(2, ...)` is equivalent to multiplying by 4. - let encodedLength := shl(2, div(add(dataLength, 2), 3)) - - // Set `result` to point to the start of the free memory. - result := mload(0x40) - - // Store the table into the scratch space. - // Offsetted by -1 byte so that the `mload` will load the character. - // We will rewrite the free memory pointer at `0x40` later with - // the allocated size. - // The magic constant 0x0670 will turn "-_" into "+/". - mstore(0x1f, "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdef") - mstore(0x3f, xor("ghijklmnopqrstuvwxyz0123456789-_", mul(iszero(fileSafe), 0x0670))) - - // Skip the first slot, which stores the length. - let ptr := add(result, 0x20) - let end := add(ptr, encodedLength) - - // Run over the input, 3 bytes at a time. - for {} 1 {} { - data := add(data, 3) // Advance 3 bytes. - let input := mload(data) - - // Write 4 bytes. Optimized for fewer stack operations. - mstore8(0, mload(and(shr(18, input), 0x3F))) - mstore8(1, mload(and(shr(12, input), 0x3F))) - mstore8(2, mload(and(shr(6, input), 0x3F))) - mstore8(3, mload(and(input, 0x3F))) - mstore(ptr, mload(0x00)) - - ptr := add(ptr, 4) // Advance 4 bytes. - if iszero(lt(ptr, end)) { break } - } - mstore(0x40, add(end, 0x20)) // Allocate the memory. - // Equivalent to `o = [0, 2, 1][dataLength % 3]`. - let o := div(2, mod(dataLength, 3)) - // Offset `ptr` and pad with '='. We can simply write over the end. - mstore(sub(ptr, o), shl(240, 0x3d3d)) - // Set `o` to zero if there is padding. - o := mul(iszero(iszero(noPadding)), o) - mstore(sub(ptr, o), 0) // Zeroize the slot after the string. - mstore(result, sub(encodedLength, o)) // Store the length. - } - } - } - - /// @dev Encodes `data` using the base64 encoding described in RFC 4648. - /// Equivalent to `encode(data, false, false)`. - function encode(bytes memory data) internal pure returns (string memory result) { - result = encode(data, false, false); - } - - /// @dev Encodes `data` using the base64 encoding described in RFC 4648. - /// Equivalent to `encode(data, fileSafe, false)`. - function encode(bytes memory data, bool fileSafe) - internal - pure - returns (string memory result) - { - result = encode(data, fileSafe, false); - } - - /// @dev Decodes base64 encoded `data`. - /// - /// Supports: - /// - RFC 4648 (both standard and file-safe mode). - /// - RFC 3501 (63: ','). - /// - /// Does not support: - /// - Line breaks. - /// - /// Note: For performance reasons, - /// this function will NOT revert on invalid `data` inputs. - /// Outputs for invalid inputs will simply be undefined behaviour. - /// It is the user's responsibility to ensure that the `data` - /// is a valid base64 encoded string. - function decode(string memory data) internal pure returns (bytes memory result) { - /// @solidity memory-safe-assembly - assembly { - let dataLength := mload(data) - - if dataLength { - let decodedLength := mul(shr(2, dataLength), 3) - - for {} 1 {} { - // If padded. - if iszero(and(dataLength, 3)) { - let t := xor(mload(add(data, dataLength)), 0x3d3d) - // forgefmt: disable-next-item - decodedLength := sub( - decodedLength, - add(iszero(byte(30, t)), iszero(byte(31, t))) - ) - break - } - // If non-padded. - decodedLength := add(decodedLength, sub(and(dataLength, 3), 1)) - break - } - result := mload(0x40) - - // Write the length of the bytes. - mstore(result, decodedLength) - - // Skip the first slot, which stores the length. - let ptr := add(result, 0x20) - let end := add(ptr, decodedLength) - - // Load the table into the scratch space. - // Constants are optimized for smaller bytecode with zero gas overhead. - // `m` also doubles as the mask of the upper 6 bits. - let m := 0xfc000000fc00686c7074787c8084888c9094989ca0a4a8acb0b4b8bcc0c4c8cc - mstore(0x5b, m) - mstore(0x3b, 0x04080c1014181c2024282c3034383c4044484c5054585c6064) - mstore(0x1a, 0xf8fcf800fcd0d4d8dce0e4e8ecf0f4) - - for {} 1 {} { - // Read 4 bytes. - data := add(data, 4) - let input := mload(data) - - // Write 3 bytes. - // forgefmt: disable-next-item - mstore(ptr, or( - and(m, mload(byte(28, input))), - shr(6, or( - and(m, mload(byte(29, input))), - shr(6, or( - and(m, mload(byte(30, input))), - shr(6, mload(byte(31, input))) - )) - )) - )) - ptr := add(ptr, 3) - if iszero(lt(ptr, end)) { break } - } - mstore(0x40, add(end, 0x20)) // Allocate the memory. - mstore(end, 0) // Zeroize the slot after the bytes. - mstore(0x60, 0) // Restore the zero slot. - } - } - } -} diff --git a/lib/solady/src/utils/CREATE3.sol b/lib/solady/src/utils/CREATE3.sol deleted file mode 100644 index dc8dcc8..0000000 --- a/lib/solady/src/utils/CREATE3.sol +++ /dev/null @@ -1,153 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Deploy to deterministic addresses without an initcode factor. -/// @author Solady (https://github.com/vectorized/solmady/blob/main/src/utils/CREATE3.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/CREATE3.sol) -/// @author Modified from 0xSequence (https://github.com/0xSequence/create3/blob/master/contracts/Create3.sol) -library CREATE3 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Unable to deploy the contract. - error DeploymentFailed(); - - /// @dev Unable to initialize the contract. - error InitializationFailed(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* BYTECODE CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /** - * -------------------------------------------------------------------+ - * Opcode | Mnemonic | Stack | Memory | - * -------------------------------------------------------------------| - * 36 | CALLDATASIZE | cds | | - * 3d | RETURNDATASIZE | 0 cds | | - * 3d | RETURNDATASIZE | 0 0 cds | | - * 37 | CALLDATACOPY | | [0..cds): calldata | - * 36 | CALLDATASIZE | cds | [0..cds): calldata | - * 3d | RETURNDATASIZE | 0 cds | [0..cds): calldata | - * 34 | CALLVALUE | value 0 cds | [0..cds): calldata | - * f0 | CREATE | newContract | [0..cds): calldata | - * -------------------------------------------------------------------| - * Opcode | Mnemonic | Stack | Memory | - * -------------------------------------------------------------------| - * 67 bytecode | PUSH8 bytecode | bytecode | | - * 3d | RETURNDATASIZE | 0 bytecode | | - * 52 | MSTORE | | [0..8): bytecode | - * 60 0x08 | PUSH1 0x08 | 0x08 | [0..8): bytecode | - * 60 0x18 | PUSH1 0x18 | 0x18 0x08 | [0..8): bytecode | - * f3 | RETURN | | [0..8): bytecode | - * -------------------------------------------------------------------+ - */ - - /// @dev The proxy bytecode. - uint256 private constant _PROXY_BYTECODE = 0x67363d3d37363d34f03d5260086018f3; - - /// @dev Hash of the `_PROXY_BYTECODE`. - /// Equivalent to `keccak256(abi.encodePacked(hex"67363d3d37363d34f03d5260086018f3"))`. - bytes32 private constant _PROXY_BYTECODE_HASH = - 0x21c35dbe1b344a2488cf3321d6ce542f8e9f305544ff09e4993a62319a497c1f; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CREATE3 OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Deploys `creationCode` deterministically with a `salt`. - /// The deployed contract is funded with `value` (in wei) ETH. - /// Returns the deterministic address of the deployed contract, - /// which solely depends on `salt`. - function deploy(bytes32 salt, bytes memory creationCode, uint256 value) - internal - returns (address deployed) - { - /// @solidity memory-safe-assembly - assembly { - // Store the `_PROXY_BYTECODE` into scratch space. - mstore(0x00, _PROXY_BYTECODE) - // Deploy a new contract with our pre-made bytecode via CREATE2. - let proxy := create2(0, 0x10, 0x10, salt) - - // If the result of `create2` is the zero address, revert. - if iszero(proxy) { - // Store the function selector of `DeploymentFailed()`. - mstore(0x00, 0x30116425) - // Revert with (offset, size). - revert(0x1c, 0x04) - } - - // Store the proxy's address. - mstore(0x14, proxy) - // 0xd6 = 0xc0 (short RLP prefix) + 0x16 (length of: 0x94 ++ proxy ++ 0x01). - // 0x94 = 0x80 + 0x14 (0x14 = the length of an address, 20 bytes, in hex). - mstore(0x00, 0xd694) - // Nonce of the proxy contract (1). - mstore8(0x34, 0x01) - - deployed := keccak256(0x1e, 0x17) - - // If the `call` fails, revert. - if iszero( - call( - gas(), // Gas remaining. - proxy, // Proxy's address. - value, // Ether value. - add(creationCode, 0x20), // Start of `creationCode`. - mload(creationCode), // Length of `creationCode`. - 0x00, // Offset of output. - 0x00 // Length of output. - ) - ) { - // Store the function selector of `InitializationFailed()`. - mstore(0x00, 0x19b991a8) - // Revert with (offset, size). - revert(0x1c, 0x04) - } - - // If the code size of `deployed` is zero, revert. - if iszero(extcodesize(deployed)) { - // Store the function selector of `InitializationFailed()`. - mstore(0x00, 0x19b991a8) - // Revert with (offset, size). - revert(0x1c, 0x04) - } - } - } - - /// @dev Returns the deterministic address for `salt` with `deployer`. - function getDeployed(bytes32 salt, address deployer) internal pure returns (address deployed) { - /// @solidity memory-safe-assembly - assembly { - // Cache the free memory pointer. - let m := mload(0x40) - // Store `deployer`. - mstore(0x00, deployer) - // Store the prefix. - mstore8(0x0b, 0xff) - // Store the salt. - mstore(0x20, salt) - // Store the bytecode hash. - mstore(0x40, _PROXY_BYTECODE_HASH) - - // Store the proxy's address. - mstore(0x14, keccak256(0x0b, 0x55)) - // Restore the free memory pointer. - mstore(0x40, m) - // 0xd6 = 0xc0 (short RLP prefix) + 0x16 (length of: 0x94 ++ proxy ++ 0x01). - // 0x94 = 0x80 + 0x14 (0x14 = the length of an address, 20 bytes, in hex). - mstore(0x00, 0xd694) - // Nonce of the proxy contract (1). - mstore8(0x34, 0x01) - - deployed := keccak256(0x1e, 0x17) - } - } - - /// @dev Returns the deterministic address for `salt`. - function getDeployed(bytes32 salt) internal view returns (address deployed) { - deployed = getDeployed(salt, address(this)); - } -} diff --git a/lib/solady/src/utils/Clone.sol b/lib/solady/src/utils/Clone.sol deleted file mode 100644 index 697a1ea..0000000 --- a/lib/solady/src/utils/Clone.sol +++ /dev/null @@ -1,387 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Class with helper read functions for clone with immutable args. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/Clone.sol) -/// @author Adapted from clones with immutable args by zefram.eth, Saw-mon & Natalie -/// (https://github.com/Saw-mon-and-Natalie/clones-with-immutable-args) -abstract contract Clone { - /// @dev Reads all of the immutable args. - function _getArgBytes() internal pure returns (bytes memory arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := mload(0x40) - let length := sub(calldatasize(), add(2, offset)) // 2 bytes are used for the length. - mstore(arg, length) // Store the length. - calldatacopy(add(arg, 0x20), offset, length) - let o := add(add(arg, 0x20), length) - mstore(o, 0) // Zeroize the slot after the bytes. - mstore(0x40, add(o, 0x20)) // Allocate the memory. - } - } - - /// @dev Reads an immutable arg with type bytes. - function _getArgBytes(uint256 argOffset, uint256 length) - internal - pure - returns (bytes memory arg) - { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := mload(0x40) - mstore(arg, length) // Store the length. - calldatacopy(add(arg, 0x20), add(offset, argOffset), length) - let o := add(add(arg, 0x20), length) - mstore(o, 0) // Zeroize the slot after the bytes. - mstore(0x40, add(o, 0x20)) // Allocate the memory. - } - } - - /// @dev Reads an immutable arg with type address. - function _getArgAddress(uint256 argOffset) internal pure returns (address arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(96, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads a uint256 array stored in the immutable args. - function _getArgUint256Array(uint256 argOffset, uint256 length) - internal - pure - returns (uint256[] memory arg) - { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := mload(0x40) - mstore(arg, length) // Store the length. - calldatacopy(add(arg, 0x20), add(offset, argOffset), shl(5, length)) - mstore(0x40, add(add(arg, 0x20), shl(5, length))) // Allocate the memory. - } - } - - /// @dev Reads a bytes32 array stored in the immutable args. - function _getArgBytes32Array(uint256 argOffset, uint256 length) - internal - pure - returns (bytes32[] memory arg) - { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := mload(0x40) - mstore(arg, length) // Store the length. - calldatacopy(add(arg, 0x20), add(offset, argOffset), shl(5, length)) - mstore(0x40, add(add(arg, 0x20), shl(5, length))) // Allocate the memory. - } - } - - /// @dev Reads an immutable arg with type bytes32. - function _getArgBytes32(uint256 argOffset) internal pure returns (bytes32 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := calldataload(add(offset, argOffset)) - } - } - - /// @dev Reads an immutable arg with type uint256. - function _getArgUint256(uint256 argOffset) internal pure returns (uint256 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := calldataload(add(offset, argOffset)) - } - } - - /// @dev Reads an immutable arg with type uint248. - function _getArgUint248(uint256 argOffset) internal pure returns (uint248 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(8, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint240. - function _getArgUint240(uint256 argOffset) internal pure returns (uint240 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(16, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint232. - function _getArgUint232(uint256 argOffset) internal pure returns (uint232 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(24, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint224. - function _getArgUint224(uint256 argOffset) internal pure returns (uint224 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(0x20, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint216. - function _getArgUint216(uint256 argOffset) internal pure returns (uint216 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(40, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint208. - function _getArgUint208(uint256 argOffset) internal pure returns (uint208 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(48, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint200. - function _getArgUint200(uint256 argOffset) internal pure returns (uint200 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(56, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint192. - function _getArgUint192(uint256 argOffset) internal pure returns (uint192 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(64, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint184. - function _getArgUint184(uint256 argOffset) internal pure returns (uint184 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(72, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint176. - function _getArgUint176(uint256 argOffset) internal pure returns (uint176 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(80, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint168. - function _getArgUint168(uint256 argOffset) internal pure returns (uint168 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(88, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint160. - function _getArgUint160(uint256 argOffset) internal pure returns (uint160 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(96, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint152. - function _getArgUint152(uint256 argOffset) internal pure returns (uint152 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(104, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint144. - function _getArgUint144(uint256 argOffset) internal pure returns (uint144 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(112, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint136. - function _getArgUint136(uint256 argOffset) internal pure returns (uint136 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(120, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint128. - function _getArgUint128(uint256 argOffset) internal pure returns (uint128 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(128, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint120. - function _getArgUint120(uint256 argOffset) internal pure returns (uint120 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(136, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint112. - function _getArgUint112(uint256 argOffset) internal pure returns (uint112 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(144, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint104. - function _getArgUint104(uint256 argOffset) internal pure returns (uint104 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(152, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint96. - function _getArgUint96(uint256 argOffset) internal pure returns (uint96 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(160, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint88. - function _getArgUint88(uint256 argOffset) internal pure returns (uint88 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(168, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint80. - function _getArgUint80(uint256 argOffset) internal pure returns (uint80 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(176, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint72. - function _getArgUint72(uint256 argOffset) internal pure returns (uint72 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(184, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint64. - function _getArgUint64(uint256 argOffset) internal pure returns (uint64 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(192, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint56. - function _getArgUint56(uint256 argOffset) internal pure returns (uint56 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(200, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint48. - function _getArgUint48(uint256 argOffset) internal pure returns (uint48 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(208, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint40. - function _getArgUint40(uint256 argOffset) internal pure returns (uint40 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(216, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint32. - function _getArgUint32(uint256 argOffset) internal pure returns (uint32 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(224, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint24. - function _getArgUint24(uint256 argOffset) internal pure returns (uint24 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(232, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint16. - function _getArgUint16(uint256 argOffset) internal pure returns (uint16 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(240, calldataload(add(offset, argOffset))) - } - } - - /// @dev Reads an immutable arg with type uint8. - function _getArgUint8(uint256 argOffset) internal pure returns (uint8 arg) { - uint256 offset = _getImmutableArgsOffset(); - /// @solidity memory-safe-assembly - assembly { - arg := shr(248, calldataload(add(offset, argOffset))) - } - } - - /// @return offset The offset of the packed immutable args in calldata. - function _getImmutableArgsOffset() internal pure returns (uint256 offset) { - /// @solidity memory-safe-assembly - assembly { - offset := sub(calldatasize(), shr(240, calldataload(sub(calldatasize(), 2)))) - } - } -} diff --git a/lib/solady/src/utils/DateTimeLib.sol b/lib/solady/src/utils/DateTimeLib.sol deleted file mode 100644 index 005ba76..0000000 --- a/lib/solady/src/utils/DateTimeLib.sol +++ /dev/null @@ -1,516 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for date time operations. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/DateTimeLib.sol) -/// -/// Conventions: -/// --------------------------------------------------------------------+ -/// Unit | Range | Notes | -/// --------------------------------------------------------------------| -/// timestamp | 0..0x1e18549868c76ff | Unix timestamp. | -/// epochDay | 0..0x16d3e098039 | Days since 1970-01-01. | -/// year | 1970..0xffffffff | Gregorian calendar year. | -/// month | 1..12 | Gregorian calendar month. | -/// day | 1..31 | Gregorian calendar day of month. | -/// weekday | 1..7 | The day of the week (1-indexed). | -/// --------------------------------------------------------------------+ -/// All timestamps of days are rounded down to 00:00:00 UTC. -library DateTimeLib { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // Weekdays are 1-indexed for a traditional rustic feel. - - // "And on the seventh day God finished his work that he had done, - // and he rested on the seventh day from all his work that he had done." - // -- Genesis 2:2 - - uint256 internal constant MON = 1; - uint256 internal constant TUE = 2; - uint256 internal constant WED = 3; - uint256 internal constant THU = 4; - uint256 internal constant FRI = 5; - uint256 internal constant SAT = 6; - uint256 internal constant SUN = 7; - - // Months and days of months are 1-indexed for ease of use. - - uint256 internal constant JAN = 1; - uint256 internal constant FEB = 2; - uint256 internal constant MAR = 3; - uint256 internal constant APR = 4; - uint256 internal constant MAY = 5; - uint256 internal constant JUN = 6; - uint256 internal constant JUL = 7; - uint256 internal constant AUG = 8; - uint256 internal constant SEP = 9; - uint256 internal constant OCT = 10; - uint256 internal constant NOV = 11; - uint256 internal constant DEC = 12; - - // These limits are large enough for most practical purposes. - // Inputs that exceed these limits result in undefined behavior. - - uint256 internal constant MAX_SUPPORTED_YEAR = 0xffffffff; - uint256 internal constant MAX_SUPPORTED_EPOCH_DAY = 0x16d3e098039; - uint256 internal constant MAX_SUPPORTED_TIMESTAMP = 0x1e18549868c76ff; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* DATE TIME OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the number of days since 1970-01-01 from (`year`,`month`,`day`). - /// See: https://howardhinnant.github.io/date_algorithms.html - /// Note: Inputs outside the supported ranges result in undefined behavior. - /// Use {isSupportedDate} to check if the inputs are supported. - function dateToEpochDay(uint256 year, uint256 month, uint256 day) - internal - pure - returns (uint256 epochDay) - { - /// @solidity memory-safe-assembly - assembly { - year := sub(year, lt(month, 3)) - let doy := add(shr(11, add(mul(62719, mod(add(month, 9), 12)), 769)), day) - let yoe := mod(year, 400) - let doe := sub(add(add(mul(yoe, 365), shr(2, yoe)), doy), div(yoe, 100)) - epochDay := sub(add(mul(div(year, 400), 146097), doe), 719469) - } - } - - /// @dev Returns (`year`,`month`,`day`) from the number of days since 1970-01-01. - /// Note: Inputs outside the supported ranges result in undefined behavior. - /// Use {isSupportedDays} to check if the inputs is supported. - function epochDayToDate(uint256 epochDay) - internal - pure - returns (uint256 year, uint256 month, uint256 day) - { - /// @solidity memory-safe-assembly - assembly { - epochDay := add(epochDay, 719468) - let doe := mod(epochDay, 146097) - let yoe := - div(sub(sub(add(doe, div(doe, 36524)), div(doe, 1460)), eq(doe, 146096)), 365) - let doy := sub(doe, sub(add(mul(365, yoe), shr(2, yoe)), div(yoe, 100))) - let mp := div(add(mul(5, doy), 2), 153) - day := add(sub(doy, shr(11, add(mul(mp, 62719), 769))), 1) - month := sub(add(mp, 3), mul(gt(mp, 9), 12)) - year := add(add(yoe, mul(div(epochDay, 146097), 400)), lt(month, 3)) - } - } - - /// @dev Returns the unix timestamp from (`year`,`month`,`day`). - /// Note: Inputs outside the supported ranges result in undefined behavior. - /// Use {isSupportedDate} to check if the inputs are supported. - function dateToTimestamp(uint256 year, uint256 month, uint256 day) - internal - pure - returns (uint256 result) - { - unchecked { - result = dateToEpochDay(year, month, day) * 86400; - } - } - - /// @dev Returns (`year`,`month`,`day`) from the given unix timestamp. - /// Note: Inputs outside the supported ranges result in undefined behavior. - /// Use {isSupportedTimestamp} to check if the inputs are supported. - function timestampToDate(uint256 timestamp) - internal - pure - returns (uint256 year, uint256 month, uint256 day) - { - (year, month, day) = epochDayToDate(timestamp / 86400); - } - - /// @dev Returns the unix timestamp from - /// (`year`,`month`,`day`,`hour`,`minute`,`second`). - /// Note: Inputs outside the supported ranges result in undefined behavior. - /// Use {isSupportedDateTime} to check if the inputs are supported. - function dateTimeToTimestamp( - uint256 year, - uint256 month, - uint256 day, - uint256 hour, - uint256 minute, - uint256 second - ) internal pure returns (uint256 result) { - unchecked { - result = dateToEpochDay(year, month, day) * 86400 + hour * 3600 + minute * 60 + second; - } - } - - /// @dev Returns (`year`,`month`,`day`,`hour`,`minute`,`second`) - /// from the given unix timestamp. - /// Note: Inputs outside the supported ranges result in undefined behavior. - /// Use {isSupportedTimestamp} to check if the inputs are supported. - function timestampToDateTime(uint256 timestamp) - internal - pure - returns ( - uint256 year, - uint256 month, - uint256 day, - uint256 hour, - uint256 minute, - uint256 second - ) - { - unchecked { - (year, month, day) = epochDayToDate(timestamp / 86400); - uint256 secs = timestamp % 86400; - hour = secs / 3600; - secs = secs % 3600; - minute = secs / 60; - second = secs % 60; - } - } - - /// @dev Returns if the `year` is leap. - function isLeapYear(uint256 year) internal pure returns (bool leap) { - /// @solidity memory-safe-assembly - assembly { - leap := iszero(and(add(mul(iszero(mod(year, 25)), 12), 3), year)) - } - } - - /// @dev Returns number of days in given `month` of `year`. - function daysInMonth(uint256 year, uint256 month) internal pure returns (uint256 result) { - bool flag = isLeapYear(year); - /// @solidity memory-safe-assembly - assembly { - // `daysInMonths = [31,28,31,30,31,30,31,31,30,31,30,31]`. - // `result = daysInMonths[month - 1] + isLeapYear(year)`. - result := - add(byte(month, shl(152, 0x1F1C1F1E1F1E1F1F1E1F1E1F)), and(eq(month, 2), flag)) - } - } - - /// @dev Returns the weekday from the unix timestamp. - /// Monday: 1, Tuesday: 2, ....., Sunday: 7. - function weekday(uint256 timestamp) internal pure returns (uint256 result) { - unchecked { - result = ((timestamp / 86400 + 3) % 7) + 1; - } - } - - /// @dev Returns if (`year`,`month`,`day`) is a supported date. - /// - `1970 <= year <= MAX_SUPPORTED_YEAR`. - /// - `1 <= month <= 12`. - /// - `1 <= day <= daysInMonth(year, month)`. - function isSupportedDate(uint256 year, uint256 month, uint256 day) - internal - pure - returns (bool result) - { - uint256 md = daysInMonth(year, month); - /// @solidity memory-safe-assembly - assembly { - let w := not(0) - result := - and( - lt(sub(year, 1970), sub(MAX_SUPPORTED_YEAR, 1969)), - and(lt(add(month, w), 12), lt(add(day, w), md)) - ) - } - } - - /// @dev Returns if (`year`,`month`,`day`,`hour`,`minute`,`second`) is a supported date time. - /// - `1970 <= year <= MAX_SUPPORTED_YEAR`. - /// - `1 <= month <= 12`. - /// - `1 <= day <= daysInMonth(year, month)`. - /// - `hour < 24`. - /// - `minute < 60`. - /// - `second < 60`. - function isSupportedDateTime( - uint256 year, - uint256 month, - uint256 day, - uint256 hour, - uint256 minute, - uint256 second - ) internal pure returns (bool result) { - if (isSupportedDate(year, month, day)) { - /// @solidity memory-safe-assembly - assembly { - result := and(lt(hour, 24), and(lt(minute, 60), lt(second, 60))) - } - } - } - - /// @dev Returns if `epochDay` is a supported unix epoch day. - function isSupportedEpochDay(uint256 epochDay) internal pure returns (bool result) { - unchecked { - result = epochDay < MAX_SUPPORTED_EPOCH_DAY + 1; - } - } - - /// @dev Returns if `timestamp` is a supported unix timestamp. - function isSupportedTimestamp(uint256 timestamp) internal pure returns (bool result) { - unchecked { - result = timestamp < MAX_SUPPORTED_TIMESTAMP + 1; - } - } - - /// @dev Returns the unix timestamp of the given `n`th weekday `wd`, in `month` of `year`. - /// Example: 3rd Friday of Feb 2022 is `nthWeekdayInMonthOfYearTimestamp(2022, 2, 3, 5)` - /// Note: `n` is 1-indexed for traditional consistency. - /// Invalid weekdays (i.e. `wd == 0 || wd > 7`) result in undefined behavior. - function nthWeekdayInMonthOfYearTimestamp(uint256 year, uint256 month, uint256 n, uint256 wd) - internal - pure - returns (uint256 result) - { - uint256 d = dateToEpochDay(year, month, 1); - uint256 md = daysInMonth(year, month); - /// @solidity memory-safe-assembly - assembly { - let diff := sub(wd, add(mod(add(d, 3), 7), 1)) - let date := add(mul(sub(n, 1), 7), add(mul(gt(diff, 6), 7), diff)) - result := mul(mul(86400, add(date, d)), and(lt(date, md), iszero(iszero(n)))) - } - } - - /// @dev Returns the unix timestamp of the most recent Monday. - function mondayTimestamp(uint256 timestamp) internal pure returns (uint256 result) { - uint256 t = timestamp; - /// @solidity memory-safe-assembly - assembly { - let day := div(t, 86400) - result := mul(mul(sub(day, mod(add(day, 3), 7)), 86400), gt(t, 345599)) - } - } - - /// @dev Returns whether the unix timestamp falls on a Saturday or Sunday. - /// To check whether it is a week day, just take the negation of the result. - function isWeekEnd(uint256 timestamp) internal pure returns (bool result) { - result = weekday(timestamp) > FRI; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* DATE TIME ARITHMETIC OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Adds `numYears` to the unix timestamp, and returns the result. - /// Note: The result will share the same Gregorian calendar month, - /// but different Gregorian calendar years for non-zero `numYears`. - /// If the Gregorian calendar month of the result has less days - /// than the Gregorian calendar month day of the `timestamp`, - /// the result's month day will be the maximum possible value for the month. - /// (e.g. from 29th Feb to 28th Feb) - function addYears(uint256 timestamp, uint256 numYears) internal pure returns (uint256 result) { - (uint256 year, uint256 month, uint256 day) = epochDayToDate(timestamp / 86400); - result = _offsetted(year + numYears, month, day, timestamp); - } - - /// @dev Adds `numMonths` to the unix timestamp, and returns the result. - /// Note: If the Gregorian calendar month of the result has less days - /// than the Gregorian calendar month day of the `timestamp`, - /// the result's month day will be the maximum possible value for the month. - /// (e.g. from 29th Feb to 28th Feb) - function addMonths(uint256 timestamp, uint256 numMonths) - internal - pure - returns (uint256 result) - { - (uint256 year, uint256 month, uint256 day) = epochDayToDate(timestamp / 86400); - month = _sub(month + numMonths, 1); - result = _offsetted(year + month / 12, _add(month % 12, 1), day, timestamp); - } - - /// @dev Adds `numDays` to the unix timestamp, and returns the result. - function addDays(uint256 timestamp, uint256 numDays) internal pure returns (uint256 result) { - result = timestamp + numDays * 86400; - } - - /// @dev Adds `numHours` to the unix timestamp, and returns the result. - function addHours(uint256 timestamp, uint256 numHours) internal pure returns (uint256 result) { - result = timestamp + numHours * 3600; - } - - /// @dev Adds `numMinutes` to the unix timestamp, and returns the result. - function addMinutes(uint256 timestamp, uint256 numMinutes) - internal - pure - returns (uint256 result) - { - result = timestamp + numMinutes * 60; - } - - /// @dev Adds `numSeconds` to the unix timestamp, and returns the result. - function addSeconds(uint256 timestamp, uint256 numSeconds) - internal - pure - returns (uint256 result) - { - result = timestamp + numSeconds; - } - - /// @dev Subtracts `numYears` from the unix timestamp, and returns the result. - /// Note: The result will share the same Gregorian calendar month, - /// but different Gregorian calendar years for non-zero `numYears`. - /// If the Gregorian calendar month of the result has less days - /// than the Gregorian calendar month day of the `timestamp`, - /// the result's month day will be the maximum possible value for the month. - /// (e.g. from 29th Feb to 28th Feb) - function subYears(uint256 timestamp, uint256 numYears) internal pure returns (uint256 result) { - (uint256 year, uint256 month, uint256 day) = epochDayToDate(timestamp / 86400); - result = _offsetted(year - numYears, month, day, timestamp); - } - - /// @dev Subtracts `numYears` from the unix timestamp, and returns the result. - /// Note: If the Gregorian calendar month of the result has less days - /// than the Gregorian calendar month day of the `timestamp`, - /// the result's month day will be the maximum possible value for the month. - /// (e.g. from 29th Feb to 28th Feb) - function subMonths(uint256 timestamp, uint256 numMonths) - internal - pure - returns (uint256 result) - { - (uint256 year, uint256 month, uint256 day) = epochDayToDate(timestamp / 86400); - uint256 yearMonth = _totalMonths(year, month) - _add(numMonths, 1); - result = _offsetted(yearMonth / 12, _add(yearMonth % 12, 1), day, timestamp); - } - - /// @dev Subtracts `numDays` from the unix timestamp, and returns the result. - function subDays(uint256 timestamp, uint256 numDays) internal pure returns (uint256 result) { - result = timestamp - numDays * 86400; - } - - /// @dev Subtracts `numHours` from the unix timestamp, and returns the result. - function subHours(uint256 timestamp, uint256 numHours) internal pure returns (uint256 result) { - result = timestamp - numHours * 3600; - } - - /// @dev Subtracts `numMinutes` from the unix timestamp, and returns the result. - function subMinutes(uint256 timestamp, uint256 numMinutes) - internal - pure - returns (uint256 result) - { - result = timestamp - numMinutes * 60; - } - - /// @dev Subtracts `numSeconds` from the unix timestamp, and returns the result. - function subSeconds(uint256 timestamp, uint256 numSeconds) - internal - pure - returns (uint256 result) - { - result = timestamp - numSeconds; - } - - /// @dev Returns the difference in Gregorian calendar years - /// between `fromTimestamp` and `toTimestamp`. - /// Note: Even if the true time difference is less than a year, - /// the difference can be non-zero is the timestamps are - /// from different Gregorian calendar years - function diffYears(uint256 fromTimestamp, uint256 toTimestamp) - internal - pure - returns (uint256 result) - { - toTimestamp - fromTimestamp; - (uint256 fromYear,,) = epochDayToDate(fromTimestamp / 86400); - (uint256 toYear,,) = epochDayToDate(toTimestamp / 86400); - result = _sub(toYear, fromYear); - } - - /// @dev Returns the difference in Gregorian calendar months - /// between `fromTimestamp` and `toTimestamp`. - /// Note: Even if the true time difference is less than a month, - /// the difference can be non-zero is the timestamps are - /// from different Gregorian calendar months. - function diffMonths(uint256 fromTimestamp, uint256 toTimestamp) - internal - pure - returns (uint256 result) - { - toTimestamp - fromTimestamp; - (uint256 fromYear, uint256 fromMonth,) = epochDayToDate(fromTimestamp / 86400); - (uint256 toYear, uint256 toMonth,) = epochDayToDate(toTimestamp / 86400); - result = _sub(_totalMonths(toYear, toMonth), _totalMonths(fromYear, fromMonth)); - } - - /// @dev Returns the difference in days between `fromTimestamp` and `toTimestamp`. - function diffDays(uint256 fromTimestamp, uint256 toTimestamp) - internal - pure - returns (uint256 result) - { - result = (toTimestamp - fromTimestamp) / 86400; - } - - /// @dev Returns the difference in hours between `fromTimestamp` and `toTimestamp`. - function diffHours(uint256 fromTimestamp, uint256 toTimestamp) - internal - pure - returns (uint256 result) - { - result = (toTimestamp - fromTimestamp) / 3600; - } - - /// @dev Returns the difference in minutes between `fromTimestamp` and `toTimestamp`. - function diffMinutes(uint256 fromTimestamp, uint256 toTimestamp) - internal - pure - returns (uint256 result) - { - result = (toTimestamp - fromTimestamp) / 60; - } - - /// @dev Returns the difference in seconds between `fromTimestamp` and `toTimestamp`. - function diffSeconds(uint256 fromTimestamp, uint256 toTimestamp) - internal - pure - returns (uint256 result) - { - result = toTimestamp - fromTimestamp; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PRIVATE HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Unchecked arithmetic for computing the total number of months. - function _totalMonths(uint256 numYears, uint256 numMonths) - private - pure - returns (uint256 total) - { - unchecked { - total = numYears * 12 + numMonths; - } - } - - /// @dev Unchecked arithmetic for adding two numbers. - function _add(uint256 a, uint256 b) private pure returns (uint256 c) { - unchecked { - c = a + b; - } - } - - /// @dev Unchecked arithmetic for subtracting two numbers. - function _sub(uint256 a, uint256 b) private pure returns (uint256 c) { - unchecked { - c = a - b; - } - } - - /// @dev Returns the offsetted timestamp. - function _offsetted(uint256 year, uint256 month, uint256 day, uint256 timestamp) - private - pure - returns (uint256 result) - { - uint256 dm = daysInMonth(year, month); - if (day >= dm) { - day = dm; - } - result = dateToEpochDay(year, month, day) * 86400 + (timestamp % 86400); - } -} diff --git a/lib/solady/src/utils/DynamicBufferLib.sol b/lib/solady/src/utils/DynamicBufferLib.sol deleted file mode 100644 index b8b94d2..0000000 --- a/lib/solady/src/utils/DynamicBufferLib.sol +++ /dev/null @@ -1,226 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for buffers with automatic capacity resizing. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/DynamicBuffer.sol) -/// @author Modified from cozyco (https://github.com/samkingco/cozyco/blob/main/contracts/utils/DynamicBuffer.sol) -library DynamicBufferLib { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STRUCTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Type to represent a dynamic buffer in memory. - /// You can directly assign to `data`, and the `p` function will - /// take care of the memory allocation. - struct DynamicBuffer { - bytes data; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Reserves at least `minimum` amount of contiguous memory. - function reserve(DynamicBuffer memory buffer, uint256 minimum) - internal - pure - returns (DynamicBuffer memory result) - { - _deallocate(result); - result = buffer; - uint256 n = buffer.data.length; - if (minimum > n) { - uint256 i = 0x40; - do {} while ((i <<= 1) < minimum); - bytes memory data; - /// @solidity memory-safe-assembly - assembly { - data := 0x00 - mstore(data, sub(i, n)) - } - result = p(result, data); - } - } - - /// @dev Clears the buffer without deallocating the memory. - function clear(DynamicBuffer memory buffer) - internal - pure - returns (DynamicBuffer memory result) - { - _deallocate(result); - /// @solidity memory-safe-assembly - assembly { - mstore(mload(buffer), 0) - } - result = buffer; - } - - /// @dev Returns a string pointing to the underlying bytes data. - /// Note: The string WILL change if the buffer is updated. - function s(DynamicBuffer memory buffer) internal pure returns (string memory) { - return string(buffer.data); - } - - /// @dev Appends `data` to `buffer`. - /// Returns the same buffer, so that it can be used for function chaining. - function p(DynamicBuffer memory buffer, bytes memory data) - internal - pure - returns (DynamicBuffer memory result) - { - _deallocate(result); - result = buffer; - if (data.length == 0) return result; - /// @solidity memory-safe-assembly - assembly { - let w := not(0x1f) - let bufData := mload(buffer) - let bufDataLen := mload(bufData) - let newBufDataLen := add(mload(data), bufDataLen) - // Some random prime number to multiply `cap`, so that - // we know that the `cap` is for a dynamic buffer. - // Selected to be larger than any memory pointer realistically. - let prime := 1621250193422201 - let cap := mload(add(bufData, w)) // `mload(sub(bufData, 0x20))`. - // Extract `cap`, initializing it to zero if it is not a multiple of `prime`. - cap := mul(div(cap, prime), iszero(mod(cap, prime))) - - // Expand / Reallocate memory if required. - // Note that we need to allocate an extra word for the length, and - // and another extra word as a safety word (giving a total of 0x40 bytes). - // Without the safety word, the backwards copying can cause a buffer overflow. - for {} iszero(lt(newBufDataLen, cap)) {} { - // Approximately more than double the capacity to ensure more than enough space. - let newCap := and(add(cap, add(or(cap, newBufDataLen), 0x20)), w) - // If the memory is contiguous, we can simply expand it. - if iszero(or(xor(mload(0x40), add(bufData, add(0x40, cap))), eq(bufData, 0x60))) { - // Store `cap * prime` in the word before the length. - mstore(add(bufData, w), mul(prime, newCap)) - mstore(0x40, add(bufData, add(0x40, newCap))) // Expand the memory allocation. - break - } - // Set the `newBufData` to point to the word after `cap`. - let newBufData := add(mload(0x40), 0x20) - mstore(0x40, add(newBufData, add(0x40, newCap))) // Reallocate the memory. - mstore(buffer, newBufData) // Store the `newBufData`. - // Copy `bufData` one word at a time, backwards. - for { let o := and(add(bufDataLen, 0x20), w) } 1 {} { - mstore(add(newBufData, o), mload(add(bufData, o))) - o := add(o, w) // `sub(o, 0x20)`. - if iszero(o) { break } - } - // Store `cap * prime` in the word before the length. - mstore(add(newBufData, w), mul(prime, newCap)) - bufData := newBufData // Assign `newBufData` to `bufData`. - break - } - // If it's a reserve operation, set the variables to skip the appending. - if iszero(data) { - mstore(data, data) - newBufDataLen := bufDataLen - } - // Copy `data` one word at a time, backwards. - for { let o := and(add(mload(data), 0x20), w) } 1 {} { - mstore(add(add(bufData, bufDataLen), o), mload(add(data, o))) - o := add(o, w) // `sub(o, 0x20)`. - if iszero(o) { break } - } - mstore(add(add(bufData, 0x20), newBufDataLen), 0) // Zeroize the word after the buffer. - mstore(bufData, newBufDataLen) // Store the length. - } - } - - /// @dev Appends `data0`, `data1` to `buffer`. - /// Returns the same buffer, so that it can be used for function chaining. - function p(DynamicBuffer memory buffer, bytes memory data0, bytes memory data1) - internal - pure - returns (DynamicBuffer memory result) - { - _deallocate(result); - result = p(p(buffer, data0), data1); - } - - /// @dev Appends `data0` .. `data2` to `buffer`. - /// Returns the same buffer, so that it can be used for function chaining. - function p( - DynamicBuffer memory buffer, - bytes memory data0, - bytes memory data1, - bytes memory data2 - ) internal pure returns (DynamicBuffer memory result) { - _deallocate(result); - result = p(p(p(buffer, data0), data1), data2); - } - - /// @dev Appends `data0` .. `data3` to `buffer`. - /// Returns the same buffer, so that it can be used for function chaining. - function p( - DynamicBuffer memory buffer, - bytes memory data0, - bytes memory data1, - bytes memory data2, - bytes memory data3 - ) internal pure returns (DynamicBuffer memory result) { - _deallocate(result); - result = p(p(p(p(buffer, data0), data1), data2), data3); - } - - /// @dev Appends `data0` .. `data4` to `buffer`. - /// Returns the same buffer, so that it can be used for function chaining. - function p( - DynamicBuffer memory buffer, - bytes memory data0, - bytes memory data1, - bytes memory data2, - bytes memory data3, - bytes memory data4 - ) internal pure returns (DynamicBuffer memory result) { - _deallocate(result); - result = p(p(p(p(p(buffer, data0), data1), data2), data3), data4); - } - - /// @dev Appends `data0` .. `data5` to `buffer`. - /// Returns the same buffer, so that it can be used for function chaining. - function p( - DynamicBuffer memory buffer, - bytes memory data0, - bytes memory data1, - bytes memory data2, - bytes memory data3, - bytes memory data4, - bytes memory data5 - ) internal pure returns (DynamicBuffer memory result) { - _deallocate(result); - result = p(p(p(p(p(p(buffer, data0), data1), data2), data3), data4), data5); - } - - /// @dev Appends `data0` .. `data6` to `buffer`. - /// Returns the same buffer, so that it can be used for function chaining. - function p( - DynamicBuffer memory buffer, - bytes memory data0, - bytes memory data1, - bytes memory data2, - bytes memory data3, - bytes memory data4, - bytes memory data5, - bytes memory data6 - ) internal pure returns (DynamicBuffer memory result) { - _deallocate(result); - result = p(p(p(p(p(p(p(buffer, data0), data1), data2), data3), data4), data5), data6); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PRIVATE HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Helper for deallocating a automatically allocated `buffer` pointer. - function _deallocate(DynamicBuffer memory result) private pure { - /// @solidity memory-safe-assembly - assembly { - mstore(0x40, result) // Deallocate, as we have already allocated. - } - } -} diff --git a/lib/solady/src/utils/ECDSA.sol b/lib/solady/src/utils/ECDSA.sol deleted file mode 100644 index 9462498..0000000 --- a/lib/solady/src/utils/ECDSA.sol +++ /dev/null @@ -1,411 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Gas optimized ECDSA wrapper. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/ECDSA.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/ECDSA.sol) -/// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/cryptography/ECDSA.sol) -/// -/// @dev Note: -/// - The recovery functions use the ecrecover precompile (0x1). -/// - As of Solady version 0.0.68, the `recover` variants will revert upon recovery failure. -/// This is for more safety by default. -/// Use the `tryRecover` variants if you need to get the zero address back -/// upon recovery failure instead. -/// - As of Solady version 0.0.134, all `bytes signature` variants accept both -/// regular 65-byte `(r, s, v)` and EIP-2098 `(r, vs)` short form signatures. -/// See: https://eips.ethereum.org/EIPS/eip-2098 -/// This is for calldata efficiency on smart accounts prevalent on L2s. -/// -/// WARNING! Do NOT use signatures as unique identifiers: -/// - Use a nonce in the digest to prevent replay attacks on the same contract. -/// - Use EIP-712 for the digest to prevent replay attacks across different chains and contracts. -/// EIP-712 also enables readable signing of typed data for better user safety. -/// This implementation does NOT check if a signature is non-malleable. -library ECDSA { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The signature is invalid. - error InvalidSignature(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* RECOVERY OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Recovers the signer's address from a message digest `hash`, and the `signature`. - function recover(bytes32 hash, bytes memory signature) internal view returns (address result) { - /// @solidity memory-safe-assembly - assembly { - result := 1 - let m := mload(0x40) // Cache the free memory pointer. - for {} 1 {} { - mstore(0x00, hash) - mstore(0x40, mload(add(signature, 0x20))) // `r`. - if eq(mload(signature), 64) { - let vs := mload(add(signature, 0x40)) - mstore(0x20, add(shr(255, vs), 27)) // `v`. - mstore(0x60, shr(1, shl(1, vs))) // `s`. - break - } - if eq(mload(signature), 65) { - mstore(0x20, byte(0, mload(add(signature, 0x60)))) // `v`. - mstore(0x60, mload(add(signature, 0x40))) // `s`. - break - } - result := 0 - break - } - result := - mload( - staticcall( - gas(), // Amount of gas left for the transaction. - result, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x01, // Start of output. - 0x20 // Size of output. - ) - ) - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - if iszero(returndatasize()) { - mstore(0x00, 0x8baa579f) // `InvalidSignature()`. - revert(0x1c, 0x04) - } - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - } - } - - /// @dev Recovers the signer's address from a message digest `hash`, and the `signature`. - function recoverCalldata(bytes32 hash, bytes calldata signature) - internal - view - returns (address result) - { - /// @solidity memory-safe-assembly - assembly { - result := 1 - let m := mload(0x40) // Cache the free memory pointer. - mstore(0x00, hash) - for {} 1 {} { - if eq(signature.length, 64) { - let vs := calldataload(add(signature.offset, 0x20)) - mstore(0x20, add(shr(255, vs), 27)) // `v`. - mstore(0x40, calldataload(signature.offset)) // `r`. - mstore(0x60, shr(1, shl(1, vs))) // `s`. - break - } - if eq(signature.length, 65) { - mstore(0x20, byte(0, calldataload(add(signature.offset, 0x40)))) // `v`. - calldatacopy(0x40, signature.offset, 0x40) // Copy `r` and `s`. - break - } - result := 0 - break - } - result := - mload( - staticcall( - gas(), // Amount of gas left for the transaction. - result, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x01, // Start of output. - 0x20 // Size of output. - ) - ) - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - if iszero(returndatasize()) { - mstore(0x00, 0x8baa579f) // `InvalidSignature()`. - revert(0x1c, 0x04) - } - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - } - } - - /// @dev Recovers the signer's address from a message digest `hash`, - /// and the EIP-2098 short form signature defined by `r` and `vs`. - function recover(bytes32 hash, bytes32 r, bytes32 vs) internal view returns (address result) { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Cache the free memory pointer. - mstore(0x00, hash) - mstore(0x20, add(shr(255, vs), 27)) // `v`. - mstore(0x40, r) - mstore(0x60, shr(1, shl(1, vs))) // `s`. - result := - mload( - staticcall( - gas(), // Amount of gas left for the transaction. - 1, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x01, // Start of output. - 0x20 // Size of output. - ) - ) - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - if iszero(returndatasize()) { - mstore(0x00, 0x8baa579f) // `InvalidSignature()`. - revert(0x1c, 0x04) - } - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - } - } - - /// @dev Recovers the signer's address from a message digest `hash`, - /// and the signature defined by `v`, `r`, `s`. - function recover(bytes32 hash, uint8 v, bytes32 r, bytes32 s) - internal - view - returns (address result) - { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Cache the free memory pointer. - mstore(0x00, hash) - mstore(0x20, and(v, 0xff)) - mstore(0x40, r) - mstore(0x60, s) - result := - mload( - staticcall( - gas(), // Amount of gas left for the transaction. - 1, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x01, // Start of output. - 0x20 // Size of output. - ) - ) - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - if iszero(returndatasize()) { - mstore(0x00, 0x8baa579f) // `InvalidSignature()`. - revert(0x1c, 0x04) - } - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* TRY-RECOVER OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // WARNING! - // These functions will NOT revert upon recovery failure. - // Instead, they will return the zero address upon recovery failure. - // It is critical that the returned address is NEVER compared against - // a zero address (e.g. an uninitialized address variable). - - /// @dev Recovers the signer's address from a message digest `hash`, and the `signature`. - function tryRecover(bytes32 hash, bytes memory signature) - internal - view - returns (address result) - { - /// @solidity memory-safe-assembly - assembly { - result := 1 - let m := mload(0x40) // Cache the free memory pointer. - for {} 1 {} { - mstore(0x00, hash) - mstore(0x40, mload(add(signature, 0x20))) // `r`. - if eq(mload(signature), 64) { - let vs := mload(add(signature, 0x40)) - mstore(0x20, add(shr(255, vs), 27)) // `v`. - mstore(0x60, shr(1, shl(1, vs))) // `s`. - break - } - if eq(mload(signature), 65) { - mstore(0x20, byte(0, mload(add(signature, 0x60)))) // `v`. - mstore(0x60, mload(add(signature, 0x40))) // `s`. - break - } - result := 0 - break - } - pop( - staticcall( - gas(), // Amount of gas left for the transaction. - result, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x40, // Start of output. - 0x20 // Size of output. - ) - ) - mstore(0x60, 0) // Restore the zero slot. - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - result := mload(xor(0x60, returndatasize())) - mstore(0x40, m) // Restore the free memory pointer. - } - } - - /// @dev Recovers the signer's address from a message digest `hash`, and the `signature`. - function tryRecoverCalldata(bytes32 hash, bytes calldata signature) - internal - view - returns (address result) - { - /// @solidity memory-safe-assembly - assembly { - result := 1 - let m := mload(0x40) // Cache the free memory pointer. - mstore(0x00, hash) - for {} 1 {} { - if eq(signature.length, 64) { - let vs := calldataload(add(signature.offset, 0x20)) - mstore(0x20, add(shr(255, vs), 27)) // `v`. - mstore(0x40, calldataload(signature.offset)) // `r`. - mstore(0x60, shr(1, shl(1, vs))) // `s`. - break - } - if eq(signature.length, 65) { - mstore(0x20, byte(0, calldataload(add(signature.offset, 0x40)))) // `v`. - calldatacopy(0x40, signature.offset, 0x40) // Copy `r` and `s`. - break - } - result := 0 - break - } - pop( - staticcall( - gas(), // Amount of gas left for the transaction. - result, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x40, // Start of output. - 0x20 // Size of output. - ) - ) - mstore(0x60, 0) // Restore the zero slot. - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - result := mload(xor(0x60, returndatasize())) - mstore(0x40, m) // Restore the free memory pointer. - } - } - - /// @dev Recovers the signer's address from a message digest `hash`, - /// and the EIP-2098 short form signature defined by `r` and `vs`. - function tryRecover(bytes32 hash, bytes32 r, bytes32 vs) - internal - view - returns (address result) - { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Cache the free memory pointer. - mstore(0x00, hash) - mstore(0x20, add(shr(255, vs), 27)) // `v`. - mstore(0x40, r) - mstore(0x60, shr(1, shl(1, vs))) // `s`. - pop( - staticcall( - gas(), // Amount of gas left for the transaction. - 1, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x40, // Start of output. - 0x20 // Size of output. - ) - ) - mstore(0x60, 0) // Restore the zero slot. - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - result := mload(xor(0x60, returndatasize())) - mstore(0x40, m) // Restore the free memory pointer. - } - } - - /// @dev Recovers the signer's address from a message digest `hash`, - /// and the signature defined by `v`, `r`, `s`. - function tryRecover(bytes32 hash, uint8 v, bytes32 r, bytes32 s) - internal - view - returns (address result) - { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Cache the free memory pointer. - mstore(0x00, hash) - mstore(0x20, and(v, 0xff)) - mstore(0x40, r) - mstore(0x60, s) - pop( - staticcall( - gas(), // Amount of gas left for the transaction. - 1, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x40, // Start of output. - 0x20 // Size of output. - ) - ) - mstore(0x60, 0) // Restore the zero slot. - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - result := mload(xor(0x60, returndatasize())) - mstore(0x40, m) // Restore the free memory pointer. - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* HASHING OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns an Ethereum Signed Message, created from a `hash`. - /// This produces a hash corresponding to the one signed with the - /// [`eth_sign`](https://eth.wiki/json-rpc/API#eth_sign) - /// JSON-RPC method as part of EIP-191. - function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, hash) // Store into scratch space for keccak256. - mstore(0x00, "\x00\x00\x00\x00\x19Ethereum Signed Message:\n32") // 28 bytes. - result := keccak256(0x04, 0x3c) // `32 * 2 - (32 - 28) = 60 = 0x3c`. - } - } - - /// @dev Returns an Ethereum Signed Message, created from `s`. - /// This produces a hash corresponding to the one signed with the - /// [`eth_sign`](https://eth.wiki/json-rpc/API#eth_sign) - /// JSON-RPC method as part of EIP-191. - /// Note: Supports lengths of `s` up to 999999 bytes. - function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - let sLength := mload(s) - let o := 0x20 - mstore(o, "\x19Ethereum Signed Message:\n") // 26 bytes, zero-right-padded. - mstore(0x00, 0x00) - // Convert the `s.length` to ASCII decimal representation: `base10(s.length)`. - for { let temp := sLength } 1 {} { - o := sub(o, 1) - mstore8(o, add(48, mod(temp, 10))) - temp := div(temp, 10) - if iszero(temp) { break } - } - let n := sub(0x3a, o) // Header length: `26 + 32 - o`. - // Throw an out-of-offset error (consumes all gas) if the header exceeds 32 bytes. - returndatacopy(returndatasize(), returndatasize(), gt(n, 0x20)) - mstore(s, or(mload(0x00), mload(n))) // Temporarily store the header. - result := keccak256(add(s, sub(0x20, n)), add(n, sLength)) - mstore(s, sLength) // Restore the length. - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EMPTY CALLDATA HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns an empty calldata bytes. - function emptySignature() internal pure returns (bytes calldata signature) { - /// @solidity memory-safe-assembly - assembly { - signature.length := 0 - } - } -} diff --git a/lib/solady/src/utils/EIP712.sol b/lib/solady/src/utils/EIP712.sol deleted file mode 100644 index f77ee14..0000000 --- a/lib/solady/src/utils/EIP712.sol +++ /dev/null @@ -1,208 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Contract for EIP-712 typed structured data hashing and signing. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/EIP712.sol) -/// @author Modified from Solbase (https://github.com/Sol-DAO/solbase/blob/main/src/utils/EIP712.sol) -/// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/cryptography/EIP712.sol) -/// -/// @dev Note, this implementation: -/// - Uses `address(this)` for the `verifyingContract` field. -/// - Does NOT use the optional EIP-712 salt. -/// - Does NOT use any EIP-712 extensions. -/// This is for simplicity and to save gas. -/// If you need to customize, please fork / modify accordingly. -abstract contract EIP712 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTANTS AND IMMUTABLES */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev `keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)")`. - bytes32 internal constant _DOMAIN_TYPEHASH = - 0x8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f; - - uint256 private immutable _cachedThis; - uint256 private immutable _cachedChainId; - bytes32 private immutable _cachedNameHash; - bytes32 private immutable _cachedVersionHash; - bytes32 private immutable _cachedDomainSeparator; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTRUCTOR */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Cache the hashes for cheaper runtime gas costs. - /// In the case of upgradeable contracts (i.e. proxies), - /// or if the chain id changes due to a hard fork, - /// the domain separator will be seamlessly calculated on-the-fly. - constructor() { - _cachedThis = uint256(uint160(address(this))); - _cachedChainId = block.chainid; - - string memory name; - string memory version; - if (!_domainNameAndVersionMayChange()) (name, version) = _domainNameAndVersion(); - bytes32 nameHash = _domainNameAndVersionMayChange() ? bytes32(0) : keccak256(bytes(name)); - bytes32 versionHash = - _domainNameAndVersionMayChange() ? bytes32(0) : keccak256(bytes(version)); - _cachedNameHash = nameHash; - _cachedVersionHash = versionHash; - - bytes32 separator; - if (!_domainNameAndVersionMayChange()) { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Load the free memory pointer. - mstore(m, _DOMAIN_TYPEHASH) - mstore(add(m, 0x20), nameHash) - mstore(add(m, 0x40), versionHash) - mstore(add(m, 0x60), chainid()) - mstore(add(m, 0x80), address()) - separator := keccak256(m, 0xa0) - } - } - _cachedDomainSeparator = separator; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* FUNCTIONS TO OVERRIDE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Please override this function to return the domain name and version. - /// ``` - /// function _domainNameAndVersion() - /// internal - /// pure - /// virtual - /// returns (string memory name, string memory version) - /// { - /// name = "Solady"; - /// version = "1"; - /// } - /// ``` - /// - /// Note: If the returned result may change after the contract has been deployed, - /// you must override `_domainNameAndVersionMayChange()` to return true. - function _domainNameAndVersion() - internal - view - virtual - returns (string memory name, string memory version); - - /// @dev Returns if `_domainNameAndVersion()` may change - /// after the contract has been deployed (i.e. after the constructor). - /// Default: false. - function _domainNameAndVersionMayChange() internal pure virtual returns (bool result) {} - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* HASHING OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the EIP-712 domain separator. - function _domainSeparator() internal view virtual returns (bytes32 separator) { - if (_domainNameAndVersionMayChange()) { - separator = _buildDomainSeparator(); - } else { - separator = _cachedDomainSeparator; - if (_cachedDomainSeparatorInvalidated()) separator = _buildDomainSeparator(); - } - } - - /// @dev Returns the hash of the fully encoded EIP-712 message for this domain, - /// given `structHash`, as defined in - /// https://eips.ethereum.org/EIPS/eip-712#definition-of-hashstruct. - /// - /// The hash can be used together with {ECDSA-recover} to obtain the signer of a message: - /// ``` - /// bytes32 digest = _hashTypedData(keccak256(abi.encode( - /// keccak256("Mail(address to,string contents)"), - /// mailTo, - /// keccak256(bytes(mailContents)) - /// ))); - /// address signer = ECDSA.recover(digest, signature); - /// ``` - function _hashTypedData(bytes32 structHash) internal view virtual returns (bytes32 digest) { - // We will use `digest` to store the domain separator to save a bit of gas. - if (_domainNameAndVersionMayChange()) { - digest = _buildDomainSeparator(); - } else { - digest = _cachedDomainSeparator; - if (_cachedDomainSeparatorInvalidated()) digest = _buildDomainSeparator(); - } - /// @solidity memory-safe-assembly - assembly { - // Compute the digest. - mstore(0x00, 0x1901000000000000) // Store "\x19\x01". - mstore(0x1a, digest) // Store the domain separator. - mstore(0x3a, structHash) // Store the struct hash. - digest := keccak256(0x18, 0x42) - // Restore the part of the free memory slot that was overwritten. - mstore(0x3a, 0) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EIP-5267 OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev See: https://eips.ethereum.org/EIPS/eip-5267 - function eip712Domain() - public - view - virtual - returns ( - bytes1 fields, - string memory name, - string memory version, - uint256 chainId, - address verifyingContract, - bytes32 salt, - uint256[] memory extensions - ) - { - fields = hex"0f"; // `0b01111`. - (name, version) = _domainNameAndVersion(); - chainId = block.chainid; - verifyingContract = address(this); - salt = salt; // `bytes32(0)`. - extensions = extensions; // `new uint256[](0)`. - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PRIVATE HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the EIP-712 domain separator. - function _buildDomainSeparator() private view returns (bytes32 separator) { - // We will use `separator` to store the name hash to save a bit of gas. - bytes32 versionHash; - if (_domainNameAndVersionMayChange()) { - (string memory name, string memory version) = _domainNameAndVersion(); - separator = keccak256(bytes(name)); - versionHash = keccak256(bytes(version)); - } else { - separator = _cachedNameHash; - versionHash = _cachedVersionHash; - } - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Load the free memory pointer. - mstore(m, _DOMAIN_TYPEHASH) - mstore(add(m, 0x20), separator) // Name hash. - mstore(add(m, 0x40), versionHash) - mstore(add(m, 0x60), chainid()) - mstore(add(m, 0x80), address()) - separator := keccak256(m, 0xa0) - } - } - - /// @dev Returns if the cached domain separator has been invalidated. - function _cachedDomainSeparatorInvalidated() private view returns (bool result) { - uint256 cachedChainId = _cachedChainId; - uint256 cachedThis = _cachedThis; - /// @solidity memory-safe-assembly - assembly { - result := iszero(and(eq(chainid(), cachedChainId), eq(address(), cachedThis))) - } - } -} diff --git a/lib/solady/src/utils/ERC1967Factory.sol b/lib/solady/src/utils/ERC1967Factory.sol deleted file mode 100644 index 5d13141..0000000 --- a/lib/solady/src/utils/ERC1967Factory.sol +++ /dev/null @@ -1,419 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Factory for deploying and managing ERC1967 proxy contracts. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/ERC1967Factory.sol) -/// @author jtriley-eth (https://github.com/jtriley-eth/minimum-viable-proxy) -contract ERC1967Factory { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The caller is not authorized to call the function. - error Unauthorized(); - - /// @dev The proxy deployment failed. - error DeploymentFailed(); - - /// @dev The upgrade failed. - error UpgradeFailed(); - - /// @dev The salt does not start with the caller. - error SaltDoesNotStartWithCaller(); - - /// @dev `bytes4(keccak256(bytes("Unauthorized()")))`. - uint256 internal constant _UNAUTHORIZED_ERROR_SELECTOR = 0x82b42900; - - /// @dev `bytes4(keccak256(bytes("DeploymentFailed()")))`. - uint256 internal constant _DEPLOYMENT_FAILED_ERROR_SELECTOR = 0x30116425; - - /// @dev `bytes4(keccak256(bytes("UpgradeFailed()")))`. - uint256 internal constant _UPGRADE_FAILED_ERROR_SELECTOR = 0x55299b49; - - /// @dev `bytes4(keccak256(bytes("SaltDoesNotStartWithCaller()")))`. - uint256 internal constant _SALT_DOES_NOT_START_WITH_CALLER_ERROR_SELECTOR = 0x2f634836; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EVENTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The admin of a proxy contract has been changed. - event AdminChanged(address indexed proxy, address indexed admin); - - /// @dev The implementation for a proxy has been upgraded. - event Upgraded(address indexed proxy, address indexed implementation); - - /// @dev A proxy has been deployed. - event Deployed(address indexed proxy, address indexed implementation, address indexed admin); - - /// @dev `keccak256(bytes("AdminChanged(address,address)"))`. - uint256 internal constant _ADMIN_CHANGED_EVENT_SIGNATURE = - 0x7e644d79422f17c01e4894b5f4f588d331ebfa28653d42ae832dc59e38c9798f; - - /// @dev `keccak256(bytes("Upgraded(address,address)"))`. - uint256 internal constant _UPGRADED_EVENT_SIGNATURE = - 0x5d611f318680d00598bb735d61bacf0c514c6b50e1e5ad30040a4df2b12791c7; - - /// @dev `keccak256(bytes("Deployed(address,address,address)"))`. - uint256 internal constant _DEPLOYED_EVENT_SIGNATURE = - 0xc95935a66d15e0da5e412aca0ad27ae891d20b2fb91cf3994b6a3bf2b8178082; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STORAGE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // The admin slot for a `proxy` is `shl(96, proxy)`. - - /// @dev The ERC-1967 storage slot for the implementation in the proxy. - /// `uint256(keccak256("eip1967.proxy.implementation")) - 1`. - uint256 internal constant _IMPLEMENTATION_SLOT = - 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ADMIN FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the admin of the proxy. - function adminOf(address proxy) public view returns (address admin) { - assembly { - admin := sload(shl(96, proxy)) - } - } - - /// @dev Sets the admin of the proxy. - /// The caller of this function must be the admin of the proxy on this factory. - function changeAdmin(address proxy, address admin) public { - assembly { - // Check if the caller is the admin of the proxy. - if iszero(eq(sload(shl(96, proxy)), caller())) { - mstore(0x00, _UNAUTHORIZED_ERROR_SELECTOR) - revert(0x1c, 0x04) - } - // Store the admin for the proxy. - sstore(shl(96, proxy), admin) - // Emit the {AdminChanged} event. - log3(0, 0, _ADMIN_CHANGED_EVENT_SIGNATURE, proxy, admin) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* UPGRADE FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Upgrades the proxy to point to `implementation`. - /// The caller of this function must be the admin of the proxy on this factory. - function upgrade(address proxy, address implementation) public payable { - upgradeAndCall(proxy, implementation, _emptyData()); - } - - /// @dev Upgrades the proxy to point to `implementation`. - /// Then, calls the proxy with abi encoded `data`. - /// The caller of this function must be the admin of the proxy on this factory. - function upgradeAndCall(address proxy, address implementation, bytes calldata data) - public - payable - { - assembly { - // Check if the caller is the admin of the proxy. - if iszero(eq(sload(shl(96, proxy)), caller())) { - mstore(0x00, _UNAUTHORIZED_ERROR_SELECTOR) - revert(0x1c, 0x04) - } - // Set up the calldata to upgrade the proxy. - let m := mload(0x40) - mstore(m, implementation) - mstore(add(m, 0x20), _IMPLEMENTATION_SLOT) - calldatacopy(add(m, 0x40), data.offset, data.length) - // Try upgrading the proxy and revert upon failure. - if iszero(call(gas(), proxy, callvalue(), m, add(0x40, data.length), 0x00, 0x00)) { - // Revert with the `UpgradeFailed` selector if there is no error returndata. - if iszero(returndatasize()) { - mstore(0x00, _UPGRADE_FAILED_ERROR_SELECTOR) - revert(0x1c, 0x04) - } - // Otherwise, bubble up the returned error. - returndatacopy(0x00, 0x00, returndatasize()) - revert(0x00, returndatasize()) - } - // Emit the {Upgraded} event. - log3(0, 0, _UPGRADED_EVENT_SIGNATURE, proxy, implementation) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* DEPLOY FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Deploys a proxy for `implementation`, with `admin`, - /// and returns its address. - /// The value passed into this function will be forwarded to the proxy. - function deploy(address implementation, address admin) public payable returns (address proxy) { - proxy = deployAndCall(implementation, admin, _emptyData()); - } - - /// @dev Deploys a proxy for `implementation`, with `admin`, - /// and returns its address. - /// The value passed into this function will be forwarded to the proxy. - /// Then, calls the proxy with abi encoded `data`. - function deployAndCall(address implementation, address admin, bytes calldata data) - public - payable - returns (address proxy) - { - proxy = _deploy(implementation, admin, bytes32(0), false, data); - } - - /// @dev Deploys a proxy for `implementation`, with `admin`, `salt`, - /// and returns its deterministic address. - /// The value passed into this function will be forwarded to the proxy. - function deployDeterministic(address implementation, address admin, bytes32 salt) - public - payable - returns (address proxy) - { - proxy = deployDeterministicAndCall(implementation, admin, salt, _emptyData()); - } - - /// @dev Deploys a proxy for `implementation`, with `admin`, `salt`, - /// and returns its deterministic address. - /// The value passed into this function will be forwarded to the proxy. - /// Then, calls the proxy with abi encoded `data`. - function deployDeterministicAndCall( - address implementation, - address admin, - bytes32 salt, - bytes calldata data - ) public payable returns (address proxy) { - assembly { - // If the salt does not start with the zero address or the caller. - if iszero(or(iszero(shr(96, salt)), eq(caller(), shr(96, salt)))) { - mstore(0x00, _SALT_DOES_NOT_START_WITH_CALLER_ERROR_SELECTOR) - revert(0x1c, 0x04) - } - } - proxy = _deploy(implementation, admin, salt, true, data); - } - - /// @dev Deploys the proxy, with optionality to deploy deterministically with a `salt`. - function _deploy( - address implementation, - address admin, - bytes32 salt, - bool useSalt, - bytes calldata data - ) internal returns (address proxy) { - bytes32 m = _initCode(); - assembly { - // Create the proxy. - switch useSalt - case 0 { proxy := create(0, add(m, 0x13), 0x88) } - default { proxy := create2(0, add(m, 0x13), 0x88, salt) } - // Revert if the creation fails. - if iszero(proxy) { - mstore(0x00, _DEPLOYMENT_FAILED_ERROR_SELECTOR) - revert(0x1c, 0x04) - } - - // Set up the calldata to set the implementation of the proxy. - mstore(m, implementation) - mstore(add(m, 0x20), _IMPLEMENTATION_SLOT) - calldatacopy(add(m, 0x40), data.offset, data.length) - // Try setting the implementation on the proxy and revert upon failure. - if iszero(call(gas(), proxy, callvalue(), m, add(0x40, data.length), 0x00, 0x00)) { - // Revert with the `DeploymentFailed` selector if there is no error returndata. - if iszero(returndatasize()) { - mstore(0x00, _DEPLOYMENT_FAILED_ERROR_SELECTOR) - revert(0x1c, 0x04) - } - // Otherwise, bubble up the returned error. - returndatacopy(0x00, 0x00, returndatasize()) - revert(0x00, returndatasize()) - } - - // Store the admin for the proxy. - sstore(shl(96, proxy), admin) - - // Emit the {Deployed} event. - log4(0, 0, _DEPLOYED_EVENT_SIGNATURE, proxy, implementation, admin) - } - } - - /// @dev Returns the address of the proxy deployed with `salt`. - function predictDeterministicAddress(bytes32 salt) public view returns (address predicted) { - bytes32 hash = initCodeHash(); - assembly { - // Compute and store the bytecode hash. - mstore8(0x00, 0xff) // Write the prefix. - mstore(0x35, hash) - mstore(0x01, shl(96, address())) - mstore(0x15, salt) - // Note: `predicted` has dirty upper 96 bits. We won't clean it here - // as it will be automatically cleaned when it is copied into the returndata. - // Please clean as needed if used in other inline assembly blocks. - predicted := keccak256(0x00, 0x55) - // Restore the part of the free memory pointer that has been overwritten. - mstore(0x35, 0) - } - } - - /// @dev Returns the initialization code hash of the proxy. - /// Used for mining vanity addresses with create2crunch. - function initCodeHash() public view returns (bytes32 result) { - bytes32 m = _initCode(); - assembly { - result := keccak256(add(m, 0x13), 0x88) - } - } - - /// @dev Returns a pointer to the initialization code of a proxy created via this factory. - function _initCode() internal view returns (bytes32 m) { - assembly { - /** - * -------------------------------------------------------------------------------------+ - * CREATION (9 bytes) | - * -------------------------------------------------------------------------------------| - * Opcode | Mnemonic | Stack | Memory | - * -------------------------------------------------------------------------------------| - * 60 runSize | PUSH1 runSize | r | | - * 3d | RETURNDATASIZE | 0 r | | - * 81 | DUP2 | r 0 r | | - * 60 offset | PUSH1 offset | o r 0 r | | - * 3d | RETURNDATASIZE | 0 o r 0 r | | - * 39 | CODECOPY | 0 r | [0..runSize): runtime code | - * f3 | RETURN | | [0..runSize): runtime code | - * -------------------------------------------------------------------------------------| - * RUNTIME (127 bytes) | - * -------------------------------------------------------------------------------------| - * Opcode | Mnemonic | Stack | Memory | - * -------------------------------------------------------------------------------------| - * | - * ::: keep some values in stack :::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | 0 | | - * 3d | RETURNDATASIZE | 0 0 | | - * | - * ::: check if caller is factory ::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 33 | CALLER | c 0 0 | | - * 73 factory | PUSH20 factory | f c 0 0 | | - * 14 | EQ | isf 0 0 | | - * 60 0x57 | PUSH1 0x57 | dest isf 0 0 | | - * 57 | JUMPI | 0 0 | | - * | - * ::: copy calldata to memory :::::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 36 | CALLDATASIZE | cds 0 0 | | - * 3d | RETURNDATASIZE | 0 cds 0 0 | | - * 3d | RETURNDATASIZE | 0 0 cds 0 0 | | - * 37 | CALLDATACOPY | 0 0 | [0..calldatasize): calldata | - * | - * ::: delegatecall to implementation ::::::::::::::::::::::::::::::::::::::::::::::::: | - * 36 | CALLDATASIZE | cds 0 0 | [0..calldatasize): calldata | - * 3d | RETURNDATASIZE | 0 cds 0 0 | [0..calldatasize): calldata | - * 7f slot | PUSH32 slot | s 0 cds 0 0 | [0..calldatasize): calldata | - * 54 | SLOAD | i 0 cds 0 0 | [0..calldatasize): calldata | - * 5a | GAS | g i 0 cds 0 0 | [0..calldatasize): calldata | - * f4 | DELEGATECALL | succ | [0..calldatasize): calldata | - * | - * ::: copy returndata to memory :::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | rds succ | [0..calldatasize): calldata | - * 60 0x00 | PUSH1 0x00 | 0 rds succ | [0..calldatasize): calldata | - * 80 | DUP1 | 0 0 rds succ | [0..calldatasize): calldata | - * 3e | RETURNDATACOPY | succ | [0..returndatasize): returndata | - * | - * ::: branch on delegatecall status :::::::::::::::::::::::::::::::::::::::::::::::::: | - * 60 0x52 | PUSH1 0x52 | dest succ | [0..returndatasize): returndata | - * 57 | JUMPI | | [0..returndatasize): returndata | - * | - * ::: delegatecall failed, revert :::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | rds | [0..returndatasize): returndata | - * 60 0x00 | PUSH1 0x00 | 0 rds | [0..returndatasize): returndata | - * fd | REVERT | | [0..returndatasize): returndata | - * | - * ::: delegatecall succeeded, return ::::::::::::::::::::::::::::::::::::::::::::::::: | - * 5b | JUMPDEST | | [0..returndatasize): returndata | - * 3d | RETURNDATASIZE | rds | [0..returndatasize): returndata | - * 60 0x00 | PUSH1 0x00 | 0 rds | [0..returndatasize): returndata | - * f3 | RETURN | | [0..returndatasize): returndata | - * | - * ::: set new implementation (caller is factory) ::::::::::::::::::::::::::::::::::::: | - * 5b | JUMPDEST | 0 0 | | - * 3d | RETURNDATASIZE | 0 0 0 | | - * 35 | CALLDATALOAD | impl 0 0 | | - * 06 0x20 | PUSH1 0x20 | w impl 0 0 | | - * 35 | CALLDATALOAD | slot impl 0 0 | | - * 55 | SSTORE | 0 0 | | - * | - * ::: no extra calldata, return :::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 60 0x40 | PUSH1 0x40 | 2w 0 0 | | - * 80 | DUP1 | 2w 2w 0 0 | | - * 36 | CALLDATASIZE | cds 2w 2w 0 0 | | - * 11 | GT | gt 2w 0 0 | | - * 15 | ISZERO | lte 2w 0 0 | | - * 60 0x52 | PUSH1 0x52 | dest lte 2w 0 0 | | - * 57 | JUMPI | 2w 0 0 | | - * | - * ::: copy extra calldata to memory :::::::::::::::::::::::::::::::::::::::::::::::::: | - * 36 | CALLDATASIZE | cds 2w 0 0 | | - * 03 | SUB | t 0 0 | | - * 80 | DUP1 | t t 0 0 | | - * 60 0x40 | PUSH1 0x40 | 2w t t 0 0 | | - * 3d | RETURNDATASIZE | 0 2w t t 0 0 | | - * 37 | CALLDATACOPY | t 0 0 | [0..t): extra calldata | - * | - * ::: delegatecall to implementation ::::::::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | 0 t 0 0 | [0..t): extra calldata | - * 3d | RETURNDATASIZE | 0 0 t 0 0 | [0..t): extra calldata | - * 35 | CALLDATALOAD | i 0 t 0 0 | [0..t): extra calldata | - * 5a | GAS | g i 0 t 0 0 | [0..t): extra calldata | - * f4 | DELEGATECALL | succ | [0..t): extra calldata | - * | - * ::: copy returndata to memory :::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | rds succ | [0..t): extra calldata | - * 60 0x00 | PUSH1 0x00 | 0 rds succ | [0..t): extra calldata | - * 80 | DUP1 | 0 0 rds succ | [0..t): extra calldata | - * 3e | RETURNDATACOPY | succ | [0..returndatasize): returndata | - * | - * ::: branch on delegatecall status :::::::::::::::::::::::::::::::::::::::::::::::::: | - * 60 0x52 | PUSH1 0x52 | dest succ | [0..returndatasize): returndata | - * 57 | JUMPI | | [0..returndatasize): returndata | - * | - * ::: delegatecall failed, revert :::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | rds | [0..returndatasize): returndata | - * 60 0x00 | PUSH1 0x00 | 0 rds | [0..returndatasize): returndata | - * fd | REVERT | | [0..returndatasize): returndata | - * -------------------------------------------------------------------------------------+ - */ - - m := mload(0x40) - // forgefmt: disable-start - switch shr(112, address()) - case 0 { - // If the factory's address has six or more leading zero bytes. - mstore(add(m, 0x75), 0x604c573d6000fd) // 7 - mstore(add(m, 0x6e), 0x3d3560203555604080361115604c5736038060403d373d3d355af43d6000803e) // 32 - mstore(add(m, 0x4e), 0x3735a920a3ca505d382bbc545af43d6000803e604c573d6000fd5b3d6000f35b) // 32 - mstore(add(m, 0x2e), 0x14605157363d3d37363d7f360894a13ba1a3210667c828492db98dca3e2076cc) // 32 - mstore(add(m, 0x0e), address()) // 14 - mstore(m, 0x60793d8160093d39f33d3d336d) // 9 + 4 - } - default { - mstore(add(m, 0x7b), 0x6052573d6000fd) // 7 - mstore(add(m, 0x74), 0x3d356020355560408036111560525736038060403d373d3d355af43d6000803e) // 32 - mstore(add(m, 0x54), 0x3735a920a3ca505d382bbc545af43d6000803e6052573d6000fd5b3d6000f35b) // 32 - mstore(add(m, 0x34), 0x14605757363d3d37363d7f360894a13ba1a3210667c828492db98dca3e2076cc) // 32 - mstore(add(m, 0x14), address()) // 20 - mstore(m, 0x607f3d8160093d39f33d3d3373) // 9 + 4 - } - // forgefmt: disable-end - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Helper function to return an empty bytes calldata. - function _emptyData() internal pure returns (bytes calldata data) { - assembly { - data.length := 0 - } - } -} diff --git a/lib/solady/src/utils/ERC1967FactoryConstants.sol b/lib/solady/src/utils/ERC1967FactoryConstants.sol deleted file mode 100644 index 61ae337..0000000 --- a/lib/solady/src/utils/ERC1967FactoryConstants.sol +++ /dev/null @@ -1,45 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice The address and bytecode of the canonical ERC1967Factory deployment. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/ERC1967FactoryLib.sol) -/// @author jtriley-eth (https://github.com/jtriley-eth/minimum-viable-proxy) -/// -/// @dev The canonical ERC1967Factory is deployed permissionlessly via -/// 0age's ImmutableCreate2Factory located at 0x0000000000FFe8B47B3e2130213B802212439497. -/// -/// `ADDRESS = immutableCreate2Factory.safeCreate2(SALT, INITCODE)` -/// -/// If the canonical ERC1967Factory has not been deployed on your EVM chain of choice, -/// please feel free to deploy via 0age's ImmutableCreate2Factory. -/// -/// If 0age's ImmutableCreate2Factory has not been deployed on your EVM chain of choice, -/// please refer to 0age's ImmutableCreate2Factory deployment instructions at: -/// https://github.com/ProjectOpenSea/seaport/blob/main/docs/Deployment.md -/// -/// Contract verification: -/// - Source code: -/// https://github.com/Vectorized/solady/blob/5212e50fef1f2ff1b1b5e03a5d276a0d23c02713/src/utils/ERC1967Factory.sol -/// (The EXACT source code is required. Use the file at the commit instead of the latest copy.) -/// - Optimization Enabled: Yes with 1000000 runs -/// - Compiler Version: v0.8.19+commit.7dd6d404 -/// - Other Settings: default evmVersion, MIT license -library ERC1967FactoryConstants { - /// @dev The canonical ERC1967Factory address for EVM chains. - address internal constant ADDRESS = 0x0000000000006396FF2a80c067f99B3d2Ab4Df24; - - /// @dev The canonical ERC1967Factory bytecode for EVM chains. - /// Useful for forge tests: - /// `vm.etch(ADDRESS, BYTECODE)`. - bytes internal constant BYTECODE = - hex"6080604052600436106100b15760003560e01c8063545e7c611161006957806399a88ec41161004e57806399a88ec41461019d578063a97b90d5146101b0578063db4c545e146101c357600080fd5b8063545e7c61146101775780639623609d1461018a57600080fd5b80633729f9221161009a5780633729f922146101315780634314f120146101445780635414dff01461015757600080fd5b80631acfd02a146100b65780632abbef15146100d8575b600080fd5b3480156100c257600080fd5b506100d66100d1366004610604565b6101e6565b005b3480156100e457600080fd5b506101076100f3366004610637565b30600c908152600091909152602090205490565b60405173ffffffffffffffffffffffffffffffffffffffff90911681526020015b60405180910390f35b61010761013f366004610652565b610237565b6101076101523660046106d7565b61024e565b34801561016357600080fd5b50610107610172366004610738565b610267565b610107610185366004610604565b61029a565b6100d66101983660046106d7565b6102af565b6100d66101ab366004610604565b61035f565b6101076101be366004610751565b610370565b3480156101cf57600080fd5b506101d86103a9565b604051908152602001610128565b30600c52816000526020600c2033815414610209576382b429006000526004601cfd5b81905580827f7e644d79422f17c01e4894b5f4f588d331ebfa28653d42ae832dc59e38c9798f600080a35050565b60006102468484843685610370565b949350505050565b600061025e8585838087876103c2565b95945050505050565b6000806102726103a9565b905060ff600053806035523060601b6001528260155260556000209150600060355250919050565b60006102a88383368461024e565b9392505050565b30600c5283600052336020600c2054146102d1576382b429006000526004601cfd5b6040518381527f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc602082015281836040830137600080836040018334895af1610331573d610327576355299b496000526004601cfd5b3d6000803e3d6000fd5b5082847f5d611f318680d00598bb735d61bacf0c514c6b50e1e5ad30040a4df2b12791c7600080a350505050565b61036c82823660006102af565b5050565b60008360601c33148460601c151761039057632f6348366000526004601cfd5b61039f868686600187876103c2565b9695505050505050565b6000806103b461049c565b608960139091012092915050565b6000806103cd61049c565b90508480156103e757866089601384016000f592506103f3565b6089601383016000f092505b50816104075763301164256000526004601cfd5b8781527f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc602082015282846040830137600080846040018334865af161045a573d6103275763301164256000526004601cfd5b30600c5281600052866020600c20558688837fc95935a66d15e0da5e412aca0ad27ae891d20b2fb91cf3994b6a3bf2b8178082600080a4509695505050505050565b6040513060701c801561054257666052573d6000fd607b8301527f3d356020355560408036111560525736038060403d373d3d355af43d6000803e60748301527f3735a920a3ca505d382bbc545af43d6000803e6052573d6000fd5b3d6000f35b60548301527f14605757363d3d37363d7f360894a13ba1a3210667c828492db98dca3e2076cc60348301523060148301526c607f3d8160093d39f33d3d337382525090565b66604c573d6000fd60758301527f3d3560203555604080361115604c5736038060403d373d3d355af43d6000803e606e8301527f3735a920a3ca505d382bbc545af43d6000803e604c573d6000fd5b3d6000f35b604e8301527f14605157363d3d37363d7f360894a13ba1a3210667c828492db98dca3e2076cc602e83015230600e8301526c60793d8160093d39f33d3d336d82525090565b803573ffffffffffffffffffffffffffffffffffffffff811681146105ff57600080fd5b919050565b6000806040838503121561061757600080fd5b610620836105db565b915061062e602084016105db565b90509250929050565b60006020828403121561064957600080fd5b6102a8826105db565b60008060006060848603121561066757600080fd5b610670846105db565b925061067e602085016105db565b9150604084013590509250925092565b60008083601f8401126106a057600080fd5b50813567ffffffffffffffff8111156106b857600080fd5b6020830191508360208285010111156106d057600080fd5b9250929050565b600080600080606085870312156106ed57600080fd5b6106f6856105db565b9350610704602086016105db565b9250604085013567ffffffffffffffff81111561072057600080fd5b61072c8782880161068e565b95989497509550505050565b60006020828403121561074a57600080fd5b5035919050565b60008060008060006080868803121561076957600080fd5b610772866105db565b9450610780602087016105db565b935060408601359250606086013567ffffffffffffffff8111156107a357600080fd5b6107af8882890161068e565b96999598509396509294939250505056fea26469706673582212200ac7c3ccbc2d311c48bf5465b021542e0e306fe3c462c060ba6a3d2f81ff6c5f64736f6c63430008130033"; - - /// @dev The initcode used to deploy the canonical ERC1967Factory. - bytes internal constant INITCODE = abi.encodePacked( - hex"608060405234801561001057600080fd5b506107f6806100206000396000f3fe", BYTECODE - ); - - /// @dev For deterministic deployment via 0age's ImmutableCreate2Factory. - bytes32 internal constant SALT = - 0x0000000000000000000000000000000000000000e75e4f228818c80007508f33; -} diff --git a/lib/solady/src/utils/FixedPointMathLib.sol b/lib/solady/src/utils/FixedPointMathLib.sol deleted file mode 100644 index 590f809..0000000 --- a/lib/solady/src/utils/FixedPointMathLib.sol +++ /dev/null @@ -1,993 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Arithmetic library with operations for fixed-point numbers. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/FixedPointMathLib.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/FixedPointMathLib.sol) -library FixedPointMathLib { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The operation failed, as the output exceeds the maximum value of uint256. - error ExpOverflow(); - - /// @dev The operation failed, as the output exceeds the maximum value of uint256. - error FactorialOverflow(); - - /// @dev The operation failed, due to an overflow. - error RPowOverflow(); - - /// @dev The mantissa is too big to fit. - error MantissaOverflow(); - - /// @dev The operation failed, due to an multiplication overflow. - error MulWadFailed(); - - /// @dev The operation failed, either due to a - /// multiplication overflow, or a division by a zero. - error DivWadFailed(); - - /// @dev The multiply-divide operation failed, either due to a - /// multiplication overflow, or a division by a zero. - error MulDivFailed(); - - /// @dev The division failed, as the denominator is zero. - error DivFailed(); - - /// @dev The full precision multiply-divide operation failed, either due - /// to the result being larger than 256 bits, or a division by a zero. - error FullMulDivFailed(); - - /// @dev The output is undefined, as the input is less-than-or-equal to zero. - error LnWadUndefined(); - - /// @dev The input outside the acceptable domain. - error OutOfDomain(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The scalar of ETH and most ERC20s. - uint256 internal constant WAD = 1e18; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* SIMPLIFIED FIXED POINT OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Equivalent to `(x * y) / WAD` rounded down. - function mulWad(uint256 x, uint256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - // Equivalent to `require(y == 0 || x <= type(uint256).max / y)`. - if mul(y, gt(x, div(not(0), y))) { - mstore(0x00, 0xbac65e5b) // `MulWadFailed()`. - revert(0x1c, 0x04) - } - z := div(mul(x, y), WAD) - } - } - - /// @dev Equivalent to `(x * y) / WAD` rounded up. - function mulWadUp(uint256 x, uint256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - // Equivalent to `require(y == 0 || x <= type(uint256).max / y)`. - if mul(y, gt(x, div(not(0), y))) { - mstore(0x00, 0xbac65e5b) // `MulWadFailed()`. - revert(0x1c, 0x04) - } - z := add(iszero(iszero(mod(mul(x, y), WAD))), div(mul(x, y), WAD)) - } - } - - /// @dev Equivalent to `(x * WAD) / y` rounded down. - function divWad(uint256 x, uint256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - // Equivalent to `require(y != 0 && (WAD == 0 || x <= type(uint256).max / WAD))`. - if iszero(mul(y, iszero(mul(WAD, gt(x, div(not(0), WAD)))))) { - mstore(0x00, 0x7c5f487d) // `DivWadFailed()`. - revert(0x1c, 0x04) - } - z := div(mul(x, WAD), y) - } - } - - /// @dev Equivalent to `(x * WAD) / y` rounded up. - function divWadUp(uint256 x, uint256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - // Equivalent to `require(y != 0 && (WAD == 0 || x <= type(uint256).max / WAD))`. - if iszero(mul(y, iszero(mul(WAD, gt(x, div(not(0), WAD)))))) { - mstore(0x00, 0x7c5f487d) // `DivWadFailed()`. - revert(0x1c, 0x04) - } - z := add(iszero(iszero(mod(mul(x, WAD), y))), div(mul(x, WAD), y)) - } - } - - /// @dev Equivalent to `x` to the power of `y`. - /// because `x ** y = (e ** ln(x)) ** y = e ** (ln(x) * y)`. - function powWad(int256 x, int256 y) internal pure returns (int256) { - // Using `ln(x)` means `x` must be greater than 0. - return expWad((lnWad(x) * y) / int256(WAD)); - } - - /// @dev Returns `exp(x)`, denominated in `WAD`. - function expWad(int256 x) internal pure returns (int256 r) { - unchecked { - // When the result is less than 0.5 we return zero. - // This happens when `x <= floor(log(0.5e18) * 1e18) ≈ -42e18`. - if (x <= -42139678854452767551) return r; - - /// @solidity memory-safe-assembly - assembly { - // When the result is greater than `(2**255 - 1) / 1e18` we can not represent it as - // an int. This happens when `x >= floor(log((2**255 - 1) / 1e18) * 1e18) ≈ 135`. - if iszero(slt(x, 135305999368893231589)) { - mstore(0x00, 0xa37bfec9) // `ExpOverflow()`. - revert(0x1c, 0x04) - } - } - - // `x` is now in the range `(-42, 136) * 1e18`. Convert to `(-42, 136) * 2**96` - // for more intermediate precision and a binary basis. This base conversion - // is a multiplication by 1e18 / 2**96 = 5**18 / 2**78. - x = (x << 78) / 5 ** 18; - - // Reduce range of x to (-½ ln 2, ½ ln 2) * 2**96 by factoring out powers - // of two such that exp(x) = exp(x') * 2**k, where k is an integer. - // Solving this gives k = round(x / log(2)) and x' = x - k * log(2). - int256 k = ((x << 96) / 54916777467707473351141471128 + 2 ** 95) >> 96; - x = x - k * 54916777467707473351141471128; - - // `k` is in the range `[-61, 195]`. - - // Evaluate using a (6, 7)-term rational approximation. - // `p` is made monic, we'll multiply by a scale factor later. - int256 y = x + 1346386616545796478920950773328; - y = ((y * x) >> 96) + 57155421227552351082224309758442; - int256 p = y + x - 94201549194550492254356042504812; - p = ((p * y) >> 96) + 28719021644029726153956944680412240; - p = p * x + (4385272521454847904659076985693276 << 96); - - // We leave `p` in `2**192` basis so we don't need to scale it back up for the division. - int256 q = x - 2855989394907223263936484059900; - q = ((q * x) >> 96) + 50020603652535783019961831881945; - q = ((q * x) >> 96) - 533845033583426703283633433725380; - q = ((q * x) >> 96) + 3604857256930695427073651918091429; - q = ((q * x) >> 96) - 14423608567350463180887372962807573; - q = ((q * x) >> 96) + 26449188498355588339934803723976023; - - /// @solidity memory-safe-assembly - assembly { - // Div in assembly because solidity adds a zero check despite the unchecked. - // The q polynomial won't have zeros in the domain as all its roots are complex. - // No scaling is necessary because p is already `2**96` too large. - r := sdiv(p, q) - } - - // r should be in the range `(0.09, 0.25) * 2**96`. - - // We now need to multiply r by: - // - The scale factor `s ≈ 6.031367120`. - // - The `2**k` factor from the range reduction. - // - The `1e18 / 2**96` factor for base conversion. - // We do this all at once, with an intermediate result in `2**213` - // basis, so the final right shift is always by a positive amount. - r = int256( - (uint256(r) * 3822833074963236453042738258902158003155416615667) >> uint256(195 - k) - ); - } - } - - /// @dev Returns `ln(x)`, denominated in `WAD`. - function lnWad(int256 x) internal pure returns (int256 r) { - /// @solidity memory-safe-assembly - assembly { - if iszero(sgt(x, 0)) { - mstore(0x00, 0x1615e638) // `LnWadUndefined()`. - revert(0x1c, 0x04) - } - // We want to convert `x` from `10**18` fixed point to `2**96` fixed point. - // We do this by multiplying by `2**96 / 10**18`. But since - // `ln(x * C) = ln(x) + ln(C)`, we can simply do nothing here - // and add `ln(2**96 / 10**18)` at the end. - - // Compute `k = log2(x) - 96`, `t = 159 - k = 255 - log2(x) = 255 ^ log2(x)`. - let t := shl(7, lt(0xffffffffffffffffffffffffffffffff, x)) - t := or(t, shl(6, lt(0xffffffffffffffff, shr(t, x)))) - t := or(t, shl(5, lt(0xffffffff, shr(t, x)))) - t := or(t, shl(4, lt(0xffff, shr(t, x)))) - t := or(t, shl(3, lt(0xff, shr(t, x)))) - // forgefmt: disable-next-item - t := xor(t, byte(and(0x1f, shr(shr(t, x), 0x8421084210842108cc6318c6db6d54be)), - 0xf8f9f9faf9fdfafbf9fdfcfdfafbfcfef9fafdfafcfcfbfefafafcfbffffffff)) - - // Reduce range of x to (1, 2) * 2**96 - // ln(2^k * x) = k * ln(2) + ln(x) - x := shr(159, shl(t, x)) - - // Evaluate using a (8, 8)-term rational approximation. - // `p` is made monic, we will multiply by a scale factor later. - // forgefmt: disable-next-item - let p := sub( // This heavily nested expression is to avoid stack-too-deep for via-ir. - sar(96, mul(add(43456485725739037958740375743393, - sar(96, mul(add(24828157081833163892658089445524, - sar(96, mul(add(3273285459638523848632254066296, - x), x))), x))), x)), 11111509109440967052023855526967) - p := sub(sar(96, mul(p, x)), 45023709667254063763336534515857) - p := sub(sar(96, mul(p, x)), 14706773417378608786704636184526) - p := sub(mul(p, x), shl(96, 795164235651350426258249787498)) - - // We leave `p` in `2**192` basis so we don't need to scale it back up for the division. - // `q` is monic by convention. - let q := add(5573035233440673466300451813936, x) - q := add(71694874799317883764090561454958, sar(96, mul(x, q))) - q := add(283447036172924575727196451306956, sar(96, mul(x, q))) - q := add(401686690394027663651624208769553, sar(96, mul(x, q))) - q := add(204048457590392012362485061816622, sar(96, mul(x, q))) - q := add(31853899698501571402653359427138, sar(96, mul(x, q))) - q := add(909429971244387300277376558375, sar(96, mul(x, q))) - - // `r` is in the range `(0, 0.125) * 2**96`. - - // Finalization, we need to: - // - Multiply by the scale factor `s = 5.549…`. - // - Add `ln(2**96 / 10**18)`. - // - Add `k * ln(2)`. - // - Multiply by `10**18 / 2**96 = 5**18 >> 78`. - - // The q polynomial is known not to have zeros in the domain. - // No scaling required because p is already `2**96` too large. - r := sdiv(p, q) - // Multiply by the scaling factor: `s * 5e18 * 2**96`, base is now `5**18 * 2**192`. - r := mul(1677202110996718588342820967067443963516166, r) - // Add `ln(2) * k * 5e18 * 2**192`. - // forgefmt: disable-next-item - r := add(mul(16597577552685614221487285958193947469193820559219878177908093499208371, sub(159, t)), r) - // Add `ln(2**96 / 10**18) * 5e18 * 2**192`. - r := add(600920179829731861736702779321621459595472258049074101567377883020018308, r) - // Base conversion: mul `2**18 / 2**192`. - r := sar(174, r) - } - } - - /// @dev Returns `W_0(x)`, denominated in `WAD`. - /// See: https://en.wikipedia.org/wiki/Lambert_W_function - /// a.k.a. Product log function. This is an approximation of the principal branch. - function lambertW0Wad(int256 x) internal pure returns (int256 w) { - if ((w = x) <= -367879441171442322) revert OutOfDomain(); // `x` less than `-1/e`. - uint256 c; // Whether we need to avoid catastrophic cancellation. - uint256 i = 4; // Number of iterations. - if (w <= 0x1ffffffffffff) { - if (-0x4000000000000 <= w) { - i = 1; // Inputs near zero only take one step to converge. - } else if (w <= -0x3ffffffffffffff) { - i = 32; // Inputs near `-1/e` take very long to converge. - } - } else if (w >> 63 == 0) { - /// @solidity memory-safe-assembly - assembly { - // Inline log2 for more performance, since the range is small. - let v := shr(49, w) - let l := shl(3, lt(0xff, v)) - // forgefmt: disable-next-item - l := add(or(l, byte(and(0x1f, shr(shr(l, v), 0x8421084210842108cc6318c6db6d54be)), - 0x0706060506020504060203020504030106050205030304010505030400000000)), 49) - w := sdiv(shl(l, 7), byte(sub(l, 31), 0x0303030303030303040506080c13)) - c := gt(l, 60) - i := add(2, add(gt(l, 53), c)) - } - } else { - // `ln(x) - ln(ln(x)) + b * ln(ln(x)) / ln(x)`. - int256 ll = lnWad(w = lnWad(w)); - /// @solidity memory-safe-assembly - assembly { - w := add(sdiv(mul(ll, 1023715080943847266), w), sub(w, ll)) - i := add(3, iszero(shr(68, x))) - c := iszero(shr(143, x)) - } - if (c == 0) { - int256 wad = int256(WAD); - int256 p = x; - // If `x` is big, use Newton's so that intermediate values won't overflow. - do { - int256 e = expWad(w); - /// @solidity memory-safe-assembly - assembly { - let t := mul(w, div(e, wad)) - w := sub(w, sdiv(sub(t, x), div(add(e, t), wad))) - i := sub(i, 1) - } - if (p <= w) break; - p = w; - } while (i != 0); - /// @solidity memory-safe-assembly - assembly { - w := sub(w, sgt(w, 2)) - } - return w; - } - } - // forgefmt: disable-next-item - unchecked { - int256 wad = int256(WAD); - int256 p = x; - do { // Otherwise, use Halley's for faster convergence. - int256 e = expWad(w); - /// @solidity memory-safe-assembly - assembly { - let t := add(w, wad) - let s := sub(mul(w, e), mul(x, wad)) - w := sub(w, sdiv(mul(s, wad), sub(mul(e, t), sdiv(mul(add(t, wad), s), add(t, t))))) - } - if (p <= w) break; - p = w; - } while (--i != c); - /// @solidity memory-safe-assembly - assembly { - w := sub(w, sgt(w, 2)) - } - // For certain ranges of `x`, we'll use the quadratic-rate recursive formula of - // R. Iacono and J.P. Boyd for the last iteration, to avoid catastrophic cancellation. - if (c != 0) { - /// @solidity memory-safe-assembly - assembly { - x := sdiv(mul(x, wad), w) - } - x = (w * (wad + lnWad(x))); - /// @solidity memory-safe-assembly - assembly { - w := sdiv(x, add(wad, w)) - } - } - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* GENERAL NUMBER UTILITIES */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Calculates `floor(a * b / d)` with full precision. - /// Throws if result overflows a uint256 or when `d` is zero. - /// Credit to Remco Bloemen under MIT license: https://2π.com/21/muldiv - function fullMulDiv(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - for {} 1 {} { - // 512-bit multiply `[p1 p0] = x * y`. - // Compute the product mod `2**256` and mod `2**256 - 1` - // then use the Chinese Remainder Theorem to reconstruct - // the 512 bit result. The result is stored in two 256 - // variables such that `product = p1 * 2**256 + p0`. - - // Least significant 256 bits of the product. - let p0 := mul(x, y) - let mm := mulmod(x, y, not(0)) - // Most significant 256 bits of the product. - let p1 := sub(mm, add(p0, lt(mm, p0))) - - // Handle non-overflow cases, 256 by 256 division. - if iszero(p1) { - if iszero(d) { - mstore(0x00, 0xae47f702) // `FullMulDivFailed()`. - revert(0x1c, 0x04) - } - result := div(p0, d) - break - } - - // Make sure the result is less than `2**256`. Also prevents `d == 0`. - if iszero(gt(d, p1)) { - mstore(0x00, 0xae47f702) // `FullMulDivFailed()`. - revert(0x1c, 0x04) - } - - /*------------------- 512 by 256 division --------------------*/ - - // Make division exact by subtracting the remainder from `[p1 p0]`. - // Compute remainder using mulmod. - let r := mulmod(x, y, d) - // `t` is the least significant bit of `d`. - // Always greater or equal to 1. - let t := and(d, sub(0, d)) - // Divide `d` by `t`, which is a power of two. - d := div(d, t) - // Invert `d mod 2**256` - // Now that `d` is an odd number, it has an inverse - // modulo `2**256` such that `d * inv = 1 mod 2**256`. - // Compute the inverse by starting with a seed that is correct - // correct for four bits. That is, `d * inv = 1 mod 2**4`. - let inv := xor(mul(3, d), 2) - // Now use Newton-Raphson iteration to improve the precision. - // Thanks to Hensel's lifting lemma, this also works in modular - // arithmetic, doubling the correct bits in each step. - inv := mul(inv, sub(2, mul(d, inv))) // inverse mod 2**8 - inv := mul(inv, sub(2, mul(d, inv))) // inverse mod 2**16 - inv := mul(inv, sub(2, mul(d, inv))) // inverse mod 2**32 - inv := mul(inv, sub(2, mul(d, inv))) // inverse mod 2**64 - inv := mul(inv, sub(2, mul(d, inv))) // inverse mod 2**128 - result := - mul( - // Divide [p1 p0] by the factors of two. - // Shift in bits from `p1` into `p0`. For this we need - // to flip `t` such that it is `2**256 / t`. - or(mul(sub(p1, gt(r, p0)), add(div(sub(0, t), t), 1)), div(sub(p0, r), t)), - // inverse mod 2**256 - mul(inv, sub(2, mul(d, inv))) - ) - break - } - } - } - - /// @dev Calculates `floor(x * y / d)` with full precision, rounded up. - /// Throws if result overflows a uint256 or when `d` is zero. - /// Credit to Uniswap-v3-core under MIT license: - /// https://github.com/Uniswap/v3-core/blob/contracts/libraries/FullMath.sol - function fullMulDivUp(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 result) { - result = fullMulDiv(x, y, d); - /// @solidity memory-safe-assembly - assembly { - if mulmod(x, y, d) { - result := add(result, 1) - if iszero(result) { - mstore(0x00, 0xae47f702) // `FullMulDivFailed()`. - revert(0x1c, 0x04) - } - } - } - } - - /// @dev Returns `floor(x * y / d)`. - /// Reverts if `x * y` overflows, or `d` is zero. - function mulDiv(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - // Equivalent to require(d != 0 && (y == 0 || x <= type(uint256).max / y)) - if iszero(mul(d, iszero(mul(y, gt(x, div(not(0), y)))))) { - mstore(0x00, 0xad251c27) // `MulDivFailed()`. - revert(0x1c, 0x04) - } - z := div(mul(x, y), d) - } - } - - /// @dev Returns `ceil(x * y / d)`. - /// Reverts if `x * y` overflows, or `d` is zero. - function mulDivUp(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - // Equivalent to require(d != 0 && (y == 0 || x <= type(uint256).max / y)) - if iszero(mul(d, iszero(mul(y, gt(x, div(not(0), y)))))) { - mstore(0x00, 0xad251c27) // `MulDivFailed()`. - revert(0x1c, 0x04) - } - z := add(iszero(iszero(mod(mul(x, y), d))), div(mul(x, y), d)) - } - } - - /// @dev Returns `ceil(x / d)`. - /// Reverts if `d` is zero. - function divUp(uint256 x, uint256 d) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - if iszero(d) { - mstore(0x00, 0x65244e4e) // `DivFailed()`. - revert(0x1c, 0x04) - } - z := add(iszero(iszero(mod(x, d))), div(x, d)) - } - } - - /// @dev Returns `max(0, x - y)`. - function zeroFloorSub(uint256 x, uint256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := mul(gt(x, y), sub(x, y)) - } - } - - /// @dev Exponentiate `x` to `y` by squaring, denominated in base `b`. - /// Reverts if the computation overflows. - function rpow(uint256 x, uint256 y, uint256 b) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := mul(b, iszero(y)) // `0 ** 0 = 1`. Otherwise, `0 ** n = 0`. - if x { - z := xor(b, mul(xor(b, x), and(y, 1))) // `z = isEven(y) ? scale : x` - let half := shr(1, b) // Divide `b` by 2. - // Divide `y` by 2 every iteration. - for { y := shr(1, y) } y { y := shr(1, y) } { - let xx := mul(x, x) // Store x squared. - let xxRound := add(xx, half) // Round to the nearest number. - // Revert if `xx + half` overflowed, or if `x ** 2` overflows. - if or(lt(xxRound, xx), shr(128, x)) { - mstore(0x00, 0x49f7642b) // `RPowOverflow()`. - revert(0x1c, 0x04) - } - x := div(xxRound, b) // Set `x` to scaled `xxRound`. - // If `y` is odd: - if and(y, 1) { - let zx := mul(z, x) // Compute `z * x`. - let zxRound := add(zx, half) // Round to the nearest number. - // If `z * x` overflowed or `zx + half` overflowed: - if or(xor(div(zx, x), z), lt(zxRound, zx)) { - // Revert if `x` is non-zero. - if iszero(iszero(x)) { - mstore(0x00, 0x49f7642b) // `RPowOverflow()`. - revert(0x1c, 0x04) - } - } - z := div(zxRound, b) // Return properly scaled `zxRound`. - } - } - } - } - } - - /// @dev Returns the square root of `x`. - function sqrt(uint256 x) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - // `floor(sqrt(2**15)) = 181`. `sqrt(2**15) - 181 = 2.84`. - z := 181 // The "correct" value is 1, but this saves a multiplication later. - - // This segment is to get a reasonable initial estimate for the Babylonian method. With a bad - // start, the correct # of bits increases ~linearly each iteration instead of ~quadratically. - - // Let `y = x / 2**r`. We check `y >= 2**(k + 8)` - // but shift right by `k` bits to ensure that if `x >= 256`, then `y >= 256`. - let r := shl(7, lt(0xffffffffffffffffffffffffffffffffff, x)) - r := or(r, shl(6, lt(0xffffffffffffffffff, shr(r, x)))) - r := or(r, shl(5, lt(0xffffffffff, shr(r, x)))) - r := or(r, shl(4, lt(0xffffff, shr(r, x)))) - z := shl(shr(1, r), z) - - // Goal was to get `z*z*y` within a small factor of `x`. More iterations could - // get y in a tighter range. Currently, we will have y in `[256, 256*(2**16))`. - // We ensured `y >= 256` so that the relative difference between `y` and `y+1` is small. - // That's not possible if `x < 256` but we can just verify those cases exhaustively. - - // Now, `z*z*y <= x < z*z*(y+1)`, and `y <= 2**(16+8)`, and either `y >= 256`, or `x < 256`. - // Correctness can be checked exhaustively for `x < 256`, so we assume `y >= 256`. - // Then `z*sqrt(y)` is within `sqrt(257)/sqrt(256)` of `sqrt(x)`, or about 20bps. - - // For `s` in the range `[1/256, 256]`, the estimate `f(s) = (181/1024) * (s+1)` - // is in the range `(1/2.84 * sqrt(s), 2.84 * sqrt(s))`, - // with largest error when `s = 1` and when `s = 256` or `1/256`. - - // Since `y` is in `[256, 256*(2**16))`, let `a = y/65536`, so that `a` is in `[1/256, 256)`. - // Then we can estimate `sqrt(y)` using - // `sqrt(65536) * 181/1024 * (a + 1) = 181/4 * (y + 65536)/65536 = 181 * (y + 65536)/2**18`. - - // There is no overflow risk here since `y < 2**136` after the first branch above. - z := shr(18, mul(z, add(shr(r, x), 65536))) // A `mul()` is saved from starting `z` at 181. - - // Given the worst case multiplicative error of 2.84 above, 7 iterations should be enough. - z := shr(1, add(z, div(x, z))) - z := shr(1, add(z, div(x, z))) - z := shr(1, add(z, div(x, z))) - z := shr(1, add(z, div(x, z))) - z := shr(1, add(z, div(x, z))) - z := shr(1, add(z, div(x, z))) - z := shr(1, add(z, div(x, z))) - - // If `x+1` is a perfect square, the Babylonian method cycles between - // `floor(sqrt(x))` and `ceil(sqrt(x))`. This statement ensures we return floor. - // See: https://en.wikipedia.org/wiki/Integer_square_root#Using_only_integer_division - z := sub(z, lt(div(x, z), z)) - } - } - - /// @dev Returns the cube root of `x`. - /// Credit to bout3fiddy and pcaversaccio under AGPLv3 license: - /// https://github.com/pcaversaccio/snekmate/blob/main/src/utils/Math.vy - function cbrt(uint256 x) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - let r := shl(7, lt(0xffffffffffffffffffffffffffffffff, x)) - r := or(r, shl(6, lt(0xffffffffffffffff, shr(r, x)))) - r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) - r := or(r, shl(4, lt(0xffff, shr(r, x)))) - r := or(r, shl(3, lt(0xff, shr(r, x)))) - - z := div(shl(div(r, 3), shl(lt(0xf, shr(r, x)), 0xf)), xor(7, mod(r, 3))) - - z := div(add(add(div(x, mul(z, z)), z), z), 3) - z := div(add(add(div(x, mul(z, z)), z), z), 3) - z := div(add(add(div(x, mul(z, z)), z), z), 3) - z := div(add(add(div(x, mul(z, z)), z), z), 3) - z := div(add(add(div(x, mul(z, z)), z), z), 3) - z := div(add(add(div(x, mul(z, z)), z), z), 3) - z := div(add(add(div(x, mul(z, z)), z), z), 3) - - z := sub(z, lt(div(x, mul(z, z)), z)) - } - } - - /// @dev Returns the square root of `x`, denominated in `WAD`. - function sqrtWad(uint256 x) internal pure returns (uint256 z) { - unchecked { - z = 10 ** 9; - if (x <= type(uint256).max / 10 ** 36 - 1) { - x *= 10 ** 18; - z = 1; - } - z *= sqrt(x); - } - } - - /// @dev Returns the cube root of `x`, denominated in `WAD`. - function cbrtWad(uint256 x) internal pure returns (uint256 z) { - unchecked { - z = 10 ** 12; - if (x <= (type(uint256).max / 10 ** 36) * 10 ** 18 - 1) { - if (x >= type(uint256).max / 10 ** 36) { - x *= 10 ** 18; - z = 10 ** 6; - } else { - x *= 10 ** 36; - z = 1; - } - } - z *= cbrt(x); - } - } - - /// @dev Returns the factorial of `x`. - function factorial(uint256 x) internal pure returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - if iszero(lt(x, 58)) { - mstore(0x00, 0xaba0f2a2) // `FactorialOverflow()`. - revert(0x1c, 0x04) - } - for { result := 1 } x { x := sub(x, 1) } { result := mul(result, x) } - } - } - - /// @dev Returns the log2 of `x`. - /// Equivalent to computing the index of the most significant bit (MSB) of `x`. - /// Returns 0 if `x` is zero. - function log2(uint256 x) internal pure returns (uint256 r) { - /// @solidity memory-safe-assembly - assembly { - r := shl(7, lt(0xffffffffffffffffffffffffffffffff, x)) - r := or(r, shl(6, lt(0xffffffffffffffff, shr(r, x)))) - r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) - r := or(r, shl(4, lt(0xffff, shr(r, x)))) - r := or(r, shl(3, lt(0xff, shr(r, x)))) - // forgefmt: disable-next-item - r := or(r, byte(and(0x1f, shr(shr(r, x), 0x8421084210842108cc6318c6db6d54be)), - 0x0706060506020504060203020504030106050205030304010505030400000000)) - } - } - - /// @dev Returns the log2 of `x`, rounded up. - /// Returns 0 if `x` is zero. - function log2Up(uint256 x) internal pure returns (uint256 r) { - r = log2(x); - /// @solidity memory-safe-assembly - assembly { - r := add(r, lt(shl(r, 1), x)) - } - } - - /// @dev Returns the log10 of `x`. - /// Returns 0 if `x` is zero. - function log10(uint256 x) internal pure returns (uint256 r) { - /// @solidity memory-safe-assembly - assembly { - if iszero(lt(x, 100000000000000000000000000000000000000)) { - x := div(x, 100000000000000000000000000000000000000) - r := 38 - } - if iszero(lt(x, 100000000000000000000)) { - x := div(x, 100000000000000000000) - r := add(r, 20) - } - if iszero(lt(x, 10000000000)) { - x := div(x, 10000000000) - r := add(r, 10) - } - if iszero(lt(x, 100000)) { - x := div(x, 100000) - r := add(r, 5) - } - r := add(r, add(gt(x, 9), add(gt(x, 99), add(gt(x, 999), gt(x, 9999))))) - } - } - - /// @dev Returns the log10 of `x`, rounded up. - /// Returns 0 if `x` is zero. - function log10Up(uint256 x) internal pure returns (uint256 r) { - r = log10(x); - /// @solidity memory-safe-assembly - assembly { - r := add(r, lt(exp(10, r), x)) - } - } - - /// @dev Returns the log256 of `x`. - /// Returns 0 if `x` is zero. - function log256(uint256 x) internal pure returns (uint256 r) { - /// @solidity memory-safe-assembly - assembly { - r := shl(7, lt(0xffffffffffffffffffffffffffffffff, x)) - r := or(r, shl(6, lt(0xffffffffffffffff, shr(r, x)))) - r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) - r := or(r, shl(4, lt(0xffff, shr(r, x)))) - r := or(shr(3, r), lt(0xff, shr(r, x))) - } - } - - /// @dev Returns the log256 of `x`, rounded up. - /// Returns 0 if `x` is zero. - function log256Up(uint256 x) internal pure returns (uint256 r) { - r = log256(x); - /// @solidity memory-safe-assembly - assembly { - r := add(r, lt(shl(shl(3, r), 1), x)) - } - } - - /// @dev Returns the scientific notation format `mantissa * 10 ** exponent` of `x`. - /// Useful for compressing prices (e.g. using 25 bit mantissa and 7 bit exponent). - function sci(uint256 x) internal pure returns (uint256 mantissa, uint256 exponent) { - /// @solidity memory-safe-assembly - assembly { - mantissa := x - if mantissa { - if iszero(mod(mantissa, 1000000000000000000000000000000000)) { - mantissa := div(mantissa, 1000000000000000000000000000000000) - exponent := 33 - } - if iszero(mod(mantissa, 10000000000000000000)) { - mantissa := div(mantissa, 10000000000000000000) - exponent := add(exponent, 19) - } - if iszero(mod(mantissa, 1000000000000)) { - mantissa := div(mantissa, 1000000000000) - exponent := add(exponent, 12) - } - if iszero(mod(mantissa, 1000000)) { - mantissa := div(mantissa, 1000000) - exponent := add(exponent, 6) - } - if iszero(mod(mantissa, 10000)) { - mantissa := div(mantissa, 10000) - exponent := add(exponent, 4) - } - if iszero(mod(mantissa, 100)) { - mantissa := div(mantissa, 100) - exponent := add(exponent, 2) - } - if iszero(mod(mantissa, 10)) { - mantissa := div(mantissa, 10) - exponent := add(exponent, 1) - } - } - } - } - - /// @dev Convenience function for packing `x` into a smaller number using `sci`. - /// The `mantissa` will be in bits [7..255] (the upper 249 bits). - /// The `exponent` will be in bits [0..6] (the lower 7 bits). - /// Use `SafeCastLib` to safely ensure that the `packed` number is small - /// enough to fit in the desired unsigned integer type: - /// ``` - /// uint32 packed = SafeCastLib.toUint32(FixedPointMathLib.packSci(777 ether)); - /// ``` - function packSci(uint256 x) internal pure returns (uint256 packed) { - (x, packed) = sci(x); // Reuse for `mantissa` and `exponent`. - /// @solidity memory-safe-assembly - assembly { - if shr(249, x) { - mstore(0x00, 0xce30380c) // `MantissaOverflow()`. - revert(0x1c, 0x04) - } - packed := or(shl(7, x), packed) - } - } - - /// @dev Convenience function for unpacking a packed number from `packSci`. - function unpackSci(uint256 packed) internal pure returns (uint256 unpacked) { - unchecked { - unpacked = (packed >> 7) * 10 ** (packed & 0x7f); - } - } - - /// @dev Returns the average of `x` and `y`. - function avg(uint256 x, uint256 y) internal pure returns (uint256 z) { - unchecked { - z = (x & y) + ((x ^ y) >> 1); - } - } - - /// @dev Returns the average of `x` and `y`. - function avg(int256 x, int256 y) internal pure returns (int256 z) { - unchecked { - z = (x >> 1) + (y >> 1) + (((x & 1) + (y & 1)) >> 1); - } - } - - /// @dev Returns the absolute value of `x`. - function abs(int256 x) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := xor(sub(0, shr(255, x)), add(sub(0, shr(255, x)), x)) - } - } - - /// @dev Returns the absolute distance between `x` and `y`. - function dist(int256 x, int256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := xor(mul(xor(sub(y, x), sub(x, y)), sgt(x, y)), sub(y, x)) - } - } - - /// @dev Returns the minimum of `x` and `y`. - function min(uint256 x, uint256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := xor(x, mul(xor(x, y), lt(y, x))) - } - } - - /// @dev Returns the minimum of `x` and `y`. - function min(int256 x, int256 y) internal pure returns (int256 z) { - /// @solidity memory-safe-assembly - assembly { - z := xor(x, mul(xor(x, y), slt(y, x))) - } - } - - /// @dev Returns the maximum of `x` and `y`. - function max(uint256 x, uint256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := xor(x, mul(xor(x, y), gt(y, x))) - } - } - - /// @dev Returns the maximum of `x` and `y`. - function max(int256 x, int256 y) internal pure returns (int256 z) { - /// @solidity memory-safe-assembly - assembly { - z := xor(x, mul(xor(x, y), sgt(y, x))) - } - } - - /// @dev Returns `x`, bounded to `minValue` and `maxValue`. - function clamp(uint256 x, uint256 minValue, uint256 maxValue) - internal - pure - returns (uint256 z) - { - /// @solidity memory-safe-assembly - assembly { - z := xor(x, mul(xor(x, minValue), gt(minValue, x))) - z := xor(z, mul(xor(z, maxValue), lt(maxValue, z))) - } - } - - /// @dev Returns `x`, bounded to `minValue` and `maxValue`. - function clamp(int256 x, int256 minValue, int256 maxValue) internal pure returns (int256 z) { - /// @solidity memory-safe-assembly - assembly { - z := xor(x, mul(xor(x, minValue), sgt(minValue, x))) - z := xor(z, mul(xor(z, maxValue), slt(maxValue, z))) - } - } - - /// @dev Returns greatest common divisor of `x` and `y`. - function gcd(uint256 x, uint256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - for { z := x } y {} { - let t := y - y := mod(z, y) - z := t - } - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* RAW NUMBER OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns `x + y`, without checking for overflow. - function rawAdd(uint256 x, uint256 y) internal pure returns (uint256 z) { - unchecked { - z = x + y; - } - } - - /// @dev Returns `x + y`, without checking for overflow. - function rawAdd(int256 x, int256 y) internal pure returns (int256 z) { - unchecked { - z = x + y; - } - } - - /// @dev Returns `x - y`, without checking for underflow. - function rawSub(uint256 x, uint256 y) internal pure returns (uint256 z) { - unchecked { - z = x - y; - } - } - - /// @dev Returns `x - y`, without checking for underflow. - function rawSub(int256 x, int256 y) internal pure returns (int256 z) { - unchecked { - z = x - y; - } - } - - /// @dev Returns `x * y`, without checking for overflow. - function rawMul(uint256 x, uint256 y) internal pure returns (uint256 z) { - unchecked { - z = x * y; - } - } - - /// @dev Returns `x * y`, without checking for overflow. - function rawMul(int256 x, int256 y) internal pure returns (int256 z) { - unchecked { - z = x * y; - } - } - - /// @dev Returns `x / y`, returning 0 if `y` is zero. - function rawDiv(uint256 x, uint256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := div(x, y) - } - } - - /// @dev Returns `x / y`, returning 0 if `y` is zero. - function rawSDiv(int256 x, int256 y) internal pure returns (int256 z) { - /// @solidity memory-safe-assembly - assembly { - z := sdiv(x, y) - } - } - - /// @dev Returns `x % y`, returning 0 if `y` is zero. - function rawMod(uint256 x, uint256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := mod(x, y) - } - } - - /// @dev Returns `x % y`, returning 0 if `y` is zero. - function rawSMod(int256 x, int256 y) internal pure returns (int256 z) { - /// @solidity memory-safe-assembly - assembly { - z := smod(x, y) - } - } - - /// @dev Returns `(x + y) % d`, return 0 if `d` if zero. - function rawAddMod(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := addmod(x, y, d) - } - } - - /// @dev Returns `(x * y) % d`, return 0 if `d` if zero. - function rawMulMod(uint256 x, uint256 y, uint256 d) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := mulmod(x, y, d) - } - } -} diff --git a/lib/solady/src/utils/GasBurnerLib.sol b/lib/solady/src/utils/GasBurnerLib.sol deleted file mode 100644 index 6150073..0000000 --- a/lib/solady/src/utils/GasBurnerLib.sol +++ /dev/null @@ -1,25 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for burning gas without reverting. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/GasBurnerLib.sol) -library GasBurnerLib { - /// @dev Burns approximately `x` amount of gas. - /// Intended for Contract Secured Revenue (CSR). - /// - /// Recommendation: pass in an admin-controlled dynamic value instead of a hardcoded one. - /// This is so that you can adjust your contract as needed depending on market conditions, - /// and to give you and your users a leeway in case the L2 chain change the rules. - function burn(uint256 x) internal pure { - /// @solidity memory-safe-assembly - assembly { - mstore(0x10, or(1, x)) - let n := mul(gt(x, 120), div(x, 91)) - // We use keccak256 instead of blake2f precompile for better widespread compatibility. - for { let i := 0 } iszero(eq(i, n)) { i := add(i, 1) } { - mstore(0x10, keccak256(0x10, 0x10)) // Yes. - } - if iszero(mload(0x10)) { invalid() } - } - } -} diff --git a/lib/solady/src/utils/JSONParserLib.sol b/lib/solady/src/utils/JSONParserLib.sol deleted file mode 100644 index eaa7649..0000000 --- a/lib/solady/src/utils/JSONParserLib.sol +++ /dev/null @@ -1,815 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for parsing JSONs. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/JSONParserLib.sol) -library JSONParserLib { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The input is invalid. - error ParsingFailed(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // There are 6 types of variables in JSON (excluding undefined). - - /// @dev For denoting that an item has not been initialized. - /// A item returned from `parse` will never be of an undefined type. - /// Parsing a invalid JSON string will simply revert. - uint8 internal constant TYPE_UNDEFINED = 0; - - /// @dev Type representing an array (e.g. `[1,2,3]`). - uint8 internal constant TYPE_ARRAY = 1; - - /// @dev Type representing an object (e.g. `{"a":"A","b":"B"}`). - uint8 internal constant TYPE_OBJECT = 2; - - /// @dev Type representing a number (e.g. `-1.23e+21`). - uint8 internal constant TYPE_NUMBER = 3; - - /// @dev Type representing a string (e.g. `"hello"`). - uint8 internal constant TYPE_STRING = 4; - - /// @dev Type representing a boolean (i.e. `true` or `false`). - uint8 internal constant TYPE_BOOLEAN = 5; - - /// @dev Type representing null (i.e. `null`). - uint8 internal constant TYPE_NULL = 6; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STRUCTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev A pointer to a parsed JSON node. - struct Item { - // Do NOT modify the `_data` directly. - uint256 _data; - } - - // Private constants for packing `_data`. - - uint256 private constant _BITPOS_STRING = 32 * 7 - 8; - uint256 private constant _BITPOS_KEY_LENGTH = 32 * 6 - 8; - uint256 private constant _BITPOS_KEY = 32 * 5 - 8; - uint256 private constant _BITPOS_VALUE_LENGTH = 32 * 4 - 8; - uint256 private constant _BITPOS_VALUE = 32 * 3 - 8; - uint256 private constant _BITPOS_CHILD = 32 * 2 - 8; - uint256 private constant _BITPOS_SIBLING_OR_PARENT = 32 * 1 - 8; - uint256 private constant _BITMASK_POINTER = 0xffffffff; - uint256 private constant _BITMASK_TYPE = 7; - uint256 private constant _KEY_INITED = 1 << 3; - uint256 private constant _VALUE_INITED = 1 << 4; - uint256 private constant _CHILDREN_INITED = 1 << 5; - uint256 private constant _PARENT_IS_ARRAY = 1 << 6; - uint256 private constant _PARENT_IS_OBJECT = 1 << 7; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* JSON PARSING OPERATION */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Parses the JSON string `s`, and returns the root. - /// Reverts if `s` is not a valid JSON as specified in RFC 8259. - /// Object items WILL simply contain all their children, inclusive of repeated keys, - /// in the same order which they appear in the JSON string. - /// - /// Note: For efficiency, this function WILL NOT make a copy of `s`. - /// The parsed tree WILL contain offsets to `s`. - /// Do NOT pass in a string that WILL be modified later on. - function parse(string memory s) internal pure returns (Item memory result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x40, result) // We will use our own allocation instead. - } - bytes32 r = _query(_toInput(s), 255); - /// @solidity memory-safe-assembly - assembly { - result := r - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* JSON ITEM OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // Note: - // - An item is a node in the JSON tree. - // - The value of a string item WILL be double-quoted, JSON encoded. - // - We make a distinction between `index` and `key`. - // - Items in arrays are located by `index` (uint256). - // - Items in objects are located by `key` (string). - // - Keys are always strings, double-quoted, JSON encoded. - // - // These design choices are made to balance between efficiency and ease-of-use. - - /// @dev Returns the string value of the item. - /// This is its exact string representation in the original JSON string. - /// The returned string WILL have leading and trailing whitespace trimmed. - /// All inner whitespace WILL be preserved, exactly as it is in the original JSON string. - /// If the item's type is string, the returned string WILL be double-quoted, JSON encoded. - /// - /// Note: This function lazily instantiates and caches the returned string. - /// Do NOT modify the returned string. - function value(Item memory item) internal pure returns (string memory result) { - bytes32 r = _query(_toInput(item), 0); - /// @solidity memory-safe-assembly - assembly { - result := r - } - } - - /// @dev Returns the index of the item in the array. - /// It the item's parent is not an array, returns 0. - function index(Item memory item) internal pure returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - if and(mload(item), _PARENT_IS_ARRAY) { - result := and(_BITMASK_POINTER, shr(_BITPOS_KEY, mload(item))) - } - } - } - - /// @dev Returns the key of the item in the object. - /// It the item's parent is not an object, returns an empty string. - /// The returned string WILL be double-quoted, JSON encoded. - /// - /// Note: This function lazily instantiates and caches the returned string. - /// Do NOT modify the returned string. - function key(Item memory item) internal pure returns (string memory result) { - if (item._data & _PARENT_IS_OBJECT != 0) { - bytes32 r = _query(_toInput(item), 1); - /// @solidity memory-safe-assembly - assembly { - result := r - } - } - } - - /// @dev Returns the key of the item in the object. - /// It the item is neither an array nor object, returns an empty array. - /// - /// Note: This function lazily instantiates and caches the returned array. - /// Do NOT modify the returned array. - function children(Item memory item) internal pure returns (Item[] memory result) { - bytes32 r = _query(_toInput(item), 3); - /// @solidity memory-safe-assembly - assembly { - result := r - } - } - - /// @dev Returns the number of children. - /// It the item is neither an array nor object, returns zero. - function size(Item memory item) internal pure returns (uint256 result) { - bytes32 r = _query(_toInput(item), 3); - /// @solidity memory-safe-assembly - assembly { - result := mload(r) - } - } - - /// @dev Returns the item at index `i` for (array). - /// If `item` is not an array, the result's type WILL be undefined. - /// If there is no item with the index, the result's type WILL be undefined. - function at(Item memory item, uint256 i) internal pure returns (Item memory result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x40, result) // Free the default allocation. We'll allocate manually. - } - bytes32 r = _query(_toInput(item), 3); - /// @solidity memory-safe-assembly - assembly { - result := mload(add(add(r, 0x20), shl(5, i))) - if iszero(and(lt(i, mload(r)), eq(and(mload(item), _BITMASK_TYPE), TYPE_ARRAY))) { - result := 0x60 // Reset to the zero pointer. - } - } - } - - /// @dev Returns the item at key `k` for (object). - /// If `item` is not an object, the result's type WILL be undefined. - /// The key MUST be double-quoted, JSON encoded. This is for efficiency reasons. - /// - Correct : `item.at('"k"')`. - /// - Wrong : `item.at("k")`. - /// For duplicated keys, the last item with the key WILL be returned. - /// If there is no item with the key, the result's type WILL be undefined. - function at(Item memory item, string memory k) internal pure returns (Item memory result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x40, result) // Free the default allocation. We'll allocate manually. - result := 0x60 // Initialize to the zero pointer. - } - if (isObject(item)) { - bytes32 kHash = keccak256(bytes(k)); - Item[] memory r = children(item); - // We'll just do a linear search. The alternatives are very bloated. - for (uint256 i = r.length << 5; i != 0;) { - /// @solidity memory-safe-assembly - assembly { - item := mload(add(r, i)) - i := sub(i, 0x20) - } - if (keccak256(bytes(key(item))) != kHash) continue; - result = item; - break; - } - } - } - - /// @dev Returns the item's type. - function getType(Item memory item) internal pure returns (uint8 result) { - result = uint8(item._data & _BITMASK_TYPE); - } - - /// Note: All types are mutually exclusive. - - /// @dev Returns whether the item is of type undefined. - function isUndefined(Item memory item) internal pure returns (bool result) { - result = item._data & _BITMASK_TYPE == TYPE_UNDEFINED; - } - - /// @dev Returns whether the item is of type array. - function isArray(Item memory item) internal pure returns (bool result) { - result = item._data & _BITMASK_TYPE == TYPE_ARRAY; - } - - /// @dev Returns whether the item is of type object. - function isObject(Item memory item) internal pure returns (bool result) { - result = item._data & _BITMASK_TYPE == TYPE_OBJECT; - } - - /// @dev Returns whether the item is of type number. - function isNumber(Item memory item) internal pure returns (bool result) { - result = item._data & _BITMASK_TYPE == TYPE_NUMBER; - } - - /// @dev Returns whether the item is of type string. - function isString(Item memory item) internal pure returns (bool result) { - result = item._data & _BITMASK_TYPE == TYPE_STRING; - } - - /// @dev Returns whether the item is of type boolean. - function isBoolean(Item memory item) internal pure returns (bool result) { - result = item._data & _BITMASK_TYPE == TYPE_BOOLEAN; - } - - /// @dev Returns whether the item is of type null. - function isNull(Item memory item) internal pure returns (bool result) { - result = item._data & _BITMASK_TYPE == TYPE_NULL; - } - - /// @dev Returns the item's parent. - /// If the item does not have a parent, the result's type will be undefined. - function parent(Item memory item) internal pure returns (Item memory result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x40, result) // Free the default allocation. We've already allocated. - result := and(shr(_BITPOS_SIBLING_OR_PARENT, mload(item)), _BITMASK_POINTER) - if iszero(result) { result := 0x60 } // Reset to the zero pointer. - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* UTILITY FUNCTIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Parses an unsigned integer from a string (in decimal, i.e. base 10). - /// Reverts if `s` is not a valid uint256 string matching the RegEx `^[0-9]+$`, - /// or if the parsed number is too big for a uint256. - function parseUint(string memory s) internal pure returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - let n := mload(s) - let preMulOverflowThres := div(not(0), 10) - for { let i := 0 } 1 {} { - i := add(i, 1) - let digit := sub(and(mload(add(s, i)), 0xff), 48) - let mulOverflowed := gt(result, preMulOverflowThres) - let product := mul(10, result) - result := add(product, digit) - n := mul(n, iszero(or(or(mulOverflowed, lt(result, product)), gt(digit, 9)))) - if iszero(lt(i, n)) { break } - } - if iszero(n) { - mstore(0x00, 0x10182796) // `ParsingFailed()`. - revert(0x1c, 0x04) - } - } - } - - /// @dev Parses a signed integer from a string (in decimal, i.e. base 10). - /// Reverts if `s` is not a valid int256 string matching the RegEx `^[+-]?[0-9]+$`, - /// or if the parsed number is too big for a int256. - function parseInt(string memory s) internal pure returns (int256 result) { - uint256 n = bytes(s).length; - uint256 sign; - uint256 isNegative; - /// @solidity memory-safe-assembly - assembly { - if n { - let c := and(mload(add(s, 1)), 0xff) - isNegative := eq(c, 45) - if or(eq(c, 43), isNegative) { - sign := c - s := add(s, 1) - mstore(s, sub(n, 1)) - } - if iszero(or(sign, lt(sub(c, 48), 10))) { s := 0x60 } - } - } - uint256 x = parseUint(s); - /// @solidity memory-safe-assembly - assembly { - if shr(255, x) { - mstore(0x00, 0x10182796) // `ParsingFailed()`. - revert(0x1c, 0x04) - } - if sign { - mstore(s, sign) - s := sub(s, 1) - mstore(s, n) - } - result := xor(x, mul(xor(x, add(not(x), 1)), isNegative)) - } - } - - /// @dev Parses an unsigned integer from a string (in hexadecimal, i.e. base 16). - /// Reverts if `s` is not a valid uint256 hex string matching the RegEx - /// `^(0[xX])?[0-9a-fA-F]+$`, or if the parsed number is too big for a uint256. - function parseUintFromHex(string memory s) internal pure returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - let n := mload(s) - // Skip two if starts with '0x' or '0X'. - let i := shl(1, and(eq(0x3078, or(shr(240, mload(add(s, 0x20))), 0x20)), gt(n, 1))) - for {} 1 {} { - i := add(i, 1) - let c := - byte( - and(0x1f, shr(and(mload(add(s, i)), 0xff), 0x3e4088843e41bac000000000000)), - 0x3010a071000000b0104040208000c05090d060e0f - ) - n := mul(n, iszero(or(iszero(c), shr(252, result)))) - result := add(shl(4, result), sub(c, 1)) - if iszero(lt(i, n)) { break } - } - if iszero(n) { - mstore(0x00, 0x10182796) // `ParsingFailed()`. - revert(0x1c, 0x04) - } - } - } - - /// @dev Decodes a JSON encoded string. - /// The string MUST be double-quoted, JSON encoded. - /// Reverts if the string is invalid. - /// As you can see, it's pretty complex for a deceptively simple looking task. - function decodeString(string memory s) internal pure returns (string memory result) { - /// @solidity memory-safe-assembly - assembly { - function fail() { - mstore(0x00, 0x10182796) // `ParsingFailed()`. - revert(0x1c, 0x04) - } - - function decodeUnicodeEscapeSequence(pIn_, end_) -> _unicode, _pOut { - _pOut := add(pIn_, 4) - let b_ := iszero(gt(_pOut, end_)) - let t_ := mload(pIn_) // Load the whole word. - for { let i_ := 0 } iszero(eq(i_, 4)) { i_ := add(i_, 1) } { - let c_ := sub(byte(i_, t_), 48) - if iszero(and(shr(c_, 0x7e0000007e03ff), b_)) { fail() } // Not hexadecimal. - c_ := sub(c_, add(mul(gt(c_, 16), 7), shl(5, gt(c_, 48)))) - _unicode := add(shl(4, _unicode), c_) - } - } - - function decodeUnicodeCodePoint(pIn_, end_) -> _unicode, _pOut { - _unicode, _pOut := decodeUnicodeEscapeSequence(pIn_, end_) - if iszero(or(lt(_unicode, 0xd800), gt(_unicode, 0xdbff))) { - let t_ := mload(_pOut) // Load the whole word. - end_ := mul(end_, eq(shr(240, t_), 0x5c75)) // Fail if not starting with '\\u'. - t_, _pOut := decodeUnicodeEscapeSequence(add(_pOut, 2), end_) - _unicode := add(0x10000, add(shl(10, and(0x3ff, _unicode)), and(0x3ff, t_))) - } - } - - function appendCodePointAsUTF8(pIn_, c_) -> _pOut { - if iszero(gt(c_, 0x7f)) { - mstore8(pIn_, c_) - _pOut := add(pIn_, 1) - leave - } - mstore8(0x1f, c_) - mstore8(0x1e, shr(6, c_)) - if iszero(gt(c_, 0x7ff)) { - mstore(pIn_, shl(240, or(0xc080, and(0x1f3f, mload(0x00))))) - _pOut := add(pIn_, 2) - leave - } - mstore8(0x1d, shr(12, c_)) - if iszero(gt(c_, 0xffff)) { - mstore(pIn_, shl(232, or(0xe08080, and(0x0f3f3f, mload(0x00))))) - _pOut := add(pIn_, 3) - leave - } - mstore8(0x1c, shr(18, c_)) - mstore(pIn_, shl(224, or(0xf0808080, and(0x073f3f3f, mload(0x00))))) - _pOut := add(pIn_, shl(2, lt(c_, 0x110000))) - } - - function chr(p_) -> _c { - _c := byte(0, mload(p_)) - } - - let n := mload(s) - let end := add(add(s, n), 0x1f) - if iszero(and(gt(n, 1), eq(0x2222, or(and(0xff00, mload(add(s, 2))), chr(end))))) { - fail() // Fail if not double-quoted. - } - let out := add(mload(0x40), 0x20) - for { let curr := add(s, 0x21) } iszero(eq(curr, end)) {} { - let c := chr(curr) - curr := add(curr, 1) - // Not '\\'. - if iszero(eq(c, 92)) { - // Not '"'. - if iszero(eq(c, 34)) { - mstore8(out, c) - out := add(out, 1) - continue - } - curr := end - } - if iszero(eq(curr, end)) { - let escape := chr(curr) - curr := add(curr, 1) - // '"', '/', '\\'. - if and(shr(escape, 0x100000000000800400000000), 1) { - mstore8(out, escape) - out := add(out, 1) - continue - } - // 'u'. - if eq(escape, 117) { - escape, curr := decodeUnicodeCodePoint(curr, end) - out := appendCodePointAsUTF8(out, escape) - continue - } - // `{'b':'\b', 'f':'\f', 'n':'\n', 'r':'\r', 't':'\t'}`. - escape := byte(sub(escape, 85), 0x080000000c000000000000000a0000000d0009) - if escape { - mstore8(out, escape) - out := add(out, 1) - continue - } - } - fail() - break - } - mstore(out, 0) // Zeroize the last slot. - result := mload(0x40) - mstore(result, sub(out, add(result, 0x20))) // Store the length. - mstore(0x40, add(out, 0x20)) // Allocate the memory. - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PRIVATE HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Performs a query on the input with the given mode. - function _query(bytes32 input, uint256 mode) private pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - function fail() { - mstore(0x00, 0x10182796) // `ParsingFailed()`. - revert(0x1c, 0x04) - } - - function chr(p_) -> _c { - _c := byte(0, mload(p_)) - } - - function skipWhitespace(pIn_, end_) -> _pOut { - for { _pOut := pIn_ } 1 { _pOut := add(_pOut, 1) } { - if iszero(and(shr(chr(_pOut), 0x100002600), 1)) { leave } // Not in ' \n\r\t'. - } - } - - function setP(packed_, bitpos_, p_) -> _packed { - // Perform an out-of-gas revert if `p_` exceeds `_BITMASK_POINTER`. - returndatacopy(returndatasize(), returndatasize(), gt(p_, _BITMASK_POINTER)) - _packed := or(and(not(shl(bitpos_, _BITMASK_POINTER)), packed_), shl(bitpos_, p_)) - } - - function getP(packed_, bitpos_) -> _p { - _p := and(_BITMASK_POINTER, shr(bitpos_, packed_)) - } - - function mallocItem(s_, packed_, pStart_, pCurr_, type_) -> _item { - _item := mload(0x40) - // forgefmt: disable-next-item - packed_ := setP(setP(packed_, _BITPOS_VALUE, sub(pStart_, add(s_, 0x20))), - _BITPOS_VALUE_LENGTH, sub(pCurr_, pStart_)) - mstore(_item, or(packed_, type_)) - mstore(0x40, add(_item, 0x20)) // Allocate memory. - } - - function parseValue(s_, sibling_, pIn_, end_) -> _item, _pOut { - let packed_ := setP(mload(0x00), _BITPOS_SIBLING_OR_PARENT, sibling_) - _pOut := skipWhitespace(pIn_, end_) - if iszero(lt(_pOut, end_)) { leave } - for { let c_ := chr(_pOut) } 1 {} { - // If starts with '"'. - if eq(c_, 34) { - let pStart_ := _pOut - _pOut := parseStringSub(s_, packed_, _pOut, end_) - _item := mallocItem(s_, packed_, pStart_, _pOut, TYPE_STRING) - break - } - // If starts with '['. - if eq(c_, 91) { - _item, _pOut := parseArray(s_, packed_, _pOut, end_) - break - } - // If starts with '{'. - if eq(c_, 123) { - _item, _pOut := parseObject(s_, packed_, _pOut, end_) - break - } - // If starts with any in '0123456789-'. - if and(shr(c_, shl(45, 0x1ff9)), 1) { - _item, _pOut := parseNumber(s_, packed_, _pOut, end_) - break - } - if iszero(gt(add(_pOut, 4), end_)) { - let pStart_ := _pOut - let w_ := shr(224, mload(_pOut)) - // 'true' in hex format. - if eq(w_, 0x74727565) { - _pOut := add(_pOut, 4) - _item := mallocItem(s_, packed_, pStart_, _pOut, TYPE_BOOLEAN) - break - } - // 'null' in hex format. - if eq(w_, 0x6e756c6c) { - _pOut := add(_pOut, 4) - _item := mallocItem(s_, packed_, pStart_, _pOut, TYPE_NULL) - break - } - } - if iszero(gt(add(_pOut, 5), end_)) { - let pStart_ := _pOut - let w_ := shr(216, mload(_pOut)) - // 'false' in hex format. - if eq(w_, 0x66616c7365) { - _pOut := add(_pOut, 5) - _item := mallocItem(s_, packed_, pStart_, _pOut, TYPE_BOOLEAN) - break - } - } - fail() - break - } - _pOut := skipWhitespace(_pOut, end_) - } - - function parseArray(s_, packed_, pIn_, end_) -> _item, _pOut { - let j_ := 0 - for { _pOut := add(pIn_, 1) } 1 { _pOut := add(_pOut, 1) } { - if iszero(lt(_pOut, end_)) { fail() } - if iszero(_item) { - _pOut := skipWhitespace(_pOut, end_) - if eq(chr(_pOut), 93) { break } // ']'. - } - _item, _pOut := parseValue(s_, _item, _pOut, end_) - if _item { - // forgefmt: disable-next-item - mstore(_item, setP(or(_PARENT_IS_ARRAY, mload(_item)), - _BITPOS_KEY, j_)) - j_ := add(j_, 1) - let c_ := chr(_pOut) - if eq(c_, 93) { break } // ']'. - if eq(c_, 44) { continue } // ','. - } - _pOut := end_ - } - _pOut := add(_pOut, 1) - packed_ := setP(packed_, _BITPOS_CHILD, _item) - _item := mallocItem(s_, packed_, pIn_, _pOut, TYPE_ARRAY) - } - - function parseObject(s_, packed_, pIn_, end_) -> _item, _pOut { - for { _pOut := add(pIn_, 1) } 1 { _pOut := add(_pOut, 1) } { - if iszero(lt(_pOut, end_)) { fail() } - if iszero(_item) { - _pOut := skipWhitespace(_pOut, end_) - if eq(chr(_pOut), 125) { break } // '}'. - } - _pOut := skipWhitespace(_pOut, end_) - let pKeyStart_ := _pOut - let pKeyEnd_ := parseStringSub(s_, _item, _pOut, end_) - _pOut := skipWhitespace(pKeyEnd_, end_) - // If ':'. - if eq(chr(_pOut), 58) { - _item, _pOut := parseValue(s_, _item, add(_pOut, 1), end_) - if _item { - // forgefmt: disable-next-item - mstore(_item, setP(setP(or(_PARENT_IS_OBJECT, mload(_item)), - _BITPOS_KEY_LENGTH, sub(pKeyEnd_, pKeyStart_)), - _BITPOS_KEY, sub(pKeyStart_, add(s_, 0x20)))) - let c_ := chr(_pOut) - if eq(c_, 125) { break } // '}'. - if eq(c_, 44) { continue } // ','. - } - } - _pOut := end_ - } - _pOut := add(_pOut, 1) - packed_ := setP(packed_, _BITPOS_CHILD, _item) - _item := mallocItem(s_, packed_, pIn_, _pOut, TYPE_OBJECT) - } - - function checkStringU(p_, o_) { - // If not in '0123456789abcdefABCDEF', revert. - if iszero(and(shr(sub(chr(add(p_, o_)), 48), 0x7e0000007e03ff), 1)) { fail() } - if iszero(eq(o_, 5)) { checkStringU(p_, add(o_, 1)) } - } - - function parseStringSub(s_, packed_, pIn_, end_) -> _pOut { - if iszero(lt(pIn_, end_)) { fail() } - for { _pOut := add(pIn_, 1) } 1 {} { - let c_ := chr(_pOut) - if eq(c_, 34) { break } // '"'. - // Not '\'. - if iszero(eq(c_, 92)) { - _pOut := add(_pOut, 1) - continue - } - c_ := chr(add(_pOut, 1)) - // '"', '\', '//', 'b', 'f', 'n', 'r', 't'. - if and(shr(sub(c_, 34), 0x510110400000000002001), 1) { - _pOut := add(_pOut, 2) - continue - } - // 'u'. - if eq(c_, 117) { - checkStringU(_pOut, 2) - _pOut := add(_pOut, 6) - continue - } - _pOut := end_ - break - } - if iszero(lt(_pOut, end_)) { fail() } - _pOut := add(_pOut, 1) - } - - function skip0To9s(pIn_, end_, atLeastOne_) -> _pOut { - for { _pOut := pIn_ } 1 { _pOut := add(_pOut, 1) } { - if iszero(lt(sub(chr(_pOut), 48), 10)) { break } // Not '0'..'9'. - } - if and(atLeastOne_, eq(pIn_, _pOut)) { fail() } - } - - function parseNumber(s_, packed_, pIn_, end_) -> _item, _pOut { - _pOut := pIn_ - if eq(chr(_pOut), 45) { _pOut := add(_pOut, 1) } // '-'. - if iszero(lt(sub(chr(_pOut), 48), 10)) { fail() } // Not '0'..'9'. - let c_ := chr(_pOut) - _pOut := add(_pOut, 1) - if iszero(eq(c_, 48)) { _pOut := skip0To9s(_pOut, end_, 0) } // Not '0'. - if eq(chr(_pOut), 46) { _pOut := skip0To9s(add(_pOut, 1), end_, 1) } // '.'. - let t_ := mload(_pOut) - // 'E', 'e'. - if eq(or(0x20, byte(0, t_)), 101) { - // forgefmt: disable-next-item - _pOut := skip0To9s(add(byte(sub(byte(1, t_), 14), 0x010001), // '+', '-'. - add(_pOut, 1)), end_, 1) - } - _item := mallocItem(s_, packed_, pIn_, _pOut, TYPE_NUMBER) - } - - function copyStr(s_, offset_, len_) -> _sCopy { - _sCopy := mload(0x40) - s_ := add(s_, offset_) - let w_ := not(0x1f) - for { let i_ := and(add(len_, 0x1f), w_) } 1 {} { - mstore(add(_sCopy, i_), mload(add(s_, i_))) - i_ := add(i_, w_) // `sub(i_, 0x20)`. - if iszero(i_) { break } - } - mstore(_sCopy, len_) // Copy the length. - mstore(add(add(_sCopy, 0x20), len_), 0) // Zeroize the last slot. - mstore(0x40, add(add(_sCopy, 0x40), len_)) // Allocate memory. - } - - function value(item_) -> _value { - let packed_ := mload(item_) - _value := getP(packed_, _BITPOS_VALUE) // The offset in the string. - if iszero(and(_VALUE_INITED, packed_)) { - let s_ := getP(packed_, _BITPOS_STRING) - _value := copyStr(s_, _value, getP(packed_, _BITPOS_VALUE_LENGTH)) - packed_ := setP(packed_, _BITPOS_VALUE, _value) - mstore(s_, or(_VALUE_INITED, packed_)) - } - } - - function children(item_) -> _arr { - _arr := 0x60 // Initialize to the zero pointer. - let packed_ := mload(item_) - for {} iszero(gt(and(_BITMASK_TYPE, packed_), TYPE_OBJECT)) {} { - if or(iszero(packed_), iszero(item_)) { break } - if and(packed_, _CHILDREN_INITED) { - _arr := getP(packed_, _BITPOS_CHILD) - break - } - _arr := mload(0x40) - let o_ := add(_arr, 0x20) - for { let h_ := getP(packed_, _BITPOS_CHILD) } h_ {} { - mstore(o_, h_) - let q_ := mload(h_) - let y_ := getP(q_, _BITPOS_SIBLING_OR_PARENT) - mstore(h_, setP(q_, _BITPOS_SIBLING_OR_PARENT, item_)) - h_ := y_ - o_ := add(o_, 0x20) - } - let w_ := not(0x1f) - let n_ := add(w_, sub(o_, _arr)) - mstore(_arr, shr(5, n_)) - mstore(0x40, o_) // Allocate memory. - packed_ := setP(packed_, _BITPOS_CHILD, _arr) - mstore(item_, or(_CHILDREN_INITED, packed_)) - // Reverse the array. - if iszero(lt(n_, 0x40)) { - let lo_ := add(_arr, 0x20) - let hi_ := add(_arr, n_) - for {} 1 {} { - let temp_ := mload(lo_) - mstore(lo_, mload(hi_)) - mstore(hi_, temp_) - hi_ := add(hi_, w_) - lo_ := add(lo_, 0x20) - if iszero(lt(lo_, hi_)) { break } - } - } - break - } - } - - function getStr(item_, bitpos_, bitposLength_, bitmaskInited_) -> _result { - _result := 0x60 // Initialize to the zero pointer. - let packed_ := mload(item_) - if or(iszero(item_), iszero(packed_)) { leave } - _result := getP(packed_, bitpos_) - if iszero(and(bitmaskInited_, packed_)) { - let s_ := getP(packed_, _BITPOS_STRING) - _result := copyStr(s_, _result, getP(packed_, bitposLength_)) - mstore(item_, or(bitmaskInited_, setP(packed_, bitpos_, _result))) - } - } - - switch mode - // Get value. - case 0 { result := getStr(input, _BITPOS_VALUE, _BITPOS_VALUE_LENGTH, _VALUE_INITED) } - // Get key. - case 1 { result := getStr(input, _BITPOS_KEY, _BITPOS_KEY_LENGTH, _KEY_INITED) } - // Get children. - case 3 { result := children(input) } - // Parse. - default { - let p := add(input, 0x20) - let e := add(p, mload(input)) - if iszero(eq(p, e)) { - let c := chr(e) - mstore8(e, 34) // Place a '"' at the end to speed up parsing. - // The `34 << 248` makes `mallocItem` preserve '"' at the end. - mstore(0x00, setP(shl(248, 34), _BITPOS_STRING, input)) - result, p := parseValue(input, 0, p, e) - mstore8(e, c) // Restore the original char at the end. - } - if or(lt(p, e), iszero(result)) { fail() } - } - } - } - - /// @dev Casts the input to a bytes32. - function _toInput(string memory input) private pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - result := input - } - } - - /// @dev Casts the input to a bytes32. - function _toInput(Item memory input) private pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - result := input - } - } -} diff --git a/lib/solady/src/utils/LibBit.sol b/lib/solady/src/utils/LibBit.sol deleted file mode 100644 index 0c7dba5..0000000 --- a/lib/solady/src/utils/LibBit.sol +++ /dev/null @@ -1,184 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for bit twiddling and boolean operations. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/LibBit.sol) -/// @author Inspired by (https://graphics.stanford.edu/~seander/bithacks.html) -library LibBit { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* BIT TWIDDLING OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Find last set. - /// Returns the index of the most significant bit of `x`, - /// counting from the least significant bit position. - /// If `x` is zero, returns 256. - function fls(uint256 x) internal pure returns (uint256 r) { - /// @solidity memory-safe-assembly - assembly { - r := or(shl(8, iszero(x)), shl(7, lt(0xffffffffffffffffffffffffffffffff, x))) - r := or(r, shl(6, lt(0xffffffffffffffff, shr(r, x)))) - r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) - r := or(r, shl(4, lt(0xffff, shr(r, x)))) - r := or(r, shl(3, lt(0xff, shr(r, x)))) - // forgefmt: disable-next-item - r := or(r, byte(and(0x1f, shr(shr(r, x), 0x8421084210842108cc6318c6db6d54be)), - 0x0706060506020504060203020504030106050205030304010505030400000000)) - } - } - - /// @dev Count leading zeros. - /// Returns the number of zeros preceding the most significant one bit. - /// If `x` is zero, returns 256. - function clz(uint256 x) internal pure returns (uint256 r) { - /// @solidity memory-safe-assembly - assembly { - r := shl(7, lt(0xffffffffffffffffffffffffffffffff, x)) - r := or(r, shl(6, lt(0xffffffffffffffff, shr(r, x)))) - r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) - r := or(r, shl(4, lt(0xffff, shr(r, x)))) - r := or(r, shl(3, lt(0xff, shr(r, x)))) - // forgefmt: disable-next-item - r := add(xor(r, byte(and(0x1f, shr(shr(r, x), 0x8421084210842108cc6318c6db6d54be)), - 0xf8f9f9faf9fdfafbf9fdfcfdfafbfcfef9fafdfafcfcfbfefafafcfbffffffff)), iszero(x)) - } - } - - /// @dev Find first set. - /// Returns the index of the least significant bit of `x`, - /// counting from the least significant bit position. - /// If `x` is zero, returns 256. - /// Equivalent to `ctz` (count trailing zeros), which gives - /// the number of zeros following the least significant one bit. - function ffs(uint256 x) internal pure returns (uint256 r) { - /// @solidity memory-safe-assembly - assembly { - // Isolate the least significant bit. - let b := and(x, add(not(x), 1)) - - r := or(shl(8, iszero(x)), shl(7, lt(0xffffffffffffffffffffffffffffffff, b))) - r := or(r, shl(6, lt(0xffffffffffffffff, shr(r, b)))) - r := or(r, shl(5, lt(0xffffffff, shr(r, b)))) - - // For the remaining 32 bits, use a De Bruijn lookup. - // forgefmt: disable-next-item - r := or(r, byte(and(div(0xd76453e0, shr(r, b)), 0x1f), - 0x001f0d1e100c1d070f090b19131c1706010e11080a1a141802121b1503160405)) - } - } - - /// @dev Returns the number of set bits in `x`. - function popCount(uint256 x) internal pure returns (uint256 c) { - /// @solidity memory-safe-assembly - assembly { - let max := not(0) - let isMax := eq(x, max) - x := sub(x, and(shr(1, x), div(max, 3))) - x := add(and(x, div(max, 5)), and(shr(2, x), div(max, 5))) - x := and(add(x, shr(4, x)), div(max, 17)) - c := or(shl(8, isMax), shr(248, mul(x, div(max, 255)))) - } - } - - /// @dev Returns whether `x` is a power of 2. - function isPo2(uint256 x) internal pure returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - // Equivalent to `x && !(x & (x - 1))`. - result := iszero(add(and(x, sub(x, 1)), iszero(x))) - } - } - - /// @dev Returns `x` reversed at the bit level. - function reverseBits(uint256 x) internal pure returns (uint256 r) { - /// @solidity memory-safe-assembly - assembly { - // Computing masks on-the-fly reduces bytecode size by about 500 bytes. - let m := not(0) - r := x - for { let s := 128 } 1 {} { - m := xor(m, shl(s, m)) - r := or(and(shr(s, r), m), and(shl(s, r), not(m))) - s := shr(1, s) - if iszero(s) { break } - } - } - } - - /// @dev Returns `x` reversed at the byte level. - function reverseBytes(uint256 x) internal pure returns (uint256 r) { - /// @solidity memory-safe-assembly - assembly { - // Computing masks on-the-fly reduces bytecode size by about 200 bytes. - let m := not(0) - r := x - for { let s := 128 } 1 {} { - m := xor(m, shl(s, m)) - r := or(and(shr(s, r), m), and(shl(s, r), not(m))) - s := shr(1, s) - if eq(s, 4) { break } - } - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* BOOLEAN OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // A Solidity bool on the stack or memory is represented as a 256-bit word. - // Non-zero values are true, zero is false. - // A clean bool is either 0 (false) or 1 (true) under the hood. - // Usually, if not always, the bool result of a regular Solidity expression, - // or the argument of a public/external function will be a clean bool. - // You can usually use the raw variants for more performance. - // If uncertain, test (best with exact compiler settings). - // Or use the non-raw variants (compiler can sometimes optimize out the double `iszero`s). - - /// @dev Returns `x & y`. Inputs must be clean. - function rawAnd(bool x, bool y) internal pure returns (bool z) { - /// @solidity memory-safe-assembly - assembly { - z := and(x, y) - } - } - - /// @dev Returns `x & y`. - function and(bool x, bool y) internal pure returns (bool z) { - /// @solidity memory-safe-assembly - assembly { - z := and(iszero(iszero(x)), iszero(iszero(y))) - } - } - - /// @dev Returns `x | y`. Inputs must be clean. - function rawOr(bool x, bool y) internal pure returns (bool z) { - /// @solidity memory-safe-assembly - assembly { - z := or(x, y) - } - } - - /// @dev Returns `x | y`. - function or(bool x, bool y) internal pure returns (bool z) { - /// @solidity memory-safe-assembly - assembly { - z := or(iszero(iszero(x)), iszero(iszero(y))) - } - } - - /// @dev Returns 1 if `b` is true, else 0. Input must be clean. - function rawToUint(bool b) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := b - } - } - - /// @dev Returns 1 if `b` is true, else 0. - function toUint(bool b) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := iszero(iszero(b)) - } - } -} diff --git a/lib/solady/src/utils/LibBitmap.sol b/lib/solady/src/utils/LibBitmap.sol deleted file mode 100644 index a333b5b..0000000 --- a/lib/solady/src/utils/LibBitmap.sol +++ /dev/null @@ -1,199 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {LibBit} from "./LibBit.sol"; - -/// @notice Library for storage of packed unsigned booleans. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/LibBitmap.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/LibBitmap.sol) -/// @author Modified from Solidity-Bits (https://github.com/estarriolvetch/solidity-bits/blob/main/contracts/BitMaps.sol) -library LibBitmap { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The constant returned when a bitmap scan does not find a result. - uint256 internal constant NOT_FOUND = type(uint256).max; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STRUCTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev A bitmap in storage. - struct Bitmap { - mapping(uint256 => uint256) map; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the boolean value of the bit at `index` in `bitmap`. - function get(Bitmap storage bitmap, uint256 index) internal view returns (bool isSet) { - // It is better to set `isSet` to either 0 or 1, than zero vs non-zero. - // Both cost the same amount of gas, but the former allows the returned value - // to be reused without cleaning the upper bits. - uint256 b = (bitmap.map[index >> 8] >> (index & 0xff)) & 1; - /// @solidity memory-safe-assembly - assembly { - isSet := b - } - } - - /// @dev Updates the bit at `index` in `bitmap` to true. - function set(Bitmap storage bitmap, uint256 index) internal { - bitmap.map[index >> 8] |= (1 << (index & 0xff)); - } - - /// @dev Updates the bit at `index` in `bitmap` to false. - function unset(Bitmap storage bitmap, uint256 index) internal { - bitmap.map[index >> 8] &= ~(1 << (index & 0xff)); - } - - /// @dev Flips the bit at `index` in `bitmap`. - /// Returns the boolean result of the flipped bit. - function toggle(Bitmap storage bitmap, uint256 index) internal returns (bool newIsSet) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, bitmap.slot) - mstore(0x00, shr(8, index)) - let storageSlot := keccak256(0x00, 0x40) - let shift := and(index, 0xff) - let storageValue := xor(sload(storageSlot), shl(shift, 1)) - // It makes sense to return the `newIsSet`, - // as it allow us to skip an additional warm `sload`, - // and it costs minimal gas (about 15), - // which may be optimized away if the returned value is unused. - newIsSet := and(1, shr(shift, storageValue)) - sstore(storageSlot, storageValue) - } - } - - /// @dev Updates the bit at `index` in `bitmap` to `shouldSet`. - function setTo(Bitmap storage bitmap, uint256 index, bool shouldSet) internal { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, bitmap.slot) - mstore(0x00, shr(8, index)) - let storageSlot := keccak256(0x00, 0x40) - let storageValue := sload(storageSlot) - let shift := and(index, 0xff) - sstore( - storageSlot, - // Unsets the bit at `shift` via `and`, then sets its new value via `or`. - or(and(storageValue, not(shl(shift, 1))), shl(shift, iszero(iszero(shouldSet)))) - ) - } - } - - /// @dev Consecutively sets `amount` of bits starting from the bit at `start`. - function setBatch(Bitmap storage bitmap, uint256 start, uint256 amount) internal { - /// @solidity memory-safe-assembly - assembly { - let max := not(0) - let shift := and(start, 0xff) - mstore(0x20, bitmap.slot) - mstore(0x00, shr(8, start)) - if iszero(lt(add(shift, amount), 257)) { - let storageSlot := keccak256(0x00, 0x40) - sstore(storageSlot, or(sload(storageSlot), shl(shift, max))) - let bucket := add(mload(0x00), 1) - let bucketEnd := add(mload(0x00), shr(8, add(amount, shift))) - amount := and(add(amount, shift), 0xff) - shift := 0 - for {} iszero(eq(bucket, bucketEnd)) { bucket := add(bucket, 1) } { - mstore(0x00, bucket) - sstore(keccak256(0x00, 0x40), max) - } - mstore(0x00, bucket) - } - let storageSlot := keccak256(0x00, 0x40) - sstore(storageSlot, or(sload(storageSlot), shl(shift, shr(sub(256, amount), max)))) - } - } - - /// @dev Consecutively unsets `amount` of bits starting from the bit at `start`. - function unsetBatch(Bitmap storage bitmap, uint256 start, uint256 amount) internal { - /// @solidity memory-safe-assembly - assembly { - let shift := and(start, 0xff) - mstore(0x20, bitmap.slot) - mstore(0x00, shr(8, start)) - if iszero(lt(add(shift, amount), 257)) { - let storageSlot := keccak256(0x00, 0x40) - sstore(storageSlot, and(sload(storageSlot), not(shl(shift, not(0))))) - let bucket := add(mload(0x00), 1) - let bucketEnd := add(mload(0x00), shr(8, add(amount, shift))) - amount := and(add(amount, shift), 0xff) - shift := 0 - for {} iszero(eq(bucket, bucketEnd)) { bucket := add(bucket, 1) } { - mstore(0x00, bucket) - sstore(keccak256(0x00, 0x40), 0) - } - mstore(0x00, bucket) - } - let storageSlot := keccak256(0x00, 0x40) - sstore( - storageSlot, and(sload(storageSlot), not(shl(shift, shr(sub(256, amount), not(0))))) - ) - } - } - - /// @dev Returns number of set bits within a range by - /// scanning `amount` of bits starting from the bit at `start`. - function popCount(Bitmap storage bitmap, uint256 start, uint256 amount) - internal - view - returns (uint256 count) - { - unchecked { - uint256 bucket = start >> 8; - uint256 shift = start & 0xff; - if (!(amount + shift < 257)) { - count = LibBit.popCount(bitmap.map[bucket] >> shift); - uint256 bucketEnd = bucket + ((amount + shift) >> 8); - amount = (amount + shift) & 0xff; - shift = 0; - for (++bucket; bucket != bucketEnd; ++bucket) { - count += LibBit.popCount(bitmap.map[bucket]); - } - } - count += LibBit.popCount((bitmap.map[bucket] >> shift) << (256 - amount)); - } - } - - /// @dev Returns the index of the most significant set bit before the bit at `before`. - /// If no set bit is found, returns `NOT_FOUND`. - function findLastSet(Bitmap storage bitmap, uint256 before) - internal - view - returns (uint256 setBitIndex) - { - uint256 bucket; - uint256 bucketBits; - /// @solidity memory-safe-assembly - assembly { - setBitIndex := not(0) - bucket := shr(8, before) - mstore(0x00, bucket) - mstore(0x20, bitmap.slot) - let offset := and(0xff, not(before)) // `256 - (255 & before) - 1`. - bucketBits := shr(offset, shl(offset, sload(keccak256(0x00, 0x40)))) - if iszero(or(bucketBits, iszero(bucket))) { - for {} 1 {} { - bucket := add(bucket, setBitIndex) // `sub(bucket, 1)`. - mstore(0x00, bucket) - bucketBits := sload(keccak256(0x00, 0x40)) - if or(bucketBits, iszero(bucket)) { break } - } - } - } - if (bucketBits != 0) { - setBitIndex = (bucket << 8) | LibBit.fls(bucketBits); - /// @solidity memory-safe-assembly - assembly { - setBitIndex := or(setBitIndex, sub(0, gt(setBitIndex, before))) - } - } - } -} diff --git a/lib/solady/src/utils/LibClone.sol b/lib/solady/src/utils/LibClone.sol deleted file mode 100644 index abf3fbc..0000000 --- a/lib/solady/src/utils/LibClone.sol +++ /dev/null @@ -1,836 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Minimal proxy library. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/LibClone.sol) -/// @author Minimal proxy by 0age (https://github.com/0age) -/// @author Clones with immutable args by wighawag, zefram.eth, Saw-mon & Natalie -/// (https://github.com/Saw-mon-and-Natalie/clones-with-immutable-args) -/// @author Minimal ERC1967 proxy by jtriley-eth (https://github.com/jtriley-eth/minimum-viable-proxy) -/// -/// @dev Minimal proxy: -/// Although the sw0nt pattern saves 5 gas over the erc-1167 pattern during runtime, -/// it is not supported out-of-the-box on Etherscan. Hence, we choose to use the 0age pattern, -/// which saves 4 gas over the erc-1167 pattern during runtime, and has the smallest bytecode. -/// -/// @dev Minimal proxy (PUSH0 variant): -/// This is a new minimal proxy that uses the PUSH0 opcode introduced during Shanghai. -/// It is optimized first for minimal runtime gas, then for minimal bytecode. -/// The PUSH0 clone functions are intentionally postfixed with a jarring "_PUSH0" as -/// many EVM chains may not support the PUSH0 opcode in the early months after Shanghai. -/// Please use with caution. -/// -/// @dev Clones with immutable args (CWIA): -/// The implementation of CWIA here implements a `receive()` method that emits the -/// `ReceiveETH(uint256)` event. This skips the `DELEGATECALL` when there is no calldata, -/// enabling us to accept hard gas-capped `sends` & `transfers` for maximum backwards -/// composability. The minimal proxy implementation does not offer this feature. -/// -/// @dev Minimal ERC1967 proxy: -/// An minimal ERC1967 proxy, intended to be upgraded with UUPS. -/// This is NOT the same as ERC1967Factory's transparent proxy, which includes admin logic. -library LibClone { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Unable to deploy the clone. - error DeploymentFailed(); - - /// @dev The salt must start with either the zero address or `by`. - error SaltDoesNotStartWith(); - - /// @dev The ETH transfer has failed. - error ETHTransferFailed(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* MINIMAL PROXY OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Deploys a clone of `implementation`. - function clone(address implementation) internal returns (address instance) { - instance = clone(0, implementation); - } - - /// @dev Deploys a clone of `implementation`. - function clone(uint256 value, address implementation) internal returns (address instance) { - /// @solidity memory-safe-assembly - assembly { - /** - * --------------------------------------------------------------------------+ - * CREATION (9 bytes) | - * --------------------------------------------------------------------------| - * Opcode | Mnemonic | Stack | Memory | - * --------------------------------------------------------------------------| - * 60 runSize | PUSH1 runSize | r | | - * 3d | RETURNDATASIZE | 0 r | | - * 81 | DUP2 | r 0 r | | - * 60 offset | PUSH1 offset | o r 0 r | | - * 3d | RETURNDATASIZE | 0 o r 0 r | | - * 39 | CODECOPY | 0 r | [0..runSize): runtime code | - * f3 | RETURN | | [0..runSize): runtime code | - * --------------------------------------------------------------------------| - * RUNTIME (44 bytes) | - * --------------------------------------------------------------------------| - * Opcode | Mnemonic | Stack | Memory | - * --------------------------------------------------------------------------| - * | - * ::: keep some values in stack ::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | 0 | | - * 3d | RETURNDATASIZE | 0 0 | | - * 3d | RETURNDATASIZE | 0 0 0 | | - * 3d | RETURNDATASIZE | 0 0 0 0 | | - * | - * ::: copy calldata to memory ::::::::::::::::::::::::::::::::::::::::::::: | - * 36 | CALLDATASIZE | cds 0 0 0 0 | | - * 3d | RETURNDATASIZE | 0 cds 0 0 0 0 | | - * 3d | RETURNDATASIZE | 0 0 cds 0 0 0 0 | | - * 37 | CALLDATACOPY | 0 0 0 0 | [0..cds): calldata | - * | - * ::: delegate call to the implementation contract :::::::::::::::::::::::: | - * 36 | CALLDATASIZE | cds 0 0 0 0 | [0..cds): calldata | - * 3d | RETURNDATASIZE | 0 cds 0 0 0 0 | [0..cds): calldata | - * 73 addr | PUSH20 addr | addr 0 cds 0 0 0 0 | [0..cds): calldata | - * 5a | GAS | gas addr 0 cds 0 0 0 0 | [0..cds): calldata | - * f4 | DELEGATECALL | success 0 0 | [0..cds): calldata | - * | - * ::: copy return data to memory :::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | rds success 0 0 | [0..cds): calldata | - * 3d | RETURNDATASIZE | rds rds success 0 0 | [0..cds): calldata | - * 93 | SWAP4 | 0 rds success 0 rds | [0..cds): calldata | - * 80 | DUP1 | 0 0 rds success 0 rds | [0..cds): calldata | - * 3e | RETURNDATACOPY | success 0 rds | [0..rds): returndata | - * | - * 60 0x2a | PUSH1 0x2a | 0x2a success 0 rds | [0..rds): returndata | - * 57 | JUMPI | 0 rds | [0..rds): returndata | - * | - * ::: revert :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * fd | REVERT | | [0..rds): returndata | - * | - * ::: return :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 5b | JUMPDEST | 0 rds | [0..rds): returndata | - * f3 | RETURN | | [0..rds): returndata | - * --------------------------------------------------------------------------+ - */ - - mstore(0x21, 0x5af43d3d93803e602a57fd5bf3) - mstore(0x14, implementation) - mstore(0x00, 0x602c3d8160093d39f33d3d3d3d363d3d37363d73) - instance := create(value, 0x0c, 0x35) - if iszero(instance) { - mstore(0x00, 0x30116425) // `DeploymentFailed()`. - revert(0x1c, 0x04) - } - mstore(0x21, 0) // Restore the overwritten part of the free memory pointer. - } - } - - /// @dev Deploys a deterministic clone of `implementation` with `salt`. - function cloneDeterministic(address implementation, bytes32 salt) - internal - returns (address instance) - { - instance = cloneDeterministic(0, implementation, salt); - } - - /// @dev Deploys a deterministic clone of `implementation` with `salt`. - function cloneDeterministic(uint256 value, address implementation, bytes32 salt) - internal - returns (address instance) - { - /// @solidity memory-safe-assembly - assembly { - mstore(0x21, 0x5af43d3d93803e602a57fd5bf3) - mstore(0x14, implementation) - mstore(0x00, 0x602c3d8160093d39f33d3d3d3d363d3d37363d73) - instance := create2(value, 0x0c, 0x35, salt) - if iszero(instance) { - mstore(0x00, 0x30116425) // `DeploymentFailed()`. - revert(0x1c, 0x04) - } - mstore(0x21, 0) // Restore the overwritten part of the free memory pointer. - } - } - - /// @dev Returns the initialization code hash of the clone of `implementation`. - /// Used for mining vanity addresses with create2crunch. - function initCodeHash(address implementation) internal pure returns (bytes32 hash) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x21, 0x5af43d3d93803e602a57fd5bf3) - mstore(0x14, implementation) - mstore(0x00, 0x602c3d8160093d39f33d3d3d3d363d3d37363d73) - hash := keccak256(0x0c, 0x35) - mstore(0x21, 0) // Restore the overwritten part of the free memory pointer. - } - } - - /// @dev Returns the address of the deterministic clone of `implementation`, - /// with `salt` by `deployer`. - /// Note: The returned result has dirty upper 96 bits. Please clean if used in assembly. - function predictDeterministicAddress(address implementation, bytes32 salt, address deployer) - internal - pure - returns (address predicted) - { - bytes32 hash = initCodeHash(implementation); - predicted = predictDeterministicAddress(hash, salt, deployer); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* MINIMAL PROXY OPERATIONS (PUSH0 VARIANT) */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Deploys a PUSH0 clone of `implementation`. - function clone_PUSH0(address implementation) internal returns (address instance) { - instance = clone_PUSH0(0, implementation); - } - - /// @dev Deploys a PUSH0 clone of `implementation`. - function clone_PUSH0(uint256 value, address implementation) - internal - returns (address instance) - { - /// @solidity memory-safe-assembly - assembly { - /** - * --------------------------------------------------------------------------+ - * CREATION (9 bytes) | - * --------------------------------------------------------------------------| - * Opcode | Mnemonic | Stack | Memory | - * --------------------------------------------------------------------------| - * 60 runSize | PUSH1 runSize | r | | - * 5f | PUSH0 | 0 r | | - * 81 | DUP2 | r 0 r | | - * 60 offset | PUSH1 offset | o r 0 r | | - * 5f | PUSH0 | 0 o r 0 r | | - * 39 | CODECOPY | 0 r | [0..runSize): runtime code | - * f3 | RETURN | | [0..runSize): runtime code | - * --------------------------------------------------------------------------| - * RUNTIME (45 bytes) | - * --------------------------------------------------------------------------| - * Opcode | Mnemonic | Stack | Memory | - * --------------------------------------------------------------------------| - * | - * ::: keep some values in stack ::::::::::::::::::::::::::::::::::::::::::: | - * 5f | PUSH0 | 0 | | - * 5f | PUSH0 | 0 0 | | - * | - * ::: copy calldata to memory ::::::::::::::::::::::::::::::::::::::::::::: | - * 36 | CALLDATASIZE | cds 0 0 | | - * 5f | PUSH0 | 0 cds 0 0 | | - * 5f | PUSH0 | 0 0 cds 0 0 | | - * 37 | CALLDATACOPY | 0 0 | [0..cds): calldata | - * | - * ::: delegate call to the implementation contract :::::::::::::::::::::::: | - * 36 | CALLDATASIZE | cds 0 0 | [0..cds): calldata | - * 5f | PUSH0 | 0 cds 0 0 | [0..cds): calldata | - * 73 addr | PUSH20 addr | addr 0 cds 0 0 | [0..cds): calldata | - * 5a | GAS | gas addr 0 cds 0 0 | [0..cds): calldata | - * f4 | DELEGATECALL | success | [0..cds): calldata | - * | - * ::: copy return data to memory :::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | rds success | [0..cds): calldata | - * 5f | PUSH0 | 0 rds success | [0..cds): calldata | - * 5f | PUSH0 | 0 0 rds success | [0..cds): calldata | - * 3e | RETURNDATACOPY | success | [0..rds): returndata | - * | - * 60 0x29 | PUSH1 0x29 | 0x29 success | [0..rds): returndata | - * 57 | JUMPI | | [0..rds): returndata | - * | - * ::: revert :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | rds | [0..rds): returndata | - * 5f | PUSH0 | 0 rds | [0..rds): returndata | - * fd | REVERT | | [0..rds): returndata | - * | - * ::: return :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 5b | JUMPDEST | | [0..rds): returndata | - * 3d | RETURNDATASIZE | rds | [0..rds): returndata | - * 5f | PUSH0 | 0 rds | [0..rds): returndata | - * f3 | RETURN | | [0..rds): returndata | - * --------------------------------------------------------------------------+ - */ - - mstore(0x24, 0x5af43d5f5f3e6029573d5ffd5b3d5ff3) // 16 - mstore(0x14, implementation) // 20 - mstore(0x00, 0x602d5f8160095f39f35f5f365f5f37365f73) // 9 + 9 - instance := create(value, 0x0e, 0x36) - if iszero(instance) { - mstore(0x00, 0x30116425) // `DeploymentFailed()`. - revert(0x1c, 0x04) - } - mstore(0x24, 0) // Restore the overwritten part of the free memory pointer. - } - } - - /// @dev Deploys a deterministic PUSH0 clone of `implementation` with `salt`. - function cloneDeterministic_PUSH0(address implementation, bytes32 salt) - internal - returns (address instance) - { - instance = cloneDeterministic_PUSH0(0, implementation, salt); - } - - /// @dev Deploys a deterministic PUSH0 clone of `implementation` with `salt`. - function cloneDeterministic_PUSH0(uint256 value, address implementation, bytes32 salt) - internal - returns (address instance) - { - /// @solidity memory-safe-assembly - assembly { - mstore(0x24, 0x5af43d5f5f3e6029573d5ffd5b3d5ff3) // 16 - mstore(0x14, implementation) // 20 - mstore(0x00, 0x602d5f8160095f39f35f5f365f5f37365f73) // 9 + 9 - instance := create2(value, 0x0e, 0x36, salt) - if iszero(instance) { - mstore(0x00, 0x30116425) // `DeploymentFailed()`. - revert(0x1c, 0x04) - } - mstore(0x24, 0) // Restore the overwritten part of the free memory pointer. - } - } - - /// @dev Returns the initialization code hash of the PUSH0 clone of `implementation`. - /// Used for mining vanity addresses with create2crunch. - function initCodeHash_PUSH0(address implementation) internal pure returns (bytes32 hash) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x24, 0x5af43d5f5f3e6029573d5ffd5b3d5ff3) // 16 - mstore(0x14, implementation) // 20 - mstore(0x00, 0x602d5f8160095f39f35f5f365f5f37365f73) // 9 + 9 - hash := keccak256(0x0e, 0x36) - mstore(0x24, 0) // Restore the overwritten part of the free memory pointer. - } - } - - /// @dev Returns the address of the deterministic PUSH0 clone of `implementation`, - /// with `salt` by `deployer`. - /// Note: The returned result has dirty upper 96 bits. Please clean if used in assembly. - function predictDeterministicAddress_PUSH0( - address implementation, - bytes32 salt, - address deployer - ) internal pure returns (address predicted) { - bytes32 hash = initCodeHash_PUSH0(implementation); - predicted = predictDeterministicAddress(hash, salt, deployer); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CLONES WITH IMMUTABLE ARGS OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // Note: This implementation of CWIA differs from the original implementation. - // If the calldata is empty, it will emit a `ReceiveETH(uint256)` event and skip the `DELEGATECALL`. - - /// @dev Deploys a clone of `implementation` with immutable arguments encoded in `data`. - function clone(address implementation, bytes memory data) internal returns (address instance) { - instance = clone(0, implementation, data); - } - - /// @dev Deploys a clone of `implementation` with immutable arguments encoded in `data`. - function clone(uint256 value, address implementation, bytes memory data) - internal - returns (address instance) - { - assembly { - // Compute the boundaries of the data and cache the memory slots around it. - let mBefore3 := mload(sub(data, 0x60)) - let mBefore2 := mload(sub(data, 0x40)) - let mBefore1 := mload(sub(data, 0x20)) - let dataLength := mload(data) - let dataEnd := add(add(data, 0x20), dataLength) - let mAfter1 := mload(dataEnd) - - // +2 bytes for telling how much data there is appended to the call. - let extraLength := add(dataLength, 2) - // The `creationSize` is `extraLength + 108` - // The `runSize` is `creationSize - 10`. - - /** - * ---------------------------------------------------------------------------------------------------+ - * CREATION (10 bytes) | - * ---------------------------------------------------------------------------------------------------| - * Opcode | Mnemonic | Stack | Memory | - * ---------------------------------------------------------------------------------------------------| - * 61 runSize | PUSH2 runSize | r | | - * 3d | RETURNDATASIZE | 0 r | | - * 81 | DUP2 | r 0 r | | - * 60 offset | PUSH1 offset | o r 0 r | | - * 3d | RETURNDATASIZE | 0 o r 0 r | | - * 39 | CODECOPY | 0 r | [0..runSize): runtime code | - * f3 | RETURN | | [0..runSize): runtime code | - * ---------------------------------------------------------------------------------------------------| - * RUNTIME (98 bytes + extraLength) | - * ---------------------------------------------------------------------------------------------------| - * Opcode | Mnemonic | Stack | Memory | - * ---------------------------------------------------------------------------------------------------| - * | - * ::: if no calldata, emit event & return w/o `DELEGATECALL` ::::::::::::::::::::::::::::::::::::::: | - * 36 | CALLDATASIZE | cds | | - * 60 0x2c | PUSH1 0x2c | 0x2c cds | | - * 57 | JUMPI | | | - * 34 | CALLVALUE | cv | | - * 3d | RETURNDATASIZE | 0 cv | | - * 52 | MSTORE | | [0..0x20): callvalue | - * 7f sig | PUSH32 0x9e.. | sig | [0..0x20): callvalue | - * 59 | MSIZE | 0x20 sig | [0..0x20): callvalue | - * 3d | RETURNDATASIZE | 0 0x20 sig | [0..0x20): callvalue | - * a1 | LOG1 | | [0..0x20): callvalue | - * 00 | STOP | | [0..0x20): callvalue | - * 5b | JUMPDEST | | | - * | - * ::: copy calldata to memory :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 36 | CALLDATASIZE | cds | | - * 3d | RETURNDATASIZE | 0 cds | | - * 3d | RETURNDATASIZE | 0 0 cds | | - * 37 | CALLDATACOPY | | [0..cds): calldata | - * | - * ::: keep some values in stack :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | 0 | [0..cds): calldata | - * 3d | RETURNDATASIZE | 0 0 | [0..cds): calldata | - * 3d | RETURNDATASIZE | 0 0 0 | [0..cds): calldata | - * 3d | RETURNDATASIZE | 0 0 0 0 | [0..cds): calldata | - * 61 extra | PUSH2 extra | e 0 0 0 0 | [0..cds): calldata | - * | - * ::: copy extra data to memory :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 80 | DUP1 | e e 0 0 0 0 | [0..cds): calldata | - * 60 0x62 | PUSH1 0x62 | 0x62 e e 0 0 0 0 | [0..cds): calldata | - * 36 | CALLDATASIZE | cds 0x62 e e 0 0 0 0 | [0..cds): calldata | - * 39 | CODECOPY | e 0 0 0 0 | [0..cds): calldata, [cds..cds+e): extraData | - * | - * ::: delegate call to the implementation contract ::::::::::::::::::::::::::::::::::::::::::::::::: | - * 36 | CALLDATASIZE | cds e 0 0 0 0 | [0..cds): calldata, [cds..cds+e): extraData | - * 01 | ADD | cds+e 0 0 0 0 | [0..cds): calldata, [cds..cds+e): extraData | - * 3d | RETURNDATASIZE | 0 cds+e 0 0 0 0 | [0..cds): calldata, [cds..cds+e): extraData | - * 73 addr | PUSH20 addr | addr 0 cds+e 0 0 0 0 | [0..cds): calldata, [cds..cds+e): extraData | - * 5a | GAS | gas addr 0 cds+e 0 0 0 0 | [0..cds): calldata, [cds..cds+e): extraData | - * f4 | DELEGATECALL | success 0 0 | [0..cds): calldata, [cds..cds+e): extraData | - * | - * ::: copy return data to memory ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | rds success 0 0 | [0..cds): calldata, [cds..cds+e): extraData | - * 3d | RETURNDATASIZE | rds rds success 0 0 | [0..cds): calldata, [cds..cds+e): extraData | - * 93 | SWAP4 | 0 rds success 0 rds | [0..cds): calldata, [cds..cds+e): extraData | - * 80 | DUP1 | 0 0 rds success 0 rds | [0..cds): calldata, [cds..cds+e): extraData | - * 3e | RETURNDATACOPY | success 0 rds | [0..rds): returndata | - * | - * 60 0x60 | PUSH1 0x60 | 0x60 success 0 rds | [0..rds): returndata | - * 57 | JUMPI | 0 rds | [0..rds): returndata | - * | - * ::: revert ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * fd | REVERT | | [0..rds): returndata | - * | - * ::: return ::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 5b | JUMPDEST | 0 rds | [0..rds): returndata | - * f3 | RETURN | | [0..rds): returndata | - * ---------------------------------------------------------------------------------------------------+ - */ - - mstore(data, 0x5af43d3d93803e606057fd5bf3) // Write the bytecode before the data. - mstore(sub(data, 0x0d), implementation) // Write the address of the implementation. - // Write the rest of the bytecode. - mstore( - sub(data, 0x21), - or(shl(0x48, extraLength), 0x593da1005b363d3d373d3d3d3d610000806062363936013d73) - ) - // `keccak256("ReceiveETH(uint256)")` - mstore( - sub(data, 0x3a), 0x9e4ac34f21c619cefc926c8bd93b54bf5a39c7ab2127a895af1cc0691d7e3dff - ) - mstore( - // Do a out-of-gas revert if `extraLength` is too big. 0xffff - 0x62 + 0x01 = 0xff9e. - // The actual EVM limit may be smaller and may change over time. - sub(data, add(0x59, lt(extraLength, 0xff9e))), - or(shl(0x78, add(extraLength, 0x62)), 0xfd6100003d81600a3d39f336602c57343d527f) - ) - mstore(dataEnd, shl(0xf0, extraLength)) - - instance := create(value, sub(data, 0x4c), add(extraLength, 0x6c)) - if iszero(instance) { - mstore(0x00, 0x30116425) // `DeploymentFailed()`. - revert(0x1c, 0x04) - } - - // Restore the overwritten memory surrounding `data`. - mstore(dataEnd, mAfter1) - mstore(data, dataLength) - mstore(sub(data, 0x20), mBefore1) - mstore(sub(data, 0x40), mBefore2) - mstore(sub(data, 0x60), mBefore3) - } - } - - /// @dev Deploys a deterministic clone of `implementation` - /// with immutable arguments encoded in `data` and `salt`. - function cloneDeterministic(address implementation, bytes memory data, bytes32 salt) - internal - returns (address instance) - { - instance = cloneDeterministic(0, implementation, data, salt); - } - - /// @dev Deploys a deterministic clone of `implementation` - /// with immutable arguments encoded in `data` and `salt`. - function cloneDeterministic( - uint256 value, - address implementation, - bytes memory data, - bytes32 salt - ) internal returns (address instance) { - assembly { - // Compute the boundaries of the data and cache the memory slots around it. - let mBefore3 := mload(sub(data, 0x60)) - let mBefore2 := mload(sub(data, 0x40)) - let mBefore1 := mload(sub(data, 0x20)) - let dataLength := mload(data) - let dataEnd := add(add(data, 0x20), dataLength) - let mAfter1 := mload(dataEnd) - - // +2 bytes for telling how much data there is appended to the call. - let extraLength := add(dataLength, 2) - - mstore(data, 0x5af43d3d93803e606057fd5bf3) // Write the bytecode before the data. - mstore(sub(data, 0x0d), implementation) // Write the address of the implementation. - // Write the rest of the bytecode. - mstore( - sub(data, 0x21), - or(shl(0x48, extraLength), 0x593da1005b363d3d373d3d3d3d610000806062363936013d73) - ) - // `keccak256("ReceiveETH(uint256)")` - mstore( - sub(data, 0x3a), 0x9e4ac34f21c619cefc926c8bd93b54bf5a39c7ab2127a895af1cc0691d7e3dff - ) - mstore( - // Do a out-of-gas revert if `extraLength` is too big. 0xffff - 0x62 + 0x01 = 0xff9e. - // The actual EVM limit may be smaller and may change over time. - sub(data, add(0x59, lt(extraLength, 0xff9e))), - or(shl(0x78, add(extraLength, 0x62)), 0xfd6100003d81600a3d39f336602c57343d527f) - ) - mstore(dataEnd, shl(0xf0, extraLength)) - - instance := create2(value, sub(data, 0x4c), add(extraLength, 0x6c), salt) - if iszero(instance) { - mstore(0x00, 0x30116425) // `DeploymentFailed()`. - revert(0x1c, 0x04) - } - - // Restore the overwritten memory surrounding `data`. - mstore(dataEnd, mAfter1) - mstore(data, dataLength) - mstore(sub(data, 0x20), mBefore1) - mstore(sub(data, 0x40), mBefore2) - mstore(sub(data, 0x60), mBefore3) - } - } - - /// @dev Returns the initialization code hash of the clone of `implementation` - /// using immutable arguments encoded in `data`. - /// Used for mining vanity addresses with create2crunch. - function initCodeHash(address implementation, bytes memory data) - internal - pure - returns (bytes32 hash) - { - assembly { - // Compute the boundaries of the data and cache the memory slots around it. - let mBefore3 := mload(sub(data, 0x60)) - let mBefore2 := mload(sub(data, 0x40)) - let mBefore1 := mload(sub(data, 0x20)) - let dataLength := mload(data) - let dataEnd := add(add(data, 0x20), dataLength) - let mAfter1 := mload(dataEnd) - - // Do a out-of-gas revert if `dataLength` is too big. 0xffff - 0x02 - 0x62 = 0xff9b. - // The actual EVM limit may be smaller and may change over time. - returndatacopy(returndatasize(), returndatasize(), gt(dataLength, 0xff9b)) - - // +2 bytes for telling how much data there is appended to the call. - let extraLength := add(dataLength, 2) - - mstore(data, 0x5af43d3d93803e606057fd5bf3) // Write the bytecode before the data. - mstore(sub(data, 0x0d), implementation) // Write the address of the implementation. - // Write the rest of the bytecode. - mstore( - sub(data, 0x21), - or(shl(0x48, extraLength), 0x593da1005b363d3d373d3d3d3d610000806062363936013d73) - ) - // `keccak256("ReceiveETH(uint256)")` - mstore( - sub(data, 0x3a), 0x9e4ac34f21c619cefc926c8bd93b54bf5a39c7ab2127a895af1cc0691d7e3dff - ) - mstore( - sub(data, 0x5a), - or(shl(0x78, add(extraLength, 0x62)), 0x6100003d81600a3d39f336602c57343d527f) - ) - mstore(dataEnd, shl(0xf0, extraLength)) - - hash := keccak256(sub(data, 0x4c), add(extraLength, 0x6c)) - - // Restore the overwritten memory surrounding `data`. - mstore(dataEnd, mAfter1) - mstore(data, dataLength) - mstore(sub(data, 0x20), mBefore1) - mstore(sub(data, 0x40), mBefore2) - mstore(sub(data, 0x60), mBefore3) - } - } - - /// @dev Returns the address of the deterministic clone of - /// `implementation` using immutable arguments encoded in `data`, with `salt`, by `deployer`. - /// Note: The returned result has dirty upper 96 bits. Please clean if used in assembly. - function predictDeterministicAddress( - address implementation, - bytes memory data, - bytes32 salt, - address deployer - ) internal pure returns (address predicted) { - bytes32 hash = initCodeHash(implementation, data); - predicted = predictDeterministicAddress(hash, salt, deployer); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* MINIMAL ERC1967 PROXY OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // Note: The ERC1967 proxy here is intended to be upgraded with UUPS. - // This is NOT the same as ERC1967Factory's transparent proxy, which includes admin logic. - - /// @dev Deploys a minimal ERC1967 proxy with `implementation`. - function deployERC1967(address implementation) internal returns (address instance) { - instance = deployERC1967(0, implementation); - } - - /// @dev Deploys a minimal ERC1967 proxy with `implementation`. - function deployERC1967(uint256 value, address implementation) - internal - returns (address instance) - { - /// @solidity memory-safe-assembly - assembly { - /** - * ---------------------------------------------------------------------------------+ - * CREATION (34 bytes) | - * ---------------------------------------------------------------------------------| - * Opcode | Mnemonic | Stack | Memory | - * ---------------------------------------------------------------------------------| - * 60 runSize | PUSH1 runSize | r | | - * 3d | RETURNDATASIZE | 0 r | | - * 81 | DUP2 | r 0 r | | - * 60 offset | PUSH1 offset | o r 0 r | | - * 3d | RETURNDATASIZE | 0 o r 0 r | | - * 39 | CODECOPY | 0 r | [0..runSize): runtime code | - * 73 impl | PUSH20 impl | impl 0 r | [0..runSize): runtime code | - * 60 slotPos | PUSH1 slotPos | slotPos impl 0 r | [0..runSize): runtime code | - * 51 | MLOAD | slot impl 0 r | [0..runSize): runtime code | - * 55 | SSTORE | 0 r | [0..runSize): runtime code | - * f3 | RETURN | | [0..runSize): runtime code | - * ---------------------------------------------------------------------------------| - * RUNTIME (62 bytes) | - * ---------------------------------------------------------------------------------| - * Opcode | Mnemonic | Stack | Memory | - * ---------------------------------------------------------------------------------| - * | - * ::: copy calldata to memory :::::::::::::::::::::::::::::::::::::::::::::::::::: | - * 36 | CALLDATASIZE | cds | | - * 3d | RETURNDATASIZE | 0 cds | | - * 3d | RETURNDATASIZE | 0 0 cds | | - * 37 | CALLDATACOPY | | [0..calldatasize): calldata | - * | - * ::: delegatecall to implementation ::::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | 0 | | - * 3d | RETURNDATASIZE | 0 0 | | - * 36 | CALLDATASIZE | cds 0 0 | [0..calldatasize): calldata | - * 3d | RETURNDATASIZE | 0 cds 0 0 | [0..calldatasize): calldata | - * 7f slot | PUSH32 slot | s 0 cds 0 0 | [0..calldatasize): calldata | - * 54 | SLOAD | i 0 cds 0 0 | [0..calldatasize): calldata | - * 5a | GAS | g i 0 cds 0 0 | [0..calldatasize): calldata | - * f4 | DELEGATECALL | succ | [0..calldatasize): calldata | - * | - * ::: copy returndata to memory :::::::::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | rds succ | [0..calldatasize): calldata | - * 60 0x00 | PUSH1 0x00 | 0 rds succ | [0..calldatasize): calldata | - * 80 | DUP1 | 0 0 rds succ | [0..calldatasize): calldata | - * 3e | RETURNDATACOPY | succ | [0..returndatasize): returndata | - * | - * ::: branch on delegatecall status :::::::::::::::::::::::::::::::::::::::::::::: | - * 60 0x38 | PUSH1 0x38 | dest succ | [0..returndatasize): returndata | - * 57 | JUMPI | | [0..returndatasize): returndata | - * | - * ::: delegatecall failed, revert :::::::::::::::::::::::::::::::::::::::::::::::: | - * 3d | RETURNDATASIZE | rds | [0..returndatasize): returndata | - * 60 0x00 | PUSH1 0x00 | 0 rds | [0..returndatasize): returndata | - * fd | REVERT | | [0..returndatasize): returndata | - * | - * ::: delegatecall succeeded, return ::::::::::::::::::::::::::::::::::::::::::::: | - * 5b | JUMPDEST | | [0..returndatasize): returndata | - * 3d | RETURNDATASIZE | rds | [0..returndatasize): returndata | - * 60 0x00 | PUSH1 0x00 | 0 rds | [0..returndatasize): returndata | - * f3 | RETURN | | [0..returndatasize): returndata | - * ---------------------------------------------------------------------------------+ - */ - - let m := mload(0x40) // Cache the free memory pointer. - mstore(0x60, 0xcc3735a920a3ca505d382bbc545af43d6000803e6038573d6000fd5b3d6000f3) - mstore(0x40, 0x5155f3363d3d373d3d363d7f360894a13ba1a3210667c828492db98dca3e2076) - mstore(0x20, 0x6009) - mstore(0x1e, implementation) - mstore(0x0a, 0x603d3d8160223d3973) - instance := create(value, 0x21, 0x5f) - if iszero(instance) { - mstore(0x00, 0x30116425) // `DeploymentFailed()`. - revert(0x1c, 0x04) - } - mstore(0x40, m) // Restore the free memory pointer. - mstore(0x60, 0) // Restore the zero slot. - } - } - - /// @dev Deploys a deterministic minimal ERC1967 proxy with `implementation` and `salt`. - function deployDeterministicERC1967(address implementation, bytes32 salt) - internal - returns (address instance) - { - instance = deployDeterministicERC1967(0, implementation, salt); - } - - /// @dev Deploys a deterministic minimal ERC1967 proxy with `implementation` and `salt`. - function deployDeterministicERC1967(uint256 value, address implementation, bytes32 salt) - internal - returns (address instance) - { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Cache the free memory pointer. - mstore(0x60, 0xcc3735a920a3ca505d382bbc545af43d6000803e6038573d6000fd5b3d6000f3) - mstore(0x40, 0x5155f3363d3d373d3d363d7f360894a13ba1a3210667c828492db98dca3e2076) - mstore(0x20, 0x6009) - mstore(0x1e, implementation) - mstore(0x0a, 0x603d3d8160223d3973) - instance := create2(value, 0x21, 0x5f, salt) - if iszero(instance) { - mstore(0x00, 0x30116425) // `DeploymentFailed()`. - revert(0x1c, 0x04) - } - mstore(0x40, m) // Restore the free memory pointer. - mstore(0x60, 0) // Restore the zero slot. - } - } - - /// @dev Creates a deterministic minimal ERC1967 proxy with `implementation` and `salt`. - /// Note: This method is intended for use in ERC4337 factories, - /// which are expected to NOT revert if the proxy is already deployed. - function createDeterministicERC1967(address implementation, bytes32 salt) - internal - returns (bool alreadyDeployed, address instance) - { - return createDeterministicERC1967(0, implementation, salt); - } - - /// @dev Creates a deterministic minimal ERC1967 proxy with `implementation` and `salt`. - /// Note: This method is intended for use in ERC4337 factories, - /// which are expected to NOT revert if the proxy is already deployed. - function createDeterministicERC1967(uint256 value, address implementation, bytes32 salt) - internal - returns (bool alreadyDeployed, address instance) - { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Cache the free memory pointer. - mstore(0x60, 0xcc3735a920a3ca505d382bbc545af43d6000803e6038573d6000fd5b3d6000f3) - mstore(0x40, 0x5155f3363d3d373d3d363d7f360894a13ba1a3210667c828492db98dca3e2076) - mstore(0x20, 0x6009) - mstore(0x1e, implementation) - mstore(0x0a, 0x603d3d8160223d3973) - // Compute and store the bytecode hash. - mstore(add(m, 0x35), keccak256(0x21, 0x5f)) - mstore(m, shl(88, address())) - mstore8(m, 0xff) // Write the prefix. - mstore(add(m, 0x15), salt) - instance := keccak256(m, 0x55) - for {} 1 {} { - if iszero(extcodesize(instance)) { - instance := create2(value, 0x21, 0x5f, salt) - if iszero(instance) { - mstore(0x00, 0x30116425) // `DeploymentFailed()`. - revert(0x1c, 0x04) - } - break - } - alreadyDeployed := 1 - if iszero(value) { break } - if iszero(call(gas(), instance, value, codesize(), 0x00, codesize(), 0x00)) { - mstore(0x00, 0xb12d13eb) // `ETHTransferFailed()`. - revert(0x1c, 0x04) - } - break - } - mstore(0x40, m) // Restore the free memory pointer. - mstore(0x60, 0) // Restore the zero slot. - } - } - - /// @dev Returns the initialization code hash of the clone of `implementation` - /// using immutable arguments encoded in `data`. - /// Used for mining vanity addresses with create2crunch. - function initCodeHashERC1967(address implementation) internal pure returns (bytes32 hash) { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Cache the free memory pointer. - mstore(0x60, 0xcc3735a920a3ca505d382bbc545af43d6000803e6038573d6000fd5b3d6000f3) - mstore(0x40, 0x5155f3363d3d373d3d363d7f360894a13ba1a3210667c828492db98dca3e2076) - mstore(0x20, 0x6009) - mstore(0x1e, implementation) - mstore(0x0a, 0x603d3d8160223d3973) - hash := keccak256(0x21, 0x5f) - mstore(0x40, m) // Restore the free memory pointer. - mstore(0x60, 0) // Restore the zero slot. - } - } - - /// @dev Returns the address of the deterministic clone of - /// `implementation` using immutable arguments encoded in `data`, with `salt`, by `deployer`. - /// Note: The returned result has dirty upper 96 bits. Please clean if used in assembly. - function predictDeterministicAddressERC1967( - address implementation, - bytes32 salt, - address deployer - ) internal pure returns (address predicted) { - bytes32 hash = initCodeHashERC1967(implementation); - predicted = predictDeterministicAddress(hash, salt, deployer); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* OTHER OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the address when a contract with initialization code hash, - /// `hash`, is deployed with `salt`, by `deployer`. - /// Note: The returned result has dirty upper 96 bits. Please clean if used in assembly. - function predictDeterministicAddress(bytes32 hash, bytes32 salt, address deployer) - internal - pure - returns (address predicted) - { - /// @solidity memory-safe-assembly - assembly { - // Compute and store the bytecode hash. - mstore8(0x00, 0xff) // Write the prefix. - mstore(0x35, hash) - mstore(0x01, shl(96, deployer)) - mstore(0x15, salt) - predicted := keccak256(0x00, 0x55) - mstore(0x35, 0) // Restore the overwritten part of the free memory pointer. - } - } - - /// @dev Requires that `salt` starts with either the zero address or `by`. - function checkStartsWith(bytes32 salt, address by) internal pure { - /// @solidity memory-safe-assembly - assembly { - // If the salt does not start with the zero address or `by`. - if iszero(or(iszero(shr(96, salt)), eq(shr(96, shl(96, by)), shr(96, salt)))) { - mstore(0x00, 0x0c4549ef) // `SaltDoesNotStartWith()`. - revert(0x1c, 0x04) - } - } - } -} diff --git a/lib/solady/src/utils/LibMap.sol b/lib/solady/src/utils/LibMap.sol deleted file mode 100644 index edac97a..0000000 --- a/lib/solady/src/utils/LibMap.sol +++ /dev/null @@ -1,309 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for storage of packed unsigned integers. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/LibMap.sol) -library LibMap { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STRUCTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev A uint8 map in storage. - struct Uint8Map { - mapping(uint256 => uint256) map; - } - - /// @dev A uint16 map in storage. - struct Uint16Map { - mapping(uint256 => uint256) map; - } - - /// @dev A uint32 map in storage. - struct Uint32Map { - mapping(uint256 => uint256) map; - } - - /// @dev A uint40 map in storage. Useful for storing timestamps up to 34841 A.D. - struct Uint40Map { - mapping(uint256 => uint256) map; - } - - /// @dev A uint64 map in storage. - struct Uint64Map { - mapping(uint256 => uint256) map; - } - - /// @dev A uint128 map in storage. - struct Uint128Map { - mapping(uint256 => uint256) map; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* GETTERS / SETTERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the uint8 value at `index` in `map`. - function get(Uint8Map storage map, uint256 index) internal view returns (uint8 result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, map.slot) - mstore(0x00, shr(5, index)) - result := byte(and(31, not(index)), sload(keccak256(0x00, 0x40))) - } - } - - /// @dev Updates the uint8 value at `index` in `map`. - function set(Uint8Map storage map, uint256 index, uint8 value) internal { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, map.slot) - mstore(0x00, shr(5, index)) - let s := keccak256(0x00, 0x40) // Storage slot. - mstore(0x00, sload(s)) - mstore8(and(31, not(index)), value) - sstore(s, mload(0x00)) - } - } - - /// @dev Returns the uint16 value at `index` in `map`. - function get(Uint16Map storage map, uint256 index) internal view returns (uint16 result) { - result = uint16(map.map[index >> 4] >> ((index & 15) << 4)); - } - - /// @dev Updates the uint16 value at `index` in `map`. - function set(Uint16Map storage map, uint256 index, uint16 value) internal { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, map.slot) - mstore(0x00, shr(4, index)) - let s := keccak256(0x00, 0x40) // Storage slot. - let o := shl(4, and(index, 15)) // Storage slot offset (bits). - let v := sload(s) // Storage slot value. - let m := 0xffff // Value mask. - sstore(s, xor(v, shl(o, and(m, xor(shr(o, v), value))))) - } - } - - /// @dev Returns the uint32 value at `index` in `map`. - function get(Uint32Map storage map, uint256 index) internal view returns (uint32 result) { - result = uint32(map.map[index >> 3] >> ((index & 7) << 5)); - } - - /// @dev Updates the uint32 value at `index` in `map`. - function set(Uint32Map storage map, uint256 index, uint32 value) internal { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, map.slot) - mstore(0x00, shr(3, index)) - let s := keccak256(0x00, 0x40) // Storage slot. - let o := shl(5, and(index, 7)) // Storage slot offset (bits). - let v := sload(s) // Storage slot value. - let m := 0xffffffff // Value mask. - sstore(s, xor(v, shl(o, and(m, xor(shr(o, v), value))))) - } - } - - /// @dev Returns the uint40 value at `index` in `map`. - function get(Uint40Map storage map, uint256 index) internal view returns (uint40 result) { - unchecked { - result = uint40(map.map[index / 6] >> ((index % 6) * 40)); - } - } - - /// @dev Updates the uint40 value at `index` in `map`. - function set(Uint40Map storage map, uint256 index, uint40 value) internal { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, map.slot) - mstore(0x00, div(index, 6)) - let s := keccak256(0x00, 0x40) // Storage slot. - let o := mul(40, mod(index, 6)) // Storage slot offset (bits). - let v := sload(s) // Storage slot value. - let m := 0xffffffffff // Value mask. - sstore(s, xor(v, shl(o, and(m, xor(shr(o, v), value))))) - } - } - - /// @dev Returns the uint64 value at `index` in `map`. - function get(Uint64Map storage map, uint256 index) internal view returns (uint64 result) { - result = uint64(map.map[index >> 2] >> ((index & 3) << 6)); - } - - /// @dev Updates the uint64 value at `index` in `map`. - function set(Uint64Map storage map, uint256 index, uint64 value) internal { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, map.slot) - mstore(0x00, shr(2, index)) - let s := keccak256(0x00, 0x40) // Storage slot. - let o := shl(6, and(index, 3)) // Storage slot offset (bits). - let v := sload(s) // Storage slot value. - let m := 0xffffffffffffffff // Value mask. - sstore(s, xor(v, shl(o, and(m, xor(shr(o, v), value))))) - } - } - - /// @dev Returns the uint128 value at `index` in `map`. - function get(Uint128Map storage map, uint256 index) internal view returns (uint128 result) { - result = uint128(map.map[index >> 1] >> ((index & 1) << 7)); - } - - /// @dev Updates the uint128 value at `index` in `map`. - function set(Uint128Map storage map, uint256 index, uint128 value) internal { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, map.slot) - mstore(0x00, shr(1, index)) - let s := keccak256(0x00, 0x40) // Storage slot. - let o := shl(7, and(index, 1)) // Storage slot offset (bits). - let v := sload(s) // Storage slot value. - let m := 0xffffffffffffffffffffffffffffffff // Value mask. - sstore(s, xor(v, shl(o, and(m, xor(shr(o, v), value))))) - } - } - - /// @dev Returns the value at `index` in `map`. - function get(mapping(uint256 => uint256) storage map, uint256 index, uint256 bitWidth) - internal - view - returns (uint256 result) - { - unchecked { - uint256 d = _rawDiv(256, bitWidth); // Bucket size. - uint256 m = (1 << bitWidth) - 1; // Value mask. - result = (map[_rawDiv(index, d)] >> (_rawMod(index, d) * bitWidth)) & m; - } - } - - /// @dev Updates the value at `index` in `map`. - function set( - mapping(uint256 => uint256) storage map, - uint256 index, - uint256 value, - uint256 bitWidth - ) internal { - unchecked { - uint256 d = _rawDiv(256, bitWidth); // Bucket size. - uint256 m = (1 << bitWidth) - 1; // Value mask. - uint256 o = _rawMod(index, d) * bitWidth; // Storage slot offset (bits). - map[_rawDiv(index, d)] ^= (((map[_rawDiv(index, d)] >> o) ^ value) & m) << o; - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* BINARY SEARCH */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // The following functions search in the range of [`start`, `end`) - // (i.e. `start <= index < end`). - // The range must be sorted in ascending order. - // `index` precedence: equal to > nearest before > nearest after. - // An invalid search range will simply return `(found = false, index = start)`. - - /// @dev Returns whether `map` contains `needle`, and the index of `needle`. - function searchSorted(Uint8Map storage map, uint8 needle, uint256 start, uint256 end) - internal - view - returns (bool found, uint256 index) - { - return searchSorted(map.map, needle, start, end, 8); - } - - /// @dev Returns whether `map` contains `needle`, and the index of `needle`. - function searchSorted(Uint16Map storage map, uint16 needle, uint256 start, uint256 end) - internal - view - returns (bool found, uint256 index) - { - return searchSorted(map.map, needle, start, end, 16); - } - - /// @dev Returns whether `map` contains `needle`, and the index of `needle`. - function searchSorted(Uint32Map storage map, uint32 needle, uint256 start, uint256 end) - internal - view - returns (bool found, uint256 index) - { - return searchSorted(map.map, needle, start, end, 32); - } - - /// @dev Returns whether `map` contains `needle`, and the index of `needle`. - function searchSorted(Uint40Map storage map, uint40 needle, uint256 start, uint256 end) - internal - view - returns (bool found, uint256 index) - { - return searchSorted(map.map, needle, start, end, 40); - } - - /// @dev Returns whether `map` contains `needle`, and the index of `needle`. - function searchSorted(Uint64Map storage map, uint64 needle, uint256 start, uint256 end) - internal - view - returns (bool found, uint256 index) - { - return searchSorted(map.map, needle, start, end, 64); - } - - /// @dev Returns whether `map` contains `needle`, and the index of `needle`. - function searchSorted(Uint128Map storage map, uint128 needle, uint256 start, uint256 end) - internal - view - returns (bool found, uint256 index) - { - return searchSorted(map.map, needle, start, end, 128); - } - - /// @dev Returns whether `map` contains `needle`, and the index of `needle`. - function searchSorted( - mapping(uint256 => uint256) storage map, - uint256 needle, - uint256 start, - uint256 end, - uint256 bitWidth - ) internal view returns (bool found, uint256 index) { - unchecked { - if (start >= end) end = start; - uint256 t; - uint256 o = start - 1; // Offset to derive the actual index. - uint256 l = 1; // Low. - uint256 d = _rawDiv(256, bitWidth); // Bucket size. - uint256 m = (1 << bitWidth) - 1; // Value mask. - uint256 h = end - start; // High. - while (true) { - index = (l & h) + ((l ^ h) >> 1); - if (l > h) break; - t = (map[_rawDiv(index + o, d)] >> (_rawMod(index + o, d) * bitWidth)) & m; - if (t == needle) break; - if (needle <= t) h = index - 1; - else l = index + 1; - } - /// @solidity memory-safe-assembly - assembly { - m := or(iszero(index), iszero(bitWidth)) - found := iszero(or(xor(t, needle), m)) - index := add(o, xor(index, mul(xor(index, 1), m))) - } - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PRIVATE HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns `x / y`, returning 0 if `y` is zero. - function _rawDiv(uint256 x, uint256 y) private pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := div(x, y) - } - } - - /// @dev Returns `x % y`, returning 0 if `y` is zero. - function _rawMod(uint256 x, uint256 y) private pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - z := mod(x, y) - } - } -} diff --git a/lib/solady/src/utils/LibPRNG.sol b/lib/solady/src/utils/LibPRNG.sol deleted file mode 100644 index 6e1d9da..0000000 --- a/lib/solady/src/utils/LibPRNG.sol +++ /dev/null @@ -1,153 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for generating pseudorandom numbers. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/LibPRNG.sol) -library LibPRNG { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STRUCTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev A pseudorandom number state in memory. - struct PRNG { - uint256 state; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Seeds the `prng` with `state`. - function seed(PRNG memory prng, uint256 state) internal pure { - /// @solidity memory-safe-assembly - assembly { - mstore(prng, state) - } - } - - /// @dev Returns the next pseudorandom uint256. - /// All bits of the returned uint256 pass the NIST Statistical Test Suite. - function next(PRNG memory prng) internal pure returns (uint256 result) { - // We simply use `keccak256` for a great balance between - // runtime gas costs, bytecode size, and statistical properties. - // - // A high-quality LCG with a 32-byte state - // is only about 30% more gas efficient during runtime, - // but requires a 32-byte multiplier, which can cause bytecode bloat - // when this function is inlined. - // - // Using this method is about 2x more efficient than - // `nextRandomness = uint256(keccak256(abi.encode(randomness)))`. - /// @solidity memory-safe-assembly - assembly { - result := keccak256(prng, 0x20) - mstore(prng, result) - } - } - - /// @dev Returns a pseudorandom uint256, uniformly distributed - /// between 0 (inclusive) and `upper` (exclusive). - /// If your modulus is big, this method is recommended - /// for uniform sampling to avoid modulo bias. - /// For uniform sampling across all uint256 values, - /// or for small enough moduli such that the bias is neligible, - /// use {next} instead. - function uniform(PRNG memory prng, uint256 upper) internal pure returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - for {} 1 {} { - result := keccak256(prng, 0x20) - mstore(prng, result) - if iszero(lt(result, mod(sub(0, upper), upper))) { break } - } - result := mod(result, upper) - } - } - - /// @dev Shuffles the array in-place with Fisher-Yates shuffle. - function shuffle(PRNG memory prng, uint256[] memory a) internal pure { - /// @solidity memory-safe-assembly - assembly { - let n := mload(a) - let w := not(0) - let mask := shr(128, w) - if n { - for { a := add(a, 0x20) } 1 {} { - // We can just directly use `keccak256`, cuz - // the other approaches don't save much. - let r := keccak256(prng, 0x20) - mstore(prng, r) - - // Note that there will be a very tiny modulo bias - // if the length of the array is not a power of 2. - // For all practical purposes, it is negligible - // and will not be a fairness or security concern. - { - let j := add(a, shl(5, mod(shr(128, r), n))) - n := add(n, w) // `sub(n, 1)`. - if iszero(n) { break } - - let i := add(a, shl(5, n)) - let t := mload(i) - mstore(i, mload(j)) - mstore(j, t) - } - - { - let j := add(a, shl(5, mod(and(r, mask), n))) - n := add(n, w) // `sub(n, 1)`. - if iszero(n) { break } - - let i := add(a, shl(5, n)) - let t := mload(i) - mstore(i, mload(j)) - mstore(j, t) - } - } - } - } - } - - /// @dev Shuffles the bytes in-place with Fisher-Yates shuffle. - function shuffle(PRNG memory prng, bytes memory a) internal pure { - /// @solidity memory-safe-assembly - assembly { - let n := mload(a) - let w := not(0) - let mask := shr(128, w) - if n { - let b := add(a, 0x01) - for { a := add(a, 0x20) } 1 {} { - // We can just directly use `keccak256`, cuz - // the other approaches don't save much. - let r := keccak256(prng, 0x20) - mstore(prng, r) - - // Note that there will be a very tiny modulo bias - // if the length of the array is not a power of 2. - // For all practical purposes, it is negligible - // and will not be a fairness or security concern. - { - let o := mod(shr(128, r), n) - n := add(n, w) // `sub(n, 1)`. - if iszero(n) { break } - - let t := mload(add(b, n)) - mstore8(add(a, n), mload(add(b, o))) - mstore8(add(a, o), t) - } - - { - let o := mod(and(r, mask), n) - n := add(n, w) // `sub(n, 1)`. - if iszero(n) { break } - - let t := mload(add(b, n)) - mstore8(add(a, n), mload(add(b, o))) - mstore8(add(a, o), t) - } - } - } - } - } -} diff --git a/lib/solady/src/utils/LibRLP.sol b/lib/solady/src/utils/LibRLP.sol deleted file mode 100644 index 4f489c8..0000000 --- a/lib/solady/src/utils/LibRLP.sol +++ /dev/null @@ -1,71 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for computing contract addresses from their deployer and nonce. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/LibRLP.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/LibRLP.sol) -library LibRLP { - /// @dev Returns the address where a contract will be stored if deployed via - /// `deployer` with `nonce` using the `CREATE` opcode. - /// For the specification of the Recursive Length Prefix (RLP) - /// encoding scheme, please refer to p. 19 of the Ethereum Yellow Paper - /// (https://ethereum.github.io/yellowpaper/paper.pdf) - /// and the Ethereum Wiki (https://eth.wiki/fundamentals/rlp). - /// - /// Based on the EIP-161 (https://github.com/ethereum/EIPs/blob/master/EIPS/eip-161.md) - /// specification, all contract accounts on the Ethereum mainnet are initiated with - /// `nonce = 1`. Thus, the first contract address created by another contract - /// is calculated with a non-zero nonce. - /// - /// The theoretical allowed limit, based on EIP-2681 - /// (https://eips.ethereum.org/EIPS/eip-2681), for an account nonce is 2**64-2. - /// - /// Caution! This function will NOT check that the nonce is within the theoretical range. - /// This is for performance, as exceeding the range is extremely impractical. - /// It is the user's responsibility to ensure that the nonce is valid - /// (e.g. no dirty bits after packing / unpacking). - /// - /// Note: The returned result has dirty upper 96 bits. Please clean if used in assembly. - function computeAddress(address deployer, uint256 nonce) - internal - pure - returns (address deployed) - { - /// @solidity memory-safe-assembly - assembly { - for {} 1 {} { - // The integer zero is treated as an empty byte string, - // and as a result it only has a length prefix, 0x80, - // computed via `0x80 + 0`. - - // A one-byte integer in the [0x00, 0x7f] range uses its - // own value as a length prefix, - // there is no additional `0x80 + length` prefix that precedes it. - if iszero(gt(nonce, 0x7f)) { - mstore(0x00, deployer) - // Using `mstore8` instead of `or` naturally cleans - // any dirty upper bits of `deployer`. - mstore8(0x0b, 0x94) - mstore8(0x0a, 0xd6) - // `shl` 7 is equivalent to multiplying by 0x80. - mstore8(0x20, or(shl(7, iszero(nonce)), nonce)) - deployed := keccak256(0x0a, 0x17) - break - } - let i := 8 - // Just use a loop to generalize all the way with minimal bytecode size. - for {} shr(i, nonce) { i := add(i, 8) } {} - // `shr` 3 is equivalent to dividing by 8. - i := shr(3, i) - // Store in descending slot sequence to overlap the values correctly. - mstore(i, nonce) - mstore(0x00, shl(8, deployer)) - mstore8(0x1f, add(0x80, i)) - mstore8(0x0a, 0x94) - mstore8(0x09, add(0xd6, i)) - deployed := keccak256(0x09, add(0x17, i)) - break - } - } - } -} diff --git a/lib/solady/src/utils/LibSort.sol b/lib/solady/src/utils/LibSort.sol deleted file mode 100644 index 97789bf..0000000 --- a/lib/solady/src/utils/LibSort.sol +++ /dev/null @@ -1,699 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Optimized sorts and operations for sorted arrays. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/Sort.sol) -library LibSort { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INSERTION SORT */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // - Faster on small arrays (32 or lesser elements). - // - Faster on almost sorted arrays. - // - Smaller bytecode. - // - May be suitable for view functions intended for off-chain querying. - - /// @dev Sorts the array in-place with insertion sort. - function insertionSort(uint256[] memory a) internal pure { - /// @solidity memory-safe-assembly - assembly { - let n := mload(a) // Length of `a`. - mstore(a, 0) // For insertion sort's inner loop to terminate. - let h := add(a, shl(5, n)) // High slot. - let s := 0x20 - let w := not(0x1f) - for { let i := add(a, s) } 1 {} { - i := add(i, s) - if gt(i, h) { break } - let k := mload(i) // Key. - let j := add(i, w) // The slot before the current slot. - let v := mload(j) // The value of `j`. - if iszero(gt(v, k)) { continue } - for {} 1 {} { - mstore(add(j, s), v) - j := add(j, w) // `sub(j, 0x20)`. - v := mload(j) - if iszero(gt(v, k)) { break } - } - mstore(add(j, s), k) - } - mstore(a, n) // Restore the length of `a`. - } - } - - /// @dev Sorts the array in-place with insertion sort. - function insertionSort(int256[] memory a) internal pure { - _flipSign(a); - insertionSort(_toUints(a)); - _flipSign(a); - } - - /// @dev Sorts the array in-place with insertion sort. - function insertionSort(address[] memory a) internal pure { - insertionSort(_toUints(a)); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* INTRO-QUICKSORT */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // - Faster on larger arrays (more than 32 elements). - // - Robust performance. - // - Larger bytecode. - - /// @dev Sorts the array in-place with intro-quicksort. - function sort(uint256[] memory a) internal pure { - /// @solidity memory-safe-assembly - assembly { - let w := not(0x1f) - let s := 0x20 - let n := mload(a) // Length of `a`. - mstore(a, 0) // For insertion sort's inner loop to terminate. - - // Let the stack be the start of the free memory. - let stack := mload(0x40) - - for {} iszero(lt(n, 2)) {} { - // Push `l` and `h` to the stack. - // The `shl` by 5 is equivalent to multiplying by `0x20`. - let l := add(a, s) - let h := add(a, shl(5, n)) - - let j := l - // forgefmt: disable-next-item - for {} iszero(or(eq(j, h), gt(mload(j), mload(add(j, s))))) {} { - j := add(j, s) - } - // If the array is already sorted. - if eq(j, h) { break } - - j := h - // forgefmt: disable-next-item - for {} iszero(gt(mload(j), mload(add(j, w)))) {} { - j := add(j, w) // `sub(j, 0x20)`. - } - // If the array is reversed sorted. - if eq(j, l) { - for {} 1 {} { - let t := mload(l) - mstore(l, mload(h)) - mstore(h, t) - h := add(h, w) // `sub(h, 0x20)`. - l := add(l, s) - if iszero(lt(l, h)) { break } - } - break - } - - // Push `l` and `h` onto the stack. - mstore(stack, l) - mstore(add(stack, s), h) - stack := add(stack, 0x40) - break - } - - for { let stackBottom := mload(0x40) } iszero(eq(stack, stackBottom)) {} { - // Pop `l` and `h` from the stack. - stack := sub(stack, 0x40) - let l := mload(stack) - let h := mload(add(stack, s)) - - // Do insertion sort if `h - l <= 0x20 * 12`. - // Threshold is fine-tuned via trial and error. - if iszero(gt(sub(h, l), 0x180)) { - // Hardcode sort the first 2 elements. - let i := add(l, s) - if iszero(lt(mload(l), mload(i))) { - let t := mload(i) - mstore(i, mload(l)) - mstore(l, t) - } - for {} 1 {} { - i := add(i, s) - if gt(i, h) { break } - let k := mload(i) // Key. - let j := add(i, w) // The slot before the current slot. - let v := mload(j) // The value of `j`. - if iszero(gt(v, k)) { continue } - for {} 1 {} { - mstore(add(j, s), v) - j := add(j, w) - v := mload(j) - if iszero(gt(v, k)) { break } - } - mstore(add(j, s), k) - } - continue - } - // Pivot slot is the average of `l` and `h`. - let p := add(shl(5, shr(6, add(l, h))), and(31, l)) - // Median of 3 with sorting. - { - function swap(a_, b_) -> _b, _a { - _b := a_ - _a := b_ - } - let e0 := mload(l) - let e1 := mload(h) - if iszero(lt(e0, e1)) { e1, e0 := swap(e0, e1) } - let e2 := mload(p) - if iszero(lt(e2, e1)) { e2, e1 := swap(e1, e2) } - if iszero(lt(e0, e2)) { e2, e0 := swap(e0, e2) } - mstore(p, e2) - mstore(h, e1) - mstore(l, e0) - } - // Hoare's partition. - { - // The value of the pivot slot. - let x := mload(p) - p := h - for { let i := l } 1 {} { - for {} 1 {} { - i := add(i, s) - if iszero(gt(x, mload(i))) { break } - } - let j := p - for {} 1 {} { - j := add(j, w) - if iszero(lt(x, mload(j))) { break } - } - p := j - if iszero(lt(i, p)) { break } - // Swap slots `i` and `p`. - let t := mload(i) - mstore(i, mload(p)) - mstore(p, t) - } - } - // If slice on right of pivot is non-empty, push onto stack. - { - mstore(stack, add(p, s)) - // Skip `mstore(add(stack, 0x20), h)`, as it is already on the stack. - stack := add(stack, shl(6, lt(add(p, s), h))) - } - // If slice on left of pivot is non-empty, push onto stack. - { - mstore(stack, l) - mstore(add(stack, s), p) - stack := add(stack, shl(6, gt(p, l))) - } - } - mstore(a, n) // Restore the length of `a`. - } - } - - /// @dev Sorts the array in-place with intro-quicksort. - function sort(int256[] memory a) internal pure { - _flipSign(a); - sort(_toUints(a)); - _flipSign(a); - } - - /// @dev Sorts the array in-place with intro-quicksort. - function sort(address[] memory a) internal pure { - sort(_toUints(a)); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* OTHER USEFUL OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // For performance, the `uniquifySorted` methods will not revert if the - // array is not sorted -- it will simply remove consecutive duplicate elements. - - /// @dev Removes duplicate elements from a ascendingly sorted memory array. - function uniquifySorted(uint256[] memory a) internal pure { - /// @solidity memory-safe-assembly - assembly { - // If the length of `a` is greater than 1. - if iszero(lt(mload(a), 2)) { - let x := add(a, 0x20) - let y := add(a, 0x40) - let end := add(a, shl(5, add(mload(a), 1))) - for {} 1 {} { - if iszero(eq(mload(x), mload(y))) { - x := add(x, 0x20) - mstore(x, mload(y)) - } - y := add(y, 0x20) - if eq(y, end) { break } - } - mstore(a, shr(5, sub(x, a))) - } - } - } - - /// @dev Removes duplicate elements from a ascendingly sorted memory array. - function uniquifySorted(int256[] memory a) internal pure { - uniquifySorted(_toUints(a)); - } - - /// @dev Removes duplicate elements from a ascendingly sorted memory array. - function uniquifySorted(address[] memory a) internal pure { - uniquifySorted(_toUints(a)); - } - - /// @dev Returns whether `a` contains `needle`, and the index of `needle`. - /// `index` precedence: equal to > nearest before > nearest after. - function searchSorted(uint256[] memory a, uint256 needle) - internal - pure - returns (bool found, uint256 index) - { - (found, index) = _searchSorted(a, needle, 0); - } - - /// @dev Returns whether `a` contains `needle`, and the index of `needle`. - /// `index` precedence: equal to > nearest before > nearest after. - function searchSorted(int256[] memory a, int256 needle) - internal - pure - returns (bool found, uint256 index) - { - (found, index) = _searchSorted(_toUints(a), uint256(needle), 1 << 255); - } - - /// @dev Returns whether `a` contains `needle`, and the index of `needle`. - /// `index` precedence: equal to > nearest before > nearest after. - function searchSorted(address[] memory a, address needle) - internal - pure - returns (bool found, uint256 index) - { - (found, index) = _searchSorted(_toUints(a), uint256(uint160(needle)), 0); - } - - /// @dev Reverses the array in-place. - function reverse(uint256[] memory a) internal pure { - /// @solidity memory-safe-assembly - assembly { - if iszero(lt(mload(a), 2)) { - let s := 0x20 - let w := not(0x1f) - let h := add(a, shl(5, mload(a))) - for { a := add(a, s) } 1 {} { - let t := mload(a) - mstore(a, mload(h)) - mstore(h, t) - h := add(h, w) - a := add(a, s) - if iszero(lt(a, h)) { break } - } - } - } - } - - /// @dev Reverses the array in-place. - function reverse(int256[] memory a) internal pure { - reverse(_toUints(a)); - } - - /// @dev Reverses the array in-place. - function reverse(address[] memory a) internal pure { - reverse(_toUints(a)); - } - - /// @dev Returns whether the array is sorted in ascending order. - function isSorted(uint256[] memory a) internal pure returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - result := 1 - if iszero(lt(mload(a), 2)) { - let end := add(a, shl(5, mload(a))) - for { a := add(a, 0x20) } 1 {} { - let p := mload(a) - a := add(a, 0x20) - result := iszero(gt(p, mload(a))) - if iszero(mul(result, xor(a, end))) { break } - } - } - } - } - - /// @dev Returns whether the array is sorted in ascending order. - function isSorted(int256[] memory a) internal pure returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - result := 1 - if iszero(lt(mload(a), 2)) { - let end := add(a, shl(5, mload(a))) - for { a := add(a, 0x20) } 1 {} { - let p := mload(a) - a := add(a, 0x20) - result := iszero(sgt(p, mload(a))) - if iszero(mul(result, xor(a, end))) { break } - } - } - } - } - - /// @dev Returns whether the array is sorted in ascending order. - function isSorted(address[] memory a) internal pure returns (bool result) { - result = isSorted(_toUints(a)); - } - - /// @dev Returns whether the array is strictly ascending (sorted and uniquified). - function isSortedAndUniquified(uint256[] memory a) internal pure returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - result := 1 - if iszero(lt(mload(a), 2)) { - let end := add(a, shl(5, mload(a))) - for { a := add(a, 0x20) } 1 {} { - let p := mload(a) - a := add(a, 0x20) - result := lt(p, mload(a)) - if iszero(mul(result, xor(a, end))) { break } - } - } - } - } - - /// @dev Returns whether the array is strictly ascending (sorted and uniquified). - function isSortedAndUniquified(int256[] memory a) internal pure returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - result := 1 - if iszero(lt(mload(a), 2)) { - let end := add(a, shl(5, mload(a))) - for { a := add(a, 0x20) } 1 {} { - let p := mload(a) - a := add(a, 0x20) - result := slt(p, mload(a)) - if iszero(mul(result, xor(a, end))) { break } - } - } - } - } - - /// @dev Returns whether the array is strictly ascending (sorted and uniquified). - function isSortedAndUniquified(address[] memory a) internal pure returns (bool result) { - result = isSortedAndUniquified(_toUints(a)); - } - - /// @dev Returns the sorted set difference of `a` and `b`. - /// Note: Behaviour is undefined if inputs are not sorted and uniquified. - function difference(uint256[] memory a, uint256[] memory b) - internal - pure - returns (uint256[] memory c) - { - c = _difference(a, b, 0); - } - - /// @dev Returns the sorted set difference between `a` and `b`. - /// Note: Behaviour is undefined if inputs are not sorted and uniquified. - function difference(int256[] memory a, int256[] memory b) - internal - pure - returns (int256[] memory c) - { - c = _toInts(_difference(_toUints(a), _toUints(b), 1 << 255)); - } - - /// @dev Returns the sorted set difference between `a` and `b`. - /// Note: Behaviour is undefined if inputs are not sorted and uniquified. - function difference(address[] memory a, address[] memory b) - internal - pure - returns (address[] memory c) - { - c = _toAddresses(_difference(_toUints(a), _toUints(b), 0)); - } - - /// @dev Returns the sorted set intersection between `a` and `b`. - /// Note: Behaviour is undefined if inputs are not sorted and uniquified. - function intersection(uint256[] memory a, uint256[] memory b) - internal - pure - returns (uint256[] memory c) - { - c = _intersection(a, b, 0); - } - - /// @dev Returns the sorted set intersection between `a` and `b`. - /// Note: Behaviour is undefined if inputs are not sorted and uniquified. - function intersection(int256[] memory a, int256[] memory b) - internal - pure - returns (int256[] memory c) - { - c = _toInts(_intersection(_toUints(a), _toUints(b), 1 << 255)); - } - - /// @dev Returns the sorted set intersection between `a` and `b`. - /// Note: Behaviour is undefined if inputs are not sorted and uniquified. - function intersection(address[] memory a, address[] memory b) - internal - pure - returns (address[] memory c) - { - c = _toAddresses(_intersection(_toUints(a), _toUints(b), 0)); - } - - /// @dev Returns the sorted set union of `a` and `b`. - /// Note: Behaviour is undefined if inputs are not sorted and uniquified. - function union(uint256[] memory a, uint256[] memory b) - internal - pure - returns (uint256[] memory c) - { - c = _union(a, b, 0); - } - - /// @dev Returns the sorted set union of `a` and `b`. - /// Note: Behaviour is undefined if inputs are not sorted and uniquified. - function union(int256[] memory a, int256[] memory b) - internal - pure - returns (int256[] memory c) - { - c = _toInts(_union(_toUints(a), _toUints(b), 1 << 255)); - } - - /// @dev Returns the sorted set union between `a` and `b`. - /// Note: Behaviour is undefined if inputs are not sorted and uniquified. - function union(address[] memory a, address[] memory b) - internal - pure - returns (address[] memory c) - { - c = _toAddresses(_union(_toUints(a), _toUints(b), 0)); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PRIVATE HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Reinterpret cast to an uint256 array. - function _toUints(int256[] memory a) private pure returns (uint256[] memory casted) { - /// @solidity memory-safe-assembly - assembly { - casted := a - } - } - - /// @dev Reinterpret cast to an uint256 array. - function _toUints(address[] memory a) private pure returns (uint256[] memory casted) { - /// @solidity memory-safe-assembly - assembly { - // As any address written to memory will have the upper 96 bits - // of the word zeroized (as per Solidity spec), we can directly - // compare these addresses as if they are whole uint256 words. - casted := a - } - } - - /// @dev Reinterpret cast to an int array. - function _toInts(uint256[] memory a) private pure returns (int256[] memory casted) { - /// @solidity memory-safe-assembly - assembly { - casted := a - } - } - - /// @dev Reinterpret cast to an address array. - function _toAddresses(uint256[] memory a) private pure returns (address[] memory casted) { - /// @solidity memory-safe-assembly - assembly { - casted := a - } - } - - /// @dev Converts an array of signed integers to unsigned - /// integers suitable for sorting or vice versa. - function _flipSign(int256[] memory a) private pure { - /// @solidity memory-safe-assembly - assembly { - let w := shl(255, 1) - for { let end := add(a, shl(5, mload(a))) } iszero(eq(a, end)) {} { - a := add(a, 0x20) - mstore(a, add(mload(a), w)) - } - } - } - - /// @dev Returns whether `a` contains `needle`, and the index of `needle`. - /// `index` precedence: equal to > nearest before > nearest after. - function _searchSorted(uint256[] memory a, uint256 needle, uint256 signed) - private - pure - returns (bool found, uint256 index) - { - /// @solidity memory-safe-assembly - assembly { - let w := not(0) - let l := 1 - let h := mload(a) - let t := 0 - for { needle := add(signed, needle) } 1 {} { - index := shr(1, add(l, h)) - t := add(signed, mload(add(a, shl(5, index)))) - if or(gt(l, h), eq(t, needle)) { break } - // Decide whether to search the left or right half. - if iszero(gt(needle, t)) { - h := add(index, w) - continue - } - l := add(index, 1) - } - // `index` will be zero in the case of an empty array, - // or when the value is less than the smallest value in the array. - found := eq(t, needle) - t := iszero(iszero(index)) - index := mul(add(index, w), t) - found := and(found, t) - } - } - - /// @dev Returns the sorted set difference of `a` and `b`. - /// Note: Behaviour is undefined if inputs are not sorted and uniquified. - function _difference(uint256[] memory a, uint256[] memory b, uint256 signed) - private - pure - returns (uint256[] memory c) - { - /// @solidity memory-safe-assembly - assembly { - let s := 0x20 - let aEnd := add(a, shl(5, mload(a))) - let bEnd := add(b, shl(5, mload(b))) - c := mload(0x40) // Set `c` to the free memory pointer. - a := add(a, s) - b := add(b, s) - let k := c - for {} iszero(or(gt(a, aEnd), gt(b, bEnd))) {} { - let u := mload(a) - let v := mload(b) - if iszero(xor(u, v)) { - a := add(a, s) - b := add(b, s) - continue - } - if iszero(lt(add(u, signed), add(v, signed))) { - b := add(b, s) - continue - } - k := add(k, s) - mstore(k, u) - a := add(a, s) - } - for {} iszero(gt(a, aEnd)) {} { - k := add(k, s) - mstore(k, mload(a)) - a := add(a, s) - } - mstore(c, shr(5, sub(k, c))) // Store the length of `c`. - mstore(0x40, add(k, s)) // Allocate the memory for `c`. - } - } - - /// @dev Returns the sorted set intersection between `a` and `b`. - /// Note: Behaviour is undefined if inputs are not sorted and uniquified. - function _intersection(uint256[] memory a, uint256[] memory b, uint256 signed) - private - pure - returns (uint256[] memory c) - { - /// @solidity memory-safe-assembly - assembly { - let s := 0x20 - let aEnd := add(a, shl(5, mload(a))) - let bEnd := add(b, shl(5, mload(b))) - c := mload(0x40) // Set `c` to the free memory pointer. - a := add(a, s) - b := add(b, s) - let k := c - for {} iszero(or(gt(a, aEnd), gt(b, bEnd))) {} { - let u := mload(a) - let v := mload(b) - if iszero(xor(u, v)) { - k := add(k, s) - mstore(k, u) - a := add(a, s) - b := add(b, s) - continue - } - if iszero(lt(add(u, signed), add(v, signed))) { - b := add(b, s) - continue - } - a := add(a, s) - } - mstore(c, shr(5, sub(k, c))) // Store the length of `c`. - mstore(0x40, add(k, s)) // Allocate the memory for `c`. - } - } - - /// @dev Returns the sorted set union of `a` and `b`. - /// Note: Behaviour is undefined if inputs are not sorted and uniquified. - function _union(uint256[] memory a, uint256[] memory b, uint256 signed) - private - pure - returns (uint256[] memory c) - { - /// @solidity memory-safe-assembly - assembly { - let s := 0x20 - let aEnd := add(a, shl(5, mload(a))) - let bEnd := add(b, shl(5, mload(b))) - c := mload(0x40) // Set `c` to the free memory pointer. - a := add(a, s) - b := add(b, s) - let k := c - for {} iszero(or(gt(a, aEnd), gt(b, bEnd))) {} { - let u := mload(a) - let v := mload(b) - if iszero(xor(u, v)) { - k := add(k, s) - mstore(k, u) - a := add(a, s) - b := add(b, s) - continue - } - if iszero(lt(add(u, signed), add(v, signed))) { - k := add(k, s) - mstore(k, v) - b := add(b, s) - continue - } - k := add(k, s) - mstore(k, u) - a := add(a, s) - } - for {} iszero(gt(a, aEnd)) {} { - k := add(k, s) - mstore(k, mload(a)) - a := add(a, s) - } - for {} iszero(gt(b, bEnd)) {} { - k := add(k, s) - mstore(k, mload(b)) - b := add(b, s) - } - mstore(c, shr(5, sub(k, c))) // Store the length of `c`. - mstore(0x40, add(k, s)) // Allocate the memory for `c`. - } - } -} diff --git a/lib/solady/src/utils/LibString.sol b/lib/solady/src/utils/LibString.sol deleted file mode 100644 index 7f5cb60..0000000 --- a/lib/solady/src/utils/LibString.sol +++ /dev/null @@ -1,1163 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for converting numbers into strings and other string operations. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/LibString.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/LibString.sol) -library LibString { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The `length` of the output is too small to contain all the hex digits. - error HexLengthInsufficient(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The constant returned when the `search` is not found in the string. - uint256 internal constant NOT_FOUND = type(uint256).max; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* DECIMAL OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the base 10 decimal representation of `value`. - function toString(uint256 value) internal pure returns (string memory str) { - /// @solidity memory-safe-assembly - assembly { - // The maximum value of a uint256 contains 78 digits (1 byte per digit), but - // we allocate 0xa0 bytes to keep the free memory pointer 32-byte word aligned. - // We will need 1 word for the trailing zeros padding, 1 word for the length, - // and 3 words for a maximum of 78 digits. - str := add(mload(0x40), 0x80) - // Update the free memory pointer to allocate. - mstore(0x40, add(str, 0x20)) - // Zeroize the slot after the string. - mstore(str, 0) - - // Cache the end of the memory to calculate the length later. - let end := str - - let w := not(0) // Tsk. - // We write the string from rightmost digit to leftmost digit. - // The following is essentially a do-while loop that also handles the zero case. - for { let temp := value } 1 {} { - str := add(str, w) // `sub(str, 1)`. - // Write the character to the pointer. - // The ASCII index of the '0' character is 48. - mstore8(str, add(48, mod(temp, 10))) - // Keep dividing `temp` until zero. - temp := div(temp, 10) - if iszero(temp) { break } - } - - let length := sub(end, str) - // Move the pointer 32 bytes leftwards to make room for the length. - str := sub(str, 0x20) - // Store the length. - mstore(str, length) - } - } - - /// @dev Returns the base 10 decimal representation of `value`. - function toString(int256 value) internal pure returns (string memory str) { - if (value >= 0) { - return toString(uint256(value)); - } - unchecked { - str = toString(uint256(-value)); - } - /// @solidity memory-safe-assembly - assembly { - // We still have some spare memory space on the left, - // as we have allocated 3 words (96 bytes) for up to 78 digits. - let length := mload(str) // Load the string length. - mstore(str, 0x2d) // Store the '-' character. - str := sub(str, 1) // Move back the string pointer by a byte. - mstore(str, add(length, 1)) // Update the string length. - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* HEXADECIMAL OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the hexadecimal representation of `value`, - /// left-padded to an input length of `length` bytes. - /// The output is prefixed with "0x" encoded using 2 hexadecimal digits per byte, - /// giving a total length of `length * 2 + 2` bytes. - /// Reverts if `length` is too small for the output to contain all the digits. - function toHexString(uint256 value, uint256 length) internal pure returns (string memory str) { - str = toHexStringNoPrefix(value, length); - /// @solidity memory-safe-assembly - assembly { - let strLength := add(mload(str), 2) // Compute the length. - mstore(str, 0x3078) // Write the "0x" prefix. - str := sub(str, 2) // Move the pointer. - mstore(str, strLength) // Write the length. - } - } - - /// @dev Returns the hexadecimal representation of `value`, - /// left-padded to an input length of `length` bytes. - /// The output is prefixed with "0x" encoded using 2 hexadecimal digits per byte, - /// giving a total length of `length * 2` bytes. - /// Reverts if `length` is too small for the output to contain all the digits. - function toHexStringNoPrefix(uint256 value, uint256 length) - internal - pure - returns (string memory str) - { - /// @solidity memory-safe-assembly - assembly { - // We need 0x20 bytes for the trailing zeros padding, `length * 2` bytes - // for the digits, 0x02 bytes for the prefix, and 0x20 bytes for the length. - // We add 0x20 to the total and round down to a multiple of 0x20. - // (0x20 + 0x20 + 0x02 + 0x20) = 0x62. - str := add(mload(0x40), and(add(shl(1, length), 0x42), not(0x1f))) - // Allocate the memory. - mstore(0x40, add(str, 0x20)) - // Zeroize the slot after the string. - mstore(str, 0) - - // Cache the end to calculate the length later. - let end := str - // Store "0123456789abcdef" in scratch space. - mstore(0x0f, 0x30313233343536373839616263646566) - - let start := sub(str, add(length, length)) - let w := not(1) // Tsk. - let temp := value - // We write the string from rightmost digit to leftmost digit. - // The following is essentially a do-while loop that also handles the zero case. - for {} 1 {} { - str := add(str, w) // `sub(str, 2)`. - mstore8(add(str, 1), mload(and(temp, 15))) - mstore8(str, mload(and(shr(4, temp), 15))) - temp := shr(8, temp) - if iszero(xor(str, start)) { break } - } - - if temp { - // Store the function selector of `HexLengthInsufficient()`. - mstore(0x00, 0x2194895a) - // Revert with (offset, size). - revert(0x1c, 0x04) - } - - // Compute the string's length. - let strLength := sub(end, str) - // Move the pointer and write the length. - str := sub(str, 0x20) - mstore(str, strLength) - } - } - - /// @dev Returns the hexadecimal representation of `value`. - /// The output is prefixed with "0x" and encoded using 2 hexadecimal digits per byte. - /// As address are 20 bytes long, the output will left-padded to have - /// a length of `20 * 2 + 2` bytes. - function toHexString(uint256 value) internal pure returns (string memory str) { - str = toHexStringNoPrefix(value); - /// @solidity memory-safe-assembly - assembly { - let strLength := add(mload(str), 2) // Compute the length. - mstore(str, 0x3078) // Write the "0x" prefix. - str := sub(str, 2) // Move the pointer. - mstore(str, strLength) // Write the length. - } - } - - /// @dev Returns the hexadecimal representation of `value`. - /// The output is prefixed with "0x". - /// The output excludes leading "0" from the `toHexString` output. - /// `0x00: "0x0", 0x01: "0x1", 0x12: "0x12", 0x123: "0x123"`. - function toMinimalHexString(uint256 value) internal pure returns (string memory str) { - str = toHexStringNoPrefix(value); - /// @solidity memory-safe-assembly - assembly { - let o := eq(byte(0, mload(add(str, 0x20))), 0x30) // Whether leading zero is present. - let strLength := add(mload(str), 2) // Compute the length. - mstore(add(str, o), 0x3078) // Write the "0x" prefix, accounting for leading zero. - str := sub(add(str, o), 2) // Move the pointer, accounting for leading zero. - mstore(str, sub(strLength, o)) // Write the length, accounting for leading zero. - } - } - - /// @dev Returns the hexadecimal representation of `value`. - /// The output excludes leading "0" from the `toHexStringNoPrefix` output. - /// `0x00: "0", 0x01: "1", 0x12: "12", 0x123: "123"`. - function toMinimalHexStringNoPrefix(uint256 value) internal pure returns (string memory str) { - str = toHexStringNoPrefix(value); - /// @solidity memory-safe-assembly - assembly { - let o := eq(byte(0, mload(add(str, 0x20))), 0x30) // Whether leading zero is present. - let strLength := mload(str) // Get the length. - str := add(str, o) // Move the pointer, accounting for leading zero. - mstore(str, sub(strLength, o)) // Write the length, accounting for leading zero. - } - } - - /// @dev Returns the hexadecimal representation of `value`. - /// The output is encoded using 2 hexadecimal digits per byte. - /// As address are 20 bytes long, the output will left-padded to have - /// a length of `20 * 2` bytes. - function toHexStringNoPrefix(uint256 value) internal pure returns (string memory str) { - /// @solidity memory-safe-assembly - assembly { - // We need 0x20 bytes for the trailing zeros padding, 0x20 bytes for the length, - // 0x02 bytes for the prefix, and 0x40 bytes for the digits. - // The next multiple of 0x20 above (0x20 + 0x20 + 0x02 + 0x40) is 0xa0. - str := add(mload(0x40), 0x80) - // Allocate the memory. - mstore(0x40, add(str, 0x20)) - // Zeroize the slot after the string. - mstore(str, 0) - - // Cache the end to calculate the length later. - let end := str - // Store "0123456789abcdef" in scratch space. - mstore(0x0f, 0x30313233343536373839616263646566) - - let w := not(1) // Tsk. - // We write the string from rightmost digit to leftmost digit. - // The following is essentially a do-while loop that also handles the zero case. - for { let temp := value } 1 {} { - str := add(str, w) // `sub(str, 2)`. - mstore8(add(str, 1), mload(and(temp, 15))) - mstore8(str, mload(and(shr(4, temp), 15))) - temp := shr(8, temp) - if iszero(temp) { break } - } - - // Compute the string's length. - let strLength := sub(end, str) - // Move the pointer and write the length. - str := sub(str, 0x20) - mstore(str, strLength) - } - } - - /// @dev Returns the hexadecimal representation of `value`. - /// The output is prefixed with "0x", encoded using 2 hexadecimal digits per byte, - /// and the alphabets are capitalized conditionally according to - /// https://eips.ethereum.org/EIPS/eip-55 - function toHexStringChecksummed(address value) internal pure returns (string memory str) { - str = toHexString(value); - /// @solidity memory-safe-assembly - assembly { - let mask := shl(6, div(not(0), 255)) // `0b010000000100000000 ...` - let o := add(str, 0x22) - let hashed := and(keccak256(o, 40), mul(34, mask)) // `0b10001000 ... ` - let t := shl(240, 136) // `0b10001000 << 240` - for { let i := 0 } 1 {} { - mstore(add(i, i), mul(t, byte(i, hashed))) - i := add(i, 1) - if eq(i, 20) { break } - } - mstore(o, xor(mload(o), shr(1, and(mload(0x00), and(mload(o), mask))))) - o := add(o, 0x20) - mstore(o, xor(mload(o), shr(1, and(mload(0x20), and(mload(o), mask))))) - } - } - - /// @dev Returns the hexadecimal representation of `value`. - /// The output is prefixed with "0x" and encoded using 2 hexadecimal digits per byte. - function toHexString(address value) internal pure returns (string memory str) { - str = toHexStringNoPrefix(value); - /// @solidity memory-safe-assembly - assembly { - let strLength := add(mload(str), 2) // Compute the length. - mstore(str, 0x3078) // Write the "0x" prefix. - str := sub(str, 2) // Move the pointer. - mstore(str, strLength) // Write the length. - } - } - - /// @dev Returns the hexadecimal representation of `value`. - /// The output is encoded using 2 hexadecimal digits per byte. - function toHexStringNoPrefix(address value) internal pure returns (string memory str) { - /// @solidity memory-safe-assembly - assembly { - str := mload(0x40) - - // Allocate the memory. - // We need 0x20 bytes for the trailing zeros padding, 0x20 bytes for the length, - // 0x02 bytes for the prefix, and 0x28 bytes for the digits. - // The next multiple of 0x20 above (0x20 + 0x20 + 0x02 + 0x28) is 0x80. - mstore(0x40, add(str, 0x80)) - - // Store "0123456789abcdef" in scratch space. - mstore(0x0f, 0x30313233343536373839616263646566) - - str := add(str, 2) - mstore(str, 40) - - let o := add(str, 0x20) - mstore(add(o, 40), 0) - - value := shl(96, value) - - // We write the string from rightmost digit to leftmost digit. - // The following is essentially a do-while loop that also handles the zero case. - for { let i := 0 } 1 {} { - let p := add(o, add(i, i)) - let temp := byte(i, value) - mstore8(add(p, 1), mload(and(temp, 15))) - mstore8(p, mload(shr(4, temp))) - i := add(i, 1) - if eq(i, 20) { break } - } - } - } - - /// @dev Returns the hex encoded string from the raw bytes. - /// The output is encoded using 2 hexadecimal digits per byte. - function toHexString(bytes memory raw) internal pure returns (string memory str) { - str = toHexStringNoPrefix(raw); - /// @solidity memory-safe-assembly - assembly { - let strLength := add(mload(str), 2) // Compute the length. - mstore(str, 0x3078) // Write the "0x" prefix. - str := sub(str, 2) // Move the pointer. - mstore(str, strLength) // Write the length. - } - } - - /// @dev Returns the hex encoded string from the raw bytes. - /// The output is encoded using 2 hexadecimal digits per byte. - function toHexStringNoPrefix(bytes memory raw) internal pure returns (string memory str) { - /// @solidity memory-safe-assembly - assembly { - let length := mload(raw) - str := add(mload(0x40), 2) // Skip 2 bytes for the optional prefix. - mstore(str, add(length, length)) // Store the length of the output. - - // Store "0123456789abcdef" in scratch space. - mstore(0x0f, 0x30313233343536373839616263646566) - - let o := add(str, 0x20) - let end := add(raw, length) - - for {} iszero(eq(raw, end)) {} { - raw := add(raw, 1) - mstore8(add(o, 1), mload(and(mload(raw), 15))) - mstore8(o, mload(and(shr(4, mload(raw)), 15))) - o := add(o, 2) - } - mstore(o, 0) // Zeroize the slot after the string. - mstore(0x40, add(o, 0x20)) // Allocate the memory. - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* RUNE STRING OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the number of UTF characters in the string. - function runeCount(string memory s) internal pure returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - if mload(s) { - mstore(0x00, div(not(0), 255)) - mstore(0x20, 0x0202020202020202020202020202020202020202020202020303030304040506) - let o := add(s, 0x20) - let end := add(o, mload(s)) - for { result := 1 } 1 { result := add(result, 1) } { - o := add(o, byte(0, mload(shr(250, mload(o))))) - if iszero(lt(o, end)) { break } - } - } - } - } - - /// @dev Returns if this string is a 7-bit ASCII string. - /// (i.e. all characters codes are in [0..127]) - function is7BitASCII(string memory s) internal pure returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - let mask := shl(7, div(not(0), 255)) - result := 1 - let n := mload(s) - if n { - let o := add(s, 0x20) - let end := add(o, n) - let last := mload(end) - mstore(end, 0) - for {} 1 {} { - if and(mask, mload(o)) { - result := 0 - break - } - o := add(o, 0x20) - if iszero(lt(o, end)) { break } - } - mstore(end, last) - } - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* BYTE STRING OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // For performance and bytecode compactness, all indices of the following operations - // are byte (ASCII) offsets, not UTF character offsets. - - /// @dev Returns `subject` all occurrences of `search` replaced with `replacement`. - function replace(string memory subject, string memory search, string memory replacement) - internal - pure - returns (string memory result) - { - /// @solidity memory-safe-assembly - assembly { - let subjectLength := mload(subject) - let searchLength := mload(search) - let replacementLength := mload(replacement) - - subject := add(subject, 0x20) - search := add(search, 0x20) - replacement := add(replacement, 0x20) - result := add(mload(0x40), 0x20) - - let subjectEnd := add(subject, subjectLength) - if iszero(gt(searchLength, subjectLength)) { - let subjectSearchEnd := add(sub(subjectEnd, searchLength), 1) - let h := 0 - if iszero(lt(searchLength, 0x20)) { h := keccak256(search, searchLength) } - let m := shl(3, sub(0x20, and(searchLength, 0x1f))) - let s := mload(search) - for {} 1 {} { - let t := mload(subject) - // Whether the first `searchLength % 32` bytes of - // `subject` and `search` matches. - if iszero(shr(m, xor(t, s))) { - if h { - if iszero(eq(keccak256(subject, searchLength), h)) { - mstore(result, t) - result := add(result, 1) - subject := add(subject, 1) - if iszero(lt(subject, subjectSearchEnd)) { break } - continue - } - } - // Copy the `replacement` one word at a time. - for { let o := 0 } 1 {} { - mstore(add(result, o), mload(add(replacement, o))) - o := add(o, 0x20) - if iszero(lt(o, replacementLength)) { break } - } - result := add(result, replacementLength) - subject := add(subject, searchLength) - if searchLength { - if iszero(lt(subject, subjectSearchEnd)) { break } - continue - } - } - mstore(result, t) - result := add(result, 1) - subject := add(subject, 1) - if iszero(lt(subject, subjectSearchEnd)) { break } - } - } - - let resultRemainder := result - result := add(mload(0x40), 0x20) - let k := add(sub(resultRemainder, result), sub(subjectEnd, subject)) - // Copy the rest of the string one word at a time. - for {} lt(subject, subjectEnd) {} { - mstore(resultRemainder, mload(subject)) - resultRemainder := add(resultRemainder, 0x20) - subject := add(subject, 0x20) - } - result := sub(result, 0x20) - let last := add(add(result, 0x20), k) // Zeroize the slot after the string. - mstore(last, 0) - mstore(0x40, add(last, 0x20)) // Allocate the memory. - mstore(result, k) // Store the length. - } - } - - /// @dev Returns the byte index of the first location of `search` in `subject`, - /// searching from left to right, starting from `from`. - /// Returns `NOT_FOUND` (i.e. `type(uint256).max`) if the `search` is not found. - function indexOf(string memory subject, string memory search, uint256 from) - internal - pure - returns (uint256 result) - { - /// @solidity memory-safe-assembly - assembly { - for { let subjectLength := mload(subject) } 1 {} { - if iszero(mload(search)) { - if iszero(gt(from, subjectLength)) { - result := from - break - } - result := subjectLength - break - } - let searchLength := mload(search) - let subjectStart := add(subject, 0x20) - - result := not(0) // Initialize to `NOT_FOUND`. - - subject := add(subjectStart, from) - let end := add(sub(add(subjectStart, subjectLength), searchLength), 1) - - let m := shl(3, sub(0x20, and(searchLength, 0x1f))) - let s := mload(add(search, 0x20)) - - if iszero(and(lt(subject, end), lt(from, subjectLength))) { break } - - if iszero(lt(searchLength, 0x20)) { - for { let h := keccak256(add(search, 0x20), searchLength) } 1 {} { - if iszero(shr(m, xor(mload(subject), s))) { - if eq(keccak256(subject, searchLength), h) { - result := sub(subject, subjectStart) - break - } - } - subject := add(subject, 1) - if iszero(lt(subject, end)) { break } - } - break - } - for {} 1 {} { - if iszero(shr(m, xor(mload(subject), s))) { - result := sub(subject, subjectStart) - break - } - subject := add(subject, 1) - if iszero(lt(subject, end)) { break } - } - break - } - } - } - - /// @dev Returns the byte index of the first location of `search` in `subject`, - /// searching from left to right. - /// Returns `NOT_FOUND` (i.e. `type(uint256).max`) if the `search` is not found. - function indexOf(string memory subject, string memory search) - internal - pure - returns (uint256 result) - { - result = indexOf(subject, search, 0); - } - - /// @dev Returns the byte index of the first location of `search` in `subject`, - /// searching from right to left, starting from `from`. - /// Returns `NOT_FOUND` (i.e. `type(uint256).max`) if the `search` is not found. - function lastIndexOf(string memory subject, string memory search, uint256 from) - internal - pure - returns (uint256 result) - { - /// @solidity memory-safe-assembly - assembly { - for {} 1 {} { - result := not(0) // Initialize to `NOT_FOUND`. - let searchLength := mload(search) - if gt(searchLength, mload(subject)) { break } - let w := result - - let fromMax := sub(mload(subject), searchLength) - if iszero(gt(fromMax, from)) { from := fromMax } - - let end := add(add(subject, 0x20), w) - subject := add(add(subject, 0x20), from) - if iszero(gt(subject, end)) { break } - // As this function is not too often used, - // we shall simply use keccak256 for smaller bytecode size. - for { let h := keccak256(add(search, 0x20), searchLength) } 1 {} { - if eq(keccak256(subject, searchLength), h) { - result := sub(subject, add(end, 1)) - break - } - subject := add(subject, w) // `sub(subject, 1)`. - if iszero(gt(subject, end)) { break } - } - break - } - } - } - - /// @dev Returns the byte index of the first location of `search` in `subject`, - /// searching from right to left. - /// Returns `NOT_FOUND` (i.e. `type(uint256).max`) if the `search` is not found. - function lastIndexOf(string memory subject, string memory search) - internal - pure - returns (uint256 result) - { - result = lastIndexOf(subject, search, uint256(int256(-1))); - } - - /// @dev Returns true if `search` is found in `subject`, false otherwise. - function contains(string memory subject, string memory search) internal pure returns (bool) { - return indexOf(subject, search) != NOT_FOUND; - } - - /// @dev Returns whether `subject` starts with `search`. - function startsWith(string memory subject, string memory search) - internal - pure - returns (bool result) - { - /// @solidity memory-safe-assembly - assembly { - let searchLength := mload(search) - // Just using keccak256 directly is actually cheaper. - // forgefmt: disable-next-item - result := and( - iszero(gt(searchLength, mload(subject))), - eq( - keccak256(add(subject, 0x20), searchLength), - keccak256(add(search, 0x20), searchLength) - ) - ) - } - } - - /// @dev Returns whether `subject` ends with `search`. - function endsWith(string memory subject, string memory search) - internal - pure - returns (bool result) - { - /// @solidity memory-safe-assembly - assembly { - let searchLength := mload(search) - let subjectLength := mload(subject) - // Whether `search` is not longer than `subject`. - let withinRange := iszero(gt(searchLength, subjectLength)) - // Just using keccak256 directly is actually cheaper. - // forgefmt: disable-next-item - result := and( - withinRange, - eq( - keccak256( - // `subject + 0x20 + max(subjectLength - searchLength, 0)`. - add(add(subject, 0x20), mul(withinRange, sub(subjectLength, searchLength))), - searchLength - ), - keccak256(add(search, 0x20), searchLength) - ) - ) - } - } - - /// @dev Returns `subject` repeated `times`. - function repeat(string memory subject, uint256 times) - internal - pure - returns (string memory result) - { - /// @solidity memory-safe-assembly - assembly { - let subjectLength := mload(subject) - if iszero(or(iszero(times), iszero(subjectLength))) { - subject := add(subject, 0x20) - result := mload(0x40) - let output := add(result, 0x20) - for {} 1 {} { - // Copy the `subject` one word at a time. - for { let o := 0 } 1 {} { - mstore(add(output, o), mload(add(subject, o))) - o := add(o, 0x20) - if iszero(lt(o, subjectLength)) { break } - } - output := add(output, subjectLength) - times := sub(times, 1) - if iszero(times) { break } - } - mstore(output, 0) // Zeroize the slot after the string. - let resultLength := sub(output, add(result, 0x20)) - mstore(result, resultLength) // Store the length. - // Allocate the memory. - mstore(0x40, add(result, add(resultLength, 0x20))) - } - } - } - - /// @dev Returns a copy of `subject` sliced from `start` to `end` (exclusive). - /// `start` and `end` are byte offsets. - function slice(string memory subject, uint256 start, uint256 end) - internal - pure - returns (string memory result) - { - /// @solidity memory-safe-assembly - assembly { - let subjectLength := mload(subject) - if iszero(gt(subjectLength, end)) { end := subjectLength } - if iszero(gt(subjectLength, start)) { start := subjectLength } - if lt(start, end) { - result := mload(0x40) - let resultLength := sub(end, start) - mstore(result, resultLength) - subject := add(subject, start) - let w := not(0x1f) - // Copy the `subject` one word at a time, backwards. - for { let o := and(add(resultLength, 0x1f), w) } 1 {} { - mstore(add(result, o), mload(add(subject, o))) - o := add(o, w) // `sub(o, 0x20)`. - if iszero(o) { break } - } - // Zeroize the slot after the string. - mstore(add(add(result, 0x20), resultLength), 0) - // Allocate memory for the length and the bytes, - // rounded up to a multiple of 32. - mstore(0x40, add(result, and(add(resultLength, 0x3f), w))) - } - } - } - - /// @dev Returns a copy of `subject` sliced from `start` to the end of the string. - /// `start` is a byte offset. - function slice(string memory subject, uint256 start) - internal - pure - returns (string memory result) - { - result = slice(subject, start, uint256(int256(-1))); - } - - /// @dev Returns all the indices of `search` in `subject`. - /// The indices are byte offsets. - function indicesOf(string memory subject, string memory search) - internal - pure - returns (uint256[] memory result) - { - /// @solidity memory-safe-assembly - assembly { - let subjectLength := mload(subject) - let searchLength := mload(search) - - if iszero(gt(searchLength, subjectLength)) { - subject := add(subject, 0x20) - search := add(search, 0x20) - result := add(mload(0x40), 0x20) - - let subjectStart := subject - let subjectSearchEnd := add(sub(add(subject, subjectLength), searchLength), 1) - let h := 0 - if iszero(lt(searchLength, 0x20)) { h := keccak256(search, searchLength) } - let m := shl(3, sub(0x20, and(searchLength, 0x1f))) - let s := mload(search) - for {} 1 {} { - let t := mload(subject) - // Whether the first `searchLength % 32` bytes of - // `subject` and `search` matches. - if iszero(shr(m, xor(t, s))) { - if h { - if iszero(eq(keccak256(subject, searchLength), h)) { - subject := add(subject, 1) - if iszero(lt(subject, subjectSearchEnd)) { break } - continue - } - } - // Append to `result`. - mstore(result, sub(subject, subjectStart)) - result := add(result, 0x20) - // Advance `subject` by `searchLength`. - subject := add(subject, searchLength) - if searchLength { - if iszero(lt(subject, subjectSearchEnd)) { break } - continue - } - } - subject := add(subject, 1) - if iszero(lt(subject, subjectSearchEnd)) { break } - } - let resultEnd := result - // Assign `result` to the free memory pointer. - result := mload(0x40) - // Store the length of `result`. - mstore(result, shr(5, sub(resultEnd, add(result, 0x20)))) - // Allocate memory for result. - // We allocate one more word, so this array can be recycled for {split}. - mstore(0x40, add(resultEnd, 0x20)) - } - } - } - - /// @dev Returns a arrays of strings based on the `delimiter` inside of the `subject` string. - function split(string memory subject, string memory delimiter) - internal - pure - returns (string[] memory result) - { - uint256[] memory indices = indicesOf(subject, delimiter); - /// @solidity memory-safe-assembly - assembly { - let w := not(0x1f) - let indexPtr := add(indices, 0x20) - let indicesEnd := add(indexPtr, shl(5, add(mload(indices), 1))) - mstore(add(indicesEnd, w), mload(subject)) - mstore(indices, add(mload(indices), 1)) - let prevIndex := 0 - for {} 1 {} { - let index := mload(indexPtr) - mstore(indexPtr, 0x60) - if iszero(eq(index, prevIndex)) { - let element := mload(0x40) - let elementLength := sub(index, prevIndex) - mstore(element, elementLength) - // Copy the `subject` one word at a time, backwards. - for { let o := and(add(elementLength, 0x1f), w) } 1 {} { - mstore(add(element, o), mload(add(add(subject, prevIndex), o))) - o := add(o, w) // `sub(o, 0x20)`. - if iszero(o) { break } - } - // Zeroize the slot after the string. - mstore(add(add(element, 0x20), elementLength), 0) - // Allocate memory for the length and the bytes, - // rounded up to a multiple of 32. - mstore(0x40, add(element, and(add(elementLength, 0x3f), w))) - // Store the `element` into the array. - mstore(indexPtr, element) - } - prevIndex := add(index, mload(delimiter)) - indexPtr := add(indexPtr, 0x20) - if iszero(lt(indexPtr, indicesEnd)) { break } - } - result := indices - if iszero(mload(delimiter)) { - result := add(indices, 0x20) - mstore(result, sub(mload(indices), 2)) - } - } - } - - /// @dev Returns a concatenated string of `a` and `b`. - /// Cheaper than `string.concat()` and does not de-align the free memory pointer. - function concat(string memory a, string memory b) - internal - pure - returns (string memory result) - { - /// @solidity memory-safe-assembly - assembly { - let w := not(0x1f) - result := mload(0x40) - let aLength := mload(a) - // Copy `a` one word at a time, backwards. - for { let o := and(add(aLength, 0x20), w) } 1 {} { - mstore(add(result, o), mload(add(a, o))) - o := add(o, w) // `sub(o, 0x20)`. - if iszero(o) { break } - } - let bLength := mload(b) - let output := add(result, aLength) - // Copy `b` one word at a time, backwards. - for { let o := and(add(bLength, 0x20), w) } 1 {} { - mstore(add(output, o), mload(add(b, o))) - o := add(o, w) // `sub(o, 0x20)`. - if iszero(o) { break } - } - let totalLength := add(aLength, bLength) - let last := add(add(result, 0x20), totalLength) - // Zeroize the slot after the string. - mstore(last, 0) - // Stores the length. - mstore(result, totalLength) - // Allocate memory for the length and the bytes, - // rounded up to a multiple of 32. - mstore(0x40, and(add(last, 0x1f), w)) - } - } - - /// @dev Returns a copy of the string in either lowercase or UPPERCASE. - /// WARNING! This function is only compatible with 7-bit ASCII strings. - function toCase(string memory subject, bool toUpper) - internal - pure - returns (string memory result) - { - /// @solidity memory-safe-assembly - assembly { - let length := mload(subject) - if length { - result := add(mload(0x40), 0x20) - subject := add(subject, 1) - let flags := shl(add(70, shl(5, toUpper)), 0x3ffffff) - let w := not(0) - for { let o := length } 1 {} { - o := add(o, w) - let b := and(0xff, mload(add(subject, o))) - mstore8(add(result, o), xor(b, and(shr(b, flags), 0x20))) - if iszero(o) { break } - } - result := mload(0x40) - mstore(result, length) // Store the length. - let last := add(add(result, 0x20), length) - mstore(last, 0) // Zeroize the slot after the string. - mstore(0x40, add(last, 0x20)) // Allocate the memory. - } - } - } - - /// @dev Returns a string from a small bytes32 string. - /// `smallString` must be null terminated, or behavior will be undefined. - function fromSmallString(bytes32 smallString) internal pure returns (string memory result) { - if (smallString == bytes32(0)) return result; - /// @solidity memory-safe-assembly - assembly { - result := mload(0x40) - let n := 0 - for {} 1 {} { - n := add(n, 1) - if iszero(byte(n, smallString)) { break } // Scan for '\0'. - } - mstore(result, n) - let o := add(result, 0x20) - mstore(o, smallString) - mstore(add(o, n), 0) - mstore(0x40, add(result, 0x40)) - } - } - - /// @dev Returns a lowercased copy of the string. - /// WARNING! This function is only compatible with 7-bit ASCII strings. - function lower(string memory subject) internal pure returns (string memory result) { - result = toCase(subject, false); - } - - /// @dev Returns an UPPERCASED copy of the string. - /// WARNING! This function is only compatible with 7-bit ASCII strings. - function upper(string memory subject) internal pure returns (string memory result) { - result = toCase(subject, true); - } - - /// @dev Escapes the string to be used within HTML tags. - function escapeHTML(string memory s) internal pure returns (string memory result) { - /// @solidity memory-safe-assembly - assembly { - let end := add(s, mload(s)) - result := add(mload(0x40), 0x20) - // Store the bytes of the packed offsets and strides into the scratch space. - // `packed = (stride << 5) | offset`. Max offset is 20. Max stride is 6. - mstore(0x1f, 0x900094) - mstore(0x08, 0xc0000000a6ab) - // Store ""&'<>" into the scratch space. - mstore(0x00, shl(64, 0x2671756f743b26616d703b262333393b266c743b2667743b)) - for {} iszero(eq(s, end)) {} { - s := add(s, 1) - let c := and(mload(s), 0xff) - // Not in `["\"","'","&","<",">"]`. - if iszero(and(shl(c, 1), 0x500000c400000000)) { - mstore8(result, c) - result := add(result, 1) - continue - } - let t := shr(248, mload(c)) - mstore(result, mload(and(t, 0x1f))) - result := add(result, shr(5, t)) - } - let last := result - mstore(last, 0) // Zeroize the slot after the string. - result := mload(0x40) - mstore(result, sub(last, add(result, 0x20))) // Store the length. - mstore(0x40, add(last, 0x20)) // Allocate the memory. - } - } - - /// @dev Escapes the string to be used within double-quotes in a JSON. - /// If `addDoubleQuotes` is true, the result will be enclosed in double-quotes. - function escapeJSON(string memory s, bool addDoubleQuotes) - internal - pure - returns (string memory result) - { - /// @solidity memory-safe-assembly - assembly { - let end := add(s, mload(s)) - result := add(mload(0x40), 0x20) - if addDoubleQuotes { - mstore8(result, 34) - result := add(1, result) - } - // Store "\\u0000" in scratch space. - // Store "0123456789abcdef" in scratch space. - // Also, store `{0x08:"b", 0x09:"t", 0x0a:"n", 0x0c:"f", 0x0d:"r"}`. - // into the scratch space. - mstore(0x15, 0x5c75303030303031323334353637383961626364656662746e006672) - // Bitmask for detecting `["\"","\\"]`. - let e := or(shl(0x22, 1), shl(0x5c, 1)) - for {} iszero(eq(s, end)) {} { - s := add(s, 1) - let c := and(mload(s), 0xff) - if iszero(lt(c, 0x20)) { - if iszero(and(shl(c, 1), e)) { - // Not in `["\"","\\"]`. - mstore8(result, c) - result := add(result, 1) - continue - } - mstore8(result, 0x5c) // "\\". - mstore8(add(result, 1), c) - result := add(result, 2) - continue - } - if iszero(and(shl(c, 1), 0x3700)) { - // Not in `["\b","\t","\n","\f","\d"]`. - mstore8(0x1d, mload(shr(4, c))) // Hex value. - mstore8(0x1e, mload(and(c, 15))) // Hex value. - mstore(result, mload(0x19)) // "\\u00XX". - result := add(result, 6) - continue - } - mstore8(result, 0x5c) // "\\". - mstore8(add(result, 1), mload(add(c, 8))) - result := add(result, 2) - } - if addDoubleQuotes { - mstore8(result, 34) - result := add(1, result) - } - let last := result - mstore(last, 0) // Zeroize the slot after the string. - result := mload(0x40) - mstore(result, sub(last, add(result, 0x20))) // Store the length. - mstore(0x40, add(last, 0x20)) // Allocate the memory. - } - } - - /// @dev Escapes the string to be used within double-quotes in a JSON. - function escapeJSON(string memory s) internal pure returns (string memory result) { - result = escapeJSON(s, false); - } - - /// @dev Returns whether `a` equals `b`. - function eq(string memory a, string memory b) internal pure returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - result := eq(keccak256(add(a, 0x20), mload(a)), keccak256(add(b, 0x20), mload(b))) - } - } - - /// @dev Returns whether `a` equals `b`. For small strings up to 32 bytes. - /// `b` must be null terminated, or behavior will be undefined. - function eqs(string memory a, bytes32 b) internal pure returns (bool result) { - /// @solidity memory-safe-assembly - assembly { - // These should be evaluated on compile time, as far as possible. - let x := and(b, add(not(b), 1)) - let r := or(shl(8, iszero(b)), shl(7, iszero(iszero(shr(128, x))))) - r := or(r, shl(6, iszero(iszero(shr(64, shr(r, x)))))) - r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) - r := or(r, shl(4, lt(0xffff, shr(r, x)))) - r := or(r, shl(3, lt(0xff, shr(r, x)))) - result := gt(eq(mload(a), sub(32, shr(3, r))), shr(r, xor(b, mload(add(a, 0x20))))) - } - } - - /// @dev Packs a single string with its length into a single word. - /// Returns `bytes32(0)` if the length is zero or greater than 31. - function packOne(string memory a) internal pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - // We don't need to zero right pad the string, - // since this is our own custom non-standard packing scheme. - result := - mul( - // Load the length and the bytes. - mload(add(a, 0x1f)), - // `length != 0 && length < 32`. Abuses underflow. - // Assumes that the length is valid and within the block gas limit. - lt(sub(mload(a), 1), 0x1f) - ) - } - } - - /// @dev Unpacks a string packed using {packOne}. - /// Returns the empty string if `packed` is `bytes32(0)`. - /// If `packed` is not an output of {packOne}, the output behavior is undefined. - function unpackOne(bytes32 packed) internal pure returns (string memory result) { - /// @solidity memory-safe-assembly - assembly { - // Grab the free memory pointer. - result := mload(0x40) - // Allocate 2 words (1 for the length, 1 for the bytes). - mstore(0x40, add(result, 0x40)) - // Zeroize the length slot. - mstore(result, 0) - // Store the length and bytes. - mstore(add(result, 0x1f), packed) - // Right pad with zeroes. - mstore(add(add(result, 0x20), mload(result)), 0) - } - } - - /// @dev Packs two strings with their lengths into a single word. - /// Returns `bytes32(0)` if combined length is zero or greater than 30. - function packTwo(string memory a, string memory b) internal pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - let aLength := mload(a) - // We don't need to zero right pad the strings, - // since this is our own custom non-standard packing scheme. - result := - mul( - // Load the length and the bytes of `a` and `b`. - or( - shl(shl(3, sub(0x1f, aLength)), mload(add(a, aLength))), - mload(sub(add(b, 0x1e), aLength)) - ), - // `totalLength != 0 && totalLength < 31`. Abuses underflow. - // Assumes that the lengths are valid and within the block gas limit. - lt(sub(add(aLength, mload(b)), 1), 0x1e) - ) - } - } - - /// @dev Unpacks strings packed using {packTwo}. - /// Returns the empty strings if `packed` is `bytes32(0)`. - /// If `packed` is not an output of {packTwo}, the output behavior is undefined. - function unpackTwo(bytes32 packed) - internal - pure - returns (string memory resultA, string memory resultB) - { - /// @solidity memory-safe-assembly - assembly { - // Grab the free memory pointer. - resultA := mload(0x40) - resultB := add(resultA, 0x40) - // Allocate 2 words for each string (1 for the length, 1 for the byte). Total 4 words. - mstore(0x40, add(resultB, 0x40)) - // Zeroize the length slots. - mstore(resultA, 0) - mstore(resultB, 0) - // Store the lengths and bytes. - mstore(add(resultA, 0x1f), packed) - mstore(add(resultB, 0x1f), mload(add(add(resultA, 0x20), mload(resultA)))) - // Right pad with zeroes. - mstore(add(add(resultA, 0x20), mload(resultA)), 0) - mstore(add(add(resultB, 0x20), mload(resultB)), 0) - } - } - - /// @dev Directly returns `a` without copying. - function directReturn(string memory a) internal pure { - assembly { - // Assumes that the string does not start from the scratch space. - let retStart := sub(a, 0x20) - let retSize := add(mload(a), 0x40) - // Right pad with zeroes. Just in case the string is produced - // by a method that doesn't zero right pad. - mstore(add(retStart, retSize), 0) - // Store the return offset. - mstore(retStart, 0x20) - // End the transaction, returning the string. - return(retStart, retSize) - } - } -} diff --git a/lib/solady/src/utils/LibZip.sol b/lib/solady/src/utils/LibZip.sol deleted file mode 100644 index 0fe8d82..0000000 --- a/lib/solady/src/utils/LibZip.sol +++ /dev/null @@ -1,279 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for compressing and decompressing bytes. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/LibZip.sol) -/// @author Calldata compression by clabby (https://github.com/clabby/op-kompressor) -/// @author FastLZ by ariya (https://github.com/ariya/FastLZ) -/// -/// @dev Note: -/// The accompanying solady.js library includes implementations of -/// FastLZ and calldata operations for convenience. -library LibZip { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* FAST LZ OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // LZ77 implementation based on FastLZ. - // Equivalent to level 1 compression and decompression at the following commit: - // https://github.com/ariya/FastLZ/commit/344eb4025f9ae866ebf7a2ec48850f7113a97a42 - // Decompression is backwards compatible. - - /// @dev Returns the compressed `data`. - function flzCompress(bytes memory data) internal pure returns (bytes memory result) { - /// @solidity memory-safe-assembly - assembly { - function ms8(d_, v_) -> _d { - mstore8(d_, v_) - _d := add(d_, 1) - } - function u24(p_) -> _u { - let w := mload(p_) - _u := or(shl(16, byte(2, w)), or(shl(8, byte(1, w)), byte(0, w))) - } - function cmp(p_, q_, e_) -> _l { - for { e_ := sub(e_, q_) } lt(_l, e_) { _l := add(_l, 1) } { - e_ := mul(iszero(byte(0, xor(mload(add(p_, _l)), mload(add(q_, _l))))), e_) - } - } - function literals(runs_, src_, dest_) -> _o { - for { _o := dest_ } iszero(lt(runs_, 0x20)) { runs_ := sub(runs_, 0x20) } { - mstore(ms8(_o, 31), mload(src_)) - _o := add(_o, 0x21) - src_ := add(src_, 0x20) - } - if iszero(runs_) { leave } - mstore(ms8(_o, sub(runs_, 1)), mload(src_)) - _o := add(1, add(_o, runs_)) - } - function match(l_, d_, o_) -> _o { - for { d_ := sub(d_, 1) } iszero(lt(l_, 263)) { l_ := sub(l_, 262) } { - o_ := ms8(ms8(ms8(o_, add(224, shr(8, d_))), 253), and(0xff, d_)) - } - if iszero(lt(l_, 7)) { - _o := ms8(ms8(ms8(o_, add(224, shr(8, d_))), sub(l_, 7)), and(0xff, d_)) - leave - } - _o := ms8(ms8(o_, add(shl(5, l_), shr(8, d_))), and(0xff, d_)) - } - function setHash(i_, v_) { - let p := add(mload(0x40), shl(2, i_)) - mstore(p, xor(mload(p), shl(224, xor(shr(224, mload(p)), v_)))) - } - function getHash(i_) -> _h { - _h := shr(224, mload(add(mload(0x40), shl(2, i_)))) - } - function hash(v_) -> _r { - _r := and(shr(19, mul(2654435769, v_)), 0x1fff) - } - function setNextHash(ip_, ipStart_) -> _ip { - setHash(hash(u24(ip_)), sub(ip_, ipStart_)) - _ip := add(ip_, 1) - } - codecopy(mload(0x40), codesize(), 0x8000) // Zeroize the hashmap. - let op := add(mload(0x40), 0x8000) - let a := add(data, 0x20) - let ipStart := a - let ipLimit := sub(add(ipStart, mload(data)), 13) - for { let ip := add(2, a) } lt(ip, ipLimit) {} { - let r := 0 - let d := 0 - for {} 1 {} { - let s := u24(ip) - let h := hash(s) - r := add(ipStart, getHash(h)) - setHash(h, sub(ip, ipStart)) - d := sub(ip, r) - if iszero(lt(ip, ipLimit)) { break } - ip := add(ip, 1) - if iszero(gt(d, 0x1fff)) { if eq(s, u24(r)) { break } } - } - if iszero(lt(ip, ipLimit)) { break } - ip := sub(ip, 1) - if gt(ip, a) { op := literals(sub(ip, a), a, op) } - let l := cmp(add(r, 3), add(ip, 3), add(ipLimit, 9)) - op := match(l, d, op) - ip := setNextHash(setNextHash(add(ip, l), ipStart), ipStart) - a := ip - } - op := literals(sub(add(ipStart, mload(data)), a), a, op) - result := mload(0x40) - let t := add(result, 0x8000) - let n := sub(op, t) - mstore(result, n) // Store the length. - // Copy the result to compact the memory, overwriting the hashmap. - let o := add(result, 0x20) - for { let i } lt(i, n) { i := add(i, 0x20) } { mstore(add(o, i), mload(add(t, i))) } - mstore(add(o, n), 0) // Zeroize the slot after the string. - mstore(0x40, add(add(o, n), 0x20)) // Allocate the memory. - } - } - - /// @dev Returns the decompressed `data`. - function flzDecompress(bytes memory data) internal pure returns (bytes memory result) { - /// @solidity memory-safe-assembly - assembly { - let n := 0 - let end := add(add(data, 0x20), mload(data)) - result := mload(0x40) - let op := add(result, 0x20) - for { data := add(data, 0x20) } lt(data, end) {} { - let w := mload(data) - let c := byte(0, w) - let t := shr(5, c) - if iszero(t) { - mstore(add(op, n), mload(add(data, 1))) - data := add(data, add(2, c)) - n := add(n, add(1, c)) - continue - } - let g := eq(t, 7) - let l := add(2, xor(t, mul(g, xor(t, add(7, byte(1, w)))))) - for { - let s := add(add(shl(8, and(0x1f, c)), byte(add(1, g), w)), 1) - let r := add(op, sub(n, s)) - let o := add(op, n) - let f := xor(s, mul(gt(s, 0x20), xor(s, 0x20))) - let j := 0 - } 1 {} { - mstore(add(o, j), mload(add(r, j))) - j := add(j, f) - if iszero(lt(j, l)) { break } - } - data := add(data, add(2, g)) - n := add(n, l) - } - mstore(result, n) // Store the length. - let o := add(add(result, 0x20), n) - mstore(o, 0) // Zeroize the slot after the string. - mstore(0x40, add(o, 0x20)) // Allocate the memory. - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CALLDATA OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // Calldata compression and decompression using selective run length encoding: - // - Sequences of 0x00 (up to 128 consecutive). - // - Sequences of 0xff (up to 32 consecutive). - // - // A run length encoded block consists of two bytes: - // (0) 0x00 - // (1) A control byte with the following bit layout: - // - [7] `0: 0x00, 1: 0xff`. - // - [0..6] `runLength - 1`. - // - // The first 4 bytes are bitwise negated so that the compressed calldata - // can be dispatched into the `fallback` and `receive` functions. - - /// @dev Returns the compressed `data`. - function cdCompress(bytes memory data) internal pure returns (bytes memory result) { - /// @solidity memory-safe-assembly - assembly { - function rle(v_, o_, d_) -> _o, _d { - mstore(o_, shl(240, or(and(0xff, add(d_, 0xff)), and(0x80, v_)))) - _o := add(o_, 2) - } - result := mload(0x40) - let o := add(result, 0x20) - let z := 0 // Number of consecutive 0x00. - let y := 0 // Number of consecutive 0xff. - for { let end := add(data, mload(data)) } iszero(eq(data, end)) {} { - data := add(data, 1) - let c := byte(31, mload(data)) - if iszero(c) { - if y { o, y := rle(0xff, o, y) } - z := add(z, 1) - if eq(z, 0x80) { o, z := rle(0x00, o, 0x80) } - continue - } - if eq(c, 0xff) { - if z { o, z := rle(0x00, o, z) } - y := add(y, 1) - if eq(y, 0x20) { o, y := rle(0xff, o, 0x20) } - continue - } - if y { o, y := rle(0xff, o, y) } - if z { o, z := rle(0x00, o, z) } - mstore8(o, c) - o := add(o, 1) - } - if y { o, y := rle(0xff, o, y) } - if z { o, z := rle(0x00, o, z) } - // Bitwise negate the first 4 bytes. - mstore(add(result, 4), not(mload(add(result, 4)))) - mstore(result, sub(o, add(result, 0x20))) // Store the length. - mstore(o, 0) // Zeroize the slot after the string. - mstore(0x40, add(o, 0x20)) // Allocate the memory. - } - } - - /// @dev Returns the decompressed `data`. - function cdDecompress(bytes memory data) internal pure returns (bytes memory result) { - /// @solidity memory-safe-assembly - assembly { - if mload(data) { - result := mload(0x40) - let o := add(result, 0x20) - let s := add(data, 4) - let v := mload(s) - let end := add(data, mload(data)) - mstore(s, not(v)) // Bitwise negate the first 4 bytes. - for {} lt(data, end) {} { - data := add(data, 1) - let c := byte(31, mload(data)) - if iszero(c) { - data := add(data, 1) - let d := byte(31, mload(data)) - // Fill with either 0xff or 0x00. - mstore(o, not(0)) - if iszero(gt(d, 0x7f)) { codecopy(o, codesize(), add(d, 1)) } - o := add(o, add(and(d, 0x7f), 1)) - continue - } - mstore8(o, c) - o := add(o, 1) - } - mstore(s, v) // Restore the first 4 bytes. - mstore(result, sub(o, add(result, 0x20))) // Store the length. - mstore(o, 0) // Zeroize the slot after the string. - mstore(0x40, add(o, 0x20)) // Allocate the memory. - } - } - } - - /// @dev To be called in the `receive` and `fallback` functions. - /// ``` - /// receive() external payable { LibZip.cdFallback(); } - /// fallback() external payable { LibZip.cdFallback(); } - /// ``` - /// For efficiency, this function will directly return the results, terminating the context. - /// If called internally, it must be called at the end of the function. - function cdFallback() internal { - assembly { - if iszero(calldatasize()) { return(calldatasize(), calldatasize()) } - let o := 0 - let f := not(3) // For negating the first 4 bytes. - for { let i := 0 } lt(i, calldatasize()) {} { - let c := byte(0, xor(add(i, f), calldataload(i))) - i := add(i, 1) - if iszero(c) { - let d := byte(0, xor(add(i, f), calldataload(i))) - i := add(i, 1) - // Fill with either 0xff or 0x00. - mstore(o, not(0)) - if iszero(gt(d, 0x7f)) { codecopy(o, codesize(), add(d, 1)) } - o := add(o, add(and(d, 0x7f), 1)) - continue - } - mstore8(o, c) - o := add(o, 1) - } - let success := delegatecall(gas(), address(), 0x00, o, codesize(), 0x00) - returndatacopy(0x00, 0x00, returndatasize()) - if iszero(success) { revert(0x00, returndatasize()) } - return(0x00, returndatasize()) - } - } -} diff --git a/lib/solady/src/utils/MerkleProofLib.sol b/lib/solady/src/utils/MerkleProofLib.sol deleted file mode 100644 index 967eb98..0000000 --- a/lib/solady/src/utils/MerkleProofLib.sol +++ /dev/null @@ -1,309 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Gas optimized verification of proof of inclusion for a leaf in a Merkle tree. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/MerkleProofLib.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/MerkleProofLib.sol) -/// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/cryptography/MerkleProof.sol) -library MerkleProofLib { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* MERKLE PROOF VERIFICATION OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns whether `leaf` exists in the Merkle tree with `root`, given `proof`. - function verify(bytes32[] memory proof, bytes32 root, bytes32 leaf) - internal - pure - returns (bool isValid) - { - /// @solidity memory-safe-assembly - assembly { - if mload(proof) { - // Initialize `offset` to the offset of `proof` elements in memory. - let offset := add(proof, 0x20) - // Left shift by 5 is equivalent to multiplying by 0x20. - let end := add(offset, shl(5, mload(proof))) - // Iterate over proof elements to compute root hash. - for {} 1 {} { - // Slot of `leaf` in scratch space. - // If the condition is true: 0x20, otherwise: 0x00. - let scratch := shl(5, gt(leaf, mload(offset))) - // Store elements to hash contiguously in scratch space. - // Scratch space is 64 bytes (0x00 - 0x3f) and both elements are 32 bytes. - mstore(scratch, leaf) - mstore(xor(scratch, 0x20), mload(offset)) - // Reuse `leaf` to store the hash to reduce stack operations. - leaf := keccak256(0x00, 0x40) - offset := add(offset, 0x20) - if iszero(lt(offset, end)) { break } - } - } - isValid := eq(leaf, root) - } - } - - /// @dev Returns whether `leaf` exists in the Merkle tree with `root`, given `proof`. - function verifyCalldata(bytes32[] calldata proof, bytes32 root, bytes32 leaf) - internal - pure - returns (bool isValid) - { - /// @solidity memory-safe-assembly - assembly { - if proof.length { - // Left shift by 5 is equivalent to multiplying by 0x20. - let end := add(proof.offset, shl(5, proof.length)) - // Initialize `offset` to the offset of `proof` in the calldata. - let offset := proof.offset - // Iterate over proof elements to compute root hash. - for {} 1 {} { - // Slot of `leaf` in scratch space. - // If the condition is true: 0x20, otherwise: 0x00. - let scratch := shl(5, gt(leaf, calldataload(offset))) - // Store elements to hash contiguously in scratch space. - // Scratch space is 64 bytes (0x00 - 0x3f) and both elements are 32 bytes. - mstore(scratch, leaf) - mstore(xor(scratch, 0x20), calldataload(offset)) - // Reuse `leaf` to store the hash to reduce stack operations. - leaf := keccak256(0x00, 0x40) - offset := add(offset, 0x20) - if iszero(lt(offset, end)) { break } - } - } - isValid := eq(leaf, root) - } - } - - /// @dev Returns whether all `leaves` exist in the Merkle tree with `root`, - /// given `proof` and `flags`. - /// - /// Note: - /// - Breaking the invariant `flags.length == (leaves.length - 1) + proof.length` - /// will always return false. - /// - The sum of the lengths of `proof` and `leaves` must never overflow. - /// - Any non-zero word in the `flags` array is treated as true. - /// - The memory offset of `proof` must be non-zero - /// (i.e. `proof` is not pointing to the scratch space). - function verifyMultiProof( - bytes32[] memory proof, - bytes32 root, - bytes32[] memory leaves, - bool[] memory flags - ) internal pure returns (bool isValid) { - // Rebuilds the root by consuming and producing values on a queue. - // The queue starts with the `leaves` array, and goes into a `hashes` array. - // After the process, the last element on the queue is verified - // to be equal to the `root`. - // - // The `flags` array denotes whether the sibling - // should be popped from the queue (`flag == true`), or - // should be popped from the `proof` (`flag == false`). - /// @solidity memory-safe-assembly - assembly { - // Cache the lengths of the arrays. - let leavesLength := mload(leaves) - let proofLength := mload(proof) - let flagsLength := mload(flags) - - // Advance the pointers of the arrays to point to the data. - leaves := add(0x20, leaves) - proof := add(0x20, proof) - flags := add(0x20, flags) - - // If the number of flags is correct. - for {} eq(add(leavesLength, proofLength), add(flagsLength, 1)) {} { - // For the case where `proof.length + leaves.length == 1`. - if iszero(flagsLength) { - // `isValid = (proof.length == 1 ? proof[0] : leaves[0]) == root`. - isValid := eq(mload(xor(leaves, mul(xor(proof, leaves), proofLength))), root) - break - } - - // The required final proof offset if `flagsLength` is not zero, otherwise zero. - let proofEnd := add(proof, shl(5, proofLength)) - // We can use the free memory space for the queue. - // We don't need to allocate, since the queue is temporary. - let hashesFront := mload(0x40) - // Copy the leaves into the hashes. - // Sometimes, a little memory expansion costs less than branching. - // Should cost less, even with a high free memory offset of 0x7d00. - leavesLength := shl(5, leavesLength) - for { let i := 0 } iszero(eq(i, leavesLength)) { i := add(i, 0x20) } { - mstore(add(hashesFront, i), mload(add(leaves, i))) - } - // Compute the back of the hashes. - let hashesBack := add(hashesFront, leavesLength) - // This is the end of the memory for the queue. - // We recycle `flagsLength` to save on stack variables (sometimes save gas). - flagsLength := add(hashesBack, shl(5, flagsLength)) - - for {} 1 {} { - // Pop from `hashes`. - let a := mload(hashesFront) - // Pop from `hashes`. - let b := mload(add(hashesFront, 0x20)) - hashesFront := add(hashesFront, 0x40) - - // If the flag is false, load the next proof, - // else, pops from the queue. - if iszero(mload(flags)) { - // Loads the next proof. - b := mload(proof) - proof := add(proof, 0x20) - // Unpop from `hashes`. - hashesFront := sub(hashesFront, 0x20) - } - - // Advance to the next flag. - flags := add(flags, 0x20) - - // Slot of `a` in scratch space. - // If the condition is true: 0x20, otherwise: 0x00. - let scratch := shl(5, gt(a, b)) - // Hash the scratch space and push the result onto the queue. - mstore(scratch, a) - mstore(xor(scratch, 0x20), b) - mstore(hashesBack, keccak256(0x00, 0x40)) - hashesBack := add(hashesBack, 0x20) - if iszero(lt(hashesBack, flagsLength)) { break } - } - isValid := - and( - // Checks if the last value in the queue is same as the root. - eq(mload(sub(hashesBack, 0x20)), root), - // And whether all the proofs are used, if required. - eq(proofEnd, proof) - ) - break - } - } - } - - /// @dev Returns whether all `leaves` exist in the Merkle tree with `root`, - /// given `proof` and `flags`. - /// - /// Note: - /// - Breaking the invariant `flags.length == (leaves.length - 1) + proof.length` - /// will always return false. - /// - Any non-zero word in the `flags` array is treated as true. - /// - The calldata offset of `proof` must be non-zero - /// (i.e. `proof` is from a regular Solidity function with a 4-byte selector). - function verifyMultiProofCalldata( - bytes32[] calldata proof, - bytes32 root, - bytes32[] calldata leaves, - bool[] calldata flags - ) internal pure returns (bool isValid) { - // Rebuilds the root by consuming and producing values on a queue. - // The queue starts with the `leaves` array, and goes into a `hashes` array. - // After the process, the last element on the queue is verified - // to be equal to the `root`. - // - // The `flags` array denotes whether the sibling - // should be popped from the queue (`flag == true`), or - // should be popped from the `proof` (`flag == false`). - /// @solidity memory-safe-assembly - assembly { - // If the number of flags is correct. - for {} eq(add(leaves.length, proof.length), add(flags.length, 1)) {} { - // For the case where `proof.length + leaves.length == 1`. - if iszero(flags.length) { - // `isValid = (proof.length == 1 ? proof[0] : leaves[0]) == root`. - // forgefmt: disable-next-item - isValid := eq( - calldataload( - xor(leaves.offset, mul(xor(proof.offset, leaves.offset), proof.length)) - ), - root - ) - break - } - - // The required final proof offset if `flagsLength` is not zero, otherwise zero. - let proofEnd := add(proof.offset, shl(5, proof.length)) - // We can use the free memory space for the queue. - // We don't need to allocate, since the queue is temporary. - let hashesFront := mload(0x40) - // Copy the leaves into the hashes. - // Sometimes, a little memory expansion costs less than branching. - // Should cost less, even with a high free memory offset of 0x7d00. - calldatacopy(hashesFront, leaves.offset, shl(5, leaves.length)) - // Compute the back of the hashes. - let hashesBack := add(hashesFront, shl(5, leaves.length)) - // This is the end of the memory for the queue. - // We recycle `flagsLength` to save on stack variables (sometimes save gas). - flags.length := add(hashesBack, shl(5, flags.length)) - - // We don't need to make a copy of `proof.offset` or `flags.offset`, - // as they are pass-by-value (this trick may not always save gas). - - for {} 1 {} { - // Pop from `hashes`. - let a := mload(hashesFront) - // Pop from `hashes`. - let b := mload(add(hashesFront, 0x20)) - hashesFront := add(hashesFront, 0x40) - - // If the flag is false, load the next proof, - // else, pops from the queue. - if iszero(calldataload(flags.offset)) { - // Loads the next proof. - b := calldataload(proof.offset) - proof.offset := add(proof.offset, 0x20) - // Unpop from `hashes`. - hashesFront := sub(hashesFront, 0x20) - } - - // Advance to the next flag offset. - flags.offset := add(flags.offset, 0x20) - - // Slot of `a` in scratch space. - // If the condition is true: 0x20, otherwise: 0x00. - let scratch := shl(5, gt(a, b)) - // Hash the scratch space and push the result onto the queue. - mstore(scratch, a) - mstore(xor(scratch, 0x20), b) - mstore(hashesBack, keccak256(0x00, 0x40)) - hashesBack := add(hashesBack, 0x20) - if iszero(lt(hashesBack, flags.length)) { break } - } - isValid := - and( - // Checks if the last value in the queue is same as the root. - eq(mload(sub(hashesBack, 0x20)), root), - // And whether all the proofs are used, if required. - eq(proofEnd, proof.offset) - ) - break - } - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EMPTY CALLDATA HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns an empty calldata bytes32 array. - function emptyProof() internal pure returns (bytes32[] calldata proof) { - /// @solidity memory-safe-assembly - assembly { - proof.length := 0 - } - } - - /// @dev Returns an empty calldata bytes32 array. - function emptyLeaves() internal pure returns (bytes32[] calldata leaves) { - /// @solidity memory-safe-assembly - assembly { - leaves.length := 0 - } - } - - /// @dev Returns an empty calldata bool array. - function emptyFlags() internal pure returns (bool[] calldata flags) { - /// @solidity memory-safe-assembly - assembly { - flags.length := 0 - } - } -} diff --git a/lib/solady/src/utils/MetadataReaderLib.sol b/lib/solady/src/utils/MetadataReaderLib.sol deleted file mode 100644 index e44c249..0000000 --- a/lib/solady/src/utils/MetadataReaderLib.sol +++ /dev/null @@ -1,150 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for reading contract metadata robustly. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/MetadataReaderLib.sol) -library MetadataReaderLib { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* METADATA READING OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // Best-effort string reading operations. - // Should NOT revert as long as sufficient gas is provided. - // - // Performs the following in order: - // 1. Returns the empty string for the following cases: - // - Reverts. - // - No returndata (e.g. function returns nothing, EOA). - // - Returns empty string. - // 2. Attempts to `abi.decode` the returndata into a string. - // 3. With any remaining gas, scans the returndata from start to end for the - // null byte '\0', to interpret the returndata as a null-terminated string. - - /// @dev Equivalent to `readString(abi.encodeWithSignature("name()"), limit)`. - function readName(address target, uint256 limit) internal view returns (string memory) { - return _string(target, _ptr(0x06fdde03), limit); - } - - /// @dev Equivalent to `readString(abi.encodeWithSignature("symbol()"), limit)`. - function readSymbol(address target, uint256 limit) internal view returns (string memory) { - return _string(target, _ptr(0x95d89b41), limit); - } - - /// @dev Performs a best-effort string query on `target` with `data` as the calldata. - /// The string will be truncated to `limit` bytes. - function readString(address target, bytes memory data, uint256 limit) - internal - view - returns (string memory) - { - return _string(target, _ptr(data), limit); - } - - // Best-effort unsigned integer reading operations. - // Should NOT revert as long as sufficient gas is provided. - // - // Performs the following in order: - // 1. Attempts to `abi.decode` the result into a uint256 - // (equivalent across all Solidity uint types, downcast as needed). - // 2. Returns zero for the following cases: - // - Reverts. - // - No returndata (e.g. function returns nothing, EOA). - // - Returns zero. - // - `abi.decode` failure. - - /// @dev Equivalent to `uint8(readUint(abi.encodeWithSignature("decimal()")))`. - function readDecimals(address target) internal view returns (uint8) { - return uint8(_uint(target, _ptr(0x313ce567))); - } - - /// @dev Performs a best-effort uint query on `target` with `data` as the calldata. - function readUint(address target, bytes memory data) internal view returns (uint256) { - return _uint(target, _ptr(data)); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PRIVATE HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Attempts to read and return a string at `target`. - function _string(address target, bytes32 ptr, uint256 limit) - private - view - returns (string memory result) - { - /// @solidity memory-safe-assembly - assembly { - function min(x_, y_) -> _z { - _z := xor(x_, mul(xor(x_, y_), lt(y_, x_))) - } - for {} staticcall(gas(), target, add(ptr, 0x20), mload(ptr), 0x00, 0x20) {} { - let m := mload(0x40) // Grab the free memory pointer. - let s := add(0x20, m) // Start of the string's bytes in memory. - // Attempt to `abi.decode` if the returndatasize is greater or equal to 64. - if iszero(lt(returndatasize(), 0x40)) { - let o := mload(0x00) // Load the string's offset in the returndata. - // If the string's offset is within bounds. - if iszero(gt(o, sub(returndatasize(), 0x20))) { - returndatacopy(m, o, 0x20) // Copy the string's length. - // If the full string's end is within bounds. - // Note: If the full string doesn't fit, the `abi.decode` must be aborted - // for compliance purposes, regardless if the truncated string can fit. - if iszero(gt(mload(m), sub(returndatasize(), add(o, 0x20)))) { - let n := min(mload(m), limit) // Truncate if needed. - mstore(m, n) // Overwrite the length. - returndatacopy(s, add(o, 0x20), n) // Copy the string's bytes. - mstore(add(s, n), 0) // Zeroize the slot after the string. - mstore(0x40, add(0x20, add(s, n))) // Allocate memory for the string. - result := m - break - } - } - } - // Try interpreting as a null-terminated string. - let n := min(returndatasize(), limit) // Truncate if needed. - returndatacopy(s, 0, n) // Copy the string's bytes. - mstore8(add(s, n), 0) // Place a '\0' at the end. - let i := s // Pointer to the next byte to scan. - for {} byte(0, mload(i)) { i := add(i, 1) } {} // Scan for '\0'. - mstore(m, sub(i, s)) // Store the string's length. - mstore(i, 0) // Zeroize the slot after the string. - mstore(0x40, add(0x20, i)) // Allocate memory for the string. - result := m - break - } - } - } - - /// @dev Attempts to read and return a uint at `target`. - function _uint(address target, bytes32 ptr) private view returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - result := - mul( - mload(0x20), - and( // The arguments of `and` are evaluated from right to left. - gt(returndatasize(), 0x1f), // At least 32 bytes returned. - staticcall(gas(), target, add(ptr, 0x20), mload(ptr), 0x20, 0x20) - ) - ) - } - } - - /// @dev Casts the function selector `s` into a pointer. - function _ptr(uint256 s) private pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - // Layout the calldata in the scratch space for temporary usage. - mstore(0x04, s) // Store the function selector. - mstore(result, 4) // Store the length. - } - } - - /// @dev Casts the `data` into a pointer. - function _ptr(bytes memory data) private pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - result := data - } - } -} diff --git a/lib/solady/src/utils/MinHeapLib.sol b/lib/solady/src/utils/MinHeapLib.sol deleted file mode 100644 index 6facfa9..0000000 --- a/lib/solady/src/utils/MinHeapLib.sol +++ /dev/null @@ -1,204 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for managing a min-heap in storage. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/MinHeapLib.sol) -library MinHeapLib { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The heap is empty. - error HeapIsEmpty(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STRUCTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev A heap in storage. - struct Heap { - uint256[] data; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // Tips: - // - To use as a max-map, negate the values. - // - If use on tuples, pack the tuple values into a single integer. - // - To use on signed integers, convert the signed integers into - // their ordered unsigned counterparts via `uint256(x) + (1 << 255)`. - - /// @dev Returns the minimum value of the heap. - /// Reverts if the heap is empty. - function root(Heap storage heap) internal view returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - if iszero(sload(heap.slot)) { - mstore(0x00, 0xa6ca772e) // Store the function selector of `HeapIsEmpty()`. - revert(0x1c, 0x04) // Revert with (offset, size). - } - mstore(0x00, heap.slot) - result := sload(keccak256(0x00, 0x20)) - } - } - - /// @dev Returns the number of items in the heap. - function length(Heap storage heap) internal view returns (uint256) { - return heap.data.length; - } - - /// @dev Pushes the `value` onto the min-heap. - function push(Heap storage heap, uint256 value) internal { - _set(heap, value, 0, 4); - } - - /// @dev Pops the minimum value from the min-heap. - /// Reverts if the heap is empty. - function pop(Heap storage heap) internal returns (uint256 popped) { - (,, popped) = _set(heap, 0, 0, 3); - } - - /// @dev Pushes the `value` onto the min-heap, and pops the minimum value. - function pushPop(Heap storage heap, uint256 value) internal returns (uint256 popped) { - (,, popped) = _set(heap, value, 0, 2); - } - - /// @dev Pops the minimum value, and pushes the new `value` onto the min-heap. - /// Reverts if the heap is empty. - function replace(Heap storage heap, uint256 value) internal returns (uint256 popped) { - (,, popped) = _set(heap, value, 0, 1); - } - - /// @dev Pushes the `value` onto the min-heap, and pops the minimum value - /// if the length of the heap exceeds `maxLength`. - /// - /// Reverts if `maxLength` is zero. - /// - /// - If the queue is not full: - /// (`success` = true, `hasPopped` = false, `popped` = 0) - /// - If the queue is full, and `value` is not greater than the minimum value: - /// (`success` = false, `hasPopped` = false, `popped` = 0) - /// - If the queue is full, and `value` is greater than the minimum value: - /// (`success` = true, `hasPopped` = true, `popped` = ) - /// - /// Useful for implementing a bounded priority queue. - function enqueue(Heap storage heap, uint256 value, uint256 maxLength) - internal - returns (bool success, bool hasPopped, uint256 popped) - { - (success, hasPopped, popped) = _set(heap, value, maxLength, 0); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PRIVATE HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Helper function for heap operations. - /// Designed for code conciseness, bytecode compactness, and decent performance. - function _set(Heap storage heap, uint256 value, uint256 maxLength, uint256 mode) - private - returns (bool success, bool hasPopped, uint256 popped) - { - /// @solidity memory-safe-assembly - assembly { - let n := sload(heap.slot) - // Compute the array storage slot offset. - mstore(0x00, heap.slot) - let sOffset := keccak256(0x00, 0x20) - - let pos := 0 - let childPos := not(0) - // Operations are ordered from most likely usage to least likely usage. - for {} 1 { - mstore(0x00, 0xa6ca772e) // Store the function selector of `HeapIsEmpty()`. - revert(0x1c, 0x04) // Revert with (offset, size). - } { - // `enqueue`. - if iszero(mode) { - if iszero(maxLength) { continue } - if iszero(eq(n, maxLength)) { - // If queue is not full. - success := 1 - // Increment and update the length. - pos := n - sstore(heap.slot, add(pos, 1)) - childPos := add(childPos, childPos) - break - } - let r := sload(sOffset) - if iszero(lt(r, value)) { break } - success := 1 - hasPopped := 1 - childPos := 1 - popped := r - break - } - // `replace`. - if eq(mode, 1) { - if iszero(n) { continue } - popped := sload(sOffset) - childPos := 1 - break - } - // `pushPop`. - if eq(mode, 2) { - popped := value - if iszero(n) { break } - let r := sload(sOffset) - if iszero(lt(r, value)) { break } - popped := r - childPos := 1 - break - } - // `pop`. - if eq(mode, 3) { - if iszero(n) { continue } - // Decrement and update the length. - n := sub(n, 1) - sstore(heap.slot, n) - // Set the `value` to the last item. - value := sload(add(sOffset, n)) - popped := value - if iszero(n) { break } - popped := sload(sOffset) - childPos := 1 - break - } - // `push`. - { - // Increment and update the length. - pos := n - sstore(heap.slot, add(pos, 1)) - childPos := add(childPos, childPos) - break - } - } - - for {} lt(childPos, n) {} { - let child := sload(add(sOffset, childPos)) - let rightPos := add(childPos, 1) - let right := sload(add(sOffset, rightPos)) - if iszero(and(lt(rightPos, n), iszero(lt(child, right)))) { - right := child - rightPos := childPos - } - sstore(add(sOffset, pos), right) - pos := rightPos - childPos := add(shl(1, pos), 1) - } - - for {} pos {} { - let parentPos := shr(1, sub(pos, 1)) - let parent := sload(add(sOffset, parentPos)) - if iszero(lt(value, parent)) { break } - sstore(add(sOffset, pos), parent) - pos := parentPos - } - - // If `childPos` is not `not(0)`. - if add(childPos, 1) { sstore(add(sOffset, pos), value) } - } - } -} diff --git a/lib/solady/src/utils/Multicallable.sol b/lib/solady/src/utils/Multicallable.sol deleted file mode 100644 index 1c0a5c4..0000000 --- a/lib/solady/src/utils/Multicallable.sol +++ /dev/null @@ -1,66 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Contract that enables a single call to call multiple methods on itself. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/Multicallable.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/Multicallable.sol) -abstract contract Multicallable { - /// @dev Apply `DELEGATECALL` with the current contract to each calldata in `data`, - /// and store the `abi.encode` formatted results of each `DELEGATECALL` into `results`. - /// If any of the `DELEGATECALL`s reverts, the entire context is reverted, - /// and the error is bubbled up. - /// - /// This function is deliberately made non-payable to guard against double-spending. - /// (See: https://www.paradigm.xyz/2021/08/two-rights-might-make-a-wrong) - /// - /// For efficiency, this function will directly return the results, terminating the context. - /// If called internally, it must be called at the end of a function - /// that returns `(bytes[] memory)`. - function multicall(bytes[] calldata data) public virtual returns (bytes[] memory) { - assembly { - mstore(0x00, 0x20) - mstore(0x20, data.length) // Store `data.length` into `results`. - // Early return if no data. - if iszero(data.length) { return(0x00, 0x40) } - - let results := 0x40 - // `shl` 5 is equivalent to multiplying by 0x20. - let end := shl(5, data.length) - // Copy the offsets from calldata into memory. - calldatacopy(0x40, data.offset, end) - // Offset into `results`. - let resultsOffset := end - // Pointer to the end of `results`. - end := add(results, end) - - for {} 1 {} { - // The offset of the current bytes in the calldata. - let o := add(data.offset, mload(results)) - let m := add(resultsOffset, 0x40) - // Copy the current bytes from calldata to the memory. - calldatacopy( - m, - add(o, 0x20), // The offset of the current bytes' bytes. - calldataload(o) // The length of the current bytes. - ) - if iszero(delegatecall(gas(), address(), m, calldataload(o), codesize(), 0x00)) { - // Bubble up the revert if the delegatecall reverts. - returndatacopy(0x00, 0x00, returndatasize()) - revert(0x00, returndatasize()) - } - // Append the current `resultsOffset` into `results`. - mstore(results, resultsOffset) - results := add(results, 0x20) - // Append the `returndatasize()`, and the return data. - mstore(m, returndatasize()) - returndatacopy(add(m, 0x20), 0x00, returndatasize()) - // Advance the `resultsOffset` by `returndatasize() + 0x20`, - // rounded up to the next multiple of 32. - resultsOffset := - and(add(add(resultsOffset, returndatasize()), 0x3f), 0xffffffffffffffe0) - if iszero(lt(results, end)) { break } - } - return(0x00, add(resultsOffset, 0x40)) - } - } -} diff --git a/lib/solady/src/utils/RedBlackTreeLib.sol b/lib/solady/src/utils/RedBlackTreeLib.sol deleted file mode 100644 index 6ed7dfb..0000000 --- a/lib/solady/src/utils/RedBlackTreeLib.sol +++ /dev/null @@ -1,729 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Library for managing a red-black-tree in storage. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/RedBlackTreeLib.sol) -/// @author Modified from BokkyPooBahsRedBlackTreeLibrary -/// (https://github.com/bokkypoobah/BokkyPooBahsRedBlackTreeLibrary) -/// @dev This red-black-tree does not support the zero (i.e. empty) value. -library RedBlackTreeLib { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The value cannot be zero. - error ValueIsEmpty(); - - /// @dev Cannot insert a value that already exists. - error ValueAlreadyExists(); - - /// @dev Cannot remove a value that does not exist. - error ValueDoesNotExist(); - - /// @dev The pointer is out of bounds. - error PointerOutOfBounds(); - - /// @dev The tree is full. - error TreeIsFull(); - - /// @dev `bytes4(keccak256(bytes("ValueAlreadyExists()")))`. - uint256 internal constant ERROR_VALUE_ALREADY_EXISTS = 0xbb33e6ac; - - /// @dev `bytes4(keccak256(bytes("ValueDoesNotExist()")))`. - uint256 internal constant ERROR_VALUE_DOES_NOT_EXISTS = 0xb113638a; - - /// @dev `bytes4(keccak256(bytes("PointerOutOfBounds()")))`. - uint256 internal constant ERROR_POINTER_OUT_OF_BOUNDS = 0xccd52fbc; - - /// @dev `bytes4(keccak256(bytes("TreeIsFull()")))`. - uint256 internal constant ERROR_TREE_IS_FULL = 0xed732d0c; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STRUCTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev A red-black-tree in storage. - struct Tree { - uint256 _spacer; - } - - // Custom storage: - // ``` - // mstore(0x20, tree.slot) - // mstore(0x00, _NODES_SLOT_SEED) - // let nodes := shl(_NODES_SLOT_SHIFT, keccak256(0x00, 0x40)) - // - // let root := shr(128, sload(nodes)) - // let totalNodes := and(sload(nodes), _BITMASK_KEY) - // - // let nodePacked := sload(or(nodes, nodeIndex)) - // let nodeLeft := and(nodePacked, _BITMASK_KEY) - // let nodeRight := and(shr(_BITPOS_RIGHT, nodePacked), _BITMASK_KEY) - // let nodeParent := and(shr(_BITPOS_PARENT, nodePacked), _BITMASK_KEY) - // let nodeRed := and(shr(_BITPOS_RED, nodePacked), 1) - // - // let nodeValue := shr(_BITPOS_PACKED_VALUE, nodePacked) - // if iszero(nodeValue) { - // nodeValue := sload(or(_BIT_FULL_VALUE_SLOT, or(nodes, nodeIndex))) - // } - // ``` - - uint256 private constant _NODES_SLOT_SEED = 0x1dc27bb5462fdadcb; - uint256 private constant _NODES_SLOT_SHIFT = 32; - uint256 private constant _BITMASK_KEY = (1 << 31) - 1; - uint256 private constant _BITPOS_LEFT = 0; - uint256 private constant _BITPOS_RIGHT = 31; - uint256 private constant _BITPOS_PARENT = 31 * 2; - uint256 private constant _BITPOS_RED = 31 * 3; - uint256 private constant _BITMASK_RED = 1 << (31 * 3); - uint256 private constant _BITPOS_PACKED_VALUE = 96; - uint256 private constant _BITMASK_PACKED_VALUE = (1 << 160) - 1; - uint256 private constant _BIT_FULL_VALUE_SLOT = 1 << 31; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns the number of unique values in the tree. - function size(Tree storage tree) internal view returns (uint256 result) { - uint256 nodes = _nodes(tree); - /// @solidity memory-safe-assembly - assembly { - result := and(sload(nodes), _BITMASK_KEY) - } - } - - /// @dev Returns a pointer to the value `x`. - /// If the value `x` is not in the tree, the returned pointer will be empty. - function find(Tree storage tree, uint256 x) internal view returns (bytes32 result) { - (uint256 nodes,, uint256 key) = _find(tree, x); - result = _pack(nodes, key); - } - - /// @dev Returns a pointer to the nearest value to `x`. - /// In a tie-breaker, the returned pointer will point to the smaller value. - /// If the tree is empty, the returned pointer will be empty. - function nearest(Tree storage tree, uint256 x) internal view returns (bytes32 result) { - (uint256 nodes, uint256 cursor, uint256 key) = _find(tree, x); - unchecked { - if (cursor == 0) return result; // Nothing found -- empty tree. - if (key != 0) return _pack(nodes, key); // Exact match. - bytes32 a = _pack(nodes, cursor); - uint256 aValue = value(a); - bytes32 b = x < aValue ? prev(a) : next(a); - if (b == bytes32(0)) return a; // Only node found. - uint256 bValue = value(b); - uint256 aDist = x < aValue ? aValue - x : x - aValue; - uint256 bDist = x < bValue ? bValue - x : x - bValue; - if (aDist == bDist) return aValue < bValue ? a : b; // Tie-breaker. - return aDist < bDist ? a : b; - } - } - - /// @dev Returns a pointer to the nearest value lesser or equal to `x`. - /// If there is no value lesser or equal to `x`, the returned pointer will be empty. - function nearestBefore(Tree storage tree, uint256 x) internal view returns (bytes32 result) { - (uint256 nodes, uint256 cursor, uint256 key) = _find(tree, x); - if (cursor == 0) return result; // Nothing found -- empty tree. - if (key != 0) return _pack(nodes, key); // Exact match. - bytes32 a = _pack(nodes, cursor); - return value(a) < x ? a : prev(a); - } - - /// @dev Returns a pointer to the nearest value greater or equal to `x`. - /// If there is no value greater or equal to `x`, the returned pointer will be empty. - function nearestAfter(Tree storage tree, uint256 x) internal view returns (bytes32 result) { - (uint256 nodes, uint256 cursor, uint256 key) = _find(tree, x); - if (cursor == 0) return result; // Nothing found -- empty tree. - if (key != 0) return _pack(nodes, key); // Exact match. - bytes32 a = _pack(nodes, cursor); - return value(a) > x ? a : next(a); - } - - /// @dev Returns whether the value `x` exists. - function exists(Tree storage tree, uint256 x) internal view returns (bool result) { - (,, uint256 key) = _find(tree, x); - result = key != 0; - } - - /// @dev Inserts the value `x` into the tree. - /// Reverts if the value `x` already exists. - function insert(Tree storage tree, uint256 x) internal { - uint256 err = tryInsert(tree, x); - if (err != 0) _revert(err); - } - - /// @dev Inserts the value `x` into the tree. - /// Returns a non-zero error code upon failure instead of reverting - /// (except for reverting if `x` is an empty value). - function tryInsert(Tree storage tree, uint256 x) internal returns (uint256 err) { - (uint256 nodes, uint256 cursor, uint256 key) = _find(tree, x); - err = _update(nodes, cursor, key, x, 0); - } - - /// @dev Removes the value `x` from the tree. - /// Reverts if the value does not exist. - function remove(Tree storage tree, uint256 x) internal { - uint256 err = tryRemove(tree, x); - if (err != 0) _revert(err); - } - - /// @dev Removes the value `x` from the tree. - /// Returns a non-zero error code upon failure instead of reverting - /// (except for reverting if `x` is an empty value). - function tryRemove(Tree storage tree, uint256 x) internal returns (uint256 err) { - (uint256 nodes,, uint256 key) = _find(tree, x); - err = _update(nodes, 0, key, 0, 1); - } - - /// @dev Removes the value at pointer `ptr` from the tree. - /// Reverts if `ptr` is empty (i.e. value does not exist), - /// or if `ptr` is out of bounds. - /// After removal, `ptr` may point to another existing value. - /// For safety, do not reuse `ptr` after calling remove on it. - function remove(bytes32 ptr) internal { - uint256 err = tryRemove(ptr); - if (err != 0) _revert(err); - } - - /// @dev Removes the value at pointer `ptr` from the tree. - /// Returns a non-zero error code upon failure instead of reverting. - function tryRemove(bytes32 ptr) internal returns (uint256 err) { - (uint256 nodes, uint256 key) = _unpack(ptr); - err = _update(nodes, 0, key, 0, 1); - } - - /// @dev Clears the entire tree. All data will be deleted from storage. - function clear(Tree storage tree) internal { - uint256 nodes = _nodes(tree); - /// @solidity memory-safe-assembly - assembly { - let totalNodes := and(sload(nodes), _BITMASK_KEY) - for { let i := 1 } iszero(gt(i, totalNodes)) { i := add(i, 1) } { - let ptr := or(nodes, i) - if iszero(shr(_BITPOS_PACKED_VALUE, sload(ptr))) { - sstore(or(ptr, _BIT_FULL_VALUE_SLOT), 0) - } - sstore(ptr, 0) - } - sstore(nodes, 0) - } - } - - /// @dev Returns the value at pointer `ptr`. - /// If `ptr` is empty, the result will be zero. - function value(bytes32 ptr) internal view returns (uint256 result) { - if (ptr == bytes32(0)) return result; - /// @solidity memory-safe-assembly - assembly { - let packed := sload(ptr) - result := shr(_BITPOS_PACKED_VALUE, packed) - if iszero(result) { result := sload(or(ptr, _BIT_FULL_VALUE_SLOT)) } - } - } - - /// @dev Returns a pointer to the smallest value in the tree. - /// If the tree is empty, the returned pointer will be empty. - function first(Tree storage tree) internal view returns (bytes32 result) { - result = _end(tree, _BITPOS_LEFT); - } - - /// @dev Returns a pointer to the largest value in the tree. - /// If the tree is empty, the returned pointer will be empty. - function last(Tree storage tree) internal view returns (bytes32 result) { - result = _end(tree, _BITPOS_RIGHT); - } - - /// @dev Returns the pointer to the next largest value. - /// If there is no next value, or if `ptr` is empty, - /// the returned pointer will be empty. - function next(bytes32 ptr) internal view returns (bytes32 result) { - result = _step(ptr, _BITPOS_LEFT, _BITPOS_RIGHT); - } - - /// @dev Returns the pointer to the next smallest value. - /// If there is no previous value, or if `ptr` is empty, - /// the returned pointer will be empty. - function prev(bytes32 ptr) internal view returns (bytes32 result) { - result = _step(ptr, _BITPOS_RIGHT, _BITPOS_LEFT); - } - - /// @dev Returns whether the pointer is empty. - function isEmpty(bytes32 ptr) internal pure returns (bool result) { - result = ptr == bytes32(0); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PRIVATE HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Unpacks the pointer `ptr` to its components. - function _unpack(bytes32 ptr) private pure returns (uint256 nodes, uint256 key) { - /// @solidity memory-safe-assembly - assembly { - nodes := shl(_NODES_SLOT_SHIFT, shr(_NODES_SLOT_SHIFT, ptr)) - key := and(_BITMASK_KEY, ptr) - } - } - - /// @dev Packs `nodes` and `key` into a single pointer. - function _pack(uint256 nodes, uint256 key) private pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - result := mul(or(nodes, key), iszero(iszero(key))) - } - } - - /// @dev Returns the pointer to either end of the tree. - function _end(Tree storage tree, uint256 L) private view returns (bytes32 result) { - uint256 nodes = _nodes(tree); - /// @solidity memory-safe-assembly - assembly { - result := shr(128, sload(nodes)) - if result { - for {} 1 {} { - let packed := sload(or(nodes, result)) - let left := and(shr(L, packed), _BITMASK_KEY) - if iszero(left) { break } - result := left - } - } - } - result = _pack(nodes, uint256(result)); - } - - /// @dev Step the pointer `ptr` forwards or backwards. - function _step(bytes32 ptr, uint256 L, uint256 R) private view returns (bytes32 result) { - if (ptr == bytes32(0)) return ptr; - (uint256 nodes, uint256 target) = _unpack(ptr); - /// @solidity memory-safe-assembly - assembly { - let packed := sload(ptr) - for { result := and(shr(R, packed), _BITMASK_KEY) } 1 {} { - if iszero(result) { - result := and(shr(_BITPOS_PARENT, packed), _BITMASK_KEY) - for {} 1 {} { - if iszero(result) { break } - packed := sload(or(nodes, result)) - if iszero(eq(target, and(shr(R, packed), _BITMASK_KEY))) { break } - target := result - result := and(shr(_BITPOS_PARENT, packed), _BITMASK_KEY) - } - break - } - for {} 1 {} { - packed := sload(or(nodes, result)) - let left := and(shr(L, packed), _BITMASK_KEY) - if iszero(left) { break } - result := left - } - break - } - } - result = _pack(nodes, uint256(result)); - } - - /// @dev Inserts or delete the value `x` from the tree. - function _update(uint256 nodes, uint256 cursor, uint256 key, uint256 x, uint256 mode) - private - returns (uint256 err) - { - /// @solidity memory-safe-assembly - assembly { - function getKey(packed_, bitpos_) -> index_ { - index_ := and(shr(bitpos_, packed_), _BITMASK_KEY) - } - - function setKey(packed_, bitpos_, key_) -> result_ { - result_ := or(and(not(shl(bitpos_, _BITMASK_KEY)), packed_), shl(bitpos_, key_)) - } - - function setRed(packed_, red_) -> result_ { - result_ := or(and(not(_BITMASK_RED), packed_), shl(_BITPOS_RED, red_)) - } - - function isRed(packed_) -> red_ { - red_ := and(_BITMASK_RED, packed_) - } - - function copyRed(packed_, fromPacked_) -> result_ { - result_ := or(and(not(_BITMASK_RED), packed_), and(_BITMASK_RED, fromPacked_)) - } - - function rotate(nodes_, key_, L, R) { - let packed_ := sload(or(nodes_, key_)) - let cursor_ := getKey(packed_, R) - let parent_ := getKey(packed_, _BITPOS_PARENT) - let cursorPacked_ := sload(or(nodes_, cursor_)) - let cursorLeft_ := getKey(cursorPacked_, L) - - if cursorLeft_ { - let s_ := or(nodes_, cursorLeft_) - sstore(s_, setKey(sload(s_), _BITPOS_PARENT, key_)) - } - - for {} 1 {} { - if iszero(parent_) { - mstore(0x00, cursor_) - break - } - let s_ := or(nodes_, parent_) - let parentPacked_ := sload(s_) - if eq(key_, getKey(parentPacked_, L)) { - sstore(s_, setKey(parentPacked_, L, cursor_)) - break - } - sstore(s_, setKey(parentPacked_, R, cursor_)) - break - } - packed_ := setKey(packed_, R, cursorLeft_) - sstore(or(nodes_, key_), setKey(packed_, _BITPOS_PARENT, cursor_)) - cursorPacked_ := setKey(cursorPacked_, _BITPOS_PARENT, parent_) - sstore(or(nodes_, cursor_), setKey(cursorPacked_, L, key_)) - } - - function insertFixup(nodes_, key_) { - for {} 1 {} { - if eq(key_, mload(0x00)) { break } - let packed_ := sload(or(nodes_, key_)) - let parent_ := getKey(packed_, _BITPOS_PARENT) - let parentPacked_ := sload(or(nodes_, parent_)) - if iszero(isRed(parentPacked_)) { break } - - let grandParent_ := getKey(parentPacked_, _BITPOS_PARENT) - let grandParentPacked_ := sload(or(nodes_, grandParent_)) - - let R := mul(eq(parent_, getKey(grandParentPacked_, 0)), _BITPOS_RIGHT) - let L := xor(R, _BITPOS_RIGHT) - - let cursor_ := getKey(grandParentPacked_, R) - let cursorPacked_ := sload(or(nodes_, cursor_)) - if iszero(isRed(cursorPacked_)) { - if eq(key_, getKey(parentPacked_, R)) { - key_ := parent_ - rotate(nodes_, key_, L, R) - } - parent_ := getKey(sload(or(nodes_, key_)), _BITPOS_PARENT) - parentPacked_ := sload(or(nodes_, parent_)) - sstore(or(nodes_, parent_), setRed(parentPacked_, 0)) - grandParent_ := getKey(parentPacked_, _BITPOS_PARENT) - let s_ := or(nodes_, grandParent_) - sstore(s_, setRed(sload(s_), 1)) - rotate(nodes_, grandParent_, R, L) - continue - } - sstore(or(nodes_, parent_), setRed(parentPacked_, 0)) - sstore(or(nodes_, cursor_), setRed(cursorPacked_, 0)) - sstore(or(nodes_, grandParent_), setRed(grandParentPacked_, 1)) - key_ := grandParent_ - } - let root_ := mload(0x00) - sstore(or(nodes_, root_), setRed(sload(or(nodes_, root_)), 0)) - } - - function insert(nodes_, cursor_, key_, x_) -> err_ { - if key_ { - err_ := ERROR_VALUE_ALREADY_EXISTS - leave - } - - let totalNodes_ := add(shr(128, mload(0x20)), 1) - - if gt(totalNodes_, _BITMASK_KEY) { - err_ := ERROR_TREE_IS_FULL - leave - } - - mstore(0x20, shl(128, totalNodes_)) - - let packed_ := or(_BITMASK_RED, shl(_BITPOS_PARENT, cursor_)) - let nodePointer_ := or(nodes_, totalNodes_) - - for {} 1 {} { - if iszero(gt(x_, _BITMASK_PACKED_VALUE)) { - packed_ := or(shl(_BITPOS_PACKED_VALUE, x_), packed_) - break - } - sstore(or(nodePointer_, _BIT_FULL_VALUE_SLOT), x_) - break - } - sstore(nodePointer_, packed_) - - for {} 1 {} { - if iszero(cursor_) { - mstore(0x00, totalNodes_) - break - } - let s_ := or(nodes_, cursor_) - let cursorPacked_ := sload(s_) - let cursorValue_ := shr(_BITPOS_PACKED_VALUE, cursorPacked_) - if iszero(cursorValue_) { cursorValue_ := sload(or(s_, _BIT_FULL_VALUE_SLOT)) } - if iszero(lt(x_, cursorValue_)) { - sstore(s_, setKey(cursorPacked_, _BITPOS_RIGHT, totalNodes_)) - break - } - sstore(s_, setKey(cursorPacked_, _BITPOS_LEFT, totalNodes_)) - break - } - insertFixup(nodes_, totalNodes_) - } - - function removeFixup(nodes_, key_) { - for {} 1 {} { - if eq(key_, mload(0x00)) { break } - let packed_ := sload(or(nodes_, key_)) - if isRed(packed_) { break } - - let parent_ := getKey(packed_, _BITPOS_PARENT) - let parentPacked_ := sload(or(nodes_, parent_)) - - let R := mul(eq(key_, getKey(parentPacked_, 0)), _BITPOS_RIGHT) - let L := xor(R, _BITPOS_RIGHT) - - let cursor_ := getKey(parentPacked_, R) - let cursorPacked_ := sload(or(nodes_, cursor_)) - - if isRed(cursorPacked_) { - sstore(or(nodes_, cursor_), setRed(cursorPacked_, 0)) - sstore(or(nodes_, parent_), setRed(parentPacked_, 1)) - rotate(nodes_, parent_, L, R) - cursor_ := getKey(sload(or(nodes_, parent_)), R) - cursorPacked_ := sload(or(nodes_, cursor_)) - } - - let cursorLeft_ := getKey(cursorPacked_, L) - let cursorLeftPacked_ := sload(or(nodes_, cursorLeft_)) - let cursorRight_ := getKey(cursorPacked_, R) - let cursorRightPacked_ := sload(or(nodes_, cursorRight_)) - - if iszero(or(isRed(cursorLeftPacked_), isRed(cursorRightPacked_))) { - sstore(or(nodes_, cursor_), setRed(cursorPacked_, 1)) - key_ := parent_ - continue - } - - if iszero(isRed(cursorRightPacked_)) { - sstore(or(nodes_, cursorLeft_), setRed(cursorLeftPacked_, 0)) - sstore(or(nodes_, cursor_), setRed(cursorPacked_, 1)) - rotate(nodes_, cursor_, R, L) - cursor_ := getKey(sload(or(nodes_, parent_)), R) - cursorPacked_ := sload(or(nodes_, cursor_)) - cursorRight_ := getKey(cursorPacked_, R) - cursorRightPacked_ := sload(or(nodes_, cursorRight_)) - } - - parentPacked_ := sload(or(nodes_, parent_)) - sstore(or(nodes_, cursor_), copyRed(cursorPacked_, parentPacked_)) - sstore(or(nodes_, parent_), setRed(parentPacked_, 0)) - sstore(or(nodes_, cursorRight_), setRed(cursorRightPacked_, 0)) - rotate(nodes_, parent_, L, R) - break - } - sstore(or(nodes_, key_), setRed(sload(or(nodes_, key_)), 0)) - } - - function removeLast(nodes_, cursor_) { - let last_ := shr(128, mload(0x20)) - let lastPacked_ := sload(or(nodes_, last_)) - let lastValue_ := shr(_BITPOS_PACKED_VALUE, lastPacked_) - let lastFullValue_ := 0 - if iszero(lastValue_) { - lastValue_ := sload(or(_BIT_FULL_VALUE_SLOT, or(nodes_, last_))) - lastFullValue_ := lastValue_ - } - - let cursorPacked_ := sload(or(nodes_, cursor_)) - let cursorValue_ := shr(_BITPOS_PACKED_VALUE, cursorPacked_) - let cursorFullValue_ := 0 - if iszero(cursorValue_) { - cursorValue_ := sload(or(_BIT_FULL_VALUE_SLOT, or(nodes_, cursor_))) - cursorFullValue_ := cursorValue_ - } - - if iszero(eq(lastValue_, cursorValue_)) { - sstore(or(nodes_, cursor_), lastPacked_) - if iszero(eq(lastFullValue_, cursorFullValue_)) { - sstore(or(_BIT_FULL_VALUE_SLOT, or(nodes_, cursor_)), lastFullValue_) - } - for { let lastParent_ := getKey(lastPacked_, _BITPOS_PARENT) } 1 {} { - if iszero(lastParent_) { - mstore(0x00, cursor_) - break - } - let s_ := or(nodes_, lastParent_) - let p_ := sload(s_) - let t_ := iszero(eq(last_, getKey(p_, _BITPOS_LEFT))) - sstore(s_, setKey(p_, mul(t_, _BITPOS_RIGHT), cursor_)) - break - } - let lastRight_ := getKey(lastPacked_, _BITPOS_RIGHT) - if lastRight_ { - let s_ := or(nodes_, lastRight_) - sstore(s_, setKey(sload(s_), _BITPOS_PARENT, cursor_)) - } - let lastLeft_ := getKey(lastPacked_, _BITPOS_LEFT) - if lastLeft_ { - let s_ := or(nodes_, lastLeft_) - sstore(s_, setKey(sload(s_), _BITPOS_PARENT, cursor_)) - } - } - sstore(or(nodes_, last_), 0) - if lastFullValue_ { sstore(or(_BIT_FULL_VALUE_SLOT, or(nodes_, last_)), 0) } - - mstore(0x20, shl(128, sub(last_, 1))) - } - - function remove(nodes_, key_) -> err_ { - let last_ := shr(128, mload(0x20)) - - if gt(key_, last_) { - err_ := ERROR_POINTER_OUT_OF_BOUNDS - leave - } - if iszero(key_) { - err_ := ERROR_VALUE_DOES_NOT_EXISTS - leave - } - - let cursor_ := 0 - - for {} 1 {} { - let packed_ := sload(or(nodes_, key_)) - let left_ := getKey(packed_, _BITPOS_LEFT) - let right_ := getKey(packed_, _BITPOS_RIGHT) - if iszero(mul(left_, right_)) { - cursor_ := key_ - break - } - cursor_ := right_ - for {} 1 {} { - let cursorLeft_ := getKey(sload(or(nodes_, cursor_)), _BITPOS_LEFT) - if iszero(cursorLeft_) { break } - cursor_ := cursorLeft_ - } - break - } - - let cursorPacked_ := sload(or(nodes_, cursor_)) - let probe_ := getKey(cursorPacked_, _BITPOS_LEFT) - if iszero(probe_) { probe_ := getKey(cursorPacked_, _BITPOS_RIGHT) } - - for { let yParent_ := getKey(cursorPacked_, _BITPOS_PARENT) } 1 {} { - let probeSlot_ := or(nodes_, probe_) - sstore(probeSlot_, setKey(sload(probeSlot_), _BITPOS_PARENT, yParent_)) - - if iszero(yParent_) { - mstore(0x00, probe_) - break - } - let s_ := or(nodes_, yParent_) - let p_ := sload(s_) - let t_ := iszero(eq(cursor_, getKey(p_, _BITPOS_LEFT))) - sstore(s_, setKey(p_, mul(t_, _BITPOS_RIGHT), probe_)) - break - } - - let skipFixup_ := isRed(cursorPacked_) - - if iszero(eq(cursor_, key_)) { - let packed_ := sload(or(nodes_, key_)) - let parent_ := getKey(packed_, _BITPOS_PARENT) - for {} 1 {} { - if iszero(parent_) { - mstore(0x00, cursor_) - break - } - let s_ := or(nodes_, parent_) - let p_ := sload(s_) - let t_ := iszero(eq(key_, getKey(p_, _BITPOS_LEFT))) - sstore(s_, setKey(p_, mul(t_, _BITPOS_RIGHT), cursor_)) - break - } - - let left_ := getKey(packed_, _BITPOS_LEFT) - let leftSlot_ := or(nodes_, left_) - sstore(leftSlot_, setKey(sload(leftSlot_), _BITPOS_PARENT, cursor_)) - - let right_ := getKey(packed_, _BITPOS_RIGHT) - let rightSlot_ := or(nodes_, right_) - sstore(rightSlot_, setKey(sload(rightSlot_), _BITPOS_PARENT, cursor_)) - - let m_ := sub(shl(_BITPOS_PACKED_VALUE, 1), 1) - sstore( - or(nodes_, cursor_), - xor(cursorPacked_, and(xor(packed_, cursorPacked_), m_)) - ) - - let t_ := cursor_ - cursor_ := key_ - key_ := t_ - } - if iszero(skipFixup_) { removeFixup(nodes_, probe_) } - - removeLast(nodes_, cursor_) - } - - mstore(0x10, sload(nodes)) - - for {} 1 {} { - if iszero(mode) { - err := insert(nodes, cursor, key, x) - break - } - err := remove(nodes, key) - break - } - - sstore(nodes, mload(0x10)) - } - } - - /// @dev Returns the pointer to the `nodes` for the tree. - function _nodes(Tree storage tree) private pure returns (uint256 nodes) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, tree.slot) - mstore(0x00, _NODES_SLOT_SEED) - nodes := shl(_NODES_SLOT_SHIFT, keccak256(0x00, 0x40)) - } - } - - /// @dev Finds `x` in `tree`. The `key` will be zero if `x` is not found. - function _find(Tree storage tree, uint256 x) - private - view - returns (uint256 nodes, uint256 cursor, uint256 key) - { - if (x == 0) _revert(0xc94f1877); // `ValueIsEmpty()`. - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, tree.slot) - mstore(0x00, _NODES_SLOT_SEED) - nodes := shl(_NODES_SLOT_SHIFT, keccak256(0x00, 0x40)) - - mstore(0x01, _BITPOS_RIGHT) - for { let probe := shr(128, sload(nodes)) } probe {} { - cursor := probe - let nodePacked := sload(or(nodes, probe)) - let nodeValue := shr(_BITPOS_PACKED_VALUE, nodePacked) - if iszero(nodeValue) { - nodeValue := sload(or(or(nodes, probe), _BIT_FULL_VALUE_SLOT)) - } - if eq(nodeValue, x) { - key := cursor - break - } - probe := and(shr(mload(gt(x, nodeValue)), nodePacked), _BITMASK_KEY) - } - } - } - - /// @dev Helper to revert `err` efficiently. - function _revert(uint256 err) private pure { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, err) - revert(0x1c, 0x04) - } - } -} diff --git a/lib/solady/src/utils/SSTORE2.sol b/lib/solady/src/utils/SSTORE2.sol deleted file mode 100644 index bc22e66..0000000 --- a/lib/solady/src/utils/SSTORE2.sol +++ /dev/null @@ -1,271 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Read and write to persistent storage at a fraction of the cost. -/// @author Solady (https://github.com/vectorized/solmady/blob/main/src/utils/SSTORE2.sol) -/// @author Saw-mon-and-Natalie (https://github.com/Saw-mon-and-Natalie) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/SSTORE2.sol) -/// @author Modified from 0xSequence (https://github.com/0xSequence/sstore2/blob/master/contracts/SSTORE2.sol) -library SSTORE2 { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev We skip the first byte as it's a STOP opcode, - /// which ensures the contract can't be called. - uint256 internal constant DATA_OFFSET = 1; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Unable to deploy the storage contract. - error DeploymentFailed(); - - /// @dev The storage contract address is invalid. - error InvalidPointer(); - - /// @dev Attempt to read outside of the storage contract's bytecode bounds. - error ReadOutOfBounds(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* WRITE LOGIC */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Writes `data` into the bytecode of a storage contract and returns its address. - function write(bytes memory data) internal returns (address pointer) { - /// @solidity memory-safe-assembly - assembly { - let originalDataLength := mload(data) - - // Add 1 to data size since we are prefixing it with a STOP opcode. - let dataSize := add(originalDataLength, DATA_OFFSET) - - /** - * ------------------------------------------------------------------------------+ - * Opcode | Mnemonic | Stack | Memory | - * ------------------------------------------------------------------------------| - * 61 dataSize | PUSH2 dataSize | dataSize | | - * 80 | DUP1 | dataSize dataSize | | - * 60 0xa | PUSH1 0xa | 0xa dataSize dataSize | | - * 3D | RETURNDATASIZE | 0 0xa dataSize dataSize | | - * 39 | CODECOPY | dataSize | [0..dataSize): code | - * 3D | RETURNDATASIZE | 0 dataSize | [0..dataSize): code | - * F3 | RETURN | | [0..dataSize): code | - * 00 | STOP | | | - * ------------------------------------------------------------------------------+ - * @dev Prefix the bytecode with a STOP opcode to ensure it cannot be called. - * Also PUSH2 is used since max contract size cap is 24,576 bytes which is less than 2 ** 16. - */ - mstore( - // Do a out-of-gas revert if `dataSize` is more than 2 bytes. - // The actual EVM limit may be smaller and may change over time. - add(data, gt(dataSize, 0xffff)), - // Left shift `dataSize` by 64 so that it lines up with the 0000 after PUSH2. - or(0xfd61000080600a3d393df300, shl(0x40, dataSize)) - ) - - // Deploy a new contract with the generated creation code. - pointer := create(0, add(data, 0x15), add(dataSize, 0xa)) - - // If `pointer` is zero, revert. - if iszero(pointer) { - // Store the function selector of `DeploymentFailed()`. - mstore(0x00, 0x30116425) - // Revert with (offset, size). - revert(0x1c, 0x04) - } - - // Restore original length of the variable size `data`. - mstore(data, originalDataLength) - } - } - - /// @dev Writes `data` into the bytecode of a storage contract with `salt` - /// and returns its deterministic address. - function writeDeterministic(bytes memory data, bytes32 salt) - internal - returns (address pointer) - { - /// @solidity memory-safe-assembly - assembly { - let originalDataLength := mload(data) - let dataSize := add(originalDataLength, DATA_OFFSET) - - mstore( - // Do a out-of-gas revert if `dataSize` is more than 2 bytes. - // The actual EVM limit may be smaller and may change over time. - add(data, gt(dataSize, 0xffff)), - // Left shift `dataSize` by 64 so that it lines up with the 0000 after PUSH2. - or(0xfd61000080600a3d393df300, shl(0x40, dataSize)) - ) - - // Deploy a new contract with the generated creation code. - pointer := create2(0, add(data, 0x15), add(dataSize, 0xa), salt) - - // If `pointer` is zero, revert. - if iszero(pointer) { - // Store the function selector of `DeploymentFailed()`. - mstore(0x00, 0x30116425) - // Revert with (offset, size). - revert(0x1c, 0x04) - } - - // Restore original length of the variable size `data`. - mstore(data, originalDataLength) - } - } - - /// @dev Returns the initialization code hash of the storage contract for `data`. - /// Used for mining vanity addresses with create2crunch. - function initCodeHash(bytes memory data) internal pure returns (bytes32 hash) { - /// @solidity memory-safe-assembly - assembly { - let originalDataLength := mload(data) - let dataSize := add(originalDataLength, DATA_OFFSET) - - // Do a out-of-gas revert if `dataSize` is more than 2 bytes. - // The actual EVM limit may be smaller and may change over time. - returndatacopy(returndatasize(), returndatasize(), shr(16, dataSize)) - - mstore(data, or(0x61000080600a3d393df300, shl(0x40, dataSize))) - - hash := keccak256(add(data, 0x15), add(dataSize, 0xa)) - - // Restore original length of the variable size `data`. - mstore(data, originalDataLength) - } - } - - /// @dev Returns the address of the storage contract for `data` - /// deployed with `salt` by `deployer`. - /// Note: The returned result has dirty upper 96 bits. Please clean if used in assembly. - function predictDeterministicAddress(bytes memory data, bytes32 salt, address deployer) - internal - pure - returns (address predicted) - { - bytes32 hash = initCodeHash(data); - /// @solidity memory-safe-assembly - assembly { - // Compute and store the bytecode hash. - mstore8(0x00, 0xff) // Write the prefix. - mstore(0x35, hash) - mstore(0x01, shl(96, deployer)) - mstore(0x15, salt) - predicted := keccak256(0x00, 0x55) - // Restore the part of the free memory pointer that has been overwritten. - mstore(0x35, 0) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* READ LOGIC */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns all the `data` from the bytecode of the storage contract at `pointer`. - function read(address pointer) internal view returns (bytes memory data) { - /// @solidity memory-safe-assembly - assembly { - let pointerCodesize := extcodesize(pointer) - if iszero(pointerCodesize) { - // Store the function selector of `InvalidPointer()`. - mstore(0x00, 0x11052bb4) - // Revert with (offset, size). - revert(0x1c, 0x04) - } - // Offset all indices by 1 to skip the STOP opcode. - let size := sub(pointerCodesize, DATA_OFFSET) - - // Get the pointer to the free memory and allocate - // enough 32-byte words for the data and the length of the data, - // then copy the code to the allocated memory. - // Masking with 0xffe0 will suffice, since contract size is less than 16 bits. - data := mload(0x40) - mstore(0x40, add(data, and(add(size, 0x3f), 0xffe0))) - mstore(data, size) - mstore(add(add(data, 0x20), size), 0) // Zeroize the last slot. - extcodecopy(pointer, add(data, 0x20), DATA_OFFSET, size) - } - } - - /// @dev Returns the `data` from the bytecode of the storage contract at `pointer`, - /// from the byte at `start`, to the end of the data stored. - function read(address pointer, uint256 start) internal view returns (bytes memory data) { - /// @solidity memory-safe-assembly - assembly { - let pointerCodesize := extcodesize(pointer) - if iszero(pointerCodesize) { - // Store the function selector of `InvalidPointer()`. - mstore(0x00, 0x11052bb4) - // Revert with (offset, size). - revert(0x1c, 0x04) - } - - // If `!(pointer.code.size > start)`, reverts. - // This also handles the case where `start + DATA_OFFSET` overflows. - if iszero(gt(pointerCodesize, start)) { - // Store the function selector of `ReadOutOfBounds()`. - mstore(0x00, 0x84eb0dd1) - // Revert with (offset, size). - revert(0x1c, 0x04) - } - let size := sub(pointerCodesize, add(start, DATA_OFFSET)) - - // Get the pointer to the free memory and allocate - // enough 32-byte words for the data and the length of the data, - // then copy the code to the allocated memory. - // Masking with 0xffe0 will suffice, since contract size is less than 16 bits. - data := mload(0x40) - mstore(0x40, add(data, and(add(size, 0x3f), 0xffe0))) - mstore(data, size) - mstore(add(add(data, 0x20), size), 0) // Zeroize the last slot. - extcodecopy(pointer, add(data, 0x20), add(start, DATA_OFFSET), size) - } - } - - /// @dev Returns the `data` from the bytecode of the storage contract at `pointer`, - /// from the byte at `start`, to the byte at `end` (exclusive) of the data stored. - function read(address pointer, uint256 start, uint256 end) - internal - view - returns (bytes memory data) - { - /// @solidity memory-safe-assembly - assembly { - let pointerCodesize := extcodesize(pointer) - if iszero(pointerCodesize) { - // Store the function selector of `InvalidPointer()`. - mstore(0x00, 0x11052bb4) - // Revert with (offset, size). - revert(0x1c, 0x04) - } - - // If `!(pointer.code.size > end) || (start > end)`, revert. - // This also handles the cases where - // `end + DATA_OFFSET` or `start + DATA_OFFSET` overflows. - if iszero( - and( - gt(pointerCodesize, end), // Within bounds. - iszero(gt(start, end)) // Valid range. - ) - ) { - // Store the function selector of `ReadOutOfBounds()`. - mstore(0x00, 0x84eb0dd1) - // Revert with (offset, size). - revert(0x1c, 0x04) - } - let size := sub(end, start) - - // Get the pointer to the free memory and allocate - // enough 32-byte words for the data and the length of the data, - // then copy the code to the allocated memory. - // Masking with 0xffe0 will suffice, since contract size is less than 16 bits. - data := mload(0x40) - mstore(0x40, add(data, and(add(size, 0x3f), 0xffe0))) - mstore(data, size) - mstore(add(add(data, 0x20), size), 0) // Zeroize the last slot. - extcodecopy(pointer, add(data, 0x20), add(start, DATA_OFFSET), size) - } - } -} diff --git a/lib/solady/src/utils/SafeCastLib.sol b/lib/solady/src/utils/SafeCastLib.sol deleted file mode 100644 index 8129629..0000000 --- a/lib/solady/src/utils/SafeCastLib.sol +++ /dev/null @@ -1,390 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Safe integer casting library that reverts on overflow. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/SafeCastLib.sol) -/// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/math/SafeCast.sol) -library SafeCastLib { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - error Overflow(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* UNSIGNED INTEGER SAFE CASTING OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - function toUint8(uint256 x) internal pure returns (uint8) { - if (x >= 1 << 8) _revertOverflow(); - return uint8(x); - } - - function toUint16(uint256 x) internal pure returns (uint16) { - if (x >= 1 << 16) _revertOverflow(); - return uint16(x); - } - - function toUint24(uint256 x) internal pure returns (uint24) { - if (x >= 1 << 24) _revertOverflow(); - return uint24(x); - } - - function toUint32(uint256 x) internal pure returns (uint32) { - if (x >= 1 << 32) _revertOverflow(); - return uint32(x); - } - - function toUint40(uint256 x) internal pure returns (uint40) { - if (x >= 1 << 40) _revertOverflow(); - return uint40(x); - } - - function toUint48(uint256 x) internal pure returns (uint48) { - if (x >= 1 << 48) _revertOverflow(); - return uint48(x); - } - - function toUint56(uint256 x) internal pure returns (uint56) { - if (x >= 1 << 56) _revertOverflow(); - return uint56(x); - } - - function toUint64(uint256 x) internal pure returns (uint64) { - if (x >= 1 << 64) _revertOverflow(); - return uint64(x); - } - - function toUint72(uint256 x) internal pure returns (uint72) { - if (x >= 1 << 72) _revertOverflow(); - return uint72(x); - } - - function toUint80(uint256 x) internal pure returns (uint80) { - if (x >= 1 << 80) _revertOverflow(); - return uint80(x); - } - - function toUint88(uint256 x) internal pure returns (uint88) { - if (x >= 1 << 88) _revertOverflow(); - return uint88(x); - } - - function toUint96(uint256 x) internal pure returns (uint96) { - if (x >= 1 << 96) _revertOverflow(); - return uint96(x); - } - - function toUint104(uint256 x) internal pure returns (uint104) { - if (x >= 1 << 104) _revertOverflow(); - return uint104(x); - } - - function toUint112(uint256 x) internal pure returns (uint112) { - if (x >= 1 << 112) _revertOverflow(); - return uint112(x); - } - - function toUint120(uint256 x) internal pure returns (uint120) { - if (x >= 1 << 120) _revertOverflow(); - return uint120(x); - } - - function toUint128(uint256 x) internal pure returns (uint128) { - if (x >= 1 << 128) _revertOverflow(); - return uint128(x); - } - - function toUint136(uint256 x) internal pure returns (uint136) { - if (x >= 1 << 136) _revertOverflow(); - return uint136(x); - } - - function toUint144(uint256 x) internal pure returns (uint144) { - if (x >= 1 << 144) _revertOverflow(); - return uint144(x); - } - - function toUint152(uint256 x) internal pure returns (uint152) { - if (x >= 1 << 152) _revertOverflow(); - return uint152(x); - } - - function toUint160(uint256 x) internal pure returns (uint160) { - if (x >= 1 << 160) _revertOverflow(); - return uint160(x); - } - - function toUint168(uint256 x) internal pure returns (uint168) { - if (x >= 1 << 168) _revertOverflow(); - return uint168(x); - } - - function toUint176(uint256 x) internal pure returns (uint176) { - if (x >= 1 << 176) _revertOverflow(); - return uint176(x); - } - - function toUint184(uint256 x) internal pure returns (uint184) { - if (x >= 1 << 184) _revertOverflow(); - return uint184(x); - } - - function toUint192(uint256 x) internal pure returns (uint192) { - if (x >= 1 << 192) _revertOverflow(); - return uint192(x); - } - - function toUint200(uint256 x) internal pure returns (uint200) { - if (x >= 1 << 200) _revertOverflow(); - return uint200(x); - } - - function toUint208(uint256 x) internal pure returns (uint208) { - if (x >= 1 << 208) _revertOverflow(); - return uint208(x); - } - - function toUint216(uint256 x) internal pure returns (uint216) { - if (x >= 1 << 216) _revertOverflow(); - return uint216(x); - } - - function toUint224(uint256 x) internal pure returns (uint224) { - if (x >= 1 << 224) _revertOverflow(); - return uint224(x); - } - - function toUint232(uint256 x) internal pure returns (uint232) { - if (x >= 1 << 232) _revertOverflow(); - return uint232(x); - } - - function toUint240(uint256 x) internal pure returns (uint240) { - if (x >= 1 << 240) _revertOverflow(); - return uint240(x); - } - - function toUint248(uint256 x) internal pure returns (uint248) { - if (x >= 1 << 248) _revertOverflow(); - return uint248(x); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* SIGNED INTEGER SAFE CASTING OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - function toInt8(int256 x) internal pure returns (int8) { - int8 y = int8(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt16(int256 x) internal pure returns (int16) { - int16 y = int16(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt24(int256 x) internal pure returns (int24) { - int24 y = int24(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt32(int256 x) internal pure returns (int32) { - int32 y = int32(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt40(int256 x) internal pure returns (int40) { - int40 y = int40(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt48(int256 x) internal pure returns (int48) { - int48 y = int48(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt56(int256 x) internal pure returns (int56) { - int56 y = int56(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt64(int256 x) internal pure returns (int64) { - int64 y = int64(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt72(int256 x) internal pure returns (int72) { - int72 y = int72(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt80(int256 x) internal pure returns (int80) { - int80 y = int80(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt88(int256 x) internal pure returns (int88) { - int88 y = int88(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt96(int256 x) internal pure returns (int96) { - int96 y = int96(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt104(int256 x) internal pure returns (int104) { - int104 y = int104(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt112(int256 x) internal pure returns (int112) { - int112 y = int112(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt120(int256 x) internal pure returns (int120) { - int120 y = int120(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt128(int256 x) internal pure returns (int128) { - int128 y = int128(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt136(int256 x) internal pure returns (int136) { - int136 y = int136(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt144(int256 x) internal pure returns (int144) { - int144 y = int144(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt152(int256 x) internal pure returns (int152) { - int152 y = int152(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt160(int256 x) internal pure returns (int160) { - int160 y = int160(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt168(int256 x) internal pure returns (int168) { - int168 y = int168(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt176(int256 x) internal pure returns (int176) { - int176 y = int176(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt184(int256 x) internal pure returns (int184) { - int184 y = int184(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt192(int256 x) internal pure returns (int192) { - int192 y = int192(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt200(int256 x) internal pure returns (int200) { - int200 y = int200(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt208(int256 x) internal pure returns (int208) { - int208 y = int208(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt216(int256 x) internal pure returns (int216) { - int216 y = int216(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt224(int256 x) internal pure returns (int224) { - int224 y = int224(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt232(int256 x) internal pure returns (int232) { - int232 y = int232(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt240(int256 x) internal pure returns (int240) { - int240 y = int240(x); - if (x != y) _revertOverflow(); - return y; - } - - function toInt248(int256 x) internal pure returns (int248) { - int248 y = int248(x); - if (x != y) _revertOverflow(); - return y; - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* OTHER SAFE CASTING OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - function toInt256(uint256 x) internal pure returns (int256) { - if (x >= 1 << 255) _revertOverflow(); - return int256(x); - } - - function toUint256(int256 x) internal pure returns (uint256) { - if (x < 0) _revertOverflow(); - return uint256(x); - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* PRIVATE HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - function _revertOverflow() private pure { - /// @solidity memory-safe-assembly - assembly { - // Store the function selector of `Overflow()`. - mstore(0x00, 0x35278d12) - // Revert with (offset, size). - revert(0x1c, 0x04) - } - } -} diff --git a/lib/solady/src/utils/SafeTransferLib.sol b/lib/solady/src/utils/SafeTransferLib.sol deleted file mode 100644 index 6d107dd..0000000 --- a/lib/solady/src/utils/SafeTransferLib.sol +++ /dev/null @@ -1,374 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Safe ETH and ERC20 transfer library that gracefully handles missing return values. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/SafeTransferLib.sol) -/// @author Modified from Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/SafeTransferLib.sol) -/// -/// @dev Note: -/// - For ETH transfers, please use `forceSafeTransferETH` for DoS protection. -/// - For ERC20s, this implementation won't check that a token has code, -/// responsibility is delegated to the caller. -library SafeTransferLib { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The ETH transfer has failed. - error ETHTransferFailed(); - - /// @dev The ERC20 `transferFrom` has failed. - error TransferFromFailed(); - - /// @dev The ERC20 `transfer` has failed. - error TransferFailed(); - - /// @dev The ERC20 `approve` has failed. - error ApproveFailed(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CONSTANTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Suggested gas stipend for contract receiving ETH that disallows any storage writes. - uint256 internal constant GAS_STIPEND_NO_STORAGE_WRITES = 2300; - - /// @dev Suggested gas stipend for contract receiving ETH to perform a few - /// storage reads and writes, but low enough to prevent griefing. - uint256 internal constant GAS_STIPEND_NO_GRIEF = 100000; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ETH OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - // If the ETH transfer MUST succeed with a reasonable gas budget, use the force variants. - // - // The regular variants: - // - Forwards all remaining gas to the target. - // - Reverts if the target reverts. - // - Reverts if the current contract has insufficient balance. - // - // The force variants: - // - Forwards with an optional gas stipend - // (defaults to `GAS_STIPEND_NO_GRIEF`, which is sufficient for most cases). - // - If the target reverts, or if the gas stipend is exhausted, - // creates a temporary contract to force send the ETH via `SELFDESTRUCT`. - // Future compatible with `SENDALL`: https://eips.ethereum.org/EIPS/eip-4758. - // - Reverts if the current contract has insufficient balance. - // - // The try variants: - // - Forwards with a mandatory gas stipend. - // - Instead of reverting, returns whether the transfer succeeded. - - /// @dev Sends `amount` (in wei) ETH to `to`. - function safeTransferETH(address to, uint256 amount) internal { - /// @solidity memory-safe-assembly - assembly { - if iszero(call(gas(), to, amount, codesize(), 0x00, codesize(), 0x00)) { - mstore(0x00, 0xb12d13eb) // `ETHTransferFailed()`. - revert(0x1c, 0x04) - } - } - } - - /// @dev Sends all the ETH in the current contract to `to`. - function safeTransferAllETH(address to) internal { - /// @solidity memory-safe-assembly - assembly { - // Transfer all the ETH and check if it succeeded or not. - if iszero(call(gas(), to, selfbalance(), codesize(), 0x00, codesize(), 0x00)) { - mstore(0x00, 0xb12d13eb) // `ETHTransferFailed()`. - revert(0x1c, 0x04) - } - } - } - - /// @dev Force sends `amount` (in wei) ETH to `to`, with a `gasStipend`. - function forceSafeTransferETH(address to, uint256 amount, uint256 gasStipend) internal { - /// @solidity memory-safe-assembly - assembly { - if lt(selfbalance(), amount) { - mstore(0x00, 0xb12d13eb) // `ETHTransferFailed()`. - revert(0x1c, 0x04) - } - if iszero(call(gasStipend, to, amount, codesize(), 0x00, codesize(), 0x00)) { - mstore(0x00, to) // Store the address in scratch space. - mstore8(0x0b, 0x73) // Opcode `PUSH20`. - mstore8(0x20, 0xff) // Opcode `SELFDESTRUCT`. - if iszero(create(amount, 0x0b, 0x16)) { revert(codesize(), codesize()) } // For gas estimation. - } - } - } - - /// @dev Force sends all the ETH in the current contract to `to`, with a `gasStipend`. - function forceSafeTransferAllETH(address to, uint256 gasStipend) internal { - /// @solidity memory-safe-assembly - assembly { - if iszero(call(gasStipend, to, selfbalance(), codesize(), 0x00, codesize(), 0x00)) { - mstore(0x00, to) // Store the address in scratch space. - mstore8(0x0b, 0x73) // Opcode `PUSH20`. - mstore8(0x20, 0xff) // Opcode `SELFDESTRUCT`. - if iszero(create(selfbalance(), 0x0b, 0x16)) { revert(codesize(), codesize()) } // For gas estimation. - } - } - } - - /// @dev Force sends `amount` (in wei) ETH to `to`, with `GAS_STIPEND_NO_GRIEF`. - function forceSafeTransferETH(address to, uint256 amount) internal { - /// @solidity memory-safe-assembly - assembly { - if lt(selfbalance(), amount) { - mstore(0x00, 0xb12d13eb) // `ETHTransferFailed()`. - revert(0x1c, 0x04) - } - if iszero(call(GAS_STIPEND_NO_GRIEF, to, amount, codesize(), 0x00, codesize(), 0x00)) { - mstore(0x00, to) // Store the address in scratch space. - mstore8(0x0b, 0x73) // Opcode `PUSH20`. - mstore8(0x20, 0xff) // Opcode `SELFDESTRUCT`. - if iszero(create(amount, 0x0b, 0x16)) { revert(codesize(), codesize()) } // For gas estimation. - } - } - } - - /// @dev Force sends all the ETH in the current contract to `to`, with `GAS_STIPEND_NO_GRIEF`. - function forceSafeTransferAllETH(address to) internal { - /// @solidity memory-safe-assembly - assembly { - // forgefmt: disable-next-item - if iszero(call(GAS_STIPEND_NO_GRIEF, to, selfbalance(), codesize(), 0x00, codesize(), 0x00)) { - mstore(0x00, to) // Store the address in scratch space. - mstore8(0x0b, 0x73) // Opcode `PUSH20`. - mstore8(0x20, 0xff) // Opcode `SELFDESTRUCT`. - if iszero(create(selfbalance(), 0x0b, 0x16)) { revert(codesize(), codesize()) } // For gas estimation. - } - } - } - - /// @dev Sends `amount` (in wei) ETH to `to`, with a `gasStipend`. - function trySafeTransferETH(address to, uint256 amount, uint256 gasStipend) - internal - returns (bool success) - { - /// @solidity memory-safe-assembly - assembly { - success := call(gasStipend, to, amount, codesize(), 0x00, codesize(), 0x00) - } - } - - /// @dev Sends all the ETH in the current contract to `to`, with a `gasStipend`. - function trySafeTransferAllETH(address to, uint256 gasStipend) - internal - returns (bool success) - { - /// @solidity memory-safe-assembly - assembly { - success := call(gasStipend, to, selfbalance(), codesize(), 0x00, codesize(), 0x00) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC20 OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Sends `amount` of ERC20 `token` from `from` to `to`. - /// Reverts upon failure. - /// - /// The `from` account must have at least `amount` approved for - /// the current contract to manage. - function safeTransferFrom(address token, address from, address to, uint256 amount) internal { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Cache the free memory pointer. - mstore(0x60, amount) // Store the `amount` argument. - mstore(0x40, to) // Store the `to` argument. - mstore(0x2c, shl(96, from)) // Store the `from` argument. - mstore(0x0c, 0x23b872dd000000000000000000000000) // `transferFrom(address,address,uint256)`. - // Perform the transfer, reverting upon failure. - if iszero( - and( // The arguments of `and` are evaluated from right to left. - or(eq(mload(0x00), 1), iszero(returndatasize())), // Returned 1 or nothing. - call(gas(), token, 0, 0x1c, 0x64, 0x00, 0x20) - ) - ) { - mstore(0x00, 0x7939f424) // `TransferFromFailed()`. - revert(0x1c, 0x04) - } - mstore(0x60, 0) // Restore the zero slot to zero. - mstore(0x40, m) // Restore the free memory pointer. - } - } - - /// @dev Sends all of ERC20 `token` from `from` to `to`. - /// Reverts upon failure. - /// - /// The `from` account must have their entire balance approved for - /// the current contract to manage. - function safeTransferAllFrom(address token, address from, address to) - internal - returns (uint256 amount) - { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) // Cache the free memory pointer. - mstore(0x40, to) // Store the `to` argument. - mstore(0x2c, shl(96, from)) // Store the `from` argument. - mstore(0x0c, 0x70a08231000000000000000000000000) // `balanceOf(address)`. - // Read the balance, reverting upon failure. - if iszero( - and( // The arguments of `and` are evaluated from right to left. - gt(returndatasize(), 0x1f), // At least 32 bytes returned. - staticcall(gas(), token, 0x1c, 0x24, 0x60, 0x20) - ) - ) { - mstore(0x00, 0x7939f424) // `TransferFromFailed()`. - revert(0x1c, 0x04) - } - mstore(0x00, 0x23b872dd) // `transferFrom(address,address,uint256)`. - amount := mload(0x60) // The `amount` is already at 0x60. We'll need to return it. - // Perform the transfer, reverting upon failure. - if iszero( - and( // The arguments of `and` are evaluated from right to left. - or(eq(mload(0x00), 1), iszero(returndatasize())), // Returned 1 or nothing. - call(gas(), token, 0, 0x1c, 0x64, 0x00, 0x20) - ) - ) { - mstore(0x00, 0x7939f424) // `TransferFromFailed()`. - revert(0x1c, 0x04) - } - mstore(0x60, 0) // Restore the zero slot to zero. - mstore(0x40, m) // Restore the free memory pointer. - } - } - - /// @dev Sends `amount` of ERC20 `token` from the current contract to `to`. - /// Reverts upon failure. - function safeTransfer(address token, address to, uint256 amount) internal { - /// @solidity memory-safe-assembly - assembly { - mstore(0x14, to) // Store the `to` argument. - mstore(0x34, amount) // Store the `amount` argument. - mstore(0x00, 0xa9059cbb000000000000000000000000) // `transfer(address,uint256)`. - // Perform the transfer, reverting upon failure. - if iszero( - and( // The arguments of `and` are evaluated from right to left. - or(eq(mload(0x00), 1), iszero(returndatasize())), // Returned 1 or nothing. - call(gas(), token, 0, 0x10, 0x44, 0x00, 0x20) - ) - ) { - mstore(0x00, 0x90b8ec18) // `TransferFailed()`. - revert(0x1c, 0x04) - } - mstore(0x34, 0) // Restore the part of the free memory pointer that was overwritten. - } - } - - /// @dev Sends all of ERC20 `token` from the current contract to `to`. - /// Reverts upon failure. - function safeTransferAll(address token, address to) internal returns (uint256 amount) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, 0x70a08231) // Store the function selector of `balanceOf(address)`. - mstore(0x20, address()) // Store the address of the current contract. - // Read the balance, reverting upon failure. - if iszero( - and( // The arguments of `and` are evaluated from right to left. - gt(returndatasize(), 0x1f), // At least 32 bytes returned. - staticcall(gas(), token, 0x1c, 0x24, 0x34, 0x20) - ) - ) { - mstore(0x00, 0x90b8ec18) // `TransferFailed()`. - revert(0x1c, 0x04) - } - mstore(0x14, to) // Store the `to` argument. - amount := mload(0x34) // The `amount` is already at 0x34. We'll need to return it. - mstore(0x00, 0xa9059cbb000000000000000000000000) // `transfer(address,uint256)`. - // Perform the transfer, reverting upon failure. - if iszero( - and( // The arguments of `and` are evaluated from right to left. - or(eq(mload(0x00), 1), iszero(returndatasize())), // Returned 1 or nothing. - call(gas(), token, 0, 0x10, 0x44, 0x00, 0x20) - ) - ) { - mstore(0x00, 0x90b8ec18) // `TransferFailed()`. - revert(0x1c, 0x04) - } - mstore(0x34, 0) // Restore the part of the free memory pointer that was overwritten. - } - } - - /// @dev Sets `amount` of ERC20 `token` for `to` to manage on behalf of the current contract. - /// Reverts upon failure. - function safeApprove(address token, address to, uint256 amount) internal { - /// @solidity memory-safe-assembly - assembly { - mstore(0x14, to) // Store the `to` argument. - mstore(0x34, amount) // Store the `amount` argument. - mstore(0x00, 0x095ea7b3000000000000000000000000) // `approve(address,uint256)`. - // Perform the approval, reverting upon failure. - if iszero( - and( // The arguments of `and` are evaluated from right to left. - or(eq(mload(0x00), 1), iszero(returndatasize())), // Returned 1 or nothing. - call(gas(), token, 0, 0x10, 0x44, 0x00, 0x20) - ) - ) { - mstore(0x00, 0x3e3f8f73) // `ApproveFailed()`. - revert(0x1c, 0x04) - } - mstore(0x34, 0) // Restore the part of the free memory pointer that was overwritten. - } - } - - /// @dev Sets `amount` of ERC20 `token` for `to` to manage on behalf of the current contract. - /// If the initial attempt to approve fails, attempts to reset the approved amount to zero, - /// then retries the approval again (some tokens, e.g. USDT, requires this). - /// Reverts upon failure. - function safeApproveWithRetry(address token, address to, uint256 amount) internal { - /// @solidity memory-safe-assembly - assembly { - mstore(0x14, to) // Store the `to` argument. - mstore(0x34, amount) // Store the `amount` argument. - mstore(0x00, 0x095ea7b3000000000000000000000000) // `approve(address,uint256)`. - // Perform the approval, retrying upon failure. - if iszero( - and( // The arguments of `and` are evaluated from right to left. - or(eq(mload(0x00), 1), iszero(returndatasize())), // Returned 1 or nothing. - call(gas(), token, 0, 0x10, 0x44, 0x00, 0x20) - ) - ) { - mstore(0x34, 0) // Store 0 for the `amount`. - mstore(0x00, 0x095ea7b3000000000000000000000000) // `approve(address,uint256)`. - pop(call(gas(), token, 0, 0x10, 0x44, codesize(), 0x00)) // Reset the approval. - mstore(0x34, amount) // Store back the original `amount`. - // Retry the approval, reverting upon failure. - if iszero( - and( - or(eq(mload(0x00), 1), iszero(returndatasize())), // Returned 1 or nothing. - call(gas(), token, 0, 0x10, 0x44, 0x00, 0x20) - ) - ) { - mstore(0x00, 0x3e3f8f73) // `ApproveFailed()`. - revert(0x1c, 0x04) - } - } - mstore(0x34, 0) // Restore the part of the free memory pointer that was overwritten. - } - } - - /// @dev Returns the amount of ERC20 `token` owned by `account`. - /// Returns zero if the `token` does not exist. - function balanceOf(address token, address account) internal view returns (uint256 amount) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x14, account) // Store the `account` argument. - mstore(0x00, 0x70a08231000000000000000000000000) // `balanceOf(address)`. - amount := - mul( - mload(0x20), - and( // The arguments of `and` are evaluated from right to left. - gt(returndatasize(), 0x1f), // At least 32 bytes returned. - staticcall(gas(), token, 0x10, 0x24, 0x20, 0x20) - ) - ) - } - } -} diff --git a/lib/solady/src/utils/SignatureCheckerLib.sol b/lib/solady/src/utils/SignatureCheckerLib.sol deleted file mode 100644 index 6f9350f..0000000 --- a/lib/solady/src/utils/SignatureCheckerLib.sol +++ /dev/null @@ -1,545 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice Signature verification helper that supports both ECDSA signatures from EOAs -/// and ERC1271 signatures from smart contract wallets like Argent and Gnosis safe. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/SignatureCheckerLib.sol) -/// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/cryptography/SignatureChecker.sol) -/// -/// @dev Note: -/// - The signature checking functions use the ecrecover precompile (0x1). -/// - The `bytes memory signature` variants use the identity precompile (0x4) -/// to copy memory internally. -/// - Unlike ECDSA signatures, contract signatures are revocable. -/// - As of Solady version 0.0.134, all `bytes signature` variants accept both -/// regular 65-byte `(r, s, v)` and EIP-2098 `(r, vs)` short form signatures. -/// See: https://eips.ethereum.org/EIPS/eip-2098 -/// This is for calldata efficiency on smart accounts prevalent on L2s. -/// -/// WARNING! Do NOT use signatures as unique identifiers: -/// - Use a nonce in the digest to prevent replay attacks on the same contract. -/// - Use EIP-712 for the digest to prevent replay attacks across different chains and contracts. -/// EIP-712 also enables readable signing of typed data for better user safety. -/// This implementation does NOT check if a signature is non-malleable. -library SignatureCheckerLib { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* SIGNATURE CHECKING OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns whether `signature` is valid for `signer` and `hash`. - /// If `signer` is a smart contract, the signature is validated with ERC1271. - /// Otherwise, the signature is validated with `ECDSA.recover`. - function isValidSignatureNow(address signer, bytes32 hash, bytes memory signature) - internal - view - returns (bool isValid) - { - /// @solidity memory-safe-assembly - assembly { - // Clean the upper 96 bits of `signer` in case they are dirty. - for { signer := shr(96, shl(96, signer)) } signer {} { - let m := mload(0x40) - mstore(0x00, hash) - mstore(0x40, mload(add(signature, 0x20))) // `r`. - if eq(mload(signature), 64) { - let vs := mload(add(signature, 0x40)) - mstore(0x20, add(shr(255, vs), 27)) // `v`. - mstore(0x60, shr(1, shl(1, vs))) // `s`. - let t := - staticcall( - gas(), // Amount of gas left for the transaction. - 1, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x01, // Start of output. - 0x20 // Size of output. - ) - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - if iszero(or(iszero(returndatasize()), xor(signer, mload(t)))) { - isValid := 1 - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - break - } - } - if eq(mload(signature), 65) { - mstore(0x20, byte(0, mload(add(signature, 0x60)))) // `v`. - mstore(0x60, mload(add(signature, 0x40))) // `s`. - let t := - staticcall( - gas(), // Amount of gas left for the transaction. - 1, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x01, // Start of output. - 0x20 // Size of output. - ) - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - if iszero(or(iszero(returndatasize()), xor(signer, mload(t)))) { - isValid := 1 - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - break - } - } - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - - let f := shl(224, 0x1626ba7e) - mstore(m, f) // `bytes4(keccak256("isValidSignature(bytes32,bytes)"))`. - mstore(add(m, 0x04), hash) - let d := add(m, 0x24) - mstore(d, 0x40) // The offset of the `signature` in the calldata. - // Copy the `signature` over. - let n := add(0x20, mload(signature)) - pop(staticcall(gas(), 4, signature, n, add(m, 0x44), n)) - // forgefmt: disable-next-item - isValid := and( - // Whether the returndata is the magic value `0x1626ba7e` (left-aligned). - eq(mload(d), f), - // Whether the staticcall does not revert. - // This must be placed at the end of the `and` clause, - // as the arguments are evaluated from right to left. - staticcall( - gas(), // Remaining gas. - signer, // The `signer` address. - m, // Offset of calldata in memory. - add(returndatasize(), 0x44), // Length of calldata in memory. - d, // Offset of returndata. - 0x20 // Length of returndata to write. - ) - ) - break - } - } - } - - /// @dev Returns whether `signature` is valid for `signer` and `hash`. - /// If `signer` is a smart contract, the signature is validated with ERC1271. - /// Otherwise, the signature is validated with `ECDSA.recover`. - function isValidSignatureNowCalldata(address signer, bytes32 hash, bytes calldata signature) - internal - view - returns (bool isValid) - { - /// @solidity memory-safe-assembly - assembly { - // Clean the upper 96 bits of `signer` in case they are dirty. - for { signer := shr(96, shl(96, signer)) } signer {} { - let m := mload(0x40) - mstore(0x00, hash) - if eq(signature.length, 64) { - let vs := calldataload(add(signature.offset, 0x20)) - mstore(0x20, add(shr(255, vs), 27)) // `v`. - mstore(0x40, calldataload(signature.offset)) // `r`. - mstore(0x60, shr(1, shl(1, vs))) // `s`. - let t := - staticcall( - gas(), // Amount of gas left for the transaction. - 1, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x01, // Start of output. - 0x20 // Size of output. - ) - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - if iszero(or(iszero(returndatasize()), xor(signer, mload(t)))) { - isValid := 1 - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - break - } - } - if eq(signature.length, 65) { - mstore(0x20, byte(0, calldataload(add(signature.offset, 0x40)))) // `v`. - calldatacopy(0x40, signature.offset, 0x40) // `r`, `s`. - let t := - staticcall( - gas(), // Amount of gas left for the transaction. - 1, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x01, // Start of output. - 0x20 // Size of output. - ) - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - if iszero(or(iszero(returndatasize()), xor(signer, mload(t)))) { - isValid := 1 - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - break - } - } - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - - let f := shl(224, 0x1626ba7e) - mstore(m, f) // `bytes4(keccak256("isValidSignature(bytes32,bytes)"))`. - mstore(add(m, 0x04), hash) - let d := add(m, 0x24) - mstore(d, 0x40) // The offset of the `signature` in the calldata. - mstore(add(m, 0x44), signature.length) - // Copy the `signature` over. - calldatacopy(add(m, 0x64), signature.offset, signature.length) - // forgefmt: disable-next-item - isValid := and( - // Whether the returndata is the magic value `0x1626ba7e` (left-aligned). - eq(mload(d), f), - // Whether the staticcall does not revert. - // This must be placed at the end of the `and` clause, - // as the arguments are evaluated from right to left. - staticcall( - gas(), // Remaining gas. - signer, // The `signer` address. - m, // Offset of calldata in memory. - add(signature.length, 0x64), // Length of calldata in memory. - d, // Offset of returndata. - 0x20 // Length of returndata to write. - ) - ) - break - } - } - } - - /// @dev Returns whether the signature (`r`, `vs`) is valid for `signer` and `hash`. - /// If `signer` is a smart contract, the signature is validated with ERC1271. - /// Otherwise, the signature is validated with `ECDSA.recover`. - function isValidSignatureNow(address signer, bytes32 hash, bytes32 r, bytes32 vs) - internal - view - returns (bool isValid) - { - /// @solidity memory-safe-assembly - assembly { - // Clean the upper 96 bits of `signer` in case they are dirty. - for { signer := shr(96, shl(96, signer)) } signer {} { - let m := mload(0x40) - mstore(0x00, hash) - mstore(0x20, add(shr(255, vs), 27)) // `v`. - mstore(0x40, r) // `r`. - mstore(0x60, shr(1, shl(1, vs))) // `s`. - let t := - staticcall( - gas(), // Amount of gas left for the transaction. - 1, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x01, // Start of output. - 0x20 // Size of output. - ) - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - if iszero(or(iszero(returndatasize()), xor(signer, mload(t)))) { - isValid := 1 - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - break - } - - let f := shl(224, 0x1626ba7e) - mstore(m, f) // `bytes4(keccak256("isValidSignature(bytes32,bytes)"))`. - mstore(add(m, 0x04), hash) - let d := add(m, 0x24) - mstore(d, 0x40) // The offset of the `signature` in the calldata. - mstore(add(m, 0x44), 65) // Length of the signature. - mstore(add(m, 0x64), r) // `r`. - mstore(add(m, 0x84), mload(0x60)) // `s`. - mstore8(add(m, 0xa4), mload(0x20)) // `v`. - // forgefmt: disable-next-item - isValid := and( - // Whether the returndata is the magic value `0x1626ba7e` (left-aligned). - eq(mload(d), f), - // Whether the staticcall does not revert. - // This must be placed at the end of the `and` clause, - // as the arguments are evaluated from right to left. - staticcall( - gas(), // Remaining gas. - signer, // The `signer` address. - m, // Offset of calldata in memory. - 0xa5, // Length of calldata in memory. - d, // Offset of returndata. - 0x20 // Length of returndata to write. - ) - ) - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - break - } - } - } - - /// @dev Returns whether the signature (`v`, `r`, `s`) is valid for `signer` and `hash`. - /// If `signer` is a smart contract, the signature is validated with ERC1271. - /// Otherwise, the signature is validated with `ECDSA.recover`. - function isValidSignatureNow(address signer, bytes32 hash, uint8 v, bytes32 r, bytes32 s) - internal - view - returns (bool isValid) - { - /// @solidity memory-safe-assembly - assembly { - // Clean the upper 96 bits of `signer` in case they are dirty. - for { signer := shr(96, shl(96, signer)) } signer {} { - let m := mload(0x40) - mstore(0x00, hash) - mstore(0x20, and(v, 0xff)) // `v`. - mstore(0x40, r) // `r`. - mstore(0x60, s) // `s`. - let t := - staticcall( - gas(), // Amount of gas left for the transaction. - 1, // Address of `ecrecover`. - 0x00, // Start of input. - 0x80, // Size of input. - 0x01, // Start of output. - 0x20 // Size of output. - ) - // `returndatasize()` will be `0x20` upon success, and `0x00` otherwise. - if iszero(or(iszero(returndatasize()), xor(signer, mload(t)))) { - isValid := 1 - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - break - } - - let f := shl(224, 0x1626ba7e) - mstore(m, f) // `bytes4(keccak256("isValidSignature(bytes32,bytes)"))`. - mstore(add(m, 0x04), hash) - let d := add(m, 0x24) - mstore(d, 0x40) // The offset of the `signature` in the calldata. - mstore(add(m, 0x44), 65) // Length of the signature. - mstore(add(m, 0x64), r) // `r`. - mstore(add(m, 0x84), s) // `s`. - mstore8(add(m, 0xa4), v) // `v`. - // forgefmt: disable-next-item - isValid := and( - // Whether the returndata is the magic value `0x1626ba7e` (left-aligned). - eq(mload(d), f), - // Whether the staticcall does not revert. - // This must be placed at the end of the `and` clause, - // as the arguments are evaluated from right to left. - staticcall( - gas(), // Remaining gas. - signer, // The `signer` address. - m, // Offset of calldata in memory. - 0xa5, // Length of calldata in memory. - d, // Offset of returndata. - 0x20 // Length of returndata to write. - ) - ) - mstore(0x60, 0) // Restore the zero slot. - mstore(0x40, m) // Restore the free memory pointer. - break - } - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* ERC1271 OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns whether `signature` is valid for `hash` for an ERC1271 `signer` contract. - function isValidERC1271SignatureNow(address signer, bytes32 hash, bytes memory signature) - internal - view - returns (bool isValid) - { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) - let f := shl(224, 0x1626ba7e) - mstore(m, f) // `bytes4(keccak256("isValidSignature(bytes32,bytes)"))`. - mstore(add(m, 0x04), hash) - let d := add(m, 0x24) - mstore(d, 0x40) // The offset of the `signature` in the calldata. - // Copy the `signature` over. - let n := add(0x20, mload(signature)) - pop(staticcall(gas(), 4, signature, n, add(m, 0x44), n)) - // forgefmt: disable-next-item - isValid := and( - // Whether the returndata is the magic value `0x1626ba7e` (left-aligned). - eq(mload(d), f), - // Whether the staticcall does not revert. - // This must be placed at the end of the `and` clause, - // as the arguments are evaluated from right to left. - staticcall( - gas(), // Remaining gas. - signer, // The `signer` address. - m, // Offset of calldata in memory. - add(returndatasize(), 0x44), // Length of calldata in memory. - d, // Offset of returndata. - 0x20 // Length of returndata to write. - ) - ) - } - } - - /// @dev Returns whether `signature` is valid for `hash` for an ERC1271 `signer` contract. - function isValidERC1271SignatureNowCalldata( - address signer, - bytes32 hash, - bytes calldata signature - ) internal view returns (bool isValid) { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) - let f := shl(224, 0x1626ba7e) - mstore(m, f) // `bytes4(keccak256("isValidSignature(bytes32,bytes)"))`. - mstore(add(m, 0x04), hash) - let d := add(m, 0x24) - mstore(d, 0x40) // The offset of the `signature` in the calldata. - mstore(add(m, 0x44), signature.length) - // Copy the `signature` over. - calldatacopy(add(m, 0x64), signature.offset, signature.length) - // forgefmt: disable-next-item - isValid := and( - // Whether the returndata is the magic value `0x1626ba7e` (left-aligned). - eq(mload(d), f), - // Whether the staticcall does not revert. - // This must be placed at the end of the `and` clause, - // as the arguments are evaluated from right to left. - staticcall( - gas(), // Remaining gas. - signer, // The `signer` address. - m, // Offset of calldata in memory. - add(signature.length, 0x64), // Length of calldata in memory. - d, // Offset of returndata. - 0x20 // Length of returndata to write. - ) - ) - } - } - - /// @dev Returns whether the signature (`r`, `vs`) is valid for `hash` - /// for an ERC1271 `signer` contract. - function isValidERC1271SignatureNow(address signer, bytes32 hash, bytes32 r, bytes32 vs) - internal - view - returns (bool isValid) - { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) - let f := shl(224, 0x1626ba7e) - mstore(m, f) // `bytes4(keccak256("isValidSignature(bytes32,bytes)"))`. - mstore(add(m, 0x04), hash) - let d := add(m, 0x24) - mstore(d, 0x40) // The offset of the `signature` in the calldata. - mstore(add(m, 0x44), 65) // Length of the signature. - mstore(add(m, 0x64), r) // `r`. - mstore(add(m, 0x84), shr(1, shl(1, vs))) // `s`. - mstore8(add(m, 0xa4), add(shr(255, vs), 27)) // `v`. - // forgefmt: disable-next-item - isValid := and( - // Whether the returndata is the magic value `0x1626ba7e` (left-aligned). - eq(mload(d), f), - // Whether the staticcall does not revert. - // This must be placed at the end of the `and` clause, - // as the arguments are evaluated from right to left. - staticcall( - gas(), // Remaining gas. - signer, // The `signer` address. - m, // Offset of calldata in memory. - 0xa5, // Length of calldata in memory. - d, // Offset of returndata. - 0x20 // Length of returndata to write. - ) - ) - } - } - - /// @dev Returns whether the signature (`v`, `r`, `s`) is valid for `hash` - /// for an ERC1271 `signer` contract. - function isValidERC1271SignatureNow(address signer, bytes32 hash, uint8 v, bytes32 r, bytes32 s) - internal - view - returns (bool isValid) - { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) - let f := shl(224, 0x1626ba7e) - mstore(m, f) // `bytes4(keccak256("isValidSignature(bytes32,bytes)"))`. - mstore(add(m, 0x04), hash) - let d := add(m, 0x24) - mstore(d, 0x40) // The offset of the `signature` in the calldata. - mstore(add(m, 0x44), 65) // Length of the signature. - mstore(add(m, 0x64), r) // `r`. - mstore(add(m, 0x84), s) // `s`. - mstore8(add(m, 0xa4), v) // `v`. - // forgefmt: disable-next-item - isValid := and( - // Whether the returndata is the magic value `0x1626ba7e` (left-aligned). - eq(mload(d), f), - // Whether the staticcall does not revert. - // This must be placed at the end of the `and` clause, - // as the arguments are evaluated from right to left. - staticcall( - gas(), // Remaining gas. - signer, // The `signer` address. - m, // Offset of calldata in memory. - 0xa5, // Length of calldata in memory. - d, // Offset of returndata. - 0x20 // Length of returndata to write. - ) - ) - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* HASHING OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns an Ethereum Signed Message, created from a `hash`. - /// This produces a hash corresponding to the one signed with the - /// [`eth_sign`](https://eth.wiki/json-rpc/API#eth_sign) - /// JSON-RPC method as part of EIP-191. - function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, hash) // Store into scratch space for keccak256. - mstore(0x00, "\x00\x00\x00\x00\x19Ethereum Signed Message:\n32") // 28 bytes. - result := keccak256(0x04, 0x3c) // `32 * 2 - (32 - 28) = 60 = 0x3c`. - } - } - - /// @dev Returns an Ethereum Signed Message, created from `s`. - /// This produces a hash corresponding to the one signed with the - /// [`eth_sign`](https://eth.wiki/json-rpc/API#eth_sign) - /// JSON-RPC method as part of EIP-191. - /// Note: Supports lengths of `s` up to 999999 bytes. - function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - let sLength := mload(s) - let o := 0x20 - mstore(o, "\x19Ethereum Signed Message:\n") // 26 bytes, zero-right-padded. - mstore(0x00, 0x00) - // Convert the `s.length` to ASCII decimal representation: `base10(s.length)`. - for { let temp := sLength } 1 {} { - o := sub(o, 1) - mstore8(o, add(48, mod(temp, 10))) - temp := div(temp, 10) - if iszero(temp) { break } - } - let n := sub(0x3a, o) // Header length: `26 + 32 - o`. - // Throw an out-of-offset error (consumes all gas) if the header exceeds 32 bytes. - returndatacopy(returndatasize(), returndatasize(), gt(n, 0x20)) - mstore(s, or(mload(0x00), mload(n))) // Temporarily store the header. - result := keccak256(add(s, sub(0x20, n)), add(n, sLength)) - mstore(s, sLength) // Restore the length. - } - } - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EMPTY CALLDATA HELPERS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Returns an empty calldata bytes. - function emptySignature() internal pure returns (bytes calldata signature) { - /// @solidity memory-safe-assembly - assembly { - signature.length := 0 - } - } -} diff --git a/lib/solady/src/utils/UUPSUpgradeable.sol b/lib/solady/src/utils/UUPSUpgradeable.sol deleted file mode 100644 index 368699b..0000000 --- a/lib/solady/src/utils/UUPSUpgradeable.sol +++ /dev/null @@ -1,142 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @notice UUPS proxy mixin. -/// @author Solady (https://github.com/vectorized/solady/blob/main/src/utils/UUPSUpgradeable.sol) -/// @author Modified from OpenZeppelin -/// (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/proxy/utils/UUPSUpgradeable.sol) -/// -/// Note: -/// - This implementation is intended to be used with ERC1967 proxies. -/// See: `LibClone.deployERC1967` and related functions. -/// - This implementation is NOT compatible with legacy OpenZeppelin proxies -/// which do not store the implementation at `_ERC1967_IMPLEMENTATION_SLOT`. -abstract contract UUPSUpgradeable { - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* CUSTOM ERRORS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The upgrade failed. - error UpgradeFailed(); - - /// @dev The call is from an unauthorized call context. - error UnauthorizedCallContext(); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* IMMUTABLES */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev For checking if the context is a delegate call. - uint256 private immutable __self = uint256(uint160(address(this))); - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* EVENTS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Emitted when the proxy's implementation is upgraded. - event Upgraded(address indexed implementation); - - /// @dev `keccak256(bytes("Upgraded(address)"))`. - uint256 private constant _UPGRADED_EVENT_SIGNATURE = - 0xbc7cd75a20ee27fd9adebab32041f755214dbc6bffa90cc0225b39da2e5c2d3b; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* STORAGE */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev The ERC-1967 storage slot for the implementation in the proxy. - /// `uint256(keccak256("eip1967.proxy.implementation")) - 1`. - bytes32 internal constant _ERC1967_IMPLEMENTATION_SLOT = - 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc; - - /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ - /* UUPS OPERATIONS */ - /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ - - /// @dev Please override this function to check if `msg.sender` is authorized - /// to upgrade the proxy to `newImplementation`, reverting if not. - /// ``` - /// function _authorizeUpgrade(address) internal override onlyOwner {} - /// ``` - function _authorizeUpgrade(address newImplementation) internal virtual; - - /// @dev Returns the storage slot used by the implementation, - /// as specified in [ERC1822](https://eips.ethereum.org/EIPS/eip-1822). - /// - /// Note: The `notDelegated` modifier prevents accidental upgrades to - /// an implementation that is a proxy contract. - function proxiableUUID() public view virtual notDelegated returns (bytes32) { - // This function must always return `_ERC1967_IMPLEMENTATION_SLOT` to comply with ERC1967. - return _ERC1967_IMPLEMENTATION_SLOT; - } - - /// @dev Upgrades the proxy's implementation to `newImplementation`. - /// Emits a {Upgraded} event. - /// - /// Note: Passing in empty `data` skips the delegatecall to `newImplementation`. - function upgradeToAndCall(address newImplementation, bytes calldata data) - public - payable - virtual - onlyProxy - { - _authorizeUpgrade(newImplementation); - /// @solidity memory-safe-assembly - assembly { - newImplementation := shr(96, shl(96, newImplementation)) // Clears upper 96 bits. - mstore(0x01, 0x52d1902d) // `proxiableUUID()`. - let s := _ERC1967_IMPLEMENTATION_SLOT - // Check if `newImplementation` implements `proxiableUUID` correctly. - if iszero(eq(mload(staticcall(gas(), newImplementation, 0x1d, 0x04, 0x01, 0x20)), s)) { - mstore(0x01, 0x55299b49) // `UpgradeFailed()`. - revert(0x1d, 0x04) - } - // Emit the {Upgraded} event. - log2(codesize(), 0x00, _UPGRADED_EVENT_SIGNATURE, newImplementation) - sstore(s, newImplementation) // Updates the implementation. - - // Perform a delegatecall to `newImplementation` if `data` is non-empty. - if data.length { - // Forwards the `data` to `newImplementation` via delegatecall. - let m := mload(0x40) - calldatacopy(m, data.offset, data.length) - if iszero(delegatecall(gas(), newImplementation, m, data.length, codesize(), 0x00)) - { - // Bubble up the revert if the call reverts. - returndatacopy(m, 0x00, returndatasize()) - revert(m, returndatasize()) - } - } - } - } - - /// @dev Requires that the execution is performed through a proxy. - modifier onlyProxy() { - uint256 s = __self; - /// @solidity memory-safe-assembly - assembly { - // To enable use cases with an immutable default implementation in the bytecode, - // (see: ERC6551Proxy), we don't require that the proxy address must match the - // value stored in the implementation slot, which may not be initialized. - if eq(s, address()) { - mstore(0x00, 0x9f03a026) // `UnauthorizedCallContext()`. - revert(0x1c, 0x04) - } - } - _; - } - - /// @dev Requires that the execution is NOT performed via delegatecall. - /// This is the opposite of `onlyProxy`. - modifier notDelegated() { - uint256 s = __self; - /// @solidity memory-safe-assembly - assembly { - if iszero(eq(s, address())) { - mstore(0x00, 0x9f03a026) // `UnauthorizedCallContext()`. - revert(0x1c, 0x04) - } - } - _; - } -} diff --git a/lib/solady/test/Base64.t.sol b/lib/solady/test/Base64.t.sol deleted file mode 100644 index 87c62da..0000000 --- a/lib/solady/test/Base64.t.sol +++ /dev/null @@ -1,119 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {Base64} from "../src/utils/Base64.sol"; -import {LibString} from "../src/utils/LibString.sol"; - -contract Base64Test is SoladyTest { - function testBase64EncodeEmptyString() public { - _testBase64Encode("", ""); - } - - function testBase64EncodeShortStrings() public { - _testBase64Encode("M", "TQ=="); - _testBase64Encode("Mi", "TWk="); - _testBase64Encode("Mil", "TWls"); - _testBase64Encode("Mila", "TWlsYQ=="); - _testBase64Encode("Milad", "TWlsYWQ="); - _testBase64Encode("Milady", "TWlsYWR5"); - } - - function testBase64EncodeToStringWithDoublePadding() public { - _testBase64Encode("test", "dGVzdA=="); - } - - function testBase64EncodeToStringWithSinglePadding() public { - _testBase64Encode("test1", "dGVzdDE="); - } - - function testBase64EncodeToStringWithNoPadding() public { - _testBase64Encode("test12", "dGVzdDEy"); - } - - function testBase64EncodeSentence() public { - _testBase64Encode( - "Lorem ipsum dolor sit amet, consectetur adipiscing elit.", - "TG9yZW0gaXBzdW0gZG9sb3Igc2l0IGFtZXQsIGNvbnNlY3RldHVyIGFkaXBpc2NpbmcgZWxpdC4=" - ); - } - - function testBase64WordBoundary() public { - // Base64.encode allocates memory in multiples of 32 bytes. - // This checks if the amount of memory allocated is enough. - _testBase64Encode("012345678901234567890", "MDEyMzQ1Njc4OTAxMjM0NTY3ODkw"); - _testBase64Encode("0123456789012345678901", "MDEyMzQ1Njc4OTAxMjM0NTY3ODkwMQ=="); - _testBase64Encode("01234567890123456789012", "MDEyMzQ1Njc4OTAxMjM0NTY3ODkwMTI="); - _testBase64Encode("012345678901234567890123", "MDEyMzQ1Njc4OTAxMjM0NTY3ODkwMTIz"); - _testBase64Encode("0123456789012345678901234", "MDEyMzQ1Njc4OTAxMjM0NTY3ODkwMTIzNA=="); - } - - function _testBase64Encode(string memory input, string memory output) private { - assertEq(Base64.encode(bytes(input)), output); - } - - function testBase64EncodeDecode(bytes memory input) public { - string memory encoded = Base64.encode(input); - bytes memory decoded = Base64.decode(encoded); - - assertEq(input, decoded); - } - - function testBase64DecodeShortStringGas() public { - assertEq(Base64.decode("TWlsYWR5").length, 6); - } - - function testBase64DecodeSentenceGas() public { - assertEq( - Base64.decode( - "TG9yZW0gaXBzdW0gZG9sb3Igc2l0IGFtZXQsIGNvbnNlY3RldHVyIGFkaXBpc2NpbmcgZWxpdC4=" - ).length, - 56 - ); - } - - function testBase64EncodeDecodeAltModes(bytes memory input) public brutalizeMemory { - for (uint256 i; i < 2; ++i) { - _misalignFreeMemoryPointer(); - string memory encoded = Base64.encode(input); - _checkMemory(encoded); - - if (_random() & 1 == 0) { - encoded = LibString.replace(encoded, "=", ""); - } - if (_random() & 1 == 0) { - encoded = LibString.replace(encoded, "/", ","); - } - if (_random() & 1 == 0) { - encoded = LibString.replace(encoded, "/", "_"); - } - if (_random() & 1 == 0) { - encoded = LibString.replace(encoded, "+", "-"); - } - - _misalignFreeMemoryPointer(); - bytes memory decoded = Base64.decode(encoded); - _checkMemory(decoded); - - assertEq(input, decoded); - - input = abi.encode(encoded); - } - } - - function testBase64EncodeFileSafeAndNoPadding(bytes memory input, bool fileSafe, bool noPadding) - public - { - string memory expectedEncoded = Base64.encode(input); - - if (fileSafe) { - expectedEncoded = LibString.replace(expectedEncoded, "+", "-"); - expectedEncoded = LibString.replace(expectedEncoded, "/", "_"); - } - if (noPadding) { - expectedEncoded = LibString.replace(expectedEncoded, "=", ""); - } - - assertEq(Base64.encode(input, fileSafe, noPadding), expectedEncoded); - } -} diff --git a/lib/solady/test/CREATE3.t.sol b/lib/solady/test/CREATE3.t.sol deleted file mode 100644 index 82693bd..0000000 --- a/lib/solady/test/CREATE3.t.sol +++ /dev/null @@ -1,121 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {WETH} from "../src/tokens/WETH.sol"; -import {MockERC20} from "./utils/mocks/MockERC20.sol"; -import {MockCd} from "./utils/mocks/MockCd.sol"; - -import {CREATE3} from "../src/utils/CREATE3.sol"; - -import "./utils/SoladyTest.sol"; - -contract CREATE3Test is SoladyTest { - function testDeployERC20() public { - bytes32 salt = keccak256(bytes("A salt!")); - - MockERC20 deployed = MockERC20( - this.deploy( - salt, - abi.encodePacked(type(MockERC20).creationCode, abi.encode("Mock Token", "MOCK", 18)), - 0 - ) - ); - - assertEq(address(deployed), CREATE3.getDeployed(salt)); - - assertEq(deployed.name(), "Mock Token"); - assertEq(deployed.symbol(), "MOCK"); - assertEq(deployed.decimals(), 18); - } - - function testDeployedUpperBitsSafeForPlainSolidity() public { - bytes32 salt = keccak256(bytes("A salt!")); - address deployed = CREATE3.getDeployed(salt); - uint256 someNumber = 123456789; - uint256 packed = (someNumber << 160) | uint160(deployed); - uint256 someNumberUnpacked = packed >> 160; - assertEq(someNumber, someNumberUnpacked); - } - - function testDoubleDeploySameBytecodeReverts() public { - bytes32 salt = keccak256(bytes("Salty...")); - - this.deploy(salt, type(MockCd).creationCode, 0); - vm.expectRevert(CREATE3.DeploymentFailed.selector); - this.deploy(salt, type(MockCd).creationCode, 0); - } - - function testDoubleDeployDifferentBytecodeReverts() public { - bytes32 salt = keccak256(bytes("and sweet!")); - - this.deploy(salt, type(WETH).creationCode, 0); - vm.expectRevert(CREATE3.DeploymentFailed.selector); - this.deploy(salt, type(MockCd).creationCode, 0); - } - - function testDeployERC20( - bytes32 salt, - string calldata name, - string calldata symbol, - uint8 decimals - ) public { - MockERC20 deployed = MockERC20( - this.deploy( - salt, - abi.encodePacked(type(MockERC20).creationCode, abi.encode(name, symbol, decimals)), - 0 - ) - ); - - assertEq(address(deployed), CREATE3.getDeployed(salt)); - - assertEq(deployed.name(), name); - assertEq(deployed.symbol(), symbol); - assertEq(deployed.decimals(), decimals); - } - - function testDoubleDeploySameBytecodeReverts(bytes32 salt, bytes calldata bytecode) public { - bytes memory creationCode = _creationCode(bytecode); - this.deploy(salt, creationCode, 0); - vm.expectRevert(CREATE3.DeploymentFailed.selector); - this.deploy(salt, creationCode, 0); - } - - function testDoubleDeployDifferentBytecodeReverts( - bytes32 salt, - bytes memory bytecode1, - bytes memory bytecode2 - ) public { - this.deploy(salt, _creationCode(bytecode1), 0); - vm.expectRevert(CREATE3.DeploymentFailed.selector); - this.deploy(salt, _creationCode(bytecode2), 0); - } - - function deploy(bytes32 salt, bytes calldata creationCode, uint256 value) - external - returns (address) - { - return CREATE3.deploy(salt, creationCode, value); - } - - function _creationCode(bytes memory bytecode) internal pure returns (bytes memory result) { - /// @solidity memory-safe-assembly - assembly { - // Trim the length if needed. - let length := mload(bytecode) - let maxLength := 24566 // `24576 - 0xa`. - if iszero(lt(length, maxLength)) { mstore(bytecode, maxLength) } - // The following snippet is from SSTORE2. - result := mload(0x40) - length := mload(bytecode) - let dataSize := add(length, 1) - mstore(0x40, and(add(add(result, dataSize), 0x60), not(0x1f))) - mstore(add(result, 0x0b), or(0x61000080600a3d393df300, shl(0x40, dataSize))) - mstore(result, add(dataSize, 0xa)) // Store the length of result. - // Copy the bytes over. - for { let i := 0 } lt(i, length) { i := add(i, 0x20) } { - mstore(add(add(bytecode, 0x20), i), mload(add(add(result, 0x2b), i))) - } - } - } -} diff --git a/lib/solady/test/DateTimeLib.t.sol b/lib/solady/test/DateTimeLib.t.sol deleted file mode 100644 index 099412a..0000000 --- a/lib/solady/test/DateTimeLib.t.sol +++ /dev/null @@ -1,796 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "../test/utils/SoladyTest.sol"; -import {DateTimeLib} from "../src/utils/DateTimeLib.sol"; - -contract DateTimeLibTest is SoladyTest { - struct DateTime { - uint256 year; - uint256 month; - uint256 day; - uint256 hour; - uint256 minute; - uint256 second; - } - - function testDateTimeMaxSupported() public { - DateTime memory d; - assertEq( - DateTimeLib.dateToEpochDay(DateTimeLib.MAX_SUPPORTED_YEAR, 12, 31), - DateTimeLib.MAX_SUPPORTED_EPOCH_DAY - ); - assertEq( - DateTimeLib.dateToTimestamp(DateTimeLib.MAX_SUPPORTED_YEAR, 12, 31) + 86400 - 1, - DateTimeLib.MAX_SUPPORTED_TIMESTAMP - ); - (d.year, d.month, d.day) = DateTimeLib.timestampToDate(DateTimeLib.MAX_SUPPORTED_TIMESTAMP); - assertTrue(d.year == DateTimeLib.MAX_SUPPORTED_YEAR && d.month == 12 && d.day == 31); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(DateTimeLib.MAX_SUPPORTED_EPOCH_DAY); - assertTrue(d.year == DateTimeLib.MAX_SUPPORTED_YEAR && d.month == 12 && d.day == 31); - (d.year, d.month, d.day) = - DateTimeLib.timestampToDate(DateTimeLib.MAX_SUPPORTED_TIMESTAMP + 1); - assertFalse(d.year == DateTimeLib.MAX_SUPPORTED_YEAR && d.month == 12 && d.day == 31); - (d.year, d.month, d.day) = - DateTimeLib.epochDayToDate(DateTimeLib.MAX_SUPPORTED_EPOCH_DAY + 1); - assertFalse(d.year == DateTimeLib.MAX_SUPPORTED_YEAR && d.month == 12 && d.day == 31); - } - - function testDateToEpochDay() public { - assertEq(DateTimeLib.dateToEpochDay(1970, 1, 1), 0); - assertEq(DateTimeLib.dateToEpochDay(1970, 1, 2), 1); - assertEq(DateTimeLib.dateToEpochDay(1970, 2, 1), 31); - assertEq(DateTimeLib.dateToEpochDay(1970, 3, 1), 59); - assertEq(DateTimeLib.dateToEpochDay(1970, 4, 1), 90); - assertEq(DateTimeLib.dateToEpochDay(1970, 5, 1), 120); - assertEq(DateTimeLib.dateToEpochDay(1970, 6, 1), 151); - assertEq(DateTimeLib.dateToEpochDay(1970, 7, 1), 181); - assertEq(DateTimeLib.dateToEpochDay(1970, 8, 1), 212); - assertEq(DateTimeLib.dateToEpochDay(1970, 9, 1), 243); - assertEq(DateTimeLib.dateToEpochDay(1970, 10, 1), 273); - assertEq(DateTimeLib.dateToEpochDay(1970, 11, 1), 304); - assertEq(DateTimeLib.dateToEpochDay(1970, 12, 1), 334); - assertEq(DateTimeLib.dateToEpochDay(1970, 12, 31), 364); - assertEq(DateTimeLib.dateToEpochDay(1971, 1, 1), 365); - assertEq(DateTimeLib.dateToEpochDay(1980, 11, 3), 3959); - assertEq(DateTimeLib.dateToEpochDay(2000, 3, 1), 11017); - assertEq(DateTimeLib.dateToEpochDay(2355, 12, 31), 140982); - assertEq(DateTimeLib.dateToEpochDay(99999, 12, 31), 35804721); - assertEq(DateTimeLib.dateToEpochDay(100000, 12, 31), 35805087); - assertEq(DateTimeLib.dateToEpochDay(604800, 2, 29), 220179195); - assertEq(DateTimeLib.dateToEpochDay(1667347200, 2, 29), 608985340227); - assertEq(DateTimeLib.dateToEpochDay(1667952000, 2, 29), 609206238891); - } - - function testDateToEpochDayGas() public { - unchecked { - uint256 sum; - for (uint256 i; i < 256; ++i) { - uint256 year = _bound(_random(), 1970, DateTimeLib.MAX_SUPPORTED_YEAR); - uint256 month = _bound(_random(), 1, 12); - uint256 md = DateTimeLib.daysInMonth(year, month); - uint256 day = _bound(_random(), 1, md); - uint256 epochDay = DateTimeLib.dateToEpochDay(year, month, day); - sum += epochDay; - } - assertTrue(sum != 0); - } - } - - function testDateToEpochDayGas2() public { - unchecked { - uint256 sum; - for (uint256 i; i < 256; ++i) { - uint256 year = _bound(_random(), 1970, DateTimeLib.MAX_SUPPORTED_YEAR); - uint256 month = _bound(_random(), 1, 12); - uint256 md = DateTimeLib.daysInMonth(year, month); - uint256 day = _bound(_random(), 1, md); - uint256 epochDay = _dateToEpochDayOriginal2(year, month, day); - sum += epochDay; - } - assertTrue(sum != 0); - } - } - - function testEpochDayToDateGas() public { - unchecked { - uint256 sum; - for (uint256 i; i < 256; ++i) { - uint256 epochDay = _bound(_random(), 0, DateTimeLib.MAX_SUPPORTED_EPOCH_DAY); - (uint256 year, uint256 month, uint256 day) = DateTimeLib.epochDayToDate(epochDay); - sum += year + month + day; - } - assertTrue(sum != 0); - } - } - - function testEpochDayToDateGas2() public { - unchecked { - uint256 sum; - for (uint256 i; i < 256; ++i) { - uint256 epochDay = _bound(_random(), 0, DateTimeLib.MAX_SUPPORTED_EPOCH_DAY); - (uint256 year, uint256 month, uint256 day) = _epochDayToDateOriginal2(epochDay); - sum += year + month + day; - } - assertTrue(sum != 0); - } - } - - function testDateToEpochDayDifferential(DateTime memory d) public { - d.year = _bound(d.year, 1970, DateTimeLib.MAX_SUPPORTED_YEAR); - d.month = _bound(d.month, 1, 12); - d.day = _bound(d.day, 1, DateTimeLib.daysInMonth(d.year, d.month)); - uint256 expectedResult = _dateToEpochDayOriginal(d.year, d.month, d.day); - assertEq(DateTimeLib.dateToEpochDay(d.year, d.month, d.day), expectedResult); - } - - function testDateToEpochDayDifferential2(DateTime memory d) public { - d.year = _bound(d.year, 1970, DateTimeLib.MAX_SUPPORTED_YEAR); - d.month = _bound(d.month, 1, 12); - d.day = _bound(d.day, 1, DateTimeLib.daysInMonth(d.year, d.month)); - uint256 expectedResult = _dateToEpochDayOriginal2(d.year, d.month, d.day); - assertEq(DateTimeLib.dateToEpochDay(d.year, d.month, d.day), expectedResult); - } - - function testEpochDayToDateDifferential(uint256 timestamp) public { - timestamp = _bound(timestamp, 0, DateTimeLib.MAX_SUPPORTED_TIMESTAMP); - DateTime memory a; - DateTime memory b; - (a.year, a.month, a.day) = _epochDayToDateOriginal(timestamp); - (b.year, b.month, b.day) = DateTimeLib.epochDayToDate(timestamp); - assertTrue(a.year == b.year && a.month == b.month && a.day == b.day); - } - - function testEpochDayToDateDifferential2(uint256 timestamp) public { - timestamp = _bound(timestamp, 0, DateTimeLib.MAX_SUPPORTED_TIMESTAMP); - DateTime memory a; - DateTime memory b; - (a.year, a.month, a.day) = _epochDayToDateOriginal2(timestamp); - (b.year, b.month, b.day) = DateTimeLib.epochDayToDate(timestamp); - assertTrue(a.year == b.year && a.month == b.month && a.day == b.day); - } - - function testDaysToDate() public { - DateTime memory d; - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(0); - assertTrue(d.year == 1970 && d.month == 1 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(31); - assertTrue(d.year == 1970 && d.month == 2 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(59); - assertTrue(d.year == 1970 && d.month == 3 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(90); - assertTrue(d.year == 1970 && d.month == 4 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(120); - assertTrue(d.year == 1970 && d.month == 5 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(151); - assertTrue(d.year == 1970 && d.month == 6 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(181); - assertTrue(d.year == 1970 && d.month == 7 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(212); - assertTrue(d.year == 1970 && d.month == 8 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(243); - assertTrue(d.year == 1970 && d.month == 9 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(273); - assertTrue(d.year == 1970 && d.month == 10 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(304); - assertTrue(d.year == 1970 && d.month == 11 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(334); - assertTrue(d.year == 1970 && d.month == 12 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(365); - assertTrue(d.year == 1971 && d.month == 1 && d.day == 1); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(10987); - assertTrue(d.year == 2000 && d.month == 1 && d.day == 31); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(18321); - assertTrue(d.year == 2020 && d.month == 2 && d.day == 29); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(156468); - assertTrue(d.year == 2398 && d.month == 5 && d.day == 25); - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(35805087); - assertTrue(d.year == 100000 && d.month == 12 && d.day == 31); - } - - function testEpochDayToDate(uint256 epochDay) public { - DateTime memory d; - (d.year, d.month, d.day) = DateTimeLib.epochDayToDate(epochDay); - assertEq(epochDay, DateTimeLib.dateToEpochDay(d.year, d.month, d.day)); - } - - function testDateToAndFroEpochDay(DateTime memory a) public { - a.year = _bound(a.year, 1970, DateTimeLib.MAX_SUPPORTED_YEAR); - a.month = _bound(a.month, 1, 12); - uint256 md = DateTimeLib.daysInMonth(a.year, a.month); - a.day = _bound(a.day, 1, md); - uint256 epochDay = DateTimeLib.dateToEpochDay(a.year, a.month, a.day); - DateTime memory b; - (b.year, b.month, b.day) = DateTimeLib.epochDayToDate(epochDay); - assertTrue(a.year == b.year && a.month == b.month && a.day == b.day); - } - - function testDateTimeToAndFroTimestamp(DateTime memory a) public { - a.year = _bound(a.year, 1970, DateTimeLib.MAX_SUPPORTED_YEAR); - a.month = _bound(a.month, 1, 12); - uint256 md = DateTimeLib.daysInMonth(a.year, a.month); - a.day = _bound(a.day, 1, md); - a.hour = _bound(a.hour, 0, 23); - a.minute = _bound(a.minute, 0, 59); - a.second = _bound(a.second, 0, 59); - uint256 timestamp = - DateTimeLib.dateTimeToTimestamp(a.year, a.month, a.day, a.hour, a.minute, a.second); - DateTime memory b; - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(timestamp); - assertTrue(a.year == b.year && a.month == b.month && a.day == b.day); - assertTrue(a.hour == b.hour && a.minute == b.minute && a.second == b.second); - } - - function testDateToAndFroEpochDay() public { - unchecked { - for (uint256 i; i < 256; ++i) { - uint256 year = _bound(_random(), 1970, DateTimeLib.MAX_SUPPORTED_YEAR); - uint256 month = _bound(_random(), 1, 12); - uint256 md = DateTimeLib.daysInMonth(year, month); - uint256 day = _bound(_random(), 1, md); - uint256 epochDay = DateTimeLib.dateToEpochDay(year, month, day); - (uint256 y, uint256 m, uint256 d) = DateTimeLib.epochDayToDate(epochDay); - assertTrue(year == y && month == m && day == d); - } - } - } - - function testDateToAndFroTimestamp() public { - unchecked { - for (uint256 i; i < 256; ++i) { - uint256 year = _bound(_random(), 1970, DateTimeLib.MAX_SUPPORTED_YEAR); - uint256 month = _bound(_random(), 1, 12); - uint256 md = DateTimeLib.daysInMonth(year, month); - uint256 day = _bound(_random(), 1, md); - uint256 timestamp = DateTimeLib.dateToTimestamp(year, month, day); - assertEq(timestamp, DateTimeLib.dateToEpochDay(year, month, day) * 86400); - (uint256 y, uint256 m, uint256 d) = DateTimeLib.timestampToDate(timestamp); - assertTrue(year == y && month == m && day == d); - } - } - } - - function testIsLeapYear() public { - assertTrue(DateTimeLib.isLeapYear(2000)); - assertTrue(DateTimeLib.isLeapYear(2024)); - assertTrue(DateTimeLib.isLeapYear(2048)); - assertTrue(DateTimeLib.isLeapYear(2072)); - assertTrue(DateTimeLib.isLeapYear(2104)); - assertTrue(DateTimeLib.isLeapYear(2128)); - assertTrue(DateTimeLib.isLeapYear(10032)); - assertTrue(DateTimeLib.isLeapYear(10124)); - assertTrue(DateTimeLib.isLeapYear(10296)); - assertTrue(DateTimeLib.isLeapYear(10400)); - assertTrue(DateTimeLib.isLeapYear(10916)); - } - - function testIsLeapYear(uint256 year) public { - assertEq( - DateTimeLib.isLeapYear(year), (year % 4 == 0) && (year % 100 != 0 || year % 400 == 0) - ); - } - - function testDaysInMonth() public { - assertEq(DateTimeLib.daysInMonth(2022, 1), 31); - assertEq(DateTimeLib.daysInMonth(2022, 2), 28); - assertEq(DateTimeLib.daysInMonth(2022, 3), 31); - assertEq(DateTimeLib.daysInMonth(2022, 4), 30); - assertEq(DateTimeLib.daysInMonth(2022, 5), 31); - assertEq(DateTimeLib.daysInMonth(2022, 6), 30); - assertEq(DateTimeLib.daysInMonth(2022, 7), 31); - assertEq(DateTimeLib.daysInMonth(2022, 8), 31); - assertEq(DateTimeLib.daysInMonth(2022, 9), 30); - assertEq(DateTimeLib.daysInMonth(2022, 10), 31); - assertEq(DateTimeLib.daysInMonth(2022, 11), 30); - assertEq(DateTimeLib.daysInMonth(2022, 12), 31); - assertEq(DateTimeLib.daysInMonth(2024, 1), 31); - assertEq(DateTimeLib.daysInMonth(2024, 2), 29); - assertEq(DateTimeLib.daysInMonth(1900, 2), 28); - } - - function testDaysInMonth(uint256 year, uint256 month) public { - month = _bound(month, 1, 12); - if (DateTimeLib.isLeapYear(year) && month == 2) { - assertEq(DateTimeLib.daysInMonth(year, month), 29); - } else if ( - month == 1 || month == 3 || month == 5 || month == 7 || month == 8 || month == 10 - || month == 12 - ) { - assertEq(DateTimeLib.daysInMonth(year, month), 31); - } else if (month == 2) { - assertEq(DateTimeLib.daysInMonth(year, month), 28); - } else { - assertEq(DateTimeLib.daysInMonth(year, month), 30); - } - } - - function testWeekday() public { - assertEq(DateTimeLib.weekday(1), 4); - assertEq(DateTimeLib.weekday(86400), 5); - assertEq(DateTimeLib.weekday(86401), 5); - assertEq(DateTimeLib.weekday(172800), 6); - assertEq(DateTimeLib.weekday(259200), 7); - assertEq(DateTimeLib.weekday(345600), 1); - assertEq(DateTimeLib.weekday(432000), 2); - assertEq(DateTimeLib.weekday(518400), 3); - } - - function testDayOfWeek() public { - uint256 timestamp = 0; - uint256 weekday = 3; - unchecked { - for (uint256 i = 0; i < 1000; ++i) { - assertEq(DateTimeLib.weekday(timestamp) - 1, weekday); - timestamp += 86400; - weekday = (weekday + 1) % 7; - } - } - } - - function testIsSupportedDateTrue() public { - assertTrue(DateTimeLib.isSupportedDate(1970, 1, 1)); - assertTrue(DateTimeLib.isSupportedDate(1971, 5, 31)); - assertTrue(DateTimeLib.isSupportedDate(1971, 6, 30)); - assertTrue(DateTimeLib.isSupportedDate(1971, 12, 31)); - assertTrue(DateTimeLib.isSupportedDate(1972, 2, 28)); - assertTrue(DateTimeLib.isSupportedDate(1972, 4, 30)); - assertTrue(DateTimeLib.isSupportedDate(1972, 5, 31)); - assertTrue(DateTimeLib.isSupportedDate(2000, 2, 29)); - assertTrue(DateTimeLib.isSupportedDate(DateTimeLib.MAX_SUPPORTED_YEAR, 5, 31)); - } - - function testIsSupportedDateFalse() public { - assertFalse(DateTimeLib.isSupportedDate(0, 0, 0)); - assertFalse(DateTimeLib.isSupportedDate(1970, 0, 0)); - assertFalse(DateTimeLib.isSupportedDate(1970, 1, 0)); - assertFalse(DateTimeLib.isSupportedDate(1969, 1, 1)); - assertFalse(DateTimeLib.isSupportedDate(1800, 1, 1)); - assertFalse(DateTimeLib.isSupportedDate(1970, 13, 1)); - assertFalse(DateTimeLib.isSupportedDate(1700, 13, 1)); - assertFalse(DateTimeLib.isSupportedDate(1970, 15, 32)); - assertFalse(DateTimeLib.isSupportedDate(1970, 1, 32)); - assertFalse(DateTimeLib.isSupportedDate(1970, 13, 1)); - assertFalse(DateTimeLib.isSupportedDate(1879, 1, 1)); - assertFalse(DateTimeLib.isSupportedDate(1970, 4, 31)); - assertFalse(DateTimeLib.isSupportedDate(1970, 6, 31)); - assertFalse(DateTimeLib.isSupportedDate(1970, 7, 32)); - assertFalse(DateTimeLib.isSupportedDate(2000, 2, 30)); - assertFalse(DateTimeLib.isSupportedDate(DateTimeLib.MAX_SUPPORTED_YEAR + 1, 5, 31)); - assertFalse(DateTimeLib.isSupportedDate(type(uint256).max, 5, 31)); - } - - function testIsSupportedDateTime(DateTime memory a) public { - a.month = _bound(a.month, 0, 20); - a.day = _bound(a.day, 0, 50); - a.hour = _bound(a.hour, 0, 50); - a.minute = _bound(a.minute, 0, 100); - a.second = _bound(a.second, 0, 100); - bool isSupported = (1970 <= a.year && a.year <= DateTimeLib.MAX_SUPPORTED_YEAR) - && (1 <= a.month && a.month <= 12) - && (1 <= a.day && a.day <= DateTimeLib.daysInMonth(a.year, a.month)) && (a.hour < 24) - && (a.minute < 60) && (a.second < 60); - assertEq( - DateTimeLib.isSupportedDateTime(a.year, a.month, a.day, a.hour, a.minute, a.second), - isSupported - ); - } - - function testIsSupportedEpochDayTrue() public { - assertTrue(DateTimeLib.isSupportedEpochDay(0)); - assertTrue(DateTimeLib.isSupportedEpochDay(DateTimeLib.MAX_SUPPORTED_EPOCH_DAY)); - } - - function testIsSupportedEpochDayFalse() public { - assertFalse(DateTimeLib.isSupportedEpochDay(DateTimeLib.MAX_SUPPORTED_EPOCH_DAY + 1)); - assertFalse(DateTimeLib.isSupportedEpochDay(DateTimeLib.MAX_SUPPORTED_EPOCH_DAY + 2)); - } - - function testIsSupportedTimestampTrue() public { - assertTrue(DateTimeLib.isSupportedTimestamp(0)); - assertTrue(DateTimeLib.isSupportedTimestamp(DateTimeLib.MAX_SUPPORTED_TIMESTAMP)); - } - - function testIsSupportedTimestampFalse() public { - assertFalse(DateTimeLib.isSupportedTimestamp(DateTimeLib.MAX_SUPPORTED_TIMESTAMP + 1)); - assertFalse(DateTimeLib.isSupportedTimestamp(DateTimeLib.MAX_SUPPORTED_TIMESTAMP + 2)); - } - - function testNthWeekdayInMonthOfYearTimestamp() public { - uint256 wd; - // 1st 2nd 3rd 4th monday in November 2022. - wd = DateTimeLib.MON; - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 11, 1, wd), 1667779200); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 11, 2, wd), 1668384000); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 11, 3, wd), 1668988800); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 11, 4, wd), 1669593600); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 11, 5, wd), 0); - - // 1st... 5th Wednesday in November 2022. - wd = DateTimeLib.WED; - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 11, 1, wd), 1667347200); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 11, 2, wd), 1667952000); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 11, 3, wd), 1668556800); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 11, 4, wd), 1669161600); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 11, 5, wd), 1669766400); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 11, 6, wd), 0); - - // 1st... 5th Friday in December 2022. - wd = DateTimeLib.FRI; - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 12, 1, wd), 1669939200); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 12, 2, wd), 1670544000); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 12, 3, wd), 1671148800); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 12, 4, wd), 1671753600); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 12, 5, wd), 1672358400); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2022, 12, 6, wd), 0); - - // 1st... 5th Sunday in January 2023. - wd = DateTimeLib.SUN; - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2023, 1, 1, wd), 1672531200); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2023, 1, 2, wd), 1673136000); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2023, 1, 3, wd), 1673740800); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2023, 1, 4, wd), 1674345600); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2023, 1, 5, wd), 1674950400); - assertEq(DateTimeLib.nthWeekdayInMonthOfYearTimestamp(2023, 1, 6, wd), 0); - } - - function testNthWeekdayInMonthOfYearTimestamp( - uint256 year, - uint256 month, - uint256 n, - uint256 weekday - ) public { - unchecked { - year = _bound(year, 1970, DateTimeLib.MAX_SUPPORTED_YEAR); - month = _bound(month, 1, 12); - n = _bound(n, 1, 10); - weekday = _bound(weekday, 1, 7); - // Count number of weekdays for the month in the year. - uint256 md = DateTimeLib.daysInMonth(year, month); - uint256 timestamp = DateTimeLib.dateToTimestamp(year, month, 1); - uint256 m; - uint256 found; - for (uint256 i; i < md;) { - if (DateTimeLib.weekday(timestamp) == weekday) { - if (++m == n) { - found = 1; - break; - } - } - if (m == 0) { - timestamp += 86400; - i += 1; - } else { - timestamp += 86400 * 7; - i += 7; - } - } - assertEq( - DateTimeLib.nthWeekdayInMonthOfYearTimestamp(year, month, n, weekday), - found * timestamp - ); - } - } - - function testMondayTimestamp() public { - // Thursday 01 January 1970 -> 0 - assertEq(DateTimeLib.mondayTimestamp(0), 0); - // Friday 02 January 1970 -> 86400 - assertEq(DateTimeLib.mondayTimestamp(86400), 0); - // Saturday 03 January 1970 -> 172800 - assertEq(DateTimeLib.mondayTimestamp(172800), 0); - // Sunday 04 January 1970 -> 259200 - assertEq(DateTimeLib.mondayTimestamp(259200), 0); - // Monday 05 January 19700 -> 345600 - assertEq(DateTimeLib.mondayTimestamp(345600), 345600); - // Monday 07 November 2022 -> 1667779200 - assertEq(DateTimeLib.mondayTimestamp(1667779200), 1667779200); - // Sunday 06 November 2022 -> 1667692800 - assertEq(DateTimeLib.mondayTimestamp(1667692800), 1667174400); - // Saturday 05 November 2022 -> 1667606400 - assertEq(DateTimeLib.mondayTimestamp(1667606400), 1667174400); - // Friday 04 November 2022 -> 1667520000 - assertEq(DateTimeLib.mondayTimestamp(1667520000), 1667174400); - // Thursday 03 November 2022 -> 1667433600 - assertEq(DateTimeLib.mondayTimestamp(1667433600), 1667174400); - // Wednesday 02 November 2022 -> 1667347200 - assertEq(DateTimeLib.mondayTimestamp(1667347200), 1667174400); - // Tuesday 01 November 2022 -> 1667260800 - assertEq(DateTimeLib.mondayTimestamp(1667260800), 1667174400); - // Monday 01 November 2022 -> 1667260800 - assertEq(DateTimeLib.mondayTimestamp(1667174400), 1667174400); - } - - function testMondayTimestamp(uint256 timestamp) public { - uint256 day = timestamp / 86400; - uint256 weekday = (day + 3) % 7; - assertEq( - DateTimeLib.mondayTimestamp(timestamp), timestamp > 345599 ? (day - weekday) * 86400 : 0 - ); - } - - function testIsWeekEnd(uint256 timestamp) public { - timestamp = _bound(timestamp, 0, DateTimeLib.MAX_SUPPORTED_TIMESTAMP); - uint256 weekday = DateTimeLib.weekday(timestamp); - assertEq( - DateTimeLib.isWeekEnd(timestamp), - weekday == DateTimeLib.SAT || weekday == DateTimeLib.SUN - ); - } - - function testAddSubDiffYears(uint256 timestamp, uint256 numYears) public { - uint256 maxNumYears = 1000000; - numYears = _bound(numYears, 0, maxNumYears); - timestamp = - _bound(timestamp, 0, DateTimeLib.MAX_SUPPORTED_TIMESTAMP - maxNumYears * 366 * 86400); - uint256 result = DateTimeLib.addYears(timestamp, numYears); - DateTime memory a; - DateTime memory b; - (a.year, a.month, a.day, a.hour, a.minute, a.second) = - DateTimeLib.timestampToDateTime(timestamp); - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(result); - if (numYears != 0) assertTrue(a.year != b.year); - if (a.day <= 28) assertEq(a.day, b.day); - assertTrue(a.month == b.month); - assertTrue(a.hour == b.hour && a.minute == b.minute && a.second == b.second); - uint256 diff = DateTimeLib.diffYears(timestamp, result); - assertTrue(diff == numYears); - result = DateTimeLib.subYears(result, numYears); - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(result); - assertTrue(a.year == b.year && a.month == b.month); - assertTrue(a.hour == b.hour && a.minute == b.minute && a.second == b.second); - } - - function testDateTimeArithmeticReverts() public { - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.addYears(2 ** 128 - 1, 2 ** 255 - 1); - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.subYears(2 ** 128 - 1, 2 ** 255 - 1); - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.diffYears(2 ** 128 - 1, 2 ** 127 - 1); - - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.addMonths(2 ** 128 - 1, 2 ** 255 - 1); - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.subMonths(2 ** 128 - 1, 2 ** 255 - 1); - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.diffMonths(2 ** 128 - 1, 2 ** 127 - 1); - - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.addDays(2 ** 128 - 1, 2 ** 255 - 1); - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.subDays(2 ** 128 - 1, 2 ** 255 - 1); - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.diffDays(2 ** 128 - 1, 2 ** 127 - 1); - - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.addHours(2 ** 128 - 1, 2 ** 255 - 1); - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.subHours(2 ** 128 - 1, 2 ** 255 - 1); - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.diffHours(2 ** 128 - 1, 2 ** 127 - 1); - - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.addMinutes(2 ** 128 - 1, 2 ** 255 - 1); - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.subMinutes(2 ** 128 - 1, 2 ** 255 - 1); - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.diffMinutes(2 ** 128 - 1, 2 ** 127 - 1); - - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.addSeconds(2 ** 128 - 1, 2 ** 255 - 1); - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.subSeconds(2 ** 128 - 1, 2 ** 255 - 1); - vm.expectRevert(stdError.arithmeticError); - DateTimeLib.diffSeconds(2 ** 128 - 1, 2 ** 127 - 1); - } - - function testAddSubDiffMonths(uint256 timestamp, uint256 numMonths) public { - uint256 maxNumMonths = 1000000; - numMonths = _bound(numMonths, 0, maxNumMonths); - timestamp = - _bound(timestamp, 0, DateTimeLib.MAX_SUPPORTED_TIMESTAMP - maxNumMonths * 32 * 86400); - uint256 result = DateTimeLib.addMonths(timestamp, numMonths); - DateTime memory a; - DateTime memory b; - (a.year, a.month, a.day, a.hour, a.minute, a.second) = - DateTimeLib.timestampToDateTime(timestamp); - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(result); - if (numMonths != 0) assertTrue(a.year != b.year || a.month != b.month); - if (a.day <= 28) assertEq(a.day, b.day); - assertTrue(a.hour == b.hour && a.minute == b.minute && a.second == b.second); - uint256 diff = DateTimeLib.diffMonths(timestamp, result); - assertTrue(diff == numMonths); - result = DateTimeLib.subMonths(result, numMonths); - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(result); - assertTrue(a.year == b.year && a.month == b.month); - assertTrue(a.hour == b.hour && a.minute == b.minute && a.second == b.second); - } - - function testAddSubDiffDays(uint256 timestamp, uint256 numDays) public { - uint256 maxNumDays = 100000000; - numDays = _bound(numDays, 0, maxNumDays); - timestamp = _bound(timestamp, 0, DateTimeLib.MAX_SUPPORTED_TIMESTAMP - maxNumDays * 86400); - uint256 result = DateTimeLib.addDays(timestamp, numDays); - DateTime memory a; - DateTime memory b; - (a.year, a.month, a.day, a.hour, a.minute, a.second) = - DateTimeLib.timestampToDateTime(timestamp); - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(result); - if (numDays != 0) { - assertTrue(a.year != b.year || a.month != b.month || a.day != b.day); - } - assertTrue(a.hour == b.hour && a.minute == b.minute && a.second == b.second); - uint256 diff = DateTimeLib.diffDays(timestamp, result); - assertTrue(diff == numDays); - result = DateTimeLib.subDays(result, numDays); - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(result); - assertTrue(a.year == b.year && a.month == b.month); - assertTrue(a.hour == b.hour && a.minute == b.minute && a.second == b.second); - } - - function testAddSubDiffHours(uint256 timestamp, uint256 numHours) public { - uint256 maxNumHours = 10000000000; - numHours = _bound(numHours, 0, maxNumHours); - timestamp = _bound(timestamp, 0, DateTimeLib.MAX_SUPPORTED_TIMESTAMP - maxNumHours * 3600); - uint256 result = DateTimeLib.addHours(timestamp, numHours); - DateTime memory a; - DateTime memory b; - (a.year, a.month, a.day, a.hour, a.minute, a.second) = - DateTimeLib.timestampToDateTime(timestamp); - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(result); - if (numHours != 0) { - assertTrue(a.year != b.year || a.month != b.month || a.day != b.day || a.hour != b.hour); - } - assertTrue(a.minute == b.minute && a.second == b.second); - uint256 diff = DateTimeLib.diffHours(timestamp, result); - assertTrue(diff == numHours); - result = DateTimeLib.subHours(result, numHours); - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(result); - assertTrue(a.year == b.year && a.month == b.month); - assertTrue(a.hour == b.hour && a.minute == b.minute && a.second == b.second); - } - - function testAddSubDiffMinutes(uint256 timestamp, uint256 numMinutes) public { - uint256 maxNumMinutes = 10000000000; - numMinutes = _bound(numMinutes, 0, maxNumMinutes); - timestamp = _bound(timestamp, 0, DateTimeLib.MAX_SUPPORTED_TIMESTAMP - maxNumMinutes * 60); - uint256 result = DateTimeLib.addMinutes(timestamp, numMinutes); - DateTime memory a; - DateTime memory b; - (a.year, a.month, a.day, a.hour, a.minute, a.second) = - DateTimeLib.timestampToDateTime(timestamp); - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(result); - if (numMinutes != 0) { - assertTrue( - (a.year != b.year || a.month != b.month || a.day != b.day) - || (a.hour != b.hour || a.minute != b.minute) - ); - } - assertTrue(a.second == b.second); - uint256 diff = DateTimeLib.diffMinutes(timestamp, result); - assertTrue(diff == numMinutes); - result = DateTimeLib.subMinutes(result, numMinutes); - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(result); - assertTrue(a.year == b.year && a.month == b.month); - assertTrue(a.hour == b.hour && a.minute == b.minute && a.second == b.second); - } - - function testAddSubDiffSeconds(uint256 timestamp, uint256 numSeconds) public { - uint256 maxNumSeconds = 1000000000000; - numSeconds = _bound(numSeconds, 0, maxNumSeconds); - timestamp = _bound(timestamp, 0, DateTimeLib.MAX_SUPPORTED_TIMESTAMP - maxNumSeconds); - uint256 result = DateTimeLib.addSeconds(timestamp, numSeconds); - DateTime memory a; - DateTime memory b; - (a.year, a.month, a.day, a.hour, a.minute, a.second) = - DateTimeLib.timestampToDateTime(timestamp); - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(result); - if (numSeconds != 0) { - assertTrue( - (a.year != b.year || a.month != b.month || a.day != b.day) - || (a.hour != b.hour || a.minute != b.minute || a.second != b.second) - ); - } - uint256 diff = DateTimeLib.diffSeconds(timestamp, result); - assertTrue(diff == numSeconds); - result = DateTimeLib.subSeconds(result, numSeconds); - (b.year, b.month, b.day, b.hour, b.minute, b.second) = - DateTimeLib.timestampToDateTime(result); - assertTrue(a.year == b.year && a.month == b.month); - assertTrue(a.hour == b.hour && a.minute == b.minute && a.second == b.second); - } - - function _dateToEpochDayOriginal(uint256 year, uint256 month, uint256 day) - internal - pure - returns (uint256) - { - unchecked { - if (month <= 2) { - year -= 1; - } - uint256 era = year / 400; - uint256 yoe = year - era * 400; - uint256 doy = (153 * (month > 2 ? month - 3 : month + 9) + 2) / 5 + day - 1; - uint256 doe = yoe * 365 + yoe / 4 - yoe / 100 + doy; - return era * 146097 + doe - 719468; - } - } - - function _dateToEpochDayOriginal2(uint256 year, uint256 month, uint256 day) - internal - pure - returns (uint256 _days) - { - unchecked { - int256 _year = int256(year); - int256 _month = int256(month); - int256 _day = int256(day); - - int256 _m = (_month - 14) / 12; - int256 __days = _day - 32075 + ((1461 * (_year + 4800 + _m)) / 4) - + ((367 * (_month - 2 - _m * 12)) / 12) - ((3 * ((_year + 4900 + _m) / 100)) / 4) - - 2440588; - - _days = uint256(__days); - } - } - - function _epochDayToDateOriginal(uint256 timestamp) - internal - pure - returns (uint256 year, uint256 month, uint256 day) - { - unchecked { - timestamp += 719468; - uint256 era = timestamp / 146097; - uint256 doe = timestamp - era * 146097; - uint256 yoe = (doe - doe / 1460 + doe / 36524 - doe / 146096) / 365; - year = yoe + era * 400; - uint256 doy = doe - (365 * yoe + yoe / 4 - yoe / 100); - uint256 mp = (5 * doy + 2) / 153; - day = doy - (153 * mp + 2) / 5 + 1; - month = mp < 10 ? mp + 3 : mp - 9; - if (month <= 2) { - year += 1; - } - } - } - - function _epochDayToDateOriginal2(uint256 _days) - internal - pure - returns (uint256 year, uint256 month, uint256 day) - { - unchecked { - int256 __days = int256(_days); - - int256 L = __days + 68569 + 2440588; - int256 N = (4 * L) / 146097; - L = L - (146097 * N + 3) / 4; - int256 _year = (4000 * (L + 1)) / 1461001; - L = L - (1461 * _year) / 4 + 31; - int256 _month = (80 * L) / 2447; - int256 _day = L - (2447 * _month) / 80; - L = _month / 11; - _month = _month + 2 - 12 * L; - _year = 100 * (N - 49) + _year + L; - - year = uint256(_year); - month = uint256(_month); - day = uint256(_day); - } - } -} diff --git a/lib/solady/test/DynamicBufferLib.t.sol b/lib/solady/test/DynamicBufferLib.t.sol deleted file mode 100644 index 281553f..0000000 --- a/lib/solady/test/DynamicBufferLib.t.sol +++ /dev/null @@ -1,356 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {DynamicBufferLib} from "../src/utils/DynamicBufferLib.sol"; - -contract DynamicBufferLibTest is SoladyTest { - using DynamicBufferLib for DynamicBufferLib.DynamicBuffer; - - function testClear(uint256) public { - DynamicBufferLib.DynamicBuffer memory buffer; - bytes memory b0 = _generateRandomBytes(128, _random()); - bytes memory b1 = _generateRandomBytes(256, _random()); - bytes memory emptyBytes; - assertEq(buffer.data.length, 0); - assertEq(emptyBytes.length, 0); - if (_random() & 1 == 0) buffer.clear(); - assertEq(buffer.data.length, 0); - assertEq(emptyBytes.length, 0); - buffer.clear().p(b0); - assertEq(buffer.data, b0); - assertEq(emptyBytes.length, 0); - uint256 n0 = _bound(_random(), 0, 1024); - uint256 n1 = _bound(_random(), 0, 4096); - buffer.reserve(n0).p(b1).clear().reserve(n1); - assertEq(buffer.data.length, 0); - assertEq(emptyBytes.length, 0); - buffer.p(b1); - assertEq(buffer.data, b1); - assertEq(emptyBytes.length, 0); - buffer.p(b0); - assertEq(buffer.data, abi.encodePacked(b1, b0)); - assertEq(emptyBytes.length, 0); - buffer.clear(); - } - - function testDynamicBufferReserveFromEmpty() public { - uint256 m = _freeMemoryPointer(); - DynamicBufferLib.DynamicBuffer memory buffer; - assertEq(_freeMemoryPointer(), m + 0x20); - buffer.reserve(0x200); - assertTrue(_freeMemoryPointer() > m + 0x20); - assertTrue(_freeMemoryPointer() < 0xffff); - m = _freeMemoryPointer(); - buffer.reserve(0x200); - assertEq(_freeMemoryPointer(), m); - buffer.reserve(0x200); - assertEq(_freeMemoryPointer(), m); - } - - function testDynamicBufferReserveFromEmpty2() public { - DynamicBufferLib.DynamicBuffer memory buffer; - _incrementFreeMemoryPointer(); - buffer.reserve(0x200); - uint256 m = _freeMemoryPointer(); - buffer.reserve(0x200); - assertEq(_freeMemoryPointer(), m); - buffer.reserve(0x200); - assertEq(_freeMemoryPointer(), m); - } - - function testDynamicBufferReserveFromEmpty3(bytes calldata b, uint256 t) public { - DynamicBufferLib.DynamicBuffer memory buffer; - if (t & 1 == 0) _incrementFreeMemoryPointer(); - if (t & 2 == 0) buffer.p(_generateRandomBytes((t >> 32) & 0xff, 1)); - if (t & 4 == 0) buffer.p(b); - assertTrue(_freeMemoryPointer() < 0xffffff); - uint256 r = t >> 240; - buffer.reserve(r); - assertTrue(_freeMemoryPointer() < 0xffffff); - uint256 m = _freeMemoryPointer(); - buffer.reserve(r); - assertEq(_freeMemoryPointer(), m); - buffer.reserve(r); - assertEq(_freeMemoryPointer(), m); - } - - function _incrementFreeMemoryPointer() internal pure { - /// @solidity memory-safe-assembly - assembly { - mstore(0x40, add(mload(0x40), 0x20)) - } - } - - function _freeMemoryPointer() internal pure returns (uint256 m) { - /// @solidity memory-safe-assembly - assembly { - m := mload(0x40) - } - } - - function _bufferLocation(DynamicBufferLib.DynamicBuffer memory buffer) - internal - pure - returns (uint256 result) - { - /// @solidity memory-safe-assembly - assembly { - result := mload(buffer) - } - } - - function testDynamicBuffer(uint256) public brutalizeMemory { - unchecked { - if (_random() & 7 == 0) _misalignFreeMemoryPointer(); - DynamicBufferLib.DynamicBuffer memory bufferA; - DynamicBufferLib.DynamicBuffer memory bufferB; - uint256 z = _bound(_random(), 32, 4096); - if (_random() & 7 == 0) bufferA.reserve(_random() % z); - if (_random() & 7 == 0) bufferB.reserve(_random() % z); - uint256 r = _random() % 3; - uint256 o = _bound(_random(), 0, 32); - uint256 n = _bound(_random(), 5, _random() & 7 == 0 ? 64 : 8); - z = z + z; - - if (r == 0) { - for (uint256 i; i != n; ++i) { - if (_random() & 7 == 0) bufferA.reserve(_random() % z); - bufferA.p(_generateRandomBytes(i + o, i + z)); - } - for (uint256 i; i != n; ++i) { - if (_random() & 7 == 0) bufferB.reserve(_random() % z); - bufferB.p(_generateRandomBytes(i + o, i + z)); - } - } else if (r == 1) { - for (uint256 i; i != n; ++i) { - if (_random() & 7 == 0) bufferB.reserve(_random() % z); - bufferB.p(_generateRandomBytes(i + o, i + z)); - } - for (uint256 i; i != n; ++i) { - if (_random() & 7 == 0) bufferA.reserve(_random() % z); - bufferA.p(_generateRandomBytes(i + o, i + z)); - } - } else { - uint256 mode; - for (uint256 i; i != n; ++i) { - if (_random() & 7 == 0) mode ^= 1; - if (mode == 0) { - if (_random() & 7 == 0) bufferA.reserve(_random() % z); - bufferA.p(_generateRandomBytes(i + o, i + z)); - if (_random() & 7 == 0) bufferB.reserve(_random() % z); - bufferB.p(_generateRandomBytes(i + o, i + z)); - } else { - if (_random() & 7 == 0) bufferB.reserve(_random() % z); - bufferB.p(_generateRandomBytes(i + o, i + z)); - if (_random() & 7 == 0) bufferA.reserve(_random() % z); - bufferA.p(_generateRandomBytes(i + o, i + z)); - } - } - } - - bytes memory expected; - for (uint256 i; i != n; ++i) { - expected = bytes.concat(expected, _generateRandomBytes(i + o, i + z)); - } - assertEq(bufferA.data, expected); - assertEq(bufferB.data, expected); - } - } - - function _generateRandomBytes(uint256 n, uint256 seed) - internal - pure - returns (bytes memory result) - { - /// @solidity memory-safe-assembly - assembly { - if n { - result := mload(0x40) - mstore(result, n) - mstore(0x00, seed) - for { let i := 0 } lt(i, n) { i := add(i, 0x20) } { - mstore(0x20, i) - mstore(add(add(result, 0x20), i), keccak256(0x00, 0x40)) - } - mstore(0x40, add(add(result, 0x20), n)) - } - } - } - - function testDynamicBuffer(bytes[] memory inputs, uint256 t) public brutalizeMemory { - _boundInputs(inputs); - - uint256 sharedLocation; - if ((t >> 128) & 1 == 0) { - bytes memory first = _generateRandomBytes((t & 0xff | 1), t); - bytes memory expectedResult = first; - for (uint256 i; i < inputs.length; ++i) { - expectedResult = bytes.concat(expectedResult, inputs[i]); - } - DynamicBufferLib.DynamicBuffer memory buffer; - buffer.p(first); - uint256 location = _bufferLocation(buffer); - for (uint256 i; i < inputs.length; ++i) { - buffer.p(inputs[i]); - assertEq(_bufferLocation(buffer), location); - _checkMemory(buffer.data); - } - assertEq(buffer.data, expectedResult); - sharedLocation = _bufferLocation(buffer); - } - - if ((t >> 129) & 1 == 0) { - if ((t >> 16) & 7 == 0) _misalignFreeMemoryPointer(); - DynamicBufferLib.DynamicBuffer memory buffer; - if ((t >> 160) & 3 == 0) _incrementFreeMemoryPointer(); - if ((t >> 130) & 1 == 0 && sharedLocation != 0) { - /// @solidity memory-safe-assembly - assembly { - mstore(buffer, sharedLocation) - } - buffer.clear(); - } - if ((t >> 162) & 3 == 0) _incrementFreeMemoryPointer(); - if ((t >> 32) & 3 == 0) { - buffer.reserve((t >> 128) % 1024); - } - - unchecked { - uint256 expectedLength; - uint256 start; - if (t & 1 == 0) { - if (inputs.length > 0) { - expectedLength = inputs[0].length; - buffer.data = inputs[0]; - start = 1; - } - } - for (uint256 i = start; i < inputs.length; ++i) { - expectedLength += inputs[i].length; - // Manually store the t in the next free memory word, - // and then check if p will corrupt it - // (in the case of insufficient memory allocation). - uint256 corruptCheckSlot; - /// @solidity memory-safe-assembly - assembly { - corruptCheckSlot := mload(0x40) - mstore(corruptCheckSlot, t) - mstore(0x40, add(corruptCheckSlot, 0x20)) - } - buffer.p(inputs[i]); - if ((t >> 48) & 7 == 0 && expectedLength != 0) { - buffer.reserve((t >> 160) % (expectedLength * 2)); - } - assertEq(buffer.data.length, expectedLength); - _checkMemory(buffer.data); - bool isCorrupted; - /// @solidity memory-safe-assembly - assembly { - isCorrupted := iszero(eq(t, mload(corruptCheckSlot))) - } - assertFalse(isCorrupted); - } - } - - bytes memory expectedResult; - unchecked { - for (uint256 i; i < inputs.length; ++i) { - expectedResult = bytes.concat(expectedResult, inputs[i]); - } - } - - assertEq(keccak256(buffer.data), keccak256(expectedResult)); - } - } - - function testJoinWithConcat() public { - bytes memory expectedResult; - (bytes[] memory chunks, bytes32 joinedHash) = _getChunks(); - unchecked { - for (uint256 i; i < chunks.length; ++i) { - expectedResult = bytes.concat(expectedResult, chunks[i]); - } - } - assertEq(keccak256(expectedResult), joinedHash); - } - - function testJoinWithDynamicBuffer() public { - DynamicBufferLib.DynamicBuffer memory buffer; - (bytes[] memory chunks, bytes32 joinedHash) = _getChunks(); - unchecked { - for (uint256 i; i < chunks.length; ++i) { - buffer.p(chunks[i]); - } - } - assertEq(keccak256(buffer.data), joinedHash); - } - - function testDynamicBufferChaining() public { - DynamicBufferLib.DynamicBuffer memory bufferA; - DynamicBufferLib.DynamicBuffer memory bufferB; - bufferA = bufferB.p("0", "1"); - _checkSamePointers(bufferA, bufferB); - bufferA = bufferB.p("0", "1", "2"); - _checkSamePointers(bufferA, bufferB); - bufferA = bufferB.p("0", "1", "2", "3"); - _checkSamePointers(bufferA, bufferB); - bufferA = bufferB.p("0", "1", "2", "3", "4"); - _checkSamePointers(bufferA, bufferB); - bufferA = bufferB.p("0", "1", "2", "3", "4", "5"); - _checkSamePointers(bufferA, bufferB); - bufferA = bufferB.p("0", "1", "2", "3", "4", "5", "6"); - _checkSamePointers(bufferA, bufferB); - assertEq(bufferA.data, "010120123012340123450123456"); - assertEq(bufferB.data, "010120123012340123450123456"); - } - - function _checkSamePointers( - DynamicBufferLib.DynamicBuffer memory a, - DynamicBufferLib.DynamicBuffer memory b - ) internal { - bool isSamePointer; - assembly { - isSamePointer := eq(a, b) - } - assertTrue(isSamePointer); - } - - function _getChunks() internal pure returns (bytes[] memory chunks, bytes32 joinedHash) { - chunks = new bytes[](20); - chunks[0] = bytes( - "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua." - ); - chunks[1] = bytes("Vitae suscipit tellus mauris a diam maecenas sed enim ut."); - chunks[2] = bytes("Nisl nisi scelerisque eu ultrices vitae auctor eu augue."); - chunks[3] = bytes("Et pharetra pharetra massa massa ultricies mi quis."); - chunks[4] = bytes("Ullamcorper malesuada proin libero nunc."); - chunks[5] = bytes("Tempus imperdiet nulla malesuada pellentesque."); - chunks[6] = bytes("Nunc congue nisi vitae suscipit tellus mauris."); - chunks[7] = bytes("Eu augue ut lectus arcu."); - chunks[8] = bytes("Natoque penatibus et magnis dis parturient montes nascetur."); - chunks[9] = bytes("Convallis posuere morbi leo urna."); - - chunks[15] = bytes("Hehe"); - - joinedHash = 0x166b0e99fea53034ed188896344996efc141b922127f90922905e478cb26b312; - } - - function _boundInputs(bytes[] memory inputs) internal pure { - // Limit the total number of inputs. - /// @solidity memory-safe-assembly - assembly { - if gt(mload(inputs), 16) { mstore(inputs, 16) } - } - unchecked { - // Limit the lengths of the inputs. - for (uint256 i; i < inputs.length; ++i) { - bytes memory x = inputs[i]; - /// @solidity memory-safe-assembly - assembly { - if gt(mload(x), 128) { mstore(x, 128) } - } - } - } - } -} diff --git a/lib/solady/test/ECDSA.t.sol b/lib/solady/test/ECDSA.t.sol deleted file mode 100644 index 3f6307f..0000000 --- a/lib/solady/test/ECDSA.t.sol +++ /dev/null @@ -1,452 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {ECDSA} from "../src/utils/ECDSA.sol"; -import {LibString} from "../src/utils/LibString.sol"; - -contract ECDSATest is SoladyTest { - using ECDSA for bytes32; - using ECDSA for bytes; - - bytes32 constant TEST_MESSAGE = - 0x7dbaf558b0a1a5dc7a67202117ab143c1d8605a983e4a743bc06fcc03162dc0d; - - bytes32 constant WRONG_MESSAGE = - 0x2d0828dd7c97cff316356da3c16c68ba2316886a0e05ebafb8291939310d51a3; - - address constant SIGNER = 0x70997970C51812dc3A010C7d01b50e0d17dc79C8; - - address constant V0_SIGNER = 0x2cc1166f6212628A0deEf2B33BEFB2187D35b86c; - - address constant V1_SIGNER = 0x1E318623aB09Fe6de3C9b8672098464Aeda9100E; - - function testTryRecoverWithInvalidShortSignatureReturnsZero() public { - bytes memory signature = hex"1234"; - assertTrue(this.tryRecover(TEST_MESSAGE, signature) == address(0)); - } - - function testTryRecoverWithInvalidLongSignatureReturnsZero() public { - bytes memory signature = - hex"01234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789"; - assertTrue(this.tryRecover(TEST_MESSAGE, signature) == address(0)); - } - - function testTryRecoverWithValidSignature() public { - bytes memory signature = - hex"8688e590483917863a35ef230c0f839be8418aa4ee765228eddfcea7fe2652815db01c2c84b0ec746e1b74d97475c599b3d3419fa7181b4e01de62c02b721aea1b"; - assertTrue(this.tryRecover(TEST_MESSAGE.toEthSignedMessageHash(), signature) == SIGNER); - } - - function testTryRecoverWithWrongSigner() public { - bytes memory signature = - hex"8688e590483917863a35ef230c0f839be8418aa4ee765228eddfcea7fe2652815db01c2c84b0ec746e1b74d97475c599b3d3419fa7181b4e01de62c02b721aea1b"; - assertTrue(this.tryRecover(WRONG_MESSAGE.toEthSignedMessageHash(), signature) != SIGNER); - } - - function testTryRecoverWithInvalidSignature() public { - bytes memory signature = - hex"332ce75a821c982f9127538858900d87d3ec1f9f737338ad67cad133fa48feff48e6fa0c18abc62e42820f05943e47af3e9fbe306ce74d64094bdf1691ee53e01c"; - assertTrue(this.tryRecover(TEST_MESSAGE.toEthSignedMessageHash(), signature) != SIGNER); - } - - function testTryRecoverWithV0SignatureWithVersion00ReturnsZero() public { - bytes memory signature = - hex"5d99b6f7f6d1f73d1a26497f2b1c89b24c0993913f86e9a2d02cd69887d9c94f3c880358579d811b21dd1b7fd9bb01c1d81d10e69f0384e675c32b39643be89200"; - assertTrue(this.tryRecover(TEST_MESSAGE, signature) == address(0)); - } - - function testTryRecoverWithV0SignatureWithVersion27() public { - bytes memory signature = - hex"5d99b6f7f6d1f73d1a26497f2b1c89b24c0993913f86e9a2d02cd69887d9c94f3c880358579d811b21dd1b7fd9bb01c1d81d10e69f0384e675c32b39643be8921b"; - assertTrue(this.tryRecover(TEST_MESSAGE, signature) == V0_SIGNER); - } - - function testTryRecoverWithV0SignatureWithWrongVersionReturnsZero() public { - bytes memory signature = - hex"5d99b6f7f6d1f73d1a26497f2b1c89b24c0993913f86e9a2d02cd69887d9c94f3c880358579d811b21dd1b7fd9bb01c1d81d10e69f0384e675c32b39643be89202"; - assertTrue(this.tryRecover(TEST_MESSAGE, signature) == address(0)); - } - - function testTryRecoverWithV0SignatureWithShortEIP2098Format() public { - bytes32 r = 0x5d99b6f7f6d1f73d1a26497f2b1c89b24c0993913f86e9a2d02cd69887d9c94f; - bytes32 vs = 0x3c880358579d811b21dd1b7fd9bb01c1d81d10e69f0384e675c32b39643be892; - assertTrue(this.tryRecover(TEST_MESSAGE, r, vs) == V0_SIGNER); - } - - function testTryRecoverWithV0SignatureWithShortEIP2098FormatAsCalldata() public { - bytes memory signature = - hex"5d99b6f7f6d1f73d1a26497f2b1c89b24c0993913f86e9a2d02cd69887d9c94f3c880358579d811b21dd1b7fd9bb01c1d81d10e69f0384e675c32b39643be892"; - assertTrue(this.tryRecover(TEST_MESSAGE, signature) == V0_SIGNER); - } - - function testTryRecoverWithV1SignatureWithVersion01ReturnsZero() public { - bytes memory signature = - hex"331fe75a821c982f9127538858900d87d3ec1f9f737338ad67cad133fa48feff48e6fa0c18abc62e42820f05943e47af3e9fbe306ce74d64094bdf1691ee53e001"; - assertTrue(this.tryRecover(TEST_MESSAGE, signature) == address(0)); - } - - function testTryRecoverWithV1SignatureWithVersion28() public { - bytes memory signature = - hex"331fe75a821c982f9127538858900d87d3ec1f9f737338ad67cad133fa48feff48e6fa0c18abc62e42820f05943e47af3e9fbe306ce74d64094bdf1691ee53e01c"; - assertTrue(this.tryRecover(TEST_MESSAGE, signature) == V1_SIGNER); - } - - function testTryRecoverWithV1SignatureWithWrongVersionReturnsZero() public { - bytes memory signature = - hex"331fe75a821c982f9127538858900d87d3ec1f9f737338ad67cad133fa48feff48e6fa0c18abc62e42820f05943e47af3e9fbe306ce74d64094bdf1691ee53e002"; - assertTrue(this.tryRecover(TEST_MESSAGE, signature) == address(0)); - } - - function testTryRecoverWithV1SignatureWithShortEIP2098Format() public { - bytes32 r = 0x331fe75a821c982f9127538858900d87d3ec1f9f737338ad67cad133fa48feff; - bytes32 vs = 0xc8e6fa0c18abc62e42820f05943e47af3e9fbe306ce74d64094bdf1691ee53e0; - assertTrue(this.tryRecover(TEST_MESSAGE, r, vs) == V1_SIGNER); - } - - function testTryRecoverWithV1SignatureWithShortEIP2098FormatAsCalldata() public { - bytes memory signature = - hex"331fe75a821c982f9127538858900d87d3ec1f9f737338ad67cad133fa48feffc8e6fa0c18abc62e42820f05943e47af3e9fbe306ce74d64094bdf1691ee53e0"; - assertTrue(this.tryRecover(TEST_MESSAGE, signature) == V1_SIGNER); - } - - function testRecoverWithInvalidShortSignatureReturnsZero() public { - bytes memory signature = hex"1234"; - vm.expectRevert(ECDSA.InvalidSignature.selector); - this.recover(TEST_MESSAGE, signature); - } - - function testRecoverWithInvalidLongSignatureReverts() public { - bytes memory signature = - hex"01234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789"; - vm.expectRevert(ECDSA.InvalidSignature.selector); - this.recover(TEST_MESSAGE, signature); - } - - function testRecoverWithValidSignature() public { - bytes memory signature = - hex"8688e590483917863a35ef230c0f839be8418aa4ee765228eddfcea7fe2652815db01c2c84b0ec746e1b74d97475c599b3d3419fa7181b4e01de62c02b721aea1b"; - address recovered = this.recover(TEST_MESSAGE.toEthSignedMessageHash(), signature); - assertTrue(recovered == SIGNER); - } - - function testRecoverWithWrongSigner() public { - bytes memory signature = - hex"8688e590483917863a35ef230c0f839be8418aa4ee765228eddfcea7fe2652815db01c2c84b0ec746e1b74d97475c599b3d3419fa7181b4e01de62c02b721aea1b"; - assertTrue(this.recover(WRONG_MESSAGE.toEthSignedMessageHash(), signature) != SIGNER); - } - - function testRecoverWithInvalidSignatureReverts() public { - bytes memory signature = - hex"332ce75a821c982f9127538858900d87d3ec1f9f737338ad67cad133fa48feff48e6fa0c18abc62e42820f05943e47af3e9fbe306ce74d64094bdf1691ee53e01c"; - vm.expectRevert(ECDSA.InvalidSignature.selector); - this.recover(TEST_MESSAGE.toEthSignedMessageHash(), signature); - } - - function testRecoverWithV0SignatureWithVersion00Reverts() public { - bytes memory signature = - hex"5d99b6f7f6d1f73d1a26497f2b1c89b24c0993913f86e9a2d02cd69887d9c94f3c880358579d811b21dd1b7fd9bb01c1d81d10e69f0384e675c32b39643be89200"; - vm.expectRevert(ECDSA.InvalidSignature.selector); - this.recover(TEST_MESSAGE, signature); - } - - function testRecoverWithV0SignatureWithVersion27() public { - bytes memory signature = - hex"5d99b6f7f6d1f73d1a26497f2b1c89b24c0993913f86e9a2d02cd69887d9c94f3c880358579d811b21dd1b7fd9bb01c1d81d10e69f0384e675c32b39643be8921b"; - assertTrue(this.recover(TEST_MESSAGE, signature) == V0_SIGNER); - } - - function testRecoverWithV0SignatureWithWrongVersionReverts() public { - bytes memory signature = - hex"5d99b6f7f6d1f73d1a26497f2b1c89b24c0993913f86e9a2d02cd69887d9c94f3c880358579d811b21dd1b7fd9bb01c1d81d10e69f0384e675c32b39643be89202"; - vm.expectRevert(ECDSA.InvalidSignature.selector); - this.recover(TEST_MESSAGE, signature); - } - - function testRecoverWithV0SignatureWithShortEIP2098Format() public { - bytes32 r = 0x5d99b6f7f6d1f73d1a26497f2b1c89b24c0993913f86e9a2d02cd69887d9c94f; - bytes32 vs = 0x3c880358579d811b21dd1b7fd9bb01c1d81d10e69f0384e675c32b39643be892; - assertTrue(this.recover(TEST_MESSAGE, r, vs) == V0_SIGNER); - } - - function testRecoverWithV0SignatureWithShortEIP2098FormatAsCalldata() public { - bytes memory signature = - hex"5d99b6f7f6d1f73d1a26497f2b1c89b24c0993913f86e9a2d02cd69887d9c94f3c880358579d811b21dd1b7fd9bb01c1d81d10e69f0384e675c32b39643be892"; - this.recover(TEST_MESSAGE, signature); - } - - function testRecoverWithV1SignatureWithVersion01Reverts() public { - bytes memory signature = - hex"331fe75a821c982f9127538858900d87d3ec1f9f737338ad67cad133fa48feff48e6fa0c18abc62e42820f05943e47af3e9fbe306ce74d64094bdf1691ee53e001"; - vm.expectRevert(ECDSA.InvalidSignature.selector); - this.recover(TEST_MESSAGE, signature); - } - - function testRecoverWithV1SignatureWithVersion28() public { - bytes memory signature = - hex"331fe75a821c982f9127538858900d87d3ec1f9f737338ad67cad133fa48feff48e6fa0c18abc62e42820f05943e47af3e9fbe306ce74d64094bdf1691ee53e01c"; - assertTrue(this.recover(TEST_MESSAGE, signature) == V1_SIGNER); - } - - function testRecoverWithV1SignatureWithWrongVersionReverts() public { - bytes memory signature = - hex"331fe75a821c982f9127538858900d87d3ec1f9f737338ad67cad133fa48feff48e6fa0c18abc62e42820f05943e47af3e9fbe306ce74d64094bdf1691ee53e002"; - vm.expectRevert(ECDSA.InvalidSignature.selector); - this.recover(TEST_MESSAGE, signature); - } - - function testRecoverWithV1SignatureWithShortEIP2098Format() public { - bytes32 r = 0x331fe75a821c982f9127538858900d87d3ec1f9f737338ad67cad133fa48feff; - bytes32 vs = 0xc8e6fa0c18abc62e42820f05943e47af3e9fbe306ce74d64094bdf1691ee53e0; - assertTrue(this.recover(TEST_MESSAGE, r, vs) == V1_SIGNER); - } - - function testRecoverWithV1SignatureWithShortEIP2098FormatAsCalldata() public { - bytes memory signature = - hex"331fe75a821c982f9127538858900d87d3ec1f9f737338ad67cad133fa48feffc8e6fa0c18abc62e42820f05943e47af3e9fbe306ce74d64094bdf1691ee53e0"; - this.recover(TEST_MESSAGE, signature); - } - - struct _CheckSignatureTestTemps { - bytes argsSignature; - bytes encodedCalldataArgs; - address signer; - bool expected; - bool[2] success; - bytes[2] result; - bytes4 s; - address recovered; - } - - function _checkSignature( - address signer, - bytes32 digest, - uint8 v, - bytes32 r, - bytes32 s, - bool expected - ) internal { - _CheckSignatureTestTemps memory t; - t.signer = signer; - t.expected = expected; - - t.argsSignature = "(bytes32,uint8,bytes32,bytes32)"; - t.encodedCalldataArgs = abi.encode(digest, v, r, s); - _checkSignature(t); - - if (v == 27 || v == 28) { - bytes32 vs = bytes32((v == 28 ? 1 << 255 : 0) | uint256(s)); - t.argsSignature = "(bytes32,bytes32,bytes32)"; - t.encodedCalldataArgs = abi.encode(digest, r, vs); - _checkSignature(t); - } - - if (_random() & 1 == 0) { - t.argsSignature = "(bytes32,bytes)"; - t.encodedCalldataArgs = abi.encode(digest, abi.encodePacked(r, s, v)); - _checkSignature(t); - } - } - - function _checkSignature(_CheckSignatureTestTemps memory t) internal { - t.s = bytes4(keccak256(abi.encodePacked("tryRecover", t.argsSignature))); - (t.success[0], t.result[0]) = - address(this).call(abi.encodePacked(t.s, t.encodedCalldataArgs)); - t.recovered = t.success[0] ? abi.decode(t.result[0], (address)) : address(0); - assertEq(t.recovered == t.signer, t.expected); - - t.s = bytes4(keccak256(abi.encodePacked("tryRecoverBrutalized", t.argsSignature))); - (t.success[1], t.result[1]) = - address(this).call(abi.encodePacked(t.s, t.encodedCalldataArgs)); - t.recovered = t.success[1] ? abi.decode(t.result[1], (address)) : address(0); - assertEq(t.recovered == t.signer, t.expected); - - t.s = bytes4(keccak256(abi.encodePacked("recover", t.argsSignature))); - (t.success[0], t.result[0]) = - address(this).call(abi.encodePacked(t.s, t.encodedCalldataArgs)); - - t.s = bytes4(keccak256(abi.encodePacked("recoverBrutalized", t.argsSignature))); - (t.success[1], t.result[1]) = - address(this).call(abi.encodePacked(t.s, t.encodedCalldataArgs)); - - assertEq(t.success[0], t.success[1]); - assertEq(t.result[0], t.result[1]); - - if (t.success[0]) { - t.recovered = abi.decode(t.result[0], (address)); - assertEq(t.recovered == t.signer, t.expected); - } - } - - function testRecoverAndTryRecover(bytes32 digest) public { - (address signer, uint256 privateKey) = _randomSigner(); - - (uint8 v, bytes32 r, bytes32 s) = vm.sign(privateKey, digest); - if (_random() & 7 == 0) { - _checkSignature(signer, digest, v, r, s, true); - } - - uint8 vc = v ^ uint8(_random() & 0xff); - bytes32 rc = bytes32(uint256(r) ^ _random()); - bytes32 sc = bytes32(uint256(s) ^ _random()); - bool anyCorrupted = vc != v || rc != r || sc != s; - _checkSignature(signer, digest, vc, rc, sc, !anyCorrupted); - } - - function testBytes32ToEthSignedMessageHash() public { - assertEq( - TEST_MESSAGE.toEthSignedMessageHash(), - bytes32(0x7d768af957ef8cbf6219a37e743d5546d911dae3e46449d8a5810522db2ef65e) - ); - } - - function testBytesToEthSignedMessageHashShort() public { - bytes memory message = hex"61626364"; - assertEq( - message.toEthSignedMessageHash(), - bytes32(0xefd0b51a9c4e5f3449f4eeacb195bf48659fbc00d2f4001bf4c088ba0779fb33) - ); - } - - function testBytesToEthSignedMessageHashEmpty() public { - bytes memory message = hex""; - assertEq( - message.toEthSignedMessageHash(), - bytes32(0x5f35dce98ba4fba25530a026ed80b2cecdaa31091ba4958b99b52ea1d068adad) - ); - } - - function testBytesToEthSignedMessageHashLong() public { - bytes memory message = - hex"4142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a3031323334353637383921402324255e262a28292d3d5b5d7b7d"; - assertEq( - message.toEthSignedMessageHash(), - bytes32(0xa46dbedd405cff161b6e80c17c8567597621d9f4c087204201097cb34448e71b) - ); - } - - function testBytesToEthSignedMessageHash() public { - _testBytesToEthSignedMessageHash(999999); - _testBytesToEthSignedMessageHash(135790); - _testBytesToEthSignedMessageHash(99999); - _testBytesToEthSignedMessageHash(88888); - _testBytesToEthSignedMessageHash(3210); - _testBytesToEthSignedMessageHash(111); - _testBytesToEthSignedMessageHash(22); - _testBytesToEthSignedMessageHash(1); - _testBytesToEthSignedMessageHash(0); - } - - function testBytesToEthSignedMessageHashExceedsMaxLengthReverts() public { - vm.expectRevert(); - _testBytesToEthSignedMessageHash(999999 + 1); - } - - function _testBytesToEthSignedMessageHash(uint256 n) internal brutalizeMemory { - bytes memory message; - /// @solidity memory-safe-assembly - assembly { - message := mload(0x40) - mstore(message, n) - mstore(0x40, add(add(message, 0x20), n)) - } - assertEq( - message.toEthSignedMessageHash(), - keccak256( - abi.encodePacked("\x19Ethereum Signed Message:\n", LibString.toString(n), message) - ) - ); - /// @solidity memory-safe-assembly - assembly { - mstore(0x40, message) - } - } - - function tryRecover(bytes32 hash, bytes calldata signature) external returns (address result) { - result = ECDSA.tryRecoverCalldata(hash, signature); - assertEq(ECDSA.tryRecover(hash, signature), result); - } - - function tryRecover(bytes32 hash, uint8 v, bytes32 r, bytes32 s) - external - view - returns (address) - { - return ECDSA.tryRecover(hash, v, r, s); - } - - function tryRecover(bytes32 hash, bytes32 r, bytes32 vs) external view returns (address) { - return ECDSA.tryRecover(hash, r, vs); - } - - function tryRecoverBrutalized(bytes32 hash, bytes calldata signature) - external - brutalizeMemory - returns (address result) - { - result = ECDSA.tryRecoverCalldata(hash, signature); - assertEq(ECDSA.tryRecover(hash, signature), result); - } - - function tryRecoverBrutalized(bytes32 hash, uint8 v, bytes32 r, bytes32 s) - external - view - brutalizeMemory - returns (address) - { - return ECDSA.tryRecover(hash, v, r, s); - } - - function tryRecoverBrutalized(bytes32 hash, bytes32 r, bytes32 vs) - external - view - brutalizeMemory - returns (address) - { - return ECDSA.tryRecover(hash, r, vs); - } - - function recover(bytes32 hash, bytes calldata signature) external returns (address result) { - result = ECDSA.recoverCalldata(hash, signature); - assertEq(ECDSA.recover(hash, signature), result); - } - - function recover(bytes32 hash, bytes32 r, bytes32 vs) external view returns (address) { - return ECDSA.recover(hash, r, vs); - } - - function recover(bytes32 hash, uint8 v, bytes32 r, bytes32 s) external view returns (address) { - return ECDSA.recover(hash, v, r, s); - } - - function recoverBrutalized(bytes32 hash, bytes calldata signature) - external - brutalizeMemory - returns (address result) - { - result = ECDSA.recoverCalldata(hash, signature); - assertEq(ECDSA.recover(hash, signature), result); - } - - function recoverBrutalized(bytes32 hash, bytes32 r, bytes32 vs) - external - view - brutalizeMemory - returns (address) - { - return ECDSA.recover(hash, r, vs); - } - - function recoverBrutalized(bytes32 hash, uint8 v, bytes32 r, bytes32 s) - external - view - brutalizeMemory - returns (address) - { - return ECDSA.recover(hash, v, r, s); - } - - function testEmptyCalldataHelpers() public { - assertFalse(ECDSA.tryRecover(bytes32(0), ECDSA.emptySignature()) == address(1)); - } -} diff --git a/lib/solady/test/EIP712.t.sol b/lib/solady/test/EIP712.t.sol deleted file mode 100644 index d0f69af..0000000 --- a/lib/solady/test/EIP712.t.sol +++ /dev/null @@ -1,173 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {MockEIP712} from "./utils/mocks/MockEIP712.sol"; -import {MockEIP712Dynamic} from "./utils/mocks/MockEIP712Dynamic.sol"; -import {LibClone} from "../src/utils/LibClone.sol"; - -contract EIP712Test is SoladyTest { - MockEIP712 mock; - MockEIP712 mockClone; - MockEIP712Dynamic mockDynamic; - MockEIP712Dynamic mockDynamicClone; - - function setUp() public { - mock = new MockEIP712(); - mockClone = MockEIP712(LibClone.clone(address(mock))); - mockDynamic = new MockEIP712Dynamic("Milady", "1"); - mockDynamicClone = MockEIP712Dynamic(LibClone.clone(address(mockDynamic))); - } - - function testHashTypedData() public { - _testHashTypedDataOnClone(mock); - } - - function testHashTypedDataOnClone() public { - _testHashTypedDataOnClone(mockClone); - } - - function testHashTypedDataOnDynamic() public { - _testHashTypedDataOnClone(MockEIP712(address(mockDynamic))); - } - - function testHashTypedDataOnCloneDynamic() public { - _testHashTypedDataOnClone(MockEIP712(address(mockDynamicClone))); - } - - function testHashTypedDataWithChaindIdChange() public { - _testHashTypedDataOnClone(mock); - vm.chainId(32123); - _testHashTypedDataOnClone(mock); - } - - function testHashTypedDataOnCloneWithChaindIdChange() public { - _testHashTypedDataOnClone(mockClone); - vm.chainId(32123); - _testHashTypedDataOnClone(mockClone); - } - - function testHashTypedDataOnDynamicWithChaindIdChange() public { - _testHashTypedDataOnClone(MockEIP712(address(mockDynamic))); - vm.chainId(32123); - _testHashTypedDataOnClone(MockEIP712(address(mockDynamic))); - } - - function testHashTypedDataOnCloneDynamicWithChaindIdChange() public { - _testHashTypedDataOnClone(MockEIP712(address(mockDynamicClone))); - vm.chainId(32123); - _testHashTypedDataOnClone(MockEIP712(address(mockDynamicClone))); - } - - function _testHashTypedDataOnClone(MockEIP712 mockToTest) internal { - (address signer, uint256 privateKey) = _randomSigner(); - - (address to,) = _randomSigner(); - - string memory message = "Hello Milady!"; - - bytes32 structHash = - keccak256(abi.encode("Message(address to,string message)", to, message)); - bytes32 expectedDigest = - keccak256(abi.encodePacked("\x19\x01", mockToTest.DOMAIN_SEPARATOR(), structHash)); - - assertEq(mockToTest.hashTypedData(structHash), expectedDigest); - - (uint8 v, bytes32 r, bytes32 s) = vm.sign(privateKey, expectedDigest); - - address recoveredAddress = ecrecover(expectedDigest, v, r, s); - - assertEq(recoveredAddress, signer); - } - - function testDomainSeparator() public { - _testDomainSeparator(mock); - } - - function testDomainSeparatorOnClone() public { - _testDomainSeparator(mockClone); - } - - function testDomainSeparatorWithChainIdChange() public { - _testDomainSeparator(mock); - vm.chainId(32123); - _testDomainSeparator(mock); - } - - function testDomainSeparatorOnCloneWithChainIdChange() public { - _testDomainSeparator(mockClone); - vm.chainId(32123); - _testDomainSeparator(mockClone); - } - - function testDomainSeparatorOnDynamicWithChainIdChange() public { - _testDomainSeparator(MockEIP712(address(mockDynamic))); - vm.chainId(32123); - _testDomainSeparator(MockEIP712(address(mockDynamic))); - mockDynamic.setDomainNameAndVersion("Remilio", "2"); - _testDomainSeparator(MockEIP712(address(mockDynamic)), "Remilio", "2"); - } - - function testDomainSeparatorOnCloneDynamicWithChainIdChange() public { - mockDynamicClone.setDomainNameAndVersion("Milady", "1"); - _testDomainSeparator(MockEIP712(address(mockDynamicClone))); - vm.chainId(32123); - _testDomainSeparator(MockEIP712(address(mockDynamicClone))); - mockDynamicClone.setDomainNameAndVersion("Remilio", "2"); - _testDomainSeparator(MockEIP712(address(mockDynamicClone)), "Remilio", "2"); - } - - function _testDomainSeparator(MockEIP712 mockToTest, bytes memory name, bytes memory version) - internal - { - bytes32 expectedDomainSeparator = keccak256( - abi.encode( - keccak256( - "EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)" - ), - keccak256(name), - keccak256(version), - block.chainid, - address(mockToTest) - ) - ); - - assertEq(mockToTest.DOMAIN_SEPARATOR(), expectedDomainSeparator); - } - - function _testDomainSeparator(MockEIP712 mockToTest) internal { - _testDomainSeparator(mockToTest, "Milady", "1"); - } - - function testEIP5267() public { - _testEIP5267(mock); - _testEIP5267(mockClone); - vm.chainId(32123); - _testEIP5267(mock); - _testEIP5267(mockClone); - } - - struct _testEIP5267Variables { - bytes1 fields; - string name; - string version; - uint256 chainId; - address verifyingContract; - bytes32 salt; - uint256[] extensions; - } - - function _testEIP5267(MockEIP712 mockToTest) public { - _testEIP5267Variables memory t; - (t.fields, t.name, t.version, t.chainId, t.verifyingContract, t.salt, t.extensions) = - mockToTest.eip712Domain(); - - assertEq(t.fields, hex"0f"); - assertEq(t.name, "Milady"); - assertEq(t.version, "1"); - assertEq(t.chainId, block.chainid); - assertEq(t.verifyingContract, address(mockToTest)); - assertEq(t.salt, bytes32(0)); - assertEq(t.extensions, new uint256[](0)); - } -} diff --git a/lib/solady/test/ERC1155.t.sol b/lib/solady/test/ERC1155.t.sol deleted file mode 100644 index f4e9434..0000000 --- a/lib/solady/test/ERC1155.t.sol +++ /dev/null @@ -1,1234 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; - -import {ERC1155, MockERC1155} from "./utils/mocks/MockERC1155.sol"; - -abstract contract ERC1155TokenReceiver { - function onERC1155Received(address, address, uint256, uint256, bytes calldata) - external - virtual - returns (bytes4) - { - return ERC1155TokenReceiver.onERC1155Received.selector; - } - - function onERC1155BatchReceived( - address, - address, - uint256[] calldata, - uint256[] calldata, - bytes calldata - ) external virtual returns (bytes4) { - return ERC1155TokenReceiver.onERC1155BatchReceived.selector; - } -} - -contract ERC1155Recipient is ERC1155TokenReceiver { - address public operator; - address public from; - uint256 public id; - uint256 public amount; - bytes public mintData; - - function onERC1155Received( - address _operator, - address _from, - uint256 _id, - uint256 _amount, - bytes calldata _data - ) public override returns (bytes4) { - operator = _operator; - from = _from; - id = _id; - amount = _amount; - mintData = _data; - - return ERC1155TokenReceiver.onERC1155Received.selector; - } - - address public batchOperator; - address public batchFrom; - uint256[] internal _batchIds; - uint256[] internal _batchAmounts; - bytes public batchData; - - function batchIds() external view returns (uint256[] memory) { - return _batchIds; - } - - function batchAmounts() external view returns (uint256[] memory) { - return _batchAmounts; - } - - function onERC1155BatchReceived( - address _operator, - address _from, - uint256[] calldata _ids, - uint256[] calldata _amounts, - bytes calldata _data - ) external override returns (bytes4) { - batchOperator = _operator; - batchFrom = _from; - _batchIds = _ids; - _batchAmounts = _amounts; - batchData = _data; - - return ERC1155TokenReceiver.onERC1155BatchReceived.selector; - } -} - -contract RevertingERC1155Recipient is ERC1155TokenReceiver { - function onERC1155Received(address, address, uint256, uint256, bytes calldata) - public - pure - override - returns (bytes4) - { - revert(string(abi.encodePacked(ERC1155TokenReceiver.onERC1155Received.selector))); - } - - function onERC1155BatchReceived( - address, - address, - uint256[] calldata, - uint256[] calldata, - bytes calldata - ) external pure override returns (bytes4) { - revert(string(abi.encodePacked(ERC1155TokenReceiver.onERC1155BatchReceived.selector))); - } -} - -contract WrongReturnDataERC1155Recipient is ERC1155TokenReceiver { - function onERC1155Received(address, address, uint256, uint256, bytes calldata) - public - pure - override - returns (bytes4) - { - return 0xCAFEBEEF; - } - - function onERC1155BatchReceived( - address, - address, - uint256[] calldata, - uint256[] calldata, - bytes calldata - ) external pure override returns (bytes4) { - return 0xCAFEBEEF; - } -} - -contract NonERC1155Recipient {} - -contract MockERC1155WithHooks is MockERC1155 { - uint256 public beforeCounter; - uint256 public afterCounter; - - function _useBeforeTokenTransfer() internal view virtual override returns (bool) { - return true; - } - - function _useAfterTokenTransfer() internal view virtual override returns (bool) { - return true; - } - - function _beforeTokenTransfer( - address, - address, - uint256[] memory, - uint256[] memory, - bytes memory - ) internal virtual override { - beforeCounter++; - } - - function _afterTokenTransfer(address, address, uint256[] memory, uint256[] memory, bytes memory) - internal - virtual - override - { - afterCounter++; - } -} - -contract ERC1155HooksTest is SoladyTest, ERC1155TokenReceiver { - uint256 public expectedBeforeCounter; - uint256 public expectedAfterCounter; - - function _checkCounters() internal view { - require( - expectedBeforeCounter == MockERC1155WithHooks(msg.sender).beforeCounter(), - "Before counter mismatch." - ); - require( - expectedAfterCounter == MockERC1155WithHooks(msg.sender).afterCounter(), - "After counter mismatch." - ); - } - - function onERC1155Received(address, address, uint256, uint256, bytes calldata) - external - virtual - override - returns (bytes4) - { - _checkCounters(); - return ERC1155TokenReceiver.onERC1155Received.selector; - } - - function onERC1155BatchReceived( - address, - address, - uint256[] calldata, - uint256[] calldata, - bytes calldata - ) external virtual override returns (bytes4) { - _checkCounters(); - return ERC1155TokenReceiver.onERC1155BatchReceived.selector; - } - - function _testHooks(MockERC1155WithHooks token) internal { - address from = _randomNonZeroAddress(); - expectedBeforeCounter++; - expectedAfterCounter++; - token.mint(address(this), 1, 1000, ""); - - expectedBeforeCounter++; - expectedAfterCounter++; - token.safeTransferFrom(address(this), from, 1, 1000, ""); - - vm.prank(from); - expectedBeforeCounter++; - expectedAfterCounter++; - token.safeTransferFrom(from, address(this), 1, 1, ""); - - vm.prank(from); - expectedBeforeCounter++; - expectedAfterCounter++; - token.directSafeTransferFrom(from, address(this), 1, 1, ""); - - uint256[] memory ids = new uint256[](1); - uint256[] memory amounts = new uint256[](1); - ids[0] = 1; - amounts[0] = 1; - - vm.prank(from); - expectedBeforeCounter++; - expectedAfterCounter++; - token.safeBatchTransferFrom(from, address(this), ids, amounts, ""); - - vm.prank(from); - expectedBeforeCounter++; - expectedAfterCounter++; - token.directSafeBatchTransferFrom(from, address(this), ids, amounts, ""); - } - - function testERC1155Hooks() public { - MockERC1155WithHooks token = new MockERC1155WithHooks(); - - for (uint256 i; i < 32; ++i) { - _testHooks(token); - } - } -} - -contract ERC1155Test is SoladyTest, ERC1155TokenReceiver { - MockERC1155 token; - - event TransferSingle( - address indexed operator, - address indexed from, - address indexed to, - uint256 id, - uint256 amount - ); - - event TransferBatch( - address indexed operator, - address indexed from, - address indexed to, - uint256[] ids, - uint256[] amounts - ); - - event ApprovalForAll(address indexed owner, address indexed operator, bool isApproved); - - mapping(address => mapping(uint256 => uint256)) public userMintAmounts; - mapping(address => mapping(uint256 => uint256)) public userTransferOrBurnAmounts; - - struct _TestTemps { - address from; - address to; - uint256 n; - uint256[] ids; - uint256[] mintAmounts; - uint256[] transferAmounts; - uint256[] burnAmounts; - uint256 id; - uint256 mintAmount; - uint256 transferAmount; - uint256 burnAmount; - bytes mintData; - bytes burnData; - bytes transferData; - } - - function _randomBytes() internal returns (bytes memory b) { - uint256 r = _random(); - /// @solidity memory-safe-assembly - assembly { - b := mload(0x40) - mstore(b, mod(r, 65)) - let t := add(b, 0x20) - mstore(t, r) - mstore(add(b, 0x40), keccak256(t, 0x20)) - mstore(0x40, add(b, 0x60)) - } - } - - function _randomArray(uint256 n) internal returns (uint256[] memory a) { - /// @solidity memory-safe-assembly - assembly { - a := mload(0x40) - mstore(a, n) - mstore(0x40, add(add(a, 0x20), shl(5, n))) - } - unchecked { - for (uint256 i; i != n; ++i) { - a[i] = _random(); - } - } - } - - function _testTemps() internal returns (_TestTemps memory t) { - unchecked { - t.from = _randomNonZeroAddress(); - do { - t.to = _randomNonZeroAddress(); - } while (t.from == t.to); - uint256 n = _random() % 4; - t.n = n; - t.ids = _randomArray(n); - t.mintAmounts = _randomArray(n); - t.transferAmounts = _randomArray(n); - t.burnAmounts = _randomArray(n); - t.mintData = _randomBytes(); - t.burnData = _randomBytes(); - t.transferData = _randomBytes(); - t.id = _random(); - t.transferAmount = _random(); - t.burnAmount = _random(); - t.mintAmount = _random(); - } - } - - function _safeTransferFrom( - address from, - address to, - uint256 id, - uint256 amount, - bytes memory data - ) internal { - if (_random() % 2 == 0) { - token.safeTransferFrom(from, to, id, amount, data); - } else { - token.directSafeTransferFrom(from, to, id, amount, data); - } - } - - function _safeBatchTransferFrom( - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) internal { - if (_random() % 2 == 0) { - token.safeBatchTransferFrom(from, to, ids, amounts, data); - } else { - token.directSafeBatchTransferFrom(from, to, ids, amounts, data); - } - } - - function _setApprovalForAll(address operator, bool approved) internal { - if (_random() % 2 == 0) { - token.setApprovalForAll(operator, approved); - } else { - token.directSetApprovalForAll(operator, approved); - } - } - - function _expectMintEvent(address to, uint256 id, uint256 amount) internal { - _expectMintEvent(address(this), to, id, amount); - } - - function _expectMintEvent(address operator, address to, uint256 id, uint256 amount) internal { - _expectTransferEvent(operator, address(0), to, id, amount); - } - - function _expectBurnEvent(address from, uint256 id, uint256 amount) internal { - _expectBurnEvent(address(this), from, id, amount); - } - - function _expectBurnEvent(address operator, address from, uint256 id, uint256 amount) - internal - { - _expectTransferEvent(operator, from, address(0), id, amount); - } - - function _expectTransferEvent(address from, address to, uint256 id, uint256 amount) internal { - _expectTransferEvent(address(this), from, to, id, amount); - } - - function _expectTransferEvent( - address operator, - address from, - address to, - uint256 id, - uint256 amount - ) internal { - vm.expectEmit(true, true, true, true); - emit TransferSingle(operator, from, to, id, amount); - } - - function _expectMintEvent(address to, uint256[] memory ids, uint256[] memory amounts) - internal - { - _expectMintEvent(address(this), to, ids, amounts); - } - - function _expectMintEvent( - address operator, - address to, - uint256[] memory ids, - uint256[] memory amounts - ) internal { - _expectTransferEvent(operator, address(0), to, ids, amounts); - } - - function _expectBurnEvent(address from, uint256[] memory ids, uint256[] memory amounts) - internal - { - _expectBurnEvent(address(this), from, ids, amounts); - } - - function _expectBurnEvent( - address operator, - address from, - uint256[] memory ids, - uint256[] memory amounts - ) internal { - _expectTransferEvent(operator, from, address(0), ids, amounts); - } - - function _expectTransferEvent( - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts - ) internal { - _expectTransferEvent(address(this), from, to, ids, amounts); - } - - function _expectTransferEvent( - address operator, - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts - ) internal { - vm.expectEmit(true, true, true, true); - emit TransferBatch(operator, from, to, ids, amounts); - } - - function _expectApprovalForAllEvent(address operator, bool isApproved) internal { - _expectApprovalForAllEvent(address(this), operator, isApproved); - } - - function _expectApprovalForAllEvent(address owner, address operator, bool isApproved) - internal - { - vm.expectEmit(true, true, true, true); - emit ApprovalForAll(owner, operator, isApproved); - } - - function setUp() public { - token = new MockERC1155(); - } - - function testDirectSetApprovalForAll(address by, address operator, bool approved) public { - _expectApprovalForAllEvent(by, operator, approved); - vm.prank(by); - token.directSetApprovalForAll(operator, approved); - } - - function testAuthorizedEquivalence(address by, address from, bool isApprovedAccount) public { - bool a = true; - bool b = true; - /// @solidity memory-safe-assembly - assembly { - if by { if iszero(eq(by, from)) { a := isApprovedAccount } } - if iszero(or(iszero(by), eq(by, from))) { b := isApprovedAccount } - } - assertEq(a, b); - } - - function testMintToEOA(uint256) public { - _TestTemps memory t = _testTemps(); - - _expectMintEvent(t.to, t.id, t.mintAmount); - token.mint(t.to, t.id, t.mintAmount, t.mintData); - - assertEq(token.balanceOf(t.to, t.id), t.mintAmount); - } - - function testMintToERC1155Recipient(uint256) public { - _TestTemps memory t = _testTemps(); - - ERC1155Recipient to = new ERC1155Recipient(); - - _expectMintEvent(address(to), t.id, t.mintAmount); - token.mint(address(to), t.id, t.mintAmount, t.mintData); - - assertEq(token.balanceOf(address(to), t.id), t.mintAmount); - - assertEq(to.operator(), address(this)); - assertEq(to.from(), address(0)); - assertEq(to.id(), t.id); - assertEq(to.mintData(), t.mintData); - } - - function testBatchMintToEOA(uint256) public { - _TestTemps memory t = _testTemps(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[t.to][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - - t.mintAmounts[i] = mintAmount; - - userMintAmounts[t.to][id] += mintAmount; - } - - _expectMintEvent(t.to, t.ids, t.mintAmounts); - token.batchMint(t.to, t.ids, t.mintAmounts, t.mintData); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - assertEq(token.balanceOf(t.to, id), userMintAmounts[t.to][id]); - } - } - - function testBatchMintToERC1155Recipient(uint256) public { - _TestTemps memory t = _testTemps(); - - ERC1155Recipient to = new ERC1155Recipient(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[address(to)][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - - t.mintAmounts[i] = mintAmount; - - userMintAmounts[address(to)][id] += mintAmount; - } - - _expectMintEvent(address(to), t.ids, t.mintAmounts); - token.batchMint(address(to), t.ids, t.mintAmounts, t.mintData); - - assertEq(to.batchOperator(), address(this)); - assertEq(to.batchFrom(), address(0)); - assertEq(to.batchIds(), t.ids); - assertEq(to.batchAmounts(), t.mintAmounts); - assertEq(to.batchData(), t.mintData); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - assertEq(token.balanceOf(address(to), id), userMintAmounts[address(to)][id]); - } - } - - function testBurn(uint256) public { - _TestTemps memory t = _testTemps(); - - t.burnAmount = _bound(t.burnAmount, 0, t.mintAmount); - - _expectMintEvent(t.to, t.id, t.mintAmount); - token.mint(t.to, t.id, t.mintAmount, t.mintData); - - if (_random() % 2 == 0) { - _expectBurnEvent(t.to, t.id, t.burnAmount); - token.uncheckedBurn(t.to, t.id, t.burnAmount); - } else if (_random() % 8 == 0) { - vm.expectRevert(ERC1155.NotOwnerNorApproved.selector); - token.burn(t.to, t.id, t.burnAmount); - return; - } else { - vm.prank(t.to); - _setApprovalForAll(address(this), true); - - _expectBurnEvent(t.to, t.id, t.burnAmount); - token.burn(t.to, t.id, t.burnAmount); - } - - assertEq(token.balanceOf(t.to, t.id), t.mintAmount - t.burnAmount); - } - - function testBatchBurn(uint256) public { - _TestTemps memory t = _testTemps(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[t.to][id]; - - t.mintAmounts[i] = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - t.burnAmounts[i] = _bound(t.burnAmounts[i], 0, t.mintAmounts[i]); - - userMintAmounts[t.to][id] += t.mintAmounts[i]; - userTransferOrBurnAmounts[t.to][id] += t.burnAmounts[i]; - } - - _expectMintEvent(t.to, t.ids, t.mintAmounts); - token.batchMint(t.to, t.ids, t.mintAmounts, t.mintData); - - if (_random() % 2 == 0) { - _expectBurnEvent(t.to, t.ids, t.burnAmounts); - token.uncheckedBatchBurn(t.to, t.ids, t.burnAmounts); - } else if (_random() % 8 == 0) { - vm.expectRevert(ERC1155.NotOwnerNorApproved.selector); - token.batchBurn(t.to, t.ids, t.burnAmounts); - return; - } else { - vm.prank(t.to); - _setApprovalForAll(address(this), true); - - _expectBurnEvent(t.to, t.ids, t.burnAmounts); - token.batchBurn(t.to, t.ids, t.burnAmounts); - } - - for (uint256 i = 0; i < t.ids.length; i++) { - uint256 id = t.ids[i]; - - assertEq( - token.balanceOf(t.to, id), - userMintAmounts[t.to][id] - userTransferOrBurnAmounts[t.to][id] - ); - } - } - - function testApproveAll(address to, bool approved) public { - _expectApprovalForAllEvent(to, approved); - _setApprovalForAll(to, approved); - assertEq(token.isApprovedForAll(address(this), to), approved); - } - - function testSafeTransferFromToEOA(uint256) public { - _TestTemps memory t = _testTemps(); - - t.transferAmount = _bound(t.transferAmount, 0, t.mintAmount); - - _expectMintEvent(t.from, t.id, t.mintAmount); - token.mint(t.from, t.id, t.mintAmount, t.mintData); - - if (_random() % 2 == 0) { - _expectTransferEvent(t.from, t.to, t.id, t.transferAmount); - token.uncheckedSafeTransferFrom(t.from, t.to, t.id, t.transferAmount, t.transferData); - } else if (_random() % 8 == 0) { - vm.expectRevert(ERC1155.NotOwnerNorApproved.selector); - _safeTransferFrom(t.from, t.to, t.id, t.transferAmount, t.transferData); - return; - } else { - vm.prank(t.from); - _setApprovalForAll(address(this), true); - - _expectTransferEvent(t.from, t.to, t.id, t.transferAmount); - _safeTransferFrom(t.from, t.to, t.id, t.transferAmount, t.transferData); - } - - if (t.to == t.from) { - assertEq(token.balanceOf(t.to, t.id), t.mintAmount); - } else { - assertEq(token.balanceOf(t.to, t.id), t.transferAmount); - assertEq(token.balanceOf(t.from, t.id), t.mintAmount - t.transferAmount); - } - } - - function testSafeTransferFromToERC1155Recipient(uint256) public { - _TestTemps memory t = _testTemps(); - ERC1155Recipient to = new ERC1155Recipient(); - - t.transferAmount = _bound(t.transferAmount, 0, t.mintAmount); - - _expectMintEvent(t.from, t.id, t.mintAmount); - token.mint(t.from, t.id, t.mintAmount, t.mintData); - - vm.prank(t.from); - _setApprovalForAll(address(this), true); - - _expectTransferEvent(t.from, address(to), t.id, t.transferAmount); - _safeTransferFrom(t.from, address(to), t.id, t.transferAmount, t.transferData); - - assertEq(to.operator(), address(this)); - assertEq(to.from(), t.from); - assertEq(to.id(), t.id); - assertEq(to.mintData(), t.transferData); - - assertEq(token.balanceOf(address(to), t.id), t.transferAmount); - assertEq(token.balanceOf(t.from, t.id), t.mintAmount - t.transferAmount); - } - - function testSafeTransferFromSelf(uint256) public { - _TestTemps memory t = _testTemps(); - - t.transferAmount = _bound(t.transferAmount, 0, t.mintAmount); - - _expectMintEvent(address(this), t.id, t.mintAmount); - token.mint(address(this), t.id, t.mintAmount, t.mintData); - - _expectTransferEvent(address(this), t.to, t.id, t.transferAmount); - _safeTransferFrom(address(this), t.to, t.id, t.transferAmount, t.transferData); - - assertEq(token.balanceOf(t.to, t.id), t.transferAmount); - assertEq(token.balanceOf(address(this), t.id), t.mintAmount - t.transferAmount); - } - - function testSafeBatchTransfer() public { - for (uint256 i; i != 8; ++i) { - testSafeTransferFromToEOA(_random()); - testSafeBatchTransferFromToERC1155Recipient(_random()); - } - } - - function testSafeBatchTransferFromToEOA(uint256) public { - _TestTemps memory t = _testTemps(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[t.from][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = _bound(t.transferAmounts[i], 0, mintAmount); - - t.mintAmounts[i] = mintAmount; - t.transferAmounts[i] = transferAmount; - - userMintAmounts[t.from][id] += mintAmount; - userTransferOrBurnAmounts[t.from][id] += transferAmount; - } - _expectMintEvent(t.from, t.ids, t.mintAmounts); - token.batchMint(t.from, t.ids, t.mintAmounts, t.mintData); - - if (_random() % 2 == 0) { - _expectTransferEvent(t.from, t.to, t.ids, t.transferAmounts); - token.uncheckedSafeBatchTransferFrom( - t.from, t.to, t.ids, t.transferAmounts, t.transferData - ); - } else if (_random() % 8 == 0) { - vm.expectRevert(ERC1155.NotOwnerNorApproved.selector); - _safeBatchTransferFrom(t.from, t.to, t.ids, t.transferAmounts, t.transferData); - return; - } else { - vm.prank(t.from); - _setApprovalForAll(address(this), true); - - _expectTransferEvent(t.from, t.to, t.ids, t.transferAmounts); - _safeBatchTransferFrom(t.from, t.to, t.ids, t.transferAmounts, t.transferData); - } - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - assertEq(token.balanceOf(t.to, id), userTransferOrBurnAmounts[t.from][id]); - assertEq( - token.balanceOf(t.from, id), - userMintAmounts[t.from][id] - userTransferOrBurnAmounts[t.from][id] - ); - } - } - - function testSafeBatchTransferFromToERC1155Recipient(uint256) public { - _TestTemps memory t = _testTemps(); - - ERC1155Recipient to = new ERC1155Recipient(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[t.from][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = _bound(t.transferAmounts[i], 0, mintAmount); - - t.mintAmounts[i] = mintAmount; - t.transferAmounts[i] = transferAmount; - - userMintAmounts[t.from][id] += mintAmount; - userTransferOrBurnAmounts[t.from][id] += transferAmount; - } - - _expectMintEvent(t.from, t.ids, t.mintAmounts); - token.batchMint(t.from, t.ids, t.mintAmounts, t.mintData); - - vm.prank(t.from); - _setApprovalForAll(address(this), true); - - _expectTransferEvent(t.from, address(to), t.ids, t.transferAmounts); - _safeBatchTransferFrom(t.from, address(to), t.ids, t.transferAmounts, t.transferData); - - assertEq(to.batchOperator(), address(this)); - assertEq(to.batchFrom(), t.from); - assertEq(to.batchIds(), t.ids); - assertEq(to.batchAmounts(), t.transferAmounts); - assertEq(to.batchData(), t.transferData); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - uint256 transferAmount = userTransferOrBurnAmounts[t.from][id]; - - assertEq(token.balanceOf(address(to), id), transferAmount); - assertEq(token.balanceOf(t.from, id), userMintAmounts[t.from][id] - transferAmount); - } - } - - function testBatchBalanceOf(uint256) public { - _TestTemps memory t = _testTemps(); - - address[] memory tos = new address[](t.n); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - address to = _randomNonZeroAddress(); - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[to][id]; - - tos[i] = to; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - - token.mint(to, id, mintAmount, t.mintData); - - userMintAmounts[to][id] += mintAmount; - } - - uint256[] memory balances = token.balanceOfBatch(tos, t.ids); - - for (uint256 i = 0; i != t.n; i++) { - assertEq(balances[i], token.balanceOf(tos[i], t.ids[i])); - } - } - - function testMintToZeroReverts(uint256) public { - vm.expectRevert(ERC1155.TransferToZeroAddress.selector); - token.mint(address(0), _random(), _random(), _randomBytes()); - } - - function testMintToNonERC155RecipientReverts(uint256) public { - address to = address(new NonERC1155Recipient()); - vm.expectRevert(ERC1155.TransferToNonERC1155ReceiverImplementer.selector); - token.mint(to, _random(), _random(), _randomBytes()); - } - - function testMintToRevertingERC155RecipientReverts(uint256) public { - address to = address(new RevertingERC1155Recipient()); - vm.expectRevert(abi.encodePacked(ERC1155TokenReceiver.onERC1155Received.selector)); - token.mint(to, _random(), _random(), _randomBytes()); - } - - function testMintToWrongReturnDataERC155RecipientReverts(uint256) public { - address to = address(new WrongReturnDataERC1155Recipient()); - vm.expectRevert(ERC1155.TransferToNonERC1155ReceiverImplementer.selector); - token.mint(to, _random(), _random(), _randomBytes()); - } - - function testBurnInsufficientBalanceReverts(uint256) public { - _TestTemps memory t = _testTemps(); - while (t.mintAmount == type(uint256).max) t.mintAmount = _random(); - t.burnAmount = _bound(t.burnAmount, t.mintAmount + 1, type(uint256).max); - - token.mint(t.to, t.id, t.mintAmount, t.mintData); - - vm.prank(t.to); - _setApprovalForAll(address(this), true); - - vm.expectRevert(ERC1155.InsufficientBalance.selector); - token.burn(t.to, t.id, t.burnAmount); - } - - function testSafeTransferFromInsufficientBalanceReverts(uint256) public { - _TestTemps memory t = _testTemps(); - while (t.mintAmount == type(uint256).max) t.mintAmount = _random(); - - t.transferAmount = _bound(t.transferAmount, t.mintAmount + 1, type(uint256).max); - - token.mint(t.from, t.id, t.mintAmount, t.mintData); - - vm.prank(t.from); - _setApprovalForAll(address(this), true); - - vm.expectRevert(ERC1155.InsufficientBalance.selector); - _safeTransferFrom(t.from, t.to, t.id, t.transferAmount, t.transferData); - } - - function testSafeTransferFromSelfInsufficientBalanceReverts(uint256) public { - _TestTemps memory t = _testTemps(); - while (t.mintAmount == type(uint256).max) t.mintAmount = _random(); - - t.transferAmount = _bound(t.transferAmount, t.mintAmount + 1, type(uint256).max); - - token.mint(address(this), t.id, t.mintAmount, t.mintData); - - vm.expectRevert(ERC1155.InsufficientBalance.selector); - _safeTransferFrom(address(this), t.to, t.id, t.transferAmount, t.transferData); - } - - function testSafeTransferFromToZeroReverts(uint256) public { - _TestTemps memory t = _testTemps(); - t.transferAmount = _bound(t.transferAmount, 0, t.mintAmount); - - token.mint(address(this), t.id, t.mintAmount, t.mintData); - - vm.expectRevert(ERC1155.TransferToZeroAddress.selector); - _safeTransferFrom(address(this), address(0), t.id, t.transferAmount, t.transferData); - } - - function testSafeTransferFromToNonERC155RecipientReverts(uint256) public { - _TestTemps memory t = _testTemps(); - t.transferAmount = _bound(t.transferAmount, 0, t.mintAmount); - - token.mint(address(this), t.id, t.mintAmount, t.mintData); - t.to = address(new NonERC1155Recipient()); - - vm.expectRevert(ERC1155.TransferToNonERC1155ReceiverImplementer.selector); - _safeTransferFrom(address(this), t.to, t.id, t.transferAmount, t.transferData); - } - - function testSafeTransferFromToRevertingERC1155RecipientReverts(uint256) public { - _TestTemps memory t = _testTemps(); - t.transferAmount = _bound(t.transferAmount, 0, t.mintAmount); - - token.mint(address(this), t.id, t.mintAmount, t.mintData); - t.to = address(new RevertingERC1155Recipient()); - - vm.expectRevert(abi.encodePacked(ERC1155TokenReceiver.onERC1155Received.selector)); - _safeTransferFrom(address(this), t.to, t.id, t.transferAmount, t.transferData); - } - - function testSafeTransferFromToWrongReturnDataERC1155RecipientReverts(uint256) public { - _TestTemps memory t = _testTemps(); - t.transferAmount = _bound(t.transferAmount, 0, t.mintAmount); - - token.mint(address(this), t.id, t.mintAmount, t.mintData); - t.to = address(new WrongReturnDataERC1155Recipient()); - - vm.expectRevert(ERC1155.TransferToNonERC1155ReceiverImplementer.selector); - _safeTransferFrom(address(this), t.to, t.id, t.transferAmount, t.transferData); - } - - function testSafeBatchTransferInsufficientBalanceReverts(uint256) public { - _TestTemps memory t = _testTemps(); - - while (t.n == 0) t = _testTemps(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[t.from][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - if (mintAmount == type(uint256).max) return; - uint256 transferAmount = _bound(t.transferAmounts[i], mintAmount + 1, type(uint256).max); - - t.mintAmounts[i] = mintAmount; - t.transferAmounts[i] = transferAmount; - - userMintAmounts[t.from][id] += mintAmount; - } - - token.batchMint(t.from, t.ids, t.mintAmounts, t.mintData); - - vm.prank(t.from); - _setApprovalForAll(address(this), true); - - vm.expectRevert(ERC1155.InsufficientBalance.selector); - _safeBatchTransferFrom(t.from, t.to, t.ids, t.transferAmounts, t.transferData); - } - - function testSafeBatchTransferFromToZeroReverts(uint256) public { - _TestTemps memory t = _testTemps(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[t.from][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = _bound(t.transferAmounts[i], 0, mintAmount); - - t.mintAmounts[i] = mintAmount; - t.transferAmounts[i] = transferAmount; - - userMintAmounts[t.from][id] += mintAmount; - } - - token.batchMint(t.from, t.ids, t.mintAmounts, t.mintData); - - vm.prank(t.from); - _setApprovalForAll(address(this), true); - - vm.expectRevert(ERC1155.TransferToZeroAddress.selector); - _safeBatchTransferFrom(t.from, address(0), t.ids, t.transferAmounts, t.transferData); - } - - function testSafeBatchTransferFromToNonERC1155RecipientReverts(uint256) public { - _TestTemps memory t = _testTemps(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[t.from][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = _bound(t.transferAmounts[i], 0, mintAmount); - - t.mintAmounts[i] = mintAmount; - t.transferAmounts[i] = transferAmount; - - userMintAmounts[t.from][id] += mintAmount; - } - - token.batchMint(t.from, t.ids, t.mintAmounts, t.mintData); - - vm.prank(t.from); - _setApprovalForAll(address(this), true); - - t.to = address(new NonERC1155Recipient()); - - vm.expectRevert(ERC1155.TransferToNonERC1155ReceiverImplementer.selector); - _safeBatchTransferFrom(t.from, t.to, t.ids, t.transferAmounts, t.transferData); - } - - function testSafeBatchTransferFromToRevertingERC1155RecipientReverts(uint256) public { - _TestTemps memory t = _testTemps(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[t.from][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = _bound(t.transferAmounts[i], 0, mintAmount); - - t.mintAmounts[i] = mintAmount; - t.transferAmounts[i] = transferAmount; - - userMintAmounts[t.from][id] += mintAmount; - } - - token.batchMint(t.from, t.ids, t.mintAmounts, t.mintData); - - vm.prank(t.from); - _setApprovalForAll(address(this), true); - - t.to = address(new RevertingERC1155Recipient()); - vm.expectRevert(abi.encodePacked(ERC1155TokenReceiver.onERC1155BatchReceived.selector)); - _safeBatchTransferFrom(t.from, t.to, t.ids, t.transferAmounts, t.transferData); - } - - function testSafeBatchTransferFromToWrongReturnDataERC1155RecipientReverts(uint256) public { - _TestTemps memory t = _testTemps(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[t.from][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = _bound(t.transferAmounts[i], 0, mintAmount); - - t.mintAmounts[i] = mintAmount; - t.transferAmounts[i] = transferAmount; - - userMintAmounts[t.from][id] += mintAmount; - } - - token.batchMint(t.from, t.ids, t.mintAmounts, t.mintData); - - vm.prank(t.from); - _setApprovalForAll(address(this), true); - - t.to = address(new WrongReturnDataERC1155Recipient()); - vm.expectRevert(ERC1155.TransferToNonERC1155ReceiverImplementer.selector); - _safeBatchTransferFrom(t.from, t.to, t.ids, t.transferAmounts, t.transferData); - } - - function testSafeBatchTransferFromWithArrayLengthMismatchReverts(uint256) public { - uint256[] memory ids = new uint256[](_random() % 4); - uint256[] memory mintAmounts = new uint256[](_random() % 4); - - if (ids.length == mintAmounts.length) return; - - address from = address(0xABCD); - - vm.expectRevert(ERC1155.ArrayLengthsMismatch.selector); - token.batchMint(from, ids, mintAmounts, _randomBytes()); - - uint256[] memory transferAmounts = new uint256[](_random() % 4); - if (ids.length == transferAmounts.length) return; - - vm.prank(from); - _setApprovalForAll(address(this), true); - - address to = _randomNonZeroAddress(); - - vm.expectRevert(ERC1155.ArrayLengthsMismatch.selector); - _safeBatchTransferFrom(from, to, ids, transferAmounts, _randomBytes()); - } - - function testBatchMintToZeroReverts(uint256) public { - _TestTemps memory t = _testTemps(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[address(0)][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - - t.mintAmounts[i] = mintAmount; - - userMintAmounts[address(0)][id] += mintAmount; - } - - vm.expectRevert(ERC1155.TransferToZeroAddress.selector); - token.batchMint(address(0), t.ids, t.mintAmounts, t.mintData); - } - - function testBatchMintToNonERC1155RecipientReverts(uint256) public { - _TestTemps memory t = _testTemps(); - - NonERC1155Recipient to = new NonERC1155Recipient(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[address(to)][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - - t.mintAmounts[i] = mintAmount; - - userMintAmounts[address(to)][id] += mintAmount; - } - - vm.expectRevert(ERC1155.TransferToNonERC1155ReceiverImplementer.selector); - token.batchMint(address(to), t.ids, t.mintAmounts, t.mintData); - } - - function testBatchMintToRevertingERC1155RecipientReverts(uint256) public { - _TestTemps memory t = _testTemps(); - - RevertingERC1155Recipient to = new RevertingERC1155Recipient(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[address(to)][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - - t.mintAmounts[i] = mintAmount; - - userMintAmounts[address(to)][id] += mintAmount; - } - vm.expectRevert(abi.encodePacked(ERC1155TokenReceiver.onERC1155BatchReceived.selector)); - token.batchMint(address(to), t.ids, t.mintAmounts, t.mintData); - } - - function testBatchMintToWrongReturnDataERC1155RecipientReverts(uint256) public { - _TestTemps memory t = _testTemps(); - - WrongReturnDataERC1155Recipient to = new WrongReturnDataERC1155Recipient(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[address(to)][id]; - - uint256 mintAmount = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - - t.mintAmounts[i] = mintAmount; - - userMintAmounts[address(to)][id] += mintAmount; - } - vm.expectRevert(ERC1155.TransferToNonERC1155ReceiverImplementer.selector); - token.batchMint(address(to), t.ids, t.mintAmounts, t.mintData); - } - - function testBatchMintWithArrayMismatchReverts(uint256) public { - uint256[] memory ids = new uint256[](_random() % 4); - uint256[] memory amounts = new uint256[](_random() % 4); - - if (ids.length == amounts.length) return; - - address to = _randomNonZeroAddress(); - - vm.expectRevert(ERC1155.ArrayLengthsMismatch.selector); - token.batchMint(to, ids, amounts, _randomBytes()); - } - - function testBatchBurnInsufficientBalanceReverts(uint256) public { - _TestTemps memory t = _testTemps(); - - while (t.n == 0) t = _testTemps(); - - for (uint256 i = 0; i != t.n; i++) { - uint256 id = t.ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[t.to][id]; - - t.mintAmounts[i] = _bound(t.mintAmounts[i], 0, remainingMintAmountForId); - if (t.mintAmounts[i] == type(uint256).max) return; - t.burnAmounts[i] = _bound(t.burnAmounts[i], t.mintAmounts[i] + 1, type(uint256).max); - - userMintAmounts[t.to][id] += t.mintAmounts[i]; - } - - token.batchMint(t.to, t.ids, t.mintAmounts, t.mintData); - - vm.prank(t.to); - _setApprovalForAll(address(this), true); - - vm.expectRevert(ERC1155.InsufficientBalance.selector); - token.batchBurn(t.to, t.ids, t.burnAmounts); - } - - function testBatchBurnWithArrayLengthMismatchReverts(uint256) public { - _TestTemps memory t = _testTemps(); - - if (t.ids.length == t.burnAmounts.length) t.burnAmounts = _randomArray(t.n + 1); - - vm.expectRevert(ERC1155.ArrayLengthsMismatch.selector); - token.batchBurn(t.to, t.ids, t.burnAmounts); - } - - function testBalanceOfBatchWithArrayMismatchReverts(uint256) public { - address[] memory tos = new address[](_random() % 4); - uint256[] memory ids = new uint256[](_random() % 4); - if (tos.length == ids.length) return; - - vm.expectRevert(ERC1155.ArrayLengthsMismatch.selector); - token.balanceOfBatch(tos, ids); - } -} diff --git a/lib/solady/test/ERC1967Factory.t.sol b/lib/solady/test/ERC1967Factory.t.sol deleted file mode 100644 index a19ea56..0000000 --- a/lib/solady/test/ERC1967Factory.t.sol +++ /dev/null @@ -1,302 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {MockImplementation} from "./utils/mocks/MockImplementation.sol"; -import {ERC1967Factory} from "../src/utils/ERC1967Factory.sol"; -import {ERC1967FactoryConstants} from "../src/utils/ERC1967FactoryConstants.sol"; - -contract ERC1967FactoryTest is SoladyTest { - event AdminChanged(address indexed proxy, address indexed admin); - - event Upgraded(address indexed proxy, address indexed implementation); - - event Deployed(address indexed proxy, address indexed implementation, address indexed admin); - - ERC1967Factory factory; - address implementation0; - address implementation1; - - struct _TestTemps { - uint256 key; - uint256 value; - uint256 msgValue; - bytes32 salt; - address predictedProxy; - address proxy; - } - - function _testTemps() internal returns (_TestTemps memory t) { - t.key = _random(); - t.value = _random(); - t.msgValue = _bound(_random(), 0, uint256(type(uint96).max)); - t.salt = bytes32(_random() & uint256(type(uint96).max)); - } - - function setUp() public { - factory = new ERC1967Factory(); - implementation0 = address(new MockImplementation()); - implementation1 = address(new MockImplementation()); - } - - modifier withFactories() { - _; - { - address minedFactoryAddress = 0x0000000000001122334455667788990011223344; - vm.etch(minedFactoryAddress, address(factory).code); - factory = ERC1967Factory(minedFactoryAddress); - } - _; - } - - function testDeploy() public withFactories { - (address admin,) = _randomSigner(); - - vm.prank(admin); - address proxy = factory.deploy(implementation0, admin); - _checkProxyBytecode(proxy); - - assertEq(factory.adminOf(proxy), admin); - assertTrue(proxy != address(0)); - assertTrue(proxy.code.length > 0); - _checkImplementationSlot(proxy, implementation0); - } - - function testDeployBrutalized(uint256) public withFactories { - (address admin,) = _randomSigner(); - address implementation = implementation0; - bool brutalized; - bool success; - address f = address(factory); - /// @solidity memory-safe-assembly - assembly { - calldatacopy(0x00, 0x00, 0x40) - brutalized := eq(and(mload(0x00), 1), 0) - if brutalized { - // Extremely unlikely that all 96 upper bits will be zero. - admin := or(shl(160, keccak256(0x00, 0x20)), admin) - implementation := or(shl(160, keccak256(0x00, 0x40)), implementation) - } - let m := mload(0x40) - mstore(m, 0x545e7c61) // `deploy(address, address)`. - mstore(add(m, 0x20), implementation) - mstore(add(m, 0x40), admin) - mstore(0x00, 0) - // Basically, we want to demonstrate that Solidity has checks - // to reject dirty upper bits for addresses. - success := call(gas(), f, 0, add(m, 0x1c), 0x44, 0x00, 0x20) - // If the call is successful, there will be a deployment. - if and(success, iszero(mload(0x00))) { revert(0, 0) } - } - assertEq(brutalized, !success); - } - - function testDeployAndCall(uint256) public withFactories { - (address admin,) = _randomSigner(); - _TestTemps memory t = _testTemps(); - - bytes memory data = abi.encodeWithSignature("setValue(uint256,uint256)", t.key, t.value); - vm.deal(admin, type(uint128).max); - vm.prank(admin); - address proxy = factory.deployAndCall{value: t.msgValue}(implementation0, admin, data); - - assertEq(factory.adminOf(proxy), admin); - assertTrue(proxy != address(0)); - assertTrue(proxy.code.length > 0); - _checkImplementationSlot(proxy, implementation0); - assertEq(MockImplementation(proxy).getValue(t.key), t.value); - assertEq(proxy.balance, t.msgValue); - } - - function testDeployDeterministicAndCall(uint256) public withFactories { - (address admin,) = _randomSigner(); - _TestTemps memory t = _testTemps(); - - t.predictedProxy = factory.predictDeterministicAddress(t.salt); - bytes memory data = abi.encodeWithSignature("setValue(uint256,uint256)", t.key, t.value); - vm.deal(admin, type(uint128).max); - vm.prank(admin); - if (_random() % 8 == 0) { - t.salt = keccak256(abi.encode(_random())); - vm.expectRevert(ERC1967Factory.SaltDoesNotStartWithCaller.selector); - t.proxy = factory.deployDeterministicAndCall{value: t.msgValue}( - implementation0, admin, t.salt, data - ); - return; - } else { - vm.expectEmit(true, true, true, true); - emit Deployed(t.predictedProxy, implementation0, admin); - t.proxy = factory.deployDeterministicAndCall{value: t.msgValue}( - implementation0, admin, t.salt, data - ); - assertEq(t.proxy, t.predictedProxy); - } - - assertEq(factory.adminOf(t.proxy), admin); - assertTrue(t.proxy != address(0)); - assertTrue(t.proxy.code.length > 0); - _checkImplementationSlot(t.proxy, implementation0); - assertEq(MockImplementation(t.proxy).getValue(t.key), t.value); - assertEq(t.proxy.balance, t.msgValue); - } - - function testDeployAndCallWithRevert() public withFactories { - (address admin,) = _randomSigner(); - - bytes memory data = abi.encodeWithSignature("fails()"); - vm.expectRevert(MockImplementation.Fail.selector); - factory.deployAndCall(implementation0, admin, data); - } - - function testProxySucceeds() public withFactories { - (address admin,) = _randomSigner(); - uint256 a = 1; - - MockImplementation proxy = MockImplementation(factory.deploy(implementation0, admin)); - - assertEq(proxy.succeeds(a), a); - } - - function testProxyFails() public withFactories { - (address admin,) = _randomSigner(); - - address proxy = factory.deploy(implementation0, admin); - - vm.expectRevert(MockImplementation.Fail.selector); - MockImplementation(proxy).fails(); - } - - function testChangeAdmin() public withFactories { - (address admin, address newAdmin) = _randomAccounts(); - - vm.prank(admin); - address proxy = factory.deploy(implementation0, admin); - - vm.expectEmit(true, true, true, true, address(factory)); - emit AdminChanged(proxy, newAdmin); - - vm.prank(admin); - factory.changeAdmin(proxy, newAdmin); - - assertEq(factory.adminOf(proxy), newAdmin); - } - - function testChangeAdminUnauthorized() public withFactories { - (address admin, address sussyAccount) = _randomAccounts(); - - vm.prank(admin); - address proxy = factory.deploy(implementation0, admin); - - vm.expectRevert(ERC1967Factory.Unauthorized.selector); - - vm.prank(sussyAccount); - factory.changeAdmin(proxy, sussyAccount); - } - - function testUpgrade() public withFactories { - (address admin,) = _randomSigner(); - - vm.prank(admin); - address proxy = factory.deploy(implementation0, admin); - - vm.expectEmit(true, true, true, true, address(factory)); - emit Upgraded(proxy, implementation1); - - vm.prank(admin); - factory.upgrade(proxy, implementation1); - - _checkImplementationSlot(proxy, implementation1); - } - - function testUpgradeAndCall() public withFactories { - (address admin,) = _randomSigner(); - _TestTemps memory t = _testTemps(); - - vm.prank(admin); - address proxy = factory.deploy(implementation0, admin); - - vm.expectEmit(true, true, true, true, address(factory)); - emit Upgraded(proxy, implementation1); - - vm.prank(admin); - vm.deal(admin, type(uint128).max); - bytes memory data = abi.encodeWithSignature("setValue(uint256,uint256)", t.key, t.value); - factory.upgradeAndCall{value: t.msgValue}(proxy, implementation1, data); - - _checkImplementationSlot(proxy, implementation1); - uint256 gasBefore = gasleft(); - uint256 storedValue = MockImplementation(proxy).getValue(t.key); - unchecked { - console.log(gasBefore - gasleft()); - } - assertEq(storedValue, t.value); - assertEq(proxy.balance, t.msgValue); - } - - function testUpgradeAndCallWithRevert() public withFactories { - (address admin,) = _randomSigner(); - - vm.prank(admin); - address proxy = factory.deploy(implementation0, admin); - - vm.prank(admin); - vm.expectRevert(MockImplementation.Fail.selector); - factory.upgradeAndCall(proxy, implementation1, abi.encodeWithSignature("fails()")); - } - - function testUpgradeUnauthorized() public withFactories { - (address admin, address sussyAccount) = _randomAccounts(); - - vm.prank(admin); - address proxy = factory.deploy(implementation0, admin); - - vm.expectRevert(ERC1967Factory.Unauthorized.selector); - vm.prank(sussyAccount); - factory.upgrade(proxy, implementation1); - - vm.expectRevert(ERC1967Factory.Unauthorized.selector); - vm.prank(address(uint160(admin) ^ 1)); - factory.upgrade(proxy, implementation1); - - vm.prank(admin); - factory.upgrade(proxy, implementation1); - } - - function testUpgradeWithCorruptedProxy() public withFactories { - (address admin,) = _randomSigner(); - - vm.prank(admin); - address proxy = factory.deploy(implementation0, admin); - - vm.expectRevert(ERC1967Factory.Unauthorized.selector); - vm.prank(admin); - factory.upgrade(address(uint160(proxy) ^ 1), implementation1); - - _checkImplementationSlot(proxy, implementation0); - } - - function testFactoryDeployment() public { - address deployment = - _safeCreate2(ERC1967FactoryConstants.SALT, ERC1967FactoryConstants.INITCODE); - assertEq(deployment, ERC1967FactoryConstants.ADDRESS); - assertEq(deployment.code, ERC1967FactoryConstants.BYTECODE); - } - - function _randomAccounts() internal returns (address a, address b) { - (a,) = _randomSigner(); - do { - (b,) = _randomSigner(); - } while (a == b); - } - - function _checkImplementationSlot(address proxy, address implementation) internal { - bytes32 slot = bytes32(uint256(keccak256("eip1967.proxy.implementation")) - 1); - assertEq(vm.load(proxy, slot), bytes32(uint256(uint160(implementation)))); - } - - function _checkProxyBytecode(address proxy) internal { - bytes memory code = address(proxy).code; - assertEq(uint8(bytes1(code[code.length - 1])), 0xfd); - assertTrue(code.length == 127 || code.length == 121); - } -} diff --git a/lib/solady/test/ERC20.t.sol b/lib/solady/test/ERC20.t.sol deleted file mode 100644 index 9c69c4d..0000000 --- a/lib/solady/test/ERC20.t.sol +++ /dev/null @@ -1,482 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import "./utils/InvariantTest.sol"; - -import {ERC20, MockERC20} from "./utils/mocks/MockERC20.sol"; - -contract ERC20Test is SoladyTest { - MockERC20 token; - - bytes32 constant PERMIT_TYPEHASH = keccak256( - "Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)" - ); - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - struct _TestTemps { - address owner; - address to; - uint256 amount; - uint256 deadline; - uint8 v; - bytes32 r; - bytes32 s; - uint256 privateKey; - uint256 nonce; - } - - function _testTemps() internal returns (_TestTemps memory t) { - (t.owner, t.privateKey) = _randomSigner(); - t.to = _randomNonZeroAddress(); - t.amount = _random(); - t.deadline = _random(); - } - - function setUp() public { - token = new MockERC20("Token", "TKN", 18); - } - - function testMetadata() public { - assertEq(token.name(), "Token"); - assertEq(token.symbol(), "TKN"); - assertEq(token.decimals(), 18); - } - - function testMint() public { - vm.expectEmit(true, true, true, true); - emit Transfer(address(0), address(0xBEEF), 1e18); - token.mint(address(0xBEEF), 1e18); - - assertEq(token.totalSupply(), 1e18); - assertEq(token.balanceOf(address(0xBEEF)), 1e18); - } - - function testBurn() public { - token.mint(address(0xBEEF), 1e18); - - vm.expectEmit(true, true, true, true); - emit Transfer(address(0xBEEF), address(0), 0.9e18); - token.burn(address(0xBEEF), 0.9e18); - - assertEq(token.totalSupply(), 1e18 - 0.9e18); - assertEq(token.balanceOf(address(0xBEEF)), 0.1e18); - } - - function testApprove() public { - vm.expectEmit(true, true, true, true); - emit Approval(address(this), address(0xBEEF), 1e18); - assertTrue(token.approve(address(0xBEEF), 1e18)); - - assertEq(token.allowance(address(this), address(0xBEEF)), 1e18); - } - - function testTransfer() public { - token.mint(address(this), 1e18); - - vm.expectEmit(true, true, true, true); - emit Transfer(address(this), address(0xBEEF), 1e18); - assertTrue(token.transfer(address(0xBEEF), 1e18)); - assertEq(token.totalSupply(), 1e18); - - assertEq(token.balanceOf(address(this)), 0); - assertEq(token.balanceOf(address(0xBEEF)), 1e18); - } - - function testTransferFrom() public { - address from = address(0xABCD); - - token.mint(from, 1e18); - - vm.prank(from); - token.approve(address(this), 1e18); - - vm.expectEmit(true, true, true, true); - emit Transfer(from, address(0xBEEF), 1e18); - assertTrue(token.transferFrom(from, address(0xBEEF), 1e18)); - assertEq(token.totalSupply(), 1e18); - - assertEq(token.allowance(from, address(this)), 0); - - assertEq(token.balanceOf(from), 0); - assertEq(token.balanceOf(address(0xBEEF)), 1e18); - } - - function testInfiniteApproveTransferFrom() public { - address from = address(0xABCD); - - token.mint(from, 1e18); - - vm.prank(from); - token.approve(address(this), type(uint256).max); - - assertTrue(token.transferFrom(from, address(0xBEEF), 1e18)); - assertEq(token.totalSupply(), 1e18); - - assertEq(token.allowance(from, address(this)), type(uint256).max); - - assertEq(token.balanceOf(from), 0); - assertEq(token.balanceOf(address(0xBEEF)), 1e18); - } - - function testPermit() public { - _TestTemps memory t = _testTemps(); - t.deadline = block.timestamp; - - _signPermit(t); - - _expectPermitEmitApproval(t); - _permit(t); - - _checkAllowanceAndNonce(t); - } - - function testMintOverMaxUintReverts() public { - token.mint(address(this), type(uint256).max); - vm.expectRevert(ERC20.TotalSupplyOverflow.selector); - token.mint(address(this), 1); - } - - function testTransferInsufficientBalanceReverts() public { - token.mint(address(this), 0.9e18); - vm.expectRevert(ERC20.InsufficientBalance.selector); - token.transfer(address(0xBEEF), 1e18); - } - - function testTransferFromInsufficientAllowanceReverts() public { - address from = address(0xABCD); - - token.mint(from, 1e18); - - vm.prank(from); - token.approve(address(this), 0.9e18); - - vm.expectRevert(ERC20.InsufficientAllowance.selector); - token.transferFrom(from, address(0xBEEF), 1e18); - } - - function testTransferFromInsufficientBalanceReverts() public { - address from = address(0xABCD); - - token.mint(from, 0.9e18); - - vm.prank(from); - token.approve(address(this), 1e18); - - vm.expectRevert(ERC20.InsufficientBalance.selector); - token.transferFrom(from, address(0xBEEF), 1e18); - } - - function testMint(address to, uint256 amount) public { - vm.expectEmit(true, true, true, true); - emit Transfer(address(0), to, amount); - token.mint(to, amount); - - assertEq(token.totalSupply(), amount); - assertEq(token.balanceOf(to), amount); - } - - function testBurn(address from, uint256 mintAmount, uint256 burnAmount) public { - burnAmount = _bound(burnAmount, 0, mintAmount); - - token.mint(from, mintAmount); - vm.expectEmit(true, true, true, true); - emit Transfer(from, address(0), burnAmount); - token.burn(from, burnAmount); - - assertEq(token.totalSupply(), mintAmount - burnAmount); - assertEq(token.balanceOf(from), mintAmount - burnAmount); - } - - function testApprove(address to, uint256 amount) public { - assertTrue(token.approve(to, amount)); - - assertEq(token.allowance(address(this), to), amount); - } - - function testTransfer(address to, uint256 amount) public { - token.mint(address(this), amount); - - vm.expectEmit(true, true, true, true); - emit Transfer(address(this), to, amount); - assertTrue(token.transfer(to, amount)); - assertEq(token.totalSupply(), amount); - - if (address(this) == to) { - assertEq(token.balanceOf(address(this)), amount); - } else { - assertEq(token.balanceOf(address(this)), 0); - assertEq(token.balanceOf(to), amount); - } - } - - function testTransferFrom( - address spender, - address from, - address to, - uint256 approval, - uint256 amount - ) public { - amount = _bound(amount, 0, approval); - - token.mint(from, amount); - assertEq(token.balanceOf(from), amount); - - vm.prank(from); - token.approve(spender, approval); - - vm.expectEmit(true, true, true, true); - emit Transfer(from, to, amount); - vm.prank(spender); - assertTrue(token.transferFrom(from, to, amount)); - assertEq(token.totalSupply(), amount); - - if (approval == type(uint256).max) { - assertEq(token.allowance(from, spender), approval); - } else { - assertEq(token.allowance(from, spender), approval - amount); - } - - if (from == to) { - assertEq(token.balanceOf(from), amount); - } else { - assertEq(token.balanceOf(from), 0); - assertEq(token.balanceOf(to), amount); - } - } - - function testDirectTransfer(uint256) public { - _TestTemps memory t = _testTemps(); - while (t.owner == t.to) (t.to,) = _randomSigner(); - - uint256 totalSupply = _random(); - token.mint(t.owner, totalSupply); - assertEq(token.balanceOf(t.owner), totalSupply); - assertEq(token.balanceOf(t.to), 0); - if (t.amount > totalSupply) { - vm.expectRevert(ERC20.InsufficientBalance.selector); - token.directTransfer(t.owner, t.to, t.amount); - } else { - vm.expectEmit(true, true, true, true); - emit Transfer(t.owner, t.to, t.amount); - token.directTransfer(t.owner, t.to, t.amount); - assertEq(token.balanceOf(t.owner), totalSupply - t.amount); - assertEq(token.balanceOf(t.to), t.amount); - } - } - - function testDirectSpendAllowance(uint256) public { - _TestTemps memory t = _testTemps(); - uint256 allowance = _random(); - vm.prank(t.owner); - token.approve(t.to, allowance); - assertEq(token.allowance(t.owner, t.to), allowance); - if (allowance == type(uint256).max) { - token.directSpendAllowance(t.owner, t.to, t.amount); - assertEq(token.allowance(t.owner, t.to), allowance); - } else if (t.amount > allowance) { - vm.expectRevert(ERC20.InsufficientAllowance.selector); - token.directSpendAllowance(t.owner, t.to, t.amount); - } else { - token.directSpendAllowance(t.owner, t.to, t.amount); - assertEq(token.allowance(t.owner, t.to), allowance - t.amount); - } - } - - function testPermit(uint256) public { - _TestTemps memory t = _testTemps(); - if (t.deadline < block.timestamp) t.deadline = block.timestamp; - - _signPermit(t); - - _expectPermitEmitApproval(t); - _permit(t); - - _checkAllowanceAndNonce(t); - } - - function _checkAllowanceAndNonce(_TestTemps memory t) internal { - assertEq(token.allowance(t.owner, t.to), t.amount); - assertEq(token.nonces(t.owner), t.nonce + 1); - } - - function testBurnInsufficientBalanceReverts(address to, uint256 mintAmount, uint256 burnAmount) - public - { - if (mintAmount == type(uint256).max) mintAmount--; - burnAmount = _bound(burnAmount, mintAmount + 1, type(uint256).max); - - token.mint(to, mintAmount); - vm.expectRevert(ERC20.InsufficientBalance.selector); - token.burn(to, burnAmount); - } - - function testTransferInsufficientBalanceReverts( - address to, - uint256 mintAmount, - uint256 sendAmount - ) public { - if (mintAmount == type(uint256).max) mintAmount--; - sendAmount = _bound(sendAmount, mintAmount + 1, type(uint256).max); - - token.mint(address(this), mintAmount); - vm.expectRevert(ERC20.InsufficientBalance.selector); - token.transfer(to, sendAmount); - } - - function testTransferFromInsufficientAllowanceReverts( - address to, - uint256 approval, - uint256 amount - ) public { - if (approval == type(uint256).max) approval--; - amount = _bound(amount, approval + 1, type(uint256).max); - - address from = address(0xABCD); - - token.mint(from, amount); - - vm.prank(from); - token.approve(address(this), approval); - - vm.expectRevert(ERC20.InsufficientAllowance.selector); - token.transferFrom(from, to, amount); - } - - function testTransferFromInsufficientBalanceReverts( - address to, - uint256 mintAmount, - uint256 sendAmount - ) public { - if (mintAmount == type(uint256).max) mintAmount--; - sendAmount = _bound(sendAmount, mintAmount + 1, type(uint256).max); - - address from = address(0xABCD); - - token.mint(from, mintAmount); - - vm.prank(from); - token.approve(address(this), sendAmount); - - vm.expectRevert(ERC20.InsufficientBalance.selector); - token.transferFrom(from, to, sendAmount); - } - - function testPermitBadNonceReverts(uint256) public { - _TestTemps memory t = _testTemps(); - if (t.deadline < block.timestamp) t.deadline = block.timestamp; - while (t.nonce == 0) t.nonce = _random(); - - _signPermit(t); - - vm.expectRevert(ERC20.InvalidPermit.selector); - _permit(t); - } - - function testPermitBadDeadlineReverts(uint256) public { - _TestTemps memory t = _testTemps(); - if (t.deadline == type(uint256).max) t.deadline--; - if (t.deadline < block.timestamp) t.deadline = block.timestamp; - - _signPermit(t); - - vm.expectRevert(ERC20.InvalidPermit.selector); - t.deadline += 1; - _permit(t); - } - - function testPermitPastDeadlineReverts(uint256) public { - _TestTemps memory t = _testTemps(); - t.deadline = _bound(t.deadline, 0, block.timestamp - 1); - - _signPermit(t); - - vm.expectRevert(ERC20.PermitExpired.selector); - _permit(t); - } - - function testPermitReplayReverts(uint256) public { - _TestTemps memory t = _testTemps(); - if (t.deadline < block.timestamp) t.deadline = block.timestamp; - - _signPermit(t); - - _expectPermitEmitApproval(t); - _permit(t); - vm.expectRevert(ERC20.InvalidPermit.selector); - _permit(t); - } - - function _signPermit(_TestTemps memory t) internal view { - bytes32 innerHash = - keccak256(abi.encode(PERMIT_TYPEHASH, t.owner, t.to, t.amount, t.nonce, t.deadline)); - bytes32 domainSeparator = token.DOMAIN_SEPARATOR(); - bytes32 outerHash = keccak256(abi.encodePacked("\x19\x01", domainSeparator, innerHash)); - (t.v, t.r, t.s) = vm.sign(t.privateKey, outerHash); - } - - function _expectPermitEmitApproval(_TestTemps memory t) internal { - vm.expectEmit(true, true, true, true); - emit Approval(t.owner, t.to, t.amount); - } - - function _permit(_TestTemps memory t) internal { - address token_ = address(token); - /// @solidity memory-safe-assembly - assembly { - let m := mload(sub(t, 0x20)) - mstore(sub(t, 0x20), 0xd505accf) - pop(call(gas(), token_, 0, sub(t, 0x04), 0xe4, 0x00, 0x00)) - mstore(sub(t, 0x20), m) - } - } -} - -contract ERC20Invariants is SoladyTest, InvariantTest { - BalanceSum balanceSum; - MockERC20 token; - - function setUp() public { - token = new MockERC20("Token", "TKN", 18); - balanceSum = new BalanceSum(token); - _addTargetContract(address(balanceSum)); - } - - function invariantBalanceSum() public { - assertEq(token.totalSupply(), balanceSum.sum()); - } -} - -contract BalanceSum { - MockERC20 token; - uint256 public sum; - - constructor(MockERC20 _token) { - token = _token; - } - - function mint(address from, uint256 amount) public { - token.mint(from, amount); - sum += amount; - } - - function burn(address from, uint256 amount) public { - token.burn(from, amount); - sum -= amount; - } - - function approve(address to, uint256 amount) public { - token.approve(to, amount); - } - - function transferFrom(address from, address to, uint256 amount) public { - token.transferFrom(from, to, amount); - } - - function transfer(address to, uint256 amount) public { - token.transfer(to, amount); - } -} diff --git a/lib/solady/test/ERC2981.t.sol b/lib/solady/test/ERC2981.t.sol deleted file mode 100644 index 83340dc..0000000 --- a/lib/solady/test/ERC2981.t.sol +++ /dev/null @@ -1,140 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; - -import {ERC2981, MockERC2981} from "./utils/mocks/MockERC2981.sol"; - -contract ERC2981Test is SoladyTest { - MockERC2981 token; - - function setUp() public { - token = new MockERC2981(); - } - - struct _TestTemps { - uint256 feeDenominator; - address[2] receivers; - uint256[2] tokenIds; - uint256[2] salePrices; - uint256[2] royaltyFractions; - address defaultReceiver; - uint256 defaultRoyaltyFraction; - } - - function _testTemps() internal returns (_TestTemps memory t) { - t.feeDenominator = token.feeDenominator(); - t.tokenIds[0] = _random(); - do { - t.tokenIds[1] = _random(); - } while (t.tokenIds[0] == t.tokenIds[1]); - t.receivers[0] = _randomNonZeroAddress(); - do { - t.receivers[1] = _randomNonZeroAddress(); - } while (t.receivers[0] == t.receivers[1]); - t.salePrices[0] = _bound(_random(), 0, type(uint160).max); - t.salePrices[1] = _bound(_random(), 0, type(uint160).max); - t.defaultReceiver = _randomNonZeroAddress(); - t.defaultRoyaltyFraction = _bound(_random(), 0, t.feeDenominator); - t.royaltyFractions[0] = _bound(_random(), 0, t.feeDenominator); - t.royaltyFractions[1] = _bound(_random(), 0, t.feeDenominator); - } - - function testRoyaltyOverflowCheckDifferential(uint256 x, uint256 y) public { - unchecked { - bool expected = x != 0 && (x * y) / x != y; - bool computed; - /// @solidity memory-safe-assembly - assembly { - computed := mul(y, gt(x, div(not(0), y))) - } - assertEq(computed, expected); - } - } - - function testSetAndGetRoyaltyInfo(uint256) public { - _TestTemps memory t = _testTemps(); - - if (_random() % 16 == 0) _checkReverts(t); - - _checkRoyaltyInfoIsZero(t); - - token.setDefaultRoyalty(t.defaultReceiver, uint96(t.defaultRoyaltyFraction)); - _checkRoyaltyInfoIsDefault(t, 0); - _checkRoyaltyInfoIsDefault(t, 1); - - token.setTokenRoyalty(t.tokenIds[0], t.receivers[0], uint96(t.royaltyFractions[0])); - _checkRoyaltyInfo(t, 0); - _checkRoyaltyInfoIsDefault(t, 1); - token.setTokenRoyalty(t.tokenIds[1], t.receivers[1], uint96(t.royaltyFractions[1])); - _checkRoyaltyInfo(t, 0); - _checkRoyaltyInfo(t, 1); - - if (_random() % 16 == 0) _checkReverts(t); - - token.resetTokenRoyalty(t.tokenIds[0]); - _checkRoyaltyInfoIsDefault(t, 0); - _checkRoyaltyInfo(t, 1); - token.resetTokenRoyalty(t.tokenIds[1]); - _checkRoyaltyInfoIsDefault(t, 0); - _checkRoyaltyInfoIsDefault(t, 1); - - if (_random() % 16 == 0) _checkReverts(t); - - token.deleteDefaultRoyalty(); - - _checkRoyaltyInfoIsZero(t); - - if (_random() % 16 == 0) _checkReverts(t); - } - - function _getInvalidFeeNumerator(_TestTemps memory t) internal returns (uint96 r) { - while (true) { - r = uint96(_random()); - if (r > t.feeDenominator) break; - } - } - - function _checkReverts(_TestTemps memory t) internal { - vm.expectRevert(ERC2981.RoyaltyReceiverIsZeroAddress.selector); - token.setDefaultRoyalty(address(0), 1); - vm.expectRevert(ERC2981.RoyaltyOverflow.selector); - token.setDefaultRoyalty(t.defaultReceiver, _getInvalidFeeNumerator(t)); - - vm.expectRevert(ERC2981.RoyaltyReceiverIsZeroAddress.selector); - token.setTokenRoyalty(t.tokenIds[0], address(0), 1); - vm.expectRevert(ERC2981.RoyaltyOverflow.selector); - token.setTokenRoyalty(t.tokenIds[0], t.receivers[0], _getInvalidFeeNumerator(t)); - - vm.expectRevert(ERC2981.RoyaltyReceiverIsZeroAddress.selector); - token.setTokenRoyalty(t.tokenIds[1], address(0), 1); - vm.expectRevert(ERC2981.RoyaltyOverflow.selector); - token.setTokenRoyalty(t.tokenIds[1], t.receivers[1], _getInvalidFeeNumerator(t)); - } - - function _checkRoyaltyInfoIsZero(_TestTemps memory t) internal { - _checkRoyaltyInfo(t, 0, address(0), 0); - _checkRoyaltyInfo(t, 1, address(0), 0); - } - - function _checkRoyaltyInfoIsDefault(_TestTemps memory t, uint256 i) internal { - uint256 expected = t.salePrices[i] * t.defaultRoyaltyFraction / t.feeDenominator; - _checkRoyaltyInfo(t, i, t.defaultReceiver, expected); - } - - function _checkRoyaltyInfo(_TestTemps memory t, uint256 i) internal { - uint256 expected = t.salePrices[i] * t.royaltyFractions[i] / t.feeDenominator; - _checkRoyaltyInfo(t, i, t.receivers[i], expected); - } - - function _checkRoyaltyInfo( - _TestTemps memory t, - uint256 i, - address expectedReceiver, - uint256 expectedAmount - ) internal { - (address receiver, uint256 amount) = token.royaltyInfo(t.tokenIds[i], t.salePrices[i]); - assertEq(receiver, expectedReceiver); - assertEq(amount, expectedAmount); - } -} diff --git a/lib/solady/test/ERC4337.t.sol b/lib/solady/test/ERC4337.t.sol deleted file mode 100644 index a54370b..0000000 --- a/lib/solady/test/ERC4337.t.sol +++ /dev/null @@ -1,555 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {Ownable, SignatureCheckerLib} from "../src/accounts/ERC4337.sol"; -import {ERC4337, MockERC4337} from "./utils/mocks/MockERC4337.sol"; -import {MockEntryPoint} from "./utils/mocks/MockEntryPoint.sol"; -import {MockERC721} from "./utils/mocks/MockERC721.sol"; -import {MockERC1155} from "./utils/mocks/MockERC1155.sol"; -import {MockERC1271Wallet} from "./utils/mocks/MockERC1271Wallet.sol"; -import {LibClone} from "../src/utils/LibClone.sol"; -import {LibString} from "../src/utils/LibString.sol"; -import {LibZip} from "../src/utils/LibZip.sol"; - -contract Target { - error TargetError(bytes data); - - bytes32 public datahash; - - bytes public data; - - function setData(bytes memory data_) public payable returns (bytes memory) { - data = data_; - datahash = keccak256(data_); - return data_; - } - - function revertWithTargetError(bytes memory data_) public payable { - revert TargetError(data_); - } - - function changeOwnerSlotValue(bool change) public payable { - /// @solidity memory-safe-assembly - assembly { - if change { sstore(not(0x8b78c6d8), 0x112233) } - } - } -} - -contract ERC4337Test is SoladyTest { - event OwnershipTransferred(address indexed oldOwner, address indexed newOwner); - - // By right, this should be the keccak256 of some long-ass string: - // (e.g. `keccak256("Parent(bytes32 childHash,Mail child)Mail(Person from,Person to,string contents)Person(string name,address wallet)")`). - // But I'm lazy and will use something randomish here. - bytes32 internal constant _PARENT_TYPEHASH = - 0xd61db970ec8a2edc5f9fd31d876abe01b785909acb16dcd4baaf3b434b4c439b; - - // By right, this should be a proper domain separator, but I'm lazy. - bytes32 internal constant _DOMAIN_SEP_B = - 0xa1a044077d7677adbbfa892ded5390979b33993e0e2a457e3f974bbcda53821b; - - address internal constant _ENTRY_POINT = 0x5FF137D4b0FDCD49DcA30c7CF57E578a026d2789; - - address erc4337; - - MockERC4337 account; - - function setUp() public { - // Etch something onto `_ENTRY_POINT` such that we can deploy the account implementation. - vm.etch(_ENTRY_POINT, hex"00"); - erc4337 = address(new MockERC4337()); - account = MockERC4337(payable(LibClone.deployERC1967(erc4337))); - } - - function testInitializer() public { - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(0), address(this)); - account.initialize(address(this)); - assertEq(account.owner(), address(this)); - vm.expectRevert(Ownable.AlreadyInitialized.selector); - account.initialize(address(this)); - - address newOwner = _randomNonZeroAddress(); - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(this), newOwner); - account.transferOwnership(newOwner); - assertEq(account.owner(), newOwner); - - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(newOwner, address(this)); - vm.prank(newOwner); - account.transferOwnership(address(this)); - - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(this), address(0)); - account.renounceOwnership(); - assertEq(account.owner(), address(0)); - - vm.expectRevert(Ownable.AlreadyInitialized.selector); - account.initialize(address(this)); - assertEq(account.owner(), address(0)); - - account = MockERC4337(payable(LibClone.deployERC1967(erc4337))); - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(0), address(0)); - account.initialize(address(0)); - assertEq(account.owner(), address(0)); - - vm.expectRevert(Ownable.AlreadyInitialized.selector); - account.initialize(address(this)); - assertEq(account.owner(), address(0)); - - account = MockERC4337(payable(LibClone.deployERC1967(erc4337))); - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(0), address(1)); - account.initialize(address(1)); - assertEq(account.owner(), address(1)); - - vm.expectRevert(Ownable.AlreadyInitialized.selector); - account.initialize(address(this)); - assertEq(account.owner(), address(1)); - } - - function testExecute() public { - vm.deal(address(account), 1 ether); - account.initialize(address(this)); - - address target = address(new Target()); - bytes memory data = _randomBytes(111); - account.execute(target, 123, abi.encodeWithSignature("setData(bytes)", data)); - assertEq(Target(target).datahash(), keccak256(data)); - assertEq(target.balance, 123); - - vm.prank(_randomNonZeroAddress()); - vm.expectRevert(Ownable.Unauthorized.selector); - account.execute(target, 123, abi.encodeWithSignature("setData(bytes)", data)); - - vm.expectRevert(abi.encodeWithSignature("TargetError(bytes)", data)); - account.execute(target, 123, abi.encodeWithSignature("revertWithTargetError(bytes)", data)); - } - - function testExecuteBatch() public { - vm.deal(address(account), 1 ether); - account.initialize(address(this)); - - ERC4337.Call[] memory calls = new ERC4337.Call[](2); - calls[0].target = address(new Target()); - calls[1].target = address(new Target()); - calls[0].value = 123; - calls[1].value = 456; - calls[0].data = abi.encodeWithSignature("setData(bytes)", _randomBytes(111)); - calls[1].data = abi.encodeWithSignature("setData(bytes)", _randomBytes(222)); - - account.executeBatch(calls); - assertEq(Target(calls[0].target).datahash(), keccak256(_randomBytes(111))); - assertEq(Target(calls[1].target).datahash(), keccak256(_randomBytes(222))); - assertEq(calls[0].target.balance, 123); - assertEq(calls[1].target.balance, 456); - - calls[1].data = abi.encodeWithSignature("revertWithTargetError(bytes)", _randomBytes(111)); - vm.expectRevert(abi.encodeWithSignature("TargetError(bytes)", _randomBytes(111))); - account.executeBatch(calls); - } - - function testExecuteBatch(uint256 r) public { - vm.deal(address(account), 1 ether); - account.initialize(address(this)); - - unchecked { - uint256 n = r & 3; - ERC4337.Call[] memory calls = new ERC4337.Call[](n); - - for (uint256 i; i != n; ++i) { - uint256 v = _random() & 0xff; - calls[i].target = address(new Target()); - calls[i].value = v; - calls[i].data = abi.encodeWithSignature("setData(bytes)", _randomBytes(v)); - } - - bytes[] memory results; - if (_random() & 1 == 0) { - results = account.executeBatch(_random(), calls); - } else { - results = account.executeBatch(calls); - } - - assertEq(results.length, n); - for (uint256 i; i != n; ++i) { - uint256 v = calls[i].value; - assertEq(Target(calls[i].target).datahash(), keccak256(_randomBytes(v))); - assertEq(calls[i].target.balance, v); - assertEq(abi.decode(results[i], (bytes)), _randomBytes(v)); - } - } - } - - function testDelegateExecute() public { - testDelegateExecute(123); - } - - function testDelegateExecute(uint256 r) public { - vm.deal(address(account), 1 ether); - account.initialize(address(this)); - - address delegate = address(new Target()); - - bytes memory data; - data = abi.encodeWithSignature("setData(bytes)", _randomBytes(r)); - data = account.delegateExecute(delegate, data); - assertEq(abi.decode(data, (bytes)), _randomBytes(r)); - data = account.delegateExecute(delegate, abi.encodeWithSignature("datahash()")); - assertEq(abi.decode(data, (bytes32)), keccak256(_randomBytes(r))); - data = account.delegateExecute(delegate, abi.encodeWithSignature("data()")); - assertEq(abi.decode(data, (bytes)), _randomBytes(r)); - } - - function testDelegateExecuteRevertsIfOwnerSlotValueChanged() public { - account.initialize(address(this)); - - address delegate = address(new Target()); - - bytes memory data; - data = abi.encodeWithSignature("changeOwnerSlotValue(bool)", false); - account.delegateExecute(delegate, data); - vm.expectRevert(); - data = abi.encodeWithSignature("changeOwnerSlotValue(bool)", true); - account.delegateExecute(delegate, data); - data = abi.encodeWithSignature("changeOwnerSlotValue(bool)", false); - account.delegateExecute(delegate, data); - } - - function testDepositFunctions() public { - vm.deal(address(account), 1 ether); - account.initialize(address(this)); - - vm.etch(account.entryPoint(), address(new MockEntryPoint()).code); - assertEq(account.getDeposit(), 0); - account.addDeposit{value: 123}(); - assertEq(account.getDeposit(), 123); - address to = _randomNonZeroAddress(); - assertEq(to.balance, 0); - account.withdrawDepositTo(to, 12); - assertEq(to.balance, 12); - assertEq(account.getDeposit(), 123 - 12); - } - - function testCdFallback() public { - vm.deal(address(account), 1 ether); - account.initialize(address(this)); - - vm.etch(account.entryPoint(), address(new MockEntryPoint()).code); - assertEq(account.getDeposit(), 0); - - bytes memory data = LibZip.cdCompress(abi.encodeWithSignature("addDeposit()")); - (bool success,) = address(account).call{value: 123}(data); - assertTrue(success); - assertEq(account.getDeposit(), 123); - } - - function testCdFallback2() public { - vm.deal(address(account), 1 ether); - account.initialize(address(this)); - - vm.etch(account.entryPoint(), address(new MockEntryPoint()).code); - assertEq(account.getDeposit(), 0); - - ERC4337.Call[] memory calls = new ERC4337.Call[](2); - calls[0].target = address(new Target()); - calls[1].target = address(new Target()); - calls[0].value = 123; - calls[1].value = 456; - calls[0].data = abi.encodeWithSignature("setData(bytes)", _randomBytes(111)); - calls[1].data = abi.encodeWithSignature("setData(bytes)", _randomBytes(222)); - - bytes memory data = LibZip.cdCompress( - abi.encodeWithSignature("executeBatch((address,uint256,bytes)[])", calls) - ); - (bool success,) = address(account).call(data); - assertTrue(success); - assertEq(Target(calls[0].target).datahash(), keccak256(_randomBytes(111))); - assertEq(Target(calls[1].target).datahash(), keccak256(_randomBytes(222))); - assertEq(calls[0].target.balance, 123); - assertEq(calls[1].target.balance, 456); - } - - struct _TestTemps { - bytes32 userOpHash; - bytes32 hash; - address signer; - uint256 privateKey; - uint8 v; - bytes32 r; - bytes32 s; - uint256 missingAccountFunds; - } - - function testValidateUserOp() public { - _TestTemps memory t; - t.userOpHash = keccak256("123"); - (t.signer, t.privateKey) = _randomSigner(); - (t.v, t.r, t.s) = - vm.sign(t.privateKey, SignatureCheckerLib.toEthSignedMessageHash(t.userOpHash)); - t.missingAccountFunds = 456; - vm.deal(address(account), 1 ether); - assertEq(address(account).balance, 1 ether); - - account.initialize(t.signer); - - vm.etch(account.entryPoint(), address(new MockEntryPoint()).code); - MockEntryPoint ep = MockEntryPoint(payable(account.entryPoint())); - - ERC4337.UserOperation memory userOp; - // Success returns 0. - userOp.signature = abi.encodePacked(t.r, t.s, t.v); - assertEq( - ep.validateUserOp(address(account), userOp, t.userOpHash, t.missingAccountFunds), 0 - ); - assertEq(address(ep).balance, t.missingAccountFunds); - // Failure returns 1. - userOp.signature = abi.encodePacked(t.r, bytes32(uint256(t.s) ^ 1), t.v); - assertEq( - ep.validateUserOp(address(account), userOp, t.userOpHash, t.missingAccountFunds), 1 - ); - assertEq(address(ep).balance, t.missingAccountFunds * 2); - // Not entry point reverts. - vm.expectRevert(Ownable.Unauthorized.selector); - account.validateUserOp(userOp, t.userOpHash, t.missingAccountFunds); - } - - function testIsValidSignature() public { - _TestTemps memory t; - t.hash = keccak256("123"); - (t.signer, t.privateKey) = _randomSigner(); - (t.v, t.r, t.s) = vm.sign(t.privateKey, _toERC1271Hash(t.hash)); - - account.initialize(t.signer); - - bytes memory signature = - abi.encodePacked(t.r, t.s, t.v, _PARENT_TYPEHASH, _DOMAIN_SEP_B, t.hash); - assertEq(account.isValidSignature(_toChildHash(t.hash), signature), bytes4(0x1626ba7e)); - - unchecked { - uint256 vs = uint256(t.s) | uint256(t.v - 27) << 255; - signature = abi.encodePacked(t.r, vs, _PARENT_TYPEHASH, _DOMAIN_SEP_B, t.hash); - assertEq(account.isValidSignature(_toChildHash(t.hash), signature), bytes4(0x1626ba7e)); - } - - signature = - abi.encodePacked(t.r, t.s, t.v, uint256(_PARENT_TYPEHASH) ^ 1, _DOMAIN_SEP_B, t.hash); - assertEq(account.isValidSignature(_toChildHash(t.hash), signature), bytes4(0xffffffff)); - - signature = - abi.encodePacked(t.r, t.s, t.v, _PARENT_TYPEHASH, uint256(_DOMAIN_SEP_B) ^ 1, t.hash); - assertEq(account.isValidSignature(_toChildHash(t.hash), signature), bytes4(0xffffffff)); - - signature = - abi.encodePacked(t.r, t.s, t.v, _PARENT_TYPEHASH, _DOMAIN_SEP_B, uint256(t.hash) ^ 1); - assertEq(account.isValidSignature(_toChildHash(t.hash), signature), bytes4(0xffffffff)); - - signature = abi.encodePacked(t.r, t.s, t.v, _PARENT_TYPEHASH); - assertEq(account.isValidSignature(t.hash, signature), bytes4(0xffffffff)); - - signature = abi.encodePacked(t.r, t.s, _PARENT_TYPEHASH); - assertEq(account.isValidSignature(t.hash, signature), bytes4(0xffffffff)); - - signature = abi.encodePacked(t.r, _PARENT_TYPEHASH); - assertEq(account.isValidSignature(t.hash, signature), bytes4(0xffffffff)); - - signature = ""; - assertEq(account.isValidSignature(t.hash, signature), bytes4(0xffffffff)); - } - - function testIsValidSignaturePersonalSign() public { - _TestTemps memory t; - t.hash = keccak256("123"); - (t.signer, t.privateKey) = _randomSigner(); - (t.v, t.r, t.s) = vm.sign(t.privateKey, _toERC1271HashPersonalSign(t.hash)); - - account.initialize(t.signer); - - bytes memory signature = abi.encodePacked(t.r, t.s, t.v, _PARENT_TYPEHASH); - assertEq(account.isValidSignature(t.hash, signature), bytes4(0x1626ba7e)); - - unchecked { - uint256 vs = uint256(t.s) | uint256(t.v - 27) << 255; - signature = abi.encodePacked(t.r, vs, _PARENT_TYPEHASH); - assertEq(account.isValidSignature(t.hash, signature), bytes4(0x1626ba7e)); - } - - signature = abi.encodePacked(t.r, t.s, _PARENT_TYPEHASH, _DOMAIN_SEP_B, t.hash); - assertEq(account.isValidSignature(t.hash, signature), bytes4(0xffffffff)); - - signature = abi.encodePacked(t.r, t.s, _PARENT_TYPEHASH, _DOMAIN_SEP_B); - assertEq(account.isValidSignature(t.hash, signature), bytes4(0xffffffff)); - - signature = abi.encodePacked(t.r, t.s, _PARENT_TYPEHASH); - assertEq(account.isValidSignature(t.hash, signature), bytes4(0xffffffff)); - - signature = abi.encodePacked(t.r, _PARENT_TYPEHASH); - assertEq(account.isValidSignature(t.hash, signature), bytes4(0xffffffff)); - - signature = ""; - assertEq(account.isValidSignature(t.hash, signature), bytes4(0xffffffff)); - } - - function testIsValidSignatureWrapped() public { - _TestTemps memory t; - t.hash = keccak256("123"); - (t.signer, t.privateKey) = _randomSigner(); - (t.v, t.r, t.s) = vm.sign(t.privateKey, _toERC1271Hash(t.hash)); - - MockERC1271Wallet wrappedSigner = new MockERC1271Wallet(t.signer); - account.initialize(address(wrappedSigner)); - - bytes memory signature = - abi.encodePacked(t.r, t.s, t.v, _PARENT_TYPEHASH, _DOMAIN_SEP_B, t.hash); - assertEq(account.isValidSignature(_toChildHash(t.hash), signature), bytes4(0x1626ba7e)); - } - - function _toERC1271Hash(bytes32 child) internal view returns (bytes32) { - bytes32 domainSeparator = keccak256( - abi.encode( - keccak256( - "EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)" - ), - keccak256("Milady"), - keccak256("1"), - block.chainid, - address(account) - ) - ); - bytes32 parentStructHash = - keccak256(abi.encode(_PARENT_TYPEHASH, _toChildHash(child), child)); - return keccak256(abi.encodePacked("\x19\x01", domainSeparator, parentStructHash)); - } - - function _toChildHash(bytes32 child) internal pure returns (bytes32) { - return keccak256(abi.encodePacked(hex"1901", _DOMAIN_SEP_B, child)); - } - - function _toERC1271HashPersonalSign(bytes32 childHash) internal view returns (bytes32) { - bytes32 domainSeparator = keccak256( - abi.encode( - keccak256( - "EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)" - ), - keccak256("Milady"), - keccak256("1"), - block.chainid, - address(account) - ) - ); - bytes32 parentStructHash = keccak256(abi.encode(_PARENT_TYPEHASH, childHash)); - return keccak256(abi.encodePacked("\x19\x01", domainSeparator, parentStructHash)); - } - - function testETHReceived() public { - (bool success,) = address(account).call{value: 1 ether}(""); - assertTrue(success); - } - - function testOnERC721Received() public { - address alice = _randomNonZeroAddress(); - MockERC721 erc721 = new MockERC721(); - erc721.mint(alice, 1); - vm.prank(alice); - erc721.safeTransferFrom(alice, address(account), 1); - } - - function testOnERC1155Received() public { - address alice = _randomNonZeroAddress(); - MockERC1155 erc1155 = new MockERC1155(); - erc1155.mint(alice, 1, 1, ""); - vm.prank(alice); - erc1155.safeTransferFrom(alice, address(account), 1, 1, ""); - } - - function testOnERC1155BatchReceived() public { - address alice = _randomNonZeroAddress(); - MockERC1155 erc1155 = new MockERC1155(); - erc1155.mint(alice, 1, 1, ""); - uint256[] memory ids = new uint256[](1); - ids[0] = 1; - uint256[] memory amts = new uint256[](1); - amts[0] = 1; - vm.prank(alice); - erc1155.safeBatchTransferFrom(alice, address(account), ids, amts, ""); - } - - function testDirectStorage() public { - bytes32 storageSlot = bytes32(uint256(123)); - bytes32 storageValue = bytes32(uint256(456)); - - vm.expectRevert(Ownable.Unauthorized.selector); - account.storageStore(storageSlot, storageValue); - - account.initialize(address(this)); - assertEq(account.storageLoad(storageSlot), bytes32(0)); - account.storageStore(storageSlot, storageValue); - assertEq(account.storageLoad(storageSlot), storageValue); - } - - function testOwnerRecovery() public { - ERC4337.UserOperation memory userOp; - - userOp.sender = address(account); - userOp.nonce = 4337; - - // `bob` is set as recovery. - address bob = address(0xb); - userOp.callData = abi.encodeWithSelector( - ERC4337.execute.selector, - address(account), - 0, - abi.encodeWithSelector(Ownable.completeOwnershipHandover.selector, bob) - ); - - // `bob` must accept recovery. - // IRL this would follow need. - vm.prank(bob); - account.requestOwnershipHandover(); - - _TestTemps memory t; - t.userOpHash = keccak256(abi.encode(userOp)); - (t.signer, t.privateKey) = _randomSigner(); - (t.v, t.r, t.s) = - vm.sign(t.privateKey, SignatureCheckerLib.toEthSignedMessageHash(t.userOpHash)); - - t.missingAccountFunds = 456; - vm.deal(address(account), 1 ether); - - account.initialize(t.signer); - assertEq(account.owner(), t.signer); - - vm.etch(account.entryPoint(), address(new MockEntryPoint()).code); - MockEntryPoint ep = MockEntryPoint(payable(account.entryPoint())); - - // Success returns 0. - userOp.signature = abi.encodePacked(t.r, t.s, t.v); - assertEq( - ep.validateUserOp(address(account), userOp, t.userOpHash, t.missingAccountFunds), 0 - ); - // Check recovery to `bob`. - vm.prank(address(ep)); - (bool success,) = address(account).call(userOp.callData); - assertTrue(success); - assertEq(account.owner(), bob); - } - - function _randomBytes(uint256 seed) internal pure returns (bytes memory result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, seed) - let r := keccak256(0x00, 0x20) - if lt(byte(2, r), 0x20) { - result := mload(0x40) - let n := and(r, 0x7f) - mstore(result, n) - codecopy(add(result, 0x20), byte(1, r), add(n, 0x40)) - mstore(0x40, add(add(result, 0x40), n)) - } - } - } -} diff --git a/lib/solady/test/ERC4337Factory.t.sol b/lib/solady/test/ERC4337Factory.t.sol deleted file mode 100644 index 33eb422..0000000 --- a/lib/solady/test/ERC4337Factory.t.sol +++ /dev/null @@ -1,75 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {MockERC4337} from "./utils/mocks/MockERC4337.sol"; -import {ERC4337Factory} from "../src/accounts/ERC4337Factory.sol"; -import {LibClone} from "../src/utils/LibClone.sol"; - -contract ERC4337FactoryTest is SoladyTest { - address internal constant _ENTRY_POINT = 0x5FF137D4b0FDCD49DcA30c7CF57E578a026d2789; - - ERC4337Factory factory; - - MockERC4337 erc4337; - - function setUp() public { - // Etch something onto `_ENTRY_POINT` such that we can deploy the account implementation. - vm.etch(_ENTRY_POINT, hex"00"); - erc4337 = new MockERC4337(); - factory = new ERC4337Factory(address(erc4337)); - } - - function testDeployDeterministic(uint256) public { - vm.deal(address(this), 100 ether); - address owner = _randomNonZeroAddress(); - uint256 initialValue = _random() % 100 ether; - bytes32 salt = _random() % 8 == 0 ? bytes32(_random()) : bytes32(uint256(uint96(_random()))); - address account; - if (uint256(salt) >> 96 != uint160(owner) && uint256(salt) >> 96 != 0) { - vm.expectRevert(LibClone.SaltDoesNotStartWith.selector); - account = factory.createAccount{value: initialValue}(owner, salt); - return; - } else { - account = factory.createAccount{value: initialValue}(owner, salt); - } - assertEq(address(account).balance, initialValue); - assertEq(MockERC4337(payable(account)).owner(), owner); - _checkImplementationSlot(account, address(erc4337)); - } - - function testCreateAccountRepeatedDeployment() public { - bytes32 salt = bytes32(_random() & uint256(type(uint96).max)); - address expectedInstance = factory.getAddress(salt); - address instance = factory.createAccount{value: 123}(address(0xABCD), salt); - assertEq(instance.balance, 123); - assertEq(factory.createAccount{value: 456}(address(0xABCD), salt), instance); - assertEq(factory.createAccount(address(0xABCD), salt), instance); - assertEq(instance.balance, 123 + 456); - assertEq(expectedInstance, instance); - } - - function testCreateAccountRepeatedDeployment(uint256) public { - address owner = _randomNonZeroAddress(); - bytes32 salt = - bytes32((_random() & uint256(type(uint96).max)) | (uint256(uint160(owner)) << 96)); - address expectedInstance = factory.getAddress(salt); - address notOwner = _randomNonZeroAddress(); - while (owner == notOwner) notOwner = _randomNonZeroAddress(); - vm.expectRevert(LibClone.SaltDoesNotStartWith.selector); - factory.createAccount{value: 123}(notOwner, salt); - address instance = factory.createAccount{value: 123}(owner, salt); - assertEq(instance.balance, 123); - vm.expectRevert(LibClone.SaltDoesNotStartWith.selector); - factory.createAccount{value: 123}(notOwner, salt); - assertEq(factory.createAccount{value: 456}(owner, salt), instance); - assertEq(factory.createAccount(owner, salt), instance); - assertEq(instance.balance, 123 + 456); - assertEq(expectedInstance, instance); - } - - function _checkImplementationSlot(address proxy, address implementation_) internal { - bytes32 slot = bytes32(uint256(keccak256("eip1967.proxy.implementation")) - 1); - assertEq(vm.load(proxy, slot), bytes32(uint256(uint160(implementation_)))); - } -} diff --git a/lib/solady/test/ERC4626.t.sol b/lib/solady/test/ERC4626.t.sol deleted file mode 100644 index c84a0cb..0000000 --- a/lib/solady/test/ERC4626.t.sol +++ /dev/null @@ -1,556 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; - -import {ERC20, MockERC20} from "./utils/mocks/MockERC20.sol"; -import {ERC4626, MockERC4626} from "./utils/mocks/MockERC4626.sol"; -import {SafeTransferLib} from "../src/utils/SafeTransferLib.sol"; -import {FixedPointMathLib} from "../src/utils/FixedPointMathLib.sol"; - -contract ERC4626Test is SoladyTest { - MockERC20 underlying; - MockERC4626 vault; - - event Deposit(address indexed by, address indexed owner, uint256 assets, uint256 shares); - - event Withdraw( - address indexed by, - address indexed to, - address indexed owner, - uint256 assets, - uint256 shares - ); - - function setUp() public { - underlying = new MockERC20("Mock Token", "TKN", 18); - vault = new MockERC4626(address(underlying), "Mock Token Vault", "vwTKN", false, 0); - } - - function _replaceWithVirtualSharesVault(uint8 decimalsOffset) internal { - vault = new MockERC4626(address(underlying), "VSV", "VSVTKN", true, decimalsOffset); - } - - function _replaceWithVirtualSharesVault() internal { - _replaceWithVirtualSharesVault(0); - } - - function testDifferentialFullMulDiv(uint256 x, uint256 y, uint256 d) public { - d = type(uint256).max - d % 4; - (bool success0,) = address(this).call( - abi.encodeWithSignature("fullMulDivChecked(uint256,uint256,uint256)", x, y, d) - ); - (bool success1,) = address(this).call( - abi.encodeWithSignature("fullMulDivUnchecked(uint256,uint256,uint256)", x, y, d) - ); - if (d == type(uint256).max) { - assertFalse(success0); - assertFalse(success1); - } - assertEq(success0, success1); - } - - function fullMulDivChecked(uint256 x, uint256 y, uint256 d) public pure { - FixedPointMathLib.fullMulDiv(x, y, d + 1); - } - - function fullMulDivUnchecked(uint256 x, uint256 y, uint256 d) public pure { - unchecked { - FixedPointMathLib.fullMulDiv(x, y, d + 1); - } - } - - function testMetadata() public { - assertEq(vault.name(), "Mock Token Vault"); - assertEq(vault.symbol(), "vwTKN"); - assertEq(vault.decimals(), 18); - } - - function testUseVirtualShares() public { - assertEq(vault.useVirtualShares(), false); - _replaceWithVirtualSharesVault(); - assertEq(vault.useVirtualShares(), true); - assertEq(vault.decimals(), 18); - _replaceWithVirtualSharesVault(1); - assertEq(vault.decimals(), 19); - } - - function testTryGetAssetDecimals() public { - unchecked { - for (uint256 i = 0; i < 5; ++i) { - _testTryGetAssetDecimals(uint8(i)); - } - for (uint256 i = 125; i < 130; ++i) { - _testTryGetAssetDecimals(uint8(i)); - } - for (uint256 i = 250; i < 256; ++i) { - _testTryGetAssetDecimals(uint8(i)); - } - } - vault = new MockERC4626(address(this), "", "", false, 0); - assertEq(vault.decimals(), 18); - } - - function _testTryGetAssetDecimals(uint8 i) internal { - underlying = new MockERC20("", "", i); - assertEq(underlying.decimals(), i); - vault = new MockERC4626(address(underlying), "", "", false, 0); - assertEq(vault.decimals(), i); - } - - function testSingleDepositWithdraw(uint128 amount) public { - if (amount == 0) amount = 1; - - uint256 aliceUnderlyingAmount = amount; - - address alice = address(0xABCD); - - underlying.mint(alice, aliceUnderlyingAmount); - - vm.prank(alice); - underlying.approve(address(vault), aliceUnderlyingAmount); - assertEq(underlying.allowance(alice, address(vault)), aliceUnderlyingAmount); - - uint256 alicePreDepositBal = underlying.balanceOf(alice); - - vm.prank(alice); - uint256 aliceShareAmount = vault.deposit(aliceUnderlyingAmount, alice); - - assertEq(vault.afterDepositHookCalledCounter(), 1); - - // Expect exchange rate to be 1:1 on initial deposit. - assertEq(aliceUnderlyingAmount, aliceShareAmount); - assertEq(vault.previewWithdraw(aliceShareAmount), aliceUnderlyingAmount); - assertEq(vault.previewDeposit(aliceUnderlyingAmount), aliceShareAmount); - assertEq(vault.totalSupply(), aliceShareAmount); - assertEq(vault.totalAssets(), aliceUnderlyingAmount); - assertEq(vault.balanceOf(alice), aliceShareAmount); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), aliceUnderlyingAmount); - assertEq(underlying.balanceOf(alice), alicePreDepositBal - aliceUnderlyingAmount); - - vm.prank(alice); - vault.withdraw(aliceUnderlyingAmount, alice, alice); - - assertEq(vault.beforeWithdrawHookCalledCounter(), 1); - - assertEq(vault.totalAssets(), 0); - assertEq(vault.balanceOf(alice), 0); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), 0); - assertEq(underlying.balanceOf(alice), alicePreDepositBal); - } - - function testSingleMintRedeem(uint128 amount) public { - if (amount == 0) amount = 1; - - uint256 aliceShareAmount = amount; - - address alice = address(0xABCD); - - underlying.mint(alice, aliceShareAmount); - - vm.prank(alice); - underlying.approve(address(vault), aliceShareAmount); - assertEq(underlying.allowance(alice, address(vault)), aliceShareAmount); - - uint256 alicePreDepositBal = underlying.balanceOf(alice); - - vm.prank(alice); - uint256 aliceUnderlyingAmount = vault.mint(aliceShareAmount, alice); - - assertEq(vault.afterDepositHookCalledCounter(), 1); - - // Expect exchange rate to be 1:1 on initial mint. - assertEq(aliceShareAmount, aliceUnderlyingAmount); - assertEq(vault.previewWithdraw(aliceShareAmount), aliceUnderlyingAmount); - assertEq(vault.previewDeposit(aliceUnderlyingAmount), aliceShareAmount); - assertEq(vault.totalSupply(), aliceShareAmount); - assertEq(vault.totalAssets(), aliceUnderlyingAmount); - assertEq(vault.balanceOf(alice), aliceUnderlyingAmount); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), aliceUnderlyingAmount); - assertEq(underlying.balanceOf(alice), alicePreDepositBal - aliceUnderlyingAmount); - - vm.prank(alice); - vault.redeem(aliceShareAmount, alice, alice); - - assertEq(vault.beforeWithdrawHookCalledCounter(), 1); - - assertEq(vault.totalAssets(), 0); - assertEq(vault.balanceOf(alice), 0); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), 0); - assertEq(underlying.balanceOf(alice), alicePreDepositBal); - } - - function testMultipleMintDepositRedeemWithdraw() public { - _testMultipleMintDepositRedeemWithdraw(0); - } - - function testVirtualSharesMultipleMintDepositRedeemWithdraw() public { - _replaceWithVirtualSharesVault(); - _testMultipleMintDepositRedeemWithdraw(1); - } - - struct _TestTemps { - address alice; - address bob; - uint256 mutationUnderlyingAmount; - uint256 aliceUnderlyingAmount; - uint256 aliceShareAmount; - uint256 bobShareAmount; - uint256 bobUnderlyingAmount; - uint256 preMutationShareBal; - uint256 preMutationBal; - } - - function _testMultipleMintDepositRedeemWithdraw(uint256 slippage) public { - // Scenario: - // A = Alice, B = Bob - // ________________________________________________________ - // | Vault shares | A share | A assets | B share | B assets | - // |========================================================| - // | 1. Alice mints 2000 shares (costs 2000 tokens) | - // |--------------|---------|----------|---------|----------| - // | 2000 | 2000 | 2000 | 0 | 0 | - // |--------------|---------|----------|---------|----------| - // | 2. Bob deposits 4000 tokens (mints 4000 shares) | - // |--------------|---------|----------|---------|----------| - // | 6000 | 2000 | 2000 | 4000 | 4000 | - // |--------------|---------|----------|---------|----------| - // | 3. Vault mutates by +3000 tokens... | - // | (simulated yield returned from strategy)... | - // |--------------|---------|----------|---------|----------| - // | 6000 | 2000 | 3000 | 4000 | 6000 | - // |--------------|---------|----------|---------|----------| - // | 4. Alice deposits 2000 tokens (mints 1333 shares) | - // |--------------|---------|----------|---------|----------| - // | 7333 | 3333 | 4999 | 4000 | 6000 | - // |--------------|---------|----------|---------|----------| - // | 5. Bob mints 2000 shares (costs 3001 assets) | - // | NOTE: Bob's assets spent got rounded up | - // | NOTE: Alice's vault assets got rounded up | - // |--------------|---------|----------|---------|----------| - // | 9333 | 3333 | 5000 | 6000 | 9000 | - // |--------------|---------|----------|---------|----------| - // | 6. Vault mutates by +3000 tokens... | - // | (simulated yield returned from strategy) | - // | NOTE: Vault holds 17001 tokens, but sum of | - // | assetsOf() is 17000. | - // |--------------|---------|----------|---------|----------| - // | 9333 | 3333 | 6071 | 6000 | 10929 | - // |--------------|---------|----------|---------|----------| - // | 7. Alice redeem 1333 shares (2428 assets) | - // |--------------|---------|----------|---------|----------| - // | 8000 | 2000 | 3643 | 6000 | 10929 | - // |--------------|---------|----------|---------|----------| - // | 8. Bob withdraws 2928 assets (1608 shares) | - // |--------------|---------|----------|---------|----------| - // | 6392 | 2000 | 3643 | 4392 | 8000 | - // |--------------|---------|----------|---------|----------| - // | 9. Alice withdraws 3643 assets (2000 shares) | - // | NOTE: Bob's assets have been rounded back up | - // |--------------|---------|----------|---------|----------| - // | 4392 | 0 | 0 | 4392 | 8001 | - // |--------------|---------|----------|---------|----------| - // | 10. Bob redeem 4392 shares (8001 tokens) | - // |--------------|---------|----------|---------|----------| - // | 0 | 0 | 0 | 0 | 0 | - // |______________|_________|__________|_________|__________| - - _TestTemps memory t; - t.alice = address(0x9988776655443322110000112233445566778899); - t.bob = address(0x1122334455667788990000998877665544332211); - - t.mutationUnderlyingAmount = 3000; - - underlying.mint(t.alice, 4000); - - vm.prank(t.alice); - underlying.approve(address(vault), 4000); - - assertEq(underlying.allowance(t.alice, address(vault)), 4000); - - underlying.mint(t.bob, 7001); - - vm.prank(t.bob); - underlying.approve(address(vault), 7001); - - assertEq(underlying.allowance(t.bob, address(vault)), 7001); - - // 1. Alice mints 2000 shares (costs 2000 tokens) - vm.prank(t.alice); - vm.expectEmit(true, true, true, true); - emit Deposit(t.alice, t.alice, 2000, 2000); - t.aliceUnderlyingAmount = vault.mint(2000, t.alice); - - t.aliceShareAmount = vault.previewDeposit(t.aliceUnderlyingAmount); - assertEq(vault.afterDepositHookCalledCounter(), 1); - - // Expect to have received the requested mint amount. - assertEq(t.aliceShareAmount, 2000); - assertEq(vault.balanceOf(t.alice), t.aliceShareAmount); - assertEq(vault.convertToAssets(vault.balanceOf(t.alice)), t.aliceUnderlyingAmount); - assertEq(vault.convertToShares(t.aliceUnderlyingAmount), vault.balanceOf(t.alice)); - - // Expect a 1:1 ratio before mutation. - assertEq(t.aliceUnderlyingAmount, 2000); - - // Sanity check. - assertEq(vault.totalSupply(), t.aliceShareAmount); - assertEq(vault.totalAssets(), t.aliceUnderlyingAmount); - - // 2. Bob deposits 4000 tokens (mints 4000 shares) - vm.prank(t.bob); - vm.expectEmit(true, true, true, true); - emit Deposit(t.bob, t.bob, 4000, 4000); - t.bobShareAmount = vault.deposit(4000, t.bob); - t.bobUnderlyingAmount = vault.previewWithdraw(t.bobShareAmount); - assertEq(vault.afterDepositHookCalledCounter(), 2); - - // Expect to have received the requested underlying amount. - assertEq(t.bobUnderlyingAmount, 4000); - assertEq(vault.balanceOf(t.bob), t.bobShareAmount); - assertEq(vault.convertToAssets(vault.balanceOf(t.bob)), t.bobUnderlyingAmount); - assertEq(vault.convertToShares(t.bobUnderlyingAmount), vault.balanceOf(t.bob)); - - // Expect a 1:1 ratio before mutation. - assertEq(t.bobShareAmount, t.bobUnderlyingAmount); - - // Sanity check. - t.preMutationShareBal = t.aliceShareAmount + t.bobShareAmount; - t.preMutationBal = t.aliceUnderlyingAmount + t.bobUnderlyingAmount; - assertEq(vault.totalSupply(), t.preMutationShareBal); - assertEq(vault.totalAssets(), t.preMutationBal); - assertEq(vault.totalSupply(), 6000); - assertEq(vault.totalAssets(), 6000); - - // 3. Vault mutates by +3000 tokens... | - // (simulated yield returned from strategy)... - // The Vault now contains more tokens than deposited which causes the exchange rate to change. - // Alice share is 33.33% of the Vault, Bob 66.66% of the Vault. - // Alice's share count stays the same but the underlying amount changes from 2000 to 3000. - // Bob's share count stays the same but the underlying amount changes from 4000 to 6000. - underlying.mint(address(vault), t.mutationUnderlyingAmount); - assertEq(vault.totalSupply(), t.preMutationShareBal); - assertEq(vault.totalAssets(), t.preMutationBal + t.mutationUnderlyingAmount); - assertEq(vault.balanceOf(t.alice), t.aliceShareAmount); - assertEq( - vault.convertToAssets(vault.balanceOf(t.alice)), - t.aliceUnderlyingAmount + (t.mutationUnderlyingAmount / 3) * 1 - slippage - ); - assertEq(vault.balanceOf(t.bob), t.bobShareAmount); - assertEq( - vault.convertToAssets(vault.balanceOf(t.bob)), - t.bobUnderlyingAmount + (t.mutationUnderlyingAmount / 3) * 2 - slippage - ); - - // 4. Alice deposits 2000 tokens (mints 1333 shares) - vm.prank(t.alice); - vault.deposit(2000, t.alice); - - assertEq(vault.totalSupply(), 7333); - assertEq(vault.balanceOf(t.alice), 3333); - assertEq(vault.convertToAssets(vault.balanceOf(t.alice)), 4999); - assertEq(vault.balanceOf(t.bob), 4000); - assertEq(vault.convertToAssets(vault.balanceOf(t.bob)), 6000); - - // 5. Bob mints 2000 shares (costs 3001 assets) - // NOTE: Bob's assets spent got rounded up - // NOTE: Alices's vault assets got rounded up - vm.prank(t.bob); - vault.mint(2000, t.bob); - - assertEq(vault.totalSupply(), 9333); - assertEq(vault.balanceOf(t.alice), 3333); - assertEq(vault.convertToAssets(vault.balanceOf(t.alice)), 5000 - slippage); - assertEq(vault.balanceOf(t.bob), 6000); - assertEq(vault.convertToAssets(vault.balanceOf(t.bob)), 9000); - - // Sanity checks: - // Alice and t.bob should have spent all their tokens now - assertEq(underlying.balanceOf(t.alice), 0); - assertEq(underlying.balanceOf(t.bob) - slippage, 0); - // Assets in vault: 4k (t.alice) + 7k (t.bob) + 3k (yield) + 1 (round up) - assertEq(vault.totalAssets(), 14001 - slippage); - - // 6. Vault mutates by +3000 tokens - // NOTE: Vault holds 17001 tokens, but sum of assetsOf() is 17000. - underlying.mint(address(vault), t.mutationUnderlyingAmount); - assertEq(vault.convertToAssets(vault.balanceOf(t.alice)), 6071 - slippage); - assertEq(vault.convertToAssets(vault.balanceOf(t.bob)), 10929 - slippage); - assertEq(vault.totalSupply(), 9333); - assertEq(vault.totalAssets(), 17001 - slippage); - - // 7. Alice redeem 1333 shares (2428 assets) - vm.prank(t.alice); - vault.redeem(1333, t.alice, t.alice); - - assertEq(underlying.balanceOf(t.alice), 2428 - slippage); - assertEq(vault.totalSupply(), 8000); - assertEq(vault.totalAssets(), 14573); - assertEq(vault.balanceOf(t.alice), 2000); - assertEq(vault.convertToAssets(vault.balanceOf(t.alice)), 3643); - assertEq(vault.balanceOf(t.bob), 6000); - assertEq(vault.convertToAssets(vault.balanceOf(t.bob)), 10929); - - // 8. Bob withdraws 2929 assets (1608 shares) - vm.prank(t.bob); - vault.withdraw(2929, t.bob, t.bob); - - assertEq(underlying.balanceOf(t.bob) - slippage, 2929); - assertEq(vault.totalSupply(), 6392); - assertEq(vault.totalAssets(), 11644); - assertEq(vault.balanceOf(t.alice), 2000); - assertEq(vault.convertToAssets(vault.balanceOf(t.alice)), 3643); - assertEq(vault.balanceOf(t.bob), 4392); - assertEq(vault.convertToAssets(vault.balanceOf(t.bob)), 8000); - - // 9. Alice withdraws 3643 assets (2000 shares) - // NOTE: Bob's assets have been rounded back up - vm.prank(t.alice); - vm.expectEmit(true, true, true, true); - emit Withdraw(t.alice, t.alice, t.alice, 3643, 2000); - vault.withdraw(3643, t.alice, t.alice); - assertEq(underlying.balanceOf(t.alice), 6071 - slippage); - assertEq(vault.totalSupply(), 4392); - assertEq(vault.totalAssets(), 8001); - assertEq(vault.balanceOf(t.alice), 0); - assertEq(vault.convertToAssets(vault.balanceOf(t.alice)), 0); - assertEq(vault.balanceOf(t.bob), 4392); - assertEq(vault.convertToAssets(vault.balanceOf(t.bob)), 8001 - slippage); - - // 10. Bob redeem 4392 shares (8001 tokens) - vm.prank(t.bob); - vm.expectEmit(true, true, true, true); - emit Withdraw(t.bob, t.bob, t.bob, 8001 - slippage, 4392); - vault.redeem(4392, t.bob, t.bob); - assertEq(underlying.balanceOf(t.bob), 10930); - assertEq(vault.totalSupply(), 0); - assertEq(vault.totalAssets() - slippage, 0); - assertEq(vault.balanceOf(t.alice), 0); - assertEq(vault.convertToAssets(vault.balanceOf(t.alice)), 0); - assertEq(vault.balanceOf(t.bob), 0); - assertEq(vault.convertToAssets(vault.balanceOf(t.bob)), 0); - - // Sanity check - assertEq(underlying.balanceOf(address(vault)) - slippage, 0); - } - - function testDepositWithNotEnoughApprovalReverts() public { - underlying.mint(address(this), 0.5e18); - underlying.approve(address(vault), 0.5e18); - assertEq(underlying.allowance(address(this), address(vault)), 0.5e18); - - vm.expectRevert(SafeTransferLib.TransferFromFailed.selector); - vault.deposit(1e18, address(this)); - } - - function testWithdrawWithNotEnoughUnderlyingAmountReverts() public { - underlying.mint(address(this), 0.5e18); - underlying.approve(address(vault), 0.5e18); - - vault.deposit(0.5e18, address(this)); - - vm.expectRevert(ERC4626.WithdrawMoreThanMax.selector); - vault.withdraw(1e18, address(this), address(this)); - } - - function testRedeemWithNotEnoughShareAmountReverts() public { - underlying.mint(address(this), 0.5e18); - underlying.approve(address(vault), 0.5e18); - - vault.deposit(0.5e18, address(this)); - - vm.expectRevert(ERC4626.RedeemMoreThanMax.selector); - vault.redeem(1e18, address(this), address(this)); - } - - function testWithdrawWithNoUnderlyingAmountReverts() public { - vm.expectRevert(ERC4626.WithdrawMoreThanMax.selector); - vault.withdraw(1e18, address(this), address(this)); - } - - function testRedeemWithNoShareAmountReverts() public { - vm.expectRevert(ERC4626.RedeemMoreThanMax.selector); - vault.redeem(1e18, address(this), address(this)); - } - - function testDepositWithNoApprovalReverts() public { - vm.expectRevert(SafeTransferLib.TransferFromFailed.selector); - vault.deposit(1e18, address(this)); - } - - function testMintWithNoApprovalReverts() public { - vm.expectRevert(SafeTransferLib.TransferFromFailed.selector); - vault.mint(1e18, address(this)); - } - - function testMintZero() public { - vault.mint(0, address(this)); - - assertEq(vault.balanceOf(address(this)), 0); - assertEq(vault.convertToAssets(vault.balanceOf(address(this))), 0); - assertEq(vault.totalSupply(), 0); - assertEq(vault.totalAssets(), 0); - } - - function testWithdrawZero() public { - vault.withdraw(0, address(this), address(this)); - - assertEq(vault.balanceOf(address(this)), 0); - assertEq(vault.convertToAssets(vault.balanceOf(address(this))), 0); - assertEq(vault.totalSupply(), 0); - assertEq(vault.totalAssets(), 0); - } - - function testVaultInteractionsForSomeoneElse() public { - // init 2 users with a 1e18 balance - address alice = address(0xABCD); - address bob = address(0xDCBA); - underlying.mint(alice, 1e18); - underlying.mint(bob, 1e18); - - vm.prank(alice); - underlying.approve(address(vault), 1e18); - - vm.prank(bob); - underlying.approve(address(vault), 1e18); - - // alice deposits 1e18 for bob - vm.prank(alice); - vm.expectEmit(true, true, true, true); - emit Deposit(alice, bob, 1e18, 1e18); - vault.deposit(1e18, bob); - - assertEq(vault.balanceOf(alice), 0); - assertEq(vault.balanceOf(bob), 1e18); - assertEq(underlying.balanceOf(alice), 0); - - // bob mint 1e18 for alice - vm.prank(bob); - vm.expectEmit(true, true, true, true); - emit Deposit(bob, alice, 1e18, 1e18); - vault.mint(1e18, alice); - assertEq(vault.balanceOf(alice), 1e18); - assertEq(vault.balanceOf(bob), 1e18); - assertEq(underlying.balanceOf(bob), 0); - - // alice redeem 1e18 for bob - vm.prank(alice); - vm.expectEmit(true, true, true, true); - emit Withdraw(alice, bob, alice, 1e18, 1e18); - vault.redeem(1e18, bob, alice); - - assertEq(vault.balanceOf(alice), 0); - assertEq(vault.balanceOf(bob), 1e18); - assertEq(underlying.balanceOf(bob), 1e18); - - // bob withdraw 1e18 for alice - vm.prank(bob); - vm.expectEmit(true, true, true, true); - emit Withdraw(bob, alice, bob, 1e18, 1e18); - vault.withdraw(1e18, alice, bob); - - assertEq(vault.balanceOf(alice), 0); - assertEq(vault.balanceOf(bob), 0); - assertEq(underlying.balanceOf(alice), 1e18); - } -} diff --git a/lib/solady/test/ERC6551.t.sol b/lib/solady/test/ERC6551.t.sol deleted file mode 100644 index 5173008..0000000 --- a/lib/solady/test/ERC6551.t.sol +++ /dev/null @@ -1,415 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {SignatureCheckerLib} from "../src/utils/SignatureCheckerLib.sol"; -import {ERC6551Proxy} from "../src/accounts/ERC6551Proxy.sol"; -import {ERC6551, MockERC6551, MockERC6551V2} from "./utils/mocks/MockERC6551.sol"; -import {MockERC6551Registry} from "./utils/mocks/MockERC6551Registry.sol"; -import {MockERC721} from "./utils/mocks/MockERC721.sol"; -import {MockERC1155} from "./utils/mocks/MockERC1155.sol"; -import {LibZip} from "../src/utils/LibZip.sol"; -import {LibClone} from "../src/utils/LibClone.sol"; -import {LibString} from "../src/utils/LibString.sol"; - -contract Target { - error TargetError(bytes data); - - bytes32 public datahash; - - bytes public data; - - function setData(bytes memory data_) public payable returns (bytes memory) { - data = data_; - datahash = keccak256(data_); - return data_; - } - - function revertWithTargetError(bytes memory data_) public payable { - revert TargetError(data_); - } -} - -contract ERC6551Test is SoladyTest { - MockERC6551Registry internal _registry; - - address internal _erc6551; - - address internal _erc721; - - address internal _proxy; - - // By right, this should be the keccak256 of some long-ass string: - // (e.g. `keccak256("Parent(bytes32 childHash,Mail child)Mail(Person from,Person to,string contents)Person(string name,address wallet)")`). - // But I'm lazy and will use something randomish here. - bytes32 internal constant _PARENT_TYPEHASH = - 0xd61db970ec8a2edc5f9fd31d876abe01b785909acb16dcd4baaf3b434b4c439b; - - // By right, this should be a proper domain separator, but I'm lazy. - bytes32 internal constant _DOMAIN_SEP_B = - 0xa1a044077d7677adbbfa892ded5390979b33993e0e2a457e3f974bbcda53821b; - - bytes32 internal constant _ERC1967_IMPLEMENTATION_SLOT = - 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc; - - struct _TestTemps { - address owner; - uint256 chainId; - uint256 tokenId; - bytes32 salt; - MockERC6551 account; - address signer; - uint256 privateKey; - uint8 v; - bytes32 r; - bytes32 s; - } - - function setUp() public { - _registry = new MockERC6551Registry(); - _erc6551 = address(new MockERC6551()); - _erc721 = address(new MockERC721()); - _proxy = address(new ERC6551Proxy(_erc6551)); - } - - function _testTempsMint(address owner) internal returns (uint256 tokenId) { - while (true) { - tokenId = _random() % 8 == 0 ? _random() % 32 : _random(); - (bool success,) = - _erc721.call(abi.encodeWithSignature("mint(address,uint256)", owner, tokenId)); - if (success) return tokenId; - } - } - - function _testTemps() internal returns (_TestTemps memory t) { - t.owner = _randomNonZeroAddress(); - t.tokenId = _testTempsMint(t.owner); - t.chainId = block.chainid; - t.salt = bytes32(_random()); - address account = _registry.createAccount(_proxy, t.salt, t.chainId, _erc721, t.tokenId); - t.account = MockERC6551(payable(account)); - } - - function testDeployERC6551Proxy() public { - console.log(LibString.toHexString(address(new ERC6551Proxy(_erc6551)).code)); - } - - function testInitializeERC6551ProxyImplementation() public { - address account = address(_testTemps().account); - (bool success,) = account.call(""); - assertTrue(success); - bytes32 implementationSlotValue = bytes32(uint256(uint160(_erc6551))); - assertEq(vm.load(account, _ERC1967_IMPLEMENTATION_SLOT), implementationSlotValue); - } - - function testDeployERC6551(uint256) public { - _TestTemps memory t = _testTemps(); - (uint256 chainId, address tokenContract, uint256 tokenId) = t.account.token(); - assertEq(chainId, t.chainId); - assertEq(tokenContract, _erc721); - assertEq(tokenId, t.tokenId); - address owner = t.account.owner(); - assertEq(owner, t.owner); - if (_random() % 8 == 0) { - vm.prank(owner); - address newOnwer = _randomNonZeroAddress(); - MockERC721(_erc721).transferFrom(owner, newOnwer, t.tokenId); - assertEq(t.account.owner(), newOnwer); - } - } - - function testOnERC721ReceivedCycles() public { - unchecked { - uint256 n = 8; - _TestTemps[] memory t = new _TestTemps[](n); - for (uint256 i; i != n; ++i) { - t[i] = _testTemps(); - if (i != 0) { - vm.prank(t[i].owner); - MockERC721(_erc721).safeTransferFrom( - t[i].owner, address(t[i - 1].account), t[i].tokenId - ); - t[i].owner = address(t[i - 1].account); - } - } - for (uint256 i; i != n; ++i) { - for (uint256 j = i; j != n; ++j) { - vm.prank(t[i].owner); - vm.expectRevert(ERC6551.SelfOwnDetected.selector); - MockERC721(_erc721).safeTransferFrom( - t[i].owner, address(t[j].account), t[i].tokenId - ); - } - } - - _TestTemps memory u = _testTemps(); - vm.prank(u.owner); - MockERC721(_erc721).safeTransferFrom(u.owner, address(t[n - 1].account), u.tokenId); - } - } - - function testOnERC721ReceivedCyclesWithDifferentChainIds(uint256) public { - _TestTemps[] memory t = new _TestTemps[](3); - unchecked { - for (uint256 i; i != 3; ++i) { - vm.chainId(i); - t[i] = _testTemps(); - if (i != 0) { - vm.prank(t[i].owner); - MockERC721(_erc721).safeTransferFrom( - t[i].owner, address(t[i - 1].account), t[i].tokenId - ); - t[i].owner = address(t[i - 1].account); - } - } - } - unchecked { - vm.chainId(_random() % 3); - uint256 i = _random() % 3; - uint256 j = _random() % 3; - while (j == i) j = _random() % 3; - vm.prank(t[i].owner); - MockERC721(_erc721).safeTransferFrom(t[i].owner, address(t[j].account), t[i].tokenId); - } - } - - function testOnERC721Received() public { - _TestTemps memory t = _testTemps(); - address alice = _randomNonZeroAddress(); - MockERC721 erc721 = new MockERC721(); - erc721.mint(alice, 1); - vm.prank(alice); - if (alice != address(t.account)) { - erc721.safeTransferFrom(alice, address(t.account), 1); - } - } - - function testOnERC1155Received() public { - _TestTemps memory t = _testTemps(); - address alice = _randomNonZeroAddress(); - MockERC1155 erc1155 = new MockERC1155(); - erc1155.mint(alice, 1, 1, ""); - vm.prank(alice); - erc1155.safeTransferFrom(alice, address(t.account), 1, 1, ""); - } - - function testOnERC1155BatchReceived() public { - _TestTemps memory t = _testTemps(); - address alice = _randomNonZeroAddress(); - MockERC1155 erc1155 = new MockERC1155(); - erc1155.mint(alice, 1, 1, ""); - uint256[] memory ids = new uint256[](1); - ids[0] = 1; - uint256[] memory amts = new uint256[](1); - amts[0] = 1; - vm.prank(alice); - erc1155.safeBatchTransferFrom(alice, address(t.account), ids, amts, ""); - } - - function testExecute() public { - _TestTemps memory t = _testTemps(); - vm.deal(address(t.account), 1 ether); - - address target = address(new Target()); - bytes memory data = _randomBytes(111); - - assertEq(t.account.state(), 0); - - vm.prank(t.owner); - t.account.execute(target, 123, abi.encodeWithSignature("setData(bytes)", data), 0); - assertEq(Target(target).datahash(), keccak256(data)); - assertEq(target.balance, 123); - - vm.prank(_randomNonZeroAddress()); - vm.expectRevert(ERC6551.Unauthorized.selector); - t.account.execute(target, 123, abi.encodeWithSignature("setData(bytes)", data), 0); - - vm.prank(t.owner); - vm.expectRevert(abi.encodeWithSignature("TargetError(bytes)", data)); - t.account.execute( - target, 123, abi.encodeWithSignature("revertWithTargetError(bytes)", data), 0 - ); - - vm.prank(t.owner); - vm.expectRevert(ERC6551.OperationNotSupported.selector); - t.account.execute( - target, 123, abi.encodeWithSignature("revertWithTargetError(bytes)", data), 1 - ); - - assertEq(t.account.state(), 1); - } - - function testExecuteBatch() public { - _TestTemps memory t = _testTemps(); - vm.deal(address(t.account), 1 ether); - - ERC6551.Call[] memory calls = new ERC6551.Call[](2); - calls[0].target = address(new Target()); - calls[1].target = address(new Target()); - calls[0].value = 123; - calls[1].value = 456; - calls[0].data = abi.encodeWithSignature("setData(bytes)", _randomBytes(111)); - calls[1].data = abi.encodeWithSignature("setData(bytes)", _randomBytes(222)); - - assertEq(t.account.state(), 0); - - vm.prank(t.owner); - t.account.executeBatch(calls, 0); - assertEq(Target(calls[0].target).datahash(), keccak256(_randomBytes(111))); - assertEq(Target(calls[1].target).datahash(), keccak256(_randomBytes(222))); - assertEq(calls[0].target.balance, 123); - assertEq(calls[1].target.balance, 456); - - calls[1].data = abi.encodeWithSignature("revertWithTargetError(bytes)", _randomBytes(111)); - vm.expectRevert(abi.encodeWithSignature("TargetError(bytes)", _randomBytes(111))); - vm.prank(t.owner); - t.account.executeBatch(calls, 0); - - vm.prank(t.owner); - vm.expectRevert(ERC6551.OperationNotSupported.selector); - t.account.executeBatch(calls, 1); - - assertEq(t.account.state(), 1); - } - - function testExecuteBatch(uint256 r) public { - _TestTemps memory t = _testTemps(); - vm.deal(address(t.account), 1 ether); - - assertEq(t.account.state(), 0); - - unchecked { - uint256 n = r & 3; - ERC6551.Call[] memory calls = new ERC6551.Call[](n); - - for (uint256 i; i != n; ++i) { - uint256 v = _random() & 0xff; - calls[i].target = address(new Target()); - calls[i].value = v; - calls[i].data = abi.encodeWithSignature("setData(bytes)", _randomBytes(v)); - } - - bytes[] memory results; - if (_random() & 1 == 0) { - vm.prank(t.owner); - results = t.account.executeBatch(_random(), calls, 0); - } else { - vm.prank(t.owner); - results = t.account.executeBatch(calls, 0); - } - - assertEq(results.length, n); - for (uint256 i; i != n; ++i) { - uint256 v = calls[i].value; - assertEq(Target(calls[i].target).datahash(), keccak256(_randomBytes(v))); - assertEq(calls[i].target.balance, v); - assertEq(abi.decode(results[i], (bytes)), _randomBytes(v)); - } - } - - assertEq(t.account.state(), 1); - } - - function testUpgrade() public { - _TestTemps memory t = _testTemps(); - address anotherImplementation = address(new MockERC6551V2()); - vm.expectRevert(ERC6551.Unauthorized.selector); - t.account.upgradeToAndCall(anotherImplementation, bytes("")); - assertEq(t.account.state(), 0); - assertEq(t.account.mockId(), "1"); - - vm.prank(t.owner); - t.account.upgradeToAndCall(anotherImplementation, bytes("")); - assertEq(t.account.state(), 1); - assertEq(t.account.mockId(), "2"); - - vm.prank(t.owner); - t.account.upgradeToAndCall(_erc6551, bytes("")); - assertEq(t.account.state(), 2); - assertEq(t.account.mockId(), "1"); - } - - function testSupportsInterface() public { - _TestTemps memory t = _testTemps(); - assertTrue(t.account.supportsInterface(0x01ffc9a7)); - assertTrue(t.account.supportsInterface(0x6faff5f1)); - assertTrue(t.account.supportsInterface(0x74420f4c)); - assertFalse(t.account.supportsInterface(0x00000001)); - } - - function testCdFallback() public { - _TestTemps memory t = _testTemps(); - vm.deal(t.owner, 1 ether); - - ERC6551.Call[] memory calls = new ERC6551.Call[](2); - calls[0].target = address(new Target()); - calls[1].target = address(new Target()); - calls[0].value = 123; - calls[1].value = 456; - calls[0].data = abi.encodeWithSignature("setData(bytes)", _randomBytes(111)); - calls[1].data = abi.encodeWithSignature("setData(bytes)", _randomBytes(222)); - - bytes memory data = LibZip.cdCompress( - abi.encodeWithSignature("executeBatch((address,uint256,bytes)[],uint8)", calls, 0) - ); - vm.prank(t.owner); - (bool success,) = address(t.account).call{value: 1 ether}(data); - assertTrue(success); - assertEq(Target(calls[0].target).datahash(), keccak256(_randomBytes(111))); - assertEq(Target(calls[1].target).datahash(), keccak256(_randomBytes(222))); - assertEq(calls[0].target.balance, 123); - assertEq(calls[1].target.balance, 456); - } - - function testIsValidSignature() public { - _TestTemps memory t = _testTemps(); - (t.signer, t.privateKey) = _randomSigner(); - (t.v, t.r, t.s) = - vm.sign(t.privateKey, _toERC1271Hash(address(t.account), keccak256("123"))); - - vm.prank(t.owner); - MockERC721(_erc721).safeTransferFrom(t.owner, t.signer, t.tokenId); - - bytes32 hash = keccak256("123"); - bytes memory signature = - abi.encodePacked(t.r, t.s, t.v, _PARENT_TYPEHASH, _DOMAIN_SEP_B, hash); - // Success returns `0x1626ba7e`. - assertEq(t.account.isValidSignature(_toChildHash(hash), signature), bytes4(0x1626ba7e)); - } - - function _toERC1271Hash(address account, bytes32 child) internal view returns (bytes32) { - bytes32 domainSeparator = keccak256( - abi.encode( - keccak256( - "EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)" - ), - keccak256("Milady"), - keccak256("1"), - block.chainid, - address(account) - ) - ); - bytes32 parentStructHash = - keccak256(abi.encode(_PARENT_TYPEHASH, _toChildHash(child), child)); - return keccak256(abi.encodePacked("\x19\x01", domainSeparator, parentStructHash)); - } - - function _toChildHash(bytes32 child) internal pure returns (bytes32) { - return keccak256(abi.encodePacked(hex"1901", _DOMAIN_SEP_B, child)); - } - - function _randomBytes(uint256 seed) internal pure returns (bytes memory result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, seed) - let r := keccak256(0x00, 0x20) - if lt(byte(2, r), 0x20) { - result := mload(0x40) - let n := and(r, 0x7f) - mstore(result, n) - codecopy(add(result, 0x20), byte(1, r), add(n, 0x40)) - mstore(0x40, add(add(result, 0x40), n)) - } - } - } -} diff --git a/lib/solady/test/ERC6909.t.sol b/lib/solady/test/ERC6909.t.sol deleted file mode 100644 index df94587..0000000 --- a/lib/solady/test/ERC6909.t.sol +++ /dev/null @@ -1,535 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; - -import {ERC6909, MockERC6909} from "./utils/mocks/MockERC6909.sol"; - -contract ERC6909Test is SoladyTest { - MockERC6909 token; - - event Transfer( - address by, address indexed from, address indexed to, uint256 indexed id, uint256 amount - ); - - event OperatorSet(address indexed owner, address indexed spender, bool approved); - - event Approval( - address indexed owner, address indexed spender, uint256 indexed id, uint256 amount - ); - - function setUp() public { - token = new MockERC6909(); - } - - function testMetadata() public { - assertEq(token.name(), "Solady Token"); - assertEq(token.symbol(), "ST"); - } - - function testMint() public { - vm.expectEmit(true, true, true, true); - emit Transfer(address(this), address(0), address(0xBEEF), 1, 1e18); - - token.mint(address(0xBEEF), 1, 1e18); - assertEq(token.balanceOf(address(0xBEEF), 1), 1e18); - } - - function testDecimals() public { - assertEq(token.decimals(1), 18); - } - - function testBurn() public { - token.mint(address(0xBEEF), 1, 1e18); - - vm.expectEmit(true, true, true, true); - emit Transfer(address(this), address(0xBEEF), address(0), 1, 0.9e18); - token.burn(address(0xBEEF), 1, 0.9e18); - - assertEq(token.balanceOf(address(0xBEEF), 1), 0.1e18); - } - - function testApprove() public { - vm.expectEmit(true, true, true, true); - emit Approval(address(this), address(0xBEEF), 1, 1e18); - assertTrue(token.approve(address(0xBEEF), 1, 1e18)); - - assertEq(token.allowance(address(this), address(0xBEEF), 1), 1e18); - } - - function testTransfer() public { - token.mint(address(this), 1, 1e18); - - assertEq(token.balanceOf(address(this), 1), 1e18); - - vm.expectEmit(true, true, true, true); - emit Transfer(address(this), address(this), address(0xBEEF), 1, 1e18); - assertTrue(token.transfer(address(0xBEEF), 1, 1e18)); - assertEq(token.balanceOf(address(this), 1), 0); - assertEq(token.balanceOf(address(0xBEEF), 1), 1e18); - } - - function testTransferFrom() public { - address from = address(0xABCD); - - token.mint(from, 1, 1e18); - - _approve(from, address(this), 1, 1e18); - - vm.expectEmit(true, true, true, true); - emit Transfer(address(this), from, address(0xBEEF), 1, 1e18); - assertTrue(token.transferFrom(from, address(0xBEEF), 1, 1e18)); - - assertEq(token.allowance(from, address(this), 1), 0); - - assertEq(token.balanceOf(from, 1), 0); - assertEq(token.balanceOf(address(0xBEEF), 1), 1e18); - } - - function testInfiniteApproveTransferFrom() public { - address from = address(0xABCD); - - token.mint(from, 1, 1e18); - - _approve(from, address(this), 1, type(uint256).max); - - vm.expectEmit(true, true, true, true); - emit Transfer(address(this), from, address(0xBEEF), 1, 1e18); - assertTrue(token.transferFrom(from, address(0xBEEF), 1, 1e18)); - - assertEq(token.allowance(from, address(this), 1), type(uint256).max); - - assertEq(token.balanceOf(from, 1), 0); - assertEq(token.balanceOf(address(0xBEEF), 1), 1e18); - } - - function testOperatorTransferFrom() public { - address from = address(0xABcD); - - token.mint(from, 1, 1e18); - - _setOperator(from, address(this), true); - - vm.expectEmit(true, true, true, true); - emit Transfer(address(this), from, address(0xBEEF), 1, 1e18); - assertTrue(token.transferFrom(from, address(0xBEEF), 1, 1e18)); - - assertEq(token.balanceOf(from, 1), 0); - assertEq(token.balanceOf(address(0xBEEF), 1), 1e18); - } - - function testSetOperator() public { - assertEq(token.isOperator(address(this), address(0xBEEF)), false); - - vm.expectEmit(true, true, true, true); - emit OperatorSet(address(this), address(0xBEEF), true); - token.setOperator(address(0xBEEF), true); - assertEq(token.isOperator(address(this), address(0xBEEF)), true); - } - - function testTokenURI() public { - token.mint(address(0xBEEF), 1, 1e18); - assertEq(token.tokenURI(1), "http://solady.org/1"); - } - - function testMintOverMaxUintReverts() public { - token.mint(address(this), 1, type(uint256).max); - vm.expectRevert(ERC6909.BalanceOverflow.selector); - token.mint(address(this), 1, 1); - } - - function testTransferOverMaxUintReverts() public { - token.mint(address(this), 1, type(uint256).max); - token.transfer(address(0xBEEF), 1, type(uint256).max); - token.mint(address(this), 1, 1); - vm.expectRevert(ERC6909.BalanceOverflow.selector); - token.transfer(address(0xBEEF), 1, 1); - } - - function testTransferFromOverMaxUintReverts() public { - address from = address(0xABCD); - - _approve(from, address(this), 1, type(uint256).max); - - token.mint(from, 1, type(uint256).max); - token.transferFrom(from, address(0xBEEF), 1, type(uint256).max); - - token.mint(from, 1, 1); - vm.expectRevert(ERC6909.BalanceOverflow.selector); - token.transferFrom(from, address(0xBEEF), 1, 1); - } - - function testTransferInsufficientBalanceReverts() public { - token.mint(address(this), 1, 0.9e18); - _expectInsufficientBalanceRevert(); - token.transfer(address(0xBEEF), 1, 1e18); - } - - function testTransferFromInsufficientPermission() public { - address from = address(0xABCD); - - token.mint(from, 1, 1e18); - - _approve(from, address(this), 1, 0.9e18); - - _expectInsufficientPermissionRevert(); - token.transferFrom(from, address(0xBEEF), 1, 1e18); - } - - function testTransferFromInsufficientBalanceReverts() public { - address from = address(0xABCD); - - token.mint(from, 1, 0.9e18); - - _approve(from, address(this), 1, 1e18); - - _expectInsufficientBalanceRevert(); - token.transferFrom(from, address(0xBEEF), 1, 1e18); - } - - function testMint(address to, uint256 id, uint256 amount) public { - vm.expectEmit(true, true, true, true); - emit Transfer(address(this), address(0), to, id, amount); - token.mint(to, id, amount); - - assertEq(token.balanceOf(to, id), amount); - } - - function testBurn(address from, uint256 id, uint256 mintAmount, uint256 burnAmount) public { - burnAmount = _bound(burnAmount, 0, mintAmount); - - token.mint(from, id, mintAmount); - vm.expectEmit(true, true, true, true); - emit Transfer(address(this), from, address(0), id, burnAmount); - token.burn(from, id, burnAmount); - - assertEq(token.balanceOf(from, id), mintAmount - burnAmount); - } - - function testApprove(address to, uint256 id, uint256 amount) public { - _approve(address(this), to, id, amount); - } - - function testTransfer(address to, uint256 id, uint256 amount) public { - token.mint(address(this), id, amount); - - vm.expectEmit(true, true, true, true); - emit Transfer(address(this), address(this), to, id, amount); - assertTrue(token.transfer(to, id, amount)); - - if (address(this) == to) { - assertEq(token.balanceOf(address(this), id), amount); - } else { - assertEq(token.balanceOf(address(this), id), 0); - assertEq(token.balanceOf(to, id), amount); - } - } - - function testTransferFrom( - address spender, - address from, - address to, - uint256 id, - uint256 approval, - uint256 amount - ) public { - amount = _bound(amount, 0, approval); - - token.mint(from, id, amount); - assertEq(token.balanceOf(from, id), amount); - - _approve(from, spender, id, approval); - - vm.expectEmit(true, true, true, true); - emit Transfer(spender, from, to, id, amount); - vm.prank(spender); - assertTrue(token.transferFrom(from, to, id, amount)); - - if (approval == type(uint256).max) { - assertEq(token.allowance(from, spender, id), approval); - } else { - assertEq(token.allowance(from, spender, id), approval - amount); - } - - if (from == to) { - assertEq(token.balanceOf(from, id), amount); - } else { - assertEq(token.balanceOf(from, id), 0); - assertEq(token.balanceOf(to, id), amount); - } - } - - function testSetOperator(address owner, address spender, bool approved) public { - _setOperator(owner, spender, approved); - } - - function testMintOverMaxUintReverts(address to, uint256 id, uint256 amount0, uint256 amount1) - public - { - amount0 = _bound(amount0, 1, type(uint256).max); - amount1 = _bound(amount1, type(uint256).max - amount0 + 1, type(uint256).max); - token.mint(to, id, amount0); - - vm.expectRevert(ERC6909.BalanceOverflow.selector); - token.mint(to, id, amount1); - } - - function testBurnInsufficientBalanceReverts( - address to, - uint256 mintAmount, - uint256 id, - uint256 burnAmount - ) public { - if (mintAmount == type(uint256).max) mintAmount--; - burnAmount = _bound(burnAmount, mintAmount + 1, type(uint256).max); - - token.mint(to, id, mintAmount); - - _expectInsufficientBalanceRevert(); - token.burn(to, id, burnAmount); - } - - function testTransferOverMaxUintReverts( - address to, - uint256 id, - uint256 amount0, - uint256 amount1 - ) public { - amount0 = _bound(amount0, 1, type(uint256).max); - amount1 = _bound(amount1, type(uint256).max - amount0 + 1, type(uint256).max); - - token.mint(address(this), id, amount0); - token.transfer(to, id, amount0); - token.mint(address(this), id, amount1); - - vm.expectRevert(ERC6909.BalanceOverflow.selector); - token.transfer(to, id, amount1); - } - - function testTransferInsufficientBalanceReverts( - address to, - uint256 id, - uint256 mintAmount, - uint256 sendAmount - ) public { - if (mintAmount == type(uint256).max) mintAmount--; - sendAmount = _bound(sendAmount, mintAmount + 1, type(uint256).max); - - token.mint(address(this), id, mintAmount); - - _expectInsufficientBalanceRevert(); - token.transfer(to, id, sendAmount); - } - - function testTransferFromOverMaxUintReverts( - address to, - uint256 id, - uint256 amount0, - uint256 amount1 - ) public { - amount0 = _bound(amount0, 1, type(uint256).max); - amount1 = _bound(amount1, type(uint256).max - amount0 + 1, type(uint256).max); - - address from = address(0xABCD); - - token.mint(from, id, amount0); - _approve(from, address(this), id, amount0); - - token.transferFrom(from, to, id, amount0); - - token.mint(from, id, amount1); - _approve(from, address(this), id, amount1); - - vm.expectRevert(ERC6909.BalanceOverflow.selector); - token.transferFrom(from, to, id, amount1); - } - - function testTransferFromInsufficientAllowanceReverts( - address to, - uint256 id, - uint256 approval, - uint256 amount - ) public { - if (approval == type(uint256).max) approval--; - amount = _bound(amount, approval + 1, type(uint256).max); - - address from = address(0xABCD); - - token.mint(from, amount, id); - - _approve(from, address(this), id, approval); - - _expectInsufficientPermissionRevert(); - token.transferFrom(from, to, id, amount); - } - - function testTransferFromInsufficientBalanceReverts( - address to, - uint256 id, - uint256 mintAmount, - uint256 sendAmount - ) public { - if (mintAmount == type(uint256).max) mintAmount--; - sendAmount = _bound(sendAmount, mintAmount + 1, type(uint256).max); - - address from = address(0xABCD); - - token.mint(from, id, mintAmount); - - _approve(from, address(this), id, sendAmount); - - _expectInsufficientBalanceRevert(); - token.transferFrom(from, to, id, sendAmount); - } - - function testTransferFromCallerIsNotOperator(address to, uint256 id, uint256 amount) public { - amount = _bound(amount, 1, type(uint256).max); - - address from = address(0xABCD); - - token.mint(from, id, amount); - - _expectInsufficientPermissionRevert(); - token.transferFrom(from, to, id, amount); - } - - struct _TestTemps { - uint256 id; - uint256 allowance; - bool isOperator; - uint256 balance; - uint256 amount; - address by; - address from; - address to; - bool success; - } - - function testDirectSetOperator() public { - _directSetOperator(address(1), address(2), true); - } - - function testDirectApprove() public { - _directApprove(address(1), address(2), 1, 123); - } - - function testDirectTransfer() public { - token.mint(address(2), 1, 1); - vm.prank(address(2)); - token.approve(address(1), 1, 1); - token.directTransferFrom(address(1), address(2), address(3), 1, 1); - } - - function testDirectFunctions(uint256) public { - _TestTemps memory t; - t.id = _random(); - t.by = _random() % 16 == 0 ? address(0) : _randomAddress(); - t.from = _randomAddress(); - t.to = _randomAddress(); - - for (uint256 q; q != 2; ++q) { - t.success = false; - t.allowance = _random(); - t.balance = _random(); - t.amount = _random(); - t.isOperator = _random() % 4 == 0; - t.id ^= 1; - - token.mint(t.from, t.id, t.balance); - if (_random() % 2 == 0) { - _directSetOperator(t.from, t.by, t.isOperator); - _directApprove(t.from, t.by, t.id, t.allowance); - } else { - _setOperator(t.from, t.by, t.isOperator); - _directApprove(t.from, t.by, t.id, t.allowance); - } - - if (t.balance >= t.amount) { - if (t.by == address(0) || t.isOperator || t.allowance >= t.amount) { - t.success = true; - } else { - _expectInsufficientPermissionRevert(); - } - } else { - if (t.by == address(0) || t.isOperator || t.allowance >= t.amount) { - _expectInsufficientBalanceRevert(); - } else { - _expectInsufficientPermissionRevert(); - } - } - - if (t.by == address(0) && _random() % 4 == 0) { - if (t.success) { - vm.expectEmit(true, true, true, true); - emit Transfer(t.from, t.from, t.to, t.id, t.amount); - } - vm.prank(t.from); - token.transfer(t.to, t.id, t.amount); - } else if (t.by != address(0) && _random() % 4 == 0) { - if (t.success) { - vm.expectEmit(true, true, true, true); - emit Transfer(t.by, t.from, t.to, t.id, t.amount); - } - vm.prank(t.by); - token.transferFrom(t.from, t.to, t.id, t.amount); - } else { - if (t.success) { - vm.expectEmit(true, true, true, true); - emit Transfer(t.by, t.from, t.to, t.id, t.amount); - } - token.directTransferFrom(t.by, t.from, t.to, t.id, t.amount); - } - - if (t.by == address(0) || t.isOperator || t.allowance == type(uint256).max) { - assertEq(token.allowance(t.from, t.by, t.id), t.allowance); - } - - if (t.success) { - if (t.to == t.from) { - assertEq(token.balanceOf(t.to, t.id), t.balance); - } else { - assertEq(token.balanceOf(t.from, t.id), t.balance - t.amount); - assertEq(token.balanceOf(t.to, t.id), t.amount); - } - } - } - } - - function _expectInsufficientBalanceRevert() internal { - vm.expectRevert(ERC6909.InsufficientBalance.selector); - } - - function _expectInsufficientPermissionRevert() internal { - vm.expectRevert(ERC6909.InsufficientPermission.selector); - } - - function _approve(address owner, address spender, uint256 id, uint256 amount) internal { - vm.prank(owner); - vm.expectEmit(true, true, true, true); - emit Approval(owner, spender, id, amount); - token.approve(spender, id, amount); - assertEq(token.allowance(owner, spender, id), amount); - } - - function _setOperator(address owner, address operator, bool approved) internal { - vm.prank(owner); - vm.expectEmit(true, true, true, true); - emit OperatorSet(owner, operator, approved); - token.directSetOperator(owner, operator, approved); - assertEq(token.isOperator(owner, operator), approved); - } - - function _directApprove(address owner, address spender, uint256 id, uint256 amount) internal { - vm.expectEmit(true, true, true, true); - emit Approval(owner, spender, id, amount); - token.directApprove(owner, spender, id, amount); - assertEq(token.allowance(owner, spender, id), amount); - } - - function _directSetOperator(address owner, address operator, bool approved) internal { - vm.expectEmit(true, true, true, true); - emit OperatorSet(owner, operator, approved); - token.directSetOperator(owner, operator, approved); - assertEq(token.isOperator(owner, operator), approved); - } -} diff --git a/lib/solady/test/ERC721.t.sol b/lib/solady/test/ERC721.t.sol deleted file mode 100644 index 5591b2b..0000000 --- a/lib/solady/test/ERC721.t.sol +++ /dev/null @@ -1,976 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; - -import {ERC721, MockERC721} from "./utils/mocks/MockERC721.sol"; - -abstract contract ERC721TokenReceiver { - function onERC721Received(address, address, uint256, bytes calldata) - external - virtual - returns (bytes4) - { - return ERC721TokenReceiver.onERC721Received.selector; - } -} - -contract ERC721Recipient is ERC721TokenReceiver { - address public operator; - address public from; - uint256 public id; - bytes public data; - - function onERC721Received(address _operator, address _from, uint256 _id, bytes calldata _data) - public - virtual - override - returns (bytes4) - { - operator = _operator; - from = _from; - id = _id; - data = _data; - - return ERC721TokenReceiver.onERC721Received.selector; - } -} - -contract RevertingERC721Recipient is ERC721TokenReceiver { - function onERC721Received(address, address, uint256, bytes calldata) - public - virtual - override - returns (bytes4) - { - revert(string(abi.encodePacked(ERC721TokenReceiver.onERC721Received.selector))); - } -} - -contract WrongReturnDataERC721Recipient is ERC721TokenReceiver { - function onERC721Received(address, address, uint256, bytes calldata) - public - virtual - override - returns (bytes4) - { - return 0xCAFEBEEF; - } -} - -contract NonERC721Recipient {} - -contract MockERC721WithHooks is MockERC721 { - uint256 public beforeCounter; - uint256 public afterCounter; - - function _beforeTokenTransfer(address, address, uint256) internal virtual override { - beforeCounter++; - } - - function _afterTokenTransfer(address, address, uint256) internal virtual override { - afterCounter++; - } -} - -contract ERC721HooksTest is SoladyTest, ERC721TokenReceiver { - uint256 public expectedBeforeCounter; - uint256 public expectedAfterCounter; - uint256 public ticker; - - function _checkCounters() internal view { - require( - expectedBeforeCounter == MockERC721WithHooks(msg.sender).beforeCounter(), - "Before counter mismatch." - ); - require( - expectedAfterCounter == MockERC721WithHooks(msg.sender).afterCounter(), - "After counter mismatch." - ); - } - - function onERC721Received(address, address, uint256, bytes calldata) - external - virtual - override - returns (bytes4) - { - _checkCounters(); - return ERC721TokenReceiver.onERC721Received.selector; - } - - function _testHooks(MockERC721WithHooks token) internal { - address from = _randomNonZeroAddress(); - uint256 tokenId = - uint256(keccak256(abi.encode(expectedBeforeCounter, expectedAfterCounter))); - expectedBeforeCounter++; - expectedAfterCounter++; - token.mint(address(this), tokenId); - - expectedBeforeCounter++; - expectedAfterCounter++; - token.transferFrom(address(this), from, tokenId); - - expectedBeforeCounter++; - expectedAfterCounter++; - uint256 r = ticker < 4 ? ticker : _random() % 4; - vm.prank(from); - if (r == 0) { - token.safeTransferFrom(from, address(this), tokenId); - } else if (r == 1) { - token.safeTransferFrom(from, address(this), tokenId, ""); - } else if (r == 2) { - token.directSafeTransferFrom(from, address(this), tokenId); - } else if (r == 3) { - token.directSafeTransferFrom(from, address(this), tokenId, ""); - } else { - revert(); - } - } - - function testERC721Hooks() public { - MockERC721WithHooks token = new MockERC721WithHooks(); - - for (uint256 i; i < 32; ++i) { - _testHooks(token); - } - } -} - -contract ERC721Test is SoladyTest { - MockERC721 token; - - uint256 private constant _ERC721_MASTER_SLOT_SEED = 0x7d8825530a5a2e7a << 192; - - event Transfer(address indexed from, address indexed to, uint256 indexed id); - - event Approval(address indexed owner, address indexed approved, uint256 indexed id); - - event ApprovalForAll(address indexed owner, address indexed operator, bool approved); - - function setUp() public { - token = new MockERC721(); - } - - function _expectMintEvent(address to, uint256 id) internal { - _expectTransferEvent(address(0), to, id); - } - - function _expectBurnEvent(address from, uint256 id) internal { - _expectTransferEvent(from, address(0), id); - } - - function _expectTransferEvent(address from, address to, uint256 id) internal { - vm.expectEmit(true, true, true, true); - emit Transfer(from, to, id); - } - - function _expectApprovalEvent(address owner, address approved, uint256 id) internal { - vm.expectEmit(true, true, true, true); - emit Approval(owner, approved, id); - } - - function _expectApprovalForAllEvent(address owner, address operator, bool approved) internal { - vm.expectEmit(true, true, true, true); - emit ApprovalForAll(owner, operator, approved); - } - - function _aux(address owner) internal pure returns (uint224 result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, owner) - result := shr(32, shl(32, keccak256(0x0c, 0x14))) - } - } - - function _extraData(uint256 id) internal pure returns (uint96 result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, id) - result := shr(160, shl(160, keccak256(0x00, 0x20))) - } - } - - function _transferFrom(address from, address to, uint256 id) internal { - if (_random() % 2 == 0) { - token.transferFrom(from, to, id); - } else { - token.directTransferFrom(from, to, id); - } - } - - function _safeTransferFrom(address from, address to, uint256 id) internal { - if (_random() % 2 == 0) { - token.safeTransferFrom(from, to, id); - } else { - token.directSafeTransferFrom(from, to, id); - } - } - - function _safeTransferFrom(address from, address to, uint256 id, bytes memory data) internal { - if (_random() % 2 == 0) { - token.safeTransferFrom(from, to, id, data); - } else { - token.directSafeTransferFrom(from, to, id, data); - } - } - - function _approve(address spender, uint256 id) internal { - if (_random() % 2 == 0) { - token.approve(spender, id); - } else { - token.directApprove(spender, id); - } - } - - function _setApprovalForAll(address operator, bool approved) internal { - if (_random() % 2 == 0) { - token.setApprovalForAll(operator, approved); - } else { - token.directSetApprovalForAll(operator, approved); - } - } - - function _ownerOf(uint256 id) internal returns (address) { - if (_random() % 2 == 0) { - return token.ownerOf(id); - } else { - return token.directOwnerOf(id); - } - } - - function _getApproved(uint256 id) internal returns (address) { - if (_random() % 2 == 0) { - return token.getApproved(id); - } else { - return token.directGetApproved(id); - } - } - - function _owners() internal returns (address a, address b) { - a = _randomNonZeroAddress(); - b = _randomNonZeroAddress(); - while (a == b) b = _randomNonZeroAddress(); - } - - function testSafetyOfCustomStorage(uint256 id0, uint256 id1) public { - bool safe; - while (id0 == id1) id1 = _random(); - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, id0) - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - let slot0 := add(id0, add(id0, keccak256(0x00, 0x20))) - let slot2 := add(1, slot0) - mstore(0x00, id1) - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - let slot1 := add(id1, add(id1, keccak256(0x00, 0x20))) - let slot3 := add(1, slot1) - safe := 1 - if eq(slot0, slot1) { safe := 0 } - if eq(slot0, slot2) { safe := 0 } - if eq(slot0, slot3) { safe := 0 } - if eq(slot1, slot2) { safe := 0 } - if eq(slot1, slot3) { safe := 0 } - if eq(slot2, slot3) { safe := 0 } - } - require(safe, "Custom storage not safe"); - } - - function testAuthorizedEquivalence(address by, bool isOwnerOrOperator, bool isApprovedAccount) - public - { - bool a = true; - bool b = true; - /// @solidity memory-safe-assembly - assembly { - if by { if iszero(isOwnerOrOperator) { a := isApprovedAccount } } - if iszero(or(iszero(by), isOwnerOrOperator)) { b := isApprovedAccount } - } - assertEq(a, b); - } - - function testCannotExceedMaxBalance() public { - bytes32 balanceSlot; - (address owner0, address owner1) = _owners(); - - /// @solidity memory-safe-assembly - assembly { - mstore(0x1c, _ERC721_MASTER_SLOT_SEED) - mstore(0x00, owner0) - balanceSlot := keccak256(0x0c, 0x1c) - } - - vm.store(address(token), balanceSlot, bytes32(uint256(0xfffffffe))); - token.setAux(owner0, type(uint224).max); - assertEq(token.balanceOf(owner0), 0xfffffffe); - assertEq(token.getAux(owner0), type(uint224).max); - token.mint(owner0, 0); - assertEq(token.balanceOf(owner0), 0xffffffff); - - vm.expectRevert(ERC721.AccountBalanceOverflow.selector); - token.mint(owner0, 1); - - token.uncheckedBurn(0); - assertEq(token.balanceOf(owner0), 0xfffffffe); - - token.mint(owner1, 0); - vm.prank(owner1); - _transferFrom(owner1, owner0, 0); - - token.mint(owner1, 1); - vm.expectRevert(ERC721.AccountBalanceOverflow.selector); - vm.prank(owner1); - _transferFrom(owner1, owner0, 1); - assertEq(token.getAux(owner0), type(uint224).max); - } - - function testMint(uint256 id) public { - address owner = _randomNonZeroAddress(); - - _expectMintEvent(owner, id); - token.mint(owner, id); - - assertEq(token.balanceOf(owner), 1); - assertEq(_ownerOf(id), owner); - } - - function testBurn(uint256 id) public { - address owner = _randomNonZeroAddress(); - - _expectMintEvent(owner, id); - token.mint(owner, id); - - if (_random() % 2 == 0) { - _expectBurnEvent(owner, id); - token.uncheckedBurn(id); - } else { - vm.expectRevert(ERC721.NotOwnerNorApproved.selector); - token.burn(id); - uint256 r = _random() % 3; - if (r == 0) { - vm.prank(owner); - _transferFrom(owner, address(this), id); - _expectBurnEvent(address(this), id); - token.burn(id); - } - if (r == 1) { - vm.prank(owner); - _setApprovalForAll(address(this), true); - _expectBurnEvent(owner, id); - token.burn(id); - } - if (r == 2) { - vm.prank(owner); - _approve(address(this), id); - _expectBurnEvent(owner, id); - token.burn(id); - } - } - - assertEq(token.balanceOf(owner), 0); - - vm.expectRevert(ERC721.TokenDoesNotExist.selector); - _ownerOf(id); - } - - function testTransferFrom() public { - address owner = _randomNonZeroAddress(); - token.mint(owner, 0); - vm.prank(owner); - token.transferFrom(owner, address(this), 0); - } - - function testEverything(uint256) public { - address[2] memory owners; - uint256[][2] memory tokens; - - unchecked { - (owners[0], owners[1]) = _owners(); - for (uint256 j; j != 2; ++j) { - tokens[j] = new uint256[](_random() % 3); - } - - for (uint256 j; j != 2; ++j) { - token.setAux(owners[j], _aux(owners[j])); - for (uint256 i; i != tokens[j].length;) { - uint256 id = _random(); - if (!token.exists(id)) { - tokens[j][i++] = id; - _expectMintEvent(owners[j], id); - token.mint(owners[j], id); - token.setExtraData(id, _extraData(id)); - } - } - } - for (uint256 j; j != 2; ++j) { - assertEq(token.balanceOf(owners[j]), tokens[j].length); - for (uint256 i; i != tokens[j].length; ++i) { - vm.prank(owners[j]); - _expectApprovalEvent(owners[j], address(this), tokens[j][i]); - _approve(address(this), tokens[j][i]); - } - } - for (uint256 j; j != 2; ++j) { - for (uint256 i; i != tokens[j].length; ++i) { - assertEq(_getApproved(tokens[j][i]), address(this)); - uint256 fromBalanceBefore = token.balanceOf(owners[j]); - uint256 toBalanceBefore = token.balanceOf(owners[j ^ 1]); - _expectTransferEvent(owners[j], owners[j ^ 1], tokens[j][i]); - _transferFrom(owners[j], owners[j ^ 1], tokens[j][i]); - assertEq(token.balanceOf(owners[j]), fromBalanceBefore - 1); - assertEq(token.balanceOf(owners[j ^ 1]), toBalanceBefore + 1); - assertEq(_getApproved(tokens[j][i]), address(0)); - } - } - for (uint256 j; j != 2; ++j) { - for (uint256 i; i != tokens[j].length; ++i) { - assertEq(_ownerOf(tokens[j][i]), owners[j ^ 1]); - assertEq(token.getExtraData(tokens[j][i]), _extraData(tokens[j][i])); - } - } - if (_random() % 2 == 0) { - for (uint256 j; j != 2; ++j) { - for (uint256 i; i != tokens[j].length; ++i) { - vm.expectRevert(ERC721.NotOwnerNorApproved.selector); - _transferFrom(owners[j ^ 1], owners[j], tokens[j][i]); - vm.prank(owners[j ^ 1]); - _expectApprovalEvent(owners[j ^ 1], address(this), tokens[j][i]); - _approve(address(this), tokens[j][i]); - _expectTransferEvent(owners[j ^ 1], owners[j], tokens[j][i]); - _transferFrom(owners[j ^ 1], owners[j], tokens[j][i]); - } - } - } else { - for (uint256 j; j != 2; ++j) { - vm.prank(owners[j ^ 1]); - _expectApprovalForAllEvent(owners[j ^ 1], address(this), true); - token.setApprovalForAll(address(this), true); - for (uint256 i; i != tokens[j].length; ++i) { - _expectTransferEvent(owners[j ^ 1], owners[j], tokens[j][i]); - _transferFrom(owners[j ^ 1], owners[j], tokens[j][i]); - } - } - } - for (uint256 j; j != 2; ++j) { - assertEq(token.getAux(owners[j]), _aux(owners[j])); - for (uint256 i; i != tokens[j].length; ++i) { - assertEq(_ownerOf(tokens[j][i]), owners[j]); - assertEq(token.getExtraData(tokens[j][i]), _extraData(tokens[j][i])); - } - } - for (uint256 j; j != 2; ++j) { - for (uint256 i; i != tokens[j].length; ++i) { - token.uncheckedBurn(tokens[j][i]); - } - } - for (uint256 j; j != 2; ++j) { - assertEq(token.balanceOf(owners[j]), 0); - for (uint256 i; i != tokens[j].length; ++i) { - assertEq(token.getExtraData(tokens[j][i]), _extraData(tokens[j][i])); - } - } - } - } - - function testIsApprovedOrOwner(uint256 id) public { - (address owner0, address owner1) = _owners(); - - vm.expectRevert(ERC721.TokenDoesNotExist.selector); - token.isApprovedOrOwner(owner0, id); - - token.mint(owner0, id); - assertEq(token.isApprovedOrOwner(owner0, id), true); - - vm.prank(owner0); - _transferFrom(owner0, owner1, id); - assertEq(token.isApprovedOrOwner(owner0, id), false); - - vm.prank(owner1); - _setApprovalForAll(owner0, true); - assertEq(token.isApprovedOrOwner(owner0, id), true); - - vm.prank(owner1); - _setApprovalForAll(owner0, false); - assertEq(token.isApprovedOrOwner(owner0, id), false); - - vm.prank(owner1); - _approve(owner0, id); - assertEq(token.isApprovedOrOwner(owner0, id), true); - } - - function testExtraData(uint256 id) public { - (address owner0, address owner1) = _owners(); - - bool setExtraData = _random() % 2 == 0; - uint96 extraData = uint96(_bound(_random(), 0, type(uint96).max)); - if (setExtraData) { - token.setExtraData(id, extraData); - } - _expectMintEvent(owner0, id); - token.mint(owner0, id); - if (setExtraData) { - assertEq(token.getExtraData(id), extraData); - } else { - assertEq(token.getExtraData(id), 0); - } - - vm.prank(owner0); - _expectTransferEvent(owner0, owner1, id); - _transferFrom(owner0, owner1, id); - if (setExtraData) { - assertEq(token.getExtraData(id), extraData); - } else { - assertEq(token.getExtraData(id), 0); - } - assertEq(_ownerOf(id), owner1); - - if (_random() % 2 == 0) { - extraData = uint96(_bound(_random(), 0, type(uint96).max)); - token.setExtraData(id, extraData); - setExtraData = true; - } - - _expectBurnEvent(owner1, id); - token.uncheckedBurn(id); - if (setExtraData) { - assertEq(token.getExtraData(id), extraData); - } else { - assertEq(token.getExtraData(id), 0); - } - vm.expectRevert(ERC721.TokenDoesNotExist.selector); - _ownerOf(id); - } - - function testExtraData2(uint256 id0, uint256 id1) public { - while (id0 == id1) id1 = _random(); - token.setExtraData(id0, _extraData(id0)); - token.setExtraData(id1, _extraData(id1)); - assertEq(token.getExtraData(id0), _extraData(id0)); - assertEq(token.getExtraData(id1), _extraData(id1)); - } - - function testAux(uint256) public { - (address owner0, address owner1) = _owners(); - - bool setAux = _random() % 2 == 0; - if (setAux) { - token.setAux(owner0, _aux(owner0)); - token.setAux(owner1, _aux(owner1)); - } - - for (uint256 i; i < 2; ++i) { - _expectMintEvent(owner0, i * 2 + 0); - token.mint(owner0, i * 2 + 0); - assertEq(token.balanceOf(owner0), i + 1); - - _expectMintEvent(owner1, i * 2 + 1); - token.mint(owner1, i * 2 + 1); - assertEq(token.balanceOf(owner1), i + 1); - - if (setAux) { - assertEq(token.getAux(owner0), _aux(owner0)); - assertEq(token.getAux(owner1), _aux(owner1)); - } else { - assertEq(token.getAux(owner0), 0); - assertEq(token.getAux(owner1), 0); - } - } - - for (uint256 i; i < 2; ++i) { - _expectBurnEvent(owner0, i * 2 + 0); - token.uncheckedBurn(i * 2 + 0); - assertEq(token.balanceOf(owner0), 1 - i); - - _expectBurnEvent(owner1, i * 2 + 1); - token.uncheckedBurn(i * 2 + 1); - assertEq(token.balanceOf(owner1), 1 - i); - - if (setAux) { - assertEq(token.getAux(owner0), _aux(owner0)); - assertEq(token.getAux(owner1), _aux(owner1)); - } else { - assertEq(token.getAux(owner0), 0); - assertEq(token.getAux(owner1), 0); - } - } - } - - function testApprove(uint256 id) public { - (address spender,) = _randomSigner(); - - token.mint(address(this), id); - - _expectApprovalEvent(address(this), spender, id); - _approve(spender, id); - assertEq(_getApproved(id), spender); - } - - function testApproveBurn(uint256 id) public { - (address spender,) = _randomSigner(); - - token.mint(address(this), id); - - _approve(spender, id); - - token.uncheckedBurn(id); - - assertEq(token.balanceOf(address(this)), 0); - - vm.expectRevert(ERC721.TokenDoesNotExist.selector); - _getApproved(id); - - vm.expectRevert(ERC721.TokenDoesNotExist.selector); - _ownerOf(id); - } - - function testApproveAll(uint256) public { - (address operator,) = _randomSigner(); - bool approved = _random() % 2 == 0; - _expectApprovalForAllEvent(address(this), operator, approved); - _setApprovalForAll(operator, approved); - assertEq(token.isApprovedForAll(address(this), operator), approved); - } - - function testTransferFrom(uint256 id) public { - (address from, address to) = _owners(); - - token.mint(from, id); - - if (_random() % 2 == 0) { - uint256 r = _random() % 3; - if (r == 0) { - vm.prank(from); - _approve(address(this), id); - _expectTransferEvent(from, to, id); - _transferFrom(from, to, id); - } - if (r == 1) { - vm.prank(from); - _setApprovalForAll(address(this), true); - _expectTransferEvent(from, to, id); - _transferFrom(from, to, id); - } - if (r == 2) { - vm.prank(from); - _expectTransferEvent(from, address(this), id); - _transferFrom(from, address(this), id); - _expectTransferEvent(address(this), to, id); - _transferFrom(address(this), to, id); - } - } else { - (address temp,) = _randomSigner(); - while (temp == from || temp == to) (temp,) = _randomSigner(); - if (_random() % 2 == 0) { - _expectTransferEvent(from, temp, id); - token.uncheckedTransferFrom(from, temp, id); - } else { - vm.prank(from); - _expectTransferEvent(from, temp, id); - _transferFrom(from, temp, id); - } - _expectTransferEvent(temp, to, id); - token.uncheckedTransferFrom(temp, to, id); - } - - assertEq(_getApproved(id), address(0)); - assertEq(_ownerOf(id), to); - assertEq(token.balanceOf(to), 1); - assertEq(token.balanceOf(from), 0); - } - - function testTransferFromSelf(uint256 id) public { - (address to,) = _randomSigner(); - - token.mint(address(this), id); - - _transferFrom(address(this), to, id); - - assertEq(_getApproved(id), address(0)); - assertEq(_ownerOf(id), to); - assertEq(token.balanceOf(to), 1); - assertEq(token.balanceOf(address(this)), 0); - } - - function testTransferFromApproveAll(uint256 id) public { - (address from, address to) = _owners(); - - token.mint(from, id); - - vm.prank(from); - _setApprovalForAll(address(this), true); - - _transferFrom(from, to, id); - - assertEq(_getApproved(id), address(0)); - assertEq(_ownerOf(id), to); - assertEq(token.balanceOf(to), 1); - assertEq(token.balanceOf(from), 0); - } - - function testSafeTransferFromToEOA(uint256 id) public { - (address from, address to) = _owners(); - - token.mint(from, id); - - vm.prank(from); - _setApprovalForAll(address(this), true); - - _safeTransferFrom(from, to, id); - - assertEq(_getApproved(id), address(0)); - assertEq(_ownerOf(id), to); - assertEq(token.balanceOf(to), 1); - assertEq(token.balanceOf(from), 0); - } - - function testSafeTransferFromToERC721Recipient(uint256 id) public { - (address from,) = _randomSigner(); - - ERC721Recipient recipient = new ERC721Recipient(); - - token.mint(from, id); - - vm.prank(from); - _setApprovalForAll(address(this), true); - - _safeTransferFrom(from, address(recipient), id); - - assertEq(_getApproved(id), address(0)); - assertEq(_ownerOf(id), address(recipient)); - assertEq(token.balanceOf(address(recipient)), 1); - assertEq(token.balanceOf(from), 0); - - assertEq(recipient.operator(), address(this)); - assertEq(recipient.from(), from); - assertEq(recipient.id(), id); - assertEq(recipient.data(), ""); - } - - function testSafeTransferFromToERC721RecipientWithData(uint256 id, bytes memory data) public { - (address from,) = _randomSigner(); - - ERC721Recipient recipient = new ERC721Recipient(); - - token.mint(from, id); - - vm.prank(from); - _setApprovalForAll(address(this), true); - - _safeTransferFrom(from, address(recipient), id, data); - - assertEq(recipient.data(), data); - assertEq(recipient.id(), id); - assertEq(recipient.operator(), address(this)); - assertEq(recipient.from(), from); - - assertEq(_getApproved(id), address(0)); - assertEq(_ownerOf(id), address(recipient)); - assertEq(token.balanceOf(address(recipient)), 1); - assertEq(token.balanceOf(from), 0); - } - - function testSafeMintToEOA(uint256 id) public { - (address to,) = _randomSigner(); - - token.safeMint(to, id); - - assertEq(_ownerOf(id), address(to)); - assertEq(token.balanceOf(address(to)), 1); - } - - function testSafeMintToERC721Recipient(uint256 id) public { - ERC721Recipient to = new ERC721Recipient(); - - token.safeMint(address(to), id); - - assertEq(_ownerOf(id), address(to)); - assertEq(token.balanceOf(address(to)), 1); - - assertEq(to.operator(), address(this)); - assertEq(to.from(), address(0)); - assertEq(to.id(), id); - assertEq(to.data(), ""); - } - - function testSafeMintToERC721RecipientWithData(uint256 id, bytes memory data) public { - ERC721Recipient to = new ERC721Recipient(); - - token.safeMint(address(to), id, data); - - assertEq(_ownerOf(id), address(to)); - assertEq(token.balanceOf(address(to)), 1); - - assertEq(to.operator(), address(this)); - assertEq(to.from(), address(0)); - assertEq(to.id(), id); - assertEq(to.data(), data); - } - - function testMintToZeroReverts(uint256 id) public { - vm.expectRevert(ERC721.TransferToZeroAddress.selector); - token.mint(address(0), id); - } - - function testDoubleMintReverts(uint256 id) public { - (address to,) = _randomSigner(); - - token.mint(to, id); - vm.expectRevert(ERC721.TokenAlreadyExists.selector); - token.mint(to, id); - } - - function testBurnNonExistentReverts(uint256 id) public { - vm.expectRevert(ERC721.TokenDoesNotExist.selector); - token.uncheckedBurn(id); - } - - function testDoubleBurnReverts(uint256 id) public { - (address to,) = _randomSigner(); - - token.mint(to, id); - - token.uncheckedBurn(id); - vm.expectRevert(ERC721.TokenDoesNotExist.selector); - token.uncheckedBurn(id); - } - - function testApproveNonExistentReverts(uint256 id, address to) public { - vm.expectRevert(ERC721.TokenDoesNotExist.selector); - _approve(to, id); - } - - function testApproveUnauthorizedReverts(uint256 id) public { - (address owner, address to) = _owners(); - - token.mint(owner, id); - vm.expectRevert(ERC721.NotOwnerNorApproved.selector); - _approve(to, id); - } - - function testTransferFromNotExistentReverts(address from, address to, uint256 id) public { - vm.expectRevert(ERC721.TokenDoesNotExist.selector); - _transferFrom(from, to, id); - } - - function testTransferFromWrongFromReverts(address to, uint256 id) public { - (address owner, address from) = _owners(); - - token.mint(owner, id); - vm.expectRevert(ERC721.TransferFromIncorrectOwner.selector); - _transferFrom(from, to, id); - } - - function testTransferFromToZeroReverts(uint256 id) public { - token.mint(address(this), id); - - vm.expectRevert(ERC721.TransferToZeroAddress.selector); - _transferFrom(address(this), address(0), id); - } - - function testTransferFromNotOwner(uint256 id) public { - (address from, address to) = _owners(); - - token.mint(from, id); - - vm.expectRevert(ERC721.NotOwnerNorApproved.selector); - _transferFrom(from, to, id); - } - - function testSafeTransferFromToNonERC721RecipientReverts(uint256 id) public { - token.mint(address(this), id); - address to = address(new NonERC721Recipient()); - vm.expectRevert(ERC721.TransferToNonERC721ReceiverImplementer.selector); - _safeTransferFrom(address(this), address(to), id); - } - - function testSafeTransferFromToNonERC721RecipientWithDataReverts(uint256 id, bytes memory data) - public - { - token.mint(address(this), id); - address to = address(new NonERC721Recipient()); - vm.expectRevert(ERC721.TransferToNonERC721ReceiverImplementer.selector); - _safeTransferFrom(address(this), to, id, data); - } - - function testSafeTransferFromToRevertingERC721RecipientReverts(uint256 id) public { - token.mint(address(this), id); - address to = address(new RevertingERC721Recipient()); - vm.expectRevert(abi.encodePacked(ERC721TokenReceiver.onERC721Received.selector)); - _safeTransferFrom(address(this), to, id); - } - - function testSafeTransferFromToRevertingERC721RecipientWithDataReverts( - uint256 id, - bytes memory data - ) public { - token.mint(address(this), id); - address to = address(new RevertingERC721Recipient()); - vm.expectRevert(abi.encodePacked(ERC721TokenReceiver.onERC721Received.selector)); - _safeTransferFrom(address(this), to, id, data); - } - - function testSafeTransferFromToERC721RecipientWithWrongReturnDataReverts(uint256 id) public { - token.mint(address(this), id); - address to = address(new WrongReturnDataERC721Recipient()); - vm.expectRevert(ERC721.TransferToNonERC721ReceiverImplementer.selector); - _safeTransferFrom(address(this), to, id); - } - - function testSafeTransferFromToERC721RecipientWithWrongReturnDataWithDataReverts( - uint256 id, - bytes memory data - ) public { - token.mint(address(this), id); - address to = address(new WrongReturnDataERC721Recipient()); - vm.expectRevert(ERC721.TransferToNonERC721ReceiverImplementer.selector); - _safeTransferFrom(address(this), to, id, data); - } - - function testSafeMintToNonERC721RecipientReverts(uint256 id) public { - address to = address(new NonERC721Recipient()); - vm.expectRevert(ERC721.TransferToNonERC721ReceiverImplementer.selector); - token.safeMint(to, id); - } - - function testSafeMintToNonERC721RecipientWithDataReverts(uint256 id, bytes memory data) - public - { - address to = address(new NonERC721Recipient()); - vm.expectRevert(ERC721.TransferToNonERC721ReceiverImplementer.selector); - token.safeMint(to, id, data); - } - - function testSafeMintToRevertingERC721RecipientReverts(uint256 id) public { - address to = address(new RevertingERC721Recipient()); - vm.expectRevert(abi.encodePacked(ERC721TokenReceiver.onERC721Received.selector)); - token.safeMint(to, id); - } - - function testSafeMintToRevertingERC721RecipientWithDataReverts(uint256 id, bytes memory data) - public - { - address to = address(new RevertingERC721Recipient()); - vm.expectRevert(abi.encodePacked(ERC721TokenReceiver.onERC721Received.selector)); - token.safeMint(to, id, data); - } - - function testSafeMintToERC721RecipientWithWrongReturnData(uint256 id) public { - address to = address(new WrongReturnDataERC721Recipient()); - vm.expectRevert(ERC721.TransferToNonERC721ReceiverImplementer.selector); - token.safeMint(to, id); - } - - function testSafeMintToERC721RecipientWithWrongReturnDataWithData(uint256 id, bytes memory data) - public - { - address to = address(new WrongReturnDataERC721Recipient()); - vm.expectRevert(ERC721.TransferToNonERC721ReceiverImplementer.selector); - token.safeMint(to, id, data); - } - - function testOwnerOfNonExistent(uint256 id) public { - vm.expectRevert(ERC721.TokenDoesNotExist.selector); - _ownerOf(id); - } -} diff --git a/lib/solady/test/FixedPointMathLib.t.sol b/lib/solady/test/FixedPointMathLib.t.sol deleted file mode 100644 index 62f8bc3..0000000 --- a/lib/solady/test/FixedPointMathLib.t.sol +++ /dev/null @@ -1,1750 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {FixedPointMathLib} from "../src/utils/FixedPointMathLib.sol"; - -contract FixedPointMathLibTest is SoladyTest { - function testExpWad() public { - assertEq(FixedPointMathLib.expWad(-42139678854452767551), 0); - - assertEq(FixedPointMathLib.expWad(-3e18), 49787068367863942); - assertEq(FixedPointMathLib.expWad(-2e18), 135335283236612691); - assertEq(FixedPointMathLib.expWad(-1e18), 367879441171442321); - - assertEq(FixedPointMathLib.expWad(-0.5e18), 606530659712633423); - assertEq(FixedPointMathLib.expWad(-0.3e18), 740818220681717866); - - assertEq(FixedPointMathLib.expWad(0), 1000000000000000000); - - assertEq(FixedPointMathLib.expWad(0.3e18), 1349858807576003103); - assertEq(FixedPointMathLib.expWad(0.5e18), 1648721270700128146); - - assertEq(FixedPointMathLib.expWad(1e18), 2718281828459045235); - assertEq(FixedPointMathLib.expWad(2e18), 7389056098930650227); - assertEq(FixedPointMathLib.expWad(3e18), 20085536923187667741); - // True value: 20085536923187667740.92 - - assertEq(FixedPointMathLib.expWad(10e18), 220264657948067165169_80); - // True value: 22026465794806716516957.90 - // Relative error 9.987984547746668e-22 - - assertEq(FixedPointMathLib.expWad(50e18), 5184705528587072464_148529318587763226117); - // True value: 5184705528587072464_087453322933485384827.47 - // Relative error: 1.1780031733243328e-20 - - assertEq( - FixedPointMathLib.expWad(100e18), - 268811714181613544841_34666106240937146178367581647816351662017 - ); - // True value: 268811714181613544841_26255515800135873611118773741922415191608 - // Relative error: 3.128803544297531e-22 - - assertEq( - FixedPointMathLib.expWad(135305999368893231588), - 578960446186580976_50144101621524338577433870140581303254786265309376407432913 - ); - // True value: 578960446186580976_49816762928942336782129491980154662247847962410455084893091 - // Relative error: 5.653904247484822e-21 - } - - // Notes on lambertW0Wad: - // - // If you want to attempt finding a better approximation, look at - // https://github.com/recmo/experiment-solexp/blob/main/approximate_mpmath.ipynb - // I somehow can't get it to reproduce the approximation constants for `lnWad`. - // Let me know if you can get the code to reproduce the approximation constants for `lnWad`. - - event TestingLambertW0WadMonotonicallyIncreasing( - int256 a, int256 b, int256 w0a, int256 w0b, bool success, uint256 gasUsed - ); - - event LogUint(string name, uint256 value); - event LogInt(string name, int256 value); - - int256 internal constant _ONE_DIV_EXP = 367879441171442321; - int256 internal constant _LAMBERT_W0_MIN = -367879441171442321; - int256 internal constant _EXP = 2718281828459045235; - int256 internal constant _WAD = 10 ** 18; - - function testLambertW0WadKnownValues() public { - _checkLambertW0Wad(0, 0); - _checkLambertW0Wad(1, 1); - _checkLambertW0Wad(2, 2); - _checkLambertW0Wad(3, 2); - _checkLambertW0Wad(131071, 131070); - _checkLambertW0Wad(17179869183, 17179868887); - _checkLambertW0Wad(1000000000000000000, 567143290409783872); - _checkLambertW0Wad(-3678794411715, -3678807945318); - _checkLambertW0Wad(_LAMBERT_W0_MIN, -999999999741585709); - // These are exact values. - _checkLambertW0Wad(2 ** 255 - 1, 130435123404408416612); - _checkLambertW0Wad(2 ** 254 - 1, 129747263755102316133); - _checkLambertW0Wad(2 ** 253 - 1, 129059431996357330139); - _checkLambertW0Wad(2 ** 252 - 1, 128371628422812486425); - _checkLambertW0Wad(2 ** 251 - 1, 127683853333788079721); - _checkLambertW0Wad(2 ** 250 - 1, 126996107033385166927); - _checkLambertW0Wad(2 ** 249 - 1, 126308389830587715420); - _checkLambertW0Wad(2 ** 248 - 1, 125620702039367489656); - _checkLambertW0Wad(2 ** 247 - 1, 124933043978791764502); - _checkLambertW0Wad(2 ** 246 - 1, 124245415973133957088); - _checkLambertW0Wad(2 ** 245 - 1, 123557818351987272451); - _checkLambertW0Wad(2 ** 244 - 1, 122870251450381461880); - _checkLambertW0Wad(2 ** 243 - 1, 122182715608902796703); - _checkLambertW0Wad(2 ** 242 - 1, 121495211173817364188); - _checkLambertW0Wad(2 ** 241 - 1, 120807738497197796422); - _checkLambertW0Wad(2 ** 240 - 1, 120120297937053547320); - _checkLambertW0Wad(2 ** 239 - 1, 119432889857464837488); - _checkLambertW0Wad(2 ** 238 - 1, 118745514628720391363); - _checkLambertW0Wad(2 ** 237 - 1, 118058172627459096009); - _checkLambertW0Wad(2 ** 236 - 1, 117370864236815716134); - _checkLambertW0Wad(2 ** 235 - 1, 116683589846570805279); - _checkLambertW0Wad(2 ** 234 - 1, 115996349853304958814); - _checkLambertW0Wad(2 ** 233 - 1, 115309144660557560280); - _checkLambertW0Wad(2 ** 232 - 1, 114621974678990178815); - _checkLambertW0Wad(2 ** 231 - 1, 113934840326554781918); - _checkLambertW0Wad(2 ** 230 - 1, 113247742028666934564); - _checkLambertW0Wad(2 ** 229 - 1, 112560680218384162820); - _checkLambertW0Wad(2 ** 228 - 1, 111873655336589667598); - _checkLambertW0Wad(2 ** 227 - 1, 111186667832181581935); - _checkLambertW0Wad(2 ** 226 - 1, 110499718162267973459); - _checkLambertW0Wad(2 ** 225 - 1, 109812806792367802251); - _checkLambertW0Wad(2 ** 224 - 1, 109125934196618053331); - _checkLambertW0Wad(2 ** 223 - 1, 108439100857987272488); - _checkLambertW0Wad(2 ** 222 - 1, 107752307268495744067); - _checkLambertW0Wad(2 ** 221 - 1, 107065553929442559763); - _checkLambertW0Wad(2 ** 220 - 1, 106378841351639838444); - _checkLambertW0Wad(2 ** 219 - 1, 105692170055654368478); - _checkLambertW0Wad(2 ** 218 - 1, 105005540572056956171); - _checkLambertW0Wad(2 ** 217 - 1, 104318953441679776592); - _checkLambertW0Wad(2 ** 216 - 1, 103632409215882036434); - _checkLambertW0Wad(2 ** 215 - 1, 102945908456824272609); - _checkLambertW0Wad(2 ** 214 - 1, 102259451737751625038); - _checkLambertW0Wad(2 ** 213 - 1, 101573039643286437675); - _checkLambertW0Wad(2 ** 212 - 1, 100886672769730558166); - _checkLambertW0Wad(2 ** 211 - 1, 100200351725377723788); - _checkLambertW0Wad(2 ** 210 - 1, 99514077130836439501); - _checkLambertW0Wad(2 ** 209 - 1, 98827849619363773067); - _checkLambertW0Wad(2 ** 208 - 1, 98141669837210512407); - _checkLambertW0Wad(2 ** 207 - 1, 97455538443978151616); - _checkLambertW0Wad(2 ** 206 - 1, 96769456112988194563); - _checkLambertW0Wad(2 ** 205 - 1, 96083423531664288650); - _checkLambertW0Wad(2 ** 204 - 1, 95397441401927726359); - _checkLambertW0Wad(2 ** 203 - 1, 94711510440606878644); - _checkLambertW0Wad(2 ** 202 - 1, 94025631379861152095); - _checkLambertW0Wad(2 ** 201 - 1, 93339804967620091367); - _checkLambertW0Wad(2 ** 200 - 1, 92654031968038279517); - _checkLambertW0Wad(2 ** 199 - 1, 91968313161966721893); - _checkLambertW0Wad(2 ** 198 - 1, 91282649347441434152); - _checkLambertW0Wad(2 ** 197 - 1, 90597041340189991908); - _checkLambertW0Wad(2 ** 196 - 1, 89911489974156838659); - _checkLambertW0Wad(2 ** 195 - 1, 89225996102048190100); - _checkLambertW0Wad(2 ** 194 - 1, 88540560595897416858); - _checkLambertW0Wad(2 ** 193 - 1, 87855184347651834275); - _checkLambertW0Wad(2 ** 192 - 1, 87169868269781877263); - _checkLambertW0Wad(2 ** 191 - 1, 86484613295913690725); - _checkLambertW0Wad(2 ** 190 - 1, 85799420381486221653); - _checkLambertW0Wad(2 ** 189 - 1, 85114290504433958190); - _checkLambertW0Wad(2 ** 188 - 1, 84429224665896523735); - _checkLambertW0Wad(2 ** 187 - 1, 83744223890956400983); - _checkLambertW0Wad(2 ** 186 - 1, 83059289229406131801); - _checkLambertW0Wad(2 ** 185 - 1, 82374421756546414467); - _checkLambertW0Wad(2 ** 184 - 1, 81689622574016600237); - _checkLambertW0Wad(2 ** 183 - 1, 81004892810659176931); - _checkLambertW0Wad(2 ** 182 - 1, 80320233623419918558); - _checkLambertW0Wad(2 ** 181 - 1, 79635646198285477393); - _checkLambertW0Wad(2 ** 180 - 1, 78951131751260298782); - _checkLambertW0Wad(2 ** 179 - 1, 78266691529384849812); - _checkLambertW0Wad(2 ** 178 - 1, 77582326811797271395); - _checkLambertW0Wad(2 ** 177 - 1, 76898038910840689756); - _checkLambertW0Wad(2 ** 176 - 1, 76213829173218558571); - _checkLambertW0Wad(2 ** 175 - 1, 75529698981200547567); - _checkLambertW0Wad(2 ** 174 - 1, 74845649753881648207); - _checkLambertW0Wad(2 ** 173 - 1, 74161682948497332759); - _checkLambertW0Wad(2 ** 172 - 1, 73477800061797780656); - _checkLambertW0Wad(2 ** 171 - 1, 72794002631484376331); - _checkLambertW0Wad(2 ** 170 - 1, 72110292237711886966); - _checkLambertW0Wad(2 ** 169 - 1, 71426670504659947705); - _checkLambertW0Wad(2 ** 168 - 1, 70743139102177717275); - _checkLambertW0Wad(2 ** 167 - 1, 70059699747505819935); - _checkLambertW0Wad(2 ** 166 - 1, 69376354207079961679); - _checkLambertW0Wad(2 ** 165 - 1, 68693104298420901379); - _checkLambertW0Wad(2 ** 164 - 1, 68009951892115772747); - _checkLambertW0Wad(2 ** 163 - 1, 67326898913896092682); - _checkLambertW0Wad(2 ** 162 - 1, 66643947346818157796); - _checkLambertW0Wad(2 ** 161 - 1, 65961099233551926143); - _checkLambertW0Wad(2 ** 160 - 1, 65278356678784907905); - _checkLambertW0Wad(2 ** 159 - 1, 64595721851748049983); - _checkLambertW0Wad(2 ** 158 - 1, 63913196988871098107); - _checkLambertW0Wad(2 ** 157 - 1, 63230784396575459844); - _checkLambertW0Wad(2 ** 156 - 1, 62548486454213176429); - _checkLambertW0Wad(2 ** 155 - 1, 61866305617161244980); - _checkLambertW0Wad(2 ** 154 - 1, 61184244420081220067); - _checkLambertW0Wad(2 ** 153 - 1, 60502305480354769865); - _checkLambertW0Wad(2 ** 152 - 1, 59820491501706673077); - _checkLambertW0Wad(2 ** 151 - 1, 59138805278027624755); - _checkLambertW0Wad(2 ** 150 - 1, 58457249697410179101); - _checkLambertW0Wad(2 ** 149 - 1, 57775827746412203235); - _checkLambertW0Wad(2 ** 148 - 1, 57094542514563356374); - _checkLambertW0Wad(2 ** 147 - 1, 56413397199131353678); - _checkLambertW0Wad(2 ** 146 - 1, 55732395110166133991); - _checkLambertW0Wad(2 ** 145 - 1, 55051539675841537897); - _checkLambertW0Wad(2 ** 144 - 1, 54370834448115730535); - _checkLambertW0Wad(2 ** 143 - 1, 53690283108733387465); - _checkLambertW0Wad(2 ** 142 - 1, 53009889475594618649); - _checkLambertW0Wad(2 ** 141 - 1, 52329657509517754228); - _checkLambertW0Wad(2 ** 140 - 1, 51649591321425477661); - _checkLambertW0Wad(2 ** 139 - 1, 50969695179986390948); - _checkLambertW0Wad(2 ** 138 - 1, 50289973519746960243); - _checkLambertW0Wad(2 ** 137 - 1, 49610430949791948630); - _checkLambertW0Wad(2 ** 136 - 1, 48931072262974930811); - _checkLambertW0Wad(2 ** 135 - 1, 48251902445764340905); - _checkLambertW0Wad(2 ** 134 - 1, 47572926688754773801); - _checkLambertW0Wad(2 ** 133 - 1, 46894150397897992742); - _checkLambertW0Wad(2 ** 132 - 1, 46215579206513348095); - _checkLambertW0Wad(2 ** 131 - 1, 45537218988143149666); - _checkLambertW0Wad(2 ** 130 - 1, 44859075870325031417); - _checkLambertW0Wad(2 ** 129 - 1, 44181156249360587882); - _checkLambertW0Wad(2 ** 128 - 1, 43503466806167642613); - _checkLambertW0Wad(2 ** 127 - 1, 42826014523312541917); - _checkLambertW0Wad(2 ** 126 - 1, 42148806703328979292); - _checkLambertW0Wad(2 ** 125 - 1, 41471850988441194251); - _checkLambertW0Wad(2 ** 124 - 1, 40795155381822122767); - _checkLambertW0Wad(2 ** 123 - 1, 40118728270531400808); - _checkLambertW0Wad(2 ** 122 - 1, 39442578450294263667); - _checkLambertW0Wad(2 ** 121 - 1, 38766715152300604375); - _checkLambertW0Wad(2 ** 120 - 1, 38091148072224059569); - _checkLambertW0Wad(2 ** 119 - 1, 37415887401684336100); - _checkLambertW0Wad(2 ** 118 - 1, 36740943862402491609); - _checkLambertW0Wad(2 ** 117 - 1, 36066328743329022902); - _checkLambertW0Wad(2 ** 116 - 1, 35392053941058967434); - _checkLambertW0Wad(2 ** 115 - 1, 34718132003887455986); - _checkLambertW0Wad(2 ** 114 - 1, 34044576179904059477); - _checkLambertW0Wad(2 ** 113 - 1, 33371400469575784902); - _checkLambertW0Wad(2 ** 112 - 1, 32698619683327803297); - _checkLambertW0Wad(2 ** 111 - 1, 32026249504699254799); - _checkLambertW0Wad(2 ** 110 - 1, 31354306559730344521); - _checkLambertW0Wad(2 ** 109 - 1, 30682808493328298780); - _checkLambertW0Wad(2 ** 108 - 1, 30011774053465850808); - _checkLambertW0Wad(2 ** 107 - 1, 29341223184189485097); - _checkLambertW0Wad(2 ** 106 - 1, 28671177128558970924); - _checkLambertW0Wad(2 ** 105 - 1, 28001658542808735364); - _checkLambertW0Wad(2 ** 104 - 1, 27332691623220201135); - _checkLambertW0Wad(2 ** 103 - 1, 26664302247428250682); - _checkLambertW0Wad(2 ** 102 - 1, 25996518132161712657); - _checkLambertW0Wad(2 ** 101 - 1, 25329369009746106264); - _checkLambertW0Wad(2 ** 100 - 1, 24662886826087826761); - _checkLambertW0Wad(2 ** 99 - 1, 23997105963326166352); - _checkLambertW0Wad(2 ** 98 - 1, 23332063490900058530); - _checkLambertW0Wad(2 ** 97 - 1, 22667799449451523321); - _checkLambertW0Wad(2 ** 96 - 1, 22004357172804292983); - _checkLambertW0Wad(2 ** 95 - 1, 21341783654247925671); - _checkLambertW0Wad(2 ** 94 - 1, 20680129964567978803); - _checkLambertW0Wad(2 ** 93 - 1, 20019451730746615034); - _checkLambertW0Wad(2 ** 92 - 1, 19359809686086176343); - _checkLambertW0Wad(2 ** 91 - 1, 18701270304772358157); - _checkLambertW0Wad(2 ** 90 - 1, 18043906536712772323); - _checkLambertW0Wad(2 ** 89 - 1, 17387798662016868795); - _checkLambertW0Wad(2 ** 88 - 1, 16733035288929945451); - _checkLambertW0Wad(2 ** 87 - 1, 16079714524670107222); - _checkLambertW0Wad(2 ** 86 - 1, 15427945355807184379); - _checkLambertW0Wad(2 ** 85 - 1, 14777849284057868231); - _checkLambertW0Wad(2 ** 84 - 1, 14129562275318189632); - _checkLambertW0Wad(2 ** 83 - 1, 13483237095324880705); - _checkLambertW0Wad(2 ** 82 - 1, 12839046125789215063); - _checkLambertW0Wad(2 ** 81 - 1, 12197184781931118579); - _checkLambertW0Wad(2 ** 80 - 1, 11557875688514566228 - 1); - _checkLambertW0Wad(2 ** 79 - 1, 10921373820226202580); - _checkLambertW0Wad(2 ** 78 - 1, 10287972878516218499); - _checkLambertW0Wad(2 ** 77 - 1, 9658013267990184319); - _checkLambertW0Wad(2 ** 76 - 1, 9031892161491509531); - _checkLambertW0Wad(2 ** 75 - 1, 8410076319328428686); - _checkLambertW0Wad(2 ** 74 - 1, 7793118576966979948); - _checkLambertW0Wad(2 ** 73 - 1, 7181679269695846234); - _checkLambertW0Wad(2 ** 72 - 1, 6576554370186862926); - _checkLambertW0Wad(2 ** 71 - 1, 5978712844468804878); - _checkLambertW0Wad(2 ** 70 - 1, 5389346779005776683); - _checkLambertW0Wad(2 ** 69 - 1, 4809939316762921936); - _checkLambertW0Wad(2 ** 68 - 1, 4242357480017482271); - _checkLambertW0Wad(2 ** 67 - 1, 3688979548845126287); - _checkLambertW0Wad(2 ** 66 - 1, 3152869312105232629); - _checkLambertW0Wad(2 ** 65 - 1, 2638010157689274059); - _checkLambertW0Wad(2 ** 64 - 1, 2149604165721149566); - _checkLambertW0Wad(2 ** 63 - 1, 1694407549795038335); - _checkLambertW0Wad(2 ** 62 - 1, 1280973323147500590); - _checkLambertW0Wad(2 ** 61 - 1, 919438481612859603); - _checkLambertW0Wad(2 ** 60 - 1, 620128202996354327); - _checkLambertW0Wad(2 ** 59 - 1, 390213425026895126); - _checkLambertW0Wad(2 ** 58 - 1, 229193491169149614); - _checkLambertW0Wad(2 ** 57 - 1, 126935310044982397); - _checkLambertW0Wad(2 ** 56 - 1, 67363429834711483); - _checkLambertW0Wad(2 ** 55 - 1, 34796675828817814); - _checkLambertW0Wad(2 ** 54 - 1, 17698377658513340); - _checkLambertW0Wad(2 ** 53 - 1, 8927148493627578); - _checkLambertW0Wad(2 ** 52 - 1, 4483453146102402); - _checkLambertW0Wad(2 ** 51 - 1, 2246746269994097); - _checkLambertW0Wad(2 ** 50 - 1, 1124634392838166); - _checkLambertW0Wad(2 ** 49 - 1, 562633308112667); - _checkLambertW0Wad(2 ** 48 - 1, 281395781982528); - _checkLambertW0Wad(2 ** 47 - 1, 140717685495042); - _checkLambertW0Wad(2 ** 46 - 1, 70363792940114); - _checkLambertW0Wad(2 ** 45 - 1, 35183134214121); - _checkLambertW0Wad(2 ** 44 - 1, 17591876567571); - _checkLambertW0Wad(2 ** 43 - 1, 8796015651975); - _checkLambertW0Wad(2 ** 42 - 1, 4398027168417); - _checkLambertW0Wad(2 ** 41 - 1, 2199018419863); - _checkLambertW0Wad(2 ** 40 - 1, 1099510418851); - _checkLambertW0Wad(2 ** 39 - 1, 549755511655); - _checkLambertW0Wad(2 ** 38 - 1, 274877831385); - _checkLambertW0Wad(2 ** 37 - 1, 137438934581); - _checkLambertW0Wad(2 ** 36 - 1, 68719472012); - _checkLambertW0Wad(2 ** 35 - 1, 34359737186); - _checkLambertW0Wad(2 ** 34 - 1, 17179868887); - _checkLambertW0Wad(2 ** 33 - 1, 8589934517); - _checkLambertW0Wad(2 ** 32 - 1, 4294967276); - _checkLambertW0Wad(2 ** 31 - 1, 2147483642); - _checkLambertW0Wad(2 ** 30 - 1, 1073741821); - _checkLambertW0Wad(2 ** 29 - 1, 536870910); - _checkLambertW0Wad(2 ** 28 - 1, 268435454); - _checkLambertW0Wad(2 ** 27 - 1, 134217726); - _checkLambertW0Wad(2 ** 26 - 1, 67108862); - _checkLambertW0Wad(2 ** 25 - 1, 33554430); - _checkLambertW0Wad(2 ** 24 - 1, 16777214); - _checkLambertW0Wad(2 ** 23 - 1, 8388606); - _checkLambertW0Wad(2 ** 22 - 1, 4194302); - _checkLambertW0Wad(2 ** 21 - 1, 2097150); - _checkLambertW0Wad(2 ** 20 - 1, 1048574); - _checkLambertW0Wad(2 ** 19 - 1, 524286); - _checkLambertW0Wad(2 ** 18 - 1, 262142); - _checkLambertW0Wad(2 ** 17 - 1, 131070); - _checkLambertW0Wad(2 ** 16 - 1, 65534); - _checkLambertW0Wad(2 ** 15 - 1, 32766); - _checkLambertW0Wad(2 ** 14 - 1, 16382); - _checkLambertW0Wad(2 ** 13 - 1, 8190); - _checkLambertW0Wad(2 ** 12 - 1, 4094); - _checkLambertW0Wad(2 ** 11 - 1, 2046); - _checkLambertW0Wad(2 ** 10 - 1, 1022); - _checkLambertW0Wad(2 ** 9 - 1, 510); - _checkLambertW0Wad(2 ** 8 - 1, 254); - } - - function testLambertW0WadRevertsForOutOfDomain() public { - FixedPointMathLib.lambertW0Wad(_LAMBERT_W0_MIN); - for (int256 i = 0; i <= 10; ++i) { - vm.expectRevert(FixedPointMathLib.OutOfDomain.selector); - FixedPointMathLib.lambertW0Wad(_LAMBERT_W0_MIN - 1 - i); - } - vm.expectRevert(FixedPointMathLib.OutOfDomain.selector); - FixedPointMathLib.lambertW0Wad(-type(int256).max); - } - - function _checkLambertW0Wad(int256 x, int256 expected) internal { - unchecked { - uint256 gasBefore = gasleft(); - int256 w = FixedPointMathLib.lambertW0Wad(x); - uint256 gasUsed = gasBefore - gasleft(); - emit LogInt("x", x); - emit LogUint("gasUsed", gasUsed); - assertEq(w, expected); - } - } - - function testLambertW0WadAccuracy() public { - testLambertW0WadAccuracy(uint184(int184(_testLamberW0WadAccuracyThres()))); - testLambertW0WadAccuracy(2 ** 184 - 1); - } - - function testLambertW0WadAccuracy(uint184 a) public { - int256 x = int256(int184(a)); - if (x >= _testLamberW0WadAccuracyThres()) { - int256 l = FixedPointMathLib.lnWad(x); - int256 r = x * l / _WAD; - int256 w = FixedPointMathLib.lambertW0Wad(r); - assertLt(FixedPointMathLib.abs(l - w), 0xff); - } - } - - function _testLamberW0WadAccuracyThres() internal pure returns (int256) { - unchecked { - return _ONE_DIV_EXP + _ONE_DIV_EXP * 0.01 ether / 1 ether; - } - } - - function testLambertW0WadWithinBounds(int256 x) public { - if (x <= 0) x = _boundLambertW0WadInput(x); - int256 w = FixedPointMathLib.lambertW0Wad(x); - assertTrue(w <= x); - unchecked { - if (x > _EXP) { - int256 l = FixedPointMathLib.lnWad(x); - assertGt(l, 0); - int256 ll = FixedPointMathLib.lnWad(l); - int256 q = ll * _WAD; - int256 lower = l - ll + q / (2 * l); - assertLt(lower, w + 1); - int256 upper = l - ll + (q * _EXP) / (l * (_EXP - _WAD)) + 1; - assertLt(w, upper); - } - } - } - - function testLambertW0WadWithinBounds() public { - testLambertW0WadWithinBounds(_EXP - 1); - testLambertW0WadWithinBounds(_EXP); - testLambertW0WadWithinBounds(_EXP + 1); - testLambertW0WadWithinBounds(type(int256).max); - } - - function testLambertW0WadMonotonicallyIncreasing() public { - unchecked { - for (uint256 i; i <= 256; ++i) { - uint256 x = 1 << i; - testLambertW0WadMonotonicallyIncreasingAround(int256(x)); - testLambertW0WadMonotonicallyIncreasingAround(int256(x - 1)); - } - for (uint256 i; i <= 57; ++i) { - uint256 x = 1 << i; - testLambertW0WadMonotonicallyIncreasingAround(-int256(x)); - testLambertW0WadMonotonicallyIncreasingAround(-int256(x - 1)); - } - } - } - - function testLambertW0WadMonotonicallyIncreasing2() public { - // These are some problematic values gathered over the attempts. - // Some might not be problematic now. - _testLambertW0WadMonoAround(0x598cdf77327d789dc); - _testLambertW0WadMonoAround(0x3c8d97dfe4afb1b05); - _testLambertW0WadMonoAround(0x56a147b480c03cc22); - _testLambertW0WadMonoAround(0x3136f439c231d0bb9); - _testLambertW0WadMonoAround(0x2ae7cff17ef2469a1); - _testLambertW0WadMonoAround(0x1de668fd7afcf61cc); - _testLambertW0WadMonoAround(0x15024b2a35f2cdd95); - _testLambertW0WadMonoAround(0x11a65ae94b59590f9); - _testLambertW0WadMonoAround(0xf0c2c82174dffb7e); - _testLambertW0WadMonoAround(0xed3e56938cb11626); - _testLambertW0WadMonoAround(0xecf5c4e511142439); - _testLambertW0WadMonoAround(0xc0755fa2b4033cb0); - _testLambertW0WadMonoAround(0xa235db282ea4edc6); - _testLambertW0WadMonoAround(0x9ff2ec5c26eec112); - _testLambertW0WadMonoAround(0xa0c3c4e36f4415f1); - _testLambertW0WadMonoAround(0x9b9f0e8d61287782); - _testLambertW0WadMonoAround(0x7df719d1a4a7b8ad); - _testLambertW0WadMonoAround(0x7c881679a1464d25); - _testLambertW0WadMonoAround(0x7bec47487071495a); - _testLambertW0WadMonoAround(0x7be31c75fc717f9f); - _testLambertW0WadMonoAround(0x7bbb4e0716eeca53); - _testLambertW0WadMonoAround(0x78e59d40a92b443b); - _testLambertW0WadMonoAround(0x77658c4ad3af717d); - _testLambertW0WadMonoAround(0x75ae9afa425919fe); - _testLambertW0WadMonoAround(0x7526092d05bef41f); - _testLambertW0WadMonoAround(0x52896fe82be03dfe); - _testLambertW0WadMonoAround(0x4f05b0ddf3b71a19); - _testLambertW0WadMonoAround(0x3094b0feb93943fd); - _testLambertW0WadMonoAround(0x2ef215ae6701c40e); - _testLambertW0WadMonoAround(0x2ebd1c82095d6a92); - _testLambertW0WadMonoAround(0x2e520a4e670d52bb); - _testLambertW0WadMonoAround(0xfc2f004412e5ce69); - _testLambertW0WadMonoAround(0x158bc0b201103a7fc); - _testLambertW0WadMonoAround(0x39280df60945c436b); - _testLambertW0WadMonoAround(0x47256e5d374b35f74); - _testLambertW0WadMonoAround(0x2b9568ffb08c155a4); - _testLambertW0WadMonoAround(0x1b60b07806956f34d); - _testLambertW0WadMonoAround(0x21902755d1eee824c); - _testLambertW0WadMonoAround(0x6e15c8a6ee6e4fca4); - _testLambertW0WadMonoAround(0x5b13067d92d8e49c6); - _testLambertW0WadMonoAround(0x2826ebc1fce90cf6e); - _testLambertW0WadMonoAround(0x215eb5aa1041510a4); - _testLambertW0WadMonoAround(0x47b20347b57504c32); - _testLambertW0WadMonoAround(0x75e8fd53f8c90f95a); - _testLambertW0WadMonoAround(0x43e8d80f9af282627); - _testLambertW0WadMonoAround(0x3cf555b5fd4f20615); - _testLambertW0WadMonoAround(0xaff4b8b52f8355e6e); - _testLambertW0WadMonoAround(0x529e89e77ae046255); - _testLambertW0WadMonoAround(0x1f0289433f07cbf53b); - _testLambertW0WadMonoAround(0xc1f6e56c2001d9432); - _testLambertW0WadMonoAround(0x5e4117305c6e33ebc); - _testLambertW0WadMonoAround(0x2b416472dce2ea26d); - _testLambertW0WadMonoAround(0x71f55956ef3326067); - _testLambertW0WadMonoAround(0x35d9d57c965eb82c6); - _testLambertW0WadMonoAround(0x184f520f19335f25d); - _testLambertW0WadMonoAround(0x3c4bb8f445abe21a7); - _testLambertW0WadMonoAround(0x573e3b3e06e208201); - _testLambertW0WadMonoAround(0x184f520f19335f25d); - _testLambertW0WadMonoAround(0x573e3b3e06e208201); - _testLambertW0WadMonoAround(0x61e511ba00db632a4); - _testLambertW0WadMonoAround(0x12731b97bde57933d); - _testLambertW0WadMonoAround(0x79c29b05cf39be374); - _testLambertW0WadMonoAround(0x390fcd4186ac250b3); - _testLambertW0WadMonoAround(0x69c74b5975fd4832a); - _testLambertW0WadMonoAround(0x59db219a7048121bd); - _testLambertW0WadMonoAround(0x28f2adc4fab331d251); - _testLambertW0WadMonoAround(0x7be91527cc31769c); - _testLambertW0WadMonoAround(0x2ef215ae6701c40f); - _testLambertW0WadMonoAround(0x1240541334cfadd81); - _testLambertW0WadMonoAround(0x2a79eccb3d5f4faaed); - _testLambertW0WadMonoAround(0x7470d50c23bfd30e0); - _testLambertW0WadMonoAround(0x313386f14a7f95af9); - _testLambertW0WadMonoAround(0x2a60f3b64c57088e9); - _testLambertW0WadMonoAround(0x381298f7aa53edfe0); - _testLambertW0WadMonoAround(0x5cbfac5d7a1770806); - _testLambertW0WadMonoAround(0x19e46d1b5e6aba57e); - _testLambertW0WadMonoAround(0x19ff86906ae47c70a); - _testLambertW0WadMonoAround(0x164684654d9ca54ea1); - _testLambertW0WadMonoAround(0x99337fa75e803139); - _testLambertW0WadMonoAround(0x6fa0a50fcb8a95b97e); - _testLambertW0WadMonoAround(0xa117a195e06c3fd531); - _testLambertW0WadMonoAround(0x305da7073093bd8a07); - _testLambertW0WadMonoAround(0x98582b07fd3c6b64); - _testLambertW0WadMonoAround(0x1e824d2a367d9ce65); - _testLambertW0WadMonoAround(0x7bea796d633b386a); - _testLambertW0WadMonoAround(0x2fff5c38c6b2a2cd); - _testLambertW0WadMonoAround(0x198af4e7ffee1df7627); - _testLambertW0WadMonoAround(0x8ea8a7b6f7c7424d8d); - _testLambertW0WadMonoAround(0x11e504fa805e54e2ed8); - _testLambertW0WadMonoAround(0x3e5f2a7801badcdabd); - _testLambertW0WadMonoAround(0x1b7aaad69ac8770a3be); - _testLambertW0WadMonoAround(0x658acb00d525f3d345); - _testLambertW0WadMonoAround(0xd994d6447146880183f); - _testLambertW0WadMonoAround(0x2e07a342d7b1bc1a5ae); - } - - function testLambertW0WadMonoDebug() public { - unchecked { - for (int256 i = -9; i <= 9; ++i) { - _testLambertW0WadMonoAround(0x2e07a342d7b1bc1a5ae + i); - } - } - } - - function _testLambertW0WadMonoAround(int256 x) internal { - emit LogInt("x", x); - emit LogUint("log2(x)", FixedPointMathLib.log2(uint256(x))); - testLambertW0WadMonotonicallyIncreasingAround(x); - } - - function testLambertW0WadMonotonicallyIncreasingAround2(uint96 t) public { - int256 x = int256(uint256(t)); - testLambertW0WadMonotonicallyIncreasingAround(x); - if (t & 0xff == 0xab) { - _testLambertW0WadMonoFocus(x, 0, 0x1ffffffffffff, 0xffffffffffffffffff); - _testLambertW0WadMonoFocus(x, 1, 0x1fffffffffffff, 0xffffffffffffffffff); - _testLambertW0WadMonoFocus(x, 2, 0xfffffffffffffff, 0xffffffffffffffffff); - _testLambertW0WadMonoFocus(x, 3, 0xffffffffffffffff, 0xfffffffffffffffff); - _testLambertW0WadMonoFocus(x, 4, 0xffffffffffffffff, 0xfffffffffffffffff); - _testLambertW0WadMonoFocus(x, 5, 0xffffffffffffffff, 0xffffffffffffffffff); - _testLambertW0WadMonoFocus(x, 6, 0xffffffffffffffff, 0xffffffffffffffffff); - _testLambertW0WadMonoFocus(x, 7, 0xffffffffffffffff, 0xfffffffffffffffffff); - _testLambertW0WadMonoFocus(x, 8, 0xffffffffffffffff, 0xfffffffffffffffffff); - _testLambertW0WadMonoFocus(x, 9, 0xffffffffffffffff, 0xffffffffffffffffffff); - } - } - - function _testLambertW0WadMonoFocus(int256 t, int256 i, int256 low, int256 mask) internal { - int256 x; - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, t) - mstore(0x20, i) - x := and(keccak256(0x00, 0x40), mask) - } - do { - testLambertW0WadMonotonicallyIncreasingAround(x); - x >>= 1; - } while (x >= low); - } - - function testLambertW0WadMonotonicallyIncreasingAround(int256 t) public { - if (t < _LAMBERT_W0_MIN) t = _boundLambertW0WadInput(t); - unchecked { - int256 end = t + 2; - for (int256 x = t - 2; x != end; ++x) { - testLambertW0WadMonotonicallyIncreasing(x, x + 1); - } - } - } - - function testLambertW0WadMonotonicallyIncreasing(int256 a, int256 b) public { - if (a < _LAMBERT_W0_MIN) a = _boundLambertW0WadInput(a); - if (b < _LAMBERT_W0_MIN) b = _boundLambertW0WadInput(b); - if (a > b) { - int256 t = b; - b = a; - a = t; - } - unchecked { - uint256 gasBefore = gasleft(); - int256 w0a = FixedPointMathLib.lambertW0Wad(a); - uint256 gasUsed = gasBefore - gasleft(); - int256 w0b = FixedPointMathLib.lambertW0Wad(b); - bool success = w0a <= w0b; - emit TestingLambertW0WadMonotonicallyIncreasing(a, b, w0a, w0b, success, gasUsed); - if (!success) { - emit LogUint("log2(a)", FixedPointMathLib.log2(uint256(a))); - emit LogUint("log2(b)", FixedPointMathLib.log2(uint256(b))); - emit LogUint("log2(w0a)", FixedPointMathLib.log2(uint256(w0a))); - emit LogUint("log2(w0b)", FixedPointMathLib.log2(uint256(w0b))); - assertTrue(success); - } - } - } - - function _boundLambertW0WadInput(int256 x) internal pure returns (int256 result) { - /// @solidity memory-safe-assembly - assembly { - result := shr(1, shl(1, not(x))) - } - } - - function testMulWad() public { - assertEq(FixedPointMathLib.mulWad(2.5e18, 0.5e18), 1.25e18); - assertEq(FixedPointMathLib.mulWad(3e18, 1e18), 3e18); - assertEq(FixedPointMathLib.mulWad(369, 271), 0); - } - - function testMulWadEdgeCases() public { - assertEq(FixedPointMathLib.mulWad(0, 1e18), 0); - assertEq(FixedPointMathLib.mulWad(1e18, 0), 0); - assertEq(FixedPointMathLib.mulWad(0, 0), 0); - } - - function testMulWadUp() public { - assertEq(FixedPointMathLib.mulWadUp(2.5e18, 0.5e18), 1.25e18); - assertEq(FixedPointMathLib.mulWadUp(3e18, 1e18), 3e18); - assertEq(FixedPointMathLib.mulWadUp(369, 271), 1); - } - - function testMulWadUpEdgeCases() public { - assertEq(FixedPointMathLib.mulWadUp(0, 1e18), 0); - assertEq(FixedPointMathLib.mulWadUp(1e18, 0), 0); - assertEq(FixedPointMathLib.mulWadUp(0, 0), 0); - } - - function testDivWad() public { - assertEq(FixedPointMathLib.divWad(1.25e18, 0.5e18), 2.5e18); - assertEq(FixedPointMathLib.divWad(3e18, 1e18), 3e18); - assertEq(FixedPointMathLib.divWad(2, 100000000000000e18), 0); - } - - function testDivWadEdgeCases() public { - assertEq(FixedPointMathLib.divWad(0, 1e18), 0); - } - - function testDivWadZeroDenominatorReverts() public { - vm.expectRevert(FixedPointMathLib.DivWadFailed.selector); - FixedPointMathLib.divWad(1e18, 0); - } - - function testDivWadUp() public { - assertEq(FixedPointMathLib.divWadUp(1.25e18, 0.5e18), 2.5e18); - assertEq(FixedPointMathLib.divWadUp(3e18, 1e18), 3e18); - assertEq(FixedPointMathLib.divWadUp(2, 100000000000000e18), 1); - unchecked { - for (uint256 i; i < 10; ++i) { - assertEq(FixedPointMathLib.divWadUp(2, 100000000000000e18), 1); - } - } - } - - function testDivWadUpEdgeCases() public { - assertEq(FixedPointMathLib.divWadUp(0, 1e18), 0); - } - - function testDivWadUpZeroDenominatorReverts() public { - vm.expectRevert(FixedPointMathLib.DivWadFailed.selector); - FixedPointMathLib.divWadUp(1e18, 0); - } - - function testMulDiv() public { - assertEq(FixedPointMathLib.mulDiv(2.5e27, 0.5e27, 1e27), 1.25e27); - assertEq(FixedPointMathLib.mulDiv(2.5e18, 0.5e18, 1e18), 1.25e18); - assertEq(FixedPointMathLib.mulDiv(2.5e8, 0.5e8, 1e8), 1.25e8); - assertEq(FixedPointMathLib.mulDiv(369, 271, 1e2), 999); - - assertEq(FixedPointMathLib.mulDiv(1e27, 1e27, 2e27), 0.5e27); - assertEq(FixedPointMathLib.mulDiv(1e18, 1e18, 2e18), 0.5e18); - assertEq(FixedPointMathLib.mulDiv(1e8, 1e8, 2e8), 0.5e8); - - assertEq(FixedPointMathLib.mulDiv(2e27, 3e27, 2e27), 3e27); - assertEq(FixedPointMathLib.mulDiv(3e18, 2e18, 3e18), 2e18); - assertEq(FixedPointMathLib.mulDiv(2e8, 3e8, 2e8), 3e8); - } - - function testMulDivEdgeCases() public { - assertEq(FixedPointMathLib.mulDiv(0, 1e18, 1e18), 0); - assertEq(FixedPointMathLib.mulDiv(1e18, 0, 1e18), 0); - assertEq(FixedPointMathLib.mulDiv(0, 0, 1e18), 0); - } - - function testMulDivZeroDenominatorReverts() public { - vm.expectRevert(FixedPointMathLib.MulDivFailed.selector); - FixedPointMathLib.mulDiv(1e18, 1e18, 0); - } - - function testMulDivUp() public { - assertEq(FixedPointMathLib.mulDivUp(2.5e27, 0.5e27, 1e27), 1.25e27); - assertEq(FixedPointMathLib.mulDivUp(2.5e18, 0.5e18, 1e18), 1.25e18); - assertEq(FixedPointMathLib.mulDivUp(2.5e8, 0.5e8, 1e8), 1.25e8); - assertEq(FixedPointMathLib.mulDivUp(369, 271, 1e2), 1000); - - assertEq(FixedPointMathLib.mulDivUp(1e27, 1e27, 2e27), 0.5e27); - assertEq(FixedPointMathLib.mulDivUp(1e18, 1e18, 2e18), 0.5e18); - assertEq(FixedPointMathLib.mulDivUp(1e8, 1e8, 2e8), 0.5e8); - - assertEq(FixedPointMathLib.mulDivUp(2e27, 3e27, 2e27), 3e27); - assertEq(FixedPointMathLib.mulDivUp(3e18, 2e18, 3e18), 2e18); - assertEq(FixedPointMathLib.mulDivUp(2e8, 3e8, 2e8), 3e8); - } - - function testMulDivUpEdgeCases() public { - assertEq(FixedPointMathLib.mulDivUp(0, 1e18, 1e18), 0); - assertEq(FixedPointMathLib.mulDivUp(1e18, 0, 1e18), 0); - assertEq(FixedPointMathLib.mulDivUp(0, 0, 1e18), 0); - } - - function testMulDivUpZeroDenominator() public { - vm.expectRevert(FixedPointMathLib.MulDivFailed.selector); - FixedPointMathLib.mulDivUp(1e18, 1e18, 0); - } - - function testLnWad() public { - assertEq(FixedPointMathLib.lnWad(1e18), 0); - - // Actual: 999999999999999999.8674576… - assertEq(FixedPointMathLib.lnWad(2718281828459045235), 999999999999999999); - - // Actual: 2461607324344817917.963296… - assertEq(FixedPointMathLib.lnWad(11723640096265400935), 2461607324344817918); - } - - function testLnWadSmall() public { - // Actual: -41446531673892822312.3238461… - assertEq(FixedPointMathLib.lnWad(1), -41446531673892822313); - - // Actual: -37708862055609454006.40601608… - assertEq(FixedPointMathLib.lnWad(42), -37708862055609454007); - - // Actual: -32236191301916639576.251880365581… - assertEq(FixedPointMathLib.lnWad(1e4), -32236191301916639577); - - // Actual: -20723265836946411156.161923092… - assertEq(FixedPointMathLib.lnWad(1e9), -20723265836946411157); - } - - function testLnWadBig() public { - // Actual: 135305999368893231589.070344787… - assertEq(FixedPointMathLib.lnWad(2 ** 255 - 1), 135305999368893231589); - - // Actual: 76388489021297880288.605614463571… - assertEq(FixedPointMathLib.lnWad(2 ** 170), 76388489021297880288); - - // Actual: 47276307437780177293.081865… - assertEq(FixedPointMathLib.lnWad(2 ** 128), 47276307437780177293); - } - - function testLnWadNegativeReverts() public { - vm.expectRevert(FixedPointMathLib.LnWadUndefined.selector); - FixedPointMathLib.lnWad(-1); - FixedPointMathLib.lnWad(-2 ** 255); - } - - function testLnWadOverflowReverts() public { - vm.expectRevert(FixedPointMathLib.LnWadUndefined.selector); - FixedPointMathLib.lnWad(0); - } - - function testRPow() public { - assertEq(FixedPointMathLib.rpow(0, 0, 0), 0); - assertEq(FixedPointMathLib.rpow(1, 0, 0), 0); - assertEq(FixedPointMathLib.rpow(0, 1, 0), 0); - assertEq(FixedPointMathLib.rpow(0, 0, 1), 1); - assertEq(FixedPointMathLib.rpow(1, 1, 0), 1); - assertEq(FixedPointMathLib.rpow(1, 1, 1), 1); - assertEq(FixedPointMathLib.rpow(2e27, 0, 1e27), 1e27); - assertEq(FixedPointMathLib.rpow(2e27, 2, 1e27), 4e27); - assertEq(FixedPointMathLib.rpow(2e18, 2, 1e18), 4e18); - assertEq(FixedPointMathLib.rpow(2e8, 2, 1e8), 4e8); - assertEq(FixedPointMathLib.rpow(8, 3, 1), 512); - } - - function testRPowOverflowReverts() public { - vm.expectRevert(FixedPointMathLib.RPowOverflow.selector); - FixedPointMathLib.rpow(2, type(uint128).max, 1); - FixedPointMathLib.rpow(type(uint128).max, 3, 1); - } - - function testSqrt() public { - assertEq(FixedPointMathLib.sqrt(0), 0); - assertEq(FixedPointMathLib.sqrt(1), 1); - assertEq(FixedPointMathLib.sqrt(2704), 52); - assertEq(FixedPointMathLib.sqrt(110889), 333); - assertEq(FixedPointMathLib.sqrt(32239684), 5678); - unchecked { - for (uint256 i = 100; i < 200; ++i) { - assertEq(FixedPointMathLib.sqrt(i * i), i); - } - } - } - - function testSqrtWad() public { - assertEq(FixedPointMathLib.sqrtWad(0), 0); - assertEq(FixedPointMathLib.sqrtWad(1), 10 ** 9); - assertEq(FixedPointMathLib.sqrtWad(2), 1414213562); - assertEq(FixedPointMathLib.sqrtWad(4), 2000000000); - assertEq(FixedPointMathLib.sqrtWad(8), 2828427124); - assertEq(FixedPointMathLib.sqrtWad(16), 4000000000); - assertEq(FixedPointMathLib.sqrtWad(32), 5656854249); - assertEq(FixedPointMathLib.sqrtWad(64), 8000000000); - assertEq(FixedPointMathLib.sqrtWad(10 ** 18), 10 ** 18); - assertEq(FixedPointMathLib.sqrtWad(4 * 10 ** 18), 2 * 10 ** 18); - assertEq(FixedPointMathLib.sqrtWad(type(uint8).max), 15968719422); - assertEq(FixedPointMathLib.sqrtWad(type(uint16).max), 255998046867); - assertEq(FixedPointMathLib.sqrtWad(type(uint32).max), 65535999992370); - assertEq(FixedPointMathLib.sqrtWad(type(uint64).max), 4294967295999999999); - assertEq(FixedPointMathLib.sqrtWad(type(uint128).max), 18446744073709551615999999999); - assertEq( - FixedPointMathLib.sqrtWad(type(uint256).max), - 340282366920938463463374607431768211455000000000 - ); - } - - function testCbrt() public { - assertEq(FixedPointMathLib.cbrt(0), 0); - assertEq(FixedPointMathLib.cbrt(1), 1); - assertEq(FixedPointMathLib.cbrt(2), 1); - assertEq(FixedPointMathLib.cbrt(3), 1); - assertEq(FixedPointMathLib.cbrt(9), 2); - assertEq(FixedPointMathLib.cbrt(27), 3); - assertEq(FixedPointMathLib.cbrt(80), 4); - assertEq(FixedPointMathLib.cbrt(81), 4); - assertEq(FixedPointMathLib.cbrt(10 ** 18), 10 ** 6); - assertEq(FixedPointMathLib.cbrt(8 * 10 ** 18), 2 * 10 ** 6); - assertEq(FixedPointMathLib.cbrt(9 * 10 ** 18), 2080083); - assertEq(FixedPointMathLib.cbrt(type(uint8).max), 6); - assertEq(FixedPointMathLib.cbrt(type(uint16).max), 40); - assertEq(FixedPointMathLib.cbrt(type(uint32).max), 1625); - assertEq(FixedPointMathLib.cbrt(type(uint64).max), 2642245); - assertEq(FixedPointMathLib.cbrt(type(uint128).max), 6981463658331); - assertEq(FixedPointMathLib.cbrt(type(uint256).max), 48740834812604276470692694); - } - - function testCbrtWad() public { - assertEq(FixedPointMathLib.cbrtWad(0), 0); - assertEq(FixedPointMathLib.cbrtWad(1), 10 ** 12); - assertEq(FixedPointMathLib.cbrtWad(2), 1259921049894); - assertEq(FixedPointMathLib.cbrtWad(3), 1442249570307); - assertEq(FixedPointMathLib.cbrtWad(9), 2080083823051); - assertEq(FixedPointMathLib.cbrtWad(27), 3000000000000); - assertEq(FixedPointMathLib.cbrtWad(80), 4308869380063); - assertEq(FixedPointMathLib.cbrtWad(81), 4326748710922); - assertEq(FixedPointMathLib.cbrtWad(10 ** 18), 10 ** 18); - assertEq(FixedPointMathLib.cbrtWad(8 * 10 ** 18), 2 * 10 ** 18); - assertEq(FixedPointMathLib.cbrtWad(9 * 10 ** 18), 2080083823051904114); - assertEq(FixedPointMathLib.cbrtWad(type(uint8).max), 6341325705384); - assertEq(FixedPointMathLib.cbrtWad(type(uint16).max), 40317268530317); - assertEq(FixedPointMathLib.cbrtWad(type(uint32).max), 1625498677089280); - assertEq(FixedPointMathLib.cbrtWad(type(uint64).max), 2642245949629133047); - assertEq(FixedPointMathLib.cbrtWad(type(uint128).max), 6981463658331559092288464); - assertEq( - FixedPointMathLib.cbrtWad(type(uint256).max), 48740834812604276470692694000000000000 - ); - } - - function testLog2() public { - assertEq(FixedPointMathLib.log2(0), 0); - assertEq(FixedPointMathLib.log2(2), 1); - assertEq(FixedPointMathLib.log2(4), 2); - assertEq(FixedPointMathLib.log2(1024), 10); - assertEq(FixedPointMathLib.log2(1048576), 20); - assertEq(FixedPointMathLib.log2(1073741824), 30); - for (uint256 i = 1; i < 255; i++) { - assertEq(FixedPointMathLib.log2((1 << i) - 1), i - 1); - assertEq(FixedPointMathLib.log2((1 << i)), i); - assertEq(FixedPointMathLib.log2((1 << i) + 1), i); - } - } - - function testLog2Differential(uint256 x) public { - assertEq(FixedPointMathLib.log2(x), _log2Original(x)); - } - - function _log2Original(uint256 value) internal pure returns (uint256) { - uint256 result = 0; - unchecked { - if (value >> 128 > 0) { - value >>= 128; - result += 128; - } - if (value >> 64 > 0) { - value >>= 64; - result += 64; - } - if (value >> 32 > 0) { - value >>= 32; - result += 32; - } - if (value >> 16 > 0) { - value >>= 16; - result += 16; - } - if (value >> 8 > 0) { - value >>= 8; - result += 8; - } - if (value >> 4 > 0) { - value >>= 4; - result += 4; - } - if (value >> 2 > 0) { - value >>= 2; - result += 2; - } - if (value >> 1 > 0) { - result += 1; - } - } - return result; - } - - function testLog2Up() public { - assertEq(FixedPointMathLib.log2Up(0), 0); - assertEq(FixedPointMathLib.log2Up(1), 0); - assertEq(FixedPointMathLib.log2Up(2), 1); - assertEq(FixedPointMathLib.log2Up(2 + 1), 2); - assertEq(FixedPointMathLib.log2Up(4), 2); - assertEq(FixedPointMathLib.log2Up(4 + 1), 3); - assertEq(FixedPointMathLib.log2Up(4 + 2), 3); - assertEq(FixedPointMathLib.log2Up(1024), 10); - assertEq(FixedPointMathLib.log2Up(1024 + 1), 11); - assertEq(FixedPointMathLib.log2Up(1048576), 20); - assertEq(FixedPointMathLib.log2Up(1048576 + 1), 21); - assertEq(FixedPointMathLib.log2Up(1073741824), 30); - assertEq(FixedPointMathLib.log2Up(1073741824 + 1), 31); - for (uint256 i = 2; i < 255; i++) { - assertEq(FixedPointMathLib.log2Up((1 << i) - 1), i); - assertEq(FixedPointMathLib.log2Up((1 << i)), i); - assertEq(FixedPointMathLib.log2Up((1 << i) + 1), i + 1); - } - } - - function testAvg() public { - assertEq(FixedPointMathLib.avg(uint256(5), uint256(6)), uint256(5)); - assertEq(FixedPointMathLib.avg(uint256(0), uint256(1)), uint256(0)); - assertEq(FixedPointMathLib.avg(uint256(45645465), uint256(4846513)), uint256(25245989)); - } - - function testAvgSigned() public { - assertEq(FixedPointMathLib.avg(int256(5), int256(6)), int256(5)); - assertEq(FixedPointMathLib.avg(int256(0), int256(1)), int256(0)); - assertEq(FixedPointMathLib.avg(int256(45645465), int256(4846513)), int256(25245989)); - - assertEq(FixedPointMathLib.avg(int256(5), int256(-6)), int256(-1)); - assertEq(FixedPointMathLib.avg(int256(0), int256(-1)), int256(-1)); - assertEq(FixedPointMathLib.avg(int256(45645465), int256(-4846513)), int256(20399476)); - } - - function testAvgEdgeCase() public { - assertEq(FixedPointMathLib.avg(uint256(2 ** 256 - 1), uint256(1)), uint256(2 ** 255)); - assertEq(FixedPointMathLib.avg(uint256(2 ** 256 - 1), uint256(10)), uint256(2 ** 255 + 4)); - assertEq( - FixedPointMathLib.avg(uint256(2 ** 256 - 1), uint256(2 ** 256 - 1)), - uint256(2 ** 256 - 1) - ); - } - - function testAbs() public { - assertEq(FixedPointMathLib.abs(0), 0); - assertEq(FixedPointMathLib.abs(-5), 5); - assertEq(FixedPointMathLib.abs(5), 5); - assertEq(FixedPointMathLib.abs(-1155656654), 1155656654); - assertEq(FixedPointMathLib.abs(621356166516546561651), 621356166516546561651); - } - - function testDist() public { - assertEq(FixedPointMathLib.dist(0, 0), 0); - assertEq(FixedPointMathLib.dist(-5, -4), 1); - assertEq(FixedPointMathLib.dist(5, 46), 41); - assertEq(FixedPointMathLib.dist(46, 5), 41); - assertEq(FixedPointMathLib.dist(-1155656654, 6544844), 1162201498); - assertEq(FixedPointMathLib.dist(-848877, -8447631456), 8446782579); - } - - function testDistEdgeCases() public { - assertEq(FixedPointMathLib.dist(type(int256).min, type(int256).max), type(uint256).max); - assertEq( - FixedPointMathLib.dist(type(int256).min, 0), - 0x8000000000000000000000000000000000000000000000000000000000000000 - ); - assertEq( - FixedPointMathLib.dist(type(int256).max, 5), - 0x7ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffa - ); - assertEq( - FixedPointMathLib.dist(type(int256).min, -5), - 0x7ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffb - ); - } - - function testAbsEdgeCases() public { - assertEq(FixedPointMathLib.abs(-(2 ** 255 - 1)), (2 ** 255 - 1)); - assertEq(FixedPointMathLib.abs((2 ** 255 - 1)), (2 ** 255 - 1)); - } - - function testGcd() public { - assertEq(FixedPointMathLib.gcd(0, 0), 0); - assertEq(FixedPointMathLib.gcd(85, 0), 85); - assertEq(FixedPointMathLib.gcd(0, 2), 2); - assertEq(FixedPointMathLib.gcd(56, 45), 1); - assertEq(FixedPointMathLib.gcd(12, 28), 4); - assertEq(FixedPointMathLib.gcd(12, 1), 1); - assertEq(FixedPointMathLib.gcd(486516589451122, 48656), 2); - assertEq(FixedPointMathLib.gcd(2 ** 254 - 4, 2 ** 128 - 1), 15); - assertEq(FixedPointMathLib.gcd(3, 26017198113384995722614372765093167890), 1); - unchecked { - for (uint256 i = 2; i < 10; ++i) { - assertEq(FixedPointMathLib.gcd(31 * (1 << i), 31), 31); - } - } - } - - function testFullMulDiv() public { - assertEq(FixedPointMathLib.fullMulDiv(0, 0, 1), 0); - assertEq(FixedPointMathLib.fullMulDiv(4, 4, 2), 8); - assertEq(FixedPointMathLib.fullMulDiv(2 ** 200, 2 ** 200, 2 ** 200), 2 ** 200); - } - - function testFullMulDivUpRevertsIfRoundedUpResultOverflowsCase1() public { - vm.expectRevert(FixedPointMathLib.FullMulDivFailed.selector); - FixedPointMathLib.fullMulDivUp( - 535006138814359, 432862656469423142931042426214547535783388063929571229938474969, 2 - ); - } - - function testFullMulDivUpRevertsIfRoundedUpResultOverflowsCase2() public { - vm.expectRevert(FixedPointMathLib.FullMulDivFailed.selector); - FixedPointMathLib.fullMulDivUp( - 115792089237316195423570985008687907853269984659341747863450311749907997002549, - 115792089237316195423570985008687907853269984659341747863450311749907997002550, - 115792089237316195423570985008687907853269984653042931687443039491902864365164 - ); - } - - function testFullMulDiv(uint256 a, uint256 b, uint256 d) public returns (uint256 result) { - if (d == 0) { - vm.expectRevert(FixedPointMathLib.FullMulDivFailed.selector); - FixedPointMathLib.fullMulDiv(a, b, d); - return 0; - } - - // Compute a * b in Chinese Remainder Basis - uint256 expectedA; - uint256 expectedB; - unchecked { - expectedA = a * b; - expectedB = mulmod(a, b, 2 ** 256 - 1); - } - - // Construct a * b - uint256 prod0; // Least significant 256 bits of the product - uint256 prod1; // Most significant 256 bits of the product - assembly { - let mm := mulmod(a, b, not(0)) - prod0 := mul(a, b) - prod1 := sub(sub(mm, prod0), lt(mm, prod0)) - } - if (prod1 >= d) { - vm.expectRevert(FixedPointMathLib.FullMulDivFailed.selector); - FixedPointMathLib.fullMulDiv(a, b, d); - return 0; - } - - uint256 q = FixedPointMathLib.fullMulDiv(a, b, d); - uint256 r = mulmod(a, b, d); - - // Compute q * d + r in Chinese Remainder Basis - uint256 actualA; - uint256 actualB; - unchecked { - actualA = q * d + r; - actualB = addmod(mulmod(q, d, 2 ** 256 - 1), r, 2 ** 256 - 1); - } - - assertEq(actualA, expectedA); - assertEq(actualB, expectedB); - return q; - } - - function testFullMulDivUp(uint256 a, uint256 b, uint256 d) public { - uint256 fullMulDivResult = testFullMulDiv(a, b, d); - if (fullMulDivResult != 0) { - uint256 expectedResult = fullMulDivResult; - if (mulmod(a, b, d) > 0) { - if (!(fullMulDivResult < type(uint256).max)) { - vm.expectRevert(FixedPointMathLib.FullMulDivFailed.selector); - FixedPointMathLib.fullMulDivUp(a, b, d); - return; - } - expectedResult++; - } - assertEq(FixedPointMathLib.fullMulDivUp(a, b, d), expectedResult); - } - } - - function testMulWad(uint256 x, uint256 y) public { - // Ignore cases where x * y overflows. - unchecked { - if (x != 0 && (x * y) / x != y) return; - } - - assertEq(FixedPointMathLib.mulWad(x, y), (x * y) / 1e18); - } - - function testMulWadOverflowReverts(uint256 x, uint256 y) public { - // Ignore cases where x * y does not overflow. - unchecked { - vm.assume(x != 0 && (x * y) / x != y); - } - vm.expectRevert(FixedPointMathLib.MulWadFailed.selector); - FixedPointMathLib.mulWad(x, y); - } - - function testMulWadUp(uint256 x, uint256 y) public { - // Ignore cases where x * y overflows. - unchecked { - if (x != 0 && (x * y) / x != y) return; - } - - assertEq(FixedPointMathLib.mulWadUp(x, y), x * y == 0 ? 0 : (x * y - 1) / 1e18 + 1); - } - - function testMulWadUpOverflowReverts(uint256 x, uint256 y) public { - // Ignore cases where x * y does not overflow. - unchecked { - vm.assume(x != 0 && !((x * y) / x == y)); - } - vm.expectRevert(FixedPointMathLib.MulWadFailed.selector); - FixedPointMathLib.mulWadUp(x, y); - } - - function testDivWad(uint256 x, uint256 y) public { - // Ignore cases where x * WAD overflows or y is 0. - unchecked { - if (y == 0 || (x != 0 && (x * 1e18) / 1e18 != x)) return; - } - - assertEq(FixedPointMathLib.divWad(x, y), (x * 1e18) / y); - } - - function testDivWadOverflowReverts(uint256 x, uint256 y) public { - // Ignore cases where x * WAD does not overflow or y is 0. - unchecked { - vm.assume(y != 0 && (x * 1e18) / 1e18 != x); - } - vm.expectRevert(FixedPointMathLib.DivWadFailed.selector); - FixedPointMathLib.divWad(x, y); - } - - function testDivWadZeroDenominatorReverts(uint256 x) public { - vm.expectRevert(FixedPointMathLib.DivWadFailed.selector); - FixedPointMathLib.divWad(x, 0); - } - - function testDivWadUp(uint256 x, uint256 y) public { - // Ignore cases where x * WAD overflows or y is 0. - unchecked { - if (y == 0 || (x != 0 && (x * 1e18) / 1e18 != x)) return; - } - - assertEq(FixedPointMathLib.divWadUp(x, y), x == 0 ? 0 : (x * 1e18 - 1) / y + 1); - } - - function testDivWadUpOverflowReverts(uint256 x, uint256 y) public { - // Ignore cases where x * WAD does not overflow or y is 0. - unchecked { - vm.assume(y != 0 && (x * 1e18) / 1e18 != x); - } - vm.expectRevert(FixedPointMathLib.DivWadFailed.selector); - FixedPointMathLib.divWadUp(x, y); - } - - function testDivWadUpZeroDenominatorReverts(uint256 x) public { - vm.expectRevert(FixedPointMathLib.DivWadFailed.selector); - FixedPointMathLib.divWadUp(x, 0); - } - - function testMulDiv(uint256 x, uint256 y, uint256 denominator) public { - // Ignore cases where x * y overflows or denominator is 0. - unchecked { - if (denominator == 0 || (x != 0 && (x * y) / x != y)) return; - } - - assertEq(FixedPointMathLib.mulDiv(x, y, denominator), (x * y) / denominator); - } - - function testMulDivOverflowReverts(uint256 x, uint256 y, uint256 denominator) public { - // Ignore cases where x * y does not overflow or denominator is 0. - unchecked { - vm.assume(denominator != 0 && x != 0 && (x * y) / x != y); - } - vm.expectRevert(FixedPointMathLib.MulDivFailed.selector); - FixedPointMathLib.mulDiv(x, y, denominator); - } - - function testMulDivZeroDenominatorReverts(uint256 x, uint256 y) public { - vm.expectRevert(FixedPointMathLib.MulDivFailed.selector); - FixedPointMathLib.mulDiv(x, y, 0); - } - - function testMulDivUp(uint256 x, uint256 y, uint256 denominator) public { - // Ignore cases where x * y overflows or denominator is 0. - unchecked { - if (denominator == 0 || (x != 0 && (x * y) / x != y)) return; - } - - assertEq( - FixedPointMathLib.mulDivUp(x, y, denominator), - x * y == 0 ? 0 : (x * y - 1) / denominator + 1 - ); - } - - function testMulDivUpOverflowReverts(uint256 x, uint256 y, uint256 denominator) public { - // Ignore cases where x * y does not overflow or denominator is 0. - unchecked { - vm.assume(denominator != 0 && x != 0 && (x * y) / x != y); - } - vm.expectRevert(FixedPointMathLib.MulDivFailed.selector); - FixedPointMathLib.mulDivUp(x, y, denominator); - } - - function testMulDivUpZeroDenominatorReverts(uint256 x, uint256 y) public { - vm.expectRevert(FixedPointMathLib.MulDivFailed.selector); - FixedPointMathLib.mulDivUp(x, y, 0); - } - - function testCbrt(uint256 x) public { - uint256 root = FixedPointMathLib.cbrt(x); - uint256 next = root + 1; - - // Ignore cases where `next * next * next` or `next * next` overflows. - unchecked { - if (next * next * next < next * next) return; - if (next * next < next) return; - } - - assertTrue(root * root * root <= x && next * next * next > x); - } - - function testCbrtWad(uint256 x) public { - uint256 result = FixedPointMathLib.cbrtWad(x); - uint256 floor = FixedPointMathLib.cbrt(x); - assertTrue(result >= floor * 10 ** 12 && result <= (floor + 1) * 10 ** 12); - assertEq(result / 10 ** 12, floor); - } - - function testCbrtBack(uint256 x) public { - unchecked { - x = _bound(x, 0, 48740834812604276470692694); - while (x != 0) { - assertEq(FixedPointMathLib.cbrt(x * x * x), x); - x >>= 1; - } - } - } - - function testSqrt(uint256 x) public { - uint256 root = FixedPointMathLib.sqrt(x); - uint256 next = root + 1; - - // Ignore cases where `next * next` overflows. - unchecked { - if (next * next < next) return; - } - - assertTrue(root * root <= x && next * next > x); - } - - function testSqrtWad(uint256 x) public { - uint256 result = FixedPointMathLib.sqrtWad(x); - uint256 floor = FixedPointMathLib.sqrt(x); - assertTrue(result >= floor * 10 ** 9 && result <= (floor + 1) * 10 ** 9); - assertEq(result / 10 ** 9, floor); - } - - function testSqrtBack(uint256 x) public { - unchecked { - x >>= 128; - while (x != 0) { - assertEq(FixedPointMathLib.sqrt(x * x), x); - x >>= 1; - } - } - } - - function testSqrtHashed(uint256 x) public { - testSqrtBack(uint256(keccak256(abi.encode(x)))); - } - - function testSqrtHashedSingle() public { - testSqrtHashed(123); - } - - function testMin(uint256 x, uint256 y) public { - uint256 z = x < y ? x : y; - assertEq(FixedPointMathLib.min(x, y), z); - } - - function testMinBrutalized(uint256 x, uint256 y) public { - uint32 xCasted; - uint32 yCasted; - /// @solidity memory-safe-assembly - assembly { - xCasted := x - yCasted := y - } - uint256 expected = xCasted < yCasted ? xCasted : yCasted; - assertEq(FixedPointMathLib.min(xCasted, yCasted), expected); - assertEq(FixedPointMathLib.min(uint32(x), uint32(y)), expected); - expected = uint32(x) < uint32(y) ? uint32(x) : uint32(y); - assertEq(FixedPointMathLib.min(xCasted, yCasted), expected); - } - - function testMinSigned(int256 x, int256 y) public { - int256 z = x < y ? x : y; - assertEq(FixedPointMathLib.min(x, y), z); - } - - function testMax(uint256 x, uint256 y) public { - uint256 z = x > y ? x : y; - assertEq(FixedPointMathLib.max(x, y), z); - } - - function testMaxSigned(int256 x, int256 y) public { - int256 z = x > y ? x : y; - assertEq(FixedPointMathLib.max(x, y), z); - } - - function testMaxCasted(uint32 x, uint32 y, uint256 brutalizer) public { - uint32 z = x > y ? x : y; - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, brutalizer) - mstore(0x20, 1) - x := or(shl(32, keccak256(0x00, 0x40)), x) - mstore(0x20, 2) - y := or(shl(32, keccak256(0x00, 0x40)), y) - } - assertTrue(FixedPointMathLib.max(x, y) == z); - } - - function testZeroFloorSub(uint256 x, uint256 y) public { - uint256 z = x > y ? x - y : 0; - assertEq(FixedPointMathLib.zeroFloorSub(x, y), z); - } - - function testZeroFloorSubCasted(uint32 x, uint32 y, uint256 brutalizer) public { - uint256 z = x > y ? x - y : 0; - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, brutalizer) - mstore(0x20, 1) - x := or(shl(32, keccak256(0x00, 0x40)), x) - mstore(0x20, 2) - y := or(shl(32, keccak256(0x00, 0x40)), y) - } - assertTrue(FixedPointMathLib.zeroFloorSub(x, y) == z); - } - - function testDist(int256 x, int256 y) public { - uint256 z; - unchecked { - if (x > y) { - z = uint256(x - y); - } else { - z = uint256(y - x); - } - } - assertEq(FixedPointMathLib.dist(x, y), z); - } - - function testAbs(int256 x) public { - uint256 z = uint256(x); - if (x < 0) { - if (x == type(int256).min) { - z = uint256(type(int256).max) + 1; - } else { - z = uint256(-x); - } - } - assertEq(FixedPointMathLib.abs(x), z); - } - - function testGcd(uint256 x, uint256 y) public { - assertEq(FixedPointMathLib.gcd(x, y), _gcd(x, y)); - } - - function testClamp(uint256 x, uint256 minValue, uint256 maxValue) public { - uint256 clamped = x; - if (clamped < minValue) { - clamped = minValue; - } - if (clamped > maxValue) { - clamped = maxValue; - } - assertEq(FixedPointMathLib.clamp(x, minValue, maxValue), clamped); - } - - function testClampSigned(int256 x, int256 minValue, int256 maxValue) public { - int256 clamped = x; - if (clamped < minValue) { - clamped = minValue; - } - if (clamped > maxValue) { - clamped = maxValue; - } - assertEq(FixedPointMathLib.clamp(x, minValue, maxValue), clamped); - } - - function testFactorial() public { - uint256 result = 1; - assertEq(FixedPointMathLib.factorial(0), result); - unchecked { - for (uint256 i = 1; i != 58; ++i) { - result = result * i; - assertEq(FixedPointMathLib.factorial(i), result); - } - } - vm.expectRevert(FixedPointMathLib.FactorialOverflow.selector); - FixedPointMathLib.factorial(58); - } - - function testFactorialOriginal() public { - uint256 result = 1; - assertEq(_factorialOriginal(0), result); - unchecked { - for (uint256 i = 1; i != 58; ++i) { - result = result * i; - assertEq(_factorialOriginal(i), result); - } - } - } - - function _factorialOriginal(uint256 x) internal pure returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - result := 1 - for {} x {} { - result := mul(result, x) - x := sub(x, 1) - } - } - } - - function _gcd(uint256 x, uint256 y) internal pure returns (uint256 result) { - if (y == 0) { - return x; - } else { - return _gcd(y, x % y); - } - } - - function testRawAdd(uint256 x, uint256 y) public { - uint256 z; - /// @solidity memory-safe-assembly - assembly { - z := add(x, y) - } - assertEq(FixedPointMathLib.rawAdd(x, y), z); - } - - function testRawAdd(int256 x, int256 y) public { - int256 z; - /// @solidity memory-safe-assembly - assembly { - z := add(x, y) - } - assertEq(FixedPointMathLib.rawAdd(x, y), z); - } - - function testRawSub(uint256 x, uint256 y) public { - uint256 z; - /// @solidity memory-safe-assembly - assembly { - z := sub(x, y) - } - assertEq(FixedPointMathLib.rawSub(x, y), z); - } - - function testRawSub(int256 x, int256 y) public { - int256 z; - /// @solidity memory-safe-assembly - assembly { - z := sub(x, y) - } - assertEq(FixedPointMathLib.rawSub(x, y), z); - } - - function testRawMul(uint256 x, uint256 y) public { - uint256 z; - /// @solidity memory-safe-assembly - assembly { - z := mul(x, y) - } - assertEq(FixedPointMathLib.rawMul(x, y), z); - } - - function testRawMul(int256 x, int256 y) public { - int256 z; - /// @solidity memory-safe-assembly - assembly { - z := mul(x, y) - } - assertEq(FixedPointMathLib.rawMul(x, y), z); - } - - function testRawDiv(uint256 x, uint256 y) public { - uint256 z; - /// @solidity memory-safe-assembly - assembly { - z := div(x, y) - } - assertEq(FixedPointMathLib.rawDiv(x, y), z); - } - - function testRawSDiv(int256 x, int256 y) public { - int256 z; - /// @solidity memory-safe-assembly - assembly { - z := sdiv(x, y) - } - assertEq(FixedPointMathLib.rawSDiv(x, y), z); - } - - function testRawMod(uint256 x, uint256 y) public { - uint256 z; - /// @solidity memory-safe-assembly - assembly { - z := mod(x, y) - } - assertEq(FixedPointMathLib.rawMod(x, y), z); - } - - function testRawSMod(int256 x, int256 y) public { - int256 z; - /// @solidity memory-safe-assembly - assembly { - z := smod(x, y) - } - assertEq(FixedPointMathLib.rawSMod(x, y), z); - } - - function testRawAddMod(uint256 x, uint256 y, uint256 denominator) public { - uint256 z; - /// @solidity memory-safe-assembly - assembly { - z := addmod(x, y, denominator) - } - assertEq(FixedPointMathLib.rawAddMod(x, y, denominator), z); - } - - function testRawMulMod(uint256 x, uint256 y, uint256 denominator) public { - uint256 z; - /// @solidity memory-safe-assembly - assembly { - z := mulmod(x, y, denominator) - } - assertEq(FixedPointMathLib.rawMulMod(x, y, denominator), z); - } - - function testLog10() public { - assertEq(FixedPointMathLib.log10(0), 0); - assertEq(FixedPointMathLib.log10(1), 0); - assertEq(FixedPointMathLib.log10(type(uint256).max), 77); - unchecked { - for (uint256 i = 1; i <= 77; ++i) { - uint256 x = 10 ** i; - assertEq(FixedPointMathLib.log10(x), i); - assertEq(FixedPointMathLib.log10(x - 1), i - 1); - assertEq(FixedPointMathLib.log10(x + 1), i); - } - } - } - - function testLog10(uint256 i, uint256 j) public { - i = _bound(i, 0, 77); - uint256 low = 10 ** i; - uint256 high = i == 77 ? type(uint256).max : (10 ** (i + 1)) - 1; - uint256 x = _bound(j, low, high); - assertEq(FixedPointMathLib.log10(x), i); - } - - function testLog10Up() public { - assertEq(FixedPointMathLib.log10Up(0), 0); - assertEq(FixedPointMathLib.log10Up(1), 0); - assertEq(FixedPointMathLib.log10Up(9), 1); - assertEq(FixedPointMathLib.log10Up(10), 1); - assertEq(FixedPointMathLib.log10Up(99), 2); - assertEq(FixedPointMathLib.log10Up(100), 2); - assertEq(FixedPointMathLib.log10Up(999), 3); - assertEq(FixedPointMathLib.log10Up(1000), 3); - assertEq(FixedPointMathLib.log10Up(10 ** 77), 77); - assertEq(FixedPointMathLib.log10Up(10 ** 77 + 1), 78); - assertEq(FixedPointMathLib.log10Up(type(uint256).max), 78); - } - - function testLog256() public { - assertEq(FixedPointMathLib.log256(0), 0); - assertEq(FixedPointMathLib.log256(1), 0); - assertEq(FixedPointMathLib.log256(256), 1); - assertEq(FixedPointMathLib.log256(type(uint256).max), 31); - unchecked { - for (uint256 i = 1; i <= 31; ++i) { - uint256 x = 256 ** i; - assertEq(FixedPointMathLib.log256(x), i); - assertEq(FixedPointMathLib.log256(x - 1), i - 1); - assertEq(FixedPointMathLib.log256(x + 1), i); - } - } - } - - function testLog256(uint256 i, uint256 j) public { - i = _bound(i, 0, 31); - uint256 low = 256 ** i; - uint256 high = i == 31 ? type(uint256).max : (256 ** (i + 1)) - 1; - uint256 x = _bound(j, low, high); - assertEq(FixedPointMathLib.log256(x), i); - } - - function testLog256Up() public { - assertEq(FixedPointMathLib.log256Up(0), 0); - assertEq(FixedPointMathLib.log256Up(0x01), 0); - assertEq(FixedPointMathLib.log256Up(0x02), 1); - assertEq(FixedPointMathLib.log256Up(0xff), 1); - assertEq(FixedPointMathLib.log256Up(0x0100), 1); - assertEq(FixedPointMathLib.log256Up(0x0101), 2); - assertEq(FixedPointMathLib.log256Up(0xffff), 2); - assertEq(FixedPointMathLib.log256Up(0x010000), 2); - assertEq(FixedPointMathLib.log256Up(0x010001), 3); - assertEq(FixedPointMathLib.log256Up(type(uint256).max - 1), 32); - assertEq(FixedPointMathLib.log256Up(type(uint256).max), 32); - } - - function testSci() public { - _testSci(0, 0, 0); - _testSci(1, 1, 0); - _testSci(13, 13, 0); - _testSci(130, 13, 1); - _testSci(1300, 13, 2); - unchecked { - uint256 a = 103; - uint256 exponent = 0; - uint256 m = 1; - uint256 n = 78 - FixedPointMathLib.log10Up(a); - for (uint256 i; i < n; ++i) { - _testSci(a * m, a, exponent); - exponent += 1; - m *= 10; - } - } - _testSci(10 ** 77, 1, 77); - _testSci(2 * (10 ** 76), 2, 76); - _testSci(9 * (10 ** 76), 9, 76); - unchecked { - for (uint256 i; i < 32; ++i) { - testSci(11 + i * i * 100); - } - for (uint256 i; i < 500; ++i) { - _testSci(0, 0, 0); - } - } - unchecked { - uint256 x = 30000000000000000000000000000000000000000000000001; - _testSci(x, x, 0); - } - } - - function testSci(uint256 a) public { - unchecked { - while (a % 10 == 0) a = _random(); - uint256 exponent = 0; - uint256 m = 1; - uint256 n = 78 - FixedPointMathLib.log10Up(a); - for (uint256 i; i < n; ++i) { - _testSci(a * m, a, exponent); - uint256 x = a * 10 ** exponent; - assertEq(x, a * m); - exponent += 1; - m *= 10; - } - } - } - - function testSci2(uint256 x) public { - unchecked { - (uint256 mantissa, uint256 exponent) = FixedPointMathLib.sci(x); - assertEq(x % 10 ** exponent, 0); - if (x != 0) { - assertTrue(x % 10 ** (exponent + 1) > 0); - assertTrue(mantissa % 10 != 0); - } else { - assertEq(mantissa, 0); - assertEq(exponent, 0); - } - } - } - - function _testSci(uint256 x, uint256 expectedMantissa, uint256 expectedExponent) internal { - (uint256 mantissa, uint256 exponent) = FixedPointMathLib.sci(x); - assertEq(mantissa, expectedMantissa); - assertEq(exponent, expectedExponent); - } - - function testPackUnpackSci(uint256) public { - unchecked { - uint256 x = (_random() & 0x1) * 10 ** (_random() % 70); - uint8 packed = uint8(FixedPointMathLib.packSci(x)); - uint256 unpacked = FixedPointMathLib.unpackSci(packed); - assertEq(unpacked, x); - } - unchecked { - uint256 x = (_random() & 0x1ff) * 10 ** (_random() % 70); - uint16 packed = uint16(FixedPointMathLib.packSci(x)); - uint256 unpacked = FixedPointMathLib.unpackSci(packed); - assertEq(unpacked, x); - } - unchecked { - uint256 x = (_random() & 0x1ffffff) * 10 ** (_random() % 70); - uint32 packed = uint32(FixedPointMathLib.packSci(x)); - uint256 unpacked = FixedPointMathLib.unpackSci(packed); - assertEq(unpacked, x); - } - unchecked { - uint256 x = (_random() & 0x1ffffffffffffff) * 10 ** (_random() % 60); - uint64 packed = uint64(FixedPointMathLib.packSci(x)); - uint256 unpacked = FixedPointMathLib.unpackSci(packed); - assertEq(unpacked, x); - } - unchecked { - uint256 x = (_random() * 10 ** (_random() % 78)) & ((1 << 249) - 1); - uint256 packed = FixedPointMathLib.packSci(x); - uint256 unpacked = FixedPointMathLib.unpackSci(packed); - assertEq(unpacked, x); - } - } - - function testPackUnpackSci() public { - uint256 mantissaSize = 249; - unchecked { - for (uint256 i; i <= mantissaSize; ++i) { - uint256 x = (1 << i) - 1; - uint256 packed = FixedPointMathLib.packSci(x); - uint256 unpacked = FixedPointMathLib.unpackSci(packed); - assertEq(unpacked, x); - } - } - unchecked { - uint256 x = (1 << (mantissaSize + 1)) - 1; - vm.expectRevert(FixedPointMathLib.MantissaOverflow.selector); - FixedPointMathLib.packSci(x); - } - } -} diff --git a/lib/solady/test/GasBurnerLib.t.sol b/lib/solady/test/GasBurnerLib.t.sol deleted file mode 100644 index aa92057..0000000 --- a/lib/solady/test/GasBurnerLib.t.sol +++ /dev/null @@ -1,31 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {GasBurnerLib} from "../src/utils/GasBurnerLib.sol"; - -contract GasBurnerLibTest is SoladyTest { - event LogGasBurn(uint256 required, uint256 actual); - - function testBurnGas() public { - _testBurnGas(0); - _testBurnGas(1); - _testBurnGas(110); - _testBurnGas(119); - _testBurnGas(120); - _testBurnGas(121); - _testBurnGas(300); - for (uint256 x = 300; x < 9000; x += 32) { - _testBurnGas(x); - } - } - - function _testBurnGas(uint256 x) internal { - unchecked { - uint256 gasBefore = gasleft(); - GasBurnerLib.burn(x); - uint256 gasAfter = gasleft(); - emit LogGasBurn(x, gasBefore - gasAfter); - } - } -} diff --git a/lib/solady/test/JSONParserLib.t.sol b/lib/solady/test/JSONParserLib.t.sol deleted file mode 100644 index e6adf97..0000000 --- a/lib/solady/test/JSONParserLib.t.sol +++ /dev/null @@ -1,788 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {JSONParserLib} from "../src/utils/JSONParserLib.sol"; -import {LibString} from "../src/utils/LibString.sol"; -import {Base64} from "../src/utils/Base64.sol"; - -contract JSONParserLibTest is SoladyTest { - using JSONParserLib for *; - - function testParseInvalidReverts() public { - _checkParseReverts(""); - _checkParseReverts("e"); - _checkParseReverts("abc"); - _checkParseReverts("1,2"); - _checkParseReverts("["); - _checkParseReverts("]"); - _checkParseReverts("{"); - _checkParseReverts("}"); - _checkParseReverts("[[]"); - _checkParseReverts("[]["); - _checkParseReverts("[][]"); - _checkParseReverts("[],[]"); - _checkParseReverts("[1,2"); - _checkParseReverts("1,2]"); - _checkParseReverts("[1"); - _checkParseReverts("1]"); - _checkParseReverts("[1,"); - _checkParseReverts("{}{"); - _checkParseReverts("{}{}"); - _checkParseReverts("{},{}"); - _checkParseReverts("{]"); - _checkParseReverts("{{}"); - _checkParseReverts("{}}"); - _checkParseReverts("[,]"); - _checkParseReverts("[0,]"); - _checkParseReverts("[0,1,]"); - _checkParseReverts("[0,,]"); - _checkParseReverts("[0,,1]"); - _checkParseReverts("[,0]"); - _checkParseReverts("{,}"); - _checkParseReverts('{"a"}'); - _checkParseReverts('{"a":"A",}'); - _checkParseReverts('{"a":"A","b":"B",}'); - _checkParseReverts('{"a":"A"b":"B"}'); - _checkParseReverts('{"a":"A",,"b":"B"}'); - _checkParseReverts('{,"a":"A","b":"B"}'); - _checkParseReverts('{"a"::"A"}'); - _checkParseReverts('{"a","A"}'); - _checkParseReverts("{1}"); - _checkParseReverts("{:}"); - } - - function testParseInvalidNumberReverts() public { - _checkParseReverts("01234567890"); - _checkParseReverts("-1.234567890e-a"); - _checkParseReverts("-1.234567890e-"); - _checkParseReverts("-1.234567890e+a"); - _checkParseReverts("-1.234567890e+"); - _checkParseReverts("-1.234567890z"); - _checkParseReverts("-1.234567890e"); - _checkParseReverts("-00.234567890"); - _checkParseReverts("-.234567890"); - _checkParseReverts("-00"); - _checkParseReverts("--0"); - _checkParseReverts("00"); - _checkParseReverts("0."); - _checkParseReverts("0..12"); - _checkParseReverts("0.0e"); - _checkParseReverts("."); - _checkParseReverts("-"); - _checkParseReverts("+"); - _checkParseReverts("e"); - _checkParseReverts("+123"); - _checkParseReverts(".123"); - _checkParseReverts("e123"); - _checkParseReverts("1 e 1"); - _checkParseReverts("-1.e+0"); - _checkParseReverts("0x"); - } - - function _checkParseReverts(string memory trimmed) internal { - vm.expectRevert(JSONParserLib.ParsingFailed.selector); - this.parsedValue(trimmed); - string memory s = trimmed; - for (uint256 i; i != 4; ++i) { - vm.expectRevert(JSONParserLib.ParsingFailed.selector); - this.parsedValue(_padWhiteSpace(s, i)); - } - } - - function parsedValue(string memory s) public view miniBrutalizeMemory returns (string memory) { - s = s.parse().value(); - _checkMemory(s); - return s; - } - - function testParseNumber() public { - _checkParseNumber("0"); - _checkParseNumber("-0"); - _checkParseNumber("-1.2e+0"); - _checkParseNumber("-1.2e+00"); - _checkParseNumber("-1.2e+001"); - _checkParseNumber("-1.2e+22"); - _checkParseNumber("-1.2e-22"); - _checkParseNumber("-1.2e22"); - _checkParseNumber("0.1"); - _checkParseNumber("0.123"); - _checkParseNumber("12345678901234567890123456789012345678901234567890"); - _checkParseNumber("12345e12345678901234567890123456789012345678901234567890"); - _checkParseNumber("1234567890"); - _checkParseNumber("123"); - _checkParseNumber("1"); - } - - function _checkParseNumber(string memory trimmed) internal { - _checkSoloNumber(trimmed.parse(), trimmed); - string memory s = trimmed; - for (uint256 i; i != 4; ++i) { - _checkSoloNumber(_padWhiteSpace(s, i).parse(), trimmed); - } - } - - function _checkSoloNumber(JSONParserLib.Item memory item, string memory trimmed) internal { - for (uint256 i; i != 2; ++i) { - assertEq(item.getType(), JSONParserLib.TYPE_NUMBER); - assertEq(item.isNumber(), true); - assertEq(item.value(), trimmed); - _checkItemIsSolo(item); - } - } - - function testParseEmptyArrays() public { - _checkParseEmptyArray("[]"); - _checkParseEmptyArray("[ ]"); - _checkParseEmptyArray("[ ]"); - } - - function _checkParseEmptyArray(string memory trimmed) internal { - _checkSoloEmptyArray(trimmed.parse(), trimmed); - string memory s = trimmed; - for (uint256 i; i != 16; ++i) { - _checkSoloEmptyArray(_padWhiteSpace(s, i).parse(), trimmed); - } - } - - function _checkSoloEmptyArray(JSONParserLib.Item memory item, string memory trimmed) internal { - for (uint256 i; i != 2; ++i) { - assertEq(item.getType(), JSONParserLib.TYPE_ARRAY); - assertEq(item.isArray(), true); - assertEq(item.value(), trimmed); - _checkItemIsSolo(item); - } - } - - function testParseEmptyObjects() public { - _checkParseEmptyObject("{}"); - _checkParseEmptyObject("{ }"); - _checkParseEmptyObject("{ }"); - } - - function _checkParseEmptyObject(string memory trimmed) internal { - _checkSoloEmptyObject(trimmed.parse(), trimmed); - string memory s = trimmed; - for (uint256 i; i != 16; ++i) { - _checkSoloEmptyObject(_padWhiteSpace(s, i).parse(), trimmed); - } - } - - function _checkSoloEmptyObject(JSONParserLib.Item memory item, string memory trimmed) - internal - { - for (uint256 i; i != 2; ++i) { - assertEq(item.getType(), JSONParserLib.TYPE_OBJECT); - assertEq(item.isObject(), true); - assertEq(item.value(), trimmed); - _checkItemIsSolo(item); - } - } - - function _padWhiteSpace(string memory s, uint256 r) internal pure returns (string memory) { - unchecked { - uint256 q = r; - r = r % 3; - string memory p = r == 0 ? " " : r == 1 ? "\t" : r == 2 ? "\r" : "\n"; - q = 1 + q / 3; - for (uint256 i; i != q; ++i) { - s = string(abi.encodePacked(p, s, p)); - } - return s; - } - } - - function testParseSimpleUintArray() public { - string memory s; - JSONParserLib.Item memory item; - - for (uint256 k; k != 9; ++k) { - uint256 o = k == 0 ? 0 : 1 << (17 * k); - string memory p = _padWhiteSpace("", k); - for (uint256 j; j != 5; ++j) { - s = "["; - for (uint256 i; i != j; ++i) { - string memory x = LibString.toString(o + i); - if (i == 0) { - s = string(abi.encodePacked(s, p, x)); - } else { - s = string(abi.encodePacked(s, p, ",", p, x)); - } - } - s = string(abi.encodePacked(s, "]")); - item = s.parse(); - assertEq(item.isArray(), true); - assertEq(item.size(), j); - for (uint256 i; i != j; ++i) { - string memory x = LibString.toString(o + i); - assertEq(item.children()[i].value(), x); - assertEq(item.children()[i].parent()._data, item._data); - assertEq(item.children()[i].parent().isArray(), true); - assertEq(item.at(i)._data, item.children()[i]._data); - assertEq(item.at(LibString.toString(i))._data, 0); - } - } - } - } - - function testEmptyItem() public { - JSONParserLib.Item memory item; - assertEq(item.value(), ""); - assertEq(item.isUndefined(), true); - assertEq(item.parent().isUndefined(), true); - assertEq(item.parent().parent().isUndefined(), true); - assertEq(item.key(), ""); - assertEq(item.at(0).isUndefined(), true); - assertEq(item.at(0).at(0).isUndefined(), true); - } - - function testParseSimpleArray() public { - string memory s = '["hehe",12,"haha"]'; - JSONParserLib.Item memory item = s.parse(); - - assertEq(item.isArray(), true); - assertEq(item.size(), 3); - _checkItemHasNoParent(item); - - assertEq(item.children()[0].value(), '"hehe"'); - assertEq(item.children()[0].index(), 0); - assertEq(item.children()[0].getType(), JSONParserLib.TYPE_STRING); - assertEq(item.children()[0].key(), ""); - assertEq(item.children()[0].parent()._data, item._data); - assertEq(item.children()[0].parent().isArray(), true); - - assertEq(item.children()[1].value(), "12"); - assertEq(item.children()[1].index(), 1); - assertEq(item.children()[1].key(), ""); - assertEq(item.children()[1].getType(), JSONParserLib.TYPE_NUMBER); - assertEq(item.children()[1].parent()._data, item._data); - assertEq(item.children()[1].parent().isArray(), true); - - assertEq(item.children()[2].value(), '"haha"'); - assertEq(item.children()[2].index(), 2); - assertEq(item.children()[2].getType(), JSONParserLib.TYPE_STRING); - assertEq(item.children()[2].key(), ""); - assertEq(item.children()[2].parent()._data, item._data); - assertEq(item.children()[2].parent().isArray(), true); - - assertEq(item.at(0)._data, item.children()[0]._data); - assertEq(item.at(1)._data, item.children()[1]._data); - assertEq(item.at(2)._data, item.children()[2]._data); - assertEq(item.at(3)._data, 0); - } - - function testParseSpecials() public miniBrutalizeMemory { - string memory s; - JSONParserLib.Item memory item; - - for (uint256 k; k < 9; ++k) { - s = _padWhiteSpace("true", k); - item = s.parse(); - assertEq(item.getType(), JSONParserLib.TYPE_BOOLEAN); - assertEq(item.isBoolean(), true); - assertEq(item.isNull(), false); - assertEq(item.value(), "true"); - assertEq(item.parent().isUndefined(), true); - _checkItemIsSolo(item); - - s = _padWhiteSpace("false", k); - item = s.parse(); - assertEq(item.getType(), JSONParserLib.TYPE_BOOLEAN); - assertEq(item.isBoolean(), true); - assertEq(item.isNull(), false); - assertEq(item.value(), "false"); - _checkItemIsSolo(item); - - s = _padWhiteSpace("null", k); - item = s.parse(); - assertEq(item.getType(), JSONParserLib.TYPE_NULL); - assertEq(item.isBoolean(), false); - assertEq(item.isNull(), true); - assertEq(item.value(), "null"); - _checkItemIsSolo(item); - } - - for (uint256 k; k != 4; ++k) { - if (k == 0) s = "[true,false,null]"; - if (k == 1) s = "[ true , false , null ]"; - if (k == 2) s = '{"A":true,"B":false,"C":null}'; - if (k == 3) s = '{ "A" : true , "B" : false , "C" : null }'; - item = s.parse(); - assertEq(item.size(), 3); - assertEq(item.children()[0].getType(), JSONParserLib.TYPE_BOOLEAN); - assertEq(item.children()[0].value(), "true"); - assertEq(item.children()[1].getType(), JSONParserLib.TYPE_BOOLEAN); - assertEq(item.children()[1].value(), "false"); - assertEq(item.children()[2].getType(), JSONParserLib.TYPE_NULL); - assertEq(item.children()[2].value(), "null"); - if (k == 0 || k == 1) { - for (uint256 i; i != 3; ++i) { - assertEq(item.children()[i].parent()._data, item._data); - assertEq(item.children()[i].parent().isArray(), true); - assertEq(item.children()[i].parent().isArray(), true); - assertEq(item.children()[i].index(), i); - assertEq(item.children()[i].key(), ""); - } - } - if (k == 2 || k == 3) { - for (uint256 i; i != 3; ++i) { - assertEq(item.children()[i].parent()._data, item._data); - assertEq(item.children()[i].parent().isObject(), true); - assertEq(item.children()[i].index(), 0); - } - assertEq(item.children()[0].key(), '"A"'); - assertEq(item.children()[1].key(), '"B"'); - assertEq(item.children()[2].key(), '"C"'); - } - } - } - - function testParseObject() public { - string memory s; - JSONParserLib.Item memory item; - - s = '{"b": "B", "_": "x", "hehe": "HEHE", "_": "y", "v": 12345, "_": "z"}'; - item = s.parse(); - - assertEq(item.isObject(), true); - assertEq(item.size(), 6); - - for (uint256 i; i < item.size(); ++i) { - assertEq(item.at(i).isUndefined(), true); - assertEq(item.children()[i].parent()._data, item._data); - } - assertEq(item.at('"_"').value(), '"z"'); - assertEq(item.at('"b"').value(), '"B"'); - assertEq(item.at('"v"').value(), "12345"); - assertEq(item.at('"hehe"').value(), '"HEHE"'); - assertEq(item.at('"m"').value(), ""); - assertEq(item.at('"m"').isUndefined(), true); - } - - function testParseValidObjectDoesNotRevert(string memory key, string memory value) public { - _limitStringLength(key); - _limitStringLength(value); - string memory s = string( - abi.encodePacked( - '{"', LibString.escapeJSON(key), '":"', LibString.escapeJSON(value), '"}' - ) - ); - this.parsedValue(s); - } - - function testParseRecursiveObject() public miniBrutalizeMemory { - string memory s; - JSONParserLib.Item memory item; - - s = '{ "": [1,2, {"m": "M"}, {},[]], "X": {"hehe": "1", "h": [true,false, null], "": 0} }'; - item = s.parse(); - - assertEq(item.isObject(), true); - assertEq(item.children()[0].key(), '""'); - assertEq(item.children()[0].index(), 0); - assertEq(item.children()[0].children()[0].value(), "1"); - assertEq(item.children()[0].children()[1].value(), "2"); - assertEq(item.children()[0].children()[2].value(), '{"m": "M"}'); - assertEq(item.children()[0].children()[2].children()[0].key(), '"m"'); - assertEq(item.children()[0].children()[2].children()[0].value(), '"M"'); - - JSONParserLib.Item memory c = item.children()[0].children()[2].children()[0]; - assertEq(c.parent().parent().parent()._data, item._data); - assertEq(c.parent().parent().parent().value(), item.value()); - assertEq(c.parent().parent().parent().parent().isUndefined(), true); - - assertEq(item.children()[1].key(), '"X"'); - assertEq(item.children()[1].index(), 0); - assertEq(item.children()[1].value(), '{"hehe": "1", "h": [true,false, null], "": 0}'); - assertEq(item.children()[1].children()[0].key(), '"hehe"'); - assertEq(item.children()[1].children()[0].value(), '"1"'); - - assertEq(item.children()[1].children()[1].key(), '"h"'); - assertEq(item.children()[1].children()[1].value(), "[true,false, null]"); - assertEq(item.children()[1].children()[1].children()[0].value(), "true"); - assertEq(item.children()[1].children()[1].children()[0].isBoolean(), true); - assertEq(item.children()[1].children()[1].children()[1].value(), "false"); - assertEq(item.children()[1].children()[1].children()[1].isBoolean(), true); - assertEq(item.children()[1].children()[1].children()[2].value(), "null"); - assertEq(item.children()[1].children()[1].children()[2].isNull(), true); - - assertEq(item.children()[1].children()[2].key(), '""'); - assertEq(item.children()[1].children()[2].value(), "0"); - assertEq(item.children()[1].children()[2].size(), 0); - - s = "[[[[[[[]]]]]]]"; - item = s.parse(); - assertEq(item.isArray(), true); - - s = '{"a":[[[{"z":"Z"}]]]}'; - item = s.parse(); - assertEq(item.isObject(), true); - } - - function testParseString() public { - _checkParseString('""'); - _checkParseString('"a"'); - _checkParseString('"ab"'); - _checkParseString('"012345678901234567890123456789"'); - _checkParseString('"0123456789012345678901234567890"'); - _checkParseString('"01234567890123456789012345678901"'); - _checkParseString('"012345678901234567890123456789012"'); - _checkParseString('"0123456789012345678901234567890123"'); - _checkParseString('" d"'); - _checkParseString('"d "'); - _checkParseString('" d "'); - _checkParseString('"\\""'); - _checkParseString('"\\\\"'); - _checkParseString('"\\/"'); - _checkParseString('"\\b"'); - _checkParseString('"\\f"'); - _checkParseString('"\\n"'); - _checkParseString('"\\r"'); - _checkParseString('"\\t"'); - _checkParseString('" \\u1234 \\"\\"\\\\ \\b\\f \\n\\r "'); - _checkParseString('"\\u1234"'); - _checkParseString('"\\uabcd"'); - _checkParseString('"\\uABCD"'); - _checkParseString('"\\uef00"'); - _checkParseString('"\\u00EF"'); - _checkParseString('"\\u1234 "'); - _checkParseString('"\\uabcd "'); - _checkParseString('"\\uABCD "'); - _checkParseString('"\\uef00 "'); - _checkParseString('"\\u00EF "'); - } - - function _checkParseString(string memory s) internal { - JSONParserLib.Item memory item; - assertEq(this.parsedValue(s), s); - for (uint256 k; k != 4; ++k) { - item = _padWhiteSpace(s, k).parse(); - assertEq(item.value(), s); - assertEq(item.isString(), true); - assertEq(item.value(), s); - _checkItemIsSolo(item); - } - } - - function testParseInvalidStringReverts() public { - _checkParseReverts('"'); - _checkParseReverts('"""'); - _checkParseReverts('""""'); - _checkParseReverts('"""""'); - _checkParseReverts('"abc" "'); - _checkParseReverts('"abc" ""'); - _checkParseReverts('"abc""abc"'); - _checkParseReverts('"\\"'); - _checkParseReverts('"\\\\\\"'); - _checkParseReverts('"\\u"'); - _checkParseReverts('"\\u1"'); - _checkParseReverts('"\\u12"'); - _checkParseReverts('"\\u123"'); - _checkParseReverts('"\\uxxxx"'); - _checkParseReverts('"\\u012g"'); - _checkParseReverts('"\\u1234'); - } - - function _checkItemIsSolo(JSONParserLib.Item memory item) internal { - _checkItemHasNoParent(item); - assertEq(item.size(), 0); - } - - function _checkItemHasNoParent(JSONParserLib.Item memory item) internal { - assertEq(item.parent().isUndefined(), true); - assertEq(item.parent()._data, 0); - assertEq(item.key(), ""); - assertEq(item.index(), 0); - assertEq(item.parent().isObject(), false); - assertEq(item.parent().isArray(), false); - assertEq(item.isUndefined(), false); - } - - function testParseGas() public { - string memory s = - '{"animation_url":"","artist":"Daniel Allan","artwork":{"mimeType":"image/gif","uri":"ar://J5NZ-e2NUcQj1OuuhpTjAKtdW_nqwnwo5FypF_a6dE4","nft":null},"attributes":[{"trait_type":"Criteria","value":"Song Edition"}],"bpm":null,"credits":null,"description":"Criteria is an 8-track project between Daniel Allan and Reo Cragun.\\n\\nA fusion of electronic music and hip-hop - Criteria brings together the best of both worlds and is meant to bring web3 music to a wider audience.\\n\\nThe collection consists of 2500 editions with activations across Sound, Bonfire, OnCyber, Spinamp and Arpeggi.","duration":105,"external_url":"https://www.sound.xyz/danielallan/criteria","genre":"Pop","image":"ar://J5NZ-e2NUcQj1OuuhpTjAKtdW_nqwnwo5FypF_a6dE4","isrc":null,"key":null,"license":null,"locationCreated":null,"losslessAudio":"","lyrics":null,"mimeType":"audio/wave","nftSerialNumber":11,"name":"Criteria #11","originalReleaseDate":null,"project":null,"publisher":null,"recordLabel":null,"tags":null,"title":"Criteria","trackNumber":1,"version":"sound-edition-20220930","visualizer":null}'; - JSONParserLib.Item memory item = s.parse(); - assertEq(item.isObject(), true); - bytes32 expectedHash = 0x6c3276c7005f50c82624fb28f9748f0fb6f0b364234e4823178f964315b41567; - assertEq(keccak256(bytes(item.at('"description"').value())), expectedHash); - } - - function testParseUintFromHex() public { - unchecked { - for (uint256 i; i != 9; ++i) { - _checkParseUintFromHex(LibString.toString(i), i); - } - } - _checkParseUintFromHex("a", 0xa); - _checkParseUintFromHex("f", 0xf); - _checkParseUintFromHex("ff", 0xff); - _checkParseUintFromHex("fff", 0xfff); - _checkParseUintFromHex( - "ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff", type(uint256).max - ); - _checkParseUintFromHex( - "1234567890abcdef1234567890abcdef1234567890abcdef1234567890abcdef", - 0x1234567890abcdef1234567890abcdef1234567890abcdef1234567890abcdef - ); - _checkParseUintFromHex( - "fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff", - 0xfffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff - ); - _checkParseUintFromHex( - "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef", - 0x0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef - ); - } - - function _checkParseUintFromHex(string memory s, uint256 x) internal { - _checkParseUintFromHexSub(LibString.lower(s), x); - _checkParseUintFromHexSub(LibString.upper(s), x); - } - - function _checkParseUintFromHexSub(string memory s, uint256 x) internal { - assertEq(this.parseUintFromHex(s), x); - assertEq(this.parseUintFromHex(LibString.concat("0x", s)), x); - assertEq(this.parseUintFromHex(LibString.concat("0x0", s)), x); - assertEq(this.parseUintFromHex(LibString.concat("0X00", s)), x); - assertEq(this.parseUintFromHex(LibString.concat("0x000", s)), x); - assertEq(this.parseUintFromHex(LibString.concat("0X", s)), x); - assertEq(this.parseUintFromHex(LibString.concat("0", s)), x); - assertEq(this.parseUintFromHex(LibString.concat("00", s)), x); - assertEq(this.parseUintFromHex(LibString.concat("000", s)), x); - } - - function parseUintFromHex(string memory s) public pure returns (uint256) { - return s.parseUintFromHex(); - } - - function testParseInvalidUintFromHexReverts() public { - _checkParseInvalidUintFromHexReverts(""); - _checkParseInvalidUintFromHexReverts("+"); - _checkParseInvalidUintFromHexReverts(" 0"); - _checkParseInvalidUintFromHexReverts("0 "); - _checkParseInvalidUintFromHexReverts(" 12"); - _checkParseInvalidUintFromHexReverts("00x12"); - _checkParseInvalidUintFromHexReverts(" 0x12"); - _checkParseInvalidUintFromHexReverts("-0x12"); - _checkParseInvalidUintFromHexReverts("0x123g"); - _checkParseInvalidUintFromHexReverts("123g"); - _checkParseInvalidUintFromHexReverts("z"); - _checkParseInvalidUintFromHexReverts( - "1ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" - ); - _checkParseInvalidUintFromHexReverts( - "10000000000000000000000000000000000000000000000000000000000000000" - ); - _checkParseInvalidUintFromHexReverts( - "ff0000000000000000000000000000000000000000000000000000000000000000" - ); - } - - function _checkParseInvalidUintFromHexReverts(string memory s) internal { - vm.expectRevert(JSONParserLib.ParsingFailed.selector); - this.parseUintFromHex(s); - } - - function testParseUint() public { - assertEq(this.parseUint("0"), 0); - assertEq(this.parseUint("1"), 1); - assertEq(this.parseUint("123"), 123); - assertEq(this.parseUint("0123"), 123); - assertEq(this.parseUint("000123"), 123); - assertEq(this.parseUint("12345678901234567890"), 12345678901234567890); - string memory s; - s = "115792089237316195423570985008687907853269984665640564039457584007913129639935"; - assertEq(this.parseUint(s), type(uint256).max); - } - - function testParseInvalidUintReverts() public { - _checkParseInvalidUintReverts(""); - _checkParseInvalidUintReverts("-"); - _checkParseInvalidUintReverts("a"); - _checkParseInvalidUintReverts(" "); - _checkParseInvalidUintReverts(" 123 "); - _checkParseInvalidUintReverts("123:"); - _checkParseInvalidUintReverts(":"); - string memory s; - s = "115792089237316195423570985008687907853269984665640564039457584007913129639936"; - _checkParseInvalidUintReverts(s); - s = "115792089237316195423570985008687907853269984665640564039457584007913129639937"; - _checkParseInvalidUintReverts(s); - s = "115792089237316195423570985008687907853269984665640564039457584007913129639999"; - _checkParseInvalidUintReverts(s); - s = "115792089237316195423570985008687907853269984665640564039457584007913129640001"; - _checkParseInvalidUintReverts(s); - s = "115792089237316195423570985008687907853269984665640564039457584007913129640035"; - _checkParseInvalidUintReverts(s); - s = "215792089237316195423570985008687907853269984665640564039457584007913129639935"; - _checkParseInvalidUintReverts(s); - s = "222222222222222222222222222222222222222222222222222222222222222222222222222222"; - _checkParseInvalidUintReverts(s); - s = "1215792089237316195423570985008687907853269984665640564039457584007913129639935"; - _checkParseInvalidUintReverts(s); - } - - function _checkParseInvalidUintReverts(string memory s) internal { - vm.expectRevert(JSONParserLib.ParsingFailed.selector); - this.parseUint(s); - } - - function parseUint(string memory s) public view miniBrutalizeMemory returns (uint256) { - return s.parseUint(); - } - - function testParseInt() public { - _checkParseInt("0", 0); - _checkParseInt("1", 1); - _checkParseInt("+1", 1); - _checkParseInt("+01", 1); - _checkParseInt("+001", 1); - _checkParseInt("+0", 0); - _checkParseInt("+1", 1); - _checkParseInt("+12", 12); - _checkParseInt("-12", -12); - string memory s; - s = "-57896044618658097711785492504343953926634992332820282019728792003956564819967"; - _checkParseInt(s, -type(int256).max); - s = "+57896044618658097711785492504343953926634992332820282019728792003956564819967"; - _checkParseInt(s, type(int256).max); - s = "57896044618658097711785492504343953926634992332820282019728792003956564819967"; - _checkParseInt(s, type(int256).max); - } - - function testParseInvalidIntReverts() public { - _checkParseInvalidIntReverts(""); - _checkParseInvalidIntReverts("-"); - _checkParseInvalidIntReverts("+"); - _checkParseInvalidIntReverts("--"); - _checkParseInvalidIntReverts("++"); - _checkParseInvalidIntReverts("a"); - _checkParseInvalidIntReverts(" "); - _checkParseInvalidIntReverts(" 123 "); - _checkParseInvalidIntReverts("123:"); - _checkParseInvalidIntReverts(":"); - _checkParseInvalidIntReverts(":123"); - string memory s; - s = "-57896044618658097711785492504343953926634992332820282019728792003956564819968"; - _checkParseInvalidIntReverts(s); - s = "+57896044618658097711785492504343953926634992332820282019728792003956564819968"; - _checkParseInvalidIntReverts(s); - } - - function _checkParseInt(string memory s, int256 x) internal { - bytes32 hashBefore = keccak256(bytes(s)); - assertEq(this.parseInt(s), x); - assertEq(keccak256(bytes(s)), hashBefore); - } - - function _checkParseInvalidIntReverts(string memory s) internal { - vm.expectRevert(JSONParserLib.ParsingFailed.selector); - this.parseInt(s); - } - - function parseInt(string memory s) public view miniBrutalizeMemory returns (int256) { - return s.parseInt(); - } - - function testDecodeString() public { - assertEq(this.decodeString('""'), ""); - assertEq(this.decodeString('"abc"'), "abc"); - assertEq(this.decodeString('" abc "'), " abc "); - assertEq(this.decodeString('"\\""'), '"'); - assertEq(this.decodeString('"\\/"'), "/"); - assertEq(this.decodeString('"\\\\"'), "\\"); - assertEq(this.decodeString('"\\b"'), hex"08"); - assertEq(this.decodeString('"\\f"'), hex"0c"); - assertEq(this.decodeString('"\\n"'), "\n"); - assertEq(this.decodeString('"\\r"'), "\r"); - assertEq(this.decodeString('"\\t"'), "\t"); - assertEq(this.decodeString('"\\u0020"'), " "); - bytes32 expectedHash; - expectedHash = 0x40b2b6558413427ef2da03b1452640d701458e0ce57114db6b7423ae3b5fe857; - assertEq(keccak256(bytes(this.decodeString('"\\u039e"'))), expectedHash); // Greek uppercase Xi. - expectedHash = 0xecab436111d5a82d983bd4630c03c83f424d2a2dd8465c31fd950b9ec8d005fb; - assertEq(keccak256(bytes(this.decodeString('"\\u2661"'))), expectedHash); // Heart. - expectedHash = 0x367c272ea502ac6e9f085c1baddc52d0ac0224f1b7d1e8621202620efa3ba084; - assertEq(keccak256(bytes(this.decodeString('"\\uD83D\\ude00"'))), expectedHash); // Smiley emoji. - } - - function testDecodeEncodedStringDoesNotRevert(string memory s) public { - _limitStringLength(s); - s = string(abi.encodePacked('"', LibString.escapeJSON(s), '"')); - this.decodeString(s); - assertEq(this.parsedValue(s), s); - } - - function _limitStringLength(string memory s) internal { - uint256 r = _random(); - /// @solidity memory-safe-assembly - assembly { - let limit := 16 - if eq(1, and(r, 3)) { limit := 80 } - let n := mload(s) - if gt(n, limit) { mstore(s, limit) } - } - } - - function testDecodeInvalidStringReverts() public { - _checkDecodeInvalidStringReverts(""); - _checkDecodeInvalidStringReverts('"'); - _checkDecodeInvalidStringReverts(' "" '); - _checkDecodeInvalidStringReverts(' ""'); - _checkDecodeInvalidStringReverts('"" '); - _checkDecodeInvalidStringReverts('"\\z"'); - _checkDecodeInvalidStringReverts('"\\u"'); - _checkDecodeInvalidStringReverts('"\\u1"'); - _checkDecodeInvalidStringReverts('"\\u111"'); - _checkDecodeInvalidStringReverts('"\\uxxxx"'); - _checkDecodeInvalidStringReverts('"\\uD83D"'); // Only half of a Smiley emoji. - } - - function _checkDecodeInvalidStringReverts(string memory s) internal { - vm.expectRevert(JSONParserLib.ParsingFailed.selector); - this.decodeString(s); - } - - function decodeString(string memory s) - public - view - miniBrutalizeMemory - returns (string memory) - { - return JSONParserLib.decodeString(s); - } - - function testParseUint(uint256 x) public { - string memory s = LibString.toString(x); - assertEq(this.parsedValue(s), s); - assertEq(this.parseUint(s), x); - } - - function testParseJWTGas() public { - string memory jwt = - "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c"; - string[] memory jwtSplitted = LibString.split(jwt, "."); - JSONParserLib.Item memory header = - JSONParserLib.parse(string(Base64.decode(jwtSplitted[0]))); - JSONParserLib.Item memory payload = - JSONParserLib.parse(string(Base64.decode(jwtSplitted[1]))); - assertEq(jwtSplitted.length, 3); - assertEq(jwtSplitted[2], "SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c"); - assertEq(header.at('"alg"').value(), '"HS256"'); - assertEq(header.at('"typ"').value(), '"JWT"'); - assertEq(payload.at('"sub"').value(), '"1234567890"'); - assertEq(payload.at('"name"').value(), '"John Doe"'); - assertEq(JSONParserLib.parseUint(payload.at('"iat"').value()), 1516239022); - } - - modifier miniBrutalizeMemory() { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, gas()) - mstore(0x00, keccak256(0x00, 0x20)) - mstore(0x20, not(mload(0x00))) - codecopy(mload(0x40), 0, codesize()) - } - _; - } -} diff --git a/lib/solady/test/LibBit.t.sol b/lib/solady/test/LibBit.t.sol deleted file mode 100644 index c235a91..0000000 --- a/lib/solady/test/LibBit.t.sol +++ /dev/null @@ -1,205 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {LibBit} from "../src/utils/LibBit.sol"; - -contract LibBitTest is SoladyTest { - function testFLS() public { - assertEq(LibBit.fls(0xff << 3), 10); - for (uint256 i = 1; i < 255; i++) { - assertEq(LibBit.fls((1 << i) - 1), i - 1); - assertEq(LibBit.fls((1 << i)), i); - assertEq(LibBit.fls((1 << i) + 1), i); - } - assertEq(LibBit.fls(0), 256); - } - - function testCLZ() public { - for (uint256 i = 1; i < 255; i++) { - assertEq(LibBit.clz((1 << i) - 1), 255 - (i - 1)); - assertEq(LibBit.clz((1 << i)), 255 - i); - assertEq(LibBit.clz((1 << i) + 1), 255 - i); - } - assertEq(LibBit.clz(0), 256); - } - - function testFFS() public { - assertEq(LibBit.ffs(0xff << 3), 3); - uint256 brutalizer = uint256(keccak256(abi.encode(address(this), block.timestamp))); - for (uint256 i = 0; i < 256; i++) { - assertEq(LibBit.ffs(1 << i), i); - assertEq(LibBit.ffs(type(uint256).max << i), i); - assertEq(LibBit.ffs((brutalizer | 1) << i), i); - } - assertEq(LibBit.ffs(0), 256); - } - - function testPopCount(uint256 x) public { - uint256 c; - unchecked { - for (uint256 t = x; t != 0; c++) { - t &= t - 1; - } - } - assertEq(LibBit.popCount(x), c); - } - - function testPopCount() public { - unchecked { - for (uint256 i = 1; i < 256; ++i) { - assertEq(LibBit.popCount((1 << i) | 1), 2); - } - } - } - - function testIsPo2(uint8 a, uint8 b) public { - unchecked { - uint256 x = (1 << uint256(a)) | (1 << uint256(b)); - if (a == b) { - assertTrue(LibBit.isPo2(x)); - } else { - assertFalse(LibBit.isPo2(x)); - } - } - } - - function testIsPo2(uint256 x) public { - uint256 c; - unchecked { - for (uint256 t = x; t != 0; c++) { - t &= t - 1; - } - } - assertEq(LibBit.isPo2(x), c == 1); - } - - function testIsPo2() public { - assertFalse(LibBit.isPo2(0)); - assertFalse(LibBit.isPo2(type(uint256).max)); - unchecked { - for (uint256 i; i < 256; ++i) { - uint256 x = 1 << i; - assertTrue(LibBit.isPo2(x)); - assertFalse(LibBit.isPo2(~x)); - } - } - } - - function testAnd(bool x, bool y) public { - assertEq(LibBit.and(x, y), x && y); - assertEq(LibBit.rawAnd(x, y), LibBit.and(x, y)); - } - - function testAnd() public { - unchecked { - for (uint256 t; t != 100; ++t) { - uint256 i = _random(); - uint256 j = _random(); - uint256 k = _random(); - bool a = i < j; - bool b = j < k; - assertEq(LibBit.and(a, b), i < j && j < k); - } - } - } - - function testOr(bool x, bool y) public { - assertEq(LibBit.or(x, y), x || y); - assertEq(LibBit.rawOr(x, y), LibBit.or(x, y)); - } - - function testOr() public { - unchecked { - for (uint256 t; t != 100; ++t) { - uint256 i = _random(); - uint256 j = _random(); - uint256 k = _random(); - bool a = i < j; - bool b = j < k; - assertEq(LibBit.or(a, b), i < j || j < k); - } - } - } - - function testAutoClean(uint256 x, uint256 y) public { - bool xCasted; - bool yCasted; - /// @solidity memory-safe-assembly - assembly { - xCasted := x - yCasted := y - } - bool result = LibBit.and(true, LibBit.or(xCasted, yCasted)); - assertEq(result, xCasted || yCasted); - } - - function testReturnsBool() public { - bool result; - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, 0x40b98a2f) - mstore(0x20, 123) - pop(staticcall(gas(), address(), 0x1c, 0x24, 0x00, 0x20)) - result := eq(mload(0x00), 1) - } - assertTrue(result); - } - - function returnsBool(uint256 i) public pure returns (bool b) { - /// @solidity memory-safe-assembly - assembly { - b := i - } - } - - function testPassInBool() public { - bool result; - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, 0x59a3028a) - mstore(0x20, 1) - pop(staticcall(gas(), address(), 0x1c, 0x24, 0x00, 0x20)) - result := eq(mload(0x00), 1) - } - assertTrue(result); - } - - function acceptsBool(bool) public pure returns (bool) { - return true; - } - - function testBoolToUint(bool b) public { - uint256 z; - /// @solidity memory-safe-assembly - assembly { - z := b - } - assertEq(LibBit.toUint(b), z); - assertEq(LibBit.rawToUint(b), z); - } - - function testReverseBitsDifferential(uint256 x) public { - assertEq(LibBit.reverseBits(x), _reverseBitsOriginal(x)); - } - - function _reverseBitsOriginal(uint256 x) internal pure returns (uint256 r) { - unchecked { - for (uint256 i; i != 256; ++i) { - r = (r << 1) | ((x >> i) & 1); - } - } - } - - function testReverseBytesDifferential(uint256 x) public { - assertEq(LibBit.reverseBytes(x), _reverseBytesOriginal(x)); - } - - function _reverseBytesOriginal(uint256 x) internal pure returns (uint256 r) { - /// @solidity memory-safe-assembly - assembly { - for { let i := 0 } lt(i, 32) { i := add(i, 1) } { mstore8(i, byte(sub(31, i), x)) } - r := mload(0x00) - } - } -} diff --git a/lib/solady/test/LibBitmap.t.sol b/lib/solady/test/LibBitmap.t.sol deleted file mode 100644 index 1e97cf8..0000000 --- a/lib/solady/test/LibBitmap.t.sol +++ /dev/null @@ -1,350 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {LibBitmap} from "../src/utils/LibBitmap.sol"; -import {LibBit} from "../src/utils/LibBit.sol"; - -contract LibBitmapTest is SoladyTest { - using LibBitmap for LibBitmap.Bitmap; - - error AlreadyClaimed(); - - LibBitmap.Bitmap bitmap; - - function get(uint256 index) public view returns (bool result) { - result = bitmap.get(index); - } - - function set(uint256 index) public { - bitmap.set(index); - } - - function unset(uint256 index) public { - bitmap.unset(index); - } - - function toggle(uint256 index) public { - bitmap.toggle(index); - } - - function setTo(uint256 index, bool shouldSet) public { - bitmap.setTo(index, shouldSet); - } - - function claimWithGetSet(uint256 index) public { - if (bitmap.get(index)) { - revert AlreadyClaimed(); - } - bitmap.set(index); - } - - function claimWithToggle(uint256 index) public { - if (bitmap.toggle(index) == false) { - revert AlreadyClaimed(); - } - } - - function testBitmapGet() public { - testBitmapGet(111111); - } - - function testBitmapGet(uint256 index) public { - assertFalse(get(index)); - } - - function testBitmapSetAndGet(uint256 index) public { - set(index); - bool result = get(index); - bool resultIsOne; - /// @solidity memory-safe-assembly - assembly { - resultIsOne := eq(result, 1) - } - assertTrue(result); - assertTrue(resultIsOne); - } - - function testBitmapSet() public { - testBitmapSet(222222); - } - - function testBitmapSet(uint256 index) public { - set(index); - assertTrue(get(index)); - } - - function testBitmapUnset() public { - testBitmapSet(333333); - } - - function testBitmapUnset(uint256 index) public { - set(index); - assertTrue(get(index)); - unset(index); - assertFalse(get(index)); - } - - function testBitmapSetTo() public { - testBitmapSetTo(555555, true, 0); - testBitmapSetTo(555555, false, 0); - } - - function testBitmapSetTo(uint256 index, bool shouldSet, uint256 randomness) public { - bool shouldSetBrutalized; - /// @solidity memory-safe-assembly - assembly { - if shouldSet { shouldSetBrutalized := or(iszero(randomness), randomness) } - } - setTo(index, shouldSetBrutalized); - assertEq(get(index), shouldSet); - } - - function testBitmapSetTo(uint256 index, uint256 randomness) public { - randomness = _random(); - unchecked { - for (uint256 i; i < 5; ++i) { - bool shouldSet; - /// @solidity memory-safe-assembly - assembly { - shouldSet := and(shr(i, randomness), 1) - } - testBitmapSetTo(index, shouldSet, _random()); - } - } - } - - function testBitmapToggle() public { - testBitmapToggle(777777, true); - testBitmapToggle(777777, false); - } - - function testBitmapToggle(uint256 index, bool initialValue) public { - setTo(index, initialValue); - assertEq(get(index), initialValue); - toggle(index); - assertEq(get(index), !initialValue); - } - - function testBitmapClaimWithGetSet() public { - uint256 index = 888888; - this.claimWithGetSet(index); - vm.expectRevert(AlreadyClaimed.selector); - this.claimWithGetSet(index); - } - - function testBitmapClaimWithToggle() public { - uint256 index = 999999; - this.claimWithToggle(index); - vm.expectRevert(AlreadyClaimed.selector); - this.claimWithToggle(index); - } - - function testBitmapSetBatchWithinSingleBucket() public { - _testBitmapSetBatch(257, 30); - } - - function testBitmapSetBatchAcrossMultipleBuckets() public { - _testBitmapSetBatch(10, 512); - } - - function testBitmapSetBatch() public { - unchecked { - for (uint256 i; i < 8; ++i) { - uint256 start = _random(); - uint256 amount = _random(); - _testBitmapSetBatch(start, amount); - } - } - } - - function testBitmapUnsetBatchWithinSingleBucket() public { - _testBitmapUnsetBatch(257, 30); - } - - function testBitmapUnsetBatchAcrossMultipleBuckets() public { - _testBitmapUnsetBatch(10, 512); - } - - function testBitmapUnsetBatch() public { - unchecked { - for (uint256 i; i < 8; ++i) { - uint256 start = _random(); - uint256 amount = _random(); - _testBitmapUnsetBatch(start, amount); - } - } - } - - function testBitmapPopCountWithinSingleBucket() public { - _testBitmapPopCount(1, 150); - } - - function testBitmapPopCountAcrossMultipleBuckets() public { - _testBitmapPopCount(10, 512); - } - - function testBitmapPopCount(uint256, uint256 start, uint256 amount) public { - unchecked { - uint256 n = 1000; - uint256 expectedCount; - _resetBitmap(0, n / 256 + 1); - - (start, amount) = _boundStartAndAmount(start, amount, n); - - uint256 jPrev = 0xff + 1; - uint256 j = _random() & 0xff; - while (true) { - bitmap.set(j); - if (j != jPrev && start <= j && j < start + amount) { - expectedCount += 1; - } - if (start + amount <= j && _random() & 7 == 0) break; - jPrev = j; - j += _random() & 0xff; - } - assertEq(bitmap.popCount(start, amount), expectedCount); - } - } - - function testBitmapPopCount() public { - unchecked { - for (uint256 i; i < 8; ++i) { - uint256 start = _random(); - uint256 amount = _random(); - testBitmapPopCount(start, amount, _random()); - } - } - } - - function testBitmapFindLastSet() public { - unchecked { - bitmap.unsetBatch(0, 2000); - bitmap.set(1000); - for (uint256 i = 0; i < 1000; ++i) { - assertEq(bitmap.findLastSet(i), LibBitmap.NOT_FOUND); - } - bitmap.set(100); - bitmap.set(10); - for (uint256 i = 0; i < 10; ++i) { - assertEq(bitmap.findLastSet(i), LibBitmap.NOT_FOUND); - } - for (uint256 i = 10; i < 100; ++i) { - assertEq(bitmap.findLastSet(i), 10); - } - for (uint256 i = 100; i < 600; ++i) { - assertEq(bitmap.findLastSet(i), 100); - } - for (uint256 i = 1000; i < 1100; ++i) { - assertEq(bitmap.findLastSet(i), 1000); - } - bitmap.set(0); - for (uint256 i = 0; i < 10; ++i) { - assertEq(bitmap.findLastSet(i), 0); - } - } - } - - function testBitmapFindLastSet(uint256 before, uint256 randomness) public { - uint256 n = 1000; - unchecked { - _resetBitmap(0, n / 256 + 1); - before = before % n; - randomness = _random() % n; - } - bitmap.set(randomness); - if (randomness <= before) { - assertEq(bitmap.findLastSet(before), randomness); - uint256 nextLcg = _random(); - bitmap.set(nextLcg); - if (nextLcg <= before) { - assertEq(bitmap.findLastSet(before), (randomness < nextLcg ? nextLcg : randomness)); - } - } else { - assertEq(bitmap.findLastSet(before), LibBitmap.NOT_FOUND); - uint256 nextLcg = _random(); - bitmap.set(nextLcg); - if (nextLcg <= before) { - assertEq(bitmap.findLastSet(before), nextLcg); - } else { - assertEq(bitmap.findLastSet(before), LibBitmap.NOT_FOUND); - } - } - } - - function _testBitmapSetBatch(uint256 start, uint256 amount) internal { - uint256 n = 1000; - (start, amount) = _boundStartAndAmount(start, amount, n); - - unchecked { - _resetBitmap(0, n / 256 + 1); - bitmap.setBatch(start, amount); - for (uint256 i; i < n; ++i) { - if (i < start) { - assertFalse(bitmap.get(i)); - } else if (i < start + amount) { - assertTrue(bitmap.get(i)); - } else { - assertFalse(bitmap.get(i)); - } - } - } - } - - function _testBitmapUnsetBatch(uint256 start, uint256 amount) internal { - uint256 n = 1000; - (start, amount) = _boundStartAndAmount(start, amount, n); - - unchecked { - _resetBitmap(type(uint256).max, n / 256 + 1); - bitmap.unsetBatch(start, amount); - for (uint256 i; i < n; ++i) { - if (i < start) { - assertTrue(bitmap.get(i)); - } else if (i < start + amount) { - assertFalse(bitmap.get(i)); - } else { - assertTrue(bitmap.get(i)); - } - } - } - } - - function _testBitmapPopCount(uint256 start, uint256 amount) internal { - uint256 n = 1000; - (start, amount) = _boundStartAndAmount(start, amount, n); - - unchecked { - _resetBitmap(0, n / 256 + 1); - bitmap.setBatch(start, amount); - assertEq(bitmap.popCount(0, n), amount); - if (start > 0) { - assertEq(bitmap.popCount(0, start - 1), 0); - } - if (start + amount < n) { - assertEq(bitmap.popCount(start + amount, n - (start + amount)), 0); - } - } - } - - function _boundStartAndAmount(uint256 start, uint256 amount, uint256 n) - private - pure - returns (uint256 boundedStart, uint256 boundedAmount) - { - unchecked { - boundedStart = start % n; - uint256 end = boundedStart + (amount % n); - if (end > n) end = n; - boundedAmount = end - boundedStart; - } - } - - function _resetBitmap(uint256 bucketValue, uint256 bucketEnd) private { - unchecked { - for (uint256 i; i < bucketEnd; ++i) { - bitmap.map[i] = bucketValue; - } - } - } -} diff --git a/lib/solady/test/LibClone.t.sol b/lib/solady/test/LibClone.t.sol deleted file mode 100644 index 4af3450..0000000 --- a/lib/solady/test/LibClone.t.sol +++ /dev/null @@ -1,422 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {LibClone} from "../src/utils/LibClone.sol"; -import {Clone} from "../src/utils/Clone.sol"; -import {SafeTransferLib} from "../src/utils/SafeTransferLib.sol"; - -contract LibCloneTest is SoladyTest, Clone { - error CustomError(uint256 currentValue); - - event ReceiveETH(uint256 amount); - - uint256 public value; - - mapping(bytes32 => bool) saltIsUsed; - - bytes32 internal constant _ERC1967_IMPLEMENTATION_SLOT = - 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc; - - function setValue(uint256 value_) public { - value = value_; - } - - function revertWithError() public view { - revert CustomError(value); - } - - function getCalldataHash() public pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - let extraLength := shr(0xf0, calldataload(sub(calldatasize(), 2))) - if iszero(lt(extraLength, 2)) { - let offset := sub(calldatasize(), extraLength) - let m := mload(0x40) - calldatacopy(m, offset, sub(extraLength, 2)) - result := keccak256(m, sub(extraLength, 2)) - } - } - } - - function _canReceiveETHCorrectly(address clone, uint256 deposit) internal { - deposit = deposit % 1 ether; - - vm.deal(address(this), deposit * 2); - - vm.expectEmit(true, true, true, true); - emit ReceiveETH(deposit); - SafeTransferLib.safeTransferETH(clone, deposit); - assertEq(clone.balance, deposit); - - vm.expectEmit(true, true, true, true); - emit ReceiveETH(deposit); - payable(clone).transfer(deposit); - assertEq(clone.balance, deposit * 2); - } - - function _shouldBehaveLikeClone(address clone, uint256 value_) internal { - assertTrue(clone != address(0)); - - uint256 thisValue = this.value(); - if (thisValue == value_) { - value_ ^= 1; - } - LibCloneTest(clone).setValue(value_); - assertEq(value_, LibCloneTest(clone).value()); - assertEq(thisValue, this.value()); - vm.expectRevert(abi.encodeWithSelector(CustomError.selector, value_)); - LibCloneTest(clone).revertWithError(); - } - - function testDeployERC1967(uint256 value_) public { - address clone = LibClone.deployERC1967(address(this)); - _shouldBehaveLikeClone(clone, value_); - assertEq( - vm.load(clone, _ERC1967_IMPLEMENTATION_SLOT), bytes32(uint256(uint160(address(this)))) - ); - } - - function testDeployERC1967() public { - testDeployERC1967(1); - } - - function testClone(uint256 value_) public { - address clone = LibClone.clone(address(this)); - _shouldBehaveLikeClone(clone, value_); - } - - function testClone() public { - testClone(1); - } - - function testCloneDeterministic(uint256 value_, bytes32 salt) public { - if (saltIsUsed[salt]) { - vm.expectRevert(LibClone.DeploymentFailed.selector); - this.cloneDeterministic(address(this), salt); - return; - } - - address clone = this.cloneDeterministic(address(this), salt); - saltIsUsed[salt] = true; - - _shouldBehaveLikeClone(clone, value_); - - address predicted = LibClone.predictDeterministicAddress(address(this), salt, address(this)); - assertEq(clone, predicted); - } - - function cloneDeterministic(address implementation, bytes32 salt) external returns (address) { - return LibClone.cloneDeterministic(_brutalized(implementation), salt); - } - - function cloneDeterministic(address implementation, bytes calldata data, bytes32 salt) - external - returns (address) - { - return LibClone.cloneDeterministic(_brutalized(implementation), data, salt); - } - - function testCloneDeterministicRevertsIfAddressAlreadyUsed() public { - testCloneDeterministic(1, keccak256("a")); - testCloneDeterministic(1, keccak256("a")); - } - - function testCloneDeterministic() public { - testCloneDeterministic(1, keccak256("b")); - } - - function testDeployDeterministicERC1967(uint256 value_, bytes32 salt) public { - if (saltIsUsed[salt]) { - vm.expectRevert(LibClone.DeploymentFailed.selector); - this.deployDeterministicERC1967(address(this), salt); - return; - } - - address clone = this.deployDeterministicERC1967(address(this), salt); - saltIsUsed[salt] = true; - - _shouldBehaveLikeClone(clone, value_); - - address predicted = - LibClone.predictDeterministicAddressERC1967(address(this), salt, address(this)); - assertEq(clone, predicted); - - assertEq( - vm.load(clone, _ERC1967_IMPLEMENTATION_SLOT), bytes32(uint256(uint160(address(this)))) - ); - } - - function deployDeterministicERC1967(address implementation, bytes32 salt) - external - returns (address) - { - return LibClone.deployDeterministicERC1967(_brutalized(implementation), salt); - } - - function testDeployDeterministicERC1967() public { - testDeployDeterministicERC1967(1, keccak256("b")); - } - - function getArgBytes(uint256 argOffset, uint256 length) public pure returns (bytes memory) { - return _getArgBytes(argOffset, length); - } - - function getArgAddress(uint256 argOffset) public pure returns (address) { - return _getArgAddress(argOffset); - } - - function getArgUint256(uint256 argOffset) public pure returns (uint256) { - uint256 result = _getArgUint256(argOffset); - unchecked { - require(bytes32(result) == _getArgBytes32(argOffset)); - require(uint248(result) == _getArgUint248(argOffset + 1)); - require(uint240(result) == _getArgUint240(argOffset + 2)); - require(uint232(result) == _getArgUint232(argOffset + 3)); - require(uint224(result) == _getArgUint224(argOffset + 4)); - require(uint216(result) == _getArgUint216(argOffset + 5)); - require(uint208(result) == _getArgUint208(argOffset + 6)); - require(uint200(result) == _getArgUint200(argOffset + 7)); - require(uint192(result) == _getArgUint192(argOffset + 8)); - require(uint184(result) == _getArgUint184(argOffset + 9)); - require(uint176(result) == _getArgUint176(argOffset + 10)); - require(uint168(result) == _getArgUint168(argOffset + 11)); - require(uint160(result) == _getArgUint160(argOffset + 12)); - require(uint152(result) == _getArgUint152(argOffset + 13)); - require(uint144(result) == _getArgUint144(argOffset + 14)); - require(uint136(result) == _getArgUint136(argOffset + 15)); - require(uint128(result) == _getArgUint128(argOffset + 16)); - require(uint120(result) == _getArgUint120(argOffset + 17)); - require(uint112(result) == _getArgUint112(argOffset + 18)); - require(uint104(result) == _getArgUint104(argOffset + 19)); - require(uint96(result) == _getArgUint96(argOffset + 20)); - require(uint88(result) == _getArgUint88(argOffset + 21)); - require(uint80(result) == _getArgUint80(argOffset + 22)); - require(uint72(result) == _getArgUint72(argOffset + 23)); - require(uint64(result) == _getArgUint64(argOffset + 24)); - require(uint56(result) == _getArgUint56(argOffset + 25)); - require(uint48(result) == _getArgUint48(argOffset + 26)); - require(uint40(result) == _getArgUint40(argOffset + 27)); - require(uint32(result) == _getArgUint32(argOffset + 28)); - require(uint24(result) == _getArgUint24(argOffset + 29)); - require(uint16(result) == _getArgUint16(argOffset + 30)); - require(uint8(result) == _getArgUint8(argOffset + 31)); - } - return result; - } - - function getArgUint256Array(uint256 argOffset, uint256 length) - public - pure - returns (uint256[] memory) - { - uint256[] memory result = _getArgUint256Array(argOffset, length); - bytes32 hash = keccak256(abi.encode(_getArgBytes32Array(argOffset, length))); - require(keccak256(abi.encode(result)) == hash); - return result; - } - - function getArgUint64(uint256 argOffset) public pure returns (uint64) { - return _getArgUint64(argOffset); - } - - function getArgUint8(uint256 argOffset) public pure returns (uint8) { - return _getArgUint8(argOffset); - } - - function testCloneWithImmutableArgs( - uint256 value_, - address argAddress, - uint256 argUint256, - uint256[] memory argUint256Array, - uint64 argUint64, - uint8 argUint8 - ) public { - bytes memory data = - abi.encodePacked(argAddress, argUint256, argUint256Array, argUint64, argUint8); - LibCloneTest clone = LibCloneTest(LibClone.clone(address(this), data)); - _shouldBehaveLikeClone(address(clone), value_); - - // For avoiding stack too deep. Also, no risk of overflow. - unchecked { - uint256 argOffset; - assertEq(clone.getArgAddress(argOffset), argAddress); - argOffset += 20; - assertEq(clone.getArgUint256(argOffset), argUint256); - argOffset += 32; - assertEq(clone.getArgUint256Array(argOffset, argUint256Array.length), argUint256Array); - argOffset += 32 * argUint256Array.length; - assertEq(clone.getArgUint64(argOffset), argUint64); - argOffset += 8; - assertEq(clone.getArgUint8(argOffset), argUint8); - } - } - - function testCloneWithImmutableArgs() public { - uint256[] memory argUint256Array = new uint256[](2); - argUint256Array[0] = 111; - argUint256Array[1] = 222; - testCloneWithImmutableArgs(1, address(uint160(0xB00Ba5)), 8, argUint256Array, 7, 6); - } - - function testCloneDeteministicWithImmutableArgs( - address argAddress, - uint256 argUint256, - uint256[] memory argUint256Array, - bytes memory argBytes, - uint64 argUint64, - uint8 argUint8, - uint256 deposit - ) public { - bytes memory data; - bytes32 salt; - - // For avoiding stack too deep. - unchecked { - // Recycle for the salt. - salt = bytes32(argUint256 + 123); - - data = abi.encodePacked( - argUint256, - argAddress, - argUint256, - argUint256Array, - argBytes, - argUint64, - argUint8, - argUint256 - ); - - bytes32 saltKey = keccak256(abi.encode(data, salt)); - if (saltIsUsed[saltKey]) { - vm.expectRevert(LibClone.DeploymentFailed.selector); - LibCloneTest(this.cloneDeterministic(address(this), data, salt)); - return; - } - saltIsUsed[saltKey] = true; - } - - bytes32 dataHashBefore = keccak256(data); - - LibCloneTest clone = LibCloneTest(this.cloneDeterministic(address(this), data, salt)); - // Check that memory management is done properly. - assertEq(keccak256(data), dataHashBefore); - - _shouldBehaveLikeClone(address(clone), argUint256); - _canReceiveETHCorrectly(address(clone), deposit); - - // For avoiding stack too deep. Also, no risk of overflow. - unchecked { - uint256 argOffset; - assertEq(clone.getArgUint256(argOffset), argUint256); - argOffset += (256 / 8); - assertEq(clone.getArgAddress(argOffset), argAddress); - argOffset += (160 / 8); - assertEq(clone.getArgUint256(argOffset), argUint256); - argOffset += (256 / 8); - assertEq(clone.getArgUint256Array(argOffset, argUint256Array.length), argUint256Array); - argOffset += (256 / 8) * argUint256Array.length; - assertEq(clone.getArgBytes(argOffset, argBytes.length), argBytes); - argOffset += (8 / 8) * argBytes.length; - assertEq(clone.getArgUint64(argOffset), argUint64); - argOffset += (64 / 8); - assertEq(clone.getArgUint8(argOffset), argUint8); - argOffset += (8 / 8); - assertEq(clone.getArgUint256(argOffset), argUint256); - } - - { - address predicted = - LibClone.predictDeterministicAddress(address(this), data, salt, address(this)); - assertEq(address(clone), predicted); - } - - // Check that memory management is done properly. - assertEq(keccak256(data), dataHashBefore); - - assertEq(clone.getCalldataHash(), dataHashBefore); - } - - function testCloneDeteministicWithImmutableArgs() public { - uint256[] memory argUint256Array = new uint256[](2); - argUint256Array[0] = uint256(keccak256("zero")); - argUint256Array[1] = uint256(keccak256("one")); - bytes memory argBytes = bytes("Teehee"); - testCloneDeteministicWithImmutableArgs( - address(uint160(uint256(keccak256("argAddress")))), - uint256(keccak256("argUint256")), - argUint256Array, - argBytes, - uint64(uint256(keccak256("argUint64"))), - uint8(uint256(keccak256("argUint8"))), - uint256(keccak256("deposit")) - ); - } - - function testStartsWith(uint256) public { - uint256 noise = _random() >> 160; - this.checkStartsWith(bytes32(noise), address(0)); - - address by = _randomNonZeroAddress(); - this.checkStartsWith(bytes32((uint256(uint160(by)) << 96) | noise), by); - - address notBy; - while (by == notBy) notBy = _randomNonZeroAddress(); - vm.expectRevert(LibClone.SaltDoesNotStartWith.selector); - this.checkStartsWith(bytes32((uint256(uint160(by)) << 96) | noise), notBy); - } - - function checkStartsWith(bytes32 salt, address by) public view { - LibClone.checkStartsWith(salt, _brutalized(by)); - } - - function _brutalized(address a) internal view returns (address result) { - /// @solidity memory-safe-assembly - assembly { - result := or(a, shl(160, gas())) - } - } - - function testCloneWithImmutableArgsRevertsIfDataTooBig() public { - uint256 n = 0xff9b; - bytes memory data = _dummyData(n); - - address clone = this.cloneDeterministic(address(this), data, bytes32(gasleft())); - _shouldBehaveLikeClone(clone, 1); - assertEq(LibCloneTest(clone).argBytesHash(), keccak256(data)); - - vm.expectRevert(); - this.cloneDeterministic(address(this), _dummyData(n + 1), bytes32(gasleft())); - } - - function testInitialDeposit() public { - vm.deal(address(this), 1 ether); - address t = address(this); - assertEq(LibClone.clone(123, t).balance, 123); - assertEq(LibClone.cloneDeterministic(123, t, bytes32(gasleft())).balance, 123); - assertEq(LibClone.clone(123, t, "").balance, 123); - assertEq(LibClone.cloneDeterministic(123, t, "", bytes32(gasleft())).balance, 123); - assertEq(LibClone.deployERC1967(123, t).balance, 123); - assertEq(LibClone.deployDeterministicERC1967(123, t, bytes32(gasleft())).balance, 123); - } - - function argBytesHash() public pure returns (bytes32) { - return keccak256(_getArgBytes()); - } - - function _dummyData(uint256 n) internal pure returns (bytes memory result) { - /// @solidity memory-safe-assembly - assembly { - result := mload(0x40) - mstore(result, n) - mstore(0x00, n) - mstore(0x20, 1) - mstore(add(0x20, result), keccak256(0x00, 0x40)) - mstore(0x20, 2) - mstore(add(add(0x20, result), n), keccak256(0x00, 0x40)) - mstore(0x20, 3) - mstore(add(result, n), keccak256(0x00, 0x40)) - mstore(0x40, add(add(0x20, result), n)) - } - } -} diff --git a/lib/solady/test/LibMap.t.sol b/lib/solady/test/LibMap.t.sol deleted file mode 100644 index 91800ed..0000000 --- a/lib/solady/test/LibMap.t.sol +++ /dev/null @@ -1,649 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {LibMap} from "../src/utils/LibMap.sol"; - -contract LibMapTest is SoladyTest { - using LibMap for *; - - uint8[0xffffffffffffffff] bigUint8ArrayMap; - - LibMap.Uint8Map[2] uint8s; - - LibMap.Uint16Map[2] uint16s; - - LibMap.Uint32Map[2] uint32s; - - LibMap.Uint40Map[2] uint40s; - - LibMap.Uint64Map[2] uint64s; - - LibMap.Uint128Map[2] uint128s; - - mapping(uint256 => LibMap.Uint32Map) uint32Maps; - - mapping(uint256 => mapping(uint256 => uint256)) generalMaps; - - mapping(uint256 => uint256) filled; - - struct _TestTemps { - uint256 i0; - uint256 i1; - uint256 v0; - uint256 v1; - } - - function _testTemps() internal returns (_TestTemps memory t) { - uint256 r = _random(); - t.i0 = (r >> 8) & 31; - t.i1 = (r >> 16) & 31; - t.v0 = _random(); - t.v1 = _random(); - } - - function getUint8(uint256 index) public view returns (uint8 result) { - result = uint8s[0].get(index); - } - - function setUint8(uint256 index, uint8 value) public { - uint8s[0].set(index, value); - } - - function getUint8FromBigArray(uint256 index) public view returns (uint8 result) { - result = bigUint8ArrayMap[index]; - } - - function setUint8FromBigArray(uint256 index, uint8 value) public { - bigUint8ArrayMap[index] = value; - } - - function testMapSetUint8() public { - this.setUint8(111111, 123); - } - - function testMapGetUint8() public { - assertEq(this.getUint8(222222), uint8(0)); - } - - function testMapSetUint8FromBigArray() public { - this.setUint8FromBigArray(111111, 123); - } - - function testMapGetFromBigArray() public { - assertEq(this.getUint8FromBigArray(222222), uint8(0)); - } - - function testUint8MapSetAndGet(uint256) public { - uint8 u = uint8(_random()); - uint8s[0].set(0, u); - assertEq(uint8s[0].map[0], u); - unchecked { - for (uint256 t; t < 8; ++t) { - uint256 r = _random(); - uint8 casted; - /// @solidity memory-safe-assembly - assembly { - casted := r - } - uint256 index = _random() % 32; - uint8s[0].set(index, casted); - assertEq(uint8s[0].get(index), casted); - } - } - } - - function testUint8MapSetAndGet() public { - unchecked { - for (uint256 t; t < 16; ++t) { - uint256 n = 64; - uint8 casted; - uint256 r = _random(); - for (uint256 i; i < n; ++i) { - /// @solidity memory-safe-assembly - assembly { - casted := or(add(mul(n, t), i), r) - } - uint8s[0].set(i, casted); - } - for (uint256 i; i < n; ++i) { - /// @solidity memory-safe-assembly - assembly { - casted := or(add(mul(n, t), i), r) - } - assertEq(uint8s[0].get(i), casted); - } - } - } - } - - function testUint8MapSetAndGet2(uint256) public { - _TestTemps memory t = _testTemps(); - uint8s[0].set(t.i0, uint8(t.v0)); - uint8s[1].set(t.i1, uint8(t.v1)); - assertEq(uint8s[0].get(t.i0), uint8(t.v0)); - assertEq(uint8s[1].get(t.i1), uint8(t.v1)); - } - - function testUint16MapSetAndGet(uint256) public { - uint16 u = uint16(_random()); - uint16s[0].set(0, u); - assertEq(uint16s[0].map[0], u); - unchecked { - for (uint256 t; t < 8; ++t) { - uint256 r = _random(); - uint16 casted; - /// @solidity memory-safe-assembly - assembly { - casted := r - } - uint256 index = _random() % 32; - uint16s[0].set(index, casted); - assertEq(uint16s[0].get(index), casted); - } - } - } - - function testUint16MapSetAndGet() public { - unchecked { - for (uint256 t; t < 16; ++t) { - uint256 n = 64; - uint16 casted; - uint256 r = _random(); - for (uint256 i; i < n; ++i) { - /// @solidity memory-safe-assembly - assembly { - casted := or(add(mul(n, t), i), r) - } - uint16s[0].set(i, casted); - } - for (uint256 i; i < n; ++i) { - /// @solidity memory-safe-assembly - assembly { - casted := or(add(mul(n, t), i), r) - } - assertEq(uint16s[0].get(i), casted); - } - } - } - } - - function testUint16MapSetAndGet2(uint256) public { - _TestTemps memory t = _testTemps(); - uint16s[0].set(t.i0, uint16(t.v0)); - uint16s[1].set(t.i1, uint16(t.v1)); - assertEq(uint16s[0].get(t.i0), uint16(t.v0)); - assertEq(uint16s[1].get(t.i1), uint16(t.v1)); - } - - function testUint32MapSetAndGet(uint256) public { - uint32 u = uint32(_random()); - uint32s[0].set(0, u); - assertEq(uint32s[0].map[0], u); - unchecked { - for (uint256 t; t < 8; ++t) { - uint256 r = _random(); - uint32 casted; - /// @solidity memory-safe-assembly - assembly { - casted := r - } - uint256 index = _random() % 32; - uint32s[0].set(index, casted); - assertEq(uint32s[0].get(index), casted); - } - } - } - - function testUint32MapSetAndGet() public { - unchecked { - for (uint256 t; t < 16; ++t) { - uint256 n = 64; - uint32 casted; - uint256 r = _random(); - for (uint256 i; i < n; ++i) { - /// @solidity memory-safe-assembly - assembly { - casted := or(add(mul(n, t), i), r) - } - uint32s[0].set(i, casted); - } - for (uint256 i; i < n; ++i) { - /// @solidity memory-safe-assembly - assembly { - casted := or(add(mul(n, t), i), r) - } - assertEq(uint32s[0].get(i), casted); - } - } - } - } - - function testUint32MapSetAndGet2(uint256) public { - _TestTemps memory t = _testTemps(); - uint32s[0].set(t.i0, uint32(t.v0)); - uint32s[1].set(t.i1, uint32(t.v1)); - assertEq(uint32s[0].get(t.i0), uint32(t.v0)); - assertEq(uint32s[1].get(t.i1), uint32(t.v1)); - } - - function testUint40MapSetAndGet(uint256) public { - uint40 u = uint40(_random()); - uint40s[0].set(0, u); - assertEq(uint40s[0].map[0], u); - unchecked { - for (uint256 t; t < 8; ++t) { - uint256 r = _random(); - uint40 casted; - /// @solidity memory-safe-assembly - assembly { - casted := r - } - uint256 index = _random() % 32; - uint40s[0].set(index, casted); - assertEq(uint40s[0].get(index), casted); - } - } - } - - function testUint40MapSetAndGet() public { - unchecked { - for (uint256 t; t < 16; ++t) { - uint256 n = 64; - uint40 casted; - uint256 r = _random(); - for (uint256 i; i < n; ++i) { - /// @solidity memory-safe-assembly - assembly { - casted := or(add(mul(n, t), i), r) - } - uint40s[0].set(i, casted); - } - for (uint256 i; i < n; ++i) { - /// @solidity memory-safe-assembly - assembly { - casted := or(add(mul(n, t), i), r) - } - assertEq(uint40s[0].get(i), casted); - } - } - } - } - - function testUint40MapSetAndGet2(uint256) public { - _TestTemps memory t = _testTemps(); - uint40s[0].set(t.i0, uint40(t.v0)); - uint40s[1].set(t.i1, uint40(t.v1)); - assertEq(uint40s[0].get(t.i0), uint40(t.v0)); - assertEq(uint40s[1].get(t.i1), uint40(t.v1)); - } - - function testUint64MapSetAndGet(uint256) public { - uint64 u = uint64(_random()); - uint64s[0].set(0, u); - assertEq(uint64s[0].map[0], u); - unchecked { - for (uint256 t; t < 8; ++t) { - uint256 r = _random(); - uint64 casted; - /// @solidity memory-safe-assembly - assembly { - casted := r - } - uint256 index = _random() % 32; - uint64s[0].set(index, casted); - assertEq(uint64s[0].get(index), casted); - } - } - } - - function testUint64MapSetAndGet() public { - unchecked { - for (uint256 t; t < 16; ++t) { - uint256 n = 64; - uint64 casted; - uint256 r = _random(); - for (uint256 i; i < n; ++i) { - /// @solidity memory-safe-assembly - assembly { - casted := or(add(mul(n, t), i), r) - } - uint64s[0].set(i, casted); - } - for (uint256 i; i < n; ++i) { - /// @solidity memory-safe-assembly - assembly { - casted := or(add(mul(n, t), i), r) - } - assertEq(uint64s[0].get(i), casted); - } - } - } - } - - function testUint64MapSetAndGet2(uint256) public { - _TestTemps memory t = _testTemps(); - uint64s[0].set(t.i0, uint64(t.v0)); - uint64s[1].set(t.i1, uint64(t.v1)); - assertEq(uint64s[0].get(t.i0), uint64(t.v0)); - assertEq(uint64s[1].get(t.i1), uint64(t.v1)); - } - - function testUint128MapSetAndGet(uint256) public { - uint128 u = uint128(_random()); - uint128s[0].set(0, u); - assertEq(uint128s[0].map[0], u); - unchecked { - for (uint256 t; t < 8; ++t) { - uint256 r = _random(); - uint128 casted; - /// @solidity memory-safe-assembly - assembly { - casted := r - } - uint256 index = _random() % 32; - uint128s[0].set(index, casted); - assertEq(uint128s[0].get(index), casted); - } - } - } - - function testUint128MapSetAndGet() public { - unchecked { - for (uint256 t; t < 16; ++t) { - uint256 n = 64; - uint128 casted; - uint256 r = _random(); - for (uint256 i; i < n; ++i) { - /// @solidity memory-safe-assembly - assembly { - casted := or(add(mul(n, t), i), r) - } - uint128s[0].set(i, casted); - } - for (uint256 i; i < n; ++i) { - /// @solidity memory-safe-assembly - assembly { - casted := or(add(mul(n, t), i), r) - } - assertEq(uint128s[0].get(i), casted); - } - } - } - } - - function testUint128MapSetAndGet2(uint256) public { - _TestTemps memory t = _testTemps(); - uint128s[0].set(t.i0, uint128(t.v0)); - uint128s[1].set(t.i1, uint128(t.v1)); - assertEq(uint128s[0].get(t.i0), uint128(t.v0)); - assertEq(uint128s[1].get(t.i1), uint128(t.v1)); - } - - function testUint32Maps(uint256) public { - unchecked { - uint256 a0 = _random(); - uint256 a1 = _random() % 2 == 0 ? a0 + _random() % 4 : a0 - _random() % 4; - uint256 b0 = _random(); - uint256 b1 = _random() % 2 == 0 ? b0 + _random() % 4 : b0 - _random() % 4; - if (a0 == a1 && b1 == b0) { - if (_random() % 2 == 0) { - if (_random() % 2 == 0) b1++; - else a0++; - } else { - if (_random() % 2 == 0) b1--; - else a0--; - } - } - uint256 c0 = _random(); - uint256 c1 = _random(); - uint32 c0Casted; - uint32 c1Casted; - /// @solidity memory-safe-assembly - assembly { - c0Casted := c0 - c1Casted := c1 - } - assertEq(uint32Maps[a0].get(b0), 0); - assertEq(uint32Maps[a1].get(b1), 0); - uint32Maps[a0].set(b0, c0Casted); - uint32Maps[a1].set(b1, c1Casted); - assertEq(uint32Maps[a0].get(b0), uint32(c0)); - assertEq(uint32Maps[a1].get(b1), uint32(c1)); - } - } - - struct _SearchSortedTestVars { - uint256 o; - uint256 n; - uint256 end; - bool found; - uint256 index; - uint256 randomIndex; - uint256 randomIndexValue; - uint256[] values; - } - - function _searchSortedTestVars(mapping(uint256 => uint256) storage map, uint256 bitWidth) - internal - returns (_SearchSortedTestVars memory t) - { - unchecked { - t.n = 1 + _random() % 7 + (_random() % 8 == 0 ? _random() % 64 : 0); - if (_random() % 2 == 0) { - t.o = type(uint256).max - t.n; - t.end = t.o + t.n; - assertEq(t.end, type(uint256).max); - } else { - t.o = _random() % 4 + (_random() % 8 == 0 ? type(uint256).max - 256 : 0); - t.end = t.o + t.n; - } - uint256 v = _random() % 4; - uint256 b = (_random() % 2) * (_random() << 7); - uint256 valueMask = (1 << bitWidth) - 1; - for (uint256 i; i != t.n; ++i) { - map.set(t.o + i, b | v, bitWidth); - filled.set((b | v) & valueMask, 1, 1); - v += 1 + _random() % 2; - } - t.randomIndex = t.o + _random() % t.n; - t.randomIndexValue = map.get(t.randomIndex, bitWidth); - - if (t.o > 0) map.set(t.o - 1, _random(), bitWidth); - if (t.end < type(uint256).max) map.set(t.end, _random(), bitWidth); - - uint256 notFoundValue = _generateNotFoundValue(t.o); - - (t.found, t.index) = map.searchSorted(notFoundValue, t.o, t.end, bitWidth); - assertFalse(t.found); - assertEq(t.index, _nearestIndexBefore(map, notFoundValue, t.o, t.n, bitWidth)); - - uint256 end = t.o - (t.o > 0 ? _random() % t.o : 0); - (t.found, t.index) = map.searchSorted(t.randomIndexValue, t.o, end, bitWidth); - assertFalse(t.found); - assertEq(t.index, t.o); - - (t.found, t.index) = map.searchSorted(t.randomIndexValue, t.o, t.end, bitWidth); - assertTrue(t.found); - assertEq(t.index, t.randomIndex); - } - } - - function _generateNotFoundValue(uint256 o) internal returns (uint256 notFoundValue) { - unchecked { - uint256 max = 32; - do { - notFoundValue = o + _random() % max; - max += 8; - } while (filled.get(notFoundValue, 1) == 1); - } - } - - function _nearestIndexBefore( - mapping(uint256 => uint256) storage map, - uint256 x, - uint256 o, - uint256 n, - uint256 bitWidth - ) internal view returns (uint256 nearestIndex) { - unchecked { - nearestIndex = o; - uint256 nearestDist = type(uint256).max; - for (uint256 i; i != n; ++i) { - uint256 y = map.get(o + i, bitWidth); - if (y > x) continue; - uint256 dist = x - y; - if (dist < nearestDist) { - nearestIndex = o + i; - nearestDist = dist; - } - } - } - } - - function testUint8MapSearchSorted(uint256) public { - unchecked { - LibMap.Uint8Map storage m = uint8s[0]; - _SearchSortedTestVars memory t = _searchSortedTestVars(m.map, 8); - assertEq(m.get(t.randomIndex), t.randomIndexValue); - (bool found, uint256 index) = m.searchSorted(uint8(t.randomIndexValue), t.o, t.end); - assertTrue(found == t.found && index == t.index); - } - } - - function testUint16MapSearchSorted(uint256) public { - unchecked { - LibMap.Uint16Map storage m = uint16s[0]; - _SearchSortedTestVars memory t = _searchSortedTestVars(m.map, 16); - assertEq(m.get(t.randomIndex), t.randomIndexValue); - (bool found, uint256 index) = m.searchSorted(uint16(t.randomIndexValue), t.o, t.end); - assertTrue(found == t.found && index == t.index); - } - } - - function testUint32MapSearchSorted(uint256) public { - unchecked { - LibMap.Uint32Map storage m = uint32s[0]; - _SearchSortedTestVars memory t = _searchSortedTestVars(m.map, 32); - assertEq(m.get(t.randomIndex), t.randomIndexValue); - (bool found, uint256 index) = m.searchSorted(uint32(t.randomIndexValue), t.o, t.end); - assertTrue(found == t.found && index == t.index); - } - } - - function testUint40MapSearchSorted(uint256) public { - unchecked { - LibMap.Uint40Map storage m = uint40s[0]; - _SearchSortedTestVars memory t = _searchSortedTestVars(m.map, 40); - assertEq(m.get(t.randomIndex), t.randomIndexValue); - (bool found, uint256 index) = m.searchSorted(uint40(t.randomIndexValue), t.o, t.end); - assertTrue(found == t.found && index == t.index); - } - } - - function testUint64MapSearchSorted(uint256) public { - unchecked { - LibMap.Uint64Map storage m = uint64s[0]; - _SearchSortedTestVars memory t = _searchSortedTestVars(m.map, 64); - assertEq(m.get(t.randomIndex), t.randomIndexValue); - (bool found, uint256 index) = m.searchSorted(uint64(t.randomIndexValue), t.o, t.end); - assertTrue(found == t.found && index == t.index); - } - } - - function testUint128MapSearchSorted(uint256) public { - unchecked { - LibMap.Uint128Map storage m = uint128s[0]; - _SearchSortedTestVars memory t = _searchSortedTestVars(m.map, 128); - assertEq(m.get(t.randomIndex), t.randomIndexValue); - (bool found, uint256 index) = m.searchSorted(uint128(t.randomIndexValue), t.o, t.end); - assertTrue(found == t.found && index == t.index); - } - } - - function testGeneralMapSearchSorted(uint256) public { - unchecked { - mapping(uint256 => uint256) storage m = generalMaps[0]; - uint256 bitWidth = _bound(_random(), 8, 256); - _searchSortedTestVars(m, bitWidth); - } - } - - function testGeneralMapFunctionsWithSmallBitWidths(uint256) public { - unchecked { - uint256 bitWidth = 1 + _random() % 6; - uint256 valueMask = (1 << bitWidth) - 1; - uint256 o = _random() % 64 + (_random() % 8 == 0 ? type(uint256).max - 256 : 0); - uint256 n = _random() % 9; - for (uint256 k; k != 2; ++k) { - for (uint256 i; i != n; ++i) { - uint256 j = o + i * 2; - generalMaps[k].set(j, _hash(j), bitWidth); - } - } - for (uint256 k; k != 2; ++k) { - for (uint256 i; i != n; ++i) { - uint256 j = o + i * 2 + 1; - generalMaps[k].set(j, _hash(j), bitWidth); - } - } - for (uint256 k; k != 2; ++k) { - for (uint256 i; i != n; ++i) { - uint256 j = o + i * 2; - assertEq(generalMaps[k].get(j, bitWidth), _hash(j) & valueMask); - j = j + 1; - assertEq(generalMaps[k].get(j, bitWidth), _hash(j) & valueMask); - } - } - } - } - - function _hash(uint256 x) internal pure returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, x) - result := keccak256(0x00, 0x20) - } - } - - function testGeneralMapFunctionsWithZeroBitWidth() public { - unchecked { - mapping(uint256 => uint256) storage m = generalMaps[0]; - for (uint256 j; j < 3; ++j) { - for (uint256 i; i < 3; ++i) { - m.set(i, j + 1, 0); - assertEq(m.get(i, 0), 0); - (bool found, uint256 index) = m.searchSorted(i, j, j + 2, 0); - assertFalse(found); - assertEq(index, j); - } - } - } - } - - function testGeneralMapFunctionsGas() public { - unchecked { - mapping(uint256 => uint256) storage m = generalMaps[0]; - for (uint256 i; i != 1000; ++i) { - m.set(i, i + 1, 32); - assertEq(m.get(i, 32), i + 1); - } - for (uint256 j = 1; j < 900; j += 37) { - (bool found, uint256 index) = m.searchSorted(j, 0, 1000, 32); - assertTrue(found); - assertEq(index, j - 1); - } - } - } - - function testFoundStatementDifferential(uint256 t, uint256 needle, uint256 index) public { - bool a; - bool b; - /// @solidity memory-safe-assembly - assembly { - a := and(eq(t, needle), iszero(iszero(index))) - b := iszero(or(xor(t, needle), iszero(index))) - } - assertEq(a, b); - } -} diff --git a/lib/solady/test/LibPRNG.t.sol b/lib/solady/test/LibPRNG.t.sol deleted file mode 100644 index 79bc2f6..0000000 --- a/lib/solady/test/LibPRNG.t.sol +++ /dev/null @@ -1,171 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {LibPRNG} from "../src/utils/LibPRNG.sol"; -import {LibSort} from "../src/utils/LibSort.sol"; - -contract LibPRNGTest is SoladyTest { - using LibPRNG for *; - - function testPRNGNext() public { - unchecked { - // Super unlikely to fail. - for (uint256 i; i < 32; ++i) { - LibPRNG.PRNG memory prng; - prng.seed(i); - uint256 r0 = prng.next(); - uint256 r1 = prng.next(); - uint256 r2 = prng.next(); - assertTrue(r0 != r1); - assertTrue(r1 != r2); - prng.seed(i * 2); - uint256 r3 = prng.next(); - assertTrue(r2 != r3); - } - } - } - - function testPRNGUniform() public { - unchecked { - LibPRNG.PRNG memory prng; - for (uint256 i = 1; i < 32; ++i) { - for (uint256 j; j < 32; ++j) { - assertTrue(prng.uniform(i) < i); - } - } - for (uint256 i; i < 32; ++i) { - assertTrue(prng.uniform(0) == 0); - } - // Super unlikely to fail. - uint256 previous; - for (uint256 i = 128; i < 256; ++i) { - uint256 n = 1 << i; - for (uint256 j; j < 8; ++j) { - uint256 r = prng.uniform(n); - assertTrue(r < n); - assertTrue(r != previous); - previous = r; - } - } - } - } - - function testPRNGShuffleGas() public pure { - unchecked { - uint256[] memory a = new uint256[](10000); - LibPRNG.PRNG memory prng; - prng.shuffle(a); - } - } - - function testPRNGShuffleBytesGas() public pure { - unchecked { - bytes memory a = new bytes(10000); - LibPRNG.PRNG memory prng; - prng.shuffle(a); - } - } - - function testPRNGShuffle() public { - unchecked { - LibPRNG.PRNG memory prng; - for (uint256 s = 1; s < 9; ++s) { - uint256 n = 1 << s; // 2, 4, 8, 16, ... - uint256[] memory a = new uint256[](n); - for (uint256 i; i < n; ++i) { - a[i] = i; - } - bytes32 hashBefore = keccak256(abi.encode(a)); - for (;;) { - prng.shuffle(a); - bytes32 hashAfterShuffle = keccak256(abi.encode(a)); - LibSort.sort(a); - bytes32 hashAfterSort = keccak256(abi.encode(a)); - assertTrue(hashBefore == hashAfterSort); - if (hashBefore != hashAfterShuffle) break; - } - } - // Checking that we won't crash. - for (uint256 n = 0; n < 2; ++n) { - uint256[] memory a = new uint256[](n); - prng.shuffle(a); - } - } - } - - function testPRNGShuffleBytes() public { - unchecked { - LibPRNG.PRNG memory prng; - for (uint256 s = 1; s < 9; ++s) { - uint256 n = 1 << s; // 2, 4, 8, 16, ... - bytes memory a = new bytes(n); - for (uint256 i; i < n; ++i) { - a[i] = bytes1(uint8(i & 0xff)); - } - bytes32 hashBefore = keccak256(abi.encode(a)); - uint256 checksumBefore = _bytesOrderAgnosticChecksum(a); - for (uint256 i; i < 30; ++i) { - prng.shuffle(a); - assertEq(_bytesOrderAgnosticChecksum(a), checksumBefore); - bytes32 hashAfterShuffle = keccak256(abi.encode(a)); - if (hashBefore != hashAfterShuffle) break; - } - } - // Checking that we won't crash. - for (uint256 n = 0; n < 2; ++n) { - uint256[] memory a = new uint256[](n); - prng.shuffle(a); - } - } - } - - function testLCGGas() public { - unchecked { - uint256 randomness; - for (uint256 i; i < 256; i++) { - randomness = _stepLCG(randomness); - } - assertTrue(randomness != 0); - } - } - - function testPRNGGas() public { - unchecked { - LibPRNG.PRNG memory prng; - uint256 randomness; - for (uint256 i; i < 256; i++) { - randomness = prng.next(); - } - assertTrue(randomness != 0); - } - } - - function _bytesOrderAgnosticChecksum(bytes memory a) internal pure returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - for { let n := mload(a) } n { n := sub(n, 1) } { - result := add(result, and(mload(add(a, n)), 0xff)) - } - } - } - - // This is for demonstrating that the gas savings - // over the `keccak256` approach isn't that much. - // The multiplier and the increment are chosen for good enough - // statistical test results. - // - // See: https://github.com/stevenang/randomness_testsuite - // See: https://www.pcg-random.org/posts/does-it-beat-the-minimal-standard.html - // - // The xorshift is required because the raw 128 lower bits - // of the LCG alone will not pass the tests. - function _stepLCG(uint256 state) private pure returns (uint256 randomness) { - /// @solidity memory-safe-assembly - assembly { - let a := 0xd6aad120322a96acae4ccfaf5fcd4bbfda3f2f3001db6837c0981639faa68d8d - state := add(mul(state, a), 83) - randomness := xor(state, shr(128, state)) - } - } -} diff --git a/lib/solady/test/LibRLP.t.sol b/lib/solady/test/LibRLP.t.sol deleted file mode 100644 index f01551f..0000000 --- a/lib/solady/test/LibRLP.t.sol +++ /dev/null @@ -1,383 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {LibRLP} from "../src/utils/LibRLP.sol"; - -contract LibRLPTest is SoladyTest { - function testComputeAddressDifferential(address deployer, uint256 nonce) public { - assertEq(LibRLP.computeAddress(deployer, nonce), computeAddressOriginal(deployer, nonce)); - } - - function testComputeAddressForSmallNonces() public { - address deployer = address(1); - assertTrue(LibRLP.computeAddress(deployer, 1) != address(0)); - assertTrue(LibRLP.computeAddress(deployer, 0x7f) != address(0)); - assertTrue(LibRLP.computeAddress(deployer, 0xff) != address(0)); - } - - function testComputeAddressOriginalForSmallNonces() public { - address deployer = address(1); - assertTrue(computeAddressOriginal(deployer, 1) != address(0)); - assertTrue(computeAddressOriginal(deployer, 0x7f) != address(0)); - assertTrue(computeAddressOriginal(deployer, 0xff) != address(0)); - } - - function testComputeAddressForLargeNonces() public { - address deployer = address(1); - assertTrue(LibRLP.computeAddress(deployer, 0xffffffff) != address(0)); - assertTrue(LibRLP.computeAddress(deployer, 0xffffffffffffff) != address(0)); - assertTrue(LibRLP.computeAddress(deployer, 0xffffffffffffffff) != address(0)); - } - - function testComputeAddressOriginalForLargeNonces() public { - address deployer = address(1); - assertTrue(computeAddressOriginal(deployer, 0xffffffff) != address(0)); - assertTrue(computeAddressOriginal(deployer, 0xffffffffffffff) != address(0)); - assertTrue(computeAddressOriginal(deployer, 0xffffffffffffffff) != address(0)); - } - - function computeAddressOriginal(address deployer, uint256 nonce) - internal - pure - returns (address) - { - // Although the theoretical allowed limit, based on EIP-2681, - // for an account nonce is 2**64-2: https://eips.ethereum.org/EIPS/eip-2681, - // we just test all the way to 2**256-1 to ensure that the computeAddress function does not revert - // for whatever nonce we provide. - // forgefmt: disable-next-item - if (nonce == 0x00) { - return address( - uint160(uint256(keccak256(abi.encodePacked(bytes1(0xd6), bytes1(0x94), deployer, bytes1(0x80))))) - ); - } - // forgefmt: disable-next-item - if (nonce <= 0x7f) { - return address( - uint160(uint256(keccak256(abi.encodePacked(bytes1(0xd6), bytes1(0x94), deployer, uint8(nonce))))) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint8).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xd7), bytes1(0x94), deployer, bytes1(0x81), uint8(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint16).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xd8), bytes1(0x94), deployer, bytes1(0x82), uint16(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint24).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xd9), bytes1(0x94), deployer, bytes1(0x83), uint24(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint32).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xda), bytes1(0x94), deployer, bytes1(0x84), uint32(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint40).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xdb), bytes1(0x94), deployer, bytes1(0x85), uint40(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint48).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xdc), bytes1(0x94), deployer, bytes1(0x86), uint48(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint56).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xdd), bytes1(0x94), deployer, bytes1(0x87), uint56(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint64).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xde), bytes1(0x94), deployer, bytes1(0x88), uint64(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint72).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xdf), bytes1(0x94), deployer, bytes1(0x89), uint72(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint80).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xe0), bytes1(0x94), deployer, bytes1(0x8a), uint80(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint88).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xe1), bytes1(0x94), deployer, bytes1(0x8b), uint88(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint96).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xe2), bytes1(0x94), deployer, bytes1(0x8c), uint96(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint104).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xe3), bytes1(0x94), deployer, bytes1(0x8d), uint104(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint112).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xe4), bytes1(0x94), deployer, bytes1(0x8e), uint112(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint120).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xe5), bytes1(0x94), deployer, bytes1(0x8f), uint120(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint128).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xe6), bytes1(0x94), deployer, bytes1(0x90), uint128(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint136).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xe7), bytes1(0x94), deployer, bytes1(0x91), uint136(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint144).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xe8), bytes1(0x94), deployer, bytes1(0x92), uint144(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint152).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xe9), bytes1(0x94), deployer, bytes1(0x93), uint152(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint160).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xea), bytes1(0x94), deployer, bytes1(0x94), uint160(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint168).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xeb), bytes1(0x94), deployer, bytes1(0x95), uint168(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint176).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xec), bytes1(0x94), deployer, bytes1(0x96), uint176(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint184).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xed), bytes1(0x94), deployer, bytes1(0x97), uint184(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint192).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xee), bytes1(0x94), deployer, bytes1(0x98), uint192(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint200).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xef), bytes1(0x94), deployer, bytes1(0x99), uint200(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint208).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xf0), bytes1(0x94), deployer, bytes1(0x9a), uint208(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint216).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xf1), bytes1(0x94), deployer, bytes1(0x9b), uint216(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint224).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xf2), bytes1(0x94), deployer, bytes1(0x9c), uint224(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint232).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xf3), bytes1(0x94), deployer, bytes1(0x9d), uint232(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint240).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xf4), bytes1(0x94), deployer, bytes1(0x9e), uint240(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint248).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xf5), bytes1(0x94), deployer, bytes1(0x9f), uint248(nonce))) - ) - ) - ); - } - // forgefmt: disable-next-item - if (nonce <= type(uint256).max) { - return address( - uint160( - uint256( - keccak256(abi.encodePacked(bytes1(0xf6), bytes1(0x94), deployer, bytes1(0xa0), uint256(nonce))) - ) - ) - ); - } - revert(); - } -} diff --git a/lib/solady/test/LibSort.t.sol b/lib/solady/test/LibSort.t.sol deleted file mode 100644 index 9dd557a..0000000 --- a/lib/solady/test/LibSort.t.sol +++ /dev/null @@ -1,1250 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import "src/utils/LibSort.sol"; - -contract LibSortTest is SoladyTest { - function testInsertionSortAddressesDifferential(uint256) public { - unchecked { - uint256 n = _random() % 32 == 0 ? _randomArrayLength() : _random() % 4; - address[] memory a = _randomAddresses(n); - // Make a copy of the `a` and perform insertion sort on it. - address[] memory aCopy = _copy(a); - for (uint256 i = 1; i < n; ++i) { - address key = aCopy[i]; - uint256 j = i; - while (j != 0 && aCopy[j - 1] > key) { - aCopy[j] = aCopy[j - 1]; - --j; - } - aCopy[j] = key; - } - LibSort.insertionSort(a); - assertEq(a, aCopy); - } - } - - function testInsertionSortPsuedorandom(uint256) public { - unchecked { - uint256[] memory a = _randomUints(32); - LibSort.insertionSort(a); - assertTrue(_isSorted(a)); - } - } - - function testInsertionSortPsuedorandom() public { - testInsertionSortPsuedorandom(123456789); - } - - function testSortChecksumed(uint256) public { - unchecked { - uint256 n = _randomArrayLength(); - uint256[] memory a = _randomUints(n); - uint256 checksum; - for (uint256 i = 0; i != n; ++i) { - checksum += a[i]; - } - LibSort.sort(a); - uint256 checksumAfterSort; - for (uint256 i = 0; i != n; ++i) { - checksumAfterSort += a[i]; - } - assertEq(checksum, checksumAfterSort); - assertTrue(_isSorted(a)); - } - } - - function testSortDifferential(uint256) public { - unchecked { - uint256[] memory a = _randomUints(_randomArrayLength()); - // Make a copy of the `a` and perform insertion sort on it. - uint256[] memory aCopy = _copy(a); - LibSort.insertionSort(aCopy); - LibSort.sort(a); - assertEq(a, aCopy); - } - } - - function testSort(uint256) public { - unchecked { - uint256[] memory a = _randomUints(_randomArrayLength()); - LibSort.sort(a); - assertTrue(_isSorted(a)); - } - } - - function testSortBasicCase() public { - unchecked { - uint256[] memory a = new uint256[](2); - a[0] = 3; - a[1] = 0; - LibSort.sort(a); - assertTrue(_isSorted(a)); - } - } - - function testSortPsuedorandom(uint256) public { - unchecked { - uint256[] memory a = _randomUints(100); - LibSort.sort(a); - assertTrue(_isSorted(a)); - } - } - - function testSortPsuedorandom() public { - testSortPsuedorandom(123456789); - } - - function testSortPsuedorandomNonuniform(uint256) public { - unchecked { - uint256[] memory a = new uint256[](100); - for (uint256 i; i < a.length; ++i) { - a[i] = _random() << (i & 8 == 0 ? 128 : 0); - } - LibSort.sort(a); - assertTrue(_isSorted(a)); - } - } - - function testSortPsuedorandomNonuniform() public { - testSortPsuedorandomNonuniform(123456789); - } - - function testSortSorted() public { - unchecked { - uint256 n = 100; - uint256[] memory a = new uint256[](n); - for (uint256 i; i != n; ++i) { - a[i] = i; - } - LibSort.sort(a); - assertTrue(_isSorted(a)); - } - } - - function testSortReversed() public { - unchecked { - uint256 n = 100; - uint256[] memory a = new uint256[](n); - for (uint256 i; i != n; ++i) { - a[i] = 999 - i; - } - LibSort.sort(a); - assertTrue(_isSorted(a)); - } - } - - function testSortMostlySame() public { - unchecked { - uint256 n = 100; - uint256[] memory a = new uint256[](n); - for (uint256 i; i != n; ++i) { - a[i] = i % 8 == 0 ? i : 0; - } - LibSort.sort(a); - assertTrue(_isSorted(a)); - } - } - - function testSortTestOverhead() public { - unchecked { - uint256 n = 100; - uint256[] memory a = new uint256[](n); - uint256 mask = (1 << 128) - 1; - for (uint256 i; i != n; ++i) { - a[i] = (i << 128) | (_random() & mask); - } - assertTrue(_isSorted(a)); - } - } - - function testSortAddressesPsuedorandomBrutalizeUpperBits() public { - unchecked { - uint256 n = 100; - address[] memory a = new address[](n); - for (uint256 i; i != n; ++i) { - address addr = address(uint160(_random())); - uint256 randomness = _random(); - /// @solidity memory-safe-assembly - assembly { - addr := or(addr, shl(160, randomness)) - } - a[i] = addr; - } - LibSort.sort(a); - assertTrue(_isSorted(a)); - } - } - - function testSortAddressesDifferential(uint256) public { - unchecked { - uint256 n = _randomArrayLength(); - uint256[] memory aRaw = _randomUints(n); - address[] memory a = new address[](n); - for (uint256 i; i != n; ++i) { - address addr; - uint256 addrRaw = aRaw[i]; - /// @solidity memory-safe-assembly - assembly { - addr := addrRaw - } - a[i] = addr; - } - // Make a copy of the `a` and perform insertion sort on it. - address[] memory aCopy = _copy(a); - LibSort.insertionSort(aCopy); - LibSort.sort(a); - assertEq(a, aCopy); - } - } - - function testSortAddressesPsuedorandom(uint256) public { - unchecked { - address[] memory a = _randomAddresses(100); - LibSort.sort(a); - assertTrue(_isSorted(a)); - } - } - - function testSortAddressesPsuedorandom() public { - testSortAddressesPsuedorandom(123456789); - } - - function testSortAddressesSorted() public { - unchecked { - uint256 n = 100; - address[] memory a = new address[](n); - for (uint256 i; i != n; ++i) { - a[i] = address(uint160(i)); - } - LibSort.sort(a); - assertTrue(_isSorted(a)); - } - } - - function testSortAddressesReversed() public { - unchecked { - uint256 n = 100; - address[] memory a = new address[](n); - for (uint256 i; i != n; ++i) { - a[i] = address(uint160(999 - i)); - } - LibSort.sort(a); - assertTrue(_isSorted(a)); - } - } - - function testSortOriginalPsuedorandom(uint256) public { - unchecked { - uint256 n = 100; - uint256[] memory a = new uint256[](n); - for (uint256 i; i != n; ++i) { - a[i] = _random(); - } - _sortOriginal(a); - assertTrue(_isSorted(a)); - } - } - - function testSortOriginalPsuedorandom() public { - testSortOriginalPsuedorandom(123456789); - } - - function testSortOriginalSorted() public { - unchecked { - uint256 n = 100; - uint256[] memory a = new uint256[](n); - for (uint256 i; i != n; ++i) { - a[i] = i; - } - _sortOriginal(a); - assertTrue(_isSorted(a)); - } - } - - function testSortOriginalReversed() public { - unchecked { - uint256 n = 100; - uint256[] memory a = new uint256[](n); - for (uint256 i; i != n; ++i) { - a[i] = 999 - i; - } - _sortOriginal(a); - assertTrue(_isSorted(a)); - } - } - - function testSortOriginalMostlySame() public { - unchecked { - uint256 n = 100; - uint256[] memory a = new uint256[](n); - for (uint256 i; i != n; ++i) { - a[i] = i % 8 == 0 ? i : 0; - } - _sortOriginal(a); - assertTrue(_isSorted(a)); - } - } - - function testUniquifySorted() public { - uint256[] memory a = new uint256[](5); - a[0] = 1; - a[1] = 1; - a[2] = 3; - a[3] = 3; - a[4] = 5; - LibSort.uniquifySorted(a); - assertTrue(_isSortedAndUniquified(a)); - assertEq(a.length, 3); - } - - function testUniquifySortedWithEmptyArray() public { - uint256[] memory a = new uint256[](0); - LibSort.uniquifySorted(a); - assertTrue(_isSortedAndUniquified(a)); - assertEq(a.length, 0); - } - - function testUniquifySortedAddress() public { - address[] memory a = new address[](10); - a[0] = address(0x1efF47bc3a10a45D4B230B5d10E37751FE6AA718); - a[1] = address(0x1efF47bc3a10a45D4B230B5d10E37751FE6AA718); - a[2] = address(0x1efF47bC3A10a45d4b630B5D10E37751FE6aA718); - a[3] = address(0x2B5AD5c4795c026514f8317c7a215E218DcCD6cF); - a[4] = address(0x6813Eb9362372EEF6200f3b1dbC3f819671cBA69); - a[5] = address(0x6813eb9362372Eef6200f3B1dbC3f819671cbA70); - a[6] = address(0xe1AB8145F7E55DC933d51a18c793F901A3A0b276); - a[7] = address(0xe1AB8145F7E55DC933d51a18c793F901A3A0b276); - a[8] = address(0xE1Ab8145F7e55Dc933D61a18c793f901A3a0B276); - a[9] = address(0xe1ab8145f7E55Dc933D61A18c793f901A3A0B288); - LibSort.uniquifySorted(a); - assertTrue(_isSortedAndUniquified(a)); - assertEq(a.length, 8); - } - - function testUniquifySorted(uint256) public { - uint256[] memory a = _randomUints(_randomArrayLength()); - LibSort.sort(a); - LibSort.uniquifySorted(a); - assertTrue(_isSortedAndUniquified(a)); - } - - function testUniquifySortedAddress(uint256) public { - address[] memory a = _randomAddresses(_randomArrayLength()); - LibSort.sort(a); - LibSort.uniquifySorted(a); - assertTrue(_isSortedAndUniquified(a)); - } - - function testUniquifySortedDifferential(uint256) public { - uint256 n = _randomArrayLength(); - uint256[] memory a = _randomUints(n); - LibSort.sort(a); - uint256[] memory aCopy = new uint256[](n); - for (uint256 i; i != n; ++i) { - aCopy[i] = a[i]; - } - LibSort.uniquifySorted(a); - _uniquifyOriginal(aCopy); - assertEq(a, aCopy); - } - - function testSearchSortedBasicCases() public { - uint256[] memory a = new uint256[](6); - a[0] = 0; - a[1] = 1; - a[2] = 2; - a[3] = 3; - a[4] = 4; - a[5] = 5; - (bool found, uint256 index) = LibSort.searchSorted(a, 2); - assertTrue(found); - assertEq(index, 2); - - a[0] = 0; - a[1] = 1; - a[2] = 2; - a[3] = 3; - a[4] = 4; - a[5] = 5; - (found, index) = LibSort.searchSorted(a, 5); - assertTrue(found); - assertEq(index, 5); - } - - function testSearchSortedEdgeCases() public { - uint256[] memory a = new uint256[](1); - a[0] = 2; - (bool found, uint256 index) = LibSort.searchSorted(a, 1); - assertFalse(found); - - a = new uint256[](2); - a[0] = 45; - a[1] = 46; - (found, index) = LibSort.searchSorted(a, 2); - assertFalse(found); - } - - function testSearchSortedWithEmptyArray() public { - uint256[] memory a = new uint256[](0); - (bool found, uint256 index) = LibSort.searchSorted(a, 1); - assertFalse(found); - assertEq(index, 0); - } - - function testSearchSortedElementNotInArray() public { - uint256[] memory a = new uint256[](5); - a[0] = 1; - a[1] = 2; - a[2] = 3; - a[3] = 4; - a[4] = 5; - (bool found, uint256 index) = LibSort.searchSorted(a, 0); - assertFalse(found); - assertEq(index, 0); - - a[0] = 15; - a[1] = 25; - a[2] = 35; - a[3] = 45; - a[4] = 55; - (found, index) = LibSort.searchSorted(a, 10); - assertFalse(found); - assertEq(index, 0); - (found, index) = LibSort.searchSorted(a, 20); - assertFalse(found); - assertEq(index, 0); - (found, index) = LibSort.searchSorted(a, 30); - assertFalse(found); - assertEq(index, 1); - (found, index) = LibSort.searchSorted(a, 40); - assertFalse(found); - assertEq(index, 2); - (found, index) = LibSort.searchSorted(a, 50); - assertFalse(found); - assertEq(index, 3); - (found, index) = LibSort.searchSorted(a, 60); - assertFalse(found); - assertEq(index, 4); - } - - function testSearchSortedElementInArray(uint256) public { - unchecked { - _misalignFreeMemoryPointer(); - uint256[] memory a = _randomUints(_randomNonZeroArrayLength()); - LibSort.sort(a); - if (_random() % 2 == 0) { - LibSort.uniquifySorted(a); - } - uint256 randomIndex = _random() % a.length; - uint256 value = a[randomIndex]; - (bool found, uint256 index) = LibSort.searchSorted(a, value); - assertTrue(found); - assertEq(a[index], value); - } - } - - function testSearchSortedElementNotInArray(uint256) public { - unchecked { - _misalignFreeMemoryPointer(); - uint256[] memory a = _randomUints(_randomNonZeroArrayLength()); - LibSort.sort(a); - if (_random() % 2 == 0) { - LibSort.uniquifySorted(a); - } - uint256 randomIndex = _random() % a.length; - uint256 missingValue; - if (_random() % 2 == 0) { - if (_random() % 2 == 0) { - missingValue = a[randomIndex] + 1; - if (missingValue == 0) return; - } else { - missingValue = a[randomIndex] - 1; - if (missingValue == type(uint256).max) return; - } - if (_exists(a, missingValue)) return; - (bool found, uint256 index) = LibSort.searchSorted(a, missingValue); - assertFalse(found); - assertEq(a[index], a[_nearestIndexBefore(a, missingValue)]); - } else { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, missingValue) - missingValue := keccak256(0x00, 0x20) - } - (bool found,) = LibSort.searchSorted(a, missingValue); - assertFalse(found); - } - } - } - - function _exists(uint256[] memory a, uint256 x) internal pure returns (bool result) { - unchecked { - uint256 n = a.length; - for (uint256 i; i != n; ++i) { - if (a[i] == x) { - return true; - } - } - return false; - } - } - - function _nearestIndexBefore(uint256[] memory a, uint256 x) - internal - pure - returns (uint256 nearestIndex) - { - unchecked { - uint256 nearestDist = type(uint256).max; - uint256 n = a.length; - for (uint256 i; i != n; ++i) { - uint256 y = a[i]; - if (y > x) continue; - uint256 dist = x - y; - if (dist < nearestDist) { - nearestIndex = i; - nearestDist = dist; - } - } - } - } - - function testSearchSorted() public { - unchecked { - uint256 n = 100; - uint256[] memory a = new uint256[](n); - for (uint256 i; i != n; i++) { - a[i] = i; - } - for (uint256 i; i != n; i++) { - (bool found, uint256 index) = LibSort.searchSorted(a, i); - assertTrue(found); - assertEq(index, i); - } - } - } - - function testSearchSortedDifferential(uint256) public { - uint256[] memory a = _randomUints(_randomArrayLength()); - uint256 needle = _random(); - if (_random() % 2 == 0 && a.length != 0) { - needle = a[_random() % a.length]; - } - (bool found, uint256 index) = LibSort.searchSorted(a, needle); - if (found) { - assertEq(a[index], needle); - } - LibSort.sort(a); - (found, index) = LibSort.searchSorted(a, needle); - assertEq(found, _exists(a, needle)); - if (found) { - assertEq(a[index], needle); - } - } - - function testSearchSortedInts() public { - unchecked { - uint256 n = 100; - int256[] memory a = new int256[](n); - for (uint256 i = 0; i != n; i++) { - a[i] = int256(i) - 50; - } - for (uint256 i; i != n; i++) { - (bool found, uint256 index) = LibSort.searchSorted(a, int256(i) - 50); - assertTrue(found); - assertEq(index, i); - } - } - } - - function testSearchSortedInts(int256[] memory a, int256 needle) public { - (bool found, uint256 index) = LibSort.searchSorted(a, needle); - if (found) { - assertEq(a[index], needle); - } - } - - function testSearchSortedAddresses() public { - unchecked { - uint256 n = 100; - address[] memory a = new address[](n); - for (uint256 i; i != n; i++) { - a[i] = address(uint160(i)); - } - for (uint256 i; i != n; i++) { - (bool found, uint256 index) = LibSort.searchSorted(a, address(uint160(i))); - assertTrue(found); - assertEq(index, i); - (found,) = LibSort.searchSorted(a, address(uint160(i + n))); - assertFalse(found); - } - } - } - - function testInsertionSortInts() public { - unchecked { - for (uint256 t; t != 16; ++t) { - int256[] memory a = _randomInts(_bound(_random(), 0, 8)); - LibSort.insertionSort(a); - assertTrue(_isSorted(a)); - } - } - } - - function testSortInts() public { - unchecked { - for (uint256 t; t != 16; ++t) { - int256[] memory a = _randomInts(_bound(_random(), 0, 64)); - LibSort.insertionSort(a); - assertTrue(_isSorted(a)); - } - } - } - - function testTwoComplementConversionSort(int256 a, int256 b) public { - uint256 w = 1 << 255; - /// @solidity memory-safe-assembly - assembly { - let aConverted := add(a, w) - let bConverted := add(b, w) - if iszero(lt(aConverted, bConverted)) { - let t := aConverted - aConverted := bConverted - bConverted := t - } - a := add(aConverted, w) - b := add(bConverted, w) - } - assertTrue(a <= b); - } - - function testReverse() public { - unchecked { - for (uint256 t; t != 16; ++t) { - uint256 n = _bound(_random(), 0, 8); - uint256[] memory a = new uint256[](n); - uint256[] memory reversed = new uint256[](n); - for (uint256 i; i != n; ++i) { - reversed[n - 1 - i] = (a[i] = _random()); - } - bytes32 originalHash = keccak256(abi.encode(a)); - LibSort.reverse(a); - assertEq(a, reversed); - LibSort.reverse(a); - assertEq(originalHash, keccak256(abi.encode(a))); - } - } - } - - function testSortedUnionDifferential(uint256) public { - (uint256[] memory a, uint256[] memory b) = _randomUintsPair(); - uint256[] memory c = LibSort.union(a, b); - assertTrue(_isSorted(c)); - assertEq(c, _unionOriginal(a, b)); - } - - function testSortedUnionDifferential() public { - unchecked { - for (uint256 t; t != 16; ++t) { - testSortedUnionDifferential(t); - } - } - } - - function testSortedUnionDifferentialInt(uint256) public { - (int256[] memory a, int256[] memory b) = _randomIntsPair(); - int256[] memory c = LibSort.union(a, b); - assertTrue(_isSorted(c)); - assertEq(c, _unionOriginal(a, b)); - } - - function testSortedIntersectionDifferential(uint256) public { - (uint256[] memory a, uint256[] memory b) = _randomUintsPair(); - uint256[] memory c = LibSort.intersection(a, b); - assertTrue(_isSorted(c)); - assertEq(c, _intersectionOriginal(a, b)); - } - - function testSortedIntersectionDifferential() public { - unchecked { - for (uint256 t; t != 16; ++t) { - testSortedIntersectionDifferential(t); - } - } - } - - function testSortedIntersectionDifferentialInt(uint256) public { - (int256[] memory a, int256[] memory b) = _randomIntsPair(); - int256[] memory c = LibSort.intersection(a, b); - assertTrue(_isSorted(c)); - assertEq(c, _intersectionOriginal(a, b)); - } - - function testSortedDifferenceDifferential(uint256) public { - (uint256[] memory a, uint256[] memory b) = _randomUintsPair(); - uint256[] memory c = LibSort.difference(a, b); - assertTrue(_isSorted(c)); - assertEq(c, _differenceOriginal(a, b)); - } - - function testSortedDifferenceDifferential() public { - unchecked { - for (uint256 t; t != 16; ++t) { - testSortedDifferenceDifferential(t); - } - } - } - - function testSortedDifferenceDifferentialInt(uint256) public { - (int256[] memory a, int256[] memory b) = _randomIntsPair(); - int256[] memory c = LibSort.difference(a, b); - assertTrue(_isSorted(c)); - assertEq(c, _differenceOriginal(a, b)); - } - - function testSortedDifferenceUnionIntersection(uint256) public { - unchecked { - bool found; - (uint256[] memory a, uint256[] memory b) = _randomUintsPair(); - - uint256[] memory aSubB = LibSort.difference(a, b); - assertTrue(_isSorted(aSubB)); - for (uint256 i; i != aSubB.length; ++i) { - (found,) = LibSort.searchSorted(a, aSubB[i]); - assertTrue(found); - (found,) = LibSort.searchSorted(b, aSubB[i]); - assertFalse(found); - } - for (uint256 i; i != b.length; ++i) { - (found,) = LibSort.searchSorted(aSubB, b[i]); - assertFalse(found); - } - - uint256[] memory bSubA = LibSort.difference(b, a); - assertTrue(_isSorted(bSubA)); - for (uint256 i; i != bSubA.length; ++i) { - (found,) = LibSort.searchSorted(b, bSubA[i]); - assertTrue(found); - (found,) = LibSort.searchSorted(a, bSubA[i]); - assertFalse(found); - } - for (uint256 i; i != a.length; ++i) { - (found,) = LibSort.searchSorted(bSubA, a[i]); - assertFalse(found); - } - - uint256[] memory aIntersectionB = LibSort.intersection(a, b); - for (uint256 i; i != aIntersectionB.length; ++i) { - (found,) = LibSort.searchSorted(b, aIntersectionB[i]); - assertTrue(found); - (found,) = LibSort.searchSorted(a, aIntersectionB[i]); - assertTrue(found); - } - - uint256[] memory aUnionB = LibSort.union(a, b); - uint256[] memory aSubBUnionBSubA = LibSort.union(aSubB, bSubA); - uint256[] memory emptySet; - assertEq(emptySet, LibSort.intersection(aSubB, bSubA)); - assertEq(emptySet, LibSort.intersection(aSubBUnionBSubA, aIntersectionB)); - assertEq(a, LibSort.union(aIntersectionB, aSubB)); - assertEq(b, LibSort.union(aIntersectionB, bSubA)); - assertEq(aIntersectionB, LibSort.intersection(b, a)); - assertEq(aUnionB, LibSort.union(b, a)); - assertEq(LibSort.union(aSubB, b), LibSort.union(b, aSubB)); - assertEq(LibSort.union(bSubA, a), LibSort.union(a, bSubA)); - assertEq(aUnionB, LibSort.union(aSubBUnionBSubA, aIntersectionB)); - } - } - - function testIsSortedDifferential(uint256) public { - unchecked { - uint256 n = _random() % 2 == 0 ? _random() % 4 : _randomArrayLength(); - uint256[] memory a = new uint256[](n); - for (uint256 i; i != n; ++i) { - a[i] = _random() % 4; - } - assertEq(LibSort.isSorted(a), _isSorted(a)); - LibSort.sort(a); - assertEq(LibSort.isSorted(a), _isSorted(a)); - if (n != 0) { - a[_random() % n] = 0; - if (_random() % 2 == 0) { - a[_random() % n] = a[_random() % n]; - } - } - assertEq(LibSort.isSorted(a), _isSorted(a)); - } - } - - function testIsSortedIntsDifferential(uint256) public { - unchecked { - uint256 n = _random() % 2 == 0 ? _random() % 4 : _randomArrayLength(); - int256[] memory a = new int256[](n); - for (uint256 i; i != n; ++i) { - a[i] = int256(_random() % 4); - if (_random() % 2 == 0) { - a[i] = -a[i]; - } - } - assertEq(LibSort.isSorted(a), _isSorted(a)); - LibSort.sort(a); - assertEq(LibSort.isSorted(a), _isSorted(a)); - if (n != 0) { - a[_random() % n] = 0; - if (_random() % 2 == 0) { - a[_random() % n] = a[_random() % n]; - } - } - assertEq(LibSort.isSorted(a), _isSorted(a)); - } - } - - function testIsSortedAddressesDifferential(uint256) public { - unchecked { - uint256 n = _random() % 2 == 0 ? _random() % 4 : _randomArrayLength(); - address[] memory a = new address[](n); - for (uint256 i; i != n; ++i) { - a[i] = address(uint160(_random() % 4)); - } - assertEq(LibSort.isSorted(a), _isSorted(a)); - LibSort.sort(a); - assertEq(LibSort.isSorted(a), _isSorted(a)); - if (n != 0) { - a[_random() % n] = address(0); - if (_random() % 2 == 0) { - a[_random() % n] = a[_random() % n]; - } - } - assertEq(LibSort.isSorted(a), _isSorted(a)); - } - } - - function testIsSortedAndUniquifiedDifferential(uint256) public { - unchecked { - uint256 n = _random() % 2 == 0 ? _random() % 4 : _randomArrayLength(); - uint256[] memory a = new uint256[](n); - for (uint256 i; i != n; ++i) { - a[i] = _random() % 4; - } - assertEq(LibSort.isSortedAndUniquified(a), _isSortedAndUniquified(a)); - LibSort.sort(a); - assertEq(LibSort.isSortedAndUniquified(a), _isSortedAndUniquified(a)); - if (n != 0) { - a[_random() % n] = 0; - if (_random() % 2 == 0) { - a[_random() % n] = a[_random() % n]; - } - } - assertEq(LibSort.isSortedAndUniquified(a), _isSortedAndUniquified(a)); - } - } - - function testIsSortedAndUniquifiedIntsDifferential(uint256) public { - unchecked { - uint256 n = _random() % 2 == 0 ? _random() % 4 : _randomArrayLength(); - int256[] memory a = new int256[](n); - for (uint256 i; i != n; ++i) { - a[i] = int256(_random() % 4); - if (_random() % 2 == 0) { - a[i] = -a[i]; - } - } - assertEq(LibSort.isSortedAndUniquified(a), _isSortedAndUniquified(a)); - LibSort.sort(a); - assertEq(LibSort.isSortedAndUniquified(a), _isSortedAndUniquified(a)); - if (n != 0) { - a[_random() % n] = 0; - if (_random() % 2 == 0) { - a[_random() % n] = a[_random() % n]; - } - } - assertEq(LibSort.isSortedAndUniquified(a), _isSortedAndUniquified(a)); - } - } - - function testIsSortedAndUniquifiedAddressesDifferential(uint256) public { - unchecked { - uint256 n = _random() % 2 == 0 ? _random() % 4 : _randomArrayLength(); - address[] memory a = new address[](n); - for (uint256 i; i != n; ++i) { - a[i] = address(uint160(_random() % 4)); - } - assertEq(LibSort.isSortedAndUniquified(a), _isSortedAndUniquified(a)); - LibSort.sort(a); - assertEq(LibSort.isSortedAndUniquified(a), _isSortedAndUniquified(a)); - if (n != 0) { - a[_random() % n] = address(0); - if (_random() % 2 == 0) { - a[_random() % n] = a[_random() % n]; - } - } - assertEq(LibSort.isSortedAndUniquified(a), _isSortedAndUniquified(a)); - } - } - - function _unionOriginal(uint256[] memory a, uint256[] memory b) - private - pure - returns (uint256[] memory c) - { - unchecked { - c = new uint256[](a.length + b.length); - uint256 o; - for (uint256 i; i != a.length; ++i) { - c[o++] = a[i]; - } - for (uint256 i; i != b.length; ++i) { - c[o++] = b[i]; - } - LibSort.insertionSort(c); - LibSort.uniquifySorted(c); - } - } - - function _unionOriginal(int256[] memory a, int256[] memory b) - private - pure - returns (int256[] memory c) - { - unchecked { - c = new int256[](a.length + b.length); - uint256 o; - for (uint256 i; i != a.length; ++i) { - c[o++] = a[i]; - } - for (uint256 i; i != b.length; ++i) { - c[o++] = b[i]; - } - LibSort.insertionSort(c); - LibSort.uniquifySorted(c); - } - } - - function _intersectionOriginal(uint256[] memory a, uint256[] memory b) - private - pure - returns (uint256[] memory c) - { - unchecked { - c = new uint256[](a.length + b.length); - uint256 o; - bool found; - for (uint256 i; i != a.length; ++i) { - (found,) = LibSort.searchSorted(b, a[i]); - if (found) c[o++] = a[i]; - } - /// @solidity memory-safe-assembly - assembly { - mstore(c, o) - } - LibSort.insertionSort(c); - LibSort.uniquifySorted(c); - } - } - - function _intersectionOriginal(int256[] memory a, int256[] memory b) - private - pure - returns (int256[] memory c) - { - unchecked { - c = new int256[](a.length + b.length); - uint256 o; - bool found; - for (uint256 i; i != a.length; ++i) { - (found,) = LibSort.searchSorted(b, a[i]); - if (found) c[o++] = a[i]; - } - /// @solidity memory-safe-assembly - assembly { - mstore(c, o) - } - LibSort.insertionSort(c); - LibSort.uniquifySorted(c); - } - } - - function _differenceOriginal(uint256[] memory a, uint256[] memory b) - private - pure - returns (uint256[] memory c) - { - unchecked { - c = new uint256[](a.length + b.length); - uint256 o; - bool found; - for (uint256 i; i != a.length; ++i) { - (found,) = LibSort.searchSorted(b, a[i]); - if (!found) c[o++] = a[i]; - } - /// @solidity memory-safe-assembly - assembly { - mstore(c, o) - } - LibSort.insertionSort(c); - LibSort.uniquifySorted(c); - } - } - - function _differenceOriginal(int256[] memory a, int256[] memory b) - private - pure - returns (int256[] memory c) - { - unchecked { - c = new int256[](a.length + b.length); - uint256 o; - bool found; - for (uint256 i; i != a.length; ++i) { - (found,) = LibSort.searchSorted(b, a[i]); - if (!found) c[o++] = a[i]; - } - /// @solidity memory-safe-assembly - assembly { - mstore(c, o) - } - LibSort.insertionSort(c); - LibSort.uniquifySorted(c); - } - } - - function _isSorted(address[] memory a) private pure returns (bool) { - unchecked { - for (uint256 i = 1; i < a.length; ++i) { - if (a[i - 1] > a[i]) return false; - } - return true; - } - } - - function _isSorted(uint256[] memory a) private pure returns (bool) { - unchecked { - for (uint256 i = 1; i < a.length; ++i) { - if (a[i - 1] > a[i]) return false; - } - return true; - } - } - - function _isSorted(int256[] memory a) private pure returns (bool) { - unchecked { - for (uint256 i = 1; i < a.length; ++i) { - if (a[i - 1] > a[i]) return false; - } - return true; - } - } - - function _isSortedAndUniquified(uint256[] memory a) private pure returns (bool) { - if (a.length == 0) { - return true; - } - unchecked { - uint256 end = a.length - 1; - for (uint256 i = 0; i != end; ++i) { - if (a[i] >= a[i + 1]) { - return false; - } - } - return true; - } - } - - function _isSortedAndUniquified(int256[] memory a) private pure returns (bool) { - if (a.length == 0) { - return true; - } - unchecked { - uint256 end = a.length - 1; - for (uint256 i = 0; i != end; ++i) { - if (a[i] >= a[i + 1]) { - return false; - } - } - return true; - } - } - - function _isSortedAndUniquified(address[] memory a) private pure returns (bool) { - if (a.length == 0) { - return true; - } - unchecked { - uint256 end = a.length - 1; - for (uint256 i = 0; i != end; ++i) { - if (a[i] >= a[i + 1]) { - return false; - } - } - return true; - } - } - - function _sortOriginal(uint256[] memory a) internal pure { - _sortOriginal(a, 0, int256(a.length - 1)); - } - - function _sortOriginal(uint256[] memory arr, int256 left, int256 right) internal pure { - int256 i = left; - int256 j = right; - if (i == j) return; - uint256 pivot = arr[uint256(left + (right - left) / 2)]; - while (i <= j) { - while (arr[uint256(i)] < pivot) { - unchecked { - ++i; - } - } - while (pivot < arr[uint256(j)]) { - unchecked { - --j; - } - } - if (i <= j) { - (arr[uint256(i)], arr[uint256(j)]) = (arr[uint256(j)], arr[uint256(i)]); - unchecked { - ++i; - --j; - } - } - } - if (left < j) _sortOriginal(arr, left, j); - if (i < right) _sortOriginal(arr, i, right); - } - - function _copy(uint256[] memory a) private view returns (uint256[] memory b) { - /// @solidity memory-safe-assembly - assembly { - b := mload(0x40) - let n := add(shl(5, mload(a)), 0x20) - pop(staticcall(gas(), 4, a, n, b, n)) - mstore(0x40, add(b, n)) - } - } - - function _copy(int256[] memory a) private view returns (int256[] memory b) { - /// @solidity memory-safe-assembly - assembly { - b := mload(0x40) - let n := add(shl(5, mload(a)), 0x20) - pop(staticcall(gas(), 4, a, n, b, n)) - mstore(0x40, add(b, n)) - } - } - - function _copy(address[] memory a) private view returns (address[] memory b) { - /// @solidity memory-safe-assembly - assembly { - b := mload(0x40) - let n := add(shl(5, mload(a)), 0x20) - pop(staticcall(gas(), 4, a, n, b, n)) - mstore(0x40, add(b, n)) - } - } - - function _randomUints(uint256 n) private returns (uint256[] memory a) { - unchecked { - _misalignFreeMemoryPointer(); - /// @solidity memory-safe-assembly - assembly { - a := mload(0x40) - mstore(a, n) - mstore(0x40, add(add(0x20, a), shl(5, n))) - } - for (uint256 i; i != n; ++i) { - a[i] = _random(); - } - } - } - - function _randomAddresses(uint256 n) private returns (address[] memory a) { - unchecked { - _misalignFreeMemoryPointer(); - /// @solidity memory-safe-assembly - assembly { - a := mload(0x40) - mstore(a, n) - mstore(0x40, add(add(0x20, a), shl(5, n))) - } - for (uint256 i; i != n; ++i) { - a[i] = address(uint160(_random())); - } - } - } - - function _randomInts(uint256 n) private returns (int256[] memory a) { - unchecked { - uint256[] memory aRaw = _randomUints(n); - /// @solidity memory-safe-assembly - assembly { - a := aRaw - } - } - } - - function _uniquifyOriginal(uint256[] memory a) private pure { - if (a.length != 0) { - unchecked { - uint256 n = a.length; - uint256 i = 0; - for (uint256 j = 1; j < n; j++) { - if (a[i] != a[j]) { - i++; - a[i] = a[j]; - } - } - /// @solidity memory-safe-assembly - assembly { - mstore(a, add(i, 1)) - } - } - } - } - - function _randomUintsPair() private returns (uint256[] memory a, uint256[] memory b) { - uint256 r = _random(); - a = _randomUints(r & 7); - b = _randomUints((r >> 128) & 7); - LibSort.insertionSort(a); - LibSort.uniquifySorted(a); - LibSort.insertionSort(b); - LibSort.uniquifySorted(b); - } - - function _randomAddressesPair() private returns (address[] memory a, address[] memory b) { - uint256 r = _random(); - a = _randomAddresses(r & 7); - b = _randomAddresses((r >> 128) & 7); - LibSort.insertionSort(a); - LibSort.uniquifySorted(a); - LibSort.insertionSort(b); - LibSort.uniquifySorted(b); - } - - function _randomIntsPair() private returns (int256[] memory a, int256[] memory b) { - uint256 r = _random(); - a = _randomInts(r & 7); - b = _randomInts((r >> 128) & 7); - LibSort.insertionSort(a); - LibSort.uniquifySorted(a); - LibSort.insertionSort(b); - LibSort.uniquifySorted(b); - } - - function _randomArrayLength() internal returns (uint256 r) { - r = _random(); - /// @solidity memory-safe-assembly - assembly { - let m := 0x070707070707070707070707070707070f0f0f0f0f0f0f1f1f1f1f1f1f3f7fff - r := and(byte(1, r), byte(and(r, 31), m)) - } - } - - function _randomNonZeroArrayLength() internal returns (uint256 r) { - do { - r = _randomArrayLength(); - } while (r == 0); - } -} diff --git a/lib/solady/test/LibString.t.sol b/lib/solady/test/LibString.t.sol deleted file mode 100644 index c0f7ce3..0000000 --- a/lib/solady/test/LibString.t.sol +++ /dev/null @@ -1,1392 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {LibString} from "../src/utils/LibString.sol"; - -contract LibStringTest is SoladyTest { - function testToStringZero() public { - assertEq(LibString.toString(uint256(0)), "0"); - } - - function testToStringPositiveNumber() public { - assertEq(LibString.toString(uint256(4132)), "4132"); - } - - function testToStringUint256Max() public { - assertEq( - LibString.toString(type(uint256).max), - "115792089237316195423570985008687907853269984665640564039457584007913129639935" - ); - } - - function testToStringZeroBrutalized() public brutalizeMemory { - string memory s0 = LibString.toString(uint256(0)); - /// @solidity memory-safe-assembly - assembly { - mstore(mload(0x40), not(0)) - mstore(0x40, add(mload(0x40), 0x20)) - } - string memory s1 = LibString.toString(uint256(0)); - /// @solidity memory-safe-assembly - assembly { - mstore(mload(0x40), not(0)) - mstore(0x40, add(mload(0x40), 0x20)) - } - assertEq(s0, "0"); - assertEq(s1, "0"); - } - - function testToStringPositiveNumberBrutalized() public brutalizeMemory { - string memory s0 = LibString.toString(uint256(4132)); - /// @solidity memory-safe-assembly - assembly { - mstore(mload(0x40), not(0)) - mstore(0x40, add(mload(0x40), 0x20)) - } - string memory s1 = LibString.toString(uint256(4132)); - /// @solidity memory-safe-assembly - assembly { - mstore(mload(0x40), not(0)) - mstore(0x40, add(mload(0x40), 0x20)) - } - assertEq(s0, "4132"); - assertEq(s1, "4132"); - } - - function testToStringUint256MaxBrutalized() public brutalizeMemory { - string memory s0 = LibString.toString(type(uint256).max); - /// @solidity memory-safe-assembly - assembly { - mstore(mload(0x40), not(0)) - mstore(0x40, add(mload(0x40), 0x20)) - } - string memory s1 = LibString.toString(type(uint256).max); - /// @solidity memory-safe-assembly - assembly { - mstore(mload(0x40), not(0)) - mstore(0x40, add(mload(0x40), 0x20)) - } - assertEq( - s0, "115792089237316195423570985008687907853269984665640564039457584007913129639935" - ); - assertEq( - s1, "115792089237316195423570985008687907853269984665640564039457584007913129639935" - ); - } - - function testToStringZeroRightPadded(uint256 x) public view brutalizeMemory { - _checkMemory(LibString.toString(x)); - } - - function testToStringSignedDifferential(int256 x) public brutalizeMemory { - assertEq(LibString.toString(x), _toStringSignedOriginal(x)); - } - - function testToStringSignedMemory(int256 x) public view brutalizeMemory { - _misalignFreeMemoryPointer(); - uint256 freeMemoryPointer; - /// @solidity memory-safe-assembly - assembly { - freeMemoryPointer := mload(0x40) - } - string memory str = LibString.toString(x); - /// @solidity memory-safe-assembly - assembly { - if lt(str, freeMemoryPointer) { revert(0, 0) } - } - _checkMemory(str); - } - - function testToStringSignedGas() public pure { - for (int256 x = -10; x < 10; ++x) { - LibString.toString(x); - } - } - - function testToStringSignedOriginalGas() public pure { - for (int256 x = -10; x < 10; ++x) { - _toStringSignedOriginal(x); - } - } - - function _toStringSignedOriginal(int256 x) internal pure returns (string memory) { - unchecked { - return x >= 0 - ? LibString.toString(uint256(x)) - : string(abi.encodePacked("-", LibString.toString(uint256(-x)))); - } - } - - function testToHexStringZero() public { - assertEq(LibString.toHexString(0), "0x00"); - } - - function testToHexStringPositiveNumber() public { - assertEq(LibString.toHexString(0x4132), "0x4132"); - } - - function testToHexStringUint256Max() public { - assertEq( - LibString.toHexString(type(uint256).max), - "0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" - ); - } - - function testToHexStringFixedLengthPositiveNumberLong() public { - assertEq( - LibString.toHexString(0x4132, 32), - "0x0000000000000000000000000000000000000000000000000000000000004132" - ); - } - - function testToHexStringFixedLengthPositiveNumberShort() public { - assertEq(LibString.toHexString(0x4132, 2), "0x4132"); - } - - function testToHexStringZeroRightPadded(uint256 x) public pure { - _checkMemory(LibString.toHexString(x)); - } - - function testToHexStringFixedLengthInsufficientLength() public { - vm.expectRevert(LibString.HexLengthInsufficient.selector); - LibString.toHexString(0x4132, 1); - } - - function testToHexStringFixedLengthUint256Max() public { - assertEq( - LibString.toHexString(type(uint256).max, 32), - "0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" - ); - } - - function testToHexStringFixedLengthZeroRightPadded(uint256 x, uint256 randomness) public pure { - uint256 minLength = (bytes(LibString.toHexString(x)).length - 2) * 2; - uint256 length = (randomness % 32) + minLength; - _checkMemory(LibString.toHexString(x, length)); - } - - function testFromAddressToHexString() public { - assertEq( - LibString.toHexString(0xA9036907dCcae6a1E0033479B12E837e5cF5a02f), - "0xa9036907dccae6a1e0033479b12e837e5cf5a02f" - ); - } - - function testAddressToHexStringZeroRightPadded(address x) public pure { - _checkMemory(LibString.toHexString(x)); - } - - function testFromAddressToHexStringWithLeadingZeros() public { - assertEq( - LibString.toHexString(0x0000E0Ca771e21bD00057F54A68C30D400000000), - "0x0000e0ca771e21bd00057f54a68c30d400000000" - ); - } - - function testToMinimalHexStringZero() public { - assertEq(LibString.toMinimalHexString(0), "0x0"); - } - - function testToMinimalHexStringPositiveNumber() public { - assertEq(LibString.toMinimalHexString(0x54132), "0x54132"); - assertEq(LibString.toMinimalHexString(0x4132), "0x4132"); - assertEq(LibString.toMinimalHexString(0x0123), "0x123"); - assertEq(LibString.toMinimalHexString(0x12), "0x12"); - assertEq(LibString.toMinimalHexString(0x1), "0x1"); - } - - function testToMinimalHexStringUint256Max() public { - assertEq( - LibString.toMinimalHexString(type(uint256).max), - "0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" - ); - } - - function testToMinimalHexStringZeroRightPadded(uint256 x) public pure { - _checkMemory(LibString.toMinimalHexString(x)); - } - - function testToMinimalHexStringNoPrefixZero() public { - assertEq(LibString.toMinimalHexStringNoPrefix(0), "0"); - } - - function testToMinimalHexStringNoPrefixPositiveNumber() public { - assertEq(LibString.toMinimalHexStringNoPrefix(0x54132), "54132"); - assertEq(LibString.toMinimalHexStringNoPrefix(0x4132), "4132"); - assertEq(LibString.toMinimalHexStringNoPrefix(0x0123), "123"); - assertEq(LibString.toMinimalHexStringNoPrefix(0x12), "12"); - assertEq(LibString.toMinimalHexStringNoPrefix(0x1), "1"); - } - - function testToMinimalHexStringNoPrefixUint256Max() public { - assertEq( - LibString.toMinimalHexStringNoPrefix(type(uint256).max), - "ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" - ); - } - - function testToMinimalHexStringNoPrefixZeroRightPadded(uint256 x) public pure { - _checkMemory(LibString.toMinimalHexStringNoPrefix(x)); - } - - function testFromAddressToHexStringChecksummed() public { - // All caps. - assertEq( - LibString.toHexStringChecksummed(0x52908400098527886E0F7030069857D2E4169EE7), - "0x52908400098527886E0F7030069857D2E4169EE7" - ); - assertEq( - LibString.toHexStringChecksummed(0x8617E340B3D01FA5F11F306F4090FD50E238070D), - "0x8617E340B3D01FA5F11F306F4090FD50E238070D" - ); - // All lower. - assertEq( - LibString.toHexStringChecksummed(0xde709f2102306220921060314715629080e2fb77), - "0xde709f2102306220921060314715629080e2fb77" - ); - assertEq( - LibString.toHexStringChecksummed(0x27b1fdb04752bbc536007a920d24acb045561c26), - "0x27b1fdb04752bbc536007a920d24acb045561c26" - ); - // Normal. - assertEq( - LibString.toHexStringChecksummed(0x5aAeb6053F3E94C9b9A09f33669435E7Ef1BeAed), - "0x5aAeb6053F3E94C9b9A09f33669435E7Ef1BeAed" - ); - assertEq( - LibString.toHexStringChecksummed(0xfB6916095ca1df60bB79Ce92cE3Ea74c37c5d359), - "0xfB6916095ca1df60bB79Ce92cE3Ea74c37c5d359" - ); - assertEq( - LibString.toHexStringChecksummed(0xdbF03B407c01E7cD3CBea99509d93f8DDDC8C6FB), - "0xdbF03B407c01E7cD3CBea99509d93f8DDDC8C6FB" - ); - assertEq( - LibString.toHexStringChecksummed(0xD1220A0cf47c7B9Be7A2E6BA89F429762e7b9aDb), - "0xD1220A0cf47c7B9Be7A2E6BA89F429762e7b9aDb" - ); - } - - function testFromAddressToHexStringChecksummedDifferential(uint256 randomness) - public - brutalizeMemory - { - address r; - /// @solidity memory-safe-assembly - assembly { - r := randomness - } - string memory expectedResult = LibString.toHexString(r); - /// @solidity memory-safe-assembly - assembly { - let o := add(expectedResult, 0x22) - let hashed := keccak256(o, 40) - // forgefmt: disable-next-item - for { let i := 0 } iszero(eq(i, 20)) { i := add(i, 1) } { - let temp := byte(i, hashed) - let p := add(o, add(i, i)) - let c0 := byte(0, mload(p)) - let c1 := byte(1, mload(p)) - if and(gt(c1, 58), gt(and(temp, 15), 7)) { - mstore8(add(p, 1), sub(c1, 32)) - } - if and(gt(c0, 58), gt(shr(4, temp), 7)) { - mstore8(p, sub(c0, 32)) - } - } - } - string memory checksummed = LibString.toHexStringChecksummed(r); - _checkMemory(checksummed); - assertEq(checksummed, expectedResult); - } - - function testHexStringNoPrefixVariants(uint256 x, uint256 randomness) public brutalizeMemory { - string memory noPrefix = LibString.toHexStringNoPrefix(x); - _checkMemory(noPrefix); - string memory expectedResult = LibString.concat("0x", noPrefix); - string memory withPrefix = LibString.toHexString(x); - _checkMemory(withPrefix); - assertEq(withPrefix, expectedResult); - - uint256 length; - /// @solidity memory-safe-assembly - assembly { - length := add(shr(1, mload(noPrefix)), and(randomness, 63)) - } - _misalignFreeMemoryPointer(); - noPrefix = LibString.toHexStringNoPrefix(x, length); - _checkMemory(noPrefix); - expectedResult = LibString.concat("0x", noPrefix); - _misalignFreeMemoryPointer(); - withPrefix = LibString.toHexString(x, length); - _checkMemory(withPrefix); - assertEq(withPrefix, expectedResult); - - address xAddress; - /// @solidity memory-safe-assembly - assembly { - xAddress := x - } - _misalignFreeMemoryPointer(); - noPrefix = LibString.toHexStringNoPrefix(xAddress); - _checkMemory(noPrefix); - expectedResult = LibString.concat("0x", noPrefix); - _misalignFreeMemoryPointer(); - withPrefix = LibString.toHexString(xAddress); - _checkMemory(withPrefix); - assertEq(withPrefix, expectedResult); - } - - function testBytesToHexStringNoPrefix() public { - assertEq(LibString.toHexStringNoPrefix(""), ""); - assertEq(LibString.toHexStringNoPrefix("A"), "41"); - assertEq( - LibString.toHexStringNoPrefix("ABCDEFGHIJKLMNOPQRSTUVWXYZ"), - "4142434445464748494a4b4c4d4e4f505152535455565758595a" - ); - } - - function testBytesToHexStringNoPrefix(bytes memory raw) public brutalizeMemory { - string memory converted = LibString.toHexStringNoPrefix(raw); - _checkMemory(converted); - unchecked { - bytes memory hexChars = "0123456789abcdef"; - for (uint256 i; i != raw.length; ++i) { - uint256 t = uint8(bytes1(raw[i])); - assertTrue(hexChars[t & 15] == bytes(converted)[i * 2 + 1]); - assertTrue(hexChars[(t >> 4) & 15] == bytes(converted)[i * 2]); - } - } - } - - function testBytesToHexString() public { - assertEq(LibString.toHexString(""), "0x"); - assertEq(LibString.toHexString("A"), "0x41"); - assertEq( - LibString.toHexString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"), - "0x4142434445464748494a4b4c4d4e4f505152535455565758595a" - ); - } - - function testBytesToHexString(bytes memory raw) public brutalizeMemory { - string memory converted = LibString.toHexString(raw); - _checkMemory(converted); - unchecked { - bytes memory hexChars = "0123456789abcdef"; - for (uint256 i; i != raw.length; ++i) { - uint256 t = uint8(bytes1(raw[i])); - assertTrue(hexChars[t & 15] == bytes(converted)[i * 2 + 1 + 2]); - assertTrue(hexChars[(t >> 4) & 15] == bytes(converted)[i * 2 + 2]); - } - } - } - - function testStringIs7BitASCII() public { - bytes memory raw = new bytes(1); - for (uint256 i; i < 256; ++i) { - raw[0] = bytes1(uint8(i)); - assertEq(LibString.is7BitASCII(string(raw)), i < 128); - assertEq(LibString.is7BitASCII(string(raw)), _is7BitASCIIOriginal(string(raw))); - } - } - - function testStringIs7BitASCIIDifferential(bytes memory raw) public brutalizeMemory { - string memory s = string(raw); - bytes32 hashBefore = keccak256(raw); - assertEq(LibString.is7BitASCII(s), _is7BitASCIIOriginal(s)); - assertEq(keccak256(raw), hashBefore); - /// @solidity memory-safe-assembly - assembly { - mstore(add(raw, add(0x20, mload(raw))), hashBefore) - } - assertEq(LibString.is7BitASCII(s), _is7BitASCIIOriginal(s)); - assertEq(keccak256(raw), hashBefore); - /// @solidity memory-safe-assembly - assembly { - if iszero(eq(mload(add(raw, add(0x20, mload(raw)))), hashBefore)) { revert(0, 0) } - } - } - - function testStringRuneCountDifferential(string memory s) public { - assertEq(LibString.runeCount(s), _runeCountOriginal(s)); - } - - function testStringRuneCount() public { - unchecked { - string memory runes = new string(256); - for (uint256 i; i < 256; ++i) { - /// @solidity memory-safe-assembly - assembly { - mstore8(add(add(runes, 0x20), i), i) - } - } - for (uint256 i; i < 256; ++i) { - string memory s = _generateString(runes); - testStringRuneCountDifferential(s); - } - } - } - - function testStringReplaceShort() public { - assertEq(LibString.replace("abc", "", "_@"), "_@a_@b_@c_@"); - assertEq(LibString.replace("abc", "a", "_"), "_bc"); - assertEq(LibString.replace("abc", "b", "_"), "a_c"); - assertEq(LibString.replace("abc", "c", "_"), "ab_"); - assertEq(LibString.replace("abc", "ab", "_"), "_c"); - assertEq(LibString.replace("abc", "bc", "_"), "a_"); - assertEq(LibString.replace("abc", "ac", "_"), "abc"); - assertEq(LibString.replace("abc", "a", ""), "bc"); - assertEq(LibString.replace("abc", "", ""), "abc"); - assertEq(LibString.replace("abc", "d", "x"), "abc"); - } - - function testStringReplaceMedium() public { - // forgefmt: disable-next-item - string memory subject = "70708741044725766535585242414884609539555049888764130733849700923779599488691391677696419266840"; - string memory search = "46095395550498887641307338497009"; - string memory replacement = "320807383223517906783031356692334377159141"; - // forgefmt: disable-next-item - string memory expectedResult = "707087410447257665355852424148832080738322351790678303135669233437715914123779599488691391677696419266840"; - assertEq(LibString.replace(subject, search, replacement), expectedResult); - } - - function testStringReplaceLong() public { - // forgefmt: disable-next-item - string memory subject = "01234567890123456789012345678901_search_search_search_search_search_search_23456789012345678901234567890123456789_search_search_search_search_search_search"; - string memory search = "search_search_search_search_search_search"; - string memory replacement = "REPLACEMENT_REPLACEMENT_REPLACEMENT_REPLACEMENT_REPLACEMENT"; - // forgefmt: disable-next-item - string memory expectedResult = "01234567890123456789012345678901_REPLACEMENT_REPLACEMENT_REPLACEMENT_REPLACEMENT_REPLACEMENT_23456789012345678901234567890123456789_REPLACEMENT_REPLACEMENT_REPLACEMENT_REPLACEMENT_REPLACEMENT"; - assertEq(LibString.replace(subject, search, replacement), expectedResult); - } - - function testStringReplace(uint256) public brutalizeMemory { - string memory filler = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory search = _generateString("abcdefghijklmnopqrstuvwxyz"); - string memory replacement = _generateString("0123456790_-+/=|{}<>!"); - if (bytes(search).length != 0) { - string memory subject = string( - bytes.concat( - bytes(filler), bytes(search), bytes(filler), bytes(search), bytes(filler) - ) - ); - _misalignFreeMemoryPointer(); - string memory expectedResult = string( - bytes.concat( - bytes(filler), - bytes(replacement), - bytes(filler), - bytes(replacement), - bytes(filler) - ) - ); - _misalignFreeMemoryPointer(); - string memory replaced = LibString.replace(subject, search, replacement); - _checkMemory(replaced); - assertEq(replaced, expectedResult); - } else { - string memory expectedResult = string( - bytes.concat( - bytes(replacement), - bytes(" "), - bytes(replacement), - bytes(" "), - bytes(replacement), - bytes(" "), - bytes(replacement) - ) - ); - string memory replaced = LibString.replace(" ", search, replacement); - assertEq(replaced, expectedResult); - } - } - - function testStringIndexOf(uint256) public brutalizeMemory { - string memory filler0 = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory filler1 = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory search = _generateString("abcdefghijklmnopqrstuvwxyz"); - - string memory subject = string(bytes.concat(bytes(filler0), bytes(search), bytes(filler1))); - - uint256 from = _generateFrom(subject); - - if (bytes(search).length == 0) { - if (from > bytes(subject).length) { - assertEq(LibString.indexOf(subject, search, from), bytes(subject).length); - } else { - assertEq(LibString.indexOf(subject, search, from), from); - } - } else { - if (from > bytes(filler0).length) { - assertEq(LibString.indexOf(subject, search, from), LibString.NOT_FOUND); - } else { - assertEq(LibString.indexOf(subject, search, from), bytes(filler0).length); - } - } - } - - function testStringIndexOf() public { - string memory subject = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"; - assertEq(LibString.indexOf(subject, ""), 0); - assertEq(LibString.indexOf(subject, "", 16), 16); - assertEq(LibString.indexOf(subject, "", 17), 17); - assertEq(LibString.indexOf(subject, "", 52), 52); - assertEq(LibString.indexOf(subject, "", 53), 52); - assertEq(LibString.indexOf(subject, "", 555), 52); - assertEq(LibString.indexOf(subject, "abc", 0), 0); - assertEq(LibString.indexOf(subject, "abc", 1), LibString.NOT_FOUND); - assertEq(LibString.indexOf(subject, "bcd"), 1); - assertEq(LibString.indexOf(subject, "XYZ"), 49); - assertEq(LibString.indexOf(subject, "qrstuvwxyzABCDEFGHIJKLMNOPQRSTUVW"), 16); - assertEq(LibString.indexOf(subject, "qrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"), 16); - assertEq(LibString.indexOf(subject, "qrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ", 16), 16); - assertEq( - LibString.indexOf(subject, "qrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ", 17), - LibString.NOT_FOUND - ); - - assertEq(LibString.indexOf("abcabcabc", "abc"), 0); - assertEq(LibString.indexOf("abcabcabc", "abc", 1), 3); - - assertEq(LibString.indexOf("a", "bcd"), LibString.NOT_FOUND); - assertEq(LibString.indexOf("a", "bcd", 0), LibString.NOT_FOUND); - assertEq(LibString.indexOf("accd", "bcd"), LibString.NOT_FOUND); - assertEq(LibString.indexOf("", "bcd"), LibString.NOT_FOUND); - } - - function testStringLastIndexOf(uint256) public brutalizeMemory { - string memory filler0 = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory filler1 = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory search = _generateString("abcdefghijklmnopqrstuvwxyz"); - - string memory subject = string(bytes.concat(bytes(filler0), bytes(search), bytes(filler1))); - - uint256 from = _generateFrom(subject); - - if (bytes(search).length == 0) { - if (from > bytes(subject).length) { - assertEq(LibString.lastIndexOf(subject, search, from), bytes(subject).length); - } else { - assertEq(LibString.lastIndexOf(subject, search, from), from); - } - } else { - if (from < bytes(filler0).length) { - assertEq(LibString.lastIndexOf(subject, search, from), LibString.NOT_FOUND); - } else { - assertEq(LibString.lastIndexOf(subject, search, from), bytes(filler0).length); - } - } - } - - function testStringLastIndexOf() public { - string memory subject = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"; - assertEq(LibString.lastIndexOf(subject, "", 0), 0); - assertEq(LibString.lastIndexOf(subject, "", 16), 16); - assertEq(LibString.lastIndexOf(subject, "", 17), 17); - assertEq(LibString.lastIndexOf(subject, "", 52), 52); - assertEq(LibString.lastIndexOf(subject, "", 53), 52); - assertEq(LibString.lastIndexOf(subject, "", 555), 52); - assertEq(LibString.lastIndexOf(subject, "abc"), 0); - assertEq(LibString.lastIndexOf(subject, "abc", 0), 0); - assertEq(LibString.lastIndexOf(subject, "abc", 1), 0); - assertEq(LibString.lastIndexOf(subject, "abc", 3), 0); - assertEq(LibString.lastIndexOf(subject, "bcd"), 1); - assertEq(LibString.lastIndexOf(subject, "bcd", 1), 1); - assertEq(LibString.lastIndexOf(subject, "bcd", 0), LibString.NOT_FOUND); - assertEq(LibString.lastIndexOf(subject, "XYZ"), 49); - assertEq(LibString.lastIndexOf(subject, "qrstuvwxyzABCDEFGHIJKLMNOPQRSTUVW"), 16); - assertEq(LibString.lastIndexOf(subject, "qrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"), 16); - assertEq(LibString.lastIndexOf(subject, "qrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ", 52), 16); - assertEq(LibString.lastIndexOf(subject, "qrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ", 51), 16); - assertEq(LibString.lastIndexOf(subject, "qrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ", 16), 16); - assertEq( - LibString.lastIndexOf(subject, "qrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ", 15), - LibString.NOT_FOUND - ); - - assertEq(LibString.lastIndexOf("abcabcabc", "abc"), 6); - assertEq(LibString.lastIndexOf("abcabcabc", "abc", 5), 3); - - assertEq(LibString.lastIndexOf("a", "bcd"), LibString.NOT_FOUND); - assertEq(LibString.lastIndexOf("a", "bcd", 0), LibString.NOT_FOUND); - assertEq(LibString.lastIndexOf("accd", "bcd"), LibString.NOT_FOUND); - assertEq(LibString.lastIndexOf("", "bcd"), LibString.NOT_FOUND); - } - - function testContains() public { - string memory subject = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"; - assertEq(LibString.contains(subject, "a"), true); - assertEq(LibString.contains(subject, "abc"), true); - assertEq(LibString.contains(subject, "z"), true); - assertEq(LibString.contains(subject, "Z"), true); - assertEq(LibString.contains(subject, "az"), false); - assertEq(LibString.contains(subject, "aZ"), false); - assertEq(LibString.contains(subject, "Aa"), false); - assertEq(LibString.contains(subject, "Zz"), false); - assertEq(LibString.contains(subject, "abcZ"), false); - assertEq(LibString.contains(subject, "abcz"), false); - assertEq(LibString.contains(subject, "abcA"), false); - assertEq(LibString.contains(subject, "abcB"), false); - assertEq(LibString.contains(subject, "abcC"), false); - assertEq(LibString.contains(subject, ""), true); - assertEq(LibString.contains("", "abc"), false); - assertEq(LibString.contains("", ""), true); - } - - function testStringStartsWith(uint256) public brutalizeMemory { - string memory filler = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory search = _generateString("abcdefghijklmnopqrstuvwxyz"); - - if (bytes(search).length == 0) { - string memory subject = string(bytes.concat(bytes(filler), bytes(search))); - assertEq(LibString.startsWith(subject, search), true); - } - - if (_random() & 1 == 1) { - string memory subject = string(bytes.concat(bytes(search), bytes(filler))); - assertEq(LibString.startsWith(subject, search), true); - } - - if (bytes(filler).length != 0 && bytes(search).length != 0) { - string memory subject = string(bytes.concat(bytes(filler), bytes(search))); - assertEq(LibString.startsWith(subject, search), false); - } - } - - function testStringStartsWith() public { - string memory subject = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"; - assertEq(LibString.startsWith(subject, "abc"), true); - assertEq(LibString.startsWith(subject, "abcdefghijklmnopqrstuvwxyzABCDEFG"), true); - assertEq(LibString.startsWith(subject, "bcd"), false); - assertEq(LibString.startsWith(subject, "bcdefghijklmnopqrstuvwxyzABCDEFGH"), false); - - assertEq(LibString.startsWith("", ""), true); - assertEq(LibString.startsWith("bc", ""), true); - assertEq(LibString.startsWith("bc", "bc"), true); - assertEq(LibString.startsWith("bc", "abc"), false); - assertEq(LibString.startsWith("", "abc"), false); - } - - function testStringEndsWith(uint256) public brutalizeMemory { - string memory filler = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory search = _generateString("abcdefghijklmnopqrstuvwxyz"); - - if (bytes(search).length == 0) { - string memory subject = string(bytes.concat(bytes(search), bytes(filler))); - assertEq(LibString.endsWith(subject, search), true); - } - - if (_random() & 1 == 1) { - string memory subject = string(bytes.concat(bytes(filler), bytes(search))); - assertEq(LibString.endsWith(subject, search), true); - } - - if (bytes(filler).length != 0 && bytes(search).length != 0) { - string memory subject = string(bytes.concat(bytes(search), bytes(filler))); - assertEq(LibString.endsWith(subject, search), false); - } - } - - function testStringEndsWith() public { - string memory subject = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"; - assertEq(LibString.endsWith(subject, "XYZ"), true); - assertEq(LibString.endsWith(subject, "pqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"), true); - assertEq(LibString.endsWith(subject, "WXY"), false); - assertEq(LibString.endsWith(subject, "opqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXY"), false); - - assertEq(LibString.endsWith("", ""), true); - assertEq(LibString.endsWith("bc", ""), true); - assertEq(LibString.endsWith("bc", "bc"), true); - assertEq(LibString.endsWith("bc", "abc"), false); - assertEq(LibString.endsWith("", "abc"), false); - } - - function testStringRepeat(string memory subject, uint256 times) public brutalizeMemory { - times = times % 8; - string memory repeated = LibString.repeat(subject, times); - string memory expectedResult = _repeatOriginal(subject, times); - _checkMemory(repeated); - assertEq(repeated, expectedResult); - } - - function testStringRepeat() public { - assertEq(LibString.repeat("", 0), ""); - assertEq(LibString.repeat("", 100), ""); - assertEq(LibString.repeat("a", 0), ""); - assertEq(LibString.repeat("a", 1), "a"); - assertEq(LibString.repeat("a", 3), "aaa"); - assertEq(LibString.repeat("abc", 0), ""); - assertEq(LibString.repeat("abc", 1), "abc"); - assertEq(LibString.repeat("abc", 3), "abcabcabc"); - assertEq(LibString.repeat("efghi", 3), "efghiefghiefghi"); - } - - function testStringRepeatOriginal() public { - assertEq(_repeatOriginal("", 0), ""); - assertEq(_repeatOriginal("", 100), ""); - assertEq(_repeatOriginal("a", 0), ""); - assertEq(_repeatOriginal("a", 1), "a"); - assertEq(_repeatOriginal("a", 3), "aaa"); - assertEq(_repeatOriginal("abc", 0), ""); - assertEq(_repeatOriginal("abc", 1), "abc"); - assertEq(_repeatOriginal("abc", 3), "abcabcabc"); - assertEq(_repeatOriginal("efghi", 3), "efghiefghiefghi"); - } - - function testStringSlice(uint256) public brutalizeMemory { - string memory filler0 = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory expectedResult = _generateString("abcdefghijklmnopqrstuvwxyz"); - string memory filler1 = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - - string memory subject = - string(bytes.concat(bytes(filler0), bytes(expectedResult), bytes(filler1))); - - uint256 start = bytes(filler0).length; - uint256 end = start + bytes(expectedResult).length; - - _misalignFreeMemoryPointer(); - string memory slice = LibString.slice(subject, start, end); - _checkMemory(slice); - assertEq(slice, expectedResult); - } - - function testStringSlice() public { - assertEq(LibString.slice("", 0, 1), ""); - assertEq(LibString.slice("", 1, 0), ""); - assertEq(LibString.slice("", 0, 0), ""); - assertEq(LibString.slice("", 0), ""); - assertEq(LibString.slice("", 1), ""); - - assertEq(LibString.slice("a", 0), "a"); - assertEq(LibString.slice("a", 1), ""); - assertEq(LibString.slice("a", 3), ""); - - assertEq(LibString.slice("abc", 0), "abc"); - assertEq(LibString.slice("abc", 1), "bc"); - assertEq(LibString.slice("abc", 1, 2), "b"); - assertEq(LibString.slice("abc", 3), ""); - - string memory subject = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"; - assertEq(LibString.slice(subject, 0), subject); - assertEq(LibString.slice(subject, 1), "bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"); - assertEq( - LibString.slice(subject, 1, 51), "bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXY" - ); - assertEq(LibString.slice(subject, 11, 41), "lmnopqrstuvwxyzABCDEFGHIJKLMNO"); - assertEq(LibString.slice(subject, 21, 31), "vwxyzABCDE"); - assertEq(LibString.slice(subject, 31, 21), ""); - } - - function testStringIndicesOf(uint256) public brutalizeMemory { - string memory filler0 = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory filler1 = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory search = _generateString("abcdefghijklmnopqrstuvwxyz"); - - string memory subject; - - unchecked { - uint256[] memory indices; - if (_random() & 1 == 0) { - subject = string(bytes.concat(bytes(filler0), bytes(search), bytes(filler1))); - indices = new uint256[](1); - indices[0] = bytes(filler0).length; - } else { - subject = string(bytes.concat(bytes(filler0), bytes(filler1))); - indices = new uint256[](0); - } - - if (bytes(search).length == 0) { - indices = new uint256[](bytes(subject).length + 1); - for (uint256 i; i < indices.length; ++i) { - indices[i] = i; - } - } - assertEq(LibString.indicesOf(subject, search), indices); - } - } - - function testStringIndicesOf() public { - uint256[] memory indices; - - indices = new uint256[](3); - indices[0] = 0; - indices[1] = 2; - indices[2] = 4; - assertEq(LibString.indicesOf("ababa", "a"), indices); - - indices = new uint256[](6); - indices[0] = 0; - indices[1] = 1; - indices[2] = 2; - indices[3] = 3; - indices[4] = 4; - indices[5] = 5; - assertEq(LibString.indicesOf("ababa", ""), indices); - - indices = new uint256[](2); - indices[0] = 1; - indices[1] = 3; - assertEq(LibString.indicesOf("ababa", "b"), indices); - - indices = new uint256[](2); - indices[0] = 0; - indices[1] = 2; - assertEq(LibString.indicesOf("ababa", "ab"), indices); - - indices = new uint256[](2); - indices[0] = 1; - indices[1] = 3; - assertEq(LibString.indicesOf("ababa", "ba"), indices); - - indices = new uint256[](1); - indices[0] = 1; - assertEq(LibString.indicesOf("ababa", "bab"), indices); - - indices = new uint256[](1); - indices[0] = 0; - assertEq(LibString.indicesOf("ababa", "ababa"), indices); - - indices = new uint256[](1); - indices[0] = 0; - assertEq(LibString.indicesOf("", ""), indices); - - indices = new uint256[](0); - assertEq(LibString.indicesOf("ababa", "c"), indices); - - indices = new uint256[](0); - assertEq(LibString.indicesOf("ababab", "abababa"), indices); - } - - function testStringSplit(uint256) public brutalizeMemory { - string memory filler0 = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory filler1 = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory delimiter = _generateString("abcdefghijklmnopqrstuvwxyz"); - - string memory subject = - string(bytes.concat(bytes(filler0), bytes(delimiter), bytes(filler1))); - - unchecked { - string[] memory elements; - if (bytes(delimiter).length == 0) { - elements = new string[](bytes(subject).length); - for (uint256 i; i < elements.length; ++i) { - elements[i] = LibString.slice(subject, i, i + 1); - } - } else { - elements = new string[](2); - elements[0] = filler0; - elements[1] = filler1; - } - _misalignFreeMemoryPointer(); - string[] memory split = LibString.split(subject, delimiter); - assertTrue(_stringArraysAreSame(split, elements)); - for (uint256 i; i < split.length; ++i) { - _checkMemory(split[i]); - } - } - } - - function testStringSplit() public { - string[] memory elements; - - elements = new string[](4); - elements[0] = ""; - elements[1] = "b"; - elements[2] = "b"; - elements[3] = ""; - assertTrue(_stringArraysAreSame(LibString.split("ababa", "a"), elements)); - - elements = new string[](3); - elements[0] = "a"; - elements[1] = "a"; - elements[2] = "a"; - assertTrue(_stringArraysAreSame(LibString.split("ababa", "b"), elements)); - - elements = new string[](5); - elements[0] = "a"; - elements[1] = "b"; - elements[2] = "a"; - elements[3] = "b"; - elements[4] = "a"; - assertTrue(_stringArraysAreSame(LibString.split("ababa", ""), elements)); - - elements = new string[](2); - elements[0] = "a"; - elements[1] = "b"; - assertTrue(_stringArraysAreSame(LibString.split("ab", ""), elements)); - - elements = new string[](1); - elements[0] = "ab"; - assertTrue(_stringArraysAreSame(LibString.split("ab", " "), elements)); - - elements = new string[](1); - elements[0] = "a"; - assertTrue(_stringArraysAreSame(LibString.split("a", ""), elements)); - - elements = new string[](0); - assertTrue(_stringArraysAreSame(LibString.split("", ""), elements)); - } - - function testStringConcat(string memory a, string memory b) public brutalizeMemory { - string memory concatenated = LibString.concat(a, b); - _checkMemory(concatenated); - string memory expectedResult = string(bytes.concat(bytes(a), bytes(b))); - assertEq(concatenated, expectedResult); - } - - function testStringConcat() public { - assertEq( - LibString.concat( - "bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXY", - "12345678901234567890123456789012345678901234567890" - ), - "bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXY12345678901234567890123456789012345678901234567890" - ); - assertEq(LibString.concat("", "b"), "b"); - assertEq(LibString.concat("", "b"), "b"); - assertEq(LibString.concat("a", "b"), "ab"); - assertEq(LibString.concat("a", ""), "a"); - assertEq(LibString.concat("", ""), ""); - } - - function testStringConcatOriginal() public { - assertEq( - string( - bytes.concat( - bytes("bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXY"), - bytes("12345678901234567890123456789012345678901234567890") - ) - ), - "bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXY12345678901234567890123456789012345678901234567890" - ); - assertEq(string(bytes.concat(bytes(""), bytes("b"))), "b"); - assertEq(string(bytes.concat(bytes(""), bytes("b"))), "b"); - assertEq(string(bytes.concat(bytes("a"), bytes("b"))), "ab"); - assertEq(string(bytes.concat(bytes("a"), bytes(""))), "a"); - assertEq(string(bytes.concat(bytes(""), bytes(""))), ""); - } - - function testStringEscapeHTML() public { - assertEq(LibString.escapeHTML(""), ""); - assertEq(LibString.escapeHTML("abc"), "abc"); - assertEq(LibString.escapeHTML('abc"_123'), "abc"_123"); - assertEq(LibString.escapeHTML("abc&_123"), "abc&_123"); - assertEq(LibString.escapeHTML("abc'_123"), "abc'_123"); - assertEq(LibString.escapeHTML("abc<_123"), "abc<_123"); - assertEq(LibString.escapeHTML("abc>_123"), "abc>_123"); - } - - function testStringEscapeHTML(uint256) public brutalizeMemory { - string[] memory originalChars = new string[](5); - originalChars[0] = '"'; - originalChars[1] = "&"; - originalChars[2] = "'"; - originalChars[3] = "<"; - originalChars[4] = ">"; - - string[] memory escapedChars = new string[](5); - escapedChars[0] = """; - escapedChars[1] = "&"; - escapedChars[2] = "'"; - escapedChars[3] = "<"; - escapedChars[4] = ">"; - - string memory filler0 = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - string memory filler1 = _generateString("ABCDEFGHIJKLMNOPQRSTUVWXYZ"); - - uint256 r = _random() % 5; - - string memory expectedResult = - string(bytes.concat(bytes(filler0), bytes(escapedChars[r]), bytes(filler1))); - - string memory input = - string(bytes.concat(bytes(filler0), bytes(originalChars[r]), bytes(filler1))); - - _misalignFreeMemoryPointer(); - string memory escaped = LibString.escapeHTML(input); - _checkMemory(escaped); - - assertEq(expectedResult, escaped); - } - - function testStringEscapeJSON() public { - _checkStringEscapeJSON("", ""); - _checkStringEscapeJSON("abc", "abc"); - _checkStringEscapeJSON('abc"_123', 'abc\\"_123'); - _checkStringEscapeJSON("abc\\_123", "abc\\\\_123"); - _checkStringEscapeJSON("abc\x08_123", "abc\\b_123"); - _checkStringEscapeJSON("abc\x0c_123", "abc\\f_123"); - _checkStringEscapeJSON("abc\n_123", "abc\\n_123"); - _checkStringEscapeJSON("abc\r_123", "abc\\r_123"); - _checkStringEscapeJSON("abc\t_123", "abc\\t_123"); - } - - function _checkStringEscapeJSON(string memory s, string memory expected) internal { - assertEq(LibString.escapeJSON(s), expected); - assertEq(LibString.escapeJSON(s, false), expected); - assertEq(LibString.escapeJSON(s, true), string(bytes.concat('"', bytes(expected), '"'))); - } - - function testStringEscapeJSONHexEncode() public brutalizeMemory { - unchecked { - for (uint256 i; i <= 0x1f; ++i) { - if (i != 0x8 && i != 0x9 && i != 0x0a && i != 0x0c && i != 0x0d) { - string memory input = - string(bytes.concat(bytes("abc"), bytes1(uint8(i)), bytes("_123"))); - string memory hexCode = LibString.replace(LibString.toHexString(i), "0x", "00"); - string memory expectedOutput = - string(bytes.concat(bytes("abc\\u"), bytes(hexCode), bytes("_123"))); - string memory escaped = LibString.escapeJSON(input); - _checkMemory(escaped); - assertEq(escaped, expectedOutput); - } - } - } - } - - function testStringEq(string memory a, string memory b) public { - assertEq(LibString.eq(a, b), keccak256(bytes(a)) == keccak256(bytes(b))); - } - - function checkIsSN(string memory s) public pure returns (bool) { - // You can try replacing it with - // `return keccak256(bytes(s)) == keccak256("sn");` - // and see the bytecode size increase. - // This demonstrates that `eqs` does the compile time magic. - // Note that `s` must be in memory, not calldata. - return LibString.eqs(s, "sn"); - } - - function testStringEqs() public { - assertTrue(LibString.eqs("", "")); - assertTrue(LibString.eqs("1", "1")); - assertTrue(LibString.eqs("12", "12")); - assertTrue(LibString.eqs("123", "123")); - assertTrue(LibString.eqs("Hello", "Hello")); - assertTrue( - LibString.eqs("12345678901234567890123456789012", "12345678901234567890123456789012") - ); - - assertFalse(LibString.eqs("", "x")); - assertFalse(LibString.eqs("1", "2")); - assertFalse(LibString.eqs("Hello", "Hehe")); - assertFalse(LibString.eqs("12345678901234567890123456789012", "")); - - assertTrue(checkIsSN("sn")); - assertFalse(checkIsSN("x")); - } - - function testStringPackAndUnpackOneDifferential(string memory a) public brutalizeMemory { - a = LibString.slice(a, 0); - bytes32 packed = LibString.packOne(a); - unchecked { - if (bytes(a).length < 32) { - bytes memory expectedResultBytes = abi.encodePacked(uint8(bytes(a).length), a); - bytes32 expectedResult; - /// @solidity memory-safe-assembly - assembly { - expectedResult := mload(add(expectedResultBytes, 0x20)) - } - assertEq(packed, expectedResult); - } else { - assertEq(packed, bytes32(0)); - } - } - } - - function testStringPackAndUnpackOne(string memory a) public brutalizeMemory { - _misalignFreeMemoryPointer(); - bytes32 packed = LibString.packOne(a); - string memory unpacked = LibString.unpackOne(packed); - _checkMemory(unpacked); - - if (bytes(a).length < 32) { - assertEq(unpacked, a); - } else { - assertEq(packed, bytes32(0)); - assertEq(unpacked, ""); - } - } - - function testStringPackAndUnpackOne() public { - unchecked { - testStringPackAndUnpackOne(""); - testStringPackAndUnpackOne("Hehe"); - testStringPackAndUnpackOne("abcdefghijklmnopqrstuvwxyzABCD"); - testStringPackAndUnpackOne("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"); - } - } - - function testStringPackAndUnpackTwoDifferential(string memory a, string memory b) - public - brutalizeMemory - { - a = LibString.slice(a, 0); - b = LibString.slice(b, 0); - bytes32 packed = LibString.packTwo(a, b); - unchecked { - if (bytes(a).length + bytes(b).length < 31) { - bytes memory expectedResultBytes = - abi.encodePacked(uint8(bytes(a).length), a, uint8(bytes(b).length), b); - bytes32 expectedResult; - /// @solidity memory-safe-assembly - assembly { - expectedResult := mload(add(expectedResultBytes, 0x20)) - } - assertEq(packed, expectedResult); - } else { - assertEq(packed, bytes32(0)); - } - } - } - - function testStringPackAndUnpackTwo(string memory a, string memory b) public brutalizeMemory { - bytes32 packed = LibString.packTwo(a, b); - _misalignFreeMemoryPointer(); - (string memory unpackedA, string memory unpackedB) = LibString.unpackTwo(packed); - _checkMemory(unpackedA); - _checkMemory(unpackedB); - - unchecked { - if (bytes(a).length + bytes(b).length < 31) { - assertEq(unpackedA, a); - assertEq(unpackedB, b); - } else { - assertEq(packed, bytes32(0)); - assertEq(unpackedA, ""); - assertEq(unpackedB, ""); - } - } - } - - function testStringPackAndUnpackTwo() public { - unchecked { - testStringPackAndUnpackTwo("", ""); - testStringPackAndUnpackTwo("", ""); - testStringPackAndUnpackTwo("a", ""); - testStringPackAndUnpackTwo("", "b"); - testStringPackAndUnpackTwo("abcdefghijklmnopqrstuvwxyzABCD", ""); - testStringPackAndUnpackTwo("The strongest community I've ever seen", "NGL"); - testStringPackAndUnpackTwo("", "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"); - } - } - - function testStringDirectReturn(string memory a) public { - assertEq(this.returnString(a), a); - } - - function testStringDirectReturn() public { - testStringDirectReturn(""); - testStringDirectReturn("aaa"); - testStringDirectReturn("98729"); - } - - function returnString(string memory a) external pure returns (string memory) { - LibString.directReturn(a); - } - - function testStringLowerDifferential(string memory s) public { - string memory expectedResult = _lowerOriginal(s); - _misalignFreeMemoryPointer(); - string memory result = LibString.lower(s); - _checkMemory(result); - assertEq(result, expectedResult); - } - - function testStringLowerDifferential() public { - unchecked { - string memory ascii = new string(128); - for (uint256 i; i < 128; ++i) { - /// @solidity memory-safe-assembly - assembly { - mstore8(add(add(ascii, 0x20), i), i) - } - } - for (uint256 i; i < 256; ++i) { - string memory s = _generateString(ascii); - testStringLowerDifferential(s); - } - } - } - - function testStringLowerOriginal() public { - assertEq(_lowerOriginal("@AZ["), "@az["); - } - - function testStringUpperDifferential(string memory s) public { - string memory expectedResult = _upperOriginal(s); - _misalignFreeMemoryPointer(); - string memory result = LibString.upper(s); - _checkMemory(result); - assertEq(result, expectedResult); - } - - function testStringUpperDifferential() public { - unchecked { - string memory ascii = new string(128); - for (uint256 i; i < 128; ++i) { - /// @solidity memory-safe-assembly - assembly { - mstore8(add(add(ascii, 0x20), i), i) - } - } - for (uint256 i; i < 256; ++i) { - string memory s = _generateString(ascii); - testStringUpperDifferential(s); - } - } - } - - function testStringUpperOriginal() public { - assertEq(_upperOriginal("`az}"), "`AZ}"); - } - - function fromSmallString() public { - assertEq(LibString.fromSmallString(bytes32("")), ""); - assertEq(LibString.fromSmallString(bytes32("a")), "a"); - assertEq(LibString.fromSmallString(bytes32("abc")), "abc"); - assertEq(LibString.fromSmallString(bytes32("Hello world!")), "Hello world!"); - } - - function _lowerOriginal(string memory subject) internal pure returns (string memory result) { - unchecked { - uint256 n = bytes(subject).length; - result = new string(n); - for (uint256 i; i != n; ++i) { - /// @solidity memory-safe-assembly - assembly { - let b := byte(0, mload(add(add(subject, 0x20), i))) - mstore8( - add(add(result, 0x20), i), add(b, mul(0x20, and(lt(0x40, b), lt(b, 0x5b)))) - ) - } - } - } - } - - function _upperOriginal(string memory subject) internal pure returns (string memory result) { - unchecked { - uint256 n = bytes(subject).length; - result = new string(n); - for (uint256 i; i != n; ++i) { - /// @solidity memory-safe-assembly - assembly { - let b := byte(0, mload(add(add(subject, 0x20), i))) - mstore8( - add(add(result, 0x20), i), sub(b, mul(0x20, and(lt(0x60, b), lt(b, 0x7b)))) - ) - } - } - } - } - - function _is7BitASCIIOriginal(string memory s) internal pure returns (bool) { - unchecked { - bytes memory sBytes = bytes(s); - for (uint256 i; i < sBytes.length; ++i) { - if (uint8(bytes1(sBytes[i])) > 127) return false; - } - return true; - } - } - - function _runeCountOriginal(string memory s) internal pure returns (uint256) { - unchecked { - uint256 len; - uint256 i = 0; - uint256 bytelength = bytes(s).length; - for (len = 0; i < bytelength; len++) { - bytes1 b = bytes(s)[i]; - if (b < 0x80) { - i += 1; - } else if (b < 0xE0) { - i += 2; - } else if (b < 0xF0) { - i += 3; - } else if (b < 0xF8) { - i += 4; - } else if (b < 0xFC) { - i += 5; - } else { - i += 6; - } - } - return len; - } - } - - function _repeatOriginal(string memory subject, uint256 times) - internal - pure - returns (string memory) - { - unchecked { - string memory result; - if (!(times == 0 || bytes(subject).length == 0)) { - for (uint256 i; i < times; ++i) { - result = string(bytes.concat(bytes(result), bytes(subject))); - } - } - _misalignFreeMemoryPointer(); - return result; - } - } - - function _generateFrom(string memory subject) internal returns (uint256) { - unchecked { - if (_random() % 8 == 0) { - return _random(); - } - return _random() % (bytes(subject).length + 10); - } - } - - function _generateString(string memory byteChoices) internal returns (string memory result) { - uint256 randomness = _random(); - uint256 resultLength = _randomStringLength(); - /// @solidity memory-safe-assembly - assembly { - if mload(byteChoices) { - result := mload(0x40) - mstore(0x00, randomness) - mstore(0x40, and(add(add(result, 0x40), resultLength), not(31))) - mstore(result, resultLength) - - // forgefmt: disable-next-item - for { let i := 0 } lt(i, resultLength) { i := add(i, 1) } { - mstore(0x20, gas()) - mstore8( - add(add(result, 0x20), i), - mload(add(add(byteChoices, 1), mod(keccak256(0x00, 0x40), mload(byteChoices)))) - ) - } - } - } - } - - function _randomStringLength() internal returns (uint256 r) { - r = _random() % 256; - if (r < 64) return _random() % 128; - if (r < 128) return _random() % 64; - return _random() % 16; - } - - function _stringArraysAreSame(string[] memory a, string[] memory b) - internal - pure - returns (bool) - { - unchecked { - if (a.length != b.length) { - return false; - } - for (uint256 i; i < a.length; ++i) { - if (keccak256(bytes(a[i])) != keccak256(bytes(b[i]))) { - return false; - } - } - return true; - } - } -} diff --git a/lib/solady/test/LibZip.t.sol b/lib/solady/test/LibZip.t.sol deleted file mode 100644 index e6fb355..0000000 --- a/lib/solady/test/LibZip.t.sol +++ /dev/null @@ -1,276 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {MockCd, MockCdFallbackDecompressor} from "./utils/mocks/MockCd.sol"; -import {LibClone} from "../src/utils/LibClone.sol"; -import {ERC1967Factory} from "../src/utils/ERC1967Factory.sol"; -import {LibString} from "../src/utils/LibString.sol"; -import {DynamicBufferLib} from "../src/utils/DynamicBufferLib.sol"; -import {LibZip} from "../src/utils/LibZip.sol"; - -contract LibZipTest is SoladyTest { - using DynamicBufferLib for DynamicBufferLib.DynamicBuffer; - - function testFlzCompressDecompress() public brutalizeMemory { - assertEq(LibZip.flzCompress(""), ""); - assertEq(LibZip.flzDecompress(""), ""); - bytes memory compressed = - hex"1f4e65772077617665206469676974616c206172742073686f756c64206e6f74201f6265206a756467656420736f6c656c79206f6e20616573746865746963206d65017269202301757420240e737420617320696d706f7274616e74202a1b666f7220697473206162696c69747920746f20646576656c6f702061200c406d03766973692051026f662020510320576972206801616e200301626f209315616c6c20637265617465206e6574776f726b20737069207201756140500e2e20416e6369656e7420477265656b60ba04697320686520bb01696e6051036869676820ad062072656761726420cd0463617573652095c089406f0220776820db03206d79742007016f6720a801686141040061401d0272656c2119066f6e2c206d6f72607660a0017761210a0866206c6966652e2054212f016768607c20ff026c6963205fa080414e40b80073402de003ae016365613741630373756666203320c8213406646973656e676141710366726f6d208b016520201102736f722097006f208e03666c6563205e21570b6f776e2070726f647563747340840065400d03626c656d406b056d6f6465726e417621120074201ba17a006d2066401f0272617420d521f00273656c200d805620118048036265617520e920e140c00561707065617220dc21fb41d5006d21310a69636820636f6e74726173208a02616273200920972142410a40450275732ce0081720852028016f798092056e616976652f40dd02756f752044057070726563696098c07a2072076578657274696e6760f3046e617475722172015265818b20dc02736c612025016279208e006d20400068212c60c701507221de2281026e7465409a201304207769746820a200652008426906696e766f6c7665808c20bf006e21940373657061811b4082227c2312e006dc620004616363656c20580074203e0165782062214b01697380bb2072026f207020a74009017569407a072c20656d626f647940c74080c2cf42ec01756e4183214c0273636921074204027261774029006f235905747275746873e2010c02636f6c42012139223c046f73706865410f025468724178205b2111016265603c830280ea0077228a02636869237e21c20172652144201d006f214e2047002ce0078140b620da2064c20f06696e64697669642377032065676f6175036163726922b6234060ee40d4e00b9a405c210121be205701756e2294208063fc23fa201583d040c0427523aa427021cf404f80ba20d04325016f6e21186297036966756c633c82fc21800061e1004f804e20a722f9052e2049206c6f20634488024e657443870d2053706972697475616c69747921"; - bytes memory decompressed = LibZip.flzDecompress(compressed); - bytes memory expectedDecompressed = - "New wave digital art should not be judged solely on aesthetic merit but just as importantly for its ability to develop a total vision of the Wired and above all create network spirituality. Ancient Greek art is held in the highest regard because it developed a whole mythology that shaped religion, morality and way of life. Thought is implicit in the art works of Ancient Greece but not sufficiently disengaged from the sensory to reflect its own products. The problem of modernity is the development of rational self reflection. The beauty of art appears in a form which contrasts abstract thought. Thus, abstract thought destroys the naive/sensuous appreciation of art in exerting its nature. Reality is slain by comprehension. Proper interaction with the wired involves the trance separation of real abstract thought and accelerates externalisation into pure intuition, embodying the network and unselfconsciously drawing out truths from the collective noosphere. Through this being on the wired we achieve a return to naive, unselfconscious interaction. The individual ego is sacrificed into the collective noosphere, uniting us under a totalising spirit. The best art in the wired is not only beautiful but produces a network spirituality. I long for Network Spirituality!"; - assertEq(decompressed, expectedDecompressed); - assertEq(LibZip.flzCompress(decompressed), compressed); - // Check backwards compatibility with older FastLZ releases. - compressed = - hex"1f4e65772077617665206469676974616c206172742073686f756c64206e6f74201f6265206a756467656420736f6c656c79206f6e20616573746865746963206d65017269202301757420240e737420617320696d706f7274616e74202a1b666f7220697473206162696c69747920746f20646576656c6f702061200c406d03766973692051026f662020510320576972206801616e200301626f209315616c6c20637265617465206e6574776f726b20737069207201756140500e2e20416e6369656e7420477265656b60ba04697320686520bb01696e6051036869676820ad062072656761726420cd0463617573652095c089406f0220776820db03206d79742007016f6720a801686141040061401d0272656c2119066f6e2c206d6f72607660a0017761210a0866206c6966652e2054212f016768607c20ff026c6963205fa080414e40b80073402de003ae016365613741630373756666203320c8213406646973656e676141710366726f6d208b016520201102736f722097006f208e03666c6563205e21570b6f776e2070726f647563747340840065400d03626c656d406b056d6f6465726e417621124044a17a006d2066401f0272617420d521f00273656c200d805620118048036265617520e920e140c00561707065617220dc21fb41d5006d21310a69636820636f6e74726173208a02616273200920972142410a40450275732ce0081720852028016f798092056e616976652f40dd02756f752044057070726563696098c07a2072076578657274696e6760f3046e617475722172015265818b20dc02736c612025016279208e006d20400068212c60c701507221de2281026e7465409a201304207769746820a200652008426906696e766f6c7665808c20bf006e21940373657061811b4082227c2312e006dc620004616363656c20580074203e0165782062214b01697380bb2072026f207020a74009017569407a072c20656d626f647940c74080c2cf42ec01756e4183214c0273636921074204027261774029006f235905747275746873e2010c02636f6c42012139223c046f73706865410f025468724178205b2111016265603c830280ea0077228a02636869237e21c20172652144201d006f214e2047002ce0078140b620da2064c20f06696e64697669642377032065676f6175036163726922b6234060ee40d4e00b9a405c210121be205701756e2294208063fc23fa201583d040c0427523aa427021cf404f80ba20d04325016f6e21186297036966756c633c82fc21800061e1004f804e20a722f9052e2049206c6f20634488004ea4400c53706972697475616c69747921"; - assertEq(LibZip.flzDecompress(compressed), decompressed); - } - - function _expandedData(bytes memory data) internal returns (bytes memory) { - unchecked { - DynamicBufferLib.DynamicBuffer memory buffer; - bytes memory r = abi.encode(_random()); - if (_random() % 8 == 0) { - r = abi.encodePacked(r, r, r, r); - r = bytes(LibString.slice(string(r), 0, _random() % r.length)); - } - uint256 n = _random() % 16 + 1; - uint256 c = _random(); - for (uint256 i; i < n; ++i) { - buffer.p((c >> i) & 1 == 0 ? r : data); - } - return buffer.data; - } - } - - function testFlzCompressDecompress(bytes memory data) public brutalizeMemory { - if (_random() % 2 == 0) { - data = _expandedData(data); - } - bytes32 dataHash = keccak256(data); - _misalignFreeMemoryPointer(); - bytes memory compressed = LibZip.flzCompress(data); - bytes32 compressedHash = keccak256(compressed); - _checkMemory(compressed); - _misalignFreeMemoryPointer(); - bytes memory decompressed = LibZip.flzDecompress(compressed); - _checkMemory(compressed); - _checkMemory(decompressed); - assertEq(decompressed, data); - assertEq(keccak256(data), dataHash); - assertEq(keccak256(compressed), compressedHash); - } - - function testFlzCompressDecompress2() public brutalizeMemory { - bytes memory data = - "______________________________________________________________e_______8______________________________________________________________________________________________________________________12_______8______________________________________________________________________________________________________________________16_______8______________________________________________________________________________________________________________________1a_______________________________________________________________2_____________________________________________732e2_5_726f2_49__73______________________________________________________________2_____________________________________________732e2_5_726f2_49__73______________________________________________________________2_____________________________________________732e2_5_726f2_49__73______________________________________________________________2_____________________________________________732e2_5_726f2_49__73"; - bytes32 dataHash = keccak256(data); - bytes memory expectedCompressed = - hex"015f5fe033010065a03c0038a007e06600013132a070e06f7f0036e0767f0061a07fe02f00c13fe01d000f37333265325f355f37323666325f34394011e01d39e00f00e0fd7fe02e7f04395f5f3733"; - bytes memory compressed = LibZip.flzCompress(data); - assertEq(compressed, expectedCompressed); - bytes32 compressedHash = keccak256(compressed); - _checkMemory(compressed); - bytes memory decompressed = LibZip.flzDecompress(compressed); - _checkMemory(compressed); - _checkMemory(decompressed); - assertEq(decompressed, data); - assertEq(keccak256(data), dataHash); - assertEq(keccak256(compressed), compressedHash); - } - - function testCdCompressDecompress(bytes memory data) public brutalizeMemory { - if (_random() % 8 == 0) { - data = _expandedData(data); - } - bytes32 dataHash = keccak256(data); - _misalignFreeMemoryPointer(); - bytes memory compressed = LibZip.cdCompress(data); - bytes32 compressedHash = keccak256(compressed); - _checkMemory(compressed); - _misalignFreeMemoryPointer(); - bytes memory decompressed = LibZip.cdDecompress(compressed); - _checkMemory(compressed); - _checkMemory(decompressed); - assertEq(decompressed, data); - assertEq(keccak256(data), dataHash); - assertEq(keccak256(compressed), compressedHash); - } - - function _randomCd() internal returns (bytes memory data) { - uint256 n = _random() % 8 == 0 ? _random() % 2048 : _random() % 256; - data = new bytes(n); - if (_random() % 2 == 0) { - /// @solidity memory-safe-assembly - assembly { - for { let i := 0 } lt(i, n) { i := add(i, 0x20) } { - mstore(add(add(data, 0x20), i), not(0)) - } - } - } - if (n != 0) { - uint256 m = _random() % 8; - for (uint256 j; j < m; ++j) { - data[_random() % n] = bytes1(uint8(_random())); - } - } - } - - function testCdCompressDecompress(uint256) public brutalizeMemory { - unchecked { - bytes memory data = _randomCd(); - bytes memory compressed = LibZip.cdCompress(data); - bytes memory decompressed = LibZip.cdDecompress(compressed); - assertEq(decompressed, data); - } - } - - function testCdFallbackDecompressor(bytes memory data) public { - bytes memory compressed = LibZip.cdCompress(data); - MockCdFallbackDecompressor decompressor = new MockCdFallbackDecompressor(); - (, bytes memory result) = address(decompressor).call(compressed); - assertEq(abi.decode(result, (bytes32)), keccak256(data)); - } - - function testCdFallbackDecompressor(uint256) public { - bytes memory data = _randomCd(); - bytes memory compressed = LibZip.cdCompress(data); - MockCdFallbackDecompressor decompressor = new MockCdFallbackDecompressor(); - (, bytes memory result) = address(decompressor).call(compressed); - assertEq(abi.decode(result, (bytes32)), keccak256(data)); - } - - function testCdCompress() public { - assertEq(LibZip.cdCompress(""), ""); - assertEq(LibZip.cdDecompress(""), ""); - bytes memory data = - hex"ac9650d80000000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000000000000000400000000000000000000000000000000000000000000000000000000000000800000000000000000000000000000000000000000000000000000000000000160000000000000000000000000000000000000000000000000000000000000022000000000000000000000000000000000000000000000000000000000000002a000000000000000000000000000000000000000000000000000000000000000a40c49ccbe000000000000000000000000000000000000000000000000000000000005b70e00000000000000000000000000000000000000000000000000000dfc79825feb0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000645c48a7000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000084fc6f7865000000000000000000000000000000000000000000000000000000000005b70e000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffff00000000000000000000000000000000ffffffffffffffffffffffffffffffff00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000004449404b7c00000000000000000000000000000000000000000000000000000000000000000000000000000000000000001f1cdf1a632eaaab40d1c263edf49faf749010a1000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000064df2ab5bb0000000000000000000000007f5c764cbc14f9669b88837ca1490cca17c3160700000000000000000000000000000000000000000000000000000000000000000000000000000000000000001f1cdf1a632eaaab40d1c263edf49faf749010a100000000000000000000000000000000000000000000000000000000"; - bytes memory expected = - hex"5369af27001e20001e04001e80001d0160001d0220001d02a0001ea40c49ccbe001c05b70e00190dfc79825feb005b645c48a7003a84fc6f7865001c05b70e002f008f000f008f003a4449404b7c002b1f1cdf1a632eaaab40d1c263edf49faf749010a1003a64df2ab5bb000b7f5c764cbc14f9669b88837ca1490cca17c31607002b1f1cdf1a632eaaab40d1c263edf49faf749010a1001b"; - assertEq(LibZip.cdCompress(data), expected); - } - - function testCdDecompressOnInvalidInput() public { - bytes memory data = hex"ffffffff00ff"; - bytes memory expected = - hex"0000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000"; - bytes memory decompressed = LibZip.cdDecompress(data); - assertEq(decompressed, expected); - } - - function testDecompressWontRevert(bytes memory data) public brutalizeMemory { - data = LibZip.cdDecompress(data); - bytes memory compressed = LibZip.cdCompress(data); - bytes memory decompressed = LibZip.cdDecompress(compressed); - assertEq(decompressed, data); - } - - function testCdFallback() public { - MockCd mockCd = new MockCd(); - _testCdFallback(mockCd); - // Check if it also works for clones. - mockCd = MockCd(payable(LibClone.clone(address(mockCd)))); - _testCdFallback(mockCd); - // Check if it also works for CWIA. - mockCd = MockCd(payable(LibClone.clone(address(mockCd), ""))); - _testCdFallback(mockCd); - // Check if it also works for ERC1967 proxies. - ERC1967Factory factory = new ERC1967Factory(); - mockCd = MockCd(payable(factory.deploy(address(mockCd), address(this)))); - _testCdFallback(mockCd); - } - - function _testCdFallback(MockCd mockCd) internal { - uint256[] memory numbers = new uint256[](100); - unchecked { - for (uint256 i; i < numbers.length; ++i) { - numbers[i] = i % 2 == 0 ? i : ~i; - } - } - assertEq(mockCd.numbersHash(), 0); - assertEq(mockCd.lastCallvalue(), 0); - assertEq(mockCd.lastCaller(), address(0)); - - uint256 callValue = 123 ether; - vm.deal(address(this), callValue * 2); - - (bool success, bytes memory result) = payable(mockCd).call{value: callValue}( - LibZip.cdCompress( - abi.encodeWithSignature("storeNumbersHash(uint256[],bool)", numbers, true) - ) - ); - - assertTrue(success); - bytes32 decodedNumbersHash = abi.decode(result, (bytes32)); - bytes32 expectedNumbersHash = keccak256(abi.encode(numbers)); - assertEq(decodedNumbersHash, expectedNumbersHash); - assertEq(mockCd.numbersHash(), expectedNumbersHash); - assertEq(mockCd.lastCallvalue(), callValue); - assertEq(mockCd.lastCaller(), address(this)); - assertEq(address(mockCd).balance, callValue); - - (success, result) = payable(mockCd).call{value: callValue}( - LibZip.cdCompress( - abi.encodeWithSignature("storeNumbersHash(uint256[],bool)", numbers, false) - ) - ); - - assertFalse(success); - assertEq(address(mockCd).balance, callValue); - assertEq(abi.encodeWithSelector(MockCd.Hash.selector, expectedNumbersHash), result); - assertEq(address(mockCd).balance, callValue); - - (success, result) = payable(mockCd).call{value: callValue}(""); - assertEq(address(mockCd).balance, callValue * 2); - assertTrue(success); - } - - function testCdFallback(bytes memory data, uint256 callValue) public brutalizeMemory { - MockCd mockCd = new MockCd(); - callValue = _bound(callValue, 0, 123 ether); - vm.deal(address(this), callValue * 2); - if (_random() % 8 == 0) { - data = _expandedData(data); - } - - (bool success, bytes memory result) = payable(mockCd).call{value: callValue}( - LibZip.cdCompress(abi.encodeWithSignature("storeDataHash(bytes,bool)", data, true)) - ); - - assertTrue(success); - bytes32 decodedDataHash = abi.decode(result, (bytes32)); - bytes32 expectedDataHash = keccak256(data); - assertEq(decodedDataHash, expectedDataHash); - assertEq(mockCd.dataHash(), expectedDataHash); - assertEq(mockCd.lastCallvalue(), callValue); - assertEq(mockCd.lastCaller(), address(this)); - assertEq(address(mockCd).balance, callValue); - - (success, result) = payable(mockCd).call{value: callValue}( - LibZip.cdCompress(abi.encodeWithSignature("storeDataHash(bytes,bool)", data, false)) - ); - - assertFalse(success); - assertEq(address(mockCd).balance, callValue); - assertEq(abi.encodeWithSelector(MockCd.Hash.selector, expectedDataHash), result); - assertEq(address(mockCd).balance, callValue); - - (success, result) = payable(mockCd).call{value: callValue}(""); - assertEq(address(mockCd).balance, callValue * 2); - assertTrue(success); - } - - function testCdFallbackMaskTrick(uint256 i, uint256 j) public { - i = _bound(i, 0, 2 ** 248 - 1); - uint256 a; - uint256 b; - /// @solidity memory-safe-assembly - assembly { - a := byte(0, xor(add(i, not(3)), j)) - b := xor(byte(i, shl(224, 0xffffffff)), byte(0, j)) - } - assertEq(a, b); - } -} diff --git a/lib/solady/test/MerkleProofLib.t.sol b/lib/solady/test/MerkleProofLib.t.sol deleted file mode 100644 index cddce6f..0000000 --- a/lib/solady/test/MerkleProofLib.t.sol +++ /dev/null @@ -1,440 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {MerkleProofLib} from "../src/utils/MerkleProofLib.sol"; -import {LibString} from "../src/utils/LibString.sol"; - -contract MerkleProofLibTest is SoladyTest { - function testVerifyProofForHeightOneTree( - bool hasProof, - bool nonEmptyProof, - bool nonEmptyRoot, - bool nonEmptyLeaf - ) public { - bytes32 root; - if (nonEmptyRoot) { - root = bytes32("a"); - } - bytes32 leaf; - if (nonEmptyLeaf) { - leaf = bytes32("a"); - } - bytes32[] memory proof; - if (hasProof) { - proof = new bytes32[](1); - proof[0] = nonEmptyProof ? bytes32("a") : bytes32(0); - } - bool isValid = leaf == root && proof.length == 0; - assertEq(this.verify(proof, root, leaf), isValid); - } - - function testVerifyProof(bytes32[] memory data, uint256 randomness) public brutalizeMemory { - vm.assume(data.length > 1); - uint256 nodeIndex = randomness % data.length; - bytes32 root = _getRoot(data); - bytes32[] memory proof = _getProof(data, nodeIndex); - bytes32 leaf = data[nodeIndex]; - - assertTrue(this.verify(proof, root, leaf)); - - // Checks verify with corrupted root returns false. - assertFalse(this.verify(proof, bytes32(uint256(root) ^ 1), leaf)); - - // Checks verify with corrupted proof returns false. - proof[0] = bytes32(uint256(proof[0]) ^ 1); - assertFalse(this.verify(proof, root, leaf)); - - // Checks verify with corrupted root and proof returns false. - assertFalse(this.verify(proof, bytes32(uint256(root) ^ 1), leaf)); - } - - function testVerifyProofBasicCaseIsValid() public { - testVerifyProofBasicCase(false, false, false, 0x00); - } - - function testVerifyProofBasicCaseIsInvalid() public { - testVerifyProofBasicCase(false, false, true, 0x00); - } - - function testVerifyMultiProofMalicious() public { - bytes32[] memory realLeaves = new bytes32[](2); - realLeaves[0] = bytes32("real leaf"); - realLeaves[1] = bytes32(0); - bytes32 root = _hashPair(realLeaves[0], realLeaves[1]); - - bytes32[] memory maliciousLeaves = new bytes32[](2); - maliciousLeaves[0] = bytes32("malicious"); - maliciousLeaves[1] = bytes32("leaves"); - bytes32[] memory maliciousProof = new bytes32[](2); - maliciousProof[0] = realLeaves[0]; - maliciousProof[1] = realLeaves[0]; - bool[] memory maliciousFlags = new bool[](3); - maliciousFlags[0] = true; - maliciousFlags[1] = true; - maliciousFlags[2] = false; - - assertFalse(this.verifyMultiProof(maliciousProof, root, maliciousLeaves, maliciousFlags)); - } - - function testVerifyProofBasicCase( - bool damageProof, - bool damageRoot, - bool damageLeaf, - bytes32 randomness - ) public { - bool noDamage = true; - uint256 ri; // Randomness index. - - bytes32[] memory proof = new bytes32[](2); - proof[0] = bytes32("b"); - proof[1] = _hashPair(bytes32("c"), bytes32(0)); - if (damageProof) { - noDamage = false; - uint256 i = uint256(uint8(randomness[ri++])) % proof.length; - proof[i] = bytes32(uint256(proof[i]) ^ 1); // Flip a bit. - } - - bytes32 root = - _hashPair(_hashPair(bytes32("a"), bytes32("b")), _hashPair(bytes32("c"), bytes32(0))); - - if (damageRoot) { - noDamage = false; - root = bytes32(uint256(root) ^ 1); // Flip a bit. - } - - bytes32 leaf = bytes32("a"); - if (damageLeaf) { - noDamage = false; - leaf = bytes32(uint256(leaf) ^ 1); // Flip a bit. - } - - assertEq(this.verify(proof, root, leaf), noDamage); - } - - function testVerifyMultiProofForSingleLeaf(bytes32[] memory data, uint256 randomness) - public - brutalizeMemory - { - vm.assume(data.length > 1); - uint256 nodeIndex = randomness % data.length; - bytes32 root = _getRoot(data); - bytes32[] memory proof = _getProof(data, nodeIndex); - bytes32[] memory leaves = new bytes32[](1); - leaves[0] = data[nodeIndex]; - bool[] memory flags = new bool[](proof.length); - - assertTrue(this.verifyMultiProof(proof, root, leaves, flags)); - - // Checks verify with corrupted root returns false. - assertFalse(this.verifyMultiProof(proof, bytes32(uint256(root) ^ 1), leaves, flags)); - - // Checks verify with corrupted proof returns false. - proof[0] = bytes32(uint256(proof[0]) ^ 1); - assertFalse(this.verifyMultiProof(proof, root, leaves, flags)); - - // Checks verify with corrupted root and proof returns false. - assertFalse(this.verifyMultiProof(proof, bytes32(uint256(root) ^ 1), leaves, flags)); - } - - function testVerifyMultiProofForHeightOneTree( - bool hasProof, - bool nonEmptyProof, - bool nonEmptyRoot, - bool hasLeaf, - bool nonEmptyLeaf, - bool[] memory flags - ) public { - bytes32 root; - if (nonEmptyRoot) { - root = bytes32("a"); - } - bytes32[] memory proof; - if (hasProof) { - proof = new bytes32[](1); - proof[0] = nonEmptyProof ? bytes32("a") : bytes32(0); - } - bytes32[] memory leaves; - if (hasLeaf) { - leaves = new bytes32[](1); - leaves[0] = nonEmptyLeaf ? bytes32("a") : bytes32(0); - } - bool leafSameAsRoot = leaves.length == 1 && leaves[0] == root; - bool proofSameAsRoot = proof.length == 1 && proof[0] == root; - bool isValid = flags.length == 0 && (leafSameAsRoot || proofSameAsRoot) - && (leaves.length + proof.length == 1); - assertEq(this.verifyMultiProof(proof, root, leaves, flags), isValid); - } - - function testVerifyMultiProofForHeightTwoTree( - bool allLeaves, - bool damageRoot, - bool damageLeaves, - bool damageProof, - bool damageFlags, - bytes32 randomness - ) public { - bool noDamage = true; - uint256 ri; // Randomness index. - - bytes32 root = _hashPair(bytes32("a"), bytes32("b")); - - bytes32[] memory proof; - bytes32[] memory leaves; - bool[] memory flags = new bool[](1); - flags[0] = allLeaves; - - if (allLeaves) { - leaves = new bytes32[](2); - leaves[0] = bytes32("a"); - leaves[1] = bytes32("b"); - } else { - leaves = new bytes32[](1); - leaves[0] = bytes32("a"); - proof = new bytes32[](1); - proof[0] = bytes32("b"); - } - - if (damageRoot) { - noDamage = false; - root = bytes32(uint256(root) ^ 1); // Flip a bit. - } - - if (damageFlags) { - noDamage = false; - flags[0] = !flags[0]; // Flip a bool. - if (uint256(uint8(randomness[ri++])) & 1 == 0) delete flags; - } - - if (damageLeaves) { - noDamage = false; - uint256 i = uint256(uint8(randomness[ri++])) % leaves.length; - leaves[i] = bytes32(uint256(leaves[i]) ^ 1); // Flip a bit. - if (uint256(uint8(randomness[ri++])) & 1 == 0) delete leaves; - } - - if (damageProof && proof.length != 0) { - noDamage = false; - proof[0] = bytes32(uint256(proof[0]) ^ 1); // Flip a bit. - if (uint256(uint8(randomness[ri++])) & 1 == 0) delete proof; - } - - assertEq(this.verifyMultiProof(proof, root, leaves, flags), noDamage); - } - - function testVerifyMultiProofIsValid() public { - testVerifyMultiProof(false, false, false, false, 0x00); - } - - function testVerifyMultiProofIsInvalid() public { - testVerifyMultiProof(false, false, true, false, 0x00); - } - - function testVerifyMultiProof( - bool damageRoot, - bool damageLeaves, - bool damageProof, - bool damageFlags, - bytes32 randomness - ) public brutalizeMemory { - bool noDamage = true; - uint256 ri; // Randomness index. - - bytes32 root = _hashPair( - _hashPair(_hashPair(bytes32("a"), bytes32("b")), _hashPair(bytes32("c"), bytes32("d"))), - _hashPair(bytes32("e"), bytes32("f")) - ); - - bytes32[] memory leaves = new bytes32[](3); - leaves[0] = bytes32("d"); - leaves[1] = bytes32("e"); - leaves[2] = bytes32("f"); - - bytes32[] memory proof = new bytes32[](2); - proof[0] = bytes32("c"); - proof[1] = _hashPair(bytes32("b"), bytes32("a")); - - bool[] memory flags = new bool[](4); - flags[0] = false; - flags[1] = true; - flags[2] = false; - flags[3] = true; - - if (damageRoot) { - noDamage = false; - root = bytes32(uint256(root) ^ 1); // Flip a bit. - } - - if (damageLeaves) { - noDamage = false; - uint256 i = uint256(uint8(randomness[ri++])) % leaves.length; - leaves[i] = bytes32(uint256(leaves[i]) ^ 1); // Flip a bit. - if (uint256(uint8(randomness[ri++])) & 1 == 0) delete leaves; - } - - if (damageProof) { - noDamage = false; - uint256 i = uint256(uint8(randomness[ri++])) % proof.length; - proof[i] = bytes32(uint256(proof[i]) ^ 1); // Flip a bit. - if (uint256(uint8(randomness[ri++])) & 1 == 0) delete proof; - } - - if (damageFlags) { - noDamage = false; - uint256 i = uint256(uint8(randomness[ri++])) % flags.length; - flags[i] = !flags[i]; // Flip a bool. - if (uint256(uint8(randomness[ri++])) & 1 == 0) delete flags; - } - - assertEq(this.verifyMultiProof(proof, root, leaves, flags), noDamage); - } - - function verify(bytes32[] calldata proof, bytes32 root, bytes32 leaf) - external - returns (bool result) - { - result = MerkleProofLib.verifyCalldata(proof, root, leaf); - assertEq(MerkleProofLib.verify(proof, root, leaf), result); - } - - function verifyMultiProof( - bytes32[] calldata proof, - bytes32 root, - bytes32[] calldata leaves, - bool[] calldata flags - ) external returns (bool result) { - uint256[] memory offsetsAndLengths = new uint256[](12); - - // Basically, we want to demonstrate that the `verifyMultiProof` does not - // change the offsets and lengths. - - /// @solidity memory-safe-assembly - assembly { - mstore(add(offsetsAndLengths, shl(5, add(1, 0))), proof.offset) - mstore(add(offsetsAndLengths, shl(5, add(1, 1))), leaves.offset) - mstore(add(offsetsAndLengths, shl(5, add(1, 2))), flags.offset) - mstore(add(offsetsAndLengths, shl(5, add(1, 3))), proof.length) - mstore(add(offsetsAndLengths, shl(5, add(1, 4))), leaves.length) - mstore(add(offsetsAndLengths, shl(5, add(1, 5))), flags.length) - } - - result = MerkleProofLib.verifyMultiProofCalldata(proof, root, leaves, flags); - - /// @solidity memory-safe-assembly - assembly { - mstore(add(offsetsAndLengths, shl(5, add(1, 6))), proof.offset) - mstore(add(offsetsAndLengths, shl(5, add(1, 7))), leaves.offset) - mstore(add(offsetsAndLengths, shl(5, add(1, 8))), flags.offset) - mstore(add(offsetsAndLengths, shl(5, add(1, 9))), proof.length) - mstore(add(offsetsAndLengths, shl(5, add(1, 10))), leaves.length) - mstore(add(offsetsAndLengths, shl(5, add(1, 11))), flags.length) - } - - assertEq(offsetsAndLengths[0], offsetsAndLengths[6]); - assertEq(offsetsAndLengths[1], offsetsAndLengths[7]); - assertEq(offsetsAndLengths[2], offsetsAndLengths[8]); - assertEq(offsetsAndLengths[3], offsetsAndLengths[9]); - assertEq(offsetsAndLengths[4], offsetsAndLengths[10]); - assertEq(offsetsAndLengths[5], offsetsAndLengths[11]); - - assertEq(MerkleProofLib.verifyMultiProof(proof, root, leaves, flags), result); - } - - // Following code is adapted from https://github.com/dmfxyz/murky/blob/main/src/common/MurkyBase.sol. - - function _getRoot(bytes32[] memory data) private pure returns (bytes32) { - require(data.length > 1); - while (data.length > 1) { - data = _hashLevel(data); - } - return data[0]; - } - - function _getProof(bytes32[] memory data, uint256 nodeIndex) - private - pure - returns (bytes32[] memory) - { - require(data.length > 1); - - bytes32[] memory result = new bytes32[](64); - uint256 pos; - - while (data.length > 1) { - unchecked { - if (nodeIndex & 0x1 == 1) { - result[pos] = data[nodeIndex - 1]; - } else if (nodeIndex + 1 == data.length) { - result[pos] = bytes32(0); - } else { - result[pos] = data[nodeIndex + 1]; - } - ++pos; - nodeIndex /= 2; - } - data = _hashLevel(data); - } - // Resize the length of the array to fit. - /// @solidity memory-safe-assembly - assembly { - mstore(result, pos) - } - - return result; - } - - function _hashLevel(bytes32[] memory data) private pure returns (bytes32[] memory) { - bytes32[] memory result; - unchecked { - uint256 length = data.length; - if (length & 0x1 == 1) { - result = new bytes32[](length / 2 + 1); - result[result.length - 1] = _hashPair(data[length - 1], bytes32(0)); - } else { - result = new bytes32[](length / 2); - } - uint256 pos = 0; - for (uint256 i = 0; i < length - 1; i += 2) { - result[pos] = _hashPair(data[i], data[i + 1]); - ++pos; - } - } - return result; - } - - function _hashPair(bytes32 left, bytes32 right) private pure returns (bytes32 result) { - /// @solidity memory-safe-assembly - assembly { - switch lt(left, right) - case 0 { - mstore(0x0, right) - mstore(0x20, left) - } - default { - mstore(0x0, left) - mstore(0x20, right) - } - result := keccak256(0x0, 0x40) - } - } - - function testEmptyCalldataHelpers() public { - assertFalse( - MerkleProofLib.verifyMultiProofCalldata( - MerkleProofLib.emptyProof(), - bytes32(0), - MerkleProofLib.emptyLeaves(), - MerkleProofLib.emptyFlags() - ) - ); - - assertFalse( - MerkleProofLib.verifyMultiProof( - MerkleProofLib.emptyProof(), - bytes32(0), - MerkleProofLib.emptyLeaves(), - MerkleProofLib.emptyFlags() - ) - ); - } -} diff --git a/lib/solady/test/MetadataReaderLib.t.sol b/lib/solady/test/MetadataReaderLib.t.sol deleted file mode 100644 index db7f8ac..0000000 --- a/lib/solady/test/MetadataReaderLib.t.sol +++ /dev/null @@ -1,305 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {MetadataReaderLib} from "../src/utils/MetadataReaderLib.sol"; -import {LibString} from "../src/utils/LibString.sol"; - -contract MetadataReaderLibTest is SoladyTest { - string internal _stringToReturn; - - uint256 internal _randomness; - - function returnsString() public view returns (string memory) { - uint256 r = _randomness; - /// @solidity memory-safe-assembly - assembly { - mstore(0x40, add(0x100, mload(0x40))) - mstore(0x00, r) - } - string memory s = _stringToReturn; - /// @solidity memory-safe-assembly - assembly { - if iszero(and(r, 1)) { - if iszero(and(r, 2)) { - mstore(sub(s, 0x40), 0x40) - return(sub(s, 0x40), add(0x60, add(mload(s), byte(2, r)))) - } - mstore(sub(s, 0x20), 0x20) - return(sub(s, 0x20), add(0x40, add(mload(s), byte(2, r)))) - } - mstore(0x00, gas()) - mstore(0x20, r) - mstore(add(mload(s), add(s, 0x20)), shr(8, keccak256(0x00, 0x40))) - return(add(s, 0x20), add(mload(s), byte(2, r))) - } - } - - function returnsEmptyString() public view returns (string memory) { - uint256 r = _randomness; - /// @solidity memory-safe-assembly - assembly { - mstore(0x40, add(0x100, mload(0x40))) - mstore(0x00, r) - } - string memory s = _stringToReturn; - /// @solidity memory-safe-assembly - assembly { - if iszero(and(r, 1)) { - if iszero(and(r, 2)) { - mstore(sub(s, 0x40), 0x41) - return(sub(s, 0x40), add(0x60, mload(s))) - } - mstore(sub(s, 0x20), 0x21) - return(sub(s, 0x20), add(0x40, mload(s))) - } - if iszero(and(r, 2)) { - let n := mload(s) - mstore(s, add(n, 1)) - if iszero(and(r, 2)) { - mstore(sub(s, 0x40), 0x40) - return(sub(s, 0x40), add(0x60, n)) - } - mstore(sub(s, 0x20), 0x20) - return(sub(s, 0x20), add(0x40, n)) - } - let m := mload(0x40) - codecopy(m, codesize(), 0x200) - mstore(m, and(63, byte(3, r))) - return(m, and(63, byte(2, r))) - } - } - - function returnsChoppedString(uint256 chop) public pure returns (string memory) { - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) - mstore(add(m, 0x00), 0x20) - mstore(add(m, 0x20), 0x20) - mstore(add(m, 0x40), "112233445566778899aa112233445566") - return(add(m, 0x00), add(0x40, chop)) - } - } - - function returnsBytes32StringA() public pure returns (bytes32) { - return bytes32(hex"4d696c616479"); - } - - function returnsBytes32StringB() public pure returns (bytes32) { - return bytes32("This string has thirty two bytes"); - } - - function returnsNothing() public pure {} - - function reverts() public pure { - revert("Lorem Ipsum"); - } - - function returnsChoppedUint(uint256 v, uint256 chop) public pure returns (uint256) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, v) - return(0x00, chop) - } - } - - function name() public view returns (string memory) { - return returnsString(); - } - - function symbol() public view returns (string memory) { - return returnsString(); - } - - function returnsUint() public view returns (uint256) { - return _randomness; - } - - function decimals() public view returns (uint8) { - return uint8(_randomness); - } - - function testReadBytes32String() public brutalizeMemory { - bytes memory data; - string memory result; - data = abi.encodeWithSignature("returnsBytes32StringA()"); - result = MetadataReaderLib.readString(address(this), data, type(uint256).max); - _checkMemory(result); - assertEq(result, "Milady"); - data = abi.encodeWithSignature("returnsBytes32StringB()"); - result = MetadataReaderLib.readString(address(this), data, type(uint256).max); - _checkMemory(result); - assertEq(result, "This string has thirty two bytes"); - data = abi.encodeWithSignature("returnsNothing()"); - result = MetadataReaderLib.readString(address(this), data, type(uint256).max); - _checkMemory(result); - assertEq(result, ""); - data = abi.encodeWithSignature("reverts()"); - result = MetadataReaderLib.readString(address(this), data, type(uint256).max); - _checkMemory(result); - assertEq(result, ""); - } - - function testReadBytes32StringTruncated() public brutalizeMemory { - bytes memory data; - string memory result; - unchecked { - data = abi.encodeWithSignature("returnsBytes32StringB()"); - for (uint256 limit; limit < 39; ++limit) { - result = MetadataReaderLib.readString(address(this), data, limit); - _checkMemory(result); - assertEq(result, LibString.slice("This string has thirty two bytes", 0, limit)); - } - } - } - - function testReadStringChopped() public { - bytes memory data; - string memory result; - data = abi.encodeWithSignature("returnsChoppedString(uint256)", uint256(32)); - result = MetadataReaderLib.readString(address(this), data, type(uint256).max); - _checkMemory(result); - assertEq(result, "112233445566778899aa112233445566"); - - for (uint256 limit; limit < 39; limit += 5) { - for (uint256 chop; chop < 39; chop += 3) { - data = abi.encodeWithSignature("returnsChoppedString(uint256)", uint256(chop)); - result = MetadataReaderLib.readString(address(this), data, limit); - _checkMemory(result); - // As long as the returndatasize is insufficient, the `abi.decode` will fail, - // and the resultant string will be empty. - // Even if the `limit` is smaller than `chop`. - string memory expected; - if (chop >= 32) { - expected = LibString.slice("112233445566778899aa112233445566", 0, limit); - } - assertEq(result, expected); - } - } - } - - function testReadString(uint256 r) public brutalizeMemory { - bytes memory data; - string memory result; - string memory s = _generateString(); - _stringToReturn = s; - _randomness = r; - data = abi.encodeWithSignature("returnsString()"); - result = MetadataReaderLib.readString(address(this), data, type(uint256).max); - _checkMemory(result); - assertEq(result, s); - result = MetadataReaderLib.readName(address(this), type(uint256).max); - _checkMemory(result); - assertEq(result, s); - result = MetadataReaderLib.readSymbol(address(this), type(uint256).max); - _checkMemory(result); - assertEq(result, s); - data = abi.encodeWithSignature("returnsEmptyString()"); - result = MetadataReaderLib.readString(address(this), data, type(uint256).max); - _checkMemory(result); - assertEq(result, ""); - data = abi.encodeWithSignature("reverts()"); - result = MetadataReaderLib.readString(address(this), data, type(uint256).max); - _checkMemory(result); - assertEq(result, ""); - data = abi.encodeWithSignature("returnsNothing()"); - result = MetadataReaderLib.readString(address(this), data, type(uint256).max); - _checkMemory(result); - assertEq(result, ""); - } - - function testReadStringTruncated(uint256 r) public brutalizeMemory { - bytes memory data; - string memory result; - string memory s = _generateString(); - _stringToReturn = s; - _randomness = r; - unchecked { - uint256 limit = _bound(_random(), 0, bytes(s).length * 2); - data = abi.encodeWithSignature("returnsString()"); - result = MetadataReaderLib.readString(address(this), data, limit); - _checkMemory(result); - assertEq(result, LibString.slice(s, 0, limit)); - } - } - - function testReadUint(uint256 r) public { - _randomness = r; - bytes memory data = abi.encodeWithSignature("returnsUint()"); - assertEq(MetadataReaderLib.readUint(address(this), data), r); - assertEq(MetadataReaderLib.readDecimals(address(this)), uint8(r)); - } - - function testReadUint() public { - bytes memory data; - uint256 result; - data = abi.encodeWithSignature("returnsNothing()"); - result = MetadataReaderLib.readUint(address(this), data); - assertEq(result, 0); - data = abi.encodeWithSignature("reverts()"); - result = MetadataReaderLib.readUint(address(this), data); - assertEq(result, 0); - - for (uint256 j; j != 8; ++j) { - for (uint256 i; i != 70; ++i) { - uint256 k = _hash(i, j); - data = abi.encodeWithSignature("returnsChoppedUint(uint256,uint256)", k, i); - result = MetadataReaderLib.readUint(address(this), data); - assertEq(result, i < 32 ? 0 : k); - } - } - } - - function testBoundsCheckDifferential(uint256) public { - uint256 rds = _bound(_random(), 0, 128); - uint256 l = _random() % 2 == 0 ? type(uint248).max : 128; - uint256 o = _bound(_random(), 0, l); - uint256 n = _bound(_random(), 0, l); - bool result; - /// @solidity memory-safe-assembly - assembly { - if iszero(lt(rds, 0x40)) { - if iszero(gt(o, sub(rds, 0x20))) { - if iszero(gt(n, sub(rds, add(o, 0x20)))) { result := 1 } - } - } - } - bool expected = rds >= 0x40 && !(o + 0x20 > rds) && !(n + o + 0x20 > rds); - assertEq(result, expected); - } - - function _hash(uint256 i, uint256 j) internal pure returns (uint256 result) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, i) - mstore(0x20, j) - result := keccak256(0x00, 0x20) - } - } - - function _generateString() internal returns (string memory result) { - uint256 randomness = _random(); - uint256 resultLength = _randomStringLength(); - /// @solidity memory-safe-assembly - assembly { - result := mload(0x40) - mstore(0x00, randomness) - mstore(0x40, and(add(add(result, 0x40), resultLength), not(31))) - mstore(result, resultLength) - - // forgefmt: disable-next-item - for { let i := 0 } lt(i, resultLength) { i := add(i, 1) } { - mstore(0x20, gas()) - let c := byte(0, keccak256(0x00, 0x40)) - mstore8(add(add(result, 0x20), i), or(c, iszero(c))) - } - } - } - - function _randomStringLength() internal returns (uint256 r) { - r = _random() % 256; - if (r < 64) return _random() % 128; - if (r < 128) return _random() % 64; - return _random() % 16; - } -} diff --git a/lib/solady/test/MinHeapLib.t.sol b/lib/solady/test/MinHeapLib.t.sol deleted file mode 100644 index 07a1efa..0000000 --- a/lib/solady/test/MinHeapLib.t.sol +++ /dev/null @@ -1,130 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {MinHeapLib} from "../src/utils/MinHeapLib.sol"; -import {LibSort} from "../src/utils/LibSort.sol"; - -contract MinHeapLibTest is SoladyTest { - using MinHeapLib for MinHeapLib.Heap; - - MinHeapLib.Heap heap0; - - MinHeapLib.Heap heap1; - - function testHeapRoot(uint256 x) public { - vm.expectRevert(MinHeapLib.HeapIsEmpty.selector); - heap0.root(); - heap0.data.push(x); - assertEq(heap0.length(), 1); - assertEq(heap0.root(), x); - } - - function testHeapPushAndPop(uint256) public { - unchecked { - uint256 n = _random() % 8; - uint256[] memory a = new uint256[](n); - - for (uint256 i; i < n; ++i) { - uint256 r = _random(); - a[i] = r; - heap0.push(r); - } - LibSort.insertionSort(a); - for (uint256 i; i < n; ++i) { - assertEq(heap0.pop(), a[i]); - } - assertEq(heap0.length(), 0); - } - } - - function testHeapPushPop(uint256) public { - unchecked { - uint256 n = _random() % 8; - for (uint256 i; i < n; ++i) { - uint256 r = _random(); - heap0.push(r); - heap1.push(r); - } - n = _random() % 8; - for (uint256 i; i < n; ++i) { - uint256 r = _random(); - uint256 popped0 = heap0.pushPop(r); - heap1.push(r); - uint256 popped1 = heap1.pop(); - assertEq(popped0, popped1); - } - } - } - - function testHeapReplace(uint256) public { - unchecked { - uint256 n = _random() % 8 + 1; - for (uint256 i; i < n; ++i) { - uint256 r = _random(); - heap0.push(r); - heap1.push(r); - } - n = _random() % 8; - for (uint256 i; i < n; ++i) { - uint256 r = _random(); - uint256 popped0 = heap0.replace(r); - uint256 popped1 = heap1.pop(); - heap1.push(r); - assertEq(popped0, popped1); - } - } - } - - function testHeapEnqueue(uint256) public { - unchecked { - uint256 maxLength = _random() % 8 + 1; - uint256 m = _random() % 32 + maxLength; - uint256[] memory a = new uint256[](m); - uint256[] memory rejected = new uint256[](m); - uint256 numRejected; - for (uint256 i; i < m; ++i) { - uint256 r = _random(); - (bool success, bool hasPopped, uint256 popped) = heap0.enqueue(r, maxLength); - if (hasPopped) { - assertEq(heap0.length(), maxLength); - assertEq(success, true); - rejected[numRejected++] = popped; - } - if (!success) { - assertEq(heap0.length(), maxLength); - rejected[numRejected++] = r; - } - a[i] = r; - } - LibSort.insertionSort(a); - /// @solidity memory-safe-assembly - assembly { - mstore(rejected, numRejected) - } - LibSort.insertionSort(rejected); - for (uint256 i; i < maxLength; ++i) { - assertEq(a[m - maxLength + i], heap0.pop()); - } - assertEq(numRejected + maxLength, m); - for (uint256 i; i < numRejected; ++i) { - assertEq(a[i], rejected[i]); - } - } - } - - function testHeapEnqueueGas(uint256) public { - unchecked { - for (uint256 i; i < 16; ++i) { - this.enqueue(i, 8); - } - for (uint256 i; i < 16; ++i) { - this.enqueue(_random() % 16, 8); - } - } - } - - function enqueue(uint256 value, uint256 maxLength) public { - heap0.enqueue(value, maxLength); - } -} diff --git a/lib/solady/test/Multicallable.t.sol b/lib/solady/test/Multicallable.t.sol deleted file mode 100644 index bbc1de8..0000000 --- a/lib/solady/test/Multicallable.t.sol +++ /dev/null @@ -1,119 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {MockMulticallable} from "./utils/mocks/MockMulticallable.sol"; - -contract MulticallableTest is SoladyTest { - MockMulticallable multicallable; - - function setUp() public { - multicallable = new MockMulticallable(); - } - - function testMulticallableRevertWithMessage(string memory revertMessage) public { - bytes[] memory data = new bytes[](1); - data[0] = - abi.encodeWithSelector(MockMulticallable.revertsWithString.selector, revertMessage); - vm.expectRevert(bytes(revertMessage)); - multicallable.multicall(data); - } - - function testMulticallableRevertWithMessage() public { - testMulticallableRevertWithMessage("Milady"); - } - - function testMulticallableRevertWithCustomError() public { - bytes[] memory data = new bytes[](1); - data[0] = abi.encodeWithSelector(MockMulticallable.revertsWithCustomError.selector); - vm.expectRevert(MockMulticallable.CustomError.selector); - multicallable.multicall(data); - } - - function testMulticallableRevertWithNothing() public { - bytes[] memory data = new bytes[](1); - data[0] = abi.encodeWithSelector(MockMulticallable.revertsWithNothing.selector); - vm.expectRevert(); - multicallable.multicall(data); - } - - function testMulticallableReturnDataIsProperlyEncoded( - uint256 a0, - uint256 b0, - uint256 a1, - uint256 b1 - ) public { - bytes[] memory data = new bytes[](2); - data[0] = abi.encodeWithSelector(MockMulticallable.returnsTuple.selector, a0, b0); - data[1] = abi.encodeWithSelector(MockMulticallable.returnsTuple.selector, a1, b1); - bytes[] memory returnedData = multicallable.multicall(data); - MockMulticallable.Tuple memory t0 = abi.decode(returnedData[0], (MockMulticallable.Tuple)); - MockMulticallable.Tuple memory t1 = abi.decode(returnedData[1], (MockMulticallable.Tuple)); - assertEq(t0.a, a0); - assertEq(t0.b, b0); - assertEq(t1.a, a1); - assertEq(t1.b, b1); - } - - function testMulticallableReturnDataIsProperlyEncoded( - string memory sIn0, - string memory sIn1, - uint256 n - ) public { - n = n % 2; - bytes[] memory dataIn = new bytes[](n); - if (n > 0) { - dataIn[0] = abi.encodeWithSelector(MockMulticallable.returnsString.selector, sIn0); - } - if (n > 1) { - dataIn[1] = abi.encodeWithSelector(MockMulticallable.returnsString.selector, sIn1); - } - bytes[] memory dataOut = multicallable.multicall(dataIn); - if (n > 0) { - assertEq(abi.decode(dataOut[0], (string)), sIn0); - } - if (n > 1) { - assertEq(abi.decode(dataOut[1], (string)), sIn1); - } - } - - function testMulticallableReturnDataIsProperlyEncoded() public { - testMulticallableReturnDataIsProperlyEncoded(0, 1, 2, 3); - } - - function testMulticallableBenchmark() public { - unchecked { - bytes[] memory data = new bytes[](10); - for (uint256 i; i != data.length; ++i) { - data[i] = abi.encodeWithSelector(MockMulticallable.returnsTuple.selector, i, i + 1); - } - bytes[] memory returnedData = multicallable.multicall(data); - assertEq(returnedData.length, data.length); - } - } - - function testMulticallableOriginalBenchmark() public { - unchecked { - bytes[] memory data = new bytes[](10); - for (uint256 i; i != data.length; ++i) { - data[i] = abi.encodeWithSelector(MockMulticallable.returnsTuple.selector, i, i + 1); - } - bytes[] memory returnedData = multicallable.multicallOriginal(data); - assertEq(returnedData.length, data.length); - } - } - - function testMulticallableWithNoData() public { - bytes[] memory data = new bytes[](0); - assertEq(multicallable.multicall(data).length, 0); - } - - function testMulticallablePreservesMsgSender() public { - address caller = address(uint160(0xbeef)); - bytes[] memory data = new bytes[](1); - data[0] = abi.encodeWithSelector(MockMulticallable.returnsSender.selector); - vm.prank(caller); - address returnedAddress = abi.decode(multicallable.multicall(data)[0], (address)); - assertEq(caller, returnedAddress); - } -} diff --git a/lib/solady/test/Ownable.t.sol b/lib/solady/test/Ownable.t.sol deleted file mode 100644 index 9e14132..0000000 --- a/lib/solady/test/Ownable.t.sol +++ /dev/null @@ -1,164 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import "./utils/mocks/MockOwnable.sol"; - -contract OwnableTest is SoladyTest { - event OwnershipTransferred(address indexed oldOwner, address indexed newOwner); - - event OwnershipHandoverRequested(address indexed pendingOwner); - - event OwnershipHandoverCanceled(address indexed pendingOwner); - - MockOwnable mockOwnable; - - function setUp() public { - mockOwnable = new MockOwnable(); - } - - function testBytecodeSize() public { - MockOwnableBytecodeSizer mock = new MockOwnableBytecodeSizer(); - assertTrue(address(mock).code.length > 0); - assertEq(mock.owner(), address(this)); - } - - function testInitializeOwnerDirect() public { - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(0), address(1)); - mockOwnable.initializeOwnerDirect(address(1)); - } - - function testSetOwnerDirect(address newOwner) public { - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(this), newOwner); - mockOwnable.setOwnerDirect(newOwner); - assertEq(mockOwnable.owner(), newOwner); - } - - function testSetOwnerDirect() public { - testSetOwnerDirect(address(1)); - } - - function testRenounceOwnership() public { - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(this), address(0)); - mockOwnable.renounceOwnership(); - assertEq(mockOwnable.owner(), address(0)); - } - - function testTransferOwnership( - address newOwner, - bool setNewOwnerToZeroAddress, - bool callerIsOwner - ) public { - assertEq(mockOwnable.owner(), address(this)); - - vm.assume(newOwner != address(this)); - - if (newOwner == address(0) || setNewOwnerToZeroAddress) { - newOwner = address(0); - vm.expectRevert(Ownable.NewOwnerIsZeroAddress.selector); - } else if (callerIsOwner) { - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(this), newOwner); - } else { - vm.prank(newOwner); - vm.expectRevert(Ownable.Unauthorized.selector); - } - - mockOwnable.transferOwnership(newOwner); - - if (newOwner != address(0) && callerIsOwner) { - assertEq(mockOwnable.owner(), newOwner); - } - } - - function testTransferOwnership() public { - testTransferOwnership(address(1), false, true); - } - - function testOnlyOwnerModifier(address nonOwner, bool callerIsOwner) public { - vm.assume(nonOwner != address(this)); - - if (!callerIsOwner) { - vm.prank(nonOwner); - vm.expectRevert(Ownable.Unauthorized.selector); - } - mockOwnable.updateFlagWithOnlyOwner(); - } - - function testHandoverOwnership(address pendingOwner) public { - vm.prank(pendingOwner); - vm.expectEmit(true, true, true, true); - emit OwnershipHandoverRequested(pendingOwner); - mockOwnable.requestOwnershipHandover(); - assertTrue(mockOwnable.ownershipHandoverExpiresAt(pendingOwner) > block.timestamp); - - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(this), pendingOwner); - - mockOwnable.completeOwnershipHandover(pendingOwner); - - assertEq(mockOwnable.owner(), pendingOwner); - } - - function testHandoverOwnership() public { - testHandoverOwnership(address(1)); - } - - function testHandoverOwnershipRevertsIfCompleteIsNotOwner() public { - address pendingOwner = address(1); - vm.prank(pendingOwner); - mockOwnable.requestOwnershipHandover(); - - vm.prank(pendingOwner); - vm.expectRevert(Ownable.Unauthorized.selector); - mockOwnable.completeOwnershipHandover(pendingOwner); - } - - function testHandoverOwnershipWithCancellation() public { - address pendingOwner = address(1); - - vm.prank(pendingOwner); - vm.expectEmit(true, true, true, true); - emit OwnershipHandoverRequested(pendingOwner); - mockOwnable.requestOwnershipHandover(); - assertTrue(mockOwnable.ownershipHandoverExpiresAt(pendingOwner) > block.timestamp); - - vm.expectEmit(true, true, true, true); - emit OwnershipHandoverCanceled(pendingOwner); - vm.prank(pendingOwner); - mockOwnable.cancelOwnershipHandover(); - assertEq(mockOwnable.ownershipHandoverExpiresAt(pendingOwner), 0); - vm.expectRevert(Ownable.NoHandoverRequest.selector); - - mockOwnable.completeOwnershipHandover(pendingOwner); - } - - function testHandoverOwnershipBeforeExpiration() public { - address pendingOwner = address(1); - vm.prank(pendingOwner); - mockOwnable.requestOwnershipHandover(); - - vm.warp(block.timestamp + mockOwnable.ownershipHandoverValidFor()); - - mockOwnable.completeOwnershipHandover(pendingOwner); - } - - function testHandoverOwnershipAfterExpiration() public { - address pendingOwner = address(1); - vm.prank(pendingOwner); - mockOwnable.requestOwnershipHandover(); - - vm.warp(block.timestamp + mockOwnable.ownershipHandoverValidFor() + 1); - - vm.expectRevert(Ownable.NoHandoverRequest.selector); - - mockOwnable.completeOwnershipHandover(pendingOwner); - } - - function testOwnershipHandoverValidForDefaultValue() public { - assertEq(mockOwnable.ownershipHandoverValidFor(), 48 * 3600); - } -} diff --git a/lib/solady/test/OwnableRoles.t.sol b/lib/solady/test/OwnableRoles.t.sol deleted file mode 100644 index f8fc7a4..0000000 --- a/lib/solady/test/OwnableRoles.t.sol +++ /dev/null @@ -1,377 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import "./utils/mocks/MockOwnableRoles.sol"; - -contract OwnableRolesTest is SoladyTest { - event OwnershipTransferred(address indexed oldOwner, address indexed newOwner); - - event OwnershipHandoverRequested(address indexed pendingOwner); - - event OwnershipHandoverCanceled(address indexed pendingOwner); - - event RolesUpdated(address indexed user, uint256 indexed roles); - - MockOwnableRoles mockOwnableRoles; - - function setUp() public { - mockOwnableRoles = new MockOwnableRoles(); - } - - function testBytecodeSize() public { - MockOwnableRolesBytecodeSizer mock = new MockOwnableRolesBytecodeSizer(); - assertTrue(address(mock).code.length > 0); - assertEq(mock.owner(), address(this)); - } - - function testInitializeOwnerDirect() public { - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(0), address(1)); - mockOwnableRoles.initializeOwnerDirect(address(1)); - } - - function testSetOwnerDirect(address newOwner) public { - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(this), newOwner); - mockOwnableRoles.setOwnerDirect(newOwner); - assertEq(mockOwnableRoles.owner(), newOwner); - } - - function testGrantAndRemoveRolesDirect( - address user, - uint256 rolesToGrant, - uint256 rolesToRemove - ) public { - mockOwnableRoles.removeRolesDirect(user, mockOwnableRoles.rolesOf(user)); - assertEq(mockOwnableRoles.rolesOf(user), 0); - mockOwnableRoles.grantRolesDirect(user, rolesToGrant); - assertEq(mockOwnableRoles.rolesOf(user), rolesToGrant); - mockOwnableRoles.removeRolesDirect(user, rolesToRemove); - assertEq(mockOwnableRoles.rolesOf(user), rolesToGrant ^ (rolesToGrant & rolesToRemove)); - } - - function testSetRolesDirect(uint256) public { - address userA = _randomNonZeroAddress(); - address userB = _randomNonZeroAddress(); - while (userA == userB) userA = _randomNonZeroAddress(); - for (uint256 t; t != 2; ++t) { - uint256 rolesA = _random(); - uint256 rolesB = _random(); - vm.expectEmit(true, true, true, true); - emit RolesUpdated(userA, rolesA); - mockOwnableRoles.setRolesDirect(userA, rolesA); - emit RolesUpdated(userB, rolesB); - mockOwnableRoles.setRolesDirect(userB, rolesB); - assertEq(mockOwnableRoles.rolesOf(userA), rolesA); - assertEq(mockOwnableRoles.rolesOf(userB), rolesB); - } - } - - function testSetOwnerDirect() public { - testSetOwnerDirect(address(1)); - } - - function testRenounceOwnership() public { - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(this), address(0)); - mockOwnableRoles.renounceOwnership(); - assertEq(mockOwnableRoles.owner(), address(0)); - } - - function testTransferOwnership( - address newOwner, - bool setNewOwnerToZeroAddress, - bool callerIsOwner - ) public { - assertEq(mockOwnableRoles.owner(), address(this)); - - vm.assume(newOwner != address(this)); - - if (newOwner == address(0) || setNewOwnerToZeroAddress) { - newOwner = address(0); - vm.expectRevert(Ownable.NewOwnerIsZeroAddress.selector); - } else if (callerIsOwner) { - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(this), newOwner); - } else { - vm.prank(newOwner); - vm.expectRevert(Ownable.Unauthorized.selector); - } - - mockOwnableRoles.transferOwnership(newOwner); - - if (newOwner != address(0) && callerIsOwner) { - assertEq(mockOwnableRoles.owner(), newOwner); - } - } - - function testTransferOwnership() public { - testTransferOwnership(address(1), false, true); - } - - function testGrantRoles() public { - vm.expectEmit(true, true, true, true); - emit RolesUpdated(address(1), 111111); - mockOwnableRoles.grantRoles(address(1), 111111); - } - - function testGrantAndRevokeOrRenounceRoles( - address user, - bool granterIsOwner, - bool useRenounce, - bool revokerIsOwner, - uint256 rolesToGrant, - uint256 rolesToRevoke - ) public { - vm.assume(user != address(this)); - - uint256 rolesAfterRevoke = rolesToGrant ^ (rolesToGrant & rolesToRevoke); - - assertTrue(rolesAfterRevoke & rolesToRevoke == 0); - assertTrue((rolesAfterRevoke | rolesToRevoke) & rolesToGrant == rolesToGrant); - - if (granterIsOwner) { - vm.expectEmit(true, true, true, true); - emit RolesUpdated(user, rolesToGrant); - } else { - vm.prank(user); - vm.expectRevert(Ownable.Unauthorized.selector); - } - mockOwnableRoles.grantRoles(user, rolesToGrant); - - if (!granterIsOwner) return; - - assertEq(mockOwnableRoles.rolesOf(user), rolesToGrant); - - if (useRenounce) { - vm.expectEmit(true, true, true, true); - emit RolesUpdated(user, rolesAfterRevoke); - vm.prank(user); - mockOwnableRoles.renounceRoles(rolesToRevoke); - } else if (revokerIsOwner) { - vm.expectEmit(true, true, true, true); - emit RolesUpdated(user, rolesAfterRevoke); - mockOwnableRoles.revokeRoles(user, rolesToRevoke); - } else { - vm.prank(user); - vm.expectRevert(Ownable.Unauthorized.selector); - mockOwnableRoles.revokeRoles(user, rolesToRevoke); - return; - } - - assertEq(mockOwnableRoles.rolesOf(user), rolesAfterRevoke); - } - - function testHasAllRoles( - address user, - uint256 rolesToGrant, - uint256 rolesToGrantBrutalizer, - uint256 rolesToCheck, - bool useSameRoles - ) public { - if (useSameRoles) { - rolesToGrant = rolesToCheck; - } - rolesToGrant |= rolesToGrantBrutalizer; - mockOwnableRoles.grantRoles(user, rolesToGrant); - - bool hasAllRoles = (rolesToGrant & rolesToCheck) == rolesToCheck; - assertEq(mockOwnableRoles.hasAllRoles(user, rolesToCheck), hasAllRoles); - } - - function testHasAnyRole(address user, uint256 rolesToGrant, uint256 rolesToCheck) public { - mockOwnableRoles.grantRoles(user, rolesToGrant); - assertEq(mockOwnableRoles.hasAnyRole(user, rolesToCheck), rolesToGrant & rolesToCheck != 0); - } - - function testRolesFromOrdinals(uint8[] memory ordinals) public { - uint256 roles; - unchecked { - for (uint256 i; i < ordinals.length; ++i) { - roles |= 1 << uint256(ordinals[i]); - } - } - assertEq(mockOwnableRoles.rolesFromOrdinals(ordinals), roles); - } - - function testRolesFromOrdinals() public { - unchecked { - for (uint256 t; t != 32; ++t) { - uint8[] memory ordinals = new uint8[](_random() % 32); - for (uint256 i; i != ordinals.length; ++i) { - uint256 randomness = _random(); - uint8 r; - assembly { - r := randomness - } - ordinals[i] = r; - } - testRolesFromOrdinals(ordinals); - } - } - } - - function testOrdinalsFromRoles(uint256 roles) public { - uint8[] memory ordinals = new uint8[](256); - uint256 n; - unchecked { - for (uint256 i; i < 256; ++i) { - if (roles & (1 << i) != 0) ordinals[n++] = uint8(i); - } - } - uint8[] memory results = mockOwnableRoles.ordinalsFromRoles(roles); - assertEq(results.length, n); - unchecked { - for (uint256 i; i < n; ++i) { - assertEq(results[i], ordinals[i]); - } - } - } - - function testOrdinalsFromRoles() public { - unchecked { - for (uint256 t; t != 32; ++t) { - testOrdinalsFromRoles(_random()); - } - } - } - - function testOnlyOwnerModifier(address nonOwner, bool callerIsOwner) public { - vm.assume(nonOwner != address(this)); - - if (!callerIsOwner) { - vm.prank(nonOwner); - vm.expectRevert(Ownable.Unauthorized.selector); - } - mockOwnableRoles.updateFlagWithOnlyOwner(); - } - - function testOnlyRolesModifier(address user, uint256 rolesToGrant, uint256 rolesToCheck) - public - { - mockOwnableRoles.grantRoles(user, rolesToGrant); - - if (rolesToGrant & rolesToCheck == 0) { - vm.expectRevert(Ownable.Unauthorized.selector); - } - vm.prank(user); - mockOwnableRoles.updateFlagWithOnlyRoles(rolesToCheck); - } - - function testOnlyOwnerOrRolesModifier( - address user, - bool callerIsOwner, - uint256 rolesToGrant, - uint256 rolesToCheck - ) public { - vm.assume(user != address(this)); - - mockOwnableRoles.grantRoles(user, rolesToGrant); - - if ((rolesToGrant & rolesToCheck == 0) && !callerIsOwner) { - vm.expectRevert(Ownable.Unauthorized.selector); - } - if (!callerIsOwner) { - vm.prank(user); - } - mockOwnableRoles.updateFlagWithOnlyOwnerOrRoles(rolesToCheck); - } - - function testOnlyRolesOrOwnerModifier( - address user, - bool callerIsOwner, - uint256 rolesToGrant, - uint256 rolesToCheck - ) public { - vm.assume(user != address(this)); - - mockOwnableRoles.grantRoles(user, rolesToGrant); - - if ((rolesToGrant & rolesToCheck == 0) && !callerIsOwner) { - vm.expectRevert(Ownable.Unauthorized.selector); - } - if (!callerIsOwner) { - vm.prank(user); - } - mockOwnableRoles.updateFlagWithOnlyRolesOrOwner(rolesToCheck); - } - - function testOnlyOwnerOrRolesModifier() public { - testOnlyOwnerOrRolesModifier(address(1), false, 1, 2); - } - - function testHandoverOwnership(address pendingOwner) public { - vm.prank(pendingOwner); - vm.expectEmit(true, true, true, true); - emit OwnershipHandoverRequested(pendingOwner); - mockOwnableRoles.requestOwnershipHandover(); - assertTrue(mockOwnableRoles.ownershipHandoverExpiresAt(pendingOwner) > block.timestamp); - - vm.expectEmit(true, true, true, true); - emit OwnershipTransferred(address(this), pendingOwner); - - mockOwnableRoles.completeOwnershipHandover(pendingOwner); - - assertEq(mockOwnableRoles.owner(), pendingOwner); - } - - function testHandoverOwnership() public { - testHandoverOwnership(address(1)); - } - - function testHandoverOwnershipRevertsIfCompleteIsNotOwner() public { - address pendingOwner = address(1); - vm.prank(pendingOwner); - mockOwnableRoles.requestOwnershipHandover(); - - vm.prank(pendingOwner); - vm.expectRevert(Ownable.Unauthorized.selector); - mockOwnableRoles.completeOwnershipHandover(pendingOwner); - } - - function testHandoverOwnershipWithCancellation() public { - address pendingOwner = address(1); - - vm.prank(pendingOwner); - vm.expectEmit(true, true, true, true); - emit OwnershipHandoverRequested(pendingOwner); - mockOwnableRoles.requestOwnershipHandover(); - assertTrue(mockOwnableRoles.ownershipHandoverExpiresAt(pendingOwner) > block.timestamp); - - vm.expectEmit(true, true, true, true); - emit OwnershipHandoverCanceled(pendingOwner); - vm.prank(pendingOwner); - mockOwnableRoles.cancelOwnershipHandover(); - assertEq(mockOwnableRoles.ownershipHandoverExpiresAt(pendingOwner), 0); - vm.expectRevert(Ownable.NoHandoverRequest.selector); - - mockOwnableRoles.completeOwnershipHandover(pendingOwner); - } - - function testHandoverOwnershipBeforeExpiration() public { - address pendingOwner = address(1); - vm.prank(pendingOwner); - mockOwnableRoles.requestOwnershipHandover(); - - vm.warp(block.timestamp + mockOwnableRoles.ownershipHandoverValidFor()); - - mockOwnableRoles.completeOwnershipHandover(pendingOwner); - } - - function testHandoverOwnershipAfterExpiration() public { - address pendingOwner = address(1); - vm.prank(pendingOwner); - mockOwnableRoles.requestOwnershipHandover(); - - vm.warp(block.timestamp + mockOwnableRoles.ownershipHandoverValidFor() + 1); - - vm.expectRevert(Ownable.NoHandoverRequest.selector); - - mockOwnableRoles.completeOwnershipHandover(pendingOwner); - } - - function testOwnershipHandoverValidForDefaultValue() public { - assertEq(mockOwnableRoles.ownershipHandoverValidFor(), 48 * 3600); - } -} diff --git a/lib/solady/test/README.md b/lib/solady/test/README.md deleted file mode 100644 index 17268ea..0000000 --- a/lib/solady/test/README.md +++ /dev/null @@ -1,5 +0,0 @@ -## WARNING! - -All test files are strictly intended for testing purposes only. - -Do NOT copy anything here into production code unless you really know what you are doing. \ No newline at end of file diff --git a/lib/solady/test/Receiver.t.sol b/lib/solady/test/Receiver.t.sol deleted file mode 100644 index 68fa005..0000000 --- a/lib/solady/test/Receiver.t.sol +++ /dev/null @@ -1,43 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {Receiver} from "../src/accounts/Receiver.sol"; -import {MockERC721} from "./utils/mocks/MockERC721.sol"; -import {MockERC1155} from "./utils/mocks/MockERC1155.sol"; -import {MockReceiver} from "./utils/mocks/MockReceiver.sol"; - -contract ReceiverTest is SoladyTest { - MockERC721 immutable erc721 = new MockERC721(); - MockERC1155 immutable erc1155 = new MockERC1155(); - MockReceiver immutable receiver = new MockReceiver(); - address immutable alice = address(bytes20("milady")); - - function setUp() public {} - - function testETHReceived() public { - payable(address(receiver)).transfer(1 ether); - } - - function testOnERC721Received() public { - erc721.mint(alice, 1); - vm.prank(alice); - erc721.safeTransferFrom(alice, address(receiver), 1); - } - - function testOnERC1155Received() public { - erc1155.mint(alice, 1, 1, ""); - vm.prank(alice); - erc1155.safeTransferFrom(alice, address(receiver), 1, 1, ""); - } - - function testOnERC1155BatchReceived() public { - erc1155.mint(alice, 1, 1, ""); - uint256[] memory ids = new uint256[](1); - ids[0] = 1; - uint256[] memory amts = new uint256[](1); - amts[0] = 1; - vm.prank(alice); - erc1155.safeBatchTransferFrom(alice, address(receiver), ids, amts, ""); - } -} diff --git a/lib/solady/test/RedBlackTree.t.sol b/lib/solady/test/RedBlackTree.t.sol deleted file mode 100644 index b3e83ea..0000000 --- a/lib/solady/test/RedBlackTree.t.sol +++ /dev/null @@ -1,570 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {LibSort} from "../src/utils/LibSort.sol"; -import {LibPRNG} from "../src/utils/LibPRNG.sol"; -import {RedBlackTreeLib} from "../src/utils/RedBlackTreeLib.sol"; - -contract RedBlackTreeLibTest is SoladyTest { - using RedBlackTreeLib for *; - using LibPRNG for *; - - RedBlackTreeLib.Tree tree; - RedBlackTreeLib.Tree tree2; - - function testRedBlackTreeInsertBenchStep() public { - unchecked { - LibPRNG.PRNG memory prng = LibPRNG.PRNG(123); - uint256 n = 128; - uint256 m = (1 << 160) - 1; - for (uint256 i; i != n; ++i) { - uint256 r = 1 | (prng.next() & m); - tree.insert(r); - } - _testIterateTree(); - } - } - - function testRedBlackTreeInsertBenchUint160() public { - unchecked { - LibPRNG.PRNG memory prng = LibPRNG.PRNG(123); - uint256 n = 128; - uint256[] memory a = _makeArray(n); - uint256 m = (1 << 160) - 1; - for (uint256 i; i != n; ++i) { - uint256 r = 1 | (prng.next() & m); - a[i] = r; - tree.insert(r); - } - } - } - - function testRedBlackTreeBenchUint160() public { - unchecked { - LibPRNG.PRNG memory prng = LibPRNG.PRNG(123); - uint256 n = 128; - uint256[] memory a = _makeArray(n); - uint256 m = (1 << 160) - 1; - for (uint256 i; i != n; ++i) { - uint256 r = 1 | (prng.next() & m); - a[i] = r; - tree.insert(r); - } - prng.shuffle(a); - for (uint256 i; i != n; ++i) { - tree.remove(a[i]); - } - assertEq(tree.size(), 0); - } - } - - function testRedBlackTreeInsertBenchUint256() public { - unchecked { - LibPRNG.PRNG memory prng = LibPRNG.PRNG(123); - uint256 n = 128; - uint256[] memory a = _makeArray(n); - for (uint256 i; i != n; ++i) { - uint256 r = 1 | prng.next(); - a[i] = r; - tree.insert(r); - } - } - } - - function testRedBlackTreeBenchUint256() public { - unchecked { - LibPRNG.PRNG memory prng = LibPRNG.PRNG(123); - uint256 n = 128; - uint256[] memory a = _makeArray(n); - for (uint256 i; i != n; ++i) { - uint256 r = 1 | prng.next(); - a[i] = r; - tree.insert(r); - } - prng.shuffle(a); - for (uint256 i; i != n; ++i) { - tree.remove(a[i]); - } - assertEq(tree.size(), 0); - } - } - - function testRedBlackTreeInsertAndRemove(uint256) public { - unchecked { - for (uint256 t; t < 2; ++t) { - _testRedBlackTreeInsertAndRemove(); - } - } - } - - function _testRemoveAndInsertBack(uint256[] memory a, uint256 n, uint256 t) internal { - unchecked { - uint256 choice = a[_random() % n]; - bytes32 ptr = tree.find(choice); - bool exists = !ptr.isEmpty(); - if (exists) { - assertEq(ptr.value(), choice); - ptr.remove(); - if (_random() % 4 == 0) tree.tryRemove(choice); - assertTrue(tree.find(choice).isEmpty()); - assertFalse(tree.exists(choice)); - } - if (t != 0) { - _testRemoveAndInsertBack(a, n, t - 1); - } - if (exists) { - tree.insert(choice); - if (_random() % 4 == 0) tree.tryInsert(choice); - assertFalse(tree.find(choice).isEmpty()); - assertTrue(tree.exists(choice)); - } - } - } - - function _testIterateTree() internal { - bytes32 ptr = tree.first(); - uint256 prevValue; - while (!ptr.isEmpty()) { - uint256 v = ptr.value(); - assertTrue(prevValue < v); - prevValue = v; - ptr = ptr.next(); - } - assertEq(ptr.next().value(), 0); - - ptr = tree.last(); - prevValue = 0; - while (!ptr.isEmpty()) { - uint256 v = ptr.value(); - assertTrue(prevValue == 0 || prevValue > v); - prevValue = v; - ptr = ptr.prev(); - } - assertEq(ptr.prev().value(), 0); - } - - function _testRedBlackTreeInsertAndRemove() internal { - uint256 n = _random() % (_random() % 128 == 0 ? 32 : 8); - uint256[] memory a = _fillTree(n); - - LibSort.sort(a); - LibSort.uniquifySorted(a); - assertEq(a.length, n); - assertEq(tree.size(), n); - - assertEq(tree2.size(), 0); - - unchecked { - uint256 i; - bytes32 ptr = tree.first(); - while (!ptr.isEmpty()) { - assertEq(a[i++], ptr.value()); - ptr = ptr.next(); - } - assertEq(ptr.next().value(), 0); - } - - unchecked { - uint256 i = n; - bytes32 ptr = tree.last(); - while (!ptr.isEmpty()) { - assertEq(a[--i], ptr.value()); - ptr = ptr.prev(); - } - assertEq(ptr.prev().value(), 0); - } - - _testIterateTree(); - - LibPRNG.PRNG memory prng = LibPRNG.PRNG(_random()); - prng.shuffle(a); - - unchecked { - uint256 m = n < 8 ? 4 : n; - for (uint256 i; i != n; ++i) { - tree.remove(a[i]); - assertEq(tree.size(), n - i - 1); - if (_random() % m == 0) { - _testIterateTree(); - } - } - } - assertEq(tree.size(), 0); - - unchecked { - if (_random() % 2 == 0) { - for (uint256 i; i != n; ++i) { - assertTrue(tree.find(a[i]).isEmpty()); - } - } - assertTrue(tree.first().isEmpty()); - assertEq(tree.first().value(), 0); - assertTrue(tree.last().isEmpty()); - assertEq(tree.last().value(), 0); - } - - assertEq(tree2.size(), 0); - } - - function testRedBlackTreeInsertAndRemove2(uint256) public { - unchecked { - uint256 n = _random() % 2 == 0 ? 16 : 32; - uint256[] memory candidates = _makeArray(n); - for (uint256 i; i != n; ++i) { - candidates[i] = _bound(_random(), 1, type(uint256).max); - } - uint256[] memory records = _makeArray(0); - uint256 mode = 0; - for (uint256 t = _random() % 32 + 1; t != 0; --t) { - uint256 r = candidates[_random() % n]; - bytes32 ptr = tree.find(r); - if (mode == 0) { - if (ptr.isEmpty()) { - tree.insert(r); - _addToArray(records, r); - } - } else { - if (!ptr.isEmpty()) { - tree.remove(r); - _removeFromArray(records, r); - } - } - if (_random() % 3 == 0) mode = _random() % 2; - } - LibSort.sort(records); - assertEq(tree.size(), records.length); - - assertEq(tree2.size(), 0); - - { - uint256 i = 0; - bytes32 ptr = tree.first(); - while (!ptr.isEmpty()) { - assertEq(records[i++], ptr.value()); - ptr = ptr.next(); - } - assertEq(ptr.next().value(), 0); - } - } - } - - function _makeArray(uint256 size, uint256 maxCap) - internal - pure - returns (uint256[] memory result) - { - /// @solidity memory-safe-assembly - assembly { - result := mload(0x40) - mstore(result, size) - mstore(0x40, add(result, shl(5, add(maxCap, 1)))) - } - } - - function _makeArray(uint256 size) internal pure returns (uint256[] memory result) { - require(size <= 512, "Size too big."); - result = _makeArray(size, 512); - } - - function _addToArray(uint256[] memory a, uint256 x) internal pure { - /// @solidity memory-safe-assembly - assembly { - let exists := 0 - let n := mload(a) - for { let i := 0 } lt(i, n) { i := add(i, 1) } { - let o := add(add(a, 0x20), shl(5, i)) - if eq(mload(o), x) { - exists := 1 - break - } - } - if iszero(exists) { - n := add(n, 1) - mstore(add(a, shl(5, n)), x) - mstore(a, n) - } - } - } - - function _removeFromArray(uint256[] memory a, uint256 x) internal pure { - /// @solidity memory-safe-assembly - assembly { - let n := mload(a) - for { let i := 0 } lt(i, n) { i := add(i, 1) } { - let o := add(add(a, 0x20), shl(5, i)) - if eq(mload(o), x) { - mstore(o, mload(add(a, shl(5, n)))) - mstore(a, sub(n, 1)) - break - } - } - } - } - - function testRedBlackTreeInsertAndRemove3() public { - unchecked { - uint256 m = type(uint256).max; - for (uint256 i; i < 256; ++i) { - tree.insert(m - i); - assertEq(tree.size(), i + 1); - } - for (uint256 i; i < 256; ++i) { - tree2.insert(i + 1); - assertEq(tree2.size(), i + 1); - } - for (uint256 i; i < 256; ++i) { - assertTrue(tree.exists(m - i)); - assertFalse(tree.exists(i + 1)); - assertTrue(tree2.exists(i + 1)); - assertFalse(tree2.exists(m - i)); - } - bytes32[] memory ptrs = new bytes32[](256); - for (uint256 i; i < 256; ++i) { - bytes32 ptr = tree.find(m - i); - ptr.remove(); - assertTrue(ptr.value() != m - i); - ptrs[i] = ptr; - assertEq(tree.size(), 256 - (i + 1)); - } - for (uint256 i; i < 256; ++i) { - assertEq(ptrs[i].value(), 0); - vm.expectRevert(RedBlackTreeLib.PointerOutOfBounds.selector); - ptrs[i].remove(); - } - for (uint256 i; i < 256; ++i) { - tree2.remove(i + 1); - assertEq(tree2.size(), 256 - (i + 1)); - } - } - } - - function testRedBlackTreeRejectsEmptyValue() public { - vm.expectRevert(RedBlackTreeLib.ValueIsEmpty.selector); - tree.insert(0); - vm.expectRevert(RedBlackTreeLib.ValueIsEmpty.selector); - tree.remove(0); - vm.expectRevert(RedBlackTreeLib.ValueIsEmpty.selector); - tree.find(0); - } - - function testRedBlackTreeRemoveViaPointer() public { - tree.insert(1); - tree.insert(2); - - bytes32 ptr = tree.find(1); - ptr.remove(); - ptr.remove(); - - vm.expectRevert(RedBlackTreeLib.PointerOutOfBounds.selector); - ptr.remove(); - - ptr = bytes32(0); - vm.expectRevert(RedBlackTreeLib.ValueDoesNotExist.selector); - ptr.remove(); - } - - function testRedBlackTreeTryInsertAndRemove() public { - tree.tryInsert(1); - tree.tryInsert(2); - assertEq(tree.size(), 2); - tree.tryInsert(1); - assertEq(tree.size(), 2); - tree.tryRemove(2); - assertEq(tree.size(), 1); - tree.tryRemove(2); - assertEq(tree.size(), 1); - } - - function testRedBlackTreeClear() public { - tree.tryInsert(1); - tree.tryInsert(2); - bytes32 ptr1 = tree.find(1); - bytes32 ptr2 = tree.find(2); - assertEq(tree.size(), 2); - tree.clear(); - assertEq(tree.size(), 0); - assertEq(ptr1.value(), 0); - assertEq(ptr2.value(), 0); - } - - function testRedBlackTreeClear(uint256) public { - unchecked { - uint256 n = _random() % (_random() % 128 == 0 ? 32 : 8); - uint256[] memory a = _fillTree(n); - - bytes32[] memory ptrs = new bytes32[](n); - for (uint256 i; i != n; ++i) { - ptrs[i] = tree.find(a[i]); - assertTrue(ptrs[i].value() != 0); - } - tree.clear(); - assertEq(tree.size(), 0); - for (uint256 i; i != n; ++i) { - assertEq(ptrs[i].value(), 0); - } - } - } - - function testRedBlackTreeTreeFullReverts() public { - tree.insert(1); - bytes32 ptr = tree.find(1); - /// @solidity memory-safe-assembly - assembly { - ptr := shl(32, shr(32, ptr)) - sstore(ptr, or(sload(ptr), sub(shl(31, 1), 1))) - } - vm.expectRevert(RedBlackTreeLib.TreeIsFull.selector); - tree.insert(2); - assertEq(tree.size(), 2 ** 31 - 1); - } - - function testRedBlackTreePointers() public { - assertTrue(tree.find(1).isEmpty()); - assertTrue(tree.find(2).isEmpty()); - - tree.insert(1); - tree.insert(2); - - assertFalse(tree.find(1).isEmpty()); - assertFalse(tree.find(2).isEmpty()); - - assertTrue(tree.find(1).prev().isEmpty()); - assertFalse(tree.find(1).next().isEmpty()); - - assertFalse(tree.find(2).prev().isEmpty()); - assertTrue(tree.find(2).next().isEmpty()); - - assertEq(tree.find(1).next(), tree.find(2)); - assertEq(tree.find(1), tree.find(2).prev()); - - assertTrue(tree.find(1).prev().isEmpty()); - assertTrue(tree.find(1).prev().prev().isEmpty()); - assertTrue(tree.find(1).prev().next().isEmpty()); - - assertTrue(tree.find(2).next().isEmpty()); - assertTrue(tree.find(2).next().next().isEmpty()); - assertTrue(tree.find(2).next().prev().isEmpty()); - - assertEq(tree.first(), tree.find(1)); - assertEq(tree.last(), tree.find(2)); - - assertTrue(tree.find(3).isEmpty()); - } - - function testRedBlackTreeNearest(uint256) public { - assertEq(tree.nearest(1), bytes32(0)); - uint256[] memory a = _fillTree(_random() % 8); - uint256 x = _bound(_random(), 1, type(uint256).max); - (uint256 nearestIndex, bool found) = _nearestIndex(a, x); - if (found) { - assertEq(tree.nearest(x).value(), a[nearestIndex]); - } else { - assertEq(tree.nearest(x), bytes32(0)); - } - } - - function _nearestIndex(uint256[] memory a, uint256 x) - internal - pure - returns (uint256 nearestIndex, bool found) - { - unchecked { - uint256 nearestValue = type(uint256).max; - uint256 nearestDist = type(uint256).max; - uint256 n = a.length; - for (uint256 i; i != n; ++i) { - uint256 y = a[i]; - uint256 dist = x < y ? y - x : x - y; - if (dist < nearestDist || (dist == nearestDist && y < nearestValue)) { - nearestIndex = i; - nearestValue = y; - nearestDist = dist; - found = true; - } - } - } - } - - function testRedBlackTreeNearestBefore(uint256) public { - assertEq(tree.nearestBefore(1), bytes32(0)); - uint256[] memory a = _fillTree(_random() % 8); - uint256 x = _bound(_random(), 1, type(uint256).max); - (uint256 nearestIndexBefore, bool found) = _nearestIndexBefore(a, x); - if (found) { - assertEq(tree.nearestBefore(x).value(), a[nearestIndexBefore]); - } else { - assertEq(tree.nearestBefore(x), bytes32(0)); - } - } - - function _nearestIndexBefore(uint256[] memory a, uint256 x) - internal - pure - returns (uint256 nearestIndex, bool found) - { - unchecked { - uint256 nearestDist = type(uint256).max; - uint256 n = a.length; - for (uint256 i; i != n; ++i) { - uint256 y = a[i]; - if (y > x) continue; - uint256 dist = x - y; - if (dist < nearestDist) { - nearestIndex = i; - nearestDist = dist; - found = true; - } - } - } - } - - function testRedBlackTreeNearestAfter(uint256) public { - assertEq(tree.nearestAfter(1), bytes32(0)); - uint256[] memory a = _fillTree(_random() % 8); - uint256 x = _bound(_random(), 1, type(uint256).max); - (uint256 nearestIndexAfter, bool found) = _nearestIndexAfter(a, x); - if (found) { - assertEq(tree.nearestAfter(x).value(), a[nearestIndexAfter]); - } else { - assertEq(tree.nearestAfter(x), bytes32(0)); - } - } - - function _nearestIndexAfter(uint256[] memory a, uint256 x) - internal - pure - returns (uint256 nearestIndex, bool found) - { - unchecked { - uint256 nearestDist = type(uint256).max; - uint256 n = a.length; - for (uint256 i; i != n; ++i) { - uint256 y = a[i]; - if (y < x) continue; - uint256 dist = y - x; - if (dist < nearestDist) { - nearestIndex = i; - nearestDist = dist; - found = true; - } - } - } - } - - function _fillTree(uint256 n) internal returns (uint256[] memory a) { - a = _makeArray(n); - unchecked { - for (uint256 i; i != n;) { - uint256 r = _bound(_random(), 1, type(uint256).max); - if (tree.find(r).isEmpty()) { - a[i++] = r; - tree.insert(r); - } - if (_random() % 4 == 0) { - _testRemoveAndInsertBack(a, i, (3 + i >> 2)); - } - } - } - } -} diff --git a/lib/solady/test/SSTORE2.t.sol b/lib/solady/test/SSTORE2.t.sol deleted file mode 100644 index 411f28a..0000000 --- a/lib/solady/test/SSTORE2.t.sol +++ /dev/null @@ -1,200 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {SSTORE2} from "../src/utils/SSTORE2.sol"; - -contract SSTORE2Test is SoladyTest { - function testWriteRead() public { - bytes memory testBytes = abi.encode("this is a test"); - - address pointer = SSTORE2.write(testBytes); - - assertEq(SSTORE2.read(pointer), testBytes); - } - - function testWriteReadFullStartBound() public { - assertEq(SSTORE2.read(SSTORE2.write(hex"11223344"), 0), hex"11223344"); - } - - function testWriteReadCustomStartBound() public { - assertEq(SSTORE2.read(SSTORE2.write(hex"11223344"), 1), hex"223344"); - } - - function testWriteReadFullBoundedRead() public { - bytes memory testBytes = abi.encode("this is a test"); - - assertEq(SSTORE2.read(SSTORE2.write(testBytes), 0, testBytes.length), testBytes); - } - - function testWriteReadCustomBounds() public { - assertEq(SSTORE2.read(SSTORE2.write(hex"11223344"), 1, 3), hex"2233"); - } - - function testWriteReadEmptyBound() public { - SSTORE2.read(SSTORE2.write(hex"11223344"), 3, 3); - } - - function testReadInvalidPointerReverts() public { - vm.expectRevert(SSTORE2.InvalidPointer.selector); - SSTORE2.read(address(1)); - } - - function testReadInvalidPointerCustomStartBoundReverts() public { - vm.expectRevert(SSTORE2.InvalidPointer.selector); - SSTORE2.read(address(1), 1); - } - - function testReadInvalidPointerCustomBoundsReverts() public { - vm.expectRevert(SSTORE2.InvalidPointer.selector); - SSTORE2.read(address(1), 2, 4); - } - - function testWriteReadOutOfStartBoundReverts() public { - address pointer = SSTORE2.write(hex"11223344"); - vm.expectRevert(SSTORE2.ReadOutOfBounds.selector); - SSTORE2.read(pointer, 41000); - } - - function testWriteReadEmptyOutOfBoundsReverts() public { - address pointer = SSTORE2.write(hex"11223344"); - vm.expectRevert(SSTORE2.ReadOutOfBounds.selector); - SSTORE2.read(pointer, 42000, 42000); - } - - function testWriteReadOutOfBoundsReverts() public { - address pointer = SSTORE2.write(hex"11223344"); - vm.expectRevert(SSTORE2.ReadOutOfBounds.selector); - SSTORE2.read(pointer, 41000, 42000); - } - - function testWriteRead(bytes calldata testBytes) public brutalizeMemory { - _misalignFreeMemoryPointer(); - bytes memory readResult = SSTORE2.read(SSTORE2.write(testBytes)); - _checkMemory(readResult); - assertEq(readResult, testBytes); - } - - function testWriteReadCustomStartBound(bytes calldata testBytes, uint256 startIndex) - public - brutalizeMemory - { - if (testBytes.length == 0) return; - - startIndex = _bound(startIndex, 0, testBytes.length); - - _misalignFreeMemoryPointer(); - bytes memory readResult = SSTORE2.read(SSTORE2.write(testBytes), startIndex); - _checkMemory(readResult); - assertEq(readResult, bytes(testBytes[startIndex:])); - } - - function testWriteReadCustomBounds( - bytes calldata testBytes, - uint256 startIndex, - uint256 endIndex - ) public brutalizeMemory { - if (testBytes.length == 0) return; - - endIndex = _bound(endIndex, 0, testBytes.length); - startIndex = _bound(startIndex, 0, testBytes.length); - - if (startIndex > endIndex) return; - - _misalignFreeMemoryPointer(); - bytes memory readResult = SSTORE2.read(SSTORE2.write(testBytes), startIndex, endIndex); - _checkMemory(readResult); - assertEq(readResult, bytes(testBytes[startIndex:endIndex])); - } - - function testReadInvalidPointerRevert(address pointer) public brutalizeMemory { - if (pointer.code.length > 0) return; - vm.expectRevert(SSTORE2.InvalidPointer.selector); - SSTORE2.read(pointer); - } - - function testReadInvalidPointerCustomStartBoundReverts(address pointer, uint256 startIndex) - public - brutalizeMemory - { - if (pointer.code.length > 0) return; - vm.expectRevert(SSTORE2.InvalidPointer.selector); - SSTORE2.read(pointer, startIndex); - } - - function testReadInvalidPointerCustomBoundsReverts( - address pointer, - uint256 startIndex, - uint256 endIndex - ) public brutalizeMemory { - if (pointer.code.length > 0) return; - vm.expectRevert(SSTORE2.InvalidPointer.selector); - SSTORE2.read(pointer, startIndex, endIndex); - } - - function testWriteReadCustomStartBoundOutOfRangeReverts( - bytes calldata testBytes, - uint256 startIndex - ) public brutalizeMemory { - startIndex = _bound(startIndex, testBytes.length + 1, type(uint256).max); - address pointer = SSTORE2.write(testBytes); - vm.expectRevert(SSTORE2.ReadOutOfBounds.selector); - SSTORE2.read(pointer, startIndex); - } - - function testWriteReadCustomBoundsOutOfRangeReverts( - bytes calldata testBytes, - uint256 startIndex, - uint256 endIndex - ) public brutalizeMemory { - endIndex = _bound(endIndex, testBytes.length + 1, type(uint256).max); - address pointer = SSTORE2.write(testBytes); - vm.expectRevert(SSTORE2.ReadOutOfBounds.selector); - SSTORE2.read(pointer, startIndex, endIndex); - } - - function testWriteReadDeterministic(bytes calldata testBytes) public brutalizeMemory { - bytes32 salt = bytes32(_random()); - address deployer = address(this); - if (_random() % 8 == 0) { - (deployer,) = _randomSigner(); - } - vm.prank(deployer); - address deterministicPointer = SSTORE2.writeDeterministic(testBytes, salt); - assertEq(SSTORE2.read(deterministicPointer), testBytes); - assertEq( - SSTORE2.predictDeterministicAddress(testBytes, salt, deployer), deterministicPointer - ); - - address pointer = SSTORE2.write(testBytes); - assertEq(pointer.code, deterministicPointer.code); - } - - function testWriteWithTooBigDataReverts() public { - bytes memory data = _dummyData(0xfffe); - address pointer = this.write(data); - assertEq(SSTORE2.read(pointer), data); - vm.expectRevert(); - pointer = this.write(_dummyData(0xffff)); - } - - function write(bytes memory data) public returns (address) { - return SSTORE2.write(data); - } - - function _dummyData(uint256 n) internal pure returns (bytes memory result) { - /// @solidity memory-safe-assembly - assembly { - result := mload(0x40) - mstore(result, n) - mstore(0x00, n) - mstore(0x20, 1) - mstore(add(0x20, result), keccak256(0x00, 0x40)) - mstore(0x20, 2) - mstore(add(add(0x20, result), n), keccak256(0x00, 0x40)) - mstore(0x20, 3) - mstore(add(result, n), keccak256(0x00, 0x40)) - mstore(0x40, add(add(0x20, result), n)) - } - } -} diff --git a/lib/solady/test/SafeCastLib.t.sol b/lib/solady/test/SafeCastLib.t.sol deleted file mode 100644 index 5b17145..0000000 --- a/lib/solady/test/SafeCastLib.t.sol +++ /dev/null @@ -1,307 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {SafeCastLib} from "../src/utils/SafeCastLib.sol"; - -contract SafeCastLibTest is SoladyTest { - function testSafeCastToUint(uint256 x) public { - assertEq(SafeCastLib.toUint8(uint8(x)), uint8(x)); - if (x >= (1 << 8)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint8(x); - assertEq(SafeCastLib.toUint16(uint16(x)), uint16(x)); - if (x >= (1 << 16)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint16(x); - assertEq(SafeCastLib.toUint24(uint24(x)), uint24(x)); - if (x >= (1 << 24)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint24(x); - assertEq(SafeCastLib.toUint32(uint32(x)), uint32(x)); - if (x >= (1 << 32)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint32(x); - assertEq(SafeCastLib.toUint40(uint40(x)), uint40(x)); - if (x >= (1 << 40)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint40(x); - assertEq(SafeCastLib.toUint48(uint48(x)), uint48(x)); - if (x >= (1 << 48)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint48(x); - assertEq(SafeCastLib.toUint56(uint56(x)), uint56(x)); - if (x >= (1 << 56)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint56(x); - assertEq(SafeCastLib.toUint64(uint64(x)), uint64(x)); - if (x >= (1 << 64)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint64(x); - assertEq(SafeCastLib.toUint72(uint72(x)), uint72(x)); - if (x >= (1 << 72)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint72(x); - assertEq(SafeCastLib.toUint80(uint80(x)), uint80(x)); - if (x >= (1 << 80)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint80(x); - assertEq(SafeCastLib.toUint88(uint88(x)), uint88(x)); - if (x >= (1 << 88)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint88(x); - assertEq(SafeCastLib.toUint96(uint96(x)), uint96(x)); - if (x >= (1 << 96)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint96(x); - assertEq(SafeCastLib.toUint104(uint104(x)), uint104(x)); - if (x >= (1 << 104)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint104(x); - assertEq(SafeCastLib.toUint112(uint112(x)), uint112(x)); - if (x >= (1 << 112)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint112(x); - assertEq(SafeCastLib.toUint120(uint120(x)), uint120(x)); - if (x >= (1 << 120)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint120(x); - assertEq(SafeCastLib.toUint128(uint128(x)), uint128(x)); - if (x >= (1 << 128)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint128(x); - assertEq(SafeCastLib.toUint136(uint136(x)), uint136(x)); - if (x >= (1 << 136)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint136(x); - assertEq(SafeCastLib.toUint144(uint144(x)), uint144(x)); - if (x >= (1 << 144)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint144(x); - assertEq(SafeCastLib.toUint152(uint152(x)), uint152(x)); - if (x >= (1 << 152)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint152(x); - assertEq(SafeCastLib.toUint160(uint160(x)), uint160(x)); - if (x >= (1 << 160)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint160(x); - assertEq(SafeCastLib.toUint168(uint168(x)), uint168(x)); - if (x >= (1 << 168)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint168(x); - assertEq(SafeCastLib.toUint176(uint176(x)), uint176(x)); - if (x >= (1 << 176)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint176(x); - assertEq(SafeCastLib.toUint184(uint184(x)), uint184(x)); - if (x >= (1 << 184)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint184(x); - assertEq(SafeCastLib.toUint192(uint192(x)), uint192(x)); - if (x >= (1 << 192)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint192(x); - assertEq(SafeCastLib.toUint200(uint200(x)), uint200(x)); - if (x >= (1 << 200)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint200(x); - assertEq(SafeCastLib.toUint208(uint208(x)), uint208(x)); - if (x >= (1 << 208)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint208(x); - assertEq(SafeCastLib.toUint216(uint216(x)), uint216(x)); - if (x >= (1 << 216)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint216(x); - assertEq(SafeCastLib.toUint224(uint224(x)), uint224(x)); - if (x >= (1 << 224)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint224(x); - assertEq(SafeCastLib.toUint232(uint232(x)), uint232(x)); - if (x >= (1 << 232)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint232(x); - assertEq(SafeCastLib.toUint240(uint240(x)), uint240(x)); - if (x >= (1 << 240)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint240(x); - assertEq(SafeCastLib.toUint248(uint248(x)), uint248(x)); - if (x >= (1 << 248)) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint248(x); - } - - function testSafeCastToUint() public { - unchecked { - for (uint256 i; i != 256; ++i) { - testSafeCastToUint(1 << i); - testSafeCastToUint(type(uint256).max >> i); - testSafeCastToUint(_random()); - } - } - } - - function testSafeCastToUintBench() public { - unchecked { - uint256 sum; - for (uint256 i; i != 127; ++i) { - sum += uint256(SafeCastLib.toUint8(i)); - sum += uint256(SafeCastLib.toUint16(i)); - sum += uint256(SafeCastLib.toUint24(i)); - sum += uint256(SafeCastLib.toUint32(i)); - sum += uint256(SafeCastLib.toUint40(i)); - sum += uint256(SafeCastLib.toUint48(i)); - sum += uint256(SafeCastLib.toUint56(i)); - sum += uint256(SafeCastLib.toUint64(i)); - sum += uint256(SafeCastLib.toUint72(i)); - sum += uint256(SafeCastLib.toUint80(i)); - sum += uint256(SafeCastLib.toUint88(i)); - sum += uint256(SafeCastLib.toUint96(i)); - sum += uint256(SafeCastLib.toUint104(i)); - sum += uint256(SafeCastLib.toUint112(i)); - sum += uint256(SafeCastLib.toUint120(i)); - sum += uint256(SafeCastLib.toUint128(i)); - sum += uint256(SafeCastLib.toUint136(i)); - sum += uint256(SafeCastLib.toUint144(i)); - sum += uint256(SafeCastLib.toUint152(i)); - sum += uint256(SafeCastLib.toUint160(i)); - sum += uint256(SafeCastLib.toUint168(i)); - sum += uint256(SafeCastLib.toUint176(i)); - sum += uint256(SafeCastLib.toUint184(i)); - sum += uint256(SafeCastLib.toUint192(i)); - sum += uint256(SafeCastLib.toUint200(i)); - sum += uint256(SafeCastLib.toUint208(i)); - sum += uint256(SafeCastLib.toUint216(i)); - sum += uint256(SafeCastLib.toUint224(i)); - sum += uint256(SafeCastLib.toUint232(i)); - sum += uint256(SafeCastLib.toUint240(i)); - sum += uint256(SafeCastLib.toUint248(i)); - } - assertTrue(sum > 100); - } - } - - function testSafeCastToInt(int256 x) public { - assertEq(SafeCastLib.toInt8(int8(x)), int8(x)); - if (int8(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt8(x); - assertEq(SafeCastLib.toInt16(int16(x)), int16(x)); - if (int16(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt16(x); - assertEq(SafeCastLib.toInt24(int24(x)), int24(x)); - if (int24(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt24(x); - assertEq(SafeCastLib.toInt32(int32(x)), int32(x)); - if (int32(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt32(x); - assertEq(SafeCastLib.toInt40(int40(x)), int40(x)); - if (int40(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt40(x); - assertEq(SafeCastLib.toInt48(int48(x)), int48(x)); - if (int48(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt48(x); - assertEq(SafeCastLib.toInt56(int56(x)), int56(x)); - if (int56(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt56(x); - assertEq(SafeCastLib.toInt64(int64(x)), int64(x)); - if (int64(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt64(x); - assertEq(SafeCastLib.toInt72(int72(x)), int72(x)); - if (int72(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt72(x); - assertEq(SafeCastLib.toInt80(int80(x)), int80(x)); - if (int80(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt80(x); - assertEq(SafeCastLib.toInt88(int88(x)), int88(x)); - if (int88(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt88(x); - assertEq(SafeCastLib.toInt96(int96(x)), int96(x)); - if (int96(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt96(x); - assertEq(SafeCastLib.toInt104(int104(x)), int104(x)); - if (int104(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt104(x); - assertEq(SafeCastLib.toInt112(int112(x)), int112(x)); - if (int112(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt112(x); - assertEq(SafeCastLib.toInt120(int120(x)), int120(x)); - if (int120(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt120(x); - assertEq(SafeCastLib.toInt128(int128(x)), int128(x)); - if (int128(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt128(x); - assertEq(SafeCastLib.toInt136(int136(x)), int136(x)); - if (int136(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt136(x); - assertEq(SafeCastLib.toInt144(int144(x)), int144(x)); - if (int144(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt144(x); - assertEq(SafeCastLib.toInt152(int152(x)), int152(x)); - if (int152(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt152(x); - assertEq(SafeCastLib.toInt160(int160(x)), int160(x)); - if (int160(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt160(x); - assertEq(SafeCastLib.toInt168(int168(x)), int168(x)); - if (int168(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt168(x); - assertEq(SafeCastLib.toInt176(int176(x)), int176(x)); - if (int176(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt176(x); - assertEq(SafeCastLib.toInt184(int184(x)), int184(x)); - if (int184(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt184(x); - assertEq(SafeCastLib.toInt192(int192(x)), int192(x)); - if (int192(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt192(x); - assertEq(SafeCastLib.toInt200(int200(x)), int200(x)); - if (int200(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt200(x); - assertEq(SafeCastLib.toInt208(int208(x)), int208(x)); - if (int208(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt208(x); - assertEq(SafeCastLib.toInt216(int216(x)), int216(x)); - if (int216(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt216(x); - assertEq(SafeCastLib.toInt224(int224(x)), int224(x)); - if (int224(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt224(x); - assertEq(SafeCastLib.toInt232(int232(x)), int232(x)); - if (int232(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt232(x); - assertEq(SafeCastLib.toInt240(int240(x)), int240(x)); - if (int240(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt240(x); - assertEq(SafeCastLib.toInt248(int248(x)), int248(x)); - if (int248(x) != x) vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt248(x); - } - - function testSafeCastToInt256(uint256 x) public { - if (x > uint256(type(int256).max)) { - vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toInt256(x); - } else { - assertEq(SafeCastLib.toInt256(x), int256(x)); - } - } - - function testSafeCastToUint256(int256 x) public { - if (x < 0) { - vm.expectRevert(SafeCastLib.Overflow.selector); - SafeCastLib.toUint256(x); - } else { - assertEq(SafeCastLib.toUint256(x), uint256(x)); - } - } - - function testSafeCastToIntBench() public { - unchecked { - int256 sum; - for (int256 i; i != 127; ++i) { - sum += int256(SafeCastLib.toInt8(i)); - sum += int256(SafeCastLib.toInt16(i)); - sum += int256(SafeCastLib.toInt24(i)); - sum += int256(SafeCastLib.toInt32(i)); - sum += int256(SafeCastLib.toInt40(i)); - sum += int256(SafeCastLib.toInt48(i)); - sum += int256(SafeCastLib.toInt56(i)); - sum += int256(SafeCastLib.toInt64(i)); - sum += int256(SafeCastLib.toInt72(i)); - sum += int256(SafeCastLib.toInt80(i)); - sum += int256(SafeCastLib.toInt88(i)); - sum += int256(SafeCastLib.toInt96(i)); - sum += int256(SafeCastLib.toInt104(i)); - sum += int256(SafeCastLib.toInt112(i)); - sum += int256(SafeCastLib.toInt120(i)); - sum += int256(SafeCastLib.toInt128(i)); - sum += int256(SafeCastLib.toInt136(i)); - sum += int256(SafeCastLib.toInt144(i)); - sum += int256(SafeCastLib.toInt152(i)); - sum += int256(SafeCastLib.toInt160(i)); - sum += int256(SafeCastLib.toInt168(i)); - sum += int256(SafeCastLib.toInt176(i)); - sum += int256(SafeCastLib.toInt184(i)); - sum += int256(SafeCastLib.toInt192(i)); - sum += int256(SafeCastLib.toInt200(i)); - sum += int256(SafeCastLib.toInt208(i)); - sum += int256(SafeCastLib.toInt216(i)); - sum += int256(SafeCastLib.toInt224(i)); - sum += int256(SafeCastLib.toInt232(i)); - sum += int256(SafeCastLib.toInt240(i)); - sum += int256(SafeCastLib.toInt248(i)); - } - assertTrue(sum > 100); - } - } -} diff --git a/lib/solady/test/SafeTransferLib.t.sol b/lib/solady/test/SafeTransferLib.t.sol deleted file mode 100644 index 81617b8..0000000 --- a/lib/solady/test/SafeTransferLib.t.sol +++ /dev/null @@ -1,748 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {MockERC20} from "./utils/mocks/MockERC20.sol"; -import {MockERC20LikeUSDT} from "./utils/mocks/MockERC20LikeUSDT.sol"; -import {MockETHRecipient} from "./utils/mocks/MockETHRecipient.sol"; -import {RevertingToken} from "./utils/weird-tokens/RevertingToken.sol"; -import {ReturnsTwoToken} from "./utils/weird-tokens/ReturnsTwoToken.sol"; -import {ReturnsFalseToken} from "./utils/weird-tokens/ReturnsFalseToken.sol"; -import {MissingReturnToken} from "./utils/weird-tokens/MissingReturnToken.sol"; -import {ReturnsTooMuchToken} from "./utils/weird-tokens/ReturnsTooMuchToken.sol"; -import {ReturnsRawBytesToken} from "./utils/weird-tokens/ReturnsRawBytesToken.sol"; -import {ReturnsTooLittleToken} from "./utils/weird-tokens/ReturnsTooLittleToken.sol"; - -import "./utils/SoladyTest.sol"; - -import {ERC20} from "../src/tokens/ERC20.sol"; -import {SafeTransferLib} from "../src/utils/SafeTransferLib.sol"; - -contract SafeTransferLibTest is SoladyTest { - uint256 constant SUCCESS = 1; - uint256 constant REVERTS_WITH_SELECTOR = 2; - uint256 constant REVERTS_WITH_ANY = 3; - - RevertingToken reverting; - ReturnsTwoToken returnsTwo; - ReturnsFalseToken returnsFalse; - MissingReturnToken missingReturn; - ReturnsTooMuchToken returnsTooMuch; - ReturnsRawBytesToken returnsRawBytes; - ReturnsTooLittleToken returnsTooLittle; - - MockERC20 erc20; - - function setUp() public { - reverting = new RevertingToken(); - returnsTwo = new ReturnsTwoToken(); - returnsFalse = new ReturnsFalseToken(); - missingReturn = new MissingReturnToken(); - returnsTooMuch = new ReturnsTooMuchToken(); - returnsRawBytes = new ReturnsRawBytesToken(); - returnsTooLittle = new ReturnsTooLittleToken(); - - erc20 = new MockERC20("StandardToken", "ST", 18); - erc20.mint(address(this), type(uint256).max); - } - - function testTransferWithMissingReturn() public { - verifySafeTransfer(address(missingReturn), address(0xBEEF), 1e18, SUCCESS); - } - - function testTransferWithStandardERC20() public { - verifySafeTransfer(address(erc20), address(0xBEEF), 1e18, SUCCESS); - } - - function testTransferWithReturnsTooMuch() public { - verifySafeTransfer(address(returnsTooMuch), address(0xBEEF), 1e18, SUCCESS); - } - - function testTransferWithNonContract() public { - SafeTransferLib.safeTransfer(address(0xBADBEEF), address(0xBEEF), 1e18); - } - - function testTransferFromWithMissingReturn() public { - verifySafeTransferFrom( - address(missingReturn), address(0xFEED), address(0xBEEF), 1e18, SUCCESS - ); - } - - function testTransferFromWithStandardERC20() public { - verifySafeTransferFrom(address(erc20), address(0xFEED), address(0xBEEF), 1e18, SUCCESS); - } - - function testTransferFromWithReturnsTooMuch() public { - verifySafeTransferFrom( - address(returnsTooMuch), address(0xFEED), address(0xBEEF), 1e18, SUCCESS - ); - } - - function testTransferFromWithNonContract() public { - SafeTransferLib.safeTransferFrom(address(0xBADBEEF), address(0xFEED), address(0xBEEF), 1e18); - } - - function testApproveWithMissingReturn() public { - verifySafeApprove(address(missingReturn), address(0xBEEF), 1e18, SUCCESS); - } - - function testApproveWithStandardERC20() public { - verifySafeApprove(address(erc20), address(0xBEEF), 1e18, SUCCESS); - } - - function testApproveWithReturnsTooMuch() public { - verifySafeApprove(address(returnsTooMuch), address(0xBEEF), 1e18, SUCCESS); - } - - function testApproveWithNonContract() public { - SafeTransferLib.safeApprove(address(0xBADBEEF), address(0xBEEF), 1e18); - } - - function testApproveWithRetryWithNonContract() public { - SafeTransferLib.safeApproveWithRetry(address(0xBADBEEF), address(0xBEEF), 1e18); - } - - function testTransferETH() public { - SafeTransferLib.safeTransferETH(address(0xBEEF), 1e18); - } - - function testTransferAllETH() public { - SafeTransferLib.safeTransferAllETH(address(0xBEEF)); - } - - function testTryTransferETH() public { - MockETHRecipient recipient = new MockETHRecipient(false, false); - bool success = SafeTransferLib.trySafeTransferETH(address(recipient), 1e18, gasleft()); - assertTrue(success); - } - - function testTryTransferAllETH() public { - MockETHRecipient recipient = new MockETHRecipient(false, false); - bool success = SafeTransferLib.trySafeTransferAllETH(address(recipient), gasleft()); - assertTrue(success); - } - - function testTryTransferETHWithNoStorageWrites() public { - MockETHRecipient recipient = new MockETHRecipient(true, false); - - { - bool success = SafeTransferLib.trySafeTransferETH( - address(recipient), 1e18, SafeTransferLib.GAS_STIPEND_NO_STORAGE_WRITES - ); - assertFalse(success); - } - - { - uint256 counterBefore = recipient.counter(); - bool success = SafeTransferLib.trySafeTransferETH( - address(recipient), 1e18, SafeTransferLib.GAS_STIPEND_NO_GRIEF - ); - assertTrue(success); - assertEq(recipient.counter(), counterBefore + 1); - } - - { - uint256 counterBefore = recipient.counter(); - bool success = SafeTransferLib.trySafeTransferETH(address(recipient), 1e18, gasleft()); - assertTrue(success); - assertEq(recipient.counter(), counterBefore + 1); - } - } - - function testTryTransferETHWithNoGrief() public { - MockETHRecipient recipient = new MockETHRecipient(false, true); - - { - bool success = SafeTransferLib.trySafeTransferETH( - address(recipient), 1e18, SafeTransferLib.GAS_STIPEND_NO_STORAGE_WRITES - ); - assertFalse(success); - assertTrue(recipient.garbage() == 0); - } - - { - bool success = SafeTransferLib.trySafeTransferETH( - address(recipient), 1e18, SafeTransferLib.GAS_STIPEND_NO_GRIEF - ); - assertFalse(success); - assertTrue(recipient.garbage() == 0); - } - - { - bool success = SafeTransferLib.trySafeTransferETH(address(recipient), 1e18, gasleft()); - assertTrue(success); - assertTrue(recipient.garbage() != 0); - } - } - - function testForceTransferETHToGriever(uint256 amount, uint256 randomness) public { - amount = amount % 1000 ether; - uint256 originalBalance = address(this).balance; - vm.deal(address(this), amount * 2); - - MockETHRecipient recipient = new MockETHRecipient(false, true); - - { - uint256 receipientBalanceBefore = address(recipient).balance; - uint256 senderBalanceBefore = address(this).balance; - uint256 r = uint256(keccak256(abi.encode(randomness))) % 3; - // Send to a griever with a gas stipend. Should not revert. - if (r == 0) { - this.forceSafeTransferETH( - address(recipient), amount, SafeTransferLib.GAS_STIPEND_NO_STORAGE_WRITES - ); - } else if (r == 1) { - this.forceSafeTransferETH( - address(recipient), amount, SafeTransferLib.GAS_STIPEND_NO_GRIEF - ); - } else { - this.forceSafeTransferETH(address(recipient), amount); - } - assertEq(address(recipient).balance - receipientBalanceBefore, amount); - assertEq(senderBalanceBefore - address(this).balance, amount); - // We use the `SELFDESTRUCT` to send, and thus the `garbage` should NOT be updated. - assertTrue(recipient.garbage() == 0); - } - - { - uint256 receipientBalanceBefore = address(recipient).balance; - uint256 senderBalanceBefore = address(this).balance; - // Send more than remaining balance without gas stipend. Should revert. - vm.expectRevert(SafeTransferLib.ETHTransferFailed.selector); - this.forceSafeTransferETH(address(recipient), address(this).balance + 1, gasleft()); - assertEq(address(recipient).balance - receipientBalanceBefore, 0); - assertEq(senderBalanceBefore - address(this).balance, 0); - // We did not send anything, and thus the `garbage` should NOT be updated. - assertTrue(recipient.garbage() == 0); - } - - { - uint256 receipientBalanceBefore = address(recipient).balance; - uint256 senderBalanceBefore = address(this).balance; - // Send all the remaining balance without gas stipend. Should not revert. - amount = address(this).balance; - this.forceSafeTransferETH(address(recipient), amount, gasleft()); - assertEq(address(recipient).balance - receipientBalanceBefore, amount); - assertEq(senderBalanceBefore - address(this).balance, amount); - // We use the normal `CALL` to send, and thus the `garbage` should be updated. - assertTrue(recipient.garbage() != 0); - } - - vm.deal(address(this), originalBalance); - } - - function testForceTransferETHToGriever() public { - testForceTransferETHToGriever(1 ether, 0); - testForceTransferETHToGriever(1 ether, 1); - testForceTransferETHToGriever(1 ether, 2); - } - - function testTransferWithReturnsFalseReverts() public { - verifySafeTransfer(address(returnsFalse), address(0xBEEF), 1e18, REVERTS_WITH_SELECTOR); - } - - function testTransferWithRevertingReverts() public { - verifySafeTransfer(address(reverting), address(0xBEEF), 1e18, REVERTS_WITH_SELECTOR); - } - - function testTransferWithReturnsTooLittleReverts() public { - verifySafeTransfer(address(returnsTooLittle), address(0xBEEF), 1e18, REVERTS_WITH_SELECTOR); - } - - function testTransferFromWithReturnsFalseReverts() public { - verifySafeTransferFrom( - address(returnsFalse), address(0xFEED), address(0xBEEF), 1e18, REVERTS_WITH_SELECTOR - ); - } - - function testTransferFromWithRevertingReverts() public { - verifySafeTransferFrom( - address(reverting), address(0xFEED), address(0xBEEF), 1e18, REVERTS_WITH_ANY - ); - } - - function testTransferFromWithReturnsTooLittleReverts() public { - verifySafeTransferFrom( - address(returnsTooLittle), address(0xFEED), address(0xBEEF), 1e18, REVERTS_WITH_SELECTOR - ); - } - - function testApproveWithReturnsFalseReverts() public { - verifySafeApprove(address(returnsFalse), address(0xBEEF), 1e18, REVERTS_WITH_SELECTOR); - } - - function testApproveWithRevertingReverts() public { - verifySafeApprove(address(reverting), address(0xBEEF), 1e18, REVERTS_WITH_SELECTOR); - } - - function testApproveWithReturnsTooLittleReverts() public { - verifySafeApprove(address(returnsTooLittle), address(0xBEEF), 1e18, REVERTS_WITH_SELECTOR); - } - - function testBalanceOfStandardERC20() public view { - erc20.balanceOf(address(this)); - } - - function testBalanceOfStandardERC20(address to, uint256 amount) public { - uint256 originalBalance = erc20.balanceOf(address(this)); - vm.assume(originalBalance >= amount); - vm.assume(to != address(this)); - - SafeTransferLib.safeTransfer(address(erc20), _brutalized(to), originalBalance - amount); - assertEq(SafeTransferLib.balanceOf(address(erc20), _brutalized(address(this))), amount); - } - - function testTransferAllWithStandardERC20() public { - SafeTransferLib.safeTransferAll(address(erc20), address(1)); - } - - function testTransferAllWithStandardERC20(address to, uint256 amount) public { - uint256 originalBalance = erc20.balanceOf(address(this)); - vm.assume(originalBalance >= amount); - vm.assume(to != address(this)); - - SafeTransferLib.safeTransfer(address(erc20), _brutalized(to), originalBalance - amount); - assertEq(erc20.balanceOf(address(this)), amount); - - assertEq(SafeTransferLib.safeTransferAll(address(erc20), _brutalized(to)), amount); - - assertEq(erc20.balanceOf(address(this)), 0); - assertEq(erc20.balanceOf(to), originalBalance); - } - - function testTransferAllFromWithStandardERC20() public { - forceApprove(address(erc20), address(this), address(this), type(uint256).max); - SafeTransferLib.safeTransferAllFrom(address(erc20), address(this), address(1)); - } - - function testTransferAllFromWithStandardERC20(address to, address from, uint256 amount) - public - { - SafeTransferLib.safeTransferAll(address(erc20), _brutalized(from)); - - uint256 originalBalance = erc20.balanceOf(from); - vm.assume(originalBalance >= amount); - vm.assume(to != from && to != address(this) && from != address(this)); - - forceApprove(address(erc20), from, address(this), type(uint256).max); - - SafeTransferLib.safeTransferFrom( - address(erc20), _brutalized(from), _brutalized(to), originalBalance - amount - ); - assertEq(erc20.balanceOf(from), amount); - - assertEq( - SafeTransferLib.safeTransferAllFrom(address(erc20), _brutalized(from), _brutalized(to)), - amount - ); - - assertEq(erc20.balanceOf(address(this)), 0); - assertEq(erc20.balanceOf(to), originalBalance); - } - - function testTransferWithMissingReturn(address to, uint256 amount) public { - verifySafeTransfer(address(missingReturn), to, amount, SUCCESS); - } - - function testTransferWithStandardERC20(address to, uint256 amount) public { - verifySafeTransfer(address(erc20), to, amount, SUCCESS); - } - - function testTransferWithReturnsTooMuch(address to, uint256 amount) public { - verifySafeTransfer(address(returnsTooMuch), to, amount, SUCCESS); - } - - function testTransferWithNonGarbage(address to, uint256 amount) public { - returnsRawBytes.setRawBytes(_generateNonGarbage()); - - verifySafeTransfer(address(returnsRawBytes), to, amount, SUCCESS); - } - - function testTransferWithNonContract(address nonContract, address to, uint256 amount) public { - if (uint256(uint160(nonContract)) <= 18 || nonContract.code.length > 0) { - return; - } - - SafeTransferLib.safeTransfer(nonContract, _brutalized(to), amount); - } - - function testTransferETHToContractWithoutFallbackReverts() public { - vm.expectRevert(SafeTransferLib.ETHTransferFailed.selector); - this.safeTransferETH(address(this), 1e18); - } - - function testTransferAllETHToContractWithoutFallbackReverts() public { - vm.expectRevert(SafeTransferLib.ETHTransferFailed.selector); - this.safeTransferAllETH(address(this)); - } - - function testTransferFromWithMissingReturn(address from, address to, uint256 amount) public { - verifySafeTransferFrom(address(missingReturn), from, to, amount, SUCCESS); - } - - function testTransferFromWithStandardERC20(address from, address to, uint256 amount) public { - verifySafeTransferFrom(address(erc20), from, to, amount, SUCCESS); - } - - function testTransferFromWithReturnsTooMuch(address from, address to, uint256 amount) public { - verifySafeTransferFrom(address(returnsTooMuch), from, to, amount, SUCCESS); - } - - function testTransferFromWithNonGarbage(address from, address to, uint256 amount) public { - returnsRawBytes.setRawBytes(_generateNonGarbage()); - - verifySafeTransferFrom(address(returnsRawBytes), from, to, amount, SUCCESS); - } - - function testTransferFromWithNonContract( - address nonContract, - address from, - address to, - uint256 amount - ) public { - if (uint256(uint160(nonContract)) <= 18 || nonContract.code.length > 0) { - return; - } - - SafeTransferLib.safeTransferFrom(nonContract, _brutalized(from), _brutalized(to), amount); - } - - function testApproveWithMissingReturn(address to, uint256 amount) public { - verifySafeApprove(address(missingReturn), to, amount, SUCCESS); - } - - function testApproveWithStandardERC20(address to, uint256 amount) public { - verifySafeApprove(address(erc20), to, amount, SUCCESS); - } - - function testApproveWithReturnsTooMuch(address to, uint256 amount) public { - verifySafeApprove(address(returnsTooMuch), to, amount, SUCCESS); - } - - function testApproveWithNonGarbage(address to, uint256 amount) public { - returnsRawBytes.setRawBytes(_generateNonGarbage()); - - verifySafeApprove(address(returnsRawBytes), to, amount, SUCCESS); - } - - function testApproveWithNonContract(address nonContract, address to, uint256 amount) public { - if (uint256(uint160(nonContract)) <= 18 || nonContract.code.length > 0) { - return; - } - - SafeTransferLib.safeApprove(nonContract, _brutalized(to), amount); - } - - function testApproveWithRetryWithNonContract(address nonContract, address to, uint256 amount) - public - { - if (uint256(uint160(nonContract)) <= 18 || nonContract.code.length > 0) { - return; - } - - SafeTransferLib.safeApproveWithRetry(nonContract, _brutalized(to), amount); - } - - function testApproveWithRetry(address to, uint256 amount0, uint256 amount1) public { - MockERC20LikeUSDT usdt = new MockERC20LikeUSDT(); - assertEq(usdt.allowance(address(this), to), 0); - SafeTransferLib.safeApproveWithRetry(address(usdt), _brutalized(to), amount0); - assertEq(usdt.allowance(address(this), to), amount0); - if (amount0 != 0 && amount1 != 0) { - verifySafeApprove(address(usdt), to, amount1, REVERTS_WITH_SELECTOR); - } - SafeTransferLib.safeApproveWithRetry(address(usdt), _brutalized(to), amount1); - assertEq(usdt.allowance(address(this), to), amount1); - } - - function testApproveWithRetry() public { - testApproveWithRetry(address(1), 123, 456); - } - - function testTransferETH(address recipient, uint256 amount) public { - // Transferring to msg.sender can fail because it's possible to overflow their ETH balance as it begins non-zero. - if ( - recipient.code.length > 0 || uint256(uint160(recipient)) <= 18 - || recipient == msg.sender - ) { - return; - } - - amount = _bound(amount, 0, address(this).balance); - - SafeTransferLib.safeTransferETH(recipient, amount); - } - - function testTransferAllETH(address recipient) public { - // Transferring to msg.sender can fail because it's possible to overflow their ETH balance as it begins non-zero. - if ( - recipient.code.length > 0 || uint256(uint160(recipient)) <= 18 - || recipient == msg.sender - ) { - return; - } - - SafeTransferLib.safeTransferAllETH(recipient); - } - - function testTransferWithReturnsFalseReverts(address to, uint256 amount) public { - verifySafeTransfer(address(returnsFalse), to, amount, REVERTS_WITH_SELECTOR); - } - - function testTransferWithRevertingReverts(address to, uint256 amount) public { - verifySafeTransfer(address(reverting), to, amount, REVERTS_WITH_SELECTOR); - } - - function testTransferWithReturnsTooLittleReverts(address to, uint256 amount) public { - verifySafeTransfer(address(returnsTooLittle), to, amount, REVERTS_WITH_SELECTOR); - } - - function testTransferWithReturnsTwoReverts(address to, uint256 amount) public { - verifySafeTransfer(address(returnsTwo), to, amount, REVERTS_WITH_SELECTOR); - } - - function testTransferWithGarbageReverts(address to, uint256 amount) public { - returnsRawBytes.setRawBytes(_generateGarbage()); - - verifySafeTransfer(address(returnsRawBytes), to, amount, REVERTS_WITH_ANY); - } - - function testTransferFromWithReturnsFalseReverts(address from, address to, uint256 amount) - public - { - verifySafeTransferFrom(address(returnsFalse), from, to, amount, REVERTS_WITH_SELECTOR); - } - - function testTransferFromWithRevertingReverts(address from, address to, uint256 amount) - public - { - verifySafeTransferFrom(address(reverting), from, to, amount, REVERTS_WITH_ANY); - } - - function testTransferFromWithReturnsTooLittleReverts(address from, address to, uint256 amount) - public - { - verifySafeTransferFrom(address(returnsTooLittle), from, to, amount, REVERTS_WITH_SELECTOR); - } - - function testTransferFromWithReturnsTwoReverts(address from, address to, uint256 amount) - public - { - verifySafeTransferFrom(address(returnsTwo), from, to, amount, REVERTS_WITH_SELECTOR); - } - - function testTransferFromWithGarbageReverts(address from, address to, uint256 amount) public { - returnsRawBytes.setRawBytes(_generateGarbage()); - - verifySafeTransferFrom(address(returnsRawBytes), from, to, amount, REVERTS_WITH_ANY); - } - - function testApproveWithReturnsFalseReverts(address to, uint256 amount) public { - verifySafeApprove(address(returnsFalse), to, amount, REVERTS_WITH_SELECTOR); - } - - function testApproveWithRevertingReverts(address to, uint256 amount) public { - verifySafeApprove(address(reverting), to, amount, REVERTS_WITH_SELECTOR); - } - - function testApproveWithReturnsTooLittleReverts(address to, uint256 amount) public { - verifySafeApprove(address(returnsTooLittle), to, amount, REVERTS_WITH_SELECTOR); - } - - function testApproveWithReturnsTwoReverts(address to, uint256 amount) public { - verifySafeApprove(address(returnsTwo), to, amount, REVERTS_WITH_SELECTOR); - } - - function testApproveWithGarbageReverts(address to, uint256 amount) public { - returnsRawBytes.setRawBytes(_generateGarbage()); - - verifySafeApprove(address(returnsRawBytes), to, amount, REVERTS_WITH_ANY); - } - - function testTransferETHToContractWithoutFallbackReverts(uint256 amount) public { - vm.expectRevert(SafeTransferLib.ETHTransferFailed.selector); - this.safeTransferETH(address(this), amount); - } - - function testTransferAllETHToContractWithoutFallbackReverts(uint256) public { - vm.expectRevert(SafeTransferLib.ETHTransferFailed.selector); - this.safeTransferAllETH(address(this)); - } - - function verifySafeTransfer(address token, address to, uint256 amount, uint256 mode) public { - if (mode == REVERTS_WITH_SELECTOR) { - vm.expectRevert(SafeTransferLib.TransferFailed.selector); - } else if (mode == REVERTS_WITH_ANY) { - (bool success,) = address(this).call( - abi.encodeWithSignature( - "verifySafeTransfer(address,address,uint256)", token, to, amount - ) - ); - assertFalse(success); - return; - } - this.verifySafeTransfer(token, to, amount); - } - - function verifySafeTransfer(address token, address to, uint256 amount) public brutalizeMemory { - uint256 preBal = ERC20(token).balanceOf(to); - if (amount == ERC20(token).balanceOf(address(this)) && _random() % 2 == 0) { - SafeTransferLib.safeTransferAll(address(token), _brutalized(to)); - } else { - SafeTransferLib.safeTransfer(address(token), _brutalized(to), amount); - } - - uint256 postBal = ERC20(token).balanceOf(to); - - if (to == address(this)) { - assertEq(preBal, postBal); - } else { - assertEq(postBal - preBal, amount); - } - } - - function verifySafeTransferFrom( - address token, - address from, - address to, - uint256 amount, - uint256 mode - ) public { - if (mode == REVERTS_WITH_SELECTOR) { - vm.expectRevert(SafeTransferLib.TransferFromFailed.selector); - } else if (mode == REVERTS_WITH_ANY) { - (bool success,) = address(this).call( - abi.encodeWithSignature( - "verifySafeTransferFrom(address,address,address,uint256)", - token, - from, - to, - amount - ) - ); - assertFalse(success); - return; - } - this.verifySafeTransferFrom(token, from, to, amount); - } - - function verifySafeTransferFrom(address token, address from, address to, uint256 amount) - public - brutalizeMemory - { - forceApprove(token, from, address(this), amount); - - // We cast to MissingReturnToken here because it won't check - // that there was return data, which accommodates all tokens. - MissingReturnToken(token).transfer(from, amount); - - uint256 preBal = ERC20(token).balanceOf(to); - if (amount == ERC20(token).balanceOf(from) && _random() % 2 == 0) { - SafeTransferLib.safeTransferAllFrom(address(token), _brutalized(from), _brutalized(to)); - } else { - SafeTransferLib.safeTransferFrom(token, _brutalized(from), _brutalized(to), amount); - } - uint256 postBal = ERC20(token).balanceOf(to); - - if (from == to) { - assertEq(preBal, postBal); - } else { - assertEq(postBal - preBal, amount); - } - } - - function verifySafeApprove(address token, address to, uint256 amount, uint256 mode) public { - if (mode == REVERTS_WITH_SELECTOR) { - vm.expectRevert(SafeTransferLib.ApproveFailed.selector); - } else if (mode == REVERTS_WITH_ANY) { - (bool success,) = address(this).call( - abi.encodeWithSignature( - "verifySafeApprove(address,address,uint256)", token, to, amount - ) - ); - assertFalse(success); - return; - } - this.verifySafeApprove(token, to, amount); - } - - function verifySafeApprove(address token, address to, uint256 amount) public { - SafeTransferLib.safeApprove(_brutalized(address(token)), _brutalized(to), amount); - - assertEq(ERC20(token).allowance(address(this), to), amount); - } - - function forceApprove(address token, address from, address to, uint256 amount) public { - if (token == address(erc20)) { - bytes32 allowanceSlot; - /// @solidity memory-safe-assembly - assembly { - mstore(0x20, to) - mstore(0x0c, 0x7f5e9f20) // `_ALLOWANCE_SLOT_SEED`. - mstore(0x00, from) - allowanceSlot := keccak256(0x0c, 0x34) - } - vm.store(token, allowanceSlot, bytes32(uint256(amount))); - } else { - vm.store( - token, - keccak256(abi.encode(to, keccak256(abi.encode(from, uint256(2))))), - bytes32(uint256(amount)) - ); - } - - assertEq(ERC20(token).allowance(from, to), amount, "wrong allowance"); - } - - function forceSafeTransferETH(address to, uint256 amount, uint256 gasStipend) public { - SafeTransferLib.forceSafeTransferETH(to, amount, gasStipend); - } - - function forceSafeTransferETH(address to, uint256 amount) public { - SafeTransferLib.forceSafeTransferETH(to, amount); - } - - function safeTransferETH(address to, uint256 amount) public { - SafeTransferLib.safeTransferETH(to, amount); - } - - function safeTransferAllETH(address to) public { - SafeTransferLib.safeTransferAllETH(to); - } - - function _generateGarbage() internal returns (bytes memory result) { - uint256 r = _random(); - /// @solidity memory-safe-assembly - assembly { - for {} 1 {} { - mstore(0x00, r) - result := mload(0x40) - let n := and(r, 0x7f) - mstore(result, n) - r := keccak256(0x00, 0x40) - mstore(add(result, 0x20), r) - mstore(0x40, add(result, 0x100)) - if and(or(lt(n, 0x20), iszero(eq(r, 1))), gt(n, 0)) { break } - } - } - } - - function _generateNonGarbage() internal returns (bytes memory result) { - uint256 r = _random(); - /// @solidity memory-safe-assembly - assembly { - if iszero(and(r, 1)) { - result := mload(0x40) - mstore(result, 0x20) - mstore(add(result, 0x20), 1) - mstore(0x40, add(result, 0x40)) - } - } - } - - function _brutalized(address a) internal pure returns (address result) { - /// @solidity memory-safe-assembly - assembly { - result := or(a, shl(160, keccak256(0x00, 0x20))) - } - } -} diff --git a/lib/solady/test/SignatureCheckerLib.t.sol b/lib/solady/test/SignatureCheckerLib.t.sol deleted file mode 100644 index 4713d22..0000000 --- a/lib/solady/test/SignatureCheckerLib.t.sol +++ /dev/null @@ -1,341 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {SignatureCheckerLib} from "../src/utils/SignatureCheckerLib.sol"; -import {ECDSA} from "../src/utils/ECDSA.sol"; -import {MockERC1271Wallet} from "./utils/mocks/MockERC1271Wallet.sol"; -import {MockERC1271Malicious} from "./utils/mocks/MockERC1271Malicious.sol"; - -contract SignatureCheckerLibTest is SoladyTest { - bytes32 constant TEST_MESSAGE = - 0x7dbaf558b0a1a5dc7a67202117ab143c1d8605a983e4a743bc06fcc03162dc0d; - - bytes32 constant WRONG_MESSAGE = - 0x2d0828dd7c97cff316356da3c16c68ba2316886a0e05ebafb8291939310d51a3; - - address constant SIGNER = 0x70997970C51812dc3A010C7d01b50e0d17dc79C8; - - address constant OTHER = address(uint160(1)); - - bytes32 constant TEST_SIGNED_MESSAGE_HASH = - 0x7d768af957ef8cbf6219a37e743d5546d911dae3e46449d8a5810522db2ef65e; - - bytes32 constant WRONG_SIGNED_MESSAGE_HASH = - 0x8cd3e659093d21364c6330514aff328218aa29c2693c5b0e96602df075561952; - - bytes constant SIGNATURE = - hex"8688e590483917863a35ef230c0f839be8418aa4ee765228eddfcea7fe2652815db01c2c84b0ec746e1b74d97475c599b3d3419fa7181b4e01de62c02b721aea1b"; - - bytes constant INVALID_SIGNATURE = - hex"7688e590483917863a35ef230c0f839be8418aa4ee765228eddfcea7fe2652815db01c2c84b0ec746e1b74d97475c599b3d3419fa7181b4e01de62c02b721aea1b"; - - MockERC1271Wallet mockERC1271Wallet; - - MockERC1271Malicious mockERC1271Malicious; - - function setUp() public { - mockERC1271Wallet = new MockERC1271Wallet(SIGNER); - mockERC1271Malicious = new MockERC1271Malicious(); - } - - function testSignatureCheckerOnEOAWithMatchingSignerAndSignature() public { - _checkSignature(SIGNER, TEST_SIGNED_MESSAGE_HASH, SIGNATURE, true); - } - - function testSignatureCheckerOnEOAWithInvalidSigner() public { - _checkSignature(OTHER, TEST_SIGNED_MESSAGE_HASH, SIGNATURE, false); - } - - function testSignatureCheckerOnEOAWithWrongSignedMessageHash() public { - _checkSignature(SIGNER, WRONG_SIGNED_MESSAGE_HASH, SIGNATURE, false); - } - - function testSignatureCheckerOnEOAWithInvalidSignature() public { - _checkSignature(SIGNER, TEST_SIGNED_MESSAGE_HASH, INVALID_SIGNATURE, false); - } - - function testSignatureCheckerOnWalletWithMatchingSignerAndSignature() public { - _checkSignatureBothModes( - address(mockERC1271Wallet), TEST_SIGNED_MESSAGE_HASH, SIGNATURE, true - ); - } - - function testSignatureCheckerOnWalletWithInvalidSigner() public { - _checkSignatureBothModes(address(this), TEST_SIGNED_MESSAGE_HASH, SIGNATURE, false); - } - - function testSignatureCheckerOnWalletWithZeroAddressSigner() public { - _checkSignatureBothModes(address(0), TEST_SIGNED_MESSAGE_HASH, SIGNATURE, false); - } - - function testSignatureCheckerOnWalletWithWrongSignedMessageHash() public { - _checkSignatureBothModes( - address(mockERC1271Wallet), WRONG_SIGNED_MESSAGE_HASH, SIGNATURE, false - ); - } - - function testSignatureCheckerOnWalletWithInvalidSignature() public { - _checkSignatureBothModes( - address(mockERC1271Wallet), TEST_SIGNED_MESSAGE_HASH, INVALID_SIGNATURE, false - ); - } - - function testSignatureCheckerOnMaliciousWallet() public { - _checkSignatureBothModes( - address(mockERC1271Malicious), WRONG_SIGNED_MESSAGE_HASH, SIGNATURE, false - ); - } - - function testSignatureChecker(bytes32 digest) public { - (address signer, uint256 privateKey) = _randomSigner(); - - (uint8 v, bytes32 r, bytes32 s) = vm.sign(privateKey, digest); - _checkSignature(signer, digest, abi.encodePacked(r, s, v), true); - - if (_random() % 8 == 0) { - assertEq( - this.isValidSignatureNowCalldata(signer, digest, abi.encodePacked(r, s, v)), true - ); - assertEq( - SignatureCheckerLib.isValidSignatureNow(signer, digest, abi.encodePacked(r, s, v)), - true - ); - assertEq( - SignatureCheckerLib.isValidSignatureNow( - signer, digest, abi.encodePacked(r, s, v + 1) - ), - false - ); - assertEq( - SignatureCheckerLib.isValidSignatureNow( - signer, digest, abi.encodePacked(r, s, v - 1) - ), - false - ); - assertEq(SignatureCheckerLib.isValidSignatureNow(signer, digest, v, r, s), true); - } - - if (_random() % 8 == 0) { - bytes32 vs; - /// @solidity memory-safe-assembly - assembly { - vs := or(shl(255, sub(v, 27)), s) - } - assertEq(SignatureCheckerLib.isValidSignatureNow(signer, digest, r, vs), true); - assertEq( - SignatureCheckerLib.isValidSignatureNow(signer, digest, abi.encode(r, vs)), true - ); - assertEq(this.isValidSignatureNowCalldata(signer, digest, abi.encode(r, vs)), true); - } - - if (_random() % 8 == 0) { - bytes32 vsc; // Corrupted `vs`. - /// @solidity memory-safe-assembly - assembly { - vsc := or(shl(255, xor(1, sub(v, 27))), s) - } - assertEq(SignatureCheckerLib.isValidSignatureNow(signer, digest, r, vsc), false); - assertEq( - SignatureCheckerLib.isValidSignatureNow(signer, digest, abi.encode(r, vsc)), false - ); - assertEq(this.isValidSignatureNowCalldata(signer, digest, abi.encode(r, vsc)), false); - } - - if (_random() % 8 == 0 && r != bytes32(0) && s != bytes32(0)) { - bytes32 rc = bytes32(uint256(r) - (_random() & 1)); // Corrupted `r`. - bytes32 sc = bytes32(uint256(s) - (_random() & 1)); // Corrupted `s`. - bool anyCorrupted = rc != r || sc != s; - _checkSignature(signer, digest, abi.encodePacked(rc, sc, v), !anyCorrupted); - } - - if (_random() % 8 == 0) { - uint8 vc = uint8(_random()); // Corrupted `v`. - while (vc == 28 || vc == 27) vc = uint8(_random()); - assertEq(SignatureCheckerLib.isValidSignatureNow(signer, digest, vc, r, s), false); - assertEq( - SignatureCheckerLib.isValidSignatureNow(signer, digest, abi.encodePacked(r, s, vc)), - false - ); - assertEq( - this.isValidSignatureNowCalldata(signer, digest, abi.encodePacked(r, s, vc)), false - ); - } - } - - function _checkSignatureBothModes( - address signer, - bytes32 hash, - bytes memory signature, - bool expectedResult - ) internal { - _checkSignature(false, signer, hash, signature, expectedResult); - _checkSignature(true, signer, hash, signature, expectedResult); - } - - function _checkSignature( - address signer, - bytes32 hash, - bytes memory signature, - bool expectedResult - ) internal { - _checkSignature(false, signer, hash, signature, expectedResult); - } - - function _checkSignature( - bool onlyERC1271, - address signer, - bytes32 hash, - bytes memory signature, - bool expectedResult - ) internal { - bool callResult; - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) - - // `bytes4(keccak256("isValidSignatureNow(address,bytes32,bytes)"))`. - mstore(m, shl(224, 0x6ccea652)) - if onlyERC1271 { - // `bytes4(keccak256("isValidERC1271SignatureNow(address,bytes32,bytes)"))`. - mstore(m, shl(224, 0x3ae5d83c)) - } - mstore(add(m, 0x04), signer) - mstore(add(m, 0x24), hash) - mstore(add(m, 0x44), 0x60) // Offset of signature in calldata. - mstore(add(m, 0x64), mload(signature)) - mstore(add(m, 0x84), mload(add(signature, 0x20))) - mstore(add(m, 0xa4), mload(add(signature, 0x40))) - mstore(add(m, 0xc4), mload(add(signature, 0x60))) - // Brutalize the bytes following the 8-bit `v`. All ones will do. - mstore(add(m, 0xc5), not(0)) - - // We have to do the call in assembly to ensure that Solidity does not - // clean up the brutalized bits. - callResult := - and( - and( - // Whether the returndata is equal to 1. - eq(mload(0x00), 1), - // Whether the returndata is exactly 0x20 bytes (1 word) long . - eq(returndatasize(), 0x20) - ), - // Whether the staticcall does not revert. - // This must be placed at the end of the `and` clause, - // as the arguments are evaluated from right to left. - staticcall( - gas(), // Remaining gas. - address(), // The current contract's address. - m, // Offset of calldata in memory. - 0xe4, // Length of calldata in memory. - 0x00, // Offset of returndata. - 0x20 // Length of returndata to write. - ) - ) - } - assertEq(callResult, expectedResult); - - uint8 v; - bytes32 r; - bytes32 s; - bytes32 vs; - /// @solidity memory-safe-assembly - assembly { - // Contaminate the upper 96 bits. - signer := or(shl(160, 1), signer) - // Extract `r`, `s`, `v`. - r := mload(add(signature, 0x20)) - s := mload(add(signature, 0x40)) - v := byte(0, mload(add(signature, 0x60))) - // Pack `vs`. - vs := or(shl(255, sub(v, 27)), s) - - // Brutalize the memory. Just all ones will do. - let m := mload(0x40) - for { let i := 0 } lt(i, 30) { i := add(i, 1) } { mstore(add(m, shl(5, i)), not(0)) } - } - - if (onlyERC1271) { - assertEq( - SignatureCheckerLib.isValidERC1271SignatureNow(signer, hash, r, vs), expectedResult - ); - assertEq( - SignatureCheckerLib.isValidERC1271SignatureNow(signer, hash, v, r, s), - expectedResult - ); - } else { - assertEq(SignatureCheckerLib.isValidSignatureNow(signer, hash, r, vs), expectedResult); - assertEq(SignatureCheckerLib.isValidSignatureNow(signer, hash, v, r, s), expectedResult); - } - } - - function isValidSignatureNow(address signer, bytes32 hash, bytes calldata signature) - external - returns (bool result) - { - bool signatureIsBrutalized; - /// @solidity memory-safe-assembly - assembly { - // Contaminate the upper 96 bits. - signer := or(shl(160, 1), signer) - // Ensure that the bytes right after the signature is brutalized. - signatureIsBrutalized := calldataload(add(signature.offset, signature.length)) - } - if (!signatureIsBrutalized) revert("Signature is not brutalized."); - - result = SignatureCheckerLib.isValidSignatureNowCalldata(signer, hash, signature); - assertEq(SignatureCheckerLib.isValidSignatureNow(signer, hash, signature), result); - } - - function isValidERC1271SignatureNow(address signer, bytes32 hash, bytes calldata signature) - external - returns (bool result) - { - bool signatureIsBrutalized; - /// @solidity memory-safe-assembly - assembly { - // Contaminate the upper 96 bits. - signer := or(shl(160, 1), signer) - // Ensure that the bytes right after the signature is brutalized. - signatureIsBrutalized := calldataload(add(signature.offset, signature.length)) - } - if (!signatureIsBrutalized) revert("Signature is not brutalized."); - - result = SignatureCheckerLib.isValidERC1271SignatureNowCalldata(signer, hash, signature); - assertEq(SignatureCheckerLib.isValidERC1271SignatureNow(signer, hash, signature), result); - } - - function isValidSignatureNowCalldata(address signer, bytes32 hash, bytes calldata signature) - external - view - returns (bool result) - { - result = SignatureCheckerLib.isValidSignatureNowCalldata(signer, hash, signature); - } - - function isValidERC1271SignatureNowCalldata( - address signer, - bytes32 hash, - bytes calldata signature - ) external view returns (bool result) { - result = SignatureCheckerLib.isValidERC1271SignatureNowCalldata(signer, hash, signature); - } - - function testEmptyCalldataHelpers() public { - assertFalse( - SignatureCheckerLib.isValidSignatureNow( - address(1), bytes32(0), SignatureCheckerLib.emptySignature() - ) - ); - } - - function testToEthSignedMessageHashDifferential(bytes32 hash) public { - assertEq( - SignatureCheckerLib.toEthSignedMessageHash(hash), ECDSA.toEthSignedMessageHash(hash) - ); - } - - function testToEthSignedMessageHashDifferential(bytes memory s) public { - assertEq(SignatureCheckerLib.toEthSignedMessageHash(s), ECDSA.toEthSignedMessageHash(s)); - } -} diff --git a/lib/solady/test/UUPSUpgradeable.t.sol b/lib/solady/test/UUPSUpgradeable.t.sol deleted file mode 100644 index d7f1d7d..0000000 --- a/lib/solady/test/UUPSUpgradeable.t.sol +++ /dev/null @@ -1,84 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import {UUPSUpgradeable} from "../src/utils/UUPSUpgradeable.sol"; -import {LibClone} from "../src/utils/LibClone.sol"; -import {MockUUPSImplementation} from "../test/utils/mocks/MockUUPSImplementation.sol"; - -contract UUPSUpgradeableTest is SoladyTest { - MockUUPSImplementation impl1; - - address proxy; - - bytes32 internal constant _ERC1967_IMPLEMENTATION_SLOT = - 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc; - - event Upgraded(address indexed implementation); - - function setUp() public { - impl1 = new MockUUPSImplementation(); - proxy = LibClone.deployERC1967(address(impl1)); - MockUUPSImplementation(proxy).initialize(address(this)); - } - - function testNotDelegatedGuard() public { - assertEq(impl1.proxiableUUID(), _ERC1967_IMPLEMENTATION_SLOT); - vm.expectRevert(UUPSUpgradeable.UnauthorizedCallContext.selector); - MockUUPSImplementation(proxy).proxiableUUID(); - } - - function testOnlyProxyGuard() public { - vm.expectRevert(UUPSUpgradeable.UnauthorizedCallContext.selector); - impl1.upgradeToAndCall(address(1), bytes("")); - } - - function testUpgradeTo() public { - MockUUPSImplementation impl2 = new MockUUPSImplementation(); - vm.expectEmit(true, true, true, true); - emit Upgraded(address(impl2)); - MockUUPSImplementation(proxy).upgradeToAndCall(address(impl2), bytes("")); - bytes32 v = vm.load(proxy, _ERC1967_IMPLEMENTATION_SLOT); - assertEq(address(uint160(uint256(v))), address(impl2)); - } - - function testUpgradeToRevertWithUnauthorized() public { - vm.prank(address(0xBEEF)); - vm.expectRevert(MockUUPSImplementation.Unauthorized.selector); - MockUUPSImplementation(proxy).upgradeToAndCall(address(0xABCD), bytes("")); - } - - function testUpgradeToRevertWithUpgradeFailed() public { - vm.expectRevert(UUPSUpgradeable.UpgradeFailed.selector); - MockUUPSImplementation(proxy).upgradeToAndCall(address(0xABCD), bytes("")); - } - - function testUpgradeToAndCall() public { - MockUUPSImplementation impl2 = new MockUUPSImplementation(); - bytes memory data = abi.encodeWithSignature("setValue(uint256)", 5); - MockUUPSImplementation(proxy).upgradeToAndCall(address(impl2), data); - bytes32 v = vm.load(proxy, _ERC1967_IMPLEMENTATION_SLOT); - assertEq(address(uint160(uint256(v))), address(impl2)); - assertEq(MockUUPSImplementation(proxy).value(), 5); - } - - function testUpgradeToAndCallRevertWithUpgradeFailed() public { - vm.expectRevert(UUPSUpgradeable.UpgradeFailed.selector); - MockUUPSImplementation(proxy).upgradeToAndCall(address(0xABCD), ""); - } - - function testUpgradeToAndCallRevertWithCustomError() public { - MockUUPSImplementation impl2 = new MockUUPSImplementation(); - bytes memory data = abi.encodeWithSignature("revertWithError()"); - vm.expectRevert( - abi.encodeWithSelector(MockUUPSImplementation.CustomError.selector, address(this)) - ); - MockUUPSImplementation(proxy).upgradeToAndCall(address(impl2), data); - } - - function testUpgradeToAndCallRevertWithUnauthorized() public { - vm.prank(address(0xBEEF)); - vm.expectRevert(MockUUPSImplementation.Unauthorized.selector); - MockUUPSImplementation(proxy).upgradeToAndCall(address(0xABCD), ""); - } -} diff --git a/lib/solady/test/WETH.t.sol b/lib/solady/test/WETH.t.sol deleted file mode 100644 index e788f76..0000000 --- a/lib/solady/test/WETH.t.sol +++ /dev/null @@ -1,202 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./utils/SoladyTest.sol"; -import "./utils/InvariantTest.sol"; - -import {SafeTransferLib} from "../src/utils/SafeTransferLib.sol"; - -import {WETH} from "../src/tokens/WETH.sol"; - -contract ContractWithoutReceive {} - -contract WETHTest is SoladyTest { - WETH weth; - - event Deposit(address indexed from, uint256 amount); - - event Withdrawal(address indexed to, uint256 amount); - - function setUp() public { - weth = new WETH(); - } - - function _expectDepositEvent(address from, uint256 amount) internal { - vm.expectEmit(true, true, true, true); - emit Deposit(from, amount); - } - - function _expectDepositEvent(uint256 amount) internal { - _expectDepositEvent(address(this), amount); - } - - function _expectWithdrawalEvent(address to, uint256 amount) internal { - vm.expectEmit(true, true, true, true); - emit Withdrawal(to, amount); - } - - function _expectWithdrawalEvent(uint256 amount) internal { - _expectWithdrawalEvent(address(this), amount); - } - - function testMetadata() public { - assertEq(weth.name(), "Wrapped Ether"); - assertEq(weth.symbol(), "WETH"); - assertEq(weth.decimals(), 18); - } - - function testFallbackDeposit() public { - assertEq(weth.balanceOf(address(this)), 0); - assertEq(weth.totalSupply(), 0); - - _expectDepositEvent(1 ether); - SafeTransferLib.safeTransferETH(address(weth), 1 ether); - - assertEq(weth.balanceOf(address(this)), 1 ether); - assertEq(weth.totalSupply(), 1 ether); - } - - function testDeposit() public { - assertEq(weth.balanceOf(address(this)), 0); - assertEq(weth.totalSupply(), 0); - - _expectDepositEvent(1 ether); - weth.deposit{value: 1 ether}(); - - assertEq(weth.balanceOf(address(this)), 1 ether); - assertEq(weth.totalSupply(), 1 ether); - } - - function testWithdraw() public { - uint256 startingBalance = address(this).balance; - - _expectDepositEvent(1 ether); - weth.deposit{value: 1 ether}(); - - _expectWithdrawalEvent(1 ether); - weth.withdraw(1 ether); - - uint256 balanceAfterWithdraw = address(this).balance; - - assertEq(balanceAfterWithdraw, startingBalance); - assertEq(weth.balanceOf(address(this)), 0); - assertEq(weth.totalSupply(), 0); - } - - function testPartialWithdraw() public { - _expectDepositEvent(1 ether); - weth.deposit{value: 1 ether}(); - - uint256 balanceBeforeWithdraw = address(this).balance; - - _expectWithdrawalEvent(0.5 ether); - weth.withdraw(0.5 ether); - - uint256 balanceAfterWithdraw = address(this).balance; - - assertEq(balanceAfterWithdraw, balanceBeforeWithdraw + 0.5 ether); - assertEq(weth.balanceOf(address(this)), 0.5 ether); - assertEq(weth.totalSupply(), 0.5 ether); - } - - function testWithdrawToContractWithoutReceiveReverts() public { - address owner = address(new ContractWithoutReceive()); - - vm.deal(owner, 1 ether); - - vm.prank(owner); - _expectDepositEvent(owner, 1 ether); - weth.deposit{value: 1 ether}(); - - assertEq(weth.balanceOf(owner), 1 ether); - - vm.expectRevert(WETH.ETHTransferFailed.selector); - vm.prank(owner); - weth.withdraw(1 ether); - } - - function testFallbackDeposit(uint256 amount) public { - amount = _bound(amount, 0, address(this).balance); - - assertEq(weth.balanceOf(address(this)), 0); - assertEq(weth.totalSupply(), 0); - - _expectDepositEvent(amount); - SafeTransferLib.safeTransferETH(address(weth), amount); - - assertEq(weth.balanceOf(address(this)), amount); - assertEq(weth.totalSupply(), amount); - } - - function testDeposit(uint256 amount) public { - amount = _bound(amount, 0, address(this).balance); - - assertEq(weth.balanceOf(address(this)), 0); - assertEq(weth.totalSupply(), 0); - - _expectDepositEvent(amount); - weth.deposit{value: amount}(); - - assertEq(weth.balanceOf(address(this)), amount); - assertEq(weth.totalSupply(), amount); - } - - function testWithdraw(uint256 depositAmount, uint256 withdrawAmount) public { - depositAmount = _bound(depositAmount, 0, address(this).balance); - withdrawAmount = _bound(withdrawAmount, 0, depositAmount); - - _expectDepositEvent(depositAmount); - weth.deposit{value: depositAmount}(); - - uint256 balanceBeforeWithdraw = address(this).balance; - - _expectWithdrawalEvent(withdrawAmount); - weth.withdraw(withdrawAmount); - - uint256 balanceAfterWithdraw = address(this).balance; - - assertEq(balanceAfterWithdraw, balanceBeforeWithdraw + withdrawAmount); - assertEq(weth.balanceOf(address(this)), depositAmount - withdrawAmount); - assertEq(weth.totalSupply(), depositAmount - withdrawAmount); - } - - receive() external payable {} -} - -contract WETHInvariants is SoladyTest, InvariantTest { - WETHTester wethTester; - WETH weth; - - function setUp() public { - weth = new WETH(); - wethTester = new WETHTester{value: address(this).balance}(weth); - - _addTargetContract(address(wethTester)); - } - - function invariantTotalSupplyEqualsBalance() public { - assertEq(address(weth).balance, weth.totalSupply()); - } -} - -contract WETHTester { - WETH weth; - - constructor(WETH _weth) payable { - weth = _weth; - } - - function deposit(uint256 amount) public { - weth.deposit{value: amount}(); - } - - function fallbackDeposit(uint256 amount) public { - SafeTransferLib.safeTransferETH(address(weth), amount); - } - - function withdraw(uint256 amount) public { - weth.withdraw(amount); - } - - receive() external payable {} -} diff --git a/lib/solady/test/utils/InvariantTest.sol b/lib/solady/test/utils/InvariantTest.sol deleted file mode 100644 index 261af92..0000000 --- a/lib/solady/test/utils/InvariantTest.sol +++ /dev/null @@ -1,15 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -contract InvariantTest { - address[] private _targets; - - function targetContracts() public view virtual returns (address[] memory) { - require(_targets.length > 0, "NO_TARGET_CONTRACTS"); - return _targets; - } - - function _addTargetContract(address newTargetContract) internal virtual { - _targets.push(newTargetContract); - } -} diff --git a/lib/solady/test/utils/SoladyTest.sol b/lib/solady/test/utils/SoladyTest.sol deleted file mode 100644 index 4969db3..0000000 --- a/lib/solady/test/utils/SoladyTest.sol +++ /dev/null @@ -1,12 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "./forge-std/Test.sol"; -import "./TestPlus.sol"; - -contract SoladyTest is Test, TestPlus { - /// @dev Alias for `_hem`. - function _bound(uint256 x, uint256 min, uint256 max) internal pure virtual returns (uint256) { - return _hem(x, min, max); - } -} diff --git a/lib/solady/test/utils/TestPlus.sol b/lib/solady/test/utils/TestPlus.sol deleted file mode 100644 index 4443f70..0000000 --- a/lib/solady/test/utils/TestPlus.sol +++ /dev/null @@ -1,351 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -contract TestPlus { - /// @dev `address(bytes20(uint160(uint256(keccak256("hevm cheat code")))))`. - address private constant _VM_ADDRESS = 0x7109709ECfa91a80626fF3989D68f67F5b1DD12D; - - /// @dev Fills the memory with junk, for more robust testing of inline assembly - /// which reads/write to the memory. - function _brutalizeMemory() private view { - // To prevent a solidity 0.8.13 bug. - // See: https://blog.soliditylang.org/2022/06/15/inline-assembly-memory-side-effects-bug - // Basically, we need to access a solidity variable from the assembly to - // tell the compiler that this assembly block is not in isolation. - uint256 zero; - /// @solidity memory-safe-assembly - assembly { - let offset := mload(0x40) // Start the offset at the free memory pointer. - calldatacopy(offset, zero, calldatasize()) - - // Fill the 64 bytes of scratch space with garbage. - mstore(zero, add(caller(), gas())) - mstore(0x20, keccak256(offset, calldatasize())) - mstore(zero, keccak256(zero, 0x40)) - - let r0 := mload(zero) - let r1 := mload(0x20) - - let cSize := add(codesize(), iszero(codesize())) - if iszero(lt(cSize, 32)) { cSize := sub(cSize, and(mload(0x02), 0x1f)) } - let start := mod(mload(0x10), cSize) - let size := mul(sub(cSize, start), gt(cSize, start)) - let times := div(0x7ffff, cSize) - if iszero(lt(times, 128)) { times := 128 } - - // Occasionally offset the offset by a pseudorandom large amount. - // Can't be too large, or we will easily get out-of-gas errors. - offset := add(offset, mul(iszero(and(r1, 0xf)), and(r0, 0xfffff))) - - // Fill the free memory with garbage. - // prettier-ignore - for { let w := not(0) } 1 {} { - mstore(offset, r0) - mstore(add(offset, 0x20), r1) - offset := add(offset, 0x40) - // We use codecopy instead of the identity precompile - // to avoid polluting the `forge test -vvvv` output with tons of junk. - codecopy(offset, start, size) - codecopy(add(offset, size), 0, start) - offset := add(offset, cSize) - times := add(times, w) // `sub(times, 1)`. - if iszero(times) { break } - } - } - } - - /// @dev Fills the memory with junk, for more robust testing of inline assembly - /// which reads/write to the memory. - modifier brutalizeMemory() { - _brutalizeMemory(); - _; - _checkMemory(); - } - - /// @dev Returns a pseudorandom random number from [0 .. 2**256 - 1] (inclusive). - /// For usage in fuzz tests, please ensure that the function has an unnamed uint256 argument. - /// e.g. `testSomething(uint256) public`. - function _random() internal returns (uint256 r) { - /// @solidity memory-safe-assembly - assembly { - // This is the keccak256 of a very long string I randomly mashed on my keyboard. - let sSlot := 0xd715531fe383f818c5f158c342925dcf01b954d24678ada4d07c36af0f20e1ee - let sValue := sload(sSlot) - - mstore(0x20, sValue) - r := keccak256(0x20, 0x40) - - // If the storage is uninitialized, initialize it to the keccak256 of the calldata. - if iszero(sValue) { - sValue := sSlot - let m := mload(0x40) - calldatacopy(m, 0, calldatasize()) - r := keccak256(m, calldatasize()) - } - sstore(sSlot, add(r, 1)) - - // Do some biased sampling for more robust tests. - // prettier-ignore - for {} 1 {} { - let d := byte(0, r) - // With a 1/256 chance, randomly set `r` to any of 0,1,2. - if iszero(d) { - r := and(r, 3) - break - } - // With a 1/2 chance, set `r` to near a random power of 2. - if iszero(and(2, d)) { - // Set `t` either `not(0)` or `xor(sValue, r)`. - let t := xor(not(0), mul(iszero(and(4, d)), not(xor(sValue, r)))) - // Set `r` to `t` shifted left or right by a random multiple of 8. - switch and(8, d) - case 0 { - if iszero(and(16, d)) { t := 1 } - r := add(shl(shl(3, and(byte(3, r), 0x1f)), t), sub(and(r, 7), 3)) - } - default { - if iszero(and(16, d)) { t := shl(255, 1) } - r := add(shr(shl(3, and(byte(3, r), 0x1f)), t), sub(and(r, 7), 3)) - } - // With a 1/2 chance, negate `r`. - if iszero(and(0x20, d)) { r := not(r) } - break - } - // Otherwise, just set `r` to `xor(sValue, r)`. - r := xor(sValue, r) - break - } - } - } - - /// @dev Returns a random signer and its private key. - function _randomSigner() internal returns (address signer, uint256 privateKey) { - uint256 privateKeyMax = 0xfffffffffffffffffffffffffffffffebaaedce6af48a03bbfd25e8cd0364140; - privateKey = _hem(_random(), 1, privateKeyMax); - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, 0xffa18649) // `addr(uint256)`. - mstore(0x20, privateKey) - if iszero(call(gas(), _VM_ADDRESS, 0, 0x1c, 0x24, 0x00, 0x20)) { revert(0, 0) } - signer := mload(0x00) - } - } - - /// @dev Returns a random address. - function _randomAddress() internal returns (address result) { - result = address(uint160(_random())); - } - - /// @dev Returns a random non-zero address. - function _randomNonZeroAddress() internal returns (address result) { - do { - result = address(uint160(_random())); - } while (result == address(0)); - } - - /// @dev Rounds up the free memory pointer to the next word boundary. - /// Sometimes, some Solidity operations cause the free memory pointer to be misaligned. - function _roundUpFreeMemoryPointer() internal pure { - // To prevent a solidity 0.8.13 bug. - // See: https://blog.soliditylang.org/2022/06/15/inline-assembly-memory-side-effects-bug - // Basically, we need to access a solidity variable from the assembly to - // tell the compiler that this assembly block is not in isolation. - uint256 twoWords = 0x40; - /// @solidity memory-safe-assembly - assembly { - mstore(twoWords, and(add(mload(twoWords), 0x1f), not(0x1f))) - } - } - - /// @dev Misaligns the free memory pointer. - /// The free memory pointer has a 1/32 chance to be aligned. - function _misalignFreeMemoryPointer() internal pure { - uint256 twoWords = 0x40; - /// @solidity memory-safe-assembly - assembly { - let m := mload(twoWords) - m := add(m, mul(and(keccak256(0x00, twoWords), 0x1f), iszero(and(m, 0x1f)))) - mstore(twoWords, m) - } - } - - /// @dev Check if the free memory pointer and the zero slot are not contaminated. - /// Useful for cases where these slots are used for temporary storage. - function _checkMemory() internal pure { - bool zeroSlotIsNotZero; - bool freeMemoryPointerOverflowed; - /// @solidity memory-safe-assembly - assembly { - // Write ones to the free memory, to make subsequent checks fail if - // insufficient memory is allocated. - mstore(mload(0x40), not(0)) - // Test at a lower, but reasonable limit for more safety room. - if gt(mload(0x40), 0xffffffff) { freeMemoryPointerOverflowed := 1 } - // Check the value of the zero slot. - zeroSlotIsNotZero := mload(0x60) - } - if (freeMemoryPointerOverflowed) revert("`0x40` overflowed!"); - if (zeroSlotIsNotZero) revert("`0x60` is not zero!"); - } - - /// @dev Check if `s`: - /// - Has sufficient memory allocated. - /// - Is zero right padded (cuz some frontends like Etherscan has issues - /// with decoding non-zero-right-padded strings). - function _checkMemory(bytes memory s) internal pure { - bool notZeroRightPadded; - bool insufficientMalloc; - /// @solidity memory-safe-assembly - assembly { - // Write ones to the free memory, to make subsequent checks fail if - // insufficient memory is allocated. - mstore(mload(0x40), not(0)) - let length := mload(s) - let lastWord := mload(add(add(s, 0x20), and(length, not(0x1f)))) - let remainder := and(length, 0x1f) - if remainder { if shl(mul(8, remainder), lastWord) { notZeroRightPadded := 1 } } - // Check if the memory allocated is sufficient. - if length { if gt(add(add(s, 0x20), length), mload(0x40)) { insufficientMalloc := 1 } } - } - if (notZeroRightPadded) revert("Not zero right padded!"); - if (insufficientMalloc) revert("Insufficient memory allocation!"); - _checkMemory(); - } - - /// @dev For checking the memory allocation for string `s`. - function _checkMemory(string memory s) internal pure { - _checkMemory(bytes(s)); - } - - /// @dev Adapted from `bound`: - /// https://github.com/foundry-rs/forge-std/blob/ff4bf7db008d096ea5a657f2c20516182252a3ed/src/StdUtils.sol#L10 - /// Differentially fuzzed tested against the original implementation. - function _hem(uint256 x, uint256 min, uint256 max) - internal - pure - virtual - returns (uint256 result) - { - require(min <= max, "Max is less than min."); - - /// @solidity memory-safe-assembly - assembly { - // prettier-ignore - for {} 1 {} { - // If `x` is between `min` and `max`, return `x` directly. - // This is to ensure that dictionary values - // do not get shifted if the min is nonzero. - // More info: https://github.com/foundry-rs/forge-std/issues/188 - if iszero(or(lt(x, min), gt(x, max))) { - result := x - break - } - - let size := add(sub(max, min), 1) - if and(iszero(gt(x, 3)), gt(size, x)) { - result := add(min, x) - break - } - - let w := not(0) - if and(iszero(lt(x, sub(0, 4))), gt(size, sub(w, x))) { - result := sub(max, sub(w, x)) - break - } - - // Otherwise, wrap x into the range [min, max], - // i.e. the range is inclusive. - if iszero(lt(x, max)) { - let d := sub(x, max) - let r := mod(d, size) - if iszero(r) { - result := max - break - } - result := add(add(min, r), w) - break - } - let d := sub(min, x) - let r := mod(d, size) - if iszero(r) { - result := min - break - } - result := add(sub(max, r), 1) - break - } - } - } - - /// @dev Deploys a contract via 0age's immutable create 2 factory for testing. - function _safeCreate2(uint256 payableAmount, bytes32 salt, bytes memory initializationCode) - internal - returns (address deploymentAddress) - { - // Canonical address of 0age's immutable create 2 factory. - address c2f = 0x0000000000FFe8B47B3e2130213B802212439497; - uint256 c2fCodeLength; - /// @solidity memory-safe-assembly - assembly { - c2fCodeLength := extcodesize(c2f) - } - if (c2fCodeLength == 0) { - bytes memory ic2fBytecode = - hex"60806040526004361061003f5760003560e01c806308508b8f1461004457806364e030871461009857806385cf97ab14610138578063a49a7c90146101bc575b600080fd5b34801561005057600080fd5b506100846004803603602081101561006757600080fd5b503573ffffffffffffffffffffffffffffffffffffffff166101ec565b604080519115158252519081900360200190f35b61010f600480360360408110156100ae57600080fd5b813591908101906040810160208201356401000000008111156100d057600080fd5b8201836020820111156100e257600080fd5b8035906020019184600183028401116401000000008311171561010457600080fd5b509092509050610217565b6040805173ffffffffffffffffffffffffffffffffffffffff9092168252519081900360200190f35b34801561014457600080fd5b5061010f6004803603604081101561015b57600080fd5b8135919081019060408101602082013564010000000081111561017d57600080fd5b82018360208201111561018f57600080fd5b803590602001918460018302840111640100000000831117156101b157600080fd5b509092509050610592565b3480156101c857600080fd5b5061010f600480360360408110156101df57600080fd5b508035906020013561069e565b73ffffffffffffffffffffffffffffffffffffffff1660009081526020819052604090205460ff1690565b600083606081901c33148061024c57507fffffffffffffffffffffffffffffffffffffffff0000000000000000000000008116155b6102a1576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004018080602001828103825260458152602001806107746045913960600191505060405180910390fd5b606084848080601f0160208091040260200160405190810160405280939291908181526020018383808284376000920182905250604051855195965090943094508b93508692506020918201918291908401908083835b6020831061033557805182527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe090920191602091820191016102f8565b51815160209384036101000a7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff018019909216911617905260408051929094018281037fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe00183528085528251928201929092207fff000000000000000000000000000000000000000000000000000000000000008383015260609890981b7fffffffffffffffffffffffffffffffffffffffff00000000000000000000000016602183015260358201969096526055808201979097528251808203909701875260750182525084519484019490942073ffffffffffffffffffffffffffffffffffffffff81166000908152938490529390922054929350505060ff16156104a7576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040180806020018281038252603f815260200180610735603f913960400191505060405180910390fd5b81602001825188818334f5955050508073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff161461053a576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004018080602001828103825260468152602001806107b96046913960600191505060405180910390fd5b50505073ffffffffffffffffffffffffffffffffffffffff8116600090815260208190526040902080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff001660011790559392505050565b6000308484846040516020018083838082843760408051919093018181037fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe001825280845281516020928301207fff000000000000000000000000000000000000000000000000000000000000008383015260609990991b7fffffffffffffffffffffffffffffffffffffffff000000000000000000000000166021820152603581019790975260558088019890985282518088039098018852607590960182525085519585019590952073ffffffffffffffffffffffffffffffffffffffff81166000908152948590529490932054939450505060ff909116159050610697575060005b9392505050565b604080517fff000000000000000000000000000000000000000000000000000000000000006020808301919091523060601b6021830152603582018590526055808301859052835180840390910181526075909201835281519181019190912073ffffffffffffffffffffffffffffffffffffffff81166000908152918290529190205460ff161561072e575060005b9291505056fe496e76616c696420636f6e7472616374206372656174696f6e202d20636f6e74726163742068617320616c7265616479206265656e206465706c6f7965642e496e76616c69642073616c74202d206669727374203230206279746573206f66207468652073616c74206d757374206d617463682063616c6c696e6720616464726573732e4661696c656420746f206465706c6f7920636f6e7472616374207573696e672070726f76696465642073616c7420616e6420696e697469616c697a6174696f6e20636f64652ea265627a7a723058202bdc55310d97c4088f18acf04253db593f0914059f0c781a9df3624dcef0d1cf64736f6c634300050a0032"; - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) - mstore(m, 0xb4d6c782) // `etch(address,bytes)`. - mstore(add(m, 0x20), c2f) - mstore(add(m, 0x40), 0x40) - let n := mload(ic2fBytecode) - mstore(add(m, 0x60), n) - for { let i := 0 } lt(i, n) { i := add(0x20, i) } { - mstore(add(add(m, 0x80), i), mload(add(add(ic2fBytecode, 0x20), i))) - } - if iszero(call(gas(), _VM_ADDRESS, 0, add(m, 0x1c), add(n, 0x64), 0x00, 0x00)) { - revert(0, 0) - } - } - } - /// @solidity memory-safe-assembly - assembly { - let m := mload(0x40) - let n := mload(initializationCode) - mstore(m, 0x64e03087) // `safeCreate2(bytes32,bytes)`. - mstore(add(m, 0x20), salt) - mstore(add(m, 0x40), 0x40) - mstore(add(m, 0x60), n) - // prettier-ignore - for { let i := 0 } lt(i, n) { i := add(i, 0x20) } { - mstore(add(add(m, 0x80), i), mload(add(add(initializationCode, 0x20), i))) - } - if iszero(call(gas(), c2f, payableAmount, add(m, 0x1c), add(n, 0x64), m, 0x20)) { - returndatacopy(m, m, returndatasize()) - revert(m, returndatasize()) - } - deploymentAddress := mload(m) - } - } - - /// @dev Deploys a contract via 0age's immutable create 2 factory for testing. - function _safeCreate2(bytes32 salt, bytes memory initializationCode) - internal - returns (address deploymentAddress) - { - deploymentAddress = _safeCreate2(0, salt, initializationCode); - } - - /// @dev This function will make forge's gas output display the approximate codesize of - /// the test contract as the amount of gas burnt. Useful for quick guess checking if - /// certain optimizations actually compiles to similar bytecode. - function test__codesize() external view { - /// @solidity memory-safe-assembly - assembly { - // If the caller is the contract itself (i.e. recursive call), burn all the gas. - if eq(caller(), address()) { invalid() } - mstore(0x00, 0xf09ff470) // Store the function selector of `test__codesize()`. - pop(staticcall(codesize(), address(), 0x1c, 0x04, 0x00, 0x00)) - } - } -} diff --git a/lib/solady/test/utils/forge-std/Script.sol b/lib/solady/test/utils/forge-std/Script.sol deleted file mode 100644 index 724dd85..0000000 --- a/lib/solady/test/utils/forge-std/Script.sol +++ /dev/null @@ -1,13 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity >=0.6.0 <0.9.0; - -import "./Vm.sol"; -import "./console.sol"; - -abstract contract Script { - bool public IS_SCRIPT = true; - address private constant VM_ADDRESS = - address(bytes20(uint160(uint256(keccak256("hevm cheat code"))))); - - Vm public constant vm = Vm(VM_ADDRESS); -} diff --git a/lib/solady/test/utils/forge-std/Test.sol b/lib/solady/test/utils/forge-std/Test.sol deleted file mode 100644 index bf92f66..0000000 --- a/lib/solady/test/utils/forge-std/Test.sol +++ /dev/null @@ -1,165 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity >=0.6.0 <0.9.0; - -import "./Script.sol"; -import "ds-test/test.sol"; - -// Wrappers around Cheatcodes to avoid footguns -abstract contract Test is DSTest, Script { - uint256 internal constant UINT256_MAX = - 115792089237316195423570985008687907853269984665640564039457584007913129639935; - - /*////////////////////////////////////////////////////////////////////////// - STD-LOGS - //////////////////////////////////////////////////////////////////////////*/ - - event log_array(uint256[] val); - event log_array(int256[] val); - event log_array(address[] val); - event log_named_array(string key, uint256[] val); - event log_named_array(string key, int256[] val); - event log_named_array(string key, address[] val); - - /*////////////////////////////////////////////////////////////////////////// - STD-ASSERTIONS - //////////////////////////////////////////////////////////////////////////*/ - - function fail(string memory err) internal virtual { - emit log_named_string("Error", err); - fail(); - } - - function assertFalse(bool data) internal virtual { - assertTrue(!data); - } - - function assertFalse(bool data, string memory err) internal virtual { - assertTrue(!data, err); - } - - function assertEq(bool a, bool b) internal { - if (a != b) { - emit log("Error: a == b not satisfied [bool]"); - emit log_named_string(" Expected", b ? "true" : "false"); - emit log_named_string(" Actual", a ? "true" : "false"); - fail(); - } - } - - function assertEq(bool a, bool b, string memory err) internal { - if (a != b) { - emit log_named_string("Error", err); - assertEq(a, b); - } - } - - function assertEq(bytes memory a, bytes memory b) internal { - assertEq0(a, b); - } - - function assertEq(bytes memory a, bytes memory b, string memory err) internal { - assertEq0(a, b, err); - } - - function assertEq(uint256[] memory a, uint256[] memory b) internal { - bool inputsEq; - /// @solidity memory-safe-assembly - assembly { - inputsEq := - eq(keccak256(a, shl(5, add(mload(a), 1))), keccak256(b, shl(5, add(mload(b), 1)))) - } - if (!inputsEq) { - emit log("Error: a == b not satisfied [uint[]]"); - emit log_named_array(" Expected", b); - emit log_named_array(" Actual", a); - fail(); - } - } - - function assertEq(int256[] memory a, int256[] memory b) internal { - bool inputsEq; - /// @solidity memory-safe-assembly - assembly { - inputsEq := - eq(keccak256(a, shl(5, add(mload(a), 1))), keccak256(b, shl(5, add(mload(b), 1)))) - } - if (!inputsEq) { - emit log("Error: a == b not satisfied [int[]]"); - emit log_named_array(" Expected", b); - emit log_named_array(" Actual", a); - fail(); - } - } - - function assertEq(address[] memory a, address[] memory b) internal { - bool inputsEq; - /// @solidity memory-safe-assembly - assembly { - inputsEq := - eq(keccak256(a, shl(5, add(mload(a), 1))), keccak256(b, shl(5, add(mload(b), 1)))) - } - if (!inputsEq) { - emit log("Error: a == b not satisfied [address[]]"); - emit log_named_array(" Expected", b); - emit log_named_array(" Actual", a); - fail(); - } - } - - function assertEq(uint256[] memory a, uint256[] memory b, string memory err) internal { - bool inputsEq; - /// @solidity memory-safe-assembly - assembly { - inputsEq := - eq(keccak256(a, shl(5, add(mload(a), 1))), keccak256(b, shl(5, add(mload(b), 1)))) - } - if (!inputsEq) { - emit log_named_string("Error", err); - assertEq(a, b); - } - } - - function assertEq(int256[] memory a, int256[] memory b, string memory err) internal { - bool inputsEq; - /// @solidity memory-safe-assembly - assembly { - inputsEq := - eq(keccak256(a, shl(5, add(mload(a), 1))), keccak256(b, shl(5, add(mload(b), 1)))) - } - if (!inputsEq) { - emit log_named_string("Error", err); - assertEq(a, b); - } - } - - function assertEq(address[] memory a, address[] memory b, string memory err) internal { - bool inputsEq; - /// @solidity memory-safe-assembly - assembly { - inputsEq := - eq(keccak256(a, shl(5, add(mload(a), 1))), keccak256(b, shl(5, add(mload(b), 1)))) - } - if (!inputsEq) { - emit log_named_string("Error", err); - assertEq(a, b); - } - } -} - -/*////////////////////////////////////////////////////////////////////////// - STD-ERRORS -//////////////////////////////////////////////////////////////////////////*/ - -library stdError { - bytes public constant assertionError = abi.encodeWithSignature("Panic(uint256)", 0x01); - bytes public constant arithmeticError = abi.encodeWithSignature("Panic(uint256)", 0x11); - bytes public constant divisionError = abi.encodeWithSignature("Panic(uint256)", 0x12); - bytes public constant enumConversionError = abi.encodeWithSignature("Panic(uint256)", 0x21); - bytes public constant encodeStorageError = abi.encodeWithSignature("Panic(uint256)", 0x22); - bytes public constant popError = abi.encodeWithSignature("Panic(uint256)", 0x31); - bytes public constant indexOOBError = abi.encodeWithSignature("Panic(uint256)", 0x32); - bytes public constant memOverflowError = abi.encodeWithSignature("Panic(uint256)", 0x41); - bytes public constant zeroVarError = abi.encodeWithSignature("Panic(uint256)", 0x51); - // DEPRECATED: Use Vm's `expectRevert` without any arguments instead - bytes public constant lowLevelError = bytes(""); // `0x` -} diff --git a/lib/solady/test/utils/forge-std/Vm.sol b/lib/solady/test/utils/forge-std/Vm.sol deleted file mode 100644 index 9d2d2f6..0000000 --- a/lib/solady/test/utils/forge-std/Vm.sol +++ /dev/null @@ -1,525 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity >=0.4.22 <0.9.0; - -interface VmSafe { - struct Log { - bytes32[] topics; - bytes data; - address emitter; - } - - struct Rpc { - string key; - string url; - } - - struct FsMetadata { - bool isDir; - bool isSymlink; - uint256 length; - bool readOnly; - uint256 modified; - uint256 accessed; - uint256 created; - } - - // Loads a storage slot from an address - function load(address target, bytes32 slot) external view returns (bytes32 data); - // Signs data - function sign(uint256 privateKey, bytes32 digest) - external - pure - returns (uint8 v, bytes32 r, bytes32 s); - // Gets the address for a given private key - function addr(uint256 privateKey) external pure returns (address keyAddr); - // Gets the nonce of an account - function getNonce(address account) external view returns (uint64 nonce); - // Performs a foreign function call via the terminal - function ffi(string[] calldata commandInput) external returns (bytes memory result); - // Sets environment variables - function setEnv(string calldata name, string calldata value) external; - // Reads environment variables, (name) => (value) - function envBool(string calldata name) external view returns (bool value); - function envUint(string calldata name) external view returns (uint256 value); - function envInt(string calldata name) external view returns (int256 value); - function envAddress(string calldata name) external view returns (address value); - function envBytes32(string calldata name) external view returns (bytes32 value); - function envString(string calldata name) external view returns (string memory value); - function envBytes(string calldata name) external view returns (bytes memory value); - // Reads environment variables as arrays - function envBool(string calldata name, string calldata delim) - external - view - returns (bool[] memory value); - function envUint(string calldata name, string calldata delim) - external - view - returns (uint256[] memory value); - function envInt(string calldata name, string calldata delim) - external - view - returns (int256[] memory value); - function envAddress(string calldata name, string calldata delim) - external - view - returns (address[] memory value); - function envBytes32(string calldata name, string calldata delim) - external - view - returns (bytes32[] memory value); - function envString(string calldata name, string calldata delim) - external - view - returns (string[] memory value); - function envBytes(string calldata name, string calldata delim) - external - view - returns (bytes[] memory value); - // Read environment variables with default value - function envOr(string calldata name, bool defaultValue) external returns (bool value); - function envOr(string calldata name, uint256 defaultValue) external returns (uint256 value); - function envOr(string calldata name, int256 defaultValue) external returns (int256 value); - function envOr(string calldata name, address defaultValue) external returns (address value); - function envOr(string calldata name, bytes32 defaultValue) external returns (bytes32 value); - function envOr(string calldata name, string calldata defaultValue) - external - returns (string memory value); - function envOr(string calldata name, bytes calldata defaultValue) - external - returns (bytes memory value); - // Read environment variables as arrays with default value - function envOr(string calldata name, string calldata delim, bool[] calldata defaultValue) - external - returns (bool[] memory value); - function envOr(string calldata name, string calldata delim, uint256[] calldata defaultValue) - external - returns (uint256[] memory value); - function envOr(string calldata name, string calldata delim, int256[] calldata defaultValue) - external - returns (int256[] memory value); - function envOr(string calldata name, string calldata delim, address[] calldata defaultValue) - external - returns (address[] memory value); - function envOr(string calldata name, string calldata delim, bytes32[] calldata defaultValue) - external - returns (bytes32[] memory value); - function envOr(string calldata name, string calldata delim, string[] calldata defaultValue) - external - returns (string[] memory value); - function envOr(string calldata name, string calldata delim, bytes[] calldata defaultValue) - external - returns (bytes[] memory value); - // Records all storage reads and writes - function record() external; - // Gets all accessed reads and write slot from a recording session, for a given address - function accesses(address target) - external - returns (bytes32[] memory readSlots, bytes32[] memory writeSlots); - // Gets the _creation_ bytecode from an artifact file. Takes in the relative path to the json file - function getCode(string calldata artifactPath) - external - view - returns (bytes memory creationBytecode); - // Gets the _deployed_ bytecode from an artifact file. Takes in the relative path to the json file - function getDeployedCode(string calldata artifactPath) - external - view - returns (bytes memory runtimeBytecode); - // Labels an address in call traces - function label(address account, string calldata newLabel) external; - // Using the address that calls the test contract, has the next call (at this call depth only) create a transaction that can later be signed and sent onchain - function broadcast() external; - // Has the next call (at this call depth only) create a transaction with the address provided as the sender that can later be signed and sent onchain - function broadcast(address signer) external; - // Has the next call (at this call depth only) create a transaction with the private key provided as the sender that can later be signed and sent onchain - function broadcast(uint256 privateKey) external; - // Using the address that calls the test contract, has all subsequent calls (at this call depth only) create transactions that can later be signed and sent onchain - function startBroadcast() external; - // Has all subsequent calls (at this call depth only) create transactions with the address provided that can later be signed and sent onchain - function startBroadcast(address signer) external; - // Has all subsequent calls (at this call depth only) create transactions with the private key provided that can later be signed and sent onchain - function startBroadcast(uint256 privateKey) external; - // Stops collecting onchain transactions - function stopBroadcast() external; - // Reads the entire content of file to string - function readFile(string calldata path) external view returns (string memory data); - // Reads the entire content of file as binary. Path is relative to the project root. - function readFileBinary(string calldata path) external view returns (bytes memory data); - // Get the path of the current project root - function projectRoot() external view returns (string memory path); - // Get the metadata for a file/directory - function fsMetadata(string calldata fileOrDir) external returns (FsMetadata memory metadata); - // Reads next line of file to string - function readLine(string calldata path) external view returns (string memory line); - // Writes data to file, creating a file if it does not exist, and entirely replacing its contents if it does. - function writeFile(string calldata path, string calldata data) external; - // Writes binary data to a file, creating a file if it does not exist, and entirely replacing its contents if it does. - // Path is relative to the project root. - function writeFileBinary(string calldata path, bytes calldata data) external; - // Writes line to file, creating a file if it does not exist. - function writeLine(string calldata path, string calldata data) external; - // Closes file for reading, resetting the offset and allowing to read it from beginning with readLine. - function closeFile(string calldata path) external; - // Removes file. This cheatcode will revert in the following situations, but is not limited to just these cases: - // - Path points to a directory. - // - The file doesn't exist. - // - The user lacks permissions to remove the file. - function removeFile(string calldata path) external; - // Convert values to a string - function toString(address value) external pure returns (string memory stringifiedValue); - function toString(bytes calldata value) - external - pure - returns (string memory stringifiedValue); - function toString(bytes32 value) external pure returns (string memory stringifiedValue); - function toString(bool value) external pure returns (string memory stringifiedValue); - function toString(uint256 value) external pure returns (string memory stringifiedValue); - function toString(int256 value) external pure returns (string memory stringifiedValue); - // Convert values from a string - function parseBytes(string calldata stringifiedValue) - external - pure - returns (bytes memory parsedValue); - function parseAddress(string calldata stringifiedValue) - external - pure - returns (address parsedValue); - function parseUint(string calldata stringifiedValue) - external - pure - returns (uint256 parsedValue); - function parseInt(string calldata stringifiedValue) - external - pure - returns (int256 parsedValue); - function parseBytes32(string calldata stringifiedValue) - external - pure - returns (bytes32 parsedValue); - function parseBool(string calldata stringifiedValue) external pure returns (bool parsedValue); - // Record all the transaction logs - function recordLogs() external; - // Gets all the recorded logs - function getRecordedLogs() external returns (Log[] memory logs); - // Derive a private key from a provided mnenomic string (or mnenomic file path) at the derivation path m/44'/60'/0'/0/{index} - function deriveKey(string calldata mnemonic, uint32 index) - external - pure - returns (uint256 privateKey); - // Derive a private key from a provided mnenomic string (or mnenomic file path) at {derivationPath}{index} - function deriveKey(string calldata mnemonic, string calldata derivationPath, uint32 index) - external - pure - returns (uint256 privateKey); - // Adds a private key to the local forge wallet and returns the address - function rememberKey(uint256 privateKey) external returns (address keyAddr); - // - // parseJson - // - // ---- - // In case the returned value is a JSON object, it's encoded as a ABI-encoded tuple. As JSON objects - // don't have the notion of ordered, but tuples do, they JSON object is encoded with it's fields ordered in - // ALPHABETICAL order. That means that in order to successfully decode the tuple, we need to define a tuple that - // encodes the fields in the same order, which is alphabetical. In the case of Solidity structs, they are encoded - // as tuples, with the attributes in the order in which they are defined. - // For example: json = { 'a': 1, 'b': 0xa4tb......3xs} - // a: uint256 - // b: address - // To decode that json, we need to define a struct or a tuple as follows: - // struct json = { uint256 a; address b; } - // If we defined a json struct with the opposite order, meaning placing the address b first, it would try to - // decode the tuple in that order, and thus fail. - // ---- - // Given a string of JSON, return it as ABI-encoded - function parseJson(string calldata json, string calldata key) - external - pure - returns (bytes memory abiEncodedData); - function parseJson(string calldata json) external pure returns (bytes memory abiEncodedData); - - // The following parseJson cheatcodes will do type coercion, for the type that they indicate. - // For example, parseJsonUint will coerce all values to a uint256. That includes stringified numbers '12' - // and hex numbers '0xEF'. - // Type coercion works ONLY for discrete values or arrays. That means that the key must return a value or array, not - // a JSON object. - function parseJsonUint(string calldata, string calldata) external returns (uint256); - function parseJsonUintArray(string calldata, string calldata) - external - returns (uint256[] memory); - function parseJsonInt(string calldata, string calldata) external returns (int256); - function parseJsonIntArray(string calldata, string calldata) - external - returns (int256[] memory); - function parseJsonBool(string calldata, string calldata) external returns (bool); - function parseJsonBoolArray(string calldata, string calldata) - external - returns (bool[] memory); - function parseJsonAddress(string calldata, string calldata) external returns (address); - function parseJsonAddressArray(string calldata, string calldata) - external - returns (address[] memory); - function parseJsonString(string calldata, string calldata) external returns (string memory); - function parseJsonStringArray(string calldata, string calldata) - external - returns (string[] memory); - function parseJsonBytes(string calldata, string calldata) external returns (bytes memory); - function parseJsonBytesArray(string calldata, string calldata) - external - returns (bytes[] memory); - function parseJsonBytes32(string calldata, string calldata) external returns (bytes32); - function parseJsonBytes32Array(string calldata, string calldata) - external - returns (bytes32[] memory); - - // Serialize a key and value to a JSON object stored in-memory that can be later written to a file - // It returns the stringified version of the specific JSON file up to that moment. - function serializeBool(string calldata objectKey, string calldata valueKey, bool value) - external - returns (string memory json); - function serializeUint(string calldata objectKey, string calldata valueKey, uint256 value) - external - returns (string memory json); - function serializeInt(string calldata objectKey, string calldata valueKey, int256 value) - external - returns (string memory json); - function serializeAddress(string calldata objectKey, string calldata valueKey, address value) - external - returns (string memory json); - function serializeBytes32(string calldata objectKey, string calldata valueKey, bytes32 value) - external - returns (string memory json); - function serializeString( - string calldata objectKey, - string calldata valueKey, - string calldata value - ) external returns (string memory json); - function serializeBytes( - string calldata objectKey, - string calldata valueKey, - bytes calldata value - ) external returns (string memory json); - - function serializeBool( - string calldata objectKey, - string calldata valueKey, - bool[] calldata values - ) external returns (string memory json); - function serializeUint( - string calldata objectKey, - string calldata valueKey, - uint256[] calldata values - ) external returns (string memory json); - function serializeInt( - string calldata objectKey, - string calldata valueKey, - int256[] calldata values - ) external returns (string memory json); - function serializeAddress( - string calldata objectKey, - string calldata valueKey, - address[] calldata values - ) external returns (string memory json); - function serializeBytes32( - string calldata objectKey, - string calldata valueKey, - bytes32[] calldata values - ) external returns (string memory json); - function serializeString( - string calldata objectKey, - string calldata valueKey, - string[] calldata values - ) external returns (string memory json); - function serializeBytes( - string calldata objectKey, - string calldata valueKey, - bytes[] calldata values - ) external returns (string memory json); - - // - // writeJson - // - // ---- - // Write a serialized JSON object to a file. If the file exists, it will be overwritten. - // Let's assume we want to write the following JSON to a file: - // - // { "boolean": true, "number": 342, "object": { "title": "finally json serialization" } } - // - // ``` - // string memory json1 = "some key"; - // vm.serializeBool(json1, "boolean", true); - // vm.serializeBool(json1, "number", uint256(342)); - // json2 = "some other key"; - // string memory output = vm.serializeString(json2, "title", "finally json serialization"); - // string memory finalJson = vm.serialize(json1, "object", output); - // vm.writeJson(finalJson, "./output/example.json"); - // ``` - // The critical insight is that every invocation of serialization will return the stringified version of the JSON - // up to that point. That means we can construct arbitrary JSON objects and then use the return stringified version - // to serialize them as values to another JSON object. - // - // json1 and json2 are simply keys used by the backend to keep track of the objects. So vm.serializeJson(json1,..) - // will find the object in-memory that is keyed by "some key". - function writeJson(string calldata json, string calldata path) external; - // Write a serialized JSON object to an **existing** JSON file, replacing a value with key = - // This is useful to replace a specific value of a JSON file, without having to parse the entire thing - function writeJson(string calldata json, string calldata path, string calldata valueKey) - external; - // Returns the RPC url for the given alias - function rpcUrl(string calldata rpcAlias) external view returns (string memory json); - // Returns all rpc urls and their aliases `[alias, url][]` - function rpcUrls() external view returns (string[2][] memory urls); - // Returns all rpc urls and their aliases as structs. - function rpcUrlStructs() external view returns (Rpc[] memory urls); - // If the condition is false, discard this run's fuzz inputs and generate new ones. - function assume(bool condition) external pure; - // Pauses gas metering (i.e. gas usage is not counted). Noop if already paused. - function pauseGasMetering() external; - // Resumes gas metering (i.e. gas usage is counted again). Noop if already on. - function resumeGasMetering() external; -} - -interface Vm is VmSafe { - // Sets block.timestamp - function warp(uint256 newTimestamp) external; - // Sets block.height - function roll(uint256 newHeight) external; - // Sets block.basefee - function fee(uint256 newBasefee) external; - // Sets block.difficulty - function difficulty(uint256 newDifficulty) external; - // Sets block.chainid - function chainId(uint256 newChainId) external; - // Stores a value to an address' storage slot. - function store(address target, bytes32 slot, bytes32 value) external; - // Sets the nonce of an account; must be higher than the current nonce of the account - function setNonce(address account, uint64 newNonce) external; - // Sets the *next* call's msg.sender to be the input address - function prank(address msgSender) external; - // Sets all subsequent calls' msg.sender to be the input address until `stopPrank` is called - function startPrank(address msgSender) external; - // Sets the *next* call's msg.sender to be the input address, and the tx.origin to be the second input - function prank(address msgSender, address txOrigin) external; - // Sets all subsequent calls' msg.sender to be the input address until `stopPrank` is called, and the tx.origin to be the second input - function startPrank(address msgSender, address txOrigin) external; - // Resets subsequent calls' msg.sender to be `address(this)` - function stopPrank() external; - // Sets an address' balance - function deal(address account, uint256 newBalance) external; - // Sets an address' code - function etch(address target, bytes calldata newRuntimeBytecode) external; - // Expects an error on next call - function expectRevert(bytes calldata revertData) external; - function expectRevert(bytes4 revertData) external; - function expectRevert() external; - - // Prepare an expected log with all four checks enabled. - // Call this function, then emit an event, then call a function. Internally after the call, we check if - // logs were emitted in the expected order with the expected topics and data. - // Second form also checks supplied address against emitting contract. - function expectEmit() external; - function expectEmit(address) external; - - // Prepare an expected log with (bool checkTopic1, bool checkTopic2, bool checkTopic3, bool checkData). - // Call this function, then emit an event, then call a function. Internally after the call, we check if - // logs were emitted in the expected order with the expected topics and data (as specified by the booleans). - // Second form also checks supplied address against emitting contract. - function expectEmit(bool checkTopic1, bool checkTopic2, bool checkTopic3, bool checkData) - external; - function expectEmit( - bool checkTopic1, - bool checkTopic2, - bool checkTopic3, - bool checkData, - address emitter - ) external; - - // Mocks a call to an address, returning specified data. - // Calldata can either be strict or a partial match, e.g. if you only - // pass a Solidity selector to the expected calldata, then the entire Solidity - // function will be mocked. - function mockCall(address callee, bytes calldata data, bytes calldata returnData) external; - // Mocks a call to an address with a specific msg.value, returning specified data. - // Calldata match takes precedence over msg.value in case of ambiguity. - function mockCall( - address callee, - uint256 msgValue, - bytes calldata data, - bytes calldata returnData - ) external; - // Clears all mocked calls - function clearMockedCalls() external; - // Expects a call to an address with the specified calldata. - // Calldata can either be a strict or a partial match - function expectCall(address callee, bytes calldata data) external; - // Expects a call to an address with the specified msg.value and calldata - function expectCall(address callee, uint256 msgValue, bytes calldata data) external; - // Expect a call to an address with the specified msg.value, gas, and calldata. - function expectCall(address callee, uint256 msgValue, uint64 gas, bytes calldata data) - external; - // Expect a call to an address with the specified msg.value and calldata, and a *minimum* amount of gas. - function expectCallMinGas(address callee, uint256 msgValue, uint64 minGas, bytes calldata data) - external; - // Sets block.coinbase - function coinbase(address newCoinbase) external; - // Snapshot the current state of the evm. - // Returns the id of the snapshot that was created. - // To revert a snapshot use `revertTo` - function snapshot() external returns (uint256 snapshotId); - // Revert the state of the EVM to a previous snapshot - // Takes the snapshot id to revert to. - // This deletes the snapshot and all snapshots taken after the given snapshot id. - function revertTo(uint256 snapshotId) external returns (bool success); - // Creates a new fork with the given endpoint and block and returns the identifier of the fork - function createFork(string calldata urlOrAlias, uint256 blockNumber) - external - returns (uint256 forkId); - // Creates a new fork with the given endpoint and the _latest_ block and returns the identifier of the fork - function createFork(string calldata urlOrAlias) external returns (uint256 forkId); - // Creates a new fork with the given endpoint and at the block the given transaction was mined in, replays all transaction mined in the block before the transaction, - // and returns the identifier of the fork - function createFork(string calldata urlOrAlias, bytes32 txHash) - external - returns (uint256 forkId); - // Creates _and_ also selects a new fork with the given endpoint and block and returns the identifier of the fork - function createSelectFork(string calldata urlOrAlias, uint256 blockNumber) - external - returns (uint256 forkId); - // Creates _and_ also selects new fork with the given endpoint and at the block the given transaction was mined in, replays all transaction mined in the block before - // the transaction, returns the identifier of the fork - function createSelectFork(string calldata urlOrAlias, bytes32 txHash) - external - returns (uint256 forkId); - // Creates _and_ also selects a new fork with the given endpoint and the latest block and returns the identifier of the fork - function createSelectFork(string calldata urlOrAlias) external returns (uint256 forkId); - // Takes a fork identifier created by `createFork` and sets the corresponding forked state as active. - function selectFork(uint256 forkId) external; - /// Returns the identifier of the currently active fork. Reverts if no fork is currently active. - function activeFork() external view returns (uint256 forkId); - // Updates the currently active fork to given block number - // This is similar to `roll` but for the currently active fork - function rollFork(uint256 blockNumber) external; - // Updates the currently active fork to given transaction - // this will `rollFork` with the number of the block the transaction was mined in and replays all transaction mined before it in the block - function rollFork(bytes32 txHash) external; - // Updates the given fork to given block number - function rollFork(uint256 forkId, uint256 blockNumber) external; - // Updates the given fork to block number of the given transaction and replays all transaction mined before it in the block - function rollFork(uint256 forkId, bytes32 txHash) external; - // Marks that the account(s) should use persistent storage across fork swaps in a multifork setup - // Meaning, changes made to the state of this account will be kept when switching forks - function makePersistent(address account) external; - function makePersistent(address account0, address account1) external; - function makePersistent(address account0, address account1, address account2) external; - function makePersistent(address[] calldata accounts) external; - // Revokes persistent status from the address, previously added via `makePersistent` - function revokePersistent(address account) external; - function revokePersistent(address[] calldata accounts) external; - // Returns true if the account is marked as persistent - function isPersistent(address account) external view returns (bool persistent); - // In forking mode, explicitly grant the given address cheatcode access - function allowCheatcodes(address account) external; - // Fetches the given transaction from the active fork and executes it on the current state - function transact(bytes32 txHash) external; - // Fetches the given transaction from the given fork and executes it on the current state - function transact(uint256 forkId, bytes32 txHash) external; -} diff --git a/lib/solady/test/utils/forge-std/console.sol b/lib/solady/test/utils/forge-std/console.sol deleted file mode 100644 index 49850e1..0000000 --- a/lib/solady/test/utils/forge-std/console.sol +++ /dev/null @@ -1,1581 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity >=0.4.22 <0.9.0; - -library console { - address constant CONSOLE_ADDRESS = address(0x000000000000000000636F6e736F6c652e6c6f67); - - function _sendLogPayload(bytes memory payload) private view { - uint256 payloadLength = payload.length; - address consoleAddress = CONSOLE_ADDRESS; - /// @solidity memory-safe-assembly - assembly { - let payloadStart := add(payload, 32) - let r := staticcall(gas(), consoleAddress, payloadStart, payloadLength, 0, 0) - } - } - - function log() internal view { - _sendLogPayload(abi.encodeWithSignature("log()")); - } - - function logInt(int256 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(int)", p0)); - } - - function logUint(uint256 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint)", p0)); - } - - function logString(string memory p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string)", p0)); - } - - function logBool(bool p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool)", p0)); - } - - function logAddress(address p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address)", p0)); - } - - function logBytes(bytes memory p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes)", p0)); - } - - function logBytes1(bytes1 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes1)", p0)); - } - - function logBytes2(bytes2 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes2)", p0)); - } - - function logBytes3(bytes3 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes3)", p0)); - } - - function logBytes4(bytes4 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes4)", p0)); - } - - function logBytes5(bytes5 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes5)", p0)); - } - - function logBytes6(bytes6 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes6)", p0)); - } - - function logBytes7(bytes7 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes7)", p0)); - } - - function logBytes8(bytes8 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes8)", p0)); - } - - function logBytes9(bytes9 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes9)", p0)); - } - - function logBytes10(bytes10 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes10)", p0)); - } - - function logBytes11(bytes11 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes11)", p0)); - } - - function logBytes12(bytes12 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes12)", p0)); - } - - function logBytes13(bytes13 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes13)", p0)); - } - - function logBytes14(bytes14 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes14)", p0)); - } - - function logBytes15(bytes15 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes15)", p0)); - } - - function logBytes16(bytes16 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes16)", p0)); - } - - function logBytes17(bytes17 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes17)", p0)); - } - - function logBytes18(bytes18 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes18)", p0)); - } - - function logBytes19(bytes19 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes19)", p0)); - } - - function logBytes20(bytes20 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes20)", p0)); - } - - function logBytes21(bytes21 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes21)", p0)); - } - - function logBytes22(bytes22 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes22)", p0)); - } - - function logBytes23(bytes23 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes23)", p0)); - } - - function logBytes24(bytes24 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes24)", p0)); - } - - function logBytes25(bytes25 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes25)", p0)); - } - - function logBytes26(bytes26 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes26)", p0)); - } - - function logBytes27(bytes27 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes27)", p0)); - } - - function logBytes28(bytes28 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes28)", p0)); - } - - function logBytes29(bytes29 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes29)", p0)); - } - - function logBytes30(bytes30 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes30)", p0)); - } - - function logBytes31(bytes31 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes31)", p0)); - } - - function logBytes32(bytes32 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bytes32)", p0)); - } - - function log(uint256 p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint)", p0)); - } - - function log(string memory p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string)", p0)); - } - - function log(bool p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool)", p0)); - } - - function log(address p0) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address)", p0)); - } - - function log(uint256 p0, uint256 p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint)", p0, p1)); - } - - function log(uint256 p0, string memory p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string)", p0, p1)); - } - - function log(uint256 p0, bool p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool)", p0, p1)); - } - - function log(uint256 p0, address p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address)", p0, p1)); - } - - function log(string memory p0, uint256 p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint)", p0, p1)); - } - - function log(string memory p0, string memory p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string)", p0, p1)); - } - - function log(string memory p0, bool p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool)", p0, p1)); - } - - function log(string memory p0, address p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address)", p0, p1)); - } - - function log(bool p0, uint256 p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint)", p0, p1)); - } - - function log(bool p0, string memory p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string)", p0, p1)); - } - - function log(bool p0, bool p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool)", p0, p1)); - } - - function log(bool p0, address p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address)", p0, p1)); - } - - function log(address p0, uint256 p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint)", p0, p1)); - } - - function log(address p0, string memory p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string)", p0, p1)); - } - - function log(address p0, bool p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool)", p0, p1)); - } - - function log(address p0, address p1) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address)", p0, p1)); - } - - function log(uint256 p0, uint256 p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,uint)", p0, p1, p2)); - } - - function log(uint256 p0, uint256 p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,string)", p0, p1, p2)); - } - - function log(uint256 p0, uint256 p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,bool)", p0, p1, p2)); - } - - function log(uint256 p0, uint256 p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,address)", p0, p1, p2)); - } - - function log(uint256 p0, string memory p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,uint)", p0, p1, p2)); - } - - function log(uint256 p0, string memory p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,string)", p0, p1, p2)); - } - - function log(uint256 p0, string memory p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,bool)", p0, p1, p2)); - } - - function log(uint256 p0, string memory p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,address)", p0, p1, p2)); - } - - function log(uint256 p0, bool p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,uint)", p0, p1, p2)); - } - - function log(uint256 p0, bool p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,string)", p0, p1, p2)); - } - - function log(uint256 p0, bool p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,bool)", p0, p1, p2)); - } - - function log(uint256 p0, bool p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,address)", p0, p1, p2)); - } - - function log(uint256 p0, address p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,uint)", p0, p1, p2)); - } - - function log(uint256 p0, address p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,string)", p0, p1, p2)); - } - - function log(uint256 p0, address p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,bool)", p0, p1, p2)); - } - - function log(uint256 p0, address p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,address)", p0, p1, p2)); - } - - function log(string memory p0, uint256 p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,uint)", p0, p1, p2)); - } - - function log(string memory p0, uint256 p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,string)", p0, p1, p2)); - } - - function log(string memory p0, uint256 p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,bool)", p0, p1, p2)); - } - - function log(string memory p0, uint256 p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,address)", p0, p1, p2)); - } - - function log(string memory p0, string memory p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,uint)", p0, p1, p2)); - } - - function log(string memory p0, string memory p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,string)", p0, p1, p2)); - } - - function log(string memory p0, string memory p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,bool)", p0, p1, p2)); - } - - function log(string memory p0, string memory p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,address)", p0, p1, p2)); - } - - function log(string memory p0, bool p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,uint)", p0, p1, p2)); - } - - function log(string memory p0, bool p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,string)", p0, p1, p2)); - } - - function log(string memory p0, bool p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,bool)", p0, p1, p2)); - } - - function log(string memory p0, bool p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,address)", p0, p1, p2)); - } - - function log(string memory p0, address p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,uint)", p0, p1, p2)); - } - - function log(string memory p0, address p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,string)", p0, p1, p2)); - } - - function log(string memory p0, address p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,bool)", p0, p1, p2)); - } - - function log(string memory p0, address p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,address)", p0, p1, p2)); - } - - function log(bool p0, uint256 p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,uint)", p0, p1, p2)); - } - - function log(bool p0, uint256 p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,string)", p0, p1, p2)); - } - - function log(bool p0, uint256 p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,bool)", p0, p1, p2)); - } - - function log(bool p0, uint256 p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,address)", p0, p1, p2)); - } - - function log(bool p0, string memory p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,uint)", p0, p1, p2)); - } - - function log(bool p0, string memory p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,string)", p0, p1, p2)); - } - - function log(bool p0, string memory p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,bool)", p0, p1, p2)); - } - - function log(bool p0, string memory p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,address)", p0, p1, p2)); - } - - function log(bool p0, bool p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint)", p0, p1, p2)); - } - - function log(bool p0, bool p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,string)", p0, p1, p2)); - } - - function log(bool p0, bool p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool)", p0, p1, p2)); - } - - function log(bool p0, bool p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,address)", p0, p1, p2)); - } - - function log(bool p0, address p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,uint)", p0, p1, p2)); - } - - function log(bool p0, address p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,string)", p0, p1, p2)); - } - - function log(bool p0, address p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,bool)", p0, p1, p2)); - } - - function log(bool p0, address p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,address)", p0, p1, p2)); - } - - function log(address p0, uint256 p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,uint)", p0, p1, p2)); - } - - function log(address p0, uint256 p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,string)", p0, p1, p2)); - } - - function log(address p0, uint256 p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,bool)", p0, p1, p2)); - } - - function log(address p0, uint256 p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,address)", p0, p1, p2)); - } - - function log(address p0, string memory p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,uint)", p0, p1, p2)); - } - - function log(address p0, string memory p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,string)", p0, p1, p2)); - } - - function log(address p0, string memory p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,bool)", p0, p1, p2)); - } - - function log(address p0, string memory p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,address)", p0, p1, p2)); - } - - function log(address p0, bool p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,uint)", p0, p1, p2)); - } - - function log(address p0, bool p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,string)", p0, p1, p2)); - } - - function log(address p0, bool p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,bool)", p0, p1, p2)); - } - - function log(address p0, bool p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,address)", p0, p1, p2)); - } - - function log(address p0, address p1, uint256 p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address,uint)", p0, p1, p2)); - } - - function log(address p0, address p1, string memory p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address,string)", p0, p1, p2)); - } - - function log(address p0, address p1, bool p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address,bool)", p0, p1, p2)); - } - - function log(address p0, address p1, address p2) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address,address)", p0, p1, p2)); - } - - function log(uint256 p0, uint256 p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,uint,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,uint,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,uint,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,uint,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,string,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, string memory p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,string,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,string,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, string memory p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,string,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,bool,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,bool,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,bool,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,bool,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,address,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, address p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,address,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,address,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, uint256 p1, address p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,uint,address,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,uint,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,uint,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,uint,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,uint,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,string,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, string memory p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,string,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,string,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, string memory p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,string,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,bool,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,bool,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,bool,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,bool,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,address,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, address p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,address,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,address,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, string memory p1, address p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,string,address,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,uint,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,uint,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,uint,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,uint,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,string,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, string memory p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,string,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,string,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, string memory p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,string,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,bool,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,bool,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,bool,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,bool,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,address,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, address p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,address,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,address,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, bool p1, address p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,bool,address,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,uint,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,uint,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,uint,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,uint,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,string,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, string memory p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,string,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,string,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, string memory p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,string,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,bool,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,bool,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,bool,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,bool,address)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,address,uint)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, address p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,address,string)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(uint,address,address,bool)", p0, p1, p2, p3)); - } - - function log(uint256 p0, address p1, address p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(uint,address,address,address)", p0, p1, p2, p3) - ); - } - - function log(string memory p0, uint256 p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,uint,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,uint,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,uint,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,uint,address)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,string,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, string memory p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,string,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,string,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, string memory p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,string,address)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,bool,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,bool,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,bool,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,bool,address)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,address,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, address p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,address,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,address,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, uint256 p1, address p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,uint,address,address)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,uint,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,uint,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,uint,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,uint,address)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,string,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, string memory p2, string memory p3) - internal - view - { - _sendLogPayload(abi.encodeWithSignature("log(string,string,string,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,string,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, string memory p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(string,string,string,address)", p0, p1, p2, p3) - ); - } - - function log(string memory p0, string memory p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,bool,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,bool,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,bool,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,bool,address)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,address,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, address p2, string memory p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(string,string,address,string)", p0, p1, p2, p3) - ); - } - - function log(string memory p0, string memory p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,string,address,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, string memory p1, address p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(string,string,address,address)", p0, p1, p2, p3) - ); - } - - function log(string memory p0, bool p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,uint,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,uint,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,uint,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,uint,address)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,string,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, string memory p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,string,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,string,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, string memory p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,string,address)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,address)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,address,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, address p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,address,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,address,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, bool p1, address p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,bool,address,address)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,uint,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,uint,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,uint,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,uint,address)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,string,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, string memory p2, string memory p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(string,address,string,string)", p0, p1, p2, p3) - ); - } - - function log(string memory p0, address p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,string,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, string memory p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(string,address,string,address)", p0, p1, p2, p3) - ); - } - - function log(string memory p0, address p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,bool,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,bool,string)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,bool,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,bool,address)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,address,uint)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, address p2, string memory p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(string,address,address,string)", p0, p1, p2, p3) - ); - } - - function log(string memory p0, address p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(string,address,address,bool)", p0, p1, p2, p3)); - } - - function log(string memory p0, address p1, address p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(string,address,address,address)", p0, p1, p2, p3) - ); - } - - function log(bool p0, uint256 p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,uint,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,uint,string)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,uint,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,uint,address)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,string,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, string memory p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,string,string)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,string,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, string memory p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,string,address)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,bool,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,bool,string)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,bool,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,bool,address)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,address,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, address p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,address,string)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,address,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, uint256 p1, address p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,uint,address,address)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,uint,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,uint,string)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,uint,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,uint,address)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,string,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, string memory p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,string,string)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,string,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, string memory p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,string,address)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,string)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,address)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,address,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, address p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,address,string)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,address,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, string memory p1, address p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,string,address,address)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint,string)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint,address)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, string memory p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,string)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, string memory p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,address)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,string)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,address)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, address p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,string)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, bool p1, address p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,address)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,uint,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,uint,string)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,uint,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,uint,address)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,string,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, string memory p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,string,string)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,string,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, string memory p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,string,address)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,string)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,address)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,address,uint)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, address p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,address,string)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(bool,address,address,bool)", p0, p1, p2, p3)); - } - - function log(bool p0, address p1, address p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(bool,address,address,address)", p0, p1, p2, p3) - ); - } - - function log(address p0, uint256 p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,uint,uint)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,uint,string)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,uint,bool)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,uint,address)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,string,uint)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, string memory p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,string,string)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,string,bool)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, string memory p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,string,address)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,bool,uint)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,bool,string)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,bool,bool)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,bool,address)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,address,uint)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, address p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,address,string)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,uint,address,bool)", p0, p1, p2, p3)); - } - - function log(address p0, uint256 p1, address p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,uint,address,address)", p0, p1, p2, p3) - ); - } - - function log(address p0, string memory p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,uint,uint)", p0, p1, p2, p3)); - } - - function log(address p0, string memory p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,uint,string)", p0, p1, p2, p3)); - } - - function log(address p0, string memory p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,uint,bool)", p0, p1, p2, p3)); - } - - function log(address p0, string memory p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,uint,address)", p0, p1, p2, p3)); - } - - function log(address p0, string memory p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,string,uint)", p0, p1, p2, p3)); - } - - function log(address p0, string memory p1, string memory p2, string memory p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,string,string,string)", p0, p1, p2, p3) - ); - } - - function log(address p0, string memory p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,string,bool)", p0, p1, p2, p3)); - } - - function log(address p0, string memory p1, string memory p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,string,string,address)", p0, p1, p2, p3) - ); - } - - function log(address p0, string memory p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,bool,uint)", p0, p1, p2, p3)); - } - - function log(address p0, string memory p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,bool,string)", p0, p1, p2, p3)); - } - - function log(address p0, string memory p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,bool,bool)", p0, p1, p2, p3)); - } - - function log(address p0, string memory p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,bool,address)", p0, p1, p2, p3)); - } - - function log(address p0, string memory p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,address,uint)", p0, p1, p2, p3)); - } - - function log(address p0, string memory p1, address p2, string memory p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,string,address,string)", p0, p1, p2, p3) - ); - } - - function log(address p0, string memory p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,string,address,bool)", p0, p1, p2, p3)); - } - - function log(address p0, string memory p1, address p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,string,address,address)", p0, p1, p2, p3) - ); - } - - function log(address p0, bool p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,uint,uint)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,uint,string)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,uint,bool)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, uint256 p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,uint,address)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,string,uint)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, string memory p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,string,string)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,string,bool)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, string memory p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,string,address)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,uint)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,string)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,bool)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, bool p2, address p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,address)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, address p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,address,uint)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, address p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,address,string)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, address p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,bool,address,bool)", p0, p1, p2, p3)); - } - - function log(address p0, bool p1, address p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,bool,address,address)", p0, p1, p2, p3) - ); - } - - function log(address p0, address p1, uint256 p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address,uint,uint)", p0, p1, p2, p3)); - } - - function log(address p0, address p1, uint256 p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address,uint,string)", p0, p1, p2, p3)); - } - - function log(address p0, address p1, uint256 p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address,uint,bool)", p0, p1, p2, p3)); - } - - function log(address p0, address p1, uint256 p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,address,uint,address)", p0, p1, p2, p3) - ); - } - - function log(address p0, address p1, string memory p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address,string,uint)", p0, p1, p2, p3)); - } - - function log(address p0, address p1, string memory p2, string memory p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,address,string,string)", p0, p1, p2, p3) - ); - } - - function log(address p0, address p1, string memory p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address,string,bool)", p0, p1, p2, p3)); - } - - function log(address p0, address p1, string memory p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,address,string,address)", p0, p1, p2, p3) - ); - } - - function log(address p0, address p1, bool p2, uint256 p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address,bool,uint)", p0, p1, p2, p3)); - } - - function log(address p0, address p1, bool p2, string memory p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address,bool,string)", p0, p1, p2, p3)); - } - - function log(address p0, address p1, bool p2, bool p3) internal view { - _sendLogPayload(abi.encodeWithSignature("log(address,address,bool,bool)", p0, p1, p2, p3)); - } - - function log(address p0, address p1, bool p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,address,bool,address)", p0, p1, p2, p3) - ); - } - - function log(address p0, address p1, address p2, uint256 p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,address,address,uint)", p0, p1, p2, p3) - ); - } - - function log(address p0, address p1, address p2, string memory p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,address,address,string)", p0, p1, p2, p3) - ); - } - - function log(address p0, address p1, address p2, bool p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,address,address,bool)", p0, p1, p2, p3) - ); - } - - function log(address p0, address p1, address p2, address p3) internal view { - _sendLogPayload( - abi.encodeWithSignature("log(address,address,address,address)", p0, p1, p2, p3) - ); - } -} diff --git a/lib/solady/test/utils/mocks/MockCd.sol b/lib/solady/test/utils/mocks/MockCd.sol deleted file mode 100644 index 8dbb4d7..0000000 --- a/lib/solady/test/utils/mocks/MockCd.sol +++ /dev/null @@ -1,80 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {LibZip} from "../../../src/utils/LibZip.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockCd { - error Hash(bytes32 h); - - bytes32 public dataHash; - bytes32 public numbersHash; - uint256 public lastCallvalue; - address public lastCaller; - - function storeDataHash(bytes calldata data, bool success) - external - payable - returns (bytes32 result) - { - result = keccak256(data); - if (!success) { - revert Hash(result); - } - dataHash = result; - lastCallvalue = msg.value; - lastCaller = msg.sender; - } - - function storeNumbersHash(uint256[] calldata numbers, bool success) - external - payable - returns (bytes32 result) - { - result = keccak256(abi.encode(numbers)); - if (!success) { - revert Hash(result); - } - numbersHash = result; - lastCallvalue = msg.value; - lastCaller = msg.sender; - } - - receive() external payable { - LibZip.cdFallback(); - } - - fallback() external payable { - LibZip.cdFallback(); - } -} - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockCdFallbackDecompressor { - receive() external payable { - _interceptCdFallback(); - LibZip.cdFallback(); - } - - fallback() external payable { - _interceptCdFallback(); - LibZip.cdFallback(); - } - - function _interceptCdFallback() internal { - assembly { - if iszero(calldatasize()) { - mstore(0x00, keccak256(0x00, 0x00)) - return(0x00, 0x20) - } - if sload(0) { - calldatacopy(0x00, 0x00, calldatasize()) - mstore(0x00, keccak256(0x00, calldatasize())) - return(0x00, 0x20) - } - sstore(0, 1) - } - } -} diff --git a/lib/solady/test/utils/mocks/MockEIP712.sol b/lib/solady/test/utils/mocks/MockEIP712.sol deleted file mode 100644 index 5bf1c6b..0000000 --- a/lib/solady/test/utils/mocks/MockEIP712.sol +++ /dev/null @@ -1,26 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "../../../src/utils/EIP712.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockEIP712 is EIP712 { - function _domainNameAndVersion() - internal - pure - override - returns (string memory name, string memory version) - { - name = "Milady"; - version = "1"; - } - - function hashTypedData(bytes32 structHash) external view returns (bytes32) { - return _hashTypedData(structHash); - } - - function DOMAIN_SEPARATOR() external view returns (bytes32) { - return _domainSeparator(); - } -} diff --git a/lib/solady/test/utils/mocks/MockEIP712Dynamic.sol b/lib/solady/test/utils/mocks/MockEIP712Dynamic.sol deleted file mode 100644 index b9051f7..0000000 --- a/lib/solady/test/utils/mocks/MockEIP712Dynamic.sol +++ /dev/null @@ -1,43 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "../../../src/utils/EIP712.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockEIP712Dynamic is EIP712 { - string private _name; - string private _version; - - constructor(string memory name, string memory version) { - _name = name; - _version = version; - } - - function setDomainNameAndVersion(string memory name, string memory version) public { - _name = name; - _version = version; - } - - function _domainNameAndVersion() - internal - view - override - returns (string memory name, string memory version) - { - name = _name; - version = _version; - } - - function _domainNameAndVersionMayChange() internal pure override returns (bool) { - return true; - } - - function hashTypedData(bytes32 structHash) external view returns (bytes32) { - return _hashTypedData(structHash); - } - - function DOMAIN_SEPARATOR() external view returns (bytes32) { - return _domainSeparator(); - } -} diff --git a/lib/solady/test/utils/mocks/MockERC1155.sol b/lib/solady/test/utils/mocks/MockERC1155.sol deleted file mode 100644 index ed398d3..0000000 --- a/lib/solady/test/utils/mocks/MockERC1155.sol +++ /dev/null @@ -1,120 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {ERC1155} from "../../../src/tokens/ERC1155.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockERC1155 is ERC1155 { - function uri(uint256) public pure virtual override returns (string memory) {} - - function mint(address to, uint256 id, uint256 amount, bytes memory data) public virtual { - _mint(_brutalized(to), id, amount, data); - } - - function batchMint( - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) public virtual { - _batchMint(_brutalized(to), ids, amounts, data); - } - - function burn(address from, uint256 id, uint256 amount) public virtual { - _burn(_brutalized(msg.sender), _brutalized(from), id, amount); - } - - function uncheckedBurn(address from, uint256 id, uint256 amount) public virtual { - _burn(_brutalized(from), id, amount); - } - - function batchBurn(address from, uint256[] memory ids, uint256[] memory amounts) - public - virtual - { - _batchBurn(_brutalized(msg.sender), _brutalized(from), ids, amounts); - } - - function uncheckedBatchBurn(address from, uint256[] memory ids, uint256[] memory amounts) - public - virtual - { - _batchBurn(_brutalized(from), ids, amounts); - } - - function safeTransferFrom( - address from, - address to, - uint256 id, - uint256 amount, - bytes calldata data - ) public virtual override { - super.safeTransferFrom(_brutalized(from), _brutalized(to), id, amount, data); - } - - function directSafeTransferFrom( - address from, - address to, - uint256 id, - uint256 amount, - bytes memory data - ) public virtual { - _safeTransfer(_brutalized(msg.sender), _brutalized(from), _brutalized(to), id, amount, data); - } - - function uncheckedSafeTransferFrom( - address from, - address to, - uint256 id, - uint256 amount, - bytes memory data - ) public virtual { - _safeTransfer(_brutalized(address(0)), _brutalized(from), _brutalized(to), id, amount, data); - } - - function safeBatchTransferFrom( - address from, - address to, - uint256[] calldata ids, - uint256[] calldata amounts, - bytes calldata data - ) public virtual override { - super.safeBatchTransferFrom(_brutalized(from), _brutalized(to), ids, amounts, data); - } - - function directSafeBatchTransferFrom( - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) public virtual { - _safeBatchTransfer( - _brutalized(msg.sender), _brutalized(from), _brutalized(to), ids, amounts, data - ); - } - - function uncheckedSafeBatchTransferFrom( - address from, - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) public virtual { - _safeBatchTransfer( - _brutalized(address(0)), _brutalized(from), _brutalized(to), ids, amounts, data - ); - } - - function directSetApprovalForAll(address operator, bool approved) public virtual { - _setApprovalForAll(_brutalized(msg.sender), _brutalized(operator), approved); - } - - function _brutalized(address a) internal view returns (address result) { - /// @solidity memory-safe-assembly - assembly { - result := or(a, shl(160, gas())) - } - } -} diff --git a/lib/solady/test/utils/mocks/MockERC1271Malicious.sol b/lib/solady/test/utils/mocks/MockERC1271Malicious.sol deleted file mode 100644 index 3428d1a..0000000 --- a/lib/solady/test/utils/mocks/MockERC1271Malicious.sol +++ /dev/null @@ -1,14 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockERC1271Malicious { - function isValidSignature(bytes32, bytes calldata) external pure returns (bytes4) { - /// @solidity memory-safe-assembly - assembly { - mstore(0, 0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff) - return(0, 32) - } - } -} diff --git a/lib/solady/test/utils/mocks/MockERC1271Wallet.sol b/lib/solady/test/utils/mocks/MockERC1271Wallet.sol deleted file mode 100644 index f4cf765..0000000 --- a/lib/solady/test/utils/mocks/MockERC1271Wallet.sol +++ /dev/null @@ -1,56 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "../../../src/utils/ECDSA.sol"; - -/// @notice A generic interface for a contract which properly accepts ERC721 tokens. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC721.sol) -abstract contract ERC721TokenReceiver { - function onERC721Received(address, address, uint256, bytes calldata) - external - virtual - returns (bytes4) - { - return ERC721TokenReceiver.onERC721Received.selector; - } -} - -/// @notice A generic interface for a contract which properly accepts ERC1155 tokens. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC1155.sol) -abstract contract ERC1155TokenReceiver { - function onERC1155Received(address, address, uint256, uint256, bytes calldata) - external - virtual - returns (bytes4) - { - return ERC1155TokenReceiver.onERC1155Received.selector; - } - - function onERC1155BatchReceived( - address, - address, - uint256[] calldata, - uint256[] calldata, - bytes calldata - ) external virtual returns (bytes4) { - return ERC1155TokenReceiver.onERC1155BatchReceived.selector; - } -} - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockERC1271Wallet is ERC721TokenReceiver, ERC1155TokenReceiver { - address signer; - - constructor(address signer_) { - signer = signer_; - } - - function isValidSignature(bytes32 hash, bytes calldata signature) - external - view - returns (bytes4) - { - return ECDSA.recover(hash, signature) == signer ? bytes4(0x1626ba7e) : bytes4(0); - } -} diff --git a/lib/solady/test/utils/mocks/MockERC20.sol b/lib/solady/test/utils/mocks/MockERC20.sol deleted file mode 100644 index e1a2765..0000000 --- a/lib/solady/test/utils/mocks/MockERC20.sol +++ /dev/null @@ -1,72 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {ERC20} from "../../../src/tokens/ERC20.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockERC20 is ERC20 { - string internal _name; - string internal _symbol; - uint8 internal _decimals; - bytes32 internal immutable _nameHash; - - constructor(string memory name_, string memory symbol_, uint8 decimals_) { - _name = name_; - _symbol = symbol_; - _decimals = decimals_; - _nameHash = keccak256(bytes(name_)); - } - - function _constantNameHash() internal view virtual override returns (bytes32) { - return _nameHash; - } - - function name() public view virtual override returns (string memory) { - return _name; - } - - function symbol() public view virtual override returns (string memory) { - return _symbol; - } - - function decimals() public view virtual override returns (uint8) { - return _decimals; - } - - function mint(address to, uint256 value) public virtual { - _mint(_brutalized(to), value); - } - - function burn(address from, uint256 value) public virtual { - _burn(_brutalized(from), value); - } - - function directTransfer(address from, address to, uint256 amount) public virtual { - _transfer(_brutalized(from), _brutalized(to), amount); - } - - function directSpendAllowance(address owner, address spender, uint256 amount) public virtual { - _spendAllowance(_brutalized(owner), _brutalized(spender), amount); - } - - function transfer(address to, uint256 amount) public virtual override returns (bool) { - return super.transfer(_brutalized(to), amount); - } - - function transferFrom(address from, address to, uint256 amount) - public - virtual - override - returns (bool) - { - return super.transferFrom(_brutalized(from), _brutalized(to), amount); - } - - function _brutalized(address a) internal view returns (address result) { - /// @solidity memory-safe-assembly - assembly { - result := or(a, shl(160, gas())) - } - } -} diff --git a/lib/solady/test/utils/mocks/MockERC20LikeUSDT.sol b/lib/solady/test/utils/mocks/MockERC20LikeUSDT.sol deleted file mode 100644 index 2089ddf..0000000 --- a/lib/solady/test/utils/mocks/MockERC20LikeUSDT.sol +++ /dev/null @@ -1,16 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {MockERC20} from "./MockERC20.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockERC20LikeUSDT is MockERC20 { - constructor() MockERC20("Tether USD", "USDT", 6) {} - - // Replicates USDT (0xdAC17F958D2ee523a2206206994597C13D831ec7) approval behavior. - function approve(address spender, uint256 amount) public virtual override returns (bool) { - require(amount == 0 || allowance(msg.sender, spender) == 0, "USDT approval failure"); - return super.approve(spender, amount); - } -} diff --git a/lib/solady/test/utils/mocks/MockERC2981.sol b/lib/solady/test/utils/mocks/MockERC2981.sol deleted file mode 100644 index 0c38c5a..0000000 --- a/lib/solady/test/utils/mocks/MockERC2981.sol +++ /dev/null @@ -1,42 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {ERC2981} from "../../../src/tokens/ERC2981.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockERC2981 is ERC2981 { - function feeDenominator() external pure returns (uint256) { - return _feeDenominator(); - } - - function setDefaultRoyalty(address receiver, uint96 feeNumerator) external { - _setDefaultRoyalty(_brutalized(receiver), _brutalized(feeNumerator)); - } - - function deleteDefaultRoyalty() external { - _deleteDefaultRoyalty(); - } - - function setTokenRoyalty(uint256 tokenId, address receiver, uint96 feeNumerator) external { - _setTokenRoyalty(tokenId, _brutalized(receiver), _brutalized(feeNumerator)); - } - - function resetTokenRoyalty(uint256 tokenId) external { - _resetTokenRoyalty(tokenId); - } - - function _brutalized(uint96 x) internal view returns (uint96 result) { - /// @solidity memory-safe-assembly - assembly { - result := or(x, shl(96, gas())) - } - } - - function _brutalized(address a) internal view returns (address result) { - /// @solidity memory-safe-assembly - assembly { - result := or(a, shl(160, gas())) - } - } -} diff --git a/lib/solady/test/utils/mocks/MockERC4337.sol b/lib/solady/test/utils/mocks/MockERC4337.sol deleted file mode 100644 index fdef803..0000000 --- a/lib/solady/test/utils/mocks/MockERC4337.sol +++ /dev/null @@ -1,50 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {ERC4337} from "../../../src/accounts/ERC4337.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockERC4337 is ERC4337 { - function withdrawDepositTo(address to, uint256 amount) public payable virtual override { - super.withdrawDepositTo(_brutalized(to), amount); - } - - function _brutalized(address a) private pure returns (address result) { - /// @solidity memory-safe-assembly - assembly { - result := or(a, shl(160, 0x0123456789abcdeffedcba98)) - } - } - - function executeBatch(uint256 filler, Call[] calldata calls) - public - payable - virtual - onlyEntryPointOrOwner - returns (bytes[] memory results) - { - /// @solidity memory-safe-assembly - assembly { - mstore(0x40, add(mload(0x40), mod(filler, 0x40))) - } - return super.executeBatch(calls); - } - - function _domainNameAndVersion() - internal - pure - override - returns (string memory, string memory) - { - return ("Milady", "1"); - } - - function hashTypedData(bytes32 structHash) external view returns (bytes32) { - return _hashTypedData(structHash); - } - - function DOMAIN_SEPARATOR() external view returns (bytes32) { - return _domainSeparator(); - } -} diff --git a/lib/solady/test/utils/mocks/MockERC4626.sol b/lib/solady/test/utils/mocks/MockERC4626.sol deleted file mode 100644 index 82cd582..0000000 --- a/lib/solady/test/utils/mocks/MockERC4626.sol +++ /dev/null @@ -1,75 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {ERC20, ERC4626} from "../../../src/tokens/ERC4626.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockERC4626 is ERC4626 { - bool public immutable useVirtualShares; - uint8 public immutable decimalsOffset; - - address internal immutable _underlying; - uint8 internal immutable _decimals; - - string internal _name; - string internal _symbol; - - uint256 public beforeWithdrawHookCalledCounter; - uint256 public afterDepositHookCalledCounter; - - constructor( - address underlying_, - string memory name_, - string memory symbol_, - bool useVirtualShares_, - uint8 decimalsOffset_ - ) { - _underlying = underlying_; - - (bool success, uint8 result) = _tryGetAssetDecimals(underlying_); - _decimals = success ? result : _DEFAULT_UNDERLYING_DECIMALS; - - _name = name_; - _symbol = symbol_; - - useVirtualShares = useVirtualShares_; - decimalsOffset = decimalsOffset_; - } - - function asset() public view virtual override returns (address) { - return _underlying; - } - - function name() public view virtual override returns (string memory) { - return _name; - } - - function symbol() public view virtual override returns (string memory) { - return _symbol; - } - - function _useVirtualShares() internal view virtual override returns (bool) { - return useVirtualShares; - } - - function _underlyingDecimals() internal view virtual override returns (uint8) { - return _decimals; - } - - function _decimalsOffset() internal view virtual override returns (uint8) { - return decimalsOffset; - } - - function _beforeWithdraw(uint256, uint256) internal override { - unchecked { - ++beforeWithdrawHookCalledCounter; - } - } - - function _afterDeposit(uint256, uint256) internal override { - unchecked { - ++afterDepositHookCalledCounter; - } - } -} diff --git a/lib/solady/test/utils/mocks/MockERC6551.sol b/lib/solady/test/utils/mocks/MockERC6551.sol deleted file mode 100644 index 663d897..0000000 --- a/lib/solady/test/utils/mocks/MockERC6551.sol +++ /dev/null @@ -1,55 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {ERC6551} from "../../../src/accounts/ERC6551.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockERC6551 is ERC6551 { - function _brutalized(address a) private pure returns (address result) { - /// @solidity memory-safe-assembly - assembly { - result := or(a, shl(160, 0x0123456789abcdeffedcba98)) - } - } - - function executeBatch(uint256 filler, Call[] calldata calls, uint8 operation) - public - payable - virtual - returns (bytes[] memory results) - { - /// @solidity memory-safe-assembly - assembly { - mstore(0x40, add(mload(0x40), mod(filler, 0x40))) - } - return super.executeBatch(calls, operation); - } - - function _domainNameAndVersion() - internal - pure - override - returns (string memory, string memory) - { - return ("Milady", "1"); - } - - function hashTypedData(bytes32 structHash) external view returns (bytes32) { - return _hashTypedData(structHash); - } - - function DOMAIN_SEPARATOR() external view returns (bytes32) { - return _domainSeparator(); - } - - function mockId() public pure virtual returns (string memory) { - return "1"; - } -} - -contract MockERC6551V2 is MockERC6551 { - function mockId() public pure virtual override(MockERC6551) returns (string memory) { - return "2"; - } -} diff --git a/lib/solady/test/utils/mocks/MockERC6551Registry.sol b/lib/solady/test/utils/mocks/MockERC6551Registry.sol deleted file mode 100644 index adf981d..0000000 --- a/lib/solady/test/utils/mocks/MockERC6551Registry.sol +++ /dev/null @@ -1,115 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockERC6551Registry { - function createAccount( - address implementation, - bytes32 salt, - uint256 chainId, - address tokenContract, - uint256 tokenId - ) external returns (address) { - assembly { - // Memory Layout: - // ---- - // 0x00 0xff (1 byte) - // 0x01 registry (address) (20 bytes) - // 0x15 salt (bytes32) (32 bytes) - // 0x35 Bytecode Hash (bytes32) (32 bytes) - // ---- - // 0x55 ERC-1167 Constructor + Header (20 bytes) - // 0x69 implementation (address) (20 bytes) - // 0x5D ERC-1167 Footer (15 bytes) - // 0x8C salt (uint256) (32 bytes) - // 0xAC chainId (uint256) (32 bytes) - // 0xCC tokenContract (address) (32 bytes) - // 0xEC tokenId (uint256) (32 bytes) - - // Silence unused variable warnings - pop(chainId) - - // Copy bytecode + constant data to memory - calldatacopy(0x8c, 0x24, 0x80) // salt, chainId, tokenContract, tokenId - mstore(0x6c, 0x5af43d82803e903d91602b57fd5bf3) // ERC-1167 footer - mstore(0x5d, implementation) // implementation - mstore(0x49, 0x3d60ad80600a3d3981f3363d3d373d3d3d363d73) // ERC-1167 constructor + header - - // Copy create2 computation data to memory - mstore8(0x00, 0xff) // 0xFF - mstore(0x35, keccak256(0x55, 0xb7)) // keccak256(bytedcode) - mstore(0x01, shl(96, address())) // registry address - mstore(0x15, salt) // salt - - // Compute account address - let computed := keccak256(0x00, 0x55) - - // If the account has not yet been deployed - if iszero(extcodesize(computed)) { - // Deploy account contract - let deployed := create2(0, 0x55, 0xb7, salt) - - // Revert if the deployment fails - if iszero(deployed) { - mstore(0x00, 0x20188a59) // `AccountCreationFailed()` - revert(0x1c, 0x04) - } - - // Store account address in memory before salt and chainId - mstore(0x6c, deployed) - - // Emit the ERC6551AccountCreated event - log4( - 0x6c, - 0x60, - // `ERC6551AccountCreated(address,address,bytes32,uint256,address,uint256)` - 0x79f19b3655ee38b1ce526556b7731a20c8f218fbda4a3990b6cc4172fdf88722, - implementation, - tokenContract, - tokenId - ) - - // Return the account address - return(0x6c, 0x20) - } - - // Otherwise, return the computed account address - mstore(0x00, shr(96, shl(96, computed))) - return(0x00, 0x20) - } - } - - function account( - address implementation, - bytes32 salt, - uint256 chainId, - address tokenContract, - uint256 tokenId - ) external view returns (address) { - assembly { - // Silence unused variable warnings - pop(chainId) - pop(tokenContract) - pop(tokenId) - - // Copy bytecode + constant data to memory - calldatacopy(0x8c, 0x24, 0x80) // salt, chainId, tokenContract, tokenId - mstore(0x6c, 0x5af43d82803e903d91602b57fd5bf3) // ERC-1167 footer - mstore(0x5d, implementation) // implementation - mstore(0x49, 0x3d60ad80600a3d3981f3363d3d373d3d3d363d73) // ERC-1167 constructor + header - - // Copy create2 computation data to memory - mstore8(0x00, 0xff) // 0xFF - mstore(0x35, keccak256(0x55, 0xb7)) // keccak256(bytedcode) - mstore(0x01, shl(96, address())) // registry address - mstore(0x15, salt) // salt - - // Store computed account address in memory - mstore(0x00, shr(96, shl(96, keccak256(0x00, 0x55)))) - - // Return computed account address - return(0x00, 0x20) - } - } -} diff --git a/lib/solady/test/utils/mocks/MockERC6909.sol b/lib/solady/test/utils/mocks/MockERC6909.sol deleted file mode 100644 index dc3abf7..0000000 --- a/lib/solady/test/utils/mocks/MockERC6909.sol +++ /dev/null @@ -1,110 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {ERC6909} from "../../../src/tokens/ERC6909.sol"; -import {LibString} from "../../../src/utils/LibString.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockERC6909 is ERC6909 { - error TokenDoesNotExist(); - - function name() public view virtual override returns (string memory) { - return "Solady Token"; - } - - function symbol() public view virtual override returns (string memory) { - return "ST"; - } - - function tokenURI(uint256 id) public view virtual override returns (string memory) { - return string(abi.encodePacked("http://solady.org/", LibString.toString(id))); - } - - function mint(address to, uint256 id, uint256 amount) public payable virtual { - _mint(_brutalized(to), id, amount); - } - - function burn(address from, uint256 id, uint256 amount) public payable virtual { - _burn(_brutalized(from), id, amount); - } - - function approve(address spender, uint256 id, uint256 amount) - public - payable - virtual - override - returns (bool) - { - return super.approve(_brutalized(spender), id, amount); - } - - function setOperator(address owner, bool approved) - public - payable - virtual - override - returns (bool) - { - /// @solidity memory-safe-assembly - assembly { - approved := mul(gas(), approved) - } - return super.setOperator(_brutalized(owner), approved); - } - - function transfer(address to, uint256 id, uint256 amount) - public - payable - virtual - override - returns (bool) - { - return super.transfer(_brutalized(to), id, amount); - } - - function transferFrom(address from, address to, uint256 id, uint256 amount) - public - payable - virtual - override - returns (bool) - { - return super.transferFrom(_brutalized(from), _brutalized(to), id, amount); - } - - function directTransferFrom(address by, address from, address to, uint256 id, uint256 amount) - public - payable - virtual - { - _transfer(_brutalized(by), _brutalized(from), _brutalized(to), id, amount); - } - - function directSetOperator(address owner, address operator, bool approved) - public - payable - virtual - { - /// @solidity memory-safe-assembly - assembly { - approved := mul(gas(), approved) - } - _setOperator(owner, operator, approved); - } - - function directApprove(address owner, address spender, uint256 id, uint256 amount) - public - payable - virtual - { - _approve(owner, spender, id, amount); - } - - function _brutalized(address a) internal view returns (address result) { - /// @solidity memory-safe-assembly - assembly { - result := or(a, shl(160, gas())) - } - } -} diff --git a/lib/solady/test/utils/mocks/MockERC721.sol b/lib/solady/test/utils/mocks/MockERC721.sol deleted file mode 100644 index 81e8df1..0000000 --- a/lib/solady/test/utils/mocks/MockERC721.sol +++ /dev/null @@ -1,141 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {ERC721} from "../../../src/tokens/ERC721.sol"; -import {LibString} from "../../../src/utils/LibString.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockERC721 is ERC721 { - function name() public view virtual override returns (string memory) { - return "TEST NFT"; - } - - function symbol() public view virtual override returns (string memory) { - return "TEST"; - } - - function tokenURI(uint256 id) public view virtual override returns (string memory) { - if (!_exists(id)) revert TokenDoesNotExist(); - return string(abi.encodePacked("https://remilio.org/remilio/json/", LibString.toString(id))); - } - - function exists(uint256 id) public view virtual returns (bool) { - return _exists(id); - } - - function mint(address to, uint256 id) public virtual { - _mint(_brutalized(to), id); - } - - function burn(uint256 id) public virtual { - _burn(msg.sender, id); - } - - function uncheckedBurn(uint256 id) public virtual { - _burn(id); - } - - function safeMint(address to, uint256 id) public virtual { - _safeMint(_brutalized(to), id); - } - - function safeMint(address to, uint256 id, bytes calldata data) public virtual { - _safeMint(_brutalized(to), id, data); - } - - function getExtraData(uint256 id) public view virtual returns (uint96) { - return _getExtraData(id); - } - - function setExtraData(uint256 id, uint96 value) public virtual { - _setExtraData(id, value); - } - - function getAux(address owner) public view virtual returns (uint224) { - return _getAux(_brutalized(owner)); - } - - function setAux(address owner, uint224 value) public virtual { - _setAux(_brutalized(owner), value); - } - - function approve(address account, uint256 id) public payable virtual override { - super.approve(_brutalized(account), id); - } - - function directApprove(address account, uint256 id) public virtual { - if (!_isApprovedOrOwner(_brutalized(msg.sender), id)) revert NotOwnerNorApproved(); - _approve(_brutalized(account), id); - } - - function setApprovalForAll(address operator, bool approved) public virtual override { - super.setApprovalForAll(_brutalized(operator), approved); - } - - function directSetApprovalForAll(address operator, bool approved) public virtual { - _setApprovalForAll(_brutalized(msg.sender), _brutalized(operator), approved); - } - - function transferFrom(address from, address to, uint256 id) public payable virtual override { - super.transferFrom(_brutalized(from), _brutalized(to), id); - } - - function uncheckedTransferFrom(address from, address to, uint256 id) public payable virtual { - _transfer(_brutalized(address(0)), _brutalized(from), _brutalized(to), id); - } - - function directTransferFrom(address from, address to, uint256 id) public virtual { - _transfer(_brutalized(msg.sender), _brutalized(from), _brutalized(to), id); - } - - function safeTransferFrom(address from, address to, uint256 id) - public - payable - virtual - override - { - super.safeTransferFrom(_brutalized(from), _brutalized(to), id); - } - - function directSafeTransferFrom(address from, address to, uint256 id) public virtual { - _safeTransfer(_brutalized(msg.sender), _brutalized(from), _brutalized(to), id); - } - - function safeTransferFrom(address from, address to, uint256 id, bytes calldata data) - public - payable - virtual - override - { - super.safeTransferFrom(_brutalized(from), _brutalized(to), id, data); - } - - function directSafeTransferFrom(address from, address to, uint256 id, bytes calldata data) - public - virtual - { - _safeTransfer(_brutalized(msg.sender), _brutalized(from), _brutalized(to), id, data); - } - - function isApprovedOrOwner(address account, uint256 id) public view virtual returns (bool) { - return _isApprovedOrOwner(_brutalized(account), id); - } - - function directOwnerOf(uint256 id) public view virtual returns (address) { - if (!_exists(id)) revert TokenDoesNotExist(); - return _ownerOf(id); - } - - function directGetApproved(uint256 id) public view virtual returns (address) { - if (!_exists(id)) revert TokenDoesNotExist(); - return _getApproved(id); - } - - function _brutalized(address a) internal view returns (address result) { - /// @solidity memory-safe-assembly - assembly { - result := or(a, shl(160, gas())) - } - } -} diff --git a/lib/solady/test/utils/mocks/MockETHRecipient.sol b/lib/solady/test/utils/mocks/MockETHRecipient.sol deleted file mode 100644 index b4012a7..0000000 --- a/lib/solady/test/utils/mocks/MockETHRecipient.sol +++ /dev/null @@ -1,40 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockETHRecipient { - bool public immutable gasGriefUponReceiveETH; - - bool public immutable updateCounterUponReceiveETH; - - uint256 public counter; - - uint256 public garbage; - - constructor(bool updateCounterUponReceiveETH_, bool gasGriefUponReceiveETH_) { - updateCounterUponReceiveETH = updateCounterUponReceiveETH_; - gasGriefUponReceiveETH = gasGriefUponReceiveETH_; - } - - receive() external payable { - if (updateCounterUponReceiveETH) { - counter += 1; - } - if (gasGriefUponReceiveETH) { - /// @solidity memory-safe-assembly - assembly { - mstore(0x00, timestamp()) - mstore(0x20, 0) - - for { let i := 0 } lt(i, 10) { i := add(i, 1) } { - let h := keccak256(0x00, 0x40) - mstore(0x00, sload(h)) - mstore(0x20, i) - sstore(add(h, 1), h) - } - sstore(garbage.slot, keccak256(0x00, 0x40)) - } - } - } -} diff --git a/lib/solady/test/utils/mocks/MockEntryPoint.sol b/lib/solady/test/utils/mocks/MockEntryPoint.sol deleted file mode 100644 index e0fc265..0000000 --- a/lib/solady/test/utils/mocks/MockEntryPoint.sol +++ /dev/null @@ -1,34 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {ERC4337} from "../../../src/accounts/ERC4337.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockEntryPoint { - mapping(address => uint256) public balanceOf; - - function depositTo(address to) public payable { - balanceOf[to] += msg.value; - } - - function withdrawTo(address to, uint256 amount) public payable { - balanceOf[msg.sender] -= amount; - (bool success,) = payable(to).call{value: amount}(""); - require(success); - } - - function validateUserOp( - address account, - ERC4337.UserOperation memory userOp, - bytes32 userOpHash, - uint256 missingAccountFunds - ) public payable returns (uint256 validationData) { - validationData = - ERC4337(payable(account)).validateUserOp(userOp, userOpHash, missingAccountFunds); - } - - receive() external payable { - depositTo(msg.sender); - } -} diff --git a/lib/solady/test/utils/mocks/MockImplementation.sol b/lib/solady/test/utils/mocks/MockImplementation.sol deleted file mode 100644 index 59ba351..0000000 --- a/lib/solady/test/utils/mocks/MockImplementation.sol +++ /dev/null @@ -1,26 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockImplementation { - error Fail(); - - mapping(uint256 => uint256) internal _values; - - function fails() external pure { - revert Fail(); - } - - function succeeds(uint256 a) external pure returns (uint256) { - return a; - } - - function setValue(uint256 key, uint256 value) external payable { - _values[key] = value; - } - - function getValue(uint256 key) external view returns (uint256) { - return _values[key]; - } -} diff --git a/lib/solady/test/utils/mocks/MockMulticallable.sol b/lib/solady/test/utils/mocks/MockMulticallable.sol deleted file mode 100644 index 1a6c197..0000000 --- a/lib/solady/test/utils/mocks/MockMulticallable.sol +++ /dev/null @@ -1,68 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import "../../../src/utils/Multicallable.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockMulticallable is Multicallable { - error CustomError(); - - struct Tuple { - uint256 a; - uint256 b; - } - - function revertsWithString(string memory e) external pure { - revert(e); - } - - function revertsWithCustomError() external pure { - revert CustomError(); - } - - function revertsWithNothing() external pure { - revert(); - } - - function returnsTuple(uint256 a, uint256 b) external pure returns (Tuple memory tuple) { - tuple = Tuple({a: a, b: b}); - } - - function returnsString(string calldata s) external pure returns (string memory) { - return s; - } - - uint256 public paid; - - function pay() external payable { - paid += msg.value; - } - - function returnsSender() external view returns (address) { - return msg.sender; - } - - function multicallOriginal(bytes[] calldata data) - public - payable - returns (bytes[] memory results) - { - unchecked { - results = new bytes[](data.length); - for (uint256 i = 0; i < data.length; i++) { - (bool success, bytes memory result) = address(this).delegatecall(data[i]); - if (!success) { - // Next 5 lines from https://ethereum.stackexchange.com/a/83577 - if (result.length < 68) revert(); - /// @solidity memory-safe-assembly - assembly { - result := add(result, 0x04) - } - revert(abi.decode(result, (string))); - } - results[i] = result; - } - } - } -} diff --git a/lib/solady/test/utils/mocks/MockOwnable.sol b/lib/solady/test/utils/mocks/MockOwnable.sol deleted file mode 100644 index c132cd9..0000000 --- a/lib/solady/test/utils/mocks/MockOwnable.sol +++ /dev/null @@ -1,97 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {Ownable} from "../../../src/auth/Ownable.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockOwnable is Ownable { - bool public flag; - - constructor() payable { - _initializeOwner(msg.sender); - - // Perform the tests on the helper functions. - - address brutalizedAddress = _brutalizedAddress(address(0)); - bool brutalizedAddressIsBrutalized; - /// @solidity memory-safe-assembly - assembly { - brutalizedAddressIsBrutalized := gt(shr(160, brutalizedAddress), 0) - } - - if (!brutalizedAddressIsBrutalized) { - revert("Setup failed"); - } - } - - function initializeOwnerDirect(address newOwner) public payable { - _initializeOwner(_brutalizedAddress(newOwner)); - } - - function setOwnerDirect(address newOwner) public payable { - _setOwner(_brutalizedAddress(newOwner)); - } - - function completeOwnershipHandover(address pendingOwner) public payable virtual override { - super.completeOwnershipHandover(_brutalizedAddress(pendingOwner)); - } - - function transferOwnership(address newOwner) public payable virtual override { - super.transferOwnership(_brutalizedAddress(newOwner)); - } - - function ownershipHandoverExpiresAt(address pendingOwner) - public - view - virtual - override - returns (uint256 result) - { - result = super.ownershipHandoverExpiresAt(_brutalizedAddress(pendingOwner)); - } - - function ownershipHandoverValidFor() public view returns (uint64 result) { - result = _ownershipHandoverValidFor(); - /// @solidity memory-safe-assembly - assembly { - // Some acrobatics to make the brutalized bits pseudorandomly - // different with every call. - mstore(0x00, or(calldataload(0), mload(0x40))) - mstore(0x20, or(caller(), mload(0x00))) - // Just brutalize the upper unused bits of the result to see if it causes any issue. - result := or(shl(64, keccak256(0x00, 0x40)), result) - mstore(0x40, add(0x20, mload(0x40))) - mstore(0x00, result) - } - } - - function updateFlagWithOnlyOwner() public payable onlyOwner { - flag = true; - } - - function _brutalizedAddress(address value) private view returns (address result) { - /// @solidity memory-safe-assembly - assembly { - // Some acrobatics to make the brutalized bits pseudorandomly - // different with every call. - mstore(0x00, or(calldataload(0), mload(0x40))) - mstore(0x20, or(caller(), mload(0x00))) - result := or(shl(160, keccak256(0x00, 0x40)), value) - mstore(0x40, add(0x20, mload(0x40))) - mstore(0x00, result) - } - } -} - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockOwnableBytecodeSizer is Ownable { - constructor() payable { - initialize(); - } - - function initialize() public payable { - _initializeOwner(msg.sender); - } -} diff --git a/lib/solady/test/utils/mocks/MockOwnableRoles.sol b/lib/solady/test/utils/mocks/MockOwnableRoles.sol deleted file mode 100644 index 8a3038f..0000000 --- a/lib/solady/test/utils/mocks/MockOwnableRoles.sol +++ /dev/null @@ -1,165 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {Ownable, OwnableRoles} from "../../../src/auth/OwnableRoles.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockOwnableRoles is OwnableRoles { - bool public flag; - - constructor() payable { - _initializeOwner(msg.sender); - - // Perform the tests on the helper functions. - - address brutalizedAddress = _brutalizedAddress(address(0)); - bool brutalizedAddressIsBrutalized; - /// @solidity memory-safe-assembly - assembly { - brutalizedAddressIsBrutalized := gt(shr(160, brutalizedAddress), 0) - } - - if (!brutalizedAddressIsBrutalized) { - revert("Setup failed"); - } - - bool badBool; - /// @solidity memory-safe-assembly - assembly { - badBool := 2 - } - - bool checkedBadBool = _checkedBool(badBool); - - if (checkedBadBool) { - revert("Setup failed"); - } - } - - function initializeOwnerDirect(address newOwner) public payable { - _initializeOwner(_brutalizedAddress(newOwner)); - } - - function setOwnerDirect(address newOwner) public payable { - _setOwner(_brutalizedAddress(newOwner)); - } - - function setRolesDirect(address user, uint256 roles) public payable { - _setRoles(_brutalizedAddress(user), roles); - } - - function grantRolesDirect(address user, uint256 roles) public payable { - _grantRoles(_brutalizedAddress(user), roles); - } - - function removeRolesDirect(address user, uint256 roles) public payable { - _removeRoles(_brutalizedAddress(user), roles); - } - - function grantRoles(address user, uint256 roles) public payable virtual override { - super.grantRoles(_brutalizedAddress(user), roles); - } - - function revokeRoles(address user, uint256 roles) public payable virtual override { - super.revokeRoles(_brutalizedAddress(user), roles); - } - - function completeOwnershipHandover(address pendingOwner) public payable virtual override { - super.completeOwnershipHandover(_brutalizedAddress(pendingOwner)); - } - - function transferOwnership(address newOwner) public payable virtual override { - super.transferOwnership(_brutalizedAddress(newOwner)); - } - - function rolesOf(address user) public view virtual override returns (uint256 result) { - result = super.rolesOf(_brutalizedAddress(user)); - } - - function ownershipHandoverExpiresAt(address pendingOwner) - public - view - virtual - override - returns (uint256 result) - { - result = super.ownershipHandoverExpiresAt(_brutalizedAddress(pendingOwner)); - } - - function ownershipHandoverValidFor() public view returns (uint64 result) { - result = _ownershipHandoverValidFor(); - /// @solidity memory-safe-assembly - assembly { - // Some acrobatics to make the brutalized bits pseudorandomly - // different with every call. - mstore(0x00, or(calldataload(0), mload(0x40))) - mstore(0x20, or(caller(), mload(0x00))) - // Just brutalize the upper unused bits of the result to see if it causes any issue. - result := or(shl(64, keccak256(0x00, 0x40)), result) - mstore(0x40, add(0x20, mload(0x40))) - mstore(0x00, result) - } - } - - function updateFlagWithOnlyOwner() public payable onlyOwner { - flag = true; - } - - function updateFlagWithOnlyRoles(uint256 roles) public payable onlyRoles(roles) { - flag = true; - } - - function updateFlagWithOnlyOwnerOrRoles(uint256 roles) public payable onlyOwnerOrRoles(roles) { - flag = true; - } - - function updateFlagWithOnlyRolesOrOwner(uint256 roles) public payable onlyRolesOrOwner(roles) { - flag = true; - } - - function rolesFromOrdinals(uint8[] memory ordinals) public pure returns (uint256 roles) { - roles = _rolesFromOrdinals(ordinals); - } - - function ordinalsFromRoles(uint256 roles) public pure returns (uint8[] memory ordinals) { - ordinals = _ordinalsFromRoles(roles); - } - - function _brutalizedAddress(address value) private view returns (address result) { - /// @solidity memory-safe-assembly - assembly { - // Some acrobatics to make the brutalized bits pseudorandomly - // different with every call. - mstore(0x00, or(calldataload(0), mload(0x40))) - mstore(0x20, or(caller(), mload(0x00))) - result := or(shl(160, keccak256(0x00, 0x40)), value) - mstore(0x40, add(0x20, mload(0x40))) - mstore(0x00, result) - } - } - - function _checkedBool(bool value) private pure returns (bool result) { - result = value; - bool resultIsOneOrZero; - /// @solidity memory-safe-assembly - assembly { - // We wanna check if the result is either 1 or 0, - // to make sure we practice good assembly politeness. - resultIsOneOrZero := lt(result, 2) - } - if (!resultIsOneOrZero) result = !result; - } -} - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockOwnableRolesBytecodeSizer is OwnableRoles { - constructor() payable { - initialize(); - } - - function initialize() public payable { - _initializeOwner(msg.sender); - } -} diff --git a/lib/solady/test/utils/mocks/MockReceiver.sol b/lib/solady/test/utils/mocks/MockReceiver.sol deleted file mode 100644 index 77cd523..0000000 --- a/lib/solady/test/utils/mocks/MockReceiver.sol +++ /dev/null @@ -1,8 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {Receiver} from "../../../src/accounts/Receiver.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockReceiver is Receiver {} diff --git a/lib/solady/test/utils/mocks/MockUUPSImplementation.sol b/lib/solady/test/utils/mocks/MockUUPSImplementation.sol deleted file mode 100644 index c153160..0000000 --- a/lib/solady/test/utils/mocks/MockUUPSImplementation.sol +++ /dev/null @@ -1,50 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -import {UUPSUpgradeable} from "../../../src/utils/UUPSUpgradeable.sol"; - -/// @dev WARNING! This mock is strictly intended for testing purposes only. -/// Do NOT copy anything here into production code unless you really know what you are doing. -contract MockUUPSImplementation is UUPSUpgradeable { - uint256 public value; - - address owner; - - error Unauthorized(); - - error CustomError(address owner_); - - function initialize(address owner_) public { - owner = owner_; - } - - modifier onlyOwner() { - if (msg.sender != owner) revert Unauthorized(); - _; - } - - function _authorizeUpgrade(address) internal override onlyOwner {} - - function revertWithError() public view { - revert CustomError(owner); - } - - function setValue(uint256 val_) public { - value = val_; - } - - function upgradeToAndCall(address newImplementation, bytes calldata data) - public - payable - override - { - super.upgradeToAndCall(_brutalized(newImplementation), data); - } - - function _brutalized(address a) private pure returns (address result) { - /// @solidity memory-safe-assembly - assembly { - result := or(a, shl(160, 0x0123456789abcdeffedcba98)) - } - } -} diff --git a/lib/solady/test/utils/weird-tokens/MissingReturnToken.sol b/lib/solady/test/utils/weird-tokens/MissingReturnToken.sol deleted file mode 100644 index e07d8bd..0000000 --- a/lib/solady/test/utils/weird-tokens/MissingReturnToken.sol +++ /dev/null @@ -1,81 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -contract MissingReturnToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "MissingReturnToken"; - - string public constant symbol = "MRT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address spender, uint256 amount) public virtual { - allowance[msg.sender][spender] = amount; - - emit Approval(msg.sender, spender, amount); - } - - function transfer(address to, uint256 amount) public virtual { - balanceOf[msg.sender] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(msg.sender, to, amount); - } - - function transferFrom(address from, address to, uint256 amount) public virtual { - uint256 allowed = allowance[from][msg.sender]; // Saves gas for limited approvals. - - if (allowed != type(uint256).max) { - allowance[from][msg.sender] = allowed - amount; - } - - balanceOf[from] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(from, to, amount); - } -} diff --git a/lib/solady/test/utils/weird-tokens/ReturnsFalseToken.sol b/lib/solady/test/utils/weird-tokens/ReturnsFalseToken.sol deleted file mode 100644 index 8dd01de..0000000 --- a/lib/solady/test/utils/weird-tokens/ReturnsFalseToken.sol +++ /dev/null @@ -1,57 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -contract ReturnsFalseToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "ReturnsFalseToken"; - - string public constant symbol = "RFT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address, uint256) public virtual returns (bool) { - return false; - } - - function transfer(address, uint256) public virtual returns (bool) { - return false; - } - - function transferFrom(address, address, uint256) public virtual returns (bool) { - return false; - } -} diff --git a/lib/solady/test/utils/weird-tokens/ReturnsRawBytesToken.sol b/lib/solady/test/utils/weird-tokens/ReturnsRawBytesToken.sol deleted file mode 100644 index 6525990..0000000 --- a/lib/solady/test/utils/weird-tokens/ReturnsRawBytesToken.sol +++ /dev/null @@ -1,116 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -contract ReturnsRawBytesToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "ReturnsRawBytesToken"; - - string public constant symbol = "RGT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - MOCK STORAGE - //////////////////////////////////////////////////////////////*/ - - bytes rawBytes; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address spender, uint256 amount) public virtual { - allowance[msg.sender][spender] = amount; - - emit Approval(msg.sender, spender, amount); - - bytes memory _rawBytes = rawBytes; - - /// @solidity memory-safe-assembly - assembly { - return(add(_rawBytes, 32), mload(_rawBytes)) - } - } - - function transfer(address to, uint256 amount) public virtual { - balanceOf[msg.sender] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(msg.sender, to, amount); - - bytes memory _rawBytes = rawBytes; - - /// @solidity memory-safe-assembly - assembly { - return(add(_rawBytes, 32), mload(_rawBytes)) - } - } - - function transferFrom(address from, address to, uint256 amount) public virtual { - uint256 allowed = allowance[from][msg.sender]; // Saves gas for limited approvals. - - if (allowed != type(uint256).max) { - allowance[from][msg.sender] = allowed - amount; - } - - balanceOf[from] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(from, to, amount); - - bytes memory _rawBytes = rawBytes; - - /// @solidity memory-safe-assembly - assembly { - return(add(_rawBytes, 32), mload(_rawBytes)) - } - } - - /*/////////////////////////////////////////////////////////////// - MOCK LOGIC - //////////////////////////////////////////////////////////////*/ - - function setRawBytes(bytes memory _rawBytes) public virtual { - rawBytes = _rawBytes; - } -} diff --git a/lib/solady/test/utils/weird-tokens/ReturnsTooLittleToken.sol b/lib/solady/test/utils/weird-tokens/ReturnsTooLittleToken.sol deleted file mode 100644 index 2c624c1..0000000 --- a/lib/solady/test/utils/weird-tokens/ReturnsTooLittleToken.sol +++ /dev/null @@ -1,69 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -contract ReturnsTooLittleToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "ReturnsTooLittleToken"; - - string public constant symbol = "RTLT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address, uint256) public virtual { - /// @solidity memory-safe-assembly - assembly { - mstore(0, 0x0100000000000000000000000000000000000000000000000000000000000000) - return(0, 8) - } - } - - function transfer(address, uint256) public virtual { - /// @solidity memory-safe-assembly - assembly { - mstore(0, 0x0100000000000000000000000000000000000000000000000000000000000000) - return(0, 8) - } - } - - function transferFrom(address, address, uint256) public virtual { - /// @solidity memory-safe-assembly - assembly { - mstore(0, 0x0100000000000000000000000000000000000000000000000000000000000000) - return(0, 8) - } - } -} diff --git a/lib/solady/test/utils/weird-tokens/ReturnsTooMuchToken.sol b/lib/solady/test/utils/weird-tokens/ReturnsTooMuchToken.sol deleted file mode 100644 index 97d16b1..0000000 --- a/lib/solady/test/utils/weird-tokens/ReturnsTooMuchToken.sol +++ /dev/null @@ -1,99 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -contract ReturnsTooMuchToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "ReturnsTooMuchToken"; - - string public constant symbol = "RTMT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address spender, uint256 amount) public virtual { - allowance[msg.sender][spender] = amount; - - emit Approval(msg.sender, spender, amount); - - /// @solidity memory-safe-assembly - assembly { - mstore(0, 1) - return(0, 4096) - } - } - - function transfer(address to, uint256 amount) public virtual { - balanceOf[msg.sender] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(msg.sender, to, amount); - - /// @solidity memory-safe-assembly - assembly { - mstore(0, 1) - return(0, 4096) - } - } - - function transferFrom(address from, address to, uint256 amount) public virtual { - uint256 allowed = allowance[from][msg.sender]; // Saves gas for limited approvals. - - if (allowed != type(uint256).max) { - allowance[from][msg.sender] = allowed - amount; - } - - balanceOf[from] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(from, to, amount); - - /// @solidity memory-safe-assembly - assembly { - mstore(0, 1) - return(0, 4096) - } - } -} diff --git a/lib/solady/test/utils/weird-tokens/ReturnsTwoToken.sol b/lib/solady/test/utils/weird-tokens/ReturnsTwoToken.sol deleted file mode 100644 index fb88d0d..0000000 --- a/lib/solady/test/utils/weird-tokens/ReturnsTwoToken.sol +++ /dev/null @@ -1,57 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -contract ReturnsTwoToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "ReturnsFalseToken"; - - string public constant symbol = "RTT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address, uint256) public virtual returns (uint256) { - return 2; - } - - function transfer(address, uint256) public virtual returns (uint256) { - return 2; - } - - function transferFrom(address, address, uint256) public virtual returns (uint256) { - return 2; - } -} diff --git a/lib/solady/test/utils/weird-tokens/RevertingToken.sol b/lib/solady/test/utils/weird-tokens/RevertingToken.sol deleted file mode 100644 index 17aca18..0000000 --- a/lib/solady/test/utils/weird-tokens/RevertingToken.sol +++ /dev/null @@ -1,57 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity ^0.8.4; - -contract RevertingToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "RevertingToken"; - - string public constant symbol = "RT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address, uint256) public virtual { - revert(); - } - - function transfer(address, uint256) public virtual { - revert(); - } - - function transferFrom(address, address, uint256) public virtual { - revert(); - } -} diff --git a/lib/solmate b/lib/solmate new file mode 160000 index 0000000..4b47a19 --- /dev/null +++ b/lib/solmate @@ -0,0 +1 @@ +Subproject commit 4b47a19038b798b4a33d9749d25e570443520647 diff --git a/lib/solmate/.gas-snapshot b/lib/solmate/.gas-snapshot deleted file mode 100644 index 13e2410..0000000 --- a/lib/solmate/.gas-snapshot +++ /dev/null @@ -1,568 +0,0 @@ -AuthTest:testCallFunctionAsOwner() (gas: 29784) -AuthTest:testCallFunctionWithPermissiveAuthority() (gas: 124292) -AuthTest:testCallFunctionWithPermissiveAuthority(address) (runs: 256, μ: 129197, ~: 129235) -AuthTest:testFailCallFunctionAsNonOwner() (gas: 15523) -AuthTest:testFailCallFunctionAsNonOwner(address) (runs: 256, μ: 15685, ~: 15685) -AuthTest:testFailCallFunctionAsOwnerWithOutOfOrderAuthority() (gas: 136021) -AuthTest:testFailCallFunctionWithRestrictiveAuthority() (gas: 129144) -AuthTest:testFailCallFunctionWithRestrictiveAuthority(address) (runs: 256, μ: 129329, ~: 129329) -AuthTest:testFailSetAuthorityAsNonOwner() (gas: 18325) -AuthTest:testFailSetAuthorityAsNonOwner(address,address) (runs: 256, μ: 18594, ~: 18594) -AuthTest:testFailSetAuthorityWithRestrictiveAuthority() (gas: 129077) -AuthTest:testFailSetAuthorityWithRestrictiveAuthority(address,address) (runs: 256, μ: 129409, ~: 129409) -AuthTest:testFailTransferOwnershipAsNonOwner() (gas: 15641) -AuthTest:testFailTransferOwnershipAsNonOwner(address,address) (runs: 256, μ: 15922, ~: 15922) -AuthTest:testFailTransferOwnershipAsOwnerWithOutOfOrderAuthority() (gas: 136159) -AuthTest:testFailTransferOwnershipAsOwnerWithOutOfOrderAuthority(address) (runs: 256, μ: 136344, ~: 136344) -AuthTest:testFailTransferOwnershipWithRestrictiveAuthority() (gas: 129338) -AuthTest:testFailTransferOwnershipWithRestrictiveAuthority(address,address) (runs: 256, μ: 129588, ~: 129588) -AuthTest:testSetAuthorityAsOwner() (gas: 32214) -AuthTest:testSetAuthorityAsOwner(address) (runs: 256, μ: 32384, ~: 32384) -AuthTest:testSetAuthorityAsOwnerWithOutOfOrderAuthority() (gas: 226419) -AuthTest:testSetAuthorityWithPermissiveAuthority() (gas: 125962) -AuthTest:testSetAuthorityWithPermissiveAuthority(address,address) (runs: 256, μ: 130862, ~: 131055) -AuthTest:testTransferOwnershipAsOwner() (gas: 15298) -AuthTest:testTransferOwnershipAsOwner(address) (runs: 256, μ: 15469, ~: 15469) -AuthTest:testTransferOwnershipWithPermissiveAuthority() (gas: 127926) -AuthTest:testTransferOwnershipWithPermissiveAuthority(address,address) (runs: 256, μ: 131000, ~: 131000) -Bytes32AddressLibTest:testFillLast12Bytes() (gas: 223) -Bytes32AddressLibTest:testFromLast20Bytes() (gas: 191) -CREATE3Test:testDeployERC20() (gas: 853207) -CREATE3Test:testDeployERC20(bytes32,string,string,uint8) (runs: 256, μ: 899408, ~: 899427) -CREATE3Test:testFailDoubleDeployDifferentBytecode() (gas: 9079256848778914176) -CREATE3Test:testFailDoubleDeployDifferentBytecode(bytes32,bytes,bytes) (runs: 256, μ: 5830806919119382084, ~: 8937393460516727580) -CREATE3Test:testFailDoubleDeploySameBytecode() (gas: 9079256848778906219) -CREATE3Test:testFailDoubleDeploySameBytecode(bytes32,bytes) (runs: 256, μ: 5516047526412694750, ~: 8937393460516728714) -CREATE3Test:testPredictDeployERC20() (gas: 2102048) -DSTestPlusTest:testBound() (gas: 14214) -DSTestPlusTest:testBound(uint256,uint256,uint256) (runs: 256, μ: 2787, ~: 2793) -DSTestPlusTest:testBrutalizeMemory() (gas: 823) -DSTestPlusTest:testFailBoundMinBiggerThanMax() (gas: 309) -DSTestPlusTest:testFailBoundMinBiggerThanMax(uint256,uint256,uint256) (runs: 256, μ: 460, ~: 460) -DSTestPlusTest:testRelApproxEqBothZeroesPasses() (gas: 425) -ERC1155Test:testApproveAll() (gas: 31009) -ERC1155Test:testApproveAll(address,bool) (runs: 256, μ: 22771, ~: 31309) -ERC1155Test:testBatchBalanceOf() (gas: 157631) -ERC1155Test:testBatchBalanceOf(address[],uint256[],uint256[],bytes) (runs: 256, μ: 3564574, ~: 2943600) -ERC1155Test:testBatchBurn() (gas: 151074) -ERC1155Test:testBatchBurn(address,uint256[],uint256[],uint256[],bytes) (runs: 256, μ: 3627558, ~: 3049938) -ERC1155Test:testBatchMintToEOA() (gas: 137337) -ERC1155Test:testBatchMintToEOA(address,uint256[],uint256[],bytes) (runs: 256, μ: 3072824, ~: 2525648) -ERC1155Test:testBatchMintToERC1155Recipient() (gas: 995703) -ERC1155Test:testBatchMintToERC1155Recipient(uint256[],uint256[],bytes) (runs: 256, μ: 7395823, ~: 6396323) -ERC1155Test:testBurn() (gas: 38598) -ERC1155Test:testBurn(address,uint256,uint256,bytes,uint256) (runs: 256, μ: 39631, ~: 42098) -ERC1155Test:testFailBalanceOfBatchWithArrayMismatch() (gas: 7933) -ERC1155Test:testFailBalanceOfBatchWithArrayMismatch(address[],uint256[]) (runs: 256, μ: 58022, ~: 58788) -ERC1155Test:testFailBatchBurnInsufficientBalance() (gas: 136156) -ERC1155Test:testFailBatchBurnInsufficientBalance(address,uint256[],uint256[],uint256[],bytes) (runs: 256, μ: 1401354, ~: 590599) -ERC1155Test:testFailBatchBurnWithArrayLengthMismatch() (gas: 135542) -ERC1155Test:testFailBatchBurnWithArrayLengthMismatch(address,uint256[],uint256[],uint256[],bytes) (runs: 256, μ: 86139, ~: 76411) -ERC1155Test:testFailBatchMintToNonERC1155Recipient() (gas: 167292) -ERC1155Test:testFailBatchMintToNonERC1155Recipient(uint256[],uint256[],bytes) (runs: 256, μ: 3190100, ~: 2673077) -ERC1155Test:testFailBatchMintToRevertingERC1155Recipient() (gas: 358811) -ERC1155Test:testFailBatchMintToRevertingERC1155Recipient(uint256[],uint256[],bytes) (runs: 256, μ: 3381638, ~: 2864613) -ERC1155Test:testFailBatchMintToWrongReturnDataERC1155Recipient() (gas: 310743) -ERC1155Test:testFailBatchMintToWrongReturnDataERC1155Recipient(uint256[],uint256[],bytes) (runs: 256, μ: 3333596, ~: 2816572) -ERC1155Test:testFailBatchMintToZero() (gas: 131737) -ERC1155Test:testFailBatchMintToZero(uint256[],uint256[],bytes) (runs: 256, μ: 3130600, ~: 2612336) -ERC1155Test:testFailBatchMintWithArrayMismatch() (gas: 9600) -ERC1155Test:testFailBatchMintWithArrayMismatch(address,uint256[],uint256[],bytes) (runs: 256, μ: 66450, ~: 66511) -ERC1155Test:testFailBurnInsufficientBalance() (gas: 34852) -ERC1155Test:testFailBurnInsufficientBalance(address,uint256,uint256,uint256,bytes) (runs: 256, μ: 35106, ~: 38209) -ERC1155Test:testFailMintToNonERC155Recipient() (gas: 68191) -ERC1155Test:testFailMintToNonERC155Recipient(uint256,uint256,bytes) (runs: 256, μ: 68507, ~: 69197) -ERC1155Test:testFailMintToRevertingERC155Recipient() (gas: 259435) -ERC1155Test:testFailMintToRevertingERC155Recipient(uint256,uint256,bytes) (runs: 256, μ: 259682, ~: 260373) -ERC1155Test:testFailMintToWrongReturnDataERC155Recipient() (gas: 259389) -ERC1155Test:testFailMintToWrongReturnDataERC155Recipient(uint256,uint256,bytes) (runs: 256, μ: 259706, ~: 260397) -ERC1155Test:testFailMintToZero() (gas: 33705) -ERC1155Test:testFailMintToZero(uint256,uint256,bytes) (runs: 256, μ: 33815, ~: 34546) -ERC1155Test:testFailSafeBatchTransferFromToNonERC1155Recipient() (gas: 321377) -ERC1155Test:testFailSafeBatchTransferFromToNonERC1155Recipient(uint256[],uint256[],uint256[],bytes,bytes) (runs: 256, μ: 3541141, ~: 2963551) -ERC1155Test:testFailSafeBatchTransferFromToRevertingERC1155Recipient() (gas: 512956) -ERC1155Test:testFailSafeBatchTransferFromToRevertingERC1155Recipient(uint256[],uint256[],uint256[],bytes,bytes) (runs: 256, μ: 3732678, ~: 3155082) -ERC1155Test:testFailSafeBatchTransferFromToWrongReturnDataERC1155Recipient() (gas: 464847) -ERC1155Test:testFailSafeBatchTransferFromToWrongReturnDataERC1155Recipient(uint256[],uint256[],uint256[],bytes,bytes) (runs: 256, μ: 3684595, ~: 3107003) -ERC1155Test:testFailSafeBatchTransferFromToZero() (gas: 286556) -ERC1155Test:testFailSafeBatchTransferFromToZero(uint256[],uint256[],uint256[],bytes,bytes) (runs: 256, μ: 3505960, ~: 2928396) -ERC1155Test:testFailSafeBatchTransferFromWithArrayLengthMismatch() (gas: 162674) -ERC1155Test:testFailSafeBatchTransferFromWithArrayLengthMismatch(address,uint256[],uint256[],uint256[],bytes,bytes) (runs: 256, μ: 79144, ~: 76711) -ERC1155Test:testFailSafeBatchTransferInsufficientBalance() (gas: 163555) -ERC1155Test:testFailSafeBatchTransferInsufficientBalance(address,uint256[],uint256[],uint256[],bytes,bytes) (runs: 256, μ: 1808833, ~: 648144) -ERC1155Test:testFailSafeTransferFromInsufficientBalance() (gas: 63245) -ERC1155Test:testFailSafeTransferFromInsufficientBalance(address,uint256,uint256,uint256,bytes,bytes) (runs: 256, μ: 62944, ~: 67391) -ERC1155Test:testFailSafeTransferFromSelfInsufficientBalance() (gas: 34297) -ERC1155Test:testFailSafeTransferFromSelfInsufficientBalance(address,uint256,uint256,uint256,bytes,bytes) (runs: 256, μ: 35671, ~: 38477) -ERC1155Test:testFailSafeTransferFromToNonERC155Recipient() (gas: 96510) -ERC1155Test:testFailSafeTransferFromToNonERC155Recipient(uint256,uint256,uint256,bytes,bytes) (runs: 256, μ: 96657, ~: 100546) -ERC1155Test:testFailSafeTransferFromToRevertingERC1155Recipient() (gas: 287731) -ERC1155Test:testFailSafeTransferFromToRevertingERC1155Recipient(uint256,uint256,uint256,bytes,bytes) (runs: 256, μ: 287828, ~: 291719) -ERC1155Test:testFailSafeTransferFromToWrongReturnDataERC1155Recipient() (gas: 239587) -ERC1155Test:testFailSafeTransferFromToWrongReturnDataERC1155Recipient(uint256,uint256,uint256,bytes,bytes) (runs: 256, μ: 239707, ~: 243598) -ERC1155Test:testFailSafeTransferFromToZero() (gas: 62014) -ERC1155Test:testFailSafeTransferFromToZero(uint256,uint256,uint256,bytes,bytes) (runs: 256, μ: 62068, ~: 66037) -ERC1155Test:testMintToEOA() (gas: 34765) -ERC1155Test:testMintToEOA(address,uint256,uint256,bytes) (runs: 256, μ: 35273, ~: 35907) -ERC1155Test:testMintToERC1155Recipient() (gas: 661411) -ERC1155Test:testMintToERC1155Recipient(uint256,uint256,bytes) (runs: 256, μ: 691094, ~: 684374) -ERC1155Test:testSafeBatchTransferFromToEOA() (gas: 297822) -ERC1155Test:testSafeBatchTransferFromToEOA(address,uint256[],uint256[],uint256[],bytes,bytes) (runs: 256, μ: 5017684, ~: 4010502) -ERC1155Test:testSafeBatchTransferFromToERC1155Recipient() (gas: 1175327) -ERC1155Test:testSafeBatchTransferFromToERC1155Recipient(uint256[],uint256[],uint256[],bytes,bytes) (runs: 256, μ: 7759463, ~: 6618414) -ERC1155Test:testSafeTransferFromSelf() (gas: 64177) -ERC1155Test:testSafeTransferFromSelf(uint256,uint256,bytes,uint256,address,bytes) (runs: 256, μ: 64681, ~: 68564) -ERC1155Test:testSafeTransferFromToEOA() (gas: 93167) -ERC1155Test:testSafeTransferFromToEOA(uint256,uint256,bytes,uint256,address,bytes) (runs: 256, μ: 93963, ~: 97450) -ERC1155Test:testSafeTransferFromToERC1155Recipient() (gas: 739583) -ERC1155Test:testSafeTransferFromToERC1155Recipient(uint256,uint256,bytes,uint256,bytes) (runs: 256, μ: 769591, ~: 765729) -ERC20Invariants:invariantBalanceSum() (runs: 256, calls: 3840, reverts: 2349) -ERC20Test:invariantMetadata() (runs: 256, calls: 3840, reverts: 2566) -ERC20Test:testApprove() (gas: 31058) -ERC20Test:testApprove(address,uint256) (runs: 256, μ: 30036, ~: 31280) -ERC20Test:testBurn() (gas: 56970) -ERC20Test:testBurn(address,uint256,uint256) (runs: 256, μ: 57875, ~: 59645) -ERC20Test:testFailBurnInsufficientBalance(address,uint256,uint256) (runs: 256, μ: 52089, ~: 55492) -ERC20Test:testFailPermitBadDeadline() (gas: 36935) -ERC20Test:testFailPermitBadDeadline(uint256,address,uint256,uint256) (runs: 256, μ: 31831, ~: 37218) -ERC20Test:testFailPermitBadNonce() (gas: 36874) -ERC20Test:testFailPermitBadNonce(uint256,address,uint256,uint256,uint256) (runs: 256, μ: 32158, ~: 37187) -ERC20Test:testFailPermitPastDeadline() (gas: 11191) -ERC20Test:testFailPermitPastDeadline(uint256,address,uint256,uint256) (runs: 256, μ: 11916, ~: 13101) -ERC20Test:testFailPermitReplay() (gas: 66274) -ERC20Test:testFailPermitReplay(uint256,address,uint256,uint256) (runs: 256, μ: 56082, ~: 66592) -ERC20Test:testFailTransferFromInsufficientAllowance() (gas: 80882) -ERC20Test:testFailTransferFromInsufficientAllowance(address,uint256,uint256) (runs: 256, μ: 79521, ~: 83393) -ERC20Test:testFailTransferFromInsufficientBalance() (gas: 81358) -ERC20Test:testFailTransferFromInsufficientBalance(address,uint256,uint256) (runs: 256, μ: 79323, ~: 83870) -ERC20Test:testFailTransferInsufficientBalance() (gas: 52806) -ERC20Test:testFailTransferInsufficientBalance(address,uint256,uint256) (runs: 256, μ: 51912, ~: 55310) -ERC20Test:testInfiniteApproveTransferFrom() (gas: 89793) -ERC20Test:testMetadata(string,string,uint8) (runs: 256, μ: 846655, ~: 840944) -ERC20Test:testMint() (gas: 53746) -ERC20Test:testMint(address,uint256) (runs: 256, μ: 51437, ~: 53925) -ERC20Test:testPermit() (gas: 63193) -ERC20Test:testPermit(uint248,address,uint256,uint256) (runs: 256, μ: 62506, ~: 63517) -ERC20Test:testTransfer() (gas: 60272) -ERC20Test:testTransfer(address,uint256) (runs: 256, μ: 57996, ~: 60484) -ERC20Test:testTransferFrom() (gas: 83777) -ERC20Test:testTransferFrom(address,uint256,uint256) (runs: 256, μ: 86993, ~: 92841) -ERC4626Test:invariantMetadata() (runs: 256, calls: 3840, reverts: 2883) -ERC4626Test:testFailDepositWithNoApproval() (gas: 13369) -ERC4626Test:testFailDepositWithNotEnoughApproval() (gas: 87005) -ERC4626Test:testFailDepositZero() (gas: 7780) -ERC4626Test:testFailMintWithNoApproval() (gas: 13308) -ERC4626Test:testFailRedeemWithNoShareAmount() (gas: 32342) -ERC4626Test:testFailRedeemWithNotEnoughShareAmount() (gas: 203643) -ERC4626Test:testFailRedeemZero() (gas: 7967) -ERC4626Test:testFailWithdrawWithNoUnderlyingAmount() (gas: 32289) -ERC4626Test:testFailWithdrawWithNotEnoughUnderlyingAmount() (gas: 203607) -ERC4626Test:testMetadata(string,string) (runs: 256, μ: 1471865, ~: 1472396) -ERC4626Test:testMintZero() (gas: 54607) -ERC4626Test:testMultipleMintDepositRedeemWithdraw() (gas: 411804) -ERC4626Test:testSingleDepositWithdraw(uint128) (runs: 256, μ: 201539, ~: 201550) -ERC4626Test:testSingleMintRedeem(uint128) (runs: 256, μ: 201465, ~: 201476) -ERC4626Test:testVaultInteractionsForSomeoneElse() (gas: 286238) -ERC4626Test:testWithdrawZero() (gas: 52468) -ERC6909Test:testApprove() (gas: 31572) -ERC6909Test:testApprove(address,uint256,uint256) (runs: 256, μ: 30860, ~: 31793) -ERC6909Test:testBurn() (gas: 35411) -ERC6909Test:testBurn(address,uint256,uint256) (runs: 256, μ: 23773, ~: 24153) -ERC6909Test:testFailMintBalanceOverflow() (gas: 31727) -ERC6909Test:testFailTransferBalanceOverflow() (gas: 85598) -ERC6909Test:testFailTransferBalanceOverflow(address,address,uint256,uint256) (runs: 256, μ: 87909, ~: 87909) -ERC6909Test:testFailTransferBalanceUnderflow() (gas: 10847) -ERC6909Test:testFailTransferBalanceUnderflow(address,address,uint256,uint256) (runs: 256, μ: 13194, ~: 13194) -ERC6909Test:testFailTransferFromBalanceOverflow() (gas: 85554) -ERC6909Test:testFailTransferFromBalanceOverflow(address,address,uint256,uint256) (runs: 256, μ: 88327, ~: 88327) -ERC6909Test:testFailTransferFromBalanceUnderflow() (gas: 10868) -ERC6909Test:testFailTransferFromBalanceUnderflow(address,address,uint256,uint256) (runs: 256, μ: 13424, ~: 13424) -ERC6909Test:testFailTransferFromNotAuthorized() (gas: 36078) -ERC6909Test:testFailTransferFromNotAuthorized(address,address,uint256,uint256) (runs: 256, μ: 38677, ~: 38677) -ERC6909Test:testMint() (gas: 31579) -ERC6909Test:testMint(address,uint256,uint256) (runs: 256, μ: 30878, ~: 31811) -ERC6909Test:testSetOperator() (gas: 31083) -ERC6909Test:testSetOperator(address,bool) (runs: 256, μ: 22873, ~: 31411) -ERC6909Test:testTransfer() (gas: 61874) -ERC6909Test:testTransfer(address,address,uint256,uint256,uint256) (runs: 256, μ: 59908, ~: 64506) -ERC6909Test:testTransferFromAsOperator() (gas: 87418) -ERC6909Test:testTransferFromAsOperator(address,address,uint256,uint256,uint256) (runs: 256, μ: 85351, ~: 90112) -ERC6909Test:testTransferFromWithApproval() (gas: 91890) -ERC6909Test:testTransferFromWithApproval(address,address,uint256,uint256,uint256) (runs: 256, μ: 88775, ~: 94697) -ERC6909Test:testTransferFromWithInfiniteApproval() (gas: 91541) -ERC6909Test:testTransferFromWithInfiniteApproval(address,address,uint256,uint256,uint256) (runs: 256, μ: 89501, ~: 94262) -ERC721Test:invariantMetadata() (runs: 256, calls: 3840, reverts: 2192) -ERC721Test:testApprove() (gas: 78427) -ERC721Test:testApprove(address,uint256) (runs: 256, μ: 78637, ~: 78637) -ERC721Test:testApproveAll() (gas: 31063) -ERC721Test:testApproveAll(address,bool) (runs: 256, μ: 22869, ~: 31407) -ERC721Test:testApproveBurn() (gas: 65550) -ERC721Test:testApproveBurn(address,uint256) (runs: 256, μ: 65609, ~: 65621) -ERC721Test:testBurn() (gas: 46107) -ERC721Test:testBurn(address,uint256) (runs: 256, μ: 46148, ~: 46160) -ERC721Test:testFailApproveUnAuthorized() (gas: 55598) -ERC721Test:testFailApproveUnAuthorized(address,uint256,address) (runs: 256, μ: 55872, ~: 55873) -ERC721Test:testFailApproveUnMinted() (gas: 10236) -ERC721Test:testFailApproveUnMinted(uint256,address) (runs: 256, μ: 10363, ~: 10363) -ERC721Test:testFailBalanceOfZeroAddress() (gas: 5555) -ERC721Test:testFailBurnUnMinted() (gas: 7857) -ERC721Test:testFailBurnUnMinted(uint256) (runs: 256, μ: 7938, ~: 7938) -ERC721Test:testFailDoubleBurn() (gas: 58943) -ERC721Test:testFailDoubleBurn(uint256,address) (runs: 256, μ: 59174, ~: 59174) -ERC721Test:testFailDoubleMint() (gas: 53286) -ERC721Test:testFailDoubleMint(uint256,address) (runs: 256, μ: 53496, ~: 53496) -ERC721Test:testFailMintToZero() (gas: 5753) -ERC721Test:testFailMintToZero(uint256) (runs: 256, μ: 5835, ~: 5835) -ERC721Test:testFailOwnerOfUnminted() (gas: 7609) -ERC721Test:testFailOwnerOfUnminted(uint256) (runs: 256, μ: 7689, ~: 7689) -ERC721Test:testFailSafeMintToERC721RecipientWithWrongReturnData() (gas: 159076) -ERC721Test:testFailSafeMintToERC721RecipientWithWrongReturnData(uint256) (runs: 256, μ: 159125, ~: 159125) -ERC721Test:testFailSafeMintToERC721RecipientWithWrongReturnDataWithData() (gas: 159831) -ERC721Test:testFailSafeMintToERC721RecipientWithWrongReturnDataWithData(uint256,bytes) (runs: 256, μ: 160231, ~: 160182) -ERC721Test:testFailSafeMintToNonERC721Recipient() (gas: 89210) -ERC721Test:testFailSafeMintToNonERC721Recipient(uint256) (runs: 256, μ: 89279, ~: 89279) -ERC721Test:testFailSafeMintToNonERC721RecipientWithData() (gas: 89995) -ERC721Test:testFailSafeMintToNonERC721RecipientWithData(uint256,bytes) (runs: 256, μ: 90420, ~: 90368) -ERC721Test:testFailSafeMintToRevertingERC721Recipient() (gas: 204743) -ERC721Test:testFailSafeMintToRevertingERC721Recipient(uint256) (runs: 256, μ: 204815, ~: 204815) -ERC721Test:testFailSafeMintToRevertingERC721RecipientWithData() (gas: 205517) -ERC721Test:testFailSafeMintToRevertingERC721RecipientWithData(uint256,bytes) (runs: 256, μ: 205964, ~: 205915) -ERC721Test:testFailSafeTransferFromToERC721RecipientWithWrongReturnData() (gas: 187276) -ERC721Test:testFailSafeTransferFromToERC721RecipientWithWrongReturnData(uint256) (runs: 256, μ: 187360, ~: 187360) -ERC721Test:testFailSafeTransferFromToERC721RecipientWithWrongReturnDataWithData() (gas: 187728) -ERC721Test:testFailSafeTransferFromToERC721RecipientWithWrongReturnDataWithData(uint256,bytes) (runs: 256, μ: 188067, ~: 188063) -ERC721Test:testFailSafeTransferFromToNonERC721Recipient() (gas: 117413) -ERC721Test:testFailSafeTransferFromToNonERC721Recipient(uint256) (runs: 256, μ: 117495, ~: 117495) -ERC721Test:testFailSafeTransferFromToNonERC721RecipientWithData() (gas: 117872) -ERC721Test:testFailSafeTransferFromToNonERC721RecipientWithData(uint256,bytes) (runs: 256, μ: 118280, ~: 118276) -ERC721Test:testFailSafeTransferFromToRevertingERC721Recipient() (gas: 233009) -ERC721Test:testFailSafeTransferFromToRevertingERC721Recipient(uint256) (runs: 256, μ: 233050, ~: 233050) -ERC721Test:testFailSafeTransferFromToRevertingERC721RecipientWithData() (gas: 233396) -ERC721Test:testFailSafeTransferFromToRevertingERC721RecipientWithData(uint256,bytes) (runs: 256, μ: 233823, ~: 233819) -ERC721Test:testFailTransferFromNotOwner() (gas: 57872) -ERC721Test:testFailTransferFromNotOwner(address,address,uint256) (runs: 256, μ: 58144, ~: 58162) -ERC721Test:testFailTransferFromToZero() (gas: 53381) -ERC721Test:testFailTransferFromToZero(uint256) (runs: 256, μ: 53463, ~: 53463) -ERC721Test:testFailTransferFromUnOwned() (gas: 8000) -ERC721Test:testFailTransferFromUnOwned(address,address,uint256) (runs: 256, μ: 8276, ~: 8241) -ERC721Test:testFailTransferFromWrongFrom() (gas: 53361) -ERC721Test:testFailTransferFromWrongFrom(address,address,address,uint256) (runs: 256, μ: 53194, ~: 53752) -ERC721Test:testMetadata(string,string) (runs: 256, μ: 1309634, ~: 1309984) -ERC721Test:testMint() (gas: 54336) -ERC721Test:testMint(address,uint256) (runs: 256, μ: 54521, ~: 54521) -ERC721Test:testSafeMintToEOA() (gas: 56993) -ERC721Test:testSafeMintToEOA(uint256,address) (runs: 256, μ: 56754, ~: 57421) -ERC721Test:testSafeMintToERC721Recipient() (gas: 427035) -ERC721Test:testSafeMintToERC721Recipient(uint256) (runs: 256, μ: 426053, ~: 427142) -ERC721Test:testSafeMintToERC721RecipientWithData() (gas: 448149) -ERC721Test:testSafeMintToERC721RecipientWithData(uint256,bytes) (runs: 256, μ: 480015, ~: 470905) -ERC721Test:testSafeTransferFromToEOA() (gas: 95666) -ERC721Test:testSafeTransferFromToEOA(uint256,address) (runs: 256, μ: 94979, ~: 96099) -ERC721Test:testSafeTransferFromToERC721Recipient() (gas: 485549) -ERC721Test:testSafeTransferFromToERC721Recipient(uint256) (runs: 256, μ: 484593, ~: 485682) -ERC721Test:testSafeTransferFromToERC721RecipientWithData() (gas: 506317) -ERC721Test:testSafeTransferFromToERC721RecipientWithData(uint256,bytes) (runs: 256, μ: 538126, ~: 529082) -ERC721Test:testTransferFrom() (gas: 86347) -ERC721Test:testTransferFrom(uint256,address) (runs: 256, μ: 86467, ~: 86475) -ERC721Test:testTransferFromApproveAll() (gas: 92898) -ERC721Test:testTransferFromApproveAll(uint256,address) (runs: 256, μ: 93181, ~: 93182) -ERC721Test:testTransferFromSelf() (gas: 64776) -ERC721Test:testTransferFromSelf(uint256,address) (runs: 256, μ: 65060, ~: 65061) -FixedPointMathLibTest:testDifferentiallyFuzzSqrt(uint256) (runs: 256, μ: 13868, ~: 6222) -FixedPointMathLibTest:testDivWadDown() (gas: 820) -FixedPointMathLibTest:testDivWadDown(uint256,uint256) (runs: 256, μ: 716, ~: 813) -FixedPointMathLibTest:testDivWadDownEdgeCases() (gas: 439) -FixedPointMathLibTest:testDivWadUp() (gas: 943) -FixedPointMathLibTest:testDivWadUp(uint256,uint256) (runs: 256, μ: 799, ~: 952) -FixedPointMathLibTest:testDivWadUpEdgeCases() (gas: 442) -FixedPointMathLibTest:testFailDivWadDownOverflow(uint256,uint256) (runs: 256, μ: 440, ~: 419) -FixedPointMathLibTest:testFailDivWadDownZeroDenominator() (gas: 332) -FixedPointMathLibTest:testFailDivWadDownZeroDenominator(uint256) (runs: 256, μ: 387, ~: 387) -FixedPointMathLibTest:testFailDivWadUpOverflow(uint256,uint256) (runs: 256, μ: 395, ~: 374) -FixedPointMathLibTest:testFailDivWadUpZeroDenominator() (gas: 332) -FixedPointMathLibTest:testFailDivWadUpZeroDenominator(uint256) (runs: 256, μ: 386, ~: 386) -FixedPointMathLibTest:testFailMulDivDownOverflow(uint256,uint256,uint256) (runs: 256, μ: 436, ~: 414) -FixedPointMathLibTest:testFailMulDivDownZeroDenominator() (gas: 328) -FixedPointMathLibTest:testFailMulDivDownZeroDenominator(uint256,uint256) (runs: 256, μ: 385, ~: 385) -FixedPointMathLibTest:testFailMulDivUpOverflow(uint256,uint256,uint256) (runs: 256, μ: 459, ~: 437) -FixedPointMathLibTest:testFailMulDivUpZeroDenominator() (gas: 329) -FixedPointMathLibTest:testFailMulDivUpZeroDenominator(uint256,uint256) (runs: 256, μ: 428, ~: 428) -FixedPointMathLibTest:testFailMulWadDownOverflow(uint256,uint256) (runs: 256, μ: 419, ~: 387) -FixedPointMathLibTest:testFailMulWadUpOverflow(uint256,uint256) (runs: 256, μ: 396, ~: 364) -FixedPointMathLibTest:testMulDivDown() (gas: 1813) -FixedPointMathLibTest:testMulDivDown(uint256,uint256,uint256) (runs: 256, μ: 680, ~: 786) -FixedPointMathLibTest:testMulDivDownEdgeCases() (gas: 686) -FixedPointMathLibTest:testMulDivUp() (gas: 2095) -FixedPointMathLibTest:testMulDivUp(uint256,uint256,uint256) (runs: 256, μ: 810, ~: 1034) -FixedPointMathLibTest:testMulDivUpEdgeCases() (gas: 785) -FixedPointMathLibTest:testMulWadDown() (gas: 823) -FixedPointMathLibTest:testMulWadDown(uint256,uint256) (runs: 256, μ: 688, ~: 803) -FixedPointMathLibTest:testMulWadDownEdgeCases() (gas: 822) -FixedPointMathLibTest:testMulWadUp() (gas: 921) -FixedPointMathLibTest:testMulWadUp(uint256,uint256) (runs: 256, μ: 834, ~: 1053) -FixedPointMathLibTest:testMulWadUpEdgeCases() (gas: 899) -FixedPointMathLibTest:testRPow() (gas: 2164) -FixedPointMathLibTest:testSqrt() (gas: 2580) -FixedPointMathLibTest:testSqrt(uint256) (runs: 256, μ: 997, ~: 1013) -FixedPointMathLibTest:testSqrtBack(uint256) (runs: 256, μ: 14998, ~: 340) -FixedPointMathLibTest:testSqrtBackHashed(uint256) (runs: 256, μ: 59001, ~: 59500) -FixedPointMathLibTest:testSqrtBackHashedSingle() (gas: 58937) -LibStringTest:testDifferentiallyFuzzToString(uint256,bytes) (runs: 256, μ: 20460, ~: 7749) -LibStringTest:testDifferentiallyFuzzToStringInt(int256,bytes) (runs: 256, μ: 20610, ~: 8980) -LibStringTest:testToString() (gas: 10069) -LibStringTest:testToStringDirty() (gas: 8145) -LibStringTest:testToStringIntNegative() (gas: 9634) -LibStringTest:testToStringIntPositive() (gas: 10481) -LibStringTest:testToStringOverwrite() (gas: 506) -MerkleProofLibTest:testValidProofSupplied() (gas: 2153) -MerkleProofLibTest:testVerifyEmptyMerkleProofSuppliedLeafAndRootDifferent() (gas: 1458) -MerkleProofLibTest:testVerifyEmptyMerkleProofSuppliedLeafAndRootSame() (gas: 1452) -MerkleProofLibTest:testVerifyInvalidProofSupplied() (gas: 2172) -MultiRolesAuthorityTest:testCanCallPublicCapability() (gas: 34204) -MultiRolesAuthorityTest:testCanCallPublicCapability(address,address,bytes4) (runs: 256, μ: 34392, ~: 34361) -MultiRolesAuthorityTest:testCanCallWithAuthorizedRole() (gas: 80416) -MultiRolesAuthorityTest:testCanCallWithAuthorizedRole(address,uint8,address,bytes4) (runs: 256, μ: 80700, ~: 80671) -MultiRolesAuthorityTest:testCanCallWithCustomAuthority() (gas: 422439) -MultiRolesAuthorityTest:testCanCallWithCustomAuthority(address,address,bytes4) (runs: 256, μ: 422858, ~: 422858) -MultiRolesAuthorityTest:testCanCallWithCustomAuthorityOverridesPublicCapability() (gas: 247388) -MultiRolesAuthorityTest:testCanCallWithCustomAuthorityOverridesPublicCapability(address,address,bytes4) (runs: 256, μ: 247841, ~: 247841) -MultiRolesAuthorityTest:testCanCallWithCustomAuthorityOverridesUserWithRole() (gas: 256546) -MultiRolesAuthorityTest:testCanCallWithCustomAuthorityOverridesUserWithRole(address,uint8,address,bytes4) (runs: 256, μ: 256878, ~: 256852) -MultiRolesAuthorityTest:testSetPublicCapabilities() (gas: 27727) -MultiRolesAuthorityTest:testSetPublicCapabilities(bytes4) (runs: 256, μ: 27837, ~: 27835) -MultiRolesAuthorityTest:testSetRoleCapabilities() (gas: 28932) -MultiRolesAuthorityTest:testSetRoleCapabilities(uint8,bytes4) (runs: 256, μ: 29075, ~: 29072) -MultiRolesAuthorityTest:testSetRoles() (gas: 28918) -MultiRolesAuthorityTest:testSetRoles(address,uint8) (runs: 256, μ: 29027, ~: 29014) -MultiRolesAuthorityTest:testSetTargetCustomAuthority() (gas: 28102) -MultiRolesAuthorityTest:testSetTargetCustomAuthority(address,address) (runs: 256, μ: 28115, ~: 28144) -OwnedTest:testCallFunctionAsNonOwner() (gas: 11344) -OwnedTest:testCallFunctionAsNonOwner(address) (runs: 256, μ: 16271, ~: 16290) -OwnedTest:testCallFunctionAsOwner() (gas: 10435) -OwnedTest:testTransferOwnership() (gas: 13123) -OwnedTest:testTransferOwnership(address) (runs: 256, μ: 13162, ~: 13192) -ReentrancyGuardTest:invariantReentrancyStatusAlways1() (runs: 256, calls: 3840, reverts: 254) -ReentrancyGuardTest:testFailUnprotectedCall() (gas: 46147) -ReentrancyGuardTest:testNoReentrancy() (gas: 7515) -ReentrancyGuardTest:testProtectedCall() (gas: 33467) -RolesAuthorityTest:testCanCallPublicCapability() (gas: 33409) -RolesAuthorityTest:testCanCallPublicCapability(address,address,bytes4) (runs: 256, μ: 33559, ~: 33532) -RolesAuthorityTest:testCanCallWithAuthorizedRole() (gas: 79995) -RolesAuthorityTest:testCanCallWithAuthorizedRole(address,uint8,address,bytes4) (runs: 256, μ: 80265, ~: 80238) -RolesAuthorityTest:testSetPublicCapabilities() (gas: 29095) -RolesAuthorityTest:testSetPublicCapabilities(address,bytes4) (runs: 256, μ: 29211, ~: 29192) -RolesAuthorityTest:testSetRoleCapabilities() (gas: 30276) -RolesAuthorityTest:testSetRoleCapabilities(uint8,address,bytes4) (runs: 256, μ: 30507, ~: 30489) -RolesAuthorityTest:testSetRoles() (gas: 29005) -RolesAuthorityTest:testSetRoles(address,uint8) (runs: 256, μ: 29118, ~: 29106) -SSTORE2Test:testFailReadInvalidPointer() (gas: 2927) -SSTORE2Test:testFailReadInvalidPointer(address,bytes) (runs: 256, μ: 3858, ~: 3892) -SSTORE2Test:testFailReadInvalidPointerCustomBounds() (gas: 3099) -SSTORE2Test:testFailReadInvalidPointerCustomBounds(address,uint256,uint256,bytes) (runs: 256, μ: 4072, ~: 4130) -SSTORE2Test:testFailReadInvalidPointerCustomStartBound() (gas: 3004) -SSTORE2Test:testFailReadInvalidPointerCustomStartBound(address,uint256,bytes) (runs: 256, μ: 3960, ~: 3988) -SSTORE2Test:testFailWriteReadCustomBoundsOutOfRange(bytes,uint256,uint256,bytes) (runs: 256, μ: 46236, ~: 43603) -SSTORE2Test:testFailWriteReadCustomStartBoundOutOfRange(bytes,uint256,bytes) (runs: 256, μ: 46017, ~: 43452) -SSTORE2Test:testFailWriteReadEmptyOutOfBounds() (gas: 34470) -SSTORE2Test:testFailWriteReadOutOfBounds() (gas: 34426) -SSTORE2Test:testFailWriteReadOutOfStartBound() (gas: 34362) -SSTORE2Test:testWriteRead() (gas: 53497) -SSTORE2Test:testWriteRead(bytes,bytes) (runs: 256, μ: 44019, ~: 41555) -SSTORE2Test:testWriteReadCustomBounds() (gas: 34869) -SSTORE2Test:testWriteReadCustomBounds(bytes,uint256,uint256,bytes) (runs: 256, μ: 28531, ~: 41976) -SSTORE2Test:testWriteReadCustomStartBound() (gas: 34740) -SSTORE2Test:testWriteReadCustomStartBound(bytes,uint256,bytes) (runs: 256, μ: 46485, ~: 44053) -SSTORE2Test:testWriteReadEmptyBound() (gas: 34677) -SSTORE2Test:testWriteReadFullBoundedRead() (gas: 53672) -SSTORE2Test:testWriteReadFullStartBound() (gas: 34764) -SafeCastLibTest:testFailSafeCastTo104() (gas: 387) -SafeCastLibTest:testFailSafeCastTo104(uint256) (runs: 256, μ: 468, ~: 468) -SafeCastLibTest:testFailSafeCastTo112() (gas: 388) -SafeCastLibTest:testFailSafeCastTo112(uint256) (runs: 256, μ: 445, ~: 445) -SafeCastLibTest:testFailSafeCastTo120() (gas: 409) -SafeCastLibTest:testFailSafeCastTo120(uint256) (runs: 256, μ: 490, ~: 490) -SafeCastLibTest:testFailSafeCastTo128() (gas: 365) -SafeCastLibTest:testFailSafeCastTo128(uint256) (runs: 256, μ: 487, ~: 487) -SafeCastLibTest:testFailSafeCastTo136() (gas: 409) -SafeCastLibTest:testFailSafeCastTo136(uint256) (runs: 256, μ: 489, ~: 489) -SafeCastLibTest:testFailSafeCastTo144() (gas: 365) -SafeCastLibTest:testFailSafeCastTo144(uint256) (runs: 256, μ: 423, ~: 423) -SafeCastLibTest:testFailSafeCastTo152() (gas: 368) -SafeCastLibTest:testFailSafeCastTo152(uint256) (runs: 256, μ: 468, ~: 468) -SafeCastLibTest:testFailSafeCastTo16() (gas: 388) -SafeCastLibTest:testFailSafeCastTo16(uint256) (runs: 256, μ: 468, ~: 468) -SafeCastLibTest:testFailSafeCastTo160() (gas: 409) -SafeCastLibTest:testFailSafeCastTo160(uint256) (runs: 256, μ: 444, ~: 444) -SafeCastLibTest:testFailSafeCastTo168() (gas: 341) -SafeCastLibTest:testFailSafeCastTo168(uint256) (runs: 256, μ: 488, ~: 488) -SafeCastLibTest:testFailSafeCastTo176() (gas: 363) -SafeCastLibTest:testFailSafeCastTo176(uint256) (runs: 256, μ: 489, ~: 489) -SafeCastLibTest:testFailSafeCastTo184() (gas: 343) -SafeCastLibTest:testFailSafeCastTo184(uint256) (runs: 256, μ: 490, ~: 490) -SafeCastLibTest:testFailSafeCastTo192() (gas: 367) -SafeCastLibTest:testFailSafeCastTo192(uint256) (runs: 256, μ: 446, ~: 446) -SafeCastLibTest:testFailSafeCastTo200() (gas: 343) -SafeCastLibTest:testFailSafeCastTo200(uint256) (runs: 256, μ: 490, ~: 490) -SafeCastLibTest:testFailSafeCastTo208() (gas: 386) -SafeCastLibTest:testFailSafeCastTo208(uint256) (runs: 256, μ: 446, ~: 446) -SafeCastLibTest:testFailSafeCastTo216() (gas: 365) -SafeCastLibTest:testFailSafeCastTo216(uint256) (runs: 256, μ: 424, ~: 424) -SafeCastLibTest:testFailSafeCastTo224() (gas: 409) -SafeCastLibTest:testFailSafeCastTo224(uint256) (runs: 256, μ: 423, ~: 423) -SafeCastLibTest:testFailSafeCastTo232() (gas: 410) -SafeCastLibTest:testFailSafeCastTo232(uint256) (runs: 256, μ: 467, ~: 467) -SafeCastLibTest:testFailSafeCastTo24() (gas: 387) -SafeCastLibTest:testFailSafeCastTo24(uint256) (runs: 256, μ: 424, ~: 424) -SafeCastLibTest:testFailSafeCastTo240() (gas: 364) -SafeCastLibTest:testFailSafeCastTo240(uint256) (runs: 256, μ: 467, ~: 467) -SafeCastLibTest:testFailSafeCastTo248() (gas: 365) -SafeCastLibTest:testFailSafeCastTo248(uint256) (runs: 256, μ: 466, ~: 466) -SafeCastLibTest:testFailSafeCastTo32() (gas: 364) -SafeCastLibTest:testFailSafeCastTo32(uint256) (runs: 256, μ: 468, ~: 468) -SafeCastLibTest:testFailSafeCastTo40() (gas: 366) -SafeCastLibTest:testFailSafeCastTo40(uint256) (runs: 256, μ: 422, ~: 422) -SafeCastLibTest:testFailSafeCastTo48() (gas: 366) -SafeCastLibTest:testFailSafeCastTo48(uint256) (runs: 256, μ: 488, ~: 488) -SafeCastLibTest:testFailSafeCastTo56() (gas: 388) -SafeCastLibTest:testFailSafeCastTo56(uint256) (runs: 256, μ: 445, ~: 445) -SafeCastLibTest:testFailSafeCastTo64() (gas: 410) -SafeCastLibTest:testFailSafeCastTo64(uint256) (runs: 256, μ: 446, ~: 446) -SafeCastLibTest:testFailSafeCastTo72() (gas: 410) -SafeCastLibTest:testFailSafeCastTo72(uint256) (runs: 256, μ: 467, ~: 467) -SafeCastLibTest:testFailSafeCastTo8() (gas: 341) -SafeCastLibTest:testFailSafeCastTo8(uint256) (runs: 256, μ: 421, ~: 421) -SafeCastLibTest:testFailSafeCastTo80() (gas: 343) -SafeCastLibTest:testFailSafeCastTo80(uint256) (runs: 256, μ: 424, ~: 424) -SafeCastLibTest:testFailSafeCastTo88() (gas: 344) -SafeCastLibTest:testFailSafeCastTo88(uint256) (runs: 256, μ: 489, ~: 489) -SafeCastLibTest:testFailSafeCastTo96() (gas: 366) -SafeCastLibTest:testFailSafeCastTo96(uint256) (runs: 256, μ: 469, ~: 469) -SafeCastLibTest:testSafeCastTo104() (gas: 515) -SafeCastLibTest:testSafeCastTo104(uint256) (runs: 256, μ: 2779, ~: 2779) -SafeCastLibTest:testSafeCastTo112() (gas: 469) -SafeCastLibTest:testSafeCastTo112(uint256) (runs: 256, μ: 2755, ~: 2755) -SafeCastLibTest:testSafeCastTo120() (gas: 491) -SafeCastLibTest:testSafeCastTo120(uint256) (runs: 256, μ: 2735, ~: 2735) -SafeCastLibTest:testSafeCastTo128() (gas: 516) -SafeCastLibTest:testSafeCastTo128(uint256) (runs: 256, μ: 2735, ~: 2735) -SafeCastLibTest:testSafeCastTo136() (gas: 470) -SafeCastLibTest:testSafeCastTo136(uint256) (runs: 256, μ: 2757, ~: 2757) -SafeCastLibTest:testSafeCastTo144() (gas: 514) -SafeCastLibTest:testSafeCastTo144(uint256) (runs: 256, μ: 2798, ~: 2798) -SafeCastLibTest:testSafeCastTo152() (gas: 494) -SafeCastLibTest:testSafeCastTo152(uint256) (runs: 256, μ: 2734, ~: 2734) -SafeCastLibTest:testSafeCastTo16() (gas: 469) -SafeCastLibTest:testSafeCastTo16(uint256) (runs: 256, μ: 2779, ~: 2779) -SafeCastLibTest:testSafeCastTo160() (gas: 491) -SafeCastLibTest:testSafeCastTo160(uint256) (runs: 256, μ: 2775, ~: 2775) -SafeCastLibTest:testSafeCastTo168() (gas: 494) -SafeCastLibTest:testSafeCastTo168(uint256) (runs: 256, μ: 2799, ~: 2799) -SafeCastLibTest:testSafeCastTo176() (gas: 493) -SafeCastLibTest:testSafeCastTo176(uint256) (runs: 256, μ: 2734, ~: 2734) -SafeCastLibTest:testSafeCastTo184() (gas: 513) -SafeCastLibTest:testSafeCastTo184(uint256) (runs: 256, μ: 2801, ~: 2801) -SafeCastLibTest:testSafeCastTo192() (gas: 494) -SafeCastLibTest:testSafeCastTo192(uint256) (runs: 256, μ: 2734, ~: 2734) -SafeCastLibTest:testSafeCastTo200() (gas: 470) -SafeCastLibTest:testSafeCastTo200(uint256) (runs: 256, μ: 2734, ~: 2734) -SafeCastLibTest:testSafeCastTo208() (gas: 472) -SafeCastLibTest:testSafeCastTo208(uint256) (runs: 256, μ: 2756, ~: 2756) -SafeCastLibTest:testSafeCastTo216() (gas: 493) -SafeCastLibTest:testSafeCastTo216(uint256) (runs: 256, μ: 2777, ~: 2777) -SafeCastLibTest:testSafeCastTo224() (gas: 469) -SafeCastLibTest:testSafeCastTo224(uint256) (runs: 256, μ: 2733, ~: 2733) -SafeCastLibTest:testSafeCastTo232() (gas: 492) -SafeCastLibTest:testSafeCastTo232(uint256) (runs: 256, μ: 2735, ~: 2735) -SafeCastLibTest:testSafeCastTo24() (gas: 515) -SafeCastLibTest:testSafeCastTo24(uint256) (runs: 256, μ: 2733, ~: 2733) -SafeCastLibTest:testSafeCastTo240() (gas: 513) -SafeCastLibTest:testSafeCastTo240(uint256) (runs: 256, μ: 2800, ~: 2800) -SafeCastLibTest:testSafeCastTo248() (gas: 472) -SafeCastLibTest:testSafeCastTo248(uint256) (runs: 256, μ: 2777, ~: 2777) -SafeCastLibTest:testSafeCastTo32() (gas: 516) -SafeCastLibTest:testSafeCastTo32(uint256) (runs: 256, μ: 2777, ~: 2777) -SafeCastLibTest:testSafeCastTo40() (gas: 517) -SafeCastLibTest:testSafeCastTo40(uint256) (runs: 256, μ: 2756, ~: 2756) -SafeCastLibTest:testSafeCastTo48() (gas: 469) -SafeCastLibTest:testSafeCastTo48(uint256) (runs: 256, μ: 2778, ~: 2778) -SafeCastLibTest:testSafeCastTo56() (gas: 470) -SafeCastLibTest:testSafeCastTo56(uint256) (runs: 256, μ: 2801, ~: 2801) -SafeCastLibTest:testSafeCastTo64() (gas: 537) -SafeCastLibTest:testSafeCastTo64(uint256) (runs: 256, μ: 2799, ~: 2799) -SafeCastLibTest:testSafeCastTo72(uint256) (runs: 256, μ: 2798, ~: 2798) -SafeCastLibTest:testSafeCastTo8() (gas: 513) -SafeCastLibTest:testSafeCastTo8(uint256) (runs: 256, μ: 2755, ~: 2755) -SafeCastLibTest:testSafeCastTo80(uint256) (runs: 256, μ: 2736, ~: 2736) -SafeCastLibTest:testSafeCastTo88(uint256) (runs: 256, μ: 2755, ~: 2755) -SafeCastLibTest:testSafeCastTo96() (gas: 536) -SafeCastLibTest:testSafeCastTo96(uint256) (runs: 256, μ: 2800, ~: 2800) -SafeTransferLibTest:testApproveWithGarbage(address,uint256,bytes,bytes) (runs: 256, μ: 3016, ~: 2231) -SafeTransferLibTest:testApproveWithMissingReturn() (gas: 30757) -SafeTransferLibTest:testApproveWithMissingReturn(address,uint256,bytes) (runs: 256, μ: 30800, ~: 31572) -SafeTransferLibTest:testApproveWithNonContract() (gas: 3041) -SafeTransferLibTest:testApproveWithNonContract(address,address,uint256,bytes) (runs: 256, μ: 4095, ~: 4123) -SafeTransferLibTest:testApproveWithReturnsTooMuch() (gas: 31140) -SafeTransferLibTest:testApproveWithReturnsTooMuch(address,uint256,bytes) (runs: 256, μ: 31268, ~: 32040) -SafeTransferLibTest:testApproveWithStandardERC20() (gas: 30888) -SafeTransferLibTest:testApproveWithStandardERC20(address,uint256,bytes) (runs: 256, μ: 30994, ~: 31766) -SafeTransferLibTest:testFailApproveWithGarbage(address,uint256,bytes,bytes) (runs: 256, μ: 85288, ~: 77915) -SafeTransferLibTest:testFailApproveWithReturnsFalse() (gas: 5633) -SafeTransferLibTest:testFailApproveWithReturnsFalse(address,uint256,bytes) (runs: 256, μ: 6486, ~: 6481) -SafeTransferLibTest:testFailApproveWithReturnsTooLittle() (gas: 5574) -SafeTransferLibTest:testFailApproveWithReturnsTooLittle(address,uint256,bytes) (runs: 256, μ: 6450, ~: 6445) -SafeTransferLibTest:testFailApproveWithReturnsTwo(address,uint256,bytes) (runs: 256, μ: 6458, ~: 6453) -SafeTransferLibTest:testFailApproveWithReverting() (gas: 5508) -SafeTransferLibTest:testFailApproveWithReverting(address,uint256,bytes) (runs: 256, μ: 6409, ~: 6404) -SafeTransferLibTest:testFailTransferETHToContractWithoutFallback() (gas: 7244) -SafeTransferLibTest:testFailTransferETHToContractWithoutFallback(uint256,bytes) (runs: 256, μ: 7758, ~: 8055) -SafeTransferLibTest:testFailTransferFromWithGarbage(address,address,uint256,bytes,bytes) (runs: 256, μ: 120098, ~: 117413) -SafeTransferLibTest:testFailTransferFromWithReturnsFalse() (gas: 13675) -SafeTransferLibTest:testFailTransferFromWithReturnsFalse(address,address,uint256,bytes) (runs: 256, μ: 14606, ~: 14600) -SafeTransferLibTest:testFailTransferFromWithReturnsTooLittle() (gas: 13556) -SafeTransferLibTest:testFailTransferFromWithReturnsTooLittle(address,address,uint256,bytes) (runs: 256, μ: 14465, ~: 14459) -SafeTransferLibTest:testFailTransferFromWithReturnsTwo(address,address,uint256,bytes) (runs: 256, μ: 14572, ~: 14566) -SafeTransferLibTest:testFailTransferFromWithReverting() (gas: 9757) -SafeTransferLibTest:testFailTransferFromWithReverting(address,address,uint256,bytes) (runs: 256, μ: 10686, ~: 10680) -SafeTransferLibTest:testFailTransferWithGarbage(address,uint256,bytes,bytes) (runs: 256, μ: 91079, ~: 83995) -SafeTransferLibTest:testFailTransferWithReturnsFalse() (gas: 8538) -SafeTransferLibTest:testFailTransferWithReturnsFalse(address,uint256,bytes) (runs: 256, μ: 9457, ~: 9452) -SafeTransferLibTest:testFailTransferWithReturnsTooLittle() (gas: 8544) -SafeTransferLibTest:testFailTransferWithReturnsTooLittle(address,uint256,bytes) (runs: 256, μ: 9397, ~: 9392) -SafeTransferLibTest:testFailTransferWithReturnsTwo(address,uint256,bytes) (runs: 256, μ: 9384, ~: 9379) -SafeTransferLibTest:testFailTransferWithReverting() (gas: 8500) -SafeTransferLibTest:testFailTransferWithReverting(address,uint256,bytes) (runs: 256, μ: 9356, ~: 9351) -SafeTransferLibTest:testTransferETH() (gas: 34592) -SafeTransferLibTest:testTransferETH(address,uint256,bytes) (runs: 256, μ: 35787, ~: 37975) -SafeTransferLibTest:testTransferFromWithGarbage(address,address,uint256,bytes,bytes) (runs: 256, μ: 3348, ~: 2253) -SafeTransferLibTest:testTransferFromWithMissingReturn() (gas: 49196) -SafeTransferLibTest:testTransferFromWithMissingReturn(address,address,uint256,bytes) (runs: 256, μ: 48997, ~: 49580) -SafeTransferLibTest:testTransferFromWithNonContract() (gas: 3047) -SafeTransferLibTest:testTransferFromWithNonContract(address,address,address,uint256,bytes) (runs: 256, μ: 4234, ~: 4240) -SafeTransferLibTest:testTransferFromWithReturnsTooMuch() (gas: 49820) -SafeTransferLibTest:testTransferFromWithReturnsTooMuch(address,address,uint256,bytes) (runs: 256, μ: 49640, ~: 50219) -SafeTransferLibTest:testTransferFromWithStandardERC20() (gas: 47612) -SafeTransferLibTest:testTransferFromWithStandardERC20(address,address,uint256,bytes) (runs: 256, μ: 47345, ~: 48031) -SafeTransferLibTest:testTransferWithGarbage(address,uint256,bytes,bytes) (runs: 256, μ: 3006, ~: 2187) -SafeTransferLibTest:testTransferWithMissingReturn() (gas: 36672) -SafeTransferLibTest:testTransferWithMissingReturn(address,uint256,bytes) (runs: 256, μ: 36539, ~: 37552) -SafeTransferLibTest:testTransferWithNonContract() (gas: 3018) -SafeTransferLibTest:testTransferWithNonContract(address,address,uint256,bytes) (runs: 256, μ: 4159, ~: 4187) -SafeTransferLibTest:testTransferWithReturnsTooMuch() (gas: 37118) -SafeTransferLibTest:testTransferWithReturnsTooMuch(address,uint256,bytes) (runs: 256, μ: 36942, ~: 37955) -SafeTransferLibTest:testTransferWithStandardERC20() (gas: 36702) -SafeTransferLibTest:testTransferWithStandardERC20(address,uint256,bytes) (runs: 256, μ: 36592, ~: 37605) -SignedWadMathTest:testFailWadDivOverflow(int256,int256) (runs: 256, μ: 347, ~: 329) -SignedWadMathTest:testFailWadDivZeroDenominator(int256) (runs: 256, μ: 296, ~: 296) -SignedWadMathTest:testFailWadMulEdgeCase() (gas: 286) -SignedWadMathTest:testFailWadMulEdgeCase2() (gas: 309) -SignedWadMathTest:testFailWadMulOverflow(int256,int256) (runs: 256, μ: 354, ~: 319) -SignedWadMathTest:testWadDiv(uint256,uint256,bool,bool) (runs: 256, μ: 5712, ~: 5714) -SignedWadMathTest:testWadMul(uint256,uint256,bool,bool) (runs: 256, μ: 5760, ~: 5762) -WETHInvariants:invariantTotalSupplyEqualsBalance() (runs: 256, calls: 3840, reverts: 1776) -WETHTest:testDeposit() (gas: 63535) -WETHTest:testDeposit(uint256) (runs: 256, μ: 63155, ~: 65880) -WETHTest:testFallbackDeposit() (gas: 63249) -WETHTest:testFallbackDeposit(uint256) (runs: 256, μ: 62879, ~: 65604) -WETHTest:testPartialWithdraw() (gas: 73281) -WETHTest:testWithdraw() (gas: 54360) -WETHTest:testWithdraw(uint256,uint256) (runs: 256, μ: 75313, ~: 78076) \ No newline at end of file diff --git a/lib/solmate/.gitattributes b/lib/solmate/.gitattributes deleted file mode 100644 index 7c5f3db..0000000 --- a/lib/solmate/.gitattributes +++ /dev/null @@ -1 +0,0 @@ -.gas-snapshot linguist-language=Julia \ No newline at end of file diff --git a/lib/solmate/.github/pull_request_template.md b/lib/solmate/.github/pull_request_template.md deleted file mode 100644 index 5cca391..0000000 --- a/lib/solmate/.github/pull_request_template.md +++ /dev/null @@ -1,13 +0,0 @@ -## Description - -Describe the changes made in your pull request here. - -## Checklist - -Ensure you completed **all of the steps** below before submitting your pull request: - -- [ ] Ran `forge snapshot`? -- [ ] Ran `npm run lint`? -- [ ] Ran `forge test`? - -_Pull requests with an incomplete checklist will be thrown out._ diff --git a/lib/solmate/.github/workflows/tests.yml b/lib/solmate/.github/workflows/tests.yml deleted file mode 100644 index 2a29890..0000000 --- a/lib/solmate/.github/workflows/tests.yml +++ /dev/null @@ -1,29 +0,0 @@ -name: Tests - -on: [push, pull_request] - -jobs: - tests: - runs-on: ubuntu-latest - steps: - - uses: actions/checkout@v2 - - - name: Install Foundry - uses: onbjerg/foundry-toolchain@v1 - with: - version: nightly - - - name: Install dependencies - run: forge install - - - name: Check contract sizes - run: forge build --sizes - - - name: Check gas snapshots - run: forge snapshot --check - - - name: Run tests - run: forge test - env: - # Only fuzz intensely if we're running this action on a push to main or for a PR going into main: - FOUNDRY_PROFILE: ${{ (github.ref == 'refs/heads/main' || github.base_ref == 'main') && 'intense' }} diff --git a/lib/solmate/.gitignore b/lib/solmate/.gitignore deleted file mode 100644 index 5dfe93f..0000000 --- a/lib/solmate/.gitignore +++ /dev/null @@ -1,3 +0,0 @@ -/cache -/node_modules -/out \ No newline at end of file diff --git a/lib/solmate/.gitmodules b/lib/solmate/.gitmodules deleted file mode 100644 index e124719..0000000 --- a/lib/solmate/.gitmodules +++ /dev/null @@ -1,3 +0,0 @@ -[submodule "lib/ds-test"] - path = lib/ds-test - url = https://github.com/dapphub/ds-test diff --git a/lib/solmate/.prettierignore b/lib/solmate/.prettierignore deleted file mode 100644 index 7951405..0000000 --- a/lib/solmate/.prettierignore +++ /dev/null @@ -1 +0,0 @@ -lib \ No newline at end of file diff --git a/lib/solmate/.prettierrc b/lib/solmate/.prettierrc deleted file mode 100644 index 15ae8a7..0000000 --- a/lib/solmate/.prettierrc +++ /dev/null @@ -1,14 +0,0 @@ -{ - "tabWidth": 2, - "printWidth": 100, - - "overrides": [ - { - "files": "*.sol", - "options": { - "tabWidth": 4, - "printWidth": 120 - } - } - ] -} diff --git a/lib/solmate/.vscode/settings.json b/lib/solmate/.vscode/settings.json deleted file mode 100644 index f91a167..0000000 --- a/lib/solmate/.vscode/settings.json +++ /dev/null @@ -1,9 +0,0 @@ -{ - "solidity.packageDefaultDependenciesContractsDirectory": "src", - "solidity.packageDefaultDependenciesDirectory": "lib", - "solidity.compileUsingRemoteVersion": "v0.8.15", - "search.exclude": { "lib": true }, - "files.associations": { - ".gas-snapshot": "julia" - } -} diff --git a/lib/solmate/LICENSE b/lib/solmate/LICENSE deleted file mode 100644 index 29ebfa5..0000000 --- a/lib/solmate/LICENSE +++ /dev/null @@ -1,661 +0,0 @@ - GNU AFFERO GENERAL PUBLIC LICENSE - Version 3, 19 November 2007 - - Copyright (C) 2007 Free Software Foundation, Inc. - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - - Preamble - - The GNU Affero General Public License is a free, copyleft license for -software and other kinds of works, specifically designed to ensure -cooperation with the community in the case of network server software. - - The licenses for most software and other practical works are designed -to take away your freedom to share and change the works. By contrast, -our General Public Licenses are intended to guarantee your freedom to -share and change all versions of a program--to make sure it remains free -software for all its users. - - When we speak of free software, we are referring to freedom, not -price. Our General Public Licenses are designed to make sure that you -have the freedom to distribute copies of free software (and charge for -them if you wish), that you receive source code or can get it if you -want it, that you can change the software or use pieces of it in new -free programs, and that you know you can do these things. - - Developers that use our General Public Licenses protect your rights -with two steps: (1) assert copyright on the software, and (2) offer -you this License which gives you legal permission to copy, distribute -and/or modify the software. - - A secondary benefit of defending all users' freedom is that -improvements made in alternate versions of the program, if they -receive widespread use, become available for other developers to -incorporate. Many developers of free software are heartened and -encouraged by the resulting cooperation. However, in the case of -software used on network servers, this result may fail to come about. -The GNU General Public License permits making a modified version and -letting the public access it on a server without ever releasing its -source code to the public. - - The GNU Affero General Public License is designed specifically to -ensure that, in such cases, the modified source code becomes available -to the community. It requires the operator of a network server to -provide the source code of the modified version running there to the -users of that server. Therefore, public use of a modified version, on -a publicly accessible server, gives the public access to the source -code of the modified version. - - An older license, called the Affero General Public License and -published by Affero, was designed to accomplish similar goals. This is -a different license, not a version of the Affero GPL, but Affero has -released a new version of the Affero GPL which permits relicensing under -this license. - - The precise terms and conditions for copying, distribution and -modification follow. - - TERMS AND CONDITIONS - - 0. Definitions. - - "This License" refers to version 3 of the GNU Affero General Public License. - - "Copyright" also means copyright-like laws that apply to other kinds of -works, such as semiconductor masks. - - "The Program" refers to any copyrightable work licensed under this -License. Each licensee is addressed as "you". "Licensees" and -"recipients" may be individuals or organizations. - - To "modify" a work means to copy from or adapt all or part of the work -in a fashion requiring copyright permission, other than the making of an -exact copy. The resulting work is called a "modified version" of the -earlier work or a work "based on" the earlier work. - - A "covered work" means either the unmodified Program or a work based -on the Program. - - To "propagate" a work means to do anything with it that, without -permission, would make you directly or secondarily liable for -infringement under applicable copyright law, except executing it on a -computer or modifying a private copy. Propagation includes copying, -distribution (with or without modification), making available to the -public, and in some countries other activities as well. - - To "convey" a work means any kind of propagation that enables other -parties to make or receive copies. Mere interaction with a user through -a computer network, with no transfer of a copy, is not conveying. - - An interactive user interface displays "Appropriate Legal Notices" -to the extent that it includes a convenient and prominently visible -feature that (1) displays an appropriate copyright notice, and (2) -tells the user that there is no warranty for the work (except to the -extent that warranties are provided), that licensees may convey the -work under this License, and how to view a copy of this License. If -the interface presents a list of user commands or options, such as a -menu, a prominent item in the list meets this criterion. - - 1. Source Code. - - The "source code" for a work means the preferred form of the work -for making modifications to it. "Object code" means any non-source -form of a work. - - A "Standard Interface" means an interface that either is an official -standard defined by a recognized standards body, or, in the case of -interfaces specified for a particular programming language, one that -is widely used among developers working in that language. - - The "System Libraries" of an executable work include anything, other -than the work as a whole, that (a) is included in the normal form of -packaging a Major Component, but which is not part of that Major -Component, and (b) serves only to enable use of the work with that -Major Component, or to implement a Standard Interface for which an -implementation is available to the public in source code form. A -"Major Component", in this context, means a major essential component -(kernel, window system, and so on) of the specific operating system -(if any) on which the executable work runs, or a compiler used to -produce the work, or an object code interpreter used to run it. - - The "Corresponding Source" for a work in object code form means all -the source code needed to generate, install, and (for an executable -work) run the object code and to modify the work, including scripts to -control those activities. However, it does not include the work's -System Libraries, or general-purpose tools or generally available free -programs which are used unmodified in performing those activities but -which are not part of the work. For example, Corresponding Source -includes interface definition files associated with source files for -the work, and the source code for shared libraries and dynamically -linked subprograms that the work is specifically designed to require, -such as by intimate data communication or control flow between those -subprograms and other parts of the work. - - The Corresponding Source need not include anything that users -can regenerate automatically from other parts of the Corresponding -Source. - - The Corresponding Source for a work in source code form is that -same work. - - 2. Basic Permissions. - - All rights granted under this License are granted for the term of -copyright on the Program, and are irrevocable provided the stated -conditions are met. This License explicitly affirms your unlimited -permission to run the unmodified Program. The output from running a -covered work is covered by this License only if the output, given its -content, constitutes a covered work. This License acknowledges your -rights of fair use or other equivalent, as provided by copyright law. - - You may make, run and propagate covered works that you do not -convey, without conditions so long as your license otherwise remains -in force. You may convey covered works to others for the sole purpose -of having them make modifications exclusively for you, or provide you -with facilities for running those works, provided that you comply with -the terms of this License in conveying all material for which you do -not control copyright. Those thus making or running the covered works -for you must do so exclusively on your behalf, under your direction -and control, on terms that prohibit them from making any copies of -your copyrighted material outside their relationship with you. - - Conveying under any other circumstances is permitted solely under -the conditions stated below. Sublicensing is not allowed; section 10 -makes it unnecessary. - - 3. Protecting Users' Legal Rights From Anti-Circumvention Law. - - No covered work shall be deemed part of an effective technological -measure under any applicable law fulfilling obligations under article -11 of the WIPO copyright treaty adopted on 20 December 1996, or -similar laws prohibiting or restricting circumvention of such -measures. - - When you convey a covered work, you waive any legal power to forbid -circumvention of technological measures to the extent such circumvention -is effected by exercising rights under this License with respect to -the covered work, and you disclaim any intention to limit operation or -modification of the work as a means of enforcing, against the work's -users, your or third parties' legal rights to forbid circumvention of -technological measures. - - 4. Conveying Verbatim Copies. - - You may convey verbatim copies of the Program's source code as you -receive it, in any medium, provided that you conspicuously and -appropriately publish on each copy an appropriate copyright notice; -keep intact all notices stating that this License and any -non-permissive terms added in accord with section 7 apply to the code; -keep intact all notices of the absence of any warranty; and give all -recipients a copy of this License along with the Program. - - You may charge any price or no price for each copy that you convey, -and you may offer support or warranty protection for a fee. - - 5. Conveying Modified Source Versions. - - You may convey a work based on the Program, or the modifications to -produce it from the Program, in the form of source code under the -terms of section 4, provided that you also meet all of these conditions: - - a) The work must carry prominent notices stating that you modified - it, and giving a relevant date. - - b) The work must carry prominent notices stating that it is - released under this License and any conditions added under section - 7. This requirement modifies the requirement in section 4 to - "keep intact all notices". - - c) You must license the entire work, as a whole, under this - License to anyone who comes into possession of a copy. This - License will therefore apply, along with any applicable section 7 - additional terms, to the whole of the work, and all its parts, - regardless of how they are packaged. This License gives no - permission to license the work in any other way, but it does not - invalidate such permission if you have separately received it. - - d) If the work has interactive user interfaces, each must display - Appropriate Legal Notices; however, if the Program has interactive - interfaces that do not display Appropriate Legal Notices, your - work need not make them do so. - - A compilation of a covered work with other separate and independent -works, which are not by their nature extensions of the covered work, -and which are not combined with it such as to form a larger program, -in or on a volume of a storage or distribution medium, is called an -"aggregate" if the compilation and its resulting copyright are not -used to limit the access or legal rights of the compilation's users -beyond what the individual works permit. Inclusion of a covered work -in an aggregate does not cause this License to apply to the other -parts of the aggregate. - - 6. Conveying Non-Source Forms. - - You may convey a covered work in object code form under the terms -of sections 4 and 5, provided that you also convey the -machine-readable Corresponding Source under the terms of this License, -in one of these ways: - - a) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by the - Corresponding Source fixed on a durable physical medium - customarily used for software interchange. - - b) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by a - written offer, valid for at least three years and valid for as - long as you offer spare parts or customer support for that product - model, to give anyone who possesses the object code either (1) a - copy of the Corresponding Source for all the software in the - product that is covered by this License, on a durable physical - medium customarily used for software interchange, for a price no - more than your reasonable cost of physically performing this - conveying of source, or (2) access to copy the - Corresponding Source from a network server at no charge. - - c) Convey individual copies of the object code with a copy of the - written offer to provide the Corresponding Source. This - alternative is allowed only occasionally and noncommercially, and - only if you received the object code with such an offer, in accord - with subsection 6b. - - d) Convey the object code by offering access from a designated - place (gratis or for a charge), and offer equivalent access to the - Corresponding Source in the same way through the same place at no - further charge. You need not require recipients to copy the - Corresponding Source along with the object code. If the place to - copy the object code is a network server, the Corresponding Source - may be on a different server (operated by you or a third party) - that supports equivalent copying facilities, provided you maintain - clear directions next to the object code saying where to find the - Corresponding Source. Regardless of what server hosts the - Corresponding Source, you remain obligated to ensure that it is - available for as long as needed to satisfy these requirements. - - e) Convey the object code using peer-to-peer transmission, provided - you inform other peers where the object code and Corresponding - Source of the work are being offered to the general public at no - charge under subsection 6d. - - A separable portion of the object code, whose source code is excluded -from the Corresponding Source as a System Library, need not be -included in conveying the object code work. - - A "User Product" is either (1) a "consumer product", which means any -tangible personal property which is normally used for personal, family, -or household purposes, or (2) anything designed or sold for incorporation -into a dwelling. In determining whether a product is a consumer product, -doubtful cases shall be resolved in favor of coverage. For a particular -product received by a particular user, "normally used" refers to a -typical or common use of that class of product, regardless of the status -of the particular user or of the way in which the particular user -actually uses, or expects or is expected to use, the product. A product -is a consumer product regardless of whether the product has substantial -commercial, industrial or non-consumer uses, unless such uses represent -the only significant mode of use of the product. - - "Installation Information" for a User Product means any methods, -procedures, authorization keys, or other information required to install -and execute modified versions of a covered work in that User Product from -a modified version of its Corresponding Source. The information must -suffice to ensure that the continued functioning of the modified object -code is in no case prevented or interfered with solely because -modification has been made. - - If you convey an object code work under this section in, or with, or -specifically for use in, a User Product, and the conveying occurs as -part of a transaction in which the right of possession and use of the -User Product is transferred to the recipient in perpetuity or for a -fixed term (regardless of how the transaction is characterized), the -Corresponding Source conveyed under this section must be accompanied -by the Installation Information. But this requirement does not apply -if neither you nor any third party retains the ability to install -modified object code on the User Product (for example, the work has -been installed in ROM). - - The requirement to provide Installation Information does not include a -requirement to continue to provide support service, warranty, or updates -for a work that has been modified or installed by the recipient, or for -the User Product in which it has been modified or installed. Access to a -network may be denied when the modification itself materially and -adversely affects the operation of the network or violates the rules and -protocols for communication across the network. - - Corresponding Source conveyed, and Installation Information provided, -in accord with this section must be in a format that is publicly -documented (and with an implementation available to the public in -source code form), and must require no special password or key for -unpacking, reading or copying. - - 7. Additional Terms. - - "Additional permissions" are terms that supplement the terms of this -License by making exceptions from one or more of its conditions. -Additional permissions that are applicable to the entire Program shall -be treated as though they were included in this License, to the extent -that they are valid under applicable law. If additional permissions -apply only to part of the Program, that part may be used separately -under those permissions, but the entire Program remains governed by -this License without regard to the additional permissions. - - When you convey a copy of a covered work, you may at your option -remove any additional permissions from that copy, or from any part of -it. (Additional permissions may be written to require their own -removal in certain cases when you modify the work.) You may place -additional permissions on material, added by you to a covered work, -for which you have or can give appropriate copyright permission. - - Notwithstanding any other provision of this License, for material you -add to a covered work, you may (if authorized by the copyright holders of -that material) supplement the terms of this License with terms: - - a) Disclaiming warranty or limiting liability differently from the - terms of sections 15 and 16 of this License; or - - b) Requiring preservation of specified reasonable legal notices or - author attributions in that material or in the Appropriate Legal - Notices displayed by works containing it; or - - c) Prohibiting misrepresentation of the origin of that material, or - requiring that modified versions of such material be marked in - reasonable ways as different from the original version; or - - d) Limiting the use for publicity purposes of names of licensors or - authors of the material; or - - e) Declining to grant rights under trademark law for use of some - trade names, trademarks, or service marks; or - - f) Requiring indemnification of licensors and authors of that - material by anyone who conveys the material (or modified versions of - it) with contractual assumptions of liability to the recipient, for - any liability that these contractual assumptions directly impose on - those licensors and authors. - - All other non-permissive additional terms are considered "further -restrictions" within the meaning of section 10. If the Program as you -received it, or any part of it, contains a notice stating that it is -governed by this License along with a term that is a further -restriction, you may remove that term. If a license document contains -a further restriction but permits relicensing or conveying under this -License, you may add to a covered work material governed by the terms -of that license document, provided that the further restriction does -not survive such relicensing or conveying. - - If you add terms to a covered work in accord with this section, you -must place, in the relevant source files, a statement of the -additional terms that apply to those files, or a notice indicating -where to find the applicable terms. - - Additional terms, permissive or non-permissive, may be stated in the -form of a separately written license, or stated as exceptions; -the above requirements apply either way. - - 8. Termination. - - You may not propagate or modify a covered work except as expressly -provided under this License. Any attempt otherwise to propagate or -modify it is void, and will automatically terminate your rights under -this License (including any patent licenses granted under the third -paragraph of section 11). - - However, if you cease all violation of this License, then your -license from a particular copyright holder is reinstated (a) -provisionally, unless and until the copyright holder explicitly and -finally terminates your license, and (b) permanently, if the copyright -holder fails to notify you of the violation by some reasonable means -prior to 60 days after the cessation. - - Moreover, your license from a particular copyright holder is -reinstated permanently if the copyright holder notifies you of the -violation by some reasonable means, this is the first time you have -received notice of violation of this License (for any work) from that -copyright holder, and you cure the violation prior to 30 days after -your receipt of the notice. - - Termination of your rights under this section does not terminate the -licenses of parties who have received copies or rights from you under -this License. If your rights have been terminated and not permanently -reinstated, you do not qualify to receive new licenses for the same -material under section 10. - - 9. Acceptance Not Required for Having Copies. - - You are not required to accept this License in order to receive or -run a copy of the Program. Ancillary propagation of a covered work -occurring solely as a consequence of using peer-to-peer transmission -to receive a copy likewise does not require acceptance. However, -nothing other than this License grants you permission to propagate or -modify any covered work. These actions infringe copyright if you do -not accept this License. Therefore, by modifying or propagating a -covered work, you indicate your acceptance of this License to do so. - - 10. Automatic Licensing of Downstream Recipients. - - Each time you convey a covered work, the recipient automatically -receives a license from the original licensors, to run, modify and -propagate that work, subject to this License. You are not responsible -for enforcing compliance by third parties with this License. - - An "entity transaction" is a transaction transferring control of an -organization, or substantially all assets of one, or subdividing an -organization, or merging organizations. If propagation of a covered -work results from an entity transaction, each party to that -transaction who receives a copy of the work also receives whatever -licenses to the work the party's predecessor in interest had or could -give under the previous paragraph, plus a right to possession of the -Corresponding Source of the work from the predecessor in interest, if -the predecessor has it or can get it with reasonable efforts. - - You may not impose any further restrictions on the exercise of the -rights granted or affirmed under this License. For example, you may -not impose a license fee, royalty, or other charge for exercise of -rights granted under this License, and you may not initiate litigation -(including a cross-claim or counterclaim in a lawsuit) alleging that -any patent claim is infringed by making, using, selling, offering for -sale, or importing the Program or any portion of it. - - 11. Patents. - - A "contributor" is a copyright holder who authorizes use under this -License of the Program or a work on which the Program is based. The -work thus licensed is called the contributor's "contributor version". - - A contributor's "essential patent claims" are all patent claims -owned or controlled by the contributor, whether already acquired or -hereafter acquired, that would be infringed by some manner, permitted -by this License, of making, using, or selling its contributor version, -but do not include claims that would be infringed only as a -consequence of further modification of the contributor version. For -purposes of this definition, "control" includes the right to grant -patent sublicenses in a manner consistent with the requirements of -this License. - - Each contributor grants you a non-exclusive, worldwide, royalty-free -patent license under the contributor's essential patent claims, to -make, use, sell, offer for sale, import and otherwise run, modify and -propagate the contents of its contributor version. - - In the following three paragraphs, a "patent license" is any express -agreement or commitment, however denominated, not to enforce a patent -(such as an express permission to practice a patent or covenant not to -sue for patent infringement). To "grant" such a patent license to a -party means to make such an agreement or commitment not to enforce a -patent against the party. - - If you convey a covered work, knowingly relying on a patent license, -and the Corresponding Source of the work is not available for anyone -to copy, free of charge and under the terms of this License, through a -publicly available network server or other readily accessible means, -then you must either (1) cause the Corresponding Source to be so -available, or (2) arrange to deprive yourself of the benefit of the -patent license for this particular work, or (3) arrange, in a manner -consistent with the requirements of this License, to extend the patent -license to downstream recipients. "Knowingly relying" means you have -actual knowledge that, but for the patent license, your conveying the -covered work in a country, or your recipient's use of the covered work -in a country, would infringe one or more identifiable patents in that -country that you have reason to believe are valid. - - If, pursuant to or in connection with a single transaction or -arrangement, you convey, or propagate by procuring conveyance of, a -covered work, and grant a patent license to some of the parties -receiving the covered work authorizing them to use, propagate, modify -or convey a specific copy of the covered work, then the patent license -you grant is automatically extended to all recipients of the covered -work and works based on it. - - A patent license is "discriminatory" if it does not include within -the scope of its coverage, prohibits the exercise of, or is -conditioned on the non-exercise of one or more of the rights that are -specifically granted under this License. You may not convey a covered -work if you are a party to an arrangement with a third party that is -in the business of distributing software, under which you make payment -to the third party based on the extent of your activity of conveying -the work, and under which the third party grants, to any of the -parties who would receive the covered work from you, a discriminatory -patent license (a) in connection with copies of the covered work -conveyed by you (or copies made from those copies), or (b) primarily -for and in connection with specific products or compilations that -contain the covered work, unless you entered into that arrangement, -or that patent license was granted, prior to 28 March 2007. - - Nothing in this License shall be construed as excluding or limiting -any implied license or other defenses to infringement that may -otherwise be available to you under applicable patent law. - - 12. No Surrender of Others' Freedom. - - If conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot convey a -covered work so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you may -not convey it at all. For example, if you agree to terms that obligate you -to collect a royalty for further conveying from those to whom you convey -the Program, the only way you could satisfy both those terms and this -License would be to refrain entirely from conveying the Program. - - 13. Remote Network Interaction; Use with the GNU General Public License. - - Notwithstanding any other provision of this License, if you modify the -Program, your modified version must prominently offer all users -interacting with it remotely through a computer network (if your version -supports such interaction) an opportunity to receive the Corresponding -Source of your version by providing access to the Corresponding Source -from a network server at no charge, through some standard or customary -means of facilitating copying of software. This Corresponding Source -shall include the Corresponding Source for any work covered by version 3 -of the GNU General Public License that is incorporated pursuant to the -following paragraph. - - Notwithstanding any other provision of this License, you have -permission to link or combine any covered work with a work licensed -under version 3 of the GNU General Public License into a single -combined work, and to convey the resulting work. The terms of this -License will continue to apply to the part which is the covered work, -but the work with which it is combined will remain governed by version -3 of the GNU General Public License. - - 14. Revised Versions of this License. - - The Free Software Foundation may publish revised and/or new versions of -the GNU Affero General Public License from time to time. Such new versions -will be similar in spirit to the present version, but may differ in detail to -address new problems or concerns. - - Each version is given a distinguishing version number. If the -Program specifies that a certain numbered version of the GNU Affero General -Public License "or any later version" applies to it, you have the -option of following the terms and conditions either of that numbered -version or of any later version published by the Free Software -Foundation. If the Program does not specify a version number of the -GNU Affero General Public License, you may choose any version ever published -by the Free Software Foundation. - - If the Program specifies that a proxy can decide which future -versions of the GNU Affero General Public License can be used, that proxy's -public statement of acceptance of a version permanently authorizes you -to choose that version for the Program. - - Later license versions may give you additional or different -permissions. However, no additional obligations are imposed on any -author or copyright holder as a result of your choosing to follow a -later version. - - 15. Disclaimer of Warranty. - - THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY -APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT -HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY -OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, -THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM -IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF -ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. Limitation of Liability. - - IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING -WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS -THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY -GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE -USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF -DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD -PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), -EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF -SUCH DAMAGES. - - 17. Interpretation of Sections 15 and 16. - - If the disclaimer of warranty and limitation of liability provided -above cannot be given local legal effect according to their terms, -reviewing courts shall apply local law that most closely approximates -an absolute waiver of all civil liability in connection with the -Program, unless a warranty or assumption of liability accompanies a -copy of the Program in return for a fee. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Programs - - If you develop a new program, and you want it to be of the greatest -possible use to the public, the best way to achieve this is to make it -free software which everyone can redistribute and change under these terms. - - To do so, attach the following notices to the program. It is safest -to attach them to the start of each source file to most effectively -state the exclusion of warranty; and each file should have at least -the "copyright" line and a pointer to where the full notice is found. - - - Copyright (C) - - This program is free software: you can redistribute it and/or modify - it under the terms of the GNU Affero General Public License as published - by the Free Software Foundation, either version 3 of the License, or - (at your option) any later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU Affero General Public License for more details. - - You should have received a copy of the GNU Affero General Public License - along with this program. If not, see . - -Also add information on how to contact you by electronic and paper mail. - - If your software can interact with users remotely through a computer -network, you should also make sure that it provides a way for users to -get its source. For example, if your program is a web application, its -interface could display a "Source" link that leads users to an archive -of the code. There are many ways you could offer source, and different -solutions will be better for different programs; see section 13 for the -specific requirements. - - You should also get your employer (if you work as a programmer) or school, -if any, to sign a "copyright disclaimer" for the program, if necessary. -For more information on this, and how to apply and follow the GNU AGPL, see -. \ No newline at end of file diff --git a/lib/solmate/README.md b/lib/solmate/README.md deleted file mode 100644 index 87fda96..0000000 --- a/lib/solmate/README.md +++ /dev/null @@ -1,70 +0,0 @@ -# solmate - -**Modern**, **opinionated**, and **gas optimized** building blocks for **smart contract development**. - -## Contracts - -```ml -auth -├─ Owned — "Simple single owner authorization" -├─ Auth — "Flexible and updatable auth pattern" -├─ authorities -│ ├─ RolesAuthority — "Role based Authority that supports up to 256 roles" -│ ├─ MultiRolesAuthority — "Flexible and target agnostic role based Authority" -mixins -├─ ERC4626 — "Minimal ERC4626 tokenized Vault implementation" -tokens -├─ WETH — "Minimalist and modern Wrapped Ether implementation" -├─ ERC20 — "Modern and gas efficient ERC20 + EIP-2612 implementation" -├─ ERC721 — "Modern, minimalist, and gas efficient ERC721 implementation" -├─ ERC1155 — "Minimalist and gas efficient standard ERC1155 implementation" -├─ ERC6909 — "Minimalist and gas efficient standard ERC6909 implementation" -utils -├─ SSTORE2 — "Library for cheaper reads and writes to persistent storage" -├─ CREATE3 — "Deploy to deterministic addresses without an initcode factor" -├─ LibString — "Library for creating string representations of uint values" -├─ SafeCastLib — "Safe unsigned integer casting lib that reverts on overflow" -├─ SignedWadMath — "Signed integer 18 decimal fixed point arithmetic library" -├─ MerkleProofLib — "Efficient merkle tree inclusion proof verification library" -├─ ReentrancyGuard — "Gas optimized reentrancy protection for smart contracts" -├─ FixedPointMathLib — "Arithmetic library with operations for fixed-point numbers" -├─ Bytes32AddressLib — "Library for converting between addresses and bytes32 values" -├─ SafeTransferLib — "Safe ERC20/ETH transfer lib that handles missing return values" -``` - -## Safety - -This is **experimental software** and is provided on an "as is" and "as available" basis. - -While each [major release has been audited](audits), these contracts are **not designed with user safety** in mind: - -- There are implicit invariants these contracts expect to hold. -- **You can easily shoot yourself in the foot if you're not careful.** -- You should thoroughly read each contract you plan to use top to bottom. - -We **do not give any warranties** and **will not be liable for any loss** incurred through any use of this codebase. - -## Installation - -To install with [**Foundry**](https://github.com/gakonst/foundry): - -```sh -forge install transmissions11/solmate -``` - -To install with [**Hardhat**](https://github.com/nomiclabs/hardhat) or [**Truffle**](https://github.com/trufflesuite/truffle): - -```sh -npm install solmate -``` - -## Acknowledgements - -These contracts were inspired by or directly modified from many sources, primarily: - -- [Gnosis](https://github.com/gnosis/gp-v2-contracts) -- [Uniswap](https://github.com/Uniswap/uniswap-lib) -- [Dappsys](https://github.com/dapphub/dappsys) -- [Dappsys V2](https://github.com/dapp-org/dappsys-v2) -- [0xSequence](https://github.com/0xSequence) -- [OpenZeppelin](https://github.com/OpenZeppelin/openzeppelin-contracts) diff --git a/lib/solmate/audits/v6-Fixed-Point-Solutions.pdf b/lib/solmate/audits/v6-Fixed-Point-Solutions.pdf deleted file mode 100644 index 5c42434..0000000 Binary files a/lib/solmate/audits/v6-Fixed-Point-Solutions.pdf and /dev/null differ diff --git a/lib/solmate/foundry.toml b/lib/solmate/foundry.toml deleted file mode 100644 index ebdfa11..0000000 --- a/lib/solmate/foundry.toml +++ /dev/null @@ -1,7 +0,0 @@ -[profile.default] -solc = "0.8.15" -bytecode_hash = "none" -optimizer_runs = 1000000 - -[profile.intense.fuzz] -runs = 10000 diff --git a/lib/solmate/lib/ds-test/.gitignore b/lib/solmate/lib/ds-test/.gitignore deleted file mode 100644 index 63f0b2c..0000000 --- a/lib/solmate/lib/ds-test/.gitignore +++ /dev/null @@ -1,3 +0,0 @@ -/.dapple -/build -/out diff --git a/lib/solmate/lib/ds-test/LICENSE b/lib/solmate/lib/ds-test/LICENSE deleted file mode 100644 index 94a9ed0..0000000 --- a/lib/solmate/lib/ds-test/LICENSE +++ /dev/null @@ -1,674 +0,0 @@ - GNU GENERAL PUBLIC LICENSE - Version 3, 29 June 2007 - - Copyright (C) 2007 Free Software Foundation, Inc. - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - - Preamble - - The GNU General Public License is a free, copyleft license for -software and other kinds of works. - - The licenses for most software and other practical works are designed -to take away your freedom to share and change the works. By contrast, -the GNU General Public License is intended to guarantee your freedom to -share and change all versions of a program--to make sure it remains free -software for all its users. We, the Free Software Foundation, use the -GNU General Public License for most of our software; it applies also to -any other work released this way by its authors. You can apply it to -your programs, too. - - When we speak of free software, we are referring to freedom, not -price. Our General Public Licenses are designed to make sure that you -have the freedom to distribute copies of free software (and charge for -them if you wish), that you receive source code or can get it if you -want it, that you can change the software or use pieces of it in new -free programs, and that you know you can do these things. - - To protect your rights, we need to prevent others from denying you -these rights or asking you to surrender the rights. Therefore, you have -certain responsibilities if you distribute copies of the software, or if -you modify it: responsibilities to respect the freedom of others. - - For example, if you distribute copies of such a program, whether -gratis or for a fee, you must pass on to the recipients the same -freedoms that you received. You must make sure that they, too, receive -or can get the source code. And you must show them these terms so they -know their rights. - - Developers that use the GNU GPL protect your rights with two steps: -(1) assert copyright on the software, and (2) offer you this License -giving you legal permission to copy, distribute and/or modify it. - - For the developers' and authors' protection, the GPL clearly explains -that there is no warranty for this free software. For both users' and -authors' sake, the GPL requires that modified versions be marked as -changed, so that their problems will not be attributed erroneously to -authors of previous versions. - - Some devices are designed to deny users access to install or run -modified versions of the software inside them, although the manufacturer -can do so. This is fundamentally incompatible with the aim of -protecting users' freedom to change the software. The systematic -pattern of such abuse occurs in the area of products for individuals to -use, which is precisely where it is most unacceptable. Therefore, we -have designed this version of the GPL to prohibit the practice for those -products. If such problems arise substantially in other domains, we -stand ready to extend this provision to those domains in future versions -of the GPL, as needed to protect the freedom of users. - - Finally, every program is threatened constantly by software patents. -States should not allow patents to restrict development and use of -software on general-purpose computers, but in those that do, we wish to -avoid the special danger that patents applied to a free program could -make it effectively proprietary. To prevent this, the GPL assures that -patents cannot be used to render the program non-free. - - The precise terms and conditions for copying, distribution and -modification follow. - - TERMS AND CONDITIONS - - 0. Definitions. - - "This License" refers to version 3 of the GNU General Public License. - - "Copyright" also means copyright-like laws that apply to other kinds of -works, such as semiconductor masks. - - "The Program" refers to any copyrightable work licensed under this -License. Each licensee is addressed as "you". "Licensees" and -"recipients" may be individuals or organizations. - - To "modify" a work means to copy from or adapt all or part of the work -in a fashion requiring copyright permission, other than the making of an -exact copy. The resulting work is called a "modified version" of the -earlier work or a work "based on" the earlier work. - - A "covered work" means either the unmodified Program or a work based -on the Program. - - To "propagate" a work means to do anything with it that, without -permission, would make you directly or secondarily liable for -infringement under applicable copyright law, except executing it on a -computer or modifying a private copy. Propagation includes copying, -distribution (with or without modification), making available to the -public, and in some countries other activities as well. - - To "convey" a work means any kind of propagation that enables other -parties to make or receive copies. Mere interaction with a user through -a computer network, with no transfer of a copy, is not conveying. - - An interactive user interface displays "Appropriate Legal Notices" -to the extent that it includes a convenient and prominently visible -feature that (1) displays an appropriate copyright notice, and (2) -tells the user that there is no warranty for the work (except to the -extent that warranties are provided), that licensees may convey the -work under this License, and how to view a copy of this License. If -the interface presents a list of user commands or options, such as a -menu, a prominent item in the list meets this criterion. - - 1. Source Code. - - The "source code" for a work means the preferred form of the work -for making modifications to it. "Object code" means any non-source -form of a work. - - A "Standard Interface" means an interface that either is an official -standard defined by a recognized standards body, or, in the case of -interfaces specified for a particular programming language, one that -is widely used among developers working in that language. - - The "System Libraries" of an executable work include anything, other -than the work as a whole, that (a) is included in the normal form of -packaging a Major Component, but which is not part of that Major -Component, and (b) serves only to enable use of the work with that -Major Component, or to implement a Standard Interface for which an -implementation is available to the public in source code form. A -"Major Component", in this context, means a major essential component -(kernel, window system, and so on) of the specific operating system -(if any) on which the executable work runs, or a compiler used to -produce the work, or an object code interpreter used to run it. - - The "Corresponding Source" for a work in object code form means all -the source code needed to generate, install, and (for an executable -work) run the object code and to modify the work, including scripts to -control those activities. However, it does not include the work's -System Libraries, or general-purpose tools or generally available free -programs which are used unmodified in performing those activities but -which are not part of the work. For example, Corresponding Source -includes interface definition files associated with source files for -the work, and the source code for shared libraries and dynamically -linked subprograms that the work is specifically designed to require, -such as by intimate data communication or control flow between those -subprograms and other parts of the work. - - The Corresponding Source need not include anything that users -can regenerate automatically from other parts of the Corresponding -Source. - - The Corresponding Source for a work in source code form is that -same work. - - 2. Basic Permissions. - - All rights granted under this License are granted for the term of -copyright on the Program, and are irrevocable provided the stated -conditions are met. This License explicitly affirms your unlimited -permission to run the unmodified Program. The output from running a -covered work is covered by this License only if the output, given its -content, constitutes a covered work. This License acknowledges your -rights of fair use or other equivalent, as provided by copyright law. - - You may make, run and propagate covered works that you do not -convey, without conditions so long as your license otherwise remains -in force. You may convey covered works to others for the sole purpose -of having them make modifications exclusively for you, or provide you -with facilities for running those works, provided that you comply with -the terms of this License in conveying all material for which you do -not control copyright. Those thus making or running the covered works -for you must do so exclusively on your behalf, under your direction -and control, on terms that prohibit them from making any copies of -your copyrighted material outside their relationship with you. - - Conveying under any other circumstances is permitted solely under -the conditions stated below. Sublicensing is not allowed; section 10 -makes it unnecessary. - - 3. Protecting Users' Legal Rights From Anti-Circumvention Law. - - No covered work shall be deemed part of an effective technological -measure under any applicable law fulfilling obligations under article -11 of the WIPO copyright treaty adopted on 20 December 1996, or -similar laws prohibiting or restricting circumvention of such -measures. - - When you convey a covered work, you waive any legal power to forbid -circumvention of technological measures to the extent such circumvention -is effected by exercising rights under this License with respect to -the covered work, and you disclaim any intention to limit operation or -modification of the work as a means of enforcing, against the work's -users, your or third parties' legal rights to forbid circumvention of -technological measures. - - 4. Conveying Verbatim Copies. - - You may convey verbatim copies of the Program's source code as you -receive it, in any medium, provided that you conspicuously and -appropriately publish on each copy an appropriate copyright notice; -keep intact all notices stating that this License and any -non-permissive terms added in accord with section 7 apply to the code; -keep intact all notices of the absence of any warranty; and give all -recipients a copy of this License along with the Program. - - You may charge any price or no price for each copy that you convey, -and you may offer support or warranty protection for a fee. - - 5. Conveying Modified Source Versions. - - You may convey a work based on the Program, or the modifications to -produce it from the Program, in the form of source code under the -terms of section 4, provided that you also meet all of these conditions: - - a) The work must carry prominent notices stating that you modified - it, and giving a relevant date. - - b) The work must carry prominent notices stating that it is - released under this License and any conditions added under section - 7. This requirement modifies the requirement in section 4 to - "keep intact all notices". - - c) You must license the entire work, as a whole, under this - License to anyone who comes into possession of a copy. This - License will therefore apply, along with any applicable section 7 - additional terms, to the whole of the work, and all its parts, - regardless of how they are packaged. This License gives no - permission to license the work in any other way, but it does not - invalidate such permission if you have separately received it. - - d) If the work has interactive user interfaces, each must display - Appropriate Legal Notices; however, if the Program has interactive - interfaces that do not display Appropriate Legal Notices, your - work need not make them do so. - - A compilation of a covered work with other separate and independent -works, which are not by their nature extensions of the covered work, -and which are not combined with it such as to form a larger program, -in or on a volume of a storage or distribution medium, is called an -"aggregate" if the compilation and its resulting copyright are not -used to limit the access or legal rights of the compilation's users -beyond what the individual works permit. Inclusion of a covered work -in an aggregate does not cause this License to apply to the other -parts of the aggregate. - - 6. Conveying Non-Source Forms. - - You may convey a covered work in object code form under the terms -of sections 4 and 5, provided that you also convey the -machine-readable Corresponding Source under the terms of this License, -in one of these ways: - - a) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by the - Corresponding Source fixed on a durable physical medium - customarily used for software interchange. - - b) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by a - written offer, valid for at least three years and valid for as - long as you offer spare parts or customer support for that product - model, to give anyone who possesses the object code either (1) a - copy of the Corresponding Source for all the software in the - product that is covered by this License, on a durable physical - medium customarily used for software interchange, for a price no - more than your reasonable cost of physically performing this - conveying of source, or (2) access to copy the - Corresponding Source from a network server at no charge. - - c) Convey individual copies of the object code with a copy of the - written offer to provide the Corresponding Source. This - alternative is allowed only occasionally and noncommercially, and - only if you received the object code with such an offer, in accord - with subsection 6b. - - d) Convey the object code by offering access from a designated - place (gratis or for a charge), and offer equivalent access to the - Corresponding Source in the same way through the same place at no - further charge. You need not require recipients to copy the - Corresponding Source along with the object code. If the place to - copy the object code is a network server, the Corresponding Source - may be on a different server (operated by you or a third party) - that supports equivalent copying facilities, provided you maintain - clear directions next to the object code saying where to find the - Corresponding Source. Regardless of what server hosts the - Corresponding Source, you remain obligated to ensure that it is - available for as long as needed to satisfy these requirements. - - e) Convey the object code using peer-to-peer transmission, provided - you inform other peers where the object code and Corresponding - Source of the work are being offered to the general public at no - charge under subsection 6d. - - A separable portion of the object code, whose source code is excluded -from the Corresponding Source as a System Library, need not be -included in conveying the object code work. - - A "User Product" is either (1) a "consumer product", which means any -tangible personal property which is normally used for personal, family, -or household purposes, or (2) anything designed or sold for incorporation -into a dwelling. In determining whether a product is a consumer product, -doubtful cases shall be resolved in favor of coverage. For a particular -product received by a particular user, "normally used" refers to a -typical or common use of that class of product, regardless of the status -of the particular user or of the way in which the particular user -actually uses, or expects or is expected to use, the product. A product -is a consumer product regardless of whether the product has substantial -commercial, industrial or non-consumer uses, unless such uses represent -the only significant mode of use of the product. - - "Installation Information" for a User Product means any methods, -procedures, authorization keys, or other information required to install -and execute modified versions of a covered work in that User Product from -a modified version of its Corresponding Source. The information must -suffice to ensure that the continued functioning of the modified object -code is in no case prevented or interfered with solely because -modification has been made. - - If you convey an object code work under this section in, or with, or -specifically for use in, a User Product, and the conveying occurs as -part of a transaction in which the right of possession and use of the -User Product is transferred to the recipient in perpetuity or for a -fixed term (regardless of how the transaction is characterized), the -Corresponding Source conveyed under this section must be accompanied -by the Installation Information. But this requirement does not apply -if neither you nor any third party retains the ability to install -modified object code on the User Product (for example, the work has -been installed in ROM). - - The requirement to provide Installation Information does not include a -requirement to continue to provide support service, warranty, or updates -for a work that has been modified or installed by the recipient, or for -the User Product in which it has been modified or installed. Access to a -network may be denied when the modification itself materially and -adversely affects the operation of the network or violates the rules and -protocols for communication across the network. - - Corresponding Source conveyed, and Installation Information provided, -in accord with this section must be in a format that is publicly -documented (and with an implementation available to the public in -source code form), and must require no special password or key for -unpacking, reading or copying. - - 7. Additional Terms. - - "Additional permissions" are terms that supplement the terms of this -License by making exceptions from one or more of its conditions. -Additional permissions that are applicable to the entire Program shall -be treated as though they were included in this License, to the extent -that they are valid under applicable law. If additional permissions -apply only to part of the Program, that part may be used separately -under those permissions, but the entire Program remains governed by -this License without regard to the additional permissions. - - When you convey a copy of a covered work, you may at your option -remove any additional permissions from that copy, or from any part of -it. (Additional permissions may be written to require their own -removal in certain cases when you modify the work.) You may place -additional permissions on material, added by you to a covered work, -for which you have or can give appropriate copyright permission. - - Notwithstanding any other provision of this License, for material you -add to a covered work, you may (if authorized by the copyright holders of -that material) supplement the terms of this License with terms: - - a) Disclaiming warranty or limiting liability differently from the - terms of sections 15 and 16 of this License; or - - b) Requiring preservation of specified reasonable legal notices or - author attributions in that material or in the Appropriate Legal - Notices displayed by works containing it; or - - c) Prohibiting misrepresentation of the origin of that material, or - requiring that modified versions of such material be marked in - reasonable ways as different from the original version; or - - d) Limiting the use for publicity purposes of names of licensors or - authors of the material; or - - e) Declining to grant rights under trademark law for use of some - trade names, trademarks, or service marks; or - - f) Requiring indemnification of licensors and authors of that - material by anyone who conveys the material (or modified versions of - it) with contractual assumptions of liability to the recipient, for - any liability that these contractual assumptions directly impose on - those licensors and authors. - - All other non-permissive additional terms are considered "further -restrictions" within the meaning of section 10. If the Program as you -received it, or any part of it, contains a notice stating that it is -governed by this License along with a term that is a further -restriction, you may remove that term. If a license document contains -a further restriction but permits relicensing or conveying under this -License, you may add to a covered work material governed by the terms -of that license document, provided that the further restriction does -not survive such relicensing or conveying. - - If you add terms to a covered work in accord with this section, you -must place, in the relevant source files, a statement of the -additional terms that apply to those files, or a notice indicating -where to find the applicable terms. - - Additional terms, permissive or non-permissive, may be stated in the -form of a separately written license, or stated as exceptions; -the above requirements apply either way. - - 8. Termination. - - You may not propagate or modify a covered work except as expressly -provided under this License. Any attempt otherwise to propagate or -modify it is void, and will automatically terminate your rights under -this License (including any patent licenses granted under the third -paragraph of section 11). - - However, if you cease all violation of this License, then your -license from a particular copyright holder is reinstated (a) -provisionally, unless and until the copyright holder explicitly and -finally terminates your license, and (b) permanently, if the copyright -holder fails to notify you of the violation by some reasonable means -prior to 60 days after the cessation. - - Moreover, your license from a particular copyright holder is -reinstated permanently if the copyright holder notifies you of the -violation by some reasonable means, this is the first time you have -received notice of violation of this License (for any work) from that -copyright holder, and you cure the violation prior to 30 days after -your receipt of the notice. - - Termination of your rights under this section does not terminate the -licenses of parties who have received copies or rights from you under -this License. If your rights have been terminated and not permanently -reinstated, you do not qualify to receive new licenses for the same -material under section 10. - - 9. Acceptance Not Required for Having Copies. - - You are not required to accept this License in order to receive or -run a copy of the Program. Ancillary propagation of a covered work -occurring solely as a consequence of using peer-to-peer transmission -to receive a copy likewise does not require acceptance. However, -nothing other than this License grants you permission to propagate or -modify any covered work. These actions infringe copyright if you do -not accept this License. Therefore, by modifying or propagating a -covered work, you indicate your acceptance of this License to do so. - - 10. Automatic Licensing of Downstream Recipients. - - Each time you convey a covered work, the recipient automatically -receives a license from the original licensors, to run, modify and -propagate that work, subject to this License. You are not responsible -for enforcing compliance by third parties with this License. - - An "entity transaction" is a transaction transferring control of an -organization, or substantially all assets of one, or subdividing an -organization, or merging organizations. If propagation of a covered -work results from an entity transaction, each party to that -transaction who receives a copy of the work also receives whatever -licenses to the work the party's predecessor in interest had or could -give under the previous paragraph, plus a right to possession of the -Corresponding Source of the work from the predecessor in interest, if -the predecessor has it or can get it with reasonable efforts. - - You may not impose any further restrictions on the exercise of the -rights granted or affirmed under this License. For example, you may -not impose a license fee, royalty, or other charge for exercise of -rights granted under this License, and you may not initiate litigation -(including a cross-claim or counterclaim in a lawsuit) alleging that -any patent claim is infringed by making, using, selling, offering for -sale, or importing the Program or any portion of it. - - 11. Patents. - - A "contributor" is a copyright holder who authorizes use under this -License of the Program or a work on which the Program is based. The -work thus licensed is called the contributor's "contributor version". - - A contributor's "essential patent claims" are all patent claims -owned or controlled by the contributor, whether already acquired or -hereafter acquired, that would be infringed by some manner, permitted -by this License, of making, using, or selling its contributor version, -but do not include claims that would be infringed only as a -consequence of further modification of the contributor version. For -purposes of this definition, "control" includes the right to grant -patent sublicenses in a manner consistent with the requirements of -this License. - - Each contributor grants you a non-exclusive, worldwide, royalty-free -patent license under the contributor's essential patent claims, to -make, use, sell, offer for sale, import and otherwise run, modify and -propagate the contents of its contributor version. - - In the following three paragraphs, a "patent license" is any express -agreement or commitment, however denominated, not to enforce a patent -(such as an express permission to practice a patent or covenant not to -sue for patent infringement). To "grant" such a patent license to a -party means to make such an agreement or commitment not to enforce a -patent against the party. - - If you convey a covered work, knowingly relying on a patent license, -and the Corresponding Source of the work is not available for anyone -to copy, free of charge and under the terms of this License, through a -publicly available network server or other readily accessible means, -then you must either (1) cause the Corresponding Source to be so -available, or (2) arrange to deprive yourself of the benefit of the -patent license for this particular work, or (3) arrange, in a manner -consistent with the requirements of this License, to extend the patent -license to downstream recipients. "Knowingly relying" means you have -actual knowledge that, but for the patent license, your conveying the -covered work in a country, or your recipient's use of the covered work -in a country, would infringe one or more identifiable patents in that -country that you have reason to believe are valid. - - If, pursuant to or in connection with a single transaction or -arrangement, you convey, or propagate by procuring conveyance of, a -covered work, and grant a patent license to some of the parties -receiving the covered work authorizing them to use, propagate, modify -or convey a specific copy of the covered work, then the patent license -you grant is automatically extended to all recipients of the covered -work and works based on it. - - A patent license is "discriminatory" if it does not include within -the scope of its coverage, prohibits the exercise of, or is -conditioned on the non-exercise of one or more of the rights that are -specifically granted under this License. You may not convey a covered -work if you are a party to an arrangement with a third party that is -in the business of distributing software, under which you make payment -to the third party based on the extent of your activity of conveying -the work, and under which the third party grants, to any of the -parties who would receive the covered work from you, a discriminatory -patent license (a) in connection with copies of the covered work -conveyed by you (or copies made from those copies), or (b) primarily -for and in connection with specific products or compilations that -contain the covered work, unless you entered into that arrangement, -or that patent license was granted, prior to 28 March 2007. - - Nothing in this License shall be construed as excluding or limiting -any implied license or other defenses to infringement that may -otherwise be available to you under applicable patent law. - - 12. No Surrender of Others' Freedom. - - If conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot convey a -covered work so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you may -not convey it at all. For example, if you agree to terms that obligate you -to collect a royalty for further conveying from those to whom you convey -the Program, the only way you could satisfy both those terms and this -License would be to refrain entirely from conveying the Program. - - 13. Use with the GNU Affero General Public License. - - Notwithstanding any other provision of this License, you have -permission to link or combine any covered work with a work licensed -under version 3 of the GNU Affero General Public License into a single -combined work, and to convey the resulting work. The terms of this -License will continue to apply to the part which is the covered work, -but the special requirements of the GNU Affero General Public License, -section 13, concerning interaction through a network will apply to the -combination as such. - - 14. Revised Versions of this License. - - The Free Software Foundation may publish revised and/or new versions of -the GNU General Public License from time to time. Such new versions will -be similar in spirit to the present version, but may differ in detail to -address new problems or concerns. - - Each version is given a distinguishing version number. If the -Program specifies that a certain numbered version of the GNU General -Public License "or any later version" applies to it, you have the -option of following the terms and conditions either of that numbered -version or of any later version published by the Free Software -Foundation. If the Program does not specify a version number of the -GNU General Public License, you may choose any version ever published -by the Free Software Foundation. - - If the Program specifies that a proxy can decide which future -versions of the GNU General Public License can be used, that proxy's -public statement of acceptance of a version permanently authorizes you -to choose that version for the Program. - - Later license versions may give you additional or different -permissions. However, no additional obligations are imposed on any -author or copyright holder as a result of your choosing to follow a -later version. - - 15. Disclaimer of Warranty. - - THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY -APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT -HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY -OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, -THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM -IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF -ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. Limitation of Liability. - - IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING -WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS -THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY -GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE -USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF -DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD -PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), -EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF -SUCH DAMAGES. - - 17. Interpretation of Sections 15 and 16. - - If the disclaimer of warranty and limitation of liability provided -above cannot be given local legal effect according to their terms, -reviewing courts shall apply local law that most closely approximates -an absolute waiver of all civil liability in connection with the -Program, unless a warranty or assumption of liability accompanies a -copy of the Program in return for a fee. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Programs - - If you develop a new program, and you want it to be of the greatest -possible use to the public, the best way to achieve this is to make it -free software which everyone can redistribute and change under these terms. - - To do so, attach the following notices to the program. It is safest -to attach them to the start of each source file to most effectively -state the exclusion of warranty; and each file should have at least -the "copyright" line and a pointer to where the full notice is found. - - - Copyright (C) - - This program is free software: you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation, either version 3 of the License, or - (at your option) any later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with this program. If not, see . - -Also add information on how to contact you by electronic and paper mail. - - If the program does terminal interaction, make it output a short -notice like this when it starts in an interactive mode: - - Copyright (C) - This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. - This is free software, and you are welcome to redistribute it - under certain conditions; type `show c' for details. - -The hypothetical commands `show w' and `show c' should show the appropriate -parts of the General Public License. Of course, your program's commands -might be different; for a GUI interface, you would use an "about box". - - You should also get your employer (if you work as a programmer) or school, -if any, to sign a "copyright disclaimer" for the program, if necessary. -For more information on this, and how to apply and follow the GNU GPL, see -. - - The GNU General Public License does not permit incorporating your program -into proprietary programs. If your program is a subroutine library, you -may consider it more useful to permit linking proprietary applications with -the library. If this is what you want to do, use the GNU Lesser General -Public License instead of this License. But first, please read -. diff --git a/lib/solmate/lib/ds-test/Makefile b/lib/solmate/lib/ds-test/Makefile deleted file mode 100644 index 661dac4..0000000 --- a/lib/solmate/lib/ds-test/Makefile +++ /dev/null @@ -1,14 +0,0 @@ -all:; dapp build - -test: - -dapp --use solc:0.4.23 build - -dapp --use solc:0.4.26 build - -dapp --use solc:0.5.17 build - -dapp --use solc:0.6.12 build - -dapp --use solc:0.7.5 build - -demo: - DAPP_SRC=demo dapp --use solc:0.7.5 build - -hevm dapp-test --verbose 3 - -.PHONY: test demo diff --git a/lib/solmate/lib/ds-test/default.nix b/lib/solmate/lib/ds-test/default.nix deleted file mode 100644 index cf65419..0000000 --- a/lib/solmate/lib/ds-test/default.nix +++ /dev/null @@ -1,4 +0,0 @@ -{ solidityPackage, dappsys }: solidityPackage { - name = "ds-test"; - src = ./src; -} diff --git a/lib/solmate/lib/ds-test/demo/demo.sol b/lib/solmate/lib/ds-test/demo/demo.sol deleted file mode 100644 index f3bb48e..0000000 --- a/lib/solmate/lib/ds-test/demo/demo.sol +++ /dev/null @@ -1,222 +0,0 @@ -// SPDX-License-Identifier: GPL-3.0-or-later -pragma solidity >=0.5.0; - -import "../src/test.sol"; - -contract DemoTest is DSTest { - function test_this() public pure { - require(true); - } - function test_logs() public { - emit log("-- log(string)"); - emit log("a string"); - - emit log("-- log_named_uint(string, uint)"); - emit log_named_uint("uint", 512); - - emit log("-- log_named_int(string, int)"); - emit log_named_int("int", -512); - - emit log("-- log_named_address(string, address)"); - emit log_named_address("address", address(this)); - - emit log("-- log_named_bytes32(string, bytes32)"); - emit log_named_bytes32("bytes32", "a string"); - - emit log("-- log_named_bytes(string, bytes)"); - emit log_named_bytes("bytes", hex"cafefe"); - - emit log("-- log_named_string(string, string)"); - emit log_named_string("string", "a string"); - - emit log("-- log_named_decimal_uint(string, uint, uint)"); - emit log_named_decimal_uint("decimal uint", 1.0e18, 18); - - emit log("-- log_named_decimal_int(string, int, uint)"); - emit log_named_decimal_int("decimal int", -1.0e18, 18); - } - event log_old_named_uint(bytes32,uint); - function test_old_logs() public { - emit log_old_named_uint("key", 500); - emit log_named_bytes32("bkey", "val"); - } - function test_trace() public view { - this.echo("string 1", "string 2"); - } - function test_multiline() public { - emit log("a multiline\\nstring"); - emit log("a multiline string"); - emit log_bytes("a string"); - emit log_bytes("a multiline\nstring"); - emit log_bytes("a multiline\\nstring"); - emit logs(hex"0000"); - emit log_named_bytes("0x0000", hex"0000"); - emit logs(hex"ff"); - } - function echo(string memory s1, string memory s2) public pure - returns (string memory, string memory) - { - return (s1, s2); - } - - function prove_this(uint x) public { - emit log_named_uint("sym x", x); - assertGt(x + 1, 0); - } - - function test_logn() public { - assembly { - log0(0x01, 0x02) - log1(0x01, 0x02, 0x03) - log2(0x01, 0x02, 0x03, 0x04) - log3(0x01, 0x02, 0x03, 0x04, 0x05) - } - } - - event MyEvent(uint, uint indexed, uint, uint indexed); - function test_events() public { - emit MyEvent(1, 2, 3, 4); - } - - function test_asserts() public { - string memory err = "this test has failed!"; - emit log("## assertTrue(bool)\n"); - assertTrue(false); - emit log("\n"); - assertTrue(false, err); - - emit log("\n## assertEq(address,address)\n"); - assertEq(address(this), msg.sender); - emit log("\n"); - assertEq(address(this), msg.sender, err); - - emit log("\n## assertEq32(bytes32,bytes32)\n"); - assertEq32("bytes 1", "bytes 2"); - emit log("\n"); - assertEq32("bytes 1", "bytes 2", err); - - emit log("\n## assertEq(bytes32,bytes32)\n"); - assertEq32("bytes 1", "bytes 2"); - emit log("\n"); - assertEq32("bytes 1", "bytes 2", err); - - emit log("\n## assertEq(uint,uint)\n"); - assertEq(uint(0), 1); - emit log("\n"); - assertEq(uint(0), 1, err); - - emit log("\n## assertEq(int,int)\n"); - assertEq(-1, -2); - emit log("\n"); - assertEq(-1, -2, err); - - emit log("\n## assertEqDecimal(int,int,uint)\n"); - assertEqDecimal(-1.0e18, -1.1e18, 18); - emit log("\n"); - assertEqDecimal(-1.0e18, -1.1e18, 18, err); - - emit log("\n## assertEqDecimal(uint,uint,uint)\n"); - assertEqDecimal(uint(1.0e18), 1.1e18, 18); - emit log("\n"); - assertEqDecimal(uint(1.0e18), 1.1e18, 18, err); - - emit log("\n## assertGt(uint,uint)\n"); - assertGt(uint(0), 0); - emit log("\n"); - assertGt(uint(0), 0, err); - - emit log("\n## assertGt(int,int)\n"); - assertGt(-1, -1); - emit log("\n"); - assertGt(-1, -1, err); - - emit log("\n## assertGtDecimal(int,int,uint)\n"); - assertGtDecimal(-2.0e18, -1.1e18, 18); - emit log("\n"); - assertGtDecimal(-2.0e18, -1.1e18, 18, err); - - emit log("\n## assertGtDecimal(uint,uint,uint)\n"); - assertGtDecimal(uint(1.0e18), 1.1e18, 18); - emit log("\n"); - assertGtDecimal(uint(1.0e18), 1.1e18, 18, err); - - emit log("\n## assertGe(uint,uint)\n"); - assertGe(uint(0), 1); - emit log("\n"); - assertGe(uint(0), 1, err); - - emit log("\n## assertGe(int,int)\n"); - assertGe(-1, 0); - emit log("\n"); - assertGe(-1, 0, err); - - emit log("\n## assertGeDecimal(int,int,uint)\n"); - assertGeDecimal(-2.0e18, -1.1e18, 18); - emit log("\n"); - assertGeDecimal(-2.0e18, -1.1e18, 18, err); - - emit log("\n## assertGeDecimal(uint,uint,uint)\n"); - assertGeDecimal(uint(1.0e18), 1.1e18, 18); - emit log("\n"); - assertGeDecimal(uint(1.0e18), 1.1e18, 18, err); - - emit log("\n## assertLt(uint,uint)\n"); - assertLt(uint(0), 0); - emit log("\n"); - assertLt(uint(0), 0, err); - - emit log("\n## assertLt(int,int)\n"); - assertLt(-1, -1); - emit log("\n"); - assertLt(-1, -1, err); - - emit log("\n## assertLtDecimal(int,int,uint)\n"); - assertLtDecimal(-1.0e18, -1.1e18, 18); - emit log("\n"); - assertLtDecimal(-1.0e18, -1.1e18, 18, err); - - emit log("\n## assertLtDecimal(uint,uint,uint)\n"); - assertLtDecimal(uint(2.0e18), 1.1e18, 18); - emit log("\n"); - assertLtDecimal(uint(2.0e18), 1.1e18, 18, err); - - emit log("\n## assertLe(uint,uint)\n"); - assertLe(uint(1), 0); - emit log("\n"); - assertLe(uint(1), 0, err); - - emit log("\n## assertLe(int,int)\n"); - assertLe(0, -1); - emit log("\n"); - assertLe(0, -1, err); - - emit log("\n## assertLeDecimal(int,int,uint)\n"); - assertLeDecimal(-1.0e18, -1.1e18, 18); - emit log("\n"); - assertLeDecimal(-1.0e18, -1.1e18, 18, err); - - emit log("\n## assertLeDecimal(uint,uint,uint)\n"); - assertLeDecimal(uint(2.0e18), 1.1e18, 18); - emit log("\n"); - assertLeDecimal(uint(2.0e18), 1.1e18, 18, err); - - emit log("\n## assertEq(string,string)\n"); - string memory s1 = "string 1"; - string memory s2 = "string 2"; - assertEq(s1, s2); - emit log("\n"); - assertEq(s1, s2, err); - - emit log("\n## assertEq0(bytes,bytes)\n"); - assertEq0(hex"abcdef01", hex"abcdef02"); - emit log("\n"); - assertEq0(hex"abcdef01", hex"abcdef02", err); - } -} - -contract DemoTestWithSetUp { - function setUp() public { - } - function test_pass() public pure { - } -} diff --git a/lib/solmate/lib/ds-test/package.json b/lib/solmate/lib/ds-test/package.json deleted file mode 100644 index 4802ada..0000000 --- a/lib/solmate/lib/ds-test/package.json +++ /dev/null @@ -1,15 +0,0 @@ -{ - "name": "ds-test", - "version": "1.0.0", - "description": "Assertions, equality checks and other test helpers ", - "bugs": "https://github.com/dapphub/ds-test/issues", - "license": "GPL-3.0", - "author": "Contributors to ds-test", - "files": [ - "src/*" - ], - "repository": { - "type": "git", - "url": "https://github.com/dapphub/ds-test.git" - } -} diff --git a/lib/solmate/lib/ds-test/src/test.sol b/lib/solmate/lib/ds-test/src/test.sol deleted file mode 100644 index 515a3bd..0000000 --- a/lib/solmate/lib/ds-test/src/test.sol +++ /dev/null @@ -1,469 +0,0 @@ -// SPDX-License-Identifier: GPL-3.0-or-later - -// This program is free software: you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation, either version 3 of the License, or -// (at your option) any later version. - -// This program is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. - -// You should have received a copy of the GNU General Public License -// along with this program. If not, see . - -pragma solidity >=0.5.0; - -contract DSTest { - event log (string); - event logs (bytes); - - event log_address (address); - event log_bytes32 (bytes32); - event log_int (int); - event log_uint (uint); - event log_bytes (bytes); - event log_string (string); - - event log_named_address (string key, address val); - event log_named_bytes32 (string key, bytes32 val); - event log_named_decimal_int (string key, int val, uint decimals); - event log_named_decimal_uint (string key, uint val, uint decimals); - event log_named_int (string key, int val); - event log_named_uint (string key, uint val); - event log_named_bytes (string key, bytes val); - event log_named_string (string key, string val); - - bool public IS_TEST = true; - bool private _failed; - - address constant HEVM_ADDRESS = - address(bytes20(uint160(uint256(keccak256('hevm cheat code'))))); - - modifier mayRevert() { _; } - modifier testopts(string memory) { _; } - - function failed() public returns (bool) { - if (_failed) { - return _failed; - } else { - bool globalFailed = false; - if (hasHEVMContext()) { - (, bytes memory retdata) = HEVM_ADDRESS.call( - abi.encodePacked( - bytes4(keccak256("load(address,bytes32)")), - abi.encode(HEVM_ADDRESS, bytes32("failed")) - ) - ); - globalFailed = abi.decode(retdata, (bool)); - } - return globalFailed; - } - } - - function fail() internal { - if (hasHEVMContext()) { - (bool status, ) = HEVM_ADDRESS.call( - abi.encodePacked( - bytes4(keccak256("store(address,bytes32,bytes32)")), - abi.encode(HEVM_ADDRESS, bytes32("failed"), bytes32(uint256(0x01))) - ) - ); - status; // Silence compiler warnings - } - _failed = true; - } - - function hasHEVMContext() internal view returns (bool) { - uint256 hevmCodeSize = 0; - assembly { - hevmCodeSize := extcodesize(0x7109709ECfa91a80626fF3989D68f67F5b1DD12D) - } - return hevmCodeSize > 0; - } - - modifier logs_gas() { - uint startGas = gasleft(); - _; - uint endGas = gasleft(); - emit log_named_uint("gas", startGas - endGas); - } - - function assertTrue(bool condition) internal { - if (!condition) { - emit log("Error: Assertion Failed"); - fail(); - } - } - - function assertTrue(bool condition, string memory err) internal { - if (!condition) { - emit log_named_string("Error", err); - assertTrue(condition); - } - } - - function assertEq(address a, address b) internal { - if (a != b) { - emit log("Error: a == b not satisfied [address]"); - emit log_named_address(" Expected", b); - emit log_named_address(" Actual", a); - fail(); - } - } - function assertEq(address a, address b, string memory err) internal { - if (a != b) { - emit log_named_string ("Error", err); - assertEq(a, b); - } - } - - function assertEq(bytes32 a, bytes32 b) internal { - if (a != b) { - emit log("Error: a == b not satisfied [bytes32]"); - emit log_named_bytes32(" Expected", b); - emit log_named_bytes32(" Actual", a); - fail(); - } - } - function assertEq(bytes32 a, bytes32 b, string memory err) internal { - if (a != b) { - emit log_named_string ("Error", err); - assertEq(a, b); - } - } - function assertEq32(bytes32 a, bytes32 b) internal { - assertEq(a, b); - } - function assertEq32(bytes32 a, bytes32 b, string memory err) internal { - assertEq(a, b, err); - } - - function assertEq(int a, int b) internal { - if (a != b) { - emit log("Error: a == b not satisfied [int]"); - emit log_named_int(" Expected", b); - emit log_named_int(" Actual", a); - fail(); - } - } - function assertEq(int a, int b, string memory err) internal { - if (a != b) { - emit log_named_string("Error", err); - assertEq(a, b); - } - } - function assertEq(uint a, uint b) internal { - if (a != b) { - emit log("Error: a == b not satisfied [uint]"); - emit log_named_uint(" Expected", b); - emit log_named_uint(" Actual", a); - fail(); - } - } - function assertEq(uint a, uint b, string memory err) internal { - if (a != b) { - emit log_named_string("Error", err); - assertEq(a, b); - } - } - function assertEqDecimal(int a, int b, uint decimals) internal { - if (a != b) { - emit log("Error: a == b not satisfied [decimal int]"); - emit log_named_decimal_int(" Expected", b, decimals); - emit log_named_decimal_int(" Actual", a, decimals); - fail(); - } - } - function assertEqDecimal(int a, int b, uint decimals, string memory err) internal { - if (a != b) { - emit log_named_string("Error", err); - assertEqDecimal(a, b, decimals); - } - } - function assertEqDecimal(uint a, uint b, uint decimals) internal { - if (a != b) { - emit log("Error: a == b not satisfied [decimal uint]"); - emit log_named_decimal_uint(" Expected", b, decimals); - emit log_named_decimal_uint(" Actual", a, decimals); - fail(); - } - } - function assertEqDecimal(uint a, uint b, uint decimals, string memory err) internal { - if (a != b) { - emit log_named_string("Error", err); - assertEqDecimal(a, b, decimals); - } - } - - function assertGt(uint a, uint b) internal { - if (a <= b) { - emit log("Error: a > b not satisfied [uint]"); - emit log_named_uint(" Value a", a); - emit log_named_uint(" Value b", b); - fail(); - } - } - function assertGt(uint a, uint b, string memory err) internal { - if (a <= b) { - emit log_named_string("Error", err); - assertGt(a, b); - } - } - function assertGt(int a, int b) internal { - if (a <= b) { - emit log("Error: a > b not satisfied [int]"); - emit log_named_int(" Value a", a); - emit log_named_int(" Value b", b); - fail(); - } - } - function assertGt(int a, int b, string memory err) internal { - if (a <= b) { - emit log_named_string("Error", err); - assertGt(a, b); - } - } - function assertGtDecimal(int a, int b, uint decimals) internal { - if (a <= b) { - emit log("Error: a > b not satisfied [decimal int]"); - emit log_named_decimal_int(" Value a", a, decimals); - emit log_named_decimal_int(" Value b", b, decimals); - fail(); - } - } - function assertGtDecimal(int a, int b, uint decimals, string memory err) internal { - if (a <= b) { - emit log_named_string("Error", err); - assertGtDecimal(a, b, decimals); - } - } - function assertGtDecimal(uint a, uint b, uint decimals) internal { - if (a <= b) { - emit log("Error: a > b not satisfied [decimal uint]"); - emit log_named_decimal_uint(" Value a", a, decimals); - emit log_named_decimal_uint(" Value b", b, decimals); - fail(); - } - } - function assertGtDecimal(uint a, uint b, uint decimals, string memory err) internal { - if (a <= b) { - emit log_named_string("Error", err); - assertGtDecimal(a, b, decimals); - } - } - - function assertGe(uint a, uint b) internal { - if (a < b) { - emit log("Error: a >= b not satisfied [uint]"); - emit log_named_uint(" Value a", a); - emit log_named_uint(" Value b", b); - fail(); - } - } - function assertGe(uint a, uint b, string memory err) internal { - if (a < b) { - emit log_named_string("Error", err); - assertGe(a, b); - } - } - function assertGe(int a, int b) internal { - if (a < b) { - emit log("Error: a >= b not satisfied [int]"); - emit log_named_int(" Value a", a); - emit log_named_int(" Value b", b); - fail(); - } - } - function assertGe(int a, int b, string memory err) internal { - if (a < b) { - emit log_named_string("Error", err); - assertGe(a, b); - } - } - function assertGeDecimal(int a, int b, uint decimals) internal { - if (a < b) { - emit log("Error: a >= b not satisfied [decimal int]"); - emit log_named_decimal_int(" Value a", a, decimals); - emit log_named_decimal_int(" Value b", b, decimals); - fail(); - } - } - function assertGeDecimal(int a, int b, uint decimals, string memory err) internal { - if (a < b) { - emit log_named_string("Error", err); - assertGeDecimal(a, b, decimals); - } - } - function assertGeDecimal(uint a, uint b, uint decimals) internal { - if (a < b) { - emit log("Error: a >= b not satisfied [decimal uint]"); - emit log_named_decimal_uint(" Value a", a, decimals); - emit log_named_decimal_uint(" Value b", b, decimals); - fail(); - } - } - function assertGeDecimal(uint a, uint b, uint decimals, string memory err) internal { - if (a < b) { - emit log_named_string("Error", err); - assertGeDecimal(a, b, decimals); - } - } - - function assertLt(uint a, uint b) internal { - if (a >= b) { - emit log("Error: a < b not satisfied [uint]"); - emit log_named_uint(" Value a", a); - emit log_named_uint(" Value b", b); - fail(); - } - } - function assertLt(uint a, uint b, string memory err) internal { - if (a >= b) { - emit log_named_string("Error", err); - assertLt(a, b); - } - } - function assertLt(int a, int b) internal { - if (a >= b) { - emit log("Error: a < b not satisfied [int]"); - emit log_named_int(" Value a", a); - emit log_named_int(" Value b", b); - fail(); - } - } - function assertLt(int a, int b, string memory err) internal { - if (a >= b) { - emit log_named_string("Error", err); - assertLt(a, b); - } - } - function assertLtDecimal(int a, int b, uint decimals) internal { - if (a >= b) { - emit log("Error: a < b not satisfied [decimal int]"); - emit log_named_decimal_int(" Value a", a, decimals); - emit log_named_decimal_int(" Value b", b, decimals); - fail(); - } - } - function assertLtDecimal(int a, int b, uint decimals, string memory err) internal { - if (a >= b) { - emit log_named_string("Error", err); - assertLtDecimal(a, b, decimals); - } - } - function assertLtDecimal(uint a, uint b, uint decimals) internal { - if (a >= b) { - emit log("Error: a < b not satisfied [decimal uint]"); - emit log_named_decimal_uint(" Value a", a, decimals); - emit log_named_decimal_uint(" Value b", b, decimals); - fail(); - } - } - function assertLtDecimal(uint a, uint b, uint decimals, string memory err) internal { - if (a >= b) { - emit log_named_string("Error", err); - assertLtDecimal(a, b, decimals); - } - } - - function assertLe(uint a, uint b) internal { - if (a > b) { - emit log("Error: a <= b not satisfied [uint]"); - emit log_named_uint(" Value a", a); - emit log_named_uint(" Value b", b); - fail(); - } - } - function assertLe(uint a, uint b, string memory err) internal { - if (a > b) { - emit log_named_string("Error", err); - assertLe(a, b); - } - } - function assertLe(int a, int b) internal { - if (a > b) { - emit log("Error: a <= b not satisfied [int]"); - emit log_named_int(" Value a", a); - emit log_named_int(" Value b", b); - fail(); - } - } - function assertLe(int a, int b, string memory err) internal { - if (a > b) { - emit log_named_string("Error", err); - assertLe(a, b); - } - } - function assertLeDecimal(int a, int b, uint decimals) internal { - if (a > b) { - emit log("Error: a <= b not satisfied [decimal int]"); - emit log_named_decimal_int(" Value a", a, decimals); - emit log_named_decimal_int(" Value b", b, decimals); - fail(); - } - } - function assertLeDecimal(int a, int b, uint decimals, string memory err) internal { - if (a > b) { - emit log_named_string("Error", err); - assertLeDecimal(a, b, decimals); - } - } - function assertLeDecimal(uint a, uint b, uint decimals) internal { - if (a > b) { - emit log("Error: a <= b not satisfied [decimal uint]"); - emit log_named_decimal_uint(" Value a", a, decimals); - emit log_named_decimal_uint(" Value b", b, decimals); - fail(); - } - } - function assertLeDecimal(uint a, uint b, uint decimals, string memory err) internal { - if (a > b) { - emit log_named_string("Error", err); - assertGeDecimal(a, b, decimals); - } - } - - function assertEq(string memory a, string memory b) internal { - if (keccak256(abi.encodePacked(a)) != keccak256(abi.encodePacked(b))) { - emit log("Error: a == b not satisfied [string]"); - emit log_named_string(" Expected", b); - emit log_named_string(" Actual", a); - fail(); - } - } - function assertEq(string memory a, string memory b, string memory err) internal { - if (keccak256(abi.encodePacked(a)) != keccak256(abi.encodePacked(b))) { - emit log_named_string("Error", err); - assertEq(a, b); - } - } - - function checkEq0(bytes memory a, bytes memory b) internal pure returns (bool ok) { - ok = true; - if (a.length == b.length) { - for (uint i = 0; i < a.length; i++) { - if (a[i] != b[i]) { - ok = false; - } - } - } else { - ok = false; - } - } - function assertEq0(bytes memory a, bytes memory b) internal { - if (!checkEq0(a, b)) { - emit log("Error: a == b not satisfied [bytes]"); - emit log_named_bytes(" Expected", b); - emit log_named_bytes(" Actual", a); - fail(); - } - } - function assertEq0(bytes memory a, bytes memory b, string memory err) internal { - if (!checkEq0(a, b)) { - emit log_named_string("Error", err); - assertEq0(a, b); - } - } -} diff --git a/lib/solmate/package-lock.json b/lib/solmate/package-lock.json deleted file mode 100644 index b99f283..0000000 --- a/lib/solmate/package-lock.json +++ /dev/null @@ -1,125 +0,0 @@ -{ - "name": "solmate", - "version": "6.2.0", - "lockfileVersion": 1, - "requires": true, - "dependencies": { - "@solidity-parser/parser": { - "version": "0.13.2", - "resolved": "https://registry.npmjs.org/@solidity-parser/parser/-/parser-0.13.2.tgz", - "integrity": "sha512-RwHnpRnfrnD2MSPveYoPh8nhofEvX7fgjHk1Oq+NNvCcLx4r1js91CO9o+F/F3fBzOCyvm8kKRTriFICX/odWw==", - "dev": true, - "requires": { - "antlr4ts": "^0.5.0-alpha.4" - } - }, - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", - "dev": true - }, - "antlr4ts": { - "version": "0.5.0-alpha.4", - "resolved": "https://registry.npmjs.org/antlr4ts/-/antlr4ts-0.5.0-alpha.4.tgz", - "integrity": "sha512-WPQDt1B74OfPv/IMS2ekXAKkTZIHl88uMetg6q3OTqgFxZ/dxDXI0EWLyZid/1Pe6hTftyg5N7gel5wNAGxXyQ==", - "dev": true - }, - "emoji-regex": { - "version": "9.2.2", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", - "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", - "dev": true - }, - "escape-string-regexp": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", - "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", - "dev": true - }, - "lru-cache": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", - "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", - "dev": true, - "requires": { - "yallist": "^4.0.0" - } - }, - "prettier": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.3.2.tgz", - "integrity": "sha512-lnJzDfJ66zkMy58OL5/NY5zp70S7Nz6KqcKkXYzn2tMVrNxvbqaBpg7H3qHaLxCJ5lNMsGuM8+ohS7cZrthdLQ==", - "dev": true - }, - "prettier-plugin-solidity": { - "version": "1.0.0-beta.16", - "resolved": "https://registry.npmjs.org/prettier-plugin-solidity/-/prettier-plugin-solidity-1.0.0-beta.16.tgz", - "integrity": "sha512-xVBcnoWpe52dNnCCbqPHC9ZrTWXcNfldf852ZD0DBcHDqVMwjHTAPEdfBVy6FczbFpVa8bmxQil+G5XkEz5WHA==", - "dev": true, - "requires": { - "@solidity-parser/parser": "^0.13.2", - "emoji-regex": "^9.2.2", - "escape-string-regexp": "^4.0.0", - "semver": "^7.3.5", - "solidity-comments-extractor": "^0.0.7", - "string-width": "^4.2.2" - } - }, - "semver": { - "version": "7.3.5", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", - "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", - "dev": true, - "requires": { - "lru-cache": "^6.0.0" - } - }, - "solidity-comments-extractor": { - "version": "0.0.7", - "resolved": "https://registry.npmjs.org/solidity-comments-extractor/-/solidity-comments-extractor-0.0.7.tgz", - "integrity": "sha512-wciNMLg/Irp8OKGrh3S2tfvZiZ0NEyILfcRCXCD4mp7SgK/i9gzLfhY2hY7VMCQJ3kH9UB9BzNdibIVMchzyYw==", - "dev": true - }, - "string-width": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", - "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", - "dev": true, - "requires": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.0" - }, - "dependencies": { - "emoji-regex": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", - "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", - "dev": true - } - } - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "dev": true, - "requires": { - "ansi-regex": "^5.0.0" - } - }, - "yallist": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", - "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", - "dev": true - } - } -} diff --git a/lib/solmate/package.json b/lib/solmate/package.json deleted file mode 100644 index e94cbdf..0000000 --- a/lib/solmate/package.json +++ /dev/null @@ -1,20 +0,0 @@ -{ - "name": "solmate", - "license": "AGPL-3.0-only", - "version": "6.2.0", - "description": "Modern, opinionated and gas optimized building blocks for smart contract development.", - "files": [ - "src/**/*.sol" - ], - "repository": { - "type": "git", - "url": "git+https://github.com/transmissions11/solmate.git" - }, - "devDependencies": { - "prettier": "^2.3.1", - "prettier-plugin-solidity": "^1.0.0-beta.13" - }, - "scripts": { - "lint": "prettier --write **.sol" - } -} diff --git a/lib/solmate/src/auth/Auth.sol b/lib/solmate/src/auth/Auth.sol deleted file mode 100644 index 3807ccd..0000000 --- a/lib/solmate/src/auth/Auth.sol +++ /dev/null @@ -1,64 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -/// @notice Provides a flexible and updatable auth pattern which is completely separate from application logic. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/auth/Auth.sol) -/// @author Modified from Dappsys (https://github.com/dapphub/ds-auth/blob/master/src/auth.sol) -abstract contract Auth { - event OwnershipTransferred(address indexed user, address indexed newOwner); - - event AuthorityUpdated(address indexed user, Authority indexed newAuthority); - - address public owner; - - Authority public authority; - - constructor(address _owner, Authority _authority) { - owner = _owner; - authority = _authority; - - emit OwnershipTransferred(msg.sender, _owner); - emit AuthorityUpdated(msg.sender, _authority); - } - - modifier requiresAuth() virtual { - require(isAuthorized(msg.sender, msg.sig), "UNAUTHORIZED"); - - _; - } - - function isAuthorized(address user, bytes4 functionSig) internal view virtual returns (bool) { - Authority auth = authority; // Memoizing authority saves us a warm SLOAD, around 100 gas. - - // Checking if the caller is the owner only after calling the authority saves gas in most cases, but be - // aware that this makes protected functions uncallable even to the owner if the authority is out of order. - return (address(auth) != address(0) && auth.canCall(user, address(this), functionSig)) || user == owner; - } - - function setAuthority(Authority newAuthority) public virtual { - // We check if the caller is the owner first because we want to ensure they can - // always swap out the authority even if it's reverting or using up a lot of gas. - require(msg.sender == owner || authority.canCall(msg.sender, address(this), msg.sig)); - - authority = newAuthority; - - emit AuthorityUpdated(msg.sender, newAuthority); - } - - function transferOwnership(address newOwner) public virtual requiresAuth { - owner = newOwner; - - emit OwnershipTransferred(msg.sender, newOwner); - } -} - -/// @notice A generic interface for a contract which provides authorization data to an Auth instance. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/auth/Auth.sol) -/// @author Modified from Dappsys (https://github.com/dapphub/ds-auth/blob/master/src/auth.sol) -interface Authority { - function canCall( - address user, - address target, - bytes4 functionSig - ) external view returns (bool); -} diff --git a/lib/solmate/src/auth/Owned.sol b/lib/solmate/src/auth/Owned.sol deleted file mode 100644 index e82b44d..0000000 --- a/lib/solmate/src/auth/Owned.sol +++ /dev/null @@ -1,44 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -/// @notice Simple single owner authorization mixin. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/auth/Owned.sol) -abstract contract Owned { - /*////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event OwnershipTransferred(address indexed user, address indexed newOwner); - - /*////////////////////////////////////////////////////////////// - OWNERSHIP STORAGE - //////////////////////////////////////////////////////////////*/ - - address public owner; - - modifier onlyOwner() virtual { - require(msg.sender == owner, "UNAUTHORIZED"); - - _; - } - - /*////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor(address _owner) { - owner = _owner; - - emit OwnershipTransferred(address(0), _owner); - } - - /*////////////////////////////////////////////////////////////// - OWNERSHIP LOGIC - //////////////////////////////////////////////////////////////*/ - - function transferOwnership(address newOwner) public virtual onlyOwner { - owner = newOwner; - - emit OwnershipTransferred(msg.sender, newOwner); - } -} diff --git a/lib/solmate/src/auth/authorities/MultiRolesAuthority.sol b/lib/solmate/src/auth/authorities/MultiRolesAuthority.sol deleted file mode 100644 index 3ff80bb..0000000 --- a/lib/solmate/src/auth/authorities/MultiRolesAuthority.sol +++ /dev/null @@ -1,123 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {Auth, Authority} from "../Auth.sol"; - -/// @notice Flexible and target agnostic role based Authority that supports up to 256 roles. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/auth/authorities/MultiRolesAuthority.sol) -contract MultiRolesAuthority is Auth, Authority { - /*////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event UserRoleUpdated(address indexed user, uint8 indexed role, bool enabled); - - event PublicCapabilityUpdated(bytes4 indexed functionSig, bool enabled); - - event RoleCapabilityUpdated(uint8 indexed role, bytes4 indexed functionSig, bool enabled); - - event TargetCustomAuthorityUpdated(address indexed target, Authority indexed authority); - - /*////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor(address _owner, Authority _authority) Auth(_owner, _authority) {} - - /*////////////////////////////////////////////////////////////// - CUSTOM TARGET AUTHORITY STORAGE - //////////////////////////////////////////////////////////////*/ - - mapping(address => Authority) public getTargetCustomAuthority; - - /*////////////////////////////////////////////////////////////// - ROLE/USER STORAGE - //////////////////////////////////////////////////////////////*/ - - mapping(address => bytes32) public getUserRoles; - - mapping(bytes4 => bool) public isCapabilityPublic; - - mapping(bytes4 => bytes32) public getRolesWithCapability; - - function doesUserHaveRole(address user, uint8 role) public view virtual returns (bool) { - return (uint256(getUserRoles[user]) >> role) & 1 != 0; - } - - function doesRoleHaveCapability(uint8 role, bytes4 functionSig) public view virtual returns (bool) { - return (uint256(getRolesWithCapability[functionSig]) >> role) & 1 != 0; - } - - /*////////////////////////////////////////////////////////////// - AUTHORIZATION LOGIC - //////////////////////////////////////////////////////////////*/ - - function canCall( - address user, - address target, - bytes4 functionSig - ) public view virtual override returns (bool) { - Authority customAuthority = getTargetCustomAuthority[target]; - - if (address(customAuthority) != address(0)) return customAuthority.canCall(user, target, functionSig); - - return - isCapabilityPublic[functionSig] || bytes32(0) != getUserRoles[user] & getRolesWithCapability[functionSig]; - } - - /*/////////////////////////////////////////////////////////////// - CUSTOM TARGET AUTHORITY CONFIGURATION LOGIC - //////////////////////////////////////////////////////////////*/ - - function setTargetCustomAuthority(address target, Authority customAuthority) public virtual requiresAuth { - getTargetCustomAuthority[target] = customAuthority; - - emit TargetCustomAuthorityUpdated(target, customAuthority); - } - - /*////////////////////////////////////////////////////////////// - PUBLIC CAPABILITY CONFIGURATION LOGIC - //////////////////////////////////////////////////////////////*/ - - function setPublicCapability(bytes4 functionSig, bool enabled) public virtual requiresAuth { - isCapabilityPublic[functionSig] = enabled; - - emit PublicCapabilityUpdated(functionSig, enabled); - } - - /*////////////////////////////////////////////////////////////// - USER ROLE ASSIGNMENT LOGIC - //////////////////////////////////////////////////////////////*/ - - function setUserRole( - address user, - uint8 role, - bool enabled - ) public virtual requiresAuth { - if (enabled) { - getUserRoles[user] |= bytes32(1 << role); - } else { - getUserRoles[user] &= ~bytes32(1 << role); - } - - emit UserRoleUpdated(user, role, enabled); - } - - /*////////////////////////////////////////////////////////////// - ROLE CAPABILITY CONFIGURATION LOGIC - //////////////////////////////////////////////////////////////*/ - - function setRoleCapability( - uint8 role, - bytes4 functionSig, - bool enabled - ) public virtual requiresAuth { - if (enabled) { - getRolesWithCapability[functionSig] |= bytes32(1 << role); - } else { - getRolesWithCapability[functionSig] &= ~bytes32(1 << role); - } - - emit RoleCapabilityUpdated(role, functionSig, enabled); - } -} diff --git a/lib/solmate/src/auth/authorities/RolesAuthority.sol b/lib/solmate/src/auth/authorities/RolesAuthority.sol deleted file mode 100644 index aa5cc71..0000000 --- a/lib/solmate/src/auth/authorities/RolesAuthority.sol +++ /dev/null @@ -1,108 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {Auth, Authority} from "../Auth.sol"; - -/// @notice Role based Authority that supports up to 256 roles. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/auth/authorities/RolesAuthority.sol) -/// @author Modified from Dappsys (https://github.com/dapphub/ds-roles/blob/master/src/roles.sol) -contract RolesAuthority is Auth, Authority { - /*////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event UserRoleUpdated(address indexed user, uint8 indexed role, bool enabled); - - event PublicCapabilityUpdated(address indexed target, bytes4 indexed functionSig, bool enabled); - - event RoleCapabilityUpdated(uint8 indexed role, address indexed target, bytes4 indexed functionSig, bool enabled); - - /*////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor(address _owner, Authority _authority) Auth(_owner, _authority) {} - - /*////////////////////////////////////////////////////////////// - ROLE/USER STORAGE - //////////////////////////////////////////////////////////////*/ - - mapping(address => bytes32) public getUserRoles; - - mapping(address => mapping(bytes4 => bool)) public isCapabilityPublic; - - mapping(address => mapping(bytes4 => bytes32)) public getRolesWithCapability; - - function doesUserHaveRole(address user, uint8 role) public view virtual returns (bool) { - return (uint256(getUserRoles[user]) >> role) & 1 != 0; - } - - function doesRoleHaveCapability( - uint8 role, - address target, - bytes4 functionSig - ) public view virtual returns (bool) { - return (uint256(getRolesWithCapability[target][functionSig]) >> role) & 1 != 0; - } - - /*////////////////////////////////////////////////////////////// - AUTHORIZATION LOGIC - //////////////////////////////////////////////////////////////*/ - - function canCall( - address user, - address target, - bytes4 functionSig - ) public view virtual override returns (bool) { - return - isCapabilityPublic[target][functionSig] || - bytes32(0) != getUserRoles[user] & getRolesWithCapability[target][functionSig]; - } - - /*////////////////////////////////////////////////////////////// - ROLE CAPABILITY CONFIGURATION LOGIC - //////////////////////////////////////////////////////////////*/ - - function setPublicCapability( - address target, - bytes4 functionSig, - bool enabled - ) public virtual requiresAuth { - isCapabilityPublic[target][functionSig] = enabled; - - emit PublicCapabilityUpdated(target, functionSig, enabled); - } - - function setRoleCapability( - uint8 role, - address target, - bytes4 functionSig, - bool enabled - ) public virtual requiresAuth { - if (enabled) { - getRolesWithCapability[target][functionSig] |= bytes32(1 << role); - } else { - getRolesWithCapability[target][functionSig] &= ~bytes32(1 << role); - } - - emit RoleCapabilityUpdated(role, target, functionSig, enabled); - } - - /*////////////////////////////////////////////////////////////// - USER ROLE ASSIGNMENT LOGIC - //////////////////////////////////////////////////////////////*/ - - function setUserRole( - address user, - uint8 role, - bool enabled - ) public virtual requiresAuth { - if (enabled) { - getUserRoles[user] |= bytes32(1 << role); - } else { - getUserRoles[user] &= ~bytes32(1 << role); - } - - emit UserRoleUpdated(user, role, enabled); - } -} diff --git a/lib/solmate/src/mixins/ERC4626.sol b/lib/solmate/src/mixins/ERC4626.sol deleted file mode 100644 index af56c15..0000000 --- a/lib/solmate/src/mixins/ERC4626.sol +++ /dev/null @@ -1,183 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {ERC20} from "../tokens/ERC20.sol"; -import {SafeTransferLib} from "../utils/SafeTransferLib.sol"; -import {FixedPointMathLib} from "../utils/FixedPointMathLib.sol"; - -/// @notice Minimal ERC4626 tokenized Vault implementation. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/mixins/ERC4626.sol) -abstract contract ERC4626 is ERC20 { - using SafeTransferLib for ERC20; - using FixedPointMathLib for uint256; - - /*////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Deposit(address indexed caller, address indexed owner, uint256 assets, uint256 shares); - - event Withdraw( - address indexed caller, - address indexed receiver, - address indexed owner, - uint256 assets, - uint256 shares - ); - - /*////////////////////////////////////////////////////////////// - IMMUTABLES - //////////////////////////////////////////////////////////////*/ - - ERC20 public immutable asset; - - constructor( - ERC20 _asset, - string memory _name, - string memory _symbol - ) ERC20(_name, _symbol, _asset.decimals()) { - asset = _asset; - } - - /*////////////////////////////////////////////////////////////// - DEPOSIT/WITHDRAWAL LOGIC - //////////////////////////////////////////////////////////////*/ - - function deposit(uint256 assets, address receiver) public virtual returns (uint256 shares) { - // Check for rounding error since we round down in previewDeposit. - require((shares = previewDeposit(assets)) != 0, "ZERO_SHARES"); - - // Need to transfer before minting or ERC777s could reenter. - asset.safeTransferFrom(msg.sender, address(this), assets); - - _mint(receiver, shares); - - emit Deposit(msg.sender, receiver, assets, shares); - - afterDeposit(assets, shares); - } - - function mint(uint256 shares, address receiver) public virtual returns (uint256 assets) { - assets = previewMint(shares); // No need to check for rounding error, previewMint rounds up. - - // Need to transfer before minting or ERC777s could reenter. - asset.safeTransferFrom(msg.sender, address(this), assets); - - _mint(receiver, shares); - - emit Deposit(msg.sender, receiver, assets, shares); - - afterDeposit(assets, shares); - } - - function withdraw( - uint256 assets, - address receiver, - address owner - ) public virtual returns (uint256 shares) { - shares = previewWithdraw(assets); // No need to check for rounding error, previewWithdraw rounds up. - - if (msg.sender != owner) { - uint256 allowed = allowance[owner][msg.sender]; // Saves gas for limited approvals. - - if (allowed != type(uint256).max) allowance[owner][msg.sender] = allowed - shares; - } - - beforeWithdraw(assets, shares); - - _burn(owner, shares); - - emit Withdraw(msg.sender, receiver, owner, assets, shares); - - asset.safeTransfer(receiver, assets); - } - - function redeem( - uint256 shares, - address receiver, - address owner - ) public virtual returns (uint256 assets) { - if (msg.sender != owner) { - uint256 allowed = allowance[owner][msg.sender]; // Saves gas for limited approvals. - - if (allowed != type(uint256).max) allowance[owner][msg.sender] = allowed - shares; - } - - // Check for rounding error since we round down in previewRedeem. - require((assets = previewRedeem(shares)) != 0, "ZERO_ASSETS"); - - beforeWithdraw(assets, shares); - - _burn(owner, shares); - - emit Withdraw(msg.sender, receiver, owner, assets, shares); - - asset.safeTransfer(receiver, assets); - } - - /*////////////////////////////////////////////////////////////// - ACCOUNTING LOGIC - //////////////////////////////////////////////////////////////*/ - - function totalAssets() public view virtual returns (uint256); - - function convertToShares(uint256 assets) public view virtual returns (uint256) { - uint256 supply = totalSupply; // Saves an extra SLOAD if totalSupply is non-zero. - - return supply == 0 ? assets : assets.mulDivDown(supply, totalAssets()); - } - - function convertToAssets(uint256 shares) public view virtual returns (uint256) { - uint256 supply = totalSupply; // Saves an extra SLOAD if totalSupply is non-zero. - - return supply == 0 ? shares : shares.mulDivDown(totalAssets(), supply); - } - - function previewDeposit(uint256 assets) public view virtual returns (uint256) { - return convertToShares(assets); - } - - function previewMint(uint256 shares) public view virtual returns (uint256) { - uint256 supply = totalSupply; // Saves an extra SLOAD if totalSupply is non-zero. - - return supply == 0 ? shares : shares.mulDivUp(totalAssets(), supply); - } - - function previewWithdraw(uint256 assets) public view virtual returns (uint256) { - uint256 supply = totalSupply; // Saves an extra SLOAD if totalSupply is non-zero. - - return supply == 0 ? assets : assets.mulDivUp(supply, totalAssets()); - } - - function previewRedeem(uint256 shares) public view virtual returns (uint256) { - return convertToAssets(shares); - } - - /*////////////////////////////////////////////////////////////// - DEPOSIT/WITHDRAWAL LIMIT LOGIC - //////////////////////////////////////////////////////////////*/ - - function maxDeposit(address) public view virtual returns (uint256) { - return type(uint256).max; - } - - function maxMint(address) public view virtual returns (uint256) { - return type(uint256).max; - } - - function maxWithdraw(address owner) public view virtual returns (uint256) { - return convertToAssets(balanceOf[owner]); - } - - function maxRedeem(address owner) public view virtual returns (uint256) { - return balanceOf[owner]; - } - - /*////////////////////////////////////////////////////////////// - INTERNAL HOOKS LOGIC - //////////////////////////////////////////////////////////////*/ - - function beforeWithdraw(uint256 assets, uint256 shares) internal virtual {} - - function afterDeposit(uint256 assets, uint256 shares) internal virtual {} -} diff --git a/lib/solmate/src/test/Auth.t.sol b/lib/solmate/src/test/Auth.t.sol deleted file mode 100644 index 9e31528..0000000 --- a/lib/solmate/src/test/Auth.t.sol +++ /dev/null @@ -1,192 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; -import {MockAuthChild} from "./utils/mocks/MockAuthChild.sol"; -import {MockAuthority} from "./utils/mocks/MockAuthority.sol"; - -import {Authority} from "../auth/Auth.sol"; - -contract OutOfOrderAuthority is Authority { - function canCall( - address, - address, - bytes4 - ) public pure override returns (bool) { - revert("OUT_OF_ORDER"); - } -} - -contract AuthTest is DSTestPlus { - MockAuthChild mockAuthChild; - - function setUp() public { - mockAuthChild = new MockAuthChild(); - } - - function testTransferOwnershipAsOwner() public { - mockAuthChild.transferOwnership(address(0xBEEF)); - assertEq(mockAuthChild.owner(), address(0xBEEF)); - } - - function testSetAuthorityAsOwner() public { - mockAuthChild.setAuthority(Authority(address(0xBEEF))); - assertEq(address(mockAuthChild.authority()), address(0xBEEF)); - } - - function testCallFunctionAsOwner() public { - mockAuthChild.updateFlag(); - } - - function testTransferOwnershipWithPermissiveAuthority() public { - mockAuthChild.setAuthority(new MockAuthority(true)); - mockAuthChild.transferOwnership(address(0)); - mockAuthChild.transferOwnership(address(this)); - } - - function testSetAuthorityWithPermissiveAuthority() public { - mockAuthChild.setAuthority(new MockAuthority(true)); - mockAuthChild.transferOwnership(address(0)); - mockAuthChild.setAuthority(Authority(address(0xBEEF))); - } - - function testCallFunctionWithPermissiveAuthority() public { - mockAuthChild.setAuthority(new MockAuthority(true)); - mockAuthChild.transferOwnership(address(0)); - mockAuthChild.updateFlag(); - } - - function testSetAuthorityAsOwnerWithOutOfOrderAuthority() public { - mockAuthChild.setAuthority(new OutOfOrderAuthority()); - mockAuthChild.setAuthority(new MockAuthority(true)); - } - - function testFailTransferOwnershipAsNonOwner() public { - mockAuthChild.transferOwnership(address(0)); - mockAuthChild.transferOwnership(address(0xBEEF)); - } - - function testFailSetAuthorityAsNonOwner() public { - mockAuthChild.transferOwnership(address(0)); - mockAuthChild.setAuthority(Authority(address(0xBEEF))); - } - - function testFailCallFunctionAsNonOwner() public { - mockAuthChild.transferOwnership(address(0)); - mockAuthChild.updateFlag(); - } - - function testFailTransferOwnershipWithRestrictiveAuthority() public { - mockAuthChild.setAuthority(new MockAuthority(false)); - mockAuthChild.transferOwnership(address(0)); - mockAuthChild.transferOwnership(address(this)); - } - - function testFailSetAuthorityWithRestrictiveAuthority() public { - mockAuthChild.setAuthority(new MockAuthority(false)); - mockAuthChild.transferOwnership(address(0)); - mockAuthChild.setAuthority(Authority(address(0xBEEF))); - } - - function testFailCallFunctionWithRestrictiveAuthority() public { - mockAuthChild.setAuthority(new MockAuthority(false)); - mockAuthChild.transferOwnership(address(0)); - mockAuthChild.updateFlag(); - } - - function testFailTransferOwnershipAsOwnerWithOutOfOrderAuthority() public { - mockAuthChild.setAuthority(new OutOfOrderAuthority()); - mockAuthChild.transferOwnership(address(0)); - } - - function testFailCallFunctionAsOwnerWithOutOfOrderAuthority() public { - mockAuthChild.setAuthority(new OutOfOrderAuthority()); - mockAuthChild.updateFlag(); - } - - function testTransferOwnershipAsOwner(address newOwner) public { - mockAuthChild.transferOwnership(newOwner); - assertEq(mockAuthChild.owner(), newOwner); - } - - function testSetAuthorityAsOwner(Authority newAuthority) public { - mockAuthChild.setAuthority(newAuthority); - assertEq(address(mockAuthChild.authority()), address(newAuthority)); - } - - function testTransferOwnershipWithPermissiveAuthority(address deadOwner, address newOwner) public { - if (deadOwner == address(this)) deadOwner = address(0); - - mockAuthChild.setAuthority(new MockAuthority(true)); - mockAuthChild.transferOwnership(deadOwner); - mockAuthChild.transferOwnership(newOwner); - } - - function testSetAuthorityWithPermissiveAuthority(address deadOwner, Authority newAuthority) public { - if (deadOwner == address(this)) deadOwner = address(0); - - mockAuthChild.setAuthority(new MockAuthority(true)); - mockAuthChild.transferOwnership(deadOwner); - mockAuthChild.setAuthority(newAuthority); - } - - function testCallFunctionWithPermissiveAuthority(address deadOwner) public { - if (deadOwner == address(this)) deadOwner = address(0); - - mockAuthChild.setAuthority(new MockAuthority(true)); - mockAuthChild.transferOwnership(deadOwner); - mockAuthChild.updateFlag(); - } - - function testFailTransferOwnershipAsNonOwner(address deadOwner, address newOwner) public { - if (deadOwner == address(this)) deadOwner = address(0); - - mockAuthChild.transferOwnership(deadOwner); - mockAuthChild.transferOwnership(newOwner); - } - - function testFailSetAuthorityAsNonOwner(address deadOwner, Authority newAuthority) public { - if (deadOwner == address(this)) deadOwner = address(0); - - mockAuthChild.transferOwnership(deadOwner); - mockAuthChild.setAuthority(newAuthority); - } - - function testFailCallFunctionAsNonOwner(address deadOwner) public { - if (deadOwner == address(this)) deadOwner = address(0); - - mockAuthChild.transferOwnership(deadOwner); - mockAuthChild.updateFlag(); - } - - function testFailTransferOwnershipWithRestrictiveAuthority(address deadOwner, address newOwner) public { - if (deadOwner == address(this)) deadOwner = address(0); - - mockAuthChild.setAuthority(new MockAuthority(false)); - mockAuthChild.transferOwnership(deadOwner); - mockAuthChild.transferOwnership(newOwner); - } - - function testFailSetAuthorityWithRestrictiveAuthority(address deadOwner, Authority newAuthority) public { - if (deadOwner == address(this)) deadOwner = address(0); - - mockAuthChild.setAuthority(new MockAuthority(false)); - mockAuthChild.transferOwnership(deadOwner); - mockAuthChild.setAuthority(newAuthority); - } - - function testFailCallFunctionWithRestrictiveAuthority(address deadOwner) public { - if (deadOwner == address(this)) deadOwner = address(0); - - mockAuthChild.setAuthority(new MockAuthority(false)); - mockAuthChild.transferOwnership(deadOwner); - mockAuthChild.updateFlag(); - } - - function testFailTransferOwnershipAsOwnerWithOutOfOrderAuthority(address deadOwner) public { - if (deadOwner == address(this)) deadOwner = address(0); - - mockAuthChild.setAuthority(new OutOfOrderAuthority()); - mockAuthChild.transferOwnership(deadOwner); - } -} diff --git a/lib/solmate/src/test/Bytes32AddressLib.t.sol b/lib/solmate/src/test/Bytes32AddressLib.t.sol deleted file mode 100644 index 6c0a3d8..0000000 --- a/lib/solmate/src/test/Bytes32AddressLib.t.sol +++ /dev/null @@ -1,22 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; - -import {Bytes32AddressLib} from "../utils/Bytes32AddressLib.sol"; - -contract Bytes32AddressLibTest is DSTestPlus { - function testFillLast12Bytes() public { - assertEq( - Bytes32AddressLib.fillLast12Bytes(0xfEEDFaCEcaFeBEEFfEEDFACecaFEBeeFfeEdfAce), - 0xfeedfacecafebeeffeedfacecafebeeffeedface000000000000000000000000 - ); - } - - function testFromLast20Bytes() public { - assertEq( - Bytes32AddressLib.fromLast20Bytes(0xfeedfacecafebeeffeedfacecafebeeffeedfacecafebeeffeedfacecafebeef), - 0xCAfeBeefFeedfAceCAFeBEEffEEDfaCecafEBeeF - ); - } -} diff --git a/lib/solmate/src/test/CREATE3.t.sol b/lib/solmate/src/test/CREATE3.t.sol deleted file mode 100644 index 7ee8fc0..0000000 --- a/lib/solmate/src/test/CREATE3.t.sol +++ /dev/null @@ -1,99 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {WETH} from "../tokens/WETH.sol"; -import {DSTestPlus} from "./utils/DSTestPlus.sol"; -import {MockERC20} from "./utils/mocks/MockERC20.sol"; -import {MockAuthChild} from "./utils/mocks/MockAuthChild.sol"; - -import {CREATE3} from "../utils/CREATE3.sol"; - -contract Factory { - function deploy(bytes32 salt) public returns (address deployed) { - deployed = CREATE3.deploy( - salt, - abi.encodePacked(type(MockERC20).creationCode, abi.encode("Mock Token", "MOCK", 18)), - 0 - ); - } -} - -contract CREATE3Test is DSTestPlus { - function testDeployERC20() public { - bytes32 salt = keccak256(bytes("A salt!")); - MockERC20 deployed = MockERC20( - CREATE3.deploy( - salt, - abi.encodePacked(type(MockERC20).creationCode, abi.encode("Mock Token", "MOCK", 18)), - 0 - ) - ); - - assertEq(address(deployed), CREATE3.getDeployed(salt)); - - assertEq(deployed.name(), "Mock Token"); - assertEq(deployed.symbol(), "MOCK"); - assertEq(deployed.decimals(), 18); - } - - function testPredictDeployERC20() public { - bytes32 salt = keccak256(bytes("A salt!")); - Factory factory = new Factory(); - - MockERC20 deployed = MockERC20( - factory.deploy(salt) - ); - - assertEq(address(deployed), CREATE3.getDeployed(salt, address(factory))); - assertTrue(address(deployed) != CREATE3.getDeployed(salt)); - - assertEq(deployed.name(), "Mock Token"); - assertEq(deployed.symbol(), "MOCK"); - assertEq(deployed.decimals(), 18); - } - - function testFailDoubleDeploySameBytecode() public { - bytes32 salt = keccak256(bytes("Salty...")); - - CREATE3.deploy(salt, type(MockAuthChild).creationCode, 0); - CREATE3.deploy(salt, type(MockAuthChild).creationCode, 0); - } - - function testFailDoubleDeployDifferentBytecode() public { - bytes32 salt = keccak256(bytes("and sweet!")); - - CREATE3.deploy(salt, type(WETH).creationCode, 0); - CREATE3.deploy(salt, type(MockAuthChild).creationCode, 0); - } - - function testDeployERC20( - bytes32 salt, - string calldata name, - string calldata symbol, - uint8 decimals - ) public { - MockERC20 deployed = MockERC20( - CREATE3.deploy(salt, abi.encodePacked(type(MockERC20).creationCode, abi.encode(name, symbol, decimals)), 0) - ); - - assertEq(address(deployed), CREATE3.getDeployed(salt)); - - assertEq(deployed.name(), name); - assertEq(deployed.symbol(), symbol); - assertEq(deployed.decimals(), decimals); - } - - function testFailDoubleDeploySameBytecode(bytes32 salt, bytes calldata bytecode) public { - CREATE3.deploy(salt, bytecode, 0); - CREATE3.deploy(salt, bytecode, 0); - } - - function testFailDoubleDeployDifferentBytecode( - bytes32 salt, - bytes calldata bytecode1, - bytes calldata bytecode2 - ) public { - CREATE3.deploy(salt, bytecode1, 0); - CREATE3.deploy(salt, bytecode2, 0); - } -} diff --git a/lib/solmate/src/test/DSTestPlus.t.sol b/lib/solmate/src/test/DSTestPlus.t.sol deleted file mode 100644 index db10860..0000000 --- a/lib/solmate/src/test/DSTestPlus.t.sol +++ /dev/null @@ -1,72 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; - -contract DSTestPlusTest is DSTestPlus { - function testBound() public { - assertEq(bound(0, 69, 69), 69); - assertEq(bound(0, 68, 69), 68); - assertEq(bound(5, 0, 4), 0); - assertEq(bound(9999, 1337, 6666), 6006); - assertEq(bound(0, type(uint256).max - 6, type(uint256).max), type(uint256).max - 6); - assertEq(bound(6, type(uint256).max - 6, type(uint256).max), type(uint256).max); - } - - function testFailBoundMinBiggerThanMax() public { - bound(5, 100, 10); - } - - function testRelApproxEqBothZeroesPasses() public { - assertRelApproxEq(0, 0, 1e18); - assertRelApproxEq(0, 0, 0); - } - - function testBound( - uint256 num, - uint256 min, - uint256 max - ) public { - if (min > max) (min, max) = (max, min); - - uint256 bounded = bound(num, min, max); - - assertGe(bounded, min); - assertLe(bounded, max); - } - - function testFailBoundMinBiggerThanMax( - uint256 num, - uint256 min, - uint256 max - ) public { - if (max == min) { - unchecked { - min++; // Overflow is handled below. - } - } - - if (max > min) (min, max) = (max, min); - - bound(num, min, max); - } - - function testBrutalizeMemory() public brutalizeMemory("FEEDFACECAFEBEEFFEEDFACECAFEBEEF") { - bytes32 scratchSpace1; - bytes32 scratchSpace2; - bytes32 freeMem1; - bytes32 freeMem2; - - assembly { - scratchSpace1 := mload(0) - scratchSpace2 := mload(32) - freeMem1 := mload(mload(0x40)) - freeMem2 := mload(add(mload(0x40), 32)) - } - - assertGt(uint256(freeMem1), 0); - assertGt(uint256(freeMem2), 0); - assertGt(uint256(scratchSpace1), 0); - assertGt(uint256(scratchSpace2), 0); - } -} diff --git a/lib/solmate/src/test/ERC1155.t.sol b/lib/solmate/src/test/ERC1155.t.sol deleted file mode 100644 index 9e32d88..0000000 --- a/lib/solmate/src/test/ERC1155.t.sol +++ /dev/null @@ -1,1777 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; -import {DSInvariantTest} from "./utils/DSInvariantTest.sol"; - -import {MockERC1155} from "./utils/mocks/MockERC1155.sol"; - -import {ERC1155TokenReceiver} from "../tokens/ERC1155.sol"; - -contract ERC1155Recipient is ERC1155TokenReceiver { - address public operator; - address public from; - uint256 public id; - uint256 public amount; - bytes public mintData; - - function onERC1155Received( - address _operator, - address _from, - uint256 _id, - uint256 _amount, - bytes calldata _data - ) public override returns (bytes4) { - operator = _operator; - from = _from; - id = _id; - amount = _amount; - mintData = _data; - - return ERC1155TokenReceiver.onERC1155Received.selector; - } - - address public batchOperator; - address public batchFrom; - uint256[] internal _batchIds; - uint256[] internal _batchAmounts; - bytes public batchData; - - function batchIds() external view returns (uint256[] memory) { - return _batchIds; - } - - function batchAmounts() external view returns (uint256[] memory) { - return _batchAmounts; - } - - function onERC1155BatchReceived( - address _operator, - address _from, - uint256[] calldata _ids, - uint256[] calldata _amounts, - bytes calldata _data - ) external override returns (bytes4) { - batchOperator = _operator; - batchFrom = _from; - _batchIds = _ids; - _batchAmounts = _amounts; - batchData = _data; - - return ERC1155TokenReceiver.onERC1155BatchReceived.selector; - } -} - -contract RevertingERC1155Recipient is ERC1155TokenReceiver { - function onERC1155Received( - address, - address, - uint256, - uint256, - bytes calldata - ) public pure override returns (bytes4) { - revert(string(abi.encodePacked(ERC1155TokenReceiver.onERC1155Received.selector))); - } - - function onERC1155BatchReceived( - address, - address, - uint256[] calldata, - uint256[] calldata, - bytes calldata - ) external pure override returns (bytes4) { - revert(string(abi.encodePacked(ERC1155TokenReceiver.onERC1155BatchReceived.selector))); - } -} - -contract WrongReturnDataERC1155Recipient is ERC1155TokenReceiver { - function onERC1155Received( - address, - address, - uint256, - uint256, - bytes calldata - ) public pure override returns (bytes4) { - return 0xCAFEBEEF; - } - - function onERC1155BatchReceived( - address, - address, - uint256[] calldata, - uint256[] calldata, - bytes calldata - ) external pure override returns (bytes4) { - return 0xCAFEBEEF; - } -} - -contract NonERC1155Recipient {} - -contract ERC1155Test is DSTestPlus, ERC1155TokenReceiver { - MockERC1155 token; - - mapping(address => mapping(uint256 => uint256)) public userMintAmounts; - mapping(address => mapping(uint256 => uint256)) public userTransferOrBurnAmounts; - - function setUp() public { - token = new MockERC1155(); - } - - function testMintToEOA() public { - token.mint(address(0xBEEF), 1337, 1, ""); - - assertEq(token.balanceOf(address(0xBEEF), 1337), 1); - } - - function testMintToERC1155Recipient() public { - ERC1155Recipient to = new ERC1155Recipient(); - - token.mint(address(to), 1337, 1, "testing 123"); - - assertEq(token.balanceOf(address(to), 1337), 1); - - assertEq(to.operator(), address(this)); - assertEq(to.from(), address(0)); - assertEq(to.id(), 1337); - assertBytesEq(to.mintData(), "testing 123"); - } - - function testBatchMintToEOA() public { - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory amounts = new uint256[](5); - amounts[0] = 100; - amounts[1] = 200; - amounts[2] = 300; - amounts[3] = 400; - amounts[4] = 500; - - token.batchMint(address(0xBEEF), ids, amounts, ""); - - assertEq(token.balanceOf(address(0xBEEF), 1337), 100); - assertEq(token.balanceOf(address(0xBEEF), 1338), 200); - assertEq(token.balanceOf(address(0xBEEF), 1339), 300); - assertEq(token.balanceOf(address(0xBEEF), 1340), 400); - assertEq(token.balanceOf(address(0xBEEF), 1341), 500); - } - - function testBatchMintToERC1155Recipient() public { - ERC1155Recipient to = new ERC1155Recipient(); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory amounts = new uint256[](5); - amounts[0] = 100; - amounts[1] = 200; - amounts[2] = 300; - amounts[3] = 400; - amounts[4] = 500; - - token.batchMint(address(to), ids, amounts, "testing 123"); - - assertEq(to.batchOperator(), address(this)); - assertEq(to.batchFrom(), address(0)); - assertUintArrayEq(to.batchIds(), ids); - assertUintArrayEq(to.batchAmounts(), amounts); - assertBytesEq(to.batchData(), "testing 123"); - - assertEq(token.balanceOf(address(to), 1337), 100); - assertEq(token.balanceOf(address(to), 1338), 200); - assertEq(token.balanceOf(address(to), 1339), 300); - assertEq(token.balanceOf(address(to), 1340), 400); - assertEq(token.balanceOf(address(to), 1341), 500); - } - - function testBurn() public { - token.mint(address(0xBEEF), 1337, 100, ""); - - token.burn(address(0xBEEF), 1337, 70); - - assertEq(token.balanceOf(address(0xBEEF), 1337), 30); - } - - function testBatchBurn() public { - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - uint256[] memory burnAmounts = new uint256[](5); - burnAmounts[0] = 50; - burnAmounts[1] = 100; - burnAmounts[2] = 150; - burnAmounts[3] = 200; - burnAmounts[4] = 250; - - token.batchMint(address(0xBEEF), ids, mintAmounts, ""); - - token.batchBurn(address(0xBEEF), ids, burnAmounts); - - assertEq(token.balanceOf(address(0xBEEF), 1337), 50); - assertEq(token.balanceOf(address(0xBEEF), 1338), 100); - assertEq(token.balanceOf(address(0xBEEF), 1339), 150); - assertEq(token.balanceOf(address(0xBEEF), 1340), 200); - assertEq(token.balanceOf(address(0xBEEF), 1341), 250); - } - - function testApproveAll() public { - token.setApprovalForAll(address(0xBEEF), true); - - assertTrue(token.isApprovedForAll(address(this), address(0xBEEF))); - } - - function testSafeTransferFromToEOA() public { - address from = address(0xABCD); - - token.mint(from, 1337, 100, ""); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeTransferFrom(from, address(0xBEEF), 1337, 70, ""); - - assertEq(token.balanceOf(address(0xBEEF), 1337), 70); - assertEq(token.balanceOf(from, 1337), 30); - } - - function testSafeTransferFromToERC1155Recipient() public { - ERC1155Recipient to = new ERC1155Recipient(); - - address from = address(0xABCD); - - token.mint(from, 1337, 100, ""); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeTransferFrom(from, address(to), 1337, 70, "testing 123"); - - assertEq(to.operator(), address(this)); - assertEq(to.from(), from); - assertEq(to.id(), 1337); - assertBytesEq(to.mintData(), "testing 123"); - - assertEq(token.balanceOf(address(to), 1337), 70); - assertEq(token.balanceOf(from, 1337), 30); - } - - function testSafeTransferFromSelf() public { - token.mint(address(this), 1337, 100, ""); - - token.safeTransferFrom(address(this), address(0xBEEF), 1337, 70, ""); - - assertEq(token.balanceOf(address(0xBEEF), 1337), 70); - assertEq(token.balanceOf(address(this), 1337), 30); - } - - function testSafeBatchTransferFromToEOA() public { - address from = address(0xABCD); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - uint256[] memory transferAmounts = new uint256[](5); - transferAmounts[0] = 50; - transferAmounts[1] = 100; - transferAmounts[2] = 150; - transferAmounts[3] = 200; - transferAmounts[4] = 250; - - token.batchMint(from, ids, mintAmounts, ""); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, address(0xBEEF), ids, transferAmounts, ""); - - assertEq(token.balanceOf(from, 1337), 50); - assertEq(token.balanceOf(address(0xBEEF), 1337), 50); - - assertEq(token.balanceOf(from, 1338), 100); - assertEq(token.balanceOf(address(0xBEEF), 1338), 100); - - assertEq(token.balanceOf(from, 1339), 150); - assertEq(token.balanceOf(address(0xBEEF), 1339), 150); - - assertEq(token.balanceOf(from, 1340), 200); - assertEq(token.balanceOf(address(0xBEEF), 1340), 200); - - assertEq(token.balanceOf(from, 1341), 250); - assertEq(token.balanceOf(address(0xBEEF), 1341), 250); - } - - function testSafeBatchTransferFromToERC1155Recipient() public { - address from = address(0xABCD); - - ERC1155Recipient to = new ERC1155Recipient(); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - uint256[] memory transferAmounts = new uint256[](5); - transferAmounts[0] = 50; - transferAmounts[1] = 100; - transferAmounts[2] = 150; - transferAmounts[3] = 200; - transferAmounts[4] = 250; - - token.batchMint(from, ids, mintAmounts, ""); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, address(to), ids, transferAmounts, "testing 123"); - - assertEq(to.batchOperator(), address(this)); - assertEq(to.batchFrom(), from); - assertUintArrayEq(to.batchIds(), ids); - assertUintArrayEq(to.batchAmounts(), transferAmounts); - assertBytesEq(to.batchData(), "testing 123"); - - assertEq(token.balanceOf(from, 1337), 50); - assertEq(token.balanceOf(address(to), 1337), 50); - - assertEq(token.balanceOf(from, 1338), 100); - assertEq(token.balanceOf(address(to), 1338), 100); - - assertEq(token.balanceOf(from, 1339), 150); - assertEq(token.balanceOf(address(to), 1339), 150); - - assertEq(token.balanceOf(from, 1340), 200); - assertEq(token.balanceOf(address(to), 1340), 200); - - assertEq(token.balanceOf(from, 1341), 250); - assertEq(token.balanceOf(address(to), 1341), 250); - } - - function testBatchBalanceOf() public { - address[] memory tos = new address[](5); - tos[0] = address(0xBEEF); - tos[1] = address(0xCAFE); - tos[2] = address(0xFACE); - tos[3] = address(0xDEAD); - tos[4] = address(0xFEED); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - token.mint(address(0xBEEF), 1337, 100, ""); - token.mint(address(0xCAFE), 1338, 200, ""); - token.mint(address(0xFACE), 1339, 300, ""); - token.mint(address(0xDEAD), 1340, 400, ""); - token.mint(address(0xFEED), 1341, 500, ""); - - uint256[] memory balances = token.balanceOfBatch(tos, ids); - - assertEq(balances[0], 100); - assertEq(balances[1], 200); - assertEq(balances[2], 300); - assertEq(balances[3], 400); - assertEq(balances[4], 500); - } - - function testFailMintToZero() public { - token.mint(address(0), 1337, 1, ""); - } - - function testFailMintToNonERC155Recipient() public { - token.mint(address(new NonERC1155Recipient()), 1337, 1, ""); - } - - function testFailMintToRevertingERC155Recipient() public { - token.mint(address(new RevertingERC1155Recipient()), 1337, 1, ""); - } - - function testFailMintToWrongReturnDataERC155Recipient() public { - token.mint(address(new RevertingERC1155Recipient()), 1337, 1, ""); - } - - function testFailBurnInsufficientBalance() public { - token.mint(address(0xBEEF), 1337, 70, ""); - token.burn(address(0xBEEF), 1337, 100); - } - - function testFailSafeTransferFromInsufficientBalance() public { - address from = address(0xABCD); - - token.mint(from, 1337, 70, ""); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeTransferFrom(from, address(0xBEEF), 1337, 100, ""); - } - - function testFailSafeTransferFromSelfInsufficientBalance() public { - token.mint(address(this), 1337, 70, ""); - token.safeTransferFrom(address(this), address(0xBEEF), 1337, 100, ""); - } - - function testFailSafeTransferFromToZero() public { - token.mint(address(this), 1337, 100, ""); - token.safeTransferFrom(address(this), address(0), 1337, 70, ""); - } - - function testFailSafeTransferFromToNonERC155Recipient() public { - token.mint(address(this), 1337, 100, ""); - token.safeTransferFrom(address(this), address(new NonERC1155Recipient()), 1337, 70, ""); - } - - function testFailSafeTransferFromToRevertingERC1155Recipient() public { - token.mint(address(this), 1337, 100, ""); - token.safeTransferFrom(address(this), address(new RevertingERC1155Recipient()), 1337, 70, ""); - } - - function testFailSafeTransferFromToWrongReturnDataERC1155Recipient() public { - token.mint(address(this), 1337, 100, ""); - token.safeTransferFrom(address(this), address(new WrongReturnDataERC1155Recipient()), 1337, 70, ""); - } - - function testFailSafeBatchTransferInsufficientBalance() public { - address from = address(0xABCD); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - - mintAmounts[0] = 50; - mintAmounts[1] = 100; - mintAmounts[2] = 150; - mintAmounts[3] = 200; - mintAmounts[4] = 250; - - uint256[] memory transferAmounts = new uint256[](5); - transferAmounts[0] = 100; - transferAmounts[1] = 200; - transferAmounts[2] = 300; - transferAmounts[3] = 400; - transferAmounts[4] = 500; - - token.batchMint(from, ids, mintAmounts, ""); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, address(0xBEEF), ids, transferAmounts, ""); - } - - function testFailSafeBatchTransferFromToZero() public { - address from = address(0xABCD); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - uint256[] memory transferAmounts = new uint256[](5); - transferAmounts[0] = 50; - transferAmounts[1] = 100; - transferAmounts[2] = 150; - transferAmounts[3] = 200; - transferAmounts[4] = 250; - - token.batchMint(from, ids, mintAmounts, ""); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, address(0), ids, transferAmounts, ""); - } - - function testFailSafeBatchTransferFromToNonERC1155Recipient() public { - address from = address(0xABCD); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - uint256[] memory transferAmounts = new uint256[](5); - transferAmounts[0] = 50; - transferAmounts[1] = 100; - transferAmounts[2] = 150; - transferAmounts[3] = 200; - transferAmounts[4] = 250; - - token.batchMint(from, ids, mintAmounts, ""); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, address(new NonERC1155Recipient()), ids, transferAmounts, ""); - } - - function testFailSafeBatchTransferFromToRevertingERC1155Recipient() public { - address from = address(0xABCD); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - uint256[] memory transferAmounts = new uint256[](5); - transferAmounts[0] = 50; - transferAmounts[1] = 100; - transferAmounts[2] = 150; - transferAmounts[3] = 200; - transferAmounts[4] = 250; - - token.batchMint(from, ids, mintAmounts, ""); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, address(new RevertingERC1155Recipient()), ids, transferAmounts, ""); - } - - function testFailSafeBatchTransferFromToWrongReturnDataERC1155Recipient() public { - address from = address(0xABCD); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - uint256[] memory transferAmounts = new uint256[](5); - transferAmounts[0] = 50; - transferAmounts[1] = 100; - transferAmounts[2] = 150; - transferAmounts[3] = 200; - transferAmounts[4] = 250; - - token.batchMint(from, ids, mintAmounts, ""); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, address(new WrongReturnDataERC1155Recipient()), ids, transferAmounts, ""); - } - - function testFailSafeBatchTransferFromWithArrayLengthMismatch() public { - address from = address(0xABCD); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - uint256[] memory transferAmounts = new uint256[](4); - transferAmounts[0] = 50; - transferAmounts[1] = 100; - transferAmounts[2] = 150; - transferAmounts[3] = 200; - - token.batchMint(from, ids, mintAmounts, ""); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, address(0xBEEF), ids, transferAmounts, ""); - } - - function testFailBatchMintToZero() public { - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - token.batchMint(address(0), ids, mintAmounts, ""); - } - - function testFailBatchMintToNonERC1155Recipient() public { - NonERC1155Recipient to = new NonERC1155Recipient(); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - token.batchMint(address(to), ids, mintAmounts, ""); - } - - function testFailBatchMintToRevertingERC1155Recipient() public { - RevertingERC1155Recipient to = new RevertingERC1155Recipient(); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - token.batchMint(address(to), ids, mintAmounts, ""); - } - - function testFailBatchMintToWrongReturnDataERC1155Recipient() public { - WrongReturnDataERC1155Recipient to = new WrongReturnDataERC1155Recipient(); - - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - token.batchMint(address(to), ids, mintAmounts, ""); - } - - function testFailBatchMintWithArrayMismatch() public { - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory amounts = new uint256[](4); - amounts[0] = 100; - amounts[1] = 200; - amounts[2] = 300; - amounts[3] = 400; - - token.batchMint(address(0xBEEF), ids, amounts, ""); - } - - function testFailBatchBurnInsufficientBalance() public { - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 50; - mintAmounts[1] = 100; - mintAmounts[2] = 150; - mintAmounts[3] = 200; - mintAmounts[4] = 250; - - uint256[] memory burnAmounts = new uint256[](5); - burnAmounts[0] = 100; - burnAmounts[1] = 200; - burnAmounts[2] = 300; - burnAmounts[3] = 400; - burnAmounts[4] = 500; - - token.batchMint(address(0xBEEF), ids, mintAmounts, ""); - - token.batchBurn(address(0xBEEF), ids, burnAmounts); - } - - function testFailBatchBurnWithArrayLengthMismatch() public { - uint256[] memory ids = new uint256[](5); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - ids[4] = 1341; - - uint256[] memory mintAmounts = new uint256[](5); - mintAmounts[0] = 100; - mintAmounts[1] = 200; - mintAmounts[2] = 300; - mintAmounts[3] = 400; - mintAmounts[4] = 500; - - uint256[] memory burnAmounts = new uint256[](4); - burnAmounts[0] = 50; - burnAmounts[1] = 100; - burnAmounts[2] = 150; - burnAmounts[3] = 200; - - token.batchMint(address(0xBEEF), ids, mintAmounts, ""); - - token.batchBurn(address(0xBEEF), ids, burnAmounts); - } - - function testFailBalanceOfBatchWithArrayMismatch() public view { - address[] memory tos = new address[](5); - tos[0] = address(0xBEEF); - tos[1] = address(0xCAFE); - tos[2] = address(0xFACE); - tos[3] = address(0xDEAD); - tos[4] = address(0xFEED); - - uint256[] memory ids = new uint256[](4); - ids[0] = 1337; - ids[1] = 1338; - ids[2] = 1339; - ids[3] = 1340; - - token.balanceOfBatch(tos, ids); - } - - function testMintToEOA( - address to, - uint256 id, - uint256 amount, - bytes memory mintData - ) public { - if (to == address(0)) to = address(0xBEEF); - - if (uint256(uint160(to)) <= 18 || to.code.length > 0) return; - - token.mint(to, id, amount, mintData); - - assertEq(token.balanceOf(to, id), amount); - } - - function testMintToERC1155Recipient( - uint256 id, - uint256 amount, - bytes memory mintData - ) public { - ERC1155Recipient to = new ERC1155Recipient(); - - token.mint(address(to), id, amount, mintData); - - assertEq(token.balanceOf(address(to), id), amount); - - assertEq(to.operator(), address(this)); - assertEq(to.from(), address(0)); - assertEq(to.id(), id); - assertBytesEq(to.mintData(), mintData); - } - - function testBatchMintToEOA( - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory mintData - ) public { - if (to == address(0)) to = address(0xBEEF); - - if (uint256(uint160(to)) <= 18 || to.code.length > 0) return; - - uint256 minLength = min2(ids.length, amounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[to][id]; - - uint256 mintAmount = bound(amounts[i], 0, remainingMintAmountForId); - - normalizedIds[i] = id; - normalizedAmounts[i] = mintAmount; - - userMintAmounts[to][id] += mintAmount; - } - - token.batchMint(to, normalizedIds, normalizedAmounts, mintData); - - for (uint256 i = 0; i < normalizedIds.length; i++) { - uint256 id = normalizedIds[i]; - - assertEq(token.balanceOf(to, id), userMintAmounts[to][id]); - } - } - - function testBatchMintToERC1155Recipient( - uint256[] memory ids, - uint256[] memory amounts, - bytes memory mintData - ) public { - ERC1155Recipient to = new ERC1155Recipient(); - - uint256 minLength = min2(ids.length, amounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[address(to)][id]; - - uint256 mintAmount = bound(amounts[i], 0, remainingMintAmountForId); - - normalizedIds[i] = id; - normalizedAmounts[i] = mintAmount; - - userMintAmounts[address(to)][id] += mintAmount; - } - - token.batchMint(address(to), normalizedIds, normalizedAmounts, mintData); - - assertEq(to.batchOperator(), address(this)); - assertEq(to.batchFrom(), address(0)); - assertUintArrayEq(to.batchIds(), normalizedIds); - assertUintArrayEq(to.batchAmounts(), normalizedAmounts); - assertBytesEq(to.batchData(), mintData); - - for (uint256 i = 0; i < normalizedIds.length; i++) { - uint256 id = normalizedIds[i]; - - assertEq(token.balanceOf(address(to), id), userMintAmounts[address(to)][id]); - } - } - - function testBurn( - address to, - uint256 id, - uint256 mintAmount, - bytes memory mintData, - uint256 burnAmount - ) public { - if (to == address(0)) to = address(0xBEEF); - - if (uint256(uint160(to)) <= 18 || to.code.length > 0) return; - - burnAmount = bound(burnAmount, 0, mintAmount); - - token.mint(to, id, mintAmount, mintData); - - token.burn(to, id, burnAmount); - - assertEq(token.balanceOf(address(to), id), mintAmount - burnAmount); - } - - function testBatchBurn( - address to, - uint256[] memory ids, - uint256[] memory mintAmounts, - uint256[] memory burnAmounts, - bytes memory mintData - ) public { - if (to == address(0)) to = address(0xBEEF); - - if (uint256(uint160(to)) <= 18 || to.code.length > 0) return; - - uint256 minLength = min3(ids.length, mintAmounts.length, burnAmounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedMintAmounts = new uint256[](minLength); - uint256[] memory normalizedBurnAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[address(to)][id]; - - normalizedIds[i] = id; - normalizedMintAmounts[i] = bound(mintAmounts[i], 0, remainingMintAmountForId); - normalizedBurnAmounts[i] = bound(burnAmounts[i], 0, normalizedMintAmounts[i]); - - userMintAmounts[address(to)][id] += normalizedMintAmounts[i]; - userTransferOrBurnAmounts[address(to)][id] += normalizedBurnAmounts[i]; - } - - token.batchMint(to, normalizedIds, normalizedMintAmounts, mintData); - - token.batchBurn(to, normalizedIds, normalizedBurnAmounts); - - for (uint256 i = 0; i < normalizedIds.length; i++) { - uint256 id = normalizedIds[i]; - - assertEq(token.balanceOf(to, id), userMintAmounts[to][id] - userTransferOrBurnAmounts[to][id]); - } - } - - function testApproveAll(address to, bool approved) public { - token.setApprovalForAll(to, approved); - - assertBoolEq(token.isApprovedForAll(address(this), to), approved); - } - - function testSafeTransferFromToEOA( - uint256 id, - uint256 mintAmount, - bytes memory mintData, - uint256 transferAmount, - address to, - bytes memory transferData - ) public { - if (to == address(0)) to = address(0xBEEF); - - if (uint256(uint160(to)) <= 18 || to.code.length > 0) return; - - transferAmount = bound(transferAmount, 0, mintAmount); - - address from = address(0xABCD); - - token.mint(from, id, mintAmount, mintData); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeTransferFrom(from, to, id, transferAmount, transferData); - - if (to == from) { - assertEq(token.balanceOf(to, id), mintAmount); - } else { - assertEq(token.balanceOf(to, id), transferAmount); - assertEq(token.balanceOf(from, id), mintAmount - transferAmount); - } - } - - function testSafeTransferFromToERC1155Recipient( - uint256 id, - uint256 mintAmount, - bytes memory mintData, - uint256 transferAmount, - bytes memory transferData - ) public { - ERC1155Recipient to = new ERC1155Recipient(); - - address from = address(0xABCD); - - transferAmount = bound(transferAmount, 0, mintAmount); - - token.mint(from, id, mintAmount, mintData); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeTransferFrom(from, address(to), id, transferAmount, transferData); - - assertEq(to.operator(), address(this)); - assertEq(to.from(), from); - assertEq(to.id(), id); - assertBytesEq(to.mintData(), transferData); - - assertEq(token.balanceOf(address(to), id), transferAmount); - assertEq(token.balanceOf(from, id), mintAmount - transferAmount); - } - - function testSafeTransferFromSelf( - uint256 id, - uint256 mintAmount, - bytes memory mintData, - uint256 transferAmount, - address to, - bytes memory transferData - ) public { - if (to == address(0)) to = address(0xBEEF); - - if (uint256(uint160(to)) <= 18 || to.code.length > 0) return; - - transferAmount = bound(transferAmount, 0, mintAmount); - - token.mint(address(this), id, mintAmount, mintData); - - token.safeTransferFrom(address(this), to, id, transferAmount, transferData); - - assertEq(token.balanceOf(to, id), transferAmount); - assertEq(token.balanceOf(address(this), id), mintAmount - transferAmount); - } - - function testSafeBatchTransferFromToEOA( - address to, - uint256[] memory ids, - uint256[] memory mintAmounts, - uint256[] memory transferAmounts, - bytes memory mintData, - bytes memory transferData - ) public { - if (to == address(0)) to = address(0xBEEF); - - if (uint256(uint160(to)) <= 18 || to.code.length > 0) return; - - address from = address(0xABCD); - - uint256 minLength = min3(ids.length, mintAmounts.length, transferAmounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedMintAmounts = new uint256[](minLength); - uint256[] memory normalizedTransferAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[from][id]; - - uint256 mintAmount = bound(mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = bound(transferAmounts[i], 0, mintAmount); - - normalizedIds[i] = id; - normalizedMintAmounts[i] = mintAmount; - normalizedTransferAmounts[i] = transferAmount; - - userMintAmounts[from][id] += mintAmount; - userTransferOrBurnAmounts[from][id] += transferAmount; - } - - token.batchMint(from, normalizedIds, normalizedMintAmounts, mintData); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, to, normalizedIds, normalizedTransferAmounts, transferData); - - for (uint256 i = 0; i < normalizedIds.length; i++) { - uint256 id = normalizedIds[i]; - - assertEq(token.balanceOf(address(to), id), userTransferOrBurnAmounts[from][id]); - assertEq(token.balanceOf(from, id), userMintAmounts[from][id] - userTransferOrBurnAmounts[from][id]); - } - } - - function testSafeBatchTransferFromToERC1155Recipient( - uint256[] memory ids, - uint256[] memory mintAmounts, - uint256[] memory transferAmounts, - bytes memory mintData, - bytes memory transferData - ) public { - address from = address(0xABCD); - - ERC1155Recipient to = new ERC1155Recipient(); - - uint256 minLength = min3(ids.length, mintAmounts.length, transferAmounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedMintAmounts = new uint256[](minLength); - uint256[] memory normalizedTransferAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[from][id]; - - uint256 mintAmount = bound(mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = bound(transferAmounts[i], 0, mintAmount); - - normalizedIds[i] = id; - normalizedMintAmounts[i] = mintAmount; - normalizedTransferAmounts[i] = transferAmount; - - userMintAmounts[from][id] += mintAmount; - userTransferOrBurnAmounts[from][id] += transferAmount; - } - - token.batchMint(from, normalizedIds, normalizedMintAmounts, mintData); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, address(to), normalizedIds, normalizedTransferAmounts, transferData); - - assertEq(to.batchOperator(), address(this)); - assertEq(to.batchFrom(), from); - assertUintArrayEq(to.batchIds(), normalizedIds); - assertUintArrayEq(to.batchAmounts(), normalizedTransferAmounts); - assertBytesEq(to.batchData(), transferData); - - for (uint256 i = 0; i < normalizedIds.length; i++) { - uint256 id = normalizedIds[i]; - uint256 transferAmount = userTransferOrBurnAmounts[from][id]; - - assertEq(token.balanceOf(address(to), id), transferAmount); - assertEq(token.balanceOf(from, id), userMintAmounts[from][id] - transferAmount); - } - } - - function testBatchBalanceOf( - address[] memory tos, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory mintData - ) public { - uint256 minLength = min3(tos.length, ids.length, amounts.length); - - address[] memory normalizedTos = new address[](minLength); - uint256[] memory normalizedIds = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - address to = tos[i] == address(0) || tos[i].code.length > 0 ? address(0xBEEF) : tos[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[to][id]; - - normalizedTos[i] = to; - normalizedIds[i] = id; - - uint256 mintAmount = bound(amounts[i], 0, remainingMintAmountForId); - - token.mint(to, id, mintAmount, mintData); - - userMintAmounts[to][id] += mintAmount; - } - - uint256[] memory balances = token.balanceOfBatch(normalizedTos, normalizedIds); - - for (uint256 i = 0; i < normalizedTos.length; i++) { - assertEq(balances[i], token.balanceOf(normalizedTos[i], normalizedIds[i])); - } - } - - function testFailMintToZero( - uint256 id, - uint256 amount, - bytes memory data - ) public { - token.mint(address(0), id, amount, data); - } - - function testFailMintToNonERC155Recipient( - uint256 id, - uint256 mintAmount, - bytes memory mintData - ) public { - token.mint(address(new NonERC1155Recipient()), id, mintAmount, mintData); - } - - function testFailMintToRevertingERC155Recipient( - uint256 id, - uint256 mintAmount, - bytes memory mintData - ) public { - token.mint(address(new RevertingERC1155Recipient()), id, mintAmount, mintData); - } - - function testFailMintToWrongReturnDataERC155Recipient( - uint256 id, - uint256 mintAmount, - bytes memory mintData - ) public { - token.mint(address(new RevertingERC1155Recipient()), id, mintAmount, mintData); - } - - function testFailBurnInsufficientBalance( - address to, - uint256 id, - uint256 mintAmount, - uint256 burnAmount, - bytes memory mintData - ) public { - burnAmount = bound(burnAmount, mintAmount + 1, type(uint256).max); - - token.mint(to, id, mintAmount, mintData); - token.burn(to, id, burnAmount); - } - - function testFailSafeTransferFromInsufficientBalance( - address to, - uint256 id, - uint256 mintAmount, - uint256 transferAmount, - bytes memory mintData, - bytes memory transferData - ) public { - address from = address(0xABCD); - - transferAmount = bound(transferAmount, mintAmount + 1, type(uint256).max); - - token.mint(from, id, mintAmount, mintData); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeTransferFrom(from, to, id, transferAmount, transferData); - } - - function testFailSafeTransferFromSelfInsufficientBalance( - address to, - uint256 id, - uint256 mintAmount, - uint256 transferAmount, - bytes memory mintData, - bytes memory transferData - ) public { - transferAmount = bound(transferAmount, mintAmount + 1, type(uint256).max); - - token.mint(address(this), id, mintAmount, mintData); - token.safeTransferFrom(address(this), to, id, transferAmount, transferData); - } - - function testFailSafeTransferFromToZero( - uint256 id, - uint256 mintAmount, - uint256 transferAmount, - bytes memory mintData, - bytes memory transferData - ) public { - transferAmount = bound(transferAmount, 0, mintAmount); - - token.mint(address(this), id, mintAmount, mintData); - token.safeTransferFrom(address(this), address(0), id, transferAmount, transferData); - } - - function testFailSafeTransferFromToNonERC155Recipient( - uint256 id, - uint256 mintAmount, - uint256 transferAmount, - bytes memory mintData, - bytes memory transferData - ) public { - transferAmount = bound(transferAmount, 0, mintAmount); - - token.mint(address(this), id, mintAmount, mintData); - token.safeTransferFrom(address(this), address(new NonERC1155Recipient()), id, transferAmount, transferData); - } - - function testFailSafeTransferFromToRevertingERC1155Recipient( - uint256 id, - uint256 mintAmount, - uint256 transferAmount, - bytes memory mintData, - bytes memory transferData - ) public { - transferAmount = bound(transferAmount, 0, mintAmount); - - token.mint(address(this), id, mintAmount, mintData); - token.safeTransferFrom( - address(this), - address(new RevertingERC1155Recipient()), - id, - transferAmount, - transferData - ); - } - - function testFailSafeTransferFromToWrongReturnDataERC1155Recipient( - uint256 id, - uint256 mintAmount, - uint256 transferAmount, - bytes memory mintData, - bytes memory transferData - ) public { - transferAmount = bound(transferAmount, 0, mintAmount); - - token.mint(address(this), id, mintAmount, mintData); - token.safeTransferFrom( - address(this), - address(new WrongReturnDataERC1155Recipient()), - id, - transferAmount, - transferData - ); - } - - function testFailSafeBatchTransferInsufficientBalance( - address to, - uint256[] memory ids, - uint256[] memory mintAmounts, - uint256[] memory transferAmounts, - bytes memory mintData, - bytes memory transferData - ) public { - address from = address(0xABCD); - - uint256 minLength = min3(ids.length, mintAmounts.length, transferAmounts.length); - - if (minLength == 0) revert(); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedMintAmounts = new uint256[](minLength); - uint256[] memory normalizedTransferAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[from][id]; - - uint256 mintAmount = bound(mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = bound(transferAmounts[i], mintAmount + 1, type(uint256).max); - - normalizedIds[i] = id; - normalizedMintAmounts[i] = mintAmount; - normalizedTransferAmounts[i] = transferAmount; - - userMintAmounts[from][id] += mintAmount; - } - - token.batchMint(from, normalizedIds, normalizedMintAmounts, mintData); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, to, normalizedIds, normalizedTransferAmounts, transferData); - } - - function testFailSafeBatchTransferFromToZero( - uint256[] memory ids, - uint256[] memory mintAmounts, - uint256[] memory transferAmounts, - bytes memory mintData, - bytes memory transferData - ) public { - address from = address(0xABCD); - - uint256 minLength = min3(ids.length, mintAmounts.length, transferAmounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedMintAmounts = new uint256[](minLength); - uint256[] memory normalizedTransferAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[from][id]; - - uint256 mintAmount = bound(mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = bound(transferAmounts[i], 0, mintAmount); - - normalizedIds[i] = id; - normalizedMintAmounts[i] = mintAmount; - normalizedTransferAmounts[i] = transferAmount; - - userMintAmounts[from][id] += mintAmount; - } - - token.batchMint(from, normalizedIds, normalizedMintAmounts, mintData); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, address(0), normalizedIds, normalizedTransferAmounts, transferData); - } - - function testFailSafeBatchTransferFromToNonERC1155Recipient( - uint256[] memory ids, - uint256[] memory mintAmounts, - uint256[] memory transferAmounts, - bytes memory mintData, - bytes memory transferData - ) public { - address from = address(0xABCD); - - uint256 minLength = min3(ids.length, mintAmounts.length, transferAmounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedMintAmounts = new uint256[](minLength); - uint256[] memory normalizedTransferAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[from][id]; - - uint256 mintAmount = bound(mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = bound(transferAmounts[i], 0, mintAmount); - - normalizedIds[i] = id; - normalizedMintAmounts[i] = mintAmount; - normalizedTransferAmounts[i] = transferAmount; - - userMintAmounts[from][id] += mintAmount; - } - - token.batchMint(from, normalizedIds, normalizedMintAmounts, mintData); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom( - from, - address(new NonERC1155Recipient()), - normalizedIds, - normalizedTransferAmounts, - transferData - ); - } - - function testFailSafeBatchTransferFromToRevertingERC1155Recipient( - uint256[] memory ids, - uint256[] memory mintAmounts, - uint256[] memory transferAmounts, - bytes memory mintData, - bytes memory transferData - ) public { - address from = address(0xABCD); - - uint256 minLength = min3(ids.length, mintAmounts.length, transferAmounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedMintAmounts = new uint256[](minLength); - uint256[] memory normalizedTransferAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[from][id]; - - uint256 mintAmount = bound(mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = bound(transferAmounts[i], 0, mintAmount); - - normalizedIds[i] = id; - normalizedMintAmounts[i] = mintAmount; - normalizedTransferAmounts[i] = transferAmount; - - userMintAmounts[from][id] += mintAmount; - } - - token.batchMint(from, normalizedIds, normalizedMintAmounts, mintData); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom( - from, - address(new RevertingERC1155Recipient()), - normalizedIds, - normalizedTransferAmounts, - transferData - ); - } - - function testFailSafeBatchTransferFromToWrongReturnDataERC1155Recipient( - uint256[] memory ids, - uint256[] memory mintAmounts, - uint256[] memory transferAmounts, - bytes memory mintData, - bytes memory transferData - ) public { - address from = address(0xABCD); - - uint256 minLength = min3(ids.length, mintAmounts.length, transferAmounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedMintAmounts = new uint256[](minLength); - uint256[] memory normalizedTransferAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[from][id]; - - uint256 mintAmount = bound(mintAmounts[i], 0, remainingMintAmountForId); - uint256 transferAmount = bound(transferAmounts[i], 0, mintAmount); - - normalizedIds[i] = id; - normalizedMintAmounts[i] = mintAmount; - normalizedTransferAmounts[i] = transferAmount; - - userMintAmounts[from][id] += mintAmount; - } - - token.batchMint(from, normalizedIds, normalizedMintAmounts, mintData); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom( - from, - address(new WrongReturnDataERC1155Recipient()), - normalizedIds, - normalizedTransferAmounts, - transferData - ); - } - - function testFailSafeBatchTransferFromWithArrayLengthMismatch( - address to, - uint256[] memory ids, - uint256[] memory mintAmounts, - uint256[] memory transferAmounts, - bytes memory mintData, - bytes memory transferData - ) public { - address from = address(0xABCD); - - if (ids.length == transferAmounts.length) revert(); - - token.batchMint(from, ids, mintAmounts, mintData); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeBatchTransferFrom(from, to, ids, transferAmounts, transferData); - } - - function testFailBatchMintToZero( - uint256[] memory ids, - uint256[] memory amounts, - bytes memory mintData - ) public { - uint256 minLength = min2(ids.length, amounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[address(0)][id]; - - uint256 mintAmount = bound(amounts[i], 0, remainingMintAmountForId); - - normalizedIds[i] = id; - normalizedAmounts[i] = mintAmount; - - userMintAmounts[address(0)][id] += mintAmount; - } - - token.batchMint(address(0), normalizedIds, normalizedAmounts, mintData); - } - - function testFailBatchMintToNonERC1155Recipient( - uint256[] memory ids, - uint256[] memory amounts, - bytes memory mintData - ) public { - NonERC1155Recipient to = new NonERC1155Recipient(); - - uint256 minLength = min2(ids.length, amounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[address(to)][id]; - - uint256 mintAmount = bound(amounts[i], 0, remainingMintAmountForId); - - normalizedIds[i] = id; - normalizedAmounts[i] = mintAmount; - - userMintAmounts[address(to)][id] += mintAmount; - } - - token.batchMint(address(to), normalizedIds, normalizedAmounts, mintData); - } - - function testFailBatchMintToRevertingERC1155Recipient( - uint256[] memory ids, - uint256[] memory amounts, - bytes memory mintData - ) public { - RevertingERC1155Recipient to = new RevertingERC1155Recipient(); - - uint256 minLength = min2(ids.length, amounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[address(to)][id]; - - uint256 mintAmount = bound(amounts[i], 0, remainingMintAmountForId); - - normalizedIds[i] = id; - normalizedAmounts[i] = mintAmount; - - userMintAmounts[address(to)][id] += mintAmount; - } - - token.batchMint(address(to), normalizedIds, normalizedAmounts, mintData); - } - - function testFailBatchMintToWrongReturnDataERC1155Recipient( - uint256[] memory ids, - uint256[] memory amounts, - bytes memory mintData - ) public { - WrongReturnDataERC1155Recipient to = new WrongReturnDataERC1155Recipient(); - - uint256 minLength = min2(ids.length, amounts.length); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[address(to)][id]; - - uint256 mintAmount = bound(amounts[i], 0, remainingMintAmountForId); - - normalizedIds[i] = id; - normalizedAmounts[i] = mintAmount; - - userMintAmounts[address(to)][id] += mintAmount; - } - - token.batchMint(address(to), normalizedIds, normalizedAmounts, mintData); - } - - function testFailBatchMintWithArrayMismatch( - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory mintData - ) public { - if (ids.length == amounts.length) revert(); - - token.batchMint(address(to), ids, amounts, mintData); - } - - function testFailBatchBurnInsufficientBalance( - address to, - uint256[] memory ids, - uint256[] memory mintAmounts, - uint256[] memory burnAmounts, - bytes memory mintData - ) public { - uint256 minLength = min3(ids.length, mintAmounts.length, burnAmounts.length); - - if (minLength == 0) revert(); - - uint256[] memory normalizedIds = new uint256[](minLength); - uint256[] memory normalizedMintAmounts = new uint256[](minLength); - uint256[] memory normalizedBurnAmounts = new uint256[](minLength); - - for (uint256 i = 0; i < minLength; i++) { - uint256 id = ids[i]; - - uint256 remainingMintAmountForId = type(uint256).max - userMintAmounts[to][id]; - - normalizedIds[i] = id; - normalizedMintAmounts[i] = bound(mintAmounts[i], 0, remainingMintAmountForId); - normalizedBurnAmounts[i] = bound(burnAmounts[i], normalizedMintAmounts[i] + 1, type(uint256).max); - - userMintAmounts[to][id] += normalizedMintAmounts[i]; - } - - token.batchMint(to, normalizedIds, normalizedMintAmounts, mintData); - - token.batchBurn(to, normalizedIds, normalizedBurnAmounts); - } - - function testFailBatchBurnWithArrayLengthMismatch( - address to, - uint256[] memory ids, - uint256[] memory mintAmounts, - uint256[] memory burnAmounts, - bytes memory mintData - ) public { - if (ids.length == burnAmounts.length) revert(); - - token.batchMint(to, ids, mintAmounts, mintData); - - token.batchBurn(to, ids, burnAmounts); - } - - function testFailBalanceOfBatchWithArrayMismatch(address[] memory tos, uint256[] memory ids) public view { - if (tos.length == ids.length) revert(); - - token.balanceOfBatch(tos, ids); - } -} diff --git a/lib/solmate/src/test/ERC20.t.sol b/lib/solmate/src/test/ERC20.t.sol deleted file mode 100644 index 1506d8c..0000000 --- a/lib/solmate/src/test/ERC20.t.sol +++ /dev/null @@ -1,531 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; -import {DSInvariantTest} from "./utils/DSInvariantTest.sol"; - -import {MockERC20} from "./utils/mocks/MockERC20.sol"; - -contract ERC20Test is DSTestPlus { - MockERC20 token; - - bytes32 constant PERMIT_TYPEHASH = - keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)"); - - function setUp() public { - token = new MockERC20("Token", "TKN", 18); - } - - function invariantMetadata() public { - assertEq(token.name(), "Token"); - assertEq(token.symbol(), "TKN"); - assertEq(token.decimals(), 18); - } - - function testMint() public { - token.mint(address(0xBEEF), 1e18); - - assertEq(token.totalSupply(), 1e18); - assertEq(token.balanceOf(address(0xBEEF)), 1e18); - } - - function testBurn() public { - token.mint(address(0xBEEF), 1e18); - token.burn(address(0xBEEF), 0.9e18); - - assertEq(token.totalSupply(), 1e18 - 0.9e18); - assertEq(token.balanceOf(address(0xBEEF)), 0.1e18); - } - - function testApprove() public { - assertTrue(token.approve(address(0xBEEF), 1e18)); - - assertEq(token.allowance(address(this), address(0xBEEF)), 1e18); - } - - function testTransfer() public { - token.mint(address(this), 1e18); - - assertTrue(token.transfer(address(0xBEEF), 1e18)); - assertEq(token.totalSupply(), 1e18); - - assertEq(token.balanceOf(address(this)), 0); - assertEq(token.balanceOf(address(0xBEEF)), 1e18); - } - - function testTransferFrom() public { - address from = address(0xABCD); - - token.mint(from, 1e18); - - hevm.prank(from); - token.approve(address(this), 1e18); - - assertTrue(token.transferFrom(from, address(0xBEEF), 1e18)); - assertEq(token.totalSupply(), 1e18); - - assertEq(token.allowance(from, address(this)), 0); - - assertEq(token.balanceOf(from), 0); - assertEq(token.balanceOf(address(0xBEEF)), 1e18); - } - - function testInfiniteApproveTransferFrom() public { - address from = address(0xABCD); - - token.mint(from, 1e18); - - hevm.prank(from); - token.approve(address(this), type(uint256).max); - - assertTrue(token.transferFrom(from, address(0xBEEF), 1e18)); - assertEq(token.totalSupply(), 1e18); - - assertEq(token.allowance(from, address(this)), type(uint256).max); - - assertEq(token.balanceOf(from), 0); - assertEq(token.balanceOf(address(0xBEEF)), 1e18); - } - - function testPermit() public { - uint256 privateKey = 0xBEEF; - address owner = hevm.addr(privateKey); - - (uint8 v, bytes32 r, bytes32 s) = hevm.sign( - privateKey, - keccak256( - abi.encodePacked( - "\x19\x01", - token.DOMAIN_SEPARATOR(), - keccak256(abi.encode(PERMIT_TYPEHASH, owner, address(0xCAFE), 1e18, 0, block.timestamp)) - ) - ) - ); - - token.permit(owner, address(0xCAFE), 1e18, block.timestamp, v, r, s); - - assertEq(token.allowance(owner, address(0xCAFE)), 1e18); - assertEq(token.nonces(owner), 1); - } - - function testFailTransferInsufficientBalance() public { - token.mint(address(this), 0.9e18); - token.transfer(address(0xBEEF), 1e18); - } - - function testFailTransferFromInsufficientAllowance() public { - address from = address(0xABCD); - - token.mint(from, 1e18); - - hevm.prank(from); - token.approve(address(this), 0.9e18); - - token.transferFrom(from, address(0xBEEF), 1e18); - } - - function testFailTransferFromInsufficientBalance() public { - address from = address(0xABCD); - - token.mint(from, 0.9e18); - - hevm.prank(from); - token.approve(address(this), 1e18); - - token.transferFrom(from, address(0xBEEF), 1e18); - } - - function testFailPermitBadNonce() public { - uint256 privateKey = 0xBEEF; - address owner = hevm.addr(privateKey); - - (uint8 v, bytes32 r, bytes32 s) = hevm.sign( - privateKey, - keccak256( - abi.encodePacked( - "\x19\x01", - token.DOMAIN_SEPARATOR(), - keccak256(abi.encode(PERMIT_TYPEHASH, owner, address(0xCAFE), 1e18, 1, block.timestamp)) - ) - ) - ); - - token.permit(owner, address(0xCAFE), 1e18, block.timestamp, v, r, s); - } - - function testFailPermitBadDeadline() public { - uint256 privateKey = 0xBEEF; - address owner = hevm.addr(privateKey); - - (uint8 v, bytes32 r, bytes32 s) = hevm.sign( - privateKey, - keccak256( - abi.encodePacked( - "\x19\x01", - token.DOMAIN_SEPARATOR(), - keccak256(abi.encode(PERMIT_TYPEHASH, owner, address(0xCAFE), 1e18, 0, block.timestamp)) - ) - ) - ); - - token.permit(owner, address(0xCAFE), 1e18, block.timestamp + 1, v, r, s); - } - - function testFailPermitPastDeadline() public { - uint256 oldTimestamp = block.timestamp; - uint256 privateKey = 0xBEEF; - address owner = hevm.addr(privateKey); - - (uint8 v, bytes32 r, bytes32 s) = hevm.sign( - privateKey, - keccak256( - abi.encodePacked( - "\x19\x01", - token.DOMAIN_SEPARATOR(), - keccak256(abi.encode(PERMIT_TYPEHASH, owner, address(0xCAFE), 1e18, 0, oldTimestamp)) - ) - ) - ); - - hevm.warp(block.timestamp + 1); - token.permit(owner, address(0xCAFE), 1e18, oldTimestamp, v, r, s); - } - - function testFailPermitReplay() public { - uint256 privateKey = 0xBEEF; - address owner = hevm.addr(privateKey); - - (uint8 v, bytes32 r, bytes32 s) = hevm.sign( - privateKey, - keccak256( - abi.encodePacked( - "\x19\x01", - token.DOMAIN_SEPARATOR(), - keccak256(abi.encode(PERMIT_TYPEHASH, owner, address(0xCAFE), 1e18, 0, block.timestamp)) - ) - ) - ); - - token.permit(owner, address(0xCAFE), 1e18, block.timestamp, v, r, s); - token.permit(owner, address(0xCAFE), 1e18, block.timestamp, v, r, s); - } - - function testMetadata( - string calldata name, - string calldata symbol, - uint8 decimals - ) public { - MockERC20 tkn = new MockERC20(name, symbol, decimals); - assertEq(tkn.name(), name); - assertEq(tkn.symbol(), symbol); - assertEq(tkn.decimals(), decimals); - } - - function testMint(address from, uint256 amount) public { - token.mint(from, amount); - - assertEq(token.totalSupply(), amount); - assertEq(token.balanceOf(from), amount); - } - - function testBurn( - address from, - uint256 mintAmount, - uint256 burnAmount - ) public { - burnAmount = bound(burnAmount, 0, mintAmount); - - token.mint(from, mintAmount); - token.burn(from, burnAmount); - - assertEq(token.totalSupply(), mintAmount - burnAmount); - assertEq(token.balanceOf(from), mintAmount - burnAmount); - } - - function testApprove(address to, uint256 amount) public { - assertTrue(token.approve(to, amount)); - - assertEq(token.allowance(address(this), to), amount); - } - - function testTransfer(address from, uint256 amount) public { - token.mint(address(this), amount); - - assertTrue(token.transfer(from, amount)); - assertEq(token.totalSupply(), amount); - - if (address(this) == from) { - assertEq(token.balanceOf(address(this)), amount); - } else { - assertEq(token.balanceOf(address(this)), 0); - assertEq(token.balanceOf(from), amount); - } - } - - function testTransferFrom( - address to, - uint256 approval, - uint256 amount - ) public { - amount = bound(amount, 0, approval); - - address from = address(0xABCD); - - token.mint(from, amount); - - hevm.prank(from); - token.approve(address(this), approval); - - assertTrue(token.transferFrom(from, to, amount)); - assertEq(token.totalSupply(), amount); - - uint256 app = from == address(this) || approval == type(uint256).max ? approval : approval - amount; - assertEq(token.allowance(from, address(this)), app); - - if (from == to) { - assertEq(token.balanceOf(from), amount); - } else { - assertEq(token.balanceOf(from), 0); - assertEq(token.balanceOf(to), amount); - } - } - - function testPermit( - uint248 privKey, - address to, - uint256 amount, - uint256 deadline - ) public { - uint256 privateKey = privKey; - if (deadline < block.timestamp) deadline = block.timestamp; - if (privateKey == 0) privateKey = 1; - - address owner = hevm.addr(privateKey); - - (uint8 v, bytes32 r, bytes32 s) = hevm.sign( - privateKey, - keccak256( - abi.encodePacked( - "\x19\x01", - token.DOMAIN_SEPARATOR(), - keccak256(abi.encode(PERMIT_TYPEHASH, owner, to, amount, 0, deadline)) - ) - ) - ); - - token.permit(owner, to, amount, deadline, v, r, s); - - assertEq(token.allowance(owner, to), amount); - assertEq(token.nonces(owner), 1); - } - - function testFailBurnInsufficientBalance( - address to, - uint256 mintAmount, - uint256 burnAmount - ) public { - burnAmount = bound(burnAmount, mintAmount + 1, type(uint256).max); - - token.mint(to, mintAmount); - token.burn(to, burnAmount); - } - - function testFailTransferInsufficientBalance( - address to, - uint256 mintAmount, - uint256 sendAmount - ) public { - sendAmount = bound(sendAmount, mintAmount + 1, type(uint256).max); - - token.mint(address(this), mintAmount); - token.transfer(to, sendAmount); - } - - function testFailTransferFromInsufficientAllowance( - address to, - uint256 approval, - uint256 amount - ) public { - amount = bound(amount, approval + 1, type(uint256).max); - - address from = address(0xABCD); - - token.mint(from, amount); - - hevm.prank(from); - token.approve(address(this), approval); - - token.transferFrom(from, to, amount); - } - - function testFailTransferFromInsufficientBalance( - address to, - uint256 mintAmount, - uint256 sendAmount - ) public { - sendAmount = bound(sendAmount, mintAmount + 1, type(uint256).max); - - address from = address(0xABCD); - - token.mint(from, mintAmount); - - hevm.prank(from); - token.approve(address(this), sendAmount); - - token.transferFrom(from, to, sendAmount); - } - - function testFailPermitBadNonce( - uint256 privateKey, - address to, - uint256 amount, - uint256 deadline, - uint256 nonce - ) public { - if (deadline < block.timestamp) deadline = block.timestamp; - if (privateKey == 0) privateKey = 1; - if (nonce == 0) nonce = 1; - - address owner = hevm.addr(privateKey); - - (uint8 v, bytes32 r, bytes32 s) = hevm.sign( - privateKey, - keccak256( - abi.encodePacked( - "\x19\x01", - token.DOMAIN_SEPARATOR(), - keccak256(abi.encode(PERMIT_TYPEHASH, owner, to, amount, nonce, deadline)) - ) - ) - ); - - token.permit(owner, to, amount, deadline, v, r, s); - } - - function testFailPermitBadDeadline( - uint256 privateKey, - address to, - uint256 amount, - uint256 deadline - ) public { - if (deadline < block.timestamp) deadline = block.timestamp; - if (privateKey == 0) privateKey = 1; - - address owner = hevm.addr(privateKey); - - (uint8 v, bytes32 r, bytes32 s) = hevm.sign( - privateKey, - keccak256( - abi.encodePacked( - "\x19\x01", - token.DOMAIN_SEPARATOR(), - keccak256(abi.encode(PERMIT_TYPEHASH, owner, to, amount, 0, deadline)) - ) - ) - ); - - token.permit(owner, to, amount, deadline + 1, v, r, s); - } - - function testFailPermitPastDeadline( - uint256 privateKey, - address to, - uint256 amount, - uint256 deadline - ) public { - deadline = bound(deadline, 0, block.timestamp - 1); - if (privateKey == 0) privateKey = 1; - - address owner = hevm.addr(privateKey); - - (uint8 v, bytes32 r, bytes32 s) = hevm.sign( - privateKey, - keccak256( - abi.encodePacked( - "\x19\x01", - token.DOMAIN_SEPARATOR(), - keccak256(abi.encode(PERMIT_TYPEHASH, owner, to, amount, 0, deadline)) - ) - ) - ); - - token.permit(owner, to, amount, deadline, v, r, s); - } - - function testFailPermitReplay( - uint256 privateKey, - address to, - uint256 amount, - uint256 deadline - ) public { - if (deadline < block.timestamp) deadline = block.timestamp; - if (privateKey == 0) privateKey = 1; - - address owner = hevm.addr(privateKey); - - (uint8 v, bytes32 r, bytes32 s) = hevm.sign( - privateKey, - keccak256( - abi.encodePacked( - "\x19\x01", - token.DOMAIN_SEPARATOR(), - keccak256(abi.encode(PERMIT_TYPEHASH, owner, to, amount, 0, deadline)) - ) - ) - ); - - token.permit(owner, to, amount, deadline, v, r, s); - token.permit(owner, to, amount, deadline, v, r, s); - } -} - -contract ERC20Invariants is DSTestPlus, DSInvariantTest { - BalanceSum balanceSum; - MockERC20 token; - - function setUp() public { - token = new MockERC20("Token", "TKN", 18); - balanceSum = new BalanceSum(token); - - addTargetContract(address(balanceSum)); - } - - function invariantBalanceSum() public { - assertEq(token.totalSupply(), balanceSum.sum()); - } -} - -contract BalanceSum { - MockERC20 token; - uint256 public sum; - - constructor(MockERC20 _token) { - token = _token; - } - - function mint(address from, uint256 amount) public { - token.mint(from, amount); - sum += amount; - } - - function burn(address from, uint256 amount) public { - token.burn(from, amount); - sum -= amount; - } - - function approve(address to, uint256 amount) public { - token.approve(to, amount); - } - - function transferFrom( - address from, - address to, - uint256 amount - ) public { - token.transferFrom(from, to, amount); - } - - function transfer(address to, uint256 amount) public { - token.transfer(to, amount); - } -} diff --git a/lib/solmate/src/test/ERC4626.t.sol b/lib/solmate/src/test/ERC4626.t.sol deleted file mode 100644 index 816c8e4..0000000 --- a/lib/solmate/src/test/ERC4626.t.sol +++ /dev/null @@ -1,446 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; - -import {MockERC20} from "./utils/mocks/MockERC20.sol"; -import {MockERC4626} from "./utils/mocks/MockERC4626.sol"; - -contract ERC4626Test is DSTestPlus { - MockERC20 underlying; - MockERC4626 vault; - - function setUp() public { - underlying = new MockERC20("Mock Token", "TKN", 18); - vault = new MockERC4626(underlying, "Mock Token Vault", "vwTKN"); - } - - function invariantMetadata() public { - assertEq(vault.name(), "Mock Token Vault"); - assertEq(vault.symbol(), "vwTKN"); - assertEq(vault.decimals(), 18); - } - - function testMetadata(string calldata name, string calldata symbol) public { - MockERC4626 vlt = new MockERC4626(underlying, name, symbol); - assertEq(vlt.name(), name); - assertEq(vlt.symbol(), symbol); - assertEq(address(vlt.asset()), address(underlying)); - } - - function testSingleDepositWithdraw(uint128 amount) public { - if (amount == 0) amount = 1; - - uint256 aliceUnderlyingAmount = amount; - - address alice = address(0xABCD); - - underlying.mint(alice, aliceUnderlyingAmount); - - hevm.prank(alice); - underlying.approve(address(vault), aliceUnderlyingAmount); - assertEq(underlying.allowance(alice, address(vault)), aliceUnderlyingAmount); - - uint256 alicePreDepositBal = underlying.balanceOf(alice); - - hevm.prank(alice); - uint256 aliceShareAmount = vault.deposit(aliceUnderlyingAmount, alice); - - assertEq(vault.afterDepositHookCalledCounter(), 1); - - // Expect exchange rate to be 1:1 on initial deposit. - assertEq(aliceUnderlyingAmount, aliceShareAmount); - assertEq(vault.previewWithdraw(aliceShareAmount), aliceUnderlyingAmount); - assertEq(vault.previewDeposit(aliceUnderlyingAmount), aliceShareAmount); - assertEq(vault.totalSupply(), aliceShareAmount); - assertEq(vault.totalAssets(), aliceUnderlyingAmount); - assertEq(vault.balanceOf(alice), aliceShareAmount); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), aliceUnderlyingAmount); - assertEq(underlying.balanceOf(alice), alicePreDepositBal - aliceUnderlyingAmount); - - hevm.prank(alice); - vault.withdraw(aliceUnderlyingAmount, alice, alice); - - assertEq(vault.beforeWithdrawHookCalledCounter(), 1); - - assertEq(vault.totalAssets(), 0); - assertEq(vault.balanceOf(alice), 0); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), 0); - assertEq(underlying.balanceOf(alice), alicePreDepositBal); - } - - function testSingleMintRedeem(uint128 amount) public { - if (amount == 0) amount = 1; - - uint256 aliceShareAmount = amount; - - address alice = address(0xABCD); - - underlying.mint(alice, aliceShareAmount); - - hevm.prank(alice); - underlying.approve(address(vault), aliceShareAmount); - assertEq(underlying.allowance(alice, address(vault)), aliceShareAmount); - - uint256 alicePreDepositBal = underlying.balanceOf(alice); - - hevm.prank(alice); - uint256 aliceUnderlyingAmount = vault.mint(aliceShareAmount, alice); - - assertEq(vault.afterDepositHookCalledCounter(), 1); - - // Expect exchange rate to be 1:1 on initial mint. - assertEq(aliceShareAmount, aliceUnderlyingAmount); - assertEq(vault.previewWithdraw(aliceShareAmount), aliceUnderlyingAmount); - assertEq(vault.previewDeposit(aliceUnderlyingAmount), aliceShareAmount); - assertEq(vault.totalSupply(), aliceShareAmount); - assertEq(vault.totalAssets(), aliceUnderlyingAmount); - assertEq(vault.balanceOf(alice), aliceUnderlyingAmount); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), aliceUnderlyingAmount); - assertEq(underlying.balanceOf(alice), alicePreDepositBal - aliceUnderlyingAmount); - - hevm.prank(alice); - vault.redeem(aliceShareAmount, alice, alice); - - assertEq(vault.beforeWithdrawHookCalledCounter(), 1); - - assertEq(vault.totalAssets(), 0); - assertEq(vault.balanceOf(alice), 0); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), 0); - assertEq(underlying.balanceOf(alice), alicePreDepositBal); - } - - function testMultipleMintDepositRedeemWithdraw() public { - // Scenario: - // A = Alice, B = Bob - // ________________________________________________________ - // | Vault shares | A share | A assets | B share | B assets | - // |========================================================| - // | 1. Alice mints 2000 shares (costs 2000 tokens) | - // |--------------|---------|----------|---------|----------| - // | 2000 | 2000 | 2000 | 0 | 0 | - // |--------------|---------|----------|---------|----------| - // | 2. Bob deposits 4000 tokens (mints 4000 shares) | - // |--------------|---------|----------|---------|----------| - // | 6000 | 2000 | 2000 | 4000 | 4000 | - // |--------------|---------|----------|---------|----------| - // | 3. Vault mutates by +3000 tokens... | - // | (simulated yield returned from strategy)... | - // |--------------|---------|----------|---------|----------| - // | 6000 | 2000 | 3000 | 4000 | 6000 | - // |--------------|---------|----------|---------|----------| - // | 4. Alice deposits 2000 tokens (mints 1333 shares) | - // |--------------|---------|----------|---------|----------| - // | 7333 | 3333 | 4999 | 4000 | 6000 | - // |--------------|---------|----------|---------|----------| - // | 5. Bob mints 2000 shares (costs 3001 assets) | - // | NOTE: Bob's assets spent got rounded up | - // | NOTE: Alice's vault assets got rounded up | - // |--------------|---------|----------|---------|----------| - // | 9333 | 3333 | 5000 | 6000 | 9000 | - // |--------------|---------|----------|---------|----------| - // | 6. Vault mutates by +3000 tokens... | - // | (simulated yield returned from strategy) | - // | NOTE: Vault holds 17001 tokens, but sum of | - // | assetsOf() is 17000. | - // |--------------|---------|----------|---------|----------| - // | 9333 | 3333 | 6071 | 6000 | 10929 | - // |--------------|---------|----------|---------|----------| - // | 7. Alice redeem 1333 shares (2428 assets) | - // |--------------|---------|----------|---------|----------| - // | 8000 | 2000 | 3643 | 6000 | 10929 | - // |--------------|---------|----------|---------|----------| - // | 8. Bob withdraws 2928 assets (1608 shares) | - // |--------------|---------|----------|---------|----------| - // | 6392 | 2000 | 3643 | 4392 | 8000 | - // |--------------|---------|----------|---------|----------| - // | 9. Alice withdraws 3643 assets (2000 shares) | - // | NOTE: Bob's assets have been rounded back up | - // |--------------|---------|----------|---------|----------| - // | 4392 | 0 | 0 | 4392 | 8001 | - // |--------------|---------|----------|---------|----------| - // | 10. Bob redeem 4392 shares (8001 tokens) | - // |--------------|---------|----------|---------|----------| - // | 0 | 0 | 0 | 0 | 0 | - // |______________|_________|__________|_________|__________| - - address alice = address(0xABCD); - address bob = address(0xDCBA); - - uint256 mutationUnderlyingAmount = 3000; - - underlying.mint(alice, 4000); - - hevm.prank(alice); - underlying.approve(address(vault), 4000); - - assertEq(underlying.allowance(alice, address(vault)), 4000); - - underlying.mint(bob, 7001); - - hevm.prank(bob); - underlying.approve(address(vault), 7001); - - assertEq(underlying.allowance(bob, address(vault)), 7001); - - // 1. Alice mints 2000 shares (costs 2000 tokens) - hevm.prank(alice); - uint256 aliceUnderlyingAmount = vault.mint(2000, alice); - - uint256 aliceShareAmount = vault.previewDeposit(aliceUnderlyingAmount); - assertEq(vault.afterDepositHookCalledCounter(), 1); - - // Expect to have received the requested mint amount. - assertEq(aliceShareAmount, 2000); - assertEq(vault.balanceOf(alice), aliceShareAmount); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), aliceUnderlyingAmount); - assertEq(vault.convertToShares(aliceUnderlyingAmount), vault.balanceOf(alice)); - - // Expect a 1:1 ratio before mutation. - assertEq(aliceUnderlyingAmount, 2000); - - // Sanity check. - assertEq(vault.totalSupply(), aliceShareAmount); - assertEq(vault.totalAssets(), aliceUnderlyingAmount); - - // 2. Bob deposits 4000 tokens (mints 4000 shares) - hevm.prank(bob); - uint256 bobShareAmount = vault.deposit(4000, bob); - uint256 bobUnderlyingAmount = vault.previewWithdraw(bobShareAmount); - assertEq(vault.afterDepositHookCalledCounter(), 2); - - // Expect to have received the requested underlying amount. - assertEq(bobUnderlyingAmount, 4000); - assertEq(vault.balanceOf(bob), bobShareAmount); - assertEq(vault.convertToAssets(vault.balanceOf(bob)), bobUnderlyingAmount); - assertEq(vault.convertToShares(bobUnderlyingAmount), vault.balanceOf(bob)); - - // Expect a 1:1 ratio before mutation. - assertEq(bobShareAmount, bobUnderlyingAmount); - - // Sanity check. - uint256 preMutationShareBal = aliceShareAmount + bobShareAmount; - uint256 preMutationBal = aliceUnderlyingAmount + bobUnderlyingAmount; - assertEq(vault.totalSupply(), preMutationShareBal); - assertEq(vault.totalAssets(), preMutationBal); - assertEq(vault.totalSupply(), 6000); - assertEq(vault.totalAssets(), 6000); - - // 3. Vault mutates by +3000 tokens... | - // (simulated yield returned from strategy)... - // The Vault now contains more tokens than deposited which causes the exchange rate to change. - // Alice share is 33.33% of the Vault, Bob 66.66% of the Vault. - // Alice's share count stays the same but the underlying amount changes from 2000 to 3000. - // Bob's share count stays the same but the underlying amount changes from 4000 to 6000. - underlying.mint(address(vault), mutationUnderlyingAmount); - assertEq(vault.totalSupply(), preMutationShareBal); - assertEq(vault.totalAssets(), preMutationBal + mutationUnderlyingAmount); - assertEq(vault.balanceOf(alice), aliceShareAmount); - assertEq( - vault.convertToAssets(vault.balanceOf(alice)), - aliceUnderlyingAmount + (mutationUnderlyingAmount / 3) * 1 - ); - assertEq(vault.balanceOf(bob), bobShareAmount); - assertEq(vault.convertToAssets(vault.balanceOf(bob)), bobUnderlyingAmount + (mutationUnderlyingAmount / 3) * 2); - - // 4. Alice deposits 2000 tokens (mints 1333 shares) - hevm.prank(alice); - vault.deposit(2000, alice); - - assertEq(vault.totalSupply(), 7333); - assertEq(vault.balanceOf(alice), 3333); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), 4999); - assertEq(vault.balanceOf(bob), 4000); - assertEq(vault.convertToAssets(vault.balanceOf(bob)), 6000); - - // 5. Bob mints 2000 shares (costs 3001 assets) - // NOTE: Bob's assets spent got rounded up - // NOTE: Alices's vault assets got rounded up - hevm.prank(bob); - vault.mint(2000, bob); - - assertEq(vault.totalSupply(), 9333); - assertEq(vault.balanceOf(alice), 3333); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), 5000); - assertEq(vault.balanceOf(bob), 6000); - assertEq(vault.convertToAssets(vault.balanceOf(bob)), 9000); - - // Sanity checks: - // Alice and bob should have spent all their tokens now - assertEq(underlying.balanceOf(alice), 0); - assertEq(underlying.balanceOf(bob), 0); - // Assets in vault: 4k (alice) + 7k (bob) + 3k (yield) + 1 (round up) - assertEq(vault.totalAssets(), 14001); - - // 6. Vault mutates by +3000 tokens - // NOTE: Vault holds 17001 tokens, but sum of assetsOf() is 17000. - underlying.mint(address(vault), mutationUnderlyingAmount); - assertEq(vault.totalAssets(), 17001); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), 6071); - assertEq(vault.convertToAssets(vault.balanceOf(bob)), 10929); - - // 7. Alice redeem 1333 shares (2428 assets) - hevm.prank(alice); - vault.redeem(1333, alice, alice); - - assertEq(underlying.balanceOf(alice), 2428); - assertEq(vault.totalSupply(), 8000); - assertEq(vault.totalAssets(), 14573); - assertEq(vault.balanceOf(alice), 2000); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), 3643); - assertEq(vault.balanceOf(bob), 6000); - assertEq(vault.convertToAssets(vault.balanceOf(bob)), 10929); - - // 8. Bob withdraws 2929 assets (1608 shares) - hevm.prank(bob); - vault.withdraw(2929, bob, bob); - - assertEq(underlying.balanceOf(bob), 2929); - assertEq(vault.totalSupply(), 6392); - assertEq(vault.totalAssets(), 11644); - assertEq(vault.balanceOf(alice), 2000); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), 3643); - assertEq(vault.balanceOf(bob), 4392); - assertEq(vault.convertToAssets(vault.balanceOf(bob)), 8000); - - // 9. Alice withdraws 3643 assets (2000 shares) - // NOTE: Bob's assets have been rounded back up - hevm.prank(alice); - vault.withdraw(3643, alice, alice); - - assertEq(underlying.balanceOf(alice), 6071); - assertEq(vault.totalSupply(), 4392); - assertEq(vault.totalAssets(), 8001); - assertEq(vault.balanceOf(alice), 0); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), 0); - assertEq(vault.balanceOf(bob), 4392); - assertEq(vault.convertToAssets(vault.balanceOf(bob)), 8001); - - // 10. Bob redeem 4392 shares (8001 tokens) - hevm.prank(bob); - vault.redeem(4392, bob, bob); - assertEq(underlying.balanceOf(bob), 10930); - assertEq(vault.totalSupply(), 0); - assertEq(vault.totalAssets(), 0); - assertEq(vault.balanceOf(alice), 0); - assertEq(vault.convertToAssets(vault.balanceOf(alice)), 0); - assertEq(vault.balanceOf(bob), 0); - assertEq(vault.convertToAssets(vault.balanceOf(bob)), 0); - - // Sanity check - assertEq(underlying.balanceOf(address(vault)), 0); - } - - function testFailDepositWithNotEnoughApproval() public { - underlying.mint(address(this), 0.5e18); - underlying.approve(address(vault), 0.5e18); - assertEq(underlying.allowance(address(this), address(vault)), 0.5e18); - - vault.deposit(1e18, address(this)); - } - - function testFailWithdrawWithNotEnoughUnderlyingAmount() public { - underlying.mint(address(this), 0.5e18); - underlying.approve(address(vault), 0.5e18); - - vault.deposit(0.5e18, address(this)); - - vault.withdraw(1e18, address(this), address(this)); - } - - function testFailRedeemWithNotEnoughShareAmount() public { - underlying.mint(address(this), 0.5e18); - underlying.approve(address(vault), 0.5e18); - - vault.deposit(0.5e18, address(this)); - - vault.redeem(1e18, address(this), address(this)); - } - - function testFailWithdrawWithNoUnderlyingAmount() public { - vault.withdraw(1e18, address(this), address(this)); - } - - function testFailRedeemWithNoShareAmount() public { - vault.redeem(1e18, address(this), address(this)); - } - - function testFailDepositWithNoApproval() public { - vault.deposit(1e18, address(this)); - } - - function testFailMintWithNoApproval() public { - vault.mint(1e18, address(this)); - } - - function testFailDepositZero() public { - vault.deposit(0, address(this)); - } - - function testMintZero() public { - vault.mint(0, address(this)); - - assertEq(vault.balanceOf(address(this)), 0); - assertEq(vault.convertToAssets(vault.balanceOf(address(this))), 0); - assertEq(vault.totalSupply(), 0); - assertEq(vault.totalAssets(), 0); - } - - function testFailRedeemZero() public { - vault.redeem(0, address(this), address(this)); - } - - function testWithdrawZero() public { - vault.withdraw(0, address(this), address(this)); - - assertEq(vault.balanceOf(address(this)), 0); - assertEq(vault.convertToAssets(vault.balanceOf(address(this))), 0); - assertEq(vault.totalSupply(), 0); - assertEq(vault.totalAssets(), 0); - } - - function testVaultInteractionsForSomeoneElse() public { - // init 2 users with a 1e18 balance - address alice = address(0xABCD); - address bob = address(0xDCBA); - underlying.mint(alice, 1e18); - underlying.mint(bob, 1e18); - - hevm.prank(alice); - underlying.approve(address(vault), 1e18); - - hevm.prank(bob); - underlying.approve(address(vault), 1e18); - - // alice deposits 1e18 for bob - hevm.prank(alice); - vault.deposit(1e18, bob); - - assertEq(vault.balanceOf(alice), 0); - assertEq(vault.balanceOf(bob), 1e18); - assertEq(underlying.balanceOf(alice), 0); - - // bob mint 1e18 for alice - hevm.prank(bob); - vault.mint(1e18, alice); - assertEq(vault.balanceOf(alice), 1e18); - assertEq(vault.balanceOf(bob), 1e18); - assertEq(underlying.balanceOf(bob), 0); - - // alice redeem 1e18 for bob - hevm.prank(alice); - vault.redeem(1e18, bob, alice); - - assertEq(vault.balanceOf(alice), 0); - assertEq(vault.balanceOf(bob), 1e18); - assertEq(underlying.balanceOf(bob), 1e18); - - // bob withdraw 1e18 for alice - hevm.prank(bob); - vault.withdraw(1e18, alice, bob); - - assertEq(vault.balanceOf(alice), 0); - assertEq(vault.balanceOf(bob), 0); - assertEq(underlying.balanceOf(alice), 1e18); - } -} diff --git a/lib/solmate/src/test/ERC6909.t.sol b/lib/solmate/src/test/ERC6909.t.sol deleted file mode 100644 index 57cdb73..0000000 --- a/lib/solmate/src/test/ERC6909.t.sol +++ /dev/null @@ -1,378 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; -import {DSInvariantTest} from "./utils/DSInvariantTest.sol"; - -import {MockERC6909} from "./utils/mocks/MockERC6909.sol"; - -contract ERC6909Test is DSTestPlus { - MockERC6909 token; - - mapping(address => mapping(uint256 => uint256)) public userMintAmounts; - mapping(address => mapping(uint256 => uint256)) public userTransferOrBurnAmounts; - - function setUp() public { - token = new MockERC6909(); - } - - function testMint() public { - token.mint(address(0xBEEF), 1337, 100); - - assertEq(token.balanceOf(address(0xBEEF), 1337), 100); - } - - function testBurn() public { - token.mint(address(0xBEEF), 1337, 100); - token.burn(address(0xBEEF), 1337, 70); - - assertEq(token.balanceOf(address(0xBEEF), 1337), 30); - } - - function testSetOperator() public { - token.setOperator(address(0xBEEF), true); - - assertTrue(token.isOperator(address(this), address(0xBEEF))); - } - - function testApprove() public { - token.approve(address(0xBEEF), 1337, 100); - - assertEq(token.allowance(address(this), address(0xBEEF), 1337), 100); - } - - function testTransfer() public { - address sender = address(0xABCD); - - token.mint(sender, 1337, 100); - - hevm.prank(sender); - token.transfer(address(0xBEEF), 1337, 70); - - assertEq(token.balanceOf(sender, 1337), 30); - assertEq(token.balanceOf(address(0xBEEF), 1337), 70); - } - - function testTransferFromWithApproval() public { - address sender = address(0xABCD); - address receiver = address(0xBEEF); - - token.mint(sender, 1337, 100); - - hevm.prank(sender); - token.approve(address(this), 1337, 100); - - token.transferFrom(sender, receiver, 1337, 70); - - assertEq(token.allowance(sender, address(this), 1337), 30); - assertEq(token.balanceOf(sender, 1337), 30); - assertEq(token.balanceOf(receiver, 1337), 70); - } - - function testTransferFromWithInfiniteApproval() public { - address sender = address(0xABCD); - address receiver = address(0xBEEF); - - token.mint(sender, 1337, 100); - - hevm.prank(sender); - token.approve(address(this), 1337, type(uint256).max); - - token.transferFrom(sender, receiver, 1337, 70); - - assertEq(token.allowance(sender, address(this), 1337), type(uint256).max); - assertEq(token.balanceOf(sender, 1337), 30); - assertEq(token.balanceOf(receiver, 1337), 70); - } - - function testTransferFromAsOperator() public { - address sender = address(0xABCD); - address receiver = address(0xBEEF); - - token.mint(sender, 1337, 100); - - hevm.prank(sender); - token.setOperator(address(this), true); - - token.transferFrom(sender, receiver, 1337, 70); - - assertEq(token.balanceOf(sender, 1337), 30); - assertEq(token.balanceOf(receiver, 1337), 70); - } - - function testFailMintBalanceOverflow() public { - token.mint(address(0xDEAD), 1337, type(uint256).max); - token.mint(address(0xDEAD), 1337, 1); - } - - function testFailTransferBalanceUnderflow() public { - address sender = address(0xABCD); - address receiver = address(0xBEEF); - - hevm.prank(sender); - token.transferFrom(sender, receiver, 1337, 1); - } - - function testFailTransferBalanceOverflow() public { - address sender = address(0xABCD); - address receiver = address(0xBEEF); - - token.mint(sender, 1337, type(uint256).max); - - hevm.prank(sender); - token.transferFrom(sender, receiver, 1337, type(uint256).max); - - token.mint(sender, 1337, 1); - - hevm.prank(sender); - token.transferFrom(sender, receiver, 1337, 1); - } - - function testFailTransferFromBalanceUnderflow() public { - address sender = address(0xABCD); - address receiver = address(0xBEEF); - - hevm.prank(sender); - token.transferFrom(sender, receiver, 1337, 1); - } - - function testFailTransferFromBalanceOverflow() public { - address sender = address(0xABCD); - address receiver = address(0xBEEF); - - token.mint(sender, 1337, type(uint256).max); - - hevm.prank(sender); - token.transferFrom(sender, receiver, 1337, type(uint256).max); - - token.mint(sender, 1337, 1); - - hevm.prank(sender); - token.transferFrom(sender, receiver, 1337, 1); - } - - function testFailTransferFromNotAuthorized() public { - address sender = address(0xABCD); - address receiver = address(0xBEEF); - - token.mint(sender, 1337, 100); - - token.transferFrom(sender, receiver, 1337, 100); - } - - function testMint( - address receiver, - uint256 id, - uint256 amount - ) public { - token.mint(receiver, id, amount); - - assertEq(token.balanceOf(receiver, id), amount); - } - - function testBurn( - address sender, - uint256 id, - uint256 amount - ) public { - token.mint(sender, id, amount); - token.burn(sender, id, amount); - - assertEq(token.balanceOf(sender, id), 0); - } - - function testSetOperator(address operator, bool approved) public { - token.setOperator(operator, approved); - - assertBoolEq(token.isOperator(address(this), operator), approved); - } - - function testApprove( - address spender, - uint256 id, - uint256 amount - ) public { - token.approve(spender, id, amount); - - assertEq(token.allowance(address(this), spender, id), amount); - } - - function testTransfer( - address sender, - address receiver, - uint256 id, - uint256 mintAmount, - uint256 transferAmount - ) public { - transferAmount = bound(transferAmount, 0, mintAmount); - - token.mint(sender, id, mintAmount); - - hevm.prank(sender); - token.transfer(receiver, id, transferAmount); - - if (sender == receiver) { - assertEq(token.balanceOf(sender, id), mintAmount); - } else { - assertEq(token.balanceOf(sender, id), mintAmount - transferAmount); - assertEq(token.balanceOf(receiver, id), transferAmount); - } - } - - function testTransferFromWithApproval( - address sender, - address receiver, - uint256 id, - uint256 mintAmount, - uint256 transferAmount - ) public { - transferAmount = bound(transferAmount, 0, mintAmount); - - token.mint(sender, id, mintAmount); - - hevm.prank(sender); - token.approve(address(this), id, mintAmount); - - token.transferFrom(sender, receiver, id, transferAmount); - - if (mintAmount == type(uint256).max) { - assertEq(token.allowance(sender, address(this), id), type(uint256).max); - } else { - assertEq(token.allowance(sender, address(this), id), mintAmount - transferAmount); - } - - if (sender == receiver) { - assertEq(token.balanceOf(sender, id), mintAmount); - } else { - assertEq(token.balanceOf(sender, id), mintAmount - transferAmount); - assertEq(token.balanceOf(receiver, id), transferAmount); - } - } - - function testTransferFromWithInfiniteApproval( - address sender, - address receiver, - uint256 id, - uint256 mintAmount, - uint256 transferAmount - ) public { - transferAmount = bound(transferAmount, 0, mintAmount); - - token.mint(sender, id, mintAmount); - - hevm.prank(sender); - token.approve(address(this), id, type(uint256).max); - - token.transferFrom(sender, receiver, id, transferAmount); - - assertEq(token.allowance(sender, address(this), id), type(uint256).max); - - if (sender == receiver) { - assertEq(token.balanceOf(sender, id), mintAmount); - } else { - assertEq(token.balanceOf(sender, id), mintAmount - transferAmount); - assertEq(token.balanceOf(receiver, id), transferAmount); - } - } - - function testTransferFromAsOperator( - address sender, - address receiver, - uint256 id, - uint256 mintAmount, - uint256 transferAmount - ) public { - transferAmount = bound(transferAmount, 0, mintAmount); - - token.mint(sender, id, mintAmount); - - hevm.prank(sender); - token.setOperator(address(this), true); - - token.transferFrom(sender, receiver, id, transferAmount); - - if (sender == receiver) { - assertEq(token.balanceOf(sender, id), mintAmount); - } else { - assertEq(token.balanceOf(sender, id), mintAmount - transferAmount); - assertEq(token.balanceOf(receiver, id), transferAmount); - } - } - - function testFailTransferBalanceUnderflow( - address sender, - address receiver, - uint256 id, - uint256 amount - ) public { - amount = bound(amount, 1, type(uint256).max); - - hevm.prank(sender); - token.transfer(receiver, id, amount); - } - - function testFailTransferBalanceOverflow( - address sender, - address receiver, - uint256 id, - uint256 amount - ) public { - amount = bound(amount, 1, type(uint256).max); - uint256 overflowAmount = type(uint256).max - amount + 1; - - token.mint(sender, id, amount); - - hevm.prank(sender); - token.transfer(receiver, id, amount); - - token.mint(sender, id, overflowAmount); - - hevm.prank(sender); - token.transfer(receiver, id, overflowAmount); - } - - function testFailTransferFromBalanceUnderflow( - address sender, - address receiver, - uint256 id, - uint256 amount - ) public { - amount = bound(amount, 1, type(uint256).max); - - hevm.prank(sender); - token.transferFrom(sender, receiver, id, amount); - } - - function testFailTransferFromBalanceOverflow( - address sender, - address receiver, - uint256 id, - uint256 amount - ) public { - amount = bound(amount, 1, type(uint256).max); - uint256 overflowAmount = type(uint256).max - amount + 1; - - token.mint(sender, id, amount); - - hevm.prank(sender); - token.transferFrom(sender, receiver, id, amount); - - token.mint(sender, id, overflowAmount); - - hevm.prank(sender); - token.transferFrom(sender, receiver, id, overflowAmount); - } - - function testFailTransferFromNotAuthorized( - address sender, - address receiver, - uint256 id, - uint256 amount - ) public { - amount = bound(amount, 1, type(uint256).max); - - token.mint(sender, id, amount); - - token.transferFrom(sender, receiver, id, amount); - } -} diff --git a/lib/solmate/src/test/ERC721.t.sol b/lib/solmate/src/test/ERC721.t.sol deleted file mode 100644 index 81de0ff..0000000 --- a/lib/solmate/src/test/ERC721.t.sol +++ /dev/null @@ -1,727 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; -import {DSInvariantTest} from "./utils/DSInvariantTest.sol"; - -import {MockERC721} from "./utils/mocks/MockERC721.sol"; - -import {ERC721TokenReceiver} from "../tokens/ERC721.sol"; - -contract ERC721Recipient is ERC721TokenReceiver { - address public operator; - address public from; - uint256 public id; - bytes public data; - - function onERC721Received( - address _operator, - address _from, - uint256 _id, - bytes calldata _data - ) public virtual override returns (bytes4) { - operator = _operator; - from = _from; - id = _id; - data = _data; - - return ERC721TokenReceiver.onERC721Received.selector; - } -} - -contract RevertingERC721Recipient is ERC721TokenReceiver { - function onERC721Received( - address, - address, - uint256, - bytes calldata - ) public virtual override returns (bytes4) { - revert(string(abi.encodePacked(ERC721TokenReceiver.onERC721Received.selector))); - } -} - -contract WrongReturnDataERC721Recipient is ERC721TokenReceiver { - function onERC721Received( - address, - address, - uint256, - bytes calldata - ) public virtual override returns (bytes4) { - return 0xCAFEBEEF; - } -} - -contract NonERC721Recipient {} - -contract ERC721Test is DSTestPlus { - MockERC721 token; - - function setUp() public { - token = new MockERC721("Token", "TKN"); - } - - function invariantMetadata() public { - assertEq(token.name(), "Token"); - assertEq(token.symbol(), "TKN"); - } - - function testMint() public { - token.mint(address(0xBEEF), 1337); - - assertEq(token.balanceOf(address(0xBEEF)), 1); - assertEq(token.ownerOf(1337), address(0xBEEF)); - } - - function testBurn() public { - token.mint(address(0xBEEF), 1337); - token.burn(1337); - - assertEq(token.balanceOf(address(0xBEEF)), 0); - - hevm.expectRevert("NOT_MINTED"); - token.ownerOf(1337); - } - - function testApprove() public { - token.mint(address(this), 1337); - - token.approve(address(0xBEEF), 1337); - - assertEq(token.getApproved(1337), address(0xBEEF)); - } - - function testApproveBurn() public { - token.mint(address(this), 1337); - - token.approve(address(0xBEEF), 1337); - - token.burn(1337); - - assertEq(token.balanceOf(address(this)), 0); - assertEq(token.getApproved(1337), address(0)); - - hevm.expectRevert("NOT_MINTED"); - token.ownerOf(1337); - } - - function testApproveAll() public { - token.setApprovalForAll(address(0xBEEF), true); - - assertTrue(token.isApprovedForAll(address(this), address(0xBEEF))); - } - - function testTransferFrom() public { - address from = address(0xABCD); - - token.mint(from, 1337); - - hevm.prank(from); - token.approve(address(this), 1337); - - token.transferFrom(from, address(0xBEEF), 1337); - - assertEq(token.getApproved(1337), address(0)); - assertEq(token.ownerOf(1337), address(0xBEEF)); - assertEq(token.balanceOf(address(0xBEEF)), 1); - assertEq(token.balanceOf(from), 0); - } - - function testTransferFromSelf() public { - token.mint(address(this), 1337); - - token.transferFrom(address(this), address(0xBEEF), 1337); - - assertEq(token.getApproved(1337), address(0)); - assertEq(token.ownerOf(1337), address(0xBEEF)); - assertEq(token.balanceOf(address(0xBEEF)), 1); - assertEq(token.balanceOf(address(this)), 0); - } - - function testTransferFromApproveAll() public { - address from = address(0xABCD); - - token.mint(from, 1337); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.transferFrom(from, address(0xBEEF), 1337); - - assertEq(token.getApproved(1337), address(0)); - assertEq(token.ownerOf(1337), address(0xBEEF)); - assertEq(token.balanceOf(address(0xBEEF)), 1); - assertEq(token.balanceOf(from), 0); - } - - function testSafeTransferFromToEOA() public { - address from = address(0xABCD); - - token.mint(from, 1337); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeTransferFrom(from, address(0xBEEF), 1337); - - assertEq(token.getApproved(1337), address(0)); - assertEq(token.ownerOf(1337), address(0xBEEF)); - assertEq(token.balanceOf(address(0xBEEF)), 1); - assertEq(token.balanceOf(from), 0); - } - - function testSafeTransferFromToERC721Recipient() public { - address from = address(0xABCD); - ERC721Recipient recipient = new ERC721Recipient(); - - token.mint(from, 1337); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeTransferFrom(from, address(recipient), 1337); - - assertEq(token.getApproved(1337), address(0)); - assertEq(token.ownerOf(1337), address(recipient)); - assertEq(token.balanceOf(address(recipient)), 1); - assertEq(token.balanceOf(from), 0); - - assertEq(recipient.operator(), address(this)); - assertEq(recipient.from(), from); - assertEq(recipient.id(), 1337); - assertBytesEq(recipient.data(), ""); - } - - function testSafeTransferFromToERC721RecipientWithData() public { - address from = address(0xABCD); - ERC721Recipient recipient = new ERC721Recipient(); - - token.mint(from, 1337); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeTransferFrom(from, address(recipient), 1337, "testing 123"); - - assertEq(token.getApproved(1337), address(0)); - assertEq(token.ownerOf(1337), address(recipient)); - assertEq(token.balanceOf(address(recipient)), 1); - assertEq(token.balanceOf(from), 0); - - assertEq(recipient.operator(), address(this)); - assertEq(recipient.from(), from); - assertEq(recipient.id(), 1337); - assertBytesEq(recipient.data(), "testing 123"); - } - - function testSafeMintToEOA() public { - token.safeMint(address(0xBEEF), 1337); - - assertEq(token.ownerOf(1337), address(address(0xBEEF))); - assertEq(token.balanceOf(address(address(0xBEEF))), 1); - } - - function testSafeMintToERC721Recipient() public { - ERC721Recipient to = new ERC721Recipient(); - - token.safeMint(address(to), 1337); - - assertEq(token.ownerOf(1337), address(to)); - assertEq(token.balanceOf(address(to)), 1); - - assertEq(to.operator(), address(this)); - assertEq(to.from(), address(0)); - assertEq(to.id(), 1337); - assertBytesEq(to.data(), ""); - } - - function testSafeMintToERC721RecipientWithData() public { - ERC721Recipient to = new ERC721Recipient(); - - token.safeMint(address(to), 1337, "testing 123"); - - assertEq(token.ownerOf(1337), address(to)); - assertEq(token.balanceOf(address(to)), 1); - - assertEq(to.operator(), address(this)); - assertEq(to.from(), address(0)); - assertEq(to.id(), 1337); - assertBytesEq(to.data(), "testing 123"); - } - - function testFailMintToZero() public { - token.mint(address(0), 1337); - } - - function testFailDoubleMint() public { - token.mint(address(0xBEEF), 1337); - token.mint(address(0xBEEF), 1337); - } - - function testFailBurnUnMinted() public { - token.burn(1337); - } - - function testFailDoubleBurn() public { - token.mint(address(0xBEEF), 1337); - - token.burn(1337); - token.burn(1337); - } - - function testFailApproveUnMinted() public { - token.approve(address(0xBEEF), 1337); - } - - function testFailApproveUnAuthorized() public { - token.mint(address(0xCAFE), 1337); - - token.approve(address(0xBEEF), 1337); - } - - function testFailTransferFromUnOwned() public { - token.transferFrom(address(0xFEED), address(0xBEEF), 1337); - } - - function testFailTransferFromWrongFrom() public { - token.mint(address(0xCAFE), 1337); - - token.transferFrom(address(0xFEED), address(0xBEEF), 1337); - } - - function testFailTransferFromToZero() public { - token.mint(address(this), 1337); - - token.transferFrom(address(this), address(0), 1337); - } - - function testFailTransferFromNotOwner() public { - token.mint(address(0xFEED), 1337); - - token.transferFrom(address(0xFEED), address(0xBEEF), 1337); - } - - function testFailSafeTransferFromToNonERC721Recipient() public { - token.mint(address(this), 1337); - - token.safeTransferFrom(address(this), address(new NonERC721Recipient()), 1337); - } - - function testFailSafeTransferFromToNonERC721RecipientWithData() public { - token.mint(address(this), 1337); - - token.safeTransferFrom(address(this), address(new NonERC721Recipient()), 1337, "testing 123"); - } - - function testFailSafeTransferFromToRevertingERC721Recipient() public { - token.mint(address(this), 1337); - - token.safeTransferFrom(address(this), address(new RevertingERC721Recipient()), 1337); - } - - function testFailSafeTransferFromToRevertingERC721RecipientWithData() public { - token.mint(address(this), 1337); - - token.safeTransferFrom(address(this), address(new RevertingERC721Recipient()), 1337, "testing 123"); - } - - function testFailSafeTransferFromToERC721RecipientWithWrongReturnData() public { - token.mint(address(this), 1337); - - token.safeTransferFrom(address(this), address(new WrongReturnDataERC721Recipient()), 1337); - } - - function testFailSafeTransferFromToERC721RecipientWithWrongReturnDataWithData() public { - token.mint(address(this), 1337); - - token.safeTransferFrom(address(this), address(new WrongReturnDataERC721Recipient()), 1337, "testing 123"); - } - - function testFailSafeMintToNonERC721Recipient() public { - token.safeMint(address(new NonERC721Recipient()), 1337); - } - - function testFailSafeMintToNonERC721RecipientWithData() public { - token.safeMint(address(new NonERC721Recipient()), 1337, "testing 123"); - } - - function testFailSafeMintToRevertingERC721Recipient() public { - token.safeMint(address(new RevertingERC721Recipient()), 1337); - } - - function testFailSafeMintToRevertingERC721RecipientWithData() public { - token.safeMint(address(new RevertingERC721Recipient()), 1337, "testing 123"); - } - - function testFailSafeMintToERC721RecipientWithWrongReturnData() public { - token.safeMint(address(new WrongReturnDataERC721Recipient()), 1337); - } - - function testFailSafeMintToERC721RecipientWithWrongReturnDataWithData() public { - token.safeMint(address(new WrongReturnDataERC721Recipient()), 1337, "testing 123"); - } - - function testFailBalanceOfZeroAddress() public view { - token.balanceOf(address(0)); - } - - function testFailOwnerOfUnminted() public view { - token.ownerOf(1337); - } - - function testMetadata(string memory name, string memory symbol) public { - MockERC721 tkn = new MockERC721(name, symbol); - - assertEq(tkn.name(), name); - assertEq(tkn.symbol(), symbol); - } - - function testMint(address to, uint256 id) public { - if (to == address(0)) to = address(0xBEEF); - - token.mint(to, id); - - assertEq(token.balanceOf(to), 1); - assertEq(token.ownerOf(id), to); - } - - function testBurn(address to, uint256 id) public { - if (to == address(0)) to = address(0xBEEF); - - token.mint(to, id); - token.burn(id); - - assertEq(token.balanceOf(to), 0); - - hevm.expectRevert("NOT_MINTED"); - token.ownerOf(id); - } - - function testApprove(address to, uint256 id) public { - if (to == address(0)) to = address(0xBEEF); - - token.mint(address(this), id); - - token.approve(to, id); - - assertEq(token.getApproved(id), to); - } - - function testApproveBurn(address to, uint256 id) public { - token.mint(address(this), id); - - token.approve(address(to), id); - - token.burn(id); - - assertEq(token.balanceOf(address(this)), 0); - assertEq(token.getApproved(id), address(0)); - - hevm.expectRevert("NOT_MINTED"); - token.ownerOf(id); - } - - function testApproveAll(address to, bool approved) public { - token.setApprovalForAll(to, approved); - - assertBoolEq(token.isApprovedForAll(address(this), to), approved); - } - - function testTransferFrom(uint256 id, address to) public { - address from = address(0xABCD); - - if (to == address(0) || to == from) to = address(0xBEEF); - - token.mint(from, id); - - hevm.prank(from); - token.approve(address(this), id); - - token.transferFrom(from, to, id); - - assertEq(token.getApproved(id), address(0)); - assertEq(token.ownerOf(id), to); - assertEq(token.balanceOf(to), 1); - assertEq(token.balanceOf(from), 0); - } - - function testTransferFromSelf(uint256 id, address to) public { - if (to == address(0) || to == address(this)) to = address(0xBEEF); - - token.mint(address(this), id); - - token.transferFrom(address(this), to, id); - - assertEq(token.getApproved(id), address(0)); - assertEq(token.ownerOf(id), to); - assertEq(token.balanceOf(to), 1); - assertEq(token.balanceOf(address(this)), 0); - } - - function testTransferFromApproveAll(uint256 id, address to) public { - address from = address(0xABCD); - - if (to == address(0) || to == from) to = address(0xBEEF); - - token.mint(from, id); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.transferFrom(from, to, id); - - assertEq(token.getApproved(id), address(0)); - assertEq(token.ownerOf(id), to); - assertEq(token.balanceOf(to), 1); - assertEq(token.balanceOf(from), 0); - } - - function testSafeTransferFromToEOA(uint256 id, address to) public { - address from = address(0xABCD); - - if (to == address(0) || to == from) to = address(0xBEEF); - - if (uint256(uint160(to)) <= 18 || to.code.length > 0) return; - - token.mint(from, id); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeTransferFrom(from, to, id); - - assertEq(token.getApproved(id), address(0)); - assertEq(token.ownerOf(id), to); - assertEq(token.balanceOf(to), 1); - assertEq(token.balanceOf(from), 0); - } - - function testSafeTransferFromToERC721Recipient(uint256 id) public { - address from = address(0xABCD); - - ERC721Recipient recipient = new ERC721Recipient(); - - token.mint(from, id); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeTransferFrom(from, address(recipient), id); - - assertEq(token.getApproved(id), address(0)); - assertEq(token.ownerOf(id), address(recipient)); - assertEq(token.balanceOf(address(recipient)), 1); - assertEq(token.balanceOf(from), 0); - - assertEq(recipient.operator(), address(this)); - assertEq(recipient.from(), from); - assertEq(recipient.id(), id); - assertBytesEq(recipient.data(), ""); - } - - function testSafeTransferFromToERC721RecipientWithData(uint256 id, bytes calldata data) public { - address from = address(0xABCD); - ERC721Recipient recipient = new ERC721Recipient(); - - token.mint(from, id); - - hevm.prank(from); - token.setApprovalForAll(address(this), true); - - token.safeTransferFrom(from, address(recipient), id, data); - - assertEq(token.getApproved(id), address(0)); - assertEq(token.ownerOf(id), address(recipient)); - assertEq(token.balanceOf(address(recipient)), 1); - assertEq(token.balanceOf(from), 0); - - assertEq(recipient.operator(), address(this)); - assertEq(recipient.from(), from); - assertEq(recipient.id(), id); - assertBytesEq(recipient.data(), data); - } - - function testSafeMintToEOA(uint256 id, address to) public { - if (to == address(0)) to = address(0xBEEF); - - if (uint256(uint160(to)) <= 18 || to.code.length > 0) return; - - token.safeMint(to, id); - - assertEq(token.ownerOf(id), address(to)); - assertEq(token.balanceOf(address(to)), 1); - } - - function testSafeMintToERC721Recipient(uint256 id) public { - ERC721Recipient to = new ERC721Recipient(); - - token.safeMint(address(to), id); - - assertEq(token.ownerOf(id), address(to)); - assertEq(token.balanceOf(address(to)), 1); - - assertEq(to.operator(), address(this)); - assertEq(to.from(), address(0)); - assertEq(to.id(), id); - assertBytesEq(to.data(), ""); - } - - function testSafeMintToERC721RecipientWithData(uint256 id, bytes calldata data) public { - ERC721Recipient to = new ERC721Recipient(); - - token.safeMint(address(to), id, data); - - assertEq(token.ownerOf(id), address(to)); - assertEq(token.balanceOf(address(to)), 1); - - assertEq(to.operator(), address(this)); - assertEq(to.from(), address(0)); - assertEq(to.id(), id); - assertBytesEq(to.data(), data); - } - - function testFailMintToZero(uint256 id) public { - token.mint(address(0), id); - } - - function testFailDoubleMint(uint256 id, address to) public { - if (to == address(0)) to = address(0xBEEF); - - token.mint(to, id); - token.mint(to, id); - } - - function testFailBurnUnMinted(uint256 id) public { - token.burn(id); - } - - function testFailDoubleBurn(uint256 id, address to) public { - if (to == address(0)) to = address(0xBEEF); - - token.mint(to, id); - - token.burn(id); - token.burn(id); - } - - function testFailApproveUnMinted(uint256 id, address to) public { - token.approve(to, id); - } - - function testFailApproveUnAuthorized( - address owner, - uint256 id, - address to - ) public { - if (owner == address(0) || owner == address(this)) owner = address(0xBEEF); - - token.mint(owner, id); - - token.approve(to, id); - } - - function testFailTransferFromUnOwned( - address from, - address to, - uint256 id - ) public { - token.transferFrom(from, to, id); - } - - function testFailTransferFromWrongFrom( - address owner, - address from, - address to, - uint256 id - ) public { - if (owner == address(0)) to = address(0xBEEF); - if (from == owner) revert(); - - token.mint(owner, id); - - token.transferFrom(from, to, id); - } - - function testFailTransferFromToZero(uint256 id) public { - token.mint(address(this), id); - - token.transferFrom(address(this), address(0), id); - } - - function testFailTransferFromNotOwner( - address from, - address to, - uint256 id - ) public { - if (from == address(this)) from = address(0xBEEF); - - token.mint(from, id); - - token.transferFrom(from, to, id); - } - - function testFailSafeTransferFromToNonERC721Recipient(uint256 id) public { - token.mint(address(this), id); - - token.safeTransferFrom(address(this), address(new NonERC721Recipient()), id); - } - - function testFailSafeTransferFromToNonERC721RecipientWithData(uint256 id, bytes calldata data) public { - token.mint(address(this), id); - - token.safeTransferFrom(address(this), address(new NonERC721Recipient()), id, data); - } - - function testFailSafeTransferFromToRevertingERC721Recipient(uint256 id) public { - token.mint(address(this), id); - - token.safeTransferFrom(address(this), address(new RevertingERC721Recipient()), id); - } - - function testFailSafeTransferFromToRevertingERC721RecipientWithData(uint256 id, bytes calldata data) public { - token.mint(address(this), id); - - token.safeTransferFrom(address(this), address(new RevertingERC721Recipient()), id, data); - } - - function testFailSafeTransferFromToERC721RecipientWithWrongReturnData(uint256 id) public { - token.mint(address(this), id); - - token.safeTransferFrom(address(this), address(new WrongReturnDataERC721Recipient()), id); - } - - function testFailSafeTransferFromToERC721RecipientWithWrongReturnDataWithData(uint256 id, bytes calldata data) - public - { - token.mint(address(this), id); - - token.safeTransferFrom(address(this), address(new WrongReturnDataERC721Recipient()), id, data); - } - - function testFailSafeMintToNonERC721Recipient(uint256 id) public { - token.safeMint(address(new NonERC721Recipient()), id); - } - - function testFailSafeMintToNonERC721RecipientWithData(uint256 id, bytes calldata data) public { - token.safeMint(address(new NonERC721Recipient()), id, data); - } - - function testFailSafeMintToRevertingERC721Recipient(uint256 id) public { - token.safeMint(address(new RevertingERC721Recipient()), id); - } - - function testFailSafeMintToRevertingERC721RecipientWithData(uint256 id, bytes calldata data) public { - token.safeMint(address(new RevertingERC721Recipient()), id, data); - } - - function testFailSafeMintToERC721RecipientWithWrongReturnData(uint256 id) public { - token.safeMint(address(new WrongReturnDataERC721Recipient()), id); - } - - function testFailSafeMintToERC721RecipientWithWrongReturnDataWithData(uint256 id, bytes calldata data) public { - token.safeMint(address(new WrongReturnDataERC721Recipient()), id, data); - } - - function testFailOwnerOfUnminted(uint256 id) public view { - token.ownerOf(id); - } -} diff --git a/lib/solmate/src/test/FixedPointMathLib.t.sol b/lib/solmate/src/test/FixedPointMathLib.t.sol deleted file mode 100644 index 789f957..0000000 --- a/lib/solmate/src/test/FixedPointMathLib.t.sol +++ /dev/null @@ -1,360 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; - -import {FixedPointMathLib} from "../utils/FixedPointMathLib.sol"; - -contract FixedPointMathLibTest is DSTestPlus { - function testMulWadDown() public { - assertEq(FixedPointMathLib.mulWadDown(2.5e18, 0.5e18), 1.25e18); - assertEq(FixedPointMathLib.mulWadDown(3e18, 1e18), 3e18); - assertEq(FixedPointMathLib.mulWadDown(369, 271), 0); - } - - function testMulWadDownEdgeCases() public { - assertEq(FixedPointMathLib.mulWadDown(0, 1e18), 0); - assertEq(FixedPointMathLib.mulWadDown(1e18, 0), 0); - assertEq(FixedPointMathLib.mulWadDown(0, 0), 0); - } - - function testMulWadUp() public { - assertEq(FixedPointMathLib.mulWadUp(2.5e18, 0.5e18), 1.25e18); - assertEq(FixedPointMathLib.mulWadUp(3e18, 1e18), 3e18); - assertEq(FixedPointMathLib.mulWadUp(369, 271), 1); - } - - function testMulWadUpEdgeCases() public { - assertEq(FixedPointMathLib.mulWadUp(0, 1e18), 0); - assertEq(FixedPointMathLib.mulWadUp(1e18, 0), 0); - assertEq(FixedPointMathLib.mulWadUp(0, 0), 0); - } - - function testDivWadDown() public { - assertEq(FixedPointMathLib.divWadDown(1.25e18, 0.5e18), 2.5e18); - assertEq(FixedPointMathLib.divWadDown(3e18, 1e18), 3e18); - assertEq(FixedPointMathLib.divWadDown(2, 100000000000000e18), 0); - } - - function testDivWadDownEdgeCases() public { - assertEq(FixedPointMathLib.divWadDown(0, 1e18), 0); - } - - function testFailDivWadDownZeroDenominator() public pure { - FixedPointMathLib.divWadDown(1e18, 0); - } - - function testDivWadUp() public { - assertEq(FixedPointMathLib.divWadUp(1.25e18, 0.5e18), 2.5e18); - assertEq(FixedPointMathLib.divWadUp(3e18, 1e18), 3e18); - assertEq(FixedPointMathLib.divWadUp(2, 100000000000000e18), 1); - } - - function testDivWadUpEdgeCases() public { - assertEq(FixedPointMathLib.divWadUp(0, 1e18), 0); - } - - function testFailDivWadUpZeroDenominator() public pure { - FixedPointMathLib.divWadUp(1e18, 0); - } - - function testMulDivDown() public { - assertEq(FixedPointMathLib.mulDivDown(2.5e27, 0.5e27, 1e27), 1.25e27); - assertEq(FixedPointMathLib.mulDivDown(2.5e18, 0.5e18, 1e18), 1.25e18); - assertEq(FixedPointMathLib.mulDivDown(2.5e8, 0.5e8, 1e8), 1.25e8); - assertEq(FixedPointMathLib.mulDivDown(369, 271, 1e2), 999); - - assertEq(FixedPointMathLib.mulDivDown(1e27, 1e27, 2e27), 0.5e27); - assertEq(FixedPointMathLib.mulDivDown(1e18, 1e18, 2e18), 0.5e18); - assertEq(FixedPointMathLib.mulDivDown(1e8, 1e8, 2e8), 0.5e8); - - assertEq(FixedPointMathLib.mulDivDown(2e27, 3e27, 2e27), 3e27); - assertEq(FixedPointMathLib.mulDivDown(3e18, 2e18, 3e18), 2e18); - assertEq(FixedPointMathLib.mulDivDown(2e8, 3e8, 2e8), 3e8); - } - - function testMulDivDownEdgeCases() public { - assertEq(FixedPointMathLib.mulDivDown(0, 1e18, 1e18), 0); - assertEq(FixedPointMathLib.mulDivDown(1e18, 0, 1e18), 0); - assertEq(FixedPointMathLib.mulDivDown(0, 0, 1e18), 0); - } - - function testFailMulDivDownZeroDenominator() public pure { - FixedPointMathLib.mulDivDown(1e18, 1e18, 0); - } - - function testMulDivUp() public { - assertEq(FixedPointMathLib.mulDivUp(2.5e27, 0.5e27, 1e27), 1.25e27); - assertEq(FixedPointMathLib.mulDivUp(2.5e18, 0.5e18, 1e18), 1.25e18); - assertEq(FixedPointMathLib.mulDivUp(2.5e8, 0.5e8, 1e8), 1.25e8); - assertEq(FixedPointMathLib.mulDivUp(369, 271, 1e2), 1000); - - assertEq(FixedPointMathLib.mulDivUp(1e27, 1e27, 2e27), 0.5e27); - assertEq(FixedPointMathLib.mulDivUp(1e18, 1e18, 2e18), 0.5e18); - assertEq(FixedPointMathLib.mulDivUp(1e8, 1e8, 2e8), 0.5e8); - - assertEq(FixedPointMathLib.mulDivUp(2e27, 3e27, 2e27), 3e27); - assertEq(FixedPointMathLib.mulDivUp(3e18, 2e18, 3e18), 2e18); - assertEq(FixedPointMathLib.mulDivUp(2e8, 3e8, 2e8), 3e8); - } - - function testMulDivUpEdgeCases() public { - assertEq(FixedPointMathLib.mulDivUp(0, 1e18, 1e18), 0); - assertEq(FixedPointMathLib.mulDivUp(1e18, 0, 1e18), 0); - assertEq(FixedPointMathLib.mulDivUp(0, 0, 1e18), 0); - } - - function testFailMulDivUpZeroDenominator() public pure { - FixedPointMathLib.mulDivUp(1e18, 1e18, 0); - } - - function testRPow() public { - assertEq(FixedPointMathLib.rpow(2e27, 2, 1e27), 4e27); - assertEq(FixedPointMathLib.rpow(2e18, 2, 1e18), 4e18); - assertEq(FixedPointMathLib.rpow(2e8, 2, 1e8), 4e8); - assertEq(FixedPointMathLib.rpow(8, 3, 1), 512); - } - - function testSqrt() public { - assertEq(FixedPointMathLib.sqrt(0), 0); - assertEq(FixedPointMathLib.sqrt(1), 1); - assertEq(FixedPointMathLib.sqrt(2704), 52); - assertEq(FixedPointMathLib.sqrt(110889), 333); - assertEq(FixedPointMathLib.sqrt(32239684), 5678); - assertEq(FixedPointMathLib.sqrt(type(uint256).max), 340282366920938463463374607431768211455); - } - - function testSqrtBackHashedSingle() public { - testSqrtBackHashed(123); - } - - function testMulWadDown(uint256 x, uint256 y) public { - // Ignore cases where x * y overflows. - unchecked { - if (x != 0 && (x * y) / x != y) return; - } - - assertEq(FixedPointMathLib.mulWadDown(x, y), (x * y) / 1e18); - } - - function testFailMulWadDownOverflow(uint256 x, uint256 y) public pure { - // Ignore cases where x * y does not overflow. - unchecked { - if ((x * y) / x == y) revert(); - } - - FixedPointMathLib.mulWadDown(x, y); - } - - function testMulWadUp(uint256 x, uint256 y) public { - // Ignore cases where x * y overflows. - unchecked { - if (x != 0 && (x * y) / x != y) return; - } - - assertEq(FixedPointMathLib.mulWadUp(x, y), x * y == 0 ? 0 : (x * y - 1) / 1e18 + 1); - } - - function testFailMulWadUpOverflow(uint256 x, uint256 y) public pure { - // Ignore cases where x * y does not overflow. - unchecked { - if ((x * y) / x == y) revert(); - } - - FixedPointMathLib.mulWadUp(x, y); - } - - function testDivWadDown(uint256 x, uint256 y) public { - // Ignore cases where x * WAD overflows or y is 0. - unchecked { - if (y == 0 || (x != 0 && (x * 1e18) / 1e18 != x)) return; - } - - assertEq(FixedPointMathLib.divWadDown(x, y), (x * 1e18) / y); - } - - function testFailDivWadDownOverflow(uint256 x, uint256 y) public pure { - // Ignore cases where x * WAD does not overflow or y is 0. - unchecked { - if (y == 0 || (x * 1e18) / 1e18 == x) revert(); - } - - FixedPointMathLib.divWadDown(x, y); - } - - function testFailDivWadDownZeroDenominator(uint256 x) public pure { - FixedPointMathLib.divWadDown(x, 0); - } - - function testDivWadUp(uint256 x, uint256 y) public { - // Ignore cases where x * WAD overflows or y is 0. - unchecked { - if (y == 0 || (x != 0 && (x * 1e18) / 1e18 != x)) return; - } - - assertEq(FixedPointMathLib.divWadUp(x, y), x == 0 ? 0 : (x * 1e18 - 1) / y + 1); - } - - function testFailDivWadUpOverflow(uint256 x, uint256 y) public pure { - // Ignore cases where x * WAD does not overflow or y is 0. - unchecked { - if (y == 0 || (x * 1e18) / 1e18 == x) revert(); - } - - FixedPointMathLib.divWadUp(x, y); - } - - function testFailDivWadUpZeroDenominator(uint256 x) public pure { - FixedPointMathLib.divWadUp(x, 0); - } - - function testMulDivDown( - uint256 x, - uint256 y, - uint256 denominator - ) public { - // Ignore cases where x * y overflows or denominator is 0. - unchecked { - if (denominator == 0 || (x != 0 && (x * y) / x != y)) return; - } - - assertEq(FixedPointMathLib.mulDivDown(x, y, denominator), (x * y) / denominator); - } - - function testFailMulDivDownOverflow( - uint256 x, - uint256 y, - uint256 denominator - ) public pure { - // Ignore cases where x * y does not overflow or denominator is 0. - unchecked { - if (denominator == 0 || (x * y) / x == y) revert(); - } - - FixedPointMathLib.mulDivDown(x, y, denominator); - } - - function testFailMulDivDownZeroDenominator(uint256 x, uint256 y) public pure { - FixedPointMathLib.mulDivDown(x, y, 0); - } - - function testMulDivUp( - uint256 x, - uint256 y, - uint256 denominator - ) public { - // Ignore cases where x * y overflows or denominator is 0. - unchecked { - if (denominator == 0 || (x != 0 && (x * y) / x != y)) return; - } - - assertEq(FixedPointMathLib.mulDivUp(x, y, denominator), x * y == 0 ? 0 : (x * y - 1) / denominator + 1); - } - - function testFailMulDivUpOverflow( - uint256 x, - uint256 y, - uint256 denominator - ) public pure { - // Ignore cases where x * y does not overflow or denominator is 0. - unchecked { - if (denominator == 0 || (x * y) / x == y) revert(); - } - - FixedPointMathLib.mulDivUp(x, y, denominator); - } - - function testFailMulDivUpZeroDenominator(uint256 x, uint256 y) public pure { - FixedPointMathLib.mulDivUp(x, y, 0); - } - - function testDifferentiallyFuzzSqrt(uint256 x) public { - assertEq(FixedPointMathLib.sqrt(x), uniswapSqrt(x)); - assertEq(FixedPointMathLib.sqrt(x), abdkSqrt(x)); - } - - function testSqrt(uint256 x) public { - uint256 root = FixedPointMathLib.sqrt(x); - uint256 next = root + 1; - - // Ignore cases where next * next overflows. - unchecked { - if (next * next < next) return; - } - - assertTrue(root * root <= x && next * next > x); - } - - function testSqrtBack(uint256 x) public { - unchecked { - x >>= 128; - while (x != 0) { - assertEq(FixedPointMathLib.sqrt(x * x), x); - x >>= 1; - } - } - } - - function testSqrtBackHashed(uint256 x) public { - testSqrtBack(uint256(keccak256(abi.encode(x)))); - } - - function uniswapSqrt(uint256 y) internal pure returns (uint256 z) { - if (y > 3) { - z = y; - uint256 x = y / 2 + 1; - while (x < z) { - z = x; - x = (y / x + x) / 2; - } - } else if (y != 0) { - z = 1; - } - } - - function abdkSqrt(uint256 x) private pure returns (uint256) { - unchecked { - if (x == 0) return 0; - else { - uint256 xx = x; - uint256 r = 1; - if (xx >= 0x100000000000000000000000000000000) { - xx >>= 128; - r <<= 64; - } - if (xx >= 0x10000000000000000) { - xx >>= 64; - r <<= 32; - } - if (xx >= 0x100000000) { - xx >>= 32; - r <<= 16; - } - if (xx >= 0x10000) { - xx >>= 16; - r <<= 8; - } - if (xx >= 0x100) { - xx >>= 8; - r <<= 4; - } - if (xx >= 0x10) { - xx >>= 4; - r <<= 2; - } - if (xx >= 0x8) { - r <<= 1; - } - r = (r + x / r) >> 1; - r = (r + x / r) >> 1; - r = (r + x / r) >> 1; - r = (r + x / r) >> 1; - r = (r + x / r) >> 1; - r = (r + x / r) >> 1; - r = (r + x / r) >> 1; // Seven iterations should be enough - uint256 r1 = x / r; - return r < r1 ? r : r1; - } - } - } -} diff --git a/lib/solmate/src/test/LibString.t.sol b/lib/solmate/src/test/LibString.t.sol deleted file mode 100644 index 36ed435..0000000 --- a/lib/solmate/src/test/LibString.t.sol +++ /dev/null @@ -1,148 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity >=0.8.0; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; - -import {LibString} from "../utils/LibString.sol"; - -contract LibStringTest is DSTestPlus { - function testToString() public { - assertEq(LibString.toString(uint256(0)), "0"); - assertEq(LibString.toString(uint256(1)), "1"); - assertEq(LibString.toString(uint256(17)), "17"); - assertEq(LibString.toString(uint256(99999999)), "99999999"); - assertEq(LibString.toString(uint256(99999999999)), "99999999999"); - assertEq(LibString.toString(uint256(2342343923423)), "2342343923423"); - assertEq(LibString.toString(uint256(98765685434567)), "98765685434567"); - } - - function testToStringIntPositive() public { - assertEq(LibString.toString(int256(0)), "0"); - assertEq(LibString.toString(int256(1)), "1"); - assertEq(LibString.toString(int256(17)), "17"); - assertEq(LibString.toString(int256(99999999)), "99999999"); - assertEq(LibString.toString(int256(99999999999)), "99999999999"); - assertEq(LibString.toString(int256(2342343923423)), "2342343923423"); - assertEq(LibString.toString(int256(98765685434567)), "98765685434567"); - } - - function testToStringIntNegative() public { - assertEq(LibString.toString(int256(-0)), "0"); - assertEq(LibString.toString(int256(-17)), "-17"); - assertEq(LibString.toString(int256(-99999999)), "-99999999"); - assertEq(LibString.toString(int256(-99999999999)), "-99999999999"); - assertEq(LibString.toString(int256(-2342343923423)), "-2342343923423"); - assertEq(LibString.toString(int256(-98765685434567)), "-98765685434567"); - } - - function testDifferentiallyFuzzToString(uint256 value, bytes calldata brutalizeWith) - public - brutalizeMemory(brutalizeWith) - { - string memory libString = LibString.toString(value); - string memory oz = toStringOZ(value); - - assertEq(bytes(libString).length, bytes(oz).length); - assertEq(libString, oz); - } - - function testDifferentiallyFuzzToStringInt(int256 value, bytes calldata brutalizeWith) - public - brutalizeMemory(brutalizeWith) - { - string memory libString = LibString.toString(value); - string memory oz = toStringOZ(value); - - assertEq(bytes(libString).length, bytes(oz).length); - assertEq(libString, oz); - } - - function testToStringOverwrite() public { - string memory str = LibString.toString(uint256(1)); - - bytes32 data; - bytes32 expected; - - assembly { - // Imagine a high level allocation writing something to the current free memory. - // Should have sufficient higher order bits for this to be visible - mstore(mload(0x40), not(0)) - // Correctly allocate 32 more bytes, to avoid more interference - mstore(0x40, add(mload(0x40), 32)) - data := mload(add(str, 32)) - - // the expected value should be the uft-8 encoding of 1 (49), - // followed by clean bits. We achieve this by taking the value and - // shifting left to the end of the 32 byte word - expected := shl(248, 49) - } - - assertEq(data, expected); - } - - function testToStringDirty() public { - uint256 freememptr; - // Make the next 4 bytes of the free memory dirty - assembly { - let dirty := not(0) - freememptr := mload(0x40) - mstore(freememptr, dirty) - mstore(add(freememptr, 32), dirty) - mstore(add(freememptr, 64), dirty) - mstore(add(freememptr, 96), dirty) - mstore(add(freememptr, 128), dirty) - } - string memory str = LibString.toString(uint256(1)); - uint256 len; - bytes32 data; - bytes32 expected; - assembly { - freememptr := str - len := mload(str) - data := mload(add(str, 32)) - // the expected value should be the uft-8 encoding of 1 (49), - // followed by clean bits. We achieve this by taking the value and - // shifting left to the end of the 32 byte word - expected := shl(248, 49) - } - emit log_named_uint("str: ", freememptr); - emit log_named_uint("len: ", len); - emit log_named_bytes32("data: ", data); - assembly { - freememptr := mload(0x40) - } - emit log_named_uint("memptr: ", freememptr); - - assertEq(data, expected); - } -} - -function toStringOZ(int256 value) pure returns (string memory) { - return string(abi.encodePacked(value < 0 ? "-" : "", toStringOZ(absOZ(value)))); -} - -function toStringOZ(uint256 value) pure returns (string memory) { - if (value == 0) { - return "0"; - } - uint256 temp = value; - uint256 digits; - while (temp != 0) { - digits++; - temp /= 10; - } - bytes memory buffer = new bytes(digits); - while (value != 0) { - digits -= 1; - buffer[digits] = bytes1(uint8(48 + uint256(value % 10))); - value /= 10; - } - return string(buffer); -} - -function absOZ(int256 n) pure returns (uint256) { - unchecked { - // must be unchecked in order to support `n = type(int256).min` - return uint256(n >= 0 ? n : -n); - } -} diff --git a/lib/solmate/src/test/MerkleProofLib.t.sol b/lib/solmate/src/test/MerkleProofLib.t.sol deleted file mode 100644 index 5279679..0000000 --- a/lib/solmate/src/test/MerkleProofLib.t.sol +++ /dev/null @@ -1,50 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity >=0.8.0; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; - -import {MerkleProofLib} from "../utils/MerkleProofLib.sol"; - -contract MerkleProofLibTest is DSTestPlus { - function testVerifyEmptyMerkleProofSuppliedLeafAndRootSame() public { - bytes32[] memory proof; - assertBoolEq(this.verify(proof, 0x00, 0x00), true); - } - - function testVerifyEmptyMerkleProofSuppliedLeafAndRootDifferent() public { - bytes32[] memory proof; - bytes32 leaf = "a"; - assertBoolEq(this.verify(proof, 0x00, leaf), false); - } - - function testValidProofSupplied() public { - // Merkle tree created from leaves ['a', 'b', 'c']. - // Leaf is 'a'. - bytes32[] memory proof = new bytes32[](2); - proof[0] = 0xb5553de315e0edf504d9150af82dafa5c4667fa618ed0a6f19c69b41166c5510; - proof[1] = 0x0b42b6393c1f53060fe3ddbfcd7aadcca894465a5a438f69c87d790b2299b9b2; - bytes32 root = 0x5842148bc6ebeb52af882a317c765fccd3ae80589b21a9b8cbf21abb630e46a7; - bytes32 leaf = 0x3ac225168df54212a25c1c01fd35bebfea408fdac2e31ddd6f80a4bbf9a5f1cb; - assertBoolEq(this.verify(proof, root, leaf), true); - } - - function testVerifyInvalidProofSupplied() public { - // Merkle tree created from leaves ['a', 'b', 'c']. - // Leaf is 'a'. - // Proof is same as testValidProofSupplied but last byte of first element is modified. - bytes32[] memory proof = new bytes32[](2); - proof[0] = 0xb5553de315e0edf504d9150af82dafa5c4667fa618ed0a6f19c69b41166c5511; - proof[1] = 0x0b42b6393c1f53060fe3ddbfcd7aadcca894465a5a438f69c87d790b2299b9b2; - bytes32 root = 0x5842148bc6ebeb52af882a317c765fccd3ae80589b21a9b8cbf21abb630e46a7; - bytes32 leaf = 0x3ac225168df54212a25c1c01fd35bebfea408fdac2e31ddd6f80a4bbf9a5f1cb; - assertBoolEq(this.verify(proof, root, leaf), false); - } - - function verify( - bytes32[] calldata proof, - bytes32 root, - bytes32 leaf - ) external pure returns (bool) { - return MerkleProofLib.verify(proof, root, leaf); - } -} diff --git a/lib/solmate/src/test/MultiRolesAuthority.t.sol b/lib/solmate/src/test/MultiRolesAuthority.t.sol deleted file mode 100644 index eedf6a0..0000000 --- a/lib/solmate/src/test/MultiRolesAuthority.t.sol +++ /dev/null @@ -1,321 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; -import {MockAuthority} from "./utils/mocks/MockAuthority.sol"; - -import {Authority} from "../auth/Auth.sol"; - -import {MultiRolesAuthority} from "../auth/authorities/MultiRolesAuthority.sol"; - -contract MultiRolesAuthorityTest is DSTestPlus { - MultiRolesAuthority multiRolesAuthority; - - function setUp() public { - multiRolesAuthority = new MultiRolesAuthority(address(this), Authority(address(0))); - } - - function testSetRoles() public { - assertFalse(multiRolesAuthority.doesUserHaveRole(address(0xBEEF), 0)); - - multiRolesAuthority.setUserRole(address(0xBEEF), 0, true); - assertTrue(multiRolesAuthority.doesUserHaveRole(address(0xBEEF), 0)); - - multiRolesAuthority.setUserRole(address(0xBEEF), 0, false); - assertFalse(multiRolesAuthority.doesUserHaveRole(address(0xBEEF), 0)); - } - - function testSetRoleCapabilities() public { - assertFalse(multiRolesAuthority.doesRoleHaveCapability(0, 0xBEEFCAFE)); - - multiRolesAuthority.setRoleCapability(0, 0xBEEFCAFE, true); - assertTrue(multiRolesAuthority.doesRoleHaveCapability(0, 0xBEEFCAFE)); - - multiRolesAuthority.setRoleCapability(0, 0xBEEFCAFE, false); - assertFalse(multiRolesAuthority.doesRoleHaveCapability(0, 0xBEEFCAFE)); - } - - function testSetPublicCapabilities() public { - assertFalse(multiRolesAuthority.isCapabilityPublic(0xBEEFCAFE)); - - multiRolesAuthority.setPublicCapability(0xBEEFCAFE, true); - assertTrue(multiRolesAuthority.isCapabilityPublic(0xBEEFCAFE)); - - multiRolesAuthority.setPublicCapability(0xBEEFCAFE, false); - assertFalse(multiRolesAuthority.isCapabilityPublic(0xBEEFCAFE)); - } - - function testSetTargetCustomAuthority() public { - assertEq(address(multiRolesAuthority.getTargetCustomAuthority(address(0xBEEF))), address(0)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xBEEF), Authority(address(0xCAFE))); - assertEq(address(multiRolesAuthority.getTargetCustomAuthority(address(0xBEEF))), address(0xCAFE)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xBEEF), Authority(address(0))); - assertEq(address(multiRolesAuthority.getTargetCustomAuthority(address(0xBEEF))), address(0)); - } - - function testCanCallWithAuthorizedRole() public { - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setUserRole(address(0xBEEF), 0, true); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setRoleCapability(0, 0xBEEFCAFE, true); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setRoleCapability(0, 0xBEEFCAFE, false); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setRoleCapability(0, 0xBEEFCAFE, true); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setUserRole(address(0xBEEF), 0, false); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - } - - function testCanCallPublicCapability() public { - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setPublicCapability(0xBEEFCAFE, true); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setPublicCapability(0xBEEFCAFE, false); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - } - - function testCanCallWithCustomAuthority() public { - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xCAFE), new MockAuthority(false)); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xCAFE), new MockAuthority(true)); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xCAFE), new MockAuthority(false)); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xCAFE), Authority(address(0))); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xCAFE), new MockAuthority(true)); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - } - - function testCanCallWithCustomAuthorityOverridesPublicCapability() public { - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setPublicCapability(0xBEEFCAFE, true); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xCAFE), new MockAuthority(false)); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xCAFE), new MockAuthority(true)); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setPublicCapability(0xBEEFCAFE, false); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xCAFE), Authority(address(0))); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setPublicCapability(0xBEEFCAFE, true); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - } - - function testCanCallWithCustomAuthorityOverridesUserWithRole() public { - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setUserRole(address(0xBEEF), 0, true); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setRoleCapability(0, 0xBEEFCAFE, true); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xCAFE), new MockAuthority(false)); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xCAFE), new MockAuthority(true)); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setUserRole(address(0xBEEF), 0, false); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setTargetCustomAuthority(address(0xCAFE), Authority(address(0))); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setUserRole(address(0xBEEF), 0, true); - assertTrue(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setRoleCapability(0, 0xBEEFCAFE, false); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - multiRolesAuthority.setUserRole(address(0xBEEF), 0, false); - assertFalse(multiRolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - } - - function testSetRoles(address user, uint8 role) public { - assertFalse(multiRolesAuthority.doesUserHaveRole(user, role)); - - multiRolesAuthority.setUserRole(user, role, true); - assertTrue(multiRolesAuthority.doesUserHaveRole(user, role)); - - multiRolesAuthority.setUserRole(user, role, false); - assertFalse(multiRolesAuthority.doesUserHaveRole(user, role)); - } - - function testSetRoleCapabilities(uint8 role, bytes4 functionSig) public { - assertFalse(multiRolesAuthority.doesRoleHaveCapability(role, functionSig)); - - multiRolesAuthority.setRoleCapability(role, functionSig, true); - assertTrue(multiRolesAuthority.doesRoleHaveCapability(role, functionSig)); - - multiRolesAuthority.setRoleCapability(role, functionSig, false); - assertFalse(multiRolesAuthority.doesRoleHaveCapability(role, functionSig)); - } - - function testSetPublicCapabilities(bytes4 functionSig) public { - assertFalse(multiRolesAuthority.isCapabilityPublic(functionSig)); - - multiRolesAuthority.setPublicCapability(functionSig, true); - assertTrue(multiRolesAuthority.isCapabilityPublic(functionSig)); - - multiRolesAuthority.setPublicCapability(functionSig, false); - assertFalse(multiRolesAuthority.isCapabilityPublic(functionSig)); - } - - function testSetTargetCustomAuthority(address user, Authority customAuthority) public { - assertEq(address(multiRolesAuthority.getTargetCustomAuthority(user)), address(0)); - - multiRolesAuthority.setTargetCustomAuthority(user, customAuthority); - assertEq(address(multiRolesAuthority.getTargetCustomAuthority(user)), address(customAuthority)); - - multiRolesAuthority.setTargetCustomAuthority(user, Authority(address(0))); - assertEq(address(multiRolesAuthority.getTargetCustomAuthority(user)), address(0)); - } - - function testCanCallWithAuthorizedRole( - address user, - uint8 role, - address target, - bytes4 functionSig - ) public { - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setUserRole(user, role, true); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setRoleCapability(role, functionSig, true); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setRoleCapability(role, functionSig, false); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setRoleCapability(role, functionSig, true); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setUserRole(user, role, false); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - } - - function testCanCallPublicCapability( - address user, - address target, - bytes4 functionSig - ) public { - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setPublicCapability(functionSig, true); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setPublicCapability(functionSig, false); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - } - - function testCanCallWithCustomAuthority( - address user, - address target, - bytes4 functionSig - ) public { - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setTargetCustomAuthority(target, new MockAuthority(false)); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setTargetCustomAuthority(target, new MockAuthority(true)); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setTargetCustomAuthority(target, new MockAuthority(false)); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setTargetCustomAuthority(target, Authority(address(0))); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setTargetCustomAuthority(target, new MockAuthority(true)); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - } - - function testCanCallWithCustomAuthorityOverridesPublicCapability( - address user, - address target, - bytes4 functionSig - ) public { - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setPublicCapability(functionSig, true); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setTargetCustomAuthority(target, new MockAuthority(false)); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setTargetCustomAuthority(target, new MockAuthority(true)); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setPublicCapability(functionSig, false); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setTargetCustomAuthority(target, Authority(address(0))); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setPublicCapability(functionSig, true); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - } - - function testCanCallWithCustomAuthorityOverridesUserWithRole( - address user, - uint8 role, - address target, - bytes4 functionSig - ) public { - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setUserRole(user, role, true); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setRoleCapability(role, functionSig, true); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setTargetCustomAuthority(target, new MockAuthority(false)); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setTargetCustomAuthority(target, new MockAuthority(true)); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setUserRole(user, role, false); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setTargetCustomAuthority(target, Authority(address(0))); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setUserRole(user, role, true); - assertTrue(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setRoleCapability(role, functionSig, false); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - - multiRolesAuthority.setUserRole(user, role, false); - assertFalse(multiRolesAuthority.canCall(user, target, functionSig)); - } -} diff --git a/lib/solmate/src/test/Owned.t.sol b/lib/solmate/src/test/Owned.t.sol deleted file mode 100644 index 08a0239..0000000 --- a/lib/solmate/src/test/Owned.t.sol +++ /dev/null @@ -1,40 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; -import {MockOwned} from "./utils/mocks/MockOwned.sol"; - -contract OwnedTest is DSTestPlus { - MockOwned mockOwned; - - function setUp() public { - mockOwned = new MockOwned(); - } - - function testTransferOwnership() public { - testTransferOwnership(address(0xBEEF)); - } - - function testCallFunctionAsNonOwner() public { - testCallFunctionAsNonOwner(address(0)); - } - - function testCallFunctionAsOwner() public { - mockOwned.updateFlag(); - } - - function testTransferOwnership(address newOwner) public { - mockOwned.transferOwnership(newOwner); - - assertEq(mockOwned.owner(), newOwner); - } - - function testCallFunctionAsNonOwner(address owner) public { - hevm.assume(owner != address(this)); - - mockOwned.transferOwnership(owner); - - hevm.expectRevert("UNAUTHORIZED"); - mockOwned.updateFlag(); - } -} diff --git a/lib/solmate/src/test/ReentrancyGuard.t.sol b/lib/solmate/src/test/ReentrancyGuard.t.sol deleted file mode 100644 index 842e367..0000000 --- a/lib/solmate/src/test/ReentrancyGuard.t.sol +++ /dev/null @@ -1,56 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; - -import {ReentrancyGuard} from "../utils/ReentrancyGuard.sol"; - -contract RiskyContract is ReentrancyGuard { - uint256 public enterTimes; - - function unprotectedCall() public { - enterTimes++; - - if (enterTimes > 1) return; - - this.protectedCall(); - } - - function protectedCall() public nonReentrant { - enterTimes++; - - if (enterTimes > 1) return; - - this.protectedCall(); - } - - function overprotectedCall() public nonReentrant {} -} - -contract ReentrancyGuardTest is DSTestPlus { - RiskyContract riskyContract; - - function setUp() public { - riskyContract = new RiskyContract(); - } - - function invariantReentrancyStatusAlways1() public { - assertEq(uint256(hevm.load(address(riskyContract), 0)), 1); - } - - function testFailUnprotectedCall() public { - riskyContract.unprotectedCall(); - - assertEq(riskyContract.enterTimes(), 1); - } - - function testProtectedCall() public { - try riskyContract.protectedCall() { - fail("Reentrancy Guard Failed To Stop Attacker"); - } catch {} - } - - function testNoReentrancy() public { - riskyContract.overprotectedCall(); - } -} diff --git a/lib/solmate/src/test/RolesAuthority.t.sol b/lib/solmate/src/test/RolesAuthority.t.sol deleted file mode 100644 index e01db7e..0000000 --- a/lib/solmate/src/test/RolesAuthority.t.sol +++ /dev/null @@ -1,148 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; -import {MockAuthority} from "./utils/mocks/MockAuthority.sol"; - -import {Authority} from "../auth/Auth.sol"; - -import {RolesAuthority} from "../auth/authorities/RolesAuthority.sol"; - -contract RolesAuthorityTest is DSTestPlus { - RolesAuthority rolesAuthority; - - function setUp() public { - rolesAuthority = new RolesAuthority(address(this), Authority(address(0))); - } - - function testSetRoles() public { - assertFalse(rolesAuthority.doesUserHaveRole(address(0xBEEF), 0)); - - rolesAuthority.setUserRole(address(0xBEEF), 0, true); - assertTrue(rolesAuthority.doesUserHaveRole(address(0xBEEF), 0)); - - rolesAuthority.setUserRole(address(0xBEEF), 0, false); - assertFalse(rolesAuthority.doesUserHaveRole(address(0xBEEF), 0)); - } - - function testSetRoleCapabilities() public { - assertFalse(rolesAuthority.doesRoleHaveCapability(0, address(0xCAFE), 0xBEEFCAFE)); - - rolesAuthority.setRoleCapability(0, address(0xCAFE), 0xBEEFCAFE, true); - assertTrue(rolesAuthority.doesRoleHaveCapability(0, address(0xCAFE), 0xBEEFCAFE)); - - rolesAuthority.setRoleCapability(0, address(0xCAFE), 0xBEEFCAFE, false); - assertFalse(rolesAuthority.doesRoleHaveCapability(0, address(0xCAFE), 0xBEEFCAFE)); - } - - function testSetPublicCapabilities() public { - assertFalse(rolesAuthority.isCapabilityPublic(address(0xCAFE), 0xBEEFCAFE)); - - rolesAuthority.setPublicCapability(address(0xCAFE), 0xBEEFCAFE, true); - assertTrue(rolesAuthority.isCapabilityPublic(address(0xCAFE), 0xBEEFCAFE)); - - rolesAuthority.setPublicCapability(address(0xCAFE), 0xBEEFCAFE, false); - assertFalse(rolesAuthority.isCapabilityPublic(address(0xCAFE), 0xBEEFCAFE)); - } - - function testCanCallWithAuthorizedRole() public { - assertFalse(rolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - rolesAuthority.setUserRole(address(0xBEEF), 0, true); - assertFalse(rolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - rolesAuthority.setRoleCapability(0, address(0xCAFE), 0xBEEFCAFE, true); - assertTrue(rolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - rolesAuthority.setRoleCapability(0, address(0xCAFE), 0xBEEFCAFE, false); - assertFalse(rolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - rolesAuthority.setRoleCapability(0, address(0xCAFE), 0xBEEFCAFE, true); - assertTrue(rolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - rolesAuthority.setUserRole(address(0xBEEF), 0, false); - assertFalse(rolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - } - - function testCanCallPublicCapability() public { - assertFalse(rolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - rolesAuthority.setPublicCapability(address(0xCAFE), 0xBEEFCAFE, true); - assertTrue(rolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - - rolesAuthority.setPublicCapability(address(0xCAFE), 0xBEEFCAFE, false); - assertFalse(rolesAuthority.canCall(address(0xBEEF), address(0xCAFE), 0xBEEFCAFE)); - } - - function testSetRoles(address user, uint8 role) public { - assertFalse(rolesAuthority.doesUserHaveRole(user, role)); - - rolesAuthority.setUserRole(user, role, true); - assertTrue(rolesAuthority.doesUserHaveRole(user, role)); - - rolesAuthority.setUserRole(user, role, false); - assertFalse(rolesAuthority.doesUserHaveRole(user, role)); - } - - function testSetRoleCapabilities( - uint8 role, - address target, - bytes4 functionSig - ) public { - assertFalse(rolesAuthority.doesRoleHaveCapability(role, target, functionSig)); - - rolesAuthority.setRoleCapability(role, target, functionSig, true); - assertTrue(rolesAuthority.doesRoleHaveCapability(role, target, functionSig)); - - rolesAuthority.setRoleCapability(role, target, functionSig, false); - assertFalse(rolesAuthority.doesRoleHaveCapability(role, target, functionSig)); - } - - function testSetPublicCapabilities(address target, bytes4 functionSig) public { - assertFalse(rolesAuthority.isCapabilityPublic(target, functionSig)); - - rolesAuthority.setPublicCapability(target, functionSig, true); - assertTrue(rolesAuthority.isCapabilityPublic(target, functionSig)); - - rolesAuthority.setPublicCapability(target, functionSig, false); - assertFalse(rolesAuthority.isCapabilityPublic(target, functionSig)); - } - - function testCanCallWithAuthorizedRole( - address user, - uint8 role, - address target, - bytes4 functionSig - ) public { - assertFalse(rolesAuthority.canCall(user, target, functionSig)); - - rolesAuthority.setUserRole(user, role, true); - assertFalse(rolesAuthority.canCall(user, target, functionSig)); - - rolesAuthority.setRoleCapability(role, target, functionSig, true); - assertTrue(rolesAuthority.canCall(user, target, functionSig)); - - rolesAuthority.setRoleCapability(role, target, functionSig, false); - assertFalse(rolesAuthority.canCall(user, target, functionSig)); - - rolesAuthority.setRoleCapability(role, target, functionSig, true); - assertTrue(rolesAuthority.canCall(user, target, functionSig)); - - rolesAuthority.setUserRole(user, role, false); - assertFalse(rolesAuthority.canCall(user, target, functionSig)); - } - - function testCanCallPublicCapability( - address user, - address target, - bytes4 functionSig - ) public { - assertFalse(rolesAuthority.canCall(user, target, functionSig)); - - rolesAuthority.setPublicCapability(target, functionSig, true); - assertTrue(rolesAuthority.canCall(user, target, functionSig)); - - rolesAuthority.setPublicCapability(target, functionSig, false); - assertFalse(rolesAuthority.canCall(user, target, functionSig)); - } -} diff --git a/lib/solmate/src/test/SSTORE2.t.sol b/lib/solmate/src/test/SSTORE2.t.sol deleted file mode 100644 index 5d97ed7..0000000 --- a/lib/solmate/src/test/SSTORE2.t.sol +++ /dev/null @@ -1,152 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; - -import {SSTORE2} from "../utils/SSTORE2.sol"; - -contract SSTORE2Test is DSTestPlus { - function testWriteRead() public { - bytes memory testBytes = abi.encode("this is a test"); - - address pointer = SSTORE2.write(testBytes); - - assertBytesEq(SSTORE2.read(pointer), testBytes); - } - - function testWriteReadFullStartBound() public { - assertBytesEq(SSTORE2.read(SSTORE2.write(hex"11223344"), 0), hex"11223344"); - } - - function testWriteReadCustomStartBound() public { - assertBytesEq(SSTORE2.read(SSTORE2.write(hex"11223344"), 1), hex"223344"); - } - - function testWriteReadFullBoundedRead() public { - bytes memory testBytes = abi.encode("this is a test"); - - assertBytesEq(SSTORE2.read(SSTORE2.write(testBytes), 0, testBytes.length), testBytes); - } - - function testWriteReadCustomBounds() public { - assertBytesEq(SSTORE2.read(SSTORE2.write(hex"11223344"), 1, 3), hex"2233"); - } - - function testWriteReadEmptyBound() public { - SSTORE2.read(SSTORE2.write(hex"11223344"), 3, 3); - } - - function testFailReadInvalidPointer() public view { - SSTORE2.read(DEAD_ADDRESS); - } - - function testFailReadInvalidPointerCustomStartBound() public view { - SSTORE2.read(DEAD_ADDRESS, 1); - } - - function testFailReadInvalidPointerCustomBounds() public view { - SSTORE2.read(DEAD_ADDRESS, 2, 4); - } - - function testFailWriteReadOutOfStartBound() public { - SSTORE2.read(SSTORE2.write(hex"11223344"), 41000); - } - - function testFailWriteReadEmptyOutOfBounds() public { - SSTORE2.read(SSTORE2.write(hex"11223344"), 42000, 42000); - } - - function testFailWriteReadOutOfBounds() public { - SSTORE2.read(SSTORE2.write(hex"11223344"), 41000, 42000); - } - - function testWriteRead(bytes calldata testBytes, bytes calldata brutalizeWith) - public - brutalizeMemory(brutalizeWith) - { - assertBytesEq(SSTORE2.read(SSTORE2.write(testBytes)), testBytes); - } - - function testWriteReadCustomStartBound( - bytes calldata testBytes, - uint256 startIndex, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - if (testBytes.length == 0) return; - - startIndex = bound(startIndex, 0, testBytes.length); - - assertBytesEq(SSTORE2.read(SSTORE2.write(testBytes), startIndex), bytes(testBytes[startIndex:])); - } - - function testWriteReadCustomBounds( - bytes calldata testBytes, - uint256 startIndex, - uint256 endIndex, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - if (testBytes.length == 0) return; - - endIndex = bound(endIndex, 0, testBytes.length); - startIndex = bound(startIndex, 0, testBytes.length); - - if (startIndex > endIndex) return; - - assertBytesEq( - SSTORE2.read(SSTORE2.write(testBytes), startIndex, endIndex), - bytes(testBytes[startIndex:endIndex]) - ); - } - - function testFailReadInvalidPointer(address pointer, bytes calldata brutalizeWith) - public - view - brutalizeMemory(brutalizeWith) - { - if (pointer.code.length > 0) revert(); - - SSTORE2.read(pointer); - } - - function testFailReadInvalidPointerCustomStartBound( - address pointer, - uint256 startIndex, - bytes calldata brutalizeWith - ) public view brutalizeMemory(brutalizeWith) { - if (pointer.code.length > 0) revert(); - - SSTORE2.read(pointer, startIndex); - } - - function testFailReadInvalidPointerCustomBounds( - address pointer, - uint256 startIndex, - uint256 endIndex, - bytes calldata brutalizeWith - ) public view brutalizeMemory(brutalizeWith) { - if (pointer.code.length > 0) revert(); - - SSTORE2.read(pointer, startIndex, endIndex); - } - - function testFailWriteReadCustomStartBoundOutOfRange( - bytes calldata testBytes, - uint256 startIndex, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - startIndex = bound(startIndex, testBytes.length + 1, type(uint256).max); - - SSTORE2.read(SSTORE2.write(testBytes), startIndex); - } - - function testFailWriteReadCustomBoundsOutOfRange( - bytes calldata testBytes, - uint256 startIndex, - uint256 endIndex, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - endIndex = bound(endIndex, testBytes.length + 1, type(uint256).max); - - SSTORE2.read(SSTORE2.write(testBytes), startIndex, endIndex); - } -} diff --git a/lib/solmate/src/test/SafeCastLib.t.sol b/lib/solmate/src/test/SafeCastLib.t.sol deleted file mode 100644 index 52a6c07..0000000 --- a/lib/solmate/src/test/SafeCastLib.t.sol +++ /dev/null @@ -1,644 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; - -import {SafeCastLib} from "../utils/SafeCastLib.sol"; - -contract SafeCastLibTest is DSTestPlus { - function testSafeCastTo248() public { - assertEq(SafeCastLib.safeCastTo248(2.5e45), 2.5e45); - assertEq(SafeCastLib.safeCastTo248(2.5e27), 2.5e27); - } - - function testSafeCastTo240() public { - assertEq(SafeCastLib.safeCastTo240(2.5e45), 2.5e45); - assertEq(SafeCastLib.safeCastTo240(2.5e27), 2.5e27); - } - - function testSafeCastTo232() public { - assertEq(SafeCastLib.safeCastTo232(2.5e45), 2.5e45); - assertEq(SafeCastLib.safeCastTo232(2.5e27), 2.5e27); - } - - function testSafeCastTo224() public { - assertEq(SafeCastLib.safeCastTo224(2.5e36), 2.5e36); - assertEq(SafeCastLib.safeCastTo224(2.5e27), 2.5e27); - } - - function testSafeCastTo216() public { - assertEq(SafeCastLib.safeCastTo216(2.5e36), 2.5e36); - assertEq(SafeCastLib.safeCastTo216(2.5e27), 2.5e27); - } - - function testSafeCastTo208() public { - assertEq(SafeCastLib.safeCastTo208(2.5e36), 2.5e36); - assertEq(SafeCastLib.safeCastTo208(2.5e27), 2.5e27); - } - - function testSafeCastTo200() public { - assertEq(SafeCastLib.safeCastTo200(2.5e36), 2.5e36); - assertEq(SafeCastLib.safeCastTo200(2.5e27), 2.5e27); - } - - function testSafeCastTo192() public { - assertEq(SafeCastLib.safeCastTo192(2.5e36), 2.5e36); - assertEq(SafeCastLib.safeCastTo192(2.5e27), 2.5e27); - } - - function testSafeCastTo184() public { - assertEq(SafeCastLib.safeCastTo184(2.5e36), 2.5e36); - assertEq(SafeCastLib.safeCastTo184(2.5e27), 2.5e27); - } - - function testSafeCastTo176() public { - assertEq(SafeCastLib.safeCastTo176(2.5e36), 2.5e36); - assertEq(SafeCastLib.safeCastTo176(2.5e27), 2.5e27); - } - - function testSafeCastTo168() public { - assertEq(SafeCastLib.safeCastTo168(2.5e36), 2.5e36); - assertEq(SafeCastLib.safeCastTo168(2.5e27), 2.5e27); - } - - function testSafeCastTo160() public { - assertEq(SafeCastLib.safeCastTo160(2.5e36), 2.5e36); - assertEq(SafeCastLib.safeCastTo160(2.5e27), 2.5e27); - } - - function testSafeCastTo152() public { - assertEq(SafeCastLib.safeCastTo152(2.5e36), 2.5e36); - assertEq(SafeCastLib.safeCastTo152(2.5e27), 2.5e27); - } - - function testSafeCastTo144() public { - assertEq(SafeCastLib.safeCastTo144(2.5e36), 2.5e36); - assertEq(SafeCastLib.safeCastTo144(2.5e27), 2.5e27); - } - - function testSafeCastTo136() public { - assertEq(SafeCastLib.safeCastTo136(2.5e36), 2.5e36); - assertEq(SafeCastLib.safeCastTo136(2.5e27), 2.5e27); - } - - function testSafeCastTo128() public { - assertEq(SafeCastLib.safeCastTo128(2.5e27), 2.5e27); - assertEq(SafeCastLib.safeCastTo128(2.5e18), 2.5e18); - } - - function testSafeCastTo120() public { - assertEq(SafeCastLib.safeCastTo120(2.5e27), 2.5e27); - assertEq(SafeCastLib.safeCastTo120(2.5e18), 2.5e18); - } - - function testSafeCastTo112() public { - assertEq(SafeCastLib.safeCastTo112(2.5e27), 2.5e27); - assertEq(SafeCastLib.safeCastTo112(2.5e18), 2.5e18); - } - - function testSafeCastTo104() public { - assertEq(SafeCastLib.safeCastTo104(2.5e27), 2.5e27); - assertEq(SafeCastLib.safeCastTo104(2.5e18), 2.5e18); - } - - function testSafeCastTo96() public { - assertEq(SafeCastLib.safeCastTo96(2.5e18), 2.5e18); - assertEq(SafeCastLib.safeCastTo96(2.5e17), 2.5e17); - } - - function testSafeCastTo64() public { - assertEq(SafeCastLib.safeCastTo64(2.5e18), 2.5e18); - assertEq(SafeCastLib.safeCastTo64(2.5e17), 2.5e17); - } - - function testSafeCastTo56() public { - assertEq(SafeCastLib.safeCastTo56(2.5e16), 2.5e16); - assertEq(SafeCastLib.safeCastTo56(2.5e15), 2.5e15); - } - - function testSafeCastTo48() public { - assertEq(SafeCastLib.safeCastTo48(2.5e12), 2.5e12); - assertEq(SafeCastLib.safeCastTo48(2.5e11), 2.5e11); - } - - function testSafeCastTo40() public { - assertEq(SafeCastLib.safeCastTo40(2.5e10), 2.5e10); - assertEq(SafeCastLib.safeCastTo40(2.5e9), 2.5e9); - } - - function testSafeCastTo32() public { - assertEq(SafeCastLib.safeCastTo32(2.5e8), 2.5e8); - assertEq(SafeCastLib.safeCastTo32(2.5e7), 2.5e7); - } - - function testSafeCastTo24() public { - assertEq(SafeCastLib.safeCastTo24(2.5e4), 2.5e4); - assertEq(SafeCastLib.safeCastTo24(2.5e3), 2.5e3); - } - - function testSafeCastTo16() public { - assertEq(SafeCastLib.safeCastTo16(2.5e3), 2.5e3); - assertEq(SafeCastLib.safeCastTo16(2.5e2), 2.5e2); - } - - function testSafeCastTo8() public { - assertEq(SafeCastLib.safeCastTo8(100), 100); - assertEq(SafeCastLib.safeCastTo8(250), 250); - } - - function testFailSafeCastTo248() public pure { - SafeCastLib.safeCastTo248(type(uint248).max + 1); - } - - function testFailSafeCastTo240() public pure { - SafeCastLib.safeCastTo240(type(uint240).max + 1); - } - - function testFailSafeCastTo232() public pure { - SafeCastLib.safeCastTo232(type(uint232).max + 1); - } - - function testFailSafeCastTo224() public pure { - SafeCastLib.safeCastTo224(type(uint224).max + 1); - } - - function testFailSafeCastTo216() public pure { - SafeCastLib.safeCastTo216(type(uint216).max + 1); - } - - function testFailSafeCastTo208() public pure { - SafeCastLib.safeCastTo208(type(uint208).max + 1); - } - - function testFailSafeCastTo200() public pure { - SafeCastLib.safeCastTo200(type(uint200).max + 1); - } - - function testFailSafeCastTo192() public pure { - SafeCastLib.safeCastTo192(type(uint192).max + 1); - } - - function testFailSafeCastTo184() public pure { - SafeCastLib.safeCastTo184(type(uint184).max + 1); - } - - function testFailSafeCastTo176() public pure { - SafeCastLib.safeCastTo176(type(uint176).max + 1); - } - - function testFailSafeCastTo168() public pure { - SafeCastLib.safeCastTo168(type(uint168).max + 1); - } - - function testFailSafeCastTo160() public pure { - SafeCastLib.safeCastTo160(type(uint160).max + 1); - } - - function testFailSafeCastTo152() public pure { - SafeCastLib.safeCastTo152(type(uint152).max + 1); - } - - function testFailSafeCastTo144() public pure { - SafeCastLib.safeCastTo144(type(uint144).max + 1); - } - - function testFailSafeCastTo136() public pure { - SafeCastLib.safeCastTo136(type(uint136).max + 1); - } - - function testFailSafeCastTo128() public pure { - SafeCastLib.safeCastTo128(type(uint128).max + 1); - } - - function testFailSafeCastTo120() public pure { - SafeCastLib.safeCastTo120(type(uint120).max + 1); - } - - function testFailSafeCastTo112() public pure { - SafeCastLib.safeCastTo112(type(uint112).max + 1); - } - - function testFailSafeCastTo104() public pure { - SafeCastLib.safeCastTo104(type(uint104).max + 1); - } - - function testFailSafeCastTo96() public pure { - SafeCastLib.safeCastTo96(type(uint96).max + 1); - } - - function testFailSafeCastTo88() public pure { - SafeCastLib.safeCastTo88(type(uint88).max + 1); - } - - function testFailSafeCastTo80() public pure { - SafeCastLib.safeCastTo80(type(uint80).max + 1); - } - - function testFailSafeCastTo72() public pure { - SafeCastLib.safeCastTo72(type(uint72).max + 1); - } - - function testFailSafeCastTo64() public pure { - SafeCastLib.safeCastTo64(type(uint64).max + 1); - } - - function testFailSafeCastTo56() public pure { - SafeCastLib.safeCastTo56(type(uint56).max + 1); - } - - function testFailSafeCastTo48() public pure { - SafeCastLib.safeCastTo48(type(uint48).max + 1); - } - - function testFailSafeCastTo40() public pure { - SafeCastLib.safeCastTo40(type(uint40).max + 1); - } - - function testFailSafeCastTo32() public pure { - SafeCastLib.safeCastTo32(type(uint32).max + 1); - } - - function testFailSafeCastTo24() public pure { - SafeCastLib.safeCastTo24(type(uint24).max + 1); - } - - function testFailSafeCastTo16() public pure { - SafeCastLib.safeCastTo16(type(uint16).max + 1); - } - - function testFailSafeCastTo8() public pure { - SafeCastLib.safeCastTo8(type(uint8).max + 1); - } - - function testSafeCastTo248(uint256 x) public { - x = bound(x, 0, type(uint248).max); - - assertEq(SafeCastLib.safeCastTo248(x), x); - } - - function testSafeCastTo240(uint256 x) public { - x = bound(x, 0, type(uint240).max); - - assertEq(SafeCastLib.safeCastTo240(x), x); - } - - function testSafeCastTo232(uint256 x) public { - x = bound(x, 0, type(uint232).max); - - assertEq(SafeCastLib.safeCastTo232(x), x); - } - - function testSafeCastTo224(uint256 x) public { - x = bound(x, 0, type(uint224).max); - - assertEq(SafeCastLib.safeCastTo224(x), x); - } - - function testSafeCastTo216(uint256 x) public { - x = bound(x, 0, type(uint216).max); - - assertEq(SafeCastLib.safeCastTo216(x), x); - } - - function testSafeCastTo208(uint256 x) public { - x = bound(x, 0, type(uint208).max); - - assertEq(SafeCastLib.safeCastTo208(x), x); - } - - function testSafeCastTo200(uint256 x) public { - x = bound(x, 0, type(uint200).max); - - assertEq(SafeCastLib.safeCastTo200(x), x); - } - - function testSafeCastTo192(uint256 x) public { - x = bound(x, 0, type(uint192).max); - - assertEq(SafeCastLib.safeCastTo192(x), x); - } - - function testSafeCastTo184(uint256 x) public { - x = bound(x, 0, type(uint184).max); - - assertEq(SafeCastLib.safeCastTo184(x), x); - } - - function testSafeCastTo176(uint256 x) public { - x = bound(x, 0, type(uint176).max); - - assertEq(SafeCastLib.safeCastTo176(x), x); - } - - function testSafeCastTo168(uint256 x) public { - x = bound(x, 0, type(uint168).max); - - assertEq(SafeCastLib.safeCastTo168(x), x); - } - - function testSafeCastTo160(uint256 x) public { - x = bound(x, 0, type(uint160).max); - - assertEq(SafeCastLib.safeCastTo160(x), x); - } - - function testSafeCastTo152(uint256 x) public { - x = bound(x, 0, type(uint152).max); - - assertEq(SafeCastLib.safeCastTo152(x), x); - } - - function testSafeCastTo144(uint256 x) public { - x = bound(x, 0, type(uint144).max); - - assertEq(SafeCastLib.safeCastTo144(x), x); - } - - function testSafeCastTo136(uint256 x) public { - x = bound(x, 0, type(uint136).max); - - assertEq(SafeCastLib.safeCastTo136(x), x); - } - - function testSafeCastTo128(uint256 x) public { - x = bound(x, 0, type(uint128).max); - - assertEq(SafeCastLib.safeCastTo128(x), x); - } - - function testSafeCastTo120(uint256 x) public { - x = bound(x, 0, type(uint120).max); - - assertEq(SafeCastLib.safeCastTo120(x), x); - } - - function testSafeCastTo112(uint256 x) public { - x = bound(x, 0, type(uint112).max); - - assertEq(SafeCastLib.safeCastTo112(x), x); - } - - function testSafeCastTo104(uint256 x) public { - x = bound(x, 0, type(uint104).max); - - assertEq(SafeCastLib.safeCastTo104(x), x); - } - - function testSafeCastTo96(uint256 x) public { - x = bound(x, 0, type(uint96).max); - - assertEq(SafeCastLib.safeCastTo96(x), x); - } - - function testSafeCastTo88(uint256 x) public { - x = bound(x, 0, type(uint88).max); - - assertEq(SafeCastLib.safeCastTo88(x), x); - } - - function testSafeCastTo80(uint256 x) public { - x = bound(x, 0, type(uint80).max); - - assertEq(SafeCastLib.safeCastTo80(x), x); - } - - function testSafeCastTo72(uint256 x) public { - x = bound(x, 0, type(uint72).max); - - assertEq(SafeCastLib.safeCastTo72(x), x); - } - - function testSafeCastTo64(uint256 x) public { - x = bound(x, 0, type(uint64).max); - - assertEq(SafeCastLib.safeCastTo64(x), x); - } - - function testSafeCastTo56(uint256 x) public { - x = bound(x, 0, type(uint56).max); - - assertEq(SafeCastLib.safeCastTo56(x), x); - } - - function testSafeCastTo48(uint256 x) public { - x = bound(x, 0, type(uint48).max); - - assertEq(SafeCastLib.safeCastTo48(x), x); - } - - function testSafeCastTo40(uint256 x) public { - x = bound(x, 0, type(uint40).max); - - assertEq(SafeCastLib.safeCastTo40(x), x); - } - - function testSafeCastTo32(uint256 x) public { - x = bound(x, 0, type(uint32).max); - - assertEq(SafeCastLib.safeCastTo32(x), x); - } - - function testSafeCastTo24(uint256 x) public { - x = bound(x, 0, type(uint24).max); - - assertEq(SafeCastLib.safeCastTo24(x), x); - } - - function testSafeCastTo16(uint256 x) public { - x = bound(x, 0, type(uint16).max); - - assertEq(SafeCastLib.safeCastTo16(x), x); - } - - function testSafeCastTo8(uint256 x) public { - x = bound(x, 0, type(uint8).max); - - assertEq(SafeCastLib.safeCastTo8(x), x); - } - - function testFailSafeCastTo248(uint256 x) public { - x = bound(x, type(uint248).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo248(x); - } - - function testFailSafeCastTo240(uint256 x) public { - x = bound(x, type(uint240).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo240(x); - } - - function testFailSafeCastTo232(uint256 x) public { - x = bound(x, type(uint232).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo232(x); - } - - function testFailSafeCastTo224(uint256 x) public { - x = bound(x, type(uint224).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo224(x); - } - - function testFailSafeCastTo216(uint256 x) public { - x = bound(x, type(uint216).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo216(x); - } - - function testFailSafeCastTo208(uint256 x) public { - x = bound(x, type(uint208).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo208(x); - } - - function testFailSafeCastTo200(uint256 x) public { - x = bound(x, type(uint200).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo200(x); - } - - function testFailSafeCastTo192(uint256 x) public { - x = bound(x, type(uint192).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo192(x); - } - - function testFailSafeCastTo184(uint256 x) public { - x = bound(x, type(uint184).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo184(x); - } - - function testFailSafeCastTo176(uint256 x) public { - x = bound(x, type(uint176).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo176(x); - } - - function testFailSafeCastTo168(uint256 x) public { - x = bound(x, type(uint168).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo168(x); - } - - function testFailSafeCastTo160(uint256 x) public { - x = bound(x, type(uint160).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo160(x); - } - - function testFailSafeCastTo152(uint256 x) public { - x = bound(x, type(uint152).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo152(x); - } - - function testFailSafeCastTo144(uint256 x) public { - x = bound(x, type(uint144).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo144(x); - } - - function testFailSafeCastTo136(uint256 x) public { - x = bound(x, type(uint136).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo136(x); - } - - function testFailSafeCastTo128(uint256 x) public { - x = bound(x, type(uint128).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo128(x); - } - - function testFailSafeCastTo120(uint256 x) public { - x = bound(x, type(uint120).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo120(x); - } - - function testFailSafeCastTo112(uint256 x) public { - x = bound(x, type(uint112).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo112(x); - } - - function testFailSafeCastTo104(uint256 x) public { - x = bound(x, type(uint104).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo104(x); - } - - function testFailSafeCastTo96(uint256 x) public { - x = bound(x, type(uint96).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo96(x); - } - - function testFailSafeCastTo88(uint256 x) public { - x = bound(x, type(uint88).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo88(x); - } - - function testFailSafeCastTo80(uint256 x) public { - x = bound(x, type(uint80).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo80(x); - } - - function testFailSafeCastTo72(uint256 x) public { - x = bound(x, type(uint72).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo72(x); - } - - function testFailSafeCastTo64(uint256 x) public { - x = bound(x, type(uint64).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo64(x); - } - - function testFailSafeCastTo56(uint256 x) public { - x = bound(x, type(uint56).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo56(x); - } - - function testFailSafeCastTo48(uint256 x) public { - x = bound(x, type(uint48).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo48(x); - } - - function testFailSafeCastTo40(uint256 x) public { - x = bound(x, type(uint40).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo40(x); - } - - function testFailSafeCastTo32(uint256 x) public { - x = bound(x, type(uint32).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo32(x); - } - - function testFailSafeCastTo24(uint256 x) public { - x = bound(x, type(uint24).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo24(x); - } - - function testFailSafeCastTo16(uint256 x) public { - x = bound(x, type(uint16).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo16(x); - } - - function testFailSafeCastTo8(uint256 x) public { - x = bound(x, type(uint8).max + 1, type(uint256).max); - - SafeCastLib.safeCastTo8(x); - } -} diff --git a/lib/solmate/src/test/SafeTransferLib.t.sol b/lib/solmate/src/test/SafeTransferLib.t.sol deleted file mode 100644 index b976e9f..0000000 --- a/lib/solmate/src/test/SafeTransferLib.t.sol +++ /dev/null @@ -1,610 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {MockERC20} from "./utils/mocks/MockERC20.sol"; -import {RevertingToken} from "./utils/weird-tokens/RevertingToken.sol"; -import {ReturnsTwoToken} from "./utils/weird-tokens/ReturnsTwoToken.sol"; -import {ReturnsFalseToken} from "./utils/weird-tokens/ReturnsFalseToken.sol"; -import {MissingReturnToken} from "./utils/weird-tokens/MissingReturnToken.sol"; -import {ReturnsTooMuchToken} from "./utils/weird-tokens/ReturnsTooMuchToken.sol"; -import {ReturnsGarbageToken} from "./utils/weird-tokens/ReturnsGarbageToken.sol"; -import {ReturnsTooLittleToken} from "./utils/weird-tokens/ReturnsTooLittleToken.sol"; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; - -import {ERC20} from "../tokens/ERC20.sol"; -import {SafeTransferLib} from "../utils/SafeTransferLib.sol"; - -contract SafeTransferLibTest is DSTestPlus { - RevertingToken reverting; - ReturnsTwoToken returnsTwo; - ReturnsFalseToken returnsFalse; - MissingReturnToken missingReturn; - ReturnsTooMuchToken returnsTooMuch; - ReturnsGarbageToken returnsGarbage; - ReturnsTooLittleToken returnsTooLittle; - - MockERC20 erc20; - - function setUp() public { - reverting = new RevertingToken(); - returnsTwo = new ReturnsTwoToken(); - returnsFalse = new ReturnsFalseToken(); - missingReturn = new MissingReturnToken(); - returnsTooMuch = new ReturnsTooMuchToken(); - returnsGarbage = new ReturnsGarbageToken(); - returnsTooLittle = new ReturnsTooLittleToken(); - - erc20 = new MockERC20("StandardToken", "ST", 18); - erc20.mint(address(this), type(uint256).max); - } - - function testTransferWithMissingReturn() public { - verifySafeTransfer(address(missingReturn), address(0xBEEF), 1e18); - } - - function testTransferWithStandardERC20() public { - verifySafeTransfer(address(erc20), address(0xBEEF), 1e18); - } - - function testTransferWithReturnsTooMuch() public { - verifySafeTransfer(address(returnsTooMuch), address(0xBEEF), 1e18); - } - - function testTransferWithNonContract() public { - SafeTransferLib.safeTransfer(ERC20(address(0xBADBEEF)), address(0xBEEF), 1e18); - } - - function testTransferFromWithMissingReturn() public { - verifySafeTransferFrom(address(missingReturn), address(0xFEED), address(0xBEEF), 1e18); - } - - function testTransferFromWithStandardERC20() public { - verifySafeTransferFrom(address(erc20), address(0xFEED), address(0xBEEF), 1e18); - } - - function testTransferFromWithReturnsTooMuch() public { - verifySafeTransferFrom(address(returnsTooMuch), address(0xFEED), address(0xBEEF), 1e18); - } - - function testTransferFromWithNonContract() public { - SafeTransferLib.safeTransferFrom(ERC20(address(0xBADBEEF)), address(0xFEED), address(0xBEEF), 1e18); - } - - function testApproveWithMissingReturn() public { - verifySafeApprove(address(missingReturn), address(0xBEEF), 1e18); - } - - function testApproveWithStandardERC20() public { - verifySafeApprove(address(erc20), address(0xBEEF), 1e18); - } - - function testApproveWithReturnsTooMuch() public { - verifySafeApprove(address(returnsTooMuch), address(0xBEEF), 1e18); - } - - function testApproveWithNonContract() public { - SafeTransferLib.safeApprove(ERC20(address(0xBADBEEF)), address(0xBEEF), 1e18); - } - - function testTransferETH() public { - SafeTransferLib.safeTransferETH(address(0xBEEF), 1e18); - } - - function testFailTransferWithReturnsFalse() public { - verifySafeTransfer(address(returnsFalse), address(0xBEEF), 1e18); - } - - function testFailTransferWithReverting() public { - verifySafeTransfer(address(reverting), address(0xBEEF), 1e18); - } - - function testFailTransferWithReturnsTooLittle() public { - verifySafeTransfer(address(returnsTooLittle), address(0xBEEF), 1e18); - } - - function testFailTransferFromWithReturnsFalse() public { - verifySafeTransferFrom(address(returnsFalse), address(0xFEED), address(0xBEEF), 1e18); - } - - function testFailTransferFromWithReverting() public { - verifySafeTransferFrom(address(reverting), address(0xFEED), address(0xBEEF), 1e18); - } - - function testFailTransferFromWithReturnsTooLittle() public { - verifySafeTransferFrom(address(returnsTooLittle), address(0xFEED), address(0xBEEF), 1e18); - } - - function testFailApproveWithReturnsFalse() public { - verifySafeApprove(address(returnsFalse), address(0xBEEF), 1e18); - } - - function testFailApproveWithReverting() public { - verifySafeApprove(address(reverting), address(0xBEEF), 1e18); - } - - function testFailApproveWithReturnsTooLittle() public { - verifySafeApprove(address(returnsTooLittle), address(0xBEEF), 1e18); - } - - function testTransferWithMissingReturn( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransfer(address(missingReturn), to, amount); - } - - function testTransferWithStandardERC20( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransfer(address(erc20), to, amount); - } - - function testTransferWithReturnsTooMuch( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransfer(address(returnsTooMuch), to, amount); - } - - function testTransferWithGarbage( - address to, - uint256 amount, - bytes memory garbage, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - if ( - (garbage.length < 32 || - (garbage[0] != 0 || - garbage[1] != 0 || - garbage[2] != 0 || - garbage[3] != 0 || - garbage[4] != 0 || - garbage[5] != 0 || - garbage[6] != 0 || - garbage[7] != 0 || - garbage[8] != 0 || - garbage[9] != 0 || - garbage[10] != 0 || - garbage[11] != 0 || - garbage[12] != 0 || - garbage[13] != 0 || - garbage[14] != 0 || - garbage[15] != 0 || - garbage[16] != 0 || - garbage[17] != 0 || - garbage[18] != 0 || - garbage[19] != 0 || - garbage[20] != 0 || - garbage[21] != 0 || - garbage[22] != 0 || - garbage[23] != 0 || - garbage[24] != 0 || - garbage[25] != 0 || - garbage[26] != 0 || - garbage[27] != 0 || - garbage[28] != 0 || - garbage[29] != 0 || - garbage[30] != 0 || - garbage[31] != bytes1(0x01))) && garbage.length != 0 - ) return; - - returnsGarbage.setGarbage(garbage); - - verifySafeTransfer(address(returnsGarbage), to, amount); - } - - function testTransferWithNonContract( - address nonContract, - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - if (uint256(uint160(nonContract)) <= 18 || nonContract.code.length > 0) return; - - SafeTransferLib.safeTransfer(ERC20(nonContract), to, amount); - } - - function testFailTransferETHToContractWithoutFallback() public { - SafeTransferLib.safeTransferETH(address(this), 1e18); - } - - function testTransferFromWithMissingReturn( - address from, - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransferFrom(address(missingReturn), from, to, amount); - } - - function testTransferFromWithStandardERC20( - address from, - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransferFrom(address(erc20), from, to, amount); - } - - function testTransferFromWithReturnsTooMuch( - address from, - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransferFrom(address(returnsTooMuch), from, to, amount); - } - - function testTransferFromWithGarbage( - address from, - address to, - uint256 amount, - bytes memory garbage, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - if ( - (garbage.length < 32 || - (garbage[0] != 0 || - garbage[1] != 0 || - garbage[2] != 0 || - garbage[3] != 0 || - garbage[4] != 0 || - garbage[5] != 0 || - garbage[6] != 0 || - garbage[7] != 0 || - garbage[8] != 0 || - garbage[9] != 0 || - garbage[10] != 0 || - garbage[11] != 0 || - garbage[12] != 0 || - garbage[13] != 0 || - garbage[14] != 0 || - garbage[15] != 0 || - garbage[16] != 0 || - garbage[17] != 0 || - garbage[18] != 0 || - garbage[19] != 0 || - garbage[20] != 0 || - garbage[21] != 0 || - garbage[22] != 0 || - garbage[23] != 0 || - garbage[24] != 0 || - garbage[25] != 0 || - garbage[26] != 0 || - garbage[27] != 0 || - garbage[28] != 0 || - garbage[29] != 0 || - garbage[30] != 0 || - garbage[31] != bytes1(0x01))) && garbage.length != 0 - ) return; - - returnsGarbage.setGarbage(garbage); - - verifySafeTransferFrom(address(returnsGarbage), from, to, amount); - } - - function testTransferFromWithNonContract( - address nonContract, - address from, - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - if (uint256(uint160(nonContract)) <= 18 || nonContract.code.length > 0) return; - - SafeTransferLib.safeTransferFrom(ERC20(nonContract), from, to, amount); - } - - function testApproveWithMissingReturn( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeApprove(address(missingReturn), to, amount); - } - - function testApproveWithStandardERC20( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeApprove(address(erc20), to, amount); - } - - function testApproveWithReturnsTooMuch( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeApprove(address(returnsTooMuch), to, amount); - } - - function testApproveWithGarbage( - address to, - uint256 amount, - bytes memory garbage, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - if ( - (garbage.length < 32 || - (garbage[0] != 0 || - garbage[1] != 0 || - garbage[2] != 0 || - garbage[3] != 0 || - garbage[4] != 0 || - garbage[5] != 0 || - garbage[6] != 0 || - garbage[7] != 0 || - garbage[8] != 0 || - garbage[9] != 0 || - garbage[10] != 0 || - garbage[11] != 0 || - garbage[12] != 0 || - garbage[13] != 0 || - garbage[14] != 0 || - garbage[15] != 0 || - garbage[16] != 0 || - garbage[17] != 0 || - garbage[18] != 0 || - garbage[19] != 0 || - garbage[20] != 0 || - garbage[21] != 0 || - garbage[22] != 0 || - garbage[23] != 0 || - garbage[24] != 0 || - garbage[25] != 0 || - garbage[26] != 0 || - garbage[27] != 0 || - garbage[28] != 0 || - garbage[29] != 0 || - garbage[30] != 0 || - garbage[31] != bytes1(0x01))) && garbage.length != 0 - ) return; - - returnsGarbage.setGarbage(garbage); - - verifySafeApprove(address(returnsGarbage), to, amount); - } - - function testApproveWithNonContract( - address nonContract, - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - if (uint256(uint160(nonContract)) <= 18 || nonContract.code.length > 0) return; - - SafeTransferLib.safeApprove(ERC20(nonContract), to, amount); - } - - function testTransferETH( - address recipient, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - // Transferring to msg.sender can fail because it's possible to overflow their ETH balance as it begins non-zero. - if (recipient.code.length > 0 || uint256(uint160(recipient)) <= 18 || recipient == msg.sender) return; - - amount = bound(amount, 0, address(this).balance); - - SafeTransferLib.safeTransferETH(recipient, amount); - } - - function testFailTransferWithReturnsFalse( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransfer(address(returnsFalse), to, amount); - } - - function testFailTransferWithReverting( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransfer(address(reverting), to, amount); - } - - function testFailTransferWithReturnsTooLittle( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransfer(address(returnsTooLittle), to, amount); - } - - function testFailTransferWithReturnsTwo( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransfer(address(returnsTwo), to, amount); - } - - function testFailTransferWithGarbage( - address to, - uint256 amount, - bytes memory garbage, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - require(garbage.length != 0 && (garbage.length < 32 || garbage[31] != bytes1(0x01))); - - returnsGarbage.setGarbage(garbage); - - verifySafeTransfer(address(returnsGarbage), to, amount); - } - - function testFailTransferFromWithReturnsFalse( - address from, - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransferFrom(address(returnsFalse), from, to, amount); - } - - function testFailTransferFromWithReverting( - address from, - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransferFrom(address(reverting), from, to, amount); - } - - function testFailTransferFromWithReturnsTooLittle( - address from, - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransferFrom(address(returnsTooLittle), from, to, amount); - } - - function testFailTransferFromWithReturnsTwo( - address from, - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeTransferFrom(address(returnsTwo), from, to, amount); - } - - function testFailTransferFromWithGarbage( - address from, - address to, - uint256 amount, - bytes memory garbage, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - require(garbage.length != 0 && (garbage.length < 32 || garbage[31] != bytes1(0x01))); - - returnsGarbage.setGarbage(garbage); - - verifySafeTransferFrom(address(returnsGarbage), from, to, amount); - } - - function testFailApproveWithReturnsFalse( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeApprove(address(returnsFalse), to, amount); - } - - function testFailApproveWithReverting( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeApprove(address(reverting), to, amount); - } - - function testFailApproveWithReturnsTooLittle( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeApprove(address(returnsTooLittle), to, amount); - } - - function testFailApproveWithReturnsTwo( - address to, - uint256 amount, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - verifySafeApprove(address(returnsTwo), to, amount); - } - - function testFailApproveWithGarbage( - address to, - uint256 amount, - bytes memory garbage, - bytes calldata brutalizeWith - ) public brutalizeMemory(brutalizeWith) { - require(garbage.length != 0 && (garbage.length < 32 || garbage[31] != bytes1(0x01))); - - returnsGarbage.setGarbage(garbage); - - verifySafeApprove(address(returnsGarbage), to, amount); - } - - function testFailTransferETHToContractWithoutFallback(uint256 amount, bytes calldata brutalizeWith) - public - brutalizeMemory(brutalizeWith) - { - SafeTransferLib.safeTransferETH(address(this), amount); - } - - function verifySafeTransfer( - address token, - address to, - uint256 amount - ) internal { - uint256 preBal = ERC20(token).balanceOf(to); - SafeTransferLib.safeTransfer(ERC20(address(token)), to, amount); - uint256 postBal = ERC20(token).balanceOf(to); - - if (to == address(this)) { - assertEq(preBal, postBal); - } else { - assertEq(postBal - preBal, amount); - } - } - - function verifySafeTransferFrom( - address token, - address from, - address to, - uint256 amount - ) internal { - forceApprove(token, from, address(this), amount); - - // We cast to MissingReturnToken here because it won't check - // that there was return data, which accommodates all tokens. - MissingReturnToken(token).transfer(from, amount); - - uint256 preBal = ERC20(token).balanceOf(to); - SafeTransferLib.safeTransferFrom(ERC20(token), from, to, amount); - uint256 postBal = ERC20(token).balanceOf(to); - - if (from == to) { - assertEq(preBal, postBal); - } else { - assertEq(postBal - preBal, amount); - } - } - - function verifySafeApprove( - address token, - address to, - uint256 amount - ) internal { - SafeTransferLib.safeApprove(ERC20(address(token)), to, amount); - - assertEq(ERC20(token).allowance(address(this), to), amount); - } - - function forceApprove( - address token, - address from, - address to, - uint256 amount - ) internal { - uint256 slot = token == address(erc20) ? 4 : 2; // Standard ERC20 name and symbol aren't constant. - - hevm.store( - token, - keccak256(abi.encode(to, keccak256(abi.encode(from, uint256(slot))))), - bytes32(uint256(amount)) - ); - - assertEq(ERC20(token).allowance(from, to), amount, "wrong allowance"); - } -} diff --git a/lib/solmate/src/test/SignedWadMath.t.sol b/lib/solmate/src/test/SignedWadMath.t.sol deleted file mode 100644 index 4849448..0000000 --- a/lib/solmate/src/test/SignedWadMath.t.sol +++ /dev/null @@ -1,74 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity >=0.8.0; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; - -import {wadMul, wadDiv} from "../utils/SignedWadMath.sol"; - -contract SignedWadMathTest is DSTestPlus { - function testWadMul( - uint256 x, - uint256 y, - bool negX, - bool negY - ) public { - x = bound(x, 0, 99999999999999e18); - y = bound(x, 0, 99999999999999e18); - - int256 xPrime = negX ? -int256(x) : int256(x); - int256 yPrime = negY ? -int256(y) : int256(y); - - assertEq(wadMul(xPrime, yPrime), (xPrime * yPrime) / 1e18); - } - - function testFailWadMulEdgeCase() public pure { - int256 x = -1; - int256 y = type(int256).min; - - wadMul(x, y); - } - - function testFailWadMulEdgeCase2() public pure { - int256 x = type(int256).min; - int256 y = -1; - - wadMul(x, y); - } - - function testFailWadMulOverflow(int256 x, int256 y) public pure { - // Ignore cases where x * y does not overflow. - unchecked { - if ((x * y) / x == y) revert(); - } - - wadMul(x, y); - } - - function testWadDiv( - uint256 x, - uint256 y, - bool negX, - bool negY - ) public { - x = bound(x, 0, 99999999e18); - y = bound(x, 1, 99999999e18); - - int256 xPrime = negX ? -int256(x) : int256(x); - int256 yPrime = negY ? -int256(y) : int256(y); - - assertEq(wadDiv(xPrime, yPrime), (xPrime * 1e18) / yPrime); - } - - function testFailWadDivOverflow(int256 x, int256 y) public pure { - // Ignore cases where x * WAD does not overflow or y is 0. - unchecked { - if (y == 0 || (x * 1e18) / 1e18 == x) revert(); - } - - wadDiv(x, y); - } - - function testFailWadDivZeroDenominator(int256 x) public pure { - wadDiv(x, 0); - } -} diff --git a/lib/solmate/src/test/WETH.t.sol b/lib/solmate/src/test/WETH.t.sol deleted file mode 100644 index a13761e..0000000 --- a/lib/solmate/src/test/WETH.t.sol +++ /dev/null @@ -1,146 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity 0.8.15; - -import {DSTestPlus} from "./utils/DSTestPlus.sol"; -import {DSInvariantTest} from "./utils/DSInvariantTest.sol"; - -import {SafeTransferLib} from "../utils/SafeTransferLib.sol"; - -import {WETH} from "../tokens/WETH.sol"; - -contract WETHTest is DSTestPlus { - WETH weth; - - function setUp() public { - weth = new WETH(); - } - - function testFallbackDeposit() public { - assertEq(weth.balanceOf(address(this)), 0); - assertEq(weth.totalSupply(), 0); - - SafeTransferLib.safeTransferETH(address(weth), 1 ether); - - assertEq(weth.balanceOf(address(this)), 1 ether); - assertEq(weth.totalSupply(), 1 ether); - } - - function testDeposit() public { - assertEq(weth.balanceOf(address(this)), 0); - assertEq(weth.totalSupply(), 0); - - weth.deposit{value: 1 ether}(); - - assertEq(weth.balanceOf(address(this)), 1 ether); - assertEq(weth.totalSupply(), 1 ether); - } - - function testWithdraw() public { - uint256 startingBalance = address(this).balance; - - weth.deposit{value: 1 ether}(); - - weth.withdraw(1 ether); - - uint256 balanceAfterWithdraw = address(this).balance; - - assertEq(balanceAfterWithdraw, startingBalance); - assertEq(weth.balanceOf(address(this)), 0); - assertEq(weth.totalSupply(), 0); - } - - function testPartialWithdraw() public { - weth.deposit{value: 1 ether}(); - - uint256 balanceBeforeWithdraw = address(this).balance; - - weth.withdraw(0.5 ether); - - uint256 balanceAfterWithdraw = address(this).balance; - - assertEq(balanceAfterWithdraw, balanceBeforeWithdraw + 0.5 ether); - assertEq(weth.balanceOf(address(this)), 0.5 ether); - assertEq(weth.totalSupply(), 0.5 ether); - } - - function testFallbackDeposit(uint256 amount) public { - amount = bound(amount, 0, address(this).balance); - - assertEq(weth.balanceOf(address(this)), 0); - assertEq(weth.totalSupply(), 0); - - SafeTransferLib.safeTransferETH(address(weth), amount); - - assertEq(weth.balanceOf(address(this)), amount); - assertEq(weth.totalSupply(), amount); - } - - function testDeposit(uint256 amount) public { - amount = bound(amount, 0, address(this).balance); - - assertEq(weth.balanceOf(address(this)), 0); - assertEq(weth.totalSupply(), 0); - - weth.deposit{value: amount}(); - - assertEq(weth.balanceOf(address(this)), amount); - assertEq(weth.totalSupply(), amount); - } - - function testWithdraw(uint256 depositAmount, uint256 withdrawAmount) public { - depositAmount = bound(depositAmount, 0, address(this).balance); - withdrawAmount = bound(withdrawAmount, 0, depositAmount); - - weth.deposit{value: depositAmount}(); - - uint256 balanceBeforeWithdraw = address(this).balance; - - weth.withdraw(withdrawAmount); - - uint256 balanceAfterWithdraw = address(this).balance; - - assertEq(balanceAfterWithdraw, balanceBeforeWithdraw + withdrawAmount); - assertEq(weth.balanceOf(address(this)), depositAmount - withdrawAmount); - assertEq(weth.totalSupply(), depositAmount - withdrawAmount); - } - - receive() external payable {} -} - -contract WETHInvariants is DSTestPlus, DSInvariantTest { - WETHTester wethTester; - WETH weth; - - function setUp() public { - weth = new WETH(); - wethTester = new WETHTester{value: address(this).balance}(weth); - - addTargetContract(address(wethTester)); - } - - function invariantTotalSupplyEqualsBalance() public { - assertEq(address(weth).balance, weth.totalSupply()); - } -} - -contract WETHTester { - WETH weth; - - constructor(WETH _weth) payable { - weth = _weth; - } - - function deposit(uint256 amount) public { - weth.deposit{value: amount}(); - } - - function fallbackDeposit(uint256 amount) public { - SafeTransferLib.safeTransferETH(address(weth), amount); - } - - function withdraw(uint256 amount) public { - weth.withdraw(amount); - } - - receive() external payable {} -} diff --git a/lib/solmate/src/test/utils/DSInvariantTest.sol b/lib/solmate/src/test/utils/DSInvariantTest.sol deleted file mode 100644 index 820775c..0000000 --- a/lib/solmate/src/test/utils/DSInvariantTest.sol +++ /dev/null @@ -1,16 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -contract DSInvariantTest { - address[] private targets; - - function targetContracts() public view virtual returns (address[] memory) { - require(targets.length > 0, "NO_TARGET_CONTRACTS"); - - return targets; - } - - function addTargetContract(address newTargetContract) internal virtual { - targets.push(newTargetContract); - } -} diff --git a/lib/solmate/src/test/utils/DSTestPlus.sol b/lib/solmate/src/test/utils/DSTestPlus.sol deleted file mode 100644 index b56d4c9..0000000 --- a/lib/solmate/src/test/utils/DSTestPlus.sol +++ /dev/null @@ -1,179 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {DSTest} from "ds-test/test.sol"; - -import {Hevm} from "./Hevm.sol"; - -/// @notice Extended testing framework for DappTools projects. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/test/utils/DSTestPlus.sol) -contract DSTestPlus is DSTest { - Hevm internal constant hevm = Hevm(HEVM_ADDRESS); - - address internal constant DEAD_ADDRESS = 0xDeaDbeefdEAdbeefdEadbEEFdeadbeEFdEaDbeeF; - - string private checkpointLabel; - uint256 private checkpointGasLeft = 1; // Start the slot warm. - - modifier brutalizeMemory(bytes memory brutalizeWith) { - /// @solidity memory-safe-assembly - assembly { - // Fill the 64 bytes of scratch space with the data. - pop( - staticcall( - gas(), // Pass along all the gas in the call. - 0x04, // Call the identity precompile address. - brutalizeWith, // Offset is the bytes' pointer. - 64, // Copy enough to only fill the scratch space. - 0, // Store the return value in the scratch space. - 64 // Scratch space is only 64 bytes in size, we don't want to write further. - ) - ) - - let size := add(mload(brutalizeWith), 32) // Add 32 to include the 32 byte length slot. - - // Fill the free memory pointer's destination with the data. - pop( - staticcall( - gas(), // Pass along all the gas in the call. - 0x04, // Call the identity precompile address. - brutalizeWith, // Offset is the bytes' pointer. - size, // We want to pass the length of the bytes. - mload(0x40), // Store the return value at the free memory pointer. - size // Since the precompile just returns its input, we reuse size. - ) - ) - } - - _; - } - - function startMeasuringGas(string memory label) internal virtual { - checkpointLabel = label; - - checkpointGasLeft = gasleft(); - } - - function stopMeasuringGas() internal virtual { - uint256 checkpointGasLeft2 = gasleft(); - - // Subtract 100 to account for the warm SLOAD in startMeasuringGas. - uint256 gasDelta = checkpointGasLeft - checkpointGasLeft2 - 100; - - emit log_named_uint(string(abi.encodePacked(checkpointLabel, " Gas")), gasDelta); - } - - function fail(string memory err) internal virtual { - emit log_named_string("Error", err); - fail(); - } - - function assertFalse(bool data) internal virtual { - assertTrue(!data); - } - - function assertUint128Eq(uint128 a, uint128 b) internal virtual { - assertEq(uint256(a), uint256(b)); - } - - function assertUint64Eq(uint64 a, uint64 b) internal virtual { - assertEq(uint256(a), uint256(b)); - } - - function assertUint96Eq(uint96 a, uint96 b) internal virtual { - assertEq(uint256(a), uint256(b)); - } - - function assertUint32Eq(uint32 a, uint32 b) internal virtual { - assertEq(uint256(a), uint256(b)); - } - - function assertBoolEq(bool a, bool b) internal virtual { - b ? assertTrue(a) : assertFalse(a); - } - - function assertApproxEq( - uint256 a, - uint256 b, - uint256 maxDelta - ) internal virtual { - uint256 delta = a > b ? a - b : b - a; - - if (delta > maxDelta) { - emit log("Error: a ~= b not satisfied [uint]"); - emit log_named_uint(" Expected", b); - emit log_named_uint(" Actual", a); - emit log_named_uint(" Max Delta", maxDelta); - emit log_named_uint(" Delta", delta); - fail(); - } - } - - function assertRelApproxEq( - uint256 a, - uint256 b, - uint256 maxPercentDelta // An 18 decimal fixed point number, where 1e18 == 100% - ) internal virtual { - if (b == 0) return assertEq(a, b); // If the expected is 0, actual must be too. - - uint256 percentDelta = ((a > b ? a - b : b - a) * 1e18) / b; - - if (percentDelta > maxPercentDelta) { - emit log("Error: a ~= b not satisfied [uint]"); - emit log_named_uint(" Expected", b); - emit log_named_uint(" Actual", a); - emit log_named_decimal_uint(" Max % Delta", maxPercentDelta, 18); - emit log_named_decimal_uint(" % Delta", percentDelta, 18); - fail(); - } - } - - function assertBytesEq(bytes memory a, bytes memory b) internal virtual { - if (keccak256(a) != keccak256(b)) { - emit log("Error: a == b not satisfied [bytes]"); - emit log_named_bytes(" Expected", b); - emit log_named_bytes(" Actual", a); - fail(); - } - } - - function assertUintArrayEq(uint256[] memory a, uint256[] memory b) internal virtual { - require(a.length == b.length, "LENGTH_MISMATCH"); - - for (uint256 i = 0; i < a.length; i++) { - assertEq(a[i], b[i]); - } - } - - function bound( - uint256 x, - uint256 min, - uint256 max - ) internal virtual returns (uint256 result) { - require(max >= min, "MAX_LESS_THAN_MIN"); - - uint256 size = max - min; - - if (size == 0) result = min; - else if (size == type(uint256).max) result = x; - else { - ++size; // Make max inclusive. - uint256 mod = x % size; - result = min + mod; - } - - emit log_named_uint("Bound Result", result); - } - - function min3( - uint256 a, - uint256 b, - uint256 c - ) internal pure returns (uint256) { - return a > b ? (b > c ? c : b) : (a > c ? c : a); - } - - function min2(uint256 a, uint256 b) internal pure returns (uint256) { - return a > b ? b : a; - } -} diff --git a/lib/solmate/src/test/utils/Hevm.sol b/lib/solmate/src/test/utils/Hevm.sol deleted file mode 100644 index 8ca0eff..0000000 --- a/lib/solmate/src/test/utils/Hevm.sol +++ /dev/null @@ -1,107 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -interface Hevm { - /// @notice Sets the block timestamp. - function warp(uint256) external; - - /// @notice Sets the block height. - function roll(uint256) external; - - /// @notice Sets the block base fee. - function fee(uint256) external; - - /// @notice Loads a storage slot from an address. - function load(address, bytes32) external returns (bytes32); - - /// @notice Stores a value to an address' storage slot. - function store( - address, - bytes32, - bytes32 - ) external; - - /// @notice Signs a digest with a private key, returns v r s. - function sign(uint256, bytes32) - external - returns ( - uint8, - bytes32, - bytes32 - ); - - /// @notice Gets address for a given private key. - function addr(uint256) external returns (address); - - /// @notice Performs a foreign function call via a terminal call. - function ffi(string[] calldata) external returns (bytes memory); - - /// @notice Sets the next call's msg.sender to be the input address. - function prank(address) external; - - /// @notice Sets all subsequent calls' msg.sender to be the input address until stopPrank is called. - function startPrank(address) external; - - /// @notice Sets the next call's msg.sender to be the input address and the tx.origin to be the second input. - function prank(address, address) external; - - /// @notice Sets all subsequent calls' msg.sender to be the input address and - /// sets tx.origin to be the second address inputted until stopPrank is called. - function startPrank(address, address) external; - - /// @notice Resets msg.sender to its original value before a prank. - function stopPrank() external; - - /// @notice Sets an address' balance. - function deal(address, uint256) external; - - /// @notice Sets an address' code. - function etch(address, bytes calldata) external; - - /// @notice Expects an error from the next call. - function expectRevert(bytes calldata) external; - - /// @notice Expects a revert from the next call. - function expectRevert(bytes4) external; - - /// @notice Record all storage reads and writes. - function record() external; - - /// @notice Gets all accessed reads and write slots from a recording session, for a given address. - function accesses(address) external returns (bytes32[] memory reads, bytes32[] memory writes); - - /// @notice Prepare an expected log with (bool checkTopic1, bool checkTopic2, bool checkTopic3, bool checkData). - /// @notice Call this function, then emit an event, then call a function. Internally after the call, we check - /// if logs were emitted in the expected order with the expected topics and data as specified by the booleans. - function expectEmit( - bool, - bool, - bool, - bool - ) external; - - /// @notice Mocks the behavior of a contract call, setting the input and output for a function. - /// @notice Calldata can either be strict or a partial match, e.g. if only passed - /// a selector to the expected calldata, then the entire function will be mocked. - function mockCall( - address, - bytes calldata, - bytes calldata - ) external; - - /// @notice Clears all mocked calls. - function clearMockedCalls() external; - - /// @notice Expect a call to an address with the specified calldata. - /// @notice Calldata can either be strict or a partial match. - function expectCall(address, bytes calldata) external; - - /// @notice Fetches the contract bytecode from its artifact file. - function getCode(string calldata) external returns (bytes memory); - - /// @notice Label an address in test traces. - function label(address addr, string calldata label) external; - - /// @notice When fuzzing, generate new inputs if the input conditional is not met. - function assume(bool) external; -} diff --git a/lib/solmate/src/test/utils/mocks/MockAuthChild.sol b/lib/solmate/src/test/utils/mocks/MockAuthChild.sol deleted file mode 100644 index d2c3276..0000000 --- a/lib/solmate/src/test/utils/mocks/MockAuthChild.sol +++ /dev/null @@ -1,12 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {Auth, Authority} from "../../../auth/Auth.sol"; - -contract MockAuthChild is Auth(msg.sender, Authority(address(0))) { - bool public flag; - - function updateFlag() public virtual requiresAuth { - flag = true; - } -} diff --git a/lib/solmate/src/test/utils/mocks/MockAuthority.sol b/lib/solmate/src/test/utils/mocks/MockAuthority.sol deleted file mode 100644 index acb3689..0000000 --- a/lib/solmate/src/test/utils/mocks/MockAuthority.sol +++ /dev/null @@ -1,20 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {Authority} from "../../../auth/Auth.sol"; - -contract MockAuthority is Authority { - bool immutable allowCalls; - - constructor(bool _allowCalls) { - allowCalls = _allowCalls; - } - - function canCall( - address, - address, - bytes4 - ) public view override returns (bool) { - return allowCalls; - } -} diff --git a/lib/solmate/src/test/utils/mocks/MockERC1155.sol b/lib/solmate/src/test/utils/mocks/MockERC1155.sol deleted file mode 100644 index ede086d..0000000 --- a/lib/solmate/src/test/utils/mocks/MockERC1155.sol +++ /dev/null @@ -1,42 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {ERC1155} from "../../../tokens/ERC1155.sol"; - -contract MockERC1155 is ERC1155 { - function uri(uint256) public pure virtual override returns (string memory) {} - - function mint( - address to, - uint256 id, - uint256 amount, - bytes memory data - ) public virtual { - _mint(to, id, amount, data); - } - - function batchMint( - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) public virtual { - _batchMint(to, ids, amounts, data); - } - - function burn( - address from, - uint256 id, - uint256 amount - ) public virtual { - _burn(from, id, amount); - } - - function batchBurn( - address from, - uint256[] memory ids, - uint256[] memory amounts - ) public virtual { - _batchBurn(from, ids, amounts); - } -} diff --git a/lib/solmate/src/test/utils/mocks/MockERC20.sol b/lib/solmate/src/test/utils/mocks/MockERC20.sol deleted file mode 100644 index fbbaef5..0000000 --- a/lib/solmate/src/test/utils/mocks/MockERC20.sol +++ /dev/null @@ -1,20 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {ERC20} from "../../../tokens/ERC20.sol"; - -contract MockERC20 is ERC20 { - constructor( - string memory _name, - string memory _symbol, - uint8 _decimals - ) ERC20(_name, _symbol, _decimals) {} - - function mint(address to, uint256 value) public virtual { - _mint(to, value); - } - - function burn(address from, uint256 value) public virtual { - _burn(from, value); - } -} diff --git a/lib/solmate/src/test/utils/mocks/MockERC4626.sol b/lib/solmate/src/test/utils/mocks/MockERC4626.sol deleted file mode 100644 index edc7d5f..0000000 --- a/lib/solmate/src/test/utils/mocks/MockERC4626.sol +++ /dev/null @@ -1,28 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {ERC20} from "../../../tokens/ERC20.sol"; -import {ERC4626} from "../../../mixins/ERC4626.sol"; - -contract MockERC4626 is ERC4626 { - uint256 public beforeWithdrawHookCalledCounter = 0; - uint256 public afterDepositHookCalledCounter = 0; - - constructor( - ERC20 _underlying, - string memory _name, - string memory _symbol - ) ERC4626(_underlying, _name, _symbol) {} - - function totalAssets() public view override returns (uint256) { - return asset.balanceOf(address(this)); - } - - function beforeWithdraw(uint256, uint256) internal override { - beforeWithdrawHookCalledCounter++; - } - - function afterDeposit(uint256, uint256) internal override { - afterDepositHookCalledCounter++; - } -} diff --git a/lib/solmate/src/test/utils/mocks/MockERC6909.sol b/lib/solmate/src/test/utils/mocks/MockERC6909.sol deleted file mode 100644 index 5f7eabe..0000000 --- a/lib/solmate/src/test/utils/mocks/MockERC6909.sol +++ /dev/null @@ -1,22 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity >=0.8.0; - -import {ERC6909} from "../../../tokens/ERC6909.sol"; - -contract MockERC6909 is ERC6909 { - function mint( - address receiver, - uint256 id, - uint256 amount - ) public virtual { - _mint(receiver, id, amount); - } - - function burn( - address sender, - uint256 id, - uint256 amount - ) public virtual { - _burn(sender, id, amount); - } -} diff --git a/lib/solmate/src/test/utils/mocks/MockERC721.sol b/lib/solmate/src/test/utils/mocks/MockERC721.sol deleted file mode 100644 index 51227c0..0000000 --- a/lib/solmate/src/test/utils/mocks/MockERC721.sol +++ /dev/null @@ -1,30 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {ERC721} from "../../../tokens/ERC721.sol"; - -contract MockERC721 is ERC721 { - constructor(string memory _name, string memory _symbol) ERC721(_name, _symbol) {} - - function tokenURI(uint256) public pure virtual override returns (string memory) {} - - function mint(address to, uint256 tokenId) public virtual { - _mint(to, tokenId); - } - - function burn(uint256 tokenId) public virtual { - _burn(tokenId); - } - - function safeMint(address to, uint256 tokenId) public virtual { - _safeMint(to, tokenId); - } - - function safeMint( - address to, - uint256 tokenId, - bytes memory data - ) public virtual { - _safeMint(to, tokenId, data); - } -} diff --git a/lib/solmate/src/test/utils/mocks/MockOwned.sol b/lib/solmate/src/test/utils/mocks/MockOwned.sol deleted file mode 100644 index 52ef918..0000000 --- a/lib/solmate/src/test/utils/mocks/MockOwned.sol +++ /dev/null @@ -1,12 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {Owned} from "../../../auth/Owned.sol"; - -contract MockOwned is Owned(msg.sender) { - bool public flag; - - function updateFlag() public virtual onlyOwner { - flag = true; - } -} diff --git a/lib/solmate/src/test/utils/weird-tokens/MissingReturnToken.sol b/lib/solmate/src/test/utils/weird-tokens/MissingReturnToken.sol deleted file mode 100644 index 23f4633..0000000 --- a/lib/solmate/src/test/utils/weird-tokens/MissingReturnToken.sol +++ /dev/null @@ -1,83 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -contract MissingReturnToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "MissingReturnToken"; - - string public constant symbol = "MRT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address spender, uint256 amount) public virtual { - allowance[msg.sender][spender] = amount; - - emit Approval(msg.sender, spender, amount); - } - - function transfer(address to, uint256 amount) public virtual { - balanceOf[msg.sender] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(msg.sender, to, amount); - } - - function transferFrom( - address from, - address to, - uint256 amount - ) public virtual { - uint256 allowed = allowance[from][msg.sender]; // Saves gas for limited approvals. - - if (allowed != type(uint256).max) allowance[from][msg.sender] = allowed - amount; - - balanceOf[from] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(from, to, amount); - } -} diff --git a/lib/solmate/src/test/utils/weird-tokens/ReturnsFalseToken.sol b/lib/solmate/src/test/utils/weird-tokens/ReturnsFalseToken.sol deleted file mode 100644 index 8139efe..0000000 --- a/lib/solmate/src/test/utils/weird-tokens/ReturnsFalseToken.sol +++ /dev/null @@ -1,61 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -contract ReturnsFalseToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "ReturnsFalseToken"; - - string public constant symbol = "RFT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address, uint256) public virtual returns (bool) { - return false; - } - - function transfer(address, uint256) public virtual returns (bool) { - return false; - } - - function transferFrom( - address, - address, - uint256 - ) public virtual returns (bool) { - return false; - } -} diff --git a/lib/solmate/src/test/utils/weird-tokens/ReturnsGarbageToken.sol b/lib/solmate/src/test/utils/weird-tokens/ReturnsGarbageToken.sol deleted file mode 100644 index 77c9575..0000000 --- a/lib/solmate/src/test/utils/weird-tokens/ReturnsGarbageToken.sol +++ /dev/null @@ -1,115 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -contract ReturnsGarbageToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "ReturnsGarbageToken"; - - string public constant symbol = "RGT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - MOCK STORAGE - //////////////////////////////////////////////////////////////*/ - - bytes garbage; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address spender, uint256 amount) public virtual { - allowance[msg.sender][spender] = amount; - - emit Approval(msg.sender, spender, amount); - - bytes memory _garbage = garbage; - - assembly { - return(add(_garbage, 32), mload(_garbage)) - } - } - - function transfer(address to, uint256 amount) public virtual { - balanceOf[msg.sender] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(msg.sender, to, amount); - - bytes memory _garbage = garbage; - - assembly { - return(add(_garbage, 32), mload(_garbage)) - } - } - - function transferFrom( - address from, - address to, - uint256 amount - ) public virtual { - uint256 allowed = allowance[from][msg.sender]; // Saves gas for limited approvals. - - if (allowed != type(uint256).max) allowance[from][msg.sender] = allowed - amount; - - balanceOf[from] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(from, to, amount); - - bytes memory _garbage = garbage; - - assembly { - return(add(_garbage, 32), mload(_garbage)) - } - } - - /*/////////////////////////////////////////////////////////////// - MOCK LOGIC - //////////////////////////////////////////////////////////////*/ - - function setGarbage(bytes memory _garbage) public virtual { - garbage = _garbage; - } -} diff --git a/lib/solmate/src/test/utils/weird-tokens/ReturnsTooLittleToken.sol b/lib/solmate/src/test/utils/weird-tokens/ReturnsTooLittleToken.sol deleted file mode 100644 index 69947c3..0000000 --- a/lib/solmate/src/test/utils/weird-tokens/ReturnsTooLittleToken.sol +++ /dev/null @@ -1,70 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -contract ReturnsTooLittleToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "ReturnsTooLittleToken"; - - string public constant symbol = "RTLT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address, uint256) public virtual { - assembly { - mstore(0, 0x0100000000000000000000000000000000000000000000000000000000000000) - return(0, 8) - } - } - - function transfer(address, uint256) public virtual { - assembly { - mstore(0, 0x0100000000000000000000000000000000000000000000000000000000000000) - return(0, 8) - } - } - - function transferFrom( - address, - address, - uint256 - ) public virtual { - assembly { - mstore(0, 0x0100000000000000000000000000000000000000000000000000000000000000) - return(0, 8) - } - } -} diff --git a/lib/solmate/src/test/utils/weird-tokens/ReturnsTooMuchToken.sol b/lib/solmate/src/test/utils/weird-tokens/ReturnsTooMuchToken.sol deleted file mode 100644 index 8774cbb..0000000 --- a/lib/solmate/src/test/utils/weird-tokens/ReturnsTooMuchToken.sol +++ /dev/null @@ -1,98 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -contract ReturnsTooMuchToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "ReturnsTooMuchToken"; - - string public constant symbol = "RTMT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address spender, uint256 amount) public virtual { - allowance[msg.sender][spender] = amount; - - emit Approval(msg.sender, spender, amount); - - assembly { - mstore(0, 1) - return(0, 4096) - } - } - - function transfer(address to, uint256 amount) public virtual { - balanceOf[msg.sender] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(msg.sender, to, amount); - - assembly { - mstore(0, 1) - return(0, 4096) - } - } - - function transferFrom( - address from, - address to, - uint256 amount - ) public virtual { - uint256 allowed = allowance[from][msg.sender]; // Saves gas for limited approvals. - - if (allowed != type(uint256).max) allowance[from][msg.sender] = allowed - amount; - - balanceOf[from] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(from, to, amount); - - assembly { - mstore(0, 1) - return(0, 4096) - } - } -} diff --git a/lib/solmate/src/test/utils/weird-tokens/ReturnsTwoToken.sol b/lib/solmate/src/test/utils/weird-tokens/ReturnsTwoToken.sol deleted file mode 100644 index ac980f8..0000000 --- a/lib/solmate/src/test/utils/weird-tokens/ReturnsTwoToken.sol +++ /dev/null @@ -1,61 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -contract ReturnsTwoToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "ReturnsFalseToken"; - - string public constant symbol = "RTT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address, uint256) public virtual returns (uint256) { - return 2; - } - - function transfer(address, uint256) public virtual returns (uint256) { - return 2; - } - - function transferFrom( - address, - address, - uint256 - ) public virtual returns (uint256) { - return 2; - } -} diff --git a/lib/solmate/src/test/utils/weird-tokens/RevertingToken.sol b/lib/solmate/src/test/utils/weird-tokens/RevertingToken.sol deleted file mode 100644 index 48ac1fa..0000000 --- a/lib/solmate/src/test/utils/weird-tokens/RevertingToken.sol +++ /dev/null @@ -1,61 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -contract RevertingToken { - /*/////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*/////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public constant name = "RevertingToken"; - - string public constant symbol = "RT"; - - uint8 public constant decimals = 18; - - /*/////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*/////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor() { - totalSupply = type(uint256).max; - balanceOf[msg.sender] = type(uint256).max; - } - - /*/////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address, uint256) public virtual { - revert(); - } - - function transfer(address, uint256) public virtual { - revert(); - } - - function transferFrom( - address, - address, - uint256 - ) public virtual { - revert(); - } -} diff --git a/lib/solmate/src/tokens/ERC1155.sol b/lib/solmate/src/tokens/ERC1155.sol deleted file mode 100644 index cff0f02..0000000 --- a/lib/solmate/src/tokens/ERC1155.sol +++ /dev/null @@ -1,257 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -/// @notice Minimalist and gas efficient standard ERC1155 implementation. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC1155.sol) -abstract contract ERC1155 { - /*////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event TransferSingle( - address indexed operator, - address indexed from, - address indexed to, - uint256 id, - uint256 amount - ); - - event TransferBatch( - address indexed operator, - address indexed from, - address indexed to, - uint256[] ids, - uint256[] amounts - ); - - event ApprovalForAll(address indexed owner, address indexed operator, bool approved); - - event URI(string value, uint256 indexed id); - - /*////////////////////////////////////////////////////////////// - ERC1155 STORAGE - //////////////////////////////////////////////////////////////*/ - - mapping(address => mapping(uint256 => uint256)) public balanceOf; - - mapping(address => mapping(address => bool)) public isApprovedForAll; - - /*////////////////////////////////////////////////////////////// - METADATA LOGIC - //////////////////////////////////////////////////////////////*/ - - function uri(uint256 id) public view virtual returns (string memory); - - /*////////////////////////////////////////////////////////////// - ERC1155 LOGIC - //////////////////////////////////////////////////////////////*/ - - function setApprovalForAll(address operator, bool approved) public virtual { - isApprovedForAll[msg.sender][operator] = approved; - - emit ApprovalForAll(msg.sender, operator, approved); - } - - function safeTransferFrom( - address from, - address to, - uint256 id, - uint256 amount, - bytes calldata data - ) public virtual { - require(msg.sender == from || isApprovedForAll[from][msg.sender], "NOT_AUTHORIZED"); - - balanceOf[from][id] -= amount; - balanceOf[to][id] += amount; - - emit TransferSingle(msg.sender, from, to, id, amount); - - require( - to.code.length == 0 - ? to != address(0) - : ERC1155TokenReceiver(to).onERC1155Received(msg.sender, from, id, amount, data) == - ERC1155TokenReceiver.onERC1155Received.selector, - "UNSAFE_RECIPIENT" - ); - } - - function safeBatchTransferFrom( - address from, - address to, - uint256[] calldata ids, - uint256[] calldata amounts, - bytes calldata data - ) public virtual { - require(ids.length == amounts.length, "LENGTH_MISMATCH"); - - require(msg.sender == from || isApprovedForAll[from][msg.sender], "NOT_AUTHORIZED"); - - // Storing these outside the loop saves ~15 gas per iteration. - uint256 id; - uint256 amount; - - for (uint256 i = 0; i < ids.length; ) { - id = ids[i]; - amount = amounts[i]; - - balanceOf[from][id] -= amount; - balanceOf[to][id] += amount; - - // An array can't have a total length - // larger than the max uint256 value. - unchecked { - ++i; - } - } - - emit TransferBatch(msg.sender, from, to, ids, amounts); - - require( - to.code.length == 0 - ? to != address(0) - : ERC1155TokenReceiver(to).onERC1155BatchReceived(msg.sender, from, ids, amounts, data) == - ERC1155TokenReceiver.onERC1155BatchReceived.selector, - "UNSAFE_RECIPIENT" - ); - } - - function balanceOfBatch(address[] calldata owners, uint256[] calldata ids) - public - view - virtual - returns (uint256[] memory balances) - { - require(owners.length == ids.length, "LENGTH_MISMATCH"); - - balances = new uint256[](owners.length); - - // Unchecked because the only math done is incrementing - // the array index counter which cannot possibly overflow. - unchecked { - for (uint256 i = 0; i < owners.length; ++i) { - balances[i] = balanceOf[owners[i]][ids[i]]; - } - } - } - - /*////////////////////////////////////////////////////////////// - ERC165 LOGIC - //////////////////////////////////////////////////////////////*/ - - function supportsInterface(bytes4 interfaceId) public view virtual returns (bool) { - return - interfaceId == 0x01ffc9a7 || // ERC165 Interface ID for ERC165 - interfaceId == 0xd9b67a26 || // ERC165 Interface ID for ERC1155 - interfaceId == 0x0e89341c; // ERC165 Interface ID for ERC1155MetadataURI - } - - /*////////////////////////////////////////////////////////////// - INTERNAL MINT/BURN LOGIC - //////////////////////////////////////////////////////////////*/ - - function _mint( - address to, - uint256 id, - uint256 amount, - bytes memory data - ) internal virtual { - balanceOf[to][id] += amount; - - emit TransferSingle(msg.sender, address(0), to, id, amount); - - require( - to.code.length == 0 - ? to != address(0) - : ERC1155TokenReceiver(to).onERC1155Received(msg.sender, address(0), id, amount, data) == - ERC1155TokenReceiver.onERC1155Received.selector, - "UNSAFE_RECIPIENT" - ); - } - - function _batchMint( - address to, - uint256[] memory ids, - uint256[] memory amounts, - bytes memory data - ) internal virtual { - uint256 idsLength = ids.length; // Saves MLOADs. - - require(idsLength == amounts.length, "LENGTH_MISMATCH"); - - for (uint256 i = 0; i < idsLength; ) { - balanceOf[to][ids[i]] += amounts[i]; - - // An array can't have a total length - // larger than the max uint256 value. - unchecked { - ++i; - } - } - - emit TransferBatch(msg.sender, address(0), to, ids, amounts); - - require( - to.code.length == 0 - ? to != address(0) - : ERC1155TokenReceiver(to).onERC1155BatchReceived(msg.sender, address(0), ids, amounts, data) == - ERC1155TokenReceiver.onERC1155BatchReceived.selector, - "UNSAFE_RECIPIENT" - ); - } - - function _batchBurn( - address from, - uint256[] memory ids, - uint256[] memory amounts - ) internal virtual { - uint256 idsLength = ids.length; // Saves MLOADs. - - require(idsLength == amounts.length, "LENGTH_MISMATCH"); - - for (uint256 i = 0; i < idsLength; ) { - balanceOf[from][ids[i]] -= amounts[i]; - - // An array can't have a total length - // larger than the max uint256 value. - unchecked { - ++i; - } - } - - emit TransferBatch(msg.sender, from, address(0), ids, amounts); - } - - function _burn( - address from, - uint256 id, - uint256 amount - ) internal virtual { - balanceOf[from][id] -= amount; - - emit TransferSingle(msg.sender, from, address(0), id, amount); - } -} - -/// @notice A generic interface for a contract which properly accepts ERC1155 tokens. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC1155.sol) -abstract contract ERC1155TokenReceiver { - function onERC1155Received( - address, - address, - uint256, - uint256, - bytes calldata - ) external virtual returns (bytes4) { - return ERC1155TokenReceiver.onERC1155Received.selector; - } - - function onERC1155BatchReceived( - address, - address, - uint256[] calldata, - uint256[] calldata, - bytes calldata - ) external virtual returns (bytes4) { - return ERC1155TokenReceiver.onERC1155BatchReceived.selector; - } -} diff --git a/lib/solmate/src/tokens/ERC20.sol b/lib/solmate/src/tokens/ERC20.sol deleted file mode 100644 index 9657044..0000000 --- a/lib/solmate/src/tokens/ERC20.sol +++ /dev/null @@ -1,206 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -/// @notice Modern and gas efficient ERC20 + EIP-2612 implementation. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC20.sol) -/// @author Modified from Uniswap (https://github.com/Uniswap/uniswap-v2-core/blob/master/contracts/UniswapV2ERC20.sol) -/// @dev Do not manually set balances without updating totalSupply, as the sum of all user balances must not exceed it. -abstract contract ERC20 { - /*////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 amount); - - event Approval(address indexed owner, address indexed spender, uint256 amount); - - /*////////////////////////////////////////////////////////////// - METADATA STORAGE - //////////////////////////////////////////////////////////////*/ - - string public name; - - string public symbol; - - uint8 public immutable decimals; - - /*////////////////////////////////////////////////////////////// - ERC20 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 public totalSupply; - - mapping(address => uint256) public balanceOf; - - mapping(address => mapping(address => uint256)) public allowance; - - /*////////////////////////////////////////////////////////////// - EIP-2612 STORAGE - //////////////////////////////////////////////////////////////*/ - - uint256 internal immutable INITIAL_CHAIN_ID; - - bytes32 internal immutable INITIAL_DOMAIN_SEPARATOR; - - mapping(address => uint256) public nonces; - - /*////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor( - string memory _name, - string memory _symbol, - uint8 _decimals - ) { - name = _name; - symbol = _symbol; - decimals = _decimals; - - INITIAL_CHAIN_ID = block.chainid; - INITIAL_DOMAIN_SEPARATOR = computeDomainSeparator(); - } - - /*////////////////////////////////////////////////////////////// - ERC20 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address spender, uint256 amount) public virtual returns (bool) { - allowance[msg.sender][spender] = amount; - - emit Approval(msg.sender, spender, amount); - - return true; - } - - function transfer(address to, uint256 amount) public virtual returns (bool) { - balanceOf[msg.sender] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(msg.sender, to, amount); - - return true; - } - - function transferFrom( - address from, - address to, - uint256 amount - ) public virtual returns (bool) { - uint256 allowed = allowance[from][msg.sender]; // Saves gas for limited approvals. - - if (allowed != type(uint256).max) allowance[from][msg.sender] = allowed - amount; - - balanceOf[from] -= amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(from, to, amount); - - return true; - } - - /*////////////////////////////////////////////////////////////// - EIP-2612 LOGIC - //////////////////////////////////////////////////////////////*/ - - function permit( - address owner, - address spender, - uint256 value, - uint256 deadline, - uint8 v, - bytes32 r, - bytes32 s - ) public virtual { - require(deadline >= block.timestamp, "PERMIT_DEADLINE_EXPIRED"); - - // Unchecked because the only math done is incrementing - // the owner's nonce which cannot realistically overflow. - unchecked { - address recoveredAddress = ecrecover( - keccak256( - abi.encodePacked( - "\x19\x01", - DOMAIN_SEPARATOR(), - keccak256( - abi.encode( - keccak256( - "Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)" - ), - owner, - spender, - value, - nonces[owner]++, - deadline - ) - ) - ) - ), - v, - r, - s - ); - - require(recoveredAddress != address(0) && recoveredAddress == owner, "INVALID_SIGNER"); - - allowance[recoveredAddress][spender] = value; - } - - emit Approval(owner, spender, value); - } - - function DOMAIN_SEPARATOR() public view virtual returns (bytes32) { - return block.chainid == INITIAL_CHAIN_ID ? INITIAL_DOMAIN_SEPARATOR : computeDomainSeparator(); - } - - function computeDomainSeparator() internal view virtual returns (bytes32) { - return - keccak256( - abi.encode( - keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"), - keccak256(bytes(name)), - keccak256("1"), - block.chainid, - address(this) - ) - ); - } - - /*////////////////////////////////////////////////////////////// - INTERNAL MINT/BURN LOGIC - //////////////////////////////////////////////////////////////*/ - - function _mint(address to, uint256 amount) internal virtual { - totalSupply += amount; - - // Cannot overflow because the sum of all user - // balances can't exceed the max uint256 value. - unchecked { - balanceOf[to] += amount; - } - - emit Transfer(address(0), to, amount); - } - - function _burn(address from, uint256 amount) internal virtual { - balanceOf[from] -= amount; - - // Cannot underflow because a user's balance - // will never be larger than the total supply. - unchecked { - totalSupply -= amount; - } - - emit Transfer(from, address(0), amount); - } -} diff --git a/lib/solmate/src/tokens/ERC6909.sol b/lib/solmate/src/tokens/ERC6909.sol deleted file mode 100644 index 15b4f30..0000000 --- a/lib/solmate/src/tokens/ERC6909.sol +++ /dev/null @@ -1,118 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity >=0.8.0; - -/// @notice Minimalist and gas efficient standard ERC6909 implementation. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC6909.sol) -abstract contract ERC6909 { - /*////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event OperatorSet(address indexed owner, address indexed operator, bool approved); - - event Approval(address indexed owner, address indexed spender, uint256 indexed id, uint256 amount); - - event Transfer(address caller, address indexed from, address indexed to, uint256 indexed id, uint256 amount); - - /*////////////////////////////////////////////////////////////// - ERC6909 STORAGE - //////////////////////////////////////////////////////////////*/ - - mapping(address => mapping(address => bool)) public isOperator; - - mapping(address => mapping(uint256 => uint256)) public balanceOf; - - mapping(address => mapping(address => mapping(uint256 => uint256))) public allowance; - - /*////////////////////////////////////////////////////////////// - ERC6909 LOGIC - //////////////////////////////////////////////////////////////*/ - - function transfer( - address receiver, - uint256 id, - uint256 amount - ) public virtual returns (bool) { - balanceOf[msg.sender][id] -= amount; - - balanceOf[receiver][id] += amount; - - emit Transfer(msg.sender, msg.sender, receiver, id, amount); - - return true; - } - - function transferFrom( - address sender, - address receiver, - uint256 id, - uint256 amount - ) public virtual returns (bool) { - if (msg.sender != sender && !isOperator[sender][msg.sender]) { - uint256 allowed = allowance[sender][msg.sender][id]; - if (allowed != type(uint256).max) allowance[sender][msg.sender][id] = allowed - amount; - } - - balanceOf[sender][id] -= amount; - - balanceOf[receiver][id] += amount; - - emit Transfer(msg.sender, sender, receiver, id, amount); - - return true; - } - - function approve( - address spender, - uint256 id, - uint256 amount - ) public virtual returns (bool) { - allowance[msg.sender][spender][id] = amount; - - emit Approval(msg.sender, spender, id, amount); - - return true; - } - - function setOperator(address operator, bool approved) public virtual returns (bool) { - isOperator[msg.sender][operator] = approved; - - emit OperatorSet(msg.sender, operator, approved); - - return true; - } - - /*////////////////////////////////////////////////////////////// - ERC165 LOGIC - //////////////////////////////////////////////////////////////*/ - - function supportsInterface(bytes4 interfaceId) public view virtual returns (bool) { - return - interfaceId == 0x01ffc9a7 || // ERC165 Interface ID for ERC165 - interfaceId == 0x0f632fb3; // ERC165 Interface ID for ERC6909 - } - - /*////////////////////////////////////////////////////////////// - INTERNAL MINT/BURN LOGIC - //////////////////////////////////////////////////////////////*/ - - function _mint( - address receiver, - uint256 id, - uint256 amount - ) internal virtual { - balanceOf[receiver][id] += amount; - - emit Transfer(msg.sender, address(0), receiver, id, amount); - } - - function _burn( - address sender, - uint256 id, - uint256 amount - ) internal virtual { - balanceOf[sender][id] -= amount; - - emit Transfer(msg.sender, sender, address(0), id, amount); - } -} diff --git a/lib/solmate/src/tokens/ERC721.sol b/lib/solmate/src/tokens/ERC721.sol deleted file mode 100644 index b47f271..0000000 --- a/lib/solmate/src/tokens/ERC721.sol +++ /dev/null @@ -1,231 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -/// @notice Modern, minimalist, and gas efficient ERC-721 implementation. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC721.sol) -abstract contract ERC721 { - /*////////////////////////////////////////////////////////////// - EVENTS - //////////////////////////////////////////////////////////////*/ - - event Transfer(address indexed from, address indexed to, uint256 indexed id); - - event Approval(address indexed owner, address indexed spender, uint256 indexed id); - - event ApprovalForAll(address indexed owner, address indexed operator, bool approved); - - /*////////////////////////////////////////////////////////////// - METADATA STORAGE/LOGIC - //////////////////////////////////////////////////////////////*/ - - string public name; - - string public symbol; - - function tokenURI(uint256 id) public view virtual returns (string memory); - - /*////////////////////////////////////////////////////////////// - ERC721 BALANCE/OWNER STORAGE - //////////////////////////////////////////////////////////////*/ - - mapping(uint256 => address) internal _ownerOf; - - mapping(address => uint256) internal _balanceOf; - - function ownerOf(uint256 id) public view virtual returns (address owner) { - require((owner = _ownerOf[id]) != address(0), "NOT_MINTED"); - } - - function balanceOf(address owner) public view virtual returns (uint256) { - require(owner != address(0), "ZERO_ADDRESS"); - - return _balanceOf[owner]; - } - - /*////////////////////////////////////////////////////////////// - ERC721 APPROVAL STORAGE - //////////////////////////////////////////////////////////////*/ - - mapping(uint256 => address) public getApproved; - - mapping(address => mapping(address => bool)) public isApprovedForAll; - - /*////////////////////////////////////////////////////////////// - CONSTRUCTOR - //////////////////////////////////////////////////////////////*/ - - constructor(string memory _name, string memory _symbol) { - name = _name; - symbol = _symbol; - } - - /*////////////////////////////////////////////////////////////// - ERC721 LOGIC - //////////////////////////////////////////////////////////////*/ - - function approve(address spender, uint256 id) public virtual { - address owner = _ownerOf[id]; - - require(msg.sender == owner || isApprovedForAll[owner][msg.sender], "NOT_AUTHORIZED"); - - getApproved[id] = spender; - - emit Approval(owner, spender, id); - } - - function setApprovalForAll(address operator, bool approved) public virtual { - isApprovedForAll[msg.sender][operator] = approved; - - emit ApprovalForAll(msg.sender, operator, approved); - } - - function transferFrom( - address from, - address to, - uint256 id - ) public virtual { - require(from == _ownerOf[id], "WRONG_FROM"); - - require(to != address(0), "INVALID_RECIPIENT"); - - require( - msg.sender == from || isApprovedForAll[from][msg.sender] || msg.sender == getApproved[id], - "NOT_AUTHORIZED" - ); - - // Underflow of the sender's balance is impossible because we check for - // ownership above and the recipient's balance can't realistically overflow. - unchecked { - _balanceOf[from]--; - - _balanceOf[to]++; - } - - _ownerOf[id] = to; - - delete getApproved[id]; - - emit Transfer(from, to, id); - } - - function safeTransferFrom( - address from, - address to, - uint256 id - ) public virtual { - transferFrom(from, to, id); - - require( - to.code.length == 0 || - ERC721TokenReceiver(to).onERC721Received(msg.sender, from, id, "") == - ERC721TokenReceiver.onERC721Received.selector, - "UNSAFE_RECIPIENT" - ); - } - - function safeTransferFrom( - address from, - address to, - uint256 id, - bytes calldata data - ) public virtual { - transferFrom(from, to, id); - - require( - to.code.length == 0 || - ERC721TokenReceiver(to).onERC721Received(msg.sender, from, id, data) == - ERC721TokenReceiver.onERC721Received.selector, - "UNSAFE_RECIPIENT" - ); - } - - /*////////////////////////////////////////////////////////////// - ERC165 LOGIC - //////////////////////////////////////////////////////////////*/ - - function supportsInterface(bytes4 interfaceId) public view virtual returns (bool) { - return - interfaceId == 0x01ffc9a7 || // ERC165 Interface ID for ERC165 - interfaceId == 0x80ac58cd || // ERC165 Interface ID for ERC721 - interfaceId == 0x5b5e139f; // ERC165 Interface ID for ERC721Metadata - } - - /*////////////////////////////////////////////////////////////// - INTERNAL MINT/BURN LOGIC - //////////////////////////////////////////////////////////////*/ - - function _mint(address to, uint256 id) internal virtual { - require(to != address(0), "INVALID_RECIPIENT"); - - require(_ownerOf[id] == address(0), "ALREADY_MINTED"); - - // Counter overflow is incredibly unrealistic. - unchecked { - _balanceOf[to]++; - } - - _ownerOf[id] = to; - - emit Transfer(address(0), to, id); - } - - function _burn(uint256 id) internal virtual { - address owner = _ownerOf[id]; - - require(owner != address(0), "NOT_MINTED"); - - // Ownership check above ensures no underflow. - unchecked { - _balanceOf[owner]--; - } - - delete _ownerOf[id]; - - delete getApproved[id]; - - emit Transfer(owner, address(0), id); - } - - /*////////////////////////////////////////////////////////////// - INTERNAL SAFE MINT LOGIC - //////////////////////////////////////////////////////////////*/ - - function _safeMint(address to, uint256 id) internal virtual { - _mint(to, id); - - require( - to.code.length == 0 || - ERC721TokenReceiver(to).onERC721Received(msg.sender, address(0), id, "") == - ERC721TokenReceiver.onERC721Received.selector, - "UNSAFE_RECIPIENT" - ); - } - - function _safeMint( - address to, - uint256 id, - bytes memory data - ) internal virtual { - _mint(to, id); - - require( - to.code.length == 0 || - ERC721TokenReceiver(to).onERC721Received(msg.sender, address(0), id, data) == - ERC721TokenReceiver.onERC721Received.selector, - "UNSAFE_RECIPIENT" - ); - } -} - -/// @notice A generic interface for a contract which properly accepts ERC721 tokens. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/ERC721.sol) -abstract contract ERC721TokenReceiver { - function onERC721Received( - address, - address, - uint256, - bytes calldata - ) external virtual returns (bytes4) { - return ERC721TokenReceiver.onERC721Received.selector; - } -} diff --git a/lib/solmate/src/tokens/WETH.sol b/lib/solmate/src/tokens/WETH.sol deleted file mode 100644 index ddf9647..0000000 --- a/lib/solmate/src/tokens/WETH.sol +++ /dev/null @@ -1,35 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {ERC20} from "./ERC20.sol"; - -import {SafeTransferLib} from "../utils/SafeTransferLib.sol"; - -/// @notice Minimalist and modern Wrapped Ether implementation. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/tokens/WETH.sol) -/// @author Inspired by WETH9 (https://github.com/dapphub/ds-weth/blob/master/src/weth9.sol) -contract WETH is ERC20("Wrapped Ether", "WETH", 18) { - using SafeTransferLib for address; - - event Deposit(address indexed from, uint256 amount); - - event Withdrawal(address indexed to, uint256 amount); - - function deposit() public payable virtual { - _mint(msg.sender, msg.value); - - emit Deposit(msg.sender, msg.value); - } - - function withdraw(uint256 amount) public virtual { - _burn(msg.sender, amount); - - emit Withdrawal(msg.sender, amount); - - msg.sender.safeTransferETH(amount); - } - - receive() external payable virtual { - deposit(); - } -} diff --git a/lib/solmate/src/utils/Bytes32AddressLib.sol b/lib/solmate/src/utils/Bytes32AddressLib.sol deleted file mode 100644 index 448fb75..0000000 --- a/lib/solmate/src/utils/Bytes32AddressLib.sol +++ /dev/null @@ -1,14 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -/// @notice Library for converting between addresses and bytes32 values. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/Bytes32AddressLib.sol) -library Bytes32AddressLib { - function fromLast20Bytes(bytes32 bytesValue) internal pure returns (address) { - return address(uint160(uint256(bytesValue))); - } - - function fillLast12Bytes(address addressValue) internal pure returns (bytes32) { - return bytes32(bytes20(addressValue)); - } -} diff --git a/lib/solmate/src/utils/CREATE3.sol b/lib/solmate/src/utils/CREATE3.sol deleted file mode 100644 index 8fe0511..0000000 --- a/lib/solmate/src/utils/CREATE3.sol +++ /dev/null @@ -1,87 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {Bytes32AddressLib} from "./Bytes32AddressLib.sol"; - -/// @notice Deploy to deterministic addresses without an initcode factor. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/CREATE3.sol) -/// @author Modified from 0xSequence (https://github.com/0xSequence/create3/blob/master/contracts/Create3.sol) -library CREATE3 { - using Bytes32AddressLib for bytes32; - - //--------------------------------------------------------------------------------// - // Opcode | Opcode + Arguments | Description | Stack View // - //--------------------------------------------------------------------------------// - // 0x36 | 0x36 | CALLDATASIZE | size // - // 0x3d | 0x3d | RETURNDATASIZE | 0 size // - // 0x3d | 0x3d | RETURNDATASIZE | 0 0 size // - // 0x37 | 0x37 | CALLDATACOPY | // - // 0x36 | 0x36 | CALLDATASIZE | size // - // 0x3d | 0x3d | RETURNDATASIZE | 0 size // - // 0x34 | 0x34 | CALLVALUE | value 0 size // - // 0xf0 | 0xf0 | CREATE | newContract // - //--------------------------------------------------------------------------------// - // Opcode | Opcode + Arguments | Description | Stack View // - //--------------------------------------------------------------------------------// - // 0x67 | 0x67XXXXXXXXXXXXXXXX | PUSH8 bytecode | bytecode // - // 0x3d | 0x3d | RETURNDATASIZE | 0 bytecode // - // 0x52 | 0x52 | MSTORE | // - // 0x60 | 0x6008 | PUSH1 08 | 8 // - // 0x60 | 0x6018 | PUSH1 18 | 24 8 // - // 0xf3 | 0xf3 | RETURN | // - //--------------------------------------------------------------------------------// - bytes internal constant PROXY_BYTECODE = hex"67_36_3d_3d_37_36_3d_34_f0_3d_52_60_08_60_18_f3"; - - bytes32 internal constant PROXY_BYTECODE_HASH = keccak256(PROXY_BYTECODE); - - function deploy( - bytes32 salt, - bytes memory creationCode, - uint256 value - ) internal returns (address deployed) { - bytes memory proxyChildBytecode = PROXY_BYTECODE; - - address proxy; - /// @solidity memory-safe-assembly - assembly { - // Deploy a new contract with our pre-made bytecode via CREATE2. - // We start 32 bytes into the code to avoid copying the byte length. - proxy := create2(0, add(proxyChildBytecode, 32), mload(proxyChildBytecode), salt) - } - require(proxy != address(0), "DEPLOYMENT_FAILED"); - - deployed = getDeployed(salt); - (bool success, ) = proxy.call{value: value}(creationCode); - require(success && deployed.code.length != 0, "INITIALIZATION_FAILED"); - } - - function getDeployed(bytes32 salt) internal view returns (address) { - return getDeployed(salt, address(this)); - } - - function getDeployed(bytes32 salt, address creator) internal pure returns (address) { - address proxy = keccak256( - abi.encodePacked( - // Prefix: - bytes1(0xFF), - // Creator: - creator, - // Salt: - salt, - // Bytecode hash: - PROXY_BYTECODE_HASH - ) - ).fromLast20Bytes(); - - return - keccak256( - abi.encodePacked( - // 0xd6 = 0xc0 (short RLP prefix) + 0x16 (length of: 0x94 ++ proxy ++ 0x01) - // 0x94 = 0x80 + 0x14 (0x14 = the length of an address, 20 bytes, in hex) - hex"d6_94", - proxy, - hex"01" // Nonce of the proxy contract (1) - ) - ).fromLast20Bytes(); - } -} diff --git a/lib/solmate/src/utils/FixedPointMathLib.sol b/lib/solmate/src/utils/FixedPointMathLib.sol deleted file mode 100644 index 6887722..0000000 --- a/lib/solmate/src/utils/FixedPointMathLib.sol +++ /dev/null @@ -1,255 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -/// @notice Arithmetic library with operations for fixed-point numbers. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/FixedPointMathLib.sol) -/// @author Inspired by USM (https://github.com/usmfum/USM/blob/master/contracts/WadMath.sol) -library FixedPointMathLib { - /*////////////////////////////////////////////////////////////// - SIMPLIFIED FIXED POINT OPERATIONS - //////////////////////////////////////////////////////////////*/ - - uint256 internal constant MAX_UINT256 = 2**256 - 1; - - uint256 internal constant WAD = 1e18; // The scalar of ETH and most ERC20s. - - function mulWadDown(uint256 x, uint256 y) internal pure returns (uint256) { - return mulDivDown(x, y, WAD); // Equivalent to (x * y) / WAD rounded down. - } - - function mulWadUp(uint256 x, uint256 y) internal pure returns (uint256) { - return mulDivUp(x, y, WAD); // Equivalent to (x * y) / WAD rounded up. - } - - function divWadDown(uint256 x, uint256 y) internal pure returns (uint256) { - return mulDivDown(x, WAD, y); // Equivalent to (x * WAD) / y rounded down. - } - - function divWadUp(uint256 x, uint256 y) internal pure returns (uint256) { - return mulDivUp(x, WAD, y); // Equivalent to (x * WAD) / y rounded up. - } - - /*////////////////////////////////////////////////////////////// - LOW LEVEL FIXED POINT OPERATIONS - //////////////////////////////////////////////////////////////*/ - - function mulDivDown( - uint256 x, - uint256 y, - uint256 denominator - ) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - // Equivalent to require(denominator != 0 && (y == 0 || x <= type(uint256).max / y)) - if iszero(mul(denominator, iszero(mul(y, gt(x, div(MAX_UINT256, y)))))) { - revert(0, 0) - } - - // Divide x * y by the denominator. - z := div(mul(x, y), denominator) - } - } - - function mulDivUp( - uint256 x, - uint256 y, - uint256 denominator - ) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - // Equivalent to require(denominator != 0 && (y == 0 || x <= type(uint256).max / y)) - if iszero(mul(denominator, iszero(mul(y, gt(x, div(MAX_UINT256, y)))))) { - revert(0, 0) - } - - // If x * y modulo the denominator is strictly greater than 0, - // 1 is added to round up the division of x * y by the denominator. - z := add(gt(mod(mul(x, y), denominator), 0), div(mul(x, y), denominator)) - } - } - - function rpow( - uint256 x, - uint256 n, - uint256 scalar - ) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - switch x - case 0 { - switch n - case 0 { - // 0 ** 0 = 1 - z := scalar - } - default { - // 0 ** n = 0 - z := 0 - } - } - default { - switch mod(n, 2) - case 0 { - // If n is even, store scalar in z for now. - z := scalar - } - default { - // If n is odd, store x in z for now. - z := x - } - - // Shifting right by 1 is like dividing by 2. - let half := shr(1, scalar) - - for { - // Shift n right by 1 before looping to halve it. - n := shr(1, n) - } n { - // Shift n right by 1 each iteration to halve it. - n := shr(1, n) - } { - // Revert immediately if x ** 2 would overflow. - // Equivalent to iszero(eq(div(xx, x), x)) here. - if shr(128, x) { - revert(0, 0) - } - - // Store x squared. - let xx := mul(x, x) - - // Round to the nearest number. - let xxRound := add(xx, half) - - // Revert if xx + half overflowed. - if lt(xxRound, xx) { - revert(0, 0) - } - - // Set x to scaled xxRound. - x := div(xxRound, scalar) - - // If n is even: - if mod(n, 2) { - // Compute z * x. - let zx := mul(z, x) - - // If z * x overflowed: - if iszero(eq(div(zx, x), z)) { - // Revert if x is non-zero. - if iszero(iszero(x)) { - revert(0, 0) - } - } - - // Round to the nearest number. - let zxRound := add(zx, half) - - // Revert if zx + half overflowed. - if lt(zxRound, zx) { - revert(0, 0) - } - - // Return properly scaled zxRound. - z := div(zxRound, scalar) - } - } - } - } - } - - /*////////////////////////////////////////////////////////////// - GENERAL NUMBER UTILITIES - //////////////////////////////////////////////////////////////*/ - - function sqrt(uint256 x) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - let y := x // We start y at x, which will help us make our initial estimate. - - z := 181 // The "correct" value is 1, but this saves a multiplication later. - - // This segment is to get a reasonable initial estimate for the Babylonian method. With a bad - // start, the correct # of bits increases ~linearly each iteration instead of ~quadratically. - - // We check y >= 2^(k + 8) but shift right by k bits - // each branch to ensure that if x >= 256, then y >= 256. - if iszero(lt(y, 0x10000000000000000000000000000000000)) { - y := shr(128, y) - z := shl(64, z) - } - if iszero(lt(y, 0x1000000000000000000)) { - y := shr(64, y) - z := shl(32, z) - } - if iszero(lt(y, 0x10000000000)) { - y := shr(32, y) - z := shl(16, z) - } - if iszero(lt(y, 0x1000000)) { - y := shr(16, y) - z := shl(8, z) - } - - // Goal was to get z*z*y within a small factor of x. More iterations could - // get y in a tighter range. Currently, we will have y in [256, 256*2^16). - // We ensured y >= 256 so that the relative difference between y and y+1 is small. - // That's not possible if x < 256 but we can just verify those cases exhaustively. - - // Now, z*z*y <= x < z*z*(y+1), and y <= 2^(16+8), and either y >= 256, or x < 256. - // Correctness can be checked exhaustively for x < 256, so we assume y >= 256. - // Then z*sqrt(y) is within sqrt(257)/sqrt(256) of sqrt(x), or about 20bps. - - // For s in the range [1/256, 256], the estimate f(s) = (181/1024) * (s+1) is in the range - // (1/2.84 * sqrt(s), 2.84 * sqrt(s)), with largest error when s = 1 and when s = 256 or 1/256. - - // Since y is in [256, 256*2^16), let a = y/65536, so that a is in [1/256, 256). Then we can estimate - // sqrt(y) using sqrt(65536) * 181/1024 * (a + 1) = 181/4 * (y + 65536)/65536 = 181 * (y + 65536)/2^18. - - // There is no overflow risk here since y < 2^136 after the first branch above. - z := shr(18, mul(z, add(y, 65536))) // A mul() is saved from starting z at 181. - - // Given the worst case multiplicative error of 2.84 above, 7 iterations should be enough. - z := shr(1, add(z, div(x, z))) - z := shr(1, add(z, div(x, z))) - z := shr(1, add(z, div(x, z))) - z := shr(1, add(z, div(x, z))) - z := shr(1, add(z, div(x, z))) - z := shr(1, add(z, div(x, z))) - z := shr(1, add(z, div(x, z))) - - // If x+1 is a perfect square, the Babylonian method cycles between - // floor(sqrt(x)) and ceil(sqrt(x)). This statement ensures we return floor. - // See: https://en.wikipedia.org/wiki/Integer_square_root#Using_only_integer_division - // Since the ceil is rare, we save gas on the assignment and repeat division in the rare case. - // If you don't care whether the floor or ceil square root is returned, you can remove this statement. - z := sub(z, lt(div(x, z), z)) - } - } - - function unsafeMod(uint256 x, uint256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - // Mod x by y. Note this will return - // 0 instead of reverting if y is zero. - z := mod(x, y) - } - } - - function unsafeDiv(uint256 x, uint256 y) internal pure returns (uint256 r) { - /// @solidity memory-safe-assembly - assembly { - // Divide x by y. Note this will return - // 0 instead of reverting if y is zero. - r := div(x, y) - } - } - - function unsafeDivUp(uint256 x, uint256 y) internal pure returns (uint256 z) { - /// @solidity memory-safe-assembly - assembly { - // Add 1 to x * y if x % y > 0. Note this will - // return 0 instead of reverting if y is zero. - z := add(gt(mod(x, y), 0), div(x, y)) - } - } -} diff --git a/lib/solmate/src/utils/LibString.sol b/lib/solmate/src/utils/LibString.sol deleted file mode 100644 index 97c89e0..0000000 --- a/lib/solmate/src/utils/LibString.sol +++ /dev/null @@ -1,77 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity >=0.8.0; - -/// @notice Efficient library for creating string representations of integers. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/LibString.sol) -/// @author Modified from Solady (https://github.com/Vectorized/solady/blob/main/src/utils/LibString.sol) -library LibString { - function toString(int256 value) internal pure returns (string memory str) { - if (value >= 0) return toString(uint256(value)); - - unchecked { - str = toString(uint256(-value)); - - /// @solidity memory-safe-assembly - assembly { - // Note: This is only safe because we over-allocate memory - // and write the string from right to left in toString(uint256), - // and thus can be sure that sub(str, 1) is an unused memory location. - - let length := mload(str) // Load the string length. - // Put the - character at the start of the string contents. - mstore(str, 45) // 45 is the ASCII code for the - character. - str := sub(str, 1) // Move back the string pointer by a byte. - mstore(str, add(length, 1)) // Update the string length. - } - } - } - - function toString(uint256 value) internal pure returns (string memory str) { - /// @solidity memory-safe-assembly - assembly { - // The maximum value of a uint256 contains 78 digits (1 byte per digit), but we allocate 160 bytes - // to keep the free memory pointer word aligned. We'll need 1 word for the length, 1 word for the - // trailing zeros padding, and 3 other words for a max of 78 digits. In total: 5 * 32 = 160 bytes. - let newFreeMemoryPointer := add(mload(0x40), 160) - - // Update the free memory pointer to avoid overriding our string. - mstore(0x40, newFreeMemoryPointer) - - // Assign str to the end of the zone of newly allocated memory. - str := sub(newFreeMemoryPointer, 32) - - // Clean the last word of memory it may not be overwritten. - mstore(str, 0) - - // Cache the end of the memory to calculate the length later. - let end := str - - // We write the string from rightmost digit to leftmost digit. - // The following is essentially a do-while loop that also handles the zero case. - // prettier-ignore - for { let temp := value } 1 {} { - // Move the pointer 1 byte to the left. - str := sub(str, 1) - - // Write the character to the pointer. - // The ASCII index of the '0' character is 48. - mstore8(str, add(48, mod(temp, 10))) - - // Keep dividing temp until zero. - temp := div(temp, 10) - - // prettier-ignore - if iszero(temp) { break } - } - - // Compute and cache the final total length of the string. - let length := sub(end, str) - - // Move the pointer 32 bytes leftwards to make room for the length. - str := sub(str, 32) - - // Store the string's length at the start of memory allocated for our string. - mstore(str, length) - } - } -} diff --git a/lib/solmate/src/utils/MerkleProofLib.sol b/lib/solmate/src/utils/MerkleProofLib.sol deleted file mode 100644 index 8fd7cbd..0000000 --- a/lib/solmate/src/utils/MerkleProofLib.sol +++ /dev/null @@ -1,48 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity >=0.8.0; - -/// @notice Gas optimized merkle proof verification library. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/MerkleProofLib.sol) -/// @author Modified from Solady (https://github.com/Vectorized/solady/blob/main/src/utils/MerkleProofLib.sol) -library MerkleProofLib { - function verify( - bytes32[] calldata proof, - bytes32 root, - bytes32 leaf - ) internal pure returns (bool isValid) { - /// @solidity memory-safe-assembly - assembly { - if proof.length { - // Left shifting by 5 is like multiplying by 32. - let end := add(proof.offset, shl(5, proof.length)) - - // Initialize offset to the offset of the proof in calldata. - let offset := proof.offset - - // Iterate over proof elements to compute root hash. - // prettier-ignore - for {} 1 {} { - // Slot where the leaf should be put in scratch space. If - // leaf > calldataload(offset): slot 32, otherwise: slot 0. - let leafSlot := shl(5, gt(leaf, calldataload(offset))) - - // Store elements to hash contiguously in scratch space. - // The xor puts calldataload(offset) in whichever slot leaf - // is not occupying, so 0 if leafSlot is 32, and 32 otherwise. - mstore(leafSlot, leaf) - mstore(xor(leafSlot, 32), calldataload(offset)) - - // Reuse leaf to store the hash to reduce stack operations. - leaf := keccak256(0, 64) // Hash both slots of scratch space. - - offset := add(offset, 32) // Shift 1 word per cycle. - - // prettier-ignore - if iszero(lt(offset, end)) { break } - } - } - - isValid := eq(leaf, root) // The proof is valid if the roots match. - } - } -} diff --git a/lib/solmate/src/utils/ReentrancyGuard.sol b/lib/solmate/src/utils/ReentrancyGuard.sol deleted file mode 100644 index 1453e24..0000000 --- a/lib/solmate/src/utils/ReentrancyGuard.sol +++ /dev/null @@ -1,19 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -/// @notice Gas optimized reentrancy protection for smart contracts. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/ReentrancyGuard.sol) -/// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/security/ReentrancyGuard.sol) -abstract contract ReentrancyGuard { - uint256 private locked = 1; - - modifier nonReentrant() virtual { - require(locked == 1, "REENTRANCY"); - - locked = 2; - - _; - - locked = 1; - } -} diff --git a/lib/solmate/src/utils/SSTORE2.sol b/lib/solmate/src/utils/SSTORE2.sol deleted file mode 100644 index 23d6980..0000000 --- a/lib/solmate/src/utils/SSTORE2.sol +++ /dev/null @@ -1,101 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -/// @notice Read and write to persistent storage at a fraction of the cost. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/SSTORE2.sol) -/// @author Modified from 0xSequence (https://github.com/0xSequence/sstore2/blob/master/contracts/SSTORE2.sol) -library SSTORE2 { - uint256 internal constant DATA_OFFSET = 1; // We skip the first byte as it's a STOP opcode to ensure the contract can't be called. - - /*////////////////////////////////////////////////////////////// - WRITE LOGIC - //////////////////////////////////////////////////////////////*/ - - function write(bytes memory data) internal returns (address pointer) { - // Prefix the bytecode with a STOP opcode to ensure it cannot be called. - bytes memory runtimeCode = abi.encodePacked(hex"00", data); - - bytes memory creationCode = abi.encodePacked( - //---------------------------------------------------------------------------------------------------------------// - // Opcode | Opcode + Arguments | Description | Stack View // - //---------------------------------------------------------------------------------------------------------------// - // 0x60 | 0x600B | PUSH1 11 | codeOffset // - // 0x59 | 0x59 | MSIZE | 0 codeOffset // - // 0x81 | 0x81 | DUP2 | codeOffset 0 codeOffset // - // 0x38 | 0x38 | CODESIZE | codeSize codeOffset 0 codeOffset // - // 0x03 | 0x03 | SUB | (codeSize - codeOffset) 0 codeOffset // - // 0x80 | 0x80 | DUP | (codeSize - codeOffset) (codeSize - codeOffset) 0 codeOffset // - // 0x92 | 0x92 | SWAP3 | codeOffset (codeSize - codeOffset) 0 (codeSize - codeOffset) // - // 0x59 | 0x59 | MSIZE | 0 codeOffset (codeSize - codeOffset) 0 (codeSize - codeOffset) // - // 0x39 | 0x39 | CODECOPY | 0 (codeSize - codeOffset) // - // 0xf3 | 0xf3 | RETURN | // - //---------------------------------------------------------------------------------------------------------------// - hex"60_0B_59_81_38_03_80_92_59_39_F3", // Returns all code in the contract except for the first 11 (0B in hex) bytes. - runtimeCode // The bytecode we want the contract to have after deployment. Capped at 1 byte less than the code size limit. - ); - - /// @solidity memory-safe-assembly - assembly { - // Deploy a new contract with the generated creation code. - // We start 32 bytes into the code to avoid copying the byte length. - pointer := create(0, add(creationCode, 32), mload(creationCode)) - } - - require(pointer != address(0), "DEPLOYMENT_FAILED"); - } - - /*////////////////////////////////////////////////////////////// - READ LOGIC - //////////////////////////////////////////////////////////////*/ - - function read(address pointer) internal view returns (bytes memory) { - return readBytecode(pointer, DATA_OFFSET, pointer.code.length - DATA_OFFSET); - } - - function read(address pointer, uint256 start) internal view returns (bytes memory) { - start += DATA_OFFSET; - - return readBytecode(pointer, start, pointer.code.length - start); - } - - function read( - address pointer, - uint256 start, - uint256 end - ) internal view returns (bytes memory) { - start += DATA_OFFSET; - end += DATA_OFFSET; - - require(pointer.code.length >= end, "OUT_OF_BOUNDS"); - - return readBytecode(pointer, start, end - start); - } - - /*////////////////////////////////////////////////////////////// - INTERNAL HELPER LOGIC - //////////////////////////////////////////////////////////////*/ - - function readBytecode( - address pointer, - uint256 start, - uint256 size - ) private view returns (bytes memory data) { - /// @solidity memory-safe-assembly - assembly { - // Get a pointer to some free memory. - data := mload(0x40) - - // Update the free memory pointer to prevent overriding our data. - // We use and(x, not(31)) as a cheaper equivalent to sub(x, mod(x, 32)). - // Adding 31 to size and running the result through the logic above ensures - // the memory pointer remains word-aligned, following the Solidity convention. - mstore(0x40, add(data, and(add(add(size, 32), 31), not(31)))) - - // Store the size of the data in the first 32 byte chunk of free memory. - mstore(data, size) - - // Copy the code into memory right after the 32 bytes we used to store the size. - extcodecopy(pointer, add(data, 32), start, size) - } - } -} diff --git a/lib/solmate/src/utils/SafeCastLib.sol b/lib/solmate/src/utils/SafeCastLib.sol deleted file mode 100644 index 9e8a2af..0000000 --- a/lib/solmate/src/utils/SafeCastLib.sol +++ /dev/null @@ -1,193 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -/// @notice Safe unsigned integer casting library that reverts on overflow. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/SafeCastLib.sol) -/// @author Modified from OpenZeppelin (https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/math/SafeCast.sol) -library SafeCastLib { - function safeCastTo248(uint256 x) internal pure returns (uint248 y) { - require(x < 1 << 248); - - y = uint248(x); - } - - function safeCastTo240(uint256 x) internal pure returns (uint240 y) { - require(x < 1 << 240); - - y = uint240(x); - } - - function safeCastTo232(uint256 x) internal pure returns (uint232 y) { - require(x < 1 << 232); - - y = uint232(x); - } - - function safeCastTo224(uint256 x) internal pure returns (uint224 y) { - require(x < 1 << 224); - - y = uint224(x); - } - - function safeCastTo216(uint256 x) internal pure returns (uint216 y) { - require(x < 1 << 216); - - y = uint216(x); - } - - function safeCastTo208(uint256 x) internal pure returns (uint208 y) { - require(x < 1 << 208); - - y = uint208(x); - } - - function safeCastTo200(uint256 x) internal pure returns (uint200 y) { - require(x < 1 << 200); - - y = uint200(x); - } - - function safeCastTo192(uint256 x) internal pure returns (uint192 y) { - require(x < 1 << 192); - - y = uint192(x); - } - - function safeCastTo184(uint256 x) internal pure returns (uint184 y) { - require(x < 1 << 184); - - y = uint184(x); - } - - function safeCastTo176(uint256 x) internal pure returns (uint176 y) { - require(x < 1 << 176); - - y = uint176(x); - } - - function safeCastTo168(uint256 x) internal pure returns (uint168 y) { - require(x < 1 << 168); - - y = uint168(x); - } - - function safeCastTo160(uint256 x) internal pure returns (uint160 y) { - require(x < 1 << 160); - - y = uint160(x); - } - - function safeCastTo152(uint256 x) internal pure returns (uint152 y) { - require(x < 1 << 152); - - y = uint152(x); - } - - function safeCastTo144(uint256 x) internal pure returns (uint144 y) { - require(x < 1 << 144); - - y = uint144(x); - } - - function safeCastTo136(uint256 x) internal pure returns (uint136 y) { - require(x < 1 << 136); - - y = uint136(x); - } - - function safeCastTo128(uint256 x) internal pure returns (uint128 y) { - require(x < 1 << 128); - - y = uint128(x); - } - - function safeCastTo120(uint256 x) internal pure returns (uint120 y) { - require(x < 1 << 120); - - y = uint120(x); - } - - function safeCastTo112(uint256 x) internal pure returns (uint112 y) { - require(x < 1 << 112); - - y = uint112(x); - } - - function safeCastTo104(uint256 x) internal pure returns (uint104 y) { - require(x < 1 << 104); - - y = uint104(x); - } - - function safeCastTo96(uint256 x) internal pure returns (uint96 y) { - require(x < 1 << 96); - - y = uint96(x); - } - - function safeCastTo88(uint256 x) internal pure returns (uint88 y) { - require(x < 1 << 88); - - y = uint88(x); - } - - function safeCastTo80(uint256 x) internal pure returns (uint80 y) { - require(x < 1 << 80); - - y = uint80(x); - } - - function safeCastTo72(uint256 x) internal pure returns (uint72 y) { - require(x < 1 << 72); - - y = uint72(x); - } - - function safeCastTo64(uint256 x) internal pure returns (uint64 y) { - require(x < 1 << 64); - - y = uint64(x); - } - - function safeCastTo56(uint256 x) internal pure returns (uint56 y) { - require(x < 1 << 56); - - y = uint56(x); - } - - function safeCastTo48(uint256 x) internal pure returns (uint48 y) { - require(x < 1 << 48); - - y = uint48(x); - } - - function safeCastTo40(uint256 x) internal pure returns (uint40 y) { - require(x < 1 << 40); - - y = uint40(x); - } - - function safeCastTo32(uint256 x) internal pure returns (uint32 y) { - require(x < 1 << 32); - - y = uint32(x); - } - - function safeCastTo24(uint256 x) internal pure returns (uint24 y) { - require(x < 1 << 24); - - y = uint24(x); - } - - function safeCastTo16(uint256 x) internal pure returns (uint16 y) { - require(x < 1 << 16); - - y = uint16(x); - } - - function safeCastTo8(uint256 x) internal pure returns (uint8 y) { - require(x < 1 << 8); - - y = uint8(x); - } -} diff --git a/lib/solmate/src/utils/SafeTransferLib.sol b/lib/solmate/src/utils/SafeTransferLib.sol deleted file mode 100644 index 93ef5a9..0000000 --- a/lib/solmate/src/utils/SafeTransferLib.sol +++ /dev/null @@ -1,128 +0,0 @@ -// SPDX-License-Identifier: AGPL-3.0-only -pragma solidity >=0.8.0; - -import {ERC20} from "../tokens/ERC20.sol"; - -/// @notice Safe ETH and ERC20 transfer library that gracefully handles missing return values. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/SafeTransferLib.sol) -/// @dev Use with caution! Some functions in this library knowingly create dirty bits at the destination of the free memory pointer. -/// @dev Note that none of the functions in this library check that a token has code at all! That responsibility is delegated to the caller. -library SafeTransferLib { - /*////////////////////////////////////////////////////////////// - ETH OPERATIONS - //////////////////////////////////////////////////////////////*/ - - function safeTransferETH(address to, uint256 amount) internal { - bool success; - - /// @solidity memory-safe-assembly - assembly { - // Transfer the ETH and store if it succeeded or not. - success := call(gas(), to, amount, 0, 0, 0, 0) - } - - require(success, "ETH_TRANSFER_FAILED"); - } - - /*////////////////////////////////////////////////////////////// - ERC20 OPERATIONS - //////////////////////////////////////////////////////////////*/ - - function safeTransferFrom( - ERC20 token, - address from, - address to, - uint256 amount - ) internal { - bool success; - - /// @solidity memory-safe-assembly - assembly { - // Get a pointer to some free memory. - let freeMemoryPointer := mload(0x40) - - // Write the abi-encoded calldata into memory, beginning with the function selector. - mstore(freeMemoryPointer, 0x23b872dd00000000000000000000000000000000000000000000000000000000) - mstore(add(freeMemoryPointer, 4), and(from, 0xffffffffffffffffffffffffffffffffffffffff)) // Append and mask the "from" argument. - mstore(add(freeMemoryPointer, 36), and(to, 0xffffffffffffffffffffffffffffffffffffffff)) // Append and mask the "to" argument. - mstore(add(freeMemoryPointer, 68), amount) // Append the "amount" argument. Masking not required as it's a full 32 byte type. - - success := and( - // Set success to whether the call reverted, if not we check it either - // returned exactly 1 (can't just be non-zero data), or had no return data. - or(and(eq(mload(0), 1), gt(returndatasize(), 31)), iszero(returndatasize())), - // We use 100 because the length of our calldata totals up like so: 4 + 32 * 3. - // We use 0 and 32 to copy up to 32 bytes of return data into the scratch space. - // Counterintuitively, this call must be positioned second to the or() call in the - // surrounding and() call or else returndatasize() will be zero during the computation. - call(gas(), token, 0, freeMemoryPointer, 100, 0, 32) - ) - } - - require(success, "TRANSFER_FROM_FAILED"); - } - - function safeTransfer( - ERC20 token, - address to, - uint256 amount - ) internal { - bool success; - - /// @solidity memory-safe-assembly - assembly { - // Get a pointer to some free memory. - let freeMemoryPointer := mload(0x40) - - // Write the abi-encoded calldata into memory, beginning with the function selector. - mstore(freeMemoryPointer, 0xa9059cbb00000000000000000000000000000000000000000000000000000000) - mstore(add(freeMemoryPointer, 4), and(to, 0xffffffffffffffffffffffffffffffffffffffff)) // Append and mask the "to" argument. - mstore(add(freeMemoryPointer, 36), amount) // Append the "amount" argument. Masking not required as it's a full 32 byte type. - - success := and( - // Set success to whether the call reverted, if not we check it either - // returned exactly 1 (can't just be non-zero data), or had no return data. - or(and(eq(mload(0), 1), gt(returndatasize(), 31)), iszero(returndatasize())), - // We use 68 because the length of our calldata totals up like so: 4 + 32 * 2. - // We use 0 and 32 to copy up to 32 bytes of return data into the scratch space. - // Counterintuitively, this call must be positioned second to the or() call in the - // surrounding and() call or else returndatasize() will be zero during the computation. - call(gas(), token, 0, freeMemoryPointer, 68, 0, 32) - ) - } - - require(success, "TRANSFER_FAILED"); - } - - function safeApprove( - ERC20 token, - address to, - uint256 amount - ) internal { - bool success; - - /// @solidity memory-safe-assembly - assembly { - // Get a pointer to some free memory. - let freeMemoryPointer := mload(0x40) - - // Write the abi-encoded calldata into memory, beginning with the function selector. - mstore(freeMemoryPointer, 0x095ea7b300000000000000000000000000000000000000000000000000000000) - mstore(add(freeMemoryPointer, 4), and(to, 0xffffffffffffffffffffffffffffffffffffffff)) // Append and mask the "to" argument. - mstore(add(freeMemoryPointer, 36), amount) // Append the "amount" argument. Masking not required as it's a full 32 byte type. - - success := and( - // Set success to whether the call reverted, if not we check it either - // returned exactly 1 (can't just be non-zero data), or had no return data. - or(and(eq(mload(0), 1), gt(returndatasize(), 31)), iszero(returndatasize())), - // We use 68 because the length of our calldata totals up like so: 4 + 32 * 2. - // We use 0 and 32 to copy up to 32 bytes of return data into the scratch space. - // Counterintuitively, this call must be positioned second to the or() call in the - // surrounding and() call or else returndatasize() will be zero during the computation. - call(gas(), token, 0, freeMemoryPointer, 68, 0, 32) - ) - } - - require(success, "APPROVE_FAILED"); - } -} diff --git a/lib/solmate/src/utils/SignedWadMath.sol b/lib/solmate/src/utils/SignedWadMath.sol deleted file mode 100644 index e7d30a4..0000000 --- a/lib/solmate/src/utils/SignedWadMath.sol +++ /dev/null @@ -1,245 +0,0 @@ -// SPDX-License-Identifier: MIT -pragma solidity >=0.8.0; - -/// @notice Signed 18 decimal fixed point (wad) arithmetic library. -/// @author Solmate (https://github.com/transmissions11/solmate/blob/main/src/utils/SignedWadMath.sol) -/// @author Modified from Remco Bloemen (https://xn--2-umb.com/22/exp-ln/index.html) - -/// @dev Will not revert on overflow, only use where overflow is not possible. -function toWadUnsafe(uint256 x) pure returns (int256 r) { - /// @solidity memory-safe-assembly - assembly { - // Multiply x by 1e18. - r := mul(x, 1000000000000000000) - } -} - -/// @dev Takes an integer amount of seconds and converts it to a wad amount of days. -/// @dev Will not revert on overflow, only use where overflow is not possible. -/// @dev Not meant for negative second amounts, it assumes x is positive. -function toDaysWadUnsafe(uint256 x) pure returns (int256 r) { - /// @solidity memory-safe-assembly - assembly { - // Multiply x by 1e18 and then divide it by 86400. - r := div(mul(x, 1000000000000000000), 86400) - } -} - -/// @dev Takes a wad amount of days and converts it to an integer amount of seconds. -/// @dev Will not revert on overflow, only use where overflow is not possible. -/// @dev Not meant for negative day amounts, it assumes x is positive. -function fromDaysWadUnsafe(int256 x) pure returns (uint256 r) { - /// @solidity memory-safe-assembly - assembly { - // Multiply x by 86400 and then divide it by 1e18. - r := div(mul(x, 86400), 1000000000000000000) - } -} - -/// @dev Will not revert on overflow, only use where overflow is not possible. -function unsafeWadMul(int256 x, int256 y) pure returns (int256 r) { - /// @solidity memory-safe-assembly - assembly { - // Multiply x by y and divide by 1e18. - r := sdiv(mul(x, y), 1000000000000000000) - } -} - -/// @dev Will return 0 instead of reverting if y is zero and will -/// not revert on overflow, only use where overflow is not possible. -function unsafeWadDiv(int256 x, int256 y) pure returns (int256 r) { - /// @solidity memory-safe-assembly - assembly { - // Multiply x by 1e18 and divide it by y. - r := sdiv(mul(x, 1000000000000000000), y) - } -} - -function wadMul(int256 x, int256 y) pure returns (int256 r) { - /// @solidity memory-safe-assembly - assembly { - // Store x * y in r for now. - r := mul(x, y) - - // Combined overflow check (`x == 0 || (x * y) / x == y`) and edge case check - // where x == -1 and y == type(int256).min, for y == -1 and x == min int256, - // the second overflow check will catch this. - // See: https://secure-contracts.com/learn_evm/arithmetic-checks.html#arithmetic-checks-for-int256-multiplication - // Combining into 1 expression saves gas as resulting bytecode will only have 1 `JUMPI` - // rather than 2. - if iszero( - and( - or(iszero(x), eq(sdiv(r, x), y)), - or(lt(x, not(0)), sgt(y, 0x8000000000000000000000000000000000000000000000000000000000000000)) - ) - ) { - revert(0, 0) - } - - // Scale the result down by 1e18. - r := sdiv(r, 1000000000000000000) - } -} - -function wadDiv(int256 x, int256 y) pure returns (int256 r) { - /// @solidity memory-safe-assembly - assembly { - // Store x * 1e18 in r for now. - r := mul(x, 1000000000000000000) - - // Equivalent to require(y != 0 && ((x * 1e18) / 1e18 == x)) - if iszero(and(iszero(iszero(y)), eq(sdiv(r, 1000000000000000000), x))) { - revert(0, 0) - } - - // Divide r by y. - r := sdiv(r, y) - } -} - -/// @dev Will not work with negative bases, only use when x is positive. -function wadPow(int256 x, int256 y) pure returns (int256) { - // Equivalent to x to the power of y because x ** y = (e ** ln(x)) ** y = e ** (ln(x) * y) - return wadExp((wadLn(x) * y) / 1e18); // Using ln(x) means x must be greater than 0. -} - -function wadExp(int256 x) pure returns (int256 r) { - unchecked { - // When the result is < 0.5 we return zero. This happens when - // x <= floor(log(0.5e18) * 1e18) ~ -42e18 - if (x <= -42139678854452767551) return 0; - - // When the result is > (2**255 - 1) / 1e18 we can not represent it as an - // int. This happens when x >= floor(log((2**255 - 1) / 1e18) * 1e18) ~ 135. - if (x >= 135305999368893231589) revert("EXP_OVERFLOW"); - - // x is now in the range (-42, 136) * 1e18. Convert to (-42, 136) * 2**96 - // for more intermediate precision and a binary basis. This base conversion - // is a multiplication by 1e18 / 2**96 = 5**18 / 2**78. - x = (x << 78) / 5**18; - - // Reduce range of x to (-½ ln 2, ½ ln 2) * 2**96 by factoring out powers - // of two such that exp(x) = exp(x') * 2**k, where k is an integer. - // Solving this gives k = round(x / log(2)) and x' = x - k * log(2). - int256 k = ((x << 96) / 54916777467707473351141471128 + 2**95) >> 96; - x = x - k * 54916777467707473351141471128; - - // k is in the range [-61, 195]. - - // Evaluate using a (6, 7)-term rational approximation. - // p is made monic, we'll multiply by a scale factor later. - int256 y = x + 1346386616545796478920950773328; - y = ((y * x) >> 96) + 57155421227552351082224309758442; - int256 p = y + x - 94201549194550492254356042504812; - p = ((p * y) >> 96) + 28719021644029726153956944680412240; - p = p * x + (4385272521454847904659076985693276 << 96); - - // We leave p in 2**192 basis so we don't need to scale it back up for the division. - int256 q = x - 2855989394907223263936484059900; - q = ((q * x) >> 96) + 50020603652535783019961831881945; - q = ((q * x) >> 96) - 533845033583426703283633433725380; - q = ((q * x) >> 96) + 3604857256930695427073651918091429; - q = ((q * x) >> 96) - 14423608567350463180887372962807573; - q = ((q * x) >> 96) + 26449188498355588339934803723976023; - - /// @solidity memory-safe-assembly - assembly { - // Div in assembly because solidity adds a zero check despite the unchecked. - // The q polynomial won't have zeros in the domain as all its roots are complex. - // No scaling is necessary because p is already 2**96 too large. - r := sdiv(p, q) - } - - // r should be in the range (0.09, 0.25) * 2**96. - - // We now need to multiply r by: - // * the scale factor s = ~6.031367120. - // * the 2**k factor from the range reduction. - // * the 1e18 / 2**96 factor for base conversion. - // We do this all at once, with an intermediate result in 2**213 - // basis, so the final right shift is always by a positive amount. - r = int256((uint256(r) * 3822833074963236453042738258902158003155416615667) >> uint256(195 - k)); - } -} - -function wadLn(int256 x) pure returns (int256 r) { - unchecked { - require(x > 0, "UNDEFINED"); - - // We want to convert x from 10**18 fixed point to 2**96 fixed point. - // We do this by multiplying by 2**96 / 10**18. But since - // ln(x * C) = ln(x) + ln(C), we can simply do nothing here - // and add ln(2**96 / 10**18) at the end. - - /// @solidity memory-safe-assembly - assembly { - r := shl(7, lt(0xffffffffffffffffffffffffffffffff, x)) - r := or(r, shl(6, lt(0xffffffffffffffff, shr(r, x)))) - r := or(r, shl(5, lt(0xffffffff, shr(r, x)))) - r := or(r, shl(4, lt(0xffff, shr(r, x)))) - r := or(r, shl(3, lt(0xff, shr(r, x)))) - r := or(r, shl(2, lt(0xf, shr(r, x)))) - r := or(r, shl(1, lt(0x3, shr(r, x)))) - r := or(r, lt(0x1, shr(r, x))) - } - - // Reduce range of x to (1, 2) * 2**96 - // ln(2^k * x) = k * ln(2) + ln(x) - int256 k = r - 96; - x <<= uint256(159 - k); - x = int256(uint256(x) >> 159); - - // Evaluate using a (8, 8)-term rational approximation. - // p is made monic, we will multiply by a scale factor later. - int256 p = x + 3273285459638523848632254066296; - p = ((p * x) >> 96) + 24828157081833163892658089445524; - p = ((p * x) >> 96) + 43456485725739037958740375743393; - p = ((p * x) >> 96) - 11111509109440967052023855526967; - p = ((p * x) >> 96) - 45023709667254063763336534515857; - p = ((p * x) >> 96) - 14706773417378608786704636184526; - p = p * x - (795164235651350426258249787498 << 96); - - // We leave p in 2**192 basis so we don't need to scale it back up for the division. - // q is monic by convention. - int256 q = x + 5573035233440673466300451813936; - q = ((q * x) >> 96) + 71694874799317883764090561454958; - q = ((q * x) >> 96) + 283447036172924575727196451306956; - q = ((q * x) >> 96) + 401686690394027663651624208769553; - q = ((q * x) >> 96) + 204048457590392012362485061816622; - q = ((q * x) >> 96) + 31853899698501571402653359427138; - q = ((q * x) >> 96) + 909429971244387300277376558375; - /// @solidity memory-safe-assembly - assembly { - // Div in assembly because solidity adds a zero check despite the unchecked. - // The q polynomial is known not to have zeros in the domain. - // No scaling required because p is already 2**96 too large. - r := sdiv(p, q) - } - - // r is in the range (0, 0.125) * 2**96 - - // Finalization, we need to: - // * multiply by the scale factor s = 5.549… - // * add ln(2**96 / 10**18) - // * add k * ln(2) - // * multiply by 10**18 / 2**96 = 5**18 >> 78 - - // mul s * 5e18 * 2**96, base is now 5**18 * 2**192 - r *= 1677202110996718588342820967067443963516166; - // add ln(2) * k * 5e18 * 2**192 - r += 16597577552685614221487285958193947469193820559219878177908093499208371 * k; - // add ln(2**96 / 10**18) * 5e18 * 2**192 - r += 600920179829731861736702779321621459595472258049074101567377883020018308; - // base conversion: mul 2**18 / 2**192 - r >>= 174; - } -} - -/// @dev Will return 0 instead of reverting if y is zero. -function unsafeDiv(int256 x, int256 y) pure returns (int256 r) { - /// @solidity memory-safe-assembly - assembly { - // Divide x by y. - r := sdiv(x, y) - } -} diff --git a/test/token-voting/LlamaTokenVotingFactory.t.sol b/test/token-voting/LlamaTokenVotingFactory.t.sol index b3a40f7..73b9354 100644 --- a/test/token-voting/LlamaTokenVotingFactory.t.sol +++ b/test/token-voting/LlamaTokenVotingFactory.t.sol @@ -118,3 +118,4 @@ contract DeployTokenVotingModule is LlamaTokenVotingFactoryTest { CORE.executeAction(actionInfo); } } +