sui_protocol_config/
lib.rs

1// Copyright (c) Mysten Labs, Inc.
2// SPDX-License-Identifier: Apache-2.0
3
4use std::{
5    collections::{BTreeMap, BTreeSet},
6    sync::{
7        Arc, LazyLock,
8        atomic::{AtomicBool, Ordering},
9    },
10};
11
12#[cfg(msim)]
13use std::cell::RefCell;
14#[cfg(not(msim))]
15use std::sync::Mutex;
16
17use clap::*;
18use fastcrypto::encoding::{Base58, Encoding, Hex};
19use move_binary_format::{
20    binary_config::{BinaryConfig, TableConfig},
21    file_format_common::VERSION_1,
22};
23use move_core_types::account_address::AccountAddress;
24use move_vm_config::verifier::VerifierConfig;
25use mysten_common::in_integration_test;
26use serde::{Deserialize, Serialize};
27use serde_with::skip_serializing_none;
28use sui_protocol_config_macros::{
29    ProtocolConfigAccessors, ProtocolConfigFeatureFlagsGetters, ProtocolConfigOverride,
30};
31use tracing::{info, warn};
32
33/// The minimum and maximum protocol versions supported by this build.
34const MIN_PROTOCOL_VERSION: u64 = 1;
35const MAX_PROTOCOL_VERSION: u64 = 128;
36
37const TESTNET_USDC: &str =
38    "0xa1ec7fc00a6f40db9693ad1415d0c193ad3906494428cf252621037bd7117e29::usdc::USDC";
39
40const MAINNET_USDC: &str =
41    "0xdba34672e30cb065b1f93e3ab55318768fd6fef66c15942c9f7cb846e2f900e7::usdc::USDC";
42const MAINNET_USDSUI: &str =
43    "0x44f838219cf67b058f3b37907b655f226153c18e33dfcd0da559a844fea9b1c1::usdsui::USDSUI";
44const MAINNET_SUI_USDE: &str =
45    "0x41d587e5336f1c86cad50d38a7136db99333bb9bda91cea4ba69115defeb1402::sui_usde::SUI_USDE";
46const MAINNET_USDY: &str =
47    "0x960b531667636f39e85867775f52f6b1f220a058c4de786905bdf761e06a56bb::usdy::USDY";
48const MAINNET_FDUSD: &str =
49    "0xf16e6b723f242ec745dfd7634ad072c42d5c1d9ac9d62a39c381303eaa57693a::fdusd::FDUSD";
50const MAINNET_AUSD: &str =
51    "0x2053d08c1e2bd02791056171aab0fd12bd7cd7efad2ab8f6b9c8902f14df2ff2::ausd::AUSD";
52const MAINNET_USDB: &str =
53    "0xe14726c336e81b32328e92afc37345d159f5b550b09fa92bd43640cfdd0a0cfd::usdb::USDB";
54
55// Record history of protocol version allocations here:
56//
57// Version 1: Original version.
58// Version 2: Framework changes, including advancing epoch_start_time in safemode.
59// Version 3: gas model v2, including all sui conservation fixes. Fix for loaded child object
60//            changes, enable package upgrades, add limits on `max_size_written_objects`,
61//            `max_size_written_objects_system_tx`
62// Version 4: New reward slashing rate. Framework changes to skip stake susbidy when the epoch
63//            length is short.
64// Version 5: Package upgrade compatibility error fix. New gas cost table. New scoring decision
65//            mechanism that includes up to f scoring authorities.
66// Version 6: Change to how bytes are charged in the gas meter, increase buffer stake to 0.5f
67// Version 7: Disallow adding new abilities to types during package upgrades,
68//            disable_invariant_violation_check_in_swap_loc,
69//            disable init functions becoming entry,
70//            hash module bytes individually before computing package digest.
71// Version 8: Disallow changing abilities and type constraints for type parameters in structs
72//            during upgrades.
73// Version 9: Limit the length of Move idenfitiers to 128.
74//            Disallow extraneous module bytes,
75//            advance_to_highest_supported_protocol_version,
76// Version 10:increase bytecode verifier `max_verifier_meter_ticks_per_function` and
77//            `max_meter_ticks_per_module` limits each from 6_000_000 to 16_000_000. sui-system
78//            framework changes.
79// Version 11: Introduce `std::type_name::get_with_original_ids` to the system frameworks. Bound max depth of values within the VM.
80// Version 12: Changes to deepbook in framework to add API for querying marketplace.
81//             Change NW Batch to use versioned metadata field.
82//             Changes to sui-system package to add PTB-friendly unstake function, and minor cleanup.
83// Version 13: System package change deprecating `0xdee9::clob` and `0xdee9::custodian`, replaced by
84//             `0xdee9::clob_v2` and `0xdee9::custodian_v2`.
85// Version 14: Introduce a config variable to allow charging of computation to be either
86//             bucket base or rounding up. The presence of `gas_rounding_step` (or `None`)
87//             decides whether rounding is applied or not.
88// Version 15: Add reordering of user transactions by gas price after consensus.
89//             Add `sui::table_vec::drop` to the framework via a system package upgrade.
90// Version 16: Enabled simplified_unwrap_then_delete feature flag, which allows the execution engine
91//             to no longer consult the object store when generating unwrapped_then_deleted in the
92//             effects; this also allows us to stop including wrapped tombstones in accumulator.
93//             Add self-matching prevention for deepbook.
94// Version 17: Enable upgraded multisig support.
95// Version 18: Introduce execution layer versioning, preserve all existing behaviour in v0.
96//             Gas minimum charges moved to be a multiplier over the reference gas price. In this
97//             protocol version the multiplier is the same as the lowest bucket of computation
98//             such that the minimum transaction cost is the same as the minimum computation
99//             bucket.
100//             Add a feature flag to indicate the changes semantics of `base_tx_cost_fixed`.
101// Version 19: Changes to sui-system package to enable liquid staking.
102//             Add limit for total size of events.
103//             Increase limit for number of events emitted to 1024.
104// Version 20: Enables the flag `narwhal_new_leader_election_schedule` for the new narwhal leader
105//             schedule algorithm for enhanced fault tolerance and sets the bad node stake threshold
106//             value. Both values are set for all the environments except mainnet.
107// Version 21: ZKLogin known providers.
108// Version 22: Child object format change.
109// Version 23: Enabling the flag `narwhal_new_leader_election_schedule` for the new narwhal leader
110//             schedule algorithm for enhanced fault tolerance and sets the bad node stake threshold
111//             value for mainnet.
112// Version 24: Re-enable simple gas conservation checks.
113//             Package publish/upgrade number in a single transaction limited.
114//             JWK / authenticator state flags.
115// Version 25: Add sui::table_vec::swap and sui::table_vec::swap_remove to system packages.
116// Version 26: New gas model version.
117//             Add support for receiving objects off of other objects in devnet only.
118// Version 28: Add sui::zklogin::verify_zklogin_id and related functions to sui framework.
119//             Enable transaction effects v2 in devnet.
120// Version 29: Add verify_legacy_zklogin_address flag to sui framework, this add ability to verify
121//             transactions from a legacy zklogin address.
122// Version 30: Enable Narwhal CertificateV2
123//             Add support for random beacon.
124//             Enable transaction effects v2 in testnet.
125//             Deprecate supported oauth providers from protocol config and rely on node config
126//             instead.
127//             In execution, has_public_transfer is recomputed when loading the object.
128//             Add support for shared obj deletion and receiving objects off of other objects in devnet only.
129// Version 31: Add support for shared object deletion in devnet only.
130//             Add support for getting object ID referenced by receiving object in sui framework.
131//             Create new execution layer version, and preserve previous behavior in v1.
132//             Update semantics of `sui::transfer::receive` and add `sui::transfer::public_receive`.
133// Version 32: Add delete functions for VerifiedID and VerifiedIssuer.
134//             Add sui::token module to sui framework.
135//             Enable transfer to object in testnet.
136//             Enable Narwhal CertificateV2 on mainnet
137//             Make critbit tree and order getters public in deepbook.
138// Version 33: Add support for `receiving_object_id` function in framework
139//             Hardened OTW check.
140//             Enable transfer-to-object in mainnet.
141//             Enable shared object deletion in testnet.
142//             Enable effects v2 in mainnet.
143// Version 34: Framework changes for random beacon.
144// Version 35: Add poseidon hash function.
145//             Enable coin deny list.
146// Version 36: Enable group operations native functions in devnet.
147//             Enable shared object deletion in mainnet.
148//             Set the consensus accepted transaction size and the included transactions size in the proposed block.
149// Version 37: Reject entry functions with mutable Random.
150// Version 38: Introduce limits for binary tables size.
151// Version 39: Allow skipped epochs for randomness updates.
152//             Extra version to fix `test_upgrade_compatibility` simtest.
153// Version 40:
154// Version 41: Enable group operations native functions in testnet and mainnet (without msm).
155// Version 42: Migrate sui framework and related code to Move 2024
156// Version 43: Introduce the upper bound delta config for a zklogin signature's max epoch.
157//             Introduce an explicit parameter for the tick limit per package (previously this was
158//             represented by the parameter for the tick limit per module).
159// Version 44: Enable consensus fork detection on mainnet.
160//             Switch between Narwhal and Mysticeti consensus in tests, devnet and testnet.
161// Version 45: Use tonic networking for Mysticeti consensus.
162//             Set min Move binary format version to 6.
163//             Enable transactions to be signed with zkLogin inside multisig signature.
164//             Add native bridge.
165//             Enable native bridge in devnet
166//             Enable Leader Scoring & Schedule Change for Mysticeti consensus on testnet.
167// Version 46: Enable native bridge in testnet
168//             Enable resharing at the same initial shared version.
169// Version 47: Deepbook changes (framework update)
170// Version 48: Use tonic networking for Mysticeti.
171//             Resolve Move abort locations to the package id instead of the runtime module ID.
172//             Enable random beacon in testnet.
173//             Use new VM when verifying framework packages.
174// Version 49: Enable Move enums on devnet.
175//             Enable VDF in devnet
176//             Enable consensus commit prologue V3 in devnet.
177//             Run Mysticeti consensus by default.
178// Version 50: Add update_node_url to native bridge,
179//             New Move stdlib integer modules
180//             Enable checkpoint batching in testnet.
181//             Prepose consensus commit prologue in checkpoints.
182//             Set number of leaders per round for Mysticeti commits.
183// Version 51: Switch to DKG V1.
184//             Enable deny list v2 on devnet.
185// Version 52: Emit `CommitteeMemberUrlUpdateEvent` when updating bridge node url.
186//             std::config native functions.
187//             Modified sui-system package to enable withdrawal of stake before it becomes active.
188//             Enable soft bundle in devnet and testnet.
189//             Core macro visibility in sui core framework.
190//             Enable checkpoint batching in mainnet.
191//             Enable Mysticeti on mainnet.
192//             Enable Leader Scoring & Schedule Change for Mysticeti consensus on mainnet.
193//             Turn on count based shared object congestion control in devnet.
194//             Enable consensus commit prologue V3 in testnet.
195//             Enable enums on testnet.
196//             Add support for passkey in devnet.
197//             Enable deny list v2 on testnet and mainnet.
198// Version 53: Add feature flag to decide whether to attempt to finalize bridge committee
199//             Enable consensus commit prologue V3 on testnet.
200//             Turn on shared object congestion control in testnet.
201//             Update stdlib natives costs
202// Version 54: Enable random beacon on mainnet.
203//             Enable soft bundle on mainnet.
204// Version 55: Enable enums on mainnet.
205//             Rethrow serialization type layout errors instead of converting them.
206// Version 56: Enable bridge on mainnet.
207//             Note: do not use version 56 for any new features.
208// Version 57: Reduce minimum number of random beacon shares.
209// Version 58: Optimize boolean binops
210//             Finalize bridge committee on mainnet.
211//             Switch to distributed vote scoring in consensus in devnet
212// Version 59: Enable round prober in consensus.
213// Version 60: Validation of public inputs for Groth16 verification.
214//             Enable configuration of maximum number of type nodes in a type layout.
215// Version 61: Switch to distributed vote scoring in consensus in testnet
216//             Further reduce minimum number of random beacon shares.
217//             Add feature flag for Mysticeti fastpath.
218// Version 62: Makes the event's sending module package upgrade-aware.
219// Version 63: Enable gas based congestion control in consensus commit.
220// Version 64: Revert congestion control change.
221// Version 65: Enable distributed vote scoring in mainnet.
222// Version 66: Revert distributed vote scoring in mainnet.
223//             Framework fix for fungible staking book-keeping.
224// Version 67: Re-enable distributed vote scoring in mainnet.
225// Version 68: Add G1Uncompressed group to group ops.
226//             Update to Move stdlib.
227//             Enable gas based congestion control with overage.
228//             Further reduce minimum number of random beacon shares.
229//             Disallow adding new modules in `deps-only` packages.
230// Version 69: Sets number of rounds allowed for fastpath voting in consensus.
231//             Enable smart ancestor selection in devnet.
232//             Enable G1Uncompressed group in testnet.
233// Version 70: Enable smart ancestor selection in testnet.
234//             Enable probing for accepted rounds in round prober in testnet
235//             Add new gas model version to update charging of native functions.
236//             Add std::uq64_64 module to Move stdlib.
237//             Improve gas/wall time efficiency of some Move stdlib vector functions
238// Version 71: [SIP-45] Enable consensus amplification.
239// Version 72: Fix issue where `convert_type_argument_error` wasn't being used in all cases.
240//             Max gas budget moved to 50_000 SUI
241//             Max gas price moved to 50 SUI
242//             Variants as type nodes.
243// Version 73: Enable new marker table version.
244//             Enable consensus garbage collection and new commit rule for devnet.
245//             Enable zstd compression for consensus tonic network in testnet.
246//             Enable smart ancestor selection in mainnet.
247//             Enable probing for accepted rounds in round prober in mainnet
248// Version 74: Enable load_nitro_attestation move function in sui framework in devnet.
249//             Enable all gas costs for load_nitro_attestation.
250//             Enable zstd compression for consensus tonic network in mainnet.
251//             Enable the new commit rule for devnet.
252// Version 75: Enable passkey auth in testnet.
253// Version 76: Deprecate Deepbook V2 order placement and deposit.
254//             Removes unnecessary child object mutations
255//             Enable passkey auth in multisig for testnet.
256// Version 77: Enable uncompressed point group ops on mainnet.
257//             Enable consensus garbage collection for testnet
258//             Enable the new consensus commit rule for testnet.
259// Version 78: Make `TxContext` Move API native
260//             Enable execution time estimate mode for congestion control on testnet.
261// Version 79: Enable median based commit timestamp in consensus on testnet.
262//             Increase threshold for bad nodes that won't be considered leaders in consensus in testnet
263//             Enable load_nitro_attestation move function in sui framework in testnet.
264//             Enable consensus garbage collection for mainnet
265//             Enable the new consensus commit rule for mainnet.
266// Version 80: Bound size of values created in the adapter.
267// Version 81: Enable median based commit timestamp in consensus on mainnet.
268//             Enforce checkpoint timestamps are non-decreasing for testnet and mainnet.
269//             Increase threshold for bad nodes that won't be considered leaders in consensus in mainnet
270// Version 82: Relax bounding of size of values created in the adapter.
271// Version 83: Resolve `TypeInput` IDs to defining ID when converting to `TypeTag`s in the adapter.
272//             Enable execution time estimate mode for congestion control on mainnet.
273//             Enable nitro attestation upgraded parsing and mainnet.
274// Version 84: Limit number of stored execution time observations between epochs.
275// Version 85: Enable party transfer in devnet.
276// Version 86: Use type tags in the object runtime and adapter instead of `Type`s.
277//             Make variant count limit explicit in protocol config.
278//             Enable party transfer in testnet.
279// Version 87: Enable better type resolution errors in the adapter.
280// Version 88: Update `sui-system` package to use `calculate_rewards` function.
281//             Define the cost for the native Move function `rgp`.
282//             Ignore execution time observations after validator stops accepting certs.
283// Version 89: Add additional signature checks
284//             Add additional linkage checks
285// Version 90: Standard library improvements.
286//             Enable `debug_fatal` on Move invariant violations.
287//             Enable passkey and passkey inside multisig for mainnet.
288// Version 91: Minor changes in Sui Framework. Include CheckpointDigest in consensus dedup key for checkpoint signatures (V2).
289// Version 92: Disable checking shared object transfer restrictions per command = false
290// Version 93: Enable CheckpointDigest in consensus dedup key for checkpoint signatures.
291// Version 94: Decrease stored observations limit by 10% to stay within system object size limit.
292//             Enable party transfer on mainnet.
293// Version 95: Change type name id base cost to 52, increase max transactions per checkpoint to 20000.
294// Version 96: Enable authority capabilities v2.
295//             Fix bug where MFP transaction shared inputs' debts were not loaded
296//             Create Coin Registry object
297//             Enable checkpoint artifacts digest in devnet.
298// Version 97: Enable additional borrow checks
299// Version 98: Add authenticated event streams support via emit_authenticated function.
300//             Add better error messages to the loader.
301// Version 99: Enable new commit handler.
302// Version 100: Framework update
303// Version 101: Framework update
304//              Set max updates per settlement txn to 100.
305// Version 103: Framework update: internal Coin methods
306// Version 104: Framework update: CoinRegistry follow up for Coin methods
307//              Enable all non-zero PCRs parsing for nitro attestation native function in Devnet and Testnet.
308// Version 105: Framework update: address aliases
309//              Enable multi-epoch transaction expiration.
310//              Enable always include required PCRs (0-4 & 8) parsing even if they are zeros for
311//              nitro attestation native function in Devnet and Testnet.
312// Version 106: Framework update: accumulator storage fund calculations
313//              Enable address balances on devnet
314// Version 108: Enable new digit based gas rounding.
315//              Support TxContext in all parameter positions.
316//              Disable entry point signature check.
317//              Enable address aliases on testnet.
318//              Enable poseidon_bn254 on mainnet.
319// Version 109: Update where we set bounds for some binary tables to be a bit more idiomatic.
320// Version 110: Enable parsing on all nonzero custom pcrs in nitro attestation parsing native
321//              function on mainnet.
322//              split_checkpoints_in_consensus_handler in devnet
323//              Enable additional validation on zkLogin public identifier.
324// Version 111: Validator metadata
325// Version 112: Enable Ristretto255 in devnet.
326// Version 113: Validate gas price >= RGP at signing for address balance gas payments.
327// Version 114: Gate seeded test overrides for checkpoint tx limit behind feature flag.
328// Version 115: Gasless transaction drop safety.
329//              Enable address aliases on mainnet.
330//              Relax ValidDuring requirement for transactions with owned inputs.
331// Version 116: Enable Display Registry.
332//              Disable defer_unpaid_amplification (debugging).
333// Version 117: Update Sui System metadata handling.
334// Version 118: Adds `transfer_migration_cap` to display registry
335// Version 119: Enable the new VM.
336// Version 120: Disallow unused jump tables
337// Version 121: Re-enable defer_unpaid_amplification (devnet + testnet).
338// Version 122: Framework update: vector::empty is deprecated.
339//              Enable bulletproofs verification on devnet.
340//              Enable defer_unpaid_amplification on mainnet.
341// Version 123: Gas accounting refresh (gas_model v13).
342// Version 124: Add timestamp_based_epoch_close feature flag and enable in tests.
343//              Fix native call double-pop in gas meter stack height tracking (gas_model v14).
344//              Limit public inputs in groth16::prepare_verifying_key.
345//              Enable address balances, free tier (gasless), and coin reservations on mainnet.
346//              Enables enable_accumulators, enable_address_balance_gas_payments,
347//              enable_authenticated_event_streams, enable_coin_reservation_obj_refs,
348//              enable_object_funds_withdraw, convert_withdrawal_compatibility_ptb_arguments,
349//              split_checkpoints_in_consensus_handler, include_checkpoint_artifacts_digest_in_summary,
350//              and enable_gasless on mainnet to bring it in line with testnet.
351//              Configure mainnet gasless allowlist with stablecoin types and $0.01 minimum
352//              transfer per stable.
353// Version 125: Enable granular_post_execution_checks.
354//              Enable timestamp_based_epoch_close on testnet.
355// Version 126: Enable early_exit_on_iffw (gates the gas-underflow fix
356//              shipped to mainnet out-of-band in #26816).
357// Version 127: Enable always_advance_dkg_to_resolution.
358//              Update gas prices for range proofs and ristretto group operations.
359//              Enable Ristretto255 group operations and bulletproofs verification on testnet.
360//              Enable timestamp_based_epoch_close on mainnet.
361// Version 128: Add `insert_before` and `insert_after` to `sui::linked_table`
362
363#[derive(Copy, Clone, Debug, Hash, Serialize, Deserialize, PartialEq, Eq, PartialOrd, Ord)]
364pub struct ProtocolVersion(u64);
365
366impl ProtocolVersion {
367    // The minimum and maximum protocol version supported by this binary. Counterintuitively, this constant may
368    // change over time as support for old protocol versions is removed from the source. This
369    // ensures that when a new network (such as a testnet) is created, its genesis committee will
370    // use a protocol version that is actually supported by the binary.
371    pub const MIN: Self = Self(MIN_PROTOCOL_VERSION);
372
373    pub const MAX: Self = Self(MAX_PROTOCOL_VERSION);
374
375    #[cfg(not(msim))]
376    pub const MAX_ALLOWED: Self = Self::MAX;
377
378    // We create one additional "fake" version in simulator builds so that we can test upgrades.
379    #[cfg(msim)]
380    pub const MAX_ALLOWED: Self = Self(MAX_PROTOCOL_VERSION + 1);
381
382    pub fn new(v: u64) -> Self {
383        Self(v)
384    }
385
386    pub const fn as_u64(&self) -> u64 {
387        self.0
388    }
389
390    // For serde deserialization - we don't define a Default impl because there isn't a single
391    // universally appropriate default value.
392    pub fn max() -> Self {
393        Self::MAX
394    }
395
396    pub fn prev(self) -> Self {
397        Self(self.0.checked_sub(1).unwrap())
398    }
399}
400
401impl From<u64> for ProtocolVersion {
402    fn from(v: u64) -> Self {
403        Self::new(v)
404    }
405}
406
407impl std::ops::Sub<u64> for ProtocolVersion {
408    type Output = Self;
409    fn sub(self, rhs: u64) -> Self::Output {
410        Self::new(self.0 - rhs)
411    }
412}
413
414impl std::ops::Add<u64> for ProtocolVersion {
415    type Output = Self;
416    fn add(self, rhs: u64) -> Self::Output {
417        Self::new(self.0 + rhs)
418    }
419}
420
421#[derive(
422    Clone, Serialize, Deserialize, Debug, Default, PartialEq, Copy, PartialOrd, Ord, Eq, ValueEnum,
423)]
424pub enum Chain {
425    Mainnet,
426    Testnet,
427    #[default]
428    Unknown,
429}
430
431impl Chain {
432    pub fn as_str(self) -> &'static str {
433        match self {
434            Chain::Mainnet => "mainnet",
435            Chain::Testnet => "testnet",
436            Chain::Unknown => "unknown",
437        }
438    }
439}
440
441pub struct Error(pub String);
442
443// TODO: There are quite a few non boolean values in the feature flags. We should move them out.
444/// Records on/off feature flags that may vary at each protocol version.
445#[derive(Default, Clone, Serialize, Deserialize, Debug, ProtocolConfigFeatureFlagsGetters)]
446struct FeatureFlags {
447    // Add feature flags here, e.g.:
448    // new_protocol_feature: bool,
449    #[serde(skip_serializing_if = "is_false")]
450    package_upgrades: bool,
451    // If true, validators will commit to the root state digest
452    // in end of epoch checkpoint proposals
453    #[serde(skip_serializing_if = "is_false")]
454    commit_root_state_digest: bool,
455    // Pass epoch start time to advance_epoch safe mode function.
456    #[serde(skip_serializing_if = "is_false")]
457    advance_epoch_start_time_in_safe_mode: bool,
458    // If true, apply the fix to correctly capturing loaded child object versions in execution's
459    // object runtime.
460    #[serde(skip_serializing_if = "is_false")]
461    loaded_child_objects_fixed: bool,
462    // If true, treat missing types in the upgraded modules when creating an upgraded package as a
463    // compatibility error.
464    #[serde(skip_serializing_if = "is_false")]
465    missing_type_is_compatibility_error: bool,
466    // If true, then the scoring decision mechanism will not get disabled when we do have more than
467    // f low scoring authorities, but it will simply flag as low scoring only up to f authorities.
468    #[serde(skip_serializing_if = "is_false")]
469    scoring_decision_with_validity_cutoff: bool,
470
471    // DEPRECATED: this was an ephemeral feature flag only used by consensus handler, which has now
472    // been deployed everywhere.
473    #[serde(skip_serializing_if = "is_false")]
474    consensus_order_end_of_epoch_last: bool,
475
476    // Disallow adding abilities to types during package upgrades.
477    #[serde(skip_serializing_if = "is_false")]
478    disallow_adding_abilities_on_upgrade: bool,
479    // Disables unnecessary invariant check in the Move VM when swapping the value out of a local
480    #[serde(skip_serializing_if = "is_false")]
481    disable_invariant_violation_check_in_swap_loc: bool,
482    // advance to highest supported protocol version at epoch change, instead of the next consecutive
483    // protocol version.
484    #[serde(skip_serializing_if = "is_false")]
485    advance_to_highest_supported_protocol_version: bool,
486    // If true, disallow entry modifiers on entry functions
487    #[serde(skip_serializing_if = "is_false")]
488    ban_entry_init: bool,
489    // If true, hash module bytes individually when calculating package digests for upgrades
490    #[serde(skip_serializing_if = "is_false")]
491    package_digest_hash_module: bool,
492    // If true, disallow changing struct type parameters during package upgrades
493    #[serde(skip_serializing_if = "is_false")]
494    disallow_change_struct_type_params_on_upgrade: bool,
495    // If true, checks no extra bytes in a compiled module
496    #[serde(skip_serializing_if = "is_false")]
497    no_extraneous_module_bytes: bool,
498    // If true, then use the versioned metadata format in narwhal entities.
499    #[serde(skip_serializing_if = "is_false")]
500    narwhal_versioned_metadata: bool,
501
502    // Enable zklogin auth
503    #[serde(skip_serializing_if = "is_false")]
504    zklogin_auth: bool,
505    // How we order transactions coming out of consensus before sending to execution.
506    #[serde(skip_serializing_if = "ConsensusTransactionOrdering::is_none")]
507    consensus_transaction_ordering: ConsensusTransactionOrdering,
508
509    // Previously, the unwrapped_then_deleted field in TransactionEffects makes a distinction between
510    // whether an object has existed in the store previously (i.e. whether there is a tombstone).
511    // Such dependency makes effects generation inefficient, and requires us to include wrapped
512    // tombstone in state root hash.
513    // To prepare for effects V2, with this flag set to true, we simplify the definition of
514    // unwrapped_then_deleted to always include unwrapped then deleted objects,
515    // regardless of their previous state in the store.
516    #[serde(skip_serializing_if = "is_false")]
517    simplified_unwrap_then_delete: bool,
518    // Enable upgraded multisig support
519    #[serde(skip_serializing_if = "is_false")]
520    upgraded_multisig_supported: bool,
521    // If true minimum txn charge is a multiplier of the gas price
522    #[serde(skip_serializing_if = "is_false")]
523    txn_base_cost_as_multiplier: bool,
524
525    // If true, the ability to delete shared objects is in effect
526    #[serde(skip_serializing_if = "is_false")]
527    shared_object_deletion: bool,
528
529    // If true, then the new algorithm for the leader election schedule will be used
530    #[serde(skip_serializing_if = "is_false")]
531    narwhal_new_leader_election_schedule: bool,
532
533    // A list of supported OIDC providers that can be used for zklogin.
534    #[serde(skip_serializing_if = "is_empty")]
535    zklogin_supported_providers: BTreeSet<String>,
536
537    // If true, use the new child object format
538    #[serde(skip_serializing_if = "is_false")]
539    loaded_child_object_format: bool,
540
541    #[serde(skip_serializing_if = "is_false")]
542    enable_jwk_consensus_updates: bool,
543
544    #[serde(skip_serializing_if = "is_false")]
545    end_of_epoch_transaction_supported: bool,
546
547    // Perform simple conservation checks keeping into account out of gas scenarios
548    // while charging for storage.
549    #[serde(skip_serializing_if = "is_false")]
550    simple_conservation_checks: bool,
551
552    // If true, use the new child object format type logging
553    #[serde(skip_serializing_if = "is_false")]
554    loaded_child_object_format_type: bool,
555
556    // Enable receiving sent objects
557    #[serde(skip_serializing_if = "is_false")]
558    receive_objects: bool,
559
560    // If true, include CheckpointDigest in consensus dedup key for checkpoint signatures (V2).
561    #[serde(skip_serializing_if = "is_false")]
562    consensus_checkpoint_signature_key_includes_digest: bool,
563
564    // Enable random beacon protocol
565    #[serde(skip_serializing_if = "is_false")]
566    random_beacon: bool,
567
568    // Enable bridge protocol
569    #[serde(skip_serializing_if = "is_false")]
570    bridge: bool,
571
572    #[serde(skip_serializing_if = "is_false")]
573    enable_effects_v2: bool,
574
575    // If true, then use CertificateV2 in narwhal.
576    #[serde(skip_serializing_if = "is_false")]
577    narwhal_certificate_v2: bool,
578
579    // If true, allow verify with legacy zklogin address
580    #[serde(skip_serializing_if = "is_false")]
581    verify_legacy_zklogin_address: bool,
582
583    // Enable throughput aware consensus submission
584    #[serde(skip_serializing_if = "is_false")]
585    throughput_aware_consensus_submission: bool,
586
587    // If true, recompute has_public_transfer from the type instead of what is stored in the object
588    #[serde(skip_serializing_if = "is_false")]
589    recompute_has_public_transfer_in_execution: bool,
590
591    // If true, multisig containing zkLogin sig is accepted.
592    #[serde(skip_serializing_if = "is_false")]
593    accept_zklogin_in_multisig: bool,
594
595    // If true, multisig containing passkey sig is accepted.
596    #[serde(skip_serializing_if = "is_false")]
597    accept_passkey_in_multisig: bool,
598
599    // If true, additional zkLogin public identifier structure is validated.
600    #[serde(skip_serializing_if = "is_false")]
601    validate_zklogin_public_identifier: bool,
602
603    // If true, consensus prologue transaction also includes the consensus output digest.
604    // It can be used to detect consensus output folk.
605    #[serde(skip_serializing_if = "is_false")]
606    include_consensus_digest_in_prologue: bool,
607
608    // If true, use the hardened OTW check
609    #[serde(skip_serializing_if = "is_false")]
610    hardened_otw_check: bool,
611
612    // If true allow calling receiving_object_id function
613    #[serde(skip_serializing_if = "is_false")]
614    allow_receiving_object_id: bool,
615
616    // Enable the poseidon hash function
617    #[serde(skip_serializing_if = "is_false")]
618    enable_poseidon: bool,
619
620    // If true, enable the coin deny list.
621    #[serde(skip_serializing_if = "is_false")]
622    enable_coin_deny_list: bool,
623
624    // Enable native functions for group operations.
625    #[serde(skip_serializing_if = "is_false")]
626    enable_group_ops_native_functions: bool,
627
628    // Enable native function for msm.
629    #[serde(skip_serializing_if = "is_false")]
630    enable_group_ops_native_function_msm: bool,
631
632    // Enable group operations for Ristretto255
633    #[serde(skip_serializing_if = "is_false")]
634    enable_ristretto255_group_ops: bool,
635
636    // Enable native functions for group operations.
637    #[serde(skip_serializing_if = "is_false")]
638    enable_verify_bulletproofs_ristretto255: bool,
639
640    // Enable nitro attestation.
641    #[serde(skip_serializing_if = "is_false")]
642    enable_nitro_attestation: bool,
643
644    // Enable upgraded parsing of nitro attestation that interprets pcrs as a map.
645    #[serde(skip_serializing_if = "is_false")]
646    enable_nitro_attestation_upgraded_parsing: bool,
647
648    // Enable upgraded parsing of nitro attestation containing all nonzero PCRs.
649    #[serde(skip_serializing_if = "is_false")]
650    enable_nitro_attestation_all_nonzero_pcrs_parsing: bool,
651
652    // Enable upgraded parsing of nitro attestation to always include required PCRs, even when all zeros.
653    #[serde(skip_serializing_if = "is_false")]
654    enable_nitro_attestation_always_include_required_pcrs_parsing: bool,
655
656    // Reject functions with mutable Random.
657    #[serde(skip_serializing_if = "is_false")]
658    reject_mutable_random_on_entry_functions: bool,
659
660    // Controls the behavior of per object congestion control in consensus handler.
661    #[serde(skip_serializing_if = "PerObjectCongestionControlMode::is_none")]
662    per_object_congestion_control_mode: PerObjectCongestionControlMode,
663
664    // The consensus protocol to be used for the epoch.
665    #[serde(skip_serializing_if = "ConsensusChoice::is_narwhal")]
666    consensus_choice: ConsensusChoice,
667
668    // Consensus network to use.
669    #[serde(skip_serializing_if = "ConsensusNetwork::is_anemo")]
670    consensus_network: ConsensusNetwork,
671
672    // If true, use the correct (<=) comparison for max_gas_payment_objects instead of (<)
673    #[serde(skip_serializing_if = "is_false")]
674    correct_gas_payment_limit_check: bool,
675
676    // Set the upper bound allowed for max_epoch in zklogin signature.
677    #[serde(skip_serializing_if = "Option::is_none")]
678    zklogin_max_epoch_upper_bound_delta: Option<u64>,
679
680    // Controls leader scoring & schedule change in Mysticeti consensus.
681    #[serde(skip_serializing_if = "is_false")]
682    mysticeti_leader_scoring_and_schedule: bool,
683
684    // Enable resharing of shared objects using the same initial shared version
685    #[serde(skip_serializing_if = "is_false")]
686    reshare_at_same_initial_version: bool,
687
688    // Resolve Move abort locations to the package id instead of the runtime module ID.
689    #[serde(skip_serializing_if = "is_false")]
690    resolve_abort_locations_to_package_id: bool,
691
692    // Enables the use of the Mysticeti committed sub dag digest to the `ConsensusCommitInfo` in checkpoints.
693    // When disabled the default digest is used instead. It's important to have this guarded behind
694    // a flag as it will lead to checkpoint forks.
695    #[serde(skip_serializing_if = "is_false")]
696    mysticeti_use_committed_subdag_digest: bool,
697
698    // Enable VDF
699    #[serde(skip_serializing_if = "is_false")]
700    enable_vdf: bool,
701
702    // Controls whether consensus handler should record consensus determined shared object version
703    // assignments in consensus commit prologue transaction.
704    // The purpose of doing this is to enable replaying transaction without transaction effects.
705    // V2 also records initial shared versions for consensus objects.
706    #[serde(skip_serializing_if = "is_false")]
707    record_consensus_determined_version_assignments_in_prologue: bool,
708    #[serde(skip_serializing_if = "is_false")]
709    record_consensus_determined_version_assignments_in_prologue_v2: bool,
710
711    // Run verification of framework upgrades using a new/fresh VM.
712    #[serde(skip_serializing_if = "is_false")]
713    fresh_vm_on_framework_upgrade: bool,
714
715    // When set to true, the consensus commit prologue transaction will be placed first
716    // in a consensus commit in checkpoints.
717    // If a checkpoint contains multiple consensus commit, say [cm1][cm2]. The each commit's
718    // consensus commit prologue will be the first transaction in each segment:
719    //     [ccp1, rest cm1][ccp2, rest cm2]
720    // The reason to prepose the prologue transaction is to provide information for transaction
721    // cancellation.
722    #[serde(skip_serializing_if = "is_false")]
723    prepend_prologue_tx_in_consensus_commit_in_checkpoints: bool,
724
725    // Set number of leaders per round for Mysticeti commits.
726    #[serde(skip_serializing_if = "Option::is_none")]
727    mysticeti_num_leaders_per_round: Option<usize>,
728
729    // Enable Soft Bundle (SIP-19).
730    #[serde(skip_serializing_if = "is_false")]
731    soft_bundle: bool,
732
733    // If true, enable the coin deny list V2.
734    #[serde(skip_serializing_if = "is_false")]
735    enable_coin_deny_list_v2: bool,
736
737    // Enable passkey auth (SIP-9)
738    #[serde(skip_serializing_if = "is_false")]
739    passkey_auth: bool,
740
741    // Use AuthorityCapabilitiesV2
742    #[serde(skip_serializing_if = "is_false")]
743    authority_capabilities_v2: bool,
744
745    // Rethrow type layout errors during serialization instead of trying to convert them.
746    #[serde(skip_serializing_if = "is_false")]
747    rethrow_serialization_type_layout_errors: bool,
748
749    // Use distributed vote leader scoring strategy in consensus.
750    #[serde(skip_serializing_if = "is_false")]
751    consensus_distributed_vote_scoring_strategy: bool,
752
753    // Probe rounds received by peers from every authority.
754    #[serde(skip_serializing_if = "is_false")]
755    consensus_round_prober: bool,
756
757    // Validate identifier inputs separately
758    #[serde(skip_serializing_if = "is_false")]
759    validate_identifier_inputs: bool,
760
761    // Disallow self identifier
762    #[serde(skip_serializing_if = "is_false")]
763    disallow_self_identifier: bool,
764
765    // Enables Mysticeti fastpath.
766    #[serde(skip_serializing_if = "is_false")]
767    mysticeti_fastpath: bool,
768
769    // If true, disable pre-consensus locking for owned objects.
770    // All transactions go through consensus, and owned object conflict detection
771    // happens post-consensus via lock acquisition.
772    #[serde(skip_serializing_if = "is_false")]
773    disable_preconsensus_locking: bool,
774
775    // Makes the event's sending module version-aware.
776    #[serde(skip_serializing_if = "is_false")]
777    relocate_event_module: bool,
778
779    // Enable uncompressed group elements in BLS123-81 G1
780    #[serde(skip_serializing_if = "is_false")]
781    uncompressed_g1_group_elements: bool,
782
783    #[serde(skip_serializing_if = "is_false")]
784    disallow_new_modules_in_deps_only_packages: bool,
785
786    // Use smart ancestor selection in consensus.
787    #[serde(skip_serializing_if = "is_false")]
788    consensus_smart_ancestor_selection: bool,
789
790    // Probe accepted rounds in round prober.
791    #[serde(skip_serializing_if = "is_false")]
792    consensus_round_prober_probe_accepted_rounds: bool,
793
794    // Enable v2 native charging for natives.
795    #[serde(skip_serializing_if = "is_false")]
796    native_charging_v2: bool,
797
798    // Enables the new logic for collecting the subdag in the consensus linearizer. The new logic does not stop the recursion at the highest
799    // committed round for each authority, but allows to commit uncommitted blocks up to gc round (excluded) for that authority.
800    #[serde(skip_serializing_if = "is_false")]
801    consensus_linearize_subdag_v2: bool,
802
803    // Properly convert certain type argument errors in the execution layer.
804    #[serde(skip_serializing_if = "is_false")]
805    convert_type_argument_error: bool,
806
807    // Variants count as nodes
808    #[serde(skip_serializing_if = "is_false")]
809    variant_nodes: bool,
810
811    // If true, enable zstd compression for consensus tonic network.
812    #[serde(skip_serializing_if = "is_false")]
813    consensus_zstd_compression: bool,
814
815    // If true, enables the optimizations for child object mutations, removing unnecessary mutations
816    #[serde(skip_serializing_if = "is_false")]
817    minimize_child_object_mutations: bool,
818
819    // If true, record the additional state digest in the consensus commit prologue.
820    #[serde(skip_serializing_if = "is_false")]
821    record_additional_state_digest_in_prologue: bool,
822
823    // If true, enable `TxContext` Move API to go native.
824    #[serde(skip_serializing_if = "is_false")]
825    move_native_context: bool,
826
827    // If true, then it (1) will not enforce monotonicity checks for a block's ancestors and (2) calculates the commit's timestamp based on the
828    // weighted by stake median timestamp of the leader's ancestors.
829    #[serde(skip_serializing_if = "is_false")]
830    consensus_median_based_commit_timestamp: bool,
831
832    // If true, enables the normalization of PTB arguments but does not yet enable splatting
833    // `Result`s of length not equal to 1
834    #[serde(skip_serializing_if = "is_false")]
835    normalize_ptb_arguments: bool,
836
837    // If true, enabled batched block sync in consensus.
838    #[serde(skip_serializing_if = "is_false")]
839    consensus_batched_block_sync: bool,
840
841    // If true, enforces checkpoint timestamps are non-decreasing.
842    #[serde(skip_serializing_if = "is_false")]
843    enforce_checkpoint_timestamp_monotonicity: bool,
844
845    // If true, enables better errors and bounds for max ptb values
846    #[serde(skip_serializing_if = "is_false")]
847    max_ptb_value_size_v2: bool,
848
849    // If true, resolves all type input ids to be defining ID based in the adapter
850    #[serde(skip_serializing_if = "is_false")]
851    resolve_type_input_ids_to_defining_id: bool,
852
853    // Enable native function for party transfer
854    #[serde(skip_serializing_if = "is_false")]
855    enable_party_transfer: bool,
856
857    // Allow objects created or mutated in system transactions to exceed the max object size limit.
858    #[serde(skip_serializing_if = "is_false")]
859    allow_unbounded_system_objects: bool,
860
861    // Signifies the cut-over of using type tags instead of `Type`s in the object runtime.
862    #[serde(skip_serializing_if = "is_false")]
863    type_tags_in_object_runtime: bool,
864
865    // Enable accumulators
866    #[serde(skip_serializing_if = "is_false")]
867    enable_accumulators: bool,
868
869    // Enable coin reservation
870    #[serde(skip_serializing_if = "is_false")]
871    enable_coin_reservation_obj_refs: bool,
872
873    // If true, create the root accumulator object in the change epoch transaction.
874    // This must be enabled and shipped before `enable_accumulators` is set to true.
875    #[serde(skip_serializing_if = "is_false")]
876    create_root_accumulator_object: bool,
877
878    // Enable authenticated event streams
879    #[serde(skip_serializing_if = "is_false")]
880    enable_authenticated_event_streams: bool,
881
882    // Enable address balance gas payments
883    #[serde(skip_serializing_if = "is_false")]
884    enable_address_balance_gas_payments: bool,
885
886    // Validate gas price >= RGP at signing for address balance gas payments
887    #[serde(skip_serializing_if = "is_false")]
888    address_balance_gas_check_rgp_at_signing: bool,
889
890    #[serde(skip_serializing_if = "is_false")]
891    address_balance_gas_reject_gas_coin_arg: bool,
892
893    // Enable multi-epoch transaction expiration (max 1 epoch difference)
894    #[serde(skip_serializing_if = "is_false")]
895    enable_multi_epoch_transaction_expiration: bool,
896
897    // Relax ValidDuring expiration requirement for transactions with owned inputs
898    #[serde(skip_serializing_if = "is_false")]
899    relax_valid_during_for_owned_inputs: bool,
900
901    // Enable statically type checked ptb execution
902    #[serde(skip_serializing_if = "is_false")]
903    enable_ptb_execution_v2: bool,
904
905    // Provide better type resolution errors in the adapter.
906    #[serde(skip_serializing_if = "is_false")]
907    better_adapter_type_resolution_errors: bool,
908
909    // If true, record the time estimate processed in the consensus commit prologue.
910    #[serde(skip_serializing_if = "is_false")]
911    record_time_estimate_processed: bool,
912
913    // If true enable additional linkage checks.
914    #[serde(skip_serializing_if = "is_false")]
915    dependency_linkage_error: bool,
916
917    // If true enable additional multisig checks.
918    #[serde(skip_serializing_if = "is_false")]
919    additional_multisig_checks: bool,
920
921    // If true, ignore execution time observations after certs are closed.
922    #[serde(skip_serializing_if = "is_false")]
923    ignore_execution_time_observations_after_certs_closed: bool,
924
925    // If true use `debug_fatal` to report invariant violations.
926    // `debug_fatal` panics in debug builds and breaks tests/behavior based on older
927    // protocol versions (see make_vec_non_existent_type_v71.move)
928    #[serde(skip_serializing_if = "is_false")]
929    debug_fatal_on_move_invariant_violation: bool,
930
931    // DO NOT ENABLE THIS FOR PRODUCTION NETWORKS. used for testing only.
932    // Allow private accumulator entrypoints
933    #[serde(skip_serializing_if = "is_false")]
934    allow_private_accumulator_entrypoints: bool,
935
936    // If true, include indirect state in the additional consensus digest.
937    #[serde(skip_serializing_if = "is_false")]
938    additional_consensus_digest_indirect_state: bool,
939
940    // Check for `init` for new modules to a package on upgrade.
941    #[serde(skip_serializing_if = "is_false")]
942    check_for_init_during_upgrade: bool,
943
944    // Check shared object transfer restrictions per command.
945    #[serde(skip_serializing_if = "is_false")]
946    per_command_shared_object_transfer_rules: bool,
947
948    // Enable including checkpoint artifacts digest in the summary.
949    #[serde(skip_serializing_if = "is_false")]
950    include_checkpoint_artifacts_digest_in_summary: bool,
951
952    // If true, use MFP txns in load initial object debts.
953    #[serde(skip_serializing_if = "is_false")]
954    use_mfp_txns_in_load_initial_object_debts: bool,
955
956    // If true, cancel randomness-using txns when DKG has failed *before* doing other congestion checks.
957    #[serde(skip_serializing_if = "is_false")]
958    cancel_for_failed_dkg_early: bool,
959
960    // If true, keep advancing the DKG state machine while DKG is pending.
961    #[serde(skip_serializing_if = "is_false")]
962    always_advance_dkg_to_resolution: bool,
963
964    // Enable coin registry protocol
965    #[serde(skip_serializing_if = "is_false")]
966    enable_coin_registry: bool,
967
968    // Use abstract size in the object runtime instead the legacy value size.
969    #[serde(skip_serializing_if = "is_false")]
970    abstract_size_in_object_runtime: bool,
971
972    // If true charge for loads into the cache (i.e., fetches from storage) in the object runtime.
973    #[serde(skip_serializing_if = "is_false")]
974    object_runtime_charge_cache_load_gas: bool,
975
976    // If true, perform additional borrow checks
977    #[serde(skip_serializing_if = "is_false")]
978    additional_borrow_checks: bool,
979
980    // If true, use the new commit handler.
981    #[serde(skip_serializing_if = "is_false")]
982    use_new_commit_handler: bool,
983
984    // If true return a better error message when we encounter a loader error.
985    #[serde(skip_serializing_if = "is_false")]
986    better_loader_errors: bool,
987
988    // If true generate layouts for dynamic fields
989    #[serde(skip_serializing_if = "is_false")]
990    generate_df_type_layouts: bool,
991
992    // If true, allow Move functions called in PTBs to return references
993    #[serde(skip_serializing_if = "is_false")]
994    allow_references_in_ptbs: bool,
995
996    // Enable display registry protocol
997    #[serde(skip_serializing_if = "is_false")]
998    enable_display_registry: bool,
999
1000    // If true, enable private generics verifier v2
1001    #[serde(skip_serializing_if = "is_false")]
1002    private_generics_verifier_v2: bool,
1003
1004    // If true, deprecate global storage ops during Move module deserialization
1005    #[serde(skip_serializing_if = "is_false")]
1006    deprecate_global_storage_ops_during_deserialization: bool,
1007
1008    // If true, enable non-exclusive writes for user transactions.
1009    // DO NOT ENABLE outside of the transaction test runner.
1010    #[serde(skip_serializing_if = "is_false")]
1011    enable_non_exclusive_writes: bool,
1012
1013    // If true, deprecate global storage ops everywhere.
1014    #[serde(skip_serializing_if = "is_false")]
1015    deprecate_global_storage_ops: bool,
1016
1017    // If true, normalize depth formula to not be empty for zero depth.
1018    #[serde(skip_serializing_if = "is_false")]
1019    normalize_depth_formula: bool,
1020
1021    // If true, skip GC'ed accept votes in CommitFinalizer.
1022    #[serde(skip_serializing_if = "is_false")]
1023    consensus_skip_gced_accept_votes: bool,
1024
1025    // If true, include cancelled randomness txns in the consensus commit prologue.
1026    #[serde(skip_serializing_if = "is_false")]
1027    include_cancelled_randomness_txns_in_prologue: bool,
1028
1029    // Enables address aliases.
1030    #[serde(skip_serializing_if = "is_false")]
1031    address_aliases: bool,
1032
1033    // Corrects signature-to-signer mapping in CheckpointContentsV2.
1034    // TODO: remove old code and deprecate once in mainnet.
1035    #[serde(skip_serializing_if = "is_false")]
1036    fix_checkpoint_signature_mapping: bool,
1037
1038    // If true, enable object funds withdraw.
1039    #[serde(skip_serializing_if = "is_false")]
1040    enable_object_funds_withdraw: bool,
1041
1042    // If true, skip GC'ed blocks in direct finalization.
1043    #[serde(skip_serializing_if = "is_false")]
1044    consensus_skip_gced_blocks_in_direct_finalization: bool,
1045
1046    // If true, uses a new rounding mechanism for gas calculations, replacing the step-based one
1047    #[serde(skip_serializing_if = "is_false")]
1048    gas_rounding_halve_digits: bool,
1049
1050    // If true, enable tx contexts in all argument positions
1051    #[serde(skip_serializing_if = "is_false")]
1052    flexible_tx_context_positions: bool,
1053
1054    // If true, disable entry point signature check.
1055    #[serde(skip_serializing_if = "is_false")]
1056    disable_entry_point_signature_check: bool,
1057
1058    // If true, convert withdrawal compatibility PTB arguments to coins at the start of the PTB.
1059    #[serde(skip_serializing_if = "is_false")]
1060    convert_withdrawal_compatibility_ptb_arguments: bool,
1061
1062    // If true, additional restrictions for hot or not entry functions are enforced.
1063    #[serde(skip_serializing_if = "is_false")]
1064    restrict_hot_or_not_entry_functions: bool,
1065
1066    // If true, split checkpoints in consensus handler.
1067    #[serde(skip_serializing_if = "is_false")]
1068    split_checkpoints_in_consensus_handler: bool,
1069
1070    // If true, always accept committed system transactions.
1071    #[serde(skip_serializing_if = "is_false")]
1072    consensus_always_accept_system_transactions: bool,
1073
1074    // If true perform consistent verification of metadata
1075    #[serde(skip_serializing_if = "is_false")]
1076    validator_metadata_verify_v2: bool,
1077
1078    // If true, defer transactions with unpaid consensus amplification
1079    // (where duplicate count exceeds gas_price / RGP + 1)
1080    #[serde(skip_serializing_if = "is_false")]
1081    defer_unpaid_amplification: bool,
1082
1083    #[serde(skip_serializing_if = "is_false")]
1084    randomize_checkpoint_tx_limit_in_tests: bool,
1085
1086    // If true, mark the gas coin as uninitialized in drop safety when there is no gas coin.
1087    #[serde(skip_serializing_if = "is_false")]
1088    gasless_transaction_drop_safety: bool,
1089
1090    // When split-checkpoints enabled, merge randomness and non-randomness schedulables together.
1091    #[serde(skip_serializing_if = "is_false")]
1092    merge_randomness_into_checkpoint: bool,
1093
1094    // If true, use coin party owner information.
1095    #[serde(skip_serializing_if = "is_false")]
1096    use_coin_party_owner: bool,
1097
1098    #[serde(skip_serializing_if = "is_false")]
1099    enable_gasless: bool,
1100
1101    #[serde(skip_serializing_if = "is_false")]
1102    gasless_verify_remaining_balance: bool,
1103
1104    #[serde(skip_serializing_if = "is_false")]
1105    disallow_jump_orphans: bool,
1106
1107    // If true, return early on type mismatch in receive_object.
1108    #[serde(skip_serializing_if = "is_false")]
1109    early_return_receive_object_mismatched_type: bool,
1110
1111    // If true, use consensus commit timestamps to determine epoch close instead of EndOfPublish voting.
1112    // Each validator transitions from AcceptAllCerts to RejectAllCerts when the consensus commit
1113    // timestamp exceeds the reconfiguration timestamp. EndOfPublish quorum still works as a
1114    // fallback for manual epoch close.
1115    #[serde(skip_serializing_if = "is_false")]
1116    timestamp_based_epoch_close: bool,
1117
1118    // If true, groth16::prepare_verifying_key checks that the verifying key has no more than
1119    // MAX_PUBLIC_INPUTS public inputs.
1120    #[serde(skip_serializing_if = "is_false")]
1121    limit_groth16_pvk_inputs: bool,
1122
1123    // If true, the funds-accumulator address-balance change invariant
1124    // (`TemporaryStore::check_address_balance_changes`) is enforced as a consensus check —
1125    // violations abort the tx via the conservation-recovery flow. When false, the check still
1126    // runs but a violation panics so unexpected violations surface during rollout.
1127    #[serde(skip_serializing_if = "is_false")]
1128    enforce_address_balance_change_invariant: bool,
1129
1130    // Enables more granular post-execution checks.
1131    #[serde(skip_serializing_if = "is_false")]
1132    granular_post_execution_checks: bool,
1133
1134    // If true, exit early for IFWW transactions.
1135    #[serde(skip_serializing_if = "is_false")]
1136    early_exit_on_iffw: bool,
1137}
1138
1139fn is_false(b: &bool) -> bool {
1140    !b
1141}
1142
1143fn is_empty(b: &BTreeSet<String>) -> bool {
1144    b.is_empty()
1145}
1146
1147fn is_zero(val: &u64) -> bool {
1148    *val == 0
1149}
1150
1151/// Ordering mechanism for transactions in one Narwhal consensus output.
1152#[derive(Default, Copy, Clone, PartialEq, Eq, Serialize, Deserialize, Debug)]
1153pub enum ConsensusTransactionOrdering {
1154    /// No ordering. Transactions are processed in the order they appear in the consensus output.
1155    #[default]
1156    None,
1157    /// Order transactions by gas price, highest first.
1158    ByGasPrice,
1159}
1160
1161impl ConsensusTransactionOrdering {
1162    pub fn is_none(&self) -> bool {
1163        matches!(self, ConsensusTransactionOrdering::None)
1164    }
1165}
1166
1167#[derive(Default, Copy, Clone, PartialEq, Eq, Serialize, Deserialize, Debug)]
1168pub struct ExecutionTimeEstimateParams {
1169    // Targeted per-object utilization as an integer percentage (1-100).
1170    pub target_utilization: u64,
1171    // Schedule up to this much extra work (in microseconds) per object,
1172    // but don't allow more than a single transaction to exceed this
1173    // burst limit.
1174    pub allowed_txn_cost_overage_burst_limit_us: u64,
1175
1176    // For separate budget for randomness-using tx, the above limits are
1177    // used with this integer-percentage scaling factor (1-100).
1178    pub randomness_scalar: u64,
1179
1180    // Absolute maximum allowed transaction duration estimate (in microseconds).
1181    pub max_estimate_us: u64,
1182
1183    // Number of the final checkpoints in an epoch whose observations should be
1184    // stored for use in the next epoch.
1185    pub stored_observations_num_included_checkpoints: u64,
1186
1187    // Absolute limit on the number of saved observations at end of epoch.
1188    pub stored_observations_limit: u64,
1189
1190    // Requires observations from at least this amount of stake in order to use
1191    // observation-based execution time estimates instead of the default.
1192    #[serde(skip_serializing_if = "is_zero")]
1193    pub stake_weighted_median_threshold: u64,
1194
1195    // For backwards compatibility with old behavior we use a zero default duration when adding
1196    // new execution time observation keys and a zero generation when loading stored observations.
1197    // This can be removed once set to "true" on mainnet.
1198    #[serde(skip_serializing_if = "is_false")]
1199    pub default_none_duration_for_new_keys: bool,
1200
1201    // Number of observations per chunk. When None, chunking is disabled.
1202    #[serde(skip_serializing_if = "Option::is_none")]
1203    pub observations_chunk_size: Option<u64>,
1204}
1205
1206// The config for per object congestion control in consensus handler.
1207#[derive(Default, Copy, Clone, PartialEq, Eq, Serialize, Deserialize, Debug)]
1208pub enum PerObjectCongestionControlMode {
1209    #[default]
1210    None, // Deprecated.
1211    TotalGasBudget,                                     // Deprecated.
1212    TotalTxCount,                                       // Deprecated.
1213    TotalGasBudgetWithCap,                              // Deprecated.
1214    ExecutionTimeEstimate(ExecutionTimeEstimateParams), // Use execution time estimate as execution cost.
1215}
1216
1217impl PerObjectCongestionControlMode {
1218    pub fn is_none(&self) -> bool {
1219        matches!(self, PerObjectCongestionControlMode::None)
1220    }
1221}
1222
1223// Configuration options for consensus algorithm.
1224#[derive(Default, Copy, Clone, PartialEq, Eq, Serialize, Deserialize, Debug)]
1225pub enum ConsensusChoice {
1226    #[default]
1227    Narwhal,
1228    SwapEachEpoch,
1229    Mysticeti,
1230}
1231
1232impl ConsensusChoice {
1233    pub fn is_narwhal(&self) -> bool {
1234        matches!(self, ConsensusChoice::Narwhal)
1235    }
1236}
1237
1238// Configuration options for consensus network.
1239#[derive(Default, Copy, Clone, PartialEq, Eq, Serialize, Deserialize, Debug)]
1240pub enum ConsensusNetwork {
1241    #[default]
1242    Anemo,
1243    Tonic,
1244}
1245
1246impl ConsensusNetwork {
1247    pub fn is_anemo(&self) -> bool {
1248        matches!(self, ConsensusNetwork::Anemo)
1249    }
1250}
1251
1252/// Constants that change the behavior of the protocol.
1253///
1254/// The value of each constant here must be fixed for a given protocol version. To change the value
1255/// of a constant, advance the protocol version, and add support for it in `get_for_version` under
1256/// the new version number.
1257/// (below).
1258///
1259/// To add a new field to this struct, use the following procedure:
1260/// - Advance the protocol version.
1261/// - Add the field as a private `Option<T>` to the struct.
1262/// - Initialize the field to `None` in prior protocol versions.
1263/// - Initialize the field to `Some(val)` for your new protocol version.
1264/// - Add a public getter that simply unwraps the field.
1265/// - Two public getters of the form `field(&self) -> field_type`
1266///     and `field_as_option(&self) -> Option<field_type>` will be automatically generated for you.
1267/// Example for a field: `new_constant: Option<u64>`
1268/// ```rust,ignore
1269///      pub fn new_constant(&self) -> u64 {
1270///         self.new_constant.expect(Self::CONSTANT_ERR_MSG)
1271///     }
1272///      pub fn new_constant_as_option(&self) -> Option<u64> {
1273///         self.new_constant.expect(Self::CONSTANT_ERR_MSG)
1274///     }
1275/// ```
1276/// With `pub fn new_constant(&self) -> u64`, if the constant is accessed in a protocol version
1277/// in which it is not defined, the validator will crash. (Crashing is necessary because
1278/// this type of error would almost always result in forking if not prevented here).
1279/// If you don't want the validator to crash, you can use the
1280/// `pub fn new_constant_as_option(&self) -> Option<u64>` getter, which will
1281/// return `None` if the field is not defined at that version.
1282/// - If you want a customized getter, you can add a method in the impl.
1283#[skip_serializing_none]
1284#[derive(Clone, Serialize, Debug, ProtocolConfigAccessors, ProtocolConfigOverride)]
1285pub struct ProtocolConfig {
1286    pub version: ProtocolVersion,
1287
1288    feature_flags: FeatureFlags,
1289
1290    // ==== Transaction input limits ====
1291    /// Maximum serialized size of a transaction (in bytes).
1292    max_tx_size_bytes: Option<u64>,
1293
1294    /// Maximum number of input objects to a transaction. Enforced by the transaction input checker
1295    max_input_objects: Option<u64>,
1296
1297    /// Max size of objects a transaction can write to disk after completion. Enforce by the Sui adapter.
1298    /// This is the sum of the serialized size of all objects written to disk.
1299    /// The max size of individual objects on the other hand is `max_move_object_size`.
1300    max_size_written_objects: Option<u64>,
1301    /// Max size of objects a system transaction can write to disk after completion. Enforce by the Sui adapter.
1302    /// Similar to `max_size_written_objects` but for system transactions.
1303    max_size_written_objects_system_tx: Option<u64>,
1304
1305    /// Maximum size of serialized transaction effects.
1306    max_serialized_tx_effects_size_bytes: Option<u64>,
1307
1308    /// Maximum size of serialized transaction effects for system transactions.
1309    max_serialized_tx_effects_size_bytes_system_tx: Option<u64>,
1310
1311    /// Maximum number of gas payment objects for a transaction.
1312    max_gas_payment_objects: Option<u32>,
1313
1314    /// Maximum number of modules in a Publish transaction.
1315    max_modules_in_publish: Option<u32>,
1316
1317    /// Maximum number of transitive dependencies in a package when publishing.
1318    max_package_dependencies: Option<u32>,
1319
1320    /// Maximum number of arguments in a move call or a ProgrammableTransaction's
1321    /// TransferObjects command.
1322    max_arguments: Option<u32>,
1323
1324    /// Maximum number of total type arguments, computed recursively.
1325    max_type_arguments: Option<u32>,
1326
1327    /// Maximum depth of an individual type argument.
1328    max_type_argument_depth: Option<u32>,
1329
1330    /// Maximum size of a Pure CallArg.
1331    max_pure_argument_size: Option<u32>,
1332
1333    /// Maximum number of Commands in a ProgrammableTransaction.
1334    max_programmable_tx_commands: Option<u32>,
1335
1336    // ==== Move VM, Move bytecode verifier, and execution limits ===
1337    /// Maximum Move bytecode version the VM understands. All older versions are accepted.
1338    move_binary_format_version: Option<u32>,
1339    min_move_binary_format_version: Option<u32>,
1340
1341    /// Configuration controlling binary tables size.
1342    binary_module_handles: Option<u16>,
1343    binary_struct_handles: Option<u16>,
1344    binary_function_handles: Option<u16>,
1345    binary_function_instantiations: Option<u16>,
1346    binary_signatures: Option<u16>,
1347    binary_constant_pool: Option<u16>,
1348    binary_identifiers: Option<u16>,
1349    binary_address_identifiers: Option<u16>,
1350    binary_struct_defs: Option<u16>,
1351    binary_struct_def_instantiations: Option<u16>,
1352    binary_function_defs: Option<u16>,
1353    binary_field_handles: Option<u16>,
1354    binary_field_instantiations: Option<u16>,
1355    binary_friend_decls: Option<u16>,
1356    binary_enum_defs: Option<u16>,
1357    binary_enum_def_instantiations: Option<u16>,
1358    binary_variant_handles: Option<u16>,
1359    binary_variant_instantiation_handles: Option<u16>,
1360
1361    /// Maximum size of the `contents` part of an object, in bytes. Enforced by the Sui adapter when effects are produced.
1362    max_move_object_size: Option<u64>,
1363
1364    // TODO: Option<increase to 500 KB. currently, publishing a package > 500 KB exceeds the max computation gas cost
1365    /// Maximum size of a Move package object, in bytes. Enforced by the Sui adapter at the end of a publish transaction.
1366    max_move_package_size: Option<u64>,
1367
1368    /// Max number of publish or upgrade commands allowed in a programmable transaction block.
1369    max_publish_or_upgrade_per_ptb: Option<u64>,
1370
1371    /// Maximum gas budget in MIST that a transaction can use.
1372    max_tx_gas: Option<u64>,
1373
1374    /// Maximum amount of the proposed gas price in MIST (defined in the transaction).
1375    max_gas_price: Option<u64>,
1376
1377    /// For aborted txns, we cap the gas price at a factor of RGP. This lowers risk of setting higher priority gas price
1378    /// if there's a chance the txn will abort.
1379    max_gas_price_rgp_factor_for_aborted_transactions: Option<u64>,
1380
1381    /// The max computation bucket for gas. This is the max that can be charged for computation.
1382    max_gas_computation_bucket: Option<u64>,
1383
1384    // Define the value used to round up computation gas charges
1385    gas_rounding_step: Option<u64>,
1386
1387    /// Maximum number of nested loops. Enforced by the Move bytecode verifier.
1388    max_loop_depth: Option<u64>,
1389
1390    /// Maximum number of type arguments that can be bound to generic type parameters. Enforced by the Move bytecode verifier.
1391    max_generic_instantiation_length: Option<u64>,
1392
1393    /// Maximum number of parameters that a Move function can have. Enforced by the Move bytecode verifier.
1394    max_function_parameters: Option<u64>,
1395
1396    /// Maximum number of basic blocks that a Move function can have. Enforced by the Move bytecode verifier.
1397    max_basic_blocks: Option<u64>,
1398
1399    /// Maximum stack size value. Enforced by the Move bytecode verifier.
1400    max_value_stack_size: Option<u64>,
1401
1402    /// Maximum number of "type nodes", a metric for how big a SignatureToken will be when expanded into a fully qualified type. Enforced by the Move bytecode verifier.
1403    max_type_nodes: Option<u64>,
1404
1405    /// Maximum number of push instructions in one function. Enforced by the Move bytecode verifier.
1406    max_push_size: Option<u64>,
1407
1408    /// Maximum number of struct definitions in a module. Enforced by the Move bytecode verifier.
1409    max_struct_definitions: Option<u64>,
1410
1411    /// Maximum number of function definitions in a module. Enforced by the Move bytecode verifier.
1412    max_function_definitions: Option<u64>,
1413
1414    /// Maximum number of fields allowed in a struct definition. Enforced by the Move bytecode verifier.
1415    max_fields_in_struct: Option<u64>,
1416
1417    /// Maximum dependency depth. Enforced by the Move linker when loading dependent modules.
1418    max_dependency_depth: Option<u64>,
1419
1420    /// Maximum number of Move events that a single transaction can emit. Enforced by the VM during execution.
1421    max_num_event_emit: Option<u64>,
1422
1423    /// Maximum number of new IDs that a single transaction can create. Enforced by the VM during execution.
1424    max_num_new_move_object_ids: Option<u64>,
1425
1426    /// Maximum number of new IDs that a single system transaction can create. Enforced by the VM during execution.
1427    max_num_new_move_object_ids_system_tx: Option<u64>,
1428
1429    /// Maximum number of IDs that a single transaction can delete. Enforced by the VM during execution.
1430    max_num_deleted_move_object_ids: Option<u64>,
1431
1432    /// Maximum number of IDs that a single system transaction can delete. Enforced by the VM during execution.
1433    max_num_deleted_move_object_ids_system_tx: Option<u64>,
1434
1435    /// Maximum number of IDs that a single transaction can transfer. Enforced by the VM during execution.
1436    max_num_transferred_move_object_ids: Option<u64>,
1437
1438    /// Maximum number of IDs that a single system transaction can transfer. Enforced by the VM during execution.
1439    max_num_transferred_move_object_ids_system_tx: Option<u64>,
1440
1441    /// Maximum size of a Move user event. Enforced by the VM during execution.
1442    max_event_emit_size: Option<u64>,
1443
1444    /// Maximum size of a Move user event. Enforced by the VM during execution.
1445    max_event_emit_size_total: Option<u64>,
1446
1447    /// Maximum length of a vector in Move. Enforced by the VM during execution, and for constants, by the verifier.
1448    max_move_vector_len: Option<u64>,
1449
1450    /// Maximum length of an `Identifier` in Move. Enforced by the bytecode verifier at signing.
1451    max_move_identifier_len: Option<u64>,
1452
1453    /// Maximum depth of a Move value within the VM.
1454    max_move_value_depth: Option<u64>,
1455
1456    /// Maximum number of variants in an enum. Enforced by the bytecode verifier at signing.
1457    max_move_enum_variants: Option<u64>,
1458
1459    /// Maximum number of back edges in Move function. Enforced by the bytecode verifier at signing.
1460    max_back_edges_per_function: Option<u64>,
1461
1462    /// Maximum number of back edges in Move module. Enforced by the bytecode verifier at signing.
1463    max_back_edges_per_module: Option<u64>,
1464
1465    /// Maximum number of meter `ticks` spent verifying a Move function. Enforced by the bytecode verifier at signing.
1466    max_verifier_meter_ticks_per_function: Option<u64>,
1467
1468    /// Maximum number of meter `ticks` spent verifying a Move module. Enforced by the bytecode verifier at signing.
1469    max_meter_ticks_per_module: Option<u64>,
1470
1471    /// Maximum number of meter `ticks` spent verifying a Move package. Enforced by the bytecode verifier at signing.
1472    max_meter_ticks_per_package: Option<u64>,
1473
1474    // === Object runtime internal operation limits ====
1475    // These affect dynamic fields
1476    /// Maximum number of cached objects in the object runtime ObjectStore. Enforced by object runtime during execution
1477    object_runtime_max_num_cached_objects: Option<u64>,
1478
1479    /// Maximum number of cached objects in the object runtime ObjectStore in system transaction. Enforced by object runtime during execution
1480    object_runtime_max_num_cached_objects_system_tx: Option<u64>,
1481
1482    /// Maximum number of stored objects accessed by object runtime ObjectStore. Enforced by object runtime during execution
1483    object_runtime_max_num_store_entries: Option<u64>,
1484
1485    /// Maximum number of stored objects accessed by object runtime ObjectStore in system transaction. Enforced by object runtime during execution
1486    object_runtime_max_num_store_entries_system_tx: Option<u64>,
1487
1488    // === Execution gas costs ====
1489    /// Base cost for any Sui transaction
1490    base_tx_cost_fixed: Option<u64>,
1491
1492    /// Additional cost for a transaction that publishes a package
1493    /// i.e., the base cost of such a transaction is base_tx_cost_fixed + package_publish_cost_fixed
1494    package_publish_cost_fixed: Option<u64>,
1495
1496    /// Cost per byte of a Move call transaction
1497    /// i.e., the cost of such a transaction is base_cost + (base_tx_cost_per_byte * size)
1498    base_tx_cost_per_byte: Option<u64>,
1499
1500    /// Cost per byte for a transaction that publishes a package
1501    package_publish_cost_per_byte: Option<u64>,
1502
1503    // Per-byte cost of reading an object during transaction execution
1504    obj_access_cost_read_per_byte: Option<u64>,
1505
1506    // Per-byte cost of writing an object during transaction execution
1507    obj_access_cost_mutate_per_byte: Option<u64>,
1508
1509    // Per-byte cost of deleting an object during transaction execution
1510    obj_access_cost_delete_per_byte: Option<u64>,
1511
1512    /// Per-byte cost charged for each input object to a transaction.
1513    /// Meant to approximate the cost of checking locks for each object
1514    // TODO: Option<I'm not sure that this cost makes sense. Checking locks is "free"
1515    // in the sense that an invalid tx that can never be committed/pay gas can
1516    // force validators to check an arbitrary number of locks. If those checks are
1517    // "free" for invalid transactions, why charge for them in valid transactions
1518    // TODO: Option<if we keep this, I think we probably want it to be a fixed cost rather
1519    // than a per-byte cost. checking an object lock should not require loading an
1520    // entire object, just consulting an ID -> tx digest map
1521    obj_access_cost_verify_per_byte: Option<u64>,
1522
1523    // Maximal nodes which are allowed when converting to a type layout.
1524    max_type_to_layout_nodes: Option<u64>,
1525
1526    // Maximal size in bytes that a PTB value can be
1527    max_ptb_value_size: Option<u64>,
1528
1529    // === Gas version. gas model ===
1530    /// Gas model version, what code we are using to charge gas
1531    gas_model_version: Option<u64>,
1532
1533    // === Storage gas costs ===
1534    /// Per-byte cost of storing an object in the Sui global object store. Some of this cost may be refundable if the object is later freed
1535    obj_data_cost_refundable: Option<u64>,
1536
1537    // Per-byte cost of storing an object in the Sui transaction log (e.g., in CertifiedTransactionEffects)
1538    // This depends on the size of various fields including the effects
1539    // TODO: Option<I don't fully understand this^ and more details would be useful
1540    obj_metadata_cost_non_refundable: Option<u64>,
1541
1542    // === Tokenomics ===
1543
1544    // TODO: Option<this should be changed to u64.
1545    /// Sender of a txn that touches an object will get this percent of the storage rebate back.
1546    /// In basis point.
1547    storage_rebate_rate: Option<u64>,
1548
1549    /// 5% of the storage fund's share of rewards are reinvested into the storage fund.
1550    /// In basis point.
1551    storage_fund_reinvest_rate: Option<u64>,
1552
1553    /// The share of rewards that will be slashed and redistributed is 50%.
1554    /// In basis point.
1555    reward_slashing_rate: Option<u64>,
1556
1557    /// Unit gas price, Mist per internal gas unit.
1558    storage_gas_price: Option<u64>,
1559
1560    /// Per-object storage cost for accumulator objects, used during end-of-epoch accounting.
1561    accumulator_object_storage_cost: Option<u64>,
1562
1563    // === Core Protocol ===
1564    /// Max number of transactions per checkpoint.
1565    /// Note that this is a protocol constant and not a config as validators must have this set to
1566    /// the same value, otherwise they *will* fork.
1567    max_transactions_per_checkpoint: Option<u64>,
1568
1569    /// Max size of a checkpoint in bytes.
1570    /// Note that this is a protocol constant and not a config as validators must have this set to
1571    /// the same value, otherwise they *will* fork.
1572    max_checkpoint_size_bytes: Option<u64>,
1573
1574    /// A protocol upgrade always requires 2f+1 stake to agree. We support a buffer of additional
1575    /// stake (as a fraction of f, expressed in basis points) that is required before an upgrade
1576    /// can happen automatically. 10000bps would indicate that complete unanimity is required (all
1577    /// 3f+1 must vote), while 0bps would indicate that 2f+1 is sufficient.
1578    buffer_stake_for_protocol_upgrade_bps: Option<u64>,
1579
1580    // === Native Function Costs ===
1581
1582    // `address` module
1583    // Cost params for the Move native function `address::from_bytes(bytes: vector<u8>)`
1584    address_from_bytes_cost_base: Option<u64>,
1585    // Cost params for the Move native function `address::to_u256(address): u256`
1586    address_to_u256_cost_base: Option<u64>,
1587    // Cost params for the Move native function `address::from_u256(u256): address`
1588    address_from_u256_cost_base: Option<u64>,
1589
1590    // `config` module
1591    // Cost params for the Move native function `read_setting_impl<Name: copy + drop + store,
1592    // SettingValue: key + store, SettingDataValue: store, Value: copy + drop + store,
1593    // >(config: address, name: address, current_epoch: u64): Option<Value>`
1594    config_read_setting_impl_cost_base: Option<u64>,
1595    config_read_setting_impl_cost_per_byte: Option<u64>,
1596
1597    // `dynamic_field` module
1598    // Cost params for the Move native function `hash_type_and_key<K: copy + drop + store>(parent: address, k: K): address`
1599    dynamic_field_hash_type_and_key_cost_base: Option<u64>,
1600    dynamic_field_hash_type_and_key_type_cost_per_byte: Option<u64>,
1601    dynamic_field_hash_type_and_key_value_cost_per_byte: Option<u64>,
1602    dynamic_field_hash_type_and_key_type_tag_cost_per_byte: Option<u64>,
1603    // Cost params for the Move native function `add_child_object<Child: key>(parent: address, child: Child)`
1604    dynamic_field_add_child_object_cost_base: Option<u64>,
1605    dynamic_field_add_child_object_type_cost_per_byte: Option<u64>,
1606    dynamic_field_add_child_object_value_cost_per_byte: Option<u64>,
1607    dynamic_field_add_child_object_struct_tag_cost_per_byte: Option<u64>,
1608    // Cost params for the Move native function `borrow_child_object_mut<Child: key>(parent: &mut UID, id: address): &mut Child`
1609    dynamic_field_borrow_child_object_cost_base: Option<u64>,
1610    dynamic_field_borrow_child_object_child_ref_cost_per_byte: Option<u64>,
1611    dynamic_field_borrow_child_object_type_cost_per_byte: Option<u64>,
1612    // Cost params for the Move native function `remove_child_object<Child: key>(parent: address, id: address): Child`
1613    dynamic_field_remove_child_object_cost_base: Option<u64>,
1614    dynamic_field_remove_child_object_child_cost_per_byte: Option<u64>,
1615    dynamic_field_remove_child_object_type_cost_per_byte: Option<u64>,
1616    // Cost params for the Move native function `has_child_object(parent: address, id: address): bool`
1617    dynamic_field_has_child_object_cost_base: Option<u64>,
1618    // Cost params for the Move native function `has_child_object_with_ty<Child: key>(parent: address, id: address): bool`
1619    dynamic_field_has_child_object_with_ty_cost_base: Option<u64>,
1620    dynamic_field_has_child_object_with_ty_type_cost_per_byte: Option<u64>,
1621    dynamic_field_has_child_object_with_ty_type_tag_cost_per_byte: Option<u64>,
1622
1623    // `event` module
1624    // Cost params for the Move native function `event::emit<T: copy + drop>(event: T)`
1625    event_emit_cost_base: Option<u64>,
1626    event_emit_value_size_derivation_cost_per_byte: Option<u64>,
1627    event_emit_tag_size_derivation_cost_per_byte: Option<u64>,
1628    event_emit_output_cost_per_byte: Option<u64>,
1629    event_emit_auth_stream_cost: Option<u64>,
1630
1631    //  `object` module
1632    // Cost params for the Move native function `borrow_uid<T: key>(obj: &T): &UID`
1633    object_borrow_uid_cost_base: Option<u64>,
1634    // Cost params for the Move native function `delete_impl(id: address)`
1635    object_delete_impl_cost_base: Option<u64>,
1636    // Cost params for the Move native function `record_new_uid(id: address)`
1637    object_record_new_uid_cost_base: Option<u64>,
1638
1639    // Transfer
1640    // Cost params for the Move native function `transfer_impl<T: key>(obj: T, recipient: address)`
1641    transfer_transfer_internal_cost_base: Option<u64>,
1642    // Cost params for the Move native function `party_transfer_impl<T: key>(obj: T, party_members: vector<address>)`
1643    transfer_party_transfer_internal_cost_base: Option<u64>,
1644    // Cost params for the Move native function `freeze_object<T: key>(obj: T)`
1645    transfer_freeze_object_cost_base: Option<u64>,
1646    // Cost params for the Move native function `share_object<T: key>(obj: T)`
1647    transfer_share_object_cost_base: Option<u64>,
1648    // Cost params for the Move native function
1649    // `receive_object<T: key>(p: &mut UID, recv: Receiving<T>T)`
1650    transfer_receive_object_cost_base: Option<u64>,
1651    transfer_receive_object_cost_per_byte: Option<u64>,
1652    transfer_receive_object_type_cost_per_byte: Option<u64>,
1653
1654    // TxContext
1655    // Cost params for the Move native function `transfer_impl<T: key>(obj: T, recipient: address)`
1656    tx_context_derive_id_cost_base: Option<u64>,
1657    tx_context_fresh_id_cost_base: Option<u64>,
1658    tx_context_sender_cost_base: Option<u64>,
1659    tx_context_epoch_cost_base: Option<u64>,
1660    tx_context_epoch_timestamp_ms_cost_base: Option<u64>,
1661    tx_context_sponsor_cost_base: Option<u64>,
1662    tx_context_rgp_cost_base: Option<u64>,
1663    tx_context_gas_price_cost_base: Option<u64>,
1664    tx_context_gas_budget_cost_base: Option<u64>,
1665    tx_context_ids_created_cost_base: Option<u64>,
1666    tx_context_replace_cost_base: Option<u64>,
1667
1668    // Types
1669    // Cost params for the Move native function `is_one_time_witness<T: drop>(_: &T): bool`
1670    types_is_one_time_witness_cost_base: Option<u64>,
1671    types_is_one_time_witness_type_tag_cost_per_byte: Option<u64>,
1672    types_is_one_time_witness_type_cost_per_byte: Option<u64>,
1673
1674    // Validator
1675    // Cost params for the Move native function `validate_metadata_bcs(metadata: vector<u8>)`
1676    validator_validate_metadata_cost_base: Option<u64>,
1677    validator_validate_metadata_data_cost_per_byte: Option<u64>,
1678
1679    // Crypto natives
1680    crypto_invalid_arguments_cost: Option<u64>,
1681    // bls12381::bls12381_min_sig_verify
1682    bls12381_bls12381_min_sig_verify_cost_base: Option<u64>,
1683    bls12381_bls12381_min_sig_verify_msg_cost_per_byte: Option<u64>,
1684    bls12381_bls12381_min_sig_verify_msg_cost_per_block: Option<u64>,
1685
1686    // bls12381::bls12381_min_pk_verify
1687    bls12381_bls12381_min_pk_verify_cost_base: Option<u64>,
1688    bls12381_bls12381_min_pk_verify_msg_cost_per_byte: Option<u64>,
1689    bls12381_bls12381_min_pk_verify_msg_cost_per_block: Option<u64>,
1690
1691    // ecdsa_k1::ecrecover
1692    ecdsa_k1_ecrecover_keccak256_cost_base: Option<u64>,
1693    ecdsa_k1_ecrecover_keccak256_msg_cost_per_byte: Option<u64>,
1694    ecdsa_k1_ecrecover_keccak256_msg_cost_per_block: Option<u64>,
1695    ecdsa_k1_ecrecover_sha256_cost_base: Option<u64>,
1696    ecdsa_k1_ecrecover_sha256_msg_cost_per_byte: Option<u64>,
1697    ecdsa_k1_ecrecover_sha256_msg_cost_per_block: Option<u64>,
1698
1699    // ecdsa_k1::decompress_pubkey
1700    ecdsa_k1_decompress_pubkey_cost_base: Option<u64>,
1701
1702    // ecdsa_k1::secp256k1_verify
1703    ecdsa_k1_secp256k1_verify_keccak256_cost_base: Option<u64>,
1704    ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_byte: Option<u64>,
1705    ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_block: Option<u64>,
1706    ecdsa_k1_secp256k1_verify_sha256_cost_base: Option<u64>,
1707    ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_byte: Option<u64>,
1708    ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_block: Option<u64>,
1709
1710    // ecdsa_r1::ecrecover
1711    ecdsa_r1_ecrecover_keccak256_cost_base: Option<u64>,
1712    ecdsa_r1_ecrecover_keccak256_msg_cost_per_byte: Option<u64>,
1713    ecdsa_r1_ecrecover_keccak256_msg_cost_per_block: Option<u64>,
1714    ecdsa_r1_ecrecover_sha256_cost_base: Option<u64>,
1715    ecdsa_r1_ecrecover_sha256_msg_cost_per_byte: Option<u64>,
1716    ecdsa_r1_ecrecover_sha256_msg_cost_per_block: Option<u64>,
1717
1718    // ecdsa_r1::secp256k1_verify
1719    ecdsa_r1_secp256r1_verify_keccak256_cost_base: Option<u64>,
1720    ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_byte: Option<u64>,
1721    ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_block: Option<u64>,
1722    ecdsa_r1_secp256r1_verify_sha256_cost_base: Option<u64>,
1723    ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_byte: Option<u64>,
1724    ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_block: Option<u64>,
1725
1726    // ecvrf::verify
1727    ecvrf_ecvrf_verify_cost_base: Option<u64>,
1728    ecvrf_ecvrf_verify_alpha_string_cost_per_byte: Option<u64>,
1729    ecvrf_ecvrf_verify_alpha_string_cost_per_block: Option<u64>,
1730
1731    // ed25519
1732    ed25519_ed25519_verify_cost_base: Option<u64>,
1733    ed25519_ed25519_verify_msg_cost_per_byte: Option<u64>,
1734    ed25519_ed25519_verify_msg_cost_per_block: Option<u64>,
1735
1736    // groth16::prepare_verifying_key
1737    groth16_prepare_verifying_key_bls12381_cost_base: Option<u64>,
1738    groth16_prepare_verifying_key_bn254_cost_base: Option<u64>,
1739
1740    // groth16::verify_groth16_proof_internal
1741    groth16_verify_groth16_proof_internal_bls12381_cost_base: Option<u64>,
1742    groth16_verify_groth16_proof_internal_bls12381_cost_per_public_input: Option<u64>,
1743    groth16_verify_groth16_proof_internal_bn254_cost_base: Option<u64>,
1744    groth16_verify_groth16_proof_internal_bn254_cost_per_public_input: Option<u64>,
1745    groth16_verify_groth16_proof_internal_public_input_cost_per_byte: Option<u64>,
1746
1747    // hash::blake2b256
1748    hash_blake2b256_cost_base: Option<u64>,
1749    hash_blake2b256_data_cost_per_byte: Option<u64>,
1750    hash_blake2b256_data_cost_per_block: Option<u64>,
1751
1752    // hash::keccak256
1753    hash_keccak256_cost_base: Option<u64>,
1754    hash_keccak256_data_cost_per_byte: Option<u64>,
1755    hash_keccak256_data_cost_per_block: Option<u64>,
1756
1757    // poseidon::poseidon_bn254
1758    poseidon_bn254_cost_base: Option<u64>,
1759    poseidon_bn254_cost_per_block: Option<u64>,
1760
1761    // group_ops
1762    group_ops_bls12381_decode_scalar_cost: Option<u64>,
1763    group_ops_bls12381_decode_g1_cost: Option<u64>,
1764    group_ops_bls12381_decode_g2_cost: Option<u64>,
1765    group_ops_bls12381_decode_gt_cost: Option<u64>,
1766    group_ops_bls12381_scalar_add_cost: Option<u64>,
1767    group_ops_bls12381_g1_add_cost: Option<u64>,
1768    group_ops_bls12381_g2_add_cost: Option<u64>,
1769    group_ops_bls12381_gt_add_cost: Option<u64>,
1770    group_ops_bls12381_scalar_sub_cost: Option<u64>,
1771    group_ops_bls12381_g1_sub_cost: Option<u64>,
1772    group_ops_bls12381_g2_sub_cost: Option<u64>,
1773    group_ops_bls12381_gt_sub_cost: Option<u64>,
1774    group_ops_bls12381_scalar_mul_cost: Option<u64>,
1775    group_ops_bls12381_g1_mul_cost: Option<u64>,
1776    group_ops_bls12381_g2_mul_cost: Option<u64>,
1777    group_ops_bls12381_gt_mul_cost: Option<u64>,
1778    group_ops_bls12381_scalar_div_cost: Option<u64>,
1779    group_ops_bls12381_g1_div_cost: Option<u64>,
1780    group_ops_bls12381_g2_div_cost: Option<u64>,
1781    group_ops_bls12381_gt_div_cost: Option<u64>,
1782    group_ops_bls12381_g1_hash_to_base_cost: Option<u64>,
1783    group_ops_bls12381_g2_hash_to_base_cost: Option<u64>,
1784    group_ops_bls12381_g1_hash_to_cost_per_byte: Option<u64>,
1785    group_ops_bls12381_g2_hash_to_cost_per_byte: Option<u64>,
1786    group_ops_bls12381_g1_msm_base_cost: Option<u64>,
1787    group_ops_bls12381_g2_msm_base_cost: Option<u64>,
1788    group_ops_bls12381_g1_msm_base_cost_per_input: Option<u64>,
1789    group_ops_bls12381_g2_msm_base_cost_per_input: Option<u64>,
1790    group_ops_bls12381_msm_max_len: Option<u32>,
1791    group_ops_bls12381_pairing_cost: Option<u64>,
1792    group_ops_bls12381_g1_to_uncompressed_g1_cost: Option<u64>,
1793    group_ops_bls12381_uncompressed_g1_to_g1_cost: Option<u64>,
1794    group_ops_bls12381_uncompressed_g1_sum_base_cost: Option<u64>,
1795    group_ops_bls12381_uncompressed_g1_sum_cost_per_term: Option<u64>,
1796    group_ops_bls12381_uncompressed_g1_sum_max_terms: Option<u64>,
1797
1798    group_ops_ristretto_decode_scalar_cost: Option<u64>,
1799    group_ops_ristretto_decode_point_cost: Option<u64>,
1800    group_ops_ristretto_scalar_add_cost: Option<u64>,
1801    group_ops_ristretto_point_add_cost: Option<u64>,
1802    group_ops_ristretto_scalar_sub_cost: Option<u64>,
1803    group_ops_ristretto_point_sub_cost: Option<u64>,
1804    group_ops_ristretto_scalar_mul_cost: Option<u64>,
1805    group_ops_ristretto_point_mul_cost: Option<u64>,
1806    group_ops_ristretto_scalar_div_cost: Option<u64>,
1807    group_ops_ristretto_point_div_cost: Option<u64>,
1808
1809    verify_bulletproofs_ristretto255_base_cost: Option<u64>,
1810    verify_bulletproofs_ristretto255_cost_per_bit_and_commitment: Option<u64>,
1811
1812    // hmac::hmac_sha3_256
1813    hmac_hmac_sha3_256_cost_base: Option<u64>,
1814    hmac_hmac_sha3_256_input_cost_per_byte: Option<u64>,
1815    hmac_hmac_sha3_256_input_cost_per_block: Option<u64>,
1816
1817    // zklogin::check_zklogin_id
1818    check_zklogin_id_cost_base: Option<u64>,
1819    // zklogin::check_zklogin_issuer
1820    check_zklogin_issuer_cost_base: Option<u64>,
1821
1822    vdf_verify_vdf_cost: Option<u64>,
1823    vdf_hash_to_input_cost: Option<u64>,
1824
1825    // nitro_attestation::load_nitro_attestation
1826    nitro_attestation_parse_base_cost: Option<u64>,
1827    nitro_attestation_parse_cost_per_byte: Option<u64>,
1828    nitro_attestation_verify_base_cost: Option<u64>,
1829    nitro_attestation_verify_cost_per_cert: Option<u64>,
1830
1831    // Stdlib costs
1832    bcs_per_byte_serialized_cost: Option<u64>,
1833    bcs_legacy_min_output_size_cost: Option<u64>,
1834    bcs_failure_cost: Option<u64>,
1835
1836    hash_sha2_256_base_cost: Option<u64>,
1837    hash_sha2_256_per_byte_cost: Option<u64>,
1838    hash_sha2_256_legacy_min_input_len_cost: Option<u64>,
1839    hash_sha3_256_base_cost: Option<u64>,
1840    hash_sha3_256_per_byte_cost: Option<u64>,
1841    hash_sha3_256_legacy_min_input_len_cost: Option<u64>,
1842    type_name_get_base_cost: Option<u64>,
1843    type_name_get_per_byte_cost: Option<u64>,
1844    type_name_id_base_cost: Option<u64>,
1845
1846    string_check_utf8_base_cost: Option<u64>,
1847    string_check_utf8_per_byte_cost: Option<u64>,
1848    string_is_char_boundary_base_cost: Option<u64>,
1849    string_sub_string_base_cost: Option<u64>,
1850    string_sub_string_per_byte_cost: Option<u64>,
1851    string_index_of_base_cost: Option<u64>,
1852    string_index_of_per_byte_pattern_cost: Option<u64>,
1853    string_index_of_per_byte_searched_cost: Option<u64>,
1854
1855    vector_empty_base_cost: Option<u64>,
1856    vector_length_base_cost: Option<u64>,
1857    vector_push_back_base_cost: Option<u64>,
1858    vector_push_back_legacy_per_abstract_memory_unit_cost: Option<u64>,
1859    vector_borrow_base_cost: Option<u64>,
1860    vector_pop_back_base_cost: Option<u64>,
1861    vector_destroy_empty_base_cost: Option<u64>,
1862    vector_swap_base_cost: Option<u64>,
1863    debug_print_base_cost: Option<u64>,
1864    debug_print_stack_trace_base_cost: Option<u64>,
1865
1866    // ==== Ephemeral (consensus only) params deleted ====
1867    //
1868    // Const params for consensus scoring decision
1869    // The scaling factor property for the MED outlier detection
1870    // scoring_decision_mad_divisor: Option<f64>,
1871    // The cutoff value for the MED outlier detection
1872    // scoring_decision_cutoff_value: Option<f64>,
1873    /// === Execution Version ===
1874    execution_version: Option<u64>,
1875
1876    // Dictates the threshold (percentage of stake) that is used to calculate the "bad" nodes to be
1877    // swapped when creating the consensus schedule. The values should be of the range [0 - 33]. Anything
1878    // above 33 (f) will not be allowed.
1879    consensus_bad_nodes_stake_threshold: Option<u64>,
1880
1881    max_jwk_votes_per_validator_per_epoch: Option<u64>,
1882    // The maximum age of a JWK in epochs before it is removed from the AuthenticatorState object.
1883    // Applied at the end of an epoch as a delta from the new epoch value, so setting this to 1
1884    // will cause the new epoch to start with JWKs from the previous epoch still valid.
1885    max_age_of_jwk_in_epochs: Option<u64>,
1886
1887    // === random beacon ===
1888    /// Maximum allowed precision loss when reducing voting weights for the random beacon
1889    /// protocol.
1890    random_beacon_reduction_allowed_delta: Option<u16>,
1891
1892    /// Minimum number of shares below which voting weights will not be reduced for the
1893    /// random beacon protocol.
1894    random_beacon_reduction_lower_bound: Option<u32>,
1895
1896    /// Consensus Round after which DKG should be aborted and randomness disabled for
1897    /// the epoch, if it hasn't already completed.
1898    random_beacon_dkg_timeout_round: Option<u32>,
1899
1900    /// Minimum interval between consecutive rounds of generated randomness.
1901    random_beacon_min_round_interval_ms: Option<u64>,
1902
1903    /// Version of the random beacon DKG protocol.
1904    /// 0 was deprecated (and currently not supported), 1 is the default version.
1905    random_beacon_dkg_version: Option<u64>,
1906
1907    /// The maximum serialised transaction size (in bytes) accepted by consensus. That should be bigger than the
1908    /// `max_tx_size_bytes` with some additional headroom.
1909    consensus_max_transaction_size_bytes: Option<u64>,
1910    /// The maximum size of transactions included in a consensus block.
1911    consensus_max_transactions_in_block_bytes: Option<u64>,
1912    /// The maximum number of transactions included in a consensus block.
1913    consensus_max_num_transactions_in_block: Option<u64>,
1914
1915    /// The maximum number of rounds where transaction voting is allowed.
1916    consensus_voting_rounds: Option<u32>,
1917
1918    /// DEPRECATED. Do not use.
1919    max_accumulated_txn_cost_per_object_in_narwhal_commit: Option<u64>,
1920
1921    /// The max number of consensus rounds a transaction can be deferred due to shared object congestion.
1922    /// Transactions will be cancelled after this many rounds.
1923    max_deferral_rounds_for_congestion_control: Option<u64>,
1924
1925    /// DEPRECATED. Do not use.
1926    max_txn_cost_overage_per_object_in_commit: Option<u64>,
1927
1928    /// DEPRECATED. Do not use.
1929    allowed_txn_cost_overage_burst_per_object_in_commit: Option<u64>,
1930
1931    /// Minimum interval of commit timestamps between consecutive checkpoints.
1932    min_checkpoint_interval_ms: Option<u64>,
1933
1934    /// Version number to use for version_specific_data in `CheckpointSummary`.
1935    checkpoint_summary_version_specific_data: Option<u64>,
1936
1937    /// The max number of transactions that can be included in a single Soft Bundle.
1938    max_soft_bundle_size: Option<u64>,
1939
1940    /// Whether to try to form bridge committee
1941    // Note: this is not a feature flag because we want to distinguish between
1942    // `None` and `Some(false)`, as committee was already finalized on Testnet.
1943    bridge_should_try_to_finalize_committee: Option<bool>,
1944
1945    /// The max accumulated txn execution cost per object in a mysticeti. Transactions
1946    /// in a commit will be deferred once their touch shared objects hit this limit,
1947    /// unless the selected congestion control mode allows overage.
1948    /// This config plays the same role as `max_accumulated_txn_cost_per_object_in_narwhal_commit`
1949    /// but for mysticeti commits due to that mysticeti has higher commit rate.
1950    max_accumulated_txn_cost_per_object_in_mysticeti_commit: Option<u64>,
1951
1952    /// As above, but separate per-commit budget for transactions that use randomness.
1953    /// If not configured, uses the setting for `max_accumulated_txn_cost_per_object_in_mysticeti_commit`.
1954    max_accumulated_randomness_txn_cost_per_object_in_mysticeti_commit: Option<u64>,
1955
1956    /// Configures the garbage collection depth for consensus. When is unset or `0` then the garbage collection
1957    /// is disabled.
1958    consensus_gc_depth: Option<u32>,
1959
1960    /// DEPRECATED. Do not use.
1961    gas_budget_based_txn_cost_cap_factor: Option<u64>,
1962
1963    /// DEPRECATED. Do not use.
1964    gas_budget_based_txn_cost_absolute_cap_commit_count: Option<u64>,
1965
1966    /// SIP-45: K in the formula `amplification_factor = max(0, gas_price / reference_gas_price - K)`.
1967    /// This is the threshold for activating consensus amplification.
1968    sip_45_consensus_amplification_threshold: Option<u64>,
1969
1970    /// DEPRECATED: this was an ephemeral feature flag only used in per-epoch tables, which has now
1971    /// been deployed everywhere.
1972    use_object_per_epoch_marker_table_v2: Option<bool>,
1973
1974    /// The number of commits to consider when computing a deterministic commit rate.
1975    consensus_commit_rate_estimation_window_size: Option<u32>,
1976
1977    /// A list of effective AliasedAddress.
1978    /// For each pair, `aliased` is allowed to act as `original` for any of the transaction digests
1979    /// listed in `tx_digests`
1980    #[serde(skip_serializing_if = "Vec::is_empty")]
1981    aliased_addresses: Vec<AliasedAddress>,
1982
1983    /// The base charge for each command in a programmable transaction. This is a fixed cost to
1984    /// account for the overhead of processing each command.
1985    translation_per_command_base_charge: Option<u64>,
1986
1987    /// The base charge for each input in a programmable transaction regardless of if it is used or
1988    /// not, or a pure/object/funds withdrawal input.
1989    translation_per_input_base_charge: Option<u64>,
1990
1991    /// The base charge for each byte of pure input in a programmable transaction.
1992    translation_pure_input_per_byte_charge: Option<u64>,
1993
1994    /// The multiplier for the number of type nodes when charging for type loading.
1995    /// This is multiplied by the number of type nodes to get the total cost.
1996    /// This should be a small number to avoid excessive gas costs for loading types.
1997    translation_per_type_node_charge: Option<u64>,
1998
1999    /// The multiplier for the number of type references when charging for type checking and reference
2000    /// checking.
2001    translation_per_reference_node_charge: Option<u64>,
2002
2003    /// The multiplier for each linkage entry when charging for linkage tables that we have
2004    /// created.
2005    translation_per_linkage_entry_charge: Option<u64>,
2006
2007    /// The maximum number of updates per settlement transaction.
2008    max_updates_per_settlement_txn: Option<u32>,
2009
2010    /// Maximum computation units allowed for a gasless transaction.
2011    gasless_max_computation_units: Option<u64>,
2012
2013    /// Allowed token types for gasless transactions, with minimum transfer sizes per token.
2014    gasless_allowed_token_types: Option<Vec<(String, u64)>>,
2015
2016    /// Maximum number of unused Pure inputs allowed in a gasless transaction.
2017    /// Object and FundsWithdrawal inputs must always be used.
2018    /// When None, there is no limit (effectively unlimited).
2019    gasless_max_unused_inputs: Option<u64>,
2020
2021    /// Maximum size in bytes of each Pure input in a gasless transaction.
2022    /// When None, there is no limit (effectively unlimited).
2023    gasless_max_pure_input_bytes: Option<u64>,
2024
2025    /// Max tps for gasless transactions. Unlimited when unset, zero when set to zero.
2026    gasless_max_tps: Option<u64>,
2027
2028    #[serde(skip_serializing_if = "Option::is_none")]
2029    #[skip_accessor]
2030    include_special_package_amendments: Option<Arc<Amendments>>,
2031
2032    /// Maximum serialized size in bytes of a gasless transaction (SenderSignedData).
2033    /// Bounds the persistent storage impact of each admitted gasless transaction.
2034    gasless_max_tx_size_bytes: Option<u64>,
2035}
2036
2037/// An aliased address.
2038#[derive(Clone, Serialize, Deserialize, Debug)]
2039pub struct AliasedAddress {
2040    /// The original address.
2041    pub original: [u8; 32],
2042    /// An aliased address which is allowed to act as the original address.
2043    pub aliased: [u8; 32],
2044    /// A list of transaction digests for which the aliasing is allowed to be in effect.
2045    pub allowed_tx_digests: Vec<[u8; 32]>,
2046}
2047
2048// feature flags
2049impl ProtocolConfig {
2050    // Add checks for feature flag support here, e.g.:
2051    // pub fn check_new_protocol_feature_supported(&self) -> Result<(), Error> {
2052    //     if self.feature_flags.new_protocol_feature_supported {
2053    //         Ok(())
2054    //     } else {
2055    //         Err(Error(format!(
2056    //             "new_protocol_feature is not supported at {:?}",
2057    //             self.version
2058    //         )))
2059    //     }
2060    // }
2061
2062    pub fn check_package_upgrades_supported(&self) -> Result<(), Error> {
2063        if self.feature_flags.package_upgrades {
2064            Ok(())
2065        } else {
2066            Err(Error(format!(
2067                "package upgrades are not supported at {:?}",
2068                self.version
2069            )))
2070        }
2071    }
2072
2073    pub fn allow_receiving_object_id(&self) -> bool {
2074        self.feature_flags.allow_receiving_object_id
2075    }
2076
2077    pub fn receiving_objects_supported(&self) -> bool {
2078        self.feature_flags.receive_objects
2079    }
2080
2081    pub fn package_upgrades_supported(&self) -> bool {
2082        self.feature_flags.package_upgrades
2083    }
2084
2085    pub fn check_commit_root_state_digest_supported(&self) -> bool {
2086        self.feature_flags.commit_root_state_digest
2087    }
2088
2089    pub fn get_advance_epoch_start_time_in_safe_mode(&self) -> bool {
2090        self.feature_flags.advance_epoch_start_time_in_safe_mode
2091    }
2092
2093    pub fn loaded_child_objects_fixed(&self) -> bool {
2094        self.feature_flags.loaded_child_objects_fixed
2095    }
2096
2097    pub fn missing_type_is_compatibility_error(&self) -> bool {
2098        self.feature_flags.missing_type_is_compatibility_error
2099    }
2100
2101    pub fn scoring_decision_with_validity_cutoff(&self) -> bool {
2102        self.feature_flags.scoring_decision_with_validity_cutoff
2103    }
2104
2105    pub fn narwhal_versioned_metadata(&self) -> bool {
2106        self.feature_flags.narwhal_versioned_metadata
2107    }
2108
2109    pub fn consensus_order_end_of_epoch_last(&self) -> bool {
2110        self.feature_flags.consensus_order_end_of_epoch_last
2111    }
2112
2113    pub fn disallow_adding_abilities_on_upgrade(&self) -> bool {
2114        self.feature_flags.disallow_adding_abilities_on_upgrade
2115    }
2116
2117    pub fn disable_invariant_violation_check_in_swap_loc(&self) -> bool {
2118        self.feature_flags
2119            .disable_invariant_violation_check_in_swap_loc
2120    }
2121
2122    pub fn advance_to_highest_supported_protocol_version(&self) -> bool {
2123        self.feature_flags
2124            .advance_to_highest_supported_protocol_version
2125    }
2126
2127    pub fn ban_entry_init(&self) -> bool {
2128        self.feature_flags.ban_entry_init
2129    }
2130
2131    pub fn package_digest_hash_module(&self) -> bool {
2132        self.feature_flags.package_digest_hash_module
2133    }
2134
2135    pub fn disallow_change_struct_type_params_on_upgrade(&self) -> bool {
2136        self.feature_flags
2137            .disallow_change_struct_type_params_on_upgrade
2138    }
2139
2140    pub fn no_extraneous_module_bytes(&self) -> bool {
2141        self.feature_flags.no_extraneous_module_bytes
2142    }
2143
2144    pub fn zklogin_auth(&self) -> bool {
2145        self.feature_flags.zklogin_auth
2146    }
2147
2148    pub fn zklogin_supported_providers(&self) -> &BTreeSet<String> {
2149        &self.feature_flags.zklogin_supported_providers
2150    }
2151
2152    pub fn consensus_transaction_ordering(&self) -> ConsensusTransactionOrdering {
2153        self.feature_flags.consensus_transaction_ordering
2154    }
2155
2156    pub fn simplified_unwrap_then_delete(&self) -> bool {
2157        self.feature_flags.simplified_unwrap_then_delete
2158    }
2159
2160    pub fn supports_upgraded_multisig(&self) -> bool {
2161        self.feature_flags.upgraded_multisig_supported
2162    }
2163
2164    pub fn txn_base_cost_as_multiplier(&self) -> bool {
2165        self.feature_flags.txn_base_cost_as_multiplier
2166    }
2167
2168    pub fn shared_object_deletion(&self) -> bool {
2169        self.feature_flags.shared_object_deletion
2170    }
2171
2172    pub fn narwhal_new_leader_election_schedule(&self) -> bool {
2173        self.feature_flags.narwhal_new_leader_election_schedule
2174    }
2175
2176    pub fn loaded_child_object_format(&self) -> bool {
2177        self.feature_flags.loaded_child_object_format
2178    }
2179
2180    pub fn enable_jwk_consensus_updates(&self) -> bool {
2181        let ret = self.feature_flags.enable_jwk_consensus_updates;
2182        if ret {
2183            // jwk updates required end-of-epoch transactions
2184            assert!(self.feature_flags.end_of_epoch_transaction_supported);
2185        }
2186        ret
2187    }
2188
2189    pub fn simple_conservation_checks(&self) -> bool {
2190        self.feature_flags.simple_conservation_checks
2191    }
2192
2193    pub fn loaded_child_object_format_type(&self) -> bool {
2194        self.feature_flags.loaded_child_object_format_type
2195    }
2196
2197    pub fn end_of_epoch_transaction_supported(&self) -> bool {
2198        let ret = self.feature_flags.end_of_epoch_transaction_supported;
2199        if !ret {
2200            // jwk updates required end-of-epoch transactions
2201            assert!(!self.feature_flags.enable_jwk_consensus_updates);
2202        }
2203        ret
2204    }
2205
2206    pub fn recompute_has_public_transfer_in_execution(&self) -> bool {
2207        self.feature_flags
2208            .recompute_has_public_transfer_in_execution
2209    }
2210
2211    // this function only exists for readability in the genesis code.
2212    pub fn create_authenticator_state_in_genesis(&self) -> bool {
2213        self.enable_jwk_consensus_updates()
2214    }
2215
2216    pub fn random_beacon(&self) -> bool {
2217        self.feature_flags.random_beacon
2218    }
2219
2220    pub fn dkg_version(&self) -> u64 {
2221        // Version 0 was deprecated and removed, the default is 1 if not set.
2222        self.random_beacon_dkg_version.unwrap_or(1)
2223    }
2224
2225    pub fn enable_bridge(&self) -> bool {
2226        let ret = self.feature_flags.bridge;
2227        if ret {
2228            // bridge required end-of-epoch transactions
2229            assert!(self.feature_flags.end_of_epoch_transaction_supported);
2230        }
2231        ret
2232    }
2233
2234    pub fn should_try_to_finalize_bridge_committee(&self) -> bool {
2235        if !self.enable_bridge() {
2236            return false;
2237        }
2238        // In the older protocol version, always try to finalize the committee.
2239        self.bridge_should_try_to_finalize_committee.unwrap_or(true)
2240    }
2241
2242    pub fn enable_effects_v2(&self) -> bool {
2243        self.feature_flags.enable_effects_v2
2244    }
2245
2246    pub fn narwhal_certificate_v2(&self) -> bool {
2247        self.feature_flags.narwhal_certificate_v2
2248    }
2249
2250    pub fn verify_legacy_zklogin_address(&self) -> bool {
2251        self.feature_flags.verify_legacy_zklogin_address
2252    }
2253
2254    pub fn accept_zklogin_in_multisig(&self) -> bool {
2255        self.feature_flags.accept_zklogin_in_multisig
2256    }
2257
2258    pub fn accept_passkey_in_multisig(&self) -> bool {
2259        self.feature_flags.accept_passkey_in_multisig
2260    }
2261
2262    pub fn validate_zklogin_public_identifier(&self) -> bool {
2263        self.feature_flags.validate_zklogin_public_identifier
2264    }
2265
2266    pub fn zklogin_max_epoch_upper_bound_delta(&self) -> Option<u64> {
2267        self.feature_flags.zklogin_max_epoch_upper_bound_delta
2268    }
2269
2270    pub fn throughput_aware_consensus_submission(&self) -> bool {
2271        self.feature_flags.throughput_aware_consensus_submission
2272    }
2273
2274    pub fn include_consensus_digest_in_prologue(&self) -> bool {
2275        self.feature_flags.include_consensus_digest_in_prologue
2276    }
2277
2278    pub fn record_consensus_determined_version_assignments_in_prologue(&self) -> bool {
2279        self.feature_flags
2280            .record_consensus_determined_version_assignments_in_prologue
2281    }
2282
2283    pub fn record_additional_state_digest_in_prologue(&self) -> bool {
2284        self.feature_flags
2285            .record_additional_state_digest_in_prologue
2286    }
2287
2288    pub fn record_consensus_determined_version_assignments_in_prologue_v2(&self) -> bool {
2289        self.feature_flags
2290            .record_consensus_determined_version_assignments_in_prologue_v2
2291    }
2292
2293    pub fn prepend_prologue_tx_in_consensus_commit_in_checkpoints(&self) -> bool {
2294        self.feature_flags
2295            .prepend_prologue_tx_in_consensus_commit_in_checkpoints
2296    }
2297
2298    pub fn hardened_otw_check(&self) -> bool {
2299        self.feature_flags.hardened_otw_check
2300    }
2301
2302    pub fn enable_poseidon(&self) -> bool {
2303        self.feature_flags.enable_poseidon
2304    }
2305
2306    pub fn enable_coin_deny_list_v1(&self) -> bool {
2307        self.feature_flags.enable_coin_deny_list
2308    }
2309
2310    pub fn enable_accumulators(&self) -> bool {
2311        self.feature_flags.enable_accumulators
2312    }
2313
2314    pub fn enable_coin_reservation_obj_refs(&self) -> bool {
2315        self.new_vm_enabled() && self.feature_flags.enable_coin_reservation_obj_refs
2316    }
2317
2318    pub fn create_root_accumulator_object(&self) -> bool {
2319        self.feature_flags.create_root_accumulator_object
2320    }
2321
2322    pub fn enable_address_balance_gas_payments(&self) -> bool {
2323        self.feature_flags.enable_address_balance_gas_payments
2324    }
2325
2326    pub fn address_balance_gas_check_rgp_at_signing(&self) -> bool {
2327        self.feature_flags.address_balance_gas_check_rgp_at_signing
2328    }
2329
2330    pub fn address_balance_gas_reject_gas_coin_arg(&self) -> bool {
2331        self.feature_flags.address_balance_gas_reject_gas_coin_arg
2332    }
2333
2334    pub fn enable_multi_epoch_transaction_expiration(&self) -> bool {
2335        self.feature_flags.enable_multi_epoch_transaction_expiration
2336    }
2337
2338    pub fn relax_valid_during_for_owned_inputs(&self) -> bool {
2339        self.feature_flags.relax_valid_during_for_owned_inputs
2340    }
2341
2342    pub fn enable_authenticated_event_streams(&self) -> bool {
2343        self.feature_flags.enable_authenticated_event_streams && self.enable_accumulators()
2344    }
2345
2346    pub fn enable_non_exclusive_writes(&self) -> bool {
2347        self.feature_flags.enable_non_exclusive_writes
2348    }
2349
2350    pub fn enable_coin_registry(&self) -> bool {
2351        self.feature_flags.enable_coin_registry
2352    }
2353
2354    pub fn enable_display_registry(&self) -> bool {
2355        self.feature_flags.enable_display_registry
2356    }
2357
2358    pub fn enable_coin_deny_list_v2(&self) -> bool {
2359        self.feature_flags.enable_coin_deny_list_v2
2360    }
2361
2362    pub fn enable_group_ops_native_functions(&self) -> bool {
2363        self.feature_flags.enable_group_ops_native_functions
2364    }
2365
2366    pub fn enable_group_ops_native_function_msm(&self) -> bool {
2367        self.feature_flags.enable_group_ops_native_function_msm
2368    }
2369
2370    pub fn enable_ristretto255_group_ops(&self) -> bool {
2371        self.feature_flags.enable_ristretto255_group_ops
2372    }
2373
2374    pub fn enable_verify_bulletproofs_ristretto255(&self) -> bool {
2375        self.feature_flags.enable_verify_bulletproofs_ristretto255
2376    }
2377
2378    pub fn reject_mutable_random_on_entry_functions(&self) -> bool {
2379        self.feature_flags.reject_mutable_random_on_entry_functions
2380    }
2381
2382    pub fn per_object_congestion_control_mode(&self) -> PerObjectCongestionControlMode {
2383        self.feature_flags.per_object_congestion_control_mode
2384    }
2385
2386    pub fn consensus_choice(&self) -> ConsensusChoice {
2387        self.feature_flags.consensus_choice
2388    }
2389
2390    pub fn consensus_network(&self) -> ConsensusNetwork {
2391        self.feature_flags.consensus_network
2392    }
2393
2394    pub fn correct_gas_payment_limit_check(&self) -> bool {
2395        self.feature_flags.correct_gas_payment_limit_check
2396    }
2397
2398    pub fn reshare_at_same_initial_version(&self) -> bool {
2399        self.feature_flags.reshare_at_same_initial_version
2400    }
2401
2402    pub fn resolve_abort_locations_to_package_id(&self) -> bool {
2403        self.feature_flags.resolve_abort_locations_to_package_id
2404    }
2405
2406    pub fn mysticeti_use_committed_subdag_digest(&self) -> bool {
2407        self.feature_flags.mysticeti_use_committed_subdag_digest
2408    }
2409
2410    pub fn enable_vdf(&self) -> bool {
2411        self.feature_flags.enable_vdf
2412    }
2413
2414    pub fn fresh_vm_on_framework_upgrade(&self) -> bool {
2415        self.feature_flags.fresh_vm_on_framework_upgrade
2416    }
2417
2418    pub fn mysticeti_num_leaders_per_round(&self) -> Option<usize> {
2419        self.feature_flags.mysticeti_num_leaders_per_round
2420    }
2421
2422    pub fn soft_bundle(&self) -> bool {
2423        self.feature_flags.soft_bundle
2424    }
2425
2426    pub fn passkey_auth(&self) -> bool {
2427        self.feature_flags.passkey_auth
2428    }
2429
2430    pub fn authority_capabilities_v2(&self) -> bool {
2431        self.feature_flags.authority_capabilities_v2
2432    }
2433
2434    pub fn max_transaction_size_bytes(&self) -> u64 {
2435        // Provide a default value if protocol config version is too low.
2436        self.consensus_max_transaction_size_bytes
2437            .unwrap_or(256 * 1024)
2438    }
2439
2440    pub fn max_transactions_in_block_bytes(&self) -> u64 {
2441        if cfg!(msim) {
2442            256 * 1024
2443        } else {
2444            self.consensus_max_transactions_in_block_bytes
2445                .unwrap_or(512 * 1024)
2446        }
2447    }
2448
2449    pub fn max_num_transactions_in_block(&self) -> u64 {
2450        if cfg!(msim) {
2451            8
2452        } else {
2453            self.consensus_max_num_transactions_in_block.unwrap_or(512)
2454        }
2455    }
2456
2457    pub fn rethrow_serialization_type_layout_errors(&self) -> bool {
2458        self.feature_flags.rethrow_serialization_type_layout_errors
2459    }
2460
2461    pub fn consensus_distributed_vote_scoring_strategy(&self) -> bool {
2462        self.feature_flags
2463            .consensus_distributed_vote_scoring_strategy
2464    }
2465
2466    pub fn consensus_round_prober(&self) -> bool {
2467        self.feature_flags.consensus_round_prober
2468    }
2469
2470    pub fn validate_identifier_inputs(&self) -> bool {
2471        self.feature_flags.validate_identifier_inputs
2472    }
2473
2474    pub fn gc_depth(&self) -> u32 {
2475        self.consensus_gc_depth.unwrap_or(0)
2476    }
2477
2478    pub fn mysticeti_fastpath(&self) -> bool {
2479        self.feature_flags.mysticeti_fastpath
2480    }
2481
2482    pub fn relocate_event_module(&self) -> bool {
2483        self.feature_flags.relocate_event_module
2484    }
2485
2486    pub fn uncompressed_g1_group_elements(&self) -> bool {
2487        self.feature_flags.uncompressed_g1_group_elements
2488    }
2489
2490    pub fn disallow_new_modules_in_deps_only_packages(&self) -> bool {
2491        self.feature_flags
2492            .disallow_new_modules_in_deps_only_packages
2493    }
2494
2495    pub fn consensus_smart_ancestor_selection(&self) -> bool {
2496        self.feature_flags.consensus_smart_ancestor_selection
2497    }
2498
2499    pub fn disable_preconsensus_locking(&self) -> bool {
2500        self.feature_flags.disable_preconsensus_locking
2501    }
2502
2503    pub fn consensus_round_prober_probe_accepted_rounds(&self) -> bool {
2504        self.feature_flags
2505            .consensus_round_prober_probe_accepted_rounds
2506    }
2507
2508    pub fn native_charging_v2(&self) -> bool {
2509        self.feature_flags.native_charging_v2
2510    }
2511
2512    pub fn consensus_linearize_subdag_v2(&self) -> bool {
2513        let res = self.feature_flags.consensus_linearize_subdag_v2;
2514        assert!(
2515            !res || self.gc_depth() > 0,
2516            "The consensus linearize sub dag V2 requires GC to be enabled"
2517        );
2518        res
2519    }
2520
2521    pub fn consensus_median_based_commit_timestamp(&self) -> bool {
2522        let res = self.feature_flags.consensus_median_based_commit_timestamp;
2523        assert!(
2524            !res || self.gc_depth() > 0,
2525            "The consensus median based commit timestamp requires GC to be enabled"
2526        );
2527        res
2528    }
2529
2530    pub fn consensus_batched_block_sync(&self) -> bool {
2531        self.feature_flags.consensus_batched_block_sync
2532    }
2533
2534    pub fn convert_type_argument_error(&self) -> bool {
2535        self.feature_flags.convert_type_argument_error
2536    }
2537
2538    pub fn variant_nodes(&self) -> bool {
2539        self.feature_flags.variant_nodes
2540    }
2541
2542    pub fn consensus_zstd_compression(&self) -> bool {
2543        self.feature_flags.consensus_zstd_compression
2544    }
2545
2546    pub fn enable_nitro_attestation(&self) -> bool {
2547        self.feature_flags.enable_nitro_attestation
2548    }
2549
2550    pub fn enable_nitro_attestation_upgraded_parsing(&self) -> bool {
2551        self.feature_flags.enable_nitro_attestation_upgraded_parsing
2552    }
2553
2554    pub fn enable_nitro_attestation_all_nonzero_pcrs_parsing(&self) -> bool {
2555        self.feature_flags
2556            .enable_nitro_attestation_all_nonzero_pcrs_parsing
2557    }
2558
2559    pub fn enable_nitro_attestation_always_include_required_pcrs_parsing(&self) -> bool {
2560        self.feature_flags
2561            .enable_nitro_attestation_always_include_required_pcrs_parsing
2562    }
2563
2564    pub fn get_consensus_commit_rate_estimation_window_size(&self) -> u32 {
2565        self.consensus_commit_rate_estimation_window_size
2566            .unwrap_or(0)
2567    }
2568
2569    pub fn consensus_num_requested_prior_commits_at_startup(&self) -> u32 {
2570        // Currently there is only one parameter driving this value. If there are multiple
2571        // things computed from prior consensus commits, this function must return the max
2572        // of all of them.
2573        let window_size = self.get_consensus_commit_rate_estimation_window_size();
2574        // Ensure we are not using past commits without recording a state digest in the prologue.
2575        assert!(window_size == 0 || self.record_additional_state_digest_in_prologue());
2576        window_size
2577    }
2578
2579    pub fn minimize_child_object_mutations(&self) -> bool {
2580        self.feature_flags.minimize_child_object_mutations
2581    }
2582
2583    pub fn move_native_context(&self) -> bool {
2584        self.feature_flags.move_native_context
2585    }
2586
2587    pub fn normalize_ptb_arguments(&self) -> bool {
2588        self.feature_flags.normalize_ptb_arguments
2589    }
2590
2591    pub fn enforce_checkpoint_timestamp_monotonicity(&self) -> bool {
2592        self.feature_flags.enforce_checkpoint_timestamp_monotonicity
2593    }
2594
2595    pub fn max_ptb_value_size_v2(&self) -> bool {
2596        self.feature_flags.max_ptb_value_size_v2
2597    }
2598
2599    pub fn resolve_type_input_ids_to_defining_id(&self) -> bool {
2600        self.feature_flags.resolve_type_input_ids_to_defining_id
2601    }
2602
2603    pub fn enable_party_transfer(&self) -> bool {
2604        self.feature_flags.enable_party_transfer
2605    }
2606
2607    pub fn allow_unbounded_system_objects(&self) -> bool {
2608        self.feature_flags.allow_unbounded_system_objects
2609    }
2610
2611    pub fn type_tags_in_object_runtime(&self) -> bool {
2612        self.feature_flags.type_tags_in_object_runtime
2613    }
2614
2615    pub fn enable_ptb_execution_v2(&self) -> bool {
2616        self.feature_flags.enable_ptb_execution_v2
2617    }
2618
2619    pub fn better_adapter_type_resolution_errors(&self) -> bool {
2620        self.feature_flags.better_adapter_type_resolution_errors
2621    }
2622
2623    pub fn record_time_estimate_processed(&self) -> bool {
2624        self.feature_flags.record_time_estimate_processed
2625    }
2626
2627    pub fn ignore_execution_time_observations_after_certs_closed(&self) -> bool {
2628        self.feature_flags
2629            .ignore_execution_time_observations_after_certs_closed
2630    }
2631
2632    pub fn dependency_linkage_error(&self) -> bool {
2633        self.feature_flags.dependency_linkage_error
2634    }
2635
2636    pub fn additional_multisig_checks(&self) -> bool {
2637        self.feature_flags.additional_multisig_checks
2638    }
2639
2640    pub fn debug_fatal_on_move_invariant_violation(&self) -> bool {
2641        self.feature_flags.debug_fatal_on_move_invariant_violation
2642    }
2643
2644    pub fn allow_private_accumulator_entrypoints(&self) -> bool {
2645        self.feature_flags.allow_private_accumulator_entrypoints
2646    }
2647
2648    pub fn additional_consensus_digest_indirect_state(&self) -> bool {
2649        self.feature_flags
2650            .additional_consensus_digest_indirect_state
2651    }
2652
2653    pub fn check_for_init_during_upgrade(&self) -> bool {
2654        self.feature_flags.check_for_init_during_upgrade
2655    }
2656
2657    pub fn per_command_shared_object_transfer_rules(&self) -> bool {
2658        self.feature_flags.per_command_shared_object_transfer_rules
2659    }
2660
2661    pub fn consensus_checkpoint_signature_key_includes_digest(&self) -> bool {
2662        self.feature_flags
2663            .consensus_checkpoint_signature_key_includes_digest
2664    }
2665
2666    pub fn include_checkpoint_artifacts_digest_in_summary(&self) -> bool {
2667        self.feature_flags
2668            .include_checkpoint_artifacts_digest_in_summary
2669    }
2670
2671    pub fn use_mfp_txns_in_load_initial_object_debts(&self) -> bool {
2672        self.feature_flags.use_mfp_txns_in_load_initial_object_debts
2673    }
2674
2675    pub fn cancel_for_failed_dkg_early(&self) -> bool {
2676        self.feature_flags.cancel_for_failed_dkg_early
2677    }
2678
2679    pub fn always_advance_dkg_to_resolution(&self) -> bool {
2680        self.feature_flags.always_advance_dkg_to_resolution
2681    }
2682
2683    pub fn abstract_size_in_object_runtime(&self) -> bool {
2684        self.feature_flags.abstract_size_in_object_runtime
2685    }
2686
2687    pub fn object_runtime_charge_cache_load_gas(&self) -> bool {
2688        self.feature_flags.object_runtime_charge_cache_load_gas
2689    }
2690
2691    pub fn additional_borrow_checks(&self) -> bool {
2692        self.feature_flags.additional_borrow_checks
2693    }
2694
2695    pub fn use_new_commit_handler(&self) -> bool {
2696        self.feature_flags.use_new_commit_handler
2697    }
2698
2699    pub fn better_loader_errors(&self) -> bool {
2700        self.feature_flags.better_loader_errors
2701    }
2702
2703    pub fn generate_df_type_layouts(&self) -> bool {
2704        self.feature_flags.generate_df_type_layouts
2705    }
2706
2707    pub fn allow_references_in_ptbs(&self) -> bool {
2708        self.feature_flags.allow_references_in_ptbs
2709    }
2710
2711    pub fn private_generics_verifier_v2(&self) -> bool {
2712        self.feature_flags.private_generics_verifier_v2
2713    }
2714
2715    pub fn deprecate_global_storage_ops_during_deserialization(&self) -> bool {
2716        self.feature_flags
2717            .deprecate_global_storage_ops_during_deserialization
2718    }
2719
2720    pub fn enable_observation_chunking(&self) -> bool {
2721        matches!(self.feature_flags.per_object_congestion_control_mode,
2722            PerObjectCongestionControlMode::ExecutionTimeEstimate(ref params)
2723                if params.observations_chunk_size.is_some()
2724        )
2725    }
2726
2727    pub fn deprecate_global_storage_ops(&self) -> bool {
2728        self.feature_flags.deprecate_global_storage_ops
2729    }
2730
2731    pub fn normalize_depth_formula(&self) -> bool {
2732        self.feature_flags.normalize_depth_formula
2733    }
2734
2735    pub fn consensus_skip_gced_accept_votes(&self) -> bool {
2736        self.feature_flags.consensus_skip_gced_accept_votes
2737    }
2738
2739    pub fn include_cancelled_randomness_txns_in_prologue(&self) -> bool {
2740        self.feature_flags
2741            .include_cancelled_randomness_txns_in_prologue
2742    }
2743
2744    pub fn address_aliases(&self) -> bool {
2745        let address_aliases = self.feature_flags.address_aliases;
2746        assert!(
2747            !address_aliases || self.mysticeti_fastpath(),
2748            "Address aliases requires Mysticeti fastpath to be enabled"
2749        );
2750        if address_aliases {
2751            assert!(
2752                self.feature_flags.disable_preconsensus_locking,
2753                "Address aliases requires CertifiedTransaction to be disabled"
2754            );
2755        }
2756        address_aliases
2757    }
2758
2759    pub fn fix_checkpoint_signature_mapping(&self) -> bool {
2760        self.feature_flags.fix_checkpoint_signature_mapping
2761    }
2762
2763    pub fn enable_object_funds_withdraw(&self) -> bool {
2764        self.feature_flags.enable_object_funds_withdraw
2765    }
2766
2767    pub fn gas_rounding_halve_digits(&self) -> bool {
2768        self.feature_flags.gas_rounding_halve_digits
2769    }
2770
2771    pub fn flexible_tx_context_positions(&self) -> bool {
2772        self.feature_flags.flexible_tx_context_positions
2773    }
2774
2775    pub fn disable_entry_point_signature_check(&self) -> bool {
2776        self.feature_flags.disable_entry_point_signature_check
2777    }
2778
2779    pub fn consensus_skip_gced_blocks_in_direct_finalization(&self) -> bool {
2780        self.feature_flags
2781            .consensus_skip_gced_blocks_in_direct_finalization
2782    }
2783
2784    pub fn convert_withdrawal_compatibility_ptb_arguments(&self) -> bool {
2785        self.feature_flags
2786            .convert_withdrawal_compatibility_ptb_arguments
2787    }
2788
2789    pub fn restrict_hot_or_not_entry_functions(&self) -> bool {
2790        self.feature_flags.restrict_hot_or_not_entry_functions
2791    }
2792
2793    pub fn split_checkpoints_in_consensus_handler(&self) -> bool {
2794        self.feature_flags.split_checkpoints_in_consensus_handler
2795    }
2796
2797    pub fn consensus_always_accept_system_transactions(&self) -> bool {
2798        self.feature_flags
2799            .consensus_always_accept_system_transactions
2800    }
2801
2802    pub fn validator_metadata_verify_v2(&self) -> bool {
2803        self.feature_flags.validator_metadata_verify_v2
2804    }
2805
2806    pub fn defer_unpaid_amplification(&self) -> bool {
2807        self.feature_flags.defer_unpaid_amplification
2808    }
2809
2810    pub fn gasless_transaction_drop_safety(&self) -> bool {
2811        self.feature_flags.gasless_transaction_drop_safety
2812    }
2813
2814    pub fn new_vm_enabled(&self) -> bool {
2815        self.execution_version.is_some_and(|v| v >= 4)
2816    }
2817
2818    pub fn merge_randomness_into_checkpoint(&self) -> bool {
2819        self.feature_flags.merge_randomness_into_checkpoint
2820    }
2821
2822    pub fn use_coin_party_owner(&self) -> bool {
2823        self.feature_flags.use_coin_party_owner
2824    }
2825
2826    pub fn enable_gasless(&self) -> bool {
2827        self.feature_flags.enable_gasless
2828    }
2829
2830    pub fn gasless_verify_remaining_balance(&self) -> bool {
2831        self.feature_flags.gasless_verify_remaining_balance
2832    }
2833
2834    pub fn gasless_allowed_token_types(&self) -> &[(String, u64)] {
2835        debug_assert!(self.gasless_allowed_token_types.is_some());
2836        self.gasless_allowed_token_types.as_deref().unwrap_or(&[])
2837    }
2838
2839    pub fn get_gasless_max_unused_inputs(&self) -> u64 {
2840        self.gasless_max_unused_inputs.unwrap_or(u64::MAX)
2841    }
2842
2843    pub fn get_gasless_max_pure_input_bytes(&self) -> u64 {
2844        self.gasless_max_pure_input_bytes.unwrap_or(u64::MAX)
2845    }
2846
2847    pub fn get_gasless_max_tx_size_bytes(&self) -> u64 {
2848        self.gasless_max_tx_size_bytes.unwrap_or(u64::MAX)
2849    }
2850
2851    pub fn disallow_jump_orphans(&self) -> bool {
2852        self.feature_flags.disallow_jump_orphans
2853    }
2854
2855    pub fn early_return_receive_object_mismatched_type(&self) -> bool {
2856        self.feature_flags
2857            .early_return_receive_object_mismatched_type
2858    }
2859
2860    pub fn include_special_package_amendments_as_option(&self) -> &Option<Arc<Amendments>> {
2861        &self.include_special_package_amendments
2862    }
2863
2864    pub fn timestamp_based_epoch_close(&self) -> bool {
2865        self.feature_flags.timestamp_based_epoch_close
2866    }
2867
2868    pub fn limit_groth16_pvk_inputs(&self) -> bool {
2869        self.feature_flags.limit_groth16_pvk_inputs
2870    }
2871
2872    pub fn enforce_address_balance_change_invariant(&self) -> bool {
2873        self.feature_flags.enforce_address_balance_change_invariant
2874    }
2875
2876    pub fn granular_post_execution_checks(&self) -> bool {
2877        self.feature_flags.granular_post_execution_checks
2878    }
2879
2880    pub fn early_exit_on_iffw(&self) -> bool {
2881        self.feature_flags.early_exit_on_iffw
2882    }
2883}
2884
2885#[cfg(not(msim))]
2886static POISON_VERSION_METHODS: AtomicBool = AtomicBool::new(false);
2887
2888// Use a thread local in sim tests for test isolation.
2889#[cfg(msim)]
2890thread_local! {
2891    static POISON_VERSION_METHODS: AtomicBool = AtomicBool::new(false);
2892}
2893
2894// Instantiations for each protocol version.
2895impl ProtocolConfig {
2896    /// Get the value ProtocolConfig that are in effect during the given protocol version.
2897    pub fn get_for_version(version: ProtocolVersion, chain: Chain) -> Self {
2898        // ProtocolVersion can be deserialized so we need to check it here as well.
2899        assert!(
2900            version >= ProtocolVersion::MIN,
2901            "Network protocol version is {:?}, but the minimum supported version by the binary is {:?}. Please upgrade the binary.",
2902            version,
2903            ProtocolVersion::MIN.0,
2904        );
2905        assert!(
2906            version <= ProtocolVersion::MAX_ALLOWED,
2907            "Network protocol version is {:?}, but the maximum supported version by the binary is {:?}. Please upgrade the binary.",
2908            version,
2909            ProtocolVersion::MAX_ALLOWED.0,
2910        );
2911
2912        let mut ret = Self::get_for_version_impl(version, chain);
2913        ret.version = version;
2914
2915        ret = Self::apply_config_override(version, ret);
2916
2917        if std::env::var("SUI_PROTOCOL_CONFIG_OVERRIDE_ENABLE").is_ok() {
2918            warn!(
2919                "overriding ProtocolConfig settings with custom settings; this may break non-local networks"
2920            );
2921            let overrides: ProtocolConfigOptional =
2922                serde_env::from_env_with_prefix("SUI_PROTOCOL_CONFIG_OVERRIDE")
2923                    .expect("failed to parse ProtocolConfig override env variables");
2924            overrides.apply_to(&mut ret);
2925        }
2926
2927        ret
2928    }
2929
2930    /// Get the value ProtocolConfig that are in effect during the given protocol version.
2931    /// Or none if the version is not supported.
2932    pub fn get_for_version_if_supported(version: ProtocolVersion, chain: Chain) -> Option<Self> {
2933        if version.0 >= ProtocolVersion::MIN.0 && version.0 <= ProtocolVersion::MAX_ALLOWED.0 {
2934            let mut ret = Self::get_for_version_impl(version, chain);
2935            ret.version = version;
2936            ret = Self::apply_config_override(version, ret);
2937            Some(ret)
2938        } else {
2939            None
2940        }
2941    }
2942
2943    #[cfg(not(msim))]
2944    pub fn poison_get_for_min_version() {
2945        POISON_VERSION_METHODS.store(true, Ordering::Relaxed);
2946    }
2947
2948    #[cfg(not(msim))]
2949    fn load_poison_get_for_min_version() -> bool {
2950        POISON_VERSION_METHODS.load(Ordering::Relaxed)
2951    }
2952
2953    #[cfg(msim)]
2954    pub fn poison_get_for_min_version() {
2955        POISON_VERSION_METHODS.with(|p| p.store(true, Ordering::Relaxed));
2956    }
2957
2958    #[cfg(msim)]
2959    fn load_poison_get_for_min_version() -> bool {
2960        POISON_VERSION_METHODS.with(|p| p.load(Ordering::Relaxed))
2961    }
2962
2963    /// Convenience to get the constants at the current minimum supported version.
2964    /// Mainly used by client code that may not yet be protocol-version aware.
2965    pub fn get_for_min_version() -> Self {
2966        if Self::load_poison_get_for_min_version() {
2967            panic!("get_for_min_version called on validator");
2968        }
2969        ProtocolConfig::get_for_version(ProtocolVersion::MIN, Chain::Unknown)
2970    }
2971
2972    /// CAREFUL! - You probably want to use `get_for_version` instead.
2973    ///
2974    /// Convenience to get the constants at the current maximum supported version.
2975    /// Mainly used by genesis. Note well that this function uses the max version
2976    /// supported locally by the node, which is not necessarily the current version
2977    /// of the network. ALSO, this function disregards chain specific config (by
2978    /// using Chain::Unknown), thereby potentially returning a protocol config that
2979    /// is incorrect for some feature flags. Definitely safe for testing and for
2980    /// protocol version 11 and prior.
2981    #[allow(non_snake_case)]
2982    pub fn get_for_max_version_UNSAFE() -> Self {
2983        if Self::load_poison_get_for_min_version() {
2984            panic!("get_for_max_version_UNSAFE called on validator");
2985        }
2986        ProtocolConfig::get_for_version(ProtocolVersion::MAX, Chain::Unknown)
2987    }
2988
2989    fn get_for_version_impl(version: ProtocolVersion, chain: Chain) -> Self {
2990        #[cfg(msim)]
2991        {
2992            // populate the fake simulator version # with a different base tx cost.
2993            if version == ProtocolVersion::MAX_ALLOWED {
2994                let mut config = Self::get_for_version_impl(version - 1, Chain::Unknown);
2995                config.base_tx_cost_fixed = Some(config.base_tx_cost_fixed() + 1000);
2996                return config;
2997            }
2998        }
2999
3000        // IMPORTANT: Never modify the value of any constant for a pre-existing protocol version.
3001        // To change the values here you must create a new protocol version with the new values!
3002        let mut cfg = Self {
3003            // will be overwritten before being returned
3004            version,
3005
3006            // All flags are disabled in V1
3007            feature_flags: Default::default(),
3008
3009            max_tx_size_bytes: Some(128 * 1024),
3010            // We need this number to be at least 100x less than `max_serialized_tx_effects_size_bytes`otherwise effects can be huge
3011            max_input_objects: Some(2048),
3012            max_serialized_tx_effects_size_bytes: Some(512 * 1024),
3013            max_serialized_tx_effects_size_bytes_system_tx: Some(512 * 1024 * 16),
3014            max_gas_payment_objects: Some(256),
3015            max_modules_in_publish: Some(128),
3016            max_package_dependencies: None,
3017            max_arguments: Some(512),
3018            max_type_arguments: Some(16),
3019            max_type_argument_depth: Some(16),
3020            max_pure_argument_size: Some(16 * 1024),
3021            max_programmable_tx_commands: Some(1024),
3022            move_binary_format_version: Some(6),
3023            min_move_binary_format_version: None,
3024            binary_module_handles: None,
3025            binary_struct_handles: None,
3026            binary_function_handles: None,
3027            binary_function_instantiations: None,
3028            binary_signatures: None,
3029            binary_constant_pool: None,
3030            binary_identifiers: None,
3031            binary_address_identifiers: None,
3032            binary_struct_defs: None,
3033            binary_struct_def_instantiations: None,
3034            binary_function_defs: None,
3035            binary_field_handles: None,
3036            binary_field_instantiations: None,
3037            binary_friend_decls: None,
3038            binary_enum_defs: None,
3039            binary_enum_def_instantiations: None,
3040            binary_variant_handles: None,
3041            binary_variant_instantiation_handles: None,
3042            max_move_object_size: Some(250 * 1024),
3043            max_move_package_size: Some(100 * 1024),
3044            max_publish_or_upgrade_per_ptb: None,
3045            max_tx_gas: Some(10_000_000_000),
3046            max_gas_price: Some(100_000),
3047            max_gas_price_rgp_factor_for_aborted_transactions: None,
3048            max_gas_computation_bucket: Some(5_000_000),
3049            max_loop_depth: Some(5),
3050            max_generic_instantiation_length: Some(32),
3051            max_function_parameters: Some(128),
3052            max_basic_blocks: Some(1024),
3053            max_value_stack_size: Some(1024),
3054            max_type_nodes: Some(256),
3055            max_push_size: Some(10000),
3056            max_struct_definitions: Some(200),
3057            max_function_definitions: Some(1000),
3058            max_fields_in_struct: Some(32),
3059            max_dependency_depth: Some(100),
3060            max_num_event_emit: Some(256),
3061            max_num_new_move_object_ids: Some(2048),
3062            max_num_new_move_object_ids_system_tx: Some(2048 * 16),
3063            max_num_deleted_move_object_ids: Some(2048),
3064            max_num_deleted_move_object_ids_system_tx: Some(2048 * 16),
3065            max_num_transferred_move_object_ids: Some(2048),
3066            max_num_transferred_move_object_ids_system_tx: Some(2048 * 16),
3067            max_event_emit_size: Some(250 * 1024),
3068            max_move_vector_len: Some(256 * 1024),
3069            max_type_to_layout_nodes: None,
3070            max_ptb_value_size: None,
3071
3072            max_back_edges_per_function: Some(10_000),
3073            max_back_edges_per_module: Some(10_000),
3074            max_verifier_meter_ticks_per_function: Some(6_000_000),
3075            max_meter_ticks_per_module: Some(6_000_000),
3076            max_meter_ticks_per_package: None,
3077
3078            object_runtime_max_num_cached_objects: Some(1000),
3079            object_runtime_max_num_cached_objects_system_tx: Some(1000 * 16),
3080            object_runtime_max_num_store_entries: Some(1000),
3081            object_runtime_max_num_store_entries_system_tx: Some(1000 * 16),
3082            base_tx_cost_fixed: Some(110_000),
3083            package_publish_cost_fixed: Some(1_000),
3084            base_tx_cost_per_byte: Some(0),
3085            package_publish_cost_per_byte: Some(80),
3086            obj_access_cost_read_per_byte: Some(15),
3087            obj_access_cost_mutate_per_byte: Some(40),
3088            obj_access_cost_delete_per_byte: Some(40),
3089            obj_access_cost_verify_per_byte: Some(200),
3090            obj_data_cost_refundable: Some(100),
3091            obj_metadata_cost_non_refundable: Some(50),
3092            gas_model_version: Some(1),
3093            storage_rebate_rate: Some(9900),
3094            storage_fund_reinvest_rate: Some(500),
3095            reward_slashing_rate: Some(5000),
3096            storage_gas_price: Some(1),
3097            accumulator_object_storage_cost: None,
3098            max_transactions_per_checkpoint: Some(10_000),
3099            max_checkpoint_size_bytes: Some(30 * 1024 * 1024),
3100
3101            // For now, perform upgrades with a bare quorum of validators.
3102            // MUSTFIX: This number should be increased to at least 2000 (20%) for mainnet.
3103            buffer_stake_for_protocol_upgrade_bps: Some(0),
3104
3105            // === Native Function Costs ===
3106            // `address` module
3107            // Cost params for the Move native function `address::from_bytes(bytes: vector<u8>)`
3108            address_from_bytes_cost_base: Some(52),
3109            // Cost params for the Move native function `address::to_u256(address): u256`
3110            address_to_u256_cost_base: Some(52),
3111            // Cost params for the Move native function `address::from_u256(u256): address`
3112            address_from_u256_cost_base: Some(52),
3113
3114            // `config` module
3115            // Cost params for the Move native function `read_setting_impl``
3116            config_read_setting_impl_cost_base: None,
3117            config_read_setting_impl_cost_per_byte: None,
3118
3119            // `dynamic_field` module
3120            // Cost params for the Move native function `hash_type_and_key<K: copy + drop + store>(parent: address, k: K): address`
3121            dynamic_field_hash_type_and_key_cost_base: Some(100),
3122            dynamic_field_hash_type_and_key_type_cost_per_byte: Some(2),
3123            dynamic_field_hash_type_and_key_value_cost_per_byte: Some(2),
3124            dynamic_field_hash_type_and_key_type_tag_cost_per_byte: Some(2),
3125            // Cost params for the Move native function `add_child_object<Child: key>(parent: address, child: Child)`
3126            dynamic_field_add_child_object_cost_base: Some(100),
3127            dynamic_field_add_child_object_type_cost_per_byte: Some(10),
3128            dynamic_field_add_child_object_value_cost_per_byte: Some(10),
3129            dynamic_field_add_child_object_struct_tag_cost_per_byte: Some(10),
3130            // Cost params for the Move native function `borrow_child_object_mut<Child: key>(parent: &mut UID, id: address): &mut Child`
3131            dynamic_field_borrow_child_object_cost_base: Some(100),
3132            dynamic_field_borrow_child_object_child_ref_cost_per_byte: Some(10),
3133            dynamic_field_borrow_child_object_type_cost_per_byte: Some(10),
3134            // Cost params for the Move native function `remove_child_object<Child: key>(parent: address, id: address): Child`
3135            dynamic_field_remove_child_object_cost_base: Some(100),
3136            dynamic_field_remove_child_object_child_cost_per_byte: Some(2),
3137            dynamic_field_remove_child_object_type_cost_per_byte: Some(2),
3138            // Cost params for the Move native function `has_child_object(parent: address, id: address): bool`
3139            dynamic_field_has_child_object_cost_base: Some(100),
3140            // Cost params for the Move native function `has_child_object_with_ty<Child: key>(parent: address, id: address): bool`
3141            dynamic_field_has_child_object_with_ty_cost_base: Some(100),
3142            dynamic_field_has_child_object_with_ty_type_cost_per_byte: Some(2),
3143            dynamic_field_has_child_object_with_ty_type_tag_cost_per_byte: Some(2),
3144
3145            // `event` module
3146            // Cost params for the Move native function `event::emit<T: copy + drop>(event: T)`
3147            event_emit_cost_base: Some(52),
3148            event_emit_value_size_derivation_cost_per_byte: Some(2),
3149            event_emit_tag_size_derivation_cost_per_byte: Some(5),
3150            event_emit_output_cost_per_byte: Some(10),
3151            event_emit_auth_stream_cost: None,
3152
3153            //  `object` module
3154            // Cost params for the Move native function `borrow_uid<T: key>(obj: &T): &UID`
3155            object_borrow_uid_cost_base: Some(52),
3156            // Cost params for the Move native function `delete_impl(id: address)`
3157            object_delete_impl_cost_base: Some(52),
3158            // Cost params for the Move native function `record_new_uid(id: address)`
3159            object_record_new_uid_cost_base: Some(52),
3160
3161            // `transfer` module
3162            // Cost params for the Move native function `transfer_impl<T: key>(obj: T, recipient: address)`
3163            transfer_transfer_internal_cost_base: Some(52),
3164            // Cost params for the Move native function `party_transfer_impl<T: key>(obj: T, party_members: vector<address>)`
3165            transfer_party_transfer_internal_cost_base: None,
3166            // Cost params for the Move native function `freeze_object<T: key>(obj: T)`
3167            transfer_freeze_object_cost_base: Some(52),
3168            // Cost params for the Move native function `share_object<T: key>(obj: T)`
3169            transfer_share_object_cost_base: Some(52),
3170            transfer_receive_object_cost_base: None,
3171            transfer_receive_object_type_cost_per_byte: None,
3172            transfer_receive_object_cost_per_byte: None,
3173
3174            // `tx_context` module
3175            // Cost params for the Move native function `transfer_impl<T: key>(obj: T, recipient: address)`
3176            tx_context_derive_id_cost_base: Some(52),
3177            tx_context_fresh_id_cost_base: None,
3178            tx_context_sender_cost_base: None,
3179            tx_context_epoch_cost_base: None,
3180            tx_context_epoch_timestamp_ms_cost_base: None,
3181            tx_context_sponsor_cost_base: None,
3182            tx_context_rgp_cost_base: None,
3183            tx_context_gas_price_cost_base: None,
3184            tx_context_gas_budget_cost_base: None,
3185            tx_context_ids_created_cost_base: None,
3186            tx_context_replace_cost_base: None,
3187
3188            // `types` module
3189            // Cost params for the Move native function `is_one_time_witness<T: drop>(_: &T): bool`
3190            types_is_one_time_witness_cost_base: Some(52),
3191            types_is_one_time_witness_type_tag_cost_per_byte: Some(2),
3192            types_is_one_time_witness_type_cost_per_byte: Some(2),
3193
3194            // `validator` module
3195            // Cost params for the Move native function `validate_metadata_bcs(metadata: vector<u8>)`
3196            validator_validate_metadata_cost_base: Some(52),
3197            validator_validate_metadata_data_cost_per_byte: Some(2),
3198
3199            // Crypto
3200            crypto_invalid_arguments_cost: Some(100),
3201            // bls12381::bls12381_min_pk_verify
3202            bls12381_bls12381_min_sig_verify_cost_base: Some(52),
3203            bls12381_bls12381_min_sig_verify_msg_cost_per_byte: Some(2),
3204            bls12381_bls12381_min_sig_verify_msg_cost_per_block: Some(2),
3205
3206            // bls12381::bls12381_min_pk_verify
3207            bls12381_bls12381_min_pk_verify_cost_base: Some(52),
3208            bls12381_bls12381_min_pk_verify_msg_cost_per_byte: Some(2),
3209            bls12381_bls12381_min_pk_verify_msg_cost_per_block: Some(2),
3210
3211            // ecdsa_k1::ecrecover
3212            ecdsa_k1_ecrecover_keccak256_cost_base: Some(52),
3213            ecdsa_k1_ecrecover_keccak256_msg_cost_per_byte: Some(2),
3214            ecdsa_k1_ecrecover_keccak256_msg_cost_per_block: Some(2),
3215            ecdsa_k1_ecrecover_sha256_cost_base: Some(52),
3216            ecdsa_k1_ecrecover_sha256_msg_cost_per_byte: Some(2),
3217            ecdsa_k1_ecrecover_sha256_msg_cost_per_block: Some(2),
3218
3219            // ecdsa_k1::decompress_pubkey
3220            ecdsa_k1_decompress_pubkey_cost_base: Some(52),
3221
3222            // ecdsa_k1::secp256k1_verify
3223            ecdsa_k1_secp256k1_verify_keccak256_cost_base: Some(52),
3224            ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_byte: Some(2),
3225            ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_block: Some(2),
3226            ecdsa_k1_secp256k1_verify_sha256_cost_base: Some(52),
3227            ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_byte: Some(2),
3228            ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_block: Some(2),
3229
3230            // ecdsa_r1::ecrecover
3231            ecdsa_r1_ecrecover_keccak256_cost_base: Some(52),
3232            ecdsa_r1_ecrecover_keccak256_msg_cost_per_byte: Some(2),
3233            ecdsa_r1_ecrecover_keccak256_msg_cost_per_block: Some(2),
3234            ecdsa_r1_ecrecover_sha256_cost_base: Some(52),
3235            ecdsa_r1_ecrecover_sha256_msg_cost_per_byte: Some(2),
3236            ecdsa_r1_ecrecover_sha256_msg_cost_per_block: Some(2),
3237
3238            // ecdsa_r1::secp256k1_verify
3239            ecdsa_r1_secp256r1_verify_keccak256_cost_base: Some(52),
3240            ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_byte: Some(2),
3241            ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_block: Some(2),
3242            ecdsa_r1_secp256r1_verify_sha256_cost_base: Some(52),
3243            ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_byte: Some(2),
3244            ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_block: Some(2),
3245
3246            // ecvrf::verify
3247            ecvrf_ecvrf_verify_cost_base: Some(52),
3248            ecvrf_ecvrf_verify_alpha_string_cost_per_byte: Some(2),
3249            ecvrf_ecvrf_verify_alpha_string_cost_per_block: Some(2),
3250
3251            // ed25519
3252            ed25519_ed25519_verify_cost_base: Some(52),
3253            ed25519_ed25519_verify_msg_cost_per_byte: Some(2),
3254            ed25519_ed25519_verify_msg_cost_per_block: Some(2),
3255
3256            // groth16::prepare_verifying_key
3257            groth16_prepare_verifying_key_bls12381_cost_base: Some(52),
3258            groth16_prepare_verifying_key_bn254_cost_base: Some(52),
3259
3260            // groth16::verify_groth16_proof_internal
3261            groth16_verify_groth16_proof_internal_bls12381_cost_base: Some(52),
3262            groth16_verify_groth16_proof_internal_bls12381_cost_per_public_input: Some(2),
3263            groth16_verify_groth16_proof_internal_bn254_cost_base: Some(52),
3264            groth16_verify_groth16_proof_internal_bn254_cost_per_public_input: Some(2),
3265            groth16_verify_groth16_proof_internal_public_input_cost_per_byte: Some(2),
3266
3267            // hash::blake2b256
3268            hash_blake2b256_cost_base: Some(52),
3269            hash_blake2b256_data_cost_per_byte: Some(2),
3270            hash_blake2b256_data_cost_per_block: Some(2),
3271
3272            // hash::keccak256
3273            hash_keccak256_cost_base: Some(52),
3274            hash_keccak256_data_cost_per_byte: Some(2),
3275            hash_keccak256_data_cost_per_block: Some(2),
3276
3277            poseidon_bn254_cost_base: None,
3278            poseidon_bn254_cost_per_block: None,
3279
3280            // hmac::hmac_sha3_256
3281            hmac_hmac_sha3_256_cost_base: Some(52),
3282            hmac_hmac_sha3_256_input_cost_per_byte: Some(2),
3283            hmac_hmac_sha3_256_input_cost_per_block: Some(2),
3284
3285            // group ops
3286            group_ops_bls12381_decode_scalar_cost: None,
3287            group_ops_bls12381_decode_g1_cost: None,
3288            group_ops_bls12381_decode_g2_cost: None,
3289            group_ops_bls12381_decode_gt_cost: None,
3290            group_ops_bls12381_scalar_add_cost: None,
3291            group_ops_bls12381_g1_add_cost: None,
3292            group_ops_bls12381_g2_add_cost: None,
3293            group_ops_bls12381_gt_add_cost: None,
3294            group_ops_bls12381_scalar_sub_cost: None,
3295            group_ops_bls12381_g1_sub_cost: None,
3296            group_ops_bls12381_g2_sub_cost: None,
3297            group_ops_bls12381_gt_sub_cost: None,
3298            group_ops_bls12381_scalar_mul_cost: None,
3299            group_ops_bls12381_g1_mul_cost: None,
3300            group_ops_bls12381_g2_mul_cost: None,
3301            group_ops_bls12381_gt_mul_cost: None,
3302            group_ops_bls12381_scalar_div_cost: None,
3303            group_ops_bls12381_g1_div_cost: None,
3304            group_ops_bls12381_g2_div_cost: None,
3305            group_ops_bls12381_gt_div_cost: None,
3306            group_ops_bls12381_g1_hash_to_base_cost: None,
3307            group_ops_bls12381_g2_hash_to_base_cost: None,
3308            group_ops_bls12381_g1_hash_to_cost_per_byte: None,
3309            group_ops_bls12381_g2_hash_to_cost_per_byte: None,
3310            group_ops_bls12381_g1_msm_base_cost: None,
3311            group_ops_bls12381_g2_msm_base_cost: None,
3312            group_ops_bls12381_g1_msm_base_cost_per_input: None,
3313            group_ops_bls12381_g2_msm_base_cost_per_input: None,
3314            group_ops_bls12381_msm_max_len: None,
3315            group_ops_bls12381_pairing_cost: None,
3316            group_ops_bls12381_g1_to_uncompressed_g1_cost: None,
3317            group_ops_bls12381_uncompressed_g1_to_g1_cost: None,
3318            group_ops_bls12381_uncompressed_g1_sum_base_cost: None,
3319            group_ops_bls12381_uncompressed_g1_sum_cost_per_term: None,
3320            group_ops_bls12381_uncompressed_g1_sum_max_terms: None,
3321
3322            group_ops_ristretto_decode_scalar_cost: None,
3323            group_ops_ristretto_decode_point_cost: None,
3324            group_ops_ristretto_scalar_add_cost: None,
3325            group_ops_ristretto_point_add_cost: None,
3326            group_ops_ristretto_scalar_sub_cost: None,
3327            group_ops_ristretto_point_sub_cost: None,
3328            group_ops_ristretto_scalar_mul_cost: None,
3329            group_ops_ristretto_point_mul_cost: None,
3330            group_ops_ristretto_scalar_div_cost: None,
3331            group_ops_ristretto_point_div_cost: None,
3332
3333            verify_bulletproofs_ristretto255_base_cost: None,
3334            verify_bulletproofs_ristretto255_cost_per_bit_and_commitment: None,
3335
3336            // zklogin::check_zklogin_id
3337            check_zklogin_id_cost_base: None,
3338            // zklogin::check_zklogin_issuer
3339            check_zklogin_issuer_cost_base: None,
3340
3341            vdf_verify_vdf_cost: None,
3342            vdf_hash_to_input_cost: None,
3343
3344            // nitro_attestation::verify_nitro_attestation
3345            nitro_attestation_parse_base_cost: None,
3346            nitro_attestation_parse_cost_per_byte: None,
3347            nitro_attestation_verify_base_cost: None,
3348            nitro_attestation_verify_cost_per_cert: None,
3349
3350            bcs_per_byte_serialized_cost: None,
3351            bcs_legacy_min_output_size_cost: None,
3352            bcs_failure_cost: None,
3353            hash_sha2_256_base_cost: None,
3354            hash_sha2_256_per_byte_cost: None,
3355            hash_sha2_256_legacy_min_input_len_cost: None,
3356            hash_sha3_256_base_cost: None,
3357            hash_sha3_256_per_byte_cost: None,
3358            hash_sha3_256_legacy_min_input_len_cost: None,
3359            type_name_get_base_cost: None,
3360            type_name_get_per_byte_cost: None,
3361            type_name_id_base_cost: None,
3362            string_check_utf8_base_cost: None,
3363            string_check_utf8_per_byte_cost: None,
3364            string_is_char_boundary_base_cost: None,
3365            string_sub_string_base_cost: None,
3366            string_sub_string_per_byte_cost: None,
3367            string_index_of_base_cost: None,
3368            string_index_of_per_byte_pattern_cost: None,
3369            string_index_of_per_byte_searched_cost: None,
3370            vector_empty_base_cost: None,
3371            vector_length_base_cost: None,
3372            vector_push_back_base_cost: None,
3373            vector_push_back_legacy_per_abstract_memory_unit_cost: None,
3374            vector_borrow_base_cost: None,
3375            vector_pop_back_base_cost: None,
3376            vector_destroy_empty_base_cost: None,
3377            vector_swap_base_cost: None,
3378            debug_print_base_cost: None,
3379            debug_print_stack_trace_base_cost: None,
3380
3381            max_size_written_objects: None,
3382            max_size_written_objects_system_tx: None,
3383
3384            // ==== Ephemeral (consensus only) params deleted ====
3385            // Const params for consensus scoring decision
3386            // scoring_decision_mad_divisor: None,
3387            // scoring_decision_cutoff_value: None,
3388
3389            // Limits the length of a Move identifier
3390            max_move_identifier_len: None,
3391            max_move_value_depth: None,
3392            max_move_enum_variants: None,
3393
3394            gas_rounding_step: None,
3395
3396            execution_version: None,
3397
3398            max_event_emit_size_total: None,
3399
3400            consensus_bad_nodes_stake_threshold: None,
3401
3402            max_jwk_votes_per_validator_per_epoch: None,
3403
3404            max_age_of_jwk_in_epochs: None,
3405
3406            random_beacon_reduction_allowed_delta: None,
3407
3408            random_beacon_reduction_lower_bound: None,
3409
3410            random_beacon_dkg_timeout_round: None,
3411
3412            random_beacon_min_round_interval_ms: None,
3413
3414            random_beacon_dkg_version: None,
3415
3416            consensus_max_transaction_size_bytes: None,
3417
3418            consensus_max_transactions_in_block_bytes: None,
3419
3420            consensus_max_num_transactions_in_block: None,
3421
3422            consensus_voting_rounds: None,
3423
3424            max_accumulated_txn_cost_per_object_in_narwhal_commit: None,
3425
3426            max_deferral_rounds_for_congestion_control: None,
3427
3428            max_txn_cost_overage_per_object_in_commit: None,
3429
3430            allowed_txn_cost_overage_burst_per_object_in_commit: None,
3431
3432            min_checkpoint_interval_ms: None,
3433
3434            checkpoint_summary_version_specific_data: None,
3435
3436            max_soft_bundle_size: None,
3437
3438            bridge_should_try_to_finalize_committee: None,
3439
3440            max_accumulated_txn_cost_per_object_in_mysticeti_commit: None,
3441
3442            max_accumulated_randomness_txn_cost_per_object_in_mysticeti_commit: None,
3443
3444            consensus_gc_depth: None,
3445
3446            gas_budget_based_txn_cost_cap_factor: None,
3447
3448            gas_budget_based_txn_cost_absolute_cap_commit_count: None,
3449
3450            sip_45_consensus_amplification_threshold: None,
3451
3452            use_object_per_epoch_marker_table_v2: None,
3453
3454            consensus_commit_rate_estimation_window_size: None,
3455
3456            aliased_addresses: vec![],
3457
3458            translation_per_command_base_charge: None,
3459            translation_per_input_base_charge: None,
3460            translation_pure_input_per_byte_charge: None,
3461            translation_per_type_node_charge: None,
3462            translation_per_reference_node_charge: None,
3463            translation_per_linkage_entry_charge: None,
3464
3465            max_updates_per_settlement_txn: None,
3466
3467            gasless_max_computation_units: None,
3468            gasless_allowed_token_types: None,
3469            gasless_max_unused_inputs: None,
3470            gasless_max_pure_input_bytes: None,
3471            gasless_max_tps: None,
3472            include_special_package_amendments: None,
3473            gasless_max_tx_size_bytes: None,
3474            // When adding a new constant, set it to None in the earliest version, like this:
3475            // new_constant: None,
3476        };
3477        for cur in 2..=version.0 {
3478            match cur {
3479                1 => unreachable!(),
3480                2 => {
3481                    cfg.feature_flags.advance_epoch_start_time_in_safe_mode = true;
3482                }
3483                3 => {
3484                    // changes for gas model
3485                    cfg.gas_model_version = Some(2);
3486                    // max gas budget is in MIST and an absolute value 50SUI
3487                    cfg.max_tx_gas = Some(50_000_000_000);
3488                    // min gas budget is in MIST and an absolute value 2000MIST or 0.000002SUI
3489                    cfg.base_tx_cost_fixed = Some(2_000);
3490                    // storage gas price multiplier
3491                    cfg.storage_gas_price = Some(76);
3492                    cfg.feature_flags.loaded_child_objects_fixed = true;
3493                    // max size of written objects during a TXn
3494                    // this is a sum of all objects written during a TXn
3495                    cfg.max_size_written_objects = Some(5 * 1000 * 1000);
3496                    // max size of written objects during a system TXn to allow for larger writes
3497                    // akin to `max_size_written_objects` but for system TXns
3498                    cfg.max_size_written_objects_system_tx = Some(50 * 1000 * 1000);
3499                    cfg.feature_flags.package_upgrades = true;
3500                }
3501                // This is the first protocol version currently possible.
3502                // Mainnet starts with version 4. Previous versions are pre mainnet and have
3503                // all been wiped out.
3504                // Every other chain is after version 4.
3505                4 => {
3506                    // Change reward slashing rate to 100%.
3507                    cfg.reward_slashing_rate = Some(10000);
3508                    // protect old and new lookup for object version
3509                    cfg.gas_model_version = Some(3);
3510                }
3511                5 => {
3512                    cfg.feature_flags.missing_type_is_compatibility_error = true;
3513                    cfg.gas_model_version = Some(4);
3514                    cfg.feature_flags.scoring_decision_with_validity_cutoff = true;
3515                    // ==== Ephemeral (consensus only) params deleted ====
3516                    // cfg.scoring_decision_mad_divisor = Some(2.3);
3517                    // cfg.scoring_decision_cutoff_value = Some(2.5);
3518                }
3519                6 => {
3520                    cfg.gas_model_version = Some(5);
3521                    cfg.buffer_stake_for_protocol_upgrade_bps = Some(5000);
3522                    cfg.feature_flags.consensus_order_end_of_epoch_last = true;
3523                }
3524                7 => {
3525                    cfg.feature_flags.disallow_adding_abilities_on_upgrade = true;
3526                    cfg.feature_flags
3527                        .disable_invariant_violation_check_in_swap_loc = true;
3528                    cfg.feature_flags.ban_entry_init = true;
3529                    cfg.feature_flags.package_digest_hash_module = true;
3530                }
3531                8 => {
3532                    cfg.feature_flags
3533                        .disallow_change_struct_type_params_on_upgrade = true;
3534                }
3535                9 => {
3536                    // Limits the length of a Move identifier
3537                    cfg.max_move_identifier_len = Some(128);
3538                    cfg.feature_flags.no_extraneous_module_bytes = true;
3539                    cfg.feature_flags
3540                        .advance_to_highest_supported_protocol_version = true;
3541                }
3542                10 => {
3543                    cfg.max_verifier_meter_ticks_per_function = Some(16_000_000);
3544                    cfg.max_meter_ticks_per_module = Some(16_000_000);
3545                }
3546                11 => {
3547                    cfg.max_move_value_depth = Some(128);
3548                }
3549                12 => {
3550                    cfg.feature_flags.narwhal_versioned_metadata = true;
3551                    if chain != Chain::Mainnet {
3552                        cfg.feature_flags.commit_root_state_digest = true;
3553                    }
3554
3555                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3556                        cfg.feature_flags.zklogin_auth = true;
3557                    }
3558                }
3559                13 => {}
3560                14 => {
3561                    cfg.gas_rounding_step = Some(1_000);
3562                    cfg.gas_model_version = Some(6);
3563                }
3564                15 => {
3565                    cfg.feature_flags.consensus_transaction_ordering =
3566                        ConsensusTransactionOrdering::ByGasPrice;
3567                }
3568                16 => {
3569                    cfg.feature_flags.simplified_unwrap_then_delete = true;
3570                }
3571                17 => {
3572                    cfg.feature_flags.upgraded_multisig_supported = true;
3573                }
3574                18 => {
3575                    cfg.execution_version = Some(1);
3576                    // Following flags are implied by this execution version.  Once support for earlier
3577                    // protocol versions is dropped, these flags can be removed:
3578                    // cfg.feature_flags.package_upgrades = true;
3579                    // cfg.feature_flags.disallow_adding_abilities_on_upgrade = true;
3580                    // cfg.feature_flags.disallow_change_struct_type_params_on_upgrade = true;
3581                    // cfg.feature_flags.loaded_child_objects_fixed = true;
3582                    // cfg.feature_flags.ban_entry_init = true;
3583                    // cfg.feature_flags.pack_digest_hash_modules = true;
3584                    cfg.feature_flags.txn_base_cost_as_multiplier = true;
3585                    // this is a multiplier of the gas price
3586                    cfg.base_tx_cost_fixed = Some(1_000);
3587                }
3588                19 => {
3589                    cfg.max_num_event_emit = Some(1024);
3590                    // We maintain the same total size limit for events, but increase the number of
3591                    // events that can be emitted.
3592                    cfg.max_event_emit_size_total = Some(
3593                        256 /* former event count limit */ * 250 * 1024, /* size limit per event */
3594                    );
3595                }
3596                20 => {
3597                    cfg.feature_flags.commit_root_state_digest = true;
3598
3599                    if chain != Chain::Mainnet {
3600                        cfg.feature_flags.narwhal_new_leader_election_schedule = true;
3601                        cfg.consensus_bad_nodes_stake_threshold = Some(20);
3602                    }
3603                }
3604
3605                21 => {
3606                    if chain != Chain::Mainnet {
3607                        cfg.feature_flags.zklogin_supported_providers = BTreeSet::from([
3608                            "Google".to_string(),
3609                            "Facebook".to_string(),
3610                            "Twitch".to_string(),
3611                        ]);
3612                    }
3613                }
3614                22 => {
3615                    cfg.feature_flags.loaded_child_object_format = true;
3616                }
3617                23 => {
3618                    cfg.feature_flags.loaded_child_object_format_type = true;
3619                    cfg.feature_flags.narwhal_new_leader_election_schedule = true;
3620                    // Taking a baby step approach, we consider only 20% by stake as bad nodes so we
3621                    // have a 80% by stake of nodes participating in the leader committee. That allow
3622                    // us for more redundancy in case we have validators under performing - since the
3623                    // responsibility is shared amongst more nodes. We can increase that once we do have
3624                    // higher confidence.
3625                    cfg.consensus_bad_nodes_stake_threshold = Some(20);
3626                }
3627                24 => {
3628                    cfg.feature_flags.simple_conservation_checks = true;
3629                    cfg.max_publish_or_upgrade_per_ptb = Some(5);
3630
3631                    cfg.feature_flags.end_of_epoch_transaction_supported = true;
3632
3633                    if chain != Chain::Mainnet {
3634                        cfg.feature_flags.enable_jwk_consensus_updates = true;
3635                        // Max of 10 votes per hour
3636                        cfg.max_jwk_votes_per_validator_per_epoch = Some(240);
3637                        cfg.max_age_of_jwk_in_epochs = Some(1);
3638                    }
3639                }
3640                25 => {
3641                    // Enable zkLogin for all providers in all networks.
3642                    cfg.feature_flags.zklogin_supported_providers = BTreeSet::from([
3643                        "Google".to_string(),
3644                        "Facebook".to_string(),
3645                        "Twitch".to_string(),
3646                    ]);
3647                    cfg.feature_flags.zklogin_auth = true;
3648
3649                    // Enable jwk consensus updates
3650                    cfg.feature_flags.enable_jwk_consensus_updates = true;
3651                    cfg.max_jwk_votes_per_validator_per_epoch = Some(240);
3652                    cfg.max_age_of_jwk_in_epochs = Some(1);
3653                }
3654                26 => {
3655                    cfg.gas_model_version = Some(7);
3656                    // Only enable receiving objects in devnet
3657                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3658                        cfg.transfer_receive_object_cost_base = Some(52);
3659                        cfg.feature_flags.receive_objects = true;
3660                    }
3661                }
3662                27 => {
3663                    cfg.gas_model_version = Some(8);
3664                }
3665                28 => {
3666                    // zklogin::check_zklogin_id
3667                    cfg.check_zklogin_id_cost_base = Some(200);
3668                    // zklogin::check_zklogin_issuer
3669                    cfg.check_zklogin_issuer_cost_base = Some(200);
3670
3671                    // Only enable effects v2 on devnet.
3672                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3673                        cfg.feature_flags.enable_effects_v2 = true;
3674                    }
3675                }
3676                29 => {
3677                    cfg.feature_flags.verify_legacy_zklogin_address = true;
3678                }
3679                30 => {
3680                    // Only enable nw certificate v2 on testnet.
3681                    if chain != Chain::Mainnet {
3682                        cfg.feature_flags.narwhal_certificate_v2 = true;
3683                    }
3684
3685                    cfg.random_beacon_reduction_allowed_delta = Some(800);
3686                    // Only enable effects v2 on devnet and testnet.
3687                    if chain != Chain::Mainnet {
3688                        cfg.feature_flags.enable_effects_v2 = true;
3689                    }
3690
3691                    // zklogin_supported_providers config is deprecated, zklogin
3692                    // signature verifier will use the fetched jwk map to determine
3693                    // whether the provider is supported based on node config.
3694                    cfg.feature_flags.zklogin_supported_providers = BTreeSet::default();
3695
3696                    cfg.feature_flags.recompute_has_public_transfer_in_execution = true;
3697                }
3698                31 => {
3699                    cfg.execution_version = Some(2);
3700                    // Only enable shared object deletion on devnet
3701                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3702                        cfg.feature_flags.shared_object_deletion = true;
3703                    }
3704                }
3705                32 => {
3706                    // enable zklogin in multisig in devnet and testnet
3707                    if chain != Chain::Mainnet {
3708                        cfg.feature_flags.accept_zklogin_in_multisig = true;
3709                    }
3710                    // enable receiving objects in devnet and testnet
3711                    if chain != Chain::Mainnet {
3712                        cfg.transfer_receive_object_cost_base = Some(52);
3713                        cfg.feature_flags.receive_objects = true;
3714                    }
3715                    // Only enable random beacon on devnet
3716                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3717                        cfg.feature_flags.random_beacon = true;
3718                        cfg.random_beacon_reduction_lower_bound = Some(1600);
3719                        cfg.random_beacon_dkg_timeout_round = Some(3000);
3720                        cfg.random_beacon_min_round_interval_ms = Some(150);
3721                    }
3722                    // Only enable consensus digest in consensus commit prologue in devnet.
3723                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3724                        cfg.feature_flags.include_consensus_digest_in_prologue = true;
3725                    }
3726
3727                    // enable nw cert v2 on mainnet
3728                    cfg.feature_flags.narwhal_certificate_v2 = true;
3729                }
3730                33 => {
3731                    cfg.feature_flags.hardened_otw_check = true;
3732                    cfg.feature_flags.allow_receiving_object_id = true;
3733
3734                    // Enable transfer-to-object in mainnet
3735                    cfg.transfer_receive_object_cost_base = Some(52);
3736                    cfg.feature_flags.receive_objects = true;
3737
3738                    // Enable shared object deletion in testnet and devnet
3739                    if chain != Chain::Mainnet {
3740                        cfg.feature_flags.shared_object_deletion = true;
3741                    }
3742
3743                    cfg.feature_flags.enable_effects_v2 = true;
3744                }
3745                34 => {}
3746                35 => {
3747                    // Add costs for poseidon::poseidon_bn254
3748                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3749                        cfg.feature_flags.enable_poseidon = true;
3750                        cfg.poseidon_bn254_cost_base = Some(260);
3751                        cfg.poseidon_bn254_cost_per_block = Some(10);
3752                    }
3753
3754                    cfg.feature_flags.enable_coin_deny_list = true;
3755                }
3756                36 => {
3757                    // Only enable group ops on devnet
3758                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3759                        cfg.feature_flags.enable_group_ops_native_functions = true;
3760                        cfg.feature_flags.enable_group_ops_native_function_msm = true;
3761                        // Next values are arbitrary in a similar way as the other crypto native functions.
3762                        cfg.group_ops_bls12381_decode_scalar_cost = Some(52);
3763                        cfg.group_ops_bls12381_decode_g1_cost = Some(52);
3764                        cfg.group_ops_bls12381_decode_g2_cost = Some(52);
3765                        cfg.group_ops_bls12381_decode_gt_cost = Some(52);
3766                        cfg.group_ops_bls12381_scalar_add_cost = Some(52);
3767                        cfg.group_ops_bls12381_g1_add_cost = Some(52);
3768                        cfg.group_ops_bls12381_g2_add_cost = Some(52);
3769                        cfg.group_ops_bls12381_gt_add_cost = Some(52);
3770                        cfg.group_ops_bls12381_scalar_sub_cost = Some(52);
3771                        cfg.group_ops_bls12381_g1_sub_cost = Some(52);
3772                        cfg.group_ops_bls12381_g2_sub_cost = Some(52);
3773                        cfg.group_ops_bls12381_gt_sub_cost = Some(52);
3774                        cfg.group_ops_bls12381_scalar_mul_cost = Some(52);
3775                        cfg.group_ops_bls12381_g1_mul_cost = Some(52);
3776                        cfg.group_ops_bls12381_g2_mul_cost = Some(52);
3777                        cfg.group_ops_bls12381_gt_mul_cost = Some(52);
3778                        cfg.group_ops_bls12381_scalar_div_cost = Some(52);
3779                        cfg.group_ops_bls12381_g1_div_cost = Some(52);
3780                        cfg.group_ops_bls12381_g2_div_cost = Some(52);
3781                        cfg.group_ops_bls12381_gt_div_cost = Some(52);
3782                        cfg.group_ops_bls12381_g1_hash_to_base_cost = Some(52);
3783                        cfg.group_ops_bls12381_g2_hash_to_base_cost = Some(52);
3784                        cfg.group_ops_bls12381_g1_hash_to_cost_per_byte = Some(2);
3785                        cfg.group_ops_bls12381_g2_hash_to_cost_per_byte = Some(2);
3786                        cfg.group_ops_bls12381_g1_msm_base_cost = Some(52);
3787                        cfg.group_ops_bls12381_g2_msm_base_cost = Some(52);
3788                        cfg.group_ops_bls12381_g1_msm_base_cost_per_input = Some(52);
3789                        cfg.group_ops_bls12381_g2_msm_base_cost_per_input = Some(52);
3790                        cfg.group_ops_bls12381_msm_max_len = Some(32);
3791                        cfg.group_ops_bls12381_pairing_cost = Some(52);
3792                    }
3793                    // Enable shared object deletion on all networks.
3794                    cfg.feature_flags.shared_object_deletion = true;
3795
3796                    cfg.consensus_max_transaction_size_bytes = Some(256 * 1024); // 256KB
3797                    cfg.consensus_max_transactions_in_block_bytes = Some(6 * 1_024 * 1024);
3798                    // 6 MB
3799                }
3800                37 => {
3801                    cfg.feature_flags.reject_mutable_random_on_entry_functions = true;
3802
3803                    // Enable consensus digest in consensus commit prologue in testnet and devnet.
3804                    if chain != Chain::Mainnet {
3805                        cfg.feature_flags.include_consensus_digest_in_prologue = true;
3806                    }
3807                }
3808                38 => {
3809                    cfg.binary_module_handles = Some(100);
3810                    cfg.binary_struct_handles = Some(300);
3811                    cfg.binary_function_handles = Some(1500);
3812                    cfg.binary_function_instantiations = Some(750);
3813                    cfg.binary_signatures = Some(1000);
3814                    // constants and identifiers are proportional to the binary size,
3815                    // and they vastly depend on the code, so we are leaving them
3816                    // reasonably high
3817                    cfg.binary_constant_pool = Some(4000);
3818                    cfg.binary_identifiers = Some(10000);
3819                    cfg.binary_address_identifiers = Some(100);
3820                    cfg.binary_struct_defs = Some(200);
3821                    cfg.binary_struct_def_instantiations = Some(100);
3822                    cfg.binary_function_defs = Some(1000);
3823                    cfg.binary_field_handles = Some(500);
3824                    cfg.binary_field_instantiations = Some(250);
3825                    cfg.binary_friend_decls = Some(100);
3826                    // reduce dependencies maximum
3827                    cfg.max_package_dependencies = Some(32);
3828                    cfg.max_modules_in_publish = Some(64);
3829                    // bump execution version
3830                    cfg.execution_version = Some(3);
3831                }
3832                39 => {
3833                    // It is important that we keep this protocol version blank due to an issue with random.move.
3834                }
3835                40 => {}
3836                41 => {
3837                    // Enable group ops and all networks (but not msm)
3838                    cfg.feature_flags.enable_group_ops_native_functions = true;
3839                    // Next values are arbitrary in a similar way as the other crypto native functions.
3840                    cfg.group_ops_bls12381_decode_scalar_cost = Some(52);
3841                    cfg.group_ops_bls12381_decode_g1_cost = Some(52);
3842                    cfg.group_ops_bls12381_decode_g2_cost = Some(52);
3843                    cfg.group_ops_bls12381_decode_gt_cost = Some(52);
3844                    cfg.group_ops_bls12381_scalar_add_cost = Some(52);
3845                    cfg.group_ops_bls12381_g1_add_cost = Some(52);
3846                    cfg.group_ops_bls12381_g2_add_cost = Some(52);
3847                    cfg.group_ops_bls12381_gt_add_cost = Some(52);
3848                    cfg.group_ops_bls12381_scalar_sub_cost = Some(52);
3849                    cfg.group_ops_bls12381_g1_sub_cost = Some(52);
3850                    cfg.group_ops_bls12381_g2_sub_cost = Some(52);
3851                    cfg.group_ops_bls12381_gt_sub_cost = Some(52);
3852                    cfg.group_ops_bls12381_scalar_mul_cost = Some(52);
3853                    cfg.group_ops_bls12381_g1_mul_cost = Some(52);
3854                    cfg.group_ops_bls12381_g2_mul_cost = Some(52);
3855                    cfg.group_ops_bls12381_gt_mul_cost = Some(52);
3856                    cfg.group_ops_bls12381_scalar_div_cost = Some(52);
3857                    cfg.group_ops_bls12381_g1_div_cost = Some(52);
3858                    cfg.group_ops_bls12381_g2_div_cost = Some(52);
3859                    cfg.group_ops_bls12381_gt_div_cost = Some(52);
3860                    cfg.group_ops_bls12381_g1_hash_to_base_cost = Some(52);
3861                    cfg.group_ops_bls12381_g2_hash_to_base_cost = Some(52);
3862                    cfg.group_ops_bls12381_g1_hash_to_cost_per_byte = Some(2);
3863                    cfg.group_ops_bls12381_g2_hash_to_cost_per_byte = Some(2);
3864                    cfg.group_ops_bls12381_g1_msm_base_cost = Some(52);
3865                    cfg.group_ops_bls12381_g2_msm_base_cost = Some(52);
3866                    cfg.group_ops_bls12381_g1_msm_base_cost_per_input = Some(52);
3867                    cfg.group_ops_bls12381_g2_msm_base_cost_per_input = Some(52);
3868                    cfg.group_ops_bls12381_msm_max_len = Some(32);
3869                    cfg.group_ops_bls12381_pairing_cost = Some(52);
3870                }
3871                42 => {}
3872                43 => {
3873                    cfg.feature_flags.zklogin_max_epoch_upper_bound_delta = Some(30);
3874                    cfg.max_meter_ticks_per_package = Some(16_000_000);
3875                }
3876                44 => {
3877                    // Enable consensus digest in consensus commit prologue on all networks..
3878                    cfg.feature_flags.include_consensus_digest_in_prologue = true;
3879                    // Switch between Narwhal and Mysticeti per epoch in tests, devnet and testnet.
3880                    if chain != Chain::Mainnet {
3881                        cfg.feature_flags.consensus_choice = ConsensusChoice::SwapEachEpoch;
3882                    }
3883                }
3884                45 => {
3885                    // Use tonic networking for consensus, in tests and devnet.
3886                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3887                        cfg.feature_flags.consensus_network = ConsensusNetwork::Tonic;
3888                    }
3889
3890                    if chain != Chain::Mainnet {
3891                        // Enable leader scoring & schedule change on testnet for mysticeti.
3892                        cfg.feature_flags.mysticeti_leader_scoring_and_schedule = true;
3893                    }
3894                    cfg.min_move_binary_format_version = Some(6);
3895                    cfg.feature_flags.accept_zklogin_in_multisig = true;
3896
3897                    // Also bumps framework snapshot to fix binop issue.
3898
3899                    // enable bridge in devnet
3900                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3901                        cfg.feature_flags.bridge = true;
3902                    }
3903                }
3904                46 => {
3905                    // enable bridge in devnet and testnet
3906                    if chain != Chain::Mainnet {
3907                        cfg.feature_flags.bridge = true;
3908                    }
3909
3910                    // Enable resharing at same initial version
3911                    cfg.feature_flags.reshare_at_same_initial_version = true;
3912                }
3913                47 => {}
3914                48 => {
3915                    // Use tonic networking for Mysticeti.
3916                    cfg.feature_flags.consensus_network = ConsensusNetwork::Tonic;
3917
3918                    // Enable resolving abort code IDs to package ID instead of runtime module ID
3919                    cfg.feature_flags.resolve_abort_locations_to_package_id = true;
3920
3921                    // Enable random beacon on testnet.
3922                    if chain != Chain::Mainnet {
3923                        cfg.feature_flags.random_beacon = true;
3924                        cfg.random_beacon_reduction_lower_bound = Some(1600);
3925                        cfg.random_beacon_dkg_timeout_round = Some(3000);
3926                        cfg.random_beacon_min_round_interval_ms = Some(200);
3927                    }
3928
3929                    // Enable the committed sub dag digest inclusion on the commit output
3930                    cfg.feature_flags.mysticeti_use_committed_subdag_digest = true;
3931                }
3932                49 => {
3933                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3934                        cfg.move_binary_format_version = Some(7);
3935                    }
3936
3937                    // enable vdf in devnet
3938                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3939                        cfg.feature_flags.enable_vdf = true;
3940                        // Set to 30x and 2x the cost of a signature verification for now. This
3941                        // should be updated along with other native crypto functions.
3942                        cfg.vdf_verify_vdf_cost = Some(1500);
3943                        cfg.vdf_hash_to_input_cost = Some(100);
3944                    }
3945
3946                    // Only enable consensus commit prologue V3 in devnet.
3947                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3948                        cfg.feature_flags
3949                            .record_consensus_determined_version_assignments_in_prologue = true;
3950                    }
3951
3952                    // Run Mysticeti consensus in testnet.
3953                    if chain != Chain::Mainnet {
3954                        cfg.feature_flags.consensus_choice = ConsensusChoice::Mysticeti;
3955                    }
3956
3957                    // Run Move verification on framework upgrades in its own VM
3958                    cfg.feature_flags.fresh_vm_on_framework_upgrade = true;
3959                }
3960                50 => {
3961                    // Enable checkpoint batching in testnet.
3962                    if chain != Chain::Mainnet {
3963                        cfg.checkpoint_summary_version_specific_data = Some(1);
3964                        cfg.min_checkpoint_interval_ms = Some(200);
3965                    }
3966
3967                    // Only enable prepose consensus commit prologue in checkpoints in devnet.
3968                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3969                        cfg.feature_flags
3970                            .prepend_prologue_tx_in_consensus_commit_in_checkpoints = true;
3971                    }
3972
3973                    cfg.feature_flags.mysticeti_num_leaders_per_round = Some(1);
3974
3975                    // Set max transaction deferral to 10 consensus rounds.
3976                    cfg.max_deferral_rounds_for_congestion_control = Some(10);
3977                }
3978                51 => {
3979                    cfg.random_beacon_dkg_version = Some(1);
3980
3981                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3982                        cfg.feature_flags.enable_coin_deny_list_v2 = true;
3983                    }
3984                }
3985                52 => {
3986                    if chain != Chain::Mainnet {
3987                        cfg.feature_flags.soft_bundle = true;
3988                        cfg.max_soft_bundle_size = Some(5);
3989                    }
3990
3991                    cfg.config_read_setting_impl_cost_base = Some(100);
3992                    cfg.config_read_setting_impl_cost_per_byte = Some(40);
3993
3994                    // Turn on shared object congestion control in devnet.
3995                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3996                        cfg.max_accumulated_txn_cost_per_object_in_narwhal_commit = Some(100);
3997                        cfg.feature_flags.per_object_congestion_control_mode =
3998                            PerObjectCongestionControlMode::TotalTxCount;
3999                    }
4000
4001                    // Enable Mysticeti on mainnet.
4002                    cfg.feature_flags.consensus_choice = ConsensusChoice::Mysticeti;
4003
4004                    // Enable leader scoring & schedule change on mainnet for mysticeti.
4005                    cfg.feature_flags.mysticeti_leader_scoring_and_schedule = true;
4006
4007                    // Enable checkpoint batching on mainnet.
4008                    cfg.checkpoint_summary_version_specific_data = Some(1);
4009                    cfg.min_checkpoint_interval_ms = Some(200);
4010
4011                    // Enable consensus commit prologue V3 in testnet.
4012                    if chain != Chain::Mainnet {
4013                        cfg.feature_flags
4014                            .record_consensus_determined_version_assignments_in_prologue = true;
4015                        cfg.feature_flags
4016                            .prepend_prologue_tx_in_consensus_commit_in_checkpoints = true;
4017                    }
4018                    // Turn on enums in testnet and devnet
4019                    if chain != Chain::Mainnet {
4020                        cfg.move_binary_format_version = Some(7);
4021                    }
4022
4023                    if chain != Chain::Testnet && chain != Chain::Mainnet {
4024                        cfg.feature_flags.passkey_auth = true;
4025                    }
4026                    cfg.feature_flags.enable_coin_deny_list_v2 = true;
4027                }
4028                53 => {
4029                    // Do not allow bridge committee to finalize on mainnet.
4030                    cfg.bridge_should_try_to_finalize_committee = Some(chain != Chain::Mainnet);
4031
4032                    // Enable consensus commit prologue V3 on mainnet.
4033                    cfg.feature_flags
4034                        .record_consensus_determined_version_assignments_in_prologue = true;
4035                    cfg.feature_flags
4036                        .prepend_prologue_tx_in_consensus_commit_in_checkpoints = true;
4037
4038                    if chain == Chain::Unknown {
4039                        cfg.feature_flags.authority_capabilities_v2 = true;
4040                    }
4041
4042                    // Turns on shared object congestion control on testnet.
4043                    if chain != Chain::Mainnet {
4044                        cfg.max_accumulated_txn_cost_per_object_in_narwhal_commit = Some(100);
4045                        cfg.max_accumulated_txn_cost_per_object_in_mysticeti_commit = Some(10);
4046                        cfg.feature_flags.per_object_congestion_control_mode =
4047                            PerObjectCongestionControlMode::TotalTxCount;
4048                    }
4049
4050                    // Adjust stdlib gas costs
4051                    cfg.bcs_per_byte_serialized_cost = Some(2);
4052                    cfg.bcs_legacy_min_output_size_cost = Some(1);
4053                    cfg.bcs_failure_cost = Some(52);
4054                    cfg.debug_print_base_cost = Some(52);
4055                    cfg.debug_print_stack_trace_base_cost = Some(52);
4056                    cfg.hash_sha2_256_base_cost = Some(52);
4057                    cfg.hash_sha2_256_per_byte_cost = Some(2);
4058                    cfg.hash_sha2_256_legacy_min_input_len_cost = Some(1);
4059                    cfg.hash_sha3_256_base_cost = Some(52);
4060                    cfg.hash_sha3_256_per_byte_cost = Some(2);
4061                    cfg.hash_sha3_256_legacy_min_input_len_cost = Some(1);
4062                    cfg.type_name_get_base_cost = Some(52);
4063                    cfg.type_name_get_per_byte_cost = Some(2);
4064                    cfg.string_check_utf8_base_cost = Some(52);
4065                    cfg.string_check_utf8_per_byte_cost = Some(2);
4066                    cfg.string_is_char_boundary_base_cost = Some(52);
4067                    cfg.string_sub_string_base_cost = Some(52);
4068                    cfg.string_sub_string_per_byte_cost = Some(2);
4069                    cfg.string_index_of_base_cost = Some(52);
4070                    cfg.string_index_of_per_byte_pattern_cost = Some(2);
4071                    cfg.string_index_of_per_byte_searched_cost = Some(2);
4072                    cfg.vector_empty_base_cost = Some(52);
4073                    cfg.vector_length_base_cost = Some(52);
4074                    cfg.vector_push_back_base_cost = Some(52);
4075                    cfg.vector_push_back_legacy_per_abstract_memory_unit_cost = Some(2);
4076                    cfg.vector_borrow_base_cost = Some(52);
4077                    cfg.vector_pop_back_base_cost = Some(52);
4078                    cfg.vector_destroy_empty_base_cost = Some(52);
4079                    cfg.vector_swap_base_cost = Some(52);
4080                }
4081                54 => {
4082                    // Enable random beacon on mainnet.
4083                    cfg.feature_flags.random_beacon = true;
4084                    cfg.random_beacon_reduction_lower_bound = Some(1000);
4085                    cfg.random_beacon_dkg_timeout_round = Some(3000);
4086                    cfg.random_beacon_min_round_interval_ms = Some(500);
4087
4088                    // Turns on shared object congestion control on mainnet.
4089                    cfg.max_accumulated_txn_cost_per_object_in_narwhal_commit = Some(100);
4090                    cfg.max_accumulated_txn_cost_per_object_in_mysticeti_commit = Some(10);
4091                    cfg.feature_flags.per_object_congestion_control_mode =
4092                        PerObjectCongestionControlMode::TotalTxCount;
4093
4094                    // Enable soft bundle on mainnet.
4095                    cfg.feature_flags.soft_bundle = true;
4096                    cfg.max_soft_bundle_size = Some(5);
4097                }
4098                55 => {
4099                    // Turn on enums mainnet
4100                    cfg.move_binary_format_version = Some(7);
4101
4102                    // Assume 1KB per transaction and 500 transactions per block.
4103                    cfg.consensus_max_transactions_in_block_bytes = Some(512 * 1024);
4104                    // Assume 20_000 TPS * 5% max stake per validator / (minimum) 4 blocks per round = 250 transactions per block maximum
4105                    // Using a higher limit that is 512, to account for bursty traffic and system transactions.
4106                    cfg.consensus_max_num_transactions_in_block = Some(512);
4107
4108                    cfg.feature_flags.rethrow_serialization_type_layout_errors = true;
4109                }
4110                56 => {
4111                    if chain == Chain::Mainnet {
4112                        cfg.feature_flags.bridge = true;
4113                    }
4114                }
4115                57 => {
4116                    // Reduce minimum number of random beacon shares.
4117                    cfg.random_beacon_reduction_lower_bound = Some(800);
4118                }
4119                58 => {
4120                    if chain == Chain::Mainnet {
4121                        cfg.bridge_should_try_to_finalize_committee = Some(true);
4122                    }
4123
4124                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4125                        // Enable distributed vote scoring for devnet
4126                        cfg.feature_flags
4127                            .consensus_distributed_vote_scoring_strategy = true;
4128                    }
4129                }
4130                59 => {
4131                    // Enable round prober in consensus.
4132                    cfg.feature_flags.consensus_round_prober = true;
4133                }
4134                60 => {
4135                    cfg.max_type_to_layout_nodes = Some(512);
4136                    cfg.feature_flags.validate_identifier_inputs = true;
4137                }
4138                61 => {
4139                    if chain != Chain::Mainnet {
4140                        // Enable distributed vote scoring for testnet
4141                        cfg.feature_flags
4142                            .consensus_distributed_vote_scoring_strategy = true;
4143                    }
4144                    // Further reduce minimum number of random beacon shares.
4145                    cfg.random_beacon_reduction_lower_bound = Some(700);
4146
4147                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4148                        // Enable Mysticeti fastpath for devnet
4149                        cfg.feature_flags.mysticeti_fastpath = true;
4150                    }
4151                }
4152                62 => {
4153                    cfg.feature_flags.relocate_event_module = true;
4154                }
4155                63 => {
4156                    cfg.feature_flags.per_object_congestion_control_mode =
4157                        PerObjectCongestionControlMode::TotalGasBudgetWithCap;
4158                    cfg.gas_budget_based_txn_cost_cap_factor = Some(400_000);
4159                    cfg.max_accumulated_txn_cost_per_object_in_mysticeti_commit = Some(18_500_000);
4160                    cfg.max_accumulated_txn_cost_per_object_in_narwhal_commit = Some(240_000_000);
4161                }
4162                64 => {
4163                    cfg.feature_flags.per_object_congestion_control_mode =
4164                        PerObjectCongestionControlMode::TotalTxCount;
4165                    cfg.max_accumulated_txn_cost_per_object_in_narwhal_commit = Some(40);
4166                    cfg.max_accumulated_txn_cost_per_object_in_mysticeti_commit = Some(3);
4167                }
4168                65 => {
4169                    // Enable distributed vote scoring for mainnet
4170                    cfg.feature_flags
4171                        .consensus_distributed_vote_scoring_strategy = true;
4172                }
4173                66 => {
4174                    if chain == Chain::Mainnet {
4175                        // Revert the distributed vote scoring for mainnet (for one protocol upgrade)
4176                        cfg.feature_flags
4177                            .consensus_distributed_vote_scoring_strategy = false;
4178                    }
4179                }
4180                67 => {
4181                    // Enable it once again.
4182                    cfg.feature_flags
4183                        .consensus_distributed_vote_scoring_strategy = true;
4184                }
4185                68 => {
4186                    cfg.group_ops_bls12381_g1_to_uncompressed_g1_cost = Some(26);
4187                    cfg.group_ops_bls12381_uncompressed_g1_to_g1_cost = Some(52);
4188                    cfg.group_ops_bls12381_uncompressed_g1_sum_base_cost = Some(26);
4189                    cfg.group_ops_bls12381_uncompressed_g1_sum_cost_per_term = Some(13);
4190                    cfg.group_ops_bls12381_uncompressed_g1_sum_max_terms = Some(2000);
4191
4192                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4193                        cfg.feature_flags.uncompressed_g1_group_elements = true;
4194                    }
4195
4196                    cfg.feature_flags.per_object_congestion_control_mode =
4197                        PerObjectCongestionControlMode::TotalGasBudgetWithCap;
4198                    cfg.gas_budget_based_txn_cost_cap_factor = Some(400_000);
4199                    cfg.max_accumulated_txn_cost_per_object_in_mysticeti_commit = Some(18_500_000);
4200                    cfg.max_accumulated_randomness_txn_cost_per_object_in_mysticeti_commit =
4201                        Some(3_700_000); // 20% of above
4202                    cfg.max_txn_cost_overage_per_object_in_commit = Some(u64::MAX);
4203                    cfg.gas_budget_based_txn_cost_absolute_cap_commit_count = Some(50);
4204
4205                    // Further reduce minimum number of random beacon shares.
4206                    cfg.random_beacon_reduction_lower_bound = Some(500);
4207
4208                    cfg.feature_flags.disallow_new_modules_in_deps_only_packages = true;
4209                }
4210                69 => {
4211                    // Sets number of rounds allowed for fastpath voting in consensus.
4212                    cfg.consensus_voting_rounds = Some(40);
4213
4214                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4215                        // Enable smart ancestor selection for devnet
4216                        cfg.feature_flags.consensus_smart_ancestor_selection = true;
4217                    }
4218
4219                    if chain != Chain::Mainnet {
4220                        cfg.feature_flags.uncompressed_g1_group_elements = true;
4221                    }
4222                }
4223                70 => {
4224                    if chain != Chain::Mainnet {
4225                        // Enable smart ancestor selection for testnet
4226                        cfg.feature_flags.consensus_smart_ancestor_selection = true;
4227                        // Enable probing for accepted rounds in round prober for testnet
4228                        cfg.feature_flags
4229                            .consensus_round_prober_probe_accepted_rounds = true;
4230                    }
4231
4232                    cfg.poseidon_bn254_cost_per_block = Some(388);
4233
4234                    cfg.gas_model_version = Some(9);
4235                    cfg.feature_flags.native_charging_v2 = true;
4236                    cfg.bls12381_bls12381_min_sig_verify_cost_base = Some(44064);
4237                    cfg.bls12381_bls12381_min_pk_verify_cost_base = Some(49282);
4238                    cfg.ecdsa_k1_secp256k1_verify_keccak256_cost_base = Some(1470);
4239                    cfg.ecdsa_k1_secp256k1_verify_sha256_cost_base = Some(1470);
4240                    cfg.ecdsa_r1_secp256r1_verify_sha256_cost_base = Some(4225);
4241                    cfg.ecdsa_r1_secp256r1_verify_keccak256_cost_base = Some(4225);
4242                    cfg.ecvrf_ecvrf_verify_cost_base = Some(4848);
4243                    cfg.ed25519_ed25519_verify_cost_base = Some(1802);
4244
4245                    // Manually changed to be "under cost"
4246                    cfg.ecdsa_r1_ecrecover_keccak256_cost_base = Some(1173);
4247                    cfg.ecdsa_r1_ecrecover_sha256_cost_base = Some(1173);
4248                    cfg.ecdsa_k1_ecrecover_keccak256_cost_base = Some(500);
4249                    cfg.ecdsa_k1_ecrecover_sha256_cost_base = Some(500);
4250
4251                    cfg.groth16_prepare_verifying_key_bls12381_cost_base = Some(53838);
4252                    cfg.groth16_prepare_verifying_key_bn254_cost_base = Some(82010);
4253                    cfg.groth16_verify_groth16_proof_internal_bls12381_cost_base = Some(72090);
4254                    cfg.groth16_verify_groth16_proof_internal_bls12381_cost_per_public_input =
4255                        Some(8213);
4256                    cfg.groth16_verify_groth16_proof_internal_bn254_cost_base = Some(115502);
4257                    cfg.groth16_verify_groth16_proof_internal_bn254_cost_per_public_input =
4258                        Some(9484);
4259
4260                    cfg.hash_keccak256_cost_base = Some(10);
4261                    cfg.hash_blake2b256_cost_base = Some(10);
4262
4263                    // group ops
4264                    cfg.group_ops_bls12381_decode_scalar_cost = Some(7);
4265                    cfg.group_ops_bls12381_decode_g1_cost = Some(2848);
4266                    cfg.group_ops_bls12381_decode_g2_cost = Some(3770);
4267                    cfg.group_ops_bls12381_decode_gt_cost = Some(3068);
4268
4269                    cfg.group_ops_bls12381_scalar_add_cost = Some(10);
4270                    cfg.group_ops_bls12381_g1_add_cost = Some(1556);
4271                    cfg.group_ops_bls12381_g2_add_cost = Some(3048);
4272                    cfg.group_ops_bls12381_gt_add_cost = Some(188);
4273
4274                    cfg.group_ops_bls12381_scalar_sub_cost = Some(10);
4275                    cfg.group_ops_bls12381_g1_sub_cost = Some(1550);
4276                    cfg.group_ops_bls12381_g2_sub_cost = Some(3019);
4277                    cfg.group_ops_bls12381_gt_sub_cost = Some(497);
4278
4279                    cfg.group_ops_bls12381_scalar_mul_cost = Some(11);
4280                    cfg.group_ops_bls12381_g1_mul_cost = Some(4842);
4281                    cfg.group_ops_bls12381_g2_mul_cost = Some(9108);
4282                    cfg.group_ops_bls12381_gt_mul_cost = Some(27490);
4283
4284                    cfg.group_ops_bls12381_scalar_div_cost = Some(91);
4285                    cfg.group_ops_bls12381_g1_div_cost = Some(5091);
4286                    cfg.group_ops_bls12381_g2_div_cost = Some(9206);
4287                    cfg.group_ops_bls12381_gt_div_cost = Some(27804);
4288
4289                    cfg.group_ops_bls12381_g1_hash_to_base_cost = Some(2962);
4290                    cfg.group_ops_bls12381_g2_hash_to_base_cost = Some(8688);
4291
4292                    cfg.group_ops_bls12381_g1_msm_base_cost = Some(62648);
4293                    cfg.group_ops_bls12381_g2_msm_base_cost = Some(131192);
4294                    cfg.group_ops_bls12381_g1_msm_base_cost_per_input = Some(1333);
4295                    cfg.group_ops_bls12381_g2_msm_base_cost_per_input = Some(3216);
4296
4297                    cfg.group_ops_bls12381_uncompressed_g1_to_g1_cost = Some(677);
4298                    cfg.group_ops_bls12381_g1_to_uncompressed_g1_cost = Some(2099);
4299                    cfg.group_ops_bls12381_uncompressed_g1_sum_base_cost = Some(77);
4300                    cfg.group_ops_bls12381_uncompressed_g1_sum_cost_per_term = Some(26);
4301
4302                    cfg.group_ops_bls12381_pairing_cost = Some(26897);
4303                    cfg.group_ops_bls12381_uncompressed_g1_sum_max_terms = Some(1200);
4304
4305                    cfg.validator_validate_metadata_cost_base = Some(20000);
4306                }
4307                71 => {
4308                    cfg.sip_45_consensus_amplification_threshold = Some(5);
4309
4310                    // Enable bursts for congestion control. (10x the per-commit budget)
4311                    cfg.allowed_txn_cost_overage_burst_per_object_in_commit = Some(185_000_000);
4312                }
4313                72 => {
4314                    cfg.feature_flags.convert_type_argument_error = true;
4315
4316                    // Invariant: max_gas_price * base_tx_cost_fixed <= max_tx_gas
4317                    // max gas budget is in MIST and an absolute value 50_000 SUI
4318                    cfg.max_tx_gas = Some(50_000_000_000_000);
4319                    // max gas price is in MIST and an absolute value 50 SUI
4320                    cfg.max_gas_price = Some(50_000_000_000);
4321
4322                    cfg.feature_flags.variant_nodes = true;
4323                }
4324                73 => {
4325                    // Enable new marker table version.
4326                    cfg.use_object_per_epoch_marker_table_v2 = Some(true);
4327
4328                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4329                        // Assuming a round rate of max 15/sec, then using a gc depth of 60 allow blocks within a window of ~4 seconds
4330                        // to be included before be considered garbage collected.
4331                        cfg.consensus_gc_depth = Some(60);
4332                    }
4333
4334                    if chain != Chain::Mainnet {
4335                        // Enable zstd compression for consensus in testnet
4336                        cfg.feature_flags.consensus_zstd_compression = true;
4337                    }
4338
4339                    // Enable smart ancestor selection for mainnet
4340                    cfg.feature_flags.consensus_smart_ancestor_selection = true;
4341                    // Enable probing for accepted rounds in round prober for mainnet
4342                    cfg.feature_flags
4343                        .consensus_round_prober_probe_accepted_rounds = true;
4344
4345                    // Increase congestion control budget.
4346                    cfg.feature_flags.per_object_congestion_control_mode =
4347                        PerObjectCongestionControlMode::TotalGasBudgetWithCap;
4348                    cfg.gas_budget_based_txn_cost_cap_factor = Some(400_000);
4349                    cfg.max_accumulated_txn_cost_per_object_in_mysticeti_commit = Some(37_000_000);
4350                    cfg.max_accumulated_randomness_txn_cost_per_object_in_mysticeti_commit =
4351                        Some(7_400_000); // 20% of above
4352                    cfg.max_txn_cost_overage_per_object_in_commit = Some(u64::MAX);
4353                    cfg.gas_budget_based_txn_cost_absolute_cap_commit_count = Some(50);
4354                    cfg.allowed_txn_cost_overage_burst_per_object_in_commit = Some(370_000_000);
4355                }
4356                74 => {
4357                    // Enable nitro attestation verify native move function for devnet
4358                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4359                        cfg.feature_flags.enable_nitro_attestation = true;
4360                    }
4361                    cfg.nitro_attestation_parse_base_cost = Some(53 * 50);
4362                    cfg.nitro_attestation_parse_cost_per_byte = Some(50);
4363                    cfg.nitro_attestation_verify_base_cost = Some(49632 * 50);
4364                    cfg.nitro_attestation_verify_cost_per_cert = Some(52369 * 50);
4365
4366                    // Enable zstd compression for consensus in mainnet
4367                    cfg.feature_flags.consensus_zstd_compression = true;
4368
4369                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4370                        cfg.feature_flags.consensus_linearize_subdag_v2 = true;
4371                    }
4372                }
4373                75 => {
4374                    if chain != Chain::Mainnet {
4375                        cfg.feature_flags.passkey_auth = true;
4376                    }
4377                }
4378                76 => {
4379                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4380                        cfg.feature_flags.record_additional_state_digest_in_prologue = true;
4381                        cfg.consensus_commit_rate_estimation_window_size = Some(10);
4382                    }
4383                    cfg.feature_flags.minimize_child_object_mutations = true;
4384
4385                    if chain != Chain::Mainnet {
4386                        cfg.feature_flags.accept_passkey_in_multisig = true;
4387                    }
4388                }
4389                77 => {
4390                    cfg.feature_flags.uncompressed_g1_group_elements = true;
4391
4392                    if chain != Chain::Mainnet {
4393                        cfg.consensus_gc_depth = Some(60);
4394                        cfg.feature_flags.consensus_linearize_subdag_v2 = true;
4395                    }
4396                }
4397                78 => {
4398                    cfg.feature_flags.move_native_context = true;
4399                    cfg.tx_context_fresh_id_cost_base = Some(52);
4400                    cfg.tx_context_sender_cost_base = Some(30);
4401                    cfg.tx_context_epoch_cost_base = Some(30);
4402                    cfg.tx_context_epoch_timestamp_ms_cost_base = Some(30);
4403                    cfg.tx_context_sponsor_cost_base = Some(30);
4404                    cfg.tx_context_gas_price_cost_base = Some(30);
4405                    cfg.tx_context_gas_budget_cost_base = Some(30);
4406                    cfg.tx_context_ids_created_cost_base = Some(30);
4407                    cfg.tx_context_replace_cost_base = Some(30);
4408                    cfg.gas_model_version = Some(10);
4409
4410                    if chain != Chain::Mainnet {
4411                        cfg.feature_flags.record_additional_state_digest_in_prologue = true;
4412                        cfg.consensus_commit_rate_estimation_window_size = Some(10);
4413
4414                        // Enable execution time estimate mode for congestion control on testnet.
4415                        cfg.feature_flags.per_object_congestion_control_mode =
4416                            PerObjectCongestionControlMode::ExecutionTimeEstimate(
4417                                ExecutionTimeEstimateParams {
4418                                    target_utilization: 30,
4419                                    allowed_txn_cost_overage_burst_limit_us: 100_000, // 100 ms
4420                                    randomness_scalar: 20,
4421                                    max_estimate_us: 1_500_000, // 1.5s
4422                                    stored_observations_num_included_checkpoints: 10,
4423                                    stored_observations_limit: u64::MAX,
4424                                    stake_weighted_median_threshold: 0,
4425                                    default_none_duration_for_new_keys: false,
4426                                    observations_chunk_size: None,
4427                                },
4428                            );
4429                    }
4430                }
4431                79 => {
4432                    if chain != Chain::Mainnet {
4433                        cfg.feature_flags.consensus_median_based_commit_timestamp = true;
4434
4435                        // Increase threshold for bad nodes that won't be considered
4436                        // leaders in consensus in testnet
4437                        cfg.consensus_bad_nodes_stake_threshold = Some(30);
4438
4439                        cfg.feature_flags.consensus_batched_block_sync = true;
4440
4441                        // Enable verify nitro attestation in testnet.
4442                        cfg.feature_flags.enable_nitro_attestation = true
4443                    }
4444                    cfg.feature_flags.normalize_ptb_arguments = true;
4445
4446                    cfg.consensus_gc_depth = Some(60);
4447                    cfg.feature_flags.consensus_linearize_subdag_v2 = true;
4448                }
4449                80 => {
4450                    cfg.max_ptb_value_size = Some(1024 * 1024);
4451                }
4452                81 => {
4453                    cfg.feature_flags.consensus_median_based_commit_timestamp = true;
4454                    cfg.feature_flags.enforce_checkpoint_timestamp_monotonicity = true;
4455                    cfg.consensus_bad_nodes_stake_threshold = Some(30)
4456                }
4457                82 => {
4458                    cfg.feature_flags.max_ptb_value_size_v2 = true;
4459                }
4460                83 => {
4461                    if chain == Chain::Mainnet {
4462                        // The address that will sign the recovery transaction.
4463                        let aliased: [u8; 32] = Hex::decode(
4464                            "0x0b2da327ba6a4cacbe75dddd50e6e8bbf81d6496e92d66af9154c61c77f7332f",
4465                        )
4466                        .unwrap()
4467                        .try_into()
4468                        .unwrap();
4469
4470                        // Allow aliasing for the two addresses that contain stolen funds.
4471                        cfg.aliased_addresses.push(AliasedAddress {
4472                            original: Hex::decode("0xcd8962dad278d8b50fa0f9eb0186bfa4cbdecc6d59377214c88d0286a0ac9562").unwrap().try_into().unwrap(),
4473                            aliased,
4474                            allowed_tx_digests: vec![
4475                                Base58::decode("B2eGLFoMHgj93Ni8dAJBfqGzo8EWSTLBesZzhEpTPA4").unwrap().try_into().unwrap(),
4476                            ],
4477                        });
4478
4479                        cfg.aliased_addresses.push(AliasedAddress {
4480                            original: Hex::decode("0xe28b50cef1d633ea43d3296a3f6b67ff0312a5f1a99f0af753c85b8b5de8ff06").unwrap().try_into().unwrap(),
4481                            aliased,
4482                            allowed_tx_digests: vec![
4483                                Base58::decode("J4QqSAgp7VrQtQpMy5wDX4QGsCSEZu3U5KuDAkbESAge").unwrap().try_into().unwrap(),
4484                            ],
4485                        });
4486                    }
4487
4488                    // These features had to be deferred to v84 for mainnet in order to ship the recovery protocol
4489                    // upgrade as a patch to 1.48
4490                    if chain != Chain::Mainnet {
4491                        cfg.feature_flags.resolve_type_input_ids_to_defining_id = true;
4492                        cfg.transfer_party_transfer_internal_cost_base = Some(52);
4493
4494                        // Enable execution time estimate mode for congestion control on mainnet.
4495                        cfg.feature_flags.record_additional_state_digest_in_prologue = true;
4496                        cfg.consensus_commit_rate_estimation_window_size = Some(10);
4497                        cfg.feature_flags.per_object_congestion_control_mode =
4498                            PerObjectCongestionControlMode::ExecutionTimeEstimate(
4499                                ExecutionTimeEstimateParams {
4500                                    target_utilization: 30,
4501                                    allowed_txn_cost_overage_burst_limit_us: 100_000, // 100 ms
4502                                    randomness_scalar: 20,
4503                                    max_estimate_us: 1_500_000, // 1.5s
4504                                    stored_observations_num_included_checkpoints: 10,
4505                                    stored_observations_limit: u64::MAX,
4506                                    stake_weighted_median_threshold: 0,
4507                                    default_none_duration_for_new_keys: false,
4508                                    observations_chunk_size: None,
4509                                },
4510                            );
4511
4512                        // Enable the new depth-first block sync logic.
4513                        cfg.feature_flags.consensus_batched_block_sync = true;
4514
4515                        // Enable nitro attestation upgraded parsing logic and enable the
4516                        // native function on mainnet.
4517                        cfg.feature_flags.enable_nitro_attestation_upgraded_parsing = true;
4518                        cfg.feature_flags.enable_nitro_attestation = true;
4519                    }
4520                }
4521                84 => {
4522                    if chain == Chain::Mainnet {
4523                        cfg.feature_flags.resolve_type_input_ids_to_defining_id = true;
4524                        cfg.transfer_party_transfer_internal_cost_base = Some(52);
4525
4526                        // Enable execution time estimate mode for congestion control on mainnet.
4527                        cfg.feature_flags.record_additional_state_digest_in_prologue = true;
4528                        cfg.consensus_commit_rate_estimation_window_size = Some(10);
4529                        cfg.feature_flags.per_object_congestion_control_mode =
4530                            PerObjectCongestionControlMode::ExecutionTimeEstimate(
4531                                ExecutionTimeEstimateParams {
4532                                    target_utilization: 30,
4533                                    allowed_txn_cost_overage_burst_limit_us: 100_000, // 100 ms
4534                                    randomness_scalar: 20,
4535                                    max_estimate_us: 1_500_000, // 1.5s
4536                                    stored_observations_num_included_checkpoints: 10,
4537                                    stored_observations_limit: u64::MAX,
4538                                    stake_weighted_median_threshold: 0,
4539                                    default_none_duration_for_new_keys: false,
4540                                    observations_chunk_size: None,
4541                                },
4542                            );
4543
4544                        // Enable the new depth-first block sync logic.
4545                        cfg.feature_flags.consensus_batched_block_sync = true;
4546
4547                        // Enable nitro attestation upgraded parsing logic and enable the
4548                        // native function on mainnet.
4549                        cfg.feature_flags.enable_nitro_attestation_upgraded_parsing = true;
4550                        cfg.feature_flags.enable_nitro_attestation = true;
4551                    }
4552
4553                    // Limit the number of stored execution time observations at end of epoch.
4554                    cfg.feature_flags.per_object_congestion_control_mode =
4555                        PerObjectCongestionControlMode::ExecutionTimeEstimate(
4556                            ExecutionTimeEstimateParams {
4557                                target_utilization: 30,
4558                                allowed_txn_cost_overage_burst_limit_us: 100_000, // 100 ms
4559                                randomness_scalar: 20,
4560                                max_estimate_us: 1_500_000, // 1.5s
4561                                stored_observations_num_included_checkpoints: 10,
4562                                stored_observations_limit: 20,
4563                                stake_weighted_median_threshold: 0,
4564                                default_none_duration_for_new_keys: false,
4565                                observations_chunk_size: None,
4566                            },
4567                        );
4568                    cfg.feature_flags.allow_unbounded_system_objects = true;
4569                }
4570                85 => {
4571                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4572                        cfg.feature_flags.enable_party_transfer = true;
4573                    }
4574
4575                    cfg.feature_flags
4576                        .record_consensus_determined_version_assignments_in_prologue_v2 = true;
4577                    cfg.feature_flags.disallow_self_identifier = true;
4578                    cfg.feature_flags.per_object_congestion_control_mode =
4579                        PerObjectCongestionControlMode::ExecutionTimeEstimate(
4580                            ExecutionTimeEstimateParams {
4581                                target_utilization: 50,
4582                                allowed_txn_cost_overage_burst_limit_us: 500_000, // 500 ms
4583                                randomness_scalar: 20,
4584                                max_estimate_us: 1_500_000, // 1.5s
4585                                stored_observations_num_included_checkpoints: 10,
4586                                stored_observations_limit: 20,
4587                                stake_weighted_median_threshold: 0,
4588                                default_none_duration_for_new_keys: false,
4589                                observations_chunk_size: None,
4590                            },
4591                        );
4592                }
4593                86 => {
4594                    cfg.feature_flags.type_tags_in_object_runtime = true;
4595                    cfg.max_move_enum_variants = Some(move_core_types::VARIANT_COUNT_MAX);
4596
4597                    // Set a stake_weighted_median_threshold for congestion control.
4598                    cfg.feature_flags.per_object_congestion_control_mode =
4599                        PerObjectCongestionControlMode::ExecutionTimeEstimate(
4600                            ExecutionTimeEstimateParams {
4601                                target_utilization: 50,
4602                                allowed_txn_cost_overage_burst_limit_us: 500_000, // 500 ms
4603                                randomness_scalar: 20,
4604                                max_estimate_us: 1_500_000, // 1.5s
4605                                stored_observations_num_included_checkpoints: 10,
4606                                stored_observations_limit: 20,
4607                                stake_weighted_median_threshold: 3334,
4608                                default_none_duration_for_new_keys: false,
4609                                observations_chunk_size: None,
4610                            },
4611                        );
4612                    // Enable party transfer for testnet.
4613                    if chain != Chain::Mainnet {
4614                        cfg.feature_flags.enable_party_transfer = true;
4615                    }
4616                }
4617                87 => {
4618                    if chain == Chain::Mainnet {
4619                        cfg.feature_flags.record_time_estimate_processed = true;
4620                    }
4621                    cfg.feature_flags.better_adapter_type_resolution_errors = true;
4622                }
4623                88 => {
4624                    cfg.feature_flags.record_time_estimate_processed = true;
4625                    cfg.tx_context_rgp_cost_base = Some(30);
4626                    cfg.feature_flags
4627                        .ignore_execution_time_observations_after_certs_closed = true;
4628
4629                    // Disable backwards compatible behavior in execution time estimator for
4630                    // new protocol version.
4631                    cfg.feature_flags.per_object_congestion_control_mode =
4632                        PerObjectCongestionControlMode::ExecutionTimeEstimate(
4633                            ExecutionTimeEstimateParams {
4634                                target_utilization: 50,
4635                                allowed_txn_cost_overage_burst_limit_us: 500_000, // 500 ms
4636                                randomness_scalar: 20,
4637                                max_estimate_us: 1_500_000, // 1.5s
4638                                stored_observations_num_included_checkpoints: 10,
4639                                stored_observations_limit: 20,
4640                                stake_weighted_median_threshold: 3334,
4641                                default_none_duration_for_new_keys: true,
4642                                observations_chunk_size: None,
4643                            },
4644                        );
4645                }
4646                89 => {
4647                    cfg.feature_flags.dependency_linkage_error = true;
4648                    cfg.feature_flags.additional_multisig_checks = true;
4649                }
4650                90 => {
4651                    // 100x RGP
4652                    cfg.max_gas_price_rgp_factor_for_aborted_transactions = Some(100);
4653                    cfg.feature_flags.debug_fatal_on_move_invariant_violation = true;
4654                    cfg.feature_flags.additional_consensus_digest_indirect_state = true;
4655                    cfg.feature_flags.accept_passkey_in_multisig = true;
4656                    cfg.feature_flags.passkey_auth = true;
4657                    cfg.feature_flags.check_for_init_during_upgrade = true;
4658
4659                    // Enable Mysticeti fastpath handlers on testnet.
4660                    if chain != Chain::Mainnet {
4661                        cfg.feature_flags.mysticeti_fastpath = true;
4662                    }
4663                }
4664                91 => {
4665                    cfg.feature_flags.per_command_shared_object_transfer_rules = true;
4666                }
4667                92 => {
4668                    cfg.feature_flags.per_command_shared_object_transfer_rules = false;
4669                }
4670                93 => {
4671                    cfg.feature_flags
4672                        .consensus_checkpoint_signature_key_includes_digest = true;
4673                }
4674                94 => {
4675                    // Decrease stored observations limit 20->18 to stay within system object size limit.
4676                    cfg.feature_flags.per_object_congestion_control_mode =
4677                        PerObjectCongestionControlMode::ExecutionTimeEstimate(
4678                            ExecutionTimeEstimateParams {
4679                                target_utilization: 50,
4680                                allowed_txn_cost_overage_burst_limit_us: 500_000, // 500 ms
4681                                randomness_scalar: 20,
4682                                max_estimate_us: 1_500_000, // 1.5s
4683                                stored_observations_num_included_checkpoints: 10,
4684                                stored_observations_limit: 18,
4685                                stake_weighted_median_threshold: 3334,
4686                                default_none_duration_for_new_keys: true,
4687                                observations_chunk_size: None,
4688                            },
4689                        );
4690
4691                    // Enable party transfer on mainnet.
4692                    cfg.feature_flags.enable_party_transfer = true;
4693                }
4694                95 => {
4695                    cfg.type_name_id_base_cost = Some(52);
4696
4697                    // Reduce the frequency of checkpoint splitting under high TPS.
4698                    cfg.max_transactions_per_checkpoint = Some(20_000);
4699                }
4700                96 => {
4701                    // Enable artifacts digest in devnet.
4702                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4703                        cfg.feature_flags
4704                            .include_checkpoint_artifacts_digest_in_summary = true;
4705                    }
4706                    cfg.feature_flags.correct_gas_payment_limit_check = true;
4707                    cfg.feature_flags.authority_capabilities_v2 = true;
4708                    cfg.feature_flags.use_mfp_txns_in_load_initial_object_debts = true;
4709                    cfg.feature_flags.cancel_for_failed_dkg_early = true;
4710                    cfg.feature_flags.enable_coin_registry = true;
4711
4712                    // Enable Mysticeti fastpath handlers on mainnet.
4713                    cfg.feature_flags.mysticeti_fastpath = true;
4714                }
4715                97 => {
4716                    cfg.feature_flags.additional_borrow_checks = true;
4717                }
4718                98 => {
4719                    cfg.event_emit_auth_stream_cost = Some(52);
4720                    cfg.feature_flags.better_loader_errors = true;
4721                    cfg.feature_flags.generate_df_type_layouts = true;
4722                }
4723                99 => {
4724                    cfg.feature_flags.use_new_commit_handler = true;
4725                }
4726                100 => {
4727                    cfg.feature_flags.private_generics_verifier_v2 = true;
4728                }
4729                101 => {
4730                    cfg.feature_flags.create_root_accumulator_object = true;
4731                    cfg.max_updates_per_settlement_txn = Some(100);
4732                    if chain != Chain::Mainnet {
4733                        cfg.feature_flags.enable_poseidon = true;
4734                    }
4735                }
4736                102 => {
4737                    // Enable execution time observation chunking and increase limit to 180.
4738                    // max_move_object_size is 250 KB, we've experientially determined that fits ~ 18 estimates
4739                    // so if we have 10 chunks, that's 2.5MB, < 8MB max_serialized_tx_effects_size_bytes_system_tx
4740                    cfg.feature_flags.per_object_congestion_control_mode =
4741                        PerObjectCongestionControlMode::ExecutionTimeEstimate(
4742                            ExecutionTimeEstimateParams {
4743                                target_utilization: 50,
4744                                allowed_txn_cost_overage_burst_limit_us: 500_000, // 500 ms
4745                                randomness_scalar: 20,
4746                                max_estimate_us: 1_500_000, // 1.5s
4747                                stored_observations_num_included_checkpoints: 10,
4748                                stored_observations_limit: 180,
4749                                stake_weighted_median_threshold: 3334,
4750                                default_none_duration_for_new_keys: true,
4751                                observations_chunk_size: Some(18),
4752                            },
4753                        );
4754                    cfg.feature_flags.deprecate_global_storage_ops = true;
4755                }
4756                103 => {}
4757                104 => {
4758                    cfg.translation_per_command_base_charge = Some(1);
4759                    cfg.translation_per_input_base_charge = Some(1);
4760                    cfg.translation_pure_input_per_byte_charge = Some(1);
4761                    cfg.translation_per_type_node_charge = Some(1);
4762                    cfg.translation_per_reference_node_charge = Some(1);
4763                    cfg.translation_per_linkage_entry_charge = Some(10);
4764                    cfg.gas_model_version = Some(11);
4765                    cfg.feature_flags.abstract_size_in_object_runtime = true;
4766                    cfg.feature_flags.object_runtime_charge_cache_load_gas = true;
4767                    cfg.dynamic_field_hash_type_and_key_cost_base = Some(52);
4768                    cfg.dynamic_field_add_child_object_cost_base = Some(52);
4769                    cfg.dynamic_field_add_child_object_value_cost_per_byte = Some(1);
4770                    cfg.dynamic_field_borrow_child_object_cost_base = Some(52);
4771                    cfg.dynamic_field_borrow_child_object_child_ref_cost_per_byte = Some(1);
4772                    cfg.dynamic_field_remove_child_object_cost_base = Some(52);
4773                    cfg.dynamic_field_remove_child_object_child_cost_per_byte = Some(1);
4774                    cfg.dynamic_field_has_child_object_cost_base = Some(52);
4775                    cfg.dynamic_field_has_child_object_with_ty_cost_base = Some(52);
4776                    cfg.feature_flags.enable_ptb_execution_v2 = true;
4777
4778                    cfg.poseidon_bn254_cost_base = Some(260);
4779
4780                    cfg.feature_flags.consensus_skip_gced_accept_votes = true;
4781
4782                    if chain != Chain::Mainnet {
4783                        cfg.feature_flags
4784                            .enable_nitro_attestation_all_nonzero_pcrs_parsing = true;
4785                    }
4786
4787                    cfg.feature_flags
4788                        .include_cancelled_randomness_txns_in_prologue = true;
4789                }
4790                105 => {
4791                    cfg.feature_flags.enable_multi_epoch_transaction_expiration = true;
4792                    cfg.feature_flags.disable_preconsensus_locking = true;
4793
4794                    if chain != Chain::Mainnet {
4795                        cfg.feature_flags
4796                            .enable_nitro_attestation_always_include_required_pcrs_parsing = true;
4797                    }
4798                }
4799                106 => {
4800                    // est. 100 bytes per object * 76 (storage_gas_price)
4801                    cfg.accumulator_object_storage_cost = Some(7600);
4802
4803                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4804                        cfg.feature_flags.enable_accumulators = true;
4805                        cfg.feature_flags.enable_address_balance_gas_payments = true;
4806                        cfg.feature_flags.enable_authenticated_event_streams = true;
4807                        cfg.feature_flags.enable_object_funds_withdraw = true;
4808                    }
4809                }
4810                107 => {
4811                    cfg.feature_flags
4812                        .consensus_skip_gced_blocks_in_direct_finalization = true;
4813
4814                    // Trigger edge cases more often in integration tests.
4815                    if in_integration_test() {
4816                        cfg.consensus_gc_depth = Some(6);
4817                        cfg.consensus_max_num_transactions_in_block = Some(8);
4818                    }
4819                }
4820                108 => {
4821                    cfg.feature_flags.gas_rounding_halve_digits = true;
4822                    cfg.feature_flags.flexible_tx_context_positions = true;
4823                    cfg.feature_flags.disable_entry_point_signature_check = true;
4824
4825                    if chain != Chain::Mainnet {
4826                        cfg.feature_flags.address_aliases = true;
4827
4828                        cfg.feature_flags.enable_accumulators = true;
4829                        cfg.feature_flags.enable_address_balance_gas_payments = true;
4830                    }
4831
4832                    cfg.feature_flags.enable_poseidon = true;
4833                }
4834                109 => {
4835                    cfg.binary_variant_handles = Some(1024);
4836                    cfg.binary_variant_instantiation_handles = Some(1024);
4837                    cfg.feature_flags.restrict_hot_or_not_entry_functions = true;
4838                }
4839                110 => {
4840                    cfg.feature_flags
4841                        .enable_nitro_attestation_all_nonzero_pcrs_parsing = true;
4842                    cfg.feature_flags
4843                        .enable_nitro_attestation_always_include_required_pcrs_parsing = true;
4844                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4845                        cfg.feature_flags.split_checkpoints_in_consensus_handler = true;
4846                    }
4847                    cfg.feature_flags.validate_zklogin_public_identifier = true;
4848                    cfg.feature_flags.fix_checkpoint_signature_mapping = true;
4849                    cfg.feature_flags
4850                        .consensus_always_accept_system_transactions = true;
4851                    if chain != Chain::Mainnet {
4852                        cfg.feature_flags.enable_object_funds_withdraw = true;
4853                    }
4854                }
4855                111 => {
4856                    cfg.feature_flags.validator_metadata_verify_v2 = true;
4857                }
4858                112 => {
4859                    cfg.group_ops_ristretto_decode_scalar_cost = Some(7);
4860                    cfg.group_ops_ristretto_decode_point_cost = Some(200);
4861                    cfg.group_ops_ristretto_scalar_add_cost = Some(10);
4862                    cfg.group_ops_ristretto_point_add_cost = Some(500);
4863                    cfg.group_ops_ristretto_scalar_sub_cost = Some(10);
4864                    cfg.group_ops_ristretto_point_sub_cost = Some(500);
4865                    cfg.group_ops_ristretto_scalar_mul_cost = Some(11);
4866                    cfg.group_ops_ristretto_point_mul_cost = Some(1200);
4867                    cfg.group_ops_ristretto_scalar_div_cost = Some(151);
4868                    cfg.group_ops_ristretto_point_div_cost = Some(2500);
4869
4870                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4871                        cfg.feature_flags.enable_ristretto255_group_ops = true;
4872                    }
4873                }
4874                113 => {
4875                    cfg.feature_flags.address_balance_gas_check_rgp_at_signing = true;
4876                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4877                        cfg.feature_flags.defer_unpaid_amplification = true;
4878                    }
4879                }
4880                114 => {
4881                    cfg.feature_flags.randomize_checkpoint_tx_limit_in_tests = true;
4882                    cfg.feature_flags.address_balance_gas_reject_gas_coin_arg = true;
4883                    if chain != Chain::Mainnet {
4884                        cfg.feature_flags.split_checkpoints_in_consensus_handler = true;
4885                        cfg.feature_flags.enable_authenticated_event_streams = true;
4886                        cfg.feature_flags
4887                            .include_checkpoint_artifacts_digest_in_summary = true;
4888                    }
4889                }
4890                115 => {
4891                    cfg.feature_flags.normalize_depth_formula = true;
4892                }
4893                116 => {
4894                    cfg.feature_flags.gasless_transaction_drop_safety = true;
4895                    cfg.feature_flags.address_aliases = true;
4896                    cfg.feature_flags.relax_valid_during_for_owned_inputs = true;
4897                    // Disabled while debugging
4898                    cfg.feature_flags.defer_unpaid_amplification = false;
4899                    cfg.feature_flags.enable_display_registry = true;
4900                }
4901                117 => {}
4902                118 => {
4903                    cfg.feature_flags.use_coin_party_owner = true;
4904                }
4905                119 => {
4906                    // Enable new VM.
4907                    cfg.execution_version = Some(4);
4908                    cfg.feature_flags.address_balance_gas_reject_gas_coin_arg = false;
4909                    cfg.feature_flags.merge_randomness_into_checkpoint = true;
4910                    if chain != Chain::Mainnet {
4911                        cfg.feature_flags.enable_gasless = true;
4912                        cfg.gasless_max_computation_units = Some(50_000);
4913                        cfg.gasless_allowed_token_types = Some(vec![]);
4914                        cfg.feature_flags.enable_coin_reservation_obj_refs = true;
4915                        cfg.feature_flags
4916                            .convert_withdrawal_compatibility_ptb_arguments = true;
4917                    }
4918                    cfg.gasless_max_unused_inputs = Some(1);
4919                    cfg.gasless_max_pure_input_bytes = Some(32);
4920                    if chain == Chain::Testnet {
4921                        cfg.gasless_allowed_token_types = Some(vec![(TESTNET_USDC.to_string(), 0)]);
4922                    }
4923                    cfg.transfer_receive_object_cost_per_byte = Some(1);
4924                    cfg.transfer_receive_object_type_cost_per_byte = Some(2);
4925                }
4926                120 => {
4927                    cfg.feature_flags.disallow_jump_orphans = true;
4928                }
4929                121 => {
4930                    // Re-enable unpaid amplification deferral protection (testnet + devnet)
4931                    if chain != Chain::Mainnet {
4932                        cfg.feature_flags.defer_unpaid_amplification = true;
4933                        cfg.gasless_max_tps = Some(50);
4934                    }
4935                    cfg.feature_flags
4936                        .early_return_receive_object_mismatched_type = true;
4937                }
4938                122 => {
4939                    // Enable unpaid amplification deferral on mainnet
4940                    cfg.feature_flags.defer_unpaid_amplification = true;
4941                    // Enable bulletproofs range proofs on devnet
4942                    cfg.verify_bulletproofs_ristretto255_base_cost = Some(30000);
4943                    cfg.verify_bulletproofs_ristretto255_cost_per_bit_and_commitment = Some(6500);
4944                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4945                        cfg.feature_flags.enable_verify_bulletproofs_ristretto255 = true;
4946                    }
4947                    cfg.feature_flags.gasless_verify_remaining_balance = true;
4948                    cfg.include_special_package_amendments = match chain {
4949                        Chain::Mainnet => Some(MAINNET_LINKAGE_AMENDMENTS.clone()),
4950                        Chain::Testnet => Some(TESTNET_LINKAGE_AMENDMENTS.clone()),
4951                        Chain::Unknown => None,
4952                    };
4953                    cfg.gasless_max_tx_size_bytes = Some(16 * 1024);
4954                    cfg.gasless_max_tps = Some(300);
4955                    cfg.gasless_max_computation_units = Some(5_000);
4956                }
4957                123 => {
4958                    cfg.gas_model_version = Some(13);
4959                }
4960                124 => {
4961                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4962                        cfg.feature_flags.timestamp_based_epoch_close = true;
4963                    }
4964                    cfg.gas_model_version = Some(14);
4965                    cfg.feature_flags.limit_groth16_pvk_inputs = true;
4966
4967                    // Bring mainnet in line with testnet: enable address balances, the
4968                    // gasless "free tier", coin reservations, and the rest of the
4969                    // accumulator/withdraw stack. These are all already enabled on
4970                    // testnet and devnet, so setting them unconditionally is a no-op
4971                    // there.
4972                    cfg.feature_flags.enable_accumulators = true;
4973                    cfg.feature_flags.enable_address_balance_gas_payments = true;
4974                    cfg.feature_flags.enable_authenticated_event_streams = true;
4975                    cfg.feature_flags.enable_coin_reservation_obj_refs = true;
4976                    cfg.feature_flags.enable_object_funds_withdraw = true;
4977                    cfg.feature_flags
4978                        .convert_withdrawal_compatibility_ptb_arguments = true;
4979                    cfg.feature_flags.split_checkpoints_in_consensus_handler = true;
4980                    cfg.feature_flags
4981                        .include_checkpoint_artifacts_digest_in_summary = true;
4982                    cfg.feature_flags.enable_gasless = true;
4983
4984                    // Set the mainnet allow-list. Testnet already has its USDC entry
4985                    // from v119, so only set this on mainnet to avoid clobbering the
4986                    // testnet value. $0.01 minimum transfer per stable; all listed
4987                    // tokens have 6 decimals.
4988                    if chain == Chain::Mainnet {
4989                        cfg.gasless_allowed_token_types = Some(vec![
4990                            (MAINNET_USDC.to_string(), 10_000),
4991                            (MAINNET_USDSUI.to_string(), 10_000),
4992                            (MAINNET_SUI_USDE.to_string(), 10_000),
4993                            (MAINNET_USDY.to_string(), 10_000),
4994                            (MAINNET_FDUSD.to_string(), 10_000),
4995                            (MAINNET_AUSD.to_string(), 10_000),
4996                            (MAINNET_USDB.to_string(), 10_000),
4997                        ]);
4998                    }
4999                }
5000                125 => {
5001                    cfg.feature_flags.granular_post_execution_checks = true;
5002                    if chain != Chain::Mainnet {
5003                        cfg.feature_flags.timestamp_based_epoch_close = true;
5004                    }
5005                }
5006                126 => {
5007                    cfg.feature_flags.early_exit_on_iffw = true;
5008                }
5009                127 => {
5010                    cfg.feature_flags.always_advance_dkg_to_resolution = true;
5011
5012                    cfg.verify_bulletproofs_ristretto255_base_cost = Some(23866);
5013                    cfg.verify_bulletproofs_ristretto255_cost_per_bit_and_commitment = Some(1324);
5014                    cfg.group_ops_ristretto_decode_scalar_cost = Some(5);
5015                    cfg.group_ops_ristretto_decode_point_cost = Some(216);
5016                    cfg.group_ops_ristretto_scalar_add_cost = Some(2);
5017                    cfg.group_ops_ristretto_point_add_cost = Some(8);
5018                    cfg.group_ops_ristretto_scalar_sub_cost = Some(2);
5019                    cfg.group_ops_ristretto_point_sub_cost = Some(8);
5020                    cfg.group_ops_ristretto_scalar_mul_cost = Some(5);
5021                    cfg.group_ops_ristretto_point_mul_cost = Some(1763);
5022                    cfg.group_ops_ristretto_scalar_div_cost = Some(557);
5023                    cfg.group_ops_ristretto_point_div_cost = Some(2244);
5024
5025                    if chain != Chain::Mainnet {
5026                        cfg.feature_flags.enable_ristretto255_group_ops = true;
5027                        cfg.feature_flags.enable_verify_bulletproofs_ristretto255 = true;
5028                    }
5029
5030                    cfg.feature_flags.timestamp_based_epoch_close = true;
5031                }
5032                128 => {}
5033                // Use this template when making changes:
5034                //
5035                //     // modify an existing constant.
5036                //     move_binary_format_version: Some(7),
5037                //
5038                //     // Add a new constant (which is set to None in prior versions).
5039                //     new_constant: Some(new_value),
5040                //
5041                //     // Remove a constant (ensure that it is never accessed during this version).
5042                //     max_move_object_size: None,
5043                _ => panic!("unsupported version {:?}", version),
5044            }
5045        }
5046
5047        cfg
5048    }
5049
5050    pub fn apply_seeded_test_overrides(&mut self, seed: &[u8; 32]) {
5051        if !self.feature_flags.randomize_checkpoint_tx_limit_in_tests
5052            || !self.feature_flags.split_checkpoints_in_consensus_handler
5053        {
5054            return;
5055        }
5056
5057        if !mysten_common::in_test_configuration() {
5058            return;
5059        }
5060
5061        use rand::{Rng, SeedableRng, rngs::StdRng};
5062        let mut rng = StdRng::from_seed(*seed);
5063        let max_txns = rng.gen_range(10..=100u64);
5064        info!("seeded test override: max_transactions_per_checkpoint = {max_txns}");
5065        self.max_transactions_per_checkpoint = Some(max_txns);
5066    }
5067
5068    // Extract the bytecode verifier config from this protocol config.
5069    // If used during signing, `signing_limits` should be set.
5070    // The third limit configures`sanity_check_with_regex_reference_safety`,
5071    // which runs the new regex-based reference safety check to check that it is strictly more
5072    // permissive than the current implementation.
5073    pub fn verifier_config(&self, signing_limits: Option<(usize, usize, usize)>) -> VerifierConfig {
5074        let (
5075            max_back_edges_per_function,
5076            max_back_edges_per_module,
5077            sanity_check_with_regex_reference_safety,
5078        ) = if let Some((
5079            max_back_edges_per_function,
5080            max_back_edges_per_module,
5081            sanity_check_with_regex_reference_safety,
5082        )) = signing_limits
5083        {
5084            (
5085                Some(max_back_edges_per_function),
5086                Some(max_back_edges_per_module),
5087                Some(sanity_check_with_regex_reference_safety),
5088            )
5089        } else {
5090            (None, None, None)
5091        };
5092
5093        let additional_borrow_checks = if signing_limits.is_some() {
5094            // always turn on additional borrow checks during signing
5095            true
5096        } else {
5097            self.additional_borrow_checks()
5098        };
5099        let deprecate_global_storage_ops = if signing_limits.is_some() {
5100            // always turn on additional vector borrow checks during signing
5101            true
5102        } else {
5103            self.deprecate_global_storage_ops()
5104        };
5105
5106        VerifierConfig {
5107            max_loop_depth: Some(self.max_loop_depth() as usize),
5108            max_generic_instantiation_length: Some(self.max_generic_instantiation_length() as usize),
5109            max_function_parameters: Some(self.max_function_parameters() as usize),
5110            max_basic_blocks: Some(self.max_basic_blocks() as usize),
5111            max_value_stack_size: self.max_value_stack_size() as usize,
5112            max_type_nodes: Some(self.max_type_nodes() as usize),
5113            max_push_size: Some(self.max_push_size() as usize),
5114            max_dependency_depth: Some(self.max_dependency_depth() as usize),
5115            max_fields_in_struct: Some(self.max_fields_in_struct() as usize),
5116            max_function_definitions: Some(self.max_function_definitions() as usize),
5117            max_data_definitions: Some(self.max_struct_definitions() as usize),
5118            max_constant_vector_len: Some(self.max_move_vector_len()),
5119            max_back_edges_per_function,
5120            max_back_edges_per_module,
5121            max_basic_blocks_in_script: None,
5122            max_identifier_len: self.max_move_identifier_len_as_option(), // Before protocol version 9, there was no limit
5123            disallow_self_identifier: self.feature_flags.disallow_self_identifier,
5124            allow_receiving_object_id: self.allow_receiving_object_id(),
5125            reject_mutable_random_on_entry_functions: self
5126                .reject_mutable_random_on_entry_functions(),
5127            bytecode_version: self.move_binary_format_version(),
5128            max_variants_in_enum: self.max_move_enum_variants_as_option(),
5129            additional_borrow_checks,
5130            better_loader_errors: self.better_loader_errors(),
5131            private_generics_verifier_v2: self.private_generics_verifier_v2(),
5132            sanity_check_with_regex_reference_safety: sanity_check_with_regex_reference_safety
5133                .map(|limit| limit as u128),
5134            deprecate_global_storage_ops,
5135            disable_entry_point_signature_check: self.disable_entry_point_signature_check(),
5136            switch_to_regex_reference_safety: false,
5137            disallow_jump_orphans: self.disallow_jump_orphans(),
5138        }
5139    }
5140
5141    pub fn binary_config(
5142        &self,
5143        override_deprecate_global_storage_ops_during_deserialization: Option<bool>,
5144    ) -> BinaryConfig {
5145        let deprecate_global_storage_ops =
5146            override_deprecate_global_storage_ops_during_deserialization
5147                .unwrap_or_else(|| self.deprecate_global_storage_ops());
5148        BinaryConfig::new(
5149            self.move_binary_format_version(),
5150            self.min_move_binary_format_version_as_option()
5151                .unwrap_or(VERSION_1),
5152            self.no_extraneous_module_bytes(),
5153            deprecate_global_storage_ops,
5154            TableConfig {
5155                module_handles: self.binary_module_handles_as_option().unwrap_or(u16::MAX),
5156                datatype_handles: self.binary_struct_handles_as_option().unwrap_or(u16::MAX),
5157                function_handles: self.binary_function_handles_as_option().unwrap_or(u16::MAX),
5158                function_instantiations: self
5159                    .binary_function_instantiations_as_option()
5160                    .unwrap_or(u16::MAX),
5161                signatures: self.binary_signatures_as_option().unwrap_or(u16::MAX),
5162                constant_pool: self.binary_constant_pool_as_option().unwrap_or(u16::MAX),
5163                identifiers: self.binary_identifiers_as_option().unwrap_or(u16::MAX),
5164                address_identifiers: self
5165                    .binary_address_identifiers_as_option()
5166                    .unwrap_or(u16::MAX),
5167                struct_defs: self.binary_struct_defs_as_option().unwrap_or(u16::MAX),
5168                struct_def_instantiations: self
5169                    .binary_struct_def_instantiations_as_option()
5170                    .unwrap_or(u16::MAX),
5171                function_defs: self.binary_function_defs_as_option().unwrap_or(u16::MAX),
5172                field_handles: self.binary_field_handles_as_option().unwrap_or(u16::MAX),
5173                field_instantiations: self
5174                    .binary_field_instantiations_as_option()
5175                    .unwrap_or(u16::MAX),
5176                friend_decls: self.binary_friend_decls_as_option().unwrap_or(u16::MAX),
5177                enum_defs: self.binary_enum_defs_as_option().unwrap_or(u16::MAX),
5178                enum_def_instantiations: self
5179                    .binary_enum_def_instantiations_as_option()
5180                    .unwrap_or(u16::MAX),
5181                variant_handles: self.binary_variant_handles_as_option().unwrap_or(u16::MAX),
5182                variant_instantiation_handles: self
5183                    .binary_variant_instantiation_handles_as_option()
5184                    .unwrap_or(u16::MAX),
5185            },
5186        )
5187    }
5188
5189    /// Override one or more settings in the config, for testing.
5190    /// This must be called at the beginning of the test, before get_for_(min|max)_version is
5191    /// called, since those functions cache their return value.
5192    #[cfg(not(msim))]
5193    pub fn apply_overrides_for_testing(
5194        override_fn: impl Fn(ProtocolVersion, Self) -> Self + Send + Sync + 'static,
5195    ) -> OverrideGuard {
5196        let mut cur = CONFIG_OVERRIDE.lock().unwrap();
5197        assert!(cur.is_none(), "config override already present");
5198        *cur = Some(Box::new(override_fn));
5199        OverrideGuard
5200    }
5201
5202    /// Override one or more settings in the config, for testing.
5203    /// This must be called at the beginning of the test, before get_for_(min|max)_version is
5204    /// called, since those functions cache their return value.
5205    #[cfg(msim)]
5206    pub fn apply_overrides_for_testing(
5207        override_fn: impl Fn(ProtocolVersion, Self) -> Self + Send + 'static,
5208    ) -> OverrideGuard {
5209        CONFIG_OVERRIDE.with(|ovr| {
5210            let mut cur = ovr.borrow_mut();
5211            assert!(cur.is_none(), "config override already present");
5212            *cur = Some(Box::new(override_fn));
5213            OverrideGuard
5214        })
5215    }
5216
5217    #[cfg(not(msim))]
5218    fn apply_config_override(version: ProtocolVersion, mut ret: Self) -> Self {
5219        if let Some(override_fn) = CONFIG_OVERRIDE.lock().unwrap().as_ref() {
5220            warn!(
5221                "overriding ProtocolConfig settings with custom settings (you should not see this log outside of tests)"
5222            );
5223            ret = override_fn(version, ret);
5224        }
5225        ret
5226    }
5227
5228    #[cfg(msim)]
5229    fn apply_config_override(version: ProtocolVersion, ret: Self) -> Self {
5230        CONFIG_OVERRIDE.with(|ovr| {
5231            if let Some(override_fn) = &*ovr.borrow() {
5232                warn!(
5233                    "overriding ProtocolConfig settings with custom settings (you should not see this log outside of tests)"
5234                );
5235                override_fn(version, ret)
5236            } else {
5237                ret
5238            }
5239        })
5240    }
5241}
5242
5243// Setters for tests.
5244// This is only needed for feature_flags. Please suffix each setter with `_for_testing`.
5245// Non-feature_flags should already have test setters defined through macros.
5246impl ProtocolConfig {
5247    pub fn set_advance_to_highest_supported_protocol_version_for_testing(&mut self, val: bool) {
5248        self.feature_flags
5249            .advance_to_highest_supported_protocol_version = val
5250    }
5251    pub fn set_commit_root_state_digest_supported_for_testing(&mut self, val: bool) {
5252        self.feature_flags.commit_root_state_digest = val
5253    }
5254    pub fn set_zklogin_auth_for_testing(&mut self, val: bool) {
5255        self.feature_flags.zklogin_auth = val
5256    }
5257    pub fn set_enable_jwk_consensus_updates_for_testing(&mut self, val: bool) {
5258        self.feature_flags.enable_jwk_consensus_updates = val
5259    }
5260    pub fn set_random_beacon_for_testing(&mut self, val: bool) {
5261        self.feature_flags.random_beacon = val
5262    }
5263
5264    pub fn set_upgraded_multisig_for_testing(&mut self, val: bool) {
5265        self.feature_flags.upgraded_multisig_supported = val
5266    }
5267    pub fn set_accept_zklogin_in_multisig_for_testing(&mut self, val: bool) {
5268        self.feature_flags.accept_zklogin_in_multisig = val
5269    }
5270
5271    pub fn set_shared_object_deletion_for_testing(&mut self, val: bool) {
5272        self.feature_flags.shared_object_deletion = val;
5273    }
5274
5275    pub fn set_narwhal_new_leader_election_schedule_for_testing(&mut self, val: bool) {
5276        self.feature_flags.narwhal_new_leader_election_schedule = val;
5277    }
5278
5279    pub fn set_receive_object_for_testing(&mut self, val: bool) {
5280        self.feature_flags.receive_objects = val
5281    }
5282    pub fn set_narwhal_certificate_v2_for_testing(&mut self, val: bool) {
5283        self.feature_flags.narwhal_certificate_v2 = val
5284    }
5285    pub fn set_verify_legacy_zklogin_address_for_testing(&mut self, val: bool) {
5286        self.feature_flags.verify_legacy_zklogin_address = val
5287    }
5288
5289    pub fn set_per_object_congestion_control_mode_for_testing(
5290        &mut self,
5291        val: PerObjectCongestionControlMode,
5292    ) {
5293        self.feature_flags.per_object_congestion_control_mode = val;
5294    }
5295
5296    pub fn set_consensus_choice_for_testing(&mut self, val: ConsensusChoice) {
5297        self.feature_flags.consensus_choice = val;
5298    }
5299
5300    pub fn set_consensus_network_for_testing(&mut self, val: ConsensusNetwork) {
5301        self.feature_flags.consensus_network = val;
5302    }
5303
5304    pub fn set_zklogin_max_epoch_upper_bound_delta_for_testing(&mut self, val: Option<u64>) {
5305        self.feature_flags.zklogin_max_epoch_upper_bound_delta = val
5306    }
5307
5308    pub fn set_disable_bridge_for_testing(&mut self) {
5309        self.feature_flags.bridge = false
5310    }
5311
5312    pub fn set_mysticeti_num_leaders_per_round_for_testing(&mut self, val: Option<usize>) {
5313        self.feature_flags.mysticeti_num_leaders_per_round = val;
5314    }
5315
5316    pub fn set_enable_soft_bundle_for_testing(&mut self, val: bool) {
5317        self.feature_flags.soft_bundle = val;
5318    }
5319
5320    pub fn set_passkey_auth_for_testing(&mut self, val: bool) {
5321        self.feature_flags.passkey_auth = val
5322    }
5323
5324    pub fn set_enable_party_transfer_for_testing(&mut self, val: bool) {
5325        self.feature_flags.enable_party_transfer = val
5326    }
5327
5328    pub fn set_consensus_distributed_vote_scoring_strategy_for_testing(&mut self, val: bool) {
5329        self.feature_flags
5330            .consensus_distributed_vote_scoring_strategy = val;
5331    }
5332
5333    pub fn set_consensus_round_prober_for_testing(&mut self, val: bool) {
5334        self.feature_flags.consensus_round_prober = val;
5335    }
5336
5337    pub fn set_disallow_new_modules_in_deps_only_packages_for_testing(&mut self, val: bool) {
5338        self.feature_flags
5339            .disallow_new_modules_in_deps_only_packages = val;
5340    }
5341
5342    pub fn set_correct_gas_payment_limit_check_for_testing(&mut self, val: bool) {
5343        self.feature_flags.correct_gas_payment_limit_check = val;
5344    }
5345
5346    pub fn set_address_aliases_for_testing(&mut self, val: bool) {
5347        self.feature_flags.address_aliases = val;
5348    }
5349
5350    pub fn set_consensus_round_prober_probe_accepted_rounds(&mut self, val: bool) {
5351        self.feature_flags
5352            .consensus_round_prober_probe_accepted_rounds = val;
5353    }
5354
5355    pub fn set_mysticeti_fastpath_for_testing(&mut self, val: bool) {
5356        self.feature_flags.mysticeti_fastpath = val;
5357    }
5358
5359    pub fn set_accept_passkey_in_multisig_for_testing(&mut self, val: bool) {
5360        self.feature_flags.accept_passkey_in_multisig = val;
5361    }
5362
5363    pub fn set_consensus_batched_block_sync_for_testing(&mut self, val: bool) {
5364        self.feature_flags.consensus_batched_block_sync = val;
5365    }
5366
5367    pub fn set_record_time_estimate_processed_for_testing(&mut self, val: bool) {
5368        self.feature_flags.record_time_estimate_processed = val;
5369    }
5370
5371    pub fn set_prepend_prologue_tx_in_consensus_commit_in_checkpoints_for_testing(
5372        &mut self,
5373        val: bool,
5374    ) {
5375        self.feature_flags
5376            .prepend_prologue_tx_in_consensus_commit_in_checkpoints = val;
5377    }
5378
5379    pub fn enable_accumulators_for_testing(&mut self) {
5380        self.feature_flags.enable_accumulators = true;
5381    }
5382
5383    pub fn disable_accumulators_for_testing(&mut self) {
5384        self.feature_flags.enable_accumulators = false;
5385        self.feature_flags.enable_address_balance_gas_payments = false;
5386    }
5387
5388    pub fn enable_coin_reservation_for_testing(&mut self) {
5389        self.feature_flags.enable_coin_reservation_obj_refs = true;
5390        self.feature_flags
5391            .convert_withdrawal_compatibility_ptb_arguments = true;
5392        // Ensure execution_version >= 4 so new_vm_enabled() returns true,
5393        // which is required for enable_coin_reservation_obj_refs() to return true.
5394        self.execution_version = Some(self.execution_version.map_or(4, |v| v.max(4)));
5395    }
5396
5397    pub fn disable_coin_reservation_for_testing(&mut self) {
5398        self.feature_flags.enable_coin_reservation_obj_refs = false;
5399        self.feature_flags
5400            .convert_withdrawal_compatibility_ptb_arguments = false;
5401    }
5402
5403    pub fn create_root_accumulator_object_for_testing(&mut self) {
5404        self.feature_flags.create_root_accumulator_object = true;
5405    }
5406
5407    pub fn disable_create_root_accumulator_object_for_testing(&mut self) {
5408        self.feature_flags.create_root_accumulator_object = false;
5409    }
5410
5411    pub fn enable_address_balance_gas_payments_for_testing(&mut self) {
5412        self.feature_flags.enable_accumulators = true;
5413        self.feature_flags.allow_private_accumulator_entrypoints = true;
5414        self.feature_flags.enable_address_balance_gas_payments = true;
5415        self.feature_flags.address_balance_gas_check_rgp_at_signing = true;
5416        self.feature_flags.address_balance_gas_reject_gas_coin_arg = false;
5417        self.execution_version = Some(self.execution_version.map_or(4, |v| v.max(4)))
5418    }
5419
5420    pub fn disable_address_balance_gas_payments_for_testing(&mut self) {
5421        self.feature_flags.enable_address_balance_gas_payments = false;
5422    }
5423
5424    pub fn enable_gasless_for_testing(&mut self) {
5425        self.enable_address_balance_gas_payments_for_testing();
5426        self.feature_flags.enable_gasless = true;
5427        self.feature_flags.gasless_verify_remaining_balance = true;
5428        self.gasless_max_computation_units = Some(5_000);
5429        self.gasless_allowed_token_types = Some(vec![]);
5430        self.gasless_max_tps = Some(1000);
5431        self.gasless_max_tx_size_bytes = Some(16 * 1024);
5432    }
5433
5434    pub fn disable_gasless_for_testing(&mut self) {
5435        self.feature_flags.enable_gasless = false;
5436        self.gasless_max_computation_units = None;
5437        self.gasless_allowed_token_types = None;
5438    }
5439
5440    pub fn enable_multi_epoch_transaction_expiration_for_testing(&mut self) {
5441        self.feature_flags.enable_multi_epoch_transaction_expiration = true;
5442    }
5443
5444    pub fn enable_authenticated_event_streams_for_testing(&mut self) {
5445        self.enable_accumulators_for_testing();
5446        self.feature_flags.enable_authenticated_event_streams = true;
5447        self.feature_flags
5448            .include_checkpoint_artifacts_digest_in_summary = true;
5449        self.feature_flags.split_checkpoints_in_consensus_handler = true;
5450    }
5451
5452    pub fn disable_authenticated_event_streams_for_testing(&mut self) {
5453        self.feature_flags.enable_authenticated_event_streams = false;
5454    }
5455
5456    pub fn disable_randomize_checkpoint_tx_limit_for_testing(&mut self) {
5457        self.feature_flags.randomize_checkpoint_tx_limit_in_tests = false;
5458    }
5459
5460    pub fn enable_non_exclusive_writes_for_testing(&mut self) {
5461        self.feature_flags.enable_non_exclusive_writes = true;
5462    }
5463
5464    pub fn set_relax_valid_during_for_owned_inputs_for_testing(&mut self, val: bool) {
5465        self.feature_flags.relax_valid_during_for_owned_inputs = val;
5466    }
5467
5468    pub fn set_ignore_execution_time_observations_after_certs_closed_for_testing(
5469        &mut self,
5470        val: bool,
5471    ) {
5472        self.feature_flags
5473            .ignore_execution_time_observations_after_certs_closed = val;
5474    }
5475
5476    pub fn set_consensus_checkpoint_signature_key_includes_digest_for_testing(
5477        &mut self,
5478        val: bool,
5479    ) {
5480        self.feature_flags
5481            .consensus_checkpoint_signature_key_includes_digest = val;
5482    }
5483
5484    pub fn set_cancel_for_failed_dkg_early_for_testing(&mut self, val: bool) {
5485        self.feature_flags.cancel_for_failed_dkg_early = val;
5486    }
5487
5488    pub fn set_always_advance_dkg_to_resolution_for_testing(&mut self, val: bool) {
5489        self.feature_flags.always_advance_dkg_to_resolution = val;
5490    }
5491
5492    pub fn set_use_mfp_txns_in_load_initial_object_debts_for_testing(&mut self, val: bool) {
5493        self.feature_flags.use_mfp_txns_in_load_initial_object_debts = val;
5494    }
5495
5496    pub fn set_authority_capabilities_v2_for_testing(&mut self, val: bool) {
5497        self.feature_flags.authority_capabilities_v2 = val;
5498    }
5499
5500    pub fn allow_references_in_ptbs_for_testing(&mut self) {
5501        self.feature_flags.allow_references_in_ptbs = true;
5502    }
5503
5504    pub fn set_consensus_skip_gced_accept_votes_for_testing(&mut self, val: bool) {
5505        self.feature_flags.consensus_skip_gced_accept_votes = val;
5506    }
5507
5508    pub fn set_enable_object_funds_withdraw_for_testing(&mut self, val: bool) {
5509        self.feature_flags.enable_object_funds_withdraw = val;
5510    }
5511
5512    pub fn set_split_checkpoints_in_consensus_handler_for_testing(&mut self, val: bool) {
5513        self.feature_flags.split_checkpoints_in_consensus_handler = val;
5514    }
5515
5516    pub fn set_merge_randomness_into_checkpoint_for_testing(&mut self, val: bool) {
5517        self.feature_flags.merge_randomness_into_checkpoint = val;
5518    }
5519}
5520
5521#[cfg(not(msim))]
5522type OverrideFn = dyn Fn(ProtocolVersion, ProtocolConfig) -> ProtocolConfig + Send + Sync;
5523
5524#[cfg(not(msim))]
5525static CONFIG_OVERRIDE: Mutex<Option<Box<OverrideFn>>> = Mutex::new(None);
5526
5527#[cfg(msim)]
5528type OverrideFn = dyn Fn(ProtocolVersion, ProtocolConfig) -> ProtocolConfig + Send;
5529
5530#[cfg(msim)]
5531thread_local! {
5532    static CONFIG_OVERRIDE: RefCell<Option<Box<OverrideFn>>> = RefCell::new(None);
5533}
5534
5535#[must_use]
5536pub struct OverrideGuard;
5537
5538#[cfg(not(msim))]
5539impl Drop for OverrideGuard {
5540    fn drop(&mut self) {
5541        info!("restoring override fn");
5542        *CONFIG_OVERRIDE.lock().unwrap() = None;
5543    }
5544}
5545
5546#[cfg(msim)]
5547impl Drop for OverrideGuard {
5548    fn drop(&mut self) {
5549        info!("restoring override fn");
5550        CONFIG_OVERRIDE.with(|ovr| {
5551            *ovr.borrow_mut() = None;
5552        });
5553    }
5554}
5555
5556/// Defines which limit got crossed.
5557/// The value which crossed the limit and value of the limit crossed are embedded
5558#[derive(PartialEq, Eq)]
5559pub enum LimitThresholdCrossed {
5560    None,
5561    Soft(u128, u128),
5562    Hard(u128, u128),
5563}
5564
5565/// Convenience function for comparing limit ranges
5566/// V::MAX must be at >= U::MAX and T::MAX
5567pub fn check_limit_in_range<T: Into<V>, U: Into<V>, V: PartialOrd + Into<u128>>(
5568    x: T,
5569    soft_limit: U,
5570    hard_limit: V,
5571) -> LimitThresholdCrossed {
5572    let x: V = x.into();
5573    let soft_limit: V = soft_limit.into();
5574
5575    debug_assert!(soft_limit <= hard_limit);
5576
5577    // It is important to preserve this comparison order because if soft_limit == hard_limit
5578    // we want LimitThresholdCrossed::Hard
5579    if x >= hard_limit {
5580        LimitThresholdCrossed::Hard(x.into(), hard_limit.into())
5581    } else if x < soft_limit {
5582        LimitThresholdCrossed::None
5583    } else {
5584        LimitThresholdCrossed::Soft(x.into(), soft_limit.into())
5585    }
5586}
5587
5588#[macro_export]
5589macro_rules! check_limit {
5590    ($x:expr, $hard:expr) => {
5591        check_limit!($x, $hard, $hard)
5592    };
5593    ($x:expr, $soft:expr, $hard:expr) => {
5594        check_limit_in_range($x as u64, $soft, $hard)
5595    };
5596}
5597
5598/// Used to check which limits were crossed if the TX is metered (not system tx)
5599/// Args are: is_metered, value_to_check, metered_limit, unmetered_limit
5600/// metered_limit is always less than or equal to unmetered_hard_limit
5601#[macro_export]
5602macro_rules! check_limit_by_meter {
5603    ($is_metered:expr, $x:expr, $metered_limit:expr, $unmetered_hard_limit:expr, $metric:expr) => {{
5604        // If this is metered, we use the metered_limit limit as the upper bound
5605        let (h, metered_str) = if $is_metered {
5606            ($metered_limit, "metered")
5607        } else {
5608            // Unmetered gets more headroom
5609            ($unmetered_hard_limit, "unmetered")
5610        };
5611        use sui_protocol_config::check_limit_in_range;
5612        let result = check_limit_in_range($x as u64, $metered_limit, h);
5613        match result {
5614            LimitThresholdCrossed::None => {}
5615            LimitThresholdCrossed::Soft(_, _) => {
5616                $metric.with_label_values(&[metered_str, "soft"]).inc();
5617            }
5618            LimitThresholdCrossed::Hard(_, _) => {
5619                $metric.with_label_values(&[metered_str, "hard"]).inc();
5620            }
5621        };
5622        result
5623    }};
5624}
5625
5626// Amendments tables
5627
5628pub type Amendments = BTreeMap<AccountAddress, BTreeMap<AccountAddress, AccountAddress>>;
5629
5630static MAINNET_LINKAGE_AMENDMENTS: LazyLock<Arc<Amendments>> =
5631    LazyLock::new(|| parse_amendments(include_str!("mainnet_amendments.json")));
5632
5633static TESTNET_LINKAGE_AMENDMENTS: LazyLock<Arc<Amendments>> =
5634    LazyLock::new(|| parse_amendments(include_str!("testnet_amendments.json")));
5635
5636fn parse_amendments(json: &str) -> Arc<Amendments> {
5637    #[derive(serde::Deserialize)]
5638    struct AmendmentEntry {
5639        root: String,
5640        deps: Vec<DepEntry>,
5641    }
5642
5643    #[derive(serde::Deserialize)]
5644    struct DepEntry {
5645        original_id: String,
5646        version_id: String,
5647    }
5648
5649    let entries: Vec<AmendmentEntry> =
5650        serde_json::from_str(json).expect("Failed to parse amendments JSON");
5651    let mut amendments = BTreeMap::new();
5652    for entry in entries {
5653        let root_id = AccountAddress::from_hex_literal(&entry.root).unwrap();
5654        let mut dep_ids = BTreeMap::new();
5655        for dep in entry.deps {
5656            let orig_id = AccountAddress::from_hex_literal(&dep.original_id).unwrap();
5657            let upgraded_id = AccountAddress::from_hex_literal(&dep.version_id).unwrap();
5658            assert!(
5659                dep_ids.insert(orig_id, upgraded_id).is_none(),
5660                "Duplicate original ID in amendments table"
5661            );
5662        }
5663        assert!(
5664            amendments.insert(root_id, dep_ids).is_none(),
5665            "Duplicate root ID in amendments table"
5666        );
5667    }
5668    Arc::new(amendments)
5669}
5670
5671#[cfg(all(test, not(msim)))]
5672mod test {
5673    use insta::assert_yaml_snapshot;
5674
5675    use super::*;
5676
5677    #[test]
5678    fn snapshot_tests() {
5679        println!("\n============================================================================");
5680        println!("!                                                                          !");
5681        println!("! IMPORTANT: never update snapshots from this test. only add new versions! !");
5682        println!("!                                                                          !");
5683        println!("============================================================================\n");
5684        for chain_id in &[Chain::Unknown, Chain::Mainnet, Chain::Testnet] {
5685            // make Chain::Unknown snapshots compatible with pre-chain-id snapshots so that we
5686            // don't break the release-time compatibility tests. Once Chain Id configs have been
5687            // released everywhere, we can remove this and only test Mainnet and Testnet
5688            let chain_str = match chain_id {
5689                Chain::Unknown => "".to_string(),
5690                _ => format!("{:?}_", chain_id),
5691            };
5692            for i in MIN_PROTOCOL_VERSION..=MAX_PROTOCOL_VERSION {
5693                let cur = ProtocolVersion::new(i);
5694                assert_yaml_snapshot!(
5695                    format!("{}version_{}", chain_str, cur.as_u64()),
5696                    ProtocolConfig::get_for_version(cur, *chain_id)
5697                );
5698            }
5699        }
5700    }
5701
5702    #[test]
5703    fn test_getters() {
5704        let prot: ProtocolConfig =
5705            ProtocolConfig::get_for_version(ProtocolVersion::new(1), Chain::Unknown);
5706        assert_eq!(
5707            prot.max_arguments(),
5708            prot.max_arguments_as_option().unwrap()
5709        );
5710    }
5711
5712    #[test]
5713    fn test_setters() {
5714        let mut prot: ProtocolConfig =
5715            ProtocolConfig::get_for_version(ProtocolVersion::new(1), Chain::Unknown);
5716        prot.set_max_arguments_for_testing(123);
5717        assert_eq!(prot.max_arguments(), 123);
5718
5719        prot.set_max_arguments_from_str_for_testing("321".to_string());
5720        assert_eq!(prot.max_arguments(), 321);
5721
5722        prot.disable_max_arguments_for_testing();
5723        assert_eq!(prot.max_arguments_as_option(), None);
5724
5725        prot.set_attr_for_testing("max_arguments".to_string(), "456".to_string());
5726        assert_eq!(prot.max_arguments(), 456);
5727    }
5728
5729    #[test]
5730    fn test_get_for_version_if_supported_applies_test_overrides() {
5731        let before =
5732            ProtocolConfig::get_for_version_if_supported(ProtocolVersion::new(1), Chain::Unknown)
5733                .unwrap();
5734
5735        assert!(!before.enable_coin_reservation_obj_refs());
5736
5737        let _guard = ProtocolConfig::apply_overrides_for_testing(|_, mut cfg| {
5738            cfg.enable_coin_reservation_for_testing();
5739            cfg
5740        });
5741
5742        let after =
5743            ProtocolConfig::get_for_version_if_supported(ProtocolVersion::new(1), Chain::Unknown)
5744                .unwrap();
5745
5746        assert!(after.enable_coin_reservation_obj_refs());
5747    }
5748
5749    #[test]
5750    #[should_panic(expected = "unsupported version")]
5751    fn max_version_test() {
5752        // When this does not panic, version higher than MAX_PROTOCOL_VERSION exists.
5753        // To fix, bump MAX_PROTOCOL_VERSION or disable this check for the version.
5754        let _ = ProtocolConfig::get_for_version_impl(
5755            ProtocolVersion::new(MAX_PROTOCOL_VERSION + 1),
5756            Chain::Unknown,
5757        );
5758    }
5759
5760    #[test]
5761    fn lookup_by_string_test() {
5762        let prot: ProtocolConfig =
5763            ProtocolConfig::get_for_version(ProtocolVersion::new(1), Chain::Unknown);
5764        // Does not exist
5765        assert!(prot.lookup_attr("some random string".to_string()).is_none());
5766
5767        assert!(
5768            prot.lookup_attr("max_arguments".to_string())
5769                == Some(ProtocolConfigValue::u32(prot.max_arguments())),
5770        );
5771
5772        // We didnt have this in version 1
5773        assert!(
5774            prot.lookup_attr("max_move_identifier_len".to_string())
5775                .is_none()
5776        );
5777
5778        // But we did in version 9
5779        let prot: ProtocolConfig =
5780            ProtocolConfig::get_for_version(ProtocolVersion::new(9), Chain::Unknown);
5781        assert!(
5782            prot.lookup_attr("max_move_identifier_len".to_string())
5783                == Some(ProtocolConfigValue::u64(prot.max_move_identifier_len()))
5784        );
5785
5786        let prot: ProtocolConfig =
5787            ProtocolConfig::get_for_version(ProtocolVersion::new(1), Chain::Unknown);
5788        // We didnt have this in version 1
5789        assert!(
5790            prot.attr_map()
5791                .get("max_move_identifier_len")
5792                .unwrap()
5793                .is_none()
5794        );
5795        // We had this in version 1
5796        assert!(
5797            prot.attr_map().get("max_arguments").unwrap()
5798                == &Some(ProtocolConfigValue::u32(prot.max_arguments()))
5799        );
5800
5801        // Check feature flags
5802        let prot: ProtocolConfig =
5803            ProtocolConfig::get_for_version(ProtocolVersion::new(1), Chain::Unknown);
5804        // Does not exist
5805        assert!(
5806            prot.feature_flags
5807                .lookup_attr("some random string".to_owned())
5808                .is_none()
5809        );
5810        assert!(
5811            !prot
5812                .feature_flags
5813                .attr_map()
5814                .contains_key("some random string")
5815        );
5816
5817        // Was false in v1
5818        assert!(
5819            prot.feature_flags
5820                .lookup_attr("package_upgrades".to_owned())
5821                == Some(false)
5822        );
5823        assert!(
5824            prot.feature_flags
5825                .attr_map()
5826                .get("package_upgrades")
5827                .unwrap()
5828                == &false
5829        );
5830        let prot: ProtocolConfig =
5831            ProtocolConfig::get_for_version(ProtocolVersion::new(4), Chain::Unknown);
5832        // Was true from v3 and up
5833        assert!(
5834            prot.feature_flags
5835                .lookup_attr("package_upgrades".to_owned())
5836                == Some(true)
5837        );
5838        assert!(
5839            prot.feature_flags
5840                .attr_map()
5841                .get("package_upgrades")
5842                .unwrap()
5843                == &true
5844        );
5845    }
5846
5847    #[test]
5848    fn limit_range_fn_test() {
5849        let low = 100u32;
5850        let high = 10000u64;
5851
5852        assert!(check_limit!(1u8, low, high) == LimitThresholdCrossed::None);
5853        assert!(matches!(
5854            check_limit!(255u16, low, high),
5855            LimitThresholdCrossed::Soft(255u128, 100)
5856        ));
5857        // This wont compile because lossy
5858        //assert!(check_limit!(100000000u128, low, high) == LimitThresholdCrossed::None);
5859        // This wont compile because lossy
5860        //assert!(check_limit!(100000000usize, low, high) == LimitThresholdCrossed::None);
5861
5862        assert!(matches!(
5863            check_limit!(2550000u64, low, high),
5864            LimitThresholdCrossed::Hard(2550000, 10000)
5865        ));
5866
5867        assert!(matches!(
5868            check_limit!(2550000u64, high, high),
5869            LimitThresholdCrossed::Hard(2550000, 10000)
5870        ));
5871
5872        assert!(matches!(
5873            check_limit!(1u8, high),
5874            LimitThresholdCrossed::None
5875        ));
5876
5877        assert!(check_limit!(255u16, high) == LimitThresholdCrossed::None);
5878
5879        assert!(matches!(
5880            check_limit!(2550000u64, high),
5881            LimitThresholdCrossed::Hard(2550000, 10000)
5882        ));
5883    }
5884
5885    #[test]
5886    fn linkage_amendments_load() {
5887        let mainnet = LazyLock::force(&MAINNET_LINKAGE_AMENDMENTS);
5888        let testnet = LazyLock::force(&TESTNET_LINKAGE_AMENDMENTS);
5889        assert!(!mainnet.is_empty(), "mainnet amendments must not be empty");
5890        assert!(!testnet.is_empty(), "testnet amendments must not be empty");
5891    }
5892
5893    #[test]
5894    fn render_scalar_fields_use_precision_safe_encoding() {
5895        use mysten_common::rpc_format::Unmetered;
5896
5897        let config = ProtocolConfig::get_for_max_version_UNSAFE();
5898        let rendered = config
5899            .render::<serde_json::Value>(&mut Unmetered)
5900            .expect("render should succeed");
5901
5902        let max_args = rendered
5903            .get("max_arguments")
5904            .expect("max_arguments set at max version");
5905        assert!(
5906            max_args.is_number(),
5907            "u32 should render as number, got {max_args:?}",
5908        );
5909
5910        let max_tx_size = rendered
5911            .get("max_tx_size_bytes")
5912            .expect("max_tx_size_bytes set at max version");
5913        assert!(
5914            max_tx_size.is_string(),
5915            "u64 should render as string, got {max_tx_size:?}",
5916        );
5917    }
5918
5919    #[test]
5920    fn render_includes_non_scalar_gasless_allowlist_as_json() {
5921        use mysten_common::rpc_format::Unmetered;
5922        use serde_json::json;
5923
5924        let mut config = ProtocolConfig::get_for_max_version_UNSAFE();
5925        config.set_gasless_allowed_token_types_for_testing(vec![
5926            ("0xa::usdc::USDC".to_string(), 10_000),
5927            ("0xb::usdt::USDT".to_string(), 0),
5928        ]);
5929
5930        let rendered = config
5931            .render::<serde_json::Value>(&mut Unmetered)
5932            .expect("render should succeed under Unmetered budget");
5933        let allowlist = rendered
5934            .get("gasless_allowed_token_types")
5935            .expect("entry should be present after the testing setter");
5936
5937        // u64 values render as strings to preserve JS precision; the tuple becomes a 2-element
5938        // JSON array.
5939        assert_eq!(
5940            allowlist,
5941            &json!([["0xa::usdc::USDC", "10000"], ["0xb::usdt::USDT", "0"],]),
5942        );
5943    }
5944
5945    #[test]
5946    fn render_targets_prost_value_for_grpc() {
5947        use mysten_common::rpc_format::Unmetered;
5948        use prost_types::value::Kind;
5949
5950        let mut config = ProtocolConfig::get_for_max_version_UNSAFE();
5951        config.set_gasless_allowed_token_types_for_testing(vec![(
5952            "0xa::usdc::USDC".to_string(),
5953            10_000,
5954        )]);
5955
5956        let rendered = config
5957            .render::<prost_types::Value>(&mut Unmetered)
5958            .expect("render to prost Value should succeed");
5959        let allowlist = rendered
5960            .get("gasless_allowed_token_types")
5961            .expect("entry should be present after the testing setter");
5962
5963        // Outer ListValue with one inner ListValue carrying [coin_type_string, amount_string].
5964        let Some(Kind::ListValue(outer)) = &allowlist.kind else {
5965            panic!(
5966                "expected ListValue at the top level, got {:?}",
5967                allowlist.kind
5968            );
5969        };
5970        assert_eq!(outer.values.len(), 1, "one allowlisted entry");
5971        let Some(Kind::ListValue(entry)) = &outer.values[0].kind else {
5972            panic!("expected each entry to be a ListValue");
5973        };
5974        assert_eq!(entry.values.len(), 2, "entry has (coin_type, amount)");
5975
5976        let Some(Kind::StringValue(coin_type)) = &entry.values[0].kind else {
5977            panic!("expected coin_type as StringValue");
5978        };
5979        assert_eq!(coin_type, "0xa::usdc::USDC");
5980
5981        // u64 amount renders as a string, not a NumberValue — this is the precision-safe path.
5982        let Some(Kind::StringValue(amount)) = &entry.values[1].kind else {
5983            panic!(
5984                "expected minimum_transfer_amount as StringValue (precision-safe u64); got {:?}",
5985                entry.values[1].kind,
5986            );
5987        };
5988        assert_eq!(amount, "10000");
5989    }
5990
5991    #[test]
5992    fn render_emits_null_for_unset_protocol_versions() {
5993        use mysten_common::rpc_format::Unmetered;
5994
5995        let config = ProtocolConfig::get_for_version(1.into(), Chain::Unknown);
5996        let rendered = config
5997            .render::<serde_json::Value>(&mut Unmetered)
5998            .expect("render should succeed");
5999        // The gasless allowlist key is present in every version's keyset, but renders as JSON
6000        // `null` for versions that predate the feature. This keeps the keyset stable across
6001        // protocol versions so clients can distinguish "unknown key" from "present but unset".
6002        let entry = rendered
6003            .get("gasless_allowed_token_types")
6004            .expect("key should be present for every protocol version");
6005        assert!(
6006            entry.is_null(),
6007            "value should be null for pre-feature protocol version, got {entry:?}",
6008        );
6009    }
6010}