-
Notifications
You must be signed in to change notification settings - Fork 0
/
nodeelim_nesdis_camd.m
82 lines (70 loc) · 2.8 KB
/
nodeelim_nesdis_camd.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
% NODEWISE_NESDIS_CAMD
% Custom reduction scheme developed as a part of this master's thesis.
% Approach is inspired by nested dissection fill-in reduction algorithm,
% but takes into consideration
%
% Steps
% 1. Bisect graph to retrieve good (small) separation trees. Mark
% separators as nodes to avoid reduction.
% 2. Reduce each leaf by computing Schur complement for leaf and all of its
% parents (all up to root). Keep separators and external nodes.
%
% See also NODEWISE_ABSTRACT, NESDIS.
function reduced_data = nodeelim_nesdis_camd( G, is_ext_node, options )
if ~isfield(options, 'nesdis_opts')
options.nesdis_opts = [];
end
[~, cp, cmember] = nesdis(G, 'sym', options.nesdis_opts);
% get tree leaves
[~, y] = treelayout(cp);
leaves = find(y == min(y));
% reduce each leaf
for n=1:length(leaves)
if options.verbose
fprintf('Reduce leaf; %i/%i\n', n, length(leaves))
end
leaf_id = leaves(n);
leaf_nodes = (cmember == leaf_id);
% get all parents of binary tree leaf
parents = zeros(1, length(y));
last_parent = leaf_id;
k = 0;
while last_parent ~= 0
k = k+1;
last_parent = cp(last_parent);
parents(k) = last_parent;
end
parent_nodes = (ismember(cmember, parents));
leaf_nodes_ints = find(leaf_nodes);
parent_nodes_ints = find(parent_nodes);
% subscript G
leaf_all_nodes = [leaf_nodes_ints parent_nodes_ints];
leafG = G(leaf_all_nodes, leaf_all_nodes);
% apply adjustments for separator connectivity outside of leaf
not_current_nodes = ~(leaf_nodes | parent_nodes);
separator_connections_compensation = sum(G(leaf_all_nodes, :), 2) - sum(G(leaf_all_nodes, not_current_nodes), 2);
leafG(1:length(leafG)+1:end) = leafG(1:length(leafG)+1:end) - separator_connections_compensation';
% compute CAMD nodewise reduction
% move rows, cols
is_ext_camd = is_ext_node | parent_nodes';
is_ext_camd = is_ext_camd(leaf_all_nodes);
o = nodewise_camd(leafG, is_ext_camd, options);
Gred = o.G;
Gred(1:length(Gred)+1:end) = Gred(1:length(Gred)+1:end) + separator_connections_compensation(o.new_nodes)';
to_replace = leaf_all_nodes(o.new_nodes);
G(leaf_all_nodes, leaf_all_nodes) = 0;
G(to_replace, to_replace) = Gred;
% TODO: get rid of these empty rows and cols?
% Live updates of sparse structure are fairly costly
% Maybe partial results may be held in different data structure
non_empty_nodes = any(G,2);
new_nodes = 1:length(G);
new_nodes = new_nodes(non_empty_nodes);
reduced_data.G = G(non_empty_nodes, non_empty_nodes);
reduced_data.new_nodes = new_nodes;
reduced_data.is_ext_node = is_ext_node(non_empty_nodes);
reduced_data.eliminated_nodes = [];
reduced_data.early_exit = 0;
reduced_data.fillin_tracker = [];
end
end