blob: 0801b932c516c55345c276a29f1c5027f08e6c0e [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
13static struct memberlist 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},
18 {"f_locals", T_OBJECT, OFF(f_locals), RO},
Guido van Rossum1d5735e1994-08-30 08:27:36 +000019 {"f_lasti", T_INT, OFF(f_lasti), RO},
20 {"f_lineno", T_INT, OFF(f_lineno), RO},
Guido van Rossumc1134821995-01-10 10:39:16 +000021 {"f_restricted",T_INT, OFF(f_restricted),RO},
Guido van Rossum1d5735e1994-08-30 08:27:36 +000022 {"f_trace", T_OBJECT, OFF(f_trace)},
Guido van Rossuma027efa1997-05-05 20:56:21 +000023 {"f_exc_type", T_OBJECT, OFF(f_exc_type)},
24 {"f_exc_value", T_OBJECT, OFF(f_exc_value)},
25 {"f_exc_traceback", T_OBJECT, OFF(f_exc_traceback)},
Guido van Rossum3f5da241990-12-20 15:06:42 +000026 {NULL} /* Sentinel */
27};
28
Guido van Rossum18752471997-04-29 14:49:28 +000029static PyObject *
Fred Drake1b190b42000-07-09 05:40:56 +000030frame_getattr(PyFrameObject *f, char *name)
Guido van Rossum3f5da241990-12-20 15:06:42 +000031{
Guido van Rossum1d5735e1994-08-30 08:27:36 +000032 if (strcmp(name, "f_locals") == 0)
Guido van Rossum18752471997-04-29 14:49:28 +000033 PyFrame_FastToLocals(f);
34 return PyMember_Get((char *)f, frame_memberlist, name);
Guido van Rossum3f5da241990-12-20 15:06:42 +000035}
36
Guido van Rossum1d5735e1994-08-30 08:27:36 +000037static int
Fred Drake1b190b42000-07-09 05:40:56 +000038frame_setattr(PyFrameObject *f, char *name, PyObject *value)
Guido van Rossum1d5735e1994-08-30 08:27:36 +000039{
Guido van Rossum18752471997-04-29 14:49:28 +000040 return PyMember_Set((char *)f, frame_memberlist, name, value);
Guido van Rossum1d5735e1994-08-30 08:27:36 +000041}
42
Guido van Rossuma9e7dc11992-10-18 18:53:57 +000043/* Stack frames are allocated and deallocated at a considerable rate.
44 In an attempt to improve the speed of function calls, we maintain a
45 separate free list of stack frames (just like integers are
46 allocated in a special way -- see intobject.c). When a stack frame
47 is on the free list, only the following members have a meaning:
48 ob_type == &Frametype
49 f_back next item on free list, or NULL
Guido van Rossumf3e85a01997-01-20 04:20:52 +000050 f_nlocals number of locals
51 f_stacksize size of value stack
Jeremy Hylton2b724da2001-01-29 22:51:52 +000052 f_size size of localsplus
Guido van Rossuma9e7dc11992-10-18 18:53:57 +000053 Note that the value and block stacks are preserved -- this can save
54 another malloc() call or two (and two free() calls as well!).
55 Also note that, unlike for integers, each frame object is a
56 malloc'ed object in its own right -- it is only the actual calls to
57 malloc() that we are trying to save here, not the administration.
58 After all, while a typical program may make millions of calls, a
59 call depth of more than 20 or 30 is probably already exceptional
60 unless the program contains run-away recursion. I hope.
61*/
62
Guido van Rossum18752471997-04-29 14:49:28 +000063static PyFrameObject *free_list = NULL;
Guido van Rossuma9e7dc11992-10-18 18:53:57 +000064
Guido van Rossum3f5da241990-12-20 15:06:42 +000065static void
Fred Drake1b190b42000-07-09 05:40:56 +000066frame_dealloc(PyFrameObject *f)
Guido van Rossum3f5da241990-12-20 15:06:42 +000067{
Jeremy Hylton30c9f392001-03-13 01:58:22 +000068 int i, slots;
Guido van Rossum7582bfb1997-02-14 16:27:29 +000069 PyObject **fastlocals;
Tim Peters5ca576e2001-06-18 22:08:13 +000070 PyObject **p;
Guido van Rossum7582bfb1997-02-14 16:27:29 +000071
Guido van Rossumd724b232000-03-13 16:01:29 +000072 Py_TRASHCAN_SAFE_BEGIN(f)
Neil Schemenauer19cd2922001-07-12 13:27:11 +000073 PyObject_GC_Fini(f);
Guido van Rossum7582bfb1997-02-14 16:27:29 +000074 /* Kill all local variables */
Jeremy Hylton30c9f392001-03-13 01:58:22 +000075 slots = f->f_nlocals + f->f_ncells + f->f_nfreevars;
Guido van Rossum7582bfb1997-02-14 16:27:29 +000076 fastlocals = f->f_localsplus;
Jeremy Hylton30c9f392001-03-13 01:58:22 +000077 for (i = slots; --i >= 0; ++fastlocals) {
Guido van Rossum18752471997-04-29 14:49:28 +000078 Py_XDECREF(*fastlocals);
Guido van Rossum7582bfb1997-02-14 16:27:29 +000079 }
80
Tim Peters5ca576e2001-06-18 22:08:13 +000081 /* Free stack */
Tim Peters8c963692001-06-23 05:26:56 +000082 if (f->f_stacktop != NULL) {
83 for (p = f->f_valuestack; p < f->f_stacktop; p++)
84 Py_XDECREF(*p);
Tim Peters5ca576e2001-06-18 22:08:13 +000085 }
Tim Peters8c963692001-06-23 05:26:56 +000086
Guido van Rossum18752471997-04-29 14:49:28 +000087 Py_XDECREF(f->f_back);
88 Py_XDECREF(f->f_code);
89 Py_XDECREF(f->f_builtins);
90 Py_XDECREF(f->f_globals);
91 Py_XDECREF(f->f_locals);
92 Py_XDECREF(f->f_trace);
Guido van Rossuma027efa1997-05-05 20:56:21 +000093 Py_XDECREF(f->f_exc_type);
94 Py_XDECREF(f->f_exc_value);
95 Py_XDECREF(f->f_exc_traceback);
Guido van Rossuma9e7dc11992-10-18 18:53:57 +000096 f->f_back = free_list;
97 free_list = f;
Guido van Rossumd724b232000-03-13 16:01:29 +000098 Py_TRASHCAN_SAFE_END(f)
Guido van Rossum3f5da241990-12-20 15:06:42 +000099}
100
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000101static int
102frame_traverse(PyFrameObject *f, visitproc visit, void *arg)
103{
104 PyObject **fastlocals, **p;
105 int i, err, slots;
106#define VISIT(o) if (o) {if ((err = visit((PyObject *)(o), arg))) return err;}
107
108 VISIT(f->f_back);
109 VISIT(f->f_code);
110 VISIT(f->f_builtins);
111 VISIT(f->f_globals);
112 VISIT(f->f_locals);
113 VISIT(f->f_trace);
114 VISIT(f->f_exc_type);
115 VISIT(f->f_exc_value);
116 VISIT(f->f_exc_traceback);
117
118 /* locals */
119 slots = f->f_nlocals + f->f_ncells + f->f_nfreevars;
120 fastlocals = f->f_localsplus;
121 for (i = slots; --i >= 0; ++fastlocals) {
122 VISIT(*fastlocals);
123 }
124
125 /* stack */
126 if (f->f_stacktop != NULL) {
127 for (p = f->f_valuestack; p < f->f_stacktop; p++)
128 VISIT(*p);
129 }
130
131 return 0;
132}
133
134static void
135frame_clear(PyFrameObject *f)
136{
137 PyObject **fastlocals, **p;
138 int i, slots;
139
140 Py_XDECREF(f->f_exc_type);
141 f->f_exc_type = NULL;
142
143 Py_XDECREF(f->f_exc_value);
144 f->f_exc_value = NULL;
145
146 Py_XDECREF(f->f_exc_traceback);
147 f->f_exc_traceback = NULL;
148
149 Py_XDECREF(f->f_trace);
150 f->f_trace = NULL;
151
152 /* locals */
153 slots = f->f_nlocals + f->f_ncells + f->f_nfreevars;
154 fastlocals = f->f_localsplus;
155 for (i = slots; --i >= 0; ++fastlocals) {
156 if (*fastlocals != NULL) {
157 Py_XDECREF(*fastlocals);
158 *fastlocals = NULL;
159 }
160 }
161
162 /* stack */
163 if (f->f_stacktop != NULL) {
164 for (p = f->f_valuestack; p < f->f_stacktop; p++) {
165 Py_XDECREF(*p);
166 *p = NULL;
167 }
168 }
169}
170
171
Guido van Rossum18752471997-04-29 14:49:28 +0000172PyTypeObject PyFrame_Type = {
173 PyObject_HEAD_INIT(&PyType_Type)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000174 0,
175 "frame",
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000176 sizeof(PyFrameObject) + PyGC_HEAD_SIZE,
Guido van Rossum3f5da241990-12-20 15:06:42 +0000177 0,
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000178 (destructor)frame_dealloc, /* tp_dealloc */
179 0, /* tp_print */
180 (getattrfunc)frame_getattr, /* tp_getattr */
181 (setattrfunc)frame_setattr, /* tp_setattr */
182 0, /* tp_compare */
183 0, /* tp_repr */
184 0, /* tp_as_number */
185 0, /* tp_as_sequence */
186 0, /* tp_as_mapping */
187 0, /* tp_hash */
188 0, /* tp_call */
189 0, /* tp_str */
190 0, /* tp_getattro */
191 0, /* tp_setattro */
192 0, /* tp_as_buffer */
193 Py_TPFLAGS_DEFAULT | Py_TPFLAGS_GC, /* tp_flags */
194 0, /* tp_doc */
195 (traverseproc)frame_traverse, /* tp_traverse */
196 (inquiry)frame_clear, /* tp_clear */
Guido van Rossum3f5da241990-12-20 15:06:42 +0000197};
198
Guido van Rossum18752471997-04-29 14:49:28 +0000199PyFrameObject *
Jeremy Hylton64949cb2001-01-25 20:06:59 +0000200PyFrame_New(PyThreadState *tstate, PyCodeObject *code, PyObject *globals,
Jeremy Hylton30c9f392001-03-13 01:58:22 +0000201 PyObject *locals)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000202{
Guido van Rossuma027efa1997-05-05 20:56:21 +0000203 PyFrameObject *back = tstate->frame;
Guido van Rossum18752471997-04-29 14:49:28 +0000204 static PyObject *builtin_object;
205 PyFrameObject *f;
206 PyObject *builtins;
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000207 int extras, ncells, nfrees;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000208
Sjoerd Mullender5b7f3cd1995-04-04 11:47:41 +0000209 if (builtin_object == NULL) {
Guido van Rossumb56933e1997-01-18 07:58:41 +0000210 builtin_object = PyString_InternFromString("__builtins__");
Sjoerd Mullender5b7f3cd1995-04-04 11:47:41 +0000211 if (builtin_object == NULL)
212 return NULL;
213 }
Guido van Rossum18752471997-04-29 14:49:28 +0000214 if ((back != NULL && !PyFrame_Check(back)) ||
215 code == NULL || !PyCode_Check(code) ||
216 globals == NULL || !PyDict_Check(globals) ||
217 (locals != NULL && !PyDict_Check(locals))) {
218 PyErr_BadInternalCall();
Guido van Rossum3f5da241990-12-20 15:06:42 +0000219 return NULL;
220 }
Jeremy Hylton64949cb2001-01-25 20:06:59 +0000221 ncells = PyTuple_GET_SIZE(code->co_cellvars);
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000222 nfrees = PyTuple_GET_SIZE(code->co_freevars);
223 extras = code->co_stacksize + code->co_nlocals + ncells + nfrees;
Guido van Rossumbde6ff71998-02-19 20:48:26 +0000224 if (back == NULL || back->f_globals != globals) {
225 builtins = PyDict_GetItem(globals, builtin_object);
226 if (builtins != NULL && PyModule_Check(builtins))
227 builtins = PyModule_GetDict(builtins);
228 }
229 else {
230 /* If we share the globals, we share the builtins.
231 Save a lookup and a call. */
232 builtins = back->f_builtins;
233 }
Guido van Rossum404b95d1997-08-05 02:09:46 +0000234 if (builtins != NULL && !PyDict_Check(builtins))
235 builtins = NULL;
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000236 if (free_list == NULL) {
Guido van Rossumb18618d2000-05-03 23:44:39 +0000237 /* PyObject_New is inlined */
Guido van Rossum18752471997-04-29 14:49:28 +0000238 f = (PyFrameObject *)
Guido van Rossumb18618d2000-05-03 23:44:39 +0000239 PyObject_MALLOC(sizeof(PyFrameObject) +
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000240 extras*sizeof(PyObject *) +
241 PyGC_HEAD_SIZE);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000242 if (f == NULL)
Guido van Rossum18752471997-04-29 14:49:28 +0000243 return (PyFrameObject *)PyErr_NoMemory();
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000244 f = (PyFrameObject *) PyObject_FROM_GC(f);
Guido van Rossumb18618d2000-05-03 23:44:39 +0000245 PyObject_INIT(f, &PyFrame_Type);
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000246 f->f_size = extras;
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000247 }
248 else {
249 f = free_list;
250 free_list = free_list->f_back;
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000251 if (f->f_size < extras) {
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000252 f = (PyFrameObject *) PyObject_AS_GC(f);
Guido van Rossum6345ac61997-10-31 20:32:13 +0000253 f = (PyFrameObject *)
Guido van Rossumb18618d2000-05-03 23:44:39 +0000254 PyObject_REALLOC(f, sizeof(PyFrameObject) +
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000255 extras*sizeof(PyObject *) +
256 PyGC_HEAD_SIZE);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000257 if (f == NULL)
Guido van Rossum18752471997-04-29 14:49:28 +0000258 return (PyFrameObject *)PyErr_NoMemory();
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000259 f = (PyFrameObject *) PyObject_FROM_GC(f);
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000260 f->f_size = extras;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000261 }
Guido van Rossum747596a1997-01-24 04:00:21 +0000262 else
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000263 extras = f->f_size;
Guido van Rossumb18618d2000-05-03 23:44:39 +0000264 PyObject_INIT(f, &PyFrame_Type);
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000265 }
Guido van Rossum404b95d1997-08-05 02:09:46 +0000266 if (builtins == NULL) {
Guido van Rossumbde6ff71998-02-19 20:48:26 +0000267 /* No builtins! Make up a minimal one. */
Guido van Rossum404b95d1997-08-05 02:09:46 +0000268 builtins = PyDict_New();
Guido van Rossumf61618c1998-10-19 14:20:20 +0000269 if (builtins == NULL || /* Give them 'None', at least. */
270 PyDict_SetItemString(builtins, "None", Py_None) < 0) {
271 Py_DECREF(f);
Guido van Rossum404b95d1997-08-05 02:09:46 +0000272 return NULL;
Guido van Rossumf61618c1998-10-19 14:20:20 +0000273 }
Guido van Rossum404b95d1997-08-05 02:09:46 +0000274 }
275 else
276 Py_XINCREF(builtins);
277 f->f_builtins = builtins;
Guido van Rossum18752471997-04-29 14:49:28 +0000278 Py_XINCREF(back);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000279 f->f_back = back;
Guido van Rossum18752471997-04-29 14:49:28 +0000280 Py_INCREF(code);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000281 f->f_code = code;
Guido van Rossum18752471997-04-29 14:49:28 +0000282 Py_INCREF(globals);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000283 f->f_globals = globals;
Guido van Rossumbdd207a1995-07-26 16:14:30 +0000284 if (code->co_flags & CO_NEWLOCALS) {
285 if (code->co_flags & CO_OPTIMIZED)
286 locals = NULL; /* Let fast_2_locals handle it */
287 else {
Guido van Rossum18752471997-04-29 14:49:28 +0000288 locals = PyDict_New();
Guido van Rossumbdd207a1995-07-26 16:14:30 +0000289 if (locals == NULL) {
Guido van Rossum18752471997-04-29 14:49:28 +0000290 Py_DECREF(f);
Guido van Rossumbdd207a1995-07-26 16:14:30 +0000291 return NULL;
292 }
Guido van Rossum3f5da241990-12-20 15:06:42 +0000293 }
Guido van Rossum3f5da241990-12-20 15:06:42 +0000294 }
Guido van Rossum2271bf71995-07-18 14:30:34 +0000295 else {
296 if (locals == NULL)
297 locals = globals;
Guido van Rossum18752471997-04-29 14:49:28 +0000298 Py_INCREF(locals);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000299 }
300 f->f_locals = locals;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000301 f->f_trace = NULL;
Guido van Rossuma027efa1997-05-05 20:56:21 +0000302 f->f_exc_type = f->f_exc_value = f->f_exc_traceback = NULL;
Guido van Rossumeb46d671997-08-02 02:59:08 +0000303 f->f_tstate = tstate;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000304
Guido van Rossum2271bf71995-07-18 14:30:34 +0000305 f->f_lasti = 0;
Guido van Rossum747596a1997-01-24 04:00:21 +0000306 f->f_lineno = code->co_firstlineno;
Guido van Rossumeb46d671997-08-02 02:59:08 +0000307 f->f_restricted = (builtins != tstate->interp->builtins);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000308 f->f_iblock = 0;
309 f->f_nlocals = code->co_nlocals;
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000310 f->f_stacksize = code->co_stacksize;
311 f->f_ncells = ncells;
312 f->f_nfreevars = nfrees;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000313
314 while (--extras >= 0)
315 f->f_localsplus[extras] = NULL;
316
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000317 f->f_valuestack = f->f_localsplus + (f->f_nlocals + ncells + nfrees);
Tim Peters8c963692001-06-23 05:26:56 +0000318 f->f_stacktop = f->f_valuestack;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000319
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000320 PyObject_GC_Init(f);
Guido van Rossum3f5da241990-12-20 15:06:42 +0000321 return f;
322}
323
Guido van Rossum3f5da241990-12-20 15:06:42 +0000324/* Block management */
325
326void
Fred Drake1b190b42000-07-09 05:40:56 +0000327PyFrame_BlockSetup(PyFrameObject *f, int type, int handler, int level)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000328{
Guido van Rossum18752471997-04-29 14:49:28 +0000329 PyTryBlock *b;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000330 if (f->f_iblock >= CO_MAXBLOCKS)
Guido van Rossum18752471997-04-29 14:49:28 +0000331 Py_FatalError("XXX block stack overflow");
Guido van Rossum3f5da241990-12-20 15:06:42 +0000332 b = &f->f_blockstack[f->f_iblock++];
333 b->b_type = type;
334 b->b_level = level;
335 b->b_handler = handler;
336}
337
Guido van Rossum18752471997-04-29 14:49:28 +0000338PyTryBlock *
Fred Drake1b190b42000-07-09 05:40:56 +0000339PyFrame_BlockPop(PyFrameObject *f)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000340{
Guido van Rossum18752471997-04-29 14:49:28 +0000341 PyTryBlock *b;
Guido van Rossumd7047b31995-01-02 19:07:15 +0000342 if (f->f_iblock <= 0)
Guido van Rossum18752471997-04-29 14:49:28 +0000343 Py_FatalError("XXX block stack underflow");
Guido van Rossum3f5da241990-12-20 15:06:42 +0000344 b = &f->f_blockstack[--f->f_iblock];
345 return b;
346}
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000347
348/* Convert between "fast" version of locals and dictionary version */
349
Guido van Rossumf68d8e52001-04-14 17:55:09 +0000350static void
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000351map_to_dict(PyObject *map, int nmap, PyObject *dict, PyObject **values,
352 int deref)
353{
354 int j;
355 for (j = nmap; --j >= 0; ) {
356 PyObject *key = PyTuple_GetItem(map, j);
357 PyObject *value = values[j];
358 if (deref)
359 value = PyCell_GET(value);
360 if (value == NULL) {
361 PyErr_Clear();
362 if (PyDict_DelItem(dict, key) != 0)
363 PyErr_Clear();
364 }
365 else {
366 if (PyDict_SetItem(dict, key, value) != 0)
367 PyErr_Clear();
368 }
369 }
370}
371
Guido van Rossum6b356e72001-04-14 17:55:41 +0000372static void
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000373dict_to_map(PyObject *map, int nmap, PyObject *dict, PyObject **values,
374 int deref, int clear)
375{
376 int j;
377 for (j = nmap; --j >= 0; ) {
378 PyObject *key = PyTuple_GetItem(map, j);
379 PyObject *value = PyDict_GetItem(dict, key);
380 Py_XINCREF(value);
381 if (deref) {
Jeremy Hylton4c889012001-05-08 04:08:59 +0000382 if (value || clear) {
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000383 if (PyCell_Set(values[j], value) < 0)
384 PyErr_Clear();
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000385 }
386 } else if (value != NULL || clear) {
387 Py_XDECREF(values[j]);
388 values[j] = value;
389 }
390 }
391}
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000392
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000393void
Fred Drake1b190b42000-07-09 05:40:56 +0000394PyFrame_FastToLocals(PyFrameObject *f)
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000395{
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000396 /* Merge fast locals into f->f_locals */
Guido van Rossum18752471997-04-29 14:49:28 +0000397 PyObject *locals, *map;
398 PyObject **fast;
399 PyObject *error_type, *error_value, *error_traceback;
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000400 int j;
401 if (f == NULL)
402 return;
Guido van Rossum2271bf71995-07-18 14:30:34 +0000403 locals = f->f_locals;
404 if (locals == NULL) {
Guido van Rossum18752471997-04-29 14:49:28 +0000405 locals = f->f_locals = PyDict_New();
Guido van Rossum2271bf71995-07-18 14:30:34 +0000406 if (locals == NULL) {
Guido van Rossum18752471997-04-29 14:49:28 +0000407 PyErr_Clear(); /* Can't report it :-( */
Guido van Rossum2271bf71995-07-18 14:30:34 +0000408 return;
409 }
410 }
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000411 if (f->f_nlocals == 0)
Guido van Rossumbdd207a1995-07-26 16:14:30 +0000412 return;
413 map = f->f_code->co_varnames;
Guido van Rossum18752471997-04-29 14:49:28 +0000414 if (!PyDict_Check(locals) || !PyTuple_Check(map))
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000415 return;
Guido van Rossum18752471997-04-29 14:49:28 +0000416 PyErr_Fetch(&error_type, &error_value, &error_traceback);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000417 fast = f->f_localsplus;
Guido van Rossum18752471997-04-29 14:49:28 +0000418 j = PyTuple_Size(map);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000419 if (j > f->f_nlocals)
420 j = f->f_nlocals;
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000421 map_to_dict(map, j, locals, fast, 0);
422 if (f->f_ncells || f->f_nfreevars) {
423 if (!(PyTuple_Check(f->f_code->co_cellvars)
424 && PyTuple_Check(f->f_code->co_freevars))) {
425 Py_DECREF(locals);
426 return;
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000427 }
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000428 map_to_dict(f->f_code->co_cellvars,
429 PyTuple_GET_SIZE(f->f_code->co_cellvars),
430 locals, fast + f->f_nlocals, 1);
431 map_to_dict(f->f_code->co_freevars,
432 PyTuple_GET_SIZE(f->f_code->co_freevars),
433 locals, fast + f->f_nlocals + f->f_ncells, 1);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000434 }
Guido van Rossum18752471997-04-29 14:49:28 +0000435 PyErr_Restore(error_type, error_value, error_traceback);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000436}
437
438void
Fred Drake1b190b42000-07-09 05:40:56 +0000439PyFrame_LocalsToFast(PyFrameObject *f, int clear)
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000440{
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000441 /* Merge f->f_locals into fast locals */
Guido van Rossum18752471997-04-29 14:49:28 +0000442 PyObject *locals, *map;
443 PyObject **fast;
444 PyObject *error_type, *error_value, *error_traceback;
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000445 int j;
446 if (f == NULL)
447 return;
448 locals = f->f_locals;
Guido van Rossum2271bf71995-07-18 14:30:34 +0000449 map = f->f_code->co_varnames;
Guido van Rossum3bb63a81997-01-20 04:29:16 +0000450 if (locals == NULL || f->f_code->co_nlocals == 0)
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000451 return;
Guido van Rossum18752471997-04-29 14:49:28 +0000452 if (!PyDict_Check(locals) || !PyTuple_Check(map))
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000453 return;
Guido van Rossum18752471997-04-29 14:49:28 +0000454 PyErr_Fetch(&error_type, &error_value, &error_traceback);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000455 fast = f->f_localsplus;
Guido van Rossum18752471997-04-29 14:49:28 +0000456 j = PyTuple_Size(map);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000457 if (j > f->f_nlocals)
458 j = f->f_nlocals;
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000459 dict_to_map(f->f_code->co_varnames, j, locals, fast, 0, clear);
460 if (f->f_ncells || f->f_nfreevars) {
461 if (!(PyTuple_Check(f->f_code->co_cellvars)
462 && PyTuple_Check(f->f_code->co_freevars)))
463 return;
464 dict_to_map(f->f_code->co_cellvars,
465 PyTuple_GET_SIZE(f->f_code->co_cellvars),
Jeremy Hylton4c889012001-05-08 04:08:59 +0000466 locals, fast + f->f_nlocals, 1, clear);
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000467 dict_to_map(f->f_code->co_freevars,
468 PyTuple_GET_SIZE(f->f_code->co_freevars),
Jeremy Hylton4c889012001-05-08 04:08:59 +0000469 locals, fast + f->f_nlocals + f->f_ncells, 1, clear);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000470 }
Guido van Rossum18752471997-04-29 14:49:28 +0000471 PyErr_Restore(error_type, error_value, error_traceback);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000472}
Guido van Rossum404b95d1997-08-05 02:09:46 +0000473
474/* Clear out the free list */
475
476void
Fred Drake1b190b42000-07-09 05:40:56 +0000477PyFrame_Fini(void)
Guido van Rossum404b95d1997-08-05 02:09:46 +0000478{
479 while (free_list != NULL) {
480 PyFrameObject *f = free_list;
481 free_list = free_list->f_back;
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000482 f = (PyFrameObject *) PyObject_AS_GC(f);
Guido van Rossumb18618d2000-05-03 23:44:39 +0000483 PyObject_DEL(f);
Guido van Rossum404b95d1997-08-05 02:09:46 +0000484 }
485}