1function [QN,UN,RN,TN,CN,XN,totiter] = solver_mam_basic_mmap_inner(sn, options, lambda)
2% [QN,UN,RN,TN,CN,XN,TOTITER] = SOLVER_MAM_BASIC_MMAP_INNER(SN, OPTIONS, LAMBDA)
4% MAM/
MMAP fork-join decomposition kernel parameterised by per-
class arrival
5% rates LAMBDA. Performs the departure-process refinement loop only;
6% population enforcement (closed networks)
is the wrapper
's responsibility.
8% Copyright (c) 2012-2026, Imperial College London
11config = options.config;
12if ~isfield(config, 'fj_sync_q_len
')
13 config.fj_sync_q_len = 2;
15if ~isfield(config, 'etaqa_trunc
')
16 config.etaqa_trunc = 8;
23V = cellsum(sn.visits);
33% Build FJ synchronization map
34fjSyncMap = sn_build_fj_sync_map(sn);
36% Prepare PH service distributions
41 case SchedStrategy.EXT
42 TN(ist,:) = sn.rates(ist,:);
43 TN(ist,isnan(TN(ist,:))) = 0;
44 case {SchedStrategy.FCFS, SchedStrategy.HOL, SchedStrategy.FCFSPRPRIO}
46 PH{ist}{k} = map_scale(PH{ist}{k}, map_mean(PH{ist}{k})/sn.nservers(ist));
47 pie{ist}{k} = map_pie(PH{ist}{k});
48 D0{ist,k} = PH{ist}{k}{1};
49 if any(isnan(D0{ist,k}))
50 D0{ist,k} = -GlobalConstants.Immediate;
52 PH{ist}{k} = map_exponential(GlobalConstants.Immediate);
55 case SchedStrategy.INF
57 pie{ist}{k} = map_pie(PH{ist}{k});
58 D0{ist,k} = PH{ist}{k}{1};
59 if any(isnan(D0{ist,k}))
60 D0{ist,k} = -GlobalConstants.Immediate;
62 PH{ist}{k} = map_exponential(GlobalConstants.Immediate);
67 PH{ist}{k} = map_scale(PH{ist}{k}, map_mean(PH{ist}{k})/sn.nservers(ist));
68 pie{ist}{k} = map_pie(PH{ist}{k});
69 D0{ist,k} = PH{ist}{k}{1};
70 if any(isnan(D0{ist,k}))
71 D0{ist,k} = -GlobalConstants.Immediate;
73 PH{ist}{k} = map_exponential(GlobalConstants.Immediate);
79it_max = options.iter_max;
81 % Initialize departure processes (node-indexed: DEP{ind,r})
85 isForkJoin = (sn.nodetype(ind) == NodeType.Fork || sn.nodetype(ind) == NodeType.Join);
86 if sn.isstation(ind) && ~isForkJoin
87 ist = sn.nodeToStation(ind);
89 if V(ist,r) > 0 && lambda(r) > 0
90 DEP{ind,r} = map_scale(PH{ist}{r}, 1 / (lambda(r) * V(ist,r)));
92 DEP{ind,r} = PH{ist}{r};
98 DEP{ind,r} = map_exponential(1/lambda(r));
100 DEP{ind,r} = map_exponential(1/GlobalConstants.Immediate);
107 % Compute arrival processes with FJ synchronization
108 ARV = solver_mam_traffic_mmap(sn, DEP, config, fjSyncMap);
112 ind = sn.stationToNode(ist);
113 switch sn.nodetype(ind)
116 TN(ist,k) = lambda(k);
122 if ~isempty(ARV{ind}) && iscell(ARV{ind})
123 if length(ARV{ind}{1}) > config.space_max
125 line_printf('\nArrival process at node %d
is now at %d states. Compressing.
', ind, length(ARV{ind}{1}));
127 ARV{ind} = mmap_compress(ARV{ind}, config);
130 finiteCapUsed = false;
132 case {SchedStrategy.FCFS, SchedStrategy.HOL, SchedStrategy.FCFSPRPRIO}
133 isFiniteCap = isfinite(sn.cap(ist));
136 [isMmck, muMmck] = mam_detect_mmck(sn, ist, K, ARV{ind});
138 aggrLambda_ist = sum(mmap_lambda(ARV{ind}), 'omitnan
');
139 exactRes = qsys_mmck(aggrLambda_ist, muMmck, sn.nservers(ist), capK);
140 meanQ_fc = exactRes.meanQueueLength;
141 lossProb_fc = exactRes.lossProbability;
143 [meanQ_fc, lossProb_fc, ~] = mam_truncate_renorm( ...
144 {ARV{ind}{[1,3:end]}}, {pie{ist}{:}}, {D0{ist,:}}, capK);
146 lambdaInflow = mmap_lambda(ARV{ind});
147 lambdaInflow(isnan(lambdaInflow)) = 0;
148 TN_eff = lambdaInflow * (1 - lossProb_fc);
150 S_actual = zeros(1, K);
152 S_actual(k) = map_mean(PH{ist}{k}) * sn.nservers(ist);
155 Savg_eff = sum(TN_eff .* S_actual, 'omitnan
') / sumTN;
156 Wq = max(0, meanQ_fc / sumTN - Savg_eff);
161 TN(ist,k) = TN_eff(k);
162 UN(ist,k) = TN(ist,k) * map_mean(PH{ist}{k});
164 RN(ist,k) = Wq + S_actual(k);
165 QN(ist,k) = TN(ist,k) * RN(ist,k);
171 finiteCapUsed = true;
173 rho_ist_classes = mmap_lambda(ARV{ind}) .* arrayfun(@(k) map_mean(PH{ist}{k}), 1:K);
174 rho_ist_classes(isnan(rho_ist_classes)) = 0;
175 rho_ist = sum(rho_ist_classes);
176 if rho_ist < 1 - GlobalConstants.FineTol
177 [Qret{1:K}, ~] = MMAPPH1FCFS({ARV{ind}{[1,3:end]}}, {pie{ist}{:}}, {D0{ist,:}}, 'ncMoms
', 1, 'ncDistr
', 2);
179 QN(ist,k) = sum(Qret{k});
182 % Saturation: bound queue lengths so the
183 % wrapper's bisection can recognise overload
184 % without NaNs propagating from MMAPPH1FCFS.
186 if isfinite(sn.njobs(k))
187 QN(ist,k) = sn.njobs(k);
189 QN(ist,k) = 1/GlobalConstants.FineTol;
193 TN(ist,:) = mmap_lambda(ARV{ind});
195 case SchedStrategy.PS
196 TN(ist,:) = mmap_lambda(ARV{ind});
198 UN(ist,k) = TN(ist,k) * S(ist,k);
200 Uden = min([1-GlobalConstants.FineTol, sum(UN(ist,:))]);
202 QN(ist,k) = UN(ist,k)/(1-Uden);
208 UN(ist,k) = TN(ist,k) * map_mean(PH{ist}{k});
209 QN(ist,k) = QN(ist,k) + TN(ist,k)*(map_mean(PH{ist}{k})*sn.nservers(ist)) * (sn.nservers(ist)-1)/sn.nservers(ist);
210 RN(ist,k) = QN(ist,k) ./ TN(ist,k);
216 case SchedStrategy.INF
217 if ~isempty(ARV{ind}) && iscell(ARV{ind})
218 TN(ist,:) = mmap_lambda(ARV{ind});
222 UN(ist,k) = S(ist,k)*TN(ist,k);
223 QN(ist,k) = TN(ist,k)*S(ist,k);
224 RN(ist,k) = S(ist,k);
227 case SchedStrategy.EXT
228 % Source: TN already set above
233 if it >= 3 && max(abs(QN(:)-QN_1(:))./(QN_1(:)+GlobalConstants.FineTol)) < options.iter_tol
237 % Update departure processes
239 ind = sn.stationToNode(ist);
240 switch sn.nodetype(ind)
242 if ~isempty(ARV{ind}) && iscell(ARV{ind})
244 case {SchedStrategy.FCFS, SchedStrategy.HOL, SchedStrategy.FCFSPRPRIO}
246 A = mmap_hide(ARV{ind}, setdiff(1:K,r));
250 etaqa_n = config.etaqa_trunc;
251 etaqa_sz = (etaqa_n+1)*na*ns;
252 rho = sum(UN(ist,:));
253 if etaqa_sz <= config.space_max && rho < 1-GlobalConstants.FineTol
255 DEP{ind,r} = qbd_depproc_etaqa(A, Srv, etaqa_n);
256 DEP{ind,r} = map_normalize(DEP{ind,r});
263 if V(ist,r) > 0 && lambda(r) > 0
264 DEP{ind,r} = map_scale(DEP{ind,r}, 1 / (lambda(r) * V(ist,r)));
267 case SchedStrategy.PS
269 A = mmap_hide(ARV{ind}, setdiff(1:K,r));
273 etaqa_n = config.etaqa_trunc;
274 etaqa_sz = (etaqa_n+1)*na*ns;
275 rho = sum(UN(ist,:));
276 if V(ist,r) > 0 && lambda(r) > 0
277 if etaqa_sz <= config.space_max && rho < 1-GlobalConstants.FineTol
279 DEP{ind,r} = qbd_depproc_etaqa_ps(A, Srv, etaqa_n);
280 DEP{ind,r} = map_normalize(DEP{ind,r});
287 DEP{ind,r} = map_scale(DEP{ind,r}, 1 / (lambda(r) * V(ist,r)));
295 DEP{ind,r} = map_exponential(1/TN(ist,r));
304 line_printf(
'\nMAM FJ parametric decomposition completed in %d iterations.', it);
307% Join: derive QN/RN from parallel branch means
308for joinIdx = find(sn.nodetype == NodeType.Join)
'
309 joinStat = sn.nodeToStation(joinIdx);
313 syncGroups = unique(fjSyncMap.nodeSync(joinIdx, :));
314 syncGroups = syncGroups(syncGroups > 0);
316 if TN(joinStat, r) <= 0
322 branchNodes = find(fjSyncMap.nodeSync(joinIdx, :) == gid);
323 branchRt = zeros(1, numel(branchNodes));
324 branchTput = zeros(1, numel(branchNodes));
326 for b = 1:numel(branchNodes)
327 branchStat = sn.nodeToStation(branchNodes(b));
328 if isnan(branchStat) || RN(branchStat, r) <= 0
332 branchRt(used) = RN(branchStat, r);
333 branchTput(used) = TN(branchStat, r);
335 branchRt = branchRt(1:used);
336 branchTput = branchTput(1:used);
337 if numel(branchRt) < 2
340 lambdai = 1 ./ branchRt;
342 for pow = 0:(numel(branchRt) - 1)
343 maxBranchRt = maxBranchRt + (-1)^pow * sum(1 ./ sum(nchoosek(lambdai, pow + 1), 2));
345 syncDelay = syncDelay + max(maxBranchRt - mean(branchRt), 0);
346 joinArrivalRate = joinArrivalRate + sum(branchTput);
348 RN(joinStat, r) = syncDelay;
349 QN(joinStat, r) = joinArrivalRate * syncDelay;