|  | /* | 
|  | * 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 <err.h> | 
|  | #include <inttypes.h> | 
|  | #include <pthread.h> | 
|  | #include <stdio.h> | 
|  | #include <string.h> | 
|  | #include <sys/mman.h> | 
|  | #include <sys/user.h> | 
|  | #include <unistd.h> | 
|  |  | 
|  | #include <gtest/gtest.h> | 
|  |  | 
|  | #include <chrono> | 
|  | #include <thread> | 
|  | #include <vector> | 
|  |  | 
|  | #include <android-base/macros.h> | 
|  |  | 
|  | #include "utils.h" | 
|  |  | 
|  | using namespace std::chrono_literals; | 
|  |  | 
|  | static void WaitUntilAllExited(pid_t* pids, size_t pid_count) { | 
|  | // Wait until all children have exited. | 
|  | bool alive = true; | 
|  | while (alive) { | 
|  | alive = false; | 
|  | for (size_t i = 0; i < pid_count; ++i) { | 
|  | if (pids[i] != 0) { | 
|  | if (kill(pids[i], 0) == 0) { | 
|  | alive = true; | 
|  | } else { | 
|  | EXPECT_EQ(errno, ESRCH); | 
|  | pids[i] = 0;  // Skip in next loop. | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | class LeakChecker { | 
|  | public: | 
|  | LeakChecker() { | 
|  | // Avoid resizing and using memory later. | 
|  | // 64Ki is the default limit on VMAs per process. | 
|  | maps_.reserve(64*1024); | 
|  | Reset(); | 
|  | } | 
|  |  | 
|  | ~LeakChecker() { | 
|  | Check(); | 
|  | } | 
|  |  | 
|  | void Reset() { | 
|  | previous_size_ = GetMappingSize(); | 
|  | } | 
|  |  | 
|  | void DumpTo(std::ostream& os) const { | 
|  | os << previous_size_; | 
|  | } | 
|  |  | 
|  | private: | 
|  | size_t previous_size_; | 
|  | std::vector<map_record> maps_; | 
|  |  | 
|  | void Check() { | 
|  | auto current_size = GetMappingSize(); | 
|  | if (current_size > previous_size_) { | 
|  | FAIL() << "increase in process map size: " << previous_size_ << " -> " << current_size; | 
|  | } | 
|  | } | 
|  |  | 
|  | size_t GetMappingSize() { | 
|  | if (!Maps::parse_maps(&maps_)) { | 
|  | err(1, "failed to parse maps"); | 
|  | } | 
|  |  | 
|  | size_t result = 0; | 
|  | for (const map_record& map : maps_) { | 
|  | result += map.addr_end - map.addr_start; | 
|  | } | 
|  |  | 
|  | return result; | 
|  | } | 
|  | }; | 
|  |  | 
|  | std::ostream& operator<<(std::ostream& os, const LeakChecker& lc) { | 
|  | lc.DumpTo(os); | 
|  | return os; | 
|  | } | 
|  |  | 
|  | // http://b/36045112 | 
|  | TEST(pthread_leak, join) { | 
|  | LeakChecker lc; | 
|  |  | 
|  | for (size_t pass = 0; pass < 2; ++pass) { | 
|  | for (int i = 0; i < 100; ++i) { | 
|  | pthread_t thread; | 
|  | ASSERT_EQ(0, pthread_create(&thread, nullptr, [](void*) -> void* { return nullptr; }, nullptr)); | 
|  | ASSERT_EQ(0, pthread_join(thread, nullptr)); | 
|  | } | 
|  |  | 
|  | // A native bridge implementation might need a warm up pass to reach a steady state. | 
|  | // http://b/37920774. | 
|  | if (pass == 0) lc.Reset(); | 
|  | } | 
|  | } | 
|  |  | 
|  | // http://b/36045112 | 
|  | TEST(pthread_leak, detach) { | 
|  | LeakChecker lc; | 
|  | constexpr int kThreadCount = 100; | 
|  |  | 
|  | // Devices with low power cores/low number of cores can not finish test in time hence decreasing | 
|  | // threads count to 90. | 
|  | // http://b/129924384. | 
|  | int threads_count = (sysconf(_SC_NPROCESSORS_CONF) > 2) ? kThreadCount : (kThreadCount - 10); | 
|  |  | 
|  | for (size_t pass = 0; pass < 1; ++pass) { | 
|  | struct thread_data { pthread_barrier_t* barrier; pid_t* tid; } threads[kThreadCount] = {}; | 
|  |  | 
|  | pthread_barrier_t barrier; | 
|  | ASSERT_EQ(pthread_barrier_init(&barrier, nullptr, threads_count + 1), 0); | 
|  |  | 
|  | // Start child threads. | 
|  | pid_t tids[kThreadCount]; | 
|  | for (int i = 0; i < threads_count; ++i) { | 
|  | threads[i] = {&barrier, &tids[i]}; | 
|  | const auto thread_function = +[](void* ptr) -> void* { | 
|  | thread_data* data = static_cast<thread_data*>(ptr); | 
|  | *data->tid = gettid(); | 
|  | pthread_barrier_wait(data->barrier); | 
|  | return nullptr; | 
|  | }; | 
|  | pthread_t thread; | 
|  | ASSERT_EQ(0, pthread_create(&thread, nullptr, thread_function, &threads[i])); | 
|  | ASSERT_EQ(0, pthread_detach(thread)); | 
|  | } | 
|  |  | 
|  | pthread_barrier_wait(&barrier); | 
|  | ASSERT_EQ(pthread_barrier_destroy(&barrier), 0); | 
|  |  | 
|  | WaitUntilAllExited(tids, arraysize(tids)); | 
|  |  | 
|  | // A native bridge implementation might need a warm up pass to reach a steady state. | 
|  | // http://b/37920774. | 
|  | if (pass == 0) lc.Reset(); | 
|  | } | 
|  | } |