1function [QN,UN,RN,TN,CN,XN,totiter,method,runtime] = solver_mam(sn, options)
2%[Q,U,R,T,C,X,totiter] = SOLVER_MAM(QN, PH, OPTIONS)
4%Copyright (c) 2012-2026, Imperial College London
7method = options.method;
8config = options.config;
16V = cellsum(sn.visits);
27 inchain = sn.inchain{c};
28 lambdas_inchain = sn.rates(sn.refstat(inchain(1)),inchain);
29 lambdas_inchain = lambdas_inchain(isfinite(lambdas_inchain));
30 lambda(inchain) = sum(lambdas_inchain);
35 chain(k) = find(sn.chains(:,k));
40 case SchedStrategy.EXT
42 case {SchedStrategy.FCFS, SchedStrategy.HOL, SchedStrategy.FCFSPRPRIO}
46 line_warning(mfilename,'The dec.mmap method does not support non-FCFS queues.\n
');
48 [QN,UN,RN,TN,CN,XN] = deal([],[],[],[],[],[]);
51 runtime = toc(Tstart);
56if all(isinf(sn.njobs)) % is open
57 % open queueing system (one node is the external world)
62 case SchedStrategy.EXT
63 TN(ist,:) = sn.rates(ist,:);
64 TN(ist,isnan(TN(ist,:)))=0;
65 case {SchedStrategy.FCFS, SchedStrategy.HOL, SchedStrategy.FCFSPRPRIO}
67 % divide service time by number of servers and put
68 % later a surrogate delay server in tandem to compensate
69 PH{ist}{k} = map_scale(PH{ist}{k}, map_mean(PH{ist}{k})/sn.nservers(ist));
70 pie{ist}{k} = map_pie(PH{ist}{k});
71 D0{ist,k} = PH{ist}{k}{1};
72 if any(isnan(D0{ist,k}))
73 D0{ist,k} = -GlobalConstants.Immediate;
75 PH{ist}{k} = map_exponential(GlobalConstants.Immediate);
81 it_max = options.iter_max;
84 % now estimate arrival processes
86 % initially form departure processes using scaled service
90 ist = sn.nodeToStation(ind);
91 DEP{ind,r} = map_scale(PH{ist}{r}, 1 / (lambda(r) * V(ind,r)) );
96 ARV = solver_mam_traffic(sn, DEP, config);
100 ind = sn.stationToNode(ist);
101 switch sn.nodetype(ind)
103 if length(ARV{ind}{1}) > config.space_max
104 line_printf('\nArrival process at node %d
is now at %d states. Compressing.
',ind,length(ARV{ind}{1}));
105 ARV{ind} = mmap_compress(ARV{ind});
107 [Qret{1:K}, ~] = MMAPPH1FCFS({ARV{ind}{[1,3:end]}}, {pie{ist}{:}}, {D0{ist,:}}, 'ncMoms
', 1, 'ncDistr
',2);
109 QN(ist,k) = sum(Qret{k});
111 TN(ist,:) = mmap_lambda(ARV{ind});
114 UN(ist,k) = TN(ist,k) * map_mean(PH{ist}{k});
115 %add number of jobs at the surrogate delay server
116 QN(ist,k) = QN(ist,k) + TN(ist,k)*(map_mean(PH{ist}{k})*sn.nservers(ist)) * (sn.nservers(ist)-1)/sn.nservers(ist);
117 RN(ist,k) = QN(ist,k) ./ TN(ist,k);
121 if it >=3 && max(abs(QN(:)-QN_1(:))./QN_1(:)) < options.iter_tol
126 ind = sn.stationToNode(ist);
127 switch sn.nodetype(ind)
129 [Ret{1:2*K}] = MMAPPH1FCFS({ARV{ind}{[1,3:end]}}, {pie{ist}{:}}, {D0{ist,:}}, 'stDistrPH
');
131 %obtain response time distribution for class r
132 %alpha = Ret{(r-1)*2+1}; T0 = Ret{(r-1)*2+2};
133 %RD = {T0,(-T0)*ones(length(alpha),1)*alpha(:)'}; %PH to MAP
135 %pieRD = map_pie(
RD);
137 %define a ph
for the arrival process of
class r
138 A = mmap_hide(ARV{ind},setdiff(1:K,r));
142 %define a ph
for the service process of
class r
147 %with probability rho the output
is the
148 %inter-arrival time+service time, with probability 1-rho
is the
150 rho = sum(UN(ist,:));
151 AQ = sum(QN(ist,:)); % queue seen upon arrival with PASTA
152 Afull = AQ/rho; % from AQ = (1-rho)*0 + rho*AQfull
153 pfullonarrival = 1 - (Afull)/(1+Afull);
155 A = mmap_scale(A,map_mean(A)-map_mean(S));
161 %
this is only
for single-
class
162 DEP0ir = [S{1}, sum(rho)*tS*pieA;
165 DEP1ir = [(1-sum(rho))*S{2}, zSA;
168 DEP{ind,r} = map_normalize({DEP0ir,DEP1ir});
170 DEP{ind,r} = map_scale(DEP{ind,r}, 1 / (lambda(r) * V(ind,r)) );
171 SCVd(ind,r) = map_scv(DEP{ind,r});
172 IDCd(ind,r) = map_idc(DEP{ind,r});
179 line_printf(
'\nMAM parametric decomposition completed in %d iterations.',it);
183 line_warning(mfilename,
'This model is not supported by SolverMAM yet. Returning with no result.\n');
186runtime = toc(Tstart);