Commit d3ba142d authored by Kevin Modzelewski's avatar Kevin Modzelewski

Identify + fix codegen bug

In certain cases we wouldn't do well if we were sure that a type error
would occur (ex indexing into what we know is None) -- we would error in
codegen instead of generating the code to throw the error at runtime.

(sneak in another travis.yml attempt)
parent 17fa11b0
...@@ -16,7 +16,7 @@ install: ...@@ -16,7 +16,7 @@ install:
- git config --global user.email "you@example.com" - git config --global user.email "you@example.com"
- git config --global user.name "Your Name" - git config --global user.name "Your Name"
- mkdir ~/pyston-build && cd ~/pyston-build - mkdir ~/pyston-build && cd ~/pyston-build
- make -C ~/pyston llvm_up - make -C $TRAVIS_BUILD_DIR llvm_up
- cmake -GNinja $TRAVIS_BUILD_DIR - cmake -GNinja $TRAVIS_BUILD_DIR
- ninja libunwind ext_cpython - ninja libunwind ext_cpython
......
...@@ -1068,6 +1068,13 @@ public: ...@@ -1068,6 +1068,13 @@ public:
return boolFromI1(emitter, cmp); return boolFromI1(emitter, cmp);
} }
CompilerVariable* getitem(IREmitter& emitter, const OpInfo& info, VAR* var, CompilerVariable* slice) override {
ConcreteCompilerVariable* converted = var->makeConverted(emitter, BOXED_FLOAT);
CompilerVariable* rtn = converted->getitem(emitter, info, slice);
converted->decvref(emitter);
return rtn;
}
CompilerVariable* binexp(IREmitter& emitter, const OpInfo& info, VAR* var, CompilerVariable* rhs, CompilerVariable* binexp(IREmitter& emitter, const OpInfo& info, VAR* var, CompilerVariable* rhs,
AST_TYPE::AST_TYPE op_type, BinExpType exp_type) override { AST_TYPE::AST_TYPE op_type, BinExpType exp_type) override {
if (rhs->getType() != INT && rhs->getType() != FLOAT) { if (rhs->getType() != INT && rhs->getType() != FLOAT) {
...@@ -1366,21 +1373,21 @@ public: ...@@ -1366,21 +1373,21 @@ public:
const std::string* attr, bool clsonly, ArgPassSpec argspec, const std::string* attr, bool clsonly, ArgPassSpec argspec,
const std::vector<CompilerVariable*>& args, const std::vector<CompilerVariable*>& args,
const std::vector<const std::string*>* keyword_names, const std::vector<const std::string*>* keyword_names,
bool raise_on_missing = true) { bool* no_attribute = NULL) {
if (!canStaticallyResolveGetattrs()) if (!canStaticallyResolveGetattrs())
return NULL; return NULL;
Box* rtattr = cls->getattr(*attr); Box* rtattr = cls->getattr(*attr);
if (rtattr == NULL) { if (rtattr == NULL) {
if (raise_on_missing) { if (no_attribute) {
*no_attribute = true;
} else {
llvm::CallSite call = emitter.createCall2(info.unw_info, g.funcs.raiseAttributeErrorStr, llvm::CallSite call = emitter.createCall2(info.unw_info, g.funcs.raiseAttributeErrorStr,
getStringConstantPtr(*getNameOfClass(cls) + "\0"), getStringConstantPtr(*getNameOfClass(cls) + "\0"),
getStringConstantPtr(*attr + '\0')); getStringConstantPtr(*attr + '\0'));
call.setDoesNotReturn(); call.setDoesNotReturn();
return undefVariable();
} else {
return NULL;
} }
return undefVariable();
} }
if (rtattr->cls != function_cls) if (rtattr->cls != function_cls)
...@@ -1459,6 +1466,7 @@ public: ...@@ -1459,6 +1466,7 @@ public:
ConcreteCompilerVariable* rtn = _call(emitter, info, linked_function, cf->code, other_args, argspec, new_args, ConcreteCompilerVariable* rtn = _call(emitter, info, linked_function, cf->code, other_args, argspec, new_args,
keyword_names, cf->spec->rtn_type); keyword_names, cf->spec->rtn_type);
assert(rtn->getType() == cf->spec->rtn_type); assert(rtn->getType() == cf->spec->rtn_type);
assert(rtn->getType() != UNDEF);
// We should provide unboxed versions of these rather than boxing then unboxing: // We should provide unboxed versions of these rather than boxing then unboxing:
// TODO is it more efficient to unbox here, or should we leave it boxed? // TODO is it more efficient to unbox here, or should we leave it boxed?
...@@ -1509,8 +1517,21 @@ public: ...@@ -1509,8 +1517,21 @@ public:
const std::string& left_side_name = getOpName(op_type); const std::string& left_side_name = getOpName(op_type);
ConcreteCompilerVariable* called_constant = tryCallattrConstant( bool no_attribute = false;
emitter, info, var, &left_side_name, true, ArgPassSpec(1, 0, 0, 0), { converted_rhs }, NULL, false); ConcreteCompilerVariable* called_constant
= tryCallattrConstant(emitter, info, var, &left_side_name, true, ArgPassSpec(1, 0, 0, 0),
{ converted_rhs }, NULL, &no_attribute);
if (no_attribute) {
assert(called_constant->getType() == UNDEF);
// Kind of hacky, but just call into getitem like normal. except...
auto r = UNKNOWN->binexp(emitter, info, var, converted_rhs, op_type, exp_type);
r->decvref(emitter);
// ... return the undef value, since that matches what the type analyzer thought we would do.
return called_constant;
}
if (called_constant) { if (called_constant) {
converted_rhs->decvref(emitter); converted_rhs->decvref(emitter);
return called_constant; return called_constant;
...@@ -1525,8 +1546,20 @@ public: ...@@ -1525,8 +1546,20 @@ public:
CompilerVariable* getitem(IREmitter& emitter, const OpInfo& info, VAR* var, CompilerVariable* slice) override { CompilerVariable* getitem(IREmitter& emitter, const OpInfo& info, VAR* var, CompilerVariable* slice) override {
static const std::string attr("__getitem__"); static const std::string attr("__getitem__");
ConcreteCompilerVariable* called_constant bool no_attribute = false;
= tryCallattrConstant(emitter, info, var, &attr, true, ArgPassSpec(1, 0, 0, 0), { slice }, NULL, false); ConcreteCompilerVariable* called_constant = tryCallattrConstant(
emitter, info, var, &attr, true, ArgPassSpec(1, 0, 0, 0), { slice }, NULL, &no_attribute);
if (no_attribute) {
assert(called_constant->getType() == UNDEF);
// Kind of hacky, but just call into getitem like normal. except...
auto r = UNKNOWN->getitem(emitter, info, var, slice);
r->decvref(emitter);
// ... return the undef value, since that matches what the type analyzer thought we would do.
return called_constant;
}
if (called_constant) if (called_constant)
return called_constant; return called_constant;
......
# Regression test: make sure that we can handle various kinds of written type errors.
# ex if we can prove that an expression would throw an error due to type issues, we should
# still compile and run the function just fine.
#
# We definitely try to support this, but it can be tricky sometimes since the different phases
# (mainly, type analysis and irgen) have to agree on the way they deal with the error. If they
# don't, irgen will crash after it detects a mismatch with type analysis.
def f(p):
if not "opaque branch condition":
k = p[1]
print k
l = len(p)
print l
t = p + 1
print t
# Run in a loop to trigger OSR:
for i in xrange(20):
f(None)
print "done"
try:
print 1.0 & 1.0
except TypeError, e:
print e
try:
print 1.0[0]
except TypeError, e:
print e
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