%% -*- erlang-indent-level: 2 -*-
%%
%% Licensed under the Apache License, Version 2.0 (the "License");
%% you may not use this file except in compliance with the License.
%% You may obtain a copy of the License at
%%
%% http://www.apache.org/licenses/LICENSE-2.0
%%
%% Unless required by applicable law or agreed to in writing, software
%% distributed under the License is distributed on an "AS IS" BASIS,
%% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
%% See the License for the specific language governing permissions and
%% limitations under the License.
%%
%%=============================================================================
-module(hipe_ig_moves).
-export([new/1,
new_move/3,
get_moves/1]).
%%-----------------------------------------------------------------------------
%% The main data structure; its fields are:
%% - movelist : mapping from temp to set of associated move numbers
%% - nrmoves : number of distinct move instructions seen so far
%% - moveinsns : list of move instructions, in descending move number order
%% - moveset : set of move instructions
-record(ig_moves, {movelist :: movelist(),
nrmoves = 0 :: non_neg_integer(),
moveinsns = [] :: [{_,_}],
moveset = gb_sets:empty() :: gb_sets:set()}).
-type movelist() :: hipe_vectors:vector(ordsets:ordset(non_neg_integer())).
%%-----------------------------------------------------------------------------
-spec new(non_neg_integer()) -> #ig_moves{}.
new(NrTemps) ->
MoveList = hipe_vectors:new(NrTemps, ordsets:new()),
#ig_moves{movelist = MoveList}.
-spec new_move(_, _, #ig_moves{}) -> #ig_moves{}.
new_move(Dst, Src, IG_moves) ->
MoveSet = IG_moves#ig_moves.moveset,
MoveInsn = {Dst, Src},
case gb_sets:is_member(MoveInsn, MoveSet) of
true ->
IG_moves;
false ->
MoveNr = IG_moves#ig_moves.nrmoves,
Movelist0 = IG_moves#ig_moves.movelist,
Movelist1 = add_movelist(MoveNr, Dst,
add_movelist(MoveNr, Src, Movelist0)),
IG_moves#ig_moves{nrmoves = MoveNr+1,
movelist = Movelist1,
moveinsns = [MoveInsn|IG_moves#ig_moves.moveinsns],
moveset = gb_sets:insert(MoveInsn, MoveSet)}
end.
-spec add_movelist(non_neg_integer(), non_neg_integer(), movelist())
-> movelist().
add_movelist(MoveNr, Temp, MoveList) ->
AssocMoves = hipe_vectors:get(MoveList, Temp),
%% XXX: MoveNr does not occur in moveList[Temp], but the new list must be an
%% ordset due to the ordsets:union in hipe_coalescing_regalloc:combine().
hipe_vectors:set(MoveList, Temp, ordsets:add_element(MoveNr, AssocMoves)).
-spec get_moves(#ig_moves{}) -> {movelist(), non_neg_integer(), tuple()}.
get_moves(IG_moves) -> % -> {MoveList, NrMoves, MoveInsns}
{IG_moves#ig_moves.movelist,
IG_moves#ig_moves.nrmoves,
list_to_tuple(lists:reverse(IG_moves#ig_moves.moveinsns))}.