mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-07-27 04:13: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
|
@ -15,10 +15,11 @@ pub mod prelude;
|
|||
pub mod bitops;
|
||||
pub mod c_str;
|
||||
pub mod cell;
|
||||
pub mod definitions;
|
||||
pub mod device_class;
|
||||
pub mod irq;
|
||||
pub mod module;
|
||||
pub mod offset_of;
|
||||
pub mod qdev;
|
||||
pub mod qom;
|
||||
pub mod sysbus;
|
||||
pub mod vmstate;
|
||||
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::BqlRefCell;
|
||||
|
||||
pub use crate::definitions::ObjectType;
|
||||
pub use crate::qom::ObjectType;
|
||||
|
|
|
@ -2,12 +2,14 @@
|
|||
// Author(s): Manos Pitsidianakis <manos.pitsidianakis@linaro.org>
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
||||
//! Bindings to create devices and access device functionality from Rust.
|
||||
|
||||
use std::ffi::CStr;
|
||||
|
||||
use crate::{
|
||||
bindings::{self, DeviceClass, DeviceState, Error, ObjectClass, Property, VMStateDescription},
|
||||
definitions::ClassInitImpl,
|
||||
prelude::*,
|
||||
qom::ClassInitImpl,
|
||||
};
|
||||
|
||||
/// Trait providing the contents of [`DeviceClass`].
|
|
@ -2,7 +2,34 @@
|
|||
// Author(s): Manos Pitsidianakis <manos.pitsidianakis@linaro.org>
|
||||
// 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};
|
||||
|
||||
|
@ -143,10 +170,9 @@ pub trait ObjectImpl: ObjectType + ClassInitImpl<Self::Class> {
|
|||
///
|
||||
/// For most superclasses, `ClassInitImpl` is provided by the `qemu-api`
|
||||
/// crate itself. The Rust implementation of methods will come from a
|
||||
/// trait like [`ObjectImpl`] or
|
||||
/// [`DeviceImpl`](crate::device_class::DeviceImpl), and `ClassInitImpl` is
|
||||
/// provided by blanket implementations that operate on all implementors of the
|
||||
/// `*Impl`* trait. For example:
|
||||
/// trait like [`ObjectImpl`] or [`DeviceImpl`](crate::qdev::DeviceImpl),
|
||||
/// and `ClassInitImpl` is provided by blanket implementations that
|
||||
/// operate on all implementors of the `*Impl`* trait. For example:
|
||||
///
|
||||
/// ```ignore
|
||||
/// 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.
|
||||
///
|
||||
/// 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).
|
||||
///
|
||||
/// 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);
|
||||
}
|
||||
|
||||
#[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
|
||||
///
|
||||
/// 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::{
|
||||
bindings::{self, DeviceClass},
|
||||
cell::bql_locked,
|
||||
definitions::ClassInitImpl,
|
||||
irq::InterruptSource,
|
||||
prelude::*,
|
||||
qom::ClassInitImpl,
|
||||
};
|
||||
|
||||
unsafe impl ObjectType for SysBusDevice {
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue