diff options
Diffstat (limited to 'lib/stdlib')
| -rw-r--r-- | lib/stdlib/doc/src/lists.xml | 25 | ||||
| -rw-r--r-- | lib/stdlib/src/beam_lib.erl | 16 | ||||
| -rw-r--r-- | lib/stdlib/src/epp.erl | 2 | ||||
| -rw-r--r-- | lib/stdlib/src/erl_pp.erl | 33 | ||||
| -rw-r--r-- | lib/stdlib/src/lists.erl | 24 | ||||
| -rw-r--r-- | lib/stdlib/src/otp_internal.erl | 177 | ||||
| -rw-r--r-- | lib/stdlib/test/epp_SUITE.erl | 4 | ||||
| -rw-r--r-- | lib/stdlib/test/erl_pp_SUITE.erl | 43 | ||||
| -rw-r--r-- | lib/stdlib/test/ets_SUITE.erl | 68 | ||||
| -rw-r--r-- | lib/stdlib/test/gen_server_SUITE.erl | 16 | ||||
| -rw-r--r-- | lib/stdlib/test/lists_SUITE.erl | 4 | ||||
| -rw-r--r-- | lib/stdlib/test/zip_SUITE.erl | 25 | 
12 files changed, 380 insertions, 57 deletions
| diff --git a/lib/stdlib/doc/src/lists.xml b/lib/stdlib/doc/src/lists.xml index b6c0fa4e05..1aff78f4fc 100644 --- a/lib/stdlib/doc/src/lists.xml +++ b/lib/stdlib/doc/src/lists.xml @@ -152,6 +152,31 @@        </desc>      </func>      <func> +      <name name="filtermap" arity="2"/> +      <fsummary>Filter and map elements which satisfy a function</fsummary> +      <desc> +        <p>Calls <c><anno>Fun</anno>(<anno>Elem</anno>)</c> on successive elements <c>Elem</c> +          of <c><anno>List1</anno></c>. <c><anno>Fun</anno>/2</c> must return either a boolean +          or a tuple <c>{true, <anno>Value</anno>}</c>. The function returns the list of elements +          for which <c><anno>Fun</anno></c> returns a new value, where a value of <c>true</c> +          is synonymous with <c>{true, <anno>Elem</anno>}</c>.</p> +        <p>That is, <c>filtermap</c> behaves as if it had been defined as follows:</p> +        <code type="none"> +filtermap(Fun, List1) -> +    lists:foldr(fun(Elem, Acc) -> +                       case Fun(Elem) of +                           false -> Acc; +                           true -> [Elem|Acc]; +                           {true,Value} -> [Value|Acc] +                       end, +                end, [], List1).</code> +        <p>Example:</p> +        <pre> +> <input>lists:filtermap(fun(X) -> case X rem 2 of 0 -> {true, X div 2}; _ -> false end end, [1,2,3,4,5]).</input> +[1,2]</pre> +      </desc> +    </func> +    <func>        <name name="flatlength" arity="1"/>        <fsummary>Length of flattened deep list</fsummary>        <desc> diff --git a/lib/stdlib/src/beam_lib.erl b/lib/stdlib/src/beam_lib.erl index fe7e0f8e60..2e9ebece0e 100644 --- a/lib/stdlib/src/beam_lib.erl +++ b/lib/stdlib/src/beam_lib.erl @@ -302,10 +302,10 @@ clear_crypto_key_fun() ->  -spec make_crypto_key(mode(), string()) ->          {binary(), binary(), binary(), binary()}. -make_crypto_key(des3_cbc, String) -> +make_crypto_key(des3_cbc=Type, String) ->      <<K1:8/binary,K2:8/binary>> = First = erlang:md5(String),      <<K3:8/binary,IVec:8/binary>> = erlang:md5([First|reverse(String)]), -    {K1,K2,K3,IVec}. +    {Type,[K1,K2,K3],IVec,8}.  %%  %%  Local functions @@ -864,20 +864,20 @@ mandatory_chunks() ->  -define(CRYPTO_KEY_SERVER, beam_lib__crypto_key_server). -decrypt_abst(Mode, Module, File, Id, AtomTable, Bin) -> +decrypt_abst(Type, Module, File, Id, AtomTable, Bin) ->      try -	KeyString = get_crypto_key({debug_info, Mode, Module, File}), -	Key = make_crypto_key(des3_cbc, KeyString), -	Term = decrypt_abst_1(Mode, Key, Bin), +	KeyString = get_crypto_key({debug_info, Type, Module, File}), +	Key = make_crypto_key(Type, KeyString), +	Term = decrypt_abst_1(Key, Bin),  	{AtomTable, {Id, Term}}      catch  	_:_ ->  	    error({key_missing_or_invalid, File, Id})      end. -decrypt_abst_1(des3_cbc, {K1, K2, K3, IVec}, Bin) -> +decrypt_abst_1({Type,Key,IVec,_BlockSize}, Bin) ->      ok = start_crypto(), -    NewBin = crypto:des3_cbc_decrypt(K1, K2, K3, IVec, Bin), +    NewBin = crypto:block_decrypt(Type, Key, IVec, Bin),      binary_to_term(NewBin).  start_crypto() -> diff --git a/lib/stdlib/src/epp.erl b/lib/stdlib/src/epp.erl index e31cd63f69..d1d060ebc8 100644 --- a/lib/stdlib/src/epp.erl +++ b/lib/stdlib/src/epp.erl @@ -601,7 +601,7 @@ enter_file2(NewF, Pname, From, St0, AtLocation) ->      %% file will depend on the order of file inclusions in the parent files      Path = [filename:dirname(Pname) | tl(St0#epp.path)],      _ = set_encoding(NewF), -    #epp{file=NewF,location=Loc,name=Pname,delta=0, +    #epp{file=NewF,location=Loc,name=Pname,name2=Pname,delta=0,           sstk=[St0|St0#epp.sstk],path=Path,macs=Ms}.  enter_file_reply(From, Name, Location, AtLocation) -> diff --git a/lib/stdlib/src/erl_pp.erl b/lib/stdlib/src/erl_pp.erl index 7c7566e4ec..c0596e5ba6 100644 --- a/lib/stdlib/src/erl_pp.erl +++ b/lib/stdlib/src/erl_pp.erl @@ -35,7 +35,7 @@                         | fun((Expr :: erl_parse:abstract_expr(),                                CurrentIndentation :: integer(),                                CurrentPrecedence :: non_neg_integer(), -                              HookFunction :: hook_function()) -> +                              Options :: options()) ->                                     io_lib:chars())).  -type(option() :: {hook, hook_function()} @@ -225,7 +225,7 @@ lattribute(module, {M,Vs}, _Opts, _State) ->  lattribute(module, M, _Opts, _State) ->      attr("module", [{var,0,pname(M)}]);  lattribute(export, Falist, _Opts, _State) -> -    call({var,0,"-export"}, [falist(Falist)], 0, none); +    call({var,0,"-export"}, [falist(Falist)], 0, options(none));  lattribute(import, Name, _Opts, _State) when is_list(Name) ->      attr("import", [{var,0,pname(Name)}]);  lattribute(import, {From,Falist}, _Opts, _State) -> @@ -240,10 +240,10 @@ lattribute(Name, Arg, #options{encoding = Encoding}, _State) ->  typeattr(Tag, {TypeName,Type,Args}, _Opts) ->      {first,leaf("-"++atom_to_list(Tag)++" "), -     typed(call({atom,0,TypeName}, Args, 0, none), Type)}. +     typed(call({atom,0,TypeName}, Args, 0, options(none)), Type)}.  ltype({ann_type,_Line,[V,T]}) -> -    typed(lexpr(V, none), T); +    typed(lexpr(V, options(none)), T);  ltype({paren_type,_Line,[T]}) ->      [$(,ltype(T),$)];  ltype({type,_Line,union,Ts}) -> @@ -253,7 +253,7 @@ ltype({type,_Line,list,[T]}) ->  ltype({type,_Line,nonempty_list,[T]}) ->      {seq,$[,$],[$,],[ltype(T),leaf("...")]};  ltype({type,Line,nil,[]}) -> -    lexpr({nil,Line}, 0, none); +    lexpr({nil,Line}, 0, options(none));  ltype({type,Line,tuple,any}) ->      simple_type({atom,Line,tuple}, []);  ltype({type,_Line,tuple,Ts}) -> @@ -261,7 +261,7 @@ ltype({type,_Line,tuple,Ts}) ->  ltype({type,_Line,record,[{atom,_,N}|Fs]}) ->      record_type(N, Fs);  ltype({type,_Line,range,[_I1,_I2]=Es}) -> -    expr_list(Es, '..', fun lexpr/2, none); +    expr_list(Es, '..', fun lexpr/2, options(none));  ltype({type,_Line,binary,[I1,I2]}) ->      binary_type(I1, I2); % except binary()  ltype({type,_Line,'fun',[]}) -> @@ -277,14 +277,14 @@ ltype({remote_type,Line,[M,F,Ts]}) ->  ltype({atom,_,T}) ->      leaf(write(T));  ltype(E) -> -    lexpr(E, 0, none). +    lexpr(E, 0, options(none)).  binary_type(I1, I2) ->      B = [[] || {integer,_,0} <- [I1]] =:= [],      U = [[] || {integer,_,0} <- [I2]] =:= [],      P = max_prec(), -    E1 = [[leaf("_:"),lexpr(I1, P, none)] || B], -    E2 = [[leaf("_:_*"),lexpr(I2, P, none)] || U], +    E1 = [[leaf("_:"),lexpr(I1, P, options(none))] || B], +    E2 = [[leaf("_:_*"),lexpr(I2, P, options(none))] || U],      {seq,'<<','>>',[$,],E1++E2}.  record_type(Name, Fields) -> @@ -294,7 +294,7 @@ field_types(Fs) ->      tuple_type(Fs, fun field_type/1).  field_type({type,_Line,field_type,[Name,Type]}) -> -    typed(lexpr(Name, none), Type). +    typed(lexpr(Name, options(none)), Type).  typed(B, {type,_,union,Ts}) ->      %% Special layout for :: followed by union. @@ -330,7 +330,8 @@ sig_type(FunType) ->      fun_type([], FunType).  guard_type(Before, Gs) -> -    Gl = {list,[{step,'when',expr_list(Gs, [$,], fun constraint/2, none)}]}, +    Opts = options(none), +    Gl = {list,[{step,'when',expr_list(Gs, [$,], fun constraint/2, Opts)}]},      {list,[{step,Before,Gl}]}.  constraint({type,_Line,constraint,[Tag,As]}, _Opts) -> @@ -345,7 +346,7 @@ type_args({type,_line,product,Ts}) ->      targs(Ts).  simple_type(Tag, Types) -> -    {first,lexpr(Tag, 0, none),targs(Types)}. +    {first,lexpr(Tag, 0, options(none)),targs(Types)}.  targs(Ts) ->      {seq,$(,$),[$,],ltypes(Ts)}. @@ -357,7 +358,7 @@ ltypes(Ts, F) ->      [F(T) || T <- Ts].  attr(Name, Args) -> -    call({var,0,format("-~s", [Name])}, Args, 0, none). +    call({var,0,format("-~s", [Name])}, Args, 0, options(none)).  pname(['' | As]) ->      [$. | pname(As)]; @@ -632,11 +633,11 @@ bit_elem_types([T | Rest]) ->      [bit_elem_type(T), $-|bit_elem_types(Rest)].  bit_elem_type({A,B}) -> -    [lexpr(erl_parse:abstract(A), none), +    [lexpr(erl_parse:abstract(A), options(none)),       $:, -     lexpr(erl_parse:abstract(B), none)]; +     lexpr(erl_parse:abstract(B), options(none))];  bit_elem_type(T) -> -    lexpr(erl_parse:abstract(T), none). +    lexpr(erl_parse:abstract(T), options(none)).  %% end of BITS diff --git a/lib/stdlib/src/lists.erl b/lib/stdlib/src/lists.erl index 961c060019..0c033acd88 100644 --- a/lib/stdlib/src/lists.erl +++ b/lib/stdlib/src/lists.erl @@ -36,7 +36,7 @@  -export([merge/3, rmerge/3, sort/2, umerge/3, rumerge/3, usort/2]).  -export([all/2,any/2,map/2,flatmap/2,foldl/3,foldr/3,filter/2, -	 partition/2,zf/2, +	 partition/2,zf/2,filtermap/2,  	 mapfoldl/3,mapfoldr/3,foreach/2,takewhile/2,dropwhile/2,splitwith/2,  	 split/2]). @@ -1291,18 +1291,28 @@ partition(Pred, [H | T], As, Bs) ->  partition(Pred, [], As, Bs) when is_function(Pred, 1) ->      {reverse(As), reverse(Bs)}. --spec zf(fun((T) -> boolean() | {'true', X}), [T]) -> [(T | X)]. +-spec filtermap(Fun, List1) -> List2 when +      Fun :: fun((Elem) -> boolean() | {'true', Value}), +      List1 :: [Elem], +      List2 :: [Elem | Value], +      Elem :: term(), +      Value :: term(). -zf(F, [Hd|Tail]) -> +filtermap(F, [Hd|Tail]) ->      case F(Hd) of  	true -> -	    [Hd|zf(F, Tail)]; +	    [Hd|filtermap(F, Tail)];  	{true,Val} -> -	    [Val|zf(F, Tail)]; +	    [Val|filtermap(F, Tail)];  	false -> -	    zf(F, Tail) +	    filtermap(F, Tail)      end; -zf(F, []) when is_function(F, 1) -> []. +filtermap(F, []) when is_function(F, 1) -> []. + +-spec zf(fun((T) -> boolean() | {'true', X}), [T]) -> [(T | X)]. + +zf(F, L) -> +    filtermap(F, L).  -spec foreach(Fun, List) -> ok when        Fun :: fun((Elem :: T) -> term()), diff --git a/lib/stdlib/src/otp_internal.erl b/lib/stdlib/src/otp_internal.erl index a4f4035c79..42a42b7fd7 100644 --- a/lib/stdlib/src/otp_internal.erl +++ b/lib/stdlib/src/otp_internal.erl @@ -66,6 +66,183 @@ obsolete_1(rpc, safe_multi_server_call, A) when A =:= 2; A =:= 3 ->      {deprecated, {rpc, multi_server_call, A}}; +%% *** CRYPTO add in R16B01 *** + +obsolete_1(crypto, md4, 1) -> +    {deprecated, {crypto, hash, 2}}; +obsolete_1(crypto, md5, 1) -> +    {deprecated, {crypto, hash, 2}}; +obsolete_1(crypto, sha, 1) -> +    {deprecated, {crypto, hash, 2}}; + +obsolete_1(crypto, md4_init, 0) -> +    {deprecated, {crypto, hash_init, 1}}; +obsolete_1(crypto, md5_init, 0) -> +    {deprecated, {crypto, hash_init, 1}}; +obsolete_1(crypto, sha_init, 0) -> +    {deprecated, {crypto, hash_init, 1}}; + +obsolete_1(crypto, md4_update, 2) -> +    {deprecated, {crypto, hash_update, 3}}; +obsolete_1(crypto, md5_update, 2) -> +    {deprecated, {crypto, hash_update, 3}}; +obsolete_1(crypto, sha_update, 2) -> +    {deprecated, {crypto, hash_update, 3}}; + +obsolete_1(crypto, md4_final, 1) -> +    {deprecated, {crypto, hash_final, 2}}; +obsolete_1(crypto, md5_final, 1) -> +    {deprecated, {crypto, hash_final, 2}}; +obsolete_1(crypto, sha_final, 1) -> +    {deprecated, {crypto, hash_final, 2}}; + +obsolete_1(crypto, md5_mac, 2) -> +    {deprecated, {crypto, hmac, 3}}; +obsolete_1(crypto, sha_mac, 2) -> +    {deprecated, {crypto, hmac, 3}}; +obsolete_1(crypto, sha_mac, 3) -> +    {deprecated, {crypto, hmac, 4}}; + +obsolete_1(crypto, sha_mac_96, 2) -> +    {deprecated, {crypto, hmac_n, 3}}; +obsolete_1(crypto, md5_mac_96, 2) -> +    {deprecated, {crypto, hmac_n, 3}}; + +obsolete_1(crypto, rsa_sign, 2) -> +    {deprecated, {crypto, sign, 4}}; +obsolete_1(crypto, rsa_sign, 3) -> +    {deprecated, {crypto, sign, 4}}; +obsolete_1(crypto, rsa_verify, 3) -> +    {deprecated, {crypto, verify, 5}}; +obsolete_1(crypto, rsa_verify, 4) -> +    {deprecated, {crypto, verify, 5}}; + +obsolete_1(crypto, dss_sign, 2) -> +    {deprecated, {crypto, sign, 4}}; +obsolete_1(crypto, dss_sign, 3) -> +    {deprecated, {crypto, sign, 4}}; + +obsolete_1(crypto, dss_verify, 3) -> +    {deprecated, {crypto, verify, 4}}; +obsolete_1(crypto, dss_verify, 4) -> +    {deprecated, {crypto, verify, 4}}; + +obsolete_1(crypto, mod_exp, 3) -> +    {deprecated, {crypto, mod_pow, 3}}; + +obsolete_1(crypto, dh_compute_key, 3) -> +    {deprecated, {crypto, compute_key, 4}}; +obsolete_1(crypto, dh_generate_key, 1) -> +    {deprecated, {crypto, generate_key, 3}}; +obsolete_1(crypto, dh_generate_key, 2) -> +    {deprecated, {crypto, generate_key, 3}}; + +obsolete_1(crypto, des_cbc_encrypt, 3) -> +    {deprecated, {crypto, block_encrypt, 4}}; +obsolete_1(crypto, des3_cbc_encrypt, 5) -> +    {deprecated, {crypto, block_encrypt, 4}}; +obsolete_1(crypto, des_ecb_encrypt, 2) -> +    {deprecated, {crypto, block_encrypt, 3}}; +obsolete_1(crypto, des_ede3_cbc_encrypt, 5) -> +    {deprecated, {crypto, block_encrypt, 4}}; +obsolete_1(crypto, des_cfb_encrypt, 3) -> +    {deprecated, {crypto, block_encrypt, 4}}; +obsolete_1(crypto, des3_cfb_encrypt, 5) -> +    {deprecated, {crypto, block_encrypt, 4}}; +obsolete_1(crypto, blowfish_ecb_encrypt, 2) -> +    {deprecated, {crypto, block_encrypt, 3}}; +obsolete_1(crypto, blowfish_cbc_encrypt, 3) -> +    {deprecated, {crypto, block_encrypt, 4}}; +obsolete_1(crypto, blowfish_cfb64_encrypt, 3) -> +    {deprecated, {crypto, block_encrypt, 4}}; +obsolete_1(crypto, blowfish_ofb64_encrypt, 3) -> +    {deprecated, {crypto, block_encrypt, 4}}; +obsolete_1(crypto, aes_cfb_128_encrypt, 3) -> +    {deprecated, {crypto, block_encrypt, 4}}; +obsolete_1(crypto, aes_cbc_128_encrypt, 3) -> +    {deprecated, {crypto, block_encrypt, 4}}; +obsolete_1(crypto, aes_cbc_256_encrypt, 3) -> +    {deprecated, {crypto, block_encrypt, 4}}; +obsolete_1(crypto,rc2_cbc_encrypt, 3) -> +    {deprecated, {crypto, block_encrypt, 4}}; +obsolete_1(crypto,rc2_40_cbc_encrypt, 3) -> +    {deprecated, {crypto, block_encrypt, 4}}; + +obsolete_1(crypto, des_cbc_decrypt, 3) -> +    {deprecated, {crypto, block_decrypt, 4}}; +obsolete_1(crypto, des3_cbc_decrypt, 5) -> +    {deprecated, {crypto, block_decrypt, 4}}; +obsolete_1(crypto, des_ecb_decrypt, 2) -> +    {deprecated, {crypto, block_decrypt, 3}}; +obsolete_1(crypto, des_ede3_cbc_decrypt, 5) -> +    {deprecated, {crypto, block_decrypt, 4}}; +obsolete_1(crypto, des_cfb_decrypt, 3) -> +    {deprecated, {crypto, block_decrypt, 4}}; +obsolete_1(crypto, des3_cfb_decrypt, 5) -> +    {deprecated, {crypto, block_decrypt, 4}}; +obsolete_1(crypto, blowfish_ecb_decrypt, 2) -> +    {deprecated, {crypto, block_decrypt, 3}}; +obsolete_1(crypto, blowfish_cbc_decrypt, 3) -> +    {deprecated, {crypto, block_decrypt, 4}}; +obsolete_1(crypto, blowfish_cfb64_decrypt, 3) -> +    {deprecated, {crypto, block_decrypt, 4}}; +obsolete_1(crypto, blowfish_ofb64_decrypt, 3) -> +    {deprecated, {crypto, block_decrypt, 4}}; +obsolete_1(crypto, aes_cfb_128_decrypt, 3) -> +    {deprecated, {crypto, block_decrypt, 4}}; +obsolete_1(crypto, aes_cbc_128_decrypt, 3) -> +    {deprecated, {crypto, block_decrypt, 4}}; +obsolete_1(crypto, aes_cbc_256_decrypt, 3) -> +    {deprecated, {crypto, block_decrypt, 4}}; +obsolete_1(crypto,rc2_cbc_decrypt, 3) -> +    {deprecated, {crypto, block_decrypt, 4}}; +obsolete_1(crypto,rc2_40_cbc_decrypt, 3) -> +    {deprecated, {crypto, block_decrypt, 4}}; + +obsolete_1(crypto, aes_ctr_stream_decrypt, 2) -> +    {deprecated, {crypto, stream_decrypt, 2}}; +obsolete_1(crypto, aes_ctr_stream_encrypt, 2) -> +    {deprecated, {crypto, stream_encrypt, 2}}; +obsolete_1(crypto, aes_ctr_decrypt, 3) -> +    {deprecated, {crypto, stream_decrypt, 2}}; +obsolete_1(crypto, aes_ctr_encrypt, 3) -> +    {deprecated, {crypto, stream_encrypt, 2}}; +obsolete_1(crypto, rc4_encrypt, 2) -> +    {deprecated, {crypto, stream_encrypt, 2}}; +obsolete_1(crypto, rc4_encrypt_with_state, 2) -> +    {deprecated, {crypto, stream_encrypt, 2}}; +obsolete_1(crypto, aes_ctr_stream_init, 2) -> +    {deprecated, {crypto, stream_init, 3}}; +obsolete_1(crypto, rc4_set_key, 1) -> +    {deprecated, {crypto, stream_init, 2}}; + +obsolete_1(crypto, rsa_private_decrypt, 3) -> +    {deprecated, {crypto, private_decrypt, 4}}; +obsolete_1(crypto, rsa_public_decrypt, 3) -> +    {deprecated, {crypto, public_decrypt, 4}}; +obsolete_1(crypto, rsa_private_encrypt, 3) -> +    {deprecated, {crypto, private_encrypt, 4}}; +obsolete_1(crypto, rsa_public_encrypt, 3) -> +    {deprecated, {crypto, public_encrypt, 4}}; + +obsolete_1(crypto, des_cfb_ivec, 2) -> +    {deprecated, {crypto, next_iv, 3}}; +obsolete_1(crypto,des_cbc_ivec, 1) -> +    {deprecated, {crypto, next_iv, 2}}; +obsolete_1(crypto, aes_cbc_ivec, 1) -> +    {deprecated, {crypto, next_iv, 2}}; + +obsolete_1(crypto,info, 0) -> +    {deprecated, {crypto, module_info, 0}}; + +obsolete_1(crypto, strong_rand_mpint, 3) -> +    {deprecated, "needed only by deprecated functions"}; +obsolete_1(crypto, erlint, 1) -> +    {deprecated, "needed only by deprecated functions"}; +obsolete_1(crypto, mpint, 1) -> +    {deprecated, "needed only by deprecated functions"}; + +  %% *** SNMP ***  obsolete_1(snmp, N, A) -> diff --git a/lib/stdlib/test/epp_SUITE.erl b/lib/stdlib/test/epp_SUITE.erl index b2f1aa955a..0cbdf76270 100644 --- a/lib/stdlib/test/epp_SUITE.erl +++ b/lib/stdlib/test/epp_SUITE.erl @@ -104,6 +104,8 @@ include_local(suite) ->  include_local(Config) when is_list(Config) ->      ?line DataDir = ?config(data_dir, Config),      ?line File = filename:join(DataDir, "include_local.erl"), +    FooHrl = filename:join([DataDir,"include","foo.hrl"]), +    BarHrl = filename:join([DataDir,"include","bar.hrl"]),      %% include_local.erl includes include/foo.hrl which      %% includes bar.hrl (also in include/) without requiring      %% any additional include path, and overriding any file @@ -111,6 +113,8 @@ include_local(Config) when is_list(Config) ->      ?line {ok, List} = epp:parse_file(File, [DataDir], []),      ?line {value, {attribute,_,a,{true,true}}} =  	lists:keysearch(a,3,List), +    [{File,1},{FooHrl,1},{BarHrl,1},{FooHrl,5},{File,5}] = +        [ FileLine || {attribute,_,file,FileLine} <- List ],      ok.  %%% Here is a little reimplementation of epp:parse_file, which times out diff --git a/lib/stdlib/test/erl_pp_SUITE.erl b/lib/stdlib/test/erl_pp_SUITE.erl index 9c0a43abcc..ff3470349e 100644 --- a/lib/stdlib/test/erl_pp_SUITE.erl +++ b/lib/stdlib/test/erl_pp_SUITE.erl @@ -49,7 +49,7 @@  	  otp_6321/1, otp_6911/1, otp_6914/1, otp_8150/1, otp_8238/1,  	  otp_8473/1, otp_8522/1, otp_8567/1, otp_8664/1, otp_9147/1, -          otp_10302/1, otp_10820/1]). +          otp_10302/1, otp_10820/1, otp_11100/1]).  %% Internal export.  -export([ehook/6]). @@ -81,7 +81,7 @@ groups() ->       {tickets, [],        [otp_6321, otp_6911, otp_6914, otp_8150, otp_8238,         otp_8473, otp_8522, otp_8567, otp_8664, otp_9147, -       otp_10302, otp_10820]}]. +       otp_10302, otp_10820, otp_11100]}].  init_per_suite(Config) ->      Config. @@ -1103,6 +1103,45 @@ file_attr_is_string("-file(\"" ++ _) -> true;  file_attr_is_string([_ | L]) ->      file_attr_is_string(L). +otp_11100(doc) -> +    "OTP-11100. Fix printing of invalid forms."; +otp_11100(suite) -> []; +otp_11100(Config) when is_list(Config) -> +    %% There are a few places where the added code ("options(none)") +    %% doesn't make a difference (pp:bit_elem_type/1 is an example). + +    %% Cannot trigger the use of the hook function with export/import. +    "-export([{fy,a}/b]).\n" = +        pf({attribute,1,export,[{{fy,a},b}]}), +    "-type foo() :: integer(INVALID-FORM:{foo,bar}:).\n" = +        pf({attribute,1,type,{foo,{type,1,integer,[{foo,bar}]},[]}}), +    pf({attribute,1,type, +        {a,{type,1,range,[{integer,1,1},{foo,bar}]},[]}}), +    "-type foo(INVALID-FORM:{foo,bar}:) :: A.\n" = +        pf({attribute,1,type,{foo,{var,1,'A'},[{foo,bar}]}}), +    "-type foo() :: (INVALID-FORM:{foo,bar}: :: []).\n" = +        pf({attribute,1,type, +            {foo,{paren_type,1, +                  [{ann_type,1,[{foo,bar},{type,1,nil,[]}]}]}, +             []}}), +    "-type foo() :: <<_:INVALID-FORM:{foo,bar}:>>.\n" = +        pf({attribute,1,type, +            {foo,{type,1,binary,[{foo,bar},{integer,1,0}]},[]}}), +    "-type foo() :: <<_:10, _:_*INVALID-FORM:{foo,bar}:>>.\n" = +        pf({attribute,1,type, +            {foo,{type,1,binary,[{integer,1,10},{foo,bar}]},[]}}), +    "-type foo() :: #r{INVALID-FORM:{foo,bar}: :: integer()}.\n" = +        pf({attribute,1,type, +            {foo,{type,1,record, +                  [{atom,1,r}, +                   {type,1,field_type, +                    [{foo,bar},{type,1,integer,[]}]}]}, +             []}}), +    ok. + +pf(Form) -> +    lists:flatten(erl_pp:form(Form,none)). +  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  compile(Config, Tests) -> diff --git a/lib/stdlib/test/ets_SUITE.erl b/lib/stdlib/test/ets_SUITE.erl index af5d5a8f21..bd69019892 100644 --- a/lib/stdlib/test/ets_SUITE.erl +++ b/lib/stdlib/test/ets_SUITE.erl @@ -75,6 +75,7 @@  -export([otp_9932/1]).  -export([otp_9423/1]).  -export([otp_10182/1]). +-export([memory_check_summary/1]).  -export([init_per_testcase/2, end_per_testcase/2]).  %% Convenience for manual testing @@ -149,7 +150,9 @@ all() ->       give_away, setopts, bad_table, types,       otp_10182,       otp_9932, -     otp_9423]. +     otp_9423, +      +     memory_check_summary]. % MUST BE LAST  groups() ->       [{new, [], @@ -185,7 +188,8 @@ init_per_suite(Config) ->  end_per_suite(_Config) ->      stop_spawn_logger(), -    catch erts_debug:set_internal_state(available_internal_state, false). +    catch erts_debug:set_internal_state(available_internal_state, false), +    ok.  init_per_group(_GroupName, Config) ->  	Config. @@ -193,6 +197,26 @@ init_per_group(_GroupName, Config) ->  end_per_group(_GroupName, Config) ->  	Config. +%% Test that we did not have "too many" failed verify_etsmem()'s +%% in the test suite. +%% verify_etsmem() may give a low number of false positives +%% as concurrent activities, such as lingering processes +%% from earlier test suites, may do unrelated ets (de)allocations. +memory_check_summary(_Config) -> +    case whereis(ets_test_spawn_logger) of +	undefined -> +	    ?t:fail("No spawn logger exist"); +	_ -> +	    ets_test_spawn_logger ! {self(), get_failed_memchecks}, +	    receive {get_failed_memchecks, FailedMemchecks} -> ok end, +	    io:format("Failed memchecks: ~p\n",[FailedMemchecks]), +	    if FailedMemchecks > 3 -> +		    ct:fail("Too many failed (~p) memchecks", [FailedMemchecks]); +	       true -> +		    ok +	    end +    end. +  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% @@ -5635,7 +5659,8 @@ verify_etsmem({MemInfo,AllTabs}) ->  	    io:format("Actual:   ~p", [MemInfo2]),  	    io:format("Changed tables before: ~p\n",[AllTabs -- AllTabs2]),  	    io:format("Changed tables after: ~p\n", [AllTabs2 -- AllTabs]), -	    ?t:fail() +	    ets_test_spawn_logger ! failed_memcheck, +	    {comment, "Failed memory check"}      end. @@ -5657,10 +5682,10 @@ stop_loopers(Loopers) ->  looper(Fun, State) ->      looper(Fun, Fun(State)). -spawn_logger(Procs) -> +spawn_logger(Procs, FailedMemchecks) ->      receive  	{new_test_proc, Proc} -> -	    spawn_logger([Proc|Procs]); +	    spawn_logger([Proc|Procs], FailedMemchecks);  	{sync_test_procs, Kill, From} ->  	    lists:foreach(fun (Proc) when From == Proc ->  				  ok; @@ -5683,7 +5708,14 @@ spawn_logger(Procs) ->  				  end  			  end, Procs),  	    From ! test_procs_synced, -	    spawn_logger([From]) +	    spawn_logger([From], FailedMemchecks); + +	failed_memcheck -> +	    spawn_logger(Procs, FailedMemchecks+1); + +	{Pid, get_failed_memchecks} -> +	    Pid ! {get_failed_memchecks, FailedMemchecks}, +	    spawn_logger(Procs, FailedMemchecks)      end.  pid_status(Pid) -> @@ -5699,7 +5731,7 @@ start_spawn_logger() ->      case whereis(ets_test_spawn_logger) of  	Pid when is_pid(Pid) -> true;  	_ -> register(ets_test_spawn_logger, -		      spawn_opt(fun () -> spawn_logger([]) end, +		      spawn_opt(fun () -> spawn_logger([], 0) end,  				[{priority, max}]))      end. @@ -5710,8 +5742,7 @@ start_spawn_logger() ->  stop_spawn_logger() ->      Mon = erlang:monitor(process, ets_test_spawn_logger),      (catch exit(whereis(ets_test_spawn_logger), kill)), -    receive {'DOWN', Mon, _, _, _} -> ok end, -    ok. +    receive {'DOWN', Mon, _, _, _} -> ok end.  wait_for_test_procs() ->      wait_for_test_procs(false). @@ -5811,7 +5842,7 @@ spawn_monitor_with_pid(Pid, Fun, N) ->  		  end) of  	Pid ->  	    {Pid, erlang:monitor(process, Pid)}; -	Other -> +	_Other ->  	    spawn_monitor_with_pid(Pid,Fun,N-1)      end. @@ -6117,11 +6148,18 @@ repeat_for_opts(F, OptGenList) ->      repeat_for_opts(F, OptGenList, []).  repeat_for_opts(F, [], Acc) -> -    lists:map(fun(Opts) -> -                    OptList = lists:filter(fun(E) -> E =/= void end, Opts), -                    io:format("Calling with options ~p\n",[OptList]), -		            F(OptList) -	          end, Acc); +    lists:foldl(fun(Opts, RV_Acc) -> +			OptList = lists:filter(fun(E) -> E =/= void end, Opts), +			io:format("Calling with options ~p\n",[OptList]), +			RV = F(OptList), +			case RV_Acc of +			    {comment,_} -> RV_Acc; +			    _ -> case RV of +				     {comment,_} -> RV; +				     _ -> [RV | RV_Acc] +				 end +			end +	          end, [], Acc);  repeat_for_opts(F, [OptList | Tail], []) when is_list(OptList) ->      repeat_for_opts(F, Tail, [[Opt] || Opt <- OptList]);  repeat_for_opts(F, [OptList | Tail], AccList) when is_list(OptList) -> diff --git a/lib/stdlib/test/gen_server_SUITE.erl b/lib/stdlib/test/gen_server_SUITE.erl index 3b6a3f38bc..a360a0809b 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-2012. All Rights Reserved. +%% Copyright Ericsson AB 1996-2013. 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 @@ -1082,13 +1082,23 @@ replace_state(Config) when is_list(Config) ->  %% Test that the time for a huge message queue is not  %% significantly slower than with an empty message queue.  call_with_huge_message_queue(Config) when is_list(Config) -> +    case test_server:is_native(gen) of +	true -> +	    {skip, +	     "gen is native - huge message queue optimization " +	     "is not implemented"}; +	false -> +	    do_call_with_huge_message_queue() +		end. + +do_call_with_huge_message_queue() ->      ?line Pid = spawn_link(fun echo_loop/0), -    ?line {Time,ok} = tc(fun() -> calls(10, Pid) end), +    ?line {Time,ok} = tc(fun() -> calls(10000, Pid) end),      ?line [self() ! {msg,N} || N <- lists:seq(1, 500000)],      erlang:garbage_collect(), -    ?line {NewTime,ok} = tc(fun() -> calls(10, Pid) end), +    ?line {NewTime,ok} = tc(fun() -> calls(10000, Pid) end),      io:format("Time for empty message queue: ~p", [Time]),      io:format("Time for huge message queue: ~p", [NewTime]), diff --git a/lib/stdlib/test/lists_SUITE.erl b/lib/stdlib/test/lists_SUITE.erl index b56f0b39d8..cd7210f8ec 100644 --- a/lib/stdlib/test/lists_SUITE.erl +++ b/lib/stdlib/test/lists_SUITE.erl @@ -2532,8 +2532,8 @@ otp_5939(Config) when is_list(Config) ->      ?line [] = lists:filter(Pred, []),      ?line {'EXIT', _} = (catch lists:partition(func, [])),      ?line {[],[]} = lists:partition(Pred, []), -    ?line {'EXIT', _} = (catch lists:zf(func, [])), -    ?line [] = lists:zf(Fun1, []), +    ?line {'EXIT', _} = (catch lists:filtermap(func, [])), +    ?line [] = lists:filtermap(Fun1, []),      ?line {'EXIT', _} = (catch lists:foreach(func, [])),      ?line ok = lists:foreach(Fun1, []),      ?line {'EXIT', _} = (catch lists:mapfoldl(func, [], [])), diff --git a/lib/stdlib/test/zip_SUITE.erl b/lib/stdlib/test/zip_SUITE.erl index 7233c061ef..a57641ef62 100644 --- a/lib/stdlib/test/zip_SUITE.erl +++ b/lib/stdlib/test/zip_SUITE.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2006-2011. All Rights Reserved. +%% Copyright Ericsson AB 2006-2013. 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 @@ -109,13 +109,32 @@ borderline_test(Size, TempDir) ->      ok.  unzip_list(Archive, Name) -> -    case os:find_executable("unzip") of -        Unzip when is_list(Unzip) -> +    case unix_unzip_exists() of +	true ->              unzip_list1(Archive, Name);          _ ->              ok      end. +%% Used to do os:find_executable() to check if unzip exists, but on +%% some hosts that would give an unzip program which did not take the +%% "-Z" option. +%% Here we check that "unzip -Z" (which should display usage) and +%% check that it exists with status 0. +unix_unzip_exists() -> +    case os:type() of +	{unix,_} -> +	    Port = open_port({spawn,"unzip -Z > /dev/null"}, [exit_status]), +	    receive +		{Port,{exit_status,0}} -> +		    true; +		{Port,{exit_status,_Fail}} -> +		    false +	    end; +	_ -> +	    false +    end. +  unzip_list1(Archive, Name) ->      Expect = Name ++ "\n",      cmd_expect("unzip -Z -1 " ++ Archive, Expect). | 
