/* * Copyright (C) 2015 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "BacktraceTest.h" struct FunctionSymbol { std::string name; uint64_t start; uint64_t end; }; static std::vector GetFunctionSymbols() { std::vector symbols = { {"unknown_start", 0, 0}, {"test_level_one", reinterpret_cast(&BacktraceTest::test_level_one_), 0}, {"test_level_two", reinterpret_cast(&BacktraceTest::test_level_two_), 0}, {"test_level_three", reinterpret_cast(&BacktraceTest::test_level_three_), 0}, {"test_level_four", reinterpret_cast(&BacktraceTest::test_level_four_), 0}, {"test_recursive_call", reinterpret_cast(&BacktraceTest::test_recursive_call_), 0}, {"test_get_context_and_wait", reinterpret_cast(&BacktraceTest::test_get_context_and_wait_), 0}, {"unknown_end", static_cast(-1), static_cast(-1)}, }; std::sort( symbols.begin(), symbols.end(), [](const FunctionSymbol& s1, const FunctionSymbol& s2) { return s1.start < s2.start; }); for (size_t i = 0; i + 1 < symbols.size(); ++i) { symbols[i].end = symbols[i + 1].start; } return symbols; } static std::string RawDataToHexString(const void* data, size_t size) { const uint8_t* p = static_cast(data); std::string s; for (size_t i = 0; i < size; ++i) { s += android::base::StringPrintf("%02x", p[i]); } return s; } static void HexStringToRawData(const char* s, std::vector* data, size_t size) { for (size_t i = 0; i < size; ++i) { int value; sscanf(s, "%02x", &value); data->push_back(value); s += 2; } } struct OfflineThreadArg { std::vector ucontext; pid_t tid; volatile int exit_flag; }; static void* OfflineThreadFunc(void* arg) { OfflineThreadArg* fn_arg = reinterpret_cast(arg); fn_arg->tid = android::base::GetThreadId(); BacktraceTest::test_get_context_and_wait_(&fn_arg->ucontext, &fn_arg->exit_flag); return nullptr; } std::string GetTestPath(const std::string& arch, const std::string& path) { return android::base::GetExecutableDirectory() + "/testdata/" + arch + '/' + path; } // This test is disable because it is for generating test data. TEST_F(BacktraceTest, DISABLED_generate_offline_testdata) { // Create a thread to generate the needed stack and registers information. const size_t stack_size = 16 * 1024; void* stack = mmap(NULL, stack_size, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0); ASSERT_NE(MAP_FAILED, stack); uint64_t stack_addr = reinterpret_cast(stack); pthread_attr_t attr; ASSERT_EQ(0, pthread_attr_init(&attr)); ASSERT_EQ(0, pthread_attr_setstack(&attr, reinterpret_cast(stack), stack_size)); pthread_t thread; OfflineThreadArg arg; arg.exit_flag = 0; ASSERT_EQ(0, pthread_create(&thread, &attr, OfflineThreadFunc, &arg)); // Wait for the offline thread to generate the stack and context information. sleep(1); // Copy the stack information. std::vector stack_data(reinterpret_cast(stack), reinterpret_cast(stack) + stack_size); arg.exit_flag = 1; ASSERT_EQ(0, pthread_join(thread, nullptr)); ASSERT_EQ(0, munmap(stack, stack_size)); std::unique_ptr map(BacktraceMap::Create(getpid())); ASSERT_TRUE(map != nullptr); backtrace_stackinfo_t stack_info; stack_info.start = stack_addr; stack_info.end = stack_addr + stack_size; stack_info.data = stack_data.data(); // Generate offline testdata. std::string testdata; // 1. Dump pid, tid testdata += android::base::StringPrintf("pid: %d tid: %d\n", getpid(), arg.tid); // 2. Dump maps for (auto it = map->begin(); it != map->end(); ++it) { const backtrace_map_t* entry = *it; testdata += android::base::StringPrintf("map: start: %" PRIx64 " end: %" PRIx64 " offset: %" PRIx64 " load_bias: %" PRIx64 " flags: %d name: %s\n", entry->start, entry->end, entry->offset, entry->load_bias, entry->flags, entry->name.c_str()); } // 3. Dump ucontext testdata += android::base::StringPrintf("ucontext: %zu ", arg.ucontext.size()); testdata += RawDataToHexString(arg.ucontext.data(), arg.ucontext.size()); testdata.push_back('\n'); // 4. Dump stack testdata += android::base::StringPrintf( "stack: start: %" PRIx64 " end: %" PRIx64 " size: %zu ", stack_info.start, stack_info.end, stack_data.size()); testdata += RawDataToHexString(stack_data.data(), stack_data.size()); testdata.push_back('\n'); // 5. Dump function symbols std::vector function_symbols = GetFunctionSymbols(); for (const auto& symbol : function_symbols) { testdata += android::base::StringPrintf("function: start: %" PRIx64 " end: %" PRIx64 " name: %s\n", symbol.start, symbol.end, symbol.name.c_str()); } ASSERT_TRUE(android::base::WriteStringToFile(testdata, "offline_testdata")); } // Return the name of the function which matches the address. Although we don't know the // exact end of each function, it is accurate enough for the tests. static std::string FunctionNameForAddress(uint64_t addr, const std::vector& symbols) { for (auto& symbol : symbols) { if (addr >= symbol.start && addr < symbol.end) { return symbol.name; } } return ""; } struct OfflineTestData { int pid; int tid; std::vector maps; std::vector ucontext; backtrace_stackinfo_t stack_info; std::vector stack; std::vector symbols; }; bool ReadOfflineTestData(const std::string offline_testdata_path, OfflineTestData* testdata) { std::string s; if (!android::base::ReadFileToString(offline_testdata_path, &s)) { return false; } // Parse offline_testdata. std::vector lines = android::base::Split(s, "\n"); for (const auto& line : lines) { if (android::base::StartsWith(line, "pid:")) { sscanf(line.c_str(), "pid: %d tid: %d", &testdata->pid, &testdata->tid); } else if (android::base::StartsWith(line, "map:")) { testdata->maps.resize(testdata->maps.size() + 1); backtrace_map_t& map = testdata->maps.back(); int pos; sscanf(line.c_str(), "map: start: %" SCNx64 " end: %" SCNx64 " offset: %" SCNx64 " load_bias: %" SCNx64 " flags: %d name: %n", &map.start, &map.end, &map.offset, &map.load_bias, &map.flags, &pos); map.name = android::base::Trim(line.substr(pos)); } else if (android::base::StartsWith(line, "ucontext:")) { size_t size; int pos; testdata->ucontext.clear(); sscanf(line.c_str(), "ucontext: %zu %n", &size, &pos); HexStringToRawData(&line[pos], &testdata->ucontext, size); } else if (android::base::StartsWith(line, "stack:")) { size_t size; int pos; sscanf(line.c_str(), "stack: start: %" SCNx64 " end: %" SCNx64 " size: %zu %n", &testdata->stack_info.start, &testdata->stack_info.end, &size, &pos); CHECK_EQ(testdata->stack_info.end - testdata->stack_info.start, size); testdata->stack.clear(); HexStringToRawData(&line[pos], &testdata->stack, size); testdata->stack_info.data = testdata->stack.data(); } else if (android::base::StartsWith(line, "function:")) { testdata->symbols.resize(testdata->symbols.size() + 1); FunctionSymbol& symbol = testdata->symbols.back(); int pos; sscanf(line.c_str(), "function: start: %" SCNx64 " end: %" SCNx64 " name: %n", &symbol.start, &symbol.end, &pos); symbol.name = line.substr(pos); } } return true; } static void BacktraceOfflineTest(std::string arch_str, const std::string& testlib_name) { const std::string testlib_path(GetTestPath(arch_str, testlib_name)); const std::string offline_testdata_path(GetTestPath(arch_str, "offline_testdata")); OfflineTestData testdata; ASSERT_TRUE(ReadOfflineTestData(offline_testdata_path, &testdata)) << "Failed " << arch_str; // Fix path of libbacktrace_testlib.so. for (auto& map : testdata.maps) { if (map.name.find("libbacktrace_test.so") != std::string::npos) { map.name = testlib_path; } } Backtrace::ArchEnum arch; if (arch_str == "arm") { arch = Backtrace::ARCH_ARM; } else if (arch_str == "arm64") { arch = Backtrace::ARCH_ARM64; } else if (arch_str == "x86") { arch = Backtrace::ARCH_X86; } else if (arch_str == "x86_64") { arch = Backtrace::ARCH_X86_64; } else { abort(); } std::unique_ptr backtrace(Backtrace::CreateOffline( arch, testdata.pid, testdata.tid, testdata.maps, testdata.stack_info)); ASSERT_TRUE(backtrace != nullptr) << "Failed " << arch_str; ASSERT_TRUE(backtrace->Unwind(0, testdata.ucontext.data())) << "Failed " << arch_str; // Collect pc values of the call stack frames. std::vector pc_values; for (size_t i = 0; i < backtrace->NumFrames(); ++i) { pc_values.push_back(backtrace->GetFrame(i)->pc); } size_t test_one_index = 0; for (size_t i = 0; i < pc_values.size(); ++i) { if (FunctionNameForAddress(pc_values[i], testdata.symbols) == "test_level_one") { test_one_index = i; break; } } ASSERT_GE(test_one_index, 3u) << "Failed " << arch_str; ASSERT_EQ("test_level_one", FunctionNameForAddress(pc_values[test_one_index], testdata.symbols)) << "Failed " << arch_str; ASSERT_EQ("test_level_two", FunctionNameForAddress(pc_values[test_one_index - 1], testdata.symbols)) << "Failed " << arch_str; ASSERT_EQ("test_level_three", FunctionNameForAddress(pc_values[test_one_index - 2], testdata.symbols)) << "Failed " << arch_str; ASSERT_EQ("test_level_four", FunctionNameForAddress(pc_values[test_one_index - 3], testdata.symbols)) << "Failed " << arch_str; } // For now, these tests can only run on the given architectures. TEST_F(BacktraceTest, offline_eh_frame) { BacktraceOfflineTest("arm64", "libbacktrace_test_eh_frame.so"); BacktraceOfflineTest("x86_64", "libbacktrace_test_eh_frame.so"); } TEST_F(BacktraceTest, offline_debug_frame) { BacktraceOfflineTest("arm", "libbacktrace_test_debug_frame.so"); BacktraceOfflineTest("x86", "libbacktrace_test_debug_frame.so"); } TEST_F(BacktraceTest, offline_gnu_debugdata) { BacktraceOfflineTest("arm", "libbacktrace_test_gnu_debugdata.so"); BacktraceOfflineTest("x86", "libbacktrace_test_gnu_debugdata.so"); } TEST_F(BacktraceTest, offline_arm_exidx) { BacktraceOfflineTest("arm", "libbacktrace_test_arm_exidx.so"); } static void LibUnwindingTest(const std::string& arch_str, const std::string& testdata_name, const std::string& testlib_name) { const std::string testlib_path(GetTestPath(arch_str, testlib_name)); struct stat st; ASSERT_EQ(0, stat(testlib_path.c_str(), &st)) << "can't find testlib " << testlib_path; const std::string offline_testdata_path(GetTestPath(arch_str, testdata_name)); OfflineTestData testdata; ASSERT_TRUE(ReadOfflineTestData(offline_testdata_path, &testdata)); // Fix path of the testlib. for (auto& map : testdata.maps) { if (map.name.find(testlib_name) != std::string::npos) { map.name = testlib_path; } } Backtrace::ArchEnum arch; if (arch_str == "arm") { arch = Backtrace::ARCH_ARM; } else if (arch_str == "arm64") { arch = Backtrace::ARCH_ARM64; } else if (arch_str == "x86") { arch = Backtrace::ARCH_X86; } else if (arch_str == "x86_64") { arch = Backtrace::ARCH_X86_64; } else { ASSERT_TRUE(false) << "Unsupported arch " << arch_str; abort(); } // Do offline backtrace. std::unique_ptr backtrace(Backtrace::CreateOffline( arch, testdata.pid, testdata.tid, testdata.maps, testdata.stack_info)); ASSERT_TRUE(backtrace != nullptr); ASSERT_TRUE(backtrace->Unwind(0, testdata.ucontext.data())); ASSERT_EQ(testdata.symbols.size(), backtrace->NumFrames()); for (size_t i = 0; i < backtrace->NumFrames(); ++i) { std::string name = FunctionNameForAddress(backtrace->GetFrame(i)->rel_pc, testdata.symbols); ASSERT_EQ(name, testdata.symbols[i].name); } ASSERT_TRUE(backtrace->GetError().error_code == BACKTRACE_UNWIND_ERROR_ACCESS_MEM_FAILED || backtrace->GetError().error_code == BACKTRACE_UNWIND_ERROR_MAP_MISSING || backtrace->GetError().error_code == BACKTRACE_UNWIND_ERROR_REPEATED_FRAME); } // This test tests the situation that ranges of functions covered by .eh_frame and .ARM.exidx // overlap with each other, which appears in /system/lib/libart.so. TEST_F(BacktraceTest, offline_unwind_mix_eh_frame_and_arm_exidx) { LibUnwindingTest("arm", "offline_testdata_for_libart", "libart.so"); } TEST_F(BacktraceTest, offline_debug_frame_with_load_bias) { LibUnwindingTest("arm", "offline_testdata_for_libandroid_runtime", "libandroid_runtime.so"); } TEST_F(BacktraceTest, offline_try_armexidx_after_debug_frame) { LibUnwindingTest("arm", "offline_testdata_for_libGLESv2_adreno", "libGLESv2_adreno.so"); } TEST_F(BacktraceTest, offline_cie_with_P_augmentation) { // Make sure we can unwind through functions with CIE entry containing P augmentation, which // makes unwinding library reading personality handler from memory. One example is // /system/lib64/libskia.so. LibUnwindingTest("arm64", "offline_testdata_for_libskia", "libskia.so"); } TEST_F(BacktraceTest, offline_empty_eh_frame_hdr) { // Make sure we can unwind through libraries with empty .eh_frame_hdr section. One example is // /vendor/lib64/egl/eglSubDriverAndroid.so. LibUnwindingTest("arm64", "offline_testdata_for_eglSubDriverAndroid", "eglSubDriverAndroid.so"); } TEST_F(BacktraceTest, offline_max_frames_limit) { // The length of callchain can reach 256 when recording an application. ASSERT_GE(MAX_BACKTRACE_FRAMES, 256); }