mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-08-07 02:10:27 +00:00
Optimize methoddescr_call
refer python/cpython@2a1b676d1f refer python/cpython@c52572319c refer python/cpython@35ecebe165 refer python/cpython@8128d5a491
This commit is contained in:
parent
73ea0ea4a6
commit
9220b67b3e
6 changed files with 51 additions and 36 deletions
3
third_party/python/Include/abstract.h
vendored
3
third_party/python/Include/abstract.h
vendored
|
@ -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);
|
||||||
|
|
6
third_party/python/Include/methodobject.h
vendored
6
third_party/python/Include/methodobject.h
vendored
|
@ -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);
|
||||||
|
|
12
third_party/python/Modules/socketmodule.c
vendored
12
third_party/python/Modules/socketmodule.c
vendored
|
@ -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);
|
||||||
|
|
2
third_party/python/Objects/abstract.c
vendored
2
third_party/python/Objects/abstract.c
vendored
|
@ -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;
|
||||||
|
|
20
third_party/python/Objects/descrobject.c
vendored
20
third_party/python/Objects/descrobject.c
vendored
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
44
third_party/python/Objects/methodobject.c
vendored
44
third_party/python/Objects/methodobject.c
vendored
|
@ -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)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue