Skip to main content

surfpool_types/
features.rs

1use std::str::FromStr;
2
3use agave_feature_set::FEATURE_NAMES;
4use serde::{Deserialize, Serialize};
5use solana_pubkey::Pubkey;
6
7/// Returns the pubkey for a known feature name (kebab-case or snake_case).
8///
9/// This covers the set of features previously exposed as CLI names.
10fn lookup_feature_by_name(name: &str) -> Option<Pubkey> {
11    use agave_feature_set::*;
12
13    // Normalize: accept both kebab-case and snake_case by converting underscores to hyphens
14    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        // "lift-cpi-caller-restriction" not available in agave-feature-set 3.1.x
67        "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
78/// Parse a feature from either a name (kebab-case or snake_case) or a base58 pubkey string,
79/// validating it against known agave feature gates.
80pub fn parse_feature_pubkey(s: &str) -> Result<Pubkey, String> {
81    // 1. Try name lookup (supports both kebab-case and snake_case)
82    if let Some(pubkey) = lookup_feature_by_name(s) {
83        return Ok(pubkey);
84    }
85
86    // 2. Try base58 pubkey parse
87    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/// Configuration for SVM features, specifying which features to enable or disable.
111#[derive(Debug, Clone, Default, PartialEq, Eq, Serialize, Deserialize)]
112pub struct SvmFeatureConfig {
113    /// Features to explicitly enable (override defaults)
114    pub enable: Vec<Pubkey>,
115    /// Features to explicitly disable (override defaults)
116    pub disable: Vec<Pubkey>,
117}
118
119impl SvmFeatureConfig {
120    /// Creates a new empty feature configuration.
121    pub fn new() -> Self {
122        Self::default()
123    }
124
125    /// Returns the default mainnet feature configuration.
126    ///
127    /// This reflects features currently active on Solana mainnet-beta.
128    /// Note: This may need periodic updates as mainnet features change.
129    /// Last updated: 2025-01-25 (queried from mainnet RPC)
130    pub fn default_mainnet_features() -> Self {
131        use agave_feature_set::*;
132
133        // Features that are NOT yet active on mainnet (should be disabled)
134        let disable = vec![
135            // Blake3 syscall not yet on mainnet
136            blake3_syscall_enabled::id(),
137            // Legacy vote deprecation not yet on mainnet
138            deprecate_legacy_vote_ixs::id(),
139            // SBPF v0 disable/reenable not yet on mainnet
140            disable_sbpf_v0_execution::id(),
141            reenable_sbpf_v0_execution::id(),
142            // ZK ElGamal disable not yet on mainnet (reenable IS active)
143            disable_zk_elgamal_proof_program::id(),
144            // Extended program checked not yet on mainnet
145            enable_extend_program_checked::id(),
146            // Loader v4 not yet on mainnet
147            enable_loader_v4::id(),
148            // SBPF v1 not yet on mainnet (v2 and v3 ARE active)
149            enable_sbpf_v1_deployment_and_execution::id(),
150            // Transaction data size formalization not yet on mainnet
151            formalize_loaded_transaction_data_size::id(),
152            // Precompile verification move not yet on mainnet
153            move_precompile_verification_to_svm::id(),
154            // Stake move instructions not yet on mainnet
155            move_stake_and_move_lamports_ixs::id(),
156            // Stake minimum delegation raise not yet on mainnet
157            stake_raise_minimum_delegation_to_1_sol::id(),
158            // New features from LiteSVM 0.9.0 / Solana SVM v3.1 (not yet on mainnet)
159            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    /// Adds a feature to enable.
173    pub fn enable(mut self, feature: Pubkey) -> Self {
174        if !self.enable.contains(&feature) {
175            self.enable.push(feature);
176        }
177        // Remove from disable if present
178        self.disable.retain(|f| f != &feature);
179        self
180    }
181
182    /// Adds a feature to disable.
183    pub fn disable(mut self, feature: Pubkey) -> Self {
184        if !self.disable.contains(&feature) {
185            self.disable.push(feature);
186        }
187        // Remove from enable if present
188        self.enable.retain(|f| f != &feature);
189        self
190    }
191
192    /// Checks if a feature should be enabled based on this configuration.
193    /// Returns None if not explicitly configured (use default).
194    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    // ==================== parse_feature_pubkey tests ====================
212
213    #[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        // Verify that a known feature name resolves correctly
234        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        // System program is not a feature gate
248        let err = parse_feature_pubkey("11111111111111111111111111111111").unwrap_err();
249        assert!(err.contains("Not a known agave feature gate"));
250    }
251
252    // ==================== SvmFeatureConfig basic tests ====================
253
254    #[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    // ==================== SvmFeatureConfig complex scenarios ====================
291
292    #[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    // ==================== Mainnet defaults tests ====================
352
353    #[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        // Features that ARE active on mainnet should not be in disable list
430        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    // ==================== Serialization tests ====================
446
447    #[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    // ==================== Edge cases ====================
460
461    #[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}