/*
This file is part of Ingen.
Copyright 2007-2012 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
#include "ingen/AtomReader.hpp"
#include "ingen/Log.hpp"
#include "ingen/Node.hpp"
#include "ingen/URIMap.hpp"
#include "lv2/lv2plug.in/ns/ext/atom/util.h"
#include "raul/Path.hpp"
namespace Ingen {
AtomReader::AtomReader(URIMap& map,
URIs& uris,
Log& log,
Forge& forge,
Interface& iface)
: _map(map)
, _uris(uris)
, _log(log)
, _forge(forge)
, _iface(iface)
{
}
void
AtomReader::get_atom(const LV2_Atom* in, Atom& out)
{
if (in) {
if (in->type == _uris.atom_URID) {
const LV2_Atom_URID* urid = (const LV2_Atom_URID*)in;
const char* uri = _map.unmap_uri(urid->body);
if (uri) {
out = _forge.alloc_uri(_map.unmap_uri(urid->body));
} else {
_log.error(fmt("Unable to unmap URID %1%\n") % urid->body);
}
} else {
out = _forge.alloc(in->size, in->type, LV2_ATOM_BODY_CONST(in));
}
}
}
void
AtomReader::get_props(const LV2_Atom_Object* obj,
Ingen::Resource::Properties& props)
{
if (obj->body.otype) {
props.insert(
std::make_pair(_uris.rdf_type,
_forge.alloc_uri(_map.unmap_uri(obj->body.otype))));
}
LV2_ATOM_OBJECT_FOREACH(obj, p) {
Atom val;
get_atom(&p->value, val);
props.insert(std::make_pair(Raul::URI(_map.unmap_uri(p->key)), val));
}
}
const char*
AtomReader::atom_to_uri(const LV2_Atom* atom)
{
if (atom && atom->type == _uris.atom_URI) {
return (const char*)LV2_ATOM_BODY_CONST(atom);
} else if (atom && atom->type == _uris.atom_URID) {
return _map.unmap_uri(((const LV2_Atom_URID*)atom)->body);
} else {
return NULL;
}
}
boost::optional
AtomReader::atom_to_path(const LV2_Atom* atom)
{
const char* uri_str = atom_to_uri(atom);
if (uri_str && Raul::URI::is_valid(uri_str) &&
Node::uri_is_path(Raul::URI(uri_str))) {
return Node::uri_to_path(Raul::URI(uri_str));
}
return boost::optional();
}
bool
AtomReader::is_message(URIs& uris, const LV2_Atom* msg)
{
if (msg->type != uris.atom_Object) {
return false;
}
const LV2_Atom_Object* obj = (const LV2_Atom_Object*)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)
{
if (msg->type != _uris.atom_Object) {
_log.warn(fmt("Unknown message type <%1%>\n")
% _map.unmap_uri(msg->type));
return false;
}
const LV2_Atom_Object* obj = (const LV2_Atom_Object*)msg;
const LV2_Atom* subject = NULL;
const LV2_Atom* number = NULL;
lv2_atom_object_get(obj,
(LV2_URID)_uris.patch_subject, &subject,
(LV2_URID)_uris.patch_sequenceNumber, &number,
NULL);
const char* subject_uri = atom_to_uri(subject);
const int32_t seq_id = ((number && number->type == _uris.atom_Int)
? ((const LV2_Atom_Int*)number)->body
: 0);
if (seq_id) {
_iface.set_response_id(seq_id);
} else {
_iface.set_response_id(-1);
}
if (obj->body.otype == _uris.patch_Get) {
_iface.get(Raul::URI(subject_uri));
} else if (obj->body.otype == _uris.patch_Delete) {
const LV2_Atom_Object* body = NULL;
lv2_atom_object_get(obj, (LV2_URID)_uris.patch_body, &body, 0);
if (subject_uri && !body) {
_iface.del(Raul::URI(subject_uri));
return true;
} else if (body && body->body.otype == _uris.ingen_Arc) {
const LV2_Atom* tail = NULL;
const LV2_Atom* head = NULL;
const LV2_Atom* incidentTo = NULL;
lv2_atom_object_get(body,
(LV2_URID)_uris.ingen_tail, &tail,
(LV2_URID)_uris.ingen_head, &head,
(LV2_URID)_uris.ingen_incidentTo, &incidentTo,
NULL);
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(*tail_path, *head_path);
} else if (subject_path && other_path) {
_iface.disconnect_all(*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 = NULL;
lv2_atom_object_get(obj, (LV2_URID)_uris.patch_body, &body, 0);
if (!body) {
_log.warn("Put message has no body\n");
return false;
} else if (!subject_uri) {
_log.warn("Put message has no subject\n");
return false;
}
if (body->body.otype == _uris.ingen_Arc) {
LV2_Atom* tail = NULL;
LV2_Atom* head = NULL;
lv2_atom_object_get(body,
(LV2_URID)_uris.ingen_tail, &tail,
(LV2_URID)_uris.ingen_head, &head,
NULL);
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(*tail_path, *head_path);
} else {
_log.warn(fmt("Arc %1% => %2% has non-path tail or head\n")
% atom_to_uri(tail) % atom_to_uri(head));
}
} else {
Ingen::Resource::Properties props;
get_props(body, props);
_iface.put(Raul::URI(subject_uri), props);
}
} 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 = NULL;
lv2_atom_object_get(obj, (LV2_URID)_uris.patch_property, &prop, 0);
if (!prop || ((const LV2_Atom*)prop)->type != _uris.atom_URID) {
_log.warn("Set message missing property\n");
return false;
}
const LV2_Atom* value = NULL;
lv2_atom_object_get(obj, (LV2_URID)_uris.patch_value, &value, 0);
if (!value) {
_log.warn("Set message missing value\n");
return false;
}
Atom atom;
get_atom(value, atom);
_iface.set_property(Raul::URI(subject_uri),
Raul::URI(_map.unmap_uri(prop->body)),
atom);
} else if (obj->body.otype == _uris.patch_Patch) {
if (!subject) {
_log.warn("Patch message has no subject\n");
return false;
}
const LV2_Atom_Object* remove = NULL;
lv2_atom_object_get(obj, (LV2_URID)_uris.patch_remove, &remove, 0);
if (!remove) {
_log.warn("Patch message has no remove\n");
return false;
}
const LV2_Atom_Object* add = NULL;
lv2_atom_object_get(obj, (LV2_URID)_uris.patch_add, &add, 0);
if (!add) {
_log.warn("Patch message has no add\n");
return false;
}
Ingen::Resource::Properties add_props;
get_props(add, add_props);
Ingen::Resource::Properties remove_props;
get_props(remove, remove_props);
_iface.delta(Raul::URI(subject_uri), remove_props, add_props);
} else if (obj->body.otype == _uris.patch_Move) {
if (!subject) {
_log.warn("Move message has no subject\n");
return false;
}
const LV2_Atom* dest = NULL;
lv2_atom_object_get(obj, (LV2_URID)_uris.patch_destination, &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(*subject_path, *dest_path);
} else if (obj->body.otype == _uris.patch_Response) {
const LV2_Atom* request = NULL;
const LV2_Atom* body = NULL;
lv2_atom_object_get(obj,
(LV2_URID)_uris.patch_request, &request,
(LV2_URID)_uris.patch_body, &body,
0);
if (!request || request->type != _uris.atom_Int) {
_log.warn("Response message has no request\n");
return false;
} else if (!body || body->type != _uris.atom_Int) {
_log.warn("Response message body is not integer\n");
return false;
}
_iface.response(((const LV2_Atom_Int*)request)->body,
(Ingen::Status)((const LV2_Atom_Int*)body)->body,
subject_uri ? subject_uri : "");
} else {
_log.warn(fmt("Unknown object type <%1%>\n")
% _map.unmap_uri(obj->body.otype));
}
return true;
}
} // namespace Ingen