use padlock_core::arch::ArchConfig;
use padlock_core::ir::{AccessPattern, Field, StructLayout, TypeInfo};
use tree_sitter::{Node, Parser};
fn c_type_size_align(ty: &str, arch: &'static ArchConfig) -> (usize, usize) {
let ty = ty.trim();
for qual in &["const ", "volatile ", "restrict ", "unsigned ", "signed "] {
if let Some(rest) = ty.strip_prefix(qual) {
return c_type_size_align(rest, arch);
}
}
match ty {
"__m64" => return (8, 8),
"__m128" | "__m128d" | "__m128i" => return (16, 16),
"__m256" | "__m256d" | "__m256i" => return (32, 32),
"__m512" | "__m512d" | "__m512i" => return (64, 64),
"float32x2_t" | "int32x2_t" | "uint32x2_t" | "int8x8_t" | "uint8x8_t" | "int16x4_t"
| "uint16x4_t" | "float64x1_t" | "int64x1_t" | "uint64x1_t" => return (8, 8),
"float32x4_t" | "int32x4_t" | "uint32x4_t" | "float64x2_t" | "int64x2_t" | "uint64x2_t"
| "int8x16_t" | "uint8x16_t" | "int16x8_t" | "uint16x8_t" => return (16, 16),
_ => {}
}
match ty {
"std::mutex"
| "std::recursive_mutex"
| "std::timed_mutex"
| "std::recursive_timed_mutex"
| "pthread_mutex_t" => return (40, 8),
"std::shared_mutex" | "std::shared_timed_mutex" => return (56, 8),
"std::condition_variable" | "pthread_cond_t" => return (48, 8),
ty if ty.starts_with("std::atomic<") && ty.ends_with('>') => {
let inner = &ty[12..ty.len() - 1];
return c_type_size_align(inner.trim(), arch);
}
_ => {} }
match ty {
"char" | "_Bool" | "bool" => (1, 1),
"short" | "short int" => (2, 2),
"int" => (4, 4),
"long" => (arch.pointer_size, arch.pointer_size),
"long long" => (8, 8),
"float" => (4, 4),
"double" => (8, 8),
"long double" => (16, 16),
"int8_t" | "uint8_t" => (1, 1),
"int16_t" | "uint16_t" => (2, 2),
"int32_t" | "uint32_t" => (4, 4),
"int64_t" | "uint64_t" => (8, 8),
"size_t" | "ssize_t" | "ptrdiff_t" | "intptr_t" | "uintptr_t" => {
(arch.pointer_size, arch.pointer_size)
}
ty if ty.ends_with('*') => (arch.pointer_size, arch.pointer_size),
_ => (arch.pointer_size, arch.pointer_size),
}
}
fn strip_bitfield_suffix(ty: &str) -> &str {
if let Some(pos) = ty.rfind(':') {
let suffix = ty[pos + 1..].trim();
if !suffix.is_empty() && suffix.bytes().all(|b| b.is_ascii_digit()) {
return ty[..pos].trim_end();
}
}
ty
}
fn simulate_layout(
fields: &mut Vec<Field>,
struct_name: String,
arch: &'static ArchConfig,
) -> StructLayout {
let mut offset = 0usize;
let mut struct_align = 1usize;
for f in fields.iter_mut() {
if f.align > 0 {
offset = offset.next_multiple_of(f.align);
}
f.offset = offset;
offset += f.size;
struct_align = struct_align.max(f.align);
}
if struct_align > 0 {
offset = offset.next_multiple_of(struct_align);
}
StructLayout {
name: struct_name,
total_size: offset,
align: struct_align,
fields: fields.drain(..).collect(),
source_file: None,
source_line: None,
arch,
is_packed: false,
is_union: false,
}
}
fn simulate_union_layout(
fields: &mut Vec<Field>,
name: String,
arch: &'static ArchConfig,
) -> StructLayout {
for f in fields.iter_mut() {
f.offset = 0;
}
let max_size = fields.iter().map(|f| f.size).max().unwrap_or(0);
let max_align = fields.iter().map(|f| f.align).max().unwrap_or(1);
let total_size = if max_align > 0 {
max_size.next_multiple_of(max_align)
} else {
max_size
};
StructLayout {
name,
total_size,
align: max_align,
fields: fields.drain(..).collect(),
source_file: None,
source_line: None,
arch,
is_packed: false,
is_union: true,
}
}
fn parse_class_specifier(
source: &str,
node: Node<'_>,
arch: &'static ArchConfig,
) -> Option<StructLayout> {
let mut class_name = "<anonymous>".to_string();
let mut base_names: Vec<String> = Vec::new();
let mut body_node: Option<Node> = None;
for i in 0..node.child_count() {
let child = node.child(i)?;
match child.kind() {
"type_identifier" => class_name = source[child.byte_range()].to_string(),
"base_class_clause" => {
for j in 0..child.child_count() {
if let Some(base) = child.child(j) {
if base.kind() == "type_identifier" {
base_names.push(source[base.byte_range()].to_string());
}
}
}
}
"field_declaration_list" => body_node = Some(child),
_ => {}
}
}
let body = body_node?;
let has_virtual = contains_virtual_keyword(source, body);
let mut raw_fields: Vec<(String, String, Option<String>)> = Vec::new();
for i in 0..body.child_count() {
if let Some(child) = body.child(i) {
if child.kind() == "field_declaration" {
if let Some((ty, fname, guard)) = parse_field_declaration(source, child) {
raw_fields.push((fname, ty, guard));
}
}
}
}
let mut fields: Vec<Field> = Vec::new();
if has_virtual {
let ps = arch.pointer_size;
fields.push(Field {
name: "__vptr".to_string(),
ty: TypeInfo::Pointer { size: ps, align: ps },
offset: 0,
size: ps,
align: ps,
source_file: None,
source_line: None,
access: AccessPattern::Unknown,
});
}
for base in &base_names {
let ps = arch.pointer_size;
fields.push(Field {
name: format!("__base_{base}"),
ty: TypeInfo::Opaque {
name: base.clone(),
size: ps,
align: ps,
},
offset: 0,
size: ps,
align: ps,
source_file: None,
source_line: None,
access: AccessPattern::Unknown,
});
}
for (fname, ty_name, guard) in raw_fields {
let base_ty = strip_bitfield_suffix(&ty_name);
let (size, align) = c_type_size_align(base_ty, arch);
let access = if let Some(g) = guard {
AccessPattern::Concurrent { guard: Some(g), is_atomic: false }
} else {
AccessPattern::Unknown
};
fields.push(Field {
name: fname,
ty: TypeInfo::Primitive { name: ty_name, size, align },
offset: 0,
size,
align,
source_file: None,
source_line: None,
access,
});
}
if fields.is_empty() {
return None;
}
Some(simulate_layout(&mut fields, class_name, arch))
}
fn contains_virtual_keyword(source: &str, node: Node<'_>) -> bool {
let mut stack = vec![node];
while let Some(n) = stack.pop() {
if n.kind() == "virtual" {
return true;
}
if n.child_count() == 0 {
let text = &source[n.byte_range()];
if text == "virtual" {
return true;
}
}
for i in (0..n.child_count()).rev() {
if let Some(child) = n.child(i) {
stack.push(child);
}
}
}
false
}
fn extract_structs_from_tree(
source: &str,
root: Node<'_>,
arch: &'static ArchConfig,
layouts: &mut Vec<StructLayout>,
) {
let cursor = root.walk();
let mut stack = vec![root];
while let Some(node) = stack.pop() {
for i in (0..node.child_count()).rev() {
if let Some(child) = node.child(i) {
stack.push(child);
}
}
match node.kind() {
"struct_specifier" => {
if let Some(layout) = parse_struct_or_union_specifier(source, node, arch, false) {
layouts.push(layout);
}
}
"union_specifier" => {
if let Some(layout) = parse_struct_or_union_specifier(source, node, arch, true) {
layouts.push(layout);
}
}
"class_specifier" => {
if let Some(layout) = parse_class_specifier(source, node, arch) {
layouts.push(layout);
}
}
_ => {}
}
}
let cursor2 = root.walk();
let mut stack2 = vec![root];
while let Some(node) = stack2.pop() {
for i in (0..node.child_count()).rev() {
if let Some(child) = node.child(i) {
stack2.push(child);
}
}
if node.kind() == "type_definition" {
if let Some(layout) = parse_typedef_struct_or_union(source, node, arch) {
let existing = layouts
.iter()
.position(|l| l.name == layout.name || l.name == "<anonymous>");
match existing {
Some(i) if layouts[i].name == "<anonymous>" => {
layouts[i] = layout;
}
None => layouts.push(layout),
_ => {}
}
}
}
}
let _ = cursor;
let _ = cursor2; }
fn parse_struct_or_union_specifier(
source: &str,
node: Node<'_>,
arch: &'static ArchConfig,
is_union: bool,
) -> Option<StructLayout> {
let mut name = "<anonymous>".to_string();
let mut body_node: Option<Node> = None;
for i in 0..node.child_count() {
let child = node.child(i)?;
match child.kind() {
"type_identifier" => name = source[child.byte_range()].to_string(),
"field_declaration_list" => body_node = Some(child),
_ => {}
}
}
let body = body_node?;
let mut raw_fields: Vec<(String, String, Option<String>)> = Vec::new();
for i in 0..body.child_count() {
let child = body.child(i)?;
if child.kind() == "field_declaration" {
if let Some((ty, fname, guard)) = parse_field_declaration(source, child) {
raw_fields.push((fname, ty, guard));
}
}
}
if raw_fields.is_empty() {
return None;
}
let mut fields: Vec<Field> = raw_fields
.into_iter()
.map(|(fname, ty_name, guard)| {
let base = strip_bitfield_suffix(&ty_name);
let (size, align) = c_type_size_align(base, arch);
let access = if let Some(g) = guard {
AccessPattern::Concurrent {
guard: Some(g),
is_atomic: false,
}
} else {
AccessPattern::Unknown
};
Field {
name: fname,
ty: TypeInfo::Primitive {
name: ty_name,
size,
align,
},
offset: 0,
size,
align,
source_file: None,
source_line: None,
access,
}
})
.collect();
if is_union {
Some(simulate_union_layout(&mut fields, name, arch))
} else {
Some(simulate_layout(&mut fields, name, arch))
}
}
fn parse_typedef_struct_or_union(
source: &str,
node: Node<'_>,
arch: &'static ArchConfig,
) -> Option<StructLayout> {
let mut specifier_node: Option<Node> = None;
let mut is_union = false;
let mut typedef_name: Option<String> = None;
for i in 0..node.child_count() {
let child = node.child(i)?;
match child.kind() {
"struct_specifier" => {
specifier_node = Some(child);
is_union = false;
}
"union_specifier" => {
specifier_node = Some(child);
is_union = true;
}
"type_identifier" => typedef_name = Some(source[child.byte_range()].to_string()),
_ => {}
}
}
let spec = specifier_node?;
let typedef_name = typedef_name?;
let mut layout = parse_struct_or_union_specifier(source, spec, arch, is_union)?;
if layout.name == "<anonymous>" {
layout.name = typedef_name;
}
Some(layout)
}
#[allow(dead_code)]
fn parse_typedef_struct(
source: &str,
node: Node<'_>,
arch: &'static ArchConfig,
) -> Option<StructLayout> {
parse_typedef_struct_or_union(source, node, arch)
}
fn extract_guard_from_c_field_text(field_source: &str) -> Option<String> {
for kw in &["guarded_by", "pt_guarded_by", "GUARDED_BY", "PT_GUARDED_BY"] {
if let Some(pos) = field_source.find(kw) {
let after = &field_source[pos + kw.len()..];
let trimmed = after.trim_start();
if trimmed.starts_with('(') {
let inner = &trimmed[1..];
if let Some(end) = inner.find(')') {
let guard = inner[..end].trim().trim_matches('"');
if !guard.is_empty() {
return Some(guard.to_string());
}
}
}
}
}
None
}
fn parse_field_declaration(
source: &str,
node: Node<'_>,
) -> Option<(String, String, Option<String>)> {
let mut ty_parts: Vec<String> = Vec::new();
let mut field_name: Option<String> = None;
let mut bit_width: Option<String> = None;
let mut attr_text = String::new();
for i in 0..node.child_count() {
let child = node.child(i)?;
match child.kind() {
"type_specifier" | "primitive_type" | "type_identifier" | "sized_type_specifier" => {
ty_parts.push(source[child.byte_range()].trim().to_string());
}
"qualified_identifier" | "template_type" => {
ty_parts.push(source[child.byte_range()].trim().to_string());
}
"struct_specifier" | "union_specifier" => {
for j in 0..child.child_count() {
if let Some(sub) = child.child(j) {
if sub.kind() == "type_identifier" {
ty_parts.push(source[sub.byte_range()].trim().to_string());
break;
}
}
}
}
"field_identifier" => {
field_name = Some(source[child.byte_range()].trim().to_string());
}
"pointer_declarator" => {
field_name = extract_identifier(source, child);
ty_parts.push("*".to_string());
}
"bitfield_clause" => {
let text = source[child.byte_range()].trim();
bit_width = Some(text.trim_start_matches(':').trim().to_string());
}
"attribute_specifier" | "attribute" => {
attr_text.push_str(source[child.byte_range()].trim());
attr_text.push(' ');
}
_ => {}
}
}
let base_ty = ty_parts.join(" ");
let fname = field_name?;
if base_ty.is_empty() {
return None;
}
let ty = if let Some(w) = bit_width {
format!("{base_ty}:{w}")
} else {
base_ty
};
let field_src = source[node.byte_range()].to_string();
let guard = extract_guard_from_c_field_text(&attr_text)
.or_else(|| extract_guard_from_c_field_text(&field_src));
Some((ty, fname, guard))
}
fn extract_identifier(source: &str, node: Node<'_>) -> Option<String> {
if node.kind() == "field_identifier" || node.kind() == "identifier" {
return Some(source[node.byte_range()].to_string());
}
for i in 0..node.child_count() {
if let Some(child) = node.child(i) {
if let Some(name) = extract_identifier(source, child) {
return Some(name);
}
}
}
None
}
pub fn parse_c(source: &str, arch: &'static ArchConfig) -> anyhow::Result<Vec<StructLayout>> {
let mut parser = Parser::new();
parser.set_language(&tree_sitter_c::language())?;
let tree = parser
.parse(source, None)
.ok_or_else(|| anyhow::anyhow!("tree-sitter parse failed"))?;
let mut layouts = Vec::new();
extract_structs_from_tree(source, tree.root_node(), arch, &mut layouts);
Ok(layouts)
}
pub fn parse_cpp(source: &str, arch: &'static ArchConfig) -> anyhow::Result<Vec<StructLayout>> {
let mut parser = Parser::new();
parser.set_language(&tree_sitter_cpp::language())?;
let tree = parser
.parse(source, None)
.ok_or_else(|| anyhow::anyhow!("tree-sitter parse failed"))?;
let mut layouts = Vec::new();
extract_structs_from_tree(source, tree.root_node(), arch, &mut layouts);
Ok(layouts)
}
#[cfg(test)]
mod tests {
use super::*;
use padlock_core::arch::X86_64_SYSV;
#[test]
fn parse_simple_c_struct() {
let src = r#"
struct Point {
int x;
int y;
};
"#;
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts.len(), 1);
assert_eq!(layouts[0].name, "Point");
assert_eq!(layouts[0].fields.len(), 2);
assert_eq!(layouts[0].fields[0].name, "x");
assert_eq!(layouts[0].fields[1].name, "y");
}
#[test]
fn parse_typedef_struct() {
let src = r#"
typedef struct {
char is_active;
double timeout;
int port;
} Connection;
"#;
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts.len(), 1);
assert_eq!(layouts[0].name, "Connection");
assert_eq!(layouts[0].fields.len(), 3);
}
#[test]
fn c_layout_computes_offsets() {
let src = "struct T { char a; double b; };";
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts.len(), 1);
let layout = &layouts[0];
assert_eq!(layout.fields[0].offset, 0);
assert_eq!(layout.fields[1].offset, 8);
assert_eq!(layout.total_size, 16);
}
#[test]
fn c_layout_detects_padding() {
let src = "struct T { char a; int b; };";
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
let gaps = padlock_core::ir::find_padding(&layouts[0]);
assert!(!gaps.is_empty());
assert_eq!(gaps[0].bytes, 3); }
#[test]
fn parse_cpp_struct() {
let src = "struct Vec3 { float x; float y; float z; };";
let layouts = parse_cpp(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts.len(), 1);
assert_eq!(layouts[0].fields.len(), 3);
}
#[test]
fn simd_sse_field_size_and_align() {
let src = "struct Vecs { __m128 a; __m256 b; };";
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts.len(), 1);
let f = &layouts[0].fields;
assert_eq!(f[0].size, 16); assert_eq!(f[0].align, 16);
assert_eq!(f[1].size, 32); assert_eq!(f[1].align, 32);
}
#[test]
fn simd_avx512_size() {
let src = "struct Wide { __m512 v; };";
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts[0].fields[0].size, 64);
assert_eq!(layouts[0].fields[0].align, 64);
}
#[test]
fn simd_padding_detected_when_small_field_before_avx() {
let src = "struct Mixed { char flag; __m256 data; };";
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
let gaps = padlock_core::ir::find_padding(&layouts[0]);
assert!(!gaps.is_empty());
assert_eq!(gaps[0].bytes, 31);
}
#[test]
fn union_fields_all_at_offset_zero() {
let src = "union Data { int i; float f; double d; };";
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts.len(), 1);
let u = &layouts[0];
assert!(u.is_union);
for field in &u.fields {
assert_eq!(
field.offset, 0,
"union field '{}' should be at offset 0",
field.name
);
}
}
#[test]
fn union_total_size_is_max_field() {
let src = "union Data { int i; float f; double d; };";
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts[0].total_size, 8);
}
#[test]
fn union_no_padding_finding() {
let src = "union Data { int i; double d; };";
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
let report = padlock_core::findings::Report::from_layouts(&layouts);
let sr = &report.structs[0];
assert!(!sr
.findings
.iter()
.any(|f| matches!(f, padlock_core::findings::Finding::PaddingWaste { .. })));
assert!(!sr
.findings
.iter()
.any(|f| matches!(f, padlock_core::findings::Finding::ReorderSuggestion { .. })));
}
#[test]
fn typedef_union_parsed() {
let src = "typedef union { int a; double b; } Value;";
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts.len(), 1);
assert_eq!(layouts[0].name, "Value");
assert!(layouts[0].is_union);
}
#[test]
fn bitfield_type_annotated_with_width() {
let src = "struct Flags { int a : 3; int b : 5; };";
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts.len(), 1);
let names: Vec<&str> = layouts[0].fields.iter().map(|f| f.name.as_str()).collect();
assert!(names.contains(&"a") && names.contains(&"b"));
let a_ty = match &layouts[0].fields[0].ty {
padlock_core::ir::TypeInfo::Primitive { name, .. } => name.clone(),
_ => panic!("expected Primitive"),
};
assert!(
a_ty.contains(':'),
"bit field type should contain ':' width annotation"
);
}
#[test]
fn bitfield_uses_storage_unit_size() {
let src = "struct S { int a : 3; };";
let layouts = parse_c(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts[0].fields[0].size, 4);
}
#[test]
fn extract_guard_from_c_guarded_by_macro() {
let text = "int value GUARDED_BY(mu);";
let guard = extract_guard_from_c_field_text(text);
assert_eq!(guard.as_deref(), Some("mu"));
}
#[test]
fn extract_guard_from_c_attribute_specifier() {
let text = "__attribute__((guarded_by(counter_lock))) uint64_t counter;";
let guard = extract_guard_from_c_field_text(text);
assert_eq!(guard.as_deref(), Some("counter_lock"));
}
#[test]
fn extract_guard_pt_guarded_by() {
let text = "int *ptr PT_GUARDED_BY(ptr_lock);";
let guard = extract_guard_from_c_field_text(text);
assert_eq!(guard.as_deref(), Some("ptr_lock"));
}
#[test]
fn no_guard_returns_none() {
let guard = extract_guard_from_c_field_text("int x;");
assert!(guard.is_none());
}
#[test]
fn c_struct_guarded_by_sets_concurrent_access() {
let text = "uint64_t readers GUARDED_BY(lock_a);";
assert_eq!(
extract_guard_from_c_field_text(text).as_deref(),
Some("lock_a")
);
}
#[test]
fn c_struct_different_guards_detected_as_false_sharing() {
use padlock_core::arch::X86_64_SYSV;
use padlock_core::ir::{AccessPattern, Field, StructLayout, TypeInfo};
let mut layout = StructLayout {
name: "S".into(),
total_size: 128,
align: 8,
fields: vec![
Field {
name: "readers".into(),
ty: TypeInfo::Primitive {
name: "uint64_t".into(),
size: 8,
align: 8,
},
offset: 0,
size: 8,
align: 8,
source_file: None,
source_line: None,
access: AccessPattern::Concurrent {
guard: Some("lock_a".into()),
is_atomic: false,
},
},
Field {
name: "writers".into(),
ty: TypeInfo::Primitive {
name: "uint64_t".into(),
size: 8,
align: 8,
},
offset: 8,
size: 8,
align: 8,
source_file: None,
source_line: None,
access: AccessPattern::Concurrent {
guard: Some("lock_b".into()),
is_atomic: false,
},
},
],
source_file: None,
source_line: None,
arch: &X86_64_SYSV,
is_packed: false,
is_union: false,
};
assert!(padlock_core::analysis::false_sharing::has_false_sharing(
&layout
));
layout.fields[1].access = AccessPattern::Concurrent {
guard: Some("lock_a".into()),
is_atomic: false,
};
assert!(!padlock_core::analysis::false_sharing::has_false_sharing(
&layout
));
}
#[test]
fn cpp_class_with_virtual_method_has_vptr() {
let src = r#"
class Widget {
virtual void draw();
int x;
int y;
};
"#;
let layouts = parse_cpp(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts.len(), 1);
let l = &layouts[0];
assert_eq!(l.fields[0].name, "__vptr");
assert_eq!(l.fields[0].size, 8); assert_eq!(l.fields[0].offset, 0);
let x = l.fields.iter().find(|f| f.name == "x").unwrap();
assert_eq!(x.offset, 8);
}
#[test]
fn cpp_class_without_virtual_has_no_vptr() {
let src = "class Plain { int a; int b; };";
let layouts = parse_cpp(src, &X86_64_SYSV).unwrap();
assert_eq!(layouts.len(), 1);
assert!(!layouts[0].fields.iter().any(|f| f.name == "__vptr"));
}
#[test]
fn cpp_struct_keyword_with_virtual_has_vptr() {
let src = "struct IFoo { virtual ~IFoo(); virtual void bar(); };";
let layouts = parse_cpp(src, &X86_64_SYSV).unwrap();
let _ = layouts; }
#[test]
fn cpp_derived_class_has_base_slot() {
let src = r#"
class Base {
int x;
};
class Derived : public Base {
int y;
};
"#;
let layouts = parse_cpp(src, &X86_64_SYSV).unwrap();
let derived = layouts.iter().find(|l| l.name == "Derived").unwrap();
assert!(
derived.fields.iter().any(|f| f.name == "__base_Base"),
"Derived should have a __base_Base field"
);
let base_field = derived.fields.iter().find(|f| f.name == "__base_Base").unwrap();
let y_field = derived.fields.iter().find(|f| f.name == "y").unwrap();
assert!(y_field.offset >= base_field.offset + base_field.size);
}
#[test]
fn cpp_class_multiple_inheritance_has_multiple_base_slots() {
let src = r#"
class A { int a; };
class B { int b; };
class C : public A, public B { int c; };
"#;
let layouts = parse_cpp(src, &X86_64_SYSV).unwrap();
let c = layouts.iter().find(|l| l.name == "C").unwrap();
assert!(c.fields.iter().any(|f| f.name == "__base_A"));
assert!(c.fields.iter().any(|f| f.name == "__base_B"));
}
#[test]
fn cpp_virtual_base_class_total_size_accounts_for_vptr() {
let src = "class V { virtual void f(); int x; };";
let layouts = parse_cpp(src, &X86_64_SYSV).unwrap();
let l = &layouts[0];
assert_eq!(l.total_size, 16);
}
}