LINE Solver
MATLAB API documentation
Loading...
Searching...
No Matches
refreshProcessRepresentations.m
1function [ph, phases] = refreshProcessRepresentations(self)
2% [PH, PHASES] = REFRESHPROCESSREPRESENTATIONS()
3
4% Copyright (c) 2012-2026, Imperial College London
5% All rights reserved.
6
7
8M = getNumberOfStations(self);
9K = getNumberOfClasses(self);
10ph = cell(M,1);
11for ist=1:M
12 ph{ist,1} = cell(1,K);
13end
14phases = zeros(M,K);
15stations = self.stations;
16for ist=1:M
17 if ist == self.getIndexSourceStation
18 ph_i = stations{ist}.getSourceRates();
19 else
20 switch class(stations{ist})
21 case 'Fork'
22 mu_i = cell(1,K);
23 phi_i = cell(1,K);
24 for r=1:K
25 mu_i{r} = NaN;
26 phi_i{r} = NaN;
27 end
28 ph_i = Coxian(mu_i,phi_i).getProcess;
29 case 'Join'
30 mu_i = cell(1,K);
31 phi_i = cell(1,K);
32 for r=1:K
33 mu_i{r} = NaN;
34 phi_i{r} = NaN;
35 ph_i{r} = Coxian(mu_i{r},phi_i{r}).getProcess;
36 end
37 otherwise
38 ph_i = stations{ist}.getServiceRates();
39 end
40 end
41 for r=1:K
42 ph{ist}{r} = ph_i{r};
43 % CyclicPoisson stores {rates, durations}, not D0/D1: always 1 active phase
44 isCyclicPoisson_ir = false;
45 if isa(stations{ist}, 'Source') && any(ist == self.getIndexSourceStation) ...
46 && length(stations{ist}.input.sourceClasses) >= r ...
47 && ~isempty(stations{ist}.input.sourceClasses{r})
48 isCyclicPoisson_ir = isa(stations{ist}.input.sourceClasses{r}{end}, 'CyclicPoisson');
49 elseif isa(stations{ist}, 'ServiceStation') ...
50 && length(stations{ist}.server.serviceProcess) >= r ...
51 && ~isempty(stations{ist}.server.serviceProcess{r})
52 isCyclicPoisson_ir = isa(stations{ist}.server.serviceProcess{r}{end}, 'CyclicPoisson');
53 end
54 if isempty(ph{ist}{r}) % fluid fails otherwise
55 phases(ist,r) = 1;
56 elseif isCyclicPoisson_ir
57 phases(ist,r) = 1;
58 elseif ~isMAP(ph{ist}{r})
59 % Non-Markovian distribution: convert to MAP representation
60 ph{ist}{r} = convertToMAP(stations{ist}, r, ph{ist}{r});
61 phases(ist,r) = length(ph{ist}{r}{1});
62 elseif any(isnan(ph{ist}{r}{1}(:))) || any(isnan(ph{ist}{r}{2}(:))) % disabled
63 phases(ist,r) = 0;
64 else
65 phases(ist,r) = length(ph{ist}{r}{1});
66 end
67 end
68end
69if ~isempty(self.sn) %&& isprop(self.sn,'mu')
70 proc = ph;
71 pie = cell(size(ph));
72 for ist=1:M
73 for r=1:K
74 map_ir = ph{ist}{r};
75 % CyclicPoisson stores {rates, durations}, not D0/D1 — skip map_pie
76 isCyclicPoisson_ir = false;
77 if isa(stations{ist}, 'Source') && any(ist == self.getIndexSourceStation) ...
78 && length(stations{ist}.input.sourceClasses) >= r ...
79 && ~isempty(stations{ist}.input.sourceClasses{r})
80 isCyclicPoisson_ir = isa(stations{ist}.input.sourceClasses{r}{end}, 'CyclicPoisson');
81 elseif isa(stations{ist}, 'ServiceStation') ...
82 && length(stations{ist}.server.serviceProcess) >= r ...
83 && ~isempty(stations{ist}.server.serviceProcess{r})
84 isCyclicPoisson_ir = isa(stations{ist}.server.serviceProcess{r}{end}, 'CyclicPoisson');
85 end
86 if ~isempty(map_ir)
87 proc{ist}{r} = map_ir;
88 if isCyclicPoisson_ir
89 pie{ist}{r} = NaN;
90 else
91 pie{ist}{r} = map_pie(map_ir);
92 end
93 else
94 pie{ist}{r} = NaN;
95 end
96 end
97 end
98 self.sn.proc = proc;
99 self.sn.pie = pie;
100 self.sn.phases = phases;
101 self.sn.phasessz = max(self.sn.phases,ones(size(self.sn.phases)));
102 self.sn.phasessz(self.sn.nodeToStation(self.sn.nodetype == NodeType.Join),:)=phases(self.sn.nodeToStation(self.sn.nodetype == NodeType.Join),:);
103 self.sn.phaseshift = [zeros(size(phases,1),1),cumsum(self.sn.phasessz,2)];
104end
105end
106
107function result = isMAP(proc)
108% ISMAP Check if a process representation is a valid MAP {D0, D1}
109%
110% A valid MAP has exactly 2 cell elements, both of which are square matrices
111% of the same size.
112result = false;
113if ~iscell(proc) || length(proc) ~= 2
114 return;
115end
116D0 = proc{1};
117D1 = proc{2};
118if ~isnumeric(D0) || ~isnumeric(D1)
119 return;
120end
121if ~ismatrix(D0) || ~ismatrix(D1)
122 return;
123end
124[m0, n0] = size(D0);
125[m1, n1] = size(D1);
126if m0 ~= n0 || m1 ~= n1 || m0 ~= m1
127 return;
128end
129result = true;
130end
131
132function MAP = convertToMAP(station, classIdx, proc)
133% CONVERTTOMAP Convert non-Markovian distribution parameters to MAP
134%
135% For non-Markovian distributions, the process representation contains
136% distribution parameters rather than {D0, D1} matrices. This function
137% converts them to an Erlang approximation.
138
139% Get the distribution object from the station
140if isa(station, 'Source')
141 dist = station.input.sourceClasses{classIdx}{end};
142else
143 dist = station.server.serviceProcess{classIdx}{end};
144end
145
146% Get mean for Erlang approximation
147targetMean = dist.getMean();
148
149% Determine number of phases based on SCV
150% For Det (SCV=0), use high number of phases; for others, match SCV
151scv = dist.getSCV();
152if scv < GlobalConstants.CoarseTol
153 % Deterministic or near-deterministic: use 20 phases
154 nPhases = 20;
155else
156 % Match SCV: for Erlang, SCV = 1/n, so n = 1/SCV
157 nPhases = max(1, ceil(1/scv));
158 nPhases = min(nPhases, 100); % Cap at 100 phases
159end
160
161% Create Erlang MAP approximation
162MAP = map_erlang(targetMean, nPhases);
163end