diff options
Diffstat (limited to 'Python/sysmodule.c')
| -rw-r--r-- | Python/sysmodule.c | 100 | 
1 files changed, 94 insertions, 6 deletions
| diff --git a/Python/sysmodule.c b/Python/sysmodule.c index 290eec1..334f5d0 100644 --- a/Python/sysmodule.c +++ b/Python/sysmodule.c @@ -632,19 +632,84 @@ processor's time-stamp counter."  static PyObject *  sys_setrecursionlimit(PyObject *self, PyObject *args)  { -    int new_limit; +    int new_limit, mark; +    PyThreadState *tstate; +      if (!PyArg_ParseTuple(args, "i:setrecursionlimit", &new_limit))          return NULL; -    if (new_limit <= 0) { + +    if (new_limit < 1) {          PyErr_SetString(PyExc_ValueError, -                        "recursion limit must be positive"); +                        "recursion limit must be greater or equal than 1"); +        return NULL; +    } + +    /* Issue #25274: When the recursion depth hits the recursion limit in +       _Py_CheckRecursiveCall(), the overflowed flag of the thread state is +       set to 1 and a RecursionError is raised. The overflowed flag is reset +       to 0 when the recursion depth goes below the low-water mark: see +       Py_LeaveRecursiveCall(). + +       Reject too low new limit if the current recursion depth is higher than +       the new low-water mark. Otherwise it may not be possible anymore to +       reset the overflowed flag to 0. */ +    mark = _Py_RecursionLimitLowerWaterMark(new_limit); +    tstate = PyThreadState_GET(); +    if (tstate->recursion_depth >= mark) { +        PyErr_Format(PyExc_RecursionError, +                     "cannot set the recursion limit to %i at " +                     "the recursion depth %i: the limit is too low", +                     new_limit, tstate->recursion_depth);          return NULL;      } +      Py_SetRecursionLimit(new_limit);      Py_INCREF(Py_None);      return Py_None;  } +static PyObject * +sys_set_coroutine_wrapper(PyObject *self, PyObject *wrapper) +{ +    if (wrapper != Py_None) { +        if (!PyCallable_Check(wrapper)) { +            PyErr_Format(PyExc_TypeError, +                         "callable expected, got %.50s", +                         Py_TYPE(wrapper)->tp_name); +            return NULL; +        } +        _PyEval_SetCoroutineWrapper(wrapper); +    } +    else { +        _PyEval_SetCoroutineWrapper(NULL); +    } +    Py_RETURN_NONE; +} + +PyDoc_STRVAR(set_coroutine_wrapper_doc, +"set_coroutine_wrapper(wrapper)\n\ +\n\ +Set a wrapper for coroutine objects." +); + +static PyObject * +sys_get_coroutine_wrapper(PyObject *self, PyObject *args) +{ +    PyObject *wrapper = _PyEval_GetCoroutineWrapper(); +    if (wrapper == NULL) { +        wrapper = Py_None; +    } +    Py_INCREF(wrapper); +    return wrapper; +} + +PyDoc_STRVAR(get_coroutine_wrapper_doc, +"get_coroutine_wrapper()\n\ +\n\ +Return the wrapper for coroutine objects set by sys.set_coroutine_wrapper." +); + +  static PyTypeObject Hash_InfoType;  PyDoc_STRVAR(hash_info_doc, @@ -772,6 +837,12 @@ static PyStructSequence_Desc windows_version_desc = {                                   via indexing, the rest are name only */  }; +/* Disable deprecation warnings about GetVersionEx as the result is +   being passed straight through to the caller, who is responsible for +   using it correctly. */ +#pragma warning(push) +#pragma warning(disable:4996) +  static PyObject *  sys_getwindowsversion(PyObject *self)  { @@ -803,6 +874,8 @@ sys_getwindowsversion(PyObject *self)      return version;  } +#pragma warning(pop) +  #endif /* MS_WINDOWS */  #ifdef HAVE_DLOPEN @@ -1121,6 +1194,16 @@ PyDoc_STRVAR(sys_clear_type_cache__doc__,  "_clear_type_cache() -> None\n\  Clear the internal type lookup cache."); +static PyObject * +sys_is_finalizing(PyObject* self, PyObject* args) +{ +    return PyBool_FromLong(_Py_Finalizing != NULL); +} + +PyDoc_STRVAR(is_finalizing_doc, +"is_finalizing()\n\ +Return True if Python is exiting."); +  static PyMethodDef sys_methods[] = {      /* Might as well keep this in alphabetic order */ @@ -1167,6 +1250,7 @@ static PyMethodDef sys_methods[] = {       getwindowsversion_doc},  #endif /* MS_WINDOWS */      {"intern",          sys_intern,     METH_VARARGS, intern_doc}, +    {"is_finalizing",   sys_is_finalizing, METH_NOARGS, is_finalizing_doc},  #ifdef USE_MALLOPT      {"mdebug",          sys_mdebug, METH_VARARGS},  #endif @@ -1196,6 +1280,10 @@ static PyMethodDef sys_methods[] = {      {"call_tracing", sys_call_tracing, METH_VARARGS, call_tracing_doc},      {"_debugmallocstats", sys_debugmallocstats, METH_NOARGS,       debugmallocstats_doc}, +    {"set_coroutine_wrapper", sys_set_coroutine_wrapper, METH_O, +     set_coroutine_wrapper_doc}, +    {"get_coroutine_wrapper", sys_get_coroutine_wrapper, METH_NOARGS, +     get_coroutine_wrapper_doc},      {NULL,              NULL}           /* sentinel */  }; @@ -1670,8 +1758,8 @@ _PySys_Init(void)      the shell already prevents that. */  #if !defined(MS_WINDOWS)      { -        struct stat sb; -        if (fstat(fileno(stdin), &sb) == 0 && +        struct _Py_stat_struct sb; +        if (_Py_fstat_noraise(fileno(stdin), &sb) == 0 &&              S_ISDIR(sb.st_mode)) {              /* There's nothing more we can do. */              /* Py_FatalError() will core dump, so just exit. */ @@ -1681,7 +1769,7 @@ _PySys_Init(void)      }  #endif -    /* stdin/stdout/stderr are now set by pythonrun.c */ +    /* stdin/stdout/stderr are set in pylifecycle.c */      SET_SYS_FROM_STRING_BORROW("__displayhook__",                                 PyDict_GetItemString(sysdict, "displayhook")); | 
