18e194c0eSPaolo Bonzini // SPDX-License-Identifier: MIT 28e194c0eSPaolo Bonzini // 38e194c0eSPaolo Bonzini // This file is based on library/core/src/cell.rs from 48e194c0eSPaolo Bonzini // Rust 1.82.0. 58e194c0eSPaolo Bonzini // 68e194c0eSPaolo Bonzini // Permission is hereby granted, free of charge, to any 78e194c0eSPaolo Bonzini // person obtaining a copy of this software and associated 88e194c0eSPaolo Bonzini // documentation files (the "Software"), to deal in the 98e194c0eSPaolo Bonzini // Software without restriction, including without 108e194c0eSPaolo Bonzini // limitation the rights to use, copy, modify, merge, 118e194c0eSPaolo Bonzini // publish, distribute, sublicense, and/or sell copies of 128e194c0eSPaolo Bonzini // the Software, and to permit persons to whom the Software 138e194c0eSPaolo Bonzini // is furnished to do so, subject to the following 148e194c0eSPaolo Bonzini // conditions: 158e194c0eSPaolo Bonzini // 168e194c0eSPaolo Bonzini // The above copyright notice and this permission notice 178e194c0eSPaolo Bonzini // shall be included in all copies or substantial portions 188e194c0eSPaolo Bonzini // of the Software. 198e194c0eSPaolo Bonzini // 208e194c0eSPaolo Bonzini // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF 218e194c0eSPaolo Bonzini // ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED 228e194c0eSPaolo Bonzini // TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A 238e194c0eSPaolo Bonzini // PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT 248e194c0eSPaolo Bonzini // SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY 258e194c0eSPaolo Bonzini // CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION 268e194c0eSPaolo Bonzini // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR 278e194c0eSPaolo Bonzini // IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER 288e194c0eSPaolo Bonzini // DEALINGS IN THE SOFTWARE. 298e194c0eSPaolo Bonzini 300b9d05e3SPaolo Bonzini //! QEMU-specific mutable containers 318e194c0eSPaolo Bonzini //! 328e194c0eSPaolo Bonzini //! Rust memory safety is based on this rule: Given an object `T`, it is only 338e194c0eSPaolo Bonzini //! possible to have one of the following: 348e194c0eSPaolo Bonzini //! 358e194c0eSPaolo Bonzini //! - Having several immutable references (`&T`) to the object (also known as 368e194c0eSPaolo Bonzini //! **aliasing**). 378e194c0eSPaolo Bonzini //! - Having one mutable reference (`&mut T`) to the object (also known as 388e194c0eSPaolo Bonzini //! **mutability**). 398e194c0eSPaolo Bonzini //! 408e194c0eSPaolo Bonzini //! This is enforced by the Rust compiler. However, there are situations where 418e194c0eSPaolo Bonzini //! this rule is not flexible enough. Sometimes it is required to have multiple 428e194c0eSPaolo Bonzini //! references to an object and yet mutate it. In particular, QEMU objects 438e194c0eSPaolo Bonzini //! usually have their pointer shared with the "outside world very early in 448e194c0eSPaolo Bonzini //! their lifetime", for example when they create their 458e194c0eSPaolo Bonzini //! [`MemoryRegion`s](crate::bindings::MemoryRegion). Therefore, individual 460b9d05e3SPaolo Bonzini //! parts of a device must be made mutable in a controlled manner; this module 470b9d05e3SPaolo Bonzini //! provides the tools to do so. 480b9d05e3SPaolo Bonzini //! 490b9d05e3SPaolo Bonzini //! ## Cell types 508e194c0eSPaolo Bonzini //! 51c596199fSPaolo Bonzini //! [`BqlCell<T>`] and [`BqlRefCell<T>`] allow doing this via the Big QEMU Lock. 52c596199fSPaolo Bonzini //! While they are essentially the same single-threaded primitives that are 53c596199fSPaolo Bonzini //! available in `std::cell`, the BQL allows them to be used from a 54c596199fSPaolo Bonzini //! multi-threaded context and to share references across threads, while 55c596199fSPaolo Bonzini //! maintaining Rust's safety guarantees. For this reason, unlike 56c596199fSPaolo Bonzini //! their `std::cell` counterparts, `BqlCell` and `BqlRefCell` implement the 57c596199fSPaolo Bonzini //! `Sync` trait. 588e194c0eSPaolo Bonzini //! 598e194c0eSPaolo Bonzini //! BQL checks are performed in debug builds but can be optimized away in 608e194c0eSPaolo Bonzini //! release builds, providing runtime safety during development with no overhead 618e194c0eSPaolo Bonzini //! in production. 628e194c0eSPaolo Bonzini //! 63c596199fSPaolo Bonzini //! The two provide different ways of handling interior mutability. 64c596199fSPaolo Bonzini //! `BqlRefCell` is best suited for data that is primarily accessed by the 65c596199fSPaolo Bonzini //! device's own methods, where multiple reads and writes can be grouped within 66c596199fSPaolo Bonzini //! a single borrow and a mutable reference can be passed around. Instead, 67c596199fSPaolo Bonzini //! [`BqlCell`] is a better choice when sharing small pieces of data with 68c596199fSPaolo Bonzini //! external code (especially C code), because it provides simple get/set 69c596199fSPaolo Bonzini //! operations that can be used one at a time. 70c596199fSPaolo Bonzini //! 71c596199fSPaolo Bonzini //! Warning: While `BqlCell` and `BqlRefCell` are similar to their `std::cell` 72c596199fSPaolo Bonzini //! counterparts, they are not interchangeable. Using `std::cell` types in 73c596199fSPaolo Bonzini //! QEMU device implementations is usually incorrect and can lead to 74c596199fSPaolo Bonzini //! thread-safety issues. 758e194c0eSPaolo Bonzini //! 765778ce99SPaolo Bonzini //! ### Example 775778ce99SPaolo Bonzini //! 785778ce99SPaolo Bonzini //! ``` 795778ce99SPaolo Bonzini //! # use qemu_api::prelude::*; 80*f117857bSPaolo Bonzini //! # use qemu_api::{cell::BqlRefCell, irq::InterruptSource, irq::IRQState}; 815778ce99SPaolo Bonzini //! # use qemu_api::{sysbus::SysBusDevice, qom::Owned, qom::ParentField}; 825778ce99SPaolo Bonzini //! # const N_GPIOS: usize = 8; 835778ce99SPaolo Bonzini //! # struct PL061Registers { /* ... */ } 845778ce99SPaolo Bonzini //! # unsafe impl ObjectType for PL061State { 855778ce99SPaolo Bonzini //! # type Class = <SysBusDevice as ObjectType>::Class; 86*f117857bSPaolo Bonzini //! # const TYPE_NAME: &'static std::ffi::CStr = c"pl061"; 875778ce99SPaolo Bonzini //! # } 885778ce99SPaolo Bonzini //! struct PL061State { 895778ce99SPaolo Bonzini //! parent_obj: ParentField<SysBusDevice>, 905778ce99SPaolo Bonzini //! 915778ce99SPaolo Bonzini //! // Configuration is read-only after initialization 925778ce99SPaolo Bonzini //! pullups: u32, 935778ce99SPaolo Bonzini //! pulldowns: u32, 945778ce99SPaolo Bonzini //! 955778ce99SPaolo Bonzini //! // Single values shared with C code use BqlCell, in this case via InterruptSource 965778ce99SPaolo Bonzini //! out: [InterruptSource; N_GPIOS], 975778ce99SPaolo Bonzini //! interrupt: InterruptSource, 985778ce99SPaolo Bonzini //! 995778ce99SPaolo Bonzini //! // Larger state accessed by device methods uses BqlRefCell or Mutex 1005778ce99SPaolo Bonzini //! registers: BqlRefCell<PL061Registers>, 1015778ce99SPaolo Bonzini //! } 1025778ce99SPaolo Bonzini //! ``` 1035778ce99SPaolo Bonzini //! 1040b9d05e3SPaolo Bonzini //! ### `BqlCell<T>` 1058e194c0eSPaolo Bonzini //! 1068e194c0eSPaolo Bonzini //! [`BqlCell<T>`] implements interior mutability by moving values in and out of 1078e194c0eSPaolo Bonzini //! the cell. That is, an `&mut T` to the inner value can never be obtained as 1088e194c0eSPaolo Bonzini //! long as the cell is shared. The value itself cannot be directly obtained 1098e194c0eSPaolo Bonzini //! without copying it, cloning it, or replacing it with something else. This 1108e194c0eSPaolo Bonzini //! type provides the following methods, all of which can be called only while 1118e194c0eSPaolo Bonzini //! the BQL is held: 1128e194c0eSPaolo Bonzini //! 1138e194c0eSPaolo Bonzini //! - For types that implement [`Copy`], the [`get`](BqlCell::get) method 1148e194c0eSPaolo Bonzini //! retrieves the current interior value by duplicating it. 1158e194c0eSPaolo Bonzini //! - For types that implement [`Default`], the [`take`](BqlCell::take) method 1168e194c0eSPaolo Bonzini //! replaces the current interior value with [`Default::default()`] and 1178e194c0eSPaolo Bonzini //! returns the replaced value. 1188e194c0eSPaolo Bonzini //! - All types have: 1198e194c0eSPaolo Bonzini //! - [`replace`](BqlCell::replace): replaces the current interior value and 1208e194c0eSPaolo Bonzini //! returns the replaced value. 1218e194c0eSPaolo Bonzini //! - [`set`](BqlCell::set): this method replaces the interior value, 1228e194c0eSPaolo Bonzini //! dropping the replaced value. 123c596199fSPaolo Bonzini //! 1240b9d05e3SPaolo Bonzini //! ### `BqlRefCell<T>` 125c596199fSPaolo Bonzini //! 126c596199fSPaolo Bonzini //! [`BqlRefCell<T>`] uses Rust's lifetimes to implement "dynamic borrowing", a 127c596199fSPaolo Bonzini //! process whereby one can claim temporary, exclusive, mutable access to the 128c596199fSPaolo Bonzini //! inner value: 129c596199fSPaolo Bonzini //! 130c596199fSPaolo Bonzini //! ```ignore 131c596199fSPaolo Bonzini //! fn clear_interrupts(&self, val: u32) { 132c596199fSPaolo Bonzini //! // A mutable borrow gives read-write access to the registers 133c596199fSPaolo Bonzini //! let mut regs = self.registers.borrow_mut(); 134c596199fSPaolo Bonzini //! let old = regs.interrupt_status(); 135c596199fSPaolo Bonzini //! regs.update_interrupt_status(old & !val); 136c596199fSPaolo Bonzini //! } 137c596199fSPaolo Bonzini //! ``` 138c596199fSPaolo Bonzini //! 139c596199fSPaolo Bonzini //! Borrows for `BqlRefCell<T>`s are tracked at _runtime_, unlike Rust's native 140c596199fSPaolo Bonzini //! reference types which are entirely tracked statically, at compile time. 141c596199fSPaolo Bonzini //! Multiple immutable borrows are allowed via [`borrow`](BqlRefCell::borrow), 142c596199fSPaolo Bonzini //! or a single mutable borrow via [`borrow_mut`](BqlRefCell::borrow_mut). The 143c596199fSPaolo Bonzini //! thread will panic if these rules are violated or if the BQL is not held. 1440b9d05e3SPaolo Bonzini //! 1450b9d05e3SPaolo Bonzini //! ## Opaque wrappers 1460b9d05e3SPaolo Bonzini //! 1470b9d05e3SPaolo Bonzini //! The cell types from the previous section are useful at the boundaries 1480b9d05e3SPaolo Bonzini //! of code that requires interior mutability. When writing glue code that 1490b9d05e3SPaolo Bonzini //! interacts directly with C structs, however, it is useful to operate 1500b9d05e3SPaolo Bonzini //! at a lower level. 1510b9d05e3SPaolo Bonzini //! 1520b9d05e3SPaolo Bonzini //! C functions often violate Rust's fundamental assumptions about memory 1530b9d05e3SPaolo Bonzini //! safety by modifying memory even if it is shared. Furthermore, C structs 1540b9d05e3SPaolo Bonzini //! often start their life uninitialized and may be populated lazily. 1550b9d05e3SPaolo Bonzini //! 1560b9d05e3SPaolo Bonzini //! For this reason, this module provides the [`Opaque<T>`] type to opt out 1570b9d05e3SPaolo Bonzini //! of Rust's usual guarantees about the wrapped type. Access to the wrapped 1580b9d05e3SPaolo Bonzini //! value is always through raw pointers, obtained via methods like 1590b9d05e3SPaolo Bonzini //! [`as_mut_ptr()`](Opaque::as_mut_ptr) and [`as_ptr()`](Opaque::as_ptr). These 1600b9d05e3SPaolo Bonzini //! pointers can then be passed to C functions or dereferenced; both actions 1610b9d05e3SPaolo Bonzini //! require `unsafe` blocks, making it clear where safety guarantees must be 1620b9d05e3SPaolo Bonzini //! manually verified. For example 1630b9d05e3SPaolo Bonzini //! 1640b9d05e3SPaolo Bonzini //! ```ignore 1650b9d05e3SPaolo Bonzini //! unsafe { 1660b9d05e3SPaolo Bonzini //! let state = Opaque::<MyStruct>::uninit(); 1670b9d05e3SPaolo Bonzini //! qemu_struct_init(state.as_mut_ptr()); 1680b9d05e3SPaolo Bonzini //! } 1690b9d05e3SPaolo Bonzini //! ``` 1700b9d05e3SPaolo Bonzini //! 1710b9d05e3SPaolo Bonzini //! [`Opaque<T>`] will usually be wrapped one level further, so that 1720b9d05e3SPaolo Bonzini //! bridge methods can be added to the wrapper: 1730b9d05e3SPaolo Bonzini //! 1740b9d05e3SPaolo Bonzini //! ```ignore 1750b9d05e3SPaolo Bonzini //! pub struct MyStruct(Opaque<bindings::MyStruct>); 1760b9d05e3SPaolo Bonzini //! 1770b9d05e3SPaolo Bonzini //! impl MyStruct { 1780b9d05e3SPaolo Bonzini //! fn new() -> Pin<Box<MyStruct>> { 1790b9d05e3SPaolo Bonzini //! let result = Box::pin(unsafe { Opaque::uninit() }); 1800b9d05e3SPaolo Bonzini //! unsafe { qemu_struct_init(result.as_mut_ptr()) }; 1810b9d05e3SPaolo Bonzini //! result 1820b9d05e3SPaolo Bonzini //! } 1830b9d05e3SPaolo Bonzini //! } 1840b9d05e3SPaolo Bonzini //! ``` 1850b9d05e3SPaolo Bonzini //! 1860b9d05e3SPaolo Bonzini //! This pattern of wrapping bindgen-generated types in [`Opaque<T>`] provides 1870b9d05e3SPaolo Bonzini //! several advantages: 1880b9d05e3SPaolo Bonzini //! 1890b9d05e3SPaolo Bonzini //! * The choice of traits to be implemented is not limited by the 1900b9d05e3SPaolo Bonzini //! bindgen-generated code. For example, [`Drop`] can be added without 1910b9d05e3SPaolo Bonzini //! disabling [`Copy`] on the underlying bindgen type 1920b9d05e3SPaolo Bonzini //! 1930b9d05e3SPaolo Bonzini //! * [`Send`] and [`Sync`] implementations can be controlled by the wrapper 1940b9d05e3SPaolo Bonzini //! type rather than being automatically derived from the C struct's layout 1950b9d05e3SPaolo Bonzini //! 1960b9d05e3SPaolo Bonzini //! * Methods can be implemented in a separate crate from the bindgen-generated 1970b9d05e3SPaolo Bonzini //! bindings 1980b9d05e3SPaolo Bonzini //! 1990b9d05e3SPaolo Bonzini //! * [`Debug`](std::fmt::Debug) and [`Display`](std::fmt::Display) 2000b9d05e3SPaolo Bonzini //! implementations can be customized to be more readable than the raw C 2010b9d05e3SPaolo Bonzini //! struct representation 2020b9d05e3SPaolo Bonzini //! 2030b9d05e3SPaolo Bonzini //! The [`Opaque<T>`] type does not include BQL validation; it is possible to 2040b9d05e3SPaolo Bonzini //! assert in the code that the right lock is taken, to use it together 2050b9d05e3SPaolo Bonzini //! with a custom lock guard type, or to let C code take the lock, as 2060b9d05e3SPaolo Bonzini //! appropriate. It is also possible to use it with non-thread-safe 2070b9d05e3SPaolo Bonzini //! types, since by default (unlike [`BqlCell`] and [`BqlRefCell`] 2080b9d05e3SPaolo Bonzini //! it is neither `Sync` nor `Send`. 2090b9d05e3SPaolo Bonzini //! 2100b9d05e3SPaolo Bonzini //! While [`Opaque<T>`] is necessary for C interop, it should be used sparingly 2110b9d05e3SPaolo Bonzini //! and only at FFI boundaries. For QEMU-specific types that need interior 2120b9d05e3SPaolo Bonzini //! mutability, prefer [`BqlCell`] or [`BqlRefCell`]. 2138e194c0eSPaolo Bonzini 214c596199fSPaolo Bonzini use std::{ 215c596199fSPaolo Bonzini cell::{Cell, UnsafeCell}, 216c596199fSPaolo Bonzini cmp::Ordering, 217c596199fSPaolo Bonzini fmt, 2180b9d05e3SPaolo Bonzini marker::{PhantomData, PhantomPinned}, 2190b9d05e3SPaolo Bonzini mem::{self, MaybeUninit}, 220c596199fSPaolo Bonzini ops::{Deref, DerefMut}, 221c596199fSPaolo Bonzini ptr::NonNull, 222c596199fSPaolo Bonzini }; 2238e194c0eSPaolo Bonzini 2248e194c0eSPaolo Bonzini use crate::bindings; 2258e194c0eSPaolo Bonzini 226d4873c5dSPaolo Bonzini /// An internal function that is used by doctests. 227d4873c5dSPaolo Bonzini pub fn bql_start_test() { 228d4873c5dSPaolo Bonzini if cfg!(MESON) { 229d4873c5dSPaolo Bonzini // SAFETY: integration tests are run with --test-threads=1, while 230d4873c5dSPaolo Bonzini // unit tests and doctests are not multithreaded and do not have 231d4873c5dSPaolo Bonzini // any BQL-protected data. Just set bql_locked to true. 232d4873c5dSPaolo Bonzini unsafe { 233d4873c5dSPaolo Bonzini bindings::rust_bql_mock_lock(); 234d4873c5dSPaolo Bonzini } 235d4873c5dSPaolo Bonzini } 236d4873c5dSPaolo Bonzini } 237d4873c5dSPaolo Bonzini 2388e194c0eSPaolo Bonzini pub fn bql_locked() -> bool { 2398e194c0eSPaolo Bonzini // SAFETY: the function does nothing but return a thread-local bool 2408e194c0eSPaolo Bonzini !cfg!(MESON) || unsafe { bindings::bql_locked() } 2418e194c0eSPaolo Bonzini } 2428e194c0eSPaolo Bonzini 243c596199fSPaolo Bonzini fn bql_block_unlock(increase: bool) { 244c596199fSPaolo Bonzini if cfg!(MESON) { 245c596199fSPaolo Bonzini // SAFETY: this only adjusts a counter 246c596199fSPaolo Bonzini unsafe { 247c596199fSPaolo Bonzini bindings::bql_block_unlock(increase); 248c596199fSPaolo Bonzini } 249c596199fSPaolo Bonzini } 250c596199fSPaolo Bonzini } 251c596199fSPaolo Bonzini 2528e194c0eSPaolo Bonzini /// A mutable memory location that is protected by the Big QEMU Lock. 2538e194c0eSPaolo Bonzini /// 2548e194c0eSPaolo Bonzini /// # Memory layout 2558e194c0eSPaolo Bonzini /// 2568e194c0eSPaolo Bonzini /// `BqlCell<T>` has the same in-memory representation as its inner type `T`. 2578e194c0eSPaolo Bonzini #[repr(transparent)] 2588e194c0eSPaolo Bonzini pub struct BqlCell<T> { 2598e194c0eSPaolo Bonzini value: UnsafeCell<T>, 2608e194c0eSPaolo Bonzini } 2618e194c0eSPaolo Bonzini 2628e194c0eSPaolo Bonzini // SAFETY: Same as for std::sync::Mutex. In the end this *is* a Mutex, 2638e194c0eSPaolo Bonzini // except it is stored out-of-line 2648e194c0eSPaolo Bonzini unsafe impl<T: Send> Send for BqlCell<T> {} 2658e194c0eSPaolo Bonzini unsafe impl<T: Send> Sync for BqlCell<T> {} 2668e194c0eSPaolo Bonzini 2678e194c0eSPaolo Bonzini impl<T: Copy> Clone for BqlCell<T> { 2688e194c0eSPaolo Bonzini #[inline] 2698e194c0eSPaolo Bonzini fn clone(&self) -> BqlCell<T> { 2708e194c0eSPaolo Bonzini BqlCell::new(self.get()) 2718e194c0eSPaolo Bonzini } 2728e194c0eSPaolo Bonzini } 2738e194c0eSPaolo Bonzini 2748e194c0eSPaolo Bonzini impl<T: Default> Default for BqlCell<T> { 2758e194c0eSPaolo Bonzini /// Creates a `BqlCell<T>`, with the `Default` value for T. 2768e194c0eSPaolo Bonzini #[inline] 2778e194c0eSPaolo Bonzini fn default() -> BqlCell<T> { 2788e194c0eSPaolo Bonzini BqlCell::new(Default::default()) 2798e194c0eSPaolo Bonzini } 2808e194c0eSPaolo Bonzini } 2818e194c0eSPaolo Bonzini 2828e194c0eSPaolo Bonzini impl<T: PartialEq + Copy> PartialEq for BqlCell<T> { 2838e194c0eSPaolo Bonzini #[inline] 2848e194c0eSPaolo Bonzini fn eq(&self, other: &BqlCell<T>) -> bool { 2858e194c0eSPaolo Bonzini self.get() == other.get() 2868e194c0eSPaolo Bonzini } 2878e194c0eSPaolo Bonzini } 2888e194c0eSPaolo Bonzini 2898e194c0eSPaolo Bonzini impl<T: Eq + Copy> Eq for BqlCell<T> {} 2908e194c0eSPaolo Bonzini 2918e194c0eSPaolo Bonzini impl<T: PartialOrd + Copy> PartialOrd for BqlCell<T> { 2928e194c0eSPaolo Bonzini #[inline] 2938e194c0eSPaolo Bonzini fn partial_cmp(&self, other: &BqlCell<T>) -> Option<Ordering> { 2948e194c0eSPaolo Bonzini self.get().partial_cmp(&other.get()) 2958e194c0eSPaolo Bonzini } 2968e194c0eSPaolo Bonzini } 2978e194c0eSPaolo Bonzini 2988e194c0eSPaolo Bonzini impl<T: Ord + Copy> Ord for BqlCell<T> { 2998e194c0eSPaolo Bonzini #[inline] 3008e194c0eSPaolo Bonzini fn cmp(&self, other: &BqlCell<T>) -> Ordering { 3018e194c0eSPaolo Bonzini self.get().cmp(&other.get()) 3028e194c0eSPaolo Bonzini } 3038e194c0eSPaolo Bonzini } 3048e194c0eSPaolo Bonzini 3058e194c0eSPaolo Bonzini impl<T> From<T> for BqlCell<T> { 3068e194c0eSPaolo Bonzini /// Creates a new `BqlCell<T>` containing the given value. 3078e194c0eSPaolo Bonzini fn from(t: T) -> BqlCell<T> { 3088e194c0eSPaolo Bonzini BqlCell::new(t) 3098e194c0eSPaolo Bonzini } 3108e194c0eSPaolo Bonzini } 3118e194c0eSPaolo Bonzini 3128e194c0eSPaolo Bonzini impl<T: fmt::Debug + Copy> fmt::Debug for BqlCell<T> { 3138e194c0eSPaolo Bonzini fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { 3148e194c0eSPaolo Bonzini self.get().fmt(f) 3158e194c0eSPaolo Bonzini } 3168e194c0eSPaolo Bonzini } 3178e194c0eSPaolo Bonzini 3188e194c0eSPaolo Bonzini impl<T: fmt::Display + Copy> fmt::Display for BqlCell<T> { 3198e194c0eSPaolo Bonzini fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { 3208e194c0eSPaolo Bonzini self.get().fmt(f) 3218e194c0eSPaolo Bonzini } 3228e194c0eSPaolo Bonzini } 3238e194c0eSPaolo Bonzini 3248e194c0eSPaolo Bonzini impl<T> BqlCell<T> { 3258e194c0eSPaolo Bonzini /// Creates a new `BqlCell` containing the given value. 3268e194c0eSPaolo Bonzini /// 3278e194c0eSPaolo Bonzini /// # Examples 3288e194c0eSPaolo Bonzini /// 3298e194c0eSPaolo Bonzini /// ``` 3308e194c0eSPaolo Bonzini /// use qemu_api::cell::BqlCell; 331d4873c5dSPaolo Bonzini /// # qemu_api::cell::bql_start_test(); 3328e194c0eSPaolo Bonzini /// 3338e194c0eSPaolo Bonzini /// let c = BqlCell::new(5); 3348e194c0eSPaolo Bonzini /// ``` 3358e194c0eSPaolo Bonzini #[inline] 3368e194c0eSPaolo Bonzini pub const fn new(value: T) -> BqlCell<T> { 3378e194c0eSPaolo Bonzini BqlCell { 3388e194c0eSPaolo Bonzini value: UnsafeCell::new(value), 3398e194c0eSPaolo Bonzini } 3408e194c0eSPaolo Bonzini } 3418e194c0eSPaolo Bonzini 3428e194c0eSPaolo Bonzini /// Sets the contained value. 3438e194c0eSPaolo Bonzini /// 3448e194c0eSPaolo Bonzini /// # Examples 3458e194c0eSPaolo Bonzini /// 3468e194c0eSPaolo Bonzini /// ``` 3478e194c0eSPaolo Bonzini /// use qemu_api::cell::BqlCell; 348d4873c5dSPaolo Bonzini /// # qemu_api::cell::bql_start_test(); 3498e194c0eSPaolo Bonzini /// 3508e194c0eSPaolo Bonzini /// let c = BqlCell::new(5); 3518e194c0eSPaolo Bonzini /// 3528e194c0eSPaolo Bonzini /// c.set(10); 3538e194c0eSPaolo Bonzini /// ``` 3548e194c0eSPaolo Bonzini #[inline] 3558e194c0eSPaolo Bonzini pub fn set(&self, val: T) { 3568e194c0eSPaolo Bonzini self.replace(val); 3578e194c0eSPaolo Bonzini } 3588e194c0eSPaolo Bonzini 3598e194c0eSPaolo Bonzini /// Replaces the contained value with `val`, and returns the old contained 3608e194c0eSPaolo Bonzini /// value. 3618e194c0eSPaolo Bonzini /// 3628e194c0eSPaolo Bonzini /// # Examples 3638e194c0eSPaolo Bonzini /// 3648e194c0eSPaolo Bonzini /// ``` 3658e194c0eSPaolo Bonzini /// use qemu_api::cell::BqlCell; 366d4873c5dSPaolo Bonzini /// # qemu_api::cell::bql_start_test(); 3678e194c0eSPaolo Bonzini /// 3688e194c0eSPaolo Bonzini /// let cell = BqlCell::new(5); 3698e194c0eSPaolo Bonzini /// assert_eq!(cell.get(), 5); 3708e194c0eSPaolo Bonzini /// assert_eq!(cell.replace(10), 5); 3718e194c0eSPaolo Bonzini /// assert_eq!(cell.get(), 10); 3728e194c0eSPaolo Bonzini /// ``` 3738e194c0eSPaolo Bonzini #[inline] 3748e194c0eSPaolo Bonzini pub fn replace(&self, val: T) -> T { 3758e194c0eSPaolo Bonzini assert!(bql_locked()); 3768e194c0eSPaolo Bonzini // SAFETY: This can cause data races if called from multiple threads, 3778e194c0eSPaolo Bonzini // but it won't happen as long as C code accesses the value 3788e194c0eSPaolo Bonzini // under BQL protection only. 3798e194c0eSPaolo Bonzini mem::replace(unsafe { &mut *self.value.get() }, val) 3808e194c0eSPaolo Bonzini } 3818e194c0eSPaolo Bonzini 3828e194c0eSPaolo Bonzini /// Unwraps the value, consuming the cell. 3838e194c0eSPaolo Bonzini /// 3848e194c0eSPaolo Bonzini /// # Examples 3858e194c0eSPaolo Bonzini /// 3868e194c0eSPaolo Bonzini /// ``` 3878e194c0eSPaolo Bonzini /// use qemu_api::cell::BqlCell; 388d4873c5dSPaolo Bonzini /// # qemu_api::cell::bql_start_test(); 3898e194c0eSPaolo Bonzini /// 3908e194c0eSPaolo Bonzini /// let c = BqlCell::new(5); 3918e194c0eSPaolo Bonzini /// let five = c.into_inner(); 3928e194c0eSPaolo Bonzini /// 3938e194c0eSPaolo Bonzini /// assert_eq!(five, 5); 3948e194c0eSPaolo Bonzini /// ``` 3958e194c0eSPaolo Bonzini pub fn into_inner(self) -> T { 3968e194c0eSPaolo Bonzini assert!(bql_locked()); 3978e194c0eSPaolo Bonzini self.value.into_inner() 3988e194c0eSPaolo Bonzini } 3998e194c0eSPaolo Bonzini } 4008e194c0eSPaolo Bonzini 4018e194c0eSPaolo Bonzini impl<T: Copy> BqlCell<T> { 4028e194c0eSPaolo Bonzini /// Returns a copy of the contained value. 4038e194c0eSPaolo Bonzini /// 4048e194c0eSPaolo Bonzini /// # Examples 4058e194c0eSPaolo Bonzini /// 4068e194c0eSPaolo Bonzini /// ``` 4078e194c0eSPaolo Bonzini /// use qemu_api::cell::BqlCell; 408d4873c5dSPaolo Bonzini /// # qemu_api::cell::bql_start_test(); 4098e194c0eSPaolo Bonzini /// 4108e194c0eSPaolo Bonzini /// let c = BqlCell::new(5); 4118e194c0eSPaolo Bonzini /// 4128e194c0eSPaolo Bonzini /// let five = c.get(); 4138e194c0eSPaolo Bonzini /// ``` 4148e194c0eSPaolo Bonzini #[inline] 4158e194c0eSPaolo Bonzini pub fn get(&self) -> T { 4168e194c0eSPaolo Bonzini assert!(bql_locked()); 4178e194c0eSPaolo Bonzini // SAFETY: This can cause data races if called from multiple threads, 4188e194c0eSPaolo Bonzini // but it won't happen as long as C code accesses the value 4198e194c0eSPaolo Bonzini // under BQL protection only. 4208e194c0eSPaolo Bonzini unsafe { *self.value.get() } 4218e194c0eSPaolo Bonzini } 4228e194c0eSPaolo Bonzini } 4238e194c0eSPaolo Bonzini 4248e194c0eSPaolo Bonzini impl<T> BqlCell<T> { 4258e194c0eSPaolo Bonzini /// Returns a raw pointer to the underlying data in this cell. 4268e194c0eSPaolo Bonzini /// 4278e194c0eSPaolo Bonzini /// # Examples 4288e194c0eSPaolo Bonzini /// 4298e194c0eSPaolo Bonzini /// ``` 4308e194c0eSPaolo Bonzini /// use qemu_api::cell::BqlCell; 431d4873c5dSPaolo Bonzini /// # qemu_api::cell::bql_start_test(); 4328e194c0eSPaolo Bonzini /// 4338e194c0eSPaolo Bonzini /// let c = BqlCell::new(5); 4348e194c0eSPaolo Bonzini /// 4358e194c0eSPaolo Bonzini /// let ptr = c.as_ptr(); 4368e194c0eSPaolo Bonzini /// ``` 4378e194c0eSPaolo Bonzini #[inline] 4388e194c0eSPaolo Bonzini pub const fn as_ptr(&self) -> *mut T { 4398e194c0eSPaolo Bonzini self.value.get() 4408e194c0eSPaolo Bonzini } 4418e194c0eSPaolo Bonzini } 4428e194c0eSPaolo Bonzini 4438e194c0eSPaolo Bonzini impl<T: Default> BqlCell<T> { 4448e194c0eSPaolo Bonzini /// Takes the value of the cell, leaving `Default::default()` in its place. 4458e194c0eSPaolo Bonzini /// 4468e194c0eSPaolo Bonzini /// # Examples 4478e194c0eSPaolo Bonzini /// 4488e194c0eSPaolo Bonzini /// ``` 4498e194c0eSPaolo Bonzini /// use qemu_api::cell::BqlCell; 450d4873c5dSPaolo Bonzini /// # qemu_api::cell::bql_start_test(); 4518e194c0eSPaolo Bonzini /// 4528e194c0eSPaolo Bonzini /// let c = BqlCell::new(5); 4538e194c0eSPaolo Bonzini /// let five = c.take(); 4548e194c0eSPaolo Bonzini /// 4558e194c0eSPaolo Bonzini /// assert_eq!(five, 5); 4568e194c0eSPaolo Bonzini /// assert_eq!(c.into_inner(), 0); 4578e194c0eSPaolo Bonzini /// ``` 4588e194c0eSPaolo Bonzini pub fn take(&self) -> T { 4598e194c0eSPaolo Bonzini self.replace(Default::default()) 4608e194c0eSPaolo Bonzini } 4618e194c0eSPaolo Bonzini } 462c596199fSPaolo Bonzini 463c596199fSPaolo Bonzini /// A mutable memory location with dynamically checked borrow rules, 464c596199fSPaolo Bonzini /// protected by the Big QEMU Lock. 465c596199fSPaolo Bonzini /// 466c596199fSPaolo Bonzini /// See the [module-level documentation](self) for more. 467c596199fSPaolo Bonzini /// 468c596199fSPaolo Bonzini /// # Memory layout 469c596199fSPaolo Bonzini /// 470c596199fSPaolo Bonzini /// `BqlRefCell<T>` starts with the same in-memory representation as its 471c596199fSPaolo Bonzini /// inner type `T`. 472c596199fSPaolo Bonzini #[repr(C)] 473c596199fSPaolo Bonzini pub struct BqlRefCell<T> { 474c596199fSPaolo Bonzini // It is important that this is the first field (which is not the case 475c596199fSPaolo Bonzini // for std::cell::BqlRefCell), so that we can use offset_of! on it. 476c596199fSPaolo Bonzini // UnsafeCell and repr(C) both prevent usage of niches. 477c596199fSPaolo Bonzini value: UnsafeCell<T>, 478c596199fSPaolo Bonzini borrow: Cell<BorrowFlag>, 479c596199fSPaolo Bonzini // Stores the location of the earliest currently active borrow. 480c596199fSPaolo Bonzini // This gets updated whenever we go from having zero borrows 481c596199fSPaolo Bonzini // to having a single borrow. When a borrow occurs, this gets included 482c596199fSPaolo Bonzini // in the panic message 483c596199fSPaolo Bonzini #[cfg(feature = "debug_cell")] 484c596199fSPaolo Bonzini borrowed_at: Cell<Option<&'static std::panic::Location<'static>>>, 485c596199fSPaolo Bonzini } 486c596199fSPaolo Bonzini 487c596199fSPaolo Bonzini // Positive values represent the number of `BqlRef` active. Negative values 488c596199fSPaolo Bonzini // represent the number of `BqlRefMut` active. Right now QEMU's implementation 489c596199fSPaolo Bonzini // does not allow to create `BqlRefMut`s that refer to distinct, nonoverlapping 490c596199fSPaolo Bonzini // components of a `BqlRefCell` (e.g., different ranges of a slice). 491c596199fSPaolo Bonzini // 492c596199fSPaolo Bonzini // `BqlRef` and `BqlRefMut` are both two words in size, and so there will likely 493c596199fSPaolo Bonzini // never be enough `BqlRef`s or `BqlRefMut`s in existence to overflow half of 494c596199fSPaolo Bonzini // the `usize` range. Thus, a `BorrowFlag` will probably never overflow or 495c596199fSPaolo Bonzini // underflow. However, this is not a guarantee, as a pathological program could 496c596199fSPaolo Bonzini // repeatedly create and then mem::forget `BqlRef`s or `BqlRefMut`s. Thus, all 497c596199fSPaolo Bonzini // code must explicitly check for overflow and underflow in order to avoid 498c596199fSPaolo Bonzini // unsafety, or at least behave correctly in the event that overflow or 499c596199fSPaolo Bonzini // underflow happens (e.g., see BorrowRef::new). 500c596199fSPaolo Bonzini type BorrowFlag = isize; 501c596199fSPaolo Bonzini const UNUSED: BorrowFlag = 0; 502c596199fSPaolo Bonzini 503c596199fSPaolo Bonzini #[inline(always)] 504c596199fSPaolo Bonzini const fn is_writing(x: BorrowFlag) -> bool { 505c596199fSPaolo Bonzini x < UNUSED 506c596199fSPaolo Bonzini } 507c596199fSPaolo Bonzini 508c596199fSPaolo Bonzini #[inline(always)] 509c596199fSPaolo Bonzini const fn is_reading(x: BorrowFlag) -> bool { 510c596199fSPaolo Bonzini x > UNUSED 511c596199fSPaolo Bonzini } 512c596199fSPaolo Bonzini 513c596199fSPaolo Bonzini impl<T> BqlRefCell<T> { 514c596199fSPaolo Bonzini /// Creates a new `BqlRefCell` containing `value`. 515c596199fSPaolo Bonzini /// 516c596199fSPaolo Bonzini /// # Examples 517c596199fSPaolo Bonzini /// 518c596199fSPaolo Bonzini /// ``` 519c596199fSPaolo Bonzini /// use qemu_api::cell::BqlRefCell; 520c596199fSPaolo Bonzini /// 521c596199fSPaolo Bonzini /// let c = BqlRefCell::new(5); 522c596199fSPaolo Bonzini /// ``` 523c596199fSPaolo Bonzini #[inline] 524c596199fSPaolo Bonzini pub const fn new(value: T) -> BqlRefCell<T> { 525c596199fSPaolo Bonzini BqlRefCell { 526c596199fSPaolo Bonzini value: UnsafeCell::new(value), 527c596199fSPaolo Bonzini borrow: Cell::new(UNUSED), 528c596199fSPaolo Bonzini #[cfg(feature = "debug_cell")] 529c596199fSPaolo Bonzini borrowed_at: Cell::new(None), 530c596199fSPaolo Bonzini } 531c596199fSPaolo Bonzini } 532c596199fSPaolo Bonzini } 533c596199fSPaolo Bonzini 534c596199fSPaolo Bonzini // This ensures the panicking code is outlined from `borrow_mut` for 535c596199fSPaolo Bonzini // `BqlRefCell`. 536c596199fSPaolo Bonzini #[inline(never)] 537c596199fSPaolo Bonzini #[cold] 538c596199fSPaolo Bonzini #[cfg(feature = "debug_cell")] 539c596199fSPaolo Bonzini fn panic_already_borrowed(source: &Cell<Option<&'static std::panic::Location<'static>>>) -> ! { 540c596199fSPaolo Bonzini // If a borrow occurred, then we must already have an outstanding borrow, 541c596199fSPaolo Bonzini // so `borrowed_at` will be `Some` 542c596199fSPaolo Bonzini panic!("already borrowed at {:?}", source.take().unwrap()) 543c596199fSPaolo Bonzini } 544c596199fSPaolo Bonzini 545c596199fSPaolo Bonzini #[inline(never)] 546c596199fSPaolo Bonzini #[cold] 547c596199fSPaolo Bonzini #[cfg(not(feature = "debug_cell"))] 548c596199fSPaolo Bonzini fn panic_already_borrowed() -> ! { 549c596199fSPaolo Bonzini panic!("already borrowed") 550c596199fSPaolo Bonzini } 551c596199fSPaolo Bonzini 552c596199fSPaolo Bonzini impl<T> BqlRefCell<T> { 553c596199fSPaolo Bonzini #[inline] 554c596199fSPaolo Bonzini #[allow(clippy::unused_self)] 555c596199fSPaolo Bonzini fn panic_already_borrowed(&self) -> ! { 556c596199fSPaolo Bonzini #[cfg(feature = "debug_cell")] 557c596199fSPaolo Bonzini { 558c596199fSPaolo Bonzini panic_already_borrowed(&self.borrowed_at) 559c596199fSPaolo Bonzini } 560c596199fSPaolo Bonzini #[cfg(not(feature = "debug_cell"))] 561c596199fSPaolo Bonzini { 562c596199fSPaolo Bonzini panic_already_borrowed() 563c596199fSPaolo Bonzini } 564c596199fSPaolo Bonzini } 565c596199fSPaolo Bonzini 566c596199fSPaolo Bonzini /// Immutably borrows the wrapped value. 567c596199fSPaolo Bonzini /// 568c596199fSPaolo Bonzini /// The borrow lasts until the returned `BqlRef` exits scope. Multiple 569c596199fSPaolo Bonzini /// immutable borrows can be taken out at the same time. 570c596199fSPaolo Bonzini /// 571c596199fSPaolo Bonzini /// # Panics 572c596199fSPaolo Bonzini /// 573c596199fSPaolo Bonzini /// Panics if the value is currently mutably borrowed. 574c596199fSPaolo Bonzini /// 575c596199fSPaolo Bonzini /// # Examples 576c596199fSPaolo Bonzini /// 577c596199fSPaolo Bonzini /// ``` 578c596199fSPaolo Bonzini /// use qemu_api::cell::BqlRefCell; 579d4873c5dSPaolo Bonzini /// # qemu_api::cell::bql_start_test(); 580c596199fSPaolo Bonzini /// 581c596199fSPaolo Bonzini /// let c = BqlRefCell::new(5); 582c596199fSPaolo Bonzini /// 583c596199fSPaolo Bonzini /// let borrowed_five = c.borrow(); 584c596199fSPaolo Bonzini /// let borrowed_five2 = c.borrow(); 585c596199fSPaolo Bonzini /// ``` 586c596199fSPaolo Bonzini /// 587c596199fSPaolo Bonzini /// An example of panic: 588c596199fSPaolo Bonzini /// 589c596199fSPaolo Bonzini /// ```should_panic 590c596199fSPaolo Bonzini /// use qemu_api::cell::BqlRefCell; 591d4873c5dSPaolo Bonzini /// # qemu_api::cell::bql_start_test(); 592c596199fSPaolo Bonzini /// 593c596199fSPaolo Bonzini /// let c = BqlRefCell::new(5); 594c596199fSPaolo Bonzini /// 595c596199fSPaolo Bonzini /// let m = c.borrow_mut(); 596c596199fSPaolo Bonzini /// let b = c.borrow(); // this causes a panic 597c596199fSPaolo Bonzini /// ``` 598c596199fSPaolo Bonzini #[inline] 599c596199fSPaolo Bonzini #[track_caller] 600c596199fSPaolo Bonzini pub fn borrow(&self) -> BqlRef<'_, T> { 601c596199fSPaolo Bonzini if let Some(b) = BorrowRef::new(&self.borrow) { 602c596199fSPaolo Bonzini // `borrowed_at` is always the *first* active borrow 603c596199fSPaolo Bonzini if b.borrow.get() == 1 { 604c596199fSPaolo Bonzini #[cfg(feature = "debug_cell")] 605c596199fSPaolo Bonzini self.borrowed_at.set(Some(std::panic::Location::caller())); 606c596199fSPaolo Bonzini } 607c596199fSPaolo Bonzini 608c596199fSPaolo Bonzini bql_block_unlock(true); 609c596199fSPaolo Bonzini 610c596199fSPaolo Bonzini // SAFETY: `BorrowRef` ensures that there is only immutable access 611c596199fSPaolo Bonzini // to the value while borrowed. 612c596199fSPaolo Bonzini let value = unsafe { NonNull::new_unchecked(self.value.get()) }; 613c596199fSPaolo Bonzini BqlRef { value, borrow: b } 614c596199fSPaolo Bonzini } else { 615c596199fSPaolo Bonzini self.panic_already_borrowed() 616c596199fSPaolo Bonzini } 617c596199fSPaolo Bonzini } 618c596199fSPaolo Bonzini 619c596199fSPaolo Bonzini /// Mutably borrows the wrapped value. 620c596199fSPaolo Bonzini /// 621c596199fSPaolo Bonzini /// The borrow lasts until the returned `BqlRefMut` or all `BqlRefMut`s 622c596199fSPaolo Bonzini /// derived from it exit scope. The value cannot be borrowed while this 623c596199fSPaolo Bonzini /// borrow is active. 624c596199fSPaolo Bonzini /// 625c596199fSPaolo Bonzini /// # Panics 626c596199fSPaolo Bonzini /// 627c596199fSPaolo Bonzini /// Panics if the value is currently borrowed. 628c596199fSPaolo Bonzini /// 629c596199fSPaolo Bonzini /// # Examples 630c596199fSPaolo Bonzini /// 631c596199fSPaolo Bonzini /// ``` 632c596199fSPaolo Bonzini /// use qemu_api::cell::BqlRefCell; 633d4873c5dSPaolo Bonzini /// # qemu_api::cell::bql_start_test(); 634c596199fSPaolo Bonzini /// 635c596199fSPaolo Bonzini /// let c = BqlRefCell::new("hello".to_owned()); 636c596199fSPaolo Bonzini /// 637c596199fSPaolo Bonzini /// *c.borrow_mut() = "bonjour".to_owned(); 638c596199fSPaolo Bonzini /// 639c596199fSPaolo Bonzini /// assert_eq!(&*c.borrow(), "bonjour"); 640c596199fSPaolo Bonzini /// ``` 641c596199fSPaolo Bonzini /// 642c596199fSPaolo Bonzini /// An example of panic: 643c596199fSPaolo Bonzini /// 644c596199fSPaolo Bonzini /// ```should_panic 645c596199fSPaolo Bonzini /// use qemu_api::cell::BqlRefCell; 646d4873c5dSPaolo Bonzini /// # qemu_api::cell::bql_start_test(); 647c596199fSPaolo Bonzini /// 648c596199fSPaolo Bonzini /// let c = BqlRefCell::new(5); 649c596199fSPaolo Bonzini /// let m = c.borrow(); 650c596199fSPaolo Bonzini /// 651c596199fSPaolo Bonzini /// let b = c.borrow_mut(); // this causes a panic 652c596199fSPaolo Bonzini /// ``` 653c596199fSPaolo Bonzini #[inline] 654c596199fSPaolo Bonzini #[track_caller] 655c596199fSPaolo Bonzini pub fn borrow_mut(&self) -> BqlRefMut<'_, T> { 656c596199fSPaolo Bonzini if let Some(b) = BorrowRefMut::new(&self.borrow) { 657c596199fSPaolo Bonzini #[cfg(feature = "debug_cell")] 658c596199fSPaolo Bonzini { 659c596199fSPaolo Bonzini self.borrowed_at.set(Some(std::panic::Location::caller())); 660c596199fSPaolo Bonzini } 661c596199fSPaolo Bonzini 662c596199fSPaolo Bonzini // SAFETY: this only adjusts a counter 663c596199fSPaolo Bonzini bql_block_unlock(true); 664c596199fSPaolo Bonzini 665c596199fSPaolo Bonzini // SAFETY: `BorrowRefMut` guarantees unique access. 666c596199fSPaolo Bonzini let value = unsafe { NonNull::new_unchecked(self.value.get()) }; 667c596199fSPaolo Bonzini BqlRefMut { 668c596199fSPaolo Bonzini value, 669c596199fSPaolo Bonzini _borrow: b, 670c596199fSPaolo Bonzini marker: PhantomData, 671c596199fSPaolo Bonzini } 672c596199fSPaolo Bonzini } else { 673c596199fSPaolo Bonzini self.panic_already_borrowed() 674c596199fSPaolo Bonzini } 675c596199fSPaolo Bonzini } 676c596199fSPaolo Bonzini 677c596199fSPaolo Bonzini /// Returns a raw pointer to the underlying data in this cell. 678c596199fSPaolo Bonzini /// 679c596199fSPaolo Bonzini /// # Examples 680c596199fSPaolo Bonzini /// 681c596199fSPaolo Bonzini /// ``` 682c596199fSPaolo Bonzini /// use qemu_api::cell::BqlRefCell; 683c596199fSPaolo Bonzini /// 684c596199fSPaolo Bonzini /// let c = BqlRefCell::new(5); 685c596199fSPaolo Bonzini /// 686c596199fSPaolo Bonzini /// let ptr = c.as_ptr(); 687c596199fSPaolo Bonzini /// ``` 688c596199fSPaolo Bonzini #[inline] 689c596199fSPaolo Bonzini pub const fn as_ptr(&self) -> *mut T { 690c596199fSPaolo Bonzini self.value.get() 691c596199fSPaolo Bonzini } 692c596199fSPaolo Bonzini } 693c596199fSPaolo Bonzini 694c596199fSPaolo Bonzini // SAFETY: Same as for std::sync::Mutex. In the end this is a Mutex that is 695c596199fSPaolo Bonzini // stored out-of-line. Even though BqlRefCell includes Cells, they are 696c596199fSPaolo Bonzini // themselves protected by the Big QEMU Lock. Furtheremore, the Big QEMU 697c596199fSPaolo Bonzini // Lock cannot be released while any borrows is active. 698c596199fSPaolo Bonzini unsafe impl<T> Send for BqlRefCell<T> where T: Send {} 699c596199fSPaolo Bonzini unsafe impl<T> Sync for BqlRefCell<T> {} 700c596199fSPaolo Bonzini 701c596199fSPaolo Bonzini impl<T: Clone> Clone for BqlRefCell<T> { 702c596199fSPaolo Bonzini /// # Panics 703c596199fSPaolo Bonzini /// 704c596199fSPaolo Bonzini /// Panics if the value is currently mutably borrowed. 705c596199fSPaolo Bonzini #[inline] 706c596199fSPaolo Bonzini #[track_caller] 707c596199fSPaolo Bonzini fn clone(&self) -> BqlRefCell<T> { 708c596199fSPaolo Bonzini BqlRefCell::new(self.borrow().clone()) 709c596199fSPaolo Bonzini } 710c596199fSPaolo Bonzini 711c596199fSPaolo Bonzini /// # Panics 712c596199fSPaolo Bonzini /// 713c596199fSPaolo Bonzini /// Panics if `source` is currently mutably borrowed. 714c596199fSPaolo Bonzini #[inline] 715c596199fSPaolo Bonzini #[track_caller] 716c596199fSPaolo Bonzini fn clone_from(&mut self, source: &Self) { 717c596199fSPaolo Bonzini self.value.get_mut().clone_from(&source.borrow()) 718c596199fSPaolo Bonzini } 719c596199fSPaolo Bonzini } 720c596199fSPaolo Bonzini 721c596199fSPaolo Bonzini impl<T: Default> Default for BqlRefCell<T> { 722c596199fSPaolo Bonzini /// Creates a `BqlRefCell<T>`, with the `Default` value for T. 723c596199fSPaolo Bonzini #[inline] 724c596199fSPaolo Bonzini fn default() -> BqlRefCell<T> { 725c596199fSPaolo Bonzini BqlRefCell::new(Default::default()) 726c596199fSPaolo Bonzini } 727c596199fSPaolo Bonzini } 728c596199fSPaolo Bonzini 729c596199fSPaolo Bonzini impl<T: PartialEq> PartialEq for BqlRefCell<T> { 730c596199fSPaolo Bonzini /// # Panics 731c596199fSPaolo Bonzini /// 732c596199fSPaolo Bonzini /// Panics if the value in either `BqlRefCell` is currently mutably 733c596199fSPaolo Bonzini /// borrowed. 734c596199fSPaolo Bonzini #[inline] 735c596199fSPaolo Bonzini fn eq(&self, other: &BqlRefCell<T>) -> bool { 736c596199fSPaolo Bonzini *self.borrow() == *other.borrow() 737c596199fSPaolo Bonzini } 738c596199fSPaolo Bonzini } 739c596199fSPaolo Bonzini 740c596199fSPaolo Bonzini impl<T: Eq> Eq for BqlRefCell<T> {} 741c596199fSPaolo Bonzini 742c596199fSPaolo Bonzini impl<T: PartialOrd> PartialOrd for BqlRefCell<T> { 743c596199fSPaolo Bonzini /// # Panics 744c596199fSPaolo Bonzini /// 745c596199fSPaolo Bonzini /// Panics if the value in either `BqlRefCell` is currently mutably 746c596199fSPaolo Bonzini /// borrowed. 747c596199fSPaolo Bonzini #[inline] 748c596199fSPaolo Bonzini fn partial_cmp(&self, other: &BqlRefCell<T>) -> Option<Ordering> { 749c596199fSPaolo Bonzini self.borrow().partial_cmp(&*other.borrow()) 750c596199fSPaolo Bonzini } 751c596199fSPaolo Bonzini } 752c596199fSPaolo Bonzini 753c596199fSPaolo Bonzini impl<T: Ord> Ord for BqlRefCell<T> { 754c596199fSPaolo Bonzini /// # Panics 755c596199fSPaolo Bonzini /// 756c596199fSPaolo Bonzini /// Panics if the value in either `BqlRefCell` is currently mutably 757c596199fSPaolo Bonzini /// borrowed. 758c596199fSPaolo Bonzini #[inline] 759c596199fSPaolo Bonzini fn cmp(&self, other: &BqlRefCell<T>) -> Ordering { 760c596199fSPaolo Bonzini self.borrow().cmp(&*other.borrow()) 761c596199fSPaolo Bonzini } 762c596199fSPaolo Bonzini } 763c596199fSPaolo Bonzini 764c596199fSPaolo Bonzini impl<T> From<T> for BqlRefCell<T> { 765c596199fSPaolo Bonzini /// Creates a new `BqlRefCell<T>` containing the given value. 766c596199fSPaolo Bonzini fn from(t: T) -> BqlRefCell<T> { 767c596199fSPaolo Bonzini BqlRefCell::new(t) 768c596199fSPaolo Bonzini } 769c596199fSPaolo Bonzini } 770c596199fSPaolo Bonzini 771c596199fSPaolo Bonzini struct BorrowRef<'b> { 772c596199fSPaolo Bonzini borrow: &'b Cell<BorrowFlag>, 773c596199fSPaolo Bonzini } 774c596199fSPaolo Bonzini 775c596199fSPaolo Bonzini impl<'b> BorrowRef<'b> { 776c596199fSPaolo Bonzini #[inline] 777c596199fSPaolo Bonzini fn new(borrow: &'b Cell<BorrowFlag>) -> Option<BorrowRef<'b>> { 778c596199fSPaolo Bonzini let b = borrow.get().wrapping_add(1); 779c596199fSPaolo Bonzini if !is_reading(b) { 780c596199fSPaolo Bonzini // Incrementing borrow can result in a non-reading value (<= 0) in these cases: 781c596199fSPaolo Bonzini // 1. It was < 0, i.e. there are writing borrows, so we can't allow a read 782c596199fSPaolo Bonzini // borrow due to Rust's reference aliasing rules 783c596199fSPaolo Bonzini // 2. It was isize::MAX (the max amount of reading borrows) and it overflowed 784c596199fSPaolo Bonzini // into isize::MIN (the max amount of writing borrows) so we can't allow an 785c596199fSPaolo Bonzini // additional read borrow because isize can't represent so many read borrows 786c596199fSPaolo Bonzini // (this can only happen if you mem::forget more than a small constant amount 787c596199fSPaolo Bonzini // of `BqlRef`s, which is not good practice) 788c596199fSPaolo Bonzini None 789c596199fSPaolo Bonzini } else { 790c596199fSPaolo Bonzini // Incrementing borrow can result in a reading value (> 0) in these cases: 791c596199fSPaolo Bonzini // 1. It was = 0, i.e. it wasn't borrowed, and we are taking the first read 792c596199fSPaolo Bonzini // borrow 793c596199fSPaolo Bonzini // 2. It was > 0 and < isize::MAX, i.e. there were read borrows, and isize is 794c596199fSPaolo Bonzini // large enough to represent having one more read borrow 795c596199fSPaolo Bonzini borrow.set(b); 796c596199fSPaolo Bonzini Some(BorrowRef { borrow }) 797c596199fSPaolo Bonzini } 798c596199fSPaolo Bonzini } 799c596199fSPaolo Bonzini } 800c596199fSPaolo Bonzini 801c596199fSPaolo Bonzini impl Drop for BorrowRef<'_> { 802c596199fSPaolo Bonzini #[inline] 803c596199fSPaolo Bonzini fn drop(&mut self) { 804c596199fSPaolo Bonzini let borrow = self.borrow.get(); 805c596199fSPaolo Bonzini debug_assert!(is_reading(borrow)); 806c596199fSPaolo Bonzini self.borrow.set(borrow - 1); 807c596199fSPaolo Bonzini bql_block_unlock(false) 808c596199fSPaolo Bonzini } 809c596199fSPaolo Bonzini } 810c596199fSPaolo Bonzini 811c596199fSPaolo Bonzini impl Clone for BorrowRef<'_> { 812c596199fSPaolo Bonzini #[inline] 813c596199fSPaolo Bonzini fn clone(&self) -> Self { 814c596199fSPaolo Bonzini BorrowRef::new(self.borrow).unwrap() 815c596199fSPaolo Bonzini } 816c596199fSPaolo Bonzini } 817c596199fSPaolo Bonzini 818c596199fSPaolo Bonzini /// Wraps a borrowed reference to a value in a `BqlRefCell` box. 819c596199fSPaolo Bonzini /// A wrapper type for an immutably borrowed value from a `BqlRefCell<T>`. 820c596199fSPaolo Bonzini /// 821c596199fSPaolo Bonzini /// See the [module-level documentation](self) for more. 822c596199fSPaolo Bonzini pub struct BqlRef<'b, T: 'b> { 823c596199fSPaolo Bonzini // NB: we use a pointer instead of `&'b T` to avoid `noalias` violations, because a 824c596199fSPaolo Bonzini // `BqlRef` argument doesn't hold immutability for its whole scope, only until it drops. 825c596199fSPaolo Bonzini // `NonNull` is also covariant over `T`, just like we would have with `&T`. 826c596199fSPaolo Bonzini value: NonNull<T>, 827c596199fSPaolo Bonzini borrow: BorrowRef<'b>, 828c596199fSPaolo Bonzini } 829c596199fSPaolo Bonzini 830c596199fSPaolo Bonzini impl<T> Deref for BqlRef<'_, T> { 831c596199fSPaolo Bonzini type Target = T; 832c596199fSPaolo Bonzini 833c596199fSPaolo Bonzini #[inline] 834c596199fSPaolo Bonzini fn deref(&self) -> &T { 835c596199fSPaolo Bonzini // SAFETY: the value is accessible as long as we hold our borrow. 836c596199fSPaolo Bonzini unsafe { self.value.as_ref() } 837c596199fSPaolo Bonzini } 838c596199fSPaolo Bonzini } 839c596199fSPaolo Bonzini 840c596199fSPaolo Bonzini impl<'b, T> BqlRef<'b, T> { 841c596199fSPaolo Bonzini /// Copies a `BqlRef`. 842c596199fSPaolo Bonzini /// 843c596199fSPaolo Bonzini /// The `BqlRefCell` is already immutably borrowed, so this cannot fail. 844c596199fSPaolo Bonzini /// 845c596199fSPaolo Bonzini /// This is an associated function that needs to be used as 846c596199fSPaolo Bonzini /// `BqlRef::clone(...)`. A `Clone` implementation or a method would 847c596199fSPaolo Bonzini /// interfere with the widespread use of `r.borrow().clone()` to clone 848c596199fSPaolo Bonzini /// the contents of a `BqlRefCell`. 849c596199fSPaolo Bonzini #[must_use] 850c596199fSPaolo Bonzini #[inline] 851c596199fSPaolo Bonzini #[allow(clippy::should_implement_trait)] 852c596199fSPaolo Bonzini pub fn clone(orig: &BqlRef<'b, T>) -> BqlRef<'b, T> { 853c596199fSPaolo Bonzini BqlRef { 854c596199fSPaolo Bonzini value: orig.value, 855c596199fSPaolo Bonzini borrow: orig.borrow.clone(), 856c596199fSPaolo Bonzini } 857c596199fSPaolo Bonzini } 858c596199fSPaolo Bonzini } 859c596199fSPaolo Bonzini 860c596199fSPaolo Bonzini impl<T: fmt::Debug> fmt::Debug for BqlRef<'_, T> { 861c596199fSPaolo Bonzini fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { 862c596199fSPaolo Bonzini (**self).fmt(f) 863c596199fSPaolo Bonzini } 864c596199fSPaolo Bonzini } 865c596199fSPaolo Bonzini 866c596199fSPaolo Bonzini impl<T: fmt::Display> fmt::Display for BqlRef<'_, T> { 867c596199fSPaolo Bonzini fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { 868c596199fSPaolo Bonzini (**self).fmt(f) 869c596199fSPaolo Bonzini } 870c596199fSPaolo Bonzini } 871c596199fSPaolo Bonzini 872c596199fSPaolo Bonzini struct BorrowRefMut<'b> { 873c596199fSPaolo Bonzini borrow: &'b Cell<BorrowFlag>, 874c596199fSPaolo Bonzini } 875c596199fSPaolo Bonzini 876c596199fSPaolo Bonzini impl<'b> BorrowRefMut<'b> { 877c596199fSPaolo Bonzini #[inline] 878c596199fSPaolo Bonzini fn new(borrow: &'b Cell<BorrowFlag>) -> Option<BorrowRefMut<'b>> { 879c596199fSPaolo Bonzini // There must currently be no existing references when borrow_mut() is 880c596199fSPaolo Bonzini // called, so we explicitly only allow going from UNUSED to UNUSED - 1. 881c596199fSPaolo Bonzini match borrow.get() { 882c596199fSPaolo Bonzini UNUSED => { 883c596199fSPaolo Bonzini borrow.set(UNUSED - 1); 884c596199fSPaolo Bonzini Some(BorrowRefMut { borrow }) 885c596199fSPaolo Bonzini } 886c596199fSPaolo Bonzini _ => None, 887c596199fSPaolo Bonzini } 888c596199fSPaolo Bonzini } 889c596199fSPaolo Bonzini } 890c596199fSPaolo Bonzini 891c596199fSPaolo Bonzini impl Drop for BorrowRefMut<'_> { 892c596199fSPaolo Bonzini #[inline] 893c596199fSPaolo Bonzini fn drop(&mut self) { 894c596199fSPaolo Bonzini let borrow = self.borrow.get(); 895c596199fSPaolo Bonzini debug_assert!(is_writing(borrow)); 896c596199fSPaolo Bonzini self.borrow.set(borrow + 1); 897c596199fSPaolo Bonzini bql_block_unlock(false) 898c596199fSPaolo Bonzini } 899c596199fSPaolo Bonzini } 900c596199fSPaolo Bonzini 901c596199fSPaolo Bonzini /// A wrapper type for a mutably borrowed value from a `BqlRefCell<T>`. 902c596199fSPaolo Bonzini /// 903c596199fSPaolo Bonzini /// See the [module-level documentation](self) for more. 904c596199fSPaolo Bonzini pub struct BqlRefMut<'b, T: 'b> { 905c596199fSPaolo Bonzini // NB: we use a pointer instead of `&'b mut T` to avoid `noalias` violations, because a 906c596199fSPaolo Bonzini // `BqlRefMut` argument doesn't hold exclusivity for its whole scope, only until it drops. 907c596199fSPaolo Bonzini value: NonNull<T>, 908c596199fSPaolo Bonzini _borrow: BorrowRefMut<'b>, 909c596199fSPaolo Bonzini // `NonNull` is covariant over `T`, so we need to reintroduce invariance. 910c596199fSPaolo Bonzini marker: PhantomData<&'b mut T>, 911c596199fSPaolo Bonzini } 912c596199fSPaolo Bonzini 913c596199fSPaolo Bonzini impl<T> Deref for BqlRefMut<'_, T> { 914c596199fSPaolo Bonzini type Target = T; 915c596199fSPaolo Bonzini 916c596199fSPaolo Bonzini #[inline] 917c596199fSPaolo Bonzini fn deref(&self) -> &T { 918c596199fSPaolo Bonzini // SAFETY: the value is accessible as long as we hold our borrow. 919c596199fSPaolo Bonzini unsafe { self.value.as_ref() } 920c596199fSPaolo Bonzini } 921c596199fSPaolo Bonzini } 922c596199fSPaolo Bonzini 923c596199fSPaolo Bonzini impl<T> DerefMut for BqlRefMut<'_, T> { 924c596199fSPaolo Bonzini #[inline] 925c596199fSPaolo Bonzini fn deref_mut(&mut self) -> &mut T { 926c596199fSPaolo Bonzini // SAFETY: the value is accessible as long as we hold our borrow. 927c596199fSPaolo Bonzini unsafe { self.value.as_mut() } 928c596199fSPaolo Bonzini } 929c596199fSPaolo Bonzini } 930c596199fSPaolo Bonzini 931c596199fSPaolo Bonzini impl<T: fmt::Debug> fmt::Debug for BqlRefMut<'_, T> { 932c596199fSPaolo Bonzini fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { 933c596199fSPaolo Bonzini (**self).fmt(f) 934c596199fSPaolo Bonzini } 935c596199fSPaolo Bonzini } 936c596199fSPaolo Bonzini 937c596199fSPaolo Bonzini impl<T: fmt::Display> fmt::Display for BqlRefMut<'_, T> { 938c596199fSPaolo Bonzini fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { 939c596199fSPaolo Bonzini (**self).fmt(f) 940c596199fSPaolo Bonzini } 941c596199fSPaolo Bonzini } 9420b9d05e3SPaolo Bonzini 9430b9d05e3SPaolo Bonzini /// Stores an opaque value that is shared with C code. 9440b9d05e3SPaolo Bonzini /// 9450b9d05e3SPaolo Bonzini /// Often, C structs can changed when calling a C function even if they are 9460b9d05e3SPaolo Bonzini /// behind a shared Rust reference, or they can be initialized lazily and have 9470b9d05e3SPaolo Bonzini /// invalid bit patterns (e.g. `3` for a [`bool`]). This goes against Rust's 9480b9d05e3SPaolo Bonzini /// strict aliasing rules, which normally prevent mutation through shared 9490b9d05e3SPaolo Bonzini /// references. 9500b9d05e3SPaolo Bonzini /// 9510b9d05e3SPaolo Bonzini /// Wrapping the struct with `Opaque<T>` ensures that the Rust compiler does not 9520b9d05e3SPaolo Bonzini /// assume the usual constraints that Rust structs require, and allows using 9530b9d05e3SPaolo Bonzini /// shared references on the Rust side. 9540b9d05e3SPaolo Bonzini /// 9550b9d05e3SPaolo Bonzini /// `Opaque<T>` is `#[repr(transparent)]`, so that it matches the memory layout 9560b9d05e3SPaolo Bonzini /// of `T`. 9570b9d05e3SPaolo Bonzini #[repr(transparent)] 9580b9d05e3SPaolo Bonzini pub struct Opaque<T> { 9590b9d05e3SPaolo Bonzini value: UnsafeCell<MaybeUninit<T>>, 9600b9d05e3SPaolo Bonzini // PhantomPinned also allows multiple references to the `Opaque<T>`, i.e. 9610b9d05e3SPaolo Bonzini // one `&mut Opaque<T>` can coexist with a `&mut T` or any number of `&T`; 9620b9d05e3SPaolo Bonzini // see https://docs.rs/pinned-aliasable/latest/pinned_aliasable/. 9630b9d05e3SPaolo Bonzini _pin: PhantomPinned, 9640b9d05e3SPaolo Bonzini } 9650b9d05e3SPaolo Bonzini 9660b9d05e3SPaolo Bonzini impl<T> Opaque<T> { 9670b9d05e3SPaolo Bonzini /// Creates a new shared reference from a C pointer 9680b9d05e3SPaolo Bonzini /// 9690b9d05e3SPaolo Bonzini /// # Safety 9700b9d05e3SPaolo Bonzini /// 9710b9d05e3SPaolo Bonzini /// The pointer must be valid, though it need not point to a valid value. 9720b9d05e3SPaolo Bonzini pub unsafe fn from_raw<'a>(ptr: *mut T) -> &'a Self { 9730b9d05e3SPaolo Bonzini let ptr = NonNull::new(ptr).unwrap().cast::<Self>(); 9740b9d05e3SPaolo Bonzini // SAFETY: Self is a transparent wrapper over T 9750b9d05e3SPaolo Bonzini unsafe { ptr.as_ref() } 9760b9d05e3SPaolo Bonzini } 9770b9d05e3SPaolo Bonzini 9780b9d05e3SPaolo Bonzini /// Creates a new opaque object with uninitialized contents. 9790b9d05e3SPaolo Bonzini /// 9800b9d05e3SPaolo Bonzini /// # Safety 9810b9d05e3SPaolo Bonzini /// 9820b9d05e3SPaolo Bonzini /// Ultimately the pointer to the returned value will be dereferenced 9830b9d05e3SPaolo Bonzini /// in another `unsafe` block, for example when passing it to a C function, 9840b9d05e3SPaolo Bonzini /// but the functions containing the dereference are usually safe. The 9850b9d05e3SPaolo Bonzini /// value returned from `uninit()` must be initialized and pinned before 9860b9d05e3SPaolo Bonzini /// calling them. 9870b9d05e3SPaolo Bonzini #[allow(clippy::missing_const_for_fn)] 9880b9d05e3SPaolo Bonzini pub unsafe fn uninit() -> Self { 9890b9d05e3SPaolo Bonzini Self { 9900b9d05e3SPaolo Bonzini value: UnsafeCell::new(MaybeUninit::uninit()), 9910b9d05e3SPaolo Bonzini _pin: PhantomPinned, 9920b9d05e3SPaolo Bonzini } 9930b9d05e3SPaolo Bonzini } 9940b9d05e3SPaolo Bonzini 9950b9d05e3SPaolo Bonzini /// Creates a new opaque object with zeroed contents. 9960b9d05e3SPaolo Bonzini /// 9970b9d05e3SPaolo Bonzini /// # Safety 9980b9d05e3SPaolo Bonzini /// 9990b9d05e3SPaolo Bonzini /// Ultimately the pointer to the returned value will be dereferenced 10000b9d05e3SPaolo Bonzini /// in another `unsafe` block, for example when passing it to a C function, 10010b9d05e3SPaolo Bonzini /// but the functions containing the dereference are usually safe. The 10020b9d05e3SPaolo Bonzini /// value returned from `uninit()` must be pinned (and possibly initialized) 10030b9d05e3SPaolo Bonzini /// before calling them. 10040b9d05e3SPaolo Bonzini #[allow(clippy::missing_const_for_fn)] 10050b9d05e3SPaolo Bonzini pub unsafe fn zeroed() -> Self { 10060b9d05e3SPaolo Bonzini Self { 10070b9d05e3SPaolo Bonzini value: UnsafeCell::new(MaybeUninit::zeroed()), 10080b9d05e3SPaolo Bonzini _pin: PhantomPinned, 10090b9d05e3SPaolo Bonzini } 10100b9d05e3SPaolo Bonzini } 10110b9d05e3SPaolo Bonzini 10120b9d05e3SPaolo Bonzini /// Returns a raw mutable pointer to the opaque data. 10130b9d05e3SPaolo Bonzini pub const fn as_mut_ptr(&self) -> *mut T { 10140b9d05e3SPaolo Bonzini UnsafeCell::get(&self.value).cast() 10150b9d05e3SPaolo Bonzini } 10160b9d05e3SPaolo Bonzini 10170b9d05e3SPaolo Bonzini /// Returns a raw pointer to the opaque data. 10180b9d05e3SPaolo Bonzini pub const fn as_ptr(&self) -> *const T { 10195df3fe06SPaolo Bonzini self.as_mut_ptr().cast_const() 10200b9d05e3SPaolo Bonzini } 10210b9d05e3SPaolo Bonzini 10220b9d05e3SPaolo Bonzini /// Returns a raw pointer to the opaque data that can be passed to a 10230b9d05e3SPaolo Bonzini /// C function as `void *`. 10240b9d05e3SPaolo Bonzini pub const fn as_void_ptr(&self) -> *mut std::ffi::c_void { 10250b9d05e3SPaolo Bonzini UnsafeCell::get(&self.value).cast() 10260b9d05e3SPaolo Bonzini } 10270b9d05e3SPaolo Bonzini 10280b9d05e3SPaolo Bonzini /// Converts a raw pointer to the wrapped type. 10290b9d05e3SPaolo Bonzini pub const fn raw_get(slot: *mut Self) -> *mut T { 10300b9d05e3SPaolo Bonzini // Compare with Linux's raw_get method, which goes through an UnsafeCell 10310b9d05e3SPaolo Bonzini // because it takes a *const Self instead. 10320b9d05e3SPaolo Bonzini slot.cast() 10330b9d05e3SPaolo Bonzini } 10340b9d05e3SPaolo Bonzini } 10350b9d05e3SPaolo Bonzini 10360b9d05e3SPaolo Bonzini impl<T> fmt::Debug for Opaque<T> { 10370b9d05e3SPaolo Bonzini fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { 10380b9d05e3SPaolo Bonzini let mut name: String = "Opaque<".to_string(); 10390b9d05e3SPaolo Bonzini name += std::any::type_name::<T>(); 10400b9d05e3SPaolo Bonzini name += ">"; 10410b9d05e3SPaolo Bonzini f.debug_tuple(&name).field(&self.as_ptr()).finish() 10420b9d05e3SPaolo Bonzini } 10430b9d05e3SPaolo Bonzini } 10440b9d05e3SPaolo Bonzini 10450b9d05e3SPaolo Bonzini impl<T: Default> Opaque<T> { 10460b9d05e3SPaolo Bonzini /// Creates a new opaque object with default contents. 10470b9d05e3SPaolo Bonzini /// 10480b9d05e3SPaolo Bonzini /// # Safety 10490b9d05e3SPaolo Bonzini /// 10500b9d05e3SPaolo Bonzini /// Ultimately the pointer to the returned value will be dereferenced 10510b9d05e3SPaolo Bonzini /// in another `unsafe` block, for example when passing it to a C function, 10520b9d05e3SPaolo Bonzini /// but the functions containing the dereference are usually safe. The 10530b9d05e3SPaolo Bonzini /// value returned from `uninit()` must be pinned before calling them. 10540b9d05e3SPaolo Bonzini pub unsafe fn new() -> Self { 10550b9d05e3SPaolo Bonzini Self { 10560b9d05e3SPaolo Bonzini value: UnsafeCell::new(MaybeUninit::new(T::default())), 10570b9d05e3SPaolo Bonzini _pin: PhantomPinned, 10580b9d05e3SPaolo Bonzini } 10590b9d05e3SPaolo Bonzini } 10600b9d05e3SPaolo Bonzini } 1061f07a5674SPaolo Bonzini 1062f07a5674SPaolo Bonzini /// Annotates [`Self`] as a transparent wrapper for another type. 1063f07a5674SPaolo Bonzini /// 1064f07a5674SPaolo Bonzini /// Usually defined via the [`qemu_api_macros::Wrapper`] derive macro. 1065f07a5674SPaolo Bonzini /// 1066f07a5674SPaolo Bonzini /// # Examples 1067f07a5674SPaolo Bonzini /// 1068f07a5674SPaolo Bonzini /// ``` 1069f07a5674SPaolo Bonzini /// # use std::mem::ManuallyDrop; 1070f07a5674SPaolo Bonzini /// # use qemu_api::cell::Wrapper; 1071f07a5674SPaolo Bonzini /// #[repr(transparent)] 1072f07a5674SPaolo Bonzini /// pub struct Example { 1073f07a5674SPaolo Bonzini /// inner: ManuallyDrop<String>, 1074f07a5674SPaolo Bonzini /// } 1075f07a5674SPaolo Bonzini /// 1076f07a5674SPaolo Bonzini /// unsafe impl Wrapper for Example { 1077f07a5674SPaolo Bonzini /// type Wrapped = String; 1078f07a5674SPaolo Bonzini /// } 1079f07a5674SPaolo Bonzini /// ``` 1080f07a5674SPaolo Bonzini /// 1081f07a5674SPaolo Bonzini /// # Safety 1082f07a5674SPaolo Bonzini /// 1083f07a5674SPaolo Bonzini /// `Self` must be a `#[repr(transparent)]` wrapper for the `Wrapped` type, 1084f07a5674SPaolo Bonzini /// whether directly or indirectly. 1085f07a5674SPaolo Bonzini /// 1086f07a5674SPaolo Bonzini /// # Methods 1087f07a5674SPaolo Bonzini /// 1088f07a5674SPaolo Bonzini /// By convention, types that implement Wrapper also implement the following 1089f07a5674SPaolo Bonzini /// methods: 1090f07a5674SPaolo Bonzini /// 1091f07a5674SPaolo Bonzini /// ```ignore 1092f07a5674SPaolo Bonzini /// pub const unsafe fn from_raw<'a>(value: *mut Self::Wrapped) -> &'a Self; 1093f07a5674SPaolo Bonzini /// pub const unsafe fn as_mut_ptr(&self) -> *mut Self::Wrapped; 1094f07a5674SPaolo Bonzini /// pub const unsafe fn as_ptr(&self) -> *const Self::Wrapped; 1095f07a5674SPaolo Bonzini /// pub const unsafe fn raw_get(slot: *mut Self) -> *const Self::Wrapped; 1096f07a5674SPaolo Bonzini /// ``` 1097f07a5674SPaolo Bonzini /// 1098f07a5674SPaolo Bonzini /// They are not defined here to allow them to be `const`. 1099f07a5674SPaolo Bonzini pub unsafe trait Wrapper { 1100f07a5674SPaolo Bonzini type Wrapped; 1101f07a5674SPaolo Bonzini } 1102f07a5674SPaolo Bonzini 1103f07a5674SPaolo Bonzini unsafe impl<T> Wrapper for Opaque<T> { 1104f07a5674SPaolo Bonzini type Wrapped = T; 1105f07a5674SPaolo Bonzini } 1106