diff options
author | jcampan@chromium.org <jcampan@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-02-04 19:38:30 +0000 |
---|---|---|
committer | jcampan@chromium.org <jcampan@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-02-04 19:38:30 +0000 |
commit | a3153c47d3b5308e7a55c0924a0baa1a6e02dcd0 (patch) | |
tree | b37c85641aef8df14c732e4b187444e040a11555 /base/win_util.cc | |
parent | a93e517bfe25f11b2c4cc7d69f386424b93a2efa (diff) | |
download | chromium_src-a3153c47d3b5308e7a55c0924a0baa1a6e02dcd0.zip chromium_src-a3153c47d3b5308e7a55c0924a0baa1a6e02dcd0.tar.gz chromium_src-a3153c47d3b5308e7a55c0924a0baa1a6e02dcd0.tar.bz2 |
This CL adds macro used to track the creation and destruction
of HWNDs, in an attempt to detect potential double-delete.
A double-delete of a HWND might be responsible for the crasher
http://crbug.com/4714
Note: this CL was previously committed and reverted because it broke the sandbox integration module.
Review URL: http://codereview.chromium.org/21032
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@9161 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'base/win_util.cc')
-rw-r--r-- | base/win_util.cc | 63 |
1 files changed, 63 insertions, 0 deletions
diff --git a/base/win_util.cc b/base/win_util.cc index cfd7f2b..e18007b 100644 --- a/base/win_util.cc +++ b/base/win_util.cc @@ -4,12 +4,15 @@ #include "base/win_util.h" +#include <map> #include <sddl.h> #include "base/logging.h" #include "base/registry.h" #include "base/scoped_handle.h" +#include "base/singleton.h" #include "base/string_util.h" +#include "base/tracked.h" namespace win_util { @@ -361,6 +364,66 @@ std::wstring FormatLastWin32Error() { return FormatMessage(GetLastError()); } +typedef std::map<HWND, tracked_objects::Location> HWNDInfoMap; +struct HWNDBirthMapTrait : public DefaultSingletonTraits<HWNDInfoMap> { +}; +struct HWNDDeathMapTrait : public DefaultSingletonTraits<HWNDInfoMap> { +}; + +void NotifyHWNDCreation(const tracked_objects::Location& from_here, HWND hwnd) { + HWNDInfoMap* birth_map = Singleton<HWNDInfoMap, HWNDBirthMapTrait>::get(); + HWNDInfoMap::iterator birth_iter = birth_map->find(hwnd); + if (birth_iter != birth_map->end()) { + birth_map->erase(birth_iter); + + // We have already seen this HWND, was it destroyed? + HWNDInfoMap* death_map = Singleton<HWNDInfoMap, HWNDDeathMapTrait>::get(); + HWNDInfoMap::iterator death_iter = death_map->find(hwnd); + if (death_iter == death_map->end()) { + // We did not get a destruction notification. The code is probably not + // calling NotifyHWNDDestruction for that HWND. + NOTREACHED() << "Creation of HWND reported for already tracked HWND. The " + "HWND destruction is probably not tracked properly. " + "Fix it!"; + } else { + death_map->erase(death_iter); + } + } + birth_map->insert(std::pair<HWND, tracked_objects::Location>(hwnd, + from_here)); +} + +void NotifyHWNDDestruction(const tracked_objects::Location& from_here, + HWND hwnd) { + HWNDInfoMap* death_map = Singleton<HWNDInfoMap, HWNDDeathMapTrait>::get(); + HWNDInfoMap::iterator death_iter = death_map->find(hwnd); + + HWNDInfoMap* birth_map = Singleton<HWNDInfoMap, HWNDBirthMapTrait>::get(); + HWNDInfoMap::iterator birth_iter = birth_map->find(hwnd); + + if (death_iter != death_map->end()) { + std::string allocation, first_delete, second_delete; + if (birth_iter != birth_map->end()) + birth_iter->second.Write(true, true, &allocation); + death_iter->second.Write(true, true, &first_delete); + from_here.Write(true, true, &second_delete); + NOTREACHED() << "Double delete of an HWND. Please file a bug with info on " + "how you got that assertion and the following information:\n" + "Double delete of HWND 0x" << hwnd << "\n" << + "Allocated at " << allocation << "\n" << + "Deleted first at " << first_delete << "\n" << + "Deleted again at " << second_delete; + death_map->erase(death_iter); + } + + if (birth_iter == birth_map->end()) { + NOTREACHED() << "Destruction of HWND reported for unknown HWND. The HWND " + "construction is probably not tracked properly. Fix it!"; + } + death_map->insert(std::pair<HWND, tracked_objects::Location>(hwnd, + from_here)); +} + } // namespace win_util #ifdef _MSC_VER |