|  | /* | 
|  | *  Copyright 2016 The WebRTC Project Authors. All rights reserved. | 
|  | * | 
|  | *  Use of this source code is governed by a BSD-style license | 
|  | *  that can be found in the LICENSE file in the root of the source | 
|  | *  tree. An additional intellectual property rights grant can be found | 
|  | *  in the file PATENTS.  All contributing project authors may | 
|  | *  be found in the AUTHORS file in the root of the source tree. | 
|  | */ | 
|  |  | 
|  | #ifndef RTC_BASE_COPYONWRITEBUFFER_H_ | 
|  | #define RTC_BASE_COPYONWRITEBUFFER_H_ | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <utility> | 
|  |  | 
|  | #include "rtc_base/buffer.h" | 
|  | #include "rtc_base/checks.h" | 
|  | #include "rtc_base/refcount.h" | 
|  | #include "rtc_base/refcountedobject.h" | 
|  | #include "rtc_base/scoped_ref_ptr.h" | 
|  |  | 
|  | namespace rtc { | 
|  |  | 
|  | class CopyOnWriteBuffer { | 
|  | public: | 
|  | // An empty buffer. | 
|  | CopyOnWriteBuffer(); | 
|  | // Copy size and contents of an existing buffer. | 
|  | CopyOnWriteBuffer(const CopyOnWriteBuffer& buf); | 
|  | // Move contents from an existing buffer. | 
|  | CopyOnWriteBuffer(CopyOnWriteBuffer&& buf); | 
|  |  | 
|  | // Construct a buffer with the specified number of uninitialized bytes. | 
|  | explicit CopyOnWriteBuffer(size_t size); | 
|  | CopyOnWriteBuffer(size_t size, size_t capacity); | 
|  |  | 
|  | // Construct a buffer and copy the specified number of bytes into it. The | 
|  | // source array may be (const) uint8_t*, int8_t*, or char*. | 
|  | template <typename T, | 
|  | typename std::enable_if< | 
|  | internal::BufferCompat<uint8_t, T>::value>::type* = nullptr> | 
|  | CopyOnWriteBuffer(const T* data, size_t size) | 
|  | : CopyOnWriteBuffer(data, size, size) {} | 
|  | template <typename T, | 
|  | typename std::enable_if< | 
|  | internal::BufferCompat<uint8_t, T>::value>::type* = nullptr> | 
|  | CopyOnWriteBuffer(const T* data, size_t size, size_t capacity) | 
|  | : CopyOnWriteBuffer(size, capacity) { | 
|  | if (buffer_) { | 
|  | std::memcpy(buffer_->data(), data, size); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Construct a buffer from the contents of an array. | 
|  | template <typename T, | 
|  | size_t N, | 
|  | typename std::enable_if< | 
|  | internal::BufferCompat<uint8_t, T>::value>::type* = nullptr> | 
|  | CopyOnWriteBuffer(const T (&array)[N])  // NOLINT: runtime/explicit | 
|  | : CopyOnWriteBuffer(array, N) {} | 
|  |  | 
|  | ~CopyOnWriteBuffer(); | 
|  |  | 
|  | // Get a pointer to the data. Just .data() will give you a (const) uint8_t*, | 
|  | // but you may also use .data<int8_t>() and .data<char>(). | 
|  | template <typename T = uint8_t, | 
|  | typename std::enable_if< | 
|  | internal::BufferCompat<uint8_t, T>::value>::type* = nullptr> | 
|  | const T* data() const { | 
|  | return cdata<T>(); | 
|  | } | 
|  |  | 
|  | // Get writable pointer to the data. This will create a copy of the underlying | 
|  | // data if it is shared with other buffers. | 
|  | template <typename T = uint8_t, | 
|  | typename std::enable_if< | 
|  | internal::BufferCompat<uint8_t, T>::value>::type* = nullptr> | 
|  | T* data() { | 
|  | RTC_DCHECK(IsConsistent()); | 
|  | if (!buffer_) { | 
|  | return nullptr; | 
|  | } | 
|  | CloneDataIfReferenced(buffer_->capacity()); | 
|  | return buffer_->data<T>(); | 
|  | } | 
|  |  | 
|  | // Get const pointer to the data. This will not create a copy of the | 
|  | // underlying data if it is shared with other buffers. | 
|  | template <typename T = uint8_t, | 
|  | typename std::enable_if< | 
|  | internal::BufferCompat<uint8_t, T>::value>::type* = nullptr> | 
|  | const T* cdata() const { | 
|  | RTC_DCHECK(IsConsistent()); | 
|  | if (!buffer_) { | 
|  | return nullptr; | 
|  | } | 
|  | return buffer_->data<T>(); | 
|  | } | 
|  |  | 
|  | size_t size() const { | 
|  | RTC_DCHECK(IsConsistent()); | 
|  | return buffer_ ? buffer_->size() : 0; | 
|  | } | 
|  |  | 
|  | size_t capacity() const { | 
|  | RTC_DCHECK(IsConsistent()); | 
|  | return buffer_ ? buffer_->capacity() : 0; | 
|  | } | 
|  |  | 
|  | CopyOnWriteBuffer& operator=(const CopyOnWriteBuffer& buf) { | 
|  | RTC_DCHECK(IsConsistent()); | 
|  | RTC_DCHECK(buf.IsConsistent()); | 
|  | if (&buf != this) { | 
|  | buffer_ = buf.buffer_; | 
|  | } | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | CopyOnWriteBuffer& operator=(CopyOnWriteBuffer&& buf) { | 
|  | RTC_DCHECK(IsConsistent()); | 
|  | RTC_DCHECK(buf.IsConsistent()); | 
|  | buffer_ = std::move(buf.buffer_); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | bool operator==(const CopyOnWriteBuffer& buf) const; | 
|  |  | 
|  | bool operator!=(const CopyOnWriteBuffer& buf) const { | 
|  | return !(*this == buf); | 
|  | } | 
|  |  | 
|  | uint8_t& operator[](size_t index) { | 
|  | RTC_DCHECK_LT(index, size()); | 
|  | return data()[index]; | 
|  | } | 
|  |  | 
|  | uint8_t operator[](size_t index) const { | 
|  | RTC_DCHECK_LT(index, size()); | 
|  | return cdata()[index]; | 
|  | } | 
|  |  | 
|  | // Replace the contents of the buffer. Accepts the same types as the | 
|  | // constructors. | 
|  | template <typename T, | 
|  | typename std::enable_if< | 
|  | internal::BufferCompat<uint8_t, T>::value>::type* = nullptr> | 
|  | void SetData(const T* data, size_t size) { | 
|  | RTC_DCHECK(IsConsistent()); | 
|  | if (!buffer_) { | 
|  | buffer_ = size > 0 ? new RefCountedObject<Buffer>(data, size) : nullptr; | 
|  | } else if (!buffer_->HasOneRef()) { | 
|  | buffer_ = new RefCountedObject<Buffer>(data, size, buffer_->capacity()); | 
|  | } else { | 
|  | buffer_->SetData(data, size); | 
|  | } | 
|  | RTC_DCHECK(IsConsistent()); | 
|  | } | 
|  |  | 
|  | template <typename T, | 
|  | size_t N, | 
|  | typename std::enable_if< | 
|  | internal::BufferCompat<uint8_t, T>::value>::type* = nullptr> | 
|  | void SetData(const T (&array)[N]) { | 
|  | SetData(array, N); | 
|  | } | 
|  |  | 
|  | void SetData(const CopyOnWriteBuffer& buf) { | 
|  | RTC_DCHECK(IsConsistent()); | 
|  | RTC_DCHECK(buf.IsConsistent()); | 
|  | if (&buf != this) { | 
|  | buffer_ = buf.buffer_; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Append data to the buffer. Accepts the same types as the constructors. | 
|  | template <typename T, | 
|  | typename std::enable_if< | 
|  | internal::BufferCompat<uint8_t, T>::value>::type* = nullptr> | 
|  | void AppendData(const T* data, size_t size) { | 
|  | RTC_DCHECK(IsConsistent()); | 
|  | if (!buffer_) { | 
|  | buffer_ = new RefCountedObject<Buffer>(data, size); | 
|  | RTC_DCHECK(IsConsistent()); | 
|  | return; | 
|  | } | 
|  |  | 
|  | CloneDataIfReferenced(std::max(buffer_->capacity(), | 
|  | buffer_->size() + size)); | 
|  | buffer_->AppendData(data, size); | 
|  | RTC_DCHECK(IsConsistent()); | 
|  | } | 
|  |  | 
|  | template <typename T, | 
|  | size_t N, | 
|  | typename std::enable_if< | 
|  | internal::BufferCompat<uint8_t, T>::value>::type* = nullptr> | 
|  | void AppendData(const T (&array)[N]) { | 
|  | AppendData(array, N); | 
|  | } | 
|  |  | 
|  | void AppendData(const CopyOnWriteBuffer& buf) { | 
|  | AppendData(buf.data(), buf.size()); | 
|  | } | 
|  |  | 
|  | // Sets the size of the buffer. If the new size is smaller than the old, the | 
|  | // buffer contents will be kept but truncated; if the new size is greater, | 
|  | // the existing contents will be kept and the new space will be | 
|  | // uninitialized. | 
|  | void SetSize(size_t size); | 
|  |  | 
|  | // Ensure that the buffer size can be increased to at least capacity without | 
|  | // further reallocation. (Of course, this operation might need to reallocate | 
|  | // the buffer.) | 
|  | void EnsureCapacity(size_t capacity); | 
|  |  | 
|  | // Resets the buffer to zero size without altering capacity. Works even if the | 
|  | // buffer has been moved from. | 
|  | void Clear(); | 
|  |  | 
|  | // Swaps two buffers. | 
|  | friend void swap(CopyOnWriteBuffer& a, CopyOnWriteBuffer& b) { | 
|  | std::swap(a.buffer_, b.buffer_); | 
|  | } | 
|  |  | 
|  | private: | 
|  | // Create a copy of the underlying data if it is referenced from other Buffer | 
|  | // objects. | 
|  | void CloneDataIfReferenced(size_t new_capacity); | 
|  |  | 
|  | // Pre- and postcondition of all methods. | 
|  | bool IsConsistent() const { | 
|  | return (!buffer_ || buffer_->capacity() > 0); | 
|  | } | 
|  |  | 
|  | // buffer_ is either null, or points to an rtc::Buffer with capacity > 0. | 
|  | scoped_refptr<RefCountedObject<Buffer>> buffer_; | 
|  | }; | 
|  |  | 
|  | }  // namespace rtc | 
|  |  | 
|  | #endif  // RTC_BASE_COPYONWRITEBUFFER_H_ |