blob: 31bab9300020812bdf4e61b51bdf98cd03cf1b0c [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";
75}
76
77type record M3uaConfig {
78 /* STP-side SCTP port for M3UA */
79 integer remote_sctp_port,
80 /* local M3UA base port on TTCN3 side */
81 integer local_sctp_port,
82 /* point code routed via this M3U */
83 integer point_code,
84 /* associated routing context */
85 integer routing_ctx
86};
87type record length (NR_M3UA+NR_M3UA_SRV) of M3uaConfig M3uaConfigs;
88
Harald Welte829dac42019-11-05 16:55:30 +010089private function M3UA_SRV(integer idx) return integer {
90 return NR_M3UA+idx;
91}
Harald Welte0db44132019-10-17 11:09:05 +020092
93type component RAW_M3UA_CT extends Test_CT {
Harald Welte829dac42019-11-05 16:55:30 +010094 port M3UA_CODEC_PT M3UA[NR_M3UA+NR_M3UA_SRV];
95 var integer g_m3ua_conn_id[NR_M3UA+NR_M3UA_SRV];
Harald Welte0db44132019-10-17 11:09:05 +020096}
97
98private template PortEvent tr_SctpAssocChange := {
99 sctpEvent := {
100 sctpAssocChange := ?
101 }
102}
103private template PortEvent tr_SctpPeerAddrChange := {
104 sctpEvent := {
105 sctpPeerAddrChange := ?
106 }
107}
Harald Welte829dac42019-11-05 16:55:30 +0100108private template PortEvent tr_SctpConnOpened := {
109 connOpened := ?
110}
111
Harald Welte0db44132019-10-17 11:09:05 +0200112
113private altstep as_m3ua_sctp() runs on RAW_M3UA_CT {
114 [] any from M3UA.receive(tr_SctpAssocChange) { repeat; }
115 [] any from M3UA.receive(tr_SctpPeerAddrChange) { repeat; }
116}
117
Harald Welte42bcc442021-02-07 21:26:30 +0100118private altstep as_m3ua_ssnm_ignore() runs on RAW_M3UA_CT {
119 var M3UA_RecvFrom rx;
120 [] any from M3UA.receive(t_M3UA_RecvFrom(tr_M3UA_SSNM)) -> value rx {
121 log("Ignoring M3UA SSNM", rx);
122 repeat;
123 }
124}
125
Harald Welte0db44132019-10-17 11:09:05 +0200126friend function f_M3UA_send(integer idx, template (present) PDU_M3UA msg, template integer stream := 0)
127runs on RAW_M3UA_CT {
128 M3UA[idx].send(t_M3UA_Send(g_m3ua_conn_id[idx], msg, stream));
129}
130
131friend function f_M3UA_exp(integer idx, template (present) PDU_M3UA msg) runs on RAW_M3UA_CT {
132 var M3UA_RecvFrom rx;
Harald Welte30812592021-02-08 09:50:46 +0100133 timer T := 5.0;
134 T.start;
Harald Welte0db44132019-10-17 11:09:05 +0200135 alt {
136 [] M3UA[idx].receive(t_M3UA_RecvFrom(msg)) {
137 setverdict(pass);
138 }
139 [] M3UA[idx].receive(t_M3UA_RecvFrom(?)) -> value rx {
140 setverdict(fail, "Received unexpected M3UA[", idx, "] ", rx,
141 "while waiting for ", msg);
142 mtc.stop;
143 }
Harald Welte30812592021-02-08 09:50:46 +0100144 [] T.timeout {
145 setverdict(fail, "Timeout waiting for M3UA[", idx, "] ", msg);
146 mtc.stop;
147 }
Harald Welte0db44132019-10-17 11:09:05 +0200148 }
149}
150
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200151private template (value) Socket ts_Socket(HostName hostName, PortNumber portNumber) := {
152 hostName := hostName,
153 portNumber := portNumber
154};
155
156private template (value) SctpTuple ts_SCTP(template (omit) integer ppid := 3,
157 template (omit) integer stream := 0,
158 template (omit) SocketList remSocks := omit) := {
159 sinfo_stream := stream,
160 sinfo_ppid := ppid,
161 remSocks := remSocks,
162 assocId := omit
163};
164
Harald Welte0db44132019-10-17 11:09:05 +0200165friend function f_M3UA_connect(integer i) runs on RAW_M3UA_CT {
166 var Result res;
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200167 var Option opt_add_local_addrs;
168 var OptionList opt_list := {};
169 var template SocketList opt_add_remote_addrs;
Harald Welte1c4e9842021-02-07 23:03:28 +0100170 var M3uaConfig m3cfg := mp_m3ua_configs[i];
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200171
172 if (lengthof(mp_local_m3ua_ip) == 0 or lengthof(mp_stp_m3ua_ip) == 0) {
173 setverdict(fail, "Empty local or remote address trying to connect SCTP socket: ",
174 mp_local_m3ua_ip, " / ", mp_stp_m3ua_ip);
175 mtc.stop;
176 }
177
178 if (lengthof(mp_local_m3ua_ip) > 1) {
179 opt_add_local_addrs.sctpAdditionalLocalAddresses := substr(mp_local_m3ua_ip, 1,
180 lengthof(mp_local_m3ua_ip) - 1); //{mp_local_m3ua_ip};
181 opt_list := {opt_add_local_addrs};
182 }
183
184 if (lengthof(mp_stp_m3ua_ip) > 1) {
185 for (var integer j := 1; j < lengthof(mp_stp_m3ua_ip); j := j + 1) {
Harald Welte1c4e9842021-02-07 23:03:28 +0100186 var Socket sk := valueof(ts_Socket(mp_stp_m3ua_ip[j], m3cfg.remote_sctp_port));
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200187 opt_add_remote_addrs[j - 1] := sk;
188 }
189 } else {
190 opt_add_remote_addrs := omit;
191 }
192
Harald Welte1c4e9842021-02-07 23:03:28 +0100193 res := M3UA_CodecPort_CtrlFunct.f_IPL4_connect(M3UA[i], mp_stp_m3ua_ip[0], m3cfg.remote_sctp_port,
194 mp_local_m3ua_ip[0], m3cfg.local_sctp_port, 0,
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200195 {sctp:=valueof(ts_SCTP(3, 0, opt_add_remote_addrs))},
196 opt_list);
Harald Welte0db44132019-10-17 11:09:05 +0200197 if (not ispresent(res.connId)) {
198 setverdict(fail, "Could not connect M3UA socket, check your configuration");
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200199 mtc.stop;
Harald Welte0db44132019-10-17 11:09:05 +0200200 }
201 g_m3ua_conn_id[i] := res.connId;
202}
203
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100204friend function f_M3UA_close(integer i) runs on RAW_M3UA_CT {
205 var Result res;
206 res := M3UA_CodecPort_CtrlFunct.f_IPL4_close(M3UA[i], g_m3ua_conn_id[i], {sctp:=valueof(ts_SCTP)});
207 g_m3ua_conn_id[i] := 0;
208}
209
Harald Welte829dac42019-11-05 16:55:30 +0100210friend function f_M3UA_listen(integer i) runs on RAW_M3UA_CT {
211 var Result res;
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200212 var Option opt_add_local_addrs;
213 var OptionList opt_list := {};
Harald Welte1c4e9842021-02-07 23:03:28 +0100214 var M3uaConfig m3cfg := mp_m3ua_configs[i];
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200215
216 if (lengthof(mp_local_m3ua_ip) == 0 ) {
217 setverdict(fail, "Empty local address trying to bind SCTP socket: ",
218 mp_local_m3ua_ip);
219 mtc.stop;
220 }
221
222 if (lengthof(mp_local_m3ua_ip) > 1) {
223 opt_add_local_addrs.sctpAdditionalLocalAddresses := substr(mp_local_m3ua_ip, 1,
224 lengthof(mp_local_m3ua_ip) - 1); //{mp_local_m3ua_ip};
225 opt_list := {opt_add_local_addrs};
226 }
227
Harald Welte1c4e9842021-02-07 23:03:28 +0100228 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 +0200229 {sctp:=valueof(ts_SCTP)}, opt_list);
Harald Welte829dac42019-11-05 16:55:30 +0100230 if (not ispresent(res.connId)) {
231 setverdict(fail, "Could not bind M3UA socket, check your configuration");
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200232 mtc.stop;
Harald Welte829dac42019-11-05 16:55:30 +0100233 }
234}
235
Harald Welte27cfb7a2021-02-08 09:58:28 +0100236friend function f_init_m3ua(boolean ignore_ssnm := true) runs on RAW_M3UA_CT {
Harald Welte0db44132019-10-17 11:09:05 +0200237 var integer i;
238
239 f_init_common();
240
241 activate(as_m3ua_sctp());
Harald Welte27cfb7a2021-02-08 09:58:28 +0100242 if (ignore_ssnm) {
243 activate(as_m3ua_ssnm_ignore());
244 }
Harald Welte0db44132019-10-17 11:09:05 +0200245
246 for (i := 0; i < NR_M3UA; i:=i+1) {
247 map(self:M3UA[i], system:M3UA_CODEC_PT);
248 f_M3UA_connect(i);
249 }
250}
251
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100252friend function f_clear_m3ua() runs on RAW_M3UA_CT {
253 var integer i;
254
255 log("Clearing M3UA...");
256
257 for (i := 0; i < NR_M3UA; i:=i+1) {
258 f_M3UA_close(i);
259 }
260 /* Wait for recovery timer to trigger and shutdown all AS: */
261 f_sleep(int2float(mp_recovery_timeout_msec)/1000.0 + 0.5);
262 setverdict(pass, "M3UA cleared");
263}
264
Harald Welte829dac42019-11-05 16:55:30 +0100265friend function f_init_m3ua_srv() runs on RAW_M3UA_CT {
266 var integer i;
267 var PortEvent sctp_evt;
268
269 for (i := NR_M3UA; i < NR_M3UA+NR_M3UA_SRV; i:=i+1) {
270 map(self:M3UA[i], system:M3UA_CODEC_PT);
271 /* bind+ listen */
272 f_M3UA_listen(i);
273 /* wait for accept() */
274 M3UA[i].receive(tr_SctpConnOpened) -> value sctp_evt {
275 g_m3ua_conn_id[i] := sctp_evt.connOpened.connId;
276 }
277 }
278}
279
280
281/***********************************************************************
282 * Test the STP in M3UA SG role (we are ASP)
283 ***********************************************************************/
284
Harald Welte0db44132019-10-17 11:09:05 +0200285/* perform an outbound ASP-UP procedure */
286friend function f_M3UA_asp_up(integer idx, template (omit) OCT4 aspid := omit) runs on RAW_M3UA_CT {
287 f_M3UA_send(idx, ts_M3UA_ASPUP(aspid));
288 f_M3UA_exp(idx, tr_M3UA_ASPUP_ACK);
289}
290
291/* perform an outbound BEAT procedure */
292friend function f_M3UA_beat(integer idx, template (omit) octetstring hbd) runs on RAW_M3UA_CT {
293 if (istemplatekind(hbd, "omit")) {
294 f_M3UA_send(idx, ts_M3UA_BEAT(omit));
295 f_M3UA_exp(idx, tr_M3UA_BEAT_ACK(omit));
296 } else {
297 f_M3UA_send(idx, ts_M3UA_BEAT(ts_M3UA_hb_data(hbd)));
298 f_M3UA_exp(idx, tr_M3UA_BEAT_ACK(tr_M3UA_hb_data(hbd)));
299 }
300}
301
302/* perform an outbound ASP-ACTIVATE procedure */
303friend function f_M3UA_asp_act(integer idx, template (omit) M3UA_Traffic_Mode_Type tmt := omit,
304 template (omit) OCT4 rctx := omit) runs on RAW_M3UA_CT {
305 f_M3UA_send(idx, ts_M3UA_ASPAC(tmt, rctx));
306 f_M3UA_exp(idx, tr_M3UA_ASPAC_ACK(tmt, rctx));
307}
308
Harald Welte27cfb7a2021-02-08 09:58:28 +0100309/* perform an outbound ASP-INACTIVATE procedure */
310friend function f_M3UA_asp_inact(integer idx, template (omit) OCT4 rctx := omit) runs on RAW_M3UA_CT {
311 f_M3UA_send(idx, ts_M3UA_ASPIA(rctx));
312 f_M3UA_exp(idx, tr_M3UA_ASPIA_ACK(rctx));
313}
314
Harald Welte0db44132019-10-17 11:09:05 +0200315/* perform outbound ASP-UP and ASP-ACT, optionally expect interemittent NOTIFY */
316friend function f_M3UA_asp_up_act(integer idx, template (omit) M3UA_Traffic_Mode_Type tmt := omit,
317 template (omit) OCT4 rctx := omit,
318 template (omit) OCT2 ntfy_after_up := c_M3UA_ST_I_AS_INACTIVE,
319 template (omit) OCT2 ntfy_after_act := c_M3UA_ST_I_AS_ACTIVE)
320runs on RAW_M3UA_CT {
321 f_M3UA_asp_up(idx, omit);
322 if (not istemplatekind(ntfy_after_up, "omit")) {
323 f_M3UA_exp(idx, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_up, *));
324 }
325 f_M3UA_asp_act(idx, tmt, rctx);
326 if (not istemplatekind(ntfy_after_act, "omit")) {
327 f_M3UA_exp(idx, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_act, *));
328 }
329}
330
331
332/* Test the ASP-UP procedure */
333testcase TC_connect_asp_up() runs on RAW_M3UA_CT {
334 f_init_m3ua();
335 f_M3UA_asp_up(0);
336 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 +0100337 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200338}
339
340/* Test the heartbeat procedure without optional heartbeat data payload */
341testcase TC_beat() runs on RAW_M3UA_CT {
342 f_init_m3ua();
343 f_M3UA_asp_up(0);
344 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_INACTIVE, *));
345 f_M3UA_beat(0, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100346 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200347}
348
349/* Test the heartbeat procedure with optional heartbeat data payload */
350testcase TC_beat_payload() runs on RAW_M3UA_CT {
351 f_init_m3ua();
352 f_M3UA_asp_up(0);
353 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_INACTIVE, *));
354 f_M3UA_beat(0, 'a1a2a3a4a5'O);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100355 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200356}
357
358/* Test the ASP-ACTIVATE procedure (without traffic-mode or routing ctx) */
359testcase TC_asp_act() runs on RAW_M3UA_CT {
360 f_init_m3ua();
361 f_M3UA_asp_up_act(0);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100362 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200363}
364
365/* Test the ASP-ACTIVATE procedure with traffic-mode override */
366testcase TC_asp_act_override() runs on RAW_M3UA_CT {
367 f_init_m3ua();
368 f_M3UA_asp_up_act(0, c_M3UA_TMT_override, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100369 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200370}
371
372/* Test the ASP-ACTIVATE procedure with traffic-mode override */
373testcase TC_asp_act_loadshare() runs on RAW_M3UA_CT {
374 f_init_m3ua();
375 f_M3UA_asp_up_act(0, c_M3UA_TMT_loadshare, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100376 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200377}
378
379/* Test the ASP-ACTIVATE procedure with traffic-mode broadcast */
380testcase TC_asp_act_broadcast() runs on RAW_M3UA_CT {
381 f_init_m3ua();
382 f_M3UA_asp_up_act(0, c_M3UA_TMT_broadcast, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100383 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200384}
385
Harald Welte15a85012020-06-13 16:35:45 +0200386/* test whether the STP accepts M3UA DATA without Routing Context IE */
387testcase TC_act_rctx_data_no_rctx() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100388 var OCT4 rctx_sender := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
389 var OCT4 pc_sender := int2oct(mp_m3ua_configs[0].point_code, 4);
390 var OCT4 rctx_receiver := int2oct(mp_m3ua_configs[1].routing_ctx, 4);
391 var OCT4 pc_receiver := int2oct(mp_m3ua_configs[1].point_code, 4);
Harald Welte15a85012020-06-13 16:35:45 +0200392
393 f_init_m3ua();
394 /* bring up the sender specifying a routing context */
395
396 f_M3UA_asp_up_act(0, rctx := rctx_sender);
397 f_M3UA_asp_up_act(1);
398
399 /* check if DATA is accepted without Routing Context IE */
400 f_test_traffic(0, omit, pc_sender, 1, rctx_receiver, pc_receiver);
401
402 f_clear_m3ua();
403}
404
Harald Welte0db44132019-10-17 11:09:05 +0200405/* Test if traffic is routed from idx_tx/pc_tx to idx_rx/pc_rx */
406private function f_test_traffic(integer idx_tx, template (omit) OCT4 rctx_sender, OCT4 pc_tx,
407 integer idx_rx, template (omit) OCT4 rctx_receiver, OCT4 pc_rx,
408 OCT1 si := '23'O, OCT1 ni := '00'O, OCT1 mp := '00'O, OCT1 sls := '00'O)
409runs on RAW_M3UA_CT {
410 var octetstring data := f_rnd_octstring(f_rnd_int(100));
411 f_M3UA_send(idx_tx, ts_M3UA_DATA(rctx_sender,
412 ts_M3UA_protocol_data(pc_tx, pc_rx, si, ni, mp, sls, data)), 1);
413 f_M3UA_exp(idx_rx, tr_M3UA_DATA(rctx_receiver,
414 tr_M3UA_protocol_data(pc_tx, pc_rx, si, ni, mp, sls, data)));
415}
416
417
418/* test "traffic-mode override" behavior */
419testcase TC_tmt_override() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100420 var OCT4 rctx_sender := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
421 var OCT4 pc_sender := int2oct(mp_m3ua_configs[0].point_code, 4);
422 var OCT4 rctx_receiver := int2oct(mp_m3ua_configs[1].routing_ctx, 4);
423 var OCT4 pc_receiver := int2oct(mp_m3ua_configs[1].point_code, 4);
Harald Welte0db44132019-10-17 11:09:05 +0200424
425 f_init_m3ua();
426
427 /* bring up the 'sender' side (single ASP in AS) */
428 f_M3UA_asp_up_act(0, omit, omit);
429
430 /* activate the first 'receiver' side ASP */
431 f_M3UA_asp_up_act(1, c_M3UA_TMT_override, rctx_receiver);
432
433 /* verify traffic is routed from sender to [sole] receiver */
434 f_test_traffic(0, rctx_sender, pc_sender, 1, rctx_receiver, pc_receiver);
435
436 /* activate the second 'receiver' side ASP (no NOTIFY as AS state doesn't change) */
437 f_M3UA_asp_up_act(2, c_M3UA_TMT_override, rctx_receiver, omit, omit);
438
439 /* we expect a NOTIFY to the *other* ASP Other/Alternat-ASP-Active */
440 f_M3UA_exp(1, tr_M3UA_NOTIFY(c_M3UA_ST_T_OTHER, c_M3UA_ST_I_ALTERNATE_ASP, *));
441
442 /* verify traffic is routed from sender to new receiver */
443 f_test_traffic(0, rctx_sender, pc_sender, 2, rctx_receiver, pc_receiver);
444
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100445 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200446}
447
448private altstep as_count_rx(integer idx, template (present) PDU_M3UA exp, inout integer counter)
449runs on RAW_M3UA_CT {
450 [] M3UA[idx].receive(t_M3UA_RecvFrom(exp)) {
451 counter := counter + 1;
452 }
453}
454
455/* test "traffic-mode load-share" behavior */
456testcase TC_tmt_loadshare() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100457 var OCT4 rctx_sender := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
458 var OCT4 pc_sender := int2oct(mp_m3ua_configs[0].point_code, 4);
459 var OCT4 rctx_receiver := int2oct(mp_m3ua_configs[1].routing_ctx, 4);
460 var OCT4 pc_receiver := int2oct(mp_m3ua_configs[1].point_code, 4);
Harald Welte0db44132019-10-17 11:09:05 +0200461 var integer i;
462
463 f_init_m3ua();
464
465 /* FIXME: configure the STP via VTY to set traffic-mode */
466
467 /* bring up the 'sender' side (single ASP in AS) */
468 f_M3UA_asp_up_act(0, omit, rctx_sender);
469
470 /* activate the first 'receiver' side ASP */
Harald Welte0db44132019-10-17 11:09:05 +0200471 f_M3UA_asp_up_act(1, c_M3UA_TMT_loadshare, omit); // TODO: rctx
472
473 /* verify traffic is routed from sender to [sole] receiver */
474 for (i := 0; i < 10; i := i+1) {
475 f_test_traffic(0, rctx_sender, pc_sender, 1, rctx_receiver, pc_receiver);
476 }
477
478 /* activate the second 'receiver' side ASP (no NOTIFY) */
479 f_M3UA_asp_up_act(2, c_M3UA_TMT_loadshare, omit, omit, omit); // TODO: rctx
480
481 /* verify traffic is routed from sender to new receiver */
482 const integer iter_per_asp := 5;
483 var integer num_rx[NR_M3UA] := { 0, 0, 0 };
484 for (i := 0; i < 2*iter_per_asp; i := i+1) {
485 var octetstring data := f_rnd_octstring(f_rnd_int(100));
486 var template (value) M3UA_Protocol_Data tx_pd;
487 var template (present) M3UA_Protocol_Data rx_pd;
488 tx_pd := ts_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
489 rx_pd := tr_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
490 f_M3UA_send(0, ts_M3UA_DATA(rctx_sender, tx_pd), 1);
491 alt {
Pau Espin Pedrol4d915782020-08-21 12:24:35 +0200492 [] as_count_rx(1, tr_M3UA_DATA(rctx_receiver, rx_pd), num_rx[1]);
493 [] as_count_rx(2, tr_M3UA_DATA(rctx_receiver, rx_pd), num_rx[2]);
Harald Welte0db44132019-10-17 11:09:05 +0200494 }
495 }
496 /* FIXME: check for extraneous messages? */
497 for (i := 1; i <= 2; i := i+1) {
498 if (num_rx[i] != iter_per_asp) {
Harald Welteb9f7bbb2019-10-29 09:07:11 +0100499 setverdict(fail, "Received ", num_rx[i], " out of expected ", iter_per_asp,
500 "M3UA DATA messages at M3UA port ", i);
Harald Welte0db44132019-10-17 11:09:05 +0200501 }
502 }
503 setverdict(pass);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100504
505 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200506}
507
508/* test "traffic-mode broadcast" behavior */
509testcase TC_tmt_broadcast() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100510 var OCT4 rctx_sender := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
511 var OCT4 pc_sender := int2oct(mp_m3ua_configs[0].point_code, 4);
512 var OCT4 rctx_receiver := int2oct(mp_m3ua_configs[1].routing_ctx, 4);
513 var OCT4 pc_receiver := int2oct(mp_m3ua_configs[1].point_code, 4);
Harald Welte0db44132019-10-17 11:09:05 +0200514 var integer i;
515
516 f_init_m3ua();
517
518 /* FIXME: configure the STP via VTY to set traffic-mode */
519
520 /* bring up the 'sender' side (single ASP in AS) */
521 f_M3UA_asp_up_act(0, omit, omit); // TODO: rctx
522
523 /* activate the first 'receiver' side ASP */
Harald Welte0db44132019-10-17 11:09:05 +0200524 f_M3UA_asp_up_act(1, c_M3UA_TMT_broadcast, omit); // TODO: rctx
525
526 /* verify traffic is routed from sender to [sole] receiver */
527 for (i := 0; i < 10; i := i+1) {
528 f_test_traffic(0, rctx_sender, pc_sender, 1, rctx_receiver, pc_receiver);
529 }
530
531 /* activate the second 'receiver' side ASP */
532 f_M3UA_asp_up_act(2, c_M3UA_TMT_broadcast, omit, omit, omit); // TODO: rctx
533
534 /* verify traffic is routed from sender to new receiver */
535 for (i := 0; i < 10; i := i+1) {
536 var octetstring data := f_rnd_octstring(f_rnd_int(100));
537 var template (value) M3UA_Protocol_Data tx_pd;
538 var template (present) M3UA_Protocol_Data rx_pd;
539 tx_pd := ts_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
540 rx_pd := tr_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
541 f_M3UA_send(0, ts_M3UA_DATA(rctx_sender, tx_pd), 1);
542 /* each message must be received both on 1 and 2 */
543 f_M3UA_exp(1, tr_M3UA_DATA(rctx_receiver, rx_pd));
544 f_M3UA_exp(2, tr_M3UA_DATA(rctx_receiver, rx_pd));
545 }
546 setverdict(pass);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100547
548 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200549}
550
551private function f_M3UA_rkm_register(OCT4 id, OCT3 dpc, OCT4 rctx,
552 template (present) OCT4 exp_status := c_M3UA_REGSTS_SUCCESS)
553runs on RAW_M3UA_CT
554{
555 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:=id, dpc:=dpc, rctx:=rctx)}));
556 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:=id, status:=exp_status, rctx:=rctx)}));
557}
558
559/* Send RKM registration; expect -EPERM as RCTX doesn't match config and dynamic not permitted */
560testcase TC_rkm_reg_static_notpermitted() runs on RAW_M3UA_CT {
561 f_init_m3ua();
562
563 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:='00000099'O, dpc:='aabbcc'O)}));
564 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:='00000099'O, status:=c_M3UA_REGSTS_ERR_EPERM,
565 rctx:=?)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100566
567 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200568}
569
570/* Send RKM registration; expect OK as RCTX does match config */
571testcase TC_rkm_reg_static_permitted() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100572 var OCT3 dpc := int2oct(mp_m3ua_configs[0].point_code, 3); // must match config
573 var OCT4 rctx := int2oct(mp_m3ua_configs[0].routing_ctx, 4); // must match config
Harald Welte0db44132019-10-17 11:09:05 +0200574
575 f_init_m3ua();
576
577 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:='10000099'O, dpc:=dpc, rctx:=rctx)}));
578 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:='10000099'O, status:=c_M3UA_REGSTS_SUCCESS,
579 rctx:=rctx)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100580
581 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200582}
583
584/* Send RKM registration; expect OK as dynamic not permitted */
585testcase TC_rkm_reg_dynamic_permitted() runs on RAW_M3UA_CT {
586 f_init_common();
587 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation dynamic-permitted");
588 f_init_m3ua();
589
590 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:='20000099'O, dpc:='aabbcc'O)}));
591 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:='20000099'O, status:=c_M3UA_REGSTS_SUCCESS,
592 rctx:=?)}));
593
594 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation static-only");
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100595
596 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200597}
598
599/* try to de-register a routing key that was never registered -> error */
600testcase TC_rkm_unreg_never_registered() runs on RAW_M3UA_CT {
601 f_init_m3ua();
Harald Welte1c4e9842021-02-07 23:03:28 +0100602 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 +0200603 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 +0100604 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200605}
606
607/* try to de-register a routing key that is invalid (non-existant) -> error */
608testcase TC_rkm_unreg_invalid() runs on RAW_M3UA_CT {
609 f_init_m3ua();
610 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(int2oct(1234,4))));
611 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 +0100612 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200613}
614
Oliver Smith326d78e2021-07-23 10:42:10 +0200615/* try to de-register a routing key that was registered -> OK */
Harald Welte0db44132019-10-17 11:09:05 +0200616testcase TC_rkm_unreg_registered() runs on RAW_M3UA_CT {
Oliver Smith326d78e2021-07-23 10:42:10 +0200617 var OCT3 dpc := int2oct(123, 3);
618 var OCT4 rctx := int2oct(1234, 4);
619
620 f_init_common();
621 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation dynamic-permitted");
Harald Welte0db44132019-10-17 11:09:05 +0200622 f_init_m3ua();
Oliver Smith326d78e2021-07-23 10:42:10 +0200623
624 /* first register the routing key */
625 f_M3UA_rkm_register(id:='30000099'O, dpc:=dpc, rctx:=rctx);
626
627 /* then try to de-register */
628 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(rctx)));
Harald Welte0db44132019-10-17 11:09:05 +0200629 f_M3UA_exp(0, tr_M3UA_DEREG_RSP({tr_M3UA_dereg_res(?,c_m3UA_DEREGSTS_SUCCESS)}));
Oliver Smith326d78e2021-07-23 10:42:10 +0200630
631 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation static-only");
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100632 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200633}
634
635/* try to de-register a routing key for an active ASP -> ERROR */
636testcase TC_rkm_unreg_active() runs on RAW_M3UA_CT {
Oliver Smithee022232021-07-23 11:01:06 +0200637 var OCT3 dpc := int2oct(123, 3);
638 var OCT4 rctx := int2oct(1234, 4);
Harald Welte0db44132019-10-17 11:09:05 +0200639
Oliver Smithee022232021-07-23 11:01:06 +0200640 f_init_common();
641 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation dynamic-permitted");
Harald Welte0db44132019-10-17 11:09:05 +0200642 f_init_m3ua();
643
644 /* first register the routing key */
645 f_M3UA_rkm_register(id:='30000099'O, dpc:=dpc, rctx:=rctx);
646
647 /* then activate the ASP */
648 f_M3UA_asp_up_act(0);
Oliver Smithee022232021-07-23 11:01:06 +0200649 f_M3UA_exp(0, tr_M3UA_DAVA({*}, rctx));
650 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_ACTIVE, *));
651 f_M3UA_exp(0, tr_M3UA_DAVA({*}, *));
Harald Welte0db44132019-10-17 11:09:05 +0200652
Oliver Smithee022232021-07-23 11:01:06 +0200653 /* then try to de-register -> ERR_ASP_ACTIVE */
Harald Welte0db44132019-10-17 11:09:05 +0200654 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(rctx)));
655 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 +0100656
Oliver Smithee022232021-07-23 11:01:06 +0200657 /* deactivate ASP and properly de-register to clean up */
658 f_M3UA_asp_inact(0);
659 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(rctx)));
660 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_PENDING, *));
661 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_PENDING, *));
662 f_M3UA_exp(0, tr_M3UA_DEREG_RSP({tr_M3UA_dereg_res(?,c_m3UA_DEREGSTS_SUCCESS)}));
663
664 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation static-only");
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100665 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200666}
667
Harald Welte829dac42019-11-05 16:55:30 +0100668/***********************************************************************
669 * Test the STP in M3UA ASP role (we are SG)
670 ***********************************************************************/
671
672/* expect/perform an inbound ASP-UP procedure */
673friend function f_M3UA_CLNT_asp_up(integer idx, template OCT4 aspid := omit) runs on RAW_M3UA_CT {
674 f_M3UA_exp(idx, tr_M3UA_ASPUP(aspid));
675 f_M3UA_send(idx, ts_M3UA_ASPUP_ACK);
676}
677
678/* expect/perform an inbound ASP-ACTIVATE procedure */
679friend function f_M3UA_CLNT_asp_act(integer idx, template M3UA_Traffic_Mode_Type tmt := omit,
680 template (omit) OCT4 rctx := omit) runs on RAW_M3UA_CT {
681 f_M3UA_exp(idx, tr_M3UA_ASPAC(tmt, rctx));
682 f_M3UA_send(idx, ts_M3UA_ASPAC_ACK(tmt, rctx));
683}
684
685/* expect/perform inbound ASP-UP and ASP-ACT, optionally send interemittent NOTIFY */
686friend function f_M3UA_CLNT_asp_up_act(integer idx, template M3UA_Traffic_Mode_Type tmt := omit,
687 template OCT4 rctx := omit,
688 template (omit) OCT2 ntfy_after_up := c_M3UA_ST_I_AS_INACTIVE,
689 template (omit) OCT2 ntfy_after_act := c_M3UA_ST_I_AS_ACTIVE)
690runs on RAW_M3UA_CT {
691 f_M3UA_CLNT_asp_up(idx, omit);
692 if (not istemplatekind(ntfy_after_up, "omit")) {
693 f_M3UA_send(idx, ts_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_up, rctx));
694 }
695 f_M3UA_CLNT_asp_act(idx, tmt, rctx);
696 if (not istemplatekind(ntfy_after_act, "omit")) {
697 f_M3UA_send(idx, ts_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_act, rctx));
698 }
699}
700
701
702/* Expect inbound connection from ASP/SCTP-client, followed by ASP-UP */
703testcase TC_clnt_connect_asp_up() runs on RAW_M3UA_CT {
704 f_init_m3ua();
705 f_init_m3ua_srv();
706
707 f_M3UA_CLNT_asp_up(M3UA_SRV(0));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100708
709 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100710}
711
712/* Expect inbound connection from ASP/SCTP-client, followed by ASP-UP + ASP-ACT */
713testcase TC_clnt_asp_act() runs on RAW_M3UA_CT {
714 f_init_m3ua();
715 f_init_m3ua_srv();
716
Harald Welte1c4e9842021-02-07 23:03:28 +0100717 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 +0100718
719 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100720}
721
Harald Welte53050cd2019-12-01 20:49:01 +0100722/* Expect inbound connection from ASP/SCTP-client, followed by ASP-UP + ASP-ACT */
723testcase TC_clnt_asp_act_tmt_loadshare() runs on RAW_M3UA_CT {
724 f_init_common();
725 f_vty_config2(VTY, {"cs7 instance 0", "as as-client m3ua"}, "traffic-mode loadshare");
726 f_init_m3ua();
727 f_init_m3ua_srv();
728
Harald Welte1c4e9842021-02-07 23:03:28 +0100729 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 +0100730
731 f_clear_m3ua();
732}
733
Harald Welte829dac42019-11-05 16:55:30 +0100734/* Test traffic being routed through "server" side STP (M3UA SG), coming back in "client"
735 * side STP (M3UA ASP) */
736testcase TC_clnt_sg_to_asp() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100737 var OCT4 rctx_sender := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
738 var OCT4 pc_sender := int2oct(mp_m3ua_configs[0].point_code, 4);
739 var OCT4 rctx_receiver := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
740 var OCT4 pc_receiver := int2oct(mp_m3ua_configs[M3UA_SRV(0)].point_code, 4);
Harald Welte829dac42019-11-05 16:55:30 +0100741
742 f_init_m3ua();
743 f_M3UA_asp_up_act(0);
744
745 f_init_m3ua_srv();
Harald Welte1c4e9842021-02-07 23:03:28 +0100746 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 +0100747
748 f_sleep(1.0);
749
750 /* verify traffic is routed from sender to [sole] receiver */
751 f_test_traffic(0, rctx_sender, pc_sender, M3UA_SRV(0), rctx_receiver, pc_receiver);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100752
753 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100754}
755
756/* Test traffic being routed through "client" side STP (M3UA ASP), coming back in "server"
757 * side STP (M3UA SG) */
758testcase TC_clnt_asp_to_sg() runs on RAW_M3UA_CT {
Harald Welte1c4e9842021-02-07 23:03:28 +0100759 var OCT4 rctx_sender := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
760 var OCT4 pc_sender := int2oct(mp_m3ua_configs[M3UA_SRV(0)].point_code, 4);
761 var OCT4 rctx_receiver := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
762 var OCT4 pc_receiver := int2oct(mp_m3ua_configs[0].point_code, 4);
Harald Welte829dac42019-11-05 16:55:30 +0100763
764 f_init_m3ua();
765 f_M3UA_asp_up_act(0);
766
767 f_init_m3ua_srv();
Harald Welte1c4e9842021-02-07 23:03:28 +0100768 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 +0100769
770 f_sleep(1.0);
771
772 /* verify traffic is routed from sender to [sole] receiver */
773 f_test_traffic(M3UA_SRV(0), rctx_sender, pc_sender, 0, rctx_receiver, pc_receiver);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100774 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100775}
776
Harald Welte27cfb7a2021-02-08 09:58:28 +0100777/* Test if ASPAC / ASPIA of one ASP generates DAVA / DUNA on other ASP */
778testcase TC_ssnm_aspac_dava_aspia_duna() runs on RAW_M3UA_CT {
779 var OCT4 rctx0 := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
780
781 f_init_m3ua(ignore_ssnm := false);
782 /* activate the first ASP */
783 f_M3UA_asp_up_act(0);
784
785 /* activate the second ASP */
786 f_M3UA_asp_up_act(1, c_M3UA_TMT_override, omit);
787 /* expect DAVA for PC of second ASP on first ASP */
788 f_M3UA_exp(0, tr_M3UA_DAVA({ts_M3UA_PC(mp_m3ua_configs[1].point_code, 0)}, rctx0));
789 /* TODO: expect no DAVA on second ASP */
790
791 /* deactivate the second ASP */
792 f_M3UA_asp_inact(1);
793 /* expect DUNA for PC of second ASP on first ASP */
794 f_M3UA_exp(0, tr_M3UA_DUNA({ts_M3UA_PC(mp_m3ua_configs[1].point_code, 0)}, rctx0));
795 /* TODO: expect no DUNA on second ASP */
796
797 f_clear_m3ua();
798}
799
800/* Test if DAVA/DUNA sent from SG to ASP-role STP gets forwarded to other ASP */
801testcase TC_ssnm_distribution_dava_duna() runs on RAW_M3UA_CT {
802 var OCT4 rctx0 := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
803 var OCT4 rctxS0 := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
804 /* some random point code whose availability we advertise */
805 var template (value) M3UA_Point_Code adv_pc := ts_M3UA_PC(1234, 0);
806
807 f_init_m3ua(ignore_ssnm := false);
808
809 /* activate the first ASP */
810 f_M3UA_asp_up_act(0);
811
812 /* activate SG-role ASP (ASP on STP) */
813 f_init_m3ua_srv();
814 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4));
815
816 /* transmit a DAVA to the remote ASP */
817 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DAVA({adv_pc}, rctxS0));
818 /* expect that to show up on other ASP */
819 f_M3UA_exp(0, tr_M3UA_DAVA({adv_pc}, rctx0));
820
821 /* transmit a DUNA to the remote ASP */
822 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DUNA({adv_pc}, rctxS0));
823 /* expect that to show up on other ASP */
824 f_M3UA_exp(0, tr_M3UA_DUNA({adv_pc}, rctx0));
825}
826
827/* Test if DAVA/DUNA sent from SG to ASP-role STP gets forwarded to other ASP */
828testcase TC_ssnm_distribution_dava_duna_multipc() runs on RAW_M3UA_CT {
829 var OCT4 rctx0 := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
830 var OCT4 rctxS0 := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
831 /* some random point code whose availability we advertise */
832 var template (value) M3UA_Point_Codes adv_pcs := { ts_M3UA_PC(1234, 0), ts_M3UA_PC(5678, 0) };
833
834 f_init_m3ua(ignore_ssnm := false);
835
836 /* activate the first ASP */
837 f_M3UA_asp_up_act(0);
838
839 /* activate SG-role ASP (ASP on STP) */
840 f_init_m3ua_srv();
841 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4));
842
843 /* transmit a DAVA to the remote ASP */
844 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DAVA(adv_pcs, rctxS0));
845 /* expect that to show up on other ASP */
846 f_M3UA_exp(0, tr_M3UA_DAVA(adv_pcs, rctx0));
847
848 /* transmit a DUNA to the remote ASP */
849 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DUNA(adv_pcs, rctxS0));
850 /* expect that to show up on other ASP */
851 f_M3UA_exp(0, tr_M3UA_DUNA(adv_pcs, rctx0));
852}
853
854/* Test if DUPU sent from SG to ASP-role STP gets forwarded to other ASP */
855testcase TC_ssnm_distribution_dupu() runs on RAW_M3UA_CT {
856 var OCT4 rctx0 := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
857 var OCT4 rctxS0 := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
858 /* some random point code whose availability we advertise */
859 var template (value) M3UA_Point_Code adv_pc := ts_M3UA_PC(1234, 0);
860
861 f_init_m3ua(ignore_ssnm := false);
862
863 /* activate the first ASP */
864 f_M3UA_asp_up_act(0);
865
866 /* activate SG-role ASP (ASP on STP) */
867 f_init_m3ua_srv();
868 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4));
869
870 /* transmit a DUPU to the remote ASP */
871 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DUPU({adv_pc}, '0102'O, 'ABCD'O, rctxS0));
872 /* expect that to show up on other ASP */
873 f_M3UA_exp(0, tr_M3UA_DUPU({adv_pc}, '0102'O, 'ABCD'O, rctx0));
874}
875
876/* Test if SCON sent from SG to ASP-role STP gets forwarded to other ASP */
877testcase TC_ssnm_distribution_scon() runs on RAW_M3UA_CT {
878 var OCT4 rctx0 := int2oct(mp_m3ua_configs[0].routing_ctx, 4);
879 var OCT4 rctxS0 := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
880 /* some random point code whose availability we advertise */
881 var template (value) M3UA_Point_Code adv_pc := ts_M3UA_PC(1234, 0);
882
883 f_init_m3ua(ignore_ssnm := false);
884
885 /* activate the first ASP */
886 f_M3UA_asp_up_act(0);
887
888 /* activate SG-role ASP (ASP on STP) */
889 f_init_m3ua_srv();
890 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4));
891
892 /* transmit a SCON to the remote ASP */
893 f_M3UA_send(M3UA_SRV(0), ts_M3UA_SCON({adv_pc}, rctxS0));
894 /* expect that to show up on other ASP */
895 f_M3UA_exp(0, tr_M3UA_SCON({adv_pc}, rctx0));
896}
Harald Welte829dac42019-11-05 16:55:30 +0100897
Harald Welte30cb4002021-05-13 20:51:37 +0200898private function f_quirk(charstring quirk) runs on RAW_M3UA_CT {
899 f_vty_config2(VTY, {"cs7 instance 0", "asp asp-client0 10002 2906 m3ua"}, "quirk " & quirk);
900}
901
902private function f_no_quirk(charstring quirk) runs on RAW_M3UA_CT {
903 f_vty_config2(VTY, {"cs7 instance 0", "asp asp-client0 10002 2906 m3ua"}, "no quirk " & quirk);
904}
905
906/* quirk 'no_notify': Expect inbound connection from ASP/SCTP-client, followed by ASP-UP + ASP-ACT */
907testcase TC_clnt_quirk_no_notify_asp_act() runs on RAW_M3UA_CT {
908 f_init_m3ua();
909 f_quirk("no_notify");
910 f_init_m3ua_srv();
911
912 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4),
913 ntfy_after_up := omit, ntfy_after_act := omit);
914 f_no_quirk("no_notify");
915 f_clear_m3ua();
916}
917
Harald Welte5ef25532021-05-13 21:10:25 +0200918/* ensure that DAUD is not supported in ASP role, as required by RFC */
919testcase TC_clnt_no_daud_in_asp() runs on RAW_M3UA_CT {
920 f_init_m3ua();
921 f_no_quirk("daud_in_asp");
922 f_init_m3ua_srv();
923
924 var OCT4 rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
925
926 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := rctx);
927
928 var template (value) M3UA_Point_Codes aff_pcs := { ts_M3UA_PC(mp_m3ua_configs[M3UA_SRV(0)].point_code) };
929 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DAUD(aff_pcs));
930 f_M3UA_exp(M3UA_SRV(0), tr_M3UA_ERR('00000004'O, omit));
931 setverdict(pass);
932
933 f_clear_m3ua();
934}
935
936/* quirk 'daud_in_asp': allowing inbound DAUD from SG in ASP role */
937testcase TC_clnt_quirk_daud_in_asp() runs on RAW_M3UA_CT {
938 f_init_m3ua();
939 f_quirk("daud_in_asp");
940 f_init_m3ua_srv();
941
942 var OCT4 rctx := int2oct(mp_m3ua_configs[M3UA_SRV(0)].routing_ctx, 4);
943
944 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := rctx);
945
946 var template (value) M3UA_Point_Codes aff_pcs := { ts_M3UA_PC(mp_m3ua_configs[M3UA_SRV(0)].point_code) };
947 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DAUD(aff_pcs));
948 f_M3UA_exp(M3UA_SRV(0), tr_M3UA_DAVA(aff_pcs));
949 setverdict(pass);
950
951 f_no_quirk("daud_in_asp");
952 f_clear_m3ua();
953}
Harald Welte0db44132019-10-17 11:09:05 +0200954
Harald Welte261aea72021-05-13 21:23:29 +0200955/* Expect a normal ASP to reject any [S]SNM messages in ASP-INACTIVE state */
956testcase TC_clnt_no_snm_inactive() runs on RAW_M3UA_CT {
957 f_init_m3ua();
958 f_quirk("no_notify");
959 f_quirk("daud_in_asp");
960 f_no_quirk("snm_inactive");
961 f_init_m3ua_srv();
962
963 /* bring ASP only UP (into INACTIVE state), but not ACTIVE! */
964 f_M3UA_CLNT_asp_up(M3UA_SRV(0));
965 f_M3UA_exp(M3UA_SRV(0), tr_M3UA_ASPAC(*, *));
966
967 var template (value) M3UA_Point_Codes aff_pcs := { ts_M3UA_PC(mp_m3ua_configs[M3UA_SRV(0)].point_code) };
968 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DAUD(aff_pcs));
969 f_M3UA_exp(M3UA_SRV(0), tr_M3UA_ERR('00000006'O, omit));
970 setverdict(pass);
971
972 f_no_quirk("no_notify");
973 f_no_quirk("daud_in_asp");
974 f_clear_m3ua();
975}
976
977/* quirk 'snm_inactive': Process [S]SNM in ASP-INACTIVE state */
978testcase TC_clnt_quirk_snm_inactive() runs on RAW_M3UA_CT {
979 f_init_m3ua();
980 f_quirk("no_notify");
981 f_quirk("daud_in_asp");
982 f_quirk("snm_inactive");
983 f_init_m3ua_srv();
984
985 /* bring ASP only UP (into INACTIVE state), but not ACTIVE! */
986 f_M3UA_CLNT_asp_up(M3UA_SRV(0));
987 f_M3UA_exp(M3UA_SRV(0), tr_M3UA_ASPAC(*, *));
988
989 var template (value) M3UA_Point_Codes aff_pcs := { ts_M3UA_PC(mp_m3ua_configs[M3UA_SRV(0)].point_code) };
990 f_M3UA_send(M3UA_SRV(0), ts_M3UA_DAUD(aff_pcs));
991 f_M3UA_exp(M3UA_SRV(0), tr_M3UA_DAVA(aff_pcs));
992 setverdict(pass);
993
994 f_no_quirk("no_notify");
995 f_no_quirk("daud_in_asp");
996 f_no_quirk("snm_inactive");
997 f_clear_m3ua();
998}
999
1000
Harald Welte0db44132019-10-17 11:09:05 +02001001control {
1002 /* M3UA Tests */
1003 execute( TC_connect_asp_up() );
1004 execute( TC_beat() );
1005 execute( TC_beat_payload() );
1006 execute( TC_asp_act() );
1007 execute( TC_asp_act_override() );
1008 execute( TC_asp_act_loadshare() );
1009 execute( TC_asp_act_broadcast() );
1010 execute( TC_tmt_override() );
1011 execute( TC_tmt_loadshare() );
1012 execute( TC_tmt_broadcast() );
Harald Welte15a85012020-06-13 16:35:45 +02001013 execute( TC_act_rctx_data_no_rctx() );
Harald Welte0db44132019-10-17 11:09:05 +02001014
1015 /* M3UA RKM tests */
1016 execute( TC_rkm_reg_static_notpermitted() );
1017 execute( TC_rkm_reg_static_permitted() );
1018 execute( TC_rkm_reg_dynamic_permitted() );
Pau Espin Pedrol6b1b0ff2021-11-16 19:24:53 +01001019 execute( TC_rkm_unreg_never_registered() );
1020
Harald Welte0db44132019-10-17 11:09:05 +02001021 execute( TC_rkm_unreg_invalid() );
1022 execute( TC_rkm_unreg_registered() );
1023 execute( TC_rkm_unreg_active() );
1024 /* TODO: test RKM with unsupported routing keys: NA, SI, OPC */
1025 /* TODO: register/unregister multiple routing contexts in one message; including mixed
1026 success/failure situations */
Harald Welte829dac42019-11-05 16:55:30 +01001027
1028 /* Test STP as SCTP client + M3UA ASP role */
1029 execute( TC_clnt_connect_asp_up() );
1030 execute( TC_clnt_asp_act() );
1031 execute( TC_clnt_sg_to_asp() );
1032 execute( TC_clnt_asp_to_sg() );
Harald Welte53050cd2019-12-01 20:49:01 +01001033
Pau Espin Pedrol6b1b0ff2021-11-16 19:24:53 +01001034 execute( TC_clnt_quirk_no_notify_asp_act() );
1035 execute( TC_clnt_no_daud_in_asp() );
1036 execute( TC_clnt_quirk_daud_in_asp() );
1037 execute( TC_clnt_no_snm_inactive() );
1038 execute( TC_clnt_quirk_snm_inactive() );
1039
Harald Welte30cb4002021-05-13 20:51:37 +02001040
Harald Welte27cfb7a2021-02-08 09:58:28 +01001041 /* M3UA SSNM tests */
1042 execute( TC_ssnm_aspac_dava_aspia_duna() );
1043 execute( TC_ssnm_distribution_dava_duna() );
1044 execute( TC_ssnm_distribution_dava_duna_multipc() );
1045 execute( TC_ssnm_distribution_dupu() );
1046 execute( TC_ssnm_distribution_scon() );
1047
Harald Welte53050cd2019-12-01 20:49:01 +01001048 /* put this one last as it changes the stp side config */
1049 execute( TC_clnt_asp_act_tmt_loadshare() );
Harald Welte0db44132019-10-17 11:09:05 +02001050}
1051
1052
1053
1054}