Skip to content

Commit 4c28354

Browse files
atheigithub-actions[bot]pgherveoubkchrggwpez
authored
revive: Include immutable storage deposit into the contracts storage_base_deposit (#7230)
This PR is centered around a main fix regarding the base deposit and a bunch of drive by or related fixtures that make sense to resolve in one go. It could be broken down more but I am constantly rebasing this PR and would appreciate getting those fixes in as-one. **This adds a multi block migration to Westend AssetHub that wipes the pallet state clean. This is necessary because of the changes to the `ContractInfo` storage item. It will not delete the child storage though. This will leave a tiny bit of garbage behind but won't cause any problems. They will just be orphaned.** ## Record the deposit for immutable data into the `storage_base_deposit` The `storage_base_deposit` are all the deposit a contract has to pay for existing. It included the deposit for its own metadata and a deposit proportional (< 1.0x) to the size of its code. However, the immutable code size was not recorded there. This would lead to the situation where on terminate this portion wouldn't be refunded staying locked into the contract. It would also make the calculation of the deposit changes on `set_code_hash` more complicated when it updates the immutable data (to be done in #6985). Reason is because it didn't know how much was payed before since the storage prices could have changed in the mean time. In order for this solution to work I needed to delay the deposit calculation for a new contract for after the contract is done executing is constructor as only then we know the immutable data size. Before, we just charged this eagerly in `charge_instantiate` before we execute the constructor. Now, we merely send the ED as free balance before the constructor in order to create the account. After the constructor is done we calculate the contract base deposit and charge it. This will make `set_code_hash` much easier to implement. As a side effect it is now legal to call `set_immutable_data` multiple times per constructor (even though I see no reason to do so). It simply overrides the immutable data with the new value. The deposit accounting will be done after the constructor returns (as mentioned above) instead of when setting the immutable data. ## Don't pre-charge for reading immutable data I noticed that we were pre-charging weight for the max allowable immutable data when reading those values and then refunding after read. This is not necessary as we know its length without reading the storage as we store it out of band in contract metadata. This makes reading it free. Less pre-charging less problems. ## Remove delegate locking Fixes #7092 This is also in the spirit of making #6985 easier to implement. The locking complicates `set_code_hash` as we might need to block settings the code hash when locks exist. Check #7092 for further rationale. ## Enforce "no terminate in constructor" eagerly We used to enforce this rule after the contract execution returned. Now we error out early in the host call. This makes it easier to be sure to argue that a contract info still exists (wasn't terminated) when a constructor successfully returns. All around this his just much simpler than dealing this check. ## Moved refcount functions to `CodeInfo` They never really made sense to exist on `Stack`. But now with the locking gone this makes even less sense. The refcount is stored inside `CodeInfo` to lets just move them there. ## Set `CodeHashLockupDepositPercent` for test runtime The test runtime was setting `CodeHashLockupDepositPercent` to zero. This was trivializing many code paths and excluded them from testing. I set it to `30%` which is our default value and fixed up all the tests that broke. This should give us confidence that the lockup doeposit collections properly works. ## Reworked the `MockExecutable` to have both a `deploy` and a `call` entry point This type used for testing could only have either entry points but not both. In order to fix the `immutable_data_set_overrides` I needed to a new function `add_both` to `MockExecutable` that allows to have both entry points. Make sure to make use of it in the future :) --------- Co-authored-by: command-bot <> Co-authored-by: cmd[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: PG Herveou <[email protected]> Co-authored-by: Bastian Köcher <[email protected]> Co-authored-by: Oliver Tale-Yazdi <[email protected]>
1 parent a883475 commit 4c28354

File tree

31 files changed

+1944
-1745
lines changed

31 files changed

+1944
-1745
lines changed

Cargo.lock

Lines changed: 2 additions & 0 deletions
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

cumulus/parachains/runtimes/assets/asset-hub-westend/Cargo.toml

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@ pallet-assets-freezer = { workspace = true }
3636
pallet-aura = { workspace = true }
3737
pallet-authorship = { workspace = true }
3838
pallet-balances = { workspace = true }
39+
pallet-migrations = { workspace = true }
3940
pallet-multisig = { workspace = true }
4041
pallet-nft-fractionalization = { workspace = true }
4142
pallet-nfts = { workspace = true }
@@ -133,6 +134,7 @@ runtime-benchmarks = [
133134
"pallet-balances/runtime-benchmarks",
134135
"pallet-collator-selection/runtime-benchmarks",
135136
"pallet-message-queue/runtime-benchmarks",
137+
"pallet-migrations/runtime-benchmarks",
136138
"pallet-multisig/runtime-benchmarks",
137139
"pallet-nft-fractionalization/runtime-benchmarks",
138140
"pallet-nfts/runtime-benchmarks",
@@ -177,6 +179,7 @@ try-runtime = [
177179
"pallet-balances/try-runtime",
178180
"pallet-collator-selection/try-runtime",
179181
"pallet-message-queue/try-runtime",
182+
"pallet-migrations/try-runtime",
180183
"pallet-multisig/try-runtime",
181184
"pallet-nft-fractionalization/try-runtime",
182185
"pallet-nfts/try-runtime",
@@ -230,6 +233,7 @@ std = [
230233
"pallet-balances/std",
231234
"pallet-collator-selection/std",
232235
"pallet-message-queue/std",
236+
"pallet-migrations/std",
233237
"pallet-multisig/std",
234238
"pallet-nft-fractionalization/std",
235239
"pallet-nfts-runtime-api/std",

cumulus/parachains/runtimes/assets/asset-hub-westend/src/lib.rs

Lines changed: 22 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -185,6 +185,7 @@ impl frame_system::Config for Runtime {
185185
type SS58Prefix = SS58Prefix;
186186
type OnSetCode = cumulus_pallet_parachain_system::ParachainSetCode<Self>;
187187
type MaxConsumers = frame_support::traits::ConstU32<16>;
188+
type MultiBlockMigrator = MultiBlockMigrations;
188189
}
189190

190191
impl cumulus_pallet_weight_reclaim::Config for Runtime {
@@ -1095,6 +1096,25 @@ impl TryFrom<RuntimeCall> for pallet_revive::Call<Runtime> {
10951096
}
10961097
}
10971098

1099+
parameter_types! {
1100+
pub MbmServiceWeight: Weight = Perbill::from_percent(80) * RuntimeBlockWeights::get().max_block;
1101+
}
1102+
1103+
impl pallet_migrations::Config for Runtime {
1104+
type RuntimeEvent = RuntimeEvent;
1105+
#[cfg(not(feature = "runtime-benchmarks"))]
1106+
type Migrations = pallet_migrations::migrations::ResetPallet<Runtime, Revive>;
1107+
// Benchmarks need mocked migrations to guarantee that they succeed.
1108+
#[cfg(feature = "runtime-benchmarks")]
1109+
type Migrations = pallet_migrations::mock_helpers::MockedMigrations;
1110+
type CursorMaxLen = ConstU32<65_536>;
1111+
type IdentifierMaxLen = ConstU32<256>;
1112+
type MigrationStatusHandler = ();
1113+
type FailedMigrationHandler = frame_support::migrations::FreezeChainOnFailedMigration;
1114+
type MaxServiceWeight = MbmServiceWeight;
1115+
type WeightInfo = weights::pallet_migrations::WeightInfo<Runtime>;
1116+
}
1117+
10981118
// Create the runtime by composing the FRAME pallets that were previously configured.
10991119
construct_runtime!(
11001120
pub enum Runtime
@@ -1106,6 +1126,7 @@ construct_runtime!(
11061126
Timestamp: pallet_timestamp = 3,
11071127
ParachainInfo: parachain_info = 4,
11081128
WeightReclaim: cumulus_pallet_weight_reclaim = 5,
1129+
MultiBlockMigrations: pallet_migrations = 6,
11091130

11101131
// Monetary stuff.
11111132
Balances: pallet_balances = 10,
@@ -1430,6 +1451,7 @@ mod benches {
14301451
[pallet_asset_conversion_tx_payment, AssetTxPayment]
14311452
[pallet_balances, Balances]
14321453
[pallet_message_queue, MessageQueue]
1454+
[pallet_migrations, MultiBlockMigrations]
14331455
[pallet_multisig, Multisig]
14341456
[pallet_nft_fractionalization, NftFractionalization]
14351457
[pallet_nfts, Nfts]

cumulus/parachains/runtimes/assets/asset-hub-westend/src/weights/mod.rs

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@ pub mod pallet_assets_pool;
3030
pub mod pallet_balances;
3131
pub mod pallet_collator_selection;
3232
pub mod pallet_message_queue;
33+
pub mod pallet_migrations;
3334
pub mod pallet_multisig;
3435
pub mod pallet_nft_fractionalization;
3536
pub mod pallet_nfts;
Lines changed: 225 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,225 @@
1+
// Copyright (C) Parity Technologies (UK) Ltd.
2+
// This file is part of Cumulus.
3+
4+
// Cumulus 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+
// Cumulus 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 Cumulus. If not, see <http://www.gnu.org/licenses/>.
16+
17+
//! Autogenerated weights for `pallet_migrations`
18+
//!
19+
//! THIS FILE WAS AUTO-GENERATED USING THE SUBSTRATE BENCHMARK CLI VERSION 32.0.0
20+
//! DATE: 2025-01-27, STEPS: `50`, REPEAT: `20`, LOW RANGE: `[]`, HIGH RANGE: `[]`
21+
//! WORST CASE MAP SIZE: `1000000`
22+
//! HOSTNAME: `17938671047b`, CPU: `Intel(R) Xeon(R) CPU @ 2.60GHz`
23+
//! WASM-EXECUTION: `Compiled`, CHAIN: `None`, DB CACHE: 1024
24+
25+
// Executed Command:
26+
// frame-omni-bencher
27+
// v1
28+
// benchmark
29+
// pallet
30+
// --extrinsic=*
31+
// --runtime=target/production/wbuild/asset-hub-westend-runtime/asset_hub_westend_runtime.wasm
32+
// --pallet=pallet_migrations
33+
// --header=/__w/polkadot-sdk/polkadot-sdk/cumulus/file_header.txt
34+
// --output=./cumulus/parachains/runtimes/assets/asset-hub-westend/src/weights
35+
// --wasm-execution=compiled
36+
// --steps=50
37+
// --repeat=20
38+
// --heap-pages=4096
39+
// --no-storage-info
40+
// --no-min-squares
41+
// --no-median-slopes
42+
43+
#![cfg_attr(rustfmt, rustfmt_skip)]
44+
#![allow(unused_parens)]
45+
#![allow(unused_imports)]
46+
#![allow(missing_docs)]
47+
48+
use frame_support::{traits::Get, weights::Weight};
49+
use core::marker::PhantomData;
50+
51+
/// Weight functions for `pallet_migrations`.
52+
pub struct WeightInfo<T>(PhantomData<T>);
53+
impl<T: frame_system::Config> pallet_migrations::WeightInfo for WeightInfo<T> {
54+
/// Storage: `MultiBlockMigrations::Cursor` (r:1 w:1)
55+
/// Proof: `MultiBlockMigrations::Cursor` (`max_values`: Some(1), `max_size`: Some(65550), added: 66045, mode: `MaxEncodedLen`)
56+
/// Storage: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
57+
/// Proof: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
58+
fn onboard_new_mbms() -> Weight {
59+
// Proof Size summary in bytes:
60+
// Measured: `171`
61+
// Estimated: `67035`
62+
// Minimum execution time: 8_697_000 picoseconds.
63+
Weight::from_parts(8_998_000, 0)
64+
.saturating_add(Weight::from_parts(0, 67035))
65+
.saturating_add(T::DbWeight::get().reads(2))
66+
.saturating_add(T::DbWeight::get().writes(1))
67+
}
68+
/// Storage: `MultiBlockMigrations::Cursor` (r:1 w:0)
69+
/// Proof: `MultiBlockMigrations::Cursor` (`max_values`: Some(1), `max_size`: Some(65550), added: 66045, mode: `MaxEncodedLen`)
70+
fn progress_mbms_none() -> Weight {
71+
// Proof Size summary in bytes:
72+
// Measured: `42`
73+
// Estimated: `67035`
74+
// Minimum execution time: 2_737_000 picoseconds.
75+
Weight::from_parts(2_813_000, 0)
76+
.saturating_add(Weight::from_parts(0, 67035))
77+
.saturating_add(T::DbWeight::get().reads(1))
78+
}
79+
/// Storage: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
80+
/// Proof: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
81+
/// Storage: `MultiBlockMigrations::Cursor` (r:0 w:1)
82+
/// Proof: `MultiBlockMigrations::Cursor` (`max_values`: Some(1), `max_size`: Some(65550), added: 66045, mode: `MaxEncodedLen`)
83+
fn exec_migration_completed() -> Weight {
84+
// Proof Size summary in bytes:
85+
// Measured: `129`
86+
// Estimated: `3594`
87+
// Minimum execution time: 6_181_000 picoseconds.
88+
Weight::from_parts(6_458_000, 0)
89+
.saturating_add(Weight::from_parts(0, 3594))
90+
.saturating_add(T::DbWeight::get().reads(1))
91+
.saturating_add(T::DbWeight::get().writes(1))
92+
}
93+
/// Storage: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
94+
/// Proof: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
95+
/// Storage: `MultiBlockMigrations::Historic` (r:1 w:0)
96+
/// Proof: `MultiBlockMigrations::Historic` (`max_values`: None, `max_size`: Some(266), added: 2741, mode: `MaxEncodedLen`)
97+
fn exec_migration_skipped_historic() -> Weight {
98+
// Proof Size summary in bytes:
99+
// Measured: `225`
100+
// Estimated: `3731`
101+
// Minimum execution time: 11_932_000 picoseconds.
102+
Weight::from_parts(12_539_000, 0)
103+
.saturating_add(Weight::from_parts(0, 3731))
104+
.saturating_add(T::DbWeight::get().reads(2))
105+
}
106+
/// Storage: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
107+
/// Proof: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
108+
/// Storage: `MultiBlockMigrations::Historic` (r:1 w:0)
109+
/// Proof: `MultiBlockMigrations::Historic` (`max_values`: None, `max_size`: Some(266), added: 2741, mode: `MaxEncodedLen`)
110+
fn exec_migration_advance() -> Weight {
111+
// Proof Size summary in bytes:
112+
// Measured: `171`
113+
// Estimated: `3731`
114+
// Minimum execution time: 11_127_000 picoseconds.
115+
Weight::from_parts(11_584_000, 0)
116+
.saturating_add(Weight::from_parts(0, 3731))
117+
.saturating_add(T::DbWeight::get().reads(2))
118+
}
119+
/// Storage: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
120+
/// Proof: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
121+
/// Storage: `MultiBlockMigrations::Historic` (r:1 w:1)
122+
/// Proof: `MultiBlockMigrations::Historic` (`max_values`: None, `max_size`: Some(266), added: 2741, mode: `MaxEncodedLen`)
123+
fn exec_migration_complete() -> Weight {
124+
// Proof Size summary in bytes:
125+
// Measured: `171`
126+
// Estimated: `3731`
127+
// Minimum execution time: 12_930_000 picoseconds.
128+
Weight::from_parts(13_272_000, 0)
129+
.saturating_add(Weight::from_parts(0, 3731))
130+
.saturating_add(T::DbWeight::get().reads(2))
131+
.saturating_add(T::DbWeight::get().writes(1))
132+
}
133+
/// Storage: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
134+
/// Proof: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
135+
/// Storage: `MultiBlockMigrations::Historic` (r:1 w:0)
136+
/// Proof: `MultiBlockMigrations::Historic` (`max_values`: None, `max_size`: Some(266), added: 2741, mode: `MaxEncodedLen`)
137+
/// Storage: `MultiBlockMigrations::Cursor` (r:0 w:1)
138+
/// Proof: `MultiBlockMigrations::Cursor` (`max_values`: Some(1), `max_size`: Some(65550), added: 66045, mode: `MaxEncodedLen`)
139+
fn exec_migration_fail() -> Weight {
140+
// Proof Size summary in bytes:
141+
// Measured: `171`
142+
// Estimated: `3731`
143+
// Minimum execution time: 13_709_000 picoseconds.
144+
Weight::from_parts(14_123_000, 0)
145+
.saturating_add(Weight::from_parts(0, 3731))
146+
.saturating_add(T::DbWeight::get().reads(2))
147+
.saturating_add(T::DbWeight::get().writes(1))
148+
}
149+
fn on_init_loop() -> Weight {
150+
// Proof Size summary in bytes:
151+
// Measured: `0`
152+
// Estimated: `0`
153+
// Minimum execution time: 162_000 picoseconds.
154+
Weight::from_parts(188_000, 0)
155+
.saturating_add(Weight::from_parts(0, 0))
156+
}
157+
/// Storage: `MultiBlockMigrations::Cursor` (r:0 w:1)
158+
/// Proof: `MultiBlockMigrations::Cursor` (`max_values`: Some(1), `max_size`: Some(65550), added: 66045, mode: `MaxEncodedLen`)
159+
fn force_set_cursor() -> Weight {
160+
// Proof Size summary in bytes:
161+
// Measured: `0`
162+
// Estimated: `0`
163+
// Minimum execution time: 2_737_000 picoseconds.
164+
Weight::from_parts(2_919_000, 0)
165+
.saturating_add(Weight::from_parts(0, 0))
166+
.saturating_add(T::DbWeight::get().writes(1))
167+
}
168+
/// Storage: `MultiBlockMigrations::Cursor` (r:0 w:1)
169+
/// Proof: `MultiBlockMigrations::Cursor` (`max_values`: Some(1), `max_size`: Some(65550), added: 66045, mode: `MaxEncodedLen`)
170+
fn force_set_active_cursor() -> Weight {
171+
// Proof Size summary in bytes:
172+
// Measured: `0`
173+
// Estimated: `0`
174+
// Minimum execution time: 3_087_000 picoseconds.
175+
Weight::from_parts(3_320_000, 0)
176+
.saturating_add(Weight::from_parts(0, 0))
177+
.saturating_add(T::DbWeight::get().writes(1))
178+
}
179+
/// Storage: `MultiBlockMigrations::Cursor` (r:1 w:0)
180+
/// Proof: `MultiBlockMigrations::Cursor` (`max_values`: Some(1), `max_size`: Some(65550), added: 66045, mode: `MaxEncodedLen`)
181+
/// Storage: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
182+
/// Proof: UNKNOWN KEY `0x583359fe0e84d953a9dd84e8addb08a5` (r:1 w:0)
183+
fn force_onboard_mbms() -> Weight {
184+
// Proof Size summary in bytes:
185+
// Measured: `147`
186+
// Estimated: `67035`
187+
// Minimum execution time: 6_470_000 picoseconds.
188+
Weight::from_parts(6_760_000, 0)
189+
.saturating_add(Weight::from_parts(0, 67035))
190+
.saturating_add(T::DbWeight::get().reads(2))
191+
}
192+
/// Storage: `MultiBlockMigrations::Historic` (r:256 w:256)
193+
/// Proof: `MultiBlockMigrations::Historic` (`max_values`: None, `max_size`: Some(266), added: 2741, mode: `MaxEncodedLen`)
194+
/// The range of component `n` is `[0, 256]`.
195+
fn clear_historic(n: u32, ) -> Weight {
196+
// Proof Size summary in bytes:
197+
// Measured: `1022 + n * (271 ±0)`
198+
// Estimated: `3834 + n * (2740 ±0)`
199+
// Minimum execution time: 15_864_000 picoseconds.
200+
Weight::from_parts(24_535_162, 0)
201+
.saturating_add(Weight::from_parts(0, 3834))
202+
// Standard Error: 8_688
203+
.saturating_add(Weight::from_parts(1_530_542, 0).saturating_mul(n.into()))
204+
.saturating_add(T::DbWeight::get().reads(1))
205+
.saturating_add(T::DbWeight::get().reads((1_u64).saturating_mul(n.into())))
206+
.saturating_add(T::DbWeight::get().writes((1_u64).saturating_mul(n.into())))
207+
.saturating_add(Weight::from_parts(0, 2740).saturating_mul(n.into()))
208+
}
209+
/// Storage: `Skipped::Metadata` (r:0 w:0)
210+
/// Proof: `Skipped::Metadata` (`max_values`: None, `max_size`: None, mode: `Measured`)
211+
/// The range of component `n` is `[0, 2048]`.
212+
fn reset_pallet_migration(n: u32, ) -> Weight {
213+
// Proof Size summary in bytes:
214+
// Measured: `1680 + n * (38 ±0)`
215+
// Estimated: `758 + n * (39 ±0)`
216+
// Minimum execution time: 2_168_000 picoseconds.
217+
Weight::from_parts(2_226_000, 0)
218+
.saturating_add(Weight::from_parts(0, 758))
219+
// Standard Error: 2_841
220+
.saturating_add(Weight::from_parts(935_438, 0).saturating_mul(n.into()))
221+
.saturating_add(T::DbWeight::get().reads((1_u64).saturating_mul(n.into())))
222+
.saturating_add(T::DbWeight::get().writes((1_u64).saturating_mul(n.into())))
223+
.saturating_add(Weight::from_parts(0, 39).saturating_mul(n.into()))
224+
}
225+
}

0 commit comments

Comments
 (0)