LINE Solver
MATLAB API documentation
Loading...
Searching...
No Matches
solver_mam.m
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)
3
4%Copyright (c) 2012-2026, Imperial College London
5%All rights reserved.
6
7method = options.method;
8config = options.config;
9totiter = NaN;
10PH = sn.proc;
11I = sn.nnodes;
12M = sn.nstations;
13K = sn.nclasses;
14C = sn.nchains;
15N = sn.njobs';
16V = cellsum(sn.visits);
17Tstart=tic;
18QN = zeros(M,K);
19UN = zeros(M,K);
20RN = zeros(M,K);
21TN = zeros(M,K);
22CN = zeros(1,K);
23XN = zeros(1,K);
24
25lambda = zeros(1,K);
26for c=1:C
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);
31end
32
33chain = zeros(1,K);
34for k=1:K
35 chain(k) = find(sn.chains(:,k));
36end
37
38for ist=1:sn.nstations
39 switch sn.sched(ist)
40 case SchedStrategy.EXT
41 % no-op
42 case {SchedStrategy.FCFS, SchedStrategy.HOL, SchedStrategy.FCFSPRPRIO, SchedStrategy.PS}
43 % no-op
44 otherwise
45 if options.verbose
46 line_warning(mfilename,'The dec.mmap method does not support this scheduling strategy.\n');
47 end
48 [QN,UN,RN,TN,CN,XN] = deal([],[],[],[],[],[]);
49 totiter = 0;
50 method = '';
51 runtime = toc(Tstart);
52 return
53 end
54end
55
56if all(isinf(sn.njobs)) % is open
57 % open queueing system (one node is the external world)
58 pie = {};
59 D0 = {};
60 for ist=1:M
61 switch sn.sched(ist)
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, SchedStrategy.PS}
66 for k=1:K
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;
74 pie{ist}{k} = 1;
75 PH{ist}{k} = map_exponential(GlobalConstants.Immediate);
76 end
77 end
78 end
79 end
80
81 it_max = options.iter_max;
82 for it=1:it_max
83 %it
84 % now estimate arrival processes
85 if it == 1
86 % initially form departure processes using scaled service
87 DEP = PH;
88 for ind=1:M
89 for r=1:K
90 ist = sn.nodeToStation(ind);
91 DEP{ind,r} = map_scale(PH{ist}{r}, 1 / (lambda(r) * V(ind,r)) );
92 end
93 end
94 end
95
96 ARV = solver_mam_traffic(sn, DEP, config);
97
98 QN_1 = QN;
99 for ist=1:M
100 ind = sn.stationToNode(ist);
101 switch sn.nodetype(ind)
102 case NodeType.Queue
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});
106 end
107 TN(ist,:) = mmap_lambda(ARV{ind});
108 switch sn.sched(ist)
109 case {SchedStrategy.FCFS, SchedStrategy.HOL, SchedStrategy.FCFSPRPRIO}
110 [Qret{1:K}, ~] = MMAPPH1FCFS({ARV{ind}{[1,3:end]}}, {pie{ist}{:}}, {D0{ist,:}}, 'ncMoms', 1, 'ncDistr',2);
111 for k=1:K
112 QN(ist,k) = sum(Qret{k});
113 end
114 case SchedStrategy.PS
115 for k=1:K
116 UN(ist,k) = TN(ist,k) * map_mean(PH{ist}{k});
117 end
118 Uden = min([1-GlobalConstants.FineTol, sum(UN(ist,:))]);
119 for k=1:K
120 QN(ist,k) = UN(ist,k)/(1-Uden);
121 end
122 end
123 end
124 for k=1:K
125 UN(ist,k) = TN(ist,k) * map_mean(PH{ist}{k});
126 %add number of jobs at the surrogate delay server
127 QN(ist,k) = QN(ist,k) + TN(ist,k)*(map_mean(PH{ist}{k})*sn.nservers(ist)) * (sn.nservers(ist)-1)/sn.nservers(ist);
128 RN(ist,k) = QN(ist,k) ./ TN(ist,k);
129 end
130 end
131
132 if it >=3 && max(abs(QN(:)-QN_1(:))./QN_1(:)) < options.iter_tol
133 break;
134 end
135
136 for ist=1:M
137 ind = sn.stationToNode(ist);
138 switch sn.nodetype(ind)
139 case NodeType.Queue
140 for r=1:K
141 % extract class-r arrival MAP
142 A = mmap_hide(ARV{ind},setdiff(1:K,r));
143 S = PH{ist}{r};
144 na = length(A{1});
145 ns = length(S{1});
146 etaqa_n = config.etaqa_trunc;
147 etaqa_sz = (etaqa_n+1)*na*ns;
148 rho = sum(UN(ist,:));
149 % use ETAQA if state space is manageable and queue is stable
150 if etaqa_sz <= config.space_max && rho < 1-GlobalConstants.FineTol
151 try
152 switch sn.sched(ist)
153 case {SchedStrategy.FCFS, SchedStrategy.HOL, SchedStrategy.FCFSPRPRIO}
154 DEP{ind,r} = qbd_depproc_etaqa(A, S, etaqa_n);
155 case SchedStrategy.PS
156 DEP{ind,r} = qbd_depproc_etaqa_ps(A, S, etaqa_n);
157 end
158 DEP{ind,r} = map_normalize(DEP{ind,r});
159 catch
160 % fall back to scaled service on ETAQA failure
161 DEP{ind,r} = PH{ist}{r};
162 end
163 else
164 DEP{ind,r} = PH{ist}{r};
165 end
166 DEP{ind,r} = map_scale(DEP{ind,r}, 1 / (lambda(r) * V(ind,r)) );
167 SCVd(ind,r) = map_scv(DEP{ind,r});
168 IDCd(ind,r) = map_idc(DEP{ind,r});
169 end
170 end
171 end
172 end
173 totiter = it;
174 if options.verbose
175 line_printf('\nMAM parametric decomposition completed in %d iterations.',it);
176 end
177else
178 if options.verbose
179 line_warning(mfilename,'This model is not supported by SolverMAM yet. Returning with no result.\n');
180 end
181end
182runtime = toc(Tstart);
183end