Commit 0f1eb426 authored by Kevin Modzelewski's avatar Kevin Modzelewski

Merge pull request #1109 from undingen/binop_subclass

implement correct binop lookup for subclasses
parents 4f3885c4 e14f5cdc
...@@ -4474,121 +4474,112 @@ extern "C" Box* runtimeCallCapi(Box* obj, ArgPassSpec argspec, Box* arg1, Box* a ...@@ -4474,121 +4474,112 @@ extern "C" Box* runtimeCallCapi(Box* obj, ArgPassSpec argspec, Box* arg1, Box* a
} }
template <Rewritable rewritable> template <Rewritable rewritable>
Box* binopInternal(Box* lhs, Box* rhs, int op_type, bool inplace, BinopRewriteArgs* rewrite_args) { static Box* binopInternalHelper(BinopRewriteArgs*& rewrite_args, BoxedString* op_name, Box* lhs, Box* rhs,
RewriterVar* r_lhs, RewriterVar* r_rhs) {
if (rewritable == NOT_REWRITABLE) { if (rewritable == NOT_REWRITABLE) {
assert(!rewrite_args); assert(!rewrite_args);
rewrite_args = NULL; rewrite_args = NULL;
} }
// TODO handle the case of the rhs being a subclass of the lhs
// this could get really annoying because you can dynamically make one type a subclass
// of the other!
assert(gc::isValidGCObject(lhs));
assert(gc::isValidGCObject(rhs));
if (rewrite_args) {
// TODO probably don't need to guard on the lhs_cls since it
// will get checked no matter what, but the check that should be
// removed is probably the later one.
// ie we should have some way of specifying what we know about the values
// of objects and their attributes, and the attributes' attributes.
rewrite_args->lhs->addAttrGuard(offsetof(Box, cls), (intptr_t)lhs->cls);
rewrite_args->rhs->addAttrGuard(offsetof(Box, cls), (intptr_t)rhs->cls);
}
struct NotImplementedHelper { struct NotImplementedHelper {
static void call(Box* r, bool was_notimplemented) { assert((r == NotImplemented) == was_notimplemented); } static void call(Box* r, bool was_notimplemented) { assert((r == NotImplemented) == was_notimplemented); }
}; };
Box* irtn = NULL; Box* rtn = NULL;
if (inplace) {
BoxedString* iop_name = getInplaceOpName(op_type);
if (rewrite_args) { if (rewrite_args) {
CallattrRewriteArgs srewrite_args(rewrite_args->rewriter, rewrite_args->lhs, rewrite_args->destination); CallattrRewriteArgs srewrite_args(rewrite_args->rewriter, r_lhs, rewrite_args->destination);
srewrite_args.arg1 = rewrite_args->rhs; srewrite_args.arg1 = r_rhs;
srewrite_args.args_guarded = true; srewrite_args.args_guarded = true;
irtn = callattrInternal1<CXX, REWRITABLE>(lhs, iop_name, CLASS_ONLY, &srewrite_args, ArgPassSpec(1), rhs); rtn = callattrInternal1<CXX, REWRITABLE>(lhs, op_name, CLASS_ONLY, &srewrite_args, ArgPassSpec(1), rhs);
if (!srewrite_args.isSuccessful()) { if (!srewrite_args.isSuccessful()) {
rewrite_args = NULL; rewrite_args = NULL;
} else if (irtn) { } else if (rtn) {
rewrite_args->out_rtn = srewrite_args.getReturn(ReturnConvention::HAS_RETURN); rewrite_args->out_rtn = srewrite_args.getReturn(ReturnConvention::HAS_RETURN);
// If we allowed a rewrite to get here, it means that we assumed that the class will return NotImplemented // If we allowed a rewrite to get here, it means that we assumed that the class will return NotImplemented
// or not based only on the types of the inputs. // or not based only on the types of the inputs.
#ifndef NDEBUG #ifndef NDEBUG
rewrite_args->rewriter->call(false, (void*)NotImplementedHelper::call, rewrite_args->out_rtn, rewrite_args->rewriter->call(false, (void*)NotImplementedHelper::call, rewrite_args->out_rtn,
rewrite_args->rewriter->loadConst(irtn == NotImplemented)); rewrite_args->rewriter->loadConst(rtn == NotImplemented));
#endif #endif
} else { } else {
srewrite_args.assertReturnConvention(ReturnConvention::NO_RETURN); srewrite_args.assertReturnConvention(ReturnConvention::NO_RETURN);
} }
} else {
irtn = callattrInternal1<CXX, NOT_REWRITABLE>(lhs, iop_name, CLASS_ONLY, NULL, ArgPassSpec(1), rhs);
}
if (irtn) { if (rewrite_args && rtn) {
if (irtn != NotImplemented) { if (rtn != NotImplemented)
if (rewrite_args) {
assert(rewrite_args->out_rtn);
rewrite_args->out_success = true; rewrite_args->out_success = true;
} else {
return irtn; // I think our guarding up to here is correct; the issue is we won't be able to complete
} else { // the rewrite since we have more guards to do, but we already did some mutations.
assert(!rewrite_args); rewrite_args->out_success = false;
rewrite_args = NULL;
REWRITE_ABORTED("");
} }
} }
// we don't need to abort the rewrite when the attribute does not exist (rtn==null) because we only rewrite
// binops when both sides are not user defined types for which we assume that they will never change.
} else {
rtn = callattrInternal1<CXX, NOT_REWRITABLE>(lhs, op_name, CLASS_ONLY, NULL, ArgPassSpec(1), rhs);
} }
BoxedString* op_name = getOpName(op_type); return rtn;
Box* lrtn; }
if (rewrite_args) {
CallattrRewriteArgs srewrite_args(rewrite_args->rewriter, rewrite_args->lhs, rewrite_args->destination);
srewrite_args.arg1 = rewrite_args->rhs;
lrtn = callattrInternal1<CXX, REWRITABLE>(lhs, op_name, CLASS_ONLY, &srewrite_args, ArgPassSpec(1), rhs);
if (!srewrite_args.isSuccessful()) { template <Rewritable rewritable>
Box* binopInternal(Box* lhs, Box* rhs, int op_type, bool inplace, BinopRewriteArgs* rewrite_args) {
if (rewritable == NOT_REWRITABLE) {
assert(!rewrite_args);
rewrite_args = NULL; rewrite_args = NULL;
} else if (lrtn) {
rewrite_args->out_rtn = srewrite_args.getReturn(ReturnConvention::HAS_RETURN);
// If we allowed a rewrite to get here, it means that we assumed that the class will return NotImplemented
// or not based only on the types of the inputs.
#ifndef NDEBUG
rewrite_args->rewriter->call(false, (void*)NotImplementedHelper::call, rewrite_args->out_rtn,
rewrite_args->rewriter->loadConst(irtn == NotImplemented));
#endif
} else {
srewrite_args.assertReturnConvention(ReturnConvention::NO_RETURN);
}
} else {
lrtn = callattrInternal1<CXX, NOT_REWRITABLE>(lhs, op_name, CLASS_ONLY, NULL, ArgPassSpec(1), rhs);
} }
assert(gc::isValidGCObject(lhs));
assert(gc::isValidGCObject(rhs));
if (lrtn) { RewriterVar* r_lhs = NULL;
if (lrtn != NotImplemented) { RewriterVar* r_rhs = NULL;
if (rewrite_args) { if (rewrite_args) {
assert(rewrite_args->out_rtn); r_lhs = rewrite_args->lhs;
rewrite_args->out_success = true; r_rhs = rewrite_args->rhs;
}
return lrtn; RewriterVar* r_lhs_cls = r_lhs->getAttr(offsetof(Box, cls));
} r_lhs_cls->addGuard((intptr_t)lhs->cls);
RewriterVar* r_rhs_cls = r_rhs->getAttr(offsetof(Box, cls));
r_rhs_cls->addGuard((intptr_t)rhs->cls);
r_lhs_cls->addAttrGuard(offsetof(BoxedClass, tp_mro), (intptr_t)lhs->cls->tp_mro);
r_rhs_cls->addAttrGuard(offsetof(BoxedClass, tp_mro), (intptr_t)rhs->cls->tp_mro);
} }
// TODO patch these cases Box* irtn = NULL;
// actually, I think our guarding up to here is correct; the issue is we won't be able to complete if (inplace) {
// the rewrite since we have more guards to do, but we already did some mutations. BoxedString* iop_name = getInplaceOpName(op_type);
if (rewrite_args) { irtn = binopInternalHelper<rewritable>(rewrite_args, iop_name, lhs, rhs, r_lhs, r_rhs);
assert(rewrite_args->out_success == false); if (irtn && irtn != NotImplemented)
rewrite_args = NULL; return irtn;
REWRITE_ABORTED("");
} }
bool should_try_reverse = true;
Box* rrtn = NULL;
if (lhs->cls != rhs->cls && isSubclass(rhs->cls, lhs->cls)) {
should_try_reverse = false;
BoxedString* rop_name = getReverseOpName(op_type); BoxedString* rop_name = getReverseOpName(op_type);
Box* rrtn = callattrInternal1<CXX, REWRITABLE>(rhs, rop_name, CLASS_ONLY, NULL, ArgPassSpec(1), lhs); rrtn = binopInternalHelper<rewritable>(rewrite_args, rop_name, rhs, lhs, r_rhs, r_lhs);
if (rrtn != NULL && rrtn != NotImplemented) if (rrtn && rrtn != NotImplemented)
return rrtn; return rrtn;
}
BoxedString* op_name = getOpName(op_type);
Box* lrtn = binopInternalHelper<rewritable>(rewrite_args, op_name, lhs, rhs, r_lhs, r_rhs);
if (lrtn && lrtn != NotImplemented)
return lrtn;
if (should_try_reverse) {
BoxedString* rop_name = getReverseOpName(op_type);
rrtn = binopInternalHelper<rewritable>(rewrite_args, rop_name, rhs, lhs, r_rhs, r_lhs);
if (rrtn && rrtn != NotImplemented)
return rrtn;
}
llvm::StringRef op_sym = getOpSymbol(op_type); llvm::StringRef op_sym = getOpSymbol(op_type);
const char* op_sym_suffix = ""; const char* op_sym_suffix = "";
...@@ -4597,6 +4588,7 @@ Box* binopInternal(Box* lhs, Box* rhs, int op_type, bool inplace, BinopRewriteAr ...@@ -4597,6 +4588,7 @@ Box* binopInternal(Box* lhs, Box* rhs, int op_type, bool inplace, BinopRewriteAr
} }
if (VERBOSITY()) { if (VERBOSITY()) {
BoxedString* rop_name = getReverseOpName(op_type);
if (inplace) { if (inplace) {
BoxedString* iop_name = getInplaceOpName(op_type); BoxedString* iop_name = getInplaceOpName(op_type);
if (irtn) if (irtn)
......
# expected: fail
# - binop ordering is wrong,
# should prefer executing a operation on the subclass first
class M(type): class M(type):
def __instancecheck__(self, rhs): def __instancecheck__(self, rhs):
print "M.instancecheck", print "M.instancecheck",
......
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