1use std::str::FromStr;
2
3use agave_feature_set::FEATURE_NAMES;
4use serde::{Deserialize, Serialize};
5use solana_pubkey::Pubkey;
6
7fn lookup_feature_by_name(name: &str) -> Option<Pubkey> {
11 use agave_feature_set::*;
12
13 let normalized = name.replace('_', "-");
15 let s = normalized.as_str();
16
17 let pubkey = match s {
18 "move-precompile-verification-to-svm" => move_precompile_verification_to_svm::id(),
19 "stricter-abi-and-runtime-constraints" => stricter_abi_and_runtime_constraints::id(),
20 "enable-bpf-loader-set-authority-checked-ix" => {
21 enable_bpf_loader_set_authority_checked_ix::id()
22 }
23 "enable-loader-v4" => enable_loader_v4::id(),
24 "deplete-cu-meter-on-vm-failure" => deplete_cu_meter_on_vm_failure::id(),
25 "abort-on-invalid-curve" => abort_on_invalid_curve::id(),
26 "blake3-syscall-enabled" => blake3_syscall_enabled::id(),
27 "curve25519-syscall-enabled" => curve25519_syscall_enabled::id(),
28 "disable-deploy-of-alloc-free-syscall" => disable_deploy_of_alloc_free_syscall::id(),
29 "disable-fees-sysvar" => disable_fees_sysvar::id(),
30 "disable-sbpf-v0-execution" => disable_sbpf_v0_execution::id(),
31 "enable-alt-bn128-compression-syscall" => enable_alt_bn128_compression_syscall::id(),
32 "enable-alt-bn128-syscall" => enable_alt_bn128_syscall::id(),
33 "enable-big-mod-exp-syscall" => enable_big_mod_exp_syscall::id(),
34 "enable-get-epoch-stake-syscall" => enable_get_epoch_stake_syscall::id(),
35 "enable-poseidon-syscall" => enable_poseidon_syscall::id(),
36 "enable-sbpf-v1-deployment-and-execution" => enable_sbpf_v1_deployment_and_execution::id(),
37 "enable-sbpf-v2-deployment-and-execution" => enable_sbpf_v2_deployment_and_execution::id(),
38 "enable-sbpf-v3-deployment-and-execution" => enable_sbpf_v3_deployment_and_execution::id(),
39 "get-sysvar-syscall-enabled" => get_sysvar_syscall_enabled::id(),
40 "last-restart-slot-sysvar" => last_restart_slot_sysvar::id(),
41 "reenable-sbpf-v0-execution" => reenable_sbpf_v0_execution::id(),
42 "remaining-compute-units-syscall-enabled" => remaining_compute_units_syscall_enabled::id(),
43 "remove-bpf-loader-incorrect-program-id" => remove_bpf_loader_incorrect_program_id::id(),
44 "move-stake-and-move-lamports-ixs" => move_stake_and_move_lamports_ixs::id(),
45 "stake-raise-minimum-delegation-to-1-sol" => stake_raise_minimum_delegation_to_1_sol::id(),
46 "deprecate-legacy-vote-ixs" => deprecate_legacy_vote_ixs::id(),
47 "mask-out-rent-epoch-in-vm-serialization" => mask_out_rent_epoch_in_vm_serialization::id(),
48 "simplify-alt-bn128-syscall-error-codes" => simplify_alt_bn128_syscall_error_codes::id(),
49 "fix-alt-bn128-multiplication-input-length" => {
50 fix_alt_bn128_multiplication_input_length::id()
51 }
52 "increase-tx-account-lock-limit" => increase_tx_account_lock_limit::id(),
53 "enable-extend-program-checked" => enable_extend_program_checked::id(),
54 "formalize-loaded-transaction-data-size" => formalize_loaded_transaction_data_size::id(),
55 "disable-zk-elgamal-proof-program" => disable_zk_elgamal_proof_program::id(),
56 "reenable-zk-elgamal-proof-program" => reenable_zk_elgamal_proof_program::id(),
57 "raise-cpi-nesting-limit-to-8" => raise_cpi_nesting_limit_to_8::id(),
58 "account-data-direct-mapping" => account_data_direct_mapping::id(),
59 "provide-instruction-data-offset-in-vm-r2" => {
60 provide_instruction_data_offset_in_vm_r2::id()
61 }
62 "increase-cpi-account-info-limit" => increase_cpi_account_info_limit::id(),
63 "vote-state-v4" => vote_state_v4::id(),
64 "poseidon-enforce-padding" => poseidon_enforce_padding::id(),
65 "fix-alt-bn128-pairing-length-check" => fix_alt_bn128_pairing_length_check::id(),
66 "remove-accounts-executable-flag-checks" => remove_accounts_executable_flag_checks::id(),
68 "loosen-cpi-size-restriction" => loosen_cpi_size_restriction::id(),
69 "disable-rent-fees-collection" => disable_rent_fees_collection::id(),
70 "deprecate-rent-exemption-threshold" => deprecate_rent_exemption_threshold::id(),
71 "replace-spl-token-with-p-token" => replace_spl_token_with_p_token::id(),
72 _ => return None,
73 };
74
75 Some(pubkey)
76}
77
78pub fn parse_feature_pubkey(s: &str) -> Result<Pubkey, String> {
81 if let Some(pubkey) = lookup_feature_by_name(s) {
83 return Ok(pubkey);
84 }
85
86 let pubkey = Pubkey::from_str(s).map_err(|_| {
88 format!(
89 "Invalid feature pubkey: '{}'. Expected a base58-encoded pubkey of a known agave feature gate.",
90 s
91 )
92 })?;
93
94 if !FEATURE_NAMES.contains_key(&pubkey) {
95 let mut available: Vec<_> = FEATURE_NAMES
96 .iter()
97 .map(|(k, name)| format!(" {} ({})", k, name))
98 .collect();
99 available.sort();
100 return Err(format!(
101 "Available features:\n{}\n\nUnknown feature: '{}'. Not a known agave feature gate. Available features listed above.",
102 pubkey,
103 available.join("\n")
104 ));
105 }
106
107 Ok(pubkey)
108}
109
110#[derive(Debug, Clone, Default, PartialEq, Eq, Serialize, Deserialize)]
112pub struct SvmFeatureConfig {
113 pub enable: Vec<Pubkey>,
115 pub disable: Vec<Pubkey>,
117}
118
119impl SvmFeatureConfig {
120 pub fn new() -> Self {
122 Self::default()
123 }
124
125 pub fn default_mainnet_features() -> Self {
131 use agave_feature_set::*;
132
133 let disable = vec![
135 blake3_syscall_enabled::id(),
137 deprecate_legacy_vote_ixs::id(),
139 disable_sbpf_v0_execution::id(),
141 reenable_sbpf_v0_execution::id(),
142 disable_zk_elgamal_proof_program::id(),
144 enable_extend_program_checked::id(),
146 enable_loader_v4::id(),
148 enable_sbpf_v1_deployment_and_execution::id(),
150 formalize_loaded_transaction_data_size::id(),
152 move_precompile_verification_to_svm::id(),
154 move_stake_and_move_lamports_ixs::id(),
156 stake_raise_minimum_delegation_to_1_sol::id(),
158 remove_accounts_executable_flag_checks::id(),
160 loosen_cpi_size_restriction::id(),
161 disable_rent_fees_collection::id(),
162 deprecate_rent_exemption_threshold::id(),
163 replace_spl_token_with_p_token::id(),
164 ];
165
166 Self {
167 enable: vec![],
168 disable,
169 }
170 }
171
172 pub fn enable(mut self, feature: Pubkey) -> Self {
174 if !self.enable.contains(&feature) {
175 self.enable.push(feature);
176 }
177 self.disable.retain(|f| f != &feature);
179 self
180 }
181
182 pub fn disable(mut self, feature: Pubkey) -> Self {
184 if !self.disable.contains(&feature) {
185 self.disable.push(feature);
186 }
187 self.enable.retain(|f| f != &feature);
189 self
190 }
191
192 pub fn is_enabled(&self, feature: &Pubkey) -> Option<bool> {
195 if self.enable.contains(feature) {
196 Some(true)
197 } else if self.disable.contains(feature) {
198 Some(false)
199 } else {
200 None
201 }
202 }
203}
204
205#[cfg(test)]
206mod tests {
207 use agave_feature_set::*;
208
209 use super::*;
210
211 #[test]
214 fn test_parse_feature_pubkey_valid() {
215 let pubkey = parse_feature_pubkey(&enable_loader_v4::id().to_string()).unwrap();
216 assert_eq!(pubkey, enable_loader_v4::id());
217 }
218
219 #[test]
220 fn test_parse_feature_name_snake_case() {
221 let pubkey = parse_feature_pubkey("enable_loader_v4").unwrap();
222 assert_eq!(pubkey, enable_loader_v4::id());
223 }
224
225 #[test]
226 fn test_parse_feature_name_kebab_case() {
227 let pubkey = parse_feature_pubkey("enable-loader-v4").unwrap();
228 assert_eq!(pubkey, enable_loader_v4::id());
229 }
230
231 #[test]
232 fn test_parse_feature_name_prefers_name_over_pubkey() {
233 let pubkey = parse_feature_pubkey("blake3_syscall_enabled").unwrap();
235 assert_eq!(pubkey, blake3_syscall_enabled::id());
236 }
237
238 #[test]
239 fn test_parse_feature_unknown_name() {
240 let err = parse_feature_pubkey("nonexistent-feature-name").unwrap_err();
241 assert!(err.contains("Invalid feature"));
242 assert!(err.contains("nonexistent-feature-name"));
243 }
244
245 #[test]
246 fn test_parse_feature_pubkey_unknown_pubkey() {
247 let err = parse_feature_pubkey("11111111111111111111111111111111").unwrap_err();
249 assert!(err.contains("Not a known agave feature gate"));
250 }
251
252 #[test]
255 fn test_feature_config_new_is_empty() {
256 let config = SvmFeatureConfig::new();
257 assert!(config.enable.is_empty());
258 assert!(config.disable.is_empty());
259 }
260
261 #[test]
262 fn test_feature_config_default_is_empty() {
263 let config = SvmFeatureConfig::default();
264 assert!(config.enable.is_empty());
265 assert!(config.disable.is_empty());
266 }
267
268 #[test]
269 fn test_feature_config_enable() {
270 let config = SvmFeatureConfig::new().enable(enable_loader_v4::id());
271 assert_eq!(config.is_enabled(&enable_loader_v4::id()), Some(true));
272 assert_eq!(config.enable.len(), 1);
273 assert!(config.disable.is_empty());
274 }
275
276 #[test]
277 fn test_feature_config_disable() {
278 let config = SvmFeatureConfig::new().disable(disable_fees_sysvar::id());
279 assert_eq!(config.is_enabled(&disable_fees_sysvar::id()), Some(false));
280 assert!(config.enable.is_empty());
281 assert_eq!(config.disable.len(), 1);
282 }
283
284 #[test]
285 fn test_feature_config_is_enabled_not_configured() {
286 let config = SvmFeatureConfig::new();
287 assert_eq!(config.is_enabled(&blake3_syscall_enabled::id()), None);
288 }
289
290 #[test]
293 fn test_feature_config_enable_then_disable() {
294 let config = SvmFeatureConfig::new()
295 .enable(enable_loader_v4::id())
296 .disable(enable_loader_v4::id());
297
298 assert_eq!(config.is_enabled(&enable_loader_v4::id()), Some(false));
299 assert!(config.enable.is_empty());
300 assert_eq!(config.disable.len(), 1);
301 }
302
303 #[test]
304 fn test_feature_config_disable_then_enable() {
305 let config = SvmFeatureConfig::new()
306 .disable(enable_loader_v4::id())
307 .enable(enable_loader_v4::id());
308
309 assert_eq!(config.is_enabled(&enable_loader_v4::id()), Some(true));
310 assert_eq!(config.enable.len(), 1);
311 assert!(config.disable.is_empty());
312 }
313
314 #[test]
315 fn test_feature_config_enable_idempotent() {
316 let config = SvmFeatureConfig::new()
317 .enable(enable_loader_v4::id())
318 .enable(enable_loader_v4::id());
319
320 assert_eq!(config.enable.len(), 1);
321 }
322
323 #[test]
324 fn test_feature_config_disable_idempotent() {
325 let config = SvmFeatureConfig::new()
326 .disable(enable_loader_v4::id())
327 .disable(enable_loader_v4::id());
328
329 assert_eq!(config.disable.len(), 1);
330 }
331
332 #[test]
333 fn test_feature_config_multiple_features() {
334 let config = SvmFeatureConfig::new()
335 .enable(enable_loader_v4::id())
336 .enable(blake3_syscall_enabled::id())
337 .disable(disable_fees_sysvar::id())
338 .disable(disable_sbpf_v0_execution::id());
339
340 assert_eq!(config.is_enabled(&enable_loader_v4::id()), Some(true));
341 assert_eq!(config.is_enabled(&blake3_syscall_enabled::id()), Some(true));
342 assert_eq!(config.is_enabled(&disable_fees_sysvar::id()), Some(false));
343 assert_eq!(
344 config.is_enabled(&disable_sbpf_v0_execution::id()),
345 Some(false)
346 );
347 assert_eq!(config.enable.len(), 2);
348 assert_eq!(config.disable.len(), 2);
349 }
350
351 #[test]
354 fn test_mainnet_features_disabled_list() {
355 let config = SvmFeatureConfig::default_mainnet_features();
356
357 assert_eq!(
358 config.is_enabled(&blake3_syscall_enabled::id()),
359 Some(false)
360 );
361 assert_eq!(
362 config.is_enabled(&deprecate_legacy_vote_ixs::id()),
363 Some(false)
364 );
365 assert_eq!(
366 config.is_enabled(&disable_sbpf_v0_execution::id()),
367 Some(false)
368 );
369 assert_eq!(
370 config.is_enabled(&reenable_sbpf_v0_execution::id()),
371 Some(false)
372 );
373 assert_eq!(
374 config.is_enabled(&disable_zk_elgamal_proof_program::id()),
375 Some(false)
376 );
377 assert_eq!(
378 config.is_enabled(&enable_extend_program_checked::id()),
379 Some(false)
380 );
381 assert_eq!(config.is_enabled(&enable_loader_v4::id()), Some(false));
382 assert_eq!(
383 config.is_enabled(&enable_sbpf_v1_deployment_and_execution::id()),
384 Some(false)
385 );
386 assert_eq!(
387 config.is_enabled(&formalize_loaded_transaction_data_size::id()),
388 Some(false)
389 );
390 assert_eq!(
391 config.is_enabled(&move_precompile_verification_to_svm::id()),
392 Some(false)
393 );
394 assert_eq!(
395 config.is_enabled(&move_stake_and_move_lamports_ixs::id()),
396 Some(false)
397 );
398 assert_eq!(
399 config.is_enabled(&stake_raise_minimum_delegation_to_1_sol::id()),
400 Some(false)
401 );
402 }
403
404 #[test]
405 fn test_mainnet_features_has_no_enables() {
406 let config = SvmFeatureConfig::default_mainnet_features();
407 assert!(config.enable.is_empty());
408 }
409
410 #[test]
411 fn test_mainnet_features_override_with_enable() {
412 let config = SvmFeatureConfig::default_mainnet_features().enable(enable_loader_v4::id());
413
414 assert_eq!(config.is_enabled(&enable_loader_v4::id()), Some(true));
415 assert_eq!(
416 config.is_enabled(&blake3_syscall_enabled::id()),
417 Some(false)
418 );
419 assert_eq!(
420 config.is_enabled(&enable_extend_program_checked::id()),
421 Some(false)
422 );
423 }
424
425 #[test]
426 fn test_mainnet_features_active_features_not_in_disable() {
427 let config = SvmFeatureConfig::default_mainnet_features();
428
429 assert_eq!(config.is_enabled(&disable_fees_sysvar::id()), None);
431 assert_eq!(config.is_enabled(&curve25519_syscall_enabled::id()), None);
432 assert_eq!(config.is_enabled(&enable_alt_bn128_syscall::id()), None);
433 assert_eq!(config.is_enabled(&enable_poseidon_syscall::id()), None);
434 assert_eq!(
435 config.is_enabled(&enable_sbpf_v2_deployment_and_execution::id()),
436 None
437 );
438 assert_eq!(
439 config.is_enabled(&enable_sbpf_v3_deployment_and_execution::id()),
440 None
441 );
442 assert_eq!(config.is_enabled(&raise_cpi_nesting_limit_to_8::id()), None);
443 }
444
445 #[test]
448 fn test_feature_config_serde_roundtrip() {
449 let config = SvmFeatureConfig::new()
450 .enable(enable_loader_v4::id())
451 .disable(disable_fees_sysvar::id());
452
453 let json = serde_json::to_string(&config).unwrap();
454 let parsed: SvmFeatureConfig = serde_json::from_str(&json).unwrap();
455
456 assert_eq!(config, parsed);
457 }
458
459 #[test]
462 fn test_feature_config_clone() {
463 let config = SvmFeatureConfig::new()
464 .enable(enable_loader_v4::id())
465 .disable(disable_fees_sysvar::id());
466
467 let cloned = config.clone();
468 assert_eq!(config, cloned);
469 }
470}