-
Notifications
You must be signed in to change notification settings - Fork 54
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Charge l1 data availability fee #86
Changes from all commits
e785dc8
18d44e2
c839e83
3b6da8a
3bdab64
2079ffb
2e9f4bc
bc6e602
bb5900c
8465a72
ff719db
c80a8fd
f6cca92
31e61ea
c302a7f
13a4266
dbe18ba
125f004
178c6f8
be9281c
6e2be8f
6807c01
e813a07
7f22a73
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Large diffs are not rendered by default.
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -2,6 +2,7 @@ | |
pragma solidity 0.8.19; | ||
|
||
import {IPriceRegistry} from "./interfaces/IPriceRegistry.sol"; | ||
import {TypeAndVersionInterface} from "../interfaces/TypeAndVersionInterface.sol"; | ||
|
||
import {OwnerIsCreator} from "./../shared/access/OwnerIsCreator.sol"; | ||
import {Internal} from "./libraries/Internal.sol"; | ||
|
@@ -11,9 +12,9 @@ import {EnumerableSet} from "../vendor/openzeppelin-solidity/v4.8.0/contracts/ut | |
|
||
/// @notice The PriceRegistry contract responsibility is to store the current gas price in USD for a given destination chain, | ||
/// and the price of a token in USD allowing the owner or priceUpdater to update this value. | ||
contract PriceRegistry is IPriceRegistry, OwnerIsCreator { | ||
contract PriceRegistry is IPriceRegistry, OwnerIsCreator, TypeAndVersionInterface { | ||
using EnumerableSet for EnumerableSet.AddressSet; | ||
using USDPriceWith18Decimals for uint192; | ||
using USDPriceWith18Decimals for uint224; | ||
|
||
error TokenNotSupported(address token); | ||
error ChainNotSupported(uint64 chain); | ||
|
@@ -29,20 +30,26 @@ contract PriceRegistry is IPriceRegistry, OwnerIsCreator { | |
event UsdPerUnitGasUpdated(uint64 indexed destChain, uint256 value, uint256 timestamp); | ||
event UsdPerTokenUpdated(address indexed token, uint256 value, uint256 timestamp); | ||
|
||
/// @dev The price, in USD with 18 decimals, of 1 unit of gas for a given destination chain. | ||
// solhint-disable-next-line chainlink-solidity/all-caps-constant-storage-variables | ||
string public constant override typeAndVersion = "PriceRegistry 1.2.0"; | ||
|
||
/// @dev The gas price per unit of gas for a given destination chain, in USD with 18 decimals. | ||
/// Multiple gas prices can be encoded into the same value. Each price takes {Internal.GAS_PRICE_BITS} bits. | ||
/// For example, if Optimism is the destination chain, gas price can include L1 base fee and L2 gas price. | ||
/// Logic to parse the price components is chain-specific, and should live in OnRamp. | ||
/// @dev Price of 1e18 is 1 USD. Examples: | ||
/// Very Expensive: 1 unit of gas costs 1 USD -> 1e18 | ||
/// Expensive: 1 unit of gas costs 0.1 USD -> 1e17 | ||
/// Cheap: 1 unit of gas costs 0.000001 USD -> 1e12 | ||
mapping(uint64 destChainSelector => Internal.TimestampedUint192Value price) | ||
mapping(uint64 destChainSelector => Internal.TimestampedPackedUint224 price) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. comments need updating |
||
private s_usdPerUnitGasByDestChainSelector; | ||
|
||
/// @dev The price, in USD with 18 decimals, per 1e18 of the smallest token denomination. | ||
/// @dev Price of 1e18 represents 1 USD per 1e18 token amount. | ||
/// 1 USDC = 1.00 USD per full token, each full token is 1e6 units -> 1 * 1e18 * 1e18 / 1e6 = 1e30 | ||
/// 1 ETH = 2,000 USD per full token, each full token is 1e18 units -> 2000 * 1e18 * 1e18 / 1e18 = 2_000e18 | ||
/// 1 LINK = 5.00 USD per full token, each full token is 1e18 units -> 5 * 1e18 * 1e18 / 1e18 = 5e18 | ||
mapping(address token => Internal.TimestampedUint192Value price) private s_usdPerToken; | ||
mapping(address token => Internal.TimestampedPackedUint224 price) private s_usdPerToken; | ||
|
||
// Price updaters are allowed to update the prices. | ||
EnumerableSet.AddressSet private s_priceUpdaters; | ||
|
@@ -63,21 +70,21 @@ contract PriceRegistry is IPriceRegistry, OwnerIsCreator { | |
// ================================================================ | ||
|
||
// @inheritdoc IPriceRegistry | ||
function getTokenPrice(address token) public view override returns (Internal.TimestampedUint192Value memory) { | ||
function getTokenPrice(address token) public view override returns (Internal.TimestampedPackedUint224 memory) { | ||
return s_usdPerToken[token]; | ||
} | ||
|
||
// @inheritdoc IPriceRegistry | ||
function getValidatedTokenPrice(address token) external view override returns (uint192) { | ||
function getValidatedTokenPrice(address token) external view override returns (uint224) { | ||
return _getValidatedTokenPrice(token); | ||
} | ||
|
||
// @inheritdoc IPriceRegistry | ||
function getTokenPrices( | ||
address[] calldata tokens | ||
) external view override returns (Internal.TimestampedUint192Value[] memory) { | ||
) external view override returns (Internal.TimestampedPackedUint224[] memory) { | ||
uint256 length = tokens.length; | ||
Internal.TimestampedUint192Value[] memory tokenPrices = new Internal.TimestampedUint192Value[](length); | ||
Internal.TimestampedPackedUint224[] memory tokenPrices = new Internal.TimestampedPackedUint224[](length); | ||
for (uint256 i = 0; i < length; ++i) { | ||
tokenPrices[i] = getTokenPrice(tokens[i]); | ||
} | ||
|
@@ -93,15 +100,15 @@ contract PriceRegistry is IPriceRegistry, OwnerIsCreator { | |
// @inheritdoc IPriceRegistry | ||
function getDestinationChainGasPrice( | ||
uint64 destChainSelector | ||
) external view override returns (Internal.TimestampedUint192Value memory) { | ||
) external view override returns (Internal.TimestampedPackedUint224 memory) { | ||
return s_usdPerUnitGasByDestChainSelector[destChainSelector]; | ||
} | ||
|
||
function getTokenAndGasPrices( | ||
address token, | ||
uint64 destChainSelector | ||
) external view override returns (uint192 tokenPrice, uint192 gasPriceValue) { | ||
Internal.TimestampedUint192Value memory gasPrice = s_usdPerUnitGasByDestChainSelector[destChainSelector]; | ||
) external view override returns (uint224 tokenPrice, uint224 gasPriceValue) { | ||
Internal.TimestampedPackedUint224 memory gasPrice = s_usdPerUnitGasByDestChainSelector[destChainSelector]; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. thoughts on an internal _getValidatedExecutionPrices analogous to the _getValidatedTokenPrice? Yields some nice symmetry validatedXX means timestamp checked for both gas and tokens There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. will include in a follow up PR that includes |
||
// We do allow a gas price of 0, but no stale or unset gas prices | ||
if (gasPrice.timestamp == 0) revert ChainNotSupported(destChainSelector); | ||
uint256 timePassed = block.timestamp - gasPrice.timestamp; | ||
|
@@ -131,8 +138,8 @@ contract PriceRegistry is IPriceRegistry, OwnerIsCreator { | |
/// not supported or the price is stale. | ||
/// @param token The address of the token to get the price for | ||
/// @return the token price | ||
function _getValidatedTokenPrice(address token) internal view returns (uint192) { | ||
Internal.TimestampedUint192Value memory tokenPrice = s_usdPerToken[token]; | ||
function _getValidatedTokenPrice(address token) internal view returns (uint224) { | ||
Internal.TimestampedPackedUint224 memory tokenPrice = s_usdPerToken[token]; | ||
if (tokenPrice.timestamp == 0 || tokenPrice.value == 0) revert TokenNotSupported(token); | ||
uint256 timePassed = block.timestamp - tokenPrice.timestamp; | ||
if (timePassed > i_stalenessThreshold) revert StaleTokenPrice(token, i_stalenessThreshold, timePassed); | ||
|
@@ -187,17 +194,17 @@ contract PriceRegistry is IPriceRegistry, OwnerIsCreator { | |
|
||
for (uint256 i = 0; i < priceUpdatesLength; ++i) { | ||
Internal.TokenPriceUpdate memory update = priceUpdates.tokenPriceUpdates[i]; | ||
s_usdPerToken[update.sourceToken] = Internal.TimestampedUint192Value({ | ||
s_usdPerToken[update.sourceToken] = Internal.TimestampedPackedUint224({ | ||
value: update.usdPerToken, | ||
timestamp: uint64(block.timestamp) | ||
timestamp: uint32(block.timestamp) | ||
}); | ||
emit UsdPerTokenUpdated(update.sourceToken, update.usdPerToken, block.timestamp); | ||
} | ||
|
||
if (priceUpdates.destChainSelector != 0) { | ||
s_usdPerUnitGasByDestChainSelector[priceUpdates.destChainSelector] = Internal.TimestampedUint192Value({ | ||
s_usdPerUnitGasByDestChainSelector[priceUpdates.destChainSelector] = Internal.TimestampedPackedUint224({ | ||
value: priceUpdates.usdPerUnitGas, | ||
timestamp: uint64(block.timestamp) | ||
timestamp: uint32(block.timestamp) | ||
}); | ||
emit UsdPerUnitGasUpdated(priceUpdates.destChainSelector, priceUpdates.usdPerUnitGas, block.timestamp); | ||
} | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -11,24 +11,29 @@ interface IPriceRegistry { | |
/// @notice Get the `tokenPrice` for a given token. | ||
/// @param token The token to get the price for. | ||
/// @return tokenPrice The tokenPrice for the given token. | ||
function getTokenPrice(address token) external view returns (Internal.TimestampedUint192Value memory); | ||
function getTokenPrice(address token) external view returns (Internal.TimestampedPackedUint224 memory); | ||
|
||
/// @notice Get the `tokenPrice` for a given token, checks if the price is valid. | ||
/// @param token The token to get the price for. | ||
/// @return tokenPrice The tokenPrice for the given token if it exists and is valid. | ||
function getValidatedTokenPrice(address token) external view returns (uint192); | ||
function getValidatedTokenPrice(address token) external view returns (uint224); | ||
|
||
/// @notice Get the `tokenPrice` for an array of tokens. | ||
/// @param tokens The tokens to get prices for. | ||
/// @return tokenPrices The tokenPrices for the given tokens. | ||
function getTokenPrices(address[] calldata tokens) external view returns (Internal.TimestampedUint192Value[] memory); | ||
|
||
/// @notice Get the `gasPrice` for a given destination chain ID. | ||
function getTokenPrices(address[] calldata tokens) external view returns (Internal.TimestampedPackedUint224[] memory); | ||
|
||
/// @notice Get an encoded `gasPrice` for a given destination chain ID. | ||
/// The 224-bit result encodes necessary gas price components. | ||
/// On L1 chains like Ethereum or Avax, the only component is the gas price. | ||
/// On Optimistic Rollups, there are two components - the L2 gas price, and L1 base fee for data availability. | ||
/// On future chains, there could be more or differing price components. | ||
/// PriceRegistry does not contain chain-specific logic to parse destination chain price components. | ||
/// @param destChainSelector The destination chain to get the price for. | ||
/// @return gasPrice The gasPrice for the given destination chain ID. | ||
/// @return gasPrice The encoded gasPrice for the given destination chain ID. | ||
function getDestinationChainGasPrice( | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. probably update the comment here? it now returns a multi-dimensional gas price |
||
uint64 destChainSelector | ||
) external view returns (Internal.TimestampedUint192Value memory); | ||
) external view returns (Internal.TimestampedPackedUint224 memory); | ||
|
||
/// @notice Gets the fee token price and the gas price, both denominated in dollars. | ||
/// @param token The source token to get the price for. | ||
|
@@ -38,7 +43,7 @@ interface IPriceRegistry { | |
function getTokenAndGasPrices( | ||
address token, | ||
uint64 destChainSelector | ||
) external view returns (uint192 tokenPrice, uint192 gasPrice); | ||
) external view returns (uint224 tokenPrice, uint224 gasPrice); | ||
|
||
/// @notice Convert a given token amount to target token amount. | ||
/// @param fromToken The given token address. | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -9,19 +9,24 @@ library Internal { | |
struct PriceUpdates { | ||
TokenPriceUpdate[] tokenPriceUpdates; | ||
uint64 destChainSelector; // ──╮ Destination chain selector | ||
uint192 usdPerUnitGas; // ─────╯ 1e18 USD per smallest unit (e.g. wei) of destination chain gas | ||
uint224 usdPerUnitGas; // ─────╯ 1e18 USD per smallest unit (e.g. wei) of destination chain gas | ||
} | ||
|
||
struct TokenPriceUpdate { | ||
address sourceToken; // Source token | ||
uint192 usdPerToken; // 1e18 USD per smallest unit of token | ||
uint224 usdPerToken; // 1e18 USD per smallest unit of token | ||
} | ||
|
||
struct TimestampedUint192Value { | ||
uint192 value; // ───────╮ The price, in 1e18 USD. | ||
uint64 timestamp; // ────╯ Timestamp of the most recent price update. | ||
struct TimestampedPackedUint224 { | ||
uint224 value; // ───────╮ Value in uint224, packed. | ||
uint32 timestamp; // ────╯ Timestamp of the most recent price update. | ||
} | ||
|
||
/// @dev Gas price is stored in 112-bit unsigned int. uint224 can pack 2 prices. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can we abstract this from price reg clients all together? Since the price reg API is breaking anyways, something like There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. spoke offline - I'm cool with opaque gas price and ramps interpreting |
||
/// When packing L1 and L2 gas prices, L1 gas price is left-shifted to the higher-order bits. | ||
/// Using uint8, which is strictly lower than 1st shift operand, to avoid shift operand type warning. | ||
uint8 public constant GAS_PRICE_BITS = 112; | ||
|
||
struct PoolUpdate { | ||
address token; // The IERC20 token address | ||
address pool; // The token pool address | ||
|
@@ -52,6 +57,17 @@ library Internal { | |
bytes32 messageId; // a hash of the message data | ||
} | ||
|
||
/// @dev EVM2EVMMessage struct has 13 fields, including 3 variable arrays. | ||
/// Each variable array takes 1 more slot to store its length. | ||
/// When abi encoded, excluding array contents, | ||
/// EVM2EVMMessage takes up a fixed number of 16 lots, 32 bytes each. | ||
/// For structs that contain arrays, 1 more slot is added to the front, reaching a total of 17. | ||
uint256 public constant MESSAGE_FIXED_BYTES = 32 * 17; | ||
|
||
/// @dev Each token transfer adds 1 EVMTokenAmount and 1 bytes. | ||
/// When abiEncoded, each EVMTokenAmount takes 2 slots, each bytes takes 2 slots. | ||
uint256 public constant MESSAGE_BYTES_PER_TOKEN = 32 * 4; | ||
|
||
function _toAny2EVMMessage( | ||
EVM2EVMMessage memory original, | ||
Client.EVMTokenAmount[] memory destTokenAmounts | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
great opportunity to add a type and version to the price reg