// Copyright (c) 2006 Joao Abecasis // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) #if !defined(HPX_UTIL_STATIC_JUN_12_2008_0934AM) #define HPX_UTIL_STATIC_JUN_12_2008_0934AM #include #include #include #include #include #include #include #include #include // for placement new namespace MapReduce { // // Provides thread-safe initialization of a single static instance of T. // // This instance is guaranteed to be constructed on static storage in a // thread-safe manner, on the first call to the constructor of static_. // // Requirements: // T is default constructible or has one argument // T::T() MUST not throw! // this is a requirement of boost::call_once. // template struct static_ : boost::noncopyable { private: struct destructor { ~destructor() { static_::get_address()->~value_type(); } }; struct default_constructor { static void construct() { new (static_::get_address()) value_type(); static destructor d; } }; template struct copy_constructor { static U const* pv; static void construct() { new (get_address()) value_type(*pv); static destructor d; } }; public: typedef T value_type; typedef typename boost::call_traits::reference reference; typedef typename boost::call_traits::const_reference const_reference; static_(Tag = Tag()) { boost::call_once(&default_constructor::construct, constructed_); } template static_(U const & val, Tag = Tag()) { copy_constructor::pv = boost::addressof(val); boost::call_once(©_constructor::construct, constructed_); } operator reference() { return this->get(); } operator const_reference() const { return this->get(); } reference get() { return *this->get_address(); } const_reference get() const { return *this->get_address(); } private: typedef typename boost::add_pointer::type pointer; static pointer get_address() { return static_cast(data_.address()); } typedef boost::aligned_storage::value> storage_type; static storage_type data_; static boost::once_flag constructed_; }; template template U const* static_::copy_constructor::pv = NULL; template typename static_::storage_type static_::data_; template boost::once_flag static_::constructed_ = BOOST_ONCE_INIT; } #endif // include guard