1function demandEst = infer_mlps(model, node, rt,
class, ql)
2% INFER_MLPS MLPS demand estimation
using sn
struct-level operations.
4% Estimates service demands at a PS queue
using the Maximum Likelihood
5%
for Processor Sharing method. Pre-builds augmented CTMC models
for
6% each unique (tagClass, aQueue) combination once, then uses
7% sn_set_service + solver_ctmc directly inside the optimization loop.
10% model - LINE Network model with delay rates set and queue rates to estimate
11% node - PS queue node (Station
object)
12% rt - response time samples (column vector, n x 1)
13%
class - class of each sample (column vector, n x 1)
14% ql - queue lengths at arrival (n x R matrix, per-
class)
17% demandEst - 1 x R vector of estimated mean service demands
19% Copyright (c) 2012-2026, Imperial College London
21% This code
is released under the 3-Clause BSD License.
23sn = model.getStruct();
25nCores = node.getNumberOfServers();
27% Get delay rates from model
30 if sn.sched(i) == SchedStrategy.INF
37 muZ(k) = sn.mu{delayIdx}{k}(1);
40% Initial point estimate
41meanQL = mean(sum(ql, 2));
42Vtilde = min(meanQL, nCores);
45 if ~isempty(rt(
class == j))
46 x0(j) = Vtilde * mean(rt(
class == j)) / meanQL;
53xUB = max(rt) * ones(1, R);
56augMuZ = [muZ, 0]; % placeholder
for tagged
class rate
58%% Pre-build augmented CTMC models
for all unique (tagClass, aQueue)
59% This eliminates Network/ClosedClass/SolverCTMC construction from the
60% optimization loop. Only rates change between iterations.
61uniqueTC = unique(
class);
62uniqueQL = unique(ql,
'rows');
63ctmcOpts = []; % will be set from first solver instance
65prebuilt = containers.Map(
'KeyType',
'char',
'ValueType',
'any');
66for u = 1:length(uniqueTC)
68 augMuZ(newR) = muZ(tc);
69 for v = 1:size(uniqueQL, 1)
72 % Population: move 1 job from tagClass to tagged class
73 N = aq; N(tc) = N(tc) - 1; N(newR) = 1;
75 % Build augmented Network model (once per combo)
76 augModel = Network('mlps_aug');
78 augNode{1} = Delay(augModel,
'Think');
79 augNode{2} = Queue(augModel,
'Queue1', SchedStrategy.PS);
80 augNode{2}.setNumberOfServers(nCores);
82 augClass = ClosedClass(augModel, sprintf(
'Class%d', r), N(r), augNode{1}, 0);
83 augNode{1}.setService(augClass, Exp(augMuZ(r)));
84 augNode{2}.setService(augClass, Exp(1)); % placeholder
86 P = augModel.initRoutingMatrix;
88 P{r} = Network.serialRouting(augNode);
92 % Use SolverCTMC to get proper options and run initial solve
93 solver = SolverCTMC(augModel);
95 ctmcOpts = solver.getOptions();
97 [infGen, eventFilt, ev] = solver.getGenerator();
98 stateSpaceAggr = solver.getStateSpaceAggr();
99 augSn = augModel.getStruct();
101 % Find tagged departure
event indices (invariant across rate changes)
102 queueNodeIdx = augNode{2}.index;
105 if ev{e}.active{1}.node == queueNodeIdx && ...
106 ev{e}.active{1}.class == newR && ...
107 ev{e}.active{1}.event == EventType.DEP
108 taggedDepIdx(end+1) = e;
112 % Find subset where tagged job
is at Queue (invariant)
113 queueStIdx = augNode{2}.stationIndex;
114 taggedColAtQueue = (queueStIdx - 1) * newR + newR;
115 subset = find(stateSpaceAggr(:, taggedColAtQueue) == 1);
117 % Extract queue state space
for subset (invariant)
118 queueCols = (queueStIdx - 1) * newR + (1:newR);
119 SSqueue = stateSpaceAggr(subset, queueCols);
121 cacheKey = mat2str([tc, aq]);
122 prebuilt(cacheKey) =
struct(...
124 'queueStIdx', queueStIdx, ...
125 'taggedDepIdx', taggedDepIdx, ...
126 'subset', subset, ...
127 'SSqueue', SSqueue, ...
133%% Optimization options
135options.Display =
'iter';
136options.LargeScale =
'on';
137options.MaxIter = 1e10;
138options.MaxFunEvals = 1e10;
139options.MaxSQPIter = 5000;
140options.TolCon = 1e-6;
141options.Algorithm =
'interior-point';
143[demandEst, ~] = fmincon(@objfun, x0, [], [], [], [], xLB, xUB, [], options);
145 function f = objfun(x)
147 rates = 1 ./ x; % x = mean demands
149 % Build cache: update rates via sn_set_service, call solver_ctmc
150 cache = containers.Map(
'KeyType',
'char',
'ValueType',
'any');
152 keys = prebuilt.keys;
153 for kk = 1:length(keys)
157 % Set augmented rates: base
classes + tagged
class
158 augRates = [rates, rates(pb.tagClass)];
160 % Update service rates in cached sn
struct (cell-of-cells format)
163 sn_upd = sn_set_service_coc(sn_upd, pb.queueStIdx, cc, augRates(cc));
166 % Run solver_ctmc with updated rates (reuses sn topology/state structure)
167 [infGen, ~, ~, Dfilt] = solver_ctmc(sn_upd, ctmcOpts);
169 % Build D1 from cached departure
event indices
170 D1 = sparse(size(infGen, 1), size(infGen, 2));
171 for di = 1:length(pb.taggedDepIdx)
172 D1 = D1 + Dfilt{pb.taggedDepIdx(di)};
175 % Extract sub-generator
using cached subset
177 A = full(MAPQ1(pb.subset, pb.subset));
179 cache(key) =
struct(
'A', A,
'SSqueue', pb.SSqueue,
'N', pb.N);
182 % Compute likelihoods
using cached CTMCs
183 ftemp = zeros(size(rt));
185 cacheKey = mat2str([
class(r), ql(r, :)]);
186 cached = cache(cacheKey);
187 ftemp(r) = log(TOL + eval_mlps_likelihood(cached.A, cached.SSqueue, cached.N, rt(r)));
195function LIKE = eval_mlps_likelihood(A, SSqueue, N, Rsam)
196% Compute MLPS likelihood from pre-built CTMC components.
198pie = zeros(1, length(A));
199idx = matchrow(SSqueue, N);
200if ~isempty(idx) && idx > 0
204MAP = {A, -A * ones(size(pie))
' * pie};
205LIKE = map_pdf(MAP, Rsam);