mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-08 11:08:53 +00:00
77aeb6a2a8
Drop support for history file version 1. ParseExecutionContext no longer contains an OperationContext because in my first implementation, ParseExecutionContext didn't have interior mutability. We should probably try to add it back. Add a few to-do style comments. Search for "todo!" and "PORTING". Co-authored-by: Xiretza <xiretza@xiretza.xyz> (complete, wildcard, expand, history, history/file) Co-authored-by: Henrik Hørlück Berg <36937807+henrikhorluck@users.noreply.github.com> (builtins/set)
204 lines
6.4 KiB
Rust
204 lines
6.4 KiB
Rust
use crate::common::CancelChecker;
|
|
use crate::env::{EnvDyn, EnvDynFFI};
|
|
use crate::env::{EnvStack, EnvStackRef, Environment};
|
|
use crate::parser::{Parser, ParserRef};
|
|
use crate::proc::JobGroupRef;
|
|
use once_cell::sync::Lazy;
|
|
use std::sync::Arc;
|
|
|
|
/// A common helper which always returns false.
|
|
pub fn no_cancel() -> bool {
|
|
false
|
|
}
|
|
|
|
// Default limits for expansion.
|
|
/// The default maximum number of items from expansion.
|
|
pub const EXPANSION_LIMIT_DEFAULT: usize = 512 * 1024;
|
|
/// A smaller limit for background operations like syntax highlighting.
|
|
pub const EXPANSION_LIMIT_BACKGROUND: usize = 512;
|
|
|
|
enum Vars<'a> {
|
|
// The parser, if this is a foreground operation. If this is a background operation, this may be
|
|
// nullptr.
|
|
Parser(ParserRef),
|
|
// A set of variables.
|
|
Vars(&'a dyn Environment),
|
|
|
|
TestOnly(ParserRef, &'a dyn Environment),
|
|
}
|
|
|
|
/// A operation_context_t is a simple property bag which wraps up data needed for highlighting,
|
|
/// expansion, completion, and more.
|
|
pub struct OperationContext<'a> {
|
|
vars: Vars<'a>,
|
|
|
|
// The limit in the number of expansions which should be produced.
|
|
pub expansion_limit: usize,
|
|
|
|
/// The job group of the parental job.
|
|
/// This is used only when expanding command substitutions. If this is set, any jobs created
|
|
/// by the command substitutions should use this tree.
|
|
pub job_group: Option<JobGroupRef>,
|
|
|
|
// A function which may be used to poll for cancellation.
|
|
pub cancel_checker: CancelChecker,
|
|
}
|
|
|
|
static nullenv: Lazy<EnvStackRef> = Lazy::new(|| Arc::pin(EnvStack::new()));
|
|
|
|
impl<'a> OperationContext<'a> {
|
|
pub fn vars(&self) -> &dyn Environment {
|
|
match &self.vars {
|
|
Vars::Parser(parser) => &*parser.variables,
|
|
Vars::Vars(vars) => *vars,
|
|
Vars::TestOnly(_, vars) => *vars,
|
|
}
|
|
}
|
|
|
|
// \return an "empty" context which contains no variables, no parser, and never cancels.
|
|
pub fn empty() -> OperationContext<'static> {
|
|
OperationContext::background(&**nullenv, EXPANSION_LIMIT_DEFAULT)
|
|
}
|
|
|
|
// \return an operation context that contains only global variables, no parser, and never
|
|
// cancels.
|
|
pub fn globals() -> OperationContext<'static> {
|
|
OperationContext::background(&**EnvStack::globals(), EXPANSION_LIMIT_DEFAULT)
|
|
}
|
|
|
|
/// Construct from a full set of properties.
|
|
pub fn foreground(
|
|
parser: ParserRef,
|
|
cancel_checker: CancelChecker,
|
|
expansion_limit: usize,
|
|
) -> OperationContext<'a> {
|
|
OperationContext {
|
|
vars: Vars::Parser(parser),
|
|
expansion_limit,
|
|
job_group: None,
|
|
cancel_checker,
|
|
}
|
|
}
|
|
|
|
pub fn test_only_foreground(
|
|
parser: ParserRef,
|
|
vars: &'a dyn Environment,
|
|
cancel_checker: CancelChecker,
|
|
) -> OperationContext<'a> {
|
|
OperationContext {
|
|
vars: Vars::TestOnly(parser, vars),
|
|
expansion_limit: EXPANSION_LIMIT_DEFAULT,
|
|
job_group: None,
|
|
cancel_checker,
|
|
}
|
|
}
|
|
|
|
/// Construct from vars alone.
|
|
pub fn background(vars: &'a dyn Environment, expansion_limit: usize) -> OperationContext<'a> {
|
|
OperationContext {
|
|
vars: Vars::Vars(vars),
|
|
expansion_limit,
|
|
job_group: None,
|
|
cancel_checker: Box::new(no_cancel),
|
|
}
|
|
}
|
|
|
|
pub fn background_with_cancel_checker(
|
|
vars: &'a dyn Environment,
|
|
cancel_checker: CancelChecker,
|
|
expansion_limit: usize,
|
|
) -> OperationContext<'a> {
|
|
OperationContext {
|
|
vars: Vars::Vars(vars),
|
|
expansion_limit,
|
|
job_group: None,
|
|
cancel_checker,
|
|
}
|
|
}
|
|
|
|
pub fn has_parser(&self) -> bool {
|
|
matches!(self.vars, Vars::Parser(_) | Vars::TestOnly(_, _))
|
|
}
|
|
pub fn maybe_parser(&self) -> Option<&Parser> {
|
|
match &self.vars {
|
|
Vars::Parser(parser) => Some(parser),
|
|
Vars::Vars(_) => None,
|
|
Vars::TestOnly(parser, _) => Some(parser),
|
|
}
|
|
}
|
|
pub fn parser(&self) -> &Parser {
|
|
match &self.vars {
|
|
Vars::Parser(parser) => parser,
|
|
Vars::Vars(_) => panic!(),
|
|
Vars::TestOnly(parser, _) => parser,
|
|
}
|
|
}
|
|
// Invoke the cancel checker. \return if we should cancel.
|
|
pub fn check_cancel(&self) -> bool {
|
|
(self.cancel_checker)()
|
|
}
|
|
}
|
|
|
|
/// \return an operation context for a background operation..
|
|
/// Crucially the operation context itself does not contain a parser.
|
|
/// It is the caller's responsibility to ensure the environment lives as long as the result.
|
|
pub fn get_bg_context(env: &EnvDyn, generation_count: u32) -> OperationContext {
|
|
let cancel_checker = move || {
|
|
// Cancel if the generation count changed.
|
|
generation_count != crate::ffi::read_generation_count()
|
|
};
|
|
OperationContext::background_with_cancel_checker(
|
|
env,
|
|
Box::new(cancel_checker),
|
|
EXPANSION_LIMIT_BACKGROUND,
|
|
)
|
|
}
|
|
|
|
#[cxx::bridge]
|
|
#[allow(clippy::needless_lifetimes)]
|
|
mod operation_context_ffi {
|
|
extern "C++" {
|
|
include!("env_fwd.h");
|
|
include!("parser.h");
|
|
#[cxx_name = "EnvStackRef"]
|
|
type EnvStackRefFFI = crate::env::EnvStackRefFFI;
|
|
#[cxx_name = "EnvDyn"]
|
|
type EnvDynFFI = crate::env::EnvDynFFI;
|
|
type Parser = crate::parser::Parser;
|
|
}
|
|
extern "Rust" {
|
|
type OperationContext<'a>;
|
|
|
|
fn empty_operation_context() -> Box<OperationContext<'static>>;
|
|
fn operation_context_globals() -> Box<OperationContext<'static>>;
|
|
fn check_cancel(&self) -> bool;
|
|
|
|
#[cxx_name = "get_bg_context"]
|
|
unsafe fn get_bg_context_ffi(
|
|
env: &EnvDynFFI,
|
|
generation_count: u32,
|
|
) -> Box<OperationContext<'_>>;
|
|
#[cxx_name = "parser_context"]
|
|
fn parser_context_ffi(parser: &Parser) -> Box<OperationContext<'static>>;
|
|
}
|
|
}
|
|
|
|
fn get_bg_context_ffi(env: &EnvDynFFI, generation_count: u32) -> Box<OperationContext<'_>> {
|
|
Box::new(get_bg_context(&env.0, generation_count))
|
|
}
|
|
|
|
fn parser_context_ffi(parser: &Parser) -> Box<OperationContext<'static>> {
|
|
Box::new(parser.context())
|
|
}
|
|
|
|
unsafe impl cxx::ExternType for OperationContext<'_> {
|
|
type Id = cxx::type_id!("OperationContext");
|
|
type Kind = cxx::kind::Opaque;
|
|
}
|
|
|
|
fn empty_operation_context() -> Box<OperationContext<'static>> {
|
|
Box::new(OperationContext::empty())
|
|
}
|
|
fn operation_context_globals() -> Box<OperationContext<'static>> {
|
|
Box::new(OperationContext::globals())
|
|
}
|