Reland "Non-SFI mode: Switch to newlib. (patchset #4 id:60001 of https://codereview...
[chromium-blink-merge.git] / gpu / command_buffer / client / cmd_buffer_helper.cc
blob038ba41fda06ae7b10e2d73b91e9423d2b1d8410
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.
5 // This file contains the implementation of the command buffer helper class.
7 #include "gpu/command_buffer/client/cmd_buffer_helper.h"
9 #include "base/logging.h"
10 #include "base/time/time.h"
11 #include "gpu/command_buffer/common/command_buffer.h"
12 #include "gpu/command_buffer/common/trace_event.h"
14 namespace gpu {
16 CommandBufferHelper::CommandBufferHelper(CommandBuffer* command_buffer)
17 : command_buffer_(command_buffer),
18 ring_buffer_id_(-1),
19 ring_buffer_size_(0),
20 entries_(NULL),
21 total_entry_count_(0),
22 immediate_entry_count_(0),
23 token_(0),
24 put_(0),
25 last_put_sent_(0),
26 #if defined(CMD_HELPER_PERIODIC_FLUSH_CHECK)
27 commands_issued_(0),
28 #endif
29 usable_(true),
30 context_lost_(false),
31 flush_automatically_(true),
32 flush_generation_(0) {
35 void CommandBufferHelper::SetAutomaticFlushes(bool enabled) {
36 flush_automatically_ = enabled;
37 CalcImmediateEntries(0);
40 bool CommandBufferHelper::IsContextLost() {
41 if (!context_lost_) {
42 context_lost_ = error::IsError(command_buffer()->GetLastError());
44 return context_lost_;
47 void CommandBufferHelper::CalcImmediateEntries(int waiting_count) {
48 DCHECK_GE(waiting_count, 0);
50 // Check if usable & allocated.
51 if (!usable() || !HaveRingBuffer()) {
52 immediate_entry_count_ = 0;
53 return;
56 // Get maximum safe contiguous entries.
57 const int32 curr_get = get_offset();
58 if (curr_get > put_) {
59 immediate_entry_count_ = curr_get - put_ - 1;
60 } else {
61 immediate_entry_count_ =
62 total_entry_count_ - put_ - (curr_get == 0 ? 1 : 0);
65 // Limit entry count to force early flushing.
66 if (flush_automatically_) {
67 int32 limit =
68 total_entry_count_ /
69 ((curr_get == last_put_sent_) ? kAutoFlushSmall : kAutoFlushBig);
71 int32 pending =
72 (put_ + total_entry_count_ - last_put_sent_) % total_entry_count_;
74 if (pending > 0 && pending >= limit) {
75 // Time to force flush.
76 immediate_entry_count_ = 0;
77 } else {
78 // Limit remaining entries, but not lower than waiting_count entries to
79 // prevent deadlock when command size is greater than the flush limit.
80 limit -= pending;
81 limit = limit < waiting_count ? waiting_count : limit;
82 immediate_entry_count_ =
83 immediate_entry_count_ > limit ? limit : immediate_entry_count_;
88 bool CommandBufferHelper::AllocateRingBuffer() {
89 if (!usable()) {
90 return false;
93 if (HaveRingBuffer()) {
94 return true;
97 int32 id = -1;
98 scoped_refptr<Buffer> buffer =
99 command_buffer_->CreateTransferBuffer(ring_buffer_size_, &id);
100 if (id < 0) {
101 ClearUsable();
102 return false;
105 ring_buffer_ = buffer;
106 ring_buffer_id_ = id;
107 command_buffer_->SetGetBuffer(id);
108 entries_ = static_cast<CommandBufferEntry*>(ring_buffer_->memory());
109 total_entry_count_ = ring_buffer_size_ / sizeof(CommandBufferEntry);
110 // Call to SetGetBuffer(id) above resets get and put offsets to 0.
111 // No need to query it through IPC.
112 put_ = 0;
113 CalcImmediateEntries(0);
114 return true;
117 void CommandBufferHelper::FreeResources() {
118 if (HaveRingBuffer()) {
119 command_buffer_->DestroyTransferBuffer(ring_buffer_id_);
120 ring_buffer_id_ = -1;
121 CalcImmediateEntries(0);
125 void CommandBufferHelper::FreeRingBuffer() {
126 CHECK((put_ == get_offset()) ||
127 error::IsError(command_buffer_->GetLastState().error));
128 FreeResources();
131 bool CommandBufferHelper::Initialize(int32 ring_buffer_size) {
132 ring_buffer_size_ = ring_buffer_size;
133 return AllocateRingBuffer();
136 CommandBufferHelper::~CommandBufferHelper() {
137 FreeResources();
140 bool CommandBufferHelper::WaitForGetOffsetInRange(int32 start, int32 end) {
141 if (!usable()) {
142 return false;
144 command_buffer_->WaitForGetOffsetInRange(start, end);
145 return command_buffer_->GetLastError() == gpu::error::kNoError;
148 void CommandBufferHelper::Flush() {
149 // Wrap put_ before flush.
150 if (put_ == total_entry_count_)
151 put_ = 0;
153 if (usable()) {
154 last_flush_time_ = base::TimeTicks::Now();
155 last_put_sent_ = put_;
156 command_buffer_->Flush(put_);
157 ++flush_generation_;
158 CalcImmediateEntries(0);
162 void CommandBufferHelper::OrderingBarrier() {
163 // Wrap put_ before setting the barrier.
164 if (put_ == total_entry_count_)
165 put_ = 0;
167 if (usable()) {
168 command_buffer_->OrderingBarrier(put_);
169 ++flush_generation_;
170 CalcImmediateEntries(0);
174 #if defined(CMD_HELPER_PERIODIC_FLUSH_CHECK)
175 void CommandBufferHelper::PeriodicFlushCheck() {
176 base::TimeTicks current_time = base::TimeTicks::Now();
177 if (current_time - last_flush_time_ >
178 base::TimeDelta::FromMicroseconds(kPeriodicFlushDelayInMicroseconds)) {
179 Flush();
182 #endif
184 // Calls Flush() and then waits until the buffer is empty. Break early if the
185 // error is set.
186 bool CommandBufferHelper::Finish() {
187 TRACE_EVENT0("gpu", "CommandBufferHelper::Finish");
188 if (!usable()) {
189 return false;
191 // If there is no work just exit.
192 if (put_ == get_offset()) {
193 return true;
195 DCHECK(HaveRingBuffer());
196 Flush();
197 if (!WaitForGetOffsetInRange(put_, put_))
198 return false;
199 DCHECK_EQ(get_offset(), put_);
201 CalcImmediateEntries(0);
203 return true;
206 // Inserts a new token into the command stream. It uses an increasing value
207 // scheme so that we don't lose tokens (a token has passed if the current token
208 // value is higher than that token). Calls Finish() if the token value wraps,
209 // which will be rare.
210 int32 CommandBufferHelper::InsertToken() {
211 AllocateRingBuffer();
212 if (!usable()) {
213 return token_;
215 DCHECK(HaveRingBuffer());
216 // Increment token as 31-bit integer. Negative values are used to signal an
217 // error.
218 token_ = (token_ + 1) & 0x7FFFFFFF;
219 cmd::SetToken* cmd = GetCmdSpace<cmd::SetToken>();
220 if (cmd) {
221 cmd->Init(token_);
222 if (token_ == 0) {
223 TRACE_EVENT0("gpu", "CommandBufferHelper::InsertToken(wrapped)");
224 // we wrapped
225 Finish();
226 DCHECK_EQ(token_, last_token_read());
229 return token_;
232 // Waits until the current token value is greater or equal to the value passed
233 // in argument.
234 void CommandBufferHelper::WaitForToken(int32 token) {
235 if (!usable() || !HaveRingBuffer()) {
236 return;
238 // Return immediately if corresponding InsertToken failed.
239 if (token < 0)
240 return;
241 if (token > token_) return; // we wrapped
242 if (last_token_read() >= token)
243 return;
244 Flush();
245 command_buffer_->WaitForTokenInRange(token, token_);
248 // Waits for available entries, basically waiting until get >= put + count + 1.
249 // It actually waits for contiguous entries, so it may need to wrap the buffer
250 // around, adding a noops. Thus this function may change the value of put_. The
251 // function will return early if an error occurs, in which case the available
252 // space may not be available.
253 void CommandBufferHelper::WaitForAvailableEntries(int32 count) {
254 AllocateRingBuffer();
255 if (!usable()) {
256 return;
258 DCHECK(HaveRingBuffer());
259 DCHECK(count < total_entry_count_);
260 if (put_ + count > total_entry_count_) {
261 // There's not enough room between the current put and the end of the
262 // buffer, so we need to wrap. We will add noops all the way to the end,
263 // but we need to make sure get wraps first, actually that get is 1 or
264 // more (since put will wrap to 0 after we add the noops).
265 DCHECK_LE(1, put_);
266 int32 curr_get = get_offset();
267 if (curr_get > put_ || curr_get == 0) {
268 TRACE_EVENT0("gpu", "CommandBufferHelper::WaitForAvailableEntries");
269 Flush();
270 if (!WaitForGetOffsetInRange(1, put_))
271 return;
272 curr_get = get_offset();
273 DCHECK_LE(curr_get, put_);
274 DCHECK_NE(0, curr_get);
276 // Insert Noops to fill out the buffer.
277 int32 num_entries = total_entry_count_ - put_;
278 while (num_entries > 0) {
279 int32 num_to_skip = std::min(CommandHeader::kMaxSize, num_entries);
280 cmd::Noop::Set(&entries_[put_], num_to_skip);
281 put_ += num_to_skip;
282 num_entries -= num_to_skip;
284 put_ = 0;
287 // Try to get 'count' entries without flushing.
288 CalcImmediateEntries(count);
289 if (immediate_entry_count_ < count) {
290 // Try again with a shallow Flush().
291 Flush();
292 CalcImmediateEntries(count);
293 if (immediate_entry_count_ < count) {
294 // Buffer is full. Need to wait for entries.
295 TRACE_EVENT0("gpu", "CommandBufferHelper::WaitForAvailableEntries1");
296 if (!WaitForGetOffsetInRange(put_ + count + 1, put_))
297 return;
298 CalcImmediateEntries(count);
299 DCHECK_GE(immediate_entry_count_, count);
305 } // namespace gpu