1 /* 2 * Copyright (C) 2016 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #include <fcntl.h> 18 #include <stdlib.h> 19 #include <sys/prctl.h> 20 #include <unistd.h> 21 22 #include <gtest/gtest.h> 23 24 #include <memunreachable/memunreachable.h> 25 26 namespace android { 27 28 class HiddenPointer { 29 public: 30 explicit HiddenPointer(size_t size = 256) { Set(malloc(size)); } 31 ~HiddenPointer() { Free(); } 32 void* Get() { return reinterpret_cast<void*>(~ptr_); } 33 void Free() { 34 free(Get()); 35 Set(nullptr); 36 } 37 38 private: 39 void Set(void* ptr) { ptr_ = ~reinterpret_cast<uintptr_t>(ptr); } 40 volatile uintptr_t ptr_; 41 }; 42 43 // Trick the compiler into thinking a value on the stack is still referenced. 44 static void Ref(void** ptr) { 45 write(0, ptr, 0); 46 } 47 48 TEST(MemunreachableTest, clean) { 49 UnreachableMemoryInfo info; 50 51 ASSERT_TRUE(LogUnreachableMemory(true, 100)); 52 53 ASSERT_TRUE(GetUnreachableMemory(info)); 54 ASSERT_EQ(0U, info.leaks.size()); 55 } 56 57 TEST(MemunreachableTest, stack) { 58 HiddenPointer hidden_ptr; 59 60 { 61 void* ptr = hidden_ptr.Get(); 62 Ref(&ptr); 63 64 UnreachableMemoryInfo info; 65 66 ASSERT_TRUE(GetUnreachableMemory(info)); 67 ASSERT_EQ(0U, info.leaks.size()); 68 69 ptr = nullptr; 70 } 71 72 { 73 UnreachableMemoryInfo info; 74 75 ASSERT_TRUE(GetUnreachableMemory(info)); 76 ASSERT_EQ(1U, info.leaks.size()); 77 } 78 79 hidden_ptr.Free(); 80 81 { 82 UnreachableMemoryInfo info; 83 84 ASSERT_TRUE(GetUnreachableMemory(info)); 85 ASSERT_EQ(0U, info.leaks.size()); 86 } 87 } 88 89 void* g_ptr; 90 91 TEST(MemunreachableTest, global) { 92 HiddenPointer hidden_ptr; 93 94 g_ptr = hidden_ptr.Get(); 95 96 { 97 UnreachableMemoryInfo info; 98 99 ASSERT_TRUE(GetUnreachableMemory(info)); 100 ASSERT_EQ(0U, info.leaks.size()); 101 } 102 103 g_ptr = nullptr; 104 105 { 106 UnreachableMemoryInfo info; 107 108 ASSERT_TRUE(GetUnreachableMemory(info)); 109 ASSERT_EQ(1U, info.leaks.size()); 110 } 111 112 hidden_ptr.Free(); 113 114 { 115 UnreachableMemoryInfo info; 116 117 ASSERT_TRUE(GetUnreachableMemory(info)); 118 ASSERT_EQ(0U, info.leaks.size()); 119 } 120 } 121 122 TEST(MemunreachableTest, tls) { 123 HiddenPointer hidden_ptr; 124 pthread_key_t key; 125 pthread_key_create(&key, nullptr); 126 127 pthread_setspecific(key, hidden_ptr.Get()); 128 129 { 130 UnreachableMemoryInfo info; 131 132 ASSERT_TRUE(GetUnreachableMemory(info)); 133 ASSERT_EQ(0U, info.leaks.size()); 134 } 135 136 pthread_setspecific(key, nullptr); 137 138 { 139 UnreachableMemoryInfo info; 140 141 ASSERT_TRUE(GetUnreachableMemory(info)); 142 ASSERT_EQ(1U, info.leaks.size()); 143 } 144 145 hidden_ptr.Free(); 146 147 { 148 UnreachableMemoryInfo info; 149 150 ASSERT_TRUE(GetUnreachableMemory(info)); 151 ASSERT_EQ(0U, info.leaks.size()); 152 } 153 154 pthread_key_delete(key); 155 } 156 157 TEST(MemunreachableTest, twice) { 158 HiddenPointer hidden_ptr; 159 160 { 161 UnreachableMemoryInfo info; 162 163 ASSERT_TRUE(GetUnreachableMemory(info)); 164 ASSERT_EQ(1U, info.leaks.size()); 165 } 166 167 { 168 UnreachableMemoryInfo info; 169 170 ASSERT_TRUE(GetUnreachableMemory(info)); 171 ASSERT_EQ(1U, info.leaks.size()); 172 } 173 174 hidden_ptr.Free(); 175 176 { 177 UnreachableMemoryInfo info; 178 179 ASSERT_TRUE(GetUnreachableMemory(info)); 180 ASSERT_EQ(0U, info.leaks.size()); 181 } 182 } 183 184 TEST(MemunreachableTest, log) { 185 HiddenPointer hidden_ptr; 186 187 ASSERT_TRUE(LogUnreachableMemory(true, 100)); 188 189 hidden_ptr.Free(); 190 191 { 192 UnreachableMemoryInfo info; 193 194 ASSERT_TRUE(GetUnreachableMemory(info)); 195 ASSERT_EQ(0U, info.leaks.size()); 196 } 197 } 198 199 TEST(MemunreachableTest, notdumpable) { 200 ASSERT_EQ(0, prctl(PR_SET_DUMPABLE, 0)); 201 202 HiddenPointer hidden_ptr; 203 204 ASSERT_TRUE(LogUnreachableMemory(true, 100)); 205 206 ASSERT_EQ(0, prctl(PR_SET_DUMPABLE, 1)); 207 } 208 209 TEST(MemunreachableTest, leak_lots) { 210 std::vector<HiddenPointer> hidden_ptrs; 211 hidden_ptrs.resize(1024); 212 213 ASSERT_TRUE(LogUnreachableMemory(true, 100)); 214 } 215 216 } // namespace android 217