#include <mdb/mdb_modapi.h>
#include <pthread.h>
#include <stddef.h>
#include <dlfcn.h>
#include <link.h>
#include <libproc.h>
#include <Python.h>
#include <frameobject.h>
#define PYDB_VERSION 1
typedef struct pydb_agent pydb_agent_t;
typedef struct pydb_iter pydb_iter_t;
typedef pydb_agent_t *(*pydb_agent_create_f)(struct ps_prochandle *P, int vers);
typedef void (*pydb_agent_destroy_f)(pydb_agent_t *py);
typedef int (*pydb_get_frameinfo_f)(pydb_agent_t *py, uintptr_t frame_addr,
char *fbuf, size_t bufsz, int verbose);
typedef pydb_iter_t *(*pydb_iter_init_f)(pydb_agent_t *py, uintptr_t addr);
typedef uintptr_t (*pydb_iter_next_f)(pydb_iter_t *iter);
typedef void (*pydb_iter_fini_f)(pydb_iter_t *iter);
static pydb_agent_create_f pydb_agent_create;
static pydb_agent_destroy_f pydb_agent_destroy;
static pydb_get_frameinfo_f pydb_get_frameinfo;
static pydb_iter_init_f pydb_frame_iter_init;
static pydb_iter_init_f pydb_interp_iter_init;
static pydb_iter_init_f pydb_thread_iter_init;
static pydb_iter_next_f pydb_iter_next;
static pydb_iter_fini_f pydb_iter_fini;
static pydb_agent_t *pydb_hdl = NULL;
static void *pydb_dlhdl = NULL;
static int
py_frame(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
char buf[1024];
int verbose = FALSE;
if (mdb_getopts(argc, argv,
'v', MDB_OPT_SETBITS, TRUE, &verbose,
NULL) != argc) {
return (DCMD_USAGE);
}
if (flags & DCMD_PIPE_OUT) {
mdb_warn("py_stack cannot output into a pipe\n");
return (DCMD_ERR);
}
if (!(flags & DCMD_ADDRSPEC)) {
mdb_warn("no address");
return (DCMD_USAGE);
}
if (pydb_get_frameinfo(pydb_hdl, addr, buf, sizeof (buf),
verbose) < 0) {
mdb_warn("Unable to find frame at address %p\n", addr);
return (DCMD_ERR);
}
mdb_printf("%s", buf);
return (DCMD_OK);
}
int
py_interp_walk_init(mdb_walk_state_t *wsp)
{
pydb_iter_t *pdi;
pdi = pydb_interp_iter_init(pydb_hdl, wsp->walk_addr);
if (pdi == NULL) {
mdb_warn("unable to create interpreter iterator\n");
return (DCMD_ERR);
}
wsp->walk_data = pdi;
return (WALK_NEXT);
}
int
py_walk_step(mdb_walk_state_t *wsp)
{
pydb_iter_t *pdi = wsp->walk_data;
uintptr_t addr;
int status;
addr = pydb_iter_next(pdi);
if (addr == 0) {
return (WALK_DONE);
}
status = wsp->walk_callback(addr, 0, wsp->walk_cbdata);
return (status);
}
void
py_walk_fini(mdb_walk_state_t *wsp)
{
pydb_iter_t *pdi = wsp->walk_data;
pydb_iter_fini(pdi);
}
int
py_thread_walk_init(mdb_walk_state_t *wsp)
{
pydb_iter_t *pdi;
pdi = pydb_thread_iter_init(pydb_hdl, wsp->walk_addr);
if (pdi == NULL) {
mdb_warn("unable to create thread iterator\n");
return (DCMD_ERR);
}
wsp->walk_data = pdi;
return (WALK_NEXT);
}
int
py_frame_walk_init(mdb_walk_state_t *wsp)
{
pydb_iter_t *pdi;
pdi = pydb_frame_iter_init(pydb_hdl, wsp->walk_addr);
if (pdi == NULL) {
mdb_warn("unable to create frame iterator\n");
return (DCMD_ERR);
}
wsp->walk_data = pdi;
return (WALK_NEXT);
}
static int
python_stack(uintptr_t addr, const PyThreadState *ts, uint_t *verbose)
{
mdb_arg_t nargv;
uint_t nargc = (verbose != NULL && *verbose) ? 1 : 0;
mdb_printf("PyThreadState: %0?p\n", addr);
nargv.a_type = MDB_TYPE_STRING;
nargv.a_un.a_str = "-v";
if (mdb_pwalk_dcmd("pyframe", "pyframe", nargc, &nargv, addr) == -1) {
mdb_warn("can't walk 'pyframe'");
return (WALK_ERR);
}
return (WALK_NEXT);
}
static int
python_thread(uintptr_t addr, const PyInterpreterState *is, uint_t *verbose)
{
if (mdb_pwalk("pythread", (mdb_walk_cb_t)python_stack, verbose,
addr) == -1) {
mdb_warn("can't walk 'pythread'");
return (WALK_ERR);
}
return (WALK_NEXT);
}
static int
py_stack(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
{
uint_t verbose = FALSE;
if (mdb_getopts(argc, argv,
'v', MDB_OPT_SETBITS, TRUE, &verbose,
NULL) != argc)
return (DCMD_USAGE);
if (flags & DCMD_PIPE_OUT) {
mdb_warn("py_stack cannot output into a pipe\n");
return (DCMD_ERR);
}
if (flags & DCMD_ADDRSPEC) {
mdb_arg_t nargv;
uint_t nargc = verbose ? 1 : 0;
nargv.a_type = MDB_TYPE_STRING;
nargv.a_un.a_str = "-v";
if (mdb_pwalk_dcmd("pyframe", "pyframe", nargc, &nargv, addr)
== -1) {
mdb_warn("can't walk 'pyframe'");
return (DCMD_ERR);
}
return (DCMD_OK);
}
if (mdb_walk("pyinterp", (mdb_walk_cb_t)python_thread,
&verbose) == -1) {
mdb_warn("can't walk 'pyinterp'");
return (DCMD_ERR);
}
return (DCMD_OK);
}
static const mdb_dcmd_t dcmds[] = {
{ "pystack", "[-v]", "print python stacks", py_stack },
{ "pyframe", "[-v]", "print python frames", py_frame },
{ NULL }
};
static const mdb_walker_t walkers[] = {
{ "pyinterp", "walk python interpreter structures",
py_interp_walk_init, py_walk_step, py_walk_fini },
{ "pythread", "given an interpreter, walk the list of python threads",
py_thread_walk_init, py_walk_step, py_walk_fini },
{ "pyframe", "given a thread state, walk the list of frame objects",
py_frame_walk_init, py_walk_step, py_walk_fini },
{ NULL }
};
static const mdb_modinfo_t modinfo = {
MDB_API_VERSION, dcmds, walkers
};
static int
python_object_iter(void *cd, const prmap_t *pmp, const char *obj)
{
char path[PATH_MAX];
char *name;
char *s1, *s2;
struct ps_prochandle *Pr = cd;
name = strstr(obj, "/libpython");
if (name) {
(void) strcpy(path, obj);
if (Pstatus(Pr)->pr_dmodel != PR_MODEL_NATIVE) {
s1 = name;
s2 = path + (s1 - obj);
(void) strcpy(s2, "/64");
s2 += 3;
(void) strcpy(s2, s1);
}
s1 = strstr(obj, ".so");
s2 = strstr(path, ".so");
(void) strcpy(s2, "_db");
s2 += 3;
(void) strcpy(s2, s1);
if ((pydb_dlhdl = dlopen(path, RTLD_LAZY|RTLD_GLOBAL)) != NULL)
return (1);
}
return (0);
}
static int
python_db_init(void)
{
struct ps_prochandle *Ph;
if (mdb_get_xdata("pshandle", &Ph, sizeof (Ph)) == -1) {
mdb_warn("couldn't read pshandle xdata\n");
dlclose(pydb_dlhdl);
pydb_dlhdl = NULL;
return (-1);
}
(void) Pobject_iter(Ph, python_object_iter, Ph);
pydb_agent_create = (pydb_agent_create_f)
dlsym(pydb_dlhdl, "pydb_agent_create");
pydb_agent_destroy = (pydb_agent_destroy_f)
dlsym(pydb_dlhdl, "pydb_agent_destroy");
pydb_get_frameinfo = (pydb_get_frameinfo_f)
dlsym(pydb_dlhdl, "pydb_get_frameinfo");
pydb_frame_iter_init = (pydb_iter_init_f)
dlsym(pydb_dlhdl, "pydb_frame_iter_init");
pydb_interp_iter_init = (pydb_iter_init_f)
dlsym(pydb_dlhdl, "pydb_interp_iter_init");
pydb_thread_iter_init = (pydb_iter_init_f)
dlsym(pydb_dlhdl, "pydb_thread_iter_init");
pydb_iter_next = (pydb_iter_next_f)dlsym(pydb_dlhdl, "pydb_iter_next");
pydb_iter_fini = (pydb_iter_fini_f)dlsym(pydb_dlhdl, "pydb_iter_fini");
if (pydb_agent_create == NULL || pydb_agent_destroy == NULL ||
pydb_get_frameinfo == NULL || pydb_frame_iter_init == NULL ||
pydb_interp_iter_init == NULL || pydb_thread_iter_init == NULL ||
pydb_iter_next == NULL || pydb_iter_fini == NULL) {
mdb_warn("couldn't load pydb functions");
dlclose(pydb_dlhdl);
pydb_dlhdl = NULL;
return (-1);
}
pydb_hdl = pydb_agent_create(Ph, PYDB_VERSION);
if (pydb_hdl == NULL) {
mdb_warn("unable to create pydb_agent");
dlclose(pydb_dlhdl);
pydb_dlhdl = NULL;
return (-1);
}
return (0);
}
static void
python_db_fini(void)
{
if (pydb_dlhdl) {
pydb_agent_destroy(pydb_hdl);
pydb_hdl = NULL;
dlclose(pydb_dlhdl);
pydb_dlhdl = NULL;
}
}
const mdb_modinfo_t *
_mdb_init(void)
{
if (python_db_init() != 0)
return (NULL);
return (&modinfo);
}
void
_mdb_fini(void)
{
python_db_fini();
}