Skip to content

Commit d6bc508

Browse files
authored
Merge pull request rust-lang#122 from oli-obk/master
prevent more deallocations of statics
2 parents 4f3fc85 + 4beb774 commit d6bc508

File tree

6 files changed

+90
-12
lines changed

6 files changed

+90
-12
lines changed

src/eval_context.rs

+15-7
Original file line numberDiff line numberDiff line change
@@ -172,7 +172,7 @@ impl<'a, 'tcx> EvalContext<'a, 'tcx> {
172172
// FIXME: cache these allocs
173173
let ptr = self.memory.allocate(s.len() as u64, 1)?;
174174
self.memory.write_bytes(ptr, s.as_bytes())?;
175-
self.memory.mark_static(ptr.alloc_id, false)?;
175+
self.memory.mark_static_initalized(ptr.alloc_id, false)?;
176176
Ok(Value::ByValPair(PrimVal::Ptr(ptr), PrimVal::from_u128(s.len() as u128)))
177177
}
178178

@@ -194,9 +194,10 @@ impl<'a, 'tcx> EvalContext<'a, 'tcx> {
194194
Str(ref s) => return self.str_to_value(s),
195195

196196
ByteStr(ref bs) => {
197+
// FIXME: cache these allocs
197198
let ptr = self.memory.allocate(bs.len() as u64, 1)?;
198199
self.memory.write_bytes(ptr, bs)?;
199-
self.memory.mark_static(ptr.alloc_id, false)?;
200+
self.memory.mark_static_initalized(ptr.alloc_id, false)?;
200201
PrimVal::Ptr(ptr)
201202
}
202203

@@ -316,19 +317,22 @@ impl<'a, 'tcx> EvalContext<'a, 'tcx> {
316317
let global_value = self.globals.get_mut(&id)
317318
.expect("global should have been cached (static)");
318319
match global_value.value {
319-
Value::ByRef(ptr) => self.memory.mark_static(ptr.alloc_id, mutable)?,
320+
Value::ByRef(ptr) => self.memory.mark_static_initalized(ptr.alloc_id, mutable)?,
320321
Value::ByVal(val) => if let PrimVal::Ptr(ptr) = val {
321-
self.memory.mark_static(ptr.alloc_id, mutable)?;
322+
self.memory.mark_static_initalized(ptr.alloc_id, mutable)?;
322323
},
323324
Value::ByValPair(val1, val2) => {
324325
if let PrimVal::Ptr(ptr) = val1 {
325-
self.memory.mark_static(ptr.alloc_id, mutable)?;
326+
self.memory.mark_static_initalized(ptr.alloc_id, mutable)?;
326327
}
327328
if let PrimVal::Ptr(ptr) = val2 {
328-
self.memory.mark_static(ptr.alloc_id, mutable)?;
329+
self.memory.mark_static_initalized(ptr.alloc_id, mutable)?;
329330
}
330331
},
331332
}
333+
// see comment on `initialized` field
334+
assert!(!global_value.initialized);
335+
global_value.initialized = true;
332336
assert!(global_value.mutable);
333337
global_value.mutable = mutable;
334338
} else {
@@ -867,8 +871,12 @@ impl<'a, 'tcx> EvalContext<'a, 'tcx> {
867871
Value::ByRef(ptr) => Lvalue::from_ptr(ptr),
868872
_ => {
869873
let ptr = self.alloc_ptr_with_substs(global_val.ty, cid.substs)?;
874+
self.memory.mark_static(ptr.alloc_id);
870875
self.write_value_to_ptr(global_val.value, ptr, global_val.ty)?;
871-
self.memory.mark_static(ptr.alloc_id, global_val.mutable)?;
876+
// see comment on `initialized` field
877+
if global_val.initialized {
878+
self.memory.mark_static_initalized(ptr.alloc_id, global_val.mutable)?;
879+
}
872880
let lval = self.globals.get_mut(&cid).expect("already checked");
873881
*lval = Global {
874882
value: Value::ByRef(ptr),

src/lvalue.rs

+6
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,11 @@ pub struct GlobalId<'tcx> {
5757
#[derive(Copy, Clone, Debug)]
5858
pub struct Global<'tcx> {
5959
pub(super) value: Value,
60+
/// Only used in `force_allocation` to ensure we don't mark the memory
61+
/// before the static is initialized. It is possible to convert a
62+
/// global which initially is `Value::ByVal(PrimVal::Undef)` and gets
63+
/// lifted to an allocation before the static is fully initialized
64+
pub(super) initialized: bool,
6065
pub(super) mutable: bool,
6166
pub(super) ty: Ty<'tcx>,
6267
}
@@ -102,6 +107,7 @@ impl<'tcx> Global<'tcx> {
102107
value: Value::ByVal(PrimVal::Undef),
103108
mutable: true,
104109
ty,
110+
initialized: false,
105111
}
106112
}
107113
}

src/memory.rs

+20-4
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,7 @@ pub struct Allocation {
3838
/// The alignment of the allocation to detect unaligned reads.
3939
pub align: u64,
4040
/// Whether the allocation may be modified.
41-
/// Use the `mark_static` method of `Memory` to ensure that an error occurs, if the memory of this
41+
/// Use the `mark_static_initalized` method of `Memory` to ensure that an error occurs, if the memory of this
4242
/// allocation is modified or deallocated in the future.
4343
pub static_kind: StaticKind,
4444
}
@@ -152,6 +152,11 @@ impl<'tcx> Function<'tcx> {
152152
pub struct Memory<'a, 'tcx> {
153153
/// Actual memory allocations (arbitrary bytes, may contain pointers into other allocations)
154154
alloc_map: HashMap<AllocId, Allocation>,
155+
/// Set of statics, constants, promoteds, vtables, ... to prevent `mark_static_initalized` from stepping
156+
/// out of its own allocations.
157+
/// This set only contains statics backed by an allocation. If they are ByVal or ByValPair they
158+
/// are not here, but will be inserted once they become ByRef.
159+
static_alloc: HashSet<AllocId>,
155160
/// Number of virtual bytes allocated
156161
memory_usage: u64,
157162
/// Maximum number of virtual bytes that may be allocated
@@ -189,6 +194,7 @@ impl<'a, 'tcx> Memory<'a, 'tcx> {
189194
memory_size: max_memory,
190195
memory_usage: 0,
191196
packed: BTreeSet::new(),
197+
static_alloc: HashSet::new(),
192198
}
193199
}
194200

@@ -624,8 +630,15 @@ impl<'a, 'tcx> Memory<'a, 'tcx> {
624630

625631
/// Reading and writing
626632
impl<'a, 'tcx> Memory<'a, 'tcx> {
627-
/// mark an allocation as static, either mutable or not
628-
pub fn mark_static(&mut self, alloc_id: AllocId, mutable: bool) -> EvalResult<'tcx> {
633+
/// mark an allocation as being the entry point to a static (see `static_alloc` field)
634+
pub fn mark_static(&mut self, alloc_id: AllocId) {
635+
if alloc_id != NEVER_ALLOC_ID && alloc_id != ZST_ALLOC_ID && !self.static_alloc.insert(alloc_id) {
636+
bug!("tried to mark an allocation ({:?}) as static twice", alloc_id);
637+
}
638+
}
639+
640+
/// mark an allocation as static and initialized, either mutable or not
641+
pub fn mark_static_initalized(&mut self, alloc_id: AllocId, mutable: bool) -> EvalResult<'tcx> {
629642
// do not use `self.get_mut(alloc_id)` here, because we might have already marked a
630643
// sub-element or have circular pointers (e.g. `Rc`-cycles)
631644
let relocations = match self.alloc_map.get_mut(&alloc_id) {
@@ -645,7 +658,10 @@ impl<'a, 'tcx> Memory<'a, 'tcx> {
645658
};
646659
// recurse into inner allocations
647660
for &alloc in relocations.values() {
648-
self.mark_static(alloc, mutable)?;
661+
// relocations into other statics are not "inner allocations"
662+
if !self.static_alloc.contains(&alloc) {
663+
self.mark_static_initalized(alloc, mutable)?;
664+
}
649665
}
650666
// put back the relocations
651667
self.alloc_map.get_mut(&alloc_id).expect("checked above").relocations = relocations;

src/vtable.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -112,7 +112,7 @@ impl<'a, 'tcx> EvalContext<'a, 'tcx> {
112112
}
113113
}
114114

115-
self.memory.mark_static(vtable.alloc_id, false)?;
115+
self.memory.mark_static_initalized(vtable.alloc_id, false)?;
116116

117117
Ok(vtable)
118118
}
+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#![allow(unused_variables)]
12+
13+
#![feature(associated_consts)]
14+
15+
#[derive(Clone, Copy, Debug)]
16+
struct Bar;
17+
18+
const BAZ: Bar = Bar;
19+
20+
#[derive(Debug)]
21+
struct Foo([Bar; 1]);
22+
23+
struct Biz;
24+
25+
impl Biz {
26+
const BAZ: Foo = Foo([BAZ; 1]);
27+
}
28+
29+
fn main() {
30+
let foo = Biz::BAZ;
31+
}

tests/run-pass/issue-5917.rs

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
// Copyright 2013 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
12+
struct T (&'static [isize]);
13+
static STATIC : T = T (&[5, 4, 3]);
14+
pub fn main () {
15+
let T(ref v) = STATIC;
16+
assert_eq!(v[0], 5);
17+
}

0 commit comments

Comments
 (0)