1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225
use ahash::HashMap;
use bit_vec::BitVec;
use nohash_hasher::IntMap;
use re_chunk_store::{ChunkStoreDiffKind, ChunkStoreEvent, ChunkStoreSubscriber};
use re_log_types::{EntityPathHash, StoreId};
use re_types::{ComponentName, ComponentNameSet};
use crate::{
ApplicableEntities, IdentifiedViewSystem, IndicatedEntities, ViewSystemIdentifier,
VisualizerSystem,
};
/// A store subscriber that keep track which entities in a store can be
/// processed by a single given visualizer type.
///
/// The list of entities is additive:
/// If an entity was at any point in time applicable to the visualizer, it will be
/// kept in the list of entities.
///
/// Applicability is determined by the visualizer's set of required components.
///
/// There's only a single entity subscriber per visualizer *type*.
/// This means that if the same visualizer is used in multiple views, only a single
/// `VisualizerEntitySubscriber` is created for all of them.
pub struct VisualizerEntitySubscriber {
/// Visualizer type this subscriber is associated with.
visualizer: ViewSystemIdentifier,
/// See [`crate::VisualizerQueryInfo::indicators`]
indicator_components: ComponentNameSet,
/// Assigns each required component an index.
required_components_indices: IntMap<ComponentName, usize>,
per_store_mapping: HashMap<StoreId, VisualizerEntityMapping>,
/// Additional filter for applicability.
applicability_filter: Box<dyn VisualizerAdditionalApplicabilityFilter>,
}
/// Additional filter for applicability on top of the default check for required components.
pub trait VisualizerAdditionalApplicabilityFilter: Send + Sync {
/// Updates the internal applicability filter state based on the given events.
///
/// Called for every update no matter whether the entity is already has all required components or not.
///
/// Returns true if the entity changed in the event is now applicable to the visualizer, false otherwise.
/// Once a entity passes this filter, it can never go back to being filtered out.
/// **This implies that the filter does not _need_ to be stateful.**
/// It is perfectly fine to return `true` only if something in the diff is regarded as applicable and false otherwise.
/// (However, if necessary, the applicability filter *can* keep track of state.)
fn update_applicability(&mut self, _event: &ChunkStoreEvent) -> bool;
}
struct DefaultVisualizerApplicabilityFilter;
impl VisualizerAdditionalApplicabilityFilter for DefaultVisualizerApplicabilityFilter {
#[inline]
fn update_applicability(&mut self, _event: &ChunkStoreEvent) -> bool {
true
}
}
#[derive(Default)]
struct VisualizerEntityMapping {
/// For each entity, which of the required components are present.
///
/// Last bit is used for the applicability filter.
///
/// In order of `required_components`.
/// If all bits are set, the entity is applicable to the visualizer.
// TODO(andreas): We could just limit the number of required components to 32 or 64 and
// then use a single u32/u64 as a bitmap.
required_component_and_filter_bitmap_per_entity: IntMap<EntityPathHash, BitVec>,
/// Which entities the visualizer can be applied to.
applicable_entities: ApplicableEntities,
/// List of all entities in this store that at some point in time had any of the indicator components.
///
/// Special case:
/// If the visualizer has no indicator components, this list will contain all entities in the store.
indicated_entities: IndicatedEntities,
}
impl VisualizerEntitySubscriber {
pub fn new<T: IdentifiedViewSystem + VisualizerSystem>(visualizer: &T) -> Self {
let visualizer_query_info = visualizer.visualizer_query_info();
Self {
visualizer: T::identifier(),
indicator_components: visualizer_query_info.indicators,
required_components_indices: visualizer_query_info
.required
.into_iter()
.enumerate()
.map(|(i, name)| (name, i))
.collect(),
per_store_mapping: Default::default(),
applicability_filter: visualizer
.applicability_filter()
.unwrap_or_else(|| Box::new(DefaultVisualizerApplicabilityFilter)),
}
}
/// List of entities that are applicable to the visualizer.
#[inline]
pub fn applicable_entities(&self, store: &StoreId) -> Option<&ApplicableEntities> {
self.per_store_mapping
.get(store)
.map(|mapping| &mapping.applicable_entities)
}
/// List of entities that at some point in time had any of the indicator components advertised by this visualizer.
///
/// Useful for quickly evaluating basic "should this visualizer apply by default"-heuristic.
/// Does *not* imply that any of the given entities is also in the applicable-set!
///
/// If the visualizer has no indicator components, this list will contain all entities in the store.
pub fn indicated_entities(&self, store: &StoreId) -> Option<&IndicatedEntities> {
self.per_store_mapping
.get(store)
.map(|mapping| &mapping.indicated_entities)
}
}
impl ChunkStoreSubscriber for VisualizerEntitySubscriber {
#[inline]
fn name(&self) -> String {
self.visualizer.as_str().to_owned()
}
#[inline]
fn as_any(&self) -> &dyn std::any::Any {
self
}
#[inline]
fn as_any_mut(&mut self) -> &mut dyn std::any::Any {
self
}
fn on_events(&mut self, events: &[ChunkStoreEvent]) {
re_tracing::profile_function!(self.visualizer);
// TODO(andreas): Need to react to store removals as well. As of writing doesn't exist yet.
for event in events {
if event.diff.kind != ChunkStoreDiffKind::Addition {
// Applicability is only additive, don't care about removals.
continue;
}
let store_mapping = self
.per_store_mapping
.entry(event.store_id.clone())
.or_default();
let entity_path = event.diff.chunk.entity_path();
// Update indicator component tracking:
if self.indicator_components.is_empty()
|| self.indicator_components.iter().any(|component_name| {
event.diff.chunk.components().contains_key(component_name)
})
{
store_mapping
.indicated_entities
.0
.insert(entity_path.clone());
}
// Update required component tracking:
let required_components_bitmap = store_mapping
.required_component_and_filter_bitmap_per_entity
.entry(entity_path.hash())
.or_insert_with(|| {
BitVec::from_elem(self.required_components_indices.len() + 1, false)
});
if required_components_bitmap.all() {
// We already know that this entity is applicable to the visualizer.
continue;
}
for (component_desc, list_array) in event.diff.chunk.components().iter_flattened() {
if let Some(index) = self
.required_components_indices
.get(&component_desc.component_name)
{
// The component might be present, but logged completely empty.
// That shouldn't count towards filling "having the required component present"!
// (Note: This happens frequently now with `Transform3D`'s component which always get logged, thus tripping of the `AxisLengthDetector`!)` )
if !list_array.values().is_empty() {
required_components_bitmap.set(*index, true);
}
}
}
let bit_index_for_filter = self.required_components_indices.len();
let custom_filter = required_components_bitmap[bit_index_for_filter];
if !custom_filter {
required_components_bitmap.set(
bit_index_for_filter,
self.applicability_filter.update_applicability(event),
);
}
if required_components_bitmap.all() {
re_log::trace!(
"Entity {:?} in store {:?} is now applicable to visualizer {:?}",
entity_path,
event.store_id,
self.visualizer
);
store_mapping
.applicable_entities
.0
.insert(entity_path.clone());
}
}
}
}