mirror of
				https://github.com/python/cpython.git
				synced 2025-10-26 11:14:33 +00:00 
			
		
		
		
	 1059548686
			
		
	
	
		1059548686
		
			
		
	
	
	
	
		
			
			This converts functions, code, str, bytes, bytearray, and memoryview objects to PyCodeObject,
and ensure that the object looks like a script.  That means no args, no return, and no closure.
_PyCode_GetPureScriptXIData() takes it a step further and ensures there are no globals.
We also add _PyObject_SupportedAsScript() to the internal C-API.
(cherry picked from commit c81fa2b9cd, AKA gh-133480)
Co-authored-by: Eric Snow <ericsnowcurrently@gmail.com>
		
	
			
		
			
				
	
	
		
			2659 lines
		
	
	
	
		
			70 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			2659 lines
		
	
	
	
		
			70 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| 
 | |
| /* API for managing interactions between isolated interpreters */
 | |
| 
 | |
| #include "Python.h"
 | |
| #include "marshal.h"              // PyMarshal_WriteObjectToString()
 | |
| #include "osdefs.h"               // MAXPATHLEN
 | |
| #include "pycore_ceval.h"         // _Py_simple_func
 | |
| #include "pycore_crossinterp.h"   // _PyXIData_t
 | |
| #include "pycore_function.h"      // _PyFunction_VerifyStateless()
 | |
| #include "pycore_initconfig.h"    // _PyStatus_OK()
 | |
| #include "pycore_namespace.h"     // _PyNamespace_New()
 | |
| #include "pycore_pythonrun.h"     // _Py_SourceAsString()
 | |
| #include "pycore_typeobject.h"    // _PyStaticType_InitBuiltin()
 | |
| 
 | |
| 
 | |
| static Py_ssize_t
 | |
| _Py_GetMainfile(char *buffer, size_t maxlen)
 | |
| {
 | |
|     // We don't expect subinterpreters to have the __main__ module's
 | |
|     // __name__ set, but proceed just in case.
 | |
|     PyThreadState *tstate = _PyThreadState_GET();
 | |
|     PyObject *module = _Py_GetMainModule(tstate);
 | |
|     if (_Py_CheckMainModule(module) < 0) {
 | |
|         return -1;
 | |
|     }
 | |
|     Py_ssize_t size = _PyModule_GetFilenameUTF8(module, buffer, maxlen);
 | |
|     Py_DECREF(module);
 | |
|     return size;
 | |
| }
 | |
| 
 | |
| 
 | |
| static PyObject *
 | |
| import_get_module(PyThreadState *tstate, const char *modname)
 | |
| {
 | |
|     PyObject *module = NULL;
 | |
|     if (strcmp(modname, "__main__") == 0) {
 | |
|         module = _Py_GetMainModule(tstate);
 | |
|         if (_Py_CheckMainModule(module) < 0) {
 | |
|             assert(_PyErr_Occurred(tstate));
 | |
|             return NULL;
 | |
|         }
 | |
|     }
 | |
|     else {
 | |
|         module = PyImport_ImportModule(modname);
 | |
|         if (module == NULL) {
 | |
|             return NULL;
 | |
|         }
 | |
|     }
 | |
|     return module;
 | |
| }
 | |
| 
 | |
| 
 | |
| static PyObject *
 | |
| runpy_run_path(const char *filename, const char *modname)
 | |
| {
 | |
|     PyObject *run_path = PyImport_ImportModuleAttrString("runpy", "run_path");
 | |
|     if (run_path == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
|     PyObject *args = Py_BuildValue("(sOs)", filename, Py_None, modname);
 | |
|     if (args == NULL) {
 | |
|         Py_DECREF(run_path);
 | |
|         return NULL;
 | |
|     }
 | |
|     PyObject *ns = PyObject_Call(run_path, args, NULL);
 | |
|     Py_DECREF(run_path);
 | |
|     Py_DECREF(args);
 | |
|     return ns;
 | |
| }
 | |
| 
 | |
| 
 | |
| static PyObject *
 | |
| pyerr_get_message(PyObject *exc)
 | |
| {
 | |
|     assert(!PyErr_Occurred());
 | |
|     PyObject *args = PyException_GetArgs(exc);
 | |
|     if (args == NULL || args == Py_None || PyObject_Size(args) < 1) {
 | |
|         return NULL;
 | |
|     }
 | |
|     if (PyUnicode_Check(args)) {
 | |
|         return args;
 | |
|     }
 | |
|     PyObject *msg = PySequence_GetItem(args, 0);
 | |
|     Py_DECREF(args);
 | |
|     if (msg == NULL) {
 | |
|         PyErr_Clear();
 | |
|         return NULL;
 | |
|     }
 | |
|     if (!PyUnicode_Check(msg)) {
 | |
|         Py_DECREF(msg);
 | |
|         return NULL;
 | |
|     }
 | |
|     return msg;
 | |
| }
 | |
| 
 | |
| #define MAX_MODNAME (255)
 | |
| #define MAX_ATTRNAME (255)
 | |
| 
 | |
| struct attributeerror_info {
 | |
|     char modname[MAX_MODNAME+1];
 | |
|     char attrname[MAX_ATTRNAME+1];
 | |
| };
 | |
| 
 | |
| static int
 | |
| _parse_attributeerror(PyObject *exc, struct attributeerror_info *info)
 | |
| {
 | |
|     assert(exc != NULL);
 | |
|     assert(PyErr_GivenExceptionMatches(exc, PyExc_AttributeError));
 | |
|     int res = -1;
 | |
| 
 | |
|     PyObject *msgobj = pyerr_get_message(exc);
 | |
|     if (msgobj == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     const char *err = PyUnicode_AsUTF8(msgobj);
 | |
| 
 | |
|     if (strncmp(err, "module '", 8) != 0) {
 | |
|         goto finally;
 | |
|     }
 | |
|     err += 8;
 | |
| 
 | |
|     const char *matched = strchr(err, '\'');
 | |
|     if (matched == NULL) {
 | |
|         goto finally;
 | |
|     }
 | |
|     Py_ssize_t len = matched - err;
 | |
|     if (len > MAX_MODNAME) {
 | |
|         goto finally;
 | |
|     }
 | |
|     (void)strncpy(info->modname, err, len);
 | |
|     info->modname[len] = '\0';
 | |
|     err = matched;
 | |
| 
 | |
|     if (strncmp(err, "' has no attribute '", 20) != 0) {
 | |
|         goto finally;
 | |
|     }
 | |
|     err += 20;
 | |
| 
 | |
|     matched = strchr(err, '\'');
 | |
|     if (matched == NULL) {
 | |
|         goto finally;
 | |
|     }
 | |
|     len = matched - err;
 | |
|     if (len > MAX_ATTRNAME) {
 | |
|         goto finally;
 | |
|     }
 | |
|     (void)strncpy(info->attrname, err, len);
 | |
|     info->attrname[len] = '\0';
 | |
|     err = matched + 1;
 | |
| 
 | |
|     if (strlen(err) > 0) {
 | |
|         goto finally;
 | |
|     }
 | |
|     res = 0;
 | |
| 
 | |
| finally:
 | |
|     Py_DECREF(msgobj);
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| #undef MAX_MODNAME
 | |
| #undef MAX_ATTRNAME
 | |
| 
 | |
| 
 | |
| /**************/
 | |
| /* exceptions */
 | |
| /**************/
 | |
| 
 | |
| typedef struct xi_exceptions exceptions_t;
 | |
| static int init_static_exctypes(exceptions_t *, PyInterpreterState *);
 | |
| static void fini_static_exctypes(exceptions_t *, PyInterpreterState *);
 | |
| static int init_heap_exctypes(exceptions_t *);
 | |
| static void fini_heap_exctypes(exceptions_t *);
 | |
| #include "crossinterp_exceptions.h"
 | |
| 
 | |
| 
 | |
| /***************************/
 | |
| /* cross-interpreter calls */
 | |
| /***************************/
 | |
| 
 | |
| int
 | |
| _Py_CallInInterpreter(PyInterpreterState *interp,
 | |
|                       _Py_simple_func func, void *arg)
 | |
| {
 | |
|     if (interp == PyInterpreterState_Get()) {
 | |
|         return func(arg);
 | |
|     }
 | |
|     // XXX Emit a warning if this fails?
 | |
|     _PyEval_AddPendingCall(interp, (_Py_pending_call_func)func, arg, 0);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| int
 | |
| _Py_CallInInterpreterAndRawFree(PyInterpreterState *interp,
 | |
|                                 _Py_simple_func func, void *arg)
 | |
| {
 | |
|     if (interp == PyInterpreterState_Get()) {
 | |
|         int res = func(arg);
 | |
|         PyMem_RawFree(arg);
 | |
|         return res;
 | |
|     }
 | |
|     // XXX Emit a warning if this fails?
 | |
|     _PyEval_AddPendingCall(interp, func, arg, _Py_PENDING_RAWFREE);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**************************/
 | |
| /* cross-interpreter data */
 | |
| /**************************/
 | |
| 
 | |
| /* registry of {type -> xidatafunc} */
 | |
| 
 | |
| /* For now we use a global registry of shareable classes.  An
 | |
|    alternative would be to add a tp_* slot for a class's
 | |
|    xidatafunc. It would be simpler and more efficient. */
 | |
| 
 | |
| static void xid_lookup_init(_PyXIData_lookup_t *);
 | |
| static void xid_lookup_fini(_PyXIData_lookup_t *);
 | |
| struct _dlcontext;
 | |
| static xidatafunc lookup_getdata(struct _dlcontext *, PyObject *);
 | |
| #include "crossinterp_data_lookup.h"
 | |
| 
 | |
| 
 | |
| /* lifecycle */
 | |
| 
 | |
| _PyXIData_t *
 | |
| _PyXIData_New(void)
 | |
| {
 | |
|     _PyXIData_t *xid = PyMem_RawCalloc(1, sizeof(_PyXIData_t));
 | |
|     if (xid == NULL) {
 | |
|         PyErr_NoMemory();
 | |
|     }
 | |
|     return xid;
 | |
| }
 | |
| 
 | |
| void
 | |
| _PyXIData_Free(_PyXIData_t *xid)
 | |
| {
 | |
|     PyInterpreterState *interp = PyInterpreterState_Get();
 | |
|     _PyXIData_Clear(interp, xid);
 | |
|     PyMem_RawFree(xid);
 | |
| }
 | |
| 
 | |
| 
 | |
| /* defining cross-interpreter data */
 | |
| 
 | |
| static inline void
 | |
| _xidata_init(_PyXIData_t *xidata)
 | |
| {
 | |
|     // If the value is being reused
 | |
|     // then _xidata_clear() should have been called already.
 | |
|     assert(xidata->data == NULL);
 | |
|     assert(xidata->obj == NULL);
 | |
|     *xidata = (_PyXIData_t){0};
 | |
|     _PyXIData_INTERPID(xidata) = -1;
 | |
| }
 | |
| 
 | |
| static inline void
 | |
| _xidata_clear(_PyXIData_t *xidata)
 | |
| {
 | |
|     // _PyXIData_t only has two members that need to be
 | |
|     // cleaned up, if set: "xidata" must be freed and "obj" must be decref'ed.
 | |
|     // In both cases the original (owning) interpreter must be used,
 | |
|     // which is the caller's responsibility to ensure.
 | |
|     if (xidata->data != NULL) {
 | |
|         if (xidata->free != NULL) {
 | |
|             xidata->free(xidata->data);
 | |
|         }
 | |
|         xidata->data = NULL;
 | |
|     }
 | |
|     Py_CLEAR(xidata->obj);
 | |
| }
 | |
| 
 | |
| void
 | |
| _PyXIData_Init(_PyXIData_t *xidata,
 | |
|                PyInterpreterState *interp,
 | |
|                void *shared, PyObject *obj,
 | |
|                xid_newobjfunc new_object)
 | |
| {
 | |
|     assert(xidata != NULL);
 | |
|     assert(new_object != NULL);
 | |
|     _xidata_init(xidata);
 | |
|     xidata->data = shared;
 | |
|     if (obj != NULL) {
 | |
|         assert(interp != NULL);
 | |
|         // released in _PyXIData_Clear()
 | |
|         xidata->obj = Py_NewRef(obj);
 | |
|     }
 | |
|     // Ideally every object would know its owning interpreter.
 | |
|     // Until then, we have to rely on the caller to identify it
 | |
|     // (but we don't need it in all cases).
 | |
|     _PyXIData_INTERPID(xidata) = (interp != NULL)
 | |
|         ? PyInterpreterState_GetID(interp)
 | |
|         : -1;
 | |
|     xidata->new_object = new_object;
 | |
| }
 | |
| 
 | |
| int
 | |
| _PyXIData_InitWithSize(_PyXIData_t *xidata,
 | |
|                        PyInterpreterState *interp,
 | |
|                        const size_t size, PyObject *obj,
 | |
|                        xid_newobjfunc new_object)
 | |
| {
 | |
|     assert(size > 0);
 | |
|     // For now we always free the shared data in the same interpreter
 | |
|     // where it was allocated, so the interpreter is required.
 | |
|     assert(interp != NULL);
 | |
|     _PyXIData_Init(xidata, interp, NULL, obj, new_object);
 | |
|     xidata->data = PyMem_RawCalloc(1, size);
 | |
|     if (xidata->data == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     xidata->free = PyMem_RawFree;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| void
 | |
| _PyXIData_Clear(PyInterpreterState *interp, _PyXIData_t *xidata)
 | |
| {
 | |
|     assert(xidata != NULL);
 | |
|     // This must be called in the owning interpreter.
 | |
|     assert(interp == NULL
 | |
|            || _PyXIData_INTERPID(xidata) == -1
 | |
|            || _PyXIData_INTERPID(xidata) == PyInterpreterState_GetID(interp));
 | |
|     _xidata_clear(xidata);
 | |
| }
 | |
| 
 | |
| 
 | |
| /* getting cross-interpreter data */
 | |
| 
 | |
| static inline void
 | |
| _set_xid_lookup_failure(PyThreadState *tstate, PyObject *obj, const char *msg,
 | |
|                         PyObject *cause)
 | |
| {
 | |
|     if (msg != NULL) {
 | |
|         assert(obj == NULL);
 | |
|         set_notshareableerror(tstate, cause, 0, msg);
 | |
|     }
 | |
|     else if (obj == NULL) {
 | |
|         msg = "object does not support cross-interpreter data";
 | |
|         set_notshareableerror(tstate, cause, 0, msg);
 | |
|     }
 | |
|     else {
 | |
|         msg = "%S does not support cross-interpreter data";
 | |
|         format_notshareableerror(tstate, cause, 0, msg, obj);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| _PyObject_CheckXIData(PyThreadState *tstate, PyObject *obj)
 | |
| {
 | |
|     dlcontext_t ctx;
 | |
|     if (get_lookup_context(tstate, &ctx) < 0) {
 | |
|         return -1;
 | |
|     }
 | |
|     xidatafunc getdata = lookup_getdata(&ctx, obj);
 | |
|     if (getdata == NULL) {
 | |
|         if (!_PyErr_Occurred(tstate)) {
 | |
|             _set_xid_lookup_failure(tstate, obj, NULL, NULL);
 | |
|         }
 | |
|         return -1;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| _check_xidata(PyThreadState *tstate, _PyXIData_t *xidata)
 | |
| {
 | |
|     // xidata->data can be anything, including NULL, so we don't check it.
 | |
| 
 | |
|     // xidata->obj may be NULL, so we don't check it.
 | |
| 
 | |
|     if (_PyXIData_INTERPID(xidata) < 0) {
 | |
|         PyErr_SetString(PyExc_SystemError, "missing interp");
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     if (xidata->new_object == NULL) {
 | |
|         PyErr_SetString(PyExc_SystemError, "missing new_object func");
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     // xidata->free may be NULL, so we don't check it.
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| int
 | |
| _PyObject_GetXIData(PyThreadState *tstate,
 | |
|                     PyObject *obj, _PyXIData_t *xidata)
 | |
| {
 | |
|     PyInterpreterState *interp = tstate->interp;
 | |
| 
 | |
|     assert(xidata->data == NULL);
 | |
|     assert(xidata->obj == NULL);
 | |
|     if (xidata->data != NULL || xidata->obj != NULL) {
 | |
|         _PyErr_SetString(tstate, PyExc_ValueError, "xidata not cleared");
 | |
|     }
 | |
| 
 | |
|     // Call the "getdata" func for the object.
 | |
|     dlcontext_t ctx;
 | |
|     if (get_lookup_context(tstate, &ctx) < 0) {
 | |
|         return -1;
 | |
|     }
 | |
|     Py_INCREF(obj);
 | |
|     xidatafunc getdata = lookup_getdata(&ctx, obj);
 | |
|     if (getdata == NULL) {
 | |
|         if (PyErr_Occurred()) {
 | |
|             Py_DECREF(obj);
 | |
|             return -1;
 | |
|         }
 | |
|         // Fall back to obj
 | |
|         Py_DECREF(obj);
 | |
|         if (!_PyErr_Occurred(tstate)) {
 | |
|             _set_xid_lookup_failure(tstate, obj, NULL, NULL);
 | |
|         }
 | |
|         return -1;
 | |
|     }
 | |
|     int res = getdata(tstate, obj, xidata);
 | |
|     Py_DECREF(obj);
 | |
|     if (res != 0) {
 | |
|         PyObject *cause = _PyErr_GetRaisedException(tstate);
 | |
|         assert(cause != NULL);
 | |
|         _set_xid_lookup_failure(tstate, obj, NULL, cause);
 | |
|         Py_XDECREF(cause);
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     // Fill in the blanks and validate the result.
 | |
|     _PyXIData_INTERPID(xidata) = PyInterpreterState_GetID(interp);
 | |
|     if (_check_xidata(tstate, xidata) != 0) {
 | |
|         (void)_PyXIData_Release(xidata);
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| /* pickle C-API */
 | |
| 
 | |
| struct _pickle_context {
 | |
|     PyThreadState *tstate;
 | |
| };
 | |
| 
 | |
| static PyObject *
 | |
| _PyPickle_Dumps(struct _pickle_context *ctx, PyObject *obj)
 | |
| {
 | |
|     PyObject *dumps = PyImport_ImportModuleAttrString("pickle", "dumps");
 | |
|     if (dumps == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
|     PyObject *bytes = PyObject_CallOneArg(dumps, obj);
 | |
|     Py_DECREF(dumps);
 | |
|     return bytes;
 | |
| }
 | |
| 
 | |
| 
 | |
| struct sync_module_result {
 | |
|     PyObject *module;
 | |
|     PyObject *loaded;
 | |
|     PyObject *failed;
 | |
| };
 | |
| 
 | |
| struct sync_module {
 | |
|     const char *filename;
 | |
|     char _filename[MAXPATHLEN+1];
 | |
|     struct sync_module_result cached;
 | |
| };
 | |
| 
 | |
| static void
 | |
| sync_module_clear(struct sync_module *data)
 | |
| {
 | |
|     data->filename = NULL;
 | |
|     Py_CLEAR(data->cached.module);
 | |
|     Py_CLEAR(data->cached.loaded);
 | |
|     Py_CLEAR(data->cached.failed);
 | |
| }
 | |
| 
 | |
| 
 | |
| struct _unpickle_context {
 | |
|     PyThreadState *tstate;
 | |
|     // We only special-case the __main__ module,
 | |
|     // since other modules behave consistently.
 | |
|     struct sync_module main;
 | |
| };
 | |
| 
 | |
| static void
 | |
| _unpickle_context_clear(struct _unpickle_context *ctx)
 | |
| {
 | |
|     sync_module_clear(&ctx->main);
 | |
| }
 | |
| 
 | |
| static struct sync_module_result
 | |
| _unpickle_context_get_module(struct _unpickle_context *ctx,
 | |
|                              const char *modname)
 | |
| {
 | |
|     if (strcmp(modname, "__main__") == 0) {
 | |
|         return ctx->main.cached;
 | |
|     }
 | |
|     else {
 | |
|         return (struct sync_module_result){
 | |
|             .failed = PyExc_NotImplementedError,
 | |
|         };
 | |
|     }
 | |
| }
 | |
| 
 | |
| static struct sync_module_result
 | |
| _unpickle_context_set_module(struct _unpickle_context *ctx,
 | |
|                              const char *modname)
 | |
| {
 | |
|     struct sync_module_result res = {0};
 | |
|     struct sync_module_result *cached = NULL;
 | |
|     const char *filename = NULL;
 | |
|     const char *run_modname = modname;
 | |
|     if (strcmp(modname, "__main__") == 0) {
 | |
|         cached = &ctx->main.cached;
 | |
|         filename = ctx->main.filename;
 | |
|         // We don't want to trigger "if __name__ == '__main__':".
 | |
|         run_modname = "<fake __main__>";
 | |
|     }
 | |
|     else {
 | |
|         res.failed = PyExc_NotImplementedError;
 | |
|         goto finally;
 | |
|     }
 | |
| 
 | |
|     res.module = import_get_module(ctx->tstate, modname);
 | |
|     if (res.module == NULL) {
 | |
|         res.failed = _PyErr_GetRaisedException(ctx->tstate);
 | |
|         assert(res.failed != NULL);
 | |
|         goto finally;
 | |
|     }
 | |
| 
 | |
|     if (filename == NULL) {
 | |
|         Py_CLEAR(res.module);
 | |
|         res.failed = PyExc_NotImplementedError;
 | |
|         goto finally;
 | |
|     }
 | |
|     res.loaded = runpy_run_path(filename, run_modname);
 | |
|     if (res.loaded == NULL) {
 | |
|         Py_CLEAR(res.module);
 | |
|         res.failed = _PyErr_GetRaisedException(ctx->tstate);
 | |
|         assert(res.failed != NULL);
 | |
|         goto finally;
 | |
|     }
 | |
| 
 | |
| finally:
 | |
|     if (cached != NULL) {
 | |
|         assert(cached->module == NULL);
 | |
|         assert(cached->loaded == NULL);
 | |
|         assert(cached->failed == NULL);
 | |
|         *cached = res;
 | |
|     }
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int
 | |
| _handle_unpickle_missing_attr(struct _unpickle_context *ctx, PyObject *exc)
 | |
| {
 | |
|     // The caller must check if an exception is set or not when -1 is returned.
 | |
|     assert(!_PyErr_Occurred(ctx->tstate));
 | |
|     assert(PyErr_GivenExceptionMatches(exc, PyExc_AttributeError));
 | |
|     struct attributeerror_info info;
 | |
|     if (_parse_attributeerror(exc, &info) < 0) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     // Get the module.
 | |
|     struct sync_module_result mod = _unpickle_context_get_module(ctx, info.modname);
 | |
|     if (mod.failed != NULL) {
 | |
|         // It must have failed previously.
 | |
|         return -1;
 | |
|     }
 | |
|     if (mod.module == NULL) {
 | |
|         mod = _unpickle_context_set_module(ctx, info.modname);
 | |
|         if (mod.failed != NULL) {
 | |
|             return -1;
 | |
|         }
 | |
|         assert(mod.module != NULL);
 | |
|     }
 | |
| 
 | |
|     // Bail out if it is unexpectedly set already.
 | |
|     if (PyObject_HasAttrString(mod.module, info.attrname)) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     // Try setting the attribute.
 | |
|     PyObject *value = NULL;
 | |
|     if (PyDict_GetItemStringRef(mod.loaded, info.attrname, &value) <= 0) {
 | |
|         return -1;
 | |
|     }
 | |
|     assert(value != NULL);
 | |
|     int res = PyObject_SetAttrString(mod.module, info.attrname, value);
 | |
|     Py_DECREF(value);
 | |
|     if (res < 0) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| _PyPickle_Loads(struct _unpickle_context *ctx, PyObject *pickled)
 | |
| {
 | |
|     PyObject *loads = PyImport_ImportModuleAttrString("pickle", "loads");
 | |
|     if (loads == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
|     PyObject *obj = PyObject_CallOneArg(loads, pickled);
 | |
|     if (ctx != NULL) {
 | |
|         while (obj == NULL) {
 | |
|             assert(_PyErr_Occurred(ctx->tstate));
 | |
|             if (!PyErr_ExceptionMatches(PyExc_AttributeError)) {
 | |
|                 // We leave other failures unhandled.
 | |
|                 break;
 | |
|             }
 | |
|             // Try setting the attr if not set.
 | |
|             PyObject *exc = _PyErr_GetRaisedException(ctx->tstate);
 | |
|             if (_handle_unpickle_missing_attr(ctx, exc) < 0) {
 | |
|                 // Any resulting exceptions are ignored
 | |
|                 // in favor of the original.
 | |
|                 _PyErr_SetRaisedException(ctx->tstate, exc);
 | |
|                 break;
 | |
|             }
 | |
|             Py_CLEAR(exc);
 | |
|             // Retry with the attribute set.
 | |
|             obj = PyObject_CallOneArg(loads, pickled);
 | |
|         }
 | |
|     }
 | |
|     Py_DECREF(loads);
 | |
|     return obj;
 | |
| }
 | |
| 
 | |
| 
 | |
| /* pickle wrapper */
 | |
| 
 | |
| struct _pickle_xid_context {
 | |
|     // __main__.__file__
 | |
|     struct {
 | |
|         const char *utf8;
 | |
|         size_t len;
 | |
|         char _utf8[MAXPATHLEN+1];
 | |
|     } mainfile;
 | |
| };
 | |
| 
 | |
| static int
 | |
| _set_pickle_xid_context(PyThreadState *tstate, struct _pickle_xid_context *ctx)
 | |
| {
 | |
|     // Set mainfile if possible.
 | |
|     Py_ssize_t len = _Py_GetMainfile(ctx->mainfile._utf8, MAXPATHLEN);
 | |
|     if (len < 0) {
 | |
|         // For now we ignore any exceptions.
 | |
|         PyErr_Clear();
 | |
|     }
 | |
|     else if (len > 0) {
 | |
|         ctx->mainfile.utf8 = ctx->mainfile._utf8;
 | |
|         ctx->mainfile.len = (size_t)len;
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| struct _shared_pickle_data {
 | |
|     _PyBytes_data_t pickled;  // Must be first if we use _PyBytes_FromXIData().
 | |
|     struct _pickle_xid_context ctx;
 | |
| };
 | |
| 
 | |
| PyObject *
 | |
| _PyPickle_LoadFromXIData(_PyXIData_t *xidata)
 | |
| {
 | |
|     PyThreadState *tstate = _PyThreadState_GET();
 | |
|     struct _shared_pickle_data *shared =
 | |
|                             (struct _shared_pickle_data *)xidata->data;
 | |
|     // We avoid copying the pickled data by wrapping it in a memoryview.
 | |
|     // The alternative is to get a bytes object using _PyBytes_FromXIData().
 | |
|     PyObject *pickled = PyMemoryView_FromMemory(
 | |
|             (char *)shared->pickled.bytes, shared->pickled.len, PyBUF_READ);
 | |
|     if (pickled == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     // Unpickle the object.
 | |
|     struct _unpickle_context ctx = {
 | |
|         .tstate = tstate,
 | |
|         .main = {
 | |
|             .filename = shared->ctx.mainfile.utf8,
 | |
|         },
 | |
|     };
 | |
|     PyObject *obj = _PyPickle_Loads(&ctx, pickled);
 | |
|     Py_DECREF(pickled);
 | |
|     _unpickle_context_clear(&ctx);
 | |
|     if (obj == NULL) {
 | |
|         PyObject *cause = _PyErr_GetRaisedException(tstate);
 | |
|         assert(cause != NULL);
 | |
|         _set_xid_lookup_failure(
 | |
|                     tstate, NULL, "object could not be unpickled", cause);
 | |
|         Py_DECREF(cause);
 | |
|     }
 | |
|     return obj;
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| _PyPickle_GetXIData(PyThreadState *tstate, PyObject *obj, _PyXIData_t *xidata)
 | |
| {
 | |
|     // Pickle the object.
 | |
|     struct _pickle_context ctx = {
 | |
|         .tstate = tstate,
 | |
|     };
 | |
|     PyObject *bytes = _PyPickle_Dumps(&ctx, obj);
 | |
|     if (bytes == NULL) {
 | |
|         PyObject *cause = _PyErr_GetRaisedException(tstate);
 | |
|         assert(cause != NULL);
 | |
|         _set_xid_lookup_failure(
 | |
|                     tstate, NULL, "object could not be pickled", cause);
 | |
|         Py_DECREF(cause);
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     // If we had an "unwrapper" mechnanism, we could call
 | |
|     // _PyObject_GetXIData() on the bytes object directly and add
 | |
|     // a simple unwrapper to call pickle.loads() on the bytes.
 | |
|     size_t size = sizeof(struct _shared_pickle_data);
 | |
|     struct _shared_pickle_data *shared =
 | |
|             (struct _shared_pickle_data *)_PyBytes_GetXIDataWrapped(
 | |
|                     tstate, bytes, size, _PyPickle_LoadFromXIData, xidata);
 | |
|     Py_DECREF(bytes);
 | |
|     if (shared == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     // If it mattered, we could skip getting __main__.__file__
 | |
|     // when "__main__" doesn't show up in the pickle bytes.
 | |
|     if (_set_pickle_xid_context(tstate, &shared->ctx) < 0) {
 | |
|         _xidata_clear(xidata);
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| /* marshal wrapper */
 | |
| 
 | |
| PyObject *
 | |
| _PyMarshal_ReadObjectFromXIData(_PyXIData_t *xidata)
 | |
| {
 | |
|     PyThreadState *tstate = _PyThreadState_GET();
 | |
|     _PyBytes_data_t *shared = (_PyBytes_data_t *)xidata->data;
 | |
|     PyObject *obj = PyMarshal_ReadObjectFromString(shared->bytes, shared->len);
 | |
|     if (obj == NULL) {
 | |
|         PyObject *cause = _PyErr_GetRaisedException(tstate);
 | |
|         assert(cause != NULL);
 | |
|         _set_xid_lookup_failure(
 | |
|                     tstate, NULL, "object could not be unmarshalled", cause);
 | |
|         Py_DECREF(cause);
 | |
|         return NULL;
 | |
|     }
 | |
|     return obj;
 | |
| }
 | |
| 
 | |
| int
 | |
| _PyMarshal_GetXIData(PyThreadState *tstate, PyObject *obj, _PyXIData_t *xidata)
 | |
| {
 | |
|     PyObject *bytes = PyMarshal_WriteObjectToString(obj, Py_MARSHAL_VERSION);
 | |
|     if (bytes == NULL) {
 | |
|         PyObject *cause = _PyErr_GetRaisedException(tstate);
 | |
|         assert(cause != NULL);
 | |
|         _set_xid_lookup_failure(
 | |
|                     tstate, NULL, "object could not be marshalled", cause);
 | |
|         Py_DECREF(cause);
 | |
|         return -1;
 | |
|     }
 | |
|     size_t size = sizeof(_PyBytes_data_t);
 | |
|     _PyBytes_data_t *shared = _PyBytes_GetXIDataWrapped(
 | |
|             tstate, bytes, size, _PyMarshal_ReadObjectFromXIData, xidata);
 | |
|     Py_DECREF(bytes);
 | |
|     if (shared == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| /* script wrapper */
 | |
| 
 | |
| static int
 | |
| verify_script(PyThreadState *tstate, PyCodeObject *co, int checked, int pure)
 | |
| {
 | |
|     // Make sure it isn't a closure and (optionally) doesn't use globals.
 | |
|     PyObject *builtins = NULL;
 | |
|     if (pure) {
 | |
|         builtins = _PyEval_GetBuiltins(tstate);
 | |
|         assert(builtins != NULL);
 | |
|     }
 | |
|     if (checked) {
 | |
|         assert(_PyCode_VerifyStateless(tstate, co, NULL, NULL, builtins) == 0);
 | |
|     }
 | |
|     else if (_PyCode_VerifyStateless(tstate, co, NULL, NULL, builtins) < 0) {
 | |
|         return -1;
 | |
|     }
 | |
|     // Make sure it doesn't have args.
 | |
|     if (co->co_argcount > 0
 | |
|         || co->co_posonlyargcount > 0
 | |
|         || co->co_kwonlyargcount > 0
 | |
|         || co->co_flags & (CO_VARARGS | CO_VARKEYWORDS))
 | |
|     {
 | |
|         _PyErr_SetString(tstate, PyExc_ValueError,
 | |
|                          "code with args not supported");
 | |
|         return -1;
 | |
|     }
 | |
|     // Make sure it doesn't return anything.
 | |
|     if (!_PyCode_ReturnsOnlyNone(co)) {
 | |
|         _PyErr_SetString(tstate, PyExc_ValueError,
 | |
|                          "code that returns a value is not a script");
 | |
|         return -1;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| get_script_xidata(PyThreadState *tstate, PyObject *obj, int pure,
 | |
|                   _PyXIData_t *xidata)
 | |
| {
 | |
|     // Get the corresponding code object.
 | |
|     PyObject *code = NULL;
 | |
|     int checked = 0;
 | |
|     if (PyCode_Check(obj)) {
 | |
|         code = obj;
 | |
|         Py_INCREF(code);
 | |
|     }
 | |
|     else if (PyFunction_Check(obj)) {
 | |
|         code = PyFunction_GET_CODE(obj);
 | |
|         assert(code != NULL);
 | |
|         Py_INCREF(code);
 | |
|         if (pure) {
 | |
|             if (_PyFunction_VerifyStateless(tstate, obj) < 0) {
 | |
|                 goto error;
 | |
|             }
 | |
|             checked = 1;
 | |
|         }
 | |
|     }
 | |
|     else {
 | |
|         const char *filename = "<script>";
 | |
|         int optimize = 0;
 | |
|         PyCompilerFlags cf = _PyCompilerFlags_INIT;
 | |
|         cf.cf_flags = PyCF_SOURCE_IS_UTF8;
 | |
|         PyObject *ref = NULL;
 | |
|         const char *script = _Py_SourceAsString(obj, "???", "???", &cf, &ref);
 | |
|         if (script == NULL) {
 | |
|             if (!_PyObject_SupportedAsScript(obj)) {
 | |
|                 // We discard the raised exception.
 | |
|                 _PyErr_Format(tstate, PyExc_TypeError,
 | |
|                               "unsupported script %R", obj);
 | |
|             }
 | |
|             goto error;
 | |
|         }
 | |
|         code = Py_CompileStringExFlags(
 | |
|                     script, filename, Py_file_input, &cf, optimize);
 | |
|         Py_XDECREF(ref);
 | |
|         if (code == NULL) {
 | |
|             goto error;
 | |
|         }
 | |
|         // Compiled text can't have args or any return statements,
 | |
|         // nor be a closure.  It can use globals though.
 | |
|         if (!pure) {
 | |
|             // We don't need to check for globals either.
 | |
|             checked = 1;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Make sure it's actually a script.
 | |
|     if (verify_script(tstate, (PyCodeObject *)code, checked, pure) < 0) {
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     // Convert the code object.
 | |
|     int res = _PyCode_GetXIData(tstate, code, xidata);
 | |
|     Py_DECREF(code);
 | |
|     if (res < 0) {
 | |
|         return -1;
 | |
|     }
 | |
|     return 0;
 | |
| 
 | |
| error:
 | |
|     Py_XDECREF(code);
 | |
|     PyObject *cause = _PyErr_GetRaisedException(tstate);
 | |
|     assert(cause != NULL);
 | |
|     _set_xid_lookup_failure(
 | |
|                 tstate, NULL, "object not a valid script", cause);
 | |
|     Py_DECREF(cause);
 | |
|     return -1;
 | |
| }
 | |
| 
 | |
| int
 | |
| _PyCode_GetScriptXIData(PyThreadState *tstate,
 | |
|                         PyObject *obj, _PyXIData_t *xidata)
 | |
| {
 | |
|     return get_script_xidata(tstate, obj, 0, xidata);
 | |
| }
 | |
| 
 | |
| int
 | |
| _PyCode_GetPureScriptXIData(PyThreadState *tstate,
 | |
|                             PyObject *obj, _PyXIData_t *xidata)
 | |
| {
 | |
|     return get_script_xidata(tstate, obj, 1, xidata);
 | |
| }
 | |
| 
 | |
| 
 | |
| /* using cross-interpreter data */
 | |
| 
 | |
| PyObject *
 | |
| _PyXIData_NewObject(_PyXIData_t *xidata)
 | |
| {
 | |
|     return xidata->new_object(xidata);
 | |
| }
 | |
| 
 | |
| static int
 | |
| _call_clear_xidata(void *data)
 | |
| {
 | |
|     _xidata_clear((_PyXIData_t *)data);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| _xidata_release(_PyXIData_t *xidata, int rawfree)
 | |
| {
 | |
|     if ((xidata->data == NULL || xidata->free == NULL) && xidata->obj == NULL) {
 | |
|         // Nothing to release!
 | |
|         if (rawfree) {
 | |
|             PyMem_RawFree(xidata);
 | |
|         }
 | |
|         else {
 | |
|             xidata->data = NULL;
 | |
|         }
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     // Switch to the original interpreter.
 | |
|     PyInterpreterState *interp = _PyInterpreterState_LookUpID(
 | |
|                                         _PyXIData_INTERPID(xidata));
 | |
|     if (interp == NULL) {
 | |
|         // The interpreter was already destroyed.
 | |
|         // This function shouldn't have been called.
 | |
|         // XXX Someone leaked some memory...
 | |
|         assert(PyErr_Occurred());
 | |
|         if (rawfree) {
 | |
|             PyMem_RawFree(xidata);
 | |
|         }
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     // "Release" the data and/or the object.
 | |
|     if (rawfree) {
 | |
|         return _Py_CallInInterpreterAndRawFree(interp, _call_clear_xidata, xidata);
 | |
|     }
 | |
|     else {
 | |
|         return _Py_CallInInterpreter(interp, _call_clear_xidata, xidata);
 | |
|     }
 | |
| }
 | |
| 
 | |
| int
 | |
| _PyXIData_Release(_PyXIData_t *xidata)
 | |
| {
 | |
|     return _xidata_release(xidata, 0);
 | |
| }
 | |
| 
 | |
| int
 | |
| _PyXIData_ReleaseAndRawFree(_PyXIData_t *xidata)
 | |
| {
 | |
|     return _xidata_release(xidata, 1);
 | |
| }
 | |
| 
 | |
| 
 | |
| /*************************/
 | |
| /* convenience utilities */
 | |
| /*************************/
 | |
| 
 | |
| static const char *
 | |
| _copy_string_obj_raw(PyObject *strobj, Py_ssize_t *p_size)
 | |
| {
 | |
|     Py_ssize_t size = -1;
 | |
|     const char *str = PyUnicode_AsUTF8AndSize(strobj, &size);
 | |
|     if (str == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (size != (Py_ssize_t)strlen(str)) {
 | |
|         PyErr_SetString(PyExc_ValueError, "found embedded NULL character");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     char *copied = PyMem_RawMalloc(size+1);
 | |
|     if (copied == NULL) {
 | |
|         PyErr_NoMemory();
 | |
|         return NULL;
 | |
|     }
 | |
|     strcpy(copied, str);
 | |
|     if (p_size != NULL) {
 | |
|         *p_size = size;
 | |
|     }
 | |
|     return copied;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int
 | |
| _convert_exc_to_TracebackException(PyObject *exc, PyObject **p_tbexc)
 | |
| {
 | |
|     PyObject *args = NULL;
 | |
|     PyObject *kwargs = NULL;
 | |
|     PyObject *create = NULL;
 | |
| 
 | |
|     // This is inspired by _PyErr_Display().
 | |
|     PyObject *tbexc_type = PyImport_ImportModuleAttrString(
 | |
|         "traceback",
 | |
|         "TracebackException");
 | |
|     if (tbexc_type == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     create = PyObject_GetAttrString(tbexc_type, "from_exception");
 | |
|     Py_DECREF(tbexc_type);
 | |
|     if (create == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     args = PyTuple_Pack(1, exc);
 | |
|     if (args == NULL) {
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     kwargs = PyDict_New();
 | |
|     if (kwargs == NULL) {
 | |
|         goto error;
 | |
|     }
 | |
|     if (PyDict_SetItemString(kwargs, "save_exc_type", Py_False) < 0) {
 | |
|         goto error;
 | |
|     }
 | |
|     if (PyDict_SetItemString(kwargs, "lookup_lines", Py_False) < 0) {
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     PyObject *tbexc = PyObject_Call(create, args, kwargs);
 | |
|     Py_DECREF(args);
 | |
|     Py_DECREF(kwargs);
 | |
|     Py_DECREF(create);
 | |
|     if (tbexc == NULL) {
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     *p_tbexc = tbexc;
 | |
|     return 0;
 | |
| 
 | |
| error:
 | |
|     Py_XDECREF(args);
 | |
|     Py_XDECREF(kwargs);
 | |
|     Py_XDECREF(create);
 | |
|     return -1;
 | |
| }
 | |
| 
 | |
| // We accommodate backports here.
 | |
| #ifndef _Py_EMPTY_STR
 | |
| # define _Py_EMPTY_STR &_Py_STR(empty)
 | |
| #endif
 | |
| 
 | |
| static const char *
 | |
| _format_TracebackException(PyObject *tbexc)
 | |
| {
 | |
|     PyObject *lines = PyObject_CallMethod(tbexc, "format", NULL);
 | |
|     if (lines == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
|     assert(_Py_EMPTY_STR != NULL);
 | |
|     PyObject *formatted_obj = PyUnicode_Join(_Py_EMPTY_STR, lines);
 | |
|     Py_DECREF(lines);
 | |
|     if (formatted_obj == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     Py_ssize_t size = -1;
 | |
|     const char *formatted = _copy_string_obj_raw(formatted_obj, &size);
 | |
|     Py_DECREF(formatted_obj);
 | |
|     // We remove trailing the newline added by TracebackException.format().
 | |
|     assert(formatted[size-1] == '\n');
 | |
|     ((char *)formatted)[size-1] = '\0';
 | |
|     return formatted;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int
 | |
| _release_xid_data(_PyXIData_t *xidata, int rawfree)
 | |
| {
 | |
|     PyObject *exc = PyErr_GetRaisedException();
 | |
|     int res = rawfree
 | |
|         ? _PyXIData_Release(xidata)
 | |
|         : _PyXIData_ReleaseAndRawFree(xidata);
 | |
|     if (res < 0) {
 | |
|         /* The owning interpreter is already destroyed. */
 | |
|         _PyXIData_Clear(NULL, xidata);
 | |
|         // XXX Emit a warning?
 | |
|         PyErr_Clear();
 | |
|     }
 | |
|     PyErr_SetRaisedException(exc);
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| 
 | |
| /***********************/
 | |
| /* exception snapshots */
 | |
| /***********************/
 | |
| 
 | |
| static int
 | |
| _excinfo_init_type_from_exception(struct _excinfo_type *info, PyObject *exc)
 | |
| {
 | |
|     /* Note that this copies directly rather than into an intermediate
 | |
|        struct and does not clear on error.  If we need that then we
 | |
|        should have a separate function to wrap this one
 | |
|        and do all that there. */
 | |
|     PyObject *strobj = NULL;
 | |
| 
 | |
|     PyTypeObject *type = Py_TYPE(exc);
 | |
|     if (type->tp_flags & _Py_TPFLAGS_STATIC_BUILTIN) {
 | |
|         assert(_Py_IsImmortal((PyObject *)type));
 | |
|         info->builtin = type;
 | |
|     }
 | |
|     else {
 | |
|         // Only builtin types are preserved.
 | |
|         info->builtin = NULL;
 | |
|     }
 | |
| 
 | |
|     // __name__
 | |
|     strobj = PyType_GetName(type);
 | |
|     if (strobj == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     info->name = _copy_string_obj_raw(strobj, NULL);
 | |
|     Py_DECREF(strobj);
 | |
|     if (info->name == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     // __qualname__
 | |
|     strobj = PyType_GetQualName(type);
 | |
|     if (strobj == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     info->qualname = _copy_string_obj_raw(strobj, NULL);
 | |
|     Py_DECREF(strobj);
 | |
|     if (info->qualname == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     // __module__
 | |
|     strobj = PyType_GetModuleName(type);
 | |
|     if (strobj == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     info->module = _copy_string_obj_raw(strobj, NULL);
 | |
|     Py_DECREF(strobj);
 | |
|     if (info->module == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| _excinfo_init_type_from_object(struct _excinfo_type *info, PyObject *exctype)
 | |
| {
 | |
|     PyObject *strobj = NULL;
 | |
| 
 | |
|     // __name__
 | |
|     strobj = PyObject_GetAttrString(exctype, "__name__");
 | |
|     if (strobj == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     info->name = _copy_string_obj_raw(strobj, NULL);
 | |
|     Py_DECREF(strobj);
 | |
|     if (info->name == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     // __qualname__
 | |
|     strobj = PyObject_GetAttrString(exctype, "__qualname__");
 | |
|     if (strobj == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     info->qualname = _copy_string_obj_raw(strobj, NULL);
 | |
|     Py_DECREF(strobj);
 | |
|     if (info->qualname == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     // __module__
 | |
|     strobj = PyObject_GetAttrString(exctype, "__module__");
 | |
|     if (strobj == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     info->module = _copy_string_obj_raw(strobj, NULL);
 | |
|     Py_DECREF(strobj);
 | |
|     if (info->module == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static void
 | |
| _excinfo_clear_type(struct _excinfo_type *info)
 | |
| {
 | |
|     if (info->builtin != NULL) {
 | |
|         assert(info->builtin->tp_flags & _Py_TPFLAGS_STATIC_BUILTIN);
 | |
|         assert(_Py_IsImmortal((PyObject *)info->builtin));
 | |
|     }
 | |
|     if (info->name != NULL) {
 | |
|         PyMem_RawFree((void *)info->name);
 | |
|     }
 | |
|     if (info->qualname != NULL) {
 | |
|         PyMem_RawFree((void *)info->qualname);
 | |
|     }
 | |
|     if (info->module != NULL) {
 | |
|         PyMem_RawFree((void *)info->module);
 | |
|     }
 | |
|     *info = (struct _excinfo_type){NULL};
 | |
| }
 | |
| 
 | |
| static void
 | |
| _excinfo_normalize_type(struct _excinfo_type *info,
 | |
|                         const char **p_module, const char **p_qualname)
 | |
| {
 | |
|     if (info->name == NULL) {
 | |
|         assert(info->builtin == NULL);
 | |
|         assert(info->qualname == NULL);
 | |
|         assert(info->module == NULL);
 | |
|         // This is inspired by TracebackException.format_exception_only().
 | |
|         *p_module = NULL;
 | |
|         *p_qualname = NULL;
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     const char *module = info->module;
 | |
|     const char *qualname = info->qualname;
 | |
|     if (qualname == NULL) {
 | |
|         qualname = info->name;
 | |
|     }
 | |
|     assert(module != NULL);
 | |
|     if (strcmp(module, "builtins") == 0) {
 | |
|         module = NULL;
 | |
|     }
 | |
|     else if (strcmp(module, "__main__") == 0) {
 | |
|         module = NULL;
 | |
|     }
 | |
|     *p_qualname = qualname;
 | |
|     *p_module = module;
 | |
| }
 | |
| 
 | |
| static void
 | |
| _PyXI_excinfo_Clear(_PyXI_excinfo *info)
 | |
| {
 | |
|     _excinfo_clear_type(&info->type);
 | |
|     if (info->msg != NULL) {
 | |
|         PyMem_RawFree((void *)info->msg);
 | |
|     }
 | |
|     if (info->errdisplay != NULL) {
 | |
|         PyMem_RawFree((void *)info->errdisplay);
 | |
|     }
 | |
|     *info = (_PyXI_excinfo){{NULL}};
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| _PyXI_excinfo_format(_PyXI_excinfo *info)
 | |
| {
 | |
|     const char *module, *qualname;
 | |
|     _excinfo_normalize_type(&info->type, &module, &qualname);
 | |
|     if (qualname != NULL) {
 | |
|         if (module != NULL) {
 | |
|             if (info->msg != NULL) {
 | |
|                 return PyUnicode_FromFormat("%s.%s: %s",
 | |
|                                             module, qualname, info->msg);
 | |
|             }
 | |
|             else {
 | |
|                 return PyUnicode_FromFormat("%s.%s", module, qualname);
 | |
|             }
 | |
|         }
 | |
|         else {
 | |
|             if (info->msg != NULL) {
 | |
|                 return PyUnicode_FromFormat("%s: %s", qualname, info->msg);
 | |
|             }
 | |
|             else {
 | |
|                 return PyUnicode_FromString(qualname);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     else if (info->msg != NULL) {
 | |
|         return PyUnicode_FromString(info->msg);
 | |
|     }
 | |
|     else {
 | |
|         Py_RETURN_NONE;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static const char *
 | |
| _PyXI_excinfo_InitFromException(_PyXI_excinfo *info, PyObject *exc)
 | |
| {
 | |
|     assert(exc != NULL);
 | |
| 
 | |
|     if (PyErr_GivenExceptionMatches(exc, PyExc_MemoryError)) {
 | |
|         _PyXI_excinfo_Clear(info);
 | |
|         return NULL;
 | |
|     }
 | |
|     const char *failure = NULL;
 | |
| 
 | |
|     if (_excinfo_init_type_from_exception(&info->type, exc) < 0) {
 | |
|         failure = "error while initializing exception type snapshot";
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     // Extract the exception message.
 | |
|     PyObject *msgobj = PyObject_Str(exc);
 | |
|     if (msgobj == NULL) {
 | |
|         failure = "error while formatting exception";
 | |
|         goto error;
 | |
|     }
 | |
|     info->msg = _copy_string_obj_raw(msgobj, NULL);
 | |
|     Py_DECREF(msgobj);
 | |
|     if (info->msg == NULL) {
 | |
|         failure = "error while copying exception message";
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     // Pickle a traceback.TracebackException.
 | |
|     PyObject *tbexc = NULL;
 | |
|     if (_convert_exc_to_TracebackException(exc, &tbexc) < 0) {
 | |
| #ifdef Py_DEBUG
 | |
|         PyErr_FormatUnraisable("Exception ignored while creating TracebackException");
 | |
| #endif
 | |
|         PyErr_Clear();
 | |
|     }
 | |
|     else {
 | |
|         info->errdisplay = _format_TracebackException(tbexc);
 | |
|         Py_DECREF(tbexc);
 | |
|         if (info->errdisplay == NULL) {
 | |
| #ifdef Py_DEBUG
 | |
|             PyErr_FormatUnraisable("Exception ignored while formatting TracebackException");
 | |
| #endif
 | |
|             PyErr_Clear();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return NULL;
 | |
| 
 | |
| error:
 | |
|     assert(failure != NULL);
 | |
|     _PyXI_excinfo_Clear(info);
 | |
|     return failure;
 | |
| }
 | |
| 
 | |
| static const char *
 | |
| _PyXI_excinfo_InitFromObject(_PyXI_excinfo *info, PyObject *obj)
 | |
| {
 | |
|     const char *failure = NULL;
 | |
| 
 | |
|     PyObject *exctype = PyObject_GetAttrString(obj, "type");
 | |
|     if (exctype == NULL) {
 | |
|         failure = "exception snapshot missing 'type' attribute";
 | |
|         goto error;
 | |
|     }
 | |
|     int res = _excinfo_init_type_from_object(&info->type, exctype);
 | |
|     Py_DECREF(exctype);
 | |
|     if (res < 0) {
 | |
|         failure = "error while initializing exception type snapshot";
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     // Extract the exception message.
 | |
|     PyObject *msgobj = PyObject_GetAttrString(obj, "msg");
 | |
|     if (msgobj == NULL) {
 | |
|         failure = "exception snapshot missing 'msg' attribute";
 | |
|         goto error;
 | |
|     }
 | |
|     info->msg = _copy_string_obj_raw(msgobj, NULL);
 | |
|     Py_DECREF(msgobj);
 | |
|     if (info->msg == NULL) {
 | |
|         failure = "error while copying exception message";
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     // Pickle a traceback.TracebackException.
 | |
|     PyObject *errdisplay = PyObject_GetAttrString(obj, "errdisplay");
 | |
|     if (errdisplay == NULL) {
 | |
|         failure = "exception snapshot missing 'errdisplay' attribute";
 | |
|         goto error;
 | |
|     }
 | |
|     info->errdisplay = _copy_string_obj_raw(errdisplay, NULL);
 | |
|     Py_DECREF(errdisplay);
 | |
|     if (info->errdisplay == NULL) {
 | |
|         failure = "error while copying exception error display";
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     return NULL;
 | |
| 
 | |
| error:
 | |
|     assert(failure != NULL);
 | |
|     _PyXI_excinfo_Clear(info);
 | |
|     return failure;
 | |
| }
 | |
| 
 | |
| static void
 | |
| _PyXI_excinfo_Apply(_PyXI_excinfo *info, PyObject *exctype)
 | |
| {
 | |
|     PyObject *tbexc = NULL;
 | |
|     if (info->errdisplay != NULL) {
 | |
|         tbexc = PyUnicode_FromString(info->errdisplay);
 | |
|         if (tbexc == NULL) {
 | |
|             PyErr_Clear();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     PyObject *formatted = _PyXI_excinfo_format(info);
 | |
|     PyErr_SetObject(exctype, formatted);
 | |
|     Py_DECREF(formatted);
 | |
| 
 | |
|     if (tbexc != NULL) {
 | |
|         PyObject *exc = PyErr_GetRaisedException();
 | |
|         if (PyObject_SetAttrString(exc, "_errdisplay", tbexc) < 0) {
 | |
| #ifdef Py_DEBUG
 | |
|             PyErr_FormatUnraisable("Exception ignored while "
 | |
|                                    "setting _errdisplay");
 | |
| #endif
 | |
|             PyErr_Clear();
 | |
|         }
 | |
|         Py_DECREF(tbexc);
 | |
|         PyErr_SetRaisedException(exc);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| _PyXI_excinfo_TypeAsObject(_PyXI_excinfo *info)
 | |
| {
 | |
|     PyObject *ns = _PyNamespace_New(NULL);
 | |
|     if (ns == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
|     int empty = 1;
 | |
| 
 | |
|     if (info->type.name != NULL) {
 | |
|         PyObject *name = PyUnicode_FromString(info->type.name);
 | |
|         if (name == NULL) {
 | |
|             goto error;
 | |
|         }
 | |
|         int res = PyObject_SetAttrString(ns, "__name__", name);
 | |
|         Py_DECREF(name);
 | |
|         if (res < 0) {
 | |
|             goto error;
 | |
|         }
 | |
|         empty = 0;
 | |
|     }
 | |
| 
 | |
|     if (info->type.qualname != NULL) {
 | |
|         PyObject *qualname = PyUnicode_FromString(info->type.qualname);
 | |
|         if (qualname == NULL) {
 | |
|             goto error;
 | |
|         }
 | |
|         int res = PyObject_SetAttrString(ns, "__qualname__", qualname);
 | |
|         Py_DECREF(qualname);
 | |
|         if (res < 0) {
 | |
|             goto error;
 | |
|         }
 | |
|         empty = 0;
 | |
|     }
 | |
| 
 | |
|     if (info->type.module != NULL) {
 | |
|         PyObject *module = PyUnicode_FromString(info->type.module);
 | |
|         if (module == NULL) {
 | |
|             goto error;
 | |
|         }
 | |
|         int res = PyObject_SetAttrString(ns, "__module__", module);
 | |
|         Py_DECREF(module);
 | |
|         if (res < 0) {
 | |
|             goto error;
 | |
|         }
 | |
|         empty = 0;
 | |
|     }
 | |
| 
 | |
|     if (empty) {
 | |
|         Py_CLEAR(ns);
 | |
|     }
 | |
| 
 | |
|     return ns;
 | |
| 
 | |
| error:
 | |
|     Py_DECREF(ns);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| _PyXI_excinfo_AsObject(_PyXI_excinfo *info)
 | |
| {
 | |
|     PyObject *ns = _PyNamespace_New(NULL);
 | |
|     if (ns == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
|     int res;
 | |
| 
 | |
|     PyObject *type = _PyXI_excinfo_TypeAsObject(info);
 | |
|     if (type == NULL) {
 | |
|         if (PyErr_Occurred()) {
 | |
|             goto error;
 | |
|         }
 | |
|         type = Py_NewRef(Py_None);
 | |
|     }
 | |
|     res = PyObject_SetAttrString(ns, "type", type);
 | |
|     Py_DECREF(type);
 | |
|     if (res < 0) {
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     PyObject *msg = info->msg != NULL
 | |
|         ? PyUnicode_FromString(info->msg)
 | |
|         : Py_NewRef(Py_None);
 | |
|     if (msg == NULL) {
 | |
|         goto error;
 | |
|     }
 | |
|     res = PyObject_SetAttrString(ns, "msg", msg);
 | |
|     Py_DECREF(msg);
 | |
|     if (res < 0) {
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     PyObject *formatted = _PyXI_excinfo_format(info);
 | |
|     if (formatted == NULL) {
 | |
|         goto error;
 | |
|     }
 | |
|     res = PyObject_SetAttrString(ns, "formatted", formatted);
 | |
|     Py_DECREF(formatted);
 | |
|     if (res < 0) {
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     if (info->errdisplay != NULL) {
 | |
|         PyObject *tbexc = PyUnicode_FromString(info->errdisplay);
 | |
|         if (tbexc == NULL) {
 | |
|             PyErr_Clear();
 | |
|         }
 | |
|         else {
 | |
|             res = PyObject_SetAttrString(ns, "errdisplay", tbexc);
 | |
|             Py_DECREF(tbexc);
 | |
|             if (res < 0) {
 | |
|                 goto error;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return ns;
 | |
| 
 | |
| error:
 | |
|     Py_DECREF(ns);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| _PyXI_InitExcInfo(_PyXI_excinfo *info, PyObject *exc)
 | |
| {
 | |
|     assert(!PyErr_Occurred());
 | |
|     if (exc == NULL || exc == Py_None) {
 | |
|         PyErr_SetString(PyExc_ValueError, "missing exc");
 | |
|         return -1;
 | |
|     }
 | |
|     const char *failure;
 | |
|     if (PyExceptionInstance_Check(exc) || PyExceptionClass_Check(exc)) {
 | |
|         failure = _PyXI_excinfo_InitFromException(info, exc);
 | |
|     }
 | |
|     else {
 | |
|         failure = _PyXI_excinfo_InitFromObject(info, exc);
 | |
|     }
 | |
|     if (failure != NULL) {
 | |
|         PyErr_SetString(PyExc_Exception, failure);
 | |
|         return -1;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| _PyXI_FormatExcInfo(_PyXI_excinfo *info)
 | |
| {
 | |
|     return _PyXI_excinfo_format(info);
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| _PyXI_ExcInfoAsObject(_PyXI_excinfo *info)
 | |
| {
 | |
|     return _PyXI_excinfo_AsObject(info);
 | |
| }
 | |
| 
 | |
| void
 | |
| _PyXI_ClearExcInfo(_PyXI_excinfo *info)
 | |
| {
 | |
|     _PyXI_excinfo_Clear(info);
 | |
| }
 | |
| 
 | |
| 
 | |
| /***************************/
 | |
| /* short-term data sharing */
 | |
| /***************************/
 | |
| 
 | |
| /* error codes */
 | |
| 
 | |
| static int
 | |
| _PyXI_ApplyErrorCode(_PyXI_errcode code, PyInterpreterState *interp)
 | |
| {
 | |
|     PyThreadState *tstate = _PyThreadState_GET();
 | |
| 
 | |
|     assert(!PyErr_Occurred());
 | |
|     switch (code) {
 | |
|     case _PyXI_ERR_NO_ERROR: _Py_FALLTHROUGH;
 | |
|     case _PyXI_ERR_UNCAUGHT_EXCEPTION:
 | |
|         // There is nothing to apply.
 | |
| #ifdef Py_DEBUG
 | |
|         Py_UNREACHABLE();
 | |
| #endif
 | |
|         return 0;
 | |
|     case _PyXI_ERR_OTHER:
 | |
|         // XXX msg?
 | |
|         PyErr_SetNone(PyExc_InterpreterError);
 | |
|         break;
 | |
|     case _PyXI_ERR_NO_MEMORY:
 | |
|         PyErr_NoMemory();
 | |
|         break;
 | |
|     case _PyXI_ERR_ALREADY_RUNNING:
 | |
|         assert(interp != NULL);
 | |
|         _PyErr_SetInterpreterAlreadyRunning();
 | |
|         break;
 | |
|     case _PyXI_ERR_MAIN_NS_FAILURE:
 | |
|         PyErr_SetString(PyExc_InterpreterError,
 | |
|                         "failed to get __main__ namespace");
 | |
|         break;
 | |
|     case _PyXI_ERR_APPLY_NS_FAILURE:
 | |
|         PyErr_SetString(PyExc_InterpreterError,
 | |
|                         "failed to apply namespace to __main__");
 | |
|         break;
 | |
|     case _PyXI_ERR_NOT_SHAREABLE:
 | |
|         _set_xid_lookup_failure(tstate, NULL, NULL, NULL);
 | |
|         break;
 | |
|     default:
 | |
| #ifdef Py_DEBUG
 | |
|         Py_UNREACHABLE();
 | |
| #else
 | |
|         PyErr_Format(PyExc_RuntimeError, "unsupported error code %d", code);
 | |
| #endif
 | |
|     }
 | |
|     assert(PyErr_Occurred());
 | |
|     return -1;
 | |
| }
 | |
| 
 | |
| /* shared exceptions */
 | |
| 
 | |
| static const char *
 | |
| _PyXI_InitError(_PyXI_error *error, PyObject *excobj, _PyXI_errcode code)
 | |
| {
 | |
|     if (error->interp == NULL) {
 | |
|         error->interp = PyInterpreterState_Get();
 | |
|     }
 | |
| 
 | |
|     const char *failure = NULL;
 | |
|     if (code == _PyXI_ERR_UNCAUGHT_EXCEPTION) {
 | |
|         // There is an unhandled exception we need to propagate.
 | |
|         failure = _PyXI_excinfo_InitFromException(&error->uncaught, excobj);
 | |
|         if (failure != NULL) {
 | |
|             // We failed to initialize error->uncaught.
 | |
|             // XXX Print the excobj/traceback?  Emit a warning?
 | |
|             // XXX Print the current exception/traceback?
 | |
|             if (PyErr_ExceptionMatches(PyExc_MemoryError)) {
 | |
|                 error->code = _PyXI_ERR_NO_MEMORY;
 | |
|             }
 | |
|             else {
 | |
|                 error->code = _PyXI_ERR_OTHER;
 | |
|             }
 | |
|             PyErr_Clear();
 | |
|         }
 | |
|         else {
 | |
|             error->code = code;
 | |
|         }
 | |
|         assert(error->code != _PyXI_ERR_NO_ERROR);
 | |
|     }
 | |
|     else {
 | |
|         // There is an error code we need to propagate.
 | |
|         assert(excobj == NULL);
 | |
|         assert(code != _PyXI_ERR_NO_ERROR);
 | |
|         error->code = code;
 | |
|         _PyXI_excinfo_Clear(&error->uncaught);
 | |
|     }
 | |
|     return failure;
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| _PyXI_ApplyError(_PyXI_error *error)
 | |
| {
 | |
|     PyThreadState *tstate = PyThreadState_Get();
 | |
|     if (error->code == _PyXI_ERR_UNCAUGHT_EXCEPTION) {
 | |
|         // Raise an exception that proxies the propagated exception.
 | |
|        return _PyXI_excinfo_AsObject(&error->uncaught);
 | |
|     }
 | |
|     else if (error->code == _PyXI_ERR_NOT_SHAREABLE) {
 | |
|         // Propagate the exception directly.
 | |
|         assert(!_PyErr_Occurred(tstate));
 | |
|         _set_xid_lookup_failure(tstate, NULL, error->uncaught.msg, NULL);
 | |
|     }
 | |
|     else {
 | |
|         // Raise an exception corresponding to the code.
 | |
|         assert(error->code != _PyXI_ERR_NO_ERROR);
 | |
|         (void)_PyXI_ApplyErrorCode(error->code, error->interp);
 | |
|         if (error->uncaught.type.name != NULL || error->uncaught.msg != NULL) {
 | |
|             // __context__ will be set to a proxy of the propagated exception.
 | |
|             PyObject *exc = PyErr_GetRaisedException();
 | |
|             _PyXI_excinfo_Apply(&error->uncaught, PyExc_InterpreterError);
 | |
|             PyObject *exc2 = PyErr_GetRaisedException();
 | |
|             PyException_SetContext(exc, exc2);
 | |
|             PyErr_SetRaisedException(exc);
 | |
|         }
 | |
|     }
 | |
|     assert(PyErr_Occurred());
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| /* shared namespaces */
 | |
| 
 | |
| /* Shared namespaces are expected to have relatively short lifetimes.
 | |
|    This means dealloc of a shared namespace will normally happen "soon".
 | |
|    Namespace items hold cross-interpreter data, which must get released.
 | |
|    If the namespace/items are cleared in a different interpreter than
 | |
|    where the items' cross-interpreter data was set then that will cause
 | |
|    pending calls to be used to release the cross-interpreter data.
 | |
|    The tricky bit is that the pending calls can happen sufficiently
 | |
|    later that the namespace/items might already be deallocated.  This is
 | |
|    a problem if the cross-interpreter data is allocated as part of a
 | |
|    namespace item.  If that's the case then we must ensure the shared
 | |
|    namespace is only cleared/freed *after* that data has been released. */
 | |
| 
 | |
| typedef struct _sharednsitem {
 | |
|     const char *name;
 | |
|     _PyXIData_t *xidata;
 | |
|     // We could have a "PyXIData _data" field, so it would
 | |
|     // be allocated as part of the item and avoid an extra allocation.
 | |
|     // However, doing so adds a bunch of complexity because we must
 | |
|     // ensure the item isn't freed before a pending call might happen
 | |
|     // in a different interpreter to release the XI data.
 | |
| } _PyXI_namespace_item;
 | |
| 
 | |
| static int
 | |
| _sharednsitem_is_initialized(_PyXI_namespace_item *item)
 | |
| {
 | |
|     if (item->name != NULL) {
 | |
|         return 1;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| _sharednsitem_init(_PyXI_namespace_item *item, PyObject *key)
 | |
| {
 | |
|     item->name = _copy_string_obj_raw(key, NULL);
 | |
|     if (item->name == NULL) {
 | |
|         assert(!_sharednsitem_is_initialized(item));
 | |
|         return -1;
 | |
|     }
 | |
|     item->xidata = NULL;
 | |
|     assert(_sharednsitem_is_initialized(item));
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| _sharednsitem_has_value(_PyXI_namespace_item *item, int64_t *p_interpid)
 | |
| {
 | |
|     if (item->xidata == NULL) {
 | |
|         return 0;
 | |
|     }
 | |
|     if (p_interpid != NULL) {
 | |
|         *p_interpid = _PyXIData_INTERPID(item->xidata);
 | |
|     }
 | |
|     return 1;
 | |
| }
 | |
| 
 | |
| static int
 | |
| _sharednsitem_set_value(_PyXI_namespace_item *item, PyObject *value)
 | |
| {
 | |
|     assert(_sharednsitem_is_initialized(item));
 | |
|     assert(item->xidata == NULL);
 | |
|     item->xidata = _PyXIData_New();
 | |
|     if (item->xidata == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     PyThreadState *tstate = PyThreadState_Get();
 | |
|     if (_PyObject_GetXIData(tstate, value, item->xidata) != 0) {
 | |
|         PyMem_RawFree(item->xidata);
 | |
|         item->xidata = NULL;
 | |
|         // The caller may want to propagate PyExc_NotShareableError
 | |
|         // if currently switched between interpreters.
 | |
|         return -1;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static void
 | |
| _sharednsitem_clear_value(_PyXI_namespace_item *item)
 | |
| {
 | |
|     _PyXIData_t *xidata = item->xidata;
 | |
|     if (xidata != NULL) {
 | |
|         item->xidata = NULL;
 | |
|         int rawfree = 1;
 | |
|         (void)_release_xid_data(xidata, rawfree);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| _sharednsitem_clear(_PyXI_namespace_item *item)
 | |
| {
 | |
|     if (item->name != NULL) {
 | |
|         PyMem_RawFree((void *)item->name);
 | |
|         item->name = NULL;
 | |
|     }
 | |
|     _sharednsitem_clear_value(item);
 | |
| }
 | |
| 
 | |
| static int
 | |
| _sharednsitem_copy_from_ns(struct _sharednsitem *item, PyObject *ns)
 | |
| {
 | |
|     assert(item->name != NULL);
 | |
|     assert(item->xidata == NULL);
 | |
|     PyObject *value = PyDict_GetItemString(ns, item->name);  // borrowed
 | |
|     if (value == NULL) {
 | |
|         if (PyErr_Occurred()) {
 | |
|             return -1;
 | |
|         }
 | |
|         // When applied, this item will be set to the default (or fail).
 | |
|         return 0;
 | |
|     }
 | |
|     if (_sharednsitem_set_value(item, value) < 0) {
 | |
|         return -1;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| _sharednsitem_apply(_PyXI_namespace_item *item, PyObject *ns, PyObject *dflt)
 | |
| {
 | |
|     PyObject *name = PyUnicode_FromString(item->name);
 | |
|     if (name == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     PyObject *value;
 | |
|     if (item->xidata != NULL) {
 | |
|         value = _PyXIData_NewObject(item->xidata);
 | |
|         if (value == NULL) {
 | |
|             Py_DECREF(name);
 | |
|             return -1;
 | |
|         }
 | |
|     }
 | |
|     else {
 | |
|         value = Py_NewRef(dflt);
 | |
|     }
 | |
|     int res = PyDict_SetItem(ns, name, value);
 | |
|     Py_DECREF(name);
 | |
|     Py_DECREF(value);
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| struct _sharedns {
 | |
|     Py_ssize_t len;
 | |
|     _PyXI_namespace_item *items;
 | |
| };
 | |
| 
 | |
| static _PyXI_namespace *
 | |
| _sharedns_new(void)
 | |
| {
 | |
|     _PyXI_namespace *ns = PyMem_RawCalloc(sizeof(_PyXI_namespace), 1);
 | |
|     if (ns == NULL) {
 | |
|         PyErr_NoMemory();
 | |
|         return NULL;
 | |
|     }
 | |
|     *ns = (_PyXI_namespace){ 0 };
 | |
|     return ns;
 | |
| }
 | |
| 
 | |
| static int
 | |
| _sharedns_is_initialized(_PyXI_namespace *ns)
 | |
| {
 | |
|     if (ns->len == 0) {
 | |
|         assert(ns->items == NULL);
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     assert(ns->len > 0);
 | |
|     assert(ns->items != NULL);
 | |
|     assert(_sharednsitem_is_initialized(&ns->items[0]));
 | |
|     assert(ns->len == 1
 | |
|            || _sharednsitem_is_initialized(&ns->items[ns->len - 1]));
 | |
|     return 1;
 | |
| }
 | |
| 
 | |
| #define HAS_COMPLETE_DATA 1
 | |
| #define HAS_PARTIAL_DATA 2
 | |
| 
 | |
| static int
 | |
| _sharedns_has_xidata(_PyXI_namespace *ns, int64_t *p_interpid)
 | |
| {
 | |
|     // We expect _PyXI_namespace to always be initialized.
 | |
|     assert(_sharedns_is_initialized(ns));
 | |
|     int res = 0;
 | |
|     _PyXI_namespace_item *item0 = &ns->items[0];
 | |
|     if (!_sharednsitem_is_initialized(item0)) {
 | |
|         return 0;
 | |
|     }
 | |
|     int64_t interpid0 = -1;
 | |
|     if (!_sharednsitem_has_value(item0, &interpid0)) {
 | |
|         return 0;
 | |
|     }
 | |
|     if (ns->len > 1) {
 | |
|         // At this point we know it is has at least partial data.
 | |
|         _PyXI_namespace_item *itemN = &ns->items[ns->len-1];
 | |
|         if (!_sharednsitem_is_initialized(itemN)) {
 | |
|             res = HAS_PARTIAL_DATA;
 | |
|             goto finally;
 | |
|         }
 | |
|         int64_t interpidN = -1;
 | |
|         if (!_sharednsitem_has_value(itemN, &interpidN)) {
 | |
|             res = HAS_PARTIAL_DATA;
 | |
|             goto finally;
 | |
|         }
 | |
|         assert(interpidN == interpid0);
 | |
|     }
 | |
|     res = HAS_COMPLETE_DATA;
 | |
|     *p_interpid = interpid0;
 | |
| 
 | |
| finally:
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| static void
 | |
| _sharedns_clear(_PyXI_namespace *ns)
 | |
| {
 | |
|     if (!_sharedns_is_initialized(ns)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     // If the cross-interpreter data were allocated as part of
 | |
|     // _PyXI_namespace_item (instead of dynamically), this is where
 | |
|     // we would need verify that we are clearing the items in the
 | |
|     // correct interpreter, to avoid a race with releasing the XI data
 | |
|     // via a pending call.  See _sharedns_has_xidata().
 | |
|     for (Py_ssize_t i=0; i < ns->len; i++) {
 | |
|         _sharednsitem_clear(&ns->items[i]);
 | |
|     }
 | |
|     PyMem_RawFree(ns->items);
 | |
|     ns->items = NULL;
 | |
|     ns->len = 0;
 | |
| }
 | |
| 
 | |
| static void
 | |
| _sharedns_free(_PyXI_namespace *ns)
 | |
| {
 | |
|     _sharedns_clear(ns);
 | |
|     PyMem_RawFree(ns);
 | |
| }
 | |
| 
 | |
| static int
 | |
| _sharedns_init(_PyXI_namespace *ns, PyObject *names)
 | |
| {
 | |
|     assert(!_sharedns_is_initialized(ns));
 | |
|     assert(names != NULL);
 | |
|     Py_ssize_t len = PyDict_CheckExact(names)
 | |
|         ? PyDict_Size(names)
 | |
|         : PySequence_Size(names);
 | |
|     if (len < 0) {
 | |
|         return -1;
 | |
|     }
 | |
|     if (len == 0) {
 | |
|         PyErr_SetString(PyExc_ValueError, "empty namespaces not allowed");
 | |
|         return -1;
 | |
|     }
 | |
|     assert(len > 0);
 | |
| 
 | |
|     // Allocate the items.
 | |
|     _PyXI_namespace_item *items =
 | |
|             PyMem_RawCalloc(sizeof(struct _sharednsitem), len);
 | |
|     if (items == NULL) {
 | |
|         PyErr_NoMemory();
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     // Fill in the names.
 | |
|     Py_ssize_t i = -1;
 | |
|     if (PyDict_CheckExact(names)) {
 | |
|         Py_ssize_t pos = 0;
 | |
|         for (i=0; i < len; i++) {
 | |
|             PyObject *key;
 | |
|             if (!PyDict_Next(names, &pos, &key, NULL)) {
 | |
|                 // This should not be possible.
 | |
|                 assert(0);
 | |
|                 goto error;
 | |
|             }
 | |
|             if (_sharednsitem_init(&items[i], key) < 0) {
 | |
|                 goto error;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     else if (PySequence_Check(names)) {
 | |
|         for (i=0; i < len; i++) {
 | |
|             PyObject *key = PySequence_GetItem(names, i);
 | |
|             if (key == NULL) {
 | |
|                 goto error;
 | |
|             }
 | |
|             int res = _sharednsitem_init(&items[i], key);
 | |
|             Py_DECREF(key);
 | |
|             if (res < 0) {
 | |
|                 goto error;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     else {
 | |
|         PyErr_SetString(PyExc_NotImplementedError,
 | |
|                         "non-sequence namespace not supported");
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     ns->items = items;
 | |
|     ns->len = len;
 | |
|     assert(_sharedns_is_initialized(ns));
 | |
|     return 0;
 | |
| 
 | |
| error:
 | |
|     for (Py_ssize_t j=0; j < i; j++) {
 | |
|         _sharednsitem_clear(&items[j]);
 | |
|     }
 | |
|     PyMem_RawFree(items);
 | |
|     assert(!_sharedns_is_initialized(ns));
 | |
|     return -1;
 | |
| }
 | |
| 
 | |
| void
 | |
| _PyXI_FreeNamespace(_PyXI_namespace *ns)
 | |
| {
 | |
|     if (!_sharedns_is_initialized(ns)) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     int64_t interpid = -1;
 | |
|     if (!_sharedns_has_xidata(ns, &interpid)) {
 | |
|         _sharedns_free(ns);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (interpid == PyInterpreterState_GetID(PyInterpreterState_Get())) {
 | |
|         _sharedns_free(ns);
 | |
|     }
 | |
|     else {
 | |
|         // If we weren't always dynamically allocating the cross-interpreter
 | |
|         // data in each item then we would need to using a pending call
 | |
|         // to call _sharedns_free(), to avoid the race between freeing
 | |
|         // the shared namespace and releasing the XI data.
 | |
|         _sharedns_free(ns);
 | |
|     }
 | |
| }
 | |
| 
 | |
| _PyXI_namespace *
 | |
| _PyXI_NamespaceFromNames(PyObject *names)
 | |
| {
 | |
|     if (names == NULL || names == Py_None) {
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     _PyXI_namespace *ns = _sharedns_new();
 | |
|     if (ns == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (_sharedns_init(ns, names) < 0) {
 | |
|         PyMem_RawFree(ns);
 | |
|         if (PySequence_Size(names) == 0) {
 | |
|             PyErr_Clear();
 | |
|         }
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     return ns;
 | |
| }
 | |
| 
 | |
| #ifndef NDEBUG
 | |
| static int _session_is_active(_PyXI_session *);
 | |
| #endif
 | |
| static void _propagate_not_shareable_error(_PyXI_session *);
 | |
| 
 | |
| int
 | |
| _PyXI_FillNamespaceFromDict(_PyXI_namespace *ns, PyObject *nsobj,
 | |
|                             _PyXI_session *session)
 | |
| {
 | |
|     // session must be entered already, if provided.
 | |
|     assert(session == NULL || _session_is_active(session));
 | |
|     assert(_sharedns_is_initialized(ns));
 | |
|     for (Py_ssize_t i=0; i < ns->len; i++) {
 | |
|         _PyXI_namespace_item *item = &ns->items[i];
 | |
|         if (_sharednsitem_copy_from_ns(item, nsobj) < 0) {
 | |
|             _propagate_not_shareable_error(session);
 | |
|             // Clear out the ones we set so far.
 | |
|             for (Py_ssize_t j=0; j < i; j++) {
 | |
|                 _sharednsitem_clear_value(&ns->items[j]);
 | |
|             }
 | |
|             return -1;
 | |
|         }
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| // All items are expected to be shareable.
 | |
| static _PyXI_namespace *
 | |
| _PyXI_NamespaceFromDict(PyObject *nsobj, _PyXI_session *session)
 | |
| {
 | |
|     // session must be entered already, if provided.
 | |
|     assert(session == NULL || _session_is_active(session));
 | |
|     if (nsobj == NULL || nsobj == Py_None) {
 | |
|         return NULL;
 | |
|     }
 | |
|     if (!PyDict_CheckExact(nsobj)) {
 | |
|         PyErr_SetString(PyExc_TypeError, "expected a dict");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     _PyXI_namespace *ns = _sharedns_new();
 | |
|     if (ns == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (_sharedns_init(ns, nsobj) < 0) {
 | |
|         if (PyDict_Size(nsobj) == 0) {
 | |
|             PyMem_RawFree(ns);
 | |
|             PyErr_Clear();
 | |
|             return NULL;
 | |
|         }
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     if (_PyXI_FillNamespaceFromDict(ns, nsobj, session) < 0) {
 | |
|         goto error;
 | |
|     }
 | |
| 
 | |
|     return ns;
 | |
| 
 | |
| error:
 | |
|     assert(PyErr_Occurred()
 | |
|            || (session != NULL && session->error_override != NULL));
 | |
|     _sharedns_free(ns);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| int
 | |
| _PyXI_ApplyNamespace(_PyXI_namespace *ns, PyObject *nsobj, PyObject *dflt)
 | |
| {
 | |
|     for (Py_ssize_t i=0; i < ns->len; i++) {
 | |
|         if (_sharednsitem_apply(&ns->items[i], nsobj, dflt) != 0) {
 | |
|             return -1;
 | |
|         }
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**********************/
 | |
| /* high-level helpers */
 | |
| /**********************/
 | |
| 
 | |
| /* enter/exit a cross-interpreter session */
 | |
| 
 | |
| static void
 | |
| _enter_session(_PyXI_session *session, PyInterpreterState *interp)
 | |
| {
 | |
|     // Set here and cleared in _exit_session().
 | |
|     assert(!session->own_init_tstate);
 | |
|     assert(session->init_tstate == NULL);
 | |
|     assert(session->prev_tstate == NULL);
 | |
|     // Set elsewhere and cleared in _exit_session().
 | |
|     assert(!session->running);
 | |
|     assert(session->main_ns == NULL);
 | |
|     // Set elsewhere and cleared in _capture_current_exception().
 | |
|     assert(session->error_override == NULL);
 | |
|     // Set elsewhere and cleared in _PyXI_ApplyCapturedException().
 | |
|     assert(session->error == NULL);
 | |
| 
 | |
|     // Switch to interpreter.
 | |
|     PyThreadState *tstate = PyThreadState_Get();
 | |
|     PyThreadState *prev = tstate;
 | |
|     if (interp != tstate->interp) {
 | |
|         tstate = _PyThreadState_NewBound(interp, _PyThreadState_WHENCE_EXEC);
 | |
|         // XXX Possible GILState issues?
 | |
|         session->prev_tstate = PyThreadState_Swap(tstate);
 | |
|         assert(session->prev_tstate == prev);
 | |
|         session->own_init_tstate = 1;
 | |
|     }
 | |
|     session->init_tstate = tstate;
 | |
|     session->prev_tstate = prev;
 | |
| }
 | |
| 
 | |
| static void
 | |
| _exit_session(_PyXI_session *session)
 | |
| {
 | |
|     PyThreadState *tstate = session->init_tstate;
 | |
|     assert(tstate != NULL);
 | |
|     assert(PyThreadState_Get() == tstate);
 | |
| 
 | |
|     // Release any of the entered interpreters resources.
 | |
|     if (session->main_ns != NULL) {
 | |
|         Py_CLEAR(session->main_ns);
 | |
|     }
 | |
| 
 | |
|     // Ensure this thread no longer owns __main__.
 | |
|     if (session->running) {
 | |
|         _PyInterpreterState_SetNotRunningMain(tstate->interp);
 | |
|         assert(!PyErr_Occurred());
 | |
|         session->running = 0;
 | |
|     }
 | |
| 
 | |
|     // Switch back.
 | |
|     assert(session->prev_tstate != NULL);
 | |
|     if (session->prev_tstate != session->init_tstate) {
 | |
|         assert(session->own_init_tstate);
 | |
|         session->own_init_tstate = 0;
 | |
|         PyThreadState_Clear(tstate);
 | |
|         PyThreadState_Swap(session->prev_tstate);
 | |
|         PyThreadState_Delete(tstate);
 | |
|     }
 | |
|     else {
 | |
|         assert(!session->own_init_tstate);
 | |
|     }
 | |
|     session->prev_tstate = NULL;
 | |
|     session->init_tstate = NULL;
 | |
| }
 | |
| 
 | |
| #ifndef NDEBUG
 | |
| static int
 | |
| _session_is_active(_PyXI_session *session)
 | |
| {
 | |
|     return (session->init_tstate != NULL);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static void
 | |
| _propagate_not_shareable_error(_PyXI_session *session)
 | |
| {
 | |
|     if (session == NULL) {
 | |
|         return;
 | |
|     }
 | |
|     PyThreadState *tstate = PyThreadState_Get();
 | |
|     PyObject *exctype = get_notshareableerror_type(tstate);
 | |
|     if (exctype == NULL) {
 | |
|         PyErr_FormatUnraisable(
 | |
|                 "Exception ignored while propagating not shareable error");
 | |
|         return;
 | |
|     }
 | |
|     if (PyErr_ExceptionMatches(exctype)) {
 | |
|         // We want to propagate the exception directly.
 | |
|         session->_error_override = _PyXI_ERR_NOT_SHAREABLE;
 | |
|         session->error_override = &session->_error_override;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| _capture_current_exception(_PyXI_session *session)
 | |
| {
 | |
|     assert(session->error == NULL);
 | |
|     if (!PyErr_Occurred()) {
 | |
|         assert(session->error_override == NULL);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     // Handle the exception override.
 | |
|     _PyXI_errcode *override = session->error_override;
 | |
|     session->error_override = NULL;
 | |
|     _PyXI_errcode errcode = override != NULL
 | |
|         ? *override
 | |
|         : _PyXI_ERR_UNCAUGHT_EXCEPTION;
 | |
| 
 | |
|     // Pop the exception object.
 | |
|     PyObject *excval = NULL;
 | |
|     if (errcode == _PyXI_ERR_UNCAUGHT_EXCEPTION) {
 | |
|         // We want to actually capture the current exception.
 | |
|         excval = PyErr_GetRaisedException();
 | |
|     }
 | |
|     else if (errcode == _PyXI_ERR_ALREADY_RUNNING) {
 | |
|         // We don't need the exception info.
 | |
|         PyErr_Clear();
 | |
|     }
 | |
|     else {
 | |
|         // We could do a variety of things here, depending on errcode.
 | |
|         // However, for now we simply capture the exception and save
 | |
|         // the errcode.
 | |
|         excval = PyErr_GetRaisedException();
 | |
|     }
 | |
| 
 | |
|     // Capture the exception.
 | |
|     _PyXI_error *err = &session->_error;
 | |
|     *err = (_PyXI_error){
 | |
|         .interp = session->init_tstate->interp,
 | |
|     };
 | |
|     const char *failure;
 | |
|     if (excval == NULL) {
 | |
|         failure = _PyXI_InitError(err, NULL, errcode);
 | |
|     }
 | |
|     else {
 | |
|         failure = _PyXI_InitError(err, excval, _PyXI_ERR_UNCAUGHT_EXCEPTION);
 | |
|         Py_DECREF(excval);
 | |
|         if (failure == NULL && override != NULL) {
 | |
|             err->code = errcode;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Handle capture failure.
 | |
|     if (failure != NULL) {
 | |
|         // XXX Make this error message more generic.
 | |
|         fprintf(stderr,
 | |
|                 "RunFailedError: script raised an uncaught exception (%s)",
 | |
|                 failure);
 | |
|         err = NULL;
 | |
|     }
 | |
| 
 | |
|     // Finished!
 | |
|     assert(!PyErr_Occurred());
 | |
|     session->error  = err;
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| _PyXI_ApplyCapturedException(_PyXI_session *session)
 | |
| {
 | |
|     assert(!PyErr_Occurred());
 | |
|     assert(session->error != NULL);
 | |
|     PyObject *res = _PyXI_ApplyError(session->error);
 | |
|     assert((res == NULL) != (PyErr_Occurred() == NULL));
 | |
|     session->error = NULL;
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| int
 | |
| _PyXI_HasCapturedException(_PyXI_session *session)
 | |
| {
 | |
|     return session->error != NULL;
 | |
| }
 | |
| 
 | |
| int
 | |
| _PyXI_Enter(_PyXI_session *session,
 | |
|             PyInterpreterState *interp, PyObject *nsupdates)
 | |
| {
 | |
|     // Convert the attrs for cross-interpreter use.
 | |
|     _PyXI_namespace *sharedns = NULL;
 | |
|     if (nsupdates != NULL) {
 | |
|         sharedns = _PyXI_NamespaceFromDict(nsupdates, NULL);
 | |
|         if (sharedns == NULL && PyErr_Occurred()) {
 | |
|             assert(session->error == NULL);
 | |
|             return -1;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Switch to the requested interpreter (if necessary).
 | |
|     _enter_session(session, interp);
 | |
|     PyThreadState *session_tstate = session->init_tstate;
 | |
|     _PyXI_errcode errcode = _PyXI_ERR_UNCAUGHT_EXCEPTION;
 | |
| 
 | |
|     // Ensure this thread owns __main__.
 | |
|     if (_PyInterpreterState_SetRunningMain(interp) < 0) {
 | |
|         // In the case where we didn't switch interpreters, it would
 | |
|         // be more efficient to leave the exception in place and return
 | |
|         // immediately.  However, life is simpler if we don't.
 | |
|         errcode = _PyXI_ERR_ALREADY_RUNNING;
 | |
|         goto error;
 | |
|     }
 | |
|     session->running = 1;
 | |
| 
 | |
|     // Cache __main__.__dict__.
 | |
|     PyObject *main_mod = _Py_GetMainModule(session_tstate);
 | |
|     if (_Py_CheckMainModule(main_mod) < 0) {
 | |
|         errcode = _PyXI_ERR_MAIN_NS_FAILURE;
 | |
|         goto error;
 | |
|     }
 | |
|     PyObject *ns = PyModule_GetDict(main_mod);  // borrowed
 | |
|     Py_DECREF(main_mod);
 | |
|     if (ns == NULL) {
 | |
|         errcode = _PyXI_ERR_MAIN_NS_FAILURE;
 | |
|         goto error;
 | |
|     }
 | |
|     session->main_ns = Py_NewRef(ns);
 | |
| 
 | |
|     // Apply the cross-interpreter data.
 | |
|     if (sharedns != NULL) {
 | |
|         if (_PyXI_ApplyNamespace(sharedns, ns, NULL) < 0) {
 | |
|             errcode = _PyXI_ERR_APPLY_NS_FAILURE;
 | |
|             goto error;
 | |
|         }
 | |
|         _PyXI_FreeNamespace(sharedns);
 | |
|     }
 | |
| 
 | |
|     errcode = _PyXI_ERR_NO_ERROR;
 | |
|     assert(!PyErr_Occurred());
 | |
|     return 0;
 | |
| 
 | |
| error:
 | |
|     assert(PyErr_Occurred());
 | |
|     // We want to propagate all exceptions here directly (best effort).
 | |
|     assert(errcode != _PyXI_ERR_UNCAUGHT_EXCEPTION);
 | |
|     session->error_override = &errcode;
 | |
|     _capture_current_exception(session);
 | |
|     _exit_session(session);
 | |
|     if (sharedns != NULL) {
 | |
|         _PyXI_FreeNamespace(sharedns);
 | |
|     }
 | |
|     return -1;
 | |
| }
 | |
| 
 | |
| void
 | |
| _PyXI_Exit(_PyXI_session *session)
 | |
| {
 | |
|     _capture_current_exception(session);
 | |
|     _exit_session(session);
 | |
| }
 | |
| 
 | |
| 
 | |
| /*********************/
 | |
| /* runtime lifecycle */
 | |
| /*********************/
 | |
| 
 | |
| int
 | |
| _Py_xi_global_state_init(_PyXI_global_state_t *state)
 | |
| {
 | |
|     assert(state != NULL);
 | |
|     xid_lookup_init(&state->data_lookup);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| void
 | |
| _Py_xi_global_state_fini(_PyXI_global_state_t *state)
 | |
| {
 | |
|     assert(state != NULL);
 | |
|     xid_lookup_fini(&state->data_lookup);
 | |
| }
 | |
| 
 | |
| int
 | |
| _Py_xi_state_init(_PyXI_state_t *state, PyInterpreterState *interp)
 | |
| {
 | |
|     assert(state != NULL);
 | |
|     assert(interp == NULL || state == _PyXI_GET_STATE(interp));
 | |
| 
 | |
|     xid_lookup_init(&state->data_lookup);
 | |
| 
 | |
|     // Initialize exceptions.
 | |
|     if (interp != NULL) {
 | |
|         if (init_static_exctypes(&state->exceptions, interp) < 0) {
 | |
|             fini_heap_exctypes(&state->exceptions);
 | |
|             return -1;
 | |
|         }
 | |
|     }
 | |
|     if (init_heap_exctypes(&state->exceptions) < 0) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| void
 | |
| _Py_xi_state_fini(_PyXI_state_t *state, PyInterpreterState *interp)
 | |
| {
 | |
|     assert(state != NULL);
 | |
|     assert(interp == NULL || state == _PyXI_GET_STATE(interp));
 | |
| 
 | |
|     fini_heap_exctypes(&state->exceptions);
 | |
|     if (interp != NULL) {
 | |
|         fini_static_exctypes(&state->exceptions, interp);
 | |
|     }
 | |
| 
 | |
|     xid_lookup_fini(&state->data_lookup);
 | |
| }
 | |
| 
 | |
| 
 | |
| PyStatus
 | |
| _PyXI_Init(PyInterpreterState *interp)
 | |
| {
 | |
|     if (_Py_IsMainInterpreter(interp)) {
 | |
|         _PyXI_global_state_t *global_state = _PyXI_GET_GLOBAL_STATE(interp);
 | |
|         if (global_state == NULL) {
 | |
|             PyErr_PrintEx(0);
 | |
|             return _PyStatus_ERR(
 | |
|                     "failed to get global cross-interpreter state");
 | |
|         }
 | |
|         if (_Py_xi_global_state_init(global_state) < 0) {
 | |
|             PyErr_PrintEx(0);
 | |
|             return _PyStatus_ERR(
 | |
|                     "failed to initialize  global cross-interpreter state");
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     _PyXI_state_t *state = _PyXI_GET_STATE(interp);
 | |
|     if (state == NULL) {
 | |
|         PyErr_PrintEx(0);
 | |
|         return _PyStatus_ERR(
 | |
|                 "failed to get interpreter's cross-interpreter state");
 | |
|     }
 | |
|     // The static types were already initialized in _PyXI_InitTypes(),
 | |
|     // so we pass in NULL here to avoid initializing them again.
 | |
|     if (_Py_xi_state_init(state, NULL) < 0) {
 | |
|         PyErr_PrintEx(0);
 | |
|         return _PyStatus_ERR(
 | |
|                 "failed to initialize interpreter's cross-interpreter state");
 | |
|     }
 | |
| 
 | |
|     return _PyStatus_OK();
 | |
| }
 | |
| 
 | |
| // _PyXI_Fini() must be called before the interpreter is cleared,
 | |
| // since we must clear some heap objects.
 | |
| 
 | |
| void
 | |
| _PyXI_Fini(PyInterpreterState *interp)
 | |
| {
 | |
|     _PyXI_state_t *state = _PyXI_GET_STATE(interp);
 | |
| #ifndef NDEBUG
 | |
|     if (state == NULL) {
 | |
|         PyErr_PrintEx(0);
 | |
|         return;
 | |
|     }
 | |
| #endif
 | |
|     // The static types will be finalized soon in _PyXI_FiniTypes(),
 | |
|     // so we pass in NULL here to avoid finalizing them right now.
 | |
|     _Py_xi_state_fini(state, NULL);
 | |
| 
 | |
|     if (_Py_IsMainInterpreter(interp)) {
 | |
|         _PyXI_global_state_t *global_state = _PyXI_GET_GLOBAL_STATE(interp);
 | |
|         _Py_xi_global_state_fini(global_state);
 | |
|     }
 | |
| }
 | |
| 
 | |
| PyStatus
 | |
| _PyXI_InitTypes(PyInterpreterState *interp)
 | |
| {
 | |
|     if (init_static_exctypes(&_PyXI_GET_STATE(interp)->exceptions, interp) < 0) {
 | |
|         PyErr_PrintEx(0);
 | |
|         return _PyStatus_ERR(
 | |
|                 "failed to initialize the cross-interpreter exception types");
 | |
|     }
 | |
|     // We would initialize heap types here too but that leads to ref leaks.
 | |
|     // Instead, we intialize them in _PyXI_Init().
 | |
|     return _PyStatus_OK();
 | |
| }
 | |
| 
 | |
| void
 | |
| _PyXI_FiniTypes(PyInterpreterState *interp)
 | |
| {
 | |
|     // We would finalize heap types here too but that leads to ref leaks.
 | |
|     // Instead, we finalize them in _PyXI_Fini().
 | |
|     fini_static_exctypes(&_PyXI_GET_STATE(interp)->exceptions, interp);
 | |
| }
 | |
| 
 | |
| 
 | |
| /*************/
 | |
| /* other API */
 | |
| /*************/
 | |
| 
 | |
| PyInterpreterState *
 | |
| _PyXI_NewInterpreter(PyInterpreterConfig *config, long *maybe_whence,
 | |
|                      PyThreadState **p_tstate, PyThreadState **p_save_tstate)
 | |
| {
 | |
|     PyThreadState *save_tstate = PyThreadState_Swap(NULL);
 | |
|     assert(save_tstate != NULL);
 | |
| 
 | |
|     PyThreadState *tstate;
 | |
|     PyStatus status = Py_NewInterpreterFromConfig(&tstate, config);
 | |
|     if (PyStatus_Exception(status)) {
 | |
|         // Since no new thread state was created, there is no exception
 | |
|         // to propagate; raise a fresh one after swapping back in the
 | |
|         // old thread state.
 | |
|         PyThreadState_Swap(save_tstate);
 | |
|         _PyErr_SetFromPyStatus(status);
 | |
|         PyObject *exc = PyErr_GetRaisedException();
 | |
|         PyErr_SetString(PyExc_InterpreterError,
 | |
|                         "sub-interpreter creation failed");
 | |
|         _PyErr_ChainExceptions1(exc);
 | |
|         return NULL;
 | |
|     }
 | |
|     assert(tstate != NULL);
 | |
|     PyInterpreterState *interp = PyThreadState_GetInterpreter(tstate);
 | |
| 
 | |
|     long whence = _PyInterpreterState_WHENCE_XI;
 | |
|     if (maybe_whence != NULL) {
 | |
|         whence = *maybe_whence;
 | |
|     }
 | |
|     _PyInterpreterState_SetWhence(interp, whence);
 | |
| 
 | |
|     if (p_tstate != NULL) {
 | |
|         // We leave the new thread state as the current one.
 | |
|         *p_tstate = tstate;
 | |
|     }
 | |
|     else {
 | |
|         // Throw away the initial tstate.
 | |
|         PyThreadState_Clear(tstate);
 | |
|         PyThreadState_Swap(save_tstate);
 | |
|         PyThreadState_Delete(tstate);
 | |
|         save_tstate = NULL;
 | |
|     }
 | |
|     if (p_save_tstate != NULL) {
 | |
|         *p_save_tstate = save_tstate;
 | |
|     }
 | |
|     return interp;
 | |
| }
 | |
| 
 | |
| void
 | |
| _PyXI_EndInterpreter(PyInterpreterState *interp,
 | |
|                      PyThreadState *tstate, PyThreadState **p_save_tstate)
 | |
| {
 | |
| #ifndef NDEBUG
 | |
|     long whence = _PyInterpreterState_GetWhence(interp);
 | |
| #endif
 | |
|     assert(whence != _PyInterpreterState_WHENCE_RUNTIME);
 | |
| 
 | |
|     if (!_PyInterpreterState_IsReady(interp)) {
 | |
|         assert(whence == _PyInterpreterState_WHENCE_UNKNOWN);
 | |
|         // PyInterpreterState_Clear() requires the GIL,
 | |
|         // which a not-ready does not have, so we don't clear it.
 | |
|         // That means there may be leaks here until clearing the
 | |
|         // interpreter is fixed.
 | |
|         PyInterpreterState_Delete(interp);
 | |
|         return;
 | |
|     }
 | |
|     assert(whence != _PyInterpreterState_WHENCE_UNKNOWN);
 | |
| 
 | |
|     PyThreadState *save_tstate = NULL;
 | |
|     PyThreadState *cur_tstate = PyThreadState_GET();
 | |
|     if (tstate == NULL) {
 | |
|         if (PyThreadState_GetInterpreter(cur_tstate) == interp) {
 | |
|             tstate = cur_tstate;
 | |
|         }
 | |
|         else {
 | |
|             tstate = _PyThreadState_NewBound(interp, _PyThreadState_WHENCE_FINI);
 | |
|             assert(tstate != NULL);
 | |
|             save_tstate = PyThreadState_Swap(tstate);
 | |
|         }
 | |
|     }
 | |
|     else {
 | |
|         assert(PyThreadState_GetInterpreter(tstate) == interp);
 | |
|         if (tstate != cur_tstate) {
 | |
|             assert(PyThreadState_GetInterpreter(cur_tstate) != interp);
 | |
|             save_tstate = PyThreadState_Swap(tstate);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     Py_EndInterpreter(tstate);
 | |
| 
 | |
|     if (p_save_tstate != NULL) {
 | |
|         save_tstate = *p_save_tstate;
 | |
|     }
 | |
|     PyThreadState_Swap(save_tstate);
 | |
| }
 |