/*
This file is part of Ingen.
Copyright 2007-2017 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 .
*/
#ifndef INGEN_ATOMFORGE_HPP
#define INGEN_ATOMFORGE_HPP
#include "ingen/memory.hpp"
#include "lv2/atom/atom.h"
#include "lv2/atom/forge.h"
#include "lv2/atom/util.h"
#include "lv2/urid/urid.h"
#include "sord/sord.h"
#include "sord/sordmm.hpp"
#include "sratom/sratom.h"
#include
#include
#include
#include
#include
namespace ingen {
/// An atom forge that writes to an automatically-resized memory buffer
class AtomForge : public LV2_Atom_Forge
{
public:
explicit AtomForge(LV2_URID_Map& map)
: LV2_Atom_Forge{}
, _sratom{sratom_new(&map)}
, _buf{static_cast(calloc(8, sizeof(LV2_Atom)))}
{
lv2_atom_forge_init(this, &map);
lv2_atom_forge_set_sink(this, c_append, c_deref, this);
}
/// Forge an atom from `node` in `model`
void read(Sord::World& world, SordModel* model, const SordNode* node)
{
sratom_read(_sratom.get(), this, world.c_obj(), model, node);
}
/// Return the top-level atom that has been forged
const LV2_Atom* atom() const { return _buf.get(); }
/// Clear the atom buffer and reset the forge
void clear()
{
lv2_atom_forge_set_sink(this, c_append, c_deref, this);
_size = 0;
*_buf = {0U, 0U};
}
/// Return the internal atom serialiser
Sratom& sratom() { return *_sratom; }
private:
struct SratomDeleter { void operator()(Sratom* s) { sratom_free(s); } };
using AtomPtr = std::unique_ptr>;
using SratomPtr = std::unique_ptr;
/// Append some data and return a reference to its start
intptr_t append(const void* data, uint32_t len) {
// Record offset of the start of this write (+1 to avoid null)
const intptr_t ref = _size + 1;
// Update size and reallocate if necessary
if (lv2_atom_pad_size(_size + len) > _capacity) {
_capacity = lv2_atom_pad_size(_size + len);
_buf = AtomPtr{static_cast(
realloc(_buf.release(), _capacity)),
FreeDeleter{}};
}
// Append new data
memcpy(reinterpret_cast(_buf.get()) + _size, data, len);
_size += len;
return ref;
}
/// Dereference a reference previously returned by append()
LV2_Atom* deref(intptr_t ref) {
/* Make some assumptions and do unnecessary math to appease
-Wcast-align. This is questionable at best, though the forge should
only dereference references to aligned atoms. */
assert((ref - 1) % sizeof(LV2_Atom) == 0);
return static_cast(_buf.get() + (ref - 1) / sizeof(LV2_Atom));
// Alternatively:
// return (LV2_Atom*)((uint8_t*)_buf + ref - 1);
}
static LV2_Atom_Forge_Ref
c_append(void* self, const void* data, uint32_t len) {
return static_cast(self)->append(data, len);
}
static LV2_Atom*
c_deref(void* self, LV2_Atom_Forge_Ref ref) {
return static_cast(self)->deref(ref);
}
size_t _size{0}; ///< Current atom size
size_t _capacity{8 * sizeof(LV2_Atom)}; ///< Allocated size of buffer
SratomPtr _sratom; ///< Atom serialiser
AtomPtr _buf; ///< Atom buffer
};
} // namespace ingen
#endif // INGEN_ATOMFORGE_HPP