1function [result, x, fval, exitflag] = mapqn_bnd_qr(params, objective_queue, objective_phase, sense)
2% MAPQN_BND_QR - General Quadratic Reduction Bounds
for MAP Queueing Networks
4% MATLAB port of the Python file bnd_qr.py
7% [result, x, fval, exitflag] = mapqn_bnd_qr(params)
8% [result, x, fval, exitflag] = mapqn_bnd_qr(params, objective_queue)
9% [result, x, fval, exitflag] = mapqn_bnd_qr(params, objective_queue, objective_phase)
10% [result, x, fval, exitflag] = mapqn_bnd_qr(params, objective_queue, objective_phase, sense)
13% params - Structure with model parameters:
14% .M - Number of queues
15% .N - Total population
16% .K - [M x 1] Number of phases
for each queue
17% .mu - {M x 1} cell, each mu{i}
is K(i) x K(i) completion rates
18% .v - {M x 1} cell, each v{i}
is K(i) x K(i) background rates
19% .r - [M x M] Routing probabilities
20% .verbose - (optional)
boolean,
default true
22% objective_queue - (optional) Queue index to optimize (1-based),
default 1
23% objective_phase - (optional) Phase index to optimize (1-based),
default 1
24% sense - (optional)
'min' or
'max',
default 'max'
27% result - Structure with results:
28% .objective - Objective function value
29% .exitflag - Solver exit flag
30% .U - [M x max(K)] Utilization matrix
31% .IT - [M x max(K)] Idle time matrix
32% .Q - [M x max(K)] Queue length matrix
33% .getP2 - Function handle to extract p2 values
34% x - Raw solution vector
35% fval - Objective function value
36% exitflag - Solver exit flag
38 if nargin < 2 || isempty(objective_queue)
41 if nargin < 3 || isempty(objective_phase)
44 if nargin < 4 || isempty(sense)
55 if isfield(params,
'verbose')
56 verbose = params.verbose;
63 % Compute transition rates q{i,j}(k,h)
67 q{i,j} = zeros(K(i), K(i));
71 q{i,j}(ki, hi) = r(i,j) * mu{i}(ki, hi);
73 q{i,j}(ki, hi) = v{i}(ki, hi) + r(i,i) * mu{i}(ki, hi);
80 %% Build variable indexing
81 if verbose; fprintf(
'Building variable index map...\n'); end
84 % U(i,k) variables: utilization at queue i, phase k
85 Uidx = zeros(M, maxK);
88 varCount = varCount + 1;
89 Uidx(i, k) = varCount;
93 % IT(i,k) variables: idle time at queue i, phase k
94 ITidx = zeros(M, maxK);
97 varCount = varCount + 1;
98 ITidx(i, k) = varCount;
102 % UP(j,k,i,h) variables: utilization products
103 UPidx = zeros(M, maxK, M, maxK);
108 varCount = varCount + 1;
109 UPidx(j, kj, i, hi) = varCount;
115 % QP(j,k,i,h) variables: queue-length products
116 QPidx = zeros(M, maxK, M, maxK);
121 varCount = varCount + 1;
122 QPidx(j, kj, i, hi) = varCount;
128 % Q(i,k) variables: mean queue length
129 Qidx = zeros(M, maxK);
132 varCount = varCount + 1;
133 Qidx(i, k) = varCount;
137 % C(j,k,i) variables: conditional queue lengths
138 Cidx = zeros(M, maxK, M);
142 varCount = varCount + 1;
143 Cidx(j, kj, i) = varCount;
148 % I_var(j,k,i) variables: conditional idle lengths
149 Iidx = zeros(M, maxK, M);
153 varCount = varCount + 1;
154 Iidx(j, kj, i) = varCount;
159 % p1(j,k,i,ni,h) variables: marginal probabilities, ni from 0 to N
160 % Stored as p1idx(j, k, i, ni+1, h)
161 p1idx = zeros(M, maxK, M, N+1, maxK);
167 varCount = varCount + 1;
168 p1idx(j, kj, i, ni+1, hi) = varCount;
175 % p1c(j,k,i,ni,h) variables: complementary marginal probabilities, ni from 0 to N
176 % Stored as p1cidx(j, k, i, ni+1, h)
177 p1cidx = zeros(M, maxK, M, N+1, maxK);
183 varCount = varCount + 1;
184 p1cidx(j, kj, i, ni+1, hi) = varCount;
191 % p2(j,nj,k,i,ni,h) variables: joint probabilities, nj from 0 to N, ni from 0 to N
192 % Stored as p2idx(j, nj+1, k, i, ni+1, h)
193 p2idx = zeros(M, N+1, maxK, M, N+1, maxK);
200 varCount = varCount + 1;
201 p2idx(j, nj+1, kj, i, ni+1, hi) = varCount;
210 if verbose; fprintf(
'Total variables: %d\n', nVars); end
213 lb = zeros(nVars, 1);
216 % Set upper bounds
for each variable type
228 ub(UPidx(j, kj, i, hi)) = 1;
229 ub(QPidx(j, kj, i, hi)) = N;
237 ub(Cidx(j, kj, i)) = N;
238 ub(Iidx(j, kj, i)) = N;
247 ub(p1idx(j, kj, i, ni+1, hi)) = 1;
248 ub(p1cidx(j, kj, i, ni+1, hi)) = 1;
260 ub(p2idx(j, nj+1, kj, i, ni+1, hi)) = 1;
274 if verbose; fprintf(
'Building constraints...\n'); end
276 %% ZER1: p1(j,k,j,0,k) = 0
for all j,k (via upper bounds)
277 if verbose; fprintf(
' ZER1 constraints...\n'); end
280 idx = p1idx(j, k, j, 0+1, k);
285 %% ZER2: p1(j,k,j,nj,h) = 0
for h ~= k, all j,k,nj (via upper bounds)
286 if verbose; fprintf(
' ZER2 constraints...\n'); end
292 idx = p1idx(j, k, j, nj+1, h);
300 %% ZER3: p1(j,k,i,N,h) = 0
for j ~= i, all j,k,i,h (via upper bounds)
301 if verbose; fprintf(
' ZER3 constraints...\n'); end
307 idx = p1idx(j, k, i, N+1, h);
315 %% ZER4: p1c(j,k,j,nj,h) = 0
for nj >= 1, all j,k,nj,h (via upper bounds)
316 if verbose; fprintf(
' ZER4 constraints...\n'); end
321 idx = p1cidx(j, k, j, nj+1, h);
328 %% CEQU: C(j,k,j) = Q(j,k)
for all j,k
329 if verbose; fprintf(
' CEQU constraints...\n'); end
332 row = zeros(1, nVars);
333 row(Cidx(j, k, j)) = 1;
334 row(Qidx(j, k)) = -1;
340 %% ONE1: sum over kj,hi,ni of (p1 + p1c) = 1
for each (j,i)
341 if verbose; fprintf(
' ONE1 constraints...\n'); end
344 row = zeros(1, nVars);
348 row(p1idx(j, kj, i, ni+1, hi)) = 1;
349 row(p1cidx(j, kj, i, ni+1, hi)) = 1;
358 %% UTLB: U(i,k) = sum over t,nt,h of p1(i,k,t,nt,h)
for each (i,k,t)
359 if verbose; fprintf(
' UTLB constraints...\n'); end
363 row = zeros(1, nVars);
367 row(p1idx(i, k, t, nt+1, h)) = -1;
376 %% UTLC: IT(i,k) = sum over t,nt,h of p1c(i,k,t,nt,h)
for each (i,k,t)
377 if verbose; fprintf(
' UTLC constraints...\n'); end
381 row = zeros(1, nVars);
382 row(ITidx(i, k)) = 1;
385 row(p1cidx(i, k, t, nt+1, h)) = -1;
394 %% QLEN: Q(i,k) = sum over ni of ni*p1(i,k,i,ni,k)
for each (i,k)
395 if verbose; fprintf(
' QLEN constraints...\n'); end
398 row = zeros(1, nVars);
401 row(p1idx(i, k, i, ni+1, k)) = row(p1idx(i, k, i, ni+1, k)) - ni;
408 %% SRVB: Service balance
409 % sum{j,h} q{i,j}(k,h)*U(i,k) = sum{j,h} q{i,j}(h,k)*U(i,h)
for each (i,k)
410 if verbose; fprintf(
' SRVB constraints...\n'); end
413 row = zeros(1, nVars);
416 row(Uidx(i, k)) = row(Uidx(i, k)) + q{i,j}(k, h);
417 row(Uidx(i, h)) = row(Uidx(i, h)) - q{i,j}(h, k);
425 %% POPC: sum over i,k of Q(i,k) = N
426 if verbose; fprintf(
' POPC constraint...\n'); end
427 row = zeros(1, nVars);
436 %% ONE: sum over k of (U(j,k) + IT(j,k)) = 1
for each j
437 if verbose; fprintf(
' ONE constraints...\n'); end
439 row = zeros(1, nVars);
442 row(ITidx(j, k)) = 1;
448 %% PCL2: sum over i,j,ni>=1,nj>=1,h,k of ni*nj*p2(i,ni,h,j,nj,k) = N^2
449 if verbose; fprintf(
' PCL2 constraint...\n'); end
450 row = zeros(1, nVars);
457 idx = p2idx(i, ni+1, h, j, nj+1, k);
458 row(idx) = row(idx) + ni * nj;
468 %% PI21: p1(j,k,i,ni,h) = sum over nj=1..N of p2(j,nj,k,i,ni,h)
469 if verbose; fprintf(
' PI21 constraints...\n'); end
475 row = zeros(1, nVars);
476 row(p1idx(j, k, i, ni+1, h)) = 1;
478 idx = p2idx(j, nj+1, k, i, ni+1, h);
479 row(idx) = row(idx) - 1;
489 %% PI22: p1c(j,k,i,ni,h) = p2(j,0,k,i,ni,h)
490 if verbose; fprintf(
' PI22 constraints...\n'); end
496 row = zeros(1, nVars);
497 row(p1cidx(j, k, i, ni+1, h)) = 1;
498 row(p2idx(j, 0+1, k, i, ni+1, h)) = -1;
507 %% PI23: p2(i,ni,h,j,nj,k) = p2(j,nj,k,i,ni,h) (symmetry)
508 % Only generate
for j < i, or (j == i and nj < ni), to avoid redundancy
509 if verbose; fprintf(
' PI23 (symmetry) constraints...\n'); end
516 % Only generate constraint
if (j,nj,k) < (i,ni,h) in lex order
517 if j < i || (j == i && nj < ni) || (j == i && nj == ni && k < h)
518 idx1 = p2idx(i, ni+1, h, j, nj+1, k);
519 idx2 = p2idx(j, nj+1, k, i, ni+1, h);
521 row = zeros(1, nVars);
535 %% UUB1: sum over k of U(i,k) <= 1
for each i (inequality)
536 if verbose; fprintf(
' UUB1 constraints...\n'); end
538 row = zeros(1, nVars);
542 Aineq = [Aineq; row];
546 %% QUB1: Q(j,k) <= N*U(j,k)
for each (j,k) (inequality)
547 if verbose; fprintf(
' QUB1 constraints...\n'); end
550 row = zeros(1, nVars);
552 row(Uidx(j, k)) = -N;
553 Aineq = [Aineq; row];
558 %% CUB1: C(j,k,i) <= sum over h of Q(i,h)
for each (j,k,i) (inequality)
559 if verbose; fprintf(
' CUB1 constraints...\n'); end
563 row = zeros(1, nVars);
564 row(Cidx(j, k, i)) = 1;
566 row(Qidx(i, h)) = -1;
568 Aineq = [Aineq; row];
574 %% CUB2: C(j,k,i) <= N*U(j,k)
for each (j,k,i) (inequality)
575 if verbose; fprintf(
' CUB2 constraints...\n'); end
579 row = zeros(1, nVars);
580 row(Cidx(j, k, i)) = 1;
581 row(Uidx(j, k)) = -N;
582 Aineq = [Aineq; row];
588 %% Build objective function
589 if verbose; fprintf(
'Building objective function...\n'); end
591 c(Uidx(objective_queue, objective_phase)) = 1;
593 if strcmp(sense,
'max')
599 fprintf('Solving LP with %d variables and %d equality + %d inequality constraints...\n', ...
600 nVars, size(Aeq, 1), size(Aineq, 1));
604 options = optimoptions('linprog', 'Display', 'final', 'Algorithm', 'interior-point');
606 options = optimoptions('linprog', 'Display', 'off', 'Algorithm', 'interior-point');
609 [x, fval, exitflag] = linprog(c, Aineq, bineq, Aeq, beq, lb, ub, options);
611 if strcmp(sense, 'max')
617 result.objective = fval;
618 result.exitflag = exitflag;
621 % Compute utilizations, idle times, queue lengths
622 result.U = zeros(M, maxK);
623 result.IT = zeros(M, maxK);
624 result.Q = zeros(M, maxK);
628 result.U(i, k) = x(Uidx(i, k));
629 result.IT(i, k) = x(ITidx(i, k));
630 result.Q(i, k) = x(Qidx(i, k));
634 % Function handle to extract p2 values: p2(j,nj,k,i,ni,h)
635 result.getP2 = @(j, nj, k, i, ni, h) x(p2idx(j, nj+1, k, i, ni+1, h));
638 if verbose; fprintf('\n=== Results ===\n'); end
639 if verbose; fprintf('Objective value: %f\n', fval); end
640 if verbose; fprintf('Exit flag: %d\n', exitflag); end
641 if exitflag > 0 && verbose
642 fprintf('\nUtilizations:\n');
644 fprintf(' Queue %d: U = [', i);
646 fprintf('%.6f ', result.U(i, k));
650 fprintf('\nIdle times:\n');
652 fprintf(' Queue %d: IT = [', i);
654 fprintf('%.6f ', result.IT(i, k));
658 fprintf('\nQueue lengths:\n');
660 fprintf(' Queue %d: Q = [', i);
662 fprintf('%.6f ', result.Q(i, k));