120 lines
3.7 KiB
Rust
120 lines
3.7 KiB
Rust
//@ run-pass
|
|
//! Test that users are able to use stable mir APIs to retrieve type information from a crate item
|
|
//! definition.
|
|
|
|
//@ ignore-stage1
|
|
//@ ignore-cross-compile
|
|
//@ ignore-remote
|
|
//@ edition: 2021
|
|
|
|
#![feature(rustc_private)]
|
|
|
|
extern crate rustc_middle;
|
|
|
|
extern crate rustc_driver;
|
|
extern crate rustc_interface;
|
|
extern crate rustc_public;
|
|
extern crate rustc_public_bridge;
|
|
|
|
use rustc_public::ty::VariantIdx;
|
|
use rustc_public::ty::{ForeignItemKind, RigidTy, Ty};
|
|
use rustc_public::*;
|
|
use rustc_public_bridge::IndexedVal;
|
|
use std::io::Write;
|
|
use std::ops::ControlFlow;
|
|
|
|
const CRATE_NAME: &str = "crate_def_ty";
|
|
|
|
/// Test if we can retrieve type information from different definitions.
|
|
fn test_def_tys() -> ControlFlow<()> {
|
|
let items = rustc_public::all_local_items();
|
|
for item in &items {
|
|
// Type from crate items.
|
|
let ty = item.ty();
|
|
match item.trimmed_name().as_str() {
|
|
"STATIC_STR" => assert!(ty.kind().is_ref()),
|
|
"CONST_U32" => assert!(ty.kind().is_integral()),
|
|
"NONE" => {
|
|
let RigidTy::Adt(adt, _) = *ty.kind().rigid().unwrap() else { panic!() };
|
|
// Definition names include the entire path.
|
|
assert_eq!(adt.name(), "std::option::Option");
|
|
// Variant name only includes the actual variant name.
|
|
// I know, probably not the best name schema. o.O
|
|
assert_eq!(adt.variant(VariantIdx::to_val(0)).unwrap().name(), "None");
|
|
}
|
|
"main" => check_fn_def(ty),
|
|
_ => unreachable!("Unexpected item: `{item:?}`"),
|
|
}
|
|
}
|
|
|
|
let foreign_items = rustc_public::local_crate().foreign_modules();
|
|
for item in foreign_items[0].module().items() {
|
|
// Type from foreign items.
|
|
let ty = item.ty();
|
|
let item_kind = item.kind();
|
|
let name = item.trimmed_name();
|
|
match item_kind {
|
|
ForeignItemKind::Fn(fn_def) => {
|
|
assert_eq!(&name, "extern_fn");
|
|
assert_eq!(ty, fn_def.ty());
|
|
check_fn_def(ty)
|
|
}
|
|
ForeignItemKind::Static(def) => {
|
|
assert_eq!(&name, "EXT_STATIC");
|
|
assert_eq!(ty, def.ty());
|
|
assert!(ty.kind().is_integral())
|
|
}
|
|
_ => unreachable!("Unexpected kind: {item_kind:?}"),
|
|
};
|
|
}
|
|
|
|
ControlFlow::Continue(())
|
|
}
|
|
|
|
fn check_fn_def(ty: Ty) {
|
|
let kind = ty.kind();
|
|
let (def, args) = kind.fn_def().expect(&format!("Expected function type, but found: {ty}"));
|
|
assert!(def.ty().kind().is_fn());
|
|
assert_eq!(def.ty_with_args(args), ty);
|
|
}
|
|
|
|
/// This test will generate and analyze a dummy crate using the stable mir.
|
|
/// For that, it will first write the dummy crate into a file.
|
|
/// Then it will create a `RustcPublic` using custom arguments and then
|
|
/// it will run the compiler.
|
|
fn main() {
|
|
let path = "defs_ty_input.rs";
|
|
generate_input(&path).unwrap();
|
|
let args = &[
|
|
"rustc".to_string(),
|
|
"-Cpanic=abort".to_string(),
|
|
"--crate-name".to_string(),
|
|
CRATE_NAME.to_string(),
|
|
path.to_string(),
|
|
];
|
|
run!(args, test_def_tys).unwrap();
|
|
}
|
|
|
|
fn generate_input(path: &str) -> std::io::Result<()> {
|
|
let mut file = std::fs::File::create(path)?;
|
|
write!(
|
|
file,
|
|
r#"
|
|
static STATIC_STR: &str = "foo";
|
|
const CONST_U32: u32 = 0u32;
|
|
static NONE: Option<i32> = Option::None;
|
|
|
|
fn main() {{
|
|
let _c = core::char::from_u32(99);
|
|
let _v = Vec::<u8>::new();
|
|
let _i = std::mem::size_of::<u8>();
|
|
}}
|
|
|
|
extern "C" {{
|
|
fn extern_fn(x: i32) -> i32;
|
|
static EXT_STATIC: i32;
|
|
}}
|
|
"#
|
|
)?;
|
|
Ok(())
|
|
}
|