ERC1155Burnable.spec 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. methods {
  2. balanceOf(address, uint256) returns uint256 envfree
  3. isApprovedForAll(address,address) returns bool envfree
  4. }
  5. /// If a method call reduces account balances, the caller must be either the
  6. /// holder of the account or approved to act on the holder's behalf.
  7. /// n.b. This rule is passing for all methods except `_burn` and `_burnBatch`,
  8. /// ordinarily internal methods that are callable by our tool only because they
  9. /// were changed to public for the purposes of verification.
  10. rule onlyHolderOrApprovedCanReduceBalance {
  11. address holder; uint256 token; uint256 amount;
  12. uint256 balanceBefore = balanceOf(holder, token);
  13. method f; env e; calldataarg args;
  14. f(e, args);
  15. uint256 balanceAfter = balanceOf(holder, token);
  16. assert balanceAfter < balanceBefore => e.msg.sender == holder || isApprovedForAll(holder, e.msg.sender),
  17. "An account balance may only be reduced by the holder or a holder-approved agent";
  18. }
  19. /// Burning a larger amount of a token must reduce that token's balance more
  20. /// than burning a smaller amount.
  21. /// n.b. This rule holds for `burnBatch` as well due to rules establishing
  22. /// appropriate equivance between `burn` and `burnBatch` methods.
  23. rule burnAmountProportionalToBalanceReduction {
  24. storage beforeBurn = lastStorage;
  25. env e;
  26. address holder; uint256 token;
  27. mathint startingBalance = balanceOf(holder, token);
  28. uint256 smallBurn; uint256 largeBurn;
  29. require smallBurn < largeBurn;
  30. // smaller burn amount
  31. burn(e, holder, token, smallBurn) at beforeBurn;
  32. mathint smallBurnBalanceChange = startingBalance - balanceOf(holder, token);
  33. // larger burn amount
  34. burn(e, holder, token, largeBurn) at beforeBurn;
  35. mathint largeBurnBalanceChange = startingBalance - balanceOf(holder, token);
  36. assert smallBurnBalanceChange < largeBurnBalanceChange,
  37. "A larger burn must lead to a larger decrease in balance";
  38. }
  39. /// Two sequential burns must be equivalent to a single burn of the sum of their
  40. /// amounts.
  41. /// n.b. This rule holds for `burnBatch` as well due to rules establishing
  42. /// appropriate equivance between `burn` and `burnBatch` methods.
  43. rule sequentialBurnsEquivalentToSingleBurnOfSum {
  44. storage beforeBurns = lastStorage;
  45. env e;
  46. address holder; uint256 token;
  47. mathint startingBalance = balanceOf(holder, token);
  48. uint256 firstBurn; uint256 secondBurn; uint256 sumBurn;
  49. require sumBurn == firstBurn + secondBurn;
  50. // sequential burns
  51. burn(e, holder, token, firstBurn) at beforeBurns;
  52. burn(e, holder, token, secondBurn);
  53. mathint sequentialBurnsBalanceChange = startingBalance - balanceOf(holder, token);
  54. // burn of sum of sequential burns
  55. burn(e, holder, token, sumBurn) at beforeBurns;
  56. mathint sumBurnBalanceChange = startingBalance - balanceOf(holder, token);
  57. assert sequentialBurnsBalanceChange == sumBurnBalanceChange,
  58. "Sequential burns must be equivalent to a burn of their sum";
  59. }
  60. /// The result of burning a single token must be equivalent whether done via
  61. /// burn or burnBatch.
  62. rule singleTokenBurnBurnBatchEquivalence {
  63. storage beforeBurn = lastStorage;
  64. env e;
  65. address holder;
  66. uint256 token; uint256 burnAmount;
  67. uint256[] tokens; uint256[] burnAmounts;
  68. mathint startingBalance = balanceOf(holder, token);
  69. require tokens.length == 1; require burnAmounts.length == 1;
  70. require tokens[0] == token; require burnAmounts[0] == burnAmount;
  71. // burning via burn
  72. burn(e, holder, token, burnAmount) at beforeBurn;
  73. mathint burnBalanceChange = startingBalance - balanceOf(holder, token);
  74. // burning via burnBatch
  75. burnBatch(e, holder, tokens, burnAmounts) at beforeBurn;
  76. mathint burnBatchBalanceChange = startingBalance - balanceOf(holder, token);
  77. assert burnBalanceChange == burnBatchBalanceChange,
  78. "Burning a single token via burn or burnBatch must be equivalent";
  79. }
  80. /// The results of burning multiple tokens must be equivalent whether done
  81. /// separately via burn or together via burnBatch.
  82. rule multipleTokenBurnBurnBatchEquivalence {
  83. storage beforeBurns = lastStorage;
  84. env e;
  85. address holder;
  86. uint256 tokenA; uint256 tokenB; uint256 tokenC;
  87. uint256 burnAmountA; uint256 burnAmountB; uint256 burnAmountC;
  88. uint256[] tokens; uint256[] burnAmounts;
  89. mathint startingBalanceA = balanceOf(holder, tokenA);
  90. mathint startingBalanceB = balanceOf(holder, tokenB);
  91. mathint startingBalanceC = balanceOf(holder, tokenC);
  92. require tokens.length == 3; require burnAmounts.length == 3;
  93. require tokens[0] == tokenA; require burnAmounts[0] == burnAmountA;
  94. require tokens[1] == tokenB; require burnAmounts[1] == burnAmountB;
  95. require tokens[2] == tokenC; require burnAmounts[2] == burnAmountC;
  96. // burning via burn
  97. burn(e, holder, tokenA, burnAmountA) at beforeBurns;
  98. burn(e, holder, tokenB, burnAmountB);
  99. burn(e, holder, tokenC, burnAmountC);
  100. mathint burnBalanceChangeA = startingBalanceA - balanceOf(holder, tokenA);
  101. mathint burnBalanceChangeB = startingBalanceB - balanceOf(holder, tokenB);
  102. mathint burnBalanceChangeC = startingBalanceC - balanceOf(holder, tokenC);
  103. // burning via burnBatch
  104. burnBatch(e, holder, tokens, burnAmounts) at beforeBurns;
  105. mathint burnBatchBalanceChangeA = startingBalanceA - balanceOf(holder, tokenA);
  106. mathint burnBatchBalanceChangeB = startingBalanceB - balanceOf(holder, tokenB);
  107. mathint burnBatchBalanceChangeC = startingBalanceC - balanceOf(holder, tokenC);
  108. assert burnBalanceChangeA == burnBatchBalanceChangeA
  109. && burnBalanceChangeB == burnBatchBalanceChangeB
  110. && burnBalanceChangeC == burnBatchBalanceChangeC,
  111. "Burning multiple tokens via burn or burnBatch must be equivalent";
  112. }
  113. /// If passed empty token and burn amount arrays, burnBatch must not change
  114. /// token balances or address permissions.
  115. rule burnBatchOnEmptyArraysChangesNothing {
  116. env e;
  117. address holder; uint256 token;
  118. address nonHolderA; address nonHolderB;
  119. uint256 startingBalance = balanceOf(holder, token);
  120. bool startingPermissionNonHolderA = isApprovedForAll(holder, nonHolderA);
  121. bool startingPermissionNonHolderB = isApprovedForAll(holder, nonHolderB);
  122. uint256[] noTokens; uint256[] noBurnAmounts;
  123. require noTokens.length == 0; require noBurnAmounts.length == 0;
  124. burnBatch(e, holder, noTokens, noBurnAmounts);
  125. uint256 endingBalance = balanceOf(holder, token);
  126. bool endingPermissionNonHolderA = isApprovedForAll(holder, nonHolderA);
  127. bool endingPermissionNonHolderB = isApprovedForAll(holder, nonHolderB);
  128. assert startingBalance == endingBalance,
  129. "burnBatch must not change token balances if passed empty arrays";
  130. assert startingPermissionNonHolderA == endingPermissionNonHolderA
  131. && startingPermissionNonHolderB == endingPermissionNonHolderB,
  132. "burnBatch must not change account permissions if passed empty arrays";
  133. }
  134. /// This rule should always fail.
  135. rule sanity {
  136. method f; env e; calldataarg args;
  137. f(e, args);
  138. assert false,
  139. "This rule should always fail";
  140. }