aboutsummaryrefslogtreecommitdiffstats
path: root/doc/lchan.msc
blob: 9b7d663c41755efd6771348a631f5a4994a1b15d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
msc {
	hscale=2;
	bts [label="MS/BTS"], bsc[label="BSC"], bsc_ts [label="BSC timeslot FSM"],
	bsc_lchan[label="BSC lchan FSM"], bsc_gscon[label="BSC conn FSM"],
	mgw_msc[label="MGW/MSC"];

	bts box mgw_msc [label="lchan allocation sequence"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_UNUSED"];

	bts rbox mgw_msc [label="Channel Request from MS"];
	bts => bsc [label="RSL Channel Request"];
	bsc box bsc [label="lchan_select_by_type(chan_type)"];
	bsc -> bsc_lchan [label="lchan_activate(lchan, FOR_MS_CHANNEL_REQUEST)"];
	bsc_lchan rbox bsc_lchan [label="Continue at\nlchan_activate()\n"];
	|||;
	|||;

	bts rbox mgw_msc [label="Channel Request from BSSMAP Assignment"];
	bsc_gscon <= mgw_msc [label="BSSMAP Assignment request"];
	bsc_gscon box bsc_gscon [label="lchan_select_by_chan_mode(chan_mode)"];
	bsc_lchan <- bsc_gscon [label="lchan_activate(lchan, FOR_ASSIGNMENT)"];
	bsc_lchan rbox bsc_lchan [label="Continue at\nlchan_activate()\n"];
	|||;
	|||;

	bts rbox mgw_msc [label="Channel Request from Handover Decision"];
	bsc note bsc [label="target lchan typically already chosen by Handover Decision"];
	bsc -> bsc_gscon [label="GSCON_EV_HO_START (intra-BSC)"];
	bsc_lchan <- bsc_gscon [label="lchan_activate(lchan, FOR_HANDOVER)"];
	bsc_lchan rbox bsc_lchan [label="Continue at\nlchan_activate()\n"];
	|||;
	|||;

	bts rbox mgw_msc [label="Channel Request from intra-BSC-MT-Handover"];
	bsc_gscon <- mgw_msc [label="BSSMAP Handover Request"];
	bsc_gscon box bsc_gscon [label="lchan_select_by_chan_mode(chan_mode)"];
	bsc box bsc [label="lchan_activate(lchan, FOR_HANDOVER)"];
	bsc_lchan rbox bsc_lchan [label="Continue at\nlchan_activate()\n"];
	|||;
	|||;
	bts rbox mgw_msc [label="lchan_activate()"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_\nWAIT_TS_READY\n(timeout: ? s, Tnnnn)"];
	|||;
	|||;
	--- [label="TCH?"];
	bsc_lchan note bsc_gscon [label="This is skipped when FOR_MS_CHANNEL_REQUEST. If the MS requests
		a TCH lchan, and we end up actually giving it a TCH because no SDCCH are available, we
		can not set up an RTP stream because there is not even an L3 conn yet."];
	bsc_lchan note bsc_gscon [label="The lchan FSM asks the conn FSM to have an MGW endpoint ready as
		early as possible. Either the conn already has such MGW endpoint from a previous lchan,
		in which case it immediately replies, or it requests one from the MGW, in which case we
		wait for a response in 'TCH? (2)' below."];
	bsc_lchan -> bsc_gscon [label="GSCON_EV_ENSURE_MGW_ENDPOINT"];
	--- [label="IF conn has user_plane.fi_bts in state ST_READY"];
	bsc_lchan <- bsc_gscon [label="LCHAN_EV_MGW_ENDPOINT_AVAILABLE"];
	bsc_lchan box bsc_lchan [label="mgw_endpoint_available = true"];
	bsc_lchan note bsc_lchan [label="lchan_activate() continues"];
	--- [label="ELSE (no MGW endpoint available yet)"];
	bsc_gscon => mgw_msc [label="CRCX (for BTS) via mgcp_conn_create()"];
	bsc_gscon abox bsc_gscon [label="ST_WAIT_CRCX_BTS\n(timeout: ? s, Tnnnn)"];
	bsc_lchan <- bsc_gscon [label="(event dispatch returns)"];
	bsc_lchan note bsc_lchan [label="lchan_activate() continues"];
	...;
	bsc_gscon note bsc_gscon [label="async:"];
	bsc_gscon <= mgw_msc [label="CRCX OK (for BTS)"];
	bsc_lchan <- bsc_gscon [label="LCHAN_EV_MGW_ENDPOINT_AVAILABLE"];
	bsc_lchan box bsc_lchan [label="mgw_endpoint_available = true"];
	bsc_lchan note bsc_lchan [label="As soon as we reach LCHAN_ST_WAIT_MGW_ENDPOINT_AVAILABLE, this triggers
		immedate action (s.b.), but until then, only the flag gets set to true."];
	...;
	--- [label="CRCX timeout"];
	bsc_gscon note bsc_gscon [label="conn FSM should fire on CRCX timeout"];
	bsc_lchan <- bsc_gscon [label="LCHAN_EV_MGW_ENDPOINT_ERROR"];
	bsc_gscon note bsc_gscon [label="conn FSM should not assume anything and wait for
		GSCON_EV_LCHAN_ALLOC_ERROR"];
	bsc_lchan rbox bsc_lchan [label="Do 'On any error'"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_UNUSED"];
	bsc_ts <- bsc_lchan [label="TS_EV_LCHAN_UNUSED"];
	--- [label="END: 'TCH?'"];
	|||;
	|||;

	bsc_lchan box bsc_lchan [label="lchan_activate() exits"];
	bsc_lchan note bsc_lchan [label="still in\nlchan_request()\nLCHAN_ST_WAIT_\nTS_READY"];
	bsc_ts <- bsc_lchan [label="TS_EV_LCHAN_REQUESTED"];
	...;
	--- [label="on error from TS or timeout:"];
	bsc_ts -> bsc_lchan [label="LCHAN_EV_TS_ERROR"];
	bsc_lchan rbox bsc_lchan [label="Do 'On any error'"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_UNUSED"];
	bsc_ts <- bsc_lchan [label="TS_EV_LCHAN_UNUSED"];
	---;
	...;
	bsc_ts abox bsc_ts [label="TS_ST_IN_USE"];
	bsc_ts -> bsc_lchan [label="LCHAN_EV_TS_READY"];
	bsc_lchan box bsc_lchan [label="lchan_fsm_\npre_lchan_activ()"];

	|||;
	|||;
	bts rbox mgw_msc [label="mode FOR_MS_CHANNEL_REQUEST"];
	bts note bsc_lchan [label="This is the simple case where the MS requested a channel, and there is no
		L3 conn to the MSC; no matter if this is SDDCH or a TCH channel type, we will not prepare
		an RTP stream."];

	bsc_lchan note bsc_lchan [label="still in lchan_fsm_\npre_lchan_activ()"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_WAIT_\nACTIV_ACK\n(timeout: ? s, Tnnnn)"];
	bts <= bsc_lchan [label="RSL Chan Activ (RSL_ACT_INTRA_IMM_ASS)"];
	bts note bsc_lchan [label="If any errors occur from now on, we don't want to send an RR Immediate
		Assignment Reject anymore."];
	bsc_lchan box bsc_lchan [label="sent_chan_activ = true"];
	...;
	--- [label="on timeout"];
	bsc_lchan rbox bsc_lchan [label="Continue at: 'On any error', 'unrecoverable'"];
	---;
	...;
	bts => bsc_lchan [label="RSL Chan Activ ACK"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_WAIT_\nRLL_ESTABLISH\nT3101"];
	bsc_lchan note bsc_lchan [label="Now the lchan is assigned, but has no L3 conn yet. On errors,
		this will either go into graceful release or into broken state, but will not trigger any
		events to a (non-existing) conn."];
	...;
	--- [label="on timeout"];
	bts <= bsc_lchan [label="RSL RF Channel Release"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_WAIT_RF_RELEASE_ACK\n(T?, 4s)"];
	---;
	...;
	bts => bsc_lchan [label="RLL Establish Ind"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_ACTIVE"];
	|||;
	|||;
	bts rbox mgw_msc [label="modes FOR_ASSIGNMENT and FOR_HANDOVER"];

	bsc_lchan note bsc_lchan [label="still in lchan_fsm_\npre_lchan_activ()"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_WAIT_\nACTIV_ACK\n(timeout: ? s, Tnnnn)"];
	bts <= bsc_lchan [label="RSL Chan Activ (RSL_ACT_INTRA_NORM_ASS)",ID=FOR_ASSIGNMENT];
	bts <= bsc_lchan [label="RSL Chan Activ (RSL_ACT_INTER_ASYNC)",ID=FOR_HANDOVER];
	...;
	--- [label="on timeout"];
	bsc_lchan rbox bsc_lchan [label="Continue at: 'On any error', 'unrecoverable'"];
	---;
	bts => bsc_lchan [label="RSL Chan Activ ACK"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_WAIT_\nRLL_ESTABLISH\nT3101"];
	...;
	--- [label="on timeout"];
	bsc_lchan -> bsc_gscon [label="GSCON_EV_LCHAN_ALLOC_ERROR"];
	bsc_lchan -> bsc_lchan [label="lchan_fsm_pre_rf_release()"];
	---;
	...;
	bts => bsc_lchan [label="RLL Establish Indication"];
	|||;

	--- [label="TCH? (2)"];
	--- [label="mgw_endpoint_available == false?"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_WAIT_\nMGW_ENDPOINT_\nAVAILABLE"];
	bsc_lchan note bsc_lchan [label="rely on conn FSM timeout; apply only a long sanity timeout."];
	...;
	bsc_gscon <= mgw_msc [label="CRCX OK (for BTS)"];
	bsc_lchan <- bsc_gscon [label="LCHAN_EV_MGW_ENDPOINT_AVAILABLE"];
	bsc_lchan box bsc_lchan [label="mgw_endpoint_available = true"];
	bsc_lchan <- bsc_lchan [label="re-invoke lchan_fsm_pre_lchan_activ()"];
	--- [label="END: 'TCH? (2)'"];
	|||;

	--- [label="is BTS using IPA Abis? (osmo-bts, ip.access)"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_WAIT_\nIPACC_CRCX_ACK\n(timeout: ? s, Tnnnn)"];
	bts <= bsc_lchan [label="IPACC CRCX"];
	...;
	--- [label="on timeout"];
	bsc_lchan -> bsc_gscon [label="GSCON_EV_LCHAN_ALLOC_ERROR"];
	bsc_lchan -> bsc_lchan [label="lchan_graceful_release()"];
	---;
	...;
	bts => bsc_lchan [label="IPACC CRCX ACK"];
	bts note bsc_lchan [label="The IPACC CRCX ACK tells us what port the IPA Abis based BTS has
		assigned to this lchan. AoIP: we need to forward this to the MGW (BTS side) with an MDCX;
		SCCPlite: we forward this to the MSC during BSSMAP Assignment Complete (TODO: is this
		correct??)"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_WAIT_\nIPACC_MDCX_ACK\n(timeout: ? s, Tnnnn)"];
	bts <= bsc_lchan [label="IPACC MDCX"];
	bts note bsc_lchan [label="The IPACC MDCX tells IPA Abis based BTSes the IP address and RTP port
		assigned by the BTS side of the MGW. AoIP: the MGW CRCX (BTS) must thus happen before
		this; SCCPlite: the RTP port is already known from the timeslot+multiplex information."];
	...;
	--- [label="on timeout"];
	bsc_lchan -> bsc_gscon [label="GSCON_EV_LCHAN_ALLOC_ERROR"];
	bsc_lchan -> bsc_lchan [label="lchan_graceful_release()"];
	---;
	...;
	bts => bsc_lchan [label="IPACC MDCX ACK"];
	--- [label="END: is BTS using IPA Abis? (osmo-bts, ip.access)"];
	|||;
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_ACTIVE"];
	bsc_lchan box bsc_lchan [label="lchan_fsm_post_lchan_activ()"];
	bsc_lchan -> bsc_gscon [label="GSCON_EV_LCHAN_ACTIVE"];
	bts <= bsc_gscon [label="RR Assignment",ID="BSSMAP Assignment Request"];
	bts <= bsc_gscon [label="RR Handover Command",ID="intra-BSC HO"];
	bsc_gscon => mgw_msc [label="BSSMAP Handover\nRequest Acknowledge",ID="inter-BSC-MT HO"];
	...;
	---[label="On error"];
	bsc_lchan rbox bsc_lchan [label="Continue at 'When the lchan is no longer used'"];
	---;
	...;

	bts => bsc_gscon [label="RR Assignment Complete",ID="BSSMAP Assignment Request"];
	bts => bsc_gscon [label="RR Handover Detect",ID="intra-BSC HO"];
	bts => bsc_gscon [label="RR Handover Accept",ID="inter-BSC-MT HO"];
	bsc_gscon note bsc_gscon [label="conn FSM takes care of MGW endpoints for BTS side (possibly
		redirect) and MSC side (possibly create). More information in e.g. assignment.msc and
		handover.msc"];

	...;
	...;
	...;

	bts rbox mgw_msc [label="When the lchan is no longer used"];
	--- [label="IF the MS or BTS release the lchan"];
	bts -> bsc_lchan [label="RLL Release Ind for SAPI=0"];
	--- [label="IF the BSC other than the conn FSM decides to release"];
	bsc -> bsc_lchan [label="LCHAN_EV_RELEASE"];
	--- [label="IF the MSC or conn FSM release the lchan"];
	bsc_lchan <- bsc_gscon [label="LCHAN_EV_RELEASE"];
	---;
	bsc note bsc_gscon [label="The LCHAN_EV_RELEASE's data pointer possibly indicates an error
		cause"];
	bsc_lchan note bsc_gscon [label="If the conn FSM requested a release, it probably has already
	forgotten about this lchan. However, if the MS/BTS initiated the release, make sure the conn FSM
	is informed:"];
	bsc_lchan box bsc_lchan [label="lchan_graceful_release()"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_WAIT_\nSAPIS_RELEASED\nT3109"];
	--- [label="TCH and got as far as Chan Activ Ack?"];
	bts <= bsc_lchan [label="RSL Deactivate SACCH"];
	---;
	bts <= bsc_lchan [label="RLL Release Request (Local End)..."];
	bts <= bsc_lchan [label="...for all SAPIs except [0]"];
	|||;
	--- [label="SAPI[0] in use?"];
	bsc_lchan note bsc_lchan [label="for bts->nokia.no_loc_rel_cnf we do not expect Release Confirm
		messages and this state immediately advances to lchan_fsm_pre_rf_release()"];
	...;
	--- [label="on timeout"];
	bsc_lchan box bsc_lchan [label="Anyway try RF Channel Release, continue
		with lchan_fsm_wait_before_rf_release()"];
	---;
	...;
	bts => bsc_lchan [label="RLL Release Confirm..."];
	bts => bsc_lchan [label="...for each SAPI except [0]"];
	bsc_lchan box bsc_lchan [label="Stay in\nLCHAN_ST_WAIT_\nSAPIS_RELEASED\nuntil only SAPI[0] remains active"];
	--- [label="END: 'SAPI[0] in use?'"];
	|||;

	bsc_lchan box bsc_lchan [label="lchan_fsm_wait_before_rf_release()"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_WAIT_\nBEFORE_RF_RELEASE\nT3111"];
	bsc_lchan -> bsc_gscon [label="GSCON_EV_FORGET_LCHAN (data=lchan)"];
	bsc_gscon note bsc_gscon [label="conn FSM immediately forgets about the lchan"];
	bsc_gscon => mgw_msc [label="BSSMAP Clear Request?"];
	...;
	bsc_lchan box bsc_lchan [label="T3111 expires"];
	bsc_lchan box bsc_lchan [label="lchan_fsm_pre_rf_release()"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_WAIT_\nRF_RELEASE_ACK\nT3111"];
	bsc_lchan box bsc_lchan [label="for each bsc_rll_req matching this lchan: disable timer, call
		cb(BSC_RLLR_IND_REL_IND)"];
	bts <= bsc_lchan [label="RSL RF Channel Release"];
	...;
	--- [label="on timeout"];
	bsc_lchan rbox bsc_lchan [label="Continue at: 'On any error', 'unrecoverable'"];
	---;
	...;
	bts => bsc_lchan [label="RSL RF Channel Release Ack"];

	bsc_lchan box bsc_lchan [label="lchan_fsm_post_rf_release()"];
	|||;
	--- [label="IF an error cause was indicated on LCHAN_EV_RELEASE"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_WAIT_\nAFTER_ERROR\n(timeout: T3111+2 s, T?)"];
	...;
	bsc_lchan box bsc_lchan [label="timer expires"];
	--- [label="END: 'an error cause was indicated on LCHAN_EV_RELEASE'"];
	|||;
	bsc_lchan box bsc_lchan [label="lchan_fsm_release_complete()"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_UNUSED"];
	bsc_ts <- bsc_lchan [label="TS_EV_LCHAN_UNUSED"];
	bsc_ts abox bsc_ts [label="TS_ST_UNUSED"];
	|||;
	|||;

	bts rbox mgw_msc [label="On any error"];
	|||;
	--- [label="IF FOR_MS_CHANNEL_REQUEST && !sent_chan_activ"];
	bts <= bsc_lchan [label="RR Immediate Assign Reject"];
	|||;
	--- [label="IF FOR_ASSIGNMENT or FOR_HANDOVER"];
	bsc_lchan -> bsc_gscon [label="GSCON_EV_LCHAN_ALLOC_ERROR"];
	bsc_gscon note bsc_gscon [label="conn FSM shall immediately 'forget' the lchan"];
	bsc_gscon => mgw_msc [label="BSSMAP\nAssignment Failure",ID=FOR_ASSIGNMENT];
	bsc_gscon => mgw_msc [label="BSSMAP\nHandover Failure",ID="inter-BSC-MT HO"];
	---;
	|||;
	--- [label="IF unrecoverable error"];
	bsc_lchan abox bsc_lchan [label="LCHAN_ST_BORKEN"];
	bsc_lchan note bsc_lchan [label="The broken state usually stays around
		until the BTS disconnects."];
	...;
	bts note bsc_lchan [label="If an ACK comes in late, for specific BTS models, we may choose to
		'repair' the lchan so that it is usable again."];
	bts -> bsc_lchan [label="Chan Release ACK"];
	bsc_lchan -> bsc_lchan [label="lchan_fsm_post_rf_release()"];
}