use rustc_data_structures::stable_hasher::{HashStable, StableHasher, ToStableHashKey}; use crate::hir::{ AttributeMap, BodyId, Crate, ForeignItemId, ImplItemId, ItemId, OwnerNodes, TraitItemId, }; use crate::hir_id::{HirId, ItemLocalId}; use rustc_span::def_id::DefPathHash; /// Requirements for a `StableHashingContext` to be used in this crate. /// This is a hack to allow using the `HashStable_Generic` derive macro /// instead of implementing everything in `rustc_middle`. pub trait HashStableContext: rustc_ast::HashStableContext + rustc_target::HashStableContext { fn hash_body_id(&mut self, _: BodyId, hasher: &mut StableHasher); } impl ToStableHashKey for HirId { type KeyType = (DefPathHash, ItemLocalId); #[inline] fn to_stable_hash_key(&self, hcx: &HirCtx) -> (DefPathHash, ItemLocalId) { let def_path_hash = self.owner.def_id.to_stable_hash_key(hcx); (def_path_hash, self.local_id) } } impl ToStableHashKey for ItemLocalId { type KeyType = ItemLocalId; #[inline] fn to_stable_hash_key(&self, _: &HirCtx) -> ItemLocalId { *self } } impl ToStableHashKey for BodyId { type KeyType = (DefPathHash, ItemLocalId); #[inline] fn to_stable_hash_key(&self, hcx: &HirCtx) -> (DefPathHash, ItemLocalId) { let BodyId { hir_id } = *self; hir_id.to_stable_hash_key(hcx) } } impl ToStableHashKey for ItemId { type KeyType = DefPathHash; #[inline] fn to_stable_hash_key(&self, hcx: &HirCtx) -> DefPathHash { self.owner_id.def_id.to_stable_hash_key(hcx) } } impl ToStableHashKey for TraitItemId { type KeyType = DefPathHash; #[inline] fn to_stable_hash_key(&self, hcx: &HirCtx) -> DefPathHash { self.owner_id.def_id.to_stable_hash_key(hcx) } } impl ToStableHashKey for ImplItemId { type KeyType = DefPathHash; #[inline] fn to_stable_hash_key(&self, hcx: &HirCtx) -> DefPathHash { self.owner_id.def_id.to_stable_hash_key(hcx) } } impl ToStableHashKey for ForeignItemId { type KeyType = DefPathHash; #[inline] fn to_stable_hash_key(&self, hcx: &HirCtx) -> DefPathHash { self.owner_id.def_id.to_stable_hash_key(hcx) } } impl HashStable for BodyId { fn hash_stable(&self, hcx: &mut HirCtx, hasher: &mut StableHasher) { hcx.hash_body_id(*self, hasher) } } // The following implementations of HashStable for `ItemId`, `TraitItemId`, and // `ImplItemId` deserve special attention. Normally we do not hash `NodeId`s within // the HIR, since they just signify a HIR nodes own path. But `ItemId` et al // are used when another item in the HIR is *referenced* and we certainly // want to pick up on a reference changing its target, so we hash the NodeIds // in "DefPath Mode". impl<'tcx, HirCtx: crate::HashStableContext> HashStable for OwnerNodes<'tcx> { fn hash_stable(&self, hcx: &mut HirCtx, hasher: &mut StableHasher) { // We ignore the `nodes` and `bodies` fields since these refer to information included in // `hash` which is hashed in the collector and used for the crate hash. // `local_id_to_def_id` is also ignored because is dependent on the body, then just hashing // the body satisfies the condition of two nodes being different have different // `hash_stable` results. let OwnerNodes { hash_including_bodies, hash_without_bodies: _, nodes: _, bodies: _, local_id_to_def_id: _, } = *self; hash_including_bodies.hash_stable(hcx, hasher); } } impl<'tcx, HirCtx: crate::HashStableContext> HashStable for AttributeMap<'tcx> { fn hash_stable(&self, hcx: &mut HirCtx, hasher: &mut StableHasher) { // We ignore the `map` since it refers to information included in `hash` which is hashed in // the collector and used for the crate hash. let AttributeMap { hash, map: _ } = *self; hash.hash_stable(hcx, hasher); } } impl HashStable for Crate<'_> { fn hash_stable(&self, hcx: &mut HirCtx, hasher: &mut StableHasher) { let Crate { owners: _, hir_hash } = self; hir_hash.hash_stable(hcx, hasher) } }