mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-07-27 04:13:53 -06:00
rust: qom: add object creation functionality
The basic object lifecycle test can now be implemented using safe code! Reviewed-by: Zhao Liu <zhao1.liu@intel.com> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
This commit is contained in:
parent
0fcccf3ff0
commit
ec3eba9896
4 changed files with 48 additions and 34 deletions
|
@ -10,11 +10,11 @@ use std::{
|
||||||
|
|
||||||
use qemu_api::{
|
use qemu_api::{
|
||||||
bindings::{
|
bindings::{
|
||||||
error_fatal, hwaddr, memory_region_init_io, qdev_init_clock_in, qdev_new,
|
error_fatal, hwaddr, memory_region_init_io, qdev_init_clock_in, qdev_prop_set_chr,
|
||||||
qdev_prop_set_chr, qemu_chr_fe_accept_input, qemu_chr_fe_ioctl, qemu_chr_fe_set_handlers,
|
qemu_chr_fe_accept_input, qemu_chr_fe_ioctl, qemu_chr_fe_set_handlers,
|
||||||
qemu_chr_fe_write_all, qemu_irq, sysbus_connect_irq, sysbus_mmio_map,
|
qemu_chr_fe_write_all, qemu_irq, sysbus_connect_irq, sysbus_mmio_map, sysbus_realize,
|
||||||
sysbus_realize_and_unref, CharBackend, Chardev, Clock, ClockEvent, MemoryRegion,
|
CharBackend, Chardev, Clock, ClockEvent, MemoryRegion, QEMUChrEvent,
|
||||||
QEMUChrEvent, CHR_IOCTL_SERIAL_SET_BREAK,
|
CHR_IOCTL_SERIAL_SET_BREAK,
|
||||||
},
|
},
|
||||||
c_str, impl_vmstate_forward,
|
c_str, impl_vmstate_forward,
|
||||||
irq::InterruptSource,
|
irq::InterruptSource,
|
||||||
|
@ -705,15 +705,18 @@ pub unsafe extern "C" fn pl011_create(
|
||||||
irq: qemu_irq,
|
irq: qemu_irq,
|
||||||
chr: *mut Chardev,
|
chr: *mut Chardev,
|
||||||
) -> *mut DeviceState {
|
) -> *mut DeviceState {
|
||||||
|
let pl011 = PL011State::new();
|
||||||
unsafe {
|
unsafe {
|
||||||
let dev: *mut DeviceState = qdev_new(PL011State::TYPE_NAME.as_ptr());
|
let dev = pl011.as_mut_ptr::<DeviceState>();
|
||||||
let sysbus: *mut SysBusDevice = dev.cast::<SysBusDevice>();
|
|
||||||
|
|
||||||
qdev_prop_set_chr(dev, c_str!("chardev").as_ptr(), chr);
|
qdev_prop_set_chr(dev, c_str!("chardev").as_ptr(), chr);
|
||||||
sysbus_realize_and_unref(sysbus, addr_of_mut!(error_fatal));
|
|
||||||
|
let sysbus = pl011.as_mut_ptr::<SysBusDevice>();
|
||||||
|
sysbus_realize(sysbus, addr_of_mut!(error_fatal));
|
||||||
sysbus_mmio_map(sysbus, 0, addr);
|
sysbus_mmio_map(sysbus, 0, addr);
|
||||||
sysbus_connect_irq(sysbus, 0, irq);
|
sysbus_connect_irq(sysbus, 0, irq);
|
||||||
dev
|
|
||||||
|
// return the pointer, which is kept alive by the QOM tree; drop owned ref
|
||||||
|
pl011.as_mut_ptr()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -12,6 +12,7 @@ pub use crate::qom::Object;
|
||||||
pub use crate::qom::ObjectCast;
|
pub use crate::qom::ObjectCast;
|
||||||
pub use crate::qom::ObjectCastMut;
|
pub use crate::qom::ObjectCastMut;
|
||||||
pub use crate::qom::ObjectDeref;
|
pub use crate::qom::ObjectDeref;
|
||||||
|
pub use crate::qom::ObjectClassMethods;
|
||||||
pub use crate::qom::ObjectMethods;
|
pub use crate::qom::ObjectMethods;
|
||||||
pub use crate::qom::ObjectType;
|
pub use crate::qom::ObjectType;
|
||||||
|
|
||||||
|
|
|
@ -66,8 +66,8 @@ pub use bindings::{Object, ObjectClass};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
bindings::{
|
bindings::{
|
||||||
self, object_dynamic_cast, object_get_class, object_get_typename, object_ref, object_unref,
|
self, object_dynamic_cast, object_get_class, object_get_typename, object_new, object_ref,
|
||||||
TypeInfo,
|
object_unref, TypeInfo,
|
||||||
},
|
},
|
||||||
cell::bql_locked,
|
cell::bql_locked,
|
||||||
};
|
};
|
||||||
|
@ -759,6 +759,24 @@ impl<T: IsA<Object>> fmt::Debug for Owned<T> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Trait for class methods exposed by the Object class. The methods can be
|
||||||
|
/// called on all objects that have the trait `IsA<Object>`.
|
||||||
|
///
|
||||||
|
/// The trait should only be used through the blanket implementation,
|
||||||
|
/// which guarantees safety via `IsA`
|
||||||
|
pub trait ObjectClassMethods: IsA<Object> {
|
||||||
|
/// Return a new reference counted instance of this class
|
||||||
|
fn new() -> Owned<Self> {
|
||||||
|
assert!(bql_locked());
|
||||||
|
// SAFETY: the object created by object_new is allocated on
|
||||||
|
// the heap and has a reference count of 1
|
||||||
|
unsafe {
|
||||||
|
let obj = &*object_new(Self::TYPE_NAME.as_ptr());
|
||||||
|
Owned::from_raw(obj.unsafe_cast::<Self>())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Trait for methods exposed by the Object class. The methods can be
|
/// Trait for methods exposed by the Object class. The methods can be
|
||||||
/// called on all objects that have the trait `IsA<Object>`.
|
/// called on all objects that have the trait `IsA<Object>`.
|
||||||
///
|
///
|
||||||
|
@ -799,4 +817,5 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<T> ObjectClassMethods for T where T: IsA<Object> {}
|
||||||
impl<R: ObjectDeref> ObjectMethods for R where R::Target: IsA<Object> {}
|
impl<R: ObjectDeref> ObjectMethods for R where R::Target: IsA<Object> {}
|
||||||
|
|
|
@ -3,8 +3,7 @@
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
use std::{
|
use std::{
|
||||||
ffi::CStr,
|
ffi::{c_void, CStr},
|
||||||
os::raw::c_void,
|
|
||||||
ptr::{addr_of, addr_of_mut},
|
ptr::{addr_of, addr_of_mut},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -15,7 +14,7 @@ use qemu_api::{
|
||||||
declare_properties, define_property,
|
declare_properties, define_property,
|
||||||
prelude::*,
|
prelude::*,
|
||||||
qdev::{DeviceClass, DeviceImpl, DeviceState, Property},
|
qdev::{DeviceClass, DeviceImpl, DeviceState, Property},
|
||||||
qom::{ClassInitImpl, ObjectImpl, Owned, ParentField},
|
qom::{ClassInitImpl, ObjectImpl, ParentField},
|
||||||
vmstate::VMStateDescription,
|
vmstate::VMStateDescription,
|
||||||
zeroable::Zeroable,
|
zeroable::Zeroable,
|
||||||
};
|
};
|
||||||
|
@ -132,10 +131,8 @@ fn init_qom() {
|
||||||
/// Create and immediately drop an instance.
|
/// Create and immediately drop an instance.
|
||||||
fn test_object_new() {
|
fn test_object_new() {
|
||||||
init_qom();
|
init_qom();
|
||||||
unsafe {
|
drop(DummyState::new());
|
||||||
object_unref(object_new(DummyState::TYPE_NAME.as_ptr()).cast());
|
drop(DummyChildState::new());
|
||||||
object_unref(object_new(DummyChildState::TYPE_NAME.as_ptr()).cast());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -143,8 +140,7 @@ fn test_object_new() {
|
||||||
/// Create, clone and then drop an instance.
|
/// Create, clone and then drop an instance.
|
||||||
fn test_clone() {
|
fn test_clone() {
|
||||||
init_qom();
|
init_qom();
|
||||||
let p: *mut DummyState = unsafe { object_new(DummyState::TYPE_NAME.as_ptr()).cast() };
|
let p = DummyState::new();
|
||||||
let p = unsafe { Owned::from_raw(p) };
|
|
||||||
assert_eq!(p.clone().typename(), "dummy");
|
assert_eq!(p.clone().typename(), "dummy");
|
||||||
drop(p);
|
drop(p);
|
||||||
}
|
}
|
||||||
|
@ -153,12 +149,8 @@ fn test_clone() {
|
||||||
/// Try invoking a method on an object.
|
/// Try invoking a method on an object.
|
||||||
fn test_typename() {
|
fn test_typename() {
|
||||||
init_qom();
|
init_qom();
|
||||||
let p: *mut DummyState = unsafe { object_new(DummyState::TYPE_NAME.as_ptr()).cast() };
|
let p = DummyState::new();
|
||||||
let p_ref: &DummyState = unsafe { &*p };
|
assert_eq!(p.typename(), "dummy");
|
||||||
assert_eq!(p_ref.typename(), "dummy");
|
|
||||||
unsafe {
|
|
||||||
object_unref(p_ref.as_object_mut_ptr().cast::<c_void>());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// a note on all "cast" tests: usually, especially for downcasts the desired
|
// a note on all "cast" tests: usually, especially for downcasts the desired
|
||||||
|
@ -173,24 +165,23 @@ fn test_typename() {
|
||||||
/// Test casts on shared references.
|
/// Test casts on shared references.
|
||||||
fn test_cast() {
|
fn test_cast() {
|
||||||
init_qom();
|
init_qom();
|
||||||
let p: *mut DummyState = unsafe { object_new(DummyState::TYPE_NAME.as_ptr()).cast() };
|
let p = DummyState::new();
|
||||||
|
let p_ptr: *mut DummyState = p.as_mut_ptr();
|
||||||
|
let p_ref: &mut DummyState = unsafe { &mut *p_ptr };
|
||||||
|
|
||||||
let p_ref: &DummyState = unsafe { &*p };
|
|
||||||
let obj_ref: &Object = p_ref.upcast();
|
let obj_ref: &Object = p_ref.upcast();
|
||||||
assert_eq!(addr_of!(*obj_ref), p.cast());
|
assert_eq!(addr_of!(*obj_ref), p_ptr.cast());
|
||||||
|
|
||||||
let sbd_ref: Option<&SysBusDevice> = obj_ref.dynamic_cast();
|
let sbd_ref: Option<&SysBusDevice> = obj_ref.dynamic_cast();
|
||||||
assert!(sbd_ref.is_none());
|
assert!(sbd_ref.is_none());
|
||||||
|
|
||||||
let dev_ref: Option<&DeviceState> = obj_ref.downcast();
|
let dev_ref: Option<&DeviceState> = obj_ref.downcast();
|
||||||
assert_eq!(addr_of!(*dev_ref.unwrap()), p.cast());
|
assert_eq!(addr_of!(*dev_ref.unwrap()), p_ptr.cast());
|
||||||
|
|
||||||
// SAFETY: the cast is wrong, but the value is only used for comparison
|
// SAFETY: the cast is wrong, but the value is only used for comparison
|
||||||
unsafe {
|
unsafe {
|
||||||
let sbd_ref: &SysBusDevice = obj_ref.unsafe_cast();
|
let sbd_ref: &SysBusDevice = obj_ref.unsafe_cast();
|
||||||
assert_eq!(addr_of!(*sbd_ref), p.cast());
|
assert_eq!(addr_of!(*sbd_ref), p_ptr.cast());
|
||||||
|
|
||||||
object_unref(p_ref.as_object_mut_ptr().cast::<c_void>());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue