diff --git a/compiler/rustc_const_eval/src/interpret/place.rs b/compiler/rustc_const_eval/src/interpret/place.rs index 81b926a1b65fa..a1c57b2c32d91 100644 --- a/compiler/rustc_const_eval/src/interpret/place.rs +++ b/compiler/rustc_const_eval/src/interpret/place.rs @@ -1021,6 +1021,25 @@ where str: &str, kind: MemoryKind, mutbl: Mutability, + ) -> InterpResult<'tcx, MPlaceTy<'tcx, M::Provenance>> { + self.allocate_str_inner(str, kind, mutbl, false) + } + + pub fn allocate_str_with_null( + &mut self, + str: &str, + kind: MemoryKind, + mutbl: Mutability, + ) -> InterpResult<'tcx, MPlaceTy<'tcx, M::Provenance>> { + self.allocate_str_inner(str, kind, mutbl, true) + } + + fn allocate_str_inner( + &mut self, + str: &str, + kind: MemoryKind, + mutbl: Mutability, + null_terminate: bool, ) -> InterpResult<'tcx, MPlaceTy<'tcx, M::Provenance>> { let tcx = self.tcx.tcx; @@ -1028,7 +1047,11 @@ where let ptr = if mutbl.is_not() { // Use dedup'd allocation function. let salt = M::get_global_alloc_salt(self, None); - let id = tcx.allocate_bytes_dedup(str.as_bytes(), salt); + let id = if null_terminate { + tcx.allocate_bytes_dedup_with_null(str.as_bytes(), salt) + } else { + tcx.allocate_bytes_dedup(str.as_bytes(), salt) + }; // Turn untagged "global" pointers (obtained via `tcx`) into the machine pointer to the allocation. M::adjust_alloc_root_pointer(&self, Pointer::from(id), Some(kind))? diff --git a/compiler/rustc_const_eval/src/util/caller_location.rs b/compiler/rustc_const_eval/src/util/caller_location.rs index 7f4c36835e449..c8b4cd16c8ea6 100644 --- a/compiler/rustc_const_eval/src/util/caller_location.rs +++ b/compiler/rustc_const_eval/src/util/caller_location.rs @@ -19,14 +19,9 @@ fn alloc_caller_location<'tcx>( let loc_details = ecx.tcx.sess.opts.unstable_opts.location_detail; // This can fail if rustc runs out of memory right here. Trying to emit an error would be // pointless, since that would require allocating more memory than these short strings. - let file = if loc_details.file { - ecx.allocate_str(filename.as_str(), MemoryKind::CallerLocation, Mutability::Not).unwrap() - } else { - // FIXME: This creates a new allocation each time. It might be preferable to - // perform this allocation only once, and re-use the `MPlaceTy`. - // See https://github.com/rust-lang/rust/pull/89920#discussion_r730012398 - ecx.allocate_str("", MemoryKind::CallerLocation, Mutability::Not).unwrap() - }; + let filename = if loc_details.file { filename.as_str() } else { "" }; + let file = + ecx.allocate_str_with_null(filename, MemoryKind::CallerLocation, Mutability::Not).unwrap(); let file = file.map_provenance(CtfeProvenance::as_immutable); let line = if loc_details.line { Scalar::from_u32(line) } else { Scalar::from_u32(0) }; let col = if loc_details.column { Scalar::from_u32(col) } else { Scalar::from_u32(0) }; diff --git a/compiler/rustc_middle/src/mir/interpret/allocation.rs b/compiler/rustc_middle/src/mir/interpret/allocation.rs index ac3baf74ca7c4..44400e21cc237 100644 --- a/compiler/rustc_middle/src/mir/interpret/allocation.rs +++ b/compiler/rustc_middle/src/mir/interpret/allocation.rs @@ -294,6 +294,24 @@ impl Allocation { Allocation::from_bytes(slice, Align::ONE, Mutability::Not) } + pub fn from_bytes_byte_aligned_immutable_with_null<'a>( + slice: impl Into>, + ) -> Self { + let slice = slice.into(); + let len = slice.len() + 1; + let size = Size::from_bytes(len); + let mut bytes = Bytes::zeroed(size, Align::ONE).unwrap(); + bytes[..slice.len()].copy_from_slice(&slice); + Self { + bytes, + provenance: ProvenanceMap::new(), + init_mask: InitMask::new(size, true), + align: Align::ONE, + mutability: Mutability::Not, + extra: (), + } + } + fn uninit_inner(size: Size, align: Align, fail: impl FnOnce() -> R) -> Result { // We raise an error if we cannot create the allocation on the host. // This results in an error that can happen non-deterministically, since the memory diff --git a/compiler/rustc_middle/src/ty/context.rs b/compiler/rustc_middle/src/ty/context.rs index b682524ae39f4..421d37480b881 100644 --- a/compiler/rustc_middle/src/ty/context.rs +++ b/compiler/rustc_middle/src/ty/context.rs @@ -1462,6 +1462,13 @@ impl<'tcx> TyCtxt<'tcx> { self.reserve_and_set_memory_dedup(alloc, salt) } + pub fn allocate_bytes_dedup_with_null(self, bytes: &[u8], salt: usize) -> interpret::AllocId { + // Create an allocation that just contains these bytes. + let alloc = interpret::Allocation::from_bytes_byte_aligned_immutable_with_null(bytes); + let alloc = self.mk_const_alloc(alloc); + self.reserve_and_set_memory_dedup(alloc, salt) + } + /// Returns a range of the start/end indices specified with the /// `rustc_layout_scalar_valid_range` attribute. // FIXME(eddyb) this is an awkward spot for this method, maybe move it?