summaryrefslogtreecommitdiffstats
path: root/webkit
diff options
context:
space:
mode:
authorviettrungluu@chromium.org <viettrungluu@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2011-01-13 18:32:51 +0000
committerviettrungluu@chromium.org <viettrungluu@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2011-01-13 18:32:51 +0000
commit23b3b7003096896b71cd047924e42d5e01ce8e49 (patch)
tree09b45bdefa8bce0da2d09f17acd2145ccdd1f1c4 /webkit
parent822809e5b5b4eddbc532b083fcaad848f255f3ef (diff)
downloadchromium_src-23b3b7003096896b71cd047924e42d5e01ce8e49.zip
chromium_src-23b3b7003096896b71cd047924e42d5e01ce8e49.tar.gz
chromium_src-23b3b7003096896b71cd047924e42d5e01ce8e49.tar.bz2
Fix Pepper URL Loader callbacks.
Add some tests (for aborting calls). BUG=none TEST=ppapi_tests Review URL: http://codereview.chromium.org/6220006 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@71334 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'webkit')
-rw-r--r--webkit/plugins/ppapi/ppb_url_loader_impl.cc80
-rw-r--r--webkit/plugins/ppapi/ppb_url_loader_impl.h25
2 files changed, 67 insertions, 38 deletions
diff --git a/webkit/plugins/ppapi/ppb_url_loader_impl.cc b/webkit/plugins/ppapi/ppb_url_loader_impl.cc
index 14b91f8..fba1b8c 100644
--- a/webkit/plugins/ppapi/ppb_url_loader_impl.cc
+++ b/webkit/plugins/ppapi/ppb_url_loader_impl.cc
@@ -1,4 +1,4 @@
-// Copyright (c) 2010 The Chromium Authors. All rights reserved.
+// Copyright (c) 2011 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.
@@ -21,6 +21,7 @@
#include "third_party/WebKit/WebKit/chromium/public/WebURLResponse.h"
#include "webkit/appcache/web_application_cache_host_impl.h"
#include "webkit/plugins/ppapi/common.h"
+#include "webkit/plugins/ppapi/plugin_module.h"
#include "webkit/plugins/ppapi/ppapi_plugin_instance.h"
#include "webkit/plugins/ppapi/ppb_url_request_info_impl.h"
#include "webkit/plugins/ppapi/ppb_url_response_info_impl.h"
@@ -232,19 +233,19 @@ PPB_URLLoader_Impl* PPB_URLLoader_Impl::AsPPB_URLLoader_Impl() {
int32_t PPB_URLLoader_Impl::Open(PPB_URLRequestInfo_Impl* request,
PP_CompletionCallback callback) {
+ int32_t rv = ValidateCallback(callback);
+ if (rv != PP_OK)
+ return rv;
+
if (loader_.get())
return PP_ERROR_INPROGRESS;
- // We only support non-blocking calls.
- if (!callback.func)
- return PP_ERROR_BADARGUMENT;
-
WebFrame* frame = GetFrame(instance_);
if (!frame)
return PP_ERROR_FAILED;
WebURLRequest web_request(request->ToWebURLRequest(frame));
- int32_t rv = CanRequest(frame, web_request.url());
+ rv = CanRequest(frame, web_request.url());
if (rv != PP_OK)
return rv;
@@ -263,28 +264,25 @@ int32_t PPB_URLLoader_Impl::Open(PPB_URLRequestInfo_Impl* request,
loader_->loadAsynchronously(web_request, this);
request_info_ = scoped_refptr<PPB_URLRequestInfo_Impl>(request);
- pending_callback_ = callback;
// Notify completion when we receive a redirect or response headers.
+ RegisterCallback(callback);
return PP_ERROR_WOULDBLOCK;
}
int32_t PPB_URLLoader_Impl::FollowRedirect(PP_CompletionCallback callback) {
- if (pending_callback_.func)
- return PP_ERROR_INPROGRESS;
-
- // We only support non-blocking calls.
- if (!callback.func)
- return PP_ERROR_BADARGUMENT;
+ int32_t rv = ValidateCallback(callback);
+ if (rv != PP_OK)
+ return rv;
WebURL redirect_url = GURL(response_info_->redirect_url());
- int32_t rv = CanRequest(GetFrame(instance_), redirect_url);
+ rv = CanRequest(GetFrame(instance_), redirect_url);
if (rv != PP_OK)
return rv;
- pending_callback_ = callback;
loader_->setDefersLoading(false); // Allow the redirect to continue.
+ RegisterCallback(callback);
return PP_ERROR_WOULDBLOCK;
}
@@ -316,16 +314,13 @@ bool PPB_URLLoader_Impl::GetDownloadProgress(
int32_t PPB_URLLoader_Impl::ReadResponseBody(char* buffer,
int32_t bytes_to_read,
PP_CompletionCallback callback) {
+ int32_t rv = ValidateCallback(callback);
+ if (rv != PP_OK)
+ return rv;
if (!response_info_ || response_info_->body())
return PP_ERROR_FAILED;
if (bytes_to_read <= 0 || !buffer)
return PP_ERROR_BADARGUMENT;
- if (pending_callback_.func)
- return PP_ERROR_INPROGRESS;
-
- // We only support non-blocking calls.
- if (!callback.func)
- return PP_ERROR_BADARGUMENT;
user_buffer_ = buffer;
user_buffer_size_ = bytes_to_read;
@@ -340,23 +335,24 @@ int32_t PPB_URLLoader_Impl::ReadResponseBody(char* buffer,
return done_status_;
}
- pending_callback_ = callback;
+ RegisterCallback(callback);
return PP_ERROR_WOULDBLOCK;
}
int32_t PPB_URLLoader_Impl::FinishStreamingToFile(
PP_CompletionCallback callback) {
+ int32_t rv = ValidateCallback(callback);
+ if (rv != PP_OK)
+ return rv;
if (!response_info_ || !response_info_->body())
return PP_ERROR_FAILED;
- if (pending_callback_.func)
- return PP_ERROR_INPROGRESS;
// We may have already reached EOF.
if (done_status_ != PP_ERROR_WOULDBLOCK)
return done_status_;
// Wait for didFinishLoading / didFail.
- pending_callback_ = callback;
+ RegisterCallback(callback);
return PP_ERROR_WOULDBLOCK;
}
@@ -367,6 +363,8 @@ void PPB_URLLoader_Impl::Close() {
WebFrame* frame = instance_->container()->element().document().frame();
frame->stopLoading();
}
+ // TODO(viettrungluu): Check what happens to the callback (probably the
+ // wrong thing). May need to post abort here. crbug.com/69457
}
void PPB_URLLoader_Impl::GrantUniversalAccess() {
@@ -431,7 +429,7 @@ void PPB_URLLoader_Impl::didReceiveData(WebURLLoader* loader,
if (user_buffer_) {
RunCallback(FillUserBuffer());
} else {
- DCHECK(!pending_callback_.func);
+ DCHECK(!pending_callback_.get() || pending_callback_->completed());
}
}
@@ -482,13 +480,31 @@ void PPB_URLLoader_Impl::InstanceDestroyed(PluginInstance* instance) {
// goes out of scope.
}
-void PPB_URLLoader_Impl::RunCallback(int32_t result) {
- if (!pending_callback_.func)
- return;
+int32_t PPB_URLLoader_Impl::ValidateCallback(PP_CompletionCallback callback) {
+ // We only support non-blocking calls.
+ if (!callback.func)
+ return PP_ERROR_BADARGUMENT;
- PP_CompletionCallback callback = {0};
- std::swap(callback, pending_callback_);
- PP_RunCompletionCallback(&callback, result);
+ if (pending_callback_.get() && !pending_callback_->completed())
+ return PP_ERROR_INPROGRESS;
+
+ return PP_OK;
+}
+
+void PPB_URLLoader_Impl::RegisterCallback(PP_CompletionCallback callback) {
+ DCHECK(callback.func);
+ DCHECK(!pending_callback_.get() || pending_callback_->completed());
+
+ PP_Resource resource_id = GetReferenceNoAddRef();
+ CHECK(resource_id);
+ pending_callback_ = new TrackedCompletionCallback(
+ module()->GetCallbackTracker(), resource_id, callback);
+}
+
+void PPB_URLLoader_Impl::RunCallback(int32_t result) {
+ scoped_refptr<TrackedCompletionCallback> callback;
+ callback.swap(pending_callback_);
+ callback->Run(result); // Will complete abortively if necessary.
}
size_t PPB_URLLoader_Impl::FillUserBuffer() {
diff --git a/webkit/plugins/ppapi/ppb_url_loader_impl.h b/webkit/plugins/ppapi/ppb_url_loader_impl.h
index 6b05adc..4a3d72e 100644
--- a/webkit/plugins/ppapi/ppb_url_loader_impl.h
+++ b/webkit/plugins/ppapi/ppb_url_loader_impl.h
@@ -1,4 +1,4 @@
-// Copyright (c) 2010 The Chromium Authors. All rights reserved.
+// Copyright (c) 2011 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.
@@ -7,10 +7,12 @@
#include <deque>
+#include "base/ref_counted.h"
#include "base/scoped_ptr.h"
#include "ppapi/c/pp_completion_callback.h"
#include "ppapi/c/trusted/ppb_url_loader_trusted.h"
#include "third_party/WebKit/WebKit/chromium/public/WebURLLoaderClient.h"
+#include "webkit/plugins/ppapi/callbacks.h"
#include "webkit/plugins/ppapi/ppapi_plugin_instance.h"
#include "webkit/plugins/ppapi/resource.h"
@@ -89,7 +91,17 @@ class PPB_URLLoader_Impl : public Resource,
PPB_URLResponseInfo_Impl* response_info() const { return response_info_; }
private:
+ // Check that |callback| is valid (only non-blocking operation is supported)
+ // and that no callback is already pending. Returns |PP_OK| if okay, else
+ // |PP_ERROR_...| to be returned to the plugin.
+ int32_t ValidateCallback(PP_CompletionCallback callback);
+
+ // Sets up |callback| as the pending callback. This should only be called once
+ // it is certain that |PP_ERROR_WOULDBLOCK| will be returned.
+ void RegisterCallback(PP_CompletionCallback callback);
+
void RunCallback(int32_t result);
+
size_t FillUserBuffer();
// Converts a WebURLResponse to a URLResponseInfo and saves it.
@@ -110,10 +122,11 @@ class PPB_URLLoader_Impl : public Resource,
bool RecordDownloadProgress() const;
bool RecordUploadProgress() const;
- // This will be NULL if the instance has been deleted but this PPB_URLLoader_Impl was
- // somehow leaked. In general, you should not need to check this for NULL.
- // However, if you see a NULL pointer crash, that means somebody is holding
- // a reference to this object longer than the PluginInstance's lifetime.
+ // This will be NULL if the instance has been deleted but this
+ // PPB_URLLoader_Impl was somehow leaked. In general, you should not need to
+ // check this for NULL. However, if you see a NULL pointer crash, that means
+ // somebody is holding a reference to this object longer than the
+ // PluginInstance's lifetime.
PluginInstance* instance_;
// If true, then the plugin instance is a full-frame plugin and we're just
@@ -122,7 +135,7 @@ class PPB_URLLoader_Impl : public Resource,
scoped_ptr<WebKit::WebURLLoader> loader_;
scoped_refptr<PPB_URLRequestInfo_Impl> request_info_;
scoped_refptr<PPB_URLResponseInfo_Impl> response_info_;
- PP_CompletionCallback pending_callback_;
+ scoped_refptr<TrackedCompletionCallback> pending_callback_;
std::deque<char> buffer_;
int64_t bytes_sent_;
int64_t total_bytes_to_be_sent_;