| // Copyright 2024, Linaro Limited |
| // Author(s): Manos Pitsidianakis <manos.pitsidianakis@linaro.org> |
| // SPDX-License-Identifier: GPL-2.0-or-later |
| |
| #![cfg_attr(not(MESON), doc = include_str!("../README.md"))] |
| |
| #[allow( |
| dead_code, |
| improper_ctypes_definitions, |
| improper_ctypes, |
| non_camel_case_types, |
| non_snake_case, |
| non_upper_case_globals, |
| unsafe_op_in_unsafe_fn, |
| clippy::missing_const_for_fn, |
| clippy::too_many_arguments, |
| clippy::approx_constant, |
| clippy::use_self, |
| clippy::useless_transmute, |
| clippy::missing_safety_doc, |
| )] |
| #[rustfmt::skip] |
| pub mod bindings; |
| |
| unsafe impl Send for bindings::Property {} |
| unsafe impl Sync for bindings::Property {} |
| unsafe impl Sync for bindings::TypeInfo {} |
| unsafe impl Sync for bindings::VMStateDescription {} |
| unsafe impl Sync for bindings::VMStateField {} |
| unsafe impl Sync for bindings::VMStateInfo {} |
| |
| pub mod c_str; |
| pub mod definitions; |
| pub mod device_class; |
| pub mod offset_of; |
| pub mod vmstate; |
| pub mod zeroable; |
| |
| use std::{ |
| alloc::{GlobalAlloc, Layout}, |
| os::raw::c_void, |
| }; |
| |
| #[cfg(HAVE_GLIB_WITH_ALIGNED_ALLOC)] |
| extern "C" { |
| fn g_aligned_alloc0( |
| n_blocks: bindings::gsize, |
| n_block_bytes: bindings::gsize, |
| alignment: bindings::gsize, |
| ) -> bindings::gpointer; |
| fn g_aligned_free(mem: bindings::gpointer); |
| } |
| |
| #[cfg(not(HAVE_GLIB_WITH_ALIGNED_ALLOC))] |
| extern "C" { |
| fn qemu_memalign(alignment: usize, size: usize) -> *mut c_void; |
| fn qemu_vfree(ptr: *mut c_void); |
| } |
| |
| extern "C" { |
| fn g_malloc0(n_bytes: bindings::gsize) -> bindings::gpointer; |
| fn g_free(mem: bindings::gpointer); |
| } |
| |
| /// An allocator that uses the same allocator as QEMU in C. |
| /// |
| /// It is enabled by default with the `allocator` feature. |
| /// |
| /// To set it up manually as a global allocator in your crate: |
| /// |
| /// ```ignore |
| /// use qemu_api::QemuAllocator; |
| /// |
| /// #[global_allocator] |
| /// static GLOBAL: QemuAllocator = QemuAllocator::new(); |
| /// ``` |
| #[derive(Clone, Copy, Debug)] |
| #[repr(C)] |
| pub struct QemuAllocator { |
| _unused: [u8; 0], |
| } |
| |
| #[cfg_attr(all(feature = "allocator", not(test)), global_allocator)] |
| pub static GLOBAL: QemuAllocator = QemuAllocator::new(); |
| |
| impl QemuAllocator { |
| // From the glibc documentation, on GNU systems, malloc guarantees 16-byte |
| // alignment on 64-bit systems and 8-byte alignment on 32-bit systems. See |
| // https://www.gnu.org/software/libc/manual/html_node/Malloc-Examples.html. |
| // This alignment guarantee also applies to Windows and Android. On Darwin |
| // and OpenBSD, the alignment is 16 bytes on both 64-bit and 32-bit systems. |
| #[cfg(all( |
| target_pointer_width = "32", |
| not(any(target_os = "macos", target_os = "openbsd")) |
| ))] |
| pub const DEFAULT_ALIGNMENT_BYTES: Option<usize> = Some(8); |
| #[cfg(all( |
| target_pointer_width = "64", |
| not(any(target_os = "macos", target_os = "openbsd")) |
| ))] |
| pub const DEFAULT_ALIGNMENT_BYTES: Option<usize> = Some(16); |
| #[cfg(all( |
| any(target_pointer_width = "32", target_pointer_width = "64"), |
| any(target_os = "macos", target_os = "openbsd") |
| ))] |
| pub const DEFAULT_ALIGNMENT_BYTES: Option<usize> = Some(16); |
| #[cfg(not(any(target_pointer_width = "32", target_pointer_width = "64")))] |
| pub const DEFAULT_ALIGNMENT_BYTES: Option<usize> = None; |
| |
| pub const fn new() -> Self { |
| Self { _unused: [] } |
| } |
| } |
| |
| impl Default for QemuAllocator { |
| fn default() -> Self { |
| Self::new() |
| } |
| } |
| |
| // Sanity check. |
| const _: [(); 8] = [(); ::core::mem::size_of::<*mut c_void>()]; |
| |
| unsafe impl GlobalAlloc for QemuAllocator { |
| unsafe fn alloc(&self, layout: Layout) -> *mut u8 { |
| if matches!(Self::DEFAULT_ALIGNMENT_BYTES, Some(default) if default.checked_rem(layout.align()) == Some(0)) |
| { |
| // SAFETY: g_malloc0() is safe to call. |
| unsafe { g_malloc0(layout.size().try_into().unwrap()).cast::<u8>() } |
| } else { |
| #[cfg(HAVE_GLIB_WITH_ALIGNED_ALLOC)] |
| { |
| // SAFETY: g_aligned_alloc0() is safe to call. |
| unsafe { |
| g_aligned_alloc0( |
| layout.size().try_into().unwrap(), |
| 1, |
| layout.align().try_into().unwrap(), |
| ) |
| .cast::<u8>() |
| } |
| } |
| #[cfg(not(HAVE_GLIB_WITH_ALIGNED_ALLOC))] |
| { |
| // SAFETY: qemu_memalign() is safe to call. |
| unsafe { qemu_memalign(layout.align(), layout.size()).cast::<u8>() } |
| } |
| } |
| } |
| |
| unsafe fn dealloc(&self, ptr: *mut u8, layout: Layout) { |
| if matches!(Self::DEFAULT_ALIGNMENT_BYTES, Some(default) if default.checked_rem(layout.align()) == Some(0)) |
| { |
| // SAFETY: `ptr` must have been allocated by Self::alloc thus a valid |
| // glib-allocated pointer, so `g_free`ing is safe. |
| unsafe { g_free(ptr.cast::<_>()) } |
| } else { |
| #[cfg(HAVE_GLIB_WITH_ALIGNED_ALLOC)] |
| { |
| // SAFETY: `ptr` must have been allocated by Self::alloc thus a valid aligned |
| // glib-allocated pointer, so `g_aligned_free`ing is safe. |
| unsafe { g_aligned_free(ptr.cast::<_>()) } |
| } |
| #[cfg(not(HAVE_GLIB_WITH_ALIGNED_ALLOC))] |
| { |
| // SAFETY: `ptr` must have been allocated by Self::alloc thus a valid aligned |
| // glib-allocated pointer, so `qemu_vfree`ing is safe. |
| unsafe { qemu_vfree(ptr.cast::<_>()) } |
| } |
| } |
| } |
| } |
| |
| #[cfg(has_offset_of)] |
| pub use core::mem::offset_of; |