| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | // types.GenericAlias -- used to represent e.g. list[int].
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Python.h"
 | 
					
						
							| 
									
										
										
										
											2023-08-24 20:25:22 +02:00
										 |  |  | #include "pycore_ceval.h"         // _PyEval_GetBuiltin()
 | 
					
						
							| 
									
										
										
										
											2023-10-17 14:30:31 +02:00
										 |  |  | #include "pycore_modsupport.h"    // _PyArg_NoKeywords()
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | #include "pycore_object.h"
 | 
					
						
							| 
									
										
										
										
											2021-07-17 22:44:10 +03:00
										 |  |  | #include "pycore_unionobject.h"   // _Py_union_type_or, _PyGenericAlias_Check
 | 
					
						
							| 
									
										
										
										
											2023-07-25 15:28:30 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-12 12:20:12 +00:00
										 |  |  | #include <stdbool.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | typedef struct { | 
					
						
							|  |  |  |     PyObject_HEAD | 
					
						
							|  |  |  |     PyObject *origin; | 
					
						
							|  |  |  |     PyObject *args; | 
					
						
							|  |  |  |     PyObject *parameters; | 
					
						
							| 
									
										
										
										
											2022-03-22 04:33:02 +08:00
										 |  |  |     PyObject *weakreflist; | 
					
						
							| 
									
										
										
										
											2022-03-12 12:20:12 +00:00
										 |  |  |     // Whether we're a starred type, e.g. *tuple[int].
 | 
					
						
							|  |  |  |     bool starred; | 
					
						
							| 
									
										
										
										
											2022-03-22 04:33:02 +08:00
										 |  |  |     vectorcallfunc vectorcall; | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | } gaobject; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-12 12:20:12 +00:00
										 |  |  | typedef struct { | 
					
						
							|  |  |  |     PyObject_HEAD | 
					
						
							|  |  |  |     PyObject *obj;  /* Set to NULL when iterator is exhausted */ | 
					
						
							|  |  |  | } gaiterobject; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | static void | 
					
						
							|  |  |  | ga_dealloc(PyObject *self) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)self; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _PyObject_GC_UNTRACK(self); | 
					
						
							| 
									
										
										
										
											2020-11-16 11:27:23 +08:00
										 |  |  |     if (alias->weakreflist != NULL) { | 
					
						
							|  |  |  |         PyObject_ClearWeakRefs((PyObject *)alias); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     Py_XDECREF(alias->origin); | 
					
						
							|  |  |  |     Py_XDECREF(alias->args); | 
					
						
							|  |  |  |     Py_XDECREF(alias->parameters); | 
					
						
							| 
									
										
										
										
											2020-07-10 12:40:38 +02:00
										 |  |  |     Py_TYPE(self)->tp_free(self); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | ga_traverse(PyObject *self, visitproc visit, void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)self; | 
					
						
							|  |  |  |     Py_VISIT(alias->origin); | 
					
						
							|  |  |  |     Py_VISIT(alias->args); | 
					
						
							|  |  |  |     Py_VISIT(alias->parameters); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  | ga_repr_item(PyUnicodeWriter *writer, PyObject *p) | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | { | 
					
						
							|  |  |  |     PyObject *qualname = NULL; | 
					
						
							|  |  |  |     PyObject *module = NULL; | 
					
						
							| 
									
										
										
										
											2023-09-17 14:23:31 +03:00
										 |  |  |     int rc; | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (p == Py_Ellipsis) { | 
					
						
							|  |  |  |         // The Ellipsis object
 | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |         rc = PyUnicodeWriter_WriteUTF8(writer, "...", 3); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         goto done; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-17 14:23:31 +03:00
										 |  |  |     if ((rc = PyObject_HasAttrWithError(p, &_Py_ID(__origin__))) > 0 && | 
					
						
							|  |  |  |         (rc = PyObject_HasAttrWithError(p, &_Py_ID(__args__))) > 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // It looks like a GenericAlias
 | 
					
						
							|  |  |  |         goto use_repr; | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-17 14:23:31 +03:00
										 |  |  |     if (rc < 0) { | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |     if (PyObject_GetOptionalAttr(p, &_Py_ID(__qualname__), &qualname) < 0) { | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (qualname == NULL) { | 
					
						
							|  |  |  |         goto use_repr; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |     if (PyObject_GetOptionalAttr(p, &_Py_ID(__module__), &module) < 0) { | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (module == NULL || module == Py_None) { | 
					
						
							|  |  |  |         goto use_repr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Looks like a class
 | 
					
						
							|  |  |  |     if (PyUnicode_Check(module) && | 
					
						
							|  |  |  |         _PyUnicode_EqualToASCIIString(module, "builtins")) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // builtins don't need a module name
 | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |         rc = PyUnicodeWriter_WriteStr(writer, qualname); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         goto done; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |         rc = PyUnicodeWriter_Format(writer, "%S.%S", module, qualname); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         goto done; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  | error: | 
					
						
							|  |  |  |     rc = -1; | 
					
						
							|  |  |  |     goto done; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | use_repr: | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |     rc = PyUnicodeWriter_WriteRepr(writer, p); | 
					
						
							|  |  |  |     goto done; | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | done: | 
					
						
							|  |  |  |     Py_XDECREF(qualname); | 
					
						
							|  |  |  |     Py_XDECREF(module); | 
					
						
							| 
									
										
										
										
											2023-09-17 14:23:31 +03:00
										 |  |  |     return rc; | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-01 03:04:50 +03:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  | ga_repr_items_list(PyUnicodeWriter *writer, PyObject *p) | 
					
						
							| 
									
										
										
										
											2023-07-01 03:04:50 +03:00
										 |  |  | { | 
					
						
							|  |  |  |     assert(PyList_CheckExact(p)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Py_ssize_t len = PyList_GET_SIZE(p); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |     if (PyUnicodeWriter_WriteChar(writer, '[') < 0) { | 
					
						
							| 
									
										
										
										
											2023-07-01 03:04:50 +03:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (Py_ssize_t i = 0; i < len; i++) { | 
					
						
							|  |  |  |         if (i > 0) { | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |             if (PyUnicodeWriter_WriteUTF8(writer, ", ", 2) < 0) { | 
					
						
							| 
									
										
										
										
											2023-07-01 03:04:50 +03:00
										 |  |  |                 return -1; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         PyObject *item = PyList_GET_ITEM(p, i); | 
					
						
							|  |  |  |         if (ga_repr_item(writer, item) < 0) { | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |     if (PyUnicodeWriter_WriteChar(writer, ']') < 0) { | 
					
						
							| 
									
										
										
										
											2023-07-01 03:04:50 +03:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | ga_repr(PyObject *self) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)self; | 
					
						
							|  |  |  |     Py_ssize_t len = PyTuple_GET_SIZE(alias->args); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |     // Estimation based on the shortest format: "int[int, int, int]"
 | 
					
						
							|  |  |  |     Py_ssize_t estimate = (len <= PY_SSIZE_T_MAX / 5) ? len * 5 : len; | 
					
						
							|  |  |  |     estimate = 3 + 1 + estimate + 1; | 
					
						
							|  |  |  |     PyUnicodeWriter *writer = PyUnicodeWriter_Create(estimate); | 
					
						
							|  |  |  |     if (writer == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-15 02:35:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-12 12:20:12 +00:00
										 |  |  |     if (alias->starred) { | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |         if (PyUnicodeWriter_WriteChar(writer, '*') < 0) { | 
					
						
							| 
									
										
										
										
											2022-03-12 12:20:12 +00:00
										 |  |  |             goto error; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |     if (ga_repr_item(writer, alias->origin) < 0) { | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |     if (PyUnicodeWriter_WriteChar(writer, '[') < 0) { | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (Py_ssize_t i = 0; i < len; i++) { | 
					
						
							|  |  |  |         if (i > 0) { | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |             if (PyUnicodeWriter_WriteUTF8(writer, ", ", 2) < 0) { | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |                 goto error; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         PyObject *p = PyTuple_GET_ITEM(alias->args, i); | 
					
						
							| 
									
										
										
										
											2023-07-01 03:04:50 +03:00
										 |  |  |         if (PyList_CheckExact(p)) { | 
					
						
							|  |  |  |             // Looks like we are working with ParamSpec's list of type args:
 | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |             if (ga_repr_items_list(writer, p) < 0) { | 
					
						
							| 
									
										
										
										
											2023-07-01 03:04:50 +03:00
										 |  |  |                 goto error; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |         else if (ga_repr_item(writer, p) < 0) { | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |             goto error; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (len == 0) { | 
					
						
							|  |  |  |         // for something like tuple[()] we should print a "()"
 | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |         if (PyUnicodeWriter_WriteUTF8(writer, "()", 2) < 0) { | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |             goto error; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |     if (PyUnicodeWriter_WriteChar(writer, ']') < 0) { | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |     return PyUnicodeWriter_Finish(writer); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | error: | 
					
						
							| 
									
										
										
										
											2024-06-20 20:35:35 +02:00
										 |  |  |     PyUnicodeWriter_Discard(writer); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Index of item in self[:len], or -1 if not found (self is a tuple)
 | 
					
						
							|  |  |  | static Py_ssize_t | 
					
						
							|  |  |  | tuple_index(PyObject *self, Py_ssize_t len, PyObject *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     for (Py_ssize_t i = 0; i < len; i++) { | 
					
						
							|  |  |  |         if (PyTuple_GET_ITEM(self, i) == item) { | 
					
						
							|  |  |  |             return i; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  | static int | 
					
						
							|  |  |  | tuple_add(PyObject *self, Py_ssize_t len, PyObject *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (tuple_index(self, len, item) < 0) { | 
					
						
							| 
									
										
										
										
											2022-11-10 22:22:02 +01:00
										 |  |  |         PyTuple_SET_ITEM(self, len, Py_NewRef(item)); | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  | static Py_ssize_t | 
					
						
							|  |  |  | tuple_extend(PyObject **dst, Py_ssize_t dstindex, | 
					
						
							|  |  |  |              PyObject **src, Py_ssize_t count) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(count >= 0); | 
					
						
							|  |  |  |     if (_PyTuple_Resize(dst, PyTuple_GET_SIZE(*dst) + count - 1) != 0) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     assert(dstindex + count <= PyTuple_GET_SIZE(*dst)); | 
					
						
							|  |  |  |     for (Py_ssize_t i = 0; i < count; ++i) { | 
					
						
							|  |  |  |         PyObject *item = src[i]; | 
					
						
							| 
									
										
										
										
											2022-11-10 22:22:02 +01:00
										 |  |  |         PyTuple_SET_ITEM(*dst, dstindex + i, Py_NewRef(item)); | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  |     } | 
					
						
							|  |  |  |     return dstindex + count; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-06 21:04:33 +03:00
										 |  |  | PyObject * | 
					
						
							|  |  |  | _Py_make_parameters(PyObject *args) | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |     Py_ssize_t nargs = PyTuple_GET_SIZE(args); | 
					
						
							|  |  |  |     Py_ssize_t len = nargs; | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     PyObject *parameters = PyTuple_New(len); | 
					
						
							|  |  |  |     if (parameters == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     Py_ssize_t iparam = 0; | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |     for (Py_ssize_t iarg = 0; iarg < nargs; iarg++) { | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         PyObject *t = PyTuple_GET_ITEM(args, iarg); | 
					
						
							| 
									
										
										
										
											2022-07-09 12:18:01 +08:00
										 |  |  |         // We don't want __parameters__ descriptor of a bare Python class.
 | 
					
						
							|  |  |  |         if (PyType_Check(t)) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-17 14:23:31 +03:00
										 |  |  |         int rc = PyObject_HasAttrWithError(t, &_Py_ID(__typing_subst__)); | 
					
						
							|  |  |  |         if (rc < 0) { | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |             Py_DECREF(parameters); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-17 14:23:31 +03:00
										 |  |  |         if (rc) { | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |             iparam += tuple_add(parameters, iparam, t); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             PyObject *subparams; | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |             if (PyObject_GetOptionalAttr(t, &_Py_ID(__parameters__), | 
					
						
							| 
									
										
										
										
											2022-02-08 13:39:07 -07:00
										 |  |  |                                      &subparams) < 0) { | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |                 Py_DECREF(parameters); | 
					
						
							|  |  |  |                 return NULL; | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |             if (subparams && PyTuple_Check(subparams)) { | 
					
						
							|  |  |  |                 Py_ssize_t len2 = PyTuple_GET_SIZE(subparams); | 
					
						
							|  |  |  |                 Py_ssize_t needed = len2 - 1 - (iarg - iparam); | 
					
						
							|  |  |  |                 if (needed > 0) { | 
					
						
							|  |  |  |                     len += needed; | 
					
						
							|  |  |  |                     if (_PyTuple_Resize(¶meters, len) < 0) { | 
					
						
							|  |  |  |                         Py_DECREF(subparams); | 
					
						
							|  |  |  |                         Py_DECREF(parameters); | 
					
						
							|  |  |  |                         return NULL; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 for (Py_ssize_t j = 0; j < len2; j++) { | 
					
						
							|  |  |  |                     PyObject *t2 = PyTuple_GET_ITEM(subparams, j); | 
					
						
							|  |  |  |                     iparam += tuple_add(parameters, iparam, t2); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             Py_XDECREF(subparams); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (iparam < len) { | 
					
						
							|  |  |  |         if (_PyTuple_Resize(¶meters, iparam) < 0) { | 
					
						
							|  |  |  |             Py_XDECREF(parameters); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return parameters; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  | /* If obj is a generic alias, substitute type variables params
 | 
					
						
							|  |  |  |    with substitutions argitems.  For example, if obj is list[T], | 
					
						
							|  |  |  |    params is (T, S), and argitems is (str, int), return list[str]. | 
					
						
							|  |  |  |    If obj doesn't have a __parameters__ attribute or that's not | 
					
						
							|  |  |  |    a non-empty tuple, return a new reference to obj. */ | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  | subs_tvars(PyObject *obj, PyObject *params, | 
					
						
							| 
									
										
										
										
											2022-06-12 16:22:01 +03:00
										 |  |  |            PyObject **argitems, Py_ssize_t nargs) | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  | { | 
					
						
							|  |  |  |     PyObject *subparams; | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |     if (PyObject_GetOptionalAttr(obj, &_Py_ID(__parameters__), &subparams) < 0) { | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (subparams && PyTuple_Check(subparams) && PyTuple_GET_SIZE(subparams)) { | 
					
						
							|  |  |  |         Py_ssize_t nparams = PyTuple_GET_SIZE(params); | 
					
						
							|  |  |  |         Py_ssize_t nsubargs = PyTuple_GET_SIZE(subparams); | 
					
						
							|  |  |  |         PyObject *subargs = PyTuple_New(nsubargs); | 
					
						
							|  |  |  |         if (subargs == NULL) { | 
					
						
							|  |  |  |             Py_DECREF(subparams); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-06-12 16:22:01 +03:00
										 |  |  |         Py_ssize_t j = 0; | 
					
						
							|  |  |  |         for (Py_ssize_t i = 0; i < nsubargs; ++i) { | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |             PyObject *arg = PyTuple_GET_ITEM(subparams, i); | 
					
						
							|  |  |  |             Py_ssize_t iparam = tuple_index(params, nparams, arg); | 
					
						
							| 
									
										
										
										
											2022-06-12 16:22:01 +03:00
										 |  |  |             if (iparam >= 0) { | 
					
						
							|  |  |  |                 PyObject *param = PyTuple_GET_ITEM(params, iparam); | 
					
						
							|  |  |  |                 arg = argitems[iparam]; | 
					
						
							|  |  |  |                 if (Py_TYPE(param)->tp_iter && PyTuple_Check(arg)) {  // TypeVarTuple
 | 
					
						
							|  |  |  |                     j = tuple_extend(&subargs, j, | 
					
						
							|  |  |  |                                     &PyTuple_GET_ITEM(arg, 0), | 
					
						
							|  |  |  |                                     PyTuple_GET_SIZE(arg)); | 
					
						
							|  |  |  |                     if (j < 0) { | 
					
						
							|  |  |  |                         return NULL; | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  |                     } | 
					
						
							| 
									
										
										
										
											2022-06-12 16:22:01 +03:00
										 |  |  |                     continue; | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-11-10 22:22:02 +01:00
										 |  |  |             PyTuple_SET_ITEM(subargs, j, Py_NewRef(arg)); | 
					
						
							| 
									
										
										
										
											2022-06-12 16:22:01 +03:00
										 |  |  |             j++; | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-06-12 16:22:01 +03:00
										 |  |  |         assert(j == PyTuple_GET_SIZE(subargs)); | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         obj = PyObject_GetItem(obj, subargs); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Py_DECREF(subargs); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         Py_INCREF(obj); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_XDECREF(subparams); | 
					
						
							|  |  |  |     return obj; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  | static int | 
					
						
							|  |  |  | _is_unpacked_typevartuple(PyObject *arg) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |     PyObject *tmp; | 
					
						
							|  |  |  |     if (PyType_Check(arg)) { // TODO: Add test
 | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |     int res = PyObject_GetOptionalAttr(arg, &_Py_ID(__typing_is_unpacked_typevartuple__), &tmp); | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  |     if (res > 0) { | 
					
						
							|  |  |  |         res = PyObject_IsTrue(tmp); | 
					
						
							|  |  |  |         Py_DECREF(tmp); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | _unpacked_tuple_args(PyObject *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyObject *result; | 
					
						
							|  |  |  |     assert(!PyType_Check(arg)); | 
					
						
							|  |  |  |     // Fast path
 | 
					
						
							|  |  |  |     if (_PyGenericAlias_Check(arg) && | 
					
						
							|  |  |  |             ((gaobject *)arg)->starred && | 
					
						
							|  |  |  |             ((gaobject *)arg)->origin == (PyObject *)&PyTuple_Type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         result = ((gaobject *)arg)->args; | 
					
						
							| 
									
										
										
										
											2022-11-10 22:22:02 +01:00
										 |  |  |         return Py_NewRef(result); | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |     if (PyObject_GetOptionalAttr(arg, &_Py_ID(__typing_unpacked_tuple_args__), &result) > 0) { | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |         if (result == Py_None) { | 
					
						
							|  |  |  |             Py_DECREF(result); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _unpack_args(PyObject *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyObject *newargs = PyList_New(0); | 
					
						
							|  |  |  |     if (newargs == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     int is_tuple = PyTuple_Check(item); | 
					
						
							|  |  |  |     Py_ssize_t nitems = is_tuple ? PyTuple_GET_SIZE(item) : 1; | 
					
						
							|  |  |  |     PyObject **argitems = is_tuple ? &PyTuple_GET_ITEM(item, 0) : &item; | 
					
						
							|  |  |  |     for (Py_ssize_t i = 0; i < nitems; i++) { | 
					
						
							|  |  |  |         item = argitems[i]; | 
					
						
							|  |  |  |         if (!PyType_Check(item)) { | 
					
						
							|  |  |  |             PyObject *subargs = _unpacked_tuple_args(item); | 
					
						
							|  |  |  |             if (subargs != NULL && | 
					
						
							|  |  |  |                 PyTuple_Check(subargs) && | 
					
						
							|  |  |  |                 !(PyTuple_GET_SIZE(subargs) && | 
					
						
							|  |  |  |                   PyTuple_GET_ITEM(subargs, PyTuple_GET_SIZE(subargs)-1) == Py_Ellipsis)) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 if (PyList_SetSlice(newargs, PY_SSIZE_T_MAX, PY_SSIZE_T_MAX, subargs) < 0) { | 
					
						
							|  |  |  |                     Py_DECREF(subargs); | 
					
						
							|  |  |  |                     Py_DECREF(newargs); | 
					
						
							|  |  |  |                     return NULL; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 Py_DECREF(subargs); | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             Py_XDECREF(subargs); | 
					
						
							|  |  |  |             if (PyErr_Occurred()) { | 
					
						
							|  |  |  |                 Py_DECREF(newargs); | 
					
						
							|  |  |  |                 return NULL; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (PyList_Append(newargs, item) < 0) { | 
					
						
							|  |  |  |             Py_DECREF(newargs); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_SETREF(newargs, PySequence_Tuple(newargs)); | 
					
						
							|  |  |  |     return newargs; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-06 21:04:33 +03:00
										 |  |  | PyObject * | 
					
						
							|  |  |  | _Py_subs_parameters(PyObject *self, PyObject *args, PyObject *parameters, PyObject *item) | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-07-06 21:04:33 +03:00
										 |  |  |     Py_ssize_t nparams = PyTuple_GET_SIZE(parameters); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     if (nparams == 0) { | 
					
						
							|  |  |  |         return PyErr_Format(PyExc_TypeError, | 
					
						
							| 
									
										
										
										
											2022-03-11 10:47:26 +02:00
										 |  |  |                             "%R is not a generic class", | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |                             self); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |     item = _unpack_args(item); | 
					
						
							|  |  |  |     for (Py_ssize_t i = 0; i < nparams; i++) { | 
					
						
							|  |  |  |         PyObject *param = PyTuple_GET_ITEM(parameters, i); | 
					
						
							| 
									
										
										
										
											2022-06-12 16:22:01 +03:00
										 |  |  |         PyObject *prepare, *tmp; | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |         if (PyObject_GetOptionalAttr(param, &_Py_ID(__typing_prepare_subst__), &prepare) < 0) { | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |             Py_DECREF(item); | 
					
						
							| 
									
										
										
										
											2022-06-12 16:22:01 +03:00
										 |  |  |             return NULL; | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-06-12 16:22:01 +03:00
										 |  |  |         if (prepare && prepare != Py_None) { | 
					
						
							|  |  |  |             if (PyTuple_Check(item)) { | 
					
						
							|  |  |  |                 tmp = PyObject_CallFunction(prepare, "OO", self, item); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |                 tmp = PyObject_CallFunction(prepare, "O(O)", self, item); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             Py_DECREF(prepare); | 
					
						
							|  |  |  |             Py_SETREF(item, tmp); | 
					
						
							|  |  |  |             if (item == NULL) { | 
					
						
							|  |  |  |                 return NULL; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-06-12 16:22:01 +03:00
										 |  |  |     int is_tuple = PyTuple_Check(item); | 
					
						
							|  |  |  |     Py_ssize_t nitems = is_tuple ? PyTuple_GET_SIZE(item) : 1; | 
					
						
							|  |  |  |     PyObject **argitems = is_tuple ? &PyTuple_GET_ITEM(item, 0) : &item; | 
					
						
							|  |  |  |     if (nitems != nparams) { | 
					
						
							|  |  |  |         Py_DECREF(item); | 
					
						
							|  |  |  |         return PyErr_Format(PyExc_TypeError, | 
					
						
							|  |  |  |                             "Too %s arguments for %R; actual %zd, expected %zd", | 
					
						
							|  |  |  |                             nitems > nparams ? "many" : "few", | 
					
						
							|  |  |  |                             self, nitems, nparams); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-07-06 21:04:33 +03:00
										 |  |  |     /* Replace all type variables (specified by parameters)
 | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |        with corresponding values specified by argitems. | 
					
						
							|  |  |  |         t = list[T];          t[int]      -> newargs = [int] | 
					
						
							|  |  |  |         t = dict[str, T];     t[int]      -> newargs = [str, int] | 
					
						
							|  |  |  |         t = dict[T, list[S]]; t[str, int] -> newargs = [str, list[int]] | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2021-07-06 21:04:33 +03:00
										 |  |  |     Py_ssize_t nargs = PyTuple_GET_SIZE(args); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     PyObject *newargs = PyTuple_New(nargs); | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |     if (newargs == NULL) { | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |         Py_DECREF(item); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  |     for (Py_ssize_t iarg = 0, jarg = 0; iarg < nargs; iarg++) { | 
					
						
							| 
									
										
										
										
											2021-07-06 21:04:33 +03:00
										 |  |  |         PyObject *arg = PyTuple_GET_ITEM(args, iarg); | 
					
						
							| 
									
										
										
										
											2022-11-01 09:33:09 +02:00
										 |  |  |         if (PyType_Check(arg)) { | 
					
						
							| 
									
										
										
										
											2022-11-10 22:22:02 +01:00
										 |  |  |             PyTuple_SET_ITEM(newargs, jarg, Py_NewRef(arg)); | 
					
						
							| 
									
										
										
										
											2022-11-01 09:33:09 +02:00
										 |  |  |             jarg++; | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  |         int unpack = _is_unpacked_typevartuple(arg); | 
					
						
							|  |  |  |         if (unpack < 0) { | 
					
						
							|  |  |  |             Py_DECREF(newargs); | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |             Py_DECREF(item); | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-03-11 10:47:26 +02:00
										 |  |  |         PyObject *subst; | 
					
						
							| 
									
										
										
										
											2023-07-12 08:57:10 +03:00
										 |  |  |         if (PyObject_GetOptionalAttr(arg, &_Py_ID(__typing_subst__), &subst) < 0) { | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |             Py_DECREF(newargs); | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |             Py_DECREF(item); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-03-11 10:47:26 +02:00
										 |  |  |         if (subst) { | 
					
						
							| 
									
										
										
										
											2021-07-06 21:04:33 +03:00
										 |  |  |             Py_ssize_t iparam = tuple_index(parameters, nparams, arg); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |             assert(iparam >= 0); | 
					
						
							| 
									
										
										
										
											2022-03-11 10:47:26 +02:00
										 |  |  |             arg = PyObject_CallOneArg(subst, argitems[iparam]); | 
					
						
							|  |  |  |             Py_DECREF(subst); | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2022-06-12 16:22:01 +03:00
										 |  |  |             arg = subs_tvars(arg, parameters, argitems, nitems); | 
					
						
							| 
									
										
										
										
											2022-03-11 10:47:26 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (arg == NULL) { | 
					
						
							|  |  |  |             Py_DECREF(newargs); | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |             Py_DECREF(item); | 
					
						
							| 
									
										
										
										
											2022-03-11 10:47:26 +02:00
										 |  |  |             return NULL; | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  |         if (unpack) { | 
					
						
							|  |  |  |             jarg = tuple_extend(&newargs, jarg, | 
					
						
							|  |  |  |                     &PyTuple_GET_ITEM(arg, 0), PyTuple_GET_SIZE(arg)); | 
					
						
							|  |  |  |             Py_DECREF(arg); | 
					
						
							|  |  |  |             if (jarg < 0) { | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |                 Py_DECREF(item); | 
					
						
							| 
									
										
										
										
											2022-04-30 08:22:46 +03:00
										 |  |  |                 return NULL; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             PyTuple_SET_ITEM(newargs, jarg, arg); | 
					
						
							|  |  |  |             jarg++; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |     Py_DECREF(item); | 
					
						
							| 
									
										
										
										
											2021-07-06 21:04:33 +03:00
										 |  |  |     return newargs; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-18 06:39:32 +03:00
										 |  |  | PyDoc_STRVAR(genericalias__doc__, | 
					
						
							| 
									
										
										
										
											2024-04-12 13:56:41 +03:00
										 |  |  | "GenericAlias(origin, args, /)\n" | 
					
						
							|  |  |  | "--\n\n" | 
					
						
							| 
									
										
										
										
											2022-04-18 06:39:32 +03:00
										 |  |  | "Represent a PEP 585 generic type\n" | 
					
						
							|  |  |  | "\n" | 
					
						
							|  |  |  | "E.g. for t = list[int], t.__origin__ is list and t.__args__ is (int,)."); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-06 21:04:33 +03:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | ga_getitem(PyObject *self, PyObject *item) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)self; | 
					
						
							|  |  |  |     // Populate __parameters__ if needed.
 | 
					
						
							|  |  |  |     if (alias->parameters == NULL) { | 
					
						
							|  |  |  |         alias->parameters = _Py_make_parameters(alias->args); | 
					
						
							|  |  |  |         if (alias->parameters == NULL) { | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PyObject *newargs = _Py_subs_parameters(self, alias->args, alias->parameters, item); | 
					
						
							|  |  |  |     if (newargs == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     PyObject *res = Py_GenericAlias(alias->origin, newargs); | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |     ((gaobject *)res)->starred = alias->starred; | 
					
						
							| 
									
										
										
										
											2020-05-04 10:56:05 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     Py_DECREF(newargs); | 
					
						
							|  |  |  |     return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyMappingMethods ga_as_mapping = { | 
					
						
							|  |  |  |     .mp_subscript = ga_getitem, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Py_hash_t | 
					
						
							|  |  |  | ga_hash(PyObject *self) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)self; | 
					
						
							|  |  |  |     // TODO: Hash in the hash for the origin
 | 
					
						
							|  |  |  |     Py_hash_t h0 = PyObject_Hash(alias->origin); | 
					
						
							|  |  |  |     if (h0 == -1) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_hash_t h1 = PyObject_Hash(alias->args); | 
					
						
							|  |  |  |     if (h1 == -1) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return h0 ^ h1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-22 04:33:02 +08:00
										 |  |  | static inline PyObject * | 
					
						
							|  |  |  | set_orig_class(PyObject *obj, PyObject *self) | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | { | 
					
						
							|  |  |  |     if (obj != NULL) { | 
					
						
							| 
									
										
										
										
											2022-03-22 04:33:02 +08:00
										 |  |  |         if (PyObject_SetAttr(obj, &_Py_ID(__orig_class__), self) < 0) { | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |             if (!PyErr_ExceptionMatches(PyExc_AttributeError) && | 
					
						
							|  |  |  |                 !PyErr_ExceptionMatches(PyExc_TypeError)) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 Py_DECREF(obj); | 
					
						
							|  |  |  |                 return NULL; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             PyErr_Clear(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return obj; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-22 04:33:02 +08:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | ga_call(PyObject *self, PyObject *args, PyObject *kwds) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)self; | 
					
						
							|  |  |  |     PyObject *obj = PyObject_Call(alias->origin, args, kwds); | 
					
						
							|  |  |  |     return set_orig_class(obj, self); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | ga_vectorcall(PyObject *self, PyObject *const *args, | 
					
						
							|  |  |  |               size_t nargsf, PyObject *kwnames) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *) self; | 
					
						
							|  |  |  |     PyObject *obj = PyVectorcall_Function(alias->origin)(alias->origin, args, nargsf, kwnames); | 
					
						
							|  |  |  |     return set_orig_class(obj, self); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | static const char* const attr_exceptions[] = { | 
					
						
							| 
									
										
										
										
											2022-06-18 11:34:57 +03:00
										 |  |  |     "__class__", | 
					
						
							| 
									
										
										
										
											2023-08-11 22:12:11 +03:00
										 |  |  |     "__bases__", | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     "__origin__", | 
					
						
							|  |  |  |     "__args__", | 
					
						
							| 
									
										
										
										
											2022-05-02 13:21:59 -06:00
										 |  |  |     "__unpacked__", | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     "__parameters__", | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |     "__typing_unpacked_tuple_args__", | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     "__mro_entries__", | 
					
						
							|  |  |  |     "__reduce_ex__",  // needed so we don't look up object.__reduce_ex__
 | 
					
						
							|  |  |  |     "__reduce__", | 
					
						
							| 
									
										
										
										
											2021-09-15 21:25:41 +03:00
										 |  |  |     "__copy__", | 
					
						
							|  |  |  |     "__deepcopy__", | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     NULL, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | ga_getattro(PyObject *self, PyObject *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)self; | 
					
						
							|  |  |  |     if (PyUnicode_Check(name)) { | 
					
						
							|  |  |  |         for (const char * const *p = attr_exceptions; ; p++) { | 
					
						
							|  |  |  |             if (*p == NULL) { | 
					
						
							|  |  |  |                 return PyObject_GetAttr(alias->origin, name); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (_PyUnicode_EqualToASCIIString(name, *p)) { | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return PyObject_GenericGetAttr(self, name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | ga_richcompare(PyObject *a, PyObject *b, int op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-07-17 22:44:10 +03:00
										 |  |  |     if (!_PyGenericAlias_Check(b) || | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         (op != Py_EQ && op != Py_NE)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         Py_RETURN_NOTIMPLEMENTED; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (op == Py_NE) { | 
					
						
							|  |  |  |         PyObject *eq = ga_richcompare(a, b, Py_EQ); | 
					
						
							|  |  |  |         if (eq == NULL) | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         Py_DECREF(eq); | 
					
						
							|  |  |  |         if (eq == Py_True) { | 
					
						
							|  |  |  |             Py_RETURN_FALSE; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             Py_RETURN_TRUE; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     gaobject *aa = (gaobject *)a; | 
					
						
							|  |  |  |     gaobject *bb = (gaobject *)b; | 
					
						
							| 
									
										
										
										
											2022-05-05 20:16:06 +03:00
										 |  |  |     if (aa->starred != bb->starred) { | 
					
						
							|  |  |  |         Py_RETURN_FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     int eq = PyObject_RichCompareBool(aa->origin, bb->origin, Py_EQ); | 
					
						
							|  |  |  |     if (eq < 0) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!eq) { | 
					
						
							|  |  |  |         Py_RETURN_FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return PyObject_RichCompare(aa->args, bb->args, Py_EQ); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | ga_mro_entries(PyObject *self, PyObject *args) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)self; | 
					
						
							|  |  |  |     return PyTuple_Pack(1, alias->origin); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | ga_instancecheck(PyObject *self, PyObject *Py_UNUSED(ignored)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyErr_SetString(PyExc_TypeError, | 
					
						
							|  |  |  |                     "isinstance() argument 2 cannot be a parameterized generic"); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | ga_subclasscheck(PyObject *self, PyObject *Py_UNUSED(ignored)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyErr_SetString(PyExc_TypeError, | 
					
						
							|  |  |  |                     "issubclass() argument 2 cannot be a parameterized generic"); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | ga_reduce(PyObject *self, PyObject *Py_UNUSED(ignored)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)self; | 
					
						
							| 
									
										
										
										
											2022-05-05 20:16:06 +03:00
										 |  |  |     if (alias->starred) { | 
					
						
							|  |  |  |         PyObject *tmp = Py_GenericAlias(alias->origin, alias->args); | 
					
						
							|  |  |  |         if (tmp != NULL) { | 
					
						
							|  |  |  |             Py_SETREF(tmp, PyObject_GetIter(tmp)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (tmp == NULL) { | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return Py_BuildValue("N(N)", _PyEval_GetBuiltin(&_Py_ID(next)), tmp); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     return Py_BuildValue("O(OO)", Py_TYPE(alias), | 
					
						
							|  |  |  |                          alias->origin, alias->args); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-16 00:58:32 +03:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | ga_dir(PyObject *self, PyObject *Py_UNUSED(ignored)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)self; | 
					
						
							|  |  |  |     PyObject *dir = PyObject_Dir(alias->origin); | 
					
						
							|  |  |  |     if (dir == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PyObject *dir_entry = NULL; | 
					
						
							|  |  |  |     for (const char * const *p = attr_exceptions; ; p++) { | 
					
						
							|  |  |  |         if (*p == NULL) { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             dir_entry = PyUnicode_FromString(*p); | 
					
						
							|  |  |  |             if (dir_entry == NULL) { | 
					
						
							|  |  |  |                 goto error; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             int contains = PySequence_Contains(dir, dir_entry); | 
					
						
							|  |  |  |             if (contains < 0) { | 
					
						
							|  |  |  |                 goto error; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (contains == 0 && PyList_Append(dir, dir_entry) < 0) { | 
					
						
							|  |  |  |                 goto error; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             Py_CLEAR(dir_entry); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return dir; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | error: | 
					
						
							|  |  |  |     Py_DECREF(dir); | 
					
						
							|  |  |  |     Py_XDECREF(dir_entry); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | static PyMethodDef ga_methods[] = { | 
					
						
							|  |  |  |     {"__mro_entries__", ga_mro_entries, METH_O}, | 
					
						
							|  |  |  |     {"__instancecheck__", ga_instancecheck, METH_O}, | 
					
						
							|  |  |  |     {"__subclasscheck__", ga_subclasscheck, METH_O}, | 
					
						
							|  |  |  |     {"__reduce__", ga_reduce, METH_NOARGS}, | 
					
						
							| 
									
										
										
										
											2020-09-16 00:58:32 +03:00
										 |  |  |     {"__dir__", ga_dir, METH_NOARGS}, | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     {0} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyMemberDef ga_members[] = { | 
					
						
							| 
									
										
										
										
											2023-07-25 15:28:30 +02:00
										 |  |  |     {"__origin__", _Py_T_OBJECT, offsetof(gaobject, origin), Py_READONLY}, | 
					
						
							|  |  |  |     {"__args__", _Py_T_OBJECT, offsetof(gaobject, args), Py_READONLY}, | 
					
						
							|  |  |  |     {"__unpacked__", Py_T_BOOL, offsetof(gaobject, starred), Py_READONLY}, | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     {0} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | ga_parameters(PyObject *self, void *unused) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)self; | 
					
						
							|  |  |  |     if (alias->parameters == NULL) { | 
					
						
							| 
									
										
										
										
											2021-07-06 21:04:33 +03:00
										 |  |  |         alias->parameters = _Py_make_parameters(alias->args); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         if (alias->parameters == NULL) { | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-11-10 22:22:02 +01:00
										 |  |  |     return Py_NewRef(alias->parameters); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | ga_unpacked_tuple_args(PyObject *self, void *unused) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)self; | 
					
						
							|  |  |  |     if (alias->starred && alias->origin == (PyObject *)&PyTuple_Type) { | 
					
						
							| 
									
										
										
										
											2022-11-10 22:22:02 +01:00
										 |  |  |         return Py_NewRef(alias->args); | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |     } | 
					
						
							|  |  |  |     Py_RETURN_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | static PyGetSetDef ga_properties[] = { | 
					
						
							| 
									
										
										
										
											2023-09-14 04:42:44 +01:00
										 |  |  |     {"__parameters__", ga_parameters, (setter)NULL, PyDoc_STR("Type variables in the GenericAlias."), NULL}, | 
					
						
							| 
									
										
										
										
											2022-05-08 18:32:32 +03:00
										 |  |  |     {"__typing_unpacked_tuple_args__", ga_unpacked_tuple_args, (setter)NULL, NULL}, | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     {0} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-14 02:38:24 +08:00
										 |  |  | /* A helper function to create GenericAlias' args tuple and set its attributes.
 | 
					
						
							| 
									
										
										
										
											2021-08-09 21:32:54 +03:00
										 |  |  |  * Returns 1 on success, 0 on failure. | 
					
						
							| 
									
										
										
										
											2020-12-14 02:38:24 +08:00
										 |  |  |  */ | 
					
						
							|  |  |  | static inline int | 
					
						
							|  |  |  | setup_ga(gaobject *alias, PyObject *origin, PyObject *args) { | 
					
						
							|  |  |  |     if (!PyTuple_Check(args)) { | 
					
						
							|  |  |  |         args = PyTuple_Pack(1, args); | 
					
						
							|  |  |  |         if (args == NULL) { | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         Py_INCREF(args); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-10 22:22:02 +01:00
										 |  |  |     alias->origin = Py_NewRef(origin); | 
					
						
							| 
									
										
										
										
											2020-12-14 02:38:24 +08:00
										 |  |  |     alias->args = args; | 
					
						
							|  |  |  |     alias->parameters = NULL; | 
					
						
							|  |  |  |     alias->weakreflist = NULL; | 
					
						
							| 
									
										
										
										
											2022-03-22 04:33:02 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (PyVectorcall_Function(origin) != NULL) { | 
					
						
							|  |  |  |         alias->vectorcall = ga_vectorcall; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         alias->vectorcall = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-14 02:38:24 +08:00
										 |  |  |     return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | ga_new(PyTypeObject *type, PyObject *args, PyObject *kwds) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-12-05 23:02:14 +07:00
										 |  |  |     if (!_PyArg_NoKeywords("GenericAlias", kwds)) { | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-24 01:25:53 +09:00
										 |  |  |     if (!_PyArg_CheckPositional("GenericAlias", PyTuple_GET_SIZE(args), 2, 2)) { | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PyObject *origin = PyTuple_GET_ITEM(args, 0); | 
					
						
							|  |  |  |     PyObject *arguments = PyTuple_GET_ITEM(args, 1); | 
					
						
							| 
									
										
										
										
											2020-12-14 02:38:24 +08:00
										 |  |  |     gaobject *self = (gaobject *)type->tp_alloc(type, 0); | 
					
						
							|  |  |  |     if (self == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!setup_ga(self, origin, arguments)) { | 
					
						
							| 
									
										
										
										
											2021-07-05 00:47:38 +08:00
										 |  |  |         Py_DECREF(self); | 
					
						
							| 
									
										
										
										
											2020-12-14 02:38:24 +08:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return (PyObject *)self; | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-09 12:00:13 +08:00
										 |  |  | static PyNumberMethods ga_as_number = { | 
					
						
							| 
									
										
										
										
											2021-07-17 22:44:10 +03:00
										 |  |  |         .nb_or = _Py_union_type_or, // Add __or__ function
 | 
					
						
							| 
									
										
										
										
											2020-11-09 12:00:13 +08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-12 12:20:12 +00:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | ga_iternext(gaiterobject *gi) { | 
					
						
							|  |  |  |     if (gi->obj == NULL) { | 
					
						
							|  |  |  |         PyErr_SetNone(PyExc_StopIteration); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     gaobject *alias = (gaobject *)gi->obj; | 
					
						
							|  |  |  |     PyObject *starred_alias = Py_GenericAlias(alias->origin, alias->args); | 
					
						
							|  |  |  |     if (starred_alias == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ((gaobject *)starred_alias)->starred = true; | 
					
						
							|  |  |  |     Py_SETREF(gi->obj, NULL); | 
					
						
							|  |  |  |     return starred_alias; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | ga_iter_dealloc(gaiterobject *gi) { | 
					
						
							|  |  |  |     PyObject_GC_UnTrack(gi); | 
					
						
							|  |  |  |     Py_XDECREF(gi->obj); | 
					
						
							|  |  |  |     PyObject_GC_Del(gi); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | ga_iter_traverse(gaiterobject *gi, visitproc visit, void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Py_VISIT(gi->obj); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | ga_iter_clear(PyObject *self) { | 
					
						
							|  |  |  |     gaiterobject *gi = (gaiterobject *)self; | 
					
						
							|  |  |  |     Py_CLEAR(gi->obj); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-05 20:16:06 +03:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | ga_iter_reduce(PyObject *self, PyObject *Py_UNUSED(ignored)) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-02-24 18:02:04 -05:00
										 |  |  |     PyObject *iter = _PyEval_GetBuiltin(&_Py_ID(iter)); | 
					
						
							| 
									
										
										
										
											2022-05-05 20:16:06 +03:00
										 |  |  |     gaiterobject *gi = (gaiterobject *)self; | 
					
						
							| 
									
										
										
										
											2023-02-24 18:02:04 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* _PyEval_GetBuiltin can invoke arbitrary code,
 | 
					
						
							|  |  |  |      * call must be before access of iterator pointers. | 
					
						
							|  |  |  |      * see issue #101765 */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (gi->obj) | 
					
						
							|  |  |  |         return Py_BuildValue("N(O)", iter, gi->obj); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         return Py_BuildValue("N(())", iter); | 
					
						
							| 
									
										
										
										
											2022-05-05 20:16:06 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyMethodDef ga_iter_methods[] = { | 
					
						
							|  |  |  |     {"__reduce__", ga_iter_reduce, METH_NOARGS}, | 
					
						
							|  |  |  |     {0} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-20 23:10:41 +09:00
										 |  |  | // gh-91632: _Py_GenericAliasIterType is exported  to be cleared
 | 
					
						
							|  |  |  | // in _PyTypes_FiniTypes.
 | 
					
						
							|  |  |  | PyTypeObject _Py_GenericAliasIterType = { | 
					
						
							| 
									
										
										
										
											2022-03-12 12:20:12 +00:00
										 |  |  |     PyVarObject_HEAD_INIT(&PyType_Type, 0) | 
					
						
							|  |  |  |     .tp_name = "generic_alias_iterator", | 
					
						
							|  |  |  |     .tp_basicsize = sizeof(gaiterobject), | 
					
						
							|  |  |  |     .tp_iter = PyObject_SelfIter, | 
					
						
							|  |  |  |     .tp_iternext = (iternextfunc)ga_iternext, | 
					
						
							|  |  |  |     .tp_traverse = (traverseproc)ga_iter_traverse, | 
					
						
							| 
									
										
										
										
											2022-05-05 20:16:06 +03:00
										 |  |  |     .tp_methods = ga_iter_methods, | 
					
						
							| 
									
										
										
										
											2022-03-12 12:20:12 +00:00
										 |  |  |     .tp_dealloc = (destructor)ga_iter_dealloc, | 
					
						
							|  |  |  |     .tp_clear = (inquiry)ga_iter_clear, | 
					
						
							|  |  |  |     .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | ga_iter(PyObject *self) { | 
					
						
							| 
									
										
										
										
											2022-04-20 23:10:41 +09:00
										 |  |  |     gaiterobject *gi = PyObject_GC_New(gaiterobject, &_Py_GenericAliasIterType); | 
					
						
							| 
									
										
										
										
											2022-03-12 12:20:12 +00:00
										 |  |  |     if (gi == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     gi->obj = Py_NewRef(self); | 
					
						
							|  |  |  |     PyObject_GC_Track(gi); | 
					
						
							|  |  |  |     return (PyObject *)gi; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | // TODO:
 | 
					
						
							|  |  |  | // - argument clinic?
 | 
					
						
							|  |  |  | // - cache?
 | 
					
						
							|  |  |  | PyTypeObject Py_GenericAliasType = { | 
					
						
							|  |  |  |     PyVarObject_HEAD_INIT(&PyType_Type, 0) | 
					
						
							|  |  |  |     .tp_name = "types.GenericAlias", | 
					
						
							| 
									
										
										
										
											2022-04-18 06:39:32 +03:00
										 |  |  |     .tp_doc = genericalias__doc__, | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     .tp_basicsize = sizeof(gaobject), | 
					
						
							|  |  |  |     .tp_dealloc = ga_dealloc, | 
					
						
							|  |  |  |     .tp_repr = ga_repr, | 
					
						
							| 
									
										
										
										
											2020-11-09 12:00:13 +08:00
										 |  |  |     .tp_as_number = &ga_as_number,  // allow X | Y of GenericAlias objs
 | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     .tp_as_mapping = &ga_as_mapping, | 
					
						
							|  |  |  |     .tp_hash = ga_hash, | 
					
						
							|  |  |  |     .tp_call = ga_call, | 
					
						
							|  |  |  |     .tp_getattro = ga_getattro, | 
					
						
							| 
									
										
										
										
											2022-03-22 04:33:02 +08:00
										 |  |  |     .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC | Py_TPFLAGS_BASETYPE | Py_TPFLAGS_HAVE_VECTORCALL, | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     .tp_traverse = ga_traverse, | 
					
						
							|  |  |  |     .tp_richcompare = ga_richcompare, | 
					
						
							| 
									
										
										
										
											2020-11-16 11:27:23 +08:00
										 |  |  |     .tp_weaklistoffset = offsetof(gaobject, weakreflist), | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     .tp_methods = ga_methods, | 
					
						
							|  |  |  |     .tp_members = ga_members, | 
					
						
							|  |  |  |     .tp_alloc = PyType_GenericAlloc, | 
					
						
							|  |  |  |     .tp_new = ga_new, | 
					
						
							|  |  |  |     .tp_free = PyObject_GC_Del, | 
					
						
							|  |  |  |     .tp_getset = ga_properties, | 
					
						
							| 
									
										
										
										
											2022-03-12 12:20:12 +00:00
										 |  |  |     .tp_iter = (getiterfunc)ga_iter, | 
					
						
							| 
									
										
										
										
											2022-03-22 04:33:02 +08:00
										 |  |  |     .tp_vectorcall_offset = offsetof(gaobject, vectorcall), | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject * | 
					
						
							|  |  |  | Py_GenericAlias(PyObject *origin, PyObject *args) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-07-05 12:10:53 +01:00
										 |  |  |     gaobject *alias = (gaobject*) PyType_GenericAlloc( | 
					
						
							|  |  |  |             (PyTypeObject *)&Py_GenericAliasType, 0); | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     if (alias == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-12-14 02:38:24 +08:00
										 |  |  |     if (!setup_ga(alias, origin, args)) { | 
					
						
							| 
									
										
										
										
											2021-07-05 00:47:38 +08:00
										 |  |  |         Py_DECREF(alias); | 
					
						
							| 
									
										
										
										
											2020-12-14 02:38:24 +08:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-07 09:50:06 -07:00
										 |  |  |     return (PyObject *)alias; | 
					
						
							|  |  |  | } |