1classdef Station < StatefulNode
2 % An abstract
class for
nodes where jobs station
4 % Copyright (c) 2012-2026, Imperial College London
12 lldScaling; % limited load-dependence scaling factors
13 lcdScaling; % limited
class-dependence scaling factors
15 patienceDistributions; % per-
class patience distributions (cell array indexed by class)
20 function self = Station(name)
21 % SELF = STATION(NAME)
23 self@StatefulNode(name);
27 self.patienceDistributions = [];
30 % don
't expose to avoid accidental call without checking the queue
31 % scheduling discipline
32 function setLimitedLoadDependence(self, alpha)
33 % SETLIMITEDLOADDEPENDENCE(self, alpha)
34 % alpha(ni) is the service rate scaling when there are ni>=1
36 self.lldScaling = alpha;
39 % don't expose to avoid accidental call without checking the queue
40 % scheduling discipline
41 function setLimitedClassDependence(self, gamma)
42 % SETLIMITEDCLASSDEPENDENCE(self, gamma)
44 % gamma(ni)
is a function handle, where ni=[ni1,...,niR]
45 %
is the service rate scaling when there are nir jobs at
46 % station i in
class r
47 if isa(gamma,'function_handle')
48 self.lcdScaling = gamma;
50 line_error(mfilename,
'Class dependence must be specified through a function handle.');
58 function self = setDropRule(self,
class, drop)
59 % SELF = SETDROPRULE(CLASS, DROPRULE)
61 self.dropRule(
class) = drop;
65 function setNumServers(self, value)
66 % SETNUMSERVERS(VALUE)
68 self.numberOfServers = value;
71 function setNumberOfServers(self, value)
72 % SETNUMBEROFSERVERS(VALUE)
74 self.numberOfServers = value;
77 function value = getNumServers(self)
78 % VALUE = GETNUMSERVERS()
80 value = self.numberOfServers;
83 function value = getNumberOfServers(self)
84 % VALUE = GETNUMBEROFSERVERS()
86 value = self.numberOfServers;
89 function setCapacity(self, value)
95 function setCap(self, value)
97 % Alias for setCapacity() for backwards compatibility
99 self.setCapacity(value);
102 function setChainCapacity(self, values)
103 % SETCHAINCAPACITY(VALUES)
105 sn = self.model.getStruct;
106 if numel(values) ~= sn.nchains
107 line_error(mfilename,'The method requires in input a capacity value for each chain.');
110 inchain = sn.inchain{c};
112 if ~self.isServiceDisabled(r)
113 self.classCap(r) = values(c);
115 self.classCap(r) = Inf;
119 self.cap = min(sum(self.classCap(self.classCap>0)), self.cap);
123 function isD = isServiceDefined(self,
class)
124 K = size(self.model.getClasses(),2);
126 switch self.server.className
131 if isempty(self.server.serviceProcess{1,r})
138 function isD = isServiceDisabled(self,
class)
139 % ISD = ISSERVICEDISABLED(CLASS)
141 switch self.server.className
145 isD = self.server.serviceProcess{1,
class}{end}.isDisabled();
148 K = size(self.model.getClasses(),2);
150 switch self.server.className
154 %isD = cellfun(@(sp) sp{end}.isDisabled, self.server.serviceProcess);
156 isD(r) = self.server.serviceProcess{1,r}{end}.isDisabled();
162 function isI = isServiceImmediate(self,
class)
163 % ISI = ISSERVICEIMMEDIATE(CLASS)
165 isI = self.server.serviceProcess{1,
class}{end}.isImmediate();
168 function R = getNumberOfServiceClasses(self)
169 % R = GETNUMBEROFSERVICECLASSES()
171 R = size(self.server.serviceProcess,2);
174 function [p] = getSelfLoopProbabilities(self)
175 % [
P] = GETSELFLOOPPROBABILITIES()
177 R = getNumberOfServiceClasses(self);
180 nOutLinks = length(self.output.outputStrategy{k}{end});
181 switch RoutingStrategy.toText(self.output.outputStrategy{k}{2})
183 p(k) = 1 / nOutLinks;
184 case RoutingStrategy.PROB
185 for t=1:nOutLinks %
for all outgoing links
186 if strcmp(self.output.outputStrategy{k}{end}{t}{1}.name, self.name)
187 p(k) = self.output.outputStrategy{k}{end}{t}{2};
195 function [
map, mu, phi] = getSourceRates(self)
196 % [PH,MU,PHI] = GETSOURCERATES()
198 nclasses = size(self.input.sourceClasses,2);
199 map = cell(1,nclasses);
200 mu = cell(1,nclasses);
201 phi = cell(1,nclasses);
203 if isempty(self.input.sourceClasses{r})
204 self.input.sourceClasses{r} = {[],ServiceStrategy.LI,Disabled.getInstance()};
205 map{r} = {[NaN],[NaN]};
208 elseif ~self.input.sourceClasses{r}{end}.isDisabled()
209 switch class(self.input.sourceClasses{r}{end})
210 case {
'Replayer',
'Trace'}
211 aph = self.input.sourceClasses{r}{end}.fitAPH;
212 map{r} =
aph.getProcess();
215 case {
'Exp',
'Coxian',
'Erlang',
'HyperExp',
'Markovian',
'APH',
'PH'}
216 map{r} = self.input.sourceClasses{r}{end}.getProcess;
217 mu{r} = self.input.sourceClasses{r}{end}.getMu;
218 phi{r} = self.input.sourceClasses{r}{end}.getPhi;
219 case {
'Det',
'Uniform',
'Pareto',
'Gamma',
'Lognormal',
'Weibull'}
220 map{r} = self.input.sourceClasses{r}{end}.getProcess();
221 mu{r} = [1/self.input.sourceClasses{r}{end}.getMean];
224 map{r} = self.input.sourceClasses{r}{end}.getProcess();
225 mu{r} = self.input.sourceClasses{r}{end}.getMu;
226 phi{r} = self.input.sourceClasses{r}{end}.getPhi;
228 map{r} = self.input.sourceClasses{r}{end}.getProcess();
229 mu{r} = self.input.sourceClasses{r}{end}.getMu;
230 phi{r} = self.input.sourceClasses{r}{end}.getPhi;
232 % leave everything empty
235 map{r} = {[NaN],[NaN]};
242 function [
map,mu,phi] = getServiceRates(self)
243 % [PH,MU,PHI] = GETSERVICERATES()
245 nclasses = size(self.server.serviceProcess,2);
246 map = cell(1,nclasses);
247 mu = cell(1,nclasses);
248 phi = cell(1,nclasses);
250 serviceProcess_r = self.server.serviceProcess{r};
251 if isempty(serviceProcess_r)
252 serviceProcess_r = {[],ServiceStrategy.LI,Disabled.getInstance()};
253 map{r} = {[NaN],[NaN]};
256 elseif serviceProcess_r{end}.isImmediate()
257 map{r} = {[-GlobalConstants.Immediate],[GlobalConstants.Immediate]};
258 mu{r} = [GlobalConstants.Immediate];
260 elseif ~serviceProcess_r{end}.isDisabled()
261 switch class(serviceProcess_r{end})
262 case {
'Det',
'Uniform',
'Pareto',
'Gamma',
'Weibull',
'Lognormal'}
263 map{r} = serviceProcess_r{end}.getProcess();
264 mu{r} = [serviceProcess_r{end}.getRate];
266 case {
'Replayer',
'Trace'}
267 aph = serviceProcess_r{end}.fitAPH;
268 map{r} =
aph.getProcess();
271 case {
'Exp',
'Coxian',
'Erlang',
'HyperExp',
'Markovian',
'APH',
'MAP',
'PH'}
272 map{r} = serviceProcess_r{end}.getProcess();
273 mu{r} = serviceProcess_r{end}.getMu;
274 phi{r} = serviceProcess_r{end}.getPhi;
276 map{r} = serviceProcess_r{end}.getProcess();
277 mu{r} = serviceProcess_r{end}.getMu;
278 phi{r} = serviceProcess_r{end}.getPhi;
281 map{r} = {[NaN],[NaN]};
288 function summary(self)
291 line_printf('\nNode: <strong>%s</strong>',self.getName);
292 line_printf('\nScheduling: %s',self.schedStrategy);
293 line_printf('\nNumber of Servers: %d',self.numberOfServers);
294 for r=1:length(self.output.outputStrategy)
295 classes = self.model.getClasses();
296 line_printf('\nRouting %s: %s',
classes{r}.name,self.output.outputStrategy{r}{2});
298 % self.input.summary;
299 % self.server.summary;
300 % self.output.summary;