Upstream version 10.39.225.0
[platform/framework/web/crosswalk.git] / src / net / disk_cache / blockfile / in_flight_backend_io.cc
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 "net/disk_cache/blockfile/in_flight_backend_io.h"
6
7 #include "base/bind.h"
8 #include "base/bind_helpers.h"
9 #include "base/compiler_specific.h"
10 #include "base/logging.h"
11 #include "base/single_thread_task_runner.h"
12 #include "net/base/net_errors.h"
13 #include "net/disk_cache/blockfile/backend_impl.h"
14 #include "net/disk_cache/blockfile/entry_impl.h"
15 #include "net/disk_cache/blockfile/histogram_macros.h"
16
17 // Provide a BackendImpl object to macros from histogram_macros.h.
18 #define CACHE_UMA_BACKEND_IMPL_OBJ backend_
19
20 namespace disk_cache {
21
22 BackendIO::BackendIO(InFlightIO* controller, BackendImpl* backend,
23                      const net::CompletionCallback& callback)
24     : BackgroundIO(controller),
25       backend_(backend),
26       callback_(callback),
27       operation_(OP_NONE),
28       entry_ptr_(NULL),
29       iterator_(NULL),
30       entry_(NULL),
31       index_(0),
32       offset_(0),
33       buf_len_(0),
34       truncate_(false),
35       offset64_(0),
36       start_(NULL) {
37   start_time_ = base::TimeTicks::Now();
38 }
39
40 // Runs on the background thread.
41 void BackendIO::ExecuteOperation() {
42   if (IsEntryOperation())
43     return ExecuteEntryOperation();
44
45   ExecuteBackendOperation();
46 }
47
48 // Runs on the background thread.
49 void BackendIO::OnIOComplete(int result) {
50   DCHECK(IsEntryOperation());
51   DCHECK_NE(result, net::ERR_IO_PENDING);
52   result_ = result;
53   NotifyController();
54 }
55
56 // Runs on the primary thread.
57 void BackendIO::OnDone(bool cancel) {
58   if (IsEntryOperation()) {
59     CACHE_UMA(TIMES, "TotalIOTime", 0, ElapsedTime());
60   }
61
62   if (!ReturnsEntry())
63     return;
64
65   if (result() == net::OK) {
66     static_cast<EntryImpl*>(*entry_ptr_)->OnEntryCreated(backend_);
67     if (cancel)
68       (*entry_ptr_)->Close();
69   }
70 }
71
72 bool BackendIO::IsEntryOperation() {
73   return operation_ > OP_MAX_BACKEND;
74 }
75
76 // Runs on the background thread.
77 void BackendIO::ReferenceEntry() {
78   entry_->AddRef();
79 }
80
81 void BackendIO::Init() {
82   operation_ = OP_INIT;
83 }
84
85 void BackendIO::OpenEntry(const std::string& key, Entry** entry) {
86   operation_ = OP_OPEN;
87   key_ = key;
88   entry_ptr_ = entry;
89 }
90
91 void BackendIO::CreateEntry(const std::string& key, Entry** entry) {
92   operation_ = OP_CREATE;
93   key_ = key;
94   entry_ptr_ = entry;
95 }
96
97 void BackendIO::DoomEntry(const std::string& key) {
98   operation_ = OP_DOOM;
99   key_ = key;
100 }
101
102 void BackendIO::DoomAllEntries() {
103   operation_ = OP_DOOM_ALL;
104 }
105
106 void BackendIO::DoomEntriesBetween(const base::Time initial_time,
107                                    const base::Time end_time) {
108   operation_ = OP_DOOM_BETWEEN;
109   initial_time_ = initial_time;
110   end_time_ = end_time;
111 }
112
113 void BackendIO::DoomEntriesSince(const base::Time initial_time) {
114   operation_ = OP_DOOM_SINCE;
115   initial_time_ = initial_time;
116 }
117
118 void BackendIO::OpenNextEntry(Rankings::Iterator* iterator,
119                               Entry** next_entry) {
120   operation_ = OP_OPEN_NEXT;
121   iterator_ = iterator;
122   entry_ptr_ = next_entry;
123 }
124
125 void BackendIO::EndEnumeration(scoped_ptr<Rankings::Iterator> iterator) {
126   operation_ = OP_END_ENUMERATION;
127   scoped_iterator_ = iterator.Pass();
128 }
129
130 void BackendIO::OnExternalCacheHit(const std::string& key) {
131   operation_ = OP_ON_EXTERNAL_CACHE_HIT;
132   key_ = key;
133 }
134
135 void BackendIO::CloseEntryImpl(EntryImpl* entry) {
136   operation_ = OP_CLOSE_ENTRY;
137   entry_ = entry;
138 }
139
140 void BackendIO::DoomEntryImpl(EntryImpl* entry) {
141   operation_ = OP_DOOM_ENTRY;
142   entry_ = entry;
143 }
144
145 void BackendIO::FlushQueue() {
146   operation_ = OP_FLUSH_QUEUE;
147 }
148
149 void BackendIO::RunTask(const base::Closure& task) {
150   operation_ = OP_RUN_TASK;
151   task_ = task;
152 }
153
154 void BackendIO::ReadData(EntryImpl* entry, int index, int offset,
155                          net::IOBuffer* buf, int buf_len) {
156   operation_ = OP_READ;
157   entry_ = entry;
158   index_ = index;
159   offset_ = offset;
160   buf_ = buf;
161   buf_len_ = buf_len;
162 }
163
164 void BackendIO::WriteData(EntryImpl* entry, int index, int offset,
165                           net::IOBuffer* buf, int buf_len, bool truncate) {
166   operation_ = OP_WRITE;
167   entry_ = entry;
168   index_ = index;
169   offset_ = offset;
170   buf_ = buf;
171   buf_len_ = buf_len;
172   truncate_ = truncate;
173 }
174
175 void BackendIO::ReadSparseData(EntryImpl* entry, int64 offset,
176                                net::IOBuffer* buf, int buf_len) {
177   operation_ = OP_READ_SPARSE;
178   entry_ = entry;
179   offset64_ = offset;
180   buf_ = buf;
181   buf_len_ = buf_len;
182 }
183
184 void BackendIO::WriteSparseData(EntryImpl* entry, int64 offset,
185                                 net::IOBuffer* buf, int buf_len) {
186   operation_ = OP_WRITE_SPARSE;
187   entry_ = entry;
188   offset64_ = offset;
189   buf_ = buf;
190   buf_len_ = buf_len;
191 }
192
193 void BackendIO::GetAvailableRange(EntryImpl* entry, int64 offset, int len,
194                                   int64* start) {
195   operation_ = OP_GET_RANGE;
196   entry_ = entry;
197   offset64_ = offset;
198   buf_len_ = len;
199   start_ = start;
200 }
201
202 void BackendIO::CancelSparseIO(EntryImpl* entry) {
203   operation_ = OP_CANCEL_IO;
204   entry_ = entry;
205 }
206
207 void BackendIO::ReadyForSparseIO(EntryImpl* entry) {
208   operation_ = OP_IS_READY;
209   entry_ = entry;
210 }
211
212 BackendIO::~BackendIO() {}
213
214 bool BackendIO::ReturnsEntry() {
215   return operation_ == OP_OPEN || operation_ == OP_CREATE ||
216       operation_ == OP_OPEN_NEXT;
217 }
218
219 base::TimeDelta BackendIO::ElapsedTime() const {
220   return base::TimeTicks::Now() - start_time_;
221 }
222
223 // Runs on the background thread.
224 void BackendIO::ExecuteBackendOperation() {
225   switch (operation_) {
226     case OP_INIT:
227       result_ = backend_->SyncInit();
228       break;
229     case OP_OPEN:
230       result_ = backend_->SyncOpenEntry(key_, entry_ptr_);
231       break;
232     case OP_CREATE:
233       result_ = backend_->SyncCreateEntry(key_, entry_ptr_);
234       break;
235     case OP_DOOM:
236       result_ = backend_->SyncDoomEntry(key_);
237       break;
238     case OP_DOOM_ALL:
239       result_ = backend_->SyncDoomAllEntries();
240       break;
241     case OP_DOOM_BETWEEN:
242       result_ = backend_->SyncDoomEntriesBetween(initial_time_, end_time_);
243       break;
244     case OP_DOOM_SINCE:
245       result_ = backend_->SyncDoomEntriesSince(initial_time_);
246       break;
247     case OP_OPEN_NEXT:
248       result_ = backend_->SyncOpenNextEntry(iterator_, entry_ptr_);
249       break;
250     case OP_END_ENUMERATION:
251       backend_->SyncEndEnumeration(scoped_iterator_.Pass());
252       result_ = net::OK;
253       break;
254     case OP_ON_EXTERNAL_CACHE_HIT:
255       backend_->SyncOnExternalCacheHit(key_);
256       result_ = net::OK;
257       break;
258     case OP_CLOSE_ENTRY:
259       entry_->Release();
260       result_ = net::OK;
261       break;
262     case OP_DOOM_ENTRY:
263       entry_->DoomImpl();
264       result_ = net::OK;
265       break;
266     case OP_FLUSH_QUEUE:
267       result_ = net::OK;
268       break;
269     case OP_RUN_TASK:
270       task_.Run();
271       result_ = net::OK;
272       break;
273     default:
274       NOTREACHED() << "Invalid Operation";
275       result_ = net::ERR_UNEXPECTED;
276   }
277   DCHECK_NE(net::ERR_IO_PENDING, result_);
278   NotifyController();
279 }
280
281 // Runs on the background thread.
282 void BackendIO::ExecuteEntryOperation() {
283   switch (operation_) {
284     case OP_READ:
285       result_ =
286           entry_->ReadDataImpl(index_, offset_, buf_.get(), buf_len_,
287                                base::Bind(&BackendIO::OnIOComplete, this));
288       break;
289     case OP_WRITE:
290       result_ =
291           entry_->WriteDataImpl(index_, offset_, buf_.get(), buf_len_,
292                                 base::Bind(&BackendIO::OnIOComplete, this),
293                                 truncate_);
294       break;
295     case OP_READ_SPARSE:
296       result_ = entry_->ReadSparseDataImpl(
297                     offset64_, buf_.get(), buf_len_,
298                     base::Bind(&BackendIO::OnIOComplete, this));
299       break;
300     case OP_WRITE_SPARSE:
301       result_ = entry_->WriteSparseDataImpl(
302                     offset64_, buf_.get(), buf_len_,
303                     base::Bind(&BackendIO::OnIOComplete, this));
304       break;
305     case OP_GET_RANGE:
306       result_ = entry_->GetAvailableRangeImpl(offset64_, buf_len_, start_);
307       break;
308     case OP_CANCEL_IO:
309       entry_->CancelSparseIOImpl();
310       result_ = net::OK;
311       break;
312     case OP_IS_READY:
313       result_ = entry_->ReadyForSparseIOImpl(
314                     base::Bind(&BackendIO::OnIOComplete, this));
315       break;
316     default:
317       NOTREACHED() << "Invalid Operation";
318       result_ = net::ERR_UNEXPECTED;
319   }
320   buf_ = NULL;
321   if (result_ != net::ERR_IO_PENDING)
322     NotifyController();
323 }
324
325 InFlightBackendIO::InFlightBackendIO(
326     BackendImpl* backend,
327     const scoped_refptr<base::SingleThreadTaskRunner>& background_thread)
328     : backend_(backend),
329       background_thread_(background_thread),
330       ptr_factory_(this) {
331 }
332
333 InFlightBackendIO::~InFlightBackendIO() {
334 }
335
336 void InFlightBackendIO::Init(const net::CompletionCallback& callback) {
337   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
338   operation->Init();
339   PostOperation(operation.get());
340 }
341
342 void InFlightBackendIO::OpenEntry(const std::string& key, Entry** entry,
343                                   const net::CompletionCallback& callback) {
344   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
345   operation->OpenEntry(key, entry);
346   PostOperation(operation.get());
347 }
348
349 void InFlightBackendIO::CreateEntry(const std::string& key, Entry** entry,
350                                     const net::CompletionCallback& callback) {
351   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
352   operation->CreateEntry(key, entry);
353   PostOperation(operation.get());
354 }
355
356 void InFlightBackendIO::DoomEntry(const std::string& key,
357                                   const net::CompletionCallback& callback) {
358   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
359   operation->DoomEntry(key);
360   PostOperation(operation.get());
361 }
362
363 void InFlightBackendIO::DoomAllEntries(
364     const net::CompletionCallback& callback) {
365   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
366   operation->DoomAllEntries();
367   PostOperation(operation.get());
368 }
369
370 void InFlightBackendIO::DoomEntriesBetween(const base::Time initial_time,
371                         const base::Time end_time,
372                         const net::CompletionCallback& callback) {
373   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
374   operation->DoomEntriesBetween(initial_time, end_time);
375   PostOperation(operation.get());
376 }
377
378 void InFlightBackendIO::DoomEntriesSince(
379     const base::Time initial_time, const net::CompletionCallback& callback) {
380   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
381   operation->DoomEntriesSince(initial_time);
382   PostOperation(operation.get());
383 }
384
385 void InFlightBackendIO::OpenNextEntry(Rankings::Iterator* iterator,
386                                       Entry** next_entry,
387                                       const net::CompletionCallback& callback) {
388   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
389   operation->OpenNextEntry(iterator, next_entry);
390   PostOperation(operation.get());
391 }
392
393 void InFlightBackendIO::EndEnumeration(
394     scoped_ptr<Rankings::Iterator> iterator) {
395   scoped_refptr<BackendIO> operation(
396       new BackendIO(this, backend_, net::CompletionCallback()));
397   operation->EndEnumeration(iterator.Pass());
398   PostOperation(operation.get());
399 }
400
401 void InFlightBackendIO::OnExternalCacheHit(const std::string& key) {
402   scoped_refptr<BackendIO> operation(
403       new BackendIO(this, backend_, net::CompletionCallback()));
404   operation->OnExternalCacheHit(key);
405   PostOperation(operation.get());
406 }
407
408 void InFlightBackendIO::CloseEntryImpl(EntryImpl* entry) {
409   scoped_refptr<BackendIO> operation(
410       new BackendIO(this, backend_, net::CompletionCallback()));
411   operation->CloseEntryImpl(entry);
412   PostOperation(operation.get());
413 }
414
415 void InFlightBackendIO::DoomEntryImpl(EntryImpl* entry) {
416   scoped_refptr<BackendIO> operation(
417       new BackendIO(this, backend_, net::CompletionCallback()));
418   operation->DoomEntryImpl(entry);
419   PostOperation(operation.get());
420 }
421
422 void InFlightBackendIO::FlushQueue(const net::CompletionCallback& callback) {
423   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
424   operation->FlushQueue();
425   PostOperation(operation.get());
426 }
427
428 void InFlightBackendIO::RunTask(
429     const base::Closure& task, const net::CompletionCallback& callback) {
430   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
431   operation->RunTask(task);
432   PostOperation(operation.get());
433 }
434
435 void InFlightBackendIO::ReadData(EntryImpl* entry, int index, int offset,
436                                  net::IOBuffer* buf, int buf_len,
437                                  const net::CompletionCallback& callback) {
438   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
439   operation->ReadData(entry, index, offset, buf, buf_len);
440   PostOperation(operation.get());
441 }
442
443 void InFlightBackendIO::WriteData(EntryImpl* entry, int index, int offset,
444                                   net::IOBuffer* buf, int buf_len,
445                                   bool truncate,
446                                   const net::CompletionCallback& callback) {
447   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
448   operation->WriteData(entry, index, offset, buf, buf_len, truncate);
449   PostOperation(operation.get());
450 }
451
452 void InFlightBackendIO::ReadSparseData(
453     EntryImpl* entry, int64 offset, net::IOBuffer* buf, int buf_len,
454     const net::CompletionCallback& callback) {
455   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
456   operation->ReadSparseData(entry, offset, buf, buf_len);
457   PostOperation(operation.get());
458 }
459
460 void InFlightBackendIO::WriteSparseData(
461     EntryImpl* entry, int64 offset, net::IOBuffer* buf, int buf_len,
462     const net::CompletionCallback& callback) {
463   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
464   operation->WriteSparseData(entry, offset, buf, buf_len);
465   PostOperation(operation.get());
466 }
467
468 void InFlightBackendIO::GetAvailableRange(
469     EntryImpl* entry, int64 offset, int len, int64* start,
470     const net::CompletionCallback& callback) {
471   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
472   operation->GetAvailableRange(entry, offset, len, start);
473   PostOperation(operation.get());
474 }
475
476 void InFlightBackendIO::CancelSparseIO(EntryImpl* entry) {
477   scoped_refptr<BackendIO> operation(
478       new BackendIO(this, backend_, net::CompletionCallback()));
479   operation->CancelSparseIO(entry);
480   PostOperation(operation.get());
481 }
482
483 void InFlightBackendIO::ReadyForSparseIO(
484     EntryImpl* entry, const net::CompletionCallback& callback) {
485   scoped_refptr<BackendIO> operation(new BackendIO(this, backend_, callback));
486   operation->ReadyForSparseIO(entry);
487   PostOperation(operation.get());
488 }
489
490 void InFlightBackendIO::WaitForPendingIO() {
491   InFlightIO::WaitForPendingIO();
492 }
493
494 void InFlightBackendIO::OnOperationComplete(BackgroundIO* operation,
495                                             bool cancel) {
496   BackendIO* op = static_cast<BackendIO*>(operation);
497   op->OnDone(cancel);
498
499   if (!op->callback().is_null() && (!cancel || op->IsEntryOperation()))
500     op->callback().Run(op->result());
501 }
502
503 void InFlightBackendIO::PostOperation(BackendIO* operation) {
504   background_thread_->PostTask(
505       FROM_HERE, base::Bind(&BackendIO::ExecuteOperation, operation));
506   OnOperationPosted(operation);
507 }
508
509 base::WeakPtr<InFlightBackendIO> InFlightBackendIO::GetWeakPtr() {
510   return ptr_factory_.GetWeakPtr();
511 }
512
513 }  // namespace