/* 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 "events/CreatePort.hpp" #include "ingen/Configuration.hpp" #include "ingen/Store.hpp" #include "ingen/URIs.hpp" #include "ingen/World.hpp" #include "ingen/Log.hpp" #include "raul/Maid.hpp" #include "raul/SharedPtr.hpp" #include "BlockFactory.hpp" #include "Broadcaster.hpp" #include "BufferFactory.hpp" #include "ControlBindings.hpp" #include "DirectDriver.hpp" #include "Driver.hpp" #include "Engine.hpp" #include "Event.hpp" #include "EventWriter.hpp" #include "GraphImpl.hpp" #include "LV2Options.hpp" #include "PostProcessor.hpp" #include "PreProcessor.hpp" #include "ProcessContext.hpp" #include "ThreadManager.hpp" #include "Worker.hpp" using namespace std; namespace Ingen { namespace Server { Raul::ThreadVar ThreadManager::flags(0); bool ThreadManager::single_threaded(true); Engine::Engine(Ingen::World* world) : _world(world) , _block_factory(new BlockFactory(world)) , _broadcaster(new Broadcaster()) , _buffer_factory(new BufferFactory(*this, world->uris())) , _control_bindings(NULL) , _event_writer(new EventWriter(*this)) , _maid(new Raul::Maid()) , _options(new LV2Options(*this)) , _pre_processor(new PreProcessor()) , _post_processor(new PostProcessor(*this)) , _root_graph(NULL) , _worker(new Worker(world->log(), event_queue_size())) , _process_context(*this) , _quit_flag(false) , _direct_driver(true) { if (!world->store()) { world->set_store(SharedPtr(new Store())); } _control_bindings = new ControlBindings(*this); _world->lv2_features().add_feature(_worker->schedule_feature()); _world->lv2_features().add_feature(_options); } Engine::~Engine() { deactivate(); const SharedPtr s = this->store(); if (s) { for (Store::iterator i = s->begin(); i != s->end(); ++i) { if (!PtrCast(i->second)->parent()) { i->second.reset(); } } } _world->set_store(SharedPtr()); delete _maid; delete _pre_processor; delete _post_processor; delete _block_factory; delete _control_bindings; delete _broadcaster; delete _event_writer; munlockall(); } SharedPtr Engine::store() const { return _world->store(); } size_t Engine::event_queue_size() const { return world()->conf().option("queue-size").get_int(); } void Engine::quit() { _quit_flag = true; } bool Engine::main_iteration() { _post_processor->process(); _maid->cleanup(); return !_quit_flag; } void Engine::set_driver(SharedPtr driver) { _driver = driver; } SampleCount Engine::event_time() { if (ThreadManager::single_threaded) { return 0; } const SampleCount start = _direct_driver ? _process_context.start() : _driver->frame_time(); /* Exactly one cycle latency (some could run ASAP if we get lucky, but not always, and a slight constant latency is far better than jittery lower (average) latency */ return start + _driver->block_length(); } static void execute_and_delete_event(ProcessContext& context, Event* ev) { ev->pre_process(); if (ev->time() < context.start()) { // Didn't get around to executing in time, oh well... ev->set_time(context.start()); } ev->execute(context); ev->post_process(); delete ev; } void Engine::init(double sample_rate, uint32_t block_length) { set_driver(SharedPtr(new DirectDriver(sample_rate, block_length))); _direct_driver = true; } bool Engine::activate() { if (!_driver) { return false; } ThreadManager::single_threaded = true; _buffer_factory->set_block_length(_driver->block_length()); _options->set(*this); _pre_processor->start(); const Ingen::URIs& uris = world()->uris(); Forge& forge = world()->forge(); // Create root graph if (!_root_graph) { _root_graph = new GraphImpl( *this, Raul::Symbol("root"), 1, NULL, _driver->sample_rate(), 1); _root_graph->set_property( uris.rdf_type, Resource::Property(uris.ingen_Graph, Resource::INTERNAL)); _root_graph->set_property( uris.ingen_polyphony, Resource::Property(_world->forge().make(int32_t(1)), Resource::INTERNAL)); _root_graph->activate(*_buffer_factory); _world->store()->add(_root_graph); _root_graph->compiled_graph(_root_graph->compile()); ProcessContext context(*this); Resource::Properties control_properties; control_properties.insert(make_pair(uris.lv2_name, forge.alloc("Control"))); control_properties.insert(make_pair(uris.rdf_type, uris.atom_AtomPort)); control_properties.insert(make_pair(uris.atom_bufferType, uris.atom_Sequence)); // Add control input Resource::Properties in_properties(control_properties); in_properties.insert(make_pair(uris.rdf_type, uris.lv2_InputPort)); in_properties.insert(make_pair(uris.lv2_index, forge.make(0))); in_properties.insert(make_pair(uris.lv2_portProperty, uris.lv2_connectionOptional)); in_properties.insert( make_pair(uris.ingen_canvasX, Resource::Property(forge.make(32.0f), Resource::EXTERNAL))); in_properties.insert( make_pair(uris.ingen_canvasY, Resource::Property(forge.make(32.0f), Resource::EXTERNAL))); SharedPtr respondee; execute_and_delete_event( context, new Events::CreatePort( *this, respondee, -1, 0, Raul::Path("/control_in"), false, in_properties)); // Add control out Resource::Properties out_properties(control_properties); out_properties.insert(make_pair(uris.rdf_type, uris.lv2_OutputPort)); out_properties.insert(make_pair(uris.lv2_index, forge.make(1))); in_properties.insert(make_pair(uris.lv2_portProperty, uris.lv2_connectionOptional)); out_properties.insert( make_pair(uris.ingen_canvasX, Resource::Property(forge.make(128.0f), Resource::EXTERNAL))); out_properties.insert( make_pair(uris.ingen_canvasY, Resource::Property(forge.make(32.0f), Resource::EXTERNAL))); execute_and_delete_event( context, new Events::CreatePort( *this, respondee, -1, 0, Raul::Path("/control_out"), true, out_properties)); } _driver->activate(); _root_graph->enable(); ThreadManager::single_threaded = false; return true; } void Engine::deactivate() { _pre_processor->join(); if (_driver) { _driver->deactivate(); } if (_root_graph) { _root_graph->deactivate(); } ThreadManager::single_threaded = true; } unsigned Engine::run(uint32_t sample_count) { _process_context.locate(_process_context.end(), sample_count); // Apply control bindings to input control_bindings()->pre_process( _process_context, _root_graph->port_impl(0)->buffer(0).get()); post_processor()->set_end_time(_process_context.end()); // Process events that came in during the last cycle // (Aiming for jitter-free 1 block event latency, ideally) const unsigned n_processed_events = process_events(); // Run root graph if (_root_graph) { _root_graph->process(_process_context); } // Emit control binding feedback control_bindings()->post_process( _process_context, _root_graph->port_impl(1)->buffer(0).get()); return n_processed_events; } bool Engine::pending_events() { return !_pre_processor->empty(); } void Engine::enqueue_event(Event* ev) { _pre_processor->event(ev); } unsigned Engine::process_events() { return _pre_processor->process(_process_context, *_post_processor); } void Engine::register_client(const Raul::URI& uri, SharedPtr client) { log().info(Raul::fmt("Registering client <%1%>\n") % uri.c_str()); _broadcaster->register_client(uri, client); } bool Engine::unregister_client(const Raul::URI& uri) { log().info(Raul::fmt("Unregistering client <%1%>\n") % uri.c_str()); return _broadcaster->unregister_client(uri); } } // namespace Server } // namespace Ingen