summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorskrul@chromium.org <skrul@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-01-04 18:40:25 +0000
committerskrul@chromium.org <skrul@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-01-04 18:40:25 +0000
commit6018becf8a0c8cb721f811b8839f6c28118698d7 (patch)
tree875c9d4881aa64c5fbd6412202ee67feb0c80ae6
parent6c74c2a80e7d2664308b932c7cd6787ef7aea20f (diff)
downloadchromium_src-6018becf8a0c8cb721f811b8839f6c28118698d7.zip
chromium_src-6018becf8a0c8cb721f811b8839f6c28118698d7.tar.gz
chromium_src-6018becf8a0c8cb721f811b8839f6c28118698d7.tar.bz2
Fix leak from cl 524003 (Change WDS to use the DB thread...)
This is a second try of cl 524003 -- the previous version caused a leak. The only change in this patch is the addition of the ui_thread_ member in TemplateURLModelTest and the corresponding initializer. This allows the DeleteOnUIThread trait of the WebDataService to do its job. Review URL: http://codereview.chromium.org/522025 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@35462 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r--chrome/browser/search_engines/template_url_model_unittest.cc25
-rw-r--r--chrome/browser/webdata/web_data_service.cc45
-rw-r--r--chrome/browser/webdata/web_data_service.h13
-rw-r--r--chrome/browser/webdata/web_data_service_unittest.cc6
4 files changed, 40 insertions, 49 deletions
diff --git a/chrome/browser/search_engines/template_url_model_unittest.cc b/chrome/browser/search_engines/template_url_model_unittest.cc
index 1c50fc1..7845421 100644
--- a/chrome/browser/search_engines/template_url_model_unittest.cc
+++ b/chrome/browser/search_engines/template_url_model_unittest.cc
@@ -4,6 +4,7 @@
#include "base/path_service.h"
#include "base/thread.h"
+#include "chrome/browser/chrome_thread.h"
#include "chrome/browser/history/history_notifications.h"
#include "chrome/browser/webdata/web_database.h"
#include "chrome/test/testing_profile.h"
@@ -34,6 +35,9 @@ class TemplateURLModelTestingProfile : public TestingProfile {
TemplateURLModelTestingProfile() : TestingProfile() {}
void SetUp() {
+ db_thread_.reset(new ChromeThread(ChromeThread::DB));
+ db_thread_->Start();
+
// Name a subdirectory of the temp directory.
ASSERT_TRUE(PathService::Get(base::DIR_TEMP, &test_dir_));
test_dir_ = test_dir_.AppendASCII("TemplateURLModelTest");
@@ -50,6 +54,12 @@ class TemplateURLModelTestingProfile : public TestingProfile {
void TearDown() {
// Clean up the test directory.
service_->Shutdown();
+ // Note that we must ensure the DB thread is stopped after WDS
+ // shutdown (so it can commit pending transactions) but before
+ // deleting the test profile directory, otherwise we may not be
+ // able to delete it due to an open transaction.
+ db_thread_->Stop();
+
ASSERT_TRUE(file_util::Delete(test_dir_, true));
ASSERT_FALSE(file_util::PathExists(test_dir_));
}
@@ -61,6 +71,7 @@ class TemplateURLModelTestingProfile : public TestingProfile {
private:
scoped_refptr<WebDataService> service_;
FilePath test_dir_;
+ scoped_ptr<ChromeThread> db_thread_;
};
// Trivial subclass of TemplateURLModel that records the last invocation of
@@ -93,7 +104,8 @@ class TestingTemplateURLModel : public TemplateURLModel {
class TemplateURLModelTest : public testing::Test,
public TemplateURLModelObserver {
public:
- TemplateURLModelTest() : changed_count_(0) {
+ TemplateURLModelTest() : ui_thread_(ChromeThread::UI, &message_loop_),
+ changed_count_(0) {
}
virtual void SetUp() {
@@ -142,10 +154,9 @@ class TemplateURLModelTest : public testing::Test,
// Blocks the caller until the service has finished servicing all pending
// requests.
void BlockTillServiceProcessesRequests() {
- // Schedule a task on the background thread that is processed after all
- // pending requests on the background thread.
- profile_->GetWebDataService(Profile::EXPLICIT_ACCESS)->thread()->
- message_loop()->PostTask(FROM_HERE, new QuitTask2());
+ // Schedule a task on the DB thread that is processed after all
+ // pending requests on the DB thread.
+ ChromeThread::PostTask(ChromeThread::DB, FROM_HERE, new QuitTask2());
// Run the current message loop. QuitTask2, when run, invokes Quit,
// which unblocks this.
MessageLoop::current()->Run();
@@ -191,6 +202,9 @@ class TemplateURLModelTest : public testing::Test,
}
MessageLoopForUI message_loop_;
+ // Needed to make the DeleteOnUIThread trait of WebDataService work
+ // properly.
+ ChromeThread ui_thread_;
scoped_ptr<TemplateURLModelTestingProfile> profile_;
scoped_ptr<TestingTemplateURLModel> model_;
int changed_count_;
@@ -756,4 +770,3 @@ TEST_F(TemplateURLModelTest, FailedInit) {
ASSERT_TRUE(model_->GetDefaultSearchProvider());
}
-
diff --git a/chrome/browser/webdata/web_data_service.cc b/chrome/browser/webdata/web_data_service.cc
index 4407cdd..d25492b 100644
--- a/chrome/browser/webdata/web_data_service.cc
+++ b/chrome/browser/webdata/web_data_service.cc
@@ -29,7 +29,7 @@ using webkit_glue::FormField;
using webkit_glue::PasswordForm;
WebDataService::WebDataService()
- : thread_(NULL),
+ : is_running_(false),
db_(NULL),
failed_init_(false),
should_commit_(false),
@@ -38,8 +38,8 @@ WebDataService::WebDataService()
}
WebDataService::~WebDataService() {
- if (thread_) {
- Shutdown();
+ if (is_running_ && db_) {
+ DLOG_ASSERT("WebDataService dtor called without Shutdown");
}
}
@@ -51,47 +51,18 @@ bool WebDataService::Init(const FilePath& profile_path) {
bool WebDataService::InitWithPath(const FilePath& path) {
path_ = path;
-
- thread_ = new base::Thread("Chrome_WebDataThread");
- if (!thread_->Start()) {
- delete thread_;
- thread_ = NULL;
- return false;
- }
-
+ is_running_ = true;
ScheduleTask(NewRunnableMethod(this,
&WebDataService::InitializeDatabaseIfNecessary));
return true;
}
-class ShutdownTask : public Task {
- public:
- explicit ShutdownTask(WebDataService* wds) : service_(wds) {
- }
- virtual void Run() {
- service_->ShutdownDatabase();
- }
-
- private:
-
- WebDataService* service_;
-};
-
void WebDataService::Shutdown() {
- if (thread_) {
- // We cannot use NewRunnableMethod() because this can be called from our
- // destructor. NewRunnableMethod() would AddRef() this instance.
- ScheduleTask(new ShutdownTask(this));
-
- // The thread destructor sends a message to terminate the thread and waits
- // until the thread has exited.
- delete thread_;
- thread_ = NULL;
- }
+ UnloadDatabase();
}
bool WebDataService::IsRunning() const {
- return thread_ != NULL;
+ return is_running_;
}
void WebDataService::UnloadDatabase() {
@@ -106,8 +77,8 @@ void WebDataService::ScheduleCommit() {
}
void WebDataService::ScheduleTask(Task* t) {
- if (thread_)
- thread_->message_loop()->PostTask(FROM_HERE, t);
+ if (is_running_)
+ ChromeThread::PostTask(ChromeThread::DB, FROM_HERE, t);
else
NOTREACHED() << "Task scheduled after Shutdown()";
}
diff --git a/chrome/browser/webdata/web_data_service.h b/chrome/browser/webdata/web_data_service.h
index 3a6c83b..4421b71 100644
--- a/chrome/browser/webdata/web_data_service.h
+++ b/chrome/browser/webdata/web_data_service.h
@@ -12,6 +12,7 @@
#include "base/file_path.h"
#include "base/lock.h"
#include "base/ref_counted.h"
+#include "chrome/browser/chrome_thread.h"
#include "chrome/browser/search_engines/template_url.h"
#include "third_party/skia/include/core/SkBitmap.h"
#include "webkit/glue/form_field.h"
@@ -145,7 +146,9 @@ template <class T> class WDObjectResult : public WDTypedResult {
class WebDataServiceConsumer;
-class WebDataService : public base::RefCountedThreadSafe<WebDataService> {
+class WebDataService
+ : public base::RefCountedThreadSafe<WebDataService,
+ ChromeThread::DeleteOnUIThread> {
public:
// All requests return an opaque handle of the following type.
@@ -429,6 +432,8 @@ class WebDataService : public base::RefCountedThreadSafe<WebDataService> {
//////////////////////////////////////////////////////////////////////////////
private:
friend class base::RefCountedThreadSafe<WebDataService>;
+ friend class ChromeThread;
+ friend class DeleteTask<WebDataService>;
friend class ShutdownTask;
typedef GenericRequest2<std::vector<const TemplateURL*>,
@@ -507,8 +512,6 @@ class WebDataService : public base::RefCountedThreadSafe<WebDataService> {
void GetWebAppImagesImpl(GenericRequest<GURL>* request);
- base::Thread* thread() { return thread_; }
-
// Schedule a task on our worker thread.
void ScheduleTask(Task* t);
@@ -518,8 +521,8 @@ class WebDataService : public base::RefCountedThreadSafe<WebDataService> {
// Return the next request handle.
int GetNextRequestHandle();
- // Our worker thread. All requests are processed from that thread.
- base::Thread* thread_;
+ // True once initialization has started.
+ bool is_running_;
// The path with which to initialize the database.
FilePath path_;
diff --git a/chrome/browser/webdata/web_data_service_unittest.cc b/chrome/browser/webdata/web_data_service_unittest.cc
index 1e26539..7c92725 100644
--- a/chrome/browser/webdata/web_data_service_unittest.cc
+++ b/chrome/browser/webdata/web_data_service_unittest.cc
@@ -71,10 +71,12 @@ class AutofillWebDataServiceConsumer: public WebDataServiceConsumer {
class WebDataServiceTest : public testing::Test {
public:
WebDataServiceTest()
- : ui_thread_(ChromeThread::UI, &message_loop_) {}
+ : ui_thread_(ChromeThread::UI, &message_loop_),
+ db_thread_(ChromeThread::DB) {}
protected:
virtual void SetUp() {
+ db_thread_.Start();
name1_ = ASCIIToUTF16("name1");
name2_ = ASCIIToUTF16("name2");
value1_ = ASCIIToUTF16("value1");
@@ -94,6 +96,7 @@ class WebDataServiceTest : public testing::Test {
wds_->Shutdown();
file_util::Delete(profile_dir_, true);
+ db_thread_.Stop();
MessageLoop::current()->PostTask(FROM_HERE, new MessageLoop::QuitTask);
MessageLoop::current()->Run();
}
@@ -110,6 +113,7 @@ class WebDataServiceTest : public testing::Test {
MessageLoopForUI message_loop_;
ChromeThread ui_thread_;
+ ChromeThread db_thread_;
string16 name1_;
string16 name2_;
string16 value1_;