mirror of
https://github.com/OpenMW/openmw.git
synced 2025-01-22 11:23:52 +00:00
265 lines
9.6 KiB
C
265 lines
9.6 KiB
C
|
/*
|
||
|
This file is part of Caelum.
|
||
|
See http://www.ogre3d.org/wiki/index.php/Caelum
|
||
|
|
||
|
Copyright (c) 2008 Caelum team. See Contributors.txt for details.
|
||
|
|
||
|
Caelum 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, either version 3 of the License, or
|
||
|
(at your option) any later version.
|
||
|
|
||
|
Caelum 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 Caelum. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*/
|
||
|
|
||
|
#ifndef CAELUM__PRIVATE_PTR_H
|
||
|
#define CAELUM__PRIVATE_PTR_H
|
||
|
|
||
|
#include "CaelumPrerequisites.h"
|
||
|
|
||
|
namespace Caelum
|
||
|
{
|
||
|
/** Default traits for Caelum::PrivatePtr.
|
||
|
*
|
||
|
* This default traits class make PrivatePtr work like std::auto_ptr.
|
||
|
* Other Traits classes can derive from this and only customize some of
|
||
|
* the functions.
|
||
|
*
|
||
|
* @see PrivatePtr
|
||
|
*/
|
||
|
template<class PointedT>
|
||
|
struct DefaultPrivatePtrTraits
|
||
|
{
|
||
|
/// The type of the inner member to hold in PrivatePtr
|
||
|
typedef PointedT* InnerPointerType;
|
||
|
|
||
|
/// Return an InnerPointerType repressenting a null value.
|
||
|
static inline const InnerPointerType getNullValue() {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/// Convert InnerPointerType to a naked PointedT.
|
||
|
static inline PointedT* getPointer (const InnerPointerType& inner) {
|
||
|
return inner;
|
||
|
}
|
||
|
|
||
|
/// Destroy the inner value (and set null).
|
||
|
static void destroy (InnerPointerType& inner)
|
||
|
{
|
||
|
delete inner;
|
||
|
inner = 0;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/** Template for smart pointers with strict unique ownership.
|
||
|
* A lot of objects in Ogre are created and destroyed through other
|
||
|
* "Manager" objects. Even though the memory for such objects is never
|
||
|
* actually leaked better lifetime control is frequently useful.
|
||
|
*
|
||
|
* PrivatePtr is very similar in behaviour to std::auto_ptr but tries
|
||
|
* to mimic Ogre::SharedPtr method names. Only one PrivatePtr must exist to
|
||
|
* a certain object at any one time. Assignment and copy construction will
|
||
|
* in fact pass away ownership and set the original PrivatePtr to null.
|
||
|
*
|
||
|
* This very limited functionality makes PrivatePtr very efficient; it should
|
||
|
* have no overhead compared to doing the same thing manually.
|
||
|
*
|
||
|
* PrivatePtr supports customization through a static traits class which
|
||
|
* can customize what happens when the PrivatePtr is destroyed. This makes
|
||
|
* it possible to use PrivatePtr classes for fine control over the lifetime
|
||
|
* of objects which are otherwise managed by an external class.
|
||
|
*
|
||
|
* @see DefaultPrivatePtrTraits
|
||
|
*/
|
||
|
template<class PointedT, typename TraitsT = DefaultPrivatePtrTraits<PointedT> >
|
||
|
class PrivatePtr
|
||
|
{
|
||
|
private:
|
||
|
/// Brings InnerPointerType as a type in this scope.
|
||
|
typedef typename TraitsT::InnerPointerType InnerPointerType;
|
||
|
|
||
|
/// Inner data member.
|
||
|
InnerPointerType mInner;
|
||
|
|
||
|
public:
|
||
|
/** Change the inner value.
|
||
|
* This will destroy the old value and gain ownership of the new value.
|
||
|
*/
|
||
|
void reset (const InnerPointerType& newInner = TraitsT::getNullValue()) {
|
||
|
if (mInner == newInner) {
|
||
|
return;
|
||
|
}
|
||
|
TraitsT::destroy (mInner);
|
||
|
mInner = newInner;
|
||
|
}
|
||
|
|
||
|
InnerPointerType release () {
|
||
|
InnerPointerType result = mInner;
|
||
|
mInner = TraitsT::getNullValue();
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
/** Constructor; always initialize to 0.
|
||
|
*/
|
||
|
PrivatePtr () { mInner = TraitsT::getNullValue (); }
|
||
|
|
||
|
/** Initializing constructur
|
||
|
*/
|
||
|
PrivatePtr (const InnerPointerType& inner) { mInner = inner; }
|
||
|
|
||
|
/** Non-virtual destructor (don't derive from this).
|
||
|
*/
|
||
|
~PrivatePtr () { setNull(); }
|
||
|
|
||
|
/** Copy constructor; clears right-hand-side.
|
||
|
*/
|
||
|
PrivatePtr (PrivatePtr& rhs)
|
||
|
{
|
||
|
if (&rhs != this) {
|
||
|
this->reset (rhs.mInner);
|
||
|
rhs.mInner = TraitsT::getNullValue ();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/** Assignment
|
||
|
*/
|
||
|
const PrivatePtr& operator= (PrivatePtr& rhs)
|
||
|
{
|
||
|
if (&rhs != this) {
|
||
|
this->reset (rhs.mInner);
|
||
|
rhs.mInner = TraitsT::getNullValue ();
|
||
|
}
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
/// Check if this is null.
|
||
|
bool isNull () const { return mInner == TraitsT::getNullValue (); }
|
||
|
|
||
|
/// Set to null and destroy contents (if any).
|
||
|
void setNull () {
|
||
|
TraitsT::destroy (mInner);
|
||
|
assert(this->isNull());
|
||
|
}
|
||
|
|
||
|
PointedT* getPointer () const { return TraitsT::getPointer (mInner); }
|
||
|
PointedT* get () const { return getPointer (); }
|
||
|
PointedT* operator-> () const { return getPointer (); }
|
||
|
PointedT& operator* () const{ return *getPointer (); }
|
||
|
};
|
||
|
|
||
|
/** PrivatePtr traits for a movable object.
|
||
|
* This kind of pointer will remove the movable from the scene and destroy it.
|
||
|
*/
|
||
|
template<class MovableT>
|
||
|
struct MovableObjectPrivatePtrTraits: public DefaultPrivatePtrTraits<MovableT>
|
||
|
{
|
||
|
typedef MovableT* InnerPointerType;
|
||
|
|
||
|
static void destroy (InnerPointerType& inner)
|
||
|
{
|
||
|
if (inner != 0) {
|
||
|
//Ogre::LogManager::getSingletonPtr ()->logMessage (
|
||
|
// "PrivatePtr: Destroying movable object " + inner->getName ());
|
||
|
inner->_getManager ()->destroyMovableObject (inner);
|
||
|
inner = 0;
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
typedef PrivatePtr<Ogre::MovableObject, MovableObjectPrivatePtrTraits<Ogre::MovableObject> > PrivateMovableObjectPtr;
|
||
|
typedef PrivatePtr<Ogre::BillboardChain, MovableObjectPrivatePtrTraits<Ogre::BillboardChain> > PrivateBillboardChainPtr;
|
||
|
typedef PrivatePtr<Ogre::BillboardSet, MovableObjectPrivatePtrTraits<Ogre::BillboardSet> > PrivateBillboardSetPtr;
|
||
|
typedef PrivatePtr<Ogre::Entity, MovableObjectPrivatePtrTraits<Ogre::Entity> > PrivateEntityPtr;
|
||
|
typedef PrivatePtr<Ogre::Light, MovableObjectPrivatePtrTraits<Ogre::Light> > PrivateLightPtr;
|
||
|
typedef PrivatePtr<Ogre::ManualObject, MovableObjectPrivatePtrTraits<Ogre::ManualObject> > PrivateManualObjectPtr;
|
||
|
typedef PrivatePtr<Ogre::ParticleSystem, MovableObjectPrivatePtrTraits<Ogre::ParticleSystem> > PrivateParticleSystemPtr;
|
||
|
|
||
|
/** PrivatePtr traits for a scene node.
|
||
|
* Scene nodes are created and destroyed through the scene manager.
|
||
|
* @see PrivatePrivateSceneNodePtr
|
||
|
*/
|
||
|
struct SceneNodePrivatePtrTraits: public DefaultPrivatePtrTraits<Ogre::SceneNode>
|
||
|
{
|
||
|
static void destroy (InnerPointerType& inner)
|
||
|
{
|
||
|
if (inner) {
|
||
|
//Ogre::LogManager::getSingletonPtr ()->logMessage (
|
||
|
// "PrivatePtr: Destroying scene node " + inner->getName ());
|
||
|
inner->getCreator ()->destroySceneNode (inner->getName ());
|
||
|
inner = 0;
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
typedef PrivatePtr<Ogre::SceneNode, SceneNodePrivatePtrTraits> PrivateSceneNodePtr;
|
||
|
|
||
|
/** PrivatePtr traits for uniquely-owned resources.
|
||
|
*
|
||
|
* All ogre resources are tracked by a resource managed by name and can
|
||
|
* be globally referenced from multiple places. This traits class allows
|
||
|
* you to hold a pointer to a resource which you create and completely
|
||
|
* control.
|
||
|
*
|
||
|
* The best example of this is a cloned material. It is frequently useful
|
||
|
* to create a clone of an existing material and tweak settings for one
|
||
|
* particular usage. After the clone is no longer useful the material must
|
||
|
* be explicitly removed from the MaterialManager. Otherwise an unloaded
|
||
|
* resource handle is leaked.
|
||
|
*
|
||
|
* When the PrivatePtr gets out of scope the resource is removed from the
|
||
|
* manager. In debug mode this will also check that there are no other
|
||
|
* references to the destroyed resource.
|
||
|
*/
|
||
|
template<class PointedT, class InnerT, class ManagerT>
|
||
|
struct PrivateResourcePtrTraits
|
||
|
{
|
||
|
typedef InnerT InnerPointerType;
|
||
|
|
||
|
static const InnerT getNullValue () {
|
||
|
return InnerT();
|
||
|
}
|
||
|
|
||
|
static PointedT* getPointer (const InnerPointerType& inner) {
|
||
|
return inner.getPointer ();
|
||
|
}
|
||
|
|
||
|
static void destroy (InnerPointerType& inner) {
|
||
|
if (!inner.isNull ()) {
|
||
|
//Ogre::LogManager::getSingletonPtr ()->logMessage (
|
||
|
// "PrivateResourcePtrTraits: Destroying owned resource"
|
||
|
// " name=" + inner->getName () +
|
||
|
// " handle=" + Ogre::StringConverter::toString (inner->getHandle ()) );
|
||
|
ManagerT::getSingletonPtr ()->remove (inner->getHandle ());
|
||
|
assert (inner.unique () && "Resource pointer not unique after destruction");
|
||
|
inner.setNull();
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
typedef PrivatePtr <
|
||
|
Ogre::Material,
|
||
|
PrivateResourcePtrTraits <
|
||
|
Ogre::Material,
|
||
|
Ogre::MaterialPtr,
|
||
|
Ogre::MaterialManager
|
||
|
>
|
||
|
> PrivateMaterialPtr;
|
||
|
|
||
|
typedef PrivatePtr <
|
||
|
Ogre::Mesh,
|
||
|
PrivateResourcePtrTraits <
|
||
|
Ogre::Mesh,
|
||
|
Ogre::MeshPtr,
|
||
|
Ogre::MeshManager
|
||
|
>
|
||
|
> PrivateMeshPtr;
|
||
|
}
|
||
|
|
||
|
#endif // CAELUM__PRIVATE_PTR_H
|