Packing.js 2.7 KB

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