Skip to content

Commit

Permalink
re-structure micro-optimizations of typemap
Browse files Browse the repository at this point in the history
I believe this should better match the usage patterns
and is able to unroll more cases and more effectively use that information
  • Loading branch information
vtjnash committed May 19, 2016
1 parent 8498363 commit f6ae227
Show file tree
Hide file tree
Showing 5 changed files with 96 additions and 85 deletions.
6 changes: 3 additions & 3 deletions src/dump.c
Original file line number Diff line number Diff line change
Expand Up @@ -1993,9 +1993,9 @@ static void jl_restore_system_image_from_stream(ios_t *f)
jl_top_module = (jl_module_t*)jl_deserialize_value(f, NULL);
jl_internal_main_module = jl_main_module;
jl_typeinf_func = (jl_function_t*)jl_deserialize_value(f, NULL);
jl_type_type->name->mt = (jl_methtable_t*)jl_deserialize_value(f, NULL);
jl_typector_type->name->mt = jl_uniontype_type->name->mt = jl_datatype_type->name->mt =
jl_type_type->name->mt;
jl_type_type_mt = (jl_methtable_t*)jl_deserialize_value(f, NULL);
jl_type_type->name->mt = jl_typector_type->name->mt = jl_uniontype_type->name->mt = jl_datatype_type->name->mt =
jl_type_type_mt;

jl_core_module = (jl_module_t*)jl_get_global(jl_main_module,
jl_symbol("Core"));
Expand Down
19 changes: 14 additions & 5 deletions src/gf.c
Original file line number Diff line number Diff line change
Expand Up @@ -106,7 +106,7 @@ const struct jl_typemap_info tfunc_cache = {

static int8_t jl_cachearg_offset(jl_methtable_t *mt)
{
return (mt == jl_type_type->name->mt) ? 0 : 1;
return (mt == jl_type_type_mt) ? 0 : 1;
}

/// ----- Insertion logic for special entries ----- ///
Expand Down Expand Up @@ -1581,10 +1581,20 @@ JL_DLLEXPORT jl_value_t *jl_apply_generic(jl_value_t **args, uint32_t nargs)
jl_method_error((jl_function_t*)F, args, nargs);
// unreachable
}
jl_value_t *res;
if (mfunc->inInference || mfunc->inCompile) {
// if inference is running on this function, return a copy
// of the function to be compiled without inference and run.
res = jl_call_unspecialized(mfunc->sparam_vals, jl_get_unspecialized(mfunc), args, nargs);
}
else {
res = jl_call_method_internal(mfunc, args, nargs);
}
JL_GC_POP();
return verify_type(res);
}
else {
mfunc = entry->func.linfo;
}

mfunc = entry->func.linfo;
#ifdef JL_TRACE
if (traceen)
jl_printf(JL_STDOUT, " at %s:%d\n", jl_symbol_name(mfunc->file), mfunc->line);
Expand All @@ -1598,7 +1608,6 @@ JL_DLLEXPORT jl_value_t *jl_apply_generic(jl_value_t **args, uint32_t nargs)
else {
res = jl_call_method_internal(mfunc, args, nargs);
}
if (tt) JL_GC_POP();
return verify_type(res);
}

Expand Down
4 changes: 3 additions & 1 deletion src/jltypes.c
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ extern "C" {

jl_datatype_t *jl_any_type;
jl_datatype_t *jl_type_type;
jl_methtable_t *jl_type_type_mt;
jl_datatype_t *jl_typename_type;
jl_datatype_t *jl_sym_type;
jl_datatype_t *jl_symbol_type;
Expand Down Expand Up @@ -3263,7 +3264,8 @@ void jl_init_types(void)
jl_any_type = jl_new_abstracttype((jl_value_t*)jl_symbol("Any"), NULL, jl_emptysvec);
jl_any_type->super = jl_any_type;
jl_type_type = jl_new_abstracttype((jl_value_t*)jl_symbol("Type"), jl_any_type, jl_emptysvec);
jl_type_type->name->mt = jl_new_method_table(jl_type_type->name->name, jl_current_module);
jl_type_type_mt = jl_new_method_table(jl_type_type->name->name, jl_current_module);
jl_type_type->name->mt = jl_type_type_mt;

// initialize them. lots of cycles.
jl_datatype_type->name = jl_new_typename(jl_symbol("DataType"));
Expand Down
17 changes: 16 additions & 1 deletion src/julia_internal.h
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,9 @@
extern "C" {
#endif

// useful constants
extern jl_methtable_t *jl_type_type_mt;

// execution of certain certain unpure
// statements is prohibited from certain
// callbacks (such as generated functions)
Expand Down Expand Up @@ -623,7 +626,19 @@ jl_typemap_entry_t *jl_typemap_insert(union jl_typemap_t *cache, jl_value_t *par
jl_typemap_entry_t *jl_typemap_assoc_by_type(union jl_typemap_t ml_or_cache, jl_tupletype_t *types, jl_svec_t **penv,
int8_t subtype_inexact__sigseq_useenv, int8_t subtype, int8_t offs);
static jl_typemap_entry_t *const INEXACT_ENTRY = (jl_typemap_entry_t*)(uintptr_t)-1;
jl_typemap_entry_t *jl_typemap_assoc_exact(union jl_typemap_t ml_or_cache, jl_value_t **args, size_t n, int8_t offs);
jl_typemap_entry_t *jl_typemap_level_assoc_exact(jl_typemap_level_t *cache, jl_value_t **args, size_t n, int8_t offs);
jl_typemap_entry_t *jl_typemap_entry_assoc_exact(jl_typemap_entry_t *mn, jl_value_t **args, size_t n);
STATIC_INLINE jl_typemap_entry_t *jl_typemap_assoc_exact(union jl_typemap_t ml_or_cache, jl_value_t **args, size_t n, int8_t offs)
{
// NOTE: This function is a huge performance hot spot!!
if (jl_typeof(ml_or_cache.unknown) == (jl_value_t*)jl_typemap_entry_type) {
return jl_typemap_entry_assoc_exact(ml_or_cache.leaf, args, n);
}
else if (jl_typeof(ml_or_cache.unknown) == (jl_value_t*)jl_typemap_level_type) {
return jl_typemap_level_assoc_exact(ml_or_cache.node, args, n, offs);
}
return NULL;
}

typedef int (*jl_typemap_visitor_fptr)(jl_typemap_entry_t *l, void *closure);
int jl_typemap_visitor(union jl_typemap_t a, jl_typemap_visitor_fptr fptr, void *closure);
Expand Down
135 changes: 60 additions & 75 deletions src/typemap.c
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
#include <unistd.h>
#endif

#define MAX_METHLIST_COUNT 24 // this can strongly affect the sysimg size and speed!
#define MAX_METHLIST_COUNT 12 // this can strongly affect the sysimg size and speed!
#define INIT_CACHE_SIZE 8 // must be a power-of-two

#ifdef __cplusplus
Expand Down Expand Up @@ -686,9 +686,35 @@ jl_typemap_entry_t *jl_typemap_assoc_by_type(union jl_typemap_t ml_or_cache, jl_
}
}

static jl_typemap_entry_t *jl_typemap_node_assoc_exact(jl_typemap_entry_t *ml, jl_value_t **args, size_t n)
jl_typemap_entry_t *jl_typemap_entry_assoc_exact(jl_typemap_entry_t *ml, jl_value_t **args, size_t n)
{
// some manually-unrolled common special cases
jl_typemap_entry_t *prev = NULL, *first = ml;
while (ml->simplesig == (void*)jl_nothing && ml->guardsigs == jl_emptysvec && ml->isleafsig) {
// use a tight loop for a long as possible
if (n == jl_datatype_nfields(ml->sig) && jl_typeof(args[0]) == jl_tparam(ml->sig, 0)) {
if (n == 1)
goto matchnm;
if (n == 2) {
if (jl_typeof(args[1]) == jl_tparam(ml->sig, 1))
goto matchnm;
}
else if (n == 3) {
if (jl_typeof(args[1]) == jl_tparam(ml->sig, 1) &&
jl_typeof(args[2]) == jl_tparam(ml->sig, 2))
goto matchnm;
}
else {
if (sig_match_leaf(args, jl_svec_data(ml->sig->parameters), n))
goto matchnm;
}
}
prev = ml;
ml = ml->next;
if (ml == (void*)jl_nothing)
return NULL;
}

while (ml != (void*)jl_nothing) {
size_t lensig = jl_datatype_nfields(ml->sig);
if (lensig == n || (ml->va && lensig <= n+1)) {
Expand Down Expand Up @@ -728,94 +754,53 @@ static jl_typemap_entry_t *jl_typemap_node_assoc_exact(jl_typemap_entry_t *ml, j
}
}
}
if (prev != NULL && ml->isleafsig && first->next != ml) {
// LRU queue: move ml from prev->next to first->next
prev->next = ml->next;
jl_gc_wb(prev, prev->next);
ml->next = first->next;
jl_gc_wb(ml, ml->next);
first->next = ml;
jl_gc_wb(first, first->next);
}
return ml;
goto matchnm;
}
nomatch:
prev = ml;
ml = ml->next;
}
return NULL;
matchnm:
if (prev != NULL && ml->isleafsig && first->next != ml) {
// LRU queue: move ml from prev->next to first->next
prev->next = ml->next;
jl_gc_wb(prev, prev->next);
ml->next = first->next;
jl_gc_wb(ml, ml->next);
first->next = ml;
jl_gc_wb(first, first->next);
}
return ml;
}

jl_typemap_entry_t *jl_typemap_assoc_exact(union jl_typemap_t ml_or_cache, jl_value_t **args, size_t n, int8_t offs)
jl_typemap_entry_t *jl_typemap_level_assoc_exact(jl_typemap_level_t *cache, jl_value_t **args, size_t n, int8_t offs)
{
// NOTE: This function is a huge performance hot spot!!
if (jl_typeof(ml_or_cache.unknown) == (jl_value_t*)jl_typemap_level_type) {
jl_typemap_level_t *cache = ml_or_cache.node;
if (n > offs) {
jl_value_t *a1 = args[offs];
jl_value_t *ty = (jl_value_t*)jl_typeof(a1);
assert(jl_is_datatype(ty));
if (ty == (jl_value_t*)jl_datatype_type && cache->targ != (void*)jl_nothing) {
ml_or_cache = mtcache_hash_lookup(cache->targ, a1, 1, offs);
jl_typemap_entry_t *ml = jl_typemap_assoc_exact(ml_or_cache, args, n, offs+1);
if (ml) return ml;
}
if (cache->arg1 != (void*)jl_nothing) {
ml_or_cache = mtcache_hash_lookup(cache->arg1, ty, 0, offs);
if (jl_typeof(ml_or_cache.unknown) == (jl_value_t*)jl_typemap_entry_type) {
if (offs < 2 && n > 1) {
// some manually-unrolled common special cases
jl_value_t *a0 = args[1 - offs];
jl_value_t *t0 = (jl_value_t*)jl_typeof(a0);
jl_typemap_entry_t *mn = ml_or_cache.leaf;
if (mn->simplesig == (void*)jl_nothing && mn->guardsigs == jl_emptysvec &&
n == jl_datatype_nfields(mn->sig) && mn->isleafsig) {
if (n == 2) {
if (jl_tparam(ml_or_cache.leaf->sig, 1 - offs) == t0)
return ml_or_cache.leaf;
}
else if (n == 3) {
jl_value_t *a2 = args[2];
if (jl_tparam(mn->sig, 1 - offs) == t0 &&
jl_tparam(mn->sig, 2) == (jl_value_t*)jl_typeof(a2))
return mn;
}
else {
if (sig_match_leaf(args, jl_svec_data(mn->sig->parameters), n))
return mn;
}
mn = mn->next;
if (n == 3 && mn != (void*)jl_nothing &&
mn->simplesig == (void*)jl_nothing && mn->guardsigs == jl_emptysvec &&
n == jl_datatype_nfields(mn->sig) && mn->isleafsig) {
if (jl_tparam(mn->sig, 1 - offs) == t0 &&
jl_tparam(mn->sig, 2) == (jl_value_t*)jl_typeof(args[2]))
return mn;
}
}
}
jl_typemap_entry_t *ml = jl_typemap_node_assoc_exact(ml_or_cache.leaf, args, n);
if (ml) return ml;
}
else if (jl_typeof(ml_or_cache.unknown) == (jl_value_t*)jl_typemap_level_type) {
jl_typemap_entry_t *ml = jl_typemap_assoc_exact(ml_or_cache, args, n, offs+1);
if (ml) return ml;
}
}
if (n > offs) {
jl_value_t *a1 = args[offs];
jl_value_t *ty = (jl_value_t*)jl_typeof(a1);
assert(jl_is_datatype(ty));
if (ty == (jl_value_t*)jl_datatype_type && cache->targ != (void*)jl_nothing) {
union jl_typemap_t ml_or_cache = mtcache_hash_lookup(cache->targ, a1, 1, offs);
jl_typemap_entry_t *ml = jl_typemap_assoc_exact(ml_or_cache, args, n, offs+1);
if (ml) return ml;
}
if (cache->arg1 != (void*)jl_nothing) {
union jl_typemap_t ml_or_cache = mtcache_hash_lookup(cache->arg1, ty, 0, offs);
jl_typemap_entry_t *ml = jl_typemap_assoc_exact(ml_or_cache, args, n, offs+1);
if (ml) return ml;
}
jl_typemap_entry_t *ml = jl_typemap_node_assoc_exact(cache->linear, args, n);
if (ml) return ml;
if (jl_typeof(ml_or_cache.unknown) != (jl_value_t*)jl_nothing)
return jl_typemap_assoc_exact(cache->any, args, n, offs+1);
return NULL;
}
else {
return jl_typemap_node_assoc_exact(ml_or_cache.leaf, args, n);
if (jl_typeof(cache->linear) != (jl_value_t*)jl_nothing) {
jl_typemap_entry_t *ml = jl_typemap_entry_assoc_exact(cache->linear, args, n);
if (ml) return ml;
}
if (jl_typeof(cache->any.unknown) != (jl_value_t*)jl_nothing)
return jl_typemap_assoc_exact(cache->any, args, n, offs+1);
return NULL;
}



// ----- Method List Insertion Management ----- //

static unsigned jl_typemap_list_count(jl_typemap_entry_t *ml)
Expand Down

0 comments on commit f6ae227

Please sign in to comment.