普通文本  |  579行  |  21.81 KB

#include "huge_page_deducer.h"  

#include "base/logging.h"
#include "compat/string.h"
#include "compat/test.h"


namespace quipper {
namespace {
using PerfEvent = PerfDataProto::PerfEvent;
using MMapEvent = PerfDataProto::MMapEvent;
using ::testing::EqualsProto;
using ::testing::Pointwise;
using ::testing::proto::Partially;

// AddMmap is a helper function to create simple MMapEvents, with which
// testcases can encode "maps" entries similar to /proc/self/maps in a tabular
// one-line-per-entry.
void AddMmap(uint32_t pid, uint64_t mmap_start, uint64_t length, uint64_t pgoff,
             const string& file, RepeatedPtrField<PerfEvent>* events) {
  MMapEvent* ev = events->Add()->mutable_mmap_event();
  ev->set_pid(pid);
  ev->set_start(mmap_start);
  ev->set_len(length);
  ev->set_pgoff(pgoff);
  ev->set_filename(file);
}

TEST(HugePageDeducer, HugePagesMappings) {
  RepeatedPtrField<PerfEvent> events;
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x40000000);
    ev->set_len(0x18000);
    ev->set_pgoff(0);
    ev->set_filename("/usr/lib/libfoo.so");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x40018000);
    ev->set_len(0x1e8000);
    ev->set_pgoff(0);
    ev->set_filename("/opt/google/chrome/chrome");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x40200000);
    ev->set_len(0x1c00000);
    ev->set_pgoff(0);
    ev->set_filename("//anon");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x41e00000);
    ev->set_len(0x4000000);
    ev->set_pgoff(0x1de8000);
    ev->set_filename("/opt/google/chrome/chrome");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(2345);
    ev->set_start(0x45e00000);
    ev->set_len(0x1e00000);
    ev->set_pgoff(0);
    ev->set_filename("//anon");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(2345);
    ev->set_start(0x47c00000);
    ev->set_len(0x4000000);
    ev->set_pgoff(0x1e00000);
    ev->set_filename("/opt/google/chrome/chrome");
  }

  DeduceHugePages(&events);
  CombineMappings(&events);

  ASSERT_GE(events.size(), 3);
  EXPECT_EQ(events.size(), 3);

  EXPECT_THAT(events,
              Pointwise(Partially(EqualsProto()),
                        {
                            "mmap_event: { start: 0x40000000 len:0x18000 "
                            "pgoff: 0 filename: '/usr/lib/libfoo.so'}",
                            "mmap_event: { start: 0x40018000 len:0x5de8000 "
                            "pgoff: 0 filename: '/opt/google/chrome/chrome'}",
                            "mmap_event: { start: 0x45e00000 len:0x5e00000 "
                            "pgoff: 0 filename: '/opt/google/chrome/chrome'}",
                        }));

  EXPECT_EQ("/usr/lib/libfoo.so", events[0].mmap_event().filename());
  EXPECT_EQ(0x40000000, events[0].mmap_event().start());
  EXPECT_EQ(0x18000, events[0].mmap_event().len());
  EXPECT_EQ(0x0, events[0].mmap_event().pgoff());

  // The split Chrome mappings should have been combined.
  EXPECT_EQ("/opt/google/chrome/chrome", events[2].mmap_event().filename());
  EXPECT_EQ(0x40018000, events[1].mmap_event().start());
  EXPECT_EQ(0x5de8000, events[1].mmap_event().len());
  EXPECT_EQ(0x0, events[1].mmap_event().pgoff());

  EXPECT_EQ("/opt/google/chrome/chrome", events[2].mmap_event().filename());
  EXPECT_EQ(0x45e00000, events[2].mmap_event().start());
  EXPECT_EQ(0x5e00000, events[2].mmap_event().len());
  EXPECT_EQ(0x0, events[2].mmap_event().pgoff());
}

enum HugepageTextStyle {
  kAnonHugepageText,
  kNoHugepageText,
};

class HugepageTextStyleDependent
    : public ::testing::TestWithParam<HugepageTextStyle> {
 protected:
  void AddHugepageTextMmap(uint32_t pid, uint64_t mmap_start, uint64_t length,
                           uint64_t pgoff, string file,
                           RepeatedPtrField<PerfEvent>* events) {
    // Various hugepage implementations and perf versions result in various
    // quirks in how hugepages are reported.

    switch (GetParam()) {
      case kNoHugepageText:
        // Do nothing; the maps are complete and file-backed
        break;
      case kAnonHugepageText:
        // exec is remapped into anonymous memory, which perf reports as
        // '//anon'. Anonymous sections have no pgoff.
        file = "//anon";
        pgoff = 0;
        break;
      default:
        CHECK(false) << "Unimplemented";
    }
    AddMmap(pid, mmap_start, length, pgoff, file, events);
  }
};

TEST_P(HugepageTextStyleDependent, OnlyOneMappingThatIsHuge) {
  RepeatedPtrField<PerfEvent> events;
  AddHugepageTextMmap(1, 0x100200000, 0x200000, 0, "file", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  // Don't check filename='file'; if it's backed by anonymous memory, it isn't
  // possible for quipper to deduce the filename without other mmaps immediately
  // adjacent.
  EXPECT_THAT(
      events,
      Pointwise(Partially(EqualsProto()),
                {"mmap_event: { start: 0x100200000 len: 0x200000 pgoff: 0}"}));
}

TEST_P(HugepageTextStyleDependent, OnlyOneMappingUnaligned) {
  RepeatedPtrField<PerfEvent> events;
  AddMmap(2, 0x200201000, 0x200000, 0, "file", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {"mmap_event: { start: 0x200201000 "
                                 "len:0x200000 pgoff: 0 filename: 'file'}"}));
}

TEST_P(HugepageTextStyleDependent, FirstPageIsHugeWithSmallTail) {
  RepeatedPtrField<PerfEvent> events;
  AddHugepageTextMmap(3, 0x300400000, 0x400000, 0, "file", &events);
  AddMmap(3, 0x300800000, 0x001000, 0x400000, "file", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {"mmap_event: { start: 0x300400000 "
                                 "len:0x401000 pgoff: 0 filename: 'file'}"}));
}

TEST_P(HugepageTextStyleDependent, DISABLED_FirstPageIsSmallWithHugeTail) {
  // This test is disabled because DeduceHugePage requires a non-zero pgoff
  // *after* a hugepage_text section in order to correctly deduce it, so it
  // is unable to deduce these cases.
  RepeatedPtrField<PerfEvent> events;
  AddMmap(4, 0x4003ff000, 0x001000, 0, "file", &events);
  AddHugepageTextMmap(4, 0x400400000, 0x200000, 0x001000, "file", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {"mmap_event: { start: 0x4003ff000 "
                                 "len:0x201000 pgoff: 0 filename: 'file'}"}));
}

TEST_P(HugepageTextStyleDependent, HugePageBetweenTwoSmallSections) {
  RepeatedPtrField<PerfEvent> events;
  AddMmap(5, 0x5003ff000, 0x001000, 0, "file", &events);
  AddHugepageTextMmap(5, 0x500400000, 0x200000, 0x001000, "file", &events);
  AddMmap(5, 0x500600000, 0x001000, 0x201000, "file", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {"mmap_event: { start: 0x5003ff000 "
                                 "len:0x202000 pgoff: 0 filename: 'file'}"}));
}

TEST_P(HugepageTextStyleDependent, HugePageSplitByEarlyMlockBetweenTwoSmall) {
  RepeatedPtrField<PerfEvent> events;
  AddMmap(6, 0x6003ff000, 0x001000, 0, "file", &events);
  AddHugepageTextMmap(6, 0x600400000, 0x3f8000, 0x001000, "file", &events);
  AddHugepageTextMmap(6, 0x6007f8000, 0x008000, 0x3f9000, "file", &events);
  AddMmap(6, 0x600800000, 0x001000, 0x401000, "file", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {"mmap_event: { start: 0x6003ff000 "
                                 "len:0x402000 pgoff: 0 filename: 'file'}"}));
}

TEST_P(HugepageTextStyleDependent, HugePageSplitByLateMlockBetweenTwoSmall) {
  RepeatedPtrField<PerfEvent> events;
  AddMmap(7, 0x7003ff000, 0x001000, 0, "file", &events);
  AddHugepageTextMmap(7, 0x700400000, 0x008000, 0x001000, "file", &events);
  AddHugepageTextMmap(7, 0x700408000, 0x3f8000, 0x009000, "file", &events);
  AddMmap(7, 0x700800000, 0x001000, 0x401000, "file", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {"mmap_event: { start: 0x7003ff000 "
                                 "len:0x402000 pgoff: 0 filename: 'file'}"}));
}

TEST_P(HugepageTextStyleDependent, HugePageSplitEvenlyByMlockBetweenTwoSmall) {
  RepeatedPtrField<PerfEvent> events;
  AddMmap(8, 0x8003ff000, 0x001000, 0, "file", &events);
  AddHugepageTextMmap(8, 0x800400000, 0x0f8000, 0x001000, "file", &events);
  AddHugepageTextMmap(8, 0x8004f8000, 0x008000, 0x0f9000, "file", &events);
  AddHugepageTextMmap(8, 0x800500000, 0x100000, 0x101000, "file", &events);
  AddMmap(8, 0x800600000, 0x001000, 0x201000, "file", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {"mmap_event: { start: 0x8003ff000 "
                                 "len:0x202000 pgoff: 0 filename: 'file'}"}));
}

TEST_P(HugepageTextStyleDependent, MultipleContiguousHugepages) {
  RepeatedPtrField<PerfEvent> events;
  AddMmap(9, 0x9003ff000, 0x001000, 0, "file", &events);
  AddHugepageTextMmap(9, 0x900400000, 0x200000, 0x001000, "file", &events);
  AddHugepageTextMmap(9, 0x900600000, 0x200000, 0x201000, "file", &events);
  AddMmap(9, 0x900800000, 0x001000, 0x401000, "file", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {"mmap_event: { start: 0x9003ff000 "
                                 "len:0x402000 pgoff: 0 filename: 'file'}"}));
}

TEST_P(HugepageTextStyleDependent, MultipleContiguousMlockSplitHugepages) {
  // Think:
  // - hugepage_text 4MiB range
  // - mlock alternating 512-KiB chunks
  RepeatedPtrField<PerfEvent> events;
  AddMmap(10, 0xa003ff000, 0x001000, 0, "file", &events);
  AddHugepageTextMmap(10, 0xa00400000, 0x080000, 0x001000, "file", &events);
  AddHugepageTextMmap(10, 0xa00480000, 0x080000, 0x081000, "file", &events);
  AddHugepageTextMmap(10, 0xa00500000, 0x080000, 0x101000, "file", &events);
  AddHugepageTextMmap(10, 0xa00580000, 0x080000, 0x181000, "file", &events);
  AddHugepageTextMmap(10, 0xa00600000, 0x080000, 0x201000, "file", &events);
  AddHugepageTextMmap(10, 0xa00680000, 0x080000, 0x281000, "file", &events);
  AddHugepageTextMmap(10, 0xa00700000, 0x080000, 0x301000, "file", &events);
  AddHugepageTextMmap(10, 0xa00780000, 0x080000, 0x381000, "file", &events);
  AddMmap(10, 0xa00800000, 0x001000, 0x401000, "file", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {"mmap_event: { start: 0xa003ff000 "
                                 "len:0x402000 pgoff: 0 filename: 'file'}"}));
}

TEST_P(HugepageTextStyleDependent, MultipleWithUnalignedInitialHugePage) {
  // Base on real program
  RepeatedPtrField<PerfEvent> events;

  AddHugepageTextMmap(11, 0x85d32e000, 0x6d2000, 0x0, "file", &events);
  AddHugepageTextMmap(11, 0x85da00000, 0x6a00000, 0x6d2000, "file", &events);
  AddMmap(11, 0x864400000, 0x200000, 0x70d2000, "file", &events);
  AddHugepageTextMmap(11, 0x864600000, 0x200000, 0x72d2000, "file", &events);
  AddMmap(11, 0x864800000, 0x600000, 0x74d2000, "file", &events);
  AddHugepageTextMmap(11, 0x864e00000, 0x200000, 0x7ad2000, "file", &events);
  AddMmap(11, 0x865000000, 0x4a000, 0x7cd2000, "file", &events);
  AddMmap(11, 0x86504a000, 0x1000, 0x7d1c000, "file", &events);
  AddMmap(11, 0xa3d368000, 0x3a96000, 0x0, "file2", &events);
  AddMmap(11, 0xa467cc000, 0x2000, 0x0, "file3", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {
                                    "mmap_event: { start: 0x85d32e000 "
                                    "len:0x7d1d000 pgoff: 0 filename: 'file'}",
                                    "mmap_event: { start: 0xa3d368000 "
                                    "len:0x3a96000 pgoff: 0 filename: 'file2'}",
                                    "mmap_event: { start: 0xa467cc000 "
                                    "len:0x2000,  pgoff: 0 filename: 'file3'}",
                                }));
}

TEST_P(HugepageTextStyleDependent, MultipleWithUnalignedInitialHugePage2) {
  // Base on real program
  RepeatedPtrField<PerfEvent> events;
  AddHugepageTextMmap(12, 0xbcff6000, 0x200000, 0x00000000, "file", &events);
  AddMmap(12, 0xbd1f6000, 0x300a000, 0x200000, "file", &events);
  AddHugepageTextMmap(12, 0xc0200000, 0x2b374000, 0x320a000, "file", &events);
  AddHugepageTextMmap(12, 0xeb574000, 0x514000, 0x2e57e000, "file", &events);
  AddHugepageTextMmap(12, 0xeba88000, 0x1d78000, 0x2ea92000, "file", &events);
  AddMmap(12, 0xed800000, 0x1200000, 0x3080a000, "file", &events);
  AddHugepageTextMmap(12, 0xeea00000, 0x200000, 0x31a0a000, "file", &events);
  AddMmap(12, 0xeec00000, 0x2800000, 0x31c0a000, "file", &events);
  AddHugepageTextMmap(12, 0xf1400000, 0x200000, 0x3440a000, "file", &events);
  AddMmap(12, 0xf1600000, 0x89f000, 0x3460a000, "file", &events);
  AddMmap(12, 0xf1e9f000, 0x1000, 0x34ea9000, "file", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {"mmap_event: { start: 0xbcff6000 "
                                 "len:0x34eaa000 pgoff: 0 filename: 'file'}"}));
}

TEST_P(HugepageTextStyleDependent, NoMmaps) {
  RepeatedPtrField<PerfEvent> events;
  events.Add();

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(EqualsProto(), std::vector<PerfEvent>(1)));
}
TEST_P(HugepageTextStyleDependent, MultipleNonMmaps) {
  RepeatedPtrField<PerfEvent> events;
  events.Add();
  events.Add();

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(EqualsProto(), std::vector<PerfEvent>(2)));
}
TEST_P(HugepageTextStyleDependent, NonMmapFirstMmap) {
  RepeatedPtrField<PerfEvent> events;
  events.Add();
  AddHugepageTextMmap(12, 0, 0x200000, 0, "file", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {"", "mmap_event: { pgoff: 0 }"}));
}
TEST_P(HugepageTextStyleDependent, NonMmapAfterLastMmap) {
  RepeatedPtrField<PerfEvent> events;
  AddHugepageTextMmap(12, 0, 0x200000, 0, "file", &events);
  events.Add();

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events, Pointwise(Partially(EqualsProto()),
                                {"mmap_event: { pgoff: 0 }", ""}));
}

INSTANTIATE_TEST_CASE_P(NoHugepageText, HugepageTextStyleDependent,
                        ::testing::Values(kNoHugepageText));
INSTANTIATE_TEST_CASE_P(AnonHugepageText, HugepageTextStyleDependent,
                        ::testing::Values(kAnonHugepageText));

TEST(HugePageDeducer, DoesNotChangeVirtuallyContiguousPgoffNonContiguous) {
  // We've seen programs with strange memory layouts having virtually contiguous
  // memory backed by non-contiguous bits of a file.
  RepeatedPtrField<PerfEvent> events;
  AddMmap(758463, 0x2f278000, 0x20000, 0, "lib0.so", &events);
  AddMmap(758463, 0x2f29d000, 0x2000, 0, "shm", &events);
  AddMmap(758463, 0x2f2a2000, 0xa000, 0, "lib1.so", &events);
  AddMmap(758463, 0x3d400000, 0x9ee000, 0, "lib2.so", &events);
  AddMmap(758463, 0x3e000000, 0x16000, 0, "lib3.so", &events);
  AddMmap(758463, 0x3e400000, 0x270000, 0x1a00000, "shm", &events);
  AddMmap(758463, 0x3e670000, 0x10000, 0x1aaac000, "shm", &events);
  AddMmap(758463, 0x3e680000, 0x10000, 0x1b410000, "shm", &events);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(events,
              Pointwise(Partially(EqualsProto()),
                        {
                            "mmap_event: { pgoff: 0 filename: 'lib0.so' }",
                            "mmap_event: { pgoff: 0 filename: 'shm' }",
                            "mmap_event: { pgoff: 0 filename: 'lib1.so' }",
                            "mmap_event: { pgoff: 0 filename: 'lib2.so' }",
                            "mmap_event: { pgoff: 0 filename: 'lib3.so' }",
                            "mmap_event: { pgoff: 0x1a00000 filename: 'shm' }",
                            "mmap_event: { pgoff: 0x1aaac000 filename: 'shm' }",
                            "mmap_event: { pgoff: 0x1b410000 filename: 'shm' }",
                        }));
}

TEST(HugePageDeducer, IgnoresDynamicMmaps) {
  // Now, let's watch a binary hugepage_text itself.
  RepeatedPtrField<PerfEvent> events;
  AddMmap(6531, 0x560d76b25000, 0x24ce000, 0, "main", &events);
  events.rbegin()->set_timestamp(700413232676401);
  AddMmap(6531, 0x7f686a1ec000, 0x24000, 0, "ld.so", &events);
  events.rbegin()->set_timestamp(700413232691935);
  AddMmap(6531, 0x7ffea5dc8000, 0x2000, 0, "[vdso]", &events);
  events.rbegin()->set_timestamp(700413232701418);
  AddMmap(6531, 0x7f686a1e3000, 0x5000, 0, "lib1.so", &events);
  events.rbegin()->set_timestamp(700413232824216);
  AddMmap(6531, 0x7f686a1a8000, 0x3a000, 0, "lib2.so", &events);
  events.rbegin()->set_timestamp(700413232854520);
  AddMmap(6531, 0x7f6869ea7000, 0x5000, 0, "lib3.so", &events);
  events.rbegin()->set_timestamp(700413248827794);
  AddMmap(6531, 0x7f6867e00000, 0x200000, 0, "/anon_hugepage (deleted)",
          &events);
  events.rbegin()->set_timestamp(700413295816043);
  AddMmap(6531, 0x7f6867c00000, 0x200000, 0, "/anon_hugepage (deleted)",
          &events);
  events.rbegin()->set_timestamp(700413305947499);
  AddMmap(6531, 0x7f68663f8000, 0x1e00000, 0x7f68663f8000, "//anon", &events);
  events.rbegin()->set_timestamp(700413306012797);
  AddMmap(6531, 0x7f6866525000, 0x1a00000, 0x7f6866525000, "//anon", &events);
  events.rbegin()->set_timestamp(700413312132909);

  DeduceHugePages(&events);
  CombineMappings(&events);

  EXPECT_THAT(
      events,
      Pointwise(
          Partially(EqualsProto()),
          {
              "mmap_event: { pgoff: 0 filename: 'main' }",
              "mmap_event: { pgoff: 0 filename: 'ld.so' }",
              "mmap_event: { pgoff: 0 filename: '[vdso]' }",
              "mmap_event: { pgoff: 0 filename: 'lib1.so' }",
              "mmap_event: { pgoff: 0 filename: 'lib2.so' }",
              "mmap_event: { pgoff: 0 filename: 'lib3.so' }",
              "mmap_event: { pgoff: 0 filename: '/anon_hugepage (deleted)' }",
              "mmap_event: { pgoff: 0 filename: '/anon_hugepage (deleted)' }",
              "mmap_event: { pgoff: 0x7f68663f8000 filename: '//anon' }",
              "mmap_event: { pgoff: 0x7f6866525000 filename: '//anon' }",
          }));
}

TEST(HugePageDeducer, Regression62446346) {
  RepeatedPtrField<PerfEvent> events;

  // Perf infers the filename is "file", but at offset 0 for
  // hugepage-backed, anonymous mappings.
  //
  // vaddr start   - vaddr end     vaddr-size    elf-offset
  // [0x55a685bfb000-55a685dfb000) (0x200000)   @ 0]:          file
  // [0x55a685dfb000-55a687c00000) (0x1e05000)  @ 0x200000]:   file
  // [0x55a687c00000-55a6a5200000) (0x1d600000) @ 0]:          file
  // [0x55a6a5200000-55a6a6400000) (0x1200000)  @ 0x1f605000]: file
  // [0x55a6a6400000-55a6a6600000) (0x200000)   @ 0]:          file
  // [0x55a6a6600000-55a6a8800000) (0x2200000)  @ 0x20a05000]: file
  // [0x55a6a8800000-55a6a8a00000) (0x200000)   @ 0]:          file
  // [0x55a6a8a00000-55a6a90ca000) (0x6ca000)   @ 0x22e05000]: file
  // [0x55a6a90ca000-55a6a90cb000) (0x1000)     @ 0x234cf000]: file
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x55a685bfb000);
    ev->set_len(0x200000);
    ev->set_pgoff(0);
    ev->set_filename("file");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x55a685dfb000);
    ev->set_len(0x1e05000);
    ev->set_pgoff(0x200000);
    ev->set_filename("file");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x55a687c00000);
    ev->set_len(0x1d600000);
    ev->set_pgoff(0);
    ev->set_filename("file");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x55a6a5200000);
    ev->set_len(0x1200000);
    ev->set_pgoff(0x1f605000);
    ev->set_filename("file");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x55a6a6400000);
    ev->set_len(0x200000);
    ev->set_pgoff(0);
    ev->set_filename("file");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x55a6a6600000);
    ev->set_len(0x2200000);
    ev->set_pgoff(0x20a05000);
    ev->set_filename("file");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x55a6a8800000);
    ev->set_len(0x200000);
    ev->set_pgoff(0);
    ev->set_filename("file");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x55a6a8a00000);
    ev->set_len(0x6ca000);
    ev->set_pgoff(0x22e05000);
    ev->set_filename("file");
  }
  {
    MMapEvent* ev = events.Add()->mutable_mmap_event();
    ev->set_pid(1234);
    ev->set_start(0x55a6a90ca000);
    ev->set_len(0x1000);
    ev->set_pgoff(0x234cf000);
    ev->set_filename("file");
  }

  DeduceHugePages(&events);
  CombineMappings(&events);

  ASSERT_EQ(1, events.size());

  EXPECT_EQ("file", events[0].mmap_event().filename());
  EXPECT_EQ(0x55a685bfb000, events[0].mmap_event().start());
  EXPECT_EQ(0x55a6a90cb000 - 0x55a685bfb000, events[0].mmap_event().len());
  EXPECT_EQ(0, events[0].mmap_event().pgoff());
}

}  // namespace
}  // namespace quipper