Skip to content

Commit 00f529d

Browse files
authored
Rollup merge of #116719 - celinval:smir-mono, r=oli-obk
Add MonoItems and Instance to stable_mir Also add a few methods to instantiate instances and get an instance definition. We're still missing support to actually monomorphize the instance body. This is related to rust-lang/project-stable-mir#36 r? ``@oli-obk`` ``@oli-obk`` is that what you were thinking? I incorporated ``@bjorn3`` idea of just adding a Shim instance definition in #116465.
2 parents a5aa52c + 364f1a3 commit 00f529d

File tree

7 files changed

+401
-45
lines changed

7 files changed

+401
-45
lines changed

compiler/rustc_smir/src/rustc_internal/mod.rs

+24-4
Original file line numberDiff line numberDiff line change
@@ -6,9 +6,11 @@
66
use crate::rustc_internal;
77
use crate::rustc_smir::Tables;
88
use rustc_data_structures::fx;
9+
use rustc_data_structures::fx::FxIndexMap;
910
use rustc_driver::{Callbacks, Compilation, RunCompiler};
1011
use rustc_interface::{interface, Queries};
1112
use rustc_middle::mir::interpret::AllocId;
13+
use rustc_middle::ty;
1214
use rustc_middle::ty::TyCtxt;
1315
use rustc_span::def_id::{CrateNum, DefId};
1416
use rustc_span::Span;
@@ -97,7 +99,7 @@ impl<'tcx> Tables<'tcx> {
9799
stable_mir::ty::Prov(self.create_alloc_id(aid))
98100
}
99101

100-
fn create_def_id(&mut self, did: DefId) -> stable_mir::DefId {
102+
pub(crate) fn create_def_id(&mut self, did: DefId) -> stable_mir::DefId {
101103
self.def_ids.create_or_fetch(did)
102104
}
103105

@@ -108,6 +110,17 @@ impl<'tcx> Tables<'tcx> {
108110
pub(crate) fn create_span(&mut self, span: Span) -> stable_mir::ty::Span {
109111
self.spans.create_or_fetch(span)
110112
}
113+
114+
pub(crate) fn instance_def(
115+
&mut self,
116+
instance: ty::Instance<'tcx>,
117+
) -> stable_mir::mir::mono::InstanceDef {
118+
self.instances.create_or_fetch(instance)
119+
}
120+
121+
pub(crate) fn static_def(&mut self, did: DefId) -> stable_mir::mir::mono::StaticDef {
122+
stable_mir::mir::mono::StaticDef(self.create_def_id(did))
123+
}
111124
}
112125

113126
pub fn crate_num(item: &stable_mir::Crate) -> CrateNum {
@@ -118,10 +131,11 @@ pub fn run(tcx: TyCtxt<'_>, f: impl FnOnce()) {
118131
stable_mir::run(
119132
Tables {
120133
tcx,
121-
def_ids: rustc_internal::IndexMap { index_map: fx::FxIndexMap::default() },
122-
alloc_ids: rustc_internal::IndexMap { index_map: fx::FxIndexMap::default() },
123-
spans: rustc_internal::IndexMap { index_map: fx::FxIndexMap::default() },
134+
def_ids: IndexMap::default(),
135+
alloc_ids: IndexMap::default(),
136+
spans: IndexMap::default(),
124137
types: vec![],
138+
instances: IndexMap::default(),
125139
},
126140
f,
127141
);
@@ -192,6 +206,12 @@ pub struct IndexMap<K, V> {
192206
index_map: fx::FxIndexMap<K, V>,
193207
}
194208

209+
impl<K, V> Default for IndexMap<K, V> {
210+
fn default() -> Self {
211+
Self { index_map: FxIndexMap::default() }
212+
}
213+
}
214+
195215
impl<K: PartialEq + Hash + Eq, V: Copy + Debug + PartialEq + IndexedVal> IndexMap<K, V> {
196216
pub fn create_or_fetch(&mut self, key: K) -> V {
197217
let len = self.index_map.len();

compiler/rustc_smir/src/rustc_smir/mod.rs

+99-26
Original file line numberDiff line numberDiff line change
@@ -13,10 +13,12 @@ use crate::rustc_smir::stable_mir::ty::{BoundRegion, EarlyBoundRegion, Region};
1313
use rustc_hir as hir;
1414
use rustc_middle::mir;
1515
use rustc_middle::mir::interpret::{alloc_range, AllocId};
16-
use rustc_middle::ty::{self, Ty, TyCtxt, Variance};
16+
use rustc_middle::mir::mono::MonoItem;
17+
use rustc_middle::ty::{self, Instance, ParamEnv, Ty, TyCtxt, Variance};
1718
use rustc_span::def_id::{CrateNum, DefId, LOCAL_CRATE};
1819
use rustc_target::abi::FieldIdx;
19-
use stable_mir::mir::{CopyNonOverlapping, Statement, UserTypeProjection, VariantIdx};
20+
use stable_mir::mir::mono::InstanceDef;
21+
use stable_mir::mir::{Body, CopyNonOverlapping, Statement, UserTypeProjection, VariantIdx};
2022
use stable_mir::ty::{
2123
FloatTy, GenericParamDef, IntTy, LineInfo, Movability, RigidTy, Span, TyKind, UintTy,
2224
};
@@ -119,29 +121,7 @@ impl<'tcx> Context for Tables<'tcx> {
119121

120122
fn mir_body(&mut self, item: stable_mir::DefId) -> stable_mir::mir::Body {
121123
let def_id = self[item];
122-
let mir = self.tcx.instance_mir(ty::InstanceDef::Item(def_id));
123-
stable_mir::mir::Body {
124-
blocks: mir
125-
.basic_blocks
126-
.iter()
127-
.map(|block| stable_mir::mir::BasicBlock {
128-
terminator: block.terminator().stable(self),
129-
statements: block
130-
.statements
131-
.iter()
132-
.map(|statement| statement.stable(self))
133-
.collect(),
134-
})
135-
.collect(),
136-
locals: mir
137-
.local_decls
138-
.iter()
139-
.map(|decl| stable_mir::mir::LocalDecl {
140-
ty: self.intern_ty(decl.ty),
141-
span: decl.source_info.span.stable(self),
142-
})
143-
.collect(),
144-
}
124+
self.tcx.instance_mir(ty::InstanceDef::Item(def_id)).stable(self)
145125
}
146126

147127
fn ty_kind(&mut self, ty: stable_mir::ty::Ty) -> TyKind {
@@ -190,6 +170,34 @@ impl<'tcx> Context for Tables<'tcx> {
190170
.collect(),
191171
}
192172
}
173+
174+
fn instance_body(&mut self, _def: InstanceDef) -> Body {
175+
todo!("Monomorphize the body")
176+
}
177+
178+
fn instance_ty(&mut self, def: InstanceDef) -> stable_mir::ty::Ty {
179+
let instance = self.instances[def];
180+
let ty = instance.ty(self.tcx, ParamEnv::empty());
181+
self.intern_ty(ty)
182+
}
183+
184+
fn instance_def_id(&mut self, def: InstanceDef) -> stable_mir::DefId {
185+
let def_id = self.instances[def].def_id();
186+
self.create_def_id(def_id)
187+
}
188+
189+
fn mono_instance(&mut self, item: stable_mir::CrateItem) -> stable_mir::mir::mono::Instance {
190+
let def_id = self[item.0];
191+
Instance::mono(self.tcx, def_id).stable(self)
192+
}
193+
194+
fn requires_monomorphization(&self, def_id: stable_mir::DefId) -> bool {
195+
let def_id = self[def_id];
196+
let generics = self.tcx.generics_of(def_id);
197+
let result = generics.requires_monomorphization(self.tcx);
198+
println!("req {result}: {def_id:?}");
199+
result
200+
}
193201
}
194202

195203
#[derive(Clone)]
@@ -224,7 +232,8 @@ pub struct Tables<'tcx> {
224232
pub def_ids: IndexMap<DefId, stable_mir::DefId>,
225233
pub alloc_ids: IndexMap<AllocId, stable_mir::AllocId>,
226234
pub spans: IndexMap<rustc_span::Span, Span>,
227-
pub types: Vec<MaybeStable<stable_mir::ty::TyKind, Ty<'tcx>>>,
235+
pub types: Vec<MaybeStable<TyKind, Ty<'tcx>>>,
236+
pub instances: IndexMap<ty::Instance<'tcx>, InstanceDef>,
228237
}
229238

230239
impl<'tcx> Tables<'tcx> {
@@ -254,6 +263,35 @@ pub(crate) trait Stable<'tcx> {
254263
fn stable(&self, tables: &mut Tables<'tcx>) -> Self::T;
255264
}
256265

266+
impl<'tcx> Stable<'tcx> for mir::Body<'tcx> {
267+
type T = stable_mir::mir::Body;
268+
269+
fn stable(&self, tables: &mut Tables<'tcx>) -> Self::T {
270+
stable_mir::mir::Body {
271+
blocks: self
272+
.basic_blocks
273+
.iter()
274+
.map(|block| stable_mir::mir::BasicBlock {
275+
terminator: block.terminator().stable(tables),
276+
statements: block
277+
.statements
278+
.iter()
279+
.map(|statement| statement.stable(tables))
280+
.collect(),
281+
})
282+
.collect(),
283+
locals: self
284+
.local_decls
285+
.iter()
286+
.map(|decl| stable_mir::mir::LocalDecl {
287+
ty: tables.intern_ty(decl.ty),
288+
span: decl.source_info.span.stable(tables),
289+
})
290+
.collect(),
291+
}
292+
}
293+
}
294+
257295
impl<'tcx> Stable<'tcx> for mir::Statement<'tcx> {
258296
type T = stable_mir::mir::Statement;
259297
fn stable(&self, tables: &mut Tables<'tcx>) -> Self::T {
@@ -1637,3 +1675,38 @@ impl<'tcx> Stable<'tcx> for DefKind {
16371675
opaque(self)
16381676
}
16391677
}
1678+
1679+
impl<'tcx> Stable<'tcx> for ty::Instance<'tcx> {
1680+
type T = stable_mir::mir::mono::Instance;
1681+
1682+
fn stable(&self, tables: &mut Tables<'tcx>) -> Self::T {
1683+
let def = tables.instance_def(*self);
1684+
let kind = match self.def {
1685+
ty::InstanceDef::Item(..) => stable_mir::mir::mono::InstanceKind::Item,
1686+
ty::InstanceDef::Intrinsic(..) => stable_mir::mir::mono::InstanceKind::Intrinsic,
1687+
ty::InstanceDef::Virtual(..) => stable_mir::mir::mono::InstanceKind::Virtual,
1688+
ty::InstanceDef::VTableShim(..)
1689+
| ty::InstanceDef::ReifyShim(..)
1690+
| ty::InstanceDef::FnPtrAddrShim(..)
1691+
| ty::InstanceDef::ClosureOnceShim { .. }
1692+
| ty::InstanceDef::ThreadLocalShim(..)
1693+
| ty::InstanceDef::DropGlue(..)
1694+
| ty::InstanceDef::CloneShim(..)
1695+
| ty::InstanceDef::FnPtrShim(..) => stable_mir::mir::mono::InstanceKind::Shim,
1696+
};
1697+
stable_mir::mir::mono::Instance { def, kind }
1698+
}
1699+
}
1700+
1701+
impl<'tcx> Stable<'tcx> for MonoItem<'tcx> {
1702+
type T = stable_mir::mir::mono::MonoItem;
1703+
1704+
fn stable(&self, tables: &mut Tables<'tcx>) -> Self::T {
1705+
use stable_mir::mir::mono::MonoItem as StableMonoItem;
1706+
match self {
1707+
MonoItem::Fn(instance) => StableMonoItem::Fn(instance.stable(tables)),
1708+
MonoItem::Static(def_id) => StableMonoItem::Static(tables.static_def(*def_id)),
1709+
MonoItem::GlobalAsm(item_id) => StableMonoItem::GlobalAsm(opaque(item_id)),
1710+
}
1711+
}
1712+
}

compiler/stable_mir/src/error.rs

+69
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,69 @@
1+
//! When things go wrong, we need some error handling.
2+
//! There are a few different types of errors in StableMIR:
3+
//!
4+
//! - [CompilerError]: This represents errors that can be raised when invoking the compiler.
5+
//! - [Error]: Generic error that represents the reason why a request that could not be fulfilled.
6+
7+
use std::fmt::{Debug, Display, Formatter};
8+
use std::{error, fmt};
9+
10+
/// An error type used to represent an error that has already been reported by the compiler.
11+
#[derive(Clone, Copy, PartialEq, Eq)]
12+
pub enum CompilerError<T> {
13+
/// Internal compiler error (I.e.: Compiler crashed).
14+
ICE,
15+
/// Compilation failed.
16+
CompilationFailed,
17+
/// Compilation was interrupted.
18+
Interrupted(T),
19+
/// Compilation skipped. This happens when users invoke rustc to retrieve information such as
20+
/// --version.
21+
Skipped,
22+
}
23+
24+
/// A generic error to represent an API request that cannot be fulfilled.
25+
#[derive(Debug)]
26+
pub struct Error(String);
27+
28+
impl Error {
29+
pub(crate) fn new(msg: String) -> Self {
30+
Self(msg)
31+
}
32+
}
33+
34+
impl Display for Error {
35+
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
36+
Display::fmt(&self.0, f)
37+
}
38+
}
39+
40+
impl<T> Display for CompilerError<T>
41+
where
42+
T: Display,
43+
{
44+
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
45+
match self {
46+
CompilerError::ICE => write!(f, "Internal Compiler Error"),
47+
CompilerError::CompilationFailed => write!(f, "Compilation Failed"),
48+
CompilerError::Interrupted(reason) => write!(f, "Compilation Interrupted: {reason}"),
49+
CompilerError::Skipped => write!(f, "Compilation Skipped"),
50+
}
51+
}
52+
}
53+
54+
impl<T> Debug for CompilerError<T>
55+
where
56+
T: Debug,
57+
{
58+
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
59+
match self {
60+
CompilerError::ICE => write!(f, "Internal Compiler Error"),
61+
CompilerError::CompilationFailed => write!(f, "Compilation Failed"),
62+
CompilerError::Interrupted(reason) => write!(f, "Compilation Interrupted: {reason:?}"),
63+
CompilerError::Skipped => write!(f, "Compilation Skipped"),
64+
}
65+
}
66+
}
67+
68+
impl error::Error for Error {}
69+
impl<T> error::Error for CompilerError<T> where T: Display + Debug {}

compiler/stable_mir/src/lib.rs

+28-15
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@
1717
//! The goal is to eventually be published on
1818
//! [crates.io](https://crates.io).
1919
20+
use crate::mir::mono::InstanceDef;
21+
use crate::mir::Body;
2022
use std::cell::Cell;
2123
use std::fmt;
2224
use std::fmt::Debug;
@@ -29,11 +31,15 @@ use self::ty::{
2931
#[macro_use]
3032
extern crate scoped_tls;
3133

34+
pub mod error;
3235
pub mod fold;
3336
pub mod mir;
3437
pub mod ty;
3538
pub mod visitor;
3639

40+
pub use error::*;
41+
use mir::mono::Instance;
42+
3743
/// Use String for now but we should replace it.
3844
pub type Symbol = String;
3945

@@ -85,20 +91,6 @@ pub type TraitDecls = Vec<TraitDef>;
8591
/// A list of impl trait decls.
8692
pub type ImplTraitDecls = Vec<ImplDef>;
8793

88-
/// An error type used to represent an error that has already been reported by the compiler.
89-
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
90-
pub enum CompilerError<T> {
91-
/// Internal compiler error (I.e.: Compiler crashed).
92-
ICE,
93-
/// Compilation failed.
94-
CompilationFailed,
95-
/// Compilation was interrupted.
96-
Interrupted(T),
97-
/// Compilation skipped. This happens when users invoke rustc to retrieve information such as
98-
/// --version.
99-
Skipped,
100-
}
101-
10294
/// Holds information about a crate.
10395
#[derive(Clone, PartialEq, Eq, Debug)]
10496
pub struct Crate {
@@ -113,7 +105,7 @@ pub type Filename = Opaque;
113105
/// Holds information about an item in the crate.
114106
/// For now, it only stores the item DefId. Use functions inside `rustc_internal` module to
115107
/// use this item.
116-
#[derive(Clone, PartialEq, Eq, Debug)]
108+
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
117109
pub struct CrateItem(pub DefId);
118110

119111
impl CrateItem {
@@ -132,6 +124,10 @@ impl CrateItem {
132124
pub fn kind(&self) -> DefKind {
133125
with(|cx| cx.def_kind(self.0))
134126
}
127+
128+
pub fn requires_monomorphization(&self) -> bool {
129+
with(|cx| cx.requires_monomorphization(self.0))
130+
}
135131
}
136132

137133
/// Return the function where execution starts if the current
@@ -220,6 +216,23 @@ pub trait Context {
220216

221217
/// Create a new `Ty` from scratch without information from rustc.
222218
fn mk_ty(&mut self, kind: TyKind) -> Ty;
219+
220+
/// Get the body of an Instance.
221+
/// FIXME: Monomorphize the body.
222+
fn instance_body(&mut self, instance: InstanceDef) -> Body;
223+
224+
/// Get the instance type with generic substitutions applied and lifetimes erased.
225+
fn instance_ty(&mut self, instance: InstanceDef) -> Ty;
226+
227+
/// Get the instance.
228+
fn instance_def_id(&mut self, instance: InstanceDef) -> DefId;
229+
230+
/// Convert a non-generic crate item into an instance.
231+
/// This function will panic if the item is generic.
232+
fn mono_instance(&mut self, item: CrateItem) -> Instance;
233+
234+
/// Item requires monomorphization.
235+
fn requires_monomorphization(&self, def_id: DefId) -> bool;
223236
}
224237

225238
// A thread local variable that stores a pointer to the tables mapping between TyCtxt

compiler/stable_mir/src/mir.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
11
mod body;
2+
pub mod mono;
23

34
pub use body::*;

0 commit comments

Comments
 (0)