#![cfg_attr(not(feature = "std"), no_std)]
use account::SYSTEM_ACCOUNT_SIZE;
use core::fmt::Display;
use fp_evm::{ExitError, PrecompileHandle};
use frame_support::traits::fungibles::Inspect;
use frame_support::traits::fungibles::{
approvals::Inspect as ApprovalInspect, metadata::Inspect as MetadataInspect,
roles::Inspect as RolesInspect,
};
use frame_support::traits::{Get, OriginTrait};
use frame_support::{
dispatch::{GetDispatchInfo, PostDispatchInfo},
sp_runtime::traits::StaticLookup,
};
use moonkit_xcm_primitives::AccountIdAssetIdConversion;
use pallet_evm::AddressMapping;
use precompile_utils::prelude::*;
use sp_runtime::traits::{Bounded, Dispatchable};
use sp_std::vec::Vec;
use pallet_moonbeam_lazy_migrations::is_migrating_foreign_assets;
use sp_core::{MaxEncodedLen, 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 type BalanceOf<Runtime, Instance = ()> = <Runtime as pallet_assets::Config<Instance>>::Balance;
pub type AssetIdOf<Runtime, Instance = ()> = <Runtime as pallet_assets::Config<Instance>>::AssetId;
pub struct Erc20AssetsPrecompileSet<Runtime, Instance: 'static = ()>(
PhantomData<(Runtime, Instance)>,
);
impl<R, V> Clone for Erc20AssetsPrecompileSet<R, V> {
fn clone(&self) -> Self {
Self(PhantomData)
}
}
impl<R, V> Default for Erc20AssetsPrecompileSet<R, V> {
fn default() -> Self {
Self(PhantomData)
}
}
impl<Runtime, Instance> Erc20AssetsPrecompileSet<Runtime, Instance> {
pub fn new() -> Self {
Self(PhantomData)
}
}
#[precompile_utils::precompile]
#[precompile::precompile_set]
#[precompile::test_concrete_types(mock::Runtime, pallet_assets::Instance1)]
impl<Runtime, Instance> Erc20AssetsPrecompileSet<Runtime, Instance>
where
Instance: eip2612::InstanceToPrefix + 'static,
Runtime: pallet_assets::Config<Instance> + pallet_evm::Config + frame_system::Config,
Runtime::RuntimeCall: Dispatchable<PostInfo = PostDispatchInfo> + GetDispatchInfo,
Runtime::RuntimeCall: From<pallet_assets::Call<Runtime, Instance>>,
<Runtime::RuntimeCall as Dispatchable>::RuntimeOrigin: From<Option<Runtime::AccountId>>,
BalanceOf<Runtime, Instance>: TryFrom<U256> + Into<U256> + solidity::Codec,
Runtime: AccountIdAssetIdConversion<Runtime::AccountId, AssetIdOf<Runtime, Instance>>,
<<Runtime as frame_system::Config>::RuntimeCall as Dispatchable>::RuntimeOrigin: OriginTrait,
AssetIdOf<Runtime, Instance>: Display,
Runtime::AccountId: Into<H160>,
<Runtime as pallet_evm::Config>::AddressMapping: AddressMapping<Runtime::AccountId>,
{
#[precompile::discriminant]
fn discriminant(address: H160, gas: u64) -> DiscriminantResult<AssetIdOf<Runtime, Instance>> {
let extra_cost = RuntimeHelper::<Runtime>::db_read_gas_cost();
if gas < extra_cost {
return DiscriminantResult::OutOfGas;
}
let account_id = Runtime::AddressMapping::into_account_id(address);
let asset_id = match Runtime::account_to_asset_id(account_id) {
Some((_, asset_id)) => asset_id,
None => return DiscriminantResult::None(extra_cost),
};
if pallet_assets::Pallet::<Runtime, Instance>::maybe_total_supply(asset_id.clone())
.is_some() && !is_migrating_foreign_assets()
{
DiscriminantResult::Some(asset_id, extra_cost)
} else {
DiscriminantResult::None(extra_cost)
}
}
#[precompile::public("totalSupply()")]
#[precompile::view]
fn total_supply(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
) -> EvmResult<U256> {
handle.record_db_read::<Runtime>(175)?;
Ok(pallet_assets::Pallet::<Runtime, Instance>::total_issuance(asset_id).into())
}
#[precompile::public("balanceOf(address)")]
#[precompile::view]
fn balance_of(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
who: Address,
) -> EvmResult<U256> {
handle.record_db_read::<Runtime>(
87 + <Runtime as pallet_assets::Config<Instance>>::Extra::max_encoded_len(),
)?;
let who: H160 = who.into();
let amount: U256 = {
let who: Runtime::AccountId = Runtime::AddressMapping::into_account_id(who);
pallet_assets::Pallet::<Runtime, Instance>::balance(asset_id, &who).into()
};
Ok(amount)
}
#[precompile::public("allowance(address,address)")]
#[precompile::view]
fn allowance(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
owner: Address,
spender: Address,
) -> EvmResult<U256> {
handle.record_db_read::<Runtime>(136)?;
let owner: H160 = owner.into();
let spender: H160 = spender.into();
let amount: U256 = {
let owner: Runtime::AccountId = Runtime::AddressMapping::into_account_id(owner);
let spender: Runtime::AccountId = Runtime::AddressMapping::into_account_id(spender);
pallet_assets::Pallet::<Runtime, Instance>::allowance(asset_id, &owner, &spender).into()
};
Ok(amount)
}
#[precompile::public("approve(address,uint256)")]
fn approve(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
spender: Address,
value: U256,
) -> EvmResult<bool> {
handle.record_log_costs_manual(3, 32)?;
let spender: H160 = spender.into();
Self::approve_inner(asset_id, handle, handle.context().caller, spender, value)?;
log3(
handle.context().address,
SELECTOR_LOG_APPROVAL,
handle.context().caller,
spender,
solidity::encode_event_data(value),
)
.record(handle)?;
Ok(true)
}
fn approve_inner(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
owner: H160,
spender: H160,
value: U256,
) -> EvmResult {
let owner = Runtime::AddressMapping::into_account_id(owner);
let spender: Runtime::AccountId = Runtime::AddressMapping::into_account_id(spender);
let amount: BalanceOf<Runtime, Instance> =
value.try_into().unwrap_or_else(|_| Bounded::max_value());
handle.record_db_read::<Runtime>(136)?;
if pallet_assets::Pallet::<Runtime, Instance>::allowance(asset_id.clone(), &owner, &spender)
!= 0u32.into()
{
RuntimeHelper::<Runtime>::try_dispatch(
handle,
Some(owner.clone()).into(),
pallet_assets::Call::<Runtime, Instance>::cancel_approval {
id: asset_id.clone().into(),
delegate: Runtime::Lookup::unlookup(spender.clone()),
},
0,
)?;
}
RuntimeHelper::<Runtime>::try_dispatch(
handle,
Some(owner).into(),
pallet_assets::Call::<Runtime, Instance>::approve_transfer {
id: asset_id.into(),
delegate: Runtime::Lookup::unlookup(spender),
amount,
},
0,
)?;
Ok(())
}
#[precompile::public("transfer(address,uint256)")]
fn transfer(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
to: Address,
value: U256,
) -> EvmResult<bool> {
handle.record_log_costs_manual(3, 32)?;
let to: H160 = to.into();
let value = Self::u256_to_amount(value).in_field("value")?;
{
let origin = Runtime::AddressMapping::into_account_id(handle.context().caller);
let to = Runtime::AddressMapping::into_account_id(to);
RuntimeHelper::<Runtime>::try_dispatch(
handle,
Some(origin).into(),
pallet_assets::Call::<Runtime, Instance>::transfer {
id: asset_id.into(),
target: Runtime::Lookup::unlookup(to),
amount: 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(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
from: Address,
to: Address,
value: U256,
) -> EvmResult<bool> {
handle.record_log_costs_manual(3, 32)?;
let from: H160 = from.into();
let to: H160 = to.into();
let value = Self::u256_to_amount(value).in_field("value")?;
{
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);
if caller != from {
RuntimeHelper::<Runtime>::try_dispatch(
handle,
Some(caller).into(),
pallet_assets::Call::<Runtime, Instance>::transfer_approved {
id: asset_id.into(),
owner: Runtime::Lookup::unlookup(from),
destination: Runtime::Lookup::unlookup(to),
amount: value,
},
SYSTEM_ACCOUNT_SIZE,
)?;
} else {
RuntimeHelper::<Runtime>::try_dispatch(
handle,
Some(from).into(),
pallet_assets::Call::<Runtime, Instance>::transfer {
id: asset_id.into(),
target: Runtime::Lookup::unlookup(to),
amount: 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(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
) -> EvmResult<UnboundedBytes> {
handle.record_db_read::<Runtime>(
50 + (2 * <Runtime as pallet_assets::Config<Instance>>::StringLimit::get()) as usize,
)?;
let name = pallet_assets::Pallet::<Runtime, Instance>::name(asset_id)
.as_slice()
.into();
Ok(name)
}
#[precompile::public("symbol()")]
#[precompile::view]
fn symbol(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
) -> EvmResult<UnboundedBytes> {
handle.record_db_read::<Runtime>(
50 + (2 * <Runtime as pallet_assets::Config<Instance>>::StringLimit::get()) as usize,
)?;
let symbol = pallet_assets::Pallet::<Runtime, Instance>::symbol(asset_id)
.as_slice()
.into();
Ok(symbol)
}
#[precompile::public("decimals()")]
#[precompile::view]
fn decimals(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
) -> EvmResult<u8> {
handle.record_db_read::<Runtime>(
50 + (2 * <Runtime as pallet_assets::Config<Instance>>::StringLimit::get()) as usize,
)?;
Ok(pallet_assets::Pallet::<Runtime, Instance>::decimals(
asset_id,
))
}
#[precompile::public("owner()")]
#[precompile::view]
fn owner(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
) -> EvmResult<Address> {
handle.record_db_read::<Runtime>(175)?;
let owner: H160 = pallet_assets::Pallet::<Runtime, Instance>::owner(asset_id)
.ok_or(revert("No owner set"))?
.into();
Ok(Address(owner))
}
#[precompile::public("issuer()")]
#[precompile::view]
fn issuer(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
) -> EvmResult<Address> {
handle.record_db_read::<Runtime>(175)?;
let issuer: H160 = pallet_assets::Pallet::<Runtime, Instance>::issuer(asset_id)
.ok_or(revert("No issuer set"))?
.into();
Ok(Address(issuer))
}
#[precompile::public("admin()")]
#[precompile::view]
fn admin(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
) -> EvmResult<Address> {
handle.record_db_read::<Runtime>(175)?;
let admin: H160 = pallet_assets::Pallet::<Runtime, Instance>::admin(asset_id)
.ok_or(revert("No admin set"))?
.into();
Ok(Address(admin))
}
#[precompile::public("freezer()")]
#[precompile::view]
fn freezer(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
) -> EvmResult<Address> {
handle.record_db_read::<Runtime>(175)?;
let freezer: H160 = pallet_assets::Pallet::<Runtime, Instance>::freezer(asset_id)
.ok_or(revert("No freezer set"))?
.into();
Ok(Address(freezer))
}
#[precompile::public("permit(address,address,uint256,uint256,uint8,bytes32,bytes32)")]
#[allow(clippy::too_many_arguments)]
fn eip2612_permit(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
owner: Address,
spender: Address,
value: U256,
deadline: U256,
v: u8,
r: H256,
s: H256,
) -> EvmResult {
<Eip2612<Runtime, Instance>>::permit(
asset_id, handle, owner, spender, value, deadline, v, r, s,
)
}
#[precompile::public("nonces(address)")]
#[precompile::view]
fn eip2612_nonces(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
owner: Address,
) -> EvmResult<U256> {
<Eip2612<Runtime, Instance>>::nonces(asset_id, handle, owner)
}
#[precompile::public("DOMAIN_SEPARATOR()")]
#[precompile::view]
fn eip2612_domain_separator(
asset_id: AssetIdOf<Runtime, Instance>,
handle: &mut impl PrecompileHandle,
) -> EvmResult<H256> {
<Eip2612<Runtime, Instance>>::domain_separator(asset_id, handle)
}
fn u256_to_amount(value: U256) -> MayRevert<BalanceOf<Runtime, Instance>> {
value
.try_into()
.map_err(|_| RevertReason::value_is_too_large("balance type").into())
}
}