diff options
author | deanm@chromium.org <deanm@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2008-11-03 19:29:36 +0000 |
---|---|---|
committer | deanm@chromium.org <deanm@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2008-11-03 19:29:36 +0000 |
commit | ec565e6000dc1304dd92f4c5c9c0bd525fc194ea (patch) | |
tree | 7f41d7d7c831fb8b3e98bf5b24d783a09f8ad509 | |
parent | 9999b28465395531a59c8c512ea80a41c7e99bbc (diff) | |
download | chromium_src-ec565e6000dc1304dd92f4c5c9c0bd525fc194ea.zip chromium_src-ec565e6000dc1304dd92f4c5c9c0bd525fc194ea.tar.gz chromium_src-ec565e6000dc1304dd92f4c5c9c0bd525fc194ea.tar.bz2 |
Tracing showed that the resource bundle lock was held for long periods of time, as much as 3.5ms. Altough the lock is not contended in this case, be more precise about when we need to hold the lock. Also clean up a static initialization to use LazyInstance.
Review URL: http://codereview.chromium.org/9246
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@4472 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r-- | chrome/common/resource_bundle.cc | 62 |
1 files changed, 37 insertions, 25 deletions
diff --git a/chrome/common/resource_bundle.cc b/chrome/common/resource_bundle.cc index 2676bd6..c7cbad1 100644 --- a/chrome/common/resource_bundle.cc +++ b/chrome/common/resource_bundle.cc @@ -146,42 +146,54 @@ SkBitmap* ResourceBundle::LoadBitmap(HINSTANCE dll_inst, int resource_id) { } SkBitmap* ResourceBundle::GetBitmapNamed(int resource_id) { - AutoLock lock_scope(lock_); - - SkImageMap::const_iterator found = skia_images_.find(resource_id); - SkBitmap* bitmap = NULL; - - // If not found load and store the image - if (found == skia_images_.end()) { - // Try to load the bitmap from the theme dll. - if (theme_dll_) - bitmap = LoadBitmap(theme_dll_, resource_id); - // We did not find the bitmap in the theme DLL, try the current one. - if (!bitmap) - bitmap = LoadBitmap(_AtlBaseModule.GetModuleInstance(), resource_id); - skia_images_[resource_id] = bitmap; - } else { - bitmap = found->second; + // Check to see if we already have the Skia image in the cache. + { + AutoLock lock_scope(lock_); + SkImageMap::const_iterator found = skia_images_.find(resource_id); + if (found != skia_images_.end()) + return found->second; } - // This bitmap will be returned when a bitmap is requested that can not be - // found. The data inside is lazily initialized, so users must lock and - static SkBitmap* empty_bitmap = NULL; + scoped_ptr<SkBitmap> bitmap; + + // Try to load the bitmap from the theme dll. + if (theme_dll_) + bitmap.reset(LoadBitmap(theme_dll_, resource_id)); + + // If we did not find the bitmap in the theme DLL, try the current one. + if (!bitmap.get()) + bitmap.reset(LoadBitmap(_AtlBaseModule.GetModuleInstance(), resource_id)); + + // We loaded successfully. Cache the Skia version of the bitmap. + if (bitmap.get()) { + AutoLock lock_scope(lock_); - // Handle the case where loading the bitmap failed. - if (!bitmap) { + // Another thread raced us, and has already cached the skia image. + if (skia_images_.count(resource_id)) + return skia_images_[resource_id]; + + skia_images_[resource_id] = bitmap.get(); + return bitmap.release(); + } + + // We failed to retrieve the bitmap, show a debugging red square. + { LOG(WARNING) << "Unable to load bitmap with id " << resource_id; - NOTREACHED(); // Want to assert in debug mode. + NOTREACHED(); // Want to assert in debug mode. + + AutoLock lock_scope(lock_); // Guard empty_bitmap initialization. + + static SkBitmap* empty_bitmap = NULL; if (!empty_bitmap) { - // The placeholder bitmap is bright red so people notice the problem. - empty_bitmap = new SkBitmap(); + // The placeholder bitmap is bright red so people notice the problem. + // This bitmap will be leaked, but this code should never be hit. + empty_bitmap = new SkBitmap(); empty_bitmap->setConfig(SkBitmap::kARGB_8888_Config, 32, 32); empty_bitmap->allocPixels(); empty_bitmap->eraseARGB(255, 255, 0, 0); } return empty_bitmap; } - return bitmap; } bool ResourceBundle::LoadImageResourceBytes(int resource_id, |