xref: /aosp_15_r20/external/bcc/src/cc/bcc_proc.c (revision 387f9dfdfa2baef462e92476d413c7bc2470293e)
1*387f9dfdSAndroid Build Coastguard Worker /*
2*387f9dfdSAndroid Build Coastguard Worker  * Copyright (c) 2016 GitHub, Inc.
3*387f9dfdSAndroid Build Coastguard Worker  *
4*387f9dfdSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*387f9dfdSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*387f9dfdSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*387f9dfdSAndroid Build Coastguard Worker  *
8*387f9dfdSAndroid Build Coastguard Worker  * http://www.apache.org/licenses/LICENSE-2.0
9*387f9dfdSAndroid Build Coastguard Worker  *
10*387f9dfdSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*387f9dfdSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*387f9dfdSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*387f9dfdSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*387f9dfdSAndroid Build Coastguard Worker  * limitations under the License.
15*387f9dfdSAndroid Build Coastguard Worker  */
16*387f9dfdSAndroid Build Coastguard Worker 
17*387f9dfdSAndroid Build Coastguard Worker #include "bcc_proc.h"
18*387f9dfdSAndroid Build Coastguard Worker 
19*387f9dfdSAndroid Build Coastguard Worker #include <ctype.h>
20*387f9dfdSAndroid Build Coastguard Worker #include <dirent.h>
21*387f9dfdSAndroid Build Coastguard Worker #include <fcntl.h>
22*387f9dfdSAndroid Build Coastguard Worker #include <limits.h>
23*387f9dfdSAndroid Build Coastguard Worker #include <math.h>
24*387f9dfdSAndroid Build Coastguard Worker #include <stdbool.h>
25*387f9dfdSAndroid Build Coastguard Worker #include <stdint.h>
26*387f9dfdSAndroid Build Coastguard Worker #include <stdio.h>
27*387f9dfdSAndroid Build Coastguard Worker #include <stdlib.h>
28*387f9dfdSAndroid Build Coastguard Worker #include <string.h>
29*387f9dfdSAndroid Build Coastguard Worker #include <sys/mman.h>
30*387f9dfdSAndroid Build Coastguard Worker #include <sys/stat.h>
31*387f9dfdSAndroid Build Coastguard Worker #include <sys/types.h>
32*387f9dfdSAndroid Build Coastguard Worker #include <unistd.h>
33*387f9dfdSAndroid Build Coastguard Worker 
34*387f9dfdSAndroid Build Coastguard Worker #include "bcc_elf.h"
35*387f9dfdSAndroid Build Coastguard Worker #include "bcc_perf_map.h"
36*387f9dfdSAndroid Build Coastguard Worker #include "bcc_zip.h"
37*387f9dfdSAndroid Build Coastguard Worker 
38*387f9dfdSAndroid Build Coastguard Worker #ifdef __x86_64__
39*387f9dfdSAndroid Build Coastguard Worker // https://www.kernel.org/doc/Documentation/x86/x86_64/mm.txt
40*387f9dfdSAndroid Build Coastguard Worker const unsigned long long kernelAddrSpace = 0x00ffffffffffffff;
41*387f9dfdSAndroid Build Coastguard Worker #else
42*387f9dfdSAndroid Build Coastguard Worker const unsigned long long kernelAddrSpace = 0x0;
43*387f9dfdSAndroid Build Coastguard Worker #endif
44*387f9dfdSAndroid Build Coastguard Worker 
bcc_procutils_which(const char * binpath)45*387f9dfdSAndroid Build Coastguard Worker char *bcc_procutils_which(const char *binpath) {
46*387f9dfdSAndroid Build Coastguard Worker   char buffer[4096];
47*387f9dfdSAndroid Build Coastguard Worker   const char *PATH;
48*387f9dfdSAndroid Build Coastguard Worker 
49*387f9dfdSAndroid Build Coastguard Worker   if (strchr(binpath, '/'))
50*387f9dfdSAndroid Build Coastguard Worker     return bcc_elf_is_exe(binpath) ? strdup(binpath) : 0;
51*387f9dfdSAndroid Build Coastguard Worker 
52*387f9dfdSAndroid Build Coastguard Worker   if (!(PATH = getenv("PATH")))
53*387f9dfdSAndroid Build Coastguard Worker     return 0;
54*387f9dfdSAndroid Build Coastguard Worker 
55*387f9dfdSAndroid Build Coastguard Worker   while (PATH) {
56*387f9dfdSAndroid Build Coastguard Worker     const char *next = strchr(PATH, ':') ?: strchr(PATH, '\0');
57*387f9dfdSAndroid Build Coastguard Worker     const size_t path_len = next - PATH;
58*387f9dfdSAndroid Build Coastguard Worker 
59*387f9dfdSAndroid Build Coastguard Worker     if (path_len) {
60*387f9dfdSAndroid Build Coastguard Worker       int ret = snprintf(buffer, sizeof(buffer), "%.*s/%s",
61*387f9dfdSAndroid Build Coastguard Worker 	                  (int)path_len, PATH, binpath);
62*387f9dfdSAndroid Build Coastguard Worker       if (ret < 0 || ret >= sizeof(buffer))
63*387f9dfdSAndroid Build Coastguard Worker         return 0;
64*387f9dfdSAndroid Build Coastguard Worker 
65*387f9dfdSAndroid Build Coastguard Worker       if (bcc_elf_is_exe(buffer))
66*387f9dfdSAndroid Build Coastguard Worker         return strdup(buffer);
67*387f9dfdSAndroid Build Coastguard Worker     }
68*387f9dfdSAndroid Build Coastguard Worker 
69*387f9dfdSAndroid Build Coastguard Worker     PATH = *next ? (next + 1) : 0;
70*387f9dfdSAndroid Build Coastguard Worker   }
71*387f9dfdSAndroid Build Coastguard Worker 
72*387f9dfdSAndroid Build Coastguard Worker   return 0;
73*387f9dfdSAndroid Build Coastguard Worker }
74*387f9dfdSAndroid Build Coastguard Worker 
75*387f9dfdSAndroid Build Coastguard Worker #define STARTS_WITH(mapname, prefix) (!strncmp(mapname, prefix, sizeof(prefix)-1))
76*387f9dfdSAndroid Build Coastguard Worker 
bcc_mapping_is_file_backed(const char * mapname)77*387f9dfdSAndroid Build Coastguard Worker int bcc_mapping_is_file_backed(const char *mapname) {
78*387f9dfdSAndroid Build Coastguard Worker   return mapname[0] && !(
79*387f9dfdSAndroid Build Coastguard Worker     STARTS_WITH(mapname, "//anon") ||
80*387f9dfdSAndroid Build Coastguard Worker     STARTS_WITH(mapname, "/dev/zero") ||
81*387f9dfdSAndroid Build Coastguard Worker     STARTS_WITH(mapname, "/anon_hugepage") ||
82*387f9dfdSAndroid Build Coastguard Worker     STARTS_WITH(mapname, "[stack") ||
83*387f9dfdSAndroid Build Coastguard Worker     STARTS_WITH(mapname, "/SYSV") ||
84*387f9dfdSAndroid Build Coastguard Worker     STARTS_WITH(mapname, "[heap]") ||
85*387f9dfdSAndroid Build Coastguard Worker     STARTS_WITH(mapname, "[vsyscall]"));
86*387f9dfdSAndroid Build Coastguard Worker }
87*387f9dfdSAndroid Build Coastguard Worker 
88*387f9dfdSAndroid Build Coastguard Worker /*
89*387f9dfdSAndroid Build Coastguard Worker Finds a file descriptor for a given inode if it's a memory-backed fd.
90*387f9dfdSAndroid Build Coastguard Worker */
_procutils_memfd_path(const int pid,const uint64_t inum)91*387f9dfdSAndroid Build Coastguard Worker static char *_procutils_memfd_path(const int pid, const uint64_t inum) {
92*387f9dfdSAndroid Build Coastguard Worker   char path_buffer[PATH_MAX + 1];
93*387f9dfdSAndroid Build Coastguard Worker   char *path = NULL;
94*387f9dfdSAndroid Build Coastguard Worker   char *dirstr;
95*387f9dfdSAndroid Build Coastguard Worker   DIR *dirstream;
96*387f9dfdSAndroid Build Coastguard Worker   struct stat sb;
97*387f9dfdSAndroid Build Coastguard Worker   struct dirent *dent;
98*387f9dfdSAndroid Build Coastguard Worker 
99*387f9dfdSAndroid Build Coastguard Worker   snprintf(path_buffer, (PATH_MAX + 1), "/proc/%d/fd", pid);
100*387f9dfdSAndroid Build Coastguard Worker   dirstr = malloc(strlen(path_buffer) + 1);
101*387f9dfdSAndroid Build Coastguard Worker   strcpy(dirstr, path_buffer);
102*387f9dfdSAndroid Build Coastguard Worker   dirstream = opendir(dirstr);
103*387f9dfdSAndroid Build Coastguard Worker 
104*387f9dfdSAndroid Build Coastguard Worker   if (dirstream == NULL) {
105*387f9dfdSAndroid Build Coastguard Worker     free(dirstr);
106*387f9dfdSAndroid Build Coastguard Worker     return NULL;
107*387f9dfdSAndroid Build Coastguard Worker   }
108*387f9dfdSAndroid Build Coastguard Worker 
109*387f9dfdSAndroid Build Coastguard Worker   while (path == NULL && (dent = readdir(dirstream)) != NULL) {
110*387f9dfdSAndroid Build Coastguard Worker     snprintf(path_buffer, (PATH_MAX + 1), "%s/%s", dirstr, dent->d_name);
111*387f9dfdSAndroid Build Coastguard Worker     if (stat(path_buffer, &sb) == -1)
112*387f9dfdSAndroid Build Coastguard Worker       continue;
113*387f9dfdSAndroid Build Coastguard Worker 
114*387f9dfdSAndroid Build Coastguard Worker     if (sb.st_ino == inum) {
115*387f9dfdSAndroid Build Coastguard Worker       char *pid_fd_path = malloc(strlen(path_buffer) + 1);
116*387f9dfdSAndroid Build Coastguard Worker       strcpy(pid_fd_path, path_buffer);
117*387f9dfdSAndroid Build Coastguard Worker       path = pid_fd_path;
118*387f9dfdSAndroid Build Coastguard Worker     }
119*387f9dfdSAndroid Build Coastguard Worker   }
120*387f9dfdSAndroid Build Coastguard Worker   closedir(dirstream);
121*387f9dfdSAndroid Build Coastguard Worker   free(dirstr);
122*387f9dfdSAndroid Build Coastguard Worker 
123*387f9dfdSAndroid Build Coastguard Worker   return path;
124*387f9dfdSAndroid Build Coastguard Worker }
125*387f9dfdSAndroid Build Coastguard Worker 
_procfs_might_be_zip_path(const char * path)126*387f9dfdSAndroid Build Coastguard Worker static int _procfs_might_be_zip_path(const char *path) {
127*387f9dfdSAndroid Build Coastguard Worker   return strstr(path, ".zip") || strstr(path, ".apk");
128*387f9dfdSAndroid Build Coastguard Worker }
129*387f9dfdSAndroid Build Coastguard Worker 
_procfs_find_zip_entry(const char * path,int pid,uint32_t * offset)130*387f9dfdSAndroid Build Coastguard Worker static char *_procfs_find_zip_entry(const char *path, int pid,
131*387f9dfdSAndroid Build Coastguard Worker                                     uint32_t *offset) {
132*387f9dfdSAndroid Build Coastguard Worker   char ns_relative_path[PATH_MAX];
133*387f9dfdSAndroid Build Coastguard Worker   int rc = snprintf(ns_relative_path, sizeof(ns_relative_path),
134*387f9dfdSAndroid Build Coastguard Worker                     "/proc/%d/root%s", pid, path);
135*387f9dfdSAndroid Build Coastguard Worker   if (rc < 0 || rc >= sizeof(ns_relative_path)) {
136*387f9dfdSAndroid Build Coastguard Worker     return NULL;
137*387f9dfdSAndroid Build Coastguard Worker   }
138*387f9dfdSAndroid Build Coastguard Worker 
139*387f9dfdSAndroid Build Coastguard Worker   struct bcc_zip_archive *archive = bcc_zip_archive_open(ns_relative_path);
140*387f9dfdSAndroid Build Coastguard Worker   if (archive == NULL) {
141*387f9dfdSAndroid Build Coastguard Worker     return NULL;
142*387f9dfdSAndroid Build Coastguard Worker   }
143*387f9dfdSAndroid Build Coastguard Worker 
144*387f9dfdSAndroid Build Coastguard Worker   struct bcc_zip_entry entry;
145*387f9dfdSAndroid Build Coastguard Worker   if (bcc_zip_archive_find_entry_at_offset(archive, *offset, &entry) ||
146*387f9dfdSAndroid Build Coastguard Worker       entry.compression) {
147*387f9dfdSAndroid Build Coastguard Worker     bcc_zip_archive_close(archive);
148*387f9dfdSAndroid Build Coastguard Worker     return NULL;
149*387f9dfdSAndroid Build Coastguard Worker   }
150*387f9dfdSAndroid Build Coastguard Worker 
151*387f9dfdSAndroid Build Coastguard Worker   char *result = malloc(strlen(path) + entry.name_length + 3);
152*387f9dfdSAndroid Build Coastguard Worker   if (result == NULL) {
153*387f9dfdSAndroid Build Coastguard Worker     bcc_zip_archive_close(archive);
154*387f9dfdSAndroid Build Coastguard Worker     return NULL;
155*387f9dfdSAndroid Build Coastguard Worker   }
156*387f9dfdSAndroid Build Coastguard Worker 
157*387f9dfdSAndroid Build Coastguard Worker   sprintf(result, "%s!/%.*s", path, entry.name_length, entry.name);
158*387f9dfdSAndroid Build Coastguard Worker   *offset -= entry.data_offset;
159*387f9dfdSAndroid Build Coastguard Worker   bcc_zip_archive_close(archive);
160*387f9dfdSAndroid Build Coastguard Worker   return result;
161*387f9dfdSAndroid Build Coastguard Worker }
162*387f9dfdSAndroid Build Coastguard Worker 
163*387f9dfdSAndroid Build Coastguard Worker // return: 0 -> callback returned < 0, stopped iterating
164*387f9dfdSAndroid Build Coastguard Worker //        -1 -> callback never indicated to stop
_procfs_maps_each_module(FILE * procmap,int pid,bcc_procutils_modulecb callback,void * payload)165*387f9dfdSAndroid Build Coastguard Worker int _procfs_maps_each_module(FILE *procmap, int pid,
166*387f9dfdSAndroid Build Coastguard Worker                              bcc_procutils_modulecb callback, void *payload) {
167*387f9dfdSAndroid Build Coastguard Worker   char buf[PATH_MAX + 1], perm[5];
168*387f9dfdSAndroid Build Coastguard Worker   char *name, *resolved_name;
169*387f9dfdSAndroid Build Coastguard Worker   mod_info mod;
170*387f9dfdSAndroid Build Coastguard Worker   uint8_t enter_ns;
171*387f9dfdSAndroid Build Coastguard Worker   while (true) {
172*387f9dfdSAndroid Build Coastguard Worker     enter_ns = 1;
173*387f9dfdSAndroid Build Coastguard Worker     buf[0] = '\0';
174*387f9dfdSAndroid Build Coastguard Worker     // From fs/proc/task_mmu.c:show_map_vma
175*387f9dfdSAndroid Build Coastguard Worker     if (fscanf(procmap, "%lx-%lx %4s %llx %lx:%lx %lu%[^\n]",
176*387f9dfdSAndroid Build Coastguard Worker           &mod.start_addr, &mod.end_addr, perm, &mod.file_offset,
177*387f9dfdSAndroid Build Coastguard Worker           &mod.dev_major, &mod.dev_minor, &mod.inode, buf) != 8)
178*387f9dfdSAndroid Build Coastguard Worker       break;
179*387f9dfdSAndroid Build Coastguard Worker 
180*387f9dfdSAndroid Build Coastguard Worker     if (perm[2] != 'x')
181*387f9dfdSAndroid Build Coastguard Worker       continue;
182*387f9dfdSAndroid Build Coastguard Worker 
183*387f9dfdSAndroid Build Coastguard Worker     name = buf;
184*387f9dfdSAndroid Build Coastguard Worker     while (isspace(*name))
185*387f9dfdSAndroid Build Coastguard Worker       name++;
186*387f9dfdSAndroid Build Coastguard Worker     mod.name = name;
187*387f9dfdSAndroid Build Coastguard Worker     if (!bcc_mapping_is_file_backed(name))
188*387f9dfdSAndroid Build Coastguard Worker       continue;
189*387f9dfdSAndroid Build Coastguard Worker 
190*387f9dfdSAndroid Build Coastguard Worker     resolved_name = NULL;
191*387f9dfdSAndroid Build Coastguard Worker     if (strstr(name, "/memfd:")) {
192*387f9dfdSAndroid Build Coastguard Worker       resolved_name = _procutils_memfd_path(pid, mod.inode);
193*387f9dfdSAndroid Build Coastguard Worker       if (resolved_name != NULL) {
194*387f9dfdSAndroid Build Coastguard Worker         enter_ns = 0;
195*387f9dfdSAndroid Build Coastguard Worker       }
196*387f9dfdSAndroid Build Coastguard Worker     } else if (_procfs_might_be_zip_path(mod.name)) {
197*387f9dfdSAndroid Build Coastguard Worker       uint32_t zip_entry_offset = mod.file_offset;
198*387f9dfdSAndroid Build Coastguard Worker       resolved_name = _procfs_find_zip_entry(mod.name, pid, &zip_entry_offset);
199*387f9dfdSAndroid Build Coastguard Worker       if (resolved_name != NULL) {
200*387f9dfdSAndroid Build Coastguard Worker         mod.file_offset = zip_entry_offset;
201*387f9dfdSAndroid Build Coastguard Worker       }
202*387f9dfdSAndroid Build Coastguard Worker     }
203*387f9dfdSAndroid Build Coastguard Worker 
204*387f9dfdSAndroid Build Coastguard Worker     if (resolved_name != NULL) {
205*387f9dfdSAndroid Build Coastguard Worker       strncpy(buf, resolved_name, PATH_MAX);
206*387f9dfdSAndroid Build Coastguard Worker       buf[PATH_MAX] = 0;
207*387f9dfdSAndroid Build Coastguard Worker       free(resolved_name);
208*387f9dfdSAndroid Build Coastguard Worker       mod.name = buf;
209*387f9dfdSAndroid Build Coastguard Worker     }
210*387f9dfdSAndroid Build Coastguard Worker 
211*387f9dfdSAndroid Build Coastguard Worker     if (callback(&mod, enter_ns, payload) < 0)
212*387f9dfdSAndroid Build Coastguard Worker       return 0;
213*387f9dfdSAndroid Build Coastguard Worker   }
214*387f9dfdSAndroid Build Coastguard Worker 
215*387f9dfdSAndroid Build Coastguard Worker   return -1;
216*387f9dfdSAndroid Build Coastguard Worker }
217*387f9dfdSAndroid Build Coastguard Worker 
bcc_procutils_each_module(int pid,bcc_procutils_modulecb callback,void * payload)218*387f9dfdSAndroid Build Coastguard Worker int bcc_procutils_each_module(int pid, bcc_procutils_modulecb callback,
219*387f9dfdSAndroid Build Coastguard Worker                               void *payload) {
220*387f9dfdSAndroid Build Coastguard Worker   char procmap_filename[128];
221*387f9dfdSAndroid Build Coastguard Worker   FILE *procmap;
222*387f9dfdSAndroid Build Coastguard Worker   snprintf(procmap_filename, sizeof(procmap_filename), "/proc/%ld/maps",
223*387f9dfdSAndroid Build Coastguard Worker            (long)pid);
224*387f9dfdSAndroid Build Coastguard Worker   procmap = fopen(procmap_filename, "r");
225*387f9dfdSAndroid Build Coastguard Worker   if (!procmap)
226*387f9dfdSAndroid Build Coastguard Worker     return -1;
227*387f9dfdSAndroid Build Coastguard Worker 
228*387f9dfdSAndroid Build Coastguard Worker   _procfs_maps_each_module(procmap, pid, callback, payload);
229*387f9dfdSAndroid Build Coastguard Worker 
230*387f9dfdSAndroid Build Coastguard Worker   // Address mapping for the entire address space maybe in /tmp/perf-<PID>.map
231*387f9dfdSAndroid Build Coastguard Worker   // This will be used if symbols aren't resolved in an earlier mapping.
232*387f9dfdSAndroid Build Coastguard Worker   char map_path[4096];
233*387f9dfdSAndroid Build Coastguard Worker   // Try perf-<PID>.map path with process's mount namespace, chroot and NSPID,
234*387f9dfdSAndroid Build Coastguard Worker   // in case it is generated by the process itself.
235*387f9dfdSAndroid Build Coastguard Worker   mod_info mod;
236*387f9dfdSAndroid Build Coastguard Worker   memset(&mod, 0, sizeof(mod_info));
237*387f9dfdSAndroid Build Coastguard Worker   if (bcc_perf_map_path(map_path, sizeof(map_path), pid)) {
238*387f9dfdSAndroid Build Coastguard Worker     mod.name = map_path;
239*387f9dfdSAndroid Build Coastguard Worker     mod.end_addr = -1;
240*387f9dfdSAndroid Build Coastguard Worker     if (callback(&mod, 1, payload) < 0)
241*387f9dfdSAndroid Build Coastguard Worker       goto done;
242*387f9dfdSAndroid Build Coastguard Worker   }
243*387f9dfdSAndroid Build Coastguard Worker   // Try perf-<PID>.map path with global root and PID, in case it is generated
244*387f9dfdSAndroid Build Coastguard Worker   // by other Process. Avoid checking mount namespace for this.
245*387f9dfdSAndroid Build Coastguard Worker   memset(&mod, 0, sizeof(mod_info));
246*387f9dfdSAndroid Build Coastguard Worker   int res = snprintf(map_path, 4096, "/tmp/perf-%d.map", pid);
247*387f9dfdSAndroid Build Coastguard Worker   if (res > 0 && res < 4096) {
248*387f9dfdSAndroid Build Coastguard Worker     mod.name = map_path;
249*387f9dfdSAndroid Build Coastguard Worker     mod.end_addr = -1;
250*387f9dfdSAndroid Build Coastguard Worker     if (callback(&mod, 0, payload) < 0)
251*387f9dfdSAndroid Build Coastguard Worker       goto done;
252*387f9dfdSAndroid Build Coastguard Worker   }
253*387f9dfdSAndroid Build Coastguard Worker 
254*387f9dfdSAndroid Build Coastguard Worker done:
255*387f9dfdSAndroid Build Coastguard Worker   fclose(procmap);
256*387f9dfdSAndroid Build Coastguard Worker   return 0;
257*387f9dfdSAndroid Build Coastguard Worker }
258*387f9dfdSAndroid Build Coastguard Worker 
bcc_procutils_each_ksym(bcc_procutils_ksymcb callback,void * payload)259*387f9dfdSAndroid Build Coastguard Worker int bcc_procutils_each_ksym(bcc_procutils_ksymcb callback, void *payload) {
260*387f9dfdSAndroid Build Coastguard Worker   char line[2048];
261*387f9dfdSAndroid Build Coastguard Worker   char *symname, *endsym, *modname, *endmod = NULL;
262*387f9dfdSAndroid Build Coastguard Worker   FILE *kallsyms;
263*387f9dfdSAndroid Build Coastguard Worker   unsigned long long addr;
264*387f9dfdSAndroid Build Coastguard Worker 
265*387f9dfdSAndroid Build Coastguard Worker   kallsyms = fopen("/proc/kallsyms", "r");
266*387f9dfdSAndroid Build Coastguard Worker   if (!kallsyms)
267*387f9dfdSAndroid Build Coastguard Worker     return -1;
268*387f9dfdSAndroid Build Coastguard Worker 
269*387f9dfdSAndroid Build Coastguard Worker   while (fgets(line, sizeof(line), kallsyms)) {
270*387f9dfdSAndroid Build Coastguard Worker     addr = strtoull(line, &symname, 16);
271*387f9dfdSAndroid Build Coastguard Worker     if (addr == 0 || addr == ULLONG_MAX)
272*387f9dfdSAndroid Build Coastguard Worker       continue;
273*387f9dfdSAndroid Build Coastguard Worker     if (addr < kernelAddrSpace)
274*387f9dfdSAndroid Build Coastguard Worker       continue;
275*387f9dfdSAndroid Build Coastguard Worker 
276*387f9dfdSAndroid Build Coastguard Worker     symname++;
277*387f9dfdSAndroid Build Coastguard Worker     // Ignore data symbols
278*387f9dfdSAndroid Build Coastguard Worker     if (*symname == 'b' || *symname == 'B' || *symname == 'd' ||
279*387f9dfdSAndroid Build Coastguard Worker         *symname == 'D' || *symname == 'r' || *symname =='R')
280*387f9dfdSAndroid Build Coastguard Worker       continue;
281*387f9dfdSAndroid Build Coastguard Worker 
282*387f9dfdSAndroid Build Coastguard Worker     endsym = (symname = symname + 2);
283*387f9dfdSAndroid Build Coastguard Worker     while (*endsym && !isspace(*endsym)) endsym++;
284*387f9dfdSAndroid Build Coastguard Worker     *endsym = '\0';
285*387f9dfdSAndroid Build Coastguard Worker 
286*387f9dfdSAndroid Build Coastguard Worker     // Parse module name if it's available
287*387f9dfdSAndroid Build Coastguard Worker     modname = endsym + 1;
288*387f9dfdSAndroid Build Coastguard Worker     while (*modname && isspace(*endsym)) modname++;
289*387f9dfdSAndroid Build Coastguard Worker 
290*387f9dfdSAndroid Build Coastguard Worker     if (*modname && *modname == '[') {
291*387f9dfdSAndroid Build Coastguard Worker       endmod = ++modname;
292*387f9dfdSAndroid Build Coastguard Worker       while (*endmod && *endmod != ']') endmod++;
293*387f9dfdSAndroid Build Coastguard Worker       if (*endmod)
294*387f9dfdSAndroid Build Coastguard Worker         *(endmod) = '\0';
295*387f9dfdSAndroid Build Coastguard Worker       else
296*387f9dfdSAndroid Build Coastguard Worker         endmod = NULL;
297*387f9dfdSAndroid Build Coastguard Worker     }
298*387f9dfdSAndroid Build Coastguard Worker 
299*387f9dfdSAndroid Build Coastguard Worker     if (!endmod)
300*387f9dfdSAndroid Build Coastguard Worker       modname = "kernel";
301*387f9dfdSAndroid Build Coastguard Worker 
302*387f9dfdSAndroid Build Coastguard Worker     callback(symname, modname, addr, payload);
303*387f9dfdSAndroid Build Coastguard Worker   }
304*387f9dfdSAndroid Build Coastguard Worker 
305*387f9dfdSAndroid Build Coastguard Worker   fclose(kallsyms);
306*387f9dfdSAndroid Build Coastguard Worker   return 0;
307*387f9dfdSAndroid Build Coastguard Worker }
308*387f9dfdSAndroid Build Coastguard Worker 
309*387f9dfdSAndroid Build Coastguard Worker #define CACHE1_HEADER "ld.so-1.7.0"
310*387f9dfdSAndroid Build Coastguard Worker #define CACHE1_HEADER_LEN (sizeof(CACHE1_HEADER) - 1)
311*387f9dfdSAndroid Build Coastguard Worker 
312*387f9dfdSAndroid Build Coastguard Worker #define CACHE2_HEADER "glibc-ld.so.cache"
313*387f9dfdSAndroid Build Coastguard Worker #define CACHE2_HEADER_LEN (sizeof(CACHE2_HEADER) - 1)
314*387f9dfdSAndroid Build Coastguard Worker #define CACHE2_VERSION "1.1"
315*387f9dfdSAndroid Build Coastguard Worker 
316*387f9dfdSAndroid Build Coastguard Worker struct ld_cache1_entry {
317*387f9dfdSAndroid Build Coastguard Worker   int32_t flags;
318*387f9dfdSAndroid Build Coastguard Worker   uint32_t key;
319*387f9dfdSAndroid Build Coastguard Worker   uint32_t value;
320*387f9dfdSAndroid Build Coastguard Worker };
321*387f9dfdSAndroid Build Coastguard Worker 
322*387f9dfdSAndroid Build Coastguard Worker struct ld_cache1 {
323*387f9dfdSAndroid Build Coastguard Worker   char header[CACHE1_HEADER_LEN];
324*387f9dfdSAndroid Build Coastguard Worker   uint32_t entry_count;
325*387f9dfdSAndroid Build Coastguard Worker   struct ld_cache1_entry entries[0];
326*387f9dfdSAndroid Build Coastguard Worker };
327*387f9dfdSAndroid Build Coastguard Worker 
328*387f9dfdSAndroid Build Coastguard Worker struct ld_cache2_entry {
329*387f9dfdSAndroid Build Coastguard Worker   int32_t flags;
330*387f9dfdSAndroid Build Coastguard Worker   uint32_t key;
331*387f9dfdSAndroid Build Coastguard Worker   uint32_t value;
332*387f9dfdSAndroid Build Coastguard Worker   uint32_t pad1_;
333*387f9dfdSAndroid Build Coastguard Worker   uint64_t pad2_;
334*387f9dfdSAndroid Build Coastguard Worker };
335*387f9dfdSAndroid Build Coastguard Worker 
336*387f9dfdSAndroid Build Coastguard Worker struct ld_cache2 {
337*387f9dfdSAndroid Build Coastguard Worker   char header[CACHE2_HEADER_LEN];
338*387f9dfdSAndroid Build Coastguard Worker   char version[3];
339*387f9dfdSAndroid Build Coastguard Worker   uint32_t entry_count;
340*387f9dfdSAndroid Build Coastguard Worker   uint32_t string_table_len;
341*387f9dfdSAndroid Build Coastguard Worker   uint32_t pad_[5];
342*387f9dfdSAndroid Build Coastguard Worker   struct ld_cache2_entry entries[0];
343*387f9dfdSAndroid Build Coastguard Worker };
344*387f9dfdSAndroid Build Coastguard Worker 
345*387f9dfdSAndroid Build Coastguard Worker static int lib_cache_count;
346*387f9dfdSAndroid Build Coastguard Worker static struct ld_lib {
347*387f9dfdSAndroid Build Coastguard Worker   char *libname;
348*387f9dfdSAndroid Build Coastguard Worker   char *path;
349*387f9dfdSAndroid Build Coastguard Worker   int flags;
350*387f9dfdSAndroid Build Coastguard Worker } * lib_cache;
351*387f9dfdSAndroid Build Coastguard Worker 
read_cache1(const char * ld_map)352*387f9dfdSAndroid Build Coastguard Worker static int read_cache1(const char *ld_map) {
353*387f9dfdSAndroid Build Coastguard Worker   struct ld_cache1 *ldcache = (struct ld_cache1 *)ld_map;
354*387f9dfdSAndroid Build Coastguard Worker   const char *ldstrings =
355*387f9dfdSAndroid Build Coastguard Worker       (const char *)(ldcache->entries + ldcache->entry_count);
356*387f9dfdSAndroid Build Coastguard Worker   uint32_t i;
357*387f9dfdSAndroid Build Coastguard Worker 
358*387f9dfdSAndroid Build Coastguard Worker   lib_cache =
359*387f9dfdSAndroid Build Coastguard Worker       (struct ld_lib *)malloc(ldcache->entry_count * sizeof(struct ld_lib));
360*387f9dfdSAndroid Build Coastguard Worker   lib_cache_count = (int)ldcache->entry_count;
361*387f9dfdSAndroid Build Coastguard Worker 
362*387f9dfdSAndroid Build Coastguard Worker   for (i = 0; i < ldcache->entry_count; ++i) {
363*387f9dfdSAndroid Build Coastguard Worker     const char *key = ldstrings + ldcache->entries[i].key;
364*387f9dfdSAndroid Build Coastguard Worker     const char *val = ldstrings + ldcache->entries[i].value;
365*387f9dfdSAndroid Build Coastguard Worker     const int flags = ldcache->entries[i].flags;
366*387f9dfdSAndroid Build Coastguard Worker 
367*387f9dfdSAndroid Build Coastguard Worker     lib_cache[i].libname = strdup(key);
368*387f9dfdSAndroid Build Coastguard Worker     lib_cache[i].path = strdup(val);
369*387f9dfdSAndroid Build Coastguard Worker     lib_cache[i].flags = flags;
370*387f9dfdSAndroid Build Coastguard Worker   }
371*387f9dfdSAndroid Build Coastguard Worker   return 0;
372*387f9dfdSAndroid Build Coastguard Worker }
373*387f9dfdSAndroid Build Coastguard Worker 
read_cache2(const char * ld_map)374*387f9dfdSAndroid Build Coastguard Worker static int read_cache2(const char *ld_map) {
375*387f9dfdSAndroid Build Coastguard Worker   struct ld_cache2 *ldcache = (struct ld_cache2 *)ld_map;
376*387f9dfdSAndroid Build Coastguard Worker   uint32_t i;
377*387f9dfdSAndroid Build Coastguard Worker 
378*387f9dfdSAndroid Build Coastguard Worker   if (memcmp(ld_map, CACHE2_HEADER, CACHE2_HEADER_LEN))
379*387f9dfdSAndroid Build Coastguard Worker     return -1;
380*387f9dfdSAndroid Build Coastguard Worker 
381*387f9dfdSAndroid Build Coastguard Worker   lib_cache =
382*387f9dfdSAndroid Build Coastguard Worker       (struct ld_lib *)malloc(ldcache->entry_count * sizeof(struct ld_lib));
383*387f9dfdSAndroid Build Coastguard Worker   lib_cache_count = (int)ldcache->entry_count;
384*387f9dfdSAndroid Build Coastguard Worker 
385*387f9dfdSAndroid Build Coastguard Worker   for (i = 0; i < ldcache->entry_count; ++i) {
386*387f9dfdSAndroid Build Coastguard Worker     const char *key = ld_map + ldcache->entries[i].key;
387*387f9dfdSAndroid Build Coastguard Worker     const char *val = ld_map + ldcache->entries[i].value;
388*387f9dfdSAndroid Build Coastguard Worker     const int flags = ldcache->entries[i].flags;
389*387f9dfdSAndroid Build Coastguard Worker 
390*387f9dfdSAndroid Build Coastguard Worker     lib_cache[i].libname = strdup(key);
391*387f9dfdSAndroid Build Coastguard Worker     lib_cache[i].path = strdup(val);
392*387f9dfdSAndroid Build Coastguard Worker     lib_cache[i].flags = flags;
393*387f9dfdSAndroid Build Coastguard Worker   }
394*387f9dfdSAndroid Build Coastguard Worker   return 0;
395*387f9dfdSAndroid Build Coastguard Worker }
396*387f9dfdSAndroid Build Coastguard Worker 
load_ld_cache(const char * cache_path)397*387f9dfdSAndroid Build Coastguard Worker static int load_ld_cache(const char *cache_path) {
398*387f9dfdSAndroid Build Coastguard Worker   struct stat st;
399*387f9dfdSAndroid Build Coastguard Worker   size_t ld_size;
400*387f9dfdSAndroid Build Coastguard Worker   const char *ld_map;
401*387f9dfdSAndroid Build Coastguard Worker   int ret, fd = open(cache_path, O_RDONLY);
402*387f9dfdSAndroid Build Coastguard Worker 
403*387f9dfdSAndroid Build Coastguard Worker   if (fd < 0)
404*387f9dfdSAndroid Build Coastguard Worker     return -1;
405*387f9dfdSAndroid Build Coastguard Worker 
406*387f9dfdSAndroid Build Coastguard Worker   if (fstat(fd, &st) < 0 || st.st_size < sizeof(struct ld_cache1)) {
407*387f9dfdSAndroid Build Coastguard Worker     close(fd);
408*387f9dfdSAndroid Build Coastguard Worker     return -1;
409*387f9dfdSAndroid Build Coastguard Worker   }
410*387f9dfdSAndroid Build Coastguard Worker 
411*387f9dfdSAndroid Build Coastguard Worker   ld_size = st.st_size;
412*387f9dfdSAndroid Build Coastguard Worker   ld_map = (const char *)mmap(NULL, ld_size, PROT_READ, MAP_PRIVATE, fd, 0);
413*387f9dfdSAndroid Build Coastguard Worker   if (ld_map == MAP_FAILED) {
414*387f9dfdSAndroid Build Coastguard Worker     close(fd);
415*387f9dfdSAndroid Build Coastguard Worker     return -1;
416*387f9dfdSAndroid Build Coastguard Worker   }
417*387f9dfdSAndroid Build Coastguard Worker 
418*387f9dfdSAndroid Build Coastguard Worker   if (memcmp(ld_map, CACHE1_HEADER, CACHE1_HEADER_LEN) == 0) {
419*387f9dfdSAndroid Build Coastguard Worker     const struct ld_cache1 *cache1 = (struct ld_cache1 *)ld_map;
420*387f9dfdSAndroid Build Coastguard Worker     size_t cache1_len = sizeof(struct ld_cache1) +
421*387f9dfdSAndroid Build Coastguard Worker                         (cache1->entry_count * sizeof(struct ld_cache1_entry));
422*387f9dfdSAndroid Build Coastguard Worker     cache1_len = (cache1_len + 0x7) & ~0x7ULL;
423*387f9dfdSAndroid Build Coastguard Worker 
424*387f9dfdSAndroid Build Coastguard Worker     if (ld_size > (cache1_len + sizeof(struct ld_cache2)))
425*387f9dfdSAndroid Build Coastguard Worker       ret = read_cache2(ld_map + cache1_len);
426*387f9dfdSAndroid Build Coastguard Worker     else
427*387f9dfdSAndroid Build Coastguard Worker       ret = read_cache1(ld_map);
428*387f9dfdSAndroid Build Coastguard Worker   } else {
429*387f9dfdSAndroid Build Coastguard Worker     ret = read_cache2(ld_map);
430*387f9dfdSAndroid Build Coastguard Worker   }
431*387f9dfdSAndroid Build Coastguard Worker 
432*387f9dfdSAndroid Build Coastguard Worker   munmap((void *)ld_map, ld_size);
433*387f9dfdSAndroid Build Coastguard Worker   close(fd);
434*387f9dfdSAndroid Build Coastguard Worker   return ret;
435*387f9dfdSAndroid Build Coastguard Worker }
436*387f9dfdSAndroid Build Coastguard Worker 
437*387f9dfdSAndroid Build Coastguard Worker #define LD_SO_CACHE "/etc/ld.so.cache"
438*387f9dfdSAndroid Build Coastguard Worker #define FLAG_TYPE_MASK 0x00ff
439*387f9dfdSAndroid Build Coastguard Worker #define TYPE_ELF_LIBC6 0x0003
440*387f9dfdSAndroid Build Coastguard Worker #define FLAG_ABI_MASK 0xff00
441*387f9dfdSAndroid Build Coastguard Worker #define ABI_SPARC_LIB64 0x0100
442*387f9dfdSAndroid Build Coastguard Worker #define ABI_IA64_LIB64 0x0200
443*387f9dfdSAndroid Build Coastguard Worker #define ABI_X8664_LIB64 0x0300
444*387f9dfdSAndroid Build Coastguard Worker #define ABI_S390_LIB64 0x0400
445*387f9dfdSAndroid Build Coastguard Worker #define ABI_POWERPC_LIB64 0x0500
446*387f9dfdSAndroid Build Coastguard Worker #define ABI_AARCH64_LIB64 0x0a00
447*387f9dfdSAndroid Build Coastguard Worker 
match_so_flags(int flags)448*387f9dfdSAndroid Build Coastguard Worker static bool match_so_flags(int flags) {
449*387f9dfdSAndroid Build Coastguard Worker   if ((flags & FLAG_TYPE_MASK) != TYPE_ELF_LIBC6)
450*387f9dfdSAndroid Build Coastguard Worker     return false;
451*387f9dfdSAndroid Build Coastguard Worker 
452*387f9dfdSAndroid Build Coastguard Worker   switch (flags & FLAG_ABI_MASK) {
453*387f9dfdSAndroid Build Coastguard Worker   case ABI_SPARC_LIB64:
454*387f9dfdSAndroid Build Coastguard Worker   case ABI_IA64_LIB64:
455*387f9dfdSAndroid Build Coastguard Worker   case ABI_X8664_LIB64:
456*387f9dfdSAndroid Build Coastguard Worker   case ABI_S390_LIB64:
457*387f9dfdSAndroid Build Coastguard Worker   case ABI_POWERPC_LIB64:
458*387f9dfdSAndroid Build Coastguard Worker   case ABI_AARCH64_LIB64:
459*387f9dfdSAndroid Build Coastguard Worker     return (sizeof(void *) == 8);
460*387f9dfdSAndroid Build Coastguard Worker   }
461*387f9dfdSAndroid Build Coastguard Worker 
462*387f9dfdSAndroid Build Coastguard Worker   return sizeof(void *) == 4;
463*387f9dfdSAndroid Build Coastguard Worker }
464*387f9dfdSAndroid Build Coastguard Worker 
which_so_in_process(const char * libname,int pid,char * libpath)465*387f9dfdSAndroid Build Coastguard Worker static bool which_so_in_process(const char* libname, int pid, char* libpath) {
466*387f9dfdSAndroid Build Coastguard Worker   int ret, found = false;
467*387f9dfdSAndroid Build Coastguard Worker   char endline[4096], *mapname = NULL, *newline;
468*387f9dfdSAndroid Build Coastguard Worker   char mappings_file[128];
469*387f9dfdSAndroid Build Coastguard Worker   const size_t search_len = strlen(libname) + strlen("/lib.");
470*387f9dfdSAndroid Build Coastguard Worker   char search1[search_len + 1];
471*387f9dfdSAndroid Build Coastguard Worker   char search2[search_len + 1];
472*387f9dfdSAndroid Build Coastguard Worker 
473*387f9dfdSAndroid Build Coastguard Worker   snprintf(mappings_file, sizeof(mappings_file), "/proc/%ld/maps", (long)pid);
474*387f9dfdSAndroid Build Coastguard Worker   FILE *fp = fopen(mappings_file, "r");
475*387f9dfdSAndroid Build Coastguard Worker   if (!fp)
476*387f9dfdSAndroid Build Coastguard Worker     return NULL;
477*387f9dfdSAndroid Build Coastguard Worker 
478*387f9dfdSAndroid Build Coastguard Worker   snprintf(search1, search_len + 1, "/lib%s.", libname);
479*387f9dfdSAndroid Build Coastguard Worker   snprintf(search2, search_len + 1, "/lib%s-", libname);
480*387f9dfdSAndroid Build Coastguard Worker 
481*387f9dfdSAndroid Build Coastguard Worker   do {
482*387f9dfdSAndroid Build Coastguard Worker     ret = fscanf(fp, "%*x-%*x %*s %*x %*s %*d");
483*387f9dfdSAndroid Build Coastguard Worker     if (!fgets(endline, sizeof(endline), fp))
484*387f9dfdSAndroid Build Coastguard Worker       break;
485*387f9dfdSAndroid Build Coastguard Worker 
486*387f9dfdSAndroid Build Coastguard Worker     mapname = endline;
487*387f9dfdSAndroid Build Coastguard Worker     newline = strchr(endline, '\n');
488*387f9dfdSAndroid Build Coastguard Worker     if (newline)
489*387f9dfdSAndroid Build Coastguard Worker       newline[0] = '\0';
490*387f9dfdSAndroid Build Coastguard Worker 
491*387f9dfdSAndroid Build Coastguard Worker     while (isspace(mapname[0])) mapname++;
492*387f9dfdSAndroid Build Coastguard Worker 
493*387f9dfdSAndroid Build Coastguard Worker     if (strstr(mapname, ".so") && (strstr(mapname, search1) ||
494*387f9dfdSAndroid Build Coastguard Worker                                    strstr(mapname, search2))) {
495*387f9dfdSAndroid Build Coastguard Worker       found = true;
496*387f9dfdSAndroid Build Coastguard Worker       memcpy(libpath, mapname, strlen(mapname) + 1);
497*387f9dfdSAndroid Build Coastguard Worker       break;
498*387f9dfdSAndroid Build Coastguard Worker     }
499*387f9dfdSAndroid Build Coastguard Worker   } while (ret != EOF);
500*387f9dfdSAndroid Build Coastguard Worker 
501*387f9dfdSAndroid Build Coastguard Worker   fclose(fp);
502*387f9dfdSAndroid Build Coastguard Worker   return found;
503*387f9dfdSAndroid Build Coastguard Worker }
504*387f9dfdSAndroid Build Coastguard Worker 
bcc_procutils_which_so(const char * libname,int pid)505*387f9dfdSAndroid Build Coastguard Worker char *bcc_procutils_which_so(const char *libname, int pid) {
506*387f9dfdSAndroid Build Coastguard Worker   const size_t soname_len = strlen(libname) + strlen("lib.so");
507*387f9dfdSAndroid Build Coastguard Worker   char soname[soname_len + 1];
508*387f9dfdSAndroid Build Coastguard Worker   char libpath[4096];
509*387f9dfdSAndroid Build Coastguard Worker   int i;
510*387f9dfdSAndroid Build Coastguard Worker 
511*387f9dfdSAndroid Build Coastguard Worker   if (strchr(libname, '/'))
512*387f9dfdSAndroid Build Coastguard Worker     return strdup(libname);
513*387f9dfdSAndroid Build Coastguard Worker 
514*387f9dfdSAndroid Build Coastguard Worker   if (pid && which_so_in_process(libname, pid, libpath))
515*387f9dfdSAndroid Build Coastguard Worker     return strdup(libpath);
516*387f9dfdSAndroid Build Coastguard Worker 
517*387f9dfdSAndroid Build Coastguard Worker   if (lib_cache_count < 0)
518*387f9dfdSAndroid Build Coastguard Worker     return NULL;
519*387f9dfdSAndroid Build Coastguard Worker 
520*387f9dfdSAndroid Build Coastguard Worker   if (!lib_cache_count && load_ld_cache(LD_SO_CACHE) < 0) {
521*387f9dfdSAndroid Build Coastguard Worker     lib_cache_count = -1;
522*387f9dfdSAndroid Build Coastguard Worker     return NULL;
523*387f9dfdSAndroid Build Coastguard Worker   }
524*387f9dfdSAndroid Build Coastguard Worker 
525*387f9dfdSAndroid Build Coastguard Worker   snprintf(soname, soname_len + 1, "lib%s.so", libname);
526*387f9dfdSAndroid Build Coastguard Worker 
527*387f9dfdSAndroid Build Coastguard Worker   for (i = 0; i < lib_cache_count; ++i) {
528*387f9dfdSAndroid Build Coastguard Worker     if (!strncmp(lib_cache[i].libname, soname, soname_len) &&
529*387f9dfdSAndroid Build Coastguard Worker         match_so_flags(lib_cache[i].flags)) {
530*387f9dfdSAndroid Build Coastguard Worker       return strdup(lib_cache[i].path);
531*387f9dfdSAndroid Build Coastguard Worker     }
532*387f9dfdSAndroid Build Coastguard Worker   }
533*387f9dfdSAndroid Build Coastguard Worker   return NULL;
534*387f9dfdSAndroid Build Coastguard Worker }
535*387f9dfdSAndroid Build Coastguard Worker 
bcc_procutils_free(const char * ptr)536*387f9dfdSAndroid Build Coastguard Worker void bcc_procutils_free(const char *ptr) {
537*387f9dfdSAndroid Build Coastguard Worker   free((void *)ptr);
538*387f9dfdSAndroid Build Coastguard Worker }
539*387f9dfdSAndroid Build Coastguard Worker 
540*387f9dfdSAndroid Build Coastguard Worker /* Detects the following languages + C. */
541*387f9dfdSAndroid Build Coastguard Worker const char *languages[] = {"java", "node", "perl", "php", "python", "ruby"};
542*387f9dfdSAndroid Build Coastguard Worker const char *language_c = "c";
543*387f9dfdSAndroid Build Coastguard Worker const int nb_languages = 6;
544*387f9dfdSAndroid Build Coastguard Worker 
bcc_procutils_language(int pid)545*387f9dfdSAndroid Build Coastguard Worker const char *bcc_procutils_language(int pid) {
546*387f9dfdSAndroid Build Coastguard Worker   char procfilename[24], line[4096], pathname[32], *str;
547*387f9dfdSAndroid Build Coastguard Worker   FILE *procfile;
548*387f9dfdSAndroid Build Coastguard Worker   int i, ret;
549*387f9dfdSAndroid Build Coastguard Worker 
550*387f9dfdSAndroid Build Coastguard Worker   /* Look for clues in the absolute path to the executable. */
551*387f9dfdSAndroid Build Coastguard Worker   snprintf(procfilename, sizeof(procfilename), "/proc/%ld/exe", (long)pid);
552*387f9dfdSAndroid Build Coastguard Worker   if (realpath(procfilename, line)) {
553*387f9dfdSAndroid Build Coastguard Worker     for (i = 0; i < nb_languages; i++)
554*387f9dfdSAndroid Build Coastguard Worker       if (strstr(line, languages[i]))
555*387f9dfdSAndroid Build Coastguard Worker         return languages[i];
556*387f9dfdSAndroid Build Coastguard Worker   }
557*387f9dfdSAndroid Build Coastguard Worker 
558*387f9dfdSAndroid Build Coastguard Worker 
559*387f9dfdSAndroid Build Coastguard Worker   snprintf(procfilename, sizeof(procfilename), "/proc/%ld/maps", (long)pid);
560*387f9dfdSAndroid Build Coastguard Worker   procfile = fopen(procfilename, "r");
561*387f9dfdSAndroid Build Coastguard Worker   if (!procfile)
562*387f9dfdSAndroid Build Coastguard Worker     return NULL;
563*387f9dfdSAndroid Build Coastguard Worker 
564*387f9dfdSAndroid Build Coastguard Worker   /* Look for clues in memory mappings. */
565*387f9dfdSAndroid Build Coastguard Worker   bool libc = false;
566*387f9dfdSAndroid Build Coastguard Worker   do {
567*387f9dfdSAndroid Build Coastguard Worker     char perm[8], dev[8];
568*387f9dfdSAndroid Build Coastguard Worker     long long begin, end, size, inode;
569*387f9dfdSAndroid Build Coastguard Worker     ret = fscanf(procfile, "%llx-%llx %s %llx %s %lld", &begin, &end, perm,
570*387f9dfdSAndroid Build Coastguard Worker                  &size, dev, &inode);
571*387f9dfdSAndroid Build Coastguard Worker     if (!fgets(line, sizeof(line), procfile))
572*387f9dfdSAndroid Build Coastguard Worker       break;
573*387f9dfdSAndroid Build Coastguard Worker     if (ret == 6) {
574*387f9dfdSAndroid Build Coastguard Worker       char *mapname = line;
575*387f9dfdSAndroid Build Coastguard Worker       char *newline = strchr(line, '\n');
576*387f9dfdSAndroid Build Coastguard Worker       if (newline)
577*387f9dfdSAndroid Build Coastguard Worker         newline[0] = '\0';
578*387f9dfdSAndroid Build Coastguard Worker       while (isspace(mapname[0])) mapname++;
579*387f9dfdSAndroid Build Coastguard Worker       for (i = 0; i < nb_languages; i++) {
580*387f9dfdSAndroid Build Coastguard Worker         snprintf(pathname, sizeof(pathname), "/lib%s", languages[i]);
581*387f9dfdSAndroid Build Coastguard Worker         if (strstr(mapname, pathname)) {
582*387f9dfdSAndroid Build Coastguard Worker           fclose(procfile);
583*387f9dfdSAndroid Build Coastguard Worker           return languages[i];
584*387f9dfdSAndroid Build Coastguard Worker 	}
585*387f9dfdSAndroid Build Coastguard Worker         if ((str = strstr(mapname, "libc")) &&
586*387f9dfdSAndroid Build Coastguard Worker             (str[4] == '-' || str[4] == '.'))
587*387f9dfdSAndroid Build Coastguard Worker           libc = true;
588*387f9dfdSAndroid Build Coastguard Worker       }
589*387f9dfdSAndroid Build Coastguard Worker     }
590*387f9dfdSAndroid Build Coastguard Worker   } while (ret && ret != EOF);
591*387f9dfdSAndroid Build Coastguard Worker 
592*387f9dfdSAndroid Build Coastguard Worker   fclose(procfile);
593*387f9dfdSAndroid Build Coastguard Worker 
594*387f9dfdSAndroid Build Coastguard Worker   /* Return C as the language if libc was found and nothing else. */
595*387f9dfdSAndroid Build Coastguard Worker   return libc ? language_c : NULL;
596*387f9dfdSAndroid Build Coastguard Worker }
597