Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(181)

Unified Diff: src/client/linux/microdump_writer/microdump_writer.cc

Issue 1796803003: Add statistics about free space to microdump format. (Closed) Base URL: https://chromium.googlesource.com/breakpad/breakpad.git@master
Patch Set: remove #include <algorithm> Created 4 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « no previous file | no next file » | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: src/client/linux/microdump_writer/microdump_writer.cc
diff --git a/src/client/linux/microdump_writer/microdump_writer.cc b/src/client/linux/microdump_writer/microdump_writer.cc
index d459d9ec441370c2e06e2e2f9a365bedf10e6353..230ef2292ec0a416d4c3d5f9538fb5621946107c 100644
--- a/src/client/linux/microdump_writer/microdump_writer.cc
+++ b/src/client/linux/microdump_writer/microdump_writer.cc
@@ -34,8 +34,6 @@
#include <sys/utsname.h>
-#include <algorithm>
-
#include "client/linux/dump_writer_common/thread_info.h"
#include "client/linux/dump_writer_common/ucontext_reader.h"
#include "client/linux/handler/exception_handler.h"
@@ -44,6 +42,7 @@
#include "client/linux/minidump_writer/linux_ptrace_dumper.h"
#include "common/linux/file_id.h"
#include "common/linux/linux_libc_support.h"
+#include "common/memory.h"
namespace {
@@ -61,6 +60,31 @@ using google_breakpad::UContextReader;
const size_t kLineBufferSize = 2048;
+bool MappingsAreAdjacent(const MappingInfo* a, const MappingInfo* b) {
+ // Because of load biasing, we can end up with a situation where two
Primiano Tucci (use gerrit) 2016/05/06 11:07:56 Hmm I thought that after the latest linker changes
Tobias Sargeant 2016/05/06 13:29:31 It definitely came up during writing this CL. When
Primiano Tucci (use gerrit) 2016/05/06 13:58:38 https://codereview.chromium.org/1218493004 And the
+ // mappings actually overlap. So we will define adjacency to also include a
+ // b start address that lies within a's address range (including starting
+ // immediately after a).
+ // Because load biasing only ever moves the start address backwards, the end
+ // address should still increase.
+ return a->start_addr <= b->start_addr &&
+ a->start_addr + a->size >= b->start_addr;
+}
+
+size_t NextOrderedMapping(
Primiano Tucci (use gerrit) 2016/05/06 11:07:56 Not sure I get the full logic here. Why is not eno
Tobias Sargeant 2016/05/06 11:33:45 size here is mappings array length, not the size o
Primiano Tucci (use gerrit) 2016/05/06 13:29:23 Oh ok I misread it. This seems overly complicated.
Tobias Sargeant 2016/05/13 16:16:54 Could do this, although you'd have to set best = c
+ const google_breakpad::wasteful_vector<MappingInfo*>& mappings,
+ size_t curr) {
+ size_t best = mappings.size();
+ for (size_t next = 0; next < mappings.size(); ++next) {
+ if (mappings[curr]->start_addr < mappings[next]->start_addr &&
+ (best == mappings.size() ||
+ mappings[next]->start_addr < mappings[best]->start_addr)) {
+ best = next;
+ }
+ }
+ return best;
+}
+
class MicrodumpWriter {
public:
MicrodumpWriter(const ExceptionHandler::CrashContext* context,
@@ -98,6 +122,7 @@ class MicrodumpWriter {
DumpProductInformation();
DumpOSInformation();
DumpGPUInformation();
+ DumpFreeSpace();
success = DumpCrashingThread();
if (success)
success = DumpMappings();
@@ -140,6 +165,23 @@ class MicrodumpWriter {
LogAppend(hexstr);
}
+ void LogAppendDecimal(unsigned int value) {
+ const int SZ = sizeof(unsigned int) * 3;
Primiano Tucci (use gerrit) 2016/05/06 11:07:56 can we just use hex for everything, for consistenc
Tobias Sargeant 2016/05/06 11:33:45 Can do, but would still need to trim leading zeros
Primiano Tucci (use gerrit) 2016/05/06 13:29:23 LogAppend<T> will emit leading zeros only to pad t
Tobias Sargeant 2016/05/13 16:16:54 I modified LogAppend<T> to optionally strip leadin
+ if (!value) {
+ LogAppend("0");
+ return;
+ }
+ char decstr[SZ + 1];
+ decstr[SZ] = '\0';
+ int i;
+ for (i = SZ - 1; i >= 0; --i) {
+ decstr[i] = '0' + value % 10;
+ value /= 10;
+ if (!value) break;
+ }
+ LogAppend(decstr+i);
+ }
+
// Stages the buffer content hex-encoded in the current line buffer.
void LogAppend(const void* buf, size_t length) {
const uint8_t* ptr = reinterpret_cast<const uint8_t*>(buf);
@@ -391,6 +433,81 @@ class MicrodumpWriter {
LogCommitLine();
}
+ bool DumpFreeSpace() {
+ const google_breakpad::wasteful_vector<MappingInfo*>& mappings =
+ dumper_->mappings();
+ if (mappings.size() == 0) return false;
+
+ // This is complicated by the fact that mappings is not in order. It should
+ // be mostly in order, however the mapping that contains the entry point for
+ // the process is always at the front of the vector.
+
+ static const int HBITS = sizeof(size_t) * 8;
+ unsigned int hole_hist[HBITS];
Primiano Tucci (use gerrit) 2016/05/06 11:07:56 can you call this hole_histogram? For a long while
Tobias Sargeant 2016/05/06 13:29:31 Done.
+ my_memset(hole_hist, 0, sizeof(hole_hist));
+
+ // Find the lowest address mapping.
+ size_t curr = 0;
+ for (size_t i = 1; i < mappings.size(); ++i) {
+ if (mappings[i]->start_addr < mappings[curr]->start_addr) curr = i;
+ }
+
+ uintptr_t lo_addr = mappings[curr]->start_addr;
+
+ unsigned int hole_cnt = 0;
+ size_t hole_max = 0;
+ size_t hole_tot = 0;
Primiano Tucci (use gerrit) 2016/05/06 11:07:56 I'd probably rename this to hole_sum (or unmapped_
Tobias Sargeant 2016/05/06 13:29:31 Done.
+
+ while (true) {
+ // Skip to the end of an adjacent run of mappings. This is an optimization
+ // for the fact that mappings is mostly sorted.
+ while (curr != mappings.size() - 1 &&
+ MappingsAreAdjacent(mappings[curr], mappings[curr + 1]))
+ ++curr;
+
+ size_t next = NextOrderedMapping(mappings, curr);
+ if (next == mappings.size())
+ break;
Primiano Tucci (use gerrit) 2016/05/06 11:07:57 isn't this going to ignore the big hole that you c
Tobias Sargeant 2016/05/06 11:33:45 It is, but we don't know exactly how big that hole
Primiano Tucci (use gerrit) 2016/05/06 13:29:23 Hmm but then you will never know if that hole is t
Tobias Sargeant 2016/05/13 16:16:54 This is what happens. The log line is: H lo_addr
+
+ uintptr_t hole_lo = mappings[curr]->start_addr + mappings[curr]->size;
+ uintptr_t hole_hi = mappings[next]->start_addr;
+
+ if (hole_hi > hole_lo) {
+ size_t hole_sz = hole_hi - hole_lo;
+ hole_tot += hole_sz;
+ hole_max = std::max(hole_sz, hole_max);
+ ++hole_cnt;
+ size_t h = 0;
+ while (hole_sz) {
Primiano Tucci (use gerrit) 2016/05/06 11:07:56 Can we just copy chrome base::Log2Floor(uint32_t n
Tobias Sargeant 2016/05/06 13:29:31 Done.
+ ++h;
+ hole_sz >>= 1;
+ }
+ ++hole_hist[h];
+ }
+ curr = next;
+ }
+
+ uintptr_t hi_addr = mappings[curr]->start_addr + mappings[curr]->size;
+
+ LogAppend("H ");
+ LogAppend(lo_addr);
+ LogAppend(" ");
+ LogAppend(hi_addr);
+
+ LogAppend(" "); LogAppendDecimal(hole_cnt);
Primiano Tucci (use gerrit) 2016/05/06 11:07:56 are you appending the count three times here? I s
Tobias Sargeant 2016/05/06 13:29:31 Certainly did!
+ LogAppend(" "); LogAppend(hole_cnt);
Primiano Tucci (use gerrit) 2016/05/06 11:07:56 Also plz newlines after each statement
Tobias Sargeant 2016/05/06 13:29:31 Done.
+ LogAppend(" "); LogAppend(hole_cnt);
+ for (unsigned int i = 0; i < HBITS; ++i) {
Primiano Tucci (use gerrit) 2016/05/06 11:07:57 can we saturate this up and down? First of all, al
Torne 2016/05/06 11:13:36 We should probably just not do this at all on 64bi
Tobias Sargeant 2016/05/06 11:33:45 I'm not sure we want to cap at 1GB. On 32 bit that
Primiano Tucci (use gerrit) 2016/05/06 13:29:23 ok sorry I completely missed the if (!hole_hist[i]
Primiano Tucci (use gerrit) 2016/05/06 13:29:23 SGTM but nothing in this CL is preventing this to
+ if (!hole_hist[i]) continue;
+ LogAppend(" ");
+ LogAppendDecimal(i);
+ LogAppend(":");
+ LogAppendDecimal(hole_hist[i]);
+ }
+ LogCommitLine();
+ return true;
+ }
+
// Write information about the mappings in effect.
bool DumpMappings() {
// First write all the mappings from the dumper
« no previous file with comments | « no previous file | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698