LINE Solver
MATLAB API documentation
Loading...
Searching...
No Matches
solver_mam_traffic_fj.m
1function ARV = solver_mam_traffic_fj(sn, DEP, config, fjSyncMap)
2% ARV = SOLVER_MAM_TRAFFIC_FJ(SN, DEP, CONFIG, FJSYNCMAP)
3% FJ-aware traffic solver extending solver_mam_traffic with mmap_max
4% synchronization at join points.
5%
6% DEP{i,r} is the departure process of class r from i in (D0,D1) format
7% fjSyncMap is built by sn_build_fj_sync_map
8%
9% Copyright (c) 2012-2026, Imperial College London
10% All rights reserved.
11
12I = sn.nnodes;
13R = sn.nclasses;
14
15if ~isfield(config, 'fj_sync_q_len')
16 config.fj_sync_q_len = 2;
17end
18
19% In this function we use indexing over all non-ClassSwitch nodes
20non_cs_classes = [];
21isNCS = zeros(1,I);
22nodeToNCS = zeros(1,I);
23NCStoNode = [];
24for ind=1:I
25 if sn.nodetype(ind) ~= NodeType.ClassSwitch
26 non_cs_classes(end+1:end+R)= ((ind-1)*R+1):(ind*R);
27 isNCS(ind) = true;
28 nodeToNCS(ind) = sum(isNCS);
29 NCStoNode(end+1) = ind;
30 else
31 isNCS(ind) = false;
32 end
33end
34
35% Hide the nodes that are class switches
36rtncs = dtmc_stochcomp(sn.rtnodes, non_cs_classes);
37Inc = I - sum(sn.nodetype == NodeType.ClassSwitch);
38
39% DEP is indexed by node (ind, r) - convert to MMAP format
40MMAP = cell(I, R);
41for ind=1:I
42 for r=1:R
43 MMAP{ind,r} = DEP{ind,r};
44 if isempty(MMAP{ind,r}) || any(any(isnan(MMAP{ind,r}{1})))
45 MMAP{ind,r} = {[0],[0],[0]}; % no arrivals from this class
46 else
47 MMAP{ind,r}{3} = MMAP{ind,r}{2};
48 end
49 end
50end
51
52ARV = cell(Inc,1);
53DEP_NCS = cell(Inc,R);
54LINKS = cell(Inc,Inc);
55
56% Build the nodeSync matrix in NCS indexing
57nodeSyncNCS = zeros(Inc, Inc);
58for ind=1:I
59 if isNCS(ind)
60 inc = nodeToNCS(ind);
61 for jnd=1:I
62 if isNCS(jnd)
63 jnc = nodeToNCS(jnd);
64 if fjSyncMap.nodeSync(ind, jnd) > 0
65 nodeSyncNCS(inc, jnc) = fjSyncMap.nodeSync(ind, jnd);
66 end
67 end
68 end
69 end
70end
71
72% First determine all outgoing flows from all nodes
73for ind=1:I
74 if isNCS(ind)
75 inc = nodeToNCS(ind);
76 switch sn.nodetype(ind)
77 case {NodeType.Source, NodeType.Delay, NodeType.Queue, NodeType.Fork, NodeType.Join}
78 % obtain departure maps (MMAP is now node-indexed)
79 if R>1
80 DEP_NCS{inc} = mmap_super({MMAP{ind,1:R}});
81 else
82 DEP_NCS{inc} = MMAP{ind,1};
83 end
84 Psplit = zeros(R,Inc*R);
85 for r=1:R
86 for jnd = 1:I
87 if isNCS(jnd)
88 jnc = nodeToNCS(jnd);
89 for s=1:R
90 Psplit(r,(jnc-1)*R+s) = rtncs((inc-1)*R+r, (jnc-1)*R+s);
91 end
92 end
93 end
94 end
95
96 [Fsplit{1:Inc}] = npfqn_traffic_split_cs(DEP_NCS{inc}, Psplit, config);
97 for jnc=1:Inc
98 LINKS{inc,jnc} = Fsplit{jnc};
99 LINKS{inc,jnc} = mmap_normalize(LINKS{inc,jnc});
100 end
101 end
102 end
103end
104
105% Then determine all incoming flows, with FJ synchronization
106for ind=1:I
107 if isNCS(ind) && sn.nodetype(ind) ~= NodeType.Source
108 inc = nodeToNCS(ind);
109
110 % Partition incoming links into sync groups and independent flows
111 syncGroupsAtNode = unique(nodeSyncNCS(inc, :));
112 syncGroupsAtNode = syncGroupsAtNode(syncGroupsAtNode > 0);
113
114 independentFlows = {};
115 syncFlows = struct();
116
117 for jnc=1:Inc
118 if isempty(LINKS{jnc,inc}) || sum(mmap_lambda(LINKS{jnc,inc})) <= GlobalConstants.FineTol
119 continue;
120 end
121 gid = nodeSyncNCS(inc, jnc);
122 if gid == 0
123 % Independent flow
124 independentFlows{end+1} = LINKS{jnc,inc};
125 else
126 % Synchronized flow — group by sync group ID
127 fname = ['g' num2str(gid)];
128 if ~isfield(syncFlows, fname)
129 syncFlows.(fname) = {};
130 end
131 syncFlows.(fname){end+1} = LINKS{jnc,inc};
132 end
133 end
134
135 % Process synchronized flows: apply mmap_max iteratively within each group
136 syncResults = {};
137 for g = 1:length(syncGroupsAtNode)
138 gid = syncGroupsAtNode(g);
139 fname = ['g' num2str(gid)];
140 if ~isfield(syncFlows, fname)
141 continue;
142 end
143 groupFlows = syncFlows.(fname);
144 if isempty(groupFlows)
145 continue;
146 end
147
148 % Start with first flow, iteratively apply mmap_max with remaining
149 syncedFlow = groupFlows{1};
150 for f = 2:length(groupFlows)
151 syncedFlow = mmap_max(syncedFlow, groupFlows{f}, config.fj_sync_q_len);
152
153 % Compress after each mmap_max step if state space is too large
154 if length(syncedFlow{1}) > config.space_max
155 syncedFlow = mmap_compress(syncedFlow, struct('method', config.compress));
156 end
157 end
158 syncResults{end+1} = syncedFlow;
159 end
160
161 % Merge synced flows with independent flows
162 allFlows = [syncResults, independentFlows];
163
164 if length(allFlows) > 1
165 ARV{ind} = npfqn_traffic_merge(allFlows, config);
166 elseif length(allFlows) == 1
167 ARV{ind} = allFlows{1};
168 else
169 % No flows — take a zero-rate link
170 for jnc=1:Inc
171 if ~isempty(LINKS{jnc,inc})
172 ARV{ind} = LINKS{jnc,inc};
173 break;
174 end
175 end
176 if isempty(ARV{ind})
177 ARV{ind} = {[0],[0],[0]};
178 end
179 end
180 else
181 ARV{ind} = [];
182 end
183end
184end
Definition mmt.m:93