// Copyright (c) 2014 OPEN CASCADE SAS // // This file is part of Open CASCADE Technology software library. // // This library is free software; you can redistribute it and/or modify it under // the terms of the GNU Lesser General Public License version 2.1 as published // by the Free Software Foundation, with special exception defined in the file // OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT // distribution for complete text of the license and disclaimer of any warranty. // // Alternatively, this file may be used under the terms of Open CASCADE // commercial license or contractual agreement. #ifndef _Standard_Handle_HeaderFile #define _Standard_Handle_HeaderFile #include #include #include #include class Standard_Transient; namespace opencascade { //! Trait yielding true if class T1 is base of T2 but not the same template struct is_base_but_not_same : std::is_base_of {}; //! Explicit specialization of is_base_of trait to workaround the //! requirement of type to be complete when T1 and T2 are the same. template struct is_base_but_not_same ::value>::type> : std::false_type {}; //! Intrusive smart pointer for use with Standard_Transient class and its descendants. //! //! This class is similar to boost::intrusive_ptr<>, with additional //! feature historically supported by Handles in OCCT: //! it has type conversion to const reference to handle to the base types, //! which allows it to be passed by reference //! in functions accepting reference to handle to base class. //! //! These casts (potentially unsafe) can be disabled by defining macro //! OCCT_HANDLE_NOCAST; if it is defined, generalized copy constructor //! and assignment operators are defined allowing to initialize handle //! of base type from handle to derived type. template class handle { public: //! STL-compliant typedef of contained type typedef T element_type; public: //! Empty constructor handle () : entity(0) {} //! Constructor from pointer to new object handle (const T *thePtr) : entity(const_cast(thePtr)) { BeginScope(); } //! Copy constructor handle (const handle& theHandle) : entity(theHandle.entity) { BeginScope(); } //! Move constructor handle (handle&& theHandle) : entity(theHandle.entity) { theHandle.entity = 0; } //! Destructor ~handle () { EndScope(); } //! Nullify the handle void Nullify() { EndScope(); } //! Check for being null bool IsNull() const { return entity == 0; } //! Reset by new pointer void reset (T* thePtr) { Assign (thePtr); } //! Assignment operator handle& operator= (const handle& theHandle) { Assign (theHandle.entity); return *this; } //! Assignment to pointer handle& operator= (const T* thePtr) { Assign (const_cast(thePtr)); return *this; } //! Move operator handle& operator= (handle&& theHandle) { std::swap (this->entity, theHandle.entity); return *this; } //! STL-like cast to pointer to referred object (note non-const). //! @sa std::shared_ptr::get() T* get() const { return static_cast(this->entity); } //! Member access operator (note non-const) T* operator-> () const { return static_cast(this->entity); } //! Dereferencing operator (note non-const) T& operator* () const { return *get(); } //! Check for equality template bool operator== (const handle& theHandle) const { return get() == theHandle.get(); } //! Check for equality template bool operator== (const T2 *thePtr) const { return get() == thePtr; } //! Check for equality template friend bool operator== (const T2 *left, const handle& right) { return left == right.get(); } //! Check for inequality template bool operator!= (const handle& theHandle) const { return get() != theHandle.get(); } //! Check for inequality template bool operator!= (const T2 *thePtr) const { return get() != thePtr; } //! Check for inequality template friend bool operator!= (const T2 *left, const handle& right) { return left != right.get(); } //! Compare operator for possible use in std::map<> etc. template bool operator< (const handle& theHandle) const { return get() < theHandle.get(); } //! Down casting operator from handle to base type template static typename std::enable_if::value, handle>::type DownCast (const handle& theObject) { return handle (dynamic_cast(const_cast(theObject.get()))); } //! Down casting operator from pointer to base type template static typename std::enable_if::value, handle>::type DownCast (const T2* thePtr) { return handle (dynamic_cast(const_cast(thePtr))); } //! For compatibility, define down casting operator from non-base type, as deprecated template Standard_DEPRECATED("down-casting from object of the same or unrelated type is meaningless") static handle DownCast (const handle& theObject, typename std::enable_if::value, void*>::type = 0) { return handle (dynamic_cast(const_cast(theObject.get()))); } //! For compatibility, define down casting operator from non-base type, as deprecated template Standard_DEPRECATED("down-casting from object of the same or unrelated type is meaningless") static handle DownCast (const T2* thePtr, typename std::enable_if::value, void*>::type = 0) { return handle (dynamic_cast(const_cast(thePtr))); } #if (defined(__clang__)) || (defined(__INTEL_COMPILER) && __INTEL_COMPILER >= 1300) || \ (defined(_MSC_VER) && _MSC_VER >= 1800) || \ (defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6))) //! Conversion to bool for use in conditional expressions explicit operator bool () const { return entity != nullptr; } #else /* fallback version for compilers not supporting explicit conversion operators (VC10, VC11, GCC below 4.5) */ //! Conversion to bool-compatible type for use in conditional expressions operator Standard_Transient* handle::* () const { return entity ? &handle::entity : 0; } #endif // Support of conversions to handle of base type: // - copy and move constructors and assignment operators if OCCT_HANDLE_NOCAST is defined // - operators of upcast to const reference to base type otherwise #if (defined(__clang__)) || (defined(__INTEL_COMPILER) && __INTEL_COMPILER >= 1206) || \ (defined(_MSC_VER) && _MSC_VER >= 1800) || \ (defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3))) #ifdef OCCT_HANDLE_NOCAST //! Generalized copy constructor. //! Constructs handle holding entity of base type (T) from the one which holds entity of derived type (T2). template ::value>::type> handle (const handle& theHandle) : entity(theHandle.entity) { BeginScope(); } //! Generalized move constructor template ::value>::type> handle (handle&& theHandle) : entity(theHandle.entity) { theHandle.entity = 0; } //! Generalized assignment operator template ::value>::type> handle operator = (const handle& theHandle) { Assign (theHandle.entity); return *this; } //! Generalized move operator template ::value>::type> handle& operator= (handle&& theHandle) { std::swap (this->entity, theHandle.entity); return *this; } #else //! Upcast to const reference to base type. template ::value>::type> operator const handle& () const { return reinterpret_cast&>(*this); } //! Upcast to non-const reference to base type. //! NB: this cast can be dangerous, but required for legacy code; see #26377 template ::value>::type> operator handle& () { return reinterpret_cast&>(*this); } #endif /* OCCT_HANDLE_NOCAST */ #else /* fallback version for compilers not supporting default arguments of function templates (VC10, VC11, GCC below 4.3) */ #ifdef OCCT_HANDLE_NOCAST //! Generalized copy constructor. //! Constructs handle holding entity of base type (T) from the one which holds entity of derived type (T2). template handle (const handle& theHandle, typename std::enable_if ::value>::type* = nullptr) : entity(theHandle.entity) { BeginScope(); } //! Generalized move constructor template handle (handle&& theHandle, typename std::enable_if ::value>::type* = nullptr) : entity(theHandle.entity) { theHandle.entity = 0; } //! Generalized assignment operator. template handle operator = (const handle& theHandle) { std::enable_if ::value, void*>::type aTypeCheckHelperVar; (void)aTypeCheckHelperVar; Assign (theHandle.entity); return *this; } //! Generalized move operator template handle& operator= (handle&& theHandle) { std::enable_if ::value, void*>::type aTypeCheckHelperVar; (void)aTypeCheckHelperVar; std::swap (this->entity, theHandle.entity); return *this; } #else //! Upcast to const reference to base type. //! NB: this implementation will cause ambiguity errors on calls to overloaded //! functions accepting handles to different types, since compatibility is //! checked in the cast code rather than ensured by SFINAE (possible with C++11) template operator const handle& () const { // error "type is not a member of enable_if" will be generated if T2 is not sub-type of T // (handle is being cast to const& to handle of non-base type) return reinterpret_cast::value, const handle&>::type>(*this); } //! Upcast to non-const reference to base type. //! NB: this cast can be dangerous, but required for legacy code; see #26377 template Standard_DEPRECATED("Passing non-const reference to handle of base type in function is unsafe; use variable of exact type") operator handle& () { // error "type is not a member of enable_if" will be generated if T2 is not sub-type of T // (handle is being cast to const& to handle of non-base type) return reinterpret_cast::value, handle&>::type>(*this); } #endif /* OCCT_HANDLE_NOCAST */ #endif /* compiler switch */ private: //! Assignment void Assign (Standard_Transient *thePtr) { if (thePtr == entity) return; EndScope(); entity = thePtr; BeginScope(); } //! Increment reference counter of referred object void BeginScope() { if (entity != 0) entity->IncrementRefCounter(); } //! Decrement reference counter and if 0, destroy referred object void EndScope() { if (entity != 0 && entity->DecrementRefCounter() == 0) entity->Delete(); entity = 0; } template friend class handle; private: Standard_Transient* entity; }; } // namespace opencascade //! Define Handle() macro #define Handle(Class) opencascade::handle //! Global method HashCode(), for use in hash maps template inline Standard_Integer HashCode (const Handle(T)& theHandle, const Standard_Integer theUpper) { return ::HashCode (const_cast(static_cast(theHandle.get())), theUpper); } //! For compatibility with previous versions of OCCT, define Handle_Class alias for opencascade::handle. #if (defined(_MSC_VER) && _MSC_VER >= 1800) //! For Visual Studio 2013+, define Handle_Class as non-template class to allow exporting this type in C++/CLI. #define DEFINE_STANDARD_HANDLECLASS(C1,C2,BC) class C1; class Handle_##C1 : public Handle(C1) \ { \ public: \ Handle_##C1() {} \ Handle_##C1(Handle(C1)&& theHandle) : Handle(C1)(theHandle) {} \ template ::value>::type> \ inline Handle_##C1(const opencascade::handle& theOther) : Handle(C1)(theOther) {} \ template ::value>::type> \ inline Handle_##C1(const T2* theOther) : Handle(C1)(theOther) {} \ template inline Handle_##C1& operator=(T theOther) { Handle(C1)::operator=(theOther); return *this; } \ }; #else //! For other compilers, use simple typedef #define DEFINE_STANDARD_HANDLECLASS(C1,C2,BC) class C1; typedef Handle(C1) Handle_##C1; #endif #define DEFINE_STANDARD_HANDLE(C1,C2) DEFINE_STANDARD_HANDLECLASS(C1,C2,Standard_Transient) #define DEFINE_STANDARD_PHANDLE(C1,C2) DEFINE_STANDARD_HANDLECLASS(C1,C2,Standard_Persistent) #endif