diff options
Diffstat (limited to 'lib/hipe')
| -rw-r--r-- | lib/hipe/cerl/cerl_prettypr.erl | 17 | ||||
| -rw-r--r-- | lib/hipe/cerl/erl_bif_types.erl | 20 | ||||
| -rw-r--r-- | lib/hipe/cerl/erl_types.erl | 99 | ||||
| -rw-r--r-- | lib/hipe/doc/src/notes.xml | 73 | ||||
| -rw-r--r-- | lib/hipe/llvm/elf_format.hrl | 2 | ||||
| -rw-r--r-- | lib/hipe/ppc/hipe_rtl_to_ppc.erl | 14 | ||||
| -rw-r--r-- | lib/hipe/rtl/hipe_rtl.erl | 18 | ||||
| -rw-r--r-- | lib/hipe/rtl/hipe_rtl_binary_match.erl | 18 | ||||
| -rw-r--r-- | lib/hipe/sparc/hipe_rtl_to_sparc.erl | 10 | ||||
| -rw-r--r-- | lib/hipe/vsn.mk | 2 | ||||
| -rw-r--r-- | lib/hipe/x86/hipe_rtl_to_x86.erl | 15 | 
11 files changed, 209 insertions, 79 deletions
| diff --git a/lib/hipe/cerl/cerl_prettypr.erl b/lib/hipe/cerl/cerl_prettypr.erl index 9a3873f46d..f4a67439d6 100644 --- a/lib/hipe/cerl/cerl_prettypr.erl +++ b/lib/hipe/cerl/cerl_prettypr.erl @@ -1,7 +1,7 @@  %% =====================================================================  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 2004-2009. All Rights Reserved. +%% Copyright Ericsson AB 2004-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 @@ -476,13 +476,20 @@ lay_literal(Node, Ctxt) ->  	    %% that could represent printable characters - we  	    %% always print an integer.  	    text(int_lit(Node)); -	V when is_binary(V) -> -	    lay_binary(c_binary([c_bitstr(abstract(B), -					  abstract(8), +        V when is_bitstring(V) -> +            Val = fun(I) when is_integer(I) -> I; +                     (B) when is_bitstring(B) -> +                          BZ = bit_size(B), <<BV:BZ>> = B, BV +                  end, +            Sz = fun(I) when is_integer(I) -> 8; +                    (B) when is_bitstring(B) -> bit_size(B) +                 end, +	    lay_binary(c_binary([c_bitstr(abstract(Val(B)), +					  abstract(Sz(B)),  					  abstract(1),  					  abstract(integer),  					  abstract([unsigned, big])) -				 || B <- binary_to_list(V)]), +				 || B <- bitstring_to_list(V)]),  		       Ctxt);  	[] ->  	    text("[]"); diff --git a/lib/hipe/cerl/erl_bif_types.erl b/lib/hipe/cerl/erl_bif_types.erl index a460f16272..74e93bf098 100644 --- a/lib/hipe/cerl/erl_bif_types.erl +++ b/lib/hipe/cerl/erl_bif_types.erl @@ -2,7 +2,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2003-2013. All Rights Reserved. +%% Copyright Ericsson AB 2003-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 @@ -1891,7 +1891,11 @@ infinity_add(neg_inf, _Number) -> neg_inf;  infinity_add(_Number, pos_inf) -> pos_inf;  infinity_add(_Number, neg_inf) -> neg_inf;  infinity_add(Number1, Number2) when is_integer(Number1), is_integer(Number2) -> -  Number1 + Number2. +  try Number1 + Number2 +  catch +    error:system_limit when Number1 < 0 -> neg_inf; +    error:system_limit -> pos_inf +  end.  infinity_mult(neg_inf, Number) ->     Greater = infinity_geq(Number, 0),  @@ -1902,7 +1906,13 @@ infinity_mult(pos_inf, Number) -> infinity_inv(infinity_mult(neg_inf, Number));  infinity_mult(Number, pos_inf) -> infinity_inv(infinity_mult(neg_inf, Number));  infinity_mult(Number, neg_inf) -> infinity_mult(neg_inf, Number);  infinity_mult(Number1, Number2) when is_integer(Number1), is_integer(Number2)-> -  Number1 * Number2. +  try Number1 * Number2 +  catch +    error:system_limit -> +      if (Number1 >= 0) =:= (Number2 >= 0) -> pos_inf; +         true -> neg_inf +      end +  end.  width({Min, Max}) -> infinity_max([width(Min), width(Max)]);  width(pos_inf) -> pos_inf; @@ -2633,7 +2643,9 @@ opaque_args(M, F, A, Xs, Opaques) ->            true ->              case t_tuple_subtypes(X, Opaques) of                unknown -> false; -              List when length(List) >= 1 -> opaque_recargs(List, Y, Opaques) +              List when length(List) >= 1 -> +                (t_is_atom(Y, Opaques) andalso +                 opaque_recargs(List, Y, Opaques))              end;            false -> t_has_opaque_subtype(X, Opaques)          end]; diff --git a/lib/hipe/cerl/erl_types.erl b/lib/hipe/cerl/erl_types.erl index 67661130a5..4215448c61 100644 --- a/lib/hipe/cerl/erl_types.erl +++ b/lib/hipe/cerl/erl_types.erl @@ -2,7 +2,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2003-2014. All Rights Reserved. +%% Copyright Ericsson AB 2003-2015. 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 @@ -72,7 +72,7 @@           t_contains_opaque/1, t_contains_opaque/2,           t_decorate_with_opaque/3,  	 t_elements/1, -	 t_find_opaque_mismatch/2, +	 t_find_opaque_mismatch/3,           t_find_unknown_opaque/3,  	 t_fixnum/0,  	 t_map/2, @@ -262,6 +262,8 @@  -define(TAG_IMMED1_SIZE, 4).  -define(BITS, (erlang:system_info(wordsize) * 8) - ?TAG_IMMED1_SIZE). +-define(MAX_TUPLE_SIZE, (1 bsl 10)). +  %%-----------------------------------------------------------------------------  %% Type tags and qualifiers  %% @@ -530,39 +532,51 @@ list_contains_opaque(List, Opaques) ->  %% The first argument of the function is the pattern and its second  %% argument the type we are matching against the pattern. --spec t_find_opaque_mismatch(erl_type(), erl_type()) -> 'error' | {'ok', erl_type(), erl_type()}. +-spec t_find_opaque_mismatch(erl_type(), erl_type(), [erl_type()]) -> +                                'error' | {'ok', erl_type(), erl_type()}. -t_find_opaque_mismatch(T1, T2) -> -  t_find_opaque_mismatch(T1, T2, T2). +t_find_opaque_mismatch(T1, T2, Opaques) -> +  t_find_opaque_mismatch(T1, T2, T2, Opaques). -t_find_opaque_mismatch(?any, _Type, _TopType) -> error; -t_find_opaque_mismatch(?none, _Type, _TopType) -> error; -t_find_opaque_mismatch(?list(T1, Tl1, _), ?list(T2, Tl2, _), TopType) -> -  t_find_opaque_mismatch_ordlists([T1, Tl1], [T2, Tl2], TopType); -t_find_opaque_mismatch(_T1, ?opaque(_) = T2, TopType) -> {ok, TopType, T2}; -t_find_opaque_mismatch(?opaque(_) = T1, _T2, TopType) -> +t_find_opaque_mismatch(?any, _Type, _TopType, _Opaques) -> error; +t_find_opaque_mismatch(?none, _Type, _TopType, _Opaques) -> error; +t_find_opaque_mismatch(?list(T1, Tl1, _), ?list(T2, Tl2, _), TopType, Opaques) -> +  t_find_opaque_mismatch_ordlists([T1, Tl1], [T2, Tl2], TopType, Opaques); +t_find_opaque_mismatch(T1, ?opaque(_) = T2, TopType, Opaques) -> +  case is_opaque_type(T2, Opaques) of +    false -> {ok, TopType, T2}; +    true -> +      t_find_opaque_mismatch(T1, t_opaque_structure(T2), TopType, Opaques) +  end; +t_find_opaque_mismatch(?opaque(_) = T1, T2, TopType, Opaques) ->    %% The generated message is somewhat misleading: -  {ok, TopType, T1}; -t_find_opaque_mismatch(?product(T1), ?product(T2), TopType) -> -  t_find_opaque_mismatch_ordlists(T1, T2, TopType); -t_find_opaque_mismatch(?tuple(T1, Arity, _), ?tuple(T2, Arity, _), TopType) -> -  t_find_opaque_mismatch_ordlists(T1, T2, TopType); -t_find_opaque_mismatch(?tuple(_, _, _) = T1, ?tuple_set(_) = T2, TopType) -> +  case is_opaque_type(T1, Opaques) of +    false -> {ok, TopType, T1}; +    true -> +      t_find_opaque_mismatch(t_opaque_structure(T1), T2, TopType, Opaques) +  end; +t_find_opaque_mismatch(?product(T1), ?product(T2), TopType, Opaques) -> +  t_find_opaque_mismatch_ordlists(T1, T2, TopType, Opaques); +t_find_opaque_mismatch(?tuple(T1, Arity, _), ?tuple(T2, Arity, _), +                       TopType, Opaques) -> +  t_find_opaque_mismatch_ordlists(T1, T2, TopType, Opaques); +t_find_opaque_mismatch(?tuple(_, _, _) = T1, ?tuple_set(_) = T2, +                       TopType, Opaques) ->    Tuples1 = t_tuple_subtypes(T1),    Tuples2 = t_tuple_subtypes(T2), -  t_find_opaque_mismatch_lists(Tuples1, Tuples2, TopType); -t_find_opaque_mismatch(T1, ?union(U2), TopType) -> -  t_find_opaque_mismatch_lists([T1], U2, TopType); -t_find_opaque_mismatch(_T1, _T2, _TopType) -> error. +  t_find_opaque_mismatch_lists(Tuples1, Tuples2, TopType, Opaques); +t_find_opaque_mismatch(T1, ?union(U2), TopType, Opaques) -> +  t_find_opaque_mismatch_lists([T1], U2, TopType, Opaques); +t_find_opaque_mismatch(_T1, _T2, _TopType, _Opaques) -> error. -t_find_opaque_mismatch_ordlists(L1, L2, TopType) -> +t_find_opaque_mismatch_ordlists(L1, L2, TopType, Opaques) ->    List = lists:zipwith(fun(T1, T2) -> -			   t_find_opaque_mismatch(T1, T2, TopType) +			   t_find_opaque_mismatch(T1, T2, TopType, Opaques)  		       end, L1, L2),    t_find_opaque_mismatch_list(List). -t_find_opaque_mismatch_lists(L1, L2, _TopType) -> -  List = [t_find_opaque_mismatch(T1, T2, T2) || T1 <- L1, T2 <- L2], +t_find_opaque_mismatch_lists(L1, L2, _TopType, Opaques) -> +  List = [t_find_opaque_mismatch(T1, T2, T2, Opaques) || T1 <- L1, T2 <- L2],    t_find_opaque_mismatch_list(List).  t_find_opaque_mismatch_list([]) -> error; @@ -1423,7 +1437,6 @@ t_number_vals(Type) ->  t_number_vals(Type, Opaques) ->    do_opaque(Type, Opaques, fun number_vals/1). -number_vals(?int_set(?any)) -> unknown;  number_vals(?int_set(Set)) -> set_to_list(Set);  number_vals(?number(_, _)) -> unknown;  number_vals(?opaque(_)) -> unknown; @@ -1759,6 +1772,8 @@ t_tuple() ->  -spec t_tuple(non_neg_integer() | [erl_type()]) -> erl_type(). +t_tuple(N) when is_integer(N), N > ?MAX_TUPLE_SIZE  -> +  t_tuple();  t_tuple(N) when is_integer(N) ->    ?tuple(lists:duplicate(N, ?any), N, ?any);  t_tuple(List) -> @@ -4215,7 +4230,7 @@ t_from_form({type, _L, any, []}, _TypeNames, _RecDict, _VarDict) ->  t_from_form({type, _L, arity, []}, _TypeNames, _RecDict, _VarDict) ->    {t_arity(), []};  t_from_form({type, _L, array, []}, TypeNames, RecDict, VarDict) -> -  builtin_type(array, t_array(), TypeNames, RecDict, VarDict); +  builtin_type(array, t_array(), [], TypeNames, RecDict, VarDict);  t_from_form({type, _L, atom, []}, _TypeNames, _RecDict, _VarDict) ->    {t_atom(), []};  t_from_form({type, _L, binary, []}, _TypeNames, _RecDict, _VarDict) -> @@ -4238,9 +4253,9 @@ t_from_form({type, _L, byte, []}, _TypeNames, _RecDict, _VarDict) ->  t_from_form({type, _L, char, []}, _TypeNames, _RecDict, _VarDict) ->    {t_char(), []};  t_from_form({type, _L, dict, []}, TypeNames, RecDict, VarDict) -> -  builtin_type(dict, t_dict(), TypeNames, RecDict, VarDict); +  builtin_type(dict, t_dict(), [], TypeNames, RecDict, VarDict);  t_from_form({type, _L, digraph, []}, TypeNames, RecDict, VarDict) -> -  builtin_type(digraph, t_digraph(), TypeNames, RecDict, VarDict); +  builtin_type(digraph, t_digraph(), [], TypeNames, RecDict, VarDict);  t_from_form({type, _L, float, []}, _TypeNames, _RecDict, _VarDict) ->    {t_float(), []};  t_from_form({type, _L, function, []}, _TypeNames, _RecDict, _VarDict) -> @@ -4257,9 +4272,9 @@ t_from_form({type, _L, 'fun', [{type, _, product, Domain}, Range]},    {T, R2} = t_from_form(Range, TypeNames, RecDict, VarDict),    {t_fun(L, T), R1 ++ R2};  t_from_form({type, _L, gb_set, []}, TypeNames, RecDict, VarDict) -> -  builtin_type(gb_set, t_gb_set(), TypeNames, RecDict, VarDict); +  builtin_type(gb_set, t_gb_set(), [], TypeNames, RecDict, VarDict);  t_from_form({type, _L, gb_tree, []}, TypeNames, RecDict, VarDict) -> -  builtin_type(gb_tree, t_gb_tree(), TypeNames, RecDict, VarDict); +  builtin_type(gb_tree, t_gb_tree(), [], TypeNames, RecDict, VarDict);  t_from_form({type, _L, identifier, []}, _TypeNames, _RecDict, _VarDict) ->    {t_identifier(), []};  t_from_form({type, _L, integer, []}, _TypeNames, _RecDict, _VarDict) -> @@ -4273,8 +4288,12 @@ t_from_form({type, _L, list, []}, _TypeNames, _RecDict, _VarDict) ->  t_from_form({type, _L, list, [Type]}, TypeNames, RecDict,  VarDict) ->    {T, R} = t_from_form(Type, TypeNames, RecDict, VarDict),    {t_list(T), R}; -t_from_form({type, _L, map, _}, TypeNames, RecDict, VarDict) -> -  builtin_type(map, t_map([]), TypeNames, RecDict, VarDict); +t_from_form({type, _L, map, As0}, TypeNames, RecDict, VarDict) -> +  As = case is_list(As0) of +         true -> As0; +         false -> [] +       end, +  builtin_type(map, t_map([]), As, TypeNames, RecDict, VarDict);  t_from_form({type, _L, mfa, []}, _TypeNames, _RecDict, _VarDict) ->    {t_mfa(), []};  t_from_form({type, _L, module, []}, _TypeNames, _RecDict, _VarDict) -> @@ -4333,7 +4352,7 @@ t_from_form({type, _L, product, Elements}, TypeNames, RecDict, VarDict) ->    {L, R} = list_from_form(Elements, TypeNames, RecDict, VarDict),    {t_product(L), R};  t_from_form({type, _L, queue, []}, TypeNames, RecDict, VarDict) -> -  builtin_type(queue, t_queue(), TypeNames, RecDict, VarDict); +  builtin_type(queue, t_queue(), [], TypeNames, RecDict, VarDict);  t_from_form({type, _L, range, [From, To]} = Type,  	    _TypeNames, _RecDict, _VarDict) ->    case {erl_eval:partial_eval(From), erl_eval:partial_eval(To)} of @@ -4346,13 +4365,13 @@ t_from_form({type, _L, record, [Name|Fields]}, TypeNames, RecDict, VarDict) ->  t_from_form({type, _L, reference, []}, _TypeNames, _RecDict, _VarDict) ->    {t_reference(), []};  t_from_form({type, _L, set, []}, TypeNames, RecDict, VarDict) -> -  builtin_type(set, t_set(), TypeNames, RecDict, VarDict); +  builtin_type(set, t_set(), [], TypeNames, RecDict, VarDict);  t_from_form({type, _L, string, []}, _TypeNames, _RecDict, _VarDict) ->    {t_string(), []};  t_from_form({type, _L, term, []}, _TypeNames, _RecDict, _VarDict) ->    {t_any(), []};  t_from_form({type, _L, tid, []}, TypeNames, RecDict, VarDict) -> -  builtin_type(tid, t_tid(), TypeNames, RecDict, VarDict); +  builtin_type(tid, t_tid(), [], TypeNames, RecDict, VarDict);  t_from_form({type, _L, timeout, []}, _TypeNames, _RecDict, _VarDict) ->    {t_timeout(), []};  t_from_form({type, _L, tuple, any}, _TypeNames, _RecDict, _VarDict) -> @@ -4369,10 +4388,10 @@ t_from_form({opaque, _L, Name, {Mod, Args, Rep}}, _TypeNames,              _RecDict, _VarDict) ->    {t_opaque(Mod, Name, Args, Rep), []}. -builtin_type(Name, Type, TypeNames, RecDict, VarDict) -> -  case lookup_type(Name, 0, RecDict) of +builtin_type(Name, Type, Args, TypeNames, RecDict, VarDict) -> +  case lookup_type(Name, length(Args), RecDict) of      {_, {_M, _T, _A}} -> -      type_from_form(Name, [], TypeNames, RecDict, VarDict); +      type_from_form(Name, Args, TypeNames, RecDict, VarDict);      error ->        {Type, []}    end. @@ -4573,7 +4592,7 @@ t_form_to_string({type, _L, iodata, []}) -> "iodata()";  t_form_to_string({type, _L, iolist, []}) -> "iolist()";  t_form_to_string({type, _L, list, [Type]}) ->     "[" ++ t_form_to_string(Type) ++ "]"; -t_form_to_string({type, _L, map, _}) -> +t_form_to_string({type, _L, map, Args}) when not is_list(Args) ->    "#{}";  t_form_to_string({type, _L, mfa, []}) -> "mfa()";  t_form_to_string({type, _L, module, []}) -> "module()"; diff --git a/lib/hipe/doc/src/notes.xml b/lib/hipe/doc/src/notes.xml index c7faf733c7..2d6fd245f7 100644 --- a/lib/hipe/doc/src/notes.xml +++ b/lib/hipe/doc/src/notes.xml @@ -30,6 +30,73 @@    </header>    <p>This document describes the changes made to HiPE.</p> +<section><title>Hipe 3.11.2</title> + +    <section><title>Fixed Bugs and Malfunctions</title> +      <list> +        <item> +          <p> +	    Fixed internal elf_format hrl file to contain valid +	    erlang</p> +          <p> +	    Own Id: OTP-12322</p> +        </item> +      </list> +    </section> + +</section> + +<section><title>Hipe 3.11.1</title> + +    <section><title>Fixed Bugs and Malfunctions</title> +      <list> +        <item> +	    <p> The pretty-printing of bitstrings has been corrected. +	    </p> +          <p> +	    Own Id: OTP-12015</p> +        </item> +        <item> +	    <p> A bug concerning <c>is_record/2,3</c> has been fixed, +	    as well as some cases where Dialyzer could crash due to +	    reaching system limits. </p> +          <p> +	    Own Id: OTP-12018</p> +        </item> +      </list> +    </section> + +</section> + +<section><title>Hipe 3.11</title> + +    <section><title>Fixed Bugs and Malfunctions</title> +      <list> +        <item> +          <p> +	    A Dialyzer crash involving analysis of Map types has now +	    been fixed.</p> +          <p> +	    Own Id: OTP-11947</p> +        </item> +      </list> +    </section> + + +    <section><title>Improvements and New Features</title> +      <list> +        <item> +          <p> +	    Handle Maps instructions get_map_elements, put_map_assoc, +	    put_map_exact in HiPE compiler.</p> +          <p> +	    Own Id: OTP-11900</p> +        </item> +      </list> +    </section> + +</section> +  <section><title>Hipe 3.10.3</title>      <section><title>Fixed Bugs and Malfunctions</title> @@ -118,9 +185,9 @@  	    "hi" := V1, a := V2, b := V3} = M2. % match keys with  	    values</c></item> </taglist></p>            <p> -	    For information on how to use Maps please see the -	    <seealso marker="doc/reference_manual:maps">Reference -	    Manual</seealso>.</p> +	    For information on how to use Maps please see Map Expressions in the +		<seealso marker="doc/reference_manual:expressions#map_expressions"> +			Reference Manual</seealso>.</p>            <p>  	    The current implementation is without the following  	    features: <taglist> <item>No variable keys</item> diff --git a/lib/hipe/llvm/elf_format.hrl b/lib/hipe/llvm/elf_format.hrl index 78592e6e2a..7a3cdfead6 100644 --- a/lib/hipe/llvm/elf_format.hrl +++ b/lib/hipe/llvm/elf_format.hrl @@ -68,7 +68,7 @@  -define(E_IDENT,     {?E_IDENT_OFFSET, ?E_IDENT_SIZE}).  -define(E_TYPE,      {?E_TYPE_OFFSET, ?E_TYPE_SIZE}).  -define(E_MACHINE,   {?E_MACHINE_OFFSET, ?E_MACHINE_SIZE}). --define(E_VERSION,   {?E_VERSION_OFFSET, ?E_VERSION_SIZE}) +-define(E_VERSION,   {?E_VERSION_OFFSET, ?E_VERSION_SIZE}).  -define(E_ENTRY,     {?E_ENTRY_OFFSET, ?E_ENTRY_SIZE}).  -define(E_PHOFF,     {?E_PHOFF_OFFSET, ?E_PHOFF_SIZE}).  -define(E_SHOFF,     {?E_SHOFF_OFFSET, ?E_SHOFF_SIZE}). diff --git a/lib/hipe/ppc/hipe_rtl_to_ppc.erl b/lib/hipe/ppc/hipe_rtl_to_ppc.erl index 7dfa56df29..a55fc137c3 100644 --- a/lib/hipe/ppc/hipe_rtl_to_ppc.erl +++ b/lib/hipe/ppc/hipe_rtl_to_ppc.erl @@ -102,10 +102,18 @@ conv_insn(I, Map, Data) ->    end.  conv_fconv(I, Map, Data) -> -  %% Dst := (double)Src, where Dst is FP reg and Src is int reg +  %% Dst := (double)Src, where Dst is FP reg and Src is GP reg or imm    {Dst, Map0} = conv_fpreg(hipe_rtl:fconv_dst(I), Map), -  {Src, Map1} = conv_src(hipe_rtl:fconv_src(I), Map0), % exclude imm src -  I2 = mk_fconv(Dst, Src), +  {Src, Map1} = conv_src(hipe_rtl:fconv_src(I), Map0), +  I2 = +    case hipe_ppc:is_temp(Src) of +      true -> +	mk_fconv(Dst, Src); +      false -> +	Tmp = new_untagged_temp(), +	mk_li(Tmp, Src, +	      mk_fconv(Dst, Tmp)) +    end,    {I2, Map1, Data}.  mk_fconv(Dst, Src) -> diff --git a/lib/hipe/rtl/hipe_rtl.erl b/lib/hipe/rtl/hipe_rtl.erl index bc61bec0bd..2f62dd79ad 100644 --- a/lib/hipe/rtl/hipe_rtl.erl +++ b/lib/hipe/rtl/hipe_rtl.erl @@ -413,11 +413,11 @@ rtl_info_update(Rtl, Info) -> Rtl#rtl{info=Info}.  %% move  %% -mk_move(Dst, Src) -> #move{dst=Dst, src=Src}. +mk_move(Dst, Src) -> false = is_fpreg(Dst), false = is_fpreg(Src), #move{dst=Dst, src=Src}.  move_dst(#move{dst=Dst}) -> Dst. -move_dst_update(M, NewDst) -> M#move{dst=NewDst}. +move_dst_update(M, NewDst) -> false = is_fpreg(NewDst), M#move{dst=NewDst}.  move_src(#move{src=Src}) -> Src. -move_src_update(M, NewSrc) -> M#move{src=NewSrc}. +move_src_update(M, NewSrc) -> false = is_fpreg(NewSrc), M#move{src=NewSrc}.  %% is_move(#move{}) -> true;  %% is_move(_) -> false. @@ -469,7 +469,11 @@ phi_remove_pred(Phi, Pred) ->    case NewArgList of      [Arg] -> %% the phi should be turned into a move instruction        {_Label,Var} = Arg, -      mk_move(phi_dst(Phi), Var); +      Dst = phi_dst(Phi), +      case {is_fpreg(Dst), is_fpreg(Var)} of +	{true, true} -> mk_fmove(Dst, Var); +	{false, false} -> mk_move(Dst, Var) +      end;    %%    io:format("~nPhi (~w) turned into move (~w) when removing pred ~w~n",[Phi,Move,Pred]),      [_|_] ->        Phi#phi{arglist=NewArgList} @@ -836,11 +840,11 @@ fp_unop_op(#fp_unop{op=Op}) -> Op.  %% fmove  %% -mk_fmove(X, Y) -> #fmove{dst=X, src=Y}. +mk_fmove(X, Y) -> true = is_fpreg(X), true = is_fpreg(Y), #fmove{dst=X, src=Y}.  fmove_dst(#fmove{dst=Dst}) -> Dst. -fmove_dst_update(M, NewDst) -> M#fmove{dst=NewDst}. +fmove_dst_update(M, NewDst) -> true = is_fpreg(NewDst), M#fmove{dst=NewDst}.  fmove_src(#fmove{src=Src}) -> Src. -fmove_src_update(M, NewSrc) -> M#fmove{src=NewSrc}. +fmove_src_update(M, NewSrc) -> true = is_fpreg(NewSrc), M#fmove{src=NewSrc}.  %%  %% fconv diff --git a/lib/hipe/rtl/hipe_rtl_binary_match.erl b/lib/hipe/rtl/hipe_rtl_binary_match.erl index 8831199244..af8903904b 100644 --- a/lib/hipe/rtl/hipe_rtl_binary_match.erl +++ b/lib/hipe/rtl/hipe_rtl_binary_match.erl @@ -990,19 +990,19 @@ unsigned_bignum(Dst1, Src, TrueLblName) ->     hipe_tagscheme:unsafe_mk_big(Dst1, Src, unsigned),     hipe_rtl:mk_goto(TrueLblName)]. -load_bytes(Dst, Base, Offset, {Signedness, _Endianess},1) -> +load_bytes(Dst, Base, Offset, {Signedness, _Endianness},1) ->    [hipe_rtl:mk_load(Dst, Base, Offset, byte, Signedness),     hipe_rtl:mk_alu(Offset, Offset, add, hipe_rtl:mk_imm(1))]; -load_bytes(Dst, Base, Offset, {Signedness, Endianess},2) -> -  case Endianess of +load_bytes(Dst, Base, Offset, {Signedness, Endianness},2) -> +  case Endianness of      big ->        hipe_rtl_arch:load_big_2(Dst, Base, Offset, Signedness);      little ->        hipe_rtl_arch:load_little_2(Dst, Base, Offset, Signedness)    end; -load_bytes(Dst, Base, Offset, {Signedness, Endianess},3) -> +load_bytes(Dst, Base, Offset, {Signedness, Endianness},3) ->    Tmp1 = hipe_rtl:mk_new_reg(), -  case Endianess of +  case Endianness of      big ->        [hipe_rtl:mk_load(Dst, Base, Offset, byte, Signedness),         hipe_rtl:mk_alu(Offset, Offset, add, hipe_rtl:mk_imm(1)), @@ -1026,18 +1026,18 @@ load_bytes(Dst, Base, Offset, {Signedness, Endianess},3) ->         hipe_rtl:mk_alu(Dst, Dst, 'or', Tmp1),         hipe_rtl:mk_alu(Offset, Offset, add, hipe_rtl:mk_imm(1))]    end;  -load_bytes(Dst, Base, Offset, {Signedness, Endianess}, 4) -> -  case Endianess of +load_bytes(Dst, Base, Offset, {Signedness, Endianness}, 4) -> +  case Endianness of      big ->        hipe_rtl_arch:load_big_4(Dst, Base, Offset, Signedness);      little ->        hipe_rtl_arch:load_little_4(Dst, Base, Offset, Signedness)    end; -load_bytes(Dst, Base, Offset, {Signedness, Endianess}, X) when X > 1 -> +load_bytes(Dst, Base, Offset, {Signedness, Endianness}, X) when X > 1 ->    [LoopLbl, EndLbl] = create_lbls(2),    [Tmp1, Limit, TmpOffset] = create_regs(3), -  case Endianess of +  case Endianness of      big ->        [hipe_rtl:mk_alu(Limit, Offset, add, hipe_rtl:mk_imm(X)),         hipe_rtl:mk_load(Dst, Base, Offset, byte, Signedness), diff --git a/lib/hipe/sparc/hipe_rtl_to_sparc.erl b/lib/hipe/sparc/hipe_rtl_to_sparc.erl index dc001f865e..fd21be3ae7 100644 --- a/lib/hipe/sparc/hipe_rtl_to_sparc.erl +++ b/lib/hipe/sparc/hipe_rtl_to_sparc.erl @@ -85,17 +85,17 @@ conv_insn(I, Map, Data) ->    end.  conv_fconv(I, Map, Data) -> -  %% Dst := (double)Src, where Dst is FP reg and Src is int reg -  {Src, Map1} = conv_src(hipe_rtl:fconv_src(I), Map), % exclude imm src +  %% Dst := (double)Src, where Dst is FP reg and Src is GP reg or imm +  {Src, Map1} = conv_src(hipe_rtl:fconv_src(I), Map),    {Dst, Map2} = conv_fpreg(hipe_rtl:fconv_dst(I), Map1),    I2 = mk_fconv(Src, Dst),    {I2, Map2, Data}.  mk_fconv(Src, Dst) ->    CSP = hipe_sparc:mk_temp(14, 'untagged'), % o6 -  Disp = hipe_sparc:mk_simm13(100), -  [hipe_sparc:mk_store('stw', Src, CSP, Disp), -   hipe_sparc:mk_pseudo_fload(CSP, Disp, Dst, true), +  Offset = 100, +  mk_store('stw', Src, CSP, Offset) ++ +  [hipe_sparc:mk_pseudo_fload(CSP, hipe_sparc:mk_simm13(Offset), Dst, true),     hipe_sparc:mk_fp_unary('fitod', Dst, Dst)].  conv_fmove(I, Map, Data) -> diff --git a/lib/hipe/vsn.mk b/lib/hipe/vsn.mk index fb7e4b91a0..4cf09830cb 100644 --- a/lib/hipe/vsn.mk +++ b/lib/hipe/vsn.mk @@ -1 +1 @@ -HIPE_VSN = 3.10.3 +HIPE_VSN = 3.11.2 diff --git a/lib/hipe/x86/hipe_rtl_to_x86.erl b/lib/hipe/x86/hipe_rtl_to_x86.erl index d77e4fed3b..36da2f4d44 100644 --- a/lib/hipe/x86/hipe_rtl_to_x86.erl +++ b/lib/hipe/x86/hipe_rtl_to_x86.erl @@ -236,7 +236,7 @@ conv_insn(I, Map, Data) ->      #fconv{} ->        {Dst, Map0} = conv_dst(hipe_rtl:fconv_dst(I), Map),        {[], Src, Map1} = conv_src(hipe_rtl:fconv_src(I), Map0), -      I2 = [hipe_x86:mk_fmove(Src, Dst)], +      I2 = conv_fconv(Dst, Src),        {I2, Map1, Data};      X ->        %% gctest?? @@ -712,6 +712,19 @@ vmap_lookup(Map, Key) ->  vmap_bind(Map, Key, Val) ->    gb_trees:insert(Key, Val, Map). +%%% Finalise the conversion of an Integer-to-Float operation. + +conv_fconv(Dst, Src) -> +  case hipe_x86:is_imm(Src) of +    false -> +      [hipe_x86:mk_fmove(Src, Dst)]; +    true -> +      %% cvtsi2sd does not allow src to be an immediate +      Tmp = new_untagged_temp(), +      [hipe_x86:mk_move(Src, Tmp), +       hipe_x86:mk_fmove(Tmp, Dst)] +  end. +  %%% Finalise the conversion of a 2-address FP operation.  conv_fp_unary(Dst, Src, FpUnOp) -> | 
