Skip to content

Commit

Permalink
refactor code that decides how to execute top-level expressions (#24600)
Browse files Browse the repository at this point in the history
also avoid inferring blocks with method definitions and other forms
used only at load time. helps #24316.
  • Loading branch information
JeffBezanson authored Dec 3, 2017
1 parent 04ae93a commit b02cf6e
Show file tree
Hide file tree
Showing 4 changed files with 98 additions and 87 deletions.
3 changes: 2 additions & 1 deletion src/gf.c
Original file line number Diff line number Diff line change
Expand Up @@ -1679,7 +1679,8 @@ jl_llvm_functions_t jl_compile_for_dispatch(jl_method_instance_t **pli, size_t w
return li->functionObjectsDecls;
}
}
if (jl_options.compile_enabled == JL_OPTIONS_COMPILE_OFF) {
if (jl_options.compile_enabled == JL_OPTIONS_COMPILE_OFF &&
jl_is_method(li->def.method)) {
jl_code_info_t *src = jl_code_for_interpreter(li);
if (!jl_code_requires_compiler(src)) {
li->inferred = (jl_value_t*)src;
Expand Down
26 changes: 16 additions & 10 deletions src/method.c
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,8 @@ extern "C" {
#endif

extern jl_value_t *jl_builtin_getfield;
jl_value_t *jl_resolve_globals(jl_value_t *expr, jl_module_t *module, jl_svec_t *sparam_vals)
static jl_value_t *resolve_globals(jl_value_t *expr, jl_module_t *module, jl_svec_t *sparam_vals,
int binding_effects)
{
if (jl_is_symbol(expr)) {
if (module == NULL)
Expand All @@ -25,7 +26,7 @@ jl_value_t *jl_resolve_globals(jl_value_t *expr, jl_module_t *module, jl_svec_t
}
else if (jl_is_expr(expr)) {
jl_expr_t *e = (jl_expr_t*)expr;
if (e->head == global_sym) {
if (e->head == global_sym && binding_effects) {
// execute the side-effects of "global x" decl immediately:
// creates uninitialized mutable binding in module for each global
jl_toplevel_eval_flex(module, expr, 0, 1);
Expand Down Expand Up @@ -116,13 +117,23 @@ jl_value_t *jl_resolve_globals(jl_value_t *expr, jl_module_t *module, jl_svec_t
}
for (; i < nargs; i++) {
// TODO: this should be making a copy, not mutating the source
jl_exprargset(e, i, jl_resolve_globals(jl_exprarg(e, i), module, sparam_vals));
jl_exprargset(e, i, resolve_globals(jl_exprarg(e, i), module, sparam_vals, binding_effects));
}
}
}
return expr;
}

void jl_resolve_globals_in_ir(jl_array_t *stmts, jl_module_t *m, jl_svec_t *sparam_vals,
int binding_effects)
{
size_t i, l = jl_array_len(stmts);
for (i = 0; i < l; i++) {
jl_value_t *stmt = jl_array_ptr_ref(stmts, i);
jl_array_ptr_set(stmts, i, resolve_globals(stmt, m, sparam_vals, binding_effects));
}
}

// copy a :lambda Expr into its CodeInfo representation,
// including popping of known meta nodes
static void jl_code_info_set_ast(jl_code_info_t *li, jl_expr_t *ast)
Expand Down Expand Up @@ -308,12 +319,7 @@ JL_DLLEXPORT jl_code_info_t *jl_code_for_staged(jl_method_instance_t *linfo)
}

jl_array_t *stmts = (jl_array_t*)func->code;
size_t i, l;
for (i = 0, l = jl_array_len(stmts); i < l; i++) {
jl_value_t *stmt = jl_array_ptr_ref(stmts, i);
stmt = jl_resolve_globals(stmt, linfo->def.method->module, linfo->sparam_vals);
jl_array_ptr_set(stmts, i, stmt);
}
jl_resolve_globals_in_ir(stmts, linfo->def.method->module, linfo->sparam_vals, 1);
}

ptls->in_pure_callback = last_in;
Expand Down Expand Up @@ -437,7 +443,7 @@ static void jl_method_set_source(jl_method_t *m, jl_code_info_t *src)
}
}
else {
st = jl_resolve_globals(st, m->module, sparam_vars);
st = resolve_globals(st, m->module, sparam_vars, 1);
}
jl_array_ptr_set(copy, i, st);
}
Expand Down
152 changes: 78 additions & 74 deletions src/toplevel.c
Original file line number Diff line number Diff line change
Expand Up @@ -298,17 +298,31 @@ JL_DLLEXPORT jl_module_t *jl_base_relative_to(jl_module_t *m)
return jl_top_module;
}

int jl_has_intrinsics(jl_value_t *v)
static void expr_attributes(jl_value_t *v, int *has_intrinsics, int *has_defs)
{
if (!jl_is_expr(v))
return 0;
return;
jl_expr_t *e = (jl_expr_t*)v;
if (e->head == toplevel_sym || e->head == copyast_sym)
return 0;
if (e->head == foreigncall_sym)
return 1;
jl_value_t *called = NULL;
if (e->head == call_sym && jl_expr_nargs(e) > 0) {
jl_sym_t *head = e->head;
if (head == toplevel_sym || head == thunk_sym) {
return;
}
else if (head == global_sym || head == const_sym || head == copyast_sym) {
// Note: `copyast` is included here since it indicates the presence of
// `quote` and probably `eval`.
*has_defs = 1;
return;
}
else if (head == method_sym || head == abstracttype_sym || head == primtype_sym ||
head == structtype_sym || jl_is_toplevel_only_expr(v)) {
*has_defs = 1;
}
else if (head == foreigncall_sym) {
*has_intrinsics = 1;
return;
}
else if (head == call_sym && jl_expr_nargs(e) > 0) {
jl_value_t *called = NULL;
jl_value_t *f = jl_exprarg(e, 0);
if (jl_is_globalref(f)) {
jl_module_t *mod = jl_globalref_mod(f);
Expand All @@ -322,37 +336,37 @@ int jl_has_intrinsics(jl_value_t *v)
else if (jl_is_quotenode(f)) {
called = jl_quotenode_value(f);
}
if (called && jl_is_intrinsic(called) && jl_unbox_int32(called) == (int)llvmcall) {
*has_intrinsics = 1;
return;
}
}
if (called && jl_is_intrinsic(called) && jl_unbox_int32(called) == (int)llvmcall)
return 1;
int i;
for (i = 0; i < jl_array_len(e->args); i++) {
jl_value_t *a = jl_exprarg(e, i);
if (jl_is_expr(a) && jl_has_intrinsics(a))
return 1;
if (jl_is_expr(a))
expr_attributes(a, has_intrinsics, has_defs);
}
return 0;
}

int jl_code_requires_compiler(jl_code_info_t *src)
{
jl_array_t *body = src->code;
assert(jl_typeis(body, jl_array_any_type));
size_t i;
int has_intrinsics = 0, has_defs = 0;
for(i=0; i < jl_array_len(body); i++) {
jl_value_t *stmt = jl_array_ptr_ref(body,i);
if (jl_has_intrinsics(stmt))
expr_attributes(stmt, &has_intrinsics, &has_defs);
if (has_intrinsics)
return 1;
}
return 0;
}

// heuristic for whether a top-level input should be evaluated with
// the compiler or the interpreter.
static int jl_eval_with_compiler_p(jl_code_info_t *src, jl_array_t *body, int compileloops, jl_module_t *m)
static void body_attributes(jl_array_t *body, int *has_intrinsics, int *has_defs, int *has_loops)
{
size_t i, maxlabl=0;
// compile if there are backwards branches and compiler is enabled
for(i=0; i < jl_array_len(body); i++) {
jl_value_t *stmt = jl_array_ptr_ref(body,i);
if (jl_is_labelnode(stmt)) {
Expand All @@ -362,41 +376,29 @@ static int jl_eval_with_compiler_p(jl_code_info_t *src, jl_array_t *body, int co
}
size_t sz = (maxlabl+1+7)/8;
char *labls = (char*)alloca(sz); memset(labls,0,sz);
*has_loops = 0;
for(i=0; i < jl_array_len(body); i++) {
jl_value_t *stmt = jl_array_ptr_ref(body,i);
if (jl_options.compile_enabled != JL_OPTIONS_COMPILE_OFF) {
if (!*has_loops) {
if (jl_is_labelnode(stmt)) {
int l = jl_labelnode_label(stmt);
labls[l/8] |= (1<<(l&7));
}
else if (compileloops && jl_is_gotonode(stmt)) {
else if (jl_is_gotonode(stmt)) {
int l = jl_gotonode_label(stmt);
if (labls[l/8]&(1<<(l&7))) {
return 1;
}
if (labls[l/8] & (1<<(l&7)))
*has_loops = 1;
}
else if (jl_is_expr(stmt)) {
if (compileloops && ((jl_expr_t*)stmt)->head==goto_ifnot_sym) {
if (((jl_expr_t*)stmt)->head==goto_ifnot_sym) {
int l = jl_unbox_long(jl_exprarg(stmt,1));
if (labls[l/8]&(1<<(l&7))) {
return 1;
}
if (labls[l/8] & (1<<(l&7)))
*has_loops = 1;
}
}
}
if (jl_has_intrinsics(stmt))
return 1;
expr_attributes(stmt, has_intrinsics, has_defs);
}
return 0;
}

static int jl_eval_expr_with_compiler_p(jl_value_t *e, int compileloops, jl_module_t *m)
{
if (jl_is_expr(e) && ((jl_expr_t*)e)->head == body_sym)
return jl_eval_with_compiler_p(NULL, ((jl_expr_t*)e)->args, compileloops, m);
if (jl_has_intrinsics(e))
return 1;
return 0;
}

static jl_module_t *call_require(jl_sym_t *var)
Expand Down Expand Up @@ -485,20 +487,15 @@ int jl_is_toplevel_only_expr(jl_value_t *e)
((jl_expr_t*)e)->head == jl_incomplete_sym);
}

jl_value_t *jl_resolve_globals(jl_value_t *expr, jl_module_t *module, jl_svec_t *sparam_vals);
static jl_method_instance_t *jl_new_thunk(jl_code_info_t *src, jl_module_t *module)
void jl_resolve_globals_in_ir(jl_array_t *stmts, jl_module_t *m, jl_svec_t *sparam_vals,
int binding_effects);

static jl_method_instance_t *method_instance_for_thunk(jl_code_info_t *src, jl_module_t *module)
{
jl_method_instance_t *li = jl_new_method_instance_uninit();
li->inferred = (jl_value_t*)src;
li->specTypes = (jl_value_t*)jl_emptytuple_type;
li->def.module = module;
jl_array_t *stmts = (jl_array_t*)src->code;
size_t i, l;
JL_GC_PUSH1(&li);
for (i = 0, l = jl_array_len(stmts); i < l; i++) {
jl_array_ptr_set(stmts, i, jl_resolve_globals(jl_array_ptr_ref(stmts, i), module, NULL));
}
JL_GC_POP();
return li;
}

Expand Down Expand Up @@ -653,9 +650,7 @@ jl_value_t *jl_toplevel_eval_flex(jl_module_t *m, jl_value_t *e, int fast, int e
}

jl_method_instance_t *li = NULL;
jl_value_t *result;
jl_code_info_t *thk = NULL;
int ewc = 0;
JL_GC_PUSH3(&li, &thk, &ex);

if (!expanded && ex->head != body_sym && ex->head != thunk_sym && ex->head != return_sym &&
Expand Down Expand Up @@ -686,42 +681,50 @@ jl_value_t *jl_toplevel_eval_flex(jl_module_t *m, jl_value_t *e, int fast, int e
jl_throw(jl_exprarg(ex,0));
}

int has_intrinsics = 0, has_defs = 0, has_loops = 0;
if (head == thunk_sym) {
thk = (jl_code_info_t*)jl_exprarg(ex,0);
assert(jl_is_code_info(thk));
assert(jl_typeis(thk->code, jl_array_any_type));
ewc = jl_eval_with_compiler_p(thk, (jl_array_t*)thk->code, fast, m);
body_attributes((jl_array_t*)thk->code, &has_intrinsics, &has_defs, &has_loops);
}
else if (head == body_sym) {
thk = expr_to_code_info((jl_value_t*)ex);
body_attributes((jl_array_t*)thk->code, &has_intrinsics, &has_defs, &has_loops);
}
else {
if (head && jl_eval_expr_with_compiler_p((jl_value_t*)ex, fast, m)) {
thk = expr_to_code_info((jl_value_t*)ex);
ewc = 1;
}
else if (head == body_sym) {
thk = expr_to_code_info((jl_value_t*)ex);
}
else {
if (jl_is_toplevel_only_expr((jl_value_t*)ex)) {
result = jl_toplevel_eval(m, (jl_value_t*)ex);
}
else {
result = jl_interpret_toplevel_expr_in(m, (jl_value_t*)ex, NULL, NULL);
}
JL_GC_POP();
return result;
}
expr_attributes((jl_value_t*)ex, &has_intrinsics, &has_defs);
}

if (ewc) {
li = jl_new_thunk(thk, m);
size_t world = jl_get_ptls_states()->world_age;
jl_type_infer(&li, world, 0);
jl_value_t *result;
if (has_intrinsics || (!has_defs && fast && has_loops &&
jl_options.compile_enabled != JL_OPTIONS_COMPILE_OFF)) {
// use codegen
if (thk == NULL)
thk = expr_to_code_info((jl_value_t*)ex);
li = method_instance_for_thunk(thk, m);
jl_resolve_globals_in_ir((jl_array_t*)thk->code, m, NULL, 0);
// Don't infer blocks containing e.g. method definitions, since it's probably not
// worthwhile and also unsound (see #24316).
// TODO: This is still not correct since an `eval` can happen elsewhere, but it
// helps in common cases.
if (!has_defs) {
size_t world = jl_get_ptls_states()->world_age;
jl_type_infer(&li, world, 0);
}
jl_value_t *dummy_f_arg = NULL;
result = jl_call_method_internal(li, &dummy_f_arg, 1);
}
else {
result = jl_interpret_toplevel_thunk(m, thk);
// use interpreter
if (thk != NULL)
result = jl_interpret_toplevel_thunk(m, thk);
else if (jl_is_toplevel_only_expr((jl_value_t*)ex))
result = jl_toplevel_eval(m, (jl_value_t*)ex);
else
result = jl_interpret_toplevel_expr_in(m, (jl_value_t*)ex, NULL, NULL);
}

JL_GC_POP();
return result;
}
Expand All @@ -733,8 +736,9 @@ JL_DLLEXPORT jl_value_t *jl_toplevel_eval(jl_module_t *m, jl_value_t *v)

JL_DLLEXPORT jl_value_t *jl_infer_thunk(jl_code_info_t *thk, jl_module_t *m)
{
jl_method_instance_t *li = jl_new_thunk(thk, m);
jl_method_instance_t *li = method_instance_for_thunk(thk, m);
JL_GC_PUSH1(&li);
jl_resolve_globals_in_ir((jl_array_t*)thk->code, m, NULL, 0);
jl_type_infer(&li, jl_get_ptls_states()->world_age, 0);
JL_GC_POP();
return li->rettype;
Expand Down
4 changes: 2 additions & 2 deletions test/misc.jl
Original file line number Diff line number Diff line change
Expand Up @@ -20,10 +20,10 @@
@test isempty(sprint(io->warn(io, "testonce", once=true)))
@test !isempty(sprint(io->warn(io, "testonce", once=true, key=hash("testonce",hash("testanother")))))
let bt = backtrace()
ws = split(chomp(sprint(warn, "test", bt)), '\n')
ws = split(chomp(sprint(io->warn(io, "test", bt = bt))), '\n')
bs = split(chomp(sprint(Base.show_backtrace, bt)), '\n')
@test contains(ws[1],"WARNING: test")
for (l,b) in zip(ws[2:end],bs)
for (l,b) in zip(ws[2:end],bs[2:end])
@test contains(l, b)
end
end
Expand Down

0 comments on commit b02cf6e

Please sign in to comment.