Commit 6fca035b authored by Kevin Modzelewski's avatar Kevin Modzelewski

Split parsing and compilation in exec

compile() is starting to work
parent 687c0f70
...@@ -351,21 +351,84 @@ Box* evalOrExec(CLFunction* cl, Box* globals, Box* boxedLocals) { ...@@ -351,21 +351,84 @@ Box* evalOrExec(CLFunction* cl, Box* globals, Box* boxedLocals) {
return astInterpretFunctionEval(cf, globals, boxedLocals); return astInterpretFunctionEval(cf, globals, boxedLocals);
} }
template <typename AST_Type> template <typename AST_Type> CLFunction* compileForEvalOrExec(AST_Type* source, std::vector<AST_stmt*>& body) {
CLFunction* compileForEvalOrExec(AST_Type* source, std::vector<AST_stmt*>& body, BoxedModule* bm) {
LOCK_REGION(codegen_rwlock.asWrite()); LOCK_REGION(codegen_rwlock.asWrite());
Timer _t("for evalOrExec()"); Timer _t("for evalOrExec()");
ScopingAnalysis* scoping = new ScopingAnalysis(source, false); ScopingAnalysis* scoping = new ScopingAnalysis(source, false);
SourceInfo* si = new SourceInfo(bm, scoping, source, body); SourceInfo* si = new SourceInfo(getCurrentModule(), scoping, source, body);
CLFunction* cl_f = new CLFunction(0, 0, false, false, si); CLFunction* cl_f = new CLFunction(0, 0, false, false, si);
return cl_f; return cl_f;
} }
// Main entrypoints for eval and exec. CLFunction* compileExec(llvm::StringRef source) {
// TODO error message if parse fails or if it isn't an expr
// TODO should have a cleaner interface that can parse the Expression directly
// TODO this memory leaks
const char* code = source.data();
AST_Module* parsedModule = parse_string(code);
AST_Suite* parsedSuite = new AST_Suite(std::move(parsedModule->interned_strings));
parsedSuite->body = parsedModule->body;
return compileForEvalOrExec(parsedSuite, parsedSuite->body);
}
Box* compile(Box* source, Box* fn, Box* type, Box** _args) {
Box* flags = _args[0];
Box* dont_inherit = _args[0];
RELEASE_ASSERT(flags == boxInt(0), "");
RELEASE_ASSERT(dont_inherit == boxInt(0), "");
// source is allowed to be an AST, unicode, or anything that supports the buffer protocol
if (source->cls == unicode_cls) {
source = PyUnicode_AsUTF8String(source);
if (!source)
throwCAPIException();
// cf.cf_flags |= PyCF_SOURCE_IS_UTF8
}
if (isSubclass(fn->cls, unicode_cls)) {
fn = _PyUnicode_AsDefaultEncodedString(fn, NULL);
if (!fn)
throwCAPIException();
}
RELEASE_ASSERT(isSubclass(fn->cls, str_cls), "");
if (isSubclass(type->cls, unicode_cls)) {
type = _PyUnicode_AsDefaultEncodedString(type, NULL);
if (!type)
throwCAPIException();
}
RELEASE_ASSERT(isSubclass(type->cls, str_cls), "");
llvm::StringRef filename_str = static_cast<BoxedString*>(fn)->s;
llvm::StringRef type_str = static_cast<BoxedString*>(type)->s;
// Due to the fact that we get the fn from the parent module:
RELEASE_ASSERT(filename_str == "<string>", "filename must currently be '<string>'");
RELEASE_ASSERT(isSubclass(source->cls, str_cls), "");
llvm::StringRef source_str = static_cast<BoxedString*>(source)->s;
CLFunction* cl;
if (type_str == "exec") {
cl = compileExec(source_str);
} else if (type_str == "eval") {
fatalOrError(NotImplemented, "unimplemented");
throwCAPIException();
} else if (type_str == "single") {
fatalOrError(NotImplemented, "unimplemented");
throwCAPIException();
} else {
raiseExcHelper(ValueError, "compile() arg 3 must be 'exec', 'eval' or 'single'");
}
return codeForCLFunction(cl);
}
Box* eval(Box* boxedCode) { Box* eval(Box* boxedCode) {
Box* boxedLocals = fastLocalsToBoxedLocals(); Box* boxedLocals = fastLocalsToBoxedLocals();
BoxedModule* module = getCurrentModule(); BoxedModule* module = getCurrentModule();
...@@ -409,7 +472,7 @@ Box* eval(Box* boxedCode) { ...@@ -409,7 +472,7 @@ Box* eval(Box* boxedCode) {
assert(globals && (globals->cls == module_cls || globals->cls == dict_cls)); assert(globals && (globals->cls == module_cls || globals->cls == dict_cls));
CLFunction* cl = compileForEvalOrExec(parsedExpr, body, module); CLFunction* cl = compileForEvalOrExec(parsedExpr, body);
return evalOrExec(cl, globals, boxedLocals); return evalOrExec(cl, globals, boxedLocals);
} }
...@@ -432,8 +495,6 @@ Box* exec(Box* boxedCode, Box* globals, Box* locals) { ...@@ -432,8 +495,6 @@ Box* exec(Box* boxedCode, Box* globals, Box* locals) {
if (locals == None) if (locals == None)
locals = NULL; locals = NULL;
// TODO boxedCode is allowed to be a tuple
// TODO need to handle passing in globals
if (locals == NULL) { if (locals == NULL) {
locals = globals; locals = globals;
} }
...@@ -446,7 +507,6 @@ Box* exec(Box* boxedCode, Box* globals, Box* locals) { ...@@ -446,7 +507,6 @@ Box* exec(Box* boxedCode, Box* globals, Box* locals) {
globals = getGlobals(); globals = getGlobals();
BoxedModule* module = getCurrentModule(); BoxedModule* module = getCurrentModule();
if (globals && globals->cls == attrwrapper_cls && unwrapAttrWrapper(globals) == module) if (globals && globals->cls == attrwrapper_cls && unwrapAttrWrapper(globals) == module)
globals = module; globals = module;
...@@ -468,14 +528,16 @@ Box* exec(Box* boxedCode, Box* globals, Box* locals) { ...@@ -468,14 +528,16 @@ Box* exec(Box* boxedCode, Box* globals, Box* locals) {
// cf.cf_flags |= PyCF_SOURCE_IS_UTF8 // cf.cf_flags |= PyCF_SOURCE_IS_UTF8
} }
// TODO same issues as in `eval` CLFunction* cl;
RELEASE_ASSERT(boxedCode->cls == str_cls, "%s", boxedCode->cls->tp_name); if (boxedCode->cls == str_cls) {
const char* code = static_cast<BoxedString*>(boxedCode)->s.data(); cl = compileExec(static_cast<BoxedString*>(boxedCode)->s);
AST_Module* parsedModule = parse_string(code); } else if (boxedCode->cls == code_cls) {
AST_Suite* parsedSuite = new AST_Suite(std::move(parsedModule->interned_strings)); cl = clfunctionFromCode(boxedCode);
parsedSuite->body = parsedModule->body; } else {
abort();
}
assert(cl);
CLFunction* cl = compileForEvalOrExec(parsedSuite, parsedSuite->body, module);
return evalOrExec(cl, globals, locals); return evalOrExec(cl, globals, locals);
} }
......
...@@ -39,6 +39,7 @@ CompiledFunction* cfForMachineFunctionName(const std::string&); ...@@ -39,6 +39,7 @@ CompiledFunction* cfForMachineFunctionName(const std::string&);
extern "C" Box* exec(Box* boxedCode, Box* globals, Box* locals); extern "C" Box* exec(Box* boxedCode, Box* globals, Box* locals);
extern "C" Box* eval(Box* boxedCode); extern "C" Box* eval(Box* boxedCode);
extern "C" Box* compile(Box* source, Box* filename, Box* mode, Box** _args /* flags, dont_inherit */);
} }
#endif #endif
...@@ -1149,6 +1149,10 @@ void setupBuiltins() { ...@@ -1149,6 +1149,10 @@ void setupBuiltins() {
builtins_module->giveAttr("execfile", builtins_module->giveAttr("execfile",
new BoxedBuiltinFunctionOrMethod(boxRTFunction((void*)execfile, UNKNOWN, 1), "execfile")); new BoxedBuiltinFunctionOrMethod(boxRTFunction((void*)execfile, UNKNOWN, 1), "execfile"));
builtins_module->giveAttr(
"compile", new BoxedBuiltinFunctionOrMethod(boxRTFunction((void*)compile, UNKNOWN, 5, 2, false, false),
"compile", { boxInt(0), boxInt(0) }));
builtins_module->giveAttr( builtins_module->giveAttr(
"map", new BoxedBuiltinFunctionOrMethod(boxRTFunction((void*)map, LIST, 1, 0, true, false), "map")); "map", new BoxedBuiltinFunctionOrMethod(boxRTFunction((void*)map, LIST, 1, 0, true, false), "map"));
builtins_module->giveAttr( builtins_module->giveAttr(
......
...@@ -96,6 +96,11 @@ Box* codeForCLFunction(CLFunction* f) { ...@@ -96,6 +96,11 @@ Box* codeForCLFunction(CLFunction* f) {
return new BoxedCode(f); return new BoxedCode(f);
} }
CLFunction* clfunctionFromCode(Box* code) {
assert(code->cls == code_cls);
return static_cast<BoxedCode*>(code)->f;
}
extern "C" PyCodeObject* PyCode_New(int, int, int, int, PyObject*, PyObject*, PyObject*, PyObject*, PyObject*, extern "C" PyCodeObject* PyCode_New(int, int, int, int, PyObject*, PyObject*, PyObject*, PyObject*, PyObject*,
PyObject*, PyObject*, PyObject*, int, PyObject*) noexcept { PyObject*, PyObject*, PyObject*, int, PyObject*) noexcept {
RELEASE_ASSERT(0, "not implemented"); RELEASE_ASSERT(0, "not implemented");
......
...@@ -86,7 +86,7 @@ extern BoxedClass* object_cls, *type_cls, *bool_cls, *int_cls, *long_cls, *float ...@@ -86,7 +86,7 @@ extern BoxedClass* object_cls, *type_cls, *bool_cls, *int_cls, *long_cls, *float
*none_cls, *instancemethod_cls, *list_cls, *slice_cls, *module_cls, *dict_cls, *tuple_cls, *file_cls, *none_cls, *instancemethod_cls, *list_cls, *slice_cls, *module_cls, *dict_cls, *tuple_cls, *file_cls,
*enumerate_cls, *xrange_cls, *member_cls, *method_cls, *closure_cls, *generator_cls, *complex_cls, *basestring_cls, *enumerate_cls, *xrange_cls, *member_cls, *method_cls, *closure_cls, *generator_cls, *complex_cls, *basestring_cls,
*property_cls, *staticmethod_cls, *classmethod_cls, *attrwrapper_cls, *pyston_getset_cls, *capi_getset_cls, *property_cls, *staticmethod_cls, *classmethod_cls, *attrwrapper_cls, *pyston_getset_cls, *capi_getset_cls,
*builtin_function_or_method_cls, *set_cls, *frozenset_cls; *builtin_function_or_method_cls, *set_cls, *frozenset_cls, *code_cls;
} }
#define unicode_cls (&PyUnicode_Type) #define unicode_cls (&PyUnicode_Type)
#define memoryview_cls (&PyMemoryView_Type) #define memoryview_cls (&PyMemoryView_Type)
...@@ -823,6 +823,7 @@ extern Box* dict_descr; ...@@ -823,6 +823,7 @@ extern Box* dict_descr;
Box* codeForFunction(BoxedFunction*); Box* codeForFunction(BoxedFunction*);
Box* codeForCLFunction(CLFunction*); Box* codeForCLFunction(CLFunction*);
CLFunction* clfunctionFromCode(Box* code);
Box* getFrame(int depth); Box* getFrame(int depth);
} }
......
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