LINE Solver
MATLAB API documentation
Loading...
Searching...
No Matches
runAnalyzer.m
1function runtime = runAnalyzer(self, options)
2% TSIM = RUN()
3
4% Copyright (c) 2012-2026, Imperial College London
5% All rights reserved.
6
7Tstart=tic;
8
9if nargin<2 %%~exist('options','var')
10 options = self.getOptions;
11end
12
13line_debug('JMT analyzer starting: lang=%s, samples=%d, seed=%d', options.lang, options.samples, options.seed);
14
15self.runAnalyzerChecks(options);
16Solver.resetRandomGeneratorSeed(options.seed);
17
18switch options.lang
19 case 'java'
20 line_debug('Using Java JMT implementation, calling JLINE SolverJMT');
21 jmodel = LINE2JLINE(self.model);
22 M = jmodel.getNumberOfStations;
23 R = jmodel.getNumberOfClasses;
24 jsolver = JLINE.SolverJMT(jmodel, options);
25 T0=tic;
26 [QN,UN,RN,WN,AN,TN] = JLINE.arrayListToResults(jsolver.getAvgTable);
27 runtime = toc(T0);
28 CN = [];
29 XN = [];
30 QN = reshape(QN',R,M)';
31 UN = reshape(UN',R,M)';
32 RN = reshape(RN',R,M)';
33 WN = reshape(WN',R,M)';
34 AN = reshape(AN',R,M)';
35 TN = reshape(TN',R,M)';
36 lG = NaN;
37 lastiter = NaN;
38 self.setAvgResults(QN,UN,RN,TN,AN,WN,CN,XN,runtime,options.method,lastiter);
39 self.result.Prob.logNormConstAggr = lG;
40 self.result.solverSpecific.sn = JLINE.from_jline_struct(jmodel);
41 self.result.Prob.logNormConstAggr = lG;
42 return
43 case 'matlab'
44 line_debug('Using MATLAB JMT implementation');
45
46 if ~isfield(options,'verbose')
47 options.verbose = 0;
48 end
49
50 if ~isfield(options,'force')
51 options.force = false;
52 end
53
54 if ~isfield(options,'keep')
55 options.keep = false;
56 end
57
58 if self.enableChecks && ~self.supports(self.model)
59 % if options.verbose
60 line_error(mfilename,'This model contains features not supported by the solver.');
61 % end
62 % runtime = toc(T0);
63 % return
64 end
65
66 if ~isfield(options,'samples')
67 options.samples = 1e4; % default: this is the samples / measure, not the total number of simulation events, which can be much larger.
68 elseif options.samples < 5e3
69 %if ~strcmpi(options.method,'jmva.ls')
70 line_warning(mfilename,'JMT requires at least 5000 samples for each metric, the current value is %d. Starting the simulation with 5000 samples.\n', options.samples);
71 %end
72 options.samples = 5e3;
73 end
74
75 if ~isfield(options,'verbose')
76 options.verbose = 0;
77 end
78
79 if ~isfield(options,'keep')
80 options.verbose = false;
81 end
82
83 if ~isfield(options,'seed')
84 options.seed = randi([1,1e6]);
85 end
86 self.seed = options.seed;
87
88 if ~isfield(options,'timespan')
89 options.timespan = [0,Inf];
90 else
91 self.maxSimulatedTime = options.timespan(2);
92 end
93
94 if ~self.model.hasInitState
95 self.model.initDefault;
96 end
97
98 self.maxSamples = options.samples;
99 % Sync xmlParser maxSamples with solver (ensures corrected sample count is used in XML)
100 self.xmlParser.maxSamples = options.samples;
101 sn = self.getStruct;
102
103 % Auto-detect transient mode: switch to 'replication' if finite timespan with default method
104 if strcmpi(options.method, 'default') && isfield(options, 'timespan') && isfinite(options.timespan(2))
105 options.method = 'replication';
106 line_debug('Finite timespan detected, automatically using replication method for transient simulation');
107 end
108
109 switch options.method
110 case {'jsim','default'}
111 if strcmpi(options.method, 'default')
112 line_debug('Default method: using JSIM discrete-event simulation\n');
113 end
114 line_debug('Using JSIM method (discrete-event simulation)');
115 fname = self.writeJSIM(sn);
116 cmd = ['java -cp "',getJMTJarPath(self),filesep,'JMT.jar" jmt.commandline.Jmt sim "',fname,'" -seed ',num2str(options.seed),' --illegal-access=permit'];
117 if options.verbose
118 line_printf('JMT model: %s\n',fname);
119 end
120 if options.verbose == VerboseLevel.DEBUG
121 line_printf('JMT command: %s\n',cmd);
122 end
123 [status, cmdoutput] = system(cmd);
124 runtime = toc(Tstart);
125 if status ~= 0 && options.verbose
126 line_printf('\nJMT command failed with status %d. Output:\n%s\n', status, cmdoutput);
127 end
128 self.getResults;
129 if ~options.keep
130 rmdir(getFilePath(self),'s');
131 end
132 %if options.verbose
133 % line_printf('\nJMT analysis (seed: %d) completed. Runtime: %f seconds.\n',options.seed,runtime);
134 %end
135 self.setAvgResults(self.result.Avg.Q,self.result.Avg.U,self.result.Avg.R,self.result.Avg.T,self.result.Avg.A,self.result.Avg.W,[],[],runtime,options.method,1);
136
137 % Set cache hit/miss ratios from JMT Cache Hit Rate metric
138 for ind = 1:sn.nnodes
139 if sn.nodetype(ind) == NodeType.Cache
140 hitclass = sn.nodeparam{ind}.hitclass;
141 nclasses = length(hitclass);
142 actualhitprob = zeros(1, nclasses);
143 actualmissprob = zeros(1, nclasses);
144
145 % Check if JMT returned Cache Hit Rate metrics
146 fieldName = sprintf('node%d', ind);
147 if isfield(self.result, 'CacheHitRate') && isfield(self.result.CacheHitRate, fieldName)
148 actualhitprob = self.result.CacheHitRate.(fieldName);
149 actualmissprob = 1 - actualhitprob;
150 end
151
152 % Set the results on the Cache node
153 self.model.nodes{ind}.setResultHitProb(actualhitprob);
154 self.model.nodes{ind}.setResultMissProb(actualmissprob);
155
156 % Also update the cached sn struct directly
157 sn.nodeparam{ind}.actualhitprob = actualhitprob;
158 sn.nodeparam{ind}.actualmissprob = actualmissprob;
159 self.model.sn.nodeparam{ind}.actualhitprob = actualhitprob;
160 self.model.sn.nodeparam{ind}.actualmissprob = actualmissprob;
161 end
162 end
163 case {'replication'}
164 line_debug('Using replication method (transient simulation)');
165 options = self.getOptions;
166 initSeed = self.options.seed;
167 initTimeSpan = self.options.timespan;
168 self.options.timespan(1) = self.options.timespan(2);
169 if isfield(options,'timespan') && isfinite(options.timespan(2))
170 tu = [];
171 validReplications = 0;
172 for it=1:options.iter_max
173 self.options.seed = initSeed + it -1;
174 TranSysStateAggr{it} = sampleSysAggr(self);
175 % Skip replications with empty or invalid time vectors
176 if isempty(TranSysStateAggr{it}.t) || ~isvector(TranSysStateAggr{it}.t)
177 line_warning(mfilename, 'Replication %d produced empty/invalid time series, skipping.', it);
178 continue;
179 end
180 validReplications = validReplications + 1;
181 if isempty(tu)
182 tu = TranSysStateAggr{it}.t;
183 else
184 % we need to limit the time series at the minimum
185 % as otherwise the predictor of the state cannot
186 % take into account constraints that exist on the
187 % state space
188 tumax = min(max(tu),max(TranSysStateAggr{it}.t));
189 tu = union(tu, TranSysStateAggr{it}.t);
190 tu = tu(tu<=tumax);
191 end
192 end
193 if validReplications == 0
194 line_error(mfilename, 'No valid replications produced. Cannot compute transient averages.');
195 return;
196 end
197 QNt = cellzeros(sn.nstations, sn.nclasses, length(tu), 2);
198 UNt = cellzeros(sn.nstations, sn.nclasses, length(tu), 2);
199 TNt = cellzeros(sn.nstations, sn.nclasses, length(tu), 2);
200 M = sn.nstations;
201 K = sn.nclasses;
202 for jst=1:M
203 for r=1:K
204 QNt{jst,r}(:,2) = tu;
205 UNt{jst,r}(:,2) = tu;
206 TNt{jst,r}(:,2) = tu;
207 for it=1:options.iter_max
208 % Skip invalid replications
209 if isempty(TranSysStateAggr{it}.t) || ~isvector(TranSysStateAggr{it}.t)
210 continue;
211 end
212 qlenAt_t = interp1(TranSysStateAggr{it}.t, TranSysStateAggr{it}.state{jst}(:,r), tu,'previous');
213 avgQlenAt_t = qlenAt_t;
214 %avgQlenAt_t = cumsum(qlenAt_t .*[0;diff(tu)])./tu;
215 avgQlenAt_t(isnan(avgQlenAt_t))=0;
216 QNt{jst,r}(:,1) = QNt{jst,r}(:,1) + (1/validReplications) * avgQlenAt_t;
217 end
218 for it=1:options.iter_max
219 % Skip invalid replications
220 if isempty(TranSysStateAggr{it}.t) || ~isvector(TranSysStateAggr{it}.t)
221 continue;
222 end
223 if isfinite(sn.nservers(jst))
224 occupancyAt_t = interp1(TranSysStateAggr{it}.t, min(TranSysStateAggr{it}.state{jst}(:,r),sn.nservers(jst)), tu,'previous')/sn.nservers(jst);
225 else % if delay we use queue-length
226 occupancyAt_t = interp1(TranSysStateAggr{it}.t, TranSysStateAggr{it}.state{jst}(:,r), tu,'previous');
227 end
228 avgOccupancyAt_t = occupancyAt_t;
229 %avgOccupancyAt_t = cumsum(occupancyAt_t .*[0;diff(tu)])./tu;
230 avgOccupancyAt_t(isnan(avgOccupancyAt_t))=0;
231 UNt{jst,r}(:,1) = UNt{jst,r}(:,1) + (1/validReplications) * avgOccupancyAt_t;
232 end
233 % for it=1:options.iter_max
234 % departures = [0;diff(TranSysStateAggr{it}.state{j}(:,r))];
235 % departures(departures>0) = 0;
236 % departuresAt_t = abs(interp1(TranSysStateAggr{it}.t, cumsum(departures), tu, 'previous'));
237 % avgDeparturesAt_t = departuresAt_t./tu;
238 % avgDeparturesAt_t(isnan(avgDeparturesAt_t))=0;
239 % TNt{j,r}(:,1) = TNt{j,r}(:,1) + (1/options.iter_max) * avgDeparturesAt_t;
240 % end
241 if isfinite(sn.nservers(jst))
242 TNt{jst,r}(:,1) = UNt{jst,r}(:,1) * sn.nservers(jst) * sn.rates(jst,r);
243 else
244 TNt{jst,r}(:,1) = UNt{jst,r}(:,1) * sn.rates(jst,r);
245 end
246 end
247 end
248 runtime = toc(Tstart);
249 RNt = [];
250 CNt = [];
251 XNt = [];
252 self.setTranAvgResults(QNt,UNt,RNt,TNt,CNt,XNt,runtime);
253 self.result.Tran.Avg.U = UNt;
254 self.result.Tran.Avg.T = TNt;
255 self.result.Tran.Avg.Q = QNt;
256 end
257 self.options.seed = initSeed;
258 self.options.timespan = initTimeSpan;
259 self.result.('solver') = getName(self);
260 self.result.runtime = runtime;
261 %if options.verbose
262 % line_printf('\nJMT analysis (seed: %d) completed. Runtime: %f seconds.\n',options.seed,runtime);
263 %end
264 case {'jmva','jmva.amva','jmva.mva','jmva.recal','jmva.comom','jmva.chow','jmva.bs','jmva.aql','jmva.lin','jmva.dmlin','jmva.ls',...
265 'jmt.jmva','jmt.jmva.mva','jmt.jmva.amva','jmt.jmva.recal','jmt.jmva.comom','jmt.jmva.chow','jmt.jmva.bs','jmt.jmva.aql','jmt.jmva.lin','jmt.jmva.dmlin','jmt.jmva.ls'}
266 line_debug('Using JMVA method: %s', options.method);
267 fname = self.writeJMVA(sn, getJMVATempPath(self), self.options);
268 cmd = ['java -cp "',getJMTJarPath(self),filesep,'JMT.jar" jmt.commandline.Jmt mva "',fname,'" -seed ',num2str(options.seed),' --illegal-access=permit'];
269 if options.verbose
270 line_printf('JMT model: %s\n',fname);
271 end
272 if options.verbose == VerboseLevel.DEBUG
273 line_printf('JMT command: %s\n',cmd);
274 end
275 [status, cmdoutput] = system(cmd);
276 runtime = toc(Tstart);
277 if status ~= 0 && options.verbose
278 line_printf('\nJMT command failed with status %d. Output:\n%s\n', status, cmdoutput);
279 end
280 self.getResults;
281 if ~options.keep
282 rmdir(getFilePath(self),'s');
283 end
284 %if options.verbose
285 % line_printf('\nJMT analysis (method: %d) completed. Runtime: %f seconds.\n',options.method,runtime);
286 %end
287 sn = self.model.getStruct();
288
289 % Compute average arrival rate at steady-state
290 TH = getAvgTputHandles(self);
291 AN = sn_get_arvr_from_tput(sn, self.result.Avg.T, TH);
292 self.setAvgResults(self.result.Avg.Q,self.result.Avg.U,self.result.Avg.R,self.result.Avg.T,AN,self.result.Avg.W,[],[],runtime,options.method,1);
293 otherwise
294 line_warning(mfilename,'This solver does not support the specified method. Setting to default.\n');
295 self.options.method = 'default';
296 runAnalyzer(self);
297 end
298end
299end