2008-10-07 05:16:02 +02:00
|
|
|
/*
|
|
|
|
|
REFARRAYIMPL.hpp - some implementations of the ref-array interface
|
|
|
|
|
|
|
|
|
|
Copyright (C) Lumiera.org
|
|
|
|
|
2008, Hermann Vosseler <Ichthyostega@web.de>
|
|
|
|
|
|
|
|
|
|
This program 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.
|
|
|
|
|
|
|
|
|
|
This program 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 more 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., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef LIB_REFARRAYIMPL_H
|
|
|
|
|
#define LIB_REFARRAYIMPL_H
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include "lib/refarray.hpp"
|
|
|
|
|
#include "proc/nobugcfg.hpp"
|
|
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
|
using std::vector;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
namespace lib {
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Wrap a vector holding objects of a subtype
|
|
|
|
|
* and provide array-like access using the interface type.
|
|
|
|
|
*/
|
2008-10-12 23:52:19 +02:00
|
|
|
template<class B, class IM = B>
|
2008-10-07 05:16:02 +02:00
|
|
|
class RefArrayVectorWrapper
|
2008-10-12 23:52:19 +02:00
|
|
|
: public RefArray<B>
|
2008-10-07 05:16:02 +02:00
|
|
|
{
|
|
|
|
|
typedef vector<IM> const& Tab;
|
|
|
|
|
Tab table_;
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
|
|
RefArrayVectorWrapper (Tab toWrap)
|
|
|
|
|
: table_(toWrap)
|
|
|
|
|
{ }
|
|
|
|
|
|
|
|
|
|
virtual size_t size() const
|
|
|
|
|
{
|
|
|
|
|
return table_.size();
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-12 23:52:19 +02:00
|
|
|
virtual B const& operator[] (uint i) const
|
2008-10-07 05:16:02 +02:00
|
|
|
{
|
|
|
|
|
REQUIRE (i < size());
|
|
|
|
|
return table_[i];
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2008-10-09 05:23:24 +02:00
|
|
|
/**
|
|
|
|
|
* This variation of the wrapper actually \em is
|
|
|
|
|
* a vector, but can act as a RefArray
|
|
|
|
|
*/
|
2008-10-12 23:52:19 +02:00
|
|
|
template<class B, class IM = B>
|
2008-10-09 05:23:24 +02:00
|
|
|
class RefArrayVector
|
|
|
|
|
: public vector<IM>,
|
2008-10-12 23:52:19 +02:00
|
|
|
public RefArrayVectorWrapper<B,IM>
|
2008-10-09 05:23:24 +02:00
|
|
|
{
|
2008-10-12 23:52:19 +02:00
|
|
|
typedef RefArrayVectorWrapper<B,IM> Wrap;
|
|
|
|
|
typedef vector<IM> Vect;
|
2008-10-09 05:23:24 +02:00
|
|
|
typedef typename Vect::size_type Size_t;
|
|
|
|
|
typedef typename Vect::value_type Val_t;
|
|
|
|
|
|
|
|
|
|
public:
|
2008-10-12 23:52:19 +02:00
|
|
|
RefArrayVector() : Vect(), Wrap((Vect&)*this) {}
|
|
|
|
|
RefArrayVector(Size_t n, Val_t const& v = Val_t()) : Vect(n,v), Wrap((Vect&)*this) {}
|
|
|
|
|
RefArrayVector(Vect const& ref) : Vect(ref), Wrap((Vect&)*this) {}
|
2008-10-09 05:23:24 +02:00
|
|
|
|
|
|
|
|
using Vect::size;
|
|
|
|
|
using Wrap::operator[];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2008-10-12 23:52:19 +02:00
|
|
|
* RefArray implementation based on a fix sized array,
|
|
|
|
|
* i.e. the storage is embedded. Embedded subclass obj
|
|
|
|
|
* either need to be default constructible or be
|
|
|
|
|
* placed directly by a factory
|
2008-10-09 05:23:24 +02:00
|
|
|
*/
|
2008-10-12 23:52:19 +02:00
|
|
|
template<class B, size_t n, class IM = B>
|
2008-10-09 05:23:24 +02:00
|
|
|
class RefArrayTable
|
2008-10-12 23:52:19 +02:00
|
|
|
: public RefArray<B>
|
2008-10-09 05:23:24 +02:00
|
|
|
{
|
2008-10-12 23:52:19 +02:00
|
|
|
char storage_[n*sizeof(IM)];
|
|
|
|
|
IM* array_;
|
2008-10-09 05:23:24 +02:00
|
|
|
|
|
|
|
|
public:
|
2008-10-12 23:52:19 +02:00
|
|
|
RefArrayTable() ///< objects created in-place by default ctor
|
|
|
|
|
: array_ (reinterpret_cast<IM*> (&storage_))
|
2008-10-09 05:23:24 +02:00
|
|
|
{
|
|
|
|
|
size_t i=0;
|
|
|
|
|
try
|
|
|
|
|
{
|
2008-10-12 23:52:19 +02:00
|
|
|
while (i<n)
|
|
|
|
|
new(&array_[i++]) IM();
|
2008-10-09 05:23:24 +02:00
|
|
|
}
|
|
|
|
|
catch(...) { cleanup(i); throw; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template<class FAC>
|
2008-10-12 23:52:19 +02:00
|
|
|
RefArrayTable(FAC& factory) ///< objects created in-place by factory
|
|
|
|
|
: array_ (reinterpret_cast<IM*> (&storage_))
|
2008-10-09 05:23:24 +02:00
|
|
|
{
|
|
|
|
|
size_t i=0;
|
|
|
|
|
try
|
|
|
|
|
{
|
2008-10-12 23:52:19 +02:00
|
|
|
while (i<n)
|
|
|
|
|
factory(&array_[i++]);
|
2008-10-09 05:23:24 +02:00
|
|
|
}
|
|
|
|
|
catch(...) { cleanup(i); throw; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
~RefArrayTable() { cleanup(); }
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
void cleanup(size_t top=n) throw()
|
|
|
|
|
{
|
|
|
|
|
while (top) array_[--top].~IM();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-10-12 23:52:19 +02:00
|
|
|
public: //-----RefArray-Interface------------
|
2008-10-09 05:23:24 +02:00
|
|
|
|
|
|
|
|
virtual size_t size() const
|
|
|
|
|
{
|
|
|
|
|
return n;
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-12 23:52:19 +02:00
|
|
|
virtual B const& operator[] (uint i) const
|
2008-10-09 05:23:24 +02:00
|
|
|
{
|
|
|
|
|
REQUIRE (i < size());
|
|
|
|
|
return array_[i];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
2008-10-07 05:16:02 +02:00
|
|
|
} // namespace lib
|
|
|
|
|
#endif
|