diff options
| author | Tyson Nottingham <tgnottingham@gmail.com> | 2020-11-03 22:23:08 -0800 |
|---|---|---|
| committer | Tyson Nottingham <tgnottingham@gmail.com> | 2020-11-18 12:49:09 -0800 |
| commit | f09d474836d14811af1f9ebead4c1649c54e4e4c (patch) | |
| tree | ddf66a7bde0377b6618015c6090a0818577f608d /compiler/rustc_data_structures/src | |
| parent | 8d2d0014922e9f541694bfe87642749239990e0e (diff) | |
| download | rust-f09d474836d14811af1f9ebead4c1649c54e4e4c.tar.gz rust-f09d474836d14811af1f9ebead4c1649c54e4e4c.zip | |
Use PackedFingerprint in DepNode to reduce memory consumption
Diffstat (limited to 'compiler/rustc_data_structures/src')
| -rw-r--r-- | compiler/rustc_data_structures/src/fingerprint.rs | 42 |
1 files changed, 42 insertions, 0 deletions
diff --git a/compiler/rustc_data_structures/src/fingerprint.rs b/compiler/rustc_data_structures/src/fingerprint.rs index ec2f9597b18..9927511e3c1 100644 --- a/compiler/rustc_data_structures/src/fingerprint.rs +++ b/compiler/rustc_data_structures/src/fingerprint.rs @@ -151,8 +151,50 @@ impl<D: rustc_serialize::Decoder> FingerprintDecoder for D { panic!("Cannot decode `Fingerprint` with `{}`", std::any::type_name::<D>()); } } + impl FingerprintDecoder for opaque::Decoder<'_> { fn decode_fingerprint(&mut self) -> Result<Fingerprint, String> { Fingerprint::decode_opaque(self) } } + +// `PackedFingerprint` wraps a `Fingerprint`. Its purpose is to, on certain +// architectures, behave like a `Fingerprint` without alignment requirements. +// This behavior is only enabled on x86 and x86_64, where the impact of +// unaligned accesses is tolerable in small doses. +// +// This may be preferable to use in large collections of structs containing +// fingerprints, as it can reduce memory consumption by preventing the padding +// that the more strictly-aligned `Fingerprint` can introduce. An application of +// this is in the query dependency graph, which contains a large collection of +// `DepNode`s. As of this writing, the size of a `DepNode` decreases by ~30% +// (from 24 bytes to 17) by using the packed representation here, which +// noticeably decreases total memory usage when compiling large crates. +#[cfg_attr(any(target_arch = "x86", target_arch = "x86_64"), repr(packed))] +#[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Copy, Hash)] +pub struct PackedFingerprint(pub Fingerprint); + +impl std::fmt::Display for PackedFingerprint { + #[inline] + fn fmt(&self, formatter: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + // Copy to avoid taking reference to packed field. + let copy = self.0; + copy.fmt(formatter) + } +} + +impl<E: rustc_serialize::Encoder> Encodable<E> for PackedFingerprint { + #[inline] + fn encode(&self, s: &mut E) -> Result<(), E::Error> { + // Copy to avoid taking reference to packed field. + let copy = self.0; + copy.encode(s) + } +} + +impl<D: rustc_serialize::Decoder> Decodable<D> for PackedFingerprint { + #[inline] + fn decode(d: &mut D) -> Result<Self, D::Error> { + Fingerprint::decode(d).map(|f| PackedFingerprint(f)) + } +} |
