srcline.c 1.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <linux/kernel.h>
  5. #include "util/util.h"
  6. #include "util/debug.h"
  7. static int addr2line(const char *dso_name, unsigned long addr,
  8. char **file, unsigned int *line_nr)
  9. {
  10. FILE *fp;
  11. char cmd[PATH_MAX];
  12. char *filename = NULL;
  13. size_t len;
  14. char *sep;
  15. int ret = 0;
  16. scnprintf(cmd, sizeof(cmd), "addr2line -e %s %016"PRIx64,
  17. dso_name, addr);
  18. fp = popen(cmd, "r");
  19. if (fp == NULL) {
  20. pr_warning("popen failed for %s\n", dso_name);
  21. return 0;
  22. }
  23. if (getline(&filename, &len, fp) < 0 || !len) {
  24. pr_warning("addr2line has no output for %s\n", dso_name);
  25. goto out;
  26. }
  27. sep = strchr(filename, '\n');
  28. if (sep)
  29. *sep = '\0';
  30. if (!strcmp(filename, "??:0")) {
  31. pr_debug("no debugging info in %s\n", dso_name);
  32. free(filename);
  33. goto out;
  34. }
  35. sep = strchr(filename, ':');
  36. if (sep) {
  37. *sep++ = '\0';
  38. *file = filename;
  39. *line_nr = strtoul(sep, NULL, 0);
  40. ret = 1;
  41. }
  42. out:
  43. pclose(fp);
  44. return ret;
  45. }
  46. char *get_srcline(const char *dso_name, unsigned long addr)
  47. {
  48. char *file;
  49. unsigned line;
  50. char *srcline = SRCLINE_UNKNOWN;
  51. size_t size;
  52. if (dso_name[0] == '[')
  53. goto out;
  54. if (!strncmp(dso_name, "/tmp/perf-", 10))
  55. goto out;
  56. if (!addr2line(dso_name, addr, &file, &line))
  57. goto out;
  58. /* just calculate actual length */
  59. size = snprintf(NULL, 0, "%s:%u", file, line) + 1;
  60. srcline = malloc(size);
  61. if (srcline)
  62. snprintf(srcline, size, "%s:%u", file, line);
  63. else
  64. srcline = SRCLINE_UNKNOWN;
  65. free(file);
  66. out:
  67. return srcline;
  68. }
  69. void free_srcline(char *srcline)
  70. {
  71. if (srcline && strcmp(srcline, SRCLINE_UNKNOWN) != 0)
  72. free(srcline);
  73. }