From 344cdcbd4f2bc7a9203b4e98da2ac349581e521a Mon Sep 17 00:00:00 2001 From: David Robillard Date: Mon, 1 Oct 2007 03:23:30 +0000 Subject: Generic audio thread generated Event sending system. git-svn-id: http://svn.drobilla.net/lad/ingen@793 a436a847-0d15-0410-975c-d299462d15a1 --- src/libs/engine/EventSink.cpp | 38 ++++++++++++++++++++++++++++++++++---- src/libs/engine/EventSink.hpp | 4 ++-- src/libs/engine/InputPort.cpp | 4 ++-- src/libs/engine/OutputPort.cpp | 5 +++-- src/libs/engine/PostProcessor.cpp | 18 ++++++++++++------ src/libs/engine/PostProcessor.hpp | 3 +++ src/libs/engine/ProcessContext.hpp | 7 ++++++- 7 files changed, 62 insertions(+), 17 deletions(-) (limited to 'src/libs') diff --git a/src/libs/engine/EventSink.cpp b/src/libs/engine/EventSink.cpp index 441e7aac..7fc7b157 100644 --- a/src/libs/engine/EventSink.cpp +++ b/src/libs/engine/EventSink.cpp @@ -24,7 +24,7 @@ using namespace std; namespace Ingen { - +#if 0 void EventSink::control_change(Port* port, FrameTime time, float val) { @@ -32,12 +32,42 @@ EventSink::control_change(Port* port, FrameTime time, float val) SendPortValueEvent ev(_engine, time, port, false, 0, val); _events.write(sizeof(ev), (uchar*)&ev); } - +#endif + +/** \a size is not size_t because an event will never be even remotely close + * to UINT32_MAX in size, so uint32_t saves wasted space on 64-bit. + */ +bool +EventSink::write(uint32_t size, const Event* ev) +{ + if (size > _events.write_space()) + return false; + _events.write(sizeof(uint32_t), (uint8_t*)&size); + _events.write(size, (uint8_t*)ev); + + return true; +} + + +/** Read the next event into event_buffer. + * + * \a event_buffer can be casted to Event* and virtual methods called. + */ bool -EventSink::read_control_change(SendPortValueEvent& ev) +EventSink::read(uint32_t event_buffer_size, uint8_t* event_buffer) { - return _events.full_read(sizeof(SendPortValueEvent), (uchar*)&ev); + uint32_t read_size; + bool success = _events.full_read(sizeof(uint32_t), (uint8_t*)&read_size); + if (!success) + return false; + + assert(read_size <= event_buffer_size); + + if (read_size > 0) + return _events.full_read(read_size, event_buffer); + else + return false; } diff --git a/src/libs/engine/EventSink.hpp b/src/libs/engine/EventSink.hpp index b7753a79..4816a069 100644 --- a/src/libs/engine/EventSink.hpp +++ b/src/libs/engine/EventSink.hpp @@ -47,9 +47,9 @@ public: /* FIXME: Figure out variable sized event queues and make this a generic * interface (ie don't add a method for every event type, crap..) */ - void control_change(Port* port, FrameTime time, float val); + bool write(uint32_t size, const Event* ev); - bool read_control_change(SendPortValueEvent& ev); + bool read(uint32_t event_buffer_size, uint8_t* event_buffer); private: Engine& _engine; diff --git a/src/libs/engine/InputPort.cpp b/src/libs/engine/InputPort.cpp index 6cc7beb4..16a1c86a 100644 --- a/src/libs/engine/InputPort.cpp +++ b/src/libs/engine/InputPort.cpp @@ -223,8 +223,8 @@ InputPort::post_process(ProcessContext& context) if (_monitor && _type == DataType::FLOAT && _buffer_size == 1) { const Sample value = ((AudioBuffer*)(*_buffers)[0])->value_at(0); if (value != _last_reported_value) { - context.event_sink().control_change(this, context.start(), - ((AudioBuffer*)(*_buffers)[0])->value_at(0)); + const SendPortValueEvent ev(context.engine(), context.start(), this, false, 0, value); + context.event_sink().write(sizeof(ev), &ev); _last_reported_value = value; } } diff --git a/src/libs/engine/OutputPort.cpp b/src/libs/engine/OutputPort.cpp index 260f3038..6d49a1d1 100644 --- a/src/libs/engine/OutputPort.cpp +++ b/src/libs/engine/OutputPort.cpp @@ -34,8 +34,9 @@ void OutputPort::post_process(ProcessContext& context) { if (_monitor && _type == DataType::FLOAT && _buffer_size == 1) { - context.event_sink().control_change(this, context.start(), - ((AudioBuffer*)(*_buffers)[0])->value_at(0)); + const Sample value = ((AudioBuffer*)(*_buffers)[0])->value_at(0); + const SendPortValueEvent ev(context.engine(), context.start(), this, false, 0, value); + context.event_sink().write(sizeof(ev), &ev); } for (uint32_t i=0; i < _poly; ++i) diff --git a/src/libs/engine/PostProcessor.cpp b/src/libs/engine/PostProcessor.cpp index d1b09271..96aee3b1 100644 --- a/src/libs/engine/PostProcessor.cpp +++ b/src/libs/engine/PostProcessor.cpp @@ -36,6 +36,8 @@ PostProcessor::PostProcessor(Engine& engine, size_t queue_size) : _engine(engine) , _max_time(0) , _events(queue_size) + , _event_buffer_size(sizeof(SendPortValueEvent)) // FIXME: make generic + , _event_buffer((uint8_t*)malloc(_event_buffer_size)) { } @@ -43,15 +45,19 @@ PostProcessor::PostProcessor(Engine& engine, size_t queue_size) void PostProcessor::process() { + const FrameTime end_time = _max_time.get(); + /* Process any audio thread generated events */ - /* FIXME: process events from all thread if parallel */ - /* TODO: obviously this needs work to be generic */ - SendPortValueEvent ev(_engine, 0, NULL, false, 0, 0.0f); - while (_engine.audio_driver()->context().event_sink().read_control_change(ev)) { - ev.post_process(); + /* FIXME: process events from all threads if parallel */ + + while (_engine.audio_driver()->context().event_sink().read( + _event_buffer_size, _event_buffer)) { + if (((Event*)_event_buffer)->time() > end_time) + break; + ((Event*)_event_buffer)->post_process(); } - FrameTime end_time = _max_time.get(); + /* Process normal events */ while ( ! _events.empty()) { diff --git a/src/libs/engine/PostProcessor.hpp b/src/libs/engine/PostProcessor.hpp index cc19cdbe..3d51136d 100644 --- a/src/libs/engine/PostProcessor.hpp +++ b/src/libs/engine/PostProcessor.hpp @@ -61,6 +61,9 @@ private: Raul::AtomicInt _max_time; //Raul::Maid& _maid; Raul::SRSWQueue _events; + uint32_t _event_buffer_size; + uint8_t* _event_buffer; + //virtual void _whipped(); }; diff --git a/src/libs/engine/ProcessContext.hpp b/src/libs/engine/ProcessContext.hpp index c85c9ff6..0f090b3b 100644 --- a/src/libs/engine/ProcessContext.hpp +++ b/src/libs/engine/ProcessContext.hpp @@ -38,7 +38,10 @@ namespace Ingen { class ProcessContext { public: - ProcessContext(Engine& engine) : _event_sink(engine, 1024) {} // FIXME: size? + ProcessContext(Engine& engine) + : _engine(engine) + , _event_sink(engine, 1024) // FIXME: size? + {} void set_time_slice(SampleCount nframes, FrameTime start, FrameTime end) { _nframes = nframes; @@ -46,6 +49,7 @@ public: _end = end; } + inline Engine& engine() const { return _engine; } inline SampleCount nframes() const { return _nframes; } inline FrameTime start() const { return _start; } inline FrameTime end() const { return _end; } @@ -53,6 +57,7 @@ public: inline EventSink& event_sink() { return _event_sink; } private: + Engine& _engine; ///< Engine we're running in SampleCount _nframes; ///< Number of actual time (Jack) frames this cycle FrameTime _start; ///< Start frame of this cycle, timeline relative FrameTime _end; ///< End frame of this cycle, timeline relative -- cgit v1.2.1