/* This file is part of Ingen. Copyright 2007-2016 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 "PreProcessor.hpp" #include "Engine.hpp" #include "Event.hpp" #include "PostProcessor.hpp" #include "PreProcessContext.hpp" #include "RunContext.hpp" #include "ThreadManager.hpp" #include "UndoStack.hpp" #include #include #include #include #include #include #include #include #include namespace ingen::server { PreProcessor::PreProcessor(Engine& engine) : _engine(engine) , _thread(&PreProcessor::run, this) {} PreProcessor::~PreProcessor() { if (_thread.joinable()) { _exit_flag = true; _sem.post(); _thread.join(); } } void PreProcessor::event(Event* const ev, Event::Mode mode) { // TODO: Probably possible to make this lock-free with CAS ThreadManager::assert_not_thread(THREAD_IS_REAL_TIME); const std::lock_guard lock{_mutex}; assert(!ev->is_prepared()); assert(!ev->next()); ev->set_mode(mode); /* Note that tail is only used here, not in process(). The head must be checked first here, since if it is null the tail pointer is junk. */ const Event* const head = _head.load(); if (!head) { _head = ev; _tail = ev; } else { _tail.load()->next(ev); _tail = ev; } _sem.post(); } unsigned PreProcessor::process(RunContext& ctx, PostProcessor& dest, size_t limit) { Event* const head = _head.load(); size_t n_processed = 0; Event* ev = head; Event* last = ev; while (ev && ev->is_prepared()) { switch (_block_state.load()) { case BlockState::UNBLOCKED: break; case BlockState::PRE_BLOCKED: if (ev->get_execution() == Event::Execution::BLOCK) { _block_state = BlockState::BLOCKED; } else if (ev->get_execution() == Event::Execution::ATOMIC) { _block_state = BlockState::PROCESSING; } break; case BlockState::BLOCKED: break; case BlockState::PRE_UNBLOCKED: assert(ev->get_execution() == Event::Execution::BLOCK); if (ev->get_execution() == Event::Execution::BLOCK) { _block_state = BlockState::PROCESSING; } break; case BlockState::PROCESSING: if (ev->get_execution() == Event::Execution::UNBLOCK) { _block_state = BlockState::UNBLOCKED; } } if (_block_state == BlockState::BLOCKED) { break; // Waiting for PRE_UNBLOCKED } if (ev->time() < ctx.start()) { ev->set_time(ctx.start()); // Too late, nudge to context start } else if (_block_state != BlockState::PROCESSING && ev->time() >= ctx.end()) { break; // Event is for a future cycle } // Execute event ev->execute(ctx); ++n_processed; // Unblock pre-processing if this is a non-bundled atomic event if (ev->get_execution() == Event::Execution::ATOMIC) { assert(_block_state.load() == BlockState::PROCESSING); _block_state = BlockState::UNBLOCKED; } // Move to next event last = ev; ev = ev->next(); if (_block_state != BlockState::PROCESSING && limit && n_processed >= limit) { break; } } if (n_processed > 0) { #ifndef NDEBUG const Engine& engine = ctx.engine(); if (engine.world().conf().option("trace").get()) { const uint64_t start = engine.cycle_start_time(ctx); const uint64_t end = engine.current_time(); fprintf(stderr, "Processed %zu events in %u us\n", n_processed, static_cast(end - start)); } #endif auto* next = last->next(); last->next(nullptr); dest.append(ctx, head, last); // Since _head was not null, we know it hasn't been changed since _head = next; /* If next is null, then _tail may now be invalid. However, it would cause a race to reset _tail here. Instead, append() checks only _head for emptiness, and resets the tail appropriately. */ } return n_processed; } void PreProcessor::run() { PreProcessContext ctx; UndoStack& undo_stack = *_engine.undo_stack(); UndoStack& redo_stack = *_engine.redo_stack(); AtomWriter undo_writer( _engine.world().uri_map(), _engine.world().uris(), undo_stack); AtomWriter redo_writer( _engine.world().uri_map(), _engine.world().uris(), redo_stack); ThreadManager::set_flag(THREAD_PRE_PROCESS); Event* back = nullptr; while (!_exit_flag) { if (!_sem.timed_wait(std::chrono::seconds(1))) { continue; } if (!back) { // Ran off end, find new unprepared back back = _head; while (back && back->is_prepared()) { back = back->next(); } } Event* const ev = back; if (!ev) { continue; } // Set block state before enqueueing event ev->mark(ctx); switch (ev->get_execution()) { case Event::Execution::NORMAL: break; case Event::Execution::ATOMIC: assert(_block_state == BlockState::UNBLOCKED); _block_state = BlockState::PRE_BLOCKED; break; case Event::Execution::BLOCK: assert(_block_state == BlockState::UNBLOCKED); _block_state = BlockState::PRE_BLOCKED; break; case Event::Execution::UNBLOCK: wait_for_block_state(BlockState::BLOCKED); _block_state = BlockState::PRE_UNBLOCKED; } // Prepare event, allowing it to be processed assert(!ev->is_prepared()); if (ev->pre_process(ctx)) { switch (ev->get_mode()) { case Event::Mode::NORMAL: case Event::Mode::REDO: undo_stack.start_entry(); ev->undo(undo_writer); undo_stack.finish_entry(); // undo_stack.save(stderr); break; case Event::Mode::UNDO: redo_stack.start_entry(); ev->undo(redo_writer); redo_stack.finish_entry(); // redo_stack.save(stderr, "redo"); break; } } assert(ev->is_prepared()); // Wait for process() if necessary if (ev->get_execution() == Event::Execution::ATOMIC) { wait_for_block_state(BlockState::UNBLOCKED); } back = ev->next(); } } } // namespace ingen::server