blob: 1c25bae48627ea905327ac178bd5d5a3d708953a [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
37modulepar {
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +020038 HostList mp_stp_m3ua_ip := { "127.0.0.1", "::1" };
39 HostList mp_local_m3ua_ip := { "127.0.0.1", "::1" };
Harald Welte0db44132019-10-17 11:09:05 +020040 integer mp_stp_m3ua_port := 2905;
Harald Welte829dac42019-11-05 16:55:30 +010041 integer mp_stp_m3ua_clnt_port := 2906;
Harald Welte0db44132019-10-17 11:09:05 +020042 integer mp_local_m3ua_port := 9999;
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +010043 integer mp_recovery_timeout_msec := 2000;
Harald Welte0db44132019-10-17 11:09:05 +020044}
45
Harald Welte829dac42019-11-05 16:55:30 +010046private const integer NR_M3UA := 3; /* number of M3UA clients in ATS */
47private const integer NR_M3UA_SRV := 1; /* number of M3UA servres in ATS */
48
49private function M3UA_SRV(integer idx) return integer {
50 return NR_M3UA+idx;
51}
Harald Welte0db44132019-10-17 11:09:05 +020052
53type component RAW_M3UA_CT extends Test_CT {
Harald Welte829dac42019-11-05 16:55:30 +010054 port M3UA_CODEC_PT M3UA[NR_M3UA+NR_M3UA_SRV];
55 var integer g_m3ua_conn_id[NR_M3UA+NR_M3UA_SRV];
Harald Welte0db44132019-10-17 11:09:05 +020056}
57
58private template PortEvent tr_SctpAssocChange := {
59 sctpEvent := {
60 sctpAssocChange := ?
61 }
62}
63private template PortEvent tr_SctpPeerAddrChange := {
64 sctpEvent := {
65 sctpPeerAddrChange := ?
66 }
67}
Harald Welte829dac42019-11-05 16:55:30 +010068private template PortEvent tr_SctpConnOpened := {
69 connOpened := ?
70}
71
Harald Welte0db44132019-10-17 11:09:05 +020072
73private altstep as_m3ua_sctp() runs on RAW_M3UA_CT {
74 [] any from M3UA.receive(tr_SctpAssocChange) { repeat; }
75 [] any from M3UA.receive(tr_SctpPeerAddrChange) { repeat; }
76}
77
Harald Welte42bcc442021-02-07 21:26:30 +010078private altstep as_m3ua_ssnm_ignore() runs on RAW_M3UA_CT {
79 var M3UA_RecvFrom rx;
80 [] any from M3UA.receive(t_M3UA_RecvFrom(tr_M3UA_SSNM)) -> value rx {
81 log("Ignoring M3UA SSNM", rx);
82 repeat;
83 }
84}
85
Harald Welte0db44132019-10-17 11:09:05 +020086friend function f_M3UA_send(integer idx, template (present) PDU_M3UA msg, template integer stream := 0)
87runs on RAW_M3UA_CT {
88 M3UA[idx].send(t_M3UA_Send(g_m3ua_conn_id[idx], msg, stream));
89}
90
91friend function f_M3UA_exp(integer idx, template (present) PDU_M3UA msg) runs on RAW_M3UA_CT {
92 var M3UA_RecvFrom rx;
Harald Welte30812592021-02-08 09:50:46 +010093 timer T := 5.0;
94 T.start;
Harald Welte0db44132019-10-17 11:09:05 +020095 alt {
96 [] M3UA[idx].receive(t_M3UA_RecvFrom(msg)) {
97 setverdict(pass);
98 }
99 [] M3UA[idx].receive(t_M3UA_RecvFrom(?)) -> value rx {
100 setverdict(fail, "Received unexpected M3UA[", idx, "] ", rx,
101 "while waiting for ", msg);
102 mtc.stop;
103 }
Harald Welte30812592021-02-08 09:50:46 +0100104 [] T.timeout {
105 setverdict(fail, "Timeout waiting for M3UA[", idx, "] ", msg);
106 mtc.stop;
107 }
Harald Welte0db44132019-10-17 11:09:05 +0200108 }
109}
110
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200111private template (value) Socket ts_Socket(HostName hostName, PortNumber portNumber) := {
112 hostName := hostName,
113 portNumber := portNumber
114};
115
116private template (value) SctpTuple ts_SCTP(template (omit) integer ppid := 3,
117 template (omit) integer stream := 0,
118 template (omit) SocketList remSocks := omit) := {
119 sinfo_stream := stream,
120 sinfo_ppid := ppid,
121 remSocks := remSocks,
122 assocId := omit
123};
124
Harald Welte0db44132019-10-17 11:09:05 +0200125friend function f_M3UA_connect(integer i) runs on RAW_M3UA_CT {
126 var Result res;
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200127 var Option opt_add_local_addrs;
128 var OptionList opt_list := {};
129 var template SocketList opt_add_remote_addrs;
130
131 if (lengthof(mp_local_m3ua_ip) == 0 or lengthof(mp_stp_m3ua_ip) == 0) {
132 setverdict(fail, "Empty local or remote address trying to connect SCTP socket: ",
133 mp_local_m3ua_ip, " / ", mp_stp_m3ua_ip);
134 mtc.stop;
135 }
136
137 if (lengthof(mp_local_m3ua_ip) > 1) {
138 opt_add_local_addrs.sctpAdditionalLocalAddresses := substr(mp_local_m3ua_ip, 1,
139 lengthof(mp_local_m3ua_ip) - 1); //{mp_local_m3ua_ip};
140 opt_list := {opt_add_local_addrs};
141 }
142
143 if (lengthof(mp_stp_m3ua_ip) > 1) {
144 for (var integer j := 1; j < lengthof(mp_stp_m3ua_ip); j := j + 1) {
145 var Socket sk := valueof(ts_Socket(mp_stp_m3ua_ip[j], mp_stp_m3ua_port));
146 opt_add_remote_addrs[j - 1] := sk;
147 }
148 } else {
149 opt_add_remote_addrs := omit;
150 }
151
152 res := M3UA_CodecPort_CtrlFunct.f_IPL4_connect(M3UA[i], mp_stp_m3ua_ip[0], mp_stp_m3ua_port,
153 mp_local_m3ua_ip[0], mp_local_m3ua_port+i, 0,
154 {sctp:=valueof(ts_SCTP(3, 0, opt_add_remote_addrs))},
155 opt_list);
Harald Welte0db44132019-10-17 11:09:05 +0200156 if (not ispresent(res.connId)) {
157 setverdict(fail, "Could not connect M3UA socket, check your configuration");
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200158 mtc.stop;
Harald Welte0db44132019-10-17 11:09:05 +0200159 }
160 g_m3ua_conn_id[i] := res.connId;
161}
162
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100163friend function f_M3UA_close(integer i) runs on RAW_M3UA_CT {
164 var Result res;
165 res := M3UA_CodecPort_CtrlFunct.f_IPL4_close(M3UA[i], g_m3ua_conn_id[i], {sctp:=valueof(ts_SCTP)});
166 g_m3ua_conn_id[i] := 0;
167}
168
Harald Welte829dac42019-11-05 16:55:30 +0100169friend function f_M3UA_listen(integer i) runs on RAW_M3UA_CT {
170 var Result res;
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200171 var Option opt_add_local_addrs;
172 var OptionList opt_list := {};
173
174 if (lengthof(mp_local_m3ua_ip) == 0 ) {
175 setverdict(fail, "Empty local address trying to bind SCTP socket: ",
176 mp_local_m3ua_ip);
177 mtc.stop;
178 }
179
180 if (lengthof(mp_local_m3ua_ip) > 1) {
181 opt_add_local_addrs.sctpAdditionalLocalAddresses := substr(mp_local_m3ua_ip, 1,
182 lengthof(mp_local_m3ua_ip) - 1); //{mp_local_m3ua_ip};
183 opt_list := {opt_add_local_addrs};
184 }
185
186 res := M3UA_CodecPort_CtrlFunct.f_IPL4_listen(M3UA[i], mp_local_m3ua_ip[0], mp_local_m3ua_port+i,
187 {sctp:=valueof(ts_SCTP)}, opt_list);
Harald Welte829dac42019-11-05 16:55:30 +0100188 if (not ispresent(res.connId)) {
189 setverdict(fail, "Could not bind M3UA socket, check your configuration");
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200190 mtc.stop;
Harald Welte829dac42019-11-05 16:55:30 +0100191 }
192}
193
Harald Welte0db44132019-10-17 11:09:05 +0200194friend function f_init_m3ua() runs on RAW_M3UA_CT {
195 var integer i;
196
197 f_init_common();
198
199 activate(as_m3ua_sctp());
Harald Welte42bcc442021-02-07 21:26:30 +0100200 activate(as_m3ua_ssnm_ignore());
Harald Welte0db44132019-10-17 11:09:05 +0200201
202 for (i := 0; i < NR_M3UA; i:=i+1) {
203 map(self:M3UA[i], system:M3UA_CODEC_PT);
204 f_M3UA_connect(i);
205 }
206}
207
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100208friend function f_clear_m3ua() runs on RAW_M3UA_CT {
209 var integer i;
210
211 log("Clearing M3UA...");
212
213 for (i := 0; i < NR_M3UA; i:=i+1) {
214 f_M3UA_close(i);
215 }
216 /* Wait for recovery timer to trigger and shutdown all AS: */
217 f_sleep(int2float(mp_recovery_timeout_msec)/1000.0 + 0.5);
218 setverdict(pass, "M3UA cleared");
219}
220
Harald Welte829dac42019-11-05 16:55:30 +0100221friend function f_init_m3ua_srv() runs on RAW_M3UA_CT {
222 var integer i;
223 var PortEvent sctp_evt;
224
225 for (i := NR_M3UA; i < NR_M3UA+NR_M3UA_SRV; i:=i+1) {
226 map(self:M3UA[i], system:M3UA_CODEC_PT);
227 /* bind+ listen */
228 f_M3UA_listen(i);
229 /* wait for accept() */
230 M3UA[i].receive(tr_SctpConnOpened) -> value sctp_evt {
231 g_m3ua_conn_id[i] := sctp_evt.connOpened.connId;
232 }
233 }
234}
235
236
237/***********************************************************************
238 * Test the STP in M3UA SG role (we are ASP)
239 ***********************************************************************/
240
Harald Welte0db44132019-10-17 11:09:05 +0200241/* perform an outbound ASP-UP procedure */
242friend function f_M3UA_asp_up(integer idx, template (omit) OCT4 aspid := omit) runs on RAW_M3UA_CT {
243 f_M3UA_send(idx, ts_M3UA_ASPUP(aspid));
244 f_M3UA_exp(idx, tr_M3UA_ASPUP_ACK);
245}
246
247/* perform an outbound BEAT procedure */
248friend function f_M3UA_beat(integer idx, template (omit) octetstring hbd) runs on RAW_M3UA_CT {
249 if (istemplatekind(hbd, "omit")) {
250 f_M3UA_send(idx, ts_M3UA_BEAT(omit));
251 f_M3UA_exp(idx, tr_M3UA_BEAT_ACK(omit));
252 } else {
253 f_M3UA_send(idx, ts_M3UA_BEAT(ts_M3UA_hb_data(hbd)));
254 f_M3UA_exp(idx, tr_M3UA_BEAT_ACK(tr_M3UA_hb_data(hbd)));
255 }
256}
257
258/* perform an outbound ASP-ACTIVATE procedure */
259friend function f_M3UA_asp_act(integer idx, template (omit) M3UA_Traffic_Mode_Type tmt := omit,
260 template (omit) OCT4 rctx := omit) runs on RAW_M3UA_CT {
261 f_M3UA_send(idx, ts_M3UA_ASPAC(tmt, rctx));
262 f_M3UA_exp(idx, tr_M3UA_ASPAC_ACK(tmt, rctx));
263}
264
265/* perform outbound ASP-UP and ASP-ACT, optionally expect interemittent NOTIFY */
266friend function f_M3UA_asp_up_act(integer idx, template (omit) M3UA_Traffic_Mode_Type tmt := omit,
267 template (omit) OCT4 rctx := omit,
268 template (omit) OCT2 ntfy_after_up := c_M3UA_ST_I_AS_INACTIVE,
269 template (omit) OCT2 ntfy_after_act := c_M3UA_ST_I_AS_ACTIVE)
270runs on RAW_M3UA_CT {
271 f_M3UA_asp_up(idx, omit);
272 if (not istemplatekind(ntfy_after_up, "omit")) {
273 f_M3UA_exp(idx, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_up, *));
274 }
275 f_M3UA_asp_act(idx, tmt, rctx);
276 if (not istemplatekind(ntfy_after_act, "omit")) {
277 f_M3UA_exp(idx, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_act, *));
278 }
279}
280
281
282/* Test the ASP-UP procedure */
283testcase TC_connect_asp_up() runs on RAW_M3UA_CT {
284 f_init_m3ua();
285 f_M3UA_asp_up(0);
286 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 +0100287 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200288}
289
290/* Test the heartbeat procedure without optional heartbeat data payload */
291testcase TC_beat() runs on RAW_M3UA_CT {
292 f_init_m3ua();
293 f_M3UA_asp_up(0);
294 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_INACTIVE, *));
295 f_M3UA_beat(0, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100296 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200297}
298
299/* Test the heartbeat procedure with optional heartbeat data payload */
300testcase TC_beat_payload() runs on RAW_M3UA_CT {
301 f_init_m3ua();
302 f_M3UA_asp_up(0);
303 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_INACTIVE, *));
304 f_M3UA_beat(0, 'a1a2a3a4a5'O);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100305 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200306}
307
308/* Test the ASP-ACTIVATE procedure (without traffic-mode or routing ctx) */
309testcase TC_asp_act() runs on RAW_M3UA_CT {
310 f_init_m3ua();
311 f_M3UA_asp_up_act(0);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100312 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200313}
314
315/* Test the ASP-ACTIVATE procedure with traffic-mode override */
316testcase TC_asp_act_override() runs on RAW_M3UA_CT {
317 f_init_m3ua();
318 f_M3UA_asp_up_act(0, c_M3UA_TMT_override, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100319 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200320}
321
322/* Test the ASP-ACTIVATE procedure with traffic-mode override */
323testcase TC_asp_act_loadshare() runs on RAW_M3UA_CT {
324 f_init_m3ua();
325 f_M3UA_asp_up_act(0, c_M3UA_TMT_loadshare, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100326 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200327}
328
329/* Test the ASP-ACTIVATE procedure with traffic-mode broadcast */
330testcase TC_asp_act_broadcast() runs on RAW_M3UA_CT {
331 f_init_m3ua();
332 f_M3UA_asp_up_act(0, c_M3UA_TMT_broadcast, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100333 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200334}
335
Harald Welte15a85012020-06-13 16:35:45 +0200336/* test whether the STP accepts M3UA DATA without Routing Context IE */
337testcase TC_act_rctx_data_no_rctx() runs on RAW_M3UA_CT {
338 var OCT4 rctx_sender := int2oct(1023, 4);
339 var OCT4 pc_sender := int2oct(23, 4);
340 var OCT4 rctx_receiver := int2oct(1042, 4);
341 var OCT4 pc_receiver := int2oct(42, 4);
342
343 f_init_m3ua();
344 /* bring up the sender specifying a routing context */
345
346 f_M3UA_asp_up_act(0, rctx := rctx_sender);
347 f_M3UA_asp_up_act(1);
348
349 /* check if DATA is accepted without Routing Context IE */
350 f_test_traffic(0, omit, pc_sender, 1, rctx_receiver, pc_receiver);
351
352 f_clear_m3ua();
353}
354
Harald Welte0db44132019-10-17 11:09:05 +0200355/* Test if traffic is routed from idx_tx/pc_tx to idx_rx/pc_rx */
356private function f_test_traffic(integer idx_tx, template (omit) OCT4 rctx_sender, OCT4 pc_tx,
357 integer idx_rx, template (omit) OCT4 rctx_receiver, OCT4 pc_rx,
358 OCT1 si := '23'O, OCT1 ni := '00'O, OCT1 mp := '00'O, OCT1 sls := '00'O)
359runs on RAW_M3UA_CT {
360 var octetstring data := f_rnd_octstring(f_rnd_int(100));
361 f_M3UA_send(idx_tx, ts_M3UA_DATA(rctx_sender,
362 ts_M3UA_protocol_data(pc_tx, pc_rx, si, ni, mp, sls, data)), 1);
363 f_M3UA_exp(idx_rx, tr_M3UA_DATA(rctx_receiver,
364 tr_M3UA_protocol_data(pc_tx, pc_rx, si, ni, mp, sls, data)));
365}
366
367
368/* test "traffic-mode override" behavior */
369testcase TC_tmt_override() runs on RAW_M3UA_CT {
370 var OCT4 rctx_sender := int2oct(1023, 4);
371 var OCT4 pc_sender := int2oct(23, 4);
372 var OCT4 rctx_receiver := int2oct(1042, 4);
373 var OCT4 pc_receiver := int2oct(42, 4);
374
375 f_init_m3ua();
376
377 /* bring up the 'sender' side (single ASP in AS) */
378 f_M3UA_asp_up_act(0, omit, omit);
379
380 /* activate the first 'receiver' side ASP */
381 f_M3UA_asp_up_act(1, c_M3UA_TMT_override, rctx_receiver);
382
383 /* verify traffic is routed from sender to [sole] receiver */
384 f_test_traffic(0, rctx_sender, pc_sender, 1, rctx_receiver, pc_receiver);
385
386 /* activate the second 'receiver' side ASP (no NOTIFY as AS state doesn't change) */
387 f_M3UA_asp_up_act(2, c_M3UA_TMT_override, rctx_receiver, omit, omit);
388
389 /* we expect a NOTIFY to the *other* ASP Other/Alternat-ASP-Active */
390 f_M3UA_exp(1, tr_M3UA_NOTIFY(c_M3UA_ST_T_OTHER, c_M3UA_ST_I_ALTERNATE_ASP, *));
391
392 /* verify traffic is routed from sender to new receiver */
393 f_test_traffic(0, rctx_sender, pc_sender, 2, rctx_receiver, pc_receiver);
394
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100395 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200396}
397
398private altstep as_count_rx(integer idx, template (present) PDU_M3UA exp, inout integer counter)
399runs on RAW_M3UA_CT {
400 [] M3UA[idx].receive(t_M3UA_RecvFrom(exp)) {
401 counter := counter + 1;
402 }
403}
404
405/* test "traffic-mode load-share" behavior */
406testcase TC_tmt_loadshare() runs on RAW_M3UA_CT {
407 var OCT4 rctx_sender := int2oct(1023, 4);
408 var OCT4 pc_sender := int2oct(23, 4);
409 var OCT4 rctx_receiver := int2oct(1042, 4);
410 var OCT4 pc_receiver := int2oct(42, 4);
411 var integer i;
412
413 f_init_m3ua();
414
415 /* FIXME: configure the STP via VTY to set traffic-mode */
416
417 /* bring up the 'sender' side (single ASP in AS) */
418 f_M3UA_asp_up_act(0, omit, rctx_sender);
419
420 /* activate the first 'receiver' side ASP */
Harald Welte0db44132019-10-17 11:09:05 +0200421 f_M3UA_asp_up_act(1, c_M3UA_TMT_loadshare, omit); // TODO: rctx
422
423 /* verify traffic is routed from sender to [sole] receiver */
424 for (i := 0; i < 10; i := i+1) {
425 f_test_traffic(0, rctx_sender, pc_sender, 1, rctx_receiver, pc_receiver);
426 }
427
428 /* activate the second 'receiver' side ASP (no NOTIFY) */
429 f_M3UA_asp_up_act(2, c_M3UA_TMT_loadshare, omit, omit, omit); // TODO: rctx
430
431 /* verify traffic is routed from sender to new receiver */
432 const integer iter_per_asp := 5;
433 var integer num_rx[NR_M3UA] := { 0, 0, 0 };
434 for (i := 0; i < 2*iter_per_asp; i := i+1) {
435 var octetstring data := f_rnd_octstring(f_rnd_int(100));
436 var template (value) M3UA_Protocol_Data tx_pd;
437 var template (present) M3UA_Protocol_Data rx_pd;
438 tx_pd := ts_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
439 rx_pd := tr_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
440 f_M3UA_send(0, ts_M3UA_DATA(rctx_sender, tx_pd), 1);
441 alt {
Pau Espin Pedrol4d915782020-08-21 12:24:35 +0200442 [] as_count_rx(1, tr_M3UA_DATA(rctx_receiver, rx_pd), num_rx[1]);
443 [] as_count_rx(2, tr_M3UA_DATA(rctx_receiver, rx_pd), num_rx[2]);
Harald Welte0db44132019-10-17 11:09:05 +0200444 }
445 }
446 /* FIXME: check for extraneous messages? */
447 for (i := 1; i <= 2; i := i+1) {
448 if (num_rx[i] != iter_per_asp) {
Harald Welteb9f7bbb2019-10-29 09:07:11 +0100449 setverdict(fail, "Received ", num_rx[i], " out of expected ", iter_per_asp,
450 "M3UA DATA messages at M3UA port ", i);
Harald Welte0db44132019-10-17 11:09:05 +0200451 }
452 }
453 setverdict(pass);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100454
455 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200456}
457
458/* test "traffic-mode broadcast" behavior */
459testcase TC_tmt_broadcast() runs on RAW_M3UA_CT {
460 var OCT4 rctx_sender := int2oct(1023, 4);
461 var OCT4 pc_sender := int2oct(23, 4);
462 var OCT4 rctx_receiver := int2oct(1042, 4);
463 var OCT4 pc_receiver := int2oct(42, 4);
464 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, omit); // TODO: rctx
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_broadcast, 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 */
482 f_M3UA_asp_up_act(2, c_M3UA_TMT_broadcast, omit, omit, omit); // TODO: rctx
483
484 /* verify traffic is routed from sender to new receiver */
485 for (i := 0; i < 10; i := i+1) {
486 var octetstring data := f_rnd_octstring(f_rnd_int(100));
487 var template (value) M3UA_Protocol_Data tx_pd;
488 var template (present) M3UA_Protocol_Data rx_pd;
489 tx_pd := ts_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
490 rx_pd := tr_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
491 f_M3UA_send(0, ts_M3UA_DATA(rctx_sender, tx_pd), 1);
492 /* each message must be received both on 1 and 2 */
493 f_M3UA_exp(1, tr_M3UA_DATA(rctx_receiver, rx_pd));
494 f_M3UA_exp(2, tr_M3UA_DATA(rctx_receiver, rx_pd));
495 }
496 setverdict(pass);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100497
498 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200499}
500
501private function f_M3UA_rkm_register(OCT4 id, OCT3 dpc, OCT4 rctx,
502 template (present) OCT4 exp_status := c_M3UA_REGSTS_SUCCESS)
503runs on RAW_M3UA_CT
504{
505 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:=id, dpc:=dpc, rctx:=rctx)}));
506 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:=id, status:=exp_status, rctx:=rctx)}));
507}
508
509/* Send RKM registration; expect -EPERM as RCTX doesn't match config and dynamic not permitted */
510testcase TC_rkm_reg_static_notpermitted() runs on RAW_M3UA_CT {
511 f_init_m3ua();
512
513 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:='00000099'O, dpc:='aabbcc'O)}));
514 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:='00000099'O, status:=c_M3UA_REGSTS_ERR_EPERM,
515 rctx:=?)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100516
517 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200518}
519
520/* Send RKM registration; expect OK as RCTX does match config */
521testcase TC_rkm_reg_static_permitted() runs on RAW_M3UA_CT {
522 var OCT3 dpc := int2oct(23, 3); // must match config
523 var OCT4 rctx := int2oct(1023, 4); // must match config
524
525 f_init_m3ua();
526
527 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:='10000099'O, dpc:=dpc, rctx:=rctx)}));
528 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:='10000099'O, status:=c_M3UA_REGSTS_SUCCESS,
529 rctx:=rctx)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100530
531 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200532}
533
534/* Send RKM registration; expect OK as dynamic not permitted */
535testcase TC_rkm_reg_dynamic_permitted() runs on RAW_M3UA_CT {
536 f_init_common();
537 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation dynamic-permitted");
538 f_init_m3ua();
539
540 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:='20000099'O, dpc:='aabbcc'O)}));
541 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:='20000099'O, status:=c_M3UA_REGSTS_SUCCESS,
542 rctx:=?)}));
543
544 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation static-only");
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100545
546 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200547}
548
549/* try to de-register a routing key that was never registered -> error */
550testcase TC_rkm_unreg_never_registered() runs on RAW_M3UA_CT {
551 f_init_m3ua();
552 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(int2oct(1023,4))));
553 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 +0100554 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200555}
556
557/* try to de-register a routing key that is invalid (non-existant) -> error */
558testcase TC_rkm_unreg_invalid() runs on RAW_M3UA_CT {
559 f_init_m3ua();
560 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(int2oct(1234,4))));
561 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 +0100562 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200563}
564
565/* try to de-register a routing key that was registered -> OK*/
566testcase TC_rkm_unreg_registered() runs on RAW_M3UA_CT {
567 f_init_m3ua();
568 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(int2oct(1023,4))));
569 f_M3UA_exp(0, tr_M3UA_DEREG_RSP({tr_M3UA_dereg_res(?,c_m3UA_DEREGSTS_SUCCESS)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100570 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200571}
572
573/* try to de-register a routing key for an active ASP -> ERROR */
574testcase TC_rkm_unreg_active() runs on RAW_M3UA_CT {
575 var OCT3 dpc := int2oct(23, 3); // must match config
576 var OCT4 rctx := int2oct(1023, 4); // must match config
577
578 f_init_m3ua();
579
580 /* first register the routing key */
581 f_M3UA_rkm_register(id:='30000099'O, dpc:=dpc, rctx:=rctx);
582
583 /* then activate the ASP */
584 f_M3UA_asp_up_act(0);
585
586 /* then try to de-regsiter */
587 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(rctx)));
588 f_M3UA_exp(0, tr_M3UA_DEREG_RSP({tr_M3UA_dereg_res(?,c_m3UA_DEREGSTS_ERR_ASP_ACTIVE)}));
589 /* FIXME: we now may have changed the state on the STP side! */
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100590
591 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200592}
593
Harald Welte829dac42019-11-05 16:55:30 +0100594/***********************************************************************
595 * Test the STP in M3UA ASP role (we are SG)
596 ***********************************************************************/
597
598/* expect/perform an inbound ASP-UP procedure */
599friend function f_M3UA_CLNT_asp_up(integer idx, template OCT4 aspid := omit) runs on RAW_M3UA_CT {
600 f_M3UA_exp(idx, tr_M3UA_ASPUP(aspid));
601 f_M3UA_send(idx, ts_M3UA_ASPUP_ACK);
602}
603
604/* expect/perform an inbound ASP-ACTIVATE procedure */
605friend function f_M3UA_CLNT_asp_act(integer idx, template M3UA_Traffic_Mode_Type tmt := omit,
606 template (omit) OCT4 rctx := omit) runs on RAW_M3UA_CT {
607 f_M3UA_exp(idx, tr_M3UA_ASPAC(tmt, rctx));
608 f_M3UA_send(idx, ts_M3UA_ASPAC_ACK(tmt, rctx));
609}
610
611/* expect/perform inbound ASP-UP and ASP-ACT, optionally send interemittent NOTIFY */
612friend function f_M3UA_CLNT_asp_up_act(integer idx, template M3UA_Traffic_Mode_Type tmt := omit,
613 template OCT4 rctx := omit,
614 template (omit) OCT2 ntfy_after_up := c_M3UA_ST_I_AS_INACTIVE,
615 template (omit) OCT2 ntfy_after_act := c_M3UA_ST_I_AS_ACTIVE)
616runs on RAW_M3UA_CT {
617 f_M3UA_CLNT_asp_up(idx, omit);
618 if (not istemplatekind(ntfy_after_up, "omit")) {
619 f_M3UA_send(idx, ts_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_up, rctx));
620 }
621 f_M3UA_CLNT_asp_act(idx, tmt, rctx);
622 if (not istemplatekind(ntfy_after_act, "omit")) {
623 f_M3UA_send(idx, ts_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_act, rctx));
624 }
625}
626
627
628/* Expect inbound connection from ASP/SCTP-client, followed by ASP-UP */
629testcase TC_clnt_connect_asp_up() runs on RAW_M3UA_CT {
630 f_init_m3ua();
631 f_init_m3ua_srv();
632
633 f_M3UA_CLNT_asp_up(M3UA_SRV(0));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100634
635 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100636}
637
638/* Expect inbound connection from ASP/SCTP-client, followed by ASP-UP + ASP-ACT */
639testcase TC_clnt_asp_act() runs on RAW_M3UA_CT {
640 f_init_m3ua();
641 f_init_m3ua_srv();
642
Harald Weltedfbca112019-12-01 18:00:48 +0100643 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(1055, 4));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100644
645 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100646}
647
Harald Welte53050cd2019-12-01 20:49:01 +0100648/* Expect inbound connection from ASP/SCTP-client, followed by ASP-UP + ASP-ACT */
649testcase TC_clnt_asp_act_tmt_loadshare() runs on RAW_M3UA_CT {
650 f_init_common();
651 f_vty_config2(VTY, {"cs7 instance 0", "as as-client m3ua"}, "traffic-mode loadshare");
652 f_init_m3ua();
653 f_init_m3ua_srv();
654
655 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), tmt := c_M3UA_TMT_loadshare, rctx := int2oct(1055, 4));
656
657 f_clear_m3ua();
658}
659
Harald Welte829dac42019-11-05 16:55:30 +0100660/* Test traffic being routed through "server" side STP (M3UA SG), coming back in "client"
661 * side STP (M3UA ASP) */
662testcase TC_clnt_sg_to_asp() runs on RAW_M3UA_CT {
663 var OCT4 rctx_sender := int2oct(1023, 4);
664 var OCT4 pc_sender := int2oct(23, 4);
665 var OCT4 rctx_receiver := int2oct(1055, 4);
666 var OCT4 pc_receiver := int2oct(55, 4);
667
668 f_init_m3ua();
669 f_M3UA_asp_up_act(0);
670
671 f_init_m3ua_srv();
Harald Weltedfbca112019-12-01 18:00:48 +0100672 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(1055, 4));
Harald Welte829dac42019-11-05 16:55:30 +0100673
674 f_sleep(1.0);
675
676 /* verify traffic is routed from sender to [sole] receiver */
677 f_test_traffic(0, rctx_sender, pc_sender, M3UA_SRV(0), rctx_receiver, pc_receiver);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100678
679 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100680}
681
682/* Test traffic being routed through "client" side STP (M3UA ASP), coming back in "server"
683 * side STP (M3UA SG) */
684testcase TC_clnt_asp_to_sg() runs on RAW_M3UA_CT {
685 var OCT4 rctx_sender := int2oct(1055, 4);
686 var OCT4 pc_sender := int2oct(55, 4);
687 var OCT4 rctx_receiver := int2oct(1023, 4);
688 var OCT4 pc_receiver := int2oct(23, 4);
689
690 f_init_m3ua();
691 f_M3UA_asp_up_act(0);
692
693 f_init_m3ua_srv();
Harald Weltedfbca112019-12-01 18:00:48 +0100694 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(1055, 4));
Harald Welte829dac42019-11-05 16:55:30 +0100695
696 f_sleep(1.0);
697
698 /* verify traffic is routed from sender to [sole] receiver */
699 f_test_traffic(M3UA_SRV(0), rctx_sender, pc_sender, 0, rctx_receiver, pc_receiver);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100700 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100701}
702
703
Harald Welte0db44132019-10-17 11:09:05 +0200704
705control {
706 /* M3UA Tests */
707 execute( TC_connect_asp_up() );
708 execute( TC_beat() );
709 execute( TC_beat_payload() );
710 execute( TC_asp_act() );
711 execute( TC_asp_act_override() );
712 execute( TC_asp_act_loadshare() );
713 execute( TC_asp_act_broadcast() );
714 execute( TC_tmt_override() );
715 execute( TC_tmt_loadshare() );
716 execute( TC_tmt_broadcast() );
Harald Welte15a85012020-06-13 16:35:45 +0200717 execute( TC_act_rctx_data_no_rctx() );
Harald Welte0db44132019-10-17 11:09:05 +0200718
719 /* M3UA RKM tests */
720 execute( TC_rkm_reg_static_notpermitted() );
721 execute( TC_rkm_reg_static_permitted() );
722 execute( TC_rkm_reg_dynamic_permitted() );
723 execute( TC_rkm_unreg_never_registered() );
724 execute( TC_rkm_unreg_invalid() );
725 execute( TC_rkm_unreg_registered() );
726 execute( TC_rkm_unreg_active() );
727 /* TODO: test RKM with unsupported routing keys: NA, SI, OPC */
728 /* TODO: register/unregister multiple routing contexts in one message; including mixed
729 success/failure situations */
Harald Welte829dac42019-11-05 16:55:30 +0100730
731 /* Test STP as SCTP client + M3UA ASP role */
732 execute( TC_clnt_connect_asp_up() );
733 execute( TC_clnt_asp_act() );
734 execute( TC_clnt_sg_to_asp() );
735 execute( TC_clnt_asp_to_sg() );
Harald Welte53050cd2019-12-01 20:49:01 +0100736
737 /* put this one last as it changes the stp side config */
738 execute( TC_clnt_asp_act_tmt_loadshare() );
Harald Welte0db44132019-10-17 11:09:05 +0200739}
740
741
742
743}