All you can cache
When using rollups the cost of a byte in the transaction is a lot more expensive than the cost of a storage slot. Therefore, it makes sense to cache as much information as possible onchain.
In this article you learn how to create and use a caching contract in such a way that any parameter value that is likely to be used multiple times will be cached and available for use (after the first time) with a much smaller number of bytes, and how to write offchain code that uses this cache.
If you want to skip the article and just see the source code, it is here. The development stack is Foundry.
Overall design
For the sake of simplicity we'll assume all transaction parameters are uint256, 32 bytes long. When we receive a transaction, we'll parse each parameter like this:
- 
If the first byte is 0xFF, take the next 32 bytes as a parameter value and write it to the cache.
- 
If the first byte is 0xFE, take the next 32 bytes as a parameter value but do not write it to the cache.
- 
For any other value, take the top four bits as the number of additional bytes, and the bottom four bits as the most significant bits of the cache key. Here are some examples: Bytes in calldata Cache key 0x0F 0x0F 0x10,0x10 0x10 0x12,0xAC 0x02AC 0x2D,0xEA, 0xD6 0x0DEAD6 
Cache manipulation
The cache is implemented in Cache.sol. Let's go over it line by line.
1// SPDX-License-Identifier: UNLICENSED2pragma solidity ^0.8.13;345contract Cache {67    bytes1 public constant INTO_CACHE = 0xFF;8    bytes1 public constant DONT_CACHE = 0xFE;These constants are used to interpret the special cases where we provide all the information and either want it written into the cache or not. Writing into the cache requires two SSTORE operations into previously unused storage slots at a cost of 22100 gas each, so we make it optional.
12    mapping(uint => uint) public val2key;A mapping between the values and their keys. This information is necessary to encode values before you send out the transaction.
1    // Location n has the value for key n+1, because we need to preserve2    // zero as "not in the cache".3    uint[] public key2val;We can use an array for the mapping from keys to values because we assign the keys, and for simplicity we do it sequentially.
1    function cacheRead(uint _key) public view returns (uint) {2        require(_key <= key2val.length, "Reading uninitialize cache entry");3        return key2val[_key-1];4    }  // cacheReadRead a value from the cache.
1    // Write a value to the cache if it's not there already2    // Only public to enable the test to work3    function cacheWrite(uint _value) public returns (uint) {4        // If the value is already in the cache, return the current key5        if (val2key[_value] != 0) {6            return val2key[_value];7        }There is no point in putting the same value in the cache more than once. If the value is already there, just return the existing key.
1        // Since 0xFE is a special case, the largest key the cache can2        // hold is 0x0D followed by 15 0xFF's. If the cache length is already that3        // large, fail.4        //                              1 2 3 4 5 6 7 8 9 A B C D E F5        require(key2val.length+1 < 0x0DFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF,6            "cache overflow");I don't think we'll ever get a cache that big (approximately 1.8*1037 entries, which would require about 1027 TB to store). However, I'm old enough to remember "640kB would always be enough". This test is very cheap.
1        // Write the value using the next key2        val2key[_value] = key2val.length+1;Add the reverse lookup (from the value to the key).
1        key2val.push(_value);Add the forward lookup (from the key to the value). Because we assign values sequentially we can just add it after the last array value.
1        return key2val.length;2    }  // cacheWriteReturn the new length of key2val, which is the cell where the new value is stored.
1    function _calldataVal(uint startByte, uint length)2        private pure returns (uint)This function reads a value from the calldata of arbitrary length (up to 32 bytes, the word size).
1    {2        uint _retVal;34        require(length < 0x21,5            "_calldataVal length limit is 32 bytes");6        require(length + startByte <= msg.data.length,7            "_calldataVal trying to read beyond calldatasize");This function is internal, so if the rest of the code is written correctly these tests are not required. However, they don't cost much so we might as well have them.
1        assembly {2            _retVal := calldataload(startByte)3        }This code is in Yul. It reads a 32 byte value from the calldata. This works even if the calldata stops before startByte+32 because uninitialized space in EVM is considered to be zero.
1        _retVal = _retVal >> (256-length*8);We don't necessarily want a 32 byte value. This gets rid of the excess bytes.
1        return _retVal;2    } // _calldataVal345    // Read a single parameter from the calldata, starting at _fromByte6    function _readParam(uint _fromByte) internal7        returns (uint _nextByte, uint _parameterValue)8    {Read a single parameter from the calldata. Note that we need to return not just the value we read, but also the location of the next byte because parameters can range from 1 byte long to 33 bytes.
1        // The first byte tells us how to interpret the rest2        uint8 _firstByte;34        _firstByte = uint8(_calldataVal(_fromByte, 1));Solidity tries to reduce the number of bugs by forbidding potentially dangerous implicit type conversions. A downgrade, for example from 256 bits to 8 bits, needs to be explicit.
12        // Read the value, but do not write it to the cache3        if (_firstByte == uint8(DONT_CACHE))4            return(_fromByte+33, _calldataVal(_fromByte+1, 32));56        // Read the value, and write it to the cache7        if (_firstByte == uint8(INTO_CACHE)) {8            uint _param = _calldataVal(_fromByte+1, 32);9            cacheWrite(_param);10            return(_fromByte+33, _param);11        }1213        // If we got here it means that we need to read from the cache1415        // Number of extra bytes to read16        uint8 _extraBytes = _firstByte / 16;Ratidza zveseTake the lower nibble and combine it with the other bytes to read the value from the cache.
1        uint _key = (uint256(_firstByte & 0x0F) << (8*_extraBytes)) +2            _calldataVal(_fromByte+1, _extraBytes);34        return (_fromByte+_extraBytes+1, cacheRead(_key));56    }  // _readParam789    // Read n parameters (functions know how many parameters they expect)10    function _readParams(uint _paramNum) internal returns (uint[] memory) {Ratidza zveseWe could get the number of parameters we have from the calldata itself, but the functions that call us know how many parameters they expect. It's easier to let them tell us.
1        // The parameters we read2        uint[] memory params = new uint[](_paramNum);34        // Parameters start at byte 4, before that it's the function signature5        uint _atByte = 4;67        for(uint i=0; i<_paramNum; i++) {8            (_atByte, params[i]) = _readParam(_atByte);9        }Ratidza zveseRead the parameters until you have the number you need. If we go past the end of the calldata, _readParams will revert the call.
12        return(params);3    }   // readParams45    // For testing _readParams, test reading four parameters6    function fourParam() public7        returns (uint256,uint256,uint256,uint256)8    {9        uint[] memory params;10        params = _readParams(4);11        return (params[0], params[1], params[2], params[3]);12    }    // fourParamRatidza zveseOne big advantage of Foundry is that it allows tests to be written in Solidity (see Testing the cache below). This makes unit tests a lot easier. This is a function that reads four parameters and returns them so the test can verify they were correct.
1    // Get a value, return bytes that will encode it (using the cache if possible)2    function encodeVal(uint _val) public view returns(bytes memory) {encodeVal is a function that offchain code calls to help create calldata that uses the cache. It receives a single value and returns the bytes that encode it. This function is a view, so it does not require a transaction and when called externally does not cost any gas.
1        uint _key = val2key[_val];23        // The value isn't in the cache yet, add it4        if (_key == 0)5            return bytes.concat(INTO_CACHE, bytes32(_val));In the EVM all uninitialized storage is assumed to be zeros. So if we look for the key for a value that isn't there, we get a zero. In that case the bytes that encode it are INTO_CACHE (so it will be cached next time), followed by the actual value.
1        // If the key is <0x10, return it as a single byte2        if (_key < 0x10)3            return bytes.concat(bytes1(uint8(_key)));Single bytes are the easiest. We just use bytes.concat to turn a bytes<n> type into a byte array which can be any length. Despite the name, it works fine when provided with just one argument.
1        // Two byte value, encoded as 0x1vvv2        if (_key < 0x1000)3            return bytes.concat(bytes2(uint16(_key) | 0x1000));When we have a key that is less than 163, we can express it in two bytes. We first convert _key, which is a 256 bit value, to a 16 bit value and use logical or to add the number of extra bytes to the first byte. Then we just it into a bytes2 value, which can be converted to bytes.
1        // There is probably a clever way to do the following lines as a loop,2        // but it's a view function so I'm optimizing for programmer time and3        // simplicity.45        if (_key < 16*256**2)6            return bytes.concat(bytes3(uint24(_key) | (0x2 * 16 * 256**2)));7        if (_key < 16*256**3)8            return bytes.concat(bytes4(uint32(_key) | (0x3 * 16 * 256**3)));9             .10             .11             .12        if (_key < 16*256**14)13            return bytes.concat(bytes15(uint120(_key) | (0xE * 16 * 256**14)));14        if (_key < 16*256**15)15            return bytes.concat(bytes16(uint128(_key) | (0xF * 16 * 256**15)));Ratidza zveseThe other values (3 bytes, 4 bytes, etc.) are handled the same way, just with different field sizes.
1        // If we get here, something is wrong.2        revert("Error in encodeVal, should not happen");If we get here it means we got a key that's not less than 16*25615. But cacheWrite limits the keys so we can't even get up to 14*25616 (which would have a first byte of 0xFE, so it would look like DONT_CACHE). But it doesn't cost us much to add a test in case a future programmer introduces a bug.
1    } // encodeVal23}  // CacheTesting the cache
One of the advantages of Foundry is that it lets you write tests in Solidity, which makes it easier to write unit tests. The tests for the Cache class are here. Because the testing code is repetitive, as tests tend to be, this article only explains the interesting parts.
1// SPDX-License-Identifier: UNLICENSED2pragma solidity ^0.8.13;34import "forge-std/Test.sol";567// Need to run `forge test -vv` for the console.8import "forge-std/console.sol";This is just boilerplate that is necessary to use the test package and console.log.
1import "src/Cache.sol";We need to know the contract we are testing.
1contract CacheTest is Test {2    Cache cache;34    function setUp() public {5        cache = new Cache();6    }The setUp function is called before each test. In this case we just create a new cache, so that our tests won't affect each other.
1    function testCaching() public {Tests are functions whose names start with test. This function checks the basic cache functionality, writing values and reading them again.
1        for(uint i=1; i<5000; i++) {2            cache.cacheWrite(i*i);3        }45        for(uint i=1; i<5000; i++) {6            assertEq(cache.cacheRead(i), i*i);This is how you do the actual testing, using assert... functions. In this case, we check that the value we wrote is the one we read. We can discard the result of cache.cacheWrite because we know that cache keys are assigned linearly.
1        }2    }    // testCaching345    // Cache the same value multiple times, ensure that the key stays6    // the same7    function testRepeatCaching() public {8        for(uint i=1; i<100; i++) {9            uint _key1 = cache.cacheWrite(i);10            uint _key2 = cache.cacheWrite(i);11            assertEq(_key1, _key2);12        }Ratidza zveseFirst we write each value twice to the cache and make sure the keys are the same (meaning the second write didn't really happen).
1        for(uint i=1; i<100; i+=3) {2            uint _key = cache.cacheWrite(i);3            assertEq(_key, i);4        }5    }    // testRepeatCachingIn theory there could be a bug that doesn't affect consecutive cache writes. So here we do some writes that aren't consecutive and see the values are still not rewritten.
1    // Read a uint from a memory buffer (to make sure we get back the parameters2    // we sent out)3    function toUint256(bytes memory _bytes, uint256 _start) internal pure4        returns (uint256)Read a 256 bit word from a bytes memory buffer. This utility function lets us verify that we receive the correct results when we run a function call that uses the cache.
1    {2        require(_bytes.length >= _start + 32, "toUint256_outOfBounds");3        uint256 tempUint;45        assembly {6            tempUint := mload(add(add(_bytes, 0x20), _start))7        }Yul does not support data structures beyond uint256, so when you refer to a more sophisticated data structure, such as the memory buffer _bytes, you get the address of that structure. Solidity stores bytes memory values as a 32 byte word that contains the length, followed by the actual bytes, so to get byte number _start we need to calculate _bytes+32+_start.
12        return tempUint;3    }     // toUint25645    // Function signature for fourParams(), courtesy of6    // https://www.4byte.directory/signatures/?bytes4_signature=0x3edc1e6d7    bytes4 constant FOUR_PARAMS = 0x3edc1e6d;89    // Just some constant values to see we're getting the correct values back10    uint256 constant VAL_A = 0xDEAD60A7;11    uint256 constant VAL_B =     0xBEEF;12    uint256 constant VAL_C =     0x600D;13    uint256 constant VAL_D = 0x600D60A7;Ratidza zveseSome constants we need for testing.
1    function testReadParam() public {Call fourParams(), a function that uses readParams, to test we can read parameters correctly.
1        address _cacheAddr = address(cache);2        bool _success;3        bytes memory _callInput;4        bytes memory _callOutput;We can't use the normal ABI mechanism to call a function using the cache, so we need to use the low level <address>.call() mechanism. That mechanism takes a bytes memory as input, and returns that (as well as a Boolean value) as output.
1        // First call, the cache is empty2        _callInput = bytes.concat(3            FOUR_PARAMS,It is useful for the same contract to support both cached functions (for calls directly from transactions) and non-cached functions (for calls from other smart contracts). To do that we need to continue to rely on the Solidity mechanism to call the correct function, instead of putting everything in a fallback function. Doing this makes composability a lot easier. A single byte would be enough to identify the function in most cases, so we are wasting three bytes (16*3=48 gas). However, as I'm writing this those 48 gas cost 0.07 cents, which is a reasonable cost of simpler, less bug prone, code.
1            // First value, add it to the cache2            cache.INTO_CACHE(),3            bytes32(VAL_A),The first value: A flag saying it's a full value that needs to be written to the cache, followed by the 32 bytes of the value. The other three values are similar, except that VAL_B isn't written to the cache and VAL_C is both the third parameter and the fourth one.
1             .2             .3             .4        );5        (_success, _callOutput) = _cacheAddr.call(_callInput);This is where we actually call the Cache contract.
1        assertEq(_success, true);We expect the call to be successful.
1        assertEq(cache.cacheRead(1), VAL_A);2        assertEq(cache.cacheRead(2), VAL_C);We start with an empty cache and then add VAL_A followed by VAL_C. We'd expect the first one to have the key 1, and the second one to have 2.
1        assertEq(toUint256(_callOutput,0), VAL_A);2        assertEq(toUint256(_callOutput,32), VAL_B);3        assertEq(toUint256(_callOutput,64), VAL_C);4        assertEq(toUint256(_callOutput,96), VAL_C);The output is the four parameters. Here we verify it is correct.
1        // Second call, we can use the cache2        _callInput = bytes.concat(3            FOUR_PARAMS,45            // First value in the Cache6            bytes1(0x01),Cache keys below 16 are just one byte.
1            // Second value, don't add it to the cache2            cache.DONT_CACHE(),3            bytes32(VAL_B),45            // Third and fourth values, same value6            bytes1(0x02),7            bytes1(0x02)8        );9        .10        .11        .12    }   // testReadParamRatidza zveseThe tests after the call are identical to those after the first call.
1    function testEncodeVal() public {This function is similar to testReadParam, except that instead of writing the parameters explicitly we use encodeVal().
1        .2        .3        .4        _callInput = bytes.concat(5            FOUR_PARAMS,6            cache.encodeVal(VAL_A),7            cache.encodeVal(VAL_B),8            cache.encodeVal(VAL_C),9            cache.encodeVal(VAL_D)10        );11        .12        .13        .14        assertEq(_callInput.length, 4+1*4);15    }   // testEncodeValRatidza zveseThe only additional test in testEncodeVal() is to verify that the length of _callInput is correct. For the first call it is 4+33*4. For the second, where every value is already in the cache, it is 4+1*4.
1    // Test encodeVal when the key is more than a single byte2    // Maximum three bytes because filling the cache to four bytes takes3    // too long.4    function testEncodeValBig() public {5        // Put a number of values in the cache.6        // To keep things simple, use key n for value n.7        for(uint i=1; i<0x1FFF; i++) {8            cache.cacheWrite(i);9        }Ratidza zveseThe testEncodeVal function above only writes four values into the cache, so the part of the function that deals with multi-byte values is not checked. But that code is complicated and error-prone.
The first part of this function is a loop that writes all the values from 1 to 0x1FFF to the cache in order, so we'll be able to encode those values and know where they are going.
1        .2        .3        .45        _callInput = bytes.concat(6            FOUR_PARAMS,7            cache.encodeVal(0x000F),   // One byte        0x0F8            cache.encodeVal(0x0010),   // Two bytes     0x10109            cache.encodeVal(0x0100),   // Two bytes     0x110010            cache.encodeVal(0x1000)    // Three bytes 0x20100011        );Ratidza zveseTest one byte, two byte, and three byte values. We don't test beyond that because it would take too long to write enough stack entries (at least 0x10000000, approximately a quarter of a billion).
1        .2        .3        .4        .5    }    // testEncodeValBig678    // Test what with an excessively small buffer we get a revert9    function testShortCalldata() public {Ratidza zveseTest what happens in the abnormal case where there aren't enough parameters.
1        .2        .3        .4        (_success, _callOutput) = _cacheAddr.call(_callInput);5        assertEq(_success, false);6    }   // testShortCalldataSince it reverts, the result we should get is false.
1    // Call with cache keys that aren't there2    function testNoCacheKey() public {3        .4        .5        .6        _callInput = bytes.concat(7            FOUR_PARAMS,89            // First value, add it to the cache10            cache.INTO_CACHE(),11            bytes32(VAL_A),1213            // Second value14            bytes1(0x0F),15            bytes2(0x1234),16            bytes11(0xA10102030405060708090A)17        );Ratidza zveseThis function gets four perfectly legitimate parameters, except that the cache is empty so there are no values there to read.
1        .2        .3        .4    // Test what with an excessively long buffer everything works file5    function testLongCalldata() public {6        address _cacheAddr = address(cache);7        bool _success;8        bytes memory _callInput;9        bytes memory _callOutput;1011        // First call, the cache is empty12        _callInput = bytes.concat(13            FOUR_PARAMS,1415            // First value, add it to the cache16            cache.INTO_CACHE(), bytes32(VAL_A),1718            // Second value, add it to the cache19            cache.INTO_CACHE(), bytes32(VAL_B),2021            // Third value, add it to the cache22            cache.INTO_CACHE(), bytes32(VAL_C),2324            // Fourth value, add it to the cache25            cache.INTO_CACHE(), bytes32(VAL_D),2627            // And another value for "good luck"28            bytes4(0x31112233)29        );Ratidza zveseThis function sends five values. We know that the fifth value is ignored because it is not a valid cache entry, which would have caused a revert had it not been included.
1        (_success, _callOutput) = _cacheAddr.call(_callInput);2        assertEq(_success, true);3        .4        .5        .6    }   // testLongCalldata78}        // CacheTest9Ratidza zveseA sample application
Writing tests in Solidity is all very well, but at the end of the day a dapp needs to be able to process requests from outside the chain to be useful. This article demonstrates how to use caching in a dapp with WORM, which stands for "Write Once, Read Many". If a key is not yet written, you can write a value to it. If the key is written already, you get a revert.
The contract
This is the contract. It mostly repeats what we have already done with Cache and CacheTest, so we only cover the parts that are interesting.
1import "./Cache.sol";23contract WORM is Cache {The easiest way to use Cache is to inherit it in our own contract.
1    function writeEntryCached() external {2        uint[] memory params = _readParams(2);3        writeEntry(params[0], params[1]);4    }    // writeEntryCachedThis function is similar to fourParam in CacheTest above. Because we don't follow the ABI specifications, it is best not to declare any parameters into the function.
1    // Make it easier to call us2    // Function signature for writeEntryCached(), courtesy of3    // https://www.4byte.directory/signatures/?bytes4_signature=0xe4e4f2d34    bytes4 constant public WRITE_ENTRY_CACHED = 0xe4e4f2d3;The external code that calls writeEntryCached will need to manually build the calldata, instead of using worm.writeEntryCached, because we do not follow the ABI specifications. Having this constant value just makes it easier to write it.
Note that even though we define WRITE_ENTRY_CACHED as a state variable, to read it externally it is necessary to use the getter function for it, worm.WRITE_ENTRY_CACHED().
1    function readEntry(uint key) public view2        returns (uint _value, address _writtenBy, uint _writtenAtBlock)The read function is a view, so it does not require a transaction and does not cost gas. As a result, there is no benefit to using the cache for the parameter. With view functions it is best to use the standard mechanism that is simpler.
The testing code
This is the testing code for the contract. Again, let us look only at what is interesting.
1    function testWReadWrite() public {2        worm.writeEntry(0xDEAD, 0x60A7);34        vm.expectRevert(bytes("entry already written"));5        worm.writeEntry(0xDEAD, 0xBEEF);This (vm.expectRevert) is how we specify in a Foundry test that the next call should fail, and the reported reason for a failure. This applies when we use the syntax <contract>.<function name>() rather than building the calldata and calling the contract using the low level interface (<contract>.call(), etc.).
1    function testReadWriteCached() public {2        uint cacheGoat = worm.cacheWrite(0x60A7);Here we use the fact that cacheWrite returns the cache key. This is not something we'd expect to use in production, because cacheWrite changes the state, and therefore can only be called during a transaction. Transactions don't have return values, if they have results those results are supposed to be emitted as events. So the cacheWrite return value is only accessible from onchain code, and onchain code does not need parameter caching.
1        (_success,) = address(worm).call(_callInput);This is how we tell Solidity that while <contract address>.call() has two return values, we only care about the first.
1        (_success,) = address(worm).call(_callInput);2        assertEq(_success, false);Since we use the low level <address>.call() function, we can't use vm.expectRevert() and have to look at the boolean success value we get from the call.
1    event EntryWritten(uint indexed key, uint indexed value);23        .4        .5        .67        _callInput = bytes.concat(8            worm.WRITE_ENTRY_CACHED(), worm.encodeVal(a), worm.encodeVal(b));9        vm.expectEmit(true, true, false, false);10        emit EntryWritten(a, b);11        (_success,) = address(worm).call(_callInput);Ratidza zveseThis is the way we verify that code emits an event correctly in Foundry.
The client
One thing you don't get with Solidity tests is JavaScript code you can cut and paste into your own application. To write that code I deployed WORM to Optimism Goerli, Optimism's new testnet. It is at address 0xd34335b1d818cee54e3323d3246bd31d94e6a78a.
You can see JavaScript code for the client here. To use it:
- 
Clone the git repository: 1git clone https://github.com/qbzzt/20220915-all-you-can-cache.git
- 
Install the necessary packages: 1cd javascript2yarn
- 
Copy the configuration file: 1cp .env.example .env
- 
Edit .envfor your configuration:Parameter Value MNEMONIC The mnemonic for an account that has enough ETH to pay for a transaction. You can get free ETH for the Optimism Goerli network here. OPTIMISM_GOERLI_URL URL to Optimism Goerli. The public endpoint, https://goerli.optimism.io, is rate limited but sufficient to what we need here
- 
Run index.js.1node index.jsThis sample application first writes an entry to WORM, displaying the calldata and a link to the transaction on Etherscan. Then it reads back that entry, and displays the key it uses and the values in the entry (value, block number, and author). 
Most of the client is normal Dapp JavaScript. So again we'll only go over the interesting parts.
1.2.3.4const main = async () => {5    const func = await worm.WRITE_ENTRY_CACHED()67    // Need a new key every time8    const key = await worm.encodeVal(Number(new Date()))A given slot can only be written into once, so we use the timestamp to make sure we don't reuse slots.
1const val = await worm.encodeVal("0x600D")23// Write an entry4const calldata = func + key.slice(2) + val.slice(2)Ethers expects the call data to be a hex string, 0x followed by an even number of hexadecimal digits. As key and val both start with 0x, we need to remove those headers.
1const tx = await worm.populateTransaction.writeEntryCached()2tx.data = calldata34sentTx = await wallet.sendTransaction(tx)As with the Solidity testing code, we cannot call a cached function normally. Instead, we need to use a lower level mechanism.
1    .2    .3    .4    // Read the entry just written5    const realKey = '0x' + key.slice(4)  // remove the FF flag6    const entryRead = await worm.readEntry(realKey)7    .8    .9    .Ratidza zveseFor reading entries we can use the normal mechanism. There's no need to use parameter caching with view functions.
Conclusion
The code in this article is a proof of concept, the purpose is to make the idea easy to understand. For a production-ready system you might want to implement some additional functionality:
- 
Handle values that aren't uint256. For example, strings.
- 
Instead of a global cache, maybe have a mapping between users and caches. Different users use different values. 
- 
Values used for addresses are distinct from those used for other purposes. It might make sense to have a separate cache just for addresses. 
- 
Currently, the cache keys are on a "first come, smallest key" algorithm. The first sixteen values can be sent as a single byte. The next 4080 values can be sent as two bytes. The next approximately million values are three bytes, etc. A production system should keep usage counters on cache entries and reorganize them so that the sixteen most common values are one byte, the next 4080 most common values two bytes, etc. However, that is a potentially dangerous operation. Imagine the following sequence of events: - 
Noam Naive calls encodeValto encode the address to which he wants to send tokens. That address is one of the first used on the application, so the encoded value is 0x06. This is aviewfunction, not a transaction, so it's between Noam and the node he uses, and nobody else knows about it
- 
Owen Owner runs the cache reordering operation. Very few people actually use that address, so it is now encoded as 0x201122. A different value, 1018, is assigned 0x06. 
- 
Noam Naive sends his tokens to 0x06. They go to the address 0x0000000000000000000000000de0b6b3a7640000, and since nobody knows the private key for that address, they are just stuck there. Noam is not happy.
 There are ways to solve this problem, and the related problem of transactions that are in the mempool during the cache reorder, but you must be aware of it. 
- 
I demonstrated caching here with Optimism, because I'm an Optimism employee and this is the rollup I know best. But it should work with any rollup that charges a minimal cost for internal processing, so that in comparison writing the transaction data to L1 is the major expense.