mirror of
				https://github.com/python/cpython.git
				synced 2025-10-26 03:04:41 +00:00 
			
		
		
		
	 0ef4ffeefd
			
		
	
	
		0ef4ffeefd
		
			
		
	
	
	
	
		
			
			The use of PySys_GetObject() and _PySys_GetAttr(), which return a borrowed reference, has been replaced by using one of the following functions, which return a strong reference and distinguish a missing attribute from an error: _PySys_GetOptionalAttr(), _PySys_GetOptionalAttrString(), _PySys_GetRequiredAttr(), and _PySys_GetRequiredAttrString().
		
			
				
	
	
		
			286 lines
		
	
	
	
		
			8.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			286 lines
		
	
	
	
		
			8.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| 
 | |
| #define _PY_INTERPRETER
 | |
| 
 | |
| #include "Python.h"
 | |
| #include "pycore_frame.h"
 | |
| #include "pycore_function.h"
 | |
| #include "pycore_global_objects.h"
 | |
| #include "pycore_compile.h"       // _PyCompile_GetUnaryIntrinsicName, etc
 | |
| #include "pycore_intrinsics.h"    // INTRINSIC_PRINT
 | |
| #include "pycore_pyerrors.h"      // _PyErr_SetString()
 | |
| #include "pycore_runtime.h"       // _Py_ID()
 | |
| #include "pycore_sysmodule.h"     // _PySys_GetRequiredAttr()
 | |
| #include "pycore_typevarobject.h" // _Py_make_typevar()
 | |
| 
 | |
| 
 | |
| /******** Unary functions ********/
 | |
| 
 | |
| static PyObject *
 | |
| no_intrinsic1(PyThreadState* tstate, PyObject *unused)
 | |
| {
 | |
|     _PyErr_SetString(tstate, PyExc_SystemError, "invalid intrinsic function");
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| print_expr(PyThreadState* Py_UNUSED(ignored), PyObject *value)
 | |
| {
 | |
|     PyObject *hook = _PySys_GetRequiredAttr(&_Py_ID(displayhook));
 | |
|     // Can't use ERROR_IF here.
 | |
|     if (hook == NULL) {
 | |
|         return NULL;
 | |
|     }
 | |
|     PyObject *res = PyObject_CallOneArg(hook, value);
 | |
|     Py_DECREF(hook);
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| static int
 | |
| import_all_from(PyThreadState *tstate, PyObject *locals, PyObject *v)
 | |
| {
 | |
|     PyObject *all, *dict, *name, *value;
 | |
|     int skip_leading_underscores = 0;
 | |
|     int pos, err;
 | |
| 
 | |
|     if (PyObject_GetOptionalAttr(v, &_Py_ID(__all__), &all) < 0) {
 | |
|         return -1; /* Unexpected error */
 | |
|     }
 | |
|     if (all == NULL) {
 | |
|         if (PyObject_GetOptionalAttr(v, &_Py_ID(__dict__), &dict) < 0) {
 | |
|             return -1;
 | |
|         }
 | |
|         if (dict == NULL) {
 | |
|             _PyErr_SetString(tstate, PyExc_ImportError,
 | |
|                     "from-import-* object has no __dict__ and no __all__");
 | |
|             return -1;
 | |
|         }
 | |
|         all = PyMapping_Keys(dict);
 | |
|         Py_DECREF(dict);
 | |
|         if (all == NULL)
 | |
|             return -1;
 | |
|         skip_leading_underscores = 1;
 | |
|     }
 | |
| 
 | |
|     for (pos = 0, err = 0; ; pos++) {
 | |
|         name = PySequence_GetItem(all, pos);
 | |
|         if (name == NULL) {
 | |
|             if (!_PyErr_ExceptionMatches(tstate, PyExc_IndexError)) {
 | |
|                 err = -1;
 | |
|             }
 | |
|             else {
 | |
|                 _PyErr_Clear(tstate);
 | |
|             }
 | |
|             break;
 | |
|         }
 | |
|         if (!PyUnicode_Check(name)) {
 | |
|             PyObject *modname = PyObject_GetAttr(v, &_Py_ID(__name__));
 | |
|             if (modname == NULL) {
 | |
|                 Py_DECREF(name);
 | |
|                 err = -1;
 | |
|                 break;
 | |
|             }
 | |
|             if (!PyUnicode_Check(modname)) {
 | |
|                 _PyErr_Format(tstate, PyExc_TypeError,
 | |
|                               "module __name__ must be a string, not %.100s",
 | |
|                               Py_TYPE(modname)->tp_name);
 | |
|             }
 | |
|             else {
 | |
|                 _PyErr_Format(tstate, PyExc_TypeError,
 | |
|                               "%s in %U.%s must be str, not %.100s",
 | |
|                               skip_leading_underscores ? "Key" : "Item",
 | |
|                               modname,
 | |
|                               skip_leading_underscores ? "__dict__" : "__all__",
 | |
|                               Py_TYPE(name)->tp_name);
 | |
|             }
 | |
|             Py_DECREF(modname);
 | |
|             Py_DECREF(name);
 | |
|             err = -1;
 | |
|             break;
 | |
|         }
 | |
|         if (skip_leading_underscores) {
 | |
|             if (PyUnicode_READ_CHAR(name, 0) == '_') {
 | |
|                 Py_DECREF(name);
 | |
|                 continue;
 | |
|             }
 | |
|         }
 | |
|         value = PyObject_GetAttr(v, name);
 | |
|         if (value == NULL)
 | |
|             err = -1;
 | |
|         else if (PyDict_CheckExact(locals))
 | |
|             err = PyDict_SetItem(locals, name, value);
 | |
|         else
 | |
|             err = PyObject_SetItem(locals, name, value);
 | |
|         Py_DECREF(name);
 | |
|         Py_XDECREF(value);
 | |
|         if (err < 0)
 | |
|             break;
 | |
|     }
 | |
|     Py_DECREF(all);
 | |
|     return err;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| import_star(PyThreadState* tstate, PyObject *from)
 | |
| {
 | |
|     _PyInterpreterFrame *frame = tstate->current_frame;
 | |
| 
 | |
|     PyObject *locals = _PyFrame_GetLocals(frame);
 | |
|     if (locals == NULL) {
 | |
|         _PyErr_SetString(tstate, PyExc_SystemError,
 | |
|                             "no locals found during 'import *'");
 | |
|         return NULL;
 | |
|     }
 | |
|     int err = import_all_from(tstate, locals, from);
 | |
|     Py_DECREF(locals);
 | |
|     if (err < 0) {
 | |
|         return NULL;
 | |
|     }
 | |
|     Py_RETURN_NONE;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| stopiteration_error(PyThreadState* tstate, PyObject *exc)
 | |
| {
 | |
|     _PyInterpreterFrame *frame = tstate->current_frame;
 | |
|     assert(frame->owner == FRAME_OWNED_BY_GENERATOR);
 | |
|     assert(PyExceptionInstance_Check(exc));
 | |
|     const char *msg = NULL;
 | |
|     if (PyErr_GivenExceptionMatches(exc, PyExc_StopIteration)) {
 | |
|         msg = "generator raised StopIteration";
 | |
|         if (_PyFrame_GetCode(frame)->co_flags & CO_ASYNC_GENERATOR) {
 | |
|             msg = "async generator raised StopIteration";
 | |
|         }
 | |
|         else if (_PyFrame_GetCode(frame)->co_flags & CO_COROUTINE) {
 | |
|             msg = "coroutine raised StopIteration";
 | |
|         }
 | |
|     }
 | |
|     else if ((_PyFrame_GetCode(frame)->co_flags & CO_ASYNC_GENERATOR) &&
 | |
|             PyErr_GivenExceptionMatches(exc, PyExc_StopAsyncIteration))
 | |
|     {
 | |
|         /* code in `gen` raised a StopAsyncIteration error:
 | |
|         raise a RuntimeError.
 | |
|         */
 | |
|         msg = "async generator raised StopAsyncIteration";
 | |
|     }
 | |
|     if (msg != NULL) {
 | |
|         PyObject *message = _PyUnicode_FromASCII(msg, strlen(msg));
 | |
|         if (message == NULL) {
 | |
|             return NULL;
 | |
|         }
 | |
|         PyObject *error = PyObject_CallOneArg(PyExc_RuntimeError, message);
 | |
|         if (error == NULL) {
 | |
|             Py_DECREF(message);
 | |
|             return NULL;
 | |
|         }
 | |
|         assert(PyExceptionInstance_Check(error));
 | |
|         PyException_SetCause(error, Py_NewRef(exc));
 | |
|         // Steal exc reference, rather than Py_NewRef+Py_DECREF
 | |
|         PyException_SetContext(error, Py_NewRef(exc));
 | |
|         Py_DECREF(message);
 | |
|         return error;
 | |
|     }
 | |
|     return Py_NewRef(exc);
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| unary_pos(PyThreadState* unused, PyObject *value)
 | |
| {
 | |
|     return PyNumber_Positive(value);
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| list_to_tuple(PyThreadState* unused, PyObject *v)
 | |
| {
 | |
|     assert(PyList_Check(v));
 | |
|     return _PyTuple_FromArray(((PyListObject *)v)->ob_item, Py_SIZE(v));
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| make_typevar(PyThreadState* Py_UNUSED(ignored), PyObject *v)
 | |
| {
 | |
|     assert(PyUnicode_Check(v));
 | |
|     return _Py_make_typevar(v, NULL, NULL);
 | |
| }
 | |
| 
 | |
| 
 | |
| #define INTRINSIC_FUNC_ENTRY(N, F) \
 | |
|     [N] = {F, #N},
 | |
| 
 | |
| const intrinsic_func1_info
 | |
| _PyIntrinsics_UnaryFunctions[] = {
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_1_INVALID, no_intrinsic1)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_PRINT, print_expr)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_IMPORT_STAR, import_star)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_STOPITERATION_ERROR, stopiteration_error)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_ASYNC_GEN_WRAP, _PyAsyncGenValueWrapperNew)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_UNARY_POSITIVE, unary_pos)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_LIST_TO_TUPLE, list_to_tuple)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_TYPEVAR, make_typevar)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_PARAMSPEC, _Py_make_paramspec)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_TYPEVARTUPLE, _Py_make_typevartuple)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_SUBSCRIPT_GENERIC, _Py_subscript_generic)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_TYPEALIAS, _Py_make_typealias)
 | |
| };
 | |
| 
 | |
| 
 | |
| /******** Binary functions ********/
 | |
| 
 | |
| static PyObject *
 | |
| no_intrinsic2(PyThreadState* tstate, PyObject *unused1, PyObject *unused2)
 | |
| {
 | |
|     _PyErr_SetString(tstate, PyExc_SystemError, "invalid intrinsic function");
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| prep_reraise_star(PyThreadState* unused, PyObject *orig, PyObject *excs)
 | |
| {
 | |
|     assert(PyList_Check(excs));
 | |
|     return _PyExc_PrepReraiseStar(orig, excs);
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| make_typevar_with_bound(PyThreadState* Py_UNUSED(ignored), PyObject *name,
 | |
|                         PyObject *evaluate_bound)
 | |
| {
 | |
|     assert(PyUnicode_Check(name));
 | |
|     return _Py_make_typevar(name, evaluate_bound, NULL);
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| make_typevar_with_constraints(PyThreadState* Py_UNUSED(ignored), PyObject *name,
 | |
|                               PyObject *evaluate_constraints)
 | |
| {
 | |
|     assert(PyUnicode_Check(name));
 | |
|     return _Py_make_typevar(name, NULL, evaluate_constraints);
 | |
| }
 | |
| 
 | |
| const intrinsic_func2_info
 | |
| _PyIntrinsics_BinaryFunctions[] = {
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_2_INVALID, no_intrinsic2)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_PREP_RERAISE_STAR, prep_reraise_star)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_TYPEVAR_WITH_BOUND, make_typevar_with_bound)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_TYPEVAR_WITH_CONSTRAINTS, make_typevar_with_constraints)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_SET_FUNCTION_TYPE_PARAMS, _Py_set_function_type_params)
 | |
|     INTRINSIC_FUNC_ENTRY(INTRINSIC_SET_TYPEPARAM_DEFAULT, _Py_set_typeparam_default)
 | |
| };
 | |
| 
 | |
| #undef INTRINSIC_FUNC_ENTRY
 | |
| 
 | |
| PyObject*
 | |
| _PyCompile_GetUnaryIntrinsicName(int index)
 | |
| {
 | |
|     if (index < 0 || index > MAX_INTRINSIC_1) {
 | |
|         return NULL;
 | |
|     }
 | |
|     return PyUnicode_FromString(_PyIntrinsics_UnaryFunctions[index].name);
 | |
| }
 | |
| 
 | |
| PyObject*
 | |
| _PyCompile_GetBinaryIntrinsicName(int index)
 | |
| {
 | |
|     if (index < 0 || index > MAX_INTRINSIC_2) {
 | |
|         return NULL;
 | |
|     }
 | |
|     return PyUnicode_FromString(_PyIntrinsics_BinaryFunctions[index].name);
 | |
| }
 |