/*
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 "Broadcaster.hpp"
#include "BufferFactory.hpp"
#include "DirectDriver.hpp"
#include "ControlBindings.hpp"
#include "Driver.hpp"
#include "Engine.hpp"
#include "Event.hpp"
#include "EventWriter.hpp"
#include "NodeFactory.hpp"
#include "PatchImpl.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)
, _broadcaster(new Broadcaster())
, _buffer_factory(new BufferFactory(*this, world->uris()))
, _control_bindings(NULL)
, _maid(new Raul::Maid())
, _node_factory(new NodeFactory(world))
, _pre_processor(new PreProcessor())
, _post_processor(new PostProcessor(*this))
, _event_writer(new EventWriter(*this))
, _root_patch(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());
}
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 _node_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());
_pre_processor->start();
const Ingen::URIs& uris = world()->uris();
Forge& forge = world()->forge();
// Create root patch
if (!_root_patch) {
_root_patch = new PatchImpl(
*this, Raul::Symbol("root"), 1, NULL, _driver->sample_rate(), 1);
_root_patch->set_property(
uris.rdf_type,
Resource::Property(uris.ingen_Patch, Resource::INTERNAL));
_root_patch->set_property(
uris.ingen_polyphony,
Resource::Property(_world->forge().make(int32_t(1)),
Resource::INTERNAL));
_root_patch->activate(*_buffer_factory);
_world->store()->add(_root_patch);
_root_patch->compiled_patch(_root_patch->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_patch->enable();
ThreadManager::single_threaded = false;
return true;
}
void
Engine::deactivate()
{
_pre_processor->join();
if (_driver) {
_driver->deactivate();
}
if (_root_patch) {
_root_patch->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_patch->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 patch
if (_root_patch) {
_root_patch->process(_process_context);
}
// Emit control binding feedback
control_bindings()->post_process(
_process_context, _root_patch->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