/* This file is part of Ingen. * Copyright 2007-2011 David Robillard * * Ingen 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. * * 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 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 "ingen/shared/LV2URIMap.hpp" #include "ingen/shared/URIs.hpp" #include "raul/log.hpp" #include "AudioBuffer.hpp" #include "BufferFactory.hpp" #include "Driver.hpp" #include "Engine.hpp" #include "ThreadManager.hpp" using namespace Raul; namespace Ingen { namespace Server { static const size_t EVENT_BYTES_PER_FRAME = 4; // FIXME BufferFactory::BufferFactory(Engine& engine, SharedPtr uris) : _engine(engine) , _uris(uris) , _silent_buffer(NULL) { assert(_uris); } BufferFactory::~BufferFactory() { _silent_buffer.reset(); free_list(_free_audio.get()); free_list(_free_control.get()); free_list(_free_object.get()); } Ingen::Forge& BufferFactory::forge() { return _engine.world()->forge(); } void BufferFactory::free_list(Buffer* head) { while (head) { Buffer* next = head->_next; delete head; head = next; } } void BufferFactory::set_block_length(SampleCount block_length) { _silent_buffer = create(_uris->atom_Sound, audio_buffer_size(block_length)); } uint32_t BufferFactory::audio_buffer_size(SampleCount nframes) { return sizeof(LV2_Atom_Vector) + (nframes * sizeof(float)); } uint32_t BufferFactory::default_buffer_size(LV2_URID type) { if (type == _uris->atom_Float) { return sizeof(LV2_Atom_Float); } else if (type == _uris->atom_Sound) { return audio_buffer_size(_engine.driver()->block_length()); } else if (type == _uris->atom_Sequence) { return _engine.driver()->block_length() * EVENT_BYTES_PER_FRAME; } else { return 0; } } BufferFactory::Ref BufferFactory::get(LV2_URID type, uint32_t capacity, bool force_create) { Raul::AtomicPtr& head_ptr = free_list(type); Buffer* try_head = NULL; if (!force_create) { Buffer* next; do { try_head = head_ptr.get(); if (!try_head) break; next = try_head->_next; } while (!head_ptr.compare_and_exchange(try_head, next)); } if (!try_head) { if (!ThreadManager::thread_is(THREAD_PROCESS)) { return create(type, capacity); } else { assert(false); error << "Failed to obtain buffer" << endl; return Ref(); } } try_head->_next = NULL; return Ref(try_head); } BufferFactory::Ref BufferFactory::silent_buffer() { return _silent_buffer; } BufferFactory::Ref BufferFactory::create(LV2_URID type, uint32_t capacity) { ThreadManager::assert_not_thread(THREAD_PROCESS); Buffer* buffer = NULL; if (capacity == 0) { capacity = default_buffer_size(type); } if (type == _uris->atom_Float) { assert(capacity >= sizeof(LV2_Atom_Float)); buffer = new AudioBuffer(*this, type, capacity); } else if (type == _uris->atom_Sound) { assert(capacity >= default_buffer_size(_uris->atom_Sound)); buffer = new AudioBuffer(*this, type, capacity); } else { buffer = new Buffer(*this, type, capacity); } buffer->atom()->type = type; assert(buffer); return Ref(buffer); } void BufferFactory::recycle(Buffer* buf) { Raul::AtomicPtr& head_ptr = free_list(buf->type()); Buffer* try_head; do { try_head = head_ptr.get(); buf->_next = try_head; } while (!head_ptr.compare_and_exchange(try_head, buf)); } } // namespace Server } // namespace Ingen