mirror of
https://github.com/radareorg/radare2.git
synced 2024-11-23 13:19:54 +00:00
165 lines
3.5 KiB
C
165 lines
3.5 KiB
C
/* radare - LGPL - Copyright 2009-2017 - pancake */
|
|
|
|
#include <r_debug.h>
|
|
|
|
R_API RDebugPid *r_debug_pid_new(const char *path, int pid, int uid, char status, ut64 pc) {
|
|
RDebugPid *p = R_NEW0 (RDebugPid);
|
|
if (!p) {
|
|
return NULL;
|
|
}
|
|
p->path = strdup (path);
|
|
p->pid = pid;
|
|
p->uid = uid;
|
|
p->status = status;
|
|
p->runnable = true;
|
|
p->pc = pc;
|
|
return p;
|
|
}
|
|
|
|
R_API RDebugPid *r_debug_pid_free(RDebugPid *pid) {
|
|
free (pid->path);
|
|
free (pid);
|
|
return NULL;
|
|
}
|
|
|
|
R_API RList *r_debug_pids(RDebug *dbg, int pid) {
|
|
if (dbg && dbg->h && dbg->h->pids) {
|
|
return dbg->h->pids (dbg, pid);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
// TODO: deprecate list/iterate functions from core apis? keep them for easiness?
|
|
R_API int r_debug_pid_list(RDebug *dbg, int pid, char fmt) {
|
|
RList *list;
|
|
RListIter *iter;
|
|
RDebugPid *p;
|
|
if (dbg && dbg->h && dbg->h->pids) {
|
|
list = dbg->h->pids (dbg, R_MAX (0, pid));
|
|
if (!list) {
|
|
return false;
|
|
}
|
|
PJ *j = pj_new ();
|
|
pj_a (j);
|
|
r_list_foreach (list, iter, p) {
|
|
switch (fmt) {
|
|
case 'j':
|
|
pj_o (j);
|
|
pj_kb (j, "current", dbg->pid == p->pid);
|
|
pj_ki (j, "ppid", p->ppid);
|
|
pj_ki (j, "pid", p->pid);
|
|
pj_ki (j, "uid", p->uid);
|
|
pj_ks (j, "status", &p->status);
|
|
pj_ks (j, "path", p->path);
|
|
pj_end (j);
|
|
break;
|
|
default:
|
|
dbg->cb_printf (" %c %d ppid:%d uid:%d %c %s\n",
|
|
dbg->pid == p->pid? '*': '-',
|
|
p->pid, p->ppid, p->uid, p->status, p->path);
|
|
break;
|
|
}
|
|
}
|
|
pj_end (j);
|
|
if (fmt == 'j') {
|
|
dbg->cb_printf (pj_string (j));
|
|
}
|
|
pj_free (j);
|
|
r_list_free (list);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
R_API int r_debug_thread_list(RDebug *dbg, int pid, char fmt) {
|
|
RList *list;
|
|
RListIter *iter;
|
|
RDebugPid *p;
|
|
RAnalFunction *fcn = NULL;
|
|
RDebugMap *map = NULL;
|
|
RStrBuf *path = NULL;
|
|
if (pid == -1) {
|
|
return false;
|
|
}
|
|
if (dbg && dbg->h && dbg->h->threads) {
|
|
list = dbg->h->threads (dbg, pid);
|
|
if (!list) {
|
|
return false;
|
|
}
|
|
PJ *j = pj_new ();
|
|
pj_a (j);
|
|
r_list_foreach (list, iter, p) {
|
|
path = r_strbuf_new ("");
|
|
if (p->pc != 0) {
|
|
map = r_debug_map_get (dbg, p->pc);
|
|
if (map && map->name && map->name[0]) {
|
|
r_strbuf_appendf (path, "%s ", map->name);
|
|
}
|
|
|
|
r_strbuf_appendf (path, "(0x%" PFMT64x ")", p->pc);
|
|
|
|
fcn = r_anal_get_fcn_in (dbg->anal, p->pc, 0);
|
|
if (fcn) {
|
|
r_strbuf_appendf (path, " in %s+0x%" PFMT64x, fcn->name, (p->pc - fcn->addr));
|
|
}
|
|
}
|
|
switch (fmt) {
|
|
case 'j':
|
|
pj_o (j);
|
|
pj_kb (j, "current", dbg->tid == p->pid);
|
|
pj_ki (j, "pid", p->pid);
|
|
pj_ks (j, "status", &p->status);
|
|
pj_ks (j, "path", r_strbuf_get (path));
|
|
pj_end (j);
|
|
break;
|
|
default:
|
|
dbg->cb_printf (" %c %d %c %s\n",
|
|
dbg->tid == p->pid? '*': '-',
|
|
p->pid, p->status, r_strbuf_get (path));
|
|
break;
|
|
}
|
|
r_strbuf_free (path);
|
|
}
|
|
pj_end (j);
|
|
if (fmt == 'j') {
|
|
dbg->cb_printf (pj_string (j));
|
|
}
|
|
pj_free (j);
|
|
r_list_free (list);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/* processes */
|
|
R_API int r_debug_pid_parent(RDebugPid *pid) {
|
|
// fork in child
|
|
return 0;
|
|
}
|
|
|
|
#if 0
|
|
R_API int r_debug_pid_del(struct r_debug_t *dbg) {
|
|
// kill da child
|
|
return true;
|
|
}
|
|
|
|
/* threads */
|
|
R_API int r_debug_pid_add_thread(struct r_debug_t *dbg) {
|
|
// create a thread in process
|
|
return true;
|
|
}
|
|
|
|
R_API int r_debug_pid_del_thread(struct r_debug_t *dbg) {
|
|
// kill a thread in process
|
|
return true;
|
|
}
|
|
#endif
|
|
|
|
/* status */
|
|
R_API int r_debug_pid_set_state(struct r_debug_t *dbg, int status) {
|
|
return true;
|
|
}
|
|
|
|
/* status */
|
|
R_API struct r_debug_pid_t *r_debug_pid_get_status(struct r_debug_t *dbg, int pid) {
|
|
return NULL;
|
|
}
|