Files
mercury/deep_profiler/callgraph.m
Julien Fischer 2c686ea193 Convert the rest of the deep profiler to four-space indentation.
Estimated hours taken: 1.5
Branches: main

Convert the rest of the deep profiler to four-space indentation.
There are no changes to any algorithms (other than introducing state
variables in a few spots).

Remove an old bug workaround.

deep_profiler/*.m:
	Convert to four-space indentation where that has not already
	been done.

	Fix some bad line breaks.

	Use state variables in a few places.

	Reposition comments according to our current coding standard.

deep_profiler/Mercury.options:
	Remove some old bug workarounds.
2005-11-08 08:12:40 +00:00

185 lines
6.2 KiB
Mathematica

%-----------------------------------------------------------------------------%
% vim: ft=mercury ts=4 sw=4 et
%-----------------------------------------------------------------------------%
% Copyright (C) 2001-2002, 2004-2005 The University of Melbourne.
% This file may only be copied under the terms of the GNU General
% Public License - see the file COPYING in the Mercury distribution.
%-----------------------------------------------------------------------------%
%
% Authors: conway, zs.
%
% This module contains the code for finding the cliques in the call graph
% described by an initial_deep structure. They are returned as a list of
% cliques, in bottom-up order.
:- module callgraph.
:- interface.
:- import_module profile.
:- import_module array.
:- import_module list.
%-----------------------------------------------------------------------------%
:- pred find_cliques(initial_deep::in, list(list(proc_dynamic_ptr))::out)
is det.
:- pred make_clique_indexes(int::in, list(list(proc_dynamic_ptr))::in,
array(list(proc_dynamic_ptr))::array_uo, array(clique_ptr)::array_uo)
is det.
%-----------------------------------------------------------------------------%
%-----------------------------------------------------------------------------%
:- implementation.
:- import_module array_util.
:- import_module cliques.
:- import_module profile.
:- import_module int.
:- import_module set.
:- import_module svarray.
% :- import_module io.
% :- import_module require.
% :- import_module string.
% :- import_module unsafe.
%-----------------------------------------------------------------------------%
find_cliques(InitDeep, BottomUpPDPtrCliqueList) :-
make_graph(InitDeep, Graph),
topological_sort(Graph, TopDownPDICliqueList),
%
% Turn each of the sets of PDIs into a list of PDPtrs. We use foldl here
% because the list may be very long and map runs out of stack space, and
% we want the final list in reverse order anyway because the propagation
% algorithm works bottom up.
%
list__foldl(accumulate_pdptr_lists, TopDownPDICliqueList,
[], BottomUpPDPtrCliqueList).
:- pred accumulate_pdptr_lists(set(int)::in, list(list(proc_dynamic_ptr))::in,
list(list(proc_dynamic_ptr))::out) is det.
accumulate_pdptr_lists(PDISet, PDPtrLists0, PDPtrLists) :-
pdi_set_to_pdptr_list(PDISet, PDPtrList),
PDPtrLists = [PDPtrList | PDPtrLists0].
:- pred pdi_set_to_pdptr_list(set(int)::in, list(proc_dynamic_ptr)::out)
is det.
pdi_set_to_pdptr_list(PDISet, PDPtrList) :-
set__to_sorted_list(PDISet, PDIList),
list__map(pdi_to_pdptr, PDIList, PDPtrList).
:- pred pdi_to_pdptr(int::in, proc_dynamic_ptr::out) is det.
pdi_to_pdptr(PDI, proc_dynamic_ptr(PDI)).
%-----------------------------------------------------------------------------%
:- pred make_graph(initial_deep::in, graph::out) is det.
make_graph(InitDeep, Graph) :-
init(Graph0),
array_foldl_from_1(add_pd_arcs(InitDeep), InitDeep ^ init_proc_dynamics,
Graph0, Graph).
:- pred add_pd_arcs(initial_deep::in, int::in, proc_dynamic::in,
graph::in, graph::out) is det.
add_pd_arcs(InitDeep, PDI, PD, !Graph) :-
CallSiteRefArray = PD ^ pd_sites,
array__to_list(CallSiteRefArray, CallSiteRefList),
list__foldl(add_call_site_arcs(InitDeep, PDI), CallSiteRefList,
!Graph).
:- pred add_call_site_arcs(initial_deep::in, int::in, call_site_array_slot::in,
graph::in, graph::out) is det.
add_call_site_arcs(InitDeep, FromPDI, CallSiteSlot, !Graph) :-
(
CallSiteSlot = normal(CSDPtr),
add_csd_arcs(InitDeep, FromPDI, CSDPtr, !Graph)
;
CallSiteSlot = multi(_, CSDPtrArray),
array__to_list(CSDPtrArray, CSDPtrs),
list__foldl(add_csd_arcs(InitDeep, FromPDI), CSDPtrs, !Graph)
).
:- pred add_csd_arcs(initial_deep::in, int::in, call_site_dynamic_ptr::in,
graph::in, graph::out) is det.
% :- pragma promise_pure(add_csd_arcs/5).
add_csd_arcs(InitDeep, FromPDI, CSDPtr, !Graph) :-
CSDPtr = call_site_dynamic_ptr(CSDI),
( CSDI > 0 ->
array__lookup(InitDeep ^ init_call_site_dynamics, CSDI, CSD),
ToPDPtr = CSD ^ csd_callee,
ToPDPtr = proc_dynamic_ptr(ToPDI),
% impure unsafe_perform_io(write_arc(FromPDI, ToPDI, CSDI)),
add_arc(!.Graph, FromPDI, ToPDI, !:Graph)
;
true
).
%-----------------------------------------------------------------------------%
make_clique_indexes(NPDs, CliqueList, Cliques, CliqueIndex) :-
Cliques = array(CliqueList),
array__init(NPDs, clique_ptr(-1), CliqueIndex0),
%
% For each clique, add entries to the CliqueIndex array, which maps every
% proc_dynamic_ptr back to the clique to which it belongs.
%
array_foldl_from_1(index_clique, Cliques, CliqueIndex0, CliqueIndex).
:- pred index_clique(int::in, list(proc_dynamic_ptr)::in,
array(clique_ptr)::array_di, array(clique_ptr)::array_uo) is det.
index_clique(CliqueNum, CliqueMembers, !CliqueIndex) :-
array_list_foldl(index_clique_member(CliqueNum), CliqueMembers,
!CliqueIndex).
:- pred index_clique_member(int::in, proc_dynamic_ptr::in,
array(clique_ptr)::array_di, array(clique_ptr)::array_uo) is det.
% :- pragma promise_pure(index_clique_member/4).
index_clique_member(CliqueNum, PDPtr, !CliqueIndex) :-
PDPtr = proc_dynamic_ptr(PDI),
% impure unsafe_perform_io(write_pdi_cn(PDI, CliqueNum)),
svarray.set(PDI, clique_ptr(CliqueNum), !CliqueIndex).
%-----------------------------------------------------------------------------%
%
% Predicates for use in debugging.
%
% :- pred write_arc(int::in, int::in, int::in, io::di, io::uo)
% is det.
%
% write_arc(FromPDI, ToPDI, CSDI, !IO) :-
% io__format("arc from pd %d to pd %d through csd %d\n",
% [i(FromPDI), i(ToPDI), i(CSDI)], !IO).
%
% :- pred write_pdi_cn(int::in, int::in, io::di, io::uo) is det.
%
% write_pdi_cn(PDI, CN, !IO) :-
% io__write_string("pdi ", !IO),
% io__write_int(PDI, !IO),
% io__write_string(" -> clique ", !IO),
% io__write_int(CN, !IO),
% io__nl(!IO),
% io__flush_output(!IO).
%
%-----------------------------------------------------------------------------%
:- end_module callgraph.
%-----------------------------------------------------------------------------%