From 1fc0552ebf8c2bb5fb6fb67efa68b64f74fc6665 Mon Sep 17 00:00:00 2001 From: Eliza Weisman Date: Wed, 15 Sep 2021 11:00:06 -0700 Subject: [PATCH] subscriber: don't use `SmallVec`s for filter fields The `DirectiveSet` type used in `EnvFilter` and `Targets` uses `SmallVec` to store the filtering directives when the `SmallVec` feature is enabled. This is intended to improve the performance of iterating over small sets of directives, by avoiding a heap pointer dereference. PR #1550 changed the directives themselves to also use `SmallVec` for storing _field_ filters. This was intended to make the same optimization for field filters. However, it had unintended consequences: an empty `SmallVec` is an array of `T` items (plus metadata), while an empty `Vec` is just a couple of words. Since _most_ filters don't have field filters, this meant that we were suddenly using a lot more space to store...nothing. This made `EnvFilter`s _much_ larger, causing problems for some users (see #1567). This branch undoes the change to `SmallVec` for field name/value filters. This takes the size of an `EnvFilter` from 5420 bytes back down to 1272 bytes. Fixes #1567 --- tracing-subscriber/src/filter/directive.rs | 20 ++++++++-------- .../src/filter/env/directive.rs | 13 +++++----- tracing-subscriber/src/filter/targets.rs | 24 +++++++++---------- 3 files changed, 28 insertions(+), 29 deletions(-) diff --git a/tracing-subscriber/src/filter/directive.rs b/tracing-subscriber/src/filter/directive.rs index 61cc00a9ec..82d1d70bba 100644 --- a/tracing-subscriber/src/filter/directive.rs +++ b/tracing-subscriber/src/filter/directive.rs @@ -13,14 +13,14 @@ pub struct ParseError { #[derive(Debug, PartialEq, Eq, Clone)] pub(crate) struct StaticDirective { pub(in crate::filter) target: Option, - pub(in crate::filter) field_names: FilterVec, + pub(in crate::filter) field_names: Vec, pub(in crate::filter) level: LevelFilter, } -#[cfg(feature = "smallvec")] -pub(in crate::filter) type FilterVec = smallvec::SmallVec<[T; 8]>; -#[cfg(not(feature = "smallvec"))] -pub(in crate::filter) type FilterVec = Vec; +// #[cfg(feature = "smallvec")] +// pub(crate) type FilterVec = smallvec::SmallVec<[T; 8]>; +// #[cfg(not(feature = "smallvec"))] +pub(crate) type FilterVec = Vec; #[derive(Debug, PartialEq, Clone)] pub(in crate::filter) struct DirectiveSet { @@ -129,7 +129,7 @@ impl DirectiveSet { impl StaticDirective { pub(in crate::filter) fn new( target: Option, - field_names: FilterVec, + field_names: Vec, level: LevelFilter, ) -> Self { Self { @@ -221,7 +221,7 @@ impl Default for StaticDirective { fn default() -> Self { StaticDirective { target: None, - field_names: FilterVec::new(), + field_names: Vec::new(), level: LevelFilter::ERROR, } } @@ -288,7 +288,7 @@ impl FromStr for StaticDirective { let mut split = part0.split("[{"); let target = split.next().map(String::from); - let mut field_names = FilterVec::new(); + let mut field_names = Vec::new(); // Directive includes fields: // * `foo[{bar}]=trace` // * `foo[{bar,baz}]=trace` @@ -326,12 +326,12 @@ impl FromStr for StaticDirective { Ok(level) => Self { level, target: None, - field_names: FilterVec::new(), + field_names: Vec::new(), }, Err(_) => Self { target: Some(String::from(part0)), level: LevelFilter::TRACE, - field_names: FilterVec::new(), + field_names: Vec::new(), }, }) } diff --git a/tracing-subscriber/src/filter/env/directive.rs b/tracing-subscriber/src/filter/env/directive.rs index 538df16111..66ca23dc41 100644 --- a/tracing-subscriber/src/filter/env/directive.rs +++ b/tracing-subscriber/src/filter/env/directive.rs @@ -1,5 +1,4 @@ -use super::FilterVec; -pub(crate) use crate::filter::directive::{ParseError, StaticDirective}; +pub(crate) use crate::filter::directive::{FilterVec, ParseError, StaticDirective}; use crate::filter::{ directive::{DirectiveSet, Match}, env::{field, FieldMap}, @@ -16,7 +15,7 @@ use tracing_core::{span, Level, Metadata}; #[cfg_attr(docsrs, doc(cfg(feature = "env-filter")))] pub struct Directive { in_span: Option, - fields: FilterVec, + fields: Vec, pub(crate) target: Option, pub(crate) level: LevelFilter, } @@ -216,12 +215,12 @@ impl FromStr for Directive { FIELD_FILTER_RE .find_iter(c.as_str()) .map(|c| c.as_str().parse()) - .collect::, _>>() + .collect::, _>>() }) - .unwrap_or_else(|| Ok(FilterVec::new())); + .unwrap_or_else(|| Ok(Vec::new())); Some((span, fields)) }) - .unwrap_or_else(|| (None, Ok(FilterVec::new()))); + .unwrap_or_else(|| (None, Ok(Vec::new()))); let level = caps .name("level") @@ -244,7 +243,7 @@ impl Default for Directive { level: LevelFilter::OFF, target: None, in_span: None, - fields: FilterVec::new(), + fields: Vec::new(), } } } diff --git a/tracing-subscriber/src/filter/targets.rs b/tracing-subscriber/src/filter/targets.rs index 6698745dfd..2c3c2a471c 100644 --- a/tracing-subscriber/src/filter/targets.rs +++ b/tracing-subscriber/src/filter/targets.rs @@ -361,11 +361,11 @@ mod tests { assert_eq!(dirs.len(), 2, "\nparsed: {:#?}", dirs); assert_eq!(dirs[0].target, Some("server".to_string())); assert_eq!(dirs[0].level, LevelFilter::DEBUG); - assert_eq!(dirs[0].field_names, FilterVec::::default()); + assert_eq!(dirs[0].field_names, Vec::::new()); assert_eq!(dirs[1].target, Some("common".to_string())); assert_eq!(dirs[1].level, LevelFilter::INFO); - assert_eq!(dirs[1].field_names, FilterVec::::default()); + assert_eq!(dirs[1].field_names, Vec::::new()); } fn expect_parse_level_directives(s: &str) { @@ -374,27 +374,27 @@ mod tests { assert_eq!(dirs[0].target, Some("crate3::mod2::mod1".to_string())); assert_eq!(dirs[0].level, LevelFilter::OFF); - assert_eq!(dirs[0].field_names, FilterVec::::default()); + assert_eq!(dirs[0].field_names, Vec::::new()); assert_eq!(dirs[1].target, Some("crate1::mod2::mod3".to_string())); assert_eq!(dirs[1].level, LevelFilter::INFO); - assert_eq!(dirs[1].field_names, FilterVec::::default()); + assert_eq!(dirs[1].field_names, Vec::::new()); assert_eq!(dirs[2].target, Some("crate1::mod2".to_string())); assert_eq!(dirs[2].level, LevelFilter::WARN); - assert_eq!(dirs[2].field_names, FilterVec::::default()); + assert_eq!(dirs[2].field_names, Vec::::new()); assert_eq!(dirs[3].target, Some("crate1::mod1".to_string())); assert_eq!(dirs[3].level, LevelFilter::ERROR); - assert_eq!(dirs[3].field_names, FilterVec::::default()); + assert_eq!(dirs[3].field_names, Vec::::new()); assert_eq!(dirs[4].target, Some("crate3".to_string())); assert_eq!(dirs[4].level, LevelFilter::TRACE); - assert_eq!(dirs[4].field_names, FilterVec::::default()); + assert_eq!(dirs[4].field_names, Vec::::new()); assert_eq!(dirs[5].target, Some("crate2".to_string())); assert_eq!(dirs[5].level, LevelFilter::DEBUG); - assert_eq!(dirs[5].field_names, FilterVec::::default()); + assert_eq!(dirs[5].field_names, Vec::::new()); } #[test] @@ -418,19 +418,19 @@ mod tests { assert_eq!(dirs.len(), 4, "\nparsed: {:#?}", dirs); assert_eq!(dirs[0].target, Some("crate1::mod2".to_string())); assert_eq!(dirs[0].level, LevelFilter::TRACE); - assert_eq!(dirs[0].field_names, FilterVec::::default()); + assert_eq!(dirs[0].field_names, Vec::::new()); assert_eq!(dirs[1].target, Some("crate1::mod1".to_string())); assert_eq!(dirs[1].level, LevelFilter::ERROR); - assert_eq!(dirs[1].field_names, FilterVec::::default()); + assert_eq!(dirs[1].field_names, Vec::::new()); assert_eq!(dirs[2].target, Some("crate3".to_string())); assert_eq!(dirs[2].level, LevelFilter::OFF); - assert_eq!(dirs[2].field_names, FilterVec::::default()); + assert_eq!(dirs[2].field_names, Vec::::new()); assert_eq!(dirs[3].target, Some("crate2".to_string())); assert_eq!(dirs[3].level, LevelFilter::DEBUG); - assert_eq!(dirs[3].field_names, FilterVec::::default()); + assert_eq!(dirs[3].field_names, Vec::::new()); } #[test]