diff options
Diffstat (limited to 'Objects/methodobject.c')
| -rw-r--r-- | Objects/methodobject.c | 148 | 
1 files changed, 120 insertions, 28 deletions
| diff --git a/Objects/methodobject.c b/Objects/methodobject.c index f0685dd606..6179aeebd0 100644 --- a/Objects/methodobject.c +++ b/Objects/methodobject.c @@ -13,6 +13,15 @@ static int numfree = 0;  #define PyCFunction_MAXFREELIST 256  #endif +/* undefine macro trampoline to PyCFunction_NewEx */ +#undef PyCFunction_New + +PyObject * +PyCFunction_New(PyMethodDef *ml, PyObject *self) +{ +    return PyCFunction_NewEx(ml, self, NULL); +} +  PyObject *  PyCFunction_NewEx(PyMethodDef *ml, PyObject *self, PyObject *module)  { @@ -20,7 +29,7 @@ PyCFunction_NewEx(PyMethodDef *ml, PyObject *self, PyObject *module)      op = free_list;      if (op != NULL) {          free_list = (PyCFunctionObject *)(op->m_self); -        PyObject_INIT(op, &PyCFunction_Type); +        (void)PyObject_INIT(op, &PyCFunction_Type);          numfree--;      }      else { @@ -70,23 +79,34 @@ PyCFunction_GetFlags(PyObject *op)  PyObject *  PyCFunction_Call(PyObject *func, PyObject *arg, PyObject *kw)  { +#define CHECK_RESULT(res) assert(res != NULL || PyErr_Occurred()) +      PyCFunctionObject* f = (PyCFunctionObject*)func;      PyCFunction meth = PyCFunction_GET_FUNCTION(func);      PyObject *self = PyCFunction_GET_SELF(func); +    PyObject *res;      Py_ssize_t size;      switch (PyCFunction_GET_FLAGS(func) & ~(METH_CLASS | METH_STATIC | METH_COEXIST)) {      case METH_VARARGS: -        if (kw == NULL || PyDict_Size(kw) == 0) -            return (*meth)(self, arg); +        if (kw == NULL || PyDict_Size(kw) == 0) { +            res = (*meth)(self, arg); +            CHECK_RESULT(res); +            return res; +        }          break;      case METH_VARARGS | METH_KEYWORDS: -        return (*(PyCFunctionWithKeywords)meth)(self, arg, kw); +        res = (*(PyCFunctionWithKeywords)meth)(self, arg, kw); +        CHECK_RESULT(res); +        return res;      case METH_NOARGS:          if (kw == NULL || PyDict_Size(kw) == 0) {              size = PyTuple_GET_SIZE(arg); -            if (size == 0) -                return (*meth)(self, NULL); +            if (size == 0) { +                res = (*meth)(self, NULL); +                CHECK_RESULT(res); +                return res; +            }              PyErr_Format(PyExc_TypeError,                  "%.200s() takes no arguments (%zd given)",                  f->m_ml->ml_name, size); @@ -96,8 +116,11 @@ PyCFunction_Call(PyObject *func, PyObject *arg, PyObject *kw)      case METH_O:          if (kw == NULL || PyDict_Size(kw) == 0) {              size = PyTuple_GET_SIZE(arg); -            if (size == 1) -                return (*meth)(self, PyTuple_GET_ITEM(arg, 0)); +            if (size == 1) { +                res = (*meth)(self, PyTuple_GET_ITEM(arg, 0)); +                CHECK_RESULT(res); +                return res; +            }              PyErr_Format(PyExc_TypeError,                  "%.200s() takes exactly one argument (%zd given)",                  f->m_ml->ml_name, size); @@ -114,6 +137,8 @@ PyCFunction_Call(PyObject *func, PyObject *arg, PyObject *kw)      PyErr_Format(PyExc_TypeError, "%.200s() takes no keyword arguments",                   f->m_ml->ml_name);      return NULL; + +#undef CHECK_RESULT  }  /* Methods (the standard built-in methods, that is) */ @@ -135,14 +160,94 @@ meth_dealloc(PyCFunctionObject *m)  }  static PyObject * +meth_reduce(PyCFunctionObject *m) +{ +    PyObject *builtins; +    PyObject *getattr; +    _Py_IDENTIFIER(getattr); + +    if (m->m_self == NULL || PyModule_Check(m->m_self)) +        return PyUnicode_FromString(m->m_ml->ml_name); + +    builtins = PyEval_GetBuiltins(); +    getattr = _PyDict_GetItemId(builtins, &PyId_getattr); +    return Py_BuildValue("O(Os)", getattr, m->m_self, m->m_ml->ml_name); +} + +static PyMethodDef meth_methods[] = { +    {"__reduce__", (PyCFunction)meth_reduce, METH_NOARGS, NULL}, +    {NULL, NULL} +}; + +/* + * finds the docstring's introspection signature. + * if present, returns a pointer pointing to the first '('. + * otherwise returns NULL. + */ +static const char *find_signature(PyCFunctionObject *m) +{ +    const char *trace = m->m_ml->ml_doc; +    const char *name = m->m_ml->ml_name; +    size_t length; +    if (!trace || !name) +        return NULL; +    length = strlen(name); +    if (strncmp(trace, name, length)) +        return NULL; +    trace += length; +    if (*trace != '(') +        return NULL; +    return trace; +} + +/* + * skips to the end of the docstring's instrospection signature. + */ +static const char *skip_signature(const char *trace) +{ +    while (*trace && *trace != '\n') +        trace++; +    return trace; +} + +static const char *skip_eols(const char *trace) +{ +    while (*trace == '\n') +        trace++; +    return trace; +} + +static PyObject * +meth_get__text_signature__(PyCFunctionObject *m, void *closure) +{ +    const char *start = find_signature(m); +    const char *trace; + +    if (!start) { +        Py_INCREF(Py_None); +        return Py_None; +    } + +    trace = skip_signature(start); +    return PyUnicode_FromStringAndSize(start, trace - start); +} + +static PyObject *  meth_get__doc__(PyCFunctionObject *m, void *closure)  { -    const char *doc = m->m_ml->ml_doc; +    const char *doc = find_signature(m); -    if (doc != NULL) -        return PyUnicode_FromString(doc); -    Py_INCREF(Py_None); -    return Py_None; +    if (doc) +        doc = skip_eols(skip_signature(doc)); +    else +        doc = m->m_ml->ml_doc; +     +    if (!doc) { +        Py_INCREF(Py_None); +        return Py_None; +    } + +    return PyUnicode_FromString(doc);  }  static PyObject * @@ -211,6 +316,7 @@ static PyGetSetDef meth_getsets [] = {      {"__name__", (getter)meth_get__name__, NULL, NULL},      {"__qualname__", (getter)meth_get__qualname__, NULL, NULL},      {"__self__", (getter)meth_get__self__, NULL, NULL}, +    {"__text_signature__", (getter)meth_get__text_signature__, NULL, NULL},      {0}  }; @@ -308,7 +414,7 @@ PyTypeObject PyCFunction_Type = {      0,                                          /* tp_weaklistoffset */      0,                                          /* tp_iter */      0,                                          /* tp_iternext */ -    0,                                          /* tp_methods */ +    meth_methods,                               /* tp_methods */      meth_members,                               /* tp_members */      meth_getsets,                               /* tp_getset */      0,                                          /* tp_base */ @@ -346,17 +452,3 @@ _PyCFunction_DebugMallocStats(FILE *out)                             "free PyCFunctionObject",                             numfree, sizeof(PyCFunctionObject));  } - -/* PyCFunction_New() is now just a macro that calls PyCFunction_NewEx(), -   but it's part of the API so we need to keep a function around that -   existing C extensions can call. -*/ - -#undef PyCFunction_New -PyAPI_FUNC(PyObject *) PyCFunction_New(PyMethodDef *, PyObject *); - -PyObject * -PyCFunction_New(PyMethodDef *ml, PyObject *self) -{ -    return PyCFunction_NewEx(ml, self, NULL); -} | 
