1function runtime = runAnalyzer(self, options)
4% Copyright (c) 2012-2026, Imperial College London
9if nargin<2 %%~exist(
'options',
'var')
10 options = self.getOptions;
13line_debug('JMT analyzer starting: lang=%s, samples=%d, seed=%d', options.lang, options.samples, options.seed);
15self.runAnalyzerChecks(options);
16Solver.resetRandomGeneratorSeed(options.seed);
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);
26 [QN,UN,RN,WN,AN,TN] = JLINE.arrayListToResults(jsolver.getAvgTable);
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)';
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;
44 line_debug('Using MATLAB JMT implementation');
46 if ~isfield(options,'verbose')
50 if ~isfield(options,'force')
51 options.force = false;
54 if ~isfield(options,'keep')
58 if self.enableChecks && ~self.supports(self.model)
60 line_error(mfilename,'This model contains features not supported by the solver.');
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);
72 options.samples = 5e3;
75 if ~isfield(options,'verbose')
79 if ~isfield(options,'keep')
80 options.verbose = false;
83 if ~isfield(options,'seed')
84 options.seed = randi([1,1e6]);
86 self.seed = options.seed;
88 if ~isfield(options,'timespan')
89 options.timespan = [0,Inf];
91 self.maxSimulatedTime = options.timespan(2);
94 if ~self.model.hasInitState
95 self.model.initDefault;
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;
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');
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');
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'];
118 line_printf('JMT model: %s\n',fname);
120 if options.verbose == VerboseLevel.DEBUG
121 line_printf('JMT command: %s\n',cmd);
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);
130 rmdir(getFilePath(self),'s');
133 % line_printf('\nJMT analysis (seed: %d) completed. Runtime: %f seconds.\n',options.seed,runtime);
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);
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);
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;
152 % Set the results on the Cache node
153 self.model.nodes{ind}.setResultHitProb(actualhitprob);
154 self.model.nodes{ind}.setResultMissProb(actualmissprob);
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;
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))
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);
180 validReplications = validReplications + 1;
182 tu = TranSysStateAggr{it}.t;
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
188 tumax = min(max(tu),max(TranSysStateAggr{it}.t));
189 tu =
union(tu, TranSysStateAggr{it}.t);
193 if validReplications == 0
194 line_error(mfilename,
'No valid replications produced. Cannot compute transient averages.');
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);
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)
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;
218 for it=1:options.iter_max
219 % Skip invalid replications
220 if isempty(TranSysStateAggr{it}.t) || ~isvector(TranSysStateAggr{it}.t)
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');
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;
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;
241 if isfinite(sn.nservers(jst))
242 TNt{jst,r}(:,1) = UNt{jst,r}(:,1) * sn.nservers(jst) * sn.rates(jst,r);
244 TNt{jst,r}(:,1) = UNt{jst,r}(:,1) * sn.rates(jst,r);
248 runtime = toc(Tstart);
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;
257 self.options.seed = initSeed;
258 self.options.timespan = initTimeSpan;
259 self.result.(
'solver') = getName(self);
260 self.result.runtime = runtime;
262 % line_printf(
'\nJMT analysis (seed: %d) completed. Runtime: %f seconds.\n',options.seed,runtime);
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'];
270 line_printf(
'JMT model: %s\n',fname);
272 if options.verbose == VerboseLevel.DEBUG
273 line_printf(
'JMT command: %s\n',cmd);
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);
282 rmdir(getFilePath(self),
's');
285 % line_printf(
'\nJMT analysis (method: %d) completed. Runtime: %f seconds.\n',options.method,runtime);
287 sn = self.model.getStruct();
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);
294 line_warning(mfilename,
'This solver does not support the specified method. Setting to default.\n');
295 self.options.method =
'default';