/*
This file is part of Mitsuba, a physically based rendering system.
Copyright (c) 2007-2012 by Wenzel Jakob and others.
Mitsuba is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License Version 3
as published by the Free Software Foundation.
Mitsuba 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, see .
*/
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#if defined(__OSX__)
# include
#endif
MTS_NAMESPACE_BEGIN
namespace mi = boost::multi_index;
/* The native TLS classes on Linux/MacOS/Windows only support a limited number
of dynamically allocated entries (usually 1024 or 1088). Furthermore, they
do not provide appropriate cleanup semantics when the TLS object or one of
the assocated threads dies. The custom TLS code provided in Mitsuba has no
such limits (caching in various subsystems of Mitsuba may create a huge amount,
so this is a big deal) as well as nice cleanup semantics. The implementation
is designed to make the \c get() operation as fast as as possible at the cost
of more involved locking when creating or destroying threads and TLS objects */
namespace detail {
/// A single TLS entry + cleanup hook
struct TLSEntry {
void *data;
void (*destructFunctor)(void *);
inline TLSEntry() : data(NULL), destructFunctor(NULL) { }
};
/// boost multi-index element to act as replacement of map
template
struct mutable_pair {
mutable_pair(const T1 &f, const T2 &s) : first(f), second(s) { }
T1 first;
mutable T2 second;
};
/// Per-thread TLS entry map
struct PerThreadData {
typedef mutable_pair MapData;
typedef mi::member key_member;
struct seq_tag {};
struct key_tag {};
typedef mi::multi_index_container, key_member>,
mi::sequenced >
>
> Map;
typedef mi::index