rust: rename qemu-api modules to follow C code a bit more

A full match would mean calling them qom::object and hw::core::qdev.  For now,
keep the names shorter but still a bit easier to find.

Reviewed-by: Zhao Liu <zhao1.liu@intel.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
This commit is contained in:
Paolo Bonzini 2024-10-29 14:15:27 +01:00
parent cb36da9bd8
commit 4aed0296b3
10 changed files with 91 additions and 56 deletions

View file

@ -11,10 +11,10 @@ use std::{
use qemu_api::{ use qemu_api::{
bindings::{self, *}, bindings::{self, *},
c_str, c_str,
definitions::ObjectImpl,
device_class::DeviceImpl,
irq::InterruptSource, irq::InterruptSource,
prelude::*, prelude::*,
qdev::DeviceImpl,
qom::ObjectImpl,
}; };
use crate::{ use crate::{

View file

@ -40,7 +40,7 @@ pub fn derive_object(input: TokenStream) -> TokenStream {
let expanded = quote! { let expanded = quote! {
::qemu_api::module_init! { ::qemu_api::module_init! {
MODULE_INIT_QOM => unsafe { MODULE_INIT_QOM => unsafe {
::qemu_api::bindings::type_register_static(&<#name as ::qemu_api::definitions::ObjectImpl>::TYPE_INFO); ::qemu_api::bindings::type_register_static(&<#name as ::qemu_api::qom::ObjectImpl>::TYPE_INFO);
} }
} }
}; };

View file

@ -19,11 +19,12 @@ _qemu_api_rs = static_library(
'src/bitops.rs', 'src/bitops.rs',
'src/cell.rs', 'src/cell.rs',
'src/c_str.rs', 'src/c_str.rs',
'src/definitions.rs',
'src/device_class.rs',
'src/irq.rs', 'src/irq.rs',
'src/module.rs',
'src/offset_of.rs', 'src/offset_of.rs',
'src/prelude.rs', 'src/prelude.rs',
'src/qdev.rs',
'src/qom.rs',
'src/sysbus.rs', 'src/sysbus.rs',
'src/vmstate.rs', 'src/vmstate.rs',
'src/zeroable.rs', 'src/zeroable.rs',

View file

@ -15,10 +15,11 @@ pub mod prelude;
pub mod bitops; pub mod bitops;
pub mod c_str; pub mod c_str;
pub mod cell; pub mod cell;
pub mod definitions;
pub mod device_class;
pub mod irq; pub mod irq;
pub mod module;
pub mod offset_of; pub mod offset_of;
pub mod qdev;
pub mod qom;
pub mod sysbus; pub mod sysbus;
pub mod vmstate; pub mod vmstate;
pub mod zeroable; pub mod zeroable;

View file

@ -0,0 +1,43 @@
// Copyright 2024, Linaro Limited
// Author(s): Manos Pitsidianakis <manos.pitsidianakis@linaro.org>
// SPDX-License-Identifier: GPL-2.0-or-later
//! Macro to register blocks of code that run as QEMU starts up.
#[macro_export]
macro_rules! module_init {
($type:ident => $body:block) => {
const _: () = {
#[used]
#[cfg_attr(
not(any(target_vendor = "apple", target_os = "windows")),
link_section = ".init_array"
)]
#[cfg_attr(target_vendor = "apple", link_section = "__DATA,__mod_init_func")]
#[cfg_attr(target_os = "windows", link_section = ".CRT$XCU")]
pub static LOAD_MODULE: extern "C" fn() = {
extern "C" fn init_fn() {
$body
}
extern "C" fn ctor_fn() {
unsafe {
$crate::bindings::register_module_init(
Some(init_fn),
$crate::bindings::module_init_type::$type,
);
}
}
ctor_fn
};
};
};
// shortcut because it's quite common that $body needs unsafe {}
($type:ident => unsafe $body:block) => {
$crate::module_init! {
$type => { unsafe { $body } }
}
};
}

View file

@ -7,4 +7,4 @@ pub use crate::bitops::IntegerExt;
pub use crate::cell::BqlCell; pub use crate::cell::BqlCell;
pub use crate::cell::BqlRefCell; pub use crate::cell::BqlRefCell;
pub use crate::definitions::ObjectType; pub use crate::qom::ObjectType;

View file

@ -2,12 +2,14 @@
// Author(s): Manos Pitsidianakis <manos.pitsidianakis@linaro.org> // Author(s): Manos Pitsidianakis <manos.pitsidianakis@linaro.org>
// SPDX-License-Identifier: GPL-2.0-or-later // SPDX-License-Identifier: GPL-2.0-or-later
//! Bindings to create devices and access device functionality from Rust.
use std::ffi::CStr; use std::ffi::CStr;
use crate::{ use crate::{
bindings::{self, DeviceClass, DeviceState, Error, ObjectClass, Property, VMStateDescription}, bindings::{self, DeviceClass, DeviceState, Error, ObjectClass, Property, VMStateDescription},
definitions::ClassInitImpl,
prelude::*, prelude::*,
qom::ClassInitImpl,
}; };
/// Trait providing the contents of [`DeviceClass`]. /// Trait providing the contents of [`DeviceClass`].

View file

@ -2,7 +2,34 @@
// Author(s): Manos Pitsidianakis <manos.pitsidianakis@linaro.org> // Author(s): Manos Pitsidianakis <manos.pitsidianakis@linaro.org>
// SPDX-License-Identifier: GPL-2.0-or-later // SPDX-License-Identifier: GPL-2.0-or-later
//! Definitions required by QEMU when registering a device. //! Bindings to access QOM functionality from Rust.
//!
//! This module provides automatic creation and registration of `TypeInfo`
//! for classes that are written in Rust, and mapping between Rust traits
//! and QOM vtables.
//!
//! # Structure of a class
//!
//! A leaf class only needs a struct holding instance state. The struct must
//! implement the [`ObjectType`] trait, as well as any `*Impl` traits that exist
//! for its superclasses.
//!
//! If a class has subclasses, it will also provide a struct for instance data,
//! with the same characteristics as for concrete classes, but it also needs
//! additional components to support virtual methods:
//!
//! * a struct for class data, for example `DeviceClass`. This corresponds to
//! the C "class struct" and holds the vtable that is used by instances of the
//! class and its subclasses. It must start with its parent's class struct.
//!
//! * a trait for virtual method implementations, for example `DeviceImpl`.
//! Child classes implement this trait to provide their own behavior for
//! virtual methods. The trait's methods take `&self` to access instance data.
//!
//! * an implementation of [`ClassInitImpl`], for example
//! `ClassInitImpl<DeviceClass>`. This fills the vtable in the class struct;
//! the source for this is the `*Impl` trait; the associated consts and
//! functions if needed are wrapped to map C types into Rust types.
use std::{ffi::CStr, os::raw::c_void}; use std::{ffi::CStr, os::raw::c_void};
@ -143,10 +170,9 @@ pub trait ObjectImpl: ObjectType + ClassInitImpl<Self::Class> {
/// ///
/// For most superclasses, `ClassInitImpl` is provided by the `qemu-api` /// For most superclasses, `ClassInitImpl` is provided by the `qemu-api`
/// crate itself. The Rust implementation of methods will come from a /// crate itself. The Rust implementation of methods will come from a
/// trait like [`ObjectImpl`] or /// trait like [`ObjectImpl`] or [`DeviceImpl`](crate::qdev::DeviceImpl),
/// [`DeviceImpl`](crate::device_class::DeviceImpl), and `ClassInitImpl` is /// and `ClassInitImpl` is provided by blanket implementations that
/// provided by blanket implementations that operate on all implementors of the /// operate on all implementors of the `*Impl`* trait. For example:
/// `*Impl`* trait. For example:
/// ///
/// ```ignore /// ```ignore
/// impl<T> ClassInitImpl<DeviceClass> for T /// impl<T> ClassInitImpl<DeviceClass> for T
@ -194,7 +220,7 @@ pub trait ClassInitImpl<T> {
/// can change them to override virtual methods of a parent class. /// can change them to override virtual methods of a parent class.
/// ///
/// The virtual method implementations usually come from another /// The virtual method implementations usually come from another
/// trait, for example [`DeviceImpl`](crate::device_class::DeviceImpl) /// trait, for example [`DeviceImpl`](crate::qdev::DeviceImpl)
/// when `T` is [`DeviceClass`](crate::bindings::DeviceClass). /// when `T` is [`DeviceClass`](crate::bindings::DeviceClass).
/// ///
/// On entry, `klass`'s parent class is initialized, while the other fields /// On entry, `klass`'s parent class is initialized, while the other fields
@ -206,44 +232,6 @@ pub trait ClassInitImpl<T> {
fn class_init(klass: &mut T); fn class_init(klass: &mut T);
} }
#[macro_export]
macro_rules! module_init {
($type:ident => $body:block) => {
const _: () = {
#[used]
#[cfg_attr(
not(any(target_vendor = "apple", target_os = "windows")),
link_section = ".init_array"
)]
#[cfg_attr(target_vendor = "apple", link_section = "__DATA,__mod_init_func")]
#[cfg_attr(target_os = "windows", link_section = ".CRT$XCU")]
pub static LOAD_MODULE: extern "C" fn() = {
extern "C" fn init_fn() {
$body
}
extern "C" fn ctor_fn() {
unsafe {
$crate::bindings::register_module_init(
Some(init_fn),
$crate::bindings::module_init_type::$type,
);
}
}
ctor_fn
};
};
};
// shortcut because it's quite common that $body needs unsafe {}
($type:ident => unsafe $body:block) => {
$crate::module_init! {
$type => { unsafe { $body } }
}
};
}
/// # Safety /// # Safety
/// ///
/// We expect the FFI user of this function to pass a valid pointer that /// We expect the FFI user of this function to pass a valid pointer that

View file

@ -9,9 +9,9 @@ pub use bindings::{SysBusDevice, SysBusDeviceClass};
use crate::{ use crate::{
bindings::{self, DeviceClass}, bindings::{self, DeviceClass},
cell::bql_locked, cell::bql_locked,
definitions::ClassInitImpl,
irq::InterruptSource, irq::InterruptSource,
prelude::*, prelude::*,
qom::ClassInitImpl,
}; };
unsafe impl ObjectType for SysBusDevice { unsafe impl ObjectType for SysBusDevice {

View file

@ -5,8 +5,8 @@
use std::ffi::CStr; use std::ffi::CStr;
use qemu_api::{ use qemu_api::{
bindings::*, c_str, declare_properties, define_property, definitions::ObjectImpl, bindings::*, c_str, declare_properties, define_property, prelude::*, qdev::DeviceImpl,
device_class::DeviceImpl, impl_device_class, prelude::*, zeroable::Zeroable, qom::ObjectImpl, zeroable::Zeroable,
}; };
#[test] #[test]