salsa 0.26.1

A generic framework for on-demand, incrementalized computation (experimental)
Documentation
use std::fmt::Debug;
use std::hash::{Hash, Hasher};
use std::num::NonZeroU32;

use crate::zalsa::Zalsa;

/// The `Id` of a salsa struct in the database [`Table`](`crate::table::Table`).
///
/// The high-order bits of an `Id` store a 32-bit generation counter, while
/// the low-order bits pack a [`PageIndex`](`crate::table::PageIndex`) and
/// [`SlotIndex`](`crate::table::SlotIndex`) within the page.
///
/// The low-order bits of `Id` are a `u32` ranging from `0..Id::MAX_U32`.
/// The maximum range is smaller than a standard `u32` to leave
/// room for niches; currently there is only one niche, so that
/// `Option<Id>` is the same size as an `Id`.
///
/// As an end-user of `Salsa` you will generally not use `Id` directly,
/// it is wrapped in new types.
#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
pub struct Id {
    index: NonZeroU32,
    generation: u32,
}

impl Id {
    pub const MAX_U32: u32 = u32::MAX - 0xFF;
    pub const MAX_USIZE: usize = Self::MAX_U32 as usize;

    /// Create a `salsa::Id` from a u32 value, without a generation. This
    /// value should be less than [`Self::MAX_U32`].
    ///
    /// In general, you should not need to create salsa ids yourself,
    /// but it can be useful if you are using the type as a general
    /// purpose "identifier" internally.
    ///
    /// # Safety
    ///
    /// The supplied value must be less than [`Self::MAX_U32`].
    #[doc(hidden)]
    #[track_caller]
    #[inline]
    pub const unsafe fn from_index(index: u32) -> Self {
        debug_assert!(index < Self::MAX_U32);

        Id {
            // SAFETY: Caller obligation.
            index: unsafe { NonZeroU32::new_unchecked(index + 1) },
            generation: 0,
        }
    }

    /// Create a `salsa::Id` from a `u64` value.
    ///
    /// This should only be used to recreate an `Id` together with `Id::as_u64`.
    ///
    /// # Safety
    ///
    /// The data bits of the supplied value must represent a valid `Id` returned
    /// by `Id::as_u64`.
    #[doc(hidden)]
    #[track_caller]
    #[inline]
    pub const unsafe fn from_bits_unchecked(bits: u64) -> Self {
        // SAFETY: Caller obligation.
        let index = unsafe { NonZeroU32::new_unchecked(bits as u32) };
        let generation = (bits >> 32) as u32;

        Id { index, generation }
    }

    /// Create a `salsa::Id` from a `u64` value.
    ///
    /// This should only be used to recreate an `Id` together with `Id::as_u64`,
    /// and may panic if the `Id` is invalid.
    #[inline]
    #[doc(hidden)]
    pub const fn from_bits(bits: u64) -> Self {
        let index = NonZeroU32::new(bits as u32).expect("attempted to create invalid `Id`");
        let generation = (bits >> 32) as u32;

        Id { index, generation }
    }

    /// Return a `u64` representation of this `Id`.
    #[inline]
    pub fn as_bits(self) -> u64 {
        u64::from(self.index.get()) | (u64::from(self.generation) << 32)
    }

    /// Returns a new `Id` with same index, but the generation incremented by one.
    ///
    /// Returns `None` if the generation would overflow, i.e. the current generation
    /// is `u32::MAX`.
    #[inline]
    pub fn next_generation(self) -> Option<Id> {
        self.generation()
            .checked_add(1)
            .map(|generation| self.with_generation(generation))
    }

    /// Mark the `Id` with a generation.
    ///
    /// This `Id` will refer to the same page and slot in the database,
    /// but will differ from other identifiers of the slot based on the
    /// provided generation.
    #[inline]
    pub const fn with_generation(self, generation: u32) -> Id {
        Id {
            index: self.index,
            generation,
        }
    }

    /// Return the index portion of this `Id`.
    #[inline]
    pub const fn index(self) -> u32 {
        self.index.get() - 1
    }

    /// Return the generation of this `Id`.
    #[inline]
    pub const fn generation(self) -> u32 {
        self.generation
    }
}

impl Hash for Id {
    fn hash<H: Hasher>(&self, state: &mut H) {
        // Convert to a `u64` to avoid dispatching multiple calls to `H::write`.
        state.write_u64(self.as_bits());
    }
}

#[cfg(feature = "persistence")]
impl serde::Serialize for Id {
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: serde::Serializer,
    {
        serde::Serialize::serialize(&self.as_bits(), serializer)
    }
}

#[cfg(feature = "persistence")]
impl<'de> serde::Deserialize<'de> for Id {
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
    where
        D: serde::Deserializer<'de>,
    {
        serde::Deserialize::deserialize(deserializer).map(Self::from_bits)
    }
}

impl Debug for Id {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        if self.generation() == 0 {
            write!(f, "Id({:x})", self.index())
        } else {
            write!(f, "Id({:x}g{:x})", self.index(), self.generation())
        }
    }
}

/// Internal salsa trait for types that can be represented as a salsa id.
pub trait AsId: Sized {
    fn as_id(&self) -> Id;
}

/// Internal Salsa trait for types that are just a newtype'd [`Id`][].
pub trait FromId {
    fn from_id(id: Id) -> Self;
}

impl AsId for Id {
    #[inline]
    fn as_id(&self) -> Id {
        *self
    }
}

impl FromId for Id {
    #[inline]
    fn from_id(id: Id) -> Self {
        id
    }
}

/// Enums cannot use [`FromId`] because they need access to the DB to tell the `TypeId` of the variant,
/// so they use this trait instead, that has a blanket implementation for `FromId`.
pub trait FromIdWithDb {
    fn from_id(id: Id, zalsa: &Zalsa) -> Self;
}

impl<T: FromId> FromIdWithDb for T {
    #[inline]
    fn from_id(id: Id, _zalsa: &Zalsa) -> Self {
        FromId::from_id(id)
    }
}