2023-03-28 15:59:51 +00:00
|
|
|
use crate::{
|
2023-04-21 22:49:16 +00:00
|
|
|
common::escape,
|
2023-04-16 01:15:37 +00:00
|
|
|
ffi::{self, parser_t, wcharz_t, wcstring_list_ffi_t},
|
2023-03-28 15:59:51 +00:00
|
|
|
global_safety::RelaxedAtomicBool,
|
2023-08-08 22:16:04 +00:00
|
|
|
wchar::prelude::*,
|
2023-04-16 01:15:37 +00:00
|
|
|
wchar_ffi::{WCharFromFFI, WCharToFFI},
|
2023-03-28 15:59:51 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#[cxx::bridge]
|
|
|
|
mod trace_ffi {
|
|
|
|
extern "C++" {
|
|
|
|
include!("wutil.h");
|
|
|
|
include!("parser.h");
|
2023-04-16 01:15:37 +00:00
|
|
|
type wcstring_list_ffi_t = super::wcstring_list_ffi_t;
|
2023-03-28 15:59:51 +00:00
|
|
|
type wcharz_t = super::wcharz_t;
|
|
|
|
type parser_t = super::parser_t;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "Rust" {
|
|
|
|
fn trace_set_enabled(do_enable: bool);
|
|
|
|
fn trace_enabled(parser: &parser_t) -> bool;
|
|
|
|
#[cxx_name = "trace_argv"]
|
2023-04-16 01:15:37 +00:00
|
|
|
fn trace_argv_ffi(parser: &parser_t, command: wcharz_t, args: &wcstring_list_ffi_t);
|
2023-03-28 15:59:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static DO_TRACE: RelaxedAtomicBool = RelaxedAtomicBool::new(false);
|
|
|
|
|
|
|
|
pub fn trace_set_enabled(do_enable: bool) {
|
|
|
|
DO_TRACE.store(do_enable);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// return whether tracing is enabled.
|
|
|
|
pub fn trace_enabled(parser: &parser_t) -> bool {
|
|
|
|
let ld = parser.ffi_libdata_pod_const();
|
|
|
|
if ld.suppress_fish_trace {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
DO_TRACE.load()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Trace an "argv": a list of arguments where the first is the command.
|
|
|
|
// Allow the `&Vec` parameter as this function only exists temporarily for the FFI
|
|
|
|
#[allow(clippy::ptr_arg)]
|
2023-04-16 01:15:37 +00:00
|
|
|
fn trace_argv_ffi(parser: &parser_t, command: wcharz_t, args: &wcstring_list_ffi_t) {
|
2023-08-08 22:16:04 +00:00
|
|
|
let command: WString = command.into();
|
|
|
|
let args: Vec<WString> = args.from_ffi();
|
|
|
|
let args_ref: Vec<&wstr> = args.iter().map(WString::as_utfstr).collect();
|
2023-03-28 15:59:51 +00:00
|
|
|
trace_argv(parser, command.as_utfstr(), &args_ref);
|
|
|
|
}
|
|
|
|
|
2023-04-22 18:10:08 +00:00
|
|
|
pub fn trace_argv<S: AsRef<wstr>>(parser: &parser_t, command: &wstr, args: &[S]) {
|
2023-03-28 15:59:51 +00:00
|
|
|
// Format into a string to prevent interleaving with flog in other threads.
|
|
|
|
// Add the + prefix.
|
|
|
|
let mut trace_text = L!("-").repeat(parser.blocks_size() - 1);
|
|
|
|
trace_text.push('>');
|
|
|
|
|
|
|
|
if !command.is_empty() {
|
|
|
|
trace_text.push(' ');
|
|
|
|
trace_text.push_utfstr(command);
|
|
|
|
}
|
|
|
|
for arg in args {
|
|
|
|
trace_text.push(' ');
|
2023-04-22 18:10:08 +00:00
|
|
|
trace_text.push_utfstr(&escape(arg.as_ref()));
|
2023-03-28 15:59:51 +00:00
|
|
|
}
|
|
|
|
trace_text.push('\n');
|
|
|
|
ffi::log_extra_to_flog_file(&trace_text.to_ffi());
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Convenience helper to trace a single string if tracing is enabled.
|
2023-04-22 18:10:08 +00:00
|
|
|
pub fn trace_if_enabled<S: AsRef<wstr>>(parser: &parser_t, command: &wstr, args: &[S]) {
|
2023-03-28 15:59:51 +00:00
|
|
|
if trace_enabled(parser) {
|
|
|
|
trace_argv(parser, command, args);
|
|
|
|
}
|
|
|
|
}
|