|  |  | 
|  | static crossinterpdatafunc _lookup_getdata_from_registry( | 
|  | PyInterpreterState *, PyObject *); | 
|  |  | 
|  | static crossinterpdatafunc | 
|  | lookup_getdata(PyInterpreterState *interp, PyObject *obj) | 
|  | { | 
|  | /* Cross-interpreter objects are looked up by exact match on the class. | 
|  | We can reassess this policy when we move from a global registry to a | 
|  | tp_* slot. */ | 
|  | return _lookup_getdata_from_registry(interp, obj); | 
|  | } | 
|  |  | 
|  | crossinterpdatafunc | 
|  | _PyCrossInterpreterData_Lookup(PyObject *obj) | 
|  | { | 
|  | PyInterpreterState *interp = PyInterpreterState_Get(); | 
|  | return lookup_getdata(interp, obj); | 
|  | } | 
|  |  | 
|  |  | 
|  | /***********************************************/ | 
|  | /* a registry of {type -> crossinterpdatafunc} */ | 
|  | /***********************************************/ | 
|  |  | 
|  | /* For now we use a global registry of shareable classes.  An | 
|  | alternative would be to add a tp_* slot for a class's | 
|  | crossinterpdatafunc. It would be simpler and more efficient.  */ | 
|  |  | 
|  |  | 
|  | /* registry lifecycle */ | 
|  |  | 
|  | static void _register_builtins_for_crossinterpreter_data(struct _xidregistry *); | 
|  |  | 
|  | static void | 
|  | _xidregistry_init(struct _xidregistry *registry) | 
|  | { | 
|  | if (registry->initialized) { | 
|  | return; | 
|  | } | 
|  | registry->initialized = 1; | 
|  |  | 
|  | if (registry->global) { | 
|  | // Registering the builtins is cheap so we don't bother doing it lazily. | 
|  | assert(registry->head == NULL); | 
|  | _register_builtins_for_crossinterpreter_data(registry); | 
|  | } | 
|  | } | 
|  |  | 
|  | static void _xidregistry_clear(struct _xidregistry *); | 
|  |  | 
|  | static void | 
|  | _xidregistry_fini(struct _xidregistry *registry) | 
|  | { | 
|  | if (!registry->initialized) { | 
|  | return; | 
|  | } | 
|  | registry->initialized = 0; | 
|  |  | 
|  | _xidregistry_clear(registry); | 
|  | } | 
|  |  | 
|  | static inline struct _xidregistry * _get_global_xidregistry(_PyRuntimeState *); | 
|  | static inline struct _xidregistry * _get_xidregistry(PyInterpreterState *); | 
|  |  | 
|  | static void | 
|  | xid_lookup_init(PyInterpreterState *interp) | 
|  | { | 
|  | if (_Py_IsMainInterpreter(interp)) { | 
|  | _xidregistry_init(_get_global_xidregistry(interp->runtime)); | 
|  | } | 
|  | _xidregistry_init(_get_xidregistry(interp)); | 
|  | } | 
|  |  | 
|  | static void | 
|  | xid_lookup_fini(PyInterpreterState *interp) | 
|  | { | 
|  | _xidregistry_fini(_get_xidregistry(interp)); | 
|  | if (_Py_IsMainInterpreter(interp)) { | 
|  | _xidregistry_fini(_get_global_xidregistry(interp->runtime)); | 
|  | } | 
|  | } | 
|  |  | 
|  |  | 
|  | /* registry thread safety */ | 
|  |  | 
|  | static void | 
|  | _xidregistry_lock(struct _xidregistry *registry) | 
|  | { | 
|  | if (registry->global) { | 
|  | PyMutex_Lock(®istry->mutex); | 
|  | } | 
|  | // else: Within an interpreter we rely on the GIL instead of a separate lock. | 
|  | } | 
|  |  | 
|  | static void | 
|  | _xidregistry_unlock(struct _xidregistry *registry) | 
|  | { | 
|  | if (registry->global) { | 
|  | PyMutex_Unlock(®istry->mutex); | 
|  | } | 
|  | } | 
|  |  | 
|  |  | 
|  | /* accessing the registry */ | 
|  |  | 
|  | static inline struct _xidregistry * | 
|  | _get_global_xidregistry(_PyRuntimeState *runtime) | 
|  | { | 
|  | return &runtime->xi.registry; | 
|  | } | 
|  |  | 
|  | static inline struct _xidregistry * | 
|  | _get_xidregistry(PyInterpreterState *interp) | 
|  | { | 
|  | return &interp->xi.registry; | 
|  | } | 
|  |  | 
|  | static inline struct _xidregistry * | 
|  | _get_xidregistry_for_type(PyInterpreterState *interp, PyTypeObject *cls) | 
|  | { | 
|  | struct _xidregistry *registry = _get_global_xidregistry(interp->runtime); | 
|  | if (cls->tp_flags & Py_TPFLAGS_HEAPTYPE) { | 
|  | registry = _get_xidregistry(interp); | 
|  | } | 
|  | return registry; | 
|  | } | 
|  |  | 
|  | static struct _xidregitem * _xidregistry_remove_entry( | 
|  | struct _xidregistry *, struct _xidregitem *); | 
|  |  | 
|  | static struct _xidregitem * | 
|  | _xidregistry_find_type(struct _xidregistry *xidregistry, PyTypeObject *cls) | 
|  | { | 
|  | struct _xidregitem *cur = xidregistry->head; | 
|  | while (cur != NULL) { | 
|  | if (cur->weakref != NULL) { | 
|  | // cur is/was a heap type. | 
|  | PyObject *registered = _PyWeakref_GET_REF(cur->weakref); | 
|  | if (registered == NULL) { | 
|  | // The weakly ref'ed object was freed. | 
|  | cur = _xidregistry_remove_entry(xidregistry, cur); | 
|  | continue; | 
|  | } | 
|  | assert(PyType_Check(registered)); | 
|  | assert(cur->cls == (PyTypeObject *)registered); | 
|  | assert(cur->cls->tp_flags & Py_TPFLAGS_HEAPTYPE); | 
|  | Py_DECREF(registered); | 
|  | } | 
|  | if (cur->cls == cls) { | 
|  | return cur; | 
|  | } | 
|  | cur = cur->next; | 
|  | } | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | static crossinterpdatafunc | 
|  | _lookup_getdata_from_registry(PyInterpreterState *interp, PyObject *obj) | 
|  | { | 
|  | PyTypeObject *cls = Py_TYPE(obj); | 
|  |  | 
|  | struct _xidregistry *xidregistry = _get_xidregistry_for_type(interp, cls); | 
|  | _xidregistry_lock(xidregistry); | 
|  |  | 
|  | struct _xidregitem *matched = _xidregistry_find_type(xidregistry, cls); | 
|  | crossinterpdatafunc func = matched != NULL ? matched->getdata : NULL; | 
|  |  | 
|  | _xidregistry_unlock(xidregistry); | 
|  | return func; | 
|  | } | 
|  |  | 
|  |  | 
|  | /* updating the registry */ | 
|  |  | 
|  | static int | 
|  | _xidregistry_add_type(struct _xidregistry *xidregistry, | 
|  | PyTypeObject *cls, crossinterpdatafunc getdata) | 
|  | { | 
|  | struct _xidregitem *newhead = PyMem_RawMalloc(sizeof(struct _xidregitem)); | 
|  | if (newhead == NULL) { | 
|  | return -1; | 
|  | } | 
|  | *newhead = (struct _xidregitem){ | 
|  | // We do not keep a reference, to avoid keeping the class alive. | 
|  | .cls = cls, | 
|  | .refcount = 1, | 
|  | .getdata = getdata, | 
|  | }; | 
|  | if (cls->tp_flags & Py_TPFLAGS_HEAPTYPE) { | 
|  | // XXX Assign a callback to clear the entry from the registry? | 
|  | newhead->weakref = PyWeakref_NewRef((PyObject *)cls, NULL); | 
|  | if (newhead->weakref == NULL) { | 
|  | PyMem_RawFree(newhead); | 
|  | return -1; | 
|  | } | 
|  | } | 
|  | newhead->next = xidregistry->head; | 
|  | if (newhead->next != NULL) { | 
|  | newhead->next->prev = newhead; | 
|  | } | 
|  | xidregistry->head = newhead; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static struct _xidregitem * | 
|  | _xidregistry_remove_entry(struct _xidregistry *xidregistry, | 
|  | struct _xidregitem *entry) | 
|  | { | 
|  | struct _xidregitem *next = entry->next; | 
|  | if (entry->prev != NULL) { | 
|  | assert(entry->prev->next == entry); | 
|  | entry->prev->next = next; | 
|  | } | 
|  | else { | 
|  | assert(xidregistry->head == entry); | 
|  | xidregistry->head = next; | 
|  | } | 
|  | if (next != NULL) { | 
|  | next->prev = entry->prev; | 
|  | } | 
|  | Py_XDECREF(entry->weakref); | 
|  | PyMem_RawFree(entry); | 
|  | return next; | 
|  | } | 
|  |  | 
|  | static void | 
|  | _xidregistry_clear(struct _xidregistry *xidregistry) | 
|  | { | 
|  | struct _xidregitem *cur = xidregistry->head; | 
|  | xidregistry->head = NULL; | 
|  | while (cur != NULL) { | 
|  | struct _xidregitem *next = cur->next; | 
|  | Py_XDECREF(cur->weakref); | 
|  | PyMem_RawFree(cur); | 
|  | cur = next; | 
|  | } | 
|  | } | 
|  |  | 
|  | int | 
|  | _PyCrossInterpreterData_RegisterClass(PyTypeObject *cls, | 
|  | crossinterpdatafunc getdata) | 
|  | { | 
|  | if (!PyType_Check(cls)) { | 
|  | PyErr_Format(PyExc_ValueError, "only classes may be registered"); | 
|  | return -1; | 
|  | } | 
|  | if (getdata == NULL) { | 
|  | PyErr_Format(PyExc_ValueError, "missing 'getdata' func"); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | int res = 0; | 
|  | PyInterpreterState *interp = _PyInterpreterState_GET(); | 
|  | struct _xidregistry *xidregistry = _get_xidregistry_for_type(interp, cls); | 
|  | _xidregistry_lock(xidregistry); | 
|  |  | 
|  | struct _xidregitem *matched = _xidregistry_find_type(xidregistry, cls); | 
|  | if (matched != NULL) { | 
|  | assert(matched->getdata == getdata); | 
|  | matched->refcount += 1; | 
|  | goto finally; | 
|  | } | 
|  |  | 
|  | res = _xidregistry_add_type(xidregistry, cls, getdata); | 
|  |  | 
|  | finally: | 
|  | _xidregistry_unlock(xidregistry); | 
|  | return res; | 
|  | } | 
|  |  | 
|  | int | 
|  | _PyCrossInterpreterData_UnregisterClass(PyTypeObject *cls) | 
|  | { | 
|  | int res = 0; | 
|  | PyInterpreterState *interp = _PyInterpreterState_GET(); | 
|  | struct _xidregistry *xidregistry = _get_xidregistry_for_type(interp, cls); | 
|  | _xidregistry_lock(xidregistry); | 
|  |  | 
|  | struct _xidregitem *matched = _xidregistry_find_type(xidregistry, cls); | 
|  | if (matched != NULL) { | 
|  | assert(matched->refcount > 0); | 
|  | matched->refcount -= 1; | 
|  | if (matched->refcount == 0) { | 
|  | (void)_xidregistry_remove_entry(xidregistry, matched); | 
|  | } | 
|  | res = 1; | 
|  | } | 
|  |  | 
|  | _xidregistry_unlock(xidregistry); | 
|  | return res; | 
|  | } | 
|  |  | 
|  |  | 
|  | /********************************************/ | 
|  | /* cross-interpreter data for builtin types */ | 
|  | /********************************************/ | 
|  |  | 
|  | // bytes | 
|  |  | 
|  | struct _shared_bytes_data { | 
|  | char *bytes; | 
|  | Py_ssize_t len; | 
|  | }; | 
|  |  | 
|  | static PyObject * | 
|  | _new_bytes_object(_PyCrossInterpreterData *data) | 
|  | { | 
|  | struct _shared_bytes_data *shared = (struct _shared_bytes_data *)(data->data); | 
|  | return PyBytes_FromStringAndSize(shared->bytes, shared->len); | 
|  | } | 
|  |  | 
|  | static int | 
|  | _bytes_shared(PyThreadState *tstate, PyObject *obj, | 
|  | _PyCrossInterpreterData *data) | 
|  | { | 
|  | if (_PyCrossInterpreterData_InitWithSize( | 
|  | data, tstate->interp, sizeof(struct _shared_bytes_data), obj, | 
|  | _new_bytes_object | 
|  | ) < 0) | 
|  | { | 
|  | return -1; | 
|  | } | 
|  | struct _shared_bytes_data *shared = (struct _shared_bytes_data *)data->data; | 
|  | if (PyBytes_AsStringAndSize(obj, &shared->bytes, &shared->len) < 0) { | 
|  | _PyCrossInterpreterData_Clear(tstate->interp, data); | 
|  | return -1; | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // str | 
|  |  | 
|  | struct _shared_str_data { | 
|  | int kind; | 
|  | const void *buffer; | 
|  | Py_ssize_t len; | 
|  | }; | 
|  |  | 
|  | static PyObject * | 
|  | _new_str_object(_PyCrossInterpreterData *data) | 
|  | { | 
|  | struct _shared_str_data *shared = (struct _shared_str_data *)(data->data); | 
|  | return PyUnicode_FromKindAndData(shared->kind, shared->buffer, shared->len); | 
|  | } | 
|  |  | 
|  | static int | 
|  | _str_shared(PyThreadState *tstate, PyObject *obj, | 
|  | _PyCrossInterpreterData *data) | 
|  | { | 
|  | if (_PyCrossInterpreterData_InitWithSize( | 
|  | data, tstate->interp, sizeof(struct _shared_str_data), obj, | 
|  | _new_str_object | 
|  | ) < 0) | 
|  | { | 
|  | return -1; | 
|  | } | 
|  | struct _shared_str_data *shared = (struct _shared_str_data *)data->data; | 
|  | shared->kind = PyUnicode_KIND(obj); | 
|  | shared->buffer = PyUnicode_DATA(obj); | 
|  | shared->len = PyUnicode_GET_LENGTH(obj); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // int | 
|  |  | 
|  | static PyObject * | 
|  | _new_long_object(_PyCrossInterpreterData *data) | 
|  | { | 
|  | return PyLong_FromSsize_t((Py_ssize_t)(data->data)); | 
|  | } | 
|  |  | 
|  | static int | 
|  | _long_shared(PyThreadState *tstate, PyObject *obj, | 
|  | _PyCrossInterpreterData *data) | 
|  | { | 
|  | /* Note that this means the size of shareable ints is bounded by | 
|  | * sys.maxsize.  Hence on 32-bit architectures that is half the | 
|  | * size of maximum shareable ints on 64-bit. | 
|  | */ | 
|  | Py_ssize_t value = PyLong_AsSsize_t(obj); | 
|  | if (value == -1 && PyErr_Occurred()) { | 
|  | if (PyErr_ExceptionMatches(PyExc_OverflowError)) { | 
|  | PyErr_SetString(PyExc_OverflowError, "try sending as bytes"); | 
|  | } | 
|  | return -1; | 
|  | } | 
|  | _PyCrossInterpreterData_Init(data, tstate->interp, (void *)value, NULL, | 
|  | _new_long_object); | 
|  | // data->obj and data->free remain NULL | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // float | 
|  |  | 
|  | static PyObject * | 
|  | _new_float_object(_PyCrossInterpreterData *data) | 
|  | { | 
|  | double * value_ptr = data->data; | 
|  | return PyFloat_FromDouble(*value_ptr); | 
|  | } | 
|  |  | 
|  | static int | 
|  | _float_shared(PyThreadState *tstate, PyObject *obj, | 
|  | _PyCrossInterpreterData *data) | 
|  | { | 
|  | if (_PyCrossInterpreterData_InitWithSize( | 
|  | data, tstate->interp, sizeof(double), NULL, | 
|  | _new_float_object | 
|  | ) < 0) | 
|  | { | 
|  | return -1; | 
|  | } | 
|  | double *shared = (double *)data->data; | 
|  | *shared = PyFloat_AsDouble(obj); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // None | 
|  |  | 
|  | static PyObject * | 
|  | _new_none_object(_PyCrossInterpreterData *data) | 
|  | { | 
|  | // XXX Singleton refcounts are problematic across interpreters... | 
|  | return Py_NewRef(Py_None); | 
|  | } | 
|  |  | 
|  | static int | 
|  | _none_shared(PyThreadState *tstate, PyObject *obj, | 
|  | _PyCrossInterpreterData *data) | 
|  | { | 
|  | _PyCrossInterpreterData_Init(data, tstate->interp, NULL, NULL, | 
|  | _new_none_object); | 
|  | // data->data, data->obj and data->free remain NULL | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // bool | 
|  |  | 
|  | static PyObject * | 
|  | _new_bool_object(_PyCrossInterpreterData *data) | 
|  | { | 
|  | if (data->data){ | 
|  | Py_RETURN_TRUE; | 
|  | } | 
|  | Py_RETURN_FALSE; | 
|  | } | 
|  |  | 
|  | static int | 
|  | _bool_shared(PyThreadState *tstate, PyObject *obj, | 
|  | _PyCrossInterpreterData *data) | 
|  | { | 
|  | _PyCrossInterpreterData_Init(data, tstate->interp, | 
|  | (void *) (Py_IsTrue(obj) ? (uintptr_t) 1 : (uintptr_t) 0), NULL, | 
|  | _new_bool_object); | 
|  | // data->obj and data->free remain NULL | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // tuple | 
|  |  | 
|  | struct _shared_tuple_data { | 
|  | Py_ssize_t len; | 
|  | _PyCrossInterpreterData **data; | 
|  | }; | 
|  |  | 
|  | static PyObject * | 
|  | _new_tuple_object(_PyCrossInterpreterData *data) | 
|  | { | 
|  | struct _shared_tuple_data *shared = (struct _shared_tuple_data *)(data->data); | 
|  | PyObject *tuple = PyTuple_New(shared->len); | 
|  | if (tuple == NULL) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | for (Py_ssize_t i = 0; i < shared->len; i++) { | 
|  | PyObject *item = _PyCrossInterpreterData_NewObject(shared->data[i]); | 
|  | if (item == NULL){ | 
|  | Py_DECREF(tuple); | 
|  | return NULL; | 
|  | } | 
|  | PyTuple_SET_ITEM(tuple, i, item); | 
|  | } | 
|  | return tuple; | 
|  | } | 
|  |  | 
|  | static void | 
|  | _tuple_shared_free(void* data) | 
|  | { | 
|  | struct _shared_tuple_data *shared = (struct _shared_tuple_data *)(data); | 
|  | #ifndef NDEBUG | 
|  | int64_t interpid = PyInterpreterState_GetID(_PyInterpreterState_GET()); | 
|  | #endif | 
|  | for (Py_ssize_t i = 0; i < shared->len; i++) { | 
|  | if (shared->data[i] != NULL) { | 
|  | assert(_PyCrossInterpreterData_INTERPID(shared->data[i]) == interpid); | 
|  | _PyCrossInterpreterData_Release(shared->data[i]); | 
|  | PyMem_RawFree(shared->data[i]); | 
|  | shared->data[i] = NULL; | 
|  | } | 
|  | } | 
|  | PyMem_Free(shared->data); | 
|  | PyMem_RawFree(shared); | 
|  | } | 
|  |  | 
|  | static int | 
|  | _tuple_shared(PyThreadState *tstate, PyObject *obj, | 
|  | _PyCrossInterpreterData *data) | 
|  | { | 
|  | Py_ssize_t len = PyTuple_GET_SIZE(obj); | 
|  | if (len < 0) { | 
|  | return -1; | 
|  | } | 
|  | struct _shared_tuple_data *shared = PyMem_RawMalloc(sizeof(struct _shared_tuple_data)); | 
|  | if (shared == NULL){ | 
|  | PyErr_NoMemory(); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | shared->len = len; | 
|  | shared->data = (_PyCrossInterpreterData **) PyMem_Calloc(shared->len, sizeof(_PyCrossInterpreterData *)); | 
|  | if (shared->data == NULL) { | 
|  | PyErr_NoMemory(); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | for (Py_ssize_t i = 0; i < shared->len; i++) { | 
|  | _PyCrossInterpreterData *data = _PyCrossInterpreterData_New(); | 
|  | if (data == NULL) { | 
|  | goto error;  // PyErr_NoMemory already set | 
|  | } | 
|  | PyObject *item = PyTuple_GET_ITEM(obj, i); | 
|  |  | 
|  | int res = -1; | 
|  | if (!_Py_EnterRecursiveCallTstate(tstate, " while sharing a tuple")) { | 
|  | res = _PyObject_GetCrossInterpreterData(item, data); | 
|  | _Py_LeaveRecursiveCallTstate(tstate); | 
|  | } | 
|  | if (res < 0) { | 
|  | PyMem_RawFree(data); | 
|  | goto error; | 
|  | } | 
|  | shared->data[i] = data; | 
|  | } | 
|  | _PyCrossInterpreterData_Init( | 
|  | data, tstate->interp, shared, obj, _new_tuple_object); | 
|  | data->free = _tuple_shared_free; | 
|  | return 0; | 
|  |  | 
|  | error: | 
|  | _tuple_shared_free(shared); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | // registration | 
|  |  | 
|  | static void | 
|  | _register_builtins_for_crossinterpreter_data(struct _xidregistry *xidregistry) | 
|  | { | 
|  | // None | 
|  | if (_xidregistry_add_type(xidregistry, (PyTypeObject *)PyObject_Type(Py_None), _none_shared) != 0) { | 
|  | Py_FatalError("could not register None for cross-interpreter sharing"); | 
|  | } | 
|  |  | 
|  | // int | 
|  | if (_xidregistry_add_type(xidregistry, &PyLong_Type, _long_shared) != 0) { | 
|  | Py_FatalError("could not register int for cross-interpreter sharing"); | 
|  | } | 
|  |  | 
|  | // bytes | 
|  | if (_xidregistry_add_type(xidregistry, &PyBytes_Type, _bytes_shared) != 0) { | 
|  | Py_FatalError("could not register bytes for cross-interpreter sharing"); | 
|  | } | 
|  |  | 
|  | // str | 
|  | if (_xidregistry_add_type(xidregistry, &PyUnicode_Type, _str_shared) != 0) { | 
|  | Py_FatalError("could not register str for cross-interpreter sharing"); | 
|  | } | 
|  |  | 
|  | // bool | 
|  | if (_xidregistry_add_type(xidregistry, &PyBool_Type, _bool_shared) != 0) { | 
|  | Py_FatalError("could not register bool for cross-interpreter sharing"); | 
|  | } | 
|  |  | 
|  | // float | 
|  | if (_xidregistry_add_type(xidregistry, &PyFloat_Type, _float_shared) != 0) { | 
|  | Py_FatalError("could not register float for cross-interpreter sharing"); | 
|  | } | 
|  |  | 
|  | // tuple | 
|  | if (_xidregistry_add_type(xidregistry, &PyTuple_Type, _tuple_shared) != 0) { | 
|  | Py_FatalError("could not register tuple for cross-interpreter sharing"); | 
|  | } | 
|  | } |