Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

release-1.9: Backports for julia 1.9.0-alpha2 / 1.9.0-beta1 #47602

Merged
merged 57 commits into from
Dec 27, 2022
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
57 commits
Select commit Hold shift + click to select a range
05c8f3b
Limit initial OpenBLAS thread count (#46844)
staticfloat Nov 15, 2022
9606388
fix #46778, precompile() for abstract but compileable signatures (#47…
JeffBezanson Nov 16, 2022
0e17295
fix #47410, syntax error with anonfn inside `elseif` and short-circui…
JeffBezanson Nov 16, 2022
9dfd76b
ensure bindings handle write barriers for ty and globalref (#47580)
vtjnash Nov 16, 2022
d85e9ac
Turn on Intel jitevents by default on Linux (#47586)
vchuravy Nov 20, 2022
b5d6b03
Doc: The default sorting alg. is stable from 1.9 (#47579)
petvana Nov 21, 2022
3bc94a9
update MPFR (#47659)
simonbyrne Nov 21, 2022
f8f0c63
Add compat note for `sortperm(x; dims)` (#47657)
mcabbott Nov 22, 2022
382661c
build: add get-lld target (#47589)
petvana Nov 23, 2022
0f271d7
Print the detailed type on heap snapshot (#47503)
gbaraldi Nov 23, 2022
c615a49
Remove typeinfer lock altogether (#46825)
pchintalapudi Nov 23, 2022
3b414c6
Fix regression in generic_bitcast with Union{} arguments. (#47605)
maleadt Nov 24, 2022
aba112c
Filesystem: `rm(; recursive=true)` should ignore `UV_EACCES` (#47668)
staticfloat Nov 24, 2022
24505fc
Fix overflow in pow5 (#47511)
LilithHafner Nov 26, 2022
6fb45be
Fix GCExt test (#47699)
vchuravy Nov 27, 2022
0865ae0
Fix REPL keybinding CTRL-Q for stdlib methods (#47637)
rashidrafeek Nov 28, 2022
cfbb86a
fix 5-arg `mul!` for vectors of vectors (#47665)
ranocha Nov 28, 2022
93587d7
Add support for "package extensions" to code loading (#47695)
KristofferC Dec 7, 2022
dc1369e
🤖 Bump the Pkg stdlib from ed6a5497e to 5d8b9ddb8 (#47828)
DilumAluthgeBot Dec 7, 2022
9b9a703
Fix and simplify inference timing logic (#47711)
pchintalapudi Nov 26, 2022
1b85c1f
Fix nth_methtable tparam of -1 when n==0 (#47666)
apaz-cli Nov 28, 2022
81f8582
Fix generator-invocation legality check for varargs generators (#47739)
Keno Nov 29, 2022
9da50d2
Bump libuv. (#47707)
maleadt Nov 30, 2022
fcf02e8
Provider cycleclock() for 32bit ARM targets (#47358)
fingolfin Nov 30, 2022
628c953
strengthen setglobal to default to release-consume ordering (#47742)
vtjnash Nov 30, 2022
dfec160
fix unescaping in `global` expressions (#47719)
simeonschaub Dec 1, 2022
d8cbffd
Refactor and document sorting dispatch (#47383)
LilithHafner Dec 3, 2022
930314e
Comment out test in subtype that causes hang due to StackOverflow(#47…
vchuravy Dec 3, 2022
01ae8b7
Prioritize build_dir for generated headers (#47783)
vchuravy Dec 4, 2022
dba443d
Set `OPENBLAS_NUM_THREADS=1` on local Distributed workers (#47803)
staticfloat Dec 5, 2022
ce7a372
Add native julia fmod (#47501)
gbaraldi Dec 6, 2022
712a123
Fix libjulia install name and libjulia-internal rpath on OS X (#47220)
jonathan-conder-sm Dec 7, 2022
d561447
Replace the `.ji` serialization with sysimage format
timholy Feb 7, 2022
f8a5cd6
Allow re-initialization and caching of foreign types (#47407)
vchuravy Nov 27, 2022
12a4863
Fix physical_memory exports. (#47859)
maleadt Dec 10, 2022
2866e26
Fix missing GC root in Symbol construction (#47865)
Keno Dec 11, 2022
0b845b1
TOML: print: handle mixed vector of dicts and non-dicts (#47876)
fonsp Dec 12, 2022
3d80653
Fixups for #47383 (fixes `runbenchmarks("sort")`) (#47822)
LilithHafner Dec 13, 2022
a548ee3
fixes for jl_rewrap_unionall
vtjnash Nov 30, 2022
31df7c8
call specialized method instance when encountering unspecialized sparams
vtjnash Nov 19, 2022
9827f1d
ensure sparams are cached correctly for widened methods
vtjnash Nov 21, 2022
a506f43
ensure types are UnionAll wrapped are cached correctly for widened Va…
vtjnash Nov 22, 2022
3e1373e
add back wordaround for `Slot objects should not occur in an AST` in …
KristofferC Dec 14, 2022
5848e99
Reduce invalidations when loading JuliaData packages (#47889)
timholy Dec 15, 2022
86e8ef9
intersect: fix a minor soundness issue with supertypes (#47813)
vtjnash Dec 16, 2022
8432d4f
make Ctrl-C during sleeping work better (#47901)
JeffBezanson Dec 16, 2022
eba98e5
revert promotions of abstract arrays inside other arrays (#47893)
KristofferC Dec 19, 2022
a16ffd6
Revert "Emit safepoints at function entry (#41616)"
Dec 20, 2022
5a684f0
Revert "improve performance issue of `@nospecialize`-d keyword func c…
Dec 20, 2022
327e081
only load extensions once dependencies have finished loading (#47927)
KristofferC Dec 19, 2022
95cb3a8
Precompile cache: always add worklist CIs (#47924)
timholy Dec 20, 2022
f17d1df
put back the old QuickSort, PartialQuickSort, and MergeSort algorithm…
LilithHafner Dec 20, 2022
3ea7f6c
add bounds check to Slices indexing (#47622)
mcabbott Dec 20, 2022
22789c0
Reduce codegen lock scope (#46836)
pchintalapudi Dec 13, 2022
9a592dd
Implement support for object caching through pkgimages (#47184)
vchuravy Dec 27, 2022
7880930
Apply `InitialOptimizations` more consistently in sorting & fix dispa…
LilithHafner Dec 22, 2022
6adc428
Restore libgcc_s symlinkin in !macOS (#47986)
antonio-rojas Dec 24, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
ensure types are UnionAll wrapped are cached correctly for widened Va…
…rarg methods

And fix a related accuracy issue in jl_isa_compileable_sig

Follow-up issue found while working on #47476

(cherry picked from commit 9e5e28f)
  • Loading branch information
vtjnash authored and KristofferC committed Dec 14, 2022
commit a506f4393f8321a834d57c7c271d0641b67d5554
167 changes: 108 additions & 59 deletions src/gf.c
Original file line number Diff line number Diff line change
Expand Up @@ -606,6 +606,46 @@ jl_value_t *jl_nth_slot_type(jl_value_t *sig, size_t i) JL_NOTSAFEPOINT
// return 1;
//}

static jl_value_t *inst_varargp_in_env(jl_value_t *decl, jl_svec_t *sparams)
{
jl_value_t *unw = jl_unwrap_unionall(decl);
jl_value_t *vm = jl_tparam(unw, jl_nparams(unw) - 1);
assert(jl_is_vararg(vm));
int nsp = jl_svec_len(sparams);
if (nsp > 0 && jl_has_free_typevars(vm)) {
JL_GC_PUSH1(&vm);
assert(jl_subtype_env_size(decl) == nsp);
vm = jl_instantiate_type_in_env(vm, (jl_unionall_t*)decl, jl_svec_data(sparams));
assert(jl_is_vararg(vm));
// rewrap_unionall(lastdeclt, sparams) if any sparams isa TypeVar
// for example, `Tuple{Vararg{Union{Nothing,Int,Val{T}}}} where T`
// and the user called it with `Tuple{Vararg{Union{Nothing,Int},N}}`, then T is unbound
jl_value_t **sp = jl_svec_data(sparams);
while (jl_is_unionall(decl)) {
jl_tvar_t *v = (jl_tvar_t*)*sp;
if (jl_is_typevar(v)) {
// must unwrap and re-wrap Vararg object explicitly here since jl_type_unionall handles it differently
jl_value_t *T = ((jl_vararg_t*)vm)->T;
jl_value_t *N = ((jl_vararg_t*)vm)->N;
int T_has_tv = T && jl_has_typevar(T, v);
int N_has_tv = N && jl_has_typevar(N, v); // n.b. JL_VARARG_UNBOUND check means this should be false
assert(!N_has_tv || N == (jl_value_t*)v);
if (T_has_tv)
vm = jl_type_unionall(v, T);
if (N_has_tv)
N = NULL;
vm = (jl_value_t*)jl_wrap_vararg(vm, N); // this cannot throw for these inputs
}
sp++;
decl = ((jl_unionall_t*)decl)->body;
nsp--;
}
assert(nsp == 0);
JL_GC_POP();
}
return vm;
}

static jl_value_t *ml_matches(jl_methtable_t *mt,
jl_tupletype_t *type, int lim, int include_ambiguous,
int intersections, size_t world, int cache_result,
Expand Down Expand Up @@ -634,10 +674,12 @@ static void jl_compilation_sig(
assert(jl_is_tuple_type(tt));
size_t i, np = jl_nparams(tt);
size_t nargs = definition->nargs; // == jl_nparams(jl_unwrap_unionall(decl));
jl_value_t *type_i = NULL;
JL_GC_PUSH1(&type_i);
for (i = 0; i < np; i++) {
jl_value_t *elt = jl_tparam(tt, i);
jl_value_t *decl_i = jl_nth_slot_type(decl, i);
jl_value_t *type_i = jl_rewrap_unionall(decl_i, decl);
type_i = jl_rewrap_unionall(decl_i, decl);
size_t i_arg = (i < nargs - 1 ? i : nargs - 1);

if (jl_is_kind(type_i)) {
Expand Down Expand Up @@ -779,55 +821,45 @@ static void jl_compilation_sig(
// supertype of any other method signatures. so far we are conservative
// and the types we find should be bigger.
if (jl_nparams(tt) >= nspec && jl_va_tuple_kind((jl_datatype_t*)decl) == JL_VARARG_UNBOUND) {
jl_svec_t *limited = jl_alloc_svec(nspec);
JL_GC_PUSH1(&limited);
if (!*newparams) *newparams = tt->parameters;
size_t i;
for (i = 0; i < nspec - 1; i++) {
jl_svecset(limited, i, jl_svecref(*newparams, i));
}
jl_value_t *lasttype = jl_svecref(*newparams, i - 1);
// if all subsequent arguments are subtypes of lasttype, specialize
type_i = jl_svecref(*newparams, nspec - 2);
// if all subsequent arguments are subtypes of type_i, specialize
// on that instead of decl. for example, if decl is
// (Any...)
// and type is
// (Symbol, Symbol, Symbol)
// then specialize as (Symbol...), but if type is
// (Symbol, Int32, Expr)
// then specialize as (Any...)
size_t j = i;
size_t j = nspec - 1;
int all_are_subtypes = 1;
for (; j < jl_svec_len(*newparams); j++) {
jl_value_t *paramj = jl_svecref(*newparams, j);
if (jl_is_vararg(paramj))
paramj = jl_unwrap_vararg(paramj);
if (!jl_subtype(paramj, lasttype)) {
if (!jl_subtype(paramj, type_i)) {
all_are_subtypes = 0;
break;
}
}
if (all_are_subtypes) {
// avoid Vararg{Type{Type{...}}}
if (jl_is_type_type(lasttype) && jl_is_type_type(jl_tparam0(lasttype)))
lasttype = (jl_value_t*)jl_type_type;
jl_svecset(limited, i, jl_wrap_vararg(lasttype, (jl_value_t*)NULL));
if (jl_is_type_type(type_i) && jl_is_type_type(jl_tparam0(type_i)))
type_i = (jl_value_t*)jl_type_type;
type_i = (jl_value_t*)jl_wrap_vararg(type_i, (jl_value_t*)NULL); // this cannot throw for these inputs
}
else {
jl_value_t *unw = jl_unwrap_unionall(decl);
jl_value_t *lastdeclt = jl_tparam(unw, jl_nparams(unw) - 1);
assert(jl_is_vararg(lastdeclt));
int nsp = jl_svec_len(sparams);
if (nsp > 0 && jl_has_free_typevars(lastdeclt)) {
assert(jl_subtype_env_size(decl) == nsp);
lastdeclt = jl_instantiate_type_in_env(lastdeclt, (jl_unionall_t*)decl, jl_svec_data(sparams));
// TODO: rewrap_unionall(lastdeclt, sparams) if any sparams isa TypeVar???
// TODO: if we made any replacements above, sparams may now be incorrect
}
jl_svecset(limited, i, lastdeclt);
type_i = inst_varargp_in_env(decl, sparams);
}
jl_svec_t *limited = jl_alloc_svec(nspec);
size_t i;
for (i = 0; i < nspec - 1; i++) {
jl_svecset(limited, i, jl_svecref(*newparams, i));
}
jl_svecset(limited, i, type_i);
*newparams = limited;
JL_GC_POP();
}
JL_GC_POP();
}

// compute whether this type signature is a possible return value from jl_compilation_sig given a concrete-type for `tt`
Expand Down Expand Up @@ -865,56 +897,58 @@ JL_DLLEXPORT int jl_isa_compileable_sig(
jl_methtable_t *kwmt = mt == jl_kwcall_mt ? jl_kwmethod_table_for(decl) : mt;
if ((jl_value_t*)mt != jl_nothing) {
// try to refine estimate of min and max
if (kwmt && kwmt != jl_type_type_mt && kwmt != jl_nonfunction_mt && kwmt != jl_kwcall_mt)
if (kwmt != NULL && kwmt != jl_type_type_mt && kwmt != jl_nonfunction_mt && kwmt != jl_kwcall_mt)
// new methods may be added, increasing nspec_min later
nspec_min = kwmt->max_args + 2 + 2 * (mt == jl_kwcall_mt);
else
// nspec is always nargs+1, regardless of the other contents of these mt
nspec_max = nspec_min;
}
int isbound = (jl_va_tuple_kind((jl_datatype_t*)decl) == JL_VARARG_UNBOUND);
int isunbound = (jl_va_tuple_kind((jl_datatype_t*)decl) == JL_VARARG_UNBOUND);
if (jl_is_vararg(jl_tparam(type, np - 1))) {
if (!isbound || np < nspec_min || np > nspec_max)
if (!isunbound || np < nspec_min || np > nspec_max)
return 0;
}
else {
if (np < nargs - 1 || (isbound && np >= nspec_max))
if (np < nargs - 1 || (isunbound && np >= nspec_max))
return 0;
}
}
else if (np != nargs || jl_is_vararg(jl_tparam(type, np - 1))) {
return 0;
}

jl_value_t *type_i = NULL;
JL_GC_PUSH1(&type_i);
for (i = 0; i < np; i++) {
jl_value_t *elt = jl_tparam(type, i);
jl_value_t *decl_i = jl_nth_slot_type((jl_value_t*)decl, i);
jl_value_t *type_i = jl_rewrap_unionall(decl_i, decl);
size_t i_arg = (i < nargs - 1 ? i : nargs - 1);

if (jl_is_vararg(elt)) {
elt = jl_unwrap_vararg(elt);
if (jl_has_free_typevars(decl_i)) {
// TODO: in this case, answer semi-conservatively that these varargs are always compilable
// we don't have the ability to get sparams, so deciding if elt
// is a potential result of jl_instantiate_type_in_env for decl_i
// for any sparams that is consistent with the rest of the arguments
// seems like it would be extremely difficult
// and hopefully the upstream code probably gave us something reasonable
continue;
}
else if (jl_egal(elt, decl_i)) {
continue;
type_i = inst_varargp_in_env(decl, sparams);
if (jl_has_free_typevars(type_i)) {
JL_GC_POP();
return 0; // something went badly wrong?
}
else if (jl_is_type_type(elt) && jl_is_type_type(jl_tparam0(elt))) {
return 0;
if (jl_egal(elt, type_i))
continue; // elt could be chosen by inst_varargp_in_env for these sparams
elt = jl_unwrap_vararg(elt);
if (jl_is_type_type(elt) && jl_is_type_type(jl_tparam0(elt))) {
JL_GC_POP();
return 0; // elt would be set equal to jl_type_type instead
}
// else, it needs to meet the usual rules
// else, elt also needs to meet the usual rules
}

jl_value_t *decl_i = jl_nth_slot_type(decl, i);
type_i = jl_rewrap_unionall(decl_i, decl);

if (i_arg > 0 && i_arg <= sizeof(definition->nospecialize) * 8 &&
(definition->nospecialize & (1 << (i_arg - 1)))) {
if (!jl_has_free_typevars(decl_i) && !jl_is_kind(decl_i)) {
if (jl_egal(elt, decl_i))
continue;
JL_GC_POP();
return 0;
}
}
Expand All @@ -923,10 +957,12 @@ JL_DLLEXPORT int jl_isa_compileable_sig(
// kind slots always get guard entries (checking for subtypes of Type)
if (jl_subtype(elt, type_i) && !jl_subtype((jl_value_t*)jl_type_type, type_i))
continue;
// TODO: other code paths that could reach here
// TODO: other code paths that could reach here?
JL_GC_POP();
return 0;
}
else if (jl_is_kind(type_i)) {
JL_GC_POP();
return 0;
}

Expand All @@ -938,22 +974,31 @@ JL_DLLEXPORT int jl_isa_compileable_sig(
continue;
if (i >= nargs && definition->isva)
continue;
JL_GC_POP();
return 0;
}
if (!iscalled && very_general_type(type_i))
if (!iscalled && very_general_type(type_i)) {
JL_GC_POP();
return 0;
if (!jl_is_datatype(elt))
}
if (!jl_is_datatype(elt)) {
JL_GC_POP();
return 0;
}

// if the declared type was not Any or Union{Type, ...},
// then the match must been with kind, such as UnionAll or DataType,
// and the result of matching the type signature
// needs to be corrected to the concrete type 'kind' (and not to Type)
jl_value_t *kind = jl_typeof(jl_tparam0(elt));
if (kind == jl_bottom_type)
if (kind == jl_bottom_type) {
JL_GC_POP();
return 0; // Type{Union{}} gets normalized to typeof(Union{})
if (jl_subtype(kind, type_i) && !jl_subtype((jl_value_t*)jl_type_type, type_i))
}
if (jl_subtype(kind, type_i) && !jl_subtype((jl_value_t*)jl_type_type, type_i)) {
JL_GC_POP();
return 0; // gets turned into a kind
}

else if (jl_is_type_type(jl_tparam0(elt)) &&
// give up on specializing static parameters for Type{Type{Type{...}}}
Expand All @@ -966,20 +1011,20 @@ JL_DLLEXPORT int jl_isa_compileable_sig(
this can be determined using a type intersection.
*/
if (i < nargs || !definition->isva) {
jl_value_t *di = jl_type_intersection(type_i, (jl_value_t*)jl_type_type);
JL_GC_PUSH1(&di);
assert(di != (jl_value_t*)jl_bottom_type);
if (jl_is_kind(di)) {
type_i = jl_type_intersection(type_i, (jl_value_t*)jl_type_type);
assert(type_i != (jl_value_t*)jl_bottom_type);
if (jl_is_kind(type_i)) {
JL_GC_POP();
return 0;
}
else if (!jl_types_equal(di, elt)) {
else if (!jl_types_equal(type_i, elt)) {
JL_GC_POP();
return 0;
}
JL_GC_POP();
continue;
}
else {
JL_GC_POP();
return 0;
}
}
Expand All @@ -1000,12 +1045,16 @@ JL_DLLEXPORT int jl_isa_compileable_sig(
// when called with a subtype of Function but is not called
if (elt == (jl_value_t*)jl_function_type)
continue;
JL_GC_POP();
return 0;
}

if (!jl_is_concrete_type(elt))
if (!jl_is_concrete_type(elt)) {
JL_GC_POP();
return 0;
}
}
JL_GC_POP();
return 1;
}

Expand Down
9 changes: 9 additions & 0 deletions test/core.jl
Original file line number Diff line number Diff line change
Expand Up @@ -7889,3 +7889,12 @@ code_typed(f47476, (Int, Int, Int, Int, Vararg{Union{Int, NTuple{2,Int}}},))
vect47476(::Type{T}) where {T} = T
@test vect47476(Type{Type{Type{Int32}}}) === Type{Type{Type{Int32}}}
@test vect47476(Type{Type{Type{Int64}}}) === Type{Type{Type{Int64}}}

g47476(::Union{Nothing,Int,Val{T}}...) where {T} = T
@test_throws UndefVarError(:T) g47476(nothing, 1, nothing, 2, nothing, 3, nothing, 4, nothing, 5)
@test g47476(nothing, 1, nothing, 2, nothing, 3, nothing, 4, nothing, 5, Val(6)) === 6
let spec = only(methods(g47476)).specializations
@test !isempty(spec)
@test any(mi -> mi !== nothing && Base.isvatuple(mi.specTypes), spec)
@test all(mi -> mi === nothing || !Base.has_free_typevars(mi.specTypes), spec)
end