2022-12-05 14:48:57 +11:00
|
|
|
//! The librashader preset C API (`libra_preset_*`).
|
2023-01-15 07:52:58 +11:00
|
|
|
use crate::ctypes::{libra_shader_preset_t};
|
2023-01-14 08:05:13 +11:00
|
|
|
use crate::error::{assert_non_null, assert_some_ptr, LibrashaderError};
|
2023-01-15 07:52:58 +11:00
|
|
|
use crate::ffi::extern_fn;
|
2022-12-05 16:06:37 +11:00
|
|
|
use librashader::presets::ShaderPreset;
|
|
|
|
use std::ffi::{c_char, CStr, CString};
|
|
|
|
use std::mem::MaybeUninit;
|
2022-12-04 11:55:27 +11:00
|
|
|
use std::ptr::NonNull;
|
|
|
|
|
2023-01-15 07:52:58 +11:00
|
|
|
/// 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,
|
|
|
|
}
|
|
|
|
|
2023-01-14 08:05:13 +11:00
|
|
|
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<libra_shader_preset_t>
|
|
|
|
) {
|
2022-12-04 10:32:10 +11:00
|
|
|
assert_non_null!(filename);
|
|
|
|
assert_non_null!(out);
|
|
|
|
|
2022-12-05 16:06:37 +11:00
|
|
|
let filename = unsafe { CStr::from_ptr(filename) };
|
2022-12-04 10:32:10 +11:00
|
|
|
let filename = filename.to_str()?;
|
|
|
|
println!("loading {filename}");
|
|
|
|
|
2023-01-14 08:05:13 +11:00
|
|
|
let preset = ShaderPreset::try_parse(filename)?;
|
2022-12-04 10:32:10 +11:00
|
|
|
unsafe {
|
2022-12-05 16:06:37 +11:00
|
|
|
out.write(MaybeUninit::new(NonNull::new(Box::into_raw(Box::new(
|
|
|
|
preset,
|
|
|
|
)))))
|
2022-12-04 10:32:10 +11:00
|
|
|
}
|
2023-01-14 08:05:13 +11:00
|
|
|
}
|
2022-12-04 10:32:10 +11:00
|
|
|
}
|
|
|
|
|
2023-01-14 08:05:13 +11:00
|
|
|
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) {
|
2022-12-04 10:32:10 +11:00
|
|
|
assert_non_null!(preset);
|
|
|
|
unsafe {
|
|
|
|
let preset_ptr = &mut *preset;
|
2022-12-04 11:55:27 +11:00
|
|
|
let preset = preset_ptr.take();
|
|
|
|
drop(Box::from_raw(preset.unwrap().as_ptr()));
|
2022-12-04 10:32:10 +11:00
|
|
|
}
|
2023-01-14 08:05:13 +11:00
|
|
|
}
|
2022-12-04 10:32:10 +11:00
|
|
|
}
|
|
|
|
|
2023-01-14 08:05:13 +11:00
|
|
|
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| {
|
2022-12-04 10:32:10 +11:00
|
|
|
let name = unsafe {
|
|
|
|
CStr::from_ptr(name)
|
|
|
|
};
|
|
|
|
|
|
|
|
let name = name.to_str()?;
|
2022-12-04 11:55:27 +11:00
|
|
|
assert_some_ptr!(mut preset);
|
2022-12-04 10:32:10 +11:00
|
|
|
|
|
|
|
if let Some(param) = preset.parameters.iter_mut().find(|c| c.name == name) {
|
|
|
|
param.value = value
|
|
|
|
}
|
2023-01-14 08:05:13 +11:00
|
|
|
}
|
2022-12-04 10:32:10 +11:00
|
|
|
}
|
|
|
|
|
2023-01-14 08:05:13 +11:00
|
|
|
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<f32>
|
|
|
|
) |name, preset| {
|
2022-12-05 16:06:37 +11:00
|
|
|
let name = unsafe { CStr::from_ptr(name) };
|
2022-12-04 10:32:10 +11:00
|
|
|
let name = name.to_str()?;
|
2022-12-04 11:55:27 +11:00
|
|
|
assert_some_ptr!(preset);
|
2022-12-05 15:54:47 +11:00
|
|
|
assert_non_null!(value);
|
|
|
|
|
2022-12-04 10:32:10 +11:00
|
|
|
if let Some(param) = preset.parameters.iter().find(|c| c.name == name) {
|
2022-12-05 16:06:37 +11:00
|
|
|
unsafe { value.write(MaybeUninit::new(param.value)) }
|
2022-12-04 10:32:10 +11:00
|
|
|
}
|
2023-01-14 08:05:13 +11:00
|
|
|
}
|
2022-12-04 10:32:10 +11:00
|
|
|
}
|
|
|
|
|
2023-01-14 08:05:13 +11:00
|
|
|
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| {
|
2022-12-05 14:37:03 +11:00
|
|
|
assert_some_ptr!(preset);
|
|
|
|
println!("{:#?}", preset);
|
2023-01-14 08:05:13 +11:00
|
|
|
}
|
2022-12-04 10:32:10 +11:00
|
|
|
}
|
|
|
|
|
2023-01-15 07:52:58 +11:00
|
|
|
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<libra_preset_parameter_list_t>
|
|
|
|
) |preset| {
|
2022-12-04 11:55:27 +11:00
|
|
|
assert_some_ptr!(preset);
|
2023-01-15 07:52:58 +11:00
|
|
|
assert_non_null!(out);
|
2022-12-04 10:32:10 +11:00
|
|
|
|
|
|
|
let iter = librashader::presets::get_parameter_meta(preset)?;
|
2023-01-15 07:52:58 +11:00
|
|
|
let mut values = Vec::new();
|
2022-12-04 10:32:10 +11:00
|
|
|
for param in iter {
|
2023-01-15 07:52:58 +11:00
|
|
|
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
|
|
|
|
})
|
2022-12-04 10:32:10 +11:00
|
|
|
}
|
2023-01-15 07:52:58 +11:00
|
|
|
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,
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
}
|
2022-12-04 10:32:10 +11:00
|
|
|
}
|
2023-01-15 07:52:58 +11:00
|
|
|
|
|
|
|
// /// 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);
|
|
|
|
// })
|
|
|
|
// }
|