mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-24 02:45:04 +00:00
168 lines
3.7 KiB
Rust
168 lines
3.7 KiB
Rust
|
use base_db::fixture::WithFixture;
|
||
|
use chalk_ir::{AdtId, TyKind};
|
||
|
use hir_def::{
|
||
|
db::DefDatabase,
|
||
|
layout::{Layout, LayoutError},
|
||
|
};
|
||
|
|
||
|
use crate::{test_db::TestDB, Interner, Substitution};
|
||
|
|
||
|
use super::layout_of_ty;
|
||
|
|
||
|
fn eval_goal(ra_fixture: &str) -> Result<Layout, LayoutError> {
|
||
|
let (db, file_id) = TestDB::with_single_file(ra_fixture);
|
||
|
let module_id = db.module_for_file(file_id);
|
||
|
let def_map = module_id.def_map(&db);
|
||
|
let scope = &def_map[module_id.local_id].scope;
|
||
|
let adt_id = scope
|
||
|
.declarations()
|
||
|
.into_iter()
|
||
|
.find_map(|x| match x {
|
||
|
hir_def::ModuleDefId::AdtId(x) => {
|
||
|
let name = match x {
|
||
|
hir_def::AdtId::StructId(x) => db.struct_data(x).name.to_string(),
|
||
|
hir_def::AdtId::UnionId(x) => db.union_data(x).name.to_string(),
|
||
|
hir_def::AdtId::EnumId(x) => db.enum_data(x).name.to_string(),
|
||
|
};
|
||
|
if name == "Goal" {
|
||
|
Some(x)
|
||
|
} else {
|
||
|
None
|
||
|
}
|
||
|
}
|
||
|
_ => None,
|
||
|
})
|
||
|
.unwrap();
|
||
|
let goal_ty = TyKind::Adt(AdtId(adt_id), Substitution::empty(Interner)).intern(Interner);
|
||
|
layout_of_ty(&db, &goal_ty)
|
||
|
}
|
||
|
|
||
|
fn check_size_and_align(ra_fixture: &str, size: u64, align: u64) {
|
||
|
let l = eval_goal(ra_fixture).unwrap();
|
||
|
assert_eq!(l.size.bytes(), size);
|
||
|
assert_eq!(l.align.abi.bytes(), align);
|
||
|
}
|
||
|
|
||
|
fn check_fail(ra_fixture: &str, e: LayoutError) {
|
||
|
let r = eval_goal(ra_fixture);
|
||
|
assert_eq!(r, Err(e));
|
||
|
}
|
||
|
|
||
|
macro_rules! size_and_align {
|
||
|
($($t:tt)*) => {
|
||
|
{
|
||
|
#[allow(dead_code)]
|
||
|
$($t)*
|
||
|
check_size_and_align(
|
||
|
stringify!($($t)*),
|
||
|
::std::mem::size_of::<Goal>() as u64,
|
||
|
::std::mem::align_of::<Goal>() as u64,
|
||
|
);
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn hello_world() {
|
||
|
size_and_align! {
|
||
|
struct Goal(i32);
|
||
|
}
|
||
|
//check_size_and_align(r#"struct Goal(i32)"#, 4, 4);
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn field_order_optimization() {
|
||
|
size_and_align! {
|
||
|
struct Goal(u8, i32, u8);
|
||
|
}
|
||
|
size_and_align! {
|
||
|
#[repr(C)]
|
||
|
struct Goal(u8, i32, u8);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn recursive() {
|
||
|
size_and_align! {
|
||
|
struct Goal {
|
||
|
left: &'static Goal,
|
||
|
right: &'static Goal,
|
||
|
}
|
||
|
}
|
||
|
size_and_align! {
|
||
|
struct BoxLike<T: ?Sized>(*mut T);
|
||
|
struct Goal(BoxLike<Goal>);
|
||
|
}
|
||
|
check_fail(
|
||
|
r#"struct Goal(Goal);"#,
|
||
|
LayoutError::UserError("infinite sized recursive type".to_string()),
|
||
|
);
|
||
|
check_fail(
|
||
|
r#"
|
||
|
struct Foo<T>(Foo<T>);
|
||
|
struct Goal(Foo<i32>);
|
||
|
"#,
|
||
|
LayoutError::UserError("infinite sized recursive type".to_string()),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn generic() {
|
||
|
size_and_align! {
|
||
|
struct Pair<A, B>(A, B);
|
||
|
struct Goal(Pair<Pair<i32, u8>, i64>);
|
||
|
}
|
||
|
size_and_align! {
|
||
|
struct X<const N: usize> {
|
||
|
field1: [i32; N],
|
||
|
field2: [u8; N],
|
||
|
}
|
||
|
struct Goal(X<1000>);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn enums() {
|
||
|
size_and_align! {
|
||
|
enum Goal {
|
||
|
Quit,
|
||
|
Move { x: i32, y: i32 },
|
||
|
ChangeColor(i32, i32, i32),
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn primitives() {
|
||
|
size_and_align! {
|
||
|
struct Goal(i32, i128, isize, usize, f32, f64, bool, char);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn tuple() {
|
||
|
size_and_align! {
|
||
|
struct Goal((), (i32, u64, bool));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn niche_optimization() {
|
||
|
check_size_and_align(
|
||
|
r#"
|
||
|
//- minicore: option
|
||
|
struct Goal(Option<&i32>);
|
||
|
"#,
|
||
|
8,
|
||
|
8,
|
||
|
);
|
||
|
check_size_and_align(
|
||
|
r#"
|
||
|
//- minicore: option
|
||
|
struct Goal(Option<Option<bool>>);
|
||
|
"#,
|
||
|
1,
|
||
|
1,
|
||
|
);
|
||
|
}
|