xref: /titanic_53/usr/src/cmd/mdb/common/modules/libpython/libpython.c (revision 2fb4439d628ad2df0775287be1abd1ed95e7d267)
1*2fb4439dSAlexander Pyhalov /*
2*2fb4439dSAlexander Pyhalov  * CDDL HEADER START
3*2fb4439dSAlexander Pyhalov  *
4*2fb4439dSAlexander Pyhalov  * The contents of this file are subject to the terms of the
5*2fb4439dSAlexander Pyhalov  * Common Development and Distribution License (the "License").
6*2fb4439dSAlexander Pyhalov  * You may not use this file except in compliance with the License.
7*2fb4439dSAlexander Pyhalov  *
8*2fb4439dSAlexander Pyhalov  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9*2fb4439dSAlexander Pyhalov  * or http://www.opensolaris.org/os/licensing.
10*2fb4439dSAlexander Pyhalov  * See the License for the specific language governing permissions
11*2fb4439dSAlexander Pyhalov  * and limitations under the License.
12*2fb4439dSAlexander Pyhalov  *
13*2fb4439dSAlexander Pyhalov  * When distributing Covered Code, include this CDDL HEADER in each
14*2fb4439dSAlexander Pyhalov  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15*2fb4439dSAlexander Pyhalov  * If applicable, add the following below this CDDL HEADER, with the
16*2fb4439dSAlexander Pyhalov  * fields enclosed by brackets "[]" replaced with your own identifying
17*2fb4439dSAlexander Pyhalov  * information: Portions Copyright [yyyy] [name of copyright owner]
18*2fb4439dSAlexander Pyhalov  *
19*2fb4439dSAlexander Pyhalov  * CDDL HEADER END
20*2fb4439dSAlexander Pyhalov  */
21*2fb4439dSAlexander Pyhalov /*
22*2fb4439dSAlexander Pyhalov  * Copyright 2010 Sun Microsystems, Inc.  All rights reserved.
23*2fb4439dSAlexander Pyhalov  * Use is subject to license terms.
24*2fb4439dSAlexander Pyhalov  */
25*2fb4439dSAlexander Pyhalov 
26*2fb4439dSAlexander Pyhalov #include <mdb/mdb_modapi.h>
27*2fb4439dSAlexander Pyhalov 
28*2fb4439dSAlexander Pyhalov #include <pthread.h>
29*2fb4439dSAlexander Pyhalov #include <stddef.h>
30*2fb4439dSAlexander Pyhalov #include <dlfcn.h>
31*2fb4439dSAlexander Pyhalov #include <link.h>
32*2fb4439dSAlexander Pyhalov #include <libproc.h>
33*2fb4439dSAlexander Pyhalov 
34*2fb4439dSAlexander Pyhalov #include <Python.h>
35*2fb4439dSAlexander Pyhalov #include <frameobject.h>
36*2fb4439dSAlexander Pyhalov 
37*2fb4439dSAlexander Pyhalov /*
38*2fb4439dSAlexander Pyhalov  * Decoding Python Stack Frames
39*2fb4439dSAlexander Pyhalov  * ============================
40*2fb4439dSAlexander Pyhalov  *
41*2fb4439dSAlexander Pyhalov  * Python2 uses a variety of objects to construct its call chain.  An address
42*2fb4439dSAlexander Pyhalov  * space may have one or more PyInterpreterState objects, which are the base
43*2fb4439dSAlexander Pyhalov  * object in the interpreter's state.  These objects are kept in a linked list
44*2fb4439dSAlexander Pyhalov  * with a head pointer named interp_head.  This makes it possible for the
45*2fb4439dSAlexander Pyhalov  * debugger to get a toehold on data structures necessary to understand the
46*2fb4439dSAlexander Pyhalov  * interpreter.  Since most of these structures are linked out of the
47*2fb4439dSAlexander Pyhalov  * InterpreterState, traversals generally start here.
48*2fb4439dSAlexander Pyhalov  *
49*2fb4439dSAlexander Pyhalov  * In order to decode a frame, the debugger needs to walk from
50*2fb4439dSAlexander Pyhalov  * PyInterpreterState down to a PyCodeObject.  The diagram below shows the
51*2fb4439dSAlexander Pyhalov  * the objects that must be examined in order to reach a leaf PyCodeObject.
52*2fb4439dSAlexander Pyhalov  *
53*2fb4439dSAlexander Pyhalov  *                +--------------------+ next   +--------------------+ next
54*2fb4439dSAlexander Pyhalov  * interp_head -> | PyInterpreterState | ---->  | PyInterpreterState | ---> ...
55*2fb4439dSAlexander Pyhalov  *                +--------------------+        +--------------------+
56*2fb4439dSAlexander Pyhalov  *                  |                            | tstate_head
57*2fb4439dSAlexander Pyhalov  *                  | tstate_head                V
58*2fb4439dSAlexander Pyhalov  *                  |                 +---------------+  frame
59*2fb4439dSAlexander Pyhalov  *                  V                 | PyThreadState | -----> ...
60*2fb4439dSAlexander Pyhalov  *  +---------------+  frame          +---------------+
61*2fb4439dSAlexander Pyhalov  *  | PyThreadState |  ---> ...
62*2fb4439dSAlexander Pyhalov  *  +---------------+
63*2fb4439dSAlexander Pyhalov  *          | next
64*2fb4439dSAlexander Pyhalov  *          V
65*2fb4439dSAlexander Pyhalov  *  +---------------+  frame    +---------------+ f_back +---------------+
66*2fb4439dSAlexander Pyhalov  *  | PyThreadState |  ------>  | PyFrameObject | -----> | PyFrameObject |
67*2fb4439dSAlexander Pyhalov  *  +---------------+           +---------------+        +---------------+
68*2fb4439dSAlexander Pyhalov  *                                      |                       |
69*2fb4439dSAlexander Pyhalov  *                                      | f_code                | f_code
70*2fb4439dSAlexander Pyhalov  *                                      V                       V
71*2fb4439dSAlexander Pyhalov  *                              +--------------+               ...
72*2fb4439dSAlexander Pyhalov  *                              | PyCodeObject |
73*2fb4439dSAlexander Pyhalov  *                              +--------------+
74*2fb4439dSAlexander Pyhalov  *                 co_filename   |      |     | co_lnotab
75*2fb4439dSAlexander Pyhalov  *                 +-------------+      |     +-------------+
76*2fb4439dSAlexander Pyhalov  *                 |           co_name  |                   |
77*2fb4439dSAlexander Pyhalov  *                 V                    V                   V
78*2fb4439dSAlexander Pyhalov  * +----------------+          +----------------+         +----------------+
79*2fb4439dSAlexander Pyhalov  * | PyStringObject |          | PyStringObject |         | PyStringObject |
80*2fb4439dSAlexander Pyhalov  * +----------------+          +----------------+         +----------------+
81*2fb4439dSAlexander Pyhalov  *
82*2fb4439dSAlexander Pyhalov  * The interp_head pointer is a list of one or more PyInterpreterState
83*2fb4439dSAlexander Pyhalov  * objects.  Each of these objects can contain one or more PyThreadState
84*2fb4439dSAlexander Pyhalov  * objects.  The PyInterpreterState object keeps a pointer to the head of the
85*2fb4439dSAlexander Pyhalov  * list of PyThreadState objects as tstate_head.
86*2fb4439dSAlexander Pyhalov  *
87*2fb4439dSAlexander Pyhalov  * Each thread keeps ahold of its stack frames.  The PyThreadState object
88*2fb4439dSAlexander Pyhalov  * has a pointer to the topmost PyFrameObject, kept in frame.  The
89*2fb4439dSAlexander Pyhalov  * successive frames on the stack are kept linked in the PyFrameObject's
90*2fb4439dSAlexander Pyhalov  * f_back pointer, with each frame pointing to its caller.
91*2fb4439dSAlexander Pyhalov  *
92*2fb4439dSAlexander Pyhalov  * In order to decode each call frame, our code needs to look at the
93*2fb4439dSAlexander Pyhalov  * PyCodeObject for each frame.  Essentially, this is the code that is
94*2fb4439dSAlexander Pyhalov  * being executed in the frame.  The PyFrameObject keeps a pointer to this
95*2fb4439dSAlexander Pyhalov  * code object in f_code.  In order to print meaningful debug information,
96*2fb4439dSAlexander Pyhalov  * it's necessary to extract the Python filename (co_filename), the
97*2fb4439dSAlexander Pyhalov  * function name (co_name), and the line number within the file
98*2fb4439dSAlexander Pyhalov  * (co_lnotab).  The filename and function are stored as strings, but the
99*2fb4439dSAlexander Pyhalov  * line number is a mapping of bytecode offsets to line numbers.  The
100*2fb4439dSAlexander Pyhalov  * description of the lnotab algorithm lives here:
101*2fb4439dSAlexander Pyhalov  *
102*2fb4439dSAlexander Pyhalov  * http://svn.python.org/projects/python/trunk/Objects/lnotab_notes.txt
103*2fb4439dSAlexander Pyhalov  *
104*2fb4439dSAlexander Pyhalov  * In order to decode the frame, the debugger needs to walk each
105*2fb4439dSAlexander Pyhalov  * InterpreterState object.  For each InterpreterState, every PyThreadState
106*2fb4439dSAlexander Pyhalov  * must be traversed.  The PyThreadState objects point to the
107*2fb4439dSAlexander Pyhalov  * PyFrameObjects.  For every thread, we must walk the frames backwards and
108*2fb4439dSAlexander Pyhalov  * decode the strings that are in the PyCodeObjects.
109*2fb4439dSAlexander Pyhalov  */
110*2fb4439dSAlexander Pyhalov 
111*2fb4439dSAlexander Pyhalov /*
112*2fb4439dSAlexander Pyhalov  * The Python-dependent debugging functionality lives in its own helper
113*2fb4439dSAlexander Pyhalov  * library.  The helper agent is provided by libpython2.[67]_db.so, which
114*2fb4439dSAlexander Pyhalov  * is also used by pstack(1) for debugging Python processes.
115*2fb4439dSAlexander Pyhalov  *
116*2fb4439dSAlexander Pyhalov  * Define needed prototypes here.
117*2fb4439dSAlexander Pyhalov  */
118*2fb4439dSAlexander Pyhalov 
119*2fb4439dSAlexander Pyhalov #define	PYDB_VERSION	1
120*2fb4439dSAlexander Pyhalov typedef struct pydb_agent pydb_agent_t;
121*2fb4439dSAlexander Pyhalov typedef struct pydb_iter pydb_iter_t;
122*2fb4439dSAlexander Pyhalov 
123*2fb4439dSAlexander Pyhalov typedef pydb_agent_t *(*pydb_agent_create_f)(struct ps_prochandle *P, int vers);
124*2fb4439dSAlexander Pyhalov typedef void (*pydb_agent_destroy_f)(pydb_agent_t *py);
125*2fb4439dSAlexander Pyhalov typedef int (*pydb_get_frameinfo_f)(pydb_agent_t *py, uintptr_t frame_addr,
126*2fb4439dSAlexander Pyhalov     char *fbuf, size_t bufsz, int verbose);
127*2fb4439dSAlexander Pyhalov typedef pydb_iter_t *(*pydb_iter_init_f)(pydb_agent_t *py, uintptr_t addr);
128*2fb4439dSAlexander Pyhalov typedef uintptr_t (*pydb_iter_next_f)(pydb_iter_t *iter);
129*2fb4439dSAlexander Pyhalov typedef void (*pydb_iter_fini_f)(pydb_iter_t *iter);
130*2fb4439dSAlexander Pyhalov 
131*2fb4439dSAlexander Pyhalov static pydb_agent_create_f pydb_agent_create;
132*2fb4439dSAlexander Pyhalov static pydb_agent_destroy_f pydb_agent_destroy;
133*2fb4439dSAlexander Pyhalov static pydb_get_frameinfo_f pydb_get_frameinfo;
134*2fb4439dSAlexander Pyhalov static pydb_iter_init_f pydb_frame_iter_init;
135*2fb4439dSAlexander Pyhalov static pydb_iter_init_f pydb_interp_iter_init;
136*2fb4439dSAlexander Pyhalov static pydb_iter_init_f pydb_thread_iter_init;
137*2fb4439dSAlexander Pyhalov static pydb_iter_next_f pydb_iter_next;
138*2fb4439dSAlexander Pyhalov static pydb_iter_fini_f pydb_iter_fini;
139*2fb4439dSAlexander Pyhalov 
140*2fb4439dSAlexander Pyhalov static pydb_agent_t *pydb_hdl = NULL;
141*2fb4439dSAlexander Pyhalov static void *pydb_dlhdl = NULL;
142*2fb4439dSAlexander Pyhalov 
143*2fb4439dSAlexander Pyhalov /*ARGSUSED*/
144*2fb4439dSAlexander Pyhalov static int
py_frame(uintptr_t addr,uint_t flags,int argc,const mdb_arg_t * argv)145*2fb4439dSAlexander Pyhalov py_frame(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
146*2fb4439dSAlexander Pyhalov {
147*2fb4439dSAlexander Pyhalov 	char buf[1024];
148*2fb4439dSAlexander Pyhalov 	int verbose = FALSE;
149*2fb4439dSAlexander Pyhalov 
150*2fb4439dSAlexander Pyhalov 	if (mdb_getopts(argc, argv,
151*2fb4439dSAlexander Pyhalov 	    'v', MDB_OPT_SETBITS, TRUE, &verbose,
152*2fb4439dSAlexander Pyhalov 	    NULL) != argc) {
153*2fb4439dSAlexander Pyhalov 		return (DCMD_USAGE);
154*2fb4439dSAlexander Pyhalov 	}
155*2fb4439dSAlexander Pyhalov 
156*2fb4439dSAlexander Pyhalov 	if (flags & DCMD_PIPE_OUT) {
157*2fb4439dSAlexander Pyhalov 		mdb_warn("py_stack cannot output into a pipe\n");
158*2fb4439dSAlexander Pyhalov 		return (DCMD_ERR);
159*2fb4439dSAlexander Pyhalov 	}
160*2fb4439dSAlexander Pyhalov 
161*2fb4439dSAlexander Pyhalov 	if (!(flags & DCMD_ADDRSPEC)) {
162*2fb4439dSAlexander Pyhalov 		mdb_warn("no address");
163*2fb4439dSAlexander Pyhalov 		return (DCMD_USAGE);
164*2fb4439dSAlexander Pyhalov 	}
165*2fb4439dSAlexander Pyhalov 
166*2fb4439dSAlexander Pyhalov 	if (pydb_get_frameinfo(pydb_hdl, addr, buf, sizeof (buf),
167*2fb4439dSAlexander Pyhalov 	    verbose) < 0) {
168*2fb4439dSAlexander Pyhalov 		mdb_warn("Unable to find frame at address %p\n", addr);
169*2fb4439dSAlexander Pyhalov 		return (DCMD_ERR);
170*2fb4439dSAlexander Pyhalov 	}
171*2fb4439dSAlexander Pyhalov 
172*2fb4439dSAlexander Pyhalov 	mdb_printf("%s", buf);
173*2fb4439dSAlexander Pyhalov 
174*2fb4439dSAlexander Pyhalov 	return (DCMD_OK);
175*2fb4439dSAlexander Pyhalov }
176*2fb4439dSAlexander Pyhalov 
177*2fb4439dSAlexander Pyhalov int
py_interp_walk_init(mdb_walk_state_t * wsp)178*2fb4439dSAlexander Pyhalov py_interp_walk_init(mdb_walk_state_t *wsp)
179*2fb4439dSAlexander Pyhalov {
180*2fb4439dSAlexander Pyhalov 	pydb_iter_t *pdi;
181*2fb4439dSAlexander Pyhalov 
182*2fb4439dSAlexander Pyhalov 	pdi = pydb_interp_iter_init(pydb_hdl, wsp->walk_addr);
183*2fb4439dSAlexander Pyhalov 
184*2fb4439dSAlexander Pyhalov 	if (pdi == NULL) {
185*2fb4439dSAlexander Pyhalov 		mdb_warn("unable to create interpreter iterator\n");
186*2fb4439dSAlexander Pyhalov 		return (DCMD_ERR);
187*2fb4439dSAlexander Pyhalov 	}
188*2fb4439dSAlexander Pyhalov 
189*2fb4439dSAlexander Pyhalov 	wsp->walk_data = pdi;
190*2fb4439dSAlexander Pyhalov 
191*2fb4439dSAlexander Pyhalov 	return (WALK_NEXT);
192*2fb4439dSAlexander Pyhalov }
193*2fb4439dSAlexander Pyhalov 
194*2fb4439dSAlexander Pyhalov int
py_walk_step(mdb_walk_state_t * wsp)195*2fb4439dSAlexander Pyhalov py_walk_step(mdb_walk_state_t *wsp)
196*2fb4439dSAlexander Pyhalov {
197*2fb4439dSAlexander Pyhalov 	pydb_iter_t *pdi = wsp->walk_data;
198*2fb4439dSAlexander Pyhalov 	uintptr_t addr;
199*2fb4439dSAlexander Pyhalov 	int status;
200*2fb4439dSAlexander Pyhalov 
201*2fb4439dSAlexander Pyhalov 	addr = pydb_iter_next(pdi);
202*2fb4439dSAlexander Pyhalov 
203*2fb4439dSAlexander Pyhalov 	if (addr == NULL) {
204*2fb4439dSAlexander Pyhalov 		return (WALK_DONE);
205*2fb4439dSAlexander Pyhalov 	}
206*2fb4439dSAlexander Pyhalov 
207*2fb4439dSAlexander Pyhalov 	status = wsp->walk_callback(addr, 0, wsp->walk_cbdata);
208*2fb4439dSAlexander Pyhalov 
209*2fb4439dSAlexander Pyhalov 	return (status);
210*2fb4439dSAlexander Pyhalov }
211*2fb4439dSAlexander Pyhalov 
212*2fb4439dSAlexander Pyhalov void
py_walk_fini(mdb_walk_state_t * wsp)213*2fb4439dSAlexander Pyhalov py_walk_fini(mdb_walk_state_t *wsp)
214*2fb4439dSAlexander Pyhalov {
215*2fb4439dSAlexander Pyhalov 	pydb_iter_t *pdi = wsp->walk_data;
216*2fb4439dSAlexander Pyhalov 	pydb_iter_fini(pdi);
217*2fb4439dSAlexander Pyhalov }
218*2fb4439dSAlexander Pyhalov 
219*2fb4439dSAlexander Pyhalov int
py_thread_walk_init(mdb_walk_state_t * wsp)220*2fb4439dSAlexander Pyhalov py_thread_walk_init(mdb_walk_state_t *wsp)
221*2fb4439dSAlexander Pyhalov {
222*2fb4439dSAlexander Pyhalov 	pydb_iter_t *pdi;
223*2fb4439dSAlexander Pyhalov 
224*2fb4439dSAlexander Pyhalov 	pdi = pydb_thread_iter_init(pydb_hdl, wsp->walk_addr);
225*2fb4439dSAlexander Pyhalov 	if (pdi == NULL) {
226*2fb4439dSAlexander Pyhalov 		mdb_warn("unable to create thread iterator\n");
227*2fb4439dSAlexander Pyhalov 		return (DCMD_ERR);
228*2fb4439dSAlexander Pyhalov 	}
229*2fb4439dSAlexander Pyhalov 
230*2fb4439dSAlexander Pyhalov 	wsp->walk_data = pdi;
231*2fb4439dSAlexander Pyhalov 
232*2fb4439dSAlexander Pyhalov 	return (WALK_NEXT);
233*2fb4439dSAlexander Pyhalov }
234*2fb4439dSAlexander Pyhalov 
235*2fb4439dSAlexander Pyhalov int
py_frame_walk_init(mdb_walk_state_t * wsp)236*2fb4439dSAlexander Pyhalov py_frame_walk_init(mdb_walk_state_t *wsp)
237*2fb4439dSAlexander Pyhalov {
238*2fb4439dSAlexander Pyhalov 	pydb_iter_t *pdi;
239*2fb4439dSAlexander Pyhalov 
240*2fb4439dSAlexander Pyhalov 	pdi = pydb_frame_iter_init(pydb_hdl, wsp->walk_addr);
241*2fb4439dSAlexander Pyhalov 	if (pdi == NULL) {
242*2fb4439dSAlexander Pyhalov 		mdb_warn("unable to create frame iterator\n");
243*2fb4439dSAlexander Pyhalov 		return (DCMD_ERR);
244*2fb4439dSAlexander Pyhalov 	}
245*2fb4439dSAlexander Pyhalov 
246*2fb4439dSAlexander Pyhalov 	wsp->walk_data = pdi;
247*2fb4439dSAlexander Pyhalov 
248*2fb4439dSAlexander Pyhalov 	return (WALK_NEXT);
249*2fb4439dSAlexander Pyhalov }
250*2fb4439dSAlexander Pyhalov 
251*2fb4439dSAlexander Pyhalov /*ARGSUSED*/
252*2fb4439dSAlexander Pyhalov static int
python_stack(uintptr_t addr,const PyThreadState * ts,uint_t * verbose)253*2fb4439dSAlexander Pyhalov python_stack(uintptr_t addr, const PyThreadState *ts, uint_t *verbose)
254*2fb4439dSAlexander Pyhalov {
255*2fb4439dSAlexander Pyhalov 	mdb_arg_t nargv;
256*2fb4439dSAlexander Pyhalov 	uint_t nargc = (verbose != NULL && *verbose) ? 1 : 0;
257*2fb4439dSAlexander Pyhalov 	/*
258*2fb4439dSAlexander Pyhalov 	 * Pass the ThreadState to the frame walker. Have frame walker
259*2fb4439dSAlexander Pyhalov 	 * call frame dcmd.
260*2fb4439dSAlexander Pyhalov 	 */
261*2fb4439dSAlexander Pyhalov 	mdb_printf("PyThreadState: %0?p\n", addr);
262*2fb4439dSAlexander Pyhalov 
263*2fb4439dSAlexander Pyhalov 	nargv.a_type = MDB_TYPE_STRING;
264*2fb4439dSAlexander Pyhalov 	nargv.a_un.a_str = "-v";
265*2fb4439dSAlexander Pyhalov 
266*2fb4439dSAlexander Pyhalov 	if (mdb_pwalk_dcmd("pyframe", "pyframe", nargc, &nargv, addr) == -1) {
267*2fb4439dSAlexander Pyhalov 		mdb_warn("can't walk 'pyframe'");
268*2fb4439dSAlexander Pyhalov 		return (WALK_ERR);
269*2fb4439dSAlexander Pyhalov 	}
270*2fb4439dSAlexander Pyhalov 
271*2fb4439dSAlexander Pyhalov 	return (WALK_NEXT);
272*2fb4439dSAlexander Pyhalov }
273*2fb4439dSAlexander Pyhalov 
274*2fb4439dSAlexander Pyhalov /*ARGSUSED*/
275*2fb4439dSAlexander Pyhalov static int
python_thread(uintptr_t addr,const PyInterpreterState * is,uint_t * verbose)276*2fb4439dSAlexander Pyhalov python_thread(uintptr_t addr, const PyInterpreterState *is, uint_t *verbose)
277*2fb4439dSAlexander Pyhalov {
278*2fb4439dSAlexander Pyhalov 	/*
279*2fb4439dSAlexander Pyhalov 	 * Pass the InterpreterState to the threadstate walker.
280*2fb4439dSAlexander Pyhalov 	 */
281*2fb4439dSAlexander Pyhalov 	if (mdb_pwalk("pythread", (mdb_walk_cb_t)python_stack, verbose,
282*2fb4439dSAlexander Pyhalov 	    addr) == -1) {
283*2fb4439dSAlexander Pyhalov 		mdb_warn("can't walk 'pythread'");
284*2fb4439dSAlexander Pyhalov 		return (WALK_ERR);
285*2fb4439dSAlexander Pyhalov 	}
286*2fb4439dSAlexander Pyhalov 
287*2fb4439dSAlexander Pyhalov 	return (WALK_NEXT);
288*2fb4439dSAlexander Pyhalov }
289*2fb4439dSAlexander Pyhalov 
290*2fb4439dSAlexander Pyhalov /*ARGSUSED*/
291*2fb4439dSAlexander Pyhalov static int
py_stack(uintptr_t addr,uint_t flags,int argc,const mdb_arg_t * argv)292*2fb4439dSAlexander Pyhalov py_stack(uintptr_t addr, uint_t flags, int argc, const mdb_arg_t *argv)
293*2fb4439dSAlexander Pyhalov {
294*2fb4439dSAlexander Pyhalov 	uint_t verbose = FALSE;
295*2fb4439dSAlexander Pyhalov 
296*2fb4439dSAlexander Pyhalov 	if (mdb_getopts(argc, argv,
297*2fb4439dSAlexander Pyhalov 	    'v', MDB_OPT_SETBITS, TRUE, &verbose,
298*2fb4439dSAlexander Pyhalov 	    NULL) != argc)
299*2fb4439dSAlexander Pyhalov 		return (DCMD_USAGE);
300*2fb4439dSAlexander Pyhalov 
301*2fb4439dSAlexander Pyhalov 	if (flags & DCMD_PIPE_OUT) {
302*2fb4439dSAlexander Pyhalov 		mdb_warn("py_stack cannot output into a pipe\n");
303*2fb4439dSAlexander Pyhalov 		return (DCMD_ERR);
304*2fb4439dSAlexander Pyhalov 	}
305*2fb4439dSAlexander Pyhalov 
306*2fb4439dSAlexander Pyhalov 	if (flags & DCMD_ADDRSPEC) {
307*2fb4439dSAlexander Pyhalov 		mdb_arg_t nargv;
308*2fb4439dSAlexander Pyhalov 		uint_t nargc = verbose ? 1 : 0;
309*2fb4439dSAlexander Pyhalov 
310*2fb4439dSAlexander Pyhalov 		nargv.a_type = MDB_TYPE_STRING;
311*2fb4439dSAlexander Pyhalov 		nargv.a_un.a_str = "-v";
312*2fb4439dSAlexander Pyhalov 
313*2fb4439dSAlexander Pyhalov 		if (mdb_pwalk_dcmd("pyframe", "pyframe", nargc, &nargv, addr)
314*2fb4439dSAlexander Pyhalov 		    == -1) {
315*2fb4439dSAlexander Pyhalov 			mdb_warn("can't walk 'pyframe'");
316*2fb4439dSAlexander Pyhalov 			return (DCMD_ERR);
317*2fb4439dSAlexander Pyhalov 		}
318*2fb4439dSAlexander Pyhalov 		return (DCMD_OK);
319*2fb4439dSAlexander Pyhalov 	}
320*2fb4439dSAlexander Pyhalov 
321*2fb4439dSAlexander Pyhalov 	if (mdb_walk("pyinterp", (mdb_walk_cb_t)python_thread,
322*2fb4439dSAlexander Pyhalov 	    &verbose) == -1) {
323*2fb4439dSAlexander Pyhalov 		mdb_warn("can't walk 'pyinterp'");
324*2fb4439dSAlexander Pyhalov 		return (DCMD_ERR);
325*2fb4439dSAlexander Pyhalov 	}
326*2fb4439dSAlexander Pyhalov 
327*2fb4439dSAlexander Pyhalov 	return (DCMD_OK);
328*2fb4439dSAlexander Pyhalov }
329*2fb4439dSAlexander Pyhalov 
330*2fb4439dSAlexander Pyhalov static const mdb_dcmd_t dcmds[] = {
331*2fb4439dSAlexander Pyhalov 	{ "pystack", "[-v]", "print python stacks", py_stack },
332*2fb4439dSAlexander Pyhalov 	{ "pyframe", "[-v]", "print python frames", py_frame },
333*2fb4439dSAlexander Pyhalov 	{ NULL }
334*2fb4439dSAlexander Pyhalov };
335*2fb4439dSAlexander Pyhalov 
336*2fb4439dSAlexander Pyhalov static const mdb_walker_t walkers[] = {
337*2fb4439dSAlexander Pyhalov 	{ "pyinterp", "walk python interpreter structures",
338*2fb4439dSAlexander Pyhalov 		py_interp_walk_init, py_walk_step, py_walk_fini },
339*2fb4439dSAlexander Pyhalov 	{ "pythread", "given an interpreter, walk the list of python threads",
340*2fb4439dSAlexander Pyhalov 		py_thread_walk_init, py_walk_step, py_walk_fini },
341*2fb4439dSAlexander Pyhalov 	{ "pyframe", "given a thread state, walk the list of frame objects",
342*2fb4439dSAlexander Pyhalov 		py_frame_walk_init, py_walk_step, py_walk_fini },
343*2fb4439dSAlexander Pyhalov 	{ NULL }
344*2fb4439dSAlexander Pyhalov };
345*2fb4439dSAlexander Pyhalov 
346*2fb4439dSAlexander Pyhalov static const mdb_modinfo_t modinfo = {
347*2fb4439dSAlexander Pyhalov 	MDB_API_VERSION, dcmds, walkers
348*2fb4439dSAlexander Pyhalov };
349*2fb4439dSAlexander Pyhalov 
350*2fb4439dSAlexander Pyhalov /*ARGSUSED*/
351*2fb4439dSAlexander Pyhalov static int
python_object_iter(void * cd,const prmap_t * pmp,const char * obj)352*2fb4439dSAlexander Pyhalov python_object_iter(void *cd, const prmap_t *pmp, const char *obj)
353*2fb4439dSAlexander Pyhalov {
354*2fb4439dSAlexander Pyhalov 	char path[PATH_MAX];
355*2fb4439dSAlexander Pyhalov 	char *name;
356*2fb4439dSAlexander Pyhalov 	char *s1, *s2;
357*2fb4439dSAlexander Pyhalov 	struct ps_prochandle *Pr = cd;
358*2fb4439dSAlexander Pyhalov 
359*2fb4439dSAlexander Pyhalov 	name = strstr(obj, "/libpython");
360*2fb4439dSAlexander Pyhalov 
361*2fb4439dSAlexander Pyhalov 	if (name) {
362*2fb4439dSAlexander Pyhalov 		(void) strcpy(path, obj);
363*2fb4439dSAlexander Pyhalov 		if (Pstatus(Pr)->pr_dmodel != PR_MODEL_NATIVE) {
364*2fb4439dSAlexander Pyhalov 			s1 = name;
365*2fb4439dSAlexander Pyhalov 			s2 = path + (s1 - obj);
366*2fb4439dSAlexander Pyhalov 			(void) strcpy(s2, "/64");
367*2fb4439dSAlexander Pyhalov 			s2 += 3;
368*2fb4439dSAlexander Pyhalov 			(void) strcpy(s2, s1);
369*2fb4439dSAlexander Pyhalov 		}
370*2fb4439dSAlexander Pyhalov 
371*2fb4439dSAlexander Pyhalov 		s1 = strstr(obj, ".so");
372*2fb4439dSAlexander Pyhalov 		s2 = strstr(path, ".so");
373*2fb4439dSAlexander Pyhalov 		(void) strcpy(s2, "_db");
374*2fb4439dSAlexander Pyhalov 		s2 += 3;
375*2fb4439dSAlexander Pyhalov 		(void) strcpy(s2, s1);
376*2fb4439dSAlexander Pyhalov 
377*2fb4439dSAlexander Pyhalov 		if ((pydb_dlhdl = dlopen(path, RTLD_LAZY|RTLD_GLOBAL)) != NULL)
378*2fb4439dSAlexander Pyhalov 			return (1);
379*2fb4439dSAlexander Pyhalov 	}
380*2fb4439dSAlexander Pyhalov 
381*2fb4439dSAlexander Pyhalov 	return (0);
382*2fb4439dSAlexander Pyhalov }
383*2fb4439dSAlexander Pyhalov 
384*2fb4439dSAlexander Pyhalov static int
python_db_init(void)385*2fb4439dSAlexander Pyhalov python_db_init(void)
386*2fb4439dSAlexander Pyhalov {
387*2fb4439dSAlexander Pyhalov 	struct ps_prochandle *Ph;
388*2fb4439dSAlexander Pyhalov 
389*2fb4439dSAlexander Pyhalov 	if (mdb_get_xdata("pshandle", &Ph, sizeof (Ph)) == -1) {
390*2fb4439dSAlexander Pyhalov 		mdb_warn("couldn't read pshandle xdata\n");
391*2fb4439dSAlexander Pyhalov 		dlclose(pydb_dlhdl);
392*2fb4439dSAlexander Pyhalov 		pydb_dlhdl = NULL;
393*2fb4439dSAlexander Pyhalov 		return (-1);
394*2fb4439dSAlexander Pyhalov 	}
395*2fb4439dSAlexander Pyhalov 
396*2fb4439dSAlexander Pyhalov 	(void) Pobject_iter(Ph, python_object_iter, Ph);
397*2fb4439dSAlexander Pyhalov 
398*2fb4439dSAlexander Pyhalov 	pydb_agent_create = (pydb_agent_create_f)
399*2fb4439dSAlexander Pyhalov 	    dlsym(pydb_dlhdl, "pydb_agent_create");
400*2fb4439dSAlexander Pyhalov 	pydb_agent_destroy = (pydb_agent_destroy_f)
401*2fb4439dSAlexander Pyhalov 	    dlsym(pydb_dlhdl, "pydb_agent_destroy");
402*2fb4439dSAlexander Pyhalov 	pydb_get_frameinfo = (pydb_get_frameinfo_f)
403*2fb4439dSAlexander Pyhalov 	    dlsym(pydb_dlhdl, "pydb_get_frameinfo");
404*2fb4439dSAlexander Pyhalov 
405*2fb4439dSAlexander Pyhalov 	pydb_frame_iter_init = (pydb_iter_init_f)
406*2fb4439dSAlexander Pyhalov 	    dlsym(pydb_dlhdl, "pydb_frame_iter_init");
407*2fb4439dSAlexander Pyhalov 	pydb_interp_iter_init = (pydb_iter_init_f)
408*2fb4439dSAlexander Pyhalov 	    dlsym(pydb_dlhdl, "pydb_interp_iter_init");
409*2fb4439dSAlexander Pyhalov 	pydb_thread_iter_init = (pydb_iter_init_f)
410*2fb4439dSAlexander Pyhalov 	    dlsym(pydb_dlhdl, "pydb_thread_iter_init");
411*2fb4439dSAlexander Pyhalov 	pydb_iter_next = (pydb_iter_next_f)dlsym(pydb_dlhdl, "pydb_iter_next");
412*2fb4439dSAlexander Pyhalov 	pydb_iter_fini = (pydb_iter_fini_f)dlsym(pydb_dlhdl, "pydb_iter_fini");
413*2fb4439dSAlexander Pyhalov 
414*2fb4439dSAlexander Pyhalov 
415*2fb4439dSAlexander Pyhalov 	if (pydb_agent_create == NULL || pydb_agent_destroy == NULL ||
416*2fb4439dSAlexander Pyhalov 	    pydb_get_frameinfo == NULL || pydb_frame_iter_init == NULL ||
417*2fb4439dSAlexander Pyhalov 	    pydb_interp_iter_init == NULL || pydb_thread_iter_init == NULL ||
418*2fb4439dSAlexander Pyhalov 	    pydb_iter_next == NULL || pydb_iter_fini == NULL) {
419*2fb4439dSAlexander Pyhalov 		mdb_warn("couldn't load pydb functions");
420*2fb4439dSAlexander Pyhalov 		dlclose(pydb_dlhdl);
421*2fb4439dSAlexander Pyhalov 		pydb_dlhdl = NULL;
422*2fb4439dSAlexander Pyhalov 		return (-1);
423*2fb4439dSAlexander Pyhalov 	}
424*2fb4439dSAlexander Pyhalov 
425*2fb4439dSAlexander Pyhalov 	pydb_hdl = pydb_agent_create(Ph, PYDB_VERSION);
426*2fb4439dSAlexander Pyhalov 	if (pydb_hdl == NULL) {
427*2fb4439dSAlexander Pyhalov 		mdb_warn("unable to create pydb_agent");
428*2fb4439dSAlexander Pyhalov 		dlclose(pydb_dlhdl);
429*2fb4439dSAlexander Pyhalov 		pydb_dlhdl = NULL;
430*2fb4439dSAlexander Pyhalov 		return (-1);
431*2fb4439dSAlexander Pyhalov 	}
432*2fb4439dSAlexander Pyhalov 
433*2fb4439dSAlexander Pyhalov 	return (0);
434*2fb4439dSAlexander Pyhalov }
435*2fb4439dSAlexander Pyhalov 
436*2fb4439dSAlexander Pyhalov static void
python_db_fini(void)437*2fb4439dSAlexander Pyhalov python_db_fini(void)
438*2fb4439dSAlexander Pyhalov {
439*2fb4439dSAlexander Pyhalov 	if (pydb_dlhdl) {
440*2fb4439dSAlexander Pyhalov 		pydb_agent_destroy(pydb_hdl);
441*2fb4439dSAlexander Pyhalov 		pydb_hdl = NULL;
442*2fb4439dSAlexander Pyhalov 
443*2fb4439dSAlexander Pyhalov 		dlclose(pydb_dlhdl);
444*2fb4439dSAlexander Pyhalov 		pydb_dlhdl = NULL;
445*2fb4439dSAlexander Pyhalov 	}
446*2fb4439dSAlexander Pyhalov }
447*2fb4439dSAlexander Pyhalov 
448*2fb4439dSAlexander Pyhalov const mdb_modinfo_t *
_mdb_init(void)449*2fb4439dSAlexander Pyhalov _mdb_init(void)
450*2fb4439dSAlexander Pyhalov {
451*2fb4439dSAlexander Pyhalov 	if (python_db_init() != 0)
452*2fb4439dSAlexander Pyhalov 		return (NULL);
453*2fb4439dSAlexander Pyhalov 
454*2fb4439dSAlexander Pyhalov 	return (&modinfo);
455*2fb4439dSAlexander Pyhalov }
456*2fb4439dSAlexander Pyhalov 
457*2fb4439dSAlexander Pyhalov void
_mdb_fini(void)458*2fb4439dSAlexander Pyhalov _mdb_fini(void)
459*2fb4439dSAlexander Pyhalov {
460*2fb4439dSAlexander Pyhalov 	python_db_fini();
461*2fb4439dSAlexander Pyhalov }
462