/*
  This file is part of Ingen.
  Copyright 2007-2015 David Robillard <http://drobilla.net/>

  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 <http://www.gnu.org/licenses/>.
*/

#ifndef INGEN_ENGINE_BUFFERFACTORY_HPP
#define INGEN_ENGINE_BUFFERFACTORY_HPP

#include <atomic>
#include <map>
#include <mutex>

#include "ingen/Atom.hpp"
#include "ingen/Forge.hpp"
#include "ingen/URIs.hpp"
#include "ingen/ingen.h"
#include "ingen/types.hpp"
#include "raul/RingBuffer.hpp"

#include "BufferRef.hpp"
#include "PortType.hpp"
#include "types.hpp"

namespace Raul { class Maid; }

namespace Ingen {

class URIs;

namespace Server {

class Engine;

class INGEN_API BufferFactory {
public:
	BufferFactory(Engine& engine, URIs& uris);
	~BufferFactory();

	static uint32_t audio_buffer_size(SampleCount nframes);

	uint32_t audio_buffer_size() const;
	uint32_t default_size(LV2_URID type) const;

	/** Dynamically allocate a new Buffer. */
	BufferRef create(LV2_URID type,
	                 LV2_URID value_type,
	                 uint32_t capacity = 0);

	/** Get a new buffer, reusing if possible, allocating if otherwise. */
	BufferRef get_buffer(LV2_URID type,
	                     LV2_URID value_type,
	                     uint32_t capacity);

	/** Claim an existing buffer, never allocates, real-time safe. */
	BufferRef claim_buffer(LV2_URID type,
	                       LV2_URID value_type,
	                       uint32_t capacity);

	/** Return a reference to a shared silent buffer. */
	BufferRef silent_buffer();

	void set_block_length(SampleCount block_length);
	void set_seq_size(uint32_t seq_size) { _seq_size = seq_size; }

	Forge&      forge();
	Raul::Maid& maid();

	URIs&   uris()   { return _uris; }
	Engine& engine() { return _engine; }

private:
	friend class Buffer;
	void recycle(Buffer* buf);

	Buffer* try_get_buffer(LV2_URID type);

	inline std::atomic<Buffer*>& free_list(LV2_URID type) {
		if (type == _uris.atom_Float) {
			return _free_control;
		} else if (type == _uris.atom_Sound) {
			return _free_audio;
		} else if (type == _uris.atom_Sequence) {
			return _free_sequence;
		} else {
			return _free_object;
		}
	}

	void free_list(Buffer* head);

	std::atomic<Buffer*> _free_audio;
	std::atomic<Buffer*> _free_control;
	std::atomic<Buffer*> _free_sequence;
	std::atomic<Buffer*> _free_object;

	std::mutex  _mutex;
	Engine&     _engine;
	URIs&       _uris;
	uint32_t    _seq_size;

	BufferRef _silent_buffer;
};

} // namespace Server
} // namespace Ingen

#endif // INGEN_ENGINE_BUFFERFACTORY_HPP