diff options
Diffstat (limited to 'lib/stdlib')
25 files changed, 617 insertions, 235 deletions
| diff --git a/lib/stdlib/doc/src/maps.xml b/lib/stdlib/doc/src/maps.xml index b37f7fd7fd..64229fa8d3 100644 --- a/lib/stdlib/doc/src/maps.xml +++ b/lib/stdlib/doc/src/maps.xml @@ -319,6 +319,23 @@ false</code>  		</func>  		<func> +			<name name="with" arity="2"/> +			<fsummary></fsummary> +			<desc> +				<p> +					Returns a new map <c><anno>Map2</anno></c> with the keys <c>K1</c> through <c>Kn</c> and their associated values from map <c><anno>Map1</anno></c>. +					Any key in <c><anno>Ks</anno></c> that does not exist in <c><anno>Map1</anno></c> are ignored. +				</p> +				<p>Example:</p> +				<code type="none"> +> Map = #{42 => value_three,1337 => "value two","a" => 1}, +  Ks = ["a",42,"other key"], +  maps:without(Ks,Map). +#{42 => value_three,"a" => 1}</code> +			</desc> +		</func> + +		<func>  			<name name="without" arity="2"/>  			<fsummary></fsummary>  			<desc> diff --git a/lib/stdlib/doc/src/notes.xml b/lib/stdlib/doc/src/notes.xml index 0421d560b6..ebc750a399 100644 --- a/lib/stdlib/doc/src/notes.xml +++ b/lib/stdlib/doc/src/notes.xml @@ -30,6 +30,98 @@    </header>    <p>This document describes the changes made to the STDLIB application.</p> +<section><title>STDLIB 2.2</title> + +    <section><title>Fixed Bugs and Malfunctions</title> +      <list> +        <item> +          <p> +	    The type spec of the FormFunc argument to +	    sys:handle_debug/4 was erroneously pointing to dbg_fun(). +	    This is now corrected and the new type is format_fun().</p> +          <p> +	    Own Id: OTP-11800</p> +        </item> +        <item> +          <p> +	    Behaviors such as gen_fsm and gen_server should always +	    invoke format_status/2 before printing the state to the +	    logs.</p> +          <p> +	    Own Id: OTP-11967</p> +        </item> +        <item> +	    <p> The documentation of <c>dets:insert_new/2</c> has +	    been corrected. (Thanks to Alexei Sholik for reporting +	    the bug.) </p> +          <p> +	    Own Id: OTP-12024</p> +        </item> +        <item> +          <p> +	    Printing a term with io_lib:format and control sequence +	    w, precision P and field width F, where F< P would +	    fail in one of the two following ways:</p> +          <p> +	    1) If P < printed length of the term, an infinite loop +	    would be entered, consuming all available memory.</p> +          <p> +	    2) If P >= printed length of the term, an exception +	    would be raised.</p> +          <p> +	    These two problems are now corrected.</p> +          <p> +	    Own Id: OTP-12041</p> +        </item> +        <item> +          <p> +	    The documentation of <c>maps:values/1</c> has been +	    corrected.</p> +          <p> +	    Own Id: OTP-12055</p> +        </item> +        <item> +          <p> +	    Expand shell functions in map expressions.</p> +          <p> +	    Own Id: OTP-12063</p> +        </item> +      </list> +    </section> + + +    <section><title>Improvements and New Features</title> +      <list> +        <item> +          <p> +	    Add maps:with/2</p> +          <p> +	    Own Id: OTP-12137</p> +        </item> +      </list> +    </section> + +</section> + +<section><title>STDLIB 2.1.1</title> + +    <section><title>Fixed Bugs and Malfunctions</title> +      <list> +        <item> +          <p> +	    OTP-11850 fixed filelib:wildcard/1 to work with broken +	    symlinks. This correction, however, introduced problems +	    since symlinks were no longer followed for functions like +	    filelib:ensure_dir/1, filelib:is_dir/1, +	    filelib:file_size/1, etc. This is now corrected.</p> +          <p> +	    Own Id: OTP-12054 Aux Id: seq12660 </p> +        </item> +      </list> +    </section> + +</section> +  <section><title>STDLIB 2.1</title>      <section><title>Fixed Bugs and Malfunctions</title> diff --git a/lib/stdlib/doc/src/string.xml b/lib/stdlib/doc/src/string.xml index c96cc95a44..b05d5cbc08 100644 --- a/lib/stdlib/doc/src/string.xml +++ b/lib/stdlib/doc/src/string.xml @@ -4,7 +4,7 @@  <erlref>    <header>      <copyright> -      <year>1996</year><year>2013</year> +      <year>1996</year><year>2014</year>        <holder>Ericsson AB. All Rights Reserved.</holder>      </copyright>      <legalnotice> @@ -124,6 +124,10 @@          <code type="none">  > tokens("abc defxxghix jkl", "x ").  ["abc", "def", "ghi", "jkl"]        </code> +	<p>Note that, as shown in the example above, two or more +	adjacent separator characters in <c><anno>String</anno></c> +	will be treated as one. That is, there will not be any empty +	strings in the resulting list of tokens.</p>        </desc>      </func>      <func> diff --git a/lib/stdlib/doc/src/sys.xml b/lib/stdlib/doc/src/sys.xml index a46fa1289f..19605f325b 100644 --- a/lib/stdlib/doc/src/sys.xml +++ b/lib/stdlib/doc/src/sys.xml @@ -4,7 +4,7 @@  <erlref>    <header>      <copyright> -      <year>1996</year><year>2013</year> +      <year>1996</year><year>2014</year>        <holder>Ericsson AB. All Rights Reserved.</holder>      </copyright>      <legalnotice> @@ -115,6 +115,9 @@      <datatype>        <name name="dbg_fun"/>      </datatype> +    <datatype> +      <name name="format_fun"/> +    </datatype>    </datatypes>    <funcs>      <func> diff --git a/lib/stdlib/src/erl_pp.erl b/lib/stdlib/src/erl_pp.erl index 82bc2c1460..3dbb5ab64c 100644 --- a/lib/stdlib/src/erl_pp.erl +++ b/lib/stdlib/src/erl_pp.erl @@ -300,7 +300,15 @@ map_pair_types(Fs) ->      tuple_type(Fs, fun map_pair_type/1).  map_pair_type({type,_Line,map_field_assoc,Ktype,Vtype}) -> -    {seq,[],[]," =>",[ltype(Ktype),ltype(Vtype)]}. +    map_assoc_typed(lexpr(Ktype, options(none)), Vtype). + +map_assoc_typed(B, {type,_,union,Ts}) -> +    {first,[B,$\s],{seq,[],[],[],map_assoc_union_type(Ts)}}; +map_assoc_typed(B, Type) -> +    {list,[{cstep,[B," =>"],ltype(Type)}]}. + +map_assoc_union_type([T|Ts]) -> +    [[leaf("=> "),ltype(T)] | ltypes(Ts, fun union_elem/1)].  record_type(Name, Fields) ->      {first,[record_name(Name)],field_types(Fields)}. diff --git a/lib/stdlib/src/erl_scan.erl b/lib/stdlib/src/erl_scan.erl index ae59d5f44f..6fd6bb888b 100644 --- a/lib/stdlib/src/erl_scan.erl +++ b/lib/stdlib/src/erl_scan.erl @@ -1075,7 +1075,7 @@ scan_number([$#|Cs]=Cs0, St, Line, Col, Toks, Ncs0) ->      Ncs = lists:reverse(Ncs0),      case catch list_to_integer(Ncs) of          B when B >= 2, B =< 1+$Z-$A+10 -> -            Bcs = ?STR(St, Ncs++[$#]), +            Bcs = Ncs++[$#],              scan_based_int(Cs, St, Line, Col, Toks, {B,[],Bcs});          B ->              Len = length(Ncs), @@ -1108,7 +1108,7 @@ scan_based_int(Cs, St, Line, Col, Toks, {B,Ncs0,Bcs}) ->      Ncs = lists:reverse(Ncs0),      case catch erlang:list_to_integer(Ncs, B) of          N when is_integer(N) -> -            tok3(Cs, St, Line, Col, Toks, integer, ?STR(St, Bcs++Ncs), N); +            tok3(Cs, St, Line, Col, Toks, integer, Bcs++Ncs, N);          _ ->              Len = length(Bcs)+length(Ncs),              Ncol = incr_column(Col, Len), diff --git a/lib/stdlib/src/filelib.erl b/lib/stdlib/src/filelib.erl index b1dabf614a..daae1fd2d2 100644 --- a/lib/stdlib/src/filelib.erl +++ b/lib/stdlib/src/filelib.erl @@ -265,7 +265,7 @@ do_wildcard(Pattern, Cwd, Mod) ->      lists:sort(Files).  do_wildcard_1({exists,File}, Mod) -> -    case eval_read_file_info(File, Mod) of +    case eval_read_link_info(File, Mod) of  	{ok,_} -> [File];  	_ -> []      end; @@ -488,7 +488,7 @@ badpattern(Reason) ->      error({badpattern,Reason}).  eval_read_file_info(File, file) -> -    file:read_link_info(File); +    file:read_file_info(File);  eval_read_file_info(File, erl_prim_loader) ->      case erl_prim_loader:read_file_info(File) of  	error -> {error, erl_prim_loader}; @@ -497,6 +497,16 @@ eval_read_file_info(File, erl_prim_loader) ->  eval_read_file_info(File, Mod) ->      Mod:read_file_info(File). +eval_read_link_info(File, file) -> +    file:read_link_info(File); +eval_read_link_info(File, erl_prim_loader) -> +    case erl_prim_loader:read_link_info(File) of +        error -> {error, erl_prim_loader}; +        Res-> Res +    end; +eval_read_link_info(File, Mod) -> +    Mod:read_link_info(File). +  eval_list_dir(Dir, file) ->      file:list_dir(Dir);  eval_list_dir(Dir, erl_prim_loader) -> diff --git a/lib/stdlib/src/gen_event.erl b/lib/stdlib/src/gen_event.erl index d39dd89d3a..40c5a93d59 100644 --- a/lib/stdlib/src/gen_event.erl +++ b/lib/stdlib/src/gen_event.erl @@ -49,8 +49,6 @@  -import(error_logger, [error_msg/2]). --define(reply(X), From ! {element(2,Tag), X}). -  -record(handler, {module             :: atom(),  		  id = false,  		  state, @@ -249,49 +247,49 @@ handle_msg(Msg, Parent, ServerName, MSL, Debug) ->  	{notify, Event} ->  	    {Hib,MSL1} = server_notify(Event, handle_event, MSL, ServerName),  	    loop(Parent, ServerName, MSL1, Debug, Hib); -	{From, Tag, {sync_notify, Event}} -> +	{_From, Tag, {sync_notify, Event}} ->  	    {Hib, MSL1} = server_notify(Event, handle_event, MSL, ServerName), -	    ?reply(ok), +	    reply(Tag, ok),  	    loop(Parent, ServerName, MSL1, Debug, Hib);  	{'EXIT', From, Reason} ->  	    MSL1 = handle_exit(From, Reason, MSL, ServerName),  	    loop(Parent, ServerName, MSL1, Debug, false); -	{From, Tag, {call, Handler, Query}} -> +	{_From, Tag, {call, Handler, Query}} ->  	    {Hib, Reply, MSL1} = server_call(Handler, Query, MSL, ServerName), -	    ?reply(Reply), +	    reply(Tag, Reply),  	    loop(Parent, ServerName, MSL1, Debug, Hib); -	{From, Tag, {add_handler, Handler, Args}} -> +	{_From, Tag, {add_handler, Handler, Args}} ->  	    {Hib, Reply, MSL1} = server_add_handler(Handler, Args, MSL), -	    ?reply(Reply), +	    reply(Tag, Reply),  	    loop(Parent, ServerName, MSL1, Debug, Hib); -	{From, Tag, {add_sup_handler, Handler, Args, SupP}} -> +	{_From, Tag, {add_sup_handler, Handler, Args, SupP}} ->  	    {Hib, Reply, MSL1} = server_add_sup_handler(Handler, Args, MSL, SupP), -	    ?reply(Reply), +	    reply(Tag, Reply),  	    loop(Parent, ServerName, MSL1, Debug, Hib); -	{From, Tag, {delete_handler, Handler, Args}} -> +	{_From, Tag, {delete_handler, Handler, Args}} ->  	    {Reply, MSL1} = server_delete_handler(Handler, Args, MSL,  						  ServerName), -	    ?reply(Reply), +	    reply(Tag, Reply),  	    loop(Parent, ServerName, MSL1, Debug, false); -	{From, Tag, {swap_handler, Handler1, Args1, Handler2, Args2}} -> +	{_From, Tag, {swap_handler, Handler1, Args1, Handler2, Args2}} ->  	    {Hib, Reply, MSL1} = server_swap_handler(Handler1, Args1, Handler2,  						     Args2, MSL, ServerName), -	    ?reply(Reply), +	    reply(Tag, Reply),  	    loop(Parent, ServerName, MSL1, Debug, Hib); -	{From, Tag, {swap_sup_handler, Handler1, Args1, Handler2, Args2, +	{_From, Tag, {swap_sup_handler, Handler1, Args1, Handler2, Args2,  		     Sup}} ->  	    {Hib, Reply, MSL1} = server_swap_handler(Handler1, Args1, Handler2,  						Args2, MSL, Sup, ServerName), -	    ?reply(Reply), +	    reply(Tag, Reply),  	    loop(Parent, ServerName, MSL1, Debug, Hib); -	{From, Tag, stop} -> +	{_From, Tag, stop} ->  	    catch terminate_server(normal, Parent, MSL, ServerName), -	    ?reply(ok); -	{From, Tag, which_handlers} -> -	    ?reply(the_handlers(MSL)), +	    reply(Tag, ok); +	{_From, Tag, which_handlers} -> +	    reply(Tag, the_handlers(MSL)),  	    loop(Parent, ServerName, MSL, Debug, false); -	{From, Tag, get_modules} -> -	    ?reply(get_modules(MSL)), +	{_From, Tag, get_modules} -> +	    reply(Tag, get_modules(MSL)),  	    loop(Parent, ServerName, MSL, Debug, false);  	Other  ->  	    {Hib, MSL1} = server_notify(Other, handle_info, MSL, ServerName), @@ -303,6 +301,9 @@ terminate_server(Reason, Parent, MSL, ServerName) ->      do_unlink(Parent, MSL),      exit(Reason). +reply({From, Ref}, Msg) -> +    From ! {Ref, Msg}. +  %% unlink the supervisor process of all supervised handlers.  %% We do not want a handler supervisor to EXIT due to the  %% termination of the event manager (server). diff --git a/lib/stdlib/src/gen_fsm.erl b/lib/stdlib/src/gen_fsm.erl index e914f7d0b2..5afe3e8b09 100644 --- a/lib/stdlib/src/gen_fsm.erl +++ b/lib/stdlib/src/gen_fsm.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1996-2013. All Rights Reserved. +%% Copyright Ericsson AB 1996-2014. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -594,7 +594,8 @@ reply(Name, {To, Tag}, Reply, Debug, StateName) ->  terminate(Reason, Name, Msg, Mod, StateName, StateData, Debug) ->      case catch Mod:terminate(Reason, StateName, StateData) of  	{'EXIT', R} -> -	    error_info(R, Name, Msg, StateName, StateData, Debug), +	    FmtStateData = format_status(terminate, Mod, get(), StateData), +	    error_info(R, Name, Msg, StateName, FmtStateData, Debug),  	    exit(R);  	_ ->  	    case Reason of @@ -605,17 +606,7 @@ terminate(Reason, Name, Msg, Mod, StateName, StateData, Debug) ->   		{shutdown,_}=Shutdown ->   		    exit(Shutdown);  		_ -> -                    FmtStateData = -                        case erlang:function_exported(Mod, format_status, 2) of -                            true -> -                                Args = [get(), StateData], -                                case catch Mod:format_status(terminate, Args) of -                                    {'EXIT', _} -> StateData; -                                    Else -> Else -                                end; -                            _ -> -                                StateData -                        end, +                    FmtStateData = format_status(terminate, Mod, get(), StateData),  		    error_info(Reason,Name,Msg,StateName,FmtStateData,Debug),  		    exit(Reason)  	    end @@ -680,21 +671,29 @@ format_status(Opt, StatusData) ->      Header = gen:format_status_header("Status for state machine",                                        Name),      Log = sys:get_debug(log, Debug, []), -    DefaultStatus = [{data, [{"StateData", StateData}]}], -    Specfic = -	case erlang:function_exported(Mod, format_status, 2) of -	    true -> -		case catch Mod:format_status(Opt,[PDict,StateData]) of -		    {'EXIT', _} -> DefaultStatus; -                    StatusList when is_list(StatusList) -> StatusList; -		    Else -> [Else] -		end; -	    _ -> -		DefaultStatus -	end, +    Specfic = format_status(Opt, Mod, PDict, StateData), +    Specfic = case format_status(Opt, Mod, PDict, StateData) of +		  S when is_list(S) -> S; +		  S -> [S] +	      end,      [{header, Header},       {data, [{"Status", SysState},  	     {"Parent", Parent},  	     {"Logged events", Log},  	     {"StateName", StateName}]} |       Specfic]. + +format_status(Opt, Mod, PDict, State) -> +    DefStatus = case Opt of +		    terminate -> State; +		    _ -> [{data, [{"StateData", State}]}] +		end, +    case erlang:function_exported(Mod, format_status, 2) of +	true -> +	    case catch Mod:format_status(Opt, [PDict, State]) of +		{'EXIT', _} -> DefStatus; +		Else -> Else +	    end; +	_ -> +	    DefStatus +    end. diff --git a/lib/stdlib/src/gen_server.erl b/lib/stdlib/src/gen_server.erl index 202a931fae..dadfe56b3d 100644 --- a/lib/stdlib/src/gen_server.erl +++ b/lib/stdlib/src/gen_server.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1996-2013. All Rights Reserved. +%% Copyright Ericsson AB 1996-2014. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -720,7 +720,8 @@ print_event(Dev, Event, Name) ->  terminate(Reason, Name, Msg, Mod, State, Debug) ->      case catch Mod:terminate(Reason, State) of  	{'EXIT', R} -> -	    error_info(R, Name, Msg, State, Debug), +	    FmtState = format_status(terminate, Mod, get(), State), +	    error_info(R, Name, Msg, FmtState, Debug),  	    exit(R);  	_ ->  	    case Reason of @@ -731,17 +732,7 @@ terminate(Reason, Name, Msg, Mod, State, Debug) ->  		{shutdown,_}=Shutdown ->  		    exit(Shutdown);  		_ -> -		    FmtState = -			case erlang:function_exported(Mod, format_status, 2) of -			    true -> -				Args = [get(), State], -				case catch Mod:format_status(terminate, Args) of -				    {'EXIT', _} -> State; -				    Else -> Else -				end; -			    _ -> -				State -			end, +		    FmtState = format_status(terminate, Mod, get(), State),  		    error_info(Reason, Name, Msg, FmtState, Debug),  		    exit(Reason)  	    end @@ -875,23 +866,29 @@ name_to_pid(Name) ->  %%-----------------------------------------------------------------  format_status(Opt, StatusData) ->      [PDict, SysState, Parent, Debug, [Name, State, Mod, _Time]] = StatusData, -    Header = gen:format_status_header("Status for generic server", -                                      Name), +    Header = gen:format_status_header("Status for generic server", Name),      Log = sys:get_debug(log, Debug, []), -    DefaultStatus = [{data, [{"State", State}]}], -    Specfic = -	case erlang:function_exported(Mod, format_status, 2) of -	    true -> -		case catch Mod:format_status(Opt, [PDict, State]) of -		    {'EXIT', _} -> DefaultStatus; -                    StatusList when is_list(StatusList) -> StatusList; -		    Else -> [Else] -		end; -	    _ -> -		DefaultStatus -	end, +    Specfic = case format_status(Opt, Mod, PDict, State) of +		  S when is_list(S) -> S; +		  S -> [S] +	      end,      [{header, Header},       {data, [{"Status", SysState},  	     {"Parent", Parent},  	     {"Logged events", Log}]} |       Specfic]. + +format_status(Opt, Mod, PDict, State) -> +    DefStatus = case Opt of +		    terminate -> State; +		    _ -> [{data, [{"State", State}]}] +		end, +    case erlang:function_exported(Mod, format_status, 2) of +	true -> +	    case catch Mod:format_status(Opt, [PDict, State]) of +		{'EXIT', _} -> DefStatus; +		Else -> Else +	    end; +	_ -> +	    DefStatus +    end. diff --git a/lib/stdlib/src/maps.erl b/lib/stdlib/src/maps.erl index 4ef1638e6d..ba4d6a5c87 100644 --- a/lib/stdlib/src/maps.erl +++ b/lib/stdlib/src/maps.erl @@ -24,6 +24,7 @@  	map/2,  	size/1,      without/2, +    with/2,      get/3      ]). @@ -133,10 +134,10 @@ to_list(_) -> erlang:nif_error(undef).  update(_,_,_) -> erlang:nif_error(undef). --spec values(Map) -> Keys when +-spec values(Map) -> Values when      Map :: map(), -    Keys :: [Key], -    Key :: term(). +    Values :: [Value], +    Value :: term().  values(_) -> erlang:nif_error(undef). @@ -201,3 +202,13 @@ size(Map) when is_map(Map) ->  without(Ks, M) when is_list(Ks), is_map(M) ->      maps:from_list([{K,V}||{K,V} <- maps:to_list(M), not lists:member(K, Ks)]). + + +-spec with(Ks, Map1) -> Map2 when +    Ks :: [K], +    Map1 :: map(), +    Map2 :: map(), +    K :: term(). + +with(Ks, M) when is_list(Ks), is_map(M) -> +    maps:from_list([{K,V}||{K,V} <- maps:to_list(M), lists:member(K, Ks)]). diff --git a/lib/stdlib/src/otp_internal.erl b/lib/stdlib/src/otp_internal.erl index c0ee8799c8..6c25beabe9 100644 --- a/lib/stdlib/src/otp_internal.erl +++ b/lib/stdlib/src/otp_internal.erl @@ -421,13 +421,13 @@ obsolete_1(ssh_cm, stop_listener, 1) ->  obsolete_1(ssh_cm, session_open, A) when A =:= 2; A =:= 4 ->      {removed,{ssh_connection,session_channel,A},"R14B"};  obsolete_1(ssh_cm, direct_tcpip, A) when A =:= 6; A =:= 8 -> -    {removed,{ssh_connection,direct_tcpip,A}}; +    {removed,{ssh_connection,direct_tcpip,A},"R14B"};  obsolete_1(ssh_cm, tcpip_forward, 3) ->      {removed,{ssh_connection,tcpip_forward,3},"R14B"};  obsolete_1(ssh_cm, cancel_tcpip_forward, 3) ->      {removed,{ssh_connection,cancel_tcpip_forward,3},"R14B"};  obsolete_1(ssh_cm, open_pty, A) when A =:= 3; A =:= 7; A =:= 9 -> -    {removed,{ssh_connection,open_pty,A},"R14"}; +    {removed,{ssh_connection,open_pty,A},"R14B"};  obsolete_1(ssh_cm, setenv, 5) ->      {removed,{ssh_connection,setenv,5},"R14B"};  obsolete_1(ssh_cm, shell, 2) -> @@ -441,11 +441,11 @@ obsolete_1(ssh_cm, winch, A) when A =:= 4; A =:= 6 ->  obsolete_1(ssh_cm, signal, 3) ->      {removed,{ssh_connection,signal,3},"R14B"};  obsolete_1(ssh_cm, attach, A) when A =:= 2; A =:= 3 -> -    {removed,{ssh,attach,A}}; +    {removed,"no longer useful; removed in R14B"};  obsolete_1(ssh_cm, detach, 2) -> -    {removed,"no longer useful; will be removed in R14B"}; +    {removed,"no longer useful; removed in R14B"};  obsolete_1(ssh_cm, set_user_ack, 4) -> -    {removed,"no longer useful; will be removed in R14B"}; +    {removed,"no longer useful; removed in R14B"};  obsolete_1(ssh_cm, adjust_window, 3) ->      {removed,{ssh_connection,adjust_window,3},"R14B"};  obsolete_1(ssh_cm, close, 2) -> @@ -461,9 +461,9 @@ obsolete_1(ssh_cm, send_ack, A) when 3 =< A, A =< 5 ->  obsolete_1(ssh_ssh, connect, A) when 1 =< A, A =< 3 ->      {removed,{ssh,shell,A},"R14B"};  obsolete_1(ssh_sshd, listen, A) when 0 =< A, A =< 3 -> -    {removed,{ssh,daemon,[1,2,3]},"R14"}; +    {removed,{ssh,daemon,[1,2,3]},"R14B"};  obsolete_1(ssh_sshd, stop, 1) -> -    {removed,{ssh,stop_listener,1}}; +    {removed,{ssh,stop_listener,1},"R14B"};  %% Added in R13A.  obsolete_1(regexp, _, _) -> diff --git a/lib/stdlib/src/proc_lib.erl b/lib/stdlib/src/proc_lib.erl index 1eb6fc2e86..bf2a4e7ac5 100644 --- a/lib/stdlib/src/proc_lib.erl +++ b/lib/stdlib/src/proc_lib.erl @@ -216,10 +216,8 @@ ensure_link(SpawnOpts) ->  init_p(Parent, Ancestors, Fun) when is_function(Fun) ->      put('$ancestors', [Parent|Ancestors]), -    {module,Mod} = erlang:fun_info(Fun, module), -    {name,Name} = erlang:fun_info(Fun, name), -    {arity,Arity} = erlang:fun_info(Fun, arity), -    put('$initial_call', {Mod,Name,Arity}), +    Mfa = erlang:fun_info_mfa(Fun), +    put('$initial_call', Mfa),      try  	Fun()      catch diff --git a/lib/stdlib/src/shell.erl b/lib/stdlib/src/shell.erl index 3b90542452..679c13f0cf 100644 --- a/lib/stdlib/src/shell.erl +++ b/lib/stdlib/src/shell.erl @@ -371,6 +371,14 @@ expand_expr({bc,L,E,Qs}, C) ->      {bc,L,expand_expr(E, C),expand_quals(Qs, C)};  expand_expr({tuple,L,Elts}, C) ->      {tuple,L,expand_exprs(Elts, C)}; +expand_expr({map,L,Es}, C) -> +    {map,L,expand_exprs(Es, C)}; +expand_expr({map,L,Arg,Es}, C) -> +    {map,L,expand_expr(Arg, C),expand_exprs(Es, C)}; +expand_expr({map_field_assoc,L,K,V}, C) -> +    {map_field_assoc,L,expand_expr(K, C),expand_expr(V, C)}; +expand_expr({map_field_exact,L,K,V}, C) -> +    {map_field_exact,L,expand_expr(K, C),expand_expr(V, C)};  expand_expr({record_index,L,Name,F}, C) ->      {record_index,L,Name,expand_expr(F, C)};  expand_expr({record,L,Name,Is}, C) -> diff --git a/lib/stdlib/src/stdlib.app.src b/lib/stdlib/src/stdlib.app.src index d388410de0..aa9899da3b 100644 --- a/lib/stdlib/src/stdlib.app.src +++ b/lib/stdlib/src/stdlib.app.src @@ -103,7 +103,7 @@                 dets]},    {applications, [kernel]},    {env, []}, -  {runtime_dependencies, ["sasl-2.4","kernel-3.0","erts-6.0","crypto-3.3", +  {runtime_dependencies, ["sasl-2.4","kernel-3.0.2","erts-6.2","crypto-3.3",  			  "compiler-5.0"]}  ]}. diff --git a/lib/stdlib/src/stdlib.appup.src b/lib/stdlib/src/stdlib.appup.src index 22eefb2514..99d9b8b431 100644 --- a/lib/stdlib/src/stdlib.appup.src +++ b/lib/stdlib/src/stdlib.appup.src @@ -17,9 +17,11 @@  %% %CopyrightEnd%  {"%VSN%",   %% Up from - max one major revision back - [{<<"2\\.0(\\.[0-9]+)*">>,[restart_new_emulator]}, %% R17 + [{<<"2\\.1(\\.[0-9]+)*">>,[restart_new_emulator]}, %% 17.1 +  {<<"2\\.0(\\.[0-9]+)*">>,[restart_new_emulator]}, %% 17.0    {<<"1\\.19(\\.[0-9]+)*">>,[restart_new_emulator]}],%% R16   %% Down to - max one major revision back - [{<<"2\\.0(\\.[0-9]+)*">>,[restart_new_emulator]}, %% R17 + [{<<"2\\.1(\\.[0-9]+)*">>,[restart_new_emulator]}, %% 17.1 +  {<<"2\\.0(\\.[0-9]+)*">>,[restart_new_emulator]}, %% 17.0    {<<"1\\.19(\\.[0-9]+)*">>,[restart_new_emulator]}] %% R16  }. diff --git a/lib/stdlib/src/sys.erl b/lib/stdlib/src/sys.erl index e25cc25f57..d3ba09ce82 100644 --- a/lib/stdlib/src/sys.erl +++ b/lib/stdlib/src/sys.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 1996-2013. All Rights Reserved. +%% Copyright Ericsson AB 1996-2014. All Rights Reserved.  %%   %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -46,7 +46,7 @@                           {N :: non_neg_integer(),                            [{Event :: system_event(),                              FuncState :: _, -                            FormFunc :: dbg_fun()}]}} +                            FormFunc :: format_fun()}]}}                        | {'statistics', {file:date_time(),                                          {'reductions', non_neg_integer()},                                          MessagesIn :: non_neg_integer(), @@ -57,6 +57,10 @@                               Event :: system_event(),                               ProcState :: _) -> 'done' | (NewFuncState :: _)). +-type format_fun()   :: fun((Device :: io:device() | file:io_device(), +			     Event :: system_event(), +			     Extra :: term()) -> any()). +  %%-----------------------------------------------------------------  %% System messages  %%----------------------------------------------------------------- @@ -346,7 +350,7 @@ handle_system_msg(SysState, Msg, From, Parent, Mod, Debug, Misc, Hib) ->  %%-----------------------------------------------------------------  -spec handle_debug(Debug, FormFunc, Extra, Event) -> [dbg_opt()] when        Debug :: [dbg_opt()], -      FormFunc :: dbg_fun(), +      FormFunc :: format_fun(),        Extra :: term(),        Event :: system_event().  handle_debug([{trace, true} | T], FormFunc, State, Event) -> diff --git a/lib/stdlib/test/erl_scan_SUITE.erl b/lib/stdlib/test/erl_scan_SUITE.erl index 35067e8116..9be9f641c8 100644 --- a/lib/stdlib/test/erl_scan_SUITE.erl +++ b/lib/stdlib/test/erl_scan_SUITE.erl @@ -204,20 +204,20 @@ reserved_words() ->      [begin           ?line {RW, true} = {RW, erl_scan:reserved_word(RW)},           S = atom_to_list(RW), -         Ts = [{RW,1}], +         Ts = [{RW,{1,1}}],           ?line test_string(S, Ts)       end || RW <- L],      ok.  atoms() -> -    ?line test_string("a -                 b", [{atom,1,a},{atom,2,b}]), -    ?line test_string("'a b'", [{atom,1,'a b'}]), -    ?line test_string("a", [{atom,1,a}]), -    ?line test_string("a@2", [{atom,1,a@2}]), -    ?line test_string([39,65,200,39], [{atom,1,'AÈ'}]), -    ?line test_string("ärlig östen", [{atom,1,ärlig},{atom,1,östen}]), +    test_string("a +                 b", [{atom,{1,1},a},{atom,{2,18},b}]), +    test_string("'a b'", [{atom,{1,1},'a b'}]), +    test_string("a", [{atom,{1,1},a}]), +    test_string("a@2", [{atom,{1,1},a@2}]), +    test_string([39,65,200,39], [{atom,{1,1},'AÈ'}]), +    test_string("ärlig östen", [{atom,{1,1},ärlig},{atom,{1,7},östen}]),      ?line {ok,[{atom,_,'$a'}],{1,6}} =          erl_scan:string("'$\\a'", {1,1}),      ?line test("'$\\a'"), @@ -230,7 +230,7 @@ punctuations() ->      %% One token at a time:      [begin           W = list_to_atom(S), -         Ts = [{W,1}], +         Ts = [{W,{1,1}}],           ?line test_string(S, Ts)       end || S <- L],      Three = ["/=:=", "<=:=", "==:=", ">=:="], % three tokens... @@ -246,53 +246,60 @@ punctuations() ->      [begin           W1 = list_to_atom(S1),           W2 = list_to_atom(S2), -         Ts = [{W1,1},{W2,1}], +         Ts = [{W1,{1,1}},{W2,{1,-L2+1}}],           ?line test_string(S, Ts) -     end || {S,[{_,S1,S2}|_]}  <- SL], +     end || {S,[{L2,S1,S2}|_]}  <- SL], -    PTs1 = [{'!',1},{'(',1},{')',1},{',',1},{';',1},{'=',1},{'[',1}, -            {']',1},{'{',1},{'|',1},{'}',1}], +    PTs1 = [{'!',{1,1}},{'(',{1,2}},{')',{1,3}},{',',{1,4}},{';',{1,5}}, +            {'=',{1,6}},{'[',{1,7}},{']',{1,8}},{'{',{1,9}},{'|',{1,10}}, +            {'}',{1,11}}],      ?line test_string("!(),;=[]{|}", PTs1), -    PTs2 = [{'#',1},{'&',1},{'*',1},{'+',1},{'/',1}, -            {':',1},{'<',1},{'>',1},{'?',1},{'@',1}, -            {'\\',1},{'^',1},{'`',1},{'~',1}], +    PTs2 = [{'#',{1,1}},{'&',{1,2}},{'*',{1,3}},{'+',{1,4}},{'/',{1,5}}, +            {':',{1,6}},{'<',{1,7}},{'>',{1,8}},{'?',{1,9}},{'@',{1,10}}, +            {'\\',{1,11}},{'^',{1,12}},{'`',{1,13}},{'~',{1,14}}],      ?line test_string("#&*+/:<>?@\\^`~", PTs2), -    ?line test_string(".. ", [{'..',1}]), -    ?line test("1 .. 2"), -    ?line test_string("...", [{'...',1}]), +    test_string(".. ", [{'..',{1,1}}]), +    test_string("1 .. 2", +                [{integer,{1,1},1},{'..',{1,3}},{integer,{1,6},2}]), +    test_string("...", [{'...',{1,1}}]),      ok.  comments() ->      ?line test("a %%\n b"),      ?line {ok,[],1} = erl_scan:string("%"),      ?line test("a %%\n b"), -    ?line {ok,[{atom,_,a},{atom,_,b}],{2,3}} = +    {ok,[{atom,{1,1},a},{atom,{2,2},b}],{2,3}} =          erl_scan:string("a %%\n b",{1,1}), -    ?line {ok,[{atom,_,a},{comment,_,"%%"},{atom,_,b}],{2,3}} = +    {ok,[{atom,{1,1},a},{comment,{1,3},"%%"},{atom,{2,2},b}],{2,3}} =          erl_scan:string("a %%\n b",{1,1}, [return_comments]), -    ?line {ok,[{atom,_,a}, -               {white_space,_," "}, -               {white_space,_,"\n "}, -               {atom,_,b}], -           {2,3}} = +    {ok,[{atom,{1,1},a}, +         {white_space,{1,2}," "}, +         {white_space,{1,5},"\n "}, +         {atom,{2,2},b}], +     {2,3}} =          erl_scan:string("a %%\n b",{1,1},[return_white_spaces]), -    ?line {ok,[{atom,_,a}, -               {white_space,_," "}, -               {comment,_,"%%"}, -               {white_space,_,"\n "}, -               {atom,_,b}], -           {2,3}} = erl_scan:string("a %%\n b",{1,1},[return]), +    {ok,[{atom,{1,1},a}, +         {white_space,{1,2}," "}, +         {comment,{1,3},"%%"}, +         {white_space,{1,5},"\n "}, +         {atom,{2,2},b}], +     {2,3}} = erl_scan:string("a %%\n b",{1,1},[return]),      ok.  errors() ->      ?line {error,{1,erl_scan,{string,$',"qa"}},1} = erl_scan:string("'qa"), %' +    {error,{{1,1},erl_scan,{string,$',"qa"}},{1,4}} = %' +        erl_scan:string("'qa", {1,1}, []), %'      ?line {error,{1,erl_scan,{string,$","str"}},1} = %"          erl_scan:string("\"str"), %" +    {error,{{1,1},erl_scan,{string,$","str"}},{1,5}} = %" +        erl_scan:string("\"str", {1,1}, []), %"      ?line {error,{1,erl_scan,char},1} = erl_scan:string("$"), -    ?line test_string([34,65,200,34], [{string,1,"AÈ"}]), -    ?line test_string("\\", [{'\\',1}]), +    {error,{{1,1},erl_scan,char},{1,2}} = erl_scan:string("$", {1,1}, []), +    test_string([34,65,200,34], [{string,{1,1},"AÈ"}]), +    test_string("\\", [{'\\',{1,1}}]),      ?line {'EXIT',_} =          (catch {foo, erl_scan:string('$\\a', {1,1})}), % type error      ?line {'EXIT',_} = @@ -304,7 +311,7 @@ errors() ->  integers() ->      [begin           I = list_to_integer(S), -         Ts = [{integer,1,I}], +         Ts = [{integer,{1,1},I}],           ?line test_string(S, Ts)       end || S <- [[N] || N <- lists:seq($0, $9)] ++ ["2323","000"] ],      ok. @@ -313,14 +320,16 @@ base_integers() ->      [begin           B = list_to_integer(BS),           I = erlang:list_to_integer(S, B), -         Ts = [{integer,1,I}], +         Ts = [{integer,{1,1},I}],           ?line test_string(BS++"#"++S, Ts)       end || {BS,S} <- [{"2","11"}, {"5","23234"}, {"12","05a"},                         {"16","abcdef"}, {"16","ABCDEF"}] ],      ?line {error,{1,erl_scan,{base,1}},1} = erl_scan:string("1#000"), +    {error,{{1,1},erl_scan,{base,1}},{1,2}} = +        erl_scan:string("1#000", {1,1}, []), -    ?line test_string("12#bc", [{integer,1,11},{atom,1,c}]), +    test_string("12#bc", [{integer,{1,1},11},{atom,{1,5},c}]),      [begin           Str = BS ++ "#" ++ S, @@ -329,40 +338,53 @@ base_integers() ->       end || {BS,S} <- [{"3","3"},{"15","f"}, {"12","c"}] ],      ?line {ok,[{integer,1,239},{'@',1}],1} = erl_scan:string("16#ef@"), -    ?line {ok,[{integer,1,14},{atom,1,g@}],1} = erl_scan:string("16#eg@"), +    {ok,[{integer,{1,1},239},{'@',{1,6}}],{1,7}} = +        erl_scan:string("16#ef@", {1,1}, []), +    {ok,[{integer,{1,1},14},{atom,{1,5},g@}],{1,7}} = +        erl_scan:string("16#eg@", {1,1}, []),      ok.  floats() ->      [begin           F = list_to_float(FS), -         Ts = [{float,1,F}], +         Ts = [{float,{1,1},F}],           ?line test_string(FS, Ts)       end || FS <- ["1.0","001.17","3.31200","1.0e0","1.0E17",                     "34.21E-18", "17.0E+14"]], -    ?line test_string("1.e2", [{integer,1,1},{'.',1},{atom,1,e2}]), +    test_string("1.e2", [{integer,{1,1},1},{'.',{1,2}},{atom,{1,3},e2}]),      ?line {error,{1,erl_scan,{illegal,float}},1} =          erl_scan:string("1.0e400"), +    {error,{{1,1},erl_scan,{illegal,float}},{1,8}} = +        erl_scan:string("1.0e400", {1,1}, []),      [begin -         ?line {error,{1,erl_scan,{illegal,float}},1} = erl_scan:string(S) +         {error,{1,erl_scan,{illegal,float}},1} = erl_scan:string(S), +         {error,{{1,1},erl_scan,{illegal,float}},{1,_}} = +             erl_scan:string(S, {1,1}, [])       end || S <- ["1.14Ea"]],      ok.  dots() -> -    Dot = [{".",    {ok,[{dot,1}],1}}, -           {". ",   {ok,[{dot,1}],1}}, -           {".\n",  {ok,[{dot,1}],2}}, -           {".%",   {ok,[{dot,1}],1}}, -           {".\210",{ok,[{dot,1}],1}}, -           {".% öh",{ok,[{dot,1}],1}}, -           {".%\n", {ok,[{dot,1}],2}}, -           {".$",   {error,{1,erl_scan,char},1}}, -           {".$\\", {error,{1,erl_scan,char},1}}, -           {".a",   {ok,[{'.',1},{atom,1,a}],1}} +    Dot = [{".",    {ok,[{dot,1}],1}, {ok,[{dot,{1,1}}],{1,2}}}, +           {". ",   {ok,[{dot,1}],1}, {ok,[{dot,{1,1}}],{1,3}}}, +           {".\n",  {ok,[{dot,1}],2}, {ok,[{dot,{1,1}}],{2,1}}}, +           {".%",   {ok,[{dot,1}],1}, {ok,[{dot,{1,1}}],{1,3}}}, +           {".\210",{ok,[{dot,1}],1}, {ok,[{dot,{1,1}}],{1,3}}}, +           {".% öh",{ok,[{dot,1}],1}, {ok,[{dot,{1,1}}],{1,6}}}, +           {".%\n", {ok,[{dot,1}],2}, {ok,[{dot,{1,1}}],{2,1}}}, +           {".$",   {error,{1,erl_scan,char},1}, +                    {error,{{1,2},erl_scan,char},{1,3}}}, +           {".$\\", {error,{1,erl_scan,char},1}, +                    {error,{{1,2},erl_scan,char},{1,4}}}, +           {".a",   {ok,[{'.',1},{atom,1,a}],1}, +                    {ok,[{'.',{1,1}},{atom,{1,2},a}],{1,3}}}            ], -    ?line [R = erl_scan:string(S) || {S, R} <- Dot], +    [begin +         R = erl_scan:string(S), +         R2 = erl_scan:string(S, {1,1}, []) +     end || {S, R, R2} <- Dot],      ?line {ok,[{dot,_}=T1],{1,2}} = erl_scan:string(".", {1,1}, text),      ?line [{column,1},{length,1},{line,1},{text,"."}] = @@ -379,55 +401,55 @@ dots() ->      ?line {error,{{1,2},erl_scan,char},{1,4}} =          erl_scan:string(".$\\", {1,1}), -    ?line test(". "), -    ?line test(".  "), -    ?line test(".\n"), -    ?line test(".\n\n"), -    ?line test(".\n\r"), -    ?line test(".\n\n\n"), -    ?line test(".\210"), -    ?line test(".%\n"), -    ?line test(".a"), - -    ?line test("%. \n. "), +    test_string(". ", [{dot,{1,1}}]), +    test_string(".  ", [{dot,{1,1}}]), +    test_string(".\n", [{dot,{1,1}}]), +    test_string(".\n\n", [{dot,{1,1}}]), +    test_string(".\n\r", [{dot,{1,1}}]), +    test_string(".\n\n\n", [{dot,{1,1}}]), +    test_string(".\210", [{dot,{1,1}}]), +    test_string(".%\n", [{dot,{1,1}}]), +    test_string(".a", [{'.',{1,1}},{atom,{1,2},a}]), + +    test_string("%. \n. ", [{dot,{2,1}}]),      ?line {more,C} = erl_scan:tokens([], "%. ",{1,1}, return), -    ?line {done,{ok,[{comment,_,"%. "}, -                     {white_space,_,"\n"}, -                     {dot,_}], -                 {2,3}}, ""} = +    {done,{ok,[{comment,{1,1},"%. "}, +               {white_space,{1,4},"\n"}, +               {dot,{2,1}}], +           {2,3}}, ""} =          erl_scan:tokens(C, "\n. ", {1,1}, return), % any loc, any options      ?line [test_string(S, R) || -              {S, R} <- [{".$\n",   [{'.',1},{char,1,$\n}]}, -                         {"$\\\n",  [{char,1,$\n}]}, -                         {"'\\\n'", [{atom,1,'\n'}]}, -                         {"$\n",    [{char,1,$\n}]}] ], +              {S, R} <- [{".$\n",   [{'.',{1,1}},{char,{1,2},$\n}]}, +                         {"$\\\n",  [{char,{1,1},$\n}]}, +                         {"'\\\n'", [{atom,{1,1},'\n'}]}, +                         {"$\n",    [{char,{1,1},$\n}]}] ],      ok.  chars() ->      [begin           L = lists:flatten(io_lib:format("$\\~.8b", [C])), -         Ts = [{char,1,C}], +         Ts = [{char,{1,1},C}],           ?line test_string(L, Ts)       end || C <- lists:seq(0, 255)],      %% Leading zeroes...      [begin           L = lists:flatten(io_lib:format("$\\~3.8.0b", [C])), -         Ts = [{char,1,C}], +         Ts = [{char,{1,1},C}],           ?line test_string(L, Ts)       end || C <- lists:seq(0, 255)],      %% $\^\n now increments the line...      [begin           L = "$\\^" ++ [C], -         Ts = [{char,1,C band 2#11111}], +         Ts = [{char,{1,1},C band 2#11111}],           ?line test_string(L, Ts)       end || C <- lists:seq(0, 255)],      [begin           L = "$\\" ++ [C], -         Ts = [{char,1,V}], +         Ts = [{char,{1,1},V}],           ?line test_string(L, Ts)       end || {C,V} <- [{$n,$\n}, {$r,$\r}, {$t,$\t}, {$v,$\v},                        {$b,$\b}, {$f,$\f}, {$e,$\e}, {$s,$\s}, @@ -440,45 +462,45 @@ chars() ->      No = EC ++ Ds ++ X ++ New,      [begin           L = "$\\" ++ [C], -         Ts = [{char,1,C}], +         Ts = [{char,{1,1},C}],           ?line test_string(L, Ts)       end || C <- lists:seq(0, 255) -- No],      [begin           L = "'$\\" ++ [C] ++ "'", -         Ts = [{atom,1,list_to_atom("$"++[C])}], +         Ts = [{atom,{1,1},list_to_atom("$"++[C])}],           ?line test_string(L, Ts)       end || C <- lists:seq(0, 255) -- No], -    ?line test_string("\"\\013a\\\n\"", [{string,1,"\va\n"}]), +    test_string("\"\\013a\\\n\"", [{string,{1,1},"\va\n"}]), -    ?line test_string("'\n'", [{atom,1,'\n'}]), -    ?line test_string("\"\n\a\"", [{string,1,"\na"}]), +    test_string("'\n'", [{atom,{1,1},'\n'}]), +    test_string("\"\n\a\"", [{string,{1,1},"\na"}]),      %% No escape      [begin           L = "$" ++ [C], -         Ts = [{char,1,C}], +         Ts = [{char,{1,1},C}],           ?line test_string(L, Ts)       end || C <- lists:seq(0, 255) -- (No ++ [$\\])], -    ?line test_string("$\n", [{char,1,$\n}]), +    test_string("$\n", [{char,{1,1},$\n}]),      ?line {error,{{1,1},erl_scan,char},{1,4}} =          erl_scan:string("$\\^",{1,1}), -    ?line test_string("$\\\n", [{char,1,$\n}]), +    test_string("$\\\n", [{char,{1,1},$\n}]),      %% Robert's scanner returns line 1: -    ?line test_string("$\\\n", [{char,1,$\n}]), -    ?line test_string("$\n\n", [{char,1,$\n}]), +    test_string("$\\\n", [{char,{1,1},$\n}]), +    test_string("$\n\n", [{char,{1,1},$\n}]),      ?line test("$\n\n"),      ok.  variables() -> -    ?line test_string("     \237_Aouåeiyäö", [{var,1,'_Aouåeiyäö'}]), -    ?line test_string("A_b_c@", [{var,1,'A_b_c@'}]), -    ?line test_string("V@2", [{var,1,'V@2'}]), -    ?line test_string("ABDÀ", [{var,1,'ABDÀ'}]), -    ?line test_string("Ärlig Östen", [{var,1,'Ärlig'},{var,1,'Östen'}]), +    test_string("     \237_Aouåeiyäö", [{var,{1,7},'_Aouåeiyäö'}]), +    test_string("A_b_c@", [{var,{1,1},'A_b_c@'}]), +    test_string("V@2", [{var,{1,1},'V@2'}]), +    test_string("ABDÀ", [{var,{1,1},'ABDÀ'}]), +    test_string("Ärlig Östen", [{var,{1,1},'Ärlig'},{var,{1,7},'Östen'}]),      ok.  eof() -> @@ -508,11 +530,25 @@ eof() ->      ?line {done,{ok,[{atom,1,a}],1},eof} =          erl_scan:tokens(C5,eof,1), +    %% With column. +    {more, C6} = erl_scan:tokens([], "a", {1,1}), +    %% An error before R13A. +    %% {done,{error,{1,erl_scan,scan},1},eof} = +    {done,{ok,[{atom,{1,1},a}],{1,2}},eof} = +        erl_scan:tokens(C6,eof,1), +      %% A dot followed by eof is special:      ?line {more, C} = erl_scan:tokens([], "a.", 1),      ?line {done,{ok,[{atom,1,a},{dot,1}],1},eof} = erl_scan:tokens(C,eof,1),      ?line {ok,[{atom,1,foo},{dot,1}],1} = erl_scan:string("foo."), +    %% With column. +    {more, CCol} = erl_scan:tokens([], "a.", {1,1}), +    {done,{ok,[{atom,{1,1},a},{dot,{1,2}}],{1,3}},eof} = +        erl_scan:tokens(CCol,eof,1), +    {ok,[{atom,{1,1},foo},{dot,{1,4}}],{1,5}} = +        erl_scan:string("foo.", {1,1}, []), +      ok.  illegal() -> @@ -816,34 +852,34 @@ unicode() ->          erl_scan:string([1089]),      ?line {error,{{1,1},erl_scan,{illegal,character}},{1,2}} =          erl_scan:string([1089], {1,1}), -    ?line {error,{1,erl_scan,{illegal,atom}},1} = +    {error,{1,erl_scan,{illegal,atom}},1} =          erl_scan:string("'a"++[1089]++"b'", 1), -    ?line {error,{{1,1},erl_scan,{illegal,atom}},{1,6}} = +    {error,{{1,1},erl_scan,{illegal,atom}},{1,6}} =          erl_scan:string("'a"++[1089]++"b'", {1,1}),      ?line test("\"a"++[1089]++"b\""), -    ?line {ok,[{char,1,1}],1} = +    {ok,[{char,1,1}],1} =          erl_scan:string([$$,$\\,$^,1089], 1), -    ?line {error,{1,erl_scan,Error},1} = +    {error,{1,erl_scan,Error},1} =          erl_scan:string("\"qa\x{aaa}", 1), -    ?line "unterminated string starting with \"qa"++[2730]++"\"" = +    "unterminated string starting with \"qa"++[2730]++"\"" =          erl_scan:format_error(Error),      ?line {error,{{1,1},erl_scan,_},{1,11}} =          erl_scan:string("\"qa\\x{aaa}",{1,1}), -    ?line {error,{{1,1},erl_scan,{illegal,atom}},{1,12}} = +    {error,{{1,1},erl_scan,{illegal,atom}},{1,12}} =          erl_scan:string("'qa\\x{aaa}'",{1,1}), -    ?line {ok,[{char,1,1089}],1} = +    {ok,[{char,1,1089}],1} =          erl_scan:string([$$,1089], 1), -    ?line {ok,[{char,1,1089}],1} = +    {ok,[{char,1,1089}],1} =          erl_scan:string([$$,$\\,1089], 1),      Qs = "$\\x{aaa}", -    ?line {ok,[{char,1,$\x{aaa}}],1} = +    {ok,[{char,1,$\x{aaa}}],1} =          erl_scan:string(Qs, 1), -    ?line {ok,[Q2],{1,9}} = +    {ok,[Q2],{1,9}} =          erl_scan:string("$\\x{aaa}", {1,1}, [text]), -    ?line [{category,char},{column,1},{length,8}, +    [{category,char},{column,1},{length,8},             {line,1},{symbol,16#aaa},{text,Qs}] =          erl_scan:token_info(Q2), @@ -1164,7 +1200,13 @@ otp_11807(Config) when is_list(Config) ->           (catch erl_parse:abstract("string", [{encoding,bad}])),     ok. -test_string(String, Expected) -> +test_string(String, ExpectedWithCol) -> +    {ok, ExpectedWithCol, _EndWithCol} = erl_scan:string(String, {1, 1}, []), +    Expected = [ begin +                     {L,_C} = element(2, T), +                     setelement(2, T, L) +                 end +                    || T <- ExpectedWithCol ],      {ok, Expected, _End} = erl_scan:string(String),      test(String). diff --git a/lib/stdlib/test/filelib_SUITE.erl b/lib/stdlib/test/filelib_SUITE.erl index 1135828fae..bd313390b3 100644 --- a/lib/stdlib/test/filelib_SUITE.erl +++ b/lib/stdlib/test/filelib_SUITE.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2005-2013. All Rights Reserved. +%% Copyright Ericsson AB 2005-2014. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -23,7 +23,8 @@  	 init_per_group/2,end_per_group/2,  	 init_per_testcase/2,end_per_testcase/2,  	 wildcard_one/1,wildcard_two/1,wildcard_errors/1, -	 fold_files/1,otp_5960/1,ensure_dir_eexist/1,symlinks/1]). +	 fold_files/1,otp_5960/1,ensure_dir_eexist/1,ensure_dir_symlink/1, +	 wildcard_symlink/1, is_file_symlink/1, file_props_symlink/1]).  -import(lists, [foreach/2]). @@ -43,7 +44,8 @@ suite() -> [{ct_hooks,[ts_install_cth]}].  all() ->       [wildcard_one, wildcard_two, wildcard_errors, -     fold_files, otp_5960, ensure_dir_eexist, symlinks]. +     fold_files, otp_5960, ensure_dir_eexist, ensure_dir_symlink, +     wildcard_symlink, is_file_symlink, file_props_symlink].  groups() ->       []. @@ -368,9 +370,28 @@ ensure_dir_eexist(Config) when is_list(Config) ->      ?line {error, eexist} = filelib:ensure_dir(NeedFileB),      ok. -symlinks(Config) when is_list(Config) -> +ensure_dir_symlink(Config) when is_list(Config) ->      PrivDir = ?config(priv_dir, Config), -    Dir = filename:join(PrivDir, ?MODULE_STRING++"_symlinks"), +    Dir = filename:join(PrivDir, "ensure_dir_symlink"), +    Name = filename:join(Dir, "same_name_as_file_and_dir"), +    ok = filelib:ensure_dir(Name), +    ok = file:write_file(Name, <<"some string\n">>), +    %% With a symlink to the directory. +    Symlink = filename:join(PrivDir, "ensure_dir_symlink_link"), +    case file:make_symlink(Dir, Symlink) of +        {error,enotsup} -> +            {skip,"Symlinks not supported on this platform"}; +        {error,eperm} -> +            {win32,_} = os:type(), +            {skip,"Windows user not privileged to create symlinks"}; +        ok -> +            SymlinkedName = filename:join(Symlink, "same_name_as_file_and_dir"), +            ok = filelib:ensure_dir(SymlinkedName) +    end. + +wildcard_symlink(Config) when is_list(Config) -> +    PrivDir = ?config(priv_dir, Config), +    Dir = filename:join(PrivDir, ?MODULE_STRING++"_wildcard_symlink"),      SubDir = filename:join(Dir, "sub"),      AFile = filename:join(SubDir, "a_file"),      Alias = filename:join(Dir, "symlink"), @@ -388,6 +409,18 @@ symlinks(Config) when is_list(Config) ->  		basenames(Dir, filelib:wildcard(filename:join(Dir, "*"))),  	    ["symlink"] =  		basenames(Dir, filelib:wildcard(filename:join(Dir, "symlink"))), +	    ["sub","symlink"] = +		basenames(Dir, filelib:wildcard(filename:join(Dir, "*"), +						erl_prim_loader)), +	    ["symlink"] = +		basenames(Dir, filelib:wildcard(filename:join(Dir, "symlink"), +						erl_prim_loader)), +	    ["sub","symlink"] = +		basenames(Dir, filelib:wildcard(filename:join(Dir, "*"), +						prim_file)), +	    ["symlink"] = +		basenames(Dir, filelib:wildcard(filename:join(Dir, "symlink"), +						prim_file)),  	    ok = file:delete(AFile),  	    %% The symlink should still be visible even when its target  	    %% has been deleted. @@ -395,6 +428,18 @@ symlinks(Config) when is_list(Config) ->  		basenames(Dir, filelib:wildcard(filename:join(Dir, "*"))),  	    ["symlink"] =  		basenames(Dir, filelib:wildcard(filename:join(Dir, "symlink"))), +	    ["sub","symlink"] = +		basenames(Dir, filelib:wildcard(filename:join(Dir, "*"), +						erl_prim_loader)), +	    ["symlink"] = +		basenames(Dir, filelib:wildcard(filename:join(Dir, "symlink"), +						erl_prim_loader)), +	    ["sub","symlink"] = +		basenames(Dir, filelib:wildcard(filename:join(Dir, "*"), +						prim_file)), +	    ["symlink"] = +		basenames(Dir, filelib:wildcard(filename:join(Dir, "symlink"), +						prim_file)),  	    ok      end. @@ -403,3 +448,60 @@ basenames(Dir, Files) ->  	 Dir = filename:dirname(F),  	 filename:basename(F)       end || F <- Files]. + +is_file_symlink(Config) -> +    PrivDir = ?config(priv_dir, Config), +    Dir = filename:join(PrivDir, ?MODULE_STRING++"_is_file_symlink"), +    SubDir = filename:join(Dir, "sub"), +    AFile = filename:join(SubDir, "a_file"), +    DirAlias = filename:join(Dir, "dir_symlink"), +    FileAlias = filename:join(Dir, "file_symlink"), +    ok = file:make_dir(Dir), +    ok = file:make_dir(SubDir), +    ok = file:write_file(AFile, "not that big\n"), +    case file:make_symlink(SubDir, DirAlias) of +	{error, enotsup} -> +	    {skip, "Links not supported on this platform"}; +	{error, eperm} -> +	    {win32,_} = os:type(), +	    {skip, "Windows user not privileged to create symlinks"}; +	ok -> +	    true = filelib:is_dir(DirAlias), +	    true = filelib:is_dir(DirAlias, erl_prim_loader), +	    true = filelib:is_dir(DirAlias, prim_file), +	    true = filelib:is_file(DirAlias), +	    true = filelib:is_file(DirAlias, erl_prim_loader), +	    true = filelib:is_file(DirAlias, prim_file), +	    ok = file:make_symlink(AFile,FileAlias), +	    true = filelib:is_file(FileAlias), +	    true = filelib:is_file(FileAlias, erl_prim_loader), +	    true = filelib:is_file(FileAlias, prim_file), +	    true = filelib:is_regular(FileAlias), +	    true = filelib:is_regular(FileAlias, erl_prim_loader), +	    true = filelib:is_regular(FileAlias, prim_file), +	    ok +    end. + +file_props_symlink(Config) -> +    PrivDir = ?config(priv_dir, Config), +    Dir = filename:join(PrivDir, ?MODULE_STRING++"_file_props_symlink"), +    AFile = filename:join(Dir, "a_file"), +    Alias = filename:join(Dir, "symlink"), +    ok = file:make_dir(Dir), +    ok = file:write_file(AFile, "not that big\n"), +    case file:make_symlink(AFile, Alias) of +	{error, enotsup} -> +	    {skip, "Links not supported on this platform"}; +	{error, eperm} -> +	    {win32,_} = os:type(), +	    {skip, "Windows user not privileged to create symlinks"}; +	ok -> +	    {_,_} = LastMod = filelib:last_modified(AFile), +	    LastMod = filelib:last_modified(Alias), +	    LastMod = filelib:last_modified(Alias, erl_prim_loader), +	    LastMod = filelib:last_modified(Alias, prim_file), +	    FileSize = filelib:file_size(AFile), +	    FileSize = filelib:file_size(Alias), +	    FileSize = filelib:file_size(Alias, erl_prim_loader), +	    FileSize = filelib:file_size(Alias, prim_file) +    end. diff --git a/lib/stdlib/test/gen_fsm_SUITE.erl b/lib/stdlib/test/gen_fsm_SUITE.erl index 8aeec07ae8..336065b258 100644 --- a/lib/stdlib/test/gen_fsm_SUITE.erl +++ b/lib/stdlib/test/gen_fsm_SUITE.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1996-2013. All Rights Reserved. +%% Copyright Ericsson AB 1996-2014. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -31,7 +31,9 @@  -export([shutdown/1]). --export([ sys1/1, call_format_status/1, error_format_status/1, get_state/1, replace_state/1]). +-export([ sys1/1, +	  call_format_status/1, error_format_status/1, terminate_crash_format/1, +	  get_state/1, replace_state/1]).  -export([hibernate/1,hiber_idle/3,hiber_wakeup/3,hiber_idle/2,hiber_wakeup/2]). @@ -66,7 +68,8 @@ groups() ->         start8, start9, start10, start11, start12]},       {abnormal, [], [abnormal1, abnormal2]},       {sys, [], -      [sys1, call_format_status, error_format_status, get_state, replace_state]}]. +      [sys1, call_format_status, error_format_status, terminate_crash_format, +       get_state, replace_state]}].  init_per_suite(Config) ->      Config. @@ -403,7 +406,7 @@ error_format_status(Config) when is_list(Config) ->      receive  	{error,_GroupLeader,{Pid,  			     "** State machine"++_, -			     [Pid,{_,_,badreturn},idle,StateData,_]}} -> +			     [Pid,{_,_,badreturn},idle,{formatted,StateData},_]}} ->  	    ok;  	Other ->  	    ?line io:format("Unexpected: ~p", [Other]), @@ -413,6 +416,29 @@ error_format_status(Config) when is_list(Config) ->      process_flag(trap_exit, OldFl),      ok. +terminate_crash_format(Config) when is_list(Config) -> +    error_logger_forwarder:register(), +    OldFl = process_flag(trap_exit, true), +    StateData = crash_terminate, +    {ok, Pid} = gen_fsm:start(gen_fsm_SUITE, {state_data, StateData}, []), +    stop_it(Pid), +    receive +	{error,_GroupLeader,{Pid, +			     "** State machine"++_, +			     [Pid,{_,_,_},idle,{formatted, StateData},_]}} -> +	    ok; +	Other -> +	    io:format("Unexpected: ~p", [Other]), +	    ?t:fail() +    after 5000 -> +	    io:format("Timeout: expected error logger msg", []), +	    ?t:fail() +    end, +    [] = ?t:messages_get(), +    process_flag(trap_exit, OldFl), +    ok. + +  get_state(Config) when is_list(Config) ->      State = self(),      {ok, Pid} = gen_fsm:start(?MODULE, {state_data, State}, []), @@ -867,7 +893,8 @@ init({state_data, StateData}) ->  init(_) ->      {ok, idle, state_data}. - +terminate(_, _State, crash_terminate) -> +    exit({crash, terminate});  terminate({From, stopped}, State, _Data) ->      From ! {self(), {stopped, State}},      ok; @@ -1005,6 +1032,6 @@ handle_sync_event({get, _Pid}, _From, State, Data) ->      {reply, {state, State, Data}, State, Data}.  format_status(terminate, [_Pdict, StateData]) -> -    StateData; +    {formatted, StateData};  format_status(normal, [_Pdict, _StateData]) ->      [format_status_called]. diff --git a/lib/stdlib/test/gen_server_SUITE.erl b/lib/stdlib/test/gen_server_SUITE.erl index 960e7f60e7..42694d8b5d 100644 --- a/lib/stdlib/test/gen_server_SUITE.erl +++ b/lib/stdlib/test/gen_server_SUITE.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1996-2013. All Rights Reserved. +%% Copyright Ericsson AB 1996-2014. All Rights Reserved.  %%  %% The contents of this file are subject to the Erlang Public License,  %% Version 1.1, (the "License"); you may not use this file except in @@ -32,7 +32,8 @@  	 spec_init_local_registered_parent/1,   	 spec_init_global_registered_parent/1,  	 otp_5854/1, hibernate/1, otp_7669/1, call_format_status/1, -	 error_format_status/1, get_state/1, replace_state/1, call_with_huge_message_queue/1 +	 error_format_status/1, terminate_crash_format/1, +	 get_state/1, replace_state/1, call_with_huge_message_queue/1  	]).  % spawn export @@ -56,7 +57,8 @@ all() ->       call_remote_n3, spec_init,       spec_init_local_registered_parent,       spec_init_global_registered_parent, otp_5854, hibernate, -     otp_7669, call_format_status, error_format_status, +     otp_7669, +     call_format_status, error_format_status, terminate_crash_format,       get_state, replace_state,       call_with_huge_message_queue]. @@ -273,7 +275,7 @@ crash(Config) when is_list(Config) ->      receive  	{error,_GroupLeader4,{Pid4,  			      "** Generic server"++_, -			      [Pid4,crash,state4,crashed]}} -> +			      [Pid4,crash,{formatted, state4},crashed]}} ->  	    ok;  	Other4a ->   	    ?line io:format("Unexpected: ~p", [Other4a]), @@ -1024,7 +1026,7 @@ error_format_status(Config) when is_list(Config) ->      receive  	{error,_GroupLeader,{Pid,  			     "** Generic server"++_, -			     [Pid,crash,State,crashed]}} -> +			     [Pid,crash,{formatted, State},crashed]}} ->  	    ok;  	Other ->  	    ?line io:format("Unexpected: ~p", [Other]), @@ -1034,6 +1036,31 @@ error_format_status(Config) when is_list(Config) ->      process_flag(trap_exit, OldFl),      ok. +%% Verify that error when terminating correctly calls our format_status/2 fun +%% +terminate_crash_format(Config) when is_list(Config) -> +    error_logger_forwarder:register(), +    OldFl = process_flag(trap_exit, true), +    State = crash_terminate, +    {ok, Pid} = gen_server:start_link(?MODULE, {state, State}, []), +    gen_server:call(Pid, stop), +    receive {'EXIT', Pid, {crash, terminate}} -> ok end, +    receive +	{error,_GroupLeader,{Pid, +			     "** Generic server"++_, +			     [Pid,stop, {formatted, State},{crash, terminate}]}} -> +	    ok; +	Other -> +	    io:format("Unexpected: ~p", [Other]), +	    ?t:fail() +    after 5000 -> +	    io:format("Timeout: expected error logger msg", []), +	    ?t:fail() +    end, +    ?t:messages_get(), +    process_flag(trap_exit, OldFl), +    ok. +  %% Verify that sys:get_state correctly returns gen_server state  %%  get_state(suite) -> @@ -1323,10 +1350,12 @@ terminate({From, stopped}, _State) ->  terminate({From, stopped_info}, _State) ->      From ! {self(), stopped_info},      ok; +terminate(_, crash_terminate) -> +    exit({crash, terminate});  terminate(_Reason, _State) ->      ok.  format_status(terminate, [_PDict, State]) -> -    State; +    {formatted, State};  format_status(normal, [_PDict, _State]) ->      format_status_called. diff --git a/lib/stdlib/test/maps_SUITE.erl b/lib/stdlib/test/maps_SUITE.erl index c826ee731a..dda20a615b 100644 --- a/lib/stdlib/test/maps_SUITE.erl +++ b/lib/stdlib/test/maps_SUITE.erl @@ -24,10 +24,7 @@  -include_lib("test_server/include/test_server.hrl"). -% Default timetrap timeout (set in init_per_testcase). -% This should be set relatively high (10-15 times the expected -% max testcasetime). --define(default_timeout, ?t:minutes(4)). +-define(default_timeout, ?t:minutes(1)).  % Test server specific exports  -export([all/0]). @@ -37,13 +34,13 @@  -export([init_per_testcase/2]).  -export([end_per_testcase/2]). --export([get3/1]). +-export([t_get_3/1,t_with_2/1,t_without_2/1]).  suite() ->      [{ct_hooks, [ts_install_cth]}].  all() -> -    [get3]. +    [t_get_3,t_with_2,t_without_2].  init_per_suite(Config) ->      Config. @@ -52,7 +49,7 @@ end_per_suite(_Config) ->      ok.  init_per_testcase(_Case, Config) -> -    ?line Dog=test_server:timetrap(?default_timeout), +    Dog=test_server:timetrap(?default_timeout),      [{watchdog, Dog}|Config].  end_per_testcase(_Case, Config) -> @@ -60,10 +57,24 @@ end_per_testcase(_Case, Config) ->      test_server:timetrap_cancel(Dog),      ok. -get3(Config) when is_list(Config) -> +t_get_3(Config) when is_list(Config) ->      Map = #{ key1 => value1, key2 => value2 },      DefaultValue = "Default value", -    ?line value1 = maps:get(key1, Map, DefaultValue), -    ?line value2 = maps:get(key2, Map, DefaultValue), -    ?line DefaultValue = maps:get(key3, Map, DefaultValue), +    value1 = maps:get(key1, Map, DefaultValue), +    value2 = maps:get(key2, Map, DefaultValue), +    DefaultValue = maps:get(key3, Map, DefaultValue), +    ok. + +t_without_2(_Config) -> +    Ki = [11,22,33,44,55,66,77,88,99], +    M0 = maps:from_list([{{k,I},{v,I}}||I<-lists:seq(1,100)]), +    M1 = maps:from_list([{{k,I},{v,I}}||I<-lists:seq(1,100) -- Ki]), +    M1 = maps:without([{k,I}||I <- Ki],M0), +    ok. + +t_with_2(_Config) -> +    Ki = [11,22,33,44,55,66,77,88,99], +    M0 = maps:from_list([{{k,I},{v,I}}||I<-lists:seq(1,100)]), +    M1 = maps:from_list([{{k,I},{v,I}}||I<-Ki]), +    M1 = maps:with([{k,I}||I <- Ki],M0),      ok. diff --git a/lib/stdlib/test/shell_SUITE.erl b/lib/stdlib/test/shell_SUITE.erl index e016432f4d..f841e2c4a6 100644 --- a/lib/stdlib/test/shell_SUITE.erl +++ b/lib/stdlib/test/shell_SUITE.erl @@ -2532,6 +2532,11 @@ otp_6554(Config) when is_list(Config) ->            "\n    end.\nok.\n" =           t(<<"begin F = fun() -> foo end, 1 end. B = F(). C = 17. b().">>), +    ?line "3: command not found" = comm_err(<<"#{v(3) => v}.">>), +    ?line "3: command not found" = comm_err(<<"#{k => v(3)}.">>), +    ?line "3: command not found" = comm_err(<<"#{v(3) := v}.">>), +    ?line "3: command not found" = comm_err(<<"#{k := v(3)}.">>), +    ?line "3: command not found" = comm_err(<<"(v(3))#{}.">>),      %% Tests I'd like to do: (you should try them manually)      %% "catch spawn_link(fun() -> timer:sleep(1000), exit(foo) end)."      %%   "** exception error: foo" should be output after 1 second diff --git a/lib/stdlib/test/stdlib_SUITE.erl b/lib/stdlib/test/stdlib_SUITE.erl index 59821220b4..3d09bd27ff 100644 --- a/lib/stdlib/test/stdlib_SUITE.erl +++ b/lib/stdlib/test/stdlib_SUITE.erl @@ -78,17 +78,29 @@ appup_test(_Config) ->  appup_tests(_App,{[],[]}) ->      {skip,"no previous releases available"}; -appup_tests(App,{OkVsns,NokVsns}) -> +appup_tests(App,{OkVsns0,NokVsns}) ->      application:load(App),      {_,_,Vsn} = lists:keyfind(App,1,application:loaded_applications()),      AppupFileName = atom_to_list(App) ++ ".appup",      AppupFile = filename:join([code:lib_dir(App),ebin,AppupFileName]),      {ok,[{Vsn,UpFrom,DownTo}=AppupScript]} = file:consult(AppupFile),      ct:log("~p~n",[AppupScript]), -    ct:log("Testing ok versions: ~p~n",[OkVsns]), +    OkVsns = +	case OkVsns0 -- [Vsn] of +	    OkVsns0 -> +		OkVsns0; +	    Ok -> +		ct:log("Current version, ~p, is same as in previous release.~n" +		       "Removing this from the list of ok versions.", +		      [Vsn]), +		Ok +	end, +    ct:log("Testing that appup allows upgrade from these versions: ~p~n", +	   [OkVsns]),      check_appup(OkVsns,UpFrom,{ok,[restart_new_emulator]}),      check_appup(OkVsns,DownTo,{ok,[restart_new_emulator]}), -    ct:log("Testing not ok versions: ~p~n",[NokVsns]), +    ct:log("Testing that appup does not allow upgrade from these versions: ~p~n", +	   [NokVsns]),      check_appup(NokVsns,UpFrom,error),      check_appup(NokVsns,DownTo,error),      ok. diff --git a/lib/stdlib/vsn.mk b/lib/stdlib/vsn.mk index 52ed78c557..b522c3ea3c 100644 --- a/lib/stdlib/vsn.mk +++ b/lib/stdlib/vsn.mk @@ -1 +1 @@ -STDLIB_VSN = 2.1 +STDLIB_VSN = 2.2 | 
