py: Expand type equality flags to 3 separate ones, fix bool/namedtuple.
Both bool and namedtuple will check against other types for equality; int,
float and complex for bool, and tuple for namedtuple. So to make them work
after the recent commit 3aab54bf43
they would
need MP_TYPE_FLAG_NEEDS_FULL_EQ_TEST set. But that makes all bool and
namedtuple equality checks less efficient because mp_obj_equal_not_equal()
could no longer short-cut x==x, and would need to try __ne__. To improve
this, this commit splits the MP_TYPE_FLAG_NEEDS_FULL_EQ_TEST flags into 3
separate flags to give types more fine-grained control over how their
equality behaves. These new flags are then used to fix bool and namedtuple
equality.
Fixes issue #5615 and #5620.
This commit is contained in:
parent
0852acfc74
commit
9ec1caf42e
6
py/obj.c
6
py/obj.c
|
@ -209,7 +209,7 @@ mp_obj_t mp_obj_equal_not_equal(mp_binary_op_t op, mp_obj_t o1, mp_obj_t o2) {
|
||||||
|
|
||||||
// Shortcut for very common cases
|
// Shortcut for very common cases
|
||||||
if (o1 == o2 &&
|
if (o1 == o2 &&
|
||||||
(mp_obj_is_small_int(o1) || !(mp_obj_get_type(o1)->flags & MP_TYPE_FLAG_NEEDS_FULL_EQ_TEST))) {
|
(mp_obj_is_small_int(o1) || !(mp_obj_get_type(o1)->flags & MP_TYPE_FLAG_EQ_NOT_REFLEXIVE))) {
|
||||||
return local_true;
|
return local_true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -250,10 +250,10 @@ mp_obj_t mp_obj_equal_not_equal(mp_binary_op_t op, mp_obj_t o1, mp_obj_t o2) {
|
||||||
// If a full equality test is not needed and the other object is a different
|
// If a full equality test is not needed and the other object is a different
|
||||||
// type then we don't need to bother trying the comparison.
|
// type then we don't need to bother trying the comparison.
|
||||||
if (type->binary_op != NULL &&
|
if (type->binary_op != NULL &&
|
||||||
((type->flags & MP_TYPE_FLAG_NEEDS_FULL_EQ_TEST) || mp_obj_get_type(o2) == type)) {
|
((type->flags & MP_TYPE_FLAG_EQ_CHECKS_OTHER_TYPE) || mp_obj_get_type(o2) == type)) {
|
||||||
// CPython is asymmetric: it will try __eq__ if there's no __ne__ but not the
|
// CPython is asymmetric: it will try __eq__ if there's no __ne__ but not the
|
||||||
// other way around. If the class doesn't need a full test we can skip __ne__.
|
// other way around. If the class doesn't need a full test we can skip __ne__.
|
||||||
if (op == MP_BINARY_OP_NOT_EQUAL && (type->flags & MP_TYPE_FLAG_NEEDS_FULL_EQ_TEST)) {
|
if (op == MP_BINARY_OP_NOT_EQUAL && (type->flags & MP_TYPE_FLAG_EQ_HAS_NEQ_TEST)) {
|
||||||
mp_obj_t r = type->binary_op(MP_BINARY_OP_NOT_EQUAL, o1, o2);
|
mp_obj_t r = type->binary_op(MP_BINARY_OP_NOT_EQUAL, o1, o2);
|
||||||
if (r != MP_OBJ_NULL) {
|
if (r != MP_OBJ_NULL) {
|
||||||
return r;
|
return r;
|
||||||
|
|
15
py/obj.h
15
py/obj.h
|
@ -445,14 +445,17 @@ typedef mp_obj_t (*mp_fun_var_t)(size_t n, const mp_obj_t *);
|
||||||
typedef mp_obj_t (*mp_fun_kw_t)(size_t n, const mp_obj_t *, mp_map_t *);
|
typedef mp_obj_t (*mp_fun_kw_t)(size_t n, const mp_obj_t *, mp_map_t *);
|
||||||
|
|
||||||
// Flags for type behaviour (mp_obj_type_t.flags)
|
// Flags for type behaviour (mp_obj_type_t.flags)
|
||||||
// If MP_TYPE_FLAG_NEEDS_FULL_EQ_TEST is clear then all the following hold:
|
// If MP_TYPE_FLAG_EQ_NOT_REFLEXIVE is clear then __eq__ is reflexive (A==A returns True).
|
||||||
// (a) the type only implements the __eq__ operator and not the __ne__ operator;
|
// If MP_TYPE_FLAG_EQ_CHECKS_OTHER_TYPE is clear then the type can't be equal to an
|
||||||
// (b) __eq__ returns a boolean result (False or True);
|
// instance of any different class that also clears this flag. If this flag is set
|
||||||
// (c) __eq__ is reflexive (A==A is True);
|
// then the type may check for equality against a different type.
|
||||||
// (d) the type can't be equal to an instance of any different class that also clears this flag.
|
// If MP_TYPE_FLAG_EQ_HAS_NEQ_TEST is clear then the type only implements the __eq__
|
||||||
|
// operator and not the __ne__ operator. If it's set then __ne__ may be implemented.
|
||||||
#define MP_TYPE_FLAG_IS_SUBCLASSED (0x0001)
|
#define MP_TYPE_FLAG_IS_SUBCLASSED (0x0001)
|
||||||
#define MP_TYPE_FLAG_HAS_SPECIAL_ACCESSORS (0x0002)
|
#define MP_TYPE_FLAG_HAS_SPECIAL_ACCESSORS (0x0002)
|
||||||
#define MP_TYPE_FLAG_NEEDS_FULL_EQ_TEST (0x0004)
|
#define MP_TYPE_FLAG_EQ_NOT_REFLEXIVE (0x0040)
|
||||||
|
#define MP_TYPE_FLAG_EQ_CHECKS_OTHER_TYPE (0x0080)
|
||||||
|
#define MP_TYPE_FLAG_EQ_HAS_NEQ_TEST (0x0010)
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
PRINT_STR = 0,
|
PRINT_STR = 0,
|
||||||
|
|
|
@ -557,8 +557,8 @@ const mp_obj_type_t mp_type_array = {
|
||||||
#if MICROPY_PY_BUILTINS_BYTEARRAY
|
#if MICROPY_PY_BUILTINS_BYTEARRAY
|
||||||
const mp_obj_type_t mp_type_bytearray = {
|
const mp_obj_type_t mp_type_bytearray = {
|
||||||
{ &mp_type_type },
|
{ &mp_type_type },
|
||||||
|
.flags = MP_TYPE_FLAG_EQ_CHECKS_OTHER_TYPE,
|
||||||
.name = MP_QSTR_bytearray,
|
.name = MP_QSTR_bytearray,
|
||||||
.flags = MP_TYPE_FLAG_NEEDS_FULL_EQ_TEST,
|
|
||||||
.print = array_print,
|
.print = array_print,
|
||||||
.make_new = bytearray_make_new,
|
.make_new = bytearray_make_new,
|
||||||
.getiter = array_iterator_new,
|
.getiter = array_iterator_new,
|
||||||
|
|
|
@ -86,6 +86,7 @@ STATIC mp_obj_t bool_binary_op(mp_binary_op_t op, mp_obj_t lhs_in, mp_obj_t rhs_
|
||||||
|
|
||||||
const mp_obj_type_t mp_type_bool = {
|
const mp_obj_type_t mp_type_bool = {
|
||||||
{ &mp_type_type },
|
{ &mp_type_type },
|
||||||
|
.flags = MP_TYPE_FLAG_EQ_CHECKS_OTHER_TYPE, // can match all numeric types
|
||||||
.name = MP_QSTR_bool,
|
.name = MP_QSTR_bool,
|
||||||
.print = bool_print,
|
.print = bool_print,
|
||||||
.make_new = bool_make_new,
|
.make_new = bool_make_new,
|
||||||
|
|
|
@ -147,8 +147,8 @@ STATIC void complex_attr(mp_obj_t self_in, qstr attr, mp_obj_t *dest) {
|
||||||
|
|
||||||
const mp_obj_type_t mp_type_complex = {
|
const mp_obj_type_t mp_type_complex = {
|
||||||
{ &mp_type_type },
|
{ &mp_type_type },
|
||||||
|
.flags = MP_TYPE_FLAG_EQ_NOT_REFLEXIVE | MP_TYPE_FLAG_EQ_CHECKS_OTHER_TYPE,
|
||||||
.name = MP_QSTR_complex,
|
.name = MP_QSTR_complex,
|
||||||
.flags = MP_TYPE_FLAG_NEEDS_FULL_EQ_TEST,
|
|
||||||
.print = complex_print,
|
.print = complex_print,
|
||||||
.make_new = complex_make_new,
|
.make_new = complex_make_new,
|
||||||
.unary_op = complex_unary_op,
|
.unary_op = complex_unary_op,
|
||||||
|
|
|
@ -185,8 +185,8 @@ STATIC mp_obj_t float_binary_op(mp_binary_op_t op, mp_obj_t lhs_in, mp_obj_t rhs
|
||||||
|
|
||||||
const mp_obj_type_t mp_type_float = {
|
const mp_obj_type_t mp_type_float = {
|
||||||
{ &mp_type_type },
|
{ &mp_type_type },
|
||||||
|
.flags = MP_TYPE_FLAG_EQ_NOT_REFLEXIVE | MP_TYPE_FLAG_EQ_CHECKS_OTHER_TYPE,
|
||||||
.name = MP_QSTR_float,
|
.name = MP_QSTR_float,
|
||||||
.flags = MP_TYPE_FLAG_NEEDS_FULL_EQ_TEST,
|
|
||||||
.print = float_print,
|
.print = float_print,
|
||||||
.make_new = float_make_new,
|
.make_new = float_make_new,
|
||||||
.unary_op = float_unary_op,
|
.unary_op = float_unary_op,
|
||||||
|
|
|
@ -155,6 +155,7 @@ mp_obj_namedtuple_type_t *mp_obj_new_namedtuple_base(size_t n_fields, mp_obj_t *
|
||||||
STATIC mp_obj_t mp_obj_new_namedtuple_type(qstr name, size_t n_fields, mp_obj_t *fields) {
|
STATIC mp_obj_t mp_obj_new_namedtuple_type(qstr name, size_t n_fields, mp_obj_t *fields) {
|
||||||
mp_obj_namedtuple_type_t *o = mp_obj_new_namedtuple_base(n_fields, fields);
|
mp_obj_namedtuple_type_t *o = mp_obj_new_namedtuple_base(n_fields, fields);
|
||||||
o->base.base.type = &mp_type_type;
|
o->base.base.type = &mp_type_type;
|
||||||
|
o->base.flags = MP_TYPE_FLAG_EQ_CHECKS_OTHER_TYPE; // can match tuple
|
||||||
o->base.name = name;
|
o->base.name = name;
|
||||||
o->base.print = namedtuple_print;
|
o->base.print = namedtuple_print;
|
||||||
o->base.make_new = namedtuple_make_new;
|
o->base.make_new = namedtuple_make_new;
|
||||||
|
|
|
@ -563,8 +563,8 @@ STATIC MP_DEFINE_CONST_DICT(frozenset_locals_dict, frozenset_locals_dict_table);
|
||||||
|
|
||||||
const mp_obj_type_t mp_type_frozenset = {
|
const mp_obj_type_t mp_type_frozenset = {
|
||||||
{ &mp_type_type },
|
{ &mp_type_type },
|
||||||
|
.flags = MP_TYPE_FLAG_EQ_CHECKS_OTHER_TYPE,
|
||||||
.name = MP_QSTR_frozenset,
|
.name = MP_QSTR_frozenset,
|
||||||
.flags = MP_TYPE_FLAG_NEEDS_FULL_EQ_TEST,
|
|
||||||
.print = set_print,
|
.print = set_print,
|
||||||
.make_new = set_make_new,
|
.make_new = set_make_new,
|
||||||
.unary_op = set_unary_op,
|
.unary_op = set_unary_op,
|
||||||
|
|
|
@ -1100,7 +1100,8 @@ mp_obj_t mp_obj_new_type(qstr name, mp_obj_t bases_tuple, mp_obj_t locals_dict)
|
||||||
// TODO might need to make a copy of locals_dict; at least that's how CPython does it
|
// TODO might need to make a copy of locals_dict; at least that's how CPython does it
|
||||||
|
|
||||||
// Basic validation of base classes
|
// Basic validation of base classes
|
||||||
uint16_t base_flags = MP_TYPE_FLAG_NEEDS_FULL_EQ_TEST;
|
uint16_t base_flags = MP_TYPE_FLAG_EQ_NOT_REFLEXIVE
|
||||||
|
| MP_TYPE_FLAG_EQ_CHECKS_OTHER_TYPE | MP_TYPE_FLAG_EQ_HAS_NEQ_TEST;
|
||||||
size_t bases_len;
|
size_t bases_len;
|
||||||
mp_obj_t *bases_items;
|
mp_obj_t *bases_items;
|
||||||
mp_obj_tuple_get(bases_tuple, &bases_len, &bases_items);
|
mp_obj_tuple_get(bases_tuple, &bases_len, &bases_items);
|
||||||
|
|
Loading…
Reference in New Issue