blob: 165121d8f8715505ff3254e28a9fd23136438c95 [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},
19 {"f_lineno", T_INT, OFF(f_lineno), RO},
Guido van Rossumc1134821995-01-10 10:39:16 +000020 {"f_restricted",T_INT, OFF(f_restricted),RO},
Guido van Rossum1d5735e1994-08-30 08:27:36 +000021 {"f_trace", T_OBJECT, OFF(f_trace)},
Guido van Rossuma027efa1997-05-05 20:56:21 +000022 {"f_exc_type", T_OBJECT, OFF(f_exc_type)},
23 {"f_exc_value", T_OBJECT, OFF(f_exc_value)},
24 {"f_exc_traceback", T_OBJECT, OFF(f_exc_traceback)},
Guido van Rossum3f5da241990-12-20 15:06:42 +000025 {NULL} /* Sentinel */
26};
27
Guido van Rossum18752471997-04-29 14:49:28 +000028static PyObject *
Tim Peters6d6c1a32001-08-02 04:15:00 +000029frame_getlocals(PyFrameObject *f, void *closure)
Guido van Rossum3f5da241990-12-20 15:06:42 +000030{
Tim Peters6d6c1a32001-08-02 04:15:00 +000031 PyFrame_FastToLocals(f);
32 Py_INCREF(f->f_locals);
33 return f->f_locals;
Guido van Rossum3f5da241990-12-20 15:06:42 +000034}
35
Guido van Rossum32d34c82001-09-20 21:45:26 +000036static PyGetSetDef frame_getsetlist[] = {
Tim Peters6d6c1a32001-08-02 04:15:00 +000037 {"f_locals", (getter)frame_getlocals, NULL, NULL},
38 {0}
39};
Guido van Rossum1d5735e1994-08-30 08:27:36 +000040
Guido van Rossuma9e7dc11992-10-18 18:53:57 +000041/* Stack frames are allocated and deallocated at a considerable rate.
42 In an attempt to improve the speed of function calls, we maintain a
43 separate free list of stack frames (just like integers are
44 allocated in a special way -- see intobject.c). When a stack frame
45 is on the free list, only the following members have a meaning:
46 ob_type == &Frametype
47 f_back next item on free list, or NULL
Guido van Rossumf3e85a01997-01-20 04:20:52 +000048 f_nlocals number of locals
49 f_stacksize size of value stack
Neil Schemenauer4f4817f2001-08-29 23:52:17 +000050 ob_size size of localsplus
Guido van Rossuma9e7dc11992-10-18 18:53:57 +000051 Note that the value and block stacks are preserved -- this can save
52 another malloc() call or two (and two free() calls as well!).
53 Also note that, unlike for integers, each frame object is a
54 malloc'ed object in its own right -- it is only the actual calls to
55 malloc() that we are trying to save here, not the administration.
56 After all, while a typical program may make millions of calls, a
57 call depth of more than 20 or 30 is probably already exceptional
58 unless the program contains run-away recursion. I hope.
Tim Petersb7ba7432002-04-13 05:21:47 +000059
60 Later, MAXFREELIST was added to bound the # of frames saved on
61 free_list. Else programs creating lots of cyclic trash involving
62 frames could provoke free_list into growing without bound.
Guido van Rossuma9e7dc11992-10-18 18:53:57 +000063*/
64
Guido van Rossum18752471997-04-29 14:49:28 +000065static PyFrameObject *free_list = NULL;
Tim Petersb7ba7432002-04-13 05:21:47 +000066static int numfree = 0; /* number of frames currently in free_list */
67#define MAXFREELIST 200 /* max value for numfree */
Guido van Rossuma9e7dc11992-10-18 18:53:57 +000068
Guido van Rossum3f5da241990-12-20 15:06:42 +000069static void
Fred Drake1b190b42000-07-09 05:40:56 +000070frame_dealloc(PyFrameObject *f)
Guido van Rossum3f5da241990-12-20 15:06:42 +000071{
Jeremy Hylton30c9f392001-03-13 01:58:22 +000072 int i, slots;
Guido van Rossum7582bfb1997-02-14 16:27:29 +000073 PyObject **fastlocals;
Tim Peters5ca576e2001-06-18 22:08:13 +000074 PyObject **p;
Guido van Rossum7582bfb1997-02-14 16:27:29 +000075
Guido van Rossumff413af2002-03-28 20:34:59 +000076 PyObject_GC_UnTrack(f);
Guido van Rossumd724b232000-03-13 16:01:29 +000077 Py_TRASHCAN_SAFE_BEGIN(f)
Guido van Rossum7582bfb1997-02-14 16:27:29 +000078 /* Kill all local variables */
Jeremy Hylton30c9f392001-03-13 01:58:22 +000079 slots = f->f_nlocals + f->f_ncells + f->f_nfreevars;
Guido van Rossum7582bfb1997-02-14 16:27:29 +000080 fastlocals = f->f_localsplus;
Jeremy Hylton30c9f392001-03-13 01:58:22 +000081 for (i = slots; --i >= 0; ++fastlocals) {
Guido van Rossum18752471997-04-29 14:49:28 +000082 Py_XDECREF(*fastlocals);
Guido van Rossum7582bfb1997-02-14 16:27:29 +000083 }
84
Tim Peters5ca576e2001-06-18 22:08:13 +000085 /* Free stack */
Tim Peters8c963692001-06-23 05:26:56 +000086 if (f->f_stacktop != NULL) {
87 for (p = f->f_valuestack; p < f->f_stacktop; p++)
88 Py_XDECREF(*p);
Tim Peters5ca576e2001-06-18 22:08:13 +000089 }
Tim Peters8c963692001-06-23 05:26:56 +000090
Guido van Rossum18752471997-04-29 14:49:28 +000091 Py_XDECREF(f->f_back);
92 Py_XDECREF(f->f_code);
93 Py_XDECREF(f->f_builtins);
94 Py_XDECREF(f->f_globals);
95 Py_XDECREF(f->f_locals);
96 Py_XDECREF(f->f_trace);
Guido van Rossuma027efa1997-05-05 20:56:21 +000097 Py_XDECREF(f->f_exc_type);
98 Py_XDECREF(f->f_exc_value);
99 Py_XDECREF(f->f_exc_traceback);
Tim Petersb7ba7432002-04-13 05:21:47 +0000100 if (numfree < MAXFREELIST) {
101 ++numfree;
102 f->f_back = free_list;
103 free_list = f;
104 }
105 else
106 PyObject_GC_Del(f);
Guido van Rossumd724b232000-03-13 16:01:29 +0000107 Py_TRASHCAN_SAFE_END(f)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000108}
109
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000110static int
111frame_traverse(PyFrameObject *f, visitproc visit, void *arg)
112{
113 PyObject **fastlocals, **p;
114 int i, err, slots;
115#define VISIT(o) if (o) {if ((err = visit((PyObject *)(o), arg))) return err;}
116
117 VISIT(f->f_back);
118 VISIT(f->f_code);
119 VISIT(f->f_builtins);
120 VISIT(f->f_globals);
121 VISIT(f->f_locals);
122 VISIT(f->f_trace);
123 VISIT(f->f_exc_type);
124 VISIT(f->f_exc_value);
125 VISIT(f->f_exc_traceback);
126
127 /* locals */
128 slots = f->f_nlocals + f->f_ncells + f->f_nfreevars;
129 fastlocals = f->f_localsplus;
130 for (i = slots; --i >= 0; ++fastlocals) {
131 VISIT(*fastlocals);
132 }
133
134 /* stack */
135 if (f->f_stacktop != NULL) {
136 for (p = f->f_valuestack; p < f->f_stacktop; p++)
137 VISIT(*p);
138 }
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000139 return 0;
140}
141
142static void
143frame_clear(PyFrameObject *f)
144{
145 PyObject **fastlocals, **p;
146 int i, slots;
147
148 Py_XDECREF(f->f_exc_type);
149 f->f_exc_type = NULL;
150
151 Py_XDECREF(f->f_exc_value);
152 f->f_exc_value = NULL;
153
154 Py_XDECREF(f->f_exc_traceback);
155 f->f_exc_traceback = NULL;
156
157 Py_XDECREF(f->f_trace);
158 f->f_trace = NULL;
159
160 /* locals */
161 slots = f->f_nlocals + f->f_ncells + f->f_nfreevars;
162 fastlocals = f->f_localsplus;
163 for (i = slots; --i >= 0; ++fastlocals) {
164 if (*fastlocals != NULL) {
165 Py_XDECREF(*fastlocals);
166 *fastlocals = NULL;
167 }
168 }
169
170 /* stack */
171 if (f->f_stacktop != NULL) {
172 for (p = f->f_valuestack; p < f->f_stacktop; p++) {
173 Py_XDECREF(*p);
174 *p = NULL;
175 }
176 }
177}
178
179
Guido van Rossum18752471997-04-29 14:49:28 +0000180PyTypeObject PyFrame_Type = {
181 PyObject_HEAD_INIT(&PyType_Type)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000182 0,
183 "frame",
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000184 sizeof(PyFrameObject),
185 sizeof(PyObject *),
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000186 (destructor)frame_dealloc, /* tp_dealloc */
187 0, /* tp_print */
Tim Peters6d6c1a32001-08-02 04:15:00 +0000188 0, /* tp_getattr */
189 0, /* tp_setattr */
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000190 0, /* tp_compare */
191 0, /* tp_repr */
192 0, /* tp_as_number */
193 0, /* tp_as_sequence */
194 0, /* tp_as_mapping */
195 0, /* tp_hash */
196 0, /* tp_call */
197 0, /* tp_str */
Tim Peters6d6c1a32001-08-02 04:15:00 +0000198 PyObject_GenericGetAttr, /* tp_getattro */
199 PyObject_GenericSetAttr, /* tp_setattro */
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000200 0, /* tp_as_buffer */
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000201 Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC,/* tp_flags */
Neil Schemenauer19cd2922001-07-12 13:27:11 +0000202 0, /* tp_doc */
203 (traverseproc)frame_traverse, /* tp_traverse */
204 (inquiry)frame_clear, /* tp_clear */
Tim Peters6d6c1a32001-08-02 04:15:00 +0000205 0, /* tp_richcompare */
206 0, /* tp_weaklistoffset */
207 0, /* tp_iter */
208 0, /* tp_iternext */
209 0, /* tp_methods */
210 frame_memberlist, /* tp_members */
211 frame_getsetlist, /* tp_getset */
212 0, /* tp_base */
213 0, /* tp_dict */
Guido van Rossum3f5da241990-12-20 15:06:42 +0000214};
215
Guido van Rossum18752471997-04-29 14:49:28 +0000216PyFrameObject *
Jeremy Hylton64949cb2001-01-25 20:06:59 +0000217PyFrame_New(PyThreadState *tstate, PyCodeObject *code, PyObject *globals,
Jeremy Hylton30c9f392001-03-13 01:58:22 +0000218 PyObject *locals)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000219{
Guido van Rossuma027efa1997-05-05 20:56:21 +0000220 PyFrameObject *back = tstate->frame;
Guido van Rossum18752471997-04-29 14:49:28 +0000221 static PyObject *builtin_object;
222 PyFrameObject *f;
223 PyObject *builtins;
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000224 int extras, ncells, nfrees;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000225
Sjoerd Mullender5b7f3cd1995-04-04 11:47:41 +0000226 if (builtin_object == NULL) {
Guido van Rossumb56933e1997-01-18 07:58:41 +0000227 builtin_object = PyString_InternFromString("__builtins__");
Sjoerd Mullender5b7f3cd1995-04-04 11:47:41 +0000228 if (builtin_object == NULL)
229 return NULL;
230 }
Guido van Rossum18752471997-04-29 14:49:28 +0000231 if ((back != NULL && !PyFrame_Check(back)) ||
232 code == NULL || !PyCode_Check(code) ||
233 globals == NULL || !PyDict_Check(globals) ||
234 (locals != NULL && !PyDict_Check(locals))) {
235 PyErr_BadInternalCall();
Guido van Rossum3f5da241990-12-20 15:06:42 +0000236 return NULL;
237 }
Jeremy Hylton64949cb2001-01-25 20:06:59 +0000238 ncells = PyTuple_GET_SIZE(code->co_cellvars);
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000239 nfrees = PyTuple_GET_SIZE(code->co_freevars);
240 extras = code->co_stacksize + code->co_nlocals + ncells + nfrees;
Guido van Rossumbde6ff71998-02-19 20:48:26 +0000241 if (back == NULL || back->f_globals != globals) {
242 builtins = PyDict_GetItem(globals, builtin_object);
243 if (builtins != NULL && PyModule_Check(builtins))
244 builtins = PyModule_GetDict(builtins);
245 }
246 else {
247 /* If we share the globals, we share the builtins.
248 Save a lookup and a call. */
249 builtins = back->f_builtins;
250 }
Guido van Rossum404b95d1997-08-05 02:09:46 +0000251 if (builtins != NULL && !PyDict_Check(builtins))
252 builtins = NULL;
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000253 if (free_list == NULL) {
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000254 f = PyObject_GC_NewVar(PyFrameObject, &PyFrame_Type, extras);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000255 if (f == NULL)
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000256 return NULL;
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000257 }
258 else {
Tim Petersb7ba7432002-04-13 05:21:47 +0000259 assert(numfree > 0);
260 --numfree;
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000261 f = free_list;
262 free_list = free_list->f_back;
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000263 if (f->ob_size < extras) {
264 f = PyObject_GC_Resize(PyFrameObject, f, extras);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000265 if (f == NULL)
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000266 return NULL;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000267 }
Guido van Rossum747596a1997-01-24 04:00:21 +0000268 else
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000269 extras = f->ob_size;
Tim Petersdeb77e82001-08-30 00:32:51 +0000270 _Py_NewReference((PyObject *)f);
Guido van Rossuma9e7dc11992-10-18 18:53:57 +0000271 }
Guido van Rossum404b95d1997-08-05 02:09:46 +0000272 if (builtins == NULL) {
Guido van Rossumbde6ff71998-02-19 20:48:26 +0000273 /* No builtins! Make up a minimal one. */
Guido van Rossum404b95d1997-08-05 02:09:46 +0000274 builtins = PyDict_New();
Guido van Rossumf61618c1998-10-19 14:20:20 +0000275 if (builtins == NULL || /* Give them 'None', at least. */
276 PyDict_SetItemString(builtins, "None", Py_None) < 0) {
277 Py_DECREF(f);
Guido van Rossum404b95d1997-08-05 02:09:46 +0000278 return NULL;
Guido van Rossumf61618c1998-10-19 14:20:20 +0000279 }
Guido van Rossum404b95d1997-08-05 02:09:46 +0000280 }
281 else
282 Py_XINCREF(builtins);
283 f->f_builtins = builtins;
Guido van Rossum18752471997-04-29 14:49:28 +0000284 Py_XINCREF(back);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000285 f->f_back = back;
Guido van Rossum18752471997-04-29 14:49:28 +0000286 Py_INCREF(code);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000287 f->f_code = code;
Guido van Rossum18752471997-04-29 14:49:28 +0000288 Py_INCREF(globals);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000289 f->f_globals = globals;
Guido van Rossumbdd207a1995-07-26 16:14:30 +0000290 if (code->co_flags & CO_NEWLOCALS) {
291 if (code->co_flags & CO_OPTIMIZED)
292 locals = NULL; /* Let fast_2_locals handle it */
293 else {
Guido van Rossum18752471997-04-29 14:49:28 +0000294 locals = PyDict_New();
Guido van Rossumbdd207a1995-07-26 16:14:30 +0000295 if (locals == NULL) {
Guido van Rossum18752471997-04-29 14:49:28 +0000296 Py_DECREF(f);
Guido van Rossumbdd207a1995-07-26 16:14:30 +0000297 return NULL;
298 }
Guido van Rossum3f5da241990-12-20 15:06:42 +0000299 }
Guido van Rossum3f5da241990-12-20 15:06:42 +0000300 }
Guido van Rossum2271bf71995-07-18 14:30:34 +0000301 else {
302 if (locals == NULL)
303 locals = globals;
Guido van Rossum18752471997-04-29 14:49:28 +0000304 Py_INCREF(locals);
Guido van Rossum2271bf71995-07-18 14:30:34 +0000305 }
306 f->f_locals = locals;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000307 f->f_trace = NULL;
Guido van Rossuma027efa1997-05-05 20:56:21 +0000308 f->f_exc_type = f->f_exc_value = f->f_exc_traceback = NULL;
Guido van Rossumeb46d671997-08-02 02:59:08 +0000309 f->f_tstate = tstate;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000310
Guido van Rossum2271bf71995-07-18 14:30:34 +0000311 f->f_lasti = 0;
Guido van Rossum747596a1997-01-24 04:00:21 +0000312 f->f_lineno = code->co_firstlineno;
Guido van Rossumeb46d671997-08-02 02:59:08 +0000313 f->f_restricted = (builtins != tstate->interp->builtins);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000314 f->f_iblock = 0;
315 f->f_nlocals = code->co_nlocals;
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000316 f->f_stacksize = code->co_stacksize;
317 f->f_ncells = ncells;
318 f->f_nfreevars = nfrees;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000319
320 while (--extras >= 0)
321 f->f_localsplus[extras] = NULL;
322
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000323 f->f_valuestack = f->f_localsplus + (f->f_nlocals + ncells + nfrees);
Tim Peters8c963692001-06-23 05:26:56 +0000324 f->f_stacktop = f->f_valuestack;
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000325 _PyObject_GC_TRACK(f);
Guido van Rossum3f5da241990-12-20 15:06:42 +0000326 return f;
327}
328
Guido van Rossum3f5da241990-12-20 15:06:42 +0000329/* Block management */
330
331void
Fred Drake1b190b42000-07-09 05:40:56 +0000332PyFrame_BlockSetup(PyFrameObject *f, int type, int handler, int level)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000333{
Guido van Rossum18752471997-04-29 14:49:28 +0000334 PyTryBlock *b;
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000335 if (f->f_iblock >= CO_MAXBLOCKS)
Guido van Rossum18752471997-04-29 14:49:28 +0000336 Py_FatalError("XXX block stack overflow");
Guido van Rossum3f5da241990-12-20 15:06:42 +0000337 b = &f->f_blockstack[f->f_iblock++];
338 b->b_type = type;
339 b->b_level = level;
340 b->b_handler = handler;
341}
342
Guido van Rossum18752471997-04-29 14:49:28 +0000343PyTryBlock *
Fred Drake1b190b42000-07-09 05:40:56 +0000344PyFrame_BlockPop(PyFrameObject *f)
Guido van Rossum3f5da241990-12-20 15:06:42 +0000345{
Guido van Rossum18752471997-04-29 14:49:28 +0000346 PyTryBlock *b;
Guido van Rossumd7047b31995-01-02 19:07:15 +0000347 if (f->f_iblock <= 0)
Guido van Rossum18752471997-04-29 14:49:28 +0000348 Py_FatalError("XXX block stack underflow");
Guido van Rossum3f5da241990-12-20 15:06:42 +0000349 b = &f->f_blockstack[--f->f_iblock];
350 return b;
351}
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000352
353/* Convert between "fast" version of locals and dictionary version */
354
Guido van Rossumf68d8e52001-04-14 17:55:09 +0000355static void
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000356map_to_dict(PyObject *map, int nmap, PyObject *dict, PyObject **values,
357 int deref)
358{
359 int j;
360 for (j = nmap; --j >= 0; ) {
Jeremy Hylton1a48ca82001-12-06 15:48:16 +0000361 PyObject *key = PyTuple_GET_ITEM(map, j);
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000362 PyObject *value = values[j];
363 if (deref)
364 value = PyCell_GET(value);
365 if (value == NULL) {
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000366 if (PyDict_DelItem(dict, key) != 0)
367 PyErr_Clear();
368 }
369 else {
370 if (PyDict_SetItem(dict, key, value) != 0)
371 PyErr_Clear();
372 }
373 }
374}
375
Guido van Rossum6b356e72001-04-14 17:55:41 +0000376static void
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000377dict_to_map(PyObject *map, int nmap, PyObject *dict, PyObject **values,
378 int deref, int clear)
379{
380 int j;
381 for (j = nmap; --j >= 0; ) {
Jeremy Hylton1a48ca82001-12-06 15:48:16 +0000382 PyObject *key = PyTuple_GET_ITEM(map, j);
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000383 PyObject *value = PyDict_GetItem(dict, key);
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000384 if (deref) {
Jeremy Hylton4c889012001-05-08 04:08:59 +0000385 if (value || clear) {
Jeremy Hylton1a48ca82001-12-06 15:48:16 +0000386 if (PyCell_GET(values[j]) != value) {
387 if (PyCell_Set(values[j], value) < 0)
388 PyErr_Clear();
389 }
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000390 }
391 } else if (value != NULL || clear) {
Jeremy Hylton1a48ca82001-12-06 15:48:16 +0000392 if (values[j] != value) {
393 Py_XINCREF(value);
394 Py_XDECREF(values[j]);
395 values[j] = value;
396 }
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000397 }
398 }
399}
Jeremy Hylton2b724da2001-01-29 22:51:52 +0000400
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000401void
Fred Drake1b190b42000-07-09 05:40:56 +0000402PyFrame_FastToLocals(PyFrameObject *f)
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000403{
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000404 /* Merge fast locals into f->f_locals */
Guido van Rossum18752471997-04-29 14:49:28 +0000405 PyObject *locals, *map;
406 PyObject **fast;
407 PyObject *error_type, *error_value, *error_traceback;
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000408 int j;
409 if (f == NULL)
410 return;
Guido van Rossum2271bf71995-07-18 14:30:34 +0000411 locals = f->f_locals;
412 if (locals == NULL) {
Guido van Rossum18752471997-04-29 14:49:28 +0000413 locals = f->f_locals = PyDict_New();
Guido van Rossum2271bf71995-07-18 14:30:34 +0000414 if (locals == NULL) {
Guido van Rossum18752471997-04-29 14:49:28 +0000415 PyErr_Clear(); /* Can't report it :-( */
Guido van Rossum2271bf71995-07-18 14:30:34 +0000416 return;
417 }
418 }
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000419 if (f->f_nlocals == 0)
Guido van Rossumbdd207a1995-07-26 16:14:30 +0000420 return;
421 map = f->f_code->co_varnames;
Guido van Rossum18752471997-04-29 14:49:28 +0000422 if (!PyDict_Check(locals) || !PyTuple_Check(map))
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000423 return;
Guido van Rossum18752471997-04-29 14:49:28 +0000424 PyErr_Fetch(&error_type, &error_value, &error_traceback);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000425 fast = f->f_localsplus;
Guido van Rossum18752471997-04-29 14:49:28 +0000426 j = PyTuple_Size(map);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000427 if (j > f->f_nlocals)
428 j = f->f_nlocals;
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000429 map_to_dict(map, j, locals, fast, 0);
430 if (f->f_ncells || f->f_nfreevars) {
431 if (!(PyTuple_Check(f->f_code->co_cellvars)
432 && PyTuple_Check(f->f_code->co_freevars))) {
433 Py_DECREF(locals);
434 return;
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000435 }
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000436 map_to_dict(f->f_code->co_cellvars,
437 PyTuple_GET_SIZE(f->f_code->co_cellvars),
438 locals, fast + f->f_nlocals, 1);
439 map_to_dict(f->f_code->co_freevars,
440 PyTuple_GET_SIZE(f->f_code->co_freevars),
441 locals, fast + f->f_nlocals + f->f_ncells, 1);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000442 }
Guido van Rossum18752471997-04-29 14:49:28 +0000443 PyErr_Restore(error_type, error_value, error_traceback);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000444}
445
446void
Fred Drake1b190b42000-07-09 05:40:56 +0000447PyFrame_LocalsToFast(PyFrameObject *f, int clear)
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000448{
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000449 /* Merge f->f_locals into fast locals */
Guido van Rossum18752471997-04-29 14:49:28 +0000450 PyObject *locals, *map;
451 PyObject **fast;
452 PyObject *error_type, *error_value, *error_traceback;
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000453 int j;
454 if (f == NULL)
455 return;
456 locals = f->f_locals;
Guido van Rossum2271bf71995-07-18 14:30:34 +0000457 map = f->f_code->co_varnames;
Guido van Rossum3bb63a81997-01-20 04:29:16 +0000458 if (locals == NULL || f->f_code->co_nlocals == 0)
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000459 return;
Guido van Rossum18752471997-04-29 14:49:28 +0000460 if (!PyDict_Check(locals) || !PyTuple_Check(map))
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000461 return;
Guido van Rossum18752471997-04-29 14:49:28 +0000462 PyErr_Fetch(&error_type, &error_value, &error_traceback);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000463 fast = f->f_localsplus;
Guido van Rossum18752471997-04-29 14:49:28 +0000464 j = PyTuple_Size(map);
Guido van Rossumf3e85a01997-01-20 04:20:52 +0000465 if (j > f->f_nlocals)
466 j = f->f_nlocals;
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000467 dict_to_map(f->f_code->co_varnames, j, locals, fast, 0, clear);
468 if (f->f_ncells || f->f_nfreevars) {
469 if (!(PyTuple_Check(f->f_code->co_cellvars)
470 && PyTuple_Check(f->f_code->co_freevars)))
471 return;
472 dict_to_map(f->f_code->co_cellvars,
473 PyTuple_GET_SIZE(f->f_code->co_cellvars),
Jeremy Hylton4c889012001-05-08 04:08:59 +0000474 locals, fast + f->f_nlocals, 1, clear);
Jeremy Hylton220ae7c2001-03-21 16:43:47 +0000475 dict_to_map(f->f_code->co_freevars,
476 PyTuple_GET_SIZE(f->f_code->co_freevars),
Jeremy Hylton4c889012001-05-08 04:08:59 +0000477 locals, fast + f->f_nlocals + f->f_ncells, 1, clear);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000478 }
Guido van Rossum18752471997-04-29 14:49:28 +0000479 PyErr_Restore(error_type, error_value, error_traceback);
Guido van Rossum1d5735e1994-08-30 08:27:36 +0000480}
Guido van Rossum404b95d1997-08-05 02:09:46 +0000481
482/* Clear out the free list */
483
484void
Fred Drake1b190b42000-07-09 05:40:56 +0000485PyFrame_Fini(void)
Guido van Rossum404b95d1997-08-05 02:09:46 +0000486{
487 while (free_list != NULL) {
488 PyFrameObject *f = free_list;
489 free_list = free_list->f_back;
Neil Schemenauer4f4817f2001-08-29 23:52:17 +0000490 PyObject_GC_Del(f);
Tim Petersb7ba7432002-04-13 05:21:47 +0000491 --numfree;
Guido van Rossum404b95d1997-08-05 02:09:46 +0000492 }
Tim Petersb7ba7432002-04-13 05:21:47 +0000493 assert(numfree == 0);
Guido van Rossum404b95d1997-08-05 02:09:46 +0000494}