Commit bda9e1d6 authored by sgurun@chromium.org's avatar sgurun@chromium.org

Fix threading issues in aw form database

BUG=285584

Review URL: https://chromiumcodereview.appspot.com/23803005

git-svn-id: svn://svn.chromium.org/chrome/trunk/src@221409 0039d316-1c4b-4281-b951-d872f2087c98
parent ad4ec417
...@@ -23,6 +23,8 @@ ...@@ -23,6 +23,8 @@
#include "content/public/browser/web_contents.h" #include "content/public/browser/web_contents.h"
#include "net/url_request/url_request_context.h" #include "net/url_request/url_request_context.h"
using content::BrowserThread;
namespace android_webview { namespace android_webview {
namespace { namespace {
...@@ -41,11 +43,11 @@ class AwResourceContext : public content::ResourceContext { ...@@ -41,11 +43,11 @@ class AwResourceContext : public content::ResourceContext {
// content::ResourceContext implementation. // content::ResourceContext implementation.
virtual net::HostResolver* GetHostResolver() OVERRIDE { virtual net::HostResolver* GetHostResolver() OVERRIDE {
DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO)); DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
return getter_->GetURLRequestContext()->host_resolver(); return getter_->GetURLRequestContext()->host_resolver();
} }
virtual net::URLRequestContext* GetRequestContext() OVERRIDE { virtual net::URLRequestContext* GetRequestContext() OVERRIDE {
DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO)); DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
return getter_->GetURLRequestContext(); return getter_->GetURLRequestContext();
} }
virtual bool AllowMicAccess(const GURL& origin) OVERRIDE { virtual bool AllowMicAccess(const GURL& origin) OVERRIDE {
...@@ -73,6 +75,9 @@ AwBrowserContext::AwBrowserContext( ...@@ -73,6 +75,9 @@ AwBrowserContext::AwBrowserContext(
user_pref_service_ready_(false) { user_pref_service_ready_(false) {
DCHECK(g_browser_context == NULL); DCHECK(g_browser_context == NULL);
g_browser_context = this; g_browser_context = this;
form_database_service_.reset(
new AwFormDatabaseService(context_storage_path_));
} }
AwBrowserContext::~AwBrowserContext() { AwBrowserContext::~AwBrowserContext() {
...@@ -146,10 +151,6 @@ AwQuotaManagerBridge* AwBrowserContext::GetQuotaManagerBridge() { ...@@ -146,10 +151,6 @@ AwQuotaManagerBridge* AwBrowserContext::GetQuotaManagerBridge() {
} }
AwFormDatabaseService* AwBrowserContext::GetFormDatabaseService() { AwFormDatabaseService* AwBrowserContext::GetFormDatabaseService() {
if (!form_database_service_) {
form_database_service_.reset(
new AwFormDatabaseService(context_storage_path_));
}
return form_database_service_.get(); return form_database_service_.get();
} }
......
...@@ -4,11 +4,13 @@ ...@@ -4,11 +4,13 @@
#include "android_webview/browser/aw_form_database_service.h" #include "android_webview/browser/aw_form_database_service.h"
#include "base/logging.h" #include "base/logging.h"
#include "base/synchronization/waitable_event.h"
#include "components/autofill/core/browser/webdata/autofill_table.h" #include "components/autofill/core/browser/webdata/autofill_table.h"
#include "components/webdata/common/webdata_constants.h" #include "components/webdata/common/webdata_constants.h"
#include "content/public/browser/browser_thread.h" #include "content/public/browser/browser_thread.h"
#include "ui/base/l10n/l10n_util_android.h" #include "ui/base/l10n/l10n_util_android.h"
using base::WaitableEvent;
using content::BrowserThread; using content::BrowserThread;
namespace { namespace {
...@@ -23,10 +25,7 @@ void DatabaseErrorCallback(sql::InitStatus status) { ...@@ -23,10 +25,7 @@ void DatabaseErrorCallback(sql::InitStatus status) {
namespace android_webview { namespace android_webview {
AwFormDatabaseService::AwFormDatabaseService(const base::FilePath path) AwFormDatabaseService::AwFormDatabaseService(const base::FilePath path) {
: pending_query_handle_(0),
has_form_data_(false),
completion_(false, false) {
web_database_ = new WebDatabaseService(path.Append(kWebDataFilename), web_database_ = new WebDatabaseService(path.Append(kWebDataFilename),
BrowserThread::GetMessageLoopProxyForThread(BrowserThread::UI), BrowserThread::GetMessageLoopProxyForThread(BrowserThread::UI),
...@@ -42,30 +41,33 @@ AwFormDatabaseService::AwFormDatabaseService(const base::FilePath path) ...@@ -42,30 +41,33 @@ AwFormDatabaseService::AwFormDatabaseService(const base::FilePath path)
} }
AwFormDatabaseService::~AwFormDatabaseService() { AwFormDatabaseService::~AwFormDatabaseService() {
CancelPendingQuery();
Shutdown(); Shutdown();
} }
void AwFormDatabaseService::Shutdown() { void AwFormDatabaseService::Shutdown() {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
DCHECK(result_map_.empty());
// TODO(sgurun) we don't run into this logic right now,
// but if we do, then we need to implement cancellation
// of pending queries.
autofill_data_->ShutdownOnUIThread(); autofill_data_->ShutdownOnUIThread();
web_database_->ShutdownDatabase(); web_database_->ShutdownDatabase();
} }
void AwFormDatabaseService::CancelPendingQuery() {
if (pending_query_handle_) {
if (autofill_data_.get())
autofill_data_->CancelRequest(pending_query_handle_);
pending_query_handle_ = 0;
}
}
scoped_refptr<autofill::AutofillWebDataService> scoped_refptr<autofill::AutofillWebDataService>
AwFormDatabaseService::get_autofill_webdata_service() { AwFormDatabaseService::get_autofill_webdata_service() {
return autofill_data_; return autofill_data_;
} }
void AwFormDatabaseService::ClearFormData() { void AwFormDatabaseService::ClearFormData() {
BrowserThread::PostTask(
BrowserThread::DB,
FROM_HERE,
base::Bind(&AwFormDatabaseService::ClearFormDataImpl,
base::Unretained(this)));
}
void AwFormDatabaseService::ClearFormDataImpl() {
base::Time begin; base::Time begin;
base::Time end = base::Time::Max(); base::Time end = base::Time::Max();
autofill_data_->RemoveFormElementsAddedBetween(begin, end); autofill_data_->RemoveFormElementsAddedBetween(begin, end);
...@@ -73,35 +75,50 @@ void AwFormDatabaseService::ClearFormData() { ...@@ -73,35 +75,50 @@ void AwFormDatabaseService::ClearFormData() {
} }
bool AwFormDatabaseService::HasFormData() { bool AwFormDatabaseService::HasFormData() {
BrowserThread::PostTask(BrowserThread::DB, FROM_HERE, WaitableEvent completion(false, false);
bool result = false;
BrowserThread::PostTask(
BrowserThread::DB,
FROM_HERE,
base::Bind(&AwFormDatabaseService::HasFormDataImpl, base::Bind(&AwFormDatabaseService::HasFormDataImpl,
base::Unretained(this))); base::Unretained(this),
completion_.Wait(); &completion,
return has_form_data_; &result));
completion.Wait();
return result;
} }
void AwFormDatabaseService::HasFormDataImpl() { void AwFormDatabaseService::HasFormDataImpl(
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB)); WaitableEvent* completion,
pending_query_handle_ = autofill_data_->HasFormElements(this); bool* result) {
WebDataServiceBase::Handle pending_query_handle =
autofill_data_->HasFormElements(this);
PendingQuery query;
query.result = result;
query.completion = completion;
result_map_[pending_query_handle] = query;
} }
void AwFormDatabaseService::OnWebDataServiceRequestDone( void AwFormDatabaseService::OnWebDataServiceRequestDone(
WebDataServiceBase::Handle h, WebDataServiceBase::Handle h,
const WDTypedResult* result) { const WDTypedResult* result) {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB)); DCHECK(BrowserThread::CurrentlyOn(BrowserThread::DB));
DCHECK_EQ(pending_query_handle_, h); bool has_form_data = false;
pending_query_handle_ = 0;
has_form_data_ = false;
if (result) { if (result) {
DCHECK_EQ(AUTOFILL_VALUE_RESULT, result->GetType()); DCHECK_EQ(AUTOFILL_VALUE_RESULT, result->GetType());
const WDResult<bool>* autofill_result = const WDResult<bool>* autofill_result =
static_cast<const WDResult<bool>*>(result); static_cast<const WDResult<bool>*>(result);
has_form_data_ = autofill_result->GetValue(); has_form_data = autofill_result->GetValue();
}
QueryMap::const_iterator it = result_map_.find(h);
if (it == result_map_.end()) {
LOG(WARNING) << "Received unexpected callback from web data service";
return;
} }
completion_.Signal(); *(it->second.result) = has_form_data;
it->second.completion->Signal();
result_map_.erase(h);
} }
} // namespace android_webview } // namespace android_webview
...@@ -7,11 +7,14 @@ ...@@ -7,11 +7,14 @@
#include "base/basictypes.h" #include "base/basictypes.h"
#include "base/files/file_path.h" #include "base/files/file_path.h"
#include "base/synchronization/waitable_event.h"
#include "components/autofill/core/browser/webdata/autofill_webdata_service.h" #include "components/autofill/core/browser/webdata/autofill_webdata_service.h"
#include "components/webdata/common/web_data_service_consumer.h" #include "components/webdata/common/web_data_service_consumer.h"
#include "components/webdata/common/web_database_service.h" #include "components/webdata/common/web_database_service.h"
namespace base {
class WaitableEvent;
};
namespace android_webview { namespace android_webview {
// Handles the database operations necessary to implement the autocomplete // Handles the database operations necessary to implement the autocomplete
...@@ -42,15 +45,16 @@ class AwFormDatabaseService : public WebDataServiceConsumer { ...@@ -42,15 +45,16 @@ class AwFormDatabaseService : public WebDataServiceConsumer {
const WDTypedResult* result) OVERRIDE; const WDTypedResult* result) OVERRIDE;
private: private:
// Cancels the currently pending WebDataService query, if there is one. struct PendingQuery {
void CancelPendingQuery(); bool* result;
base::WaitableEvent* completion;
};
typedef std::map<WebDataServiceBase::Handle, PendingQuery> QueryMap;
void HasFormDataImpl(); void ClearFormDataImpl();
void HasFormDataImpl(base::WaitableEvent* completion, bool* result);
// Stores the query handle when an async database query is executed. QueryMap result_map_;
WebDataServiceBase::Handle pending_query_handle_;
bool has_form_data_;
base::WaitableEvent completion_;
scoped_refptr<autofill::AutofillWebDataService> autofill_data_; scoped_refptr<autofill::AutofillWebDataService> autofill_data_;
scoped_refptr<WebDatabaseService> web_database_; scoped_refptr<WebDatabaseService> web_database_;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment