/* This file is part of Machina. * Copyright (C) 2007-2009 David Robillard * * Machina 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. * * Machina 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 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., * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include #include #include #include #include "redlandmm/Query.hpp" #include "machina/Loader.hpp" #include "machina/Node.hpp" #include "machina/Edge.hpp" #include "machina/Machine.hpp" #include "machina/ActionFactory.hpp" #include "machina-config.h" using namespace Raul; using namespace std; namespace Machina { Loader::Loader(Redland::World& rdf_world) : _rdf_world(rdf_world) { _rdf_world.add_prefix("xsd", "http://www.w3.org/2001/XMLSchema#"); _rdf_world.add_prefix("", "http://drobilla.net/ns/machina#"); } /** Load (create) all objects from RDF into the engine. * * @param uri URI of machine (resolvable URI to an RDF document). * @return Loaded Machine. */ SharedPtr Loader::load(const Glib::ustring& uri) { using Redland::Query; using Redland::QueryResults; using Glib::ustring; SharedPtr machine; ustring document_uri = uri; // If "URI" doesn't contain a colon, try to resolve as a filename if (uri.find(":") == ustring::npos) { raptor_uri* base_uri = raptor_new_uri((const unsigned char*)"file:."); raptor_uri* document_raptor_uri = raptor_new_uri_relative_to_base( base_uri, (const unsigned char*)uri.c_str()); if (document_raptor_uri) { document_uri = (char*)raptor_uri_as_string(document_raptor_uri); raptor_free_uri(document_raptor_uri); raptor_free_uri(base_uri); } else { raptor_free_uri(base_uri); return machine; // NULL } } const string machine_uri = string("<>"); cout << "[Loader] Loading " << document_uri << endl; machine = SharedPtr(new Machine(TimeUnit::beats(MACHINA_PPQN))); typedef std::map > Created; Created created; Redland::Model model(_rdf_world, document_uri); /* Get initial nodes */ Query query = Query(_rdf_world, ustring( "SELECT DISTINCT ?node ?duration WHERE {\n") + machine_uri + " :initialNode ?node .\n" "?node :duration ?duration .\n" "}\n"); SharedPtr results = query.run(_rdf_world, model); for (; !results->finished(); results->next()) { const char* node_id = results->get("node"); SharedPtr node( new Node( TimeStamp(TimeUnit(TimeUnit::BEATS, MACHINA_PPQN), results->get("duration").to_float()), true)); machine->add_node(node); created[node_id] = node; } /* Get remaining (non-initial) nodes */ query = Query(_rdf_world, ustring( "SELECT DISTINCT ?node ?duration WHERE {\n") + machine_uri + " :node ?node .\n" "?node :duration ?duration .\n" "}\n"); results = query.run(_rdf_world, model); for (; !results->finished(); results->next()) { const char* node_id = results->get("node"); if (created.find(node_id) == created.end()) { SharedPtr node(new Node( TimeStamp(TimeUnit(TimeUnit::BEATS, MACHINA_PPQN), results->get("duration").to_float()), false)); machine->add_node(node); created[node_id] = node; } } /* Find out which nodes are selectors */ query = Query(_rdf_world, ustring( "SELECT DISTINCT ?node WHERE {\n") + machine_uri + " :node ?node .\n" "?node a :SelectorNode .\n" "}\n"); results = query.run(_rdf_world, model); for (; !results->finished(); results->next()) { const char* node_id = results->get("node"); Created::iterator n = created.find(node_id); if (n != created.end()) n->second->set_selector(true); else cerr << "WARNING: Selector node " << node_id << " not found" << endl; } /* Get note actions */ query = Query(_rdf_world, ustring( "SELECT DISTINCT ?node ?note WHERE {\n" " ?node :enterAction [ a :MidiAction; :midiNote ?note ] ;\n" " :exitAction [ a :MidiAction; :midiNote ?note ] .\n" "}\n")); results = query.run(_rdf_world, model); for (; !results->finished(); results->next()) { Created::iterator node_i = created.find((const char*)results->get("node")); if (node_i != created.end()) { SharedPtr node = node_i->second; const int note_num = results->get("note").to_int(); if (note_num >= 0 && note_num <= 127) { node->set_enter_action(ActionFactory::note_on((unsigned char)note_num)); node->set_exit_action(ActionFactory::note_off((unsigned char)note_num)); } else { cerr << "WARNING: MIDI note number out of range, ignoring." << endl; } } else { cerr << "WARNING: Found note for unknown states. Ignoring." << endl; } } /* Get edges */ query = Query(_rdf_world, ustring( "SELECT DISTINCT ?edge ?src ?dst ?prob WHERE {\n" + machine_uri + " :edge ?edge .\n" "?edge :tail ?src ;\n" " :head ?dst ;\n" " :probability ?prob .\n }")); results = query.run(_rdf_world, model); for (; !results->finished(); results->next()) { const char* src_uri = results->get("src"); const char* dst_uri = results->get("dst"); float prob = results->get("prob").to_float(); Created::iterator src_i = created.find(src_uri); Created::iterator dst_i = created.find(dst_uri); if (src_i != created.end() && dst_i != created.end()) { const SharedPtr src = src_i->second; const SharedPtr dst = dst_i->second; SharedPtr edge(new Edge(src, dst)); edge->set_probability(prob); src->add_edge(edge); } else { cerr << "[Loader] WARNING: Ignored edge between unknown nodes " << src_uri << " -> " << dst_uri << endl; } } if (machine && machine->nodes().size() > 0) { machine->reset(machine->time()); return machine; } else { return SharedPtr(); } } } // namespace Machina