123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120 |
- // SPDX-License-Identifier: MIT
- // OpenZeppelin Contracts (last updated v5.0.2) (utils/Base64.sol)
- pragma solidity ^0.8.20;
- /**
- * @dev Provides a set of functions to operate with Base64 strings.
- */
- library Base64 {
- /**
- * @dev Base64 Encoding/Decoding Table
- * See sections 4 and 5 of https://datatracker.ietf.org/doc/html/rfc4648
- */
- string internal constant _TABLE = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
- string internal constant _TABLE_URL = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_";
- /**
- * @dev Converts a `bytes` to its Bytes64 `string` representation.
- */
- function encode(bytes memory data) internal pure returns (string memory) {
- return _encode(data, _TABLE, true);
- }
- /**
- * @dev Converts a `bytes` to its Bytes64Url `string` representation.
- */
- function encodeURL(bytes memory data) internal pure returns (string memory) {
- return _encode(data, _TABLE_URL, false);
- }
- /**
- * @dev Internal table-agnostic conversion
- */
- function _encode(bytes memory data, string memory table, bool withPadding) private pure returns (string memory) {
- /**
- * Inspired by Brecht Devos (Brechtpd) implementation - MIT licence
- * https://github.com/Brechtpd/base64/blob/e78d9fd951e7b0977ddca77d92dc85183770daf4/base64.sol
- */
- if (data.length == 0) return "";
- // If padding is enabled, the final length should be `bytes` data length divided by 3 rounded up and then
- // multiplied by 4 so that it leaves room for padding the last chunk
- // - `data.length + 2` -> Round up
- // - `/ 3` -> Number of 3-bytes chunks
- // - `4 *` -> 4 characters for each chunk
- // If padding is disabled, the final length should be `bytes` data length multiplied by 4/3 rounded up as
- // opposed to when padding is required to fill the last chunk.
- // - `4 *` -> 4 characters for each chunk
- // - `data.length + 2` -> Round up
- // - `/ 3` -> Number of 3-bytes chunks
- uint256 resultLength = withPadding ? 4 * ((data.length + 2) / 3) : (4 * data.length + 2) / 3;
- string memory result = new string(resultLength);
- /// @solidity memory-safe-assembly
- assembly {
- // Prepare the lookup table (skip the first "length" byte)
- let tablePtr := add(table, 1)
- // Prepare result pointer, jump over length
- let resultPtr := add(result, 0x20)
- let dataPtr := data
- let endPtr := add(data, mload(data))
- // In some cases, the last iteration will read bytes after the end of the data. We cache the value, and
- // set it to zero to make sure no dirty bytes are read in that section.
- let afterPtr := add(endPtr, 0x20)
- let afterCache := mload(afterPtr)
- mstore(afterPtr, 0x00)
- // Run over the input, 3 bytes at a time
- for {
- } lt(dataPtr, endPtr) {
- } {
- // Advance 3 bytes
- dataPtr := add(dataPtr, 3)
- let input := mload(dataPtr)
- // To write each character, shift the 3 byte (24 bits) chunk
- // 4 times in blocks of 6 bits for each character (18, 12, 6, 0)
- // and apply logical AND with 0x3F to bitmask the least significant 6 bits.
- // Use this as an index into the lookup table, mload an entire word
- // so the desired character is in the least significant byte, and
- // mstore8 this least significant byte into the result and continue.
- mstore8(resultPtr, mload(add(tablePtr, and(shr(18, input), 0x3F))))
- resultPtr := add(resultPtr, 1) // Advance
- mstore8(resultPtr, mload(add(tablePtr, and(shr(12, input), 0x3F))))
- resultPtr := add(resultPtr, 1) // Advance
- mstore8(resultPtr, mload(add(tablePtr, and(shr(6, input), 0x3F))))
- resultPtr := add(resultPtr, 1) // Advance
- mstore8(resultPtr, mload(add(tablePtr, and(input, 0x3F))))
- resultPtr := add(resultPtr, 1) // Advance
- }
- // Reset the value that was cached
- mstore(afterPtr, afterCache)
- if withPadding {
- // When data `bytes` is not exactly 3 bytes long
- // it is padded with `=` characters at the end
- switch mod(mload(data), 3)
- case 1 {
- mstore8(sub(resultPtr, 1), 0x3d)
- mstore8(sub(resultPtr, 2), 0x3d)
- }
- case 2 {
- mstore8(sub(resultPtr, 1), 0x3d)
- }
- }
- }
- return result;
- }
- }
|