159 lines
5.1 KiB
C++
159 lines
5.1 KiB
C++
//===-- CFCReleaser.h -------------------------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef CoreFoundationCPP_CFReleaser_h_
|
|
#define CoreFoundationCPP_CFReleaser_h_
|
|
|
|
#include <CoreFoundation/CoreFoundation.h>
|
|
|
|
#ifdef __cplusplus
|
|
|
|
#include <assert.h>
|
|
|
|
//----------------------------------------------------------------------
|
|
// Templatized CF helper class that can own any CF pointer and will
|
|
// call CFRelease() on any valid pointer it owns unless that pointer is
|
|
// explicitly released using the release() member function. This class
|
|
// is designed to mimic the std::auto_ptr<T> class and has all of the
|
|
// same functions. The one thing to watch out for is the
|
|
// CFCReleaser<T>::release() function won't actually CFRelease any owned
|
|
// pointer, it is designed to relinquish ownership of the pointer just
|
|
// like std:auto_ptr<T>::release() does.
|
|
//----------------------------------------------------------------------
|
|
template <class T>
|
|
class CFCReleaser
|
|
{
|
|
public:
|
|
//----------------------------------------------------------
|
|
// Constructor that takes a pointer to a CF object that is
|
|
// to be released when this object goes out of scope
|
|
//----------------------------------------------------------
|
|
CFCReleaser(T ptr = NULL) :
|
|
_ptr(ptr)
|
|
{
|
|
}
|
|
|
|
//----------------------------------------------------------
|
|
// Copy constructor
|
|
//
|
|
// Note that copying a CFCReleaser will not transfer
|
|
// ownership of the contained pointer, but it will bump its
|
|
// reference count. This is where this class differs from
|
|
// std::auto_ptr.
|
|
//----------------------------------------------------------
|
|
CFCReleaser(const CFCReleaser& rhs) :
|
|
_ptr(rhs.get())
|
|
{
|
|
if (get())
|
|
::CFRetain(get());
|
|
}
|
|
|
|
|
|
//----------------------------------------------------------
|
|
// The destructor will release the pointer that it contains
|
|
// if it has a valid pointer.
|
|
//----------------------------------------------------------
|
|
virtual ~CFCReleaser()
|
|
{
|
|
reset();
|
|
}
|
|
|
|
//----------------------------------------------------------
|
|
// Assignment operator.
|
|
//
|
|
// Note that assigning one CFCReleaser to another will
|
|
// not transfer ownership of the contained pointer, but it
|
|
// will bump its reference count. This is where this class
|
|
// differs from std::auto_ptr.
|
|
//----------------------------------------------------------
|
|
CFCReleaser&
|
|
operator= (const CFCReleaser<T>& rhs)
|
|
{
|
|
if (this != &rhs)
|
|
{
|
|
// Replace our owned pointer with the new one
|
|
reset(rhs.get());
|
|
// Retain the current pointer that we own
|
|
if (get())
|
|
::CFRetain(get());
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
//----------------------------------------------------------
|
|
// Get the address of the contained type in case it needs
|
|
// to be passed to a function that will fill in a pointer
|
|
// value. The function currently will assert if _ptr is not
|
|
// NULL because the only time this method should be used is
|
|
// if another function will modify the contents, and we
|
|
// could leak a pointer if this is not NULL. If the
|
|
// assertion fires, check the offending code, or call
|
|
// reset() prior to using the "ptr_address()" member to make
|
|
// sure any owned objects has CFRelease called on it.
|
|
// I had to add the "enforce_null" bool here because some
|
|
// API's require the pointer address even though they don't change it.
|
|
//----------------------------------------------------------
|
|
T*
|
|
ptr_address(bool enforce_null = true)
|
|
{
|
|
if (enforce_null)
|
|
assert (_ptr == NULL);
|
|
return &_ptr;
|
|
}
|
|
|
|
//----------------------------------------------------------
|
|
// Access the pointer itself
|
|
//----------------------------------------------------------
|
|
T
|
|
get()
|
|
{
|
|
return _ptr;
|
|
}
|
|
|
|
const T
|
|
get() const
|
|
{
|
|
return _ptr;
|
|
}
|
|
|
|
|
|
//----------------------------------------------------------
|
|
// Set a new value for the pointer and CFRelease our old
|
|
// value if we had a valid one.
|
|
//----------------------------------------------------------
|
|
void
|
|
reset(T ptr = NULL)
|
|
{
|
|
if ((_ptr != NULL) && (ptr != _ptr))
|
|
::CFRelease(_ptr);
|
|
_ptr = ptr;
|
|
}
|
|
|
|
//----------------------------------------------------------
|
|
// Release ownership without calling CFRelease. This class
|
|
// is designed to mimic std::auto_ptr<T>, so the release
|
|
// method releases ownership of the contained pointer
|
|
// and does NOT call CFRelease.
|
|
//----------------------------------------------------------
|
|
T
|
|
release()
|
|
{
|
|
T tmp = _ptr;
|
|
_ptr = NULL;
|
|
return tmp;
|
|
}
|
|
|
|
private:
|
|
T _ptr;
|
|
};
|
|
|
|
#endif // #ifdef __cplusplus
|
|
#endif // #ifndef CoreFoundationCPP_CFReleaser_h_
|
|
|