15a5110d2SManos Pitsidianakis // Copyright 2024, Linaro Limited 25a5110d2SManos Pitsidianakis // Author(s): Manos Pitsidianakis <manos.pitsidianakis@linaro.org> 35a5110d2SManos Pitsidianakis // SPDX-License-Identifier: GPL-2.0-or-later 45a5110d2SManos Pitsidianakis 54aed0296SPaolo Bonzini //! Bindings to access QOM functionality from Rust. 64aed0296SPaolo Bonzini //! 74aed0296SPaolo Bonzini //! This module provides automatic creation and registration of `TypeInfo` 84aed0296SPaolo Bonzini //! for classes that are written in Rust, and mapping between Rust traits 94aed0296SPaolo Bonzini //! and QOM vtables. 104aed0296SPaolo Bonzini //! 114aed0296SPaolo Bonzini //! # Structure of a class 124aed0296SPaolo Bonzini //! 134aed0296SPaolo Bonzini //! A leaf class only needs a struct holding instance state. The struct must 144aed0296SPaolo Bonzini //! implement the [`ObjectType`] trait, as well as any `*Impl` traits that exist 154aed0296SPaolo Bonzini //! for its superclasses. 164aed0296SPaolo Bonzini //! 174aed0296SPaolo Bonzini //! If a class has subclasses, it will also provide a struct for instance data, 184aed0296SPaolo Bonzini //! with the same characteristics as for concrete classes, but it also needs 194aed0296SPaolo Bonzini //! additional components to support virtual methods: 204aed0296SPaolo Bonzini //! 214aed0296SPaolo Bonzini //! * a struct for class data, for example `DeviceClass`. This corresponds to 224aed0296SPaolo Bonzini //! the C "class struct" and holds the vtable that is used by instances of the 234aed0296SPaolo Bonzini //! class and its subclasses. It must start with its parent's class struct. 244aed0296SPaolo Bonzini //! 254aed0296SPaolo Bonzini //! * a trait for virtual method implementations, for example `DeviceImpl`. 264aed0296SPaolo Bonzini //! Child classes implement this trait to provide their own behavior for 274aed0296SPaolo Bonzini //! virtual methods. The trait's methods take `&self` to access instance data. 284aed0296SPaolo Bonzini //! 294aed0296SPaolo Bonzini //! * an implementation of [`ClassInitImpl`], for example 304aed0296SPaolo Bonzini //! `ClassInitImpl<DeviceClass>`. This fills the vtable in the class struct; 314aed0296SPaolo Bonzini //! the source for this is the `*Impl` trait; the associated consts and 324aed0296SPaolo Bonzini //! functions if needed are wrapped to map C types into Rust types. 335a5110d2SManos Pitsidianakis 349f7d4520SPaolo Bonzini use std::{ffi::CStr, os::raw::c_void}; 355a5110d2SManos Pitsidianakis 36*716d89f9SPaolo Bonzini pub use bindings::{Object, ObjectClass}; 37*716d89f9SPaolo Bonzini 38*716d89f9SPaolo Bonzini use crate::bindings::{self, TypeInfo}; 395a5110d2SManos Pitsidianakis 401f9d52c9SPaolo Bonzini unsafe extern "C" fn rust_instance_init<T: ObjectImpl>(obj: *mut Object) { 411f9d52c9SPaolo Bonzini // SAFETY: obj is an instance of T, since rust_instance_init<T> 421f9d52c9SPaolo Bonzini // is called from QOM core as the instance_init function 431f9d52c9SPaolo Bonzini // for class T 441f9d52c9SPaolo Bonzini unsafe { T::INSTANCE_INIT.unwrap()(&mut *obj.cast::<T>()) } 451f9d52c9SPaolo Bonzini } 461f9d52c9SPaolo Bonzini 471f9d52c9SPaolo Bonzini unsafe extern "C" fn rust_instance_post_init<T: ObjectImpl>(obj: *mut Object) { 481f9d52c9SPaolo Bonzini // SAFETY: obj is an instance of T, since rust_instance_post_init<T> 491f9d52c9SPaolo Bonzini // is called from QOM core as the instance_post_init function 501f9d52c9SPaolo Bonzini // for class T 511f9d52c9SPaolo Bonzini // 521f9d52c9SPaolo Bonzini // FIXME: it's not really guaranteed that there are no backpointers to 531f9d52c9SPaolo Bonzini // obj; it's quite possible that they have been created by instance_init(). 541f9d52c9SPaolo Bonzini // The receiver should be &self, not &mut self. 551f9d52c9SPaolo Bonzini T::INSTANCE_POST_INIT.unwrap()(unsafe { &mut *obj.cast::<T>() }) 561f9d52c9SPaolo Bonzini } 571f9d52c9SPaolo Bonzini 586dd818fbSPaolo Bonzini unsafe extern "C" fn rust_class_init<T: ObjectType + ClassInitImpl<T::Class>>( 596dd818fbSPaolo Bonzini klass: *mut ObjectClass, 606dd818fbSPaolo Bonzini _data: *mut c_void, 616dd818fbSPaolo Bonzini ) { 626dd818fbSPaolo Bonzini // SAFETY: klass is a T::Class, since rust_class_init<T> 636dd818fbSPaolo Bonzini // is called from QOM core as the class_init function 646dd818fbSPaolo Bonzini // for class T 656dd818fbSPaolo Bonzini T::class_init(unsafe { &mut *klass.cast::<T::Class>() }) 666dd818fbSPaolo Bonzini } 676dd818fbSPaolo Bonzini 687bd8e3efSPaolo Bonzini /// Trait exposed by all structs corresponding to QOM objects. 691f9d52c9SPaolo Bonzini /// 701f9d52c9SPaolo Bonzini /// # Safety 711f9d52c9SPaolo Bonzini /// 727bd8e3efSPaolo Bonzini /// For classes declared in C: 731f9d52c9SPaolo Bonzini /// 747bd8e3efSPaolo Bonzini /// - `Class` and `TYPE` must match the data in the `TypeInfo`; 757bd8e3efSPaolo Bonzini /// 767bd8e3efSPaolo Bonzini /// - the first field of the struct must be of the instance type corresponding 777bd8e3efSPaolo Bonzini /// to the superclass, as declared in the `TypeInfo` 787bd8e3efSPaolo Bonzini /// 797bd8e3efSPaolo Bonzini /// - likewise, the first field of the `Class` struct must be of the class type 807bd8e3efSPaolo Bonzini /// corresponding to the superclass 817bd8e3efSPaolo Bonzini /// 827bd8e3efSPaolo Bonzini /// For classes declared in Rust and implementing [`ObjectImpl`]: 837bd8e3efSPaolo Bonzini /// 847bd8e3efSPaolo Bonzini /// - the struct must be `#[repr(C)]`; 851f9d52c9SPaolo Bonzini /// 861f9d52c9SPaolo Bonzini /// - the first field of the struct must be of the instance struct corresponding 87166e8a1fSPaolo Bonzini /// to the superclass, which is `ObjectImpl::ParentType` 887bd8e3efSPaolo Bonzini /// 897bd8e3efSPaolo Bonzini /// - likewise, the first field of the `Class` must be of the class struct 90166e8a1fSPaolo Bonzini /// corresponding to the superclass, which is `ObjectImpl::ParentType::Class`. 917bd8e3efSPaolo Bonzini pub unsafe trait ObjectType: Sized { 926dd818fbSPaolo Bonzini /// The QOM class object corresponding to this struct. This is used 936dd818fbSPaolo Bonzini /// to automatically generate a `class_init` method. 94c6c4f3e0SPaolo Bonzini type Class; 951f9d52c9SPaolo Bonzini 961f9d52c9SPaolo Bonzini /// The name of the type, which can be passed to `object_new()` to 971f9d52c9SPaolo Bonzini /// generate an instance of this type. 985a5110d2SManos Pitsidianakis const TYPE_NAME: &'static CStr; 997bd8e3efSPaolo Bonzini } 1001f9d52c9SPaolo Bonzini 1017bd8e3efSPaolo Bonzini /// Trait a type must implement to be registered with QEMU. 1026dd818fbSPaolo Bonzini pub trait ObjectImpl: ObjectType + ClassInitImpl<Self::Class> { 1031f9d52c9SPaolo Bonzini /// The parent of the type. This should match the first field of 1041f9d52c9SPaolo Bonzini /// the struct that implements `ObjectImpl`: 105166e8a1fSPaolo Bonzini type ParentType: ObjectType; 1061f9d52c9SPaolo Bonzini 1071f9d52c9SPaolo Bonzini /// Whether the object can be instantiated 108b2a48545SPaolo Bonzini const ABSTRACT: bool = false; 109b2a48545SPaolo Bonzini const INSTANCE_FINALIZE: Option<unsafe extern "C" fn(obj: *mut Object)> = None; 1103701fb22SPaolo Bonzini 1111f9d52c9SPaolo Bonzini /// Function that is called to initialize an object. The parent class will 1121f9d52c9SPaolo Bonzini /// have already been initialized so the type is only responsible for 1131f9d52c9SPaolo Bonzini /// initializing its own members. 1141f9d52c9SPaolo Bonzini /// 1151f9d52c9SPaolo Bonzini /// FIXME: The argument is not really a valid reference. `&mut 1161f9d52c9SPaolo Bonzini /// MaybeUninit<Self>` would be a better description. 1171f9d52c9SPaolo Bonzini const INSTANCE_INIT: Option<unsafe fn(&mut Self)> = None; 1181f9d52c9SPaolo Bonzini 1191f9d52c9SPaolo Bonzini /// Function that is called to finish initialization of an object, once 1201f9d52c9SPaolo Bonzini /// `INSTANCE_INIT` functions have been called. 1211f9d52c9SPaolo Bonzini const INSTANCE_POST_INIT: Option<fn(&mut Self)> = None; 1221f9d52c9SPaolo Bonzini 1236dd818fbSPaolo Bonzini /// Called on descendent classes after all parent class initialization 1246dd818fbSPaolo Bonzini /// has occurred, but before the class itself is initialized. This 1256dd818fbSPaolo Bonzini /// is only useful if a class is not a leaf, and can be used to undo 1266dd818fbSPaolo Bonzini /// the effects of copying the contents of the parent's class struct 1276dd818fbSPaolo Bonzini /// to the descendants. 1286dd818fbSPaolo Bonzini const CLASS_BASE_INIT: Option< 1296dd818fbSPaolo Bonzini unsafe extern "C" fn(klass: *mut ObjectClass, data: *mut c_void), 1306dd818fbSPaolo Bonzini > = None; 1316dd818fbSPaolo Bonzini 1323701fb22SPaolo Bonzini const TYPE_INFO: TypeInfo = TypeInfo { 1333701fb22SPaolo Bonzini name: Self::TYPE_NAME.as_ptr(), 134166e8a1fSPaolo Bonzini parent: Self::ParentType::TYPE_NAME.as_ptr(), 1353701fb22SPaolo Bonzini instance_size: core::mem::size_of::<Self>(), 1363701fb22SPaolo Bonzini instance_align: core::mem::align_of::<Self>(), 1371f9d52c9SPaolo Bonzini instance_init: match Self::INSTANCE_INIT { 1381f9d52c9SPaolo Bonzini None => None, 1391f9d52c9SPaolo Bonzini Some(_) => Some(rust_instance_init::<Self>), 1401f9d52c9SPaolo Bonzini }, 1411f9d52c9SPaolo Bonzini instance_post_init: match Self::INSTANCE_POST_INIT { 1421f9d52c9SPaolo Bonzini None => None, 1431f9d52c9SPaolo Bonzini Some(_) => Some(rust_instance_post_init::<Self>), 1441f9d52c9SPaolo Bonzini }, 1453701fb22SPaolo Bonzini instance_finalize: Self::INSTANCE_FINALIZE, 1463701fb22SPaolo Bonzini abstract_: Self::ABSTRACT, 1473701fb22SPaolo Bonzini class_size: core::mem::size_of::<Self::Class>(), 1486dd818fbSPaolo Bonzini class_init: Some(rust_class_init::<Self>), 1496dd818fbSPaolo Bonzini class_base_init: Self::CLASS_BASE_INIT, 1503701fb22SPaolo Bonzini class_data: core::ptr::null_mut(), 1513701fb22SPaolo Bonzini interfaces: core::ptr::null_mut(), 1523701fb22SPaolo Bonzini }; 153cb36da9bSPaolo Bonzini 154cb36da9bSPaolo Bonzini // methods on ObjectClass 155cb36da9bSPaolo Bonzini const UNPARENT: Option<fn(&Self)> = None; 1565a5110d2SManos Pitsidianakis } 1575a5110d2SManos Pitsidianakis 1586dd818fbSPaolo Bonzini /// Internal trait used to automatically fill in a class struct. 15993ea0896SPaolo Bonzini /// 16093ea0896SPaolo Bonzini /// Each QOM class that has virtual methods describes them in a 16193ea0896SPaolo Bonzini /// _class struct_. Class structs include a parent field corresponding 16293ea0896SPaolo Bonzini /// to the vtable of the parent class, all the way up to [`ObjectClass`]. 1636dd818fbSPaolo Bonzini /// Each QOM type has one such class struct; this trait takes care of 1646dd818fbSPaolo Bonzini /// initializing the `T` part of the class struct, for the type that 1656dd818fbSPaolo Bonzini /// implements the trait. 16693ea0896SPaolo Bonzini /// 1676dd818fbSPaolo Bonzini /// Each struct will implement this trait with `T` equal to each 1686dd818fbSPaolo Bonzini /// superclass. For example, a device should implement at least 169*716d89f9SPaolo Bonzini /// `ClassInitImpl<`[`DeviceClass`](crate::qdev::DeviceClass)`>` and 170*716d89f9SPaolo Bonzini /// `ClassInitImpl<`[`ObjectClass`]`>`. Such implementations are made 171*716d89f9SPaolo Bonzini /// in one of two ways. 1726dd818fbSPaolo Bonzini /// 1736dd818fbSPaolo Bonzini /// For most superclasses, `ClassInitImpl` is provided by the `qemu-api` 1746dd818fbSPaolo Bonzini /// crate itself. The Rust implementation of methods will come from a 1754aed0296SPaolo Bonzini /// trait like [`ObjectImpl`] or [`DeviceImpl`](crate::qdev::DeviceImpl), 1764aed0296SPaolo Bonzini /// and `ClassInitImpl` is provided by blanket implementations that 1774aed0296SPaolo Bonzini /// operate on all implementors of the `*Impl`* trait. For example: 1786dd818fbSPaolo Bonzini /// 1796dd818fbSPaolo Bonzini /// ```ignore 1806dd818fbSPaolo Bonzini /// impl<T> ClassInitImpl<DeviceClass> for T 1816dd818fbSPaolo Bonzini /// where 182cb36da9bSPaolo Bonzini /// T: ClassInitImpl<ObjectClass> + DeviceImpl, 1836dd818fbSPaolo Bonzini /// ``` 1846dd818fbSPaolo Bonzini /// 185cb36da9bSPaolo Bonzini /// The bound on `ClassInitImpl<ObjectClass>` is needed so that, 186cb36da9bSPaolo Bonzini /// after initializing the `DeviceClass` part of the class struct, 187cb36da9bSPaolo Bonzini /// the parent [`ObjectClass`] is initialized as well. 188cb36da9bSPaolo Bonzini /// 1896dd818fbSPaolo Bonzini /// The other case is when manual implementation of the trait is needed. 1906dd818fbSPaolo Bonzini /// This covers the following cases: 1916dd818fbSPaolo Bonzini /// 1926dd818fbSPaolo Bonzini /// * if a class implements a QOM interface, the Rust code _has_ to define its 1936dd818fbSPaolo Bonzini /// own class struct `FooClass` and implement `ClassInitImpl<FooClass>`. 1946dd818fbSPaolo Bonzini /// `ClassInitImpl<FooClass>`'s `class_init` method will then forward to 1956dd818fbSPaolo Bonzini /// multiple other `class_init`s, for the interfaces as well as the 1966dd818fbSPaolo Bonzini /// superclass. (Note that there is no Rust example yet for using interfaces). 1976dd818fbSPaolo Bonzini /// 1986dd818fbSPaolo Bonzini /// * for classes implemented outside the ``qemu-api`` crate, it's not possible 1996dd818fbSPaolo Bonzini /// to add blanket implementations like the above one, due to orphan rules. In 2006dd818fbSPaolo Bonzini /// that case, the easiest solution is to implement 2016dd818fbSPaolo Bonzini /// `ClassInitImpl<YourSuperclass>` for each subclass and not have a 2026dd818fbSPaolo Bonzini /// `YourSuperclassImpl` trait at all. 2036dd818fbSPaolo Bonzini /// 2046dd818fbSPaolo Bonzini /// ```ignore 2056dd818fbSPaolo Bonzini /// impl ClassInitImpl<YourSuperclass> for YourSubclass { 2066dd818fbSPaolo Bonzini /// fn class_init(klass: &mut YourSuperclass) { 2076dd818fbSPaolo Bonzini /// klass.some_method = Some(Self::some_method); 2086dd818fbSPaolo Bonzini /// <Self as ClassInitImpl<SysBusDeviceClass>>::class_init(&mut klass.parent_class); 2096dd818fbSPaolo Bonzini /// } 2106dd818fbSPaolo Bonzini /// } 2116dd818fbSPaolo Bonzini /// ``` 2126dd818fbSPaolo Bonzini /// 2136dd818fbSPaolo Bonzini /// While this method incurs a small amount of code duplication, 2146dd818fbSPaolo Bonzini /// it is generally limited to the recursive call on the last line. 2156dd818fbSPaolo Bonzini /// This is because classes defined in Rust do not need the same 2166dd818fbSPaolo Bonzini /// glue code that is needed when the classes are defined in C code. 2176dd818fbSPaolo Bonzini /// You may consider using a macro if you have many subclasses. 2186dd818fbSPaolo Bonzini pub trait ClassInitImpl<T> { 2196dd818fbSPaolo Bonzini /// Initialize `klass` to point to the virtual method implementations 2206dd818fbSPaolo Bonzini /// for `Self`. On entry, the virtual method pointers are set to 22193ea0896SPaolo Bonzini /// the default values coming from the parent classes; the function 22293ea0896SPaolo Bonzini /// can change them to override virtual methods of a parent class. 2236dd818fbSPaolo Bonzini /// 2246dd818fbSPaolo Bonzini /// The virtual method implementations usually come from another 2254aed0296SPaolo Bonzini /// trait, for example [`DeviceImpl`](crate::qdev::DeviceImpl) 226*716d89f9SPaolo Bonzini /// when `T` is [`DeviceClass`](crate::qdev::DeviceClass). 2276dd818fbSPaolo Bonzini /// 2286dd818fbSPaolo Bonzini /// On entry, `klass`'s parent class is initialized, while the other fields 2296dd818fbSPaolo Bonzini /// are all zero; it is therefore assumed that all fields in `T` can be 2306dd818fbSPaolo Bonzini /// zeroed, otherwise it would not be possible to provide the class as a 2316dd818fbSPaolo Bonzini /// `&mut T`. TODO: add a bound of [`Zeroable`](crate::zeroable::Zeroable) 2326dd818fbSPaolo Bonzini /// to T; this is more easily done once Zeroable does not require a manual 2336dd818fbSPaolo Bonzini /// implementation (Rust 1.75.0). 2346dd818fbSPaolo Bonzini fn class_init(klass: &mut T); 2355a5110d2SManos Pitsidianakis } 2365a5110d2SManos Pitsidianakis 237cb36da9bSPaolo Bonzini /// # Safety 238cb36da9bSPaolo Bonzini /// 239cb36da9bSPaolo Bonzini /// We expect the FFI user of this function to pass a valid pointer that 240cb36da9bSPaolo Bonzini /// can be downcasted to type `T`. We also expect the device is 241cb36da9bSPaolo Bonzini /// readable/writeable from one thread at any time. 242cb36da9bSPaolo Bonzini unsafe extern "C" fn rust_unparent_fn<T: ObjectImpl>(dev: *mut Object) { 243cb36da9bSPaolo Bonzini unsafe { 244cb36da9bSPaolo Bonzini assert!(!dev.is_null()); 245cb36da9bSPaolo Bonzini let state = core::ptr::NonNull::new_unchecked(dev.cast::<T>()); 246cb36da9bSPaolo Bonzini T::UNPARENT.unwrap()(state.as_ref()); 247cb36da9bSPaolo Bonzini } 248cb36da9bSPaolo Bonzini } 249cb36da9bSPaolo Bonzini 250cb36da9bSPaolo Bonzini impl<T> ClassInitImpl<ObjectClass> for T 251cb36da9bSPaolo Bonzini where 252cb36da9bSPaolo Bonzini T: ObjectImpl, 253cb36da9bSPaolo Bonzini { 254cb36da9bSPaolo Bonzini fn class_init(oc: &mut ObjectClass) { 255cb36da9bSPaolo Bonzini if <T as ObjectImpl>::UNPARENT.is_some() { 256cb36da9bSPaolo Bonzini oc.unparent = Some(rust_unparent_fn::<T>); 257cb36da9bSPaolo Bonzini } 258cb36da9bSPaolo Bonzini } 259cb36da9bSPaolo Bonzini } 260cb36da9bSPaolo Bonzini 261cb36da9bSPaolo Bonzini unsafe impl ObjectType for Object { 262cb36da9bSPaolo Bonzini type Class = ObjectClass; 263cb36da9bSPaolo Bonzini const TYPE_NAME: &'static CStr = 264cb36da9bSPaolo Bonzini unsafe { CStr::from_bytes_with_nul_unchecked(bindings::TYPE_OBJECT) }; 265cb36da9bSPaolo Bonzini } 266