/* This file is part of Ingen. Copyright 2007-2016 David Robillard 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 . */ #include "ingen/Atom.hpp" #include "ingen/AtomReader.hpp" #include "ingen/Interface.hpp" #include "ingen/Log.hpp" #include "ingen/Message.hpp" #include "ingen/Properties.hpp" #include "ingen/Resource.hpp" #include "ingen/Status.hpp" #include "ingen/URI.hpp" #include "ingen/URIMap.hpp" #include "ingen/URIs.hpp" #include "ingen/paths.hpp" #include "lv2/atom/atom.h" #include "lv2/atom/util.h" #include "raul/Path.hpp" #include #include #include #include namespace ingen { AtomReader::AtomReader(URIMap& map, URIs& uris, Log& log, Interface& iface) : _map(map) , _uris(uris) , _log(log) , _iface(iface) {} void AtomReader::get_atom(const LV2_Atom* in, Atom& out) { if (in) { if (in->type == _uris.atom_URID) { const auto* const urid = reinterpret_cast(in); const char* const uri = _map.unmap_uri(urid->body); if (uri) { out = Atom(sizeof(int32_t), _uris.atom_URID, &urid->body); } else { _log.error("Unable to unmap URID %1%\n", urid->body); } } else { out = Atom(in->size, in->type, LV2_ATOM_BODY_CONST(in)); } } } void AtomReader::get_props(const LV2_Atom_Object* obj, ingen::Properties& props) { if (obj->body.otype) { const Atom type(sizeof(int32_t), _uris.atom_URID, &obj->body.otype); props.emplace(_uris.rdf_type, type); } LV2_ATOM_OBJECT_FOREACH(obj, p) { Atom val; get_atom(&p->value, val); props.emplace(URI(_map.unmap_uri(p->key)), val); } } boost::optional AtomReader::atom_to_uri(const LV2_Atom* atom) { if (!atom) { return boost::optional(); } if (atom->type == _uris.atom_URI) { const char* str = static_cast(LV2_ATOM_BODY_CONST(atom)); if (URI::is_valid(str)) { return URI(str); } _log.warn("Invalid URI <%1%>\n", str); } else if (atom->type == _uris.atom_Path) { const char* str = static_cast(LV2_ATOM_BODY_CONST(atom)); if (!strncmp(str, "file://", 5)) { return URI(str); } return URI(std::string("file://") + str); } if (atom->type == _uris.atom_URID) { const char* str = _map.unmap_uri(reinterpret_cast(atom)->body); if (str) { return URI(str); } _log.warn("Unknown URID %1%\n", str); } return boost::optional(); } boost::optional AtomReader::atom_to_path(const LV2_Atom* atom) { boost::optional uri = atom_to_uri(atom); if (uri && uri_is_path(*uri)) { return uri_to_path(*uri); } return boost::optional(); } Resource::Graph AtomReader::atom_to_context(const LV2_Atom* atom) { Resource::Graph ctx = Resource::Graph::DEFAULT; if (atom) { boost::optional maybe_uri = atom_to_uri(atom); if (maybe_uri) { ctx = Resource::uri_to_graph(*maybe_uri); } else { _log.warn("Message has invalid context\n"); } } return ctx; } bool AtomReader::is_message(const URIs& uris, const LV2_Atom* msg) { if (msg->type != uris.atom_Object) { return false; } const auto* obj = reinterpret_cast(msg); return (obj->body.otype == uris.patch_Get || obj->body.otype == uris.patch_Delete || obj->body.otype == uris.patch_Put || obj->body.otype == uris.patch_Set || obj->body.otype == uris.patch_Patch || obj->body.otype == uris.patch_Move || obj->body.otype == uris.patch_Response); } bool AtomReader::write(const LV2_Atom* msg, int32_t default_id) { if (msg->type != _uris.atom_Object) { _log.warn("Unknown message type <%1%>\n", _map.unmap_uri(msg->type)); return false; } const auto* const obj = reinterpret_cast(msg); const LV2_Atom* subject = nullptr; const LV2_Atom* number = nullptr; lv2_atom_object_get(obj, _uris.patch_subject.urid(), &subject, _uris.patch_sequenceNumber.urid(), &number, nullptr); const boost::optional subject_uri = atom_to_uri(subject); const int32_t seq = ((number && number->type == _uris.atom_Int) ? reinterpret_cast(number)->body : default_id); if (obj->body.otype == _uris.patch_Get) { if (subject_uri) { _iface(Get{seq, *subject_uri}); } } else if (obj->body.otype == _uris.ingen_BundleStart) { _iface(BundleBegin{seq}); } else if (obj->body.otype == _uris.ingen_BundleEnd) { _iface(BundleEnd{seq}); } else if (obj->body.otype == _uris.ingen_Undo) { _iface(Undo{seq}); } else if (obj->body.otype == _uris.ingen_Redo) { _iface(Redo{seq}); } else if (obj->body.otype == _uris.patch_Delete) { const LV2_Atom_Object* body = nullptr; lv2_atom_object_get(obj, _uris.patch_body.urid(), &body, 0); if (subject_uri && !body) { _iface(Del{seq, *subject_uri}); return true; } if (body && body->body.otype == _uris.ingen_Arc) { const LV2_Atom* tail = nullptr; const LV2_Atom* head = nullptr; const LV2_Atom* incidentTo = nullptr; lv2_atom_object_get(body, _uris.ingen_tail.urid(), &tail, _uris.ingen_head.urid(), &head, _uris.ingen_incidentTo.urid(), &incidentTo, nullptr); boost::optional subject_path(atom_to_path(subject)); boost::optional tail_path(atom_to_path(tail)); boost::optional head_path(atom_to_path(head)); boost::optional other_path(atom_to_path(incidentTo)); if (tail_path && head_path) { _iface(Disconnect{seq, *tail_path, *head_path}); } else if (subject_path && other_path) { _iface(DisconnectAll{seq, *subject_path, *other_path}); } else { _log.warn("Delete of unknown object\n"); return false; } } } else if (obj->body.otype == _uris.patch_Put) { const LV2_Atom_Object* body = nullptr; const LV2_Atom* context = nullptr; lv2_atom_object_get(obj, _uris.patch_body.urid(), &body, _uris.patch_context.urid(), &context, 0); if (!body) { _log.warn("Put message has no body\n"); return false; } if (!subject_uri) { _log.warn("Put message has no subject\n"); return false; } if (body->body.otype == _uris.ingen_Arc) { LV2_Atom* tail = nullptr; LV2_Atom* head = nullptr; lv2_atom_object_get(body, _uris.ingen_tail.urid(), &tail, _uris.ingen_head.urid(), &head, nullptr); if (!tail || !head) { _log.warn("Arc has no tail or head\n"); return false; } boost::optional tail_path(atom_to_path(tail)); boost::optional head_path(atom_to_path(head)); if (tail_path && head_path) { _iface(Connect{seq, *tail_path, *head_path}); } else { _log.warn("Arc has non-path tail or head\n"); } } else { ingen::Properties props; get_props(body, props); _iface(Put{seq, *subject_uri, props, atom_to_context(context)}); } } else if (obj->body.otype == _uris.patch_Set) { if (!subject_uri) { _log.warn("Set message has no subject\n"); return false; } const LV2_Atom_URID* prop = nullptr; const LV2_Atom* value = nullptr; const LV2_Atom* context = nullptr; lv2_atom_object_get(obj, _uris.patch_property.urid(), &prop, _uris.patch_value.urid(), &value, _uris.patch_context.urid(), &context, 0); if (!prop || reinterpret_cast(prop)->type != _uris.atom_URID) { _log.warn("Set message missing property\n"); return false; } if (!value) { _log.warn("Set message missing value\n"); return false; } Atom atom; get_atom(value, atom); _iface(SetProperty{seq, *subject_uri, URI(_map.unmap_uri(prop->body)), atom, atom_to_context(context)}); } else if (obj->body.otype == _uris.patch_Patch) { if (!subject_uri) { _log.warn("Patch message has no subject\n"); return false; } const LV2_Atom_Object* remove = nullptr; const LV2_Atom_Object* add = nullptr; const LV2_Atom* context = nullptr; lv2_atom_object_get(obj, _uris.patch_remove.urid(), &remove, _uris.patch_add.urid(), &add, _uris.patch_context.urid(), &context, 0); if (!remove) { _log.warn("Patch message has no remove\n"); return false; } if (!add) { _log.warn("Patch message has no add\n"); return false; } ingen::Properties add_props; get_props(add, add_props); ingen::Properties remove_props; get_props(remove, remove_props); _iface(Delta{seq, *subject_uri, remove_props, add_props, atom_to_context(context)}); } else if (obj->body.otype == _uris.patch_Copy) { if (!subject) { _log.warn("Copy message has no subject\n"); return false; } if (!subject_uri) { _log.warn("Copy message has non-path subject\n"); return false; } const LV2_Atom* dest = nullptr; lv2_atom_object_get(obj, _uris.patch_destination.urid(), &dest, 0); if (!dest) { _log.warn("Copy message has no destination\n"); return false; } boost::optional dest_uri(atom_to_uri(dest)); if (!dest_uri) { _log.warn("Copy message has non-URI destination\n"); return false; } _iface(Copy{seq, *subject_uri, *dest_uri}); } else if (obj->body.otype == _uris.patch_Move) { if (!subject) { _log.warn("Move message has no subject\n"); return false; } const LV2_Atom* dest = nullptr; lv2_atom_object_get(obj, _uris.patch_destination.urid(), &dest, 0); if (!dest) { _log.warn("Move message has no destination\n"); return false; } boost::optional subject_path(atom_to_path(subject)); if (!subject_path) { _log.warn("Move message has non-path subject\n"); return false; } boost::optional dest_path(atom_to_path(dest)); if (!dest_path) { _log.warn("Move message has non-path destination\n"); return false; } _iface(Move{seq, *subject_path, *dest_path}); } else if (obj->body.otype == _uris.patch_Response) { const LV2_Atom* body = nullptr; lv2_atom_object_get(obj, _uris.patch_body.urid(), &body, 0); if (!number || number->type != _uris.atom_Int) { _log.warn("Response message has no sequence number\n"); return false; } if (!body || body->type != _uris.atom_Int) { _log.warn("Response message body is not integer\n"); return false; } _iface(Response{reinterpret_cast(number)->body, static_cast( reinterpret_cast(body)->body), subject_uri ? subject_uri->c_str() : ""}); } else { _log.warn("Unknown object type <%1%>\n", _map.unmap_uri(obj->body.otype)); } return true; } } // namespace ingen