Uniswap V4 CurrencyDelta Library 是一种优化的闪电记账工具,通过 transient storage 记录代币余额变化,从而节省 gas 消耗。文章详细介绍了如何计算存储 slot 地址、获取余额变化和应用余额变更的 Solidity 函数,实现了高效的代币管理。
CurrencyDelta Library 实现了闪电记账功能,通过记录某个地址的代币余额变化,来实现对代币余额的更新。由于所有操作在 transient storage
中进行,因此可以极大节省 gas。
EIP-1153 介绍了如何在合约中使用 transient storage,通过引入 tload
和 tstore
操作,可以在合约中实现对临时存储的读写操作。
根据 target
和 currency
计算出存储的 slot 地址,用于存储余额变化值。
/// @notice calculates which storage slot a delta should be stored in for a given account and currency
function _computeSlot(address target, Currency currency) internal pure returns (bytes32 hashSlot) {
assembly ("memory-safe") {
mstore(0, and(target, 0xffffffffffffffffffffffffffffffffffffffff))
mstore(32, and(currency, 0xffffffffffffffffffffffffffffffffffffffff))
hashSlot := keccak256(0, 64)
}
}
因为 target
和 currency
都是 address
类型,因此只需要保留低 20 字节的数据。
在 EVM 的内存布局中,前 64 字节是用于 hash 计算的临时区域,因此可以直接使用 mstore
指令将 target
和 currency
分别写入地址 0 和地址 32 的位置(各 32 字节)。
最后,通过 keccak256(0, 64)
对前 64 字节的数据进行哈希,得到 slot 地址。
获取某个地址和代币的记账余额。
function getDelta(Currency currency, address target) internal view returns (int256 delta) {
bytes32 hashSlot = _computeSlot(target, currency);
assembly ("memory-safe") {
delta := tload(hashSlot)
}
}
通过 _computeSlot
计算出 slot 地址,然后通过 tload
读取 slot 中的值。
应用新的余额变化值,更新指定地址和代币的余额。
输入参数:
currency
代币类型target
地址delta
余额变化值返回:
previous
更新前的余额next
更新后的余额/// @notice applies a new currency delta for a given account and currency
/// @return previous The prior value
/// @return next The modified result
function applyDelta(Currency currency, address target, int128 delta)
internal
returns (int256 previous, int256 next)
{
bytes32 hashSlot = _computeSlot(target, currency);
assembly ("memory-safe") {
previous := tload(hashSlot)
}
next = previous + delta;
assembly ("memory-safe") {
tstore(hashSlot, next)
}
}
通过 _computeSlot
计算出 slot 地址,然后通过 tload
读取 slot 中的值,得到更新前的余额。
接着,计算出新的余额,并通过 tstore
保存。
- 原文链接: github.com/adshao/public...
- 登链社区 AI 助手,为大家转译优秀英文文章,如有翻译不通的地方,还请包涵~
如果觉得我的文章对您有用,请随意打赏。你的支持将鼓励我继续创作!