blob: 13703d0cf0bb982408caa0f9964ad20e669a7db2 [file] [log] [blame]
Harald Welte0db44132019-10-17 11:09:05 +02001module STP_Tests_M3UA {
2
3/* Osmocom STP test suite in in TTCN-3
4 * (C) 2019 Harald Welte <laforge@gnumonks.org>
5 * All rights reserved.
6 *
7 * Released under the terms of GNU General Public License, Version 2 or
8 * (at your option) any later version.
9 *
10 * SPDX-License-Identifier: GPL-2.0-or-later
11 */
12
13friend module STP_Tests;
14
15import from General_Types all;
16import from Osmocom_Types all;
17import from IPL4asp_Types all;
18
19import from Osmocom_VTY_Functions all;
20
21import from M3UA_Types all;
22import from M3UA_Templates all;
23import from M3UA_CodecPort all;
24import from M3UA_CodecPort_CtrlFunct all;
25
26import from M3UA_Emulation all;
27import from MTP3asp_Types all;
28import from MTP3asp_PortType all;
29
30import from SCCP_Types all;
31import from SCCP_Templates all;
32import from SCCPasp_Types all;
33import from SCCP_Emulation all;
34
35import from STP_Tests_Common all;
36
Harald Welte829dac42019-11-05 16:55:30 +010037private const integer NR_M3UA := 3; /* number of M3UA clients in ATS */
38private const integer NR_M3UA_SRV := 1; /* number of M3UA servres in ATS */
39
Harald Welte1c4e9842021-02-07 23:03:28 +010040modulepar {
41 /* STP-side IP addresses */
42 HostList mp_stp_m3ua_ip := { "127.0.0.1", "::1" };
43 /* local IP addresses */
44 HostList mp_local_m3ua_ip := { "127.0.0.1", "::1" };
45 M3uaConfigs mp_m3ua_configs := {
46 /* as-sender: One ASP within AS */
47 {
48 remote_sctp_port := 2905,
49 local_sctp_port := 9999,
50 point_code := 23,
51 routing_ctx := 1023
52 },
53 /* as-receiver: Two ASP within AS */
54 {
55 remote_sctp_port := 2905,
56 local_sctp_port := 10000,
57 point_code := 42,
58 routing_ctx := 1042
59 }, {
60 remote_sctp_port := 2905,
61 local_sctp_port := 10001,
62 point_code := 42,
63 routing_ctx := 1042
64 },
65 /* as-client: One ASP within AS */
66 {
67 remote_sctp_port := 2906,
68 local_sctp_port := 10002,
69 point_code := 55,
70 routing_ctx := 1055
71 }
72 };
73 integer mp_recovery_timeout_msec := 2000;
74 charstring mp_sccp_service_type := "mtp3_itu";
Harald Welte30cb4002021-05-13 20:51:37 +020075
76 /* does the STP support the 'asp quirk' command? */
77 boolean mp_stp_has_asp_quirk := true;
Harald Welte1c4e9842021-02-07 23:03:28 +010078}
79
80type record M3uaConfig {
81 /* STP-side SCTP port for M3UA */
82 integer remote_sctp_port,
83 /* local M3UA base port on TTCN3 side */
84 integer local_sctp_port,
85 /* point code routed via this M3U */
86 integer point_code,
87 /* associated routing context */
88 integer routing_ctx
89};
90type record length (NR_M3UA+NR_M3UA_SRV) of M3uaConfig M3uaConfigs;
91
Harald Welte829dac42019-11-05 16:55:30 +010092private function M3UA_SRV(integer idx) return integer {
93 return NR_M3UA+idx;
94}
Harald Welte0db44132019-10-17 11:09:05 +020095
96type component RAW_M3UA_CT extends Test_CT {
Harald Welte829dac42019-11-05 16:55:30 +010097 port M3UA_CODEC_PT M3UA[NR_M3UA+NR_M3UA_SRV];
98 var integer g_m3ua_conn_id[NR_M3UA+NR_M3UA_SRV];
Harald Welte0db44132019-10-17 11:09:05 +020099}
100
101private template PortEvent tr_SctpAssocChange := {
102 sctpEvent := {
103 sctpAssocChange := ?
104 }
105}
106private template PortEvent tr_SctpPeerAddrChange := {
107 sctpEvent := {
108 sctpPeerAddrChange := ?
109 }
110}
Harald Welte829dac42019-11-05 16:55:30 +0100111private template PortEvent tr_SctpConnOpened := {
112 connOpened := ?
113}
114
Harald Welte0db44132019-10-17 11:09:05 +0200115
116private altstep as_m3ua_sctp() runs on RAW_M3UA_CT {
117 [] any from M3UA.receive(tr_SctpAssocChange) { repeat; }
118 [] any from M3UA.receive(tr_SctpPeerAddrChange) { repeat; }
119}
120
Harald Welte42bcc442021-02-07 21:26:30 +0100121private altstep as_m3ua_ssnm_ignore() runs on RAW_M3UA_CT {
122 var M3UA_RecvFrom rx;
123 [] any from M3UA.receive(t_M3UA_RecvFrom(tr_M3UA_SSNM)) -> value rx {
124 log("Ignoring M3UA SSNM", rx);
125 repeat;
126 }
127}
128
Harald Welte0db44132019-10-17 11:09:05 +0200129friend function f_M3UA_send(integer idx, template (present) PDU_M3UA msg, template integer stream := 0)
130runs on RAW_M3UA_CT {
131 M3UA[idx].send(t_M3UA_Send(g_m3ua_conn_id[idx], msg, stream));
132}
133
134friend function f_M3UA_exp(integer idx, template (present) PDU_M3UA msg) runs on RAW_M3UA_CT {
135 var M3UA_RecvFrom rx;
Harald Welte30812592021-02-08 09:50:46 +0100136 timer T := 5.0;
137 T.start;
Harald Welte0db44132019-10-17 11:09:05 +0200138 alt {
139 [] M3UA[idx].receive(t_M3UA_RecvFrom(msg)) {
140 setverdict(pass);
141 }
142 [] M3UA[idx].receive(t_M3UA_RecvFrom(?)) -> value rx {
143 setverdict(fail, "Received unexpected M3UA[", idx, "] ", rx,
144 "while waiting for ", msg);
145 mtc.stop;
146 }
Harald Welte30812592021-02-08 09:50:46 +0100147 [] T.timeout {
148 setverdict(fail, "Timeout waiting for M3UA[", idx, "] ", msg);
149 mtc.stop;
150 }
Harald Welte0db44132019-10-17 11:09:05 +0200151 }
152}
153
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200154private template (value) Socket ts_Socket(HostName hostName, PortNumber portNumber) := {
155 hostName := hostName,
156 portNumber := portNumber
157};
158
159private template (value) SctpTuple ts_SCTP(template (omit) integer ppid := 3,
160 template (omit) integer stream := 0,
161 template (omit) SocketList remSocks := omit) := {
162 sinfo_stream := stream,
163 sinfo_ppid := ppid,
164 remSocks := remSocks,
165 assocId := omit
166};
167
Harald Welte0db44132019-10-17 11:09:05 +0200168friend function f_M3UA_connect(integer i) runs on RAW_M3UA_CT {
169 var Result res;
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200170 var Option opt_add_local_addrs;
171 var OptionList opt_list := {};
172 var template SocketList opt_add_remote_addrs;
Harald Welte1c4e9842021-02-07 23:03:28 +0100173 var M3uaConfig m3cfg := mp_m3ua_configs[i];
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200174
175 if (lengthof(mp_local_m3ua_ip) == 0 or lengthof(mp_stp_m3ua_ip) == 0) {
176 setverdict(fail, "Empty local or remote address trying to connect SCTP socket: ",
177 mp_local_m3ua_ip, " / ", mp_stp_m3ua_ip);
178 mtc.stop;
179 }
180
181 if (lengthof(mp_local_m3ua_ip) > 1) {
182 opt_add_local_addrs.sctpAdditionalLocalAddresses := substr(mp_local_m3ua_ip, 1,
183 lengthof(mp_local_m3ua_ip) - 1); //{mp_local_m3ua_ip};
184 opt_list := {opt_add_local_addrs};
185 }
186
187 if (lengthof(mp_stp_m3ua_ip) > 1) {
188 for (var integer j := 1; j < lengthof(mp_stp_m3ua_ip); j := j + 1) {
Harald Welte1c4e9842021-02-07 23:03:28 +0100189 var Socket sk := valueof(ts_Socket(mp_stp_m3ua_ip[j], m3cfg.remote_sctp_port));
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200190 opt_add_remote_addrs[j - 1] := sk;
191 }
192 } else {
193 opt_add_remote_addrs := omit;
194 }
195
Harald Welte1c4e9842021-02-07 23:03:28 +0100196 res := M3UA_CodecPort_CtrlFunct.f_IPL4_connect(M3UA[i], mp_stp_m3ua_ip[0], m3cfg.remote_sctp_port,
197 mp_local_m3ua_ip[0], m3cfg.local_sctp_port, 0,
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200198 {sctp:=valueof(ts_SCTP(3, 0, opt_add_remote_addrs))},
199 opt_list);
Harald Welte0db44132019-10-17 11:09:05 +0200200 if (not ispresent(res.connId)) {
201 setverdict(fail, "Could not connect M3UA socket, check your configuration");
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200202 mtc.stop;
Harald Welte0db44132019-10-17 11:09:05 +0200203 }
204 g_m3ua_conn_id[i] := res.connId;
205}
206
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100207friend function f_M3UA_close(integer i) runs on RAW_M3UA_CT {
208 var Result res;
209 res := M3UA_CodecPort_CtrlFunct.f_IPL4_close(M3UA[i], g_m3ua_conn_id[i], {sctp:=valueof(ts_SCTP)});
210 g_m3ua_conn_id[i] := 0;
211}
212
Harald Welte829dac42019-11-05 16:55:30 +0100213friend function f_M3UA_listen(integer i) runs on RAW_M3UA_CT {
214 var Result res;
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200215 var Option opt_add_local_addrs;
216 var OptionList opt_list := {};
Harald Welte1c4e9842021-02-07 23:03:28 +0100217 var M3uaConfig m3cfg := mp_m3ua_configs[i];
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200218
219 if (lengthof(mp_local_m3ua_ip) == 0 ) {
220 setverdict(fail, "Empty local address trying to bind SCTP socket: ",
221 mp_local_m3ua_ip);
222 mtc.stop;
223 }
224
225 if (lengthof(mp_local_m3ua_ip) > 1) {
226 opt_add_local_addrs.sctpAdditionalLocalAddresses := substr(mp_local_m3ua_ip, 1,
227 lengthof(mp_local_m3ua_ip) - 1); //{mp_local_m3ua_ip};
228 opt_list := {opt_add_local_addrs};
229 }
230
Harald Welte1c4e9842021-02-07 23:03:28 +0100231 res := M3UA_CodecPort_CtrlFunct.f_IPL4_listen(M3UA[i], mp_local_m3ua_ip[0], m3cfg.local_sctp_port,
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200232 {sctp:=valueof(ts_SCTP)}, opt_list);
Harald Welte829dac42019-11-05 16:55:30 +0100233 if (not ispresent(res.connId)) {
234 setverdict(fail, "Could not bind M3UA socket, check your configuration");
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200235 mtc.stop;
Harald Welte829dac42019-11-05 16:55:30 +0100236 }
237}
238
Harald Welte27cfb7a2021-02-08 09:58:28 +0100239friend function f_init_m3ua(boolean ignore_ssnm := true) runs on RAW_M3UA_CT {
Harald Welte0db44132019-10-17 11:09:05 +0200240 var integer i;
241
242 f_init_common();
243
244 activate(as_m3ua_sctp());
Harald Welte27cfb7a2021-02-08 09:58:28 +0100245 if (ignore_ssnm) {
246 activate(as_m3ua_ssnm_ignore());
247 }
Harald Welte0db44132019-10-17 11:09:05 +0200248
249 for (i := 0; i < NR_M3UA; i:=i+1) {
250 map(self:M3UA[i], system:M3UA_CODEC_PT);
251 f_M3UA_connect(i);
252 }
253}
254
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100255friend function f_clear_m3ua() runs on RAW_M3UA_CT {
256 var integer i;
257
258 log("Clearing M3UA...");
259
260 for (i := 0; i < NR_M3UA; i:=i+1) {
261 f_M3UA_close(i);
262 }
263 /* Wait for recovery timer to trigger and shutdown all AS: */
264 f_sleep(int2float(mp_recovery_timeout_msec)/1000.0 + 0.5);
265 setverdict(pass, "M3UA cleared");
266}
267
Harald Welte829dac42019-11-05 16:55:30 +0100268friend function f_init_m3ua_srv() runs on RAW_M3UA_CT {
269 var integer i;
270 var PortEvent sctp_evt;
271
272 for (i := NR_M3UA; i < NR_M3UA+NR_M3UA_SRV; i:=i+1) {
273 map(self:M3UA[i], system:M3UA_CODEC_PT);
274 /* bind+ listen */
275 f_M3UA_listen(i);
276 /* wait for accept() */
277 M3UA[i].receive(tr_SctpConnOpened) -> value sctp_evt {
278 g_m3ua_conn_id[i] := sctp_evt.connOpened.connId;
279 }
280 }
281}
282
283
284/***********************************************************************
285 * Test the STP in M3UA SG role (we are ASP)
286 ***********************************************************************/
287
Harald Welte0db44132019-10-17 11:09:05 +0200288/* perform an outbound ASP-UP procedure */
289friend function f_M3UA_asp_up(integer idx, template (omit) OCT4 aspid := omit) runs on RAW_M3UA_CT {
290 f_M3UA_send(idx, ts_M3UA_ASPUP(aspid));
291 f_M3UA_exp(idx, tr_M3UA_ASPUP_ACK);
292}
293
294/* perform an outbound BEAT procedure */
295friend function f_M3UA_beat(integer idx, template (omit) octetstring hbd) runs on RAW_M3UA_CT {
296 if (istemplatekind(hbd, "omit")) {
297 f_M3UA_send(idx, ts_M3UA_BEAT(omit));
298 f_M3UA_exp(idx, tr_M3UA_BEAT_ACK(omit));
299 } else {
300 f_M3UA_send(idx, ts_M3UA_BEAT(ts_M3UA_hb_data(hbd)));
301 f_M3UA_exp(idx, tr_M3UA_BEAT_ACK(tr_M3UA_hb_data(hbd)));
302 }
303}
304
305/* perform an outbound ASP-ACTIVATE procedure */
306friend function f_M3UA_asp_act(integer idx, template (omit) M3UA_Traffic_Mode_Type tmt := omit,
307 template (omit) OCT4 rctx := omit) runs on RAW_M3UA_CT {
308 f_M3UA_send(idx, ts_M3UA_ASPAC(tmt, rctx));
309 f_M3UA_exp(idx, tr_M3UA_ASPAC_ACK(tmt, rctx));
310}
311
Harald Welte27cfb7a2021-02-08 09:58:28 +0100312/* perform an outbound ASP-INACTIVATE procedure */
313friend function f_M3UA_asp_inact(integer idx, template (omit) OCT4 rctx := omit) runs on RAW_M3UA_CT {
314 f_M3UA_send(idx, ts_M3UA_ASPIA(rctx));
315 f_M3UA_exp(idx, tr_M3UA_ASPIA_ACK(rctx));
316}
317
Harald Welte0db44132019-10-17 11:09:05 +0200318/* perform outbound ASP-UP and ASP-ACT, optionally expect interemittent NOTIFY */
319friend function f_M3UA_asp_up_act(integer idx, template (omit) M3UA_Traffic_Mode_Type tmt := omit,
320 template (omit) OCT4 rctx := omit,
321 template (omit) OCT2 ntfy_after_up := c_M3UA_ST_I_AS_INACTIVE,
322 template (omit) OCT2 ntfy_after_act := c_M3UA_ST_I_AS_ACTIVE)
323runs on RAW_M3UA_CT {
324 f_M3UA_asp_up(idx, omit);
325 if (not istemplatekind(ntfy_after_up, "omit")) {
326 f_M3UA_exp(idx, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_up, *));
327 }
328 f_M3UA_asp_act(idx, tmt, rctx);
329 if (not istemplatekind(ntfy_after_act, "omit")) {
330 f_M3UA_exp(idx, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_act, *));
331 }
332}
333
334
335/* Test the ASP-UP procedure */
336testcase TC_connect_asp_up() runs on RAW_M3UA_CT {
337 f_init_m3ua();
338 f_M3UA_asp_up(0);
339 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_INACTIVE, *));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100340 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200341}
342
343/* Test the heartbeat procedure without optional heartbeat data payload */
344testcase TC_beat() runs on RAW_M3UA_CT {
345 f_init_m3ua();
346 f_M3UA_asp_up(0);
347 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_INACTIVE, *));
348 f_M3UA_beat(0, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100349 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200350}
351
352/* Test the heartbeat procedure with optional heartbeat data payload */
353testcase TC_beat_payload() runs on RAW_M3UA_CT {
354 f_init_m3ua();
355 f_M3UA_asp_up(0);
356 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_INACTIVE, *));
357 f_M3UA_beat(0, 'a1a2a3a4a5'O);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100358 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200359}
360
361/* Test the ASP-ACTIVATE procedure (without traffic-mode or routing ctx) */
362testcase TC_asp_act() runs on RAW_M3UA_CT {
363 f_init_m3ua();
364 f_M3UA_asp_up_act(0);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100365 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200366}
367
368/* Test the ASP-ACTIVATE procedure with traffic-mode override */
369testcase TC_asp_act_override() runs on RAW_M3UA_CT {
370 f_init_m3ua();
371 f_M3UA_asp_up_act(0, c_M3UA_TMT_override, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100372 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200373}
374
375/* Test the ASP-ACTIVATE procedure with traffic-mode override */
376testcase TC_asp_act_loadshare() runs on RAW_M3UA_CT {
377 f_init_m3ua();
378 f_M3UA_asp_up_act(0, c_M3UA_TMT_loadshare, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100379 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200380}
381
382/* Test the ASP-ACTIVATE procedure with traffic-mode broadcast */
383testcase TC_asp_act_broadcast() runs on RAW_M3UA_CT {
384 f_init_m3ua();
385 f_M3UA_asp_up_act(0, c_M3UA_TMT_broadcast, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100386 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200387}
388
Harald Welte15a85012020-06-13 16:35:45 +0200389/* test whether the STP accepts M3UA DATA without Routing Context IE */
390testcase TC_act_rctx_data_no_rctx() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100391 var OCT4 rctx_sender := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
392 var OCT4 pc_sender := int2oct(mp_m3ua_configs[0].point_code, 4);
393 var OCT4 rctx_receiver := int2oct(mp_m3ua_configs[1].routing_ctx, 4);
394 var OCT4 pc_receiver := int2oct(mp_m3ua_configs[1].point_code, 4);
Harald Welte15a85012020-06-13 16:35:45 +0200395
396 f_init_m3ua();
397 /* bring up the sender specifying a routing context */
398
399 f_M3UA_asp_up_act(0, rctx := rctx_sender);
400 f_M3UA_asp_up_act(1);
401
402 /* check if DATA is accepted without Routing Context IE */
403 f_test_traffic(0, omit, pc_sender, 1, rctx_receiver, pc_receiver);
404
405 f_clear_m3ua();
406}
407
Harald Welte0db44132019-10-17 11:09:05 +0200408/* Test if traffic is routed from idx_tx/pc_tx to idx_rx/pc_rx */
409private function f_test_traffic(integer idx_tx, template (omit) OCT4 rctx_sender, OCT4 pc_tx,
410 integer idx_rx, template (omit) OCT4 rctx_receiver, OCT4 pc_rx,
411 OCT1 si := '23'O, OCT1 ni := '00'O, OCT1 mp := '00'O, OCT1 sls := '00'O)
412runs on RAW_M3UA_CT {
413 var octetstring data := f_rnd_octstring(f_rnd_int(100));
414 f_M3UA_send(idx_tx, ts_M3UA_DATA(rctx_sender,
415 ts_M3UA_protocol_data(pc_tx, pc_rx, si, ni, mp, sls, data)), 1);
416 f_M3UA_exp(idx_rx, tr_M3UA_DATA(rctx_receiver,
417 tr_M3UA_protocol_data(pc_tx, pc_rx, si, ni, mp, sls, data)));
418}
419
420
421/* test "traffic-mode override" behavior */
422testcase TC_tmt_override() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100423 var OCT4 rctx_sender := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
424 var OCT4 pc_sender := int2oct(mp_m3ua_configs[0].point_code, 4);
425 var OCT4 rctx_receiver := int2oct(mp_m3ua_configs[1].routing_ctx, 4);
426 var OCT4 pc_receiver := int2oct(mp_m3ua_configs[1].point_code, 4);
Harald Welte0db44132019-10-17 11:09:05 +0200427
428 f_init_m3ua();
429
430 /* bring up the 'sender' side (single ASP in AS) */
431 f_M3UA_asp_up_act(0, omit, omit);
432
433 /* activate the first 'receiver' side ASP */
434 f_M3UA_asp_up_act(1, c_M3UA_TMT_override, rctx_receiver);
435
436 /* verify traffic is routed from sender to [sole] receiver */
437 f_test_traffic(0, rctx_sender, pc_sender, 1, rctx_receiver, pc_receiver);
438
439 /* activate the second 'receiver' side ASP (no NOTIFY as AS state doesn't change) */
440 f_M3UA_asp_up_act(2, c_M3UA_TMT_override, rctx_receiver, omit, omit);
441
442 /* we expect a NOTIFY to the *other* ASP Other/Alternat-ASP-Active */
443 f_M3UA_exp(1, tr_M3UA_NOTIFY(c_M3UA_ST_T_OTHER, c_M3UA_ST_I_ALTERNATE_ASP, *));
444
445 /* verify traffic is routed from sender to new receiver */
446 f_test_traffic(0, rctx_sender, pc_sender, 2, rctx_receiver, pc_receiver);
447
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100448 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200449}
450
451private altstep as_count_rx(integer idx, template (present) PDU_M3UA exp, inout integer counter)
452runs on RAW_M3UA_CT {
453 [] M3UA[idx].receive(t_M3UA_RecvFrom(exp)) {
454 counter := counter + 1;
455 }
456}
457
458/* test "traffic-mode load-share" behavior */
459testcase TC_tmt_loadshare() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100460 var OCT4 rctx_sender := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
461 var OCT4 pc_sender := int2oct(mp_m3ua_configs[0].point_code, 4);
462 var OCT4 rctx_receiver := int2oct(mp_m3ua_configs[1].routing_ctx, 4);
463 var OCT4 pc_receiver := int2oct(mp_m3ua_configs[1].point_code, 4);
Harald Welte0db44132019-10-17 11:09:05 +0200464 var integer i;
465
466 f_init_m3ua();
467
468 /* FIXME: configure the STP via VTY to set traffic-mode */
469
470 /* bring up the 'sender' side (single ASP in AS) */
471 f_M3UA_asp_up_act(0, omit, rctx_sender);
472
473 /* activate the first 'receiver' side ASP */
Harald Welte0db44132019-10-17 11:09:05 +0200474 f_M3UA_asp_up_act(1, c_M3UA_TMT_loadshare, omit); // TODO: rctx
475
476 /* verify traffic is routed from sender to [sole] receiver */
477 for (i := 0; i < 10; i := i+1) {
478 f_test_traffic(0, rctx_sender, pc_sender, 1, rctx_receiver, pc_receiver);
479 }
480
481 /* activate the second 'receiver' side ASP (no NOTIFY) */
482 f_M3UA_asp_up_act(2, c_M3UA_TMT_loadshare, omit, omit, omit); // TODO: rctx
483
484 /* verify traffic is routed from sender to new receiver */
485 const integer iter_per_asp := 5;
486 var integer num_rx[NR_M3UA] := { 0, 0, 0 };
487 for (i := 0; i < 2*iter_per_asp; i := i+1) {
488 var octetstring data := f_rnd_octstring(f_rnd_int(100));
489 var template (value) M3UA_Protocol_Data tx_pd;
490 var template (present) M3UA_Protocol_Data rx_pd;
491 tx_pd := ts_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
492 rx_pd := tr_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
493 f_M3UA_send(0, ts_M3UA_DATA(rctx_sender, tx_pd), 1);
494 alt {
Pau Espin Pedrol4d915782020-08-21 12:24:35 +0200495 [] as_count_rx(1, tr_M3UA_DATA(rctx_receiver, rx_pd), num_rx[1]);
496 [] as_count_rx(2, tr_M3UA_DATA(rctx_receiver, rx_pd), num_rx[2]);
Harald Welte0db44132019-10-17 11:09:05 +0200497 }
498 }
499 /* FIXME: check for extraneous messages? */
500 for (i := 1; i <= 2; i := i+1) {
501 if (num_rx[i] != iter_per_asp) {
Harald Welteb9f7bbb2019-10-29 09:07:11 +0100502 setverdict(fail, "Received ", num_rx[i], " out of expected ", iter_per_asp,
503 "M3UA DATA messages at M3UA port ", i);
Harald Welte0db44132019-10-17 11:09:05 +0200504 }
505 }
506 setverdict(pass);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100507
508 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200509}
510
511/* test "traffic-mode broadcast" behavior */
512testcase TC_tmt_broadcast() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100513 var OCT4 rctx_sender := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
514 var OCT4 pc_sender := int2oct(mp_m3ua_configs[0].point_code, 4);
515 var OCT4 rctx_receiver := int2oct(mp_m3ua_configs[1].routing_ctx, 4);
516 var OCT4 pc_receiver := int2oct(mp_m3ua_configs[1].point_code, 4);
Harald Welte0db44132019-10-17 11:09:05 +0200517 var integer i;
518
519 f_init_m3ua();
520
521 /* FIXME: configure the STP via VTY to set traffic-mode */
522
523 /* bring up the 'sender' side (single ASP in AS) */
524 f_M3UA_asp_up_act(0, omit, omit); // TODO: rctx
525
526 /* activate the first 'receiver' side ASP */
Harald Welte0db44132019-10-17 11:09:05 +0200527 f_M3UA_asp_up_act(1, c_M3UA_TMT_broadcast, omit); // TODO: rctx
528
529 /* verify traffic is routed from sender to [sole] receiver */
530 for (i := 0; i < 10; i := i+1) {
531 f_test_traffic(0, rctx_sender, pc_sender, 1, rctx_receiver, pc_receiver);
532 }
533
534 /* activate the second 'receiver' side ASP */
535 f_M3UA_asp_up_act(2, c_M3UA_TMT_broadcast, omit, omit, omit); // TODO: rctx
536
537 /* verify traffic is routed from sender to new receiver */
538 for (i := 0; i < 10; i := i+1) {
539 var octetstring data := f_rnd_octstring(f_rnd_int(100));
540 var template (value) M3UA_Protocol_Data tx_pd;
541 var template (present) M3UA_Protocol_Data rx_pd;
542 tx_pd := ts_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
543 rx_pd := tr_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
544 f_M3UA_send(0, ts_M3UA_DATA(rctx_sender, tx_pd), 1);
545 /* each message must be received both on 1 and 2 */
546 f_M3UA_exp(1, tr_M3UA_DATA(rctx_receiver, rx_pd));
547 f_M3UA_exp(2, tr_M3UA_DATA(rctx_receiver, rx_pd));
548 }
549 setverdict(pass);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100550
551 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200552}
553
554private function f_M3UA_rkm_register(OCT4 id, OCT3 dpc, OCT4 rctx,
555 template (present) OCT4 exp_status := c_M3UA_REGSTS_SUCCESS)
556runs on RAW_M3UA_CT
557{
558 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:=id, dpc:=dpc, rctx:=rctx)}));
559 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:=id, status:=exp_status, rctx:=rctx)}));
560}
561
562/* Send RKM registration; expect -EPERM as RCTX doesn't match config and dynamic not permitted */
563testcase TC_rkm_reg_static_notpermitted() runs on RAW_M3UA_CT {
564 f_init_m3ua();
565
566 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:='00000099'O, dpc:='aabbcc'O)}));
567 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:='00000099'O, status:=c_M3UA_REGSTS_ERR_EPERM,
568 rctx:=?)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100569
570 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200571}
572
573/* Send RKM registration; expect OK as RCTX does match config */
574testcase TC_rkm_reg_static_permitted() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100575 var OCT3 dpc := int2oct(mp_m3ua_configs[0].point_code, 3); // must match config
576 var OCT4 rctx := int2oct(mp_m3ua_configs[0].routing_ctx, 4); // must match config
Harald Welte0db44132019-10-17 11:09:05 +0200577
578 f_init_m3ua();
579
580 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:='10000099'O, dpc:=dpc, rctx:=rctx)}));
581 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:='10000099'O, status:=c_M3UA_REGSTS_SUCCESS,
582 rctx:=rctx)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100583
584 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200585}
586
587/* Send RKM registration; expect OK as dynamic not permitted */
588testcase TC_rkm_reg_dynamic_permitted() runs on RAW_M3UA_CT {
589 f_init_common();
590 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation dynamic-permitted");
591 f_init_m3ua();
592
593 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:='20000099'O, dpc:='aabbcc'O)}));
594 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:='20000099'O, status:=c_M3UA_REGSTS_SUCCESS,
595 rctx:=?)}));
596
597 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation static-only");
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100598
599 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200600}
601
602/* try to de-register a routing key that was never registered -> error */
603testcase TC_rkm_unreg_never_registered() runs on RAW_M3UA_CT {
604 f_init_m3ua();
Harald Welte1c4e9842021-02-07 23:03:28 +0100605 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(int2oct(mp_m3ua_configs[0].routing_ctx,4))));
Harald Welte0db44132019-10-17 11:09:05 +0200606 f_M3UA_exp(0, tr_M3UA_DEREG_RSP({tr_M3UA_dereg_res(?,c_m3UA_DEREGSTS_ERR_NOT_REG)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100607 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200608}
609
610/* try to de-register a routing key that is invalid (non-existant) -> error */
611testcase TC_rkm_unreg_invalid() runs on RAW_M3UA_CT {
612 f_init_m3ua();
613 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(int2oct(1234,4))));
614 f_M3UA_exp(0, tr_M3UA_DEREG_RSP({tr_M3UA_dereg_res(?,c_m3UA_DEREGSTS_ERR_INVAL_RCTX)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100615 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200616}
617
Oliver Smith326d78e2021-07-23 10:42:10 +0200618/* try to de-register a routing key that was registered -> OK */
Harald Welte0db44132019-10-17 11:09:05 +0200619testcase TC_rkm_unreg_registered() runs on RAW_M3UA_CT {
Oliver Smith326d78e2021-07-23 10:42:10 +0200620 var OCT3 dpc := int2oct(123, 3);
621 var OCT4 rctx := int2oct(1234, 4);
622
623 f_init_common();
624 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation dynamic-permitted");
Harald Welte0db44132019-10-17 11:09:05 +0200625 f_init_m3ua();
Oliver Smith326d78e2021-07-23 10:42:10 +0200626
627 /* first register the routing key */
628 f_M3UA_rkm_register(id:='30000099'O, dpc:=dpc, rctx:=rctx);
629
630 /* then try to de-register */
631 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(rctx)));
Harald Welte0db44132019-10-17 11:09:05 +0200632 f_M3UA_exp(0, tr_M3UA_DEREG_RSP({tr_M3UA_dereg_res(?,c_m3UA_DEREGSTS_SUCCESS)}));
Oliver Smith326d78e2021-07-23 10:42:10 +0200633
634 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation static-only");
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100635 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200636}
637
638/* try to de-register a routing key for an active ASP -> ERROR */
639testcase TC_rkm_unreg_active() runs on RAW_M3UA_CT {
Oliver Smithee022232021-07-23 11:01:06 +0200640 var OCT3 dpc := int2oct(123, 3);
641 var OCT4 rctx := int2oct(1234, 4);
Harald Welte0db44132019-10-17 11:09:05 +0200642
Oliver Smithee022232021-07-23 11:01:06 +0200643 f_init_common();
644 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation dynamic-permitted");
Harald Welte0db44132019-10-17 11:09:05 +0200645 f_init_m3ua();
646
647 /* first register the routing key */
648 f_M3UA_rkm_register(id:='30000099'O, dpc:=dpc, rctx:=rctx);
649
650 /* then activate the ASP */
651 f_M3UA_asp_up_act(0);
Oliver Smithee022232021-07-23 11:01:06 +0200652 f_M3UA_exp(0, tr_M3UA_DAVA({*}, rctx));
653 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_ACTIVE, *));
654 f_M3UA_exp(0, tr_M3UA_DAVA({*}, *));
Harald Welte0db44132019-10-17 11:09:05 +0200655
Oliver Smithee022232021-07-23 11:01:06 +0200656 /* then try to de-register -> ERR_ASP_ACTIVE */
Harald Welte0db44132019-10-17 11:09:05 +0200657 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(rctx)));
658 f_M3UA_exp(0, tr_M3UA_DEREG_RSP({tr_M3UA_dereg_res(?,c_m3UA_DEREGSTS_ERR_ASP_ACTIVE)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100659
Oliver Smithee022232021-07-23 11:01:06 +0200660 /* deactivate ASP and properly de-register to clean up */
661 f_M3UA_asp_inact(0);
662 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(rctx)));
663 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_PENDING, *));
664 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_PENDING, *));
665 f_M3UA_exp(0, tr_M3UA_DEREG_RSP({tr_M3UA_dereg_res(?,c_m3UA_DEREGSTS_SUCCESS)}));
666
667 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation static-only");
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100668 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200669}
670
Harald Welte829dac42019-11-05 16:55:30 +0100671/***********************************************************************
672 * Test the STP in M3UA ASP role (we are SG)
673 ***********************************************************************/
674
675/* expect/perform an inbound ASP-UP procedure */
676friend function f_M3UA_CLNT_asp_up(integer idx, template OCT4 aspid := omit) runs on RAW_M3UA_CT {
677 f_M3UA_exp(idx, tr_M3UA_ASPUP(aspid));
678 f_M3UA_send(idx, ts_M3UA_ASPUP_ACK);
679}
680
681/* expect/perform an inbound ASP-ACTIVATE procedure */
682friend function f_M3UA_CLNT_asp_act(integer idx, template M3UA_Traffic_Mode_Type tmt := omit,
683 template (omit) OCT4 rctx := omit) runs on RAW_M3UA_CT {
684 f_M3UA_exp(idx, tr_M3UA_ASPAC(tmt, rctx));
685 f_M3UA_send(idx, ts_M3UA_ASPAC_ACK(tmt, rctx));
686}
687
688/* expect/perform inbound ASP-UP and ASP-ACT, optionally send interemittent NOTIFY */
689friend function f_M3UA_CLNT_asp_up_act(integer idx, template M3UA_Traffic_Mode_Type tmt := omit,
690 template OCT4 rctx := omit,
691 template (omit) OCT2 ntfy_after_up := c_M3UA_ST_I_AS_INACTIVE,
692 template (omit) OCT2 ntfy_after_act := c_M3UA_ST_I_AS_ACTIVE)
693runs on RAW_M3UA_CT {
694 f_M3UA_CLNT_asp_up(idx, omit);
695 if (not istemplatekind(ntfy_after_up, "omit")) {
696 f_M3UA_send(idx, ts_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_up, rctx));
697 }
698 f_M3UA_CLNT_asp_act(idx, tmt, rctx);
699 if (not istemplatekind(ntfy_after_act, "omit")) {
700 f_M3UA_send(idx, ts_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_act, rctx));
701 }
702}
703
704
705/* Expect inbound connection from ASP/SCTP-client, followed by ASP-UP */
706testcase TC_clnt_connect_asp_up() runs on RAW_M3UA_CT {
707 f_init_m3ua();
708 f_init_m3ua_srv();
709
710 f_M3UA_CLNT_asp_up(M3UA_SRV(0));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100711
712 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100713}
714
715/* Expect inbound connection from ASP/SCTP-client, followed by ASP-UP + ASP-ACT */
716testcase TC_clnt_asp_act() runs on RAW_M3UA_CT {
717 f_init_m3ua();
718 f_init_m3ua_srv();
719
Harald Welte1c4e9842021-02-07 23:03:28 +0100720 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100721
722 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100723}
724
Harald Welte53050cd2019-12-01 20:49:01 +0100725/* Expect inbound connection from ASP/SCTP-client, followed by ASP-UP + ASP-ACT */
726testcase TC_clnt_asp_act_tmt_loadshare() runs on RAW_M3UA_CT {
727 f_init_common();
728 f_vty_config2(VTY, {"cs7 instance 0", "as as-client m3ua"}, "traffic-mode loadshare");
729 f_init_m3ua();
730 f_init_m3ua_srv();
731
Harald Welte1c4e9842021-02-07 23:03:28 +0100732 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), tmt := c_M3UA_TMT_loadshare, rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4));
Harald Welte53050cd2019-12-01 20:49:01 +0100733
734 f_clear_m3ua();
735}
736
Harald Welte829dac42019-11-05 16:55:30 +0100737/* Test traffic being routed through "server" side STP (M3UA SG), coming back in "client"
738 * side STP (M3UA ASP) */
739testcase TC_clnt_sg_to_asp() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100740 var OCT4 rctx_sender := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
741 var OCT4 pc_sender := int2oct(mp_m3ua_configs[0].point_code, 4);
742 var OCT4 rctx_receiver := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
743 var OCT4 pc_receiver := int2oct(mp_m3ua_configs[M3UA_SRV(0)].point_code, 4);
Harald Welte829dac42019-11-05 16:55:30 +0100744
745 f_init_m3ua();
746 f_M3UA_asp_up_act(0);
747
748 f_init_m3ua_srv();
Harald Welte1c4e9842021-02-07 23:03:28 +0100749 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4));
Harald Welte829dac42019-11-05 16:55:30 +0100750
751 f_sleep(1.0);
752
753 /* verify traffic is routed from sender to [sole] receiver */
754 f_test_traffic(0, rctx_sender, pc_sender, M3UA_SRV(0), rctx_receiver, pc_receiver);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100755
756 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100757}
758
759/* Test traffic being routed through "client" side STP (M3UA ASP), coming back in "server"
760 * side STP (M3UA SG) */
761testcase TC_clnt_asp_to_sg() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100762 var OCT4 rctx_sender := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
763 var OCT4 pc_sender := int2oct(mp_m3ua_configs[M3UA_SRV(0)].point_code, 4);
764 var OCT4 rctx_receiver := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
765 var OCT4 pc_receiver := int2oct(mp_m3ua_configs[0].point_code, 4);
Harald Welte829dac42019-11-05 16:55:30 +0100766
767 f_init_m3ua();
768 f_M3UA_asp_up_act(0);
769
770 f_init_m3ua_srv();
Harald Welte1c4e9842021-02-07 23:03:28 +0100771 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4));
Harald Welte829dac42019-11-05 16:55:30 +0100772
773 f_sleep(1.0);
774
775 /* verify traffic is routed from sender to [sole] receiver */
776 f_test_traffic(M3UA_SRV(0), rctx_sender, pc_sender, 0, rctx_receiver, pc_receiver);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100777 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100778}
779
Harald Welte27cfb7a2021-02-08 09:58:28 +0100780/* Test if ASPAC / ASPIA of one ASP generates DAVA / DUNA on other ASP */
781testcase TC_ssnm_aspac_dava_aspia_duna() runs on RAW_M3UA_CT {
782 var OCT4 rctx0 := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
783
784 f_init_m3ua(ignore_ssnm := false);
785 /* activate the first ASP */
786 f_M3UA_asp_up_act(0);
787
788 /* activate the second ASP */
789 f_M3UA_asp_up_act(1, c_M3UA_TMT_override, omit);
790 /* expect DAVA for PC of second ASP on first ASP */
791 f_M3UA_exp(0, tr_M3UA_DAVA({ts_M3UA_PC(mp_m3ua_configs[1].point_code, 0)}, rctx0));
792 /* TODO: expect no DAVA on second ASP */
793
794 /* deactivate the second ASP */
795 f_M3UA_asp_inact(1);
796 /* expect DUNA for PC of second ASP on first ASP */
797 f_M3UA_exp(0, tr_M3UA_DUNA({ts_M3UA_PC(mp_m3ua_configs[1].point_code, 0)}, rctx0));
798 /* TODO: expect no DUNA on second ASP */
799
800 f_clear_m3ua();
801}
802
803/* Test if DAVA/DUNA sent from SG to ASP-role STP gets forwarded to other ASP */
804testcase TC_ssnm_distribution_dava_duna() runs on RAW_M3UA_CT {
805 var OCT4 rctx0 := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
806 var OCT4 rctxS0 := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
807 /* some random point code whose availability we advertise */
808 var template (value) M3UA_Point_Code adv_pc := ts_M3UA_PC(1234, 0);
809
810 f_init_m3ua(ignore_ssnm := false);
811
812 /* activate the first ASP */
813 f_M3UA_asp_up_act(0);
814
815 /* activate SG-role ASP (ASP on STP) */
816 f_init_m3ua_srv();
817 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4));
818
819 /* transmit a DAVA to the remote ASP */
820 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DAVA({adv_pc}, rctxS0));
821 /* expect that to show up on other ASP */
822 f_M3UA_exp(0, tr_M3UA_DAVA({adv_pc}, rctx0));
823
824 /* transmit a DUNA to the remote ASP */
825 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DUNA({adv_pc}, rctxS0));
826 /* expect that to show up on other ASP */
827 f_M3UA_exp(0, tr_M3UA_DUNA({adv_pc}, rctx0));
828}
829
830/* Test if DAVA/DUNA sent from SG to ASP-role STP gets forwarded to other ASP */
831testcase TC_ssnm_distribution_dava_duna_multipc() runs on RAW_M3UA_CT {
832 var OCT4 rctx0 := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
833 var OCT4 rctxS0 := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
834 /* some random point code whose availability we advertise */
835 var template (value) M3UA_Point_Codes adv_pcs := { ts_M3UA_PC(1234, 0), ts_M3UA_PC(5678, 0) };
836
837 f_init_m3ua(ignore_ssnm := false);
838
839 /* activate the first ASP */
840 f_M3UA_asp_up_act(0);
841
842 /* activate SG-role ASP (ASP on STP) */
843 f_init_m3ua_srv();
844 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4));
845
846 /* transmit a DAVA to the remote ASP */
847 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DAVA(adv_pcs, rctxS0));
848 /* expect that to show up on other ASP */
849 f_M3UA_exp(0, tr_M3UA_DAVA(adv_pcs, rctx0));
850
851 /* transmit a DUNA to the remote ASP */
852 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DUNA(adv_pcs, rctxS0));
853 /* expect that to show up on other ASP */
854 f_M3UA_exp(0, tr_M3UA_DUNA(adv_pcs, rctx0));
855}
856
857/* Test if DUPU sent from SG to ASP-role STP gets forwarded to other ASP */
858testcase TC_ssnm_distribution_dupu() runs on RAW_M3UA_CT {
859 var OCT4 rctx0 := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
860 var OCT4 rctxS0 := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
861 /* some random point code whose availability we advertise */
862 var template (value) M3UA_Point_Code adv_pc := ts_M3UA_PC(1234, 0);
863
864 f_init_m3ua(ignore_ssnm := false);
865
866 /* activate the first ASP */
867 f_M3UA_asp_up_act(0);
868
869 /* activate SG-role ASP (ASP on STP) */
870 f_init_m3ua_srv();
871 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4));
872
873 /* transmit a DUPU to the remote ASP */
874 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DUPU({adv_pc}, '0102'O, 'ABCD'O, rctxS0));
875 /* expect that to show up on other ASP */
876 f_M3UA_exp(0, tr_M3UA_DUPU({adv_pc}, '0102'O, 'ABCD'O, rctx0));
877}
878
879/* Test if SCON sent from SG to ASP-role STP gets forwarded to other ASP */
880testcase TC_ssnm_distribution_scon() runs on RAW_M3UA_CT {
881 var OCT4 rctx0 := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
882 var OCT4 rctxS0 := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
883 /* some random point code whose availability we advertise */
884 var template (value) M3UA_Point_Code adv_pc := ts_M3UA_PC(1234, 0);
885
886 f_init_m3ua(ignore_ssnm := false);
887
888 /* activate the first ASP */
889 f_M3UA_asp_up_act(0);
890
891 /* activate SG-role ASP (ASP on STP) */
892 f_init_m3ua_srv();
893 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4));
894
895 /* transmit a SCON to the remote ASP */
896 f_M3UA_send(M3UA_SRV(0), ts_M3UA_SCON({adv_pc}, rctxS0));
897 /* expect that to show up on other ASP */
898 f_M3UA_exp(0, tr_M3UA_SCON({adv_pc}, rctx0));
899}
Harald Welte829dac42019-11-05 16:55:30 +0100900
Harald Welte30cb4002021-05-13 20:51:37 +0200901private function f_quirk(charstring quirk) runs on RAW_M3UA_CT {
902 f_vty_config2(VTY, {"cs7 instance 0", "asp asp-client0 10002 2906 m3ua"}, "quirk " & quirk);
903}
904
905private function f_no_quirk(charstring quirk) runs on RAW_M3UA_CT {
906 f_vty_config2(VTY, {"cs7 instance 0", "asp asp-client0 10002 2906 m3ua"}, "no quirk " & quirk);
907}
908
909/* quirk 'no_notify': Expect inbound connection from ASP/SCTP-client, followed by ASP-UP + ASP-ACT */
910testcase TC_clnt_quirk_no_notify_asp_act() runs on RAW_M3UA_CT {
911 f_init_m3ua();
912 f_quirk("no_notify");
913 f_init_m3ua_srv();
914
915 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4),
916 ntfy_after_up := omit, ntfy_after_act := omit);
917 f_no_quirk("no_notify");
918 f_clear_m3ua();
919}
920
Harald Welte5ef25532021-05-13 21:10:25 +0200921/* ensure that DAUD is not supported in ASP role, as required by RFC */
922testcase TC_clnt_no_daud_in_asp() runs on RAW_M3UA_CT {
923 f_init_m3ua();
924 f_no_quirk("daud_in_asp");
925 f_init_m3ua_srv();
926
927 var OCT4 rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
928
929 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := rctx);
930
931 var template (value) M3UA_Point_Codes aff_pcs := { ts_M3UA_PC(mp_m3ua_configs[M3UA_SRV(0)].point_code) };
932 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DAUD(aff_pcs));
933 f_M3UA_exp(M3UA_SRV(0), tr_M3UA_ERR('00000004'O, omit));
934 setverdict(pass);
935
936 f_clear_m3ua();
937}
938
939/* quirk 'daud_in_asp': allowing inbound DAUD from SG in ASP role */
940testcase TC_clnt_quirk_daud_in_asp() runs on RAW_M3UA_CT {
941 f_init_m3ua();
942 f_quirk("daud_in_asp");
943 f_init_m3ua_srv();
944
945 var OCT4 rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
946
947 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := rctx);
948
949 var template (value) M3UA_Point_Codes aff_pcs := { ts_M3UA_PC(mp_m3ua_configs[M3UA_SRV(0)].point_code) };
950 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DAUD(aff_pcs));
951 f_M3UA_exp(M3UA_SRV(0), tr_M3UA_DAVA(aff_pcs));
952 setverdict(pass);
953
954 f_no_quirk("daud_in_asp");
955 f_clear_m3ua();
956}
Harald Welte0db44132019-10-17 11:09:05 +0200957
Harald Welte261aea72021-05-13 21:23:29 +0200958/* Expect a normal ASP to reject any [S]SNM messages in ASP-INACTIVE state */
959testcase TC_clnt_no_snm_inactive() runs on RAW_M3UA_CT {
960 f_init_m3ua();
961 f_quirk("no_notify");
962 f_quirk("daud_in_asp");
963 f_no_quirk("snm_inactive");
964 f_init_m3ua_srv();
965
966 /* bring ASP only UP (into INACTIVE state), but not ACTIVE! */
967 f_M3UA_CLNT_asp_up(M3UA_SRV(0));
968 f_M3UA_exp(M3UA_SRV(0), tr_M3UA_ASPAC(*, *));
969
970 var template (value) M3UA_Point_Codes aff_pcs := { ts_M3UA_PC(mp_m3ua_configs[M3UA_SRV(0)].point_code) };
971 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DAUD(aff_pcs));
972 f_M3UA_exp(M3UA_SRV(0), tr_M3UA_ERR('00000006'O, omit));
973 setverdict(pass);
974
975 f_no_quirk("no_notify");
976 f_no_quirk("daud_in_asp");
977 f_clear_m3ua();
978}
979
980/* quirk 'snm_inactive': Process [S]SNM in ASP-INACTIVE state */
981testcase TC_clnt_quirk_snm_inactive() runs on RAW_M3UA_CT {
982 f_init_m3ua();
983 f_quirk("no_notify");
984 f_quirk("daud_in_asp");
985 f_quirk("snm_inactive");
986 f_init_m3ua_srv();
987
988 /* bring ASP only UP (into INACTIVE state), but not ACTIVE! */
989 f_M3UA_CLNT_asp_up(M3UA_SRV(0));
990 f_M3UA_exp(M3UA_SRV(0), tr_M3UA_ASPAC(*, *));
991
992 var template (value) M3UA_Point_Codes aff_pcs := { ts_M3UA_PC(mp_m3ua_configs[M3UA_SRV(0)].point_code) };
993 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DAUD(aff_pcs));
994 f_M3UA_exp(M3UA_SRV(0), tr_M3UA_DAVA(aff_pcs));
995 setverdict(pass);
996
997 f_no_quirk("no_notify");
998 f_no_quirk("daud_in_asp");
999 f_no_quirk("snm_inactive");
1000 f_clear_m3ua();
1001}
1002
1003
Harald Welte0db44132019-10-17 11:09:05 +02001004control {
1005 /* M3UA Tests */
1006 execute( TC_connect_asp_up() );
1007 execute( TC_beat() );
1008 execute( TC_beat_payload() );
1009 execute( TC_asp_act() );
1010 execute( TC_asp_act_override() );
1011 execute( TC_asp_act_loadshare() );
1012 execute( TC_asp_act_broadcast() );
1013 execute( TC_tmt_override() );
1014 execute( TC_tmt_loadshare() );
1015 execute( TC_tmt_broadcast() );
Harald Welte15a85012020-06-13 16:35:45 +02001016 execute( TC_act_rctx_data_no_rctx() );
Harald Welte0db44132019-10-17 11:09:05 +02001017
1018 /* M3UA RKM tests */
1019 execute( TC_rkm_reg_static_notpermitted() );
1020 execute( TC_rkm_reg_static_permitted() );
1021 execute( TC_rkm_reg_dynamic_permitted() );
1022 execute( TC_rkm_unreg_never_registered() );
1023 execute( TC_rkm_unreg_invalid() );
1024 execute( TC_rkm_unreg_registered() );
1025 execute( TC_rkm_unreg_active() );
1026 /* TODO: test RKM with unsupported routing keys: NA, SI, OPC */
1027 /* TODO: register/unregister multiple routing contexts in one message; including mixed
1028 success/failure situations */
Harald Welte829dac42019-11-05 16:55:30 +01001029
1030 /* Test STP as SCTP client + M3UA ASP role */
1031 execute( TC_clnt_connect_asp_up() );
1032 execute( TC_clnt_asp_act() );
1033 execute( TC_clnt_sg_to_asp() );
1034 execute( TC_clnt_asp_to_sg() );
Harald Welte53050cd2019-12-01 20:49:01 +01001035
Harald Welte30cb4002021-05-13 20:51:37 +02001036 /* remove the condition if osmo-stp > 1.4.0 is released */
1037 if (mp_stp_has_asp_quirk) {
1038 execute( TC_clnt_quirk_no_notify_asp_act() );
Harald Welte5ef25532021-05-13 21:10:25 +02001039 execute( TC_clnt_no_daud_in_asp() );
1040 execute( TC_clnt_quirk_daud_in_asp() );
Harald Welte261aea72021-05-13 21:23:29 +02001041 execute( TC_clnt_no_snm_inactive() );
1042 execute( TC_clnt_quirk_snm_inactive() );
Harald Welte30cb4002021-05-13 20:51:37 +02001043 }
1044
Harald Welte27cfb7a2021-02-08 09:58:28 +01001045 /* M3UA SSNM tests */
1046 execute( TC_ssnm_aspac_dava_aspia_duna() );
1047 execute( TC_ssnm_distribution_dava_duna() );
1048 execute( TC_ssnm_distribution_dava_duna_multipc() );
1049 execute( TC_ssnm_distribution_dupu() );
1050 execute( TC_ssnm_distribution_scon() );
1051
Harald Welte53050cd2019-12-01 20:49:01 +01001052 /* put this one last as it changes the stp side config */
1053 execute( TC_clnt_asp_act_tmt_loadshare() );
Harald Welte0db44132019-10-17 11:09:05 +02001054}
1055
1056
1057
1058}