// Copyright 2024, Linaro Limited // Author(s): Manos Pitsidianakis // SPDX-License-Identifier: GPL-2.0-or-later use std::{ffi::CStr, ptr::addr_of}; use bql::BqlCell; use hwcore::{DeviceImpl, DeviceState, ResettablePhasesImpl, SysBusDevice}; use migration::{VMStateDescription, VMStateDescriptionBuilder}; use qom::{prelude::*, ObjectImpl, ParentField}; use util::bindings::{module_call_init, module_init_type}; // Test that macros can compile. pub const VMSTATE: VMStateDescription = VMStateDescriptionBuilder::::new() .name(c"name") .unmigratable() .build(); #[repr(C)] #[derive(qom::Object, hwcore::Device)] pub struct DummyState { parent: ParentField, #[property(rename = "migrate-clk", default = true)] migrate_clock: bool, } qom_isa!(DummyState: Object, DeviceState); pub struct DummyClass { parent_class: ::Class, } impl DummyClass { pub fn class_init(self: &mut DummyClass) { self.parent_class.class_init::(); } } unsafe impl ObjectType for DummyState { type Class = DummyClass; const TYPE_NAME: &'static CStr = c"dummy"; } impl ObjectImpl for DummyState { type ParentType = DeviceState; const ABSTRACT: bool = false; const CLASS_INIT: fn(&mut DummyClass) = DummyClass::class_init::; } impl ResettablePhasesImpl for DummyState {} impl DeviceImpl for DummyState { const VMSTATE: Option> = Some(VMSTATE); } #[repr(C)] #[derive(qom::Object, hwcore::Device)] pub struct DummyChildState { parent: ParentField, } qom_isa!(DummyChildState: Object, DeviceState, DummyState); pub struct DummyChildClass { parent_class: ::Class, } unsafe impl ObjectType for DummyChildState { type Class = DummyChildClass; const TYPE_NAME: &'static CStr = c"dummy_child"; } impl ObjectImpl for DummyChildState { type ParentType = DummyState; const ABSTRACT: bool = false; const CLASS_INIT: fn(&mut DummyChildClass) = DummyChildClass::class_init::; } impl ResettablePhasesImpl for DummyChildState {} impl DeviceImpl for DummyChildState {} impl DummyChildClass { pub fn class_init(self: &mut DummyChildClass) { self.parent_class.class_init::(); } } fn init_qom() { static ONCE: BqlCell = BqlCell::new(false); bql::start_test(); if !ONCE.get() { unsafe { module_call_init(module_init_type::MODULE_INIT_QOM); } ONCE.set(true); } } #[test] /// Create and immediately drop an instance. fn test_object_new() { init_qom(); drop(DummyState::new()); drop(DummyChildState::new()); } #[test] #[allow(clippy::redundant_clone)] /// Create, clone and then drop an instance. fn test_clone() { init_qom(); let p = DummyState::new(); assert_eq!(p.clone().typename(), "dummy"); drop(p); } #[test] /// Try invoking a method on an object. fn test_typename() { init_qom(); let p = DummyState::new(); assert_eq!(p.typename(), "dummy"); } // a note on all "cast" tests: usually, especially for downcasts the desired // class would be placed on the right, for example: // // let sbd_ref = p.dynamic_cast::(); // // Here I am doing the opposite to check that the resulting type is correct. #[test] #[allow(clippy::shadow_unrelated)] /// Test casts on shared references. fn test_cast() { init_qom(); let p = DummyState::new(); let p_ptr: *mut DummyState = p.as_mut_ptr(); let p_ref: &mut DummyState = unsafe { &mut *p_ptr }; let obj_ref: &Object = p_ref.upcast(); assert_eq!(addr_of!(*obj_ref), p_ptr.cast()); let sbd_ref: Option<&SysBusDevice> = obj_ref.dynamic_cast(); assert!(sbd_ref.is_none()); let dev_ref: Option<&DeviceState> = obj_ref.downcast(); assert_eq!(addr_of!(*dev_ref.unwrap()), p_ptr.cast()); // SAFETY: the cast is wrong, but the value is only used for comparison unsafe { let sbd_ref: &SysBusDevice = obj_ref.unsafe_cast(); assert_eq!(addr_of!(*sbd_ref), p_ptr.cast()); } }