Deposited transactions, also known as deposits are transactions which are initiated on L1, and executed on L2. This document outlines a new transaction type for deposits. It also describes how deposits are initiated on L1, along with the authorization and validation conditions on L2.
Vocabulary note: deposited transaction refers specifically to an L2 transaction, while deposit can refer to the transaction at various stages (for instance when it is deposited on L1).
Table of Contents
- The Deposited Transaction Type
- L1 Attributes Deposited Transaction
- Special Accounts on L2
- User-Deposited Transactions
Deposited transactions have the following notable distinctions from existing transaction types:
- They are derived from Layer 1 blocks, and must be included as part of the protocol.
- They do not include signature validation (see User-Deposited Transactions for the rationale).
- They buy their L2 gas on L1 and, as such, the L2 gas is not refundable.
We define a new EIP-2718 compatible transaction type with the prefix 0x7E
, and then a versioned
byte sequence. The first version has 0x00
as the version byte and then as the following fields
(rlp encoded in the order they appear here):
bytes32 sourceHash
: the source-hash, uniquely identifies the origin of the deposit.address from
: The address of the sender account.address to
: The address of the recipient account, or the null (zero-length) address if the deposited transaction is a contract creation.uint256 mint
: The ETH value to mint on L2.uint256 value
: The ETH value to send to the recipient account.bytes data
: The input data.uint64 gasLimit
: The gasLimit for the L2 transaction.
In contrast to EIP-155 transactions, this transaction type does not include signature information,
and makes the from
address explicit.
We select 0x7E
because transaction type identifiers are currently allowed to go up to 0x7F
.
Picking a high identifier minimizes the risk that the identifier will be used be claimed by another
transaction type on the L1 chain in the future. We don't pick 0x7F
itself in case it becomes used
for a variable-length encoding scheme.
We chose to add a version field to the deposit transaction to enable the protocol to upgrade the deposit transaction type without having to take another EIP-2718 transaction type selector.
The sourceHash
of a deposit transaction is computed based on the origin:
- User-deposited:
keccak256(bytes32(uint256(0)), keccak256(l1BlockHash, bytes32(uint256(l1LogIndex))))
. Where thel1BlockHash
, andl1LogIndex
all refer to the inclusion of the deposit log event on L1.l1LogIndex
is the index of the deposit event log in the combined list of log events of the block. - L1 attributes deposited:
keccak256(bytes32(uint256(1)), keccak256(l1BlockHash), bytes32(uint256(seqNumber)))
. Wherel1BlockHash
refers to the L1 block hash of which the info attributes are deposited. AndseqNumber = l2BlockNum - l2EpochStartBlockNum
, wherel2BlockNum
is the L2 block number of the inclusion of the deposit tx in L2, andl2EpochStartBlockNum
is the L2 block number of the first L2 block in the epoch.
Without a sourceHash
in a deposit, two different deposited transactions could have the same exact hash.
The outer keccak256
hashes the actual uniquely identifying information with a domain,
to avoid collisions between different types of sources.
We do not use the sender's nonce to ensure uniqueness because this would require an extra L2 EVM state read from the execution engine during block-derivation.
Although we define only one new transaction type, we can distinguish between two kinds of deposited transactions, based on their positioning in the L2 block:
- The first transaction MUST be a L1 attributes deposited transaction, followed by
- an array of zero-or-more user-deposited transactions submitted to the deposit feed contract on L1. User-deposited transactions are only present in the first block of a L2 epoch.
We only define a single new transaction type in order to minimize modifications to L1 client software, and complexity in general.
As noted above, the deposited transaction type does not include a signature for validation. Rather,
authorization is handled by the L2 chain derivation process, which when correctly
applied will only derive transactions with a from
address attested to by the logs of the L1
deposit contract.
In order to execute a deposited transaction:
First, the balance of the from
account MUST be increased by the amount of mint
.
Then, the execution environment for a deposited transaction is initialized based on the transaction's attributes, in exactly the same manner as it would be for an EIP-155 transaction.
Specifically, a new EVM call frame targeting the to
address is created with values initialized as
follows:
CALLER
andORIGIN
set tofrom
from
is unchanged from the deposit feed contract's logs (though the address may have been aliased by the deposit feed contract).
context.calldata
set todata
context.gas
set togasLimit
context.value
set tosendValue
No gas is bought on L2 and no refund is provided. The gas used for the deposit is subtracted from the gas pool on L2. Gas usage exactly matches other transaction types (including intrinsic gas). If a deposit runs out of gas or has some other failure, the mint will succeed and the nonce of the account will be increased, but no other state transition will occur.
If isSystemTransaction
in the deposit is set to true
, the gas used by the deposit is unmetered.
It must not be subtracted from the gas pool and the usedGas
field of the receipt must be set to 0.
Note for application developers: because CALLER
and ORIGIN
are set to from
, the
semantics of using the tx.origin == msg.sender
check will not work to determine whether
or not a caller is an EOA during a deposit transaction. Instead, the check could only be useful for
identifying the first call in the L2 deposit transaction. However this check does still satisfy
the common case in which developers are using this check to ensure that the CALLER
is unable to
execute code before and after the call.
Despite the lack of signature validation, we still increment the nonce of the from
account when a
deposit transaction is executed. In the context of a deposit-only roll up, this is not necessary
for transaction ordering or replay prevention, however it maintains consistency with the use of
nonces during contract creation. It may also simplify integration with downstream
tooling (such as wallets and block explorers).
An L1 attributes deposited transaction is a deposit transaction sent to the L1 attributes predeployed contract.
This transaction MUST have the following values:
from
is0xdeaddeaddeaddeaddeaddeaddeaddeaddead0001
(the address of the L1 Attributes depositor account)to
is0x4200000000000000000000000000000000000015
(the address of the L1 attributes predeployed contract).mint
is0
value
is0
gasLimit
is set to 150,000,000.isSystemTransaction
is set totrue
.data
is an ABI encoded call to the L1 attributes predeployed contract'ssetL1BlockValues()
function with correct values associated with the corresponding L1 block (cf. reference implementation).
No gas is paid for L1 attributes deposited transactions.
The L1 attributes deposit transaction involves two special purpose accounts:
- The L1 attributes depositor account
- The L1 attributes predeployed contract
The depositor account is an EOA with no known private key. It has the address
0xdeaddeaddeaddeaddeaddeaddeaddeaddead0001
. Its value is returned by the CALLER
and ORIGIN
opcodes during execution of the L1 attributes deposited transaction.
A predeployed contract on L2 at address 0x4200000000000000000000000000000000000015
, which holds
certain block variables from the corresponding L1 block in storage, so that they may be accessed
during the execution of the subsequent deposited transactions.
The predeploy stores the following values:
- L1 block attributes:
number
(uint64
)timestamp
(uint64
)basefee
(uint256
)hash
(bytes32
)
sequenceNumber
(uint64
): This equals the L2 block number relative to the start of the epoch, i.e. the L2 block distance to the L2 block height that the L1 attributes last changed, and reset to 0 at the start of a new epoch.- System configurables tied to the L1 block, see System configuration specification:
batcherHash
(bytes32
): A versioned commitment to the batch-submitter(s) currently operating.l1FeeOverhead
(uint256
): The L1 fee overhead to apply to L1 cost computation of transactions in this L2 block.l1FeeScalar
(uint256
): The L1 fee scalar to apply to L1 cost computation of transactions in this L2 block.
The contract implements an authorization scheme, such that it only accepts state-changing calls from the depositor account.
The contract has the following solidity interface, and can be interacted with according to the contract ABI specification.
A reference implementation of the L1 Attributes predeploy contract can be found in L1Block.sol.
After running yarn build
in the packages/contracts
directory, the bytecode to add to the genesis
file will be located in the deployedBytecode
field of the build artifacts file at
/packages/contracts/artifacts/contracts/L2/L1Block.sol/L1Block.json
.
User-deposited transactions are deposited transactions
generated by the L2 Chain Derivation process. The content of each user-deposited
transaction are determined by the corresponding TransactionDeposited
event emitted by the
deposit contract on L1.
from
is unchanged from the emitted value (though it may have been transformed to an alias in the deposit feed contract).to
is any 20-byte address (including the zero address)- In case of a contract creation (cf.
isCreation
), this address is always zero.
- In case of a contract creation (cf.
mint
is set to the emitted value.value
is set to the emitted value.gaslimit
is unchanged from the emitted value.isCreation
is set totrue
if the transaction is a contract creation,false
otherwise.data
is unchanged from the emitted value. Depending on the value ofisCreation
it is handled as either calldata or contract initialization code.isSystemTransaction
is set by the rollup node for certain transactions that have unmetered execution. It isfalse
for user deposited transactions
The deposit contract is deployed to L1. Deposited transactions are derived from the values in
the TransactionDeposited
event(s) emitted by the deposit contract.
The deposit contract is responsible for maintaining the guaranteed gas market, charging deposits for gas to be used on L2, and ensuring that the total amount of guaranteed gas in a single L1 block does not exceed the L2 block gas limit.
The deposit contract handles two special cases:
- A contract creation deposit, which is indicated by setting the
isCreation
flag totrue
. In the event that theto
address is non-zero, the contract will revert. - A call from a contract account, in which case the
from
value is transformed to its L2 alias.
If the caller is a contract, the address will be transformed by adding
0x1111000000000000000000000000000000001111
to it. The math is unchecked
and done on a
Solidity uint160
so the value will overflow. This prevents attacks in which a
contract on L1 has the same address as a contract on L2 but doesn't have the same code. We can safely ignore this
for EOAs because they're guaranteed to have the same "code" (i.e. no code at all). This also makes
it possible for users to interact with contracts on L2 even when the Sequencer is down.
A reference implementation of the deposit contract can be found in OptimismPortal.sol.