diff options
author | tommi@chromium.org <tommi@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2011-03-03 23:39:32 +0000 |
---|---|---|
committer | tommi@chromium.org <tommi@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2011-03-03 23:39:32 +0000 |
commit | 6c1319541aee6ff89c06457b46d46bbcbcfe868f (patch) | |
tree | 45eda090e3e17f5c7247dd4b1ec6649bfde4d04d | |
parent | 4006448408b35ca6be5aef5db1be1e966d31eee4 (diff) | |
download | chromium_src-6c1319541aee6ff89c06457b46d46bbcbcfe868f.zip chromium_src-6c1319541aee6ff89c06457b46d46bbcbcfe868f.tar.gz chromium_src-6c1319541aee6ff89c06457b46d46bbcbcfe868f.tar.bz2 |
Instrument the allocator code so that we can track down the cause of recent crashes.
BUG=74777
TEST=This should give us a clearer picture of what's going wrong in the recent courgette crashes.
Review URL: http://codereview.chromium.org/6611027
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@76834 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r-- | courgette/assembly_program.cc | 3 | ||||
-rw-r--r-- | courgette/memory_allocator.cc | 89 | ||||
-rw-r--r-- | courgette/memory_allocator.h | 18 |
3 files changed, 66 insertions, 44 deletions
diff --git a/courgette/assembly_program.cc b/courgette/assembly_program.cc index 345ca89..1dc9ecf 100644 --- a/courgette/assembly_program.cc +++ b/courgette/assembly_program.cc @@ -289,6 +289,9 @@ void AssemblyProgram::AssignRemainingIndexes(RVAToLabel* labels) { typedef void (EncodedProgram::*DefineLabelMethod)(int index, RVA value); +#if defined(OS_WIN) +__declspec(noinline) +#endif static void DefineLabels(const RVAToLabel& labels, EncodedProgram* encoded_format, DefineLabelMethod define_label) { diff --git a/courgette/memory_allocator.cc b/courgette/memory_allocator.cc index 06a0653..d45d053 100644 --- a/courgette/memory_allocator.cc +++ b/courgette/memory_allocator.cc @@ -7,9 +7,21 @@ #include <map> #include "base/file_util.h" +#include "base/stringprintf.h" #ifdef OS_WIN +// A helper function to help diagnose failures we've seen in the field. +// The function constructs a string containing the error and throws a runtime +// exception. The calling convention is set to stdcall to force argument +// passing via the stack. +__declspec(noinline) +void __stdcall RuntimeError(DWORD err) { + char buffer[20] = {0}; + wsprintfA(buffer, "err: %u", err); + throw buffer; +} + namespace courgette { // TempFile @@ -21,6 +33,13 @@ TempFile::~TempFile() { Close(); } +FilePath TempFile::PrepareTempFile() { + FilePath path; + if (!file_util::CreateTemporaryFile(&path)) + RuntimeError(::GetLastError()); + return path; +} + void TempFile::Close() { if (valid()) { base::ClosePlatformFile(file_); @@ -29,21 +48,18 @@ void TempFile::Close() { } } -bool TempFile::Create() { +void TempFile::Create() { DCHECK(file_ == base::kInvalidPlatformFileValue); - FilePath path; - if (file_util::CreateTemporaryFile(&path)) { - bool created = false; - base::PlatformFileError error_code = base::PLATFORM_FILE_OK; - int flags = base::PLATFORM_FILE_OPEN_ALWAYS | base::PLATFORM_FILE_READ | - base::PLATFORM_FILE_WRITE | - base::PLATFORM_FILE_DELETE_ON_CLOSE | - base::PLATFORM_FILE_TEMPORARY; - file_ = base::CreatePlatformFile(path, flags, &created, &error_code); - PLOG_IF(ERROR, file_ == base::kInvalidPlatformFileValue) - << "CreatePlatformFile"; - } - return valid(); + FilePath path(PrepareTempFile()); + bool created = false; + base::PlatformFileError error_code = base::PLATFORM_FILE_OK; + int flags = base::PLATFORM_FILE_OPEN_ALWAYS | base::PLATFORM_FILE_READ | + base::PLATFORM_FILE_WRITE | + base::PLATFORM_FILE_DELETE_ON_CLOSE | + base::PLATFORM_FILE_TEMPORARY; + file_ = base::CreatePlatformFile(path, flags, &created, &error_code); + if (file_ == base::kInvalidPlatformFileValue) + RuntimeError(error_code); } bool TempFile::valid() const { @@ -58,11 +74,11 @@ size_t TempFile::size() const { return size_; } -bool TempFile::SetSize(size_t size) { - bool ret = base::TruncatePlatformFile(file_, size); - if (ret) - size_ = size; - return ret; +void TempFile::SetSize(size_t size) { + bool set = base::TruncatePlatformFile(file_, size); + if (!set) + RuntimeError(::GetLastError()); + size_ = size; } // FileMapping @@ -74,19 +90,22 @@ FileMapping::~FileMapping() { Close(); } -bool FileMapping::Create(HANDLE file, size_t size) { +void FileMapping::InitializeView(size_t size) { + DCHECK(view_ == NULL); + DCHECK(mapping_ != NULL); + view_ = ::MapViewOfFile(mapping_, FILE_MAP_WRITE, 0, 0, size); + if (!view_) + RuntimeError(::GetLastError()); +} + +void FileMapping::Create(HANDLE file, size_t size) { DCHECK(file != INVALID_HANDLE_VALUE); DCHECK(!valid()); mapping_ = ::CreateFileMapping(file, NULL, PAGE_READWRITE, 0, 0, NULL); - if (mapping_) - view_ = ::MapViewOfFile(mapping_, FILE_MAP_WRITE, 0, 0, size); + if (!mapping_) + RuntimeError(::GetLastError()); - if (!valid()) { - PLOG(ERROR) << "Failed to map file"; - Close(); - } - - return valid(); + InitializeView(size); } void FileMapping::Close() { @@ -114,19 +133,17 @@ TempMapping::TempMapping() { TempMapping::~TempMapping() { } -bool TempMapping::Initialize(size_t size) { +void TempMapping::Initialize(size_t size) { // TODO(tommi): The assumption here is that the alignment of pointers (this) // is as strict or stricter than the alignment of the element type. This is // not always true, e.g. __m128 has 16-byte alignment. size += sizeof(this); - bool ret = file_.Create() && file_.SetSize(size) && - mapping_.Create(file_.handle(), size); - if (ret) { - TempMapping** write = reinterpret_cast<TempMapping**>(mapping_.view()); - write[0] = this; - } + file_.Create(); + file_.SetSize(size); + mapping_.Create(file_.handle(), size); - return ret; + TempMapping** write = reinterpret_cast<TempMapping**>(mapping_.view()); + write[0] = this; } void* TempMapping::memory() const { diff --git a/courgette/memory_allocator.h b/courgette/memory_allocator.h index 8e06e65..fce1e25 100644 --- a/courgette/memory_allocator.h +++ b/courgette/memory_allocator.h @@ -8,6 +8,7 @@ #include <memory> #include "base/basictypes.h" +#include "base/file_path.h" #include "base/logging.h" #include "base/platform_file.h" @@ -24,9 +25,9 @@ class TempFile { TempFile(); ~TempFile(); - bool Create(); + __declspec(noinline) void Create(); void Close(); - bool SetSize(size_t size); + __declspec(noinline) void SetSize(size_t size); // Returns true iff the temp file is currently open. bool valid() const; @@ -40,6 +41,8 @@ class TempFile { size_t size() const; protected: + __declspec(noinline) FilePath PrepareTempFile(); + base::PlatformFile file_; size_t size_; }; @@ -51,7 +54,7 @@ class FileMapping { ~FileMapping(); // Map a file from beginning to |size|. - bool Create(HANDLE file, size_t size); + __declspec(noinline) void Create(HANDLE file, size_t size); void Close(); // Returns true iff a mapping has been created. @@ -62,6 +65,8 @@ class FileMapping { void* view() const; protected: + __declspec(noinline) void InitializeView(size_t size); + HANDLE mapping_; void* view_; }; @@ -77,7 +82,7 @@ class TempMapping { // Creates a temporary file of size |size| and maps it into the current // process' address space. - bool Initialize(size_t size); + __declspec(noinline) void Initialize(size_t size); // Returns a writable pointer to the reserved memory. void* memory() const; @@ -179,10 +184,7 @@ class MemoryAllocator { // If either the heap allocation failed or the request exceeds the // max heap allocation threshold, we back the allocation with a temp file. TempMapping* mapping = new TempMapping(); - if (!mapping->Initialize(bytes)) { - delete mapping; - throw std::bad_alloc("TempMapping::Initialize"); - } + mapping->Initialize(bytes); mem = reinterpret_cast<uint8*>(mapping->memory()); mem[0] = static_cast<uint8>(FILE_ALLOCATION); } |