Skip to content

Commit 0e32d56

Browse files
committed
Add allocation test and a bit more documentation
1 parent bee320a commit 0e32d56

File tree

2 files changed

+126
-0
lines changed

2 files changed

+126
-0
lines changed

compiler/stable_mir/src/mir/alloc.rs

+7
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
//! This module provides methods to retrieve allocation information, such as static variables.
12
use crate::mir::mono::{Instance, StaticDef};
23
use crate::ty::{Allocation, Binder, ExistentialTraitRef, IndexedVal, Ty};
34
use crate::with;
@@ -25,6 +26,12 @@ impl From<AllocId> for GlobalAlloc {
2526
}
2627

2728
impl GlobalAlloc {
29+
/// Retrieve the allocation id for a global allocation if it exists.
30+
///
31+
/// For `[GlobalAlloc::VTable]`, this will return the allocation for the VTable of the given
32+
/// type for the optional trait if the type implements the trait.
33+
///
34+
/// This method will always return `None` for allocations other than `[GlobalAlloc::VTable]`.
2835
pub fn vtable_allocation(&self) -> Option<AllocId> {
2936
with(|cx| cx.vtable_allocation(self))
3037
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,119 @@
1+
// run-pass
2+
//! Test that users are able to use stable mir APIs to retrieve information of global allocations
3+
//! such as `vtable_allocation`.
4+
5+
// ignore-stage1
6+
// ignore-cross-compile
7+
// ignore-remote
8+
// ignore-windows-gnu mingw has troubles with linking https://github.com/rust-lang/rust/pull/116837
9+
// edition: 2021
10+
11+
#![feature(rustc_private)]
12+
#![feature(assert_matches)]
13+
#![feature(control_flow_enum)]
14+
#![feature(ascii_char, ascii_char_variants)]
15+
16+
extern crate rustc_hir;
17+
extern crate rustc_middle;
18+
#[macro_use]
19+
extern crate rustc_smir;
20+
extern crate rustc_driver;
21+
extern crate rustc_interface;
22+
extern crate stable_mir;
23+
24+
use rustc_middle::ty::TyCtxt;
25+
use rustc_smir::rustc_internal;
26+
use stable_mir::{CrateItem, CrateItems, ItemKind};
27+
use stable_mir::mir::alloc::GlobalAlloc;
28+
use stable_mir::mir::mono::StaticDef;
29+
use std::ascii::Char;
30+
use std::assert_matches::assert_matches;
31+
use std::io::Write;
32+
use std::ops::ControlFlow;
33+
34+
const CRATE_NAME: &str = "input";
35+
36+
/// This function uses the Stable MIR APIs to get information about the test crate.
37+
fn test_stable_mir(_tcx: TyCtxt<'_>) -> ControlFlow<()> {
38+
// Find items in the local crate.
39+
let items = stable_mir::all_local_items();
40+
check_foo(*get_item(&items, (ItemKind::Static, "FOO")).unwrap());
41+
check_bar(*get_item(&items, (ItemKind::Static, "BAR")).unwrap());
42+
ControlFlow::Continue(())
43+
}
44+
45+
/// Check the allocation data for static `FOO`.
46+
///
47+
/// ```no_run
48+
/// static FOO: [&str; 2] = ["hi", "there"];
49+
/// ```
50+
fn check_foo(item: CrateItem) {
51+
let def = StaticDef::try_from(item).unwrap();
52+
let alloc = def.eval_initializer().unwrap();
53+
assert_eq!(alloc.provenance.ptrs.len(), 2);
54+
55+
let alloc_id_0 = alloc.provenance.ptrs[0].1.0;
56+
assert_matches!(GlobalAlloc::from(alloc_id_0), GlobalAlloc::Memory(..));
57+
58+
let alloc_id_1 = alloc.provenance.ptrs[1].1.0;
59+
assert_matches!(GlobalAlloc::from(alloc_id_1), GlobalAlloc::Memory(..));
60+
}
61+
62+
/// Check the allocation data for static `BAR`.
63+
///
64+
/// ```no_run
65+
/// static BAR: &str = "Bar";
66+
/// ```
67+
fn check_bar(item: CrateItem) {
68+
let def = StaticDef::try_from(item).unwrap();
69+
let alloc = def.eval_initializer().unwrap();
70+
assert_eq!(alloc.provenance.ptrs.len(), 1);
71+
72+
let alloc_id_0 = alloc.provenance.ptrs[0].1.0;
73+
let GlobalAlloc::Memory(allocation) = GlobalAlloc::from(alloc_id_0) else { unreachable!() };
74+
assert_eq!(allocation.bytes.len(), 3);
75+
assert_eq!(allocation.bytes[0].unwrap(), Char::CapitalB.to_u8());
76+
assert_eq!(allocation.bytes[1].unwrap(), Char::SmallA.to_u8());
77+
assert_eq!(allocation.bytes[2].unwrap(), Char::SmallR.to_u8());
78+
}
79+
80+
// Use internal API to find a function in a crate.
81+
fn get_item<'a>(
82+
items: &'a CrateItems,
83+
item: (ItemKind, &str),
84+
) -> Option<&'a stable_mir::CrateItem> {
85+
items.iter().find(|crate_item| {
86+
(item.0 == crate_item.kind()) && crate_item.name() == item.1
87+
})
88+
}
89+
90+
/// This test will generate and analyze a dummy crate using the stable mir.
91+
/// For that, it will first write the dummy crate into a file.
92+
/// Then it will create a `StableMir` using custom arguments and then
93+
/// it will run the compiler.
94+
fn main() {
95+
let path = "alloc_input.rs";
96+
generate_input(&path).unwrap();
97+
let args = vec![
98+
"rustc".to_string(),
99+
"--crate-name".to_string(),
100+
CRATE_NAME.to_string(),
101+
path.to_string(),
102+
];
103+
run!(args, tcx, test_stable_mir(tcx)).unwrap();
104+
}
105+
106+
fn generate_input(path: &str) -> std::io::Result<()> {
107+
let mut file = std::fs::File::create(path)?;
108+
write!(
109+
file,
110+
r#"
111+
static FOO: [&str; 2] = ["hi", "there"];
112+
static BAR: &str = "Bar";
113+
114+
pub fn main() {{
115+
println!("{{FOO:?}}! {{BAR}}");
116+
}}"#
117+
)?;
118+
Ok(())
119+
}

0 commit comments

Comments
 (0)