Commit 8aa6259f authored by Kevin Modzelewski's avatar Kevin Modzelewski

int.__new__ has different behavior on ints and subclasses

I thought it was just an implementation detail, but it has
observable effects from Python.
parent 308c5d36
...@@ -753,33 +753,19 @@ extern "C" Box* intOct(BoxedInt* self) { ...@@ -753,33 +753,19 @@ extern "C" Box* intOct(BoxedInt* self) {
return new BoxedString(std::string(buf, len)); return new BoxedString(std::string(buf, len));
} }
extern "C" Box* intNew(Box* _cls, Box* val) { BoxedInt* _intNew(Box* val) {
if (!isSubclass(_cls->cls, type_cls))
raiseExcHelper(TypeError, "int.__new__(X): X is not a type object (%s)", getTypeName(_cls)->c_str());
BoxedClass* cls = static_cast<BoxedClass*>(_cls);
if (!isSubclass(cls, int_cls))
raiseExcHelper(TypeError, "int.__new__(%s): %s is not a subtype of int", getNameOfClass(cls)->c_str(),
getNameOfClass(cls)->c_str());
assert(cls->tp_basicsize >= sizeof(BoxedInt));
void* mem = gc_alloc(cls->tp_basicsize, gc::GCKind::PYTHON);
BoxedInt* rtn = ::new (mem) BoxedInt(cls, 0);
initUserAttrs(rtn, cls);
if (isSubclass(val->cls, int_cls)) { if (isSubclass(val->cls, int_cls)) {
rtn->n = static_cast<BoxedInt*>(val)->n; return new BoxedInt(int_cls, static_cast<BoxedInt*>(val)->n);
} else if (val->cls == str_cls) { } else if (val->cls == str_cls) {
BoxedString* s = static_cast<BoxedString*>(val); BoxedString* s = static_cast<BoxedString*>(val);
std::istringstream ss(s->s); std::istringstream ss(s->s);
int64_t n; int64_t n;
ss >> n; ss >> n;
rtn->n = n; return new BoxedInt(int_cls, n);
} else if (val->cls == float_cls) { } else if (val->cls == float_cls) {
double d = static_cast<BoxedFloat*>(val)->d; double d = static_cast<BoxedFloat*>(val)->d;
return new BoxedInt(int_cls, d);
rtn->n = d;
} else { } else {
static const std::string int_str("__int__"); static const std::string int_str("__int__");
Box* r = callattr(val, &int_str, CallattrFlags({.cls_only = true, .null_on_nonexistent = true }), Box* r = callattr(val, &int_str, CallattrFlags({.cls_only = true, .null_on_nonexistent = true }),
...@@ -794,8 +780,30 @@ extern "C" Box* intNew(Box* _cls, Box* val) { ...@@ -794,8 +780,30 @@ extern "C" Box* intNew(Box* _cls, Box* val) {
if (!isSubclass(r->cls, int_cls)) { if (!isSubclass(r->cls, int_cls)) {
raiseExcHelper(TypeError, "__int__ returned non-int (type %s)", r->cls->tp_name); raiseExcHelper(TypeError, "__int__ returned non-int (type %s)", r->cls->tp_name);
} }
rtn->n = static_cast<BoxedInt*>(r)->n; return static_cast<BoxedInt*>(r);
} }
}
extern "C" Box* intNew(Box* _cls, Box* val) {
if (!isSubclass(_cls->cls, type_cls))
raiseExcHelper(TypeError, "int.__new__(X): X is not a type object (%s)", getTypeName(_cls)->c_str());
BoxedClass* cls = static_cast<BoxedClass*>(_cls);
if (!isSubclass(cls, int_cls))
raiseExcHelper(TypeError, "int.__new__(%s): %s is not a subtype of int", getNameOfClass(cls)->c_str(),
getNameOfClass(cls)->c_str());
if (cls == int_cls)
return _intNew(val);
BoxedInt* n = _intNew(val);
assert(cls->tp_basicsize >= sizeof(BoxedInt));
void* mem = gc_alloc(cls->tp_basicsize, gc::GCKind::PYTHON);
BoxedInt* rtn = ::new (mem) BoxedInt(cls, 0);
initUserAttrs(rtn, cls);
rtn->n = n->n;
return rtn; return rtn;
} }
......
...@@ -29,7 +29,7 @@ class D(object): ...@@ -29,7 +29,7 @@ class D(object):
for a in (False, True, 2, 3.0, D(4), D(C(5)), D(False)): for a in (False, True, 2, 3.0, D(4), D(C(5)), D(False)):
i = int.__new__(C, a) i = int.__new__(C, a)
print type(i), i print type(i), i, type(int(a))
try: try:
int.__new__(C, D(1.0)) int.__new__(C, D(1.0))
......
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