summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDavid Robillard <d@drobilla.net>2020-11-11 23:24:34 +0100
committerDavid Robillard <d@drobilla.net>2020-11-11 23:24:34 +0100
commit4de692ccf8c97cb156029375a10fc6a2aa2928fe (patch)
tree87fcf0bb3d1829216a524b6fda13449a292d901a
parent09e544df518d3ce91744606460553c8ca8b148d3 (diff)
downloadingen-4de692ccf8c97cb156029375a10fc6a2aa2928fe.tar.gz
ingen-4de692ccf8c97cb156029375a10fc6a2aa2928fe.tar.bz2
ingen-4de692ccf8c97cb156029375a10fc6a2aa2928fe.zip
Fix potential state memory leaks
-rw-r--r--src/server/BlockImpl.hpp3
-rw-r--r--src/server/LV2Block.cpp94
-rw-r--r--src/server/LV2Block.hpp4
-rw-r--r--src/server/State.hpp33
-rw-r--r--src/server/events/CreateBlock.cpp4
-rw-r--r--src/server/events/Delta.cpp12
-rw-r--r--src/server/events/Delta.hpp3
7 files changed, 101 insertions, 52 deletions
diff --git a/src/server/BlockImpl.hpp b/src/server/BlockImpl.hpp
index a5f5963a..f6c72c8e 100644
--- a/src/server/BlockImpl.hpp
+++ b/src/server/BlockImpl.hpp
@@ -20,6 +20,7 @@
#include "BufferRef.hpp"
#include "NodeImpl.hpp"
#include "PortType.hpp"
+#include "State.hpp"
#include "types.hpp"
#include "ingen/Node.hpp"
@@ -108,7 +109,7 @@ public:
void set_enabled(bool e) { _enabled = e; }
/** Load a preset from the world for this block. */
- virtual LilvState* load_preset(const URI& uri) { return nullptr; }
+ virtual StatePtr load_preset(const URI& uri) { return {}; }
/** Restore `state`. */
virtual void
diff --git a/src/server/LV2Block.cpp b/src/server/LV2Block.cpp
index 3d180360..787488d5 100644
--- a/src/server/LV2Block.cpp
+++ b/src/server/LV2Block.cpp
@@ -454,9 +454,10 @@ LV2Block::instantiate(BufferFactory& bufs, const LilvState* state)
}
// Load initial state if no state is explicitly given
- LilvState* default_state = nullptr;
+ StatePtr default_state{};
if (!state) {
- state = default_state = load_preset(_lv2_plugin->uri());
+ default_state = load_preset(_lv2_plugin->uri());
+ state = default_state.get();
}
// Apply state
@@ -464,10 +465,6 @@ LV2Block::instantiate(BufferFactory& bufs, const LilvState* state)
apply_state(nullptr, state);
}
- if (default_state) {
- lilv_state_free(default_state);
- }
-
// FIXME: Polyphony + worker?
if (lilv_plugin_has_feature(plug, uris.work_schedule)) {
_worker_iface = static_cast<const LV2_Worker_Interface*>(
@@ -484,29 +481,33 @@ LV2Block::save_state(const FilePath& dir) const
World& world = _lv2_plugin->world();
LilvWorld* lworld = world.lilv_world();
- LilvState* state = lilv_state_new_from_instance(
- _lv2_plugin->lilv_plugin(), const_cast<LV2Block*>(this)->instance(0),
- &world.uri_map().urid_map(),
- nullptr, dir.c_str(), dir.c_str(), dir.c_str(), nullptr, nullptr,
- LV2_STATE_IS_POD|LV2_STATE_IS_PORTABLE, nullptr);
+ StatePtr state{
+ lilv_state_new_from_instance(_lv2_plugin->lilv_plugin(),
+ const_cast<LV2Block*>(this)->instance(0),
+ &world.uri_map().urid_map(),
+ nullptr,
+ dir.c_str(),
+ dir.c_str(),
+ dir.c_str(),
+ nullptr,
+ nullptr,
+ LV2_STATE_IS_POD | LV2_STATE_IS_PORTABLE,
+ nullptr)};
if (!state) {
return false;
- } else if (lilv_state_get_num_properties(state) == 0) {
- lilv_state_free(state);
+ } else if (lilv_state_get_num_properties(state.get()) == 0) {
return false;
}
lilv_state_save(lworld,
&world.uri_map().urid_map(),
&world.uri_map().urid_unmap(),
- state,
+ state.get(),
nullptr,
dir.c_str(),
"state.ttl");
- lilv_state_free(state);
-
return true;
}
@@ -518,14 +519,22 @@ LV2Block::duplicate(Engine& engine,
const SampleRate rate = engine.sample_rate();
// Get current state
- LilvState* state = lilv_state_new_from_instance(
- _lv2_plugin->lilv_plugin(), instance(0),
- &engine.world().uri_map().urid_map(),
- nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, LV2_STATE_IS_NATIVE, nullptr);
+ StatePtr state{
+ lilv_state_new_from_instance(_lv2_plugin->lilv_plugin(),
+ instance(0),
+ &engine.world().uri_map().urid_map(),
+ nullptr,
+ nullptr,
+ nullptr,
+ nullptr,
+ nullptr,
+ nullptr,
+ LV2_STATE_IS_NATIVE,
+ nullptr)};
// Duplicate and instantiate block
auto* dup = new LV2Block(_lv2_plugin, symbol, _polyphonic, parent, rate);
- if (!dup->instantiate(*engine.buffer_factory(), state)) {
+ if (!dup->instantiate(*engine.buffer_factory(), state.get())) {
delete dup;
return nullptr;
}
@@ -623,7 +632,7 @@ LV2Block::post_process(RunContext& ctx)
BlockImpl::post_process(ctx);
}
-LilvState*
+StatePtr
LV2Block::load_preset(const URI& uri)
{
World& world = _lv2_plugin->world();
@@ -634,25 +643,22 @@ LV2Block::load_preset(const URI& uri)
lilv_world_load_resource(lworld, preset);
// Load preset from world
- LV2_URID_Map* map = &world.uri_map().urid_map();
- LilvState* state = lilv_state_new_from_world(lworld, map, preset);
+ LV2_URID_Map* map = &world.uri_map().urid_map();
+ StatePtr state{lilv_state_new_from_world(lworld, map, preset)};
lilv_node_free(preset);
return state;
}
-LilvState*
+StatePtr
LV2Block::load_state(World& world, const FilePath& path)
{
LilvWorld* lworld = world.lilv_world();
const URI uri = URI(path);
LilvNode* subject = lilv_new_uri(lworld, uri.c_str());
- LilvState* state = lilv_state_new_from_file(
- lworld,
- &world.uri_map().urid_map(),
- subject,
- path.c_str());
+ StatePtr state{lilv_state_new_from_file(
+ lworld, &world.uri_map().urid_map(), subject, path.c_str())};
lilv_node_free(subject);
return state;
@@ -709,25 +715,31 @@ LV2Block::save_preset(const URI& uri,
const FilePath dirname = path.parent_path();
const FilePath basename = path.stem();
- LilvState* state = lilv_state_new_from_instance(
- _lv2_plugin->lilv_plugin(), instance(0), lmap,
- nullptr, nullptr, nullptr, path.c_str(),
- get_port_value, this, LV2_STATE_IS_NATIVE, nullptr);
+ StatePtr state{lilv_state_new_from_instance(_lv2_plugin->lilv_plugin(),
+ instance(0),
+ lmap,
+ nullptr,
+ nullptr,
+ nullptr,
+ path.c_str(),
+ get_port_value,
+ this,
+ LV2_STATE_IS_NATIVE,
+ nullptr)};
if (state) {
const Properties::const_iterator l = props.find(_uris.rdfs_label);
if (l != props.end() && l->second.type() == _uris.atom_String) {
- lilv_state_set_label(state, l->second.ptr<char>());
+ lilv_state_set_label(state.get(), l->second.ptr<char>());
}
- lilv_state_save(lworld, lmap, lunmap, state, nullptr,
+ lilv_state_save(lworld, lmap, lunmap, state.get(), nullptr,
dirname.c_str(), basename.c_str());
- const URI state_uri(lilv_node_as_uri(lilv_state_get_uri(state)));
- const std::string label(lilv_state_get_label(state)
- ? lilv_state_get_label(state)
- : basename);
- lilv_state_free(state);
+ const URI state_uri(lilv_node_as_uri(lilv_state_get_uri(state.get())));
+ const std::string label(lilv_state_get_label(state.get())
+ ? lilv_state_get_label(state.get())
+ : basename);
Resource preset(_uris, state_uri);
preset.set_property(_uris.rdf_type, _uris.pset_Preset);
diff --git a/src/server/LV2Block.hpp b/src/server/LV2Block.hpp
index 7a0a86ac..ec9c5ac3 100644
--- a/src/server/LV2Block.hpp
+++ b/src/server/LV2Block.hpp
@@ -95,7 +95,7 @@ public:
void run(RunContext& ctx) override;
void post_process(RunContext& ctx) override;
- LilvState* load_preset(const URI& uri) override;
+ StatePtr load_preset(const URI& uri) override;
void apply_state(const std::unique_ptr<Worker>& worker,
const LilvState* state) override;
@@ -108,7 +108,7 @@ public:
const BufferRef& buf,
SampleCount offset) override;
- static LilvState* load_state(World& world, const FilePath& path);
+ static StatePtr load_state(World& world, const FilePath& path);
protected:
struct Instance : public Raul::Noncopyable {
diff --git a/src/server/State.hpp b/src/server/State.hpp
new file mode 100644
index 00000000..3b745bdb
--- /dev/null
+++ b/src/server/State.hpp
@@ -0,0 +1,33 @@
+/*
+ This file is part of Ingen.
+ Copyright 2020 David Robillard <http://drobilla.net/>
+
+ Ingen is free software: you can redistribute it and/or modify it under the
+ terms of the GNU Affero General Public License as published by the Free
+ Software Foundation, either version 3 of the License, or any later version.
+
+ Ingen 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 Affero General Public License for details.
+
+ You should have received a copy of the GNU Affero General Public License
+ along with Ingen. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+#ifndef INGEN_ENGINE_STATE_HPP
+#define INGEN_ENGINE_STATE_HPP
+
+namespace ingen {
+namespace server {
+
+struct StateDeleter
+{
+ void operator()(LilvState* state) { lilv_state_free(state); };
+};
+
+using StatePtr = std::unique_ptr<LilvState, StateDeleter>;
+
+} // namespace server
+} // namespace ingen
+
+#endif // INGEN_ENGINE_STATE_HPP
diff --git a/src/server/events/CreateBlock.cpp b/src/server/events/CreateBlock.cpp
index 712a9966..4dc64bdc 100644
--- a/src/server/events/CreateBlock.cpp
+++ b/src/server/events/CreateBlock.cpp
@@ -134,7 +134,7 @@ CreateBlock::pre_process(PreProcessContext& ctx)
}
// Load state from directory if given in properties
- LilvState* state = nullptr;
+ StatePtr state{};
auto s = _properties.find(uris.state_state);
if (s != _properties.end() && s->second.type() == uris.forge.Path) {
state = LV2Block::load_state(
@@ -147,7 +147,7 @@ CreateBlock::pre_process(PreProcessContext& ctx)
polyphonic,
_graph,
_engine,
- state))) {
+ state.get()))) {
return Event::pre_process_done(Status::CREATION_FAILED, _path);
}
}
diff --git a/src/server/events/Delta.cpp b/src/server/events/Delta.cpp
index 33611b7a..911ace66 100644
--- a/src/server/events/Delta.cpp
+++ b/src/server/events/Delta.cpp
@@ -72,7 +72,7 @@ Delta::Delta(Engine& engine,
, _object(nullptr)
, _graph(nullptr)
, _binding(nullptr)
- , _state(nullptr)
+ , _state()
, _context(msg.ctx)
, _type(Type::PUT)
, _block(false)
@@ -390,8 +390,9 @@ Delta::pre_process(PreProcessContext& ctx)
if (!uri.empty()) {
op = SpecialType::PRESET;
if ((_state = block->load_preset(uri))) {
- lilv_state_emit_port_values(
- _state, s_add_set_event, this);
+ lilv_state_emit_port_values(_state.get(),
+ s_add_set_event,
+ this);
} else {
_engine.log().warn("Failed to load preset <%1%>\n", uri);
}
@@ -597,10 +598,11 @@ Delta::post_process()
if (_state) {
auto* block = dynamic_cast<BlockImpl*>(_object);
if (block) {
- block->apply_state(_engine.sync_worker(), _state);
+ block->apply_state(_engine.sync_worker(), _state.get());
block->set_enabled(true);
}
- lilv_state_free(_state);
+
+ _state.reset();
}
Broadcaster::Transfer t(*_engine.broadcaster());
diff --git a/src/server/events/Delta.hpp b/src/server/events/Delta.hpp
index 4d749f02..fc4716f2 100644
--- a/src/server/events/Delta.hpp
+++ b/src/server/events/Delta.hpp
@@ -17,6 +17,7 @@
#ifndef INGEN_EVENTS_DELTA_HPP
#define INGEN_EVENTS_DELTA_HPP
+#include "BlockImpl.hpp"
#include "ClientUpdate.hpp"
#include "CompiledGraph.hpp"
#include "ControlBindings.hpp"
@@ -119,7 +120,7 @@ private:
GraphImpl* _graph;
Raul::managed_ptr<CompiledGraph> _compiled_graph;
ControlBindings::Binding* _binding;
- LilvState* _state;
+ StatePtr _state;
Resource::Graph _context;
Type _type;