Rich comparisons. Refactored internal routine do_cmp() and added APIs
PyObject_RichCompare() and PyObject_RichCompareBool(). XXX Note: the code that checks for deeply nested rich comparisons is bogus -- it assumes the two objects are always identical, rather than using the same logic as PyObject_Compare(). I'll fix that later.
This commit is contained in:
parent
00e0f21be8
commit
e797ec1cb8
375
Objects/object.c
375
Objects/object.c
@ -308,96 +308,226 @@ PyObject_Str(PyObject *v)
|
|||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define NEW_STYLE_NUMBER(o) PyType_HasFeature((o)->ob_type, \
|
/* Map rich comparison operators to their swapped version, e.g. LT --> GT */
|
||||||
Py_TPFLAGS_NEWSTYLENUMBER)
|
static int swapped_op[] = {Py_GT, Py_GE, Py_EQ, Py_NE, Py_LT, Py_LE};
|
||||||
|
|
||||||
|
/* Try a genuine rich comparison, returning an object. Return:
|
||||||
|
NULL for exception;
|
||||||
|
NotImplemented if this particular rich comparison is not implemented or
|
||||||
|
undefined;
|
||||||
|
some object not equal to NotImplemented if it is implemented
|
||||||
|
(this latter object may not be a Boolean).
|
||||||
|
*/
|
||||||
|
static PyObject *
|
||||||
|
try_rich_compare(PyObject *v, PyObject *w, int op)
|
||||||
|
{
|
||||||
|
richcmpfunc f;
|
||||||
|
PyObject *res;
|
||||||
|
|
||||||
|
if ((f = v->ob_type->tp_richcompare) != NULL) {
|
||||||
|
res = (*f)(v, w, op);
|
||||||
|
if (res != Py_NotImplemented)
|
||||||
|
return res;
|
||||||
|
Py_DECREF(res);
|
||||||
|
}
|
||||||
|
if ((f = w->ob_type->tp_richcompare) != NULL) {
|
||||||
|
return (*f)(w, v, swapped_op[op]);
|
||||||
|
}
|
||||||
|
res = Py_NotImplemented;
|
||||||
|
Py_INCREF(res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Try a genuine rich comparison, returning an int. Return:
|
||||||
|
-1 for exception (including the case where try_rich_compare() returns an
|
||||||
|
object that's not a Boolean);
|
||||||
|
0 if the outcome is false;
|
||||||
|
1 if the outcome is true;
|
||||||
|
2 if this particular rich comparison is not implemented or undefined.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
cmp_to_int(PyObject *result)
|
try_rich_compare_bool(PyObject *v, PyObject *w, int op)
|
||||||
|
{
|
||||||
|
PyObject *res;
|
||||||
|
int ok;
|
||||||
|
|
||||||
|
if (v->ob_type->tp_richcompare == NULL &&
|
||||||
|
w->ob_type->tp_richcompare == NULL)
|
||||||
|
return 2; /* Shortcut, avoid INCREF+DECREF */
|
||||||
|
res = try_rich_compare(v, w, op);
|
||||||
|
if (res == NULL)
|
||||||
|
return -1;
|
||||||
|
if (res == Py_NotImplemented) {
|
||||||
|
Py_DECREF(res);
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
ok = PyObject_IsTrue(res);
|
||||||
|
Py_DECREF(res);
|
||||||
|
return ok;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Try rich comparisons to determine a 3-way comparison. Return:
|
||||||
|
-2 for an exception;
|
||||||
|
-1 if v < w;
|
||||||
|
0 if v == w;
|
||||||
|
1 if v > w;
|
||||||
|
2 if this particular rich comparison is not implemented or undefined.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
try_rich_to_3way_compare(PyObject *v, PyObject *w)
|
||||||
|
{
|
||||||
|
if (v->ob_type->tp_richcompare == NULL &&
|
||||||
|
w->ob_type->tp_richcompare == NULL)
|
||||||
|
return 2; /* Shortcut */
|
||||||
|
switch (try_rich_compare_bool(v, w, Py_LT)) {
|
||||||
|
case -1: /* Error */
|
||||||
|
return -1;
|
||||||
|
case 0: /* False: not less */
|
||||||
|
break;
|
||||||
|
case 1: /* True: less */
|
||||||
|
return -1;
|
||||||
|
case 2: /* NotImplemented */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
switch (try_rich_compare_bool(v, w, Py_GT)) {
|
||||||
|
case -1: /* Error */
|
||||||
|
return -1;
|
||||||
|
case 0: /* False: not greater */
|
||||||
|
break;
|
||||||
|
case 1: /* True: greater */
|
||||||
|
return 1;
|
||||||
|
case 2: /* NotImplemented */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
switch (try_rich_compare_bool(v, w, Py_EQ)) {
|
||||||
|
case -1: /* Error */
|
||||||
|
return -1;
|
||||||
|
case 0: /* False: not equal */
|
||||||
|
break;
|
||||||
|
case 1: /* True: equal */
|
||||||
|
return 0;
|
||||||
|
case 2: /* NotImplemented */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return 2; /* XXX Even if all three returned FALSE?! */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Try a 3-way comparison, returning an int. Return:
|
||||||
|
-2 for an exception;
|
||||||
|
-1 if v < w;
|
||||||
|
0 if v == w;
|
||||||
|
1 if v > w;
|
||||||
|
2 if this particular 3-way comparison is not implemented or undefined.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
try_3way_compare(PyObject *v, PyObject *w)
|
||||||
{
|
{
|
||||||
int c;
|
int c;
|
||||||
if (result == NULL)
|
cmpfunc f;
|
||||||
return -1;
|
|
||||||
if (!PyInt_Check(result)) {
|
/* Comparisons involving instances are given to instance_compare,
|
||||||
PyErr_SetString(PyExc_TypeError,
|
which has the same return conventions as this function. */
|
||||||
"comparison did not return an int");
|
|
||||||
return -1;
|
if (PyInstance_Check(v))
|
||||||
|
return (*v->ob_type->tp_compare)(v, w);
|
||||||
|
if (PyInstance_Check(w))
|
||||||
|
return (*w->ob_type->tp_compare)(v, w);
|
||||||
|
|
||||||
|
/* If the types are equal, don't bother with coercions etc. */
|
||||||
|
if (v->ob_type == w->ob_type) {
|
||||||
|
if ((f = v->ob_type->tp_compare) == NULL)
|
||||||
|
return 2;
|
||||||
|
c = (*f)(v, w);
|
||||||
|
if (PyErr_Occurred())
|
||||||
|
return -2;
|
||||||
|
return c < 0 ? -1 : c > 0 ? 1 : 0;
|
||||||
}
|
}
|
||||||
c = PyInt_AS_LONG(result);
|
|
||||||
Py_DECREF(result);
|
/* Try coercion; if it fails, give up */
|
||||||
|
c = PyNumber_CoerceEx(&v, &w);
|
||||||
|
if (c < 0)
|
||||||
|
return -2;
|
||||||
|
if (c > 0)
|
||||||
|
return 2;
|
||||||
|
|
||||||
|
/* Try v's comparison, if defined */
|
||||||
|
if ((f = v->ob_type->tp_compare) != NULL) {
|
||||||
|
c = (*f)(v, w);
|
||||||
|
Py_DECREF(v);
|
||||||
|
Py_DECREF(w);
|
||||||
|
if (PyErr_Occurred())
|
||||||
|
return -2;
|
||||||
|
return c < 0 ? -1 : c > 0 ? 1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Try w's comparison, if defined */
|
||||||
|
if ((f = w->ob_type->tp_compare) != NULL) {
|
||||||
|
c = (*f)(w, v); /* swapped! */
|
||||||
|
Py_DECREF(v);
|
||||||
|
Py_DECREF(w);
|
||||||
|
if (PyErr_Occurred())
|
||||||
|
return -2;
|
||||||
|
return c < 0 ? 1 : c > 0 ? -1 : 0; /* negated! */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* No comparison defined */
|
||||||
|
Py_DECREF(v);
|
||||||
|
Py_DECREF(w);
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Final fallback 3-way comparison, returning an int. Return:
|
||||||
|
-2 if an error occurred;
|
||||||
|
-1 if v < w;
|
||||||
|
0 if v == w;
|
||||||
|
1 if v > w.
|
||||||
|
*/
|
||||||
|
static int
|
||||||
|
default_3way_compare(PyObject *v, PyObject *w)
|
||||||
|
{
|
||||||
|
int c;
|
||||||
|
|
||||||
|
if (v->ob_type == w->ob_type) {
|
||||||
|
/* same type: compare pointers */
|
||||||
|
void *vv = v;
|
||||||
|
void *ww = w;
|
||||||
|
return (vv < ww) ? -1 : (vv > ww) ? 1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Special case for Unicode */
|
||||||
|
if (PyUnicode_Check(v) || PyUnicode_Check(w)) {
|
||||||
|
c = PyUnicode_Compare(v, w);
|
||||||
|
if (!PyErr_Occurred())
|
||||||
|
return c;
|
||||||
|
/* TypeErrors are ignored: if Unicode coercion fails due
|
||||||
|
to one of the arguments not having the right type, we
|
||||||
|
continue as defined by the coercion protocol (see
|
||||||
|
above). Luckily, decoding errors are reported as
|
||||||
|
ValueErrors and are not masked by this technique. */
|
||||||
|
if (!PyErr_ExceptionMatches(PyExc_TypeError))
|
||||||
|
return -2;
|
||||||
|
PyErr_Clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* different type: compare type names */
|
||||||
|
c = strcmp(v->ob_type->tp_name, w->ob_type->tp_name);
|
||||||
return (c < 0) ? -1 : (c > 0) ? 1 : 0;
|
return (c < 0) ? -1 : (c > 0) ? 1 : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define CHECK_TYPES(o) PyType_HasFeature((o)->ob_type, Py_TPFLAGS_CHECKTYPES)
|
||||||
|
|
||||||
static int
|
static int
|
||||||
do_cmp(PyObject *v, PyObject *w)
|
do_cmp(PyObject *v, PyObject *w)
|
||||||
{
|
{
|
||||||
PyNumberMethods *mv, *mw;
|
|
||||||
PyObject *x;
|
|
||||||
int c;
|
int c;
|
||||||
|
|
||||||
/* new style nb_cmp gets priority */
|
c = try_rich_to_3way_compare(v, w);
|
||||||
mv = v->ob_type->tp_as_number;
|
if (c < 2)
|
||||||
if (mv != NULL && NEW_STYLE_NUMBER(v) && mv->nb_cmp) {
|
|
||||||
x = (*mv->nb_cmp)(v, w);
|
|
||||||
if (x != Py_NotImplemented)
|
|
||||||
return cmp_to_int(x);
|
|
||||||
Py_DECREF(x);
|
|
||||||
}
|
|
||||||
mw = w->ob_type->tp_as_number;
|
|
||||||
if (mw != NULL && NEW_STYLE_NUMBER(w) && mw->nb_cmp) {
|
|
||||||
x = (*mw->nb_cmp)(v, w);
|
|
||||||
if (x != Py_NotImplemented)
|
|
||||||
return cmp_to_int(x);
|
|
||||||
Py_DECREF(x);
|
|
||||||
}
|
|
||||||
/* fall back to tp_compare */
|
|
||||||
if (v->ob_type == w->ob_type) {
|
|
||||||
if (v->ob_type->tp_compare != NULL) {
|
|
||||||
return (*v->ob_type->tp_compare)(v, w);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
Py_uintptr_t iv = (Py_uintptr_t)v;
|
|
||||||
Py_uintptr_t iw = (Py_uintptr_t)w;
|
|
||||||
return (iv < iw) ? -1 : (iv > iw) ? 1 : 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (PyUnicode_Check(v) || PyUnicode_Check(w)) {
|
|
||||||
c = PyUnicode_Compare(v, w);
|
|
||||||
if (c == -1 &&
|
|
||||||
PyErr_Occurred() &&
|
|
||||||
PyErr_ExceptionMatches(PyExc_TypeError))
|
|
||||||
/* TypeErrors are ignored: if Unicode coercion
|
|
||||||
fails due to one of the arguments not having
|
|
||||||
the right type, we continue as defined by the
|
|
||||||
coercion protocol (see above). Luckily,
|
|
||||||
decoding errors are reported as ValueErrors and
|
|
||||||
are not masked by this technique. */
|
|
||||||
PyErr_Clear();
|
|
||||||
else
|
|
||||||
return c;
|
return c;
|
||||||
}
|
c = try_3way_compare(v, w);
|
||||||
/* fall back to coercion */
|
if (c < 2)
|
||||||
if (mv && mw && (!NEW_STYLE_NUMBER(v) || !NEW_STYLE_NUMBER(w))) {
|
|
||||||
/* old style operand, both operations numeric, coerce */
|
|
||||||
int err = PyNumber_CoerceEx(&v, &w);
|
|
||||||
if (err < 0)
|
|
||||||
return -1;
|
|
||||||
if (err == 0) {
|
|
||||||
if (v->ob_type->tp_compare) {
|
|
||||||
c = (*v->ob_type->tp_compare)(v, w);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
Py_uintptr_t iv = (Py_uintptr_t)v;
|
|
||||||
Py_uintptr_t iw = (Py_uintptr_t)w;
|
|
||||||
c = (iv < iw) ? -1 : (iv > iw) ? 1 : 0;
|
|
||||||
}
|
|
||||||
Py_DECREF(v);
|
|
||||||
Py_DECREF(w);
|
|
||||||
return c;
|
return c;
|
||||||
}
|
return default_3way_compare(v, w);
|
||||||
}
|
|
||||||
/* last resort, use type names */
|
|
||||||
c = strcmp(v->ob_type->tp_name, w->ob_type->tp_name);
|
|
||||||
return (c < 0) ? -1: (c > 0) ? 1 : 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PyObject *_PyCompareState_Key;
|
PyObject *_PyCompareState_Key;
|
||||||
@ -514,9 +644,92 @@ PyObject_Compare(PyObject *v, PyObject *w)
|
|||||||
}
|
}
|
||||||
exit_cmp:
|
exit_cmp:
|
||||||
_PyCompareState_nesting--;
|
_PyCompareState_nesting--;
|
||||||
|
return result < 0 ? -1 : result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static PyObject *
|
||||||
|
try_3way_to_rich_compare(PyObject *v, PyObject *w, int op)
|
||||||
|
{
|
||||||
|
int c;
|
||||||
|
PyObject *result;
|
||||||
|
|
||||||
|
c = try_3way_compare(v, w);
|
||||||
|
if (c <= -2)
|
||||||
|
return NULL;
|
||||||
|
if (c >= 2)
|
||||||
|
c = default_3way_compare(v, w);
|
||||||
|
switch (op) {
|
||||||
|
case Py_LT: c = c < 0; break;
|
||||||
|
case Py_LE: c = c <= 0; break;
|
||||||
|
case Py_EQ: c = c == 0; break;
|
||||||
|
case Py_NE: c = c != 0; break;
|
||||||
|
case Py_GT: c = c > 0; break;
|
||||||
|
case Py_GE: c = c >= 0; break;
|
||||||
|
}
|
||||||
|
result = c ? Py_True : Py_False;
|
||||||
|
Py_INCREF(result);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PyObject *
|
||||||
|
do_richcmp(PyObject *v, PyObject *w, int op)
|
||||||
|
{
|
||||||
|
PyObject *res;
|
||||||
|
|
||||||
|
res = try_rich_compare(v, w, op);
|
||||||
|
if (res != Py_NotImplemented)
|
||||||
|
return res;
|
||||||
|
Py_DECREF(res);
|
||||||
|
|
||||||
|
return try_3way_to_rich_compare(v, w, op);
|
||||||
|
}
|
||||||
|
|
||||||
|
PyObject *
|
||||||
|
PyObject_RichCompare(PyObject *v, PyObject *w, int op)
|
||||||
|
{
|
||||||
|
PyObject *res;
|
||||||
|
|
||||||
|
assert(Py_LT <= op && op <= Py_GE);
|
||||||
|
|
||||||
|
if (_PyCompareState_nesting > NESTING_LIMIT) {
|
||||||
|
/* Too deeply nested -- assume equal */
|
||||||
|
/* XXX This is an unfair shortcut!
|
||||||
|
Should use the same logic as PyObject_Compare. */
|
||||||
|
switch (op) {
|
||||||
|
case Py_LT:
|
||||||
|
case Py_NE:
|
||||||
|
case Py_GT:
|
||||||
|
res = Py_False;
|
||||||
|
break;
|
||||||
|
case Py_LE:
|
||||||
|
case Py_EQ:
|
||||||
|
case Py_GE:
|
||||||
|
res = Py_True;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
Py_INCREF(res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
_PyCompareState_nesting++;
|
||||||
|
res = do_richcmp(v, w, op);
|
||||||
|
_PyCompareState_nesting--;
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
PyObject_RichCompareBool(PyObject *v, PyObject *w, int op)
|
||||||
|
{
|
||||||
|
PyObject *res = PyObject_RichCompare(v, w, op);
|
||||||
|
int ok;
|
||||||
|
|
||||||
|
if (res == NULL)
|
||||||
|
return -1;
|
||||||
|
ok = PyObject_IsTrue(res);
|
||||||
|
Py_DECREF(res);
|
||||||
|
return ok;
|
||||||
|
}
|
||||||
|
|
||||||
/* Set of hash utility functions to help maintaining the invariant that
|
/* Set of hash utility functions to help maintaining the invariant that
|
||||||
iff a==b then hash(a)==hash(b)
|
iff a==b then hash(a)==hash(b)
|
||||||
@ -804,10 +1017,11 @@ PyObject_Not(PyObject *v)
|
|||||||
|
|
||||||
/* Coerce two numeric types to the "larger" one.
|
/* Coerce two numeric types to the "larger" one.
|
||||||
Increment the reference count on each argument.
|
Increment the reference count on each argument.
|
||||||
Return -1 and raise an exception if no coercion is possible
|
Return value:
|
||||||
(and then no reference count is incremented).
|
-1 if an error occurred;
|
||||||
|
0 if the coercion succeeded (and then the reference counts are increased);
|
||||||
|
1 if no coercion is possible (and no error is raised).
|
||||||
*/
|
*/
|
||||||
|
|
||||||
int
|
int
|
||||||
PyNumber_CoerceEx(PyObject **pv, PyObject **pw)
|
PyNumber_CoerceEx(PyObject **pv, PyObject **pw)
|
||||||
{
|
{
|
||||||
@ -833,6 +1047,11 @@ PyNumber_CoerceEx(PyObject **pv, PyObject **pw)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Coerce two numeric types to the "larger" one.
|
||||||
|
Increment the reference count on each argument.
|
||||||
|
Return -1 and raise an exception if no coercion is possible
|
||||||
|
(and then no reference count is incremented).
|
||||||
|
*/
|
||||||
int
|
int
|
||||||
PyNumber_Coerce(PyObject **pv, PyObject **pw)
|
PyNumber_Coerce(PyObject **pv, PyObject **pw)
|
||||||
{
|
{
|
||||||
|
Loading…
x
Reference in New Issue
Block a user