1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-04 18:06:22 +03:00
occt/src/Standard/Standard_Handle.hxx
abv a9dde4a31b 0027104: DownCast() cannot return null for mismatched handle
Method DownCast() is made template, to be available only when argument is actually a pointer or handle to a base class.

For compatibility with existing code, method DownCast() that can be used for the same type, derived, or unrelated class (i.e. where there is no actual down casting) is still available, its use shall cause "deprecated" compiler warning.

OCCT code is updated to remove meaningless DownCast()s; a few places where DownCast() was used with argument of unrelated type are corrected.

DRAW command QAHandleCast is removed (it was useful only during redesign of handles).
2016-02-20 10:10:15 +03:00

418 lines
13 KiB
C++

// 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 <Standard_Address.hxx>
#include <Standard_Stream.hxx>
#include <Standard_Transient.hxx>
#include <type_traits>
class Standard_Transient;
namespace opencascade {
//! Trait yielding true if class T1 is base of T2 but not the same
template <class T1, class T2, class Dummy = void>
struct is_base_but_not_same : std::is_base_of <T1, T2> {};
//! Explicit specialization of is_base_of trait to workaround the
//! requirement of type to be complete when T1 and T2 are the same.
template <class T1, class T2>
struct is_base_but_not_same <T1, T2, typename std::enable_if <std::is_same <T1, T2>::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 T>
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<T*>(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<T*>(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
const T* get () const { return static_cast<const T*>(this->entity); }
//! STL-like cast to pointer to referred object
T* get () { return static_cast<T*>(this->entity); }
//! Member access operator (note non-const)
T* operator-> () const { return static_cast<T*>(this->entity); }
//! Dereferencing operator
T& operator* () { return *get(); }
//! Const dereferencing operator
const T& operator*() const { return *get(); }
//! Check for equality
template <class T2>
bool operator== (const handle<T2>& theHandle) const
{
return get() == theHandle.get();
}
//! Check for equality
template <class T2>
bool operator== (const T2 *thePtr) const
{
return get() == thePtr;
}
//! Check for equality
template <class T2>
friend bool operator== (const T2 *left, const handle& right)
{
return left == right.get();
}
//! Check for inequality
template <class T2>
bool operator!= (const handle<T2>& theHandle) const
{
return get() != theHandle.get();
}
//! Check for inequality
template <class T2>
bool operator!= (const T2 *thePtr) const
{
return get() != thePtr;
}
//! Check for inequality
template <class T2>
friend bool operator!= (const T2 *left, const handle& right)
{
return left != right.get();
}
//! Compare operator for possible use in std::map<> etc.
template <class T2>
bool operator< (const handle<T2>& theHandle) const
{
return get() < theHandle.get();
}
//! Down casting operator from handle to base type
template <class T2>
static typename std::enable_if<is_base_but_not_same<T2, T>::value, handle>::type
DownCast (const handle<T2>& theObject)
{
return handle (dynamic_cast<T*>(const_cast<T2*>(theObject.get())));
}
//! Down casting operator from pointer to base type
template <class T2>
static typename std::enable_if<is_base_but_not_same<T2, T>::value, handle>::type
DownCast (const T2* thePtr)
{
return handle (dynamic_cast<T*>(const_cast<T2*>(thePtr)));
}
//! For compatibility, define down casting operator from non-base type, as deprecated
template <class T2>
Standard_DEPRECATED("down-casting from object of the same or unrelated type is meaningless")
static handle DownCast (const handle<T2>& theObject, typename std::enable_if<!is_base_but_not_same<T2, T>::value, void*>::type = 0)
{
return handle (dynamic_cast<T*>(const_cast<T2*>(theObject.get())));
}
//! For compatibility, define down casting operator from non-base type, as deprecated
template <class T2>
Standard_DEPRECATED("down-casting from object of the same or unrelated type is meaningless")
static handle DownCast (const T2* thePtr, typename std::enable_if<!is_base_but_not_same<T2, T>::value, void*>::type = 0)
{
return handle (dynamic_cast<T*>(const_cast<T2*>(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 <class T2, typename = typename std::enable_if <is_base_but_not_same <T, T2>::value>::type>
handle (const handle<T2>& theHandle) :
entity(theHandle.entity)
{
BeginScope();
}
//! Generalized move constructor
template <class T2, typename = typename std::enable_if <is_base_but_not_same <T, T2>::value>::type>
handle (handle<T2>&& theHandle)
: entity(theHandle.entity)
{
theHandle.entity = 0;
}
//! Generalized assignment operator
template <class T2, typename = typename std::enable_if <is_base_but_not_same <T, T2>::value>::type>
handle operator = (const handle<T2>& theHandle)
{
Assign (theHandle.entity);
return *this;
}
//! Generalized move operator
template <class T2, typename = typename std::enable_if <is_base_but_not_same <T, T2>::value>::type>
handle& operator= (handle<T2>&& theHandle)
{
std::swap (this->entity, theHandle.entity);
return *this;
}
#else
//! Upcast to const reference to base type.
template <class T2, typename = typename std::enable_if<is_base_but_not_same<T2, T>::value>::type>
operator const handle<T2>& () const
{
return reinterpret_cast<const handle<T2>&>(*this);
}
//! Upcast to non-const reference to base type.
//! NB: this cast can be dangerous, but required for legacy code; see #26377
template <class T2, typename = typename std::enable_if<is_base_but_not_same<T2, T>::value>::type>
operator handle<T2>& ()
{
return reinterpret_cast<handle<T2>&>(*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 <class T2>
handle (const handle<T2>& theHandle, typename std::enable_if <is_base_but_not_same <T, T2>::value>::type* = nullptr) :
entity(theHandle.entity)
{
BeginScope();
}
//! Generalized move constructor
template <class T2>
handle (handle<T2>&& theHandle, typename std::enable_if <is_base_but_not_same <T, T2>::value>::type* = nullptr)
: entity(theHandle.entity)
{
theHandle.entity = 0;
}
//! Generalized assignment operator.
template <class T2>
handle operator = (const handle<T2>& theHandle)
{
std::enable_if <is_base_but_not_same <T, T2>::value, void*>::type aTypeCheckHelperVar;
(void)aTypeCheckHelperVar;
Assign (theHandle.entity);
return *this;
}
//! Generalized move operator
template <class T2>
handle& operator= (handle<T2>&& theHandle)
{
std::enable_if <is_base_but_not_same <T, T2>::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 <class T2>
operator const handle<T2>& () 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<typename std::enable_if<is_base_but_not_same<T2, T>::value, const handle<T2>&>::type>(*this);
}
//! Upcast to non-const reference to base type.
//! NB: this cast can be dangerous, but required for legacy code; see #26377
template <class T2>
Standard_DEPRECATED("Passing non-const reference to handle of base type in function is unsafe; use variable of exact type")
operator handle<T2>& ()
{
// 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<typename std::enable_if<is_base_but_not_same<T2, T>::value, handle<T2>&>::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 <class T2> friend class handle;
private:
Standard_Transient* entity;
};
} // namespace opencascade
//! Define Handle() macro
#define Handle(Class) opencascade::handle<Class>
//! Global method HashCode(), for use in hash maps
template <class T>
inline Standard_Integer HashCode (const Handle(T)& theHandle, const Standard_Integer theUpper)
{
return ::HashCode (const_cast<Standard_Address>(static_cast<const void*>(theHandle.get())), theUpper);
}
//! For compatibility with previous versions of OCCT, defines typedef opencascade::handle<Class> Handle(Class)
#define DEFINE_STANDARD_HANDLECLASS(C1,C2,BC) class C1; typedef Handle(C1) Handle_##C1;
#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