//@ 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 = Option::None; fn main() {{ let _c = core::char::from_u32(99); let _v = Vec::::new(); let _i = std::mem::size_of::(); }} extern "C" {{ fn extern_fn(x: i32) -> i32; static EXT_STATIC: i32; }} "# )?; Ok(()) }