mirror of
				https://github.com/python/cpython.git
				synced 2025-10-25 10:44:55 +00:00 
			
		
		
		
	 135cabd328
			
		
	
	
		135cabd328
		
			
		
	
	
	
	
		
			
			* Make internal APIs that take PyFrameConstructor take a PyFunctionObject instead. * Add reference to function to frame, borrow references to builtins and globals. * Add COPY_FREE_VARS instruction to allow specialization of calls to inner functions.
		
			
				
	
	
		
			128 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			128 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| 
 | |
| #include "Python.h"
 | |
| #include "frameobject.h"
 | |
| #include "pycore_frame.h"
 | |
| #include "pycore_object.h"        // _PyObject_GC_UNTRACK()
 | |
| 
 | |
| int
 | |
| _PyFrame_Traverse(InterpreterFrame *frame, visitproc visit, void *arg)
 | |
| {
 | |
|     Py_VISIT(frame->frame_obj);
 | |
|     Py_VISIT(frame->f_locals);
 | |
|     Py_VISIT(frame->f_func);
 | |
|     Py_VISIT(frame->f_code);
 | |
|    /* locals */
 | |
|     PyObject **locals = _PyFrame_GetLocalsArray(frame);
 | |
|     int i = 0;
 | |
|     /* locals and stack */
 | |
|     for (; i <frame->stacktop; i++) {
 | |
|         Py_VISIT(locals[i]);
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| PyFrameObject *
 | |
| _PyFrame_MakeAndSetFrameObject(InterpreterFrame *frame)
 | |
| {
 | |
|     assert(frame->frame_obj == NULL);
 | |
|     PyObject *error_type, *error_value, *error_traceback;
 | |
|     PyErr_Fetch(&error_type, &error_value, &error_traceback);
 | |
|     PyFrameObject *f = _PyFrame_New_NoTrack(frame, 0);
 | |
|     if (f == NULL) {
 | |
|         Py_XDECREF(error_type);
 | |
|         Py_XDECREF(error_value);
 | |
|         Py_XDECREF(error_traceback);
 | |
|     }
 | |
|     else {
 | |
|         PyErr_Restore(error_type, error_value, error_traceback);
 | |
|     }
 | |
|     frame->frame_obj = f;
 | |
|     return f;
 | |
| }
 | |
| 
 | |
| 
 | |
| static InterpreterFrame *
 | |
| copy_frame_to_heap(InterpreterFrame *frame)
 | |
| {
 | |
|     assert(frame->stacktop >= frame->f_code->co_nlocalsplus);
 | |
|     Py_ssize_t size = ((char*)&frame->localsplus[frame->stacktop]) - (char *)frame;
 | |
|     InterpreterFrame *copy = PyMem_Malloc(size);
 | |
|     if (copy == NULL) {
 | |
|         PyErr_NoMemory();
 | |
|         return NULL;
 | |
|     }
 | |
|     memcpy(copy, frame, size);
 | |
|     return copy;
 | |
| }
 | |
| 
 | |
| static inline void
 | |
| clear_specials(InterpreterFrame *frame)
 | |
| {
 | |
|     frame->generator = NULL;
 | |
|     Py_XDECREF(frame->frame_obj);
 | |
|     Py_XDECREF(frame->f_locals);
 | |
|     Py_DECREF(frame->f_func);
 | |
|     Py_DECREF(frame->f_code);
 | |
| }
 | |
| 
 | |
| static void
 | |
| take_ownership(PyFrameObject *f, InterpreterFrame *frame)
 | |
| {
 | |
|     assert(f->f_own_locals_memory == 0);
 | |
|     assert(frame->frame_obj == NULL);
 | |
| 
 | |
|     f->f_own_locals_memory = 1;
 | |
|     f->f_frame = frame;
 | |
|     assert(f->f_back == NULL);
 | |
|     if (frame->previous != NULL) {
 | |
|         /* Link PyFrameObjects.f_back and remove link through InterpreterFrame.previous */
 | |
|         PyFrameObject *back = _PyFrame_GetFrameObject(frame->previous);
 | |
|         if (back == NULL) {
 | |
|             /* Memory error here. */
 | |
|             assert(PyErr_ExceptionMatches(PyExc_MemoryError));
 | |
|             /* Nothing we can do about it */
 | |
|             PyErr_Clear();
 | |
|             _PyErr_WriteUnraisableMsg("Out of memory lazily allocating frame->f_back", NULL);
 | |
|         }
 | |
|         else {
 | |
|             f->f_back = (PyFrameObject *)Py_NewRef(back);
 | |
|         }
 | |
|         frame->previous = NULL;
 | |
|     }
 | |
|     if (!_PyObject_GC_IS_TRACKED((PyObject *)f)) {
 | |
|         _PyObject_GC_TRACK((PyObject *)f);
 | |
|     }
 | |
| }
 | |
| 
 | |
| int
 | |
| _PyFrame_Clear(InterpreterFrame * frame, int take)
 | |
| {
 | |
|     /* It is the responsibility of the owning generator/coroutine
 | |
|      * to have cleared the generator pointer */
 | |
|     assert(frame->generator == NULL);
 | |
|     if (frame->frame_obj) {
 | |
|         PyFrameObject *f = frame->frame_obj;
 | |
|         frame->frame_obj = NULL;
 | |
|         if (Py_REFCNT(f) > 1) {
 | |
|             if (!take) {
 | |
|                 frame = copy_frame_to_heap(frame);
 | |
|                 if (frame == NULL) {
 | |
|                     return -1;
 | |
|                 }
 | |
|             }
 | |
|             take_ownership(f, frame);
 | |
|             Py_DECREF(f);
 | |
|             return 0;
 | |
|         }
 | |
|         Py_DECREF(f);
 | |
|     }
 | |
|     assert(_PyFrame_GetStackPointer(frame) >= _PyFrame_Stackbase(frame));
 | |
|     for (int i = 0; i < frame->stacktop; i++) {
 | |
|         Py_XDECREF(frame->localsplus[i]);
 | |
|     }
 | |
|     clear_specials(frame);
 | |
|     if (take) {
 | |
|         PyMem_Free(frame);
 | |
|     }
 | |
|     return 0;
 | |
| }
 |