From fe294dfcf5b009d7dd5dd18b7a26f71eb619a893 Mon Sep 17 00:00:00 2001 From: David Robillard Date: Sun, 24 Jul 2016 19:16:10 -0400 Subject: Add lilv_state_set_metadata() This allows setting useful metadata on a state description accessible to hosts but not plugins, such as pset:bank and rdfs:comment. Based on patch from Hanspeter Portner. --- NEWS | 5 +- lilv/lilv.h | 22 +++++++ src/state.c | 194 +++++++++++++++++++++++++++++++++++-------------------- test/lilv_test.c | 12 ++++ 4 files changed, 162 insertions(+), 71 deletions(-) diff --git a/NEWS b/NEWS index ffa5429..8310abf 100644 --- a/NEWS +++ b/NEWS @@ -1,5 +1,8 @@ lilv (0.22.1) unstable; + * Add lilv_state_set_metadata() for adding state banks/comments/etc + (based on patch from Hanspeter Portner) + * Fix crash when state contains non-POD properties * Fix state file versioning * Unload contained resources when bundle is unloaded * Do not instantiate plugin when data fails to parse @@ -13,7 +16,7 @@ lilv (0.22.1) unstable; * Fix documentation installation * Fix outdated comment references to lilv_uri_to_path() - -- David Robillard Tue, 12 Jul 2016 21:22:40 -0400 + -- David Robillard Sun, 24 Jul 2016 19:16:57 -0400 lilv (0.22.0) stable; diff --git a/lilv/lilv.h b/lilv/lilv.h index 8418e9a..af3873e 100644 --- a/lilv/lilv.h +++ b/lilv/lilv.h @@ -1382,6 +1382,28 @@ LILV_API void lilv_state_set_label(LilvState* state, const char* label); +/** + Set a metadata property on `state`. + @param state The state to set the metadata for. + @param key The key to store `value` under (URID). + @param value Pointer to the value to be stored. + @param size The size of `value` in bytes. + @param type The type of `value` (URID). + @param flags LV2_State_Flags for `value`. + @return 0 on success. + + This is a generic version of lilv_state_set_label(), which sets metadata + properties visible to hosts, but not plugins. This allows storing useful + information such as comments or preset banks. +*/ +LILV_API int +lilv_state_set_metadata(LilvState* state, + uint32_t key, + const void* value, + size_t size, + uint32_t type, + uint32_t flags); + /** Function to set a port value. @param port_symbol The symbol of the port. diff --git a/src/state.c b/src/state.c index 137d1a7..f73e157 100644 --- a/src/state.c +++ b/src/state.c @@ -49,21 +49,26 @@ typedef struct { char* rel; ///< Abstract path (relative path in state dir) } PathMap; +typedef struct { + size_t n; + Property* props; +} PropertyArray; + struct LilvStateImpl { - LilvNode* plugin_uri; ///< Plugin URI - LilvNode* uri; ///< State/preset URI - char* dir; ///< Save directory (if saved) - char* file_dir; ///< Directory for files created by plugin - char* copy_dir; ///< Directory for snapshots of external files - char* link_dir; ///< Directory for links to external files - char* label; ///< State/Preset label - ZixTree* abs2rel; ///< PathMap sorted by abs - ZixTree* rel2abs; ///< PathMap sorted by rel - Property* props; ///< State properties - PortValue* values; ///< Port values - uint32_t atom_Path; ///< atom:Path URID - uint32_t n_props; ///< Number of state properties - uint32_t n_values; ///< Number of port values + LilvNode* plugin_uri; ///< Plugin URI + LilvNode* uri; ///< State/preset URI + char* dir; ///< Save directory (if saved) + char* file_dir; ///< Directory for files created by plugin + char* copy_dir; ///< Directory for snapshots of external files + char* link_dir; ///< Directory for links to external files + char* label; ///< State/Preset label + ZixTree* abs2rel; ///< PathMap sorted by abs + ZixTree* rel2abs; ///< PathMap sorted by rel + PropertyArray props; ///< State properties + PropertyArray metadata; ///< State metadata + PortValue* values; ///< Port values + uint32_t atom_Path; ///< atom:Path URID + uint32_t n_values; ///< Number of port values }; static int @@ -131,24 +136,23 @@ lilv_state_rel2abs(const LilvState* state, const char* path) return path; } -static LV2_State_Status -store_callback(LV2_State_Handle handle, - uint32_t key, - const void* value, - size_t size, - uint32_t type, - uint32_t flags) +static void +append_property(LilvState* state, + PropertyArray* array, + uint32_t key, + const void* value, + size_t size, + uint32_t type, + uint32_t flags) { - LilvState* const state = (LilvState*)handle; - state->props = (Property*)realloc( - state->props, (++state->n_props) * sizeof(Property)); - Property* const prop = &state->props[state->n_props - 1]; + array->props = (Property*)realloc( + array->props, (++array->n) * sizeof(Property)); + Property* const prop = &array->props[array->n - 1]; if ((flags & LV2_STATE_IS_POD) || type == state->atom_Path) { prop->value = malloc(size); memcpy(prop->value, value, size); } else { - LILV_WARN("Storing non-POD value\n"); prop->value = (void*)value; } @@ -156,7 +160,18 @@ store_callback(LV2_State_Handle handle, prop->key = key; prop->type = type; prop->flags = flags; +} +static LV2_State_Status +store_callback(LV2_State_Handle handle, + uint32_t key, + const void* value, + size_t size, + uint32_t type, + uint32_t flags) +{ + LilvState* const state = (LilvState*)handle; + append_property(state, &state->props, key, value, size, type, flags); return LV2_STATE_SUCCESS; } @@ -170,7 +185,7 @@ retrieve_callback(LV2_State_Handle handle, const LilvState* const state = (LilvState*)handle; const Property search_key = { NULL, 0, key, 0, 0 }; const Property* const prop = (Property*)bsearch( - &search_key, state->props, state->n_props, + &search_key, state->props.props, state->props.n, sizeof(Property), property_cmp); if (prop) { @@ -388,11 +403,11 @@ lilv_state_new_from_instance(const LilvPlugin* plugin, instance->lv2_handle, store_callback, state, flags, features); if (st) { LILV_ERRORF("Error saving plugin state: %s\n", state_strerror(st)); - free(state->props); - state->props = NULL; - state->n_props = 0; + free(state->props.props); + state->props.props = NULL; + state->props.n = 0; } else { - qsort(state->props, state->n_props, sizeof(Property), property_cmp); + qsort(state->props.props, state->props.n, sizeof(Property), property_cmp); } } @@ -576,9 +591,9 @@ new_state_from_model(LilvWorld* world, } if (prop.value) { - state->props = (Property*)realloc( - state->props, (++state->n_props) * sizeof(Property)); - state->props[state->n_props - 1] = prop; + state->props.props = (Property*)realloc( + state->props.props, (++state->props.n) * sizeof(Property)); + state->props.props[state->props.n - 1] = prop; } } sord_iter_free(props); @@ -589,7 +604,7 @@ new_state_from_model(LilvWorld* world, free((void*)chunk.buf); sratom_free(sratom); - qsort(state->props, state->n_props, sizeof(Property), property_cmp); + qsort(state->props.props, state->props.n, sizeof(Property), property_cmp); qsort(state->values, state->n_values, sizeof(PortValue), value_cmp); return state; @@ -864,6 +879,37 @@ link_exists(const char* path, void* data) return !matches; } +static void +write_property_array(const LilvState* state, + const PropertyArray* array, + Sratom* sratom, + uint32_t flags, + const SerdNode* subject, + LV2_URID_Unmap* unmap, + const char* dir) +{ + for (uint32_t i = 0; i < array->n; ++i) { + Property* prop = &array->props[i]; + const char* key = unmap->unmap(unmap->handle, prop->key); + + const SerdNode p = serd_node_from_string(SERD_URI, USTR(key)); + if (prop->type == state->atom_Path && !dir) { + const char* path = (const char*)prop->value; + const char* abs_path = lilv_state_rel2abs(state, path); + LILV_WARNF("Writing absolute path %s\n", abs_path); + sratom_write(sratom, unmap, flags, + subject, &p, prop->type, + strlen(abs_path) + 1, abs_path); + } else if (prop->flags & LV2_STATE_IS_POD || + prop->type == state->atom_Path) { + sratom_write(sratom, unmap, flags, + subject, &p, prop->type, prop->size, prop->value); + } else { + LILV_WARNF("Lost non-POD property <%s> on save\n", key); + } + } +} + static int lilv_state_write(LilvWorld* world, LV2_URID_Map* map, @@ -910,10 +956,13 @@ lilv_state_write(LilvWorld* world, (SerdEndSink)serd_writer_end_anon, writer); - // Write port values as pretty numbers - sratom_set_pretty_numbers(sratom, true); + // Write metadata + sratom_set_pretty_numbers(sratom, false); // Use precise types + write_property_array(state, &state->metadata, sratom, 0, + &subject, unmap, dir); // Write port values + sratom_set_pretty_numbers(sratom, true); // Use pretty numbers for (uint32_t i = 0; i < state->n_values; ++i) { PortValue* const value = &state->values[i]; @@ -939,35 +988,19 @@ lilv_state_write(LilvWorld* world, serd_writer_end_anon(writer, &port); } - // Write property values with precise types - sratom_set_pretty_numbers(sratom, false); - // Write properties - const SerdNode state_node = serd_node_from_string(SERD_BLANK, - USTR("2state")); - if (state->n_props > 0) { + const SerdNode body = serd_node_from_string(SERD_BLANK, USTR("body")); + if (state->props.n > 0) { p = serd_node_from_string(SERD_URI, USTR(LV2_STATE__state)); serd_writer_write_statement(writer, SERD_ANON_O_BEGIN, NULL, - &subject, &p, &state_node, NULL, NULL); + &subject, &p, &body, NULL, NULL); } - for (uint32_t i = 0; i < state->n_props; ++i) { - Property* prop = &state->props[i]; - const char* key = unmap->unmap(unmap->handle, prop->key); + sratom_set_pretty_numbers(sratom, false); // Use precise types + write_property_array(state, &state->props, sratom, SERD_ANON_CONT, + &body, unmap, dir); - p = serd_node_from_string(SERD_URI, USTR(key)); - if (prop->type == state->atom_Path && !dir) { - const char* path = (const char*)prop->value; - const char* abs_path = lilv_state_rel2abs(state, path); - sratom_write(sratom, unmap, SERD_ANON_CONT, - &state_node, &p, prop->type, - strlen(abs_path) + 1, abs_path); - } else { - sratom_write(sratom, unmap, SERD_ANON_CONT, - &state_node, &p, prop->type, prop->size, prop->value); - } - } - if (state->n_props > 0) { - serd_writer_end_anon(writer, &state_node); + if (state->props.n > 0) { + serd_writer_end_anon(writer, &body); } sratom_free(sratom); @@ -1166,13 +1199,23 @@ lilv_state_delete(LilvWorld* world, return 0; } +static void +free_property_array(PropertyArray* array) +{ + for (uint32_t i = 0; i < array->n; ++i) { + if (array->props[i].flags & LV2_STATE_IS_POD) { + free(array->props[i].value); + } + } + free(array->props); +} + LILV_API void lilv_state_free(LilvState* state) { if (state) { - for (uint32_t i = 0; i < state->n_props; ++i) { - free(state->props[i].value); - } + free_property_array(&state->props); + free_property_array(&state->metadata); for (uint32_t i = 0; i < state->n_values; ++i) { free(state->values[i].value); free(state->values[i].symbol); @@ -1181,7 +1224,6 @@ lilv_state_free(LilvState* state) lilv_node_free(state->uri); zix_tree_free(state->abs2rel); zix_tree_free(state->rel2abs); - free(state->props); free(state->values); free(state->label); free(state->dir); @@ -1199,7 +1241,7 @@ lilv_state_equals(const LilvState* a, const LilvState* b) || (a->label && !b->label) || (b->label && !a->label) || (a->label && b->label && strcmp(a->label, b->label)) - || a->n_props != b->n_props + || a->props.n != b->props.n || a->n_values != b->n_values) { return false; } @@ -1214,9 +1256,9 @@ lilv_state_equals(const LilvState* a, const LilvState* b) } } - for (uint32_t i = 0; i < a->n_props; ++i) { - Property* const ap = &a->props[i]; - Property* const bp = &b->props[i]; + for (uint32_t i = 0; i < a->props.n; ++i) { + Property* const ap = &a->props.props[i]; + Property* const bp = &b->props.props[i]; if (ap->key != bp->key || ap->type != bp->type || ap->flags != bp->flags) { @@ -1238,7 +1280,7 @@ lilv_state_equals(const LilvState* a, const LilvState* b) LILV_API unsigned lilv_state_get_num_properties(const LilvState* state) { - return state->n_props; + return state->props.n; } LILV_API const LilvNode* @@ -1266,3 +1308,15 @@ lilv_state_set_label(LilvState* state, const char* label) state->label = (char*)realloc(state->label, len + 1); memcpy(state->label, label, len + 1); } + +LILV_API int +lilv_state_set_metadata(LilvState* state, + uint32_t key, + const void* value, + size_t size, + uint32_t type, + uint32_t flags) +{ + append_property(state, &state->metadata, key, value, size, type, flags); + return LV2_STATE_SUCCESS; +} diff --git a/test/lilv_test.c b/test/lilv_test.c index ca73b42..eb06dcd 100644 --- a/test/lilv_test.c +++ b/test/lilv_test.c @@ -1661,6 +1661,18 @@ test_state(void) get_port_value, world, 0, NULL); TEST_ASSERT(lilv_state_equals(state2, state4)); + // Set some metadata properties + lilv_state_set_metadata(state, map.map(map.handle, LILV_NS_RDFS "comment"), + "This is a comment", + strlen("This is a comment") + 1, + map.map(map.handle, "http://lv2plug.in/ns/ext/atom#Literal"), + LV2_STATE_IS_POD); + lilv_state_set_metadata(state, map.map(map.handle, "http://example.org/metablob"), + "LIVEBEEF", + strlen("LIVEBEEF") + 1, + map.map(map.handle, "http://example.org/MetaBlob"), + 0); + // Save state to a directory int ret = lilv_state_save(world, &map, &unmap, state, NULL, "state/state.lv2", "state.ttl"); -- cgit v1.2.1