Index: src/client/linux/minidump_writer/linux_dumper.cc |
diff --git a/src/client/linux/minidump_writer/linux_dumper.cc b/src/client/linux/minidump_writer/linux_dumper.cc |
index 64b967b799976329a66f4bae38c3648c5c8ad9fa..a95ec7ec812c736be5cdb0a59939e4a499ca25ac 100644 |
--- a/src/client/linux/minidump_writer/linux_dumper.cc |
+++ b/src/client/linux/minidump_writer/linux_dumper.cc |
@@ -84,6 +84,173 @@ inline static bool IsMappedFileOpenUnsafe( |
namespace google_breakpad { |
+#if defined(__CHROMEOS__) |
+ |
+namespace { |
+ |
+// Recover memory mappings before writing dump on ChromeOS |
+// |
+// On Linux, breakpad relies on /proc/[pid]/maps to associate symbols from |
+// addresses. ChromeOS' hugepage implementation replaces some segments with |
+// anonymous private pages, which is a restriction of current implementation |
+// in Linux kernel at the time of writing. Thus, breakpad can no longer |
+// symbolize addresses from those text segments replaced with hugepages. |
+// |
+// This postprocess tries to recover the mappings. Because hugepages are always |
+// inserted in between some .text sections, it tries to infer the names and |
+// offsets of the segments, by looking at segments immediately precede and |
+// succeed them. |
+// |
+// For example, a text segment before hugepage optimization |
+// 02001000-03002000 r-xp /opt/google/chrome/chrome |
+// |
+// can be broken into |
+// 02001000-02200000 r-xp /opt/google/chrome/chrome |
+// 02200000-03000000 r-xp |
+// 03000000-03002000 r-xp /opt/google/chrome/chrome |
+// |
+// For more details, see: |
+// crbug.com/628040 ChromeOS' use of hugepages confuses crash symbolization |
Wez
2016/07/18 22:28:15
Do/will we have a bug filed to remove this hack if
|
+ |
+// Copied from CrOS' hugepage implementation, which is unlikely to change. |
+// The hugepage size is 2M. |
+const unsigned int kHpageShift = 21; |
+const size_t kHpageSize = (1 << kHpageShift); |
+const size_t kHpageMask = (~(kHpageSize - 1)); |
+ |
+// Find and merge anonymous r-xp segments with surrounding named segments. |
+// There are two cases: |
+ |
+// Case 1: curr, next |
Wez
2016/07/18 22:28:15
What do these "cases" mean? Do we sometimes see Ch
|
+// curr is anonymous |
+// curr is r-xp |
+// curr.size >= 2M |
+// curr.size is a multiple of 2M. |
Wez
2016/07/18 22:28:15
Do we need to be this specific about the sizes? Ho
|
+// next is backed by some file. |
+// curr and next are contiguous. |
+// offset(next) == sizeof(curr) |
+void TryRecoverMappings(MappingInfo *curr, MappingInfo *next) { |
Wez
2016/07/18 22:28:15
Looks like this function actually checks whether |
|
+ // Merged segments are marked with size = 0. |
Wez
2016/07/18 22:28:15
It's not clear from the context that it's possible
|
+ if (curr->size == 0 || next->size == 0) |
+ return; |
+ |
+ if (curr->size >= kHpageSize && |
+ curr->exec && |
+ (curr->size & kHpageMask) == curr->size && |
+ (curr->start_addr & kHpageMask) == curr->start_addr && |
+ curr->name[0] == '\0' && |
+ next->name[0] != '\0' && |
+ curr->start_addr + curr->size == next->start_addr && |
+ curr->size == next->offset) { |
Wez
2016/07/18 22:28:15
Strongly suggest turning this into a set of early-
|
+ |
+ // matched |
Wez
2016/07/18 22:28:15
nit: This comment doesn't add any information :P
|
+ my_strlcpy(curr->name, next->name, NAME_MAX); |
+ if (next->exec) { |
+ // (curr, next) |
Wez
2016/07/18 22:28:14
nit: Nor does this one ;)
|
+ curr->size += next->size; |
+ next->size = 0; |
+ } |
+ } |
+} |
+ |
+// Case 2: prev, curr, next |
+// curr is anonymous |
+// curr is r-xp |
+// curr.size >= 2M |
+// curr.size is a multiple of 2M. |
+// next and prev are backed by the same file. |
+// prev, curr and next are contiguous. |
+// offset(next) == offset(prev) + sizeof(prev) + sizeof(curr) |
+void TryRecoverMappings(MappingInfo *prev, MappingInfo *curr, |
+ MappingInfo *next) { |
+ // Merged segments are marked with size = 0. |
+ if (prev->size == 0 || curr->size == 0 || next->size == 0) |
+ return; |
+ |
+ if (curr->size >= kHpageSize && |
+ curr->exec && |
+ (curr->size & kHpageMask) == curr->size && |
+ (curr->start_addr & kHpageMask) == curr->start_addr && |
+ curr->name[0] == '\0' && |
+ next->name[0] != '\0' && |
+ curr->start_addr + curr->size == next->start_addr && |
+ prev->start_addr + prev->size == curr->start_addr && |
+ my_strncmp(prev->name, next->name, NAME_MAX) == 0 && |
+ next->offset == prev->offset + prev->size + curr->size) { |
+ |
+ // matched |
+ my_strlcpy(curr->name, prev->name, NAME_MAX); |
+ if (prev->exec) { |
+ curr->offset = prev->offset; |
+ curr->start_addr = prev->start_addr; |
+ if (next->exec) { |
+ // (prev, curr, next) |
+ curr->size += prev->size + next->size; |
+ prev->size = 0; |
+ next->size = 0; |
+ } else { |
+ // (prev, curr), next |
+ curr->size += prev->size; |
+ prev->size = 0; |
+ } |
+ } else { |
+ curr->offset = prev->offset + prev->size; |
+ if (next->exec) { |
+ // prev, (curr, next) |
+ curr->size += next->size; |
+ next->size = 0; |
+ } else { |
+ // prev, curr, next |
+ } |
+ } |
+ } |
+} |
+ |
+// mappings_ is sorted excepted for the first entry. |
+// This function tries to merge segemnts into the first entry, |
Wez
2016/07/18 22:28:15
typo: segments
|
+// then check for other sorted entries. |
+// See LinuxDumper::EnumerateMappings(). |
Wez
2016/07/18 22:28:15
nit: EnumerateMappings() doesn't have any document
|
+void CrOSPostProcessMappings(wasteful_vector<MappingInfo*>& mappings) { |
+ // Find the candidate "next" to first segment, which is the only one that |
+ // could be out-of-order. |
+ size_t l = 1; |
+ size_t r = mappings.size(); |
Wez
2016/07/18 22:28:15
nit: Please use descriptive variable names, for re
|
+ size_t next = mappings.size(); |
+ while (l < r) { |
+ int m = (l + r) / 2; |
+ if (mappings[m]->start_addr > mappings[0]->start_addr) |
+ r = next = m; |
+ else |
+ l = m + 1; |
Wez
2016/07/18 22:28:15
nit: I'd recommend using {} around the two cases h
|
+ } |
+ |
+ // Try to merge segments into the first. |
Wez
2016/07/18 22:28:15
It's not clear to me what this is actually trying
|
+ if (next < mappings.size()) { |
+ TryRecoverMappings(mappings[0], mappings[next]); |
+ if (next - 1 > 0) |
+ TryRecoverMappings(mappings[next - 1], mappings[0], mappings[next]); |
+ } |
+ |
+ // Iterate through normal, sorted cases. |
+ // Normal case 1. |
+ for (size_t i = 1; i < mappings.size() - 1; i++) |
+ TryRecoverMappings(mappings[i], mappings[i + 1]); |
+ |
+ // Normal case 2. |
+ for (size_t i = 1; i < mappings.size() - 2; i++) |
+ TryRecoverMappings(mappings[i], mappings[i + 1], mappings[i + 2]); |
Wez
2016/07/18 22:30:48
If we already ran through trying to merge (anon, m
|
+ |
+ // Collect merged (size == 0) segments. |
+ size_t f, e; |
+ for (f = e = 0; e < mappings.size(); e++) |
+ if (mappings[e]->size > 0) |
+ mappings[f++] = mappings[e]; |
+ mappings.resize(f); |
+} |
+ |
+} // namespace |
+#endif // __CHROMEOS__ |
+ |
// All interesting auvx entry types are below AT_SYSINFO_EHDR |
#define AT_MAX AT_SYSINFO_EHDR |
@@ -113,6 +280,11 @@ bool LinuxDumper::LateInit() { |
#if defined(__ANDROID__) |
LatePostprocessMappings(); |
#endif |
+ |
+#if defined(__CHROMEOS__) |
+ CrOSPostProcessMappings(mappings_); |
+#endif |
+ |
return true; |
} |