mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-02 16:18:44 +00:00
4639f7ec40
But don't do it for enum variants just yet.
254 lines
8.1 KiB
Rust
254 lines
8.1 KiB
Rust
//! Flags to enable upcoming features
|
|
|
|
use crate::ffi::wcharz_t;
|
|
use crate::wchar::wstr;
|
|
use crate::wchar_ffi::WCharToFFI;
|
|
use std::array;
|
|
use std::cell::UnsafeCell;
|
|
use std::sync::atomic::AtomicBool;
|
|
use std::sync::atomic::Ordering;
|
|
use widestring_suffix::widestrs;
|
|
|
|
#[cxx::bridge]
|
|
mod future_feature_flags_ffi {
|
|
extern "C++" {
|
|
include!("wutil.h");
|
|
type wcharz_t = super::wcharz_t;
|
|
}
|
|
|
|
/// The list of flags.
|
|
#[repr(u8)]
|
|
enum FeatureFlag {
|
|
/// Whether ^ is supported for stderr redirection.
|
|
stderr_nocaret,
|
|
|
|
/// Whether ? is supported as a glob.
|
|
qmark_noglob,
|
|
|
|
/// Whether string replace -r double-unescapes the replacement.
|
|
string_replace_backslash,
|
|
|
|
/// Whether "&" is not-special if followed by a word character.
|
|
ampersand_nobg_in_token,
|
|
}
|
|
|
|
/// Metadata about feature flags.
|
|
struct feature_metadata_t {
|
|
flag: FeatureFlag,
|
|
name: UniquePtr<CxxWString>,
|
|
groups: UniquePtr<CxxWString>,
|
|
description: UniquePtr<CxxWString>,
|
|
default_value: bool,
|
|
read_only: bool,
|
|
}
|
|
|
|
extern "Rust" {
|
|
type Features;
|
|
fn test(self: &Features, flag: FeatureFlag) -> bool;
|
|
fn set(self: &mut Features, flag: FeatureFlag, value: bool);
|
|
fn set_from_string(self: &mut Features, str: wcharz_t);
|
|
fn fish_features() -> *const Features;
|
|
fn feature_test(flag: FeatureFlag) -> bool;
|
|
fn mutable_fish_features() -> *mut Features;
|
|
fn feature_metadata() -> [feature_metadata_t; 4];
|
|
}
|
|
}
|
|
|
|
pub use future_feature_flags_ffi::{feature_metadata_t, FeatureFlag};
|
|
|
|
pub struct Features {
|
|
// Values for the flags.
|
|
// These are atomic to "fix" a race reported by tsan where tests of feature flags and other
|
|
// tests which use them conceptually race.
|
|
values: [AtomicBool; metadata.len()],
|
|
}
|
|
|
|
/// Metadata about feature flags.
|
|
struct FeatureMetadata {
|
|
/// The flag itself.
|
|
flag: FeatureFlag,
|
|
|
|
/// User-presentable short name of the feature flag.
|
|
name: &'static wstr,
|
|
|
|
/// Comma-separated list of feature groups.
|
|
groups: &'static wstr,
|
|
|
|
/// User-presentable description of the feature flag.
|
|
description: &'static wstr,
|
|
|
|
/// Default flag value.
|
|
default_value: bool,
|
|
|
|
/// Whether the value can still be changed or not.
|
|
read_only: bool,
|
|
}
|
|
|
|
impl From<&FeatureMetadata> for feature_metadata_t {
|
|
fn from(md: &FeatureMetadata) -> feature_metadata_t {
|
|
feature_metadata_t {
|
|
flag: md.flag,
|
|
name: md.name.to_ffi(),
|
|
groups: md.groups.to_ffi(),
|
|
description: md.description.to_ffi(),
|
|
default_value: md.default_value,
|
|
read_only: md.read_only,
|
|
}
|
|
}
|
|
}
|
|
|
|
/// The metadata, indexed by flag.
|
|
#[widestrs]
|
|
const metadata: [FeatureMetadata; 4] = [
|
|
FeatureMetadata {
|
|
flag: FeatureFlag::stderr_nocaret,
|
|
name: "stderr-nocaret"L,
|
|
groups: "3.0"L,
|
|
description: "^ no longer redirects stderr (historical, can no longer be changed)"L,
|
|
default_value: true,
|
|
read_only: true,
|
|
},
|
|
FeatureMetadata {
|
|
flag: FeatureFlag::qmark_noglob,
|
|
name: "qmark-noglob"L,
|
|
groups: "3.0"L,
|
|
description: "? no longer globs"L,
|
|
default_value: false,
|
|
read_only: false,
|
|
},
|
|
FeatureMetadata {
|
|
flag: FeatureFlag::string_replace_backslash,
|
|
name: "regex-easyesc"L,
|
|
groups: "3.1"L,
|
|
description: "string replace -r needs fewer \\'s"L,
|
|
default_value: true,
|
|
read_only: false,
|
|
},
|
|
FeatureMetadata {
|
|
flag: FeatureFlag::ampersand_nobg_in_token,
|
|
name: "ampersand-nobg-in-token"L,
|
|
groups: "3.4"L,
|
|
description: "& only backgrounds if followed by a separator"L,
|
|
default_value: true,
|
|
read_only: false,
|
|
},
|
|
];
|
|
|
|
/// The singleton shared feature set.
|
|
static mut global_features: *const UnsafeCell<Features> = std::ptr::null();
|
|
|
|
pub fn future_feature_flags_init() {
|
|
unsafe {
|
|
// Leak it for now.
|
|
global_features = Box::into_raw(Box::new(UnsafeCell::new(Features::new())));
|
|
}
|
|
}
|
|
|
|
impl Features {
|
|
fn new() -> Self {
|
|
Features {
|
|
values: array::from_fn(|i| AtomicBool::new(metadata[i].default_value)),
|
|
}
|
|
}
|
|
|
|
/// Return whether a flag is set.
|
|
pub fn test(&self, flag: FeatureFlag) -> bool {
|
|
self.values[flag.repr as usize].load(Ordering::SeqCst)
|
|
}
|
|
|
|
/// Set a flag.
|
|
pub fn set(&mut self, flag: FeatureFlag, value: bool) {
|
|
self.values[flag.repr as usize].store(value, Ordering::SeqCst)
|
|
}
|
|
|
|
/// Parses a comma-separated feature-flag string, updating ourselves with the values.
|
|
/// Feature names or group names may be prefixed with "no-" to disable them.
|
|
/// The special group name "all" may be used for those who like to live on the edge.
|
|
/// Unknown features are silently ignored.
|
|
#[widestrs]
|
|
pub fn set_from_string<'a>(&mut self, str: impl Into<&'a wstr>) {
|
|
let str: &wstr = str.into();
|
|
let whitespace = "\t\n\0x0B\0x0C\r "L.as_char_slice();
|
|
for entry in str.as_char_slice().split(|c| *c == ',') {
|
|
if entry.is_empty() {
|
|
continue;
|
|
}
|
|
|
|
// Trim leading and trailing whitespace
|
|
let entry = &entry[entry.iter().take_while(|c| whitespace.contains(c)).count()..];
|
|
let entry =
|
|
&entry[..entry.len() - entry.iter().take_while(|c| whitespace.contains(c)).count()];
|
|
|
|
// A "no-" prefix inverts the sense.
|
|
let (name, value) = match entry.strip_prefix("no-"L.as_char_slice()) {
|
|
Some(suffix) => (suffix, false),
|
|
None => (entry, true),
|
|
};
|
|
// Look for a feature with this name. If we don't find it, assume it's a group name and set
|
|
// all features whose group contain it. Do nothing even if the string is unrecognized; this
|
|
// is to allow uniform invocations of fish (e.g. disable a feature that is only present in
|
|
// future versions).
|
|
// The special name 'all' may be used for those who like to live on the edge.
|
|
if let Some(md) = metadata.iter().find(|md| md.name == name) {
|
|
// Only change it if it's not read-only.
|
|
// Don't complain if it is, this is typically set from a variable.
|
|
if !md.read_only {
|
|
self.set(md.flag, value);
|
|
}
|
|
} else {
|
|
for md in &metadata {
|
|
if md.groups == name || name == "all"L {
|
|
if !md.read_only {
|
|
self.set(md.flag, value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Return the global set of features for fish. This is const to prevent accidental mutation.
|
|
pub fn fish_features() -> *const Features {
|
|
unsafe { (*global_features).get() }
|
|
}
|
|
|
|
/// Perform a feature test on the global set of features.
|
|
pub fn feature_test(flag: FeatureFlag) -> bool {
|
|
unsafe { &*(*global_features).get() }.test(flag)
|
|
}
|
|
|
|
/// Return the global set of features for fish, but mutable. In general fish features should be set
|
|
/// at startup only.
|
|
pub fn mutable_fish_features() -> *mut Features {
|
|
unsafe { (*global_features).get() }
|
|
}
|
|
|
|
// The metadata, indexed by flag.
|
|
pub fn feature_metadata() -> [feature_metadata_t; metadata.len()] {
|
|
array::from_fn(|i| (&metadata[i]).into())
|
|
}
|
|
|
|
#[test]
|
|
#[widestrs]
|
|
fn test_feature_flags() {
|
|
let mut f = Features::new();
|
|
f.set_from_string("stderr-nocaret,nonsense"L);
|
|
assert!(f.test(FeatureFlag::stderr_nocaret));
|
|
f.set_from_string("stderr-nocaret,no-stderr-nocaret,nonsense"L);
|
|
assert!(f.test(FeatureFlag::stderr_nocaret));
|
|
|
|
// Ensure every metadata is represented once.
|
|
let mut counts: [usize; metadata.len()] = [0; metadata.len()];
|
|
for md in &metadata {
|
|
counts[md.flag.repr as usize] += 1;
|
|
}
|
|
for count in counts {
|
|
assert_eq!(count, 1);
|
|
}
|
|
|
|
assert_eq!(
|
|
metadata[FeatureFlag::stderr_nocaret.repr as usize].name,
|
|
"stderr-nocaret"L
|
|
);
|
|
}
|