2022-04-07 14:12:02 +02:00
|
|
|
//! Utilities for saving a [crate::plugin::Plugin]'s state. The actual state object is also exposed
|
|
|
|
//! to plugins through the [`GuiContext`][crate::prelude::GuiContext].
|
2022-01-29 14:20:14 +01:00
|
|
|
|
2022-08-18 13:55:31 +02:00
|
|
|
use anyhow::{Context, Result};
|
2022-01-29 14:20:14 +01:00
|
|
|
use serde::{Deserialize, Serialize};
|
2022-07-02 19:10:56 +02:00
|
|
|
use std::collections::{BTreeMap, HashMap};
|
2022-04-07 15:31:46 +02:00
|
|
|
use std::sync::Arc;
|
2022-03-02 15:49:40 +01:00
|
|
|
|
2022-03-03 23:23:51 +01:00
|
|
|
use crate::param::internals::{ParamPtr, Params};
|
2022-05-01 18:45:35 +02:00
|
|
|
use crate::param::{Param, ParamMut};
|
2022-08-20 15:09:54 +02:00
|
|
|
use crate::plugin::{BufferConfig, Plugin};
|
2022-01-29 14:20:14 +01:00
|
|
|
|
2022-04-07 14:12:02 +02:00
|
|
|
// These state objects are also exposed directly to the plugin so it can do its own internal preset
|
|
|
|
// management
|
|
|
|
|
2022-01-29 14:54:48 +01:00
|
|
|
/// A plain, unnormalized value for a parameter.
|
2022-04-07 16:49:39 +02:00
|
|
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
2022-01-29 14:20:14 +01:00
|
|
|
#[serde(rename_all = "snake_case")]
|
2022-04-07 14:12:02 +02:00
|
|
|
pub enum ParamValue {
|
2022-01-29 14:20:14 +01:00
|
|
|
F32(f32),
|
|
|
|
I32(i32),
|
2022-01-30 02:04:35 +01:00
|
|
|
Bool(bool),
|
2022-06-03 22:22:36 +02:00
|
|
|
/// Only used for enum parameters that have the `#[id = "..."]` attribute set.
|
|
|
|
String(String),
|
2022-01-29 14:20:14 +01:00
|
|
|
}
|
|
|
|
|
2022-04-07 14:12:02 +02:00
|
|
|
/// A plugin's state so it can be restored at a later point. This object can be serialized and
|
|
|
|
/// deserialized using serde.
|
2022-07-02 19:10:56 +02:00
|
|
|
///
|
|
|
|
/// The fields are stored as `BTreeMap`s so the order in the serialized file is consistent.
|
2022-04-07 16:49:39 +02:00
|
|
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
2022-04-07 17:39:34 +02:00
|
|
|
pub struct PluginState {
|
2022-08-20 15:09:54 +02:00
|
|
|
/// The plugin version this state was saved with. Right now this is not used, but later versions
|
|
|
|
/// of NIH-plug may allow you to modify the plugin state object directly before it is loaded to
|
|
|
|
/// allow migrating plugin states between breaking parameter changes.
|
|
|
|
///
|
|
|
|
/// # Notes
|
|
|
|
///
|
|
|
|
/// If the saved state is very old, then this field may be empty.
|
|
|
|
#[serde(default)]
|
|
|
|
pub version: String,
|
|
|
|
|
2022-09-29 12:28:56 +02:00
|
|
|
/// The plugin's parameter values. These are stored unnormalized. This means the old values will
|
2022-01-29 14:20:14 +01:00
|
|
|
/// be recalled when when the parameter's range gets increased. Doing so may still mess with
|
2022-09-29 12:28:56 +02:00
|
|
|
/// parameter automation though, depending on how the host implements that.
|
2022-07-02 19:10:56 +02:00
|
|
|
pub params: BTreeMap<String, ParamValue>,
|
2022-01-30 17:09:18 +01:00
|
|
|
/// Arbitrary fields that should be persisted together with the plugin's parameters. Any field
|
2022-03-03 23:05:01 +01:00
|
|
|
/// on the [`Params`][crate::param::internals::Params] struct that's annotated with `#[persist =
|
2022-02-05 12:56:03 +01:00
|
|
|
/// "stable_name"]` will be persisted this way.
|
2022-01-30 17:29:25 +01:00
|
|
|
///
|
2022-01-30 18:23:13 +01:00
|
|
|
/// The individual fields are also serialized as JSON so they can safely be restored
|
|
|
|
/// independently of the other fields.
|
2022-07-02 19:10:56 +02:00
|
|
|
pub fields: BTreeMap<String, String>,
|
2022-01-30 18:15:01 +01:00
|
|
|
}
|
2022-03-02 15:49:40 +01:00
|
|
|
|
2022-04-24 19:45:22 +02:00
|
|
|
/// Create a parameters iterator from the hashtables stored in the plugin wrappers. This avoids
|
|
|
|
/// having to call `.param_map()` again, which may include expensive user written code.
|
|
|
|
pub(crate) fn make_params_iter<'a>(
|
|
|
|
param_by_hash: &'a HashMap<u32, ParamPtr>,
|
|
|
|
param_id_to_hash: &'a HashMap<String, u32>,
|
|
|
|
) -> impl IntoIterator<Item = (&'a String, ParamPtr)> {
|
|
|
|
param_id_to_hash.iter().filter_map(|(param_id_str, hash)| {
|
|
|
|
let param_ptr = param_by_hash.get(hash)?;
|
|
|
|
Some((param_id_str, *param_ptr))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a getter function that gets a parameter from the hashtables stored in the plugin by
|
|
|
|
/// string ID.
|
|
|
|
pub(crate) fn make_params_getter<'a>(
|
|
|
|
param_by_hash: &'a HashMap<u32, ParamPtr>,
|
|
|
|
param_id_to_hash: &'a HashMap<String, u32>,
|
2022-04-24 20:03:30 +02:00
|
|
|
) -> impl Fn(&str) -> Option<ParamPtr> + 'a {
|
2022-04-24 19:45:22 +02:00
|
|
|
|param_id_str| {
|
|
|
|
param_id_to_hash
|
|
|
|
.get(param_id_str)
|
|
|
|
.and_then(|hash| param_by_hash.get(hash))
|
|
|
|
.copied()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-07 17:12:08 +02:00
|
|
|
/// Serialize a plugin's state to a state object. This is separate from [`serialize_json()`] to
|
2022-04-24 19:45:22 +02:00
|
|
|
/// allow passing the raw object directly to the plugin. The parameters are not pulled directly from
|
|
|
|
/// `plugin_params` by default to avoid unnecessary allocations in the `.param_map()` method, as the
|
|
|
|
/// plugin wrappers will already have a list of parameters handy. See [`make_params_iter()`].
|
2022-08-20 15:09:54 +02:00
|
|
|
pub(crate) unsafe fn serialize_object<'a, P: Plugin>(
|
2022-04-07 15:31:46 +02:00
|
|
|
plugin_params: Arc<dyn Params>,
|
2022-04-24 19:45:22 +02:00
|
|
|
params_iter: impl IntoIterator<Item = (&'a String, ParamPtr)>,
|
2022-04-07 17:39:34 +02:00
|
|
|
) -> PluginState {
|
2022-04-24 18:34:40 +02:00
|
|
|
// We'll serialize parameter values as a simple `string_param_id: display_value` map.
|
2022-05-01 18:30:30 +02:00
|
|
|
// NOTE: If the plugin is being modulated (and the plugin is a CLAP plugin in Bitwig Studio),
|
|
|
|
// then this should save the values without any modulation applied to it
|
2022-07-02 19:10:56 +02:00
|
|
|
let params: BTreeMap<_, _> = params_iter
|
2022-04-24 19:45:22 +02:00
|
|
|
.into_iter()
|
|
|
|
.map(|(param_id_str, param_ptr)| match param_ptr {
|
2022-05-01 18:30:30 +02:00
|
|
|
ParamPtr::FloatParam(p) => (
|
|
|
|
param_id_str.clone(),
|
|
|
|
ParamValue::F32((*p).unmodulated_plain_value()),
|
|
|
|
),
|
|
|
|
ParamPtr::IntParam(p) => (
|
|
|
|
param_id_str.clone(),
|
|
|
|
ParamValue::I32((*p).unmodulated_plain_value()),
|
|
|
|
),
|
|
|
|
ParamPtr::BoolParam(p) => (
|
|
|
|
param_id_str.clone(),
|
|
|
|
ParamValue::Bool((*p).unmodulated_plain_value()),
|
|
|
|
),
|
2022-03-02 15:49:40 +01:00
|
|
|
ParamPtr::EnumParam(p) => (
|
2022-06-03 22:22:36 +02:00
|
|
|
// Enums are either serialized based on the active variant's index (which may not be
|
|
|
|
// the same as the discriminator), or a custom set stable string ID. The latter
|
|
|
|
// allows the variants to be reordered.
|
2022-04-24 19:45:22 +02:00
|
|
|
param_id_str.clone(),
|
2022-06-03 22:22:36 +02:00
|
|
|
match (*p).unmodulated_plain_id() {
|
|
|
|
Some(id) => ParamValue::String(id.to_owned()),
|
|
|
|
None => ParamValue::I32((*p).unmodulated_plain_value()),
|
|
|
|
},
|
2022-03-02 15:49:40 +01:00
|
|
|
),
|
|
|
|
})
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
// The plugin can also persist arbitrary fields alongside its parameters. This is useful for
|
|
|
|
// storing things like sample data.
|
|
|
|
let fields = plugin_params.serialize_fields();
|
|
|
|
|
2022-08-20 15:09:54 +02:00
|
|
|
PluginState {
|
|
|
|
version: String::from(P::VERSION),
|
|
|
|
params,
|
|
|
|
fields,
|
|
|
|
}
|
2022-03-02 15:49:40 +01:00
|
|
|
}
|
2022-03-02 16:00:11 +01:00
|
|
|
|
|
|
|
/// Serialize a plugin's state to a vector containing JSON data. This can (and should) be shared
|
2022-08-18 13:55:31 +02:00
|
|
|
/// across plugin formats. If the `zstd` feature is enabled, then the state will be compressed using
|
|
|
|
/// Zstandard.
|
2022-08-20 15:09:54 +02:00
|
|
|
pub(crate) unsafe fn serialize_json<'a, P: Plugin>(
|
2022-04-07 17:12:08 +02:00
|
|
|
plugin_params: Arc<dyn Params>,
|
2022-04-24 19:45:22 +02:00
|
|
|
params_iter: impl IntoIterator<Item = (&'a String, ParamPtr)>,
|
2022-08-18 13:55:31 +02:00
|
|
|
) -> Result<Vec<u8>> {
|
2022-08-20 15:09:54 +02:00
|
|
|
let plugin_state = serialize_object::<P>(plugin_params, params_iter);
|
2022-08-18 13:55:31 +02:00
|
|
|
let json = serde_json::to_vec(&plugin_state).context("Could not format as JSON")?;
|
|
|
|
|
|
|
|
#[cfg(feature = "zstd")]
|
|
|
|
{
|
|
|
|
zstd::encode_all(json.as_slice(), zstd::DEFAULT_COMPRESSION_LEVEL)
|
|
|
|
.context("Could not compress state")
|
|
|
|
}
|
|
|
|
#[cfg(not(feature = "zstd"))]
|
|
|
|
{
|
|
|
|
Ok(json)
|
|
|
|
}
|
2022-04-07 17:12:08 +02:00
|
|
|
}
|
|
|
|
|
2022-04-07 17:39:34 +02:00
|
|
|
/// Deserialize a plugin's state from a [`PluginState`] object. This is used to allow the plugin to
|
|
|
|
/// do its own internal preset management. Returns `false` and logs an error if the state could not
|
|
|
|
/// be deserialized.
|
2022-03-02 16:00:11 +01:00
|
|
|
///
|
2022-04-24 19:45:22 +02:00
|
|
|
/// This uses a parameter getter function to avoid having to rebuild the parameter map, which may
|
|
|
|
/// include expensive user written code. See [`make_params_getter()`].
|
|
|
|
///
|
2022-03-02 16:00:11 +01:00
|
|
|
/// Make sure to reinitialize plugin after deserializing the state so it can react to the new
|
|
|
|
/// parameter values. The smoothers have already been reset by this function.
|
2022-04-07 17:12:08 +02:00
|
|
|
pub(crate) unsafe fn deserialize_object(
|
2022-04-07 17:39:34 +02:00
|
|
|
state: &PluginState,
|
2022-04-07 15:31:46 +02:00
|
|
|
plugin_params: Arc<dyn Params>,
|
2022-04-24 20:03:30 +02:00
|
|
|
params_getter: impl Fn(&str) -> Option<ParamPtr>,
|
2022-03-02 16:00:11 +01:00
|
|
|
current_buffer_config: Option<&BufferConfig>,
|
|
|
|
) -> bool {
|
|
|
|
let sample_rate = current_buffer_config.map(|c| c.sample_rate);
|
2022-04-07 17:12:08 +02:00
|
|
|
for (param_id_str, param_value) in &state.params {
|
2022-04-24 19:45:22 +02:00
|
|
|
let param_ptr = match params_getter(param_id_str.as_str()) {
|
2022-04-06 13:34:32 +02:00
|
|
|
Some(ptr) => ptr,
|
|
|
|
None => {
|
|
|
|
nih_debug_assert_failure!("Unknown parameter: {}", param_id_str);
|
2022-03-02 16:00:11 +01:00
|
|
|
continue;
|
|
|
|
}
|
2022-04-06 13:34:32 +02:00
|
|
|
};
|
2022-03-02 16:00:11 +01:00
|
|
|
|
2022-04-06 13:34:32 +02:00
|
|
|
match (param_ptr, param_value) {
|
2022-04-24 19:45:22 +02:00
|
|
|
(ParamPtr::FloatParam(p), ParamValue::F32(v)) => (*p).set_plain_value(*v),
|
|
|
|
(ParamPtr::IntParam(p), ParamValue::I32(v)) => (*p).set_plain_value(*v),
|
|
|
|
(ParamPtr::BoolParam(p), ParamValue::Bool(v)) => (*p).set_plain_value(*v),
|
2022-06-03 22:22:36 +02:00
|
|
|
// Enums are either serialized based on the active variant's index (which may not be the
|
|
|
|
// same as the discriminator), or a custom set stable string ID. The latter allows the
|
|
|
|
// variants to be reordered.
|
2022-04-06 13:34:32 +02:00
|
|
|
(ParamPtr::EnumParam(p), ParamValue::I32(variant_idx)) => {
|
2022-04-24 19:45:22 +02:00
|
|
|
(*p).set_plain_value(*variant_idx)
|
2022-03-23 17:36:58 +01:00
|
|
|
}
|
2022-06-03 22:22:36 +02:00
|
|
|
(ParamPtr::EnumParam(p), ParamValue::String(id)) => {
|
|
|
|
let deserialized_enum = (*p).set_from_id(id);
|
|
|
|
nih_debug_assert!(
|
|
|
|
deserialized_enum,
|
|
|
|
"Unknown ID {:?} for enum parameter \"{}\"",
|
|
|
|
id,
|
|
|
|
param_id_str,
|
|
|
|
);
|
|
|
|
}
|
2022-04-06 13:34:32 +02:00
|
|
|
(param_ptr, param_value) => {
|
|
|
|
nih_debug_assert_failure!(
|
|
|
|
"Invalid serialized value {:?} for parameter \"{}\" ({:?})",
|
|
|
|
param_value,
|
|
|
|
param_id_str,
|
|
|
|
param_ptr,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure everything starts out in sync
|
|
|
|
if let Some(sample_rate) = sample_rate {
|
|
|
|
param_ptr.update_smoother(sample_rate, true);
|
2022-03-02 16:00:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// The plugin can also persist arbitrary fields alongside its parameters. This is useful for
|
|
|
|
// storing things like sample data.
|
|
|
|
plugin_params.deserialize_fields(&state.fields);
|
|
|
|
|
|
|
|
true
|
|
|
|
}
|
2022-04-07 17:12:08 +02:00
|
|
|
|
2022-08-18 13:55:31 +02:00
|
|
|
/// Deserialize a plugin's state from a vector containing (compressed) JSON data. This can (and
|
|
|
|
/// should) be shared across plugin formats. Returns `false` and logs an error if the state could
|
|
|
|
/// not be deserialized. If the `zstd` feature is enabled, then this can
|
2022-04-07 17:12:08 +02:00
|
|
|
///
|
|
|
|
/// Make sure to reinitialize plugin after deserializing the state so it can react to the new
|
|
|
|
/// parameter values. The smoothers have already been reset by this function.
|
|
|
|
pub(crate) unsafe fn deserialize_json(
|
|
|
|
state: &[u8],
|
|
|
|
plugin_params: Arc<dyn Params>,
|
2022-04-24 20:03:30 +02:00
|
|
|
params_getter: impl Fn(&str) -> Option<ParamPtr>,
|
2022-04-07 17:12:08 +02:00
|
|
|
current_buffer_config: Option<&BufferConfig>,
|
|
|
|
) -> bool {
|
2022-08-18 13:55:31 +02:00
|
|
|
#[cfg(feature = "zstd")]
|
|
|
|
let state: PluginState = match zstd::decode_all(state) {
|
|
|
|
Ok(decompressed) => match serde_json::from_slice(decompressed.as_slice()) {
|
|
|
|
Ok(s) => {
|
|
|
|
nih_log!("Deserialized compressed");
|
|
|
|
s
|
|
|
|
}
|
|
|
|
Err(err) => {
|
|
|
|
nih_debug_assert_failure!("Error while deserializing state: {}", err);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// Uncompressed state files can still be loaded after enabling this feature to prevent
|
|
|
|
// breaking existing plugin instances
|
|
|
|
Err(zstd_err) => match serde_json::from_slice(state) {
|
|
|
|
Ok(s) => {
|
|
|
|
nih_log!("Deserialized uncompressed");
|
|
|
|
s
|
|
|
|
}
|
|
|
|
Err(json_err) => {
|
|
|
|
nih_debug_assert_failure!(
|
|
|
|
"Error while deserializing state as either compressed or uncompressed state: \
|
|
|
|
{}, {}",
|
|
|
|
zstd_err,
|
|
|
|
json_err
|
|
|
|
);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
#[cfg(not(feature = "zstd"))]
|
2022-04-07 17:39:34 +02:00
|
|
|
let state: PluginState = match serde_json::from_slice(state) {
|
2022-04-07 17:12:08 +02:00
|
|
|
Ok(s) => s,
|
|
|
|
Err(err) => {
|
|
|
|
nih_debug_assert_failure!("Error while deserializing state: {}", err);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-04-24 19:45:22 +02:00
|
|
|
deserialize_object(&state, plugin_params, params_getter, current_buffer_config)
|
2022-04-07 17:12:08 +02:00
|
|
|
}
|