/** * @file llptrto.h * @author Nat Goodspeed * @date 2008-08-19 * @brief LLPtrTo is a template helper to pick either TARGET* or -- when * TARGET is a subclass of LLRefCount or LLThreadSafeRefCount -- * LLPointer. LLPtrTo<> chooses whichever pointer type is best. * * $LicenseInfo:firstyear=2008&license=viewerlgpl$ * Second Life Viewer Source Code * Copyright (C) 2010, Linden Research, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; * version 2.1 of the License only. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * * Linden Research, Inc., 945 Battery Street, San Francisco, CA 94111 USA * $/LicenseInfo$ */ #if ! defined(LL_LLPTRTO_H) #define LL_LLPTRTO_H #include "llpointer.h" #include "llrefcount.h" // LLRefCount #include #include #include #include #include // std::shared_ptr, std::unique_ptr #include /** * LLPtrTo::type is either of two things: * * * When TARGET is a subclass of either LLRefCount or LLThreadSafeRefCount, * LLPtrTo::type is LLPointer. * * Otherwise, LLPtrTo::type is TARGET*. * * This way, a class template can use LLPtrTo::type to select an * appropriate pointer type to store. */ template struct LLPtrTo { typedef T* type; }; /// specialize for subclasses of LLRefCount template struct LLPtrTo::value >::type> { typedef LLPointer type; }; /// specialize for subclasses of LLThreadSafeRefCount template struct LLPtrTo::value >::type> { typedef LLPointer type; }; /** * LLRemovePointer::type gets you the underlying (pointee) type. */ template struct LLRemovePointer { typedef typename boost::remove_pointer::type type; }; /// specialize for LLPointer template struct LLRemovePointer< LLPointer > { typedef SOMECLASS type; }; namespace LL { /***************************************************************************** * get_ref() *****************************************************************************/ template struct GetRef { // return const ref or non-const ref, depending on whether we can bind // a non-const lvalue ref to the argument const auto& operator()(const T& obj) const { return obj; } auto& operator()(T& obj) const { return obj; } }; template struct GetRef { const auto& operator()(const T* ptr) const { return *ptr; } }; template struct GetRef { auto& operator()(T* ptr) const { return *ptr; } }; template struct GetRef< LLPointer > { auto& operator()(LLPointer ptr) const { return *ptr; } }; /// whether we're passed a pointer or a reference, return a reference template auto& get_ref(T& ptr_or_ref) { return GetRef::type>()(ptr_or_ref); } template const auto& get_ref(const T& ptr_or_ref) { return GetRef::type>()(ptr_or_ref); } /***************************************************************************** * get_ptr() *****************************************************************************/ // if T is any pointer type we recognize, return it unchanged template const T* get_ptr(const T* ptr) { return ptr; } template T* get_ptr(T* ptr) { return ptr; } template const std::shared_ptr& get_ptr(const std::shared_ptr& ptr) { return ptr; } template const std::unique_ptr& get_ptr(const std::unique_ptr& ptr) { return ptr; } template const boost::shared_ptr& get_ptr(const boost::shared_ptr& ptr) { return ptr; } template const boost::intrusive_ptr& get_ptr(const boost::intrusive_ptr& ptr) { return ptr; } template const LLPointer& get_ptr(const LLPointer& ptr) { return ptr; } // T is not any pointer type we recognize, take a pointer to the parameter template const T* get_ptr(const T& obj) { return &obj; } template T* get_ptr(T& obj) { return &obj; } } // namespace LL #endif /* ! defined(LL_LLPTRTO_H) */