From: serge-sans-paille Date: Sat, 21 May 2022 13:16:37 +0000 (+0000) Subject: GH-92898: Make _Py_Cast C++ version compatible with cast operator (gh-92951) X-Git-Tag: v3.12.0a1~1471 X-Git-Url: http://git.ipfire.org/gitweb.cgi?a=commitdiff_plain;h=5b71b519f966e1017c868ea2b27c61a5eac38c1f;p=thirdparty%2FPython%2Fcpython.git GH-92898: Make _Py_Cast C++ version compatible with cast operator (gh-92951) --- diff --git a/Include/pyport.h b/Include/pyport.h index 614a2789fb07..086ed4204e4f 100644 --- a/Include/pyport.h +++ b/Include/pyport.h @@ -26,8 +26,32 @@ // _Py_CAST(const PyObject*, expr) fails with a compiler error. #ifdef __cplusplus # define _Py_STATIC_CAST(type, expr) static_cast(expr) -# define _Py_CAST(type, expr) \ - const_cast(reinterpret_cast(expr)) + +extern "C++" { +namespace { +template +inline type _Py_reinterpret_cast_impl(expr_type *expr) { + return reinterpret_cast(expr); +} + +template +inline type _Py_reinterpret_cast_impl(expr_type const *expr) { + return reinterpret_cast(const_cast(expr)); +} + +template +inline type _Py_reinterpret_cast_impl(expr_type &expr) { + return static_cast(expr); +} + +template +inline type _Py_reinterpret_cast_impl(expr_type const &expr) { + return static_cast(const_cast(expr)); +} +} // namespace +} +# define _Py_CAST(type, expr) _Py_reinterpret_cast_impl(expr) + #else # define _Py_STATIC_CAST(type, expr) ((type)(expr)) # define _Py_CAST(type, expr) ((type)(expr)) diff --git a/Lib/test/_testcppext.cpp b/Lib/test/_testcppext.cpp index f38b4870e0ed..f6049eedd000 100644 --- a/Lib/test/_testcppext.cpp +++ b/Lib/test/_testcppext.cpp @@ -40,6 +40,15 @@ test_api_casts(PyObject *Py_UNUSED(module), PyObject *Py_UNUSED(args)) PyTypeObject *type = Py_TYPE(const_obj); assert(Py_REFCNT(const_obj) >= 1); + struct PyObjectProxy { + PyObject* obj; + operator PyObject *() { return obj; } + } proxy_obj = { obj }; + Py_INCREF(proxy_obj); + Py_DECREF(proxy_obj); + assert(Py_REFCNT(proxy_obj) >= 1); + + assert(type == &PyTuple_Type); assert(PyTuple_GET_SIZE(const_obj) == 2); PyObject *one = PyTuple_GET_ITEM(const_obj, 0);