#![cfg_attr(not(feature = "std"), no_std)]
use account::SYSTEM_ACCOUNT_SIZE;
use fp_evm::PrecompileHandle;
use frame_support::{
dispatch::{GetDispatchInfo, PostDispatchInfo},
sp_runtime::traits::{Bounded, CheckedSub, Dispatchable, StaticLookup},
storage::types::{StorageDoubleMap, StorageMap, ValueQuery},
traits::StorageInstance,
Blake2_128Concat,
};
use pallet_balances::pallet::{
Instance1, Instance10, Instance11, Instance12, Instance13, Instance14, Instance15, Instance16,
Instance2, Instance3, Instance4, Instance5, Instance6, Instance7, Instance8, Instance9,
};
use pallet_evm::AddressMapping;
use precompile_utils::prelude::*;
use sp_core::{H160, H256, U256};
use sp_std::{
convert::{TryFrom, TryInto},
marker::PhantomData,
};
mod eip2612;
use eip2612::Eip2612;
#[cfg(test)]
mod mock;
#[cfg(test)]
mod tests;
pub const SELECTOR_LOG_TRANSFER: [u8; 32] = keccak256!("Transfer(address,address,uint256)");
pub const SELECTOR_LOG_APPROVAL: [u8; 32] = keccak256!("Approval(address,address,uint256)");
pub const SELECTOR_LOG_DEPOSIT: [u8; 32] = keccak256!("Deposit(address,uint256)");
pub const SELECTOR_LOG_WITHDRAWAL: [u8; 32] = keccak256!("Withdrawal(address,uint256)");
pub trait InstanceToPrefix {
type ApprovesPrefix: StorageInstance;
type NoncesPrefix: StorageInstance;
}
macro_rules! impl_prefix {
($instance:ident, $name:literal) => {
paste::paste! {
mod [<_impl_prefix_ $instance:snake>] {
use super::*;
pub struct Approves;
impl StorageInstance for Approves {
const STORAGE_PREFIX: &'static str = "Approves";
fn pallet_prefix() -> &'static str {
$name
}
}
pub struct Nonces;
impl StorageInstance for Nonces {
const STORAGE_PREFIX: &'static str = "Nonces";
fn pallet_prefix() -> &'static str {
$name
}
}
impl InstanceToPrefix for $instance {
type ApprovesPrefix = Approves;
type NoncesPrefix = Nonces;
}
}
}
};
}
type Instance0 = ();
impl_prefix!(Instance0, "Erc20Instance0Balances");
impl_prefix!(Instance1, "Erc20Instance1Balances");
impl_prefix!(Instance2, "Erc20Instance2Balances");
impl_prefix!(Instance3, "Erc20Instance3Balances");
impl_prefix!(Instance4, "Erc20Instance4Balances");
impl_prefix!(Instance5, "Erc20Instance5Balances");
impl_prefix!(Instance6, "Erc20Instance6Balances");
impl_prefix!(Instance7, "Erc20Instance7Balances");
impl_prefix!(Instance8, "Erc20Instance8Balances");
impl_prefix!(Instance9, "Erc20Instance9Balances");
impl_prefix!(Instance10, "Erc20Instance10Balances");
impl_prefix!(Instance11, "Erc20Instance11Balances");
impl_prefix!(Instance12, "Erc20Instance12Balances");
impl_prefix!(Instance13, "Erc20Instance13Balances");
impl_prefix!(Instance14, "Erc20Instance14Balances");
impl_prefix!(Instance15, "Erc20Instance15Balances");
impl_prefix!(Instance16, "Erc20Instance16Balances");
pub type BalanceOf<Runtime, Instance = ()> =
<Runtime as pallet_balances::Config<Instance>>::Balance;
pub type ApprovesStorage<Runtime, Instance> = StorageDoubleMap<
<Instance as InstanceToPrefix>::ApprovesPrefix,
Blake2_128Concat,
<Runtime as frame_system::Config>::AccountId,
Blake2_128Concat,
<Runtime as frame_system::Config>::AccountId,
BalanceOf<Runtime, Instance>,
>;
pub type NoncesStorage<Instance> = StorageMap<
<Instance as InstanceToPrefix>::NoncesPrefix,
Blake2_128Concat,
H160,
U256,
ValueQuery,
>;
pub trait Erc20Metadata {
fn name() -> &'static str;
fn symbol() -> &'static str;
fn decimals() -> u8;
fn is_native_currency() -> bool;
}
pub struct Erc20BalancesPrecompile<Runtime, Metadata: Erc20Metadata, Instance: 'static = ()>(
PhantomData<(Runtime, Metadata, Instance)>,
);
#[precompile_utils::precompile]
impl<Runtime, Metadata, Instance> Erc20BalancesPrecompile<Runtime, Metadata, Instance>
where
Runtime: pallet_balances::Config<Instance> + pallet_evm::Config,
Runtime::RuntimeCall: Dispatchable<PostInfo = PostDispatchInfo> + GetDispatchInfo,
Runtime::RuntimeCall: From<pallet_balances::Call<Runtime, Instance>>,
<Runtime::RuntimeCall as Dispatchable>::RuntimeOrigin: From<Option<Runtime::AccountId>>,
BalanceOf<Runtime, Instance>: TryFrom<U256> + Into<U256>,
Metadata: Erc20Metadata,
Instance: InstanceToPrefix + 'static,
<Runtime as pallet_evm::Config>::AddressMapping: AddressMapping<Runtime::AccountId>,
{
#[precompile::public("totalSupply()")]
#[precompile::view]
fn total_supply(handle: &mut impl PrecompileHandle) -> EvmResult<U256> {
handle.record_db_read::<Runtime>(16)?;
Ok(pallet_balances::Pallet::<Runtime, Instance>::total_issuance().into())
}
#[precompile::public("balanceOf(address)")]
#[precompile::view]
fn balance_of(handle: &mut impl PrecompileHandle, owner: Address) -> EvmResult<U256> {
handle.record_db_read::<Runtime>(116)?;
let owner: H160 = owner.into();
let owner: Runtime::AccountId = Runtime::AddressMapping::into_account_id(owner);
Ok(pallet_balances::Pallet::<Runtime, Instance>::usable_balance(&owner).into())
}
#[precompile::public("allowance(address,address)")]
#[precompile::view]
fn allowance(
handle: &mut impl PrecompileHandle,
owner: Address,
spender: Address,
) -> EvmResult<U256> {
handle.record_db_read::<Runtime>(88)?;
let owner: H160 = owner.into();
let spender: H160 = spender.into();
let owner: Runtime::AccountId = Runtime::AddressMapping::into_account_id(owner);
let spender: Runtime::AccountId = Runtime::AddressMapping::into_account_id(spender);
Ok(ApprovesStorage::<Runtime, Instance>::get(owner, spender)
.unwrap_or_default()
.into())
}
#[precompile::public("approve(address,uint256)")]
fn approve(
handle: &mut impl PrecompileHandle,
spender: Address,
value: U256,
) -> EvmResult<bool> {
handle.record_cost(RuntimeHelper::<Runtime>::db_write_gas_cost())?;
handle.record_log_costs_manual(3, 32)?;
let spender: H160 = spender.into();
{
let caller: Runtime::AccountId =
Runtime::AddressMapping::into_account_id(handle.context().caller);
let spender: Runtime::AccountId = Runtime::AddressMapping::into_account_id(spender);
let value = Self::u256_to_amount(value).unwrap_or_else(|_| Bounded::max_value());
ApprovesStorage::<Runtime, Instance>::insert(caller, spender, value);
}
log3(
handle.context().address,
SELECTOR_LOG_APPROVAL,
handle.context().caller,
spender,
solidity::encode_event_data(value),
)
.record(handle)?;
Ok(true)
}
#[precompile::public("transfer(address,uint256)")]
fn transfer(handle: &mut impl PrecompileHandle, to: Address, value: U256) -> EvmResult<bool> {
handle.record_log_costs_manual(3, 32)?;
let to: H160 = to.into();
{
let origin = Runtime::AddressMapping::into_account_id(handle.context().caller);
let to = Runtime::AddressMapping::into_account_id(to);
let value = Self::u256_to_amount(value).in_field("value")?;
RuntimeHelper::<Runtime>::try_dispatch(
handle,
Some(origin).into(),
pallet_balances::Call::<Runtime, Instance>::transfer_allow_death {
dest: Runtime::Lookup::unlookup(to),
value,
},
SYSTEM_ACCOUNT_SIZE,
)?;
}
log3(
handle.context().address,
SELECTOR_LOG_TRANSFER,
handle.context().caller,
to,
solidity::encode_event_data(value),
)
.record(handle)?;
Ok(true)
}
#[precompile::public("transferFrom(address,address,uint256)")]
fn transfer_from(
handle: &mut impl PrecompileHandle,
from: Address,
to: Address,
value: U256,
) -> EvmResult<bool> {
handle.record_db_read::<Runtime>(88)?;
handle.record_cost(RuntimeHelper::<Runtime>::db_write_gas_cost())?;
handle.record_log_costs_manual(3, 32)?;
let from: H160 = from.into();
let to: H160 = to.into();
{
let caller: Runtime::AccountId =
Runtime::AddressMapping::into_account_id(handle.context().caller);
let from: Runtime::AccountId = Runtime::AddressMapping::into_account_id(from);
let to: Runtime::AccountId = Runtime::AddressMapping::into_account_id(to);
let value = Self::u256_to_amount(value).in_field("value")?;
if caller != from {
ApprovesStorage::<Runtime, Instance>::mutate(from.clone(), caller, |entry| {
let allowed = entry.ok_or(revert("spender not allowed"))?;
let allowed = allowed
.checked_sub(&value)
.ok_or_else(|| revert("trying to spend more than allowed"))?;
*entry = Some(allowed);
EvmResult::Ok(())
})?;
}
RuntimeHelper::<Runtime>::try_dispatch(
handle,
Some(from).into(),
pallet_balances::Call::<Runtime, Instance>::transfer_allow_death {
dest: Runtime::Lookup::unlookup(to),
value,
},
SYSTEM_ACCOUNT_SIZE,
)?;
}
log3(
handle.context().address,
SELECTOR_LOG_TRANSFER,
from,
to,
solidity::encode_event_data(value),
)
.record(handle)?;
Ok(true)
}
#[precompile::public("name()")]
#[precompile::view]
fn name(_handle: &mut impl PrecompileHandle) -> EvmResult<UnboundedBytes> {
Ok(Metadata::name().into())
}
#[precompile::public("symbol()")]
#[precompile::view]
fn symbol(_handle: &mut impl PrecompileHandle) -> EvmResult<UnboundedBytes> {
Ok(Metadata::symbol().into())
}
#[precompile::public("decimals()")]
#[precompile::view]
fn decimals(_handle: &mut impl PrecompileHandle) -> EvmResult<u8> {
Ok(Metadata::decimals())
}
#[precompile::public("deposit()")]
#[precompile::fallback]
#[precompile::payable]
fn deposit(handle: &mut impl PrecompileHandle) -> EvmResult {
if !Metadata::is_native_currency() {
return Err(RevertReason::UnknownSelector.into());
}
let caller: Runtime::AccountId =
Runtime::AddressMapping::into_account_id(handle.context().caller);
let precompile = Runtime::AddressMapping::into_account_id(handle.context().address);
let amount = Self::u256_to_amount(handle.context().apparent_value)?;
if amount.into() == U256::from(0u32) {
return Err(revert("deposited amount must be non-zero"));
}
handle.record_log_costs_manual(2, 32)?;
RuntimeHelper::<Runtime>::try_dispatch(
handle,
Some(precompile).into(),
pallet_balances::Call::<Runtime, Instance>::transfer_allow_death {
dest: Runtime::Lookup::unlookup(caller),
value: amount,
},
SYSTEM_ACCOUNT_SIZE,
)?;
log2(
handle.context().address,
SELECTOR_LOG_DEPOSIT,
handle.context().caller,
solidity::encode_event_data(handle.context().apparent_value),
)
.record(handle)?;
Ok(())
}
#[precompile::public("withdraw(uint256)")]
fn withdraw(handle: &mut impl PrecompileHandle, value: U256) -> EvmResult {
if !Metadata::is_native_currency() {
return Err(RevertReason::UnknownSelector.into());
}
handle.record_log_costs_manual(2, 32)?;
let account_amount: U256 = {
let owner: Runtime::AccountId =
Runtime::AddressMapping::into_account_id(handle.context().caller);
pallet_balances::Pallet::<Runtime, Instance>::usable_balance(&owner).into()
};
if value > account_amount {
return Err(revert("Trying to withdraw more than owned"));
}
log2(
handle.context().address,
SELECTOR_LOG_WITHDRAWAL,
handle.context().caller,
solidity::encode_event_data(value),
)
.record(handle)?;
Ok(())
}
#[precompile::public("permit(address,address,uint256,uint256,uint8,bytes32,bytes32)")]
#[allow(clippy::too_many_arguments)]
fn eip2612_permit(
handle: &mut impl PrecompileHandle,
owner: Address,
spender: Address,
value: U256,
deadline: U256,
v: u8,
r: H256,
s: H256,
) -> EvmResult {
<Eip2612<Runtime, Metadata, Instance>>::permit(
handle, owner, spender, value, deadline, v, r, s,
)
}
#[precompile::public("nonces(address)")]
#[precompile::view]
fn eip2612_nonces(handle: &mut impl PrecompileHandle, owner: Address) -> EvmResult<U256> {
<Eip2612<Runtime, Metadata, Instance>>::nonces(handle, owner)
}
#[precompile::public("DOMAIN_SEPARATOR()")]
#[precompile::view]
fn eip2612_domain_separator(handle: &mut impl PrecompileHandle) -> EvmResult<H256> {
<Eip2612<Runtime, Metadata, Instance>>::domain_separator(handle)
}
fn u256_to_amount(value: U256) -> MayRevert<BalanceOf<Runtime, Instance>> {
value
.try_into()
.map_err(|_| RevertReason::value_is_too_large("balance type").into())
}
}