1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75
// Take a look at the license at the top of the repository in the LICENSE file.
use std::{mem, ptr};
use glib::translate::*;
use crate::{resources_register, Resource};
impl Resource {
/// Creates a GResource from a reference to the binary resource bundle.
/// This will keep a reference to `data` while the resource lives, so
/// the data should not be modified or freed.
///
/// If you want to use this resource in the global resource namespace you need
/// to register it with [`resources_register()`][crate::resources_register()].
///
/// Note: `data` must be backed by memory that is at least pointer aligned.
/// Otherwise this function will internally create a copy of the memory since
/// GLib 2.56, or in older versions fail and exit the process.
///
/// If `data` is empty or corrupt, [`ResourceError::Internal`][crate::ResourceError::Internal] will be returned.
/// ## `data`
/// A [`glib::Bytes`][crate::glib::Bytes]
///
/// # Returns
///
/// a new [`Resource`][crate::Resource], or [`None`] on error
#[doc(alias = "g_resource_new_from_data")]
pub fn from_data(data: &glib::Bytes) -> Result<Resource, glib::Error> {
unsafe {
let mut error = ptr::null_mut();
// Create a copy of data if it is not pointer-aligned
// https://bugzilla.gnome.org/show_bug.cgi?id=790030
let mut data = data.clone();
let data_ptr = glib::ffi::g_bytes_get_data(data.to_glib_none().0, ptr::null_mut());
if data_ptr as usize % mem::align_of::<*const u8>() != 0 {
data = glib::Bytes::from(&*data);
}
let ret = ffi::g_resource_new_from_data(data.to_glib_none().0, &mut error);
if error.is_null() {
Ok(from_glib_full(ret))
} else {
Err(from_glib_full(error))
}
}
}
}
#[doc(hidden)]
pub fn resources_register_include_impl(bytes: &'static [u8]) -> Result<(), glib::Error> {
let bytes = glib::Bytes::from_static(bytes);
let resource = Resource::from_data(&bytes)?;
resources_register(&resource);
Ok(())
}
// rustdoc-stripper-ignore-next
/// Include gresources generated with `glib_build_tools::compile_resources` and register with glib. `path` is
/// relative to `OUTDIR`.
///
/// ```ignore
/// gio::resources_register_include!("compiled.gresource").unwrap();
/// ```
#[macro_export]
macro_rules! resources_register_include {
($path:expr) => {
$crate::resources_register_include_impl(include_bytes!(concat!(
env!("OUT_DIR"),
"/",
$path
)))
};
}