x86/amd-iommu: Add per IOMMU reference counting
[linux/fpc-iii.git] / tools / perf / util / module.c
blob0d8c85defcd2b02b1c772e323b9262346d471d33
1 #include "util.h"
2 #include "../perf.h"
3 #include "string.h"
4 #include "module.h"
6 #include <libelf.h>
7 #include <libgen.h>
8 #include <gelf.h>
9 #include <elf.h>
10 #include <dirent.h>
11 #include <sys/utsname.h>
13 static unsigned int crc32(const char *p, unsigned int len)
15 int i;
16 unsigned int crc = 0;
18 while (len--) {
19 crc ^= *p++;
20 for (i = 0; i < 8; i++)
21 crc = (crc >> 1) ^ ((crc & 1) ? 0xedb88320 : 0);
23 return crc;
26 /* module section methods */
28 struct sec_dso *sec_dso__new_dso(const char *name)
30 struct sec_dso *self = malloc(sizeof(*self) + strlen(name) + 1);
32 if (self != NULL) {
33 strcpy(self->name, name);
34 self->secs = RB_ROOT;
35 self->find_section = sec_dso__find_section;
38 return self;
41 static void sec_dso__delete_section(struct section *self)
43 free(((void *)self));
46 void sec_dso__delete_sections(struct sec_dso *self)
48 struct section *pos;
49 struct rb_node *next = rb_first(&self->secs);
51 while (next) {
52 pos = rb_entry(next, struct section, rb_node);
53 next = rb_next(&pos->rb_node);
54 rb_erase(&pos->rb_node, &self->secs);
55 sec_dso__delete_section(pos);
59 void sec_dso__delete_self(struct sec_dso *self)
61 sec_dso__delete_sections(self);
62 free(self);
65 static void sec_dso__insert_section(struct sec_dso *self, struct section *sec)
67 struct rb_node **p = &self->secs.rb_node;
68 struct rb_node *parent = NULL;
69 const u64 hash = sec->hash;
70 struct section *s;
72 while (*p != NULL) {
73 parent = *p;
74 s = rb_entry(parent, struct section, rb_node);
75 if (hash < s->hash)
76 p = &(*p)->rb_left;
77 else
78 p = &(*p)->rb_right;
80 rb_link_node(&sec->rb_node, parent, p);
81 rb_insert_color(&sec->rb_node, &self->secs);
84 struct section *sec_dso__find_section(struct sec_dso *self, const char *name)
86 struct rb_node *n;
87 u64 hash;
88 int len;
90 if (self == NULL)
91 return NULL;
93 len = strlen(name);
94 hash = crc32(name, len);
96 n = self->secs.rb_node;
98 while (n) {
99 struct section *s = rb_entry(n, struct section, rb_node);
101 if (hash < s->hash)
102 n = n->rb_left;
103 else if (hash > s->hash)
104 n = n->rb_right;
105 else {
106 if (!strcmp(name, s->name))
107 return s;
108 else
109 n = rb_next(&s->rb_node);
113 return NULL;
116 static size_t sec_dso__fprintf_section(struct section *self, FILE *fp)
118 return fprintf(fp, "name:%s vma:%llx path:%s\n",
119 self->name, self->vma, self->path);
122 size_t sec_dso__fprintf(struct sec_dso *self, FILE *fp)
124 size_t ret = fprintf(fp, "dso: %s\n", self->name);
126 struct rb_node *nd;
127 for (nd = rb_first(&self->secs); nd; nd = rb_next(nd)) {
128 struct section *pos = rb_entry(nd, struct section, rb_node);
129 ret += sec_dso__fprintf_section(pos, fp);
132 return ret;
135 static struct section *section__new(const char *name, const char *path)
137 struct section *self = calloc(1, sizeof(*self));
139 if (!self)
140 goto out_failure;
142 self->name = calloc(1, strlen(name) + 1);
143 if (!self->name)
144 goto out_failure;
146 self->path = calloc(1, strlen(path) + 1);
147 if (!self->path)
148 goto out_failure;
150 strcpy(self->name, name);
151 strcpy(self->path, path);
152 self->hash = crc32(self->name, strlen(name));
154 return self;
156 out_failure:
157 if (self) {
158 if (self->name)
159 free(self->name);
160 if (self->path)
161 free(self->path);
162 free(self);
165 return NULL;
168 /* module methods */
170 struct mod_dso *mod_dso__new_dso(const char *name)
172 struct mod_dso *self = malloc(sizeof(*self) + strlen(name) + 1);
174 if (self != NULL) {
175 strcpy(self->name, name);
176 self->mods = RB_ROOT;
177 self->find_module = mod_dso__find_module;
180 return self;
183 static void mod_dso__delete_module(struct module *self)
185 free(((void *)self));
188 void mod_dso__delete_modules(struct mod_dso *self)
190 struct module *pos;
191 struct rb_node *next = rb_first(&self->mods);
193 while (next) {
194 pos = rb_entry(next, struct module, rb_node);
195 next = rb_next(&pos->rb_node);
196 rb_erase(&pos->rb_node, &self->mods);
197 mod_dso__delete_module(pos);
201 void mod_dso__delete_self(struct mod_dso *self)
203 mod_dso__delete_modules(self);
204 free(self);
207 static void mod_dso__insert_module(struct mod_dso *self, struct module *mod)
209 struct rb_node **p = &self->mods.rb_node;
210 struct rb_node *parent = NULL;
211 const u64 hash = mod->hash;
212 struct module *m;
214 while (*p != NULL) {
215 parent = *p;
216 m = rb_entry(parent, struct module, rb_node);
217 if (hash < m->hash)
218 p = &(*p)->rb_left;
219 else
220 p = &(*p)->rb_right;
222 rb_link_node(&mod->rb_node, parent, p);
223 rb_insert_color(&mod->rb_node, &self->mods);
226 struct module *mod_dso__find_module(struct mod_dso *self, const char *name)
228 struct rb_node *n;
229 u64 hash;
230 int len;
232 if (self == NULL)
233 return NULL;
235 len = strlen(name);
236 hash = crc32(name, len);
238 n = self->mods.rb_node;
240 while (n) {
241 struct module *m = rb_entry(n, struct module, rb_node);
243 if (hash < m->hash)
244 n = n->rb_left;
245 else if (hash > m->hash)
246 n = n->rb_right;
247 else {
248 if (!strcmp(name, m->name))
249 return m;
250 else
251 n = rb_next(&m->rb_node);
255 return NULL;
258 static size_t mod_dso__fprintf_module(struct module *self, FILE *fp)
260 return fprintf(fp, "name:%s path:%s\n", self->name, self->path);
263 size_t mod_dso__fprintf(struct mod_dso *self, FILE *fp)
265 struct rb_node *nd;
266 size_t ret;
268 ret = fprintf(fp, "dso: %s\n", self->name);
270 for (nd = rb_first(&self->mods); nd; nd = rb_next(nd)) {
271 struct module *pos = rb_entry(nd, struct module, rb_node);
273 ret += mod_dso__fprintf_module(pos, fp);
276 return ret;
279 static struct module *module__new(const char *name, const char *path)
281 struct module *self = calloc(1, sizeof(*self));
283 if (!self)
284 goto out_failure;
286 self->name = calloc(1, strlen(name) + 1);
287 if (!self->name)
288 goto out_failure;
290 self->path = calloc(1, strlen(path) + 1);
291 if (!self->path)
292 goto out_failure;
294 strcpy(self->name, name);
295 strcpy(self->path, path);
296 self->hash = crc32(self->name, strlen(name));
298 return self;
300 out_failure:
301 if (self) {
302 if (self->name)
303 free(self->name);
304 if (self->path)
305 free(self->path);
306 free(self);
309 return NULL;
312 static int mod_dso__load_sections(struct module *mod)
314 int count = 0, path_len;
315 struct dirent *entry;
316 char *line = NULL;
317 char *dir_path;
318 DIR *dir;
319 size_t n;
321 path_len = strlen("/sys/module/");
322 path_len += strlen(mod->name);
323 path_len += strlen("/sections/");
325 dir_path = calloc(1, path_len + 1);
326 if (dir_path == NULL)
327 goto out_failure;
329 strcat(dir_path, "/sys/module/");
330 strcat(dir_path, mod->name);
331 strcat(dir_path, "/sections/");
333 dir = opendir(dir_path);
334 if (dir == NULL)
335 goto out_free;
337 while ((entry = readdir(dir))) {
338 struct section *section;
339 char *path, *vma;
340 int line_len;
341 FILE *file;
343 if (!strcmp(".", entry->d_name) || !strcmp("..", entry->d_name))
344 continue;
346 path = calloc(1, path_len + strlen(entry->d_name) + 1);
347 if (path == NULL)
348 break;
349 strcat(path, dir_path);
350 strcat(path, entry->d_name);
352 file = fopen(path, "r");
353 if (file == NULL) {
354 free(path);
355 break;
358 line_len = getline(&line, &n, file);
359 if (line_len < 0) {
360 free(path);
361 fclose(file);
362 break;
365 if (!line) {
366 free(path);
367 fclose(file);
368 break;
371 line[--line_len] = '\0'; /* \n */
373 vma = strstr(line, "0x");
374 if (!vma) {
375 free(path);
376 fclose(file);
377 break;
379 vma += 2;
381 section = section__new(entry->d_name, path);
382 if (!section) {
383 fprintf(stderr, "load_sections: allocation error\n");
384 free(path);
385 fclose(file);
386 break;
389 hex2u64(vma, &section->vma);
390 sec_dso__insert_section(mod->sections, section);
392 free(path);
393 fclose(file);
394 count++;
397 closedir(dir);
398 free(line);
399 free(dir_path);
401 return count;
403 out_free:
404 free(dir_path);
406 out_failure:
407 return count;
410 static int mod_dso__load_module_paths(struct mod_dso *self)
412 struct utsname uts;
413 int count = 0, len, err = -1;
414 char *line = NULL;
415 FILE *file;
416 char *dpath, *dir;
417 size_t n;
419 if (uname(&uts) < 0)
420 return err;
422 len = strlen("/lib/modules/");
423 len += strlen(uts.release);
424 len += strlen("/modules.dep");
426 dpath = calloc(1, len + 1);
427 if (dpath == NULL)
428 return err;
430 strcat(dpath, "/lib/modules/");
431 strcat(dpath, uts.release);
432 strcat(dpath, "/modules.dep");
434 file = fopen(dpath, "r");
435 if (file == NULL)
436 goto out_failure;
438 dir = dirname(dpath);
439 if (!dir)
440 goto out_failure;
441 strcat(dir, "/");
443 while (!feof(file)) {
444 struct module *module;
445 char *name, *path, *tmp;
446 FILE *modfile;
447 int line_len;
449 line_len = getline(&line, &n, file);
450 if (line_len < 0)
451 break;
453 if (!line)
454 break;
456 line[--line_len] = '\0'; /* \n */
458 path = strchr(line, ':');
459 if (!path)
460 break;
461 *path = '\0';
463 path = strdup(line);
464 if (!path)
465 break;
467 if (!strstr(path, dir)) {
468 if (strncmp(path, "kernel/", 7))
469 break;
471 free(path);
472 path = calloc(1, strlen(dir) + strlen(line) + 1);
473 if (!path)
474 break;
475 strcat(path, dir);
476 strcat(path, line);
479 modfile = fopen(path, "r");
480 if (modfile == NULL)
481 break;
482 fclose(modfile);
484 name = strdup(path);
485 if (!name)
486 break;
488 name = strtok(name, "/");
489 tmp = name;
491 while (tmp) {
492 tmp = strtok(NULL, "/");
493 if (tmp)
494 name = tmp;
497 name = strsep(&name, ".");
498 if (!name)
499 break;
501 /* Quirk: replace '-' with '_' in all modules */
502 for (len = strlen(name); len; len--) {
503 if (*(name+len) == '-')
504 *(name+len) = '_';
507 module = module__new(name, path);
508 if (!module)
509 break;
510 mod_dso__insert_module(self, module);
512 module->sections = sec_dso__new_dso("sections");
513 if (!module->sections)
514 break;
516 module->active = mod_dso__load_sections(module);
518 if (module->active > 0)
519 count++;
522 if (feof(file))
523 err = count;
524 else
525 fprintf(stderr, "load_module_paths: modules.dep parsing failure!\n");
527 out_failure:
528 if (dpath)
529 free(dpath);
530 if (file)
531 fclose(file);
532 if (line)
533 free(line);
535 return err;
538 int mod_dso__load_modules(struct mod_dso *dso)
540 int err;
542 err = mod_dso__load_module_paths(dso);
544 return err;