// Copyright 2017 The Crashpad Authors. All rights reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "util/linux/memory_map.h" #include #include #include #include #include #include "base/files/file_path.h" #include "base/strings/stringprintf.h" #include "gtest/gtest.h" #include "test/errors.h" #include "test/file.h" #include "test/multiprocess.h" #include "test/scoped_temp_dir.h" #include "util/file/file_io.h" #include "util/linux/scoped_ptrace_attach.h" #include "util/misc/clock.h" #include "util/misc/from_pointer_cast.h" #include "util/posix/scoped_mmap.h" namespace crashpad { namespace test { namespace { TEST(MemoryMap, SelfBasic) { ScopedMmap mmapping; ASSERT_TRUE(mmapping.ResetMmap(nullptr, getpagesize(), PROT_EXEC | PROT_READ, MAP_SHARED | MAP_ANON, -1, 0)); MemoryMap map; ASSERT_TRUE(map.Initialize(getpid())); auto stack_address = FromPointerCast(&map); const MemoryMap::Mapping* mapping = map.FindMapping(stack_address); ASSERT_TRUE(mapping); EXPECT_GE(stack_address, mapping->range.Base()); EXPECT_LT(stack_address, mapping->range.End()); EXPECT_TRUE(mapping->readable); EXPECT_TRUE(mapping->writable); auto code_address = FromPointerCast(getpid); mapping = map.FindMapping(code_address); ASSERT_TRUE(mapping); EXPECT_GE(code_address, mapping->range.Base()); EXPECT_LT(code_address, mapping->range.End()); EXPECT_TRUE(mapping->readable); EXPECT_FALSE(mapping->writable); EXPECT_TRUE(mapping->executable); auto mapping_address = mmapping.addr_as(); mapping = map.FindMapping(mapping_address); ASSERT_TRUE(mapping); mapping = map.FindMapping(mapping_address + mmapping.len() - 1); ASSERT_TRUE(mapping); EXPECT_EQ(mapping_address, mapping->range.Base()); EXPECT_EQ(mapping_address + mmapping.len(), mapping->range.End()); EXPECT_TRUE(mapping->readable); EXPECT_FALSE(mapping->writable); EXPECT_TRUE(mapping->executable); EXPECT_TRUE(mapping->shareable); } class MapChildTest : public Multiprocess { public: MapChildTest() : Multiprocess(), page_size_(getpagesize()) {} ~MapChildTest() {} private: void MultiprocessParent() override { LinuxVMAddress code_address; CheckedReadFileExactly( ReadPipeHandle(), &code_address, sizeof(code_address)); LinuxVMAddress stack_address; CheckedReadFileExactly( ReadPipeHandle(), &stack_address, sizeof(stack_address)); LinuxVMAddress mapped_address; CheckedReadFileExactly( ReadPipeHandle(), &mapped_address, sizeof(mapped_address)); LinuxVMAddress mapped_file_address; CheckedReadFileExactly( ReadPipeHandle(), &mapped_file_address, sizeof(mapped_file_address)); LinuxVMSize path_length; CheckedReadFileExactly(ReadPipeHandle(), &path_length, sizeof(path_length)); std::string mapped_file_name(path_length, std::string::value_type()); CheckedReadFileExactly(ReadPipeHandle(), &mapped_file_name[0], path_length); ScopedPtraceAttach attachment; attachment.ResetAttach(ChildPID()); MemoryMap map; ASSERT_TRUE(map.Initialize(ChildPID())); const MemoryMap::Mapping* mapping = map.FindMapping(code_address); ASSERT_TRUE(mapping); EXPECT_GE(code_address, mapping->range.Base()); EXPECT_LT(code_address, mapping->range.End()); EXPECT_TRUE(mapping->readable); EXPECT_TRUE(mapping->executable); EXPECT_FALSE(mapping->writable); mapping = map.FindMapping(stack_address); ASSERT_TRUE(mapping); EXPECT_GE(stack_address, mapping->range.Base()); EXPECT_LT(stack_address, mapping->range.End()); EXPECT_TRUE(mapping->readable); EXPECT_TRUE(mapping->writable); mapping = map.FindMapping(mapped_address); ASSERT_TRUE(mapping); EXPECT_EQ(mapped_address, mapping->range.Base()); EXPECT_EQ(mapped_address + page_size_, mapping->range.End()); EXPECT_FALSE(mapping->readable); EXPECT_FALSE(mapping->writable); EXPECT_FALSE(mapping->executable); EXPECT_TRUE(mapping->shareable); mapping = map.FindMapping(mapped_file_address); ASSERT_TRUE(mapping); EXPECT_EQ(mapped_file_address, mapping->range.Base()); EXPECT_EQ(mapping->offset, static_cast(page_size_)); EXPECT_TRUE(mapping->readable); EXPECT_TRUE(mapping->writable); EXPECT_FALSE(mapping->executable); EXPECT_FALSE(mapping->shareable); EXPECT_EQ(mapping->name, mapped_file_name); struct stat file_stat; ASSERT_EQ(stat(mapped_file_name.c_str(), &file_stat), 0) << ErrnoMessage("stat"); EXPECT_EQ(mapping->device, file_stat.st_dev); EXPECT_EQ(mapping->inode, file_stat.st_ino); EXPECT_EQ(map.FindMappingWithName(mapping->name), mapping); } void MultiprocessChild() override { auto code_address = FromPointerCast(getpid); CheckedWriteFile(WritePipeHandle(), &code_address, sizeof(code_address)); auto stack_address = FromPointerCast(&code_address); CheckedWriteFile(WritePipeHandle(), &stack_address, sizeof(stack_address)); ScopedMmap mapping; ASSERT_TRUE(mapping.ResetMmap( nullptr, page_size_, PROT_NONE, MAP_SHARED | MAP_ANON, -1, 0)); auto mapped_address = mapping.addr_as(); CheckedWriteFile( WritePipeHandle(), &mapped_address, sizeof(mapped_address)); ScopedTempDir temp_dir; base::FilePath path = temp_dir.path().Append(FILE_PATH_LITERAL("test_file")); ASSERT_FALSE(FileExists(path)); std::string path_string = path.value(); ScopedFileHandle handle(LoggingOpenFileForReadAndWrite( path, FileWriteMode::kReuseOrCreate, FilePermissions::kOwnerOnly)); ASSERT_TRUE(handle.is_valid()); std::string file_contents(page_size_ * 2, std::string::value_type()); CheckedWriteFile(handle.get(), file_contents.c_str(), file_contents.size()); ScopedMmap file_mapping; ASSERT_TRUE(file_mapping.ResetMmap(nullptr, page_size_, PROT_READ | PROT_WRITE, MAP_PRIVATE, handle.get(), page_size_)); auto mapped_file_address = file_mapping.addr_as(); CheckedWriteFile( WritePipeHandle(), &mapped_file_address, sizeof(mapped_file_address)); LinuxVMSize path_length = path_string.size(); CheckedWriteFile(WritePipeHandle(), &path_length, sizeof(path_length)); CheckedWriteFile(WritePipeHandle(), path_string.c_str(), path_length); CheckedReadFileAtEOF(ReadPipeHandle()); } const size_t page_size_; DISALLOW_COPY_AND_ASSIGN(MapChildTest); }; TEST(MemoryMap, MapChild) { MapChildTest test; test.Run(); } // Some systems optimize mappings by allocating new mappings inside existing // mappings with matching permissions. Work around this by allocating one large // mapping and then switching up the permissions of individual pages to force // populating more entries in the maps file. void InitializeMappings(ScopedMmap* mappings, size_t num_mappings, size_t mapping_size) { ASSERT_TRUE(mappings->ResetMmap(nullptr, mapping_size * num_mappings, PROT_READ, MAP_PRIVATE | MAP_ANON, -1, 0)); auto region = mappings->addr_as(); for (size_t index = 0; index < num_mappings; index += 2) { ASSERT_EQ(mprotect(reinterpret_cast(region + index * mapping_size), mapping_size, PROT_READ | PROT_WRITE), 0) << ErrnoMessage("mprotect"); } } void ExpectMappings(const MemoryMap& map, LinuxVMAddress region_addr, size_t num_mappings, size_t mapping_size) { for (size_t index = 0; index < num_mappings; ++index) { SCOPED_TRACE(base::StringPrintf("index %zu", index)); auto mapping_address = region_addr + index * mapping_size; const MemoryMap::Mapping* mapping = map.FindMapping(mapping_address); ASSERT_TRUE(mapping); EXPECT_EQ(mapping_address, mapping->range.Base()); EXPECT_EQ(mapping_address + mapping_size, mapping->range.End()); EXPECT_TRUE(mapping->readable); EXPECT_FALSE(mapping->shareable); EXPECT_FALSE(mapping->executable); if (index % 2 == 0) { EXPECT_TRUE(mapping->writable); } else { EXPECT_FALSE(mapping->writable); } } } TEST(MemoryMap, SelfLargeMapFile) { constexpr size_t kNumMappings = 1024; const size_t page_size = getpagesize(); ScopedMmap mappings; ASSERT_NO_FATAL_FAILURE( InitializeMappings(&mappings, kNumMappings, page_size)); MemoryMap map; ASSERT_TRUE(map.Initialize(getpid())); ExpectMappings( map, mappings.addr_as(), kNumMappings, page_size); } class MapRunningChildTest : public Multiprocess { public: MapRunningChildTest() : Multiprocess(), page_size_(getpagesize()) {} ~MapRunningChildTest() {} private: void MultiprocessParent() override { // Let the child get started LinuxVMAddress region_addr; CheckedReadFileExactly(ReadPipeHandle(), ®ion_addr, sizeof(region_addr)); for (int iter = 0; iter < 8; ++iter) { SCOPED_TRACE(base::StringPrintf("iter %d", iter)); // Let the child get back to its work SleepNanoseconds(1000); ScopedPtraceAttach attachment; attachment.ResetAttach(ChildPID()); MemoryMap map; ASSERT_TRUE(map.Initialize(ChildPID())); // We should at least find the original mappings. The extra mappings may // or not be found depending on scheduling. ExpectMappings(map, region_addr, kNumMappings, page_size_); } } void MultiprocessChild() override { ASSERT_EQ(fcntl(ReadPipeHandle(), F_SETFL, O_NONBLOCK), 0) << ErrnoMessage("fcntl"); ScopedMmap mappings; ASSERT_NO_FATAL_FAILURE( InitializeMappings(&mappings, kNumMappings, page_size_)); // Let the parent start mapping us auto region_addr = mappings.addr_as(); CheckedWriteFile(WritePipeHandle(), ®ion_addr, sizeof(region_addr)); // But don't stop there! constexpr size_t kNumExtraMappings = 256; ScopedMmap extra_mappings; while (true) { ASSERT_NO_FATAL_FAILURE( InitializeMappings(&extra_mappings, kNumExtraMappings, page_size_)); // Quit when the parent is done char c; FileOperationResult res = ReadFile(ReadPipeHandle(), &c, sizeof(c)); if (res == 0) { break; } ASSERT_EQ(errno, EAGAIN); } } static constexpr size_t kNumMappings = 1024; const size_t page_size_; DISALLOW_COPY_AND_ASSIGN(MapRunningChildTest); }; TEST(MemoryMap, MapRunningChild) { MapRunningChildTest test; test.Run(); } } // namespace } // namespace test } // namespace crashpad