Commit 2dadb472 authored by Stefan Behnel's avatar Stefan Behnel

fix a corner case where float(x) would not call x.__float__() for a float subtype

parent 110e2ae4
...@@ -476,8 +476,16 @@ static double __Pyx__PyObject_AsDouble(PyObject* obj) { ...@@ -476,8 +476,16 @@ static double __Pyx__PyObject_AsDouble(PyObject* obj) {
#if CYTHON_COMPILING_IN_PYPY #if CYTHON_COMPILING_IN_PYPY
float_value = PyNumber_Float(obj); float_value = PyNumber_Float(obj);
#else #else
if (Py_TYPE(obj)->tp_as_number && Py_TYPE(obj)->tp_as_number->nb_float) { PyNumberMethods *nb = Py_TYPE(obj)->tp_as_number;
return PyFloat_AsDouble(obj); if (likely(nb) && likely(nb->nb_float)) {
float_value = nb->nb_float(obj);
if (likely(float_value) && unlikely(!PyFloat_Check(float_value))) {
PyErr_Format(PyExc_TypeError,
"__float__ returned non-float (type %.200s)",
Py_TYPE(float_value)->tp_name);
Py_DECREF(float_value);
goto bad;
}
} else if (PyUnicode_CheckExact(obj) || PyBytes_CheckExact(obj)) { } else if (PyUnicode_CheckExact(obj) || PyBytes_CheckExact(obj)) {
#if PY_MAJOR_VERSION >= 3 #if PY_MAJOR_VERSION >= 3
float_value = PyFloat_FromString(obj); float_value = PyFloat_FromString(obj);
......
...@@ -137,3 +137,37 @@ def double_to_double_int(double x): ...@@ -137,3 +137,37 @@ def double_to_double_int(double x):
""" """
cdef double r = int(x) cdef double r = int(x)
return r return r
class MyFloat(float):
"""
>>> x = MyFloat(1.0)
>>> x
1.0
>>> float(x)
12.0
>>> x.float()
12.0
"""
def __float__(self):
return 12.0
def float(self):
return float(self)
class MyInt(int):
"""
>>> x = MyInt(1)
>>> x
1
>>> int(x)
2
>>> x.int()
2
"""
def __int__(self):
return 2
def int(self):
return int(self)
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment