diff options
Diffstat (limited to 'lib')
| -rw-r--r-- | lib/compiler/src/beam_block.erl | 3 | ||||
| -rw-r--r-- | lib/compiler/src/beam_flatten.erl | 3 | ||||
| -rw-r--r-- | lib/compiler/src/beam_split.erl | 3 | ||||
| -rw-r--r-- | lib/compiler/src/beam_utils.erl | 14 | ||||
| -rw-r--r-- | lib/compiler/src/beam_validator.erl | 4 | ||||
| -rw-r--r-- | lib/compiler/src/cerl.erl | 4 | ||||
| -rw-r--r-- | lib/compiler/src/cerl_inline.erl | 36 | ||||
| -rw-r--r-- | lib/compiler/src/cerl_trees.erl | 4 | ||||
| -rw-r--r-- | lib/compiler/src/core_pp.erl | 4 | ||||
| -rw-r--r-- | lib/compiler/src/sys_pre_expand.erl | 11 | ||||
| -rw-r--r-- | lib/compiler/src/v3_codegen.erl | 73 | ||||
| -rw-r--r-- | lib/compiler/src/v3_core.erl | 432 | ||||
| -rw-r--r-- | lib/compiler/src/v3_kernel.erl | 61 | ||||
| -rw-r--r-- | lib/compiler/src/v3_kernel.hrl | 2 | ||||
| -rw-r--r-- | lib/compiler/test/error_SUITE.erl | 25 | ||||
| -rw-r--r-- | lib/compiler/test/map_SUITE.erl | 376 | ||||
| -rw-r--r-- | lib/compiler/test/warnings_SUITE.erl | 4 | ||||
| -rw-r--r-- | lib/debugger/test/map_SUITE.erl | 487 | ||||
| -rw-r--r-- | lib/stdlib/src/erl_eval.erl | 2 | ||||
| -rw-r--r-- | lib/stdlib/src/erl_lint.erl | 93 | ||||
| -rw-r--r-- | lib/stdlib/test/erl_eval_SUITE.erl | 24 | ||||
| -rw-r--r-- | lib/stdlib/test/erl_lint_SUITE.erl | 16 | 
22 files changed, 1332 insertions, 349 deletions
| diff --git a/lib/compiler/src/beam_block.erl b/lib/compiler/src/beam_block.erl index 7a30c68593..5626aa34ab 100644 --- a/lib/compiler/src/beam_block.erl +++ b/lib/compiler/src/beam_block.erl @@ -155,7 +155,8 @@ collect(remove_message)      -> {set,[],[],remove_message};  collect({put_map,F,Op,S,D,R,{list,Puts}}) ->      {set,[D],[S|Puts],{alloc,R,{put_map,Op,F}}};  collect({get_map_elements,F,S,{list,Gets}}) -> -    {set,Gets,[S],{get_map_elements,F}}; +    {Ss,Ds} = beam_utils:spliteven(Gets), +    {set,Ds,[S|Ss],{get_map_elements,F}};  collect({'catch',R,L})       -> {set,[R],[],{'catch',L}};  collect(fclearerror)         -> {set,[],[],fclearerror};  collect({fcheckerror,{f,0}}) -> {set,[],[],fcheckerror}; diff --git a/lib/compiler/src/beam_flatten.erl b/lib/compiler/src/beam_flatten.erl index 46835bece1..05d067dc48 100644 --- a/lib/compiler/src/beam_flatten.erl +++ b/lib/compiler/src/beam_flatten.erl @@ -63,7 +63,8 @@ norm({set,[],[S,D],{set_tuple_element,I}}) -> {set_tuple_element,S,D,I};  norm({set,[D1,D2],[S],get_list})          -> {get_list,S,D1,D2};  norm({set,[D],[S|Puts],{alloc,R,{put_map,Op,F}}}) ->      {put_map,F,Op,S,D,R,{list,Puts}}; -norm({set,Gets,[S],{get_map_elements,F}}) -> +norm({set,Ds,[S|Ss],{get_map_elements,F}}) -> +    Gets = beam_utils:joineven(Ss,Ds),      {get_map_elements,F,S,{list,Gets}};  norm({set,[],[],remove_message})   -> remove_message;  norm({set,[],[],fclearerror}) -> fclearerror; diff --git a/lib/compiler/src/beam_split.erl b/lib/compiler/src/beam_split.erl index 688bba9a94..f5dba314ae 100644 --- a/lib/compiler/src/beam_split.erl +++ b/lib/compiler/src/beam_split.erl @@ -53,8 +53,9 @@ split_block([{set,[D],[S|Puts],{alloc,R,{put_map,Op,{f,Lbl}=Fail}}}|Is],  	    Bl, Acc) when Lbl =/= 0 ->      split_block(Is, [], [{put_map,Fail,Op,S,D,R,{list,Puts}}|  			 make_block(Bl, Acc)]); -split_block([{set,Gets,[S],{get_map_elements,{f,Lbl}=Fail}}|Is], Bl, Acc) +split_block([{set,Ds,[S|Ss],{get_map_elements,{f,Lbl}=Fail}}|Is], Bl, Acc)    when Lbl =/= 0 -> +    Gets = beam_utils:joineven(Ss,Ds),      split_block(Is, [], [{get_map_elements,Fail,S,{list,Gets}}|make_block(Bl, Acc)]);  split_block([{set,[R],[],{'catch',L}}|Is], Bl, Acc) ->      split_block(Is, [], [{'catch',R,L}|make_block(Bl, Acc)]); diff --git a/lib/compiler/src/beam_utils.erl b/lib/compiler/src/beam_utils.erl index 8ca368c167..e82ba82d38 100644 --- a/lib/compiler/src/beam_utils.erl +++ b/lib/compiler/src/beam_utils.erl @@ -26,6 +26,8 @@  	 code_at/2,bif_to_test/3,is_pure_test/1,  	 live_opt/1,delete_live_annos/1,combine_heap_needs/2]). +-export([joineven/2,spliteven/1]). +  -import(lists, [member/2,sort/1,reverse/1,splitwith/2]).  -record(live, @@ -832,3 +834,15 @@ x_live([_|Rs], Regs) -> x_live(Rs, Regs);  x_live([], Regs) -> Regs.  is_live(X, Regs) -> ((Regs bsr X) band 1) =:= 1. + +%% spliteven/1 +%% [1,2,3,4,5,6] -> {[1,3,5],[2,4,6]} +spliteven(Rs) -> spliteven(Rs,[],[]). +spliteven([],Ss,Ds) -> {reverse(Ss),reverse(Ds)}; +spliteven([S,D|Rs],Ss,Ds) -> +    spliteven(Rs,[S|Ss],[D|Ds]). + +%% joineven/1 +%% {[1,3,5],[2,4,6]} -> [1,2,3,4,5,6] +joineven([],[]) -> []; +joineven([S|Ss],[D|Ds]) -> [S,D|joineven(Ss,Ds)]. diff --git a/lib/compiler/src/beam_validator.erl b/lib/compiler/src/beam_validator.erl index 9d5563d13b..0acc7a227f 100644 --- a/lib/compiler/src/beam_validator.erl +++ b/lib/compiler/src/beam_validator.erl @@ -1123,7 +1123,9 @@ assert_freg_set(Fr, _) -> error({bad_source,Fr}).  %%% Maps  %% ensure that a list of literals has a strict -%% ascending term order (also meaning unique literals) +%% ascending term order (also meaning unique literals). +%% Single item lists may have registers. +assert_strict_literal_termorder([_]) -> ok;  assert_strict_literal_termorder(Ls) ->      Vs = lists:map(fun (L) -> get_literal(L) end, Ls),      case check_strict_value_termorder(Vs) of diff --git a/lib/compiler/src/cerl.erl b/lib/compiler/src/cerl.erl index 9d6768b157..7a2c3d70de 100644 --- a/lib/compiler/src/cerl.erl +++ b/lib/compiler/src/cerl.erl @@ -3063,10 +3063,12 @@ pat_vars(Node, Vs) ->  	map ->  	    pat_list_vars(map_es(Node), Vs);  	map_pair -> -	    pat_list_vars([map_pair_op(Node),map_pair_key(Node),map_pair_val(Node)],Vs); +	    %% map_pair_key is not a pattern var, excluded +	    pat_list_vars([map_pair_op(Node),map_pair_val(Node)],Vs);  	binary ->  	    pat_list_vars(binary_segments(Node), Vs);  	bitstr -> +	    %% bitstr_size is not a pattern var, excluded  	    pat_vars(bitstr_val(Node), Vs);  	alias ->  	    pat_vars(alias_pat(Node), [alias_var(Node) | Vs]) diff --git a/lib/compiler/src/cerl_inline.erl b/lib/compiler/src/cerl_inline.erl index 75740e8b9d..f8489a800b 100644 --- a/lib/compiler/src/cerl_inline.erl +++ b/lib/compiler/src/cerl_inline.erl @@ -1341,23 +1341,23 @@ i_bitstr(E, Ren, Env, S) ->      S3 = count_size(weight(bitstr), S2),      {update_c_bitstr(E, Val, Size, Unit, Type, Flags), S3}. -i_map(E, Ctx, Ren, Env, S) -> +i_map(E, Ctx, Ren, Env, S0) ->      %% Visit the segments for value. -    {M1, S1} = i(map_arg(E), value, Ren, Env, S), +    {M1, S1} = i(map_arg(E), value, Ren, Env, S0),      {Es, S2} = mapfoldl(fun (E, S) ->  		i_map_pair(E, Ctx, Ren, Env, S)  	end, S1, map_es(E)),      S3 = count_size(weight(map), S2),      {update_c_map(E, M1,Es), S3}. -i_map_pair(E, Ctx, Ren, Env, S) -> -    %% It is not necessary to visit the Op and Key fields, -    %% since these are always literals. -    {Val, S1} = i(map_pair_val(E), Ctx, Ren, Env, S), +i_map_pair(E, Ctx, Ren, Env, S0) -> +    %% It is not necessary to visit the Op field +    %% since it is always a literal. +    {Key, S1} = i(map_pair_key(E), value, Ren, Env, S0), +    {Val, S2} = i(map_pair_val(E), Ctx, Ren, Env, S1),      Op = map_pair_op(E), -    Key = map_pair_key(E), -    S2 = count_size(weight(map_pair), S1), -    {update_c_map_pair(E, Op, Key, Val), S2}. +    S3 = count_size(weight(map_pair), S2), +    {update_c_map_pair(E, Op, Key, Val), S3}.  %% This is a simplified version of `i_pattern', for lists of parameter @@ -1420,15 +1420,11 @@ i_pattern(E, Ren, Env, Ren0, Env0, S) ->  	    S2 = count_size(weight(binary), S1),  	    {update_c_binary(E, Es), S2};  	map -> -	    %% map patterns should not have args -	    M = map_arg(E), -  	    {Es, S1} = mapfoldl(fun (E, S) ->  			i_map_pair_pattern(E, Ren, Env, Ren0, Env0, S) -		end, -		S, map_es(E)), +		end, S, map_es(E)),  	    S2 = count_size(weight(map), S1), -	    {update_c_map(E, M, Es), S2}; +	    {update_c_map(E, map_arg(E), Es), S2};  	_ ->  	    case is_literal(E) of  		true -> @@ -1464,12 +1460,12 @@ i_bitstr_pattern(E, Ren, Env, Ren0, Env0, S) ->  i_map_pair_pattern(E, Ren, Env, Ren0, Env0, S) ->      %% It is not necessary to visit the Op it is always a literal. -    %% Same goes for Key -    {Val, S1} = i_pattern(map_pair_val(E), Ren, Env, Ren0, Env0, S), +    %% Key is an expression +    {Key, S1} = i(map_pair_key(E), value, Ren0, Env0, S), +    {Val, S2} = i_pattern(map_pair_val(E), Ren, Env, Ren0, Env0, S1),      Op = map_pair_op(E), %% should be 'exact' literal -    Key  = map_pair_key(E), -    S2 = count_size(weight(map_pair), S1), -    {update_c_map_pair(E, Op, Key, Val), S2}. +    S3 = count_size(weight(map_pair), S2), +    {update_c_map_pair(E, Op, Key, Val), S3}.  %% --------------------------------------------------------------------- diff --git a/lib/compiler/src/cerl_trees.erl b/lib/compiler/src/cerl_trees.erl index e53bdd4efb..b93da8e97f 100644 --- a/lib/compiler/src/cerl_trees.erl +++ b/lib/compiler/src/cerl_trees.erl @@ -520,9 +520,9 @@ variables(T, S) ->  	tuple ->  	    vars_in_list(tuple_es(T), S);  	map -> -	    vars_in_list(map_es(T), S); +	    vars_in_list([map_arg(T)|map_es(T)], S);  	map_pair -> -	    vars_in_list([map_pair_op(T),map_pair_key(T), map_pair_val(T)], S); +	    vars_in_list([map_pair_op(T),map_pair_key(T),map_pair_val(T)], S);  	'let' ->  	    Vs = variables(let_body(T), S),  	    Vs1 = var_list_names(let_vars(T)), diff --git a/lib/compiler/src/core_pp.erl b/lib/compiler/src/core_pp.erl index 83412ecdd7..03801a9b6d 100644 --- a/lib/compiler/src/core_pp.erl +++ b/lib/compiler/src/core_pp.erl @@ -125,8 +125,8 @@ format_1(#c_literal{anno=A,val=M},Ctxt) when is_map(M) ->  	_ -> assoc      end,      Cpairs = [#c_map_pair{op=#c_literal{val=Op}, -			  key=#c_literal{val=V}, -			  val=#c_literal{val=K}} || {K,V} <- Pairs], +			  key=#c_literal{val=K}, +			  val=#c_literal{val=V}} || {K,V} <- Pairs],  	format_1(#c_map{anno=A,arg=#c_literal{val=#{}},es=Cpairs},Ctxt);  format_1(#c_var{name={I,A}}, _) ->      [core_atom(I),$/,integer_to_list(A)]; diff --git a/lib/compiler/src/sys_pre_expand.erl b/lib/compiler/src/sys_pre_expand.erl index 6410b73941..f99307c865 100644 --- a/lib/compiler/src/sys_pre_expand.erl +++ b/lib/compiler/src/sys_pre_expand.erl @@ -262,9 +262,18 @@ pattern({map,Line,Ps}, St0) ->      {TPs,St1} = pattern_list(Ps, St0),      {{map,Line,TPs},St1};  pattern({map_field_exact,Line,K0,V0}, St0) -> -    {K,St1} = expr(K0, St0), +    %% Key should be treated as an expression +    %% but since expressions are not allowed yet, +    %% process it through pattern .. and handle assoc +    %% (normalise unary op integer -> integer) +    {K,St1} = pattern(K0, St0),      {V,St2} = pattern(V0, St1),      {{map_field_exact,Line,K,V},St2}; +pattern({map_field_assoc,Line,K0,V0}, St0) -> +    %% when keys are Maps +    {K,St1} = pattern(K0, St0), +    {V,St2} = pattern(V0, St1), +    {{map_field_assoc,Line,K,V},St2};  %%pattern({struct,Line,Tag,Ps}, St0) ->  %%    {TPs,TPsvs,St1} = pattern_list(Ps, St0),  %%    {{tuple,Line,[{atom,Line,Tag}|TPs]},TPsvs,St1}; diff --git a/lib/compiler/src/v3_codegen.erl b/lib/compiler/src/v3_codegen.erl index 47a357c23d..8c1a0c08ac 100644 --- a/lib/compiler/src/v3_codegen.erl +++ b/lib/compiler/src/v3_codegen.erl @@ -210,7 +210,7 @@ need_heap_0([], H, Acc) ->  need_heap_1(#l{ke={set,_,{binary,_}},i=I}, H) ->      {need_heap_need(I, H),0}; -need_heap_1(#l{ke={set,_,{map,_,_}},i=I}, H) -> +need_heap_1(#l{ke={set,_,{map,_,_,_}},i=I}, H) ->      {need_heap_need(I, H),0};  need_heap_1(#l{ke={set,_,Val}}, H) ->      %% Just pass through adding to needed heap. @@ -643,10 +643,6 @@ select_val_cg(tuple, R, [Arity,{f,Lbl}], Tf, Vf, [{label,Lbl}|Sis]) ->      [{test,is_tuple,{f,Tf},[R]},{test,test_arity,{f,Vf},[R,Arity]}|Sis];  select_val_cg(tuple, R, Vls, Tf, Vf, Sis) ->      [{test,is_tuple,{f,Tf},[R]},{select_tuple_arity,R,{f,Vf},{list,Vls}}|Sis]; -select_val_cg(map, R, [_Val,{f,Lbl}], Fail, Fail, [{label,Lbl}|Sis]) -> -    [{test,is_map,{f,Fail},[R]}|Sis]; -select_val_cg(map, R, [_Val,{f,Lbl}|_], Tf, _Vf, [{label,Lbl}|Sis]) -> -    [{test,is_map,{f,Tf},[R]}|Sis];  select_val_cg(Type, R, [Val, {f,Lbl}], Fail, Fail, [{label,Lbl}|Sis]) ->      [{test,is_eq_exact,{f,Fail},[R,{Type,Val}]}|Sis];  select_val_cg(Type, R, [Val, {f,Lbl}], Tf, Vf, [{label,Lbl}|Sis]) -> @@ -947,27 +943,34 @@ select_extract_map(Src, Vs, Fail, I, Vdb, Bef, St) ->      %% Assume keys are term-sorted      Rsrc = fetch_var(Src, Bef), -    {{HasKs,GetVs},Aft} = lists:foldr(fun -	    ({map_pair,Key,{var,V}},{{HasKsi,GetVsi},Int0}) -> +    {{HasKs,GetVs,HasVarKs,GetVarVs},Aft} = lists:foldr(fun +	    ({map_pair,{var,K},{var,V}},{{HasKsi,GetVsi,HasVarVsi,GetVarVsi},Int0}) ->  		case vdb_find(V, Vdb) of  		    {V,_,L} when L =< I -> -			{{[Key|HasKsi],GetVsi},Int0}; +			RK = fetch_var(K,Int0), +			{{HasKsi,GetVsi,[RK|HasVarVsi],GetVarVsi},Int0};  		    _Other ->  			Reg1 = put_reg(V, Int0#sr.reg),  			Int1 = Int0#sr{reg=Reg1}, -			{{HasKsi,[Key,fetch_reg(V, Reg1)|GetVsi]},Int1} +			RK = fetch_var(K,Int0), +			RV = fetch_reg(V,Reg1), +			{{HasKsi,GetVsi,HasVarVsi,[[RK,RV]|GetVarVsi]},Int1} +		end; +	    ({map_pair,Key,{var,V}},{{HasKsi,GetVsi,HasVarVsi,GetVarVsi},Int0}) -> +		case vdb_find(V, Vdb) of +		    {V,_,L} when L =< I -> +			{{[Key|HasKsi],GetVsi,HasVarVsi,GetVarVsi},Int0}; +		    _Other -> +			Reg1 = put_reg(V, Int0#sr.reg), +			Int1 = Int0#sr{reg=Reg1}, +			{{HasKsi,[Key,fetch_reg(V, Reg1)|GetVsi],HasVarVsi,GetVarVsi},Int1}  		end -	end, {{[],[]},Bef}, Vs), - -    Code = case {HasKs,GetVs} of -	{HasKs,[]} -> -	    [{test,has_map_fields,{f,Fail},Rsrc,{list,HasKs}}]; -	{[],GetVs} -> -	    [{get_map_elements,   {f,Fail},Rsrc,{list,GetVs}}]; -	{HasKs,GetVs} -> -	    [{test,has_map_fields,{f,Fail},Rsrc,{list,HasKs}}, -	     {get_map_elements,   {f,Fail},Rsrc,{list,GetVs}}] -    end, +	end, {{[],[],[],[]},Bef}, Vs), + +    Code = [{test,has_map_fields,{f,Fail},Rsrc,{list,HasKs}} || HasKs =/= []] ++ +	   [{test,has_map_fields,{f,Fail},Rsrc,{list,[K]}}   || K <- HasVarKs] ++ +	   [{get_map_elements,   {f,Fail},Rsrc,{list,GetVs}} || GetVs =/= []] ++ +	   [{get_map_elements,   {f,Fail},Rsrc,{list,[K,V]}} || [K,V] <- GetVarVs],      {Code, Aft, St}. @@ -1504,9 +1507,39 @@ set_cg([{var,R}], {binary,Segs}, Le, Vdb, Bef,      %% Now generate the complete code for constructing the binary.      Code = cg_binary(PutCode, Target, Temp, Fail, MaxRegs, Le#l.a),      {Sis++Code,Aft,St}; +% Map single variable key +set_cg([{var,R}], {map,Op,Map,[{map_pair,{var,_}=K,V}]}, Le, Vdb, Bef, +       #cg{in_catch=InCatch,bfail=Bfail}=St) -> + +    Fail = {f,Bfail}, +    {Sis,Int0} = +	case InCatch of +	    true -> adjust_stack(Bef, Le#l.i, Le#l.i+1, Vdb); +	    false -> {[],Bef} +	end, +    SrcReg = cg_reg_arg(Map,Int0), +    Line = line(Le#l.a), + +    List = [cg_reg_arg(K,Int0),cg_reg_arg(V,Int0)], + +    Live = max_reg(Bef#sr.reg), +    Int1 = Int0#sr{reg=put_reg(R, Int0#sr.reg)}, +    Aft = clear_dead(Int1, Le#l.i, Vdb), +    Target = fetch_reg(R, Int1#sr.reg), + +    I = case Op of +	assoc -> put_map_assoc; +	exact -> put_map_exact +    end, +    {Sis++[Line]++[{I,Fail,SrcReg,Target,Live,{list,List}}],Aft,St}; + +% Map (possibly) multiple literal keys  set_cg([{var,R}], {map,Op,Map,Es}, Le, Vdb, Bef,         #cg{in_catch=InCatch,bfail=Bfail}=St) -> +    %% assert key literals +    [] = [Var||{map_pair,{var,_}=Var,_} <- Es], +      Fail = {f,Bfail},      {Sis,Int0} =  	case InCatch of diff --git a/lib/compiler/src/v3_core.erl b/lib/compiler/src/v3_core.erl index 83cf76f241..3d9fc3a609 100644 --- a/lib/compiler/src/v3_core.erl +++ b/lib/compiler/src/v3_core.erl @@ -78,7 +78,7 @@  -import(ordsets, [add_element/2,del_element/2,is_element/2,  		  union/1,union/2,intersection/2,subtract/2]).  -import(cerl, [ann_c_cons/3,ann_c_cons_skel/3,ann_c_tuple/2,c_tuple/1, -	       ann_c_map/2, ann_c_map/3]). +	       ann_c_map/3]).  -include("core_parse.hrl"). @@ -169,60 +169,81 @@ form({attribute,_,_,_}=F, {Fs,As,Ws,File}, _Opts) ->  attribute({attribute,Line,Name,Val}) ->      {#c_literal{val=Name, anno=[Line]}, #c_literal{val=Val, anno=[Line]}}. +%% function_dump(module_info,_,_,_) -> ok; +%% function_dump(Name,Arity,Format,Terms) -> +%%     io:format("~w/~w " ++ Format,[Name,Arity]++Terms), +%%     ok. +  function({function,_,Name,Arity,Cs0}, Ws0, File, Opts) -> -    %%ok = io:fwrite("~p - ", [{Name,Arity}]),      St0 = #core{vcount=0,opts=Opts,ws=Ws0,file=[{file,File}]},      {B0,St1} = body(Cs0, Name, Arity, St0), -    %%ok = io:fwrite("1", []), -    %%ok = io:fwrite("~w:~p~n", [?LINE,B0]), +    %% ok = function_dump(Name,Arity,"body:~n~p~n",[B0]),      {B1,St2} = ubody(B0, St1), -    %%ok = io:fwrite("2", []), -    %%ok = io:fwrite("~w:~p~n", [?LINE,B1]), +    %% ok = function_dump(Name,Arity,"ubody:~n~p~n",[B1]),      {B2,#core{ws=Ws}} = cbody(B1, St2), -    %%ok = io:fwrite("3~n", []), -    %%ok = io:fwrite("~w:~p~n", [?LINE,B2]), +    %% ok = function_dump(Name,Arity,"cbody:~n~p~n",[B2]),      {{#c_var{name={Name,Arity}},B2},Ws}.  body(Cs0, Name, Arity, St0) ->      Anno = lineno_anno(element(2, hd(Cs0)), St0),      {Args,St1} = new_vars(Anno, Arity, St0), -    {Cs1,St2} = clauses(Cs0, St1), -    {Ps,St3} = new_vars(Arity, St2),    %Need new variables here -    Fc = function_clause(Ps, Anno, {Name,Arity}), -    {#ifun{anno=#a{anno=Anno},id=[],vars=Args,clauses=Cs1,fc=Fc},St3}. +    case clauses(Cs0, St1) of +	{Cs1,[],St2} -> +	    {Ps,St3} = new_vars(Arity, St2),    %Need new variables here +	    Fc = function_clause(Ps, Anno, {Name,Arity}), +	    {#ifun{anno=#a{anno=Anno},id=[],vars=Args,clauses=Cs1,fc=Fc},St3}; +	{Cs1,Eps,St2} -> +	    %% We have pre-expressions from patterns and +	    %% these needs to be letified before matching +	    %% since only bound variables are allowed +	    AnnoGen = #a{anno=[compiler_generated]}, +	    {Ps1,St3} = new_vars(Arity, St2),    %Need new variables here +	    Fc1 = function_clause(Ps1, Anno, {Name,Arity}), +	    {Ps2,St4} = new_vars(Arity, St3),    %Need new variables here +	    Fc2 = function_clause(Ps2, Anno, {Name,Arity}), +	    Case = #icase{anno=AnnoGen,args=Args, +			  clauses=Cs1, +			  fc=Fc2}, +	    {#ifun{anno=#a{anno=Anno},id=[],vars=Args, +		   clauses=[#iclause{anno=AnnoGen,pats=Ps1, +				     guard=[#c_literal{val=true}], +				     body=Eps ++ [Case]}], +		   fc=Fc1},St4} +    end.  %% clause(Clause, State) -> {Cclause,State} | noclause.  %% clauses([Clause], State) -> {[Cclause],State}.  %%  Convert clauses.  Trap bad pattern aliases and remove clause from  %%  clause list. -clauses([C0|Cs0], St0) -> +clauses([C0|Cs0],St0) ->      case clause(C0, St0) of -	{noclause,St} -> clauses(Cs0, St); -	{C,St1} -> -	    {Cs,St2} = clauses(Cs0, St1), -	    {[C|Cs],St2} +	{noclause,_,St} -> clauses(Cs0,St); +	{C,Eps1,St1} -> +	    {Cs,Eps2,St2} = clauses(Cs0, St1), +	    {[C|Cs],Eps1++Eps2,St2}      end; -clauses([], St) -> {[],St}. +clauses([],St) -> {[],[],St}.  clause({clause,Lc,H0,G0,B0}, St0) ->      try head(H0, St0) of -	H1 -> -	    {G1,St1} = guard(G0, St0), -	    {B1,St2} = exprs(B0, St1), -            Anno = lineno_anno(Lc, St2), -	    {#iclause{anno=#a{anno=Anno},pats=H1,guard=G1,body=B1},St2} +	{H1,Eps,St1} -> +	    {G1,St2} = guard(G0, St1), +	    {B1,St3} = exprs(B0, St2), +            Anno = lineno_anno(Lc, St3), +	    {#iclause{anno=#a{anno=Anno},pats=H1,guard=G1,body=B1},Eps,St3}      catch  	throw:nomatch ->  	    St = add_warning(Lc, nomatch, St0), -	    {noclause,St}			%Bad pattern +	    {noclause,[],St}			%Bad pattern      end.  clause_arity({clause,_,H0,_,_}) -> length(H0). -%% head([P], State) -> [P]. +%% head([P], State) -> {[P],[Cexpr],State}. -head(Ps, St) -> pattern_list(Ps, St). +head(Ps, St) -> +    pattern_list(Ps, St).  %% guard([Expr], State) -> {[Cexpr],State}.  %%  Build an explict and/or tree of guard alternatives, then traverse @@ -514,22 +535,7 @@ expr({tuple,L,Es0}, St0) ->      A = record_anno(L, St1),      {annotate_tuple(A, Es1, St1),Eps,St1};  expr({map,L,Es0}, St0) -> -    % erl_lint should make sure only #{ K => V } are allowed -    % in map construction. -    try map_pair_list(Es0, St0) of -	{Es1,Eps,St1} -> -	    A = lineno_anno(L, St1), -	    {ann_c_map(A,Es1),Eps,St1} -    catch -	throw:{bad_map,Warning} -> -	    St = add_warning(L, Warning, St0), -	    LineAnno = lineno_anno(L, St), -	    As = [#c_literal{anno=LineAnno,val=badarg}], -	    {#icall{anno=#a{anno=LineAnno},	%Must have an #a{} -		    module=#c_literal{anno=LineAnno,val=erlang}, -		    name=#c_literal{anno=LineAnno,val=error}, -		    args=As},[],St} -    end; +    map_build_pair_chain(#c_literal{val=#{}},Es0,lineno_anno(L,St0),St0);  expr({map,L,M0,Es0}, St0) ->      try expr_map(M0,Es0,lineno_anno(L, St0),St0) of  	{_,_,_}=Res -> Res @@ -562,26 +568,26 @@ expr({block,_,Es0}, St0) ->      {E1,Eps,St2} = expr(last(Es0), St1),      {E1,Es1 ++ Eps,St2};  expr({'if',L,Cs0}, St0) -> -    {Cs1,St1} = clauses(Cs0, St0), +    {Cs1,Ceps,St1} = clauses(Cs0, St0),      Lanno = lineno_anno(L, St1),      Fc = fail_clause([], Lanno, #c_literal{val=if_clause}), -    {#icase{anno=#a{anno=Lanno},args=[],clauses=Cs1,fc=Fc},[],St1}; +    {#icase{anno=#a{anno=Lanno},args=[],clauses=Cs1,fc=Fc},Ceps,St1};  expr({'case',L,E0,Cs0}, St0) ->      {E1,Eps,St1} = novars(E0, St0), -    {Cs1,St2} = clauses(Cs0, St1), +    {Cs1,Ceps,St2} = clauses(Cs0, St1),      {Fpat,St3} = new_var(St2),      Lanno = lineno_anno(L, St2),      Fc = fail_clause([Fpat], Lanno, c_tuple([#c_literal{val=case_clause},Fpat])), -    {#icase{anno=#a{anno=Lanno},args=[E1],clauses=Cs1,fc=Fc},Eps,St3}; +    {#icase{anno=#a{anno=Lanno},args=[E1],clauses=Cs1,fc=Fc},Eps++Ceps,St3};  expr({'receive',L,Cs0}, St0) -> -    {Cs1,St1} = clauses(Cs0, St0), -    {#ireceive1{anno=#a{anno=lineno_anno(L, St1)},clauses=Cs1}, [], St1}; +    {Cs1,Ceps,St1} = clauses(Cs0, St0), +    {#ireceive1{anno=#a{anno=lineno_anno(L, St1)},clauses=Cs1},Ceps, St1};  expr({'receive',L,Cs0,Te0,Tes0}, St0) ->      {Te1,Teps,St1} = novars(Te0, St0),      {Tes1,St2} = exprs(Tes0, St1), -    {Cs1,St3} = clauses(Cs0, St2), +    {Cs1,Ceps,St3} = clauses(Cs0, St2),      {#ireceive2{anno=#a{anno=lineno_anno(L, St3)}, -		clauses=Cs1,timeout=Te1,action=Tes1},Teps,St3}; +		clauses=Cs1,timeout=Te1,action=Tes1},Teps++Ceps,St3};  expr({'try',L,Es0,[],Ecs,[]}, St0) ->      %% 'try ... catch ... end'      {Es1,St1} = exprs(Es0, St0), @@ -595,7 +601,7 @@ expr({'try',L,Es0,Cs0,Ecs,[]}, St0) ->      %% 'try ... of ... catch ... end'      {Es1,St1} = exprs(Es0, St0),      {V,St2} = new_var(St1),		%This name should be arbitrary -    {Cs1,St3} = clauses(Cs0, St2), +    {Cs1,Ceps,St3} = clauses(Cs0, St2),      {Fpat,St4} = new_var(St3),      Lanno = lineno_anno(L, St4),      Fc = fail_clause([Fpat], Lanno, @@ -604,7 +610,7 @@ expr({'try',L,Es0,Cs0,Ecs,[]}, St0) ->      {#itry{anno=#a{anno=lineno_anno(L, St5)},args=Es1,  	   vars=[V],body=[#icase{anno=#a{anno=Lanno},args=[V],clauses=Cs1,fc=Fc}],  	   evars=Evs,handler=Hs}, -     [],St5}; +     Ceps,St5};  expr({'try',L,Es0,[],[],As0}, St0) ->      %% 'try ... after ... end'      {Es1,St1} = exprs(Es0, St0), @@ -673,24 +679,24 @@ expr({match,L,P0,E0}, St0) ->  	      {var,_,'_'} -> St0#core{wanted=false};  	      _ -> St0  	  end, -    {E2,Eps,St2} = novars(E1, St1), +    {E2,Eps1,St2} = novars(E1, St1),      St3 = St2#core{wanted=St0#core.wanted}, -    P2 = try -	     pattern(P1, St3) +    {P2,Eps2,St4} = try +	    pattern(P1, St3)  	 catch  	     throw:Thrown -> -		 Thrown +		{Thrown,[],St3}  	 end, -    {Fpat,St4} = new_var(St3), -    Lanno = lineno_anno(L, St4), +    {Fpat,St5} = new_var(St4), +    Lanno = lineno_anno(L, St5),      Fc = fail_clause([Fpat], Lanno, c_tuple([#c_literal{val=badmatch},Fpat])),      case P2 of  	nomatch -> -	    St = add_warning(L, nomatch, St4), +	    St = add_warning(L, nomatch, St5),  	    {#icase{anno=#a{anno=Lanno}, -		    args=[E2],clauses=[],fc=Fc},Eps,St}; +		    args=[E2],clauses=[],fc=Fc},Eps1++Eps2,St};  	Other when not is_atom(Other) -> -	    {#imatch{anno=#a{anno=Lanno},pat=P2,arg=E2,fc=Fc},Eps,St4} +	    {#imatch{anno=#a{anno=Lanno},pat=P2,arg=E2,fc=Fc},Eps1++Eps2,St5}      end;  expr({op,_,'++',{lc,Llc,E,Qs0},More}, St0) ->      %% Optimise '++' here because of the list comprehension algorithm. @@ -772,44 +778,77 @@ expr_map(M0,Es0,A,St0) ->  		    Fc = fail_clause([Fpat], A, #c_literal{val=badarg}),  		    {#icase{anno=#a{anno=A},args=[M1],clauses=Cs,fc=Fc},Mps,St3};  		{_,_} -> -		    {Es1,Eps,St2} = map_pair_list(Es0, St1), -		    {ann_c_map(A,M1,Es1),Mps++Eps,St2} +		    {M2,Eps,St2} = map_build_pair_chain(M1,Es0,A,St1), +		    {M2,Mps++Eps,St2}  	    end;  	false -> throw({bad_map,bad_map})      end. +%% Group continuous literal blocks and single variables, i.e. +%% M0#{ a := 1, b := V1, K1 := V2, K2 := 42} +%% becomes equivalent to +%% M1 = M0#{ a := 1, b := V1 }, +%% M2 = M1#{ K1 := V1 }, +%% M3 = M2#{ K2 := 42 } + +map_build_pair_chain(M,Es,A,St) -> +    %% hack, remove iset if only literal +    case map_build_pair_chain(M,Es,A,St,[]) of +	{_,[#iset{arg=#c_literal{}=Val}],St1} -> {Val,[],St1}; +	Normal -> Normal +    end. + +map_build_pair_chain(M0,[],_,St,Mps) -> +    {M0,Mps,St}; +map_build_pair_chain(M0,Es0,A,St0,Mps) -> +    % group continuous literal blocks +    % Anno = #a{anno=[compiler_generated]}, +    % order is important, we need to reverse the literals +    case map_pair_block(Es0,[],[],St0) of +	{{CesL,EspL},{[],[]},Es1,St1} -> +	    {MVar,St2} = new_var(St1), +	    Pre = [#iset{var=MVar, arg=ann_c_map(A,M0,reverse(CesL))}], +	    map_build_pair_chain(MVar,Es1,A,St2,Mps++EspL++Pre); +	{{[],[]},{CesV,EspV},Es1,St1} -> +	    {MVar,St2} = new_var(St1), +	    Pre = [#iset{var=MVar, arg=#c_map{arg=M0,es=CesV, anno=A}}], +	    map_build_pair_chain(MVar,Es1,A,St2,Mps ++ EspV++Pre); +	{{CesL,EspL},{CesV,EspV},Es1,St1} -> +	    {MVarL,St2} = new_var(St1), +	    {MVarV,St3} = new_var(St2), +	    Pre = [#iset{var=MVarL, arg=ann_c_map(A,M0,reverse(CesL))}, +		   #iset{var=MVarV, arg=#c_map{arg=MVarL,es=CesV,anno=A}}], +	    map_build_pair_chain(MVarV,Es1,A,St3,Mps++EspL++EspV++Pre) +    end. + +map_pair_block([{Op,L,K0,V0}|Es],Ces,Esp,St0) -> +    {K,Ep0,St1} = safe(K0, St0), +    {V,Ep1,St2} = safe(V0, St1), +    A = lineno_anno(L, St2), +    Pair0 = map_op_to_c_map_pair(Op), +    Pair1 = Pair0#c_map_pair{anno=A,key=K,val=V}, +    case cerl:is_literal(K) of +	true -> +	    map_pair_block(Es,[Pair1|Ces],Ep0 ++ Ep1 ++ Esp,St2); +	false -> +	    {{Ces,Esp},{[Pair1],Ep0++Ep1},Es,St2} +    end; +map_pair_block([],Ces,Esp,St) -> +    {{Ces,Esp},{[],[]},[],St}. + +map_op_to_c_map_pair(map_field_assoc) -> #c_map_pair{op=#c_literal{val=assoc}}; +map_op_to_c_map_pair(map_field_exact) -> #c_map_pair{op=#c_literal{val=exact}}. +  is_valid_map_src(#c_literal{val = M}) when is_map(M) -> true; -is_valid_map_src(#c_map{})  -> true;  is_valid_map_src(#c_var{})  -> true;  is_valid_map_src(_)         -> false. -map_pair_list(Es, St) -> -    foldr(fun -	    ({map_field_assoc,L,K0,V0}, {Ces,Esp,St0}) -> -		{K,Ep0,St1} = safe(K0, St0), -		ok = ensure_valid_map_key(K), -		{V,Ep1,St2} = safe(V0, St1), -		A = lineno_anno(L, St2), -		Pair = #c_map_pair{op=#c_literal{val=assoc},anno=A,key=K,val=V}, -		{[Pair|Ces],Ep0 ++ Ep1 ++ Esp,St2}; -	    ({map_field_exact,L,K0,V0}, {Ces,Esp,St0}) -> -		{K,Ep0,St1} = safe(K0, St0), -		ok = ensure_valid_map_key(K), -		{V,Ep1,St2} = safe(V0, St1), -		A = lineno_anno(L, St2), -		Pair = #c_map_pair{op=#c_literal{val=exact},anno=A,key=K,val=V}, -		{[Pair|Ces],Ep0 ++ Ep1 ++ Esp,St2} -	end, {[],[],St}, Es). - -ensure_valid_map_key(#c_literal{}) -> ok; -ensure_valid_map_key(_) -> throw({bad_map,bad_map_key}). -  %% try_exception([ExcpClause], St) -> {[ExcpVar],Handler,St}.  try_exception(Ecs0, St0) ->      %% Note that Tag is not needed for rethrow - it is already in Info.      {Evs,St1} = new_vars(3, St0), % Tag, Value, Info -    {Ecs1,St2} = clauses(Ecs0, St1), +    {Ecs1,Ceps,St2} = clauses(Ecs0, St1),      [_,Value,Info] = Evs,      Ec = #iclause{anno=#a{anno=[compiler_generated]},  		  pats=[c_tuple(Evs)],guard=[#c_literal{val=true}], @@ -817,15 +856,15 @@ try_exception(Ecs0, St0) ->  				 name=#c_literal{val=raise},  				 args=[Info,Value]}]},      Hs = [#icase{anno=#a{},args=[c_tuple(Evs)],clauses=Ecs1,fc=Ec}], -    {Evs,Hs,St2}. +    {Evs,Ceps++Hs,St2}.  try_after(As, St0) ->      %% See above. -    {Evs,St1} = new_vars(3, St0),		% Tag, Value, Info +    {Evs,St1} = new_vars(3, St0),	 % Tag, Value, Info      [_,Value,Info] = Evs, -    B = As ++ [#iprimop{anno=#a{},       %Must have an #a{} -			 name=#c_literal{val=raise}, -			 args=[Info,Value]}], +    B = As ++ [#iprimop{anno=#a{},       % Must have an #a{} +			name=#c_literal{val=raise}, +			args=[Info,Value]}],      Ec = #iclause{anno=#a{anno=[compiler_generated]},  		  pats=[c_tuple(Evs)],guard=[#c_literal{val=true}],  		  body=B}, @@ -959,7 +998,7 @@ bitstr({bin_element,_,E0,Size0,[Type,{unit,Unit}|Flags]}, St0) ->  fun_tq({_,_,Name}=Id, Cs0, L, St0, NameInfo) ->      Arity = clause_arity(hd(Cs0)), -    {Cs1,St1} = clauses(Cs0, St0), +    {Cs1,Ceps,St1} = clauses(Cs0, St0),      {Args,St2} = new_vars(Arity, St1),      {Ps,St3} = new_vars(Arity, St2),		%Need new variables here      Anno = lineno_anno(L, St3), @@ -967,7 +1006,7 @@ fun_tq({_,_,Name}=Id, Cs0, L, St0, NameInfo) ->      Fun = #ifun{anno=#a{anno=Anno},  		id=[{id,Id}],				%We KNOW!  		vars=Args,clauses=Cs1,fc=Fc,name=NameInfo}, -    {Fun,[],St3}. +    {Fun,Ceps,St3}.  %% lc_tq(Line, Exp, [Qualifier], Mc, State) -> {LetRec,[PreExp],State}.  %%  This TQ from Simon PJ pp 127-138.   @@ -1181,7 +1220,7 @@ is_generator(_) -> false.  generator(Line, {generate,Lg,P0,E}, Gs, St0) ->      LA = lineno_anno(Line, St0),      GA = lineno_anno(Lg, St0), -    {Head,St1} = list_gen_pattern(P0, Line, St0), +    {Head,Ceps,St1} = list_gen_pattern(P0, Line, St0),      {[Tail,Skip],St2} = new_vars(2, St1),      {Cg,St3} = lc_guard_tests(Gs, St2),      {AccPat,SkipPat} = case Head of @@ -1202,24 +1241,25 @@ generator(Line, {generate,Lg,P0,E}, Gs, St0) ->                         end,      {Ce,Pre,St4} = safe(E, St3),      Gen = #igen{anno=#a{anno=GA},acc_pat=AccPat,acc_guard=Cg,skip_pat=SkipPat, -                tail=Tail,tail_pat=#c_literal{anno=LA,val=[]},arg={Pre,Ce}}, +                tail=Tail,tail_pat=#c_literal{anno=LA,val=[]},arg={Ceps++Pre,Ce}},      {Gen,St4};  generator(Line, {b_generate,Lg,P,E}, Gs, St0) ->      LA = lineno_anno(Line, St0),      GA = lineno_anno(Lg, St0), -    Cp = #c_binary{segments=Segs} = pattern(P, St0), +    {Cp = #c_binary{segments=Segs},[],St1} = pattern(P, St0), +          %% The function append_tail_segment/2 keeps variable patterns as-is, making      %% it possible to have the same skip clause removal as with list generators. -    {AccSegs,Tail,TailSeg,St1} = append_tail_segment(Segs, St0), +    {AccSegs,Tail,TailSeg,St2} = append_tail_segment(Segs, St1),      AccPat = Cp#c_binary{segments=AccSegs}, -    {Cg,St2} = lc_guard_tests(Gs, St1), -    {SkipSegs,St3} = emasculate_segments(AccSegs, St2), +    {Cg,St3} = lc_guard_tests(Gs, St2), +    {SkipSegs,St4} = emasculate_segments(AccSegs, St3),      SkipPat = Cp#c_binary{segments=SkipSegs}, -    {Ce,Pre,St4} = safe(E, St3), +    {Ce,Pre,St5} = safe(E, St4),      Gen = #igen{anno=#a{anno=GA},acc_pat=AccPat,acc_guard=Cg,skip_pat=SkipPat,                  tail=Tail,tail_pat=#c_binary{anno=LA,segments=[TailSeg]},                  arg={Pre,Ce}}, -    {Gen,St4}. +    {Gen,St5}.  append_tail_segment(Segs, St0) ->      {Var,St} = new_var(St0), @@ -1248,9 +1288,9 @@ lc_guard_tests(Gs0, St0) ->  list_gen_pattern(P0, Line, St) ->      try -	{pattern(P0, St),St} +	pattern(P0,St)      catch  -	nomatch -> {nomatch,add_warning(Line, nomatch, St)} +	nomatch -> {nomatch,[],add_warning(Line, nomatch, St)}      end.  %%% @@ -1473,6 +1513,18 @@ force_novars(#ibinary{}=Bin, St) -> {Bin,[],St};  force_novars(Ce, St) ->      force_safe(Ce, St). + +%% safe_pattern_expr(Expr, State) -> {Cexpr,[PreExpr],State}. +%%   only literals and variables are safe expressions in patterns +safe_pattern_expr(E,St0) -> +    case safe(E,St0) of +	{#c_var{},_,_}=Safe -> Safe; +	{#c_literal{},_,_}=Safe -> Safe; +	{Ce,Eps,St1} -> +	    {V,St2} = new_var(St1), +	    {V,Eps++[#iset{var=V,arg=Ce}],St2} +    end. +  %% safe(Expr, State) -> {Safe,[PreExpr],State}.  %%  Generate an internal safe expression.  These are simples without  %%  binaries which can fail.  At this level we do not need to do a @@ -1547,84 +1599,109 @@ fold_match({match,L,P0,E0}, P) ->      {{match,L,P0,P1},E1};  fold_match(E, P) -> {P,E}. -%% pattern(Pattern, State) -> CorePat. +%% pattern(Pattern, State) -> {CorePat,[PreExp],State}.  %% Transform a pattern by removing line numbers.  We also normalise  %% aliases in patterns to standard form, {alias,Pat,[Var]}. - -pattern({var,L,V}, St) -> #c_var{anno=lineno_anno(L, St),name=V}; -pattern({char,L,C}, St) -> #c_literal{anno=lineno_anno(L, St),val=C}; -pattern({integer,L,I}, St) -> #c_literal{anno=lineno_anno(L, St),val=I}; -pattern({float,L,F}, St) -> #c_literal{anno=lineno_anno(L, St),val=F}; -pattern({atom,L,A}, St) -> #c_literal{anno=lineno_anno(L, St),val=A}; -pattern({string,L,S}, St) -> #c_literal{anno=lineno_anno(L, St),val=S}; -pattern({nil,L}, St) -> #c_literal{anno=lineno_anno(L, St),val=[]}; +%% +%% In patterns we may have expressions +%% 1) Binaries -> #c_bitstr{size=Expr} +%% 2) Maps -> #c_map_pair{key=Expr} +%% +%% Both of these may generate pre-expressions since only bound variables +%% or literals are allowed for these in core patterns. +%% +%% Therefor, we need to drag both the state and the collection of pre-expression +%% around in the whole pattern transformation tree. + +pattern({var,L,V}, St) -> {#c_var{anno=lineno_anno(L, St),name=V},[],St}; +pattern({char,L,C}, St) -> {#c_literal{anno=lineno_anno(L, St),val=C},[],St}; +pattern({integer,L,I}, St) -> {#c_literal{anno=lineno_anno(L, St),val=I},[],St}; +pattern({float,L,F}, St) -> {#c_literal{anno=lineno_anno(L, St),val=F},[],St}; +pattern({atom,L,A}, St) -> {#c_literal{anno=lineno_anno(L, St),val=A},[],St}; +pattern({string,L,S}, St) -> {#c_literal{anno=lineno_anno(L, St),val=S},[],St}; +pattern({nil,L}, St) -> {#c_literal{anno=lineno_anno(L, St),val=[]},[],St};  pattern({cons,L,H,T}, St) -> -    annotate_cons(lineno_anno(L, St), pattern(H, St), pattern(T, St), St); +    {Ph,Eps1,St1} = pattern(H, St), +    {Pt,Eps2,St2} = pattern(T, St1), +    {annotate_cons(lineno_anno(L, St), Ph, Pt, St2),Eps1++Eps2,St2};  pattern({tuple,L,Ps}, St) -> -    annotate_tuple(record_anno(L, St), pattern_list(Ps, St), St); -pattern({map,L,Ps}, St) -> -    #c_map{anno=lineno_anno(L, St), es=pattern_map_pairs(Ps, St)}; +    {Ps1,Eps,St1} = pattern_list(Ps,St), +    {annotate_tuple(record_anno(L, St), Ps1, St),Eps,St1}; +pattern({map,L,Pairs}, St0) -> +    {Ps,Eps,St1} = pattern_map_pairs(Pairs, St0), +    {#c_map{anno=lineno_anno(L, St1), es=Ps},Eps,St1};  pattern({bin,L,Ps}, St) ->      %% We don't create a #ibinary record here, since there is      %% no need to hold any used/new annotations in a pattern. -    #c_binary{anno=lineno_anno(L, St),segments=pat_bin(Ps, St)}; +    {#c_binary{anno=lineno_anno(L, St),segments=pat_bin(Ps, St)},[],St};  pattern({match,_,P1,P2}, St) -> -    pat_alias(pattern(P1, St), pattern(P2, St)). +    {Cp1,Eps1,St1} = pattern(P1,St), +    {Cp2,Eps2,St2} = pattern(P2,St1), +    {pat_alias(Cp1,Cp2),Eps1++Eps2,St2}.  %% pattern_map_pairs([MapFieldExact],State) -> [#c_map_pairs{}]  pattern_map_pairs(Ps, St) -> -    %% check literal key uniqueness (dict is needed) -    %% pattern all pairs -    {CMapPairs, Kdb} = lists:mapfoldl(fun -	    (P,Kdbi) -> -		#c_map_pair{key=Ck,val=Cv} = CMapPair = pattern_map_pair(P,St), -		K = core_lib:literal_value(Ck), -		case dict:find(K,Kdbi) of -		    {ok, Vs} -> -			{CMapPair, dict:store(K,[Cv|Vs],Kdbi)}; -		    _ -> -			{CMapPair, dict:store(K,[Cv],Kdbi)} +    %% check literal key uniqueness +    %%   - guaranteed via aliasing map pairs +    %% pattern all pairs in two steps +    %% 1) Construct Core Pattern +    %% 2) Alias Keys in Core Pattern +    {CMapPairs, {Eps,St1}} = lists:mapfoldl(fun +	    (P,{EpsM,Sti0}) -> +		{CMapPair,EpsP,Sti1} = pattern_map_pair(P,Sti0), +		{CMapPair, {EpsM++EpsP,Sti1}} +	end, {[],St}, Ps), +    {pat_alias_map_pairs(CMapPairs,[]),Eps,St1}. + +%% remove cluddering annotations +pattern_map_clean_key(#c_literal{val=V}) -> {literal,V}; +pattern_map_clean_key(#c_var{name=V}) -> {var,V}. + +pat_alias_map_pairs(Ps1,Ps2) -> +    Ps = Ps1 ++ Ps2, +    F = fun(#c_map_pair{key=Ck,val=Cv},Dbi) -> +		K = pattern_map_clean_key(Ck), +		case dict:find(K,Dbi) of +		    {ok,Cvs} -> dict:store(K,[Cv|Cvs],Dbi); +		    _        -> dict:store(K,[Cv],Dbi)  		end -	end, dict:new(), Ps), -    pattern_alias_map_pairs(CMapPairs,Kdb,dict:new(),St). - -pattern_alias_map_pairs([],_,_,_) -> []; -pattern_alias_map_pairs([#c_map_pair{key=Ck}=Pair|Pairs],Kdb,Kset,St) -> -    %% alias same keys if needed -    K = core_lib:literal_value(Ck), -    case dict:find(K,Kset) of -	{ok,processed} -> -	    pattern_alias_map_pairs(Pairs,Kdb,Kset,St); -	_ -> +	end, +    Kdb = lists:foldl(F,dict:new(),Ps), +    pat_alias_map_pairs(Ps,Kdb,sets:new()). + +pat_alias_map_pairs([],_,_) -> []; +pat_alias_map_pairs([#c_map_pair{key=Ck}=Pair|Pairs],Kdb,Set) -> +    K = pattern_map_clean_key(Ck), +    case sets:is_element(K,Set) of +	true -> +	    pat_alias_map_pairs(Pairs,Kdb,Set); +	false ->  	    Cvs = dict:fetch(K,Kdb), -	    Cv = pattern_alias_map_pair_patterns(Cvs), -	    Kset1 = dict:store(K, processed, Kset), -	    [Pair#c_map_pair{val=Cv}|pattern_alias_map_pairs(Pairs,Kdb,Kset1,St)] +	    Cv = pat_alias_map_pair_values(Cvs), +	    Set1 = sets:add_element(K,Set), +	    [Pair#c_map_pair{val=Cv}|pat_alias_map_pairs(Pairs,Kdb,Set1)]      end. -pattern_alias_map_pair_patterns([Cv]) -> Cv; -pattern_alias_map_pair_patterns([Cv1,Cv2|Cvs]) -> -    pattern_alias_map_pair_patterns([pat_alias(Cv1,Cv2)|Cvs]). - -pattern_map_pair({map_field_exact,L,K,V}, St) -> -    case expr(K,St) of -	{#c_literal{}=Key,_,_} -> -	    #c_map_pair{anno=lineno_anno(L, St), -			op=#c_literal{val=exact}, -			key=Key, -			val=pattern(V, St)}; -	_ -> -	    %% this will throw a cryptic error message -	    %% but it is better than nothing -	    throw(nomatch) -    end. +pat_alias_map_pair_values([Cv]) -> Cv; +pat_alias_map_pair_values([Cv1,Cv2|Cvs]) -> +    pat_alias_map_pair_values([pat_alias(Cv1,Cv2)|Cvs]). + +pattern_map_pair({map_field_exact,L,K,V}, St0) -> +    {Ck,EpsK,St1} = safe_pattern_expr(K,St0), +    {Cv,EpsV,St2} = pattern(V, St1), +    {#c_map_pair{anno=lineno_anno(L,St2), +		 op=#c_literal{val=exact}, +		 key=Ck, +		 val=Cv},EpsK++EpsV,St2}.  %% pat_bin([BinElement], State) -> [BinSeg].  pat_bin(Ps, St) -> [pat_segment(P, St) || P <- Ps]. -pat_segment({bin_element,_,Term,Size,[Type,{unit,Unit}|Flags]}, St) -> -    #c_bitstr{val=pattern(Term, St),size=pattern(Size, St), +pat_segment({bin_element,_,Val,Size,[Type,{unit,Unit}|Flags]}, St) -> +    {Pval,[],St1} = pattern(Val,St), +    {Psize,[],_St2} = pattern(Size,St1), +    #c_bitstr{val=Pval,size=Psize,  	      unit=#c_literal{val=Unit},  	      type=#c_literal{val=Type},  	      flags=#c_literal{val=Flags}}. @@ -1634,6 +1711,8 @@ pat_segment({bin_element,_,Term,Size,[Type,{unit,Unit}|Flags]}, St) ->  pat_alias(#c_var{name=V1}, P2) -> #c_alias{var=#c_var{name=V1},pat=P2};  pat_alias(P1, #c_var{name=V2}) -> #c_alias{var=#c_var{name=V2},pat=P1}; + +%% alias cons  pat_alias(#c_cons{}=Cons, #c_literal{anno=A,val=[H|T]}=S) ->      pat_alias(Cons, ann_c_cons_skel(A, #c_literal{anno=A,val=H},  				    S#c_literal{val=T})); @@ -1642,6 +1721,8 @@ pat_alias(#c_literal{anno=A,val=[H|T]}=S, #c_cons{}=Cons) ->  			      S#c_literal{val=T}), Cons);  pat_alias(#c_cons{anno=Anno,hd=H1,tl=T1}, #c_cons{hd=H2,tl=T2}) ->      ann_c_cons(Anno, pat_alias(H1, H2), pat_alias(T1, T2)); + +%% alias tuples  pat_alias(#c_tuple{anno=Anno,es=Es1}, #c_literal{val=T}) when is_tuple(T) ->      Es2 = [#c_literal{val=E} || E <- tuple_to_list(T)],      ann_c_tuple(Anno, pat_alias_list(Es1, Es2)); @@ -1650,6 +1731,12 @@ pat_alias(#c_literal{anno=Anno,val=T}, #c_tuple{es=Es2}) when is_tuple(T) ->      ann_c_tuple(Anno, pat_alias_list(Es1, Es2));  pat_alias(#c_tuple{anno=Anno,es=Es1}, #c_tuple{es=Es2}) ->      ann_c_tuple(Anno, pat_alias_list(Es1, Es2)); + +%% alias maps +%% There are no literals in maps patterns (patterns are always abstract) +pat_alias(#c_map{es=Es1}=M,#c_map{es=Es2}) -> +    M#c_map{es=pat_alias_map_pairs(Es1,Es2)}; +  pat_alias(#c_alias{var=V1,pat=P1},  	  #c_alias{var=V2,pat=P2}) ->      if V1 =:= V2 -> #c_alias{var=V1,pat=pat_alias(P1, P2)}; @@ -1672,9 +1759,15 @@ pat_alias_list([A1|A1s], [A2|A2s]) ->  pat_alias_list([], []) -> [];  pat_alias_list(_, _) -> throw(nomatch). -%% pattern_list([P], State) -> [P]. +%% pattern_list([P], State) -> {[P],Exprs,St} + +pattern_list([P0|Ps0], St0) -> +    {P1,Eps,St1} = pattern(P0, St0), +    {Ps1,Epsl,St2} = pattern_list(Ps0, St1), +    {[P1|Ps1], Eps ++ Epsl, St2}; +pattern_list([], St) -> +    {[],[],St}. -pattern_list(Ps, St) -> [pattern(P, St) || P <- Ps].  %% make_vars([Name]) -> [{Var,Name}]. @@ -1974,9 +2067,14 @@ upattern(#c_tuple{es=Es0}=Tuple, Ks, St0) ->  upattern(#c_map{es=Es0}=Map, Ks, St0) ->      {Es1,Esg,Esv,Eus,St1} = upattern_list(Es0, Ks, St0),      {Map#c_map{es=Es1},Esg,Esv,Eus,St1}; -upattern(#c_map_pair{op=#c_literal{val=exact},val=V0}=MapPair, Ks, St0) -> -    {V,Vg,Vv,Vu,St1} = upattern(V0, Ks, St0), -    {MapPair#c_map_pair{val=V},Vg,Vv,Vu,St1}; +upattern(#c_map_pair{op=#c_literal{val=exact},key=K0,val=V0}=Pair,Ks,St0) -> +    {V,Vg,Vn,Vu,St1} = upattern(V0, Ks, St0), +    % A variable key must be considered used here +    Ku = case K0 of +	#c_var{name=Name} -> [Name]; +	_ -> [] +    end, +    {Pair#c_map_pair{val=V},Vg,Vn,union(Ku,Vu),St1};  upattern(#c_binary{segments=Es0}=Bin, Ks, St0) ->      {Es1,Esg,Esv,Eus,St1} = upat_bin(Es0, Ks, St0),      {Bin#c_binary{segments=Es1},Esg,Esv,Eus,St1}; @@ -2347,8 +2445,6 @@ format_error(nomatch) ->      "pattern cannot possibly match";  format_error(bad_binary) ->      "binary construction will fail because of a type mismatch"; -format_error(bad_map_key) -> -    "map construction will fail because of none literal key (large binaries are not literals)";  format_error(bad_map) ->      "map construction will fail because of a type mismatch". diff --git a/lib/compiler/src/v3_kernel.erl b/lib/compiler/src/v3_kernel.erl index 40d2f72b4c..6504351c02 100644 --- a/lib/compiler/src/v3_kernel.erl +++ b/lib/compiler/src/v3_kernel.erl @@ -527,9 +527,9 @@ map_split_pairs(A, Var, Ces, Sub, St0) ->      Pairs0 = [{Op,K,V} || #c_map_pair{op=#c_literal{val=Op},key=K,val=V} <- Ces],      {Pairs,Esp,St1} = foldr(fun  	    ({Op,K0,V0}, {Ops,Espi,Sti0}) when Op =:= assoc; Op =:= exact -> -		{K,[],Sti1} = expr(K0, Sub, Sti0), -		{V,Ep,Sti2} = atomic(V0, Sub, Sti1), -		{[{Op,K,V}|Ops],Ep ++ Espi,Sti2} +		{K,Eps1,Sti1} = atomic(K0, Sub, Sti0), +		{V,Eps2,Sti2} = atomic(V0, Sub, Sti1), +		{[{Op,K,V}|Ops],Eps1 ++ Eps2 ++ Espi,Sti2}  	end, {[],[],St0}, Pairs0),      case map_group_pairs(Pairs) of @@ -577,11 +577,12 @@ map_key_is_used(K,Used) ->      dict:find(map_key_clean(K),Used).  %% Be explicit instead of using set_kanno(K,[]) -map_key_clean(#k_literal{val=V}) -> {k_literal,V}; -map_key_clean(#k_int{val=V})     -> {k_int,V}; -map_key_clean(#k_float{val=V})   -> {k_float,V}; -map_key_clean(#k_atom{val=V})    -> {k_atom,V}; -map_key_clean(#k_nil{})          -> k_nil. +map_key_clean(#k_var{name=V})    -> {var,V}; +map_key_clean(#k_literal{val=V}) -> {lit,V}; +map_key_clean(#k_int{val=V})     -> {lit,V}; +map_key_clean(#k_float{val=V})   -> {lit,V}; +map_key_clean(#k_atom{val=V})    -> {lit,V}; +map_key_clean(#k_nil{})          -> {lit,[]}.  %% call_type(Module, Function, Arity) -> call | bif | apply | error. @@ -757,23 +758,22 @@ flatten_alias(#c_alias{var=V,pat=P}) ->  flatten_alias(Pat) -> {[],Pat}.  pattern_map_pairs(Ces0, Isub, Osub0, St0) -> -    %% It is assumed that all core keys are literals -    %% It is later assumed that these keys are term sorted -    %% so we need to sort them here -    Ces1 = lists:sort(fun -	    (#c_map_pair{key=CkA},#c_map_pair{key=CkB}) -> -		A = core_lib:literal_value(CkA), -		B = core_lib:literal_value(CkB), -		erts_internal:cmp_term(A,B) < 0 -	end, Ces0),      %% pattern the pair keys and values as normal      {Kes,{Osub1,St1}} = lists:mapfoldl(fun  	    (#c_map_pair{anno=A,key=Ck,val=Cv},{Osubi0,Sti0}) -> -		{Kk,Osubi1,Sti1} = pattern(Ck, Isub, Osubi0, Sti0), -		{Kv,Osubi2,Sti2} = pattern(Cv, Isub, Osubi1, Sti1), +		{Kk,[],Sti1} = expr(Ck, Isub, Sti0), +		{Kv,Osubi2,Sti2} = pattern(Cv, Isub, Osubi0, Sti1),  		{#k_map_pair{anno=A,key=Kk,val=Kv},{Osubi2,Sti2}} -	end, {Osub0, St0}, Ces1), -    {Kes,Osub1,St1}. +	end, {Osub0, St0}, Ces0), +    %% It is later assumed that these keys are term sorted +    %% so we need to sort them here +    Kes1 = lists:sort(fun +	    (#k_map_pair{key=KkA},#k_map_pair{key=KkB}) -> +		A = map_key_clean(KkA), +		B = map_key_clean(KkB), +		erts_internal:cmp_term(A,B) < 0 +	end, Kes), +    {Kes1,Osub1,St1}.  pattern_bin(Es, Isub, Osub0, St0) ->      {Kbin,{_,Osub},St} = pattern_bin_1(Es, Isub, Osub0, St0), @@ -1550,13 +1550,11 @@ arg_val(Arg, C) ->  		    {set_kanno(S, []),U,T,Fs}  	    end;  	#k_map{op=exact,es=Es} -> -	    Keys = [begin -			#k_map_pair{key=#k_literal{val=Key}} = Pair, -			Key -		    end || Pair <- Es], -	    %% multiple keys may have the same name -	    %% do not use ordsets -	    lists:sort(fun(A,B) -> erts_internal:cmp_term(A,B) < 0 end, Keys) +	    lists:sort(fun(A,B) -> +			%% on the form K :: {'lit' | 'var', term()} +			%% lit < var as intended +			erts_internal:cmp_term(A,B) < 0 +		end, [map_key_clean(Key) || #k_map_pair{key=Key} <- Es])      end.  %% ubody_used_vars(Expr, State) -> [UsedVar] @@ -1943,6 +1941,7 @@ lit_list_vars(Ps) ->  %% pat_vars(Pattern) -> {[UsedVarName],[NewVarName]}.  %%  Return variables in a pattern.  All variables are new variables  %%  except those in the size field of binary segments. +%%  and map_pair keys  pat_vars(#k_var{name=N}) -> {[],[N]};  %%pat_vars(#k_char{}) -> {[],[]}; @@ -1967,8 +1966,10 @@ pat_vars(#k_tuple{es=Es}) ->      pat_list_vars(Es);  pat_vars(#k_map{es=Es}) ->      pat_list_vars(Es); -pat_vars(#k_map_pair{val=V}) -> -    pat_vars(V). +pat_vars(#k_map_pair{key=K,val=V}) -> +    {U1,New} = pat_vars(V), +    {[], U2} = pat_vars(K), +    {union(U1,U2),New}.  pat_list_vars(Ps) ->      foldl(fun (P, {Used0,New0}) -> diff --git a/lib/compiler/src/v3_kernel.hrl b/lib/compiler/src/v3_kernel.hrl index ab66445f73..b008285d9f 100644 --- a/lib/compiler/src/v3_kernel.hrl +++ b/lib/compiler/src/v3_kernel.hrl @@ -38,7 +38,7 @@  -record(k_nil, {anno=[]}).  -record(k_tuple, {anno=[],es}). --record(k_map, {anno=[],var,op,es}). +-record(k_map, {anno=[],var=#k_literal{val=#{}},op,es}).  -record(k_map_pair, {anno=[],key,val}).  -record(k_cons, {anno=[],hd,tl}).  -record(k_binary, {anno=[],segs}). diff --git a/lib/compiler/test/error_SUITE.erl b/lib/compiler/test/error_SUITE.erl index bd877bb528..0d23f12fb5 100644 --- a/lib/compiler/test/error_SUITE.erl +++ b/lib/compiler/test/error_SUITE.erl @@ -23,7 +23,7 @@  -export([all/0, suite/0,groups/0,init_per_suite/1, end_per_suite/1,   	 init_per_group/2,end_per_group/2,  	 head_mismatch_line/1,warnings_as_errors/1, bif_clashes/1, -	 transforms/1,forbidden_maps/1,bad_utf8/1]). +	 transforms/1,maps_warnings/1,bad_utf8/1]).  %% Used by transforms/1 test case.  -export([parse_transform/2]). @@ -37,7 +37,7 @@ all() ->  groups() ->       [{p,test_lib:parallel(),        [head_mismatch_line,warnings_as_errors,bif_clashes, -       transforms,forbidden_maps,bad_utf8]}]. +       transforms,maps_warnings,bad_utf8]}].  init_per_suite(Config) ->      Config. @@ -241,17 +241,30 @@ parse_transform(_, _) ->      error(too_bad). -forbidden_maps(Config) when is_list(Config) -> -    Ts1 = [{map_illegal_use_of_pattern, +maps_warnings(Config) when is_list(Config) -> +    Ts1 = [{map_ok_use_of_pattern,  	   <<" -              -export([t/0]). +              -export([t/1]). +              t(K) -> +                 #{K := 1 = V} = id(#{<<\"hi all\">> => 1}), +		 V. +              id(I) -> I. +             ">>, +	    [return], +	    []}, +	{map_illegal_use_of_pattern, +	   <<" +              -export([t/0,t/2]). +	      t(K,#{ K := V }) -> V.                t() ->                   V = 32,                   #{<<\"hi\",V,\"all\">> := 1} = id(#{<<\"hi all\">> => 1}).                id(I) -> I.               ">>,  	    [return], -	    {error,[{5,erl_lint,{illegal_map_key_variable,'V'}}], []}}], +	    {error,[{3,erl_lint,{unbound_var,'K'}}, +		    {6,erl_lint,illegal_map_key}],[]}} +    ],      [] = run2(Config, Ts1),      ok. diff --git a/lib/compiler/test/map_SUITE.erl b/lib/compiler/test/map_SUITE.erl index 403b7e8405..75efce9d7b 100644 --- a/lib/compiler/test/map_SUITE.erl +++ b/lib/compiler/test/map_SUITE.erl @@ -21,6 +21,7 @@      ]).  -export([ +	%% literals  	t_build_and_match_literals/1,  	t_update_literals/1,t_match_and_update_literals/1,  	t_update_map_expressions/1, @@ -32,6 +33,15 @@  	t_map_size/1,  	t_build_and_match_aliasing/1, +	%% variables +	t_build_and_match_variables/1, +	t_update_assoc_variables/1,t_update_exact_variables/1, +	t_nested_pattern_expressions/1, +	t_guard_update_variables/1, +	t_guard_sequence_variables/1, +	t_guard_sequence_mixed/1, +	t_frequency_table/1, +  	%% warnings  	t_warn_useless_build/1,  	t_warn_pair_key_overloaded/1, @@ -52,6 +62,7 @@  suite() -> [].  all() -> [ +	%% literals  	t_build_and_match_literals,  	t_update_literals, t_match_and_update_literals,  	t_update_map_expressions, @@ -62,6 +73,15 @@ all() -> [  	t_map_size,  	t_build_and_match_aliasing, +	%% variables +	t_build_and_match_variables, +	t_update_assoc_variables,t_update_exact_variables, +	t_nested_pattern_expressions, +	t_guard_update_variables, +	t_guard_sequence_variables, +	t_guard_sequence_mixed, +	t_frequency_table, +  	%% warnings  	t_warn_useless_build,  	t_warn_pair_key_overloaded, @@ -73,6 +93,7 @@ all() -> [  	t_build_and_match_nil,  	t_build_and_match_structure, +  	%% errors in 17.0-rc1  	t_update_values,          t_expand_map_update, @@ -119,6 +140,11 @@ t_build_and_match_literals(Config) when is_list(Config) ->      %% nil key      #{[]:=ok,1:=2} = id(#{[]=>ok,1=>2}), +    #{1:=2,[]:=ok,1:=2} = id(#{[]=>ok,1=>2}), + +    %% pseudo literals +    #{ -3 := yep } = id(#{ -3 => yep }), +    #{ <<0:358>> := "three" } = id(#{<<0:358>> =>"three"}),      %% error case      {'EXIT',{{badmatch,_},_}} = (catch (#{x:=3,x:=2} = id(#{x=>3}))), @@ -126,10 +152,10 @@ t_build_and_match_literals(Config) when is_list(Config) ->      {'EXIT',{{badmatch,_},_}} = (catch (#{x:=3} = id({a,b,c}))),      {'EXIT',{{badmatch,_},_}} = (catch (#{x:=3} = id(#{y=>3}))),      {'EXIT',{{badmatch,_},_}} = (catch (#{x:=3} = id(#{x=>"three"}))), -    {'EXIT',{badarg,_}} = (catch id(#{<<0:258>> =>"three"})),      {'EXIT',{{badmatch,_},_}} = (catch (#{#{"a"=>42} := 3}=id(#{#{"a"=>3}=>42}))),      ok. +  t_build_and_match_aliasing(Config) when is_list(Config) ->      M1 = id(#{a=>1,b=>2,c=>3,d=>4}),      #{c:=C1=_=_=C2} = M1, @@ -143,6 +169,19 @@ t_build_and_match_aliasing(Config) when is_list(Config) ->      M2 = id(#{"a"=>1,"b"=>2,"c"=>3,"d"=>4}),      #{"a":=A2,"a":=A2,"a":=A2,"b":=B2,"b":=B2,"b":=2} = M2,      #{"a":=_,"a":=_,"a":=_,"b":=_,"b":=_,"b":=2} = M2, + +    #{a:=A1,a:=A1,a:=A1,b:=B1,b:=B1} = #{a:=A1,a:=A1,a:=A1,b:=B1,b:=B1,b:=2} = M1, +    #{"a":=A3,"b":=B3} = #{"a":=A3,"a":=A3} = #{"b":=B3,"b":=2} = M2, + +    #{"a":=1,"b":=2,"c":=3,"d":=4} = #{"a":=A4,"b":=B4} = #{"a":=A4,"a":=A4} = #{"b":=B4,"d":=4} = M2, +    #{"a":=A5,"b":=B5} = #{"a":=A5,"a":=A5} = #{"b":=B5,"d":=4} = #{"a":=1,"b":=2,"c":=3,"d":=4} = M2, +    #{"a":=_,"b":=_} = #{"a":=_,"a":=_} = #{"b":=_,"d":=4} = #{"a":=1,"b":=2,"c":=3,"d":=4} = M2, + +    M3 = id(#{<<12:300>>=>1,<<13:300>>=>2}), +    #{<<12:300>> := V1, <<13:300>> := V2} = #{<<13:300>> := V2, <<12:300>> := V1} = M3, +    #{<<12:300>> := 1, <<13:300>> := 2} = #{<<13:300>> := _, <<12:300>> := _} = M3, +    #{<<13:300>> := _, <<12:300>> := _} = #{<<12:300>> := 1, <<13:300>> := 2} = M3, +      ok.  t_map_size(Config) when is_list(Config) -> @@ -241,11 +280,14 @@ t_update_assoc(Config) when is_list(Config) ->      #{1:=a,2:=b,3.0:=new,4:=d,5:=e} = M2,      M2 = M0#{3.0:=wrong,3.0=>new}, +    % Can't handle directly yet +    Bin = <<0:257>>, +    #{ Bin := val } = id(M0#{<<0:257>> => val}), %% binary limitation +      %% Errors cases.      BadMap = id(badmap),      {'EXIT',{badarg,_}} = (catch BadMap#{nonexisting=>val}),      {'EXIT',{badarg,_}} = (catch <<>>#{nonexisting=>val}), -    {'EXIT',{badarg,_}} = (catch M0#{<<0:257>> => val}), %% limitation      ok.  t_update_exact(Config) when is_list(Config) -> @@ -281,8 +323,10 @@ t_update_values(Config) when is_list(Config) ->      V0 = id(1337),      M0 = #{ a => 1, val => V0},      V1 = get_val(M0), -    M1 = M0#{ val := [V0,V1], "wazzup" => 42 }, +    M1 = id(M0#{ val := [V0,V1], "wazzup" => 42 }),      [1337, {some_val, 1337}] = get_val(M1), +    M2 = id(M1#{ <<42:333>> => 1337 }), +    {bin_key,1337} = get_val(M2),      N = 110,      List = [{[I,1,2,3,I],{1,2,3,"wat",I}}|| I <- lists:seq(1,N)], @@ -308,6 +352,7 @@ t_export(Config) when is_list(Config) ->  check_val(#{val1:=V1, val2:=V2},V1,V2) -> ok. +get_val(#{ <<42:333>> := V }) -> {bin_key, V};  get_val(#{ "wazzup" := _, val := V}) -> V;  get_val(#{ val := V }) -> {some_val, V}. @@ -437,7 +482,10 @@ guard_receive_loop() ->  t_list_comprehension(Config) when is_list(Config) -> -    [#{k:=1},#{k:=2},#{k:=3}] = [#{k=>I} || I <- [1,2,3]], +    [#{k:=1},#{k:=2},#{k:=3}] = id([#{k=>I} || I <- [1,2,3]]), +    Ls = id([#{<<2:301>> => I, "wat" => I + 1} || I <- [1,2,3]]), +    [#{<<2:301>>:=1,"wat":=2},#{<<2:301>>:=2,"wat":=3},#{<<2:301>>:=3,"wat":=4}] = Ls, +    [{1,2},{2,3},{3,4}] = id([{I2,I1} || #{"wat" := I1, <<2:301>> := I2} <- Ls]),      ok.  t_guard_fun(Config) when is_list(Config) -> @@ -601,5 +649,325 @@ t_build_and_match_structure(Config) when is_list(Config) ->      end,      ok. +%% simple build and match variables +t_build_and_match_variables(Config) when is_list(Config) -> +    K0 = id(#{}), +    K1 = id(1),       V1 = id(a), +    K2 = id(2),       V2 = id(b), +    K3 = id(3),       V3 = id("c"), +    K4 = id("4"),     V4 = id("d"), +    K5 = id(<<"5">>), V5 = id(<<"e">>), +    K6 = id({"6",7}), V6 = id("f"), +    K7 = id(#{ "a" => 3 }), +    #{K1:=V1} = id(#{K1=>V1}), +    #{K1:=V1,K2:=V2} = id(#{K1=>V1,K2=>V2}), +    #{K1:=V1,K2:=V2,K3:=V3} = id(#{K1=>V1,K2=>V2,K3=>V3}), +    #{K1:=V1,K2:=V2,K3:=V3,K4:=V4} = id(#{K1=>V1,K2=>V2,K3=>V3,K4=>V4}), +    #{K1:=V1,K2:=V2,K3:=V3,K4:=V4,K5:=V5} = id(#{K1=>V1,K2=>V2,K3=>V3,K4=>V4,K5=>V5}), +    #{K1:=V1,K2:=V2,K3:=V3,K4:=V4,K5:=V5,K6:=V6} = id(#{K1=>V1,K2=>V2,K3=>V3,K4=>V4,K5=>V5,K6=>V6}), + +    #{K5:=X,K5:=X=3,K4:=4} = id(#{K5=>3,K4=>4}), +    #{K5:=X,<<"5">>:=X=3,K4:=4} = id(#{K5=>3,K4=>4}), +    #{K5:=X,<<"5">>:=X=3,K4:=4} = id(#{<<"5">>=>3,K4=>4}), + +    #{ K4:=#{ K3:=#{K1:=V1, K2:=V2}}, K5:=V5} = +	id(#{ K5=>V5, K4=>#{ K3=>#{K2 => V2, K1 => V1}}}), +    #{ K4 := #{ K5 := Res }, K6 := Res} = id(#{K4=>#{K5 => 99}, K6 => 99}), + +    %% has keys +    #{a :=_,b :=_,K1:=_,K2:=_,K3:=V3,K4:=ResKey,K4:=ResKey,"4":=ResKey,"4":="ok"} = +	id(#{ a=>1, b=>1, K1=>V1, K2=>V2, K3=>V3, K4=>"nope", "4"=>"ok" }), + +    %% function +    ok = match_function_map_neg_keys(#{ -1 => a, -2 => b, -3 => c }), + +    %% map key +    #{ K0 := 42 } = id(#{ K0 => 42 }), +    #{ K7 := 42 } = id(#{ K7 => 42 }), + +    %% nil key +    KNIL = id([]), +    #{KNIL:=ok,1:=2} = id(#{KNIL=>ok,1=>2}), + +    Bin = <<0:258>>, +    #{ Bin := "three" } = id(#{<<0:258>> =>"three"}), + +    %% error case +    {'EXIT',{{badmatch,_},_}} = (catch (#{K5:=3,x:=2} = id(#{K5=>3}))), +    {'EXIT',{{badmatch,_},_}} = (catch (#{K5:=2} = id(#{K5=>3}))), +    {'EXIT',{{badmatch,_},_}} = (catch (#{K5:=3} = id({a,b,c}))), +    {'EXIT',{{badmatch,_},_}} = (catch (#{K5:=3} = id(#{K6=>3}))), +    {'EXIT',{{badmatch,_},_}} = (catch (#{K5:=3} = id(K7))), +    {'EXIT',{{badmatch,_},_}} = (catch (#{K7:=3} = id(#{K7=>42}))), +    ok. + + +match_function_map_neg_keys(#{ -1 := a, -2 := b, -3 := c }) -> ok. + +t_update_assoc_variables(Config) when is_list(Config) -> +    K1 = id(1), +    K2 = id(2), +    K3 = id(3.0), +    K4 = id(4), +    K5 = id(5), +    K6 = id(2.0), + +    M0 = #{K1=>a,K2=>b,K3=>c,K4=>d,K5=>e}, + +    M1 = M0#{K1=>42,K2=>100,K4=>[a,b,c]}, +    #{1:=42,2:=100,3.0:=c,4:=[a,b,c],5:=e} = M1, +    #{1:=42,2:=b,4:=d,5:=e,2.0:=100,K3:=c,4.0:=[a,b,c]} = M0#{1.0=>float,1:=42,2.0=>wrong,K6=>100,4.0=>[a,b,c]}, + +    M2 = M0#{K3=>new}, +    #{1:=a,2:=b,K3:=new,4:=d,5:=e} = M2, +    M2 = M0#{3.0:=wrong,K3=>new}, + +    #{ <<0:258>> := val } = id(M0#{<<0:258>> => val}), %% binary limitation + +    %% Errors cases. +    BadMap = id(badmap), +    {'EXIT',{badarg,_}} = (catch BadMap#{nonexisting=>val}), +    {'EXIT',{badarg,_}} = (catch <<>>#{nonexisting=>val}), +    ok. + +t_update_exact_variables(Config) when is_list(Config) -> +    K1 = id(1), +    K2 = id(2), +    K3 = id(3.0), +    K4 = id(4), + +    M0 = id(#{1=>a,2=>b,3.0=>c,4=>d,5=>e}), + +    M1 = M0#{K1:=42,K2:=100,K4:=[a,b,c]}, +    #{1:=42,2:=100,3.0:=c,K4:=[a,b,c],5:=e} = M1, +    M1 = M0#{K1:=wrong,1:=also_wrong,K1=>42,2=>wrong,K2:=100,4:=[a,b,c]}, + +    M2 = M0#{K3:=new}, +    #{1:=a,K2:=b,3.0:=new,K4:=d,5:=e} = M2, +    M2 = M0#{3.0=>wrong,K3:=new}, +    true = M2 =/= M0#{3=>right,3.0:=new}, +    #{ 3 := right, 3.0 := new } = M0#{3=>right,K3:=new}, + +    M3 = id(#{ 1 => val}), +    #{1 := update2,1.0 := new_val4} = M3#{ +	1.0 => new_val1, K1 := update, K1=> update3, +	K1 := update2, 1.0 := new_val2, 1.0 => new_val3, +	1.0 => new_val4 }, + +    #{ "wat" := 3, 2 := a } = id(#{ "wat" => 1, K2 => 2 }#{ K2 := a, "wat" := 3 }), + +    %% Errors cases. +    {'EXIT',{badarg,_}} = (catch ((id(nil))#{ a := b })), +    {'EXIT',{badarg,_}} = (catch M0#{nonexisting:=val}), +    {'EXIT',{badarg,_}} = (catch M0#{1.0:=v,1.0=>v2}), +    {'EXIT',{badarg,_}} = (catch M0#{42.0:=v,42:=v2}), +    {'EXIT',{badarg,_}} = (catch M0#{42=>v1,42.0:=v2,42:=v3}), +    {'EXIT',{badarg,_}} = (catch <<>>#{nonexisting:=val}), +    {'EXIT',{badarg,_}} = (catch M0#{<<0:257>> := val}), %% limitation +    ok. + +t_nested_pattern_expressions(Config) when is_list(Config) -> +    K1 = id("hello"), +    %K2 = id({ok}), +    [_,_,#{ <<"hi">> := wat, K1 := 42 }|_] = id([k,k,#{<<"hi">> => wat, K1 => 42}]), +    [_,_,#{ -1 := wat, K1 := 42 }|_] = id([k,k,#{-1 => wat, K1 => 42}]), +    [_,_,{#{ -1 := #{ {-3,<<0:300>>} := V1 }, K1 := 42 },3}|_] = id([k,k,{#{-1 => #{{-3,<<0:300>>}=>"hi"}, K1 => 42},3}]), +    "hi" = V1, +    %[k,#{ {-1,K1,[]} := {wat,K1}, K2 := 42 }|_] = id([k,#{{-1,K1,[]} => {wat,K1}, K2 => 42}]), +    %[k,#{ [-1,K2,[]] := {wat,K1}, K1 := 42 }|_] = id([k,#{[-1,K2,[]] => {wat,K1}, K1 => 42}]), + +    M0 = id(#{ <<33:333>> => 1, <<332:333>> => ok, a => ok, wat => yep, watzor => ok }), +    F0 = map_nested_pattern_funs(M0), +    F1 = F0(wat), +    F2 = F1(watzor), +    {yep,ok} = F2(M0), +    ok. + +map_nested_pattern_funs(M) -> +    K0 = id(a), +    fun(K1) -> +	    case M of +		#{ K0 := ok, K1 := yep, <<33:333>> := 1 } -> +		    fun(K2) -> +			    case M of +				#{ K2 := ok, K1 := yep, <<33:333>> := 1 } -> +				    fun +					(#{ <<332:333>> := ok, K1 := V1, K2 := V2 }) -> +					    {V1,V2} +				    end +			    end +		    end +	    end +    end. + +t_guard_update_variables(Config) when is_list(Config) -> +    error  = map_guard_update_variables(n,#{},#{}), +    first  = map_guard_update_variables(x,#{}, #{x=>first}), +    second = map_guard_update_variables(x,#{y=>old}, #{x=>second,y=>old}), +    third  = map_guard_update_variables(x,#{x=>old,y=>old}, #{x=>third,y=>old}), +    fourth = map_guard_update_variables(x,#{x=>old,y=>old}, #{x=>4,y=>new}), +    ok. + +map_guard_update_variables(K,M1,M2) when M1#{K=>first}    =:= M2 -> first; +map_guard_update_variables(K,M1,M2) when M1#{K=>second}   =:= M2 -> second; +map_guard_update_variables(K,M1,M2) when M1#{K:=third}    =:= M2 -> third; +map_guard_update_variables(K,M1,M2) when M1#{K:=4,y=>new} =:= M2 -> fourth; +map_guard_update_variables(_,_,_) -> error. + +t_guard_sequence_variables(Config) when is_list(Config) -> +    {1,"a"} = map_guard_sequence_var_1(a,#{seq=>1,a=>id("a"),b=>no}), +    {2,"b"} = map_guard_sequence_var_1(b,#{seq=>2,b=>id("b"),a=>no}), +    {3,"c"} = map_guard_sequence_var_1(a,#{seq=>3,a=>id("c"),b=>no}), +    {4,"d"} = map_guard_sequence_var_1(b,#{seq=>4,b=>id("d"),a=>no}), +    {4,4}   = map_guard_sequence_var_1(seq,#{seq=>4}), +    {4,4,y} = map_guard_sequence_var_1(seq,#{seq=>4,b=>id("d"),a=>y}), +    {5,"d"} = map_guard_sequence_var_1(b,#{seq=>5,b=>id("d"),a=>y}), + +    %% error case +    {'EXIT',{{case_clause,_},_}} = (catch map_guard_sequence_var_1("a",#{seq=>4,val=>id("e")})), +    ok. + + +map_guard_sequence_var_1(K,M) -> +    case M of +	#{seq:=1=Seq, K:=Val} -> {Seq,Val}; +	#{seq:=2=Seq, K:=Val} -> {Seq,Val}; +	#{seq:=3=Seq, K:=Val} -> {Seq,Val}; +	#{K:=4=Seq,   K:=Val1,a:=Val2} -> {Seq,Val1,Val2}; +	#{seq:=4=Seq, K:=Val} -> {Seq,Val}; +	#{K:=4=Seq,   K:=Val} -> {Seq,Val}; +	#{seq:=5=Seq, K:=Val} -> {Seq,Val} +    end. + + +t_guard_sequence_mixed(Config) when is_list(Config) -> +    M0 = id(#{ a=>1, b=>1, c=>1, d=>1, e=>1, f=>1, g=>1, h=>1 }), +    M1 = id(M0#{ d := 3 }), +    1 = map_guard_sequence_mixed(a,d,M1), +    M2 = id(M1#{ b := 2, d := 4, h := 2 }), +    2 = map_guard_sequence_mixed(a,d,M2), +    M3 = id(M2#{ b := 3, e := 5, g := 3 }), +    3 = map_guard_sequence_mixed(a,e,M3), +    M4 = id(M3#{ c := 4, e := 6, h := 1 }), +    4 = map_guard_sequence_mixed(a,e,M4), +    M5 = id(M4#{ c := 5, f := 7, g := 2 }), +    5 = map_guard_sequence_mixed(a,f,M5), +    M6 = id(M5#{ c := 6, f := 8, h := 3 }), +    6 = map_guard_sequence_mixed(a,f,M6), + +    %% error case +    {'EXIT',{{case_clause,_},_}} = (catch map_guard_sequence_mixed(a,b,M0)), +    ok. + +map_guard_sequence_mixed(K1,K2,M) -> +    case M of +	#{ K1 := 1, b := 1, K2 := 3, g := 1} -> 1; +	#{ K1 := 1, b := 2, K2 := 4, h := 2} -> 2; +	#{ K1 := 1, b := 3, K2 := 5, g := 3} -> 3; +	#{ K1 := 1, c := 4, K2 := 6, h := 1} -> 4; +	#{ K1 := 1, c := 5, K2 := 7, g := 2} -> 5; +	#{ K1 := 1, c := 6, K2 := 8, h := 3} -> 6 +    end. + + + +t_frequency_table(Config) when is_list(Config) -> +    random:seed({13,1337,54}),  % pseudo random +    N = 100000, +    Ts = rand_terms(N), +    #{ n:=N, tf := Tf } = frequency_table(Ts,#{ n=>0, tf => #{}}), +    ok = check_frequency(Ts,Tf), +    ok. + + +frequency_table([T|Ts], M) -> +    case M of +	#{ n := N, tf := #{ T := C } = F } -> +	    frequency_table(Ts,M#{ n := N + 1, tf := F#{ T := C + 1 }}); +	#{ n := N, tf := F } -> +	    frequency_table(Ts,M#{ n := N + 1, tf := F#{ T => 1 }}) +    end; +frequency_table([], M) -> M. + + +check_frequency(Ts,Tf) -> +    check_frequency(Ts,Tf,dict:new()). + +check_frequency([T|Ts],Tf,D) -> +    case dict:find(T,D) of +	error  -> check_frequency(Ts,Tf,dict:store(T,1,D)); +	{ok,C} -> check_frequency(Ts,Tf,dict:store(T,C+1,D)) +    end; +check_frequency([],Tf,D) -> +    validate_frequency(dict:to_list(D),Tf). + +validate_frequency([{T,C}|Fs],Tf) -> +    case Tf of +	#{ T := C } -> validate_frequency(Fs,Tf); +	_ -> error +    end; +validate_frequency([], _) -> ok. + + +%% aux + +rand_terms(0) -> []; +rand_terms(N) -> [rand_term()|rand_terms(N-1)]. + +rand_term() -> +    case random:uniform(6) of +	1 -> rand_binary(); +	2 -> rand_number(); +	3 -> rand_atom(); +	4 -> rand_tuple(); +	5 -> rand_list(); +	6 -> rand_map() +    end. + +rand_binary() -> +    case random:uniform(3) of +	1 -> <<>>; +	2 -> <<"hi">>; +	3 -> <<"message text larger than 64 bytes. yep, message text larger than 64 bytes.">> +    end. + +rand_number() -> +    case random:uniform(3) of +	1 -> random:uniform(5); +	2 -> float(random:uniform(5)); +	3 -> 1 bsl (63 + random:uniform(3)) +    end. + +rand_atom() -> +    case random:uniform(3) of +	1 -> hi; +	2 -> some_atom; +	3 -> some_other_atom +    end. + + +rand_tuple() -> +    case random:uniform(3) of +	1 -> {ok, rand_term()}; % careful +	2 -> {1, 2, 3}; +	3 -> {<<"yep">>, 1337} +    end. + +rand_list() -> +    case random:uniform(3) of +	1 -> "hi"; +	2 -> [1,rand_term()]; % careful +	3 -> [improper|list] +    end. + +rand_map() -> +    case random:uniform(3) of +	1 -> #{ hi => 3 }; +	2 -> #{ wat => rand_term(), other => 3 };  % careful +	3 -> #{ hi => 42, other => 42, yet_anoter => 1337 } +    end. + + +  %% Use this function to avoid compile-time evaluation of an expression.  id(I) -> I. diff --git a/lib/compiler/test/warnings_SUITE.erl b/lib/compiler/test/warnings_SUITE.erl index 0637041873..be0348a92d 100644 --- a/lib/compiler/test/warnings_SUITE.erl +++ b/lib/compiler/test/warnings_SUITE.erl @@ -601,7 +601,7 @@ maps(Config) when is_list(Config) ->             ">>,             [],  	   {warnings,[{3,v3_core,bad_map}]}}, -	   {bad_map_literal_key, +	   {ok_map_literal_key,             <<"               t() ->  		 V = id(1), @@ -614,7 +614,7 @@ maps(Config) when is_list(Config) ->  	     id(I) -> I.             ">>,             [], -	   {warnings,[{6,v3_core,nomatch}]}}], +	   []}],      run(Config, Ts),      ok. diff --git a/lib/debugger/test/map_SUITE.erl b/lib/debugger/test/map_SUITE.erl index 0076193725..e525484a8e 100644 --- a/lib/debugger/test/map_SUITE.erl +++ b/lib/debugger/test/map_SUITE.erl @@ -32,9 +32,30 @@  	t_guard_receive/1, t_guard_fun/1,  	t_list_comprehension/1,  	t_map_sort_literals/1, -	%t_size/1,  	t_map_size/1, - +	t_build_and_match_aliasing/1, + +	%% variables +	t_build_and_match_variables/1, +	t_update_assoc_variables/1,t_update_exact_variables/1, +	t_nested_pattern_expressions/1, +	t_guard_update_variables/1, +	t_guard_sequence_variables/1, +	t_guard_sequence_mixed/1, +	t_frequency_table/1, + +	%% not covered in 17.0-rc1 +	t_build_and_match_over_alloc/1, +	t_build_and_match_empty_val/1, +	t_build_and_match_val/1, +	t_build_and_match_nil/1, +	t_build_and_match_structure/1, + +	%% errors in 17.0-rc1 +	t_update_values/1, +        t_expand_map_update/1, +        t_export/1, +   	%% Specific Map BIFs  	t_bif_map_get/1,  	t_bif_map_find/1, @@ -61,8 +82,7 @@  	%% misc  	t_pdict/1, -	t_ets/1, -	t_dets/1 +	t_ets/1      ]).  -include_lib("stdlib/include/ms_transform.hrl"). @@ -77,7 +97,30 @@ all() -> [  	t_guard_bifs, t_guard_sequence, t_guard_update,  	t_guard_receive,t_guard_fun, t_list_comprehension,  	t_map_sort_literals, - +	t_build_and_match_aliasing, + +	%% variables +	t_build_and_match_variables, +	t_update_assoc_variables,t_update_exact_variables, +	t_nested_pattern_expressions, +	t_guard_update_variables, +	t_guard_sequence_variables, +	t_guard_sequence_mixed, +	t_frequency_table, + +	%% not covered in 17.0-rc1 +	t_build_and_match_over_alloc, +	t_build_and_match_empty_val, +	t_build_and_match_val, +	t_build_and_match_nil, +	t_build_and_match_structure, + + +	%% errors in 17.0-rc1 +	t_update_values, +        t_expand_map_update, +        t_export, +      	%% Specific Map BIFs  	t_bif_map_get,t_bif_map_find,t_bif_map_is_key,  	t_bif_map_keys, t_bif_map_merge, t_bif_map_new, @@ -94,7 +137,6 @@ all() -> [  	t_maps_fold, t_maps_map,  	t_maps_size, t_maps_without, -          %% Other functions  	t_pdict,  	t_ets @@ -147,18 +189,6 @@ t_build_and_match_literals(Config) when is_list(Config) ->      {'EXIT',{{badmatch,_},_}} = (catch (#{x:=3} = id(#{x=>"three"}))),      ok. - -%% Tests size(Map). -%% not implemented, perhaps it shouldn't be either - -%t_size(Config) when is_list(Config) -> -%    0 = size(#{}), -%    1 = size(#{a=>1}), -%    1 = size(#{a=>#{a=>1}}), -%    2 = size(#{a=>1, b=>2}), -%    3 = size(#{a=>1, b=>2, b=>"3"}), -%    ok. -  t_map_size(Config) when is_list(Config) ->      0 = map_size(id(#{})),      1 = map_size(id(#{a=>1})), @@ -268,6 +298,44 @@ t_update_exact(Config) when is_list(Config) ->      ok. +t_update_values(Config) when is_list(Config) -> +    V0 = id(1337), +    M0 = #{ a => 1, val => V0}, +    V1 = get_val(M0), +    M1 = id(M0#{ val := [V0,V1], "wazzup" => 42 }), +    [1337, {some_val, 1337}] = get_val(M1), +    M2 = id(M1#{ <<42:333>> => 1337 }), +    {bin_key,1337} = get_val(M2), + +    N = 110, +    List = [{[I,1,2,3,I],{1,2,3,"wat",I}}|| I <- lists:seq(1,N)], + +    {_,_,#{val2 := {1,2,3,"wat",N}, val1 := [N,1,2,3,N]}} = lists:foldl(fun +	    ({V2,V3},{Old2,Old3,Mi}) -> +		ok = check_val(Mi,Old2,Old3), +		#{ val1 := Old2, val2 := Old3 } = Mi, +		{V2,V3, Mi#{ val1 := id(V2), val2 := V1, val2 => id(V3)}} +	end, {none, none, #{val1=>none,val2=>none}},List), +    ok. + +t_expand_map_update(Config) when is_list(Config) -> +    M = #{<<"hello">> => <<"world">>}#{<<"hello">> := <<"les gens">>}, +    #{<<"hello">> := <<"les gens">>} = M, +    ok. + +t_export(Config) when is_list(Config) -> +    Raclette = id(#{}), +    case brie of brie -> Fromage = Raclette end, +    Raclette = Fromage#{}, +    ok. + +check_val(#{val1:=V1, val2:=V2},V1,V2) -> ok. + +get_val(#{ <<42:333>> := V }) -> {bin_key, V}; +get_val(#{ "wazzup" := _, val := V}) -> V; +get_val(#{ val := V }) -> {some_val, V}. + +  t_guard_bifs(Config) when is_list(Config) ->      true   = map_guard_head(#{a=>1}),      false  = map_guard_head([]), @@ -988,16 +1056,385 @@ t_ets(_Config) ->      ets:delete(Tid),      ok. -t_dets(_Config) -> +t_build_and_match_aliasing(Config) when is_list(Config) -> +    M1 = id(#{a=>1,b=>2,c=>3,d=>4}), +    #{c:=C1=_=_=C2} = M1, +    true = C1 =:= C2, +    #{a:=A,a:=A,a:=A,b:=B,b:=B} = M1, +    #{a:=A,a:=A,a:=A,b:=B,b:=B,b:=2} = M1, +    #{a:=A=1,a:=A,a:=A,b:=B=2,b:=B,b:=2} = M1, +    #{c:=C1, c:=_, c:=3, c:=_, c:=C2} = M1, +    #{c:=C=_=3=_=C} = M1, + +    M2 = id(#{"a"=>1,"b"=>2,"c"=>3,"d"=>4}), +    #{"a":=A2,"a":=A2,"a":=A2,"b":=B2,"b":=B2,"b":=2} = M2, +    #{"a":=_,"a":=_,"a":=_,"b":=_,"b":=_,"b":=2} = M2, +    ok. + +%% simple build and match variables +t_build_and_match_variables(Config) when is_list(Config) -> +    K0 = id(#{}), +    K1 = id(1),       V1 = id(a), +    K2 = id(2),       V2 = id(b), +    K3 = id(3),       V3 = id("c"), +    K4 = id("4"),     V4 = id("d"), +    K5 = id(<<"5">>), V5 = id(<<"e">>), +    K6 = id({"6",7}), V6 = id("f"), +    K7 = id(#{ "a" => 3 }), +    #{K1:=V1} = id(#{K1=>V1}), +    #{K1:=V1,K2:=V2} = id(#{K1=>V1,K2=>V2}), +    #{K1:=V1,K2:=V2,K3:=V3} = id(#{K1=>V1,K2=>V2,K3=>V3}), +    #{K1:=V1,K2:=V2,K3:=V3,K4:=V4} = id(#{K1=>V1,K2=>V2,K3=>V3,K4=>V4}), +    #{K1:=V1,K2:=V2,K3:=V3,K4:=V4,K5:=V5} = id(#{K1=>V1,K2=>V2,K3=>V3,K4=>V4,K5=>V5}), +    #{K1:=V1,K2:=V2,K3:=V3,K4:=V4,K5:=V5,K6:=V6} = id(#{K1=>V1,K2=>V2,K3=>V3,K4=>V4,K5=>V5,K6=>V6}), + +    #{K5:=X,K5:=X=3,K4:=4} = id(#{K5=>3,K4=>4}), +    #{K5:=X,<<"5">>:=X=3,K4:=4} = id(#{K5=>3,K4=>4}), +    #{K5:=X,<<"5">>:=X=3,K4:=4} = id(#{<<"5">>=>3,K4=>4}), + +    #{ K4:=#{ K3:=#{K1:=V1, K2:=V2}}, K5:=V5} = +	id(#{ K5=>V5, K4=>#{ K3=>#{K2 => V2, K1 => V1}}}), +    #{ K4 := #{ K5 := Res }, K6 := Res} = id(#{K4=>#{K5 => 99}, K6 => 99}), + +    %% has keys +    #{a :=_,b :=_,K1:=_,K2:=_,K3:=V3,K4:=ResKey,K4:=ResKey,"4":=ResKey,"4":="ok"} = +	id(#{ a=>1, b=>1, K1=>V1, K2=>V2, K3=>V3, K4=>"nope", "4"=>"ok" }), + +    %% function +    ok = match_function_map_neg_keys(#{ -1 => a, -2 => b, -3 => c }), + +    %% map key +    #{ K0 := 42 } = id(#{ K0 => 42 }), +    #{ K7 := 42 } = id(#{ K7 => 42 }), + +    %% nil key +    KNIL = id([]), +    #{KNIL:=ok,1:=2} = id(#{KNIL=>ok,1=>2}), + +    Bin = <<0:258>>, +    #{ Bin := "three" } = id(#{<<0:258>> =>"three"}), + +    %% error case +    {'EXIT',{{badmatch,_},_}} = (catch (#{K5:=3,x:=2} = id(#{K5=>3}))), +    {'EXIT',{{badmatch,_},_}} = (catch (#{K5:=2} = id(#{K5=>3}))), +    {'EXIT',{{badmatch,_},_}} = (catch (#{K5:=3} = id({a,b,c}))), +    {'EXIT',{{badmatch,_},_}} = (catch (#{K5:=3} = id(#{K6=>3}))), +    {'EXIT',{{badmatch,_},_}} = (catch (#{K5:=3} = id(K7))), +    {'EXIT',{{badmatch,_},_}} = (catch (#{K7:=3} = id(#{K7=>42}))), +    ok. + + +match_function_map_neg_keys(#{ -1 := a, -2 := b, -3 := c }) -> ok. + +t_update_assoc_variables(Config) when is_list(Config) -> +    K1 = id(1), +    K2 = id(2), +    K3 = id(3.0), +    K4 = id(4), +    K5 = id(5), +    K6 = id(2.0), + +    M0 = #{K1=>a,K2=>b,K3=>c,K4=>d,K5=>e}, + +    M1 = M0#{K1=>42,K2=>100,K4=>[a,b,c]}, +    #{1:=42,2:=100,3.0:=c,4:=[a,b,c],5:=e} = M1, +    #{1:=42,2:=b,4:=d,5:=e,2.0:=100,K3:=c,4.0:=[a,b,c]} = M0#{1.0=>float,1:=42,2.0=>wrong,K6=>100,4.0=>[a,b,c]}, + +    M2 = M0#{K3=>new}, +    #{1:=a,2:=b,K3:=new,4:=d,5:=e} = M2, +    M2 = M0#{3.0:=wrong,K3=>new}, + +    #{ <<0:258>> := val } = id(M0#{<<0:258>> => val}), %% binary limitation + +    %% Errors cases. +    BadMap = id(badmap), +    {'EXIT',{{badarg,_},_}} = (catch BadMap#{nonexisting=>val}), +    {'EXIT',{{badarg,_},_}} = (catch <<>>#{nonexisting=>val}), +    ok. + +t_update_exact_variables(Config) when is_list(Config) -> +    K1 = id(1), +    K2 = id(2), +    K3 = id(3.0), +    K4 = id(4), + +    M0 = id(#{1=>a,2=>b,3.0=>c,4=>d,5=>e}), + +    M1 = M0#{K1:=42,K2:=100,K4:=[a,b,c]}, +    #{1:=42,2:=100,3.0:=c,K4:=[a,b,c],5:=e} = M1, +    M1 = M0#{K1:=wrong,1:=also_wrong,K1=>42,2=>wrong,K2:=100,4:=[a,b,c]}, + +    M2 = M0#{K3:=new}, +    #{1:=a,K2:=b,3.0:=new,K4:=d,5:=e} = M2, +    M2 = M0#{3.0=>wrong,K3:=new}, +    true = M2 =/= M0#{3=>right,3.0:=new}, +    #{ 3 := right, 3.0 := new } = M0#{3=>right,K3:=new}, + +    M3 = id(#{ 1 => val}), +    #{1 := update2,1.0 := new_val4} = M3#{ +	1.0 => new_val1, K1 := update, K1=> update3, +	K1 := update2, 1.0 := new_val2, 1.0 => new_val3, +	1.0 => new_val4 }, + +    %% Errors cases. +    {'EXIT',{{badarg,_},_}} = (catch ((id(nil))#{ a := b })), +    {'EXIT',{{badarg,_},_}} = (catch <<>>#{nonexisting:=val}), + +    {'EXIT',{badarg,_}} = (catch M0#{nonexisting:=val}), +    {'EXIT',{badarg,_}} = (catch M0#{1.0:=v,1.0=>v2}), +    {'EXIT',{badarg,_}} = (catch M0#{42.0:=v,42:=v2}), +    {'EXIT',{badarg,_}} = (catch M0#{42=>v1,42.0:=v2,42:=v3}), +    {'EXIT',{badarg,_}} = (catch M0#{<<0:257>> := val}), %% limitation +    ok. + +t_nested_pattern_expressions(Config) when is_list(Config) -> +    K1 = id("hello"), +    %K2 = id({ok}), +    [_,_,#{ <<"hi">> := wat, K1 := 42 }|_] = id([k,k,#{<<"hi">> => wat, K1 => 42}]), +    [_,_,#{ -1 := wat, K1 := 42 }|_] = id([k,k,#{-1 => wat, K1 => 42}]), +    [_,_,{#{ -1 := #{ {-3,<<0:300>>} := V1 }, K1 := 42 },3}|_] = id([k,k,{#{-1 => #{{-3,<<0:300>>}=>"hi"}, K1 => 42},3}]), +    "hi" = V1, +    %[k,#{ {-1,K1,[]} := {wat,K1}, K2 := 42 }|_] = id([k,#{{-1,K1,[]} => {wat,K1}, K2 => 42}]), +    %[k,#{ [-1,K2,[]] := {wat,K1}, K1 := 42 }|_] = id([k,#{[-1,K2,[]] => {wat,K1}, K1 => 42}]), +    ok. + +t_guard_update_variables(Config) when is_list(Config) -> +    error  = map_guard_update_variables(n,#{},#{}), +    first  = map_guard_update_variables(x,#{}, #{x=>first}), +    second = map_guard_update_variables(x,#{y=>old}, #{x=>second,y=>old}), +    third  = map_guard_update_variables(x,#{x=>old,y=>old}, #{x=>third,y=>old}), +    fourth = map_guard_update_variables(x,#{x=>old,y=>old}, #{x=>4,y=>new}), +    ok. + +map_guard_update_variables(K,M1,M2) when M1#{K=>first}    =:= M2 -> first; +map_guard_update_variables(K,M1,M2) when M1#{K=>second}   =:= M2 -> second; +map_guard_update_variables(K,M1,M2) when M1#{K:=third}    =:= M2 -> third; +map_guard_update_variables(K,M1,M2) when M1#{K:=4,y=>new} =:= M2 -> fourth; +map_guard_update_variables(_,_,_) -> error. + +t_guard_sequence_variables(Config) when is_list(Config) -> +    {1,"a"} = map_guard_sequence_var_1(a,#{seq=>1,a=>id("a"),b=>no}), +    {2,"b"} = map_guard_sequence_var_1(b,#{seq=>2,b=>id("b"),a=>no}), +    {3,"c"} = map_guard_sequence_var_1(a,#{seq=>3,a=>id("c"),b=>no}), +    {4,"d"} = map_guard_sequence_var_1(b,#{seq=>4,b=>id("d"),a=>no}), +    {4,4}   = map_guard_sequence_var_1(seq,#{seq=>4}), +    {4,4,y} = map_guard_sequence_var_1(seq,#{seq=>4,b=>id("d"),a=>y}), +    {5,"d"} = map_guard_sequence_var_1(b,#{seq=>5,b=>id("d"),a=>y}), + +    %% error case +    {'EXIT',{{case_clause,_},_}} = (catch map_guard_sequence_var_1("a",#{seq=>4,val=>id("e")})), +    ok. + + +map_guard_sequence_var_1(K,M) -> +    case M of +	#{seq:=1=Seq, K:=Val} -> {Seq,Val}; +	#{seq:=2=Seq, K:=Val} -> {Seq,Val}; +	#{seq:=3=Seq, K:=Val} -> {Seq,Val}; +	#{K:=4=Seq,   K:=Val1,a:=Val2} -> {Seq,Val1,Val2}; +	#{seq:=4=Seq, K:=Val} -> {Seq,Val}; +	#{K:=4=Seq,   K:=Val} -> {Seq,Val}; +	#{seq:=5=Seq, K:=Val} -> {Seq,Val} +    end. + + +t_guard_sequence_mixed(Config) when is_list(Config) -> +    M0 = id(#{ a=>1, b=>1, c=>1, d=>1, e=>1, f=>1, g=>1, h=>1 }), +    M1 = id(M0#{ d := 3 }), +    1 = map_guard_sequence_mixed(a,d,M1), +    M2 = id(M1#{ b := 2, d := 4, h := 2 }), +    2 = map_guard_sequence_mixed(a,d,M2), +    M3 = id(M2#{ b := 3, e := 5, g := 3 }), +    3 = map_guard_sequence_mixed(a,e,M3), +    M4 = id(M3#{ c := 4, e := 6, h := 1 }), +    4 = map_guard_sequence_mixed(a,e,M4), +    M5 = id(M4#{ c := 5, f := 7, g := 2 }), +    5 = map_guard_sequence_mixed(a,f,M5), +    M6 = id(M5#{ c := 6, f := 8, h := 3 }), +    6 = map_guard_sequence_mixed(a,f,M6), + +    %% error case +    {'EXIT',{{case_clause,_},_}} = (catch map_guard_sequence_mixed(a,b,M0)), +    ok. + +map_guard_sequence_mixed(K1,K2,M) -> +    case M of +	#{ K1 := 1, b := 1, K2 := 3, g := 1} -> 1; +	#{ K1 := 1, b := 2, K2 := 4, h := 2} -> 2; +	#{ K1 := 1, b := 3, K2 := 5, g := 3} -> 3; +	#{ K1 := 1, c := 4, K2 := 6, h := 1} -> 4; +	#{ K1 := 1, c := 5, K2 := 7, g := 2} -> 5; +	#{ K1 := 1, c := 6, K2 := 8, h := 3} -> 6 +    end. + + + +t_frequency_table(Config) when is_list(Config) -> +    random:seed({13,1337,54}),  % pseudo random +    N = 100000, +    Ts = rand_terms(N), +    #{ n:=N, tf := Tf } = frequency_table(Ts,#{ n=>0, tf => #{}}), +    ok = check_frequency(Ts,Tf), +    ok. + + +frequency_table([T|Ts], M) -> +    case M of +	#{ n := N, tf := #{ T := C } = F } -> +	    frequency_table(Ts,M#{ n := N + 1, tf := F#{ T := C + 1 }}); +	#{ n := N, tf := F } -> +	    frequency_table(Ts,M#{ n := N + 1, tf := F#{ T => 1 }}) +    end; +frequency_table([], M) -> M. + + +check_frequency(Ts,Tf) -> +    check_frequency(Ts,Tf,dict:new()). + +check_frequency([T|Ts],Tf,D) -> +    case dict:find(T,D) of +	error  -> check_frequency(Ts,Tf,dict:store(T,1,D)); +	{ok,C} -> check_frequency(Ts,Tf,dict:store(T,C+1,D)) +    end; +check_frequency([],Tf,D) -> +    validate_frequency(dict:to_list(D),Tf). + +validate_frequency([{T,C}|Fs],Tf) -> +    case Tf of +	#{ T := C } -> validate_frequency(Fs,Tf); +	_ -> error +    end; +validate_frequency([], _) -> ok. + + +%% aux + +rand_terms(0) -> []; +rand_terms(N) -> [rand_term()|rand_terms(N-1)]. + +rand_term() -> +    case random:uniform(6) of +	1 -> rand_binary(); +	2 -> rand_number(); +	3 -> rand_atom(); +	4 -> rand_tuple(); +	5 -> rand_list(); +	6 -> rand_map() +    end. + +rand_binary() -> +    case random:uniform(3) of +	1 -> <<>>; +	2 -> <<"hi">>; +	3 -> <<"message text larger than 64 bytes. yep, message text larger than 64 bytes.">> +    end. + +rand_number() -> +    case random:uniform(3) of +	1 -> random:uniform(5); +	2 -> float(random:uniform(5)); +	3 -> 1 bsl (63 + random:uniform(3)) +    end. + +rand_atom() -> +    case random:uniform(3) of +	1 -> hi; +	2 -> some_atom; +	3 -> some_other_atom +    end. + + +rand_tuple() -> +    case random:uniform(3) of +	1 -> {ok, rand_term()}; % careful +	2 -> {1, 2, 3}; +	3 -> {<<"yep">>, 1337} +    end. + +rand_list() -> +    case random:uniform(3) of +	1 -> "hi"; +	2 -> [1,rand_term()]; % careful +	3 -> [improper|list] +    end. + +rand_map() -> +    case random:uniform(3) of +	1 -> #{ hi => 3 }; +	2 -> #{ wat => rand_term(), other => 3 };  % careful +	3 -> #{ hi => 42, other => 42, yet_anoter => 1337 } +    end. + + +t_build_and_match_over_alloc(Config) when is_list(Config) -> +    Ls = id([1,2,3]), +    V0 = [a|Ls], +    M0 = id(#{ "a" => V0 }), +    #{ "a" := V1 } = M0, +    V2 = id([c|Ls]), +    M2 = id(#{ "a" => V2 }), +    #{ "a" := V3 } = M2, +    {[a,1,2,3],[c,1,2,3]} = id({V1,V3}), +    ok. + +t_build_and_match_empty_val(Config) when is_list(Config) -> +    F = fun(#{ "hi":=_,{1,2}:=_,1337:=_}) -> ok end, +    ok = F(id(#{"hi"=>ok,{1,2}=>ok,1337=>ok})), + +    %% error case +    case (catch (F(id(#{"hi"=>ok})))) of +	{'EXIT',{function_clause,_}} -> ok; +	{'EXIT', {{case_clause,_},_}} -> {comment,inlined}; +	Other -> +	    test_server:fail({no_match, Other}) +    end. + +t_build_and_match_val(Config) when is_list(Config) -> +    F = fun +	(#{ "hi" := first,  v := V}) -> {1,V}; +	(#{ "hi" := second, v := V}) -> {2,V} +    end, + + +    {1,"hello"}  = F(id(#{"hi"=>first,v=>"hello"})), +    {2,"second"} = F(id(#{"hi"=>second,v=>"second"})), + +    %% error case +    case (catch (F(id(#{"hi"=>ok})))) of +	{'EXIT',{function_clause,_}} -> ok; +	{'EXIT', {{case_clause,_},_}} -> {comment,inlined}; +	Other -> +	    test_server:fail({no_match, Other}) +    end. + +t_build_and_match_nil(Config) when is_list(Config) -> +    %% literals removed the coverage +    V1 = id(cookie), +    V2 = id(cake), +    V3 = id(crisps), + +    #{ [] := V1, "treat" := V2, {your,treat} := V3 } = id(#{ +	    {your,treat} => V3, +	    "treat" => V2,  +	    [] => V1 }), +    #{ [] := V3, [] := V3 } = id(#{ [] => V1, [] => V3 }), +    ok. + +t_build_and_match_structure(Config) when is_list(Config) -> +    V2 = id("it"), +    S = id([42,{"hi", "=)", #{ "a" => 42, any => any, val => "get_" ++ V2}}]), + +    %% match deep map values +    V2 = case S of +	[42,{"hi",_, #{ "a" := 42, val := "get_" ++ V1, any := _ }}] -> V1 +    end, +    %% match deep map +    ok = case S of +	[42,{"hi",_, #{ }}] -> ok +    end,      ok. -getmsg(_Tracer) -> -    receive V -> V after 100 -> timeout end. -trace_collector(Msg,Parent) -> -    io:format("~p~n",[Msg]), -    Parent ! Msg, -    Parent.  %% Use this function to avoid compile-time evaluation of an expression.  id(I) -> I. diff --git a/lib/stdlib/src/erl_eval.erl b/lib/stdlib/src/erl_eval.erl index 639ddfc214..371573dc23 100644 --- a/lib/stdlib/src/erl_eval.erl +++ b/lib/stdlib/src/erl_eval.erl @@ -1172,7 +1172,7 @@ match_tuple([], _, _, Bs, _BBs) ->  match_map([{map_field_exact, _, K, V}|Fs], Map, Bs0, BBs) ->      Vm = try -	{value, Ke, _} = expr(K, new_bindings()), +	{value, Ke, _} = expr(K, Bs0),  	maps:get(Ke,Map)      catch error:_ ->  	throw(nomatch) diff --git a/lib/stdlib/src/erl_lint.erl b/lib/stdlib/src/erl_lint.erl index 6619ed5221..39f8a26fe1 100644 --- a/lib/stdlib/src/erl_lint.erl +++ b/lib/stdlib/src/erl_lint.erl @@ -239,10 +239,7 @@ format_error({too_many_arguments,Arity}) ->  		  "maximum allowed is ~w", [Arity,?MAX_ARGUMENTS]);  %% --- patterns and guards ---  format_error(illegal_pattern) -> "illegal pattern"; -format_error(illegal_map_key) -> -    "illegal map key"; -format_error({illegal_map_key_variable,K}) -> -    io_lib:format("illegal use of variable ~w in map",[K]); +format_error(illegal_map_key) -> "illegal map key in pattern";  format_error(illegal_bin_pattern) ->      "binary patterns cannot be matched in parallel using '='";  format_error(illegal_expr) -> "illegal expression"; @@ -1440,20 +1437,7 @@ pattern({cons,_Line,H,T}, Vt, Old,  Bvt, St0) ->  pattern({tuple,_Line,Ps}, Vt, Old, Bvt, St) ->      pattern_list(Ps, Vt, Old, Bvt, St);  pattern({map,_Line,Ps}, Vt, Old, Bvt, St) -> -    foldl(fun -	    ({map_field_assoc,L,_,_}, {Psvt,Bvt0,St0}) -> -		{Psvt,Bvt0,add_error(L, illegal_pattern, St0)}; -	    ({map_field_exact,L,KP,VP}, {Psvt,Bvt0,St0}) -> -		case is_valid_map_key(KP, pattern, St0) of -		    true -> -			{Pvt,Bvt1,St1} = pattern(VP, Vt, Old, Bvt, St0), -			{vtmerge_pat(Pvt, Psvt),vtmerge_pat(Bvt0, Bvt1), St1}; -		    false -> -			{Psvt,Bvt0,add_error(L, illegal_map_key, St0)}; -		    {false,variable,Var} -> -			{Psvt,Bvt0,add_error(L, {illegal_map_key_variable,Var}, St0)} -		end -	end, {[],[],St}, Ps); +    pattern_map(Ps, Vt, Old, Bvt, St);  %%pattern({struct,_Line,_Tag,Ps}, Vt, Old, Bvt, St) ->  %%    pattern_list(Ps, Vt, Old, Bvt, St);  pattern({record_index,Line,Name,Field}, _Vt, _Old, _Bvt, St) -> @@ -1607,6 +1591,21 @@ is_pattern_expr_1({op,_Line,Op,A1,A2}) ->      erl_internal:arith_op(Op, 2) andalso all(fun is_pattern_expr/1, [A1,A2]);  is_pattern_expr_1(_Other) -> false. +pattern_map(Ps, Vt, Old, Bvt, St) -> +    foldl(fun +	    ({map_field_assoc,L,_,_}, {Psvt,Bvt0,St0}) -> +		{Psvt,Bvt0,add_error(L, illegal_pattern, St0)}; +	    ({map_field_exact,L,K,V}, {Psvt,Bvt0,St0}) -> +		case is_valid_map_key(K) of +		    true -> +			{Kvt,St1} = expr(K, Vt, St0), +			{Vvt,Bvt2,St2} = pattern(V, Vt, Old, Bvt, St1), +			{vtmerge_pat(vtmerge_pat(Kvt, Vvt), Psvt), vtmerge_pat(Bvt0, Bvt2), St2}; +		    false -> +			{Psvt,Bvt0,add_error(L, illegal_map_key, St0)} +		end +	end, {[],[],St}, Ps). +  %% pattern_bin([Element], VarTable, Old, BinVarTable, State) ->  %%           {UpdVarTable,UpdBinVarTable,State}.  %%  Check a pattern group. BinVarTable are used binsize variables. @@ -2288,14 +2287,9 @@ check_assoc_fields([{map_field_assoc,_,_,_}|Fs], St) ->  check_assoc_fields([], St) ->      St. -map_fields([{Tag,Line,K,V}|Fs], Vt, St, F) when Tag =:= map_field_assoc; -                                                Tag =:= map_field_exact -> -    St1 = case is_valid_map_key(K, St) of -	true -> St; -	false -> add_error(Line, illegal_map_key, St); -	{false,variable,Var} -> add_error(Line, {illegal_map_key_variable,Var}, St) -    end, -    {Pvt,St2} = F([K,V], Vt, St1), +map_fields([{Tag,_,K,V}|Fs], Vt, St, F) when Tag =:= map_field_assoc; +                                             Tag =:= map_field_exact -> +    {Pvt,St2} = F([K,V], Vt, St),      {Vts,St3} = map_fields(Fs, Vt, St2, F),      {vtupdate(Pvt, Vts),St3};  map_fields([], Vt, St, _) -> @@ -2353,21 +2347,14 @@ is_valid_call(Call) ->          _ -> true      end. -%% is_valid_map_key(K,St) -> true | false | {false, Var::atom()} -%%   check for value expression without variables - -is_valid_map_key(K,St) -> -    is_valid_map_key(K,expr,St). -is_valid_map_key(K,Ctx,St) -> -    case expr(K,[],St) of -	{[],_} -> -	    is_valid_map_key_value(K,Ctx); -	{[Var|_],_} -> -	    {false,variable,element(1,Var)} -    end. +%% is_valid_map_key(K,St) -> true | false +%%   variables are allowed for patterns only at the top of the tree -is_valid_map_key_value(K,Ctx) -> +is_valid_map_key({var,_,_}) -> true; +is_valid_map_key(K) -> is_valid_map_key_value(K). +is_valid_map_key_value(K) ->      case K of +	{var,_,_} -> false;  	{char,_,_} -> true;  	{integer,_,_} -> true;  	{float,_,_} -> true; @@ -2375,36 +2362,36 @@ is_valid_map_key_value(K,Ctx) ->  	{nil,_} -> true;  	{atom,_,_} -> true;  	{cons,_,H,T} -> -	    is_valid_map_key_value(H,Ctx) andalso -	    is_valid_map_key_value(T,Ctx); +	    is_valid_map_key_value(H) andalso +	    is_valid_map_key_value(T);  	{tuple,_,Es} ->  	    foldl(fun(E,B) -> -			B andalso is_valid_map_key_value(E,Ctx) +			B andalso is_valid_map_key_value(E)  		end,true,Es);  	{map,_,Arg,Ps} ->  	    % only check for value expressions to be valid  	    % invalid map expressions are later checked in  	    % core and kernel -	    is_valid_map_key_value(Arg,Ctx) andalso foldl(fun +	    is_valid_map_key_value(Arg) andalso foldl(fun  		    ({Tag,_,Ke,Ve},B) when Tag =:= map_field_assoc; -					   Tag =:= map_field_exact, Ctx =:= expr -> -		    B andalso is_valid_map_key_value(Ke,Ctx) -		      andalso is_valid_map_key_value(Ve,Ctx); +					   Tag =:= map_field_exact -> +		    B andalso is_valid_map_key_value(Ke) +		      andalso is_valid_map_key_value(Ve);  		    (_,_) -> false  	    end,true,Ps);  	{map,_,Ps} ->  	    foldl(fun  		    ({Tag,_,Ke,Ve},B) when Tag =:= map_field_assoc; -					   Tag =:= map_field_exact, Ctx =:= expr -> -		    B andalso is_valid_map_key_value(Ke,Ctx) -		      andalso is_valid_map_key_value(Ve,Ctx); +					   Tag =:= map_field_exact -> +		    B andalso is_valid_map_key_value(Ke) +		      andalso is_valid_map_key_value(Ve);  		    (_,_) -> false  	    end, true, Ps);  	{record,_,_,Fs} ->  	    foldl(fun  		    ({record_field,_,Ke,Ve},B) -> -		    B andalso is_valid_map_key_value(Ke,Ctx) -		      andalso is_valid_map_key_value(Ve,Ctx) +		    B andalso is_valid_map_key_value(Ke) +		      andalso is_valid_map_key_value(Ve)  	      end,true,Fs);  	{bin,_,Es} ->  	    % only check for value expressions to be valid @@ -2412,9 +2399,9 @@ is_valid_map_key_value(K,Ctx) ->  	    % core and kernel  	    foldl(fun  		    ({bin_element,_,E,_,_},B) -> -		    B andalso is_valid_map_key_value(E,Ctx) +		    B andalso is_valid_map_key_value(E)  		end,true,Es); -	_ -> false +	Val -> is_pattern_expr(Val)      end.  %% record_def(Line, RecordName, [RecField], State) -> State. diff --git a/lib/stdlib/test/erl_eval_SUITE.erl b/lib/stdlib/test/erl_eval_SUITE.erl index b55324161b..3427f431c5 100644 --- a/lib/stdlib/test/erl_eval_SUITE.erl +++ b/lib/stdlib/test/erl_eval_SUITE.erl @@ -1458,6 +1458,30 @@ eep43(Config) when is_list(Config) ->            "lists:map(fun (X) -> X#{price := 0} end,                       [#{hello => 0, price => nil}]).",            [#{hello => 0, price => 0}]), +    check(fun () -> +		Map = #{ <<33:333>> => "wat" }, +		#{ <<33:333>> := "wat" } = Map +	  end, +	  "begin " +	  "   Map = #{ <<33:333>> => \"wat\" }, " +	  "   #{ <<33:333>> := \"wat\" } = Map  " +	  "end.", +	  #{ <<33:333>> => "wat" }), +    check(fun () -> +		K1 = 1, +		K2 = <<42:301>>, +		K3 = {3,K2}, +		Map = #{ K1 => 1, K2 => 2, K3 => 3, {2,2} => 4}, +		#{ K1 := 1, K2 := 2, K3 := 3, {2,2} := 4} = Map +	  end, +	  "begin " +	  "    K1 = 1, " +	  "    K2 = <<42:301>>, " +	  "    K3 = {3,K2}, " +	  "    Map = #{ K1 => 1, K2 => 2, K3 => 3, {2,2} => 4}, " +	  "    #{ K1 := 1, K2 := 2, K3 := 3, {2,2} := 4} = Map " +	  "end.", +	  #{ 1 => 1, <<42:301>> => 2, {3,<<42:301>>} => 3, {2,2} => 4}),      error_check("[camembert]#{}.", {badarg,[camembert]}),      error_check("#{} = 1.", {badmatch,1}),      ok. diff --git a/lib/stdlib/test/erl_lint_SUITE.erl b/lib/stdlib/test/erl_lint_SUITE.erl index 27f95bd3fa..f8a99f653a 100644 --- a/lib/stdlib/test/erl_lint_SUITE.erl +++ b/lib/stdlib/test/erl_lint_SUITE.erl @@ -3667,7 +3667,8 @@ maps(Config) ->                     g := 1 + 1,                     h := _,                     i := (_X = _Y), -                   j := (x ! y) }) -> +                   j := (x ! y), +		   <<0:300>> := 33}) ->                    {A,F}.              ">>,             [], @@ -3680,9 +3681,10 @@ maps(Config) ->  	   {errors,[{1,erl_lint,illegal_map_construction},                      {1,erl_lint,{unbound_var,'X'}}],              []}}, -	  {errors_in_map_keys, +	  {legal_map_construction,  	   <<"t(V) -> #{ a => 1,  			#{a=>V} => 2, +			#{{a,V}=>V} => 2,  			#{ \"hi\" => wazzup, hi => ho } => yep,  			[try a catch _:_ -> b end] => nope,  			ok => 1.0, @@ -3694,11 +3696,7 @@ maps(Config) ->  		      }.  	   ">>,  	   [], -	   {errors,[{2,erl_lint,{illegal_map_key_variable,'V'}}, -		    {4,erl_lint,illegal_map_key}, -		    {6,erl_lint,illegal_map_key}, -		    {8,erl_lint,illegal_map_key}, -		    {10,erl_lint,illegal_map_key}],[]}}, +	   []},  	   {errors_in_map_keys_pattern,  	   <<"t(#{ a := 2,  	           #{} := A, @@ -3709,8 +3707,8 @@ maps(Config) ->  	       A.  	   ">>,  	   [], -	   {errors,[{4,erl_lint,illegal_map_key}, -		    {6,erl_lint,{illegal_map_key_variable,'V'}}],[]}}], +	   {errors,[{4,erl_lint,illegal_map_construction}, +		    {6,erl_lint,illegal_map_key}],[]}}],      [] = run(Config, Ts),      ok. | 
