diff options
Diffstat (limited to 'lib/compiler/src/v3_kernel.erl')
| -rw-r--r-- | lib/compiler/src/v3_kernel.erl | 59 | 
1 files changed, 32 insertions, 27 deletions
| diff --git a/lib/compiler/src/v3_kernel.erl b/lib/compiler/src/v3_kernel.erl index 3b33a08cf7..f2eaa37617 100644 --- a/lib/compiler/src/v3_kernel.erl +++ b/lib/compiler/src/v3_kernel.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1999-2010. All Rights Reserved. +%% Copyright Ericsson AB 1999-2011. 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 @@ -83,8 +83,7 @@  -import(lists, [map/2,foldl/3,foldr/3,mapfoldl/3,splitwith/2,member/2,  		keymember/3,keyfind/3]).  -import(ordsets, [add_element/2,del_element/2,union/2,union/1,subtract/2]). - --compile({nowarn_deprecated_function, {erlang,hash,2}}). +-import(cerl, [c_tuple/1]).  -include("core_parse.hrl").  -include("v3_kernel.hrl"). @@ -247,7 +246,7 @@ expr(#c_var{anno=A,name={_Name,Arity}}=Fname, Sub, St) ->      %% instead of one for each occurrence as done now.      Vs = [#c_var{name=list_to_atom("V" ++ integer_to_list(V))} ||  	     V <- integers(1, Arity)], -    Fun = #c_fun{anno=A,vars=Vs,body=#c_apply{op=Fname,args=Vs}}, +    Fun = #c_fun{anno=A,vars=Vs,body=#c_apply{anno=A,op=Fname,args=Vs}},      expr(Fun, Sub, St);  expr(#c_var{anno=A,name=V}, Sub, St) ->      {#k_var{anno=A,name=get_vsub(V, Sub)},[],St}; @@ -291,7 +290,7 @@ expr(#c_binary{anno=A,segments=Cv}, Sub, St0) ->  	    Erl = #c_literal{val=erlang},  	    Name = #c_literal{val=error},  	    Args = [#c_literal{val=badarg}], -	    Error = #c_call{module=Erl,name=Name,args=Args}, +	    Error = #c_call{anno=A,module=Erl,name=Name,args=Args},  	    expr(Error, Sub, St0)      end;  expr(#c_fun{anno=A,vars=Cvs,body=Cb}, Sub0, #kern{ff=OldFF,func=Func}=St0) -> @@ -424,10 +423,11 @@ expr(#c_call{anno=A,module=M0,name=F0,args=Cargs}, Sub, St0) ->      end;  expr(#c_primop{anno=A,name=#c_literal{val=match_fail},args=Cargs0}, Sub, St0) ->      Cargs = translate_match_fail(Cargs0, Sub, A, St0), -    %% This special case will disappear.      {Kargs,Ap,St} = atomic_list(Cargs, Sub, St0),      Ar = length(Cargs), -    Call = #k_call{anno=A,op=#k_internal{name=match_fail,arity=Ar},args=Kargs}, +    Call = #k_call{anno=A,op=#k_remote{mod=#k_atom{val=erlang}, +				       name=#k_atom{val=error}, +				       arity=Ar},args=Kargs},      {Call,Ap,St};  expr(#c_primop{anno=A,name=#c_literal{val=N},args=Cargs}, Sub, St0) ->      {Kargs,Ap,St1} = atomic_list(Cargs, Sub, St0), @@ -457,14 +457,14 @@ expr(#ireceive_accept{anno=A}, _Sub, St) -> {#k_receive_accept{anno=A},[],St}.  translate_match_fail(Args, Sub, Anno, St) ->      case Args of  	[#c_tuple{es=[#c_literal{val=function_clause}|As]}] -> -	    translate_match_fail_1(Anno, Args, As, Sub, St); +	    translate_match_fail_1(Anno, As, Sub, St);  	[#c_literal{val=Tuple}] when is_tuple(Tuple) ->  	    %% The inliner may have created a literal out of  	    %% the original #c_tuple{}.  	    case tuple_to_list(Tuple) of  		[function_clause|As0] ->  		    As = [#c_literal{val=E} || E <- As0], -		    translate_match_fail_1(Anno, Args, As, Sub, St); +		    translate_match_fail_1(Anno, As, Sub, St);  		_ ->  		    Args  	    end; @@ -473,7 +473,7 @@ translate_match_fail(Args, Sub, Anno, St) ->  	    Args      end. -translate_match_fail_1(Anno, Args, As, Sub, #kern{ff=FF}) -> +translate_match_fail_1(Anno, As, Sub, #kern{ff=FF}) ->      AnnoFunc = case keyfind(function_name, 1, Anno) of  		   false ->  		       none;			%Force rewrite. @@ -483,10 +483,10 @@ translate_match_fail_1(Anno, Args, As, Sub, #kern{ff=FF}) ->      case {AnnoFunc,FF} of  	{Same,Same} ->  	    %% Still in the correct function. -	    Args; +	    translate_fc(As);  	{{F,_},F} ->  	    %% Still in the correct function. -	    Args; +	    translate_fc(As);  	_ ->  	    %% Wrong function or no function_name annotation.  	    %% @@ -495,9 +495,12 @@ translate_match_fail_1(Anno, Args, As, Sub, #kern{ff=FF}) ->  	    %% the current function). match_fail(function_clause) will  	    %% only work at the top level of the function it was originally  	    %% defined in, so we will need to rewrite it to a case_clause. -	    [#c_tuple{es=[#c_literal{val=case_clause},#c_tuple{es=As}]}] +	    [c_tuple([#c_literal{val=case_clause},c_tuple(As)])]      end. +translate_fc(Args) -> +    [#c_literal{val=function_clause},make_list(Args)]. +  %% call_type(Module, Function, Arity) -> call | bif | apply | error.  %%  Classify the call.  call_type(#c_literal{val=M}, #c_literal{val=F}, Ar) when is_atom(M), is_atom(F) -> @@ -1167,9 +1170,7 @@ select_bin_int_1(_, _, _, _) -> throw(not_possible).  select_assert_match_possible(Sz, Val, Fs) ->      EmptyBindings = erl_eval:new_bindings(), -    MatchFun = fun({integer,_,_}, NewV, Bs) when NewV =:= Val -> -		       {match,Bs} -	       end, +    MatchFun = match_fun(Val),      EvalFun = fun({integer,_,S}, B) -> {value,S,B} end,      Expr = [{bin_element,0,{integer,0,Val},{integer,0,Sz},[{unit,1}|Fs]}],      {value,Bin,EmptyBindings} = eval_bits:expr_grp(Expr, EmptyBindings, EvalFun), @@ -1184,6 +1185,11 @@ select_assert_match_possible(Sz, Val, Fs) ->  	    throw(not_possible)      end. +match_fun(Val) -> +    fun(match, {{integer,_,_},NewV,Bs}) when NewV =:= Val -> +	    {match,Bs} +    end. +  select_utf8(Val0) ->      try  	Bin = <<Val0/utf8>>, @@ -1493,7 +1499,6 @@ iletrec_funs_gen(Fs, FreeVs, St) ->  %% is_exit_expr(Kexpr) -> boolean().  %%  Test whether Kexpr always exits and never returns. -is_exit_expr(#k_call{op=#k_internal{name=match_fail,arity=1}}) -> true;  is_exit_expr(#k_receive_next{}) -> true;  is_exit_expr(_) -> false. @@ -1655,31 +1660,31 @@ uexpr(#k_catch{anno=A,body=B0}, {break,Rs0}, St0) ->      {Ns,St3} = new_vars(1 - length(Rs0), St2),      Rs1 = Rs0 ++ Ns,      {#k_catch{anno=#k{us=Bu,ns=lit_list_vars(Rs1),a=A},body=B1,ret=Rs1},Bu,St3}; -uexpr(#ifun{anno=A,vars=Vs,body=B0}=IFun, {break,Rs}, St0) -> +uexpr(#ifun{anno=A,vars=Vs,body=B0}, {break,Rs}, St0) ->      {B1,Bu,St1} = ubody(B0, return, St0),	%Return out of new function      Ns = lit_list_vars(Vs),      Free = subtract(Bu, Ns),			%Free variables in fun      Fvs = make_vars(Free),      Arity = length(Vs) + length(Free), -    {{Index,Uniq,Fname}, St3} = +    {Fname,St} =  	case lists:keyfind(id, 1, A) of  -	    {id,Id} -> -		{Id, St1}; +	    {id,{_,_,Fname0}} -> +		{Fname0,St1};  	    false -> -		%% No id annotation. Must invent one. -		I = St1#kern.fcount, -		U = erlang:hash(IFun, (1 bsl 27)-1), -		{N, St2} = new_fun_name(St1), -		{{I,U,N}, St2} +		%% No id annotation. Must invent a fun name. +		new_fun_name(St1)  	end,      Fun = #k_fdef{anno=#k{us=[],ns=[],a=A},func=Fname,arity=Arity,  		  vars=Vs ++ Fvs,body=B1}, +    %% Set dummy values for Index and Uniq -- the real values will +    %% be assigned by beam_asm. +    Index = Uniq = 0,      {#k_bif{anno=#k{us=Free,ns=lit_list_vars(Rs),a=A},   	    op=#k_internal{name=make_fun,arity=length(Free)+3},   	    args=[#k_atom{val=Fname},#k_int{val=Arity},   		  #k_int{val=Index},#k_int{val=Uniq}|Fvs],   	    ret=Rs}, -     Free,add_local_function(Fun, St3)}; +     Free,add_local_function(Fun, St)};  uexpr(Lit, {break,Rs}, St) ->      %% Transform literals to puts here.      %%ok = io:fwrite("uexpr ~w:~p~n", [?LINE,Lit]), | 
