mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-12-11 08:13:35 +00:00
b823558f4c
The existing infallible Create() is also changed to use operator new(size_t) to match the non-array delete when the ref count drops to zero. MozReview-Commit-ID: HZGjSKnV8I2 --HG-- extra : rebase_source : f327facf481fee7d2e543e3160e3aebe05885cca
105 lines
2.9 KiB
C++
105 lines
2.9 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* 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/. */
|
|
|
|
#ifndef MOZILLA_SHAREDBUFFER_H_
|
|
#define MOZILLA_SHAREDBUFFER_H_
|
|
|
|
#include "mozilla/CheckedInt.h"
|
|
#include "mozilla/mozalloc.h"
|
|
#include "mozilla/MemoryReporting.h"
|
|
#include "nsISupportsImpl.h"
|
|
|
|
namespace mozilla {
|
|
|
|
class AudioBlockBuffer;
|
|
class ThreadSharedFloatArrayBufferList;
|
|
|
|
/**
|
|
* Base class for objects with a thread-safe refcount and a virtual
|
|
* destructor.
|
|
*/
|
|
class ThreadSharedObject {
|
|
public:
|
|
NS_INLINE_DECL_THREADSAFE_REFCOUNTING(ThreadSharedObject)
|
|
|
|
bool IsShared() { return mRefCnt.get() > 1; }
|
|
|
|
virtual AudioBlockBuffer* AsAudioBlockBuffer() { return nullptr; };
|
|
virtual ThreadSharedFloatArrayBufferList* AsThreadSharedFloatArrayBufferList()
|
|
{
|
|
return nullptr;
|
|
};
|
|
|
|
virtual size_t SizeOfExcludingThis(MallocSizeOf aMallocSizeOf) const
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
virtual size_t SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const
|
|
{
|
|
return aMallocSizeOf(this) + SizeOfExcludingThis(aMallocSizeOf);
|
|
}
|
|
protected:
|
|
// Protected destructor, to discourage deletion outside of Release():
|
|
virtual ~ThreadSharedObject() {}
|
|
};
|
|
|
|
/**
|
|
* Heap-allocated chunk of arbitrary data with threadsafe refcounting.
|
|
* Typically you would allocate one of these, fill it in, and then treat it as
|
|
* immutable while it's shared.
|
|
* This only guarantees 4-byte alignment of the data. For alignment we simply
|
|
* assume that the memory from malloc is at least 4-byte aligned and the
|
|
* refcount's size is large enough that SharedBuffer's size is divisible by 4.
|
|
*/
|
|
class SharedBuffer : public ThreadSharedObject {
|
|
public:
|
|
void* Data() { return this + 1; }
|
|
|
|
static already_AddRefed<SharedBuffer> Create(size_t aSize, const fallible_t&)
|
|
{
|
|
void* m = operator new(AllocSize(aSize), fallible);
|
|
if (!m) {
|
|
return nullptr;
|
|
}
|
|
RefPtr<SharedBuffer> p = new (m) SharedBuffer();
|
|
return p.forget();
|
|
}
|
|
|
|
static already_AddRefed<SharedBuffer> Create(size_t aSize)
|
|
{
|
|
void* m = operator new(AllocSize(aSize));
|
|
RefPtr<SharedBuffer> p = new (m) SharedBuffer();
|
|
return p.forget();
|
|
}
|
|
|
|
size_t SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const override
|
|
{
|
|
return aMallocSizeOf(this) + SizeOfExcludingThis(aMallocSizeOf);
|
|
}
|
|
|
|
private:
|
|
static size_t
|
|
AllocSize(size_t aDataSize)
|
|
{
|
|
CheckedInt<size_t> size = sizeof(SharedBuffer);
|
|
size += aDataSize;
|
|
if (!size.isValid()) {
|
|
MOZ_CRASH();
|
|
}
|
|
return size.value();
|
|
}
|
|
|
|
SharedBuffer()
|
|
{
|
|
NS_ASSERTION((reinterpret_cast<char*>(this + 1) - reinterpret_cast<char*>(this)) % 4 == 0,
|
|
"SharedBuffers should be at least 4-byte aligned");
|
|
}
|
|
};
|
|
|
|
} // namespace mozilla
|
|
|
|
#endif /* MOZILLA_SHAREDBUFFER_H_ */
|