LINE Solver
MATLAB API documentation
Loading...
Searching...
No Matches
pfqn_ncld.m
1%{
2%{
3 % @file pfqn_ncld.m
4 % @brief Normalizing constant for load-dependent closed networks.
5%}
6%}
7
8%{
9%{
10 % @brief Normalizing constant for load-dependent closed networks.
11 % @fn pfqn_ncld(L, N, Z, mu, varargin)
12 % @param L Service demand matrix.
13 % @param N Population vector.
14 % @param Z Think time vector.
15 % @param mu Load-dependent rate matrix.
16 % @param varargin Optional solver parameters.
17 % @return lG Logarithm of normalizing constant.
18 % @return G Normalizing constant.
19 % @return method Method used for computation.
20%}
21%}
22function [lG,G,method] = pfqn_ncld(L,N,Z,mu,varargin)
23% [LGN,G,METHOD] = PFQN_NCLD(L,N,Z,VARARGIN)
24
25options = Solver.parseOptions(varargin, SolverNC.defaultOptions);
26lG = NaN;
27G = NaN;
28method = options.method;
29
30% backup initial parameters
31
32mu = mu(:,1:sum(N));
33% first remove empty classes
34nnzClasses = find(N);
35L = L(:,nnzClasses);
36N = N(:,nnzClasses);
37Z = Z(:,nnzClasses);
38
39% then scale demands in [0,1], importat that stays before the other
40% simplications in case both D and Z are all very small or very large in a
41% given class, in which case the may look to filter but not if all of them
42% are at the same scale
43R = length(N);
44scalevec = ones(1,R);
45for r=1:R
46 scalevec(r) = max([L(:,r);Z(:,r)]);
47end
48L = L ./ repmat(scalevec,size(L,1),1);
49Z = Z ./ scalevec;
50
51% remove stations with no demand
52Lsum = sum(L,2);
53Lmax = max(L,[],2);
54demStations = find((Lmax./Lsum)>GlobalConstants.FineTol);
55L = L(demStations,:);
56mu = mu(demStations,:);
57
58% if there is a class with jobs but with L and Z all zero
59if any(N((sum(L,1) + sum(Z,1)) == 0)>0)
60 line_warning(mfilename,'The model has no positive demands in any class.\n');
61 if isempty(Z) || sum(Z(:))<options.tol
62 lG = 0;
63 else
64 lG = - sum(factln(N)) + sum(N.*log(sum(Z,1))) + N*log(scalevec)';
65 end
66 G = NaN;
67
68 return
69end
70
71% update M and R
72[M,R]=size(L);
73
74% return immediately if the model is a degenerate case
75if isempty(L) || sum(L(:))<options.tol % all demands are zero
76 if isempty(Z) || sum(Z(:))<options.tol
77 lG = 0;
78 else
79 lG = - sum(factln(N)) + sum(N.*log(sum(Z,1))) + N*log(scalevec)';
80 end
81 return
82elseif M==1 && (isempty(Z) || sum(Z(:))<options.tol) % single node and no think time
83 lG = factln(sum(N)) - sum(factln(N)) + sum(N.*log(sum(L,1))) + N*log(scalevec)' - sum(log(mu(1:sum(N))));
84 return
85end
86
87% determine contribution from jobs that permanently loop at delay
88zeroDemandClasses = find(sum(L,1)<options.tol); % all jobs in delay
89nonzeroDemandClasses = setdiff(1:R, zeroDemandClasses);
90
91if isempty(sum(Z,1)) || all(sum(Z(:,zeroDemandClasses),1)<options.tol)
92 lGzdem = 0;
93 Nz = 0;
94else
95 if isempty(zeroDemandClasses) % for old MATLAB release compatibility
96 lGzdem = 0;
97 Nz = 0;
98 else
99 Nz = N(zeroDemandClasses);
100 lGzdem = - sum(factln(Nz)) + sum(Nz.*log(sum(Z(:,zeroDemandClasses),1))) + Nz*log(scalevec(zeroDemandClasses))';
101 end
102end
103L = L(:,nonzeroDemandClasses);
104N = N(nonzeroDemandClasses);
105Z = Z(:,nonzeroDemandClasses);
106scalevecz = scalevec(nonzeroDemandClasses);
107% compute G for classes No with non-zero demand
108if any(N<0)
109 lGnnzdem = 0;
110else
111 [lGnnzdem,method] = compute_norm_const_ld(L, N, Z, mu, options);
112end
113% scale back to original demands
114lG = lGnnzdem + lGzdem + N*log(scalevecz)';
115G = exp(lG);
116end
117
118function [lG,method] = compute_norm_const_ld(L,N,Z,mu,options)
119% LG = COMPUTE_NORM_CONST_LD(L,N,Z,OPTIONS)
120[M,R] = size(L);
121method = options.method;
122switch options.method
123 case {'default','exact'}
124 if sum(Z(:))<GlobalConstants.FineTol
125 Lz = L;
126 muz = mu;
127 else
128 D = size(Z,1); % number of delays
129 Lz = [L;Z];
130 muz = [mu; repmat(1:size(mu,2),D,1)];
131 end
132 if R==1
133 [lG] = pfqn_gldsingle(Lz, N, muz, options);
134 method = 'exact/gld';
135 elseif M==1 && any(Z>0)
136 [~,lG]= pfqn_comomrm_ld(L, N, Z, mu, options);
137 method = 'exact/comomld';
138 elseif M==2 && max(Z(:)) < GlobalConstants.FineTol
139 [~,lG] = pfqn_comomrm_ld(L, N, 0*N, mu, options);
140 method = 'exact/comomld';
141 else
142 [~,lG] = pfqn_gld(Lz, N, muz, options);
143 method = 'exact/gld';
144 end
145 case 'rd'
146 [lG] = pfqn_rd(L, N, Z, mu, options);
147 case 'nrp'
148 [lG] = pfqn_nrp(L, N, Z, mu, options);
149 case 'nrl'
150 [lG] = pfqn_nrl(L, N, Z, mu, options);
151 case 'comomld'
152 if M<=1 || sum(Z) < GlobalConstants.Zero
153 [~,lG]= pfqn_comomrm_ld(L, N, Z, mu, options);
154 else
155 line_warning(mfilename,'Load-dependent CoMoM is available only in models with a delay and m identical stations, running the ''rd'' algorithm instead.\n');
156 [lG] = pfqn_rd(L, N, Z, mu, options);
157 method = 'rd';
158 end
159 otherwise
160 line_error(mfilename,sprintf('Unrecognized method for solving load-dependent models: %s',options.method));
161end
162return
163end