From 49610584c72d727565dec88ca867c27fec952e74 Mon Sep 17 00:00:00 2001 From: Jeff Bezanson Date: Mon, 8 May 2017 22:43:00 -0400 Subject: [PATCH 1/2] speed up dispatch of functions with type parameters This restores the ability to start splitting typemaps at either argument 0 or 1, depending on whether the function type has parameters. --- base/serialize.jl | 3 +++ src/datatype.c | 3 +++ src/dump.c | 7 ++----- src/gf.c | 3 +-- src/jltypes.c | 17 ++++++++++++----- src/julia.h | 1 + 6 files changed, 22 insertions(+), 12 deletions(-) diff --git a/base/serialize.jl b/base/serialize.jl index 6a6847a850e90..1b86281cfd9b4 100644 --- a/base/serialize.jl +++ b/base/serialize.jl @@ -956,6 +956,9 @@ function deserialize_typename(s::AbstractSerializer, number) maxa = deserialize(s)::Int if makenew tn.mt = ccall(:jl_new_method_table, Any, (Any, Any), name, tn.module) + if isempty(parameters) + tn.mt.offs = 1 + end tn.mt.name = mtname tn.mt.max_args = maxa for def in defs diff --git a/src/datatype.c b/src/datatype.c index e97a26ad53bf1..9eec032d1f565 100644 --- a/src/datatype.c +++ b/src/datatype.c @@ -46,6 +46,7 @@ JL_DLLEXPORT jl_methtable_t *jl_new_method_table(jl_sym_t *name, jl_module_t *mo mt->kwsorter = NULL; mt->backedges = NULL; JL_MUTEX_INIT(&mt->writelock); + mt->offs = 0; return mt; } @@ -395,6 +396,8 @@ JL_DLLEXPORT jl_datatype_t *jl_new_datatype( if (!abstract) { tn->mt = jl_new_method_table(name, module); jl_gc_wb(tn, tn->mt); + if (jl_svec_len(parameters) == 0) + tn->mt->offs = 1; } } t->name = tn; diff --git a/src/dump.c b/src/dump.c index e25b7f5b110f7..88094baa39b06 100644 --- a/src/dump.c +++ b/src/dump.c @@ -2001,8 +2001,7 @@ static void jl_reinit_item(jl_value_t *v, int how, arraylist_t *tracee_list) case 3: { // rehash MethodTable jl_methtable_t *mt = (jl_methtable_t*)v; jl_typemap_rehash(mt->defs, 0); - // TODO: consider reverting this when we can split on Type{...} better - jl_typemap_rehash(mt->cache, 1); //(mt == jl_type_typename->mt) ? 0 : 1); + jl_typemap_rehash(mt->cache, mt->offs); if (tracee_list) arraylist_push(tracee_list, mt); break; @@ -2672,7 +2671,7 @@ void jl_init_serializer(void) jl_emptysvec, jl_emptytuple, jl_false, jl_true, jl_nothing, jl_any_type, call_sym, invoke_sym, goto_ifnot_sym, return_sym, body_sym, line_sym, - lambda_sym, jl_symbol("tuple"), assign_sym, isdefined_sym, + lambda_sym, jl_symbol("tuple"), assign_sym, isdefined_sym, jl_box_uint8(0), jl_box_uint8(1), // empirical list of very common symbols #include "common_symbols1.inc" @@ -2690,7 +2689,6 @@ void jl_init_serializer(void) jl_box_int32(30), jl_box_int32(31), jl_box_int32(32), #ifndef _P64 jl_box_int32(33), jl_box_int32(34), jl_box_int32(35), - jl_box_int32(36), jl_box_int32(37), #endif jl_box_int64(0), jl_box_int64(1), jl_box_int64(2), jl_box_int64(3), jl_box_int64(4), jl_box_int64(5), @@ -2705,7 +2703,6 @@ void jl_init_serializer(void) jl_box_int64(30), jl_box_int64(31), jl_box_int64(32), #ifdef _P64 jl_box_int64(33), jl_box_int64(34), jl_box_int64(35), - jl_box_int64(36), jl_box_int64(37), #endif jl_labelnode_type, jl_linenumbernode_type, jl_gotonode_type, jl_quotenode_type, jl_type_type, jl_bottom_type, jl_ref_type, diff --git a/src/gf.c b/src/gf.c index 3cdbb33b4c18b..3dbcf4ce8d792 100644 --- a/src/gf.c +++ b/src/gf.c @@ -134,8 +134,7 @@ const struct jl_typemap_info tfunc_cache = { static int8_t jl_cachearg_offset(jl_methtable_t *mt) { - // TODO: consider reverting this when we can split on Type{...} better - return 1; //(mt == jl_type_type_mt) ? 0 : 1; + return mt->offs; } /// ----- Insertion logic for special entries ----- /// diff --git a/src/jltypes.c b/src/jltypes.c index 45a7c25a92e50..bb291271d97a6 100644 --- a/src/jltypes.c +++ b/src/jltypes.c @@ -1609,6 +1609,7 @@ void jl_init_types(void) jl_typename_type->name = jl_new_typename_in(jl_symbol("TypeName"), core); jl_typename_type->name->wrapper = (jl_value_t*)jl_typename_type; jl_typename_type->name->mt = jl_new_method_table(jl_typename_type->name->name, core); + jl_typename_type->name->mt->offs = 1; jl_typename_type->super = jl_any_type; jl_typename_type->parameters = jl_emptysvec; jl_typename_type->name->names = jl_perm_symsvec(8, "name", "module", @@ -1629,15 +1630,17 @@ void jl_init_types(void) jl_methtable_type->name = jl_new_typename_in(jl_symbol("MethodTable"), core); jl_methtable_type->name->wrapper = (jl_value_t*)jl_methtable_type; jl_methtable_type->name->mt = jl_new_method_table(jl_methtable_type->name->name, core); + jl_methtable_type->name->mt->offs = 1; jl_methtable_type->super = jl_any_type; jl_methtable_type->parameters = jl_emptysvec; - jl_methtable_type->name->names = jl_perm_symsvec(9, "name", "defs", + jl_methtable_type->name->names = jl_perm_symsvec(10, "name", "defs", "cache", "max_args", "kwsorter", "module", - "backedges", "", ""); - jl_methtable_type->types = jl_svec(9, jl_sym_type, jl_any_type, jl_any_type, jl_any_type/*jl_long*/, + "backedges", "", "", "offs"); + jl_methtable_type->types = jl_svec(10, jl_sym_type, jl_any_type, jl_any_type, jl_any_type/*jl_long*/, jl_any_type, jl_any_type/*module*/, - jl_any_type/*any vector*/, jl_any_type/*long*/, jl_any_type/*int32*/); + jl_any_type/*any vector*/, jl_any_type/*long*/, jl_any_type/*int32*/, + jl_any_type/*uint8*/); jl_methtable_type->uid = jl_assign_type_uid(); jl_methtable_type->instance = NULL; jl_methtable_type->struct_decl = NULL; @@ -1649,6 +1652,7 @@ void jl_init_types(void) jl_sym_type->name = jl_new_typename_in(jl_symbol("Symbol"), core); jl_sym_type->name->wrapper = (jl_value_t*)jl_sym_type; jl_sym_type->name->mt = jl_new_method_table(jl_sym_type->name->name, core); + jl_sym_type->name->mt->offs = 1; jl_sym_type->super = jl_any_type; jl_sym_type->parameters = jl_emptysvec; jl_sym_type->name->names = jl_emptysvec; @@ -1665,6 +1669,7 @@ void jl_init_types(void) jl_simplevector_type->name = jl_new_typename_in(jl_symbol("SimpleVector"), core); jl_simplevector_type->name->wrapper = (jl_value_t*)jl_simplevector_type; jl_simplevector_type->name->mt = jl_new_method_table(jl_simplevector_type->name->name, core); + jl_simplevector_type->name->mt->offs = 1; jl_simplevector_type->super = jl_any_type; jl_simplevector_type->parameters = jl_emptysvec; jl_simplevector_type->name->names = jl_emptysvec; @@ -2056,10 +2061,12 @@ void jl_init_types(void) jl_svecset(jl_methtable_type->types, 6, jl_array_any_type); #ifdef __LP64__ jl_svecset(jl_methtable_type->types, 7, jl_int64_type); // unsigned long + jl_svecset(jl_methtable_type->types, 8, jl_int64_type); // uint32_t plus alignment #else jl_svecset(jl_methtable_type->types, 7, jl_int32_type); // DWORD -#endif jl_svecset(jl_methtable_type->types, 8, jl_int32_type); // uint32_t +#endif + jl_svecset(jl_methtable_type->types, 9, jl_uint8_type); jl_svecset(jl_method_type->types, 10, jl_method_instance_type); jl_svecset(jl_method_type->types, 11, jl_method_instance_type); jl_svecset(jl_method_instance_type->types, 12, jl_voidpointer_type); diff --git a/src/julia.h b/src/julia.h index af2ac4db5dc71..60e1d069b0332 100644 --- a/src/julia.h +++ b/src/julia.h @@ -463,6 +463,7 @@ typedef struct _jl_methtable_t { jl_module_t *module; // used for incremental serialization to locate original binding jl_array_t *backedges; jl_mutex_t writelock; + uint8_t offs; // 0, or 1 to skip splitting typemap on first (function) argument } jl_methtable_t; typedef struct { From ecff62420dd7b57b3545667a38eef8f5e1fb99db Mon Sep 17 00:00:00 2001 From: Jeff Bezanson Date: Tue, 9 May 2017 01:01:11 -0400 Subject: [PATCH 2/2] speed up 0-arg constructor dispatch. fixes #21730 Uses the `any` cache to skip all non-leaf slots when a later slot is splittable. --- src/codegen.cpp | 2 +- src/gf.c | 6 +-- src/julia.h | 2 +- src/julia_internal.h | 1 + src/typemap.c | 102 ++++++++++++++++++++++++------------------- 5 files changed, 63 insertions(+), 50 deletions(-) diff --git a/src/codegen.cpp b/src/codegen.cpp index a800306b9a741..d7ad5073bb310 100644 --- a/src/codegen.cpp +++ b/src/codegen.cpp @@ -4652,7 +4652,7 @@ static Function *gen_cfun_wrapper(jl_function_t *ff, jl_value_t *jlrettype, jl_t } const struct jl_typemap_info cfunction_cache = { - 1, &jl_voidpointer_type + 1, 0, &jl_voidpointer_type }; // Get the LLVM Function* for the C-callable entry point for a certain function diff --git a/src/gf.c b/src/gf.c index 3dbcf4ce8d792..d03712f769db9 100644 --- a/src/gf.c +++ b/src/gf.c @@ -123,13 +123,13 @@ void jl_call_tracer(tracer_cb callback, jl_value_t *tracee) /// ----- Definitions for various internal TypeMaps ----- /// const struct jl_typemap_info method_defs = { - 0, &jl_method_type + 0, 0, &jl_method_type }; const struct jl_typemap_info lambda_cache = { - 0, &jl_method_instance_type + 0, 1, &jl_method_instance_type }; const struct jl_typemap_info tfunc_cache = { - 1, &jl_any_type + 1, 0, &jl_any_type }; static int8_t jl_cachearg_offset(jl_methtable_t *mt) diff --git a/src/julia.h b/src/julia.h index 60e1d069b0332..155a7c17ce2ae 100644 --- a/src/julia.h +++ b/src/julia.h @@ -448,7 +448,7 @@ typedef struct _jl_typemap_level_t { struct jl_ordereddict_t arg1; struct jl_ordereddict_t targ; jl_typemap_entry_t *linear; // union jl_typemap_t (but no more levels) - union jl_typemap_t any; // type at offs is Any + union jl_typemap_t any; // type at offs is skipped; will be split later jl_value_t *key; // [nullable] } jl_typemap_level_t; diff --git a/src/julia_internal.h b/src/julia_internal.h index 3c5b800156fa9..9de811242c89d 100644 --- a/src/julia_internal.h +++ b/src/julia_internal.h @@ -899,6 +899,7 @@ STATIC_INLINE void jl_free_aligned(void *p) // passed around as self-documentation of the parameters of the type struct jl_typemap_info { int8_t unsorted; // whether this should be unsorted + int8_t simplekeys; // whether keys are restricted to simple kinds of types used in method caches jl_datatype_t **jl_contains; // the type that is being put in this }; diff --git a/src/typemap.c b/src/typemap.c index 3f8898e894d1b..c0d98de2489cf 100644 --- a/src/typemap.c +++ b/src/typemap.c @@ -16,14 +16,6 @@ extern "C" { #endif -// compute whether the specificity of this type is equivalent to Any in the sort order -static int jl_is_any(jl_value_t *t1) -{ - return (t1 == (jl_value_t*)jl_any_type || - (jl_is_typevar(t1) && - ((jl_tvar_t*)t1)->ub == (jl_value_t*)jl_any_type)); -} - // ----- Type Signature Subtype Testing ----- // static int sig_match_by_type_leaf(jl_value_t **types, jl_tupletype_t *sig, size_t n) @@ -318,7 +310,8 @@ void jl_typemap_rehash_array(struct jl_ordereddict_t *pa, jl_value_t *parent, in } mtcache_rehash(pa, 4 * next_power_of_two(len), parent, tparam, offs); } -void jl_typemap_rehash(union jl_typemap_t ml, int8_t offs) { +void jl_typemap_rehash(union jl_typemap_t ml, int8_t offs) +{ if (jl_typeof(ml.unknown) == (jl_value_t*)jl_typemap_level_type) { if (ml.node->targ.values != (void*)jl_nothing) jl_typemap_rehash_array(&ml.node->targ, ml.unknown, 1, offs); @@ -408,9 +401,9 @@ int jl_typemap_visitor(union jl_typemap_t cache, jl_typemap_visitor_fptr fptr, v if (cache.node->arg1.values != (void*)jl_nothing) if (!jl_typemap_array_visitor(&cache.node->arg1, fptr, closure)) return 0; - if (!jl_typemap_node_visitor(cache.node->linear, fptr, closure)) + if (!jl_typemap_visitor(cache.node->any, fptr, closure)) return 0; - return jl_typemap_visitor(cache.node->any, fptr, closure); + return jl_typemap_node_visitor(cache.node->linear, fptr, closure); } else { return jl_typemap_node_visitor(cache.leaf, fptr, closure); @@ -419,7 +412,7 @@ int jl_typemap_visitor(union jl_typemap_t cache, jl_typemap_visitor_fptr fptr, v // predicate to fast-test if this type is a leaf type that can exist in the cache // and does not need a more expensive linear scan to find all intersections -int is_cache_leaf(jl_value_t *ty) +static int is_cache_leaf(jl_value_t *ty) { return (jl_is_datatype(ty) && ((jl_datatype_t*)ty)->uid != 0 && !jl_is_kind(ty)); } @@ -541,9 +534,9 @@ int jl_typemap_intersection_visitor(union jl_typemap_t map, int offs, } } } - if (!jl_typemap_intersection_node_visitor(map.node->linear, closure)) + if (!jl_typemap_intersection_visitor(map.node->any, offs+1, closure)) return 0; - return jl_typemap_intersection_visitor(map.node->any, offs+1, closure); + return jl_typemap_intersection_node_visitor(map.node->linear, closure); } else { return jl_typemap_intersection_node_visitor(map.leaf, closure); @@ -681,7 +674,6 @@ static jl_typemap_entry_t *jl_typemap_lookup_by_type_(jl_typemap_entry_t *ml, jl return NULL; } - // this is the general entry point for looking up a type in the cache // (as a subtype, or with typeseq) jl_typemap_entry_t *jl_typemap_assoc_by_type(union jl_typemap_t ml_or_cache, jl_tupletype_t *types, jl_svec_t **penv, @@ -711,8 +703,6 @@ jl_typemap_entry_t *jl_typemap_assoc_by_type(union jl_typemap_t ml_or_cache, jl_ } // If there is a type at offs, look in the optimized caches if (!subtype) { - if (ty && jl_is_any(ty)) - return jl_typemap_assoc_by_type(cache->any, types, penv, inexact, subtype, offs+1, world); if (isva) // in lookup mode, want to match Vararg exactly, not as a subtype ty = NULL; } @@ -739,15 +729,13 @@ jl_typemap_entry_t *jl_typemap_assoc_by_type(union jl_typemap_t ml_or_cache, jl_ } if (!subtype && is_cache_leaf(ty)) return NULL; } + jl_typemap_entry_t *li = jl_typemap_assoc_by_type(cache->any, types, penv, inexact, subtype, offs+1, world); + if (li) return li; // Always check the list (since offs doesn't always start at 0) - if (subtype) { - jl_typemap_entry_t *li = jl_typemap_assoc_by_type_(cache->linear, types, inexact, penv, world); - if (li) return li; - return jl_typemap_assoc_by_type(cache->any, types, penv, inexact, subtype, offs+1, world); - } - else { + if (subtype) + return jl_typemap_assoc_by_type_(cache->linear, types, inexact, penv, world); + else return jl_typemap_lookup_by_type_(cache->linear, types, world); - } } else { return subtype ? @@ -851,12 +839,14 @@ jl_typemap_entry_t *jl_typemap_level_assoc_exact(jl_typemap_level_t *cache, jl_v if (ml) return ml; } } + if (cache->any.unknown != jl_nothing) { + jl_typemap_entry_t *ml = jl_typemap_assoc_exact(cache->any, args, n, offs+1, world); + if (ml) return ml; + } if (cache->linear != (jl_typemap_entry_t*)jl_nothing) { jl_typemap_entry_t *ml = jl_typemap_entry_assoc_exact(cache->linear, args, n, world); if (ml) return ml; } - if (cache->any.unknown != jl_nothing) - return jl_typemap_assoc_exact(cache->any, args, n, offs+1, world); return NULL; } @@ -873,10 +863,6 @@ static unsigned jl_typemap_list_count(jl_typemap_entry_t *ml) return count; } -static void jl_typemap_level_insert_(jl_typemap_level_t *cache, jl_typemap_entry_t *newrec, int8_t offs, const struct jl_typemap_info *tparams); -static void jl_typemap_list_insert_sorted(jl_typemap_entry_t **pml, jl_value_t *parent, - jl_typemap_entry_t *newrec, const struct jl_typemap_info *tparams); - static jl_typemap_level_t *jl_new_typemap_level(void) { jl_ptls_t ptls = jl_get_ptls_states(); @@ -893,6 +879,24 @@ static jl_typemap_level_t *jl_new_typemap_level(void) return cache; } +static void jl_typemap_level_insert_(jl_typemap_level_t *cache, jl_typemap_entry_t *newrec, int8_t offs, const struct jl_typemap_info *tparams, int lastleaf); + +// get index of last type component that can be hash split +static int last_leaf_offs(jl_tupletype_t *types) +{ + jl_value_t *ttypes = jl_unwrap_unionall((jl_value_t*)types); + int offs, l = jl_field_count(ttypes); + + for (offs = l - 1; offs >= 0; offs--) { + jl_value_t *t1 = jl_tparam(ttypes, offs); + if (jl_is_vararg_type(t1)) + continue; + if (is_cache_leaf(t1) || (jl_is_type_type(t1) && is_cache_leaf(jl_tparam0(t1)))) + return offs; + } + return -1; +} + static jl_typemap_level_t *jl_method_convert_list_to_cache(jl_typemap_entry_t *ml, jl_value_t *key, int8_t offs, const struct jl_typemap_info *tparams) { @@ -903,13 +907,16 @@ static jl_typemap_level_t *jl_method_convert_list_to_cache(jl_typemap_entry_t *m while (ml != (void*)jl_nothing) { next = ml->next; ml->next = (jl_typemap_entry_t*)jl_nothing; - jl_typemap_level_insert_(cache, ml, offs, tparams); + jl_typemap_level_insert_(cache, ml, offs, tparams, last_leaf_offs(ml->sig)); ml = next; } JL_GC_POP(); return cache; } +static void jl_typemap_list_insert_sorted(jl_typemap_entry_t **pml, jl_value_t *parent, + jl_typemap_entry_t *newrec, const struct jl_typemap_info *tparams); + static void jl_typemap_list_insert_(jl_typemap_entry_t **pml, jl_value_t *parent, jl_typemap_entry_t *newrec, const struct jl_typemap_info *tparams) { @@ -926,10 +933,10 @@ static void jl_typemap_list_insert_(jl_typemap_entry_t **pml, jl_value_t *parent static void jl_typemap_insert_generic(union jl_typemap_t *pml, jl_value_t *parent, jl_typemap_entry_t *newrec, jl_value_t *key, int8_t offs, - const struct jl_typemap_info *tparams) + const struct jl_typemap_info *tparams, int lastleaf) { if (jl_typeof(pml->unknown) == (jl_value_t*)jl_typemap_level_type) { - jl_typemap_level_insert_(pml->node, newrec, offs, tparams); + jl_typemap_level_insert_(pml->node, newrec, offs, tparams, lastleaf); return; } @@ -937,7 +944,7 @@ static void jl_typemap_insert_generic(union jl_typemap_t *pml, jl_value_t *paren if (count > MAX_METHLIST_COUNT) { pml->node = jl_method_convert_list_to_cache(pml->leaf, key, offs, tparams); jl_gc_wb(parent, pml->node); - jl_typemap_level_insert_(pml->node, newrec, offs, tparams); + jl_typemap_level_insert_(pml->node, newrec, offs, tparams, lastleaf); return; } @@ -946,16 +953,16 @@ static void jl_typemap_insert_generic(union jl_typemap_t *pml, jl_value_t *paren static int jl_typemap_array_insert_(struct jl_ordereddict_t *cache, jl_value_t *key, jl_typemap_entry_t *newrec, jl_value_t *parent, int8_t tparam, int8_t offs, - const struct jl_typemap_info *tparams) + const struct jl_typemap_info *tparams, int lastleaf) { union jl_typemap_t *pml = mtcache_hash_bp(cache, key, tparam, offs, (jl_value_t*)parent); if (pml) - jl_typemap_insert_generic(pml, (jl_value_t*)cache->values, newrec, key, offs+1, tparams); + jl_typemap_insert_generic(pml, (jl_value_t*)cache->values, newrec, key, offs+1, tparams, lastleaf); return pml != NULL; } static void jl_typemap_level_insert_(jl_typemap_level_t *cache, jl_typemap_entry_t *newrec, int8_t offs, - const struct jl_typemap_info *tparams) + const struct jl_typemap_info *tparams, int lastleaf) { jl_value_t *ttypes = jl_unwrap_unionall((jl_value_t*)newrec->sig); size_t l = jl_field_count(ttypes); @@ -975,11 +982,6 @@ static void jl_typemap_level_insert_(jl_typemap_level_t *cache, jl_typemap_entry else if (l > offs) { t1 = jl_tparam(ttypes, offs); } - // If the type at `offs` is Any, put it in the Any list - if (t1 && jl_is_any(t1)) { - jl_typemap_insert_generic(&cache->any, (jl_value_t*)cache, newrec, (jl_value_t*)jl_any_type, offs+1, tparams); - return; - } // Don't put Varargs in the optimized caches (too hard to handle in lookup and bp) if (t1 && !isva) { // if t1 != jl_typetype_type and the argument is Type{...}, this @@ -987,12 +989,22 @@ static void jl_typemap_level_insert_(jl_typemap_level_t *cache, jl_typemap_entry // the table indexed for that purpose. if (t1 != (jl_value_t*)jl_typetype_type && jl_is_type_type(t1)) { jl_value_t *a0 = jl_tparam0(t1); - if (jl_typemap_array_insert_(&cache->targ, a0, newrec, (jl_value_t*)cache, 1, offs, tparams)) + if (jl_typemap_array_insert_(&cache->targ, a0, newrec, (jl_value_t*)cache, 1, offs, tparams, lastleaf)) return; } - if (jl_typemap_array_insert_(&cache->arg1, t1, newrec, (jl_value_t*)cache, 0, offs, tparams)) + if (jl_typemap_array_insert_(&cache->arg1, t1, newrec, (jl_value_t*)cache, 0, offs, tparams, lastleaf)) return; } + assert(offs != lastleaf); + if (tparams->unsorted || tparams->simplekeys) { + // if we couldn't split on this offset but can split on a later one, skip this slot. + // Only do this for unsorted maps, or maps with restricted type keys, since there are + // cases where an apparently non-leaf argument can be more specific than a leaf argument: + // getindex(a::ConjArray{T,N,A}, i::Vararg{Int64,N}) where {T, N} more specific than + // getindex(a::ConjArray{T,N,A}, i::Int64) where {T, N} + if (offs < lastleaf) + return jl_typemap_insert_generic(&cache->any, (jl_value_t*)cache, newrec, (jl_value_t*)jl_any_type, offs+1, tparams, lastleaf); + } jl_typemap_list_insert_(&cache->linear, (jl_value_t*)cache, newrec, tparams); } @@ -1056,7 +1068,7 @@ jl_typemap_entry_t *jl_typemap_insert(union jl_typemap_t *cache, jl_value_t *par newrec->isleafsig = newrec->issimplesig = 0; } // TODO: assert that guardsigs == jl_emptysvec && simplesig == jl_nothing if isleafsig and optimize with that knowledge? - jl_typemap_insert_generic(cache, parent, newrec, NULL, offs, tparams); + jl_typemap_insert_generic(cache, parent, newrec, NULL, offs, tparams, last_leaf_offs(newrec->sig)); JL_GC_POP(); return newrec; }