mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-29 07:42:04 +00:00
2824b29025
This patch was mostly generated by running the following scripts on the codebase, with some manual changes made afterwards: # static_assert.sh #!/bin/bash # Command to convert an NSPR integer type to the equivalent standard integer type function convert() { echo "Converting $1 to $2..." find . ! -wholename "*nsprpub*" \ ! -wholename "*security/nss*" \ ! -wholename "*/.hg*" \ ! -wholename "obj-ff-dbg*" \ ! -name nsXPCOMCID.h \ ! -name prtypes.h \ -type f \ \( -iname "*.cpp" \ -o -iname "*.h" \ -o -iname "*.cc" \ -o -iname "*.mm" \) | \ xargs -n 1 `dirname $0`/assert_replacer.py #sed -i -e "s/\b$1\b/$2/g" } convert MOZ_STATIC_ASSERT static_assert hg rev --no-backup mfbt/Assertions.h \ media/webrtc/signaling/src/sipcc/core/includes/ccapi.h \ modules/libmar/src/mar_private.h \ modules/libmar/src/mar.h # assert_replacer.py #!/usr/bin/python import sys import re pattern = re.compile(r"\bMOZ_STATIC_ASSERT\b") def replaceInPlace(fname): print fname f = open(fname, "rw+") lines = f.readlines() for i in range(0, len(lines)): while True: index = re.search(pattern, lines[i]) if index != None: index = index.start() lines[i] = lines[i][0:index] + "static_assert" + lines[i][index+len("MOZ_STATIC_ASSERT"):] for j in range(i + 1, len(lines)): if lines[j].find(" ", index) == index: lines[j] = lines[j][0:index] + lines[j][index+4:] else: break else: break f.seek(0, 0) f.truncate() f.write("".join(lines)) f.close() argc = len(sys.argv) for i in range(1, argc): replaceInPlace(sys.argv[i]) --HG-- extra : rebase_source : 4b4a4047d82f2c205b9fad8d56dfc3f1afc0b045
192 lines
5.3 KiB
C++
192 lines
5.3 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
/* Weak pointer functionality, implemented as a mixin for use with any class. */
|
|
|
|
/**
|
|
* SupportsWeakPtr lets you have a pointer to an object 'Foo' without affecting
|
|
* its lifetime. It works by creating a single shared reference counted object
|
|
* (WeakReference) that each WeakPtr will access 'Foo' through. This lets 'Foo'
|
|
* clear the pointer in the WeakReference without having to know about all of
|
|
* the WeakPtrs to it and allows the WeakReference to live beyond the lifetime
|
|
* of 'Foo'.
|
|
*
|
|
* AtomicSupportsWeakPtr can be used for a variant with an atomically updated
|
|
* reference counter.
|
|
*
|
|
* The overhead of WeakPtr is that accesses to 'Foo' becomes an additional
|
|
* dereference, and an additional heap allocated pointer sized object shared
|
|
* between all of the WeakPtrs.
|
|
*
|
|
* Example of usage:
|
|
*
|
|
* // To have a class C support weak pointers, inherit from SupportsWeakPtr<C>.
|
|
* class C : public SupportsWeakPtr<C>
|
|
* {
|
|
* public:
|
|
* int num;
|
|
* void act();
|
|
* };
|
|
*
|
|
* C* ptr = new C();
|
|
*
|
|
* // Get weak pointers to ptr. The first time asWeakPtr is called
|
|
* // a reference counted WeakReference object is created that
|
|
* // can live beyond the lifetime of 'ptr'. The WeakReference
|
|
* // object will be notified of 'ptr's destruction.
|
|
* WeakPtr<C> weak = ptr->asWeakPtr();
|
|
* WeakPtr<C> other = ptr->asWeakPtr();
|
|
*
|
|
* // Test a weak pointer for validity before using it.
|
|
* if (weak) {
|
|
* weak->num = 17;
|
|
* weak->act();
|
|
* }
|
|
*
|
|
* // Destroying the underlying object clears weak pointers to it.
|
|
* delete ptr;
|
|
*
|
|
* MOZ_ASSERT(!weak, "Deleting |ptr| clears weak pointers to it.");
|
|
* MOZ_ASSERT(!other, "Deleting |ptr| clears all weak pointers to it.");
|
|
*
|
|
* WeakPtr is typesafe and may be used with any class. It is not required that
|
|
* the class be reference-counted or allocated in any particular way.
|
|
*
|
|
* The API was loosely inspired by Chromium's weak_ptr.h:
|
|
* http://src.chromium.org/svn/trunk/src/base/memory/weak_ptr.h
|
|
*/
|
|
|
|
#ifndef mozilla_WeakPtr_h
|
|
#define mozilla_WeakPtr_h
|
|
|
|
#include "mozilla/Assertions.h"
|
|
#include "mozilla/Atomics.h"
|
|
#include "mozilla/NullPtr.h"
|
|
#include "mozilla/RefPtr.h"
|
|
#include "mozilla/TypeTraits.h"
|
|
|
|
namespace mozilla {
|
|
|
|
template <typename T, class WeakReference> class WeakPtrBase;
|
|
template <typename T, class WeakReference> class SupportsWeakPtrBase;
|
|
|
|
namespace detail {
|
|
|
|
// This can live beyond the lifetime of the class derived from SupportsWeakPtrBase.
|
|
template<class T, RefCountAtomicity Atomicity>
|
|
class WeakReference : public RefCounted<WeakReference<T, Atomicity>, Atomicity>
|
|
{
|
|
public:
|
|
explicit WeakReference(T* p) : ptr(p) {}
|
|
T* get() const {
|
|
return ptr;
|
|
}
|
|
|
|
private:
|
|
friend class WeakPtrBase<T, WeakReference>;
|
|
friend class SupportsWeakPtrBase<T, WeakReference>;
|
|
void detach() {
|
|
ptr = nullptr;
|
|
}
|
|
T* ptr;
|
|
};
|
|
|
|
} // namespace detail
|
|
|
|
template <typename T, class WeakReference>
|
|
class SupportsWeakPtrBase
|
|
{
|
|
public:
|
|
WeakPtrBase<T, WeakReference> asWeakPtr() {
|
|
if (!weakRef)
|
|
weakRef = new WeakReference(static_cast<T*>(this));
|
|
return WeakPtrBase<T, WeakReference>(weakRef);
|
|
}
|
|
|
|
protected:
|
|
~SupportsWeakPtrBase() {
|
|
static_assert(IsBaseOf<SupportsWeakPtrBase<T, WeakReference>, T>::value,
|
|
"T must derive from SupportsWeakPtrBase<T, WeakReference>");
|
|
if (weakRef)
|
|
weakRef->detach();
|
|
}
|
|
|
|
private:
|
|
friend class WeakPtrBase<T, WeakReference>;
|
|
|
|
RefPtr<WeakReference> weakRef;
|
|
};
|
|
|
|
template <typename T>
|
|
class SupportsWeakPtr
|
|
: public SupportsWeakPtrBase<T, detail::WeakReference<T, detail::NonAtomicRefCount> >
|
|
{
|
|
};
|
|
|
|
template <typename T>
|
|
class AtomicSupportsWeakPtr
|
|
: public SupportsWeakPtrBase<T, detail::WeakReference<T, detail::AtomicRefCount> >
|
|
{
|
|
};
|
|
|
|
namespace detail {
|
|
|
|
template <typename T>
|
|
struct WeakReferenceCount
|
|
{
|
|
static const RefCountAtomicity atomicity =
|
|
IsBaseOf<AtomicSupportsWeakPtr<T>, T>::value
|
|
? AtomicRefCount
|
|
: NonAtomicRefCount;
|
|
};
|
|
|
|
}
|
|
|
|
template <typename T, class WeakReference>
|
|
class WeakPtrBase
|
|
{
|
|
public:
|
|
WeakPtrBase(const WeakPtrBase<T, WeakReference>& o) : ref(o.ref) {}
|
|
// Ensure that ref is dereferenceable in the uninitialized state
|
|
WeakPtrBase() : ref(new WeakReference(nullptr)) {}
|
|
|
|
operator T*() const {
|
|
return ref->get();
|
|
}
|
|
T& operator*() const {
|
|
return *ref->get();
|
|
}
|
|
|
|
T* operator->() const {
|
|
return ref->get();
|
|
}
|
|
|
|
T* get() const {
|
|
return ref->get();
|
|
}
|
|
|
|
private:
|
|
friend class SupportsWeakPtrBase<T, WeakReference>;
|
|
|
|
explicit WeakPtrBase(const RefPtr<WeakReference> &o) : ref(o) {}
|
|
|
|
RefPtr<WeakReference> ref;
|
|
};
|
|
|
|
template <typename T>
|
|
class WeakPtr : public WeakPtrBase<T, detail::WeakReference<T, detail::WeakReferenceCount<T>::atomicity> >
|
|
{
|
|
typedef WeakPtrBase<T, detail::WeakReference<T, detail::WeakReferenceCount<T>::atomicity> > Base;
|
|
public:
|
|
WeakPtr(const WeakPtr<T>& o) : Base(o) {}
|
|
WeakPtr(const Base& o) : Base(o) {}
|
|
WeakPtr() {}
|
|
};
|
|
|
|
} // namespace mozilla
|
|
|
|
#endif /* mozilla_WeakPtr_h */
|