summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDavid Robillard <d@drobilla.net>2011-05-14 21:44:47 +0000
committerDavid Robillard <d@drobilla.net>2011-05-14 21:44:47 +0000
commit3a855fb69a1757238d07326b8056329569022a12 (patch)
tree5346204b02539bece5f60df60d9e4d4830f7f9c5
parent4e49c56361ef06d75696e4eac77ba59c1ed1aa3f (diff)
Move SLV2-independent LV2 persist state implementation to lv2_state.h.
git-svn-id: svn://localhost/ardour2/branches/3.0@9515 d708f5d6-7413-0410-9779-e7cbd77b26cf
-rw-r--r--libs/ardour/ardour/lv2_plugin.h2
-rw-r--r--libs/ardour/ardour/lv2_state.h143
-rw-r--r--libs/ardour/lv2_plugin.cc136
3 files changed, 160 insertions, 121 deletions
diff --git a/libs/ardour/ardour/lv2_plugin.h b/libs/ardour/ardour/lv2_plugin.h
index 7f978b81b1..af06a07ecf 100644
--- a/libs/ardour/ardour/lv2_plugin.h
+++ b/libs/ardour/ardour/lv2_plugin.h
@@ -64,7 +64,7 @@ class LV2Plugin : public ARDOUR::Plugin
int get_parameter_descriptor (uint32_t which, ParameterDescriptor&) const;
uint32_t nth_parameter (uint32_t port, bool& ok) const;
- const void* extension_data (const char* uri);
+ const void* extension_data (const char* uri) const;
void* c_plugin();
void* c_ui();
diff --git a/libs/ardour/ardour/lv2_state.h b/libs/ardour/ardour/lv2_state.h
new file mode 100644
index 0000000000..c8e2d01f02
--- /dev/null
+++ b/libs/ardour/ardour/lv2_state.h
@@ -0,0 +1,143 @@
+/*
+ Copyright (C) 2011 Paul Davis
+ Author: David Robillard
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+*/
+
+#ifndef __ardour_lv2_state_h__
+#define __ardour_lv2_state_h__
+
+#include <stdint.h>
+#include <stdlib.h>
+
+#include <map>
+#include <string>
+
+#include "pbd/error.h"
+
+#include "ardour/uri_map.h"
+#include "lv2ext/lv2_persist.h"
+#include "rdff.h"
+
+namespace ARDOUR {
+
+struct LV2PersistState {
+ LV2PersistState(URIMap& map) : uri_map(map) {}
+
+ struct Value {
+ inline Value(uint32_t k, const void* v, size_t s, uint32_t t, uint32_t f)
+ : key(k), value(v), size(s), type(t), flags(f)
+ {}
+
+ const uint32_t key;
+ const void* value;
+ const size_t size;
+ const uint32_t type;
+ const uint32_t flags;
+ };
+
+ typedef std::map<uint32_t, std::string> URIs;
+ typedef std::map<uint32_t, Value> Values;
+
+ uint32_t file_id_to_runtime_id(uint32_t file_id) const {
+ URIs::const_iterator i = uris.find(file_id);
+ if (i == uris.end()) {
+ PBD::error << "LV2 state refers to undefined URI ID" << endmsg;
+ return 0;
+ }
+ return uri_map.uri_to_id(NULL, i->second.c_str());
+ }
+
+ int add_uri(uint32_t file_id, const char* str) {
+ // TODO: check for clashes (invalid file)
+ uris.insert(std::make_pair(file_id, str));
+ return 0;
+ }
+
+ int add_value(uint32_t file_key,
+ const void* value,
+ size_t size,
+ uint32_t file_type,
+ uint32_t flags) {
+ const uint32_t key = file_id_to_runtime_id(file_key);
+ const uint32_t type = file_id_to_runtime_id(file_type);
+ if (!key || !type) {
+ return 1;
+ }
+
+ Values::const_iterator i = values.find(key);
+ if (i != values.end()) {
+ PBD::error << "LV2 state contains duplicate keys" << endmsg;
+ return 1;
+ } else {
+ void* value_copy = malloc(size);
+ memcpy(value_copy, value, size); // FIXME: leak
+ values.insert(
+ std::make_pair(key,
+ Value(key, value_copy, size, type, flags)));
+ return 0;
+ }
+ }
+
+ void read(RDFF file) {
+ RDFFChunk* chunk = (RDFFChunk*)malloc(sizeof(RDFFChunk));
+ chunk->size = 0;
+ while (!rdff_read_chunk(file, &chunk)) {
+ if (rdff_chunk_is_uri(chunk)) {
+ RDFFURIChunk* body = (RDFFURIChunk*)chunk->data;
+ add_uri(body->id, body->uri);
+ } else if (rdff_chunk_is_triple(chunk)) {
+ RDFFTripleChunk* body = (RDFFTripleChunk*)chunk->data;
+ add_value(body->predicate,
+ body->object,
+ body->object_size,
+ body->object_type,
+ LV2_PERSIST_IS_POD | LV2_PERSIST_IS_PORTABLE);
+ }
+ }
+ free(chunk);
+ }
+
+ void write(RDFF file) {
+ // Write all referenced URIs to state file
+ for (URIs::const_iterator i = uris.begin(); i != uris.end(); ++i) {
+ rdff_write_uri(file,
+ i->first,
+ i->second.length(),
+ i->second.c_str());
+ }
+
+ // Write all values to state file
+ for (Values::const_iterator i = values.begin(); i != values.end(); ++i) {
+ const uint32_t key = i->first;
+ const LV2PersistState::Value& val = i->second;
+ rdff_write_triple(file,
+ 0,
+ key,
+ val.type,
+ val.size,
+ val.value);
+ }
+ }
+
+ URIMap& uri_map;
+ URIs uris;
+ Values values;
+};
+
+} // namespace ARDOUR
+
+#endif /* __ardour_lv2_state_h__ */
diff --git a/libs/ardour/lv2_plugin.cc b/libs/ardour/lv2_plugin.cc
index bc83e3e5fe..44030c594c 100644
--- a/libs/ardour/lv2_plugin.cc
+++ b/libs/ardour/lv2_plugin.cc
@@ -43,6 +43,7 @@
#include "ardour/debug.h"
#include "ardour/lv2_event_buffer.h"
#include "ardour/lv2_plugin.h"
+#include "ardour/lv2_state.h"
#include "ardour/session.h"
#include "i18n.h"
@@ -421,9 +422,9 @@ LV2Plugin::nth_parameter(uint32_t n, bool& ok) const
}
const void*
-LV2Plugin::extension_data (const char* uri)
+LV2Plugin::extension_data (const char* uri) const
{
- return _impl->instance->lv2_descriptor->extension_data (uri);
+ return slv2_instance_get_extension_data(_impl->instance, uri);
}
void*
@@ -438,69 +439,6 @@ LV2Plugin::c_ui ()
return _impl->ui;
}
-struct PersistValue {
- inline PersistValue(uint32_t k, const void* v, size_t s, uint32_t t, uint32_t f)
- : key(k), value(v), size(s), type(t), flags(f)
- {}
-
- const uint32_t key;
- const void* value;
- const size_t size;
- const uint32_t type;
- const bool flags;
-};
-
-struct PersistState {
- PersistState(URIMap& map) : uri_map(map) {}
-
- typedef std::map<uint32_t, std::string> URIs;
- typedef std::map<uint32_t, PersistValue> Values;
-
- uint32_t file_id_to_runtime_id(uint32_t file_id) const {
- URIs::const_iterator i = uris.find(file_id);
- if (i == uris.end()) {
- error << "LV2 state refers to undefined URI ID" << endmsg;
- return 0;
- }
- return uri_map.uri_to_id(NULL, i->second.c_str());
- }
-
- int add_uri(uint32_t file_id, const char* str) {
- // TODO: check for clashes (invalid file)
- uris.insert(make_pair(file_id, str));
- return 0;
- }
-
- int add_value(uint32_t file_key,
- const void* value,
- size_t size,
- uint32_t file_type,
- uint32_t flags) {
- const uint32_t key = file_id_to_runtime_id(file_key);
- const uint32_t type = file_id_to_runtime_id(file_type);
- if (!key || !type) {
- return 1;
- }
-
- Values::const_iterator i = values.find(key);
- if (i != values.end()) {
- error << "LV2 state contains duplicate keys" << endmsg;
- return 1;
- } else {
- void* value_copy = malloc(size);
- memcpy(value_copy, value, size); // FIXME: leak
- values.insert(
- make_pair(key,
- PersistValue(key, value_copy, size, type, flags)));
- return 0;
- }
- }
-
- URIMap& uri_map;
- URIs uris;
- Values values;
-};
-
int
LV2Plugin::lv2_persist_store_callback(void* host_data,
uint32_t key,
@@ -515,7 +453,7 @@ LV2Plugin::lv2_persist_store_callback(void* host_data,
size,
_uri_map.id_to_uri(NULL, type)));
- PersistState* state = (PersistState*)host_data;
+ LV2PersistState* state = (LV2PersistState*)host_data;
state->add_uri(key, _uri_map.id_to_uri(NULL, key));
state->add_uri(type, _uri_map.id_to_uri(NULL, type));
return state->add_value(key, value, size, type, flags);
@@ -528,8 +466,8 @@ LV2Plugin::lv2_persist_retrieve_callback(void* host_data,
uint32_t* type,
uint32_t* flags)
{
- PersistState* state = (PersistState*)host_data;
- PersistState::Values::const_iterator i = state->values.find(key);
+ LV2PersistState* state = (LV2PersistState*)host_data;
+ LV2PersistState::Values::const_iterator i = state->values.find(key);
if (i == state->values.end()) {
warning << "LV2 plugin attempted to retrieve nonexistent key: "
<< _uri_map.id_to_uri(NULL, key) << endmsg;
@@ -650,8 +588,8 @@ LV2Plugin::add_state(XMLNode* root) const
cout << "Saving LV2 plugin state to " << state_path << endl;
// Get LV2 Persist extension data from plugin instance
- LV2_Persist* persist = (LV2_Persist*)slv2_instance_get_extension_data(
- _impl->instance, "http://lv2plug.in/ns/ext/persist");
+ LV2_Persist* persist = (LV2_Persist*)extension_data(
+ "http://lv2plug.in/ns/ext/persist");
if (!persist) {
warning << string_compose(
_("Plugin \"%1\% failed to return LV2 persist data"),
@@ -660,37 +598,14 @@ LV2Plugin::add_state(XMLNode* root) const
}
// Save plugin state to state object
- PersistState state(_uri_map);
+ LV2PersistState state(_uri_map);
persist->save(_impl->instance->lv2_handle,
&LV2Plugin::lv2_persist_store_callback,
&state);
- // Open state file
+ // Write state object to RDFF file
RDFF file = rdff_open(state_path.c_str(), true);
-
- // Write all referenced URIs to state file
- for (PersistState::URIs::const_iterator i = state.uris.begin();
- i != state.uris.end(); ++i) {
- rdff_write_uri(file,
- i->first,
- i->second.length(),
- i->second.c_str());
- }
-
- // Write all values to state file
- for (PersistState::Values::const_iterator i = state.values.begin();
- i != state.values.end(); ++i) {
- const uint32_t key = i->first;
- const PersistValue& val = i->second;
- rdff_write_triple(file,
- 0,
- key,
- val.type,
- val.size,
- val.value);
- }
-
- // Close state file
+ state.write(file);
rdff_close(file);
root->add_property("state-file", state_filename);
@@ -862,32 +777,13 @@ LV2Plugin::set_state(const XMLNode& node, int version)
cout << "LV2 state path " << state_path << endl;
// Get LV2 Persist extension data from plugin instance
- LV2_Persist* persist = (LV2_Persist*)slv2_instance_get_extension_data(
- _impl->instance, "http://lv2plug.in/ns/ext/persist");
+ LV2_Persist* persist = (LV2_Persist*)extension_data(
+ "http://lv2plug.in/ns/ext/persist");
if (persist) {
cout << "Loading LV2 state from " << state_path << endl;
- RDFF file = rdff_open(state_path.c_str(), false);
-
- PersistState state(_uri_map);
-
- // Load file into state object
- RDFFChunk* chunk = (RDFFChunk*)malloc(sizeof(RDFFChunk));
- chunk->size = 0;
- while (!rdff_read_chunk(file, &chunk)) {
- if (rdff_chunk_is_uri(chunk)) {
- RDFFURIChunk* body = (RDFFURIChunk*)chunk->data;
- state.add_uri(body->id, body->uri);
- } else if (rdff_chunk_is_triple(chunk)) {
- RDFFTripleChunk* body = (RDFFTripleChunk*)chunk->data;
- state.add_value(body->predicate,
- body->object,
- body->object_size,
- body->object_type,
- LV2_PERSIST_IS_POD | LV2_PERSIST_IS_PORTABLE);
- }
- }
- free(chunk);
-
+ RDFF file = rdff_open(state_path.c_str(), false);
+ LV2PersistState state(_uri_map);
+ state.read(file);
persist->restore(_impl->instance->lv2_handle,
&LV2Plugin::lv2_persist_retrieve_callback,
&state);