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 55a5110d2SManos Pitsidianakis //! Definitions required by QEMU when registering a device. 65a5110d2SManos Pitsidianakis 79f7d4520SPaolo Bonzini use std::{ffi::CStr, os::raw::c_void}; 85a5110d2SManos Pitsidianakis 95a5110d2SManos Pitsidianakis use crate::bindings::{Object, ObjectClass, TypeInfo}; 105a5110d2SManos Pitsidianakis 111f9d52c9SPaolo Bonzini unsafe extern "C" fn rust_instance_init<T: ObjectImpl>(obj: *mut Object) { 121f9d52c9SPaolo Bonzini // SAFETY: obj is an instance of T, since rust_instance_init<T> 131f9d52c9SPaolo Bonzini // is called from QOM core as the instance_init function 141f9d52c9SPaolo Bonzini // for class T 151f9d52c9SPaolo Bonzini unsafe { T::INSTANCE_INIT.unwrap()(&mut *obj.cast::<T>()) } 161f9d52c9SPaolo Bonzini } 171f9d52c9SPaolo Bonzini 181f9d52c9SPaolo Bonzini unsafe extern "C" fn rust_instance_post_init<T: ObjectImpl>(obj: *mut Object) { 191f9d52c9SPaolo Bonzini // SAFETY: obj is an instance of T, since rust_instance_post_init<T> 201f9d52c9SPaolo Bonzini // is called from QOM core as the instance_post_init function 211f9d52c9SPaolo Bonzini // for class T 221f9d52c9SPaolo Bonzini // 231f9d52c9SPaolo Bonzini // FIXME: it's not really guaranteed that there are no backpointers to 241f9d52c9SPaolo Bonzini // obj; it's quite possible that they have been created by instance_init(). 251f9d52c9SPaolo Bonzini // The receiver should be &self, not &mut self. 261f9d52c9SPaolo Bonzini T::INSTANCE_POST_INIT.unwrap()(unsafe { &mut *obj.cast::<T>() }) 271f9d52c9SPaolo Bonzini } 281f9d52c9SPaolo Bonzini 29*6dd818fbSPaolo Bonzini unsafe extern "C" fn rust_class_init<T: ObjectType + ClassInitImpl<T::Class>>( 30*6dd818fbSPaolo Bonzini klass: *mut ObjectClass, 31*6dd818fbSPaolo Bonzini _data: *mut c_void, 32*6dd818fbSPaolo Bonzini ) { 33*6dd818fbSPaolo Bonzini // SAFETY: klass is a T::Class, since rust_class_init<T> 34*6dd818fbSPaolo Bonzini // is called from QOM core as the class_init function 35*6dd818fbSPaolo Bonzini // for class T 36*6dd818fbSPaolo Bonzini T::class_init(unsafe { &mut *klass.cast::<T::Class>() }) 37*6dd818fbSPaolo Bonzini } 38*6dd818fbSPaolo Bonzini 397bd8e3efSPaolo Bonzini /// Trait exposed by all structs corresponding to QOM objects. 401f9d52c9SPaolo Bonzini /// 411f9d52c9SPaolo Bonzini /// # Safety 421f9d52c9SPaolo Bonzini /// 437bd8e3efSPaolo Bonzini /// For classes declared in C: 441f9d52c9SPaolo Bonzini /// 457bd8e3efSPaolo Bonzini /// - `Class` and `TYPE` must match the data in the `TypeInfo`; 467bd8e3efSPaolo Bonzini /// 477bd8e3efSPaolo Bonzini /// - the first field of the struct must be of the instance type corresponding 487bd8e3efSPaolo Bonzini /// to the superclass, as declared in the `TypeInfo` 497bd8e3efSPaolo Bonzini /// 507bd8e3efSPaolo Bonzini /// - likewise, the first field of the `Class` struct must be of the class type 517bd8e3efSPaolo Bonzini /// corresponding to the superclass 527bd8e3efSPaolo Bonzini /// 537bd8e3efSPaolo Bonzini /// For classes declared in Rust and implementing [`ObjectImpl`]: 547bd8e3efSPaolo Bonzini /// 557bd8e3efSPaolo Bonzini /// - the struct must be `#[repr(C)]`; 561f9d52c9SPaolo Bonzini /// 571f9d52c9SPaolo Bonzini /// - the first field of the struct must be of the instance struct corresponding 58166e8a1fSPaolo Bonzini /// to the superclass, which is `ObjectImpl::ParentType` 597bd8e3efSPaolo Bonzini /// 607bd8e3efSPaolo Bonzini /// - likewise, the first field of the `Class` must be of the class struct 61166e8a1fSPaolo Bonzini /// corresponding to the superclass, which is `ObjectImpl::ParentType::Class`. 627bd8e3efSPaolo Bonzini pub unsafe trait ObjectType: Sized { 63*6dd818fbSPaolo Bonzini /// The QOM class object corresponding to this struct. This is used 64*6dd818fbSPaolo Bonzini /// to automatically generate a `class_init` method. 65c6c4f3e0SPaolo Bonzini type Class; 661f9d52c9SPaolo Bonzini 671f9d52c9SPaolo Bonzini /// The name of the type, which can be passed to `object_new()` to 681f9d52c9SPaolo Bonzini /// generate an instance of this type. 695a5110d2SManos Pitsidianakis const TYPE_NAME: &'static CStr; 707bd8e3efSPaolo Bonzini } 711f9d52c9SPaolo Bonzini 727bd8e3efSPaolo Bonzini /// Trait a type must implement to be registered with QEMU. 73*6dd818fbSPaolo Bonzini pub trait ObjectImpl: ObjectType + ClassInitImpl<Self::Class> { 741f9d52c9SPaolo Bonzini /// The parent of the type. This should match the first field of 751f9d52c9SPaolo Bonzini /// the struct that implements `ObjectImpl`: 76166e8a1fSPaolo Bonzini type ParentType: ObjectType; 771f9d52c9SPaolo Bonzini 781f9d52c9SPaolo Bonzini /// Whether the object can be instantiated 79b2a48545SPaolo Bonzini const ABSTRACT: bool = false; 80b2a48545SPaolo Bonzini const INSTANCE_FINALIZE: Option<unsafe extern "C" fn(obj: *mut Object)> = None; 813701fb22SPaolo Bonzini 821f9d52c9SPaolo Bonzini /// Function that is called to initialize an object. The parent class will 831f9d52c9SPaolo Bonzini /// have already been initialized so the type is only responsible for 841f9d52c9SPaolo Bonzini /// initializing its own members. 851f9d52c9SPaolo Bonzini /// 861f9d52c9SPaolo Bonzini /// FIXME: The argument is not really a valid reference. `&mut 871f9d52c9SPaolo Bonzini /// MaybeUninit<Self>` would be a better description. 881f9d52c9SPaolo Bonzini const INSTANCE_INIT: Option<unsafe fn(&mut Self)> = None; 891f9d52c9SPaolo Bonzini 901f9d52c9SPaolo Bonzini /// Function that is called to finish initialization of an object, once 911f9d52c9SPaolo Bonzini /// `INSTANCE_INIT` functions have been called. 921f9d52c9SPaolo Bonzini const INSTANCE_POST_INIT: Option<fn(&mut Self)> = None; 931f9d52c9SPaolo Bonzini 94*6dd818fbSPaolo Bonzini /// Called on descendent classes after all parent class initialization 95*6dd818fbSPaolo Bonzini /// has occurred, but before the class itself is initialized. This 96*6dd818fbSPaolo Bonzini /// is only useful if a class is not a leaf, and can be used to undo 97*6dd818fbSPaolo Bonzini /// the effects of copying the contents of the parent's class struct 98*6dd818fbSPaolo Bonzini /// to the descendants. 99*6dd818fbSPaolo Bonzini const CLASS_BASE_INIT: Option< 100*6dd818fbSPaolo Bonzini unsafe extern "C" fn(klass: *mut ObjectClass, data: *mut c_void), 101*6dd818fbSPaolo Bonzini > = None; 102*6dd818fbSPaolo Bonzini 1033701fb22SPaolo Bonzini const TYPE_INFO: TypeInfo = TypeInfo { 1043701fb22SPaolo Bonzini name: Self::TYPE_NAME.as_ptr(), 105166e8a1fSPaolo Bonzini parent: Self::ParentType::TYPE_NAME.as_ptr(), 1063701fb22SPaolo Bonzini instance_size: core::mem::size_of::<Self>(), 1073701fb22SPaolo Bonzini instance_align: core::mem::align_of::<Self>(), 1081f9d52c9SPaolo Bonzini instance_init: match Self::INSTANCE_INIT { 1091f9d52c9SPaolo Bonzini None => None, 1101f9d52c9SPaolo Bonzini Some(_) => Some(rust_instance_init::<Self>), 1111f9d52c9SPaolo Bonzini }, 1121f9d52c9SPaolo Bonzini instance_post_init: match Self::INSTANCE_POST_INIT { 1131f9d52c9SPaolo Bonzini None => None, 1141f9d52c9SPaolo Bonzini Some(_) => Some(rust_instance_post_init::<Self>), 1151f9d52c9SPaolo Bonzini }, 1163701fb22SPaolo Bonzini instance_finalize: Self::INSTANCE_FINALIZE, 1173701fb22SPaolo Bonzini abstract_: Self::ABSTRACT, 1183701fb22SPaolo Bonzini class_size: core::mem::size_of::<Self::Class>(), 119*6dd818fbSPaolo Bonzini class_init: Some(rust_class_init::<Self>), 120*6dd818fbSPaolo Bonzini class_base_init: Self::CLASS_BASE_INIT, 1213701fb22SPaolo Bonzini class_data: core::ptr::null_mut(), 1223701fb22SPaolo Bonzini interfaces: core::ptr::null_mut(), 1233701fb22SPaolo Bonzini }; 1245a5110d2SManos Pitsidianakis } 1255a5110d2SManos Pitsidianakis 126*6dd818fbSPaolo Bonzini /// Internal trait used to automatically fill in a class struct. 12793ea0896SPaolo Bonzini /// 12893ea0896SPaolo Bonzini /// Each QOM class that has virtual methods describes them in a 12993ea0896SPaolo Bonzini /// _class struct_. Class structs include a parent field corresponding 13093ea0896SPaolo Bonzini /// to the vtable of the parent class, all the way up to [`ObjectClass`]. 131*6dd818fbSPaolo Bonzini /// Each QOM type has one such class struct; this trait takes care of 132*6dd818fbSPaolo Bonzini /// initializing the `T` part of the class struct, for the type that 133*6dd818fbSPaolo Bonzini /// implements the trait. 13493ea0896SPaolo Bonzini /// 135*6dd818fbSPaolo Bonzini /// Each struct will implement this trait with `T` equal to each 136*6dd818fbSPaolo Bonzini /// superclass. For example, a device should implement at least 137*6dd818fbSPaolo Bonzini /// `ClassInitImpl<`[`DeviceClass`](crate::bindings::DeviceClass)`>`. 138*6dd818fbSPaolo Bonzini /// Such implementations are made in one of two ways. 139*6dd818fbSPaolo Bonzini /// 140*6dd818fbSPaolo Bonzini /// For most superclasses, `ClassInitImpl` is provided by the `qemu-api` 141*6dd818fbSPaolo Bonzini /// crate itself. The Rust implementation of methods will come from a 142*6dd818fbSPaolo Bonzini /// trait like [`ObjectImpl`] or 143*6dd818fbSPaolo Bonzini /// [`DeviceImpl`](crate::device_class::DeviceImpl), and `ClassInitImpl` is 144*6dd818fbSPaolo Bonzini /// provided by blanket implementations that operate on all implementors of the 145*6dd818fbSPaolo Bonzini /// `*Impl`* trait. For example: 146*6dd818fbSPaolo Bonzini /// 147*6dd818fbSPaolo Bonzini /// ```ignore 148*6dd818fbSPaolo Bonzini /// impl<T> ClassInitImpl<DeviceClass> for T 149*6dd818fbSPaolo Bonzini /// where 150*6dd818fbSPaolo Bonzini /// T: DeviceImpl, 151*6dd818fbSPaolo Bonzini /// ``` 152*6dd818fbSPaolo Bonzini /// 153*6dd818fbSPaolo Bonzini /// The other case is when manual implementation of the trait is needed. 154*6dd818fbSPaolo Bonzini /// This covers the following cases: 155*6dd818fbSPaolo Bonzini /// 156*6dd818fbSPaolo Bonzini /// * if a class implements a QOM interface, the Rust code _has_ to define its 157*6dd818fbSPaolo Bonzini /// own class struct `FooClass` and implement `ClassInitImpl<FooClass>`. 158*6dd818fbSPaolo Bonzini /// `ClassInitImpl<FooClass>`'s `class_init` method will then forward to 159*6dd818fbSPaolo Bonzini /// multiple other `class_init`s, for the interfaces as well as the 160*6dd818fbSPaolo Bonzini /// superclass. (Note that there is no Rust example yet for using interfaces). 161*6dd818fbSPaolo Bonzini /// 162*6dd818fbSPaolo Bonzini /// * for classes implemented outside the ``qemu-api`` crate, it's not possible 163*6dd818fbSPaolo Bonzini /// to add blanket implementations like the above one, due to orphan rules. In 164*6dd818fbSPaolo Bonzini /// that case, the easiest solution is to implement 165*6dd818fbSPaolo Bonzini /// `ClassInitImpl<YourSuperclass>` for each subclass and not have a 166*6dd818fbSPaolo Bonzini /// `YourSuperclassImpl` trait at all. 167*6dd818fbSPaolo Bonzini /// 168*6dd818fbSPaolo Bonzini /// ```ignore 169*6dd818fbSPaolo Bonzini /// impl ClassInitImpl<YourSuperclass> for YourSubclass { 170*6dd818fbSPaolo Bonzini /// fn class_init(klass: &mut YourSuperclass) { 171*6dd818fbSPaolo Bonzini /// klass.some_method = Some(Self::some_method); 172*6dd818fbSPaolo Bonzini /// <Self as ClassInitImpl<SysBusDeviceClass>>::class_init(&mut klass.parent_class); 173*6dd818fbSPaolo Bonzini /// } 174*6dd818fbSPaolo Bonzini /// } 175*6dd818fbSPaolo Bonzini /// ``` 176*6dd818fbSPaolo Bonzini /// 177*6dd818fbSPaolo Bonzini /// While this method incurs a small amount of code duplication, 178*6dd818fbSPaolo Bonzini /// it is generally limited to the recursive call on the last line. 179*6dd818fbSPaolo Bonzini /// This is because classes defined in Rust do not need the same 180*6dd818fbSPaolo Bonzini /// glue code that is needed when the classes are defined in C code. 181*6dd818fbSPaolo Bonzini /// You may consider using a macro if you have many subclasses. 182*6dd818fbSPaolo Bonzini pub trait ClassInitImpl<T> { 183*6dd818fbSPaolo Bonzini /// Initialize `klass` to point to the virtual method implementations 184*6dd818fbSPaolo Bonzini /// for `Self`. On entry, the virtual method pointers are set to 18593ea0896SPaolo Bonzini /// the default values coming from the parent classes; the function 18693ea0896SPaolo Bonzini /// can change them to override virtual methods of a parent class. 187*6dd818fbSPaolo Bonzini /// 188*6dd818fbSPaolo Bonzini /// The virtual method implementations usually come from another 189*6dd818fbSPaolo Bonzini /// trait, for example [`DeviceImpl`](crate::device_class::DeviceImpl) 190*6dd818fbSPaolo Bonzini /// when `T` is [`DeviceClass`](crate::bindings::DeviceClass). 191*6dd818fbSPaolo Bonzini /// 192*6dd818fbSPaolo Bonzini /// On entry, `klass`'s parent class is initialized, while the other fields 193*6dd818fbSPaolo Bonzini /// are all zero; it is therefore assumed that all fields in `T` can be 194*6dd818fbSPaolo Bonzini /// zeroed, otherwise it would not be possible to provide the class as a 195*6dd818fbSPaolo Bonzini /// `&mut T`. TODO: add a bound of [`Zeroable`](crate::zeroable::Zeroable) 196*6dd818fbSPaolo Bonzini /// to T; this is more easily done once Zeroable does not require a manual 197*6dd818fbSPaolo Bonzini /// implementation (Rust 1.75.0). 198*6dd818fbSPaolo Bonzini fn class_init(klass: &mut T); 1995a5110d2SManos Pitsidianakis } 2005a5110d2SManos Pitsidianakis 2015a5110d2SManos Pitsidianakis #[macro_export] 2025a5110d2SManos Pitsidianakis macro_rules! module_init { 203e90d4707SPaolo Bonzini ($type:ident => $body:block) => { 204e90d4707SPaolo Bonzini const _: () = { 2055a5110d2SManos Pitsidianakis #[used] 2064f752191SPaolo Bonzini #[cfg_attr( 2074f752191SPaolo Bonzini not(any(target_vendor = "apple", target_os = "windows")), 2084f752191SPaolo Bonzini link_section = ".init_array" 2094f752191SPaolo Bonzini )] 2104f752191SPaolo Bonzini #[cfg_attr(target_vendor = "apple", link_section = "__DATA,__mod_init_func")] 2115a5110d2SManos Pitsidianakis #[cfg_attr(target_os = "windows", link_section = ".CRT$XCU")] 2125a5110d2SManos Pitsidianakis pub static LOAD_MODULE: extern "C" fn() = { 213e90d4707SPaolo Bonzini extern "C" fn init_fn() { 2145a5110d2SManos Pitsidianakis $body 2155a5110d2SManos Pitsidianakis } 2165a5110d2SManos Pitsidianakis 217e90d4707SPaolo Bonzini extern "C" fn ctor_fn() { 2185a5110d2SManos Pitsidianakis unsafe { 2195a5110d2SManos Pitsidianakis $crate::bindings::register_module_init( 220e90d4707SPaolo Bonzini Some(init_fn), 221e90d4707SPaolo Bonzini $crate::bindings::module_init_type::$type, 2225a5110d2SManos Pitsidianakis ); 2235a5110d2SManos Pitsidianakis } 2245a5110d2SManos Pitsidianakis } 2255a5110d2SManos Pitsidianakis 226e90d4707SPaolo Bonzini ctor_fn 2275a5110d2SManos Pitsidianakis }; 2285a5110d2SManos Pitsidianakis }; 229e90d4707SPaolo Bonzini }; 230e90d4707SPaolo Bonzini 231e90d4707SPaolo Bonzini // shortcut because it's quite common that $body needs unsafe {} 232e90d4707SPaolo Bonzini ($type:ident => unsafe $body:block) => { 233e90d4707SPaolo Bonzini $crate::module_init! { 234e90d4707SPaolo Bonzini $type => { unsafe { $body } } 235e90d4707SPaolo Bonzini } 236e90d4707SPaolo Bonzini }; 2375a5110d2SManos Pitsidianakis } 238