/* radare - LGPL - Copyright 2014-2020 - pancake */ #include #define DB dbg->sgnls // TODO: this must be done by the debugger plugin // which is stored already in SDB.. but this is faster :P static struct { const char *k; const char *v; } signals[] = { // hardcoded from linux { "SIGHUP", "1" }, { "SIGINT", "2" }, { "SIGQUIT", "3" }, { "SIGILL", "4" }, { "SIGTRAP", "5" }, { "SIGABRT", "6" }, // { "SIGIOT", "6" }, { "SIGBUS", "7" }, { "SIGFPE", "8" }, { "SIGKILL", "9" }, { "SIGUSR1", "10" }, { "SIGSEGV", "11" }, { "SIGUSR2", "12" }, { "SIGPIPE", "13" }, { "SIGALRM", "14" }, { "SIGTERM", "15" }, { "SIGSTKFLT", "16" }, { "SIGCHLD", "17" }, { "SIGCONT", "18" }, { "SIGSTOP", "19" }, { "SIGTSTP", "20" }, { "SIGTTIN", "21" }, { "SIGTTOU", "22" }, { "SIGURG", "23" }, { "SIGXCPU", "24" }, { "SIGXFSZ", "25" }, { "SIGVTALRM", "26" }, { "SIGPROF", "27" }, { "SIGWINCH", "28" }, { "SIGIO", "29" }, { "SIGPOLL", "SIGIO" }, { "SIGLOST", "29" }, { "SIGPWR", "30" }, { "SIGSYS", "31" }, { "SIGRTMIN", "32" }, { "SIGRTMAX", "NSIG" }, { NULL } }; R_API void r_debug_signal_init(RDebug *dbg) { int i; // XXX DB = sdb_new (NULL, "signals", 0); for (i = 0; signals[i].k; i++) { sdb_set (DB, signals[i].k, signals[i].v, 0); sdb_set (DB, signals[i].v, signals[i].k, 0); } } static bool siglistcb(void *p, const char *k, const char *v) { RDebug *dbg = (RDebug *)p; if (atoi (k) > 0) { int mode = dbg->_mode; char *key = r_str_newf ("cfg.%s", k); int opt = sdb_num_get (DB, key, 0); free (key); if (opt) { r_cons_printf ("%s %s", k, v); if (opt & R_DBG_SIGNAL_CONT) { r_cons_strcat (" cont"); } if (opt & R_DBG_SIGNAL_SKIP) { r_cons_strcat (" skip"); } r_cons_newline (); } else { if (mode == 0) { r_cons_printf ("%s %s\n", k, v); } } } return true; } static bool siglistjsoncb(void *p, const char *k, const char *v) { static char key[32] = "cfg."; RDebug *dbg = (RDebug *)p; int opt; if (atoi (k) > 0) { strncpy (key + 4, k, 20); opt = (int)sdb_num_get (DB, key, 0); pj_o (dbg->pj); pj_ks (dbg->pj, "signum", k); pj_ks (dbg->pj, "name", v); pj_k (dbg->pj, "option"); if (opt & R_DBG_SIGNAL_CONT) { pj_s (dbg->pj, "cont"); } else if (opt & R_DBG_SIGNAL_SKIP) { pj_s (dbg->pj, "skip"); } else { pj_null (dbg->pj); } pj_end (dbg->pj); } return true; } R_API void r_debug_signal_list(RDebug *dbg, int mode) { dbg->_mode = mode; switch (mode) { case 0: case 1: sdb_foreach (DB, siglistcb, dbg); break; case 2: if (!dbg->pj) { return; } pj_a (dbg->pj); sdb_foreach (DB, siglistjsoncb, dbg); pj_end (dbg->pj); r_cons_println (pj_string (dbg->pj)); break; } dbg->_mode = 0; } R_API int r_debug_signal_send(RDebug *dbg, int num) { return r_sandbox_kill (dbg->pid, num); } R_API void r_debug_signal_setup(RDebug *dbg, int num, int opt) { sdb_queryf (DB, "cfg.%d=%d", num, opt); } R_API int r_debug_signal_what(RDebug *dbg, int num) { char k[32]; snprintf (k, sizeof (k), "cfg.%d", num); return sdb_num_get (DB, k, 0); } R_API int r_debug_signal_set(RDebug *dbg, int num, ut64 addr) { // TODO // r_debug_syscall (dbg, "signal", "addr"); return 0; } /* TODO rename to _kill_ -> _signal_ */ R_API RList *r_debug_kill_list(RDebug *dbg) { if (dbg->h->kill_list) { return dbg->h->kill_list (dbg); } return NULL; } R_API int r_debug_kill_setup(RDebug *dbg, int sig, int action) { eprintf ("TODO: set signal handlers of child\n"); // TODO: must inject code to call signal() #if 0 if (dbg->h->kill_setup) return dbg->h->kill_setup (dbg, sig, action); #endif // TODO: implement r_debug_kill_setup return false; }