return a pointer from interning, not just id

This commit is contained in:
Niko Matsakis 2024-05-15 05:37:34 -04:00
parent d92f2aa0a5
commit 5095d79d13
3 changed files with 81 additions and 49 deletions

View file

@ -82,7 +82,7 @@ impl InternedStruct {
fn validate_interned(&self) -> syn::Result<()> { fn validate_interned(&self) -> syn::Result<()> {
self.disallow_id_fields("interned")?; self.disallow_id_fields("interned")?;
self.require_db_lifetime()?; self.require_no_generics()?;
Ok(()) Ok(())
} }
@ -160,17 +160,17 @@ impl InternedStruct {
if field.is_clone_field() { if field.is_clone_field() {
parse_quote_spanned! { field_get_name.span() => parse_quote_spanned! { field_get_name.span() =>
#field_vis fn #field_get_name(self, db: &#db_dyn_ty) -> #field_ty { #field_vis fn #field_get_name(self, db: &#db_dyn_ty) -> #field_ty {
let (jar, runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(db); let (jar, _runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(db);
let ingredients = <#jar_ty as salsa::storage::HasIngredientsFor< #the_ident #type_generics >>::ingredient(jar); let ingredients = <#jar_ty as salsa::storage::HasIngredientsFor< #the_ident #type_generics >>::ingredient(jar);
std::clone::Clone::clone(&ingredients.data(runtime, self.0).#field_name) std::clone::Clone::clone(&ingredients.data(self.0).#field_name)
} }
} }
} else { } else {
parse_quote_spanned! { field_get_name.span() => parse_quote_spanned! { field_get_name.span() =>
#field_vis fn #field_get_name<'db>(self, db: &'db #db_dyn_ty) -> &'db #field_ty { #field_vis fn #field_get_name<'db>(self, db: &'db #db_dyn_ty) -> &'db #field_ty {
let (jar, runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(db); let (jar, _runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(db);
let ingredients = <#jar_ty as salsa::storage::HasIngredientsFor< #the_ident #type_generics >>::ingredient(jar); let ingredients = <#jar_ty as salsa::storage::HasIngredientsFor< #the_ident #type_generics >>::ingredient(jar);
&ingredients.data(runtime, self.0).#field_name &ingredients.data(self.0).#field_name
} }
} }
} }
@ -190,7 +190,7 @@ impl InternedStruct {
let ingredients = <#jar_ty as salsa::storage::HasIngredientsFor< #the_ident #type_generics >>::ingredient(jar); let ingredients = <#jar_ty as salsa::storage::HasIngredientsFor< #the_ident #type_generics >>::ingredient(jar);
Self(ingredients.intern(runtime, #data_ident { Self(ingredients.intern(runtime, #data_ident {
#(#field_names,)* #(#field_names,)*
})) }).salsa_id())
} }
}; };

View file

@ -428,7 +428,7 @@ fn fn_configuration(args: &FnArgs, item_fn: &syn::ItemFn) -> Configuration {
let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db); let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db);
let __ingredients = let __ingredients =
<_ as salsa::storage::HasIngredientsFor<#fn_ty>>::ingredient(__jar); <_ as salsa::storage::HasIngredientsFor<#fn_ty>>::ingredient(__jar);
let #key_var = __ingredients.intern_map.data(__runtime, __id).clone(); let #key_var = __ingredients.intern_map.data(__id).clone();
#recovery_fn(__db, __cycle, #key_splat) #recovery_fn(__db, __cycle, #key_splat)
} }
}; };
@ -460,7 +460,7 @@ fn fn_configuration(args: &FnArgs, item_fn: &syn::ItemFn) -> Configuration {
let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db); let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(__db);
let __ingredients = let __ingredients =
<_ as salsa::storage::HasIngredientsFor<#fn_ty>>::ingredient(__jar); <_ as salsa::storage::HasIngredientsFor<#fn_ty>>::ingredient(__jar);
let #key_var = __ingredients.intern_map.data(__runtime, __id).clone(); let #key_var = __ingredients.intern_map.data(__id).clone();
#inner_fn_name(__db, #key_splat) #inner_fn_name(__db, #key_splat)
} }
}; };
@ -652,7 +652,7 @@ fn ref_getter_fn(
{ {
let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(#db_var); let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(#db_var);
let __ingredients = <_ as salsa::storage::HasIngredientsFor<#config_ty>>::ingredient(__jar); let __ingredients = <_ as salsa::storage::HasIngredientsFor<#config_ty>>::ingredient(__jar);
let __key = __ingredients.intern_map.intern(__runtime, (#(#arg_names),*)); let __key = __ingredients.intern_map.intern_id(__runtime, (#(#arg_names),*));
__ingredients.function.fetch(#db_var, __key) __ingredients.function.fetch(#db_var, __key)
} }
}; };
@ -696,7 +696,7 @@ fn setter_fn(
{ {
let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar_mut(#db_var); let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar_mut(#db_var);
let __ingredients = <_ as salsa::storage::HasIngredientsFor<#config_ty>>::ingredient_mut(__jar); let __ingredients = <_ as salsa::storage::HasIngredientsFor<#config_ty>>::ingredient_mut(__jar);
let __key = __ingredients.intern_map.intern(__runtime, (#(#arg_names),*)); let __key = __ingredients.intern_map.intern_id(__runtime, (#(#arg_names),*));
__ingredients.function.store(__runtime, __key, #value_arg, salsa::Durability::LOW) __ingredients.function.store(__runtime, __key, #value_arg, salsa::Durability::LOW)
} }
}, },
@ -765,7 +765,7 @@ fn specify_fn(
let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(#db_var); let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(#db_var);
let __ingredients = <_ as salsa::storage::HasIngredientsFor<#config_ty>>::ingredient(__jar); let __ingredients = <_ as salsa::storage::HasIngredientsFor<#config_ty>>::ingredient(__jar);
let __key = __ingredients.intern_map.intern(__runtime, (#(#arg_names),*)); let __key = __ingredients.intern_map.intern_id(__runtime, (#(#arg_names),*));
__ingredients.function.specify_and_record(#db_var, __key, #value_arg) __ingredients.function.specify_and_record(#db_var, __key, #value_arg)
} }
}, },
@ -877,7 +877,7 @@ fn accumulated_fn(
{ {
let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(#db_var); let (__jar, __runtime) = <_ as salsa::storage::HasJar<#jar_ty>>::jar(#db_var);
let __ingredients = <_ as salsa::storage::HasIngredientsFor<#config_ty>>::ingredient(__jar); let __ingredients = <_ as salsa::storage::HasIngredientsFor<#config_ty>>::ingredient(__jar);
let __key = __ingredients.intern_map.intern(__runtime, (#(#arg_names),*)); let __key = __ingredients.intern_map.intern_id(__runtime, (#(#arg_names),*));
__ingredients.function.accumulated::<__A>(#db_var, __key) __ingredients.function.accumulated::<__A>(#db_var, __key)
} }
}; };

View file

@ -7,6 +7,7 @@ use crate::durability::Durability;
use crate::id::AsId; use crate::id::AsId;
use crate::ingredient::{fmt_index, IngredientRequiresReset}; use crate::ingredient::{fmt_index, IngredientRequiresReset};
use crate::key::DependencyIndex; use crate::key::DependencyIndex;
use crate::plumbing::transmute_lifetime;
use crate::runtime::local_state::QueryOrigin; use crate::runtime::local_state::QueryOrigin;
use crate::runtime::Runtime; use crate::runtime::Runtime;
use crate::{DatabaseKeyIndex, Id}; use crate::{DatabaseKeyIndex, Id};
@ -38,7 +39,7 @@ pub struct InternedIngredient<C: Configuration> {
/// Maps from an interned id to its data. /// Maps from an interned id to its data.
/// ///
/// Deadlock requirement: We access `value_map` while holding lock on `key_map`, but not vice versa. /// Deadlock requirement: We access `value_map` while holding lock on `key_map`, but not vice versa.
value_map: FxDashMap<Id, Box<C::Data<'static>>>, value_map: FxDashMap<Id, Box<InternedValue<C>>>,
/// counter for the next id. /// counter for the next id.
counter: AtomicCell<u32>, counter: AtomicCell<u32>,
@ -52,6 +53,15 @@ pub struct InternedIngredient<C: Configuration> {
debug_name: &'static str, debug_name: &'static str,
} }
/// Struct storing the interned fields.
pub struct InternedValue<C>
where
C: Configuration,
{
id: Id,
fields: C::Data<'static>,
}
impl<C> InternedIngredient<C> impl<C> InternedIngredient<C>
where where
C: Configuration, C: Configuration,
@ -71,12 +81,16 @@ where
unsafe { std::mem::transmute(data) } unsafe { std::mem::transmute(data) }
} }
unsafe fn from_internal_data<'db>(&'db self, data: &C::Data<'static>) -> &'db C::Data<'db> { pub fn intern_id<'db>(&'db self, runtime: &'db Runtime, data: C::Data<'db>) -> crate::Id {
unsafe { std::mem::transmute(data) } self.intern(runtime, data).salsa_id()
} }
/// Intern data to a unique id. /// Intern data to a unique reference.
pub fn intern<'db>(&'db self, runtime: &'db Runtime, data: C::Data<'db>) -> Id { pub fn intern<'db>(
&'db self,
runtime: &'db Runtime,
data: C::Data<'db>,
) -> &'db InternedValue<C> {
runtime.report_tracked_read( runtime.report_tracked_read(
DependencyIndex::for_table(self.ingredient_index), DependencyIndex::for_table(self.ingredient_index),
Durability::MAX, Durability::MAX,
@ -86,56 +100,57 @@ where
// Optimisation to only get read lock on the map if the data has already // Optimisation to only get read lock on the map if the data has already
// been interned. // been interned.
let internal_data = unsafe { self.to_internal_data(data) }; let internal_data = unsafe { self.to_internal_data(data) };
if let Some(id) = self.key_map.get(&internal_data) { if let Some(guard) = self.key_map.get(&internal_data) {
return *id; let id = *guard;
drop(guard);
return self.interned_value(id);
} }
match self.key_map.entry(internal_data.clone()) { match self.key_map.entry(internal_data.clone()) {
// Data has been interned by a racing call, use that ID instead // Data has been interned by a racing call, use that ID instead
dashmap::mapref::entry::Entry::Occupied(entry) => *entry.get(), dashmap::mapref::entry::Entry::Occupied(entry) => {
let id = *entry.get();
drop(entry);
self.interned_value(id)
}
// We won any races so should intern the data // We won any races so should intern the data
dashmap::mapref::entry::Entry::Vacant(entry) => { dashmap::mapref::entry::Entry::Vacant(entry) => {
let next_id = self.counter.fetch_add(1); let next_id = self.counter.fetch_add(1);
let next_id = Id::from_id(crate::id::Id::from_u32(next_id)); let next_id = Id::from_id(crate::id::Id::from_u32(next_id));
let old_value = self.value_map.insert(next_id, Box::new(internal_data)); let value = self
assert!( .value_map
old_value.is_none(), .entry(next_id)
"next_id is guaranteed to be unique, bar overflow" .or_insert(Box::new(InternedValue {
); id: next_id,
fields: internal_data,
}));
// SAFETY: Items are only removed from the `value_map` with an `&mut self` reference.
let value_ref = unsafe { transmute_lifetime(self, &**value) };
drop(value);
entry.insert(next_id); entry.insert(next_id);
next_id value_ref
} }
} }
} }
pub fn interned_value<'db>(&'db self, id: Id) -> &'db InternedValue<C> {
let r = self.value_map.get(&id).unwrap();
// SAFETY: Items are only removed from the `value_map` with an `&mut self` reference.
unsafe { transmute_lifetime(self, &**r) }
}
pub fn data<'db>(&'db self, id: Id) -> &'db C::Data<'db> {
self.interned_value(id).data()
}
pub fn reset(&mut self, revision: Revision) { pub fn reset(&mut self, revision: Revision) {
assert!(revision > self.reset_at); assert!(revision > self.reset_at);
self.reset_at = revision; self.reset_at = revision;
self.key_map.clear(); self.key_map.clear();
self.value_map.clear(); self.value_map.clear();
} }
#[track_caller]
pub fn data<'db>(&'db self, runtime: &'db Runtime, id: Id) -> &'db C::Data<'db> {
runtime.report_tracked_read(
DependencyIndex::for_table(self.ingredient_index),
Durability::MAX,
self.reset_at,
);
let data = match self.value_map.get(&id) {
Some(d) => d,
None => {
panic!("no data found for id `{:?}`", id)
}
};
// Unsafety clause:
//
// * Values are only removed or altered when we have `&mut self`
unsafe { self.from_internal_data(&data) }
}
} }
impl<DB: ?Sized, C> Ingredient<DB> for InternedIngredient<C> impl<DB: ?Sized, C> Ingredient<DB> for InternedIngredient<C>
@ -223,11 +238,28 @@ where
IdentityInterner { data: PhantomData } IdentityInterner { data: PhantomData }
} }
pub fn intern<'db>(&'db self, _runtime: &'db Runtime, id: C::Data<'db>) -> crate::Id { pub fn intern_id<'db>(&'db self, _runtime: &'db Runtime, id: C::Data<'db>) -> crate::Id {
id.as_id() id.as_id()
} }
pub fn data<'db>(&'db self, _runtime: &'db Runtime, id: crate::Id) -> C::Data<'db> { pub fn data<'db>(&'db self, id: crate::Id) -> C::Data<'db> {
<C::Data<'db>>::from_id(id) <C::Data<'db>>::from_id(id)
} }
} }
impl<C> InternedValue<C>
where
C: Configuration,
{
pub fn salsa_id(&self) -> Id {
self.id
}
pub fn data<'db>(&'db self) -> &'db C::Data<'db> {
unsafe { self.to_self_ref(&self.fields) }
}
unsafe fn to_self_ref<'db>(&'db self, fields: &'db C::Data<'static>) -> &'db C::Data<'db> {
unsafe { std::mem::transmute(fields) }
}
}