mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-07-31 14:23:53 -06:00
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:
parent
cb36da9bd8
commit
4aed0296b3
10 changed files with 91 additions and 56 deletions
|
@ -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::{
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -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',
|
||||||
|
|
|
@ -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;
|
||||||
|
|
43
rust/qemu-api/src/module.rs
Normal file
43
rust/qemu-api/src/module.rs
Normal 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 } }
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
|
@ -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;
|
||||||
|
|
|
@ -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`].
|
|
@ -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
|
|
@ -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 {
|
||||||
|
|
|
@ -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]
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue