//! The librashader preset C API (`libra_preset_*`). use crate::ctypes::{libra_shader_preset_t}; use crate::error::{assert_non_null, assert_some_ptr, LibrashaderError}; use crate::ffi::extern_fn; use librashader::presets::ShaderPreset; use std::ffi::{c_char, CStr, CString}; use std::mem::MaybeUninit; use std::ptr::NonNull; /// A list of preset parameters. #[repr(C)] pub struct libra_preset_parameter_list_t { /// A pointer to the parameter pub parameters: *const libra_preset_param_t, /// The number of parameters in the list pub length: u64, /// For internal use only. /// Changing this causes immediate undefined behaviour on freeing this parameter list. pub _internal_alloc: u64, } /// A preset parameter. #[repr(C)] pub struct libra_preset_param_t { /// The name of the parameter pub name: *const c_char, /// The description of the parameter. pub description: *const c_char, /// The initial value the parameter is set to. pub initial: f32, /// The minimum value that the parameter can be set to. pub minimum: f32, /// The maximum value that the parameter can be set to. pub maximum: f32, /// The step by which this parameter can be incremented or decremented. pub step: f32, } extern_fn! { /// Load a preset. /// /// ## Safety /// - `filename` must be either null or a valid, aligned pointer to a string path to the shader preset. /// - `out` must be either null, or an aligned pointer to an uninitialized or invalid `libra_shader_preset_t`. /// ## Returns /// - If any parameters are null, `out` is unchanged, and this function returns `LIBRA_ERR_INVALID_PARAMETER`. fn libra_preset_create( filename: *const c_char, out: *mut MaybeUninit ) { assert_non_null!(filename); assert_non_null!(out); let filename = unsafe { CStr::from_ptr(filename) }; let filename = filename.to_str()?; println!("loading {filename}"); let preset = ShaderPreset::try_parse(filename)?; unsafe { out.write(MaybeUninit::new(NonNull::new(Box::into_raw(Box::new( preset, ))))) } } } extern_fn! { /// Free the preset. /// /// If `preset` is null, this function does nothing. The resulting value in `preset` then becomes /// null. /// /// ## Safety /// - `preset` must be a valid and aligned pointer to a shader preset. fn libra_preset_free(preset: *mut libra_shader_preset_t) { assert_non_null!(preset); unsafe { let preset_ptr = &mut *preset; let preset = preset_ptr.take(); drop(Box::from_raw(preset.unwrap().as_ptr())); } } } extern_fn! { /// Set the value of the parameter in the preset. /// /// ## Safety /// - `preset` must be null or a valid and aligned pointer to a shader preset. /// - `name` must be null or a valid and aligned pointer to a string. fn libra_preset_set_param( preset: *mut libra_shader_preset_t, name: *const c_char, value: f32 ) |name|; mut |preset| { let name = unsafe { CStr::from_ptr(name) }; let name = name.to_str()?; assert_some_ptr!(mut preset); if let Some(param) = preset.parameters.iter_mut().find(|c| c.name == name) { param.value = value } } } extern_fn! { /// Get the value of the parameter as set in the preset. /// /// ## Safety /// - `preset` must be null or a valid and aligned pointer to a shader preset. /// - `name` must be null or a valid and aligned pointer to a string. /// - `value` may be a pointer to a uninitialized `float`. fn libra_preset_get_param( preset: *mut libra_shader_preset_t, name: *const c_char, value: *mut MaybeUninit ) |name, preset| { let name = unsafe { CStr::from_ptr(name) }; let name = name.to_str()?; assert_some_ptr!(preset); assert_non_null!(value); if let Some(param) = preset.parameters.iter().find(|c| c.name == name) { unsafe { value.write(MaybeUninit::new(param.value)) } } } } extern_fn! { /// Pretty print the shader preset. /// /// ## Safety /// - `preset` must be null or a valid and aligned pointer to a shader preset. fn libra_preset_print(preset: *mut libra_shader_preset_t) |preset| { assert_some_ptr!(preset); println!("{:#?}", preset); } } extern_fn! { /// Get a list of runtime parameter names. /// /// ## Safety /// - `preset` must be null or a valid and aligned pointer to a shader preset. /// - `out` must be an aligned pointer to a `libra_preset_parameter_list_t`. fn libra_preset_get_runtime_parameters( preset: *mut libra_shader_preset_t, out: *mut MaybeUninit ) |preset| { assert_some_ptr!(preset); assert_non_null!(out); let iter = librashader::presets::get_parameter_meta(preset)?; let mut values = Vec::new(); for param in iter { let name = CString::new(param.id) .map_err(|err| LibrashaderError::UnknownError(Box::new(err)))?; let description = CString::new(param.description) .map_err(|err| LibrashaderError::UnknownError(Box::new(err)))?; values.push(libra_preset_param_t { name: name.into_raw().cast_const(), description: description.into_raw().cast_const(), initial: param.initial, minimum: param.minimum, maximum: param.maximum, step: param.step }) } let (parts, len, cap) = values.into_raw_parts(); unsafe { out.write(MaybeUninit::new(libra_preset_parameter_list_t { parameters: parts, length: len as u64, _internal_alloc: cap as u64, })); } } } // /// Get a list of runtime parameter names. // /// // /// The caller must provide a sufficiently sized buffer. // /// If `value` is null, then size will be written to with the size of the buffer required // /// to get the parameter names. // /// ## Safety // /// - `preset` must be null or a valid and aligned pointer to a shader preset. // #[no_mangle] // pub unsafe extern "C" fn libra_preset_free_runtime_param_names( // value: MaybeUninit<*mut *const c_char>, // ) -> libra_error_t { // ffi_body!(|value| { // let iter = librashader::presets::get_parameter_meta(preset)?; // let mut c_strings = Vec::new(); // for param in iter { // let c_string = CString::new(param.id) // .map_err(|err| LibrashaderError::UnknownError(Box::new(err)))?; // c_strings.push(c_string.into_raw().cast_const()); // } // // let (parts, _len, _cap) = c_strings.into_raw_parts(); // // value.write(parts); // }) // }