与命令行相比,C中的Dtrace不会产生相同的分析结果

时间:2016-05-04 16:26:45

标签: dtrace

我想以编程方式从C中为Node.js跟踪堆栈(除了JS地址)。

以下命令为我提供了已解决的c++符号的堆栈。

sudo dtrace -C -n 'profile-101 /pid == 13221/ { ustack() }'

以下C代码仅返回Node的C / C ++代码的地址。什么是等价的?

#include <dtrace.h>
#include <signal.h>
#include <stdio.h>

struct ps_prochandle *g_pr;
static dtrace_hdl_t* g_dtp;

static int chewrec (const dtrace_probedata_t *data, const dtrace_recdesc_t *rec, void *arg) {
   // A NULL rec indicates that we've processed the last record.
   if (rec == NULL) {
      return (DTRACE_CONSUME_NEXT);
   }
   return (DTRACE_CONSUME_THIS);
}

static const char* g_prog =
  "#pragma D option switchrate=1000hz\n"
  "profile-1ms /pid == 13221/ {\n"
    "ustack();\n"
  "}";

static int g_intr;
static int g_exited;

static void intr (int signo) {
   g_intr = 1;
}


int main (int argc, char** argv) {
   int err;

   if ((g_dtp = dtrace_open(DTRACE_VERSION, 0, &err)) == NULL) {
      fprintf(stderr, "failed to initialize dtrace: %s\n", dtrace_errmsg(NULL, err));
      return -1;
   }
   printf("Dtrace initialized\n");

   (void) dtrace_setopt(g_dtp, "bufsize", "4m");
   (void) dtrace_setopt(g_dtp, "aggsize", "4m");

   printf("dtrace options set\n");

   dtrace_prog_t* prog;
   if ((prog = dtrace_program_strcompile(g_dtp, g_prog, DTRACE_PROBESPEC_NAME, DTRACE_C_CPP, 0, NULL)) == NULL) {
      fprintf(stderr, "failed to compile dtrace program\n");
      return -1;
   } else {
      printf("dtrace program compiled\n");
   }

   dtrace_proginfo_t info;
   if (dtrace_program_exec(g_dtp, prog, &info) == -1) {
      fprintf(stderr, "failed to enable dtrace probes\n");
      return -1;
   } else {
      printf("dtrace probes enabled\n");
   }

   struct sigaction act;
   (void) sigemptyset(&act.sa_mask);
   act.sa_flags = 0;
   act.sa_handler = intr;
   (void) sigaction(SIGINT, &act, NULL);
   (void) sigaction(SIGTERM, &act, NULL);

   if (dtrace_go(g_dtp) != 0) {
      fprintf(stderr, "could not start instrumentation\n");
      return -1;
   } else {
      printf("instrumentation started ..\n");
   }

   int done = 0;
   do {
      if (!g_intr && !done) {
         dtrace_sleep(g_dtp);
      }

      if (done || g_intr || g_exited) {
         done = 1;
         if (dtrace_stop(g_dtp) == -1) {
            fprintf(stderr, "could not stop tracing\n");
            return -1;
         }
      }

      switch (dtrace_work(g_dtp, stdout, NULL, chewrec, NULL)) {
         case DTRACE_WORKSTATUS_DONE:
            done = 1;
            break;
         case DTRACE_WORKSTATUS_OKAY:
            break;
         default:
            fprintf(stderr, "processing aborted");
            return -1;
      }
   } while (!done);

   printf("closing dtrace\n");
   dtrace_close(g_dtp);

   return 0;
}

1 个答案:

答案 0 :(得分:1)

来自dtrace-Mailing列表,Robert Mustacchi:

  

TL; DR 自己在用户区解析符号。

&#34;因此,所有符号解析处理始终在用户区完成。在 换句话说,内核中的DTrace只会收集地址之类的内容 您是否正在查看您使用ustack而非ustack的情况 处理程序通过jstack()动作(jstack()也只返回符号 非原生帧)。请注意,如果您想查看特定于JavaScript的内容 除了原生符号之外的符号,你想要使用jstack() 在你的例子而不是ustack()。

这些映射的执行方式会有所不同 你在什么系统上。如果你看一下DTrace对illumos的影响 打印ustack()的结果 (http://src.illumos.org/source/xref/illumos-gate/usr/src/lib/libdtrace/common/dt_consume.c#1320), 然后你会看到它使用libproc和Plookup_by_addr()函数 执行翻译。虽然值得指出这两点都不值得 是稳定的界面,虽然它们很少改变。&#34;