mirror of
https://github.com/salsa-rs/salsa.git
synced 2025-02-05 19:30:41 +00:00
127 lines
3.4 KiB
Rust
127 lines
3.4 KiB
Rust
use std::fmt;
|
|
|
|
use crate::{
|
|
cycle::CycleRecoveryStrategy,
|
|
ingredient::{fmt_index, Ingredient, IngredientRequiresReset},
|
|
key::{DatabaseKeyIndex, DependencyIndex},
|
|
runtime::{local_state::QueryOrigin, Runtime},
|
|
AsId, IngredientIndex, Revision,
|
|
};
|
|
|
|
pub trait InputId: AsId {}
|
|
impl<T: AsId> InputId for T {}
|
|
|
|
pub struct InputIngredient<Id>
|
|
where
|
|
Id: InputId,
|
|
{
|
|
ingredient_index: IngredientIndex,
|
|
counter: u32,
|
|
debug_name: &'static str,
|
|
_phantom: std::marker::PhantomData<Id>,
|
|
}
|
|
|
|
impl<Id> InputIngredient<Id>
|
|
where
|
|
Id: InputId,
|
|
{
|
|
pub fn new(index: IngredientIndex, debug_name: &'static str) -> Self {
|
|
Self {
|
|
ingredient_index: index,
|
|
counter: Default::default(),
|
|
debug_name,
|
|
_phantom: std::marker::PhantomData,
|
|
}
|
|
}
|
|
|
|
pub fn database_key_index(&self, id: Id) -> DatabaseKeyIndex {
|
|
DatabaseKeyIndex {
|
|
ingredient_index: self.ingredient_index,
|
|
key_index: id.as_id(),
|
|
}
|
|
}
|
|
|
|
pub fn new_input(&mut self, _runtime: &mut Runtime) -> Id {
|
|
let next_id = self.counter;
|
|
self.counter += 1;
|
|
Id::from_id(crate::Id::from_u32(next_id))
|
|
}
|
|
|
|
pub fn new_singleton_input(&mut self, _runtime: &mut Runtime) -> Id {
|
|
if self.counter >= 1 {
|
|
// already exists
|
|
Id::from_id(crate::Id::from_u32(self.counter - 1))
|
|
} else {
|
|
self.new_input(_runtime)
|
|
}
|
|
}
|
|
|
|
pub fn get_singleton_input(&self, _runtime: &Runtime) -> Option<Id> {
|
|
(self.counter > 0)
|
|
.then(|| Id::from_id(crate::Id::from_id(crate::Id::from_u32(self.counter - 1))))
|
|
}
|
|
}
|
|
|
|
impl<DB: ?Sized, Id> Ingredient<DB> for InputIngredient<Id>
|
|
where
|
|
Id: InputId,
|
|
{
|
|
fn maybe_changed_after(&self, _db: &DB, _input: DependencyIndex, _revision: Revision) -> bool {
|
|
// Input ingredients are just a counter, they store no data, they are immortal.
|
|
// Their *fields* are stored in function ingredients elsewhere.
|
|
false
|
|
}
|
|
|
|
fn cycle_recovery_strategy(&self) -> CycleRecoveryStrategy {
|
|
CycleRecoveryStrategy::Panic
|
|
}
|
|
|
|
fn origin(&self, _key_index: crate::Id) -> Option<QueryOrigin> {
|
|
None
|
|
}
|
|
|
|
fn mark_validated_output(
|
|
&self,
|
|
_db: &DB,
|
|
executor: DatabaseKeyIndex,
|
|
output_key: Option<crate::Id>,
|
|
) {
|
|
unreachable!(
|
|
"mark_validated_output({:?}, {:?}): input cannot be the output of a tracked function",
|
|
executor, output_key
|
|
);
|
|
}
|
|
|
|
fn remove_stale_output(
|
|
&self,
|
|
_db: &DB,
|
|
executor: DatabaseKeyIndex,
|
|
stale_output_key: Option<crate::Id>,
|
|
) {
|
|
unreachable!(
|
|
"remove_stale_output({:?}, {:?}): input cannot be the output of a tracked function",
|
|
executor, stale_output_key
|
|
);
|
|
}
|
|
|
|
fn reset_for_new_revision(&mut self) {
|
|
panic!("unexpected call to `reset_for_new_revision`")
|
|
}
|
|
|
|
fn salsa_struct_deleted(&self, _db: &DB, _id: crate::Id) {
|
|
panic!(
|
|
"unexpected call: input ingredients do not register for salsa struct deletion events"
|
|
);
|
|
}
|
|
|
|
fn fmt_index(&self, index: Option<crate::Id>, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
fmt_index(self.debug_name, index, fmt)
|
|
}
|
|
}
|
|
|
|
impl<Id> IngredientRequiresReset for InputIngredient<Id>
|
|
where
|
|
Id: InputId,
|
|
{
|
|
const RESET_ON_NEW_REVISION: bool = false;
|
|
}
|