OLD | NEW |
| (Empty) |
1 // Copyright (c) 2009 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/buffer_queue.h" | |
6 | |
7 #include "media/base/buffers.h" | |
8 | |
9 namespace media { | |
10 | |
11 BufferQueue::BufferQueue() | |
12 : data_offset_(0), | |
13 size_in_bytes_(0), | |
14 most_recent_time_() { | |
15 } | |
16 | |
17 BufferQueue::~BufferQueue() { | |
18 } | |
19 | |
20 void BufferQueue::Consume(size_t bytes_to_be_consumed) { | |
21 // Make sure user isn't trying to consume more than we have. | |
22 DCHECK(size_in_bytes_ >= bytes_to_be_consumed); | |
23 | |
24 // As we have enough data to consume, adjust |size_in_bytes_|. | |
25 size_in_bytes_ -= bytes_to_be_consumed; | |
26 | |
27 // Now consume them. | |
28 while (bytes_to_be_consumed > 0) { | |
29 // Calculate number of usable bytes in the front of the |queue_|. | |
30 size_t front_remaining = queue_.front()->GetDataSize() - data_offset_; | |
31 | |
32 // If there is enough data in our first buffer to advance into it, do so. | |
33 // Otherwise, advance into the queue. | |
34 if (front_remaining > bytes_to_be_consumed) { | |
35 data_offset_ += bytes_to_be_consumed; | |
36 bytes_to_be_consumed = 0; | |
37 // Garbage values are unavoidable, so this check will remain. | |
38 if (queue_.front()->GetTimestamp().InMicroseconds() > 0) { | |
39 int64 offset = (queue_.front()->GetDuration().InMicroseconds() * | |
40 data_offset_) / queue_.front()->GetDataSize(); | |
41 | |
42 most_recent_time_ = queue_.front()->GetTimestamp() + | |
43 base::TimeDelta::FromMicroseconds(offset); | |
44 } | |
45 } else { | |
46 data_offset_ = 0; | |
47 // Garbage values are unavoidable, so this check will remain. | |
48 if (queue_.front()->GetTimestamp().InMicroseconds() > 0) { | |
49 most_recent_time_ = queue_.front()->GetTimestamp() + | |
50 queue_.front()->GetDuration(); | |
51 } | |
52 queue_.pop_front(); | |
53 bytes_to_be_consumed -= front_remaining; | |
54 } | |
55 } | |
56 } | |
57 | |
58 size_t BufferQueue::Copy(uint8* dest, size_t bytes) { | |
59 if (bytes == 0) | |
60 return 0; | |
61 | |
62 DCHECK(!queue_.empty()); | |
63 | |
64 size_t current_remaining = 0; | |
65 const uint8* current = NULL; | |
66 size_t copied = 0; | |
67 | |
68 for (size_t i = 0; i < queue_.size() && bytes > 0; ++i) { | |
69 // Calculate number of usable bytes in the front of the |queue_|. Special | |
70 // case for front due to |data_offset_|. | |
71 if (i == 0) { | |
72 current_remaining = queue_.front()->GetDataSize() - data_offset_; | |
73 current = queue_.front()->GetData() + data_offset_; | |
74 } else { | |
75 current_remaining = queue_[i]->GetDataSize(); | |
76 current = queue_[i]->GetData(); | |
77 } | |
78 | |
79 // Prevent writing over the end of the buffer. | |
80 if (current_remaining > bytes) | |
81 current_remaining = bytes; | |
82 | |
83 memcpy(dest + copied, current, current_remaining); | |
84 | |
85 // Modify counts and pointers. | |
86 copied += current_remaining; | |
87 bytes -= current_remaining; | |
88 } | |
89 return copied; | |
90 } | |
91 | |
92 void BufferQueue::Enqueue(Buffer* buffer_in) { | |
93 if (queue_.empty() && buffer_in->GetTimestamp().InMicroseconds() > 0) { | |
94 most_recent_time_ = buffer_in->GetTimestamp(); | |
95 } | |
96 queue_.push_back(buffer_in); | |
97 size_in_bytes_ += buffer_in->GetDataSize(); | |
98 } | |
99 | |
100 base::TimeDelta BufferQueue::GetTime() { | |
101 return most_recent_time_; | |
102 } | |
103 | |
104 void BufferQueue::Clear() { | |
105 queue_.clear(); | |
106 size_in_bytes_ = 0; | |
107 data_offset_ = 0; | |
108 most_recent_time_ = base::TimeDelta(); | |
109 } | |
110 | |
111 bool BufferQueue::IsEmpty() { | |
112 // Since we keep track of the number of bytes, this is easier than calling | |
113 // into |queue_|. | |
114 return size_in_bytes_ == 0; | |
115 } | |
116 | |
117 size_t BufferQueue::SizeInBytes() { | |
118 return size_in_bytes_; | |
119 } | |
120 | |
121 } // namespace media | |
OLD | NEW |