diff options
author | finnur@chromium.org <finnur@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-03-25 03:00:49 +0000 |
---|---|---|
committer | finnur@chromium.org <finnur@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2010-03-25 03:00:49 +0000 |
commit | 27fbcccda73e7bd257c20f0bbf4b4c31af15c9c2 (patch) | |
tree | 1733e5af0ff85042f99f9165c93015e7df6b40c0 | |
parent | a25fa87a6cf2d10414454d8b8ad75ffeb13fb2cd (diff) | |
download | chromium_src-27fbcccda73e7bd257c20f0bbf4b4c31af15c9c2.zip chromium_src-27fbcccda73e7bd257c20f0bbf4b4c31af15c9c2.tar.gz chromium_src-27fbcccda73e7bd257c20f0bbf4b4c31af15c9c2.tar.bz2 |
Ensure all callers to GetFilePath are on the File thread
by DCHECKing based on the thread type.
TEST=Extensions and automated tests should work as before
BUG=http://crbug.com/38521
Review URL: http://codereview.chromium.org/1293001
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@42578 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r-- | chrome/browser/browser_process_impl.cc | 7 | ||||
-rw-r--r-- | chrome/browser/extensions/extension_dom_ui.cc | 2 | ||||
-rw-r--r-- | chrome/browser/extensions/extension_protocols.cc | 5 | ||||
-rw-r--r-- | chrome/common/extensions/extension_resource.cc | 27 | ||||
-rw-r--r-- | chrome/common/extensions/extension_resource.h | 41 |
5 files changed, 66 insertions, 16 deletions
diff --git a/chrome/browser/browser_process_impl.cc b/chrome/browser/browser_process_impl.cc index 30f0b24..afaf666 100644 --- a/chrome/browser/browser_process_impl.cc +++ b/chrome/browser/browser_process_impl.cc @@ -42,6 +42,7 @@ #include "chrome/common/chrome_constants.h" #include "chrome/common/chrome_paths.h" #include "chrome/common/chrome_switches.h" +#include "chrome/common/extensions/extension_resource.h" #include "chrome/common/extensions/extension_l10n_util.h" #include "chrome/common/notification_service.h" #include "chrome/common/pref_names.h" @@ -335,6 +336,12 @@ void BrowserProcessImpl::CreateFileThread() { if (!thread->StartWithOptions(options)) return; file_thread_.swap(thread); + + // ExtensionResource is in chrome/common, so it cannot depend on + // chrome/browser, which means it cannot lookup what the File thread is. + // We therefore store the thread ID from here so we can validate the proper + // thread usage in the ExtensionResource class. + ExtensionResource::set_file_thread_id(file_thread_->thread_id()); } void BrowserProcessImpl::CreateDBThread() { diff --git a/chrome/browser/extensions/extension_dom_ui.cc b/chrome/browser/extensions/extension_dom_ui.cc index 0a7e5d1..7ee7339 100644 --- a/chrome/browser/extensions/extension_dom_ui.cc +++ b/chrome/browser/extensions/extension_dom_ui.cc @@ -322,5 +322,5 @@ RefCountedMemory* ExtensionDOMUI::GetFaviconResourceBytes(Profile* profile, // TODO(arv): Move this off of the UI thread and onto the File thread. If // possible to do this asynchronously, use ImageLoadingTracker. return ReadFileData(extension->GetIconPath( - Extension::EXTENSION_ICON_BITTY).GetFilePath()); + Extension::EXTENSION_ICON_BITTY).GetFilePathOnAnyThreadHack()); } diff --git a/chrome/browser/extensions/extension_protocols.cc b/chrome/browser/extensions/extension_protocols.cc index b6e66f6..0dd4e7e 100644 --- a/chrome/browser/extensions/extension_protocols.cc +++ b/chrome/browser/extensions/extension_protocols.cc @@ -1,4 +1,4 @@ -// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved. +// Copyright (c) 2010 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. @@ -44,7 +44,8 @@ static URLRequestJob* CreateExtensionURLRequestJob(URLRequest* request, ExtensionResource resource(directory_path, extension_file_util::ExtensionURLToRelativeFilePath(request->url())); - return new URLRequestFileJob(request, resource.GetFilePath()); + return new URLRequestFileJob(request, + resource.GetFilePathOnAnyThreadHack()); } // Factory registered with URLRequest to create URLRequestJobs for diff --git a/chrome/common/extensions/extension_resource.cc b/chrome/common/extensions/extension_resource.cc index 8eed2e9..eda2d26 100644 --- a/chrome/common/extensions/extension_resource.cc +++ b/chrome/common/extensions/extension_resource.cc @@ -7,6 +7,10 @@ #include "base/file_util.h" #include "base/logging.h" +PlatformThreadId ExtensionResource::file_thread_id_ = 0; + +bool ExtensionResource::check_for_file_thread_ = false; + ExtensionResource::ExtensionResource() { } @@ -16,7 +20,7 @@ ExtensionResource::ExtensionResource(const FilePath& extension_root, relative_path_(relative_path) { } -const FilePath& ExtensionResource::GetFilePath() const { +const FilePath& ExtensionResource::GetFilePathOnAnyThreadHack() const { if (extension_root_.empty() || relative_path_.empty()) { DCHECK(full_resource_path_.empty()); return full_resource_path_; @@ -26,13 +30,20 @@ const FilePath& ExtensionResource::GetFilePath() const { if (!full_resource_path_.empty()) return full_resource_path_; - full_resource_path_ = GetFilePath(extension_root_, relative_path_); + full_resource_path_ = + GetFilePathOnAnyThreadHack(extension_root_, relative_path_); return full_resource_path_; } +const FilePath& ExtensionResource::GetFilePath() const { + DCHECK(!check_for_file_thread_ || + ExtensionResource::file_thread_id_ == PlatformThread::CurrentId()); + return GetFilePathOnAnyThreadHack(); +} + // static -FilePath ExtensionResource::GetFilePath(const FilePath& extension_root, - const FilePath& relative_path) { +FilePath ExtensionResource::GetFilePathOnAnyThreadHack( + const FilePath& extension_root, const FilePath& relative_path) { // We need to resolve the parent references in the extension_root // path on its own because IsParent doesn't like parent references. FilePath clean_extension_root(extension_root); @@ -57,6 +68,14 @@ FilePath ExtensionResource::GetFilePath(const FilePath& extension_root, return FilePath(); } +// static +FilePath ExtensionResource::GetFilePath( + const FilePath& extension_root, const FilePath& relative_path) { + DCHECK(!check_for_file_thread_ || + ExtensionResource::file_thread_id_ == PlatformThread::CurrentId()); + return GetFilePathOnAnyThreadHack(extension_root, relative_path); +} + // Unit-testing helpers. FilePath::StringType ExtensionResource::NormalizeSeperators( FilePath::StringType path) const { diff --git a/chrome/common/extensions/extension_resource.h b/chrome/common/extensions/extension_resource.h index e87787d..755c959 100644 --- a/chrome/common/extensions/extension_resource.h +++ b/chrome/common/extensions/extension_resource.h @@ -6,6 +6,7 @@ #define CHROME_COMMON_EXTENSIONS_EXTENSION_RESOURCE_H_ #include "base/file_path.h" +#include "base/platform_thread.h" // Represents a resource inside an extension. For example, an image, or a // JavaScript file. This is more complicated than just a simple FilePath @@ -18,20 +19,33 @@ class ExtensionResource { ExtensionResource(const FilePath& extension_root, const FilePath& relative_path); - // Returns actual path to the resource (default or locale specific). - // *** MIGHT HIT FILESYSTEM. Do not call on UI thread! *** - // NOTE: If you need to access ExtensionResources, such as bitmaps, on the UI - // thread, please use the ImageLoadingTracker, which uses the File thread. + // Returns actual path to the resource (default or locale specific). In the + // browser process, this will DCHECK if not called on the file thread. To + // easily load extension images on the UI thread, see ImageLoadingTracker. const FilePath& GetFilePath() const; - // Static version to avoid creating an instance of ExtensionResource - // when all we want is the localization code. - // *** MIGHT HIT FILESYSTEM. Do not call on UI thread! *** - // NOTE: If you need to access ExtensionResources, such as bitmaps, on the UI - // thread, please use the ImageLoadingTracker, which uses the File thread. + // Gets the physical file path for the extension resource, taking into account + // localization. In the browser process, this will DCHECK if not called on the + // file thread. To easily load extension images on the UI thread, see + // ImageLoadingTracker. static FilePath GetFilePath(const FilePath& extension_root, const FilePath& relative_path); + // Gets the file path on any thread. Unlike GetFilePath(), these can be called + // from any thread without DCHECKing. + // + // In the browser process, calling this method is almost always wrong. Use + // GetFilePath() on the file thread instead. + const FilePath& GetFilePathOnAnyThreadHack() const; + static FilePath GetFilePathOnAnyThreadHack(const FilePath& extension_root, + const FilePath& relative_path); + + // Setter for the proper thread to run file tasks on. + static void set_file_thread_id(PlatformThreadId thread_id) { + file_thread_id_ = thread_id; + check_for_file_thread_ = true; + } + // Getters const FilePath& extension_root() const { return extension_root_; } const FilePath& relative_path() const { return relative_path_; } @@ -49,6 +63,15 @@ class ExtensionResource { // Full path to extension resource. Starts empty. mutable FilePath full_resource_path_; + + // The thread id for the file thread. If set, GetFilePath() and related + // methods will DCHECK that they are on this thread when called. + static PlatformThreadId file_thread_id_; + + // Whether to check for file thread. See |file_thread_id_|. If set, + // GetFilePath() and related methods will DCHECK that they are on this thread + // when called. + static bool check_for_file_thread_; }; #endif // CHROME_COMMON_EXTENSIONS_EXTENSION_RESOURCE_H_ |