Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright 2014 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 "chrome/browser/media_galleries/fileapi/readahead_file_stream_reader.h" | |
| 6 | |
| 7 #include <algorithm> | |
| 8 | |
| 9 #include "base/message_loop/message_loop.h" | |
| 10 #include "base/numerics/safe_conversions.h" | |
| 11 #include "net/base/io_buffer.h" | |
| 12 #include "net/base/net_errors.h" | |
| 13 | |
| 14 using webkit_blob::FileStreamReader; | |
| 15 | |
| 16 namespace { | |
| 17 | |
| 18 const int kDesiredNumberOfBuffers = 2; // So we are always one buffer ahead. | |
|
vandebo (ex-Chrome)
2014/03/05 17:31:31
nit: two spaces before a comment.
tommycli
2014/03/05 18:07:29
Done.
| |
| 19 const int kBufferSize = 1024*1024; // 1MB to minimize transaction costs. | |
| 20 | |
| 21 } // namespace | |
| 22 | |
| 23 ReadaheadFileStreamReader::ReadaheadFileStreamReader(FileStreamReader* source) | |
| 24 : source_(source), | |
| 25 source_error_(0), | |
| 26 source_has_pending_read_(false), | |
| 27 current_offset_(0), | |
| 28 weak_factory_(this) { | |
| 29 } | |
| 30 | |
| 31 ReadaheadFileStreamReader::~ReadaheadFileStreamReader() {} | |
| 32 | |
| 33 int ReadaheadFileStreamReader::Read( | |
| 34 net::IOBuffer* buf, int buf_len, const net::CompletionCallback& callback) { | |
| 35 DCHECK(!pending_sink_buffer_.get()); | |
| 36 DCHECK(pending_read_callback_.is_null()); | |
| 37 | |
| 38 ReadFromSourceIfNeeded(); | |
| 39 | |
| 40 int result = FinishReadFromCacheOrStoredError(buf, buf_len); | |
| 41 | |
| 42 // We are waiting for an source read to complete, so save the request. | |
| 43 if (result == net::ERR_IO_PENDING) { | |
| 44 DCHECK(!pending_sink_buffer_.get()); | |
| 45 DCHECK(pending_read_callback_.is_null()); | |
| 46 pending_sink_buffer_ = new net::DrainableIOBuffer(buf, buf_len); | |
|
vandebo (ex-Chrome)
2014/03/05 17:31:31
You don't actually use pending_sink_buffer_ as a D
tommycli
2014/03/05 18:07:29
Done.
| |
| 47 pending_read_callback_ = callback; | |
| 48 } | |
| 49 | |
| 50 return result; | |
| 51 } | |
| 52 | |
| 53 int64 ReadaheadFileStreamReader::GetLength( | |
| 54 const net::Int64CompletionCallback& callback) { | |
| 55 return source_->GetLength(callback); | |
| 56 } | |
| 57 | |
| 58 int ReadaheadFileStreamReader::FinishReadFromCacheOrStoredError( | |
| 59 net::IOBuffer* buf, int buf_len) { | |
| 60 // If we don't have any ready cache, return the pending read code, or | |
| 61 // the stored error code. | |
| 62 if (buffers_.empty()) { | |
| 63 if (source_.get()) { | |
| 64 DCHECK(source_has_pending_read_); | |
| 65 return net::ERR_IO_PENDING; | |
| 66 } else { | |
| 67 return source_error_; | |
| 68 } | |
| 69 } | |
| 70 | |
| 71 scoped_refptr<net::DrainableIOBuffer> sink( | |
| 72 new net::DrainableIOBuffer(buf, buf_len)); | |
| 73 | |
| 74 while (sink->BytesRemaining() > 0 && !buffers_.empty()) { | |
| 75 net::DrainableIOBuffer* source_buffer = buffers_.front().get(); | |
| 76 | |
| 77 DCHECK(source_buffer->BytesRemaining() > 0); | |
| 78 | |
| 79 int copy_len = std::min(source_buffer->BytesRemaining(), | |
| 80 sink->BytesRemaining()); | |
| 81 std::copy(source_buffer->data(), source_buffer->data() + copy_len, | |
| 82 sink->data()); | |
| 83 | |
| 84 source_buffer->DidConsume(copy_len); | |
| 85 sink->DidConsume(copy_len); | |
| 86 | |
| 87 if (source_buffer->BytesRemaining() == 0) { | |
| 88 buffers_.pop(); | |
| 89 | |
| 90 // Get a new buffer to replace the one we just used up. | |
| 91 ReadFromSourceIfNeeded(); | |
| 92 } | |
| 93 } | |
| 94 | |
| 95 return sink->BytesConsumed(); | |
| 96 } | |
| 97 | |
| 98 void ReadaheadFileStreamReader::ReadFromSourceIfNeeded() { | |
| 99 if (!source_.get() || source_has_pending_read_ || | |
| 100 buffers_.size() >= kDesiredNumberOfBuffers) { | |
| 101 return; | |
| 102 } | |
| 103 | |
| 104 source_has_pending_read_ = true; | |
| 105 | |
| 106 scoped_refptr<net::IOBuffer> buf(new net::IOBuffer(kBufferSize)); | |
| 107 int result = source_->Read( | |
| 108 buf, | |
| 109 kBufferSize, | |
| 110 base::Bind(&ReadaheadFileStreamReader::OnFinishReadFromSource, | |
| 111 weak_factory_.GetWeakPtr(), buf)); | |
| 112 | |
| 113 if (result != net::ERR_IO_PENDING) { | |
|
vandebo (ex-Chrome)
2014/03/05 17:31:31
nit: predominate style in this directory is no {}'
tommycli
2014/03/05 18:07:29
Done.
| |
| 114 OnFinishReadFromSource(buf, result); | |
| 115 } | |
| 116 } | |
| 117 | |
| 118 void ReadaheadFileStreamReader::OnFinishReadFromSource(net::IOBuffer* buf, | |
| 119 int result) { | |
| 120 DCHECK(result != net::ERR_IO_PENDING); | |
| 121 DCHECK(source_has_pending_read_); | |
| 122 source_has_pending_read_ = false; | |
| 123 | |
| 124 // Either store the data read from |source_|, or store the error code. | |
| 125 if (result > 0) { | |
| 126 scoped_refptr<net::DrainableIOBuffer> drainable_buffer( | |
| 127 new net::DrainableIOBuffer(buf, result)); | |
| 128 buffers_.push(drainable_buffer); | |
|
vandebo (ex-Chrome)
2014/03/05 17:31:31
This should probably call ReadFromSourceIfNeeded()
tommycli
2014/03/05 18:07:29
Done.
| |
| 129 } else { | |
| 130 source_.reset(); | |
| 131 source_error_ = result; | |
| 132 } | |
| 133 | |
| 134 // If there's a read request waiting for the source FileStreamReader to | |
| 135 // finish reading, fulfill that request now from the cache or stored error. | |
| 136 if (pending_sink_buffer_.get()) { | |
| 137 DCHECK(!pending_read_callback_.is_null()); | |
| 138 | |
| 139 // Free the pending callback before running it, as the callback often | |
| 140 // dispatches another read. | |
| 141 scoped_refptr<net::DrainableIOBuffer> sink = pending_sink_buffer_; | |
| 142 pending_sink_buffer_ = NULL; | |
| 143 net::CompletionCallback completion_callback = pending_read_callback_; | |
| 144 pending_read_callback_.Reset(); | |
| 145 | |
| 146 completion_callback.Run( | |
| 147 FinishReadFromCacheOrStoredError(sink, sink->BytesRemaining())); | |
| 148 } | |
| 149 } | |
| OLD | NEW |