1*288bf522SAndroid Build Coastguard Worker /*
2*288bf522SAndroid Build Coastguard Worker * Copyright (C) 2018 The Android Open Source Project
3*288bf522SAndroid Build Coastguard Worker *
4*288bf522SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*288bf522SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*288bf522SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*288bf522SAndroid Build Coastguard Worker *
8*288bf522SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*288bf522SAndroid Build Coastguard Worker *
10*288bf522SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*288bf522SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*288bf522SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*288bf522SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*288bf522SAndroid Build Coastguard Worker * limitations under the License.
15*288bf522SAndroid Build Coastguard Worker */
16*288bf522SAndroid Build Coastguard Worker
17*288bf522SAndroid Build Coastguard Worker #include "dso.h"
18*288bf522SAndroid Build Coastguard Worker
19*288bf522SAndroid Build Coastguard Worker #include <gtest/gtest.h>
20*288bf522SAndroid Build Coastguard Worker
21*288bf522SAndroid Build Coastguard Worker #include <android-base/file.h>
22*288bf522SAndroid Build Coastguard Worker #include <android-base/stringprintf.h>
23*288bf522SAndroid Build Coastguard Worker #include <android-base/test_utils.h>
24*288bf522SAndroid Build Coastguard Worker
25*288bf522SAndroid Build Coastguard Worker #include "get_test_data.h"
26*288bf522SAndroid Build Coastguard Worker #include "read_apk.h"
27*288bf522SAndroid Build Coastguard Worker #include "thread_tree.h"
28*288bf522SAndroid Build Coastguard Worker #include "utils.h"
29*288bf522SAndroid Build Coastguard Worker
30*288bf522SAndroid Build Coastguard Worker using namespace simpleperf;
31*288bf522SAndroid Build Coastguard Worker using namespace simpleperf_dso_impl;
32*288bf522SAndroid Build Coastguard Worker
33*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(DebugElfFileFinder,use_build_id_list)34*288bf522SAndroid Build Coastguard Worker TEST(DebugElfFileFinder, use_build_id_list) {
35*288bf522SAndroid Build Coastguard Worker // Create a temp symdir with build_id_list.
36*288bf522SAndroid Build Coastguard Worker TemporaryDir tmpdir;
37*288bf522SAndroid Build Coastguard Worker TemporaryFile tmpfile(tmpdir.path);
38*288bf522SAndroid Build Coastguard Worker std::string data;
39*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(android::base::ReadFileToString(GetTestData(ELF_FILE), &data));
40*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(android::base::WriteStringToFile(data, tmpfile.path));
41*288bf522SAndroid Build Coastguard Worker BuildId build_id(ELF_FILE_BUILD_ID);
42*288bf522SAndroid Build Coastguard Worker std::string build_id_list = android::base::StringPrintf(
43*288bf522SAndroid Build Coastguard Worker "%s=%s\n", build_id.ToString().c_str(), android::base::Basename(tmpfile.path).c_str());
44*288bf522SAndroid Build Coastguard Worker std::string build_id_list_file = std::string(tmpdir.path) + "/build_id_list";
45*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(android::base::WriteStringToFile(build_id_list, build_id_list_file));
46*288bf522SAndroid Build Coastguard Worker
47*288bf522SAndroid Build Coastguard Worker DebugElfFileFinder finder;
48*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(finder.SetSymFsDir(tmpdir.path));
49*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("elf", false, build_id), std::string(tmpfile.path));
50*288bf522SAndroid Build Coastguard Worker unlink(build_id_list_file.c_str());
51*288bf522SAndroid Build Coastguard Worker }
52*288bf522SAndroid Build Coastguard Worker
ConvertPathSeparator(const std::string & path)53*288bf522SAndroid Build Coastguard Worker static std::string ConvertPathSeparator(const std::string& path) {
54*288bf522SAndroid Build Coastguard Worker std::string result = path;
55*288bf522SAndroid Build Coastguard Worker if (OS_PATH_SEPARATOR != '/') {
56*288bf522SAndroid Build Coastguard Worker std::replace(result.begin(), result.end(), '/', OS_PATH_SEPARATOR);
57*288bf522SAndroid Build Coastguard Worker }
58*288bf522SAndroid Build Coastguard Worker return result;
59*288bf522SAndroid Build Coastguard Worker }
60*288bf522SAndroid Build Coastguard Worker
61*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(DebugElfFileFinder,concatenating_symfs_dir)62*288bf522SAndroid Build Coastguard Worker TEST(DebugElfFileFinder, concatenating_symfs_dir) {
63*288bf522SAndroid Build Coastguard Worker DebugElfFileFinder finder;
64*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(finder.SetSymFsDir(GetTestDataDir()));
65*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.GetPathInSymFsDir("/system/libc.so"),
66*288bf522SAndroid Build Coastguard Worker GetTestDataDir() + "system" + OS_PATH_SEPARATOR + "libc.so");
67*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.GetPathInSymFsDir("/data/base.apk!/lib/base.so"),
68*288bf522SAndroid Build Coastguard Worker GetTestDataDir() + "data" + OS_PATH_SEPARATOR + "base.apk!/lib/base.so");
69*288bf522SAndroid Build Coastguard Worker
70*288bf522SAndroid Build Coastguard Worker BuildId build_id(ELF_FILE_BUILD_ID);
71*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile(ELF_FILE, false, build_id), GetTestDataDir() + ELF_FILE);
72*288bf522SAndroid Build Coastguard Worker std::string native_lib_in_apk = APK_FILE + "!/" + NATIVELIB_IN_APK;
73*288bf522SAndroid Build Coastguard Worker std::string apk_path = ConvertPathSeparator(APK_FILE);
74*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile(native_lib_in_apk, false, native_lib_build_id),
75*288bf522SAndroid Build Coastguard Worker GetTestDataDir() + apk_path + "!/" + NATIVELIB_IN_APK);
76*288bf522SAndroid Build Coastguard Worker }
77*288bf522SAndroid Build Coastguard Worker
78*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(DebugElfFileFinder,use_vdso)79*288bf522SAndroid Build Coastguard Worker TEST(DebugElfFileFinder, use_vdso) {
80*288bf522SAndroid Build Coastguard Worker DebugElfFileFinder finder;
81*288bf522SAndroid Build Coastguard Worker std::string fake_vdso32 = "fake_vdso32";
82*288bf522SAndroid Build Coastguard Worker std::string fake_vdso64 = "fake_vdso64";
83*288bf522SAndroid Build Coastguard Worker finder.SetVdsoFile(fake_vdso32, false);
84*288bf522SAndroid Build Coastguard Worker finder.SetVdsoFile(fake_vdso64, true);
85*288bf522SAndroid Build Coastguard Worker BuildId build_id;
86*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("[vdso]", false, build_id), fake_vdso32);
87*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("[vdso]", true, build_id), fake_vdso64);
88*288bf522SAndroid Build Coastguard Worker }
89*288bf522SAndroid Build Coastguard Worker
90*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(DebugElfFileFinder,add_symbol_dir)91*288bf522SAndroid Build Coastguard Worker TEST(DebugElfFileFinder, add_symbol_dir) {
92*288bf522SAndroid Build Coastguard Worker DebugElfFileFinder finder;
93*288bf522SAndroid Build Coastguard Worker ASSERT_FALSE(finder.AddSymbolDir(GetTestDataDir() + "dir_not_exist"));
94*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("elf", false, CHECK_ELF_FILE_BUILD_ID), "elf");
95*288bf522SAndroid Build Coastguard Worker std::string symfs_dir = ConvertPathSeparator(GetTestDataDir() + CORRECT_SYMFS_FOR_BUILD_ID_CHECK);
96*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(finder.AddSymbolDir(symfs_dir));
97*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("elf", false, CHECK_ELF_FILE_BUILD_ID),
98*288bf522SAndroid Build Coastguard Worker symfs_dir + OS_PATH_SEPARATOR + "elf_for_build_id_check");
99*288bf522SAndroid Build Coastguard Worker }
100*288bf522SAndroid Build Coastguard Worker
101*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(DebugElfFileFinder,allow_mismatched_build_id)102*288bf522SAndroid Build Coastguard Worker TEST(DebugElfFileFinder, allow_mismatched_build_id) {
103*288bf522SAndroid Build Coastguard Worker DebugElfFileFinder finder;
104*288bf522SAndroid Build Coastguard Worker std::string symfs_dir = ConvertPathSeparator(GetTestDataDir() + CORRECT_SYMFS_FOR_BUILD_ID_CHECK);
105*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(finder.AddSymbolDir(symfs_dir));
106*288bf522SAndroid Build Coastguard Worker // By default, can't find binary with empty or different build ids.
107*288bf522SAndroid Build Coastguard Worker BuildId empty_build_id;
108*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("/data/elf_for_build_id_check", false, empty_build_id),
109*288bf522SAndroid Build Coastguard Worker "/data/elf_for_build_id_check");
110*288bf522SAndroid Build Coastguard Worker BuildId different_build_id("01234");
111*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("/data/elf_for_build_id_check", false, different_build_id),
112*288bf522SAndroid Build Coastguard Worker "/data/elf_for_build_id_check");
113*288bf522SAndroid Build Coastguard Worker
114*288bf522SAndroid Build Coastguard Worker // With AllowMismatchedBuildId(), can find binary with empty and different build ids.
115*288bf522SAndroid Build Coastguard Worker finder.AllowMismatchedBuildId();
116*288bf522SAndroid Build Coastguard Worker empty_build_id = BuildId();
117*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("/data/elf_for_build_id_check", false, empty_build_id),
118*288bf522SAndroid Build Coastguard Worker symfs_dir + OS_PATH_SEPARATOR + "elf_for_build_id_check");
119*288bf522SAndroid Build Coastguard Worker different_build_id = BuildId("01234");
120*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("/data/elf_for_build_id_check", false, different_build_id),
121*288bf522SAndroid Build Coastguard Worker symfs_dir + OS_PATH_SEPARATOR + "elf_for_build_id_check");
122*288bf522SAndroid Build Coastguard Worker }
123*288bf522SAndroid Build Coastguard Worker
124*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(DebugElfFileFinder,build_id_list)125*288bf522SAndroid Build Coastguard Worker TEST(DebugElfFileFinder, build_id_list) {
126*288bf522SAndroid Build Coastguard Worker DebugElfFileFinder finder;
127*288bf522SAndroid Build Coastguard Worker // Find file in symfs dir with correct build_id_list.
128*288bf522SAndroid Build Coastguard Worker std::string symfs_dir = ConvertPathSeparator(GetTestDataDir() + "data/symfs_with_build_id_list");
129*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(finder.SetSymFsDir(symfs_dir));
130*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("elf", false, CHECK_ELF_FILE_BUILD_ID),
131*288bf522SAndroid Build Coastguard Worker symfs_dir + OS_PATH_SEPARATOR + "elf_for_build_id_check");
132*288bf522SAndroid Build Coastguard Worker
133*288bf522SAndroid Build Coastguard Worker // Find file in symfs_dir with wrong build_id_list.
134*288bf522SAndroid Build Coastguard Worker symfs_dir = ConvertPathSeparator(GetTestDataDir() + "data/symfs_with_wrong_build_id_list");
135*288bf522SAndroid Build Coastguard Worker finder.Reset();
136*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(finder.SetSymFsDir(symfs_dir));
137*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("elf", false, CHECK_ELF_FILE_BUILD_ID), "elf");
138*288bf522SAndroid Build Coastguard Worker }
139*288bf522SAndroid Build Coastguard Worker
140*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(DebugElfFileFinder,no_build_id)141*288bf522SAndroid Build Coastguard Worker TEST(DebugElfFileFinder, no_build_id) {
142*288bf522SAndroid Build Coastguard Worker DebugElfFileFinder finder;
143*288bf522SAndroid Build Coastguard Worker // If not given a build id, we should match an elf in symfs without build id.
144*288bf522SAndroid Build Coastguard Worker std::string symfs_dir = ConvertPathSeparator(GetTestDataDir() + "data/symfs_without_build_id");
145*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(finder.SetSymFsDir(symfs_dir));
146*288bf522SAndroid Build Coastguard Worker BuildId build_id;
147*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("elf", false, build_id), symfs_dir + OS_PATH_SEPARATOR + "elf");
148*288bf522SAndroid Build Coastguard Worker }
149*288bf522SAndroid Build Coastguard Worker
150*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(DebugElfFileFinder,find_basename_in_symfs_dir)151*288bf522SAndroid Build Coastguard Worker TEST(DebugElfFileFinder, find_basename_in_symfs_dir) {
152*288bf522SAndroid Build Coastguard Worker DebugElfFileFinder finder;
153*288bf522SAndroid Build Coastguard Worker // Find normal elf file.
154*288bf522SAndroid Build Coastguard Worker finder.SetSymFsDir(GetTestDataDir());
155*288bf522SAndroid Build Coastguard Worker BuildId build_id(ELF_FILE_BUILD_ID);
156*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("random_dir/elf", false, build_id), GetTestData("elf"));
157*288bf522SAndroid Build Coastguard Worker
158*288bf522SAndroid Build Coastguard Worker // Find embedded native library.
159*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("base.apk!/lib/x86_64/elf", false, build_id), GetTestData("elf"));
160*288bf522SAndroid Build Coastguard Worker
161*288bf522SAndroid Build Coastguard Worker // Find elf file without build id.
162*288bf522SAndroid Build Coastguard Worker std::string symfs_dir = ConvertPathSeparator(GetTestDataDir() + "data/symfs_without_build_id");
163*288bf522SAndroid Build Coastguard Worker finder.SetSymFsDir(symfs_dir);
164*288bf522SAndroid Build Coastguard Worker build_id = BuildId();
165*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(finder.FindDebugFile("random_dir/elf", false, build_id),
166*288bf522SAndroid Build Coastguard Worker symfs_dir + OS_PATH_SEPARATOR + "elf");
167*288bf522SAndroid Build Coastguard Worker }
168*288bf522SAndroid Build Coastguard Worker
169*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(DebugElfFileFinder,build_id_mismatch)170*288bf522SAndroid Build Coastguard Worker TEST(DebugElfFileFinder, build_id_mismatch) {
171*288bf522SAndroid Build Coastguard Worker DebugElfFileFinder finder;
172*288bf522SAndroid Build Coastguard Worker finder.SetSymFsDir(GetTestDataDir());
173*288bf522SAndroid Build Coastguard Worker CapturedStderr capture;
174*288bf522SAndroid Build Coastguard Worker BuildId mismatch_build_id("0c12a384a9f4a3f3659b7171ca615dbec3a81f71");
175*288bf522SAndroid Build Coastguard Worker std::string debug_file = finder.FindDebugFile(ELF_FILE, false, mismatch_build_id);
176*288bf522SAndroid Build Coastguard Worker capture.Stop();
177*288bf522SAndroid Build Coastguard Worker std::string stderr_output = capture.str();
178*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(debug_file, ELF_FILE);
179*288bf522SAndroid Build Coastguard Worker ASSERT_NE(stderr_output.find("build id mismatch"), std::string::npos);
180*288bf522SAndroid Build Coastguard Worker }
181*288bf522SAndroid Build Coastguard Worker
182*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,dex_file_dso)183*288bf522SAndroid Build Coastguard Worker TEST(dso, dex_file_dso) {
184*288bf522SAndroid Build Coastguard Worker #if defined(__linux__)
185*288bf522SAndroid Build Coastguard Worker for (DsoType dso_type : {DSO_DEX_FILE, DSO_ELF_FILE}) {
186*288bf522SAndroid Build Coastguard Worker std::unique_ptr<Dso> dso = Dso::CreateDso(dso_type, GetTestData("base.vdex"));
187*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(dso);
188*288bf522SAndroid Build Coastguard Worker dso->AddDexFileOffset(0x28);
189*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(DSO_DEX_FILE, dso->type());
190*288bf522SAndroid Build Coastguard Worker const Symbol* symbol = dso->FindSymbol(0x6c77e);
191*288bf522SAndroid Build Coastguard Worker ASSERT_NE(symbol, nullptr);
192*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(symbol->addr, static_cast<uint64_t>(0x6c77e));
193*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(symbol->len, static_cast<uint64_t>(0x16));
194*288bf522SAndroid Build Coastguard Worker ASSERT_STREQ(symbol->DemangledName(),
195*288bf522SAndroid Build Coastguard Worker "com.example.simpleperf.simpleperfexamplewithnative.MixActivity$1.run");
196*288bf522SAndroid Build Coastguard Worker uint64_t min_vaddr;
197*288bf522SAndroid Build Coastguard Worker uint64_t file_offset_of_min_vaddr;
198*288bf522SAndroid Build Coastguard Worker dso->GetMinExecutableVaddr(&min_vaddr, &file_offset_of_min_vaddr);
199*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(min_vaddr, 0);
200*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(file_offset_of_min_vaddr, 0);
201*288bf522SAndroid Build Coastguard Worker
202*288bf522SAndroid Build Coastguard Worker // Don't crash on not exist zip entry.
203*288bf522SAndroid Build Coastguard Worker dso = Dso::CreateDso(dso_type, GetTestData("base.zip!/not_exist_entry"));
204*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(dso);
205*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(nullptr, dso->FindSymbol(0));
206*288bf522SAndroid Build Coastguard Worker }
207*288bf522SAndroid Build Coastguard Worker #else
208*288bf522SAndroid Build Coastguard Worker GTEST_LOG_(INFO) << "This test only runs on linux because of libdexfile";
209*288bf522SAndroid Build Coastguard Worker #endif // defined(__linux__)
210*288bf522SAndroid Build Coastguard Worker }
211*288bf522SAndroid Build Coastguard Worker
212*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,dex_file_offsets)213*288bf522SAndroid Build Coastguard Worker TEST(dso, dex_file_offsets) {
214*288bf522SAndroid Build Coastguard Worker std::unique_ptr<Dso> dso = Dso::CreateDso(DSO_DEX_FILE, "");
215*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(dso);
216*288bf522SAndroid Build Coastguard Worker for (uint64_t offset : {0x3, 0x1, 0x5, 0x4, 0x2, 0x4, 0x3}) {
217*288bf522SAndroid Build Coastguard Worker dso->AddDexFileOffset(offset);
218*288bf522SAndroid Build Coastguard Worker }
219*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(*dso->DexFileOffsets(), std::vector<uint64_t>({0x1, 0x2, 0x3, 0x4, 0x5}));
220*288bf522SAndroid Build Coastguard Worker }
221*288bf522SAndroid Build Coastguard Worker
222*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,embedded_elf)223*288bf522SAndroid Build Coastguard Worker TEST(dso, embedded_elf) {
224*288bf522SAndroid Build Coastguard Worker const std::string file_path = GetUrlInApk(GetTestData(APK_FILE), NATIVELIB_IN_APK);
225*288bf522SAndroid Build Coastguard Worker std::unique_ptr<Dso> dso = Dso::CreateDso(DSO_ELF_FILE, file_path);
226*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(dso);
227*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(dso->Path(), file_path);
228*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(dso->GetDebugFilePath(), file_path);
229*288bf522SAndroid Build Coastguard Worker uint64_t min_vaddr;
230*288bf522SAndroid Build Coastguard Worker uint64_t file_offset_of_min_vaddr;
231*288bf522SAndroid Build Coastguard Worker dso->GetMinExecutableVaddr(&min_vaddr, &file_offset_of_min_vaddr);
232*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(min_vaddr, 0);
233*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(file_offset_of_min_vaddr, 0);
234*288bf522SAndroid Build Coastguard Worker const Symbol* symbol = dso->FindSymbol(0x9a4);
235*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(symbol != nullptr);
236*288bf522SAndroid Build Coastguard Worker ASSERT_STREQ(symbol->Name(), "Java_com_example_hellojni_HelloJni_callFunc1");
237*288bf522SAndroid Build Coastguard Worker BuildId build_id;
238*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(GetBuildIdFromDsoPath(file_path, &build_id));
239*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(build_id, native_lib_build_id);
240*288bf522SAndroid Build Coastguard Worker }
241*288bf522SAndroid Build Coastguard Worker
242*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,IpToVaddrInFile)243*288bf522SAndroid Build Coastguard Worker TEST(dso, IpToVaddrInFile) {
244*288bf522SAndroid Build Coastguard Worker std::unique_ptr<Dso> dso = Dso::CreateDso(DSO_ELF_FILE, GetTestData("libc.so"));
245*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(dso);
246*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(0xa5140, dso->IpToVaddrInFile(0xe9201140, 0xe9201000, 0xa5000));
247*288bf522SAndroid Build Coastguard Worker }
248*288bf522SAndroid Build Coastguard Worker
249*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,kernel_address_randomization)250*288bf522SAndroid Build Coastguard Worker TEST(dso, kernel_address_randomization) {
251*288bf522SAndroid Build Coastguard Worker // Use ELF_FILE as a fake kernel vmlinux.
252*288bf522SAndroid Build Coastguard Worker const std::string vmlinux_path = GetTestData(ELF_FILE);
253*288bf522SAndroid Build Coastguard Worker Dso::SetVmlinux(vmlinux_path);
254*288bf522SAndroid Build Coastguard Worker std::unique_ptr<Dso> dso = Dso::CreateDso(DSO_KERNEL, DEFAULT_KERNEL_MMAP_NAME);
255*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(dso);
256*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(dso->GetDebugFilePath(), vmlinux_path);
257*288bf522SAndroid Build Coastguard Worker // When map_start = 0, can't fix kernel address randomization. So vmlinux isn't used.
258*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(dso->IpToVaddrInFile(0x800500, 0, 0), 0x800500);
259*288bf522SAndroid Build Coastguard Worker ASSERT_FALSE(dso->IpToFileOffset(0x800500, 0, 0));
260*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(dso->FindSymbol(0x400510) == nullptr);
261*288bf522SAndroid Build Coastguard Worker
262*288bf522SAndroid Build Coastguard Worker dso = Dso::CreateDso(DSO_KERNEL, DEFAULT_KERNEL_MMAP_NAME);
263*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(dso);
264*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(dso->GetDebugFilePath(), vmlinux_path);
265*288bf522SAndroid Build Coastguard Worker // When map_start != 0, can fix kernel address randomization. So vmlinux is used.
266*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(dso->IpToVaddrInFile(0x800500, 0x800400, 0), 0x400500);
267*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(dso->IpToFileOffset(0x800500, 0x800400, 0).value(), 0x500);
268*288bf522SAndroid Build Coastguard Worker const Symbol* symbol = dso->FindSymbol(0x400510);
269*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(symbol != nullptr);
270*288bf522SAndroid Build Coastguard Worker ASSERT_STREQ(symbol->Name(), "GlobalFunc");
271*288bf522SAndroid Build Coastguard Worker }
272*288bf522SAndroid Build Coastguard Worker
273*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,find_vmlinux_in_symdirs)274*288bf522SAndroid Build Coastguard Worker TEST(dso, find_vmlinux_in_symdirs) {
275*288bf522SAndroid Build Coastguard Worker // Create a symdir.
276*288bf522SAndroid Build Coastguard Worker TemporaryDir tmpdir;
277*288bf522SAndroid Build Coastguard Worker std::string vmlinux_path = std::string(tmpdir.path) + OS_PATH_SEPARATOR + "elf";
278*288bf522SAndroid Build Coastguard Worker std::string data;
279*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(android::base::ReadFileToString(GetTestData(ELF_FILE), &data));
280*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(android::base::WriteStringToFile(data, vmlinux_path));
281*288bf522SAndroid Build Coastguard Worker
282*288bf522SAndroid Build Coastguard Worker // Find vmlinux in symbol dirs.
283*288bf522SAndroid Build Coastguard Worker Dso::SetVmlinux("");
284*288bf522SAndroid Build Coastguard Worker Dso::AddSymbolDir(tmpdir.path);
285*288bf522SAndroid Build Coastguard Worker Dso::SetBuildIds({std::make_pair(DEFAULT_KERNEL_MMAP_NAME, BuildId(ELF_FILE_BUILD_ID))});
286*288bf522SAndroid Build Coastguard Worker std::unique_ptr<Dso> dso = Dso::CreateDso(DSO_KERNEL, DEFAULT_KERNEL_MMAP_NAME);
287*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(dso);
288*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(dso->GetDebugFilePath(), vmlinux_path);
289*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(0x400927, dso->IpToVaddrInFile(0x800527, 0x800000, 0));
290*288bf522SAndroid Build Coastguard Worker
291*288bf522SAndroid Build Coastguard Worker // Find vmlinux by CreateDsoWithBuildId.
292*288bf522SAndroid Build Coastguard Worker Dso::SetBuildIds({});
293*288bf522SAndroid Build Coastguard Worker BuildId build_id(ELF_FILE_BUILD_ID);
294*288bf522SAndroid Build Coastguard Worker dso = Dso::CreateDsoWithBuildId(DSO_KERNEL, DEFAULT_KERNEL_MMAP_NAME, build_id);
295*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(dso);
296*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(dso->GetDebugFilePath(), vmlinux_path);
297*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(0x400927, dso->IpToVaddrInFile(0x800527, 0x800000, 0));
298*288bf522SAndroid Build Coastguard Worker }
299*288bf522SAndroid Build Coastguard Worker
300*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,kernel_module)301*288bf522SAndroid Build Coastguard Worker TEST(dso, kernel_module) {
302*288bf522SAndroid Build Coastguard Worker // Test finding debug files for kernel modules.
303*288bf522SAndroid Build Coastguard Worker Dso::SetSymFsDir(GetTestDataDir());
304*288bf522SAndroid Build Coastguard Worker std::vector<std::pair<std::string, BuildId>> build_ids;
305*288bf522SAndroid Build Coastguard Worker build_ids.emplace_back(ELF_FILE, BuildId(ELF_FILE_BUILD_ID));
306*288bf522SAndroid Build Coastguard Worker Dso::SetBuildIds(build_ids);
307*288bf522SAndroid Build Coastguard Worker std::unique_ptr<Dso> kernel_dso = Dso::CreateDso(DSO_KERNEL, DEFAULT_KERNEL_MMAP_NAME);
308*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(kernel_dso);
309*288bf522SAndroid Build Coastguard Worker std::unique_ptr<Dso> dso = Dso::CreateKernelModuleDso(ELF_FILE, 0, 0, kernel_dso.get());
310*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(dso->GetDebugFilePath(), GetTestData(ELF_FILE));
311*288bf522SAndroid Build Coastguard Worker }
312*288bf522SAndroid Build Coastguard Worker
313*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,kernel_module_CalculateMinVaddr)314*288bf522SAndroid Build Coastguard Worker TEST(dso, kernel_module_CalculateMinVaddr) {
315*288bf522SAndroid Build Coastguard Worker // Create fake Dso objects.
316*288bf522SAndroid Build Coastguard Worker auto kernel_dso = Dso::CreateDso(DSO_KERNEL, DEFAULT_KERNEL_MMAP_NAME);
317*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(kernel_dso);
318*288bf522SAndroid Build Coastguard Worker const uint64_t module_memory_start = 0xffffffa9bc790000ULL;
319*288bf522SAndroid Build Coastguard Worker const uint64_t module_memory_size = 0x8d7000ULL;
320*288bf522SAndroid Build Coastguard Worker TemporaryFile tmpfile;
321*288bf522SAndroid Build Coastguard Worker auto module_dso =
322*288bf522SAndroid Build Coastguard Worker Dso::CreateKernelModuleDso(tmpfile.path, module_memory_start,
323*288bf522SAndroid Build Coastguard Worker module_memory_start + module_memory_size, kernel_dso.get());
324*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(module_dso);
325*288bf522SAndroid Build Coastguard Worker
326*288bf522SAndroid Build Coastguard Worker // Provide symbol info for calculating min vaddr.
327*288bf522SAndroid Build Coastguard Worker std::vector<Symbol> kernel_symbols;
328*288bf522SAndroid Build Coastguard Worker kernel_symbols.emplace_back("fake_module_function [fake_module]", 0xffffffa9bc7a64e8ULL, 0x60c);
329*288bf522SAndroid Build Coastguard Worker kernel_dso->SetSymbols(&kernel_symbols);
330*288bf522SAndroid Build Coastguard Worker std::vector<Symbol> module_symbols;
331*288bf522SAndroid Build Coastguard Worker module_symbols.emplace_back("fake_module_function", 0x144e8, 0x60c);
332*288bf522SAndroid Build Coastguard Worker module_dso->SetSymbols(&module_symbols);
333*288bf522SAndroid Build Coastguard Worker
334*288bf522SAndroid Build Coastguard Worker // Calculate min vaddr.
335*288bf522SAndroid Build Coastguard Worker uint64_t min_vaddr;
336*288bf522SAndroid Build Coastguard Worker uint64_t memory_offset;
337*288bf522SAndroid Build Coastguard Worker module_dso->GetMinExecutableVaddr(&min_vaddr, &memory_offset);
338*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(min_vaddr, 0x144e8);
339*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(memory_offset, 0x164e8);
340*288bf522SAndroid Build Coastguard Worker
341*288bf522SAndroid Build Coastguard Worker // Use min vaddr in IpToVaddrInFile().
342*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(module_dso->IpToVaddrInFile(0xffffffa9bc7a64e8ULL, module_memory_start, 0), 0x144e8);
343*288bf522SAndroid Build Coastguard Worker }
344*288bf522SAndroid Build Coastguard Worker
345*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,symbol_map_file)346*288bf522SAndroid Build Coastguard Worker TEST(dso, symbol_map_file) {
347*288bf522SAndroid Build Coastguard Worker auto dso = Dso::CreateDso(DSO_SYMBOL_MAP_FILE, "perf-123.map");
348*288bf522SAndroid Build Coastguard Worker ASSERT_TRUE(dso);
349*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(DSO_SYMBOL_MAP_FILE, dso->type());
350*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(0x12345678, dso->IpToVaddrInFile(0x12345678, 0x0, 0x0));
351*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(0x12345678, dso->IpToVaddrInFile(0x12345678, 0xe9201000, 0xa5000));
352*288bf522SAndroid Build Coastguard Worker }
353*288bf522SAndroid Build Coastguard Worker
354*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,FunctionName)355*288bf522SAndroid Build Coastguard Worker TEST(dso, FunctionName) {
356*288bf522SAndroid Build Coastguard Worker Symbol symbol = Symbol("void ctep.v(cteo, ctgc, ctbn)", 0x0, 0x1);
357*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(symbol.FunctionName(), "ctep.v");
358*288bf522SAndroid Build Coastguard Worker symbol = Symbol("ctep.v(cteo, ctgc, ctbn)", 0x0, 0x1);
359*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(symbol.FunctionName(), "ctep.v");
360*288bf522SAndroid Build Coastguard Worker symbol = Symbol("ctep.v", 0x0, 0x1);
361*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(symbol.FunctionName(), "ctep.v");
362*288bf522SAndroid Build Coastguard Worker }
363*288bf522SAndroid Build Coastguard Worker
364*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,search_debug_file_only_when_needed)365*288bf522SAndroid Build Coastguard Worker TEST(dso, search_debug_file_only_when_needed) {
366*288bf522SAndroid Build Coastguard Worker Dso::SetBuildIds({std::make_pair("/elf", BuildId("1b12a384a9f4a3f3659b7171ca615dbec3a81f71"))});
367*288bf522SAndroid Build Coastguard Worker Dso::SetSymFsDir(GetTestDataDir());
368*288bf522SAndroid Build Coastguard Worker CapturedStderr capture;
369*288bf522SAndroid Build Coastguard Worker auto dso = Dso::CreateDso(DSO_ELF_FILE, "/elf");
370*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(capture.str().find("build id mismatch"), std::string::npos);
371*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(dso->GetDebugFilePath(), "/elf");
372*288bf522SAndroid Build Coastguard Worker ASSERT_NE(capture.str().find("build id mismatch"), std::string::npos);
373*288bf522SAndroid Build Coastguard Worker capture.Stop();
374*288bf522SAndroid Build Coastguard Worker }
375*288bf522SAndroid Build Coastguard Worker
376*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,read_symbol_warning)377*288bf522SAndroid Build Coastguard Worker TEST(dso, read_symbol_warning) {
378*288bf522SAndroid Build Coastguard Worker {
379*288bf522SAndroid Build Coastguard Worker // Don't warn when the file may not be an ELF file.
380*288bf522SAndroid Build Coastguard Worker auto dso = Dso::CreateDso(DSO_ELF_FILE, GetTestData("not_exist_file"));
381*288bf522SAndroid Build Coastguard Worker CapturedStderr capture;
382*288bf522SAndroid Build Coastguard Worker dso->LoadSymbols();
383*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(capture.str().find("failed to read symbols"), std::string::npos);
384*288bf522SAndroid Build Coastguard Worker }
385*288bf522SAndroid Build Coastguard Worker {
386*288bf522SAndroid Build Coastguard Worker // Don't warn when the file may not be an ELF file.
387*288bf522SAndroid Build Coastguard Worker auto dso = Dso::CreateDso(DSO_ELF_FILE, GetTestData("base.vdex"));
388*288bf522SAndroid Build Coastguard Worker CapturedStderr capture;
389*288bf522SAndroid Build Coastguard Worker dso->LoadSymbols();
390*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(capture.str().find("failed to read symbols"), std::string::npos);
391*288bf522SAndroid Build Coastguard Worker }
392*288bf522SAndroid Build Coastguard Worker {
393*288bf522SAndroid Build Coastguard Worker // Warn when the file is an ELF file (having a build id).
394*288bf522SAndroid Build Coastguard Worker std::string file_path = GetTestData("not_exist_file");
395*288bf522SAndroid Build Coastguard Worker Dso::SetBuildIds(
396*288bf522SAndroid Build Coastguard Worker {std::make_pair(file_path, BuildId("1b12a384a9f4a3f3659b7171ca615dbec3a81f71"))});
397*288bf522SAndroid Build Coastguard Worker auto dso = Dso::CreateDso(DSO_ELF_FILE, file_path);
398*288bf522SAndroid Build Coastguard Worker CapturedStderr capture;
399*288bf522SAndroid Build Coastguard Worker dso->LoadSymbols();
400*288bf522SAndroid Build Coastguard Worker ASSERT_NE(capture.str().find("failed to read symbols"), std::string::npos);
401*288bf522SAndroid Build Coastguard Worker }
402*288bf522SAndroid Build Coastguard Worker {
403*288bf522SAndroid Build Coastguard Worker // Don't warn when we already have symbols.
404*288bf522SAndroid Build Coastguard Worker std::string file_path = GetTestData("not_exist_file");
405*288bf522SAndroid Build Coastguard Worker Dso::SetBuildIds(
406*288bf522SAndroid Build Coastguard Worker {std::make_pair(file_path, BuildId("1b12a384a9f4a3f3659b7171ca615dbec3a81f71"))});
407*288bf522SAndroid Build Coastguard Worker auto dso = Dso::CreateDso(DSO_ELF_FILE, file_path);
408*288bf522SAndroid Build Coastguard Worker std::vector<Symbol> symbols;
409*288bf522SAndroid Build Coastguard Worker symbols.emplace_back("fake_symbol", 0x1234, 0x60);
410*288bf522SAndroid Build Coastguard Worker dso->SetSymbols(&symbols);
411*288bf522SAndroid Build Coastguard Worker CapturedStderr capture;
412*288bf522SAndroid Build Coastguard Worker dso->LoadSymbols();
413*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(capture.str().find("failed to read symbols"), std::string::npos);
414*288bf522SAndroid Build Coastguard Worker }
415*288bf522SAndroid Build Coastguard Worker }
416*288bf522SAndroid Build Coastguard Worker
417*288bf522SAndroid Build Coastguard Worker // @CddTest = 6.1/C-0-2
TEST(dso,demangle)418*288bf522SAndroid Build Coastguard Worker TEST(dso, demangle) {
419*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(Dso::Demangle("main"), "main");
420*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(Dso::Demangle("_ZN4main4main17h2a68d4d833d7495aE"), "main::main::h2a68d4d833d7495a");
421*288bf522SAndroid Build Coastguard Worker #if defined(__linux__) || defined(__darwin__)
422*288bf522SAndroid Build Coastguard Worker // Demangling rust symbol is only supported on linux and darwin.
423*288bf522SAndroid Build Coastguard Worker ASSERT_EQ(Dso::Demangle("_RNvC6_123foo3bar"), "123foo::bar");
424*288bf522SAndroid Build Coastguard Worker #endif
425*288bf522SAndroid Build Coastguard Worker }
426