solana_feature_set/
lib.rs

1//! Collection of all runtime features.
2//!
3//! Steps to add a new feature are outlined below. Note that these steps only cover
4//! the process of getting a feature into the core Solana code.
5//! - For features that are unambiguously good (ie bug fixes), these steps are sufficient.
6//! - For features that should go up for community vote (ie fee structure changes), more
7//!   information on the additional steps to follow can be found at:
8//!   <https://spl.solana.com/feature-proposal#feature-proposal-life-cycle>
9//!
10//! 1. Generate a new keypair with `solana-keygen new --outfile feature.json --no-passphrase`
11//!    - Keypairs should be held by core contributors only. If you're a non-core contributor going
12//!      through these steps, the PR process will facilitate a keypair holder being picked. That
13//!      person will generate the keypair, provide pubkey for PR, and ultimately enable the feature.
14//! 2. Add a public module for the feature, specifying keypair pubkey as the id with
15//!    `solana_pubkey::declare_id!()` within the module.
16//!    Additionally, add an entry to `FEATURE_NAMES` map.
17//! 3. Add desired logic to check for and switch on feature availability.
18//!
19//! For more information on how features are picked up, see comments for `Feature`.
20#![cfg_attr(feature = "frozen-abi", feature(min_specialization))]
21
22use {
23    lazy_static::lazy_static,
24    solana_clock::{Epoch, Slot},
25    solana_epoch_schedule::EpochSchedule,
26    solana_hash::Hash,
27    solana_pubkey::Pubkey,
28    solana_sha256_hasher::Hasher,
29    std::collections::{HashMap, HashSet},
30};
31
32pub mod deprecate_rewards_sysvar {
33    solana_pubkey::declare_id!("GaBtBJvmS4Arjj5W1NmFcyvPjsHN38UGYDq2MDwbs9Qu");
34}
35
36pub mod pico_inflation {
37    solana_pubkey::declare_id!("4RWNif6C2WCNiKVW7otP4G7dkmkHGyKQWRpuZ1pxKU5m");
38}
39
40pub mod full_inflation {
41    pub mod devnet_and_testnet {
42        solana_pubkey::declare_id!("DT4n6ABDqs6w4bnfwrXT9rsprcPf6cdDga1egctaPkLC");
43    }
44
45    pub mod mainnet {
46        pub mod certusone {
47            pub mod vote {
48                solana_pubkey::declare_id!("BzBBveUDymEYoYzcMWNQCx3cd4jQs7puaVFHLtsbB6fm");
49            }
50            pub mod enable {
51                solana_pubkey::declare_id!("7XRJcS5Ud5vxGB54JbK9N2vBZVwnwdBNeJW1ibRgD9gx");
52            }
53        }
54    }
55}
56
57pub mod secp256k1_program_enabled {
58    solana_pubkey::declare_id!("E3PHP7w8kB7np3CTQ1qQ2tW3KCtjRSXBQgW9vM2mWv2Y");
59}
60
61pub mod spl_token_v2_multisig_fix {
62    solana_pubkey::declare_id!("E5JiFDQCwyC6QfT9REFyMpfK2mHcmv1GUDySU1Ue7TYv");
63}
64
65pub mod no_overflow_rent_distribution {
66    solana_pubkey::declare_id!("4kpdyrcj5jS47CZb2oJGfVxjYbsMm2Kx97gFyZrxxwXz");
67}
68
69pub mod filter_stake_delegation_accounts {
70    solana_pubkey::declare_id!("GE7fRxmW46K6EmCD9AMZSbnaJ2e3LfqCZzdHi9hmYAgi");
71}
72
73pub mod require_custodian_for_locked_stake_authorize {
74    solana_pubkey::declare_id!("D4jsDcXaqdW8tDAWn8H4R25Cdns2YwLneujSL1zvjW6R");
75}
76
77pub mod spl_token_v2_self_transfer_fix {
78    solana_pubkey::declare_id!("BL99GYhdjjcv6ys22C9wPgn2aTVERDbPHHo4NbS3hgp7");
79}
80
81pub mod warp_timestamp_again {
82    solana_pubkey::declare_id!("GvDsGDkH5gyzwpDhxNixx8vtx1kwYHH13RiNAPw27zXb");
83}
84
85pub mod check_init_vote_data {
86    solana_pubkey::declare_id!("3ccR6QpxGYsAbWyfevEtBNGfWV4xBffxRj2tD6A9i39F");
87}
88
89pub mod secp256k1_recover_syscall_enabled {
90    solana_pubkey::declare_id!("6RvdSWHh8oh72Dp7wMTS2DBkf3fRPtChfNrAo3cZZoXJ");
91}
92
93pub mod system_transfer_zero_check {
94    solana_pubkey::declare_id!("BrTR9hzw4WBGFP65AJMbpAo64DcA3U6jdPSga9fMV5cS");
95}
96
97pub mod blake3_syscall_enabled {
98    solana_pubkey::declare_id!("HTW2pSyErTj4BV6KBM9NZ9VBUJVxt7sacNWcf76wtzb3");
99}
100
101pub mod dedupe_config_program_signers {
102    solana_pubkey::declare_id!("8kEuAshXLsgkUEdcFVLqrjCGGHVWFW99ZZpxvAzzMtBp");
103}
104
105pub mod verify_tx_signatures_len {
106    solana_pubkey::declare_id!("EVW9B5xD9FFK7vw1SBARwMA4s5eRo5eKJdKpsBikzKBz");
107}
108
109pub mod vote_stake_checked_instructions {
110    solana_pubkey::declare_id!("BcWknVcgvonN8sL4HE4XFuEVgfcee5MwxWPAgP6ZV89X");
111}
112
113pub mod rent_for_sysvars {
114    solana_pubkey::declare_id!("BKCPBQQBZqggVnFso5nQ8rQ4RwwogYwjuUt9biBjxwNF");
115}
116
117pub mod libsecp256k1_0_5_upgrade_enabled {
118    solana_pubkey::declare_id!("DhsYfRjxfnh2g7HKJYSzT79r74Afa1wbHkAgHndrA1oy");
119}
120
121pub mod tx_wide_compute_cap {
122    solana_pubkey::declare_id!("5ekBxc8itEnPv4NzGJtr8BVVQLNMQuLMNQQj7pHoLNZ9");
123}
124
125pub mod spl_token_v2_set_authority_fix {
126    solana_pubkey::declare_id!("FToKNBYyiF4ky9s8WsmLBXHCht17Ek7RXaLZGHzzQhJ1");
127}
128
129pub mod merge_nonce_error_into_system_error {
130    solana_pubkey::declare_id!("21AWDosvp3pBamFW91KB35pNoaoZVTM7ess8nr2nt53B");
131}
132
133pub mod disable_fees_sysvar {
134    solana_pubkey::declare_id!("JAN1trEUEtZjgXYzNBYHU9DYd7GnThhXfFP7SzPXkPsG");
135}
136
137pub mod stake_merge_with_unmatched_credits_observed {
138    solana_pubkey::declare_id!("meRgp4ArRPhD3KtCY9c5yAf2med7mBLsjKTPeVUHqBL");
139}
140
141pub mod zk_token_sdk_enabled {
142    solana_pubkey::declare_id!("zk1snxsc6Fh3wsGNbbHAJNHiJoYgF29mMnTSusGx5EJ");
143}
144
145pub mod curve25519_syscall_enabled {
146    solana_pubkey::declare_id!("7rcw5UtqgDTBBv2EcynNfYckgdAaH1MAsCjKgXMkN7Ri");
147}
148
149pub mod curve25519_restrict_msm_length {
150    solana_pubkey::declare_id!("eca6zf6JJRjQsYYPkBHF3N32MTzur4n2WL4QiiacPCL");
151}
152
153pub mod versioned_tx_message_enabled {
154    solana_pubkey::declare_id!("3KZZ6Ks1885aGBQ45fwRcPXVBCtzUvxhUTkwKMR41Tca");
155}
156
157pub mod libsecp256k1_fail_on_bad_count {
158    solana_pubkey::declare_id!("8aXvSuopd1PUj7UhehfXJRg6619RHp8ZvwTyyJHdUYsj");
159}
160
161pub mod libsecp256k1_fail_on_bad_count2 {
162    solana_pubkey::declare_id!("54KAoNiUERNoWWUhTWWwXgym94gzoXFVnHyQwPA18V9A");
163}
164
165pub mod instructions_sysvar_owned_by_sysvar {
166    solana_pubkey::declare_id!("H3kBSaKdeiUsyHmeHqjJYNc27jesXZ6zWj3zWkowQbkV");
167}
168
169pub mod stake_program_advance_activating_credits_observed {
170    solana_pubkey::declare_id!("SAdVFw3RZvzbo6DvySbSdBnHN4gkzSTH9dSxesyKKPj");
171}
172
173pub mod credits_auto_rewind {
174    solana_pubkey::declare_id!("BUS12ciZ5gCoFafUHWW8qaFMMtwFQGVxjsDheWLdqBE2");
175}
176
177pub mod demote_program_write_locks {
178    solana_pubkey::declare_id!("3E3jV7v9VcdJL8iYZUMax9DiDno8j7EWUVbhm9RtShj2");
179}
180
181pub mod ed25519_program_enabled {
182    solana_pubkey::declare_id!("6ppMXNYLhVd7GcsZ5uV11wQEW7spppiMVfqQv5SXhDpX");
183}
184
185pub mod return_data_syscall_enabled {
186    solana_pubkey::declare_id!("DwScAzPUjuv65TMbDnFY7AgwmotzWy3xpEJMXM3hZFaB");
187}
188
189pub mod reduce_required_deploy_balance {
190    solana_pubkey::declare_id!("EBeznQDjcPG8491sFsKZYBi5S5jTVXMpAKNDJMQPS2kq");
191}
192
193pub mod sol_log_data_syscall_enabled {
194    solana_pubkey::declare_id!("6uaHcKPGUy4J7emLBgUTeufhJdiwhngW6a1R9B7c2ob9");
195}
196
197pub mod stakes_remove_delegation_if_inactive {
198    solana_pubkey::declare_id!("HFpdDDNQjvcXnXKec697HDDsyk6tFoWS2o8fkxuhQZpL");
199}
200
201pub mod do_support_realloc {
202    solana_pubkey::declare_id!("75m6ysz33AfLA5DDEzWM1obBrnPQRSsdVQ2nRmc8Vuu1");
203}
204
205pub mod prevent_calling_precompiles_as_programs {
206    solana_pubkey::declare_id!("4ApgRX3ud6p7LNMJmsuaAcZY5HWctGPr5obAsjB3A54d");
207}
208
209pub mod optimize_epoch_boundary_updates {
210    solana_pubkey::declare_id!("265hPS8k8xJ37ot82KEgjRunsUp5w4n4Q4VwwiN9i9ps");
211}
212
213pub mod remove_native_loader {
214    solana_pubkey::declare_id!("HTTgmruMYRZEntyL3EdCDdnS6e4D5wRq1FA7kQsb66qq");
215}
216
217pub mod send_to_tpu_vote_port {
218    solana_pubkey::declare_id!("C5fh68nJ7uyKAuYZg2x9sEQ5YrVf3dkW6oojNBSc3Jvo");
219}
220
221pub mod requestable_heap_size {
222    solana_pubkey::declare_id!("CCu4boMmfLuqcmfTLPHQiUo22ZdUsXjgzPAURYaWt1Bw");
223}
224
225pub mod disable_fee_calculator {
226    solana_pubkey::declare_id!("2jXx2yDmGysmBKfKYNgLj2DQyAQv6mMk2BPh4eSbyB4H");
227}
228
229pub mod add_compute_budget_program {
230    solana_pubkey::declare_id!("4d5AKtxoh93Dwm1vHXUU3iRATuMndx1c431KgT2td52r");
231}
232
233pub mod nonce_must_be_writable {
234    solana_pubkey::declare_id!("BiCU7M5w8ZCMykVSyhZ7Q3m2SWoR2qrEQ86ERcDX77ME");
235}
236
237pub mod spl_token_v3_3_0_release {
238    solana_pubkey::declare_id!("Ftok2jhqAqxUWEiCVRrfRs9DPppWP8cgTB7NQNKL88mS");
239}
240
241pub mod leave_nonce_on_success {
242    solana_pubkey::declare_id!("E8MkiWZNNPGU6n55jkGzyj8ghUmjCHRmDFdYYFYHxWhQ");
243}
244
245pub mod reject_empty_instruction_without_program {
246    solana_pubkey::declare_id!("9kdtFSrXHQg3hKkbXkQ6trJ3Ja1xpJ22CTFSNAciEwmL");
247}
248
249pub mod fixed_memcpy_nonoverlapping_check {
250    solana_pubkey::declare_id!("36PRUK2Dz6HWYdG9SpjeAsF5F3KxnFCakA2BZMbtMhSb");
251}
252
253pub mod reject_non_rent_exempt_vote_withdraws {
254    solana_pubkey::declare_id!("7txXZZD6Um59YoLMF7XUNimbMjsqsWhc7g2EniiTrmp1");
255}
256
257pub mod evict_invalid_stakes_cache_entries {
258    solana_pubkey::declare_id!("EMX9Q7TVFAmQ9V1CggAkhMzhXSg8ECp7fHrWQX2G1chf");
259}
260
261pub mod allow_votes_to_directly_update_vote_state {
262    solana_pubkey::declare_id!("Ff8b1fBeB86q8cjq47ZhsQLgv5EkHu3G1C99zjUfAzrq");
263}
264
265pub mod max_tx_account_locks {
266    solana_pubkey::declare_id!("CBkDroRDqm8HwHe6ak9cguPjUomrASEkfmxEaZ5CNNxz");
267}
268
269pub mod require_rent_exempt_accounts {
270    solana_pubkey::declare_id!("BkFDxiJQWZXGTZaJQxH7wVEHkAmwCgSEVkrvswFfRJPD");
271}
272
273pub mod filter_votes_outside_slot_hashes {
274    solana_pubkey::declare_id!("3gtZPqvPpsbXZVCx6hceMfWxtsmrjMzmg8C7PLKSxS2d");
275}
276
277pub mod update_syscall_base_costs {
278    solana_pubkey::declare_id!("2h63t332mGCCsWK2nqqqHhN4U9ayyqhLVFvczznHDoTZ");
279}
280
281pub mod stake_deactivate_delinquent_instruction {
282    solana_pubkey::declare_id!("437r62HoAdUb63amq3D7ENnBLDhHT2xY8eFkLJYVKK4x");
283}
284
285pub mod vote_withdraw_authority_may_change_authorized_voter {
286    solana_pubkey::declare_id!("AVZS3ZsN4gi6Rkx2QUibYuSJG3S6QHib7xCYhG6vGJxU");
287}
288
289pub mod spl_associated_token_account_v1_0_4 {
290    solana_pubkey::declare_id!("FaTa4SpiaSNH44PGC4z8bnGVTkSRYaWvrBs3KTu8XQQq");
291}
292
293pub mod reject_vote_account_close_unless_zero_credit_epoch {
294    solana_pubkey::declare_id!("ALBk3EWdeAg2WAGf6GPDUf1nynyNqCdEVmgouG7rpuCj");
295}
296
297pub mod add_get_processed_sibling_instruction_syscall {
298    solana_pubkey::declare_id!("CFK1hRCNy8JJuAAY8Pb2GjLFNdCThS2qwZNe3izzBMgn");
299}
300
301pub mod bank_transaction_count_fix {
302    solana_pubkey::declare_id!("Vo5siZ442SaZBKPXNocthiXysNviW4UYPwRFggmbgAp");
303}
304
305pub mod disable_bpf_deprecated_load_instructions {
306    solana_pubkey::declare_id!("3XgNukcZWf9o3HdA3fpJbm94XFc4qpvTXc8h1wxYwiPi");
307}
308
309pub mod disable_bpf_unresolved_symbols_at_runtime {
310    solana_pubkey::declare_id!("4yuaYAj2jGMGTh1sSmi4G2eFscsDq8qjugJXZoBN6YEa");
311}
312
313pub mod record_instruction_in_transaction_context_push {
314    solana_pubkey::declare_id!("3aJdcZqxoLpSBxgeYGjPwaYS1zzcByxUDqJkbzWAH1Zb");
315}
316
317pub mod syscall_saturated_math {
318    solana_pubkey::declare_id!("HyrbKftCdJ5CrUfEti6x26Cj7rZLNe32weugk7tLcWb8");
319}
320
321pub mod check_physical_overlapping {
322    solana_pubkey::declare_id!("nWBqjr3gpETbiaVj3CBJ3HFC5TMdnJDGt21hnvSTvVZ");
323}
324
325pub mod limit_secp256k1_recovery_id {
326    solana_pubkey::declare_id!("7g9EUwj4j7CS21Yx1wvgWLjSZeh5aPq8x9kpoPwXM8n8");
327}
328
329pub mod disable_deprecated_loader {
330    solana_pubkey::declare_id!("GTUMCZ8LTNxVfxdrw7ZsDFTxXb7TutYkzJnFwinpE6dg");
331}
332
333pub mod check_slice_translation_size {
334    solana_pubkey::declare_id!("GmC19j9qLn2RFk5NduX6QXaDhVpGncVVBzyM8e9WMz2F");
335}
336
337pub mod stake_split_uses_rent_sysvar {
338    solana_pubkey::declare_id!("FQnc7U4koHqWgRvFaBJjZnV8VPg6L6wWK33yJeDp4yvV");
339}
340
341pub mod add_get_minimum_delegation_instruction_to_stake_program {
342    solana_pubkey::declare_id!("St8k9dVXP97xT6faW24YmRSYConLbhsMJA4TJTBLmMT");
343}
344
345pub mod error_on_syscall_bpf_function_hash_collisions {
346    solana_pubkey::declare_id!("8199Q2gMD2kwgfopK5qqVWuDbegLgpuFUFHCcUJQDN8b");
347}
348
349pub mod reject_callx_r10 {
350    solana_pubkey::declare_id!("3NKRSwpySNwD3TvP5pHnRmkAQRsdkXWRr1WaQh8p4PWX");
351}
352
353pub mod drop_redundant_turbine_path {
354    solana_pubkey::declare_id!("4Di3y24QFLt5QEUPZtbnjyfQKfm6ZMTfa6Dw1psfoMKU");
355}
356
357pub mod executables_incur_cpi_data_cost {
358    solana_pubkey::declare_id!("7GUcYgq4tVtaqNCKT3dho9r4665Qp5TxCZ27Qgjx3829");
359}
360
361pub mod fix_recent_blockhashes {
362    solana_pubkey::declare_id!("6iyggb5MTcsvdcugX7bEKbHV8c6jdLbpHwkncrgLMhfo");
363}
364
365pub mod update_rewards_from_cached_accounts {
366    solana_pubkey::declare_id!("28s7i3htzhahXQKqmS2ExzbEoUypg9krwvtK2M9UWXh9");
367}
368pub mod enable_partitioned_epoch_reward {
369    solana_pubkey::declare_id!("9bn2vTJUsUcnpiZWbu2woSKtTGW3ErZC9ERv88SDqQjK");
370}
371
372pub mod partitioned_epoch_rewards_superfeature {
373    solana_pubkey::declare_id!("PERzQrt5gBD1XEe2c9XdFWqwgHY3mr7cYWbm5V772V8");
374}
375
376pub mod spl_token_v3_4_0 {
377    solana_pubkey::declare_id!("Ftok4njE8b7tDffYkC5bAbCaQv5sL6jispYrprzatUwN");
378}
379
380pub mod spl_associated_token_account_v1_1_0 {
381    solana_pubkey::declare_id!("FaTa17gVKoqbh38HcfiQonPsAaQViyDCCSg71AubYZw8");
382}
383
384pub mod default_units_per_instruction {
385    solana_pubkey::declare_id!("J2QdYx8crLbTVK8nur1jeLsmc3krDbfjoxoea2V1Uy5Q");
386}
387
388pub mod stake_allow_zero_undelegated_amount {
389    solana_pubkey::declare_id!("sTKz343FM8mqtyGvYWvbLpTThw3ixRM4Xk8QvZ985mw");
390}
391
392pub mod require_static_program_ids_in_transaction {
393    solana_pubkey::declare_id!("8FdwgyHFEjhAdjWfV2vfqk7wA1g9X3fQpKH7SBpEv3kC");
394}
395
396pub mod stake_raise_minimum_delegation_to_1_sol {
397    // This is a feature-proposal *feature id*.  The feature keypair address is `GQXzC7YiSNkje6FFUk6sc2p53XRvKoaZ9VMktYzUMnpL`.
398    solana_pubkey::declare_id!("9onWzzvCzNC2jfhxxeqRgs5q7nFAAKpCUvkj6T6GJK9i");
399}
400
401pub mod stake_minimum_delegation_for_rewards {
402    solana_pubkey::declare_id!("G6ANXD6ptCSyNd9znZm7j4dEczAJCfx7Cy43oBx3rKHJ");
403}
404
405pub mod add_set_compute_unit_price_ix {
406    solana_pubkey::declare_id!("98std1NSHqXi9WYvFShfVepRdCoq1qvsp8fsR2XZtG8g");
407}
408
409pub mod disable_deploy_of_alloc_free_syscall {
410    solana_pubkey::declare_id!("79HWsX9rpnnJBPcdNURVqygpMAfxdrAirzAGAVmf92im");
411}
412
413pub mod include_account_index_in_rent_error {
414    solana_pubkey::declare_id!("2R72wpcQ7qV7aTJWUumdn8u5wmmTyXbK7qzEy7YSAgyY");
415}
416
417pub mod add_shred_type_to_shred_seed {
418    solana_pubkey::declare_id!("Ds87KVeqhbv7Jw8W6avsS1mqz3Mw5J3pRTpPoDQ2QdiJ");
419}
420
421pub mod warp_timestamp_with_a_vengeance {
422    solana_pubkey::declare_id!("3BX6SBeEBibHaVQXywdkcgyUk6evfYZkHdztXiDtEpFS");
423}
424
425pub mod separate_nonce_from_blockhash {
426    solana_pubkey::declare_id!("Gea3ZkK2N4pHuVZVxWcnAtS6UEDdyumdYt4pFcKjA3ar");
427}
428
429pub mod enable_durable_nonce {
430    solana_pubkey::declare_id!("4EJQtF2pkRyawwcTVfQutzq4Sa5hRhibF6QAK1QXhtEX");
431}
432
433pub mod vote_state_update_credit_per_dequeue {
434    solana_pubkey::declare_id!("CveezY6FDLVBToHDcvJRmtMouqzsmj4UXYh5ths5G5Uv");
435}
436
437pub mod quick_bail_on_panic {
438    solana_pubkey::declare_id!("DpJREPyuMZ5nDfU6H3WTqSqUFSXAfw8u7xqmWtEwJDcP");
439}
440
441pub mod nonce_must_be_authorized {
442    solana_pubkey::declare_id!("HxrEu1gXuH7iD3Puua1ohd5n4iUKJyFNtNxk9DVJkvgr");
443}
444
445pub mod nonce_must_be_advanceable {
446    solana_pubkey::declare_id!("3u3Er5Vc2jVcwz4xr2GJeSAXT3fAj6ADHZ4BJMZiScFd");
447}
448
449pub mod vote_authorize_with_seed {
450    solana_pubkey::declare_id!("6tRxEYKuy2L5nnv5bgn7iT28MxUbYxp5h7F3Ncf1exrT");
451}
452
453pub mod preserve_rent_epoch_for_rent_exempt_accounts {
454    solana_pubkey::declare_id!("HH3MUYReL2BvqqA3oEcAa7txju5GY6G4nxJ51zvsEjEZ");
455}
456
457pub mod enable_bpf_loader_extend_program_ix {
458    solana_pubkey::declare_id!("8Zs9W7D9MpSEtUWSQdGniZk2cNmV22y6FLJwCx53asme");
459}
460
461pub mod enable_early_verification_of_account_modifications {
462    solana_pubkey::declare_id!("7Vced912WrRnfjaiKRiNBcbuFw7RrnLv3E3z95Y4GTNc");
463}
464
465pub mod skip_rent_rewrites {
466    solana_pubkey::declare_id!("CGB2jM8pwZkeeiXQ66kBMyBR6Np61mggL7XUsmLjVcrw");
467}
468
469pub mod prevent_crediting_accounts_that_end_rent_paying {
470    solana_pubkey::declare_id!("812kqX67odAp5NFwM8D2N24cku7WTm9CHUTFUXaDkWPn");
471}
472
473pub mod cap_bpf_program_instruction_accounts {
474    solana_pubkey::declare_id!("9k5ijzTbYPtjzu8wj2ErH9v45xecHzQ1x4PMYMMxFgdM");
475}
476
477pub mod loosen_cpi_size_restriction {
478    solana_pubkey::declare_id!("GDH5TVdbTPUpRnXaRyQqiKUa7uZAbZ28Q2N9bhbKoMLm");
479}
480
481pub mod use_default_units_in_fee_calculation {
482    solana_pubkey::declare_id!("8sKQrMQoUHtQSUP83SPG4ta2JDjSAiWs7t5aJ9uEd6To");
483}
484
485pub mod compact_vote_state_updates {
486    solana_pubkey::declare_id!("86HpNqzutEZwLcPxS6EHDcMNYWk6ikhteg9un7Y2PBKE");
487}
488
489pub mod incremental_snapshot_only_incremental_hash_calculation {
490    solana_pubkey::declare_id!("25vqsfjk7Nv1prsQJmA4Xu1bN61s8LXCBGUPp8Rfy1UF");
491}
492
493pub mod disable_cpi_setting_executable_and_rent_epoch {
494    solana_pubkey::declare_id!("B9cdB55u4jQsDNsdTK525yE9dmSc5Ga7YBaBrDFvEhM9");
495}
496
497pub mod on_load_preserve_rent_epoch_for_rent_exempt_accounts {
498    solana_pubkey::declare_id!("CpkdQmspsaZZ8FVAouQTtTWZkc8eeQ7V3uj7dWz543rZ");
499}
500
501pub mod account_hash_ignore_slot {
502    solana_pubkey::declare_id!("SVn36yVApPLYsa8koK3qUcy14zXDnqkNYWyUh1f4oK1");
503}
504
505pub mod set_exempt_rent_epoch_max {
506    solana_pubkey::declare_id!("5wAGiy15X1Jb2hkHnPDCM8oB9V42VNA9ftNVFK84dEgv");
507}
508
509pub mod relax_authority_signer_check_for_lookup_table_creation {
510    solana_pubkey::declare_id!("FKAcEvNgSY79RpqsPNUV5gDyumopH4cEHqUxyfm8b8Ap");
511}
512
513pub mod stop_sibling_instruction_search_at_parent {
514    solana_pubkey::declare_id!("EYVpEP7uzH1CoXzbD6PubGhYmnxRXPeq3PPsm1ba3gpo");
515}
516
517pub mod vote_state_update_root_fix {
518    solana_pubkey::declare_id!("G74BkWBzmsByZ1kxHy44H3wjwp5hp7JbrGRuDpco22tY");
519}
520
521pub mod cap_accounts_data_allocations_per_transaction {
522    solana_pubkey::declare_id!("9gxu85LYRAcZL38We8MYJ4A9AwgBBPtVBAqebMcT1241");
523}
524
525pub mod epoch_accounts_hash {
526    solana_pubkey::declare_id!("5GpmAKxaGsWWbPp4bNXFLJxZVvG92ctxf7jQnzTQjF3n");
527}
528
529pub mod remove_deprecated_request_unit_ix {
530    solana_pubkey::declare_id!("EfhYd3SafzGT472tYQDUc4dPd2xdEfKs5fwkowUgVt4W");
531}
532
533pub mod disable_rehash_for_rent_epoch {
534    solana_pubkey::declare_id!("DTVTkmw3JSofd8CJVJte8PXEbxNQ2yZijvVr3pe2APPj");
535}
536
537pub mod increase_tx_account_lock_limit {
538    solana_pubkey::declare_id!("9LZdXeKGeBV6hRLdxS1rHbHoEUsKqesCC2ZAPTPKJAbK");
539}
540
541pub mod limit_max_instruction_trace_length {
542    solana_pubkey::declare_id!("GQALDaC48fEhZGWRj9iL5Q889emJKcj3aCvHF7VCbbF4");
543}
544
545pub mod check_syscall_outputs_do_not_overlap {
546    solana_pubkey::declare_id!("3uRVPBpyEJRo1emLCrq38eLRFGcu6uKSpUXqGvU8T7SZ");
547}
548
549pub mod enable_bpf_loader_set_authority_checked_ix {
550    solana_pubkey::declare_id!("5x3825XS7M2A3Ekbn5VGGkvFoAg5qrRWkTrY4bARP1GL");
551}
552
553pub mod enable_alt_bn128_syscall {
554    solana_pubkey::declare_id!("A16q37opZdQMCbe5qJ6xpBB9usykfv8jZaMkxvZQi4GJ");
555}
556
557pub mod simplify_alt_bn128_syscall_error_codes {
558    solana_pubkey::declare_id!("JDn5q3GBeqzvUa7z67BbmVHVdE3EbUAjvFep3weR3jxX");
559}
560
561pub mod enable_alt_bn128_compression_syscall {
562    solana_pubkey::declare_id!("EJJewYSddEEtSZHiqugnvhQHiWyZKjkFDQASd7oKSagn");
563}
564
565pub mod enable_program_redeployment_cooldown {
566    solana_pubkey::declare_id!("J4HFT8usBxpcF63y46t1upYobJgChmKyZPm5uTBRg25Z");
567}
568
569pub mod commission_updates_only_allowed_in_first_half_of_epoch {
570    solana_pubkey::declare_id!("noRuG2kzACwgaY7TVmLRnUNPLKNVQE1fb7X55YWBehp");
571}
572
573pub mod enable_turbine_fanout_experiments {
574    solana_pubkey::declare_id!("D31EFnLgdiysi84Woo3of4JMu7VmasUS3Z7j9HYXCeLY");
575}
576
577pub mod disable_turbine_fanout_experiments {
578    solana_pubkey::declare_id!("Gz1aLrbeQ4Q6PTSafCZcGWZXz91yVRi7ASFzFEr1U4sa");
579}
580
581pub mod move_serialized_len_ptr_in_cpi {
582    solana_pubkey::declare_id!("74CoWuBmt3rUVUrCb2JiSTvh6nXyBWUsK4SaMj3CtE3T");
583}
584
585pub mod update_hashes_per_tick {
586    solana_pubkey::declare_id!("3uFHb9oKdGfgZGJK9EHaAXN4USvnQtAFC13Fh5gGFS5B");
587}
588
589pub mod enable_big_mod_exp_syscall {
590    solana_pubkey::declare_id!("EBq48m8irRKuE7ZnMTLvLg2UuGSqhe8s8oMqnmja1fJw");
591}
592
593pub mod disable_builtin_loader_ownership_chains {
594    solana_pubkey::declare_id!("4UDcAfQ6EcA6bdcadkeHpkarkhZGJ7Bpq7wTAiRMjkoi");
595}
596
597pub mod cap_transaction_accounts_data_size {
598    solana_pubkey::declare_id!("DdLwVYuvDz26JohmgSbA7mjpJFgX5zP2dkp8qsF2C33V");
599}
600
601pub mod remove_congestion_multiplier_from_fee_calculation {
602    solana_pubkey::declare_id!("A8xyMHZovGXFkorFqEmVH2PKGLiBip5JD7jt4zsUWo4H");
603}
604
605pub mod enable_request_heap_frame_ix {
606    solana_pubkey::declare_id!("Hr1nUA9b7NJ6eChS26o7Vi8gYYDDwWD3YeBfzJkTbU86");
607}
608
609pub mod prevent_rent_paying_rent_recipients {
610    solana_pubkey::declare_id!("Fab5oP3DmsLYCiQZXdjyqT3ukFFPrsmqhXU4WU1AWVVF");
611}
612
613pub mod delay_visibility_of_program_deployment {
614    solana_pubkey::declare_id!("GmuBvtFb2aHfSfMXpuFeWZGHyDeCLPS79s48fmCWCfM5");
615}
616
617pub mod apply_cost_tracker_during_replay {
618    solana_pubkey::declare_id!("2ry7ygxiYURULZCrypHhveanvP5tzZ4toRwVp89oCNSj");
619}
620pub mod bpf_account_data_direct_mapping {
621    solana_pubkey::declare_id!("AjX3A4Nv2rzUuATEUWLP4rrBaBropyUnHxEvFDj1dKbx");
622}
623
624pub mod add_set_tx_loaded_accounts_data_size_instruction {
625    solana_pubkey::declare_id!("G6vbf1UBok8MWb8m25ex86aoQHeKTzDKzuZADHkShqm6");
626}
627
628pub mod switch_to_new_elf_parser {
629    solana_pubkey::declare_id!("Cdkc8PPTeTNUPoZEfCY5AyetUrEdkZtNPMgz58nqyaHD");
630}
631
632pub mod round_up_heap_size {
633    solana_pubkey::declare_id!("CE2et8pqgyQMP2mQRg3CgvX8nJBKUArMu3wfiQiQKY1y");
634}
635
636pub mod remove_bpf_loader_incorrect_program_id {
637    solana_pubkey::declare_id!("2HmTkCj9tXuPE4ueHzdD7jPeMf9JGCoZh5AsyoATiWEe");
638}
639
640pub mod include_loaded_accounts_data_size_in_fee_calculation {
641    solana_pubkey::declare_id!("EaQpmC6GtRssaZ3PCUM5YksGqUdMLeZ46BQXYtHYakDS");
642}
643
644pub mod native_programs_consume_cu {
645    solana_pubkey::declare_id!("8pgXCMNXC8qyEFypuwpXyRxLXZdpM4Qo72gJ6k87A6wL");
646}
647
648pub mod simplify_writable_program_account_check {
649    solana_pubkey::declare_id!("5ZCcFAzJ1zsFKe1KSZa9K92jhx7gkcKj97ci2DBo1vwj");
650}
651
652pub mod stop_truncating_strings_in_syscalls {
653    solana_pubkey::declare_id!("16FMCmgLzCNNz6eTwGanbyN2ZxvTBSLuQ6DZhgeMshg");
654}
655
656pub mod clean_up_delegation_errors {
657    solana_pubkey::declare_id!("Bj2jmUsM2iRhfdLLDSTkhM5UQRQvQHm57HSmPibPtEyu");
658}
659
660pub mod vote_state_add_vote_latency {
661    solana_pubkey::declare_id!("7axKe5BTYBDD87ftzWbk5DfzWMGyRvqmWTduuo22Yaqy");
662}
663
664pub mod checked_arithmetic_in_fee_validation {
665    solana_pubkey::declare_id!("5Pecy6ie6XGm22pc9d4P9W5c31BugcFBuy6hsP2zkETv");
666}
667
668pub mod last_restart_slot_sysvar {
669    solana_pubkey::declare_id!("HooKD5NC9QNxk25QuzCssB8ecrEzGt6eXEPBUxWp1LaR");
670}
671
672pub mod reduce_stake_warmup_cooldown {
673    solana_pubkey::declare_id!("GwtDQBghCTBgmX2cpEGNPxTEBUTQRaDMGTr5qychdGMj");
674}
675
676mod revise_turbine_epoch_stakes {
677    solana_pubkey::declare_id!("BTWmtJC8U5ZLMbBUUA1k6As62sYjPEjAiNAT55xYGdJU");
678}
679
680pub mod enable_poseidon_syscall {
681    solana_pubkey::declare_id!("FL9RsQA6TVUoh5xJQ9d936RHSebA1NLQqe3Zv9sXZRpr");
682}
683
684pub mod timely_vote_credits {
685    solana_pubkey::declare_id!("tvcF6b1TRz353zKuhBjinZkKzjmihXmBAHJdjNYw1sQ");
686}
687
688pub mod remaining_compute_units_syscall_enabled {
689    solana_pubkey::declare_id!("5TuppMutoyzhUSfuYdhgzD47F92GL1g89KpCZQKqedxP");
690}
691
692pub mod enable_program_runtime_v2_and_loader_v4 {
693    solana_pubkey::declare_id!("8oBxsYqnCvUTGzgEpxPcnVf7MLbWWPYddE33PftFeBBd");
694}
695
696pub mod require_rent_exempt_split_destination {
697    solana_pubkey::declare_id!("D2aip4BBr8NPWtU9vLrwrBvbuaQ8w1zV38zFLxx4pfBV");
698}
699
700pub mod better_error_codes_for_tx_lamport_check {
701    solana_pubkey::declare_id!("Ffswd3egL3tccB6Rv3XY6oqfdzn913vUcjCSnpvCKpfx");
702}
703
704pub mod update_hashes_per_tick2 {
705    solana_pubkey::declare_id!("EWme9uFqfy1ikK1jhJs8fM5hxWnK336QJpbscNtizkTU");
706}
707
708pub mod update_hashes_per_tick3 {
709    solana_pubkey::declare_id!("8C8MCtsab5SsfammbzvYz65HHauuUYdbY2DZ4sznH6h5");
710}
711
712pub mod update_hashes_per_tick4 {
713    solana_pubkey::declare_id!("8We4E7DPwF2WfAN8tRTtWQNhi98B99Qpuj7JoZ3Aikgg");
714}
715
716pub mod update_hashes_per_tick5 {
717    solana_pubkey::declare_id!("BsKLKAn1WM4HVhPRDsjosmqSg2J8Tq5xP2s2daDS6Ni4");
718}
719
720pub mod update_hashes_per_tick6 {
721    solana_pubkey::declare_id!("FKu1qYwLQSiehz644H6Si65U5ZQ2cp9GxsyFUfYcuADv");
722}
723
724pub mod validate_fee_collector_account {
725    solana_pubkey::declare_id!("prpFrMtgNmzaNzkPJg9o753fVvbHKqNrNTm76foJ2wm");
726}
727
728pub mod disable_rent_fees_collection {
729    solana_pubkey::declare_id!("CJzY83ggJHqPGDq8VisV3U91jDJLuEaALZooBrXtnnLU");
730}
731
732pub mod enable_zk_transfer_with_fee {
733    solana_pubkey::declare_id!("zkNLP7EQALfC1TYeB3biDU7akDckj8iPkvh9y2Mt2K3");
734}
735
736pub mod drop_legacy_shreds {
737    solana_pubkey::declare_id!("GV49KKQdBNaiv2pgqhS2Dy3GWYJGXMTVYbYkdk91orRy");
738}
739
740pub mod allow_commission_decrease_at_any_time {
741    solana_pubkey::declare_id!("decoMktMcnmiq6t3u7g5BfgcQu91nKZr6RvMYf9z1Jb");
742}
743
744pub mod add_new_reserved_account_keys {
745    solana_pubkey::declare_id!("8U4skmMVnF6k2kMvrWbQuRUT3qQSiTYpSjqmhmgfthZu");
746}
747
748pub mod consume_blockstore_duplicate_proofs {
749    solana_pubkey::declare_id!("6YsBCejwK96GZCkJ6mkZ4b68oP63z2PLoQmWjC7ggTqZ");
750}
751
752pub mod index_erasure_conflict_duplicate_proofs {
753    solana_pubkey::declare_id!("dupPajaLy2SSn8ko42aZz4mHANDNrLe8Nw8VQgFecLa");
754}
755
756pub mod merkle_conflict_duplicate_proofs {
757    solana_pubkey::declare_id!("mrkPjRg79B2oK2ZLgd7S3AfEJaX9B6gAF3H9aEykRUS");
758}
759
760pub mod disable_bpf_loader_instructions {
761    solana_pubkey::declare_id!("7WeS1vfPRgeeoXArLh7879YcB9mgE9ktjPDtajXeWfXn");
762}
763
764pub mod enable_zk_proof_from_account {
765    solana_pubkey::declare_id!("zkiTNuzBKxrCLMKehzuQeKZyLtX2yvFcEKMML8nExU8");
766}
767
768pub mod cost_model_requested_write_lock_cost {
769    solana_pubkey::declare_id!("wLckV1a64ngtcKPRGU4S4grVTestXjmNjxBjaKZrAcn");
770}
771
772pub mod enable_gossip_duplicate_proof_ingestion {
773    solana_pubkey::declare_id!("FNKCMBzYUdjhHyPdsKG2LSmdzH8TCHXn3ytj8RNBS4nG");
774}
775
776pub mod chained_merkle_conflict_duplicate_proofs {
777    solana_pubkey::declare_id!("chaie9S2zVfuxJKNRGkyTDokLwWxx6kD2ZLsqQHaDD8");
778}
779
780pub mod enable_chained_merkle_shreds {
781    solana_pubkey::declare_id!("7uZBkJXJ1HkuP6R3MJfZs7mLwymBcDbKdqbF51ZWLier");
782}
783
784pub mod remove_rounding_in_fee_calculation {
785    solana_pubkey::declare_id!("BtVN7YjDzNE6Dk7kTT7YTDgMNUZTNgiSJgsdzAeTg2jF");
786}
787
788pub mod enable_tower_sync_ix {
789    solana_pubkey::declare_id!("tSynMCspg4xFiCj1v3TDb4c7crMR5tSBhLz4sF7rrNA");
790}
791
792pub mod deprecate_unused_legacy_vote_plumbing {
793    solana_pubkey::declare_id!("6Uf8S75PVh91MYgPQSHnjRAPQq6an5BDv9vomrCwDqLe");
794}
795
796pub mod reward_full_priority_fee {
797    solana_pubkey::declare_id!("3opE3EzAKnUftUDURkzMgwpNgimBAypW1mNDYH4x4Zg7");
798}
799
800pub mod get_sysvar_syscall_enabled {
801    solana_pubkey::declare_id!("CLCoTADvV64PSrnR6QXty6Fwrt9Xc6EdxSJE4wLRePjq");
802}
803
804pub mod abort_on_invalid_curve {
805    solana_pubkey::declare_id!("FuS3FPfJDKSNot99ECLXtp3rueq36hMNStJkPJwWodLh");
806}
807
808pub mod migrate_feature_gate_program_to_core_bpf {
809    solana_pubkey::declare_id!("4eohviozzEeivk1y9UbrnekbAFMDQyJz5JjA9Y6gyvky");
810}
811
812pub mod vote_only_full_fec_sets {
813    solana_pubkey::declare_id!("ffecLRhhakKSGhMuc6Fz2Lnfq4uT9q3iu9ZsNaPLxPc");
814}
815
816pub mod migrate_config_program_to_core_bpf {
817    solana_pubkey::declare_id!("2Fr57nzzkLYXW695UdDxDeR5fhnZWSttZeZYemrnpGFV");
818}
819
820pub mod enable_get_epoch_stake_syscall {
821    solana_pubkey::declare_id!("FKe75t4LXxGaQnVHdUKM6DSFifVVraGZ8LyNo7oPwy1Z");
822}
823
824pub mod migrate_address_lookup_table_program_to_core_bpf {
825    solana_pubkey::declare_id!("C97eKZygrkU4JxJsZdjgbUY7iQR7rKTr4NyDWo2E5pRm");
826}
827
828pub mod zk_elgamal_proof_program_enabled {
829    solana_pubkey::declare_id!("zkhiy5oLowR7HY4zogXjCjeMXyruLqBwSWH21qcFtnv");
830}
831
832pub mod verify_retransmitter_signature {
833    solana_pubkey::declare_id!("BZ5g4hRbu5hLQQBdPyo2z9icGyJ8Khiyj3QS6dhWijTb");
834}
835
836pub mod move_stake_and_move_lamports_ixs {
837    solana_pubkey::declare_id!("7bTK6Jis8Xpfrs8ZoUfiMDPazTcdPcTWheZFJTA5Z6X4");
838}
839
840pub mod ed25519_precompile_verify_strict {
841    solana_pubkey::declare_id!("ed9tNscbWLYBooxWA7FE2B5KHWs8A6sxfY8EzezEcoo");
842}
843
844pub mod vote_only_retransmitter_signed_fec_sets {
845    solana_pubkey::declare_id!("RfEcA95xnhuwooVAhUUksEJLZBF7xKCLuqrJoqk4Zph");
846}
847
848pub mod move_precompile_verification_to_svm {
849    solana_pubkey::declare_id!("9ypxGLzkMxi89eDerRKXWDXe44UY2z4hBig4mDhNq5Dp");
850}
851
852pub mod enable_transaction_loading_failure_fees {
853    solana_pubkey::declare_id!("PaymEPK2oqwT9TXAVfadjztH2H6KfLEB9Hhd5Q5frvP");
854}
855
856pub mod enable_turbine_extended_fanout_experiments {
857    solana_pubkey::declare_id!("BZn14Liea52wtBwrXUxTv6vojuTTmfc7XGEDTXrvMD7b");
858}
859
860pub mod deprecate_legacy_vote_ixs {
861    solana_pubkey::declare_id!("depVvnQ2UysGrhwdiwU42tCadZL8GcBb1i2GYhMopQv");
862}
863
864pub mod disable_sbpf_v1_execution {
865    solana_pubkey::declare_id!("TestFeature11111111111111111111111111111111");
866}
867
868pub mod reenable_sbpf_v1_execution {
869    solana_pubkey::declare_id!("TestFeature21111111111111111111111111111111");
870}
871
872pub mod disable_account_loader_special_case {
873    solana_pubkey::declare_id!("EQUMpNFr7Nacb1sva56xn1aLfBxppEoSBH8RRVdkcD1x");
874}
875
876pub mod enable_secp256r1_precompile {
877    solana_pubkey::declare_id!("sr11RdZWgbHTHxSroPALe6zgaT5A1K9LcE4nfsZS4gi");
878}
879
880pub mod migrate_stake_program_to_core_bpf {
881    solana_pubkey::declare_id!("6M4oQ6eXneVhtLoiAr4yRYQY43eVLjrKbiDZDJc892yk");
882}
883
884pub mod deplete_cu_meter_on_vm_failure {
885    solana_pubkey::declare_id!("B7H2caeia4ZFcpE3QcgMqbiWiBtWrdBRBSJ1DY6Ktxbq");
886}
887
888pub mod raise_block_limits_to_50m {
889    solana_pubkey::declare_id!("5oMCU3JPaFLr8Zr4ct7yFA7jdk6Mw1RmB8K4u9ZbS42z");
890}
891
892pub mod reserve_minimal_cus_for_builtin_instructions {
893    solana_pubkey::declare_id!("C9oAhLxDBm3ssWtJx1yBGzPY55r2rArHmN1pbQn6HogH");
894}
895
896lazy_static! {
897    /// Map of feature identifiers to user-visible description
898    pub static ref FEATURE_NAMES: HashMap<Pubkey, &'static str> = [
899        (secp256k1_program_enabled::id(), "secp256k1 program"),
900        (deprecate_rewards_sysvar::id(), "deprecate unused rewards sysvar"),
901        (pico_inflation::id(), "pico inflation"),
902        (full_inflation::devnet_and_testnet::id(), "full inflation on devnet and testnet"),
903        (spl_token_v2_multisig_fix::id(), "spl-token multisig fix"),
904        (no_overflow_rent_distribution::id(), "no overflow rent distribution"),
905        (filter_stake_delegation_accounts::id(), "filter stake_delegation_accounts #14062"),
906        (require_custodian_for_locked_stake_authorize::id(), "require custodian to authorize withdrawer change for locked stake"),
907        (spl_token_v2_self_transfer_fix::id(), "spl-token self-transfer fix"),
908        (full_inflation::mainnet::certusone::enable::id(), "full inflation enabled by Certus One"),
909        (full_inflation::mainnet::certusone::vote::id(), "community vote allowing Certus One to enable full inflation"),
910        (warp_timestamp_again::id(), "warp timestamp again, adjust bounding to 25% fast 80% slow #15204"),
911        (check_init_vote_data::id(), "check initialized Vote data"),
912        (secp256k1_recover_syscall_enabled::id(), "secp256k1_recover syscall"),
913        (system_transfer_zero_check::id(), "perform all checks for transfers of 0 lamports"),
914        (blake3_syscall_enabled::id(), "blake3 syscall"),
915        (dedupe_config_program_signers::id(), "dedupe config program signers"),
916        (verify_tx_signatures_len::id(), "prohibit extra transaction signatures"),
917        (vote_stake_checked_instructions::id(), "vote/state program checked instructions #18345"),
918        (rent_for_sysvars::id(), "collect rent from accounts owned by sysvars"),
919        (libsecp256k1_0_5_upgrade_enabled::id(), "upgrade libsecp256k1 to v0.5.0"),
920        (tx_wide_compute_cap::id(), "transaction wide compute cap"),
921        (spl_token_v2_set_authority_fix::id(), "spl-token set_authority fix"),
922        (merge_nonce_error_into_system_error::id(), "merge NonceError into SystemError"),
923        (disable_fees_sysvar::id(), "disable fees sysvar"),
924        (stake_merge_with_unmatched_credits_observed::id(), "allow merging active stakes with unmatched credits_observed #18985"),
925        (zk_token_sdk_enabled::id(), "enable Zk Token proof program and syscalls"),
926        (curve25519_syscall_enabled::id(), "enable curve25519 syscalls"),
927        (versioned_tx_message_enabled::id(), "enable versioned transaction message processing"),
928        (libsecp256k1_fail_on_bad_count::id(), "fail libsecp256k1_verify if count appears wrong"),
929        (libsecp256k1_fail_on_bad_count2::id(), "fail libsecp256k1_verify if count appears wrong"),
930        (instructions_sysvar_owned_by_sysvar::id(), "fix owner for instructions sysvar"),
931        (stake_program_advance_activating_credits_observed::id(), "Enable advancing credits observed for activation epoch #19309"),
932        (credits_auto_rewind::id(), "Auto rewind stake's credits_observed if (accidental) vote recreation is detected #22546"),
933        (demote_program_write_locks::id(), "demote program write locks to readonly, except when upgradeable loader present #19593 #20265"),
934        (ed25519_program_enabled::id(), "enable builtin ed25519 signature verify program"),
935        (return_data_syscall_enabled::id(), "enable sol_{set,get}_return_data syscall"),
936        (reduce_required_deploy_balance::id(), "reduce required payer balance for program deploys"),
937        (sol_log_data_syscall_enabled::id(), "enable sol_log_data syscall"),
938        (stakes_remove_delegation_if_inactive::id(), "remove delegations from stakes cache when inactive"),
939        (do_support_realloc::id(), "support account data reallocation"),
940        (prevent_calling_precompiles_as_programs::id(), "prevent calling precompiles as programs"),
941        (optimize_epoch_boundary_updates::id(), "optimize epoch boundary updates"),
942        (remove_native_loader::id(), "remove support for the native loader"),
943        (send_to_tpu_vote_port::id(), "send votes to the tpu vote port"),
944        (requestable_heap_size::id(), "Requestable heap frame size"),
945        (disable_fee_calculator::id(), "deprecate fee calculator"),
946        (add_compute_budget_program::id(), "Add compute_budget_program"),
947        (nonce_must_be_writable::id(), "nonce must be writable"),
948        (spl_token_v3_3_0_release::id(), "spl-token v3.3.0 release"),
949        (leave_nonce_on_success::id(), "leave nonce as is on success"),
950        (reject_empty_instruction_without_program::id(), "fail instructions which have native_loader as program_id directly"),
951        (fixed_memcpy_nonoverlapping_check::id(), "use correct check for nonoverlapping regions in memcpy syscall"),
952        (reject_non_rent_exempt_vote_withdraws::id(), "fail vote withdraw instructions which leave the account non-rent-exempt"),
953        (evict_invalid_stakes_cache_entries::id(), "evict invalid stakes cache entries on epoch boundaries"),
954        (allow_votes_to_directly_update_vote_state::id(), "enable direct vote state update"),
955        (max_tx_account_locks::id(), "enforce max number of locked accounts per transaction"),
956        (require_rent_exempt_accounts::id(), "require all new transaction accounts with data to be rent-exempt"),
957        (filter_votes_outside_slot_hashes::id(), "filter vote slots older than the slot hashes history"),
958        (update_syscall_base_costs::id(), "update syscall base costs"),
959        (stake_deactivate_delinquent_instruction::id(), "enable the deactivate delinquent stake instruction #23932"),
960        (vote_withdraw_authority_may_change_authorized_voter::id(), "vote account withdraw authority may change the authorized voter #22521"),
961        (spl_associated_token_account_v1_0_4::id(), "SPL Associated Token Account Program release version 1.0.4, tied to token 3.3.0 #22648"),
962        (reject_vote_account_close_unless_zero_credit_epoch::id(), "fail vote account withdraw to 0 unless account earned 0 credits in last completed epoch"),
963        (add_get_processed_sibling_instruction_syscall::id(), "add add_get_processed_sibling_instruction_syscall"),
964        (bank_transaction_count_fix::id(), "fixes Bank::transaction_count to include all committed transactions, not just successful ones"),
965        (disable_bpf_deprecated_load_instructions::id(), "disable ldabs* and ldind* SBF instructions"),
966        (disable_bpf_unresolved_symbols_at_runtime::id(), "disable reporting of unresolved SBF symbols at runtime"),
967        (record_instruction_in_transaction_context_push::id(), "move the CPI stack overflow check to the end of push"),
968        (syscall_saturated_math::id(), "syscalls use saturated math"),
969        (check_physical_overlapping::id(), "check physical overlapping regions"),
970        (limit_secp256k1_recovery_id::id(), "limit secp256k1 recovery id"),
971        (disable_deprecated_loader::id(), "disable the deprecated BPF loader"),
972        (check_slice_translation_size::id(), "check size when translating slices"),
973        (stake_split_uses_rent_sysvar::id(), "stake split instruction uses rent sysvar"),
974        (add_get_minimum_delegation_instruction_to_stake_program::id(), "add GetMinimumDelegation instruction to stake program"),
975        (error_on_syscall_bpf_function_hash_collisions::id(), "error on bpf function hash collisions"),
976        (reject_callx_r10::id(), "Reject bpf callx r10 instructions"),
977        (drop_redundant_turbine_path::id(), "drop redundant turbine path"),
978        (executables_incur_cpi_data_cost::id(), "Executables incur CPI data costs"),
979        (fix_recent_blockhashes::id(), "stop adding hashes for skipped slots to recent blockhashes"),
980        (update_rewards_from_cached_accounts::id(), "update rewards from cached accounts"),
981        (enable_partitioned_epoch_reward::id(), "enable partitioned rewards at epoch boundary #32166"),
982        (spl_token_v3_4_0::id(), "SPL Token Program version 3.4.0 release #24740"),
983        (spl_associated_token_account_v1_1_0::id(), "SPL Associated Token Account Program version 1.1.0 release #24741"),
984        (default_units_per_instruction::id(), "Default max tx-wide compute units calculated per instruction"),
985        (stake_allow_zero_undelegated_amount::id(), "Allow zero-lamport undelegated amount for initialized stakes #24670"),
986        (require_static_program_ids_in_transaction::id(), "require static program ids in versioned transactions"),
987        (stake_raise_minimum_delegation_to_1_sol::id(), "Raise minimum stake delegation to 1.0 SOL #24357"),
988        (stake_minimum_delegation_for_rewards::id(), "stakes must be at least the minimum delegation to earn rewards"),
989        (add_set_compute_unit_price_ix::id(), "add compute budget ix for setting a compute unit price"),
990        (disable_deploy_of_alloc_free_syscall::id(), "disable new deployments of deprecated sol_alloc_free_ syscall"),
991        (include_account_index_in_rent_error::id(), "include account index in rent tx error #25190"),
992        (add_shred_type_to_shred_seed::id(), "add shred-type to shred seed #25556"),
993        (warp_timestamp_with_a_vengeance::id(), "warp timestamp again, adjust bounding to 150% slow #25666"),
994        (separate_nonce_from_blockhash::id(), "separate durable nonce and blockhash domains #25744"),
995        (enable_durable_nonce::id(), "enable durable nonce #25744"),
996        (vote_state_update_credit_per_dequeue::id(), "Calculate vote credits for VoteStateUpdate per vote dequeue to match credit awards for Vote instruction"),
997        (quick_bail_on_panic::id(), "quick bail on panic"),
998        (nonce_must_be_authorized::id(), "nonce must be authorized"),
999        (nonce_must_be_advanceable::id(), "durable nonces must be advanceable"),
1000        (vote_authorize_with_seed::id(), "An instruction you can use to change a vote accounts authority when the current authority is a derived key #25860"),
1001        (preserve_rent_epoch_for_rent_exempt_accounts::id(), "preserve rent epoch for rent exempt accounts #26479"),
1002        (enable_bpf_loader_extend_program_ix::id(), "enable bpf upgradeable loader ExtendProgram instruction #25234"),
1003        (skip_rent_rewrites::id(), "skip rewriting rent exempt accounts during rent collection #26491"),
1004        (enable_early_verification_of_account_modifications::id(), "enable early verification of account modifications #25899"),
1005        (disable_rehash_for_rent_epoch::id(), "on accounts hash calculation, do not try to rehash accounts #28934"),
1006        (account_hash_ignore_slot::id(), "ignore slot when calculating an account hash #28420"),
1007        (set_exempt_rent_epoch_max::id(), "set rent epoch to Epoch::MAX for rent-exempt accounts #28683"),
1008        (on_load_preserve_rent_epoch_for_rent_exempt_accounts::id(), "on bank load account, do not try to fix up rent_epoch #28541"),
1009        (prevent_crediting_accounts_that_end_rent_paying::id(), "prevent crediting rent paying accounts #26606"),
1010        (cap_bpf_program_instruction_accounts::id(), "enforce max number of accounts per bpf program instruction #26628"),
1011        (loosen_cpi_size_restriction::id(), "loosen cpi size restrictions #26641"),
1012        (use_default_units_in_fee_calculation::id(), "use default units per instruction in fee calculation #26785"),
1013        (compact_vote_state_updates::id(), "Compact vote state updates to lower block size"),
1014        (incremental_snapshot_only_incremental_hash_calculation::id(), "only hash accounts in incremental snapshot during incremental snapshot creation #26799"),
1015        (disable_cpi_setting_executable_and_rent_epoch::id(), "disable setting is_executable and_rent_epoch in CPI #26987"),
1016        (relax_authority_signer_check_for_lookup_table_creation::id(), "relax authority signer check for lookup table creation #27205"),
1017        (stop_sibling_instruction_search_at_parent::id(), "stop the search in get_processed_sibling_instruction when the parent instruction is reached #27289"),
1018        (vote_state_update_root_fix::id(), "fix root in vote state updates #27361"),
1019        (cap_accounts_data_allocations_per_transaction::id(), "cap accounts data allocations per transaction #27375"),
1020        (epoch_accounts_hash::id(), "enable epoch accounts hash calculation #27539"),
1021        (remove_deprecated_request_unit_ix::id(), "remove support for RequestUnitsDeprecated instruction #27500"),
1022        (increase_tx_account_lock_limit::id(), "increase tx account lock limit to 128 #27241"),
1023        (limit_max_instruction_trace_length::id(), "limit max instruction trace length #27939"),
1024        (check_syscall_outputs_do_not_overlap::id(), "check syscall outputs do_not overlap #28600"),
1025        (enable_bpf_loader_set_authority_checked_ix::id(), "enable bpf upgradeable loader SetAuthorityChecked instruction #28424"),
1026        (enable_alt_bn128_syscall::id(), "add alt_bn128 syscalls #27961"),
1027        (simplify_alt_bn128_syscall_error_codes::id(), "simplify alt_bn128 syscall error codes SIMD-0129"),
1028        (enable_program_redeployment_cooldown::id(), "enable program redeployment cooldown #29135"),
1029        (commission_updates_only_allowed_in_first_half_of_epoch::id(), "validator commission updates are only allowed in the first half of an epoch #29362"),
1030        (enable_turbine_fanout_experiments::id(), "enable turbine fanout experiments #29393"),
1031        (disable_turbine_fanout_experiments::id(), "disable turbine fanout experiments #29393"),
1032        (move_serialized_len_ptr_in_cpi::id(), "cpi ignore serialized_len_ptr #29592"),
1033        (update_hashes_per_tick::id(), "Update desired hashes per tick on epoch boundary"),
1034        (enable_big_mod_exp_syscall::id(), "add big_mod_exp syscall #28503"),
1035        (disable_builtin_loader_ownership_chains::id(), "disable builtin loader ownership chains #29956"),
1036        (cap_transaction_accounts_data_size::id(), "cap transaction accounts data size up to a limit #27839"),
1037        (remove_congestion_multiplier_from_fee_calculation::id(), "Remove congestion multiplier from transaction fee calculation #29881"),
1038        (enable_request_heap_frame_ix::id(), "Enable transaction to request heap frame using compute budget instruction #30076"),
1039        (prevent_rent_paying_rent_recipients::id(), "prevent recipients of rent rewards from ending in rent-paying state #30151"),
1040        (delay_visibility_of_program_deployment::id(), "delay visibility of program upgrades #30085"),
1041        (apply_cost_tracker_during_replay::id(), "apply cost tracker to blocks during replay #29595"),
1042        (add_set_tx_loaded_accounts_data_size_instruction::id(), "add compute budget instruction for setting account data size per transaction #30366"),
1043        (switch_to_new_elf_parser::id(), "switch to new ELF parser #30497"),
1044        (round_up_heap_size::id(), "round up heap size when calculating heap cost #30679"),
1045        (remove_bpf_loader_incorrect_program_id::id(), "stop incorrectly throwing IncorrectProgramId in bpf_loader #30747"),
1046        (include_loaded_accounts_data_size_in_fee_calculation::id(), "include transaction loaded accounts data size in base fee calculation #30657"),
1047        (native_programs_consume_cu::id(), "Native program should consume compute units #30620"),
1048        (simplify_writable_program_account_check::id(), "Simplify checks performed for writable upgradeable program accounts #30559"),
1049        (stop_truncating_strings_in_syscalls::id(), "Stop truncating strings in syscalls #31029"),
1050        (clean_up_delegation_errors::id(), "Return InsufficientDelegation instead of InsufficientFunds or InsufficientStake where applicable #31206"),
1051        (vote_state_add_vote_latency::id(), "replace Lockout with LandedVote (including vote latency) in vote state #31264"),
1052        (checked_arithmetic_in_fee_validation::id(), "checked arithmetic in fee validation #31273"),
1053        (bpf_account_data_direct_mapping::id(), "use memory regions to map account data into the rbpf vm instead of copying the data"),
1054        (last_restart_slot_sysvar::id(), "enable new sysvar last_restart_slot"),
1055        (reduce_stake_warmup_cooldown::id(), "reduce stake warmup cooldown from 25% to 9%"),
1056        (revise_turbine_epoch_stakes::id(), "revise turbine epoch stakes"),
1057        (enable_poseidon_syscall::id(), "Enable Poseidon syscall"),
1058        (timely_vote_credits::id(), "use timeliness of votes in determining credits to award"),
1059        (remaining_compute_units_syscall_enabled::id(), "enable the remaining_compute_units syscall"),
1060        (enable_program_runtime_v2_and_loader_v4::id(), "Enable Program-Runtime-v2 and Loader-v4 #33293"),
1061        (require_rent_exempt_split_destination::id(), "Require stake split destination account to be rent exempt"),
1062        (better_error_codes_for_tx_lamport_check::id(), "better error codes for tx lamport check #33353"),
1063        (enable_alt_bn128_compression_syscall::id(), "add alt_bn128 compression syscalls"),
1064        (update_hashes_per_tick2::id(), "Update desired hashes per tick to 2.8M"),
1065        (update_hashes_per_tick3::id(), "Update desired hashes per tick to 4.4M"),
1066        (update_hashes_per_tick4::id(), "Update desired hashes per tick to 7.6M"),
1067        (update_hashes_per_tick5::id(), "Update desired hashes per tick to 9.2M"),
1068        (update_hashes_per_tick6::id(), "Update desired hashes per tick to 10M"),
1069        (validate_fee_collector_account::id(), "validate fee collector account #33888"),
1070        (disable_rent_fees_collection::id(), "Disable rent fees collection #33945"),
1071        (enable_zk_transfer_with_fee::id(), "enable Zk Token proof program transfer with fee"),
1072        (drop_legacy_shreds::id(), "drops legacy shreds #34328"),
1073        (allow_commission_decrease_at_any_time::id(), "Allow commission decrease at any time in epoch #33843"),
1074        (consume_blockstore_duplicate_proofs::id(), "consume duplicate proofs from blockstore in consensus #34372"),
1075        (add_new_reserved_account_keys::id(), "add new unwritable reserved accounts #34899"),
1076        (index_erasure_conflict_duplicate_proofs::id(), "generate duplicate proofs for index and erasure conflicts #34360"),
1077        (merkle_conflict_duplicate_proofs::id(), "generate duplicate proofs for merkle root conflicts #34270"),
1078        (disable_bpf_loader_instructions::id(), "disable bpf loader management instructions #34194"),
1079        (enable_zk_proof_from_account::id(), "Enable zk token proof program to read proof from accounts instead of instruction data #34750"),
1080        (curve25519_restrict_msm_length::id(), "restrict curve25519 multiscalar multiplication vector lengths #34763"),
1081        (cost_model_requested_write_lock_cost::id(), "cost model uses number of requested write locks #34819"),
1082        (enable_gossip_duplicate_proof_ingestion::id(), "enable gossip duplicate proof ingestion #32963"),
1083        (enable_chained_merkle_shreds::id(), "Enable chained Merkle shreds #34916"),
1084        (remove_rounding_in_fee_calculation::id(), "Removing unwanted rounding in fee calculation #34982"),
1085        (deprecate_unused_legacy_vote_plumbing::id(), "Deprecate unused legacy vote tx plumbing"),
1086        (enable_tower_sync_ix::id(), "Enable tower sync vote instruction"),
1087        (chained_merkle_conflict_duplicate_proofs::id(), "generate duplicate proofs for chained merkle root conflicts"),
1088        (reward_full_priority_fee::id(), "Reward full priority fee to validators #34731"),
1089        (abort_on_invalid_curve::id(), "Abort when elliptic curve syscalls invoked on invalid curve id SIMD-0137"),
1090        (get_sysvar_syscall_enabled::id(), "Enable syscall for fetching Sysvar bytes #615"),
1091        (migrate_feature_gate_program_to_core_bpf::id(), "Migrate Feature Gate program to Core BPF (programify) #1003"),
1092        (vote_only_full_fec_sets::id(), "vote only full fec sets"),
1093        (migrate_config_program_to_core_bpf::id(), "Migrate Config program to Core BPF #1378"),
1094        (enable_get_epoch_stake_syscall::id(), "Enable syscall: sol_get_epoch_stake #884"),
1095        (migrate_address_lookup_table_program_to_core_bpf::id(), "Migrate Address Lookup Table program to Core BPF #1651"),
1096        (zk_elgamal_proof_program_enabled::id(), "Enable ZkElGamalProof program SIMD-0153"),
1097        (verify_retransmitter_signature::id(), "Verify retransmitter signature #1840"),
1098        (move_stake_and_move_lamports_ixs::id(), "Enable MoveStake and MoveLamports stake program instructions #1610"),
1099        (ed25519_precompile_verify_strict::id(), "Use strict verification in ed25519 precompile SIMD-0152"),
1100        (vote_only_retransmitter_signed_fec_sets::id(), "vote only on retransmitter signed fec sets"),
1101        (move_precompile_verification_to_svm::id(), "SIMD-0159: Move precompile verification into SVM"),
1102        (enable_transaction_loading_failure_fees::id(), "Enable fees for some additional transaction failures SIMD-0082"),
1103        (enable_turbine_extended_fanout_experiments::id(), "enable turbine extended fanout experiments #"),
1104        (deprecate_legacy_vote_ixs::id(), "Deprecate legacy vote instructions"),
1105        (partitioned_epoch_rewards_superfeature::id(), "replaces enable_partitioned_epoch_reward to enable partitioned rewards at epoch boundary SIMD-0118"),
1106        (disable_sbpf_v1_execution::id(), "Disables execution of SBPFv1 programs"),
1107        (reenable_sbpf_v1_execution::id(), "Re-enables execution of SBPFv1 programs"),
1108        (disable_account_loader_special_case::id(), "Disable account loader special case #3513"),
1109        (enable_secp256r1_precompile::id(), "Enable secp256r1 precompile SIMD-0075"),
1110        (migrate_stake_program_to_core_bpf::id(), "Migrate Stake program to Core BPF SIMD-0196 #3655"),
1111        (deplete_cu_meter_on_vm_failure::id(), "Deplete compute meter for vm errors SIMD-0182 #3993"),
1112        (raise_block_limits_to_50m::id(), "Raise block limit to 50M SIMD-0207"),
1113        (reserve_minimal_cus_for_builtin_instructions::id(), "Reserve minimal CUs for builtin instructions SIMD-170 #2562"),
1114        /*************** ADD NEW FEATURES HERE ***************/
1115    ]
1116    .iter()
1117    .cloned()
1118    .collect();
1119
1120    /// Unique identifier of the current software's feature set
1121    pub static ref ID: Hash = {
1122        let mut hasher = Hasher::default();
1123        let mut feature_ids = FEATURE_NAMES.keys().collect::<Vec<_>>();
1124        feature_ids.sort();
1125        for feature in feature_ids {
1126            hasher.hash(feature.as_ref());
1127        }
1128        hasher.result()
1129    };
1130}
1131
1132#[derive(Clone, PartialEq, Eq, Hash)]
1133pub struct FullInflationFeaturePair {
1134    pub vote_id: Pubkey, // Feature that grants the candidate the ability to enable full inflation
1135    pub enable_id: Pubkey, // Feature to enable full inflation by the candidate
1136}
1137
1138lazy_static! {
1139    /// Set of feature pairs that once enabled will trigger full inflation
1140    pub static ref FULL_INFLATION_FEATURE_PAIRS: HashSet<FullInflationFeaturePair> = [
1141        FullInflationFeaturePair {
1142            vote_id: full_inflation::mainnet::certusone::vote::id(),
1143            enable_id: full_inflation::mainnet::certusone::enable::id(),
1144        },
1145    ]
1146    .iter()
1147    .cloned()
1148    .collect();
1149}
1150
1151/// `FeatureSet` holds the set of currently active/inactive runtime features
1152#[cfg_attr(feature = "frozen-abi", derive(solana_frozen_abi_macro::AbiExample))]
1153#[derive(Debug, Clone, Eq, PartialEq)]
1154pub struct FeatureSet {
1155    pub active: HashMap<Pubkey, Slot>,
1156    pub inactive: HashSet<Pubkey>,
1157}
1158impl Default for FeatureSet {
1159    fn default() -> Self {
1160        // All features disabled
1161        Self {
1162            active: HashMap::new(),
1163            inactive: FEATURE_NAMES.keys().cloned().collect(),
1164        }
1165    }
1166}
1167impl FeatureSet {
1168    pub fn is_active(&self, feature_id: &Pubkey) -> bool {
1169        self.active.contains_key(feature_id)
1170    }
1171
1172    pub fn activated_slot(&self, feature_id: &Pubkey) -> Option<Slot> {
1173        self.active.get(feature_id).copied()
1174    }
1175
1176    /// List of enabled features that trigger full inflation
1177    pub fn full_inflation_features_enabled(&self) -> HashSet<Pubkey> {
1178        let mut hash_set = FULL_INFLATION_FEATURE_PAIRS
1179            .iter()
1180            .filter_map(|pair| {
1181                if self.is_active(&pair.vote_id) && self.is_active(&pair.enable_id) {
1182                    Some(pair.enable_id)
1183                } else {
1184                    None
1185                }
1186            })
1187            .collect::<HashSet<_>>();
1188
1189        if self.is_active(&full_inflation::devnet_and_testnet::id()) {
1190            hash_set.insert(full_inflation::devnet_and_testnet::id());
1191        }
1192        hash_set
1193    }
1194
1195    /// All features enabled, useful for testing
1196    pub fn all_enabled() -> Self {
1197        Self {
1198            active: FEATURE_NAMES.keys().cloned().map(|key| (key, 0)).collect(),
1199            inactive: HashSet::new(),
1200        }
1201    }
1202
1203    /// Activate a feature
1204    pub fn activate(&mut self, feature_id: &Pubkey, slot: Slot) {
1205        self.inactive.remove(feature_id);
1206        self.active.insert(*feature_id, slot);
1207    }
1208
1209    /// Deactivate a feature
1210    pub fn deactivate(&mut self, feature_id: &Pubkey) {
1211        self.active.remove(feature_id);
1212        self.inactive.insert(*feature_id);
1213    }
1214
1215    pub fn new_warmup_cooldown_rate_epoch(&self, epoch_schedule: &EpochSchedule) -> Option<Epoch> {
1216        self.activated_slot(&reduce_stake_warmup_cooldown::id())
1217            .map(|slot| epoch_schedule.get_epoch(slot))
1218    }
1219}
1220
1221#[cfg(test)]
1222mod test {
1223    use super::*;
1224
1225    #[test]
1226    fn test_full_inflation_features_enabled_devnet_and_testnet() {
1227        let mut feature_set = FeatureSet::default();
1228        assert!(feature_set.full_inflation_features_enabled().is_empty());
1229        feature_set
1230            .active
1231            .insert(full_inflation::devnet_and_testnet::id(), 42);
1232        assert_eq!(
1233            feature_set.full_inflation_features_enabled(),
1234            [full_inflation::devnet_and_testnet::id()]
1235                .iter()
1236                .cloned()
1237                .collect()
1238        );
1239    }
1240
1241    #[test]
1242    fn test_full_inflation_features_enabled() {
1243        // Normal sequence: vote_id then enable_id
1244        let mut feature_set = FeatureSet::default();
1245        assert!(feature_set.full_inflation_features_enabled().is_empty());
1246        feature_set
1247            .active
1248            .insert(full_inflation::mainnet::certusone::vote::id(), 42);
1249        assert!(feature_set.full_inflation_features_enabled().is_empty());
1250        feature_set
1251            .active
1252            .insert(full_inflation::mainnet::certusone::enable::id(), 42);
1253        assert_eq!(
1254            feature_set.full_inflation_features_enabled(),
1255            [full_inflation::mainnet::certusone::enable::id()]
1256                .iter()
1257                .cloned()
1258                .collect()
1259        );
1260
1261        // Backwards sequence: enable_id and then vote_id
1262        let mut feature_set = FeatureSet::default();
1263        assert!(feature_set.full_inflation_features_enabled().is_empty());
1264        feature_set
1265            .active
1266            .insert(full_inflation::mainnet::certusone::enable::id(), 42);
1267        assert!(feature_set.full_inflation_features_enabled().is_empty());
1268        feature_set
1269            .active
1270            .insert(full_inflation::mainnet::certusone::vote::id(), 42);
1271        assert_eq!(
1272            feature_set.full_inflation_features_enabled(),
1273            [full_inflation::mainnet::certusone::enable::id()]
1274                .iter()
1275                .cloned()
1276                .collect()
1277        );
1278    }
1279
1280    #[test]
1281    fn test_feature_set_activate_deactivate() {
1282        let mut feature_set = FeatureSet::default();
1283
1284        let feature = Pubkey::new_unique();
1285        assert!(!feature_set.is_active(&feature));
1286        feature_set.activate(&feature, 0);
1287        assert!(feature_set.is_active(&feature));
1288        feature_set.deactivate(&feature);
1289        assert!(!feature_set.is_active(&feature));
1290    }
1291}