Packing.js 2.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  1. const format = require('../format-lines');
  2. const { product } = require('../../helpers');
  3. const { SIZES } = require('./Packing.opts');
  4. // TEMPLATE
  5. const header = `\
  6. pragma solidity ^0.8.20;
  7. /**
  8. * @dev Helper library packing and unpacking multiple values into bytesXX.
  9. *
  10. * Example usage:
  11. *
  12. * \`\`\`solidity
  13. * library MyPacker {
  14. * type MyType is bytes32;
  15. *
  16. * function _pack(address account, bytes4 selector, uint64 period) external pure returns (MyType) {
  17. * bytes12 subpack = Packing.pack_4_8(selector, bytes8(period));
  18. * bytes32 pack = Packing.pack_20_12(bytes20(account), subpack);
  19. * return MyType.wrap(pack);
  20. * }
  21. *
  22. * function _unpack(MyType self) external pure returns (address, bytes4, uint64) {
  23. * bytes32 pack = MyType.unwrap(self);
  24. * return (
  25. * address(Packing.extract_32_20(pack, 0)),
  26. * Packing.extract_32_4(pack, 20),
  27. * uint64(Packing.extract_32_8(pack, 24))
  28. * );
  29. * }
  30. * }
  31. * \`\`\`
  32. */
  33. // solhint-disable func-name-mixedcase
  34. `;
  35. const errors = `\
  36. error OutOfRangeAccess();
  37. `;
  38. const pack = (left, right) => `\
  39. function pack_${left}_${right}(bytes${left} left, bytes${right} right) internal pure returns (bytes${
  40. left + right
  41. } result) {
  42. assembly ("memory-safe") {
  43. left := and(left, shl(${256 - 8 * left}, not(0)))
  44. right := and(right, shl(${256 - 8 * right}, not(0)))
  45. result := or(left, shr(${8 * left}, right))
  46. }
  47. }
  48. `;
  49. const extract = (outer, inner) => `\
  50. function extract_${outer}_${inner}(bytes${outer} self, uint8 offset) internal pure returns (bytes${inner} result) {
  51. if (offset > ${outer - inner}) revert OutOfRangeAccess();
  52. assembly ("memory-safe") {
  53. result := and(shl(mul(8, offset), self), shl(${256 - 8 * inner}, not(0)))
  54. }
  55. }
  56. `;
  57. const replace = (outer, inner) => `\
  58. function replace_${outer}_${inner}(bytes${outer} self, bytes${inner} value, uint8 offset) internal pure returns (bytes${outer} result) {
  59. bytes${inner} oldValue = extract_${outer}_${inner}(self, offset);
  60. assembly ("memory-safe") {
  61. value := and(value, shl(${256 - 8 * inner}, not(0)))
  62. result := xor(self, shr(mul(8, offset), xor(oldValue, value)))
  63. }
  64. }
  65. `;
  66. // GENERATE
  67. module.exports = format(
  68. header.trimEnd(),
  69. 'library Packing {',
  70. format(
  71. [].concat(
  72. errors,
  73. product(SIZES, SIZES)
  74. .filter(([left, right]) => SIZES.includes(left + right))
  75. .map(([left, right]) => pack(left, right)),
  76. product(SIZES, SIZES)
  77. .filter(([outer, inner]) => outer > inner)
  78. .flatMap(([outer, inner]) => [extract(outer, inner), replace(outer, inner)]),
  79. ),
  80. ).trimEnd(),
  81. '}',
  82. );