blob: 6c48d52f17172ee8c4cfaefad3e9051ccb420933 [file] [log] [blame]
jbauch13041cf2016-02-25 06:16:52 -08001/*
2 * Copyright 2016 The WebRTC Project Authors. All rights reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "rtc_base/copyonwritebuffer.h"
jbauch13041cf2016-02-25 06:16:52 -080012
13namespace rtc {
14
15CopyOnWriteBuffer::CopyOnWriteBuffer() {
16 RTC_DCHECK(IsConsistent());
17}
18
19CopyOnWriteBuffer::CopyOnWriteBuffer(const CopyOnWriteBuffer& buf)
Yves Gerey665174f2018-06-19 15:03:05 +020020 : buffer_(buf.buffer_) {}
jbauch13041cf2016-02-25 06:16:52 -080021
danilchapd8a9c532016-07-30 12:39:26 -070022CopyOnWriteBuffer::CopyOnWriteBuffer(CopyOnWriteBuffer&& buf)
Yves Gerey665174f2018-06-19 15:03:05 +020023 : buffer_(std::move(buf.buffer_)) {}
jbauch13041cf2016-02-25 06:16:52 -080024
Jeroen de Borst4f6d2332018-07-18 11:25:12 -070025CopyOnWriteBuffer::CopyOnWriteBuffer(const std::string& s)
26 : CopyOnWriteBuffer(s.data(), s.length()) {}
27
jbauch13041cf2016-02-25 06:16:52 -080028CopyOnWriteBuffer::CopyOnWriteBuffer(size_t size)
29 : buffer_(size > 0 ? new RefCountedObject<Buffer>(size) : nullptr) {
30 RTC_DCHECK(IsConsistent());
31}
32
33CopyOnWriteBuffer::CopyOnWriteBuffer(size_t size, size_t capacity)
34 : buffer_(size > 0 || capacity > 0
Yves Gerey665174f2018-06-19 15:03:05 +020035 ? new RefCountedObject<Buffer>(size, capacity)
36 : nullptr) {
jbauch13041cf2016-02-25 06:16:52 -080037 RTC_DCHECK(IsConsistent());
38}
39
40CopyOnWriteBuffer::~CopyOnWriteBuffer() = default;
41
Danil Chapovalov3626d7e2016-09-14 17:14:30 +020042bool CopyOnWriteBuffer::operator==(const CopyOnWriteBuffer& buf) const {
43 // Must either use the same buffer internally or have the same contents.
44 RTC_DCHECK(IsConsistent());
45 RTC_DCHECK(buf.IsConsistent());
46 return buffer_.get() == buf.buffer_.get() ||
Yves Gerey665174f2018-06-19 15:03:05 +020047 (buffer_.get() && buf.buffer_.get() &&
48 *buffer_.get() == *buf.buffer_.get());
Danil Chapovalov3626d7e2016-09-14 17:14:30 +020049}
50
51void CopyOnWriteBuffer::SetSize(size_t size) {
52 RTC_DCHECK(IsConsistent());
53 if (!buffer_) {
54 if (size > 0) {
55 buffer_ = new RefCountedObject<Buffer>(size);
56 }
57 RTC_DCHECK(IsConsistent());
58 return;
59 }
60
61 // Clone data if referenced.
62 if (!buffer_->HasOneRef()) {
Yves Gerey665174f2018-06-19 15:03:05 +020063 buffer_ = new RefCountedObject<Buffer>(buffer_->data(),
64 std::min(buffer_->size(), size),
65 std::max(buffer_->capacity(), size));
Danil Chapovalov3626d7e2016-09-14 17:14:30 +020066 }
67 buffer_->SetSize(size);
68 RTC_DCHECK(IsConsistent());
69}
70
71void CopyOnWriteBuffer::EnsureCapacity(size_t capacity) {
72 RTC_DCHECK(IsConsistent());
73 if (!buffer_) {
74 if (capacity > 0) {
75 buffer_ = new RefCountedObject<Buffer>(0, capacity);
76 }
77 RTC_DCHECK(IsConsistent());
78 return;
79 } else if (capacity <= buffer_->capacity()) {
80 return;
81 }
82
83 CloneDataIfReferenced(std::max(buffer_->capacity(), capacity));
84 buffer_->EnsureCapacity(capacity);
85 RTC_DCHECK(IsConsistent());
86}
87
88void CopyOnWriteBuffer::Clear() {
89 if (!buffer_)
90 return;
91
92 if (buffer_->HasOneRef()) {
93 buffer_->Clear();
94 } else {
95 buffer_ = new RefCountedObject<Buffer>(0, buffer_->capacity());
96 }
97 RTC_DCHECK(IsConsistent());
98}
99
100void CopyOnWriteBuffer::CloneDataIfReferenced(size_t new_capacity) {
101 if (buffer_->HasOneRef()) {
102 return;
103 }
104
105 buffer_ = new RefCountedObject<Buffer>(buffer_->data(), buffer_->size(),
Yves Gerey665174f2018-06-19 15:03:05 +0200106 new_capacity);
Danil Chapovalov3626d7e2016-09-14 17:14:30 +0200107 RTC_DCHECK(IsConsistent());
108}
109
jbauch13041cf2016-02-25 06:16:52 -0800110} // namespace rtc