Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cython
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Xavier Thompson
cython
Commits
3de7a4b8
Commit
3de7a4b8
authored
Apr 22, 2020
by
Stefan Behnel
Committed by
GitHub
Apr 22, 2020
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Tune some internal calls into fastcall/vectorcall (GH-3540)
parent
02773431
Changes
4
Show whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
58 additions
and
49 deletions
+58
-49
Cython/Compiler/ExprNodes.py
Cython/Compiler/ExprNodes.py
+3
-1
Cython/Utility/Coroutine.c
Cython/Utility/Coroutine.c
+6
-3
Cython/Utility/ModuleSetupCode.c
Cython/Utility/ModuleSetupCode.c
+7
-0
Cython/Utility/ObjectHandling.c
Cython/Utility/ObjectHandling.c
+42
-45
No files found.
Cython/Compiler/ExprNodes.py
View file @
3de7a4b8
...
...
@@ -6710,7 +6710,9 @@ class MergedDictNode(ExprNode):
if
item
.
type
is
not
dict_type
:
code
.
putln
(
'} else {'
)
code
.
putln
(
"%s = PyObject_CallFunctionObjArgs((PyObject*)&PyDict_Type, %s, NULL); %s"
%
(
code
.
globalstate
.
use_utility_code
(
UtilityCode
.
load_cached
(
"PyObjectCallOneArg"
,
"ObjectHandling.c"
))
code
.
putln
(
"%s = __Pyx_PyObject_CallOneArg((PyObject*)&PyDict_Type, %s); %s"
%
(
self
.
result
(),
item
.
py_result
(),
code
.
error_goto_if_null
(
self
.
result
(),
self
.
pos
)))
...
...
Cython/Utility/Coroutine.c
View file @
3de7a4b8
...
...
@@ -488,6 +488,8 @@ static int __pyx_Generator_init(void); /*proto*/
//@requires: Exceptions.c::RaiseException
//@requires: Exceptions.c::SaveResetException
//@requires: ObjectHandling.c::PyObjectCallMethod1
//@requires: ObjectHandling.c::PyObjectCallNoArg
//@requires: ObjectHandling.c::PyObjectFastCall
//@requires: ObjectHandling.c::PyObjectGetAttrStr
//@requires: ObjectHandling.c::PyObjectGetAttrStrNoError
//@requires: CommonStructures.c::FetchCommonType
...
...
@@ -905,7 +907,7 @@ static int __Pyx_Coroutine_CloseIter(__pyx_CoroutineObject *gen, PyObject *yf) {
PyErr_WriteUnraisable
(
yf
);
}
}
else
{
retval
=
PyObject_CallFunction
(
meth
,
NULL
);
retval
=
__Pyx_PyObject_CallNoArg
(
meth
);
Py_DECREF
(
meth
);
if
(
unlikely
(
!
retval
))
err
=
-
1
;
...
...
@@ -1057,10 +1059,11 @@ static PyObject *__Pyx__Coroutine_Throw(PyObject *self, PyObject *typ, PyObject
goto
throw_here
;
}
if
(
likely
(
args
))
{
ret
=
PyObject_CallObject
(
meth
,
args
);
ret
=
__Pyx_PyObject_Call
(
meth
,
args
,
NULL
);
}
else
{
// "tb" or even "val" might be NULL, but that also correctly terminates the argument list
ret
=
PyObject_CallFunctionObjArgs
(
meth
,
typ
,
val
,
tb
,
NULL
);
PyObject
*
cargs
[
4
]
=
{
NULL
,
typ
,
val
,
tb
};
ret
=
__Pyx_PyObject_FastCall
(
meth
,
cargs
+
1
,
3
|
__Pyx_PY_VECTORCALL_ARGUMENTS_OFFSET
);
}
Py_DECREF
(
meth
);
}
...
...
Cython/Utility/ModuleSetupCode.c
View file @
3de7a4b8
...
...
@@ -552,9 +552,16 @@ class __Pyx_FakeReference {
#if CYTHON_VECTORCALL
#define __pyx_vectorcallfunc vectorcallfunc
#define __Pyx_PY_VECTORCALL_ARGUMENTS_OFFSET PY_VECTORCALL_ARGUMENTS_OFFSET
#define __Pyx_PyVectorcall_NARGS(n) PyVectorcall_NARGS(n)
#elif CYTHON_BACKPORT_VECTORCALL
typedef
PyObject
*
(
*
__pyx_vectorcallfunc
)(
PyObject
*
callable
,
PyObject
*
const
*
args
,
size_t
nargsf
,
PyObject
*
kwnames
);
#define __Pyx_PY_VECTORCALL_ARGUMENTS_OFFSET ((size_t)1 << (8 * sizeof(size_t) - 1))
#define __Pyx_PyVectorcall_NARGS(n) ((n) & ~__Pyx_PY_VECTORCALL_ARGUMENTS_OFFSET)
#else
#define __Pyx_PY_VECTORCALL_ARGUMENTS_OFFSET 0
#define __Pyx_PyVectorcall_NARGS(n) (n)
#endif
#if CYTHON_COMPILING_IN_PYPY && !defined(PyObject_Malloc)
...
...
Cython/Utility/ObjectHandling.c
View file @
3de7a4b8
...
...
@@ -1023,7 +1023,7 @@ static PyObject *__Pyx_Py3ClassCreate(PyObject *metaclass, PyObject *name, PyObj
//@substitute: naming
//@requires: PyObjectGetAttrStrNoError
//@requires: CalculateMetaclass
//@requires: PyObjectCall
//@requires: PyObject
Fast
Call
//@requires: PyObjectCall2Args
//@requires: PyObjectLookupSpecial
// only in fallback code:
...
...
@@ -1035,14 +1035,9 @@ static PyObject *__Pyx_Py3MetaclassPrepare(PyObject *metaclass, PyObject *bases,
if
(
metaclass
)
{
PyObject
*
prep
=
__Pyx_PyObject_GetAttrStrNoError
(
metaclass
,
PYIDENT
(
"__prepare__"
));
if
(
prep
)
{
PyObject
*
pargs
=
PyTuple_Pack
(
2
,
name
,
bases
);
if
(
unlikely
(
!
pargs
))
{
Py_DECREF
(
prep
);
return
NULL
;
}
ns
=
PyObject_Call
(
prep
,
pargs
,
mkw
);
PyObject
*
pargs
[
3
]
=
{
NULL
,
name
,
bases
};
ns
=
__Pyx_PyObject_FastCallDict
(
prep
,
pargs
+
1
,
2
|
__Pyx_PY_VECTORCALL_ARGUMENTS_OFFSET
,
mkw
);
Py_DECREF
(
prep
);
Py_DECREF
(
pargs
);
}
else
{
if
(
unlikely
(
PyErr_Occurred
()))
return
NULL
;
...
...
@@ -1159,7 +1154,7 @@ static PyObject *__Pyx_InitSubclassPEP487(PyObject *type_obj, PyObject *mkw) {
if
(
unlikely
(
!
res
))
goto
bad
;
meth
=
res
;
}
res
=
__Pyx_PyObject_
Call
(
meth
,
$
empty_tuple
,
mkw
);
res
=
__Pyx_PyObject_
FastCallDict
(
meth
,
NULL
,
0
,
mkw
);
Py_DECREF
(
meth
);
if
(
unlikely
(
!
res
))
goto
bad
;
Py_DECREF
(
res
);
...
...
@@ -1205,7 +1200,7 @@ bad:
Py_CLEAR
(
type_obj
);
goto
done
;
}
res
=
__Pyx_PyObject_
Call
(
func
,
$
empty_tuple
,
mkw
);
res
=
__Pyx_PyObject_
FastCallDict
(
func
,
NULL
,
0
,
mkw
);
Py_DECREF
(
func
);
if
(
unlikely
(
!
res
))
Py_CLEAR
(
type_obj
);
...
...
@@ -1221,8 +1216,9 @@ done:
static
PyObject
*
__Pyx_Py3ClassCreate
(
PyObject
*
metaclass
,
PyObject
*
name
,
PyObject
*
bases
,
PyObject
*
dict
,
PyObject
*
mkw
,
int
calculate_metaclass
,
int
allow_py2_metaclass
)
{
PyObject
*
result
,
*
margs
;
PyObject
*
result
,
*
m
c_kw
args
;
PyObject
*
owned_metaclass
=
NULL
;
PyObject
*
margs
[
4
]
=
{
NULL
,
name
,
bases
,
dict
};
if
(
allow_py2_metaclass
)
{
/* honour Python2 __metaclass__ for backward compatibility */
owned_metaclass
=
PyObject_GetItem
(
dict
,
PYIDENT
(
"__metaclass__"
));
...
...
@@ -1241,17 +1237,12 @@ static PyObject *__Pyx_Py3ClassCreate(PyObject *metaclass, PyObject *name, PyObj
return
NULL
;
owned_metaclass
=
metaclass
;
}
result
=
NULL
;
margs
=
PyTuple_Pack
(
3
,
name
,
bases
,
dict
);
if
(
likely
(
margs
))
{
// Before PEP-487, type(a,b,c) did not accept any keyword arguments, so guard at least against that case.
PyObject
*
mc_kwargs
=
(
PY_VERSION_HEX
>=
0x030600A4
)
?
mkw
:
(
mc_kwargs
=
(
PY_VERSION_HEX
>=
0x030600A4
)
?
mkw
:
(
(
metaclass
==
(
PyObject
*
)
&
PyType_Type
)
?
NULL
:
mkw
);
result
=
__Pyx_PyObject_Call
(
metaclass
,
margs
,
mc_kwargs
);
Py_DECREF
(
margs
);
}
result
=
__Pyx_PyObject_FastCallDict
(
metaclass
,
margs
+
1
,
3
|
__Pyx_PY_VECTORCALL_ARGUMENTS_OFFSET
,
mc_kwargs
);
Py_XDECREF
(
owned_metaclass
);
#if PY_VERSION_HEX < 0x030600A4 && CYTHON_PEP487_INIT_SUBCLASS
if
(
likely
(
result
)
&&
likely
(
PyType_Check
(
result
)))
{
if
(
unlikely
(
__Pyx_SetNamesPEP487
(
result
)
<
0
))
{
...
...
@@ -2080,7 +2071,8 @@ bad:
/////////////// PyObjectFastCall.proto ///////////////
static
CYTHON_INLINE
PyObject
*
__Pyx_PyObject_FastCall
(
PyObject
*
func
,
PyObject
**
args
,
Py_ssize_t
nargs
);
/*proto*/
#define __Pyx_PyObject_FastCall(func, args, nargs) __Pyx_PyObject_FastCallDict(func, args, nargs, NULL)
static
CYTHON_INLINE
PyObject
*
__Pyx_PyObject_FastCallDict
(
PyObject
*
func
,
PyObject
**
args
,
Py_ssize_t
nargs
,
PyObject
*
kwargs
);
/*proto*/
/////////////// PyObjectFastCall ///////////////
//@requires: PyObjectCall
...
...
@@ -2088,7 +2080,7 @@ static CYTHON_INLINE PyObject* __Pyx_PyObject_FastCall(PyObject *func, PyObject
//@requires: PyObjectCallMethO
//@substitute: naming
static
CYTHON_INLINE
PyObject
*
__Pyx_PyObject_FastCall_fallback
(
PyObject
*
func
,
PyObject
**
args
,
Py_ssize_t
n
args
)
{
static
PyObject
*
__Pyx_PyObject_FastCall_fallback
(
PyObject
*
func
,
PyObject
**
args
,
Py_ssize_t
nargs
,
PyObject
*
kw
args
)
{
PyObject
*
argstuple
;
PyObject
*
result
;
Py_ssize_t
i
;
...
...
@@ -2099,18 +2091,19 @@ static CYTHON_INLINE PyObject* __Pyx_PyObject_FastCall_fallback(PyObject *func,
Py_INCREF
(
args
[
i
]);
PyTuple_SET_ITEM
(
argstuple
,
i
,
args
[
i
]);
}
result
=
__Pyx_PyObject_Call
(
func
,
argstuple
,
NULL
);
result
=
__Pyx_PyObject_Call
(
func
,
argstuple
,
kwargs
);
Py_DECREF
(
argstuple
);
return
result
;
}
static
CYTHON_INLINE
PyObject
*
__Pyx_PyObject_FastCall
(
PyObject
*
func
,
PyObject
**
args
,
Py_ssize_t
n
args
)
{
static
CYTHON_INLINE
PyObject
*
__Pyx_PyObject_FastCall
Dict
(
PyObject
*
func
,
PyObject
**
args
,
Py_ssize_t
_nargs
,
PyObject
*
kw
args
)
{
// Special fast paths for 0 and 1 arguments
// NOTE: in many cases, this is called with a constant value for nargs
// which is known at compile-time. So the branches below will typically
// be optimized away.
Py_ssize_t
nargs
=
__Pyx_PyVectorcall_NARGS
(
_nargs
);
#if CYTHON_COMPILING_IN_CPYTHON
if
(
nargs
==
0
)
{
if
(
nargs
==
0
&&
kwargs
==
NULL
)
{
#ifdef __Pyx_CyFunction_USED
if
(
PyCFunction_Check
(
func
)
||
__Pyx_CyFunction_Check
(
func
))
#else
...
...
@@ -2122,7 +2115,7 @@ static CYTHON_INLINE PyObject* __Pyx_PyObject_FastCall(PyObject *func, PyObject
}
}
}
else
if
(
nargs
==
1
)
{
else
if
(
nargs
==
1
&&
kwargs
==
NULL
)
{
if
(
PyCFunction_Check
(
func
))
{
if
(
likely
(
PyCFunction_GET_FLAGS
(
func
)
&
METH_O
))
{
...
...
@@ -2133,21 +2126,23 @@ static CYTHON_INLINE PyObject* __Pyx_PyObject_FastCall(PyObject *func, PyObject
#endif
#if PY_VERSION_HEX < 0x030800B1
#if CYTHON_FAST_PYCCALL
&& PY_VERSION_HEX >= 0x030700A1
#if CYTHON_FAST_PYCCALL
if
(
PyCFunction_Check
(
func
))
{
if
(
kwargs
)
{
return
_PyCFunction_FastCallDict
(
func
,
args
,
nargs
,
kwargs
);
}
else
{
return
_PyCFunction_FastCallKeywords
(
func
,
args
,
nargs
,
NULL
);
}
if
(
__Pyx_IS_TYPE
(
func
,
&
PyMethodDescr_Type
))
{
return
_PyMethodDescr_FastCallKeywords
(
func
,
args
,
nargs
,
NULL
);
}
#
elif CYTHON_FAST_PYCCALL
if
(
PyCFunction_Check
(
func
))
{
return
_Py
CFunction_FastCallDict
(
func
,
args
,
nargs
,
NULL
);
#
if PY_VERSION_HEX >= 0x030700A1
if
(
!
kwargs
&&
__Pyx_IS_TYPE
(
func
,
&
PyMethodDescr_Type
))
{
return
_Py
MethodDescr_FastCallKeywords
(
func
,
args
,
nargs
,
NULL
);
}
#endif
#endif
#if CYTHON_FAST_PYCALL
if
(
PyFunction_Check
(
func
))
{
return
__Pyx_PyFunction_FastCall
(
func
,
args
,
n
args
);
return
__Pyx_PyFunction_FastCall
Dict
(
func
,
args
,
nargs
,
kw
args
);
}
#endif
#endif
...
...
@@ -2155,20 +2150,20 @@ static CYTHON_INLINE PyObject* __Pyx_PyObject_FastCall(PyObject *func, PyObject
#if CYTHON_VECTORCALL
vectorcallfunc
f
=
_PyVectorcall_Function
(
func
);
if
(
f
)
{
return
f
(
func
,
args
,
nargs
,
NULL
);
return
f
(
func
,
args
,
nargs
,
kwargs
);
}
#elif __Pyx_CyFunction_USED && CYTHON_BACKPORT_VECTORCALL
// exclude fused functions for now
if
(
__Pyx_IS_TYPE
(
func
,
__pyx_CyFunctionType
))
{
__pyx_vectorcallfunc
f
=
__Pyx_CyFunction_func_vectorcall
(
func
);
if
(
f
)
return
f
(
func
,
args
,
nargs
,
NULL
);
if
(
f
)
return
f
(
func
,
args
,
nargs
,
kwargs
);
}
#endif
if
(
nargs
==
0
)
{
return
__Pyx_PyObject_Call
(
func
,
$
empty_tuple
,
NULL
);
return
__Pyx_PyObject_Call
(
func
,
$
empty_tuple
,
kwargs
);
}
return
__Pyx_PyObject_FastCall_fallback
(
func
,
args
,
nargs
);
return
__Pyx_PyObject_FastCall_fallback
(
func
,
args
,
nargs
,
kwargs
);
}
...
...
@@ -2493,14 +2488,14 @@ done:
/////////////// PyObjectCall2Args.proto ///////////////
static
CYTHON_
UNUSED
PyObject
*
__Pyx_PyObject_Call2Args
(
PyObject
*
function
,
PyObject
*
arg1
,
PyObject
*
arg2
);
/*proto*/
static
CYTHON_
INLINE
PyObject
*
__Pyx_PyObject_Call2Args
(
PyObject
*
function
,
PyObject
*
arg1
,
PyObject
*
arg2
);
/*proto*/
/////////////// PyObjectCall2Args ///////////////
//@requires: PyObjectFastCall
static
CYTHON_
UNUSED
PyObject
*
__Pyx_PyObject_Call2Args
(
PyObject
*
function
,
PyObject
*
arg1
,
PyObject
*
arg2
)
{
PyObject
*
args
[
2
]
=
{
arg1
,
arg2
};
return
__Pyx_PyObject_FastCall
(
function
,
args
,
2
);
static
CYTHON_
INLINE
PyObject
*
__Pyx_PyObject_Call2Args
(
PyObject
*
function
,
PyObject
*
arg1
,
PyObject
*
arg2
)
{
PyObject
*
args
[
3
]
=
{
NULL
,
arg1
,
arg2
};
return
__Pyx_PyObject_FastCall
(
function
,
args
+
1
,
2
|
__Pyx_PY_VECTORCALL_ARGUMENTS_OFFSET
);
}
...
...
@@ -2512,7 +2507,8 @@ static CYTHON_INLINE PyObject* __Pyx_PyObject_CallOneArg(PyObject *func, PyObjec
//@requires: PyObjectFastCall
static
CYTHON_INLINE
PyObject
*
__Pyx_PyObject_CallOneArg
(
PyObject
*
func
,
PyObject
*
arg
)
{
return
__Pyx_PyObject_FastCall
(
func
,
&
arg
,
1
);
PyObject
*
args
[
2
]
=
{
NULL
,
arg
};
return
__Pyx_PyObject_FastCall
(
func
,
args
+
1
,
1
|
__Pyx_PY_VECTORCALL_ARGUMENTS_OFFSET
);
}
...
...
@@ -2524,7 +2520,8 @@ static CYTHON_INLINE PyObject* __Pyx_PyObject_CallNoArg(PyObject *func); /*proto
//@requires: PyObjectFastCall
static
CYTHON_INLINE
PyObject
*
__Pyx_PyObject_CallNoArg
(
PyObject
*
func
)
{
return
__Pyx_PyObject_FastCall
(
func
,
NULL
,
0
);
PyObject
*
arg
=
NULL
;
return
__Pyx_PyObject_FastCall
(
func
,
(
&
arg
)
+
1
,
0
|
__Pyx_PY_VECTORCALL_ARGUMENTS_OFFSET
);
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment