2020-05-12 00:06:31 +00:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2020-04-14 20:44:39 +00:00
|
|
|
pragma solidity ^0.6.0;
|
|
|
|
|
|
|
|
// This interface is designed to be compatible with the Vyper version.
|
|
|
|
interface IDepositContract {
|
|
|
|
event DepositEvent(
|
|
|
|
bytes pubkey,
|
|
|
|
bytes withdrawal_credentials,
|
|
|
|
bytes amount,
|
|
|
|
bytes signature,
|
|
|
|
bytes index
|
|
|
|
);
|
|
|
|
|
|
|
|
function deposit(
|
|
|
|
bytes calldata pubkey,
|
|
|
|
bytes calldata withdrawal_credentials,
|
|
|
|
bytes calldata signature,
|
|
|
|
bytes32 deposit_data_root
|
|
|
|
) external payable;
|
2020-05-11 23:57:17 +00:00
|
|
|
|
|
|
|
function get_deposit_root() external view returns (bytes32);
|
|
|
|
|
|
|
|
function get_deposit_count() external view returns (bytes memory);
|
2020-04-14 20:44:39 +00:00
|
|
|
}
|
2020-04-14 20:57:58 +00:00
|
|
|
|
2020-05-12 00:06:31 +00:00
|
|
|
// This is a rewrite of the Vyper Eth2.0 deposit contract in Solidity.
|
|
|
|
// It tries to stay as close as possible to the original source code.
|
2020-04-14 20:57:58 +00:00
|
|
|
contract DepositContract is IDepositContract {
|
|
|
|
uint constant GWEI = 1e9;
|
|
|
|
|
|
|
|
uint constant MIN_DEPOSIT_AMOUNT = 1000000000; // Gwei
|
|
|
|
uint constant DEPOSIT_CONTRACT_TREE_DEPTH = 32;
|
2020-05-12 18:01:27 +00:00
|
|
|
uint constant MAX_DEPOSIT_COUNT = 2**DEPOSIT_CONTRACT_TREE_DEPTH - 1;
|
2020-04-14 20:57:58 +00:00
|
|
|
uint constant PUBKEY_LENGTH = 48; // bytes
|
|
|
|
uint constant WITHDRAWAL_CREDENTIALS_LENGTH = 32; // bytes
|
|
|
|
uint constant SIGNATURE_LENGTH = 96; // bytes
|
|
|
|
uint constant AMOUNT_LENGTH = 8; // bytes
|
|
|
|
|
|
|
|
bytes32[DEPOSIT_CONTRACT_TREE_DEPTH] branch;
|
|
|
|
uint256 deposit_count;
|
|
|
|
|
2020-04-14 21:46:45 +00:00
|
|
|
// TODO: use immutable for this
|
|
|
|
bytes32[DEPOSIT_CONTRACT_TREE_DEPTH] zero_hashes;
|
2020-04-14 20:57:58 +00:00
|
|
|
|
2020-04-14 21:46:45 +00:00
|
|
|
// Compute hashes in empty sparse Merkle tree
|
|
|
|
constructor() public {
|
|
|
|
for (uint height = 0; height < DEPOSIT_CONTRACT_TREE_DEPTH - 1; height++)
|
|
|
|
zero_hashes[height + 1] = sha256(abi.encodePacked(zero_hashes[height], zero_hashes[height]));
|
|
|
|
}
|
2020-04-14 20:57:58 +00:00
|
|
|
|
2020-05-11 23:57:17 +00:00
|
|
|
function get_deposit_root() override external view returns (bytes32) {
|
2020-04-14 21:46:45 +00:00
|
|
|
bytes32 node;
|
|
|
|
uint size = deposit_count;
|
|
|
|
for (uint height = 0; height < DEPOSIT_CONTRACT_TREE_DEPTH; height++) {
|
|
|
|
if ((size & 1) == 1)
|
|
|
|
node = sha256(abi.encodePacked(branch[height], node));
|
|
|
|
else
|
|
|
|
node = sha256(abi.encodePacked(node, zero_hashes[height]));
|
|
|
|
size /= 2;
|
|
|
|
}
|
|
|
|
return sha256(abi.encodePacked(
|
|
|
|
node,
|
|
|
|
to_little_endian_64(uint64(deposit_count)),
|
2020-05-12 18:04:18 +00:00
|
|
|
bytes24(0)
|
2020-04-14 21:46:45 +00:00
|
|
|
));
|
|
|
|
}
|
|
|
|
|
2020-05-11 23:57:17 +00:00
|
|
|
function get_deposit_count() override external view returns (bytes memory) {
|
2020-04-14 21:46:45 +00:00
|
|
|
return to_little_endian_64(uint64(deposit_count));
|
|
|
|
}
|
2020-04-14 20:57:58 +00:00
|
|
|
|
|
|
|
function deposit(
|
|
|
|
bytes calldata pubkey,
|
|
|
|
bytes calldata withdrawal_credentials,
|
|
|
|
bytes calldata signature,
|
|
|
|
bytes32 deposit_data_root
|
|
|
|
) override external payable {
|
|
|
|
// Avoid overflowing the Merkle tree (and prevent edge case in computing `self.branch`)
|
|
|
|
require(deposit_count < MAX_DEPOSIT_COUNT);
|
|
|
|
|
|
|
|
// Check deposit amount
|
|
|
|
uint deposit_amount = msg.value / GWEI;
|
|
|
|
require(deposit_amount >= MIN_DEPOSIT_AMOUNT);
|
2020-05-13 13:41:37 +00:00
|
|
|
require(deposit_amount < 2**64);
|
2020-04-14 20:57:58 +00:00
|
|
|
|
|
|
|
// Length checks for safety
|
|
|
|
require(pubkey.length == PUBKEY_LENGTH);
|
|
|
|
require(withdrawal_credentials.length == WITHDRAWAL_CREDENTIALS_LENGTH);
|
|
|
|
require(signature.length == SIGNATURE_LENGTH);
|
|
|
|
|
|
|
|
// FIXME: these are not the Vyper code, but should verify they are not needed
|
2020-04-16 17:12:58 +00:00
|
|
|
// assert(deposit_count <= 2**64-1);
|
2020-04-14 20:57:58 +00:00
|
|
|
|
|
|
|
// Emit `DepositEvent` log
|
|
|
|
bytes memory amount = to_little_endian_64(uint64(deposit_amount));
|
|
|
|
emit DepositEvent(
|
|
|
|
pubkey,
|
|
|
|
withdrawal_credentials,
|
|
|
|
amount,
|
|
|
|
signature,
|
|
|
|
to_little_endian_64(uint64(deposit_count))
|
|
|
|
);
|
|
|
|
|
|
|
|
// Compute deposit data root (`DepositData` hash tree root)
|
2020-05-12 18:04:18 +00:00
|
|
|
bytes32 pubkey_root = sha256(abi.encodePacked(pubkey, bytes16(0)));
|
2020-04-14 20:57:58 +00:00
|
|
|
bytes32 signature_root = sha256(abi.encodePacked(
|
|
|
|
sha256(abi.encodePacked(bytes(signature[:64]))),
|
2020-05-12 18:04:18 +00:00
|
|
|
sha256(abi.encodePacked(bytes(signature[64:]), bytes32(0)))
|
2020-04-14 20:57:58 +00:00
|
|
|
));
|
|
|
|
bytes32 node = sha256(abi.encodePacked(
|
|
|
|
sha256(abi.encodePacked(pubkey_root, withdrawal_credentials)),
|
2020-05-12 18:04:18 +00:00
|
|
|
sha256(abi.encodePacked(amount, bytes24(0), signature_root))
|
2020-04-14 20:57:58 +00:00
|
|
|
));
|
|
|
|
// Verify computed and expected deposit data roots match
|
|
|
|
require(node == deposit_data_root);
|
|
|
|
|
|
|
|
// Add deposit data root to Merkle tree (update a single `branch` node)
|
|
|
|
deposit_count += 1;
|
|
|
|
uint size = deposit_count;
|
|
|
|
for (uint height = 0; height < DEPOSIT_CONTRACT_TREE_DEPTH; height++) {
|
|
|
|
if ((size & 1) == 1) {
|
|
|
|
branch[height] = node;
|
2020-05-13 13:46:58 +00:00
|
|
|
return;
|
2020-04-14 20:57:58 +00:00
|
|
|
}
|
|
|
|
node = sha256(abi.encodePacked(branch[height], node));
|
|
|
|
size /= 2;
|
|
|
|
}
|
2020-05-13 13:46:58 +00:00
|
|
|
// As the loop should always end prematurely with the `return` statement,
|
|
|
|
// this code should be unreachable. We assert `false` just to be safe.
|
|
|
|
assert(false);
|
2020-04-14 20:57:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function to_little_endian_64(uint64 value) internal pure returns (bytes memory ret) {
|
|
|
|
// Unrolled the loop here.
|
|
|
|
ret = new bytes(8);
|
2020-05-12 17:59:15 +00:00
|
|
|
ret[0] = bytes1(uint8(value));
|
|
|
|
ret[1] = bytes1(uint8(value >> 8));
|
|
|
|
ret[2] = bytes1(uint8(value >> 16));
|
|
|
|
ret[3] = bytes1(uint8(value >> 24));
|
|
|
|
ret[4] = bytes1(uint8(value >> 32));
|
|
|
|
ret[5] = bytes1(uint8(value >> 40));
|
|
|
|
ret[6] = bytes1(uint8(value >> 48));
|
|
|
|
ret[7] = bytes1(uint8(value >> 56));
|
2020-04-14 20:57:58 +00:00
|
|
|
}
|
|
|
|
}
|