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 <gtest/gtest.h>
18 
19 #include <android-base/file.h>
20 
21 #include "command.h"
22 #include "get_test_data.h"
23 
24 using namespace simpleperf;
25 
ReportSampleCmd()26 static std::unique_ptr<Command> ReportSampleCmd() {
27   return CreateCommandInstance("report-sample");
28 }
29 
TEST(cmd_report_sample,text)30 TEST(cmd_report_sample, text) {
31   ASSERT_TRUE(ReportSampleCmd()->Run({"-i", GetTestData(PERF_DATA_WITH_SYMBOLS)}));
32 }
33 
TEST(cmd_report_sample,output_option)34 TEST(cmd_report_sample, output_option) {
35   TemporaryFile tmpfile;
36   ASSERT_TRUE(
37       ReportSampleCmd()->Run({"-i", GetTestData(PERF_DATA_WITH_SYMBOLS), "-o", tmpfile.path}));
38 }
39 
TEST(cmd_report_sample,show_callchain_option)40 TEST(cmd_report_sample, show_callchain_option) {
41   TemporaryFile tmpfile;
42   ASSERT_TRUE(ReportSampleCmd()->Run(
43       {"-i", GetTestData(CALLGRAPH_FP_PERF_DATA), "-o", tmpfile.path, "--show-callchain"}));
44 }
45 
GetProtobufReport(const std::string & test_data_file,std::string * protobuf_report,const std::vector<std::string> & extra_args={})46 static void GetProtobufReport(const std::string& test_data_file, std::string* protobuf_report,
47                               const std::vector<std::string>& extra_args = {}) {
48   TemporaryFile tmpfile;
49   TemporaryFile tmpfile2;
50   std::vector<std::string> args = {"-i", GetTestData(test_data_file), "-o", tmpfile.path,
51                                    "--protobuf"};
52   args.insert(args.end(), extra_args.begin(), extra_args.end());
53   ASSERT_TRUE(ReportSampleCmd()->Run(args));
54   ASSERT_TRUE(
55       ReportSampleCmd()->Run({"--dump-protobuf-report", tmpfile.path, "-o", tmpfile2.path}));
56   ASSERT_TRUE(android::base::ReadFileToString(tmpfile2.path, protobuf_report));
57 }
58 
TEST(cmd_report_sample,protobuf_option)59 TEST(cmd_report_sample, protobuf_option) {
60   std::string data;
61   GetProtobufReport(PERF_DATA_WITH_SYMBOLS, &data);
62   ASSERT_NE(data.find("magic: SIMPLEPERF"), std::string::npos);
63   ASSERT_NE(data.find("version: 1"), std::string::npos);
64   ASSERT_NE(data.find("file:"), std::string::npos);
65 }
66 
TEST(cmd_report_sample,no_skipped_file_id)67 TEST(cmd_report_sample, no_skipped_file_id) {
68   std::string data;
69   GetProtobufReport(PERF_DATA_WITH_WRONG_IP_IN_CALLCHAIN, &data);
70   // If wrong ips in callchain are omitted, "unknown" file path will not be generated.
71   ASSERT_EQ(data.find("unknown"), std::string::npos);
72 }
73 
TEST(cmd_report_sample,sample_has_event_count)74 TEST(cmd_report_sample, sample_has_event_count) {
75   std::string data;
76   GetProtobufReport(PERF_DATA_WITH_SYMBOLS, &data);
77   ASSERT_NE(data.find("event_count:"), std::string::npos);
78 }
79 
TEST(cmd_report_sample,has_thread_record)80 TEST(cmd_report_sample, has_thread_record) {
81   std::string data;
82   GetProtobufReport(PERF_DATA_WITH_SYMBOLS, &data);
83   ASSERT_NE(data.find("thread:"), std::string::npos);
84   ASSERT_NE(data.find("thread_name: t2"), std::string::npos);
85 }
86 
TEST(cmd_report_sample,trace_offcpu)87 TEST(cmd_report_sample, trace_offcpu) {
88   std::string data;
89   GetProtobufReport(PERF_DATA_WITH_TRACE_OFFCPU, &data);
90   ASSERT_NE(data.find("event_type: sched:sched_switch"), std::string::npos);
91 }
92 
TEST(cmd_report_sample,have_clear_callchain_end_in_protobuf_output)93 TEST(cmd_report_sample, have_clear_callchain_end_in_protobuf_output) {
94   std::string data;
95   GetProtobufReport(PERF_DATA_WITH_TRACE_OFFCPU, &data, {"--show-callchain"});
96   ASSERT_NE(data.find("__libc_init"), std::string::npos);
97   ASSERT_EQ(data.find("_start_main"), std::string::npos);
98 }
99 
TEST(cmd_report_sample,app_package_name_in_meta_info)100 TEST(cmd_report_sample, app_package_name_in_meta_info) {
101   std::string data;
102   GetProtobufReport(PERF_DATA_WITH_APP_PACKAGE_NAME, &data);
103   ASSERT_NE(data.find("app_package_name: com.google.sample.tunnel"), std::string::npos);
104 }
105 
TEST(cmd_report_sample,remove_unknown_kernel_symbols)106 TEST(cmd_report_sample, remove_unknown_kernel_symbols) {
107   std::string data;
108   // Test --remove-unknown-kernel-symbols on perf.data with kernel_symbols_available=false.
109   GetProtobufReport(PERF_DATA_WITH_KERNEL_SYMBOLS_AVAILABLE_FALSE, &data, {"--show-callchain"});
110   ASSERT_NE(data.find("time: 1368182962424044"), std::string::npos);
111   ASSERT_NE(data.find("path: [kernel.kallsyms]"), std::string::npos);
112   ASSERT_NE(data.find("path: /system/lib64/libc.so"), std::string::npos);
113   GetProtobufReport(PERF_DATA_WITH_KERNEL_SYMBOLS_AVAILABLE_FALSE, &data,
114                     {"--show-callchain", "--remove-unknown-kernel-symbols"});
115   // The sample dumped at time 1368182962424044 shouldn't be removed. Because it has user space
116   // callchains.
117   ASSERT_NE(data.find("time: 1368182962424044"), std::string::npos);
118   // Kernel callchains shouldn't be removed.
119   ASSERT_EQ(data.find("path: [kernel.kallsyms]"), std::string::npos);
120   // User space callchains still exist.
121   ASSERT_NE(data.find("path: /system/lib64/libc.so"), std::string::npos);
122 
123   // Test --remove-unknown-kernel-symbols on perf.data with kernel_symbols_available=true.
124   GetProtobufReport(PERF_DATA_WITH_KERNEL_SYMBOLS_AVAILABLE_TRUE, &data, {"--show-callchain"});
125   ASSERT_NE(data.find("time: 1368297633794862"), std::string::npos);
126   ASSERT_NE(data.find("path: [kernel.kallsyms]"), std::string::npos);
127   ASSERT_NE(data.find("symbol: binder_ioctl_write_read"), std::string::npos);
128   ASSERT_NE(data.find("path: /system/lib64/libc.so"), std::string::npos);
129   GetProtobufReport(PERF_DATA_WITH_KERNEL_SYMBOLS_AVAILABLE_TRUE, &data,
130                     {"--show-callchain", "--remove-unknown-kernel-symbols"});
131   ASSERT_NE(data.find("time: 1368297633794862"), std::string::npos);
132   ASSERT_NE(data.find("path: [kernel.kallsyms]"), std::string::npos);
133   ASSERT_NE(data.find("symbol: binder_ioctl_write_read"), std::string::npos);
134   ASSERT_NE(data.find("path: /system/lib64/libc.so"), std::string::npos);
135 }
136 
TEST(cmd_report_sample,show_art_frames_option)137 TEST(cmd_report_sample, show_art_frames_option) {
138   std::string data;
139   GetProtobufReport(PERF_DATA_WITH_INTERPRETER_FRAMES, &data, {"--show-callchain"});
140   ASSERT_EQ(data.find("artMterpAsmInstructionStart"), std::string::npos);
141   GetProtobufReport(PERF_DATA_WITH_INTERPRETER_FRAMES, &data,
142                     {"--show-callchain", "--show-art-frames"});
143   ASSERT_NE(data.find("artMterpAsmInstructionStart"), std::string::npos);
144 }
145 
TEST(cmd_report_sample,show_execution_type_option)146 TEST(cmd_report_sample, show_execution_type_option) {
147   std::string data;
148   GetProtobufReport("perf_display_bitmaps.data", &data,
149                     {"--show-callchain", "--show-execution-type"});
150   ASSERT_NE(data.find("execution_type: interpreted_jvm_method"), std::string::npos);
151   // We convert JIT frames to map to dex files. So there is no file named jit_app_cache in the
152   // report. But the execution type of a JIT frame isn't changed.
153   ASSERT_EQ(data.find("jit_app_cache"), std::string::npos);
154   ASSERT_NE(data.find("execution_type: jit_jvm_method"), std::string::npos);
155   // art_method is shown only when --show-art-frames is used.
156   ASSERT_EQ(data.find("execution_type: art_method"), std::string::npos);
157 
158   GetProtobufReport("perf_display_bitmaps.data", &data,
159                     {"--show-callchain", "--show-execution-type", "--show-art-frames"});
160   ASSERT_NE(data.find("execution_type: art_method"), std::string::npos);
161 }
162 
TEST(cmd_report_sample,show_symbols_before_and_after_demangle)163 TEST(cmd_report_sample, show_symbols_before_and_after_demangle) {
164   std::string data;
165   GetProtobufReport(PERF_DATA_WITH_INTERPRETER_FRAMES, &data, {"--show-callchain"});
166   ASSERT_NE(data.find("symbol: android::hardware::IPCThreadState::talkWithDriver(bool)"),
167             std::string::npos);
168   ASSERT_NE(data.find("mangled_symbol: _ZN7android8hardware14IPCThreadState14talkWithDriverEb"),
169             std::string::npos);
170 }
171 
TEST(cmd_report_sample,symdir_option)172 TEST(cmd_report_sample, symdir_option) {
173   std::string data;
174   GetProtobufReport(PERF_DATA_FOR_BUILD_ID_CHECK, &data);
175   ASSERT_EQ(data.find("symbol: main"), std::string::npos);
176   GetProtobufReport(PERF_DATA_FOR_BUILD_ID_CHECK, &data,
177                     {"--symdir", GetTestDataDir() + CORRECT_SYMFS_FOR_BUILD_ID_CHECK});
178   ASSERT_NE(data.find("symbol: main"), std::string::npos);
179 }
180 
TEST(cmd_report_sample,show_art_jni_methods)181 TEST(cmd_report_sample, show_art_jni_methods) {
182   std::string data;
183   GetProtobufReport("perf_display_bitmaps.data", &data, {"--show-callchain"});
184   ASSERT_NE(data.find("art::Method_invoke"), std::string::npos);
185   // Don't show art_jni_trampoline.
186   ASSERT_EQ(data.find("art_jni_trampoline"), std::string::npos);
187 }
188 
TEST(cmd_report_sample,show_unwinding_result)189 TEST(cmd_report_sample, show_unwinding_result) {
190   std::string data;
191   GetProtobufReport("perf_with_failed_unwinding_debug_info.data", &data, {"--show-callchain"});
192   ASSERT_NE(data.find("error_code: ERROR_INVALID_MAP"), std::string::npos);
193 }
194 
TEST(cmd_report_sample,proguard_mapping_file_option)195 TEST(cmd_report_sample, proguard_mapping_file_option) {
196   std::string data;
197   // Symbols aren't de-obfuscated without proguard mapping file.
198   GetProtobufReport("perf_need_proguard_mapping.data", &data, {"--show-callchain"});
199   ASSERT_EQ(data.find("androidx.fragment.app.FragmentActivity.startActivityForResult"),
200             std::string::npos);
201   ASSERT_EQ(data.find("com.example.android.displayingbitmaps.ui.ImageGridFragment.onItemClick"),
202             std::string::npos);
203   // Symbols are de-obfuscated with proguard mapping file.
204   GetProtobufReport(
205       "perf_need_proguard_mapping.data", &data,
206       {"--show-callchain", "--proguard-mapping-file", GetTestData("proguard_mapping.txt")});
207   ASSERT_NE(data.find("androidx.fragment.app.FragmentActivity.startActivityForResult"),
208             std::string::npos);
209   ASSERT_NE(data.find("com.example.android.displayingbitmaps.ui.ImageGridFragment.onItemClick"),
210             std::string::npos);
211 }
212