Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(274)

Unified Diff: android_webview/native/input_stream_reader.cc

Issue 11363123: [android_webview] Don't block the IO thread when reading from an InputStream. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: fix clang error Created 8 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: android_webview/native/input_stream_reader.cc
diff --git a/android_webview/native/input_stream_reader.cc b/android_webview/native/input_stream_reader.cc
new file mode 100644
index 0000000000000000000000000000000000000000..9b5befb1f728640d0424d034d9dbe5d7359ee30a
--- /dev/null
+++ b/android_webview/native/input_stream_reader.cc
@@ -0,0 +1,123 @@
+// Copyright (c) 2012 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "android_webview/native/input_stream_reader.h"
+
+#include "android_webview/native/input_stream.h"
+#include "base/message_loop.h"
+#include "content/public/browser/browser_thread.h"
+#include "net/base/net_errors.h"
+#include "net/http/http_byte_range.h"
+
joth 2012/11/07 16:55:38 ISTM this class is now fairly well abstracted from
mkosiba (inactive) 2012/11/15 19:18:50 Done.
+using base::android::AttachCurrentThread;
+using content::BrowserThread;
+
+namespace android_webview {
+
+InputStreamReader::InputStreamReader(
+ android_webview::InputStream* stream)
+ : stream_(stream) {
+ DCHECK(stream);
+}
+
+InputStreamReader::~InputStreamReader() {
+}
+
+void InputStreamReader::Seek(net::HttpByteRange byte_range,
+ net::CompletionCallback on_seek_completion) {
+ JNIEnv* env = AttachCurrentThread();
+ int content_size = 0;
+ DCHECK(env);
+
+ int error_code = VerifyRequestedRange(env, &byte_range, &content_size);
+ if (error_code != net::OK) {
+ PostCompletionCallback(on_seek_completion, error_code);
+ return;
+ }
+
+ error_code = SkipToRequestedRange(env, byte_range);
+ if (error_code != net::OK) {
+ PostCompletionCallback(on_seek_completion, error_code);
+ return;
+ }
+
+ DCHECK_GE(content_size, 0);
+ PostCompletionCallback(on_seek_completion, content_size);
+}
+
+void InputStreamReader::ReadRawData(
+ net::IOBuffer* dest,
+ int dest_size,
+ net::CompletionCallback on_read_completion) {
+ if (!dest_size) {
+ PostCompletionCallback(on_read_completion, 0);
+ return;
+ }
+
+ JNIEnv* env = AttachCurrentThread();
+ DCHECK(env);
+ DCHECK_GT(dest_size, 0);
+
+ int bytes_read = 0;
+ if (!stream_->Read(env, dest, dest_size, &bytes_read))
+ PostCompletionCallback(on_read_completion, net::ERR_FAILED);
+ else
+ PostCompletionCallback(on_read_completion, bytes_read);
+}
+
+int InputStreamReader::VerifyRequestedRange(
+ JNIEnv* env,
+ net::HttpByteRange* byte_range,
+ int* content_size) {
+ DCHECK(content_size);
+ int32_t size = 0;
+ if (!stream_->BytesAvailable(env, &size)) {
+ return net::ERR_FAILED;
+ }
+
+ if (size <= 0)
+ return net::OK;
+
+ // Check that the requested range was valid.
+ if (!byte_range->ComputeBounds(size)) {
+ return net::ERR_REQUEST_RANGE_NOT_SATISFIABLE;
+ }
+
+ size = byte_range->last_byte_position() -
+ byte_range->first_byte_position() + 1;
+ DCHECK_GE(size, 0);
+ *content_size = size;
+
+ return net::OK;
+}
+
+int InputStreamReader::SkipToRequestedRange(
+ JNIEnv* env,
+ const net::HttpByteRange& byte_range) {
+ // Skip to the start of the requested data. This has to be done in a loop
+ // because the underlying InputStream is not guaranteed to skip the requested
+ // number of bytes.
+ if (byte_range.IsValid() && byte_range.first_byte_position() != 0) {
+ int64_t skipped, bytes_to_skip = byte_range.first_byte_position();
+ do {
+ if (!stream_->Skip(env, bytes_to_skip, &skipped)) {
+ return net::ERR_FAILED;
+ }
+ if (skipped <= 0) {
+ return net::ERR_REQUEST_RANGE_NOT_SATISFIABLE;
+ }
+ } while ((bytes_to_skip -= skipped) > 0);
+ }
+ return net::OK;
+}
+
+void InputStreamReader::PostCompletionCallback(
+ const net::CompletionCallback& callback,
+ int result) {
+ BrowserThread::PostTask(BrowserThread::IO,
joth 2012/11/07 16:55:38 this class is a little bit shady, that you can cal
benm (inactive) 2012/11/15 17:39:01 Agree, it'd be nice if you're always called back o
mkosiba (inactive) 2012/11/15 19:18:50 Done.
+ FROM_HERE,
+ base::Bind(callback, result));
+}
+
+} // namespace android_webview

Powered by Google App Engine
This is Rietveld 408576698