• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) Huawei Technologies Co., Ltd. 2022. All rights reserved.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include <chrono>
17 #include <cerrno>
18 #include <csignal>
19 #include <cstdlib>
20 #include <cstdio>
21 #include <cstring>
22 #include <iostream>
23 
24 #include <fcntl.h>
25 #include <unistd.h>
26 #include <linux/perf_event.h>
27 #include <linux/bpf.h>
28 #include <sys/resource.h>
29 
30 #include "bpf_controller.h"
31 #include "commandline_flags.h"
32 #include "hhlog.h"
33 #include "command_helper.h"
34 #include "ipc_unix_socket.h"
35 
36 HIEBPF_DEFINE_string(help, "none", "specify what help message to print");
37 HIEBPF_DEFINE_string(output_file, "/data/local/tmp/hiebpf.data", "the file used to save hiebpf data");
38 HIEBPF_DEFINE_bool(exclude_tracer, true, "the flag indicates whether to trace the tracer itself or not");
39 HIEBPF_DEFINE_uint32(dump_events, 0, "maximum number of events to print instead of save into file");
40 HIEBPF_DEFINE_bool(unwind_stack, true, "whether enable unwinding or not");
41 HIEBPF_DEFINE_uint32(max_stack_depth, MAX_STACK_DEPTH, "max unwinding stack depth");
42 HIEBPF_DEFINE_uint32(duration, 0, "the tracer will run for <duration> seconds");
43 HIEBPF_DEFINE_array(events, string, "events to trace");
44 HIEBPF_DEFINE_array(pids, uint32_t, "pids of target process to trace");
45 HIEBPF_DEFINE_string(bpf_log_level, "NONE", "lowest level of BPF programs logs to print");
46 HIEBPF_DEFINE_string(bpf_log_file, "/data/local/tmp/bpf.log", "the file used to save BPF program logs");
47 HIEBPF_DEFINE_string(libbpf_log_level, "NONE", "lowest level of libbpf internal error messages to print");
48 HIEBPF_DEFINE_string(libbpf_log_file, "/data/local/tmp/libbpf.log",
49                     "the file used to save libbpf internal error messages");
50 HIEBPF_DEFINE_string(hhlog_level, "NONE", "lowest level of HHLog to print");
51 HIEBPF_DEFINE_string(hhlog_file, "/data/local/tmp/hhlog.txt", "the file used to save HHLog");
52 HIEBPF_DEFINE_bool(start, false, "setup and start hiebpf server to trace");
53 HIEBPF_DEFINE_bool(stop, false, "stop hiebpf server to trace");
54 
55 static BPFConfig GenBPFConfigFromFlags(const std::string &cmd);
56 static int GetHHLogLevelFromFlag();
57 static inline std::vector<pid_t> GetTargetPidsFromFlag();
58 static std::unique_ptr<BPFController> bpfController;
59 static OHOS::Developtools::Hiebpf::IpcUnixSocketServer g_ipcServer;
60 static bool g_interrupted = false;
61 
62 const uint32_t IPC_RECV_TIMEOUT = 1000;
63 
64 enum HiebpfIpcCommand:uint32_t {
65     CMD_START = 0,
66     CMD_STOP,
67     CMD_HEART_BEAT,
68     RET_OK = 0x10000,
69 };
70 
HandleIpcMessage(const void * data,size_t size)71 static void HandleIpcMessage(const void *data, size_t size)
72 {
73     if (size != sizeof(HiebpfIpcCommand)) {
74         HHLOGE(true, "unknown command");
75         return;
76     }
77 
78     HiebpfIpcCommand cmd = *(const_cast<HiebpfIpcCommand*>(reinterpret_cast<const HiebpfIpcCommand*>(data)));
79     if (cmd == HiebpfIpcCommand::CMD_STOP) {
80         g_interrupted = true;
81         if (bpfController != nullptr) {
82             bpfController->Stop();
83         }
84         cmd = HiebpfIpcCommand::RET_OK;
85         if (!g_ipcServer.SendMessage(&cmd, sizeof(HiebpfIpcCommand))) {
86             HHLOGE(true, "send reply(0x%x) failed", cmd);
87         }
88     } else if (cmd == HiebpfIpcCommand::CMD_HEART_BEAT) {
89         cmd = HiebpfIpcCommand::RET_OK;
90         if (!g_ipcServer.SendMessage(&cmd, sizeof(HiebpfIpcCommand))) {
91             HHLOGE(true, "send reply(0x%x) failed\n", cmd);
92         }
93     } else {
94         HHLOGW(true, "the command(0x%x) don't support\n", cmd);
95     }
96 }
97 
SendIpcCommand(OHOS::Developtools::Hiebpf::IpcUnixSocketClient & client,HiebpfIpcCommand cmd,uint32_t timeout)98 static bool SendIpcCommand(OHOS::Developtools::Hiebpf::IpcUnixSocketClient &client,
99                            HiebpfIpcCommand cmd, uint32_t timeout)
100 {
101     CHECK_TRUE(client.SendMessage(&cmd, sizeof(HiebpfIpcCommand)), false, "send request(0x%x) failed\n", cmd);
102 
103     size_t size = OHOS::Developtools::Hiebpf::UNIX_SOCKET_BUFFER_SIZE;
104     char buf[OHOS::Developtools::Hiebpf::UNIX_SOCKET_BUFFER_SIZE] = {0};
105     CHECK_TRUE(client.RecvMessage(buf, size, timeout), false, "recv reply failed\n");
106     if (size == 0) { // timeout
107         HHLOGE(true, "recv reply timeout(%dms)\n", timeout);
108         return false;
109     } else if (size != sizeof(HiebpfIpcCommand)) {
110         HHLOGE(true, "recv reply size error(%zu)\n", size);
111         return false;
112     }
113     HiebpfIpcCommand reply = *(reinterpret_cast<HiebpfIpcCommand*>(buf));
114     CHECK_TRUE(reply == HiebpfIpcCommand::RET_OK, false, "recv unknown reply: 0x%x\n", reply);
115     return true;
116 }
117 
StopHiebpfServer()118 static void StopHiebpfServer()
119 {
120     OHOS::Developtools::Hiebpf::IpcUnixSocketClient ipcClient;
121     if (!ipcClient.Connect()) {
122         printf("ipc client connect failed\n");
123         return;
124     }
125 
126     if (SendIpcCommand(ipcClient, HiebpfIpcCommand::CMD_STOP, IPC_RECV_TIMEOUT)) {
127         printf("send STOP command to hiebpf server sucess");
128         fflush(stdout);
129         while (SendIpcCommand(ipcClient, HiebpfIpcCommand::CMD_HEART_BEAT, IPC_RECV_TIMEOUT)) {
130             sleep(1);
131             printf(".");
132             fflush(stdout);
133         }
134         printf("\n");
135     } else {
136         printf("send STOP command to hiebpf server failed\n");
137     }
138 
139     ipcClient.Disconnect();
140 }
141 
CheckHiebpfServer()142 static bool CheckHiebpfServer()
143 {
144     sleep(1); // wait hiebpf IPC server started
145     OHOS::Developtools::Hiebpf::IpcUnixSocketClient ipcClient;
146     if (!ipcClient.Connect()) {
147         printf("ipc client connect failed\n");
148         return false;
149     }
150 
151     bool ret = false;
152     if (SendIpcCommand(ipcClient, HiebpfIpcCommand::CMD_HEART_BEAT, IPC_RECV_TIMEOUT)) {
153         ret = true;
154     }
155 
156     ipcClient.Disconnect();
157     return ret;
158 }
159 
main(int argc,char * argv[])160 int main(int argc, char* argv[])
161 {
162     std::string cmd;
163     for (int32_t i = 0; i < argc; ++i) {
164         cmd += std::string(argv[i]) + " ";
165     }
166     // parse all defined command line arguments
167     if (CommandLineFlags::ParseFlags(&argc, &argv, true) != 0) {
168         std::cout << "[ERROR] " << __FILE__ << ":"
169             << __FUNCTION__ << ":"
170             << __LINE__ << ": "
171             << "failed to parse command line flags" << std::endl;
172         return -1;
173     }
174 
175     // check if user issued help, none-zero return value means user does issue
176     // help, hence we exit right after printing the required help message.
177     if (CommandHelper::GetInstance().Start(&argc, &argv, FLAGS_help) != 0) {
178         // Note this is the workflow when user asking for command help, it's not
179         // an error. Besides, help workflow never fails.
180         return 0;
181     }
182 
183     // start HHLogger
184     if (HHLogger::GetInstance().Start(GetHHLogLevelFromFlag(), FLAGS_hhlog_file) != 0) {
185         std::cout << "[ERROR] " << __FILE__ << ":"
186             << __FUNCTION__ << ":"
187             << __LINE__ << ": "
188             << "failed to start HHLogger" << std::endl;
189         return -1;
190     }
191 
192     if (FLAGS_stop) {
193         StopHiebpfServer();
194         return 0;
195     }
196 
197     // fork() must be precede BPFController::MakeUnique(),
198     // otherwise ring_buffer__poll() will crash. why?
199     if (FLAGS_start) {
200         pid_t pid = fork();
201         if (pid == -1) {
202             printf("fork failed. %d:%s\n", errno, strerror(errno));
203             return -1;
204         } else if (pid == 0) { // child process
205             int32_t nullFd = open("/dev/null", O_WRONLY);
206             (void)dup2(nullFd, STDOUT_FILENO); // redirect stdout to /dev/null
207             g_ipcServer.SetHandleCallback(HandleIpcMessage);
208             if (!g_ipcServer.Start()) {
209                 return 0;
210             }
211         } else { // parent process
212             if (CheckHiebpfServer()) {
213                 printf("hiebpf server has started.\n");
214             } else {
215                 printf("hiebpf server start failed\n");
216             }
217             return 0;
218         }
219     }
220     HHLogger::GetInstance().LogSaver();
221     // start BPFController
222     bpfController = BPFController::MakeUnique(GenBPFConfigFromFlags(cmd));
223     if (bpfController == nullptr) {
224         std::cout << "[ERROR] " << __FILE__ << ":"
225             << __FUNCTION__ << ":"
226             << __LINE__ << ": "
227             << "failed to init BPFController" << std::endl;
228         return -1;
229     }
230 
231     if (g_interrupted) {
232         return 0;
233     }
234 
235     if (bpfController->Start() != 0) {
236         std::cout << "[ERROR] " << __FILE__ << ":"
237             << __FUNCTION__ << ":"
238             << __LINE__ << ": "
239             << "failed to start BPFEvent loop" << std::endl;
240         return -1;
241     }
242     bpfController = nullptr;
243 
244     if (FLAGS_start) {
245         g_ipcServer.Stop();
246     }
247 
248     return 0;
249 }
250 
GetTargetPidsFromFlag()251 static inline std::vector<pid_t> GetTargetPidsFromFlag()
252 {
253     std::vector<pid_t> pids {};
254     for (size_t k = 0; k < FLAGS_pids.size(); ++ k) {
255         pids.push_back(static_cast<pid_t>(FLAGS_pids[k]));
256     }
257     return pids;
258 }
259 
GetTargetEventsFromFlag()260 static inline std::set<HiebpfEventGroup> GetTargetEventsFromFlag()
261 {
262     const static std::unordered_map<std::string, HiebpfEventGroup> eventGroupMap {
263         {"fs", FS_GROUP_ALL},
264         {"fs:open", FS_GROUP_OPEN},
265         {"fs:read", FS_GROUP_READ},
266         {"fs:write", FS_GROUP_WRITE},
267         {"fs:close", FS_GROUP_CLOSE},
268         {"ptrace", MEM_GROUP_ALL},
269         {"bio", BIO_GROUP_ALL},
270     };
271 
272     std::set<HiebpfEventGroup> result {};
273     for (size_t k = 0; k < FLAGS_events.size(); ++k) {
274         const auto& it = eventGroupMap.find(FLAGS_events[k]);
275         if (it == eventGroupMap.end()) {
276             // cmdline input error, how notify cmdline?
277         } else {
278             result.insert(it->second);
279         }
280     }
281     return result;
282 }
283 
GetBPFLogLevelFromFlag()284 static inline __u32 GetBPFLogLevelFromFlag()
285 {
286     if (FLAGS_bpf_log_level.compare("DEBUG") == 0 or
287         FLAGS_bpf_log_level.compare("debug") == 0) {
288         return BPF_LOG_DEBUG;
289     }
290     if (FLAGS_bpf_log_level.compare("INFO") == 0 or
291         FLAGS_bpf_log_level.compare("info") == 0) {
292         return BPF_LOG_INFO;
293     }
294     if (FLAGS_bpf_log_level.compare("WARN") == 0 or
295         FLAGS_bpf_log_level.compare("warn") == 0) {
296         return BPF_LOG_WARN;
297     }
298     if (FLAGS_bpf_log_level.compare("ERROR") == 0 or
299         FLAGS_bpf_log_level.compare("error") == 0) {
300         return BPF_LOG_ERROR;
301     }
302     if (FLAGS_bpf_log_level.compare("FATAL") == 0 or
303         FLAGS_bpf_log_level.compare("fatal") == 0) {
304         return BPF_LOG_FATAL;
305     }
306     return BPF_LOG_NONE;
307 }
308 
GetLIBBPFLogLevelFromFlag()309 static inline int GetLIBBPFLogLevelFromFlag()
310 {
311     if (FLAGS_libbpf_log_level.compare("DEBUG") == 0 or
312         FLAGS_libbpf_log_level.compare("debug") == 0) {
313         return LIBBPF_DEBUG;
314     }
315     if (FLAGS_libbpf_log_level.compare("INFO") == 0 or
316         FLAGS_libbpf_log_level.compare("info") == 0) {
317         return LIBBPF_INFO;
318     }
319     if (FLAGS_libbpf_log_level.compare("WARN") == 0 or
320         FLAGS_libbpf_log_level.compare("warn") == 0) {
321         return LIBBPF_WARN;
322     }
323     if (FLAGS_libbpf_log_level.compare("ERROR") == 0 or
324         FLAGS_libbpf_log_level.compare("error") == 0) {
325         return LIBBPF_ERROR;
326     }
327     if (FLAGS_libbpf_log_level.compare("FATAL") == 0 or
328         FLAGS_libbpf_log_level.compare("fatal") == 0) {
329         return LIBBPF_FATAL;
330     }
331     return LIBBPF_NONE;
332 }
333 
GetHHLogLevelFromFlag()334 static int GetHHLogLevelFromFlag()
335 {
336     if (FLAGS_hhlog_level.compare("DEBUG") == 0 or
337         FLAGS_hhlog_level.compare("debug") == 0) {
338         return HHLOG_DEBUG;
339     }
340     if (FLAGS_hhlog_level.compare("INFO") == 0 or
341         FLAGS_hhlog_level.compare("info") == 0) {
342         return HHLOG_INFO;
343     }
344     if (FLAGS_hhlog_level.compare("WARN") == 0 or
345         FLAGS_hhlog_level.compare("warn") == 0) {
346         return HHLOG_WARN;
347     }
348     if (FLAGS_hhlog_level.compare("ERROR") == 0 or
349         FLAGS_hhlog_level.compare("error") == 0) {
350         return HHLOG_ERROR;
351     }
352     if (FLAGS_hhlog_level.compare("FATAL") == 0 or
353         FLAGS_hhlog_level.compare("fatal") == 0) {
354         return HHLOG_FATAL;
355     }
356     if (FLAGS_hhlog_level.compare("NONE") == 0 or
357         FLAGS_hhlog_level.compare("none") == 0) {
358         return HHLOG_NONE;
359     }
360     return HHLOG_NONE;
361 }
362 
GenBPFConfigFromFlags(const std::string & cmd)363 static inline BPFConfig GenBPFConfigFromFlags(const std::string &cmd)
364 {
365     BPFConfig config {};
366     config.cmd_ = cmd;
367     config.excludeTracer_ = FLAGS_exclude_tracer;
368     config.unwindStack_ = FLAGS_max_stack_depth == 0 ? false : FLAGS_unwind_stack;
369     config.dumpEvents_ = FLAGS_dump_events;
370     config.traceDuration_ = FLAGS_duration;
371     config.maxStackDepth_ = FLAGS_max_stack_depth == 0 ? 1 : FLAGS_max_stack_depth;
372     config.targetPids_ = GetTargetPidsFromFlag();
373     config.selectEventGroups_ = GetTargetEventsFromFlag();
374     config.BPFLogLevel_ = GetBPFLogLevelFromFlag();
375     config.LIBBPFLogLevel_ = GetLIBBPFLogLevelFromFlag();
376     config.BPFLogFile_ = FLAGS_bpf_log_file;
377     config.LIBBPFLogFile_ = FLAGS_libbpf_log_file;
378     config.outputFile_ = FLAGS_output_file;
379     return config;
380 }