Optimize methoddescr_call

refer python/cpython@2a1b676d1f
refer python/cpython@c52572319c
refer python/cpython@35ecebe165
refer python/cpython@8128d5a491
This commit is contained in:
ahgamut 2021-11-05 01:29:08 +05:30
parent 73ea0ea4a6
commit 9220b67b3e
6 changed files with 51 additions and 36 deletions

View file

@ -24,8 +24,7 @@ int _PyStack_UnpackDict(
Py_ssize_t nargs, Py_ssize_t nargs,
PyObject *kwargs, PyObject *kwargs,
PyObject ***p_stack, PyObject ***p_stack,
PyObject **p_kwnames, PyObject **p_kwnames);
PyObject *func);
PyObject *_PyObject_FastCallDict(PyObject *func, PyObject **args, PyObject *_PyObject_FastCallDict(PyObject *func, PyObject **args,
Py_ssize_t nargs, PyObject *kwargs); Py_ssize_t nargs, PyObject *kwargs);

View file

@ -93,6 +93,12 @@ typedef struct {
PyObject *m_module; /* The __module__ attribute, can be anything */ PyObject *m_module; /* The __module__ attribute, can be anything */
PyObject *m_weakreflist; /* List of weak references */ PyObject *m_weakreflist; /* List of weak references */
} PyCFunctionObject; } PyCFunctionObject;
PyObject * _PyMethodDef_RawFastCallDict(
PyMethodDef *method,
PyObject *self,
PyObject **args,
Py_ssize_t nargs,
PyObject *kwargs);
#endif #endif
int PyCFunction_ClearFreeList(void); int PyCFunction_ClearFreeList(void);

View file

@ -3867,11 +3867,15 @@ sock_sendto(PySocketSockObject *s, PyObject *args)
arglen = PyTuple_Size(args); arglen = PyTuple_Size(args);
switch (arglen) { switch (arglen) {
case 2: case 2:
PyArg_ParseTuple(args, "y*O:sendto", &pbuf, &addro); if (!PyArg_ParseTuple(args, "y*O:sendto", &pbuf, &addro)) {
return NULL;
}
break; break;
case 3: case 3:
PyArg_ParseTuple(args, "y*iO:sendto", if (!PyArg_ParseTuple(args, "y*iO:sendto",
&pbuf, &flags, &addro); &pbuf, &flags, &addro)) {
return NULL;
}
break; break;
default: default:
PyErr_Format(PyExc_TypeError, PyErr_Format(PyExc_TypeError,
@ -3879,8 +3883,6 @@ sock_sendto(PySocketSockObject *s, PyObject *args)
arglen); arglen);
return NULL; return NULL;
} }
if (PyErr_Occurred())
return NULL;
if (!IS_SELECTABLE(s)) { if (!IS_SELECTABLE(s)) {
PyBuffer_Release(&pbuf); PyBuffer_Release(&pbuf);

View file

@ -2446,7 +2446,7 @@ _PyStack_AsDict(PyObject **values, PyObject *kwnames)
int int
_PyStack_UnpackDict(PyObject **args, Py_ssize_t nargs, PyObject *kwargs, _PyStack_UnpackDict(PyObject **args, Py_ssize_t nargs, PyObject *kwargs,
PyObject ***p_stack, PyObject **p_kwnames, PyObject *func) PyObject ***p_stack, PyObject **p_kwnames)
{ {
PyObject **stack, **kwstack; PyObject **stack, **kwstack;
Py_ssize_t nkwargs; Py_ssize_t nkwargs;

View file

@ -228,15 +228,15 @@ getset_set(PyGetSetDescrObject *descr, PyObject *obj, PyObject *value)
} }
static PyObject * static PyObject *
methoddescr_call(PyMethodDescrObject *descr, PyObject *args, PyObject *kwds) methoddescr_call(PyMethodDescrObject *descr, PyObject *args, PyObject *kwargs)
{ {
Py_ssize_t argc; Py_ssize_t nargs;
PyObject *self, *func, *result, **stack; PyObject *self, *result;
/* Make sure that the first argument is acceptable as 'self' */ /* Make sure that the first argument is acceptable as 'self' */
assert(PyTuple_Check(args)); assert(PyTuple_Check(args));
argc = PyTuple_GET_SIZE(args); nargs = PyTuple_GET_SIZE(args);
if (argc < 1) { if (nargs < 1) {
PyErr_Format(PyExc_TypeError, PyErr_Format(PyExc_TypeError,
"descriptor '%V' of '%.100s' " "descriptor '%V' of '%.100s' "
"object needs an argument", "object needs an argument",
@ -257,12 +257,10 @@ methoddescr_call(PyMethodDescrObject *descr, PyObject *args, PyObject *kwds)
return NULL; return NULL;
} }
func = PyCFunction_NewEx(descr->d_method, self, NULL); result = _PyMethodDef_RawFastCallDict(descr->d_method, self,
if (func == NULL) &PyTuple_GET_ITEM(args, 1), nargs - 1,
return NULL; kwargs);
stack = &PyTuple_GET_ITEM(args, 1); result = _Py_CheckFunctionResult((PyObject *)descr, result, NULL);
result = _PyObject_FastCallDict(func, stack, argc - 1, kwds);
Py_DECREF(func);
return result; return result;
} }

View file

@ -165,17 +165,14 @@ PyCFunction_Call(PyObject *func, PyObject *args, PyObject *kwds)
} }
PyObject * PyObject *
_PyCFunction_FastCallDict(PyObject *func_obj, PyObject **args, Py_ssize_t nargs, _PyMethodDef_RawFastCallDict(PyMethodDef *method, PyObject *self, PyObject **args,
PyObject *kwargs) Py_ssize_t nargs, PyObject *kwargs)
{ {
PyCFunctionObject *func;
PyCFunction meth; PyCFunction meth;
PyObject *self;
PyObject *result; PyObject *result;
int flags; int flags;
assert(func_obj != NULL); assert(method != NULL);
assert(PyCFunction_Check(func_obj));
assert(nargs >= 0); assert(nargs >= 0);
assert(nargs == 0 || args != NULL); assert(nargs == 0 || args != NULL);
assert(kwargs == NULL || PyDict_Check(kwargs)); assert(kwargs == NULL || PyDict_Check(kwargs));
@ -185,10 +182,8 @@ _PyCFunction_FastCallDict(PyObject *func_obj, PyObject **args, Py_ssize_t nargs,
caller loses its exception */ caller loses its exception */
assert(!PyErr_Occurred()); assert(!PyErr_Occurred());
func = (PyCFunctionObject*)func_obj; meth = method->ml_meth;
meth = PyCFunction_GET_FUNCTION(func); flags = method->ml_flags & ~(METH_CLASS | METH_STATIC | METH_COEXIST);
self = PyCFunction_GET_SELF(func);
flags = PyCFunction_GET_FLAGS(func) & ~(METH_CLASS | METH_STATIC | METH_COEXIST);
switch (flags) switch (flags)
{ {
@ -199,7 +194,7 @@ _PyCFunction_FastCallDict(PyObject *func_obj, PyObject **args, Py_ssize_t nargs,
if (kwargs != NULL && PyDict_GET_SIZE(kwargs) != 0) { if (kwargs != NULL && PyDict_GET_SIZE(kwargs) != 0) {
PyErr_Format(PyExc_TypeError, "%.200s() takes no keyword arguments", PyErr_Format(PyExc_TypeError, "%.200s() takes no keyword arguments",
func->m_ml->ml_name); method->ml_name);
return NULL; return NULL;
} }
@ -210,7 +205,7 @@ _PyCFunction_FastCallDict(PyObject *func_obj, PyObject **args, Py_ssize_t nargs,
if (nargs != 1) { if (nargs != 1) {
PyErr_Format(PyExc_TypeError, PyErr_Format(PyExc_TypeError,
"%.200s() takes exactly one argument (%zd given)", "%.200s() takes exactly one argument (%zd given)",
func->m_ml->ml_name, nargs); method->ml_name, nargs);
return NULL; return NULL;
} }
@ -253,8 +248,7 @@ _PyCFunction_FastCallDict(PyObject *func_obj, PyObject **args, Py_ssize_t nargs,
PyObject *kwnames; PyObject *kwnames;
_PyCFunctionFast fastmeth = (_PyCFunctionFast)meth; _PyCFunctionFast fastmeth = (_PyCFunctionFast)meth;
if (_PyStack_UnpackDict(args, nargs, kwargs, if (_PyStack_UnpackDict(args, nargs, kwargs, &stack, &kwnames) < 0) {
&stack, &kwnames, func_obj) < 0) {
return NULL; return NULL;
} }
@ -273,17 +267,33 @@ _PyCFunction_FastCallDict(PyObject *func_obj, PyObject **args, Py_ssize_t nargs,
return NULL; return NULL;
} }
result = _Py_CheckFunctionResult(func_obj, result, NULL);
return result; return result;
no_keyword_error: no_keyword_error:
PyErr_Format(PyExc_TypeError, PyErr_Format(PyExc_TypeError,
"%.200s() takes no arguments (%zd given)", "%.200s() takes no arguments (%zd given)",
func->m_ml->ml_name, nargs); method->ml_name, nargs);
return NULL; return NULL;
} }
PyObject *
_PyCFunction_FastCallDict(PyObject *func, PyObject **args, Py_ssize_t nargs,
PyObject *kwargs)
{
PyObject *result;
assert(func != NULL);
assert(PyCFunction_Check(func));
result = _PyMethodDef_RawFastCallDict(((PyCFunctionObject*)func)->m_ml,
PyCFunction_GET_SELF(func),
args, nargs, kwargs);
result = _Py_CheckFunctionResult(func, result, NULL);
return result;
}
PyObject * PyObject *
_PyCFunction_FastCallKeywords(PyObject *func_obj, PyObject **args, _PyCFunction_FastCallKeywords(PyObject *func_obj, PyObject **args,
Py_ssize_t nargs, PyObject *kwnames) Py_ssize_t nargs, PyObject *kwnames)