diff options
Diffstat (limited to 'lib/compiler')
| -rw-r--r-- | lib/compiler/src/beam_ssa_type.erl | 16 | ||||
| -rw-r--r-- | lib/compiler/src/v3_core.erl | 3 | ||||
| -rw-r--r-- | lib/compiler/test/beam_type_SUITE.erl | 14 | ||||
| -rw-r--r-- | lib/compiler/test/fun_SUITE.erl | 20 | 
4 files changed, 45 insertions, 8 deletions
| diff --git a/lib/compiler/src/beam_ssa_type.erl b/lib/compiler/src/beam_ssa_type.erl index 06b42f1928..2ea7e89dfa 100644 --- a/lib/compiler/src/beam_ssa_type.erl +++ b/lib/compiler/src/beam_ssa_type.erl @@ -24,7 +24,7 @@  -include("beam_ssa_opt.hrl").  -import(lists, [all/2,any/2,droplast/1,foldl/3,last/1,member/2,                  keyfind/3,partition/2,reverse/1,reverse/2, -                seq/2,sort/1,split/2]). +                sort/1,split/2]).  -define(UNICODE_INT, #t_integer{elements={0,16#10FFFF}}). @@ -874,11 +874,11 @@ type(call, [#b_remote{mod=#b_literal{val=Mod},                          true ->                              none                      end; -                {#t_integer{elements={Min,Max}}, +                {#t_integer{elements={Min,_}}=IntType,                   #t_tuple{elements=Es0,size=Size}=T} -> -                    %% We know this will land between Min and Max, so kill the -                    %% types for those indexes. -                    Es = maps:without(seq(Min, Max), Es0), +                    %% Remove type information for all indices that +                    %% falls into the range of the integer. +                    Es = remove_element_info(IntType, Es0),                      case T#t_tuple.exact of                          false ->                              T#t_tuple{elements=Es,size=max(Min, Size)}; @@ -1649,6 +1649,12 @@ get_literal_from_type(nil) ->      #b_literal{val=[]};  get_literal_from_type(_) -> none. +remove_element_info(#t_integer{elements={Min,Max}}, Es) -> +    foldl(fun(El, Acc) when Min =< El, El =< Max -> +                  maps:remove(El, Acc); +             (_El, Acc) -> Acc +          end, Es, maps:keys(Es)). +  t_atom() ->      #t_atom{elements=any}. diff --git a/lib/compiler/src/v3_core.erl b/lib/compiler/src/v3_core.erl index 3699c9d22e..007a0247f4 100644 --- a/lib/compiler/src/v3_core.erl +++ b/lib/compiler/src/v3_core.erl @@ -1811,7 +1811,8 @@ force_safe(Ce, St0) ->  is_safe(#c_cons{}) -> true;  is_safe(#c_tuple{}) -> true; -is_safe(#c_var{}) -> true; +is_safe(#c_var{name={_,_}}) -> false;           %Fun. Not safe. +is_safe(#c_var{name=_}) -> true;                %Ordinary variable.  is_safe(#c_literal{}) -> true;  is_safe(_) -> false. diff --git a/lib/compiler/test/beam_type_SUITE.erl b/lib/compiler/test/beam_type_SUITE.erl index 882e281a44..0d1680fb15 100644 --- a/lib/compiler/test/beam_type_SUITE.erl +++ b/lib/compiler/test/beam_type_SUITE.erl @@ -271,8 +271,22 @@ setelement(_Config) ->      T0 = id({a,42}),      {a,_} = T0,      {b,_} = setelement(1, T0, b), +    {z,b} = do_setelement_1(<<(id(1)):32>>, {a,b}, z), +    {new,two} = do_setelement_2(<<(id(1)):1>>, {one,two}, new),      ok. +do_setelement_1(<<N:32>>, Tuple, NewValue) -> +    _ = element(N, Tuple), +    %% While updating the type for Tuple, beam_ssa_type would do: +    %%   maps:without(lists:seq(0, 4294967295), Elements) +    setelement(N, Tuple, NewValue). + +do_setelement_2(<<N:1>>, Tuple, NewValue) -> +    %% Cover the second clause in remove_element_info/2. The +    %% type for the second element will be kept. +    two = element(2, Tuple), +    setelement(N, Tuple, NewValue). +  cons(_Config) ->      [did] = cons(assigned, did), diff --git a/lib/compiler/test/fun_SUITE.erl b/lib/compiler/test/fun_SUITE.erl index 1df0a05275..7fc6195e31 100644 --- a/lib/compiler/test/fun_SUITE.erl +++ b/lib/compiler/test/fun_SUITE.erl @@ -22,7 +22,8 @@  -export([all/0, suite/0,groups/0,init_per_suite/1, end_per_suite/1,   	 init_per_group/2,end_per_group/2,  	 test1/1,overwritten_fun/1,otp_7202/1,bif_fun/1, -         external/1,eep37/1,eep37_dup/1,badarity/1,badfun/1]). +         external/1,eep37/1,eep37_dup/1,badarity/1,badfun/1, +         duplicated_fun/1]).  %% Internal exports.  -export([call_me/1,dup1/0,dup2/0]). @@ -37,7 +38,7 @@ all() ->  groups() ->      [{p,[parallel],        [test1,overwritten_fun,otp_7202,bif_fun,external,eep37, -       eep37_dup,badarity,badfun]}]. +       eep37_dup,badarity,badfun,duplicated_fun]}].  init_per_suite(Config) ->      test_lib:recompile(?MODULE), @@ -261,5 +262,20 @@ badfun(_Config) ->  expect_badfun(Term, Exit) ->      {'EXIT',{{badfun,Term},_}} = Exit. +duplicated_fun(_Config) -> +    try +        %% The following code used to crash the compiler before +        %% v3_core:is_safe/1 was corrected to consider fun variables +        %% unsafe. +        id([print_result_paths_fun = fun duplicated_fun_helper/1]), +        ct:error(should_fail) +    catch +        error:{badmatch,F} when is_function(F, 1) -> +            ok +    end. + +duplicated_fun_helper(_) -> +    ok. +  id(I) ->      I. | 
