blob: 3036ab684d939ba68d28a4f33691869430f90a18 [file] [log] [blame]
Guido van Rossumf70e43a1991-02-19 12:39:46 +00001
Guido van Rossum3f5da241990-12-20 15:06:42 +00002/* Frame object implementation */
3
Guido van Rossum18752471997-04-29 14:49:28 +00004#include "Python.h"
Guido van Rossum3f5da241990-12-20 15:06:42 +00005
6#include "compile.h"
7#include "frameobject.h"
8#include "opcode.h"
9#include "structmember.h"
10
Guido van Rossum18752471997-04-29 14:49:28 +000011#define OFF(x) offsetof(PyFrameObject, x)
Guido van Rossum3f5da241990-12-20 15:06:42 +000012
Guido van Rossum6f799372001-09-20 20:46:19 +000013static PyMemberDef frame_memberlist[] = {
Guido van Rossum1d5735e1994-08-30 08:27:36 +000014 {"f_back", T_OBJECT, OFF(f_back), RO},
15 {"f_code", T_OBJECT, OFF(f_code), RO},
Guido van Rossumc1134821995-01-10 10:39:16 +000016 {"f_builtins", T_OBJECT, OFF(f_builtins),RO},
Guido van Rossum1d5735e1994-08-30 08:27:36 +000017 {"f_globals", T_OBJECT, OFF(f_globals), RO},
Guido van Rossum1d5735e1994-08-30 08:27:36 +000018 {"f_lasti", T_INT, OFF(f_lasti), RO},
Guido van Rossumc1134821995-01-10 10:39:16 +000019 {"f_restricted",T_INT, OFF(f_restricted),RO},
Guido van Rossuma027efa1997-05-05 20:56:21 +000020 {"f_exc_type", T_OBJECT, OFF(f_exc_type)},
21 {"f_exc_value", T_OBJECT, OFF(f_exc_value)},
22 {"f_exc_traceback", T_OBJECT, OFF(f_exc_traceback)},
Guido van Rossum3f5da241990-12-20 15:06:42 +000023 {NULL} /* Sentinel */
24};
25
Guido van Rossum18752471997-04-29 14:49:28 +000026static PyObject *
Tim Peters6d6c1a32001-08-02 04:15:00 +000027frame_getlocals(PyFrameObject *f, void *closure)
Guido van Rossum3f5da241990-12-20 15:06:42 +000028{
Tim Peters6d6c1a32001-08-02 04:15:00 +000029 PyFrame_FastToLocals(f);
30 Py_INCREF(f->f_locals);
31 return f->f_locals;
Guido van Rossum3f5da241990-12-20 15:06:42 +000032}
33
Michael W. Hudsondd32a912002-08-15 14:59:02 +000034static PyObject *
35frame_getlineno(PyFrameObject *f, void *closure)
36{
37 int lineno;
38
Michael W. Hudson02ff6a92002-09-11 15:36:32 +000039 if (f->f_trace)
40 lineno = f->f_lineno;
41 else
42 lineno = PyCode_Addr2Line(f->f_code, f->f_lasti);
Michael W. Hudsondd32a912002-08-15 14:59:02 +000043
44 return PyInt_FromLong(lineno);
45}
46
Michael W. Hudson02ff6a92002-09-11 15:36:32 +000047static PyObject *
48frame_gettrace(PyFrameObject *f, void *closure)
49{
50 PyObject* trace = f->f_trace;
51
52 if (trace == NULL)
53 trace = Py_None;
54
55 Py_INCREF(trace);
56
57 return trace;
58}
59
60static int
61frame_settrace(PyFrameObject *f, PyObject* v, void *closure)
62{
63 /* We rely on f_lineno being accurate when f_trace is set. */
64
65 PyObject* old_value = f->f_trace;
66
67 Py_XINCREF(v);
68 f->f_trace = v;
69
70 if (v != NULL)
71 f->f_lineno = PyCode_Addr2Line(f->f_code, f->f_lasti);
72
73 Py_XDECREF(old_value);
74
75 return 0;
76}
77
Guido van Rossum32d34c82001-09-20 21:45:26 +000078static PyGetSetDef frame_getsetlist[] = {
Tim Peters6d6c1a32001-08-02 04:15:00 +000079 {"f_locals", (getter)frame_getlocals, NULL, NULL},
Michael W. Hudsondd32a912002-08-15 14:59:02 +000080 {"f_lineno", (getter)frame_getlineno, NULL, NULL},
Michael W. Hudson02ff6a92002-09-11 15:36:32 +000081 {"f_trace", (getter)frame_gettrace, (setter)frame_settrace, NULL},
Tim Peters6d6c1a32001-08-02 04:15:00 +000082 {0}
83};
Guido van Rossum1d5735e1994-08-30 08:27:36 +000084
Guido van Rossuma9e7dc11992-10-18 18:53:57 +000085/* Stack frames are allocated and deallocated at a considerable rate.
86 In an attempt to improve the speed of function calls, we maintain a
87 separate free list of stack frames (just like integers are
88 allocated in a special way -- see intobject.c). When a stack frame
89 is on the free list, only the following members have a meaning:
90 ob_type == &Frametype
91 f_back next item on free list, or NULL
Guido van Rossumf3e85a01997-01-20 04:20:52 +000092 f_nlocals number of locals
93 f_stacksize size of value stack
Neil Schemenauer4f4817f2001-08-29 23:52:17 +000094 ob_size size of localsplus
Guido van Rossuma9e7dc11992-10-18 18:53:57 +000095 Note that the value and block stacks are preserved -- this can save
96 another malloc() call or two (and two free() calls as well!).
97 Also note that, unlike for integers, each frame object is a
98 malloc'ed object in its own right -- it is only the actual calls to
99 malloc() that we are trying to save here, not the administration.
100 After all, while a typical program may make millions of calls, a
101 call depth of more than 20 or 30 is probably already exceptional
102 unless the program contains run-away recursion. I hope.
Tim Petersb7ba7432002-04-13 05:21:47 +0000103
104 Later, MAXFREELIST was added to bound the # of frames saved on
105 free_list. Else programs creating lots of cyclic trash involving
106 frames could provoke free_list into growing without bound.
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000107*/
108
Guido van Rossum18752471997-04-29 14:49:28 +0000109static PyFrameObject *free_list = NULL;
Tim Petersb7ba7432002-04-13 05:21:47 +0000110static int numfree = 0; /* number of frames currently in free_list */
111#define MAXFREELIST 200 /* max value for numfree */
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000112
Guido van Rossum3f5da241990-12-20 15:06:42 +0000113static void
Fred Drake1b190b42000-07-09 05:40:56 +0000114frame_dealloc(PyFrameObject *f)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000115{
Jeremy Hylton30c9f392001-03-13 01:58:22 +0000116 int i, slots;
Guido van Rossum7582bfb1997-02-14 16:27:29 +0000117 PyObject **fastlocals;
Tim Peters5ca576e2001-06-18 22:08:13 +0000118 PyObject **p;
Guido van Rossum7582bfb1997-02-14 16:27:29 +0000119
Guido van Rossumff413af2002-03-28 20:34:59 +0000120 PyObject_GC_UnTrack(f);
Guido van Rossumd724b232000-03-13 16:01:29 +0000121 Py_TRASHCAN_SAFE_BEGIN(f)
Guido van Rossum7582bfb1997-02-14 16:27:29 +0000122 /* Kill all local variables */
Jeremy Hylton30c9f392001-03-13 01:58:22 +0000123 slots = f->f_nlocals + f->f_ncells + f->f_nfreevars;
Guido van Rossum7582bfb1997-02-14 16:27:29 +0000124 fastlocals = f->f_localsplus;
Jeremy Hylton30c9f392001-03-13 01:58:22 +0000125 for (i = slots; --i >= 0; ++fastlocals) {
Guido van Rossum18752471997-04-29 14:49:28 +0000126 Py_XDECREF(*fastlocals);
Guido van Rossum7582bfb1997-02-14 16:27:29 +0000127 }
128
Tim Peters5ca576e2001-06-18 22:08:13 +0000129 /* Free stack */
Tim Peters8c963692001-06-23 05:26:56 +0000130 if (f->f_stacktop != NULL) {
131 for (p = f->f_valuestack; p < f->f_stacktop; p++)
132 Py_XDECREF(*p);
Tim Peters5ca576e2001-06-18 22:08:13 +0000133 }
Tim Peters8c963692001-06-23 05:26:56 +0000134
Guido van Rossum18752471997-04-29 14:49:28 +0000135 Py_XDECREF(f->f_back);
136 Py_XDECREF(f->f_code);
137 Py_XDECREF(f->f_builtins);
138 Py_XDECREF(f->f_globals);
139 Py_XDECREF(f->f_locals);
140 Py_XDECREF(f->f_trace);
Guido van Rossuma027efa1997-05-05 20:56:21 +0000141 Py_XDECREF(f->f_exc_type);
142 Py_XDECREF(f->f_exc_value);
143 Py_XDECREF(f->f_exc_traceback);
Tim Petersb7ba7432002-04-13 05:21:47 +0000144 if (numfree < MAXFREELIST) {
145 ++numfree;
146 f->f_back = free_list;
147 free_list = f;
148 }
149 else
150 PyObject_GC_Del(f);
Guido van Rossumd724b232000-03-13 16:01:29 +0000151 Py_TRASHCAN_SAFE_END(f)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000152}
153
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000154static int
155frame_traverse(PyFrameObject *f, visitproc visit, void *arg)
156{
157 PyObject **fastlocals, **p;
158 int i, err, slots;
159#define VISIT(o) if (o) {if ((err = visit((PyObject *)(o), arg))) return err;}
160
161 VISIT(f->f_back);
162 VISIT(f->f_code);
163 VISIT(f->f_builtins);
164 VISIT(f->f_globals);
165 VISIT(f->f_locals);
166 VISIT(f->f_trace);
167 VISIT(f->f_exc_type);
168 VISIT(f->f_exc_value);
169 VISIT(f->f_exc_traceback);
170
171 /* locals */
172 slots = f->f_nlocals + f->f_ncells + f->f_nfreevars;
173 fastlocals = f->f_localsplus;
174 for (i = slots; --i >= 0; ++fastlocals) {
175 VISIT(*fastlocals);
176 }
177
178 /* stack */
179 if (f->f_stacktop != NULL) {
180 for (p = f->f_valuestack; p < f->f_stacktop; p++)
181 VISIT(*p);
182 }
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000183 return 0;
184}
185
186static void
187frame_clear(PyFrameObject *f)
188{
189 PyObject **fastlocals, **p;
190 int i, slots;
191
192 Py_XDECREF(f->f_exc_type);
193 f->f_exc_type = NULL;
194
195 Py_XDECREF(f->f_exc_value);
196 f->f_exc_value = NULL;
197
198 Py_XDECREF(f->f_exc_traceback);
199 f->f_exc_traceback = NULL;
200
201 Py_XDECREF(f->f_trace);
202 f->f_trace = NULL;
203
204 /* locals */
205 slots = f->f_nlocals + f->f_ncells + f->f_nfreevars;
206 fastlocals = f->f_localsplus;
207 for (i = slots; --i >= 0; ++fastlocals) {
208 if (*fastlocals != NULL) {
209 Py_XDECREF(*fastlocals);
210 *fastlocals = NULL;
211 }
212 }
213
214 /* stack */
215 if (f->f_stacktop != NULL) {
216 for (p = f->f_valuestack; p < f->f_stacktop; p++) {
217 Py_XDECREF(*p);
218 *p = NULL;
219 }
220 }
221}
222
223
Guido van Rossum18752471997-04-29 14:49:28 +0000224PyTypeObject PyFrame_Type = {
225 PyObject_HEAD_INIT(&PyType_Type)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000226 0,
227 "frame",
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000228 sizeof(PyFrameObject),
229 sizeof(PyObject *),
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000230 (destructor)frame_dealloc, /* tp_dealloc */
231 0, /* tp_print */
Tim Peters6d6c1a32001-08-02 04:15:00 +0000232 0, /* tp_getattr */
233 0, /* tp_setattr */
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000234 0, /* tp_compare */
235 0, /* tp_repr */
236 0, /* tp_as_number */
237 0, /* tp_as_sequence */
238 0, /* tp_as_mapping */
239 0, /* tp_hash */
240 0, /* tp_call */
241 0, /* tp_str */
Tim Peters6d6c1a32001-08-02 04:15:00 +0000242 PyObject_GenericGetAttr, /* tp_getattro */
243 PyObject_GenericSetAttr, /* tp_setattro */
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000244 0, /* tp_as_buffer */
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000245 Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC,/* tp_flags */
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000246 0, /* tp_doc */
247 (traverseproc)frame_traverse, /* tp_traverse */
248 (inquiry)frame_clear, /* tp_clear */
Tim Peters6d6c1a32001-08-02 04:15:00 +0000249 0, /* tp_richcompare */
250 0, /* tp_weaklistoffset */
251 0, /* tp_iter */
252 0, /* tp_iternext */
253 0, /* tp_methods */
254 frame_memberlist, /* tp_members */
255 frame_getsetlist, /* tp_getset */
256 0, /* tp_base */
257 0, /* tp_dict */
Guido van Rossum3f5da241990-12-20 15:06:42 +0000258};
259
Guido van Rossum18752471997-04-29 14:49:28 +0000260PyFrameObject *
Jeremy Hylton64949cb2001-01-25 20:06:59 +0000261PyFrame_New(PyThreadState *tstate, PyCodeObject *code, PyObject *globals,
Jeremy Hylton30c9f392001-03-13 01:58:22 +0000262 PyObject *locals)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000263{
Guido van Rossuma027efa1997-05-05 20:56:21 +0000264 PyFrameObject *back = tstate->frame;
Guido van Rossum18752471997-04-29 14:49:28 +0000265 static PyObject *builtin_object;
266 PyFrameObject *f;
267 PyObject *builtins;
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000268 int extras, ncells, nfrees;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000269
Sjoerd Mullender5b7f3cd1995-04-04 11:47:41 +0000270 if (builtin_object == NULL) {
Guido van Rossumb56933e1997-01-18 07:58:41 +0000271 builtin_object = PyString_InternFromString("__builtins__");
Sjoerd Mullender5b7f3cd1995-04-04 11:47:41 +0000272 if (builtin_object == NULL)
273 return NULL;
274 }
Michael W. Hudson69734a52002-08-19 16:54:08 +0000275#ifdef Py_DEBUG
276 if (code == NULL || globals == NULL || !PyDict_Check(globals) ||
Guido van Rossum18752471997-04-29 14:49:28 +0000277 (locals != NULL && !PyDict_Check(locals))) {
278 PyErr_BadInternalCall();
Guido van Rossum3f5da241990-12-20 15:06:42 +0000279 return NULL;
280 }
Michael W. Hudson69734a52002-08-19 16:54:08 +0000281#endif
Jeremy Hylton64949cb2001-01-25 20:06:59 +0000282 ncells = PyTuple_GET_SIZE(code->co_cellvars);
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000283 nfrees = PyTuple_GET_SIZE(code->co_freevars);
284 extras = code->co_stacksize + code->co_nlocals + ncells + nfrees;
Guido van Rossumbde6ff71998-02-19 20:48:26 +0000285 if (back == NULL || back->f_globals != globals) {
286 builtins = PyDict_GetItem(globals, builtin_object);
287 if (builtins != NULL && PyModule_Check(builtins))
288 builtins = PyModule_GetDict(builtins);
289 }
290 else {
291 /* If we share the globals, we share the builtins.
292 Save a lookup and a call. */
293 builtins = back->f_builtins;
294 }
Guido van Rossum404b95d1997-08-05 02:09:46 +0000295 if (builtins != NULL && !PyDict_Check(builtins))
296 builtins = NULL;
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000297 if (free_list == NULL) {
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000298 f = PyObject_GC_NewVar(PyFrameObject, &PyFrame_Type, extras);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000299 if (f == NULL)
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000300 return NULL;
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000301 }
302 else {
Tim Petersb7ba7432002-04-13 05:21:47 +0000303 assert(numfree > 0);
304 --numfree;
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000305 f = free_list;
306 free_list = free_list->f_back;
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000307 if (f->ob_size < extras) {
308 f = PyObject_GC_Resize(PyFrameObject, f, extras);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000309 if (f == NULL)
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000310 return NULL;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000311 }
Tim Petersdeb77e82001-08-30 00:32:51 +0000312 _Py_NewReference((PyObject *)f);
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000313 }
Guido van Rossum404b95d1997-08-05 02:09:46 +0000314 if (builtins == NULL) {
Guido van Rossumbde6ff71998-02-19 20:48:26 +0000315 /* No builtins! Make up a minimal one. */
Guido van Rossum404b95d1997-08-05 02:09:46 +0000316 builtins = PyDict_New();
Guido van Rossumf61618c1998-10-19 14:20:20 +0000317 if (builtins == NULL || /* Give them 'None', at least. */
318 PyDict_SetItemString(builtins, "None", Py_None) < 0) {
319 Py_DECREF(f);
Guido van Rossum404b95d1997-08-05 02:09:46 +0000320 return NULL;
Guido van Rossumf61618c1998-10-19 14:20:20 +0000321 }
Guido van Rossum404b95d1997-08-05 02:09:46 +0000322 }
323 else
Neal Norwitzd94c28e2002-08-29 20:25:46 +0000324 Py_INCREF(builtins);
Guido van Rossum404b95d1997-08-05 02:09:46 +0000325 f->f_builtins = builtins;
Guido van Rossum18752471997-04-29 14:49:28 +0000326 Py_XINCREF(back);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000327 f->f_back = back;
Guido van Rossum18752471997-04-29 14:49:28 +0000328 Py_INCREF(code);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000329 f->f_code = code;
Guido van Rossum18752471997-04-29 14:49:28 +0000330 Py_INCREF(globals);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000331 f->f_globals = globals;
Guido van Rossumbdd207a1995-07-26 16:14:30 +0000332 if (code->co_flags & CO_NEWLOCALS) {
333 if (code->co_flags & CO_OPTIMIZED)
334 locals = NULL; /* Let fast_2_locals handle it */
335 else {
Guido van Rossum18752471997-04-29 14:49:28 +0000336 locals = PyDict_New();
Guido van Rossumbdd207a1995-07-26 16:14:30 +0000337 if (locals == NULL) {
Guido van Rossum18752471997-04-29 14:49:28 +0000338 Py_DECREF(f);
Guido van Rossumbdd207a1995-07-26 16:14:30 +0000339 return NULL;
340 }
Guido van Rossum3f5da241990-12-20 15:06:42 +0000341 }
Guido van Rossum3f5da241990-12-20 15:06:42 +0000342 }
Guido van Rossum2271bf71995-07-18 14:30:34 +0000343 else {
344 if (locals == NULL)
345 locals = globals;
Guido van Rossum18752471997-04-29 14:49:28 +0000346 Py_INCREF(locals);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000347 }
348 f->f_locals = locals;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000349 f->f_trace = NULL;
Guido van Rossuma027efa1997-05-05 20:56:21 +0000350 f->f_exc_type = f->f_exc_value = f->f_exc_traceback = NULL;
Guido van Rossumeb46d671997-08-02 02:59:08 +0000351 f->f_tstate = tstate;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000352
Michael W. Hudsondd32a912002-08-15 14:59:02 +0000353 f->f_lasti = -1;
Guido van Rossum747596a1997-01-24 04:00:21 +0000354 f->f_lineno = code->co_firstlineno;
Guido van Rossumeb46d671997-08-02 02:59:08 +0000355 f->f_restricted = (builtins != tstate->interp->builtins);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000356 f->f_iblock = 0;
357 f->f_nlocals = code->co_nlocals;
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000358 f->f_stacksize = code->co_stacksize;
359 f->f_ncells = ncells;
360 f->f_nfreevars = nfrees;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000361
Guido van Rossumf4be4272002-08-01 18:50:33 +0000362 extras = f->f_nlocals + ncells + nfrees;
363 memset(f->f_localsplus, 0, extras * sizeof(f->f_localsplus[0]));
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000364
Guido van Rossumf4be4272002-08-01 18:50:33 +0000365 f->f_valuestack = f->f_localsplus + extras;
Tim Peters8c963692001-06-23 05:26:56 +0000366 f->f_stacktop = f->f_valuestack;
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000367 _PyObject_GC_TRACK(f);
Guido van Rossum3f5da241990-12-20 15:06:42 +0000368 return f;
369}
370
Guido van Rossum3f5da241990-12-20 15:06:42 +0000371/* Block management */
372
373void
Fred Drake1b190b42000-07-09 05:40:56 +0000374PyFrame_BlockSetup(PyFrameObject *f, int type, int handler, int level)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000375{
Guido van Rossum18752471997-04-29 14:49:28 +0000376 PyTryBlock *b;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000377 if (f->f_iblock >= CO_MAXBLOCKS)
Guido van Rossum18752471997-04-29 14:49:28 +0000378 Py_FatalError("XXX block stack overflow");
Guido van Rossum3f5da241990-12-20 15:06:42 +0000379 b = &f->f_blockstack[f->f_iblock++];
380 b->b_type = type;
381 b->b_level = level;
382 b->b_handler = handler;
383}
384
Guido van Rossum18752471997-04-29 14:49:28 +0000385PyTryBlock *
Fred Drake1b190b42000-07-09 05:40:56 +0000386PyFrame_BlockPop(PyFrameObject *f)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000387{
Guido van Rossum18752471997-04-29 14:49:28 +0000388 PyTryBlock *b;
Guido van Rossumd7047b31995-01-02 19:07:15 +0000389 if (f->f_iblock <= 0)
Guido van Rossum18752471997-04-29 14:49:28 +0000390 Py_FatalError("XXX block stack underflow");
Guido van Rossum3f5da241990-12-20 15:06:42 +0000391 b = &f->f_blockstack[--f->f_iblock];
392 return b;
393}
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000394
395/* Convert between "fast" version of locals and dictionary version */
396
Guido van Rossumf68d8e52001-04-14 17:55:09 +0000397static void
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000398map_to_dict(PyObject *map, int nmap, PyObject *dict, PyObject **values,
399 int deref)
400{
401 int j;
402 for (j = nmap; --j >= 0; ) {
Jeremy Hylton1a48ca82001-12-06 15:48:16 +0000403 PyObject *key = PyTuple_GET_ITEM(map, j);
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000404 PyObject *value = values[j];
405 if (deref)
406 value = PyCell_GET(value);
407 if (value == NULL) {
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000408 if (PyDict_DelItem(dict, key) != 0)
409 PyErr_Clear();
410 }
411 else {
412 if (PyDict_SetItem(dict, key, value) != 0)
413 PyErr_Clear();
414 }
415 }
416}
417
Guido van Rossum6b356e72001-04-14 17:55:41 +0000418static void
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000419dict_to_map(PyObject *map, int nmap, PyObject *dict, PyObject **values,
420 int deref, int clear)
421{
422 int j;
423 for (j = nmap; --j >= 0; ) {
Jeremy Hylton1a48ca82001-12-06 15:48:16 +0000424 PyObject *key = PyTuple_GET_ITEM(map, j);
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000425 PyObject *value = PyDict_GetItem(dict, key);
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000426 if (deref) {
Jeremy Hylton4c889012001-05-08 04:08:59 +0000427 if (value || clear) {
Jeremy Hylton1a48ca82001-12-06 15:48:16 +0000428 if (PyCell_GET(values[j]) != value) {
429 if (PyCell_Set(values[j], value) < 0)
430 PyErr_Clear();
431 }
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000432 }
433 } else if (value != NULL || clear) {
Jeremy Hylton1a48ca82001-12-06 15:48:16 +0000434 if (values[j] != value) {
435 Py_XINCREF(value);
436 Py_XDECREF(values[j]);
437 values[j] = value;
438 }
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000439 }
440 }
441}
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000442
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000443void
Fred Drake1b190b42000-07-09 05:40:56 +0000444PyFrame_FastToLocals(PyFrameObject *f)
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000445{
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000446 /* Merge fast locals into f->f_locals */
Guido van Rossum18752471997-04-29 14:49:28 +0000447 PyObject *locals, *map;
448 PyObject **fast;
449 PyObject *error_type, *error_value, *error_traceback;
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000450 int j;
451 if (f == NULL)
452 return;
Guido van Rossum2271bf71995-07-18 14:30:34 +0000453 locals = f->f_locals;
454 if (locals == NULL) {
Guido van Rossum18752471997-04-29 14:49:28 +0000455 locals = f->f_locals = PyDict_New();
Guido van Rossum2271bf71995-07-18 14:30:34 +0000456 if (locals == NULL) {
Guido van Rossum18752471997-04-29 14:49:28 +0000457 PyErr_Clear(); /* Can't report it :-( */
Guido van Rossum2271bf71995-07-18 14:30:34 +0000458 return;
459 }
460 }
Guido van Rossumbdd207a1995-07-26 16:14:30 +0000461 map = f->f_code->co_varnames;
Guido van Rossum18752471997-04-29 14:49:28 +0000462 if (!PyDict_Check(locals) || !PyTuple_Check(map))
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000463 return;
Guido van Rossum18752471997-04-29 14:49:28 +0000464 PyErr_Fetch(&error_type, &error_value, &error_traceback);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000465 fast = f->f_localsplus;
Guido van Rossum18752471997-04-29 14:49:28 +0000466 j = PyTuple_Size(map);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000467 if (j > f->f_nlocals)
468 j = f->f_nlocals;
Jeremy Hylton24ea8d32002-04-20 04:46:55 +0000469 if (f->f_nlocals)
470 map_to_dict(map, j, locals, fast, 0);
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000471 if (f->f_ncells || f->f_nfreevars) {
472 if (!(PyTuple_Check(f->f_code->co_cellvars)
473 && PyTuple_Check(f->f_code->co_freevars))) {
474 Py_DECREF(locals);
475 return;
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000476 }
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000477 map_to_dict(f->f_code->co_cellvars,
478 PyTuple_GET_SIZE(f->f_code->co_cellvars),
479 locals, fast + f->f_nlocals, 1);
480 map_to_dict(f->f_code->co_freevars,
481 PyTuple_GET_SIZE(f->f_code->co_freevars),
482 locals, fast + f->f_nlocals + f->f_ncells, 1);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000483 }
Guido van Rossum18752471997-04-29 14:49:28 +0000484 PyErr_Restore(error_type, error_value, error_traceback);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000485}
486
487void
Fred Drake1b190b42000-07-09 05:40:56 +0000488PyFrame_LocalsToFast(PyFrameObject *f, int clear)
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000489{
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000490 /* Merge f->f_locals into fast locals */
Guido van Rossum18752471997-04-29 14:49:28 +0000491 PyObject *locals, *map;
492 PyObject **fast;
493 PyObject *error_type, *error_value, *error_traceback;
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000494 int j;
495 if (f == NULL)
496 return;
497 locals = f->f_locals;
Guido van Rossum2271bf71995-07-18 14:30:34 +0000498 map = f->f_code->co_varnames;
Jeremy Hylton24ea8d32002-04-20 04:46:55 +0000499 if (locals == NULL)
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000500 return;
Guido van Rossum18752471997-04-29 14:49:28 +0000501 if (!PyDict_Check(locals) || !PyTuple_Check(map))
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000502 return;
Guido van Rossum18752471997-04-29 14:49:28 +0000503 PyErr_Fetch(&error_type, &error_value, &error_traceback);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000504 fast = f->f_localsplus;
Guido van Rossum18752471997-04-29 14:49:28 +0000505 j = PyTuple_Size(map);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000506 if (j > f->f_nlocals)
507 j = f->f_nlocals;
Jeremy Hylton24ea8d32002-04-20 04:46:55 +0000508 if (f->f_nlocals)
509 dict_to_map(f->f_code->co_varnames, j, locals, fast, 0, clear);
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000510 if (f->f_ncells || f->f_nfreevars) {
511 if (!(PyTuple_Check(f->f_code->co_cellvars)
512 && PyTuple_Check(f->f_code->co_freevars)))
513 return;
514 dict_to_map(f->f_code->co_cellvars,
515 PyTuple_GET_SIZE(f->f_code->co_cellvars),
Jeremy Hylton4c889012001-05-08 04:08:59 +0000516 locals, fast + f->f_nlocals, 1, clear);
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000517 dict_to_map(f->f_code->co_freevars,
518 PyTuple_GET_SIZE(f->f_code->co_freevars),
Jeremy Hylton24ea8d32002-04-20 04:46:55 +0000519 locals, fast + f->f_nlocals + f->f_ncells, 1,
520 clear);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000521 }
Guido van Rossum18752471997-04-29 14:49:28 +0000522 PyErr_Restore(error_type, error_value, error_traceback);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000523}
Guido van Rossum404b95d1997-08-05 02:09:46 +0000524
525/* Clear out the free list */
526
527void
Fred Drake1b190b42000-07-09 05:40:56 +0000528PyFrame_Fini(void)
Guido van Rossum404b95d1997-08-05 02:09:46 +0000529{
530 while (free_list != NULL) {
531 PyFrameObject *f = free_list;
532 free_list = free_list->f_back;
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000533 PyObject_GC_Del(f);
Tim Petersb7ba7432002-04-13 05:21:47 +0000534 --numfree;
Guido van Rossum404b95d1997-08-05 02:09:46 +0000535 }
Tim Petersb7ba7432002-04-13 05:21:47 +0000536 assert(numfree == 0);
Guido van Rossum404b95d1997-08-05 02:09:46 +0000537}