1
// Copyright 2019-2025 PureStake Inc.
2
// This file is part of Moonbeam.
3

            
4
// Moonbeam is free software: you can redistribute it and/or modify
5
// it under the terms of the GNU General Public License as published by
6
// the Free Software Foundation, either version 3 of the License, or
7
// (at your option) any later version.
8

            
9
// Moonbeam is distributed in the hope that it will be useful,
10
// but WITHOUT ANY WARRANTY; without even the implied warranty of
11
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12
// GNU General Public License for more details.
13

            
14
// You should have received a copy of the GNU General Public License
15
// along with Moonbeam.  If not, see <http://www.gnu.org/licenses/>.
16

            
17
use super::moonbeam_weights;
18
use crate::{
19
	asset_config::ForeignAssetInstance,
20
	xcm_config::{AssetType, XcmExecutorConfig},
21
	AccountId, AssetId, AssetManager, Balances, Erc20XcmBridge, EvmForeignAssets,
22
	OpenTechCommitteeInstance, Runtime, TreasuryCouncilInstance, H160,
23
};
24
use frame_support::parameter_types;
25
use moonkit_xcm_primitives::{
26
	location_matcher::{Erc20PalletMatcher, ForeignAssetMatcher, SingleAddressMatcher},
27
	AccountIdAssetIdConversion,
28
};
29
use pallet_evm_precompile_author_mapping::AuthorMappingPrecompile;
30
use pallet_evm_precompile_balances_erc20::{Erc20BalancesPrecompile, Erc20Metadata};
31
use pallet_evm_precompile_batch::BatchPrecompile;
32
use pallet_evm_precompile_blake2::Blake2F;
33
use pallet_evm_precompile_bn128::{Bn128Add, Bn128Mul, Bn128Pairing};
34
use pallet_evm_precompile_call_permit::CallPermitPrecompile;
35
use pallet_evm_precompile_collective::CollectivePrecompile;
36
use pallet_evm_precompile_conviction_voting::ConvictionVotingPrecompile;
37
use pallet_evm_precompile_crowdloan_rewards::CrowdloanRewardsPrecompile;
38
use pallet_evm_precompile_gmp::GmpPrecompile;
39
use pallet_evm_precompile_identity::IdentityPrecompile;
40
use pallet_evm_precompile_modexp::Modexp;
41
use pallet_evm_precompile_p256verify::P256Verify;
42
use pallet_evm_precompile_parachain_staking::ParachainStakingPrecompile;
43
use pallet_evm_precompile_preimage::PreimagePrecompile;
44
use pallet_evm_precompile_proxy::{OnlyIsProxyAndProxy, ProxyPrecompile};
45
use pallet_evm_precompile_randomness::RandomnessPrecompile;
46
use pallet_evm_precompile_referenda::ReferendaPrecompile;
47
use pallet_evm_precompile_registry::PrecompileRegistry;
48
use pallet_evm_precompile_relay_encoder::RelayEncoderPrecompile;
49
use pallet_evm_precompile_relay_verifier::RelayDataVerifierPrecompile;
50
use pallet_evm_precompile_sha3fips::Sha3FIPS256;
51
use pallet_evm_precompile_simple::{ECRecover, ECRecoverPublicKey, Identity, Ripemd160, Sha256};
52
use pallet_evm_precompile_xcm::PalletXcmPrecompile;
53
use pallet_evm_precompile_xcm_transactor::{
54
	v1::XcmTransactorPrecompileV1, v2::XcmTransactorPrecompileV2, v3::XcmTransactorPrecompileV3,
55
};
56
use pallet_evm_precompile_xcm_utils::XcmUtilsPrecompile;
57
use pallet_evm_precompile_xtokens::XtokensPrecompile;
58
use pallet_evm_precompileset_assets_erc20::Erc20AssetsPrecompileSet;
59
use pallet_precompile_benchmarks::WeightInfo;
60
use precompile_utils::precompile_set::*;
61
use sp_std::prelude::*;
62
use xcm_primitives::AsAssetType;
63

            
64
parameter_types! {
65
	pub P256VerifyWeight: frame_support::weights::Weight =
66
		moonbeam_weights::pallet_precompile_benchmarks::WeightInfo::<Runtime>::p256_verify();
67
}
68

            
69
pub struct NativeErc20Metadata;
70

            
71
/// ERC20 metadata for the native token.
72
impl Erc20Metadata for NativeErc20Metadata {
73
	/// Returns the name of the token.
74
	fn name() -> &'static str {
75
		"GLMR token"
76
	}
77

            
78
	/// Returns the symbol of the token.
79
	fn symbol() -> &'static str {
80
		"GLMR"
81
	}
82

            
83
	/// Returns the decimals places of the token.
84
	fn decimals() -> u8 {
85
		18
86
	}
87

            
88
	/// Must return `true` only if it represents the main native currency of
89
	/// the network. It must be the currency used in `pallet_evm`.
90
6
	fn is_native_currency() -> bool {
91
6
		true
92
6
	}
93
}
94

            
95
/// The asset precompile address prefix. Addresses that match against this prefix will be routed
96
/// to Erc20AssetsPrecompileSet being marked as foreign
97
pub const FOREIGN_ASSET_PRECOMPILE_ADDRESS_PREFIX: &[u8] = &[255u8; 4];
98
/// The asset precompile address prefix. Addresses that match against this prefix will be routed
99
/// to Erc20AssetsPrecompileSet being marked as local
100
pub const LOCAL_ASSET_PRECOMPILE_ADDRESS_PREFIX: &[u8] = &[255u8, 255u8, 255u8, 254u8];
101

            
102
/// Const to identify ERC20_BALANCES_PRECOMPILE address
103
pub const ERC20_BALANCES_PRECOMPILE: u64 = 2050;
104

            
105
parameter_types! {
106
	pub ForeignAssetPrefix: &'static [u8] = FOREIGN_ASSET_PRECOMPILE_ADDRESS_PREFIX;
107
	pub LocalAssetPrefix: &'static [u8] = LOCAL_ASSET_PRECOMPILE_ADDRESS_PREFIX;
108
}
109

            
110
type EthereumPrecompilesChecks = (AcceptDelegateCall, CallableByContract, CallableByPrecompile);
111

            
112
// Pallet-xcm precompile types.
113
// Type that converts AssetId into Location
114
type AssetIdToLocationManager = (
115
	AsAssetType<AssetId, AssetType, AssetManager>,
116
	EvmForeignAssets,
117
);
118

            
119
// The pallet-balances address is identified by ERC20_BALANCES_PRECOMPILE const
120
type SingleAddressMatch = SingleAddressMatcher<AccountId, ERC20_BALANCES_PRECOMPILE, Balances>;
121

            
122
// Type that matches an AccountId with a foreign asset address (if any)
123
type ForeignAssetMatch = ForeignAssetMatcher<AccountId, AssetId, Runtime, AssetIdToLocationManager>;
124

            
125
// Erc20XcmBridge pallet is used to match ERC20s
126
type Erc20Match = Erc20PalletMatcher<AccountId, Erc20XcmBridge>;
127

            
128
#[precompile_utils::precompile_name_from_address]
129
type MoonbeamPrecompilesAt<R> = (
130
	// Ethereum precompiles:
131
	// We allow DELEGATECALL to stay compliant with Ethereum behavior.
132
	PrecompileAt<AddressU64<1>, ECRecover, EthereumPrecompilesChecks>,
133
	PrecompileAt<AddressU64<2>, Sha256, EthereumPrecompilesChecks>,
134
	PrecompileAt<AddressU64<3>, Ripemd160, EthereumPrecompilesChecks>,
135
	PrecompileAt<AddressU64<4>, Identity, EthereumPrecompilesChecks>,
136
	PrecompileAt<AddressU64<5>, Modexp, EthereumPrecompilesChecks>,
137
	PrecompileAt<AddressU64<6>, Bn128Add, EthereumPrecompilesChecks>,
138
	PrecompileAt<AddressU64<7>, Bn128Mul, EthereumPrecompilesChecks>,
139
	PrecompileAt<AddressU64<8>, Bn128Pairing, EthereumPrecompilesChecks>,
140
	PrecompileAt<AddressU64<9>, Blake2F, EthereumPrecompilesChecks>,
141
	// (0x100 => 256) https://github.com/ethereum/RIPs/blob/master/RIPS/rip-7212.md
142
	PrecompileAt<AddressU64<256>, P256Verify<P256VerifyWeight>, EthereumPrecompilesChecks>,
143
	// Non-Moonbeam specific nor Ethereum precompiles :
144
	PrecompileAt<AddressU64<1024>, Sha3FIPS256, (CallableByContract, CallableByPrecompile)>,
145
	RemovedPrecompileAt<AddressU64<1025>>, // Dispatch<R>
146
	PrecompileAt<AddressU64<1026>, ECRecoverPublicKey, (CallableByContract, CallableByPrecompile)>,
147
	// Moonbeam specific precompiles:
148
	PrecompileAt<
149
		AddressU64<2048>,
150
		ParachainStakingPrecompile<R>,
151
		(CallableByContract, CallableByPrecompile),
152
	>,
153
	PrecompileAt<
154
		AddressU64<2049>,
155
		CrowdloanRewardsPrecompile<R>,
156
		(CallableByContract, CallableByPrecompile),
157
	>,
158
	PrecompileAt<
159
		AddressU64<ERC20_BALANCES_PRECOMPILE>,
160
		Erc20BalancesPrecompile<R, NativeErc20Metadata>,
161
		(CallableByContract, CallableByPrecompile),
162
	>,
163
	RemovedPrecompileAt<AddressU64<2051>>, // Democracy
164
	PrecompileAt<
165
		AddressU64<2052>,
166
		XtokensPrecompile<R>,
167
		(
168
			SubcallWithMaxNesting<1>,
169
			CallableByContract,
170
			CallableByPrecompile,
171
		),
172
	>,
173
	PrecompileAt<
174
		AddressU64<2053>,
175
		RelayEncoderPrecompile<R>,
176
		(CallableByContract, CallableByPrecompile),
177
	>,
178
	PrecompileAt<
179
		AddressU64<2054>,
180
		XcmTransactorPrecompileV1<R>,
181
		(
182
			SubcallWithMaxNesting<1>,
183
			CallableByContract,
184
			CallableByPrecompile,
185
		),
186
	>,
187
	PrecompileAt<
188
		AddressU64<2055>,
189
		AuthorMappingPrecompile<R>,
190
		(CallableByContract, CallableByPrecompile),
191
	>,
192
	PrecompileAt<
193
		AddressU64<2056>,
194
		BatchPrecompile<R>,
195
		(
196
			SubcallWithMaxNesting<2>,
197
			// Batch is the only precompile allowed to call Batch.
198
			CallableByPrecompile<OnlyFrom<AddressU64<2056>>>,
199
		),
200
	>,
201
	PrecompileAt<
202
		AddressU64<2057>,
203
		RandomnessPrecompile<R>,
204
		(SubcallWithMaxNesting<0>, CallableByContract),
205
	>,
206
	PrecompileAt<
207
		AddressU64<2058>,
208
		CallPermitPrecompile<R>,
209
		(SubcallWithMaxNesting<0>, CallableByContract),
210
	>,
211
	PrecompileAt<
212
		AddressU64<2059>,
213
		ProxyPrecompile<R>,
214
		(
215
			CallableByContract<OnlyIsProxyAndProxy<R>>,
216
			SubcallWithMaxNesting<0>,
217
			// Batch is the only precompile allowed to call Proxy.
218
			CallableByPrecompile<OnlyFrom<AddressU64<2056>>>,
219
		),
220
	>,
221
	PrecompileAt<
222
		AddressU64<2060>,
223
		XcmUtilsPrecompile<R, XcmExecutorConfig>,
224
		CallableByContract<
225
			pallet_evm_precompile_xcm_utils::AllExceptXcmExecute<R, XcmExecutorConfig>,
226
		>,
227
	>,
228
	PrecompileAt<
229
		AddressU64<2061>,
230
		XcmTransactorPrecompileV2<R>,
231
		(
232
			SubcallWithMaxNesting<1>,
233
			CallableByContract,
234
			CallableByPrecompile,
235
		),
236
	>,
237
	RemovedPrecompileAt<AddressU64<2062>>, //CouncilInstance
238
	RemovedPrecompileAt<AddressU64<2063>>, // TechCommitteeInstance
239
	PrecompileAt<
240
		AddressU64<2064>,
241
		CollectivePrecompile<R, TreasuryCouncilInstance>,
242
		(CallableByContract, CallableByPrecompile),
243
	>,
244
	PrecompileAt<
245
		AddressU64<2065>,
246
		ReferendaPrecompile<R, crate::governance::custom_origins::Origin>,
247
		(CallableByContract, CallableByPrecompile),
248
	>,
249
	PrecompileAt<
250
		AddressU64<2066>,
251
		ConvictionVotingPrecompile<R>,
252
		(CallableByContract, CallableByPrecompile),
253
	>,
254
	PrecompileAt<
255
		AddressU64<2067>,
256
		PreimagePrecompile<R>,
257
		(CallableByContract, CallableByPrecompile),
258
	>,
259
	PrecompileAt<
260
		AddressU64<2068>,
261
		CollectivePrecompile<R, OpenTechCommitteeInstance>,
262
		(CallableByContract, CallableByPrecompile),
263
	>,
264
	PrecompileAt<
265
		AddressU64<2069>,
266
		PrecompileRegistry<R>,
267
		(CallableByContract, CallableByPrecompile),
268
	>,
269
	PrecompileAt<AddressU64<2070>, GmpPrecompile<R>, SubcallWithMaxNesting<0>>,
270
	PrecompileAt<
271
		AddressU64<2071>,
272
		XcmTransactorPrecompileV3<R>,
273
		(
274
			SubcallWithMaxNesting<1>,
275
			CallableByContract,
276
			CallableByPrecompile,
277
		),
278
	>,
279
	PrecompileAt<
280
		AddressU64<2072>,
281
		IdentityPrecompile<R, crate::MaxAdditionalFields>,
282
		(CallableByContract, CallableByPrecompile),
283
	>,
284
	PrecompileAt<
285
		AddressU64<2073>,
286
		RelayDataVerifierPrecompile<
287
			R,
288
			moonbeam_weights::pallet_precompile_benchmarks::WeightInfo<Runtime>,
289
		>,
290
		(CallableByContract, CallableByPrecompile),
291
	>,
292
	PrecompileAt<
293
		AddressU64<2074>,
294
		PalletXcmPrecompile<R, (SingleAddressMatch, ForeignAssetMatch, Erc20Match)>,
295
		(
296
			CallableByContract,
297
			CallableByPrecompile,
298
			SubcallWithMaxNesting<1>,
299
		),
300
	>,
301
);
302

            
303
pub struct DisabledLocalAssets<Runtime>(sp_std::marker::PhantomData<Runtime>);
304

            
305
impl<Runtime> sp_core::Get<Vec<H160>> for DisabledLocalAssets<Runtime>
306
where
307
	Runtime: frame_system::Config,
308
	Runtime::AccountId: Into<H160>,
309
	Runtime: AccountIdAssetIdConversion<Runtime::AccountId, AssetId>,
310
{
311
12333
	fn get() -> Vec<H160> {
312
12333
		vec![
313
12333
			337110116006454532607322340792629567158u128,
314
12333
			278750993613512357835566279094880339619,
315
12333
			228256396637196286254896220398224702687,
316
12333
			270195117769614861929703564202131636628,
317
12333
		]
318
12333
		.iter()
319
49332
		.map(|id| Runtime::asset_id_to_account(LOCAL_ASSET_PRECOMPILE_ADDRESS_PREFIX, *id).into())
320
12333
		.collect()
321
12333
	}
322
}
323

            
324
/// The PrecompileSet installed in the Moonbeam runtime.
325
/// We include the nine Istanbul precompiles
326
/// (https://github.com/ethereum/go-ethereum/blob/3c46f557/core/vm/contracts.go#L69)
327
/// as well as a special precompile for dispatching Substrate extrinsics
328
/// The following distribution has been decided for the precompiles
329
/// 0-1023: Ethereum Mainnet Precompiles
330
/// 1024-2047 Precompiles that are not in Ethereum Mainnet but are neither Moonbeam specific
331
/// 2048-4095 Moonbeam specific precompiles
332
pub type MoonbeamPrecompiles<R> = PrecompileSetBuilder<
333
	R,
334
	(
335
		// Skip precompiles if out of range.
336
		PrecompilesInRangeInclusive<(AddressU64<1>, AddressU64<4095>), MoonbeamPrecompilesAt<R>>,
337
		// Prefixed precompile sets (XC20)
338
		PrecompileSetStartingWith<
339
			ForeignAssetPrefix,
340
			Erc20AssetsPrecompileSet<R, ForeignAssetInstance>,
341
			(CallableByContract, CallableByPrecompile),
342
		>,
343
		RemovedPrecompilesAt<DisabledLocalAssets<R>>,
344
	),
345
>;