◐ Shell
clean mode source ↗

Specialize Py_DECREF() for Py_REF_DEBUG by vstinner · Pull Request #16781 · python/cpython

@vstinner

Simplify Py_DECREF() code for release mode: don't pass FILE and
LINE to the static inline function.

Simplify Py_DECREF() code for release mode: don't pass __FILE__ and
__LINE__ to the static inline function.

@vstinner

In Python 3.7, Py_DECREF() was a macro. I converted it to a static inline function in PR #10079 of https://bugs.python.org/issue35059 The new static inline function always require filename and lineno arguments, even in release mode. I'm not sure that all compilers are smart enough to remove the arguments, so I prefer to help compilers to ensure that the emited machine code is as efficient as Python 3.7.

@serhiy-storchaka

Why so complicated? Why not use just macros?

@vstinner

shihai1991

_Py_NegativeRefcount(filename, lineno, op);
}
#endif
_Py_RefTotal--;

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So this _Py_DEC_REFTOTAL would be converted in your plan?

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

_Py_DEC_REFTOTAL = _Py_RefTotal-- when Py_REF_DEBUG is defined. I prefer to put directly the real code, it may help debugging. It avoids the indirection of the preprocessor.

@ghost

I'm not sure that all compilers are smart enough to remove the arguments

I tested almost all versions of gcc/clang/msvc/icc:
https://godbolt.org/z/w-N-Kt

All versions have the same behavior:
If with -O1 option, compilers are smart enough to remove the arguments.
If without any option, the extra machine code was emited.

@vstinner

Why not use just macros?

Static inline functions have multiple benefits over macros. Let me give you one example.

Today we get a bug report on Fedora on Python 3.8 which uses static inline functions rather than macros. Thanks to that, the line number in the gdb traceback was more accurate and so more usefull. (Sadly, the bug report is private because it's an automated crash report and it might contain sensitive information, so I cannot share it.)

Yhg1s

#endif
_Py_RefTotal--;
if (--op->ob_refcnt == 0) {
_Py_Dealloc(op);

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This code dupe irks me. Can't you make the REF_DEBUG version call the regular version instead?

@vstinner

This code dupe irks me. Can't you make the REF_DEBUG version call the regular version instead?

_Py_DECREF() function cannot use __FILE__ and __LINE__ preprocessor magic macros. Only a macro can use them.

I wrote PR #17870 to avoid duplicated code and avoid passing __FILE__ and __LINE__ in release mode (when Py_REF_DEBUG is not set).

@vstinner