• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "media/base/seekable_buffer.h"
6 
7 #include <algorithm>
8 
9 #include "base/logging.h"
10 #include "media/base/data_buffer.h"
11 
12 namespace media {
13 
SeekableBuffer(int backward_capacity,int forward_capacity)14 SeekableBuffer::SeekableBuffer(int backward_capacity, int forward_capacity)
15     : current_buffer_offset_(0),
16       backward_capacity_(backward_capacity),
17       backward_bytes_(0),
18       forward_capacity_(forward_capacity),
19       forward_bytes_(0),
20       current_time_(kNoTimestamp()) {
21   current_buffer_ = buffers_.begin();
22 }
23 
~SeekableBuffer()24 SeekableBuffer::~SeekableBuffer() {
25 }
26 
Clear()27 void SeekableBuffer::Clear() {
28   buffers_.clear();
29   current_buffer_ = buffers_.begin();
30   current_buffer_offset_ = 0;
31   backward_bytes_ = 0;
32   forward_bytes_ = 0;
33   current_time_ = kNoTimestamp();
34 }
35 
Read(uint8 * data,int size)36 int SeekableBuffer::Read(uint8* data, int size) {
37   DCHECK(data);
38   return InternalRead(data, size, true, 0);
39 }
40 
Peek(uint8 * data,int size,int forward_offset)41 int SeekableBuffer::Peek(uint8* data, int size, int forward_offset) {
42   DCHECK(data);
43   return InternalRead(data, size, false, forward_offset);
44 }
45 
GetCurrentChunk(const uint8 ** data,int * size) const46 bool SeekableBuffer::GetCurrentChunk(const uint8** data, int* size) const {
47   BufferQueue::iterator current_buffer = current_buffer_;
48   int current_buffer_offset = current_buffer_offset_;
49   // Advance position if we are in the end of the current buffer.
50   while (current_buffer != buffers_.end() &&
51          current_buffer_offset >= (*current_buffer)->data_size()) {
52     ++current_buffer;
53     current_buffer_offset = 0;
54   }
55   if (current_buffer == buffers_.end())
56     return false;
57   *data = (*current_buffer)->data() + current_buffer_offset;
58   *size = (*current_buffer)->data_size() - current_buffer_offset;
59   return true;
60 }
61 
Append(const scoped_refptr<DataBuffer> & buffer_in)62 bool SeekableBuffer::Append(const scoped_refptr<DataBuffer>& buffer_in) {
63   if (buffers_.empty() && buffer_in->timestamp() != kNoTimestamp()) {
64     current_time_ = buffer_in->timestamp();
65   }
66 
67   // Since the forward capacity is only used to check the criteria for buffer
68   // full, we always append data to the buffer.
69   buffers_.push_back(buffer_in);
70 
71   // After we have written the first buffer, update |current_buffer_| to point
72   // to it.
73   if (current_buffer_ == buffers_.end()) {
74     DCHECK_EQ(0, forward_bytes_);
75     current_buffer_ = buffers_.begin();
76   }
77 
78   // Update the |forward_bytes_| counter since we have more bytes.
79   forward_bytes_ += buffer_in->data_size();
80 
81   // Advise the user to stop append if the amount of forward bytes exceeds
82   // the forward capacity. A false return value means the user should stop
83   // appending more data to this buffer.
84   if (forward_bytes_ >= forward_capacity_)
85     return false;
86   return true;
87 }
88 
Append(const uint8 * data,int size)89 bool SeekableBuffer::Append(const uint8* data, int size) {
90   if (size > 0) {
91     scoped_refptr<DataBuffer> data_buffer = DataBuffer::CopyFrom(data, size);
92     return Append(data_buffer);
93   } else {
94     // Return true if we have forward capacity.
95     return forward_bytes_ < forward_capacity_;
96   }
97 }
98 
Seek(int32 offset)99 bool SeekableBuffer::Seek(int32 offset) {
100   if (offset > 0)
101     return SeekForward(offset);
102   else if (offset < 0)
103     return SeekBackward(-offset);
104   return true;
105 }
106 
SeekForward(int size)107 bool SeekableBuffer::SeekForward(int size) {
108   // Perform seeking forward only if we have enough bytes in the queue.
109   if (size > forward_bytes_)
110     return false;
111 
112   // Do a read of |size| bytes.
113   int taken = InternalRead(NULL, size, true, 0);
114   DCHECK_EQ(taken, size);
115   return true;
116 }
117 
SeekBackward(int size)118 bool SeekableBuffer::SeekBackward(int size) {
119   if (size > backward_bytes_)
120     return false;
121   // Record the number of bytes taken.
122   int taken = 0;
123   // Loop until we taken enough bytes and rewind by the desired |size|.
124   while (taken < size) {
125     // |current_buffer_| can never be invalid when we are in this loop. It can
126     // only be invalid before any data is appended. The invalid case should be
127     // handled by checks before we enter this loop.
128     DCHECK(current_buffer_ != buffers_.end());
129 
130     // We try to consume at most |size| bytes in the backward direction. We also
131     // have to account for the offset we are in the current buffer, so take the
132     // minimum between the two to determine the amount of bytes to take from the
133     // current buffer.
134     int consumed = std::min(size - taken, current_buffer_offset_);
135 
136     // Decreases the offset in the current buffer since we are rewinding.
137     current_buffer_offset_ -= consumed;
138 
139     // Increase the amount of bytes taken in the backward direction. This
140     // determines when to stop the loop.
141     taken += consumed;
142 
143     // Forward bytes increases and backward bytes decreases by the amount
144     // consumed in the current buffer.
145     forward_bytes_ += consumed;
146     backward_bytes_ -= consumed;
147     DCHECK_GE(backward_bytes_, 0);
148 
149     // The current buffer pointed by current iterator has been consumed. Move
150     // the iterator backward so it points to the previous buffer.
151     if (current_buffer_offset_ == 0) {
152       if (current_buffer_ == buffers_.begin())
153         break;
154       // Move the iterator backward.
155       --current_buffer_;
156       // Set the offset into the current buffer to be the buffer size as we
157       // are preparing for rewind for next iteration.
158       current_buffer_offset_ = (*current_buffer_)->data_size();
159     }
160   }
161 
162   UpdateCurrentTime(current_buffer_, current_buffer_offset_);
163 
164   DCHECK_EQ(taken, size);
165   return true;
166 }
167 
EvictBackwardBuffers()168 void SeekableBuffer::EvictBackwardBuffers() {
169   // Advances the iterator until we hit the current pointer.
170   while (backward_bytes_ > backward_capacity_) {
171     BufferQueue::iterator i = buffers_.begin();
172     if (i == current_buffer_)
173       break;
174     scoped_refptr<DataBuffer> buffer = *i;
175     backward_bytes_ -= buffer->data_size();
176     DCHECK_GE(backward_bytes_, 0);
177 
178     buffers_.erase(i);
179   }
180 }
181 
InternalRead(uint8 * data,int size,bool advance_position,int forward_offset)182 int SeekableBuffer::InternalRead(uint8* data, int size,
183                                  bool advance_position,
184                                  int forward_offset) {
185   // Counts how many bytes are actually read from the buffer queue.
186   int taken = 0;
187 
188   BufferQueue::iterator current_buffer = current_buffer_;
189   int current_buffer_offset = current_buffer_offset_;
190 
191   int bytes_to_skip = forward_offset;
192   while (taken < size) {
193     // |current_buffer| is valid since the first time this buffer is appended
194     // with data.
195     if (current_buffer == buffers_.end())
196       break;
197 
198     scoped_refptr<DataBuffer> buffer = *current_buffer;
199 
200     int remaining_bytes_in_buffer =
201         buffer->data_size() - current_buffer_offset;
202 
203     if (bytes_to_skip == 0) {
204       // Find the right amount to copy from the current buffer referenced by
205       // |buffer|. We shall copy no more than |size| bytes in total and each
206       // single step copied no more than the current buffer size.
207       int copied = std::min(size - taken, remaining_bytes_in_buffer);
208 
209       // |data| is NULL if we are seeking forward, so there's no need to copy.
210       if (data)
211         memcpy(data + taken, buffer->data() + current_buffer_offset, copied);
212 
213       // Increase total number of bytes copied, which regulates when to end this
214       // loop.
215       taken += copied;
216 
217       // We have read |copied| bytes from the current buffer. Advances the
218       // offset.
219       current_buffer_offset += copied;
220     } else {
221       int skipped = std::min(remaining_bytes_in_buffer, bytes_to_skip);
222       current_buffer_offset += skipped;
223       bytes_to_skip -= skipped;
224     }
225 
226     // The buffer has been consumed.
227     if (current_buffer_offset == buffer->data_size()) {
228       if (advance_position) {
229         // Next buffer may not have timestamp, so we need to update current
230         // timestamp before switching to the next buffer.
231         UpdateCurrentTime(current_buffer, current_buffer_offset);
232       }
233 
234       BufferQueue::iterator next = current_buffer;
235       ++next;
236       // If we are at the last buffer, don't advance.
237       if (next == buffers_.end())
238         break;
239 
240       // Advances the iterator.
241       current_buffer = next;
242       current_buffer_offset = 0;
243     }
244   }
245 
246   if (advance_position) {
247     // We have less forward bytes and more backward bytes. Updates these
248     // counters by |taken|.
249     forward_bytes_ -= taken;
250     backward_bytes_ += taken;
251     DCHECK_GE(forward_bytes_, 0);
252     DCHECK(current_buffer_ != buffers_.end() || forward_bytes_ == 0);
253 
254     current_buffer_ = current_buffer;
255     current_buffer_offset_ = current_buffer_offset;
256 
257     UpdateCurrentTime(current_buffer_, current_buffer_offset_);
258     EvictBackwardBuffers();
259   }
260 
261   return taken;
262 }
263 
UpdateCurrentTime(BufferQueue::iterator buffer,int offset)264 void SeekableBuffer::UpdateCurrentTime(BufferQueue::iterator buffer,
265                                        int offset) {
266   // Garbage values are unavoidable, so this check will remain.
267   if (buffer != buffers_.end() &&
268       (*buffer)->timestamp() != kNoTimestamp()) {
269     int64 time_offset = ((*buffer)->duration().InMicroseconds() * offset) /
270                         (*buffer)->data_size();
271 
272     current_time_ = (*buffer)->timestamp() +
273                     base::TimeDelta::FromMicroseconds(time_offset);
274   }
275 }
276 
277 }  // namespace media
278