blob: 22272f34ac427dc018fde7897bfda9c5b7a0926c [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
78friend function f_M3UA_send(integer idx, template (present) PDU_M3UA msg, template integer stream := 0)
79runs on RAW_M3UA_CT {
80 M3UA[idx].send(t_M3UA_Send(g_m3ua_conn_id[idx], msg, stream));
81}
82
83friend function f_M3UA_exp(integer idx, template (present) PDU_M3UA msg) runs on RAW_M3UA_CT {
84 var M3UA_RecvFrom rx;
85 alt {
86 [] M3UA[idx].receive(t_M3UA_RecvFrom(msg)) {
87 setverdict(pass);
88 }
89 [] M3UA[idx].receive(t_M3UA_RecvFrom(?)) -> value rx {
90 setverdict(fail, "Received unexpected M3UA[", idx, "] ", rx,
91 "while waiting for ", msg);
92 mtc.stop;
93 }
94 }
95}
96
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +020097private template (value) Socket ts_Socket(HostName hostName, PortNumber portNumber) := {
98 hostName := hostName,
99 portNumber := portNumber
100};
101
102private template (value) SctpTuple ts_SCTP(template (omit) integer ppid := 3,
103 template (omit) integer stream := 0,
104 template (omit) SocketList remSocks := omit) := {
105 sinfo_stream := stream,
106 sinfo_ppid := ppid,
107 remSocks := remSocks,
108 assocId := omit
109};
110
Harald Welte0db44132019-10-17 11:09:05 +0200111friend function f_M3UA_connect(integer i) runs on RAW_M3UA_CT {
112 var Result res;
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200113 var Option opt_add_local_addrs;
114 var OptionList opt_list := {};
115 var template SocketList opt_add_remote_addrs;
116
117 if (lengthof(mp_local_m3ua_ip) == 0 or lengthof(mp_stp_m3ua_ip) == 0) {
118 setverdict(fail, "Empty local or remote address trying to connect SCTP socket: ",
119 mp_local_m3ua_ip, " / ", mp_stp_m3ua_ip);
120 mtc.stop;
121 }
122
123 if (lengthof(mp_local_m3ua_ip) > 1) {
124 opt_add_local_addrs.sctpAdditionalLocalAddresses := substr(mp_local_m3ua_ip, 1,
125 lengthof(mp_local_m3ua_ip) - 1); //{mp_local_m3ua_ip};
126 opt_list := {opt_add_local_addrs};
127 }
128
129 if (lengthof(mp_stp_m3ua_ip) > 1) {
130 for (var integer j := 1; j < lengthof(mp_stp_m3ua_ip); j := j + 1) {
131 var Socket sk := valueof(ts_Socket(mp_stp_m3ua_ip[j], mp_stp_m3ua_port));
132 opt_add_remote_addrs[j - 1] := sk;
133 }
134 } else {
135 opt_add_remote_addrs := omit;
136 }
137
138 res := M3UA_CodecPort_CtrlFunct.f_IPL4_connect(M3UA[i], mp_stp_m3ua_ip[0], mp_stp_m3ua_port,
139 mp_local_m3ua_ip[0], mp_local_m3ua_port+i, 0,
140 {sctp:=valueof(ts_SCTP(3, 0, opt_add_remote_addrs))},
141 opt_list);
Harald Welte0db44132019-10-17 11:09:05 +0200142 if (not ispresent(res.connId)) {
143 setverdict(fail, "Could not connect M3UA socket, check your configuration");
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200144 mtc.stop;
Harald Welte0db44132019-10-17 11:09:05 +0200145 }
146 g_m3ua_conn_id[i] := res.connId;
147}
148
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100149friend function f_M3UA_close(integer i) runs on RAW_M3UA_CT {
150 var Result res;
151 res := M3UA_CodecPort_CtrlFunct.f_IPL4_close(M3UA[i], g_m3ua_conn_id[i], {sctp:=valueof(ts_SCTP)});
152 g_m3ua_conn_id[i] := 0;
153}
154
Harald Welte829dac42019-11-05 16:55:30 +0100155friend function f_M3UA_listen(integer i) runs on RAW_M3UA_CT {
156 var Result res;
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200157 var Option opt_add_local_addrs;
158 var OptionList opt_list := {};
159
160 if (lengthof(mp_local_m3ua_ip) == 0 ) {
161 setverdict(fail, "Empty local address trying to bind SCTP socket: ",
162 mp_local_m3ua_ip);
163 mtc.stop;
164 }
165
166 if (lengthof(mp_local_m3ua_ip) > 1) {
167 opt_add_local_addrs.sctpAdditionalLocalAddresses := substr(mp_local_m3ua_ip, 1,
168 lengthof(mp_local_m3ua_ip) - 1); //{mp_local_m3ua_ip};
169 opt_list := {opt_add_local_addrs};
170 }
171
172 res := M3UA_CodecPort_CtrlFunct.f_IPL4_listen(M3UA[i], mp_local_m3ua_ip[0], mp_local_m3ua_port+i,
173 {sctp:=valueof(ts_SCTP)}, opt_list);
Harald Welte829dac42019-11-05 16:55:30 +0100174 if (not ispresent(res.connId)) {
175 setverdict(fail, "Could not bind M3UA socket, check your configuration");
Pau Espin Pedrolb51c0872020-08-21 13:42:30 +0200176 mtc.stop;
Harald Welte829dac42019-11-05 16:55:30 +0100177 }
178}
179
Harald Welte0db44132019-10-17 11:09:05 +0200180friend function f_init_m3ua() runs on RAW_M3UA_CT {
181 var integer i;
182
183 f_init_common();
184
185 activate(as_m3ua_sctp());
186
187 for (i := 0; i < NR_M3UA; i:=i+1) {
188 map(self:M3UA[i], system:M3UA_CODEC_PT);
189 f_M3UA_connect(i);
190 }
191}
192
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100193friend function f_clear_m3ua() runs on RAW_M3UA_CT {
194 var integer i;
195
196 log("Clearing M3UA...");
197
198 for (i := 0; i < NR_M3UA; i:=i+1) {
199 f_M3UA_close(i);
200 }
201 /* Wait for recovery timer to trigger and shutdown all AS: */
202 f_sleep(int2float(mp_recovery_timeout_msec)/1000.0 + 0.5);
203 setverdict(pass, "M3UA cleared");
204}
205
Harald Welte829dac42019-11-05 16:55:30 +0100206friend function f_init_m3ua_srv() runs on RAW_M3UA_CT {
207 var integer i;
208 var PortEvent sctp_evt;
209
210 for (i := NR_M3UA; i < NR_M3UA+NR_M3UA_SRV; i:=i+1) {
211 map(self:M3UA[i], system:M3UA_CODEC_PT);
212 /* bind+ listen */
213 f_M3UA_listen(i);
214 /* wait for accept() */
215 M3UA[i].receive(tr_SctpConnOpened) -> value sctp_evt {
216 g_m3ua_conn_id[i] := sctp_evt.connOpened.connId;
217 }
218 }
219}
220
221
222/***********************************************************************
223 * Test the STP in M3UA SG role (we are ASP)
224 ***********************************************************************/
225
Harald Welte0db44132019-10-17 11:09:05 +0200226/* perform an outbound ASP-UP procedure */
227friend function f_M3UA_asp_up(integer idx, template (omit) OCT4 aspid := omit) runs on RAW_M3UA_CT {
228 f_M3UA_send(idx, ts_M3UA_ASPUP(aspid));
229 f_M3UA_exp(idx, tr_M3UA_ASPUP_ACK);
230}
231
232/* perform an outbound BEAT procedure */
233friend function f_M3UA_beat(integer idx, template (omit) octetstring hbd) runs on RAW_M3UA_CT {
234 if (istemplatekind(hbd, "omit")) {
235 f_M3UA_send(idx, ts_M3UA_BEAT(omit));
236 f_M3UA_exp(idx, tr_M3UA_BEAT_ACK(omit));
237 } else {
238 f_M3UA_send(idx, ts_M3UA_BEAT(ts_M3UA_hb_data(hbd)));
239 f_M3UA_exp(idx, tr_M3UA_BEAT_ACK(tr_M3UA_hb_data(hbd)));
240 }
241}
242
243/* perform an outbound ASP-ACTIVATE procedure */
244friend function f_M3UA_asp_act(integer idx, template (omit) M3UA_Traffic_Mode_Type tmt := omit,
245 template (omit) OCT4 rctx := omit) runs on RAW_M3UA_CT {
246 f_M3UA_send(idx, ts_M3UA_ASPAC(tmt, rctx));
247 f_M3UA_exp(idx, tr_M3UA_ASPAC_ACK(tmt, rctx));
248}
249
250/* perform outbound ASP-UP and ASP-ACT, optionally expect interemittent NOTIFY */
251friend function f_M3UA_asp_up_act(integer idx, template (omit) M3UA_Traffic_Mode_Type tmt := omit,
252 template (omit) OCT4 rctx := omit,
253 template (omit) OCT2 ntfy_after_up := c_M3UA_ST_I_AS_INACTIVE,
254 template (omit) OCT2 ntfy_after_act := c_M3UA_ST_I_AS_ACTIVE)
255runs on RAW_M3UA_CT {
256 f_M3UA_asp_up(idx, omit);
257 if (not istemplatekind(ntfy_after_up, "omit")) {
258 f_M3UA_exp(idx, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_up, *));
259 }
260 f_M3UA_asp_act(idx, tmt, rctx);
261 if (not istemplatekind(ntfy_after_act, "omit")) {
262 f_M3UA_exp(idx, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_act, *));
263 }
264}
265
266
267/* Test the ASP-UP procedure */
268testcase TC_connect_asp_up() runs on RAW_M3UA_CT {
269 f_init_m3ua();
270 f_M3UA_asp_up(0);
271 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 +0100272 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200273}
274
275/* Test the heartbeat procedure without optional heartbeat data payload */
276testcase TC_beat() runs on RAW_M3UA_CT {
277 f_init_m3ua();
278 f_M3UA_asp_up(0);
279 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_INACTIVE, *));
280 f_M3UA_beat(0, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100281 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200282}
283
284/* Test the heartbeat procedure with optional heartbeat data payload */
285testcase TC_beat_payload() runs on RAW_M3UA_CT {
286 f_init_m3ua();
287 f_M3UA_asp_up(0);
288 f_M3UA_exp(0, tr_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, c_M3UA_ST_I_AS_INACTIVE, *));
289 f_M3UA_beat(0, 'a1a2a3a4a5'O);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100290 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200291}
292
293/* Test the ASP-ACTIVATE procedure (without traffic-mode or routing ctx) */
294testcase TC_asp_act() runs on RAW_M3UA_CT {
295 f_init_m3ua();
296 f_M3UA_asp_up_act(0);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100297 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200298}
299
300/* Test the ASP-ACTIVATE procedure with traffic-mode override */
301testcase TC_asp_act_override() runs on RAW_M3UA_CT {
302 f_init_m3ua();
303 f_M3UA_asp_up_act(0, c_M3UA_TMT_override, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100304 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200305}
306
307/* Test the ASP-ACTIVATE procedure with traffic-mode override */
308testcase TC_asp_act_loadshare() runs on RAW_M3UA_CT {
309 f_init_m3ua();
310 f_M3UA_asp_up_act(0, c_M3UA_TMT_loadshare, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100311 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200312}
313
314/* Test the ASP-ACTIVATE procedure with traffic-mode broadcast */
315testcase TC_asp_act_broadcast() runs on RAW_M3UA_CT {
316 f_init_m3ua();
317 f_M3UA_asp_up_act(0, c_M3UA_TMT_broadcast, omit);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100318 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200319}
320
Harald Welte15a85012020-06-13 16:35:45 +0200321/* test whether the STP accepts M3UA DATA without Routing Context IE */
322testcase TC_act_rctx_data_no_rctx() runs on RAW_M3UA_CT {
323 var OCT4 rctx_sender := int2oct(1023, 4);
324 var OCT4 pc_sender := int2oct(23, 4);
325 var OCT4 rctx_receiver := int2oct(1042, 4);
326 var OCT4 pc_receiver := int2oct(42, 4);
327
328 f_init_m3ua();
329 /* bring up the sender specifying a routing context */
330
331 f_M3UA_asp_up_act(0, rctx := rctx_sender);
332 f_M3UA_asp_up_act(1);
333
334 /* check if DATA is accepted without Routing Context IE */
335 f_test_traffic(0, omit, pc_sender, 1, rctx_receiver, pc_receiver);
336
337 f_clear_m3ua();
338}
339
Harald Welte0db44132019-10-17 11:09:05 +0200340/* Test if traffic is routed from idx_tx/pc_tx to idx_rx/pc_rx */
341private function f_test_traffic(integer idx_tx, template (omit) OCT4 rctx_sender, OCT4 pc_tx,
342 integer idx_rx, template (omit) OCT4 rctx_receiver, OCT4 pc_rx,
343 OCT1 si := '23'O, OCT1 ni := '00'O, OCT1 mp := '00'O, OCT1 sls := '00'O)
344runs on RAW_M3UA_CT {
345 var octetstring data := f_rnd_octstring(f_rnd_int(100));
346 f_M3UA_send(idx_tx, ts_M3UA_DATA(rctx_sender,
347 ts_M3UA_protocol_data(pc_tx, pc_rx, si, ni, mp, sls, data)), 1);
348 f_M3UA_exp(idx_rx, tr_M3UA_DATA(rctx_receiver,
349 tr_M3UA_protocol_data(pc_tx, pc_rx, si, ni, mp, sls, data)));
350}
351
352
353/* test "traffic-mode override" behavior */
354testcase TC_tmt_override() runs on RAW_M3UA_CT {
355 var OCT4 rctx_sender := int2oct(1023, 4);
356 var OCT4 pc_sender := int2oct(23, 4);
357 var OCT4 rctx_receiver := int2oct(1042, 4);
358 var OCT4 pc_receiver := int2oct(42, 4);
359
360 f_init_m3ua();
361
362 /* bring up the 'sender' side (single ASP in AS) */
363 f_M3UA_asp_up_act(0, omit, omit);
364
365 /* activate the first 'receiver' side ASP */
366 f_M3UA_asp_up_act(1, c_M3UA_TMT_override, rctx_receiver);
367
368 /* verify traffic is routed from sender to [sole] receiver */
369 f_test_traffic(0, rctx_sender, pc_sender, 1, rctx_receiver, pc_receiver);
370
371 /* activate the second 'receiver' side ASP (no NOTIFY as AS state doesn't change) */
372 f_M3UA_asp_up_act(2, c_M3UA_TMT_override, rctx_receiver, omit, omit);
373
374 /* we expect a NOTIFY to the *other* ASP Other/Alternat-ASP-Active */
375 f_M3UA_exp(1, tr_M3UA_NOTIFY(c_M3UA_ST_T_OTHER, c_M3UA_ST_I_ALTERNATE_ASP, *));
376
377 /* verify traffic is routed from sender to new receiver */
378 f_test_traffic(0, rctx_sender, pc_sender, 2, rctx_receiver, pc_receiver);
379
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100380 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200381}
382
383private altstep as_count_rx(integer idx, template (present) PDU_M3UA exp, inout integer counter)
384runs on RAW_M3UA_CT {
385 [] M3UA[idx].receive(t_M3UA_RecvFrom(exp)) {
386 counter := counter + 1;
387 }
388}
389
390/* test "traffic-mode load-share" behavior */
391testcase TC_tmt_loadshare() runs on RAW_M3UA_CT {
392 var OCT4 rctx_sender := int2oct(1023, 4);
393 var OCT4 pc_sender := int2oct(23, 4);
394 var OCT4 rctx_receiver := int2oct(1042, 4);
395 var OCT4 pc_receiver := int2oct(42, 4);
396 var integer i;
397
398 f_init_m3ua();
399
400 /* FIXME: configure the STP via VTY to set traffic-mode */
401
402 /* bring up the 'sender' side (single ASP in AS) */
403 f_M3UA_asp_up_act(0, omit, rctx_sender);
404
405 /* activate the first 'receiver' side ASP */
Harald Welte0db44132019-10-17 11:09:05 +0200406 f_M3UA_asp_up_act(1, c_M3UA_TMT_loadshare, omit); // TODO: rctx
407
408 /* verify traffic is routed from sender to [sole] receiver */
409 for (i := 0; i < 10; i := i+1) {
410 f_test_traffic(0, rctx_sender, pc_sender, 1, rctx_receiver, pc_receiver);
411 }
412
413 /* activate the second 'receiver' side ASP (no NOTIFY) */
414 f_M3UA_asp_up_act(2, c_M3UA_TMT_loadshare, omit, omit, omit); // TODO: rctx
415
416 /* verify traffic is routed from sender to new receiver */
417 const integer iter_per_asp := 5;
418 var integer num_rx[NR_M3UA] := { 0, 0, 0 };
419 for (i := 0; i < 2*iter_per_asp; i := i+1) {
420 var octetstring data := f_rnd_octstring(f_rnd_int(100));
421 var template (value) M3UA_Protocol_Data tx_pd;
422 var template (present) M3UA_Protocol_Data rx_pd;
423 tx_pd := ts_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
424 rx_pd := tr_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
425 f_M3UA_send(0, ts_M3UA_DATA(rctx_sender, tx_pd), 1);
426 alt {
Pau Espin Pedrol4d915782020-08-21 12:24:35 +0200427 [] as_count_rx(1, tr_M3UA_DATA(rctx_receiver, rx_pd), num_rx[1]);
428 [] as_count_rx(2, tr_M3UA_DATA(rctx_receiver, rx_pd), num_rx[2]);
Harald Welte0db44132019-10-17 11:09:05 +0200429 }
430 }
431 /* FIXME: check for extraneous messages? */
432 for (i := 1; i <= 2; i := i+1) {
433 if (num_rx[i] != iter_per_asp) {
Harald Welteb9f7bbb2019-10-29 09:07:11 +0100434 setverdict(fail, "Received ", num_rx[i], " out of expected ", iter_per_asp,
435 "M3UA DATA messages at M3UA port ", i);
Harald Welte0db44132019-10-17 11:09:05 +0200436 }
437 }
438 setverdict(pass);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100439
440 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200441}
442
443/* test "traffic-mode broadcast" behavior */
444testcase TC_tmt_broadcast() runs on RAW_M3UA_CT {
445 var OCT4 rctx_sender := int2oct(1023, 4);
446 var OCT4 pc_sender := int2oct(23, 4);
447 var OCT4 rctx_receiver := int2oct(1042, 4);
448 var OCT4 pc_receiver := int2oct(42, 4);
449 var integer i;
450
451 f_init_m3ua();
452
453 /* FIXME: configure the STP via VTY to set traffic-mode */
454
455 /* bring up the 'sender' side (single ASP in AS) */
456 f_M3UA_asp_up_act(0, omit, omit); // TODO: rctx
457
458 /* activate the first 'receiver' side ASP */
Harald Welte0db44132019-10-17 11:09:05 +0200459 f_M3UA_asp_up_act(1, c_M3UA_TMT_broadcast, omit); // TODO: rctx
460
461 /* verify traffic is routed from sender to [sole] receiver */
462 for (i := 0; i < 10; i := i+1) {
463 f_test_traffic(0, rctx_sender, pc_sender, 1, rctx_receiver, pc_receiver);
464 }
465
466 /* activate the second 'receiver' side ASP */
467 f_M3UA_asp_up_act(2, c_M3UA_TMT_broadcast, omit, omit, omit); // TODO: rctx
468
469 /* verify traffic is routed from sender to new receiver */
470 for (i := 0; i < 10; i := i+1) {
471 var octetstring data := f_rnd_octstring(f_rnd_int(100));
472 var template (value) M3UA_Protocol_Data tx_pd;
473 var template (present) M3UA_Protocol_Data rx_pd;
474 tx_pd := ts_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
475 rx_pd := tr_M3UA_protocol_data(pc_sender, pc_receiver, '23'O, '00'O, '00'O, '00'O, data);
476 f_M3UA_send(0, ts_M3UA_DATA(rctx_sender, tx_pd), 1);
477 /* each message must be received both on 1 and 2 */
478 f_M3UA_exp(1, tr_M3UA_DATA(rctx_receiver, rx_pd));
479 f_M3UA_exp(2, tr_M3UA_DATA(rctx_receiver, rx_pd));
480 }
481 setverdict(pass);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100482
483 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200484}
485
486private function f_M3UA_rkm_register(OCT4 id, OCT3 dpc, OCT4 rctx,
487 template (present) OCT4 exp_status := c_M3UA_REGSTS_SUCCESS)
488runs on RAW_M3UA_CT
489{
490 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:=id, dpc:=dpc, rctx:=rctx)}));
491 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:=id, status:=exp_status, rctx:=rctx)}));
492}
493
494/* Send RKM registration; expect -EPERM as RCTX doesn't match config and dynamic not permitted */
495testcase TC_rkm_reg_static_notpermitted() runs on RAW_M3UA_CT {
496 f_init_m3ua();
497
498 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:='00000099'O, dpc:='aabbcc'O)}));
499 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:='00000099'O, status:=c_M3UA_REGSTS_ERR_EPERM,
500 rctx:=?)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100501
502 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200503}
504
505/* Send RKM registration; expect OK as RCTX does match config */
506testcase TC_rkm_reg_static_permitted() runs on RAW_M3UA_CT {
507 var OCT3 dpc := int2oct(23, 3); // must match config
508 var OCT4 rctx := int2oct(1023, 4); // must match config
509
510 f_init_m3ua();
511
512 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:='10000099'O, dpc:=dpc, rctx:=rctx)}));
513 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:='10000099'O, status:=c_M3UA_REGSTS_SUCCESS,
514 rctx:=rctx)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100515
516 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200517}
518
519/* Send RKM registration; expect OK as dynamic not permitted */
520testcase TC_rkm_reg_dynamic_permitted() runs on RAW_M3UA_CT {
521 f_init_common();
522 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation dynamic-permitted");
523 f_init_m3ua();
524
525 f_M3UA_send(0, ts_M3UA_REG_REQ({ts_M3UA_rkey(id:='20000099'O, dpc:='aabbcc'O)}));
526 f_M3UA_exp(0, tr_M3UA_REG_RSP({tr_M3UA_reg_res(id:='20000099'O, status:=c_M3UA_REGSTS_SUCCESS,
527 rctx:=?)}));
528
529 f_vty_config2(VTY, {"cs7 instance 0"}, "xua rkm routing-key-allocation static-only");
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100530
531 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200532}
533
534/* try to de-register a routing key that was never registered -> error */
535testcase TC_rkm_unreg_never_registered() runs on RAW_M3UA_CT {
536 f_init_m3ua();
537 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(int2oct(1023,4))));
538 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 +0100539 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200540}
541
542/* try to de-register a routing key that is invalid (non-existant) -> error */
543testcase TC_rkm_unreg_invalid() runs on RAW_M3UA_CT {
544 f_init_m3ua();
545 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(int2oct(1234,4))));
546 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 +0100547 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200548}
549
550/* try to de-register a routing key that was registered -> OK*/
551testcase TC_rkm_unreg_registered() runs on RAW_M3UA_CT {
552 f_init_m3ua();
553 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(int2oct(1023,4))));
554 f_M3UA_exp(0, tr_M3UA_DEREG_RSP({tr_M3UA_dereg_res(?,c_m3UA_DEREGSTS_SUCCESS)}));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100555 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200556}
557
558/* try to de-register a routing key for an active ASP -> ERROR */
559testcase TC_rkm_unreg_active() runs on RAW_M3UA_CT {
560 var OCT3 dpc := int2oct(23, 3); // must match config
561 var OCT4 rctx := int2oct(1023, 4); // must match config
562
563 f_init_m3ua();
564
565 /* first register the routing key */
566 f_M3UA_rkm_register(id:='30000099'O, dpc:=dpc, rctx:=rctx);
567
568 /* then activate the ASP */
569 f_M3UA_asp_up_act(0);
570
571 /* then try to de-regsiter */
572 f_M3UA_send(0, ts_M3UA_DEREG_REQ(ts_M3UA_routing_ctx(rctx)));
573 f_M3UA_exp(0, tr_M3UA_DEREG_RSP({tr_M3UA_dereg_res(?,c_m3UA_DEREGSTS_ERR_ASP_ACTIVE)}));
574 /* FIXME: we now may have changed the state on the STP side! */
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100575
576 f_clear_m3ua();
Harald Welte0db44132019-10-17 11:09:05 +0200577}
578
Harald Welte829dac42019-11-05 16:55:30 +0100579/***********************************************************************
580 * Test the STP in M3UA ASP role (we are SG)
581 ***********************************************************************/
582
583/* expect/perform an inbound ASP-UP procedure */
584friend function f_M3UA_CLNT_asp_up(integer idx, template OCT4 aspid := omit) runs on RAW_M3UA_CT {
585 f_M3UA_exp(idx, tr_M3UA_ASPUP(aspid));
586 f_M3UA_send(idx, ts_M3UA_ASPUP_ACK);
587}
588
589/* expect/perform an inbound ASP-ACTIVATE procedure */
590friend function f_M3UA_CLNT_asp_act(integer idx, template M3UA_Traffic_Mode_Type tmt := omit,
591 template (omit) OCT4 rctx := omit) runs on RAW_M3UA_CT {
592 f_M3UA_exp(idx, tr_M3UA_ASPAC(tmt, rctx));
593 f_M3UA_send(idx, ts_M3UA_ASPAC_ACK(tmt, rctx));
594}
595
596/* expect/perform inbound ASP-UP and ASP-ACT, optionally send interemittent NOTIFY */
597friend function f_M3UA_CLNT_asp_up_act(integer idx, template M3UA_Traffic_Mode_Type tmt := omit,
598 template OCT4 rctx := omit,
599 template (omit) OCT2 ntfy_after_up := c_M3UA_ST_I_AS_INACTIVE,
600 template (omit) OCT2 ntfy_after_act := c_M3UA_ST_I_AS_ACTIVE)
601runs on RAW_M3UA_CT {
602 f_M3UA_CLNT_asp_up(idx, omit);
603 if (not istemplatekind(ntfy_after_up, "omit")) {
604 f_M3UA_send(idx, ts_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_up, rctx));
605 }
606 f_M3UA_CLNT_asp_act(idx, tmt, rctx);
607 if (not istemplatekind(ntfy_after_act, "omit")) {
608 f_M3UA_send(idx, ts_M3UA_NOTIFY(c_M3UA_ST_T_STATE_CHG, ntfy_after_act, rctx));
609 }
610}
611
612
613/* Expect inbound connection from ASP/SCTP-client, followed by ASP-UP */
614testcase TC_clnt_connect_asp_up() runs on RAW_M3UA_CT {
615 f_init_m3ua();
616 f_init_m3ua_srv();
617
618 f_M3UA_CLNT_asp_up(M3UA_SRV(0));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100619
620 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100621}
622
623/* Expect inbound connection from ASP/SCTP-client, followed by ASP-UP + ASP-ACT */
624testcase TC_clnt_asp_act() runs on RAW_M3UA_CT {
625 f_init_m3ua();
626 f_init_m3ua_srv();
627
Harald Weltedfbca112019-12-01 18:00:48 +0100628 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(1055, 4));
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100629
630 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100631}
632
Harald Welte53050cd2019-12-01 20:49:01 +0100633/* Expect inbound connection from ASP/SCTP-client, followed by ASP-UP + ASP-ACT */
634testcase TC_clnt_asp_act_tmt_loadshare() runs on RAW_M3UA_CT {
635 f_init_common();
636 f_vty_config2(VTY, {"cs7 instance 0", "as as-client m3ua"}, "traffic-mode loadshare");
637 f_init_m3ua();
638 f_init_m3ua_srv();
639
640 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), tmt := c_M3UA_TMT_loadshare, rctx := int2oct(1055, 4));
641
642 f_clear_m3ua();
643}
644
Harald Welte829dac42019-11-05 16:55:30 +0100645/* Test traffic being routed through "server" side STP (M3UA SG), coming back in "client"
646 * side STP (M3UA ASP) */
647testcase TC_clnt_sg_to_asp() runs on RAW_M3UA_CT {
648 var OCT4 rctx_sender := int2oct(1023, 4);
649 var OCT4 pc_sender := int2oct(23, 4);
650 var OCT4 rctx_receiver := int2oct(1055, 4);
651 var OCT4 pc_receiver := int2oct(55, 4);
652
653 f_init_m3ua();
654 f_M3UA_asp_up_act(0);
655
656 f_init_m3ua_srv();
Harald Weltedfbca112019-12-01 18:00:48 +0100657 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(1055, 4));
Harald Welte829dac42019-11-05 16:55:30 +0100658
659 f_sleep(1.0);
660
661 /* verify traffic is routed from sender to [sole] receiver */
662 f_test_traffic(0, rctx_sender, pc_sender, M3UA_SRV(0), rctx_receiver, pc_receiver);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100663
664 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100665}
666
667/* Test traffic being routed through "client" side STP (M3UA ASP), coming back in "server"
668 * side STP (M3UA SG) */
669testcase TC_clnt_asp_to_sg() runs on RAW_M3UA_CT {
670 var OCT4 rctx_sender := int2oct(1055, 4);
671 var OCT4 pc_sender := int2oct(55, 4);
672 var OCT4 rctx_receiver := int2oct(1023, 4);
673 var OCT4 pc_receiver := int2oct(23, 4);
674
675 f_init_m3ua();
676 f_M3UA_asp_up_act(0);
677
678 f_init_m3ua_srv();
Harald Weltedfbca112019-12-01 18:00:48 +0100679 f_M3UA_CLNT_asp_up_act(M3UA_SRV(0), rctx := int2oct(1055, 4));
Harald Welte829dac42019-11-05 16:55:30 +0100680
681 f_sleep(1.0);
682
683 /* verify traffic is routed from sender to [sole] receiver */
684 f_test_traffic(M3UA_SRV(0), rctx_sender, pc_sender, 0, rctx_receiver, pc_receiver);
Pau Espin Pedrol05726dd2019-11-06 15:55:57 +0100685 f_clear_m3ua();
Harald Welte829dac42019-11-05 16:55:30 +0100686}
687
688
Harald Welte0db44132019-10-17 11:09:05 +0200689
690control {
691 /* M3UA Tests */
692 execute( TC_connect_asp_up() );
693 execute( TC_beat() );
694 execute( TC_beat_payload() );
695 execute( TC_asp_act() );
696 execute( TC_asp_act_override() );
697 execute( TC_asp_act_loadshare() );
698 execute( TC_asp_act_broadcast() );
699 execute( TC_tmt_override() );
700 execute( TC_tmt_loadshare() );
701 execute( TC_tmt_broadcast() );
Harald Welte15a85012020-06-13 16:35:45 +0200702 execute( TC_act_rctx_data_no_rctx() );
Harald Welte0db44132019-10-17 11:09:05 +0200703
704 /* M3UA RKM tests */
705 execute( TC_rkm_reg_static_notpermitted() );
706 execute( TC_rkm_reg_static_permitted() );
707 execute( TC_rkm_reg_dynamic_permitted() );
708 execute( TC_rkm_unreg_never_registered() );
709 execute( TC_rkm_unreg_invalid() );
710 execute( TC_rkm_unreg_registered() );
711 execute( TC_rkm_unreg_active() );
712 /* TODO: test RKM with unsupported routing keys: NA, SI, OPC */
713 /* TODO: register/unregister multiple routing contexts in one message; including mixed
714 success/failure situations */
Harald Welte829dac42019-11-05 16:55:30 +0100715
716 /* Test STP as SCTP client + M3UA ASP role */
717 execute( TC_clnt_connect_asp_up() );
718 execute( TC_clnt_asp_act() );
719 execute( TC_clnt_sg_to_asp() );
720 execute( TC_clnt_asp_to_sg() );
Harald Welte53050cd2019-12-01 20:49:01 +0100721
722 /* put this one last as it changes the stp side config */
723 execute( TC_clnt_asp_act_tmt_loadshare() );
Harald Welte0db44132019-10-17 11:09:05 +0200724}
725
726
727
728}