forked from robince/partial-info-decomp
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcalc_pi.m
143 lines (123 loc) · 3.79 KB
/
calc_pi.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
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
function lat = calc_pi(lat,Pjoint,Icap,forcenn,normlevels)
% Calculate PI on a redundancy lattice using Williams and Beer summation
% inputs:
% lat - lattice structure
% Pjoint - full probability distribuion (axis per variable, target last
% axis)
% forcenn - threshold negative values on the lattice (default false)
% normlevels - normalize values across levels of the lattice (default
% false)
%
% if only lat provided calculate PI using existing Icap
% otherwise, recalculate Icap
% if only lat provided calculate PI using existing Icap
% otherwise, recalculate Icap
if nargin>1
s = size(Pjoint);
if lat.Nx ~= (length(s)-1)
error('Pjoint does not match lattice structure')
end
% calc Icap for each node
for ni=1:lat.Nnodes
lat.Icap(ni) = Icap(lat.A{ni}, Pjoint);
end
end
if lat.Nx>3
error('calc_pi: too many variables')
end
if nargin<4
forcenn = false;
end
if nargin<5
normlevels = false;
end
% use equation (7) from Williams and Beer to calculate
% PI at each node
lat.PI = NaN(size(lat.Icap));
% raw PI before non-disjoint normalisation
lat.PIraw = NaN(size(lat.Icap));
% ascend through levels of the lattice
Nlevels = max(lat.level);
for li=1:(Nlevels-1)
nodes = find(lat.level==li);
for ni=nodes
lat = calc_pi_node(lat,ni,forcenn,normlevels);
end
end
% don't enforce non-negativitity for top node
lat = calc_pi_node(lat,lat.top,false,normlevels);
function lat = calc_pi_node(lat,ni,nonneg,normlevels)
if nargin<3
nonneg = true;
end
children = lat.children{ni};
if isempty(children)
% no children
thsPI = lat.Icap(ni);
if nonneg
thsPI = max(thsPI,0);
end
lat.PI(ni) = thsPI;
lat.PIraw(ni) = thsPI;
return
end
all_children = recurse_children(lat,ni,[]);
if normlevels
PIchildren = normalise_levels(lat, all_children);
else
PIchildren = lat.PI(all_children);
end
thsPI = lat.Icap(ni) - sum(PIchildren);
if nonneg
thsPI = max(thsPI,0);
end
lat.PI(ni) = thsPI;
lat.PIraw(ni) = thsPI;
if ni==lat.top
lat.PI(all_children) = PIchildren;
end
function normPI = normalise_levels(lat,children)
% normalise to correct for non-additivity of non-disjoint nodes
% values for this set of children
PIraw = lat.PIraw(children);
levels = lat.level(children);
labels = lat.labels(children);
A = lat.A(children);
normPI = PIraw;
for li=1:lat.Nlevels
nodes = find(levels==li);
levelPI = PIraw(nodes);
posPInodes = nodes(abs(levelPI)>1e-15);
posPIvars = A(posPInodes);
posPIvars = cell2mat([posPIvars{:}]);
if length(posPIvars) ~= length(unique(posPIvars))
% have non-disjoint positive PI contributions at this level
% using structure of 3rd order lattice (might need more logic to
% determine pairwise disjoint-ness for higher order lattices)
if li==4
% special case level 4 for 3 variable lattice
% one node contains all variables
fullnode = find(strcmpi(labels,'{12}{13}{23}'));
if isempty(fullnode) || PIraw(fullnode)<1e-15
% all sources at this level are disjoint so no
% normalisation required
continue
elseif length(posPInodes)==1
% only {12}{13}{23} is non-zero so no normalization
% required
continue
end
% only normalise by 2 here even if more posPInodes, because
% there are only 2 disjoint copies at this level
normPI(posPInodes) = PIraw(posPInodes) ./ 2;
else
normPI(posPInodes) = PIraw(posPInodes) ./ length(posPInodes);
end
end
end
function children = recurse_children(lat,ni,children)
children = [children lat.children{ni}];
for ci=lat.children{ni}
children = recurse_children(lat,ci,children);
end
children = unique(children);