/* * Copyright (C) 2017 The Android Open Source Project * * 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 "perfetto/base/paged_memory.h" #include <stdint.h> #include "gtest/gtest.h" #include "perfetto/base/build_config.h" #include "src/base/test/vm_test_utils.h" #if !PERFETTO_BUILDFLAG(PERFETTO_OS_MACOSX) && \ !PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) && \ !PERFETTO_BUILDFLAG(PERFETTO_OS_FUCHSIA) #include <sys/resource.h> #endif namespace perfetto { namespace base { namespace { TEST(PagedMemoryTest, Basic) { const size_t kNumPages = 10; const size_t kSize = 4096 * kNumPages; void* ptr_raw = nullptr; { PagedMemory mem = PagedMemory::Allocate(kSize); ASSERT_TRUE(mem.IsValid()); ASSERT_EQ(0u, reinterpret_cast<uintptr_t>(mem.Get()) % 4096); ptr_raw = mem.Get(); for (size_t i = 0; i < kSize / sizeof(uint64_t); i++) ASSERT_EQ(0u, *(reinterpret_cast<uint64_t*>(mem.Get()) + i)); #if !PERFETTO_BUILDFLAG(PERFETTO_OS_FUCHSIA) ASSERT_TRUE(vm_test_utils::IsMapped(ptr_raw, kSize)); #endif #if PERFETTO_BUILDFLAG(PERFETTO_OS_LINUX) || \ PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID) ASSERT_TRUE(mem.AdviseDontNeed(ptr_raw, kSize)); // Make sure the pages were removed from the working set. ASSERT_FALSE(vm_test_utils::IsMapped(ptr_raw, kSize)); #endif } #if !PERFETTO_BUILDFLAG(PERFETTO_OS_FUCHSIA) // Freed memory is necessarily not mapped in to the process. ASSERT_FALSE(vm_test_utils::IsMapped(ptr_raw, kSize)); #endif } TEST(PagedMemoryTest, Uncommitted) { constexpr size_t kNumPages = 4096; constexpr size_t kSize = 4096 * kNumPages; char* ptr_raw = nullptr; { PagedMemory mem = PagedMemory::Allocate(kSize, PagedMemory::kDontCommit); ASSERT_TRUE(mem.IsValid()); ptr_raw = reinterpret_cast<char*>(mem.Get()); #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) // Windows only commits the first 1024 pages. constexpr size_t kMappedSize = 4096 * 1024; for (size_t i = 0; i < kMappedSize / sizeof(uint64_t); i++) ASSERT_EQ(0u, *(reinterpret_cast<uint64_t*>(mem.Get()) + i)); ASSERT_TRUE(vm_test_utils::IsMapped(ptr_raw, kMappedSize)); // Next page shouldn't be mapped. ASSERT_FALSE(vm_test_utils::IsMapped(ptr_raw + kMappedSize, 4096)); EXPECT_DEATH({ ptr_raw[kMappedSize] = 'x'; }, ".*"); // Commit the remaining pages. mem.EnsureCommitted(kSize); for (size_t i = kMappedSize / sizeof(uint64_t); i < kSize / sizeof(uint64_t); i++) { ASSERT_EQ(0u, *(reinterpret_cast<uint64_t*>(mem.Get()) + i)); } #elif PERFETTO_BUILDFLAG(PERFETTO_OS_FUCHSIA) // Fuchsia doesn't yet support paging. So this should be a no-op. mem.EnsureCommitted(kSize); for (size_t i = 0; i < kSize / sizeof(uint64_t); i++) ASSERT_EQ(0u, *(reinterpret_cast<uint64_t*>(mem.Get()) + i)); #else // Linux only maps on access. ASSERT_FALSE(vm_test_utils::IsMapped(ptr_raw, kSize)); // This should not have any effect. mem.EnsureCommitted(kSize); ASSERT_FALSE(vm_test_utils::IsMapped(ptr_raw, kSize)); for (size_t i = 0; i < kSize / sizeof(uint64_t); i++) ASSERT_EQ(0u, *(reinterpret_cast<uint64_t*>(mem.Get()) + i)); ASSERT_TRUE(vm_test_utils::IsMapped(ptr_raw, kSize)); #endif } #if !PERFETTO_BUILDFLAG(PERFETTO_OS_FUCHSIA) // Freed memory is necessarily not mapped in to the process. ASSERT_FALSE(vm_test_utils::IsMapped(ptr_raw, kSize)); #endif } #if defined(ADDRESS_SANITIZER) TEST(PagedMemoryTest, AccessUncommittedMemoryTriggersASAN) { EXPECT_DEATH( { constexpr size_t kNumPages = 4096; constexpr size_t kSize = 4096 * kNumPages; PagedMemory mem = PagedMemory::Allocate(kSize, PagedMemory::kDontCommit); ASSERT_TRUE(mem.IsValid()); char* ptr_raw = reinterpret_cast<char*>(mem.Get()); // Only the first 1024 pages are mapped. constexpr size_t kMappedSize = 4096 * 1024; ptr_raw[kMappedSize] = 'x'; abort(); }, "AddressSanitizer: .*"); } #endif // ADDRESS_SANITIZER TEST(PagedMemoryTest, GuardRegions) { const size_t kSize = 4096; PagedMemory mem = PagedMemory::Allocate(kSize); ASSERT_TRUE(mem.IsValid()); volatile char* raw = reinterpret_cast<char*>(mem.Get()); EXPECT_DEATH({ raw[-1] = 'x'; }, ".*"); EXPECT_DEATH({ raw[kSize] = 'x'; }, ".*"); } // Disable this on: // MacOS: because it doesn't seem to have an equivalent rlimit to bound mmap(). // Fuchsia: doesn't support rlimit. // Sanitizers: they seem to try to shadow mmaped memory and fail due to OOMs. #if !PERFETTO_BUILDFLAG(PERFETTO_OS_MACOSX) && \ !PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) && \ !PERFETTO_BUILDFLAG(PERFETTO_OS_FUCHSIA) && !defined(ADDRESS_SANITIZER) && \ !defined(LEAK_SANITIZER) && !defined(THREAD_SANITIZER) && \ !defined(MEMORY_SANITIZER) // Glibc headers hit this on RLIMIT_ macros. #pragma GCC diagnostic push #if defined(__clang__) #pragma GCC diagnostic ignored "-Wdisabled-macro-expansion" #endif TEST(PagedMemoryTest, Unchecked) { const size_t kMemLimit = 256 * 1024 * 1024l; struct rlimit limit { kMemLimit, kMemLimit }; // ASSERT_EXIT here is to spawn the test in a sub-process and avoid // propagating the setrlimit() to other test units in case of failure. ASSERT_EXIT( { ASSERT_EQ(0, setrlimit(RLIMIT_AS, &limit)); auto mem = PagedMemory::Allocate(kMemLimit * 2, PagedMemory::kMayFail); ASSERT_FALSE(mem.IsValid()); // Use _exit() instead of exit() to avoid calling destructors on child // process death, which may interfere with the parent process's test // launcher expectations. _exit(0); }, ::testing::ExitedWithCode(0), ""); } #pragma GCC diagnostic pop #endif } // namespace } // namespace base } // namespace perfetto