blob: 34090d2a834a9b6a25c4ab1b305a3cea62014a3f [file] [log] [blame]
Harald Welte4526da92020-03-05 23:08:10 +01001module PGW_Tests {
2
3import from General_Types all;
4import from Osmocom_Types all;
5import from Native_Functions all;
6
7import from GTPv2_Types all;
8import from GTPv2_Templates all;
9import from GTPv2_Emulation all;
10
11import from UECUPS_Types all;
12
13import from DNS_Helpers all;
14
Harald Weltef4001512020-04-26 21:48:34 +020015
16import from DIAMETER_Types all;
17import from DIAMETER_Templates all;
18import from DIAMETER_Emulation all;
19
20
Harald Welte4526da92020-03-05 23:08:10 +010021modulepar {
Vadim Yanitskiy589972f2022-01-20 19:38:16 +060022 charstring mp_pgw_hostname := "127.0.0.4";
Harald Welte4526da92020-03-05 23:08:10 +010023 charstring mp_local_hostname_c := "127.0.0.1";
24 charstring mp_local_hostname_u := "127.0.0.1";
Harald Weltef4001512020-04-26 21:48:34 +020025
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +060026 charstring mp_run_prog_log_path := "/tmp";
Harald Welte4526da92020-03-05 23:08:10 +010027 charstring mp_run_prog_as_user := "laforge";
28 charstring mp_ping_hostname := "10.45.0.1";
Harald Weltef4001512020-04-26 21:48:34 +020029
Vadim Yanitskiy589972f2022-01-20 19:38:16 +060030 charstring mp_pcrf_local_ip := "127.0.0.9";
Harald Weltef4001512020-04-26 21:48:34 +020031 integer mp_pcrf_local_port := 3868;
Harald Welte4526da92020-03-05 23:08:10 +010032}
33
34/* main component, we typically have one per testcase */
35type component PGW_Test_CT {
36 var GTPv2_Emulation_CT vc_GTP2;
37 port GTP2EM_PT TEID0;
Harald Weltef4001512020-04-26 21:48:34 +020038
39 /* emulated PCRF */
40 var DIAMETER_Emulation_CT vc_DIAMETER;
41 port DIAMETER_PT DIAMETER_UNIT;
42 port DIAMETEREM_PROC_PT DIAMETER_PROC;
Harald Welte4526da92020-03-05 23:08:10 +010043}
44
45/* per-session component; we typically have 1..N per testcase */
Harald Weltef4001512020-04-26 21:48:34 +020046type component PGW_Session_CT extends GTP2_ConnHdlr, DIAMETER_ConnHdlr {
Harald Welte4526da92020-03-05 23:08:10 +010047 var SessionPars g_pars;
48
49 /* TEI (Data) local side */
50 var OCT4 g_teid;
51 /* TEI (Control) local side */
52 var OCT4 g_teic;
53 /* TEI (Data) remote side */
54 var OCT4 g_teid_remote;
55 /* TEI (Control) remote side */
56 var OCT4 g_teic_remote;
57 /* GTP-U IPv4 address remote sie */
58 var OCT4 g_gtpu4_remote;
59 var OCT16 g_gtpu6_remote;
60
61 /* Address allocation */
62 var OCT4 g_ip4_addr;
63 var OCT16 g_ip6_addr;
64 var integer g_ip6_plen;
65}
66
67/* configuration data for a given Session */
68type record SessionPars {
69 hexstring imsi,
70 octetstring msisdn optional,
71 // serving network
72 integer rat_type,
73 // flags?
74 charstring apn,
75 /* Apn subscribed or non-subscribed */
76 boolean selection_mode,
77 BIT3 pdn_type,
78 /* PAA */
79 /* Max APN Restriction */
80 /* APN-AMBR */
81 octetstring pco optional,
82 octetstring epco optional,
83 /* Bearer Contexts to be created */
84
85 charstring tun_dev_name,
86 charstring tun_netns_name optional
87}
88
89template (value) SessionPars
90t_SessionPars(hexstring imsi, charstring tundev, integer rat_type := 6, charstring apn := "internet",
91 boolean selection_mode := false, BIT3 pdn_type := '001'B) := {
92 imsi := imsi,
93 msisdn := omit,
94 rat_type := rat_type,
95 apn := apn,
96 selection_mode := selection_mode,
97 pdn_type := pdn_type,
98 pco := omit,
99 epco := omit,
100 tun_dev_name := tundev,
101 tun_netns_name := tundev
102}
103
104type record BearerConfig {
105 integer eps_bearer_id
106}
107
108type function void_fn() runs on PGW_Session_CT;
109
Harald Weltef4001512020-04-26 21:48:34 +0200110friend function DiameterForwardUnitdataCallback(PDU_DIAMETER msg)
111runs on DIAMETER_Emulation_CT return template PDU_DIAMETER {
112 DIAMETER_UNIT.send(msg);
113 return omit;
114}
115
116friend function f_init_diameter(charstring id) runs on PGW_Test_CT {
117 var DIAMETEROps ops := {
118 create_cb := refers(DIAMETER_Emulation.ExpectedCreateCallback),
Vadim Yanitskiyb46f01e2021-12-06 03:23:13 +0300119 unitdata_cb := refers(DiameterForwardUnitdataCallback),
120 raw := false /* handler mode (IMSI based routing) */
Harald Weltef4001512020-04-26 21:48:34 +0200121 };
122 var DIAMETER_conn_parameters pars := {
123 remote_ip := mp_pgw_hostname,
124 remote_sctp_port := -1,
125 local_ip := mp_pcrf_local_ip,
126 local_sctp_port := mp_pcrf_local_port,
127 origin_host := "pcrf.localdomain",
128 origin_realm := "localdomain",
129 vendor_app_id := c_DIAMETER_3GPP_Gx_AID
130 };
131 vc_DIAMETER := DIAMETER_Emulation_CT.create(id);
132 map(vc_DIAMETER:DIAMETER, system:DIAMETER_CODEC_PT);
133 connect(vc_DIAMETER:DIAMETER_UNIT, self:DIAMETER_UNIT);
134 connect(vc_DIAMETER:DIAMETER_PROC, self:DIAMETER_PROC);
135 vc_DIAMETER.start(DIAMETER_Emulation.main(ops, pars, id));
136
137 f_diameter_wait_capability(DIAMETER_UNIT);
138}
139
Harald Welte4526da92020-03-05 23:08:10 +0100140private function f_init() runs on PGW_Test_CT {
141 var Gtp2EmulationCfg cfg := {
142 gtpc_bind_ip := mp_local_hostname_c,
143 gtpc_bind_port := GTP2C_PORT,
144 gtpc_remote_ip := mp_pgw_hostname,
145 gtpc_remote_port := GTP2C_PORT,
146 sgw_role := true,
147 use_gtpu_daemon := true
148 };
149
150 vc_GTP2 := GTPv2_Emulation_CT.create("GTP2_EM");
151 map(vc_GTP2:GTP2C, system:GTP2C);
152 connect(vc_GTP2:TEID0, self:TEID0);
153 vc_GTP2.start(GTPv2_Emulation.main(cfg));
Harald Weltef4001512020-04-26 21:48:34 +0200154
155 if (mp_pcrf_local_ip != "") {
156 f_init_diameter(testcasename());
157 }
Harald Welte4526da92020-03-05 23:08:10 +0100158}
159
160function f_start_handler(void_fn fn, template (omit) SessionPars pars := omit)
161runs on PGW_Test_CT return PGW_Session_CT {
162 var charstring id := testcasename();
163 var PGW_Session_CT vc_conn;
164 vc_conn := PGW_Session_CT.create(id);
165 connect(vc_conn:GTP2, vc_GTP2:CLIENT);
166 connect(vc_conn:GTP2_PROC, vc_GTP2:CLIENT_PROC);
Harald Weltef4001512020-04-26 21:48:34 +0200167
168 if (isbound(vc_DIAMETER)) {
169 connect(vc_conn:DIAMETER, vc_DIAMETER:DIAMETER_CLIENT);
170 connect(vc_conn:DIAMETER_PROC, vc_DIAMETER:DIAMETER_PROC);
171 }
172
Harald Welte4526da92020-03-05 23:08:10 +0100173 vc_conn.start(f_handler_init(fn, pars));
174 return vc_conn;
175}
176
177private function f_handler_init(void_fn fn, template (omit) SessionPars pars := omit)
178runs on PGW_Session_CT {
179 if (isvalue(pars)) {
180 g_pars := valueof(pars);
181 }
Harald Weltef4001512020-04-26 21:48:34 +0200182 if (DIAMETER_PROC.checkstate("Connected")) {
183 f_diameter_expect(g_pars.imsi);
184 }
Harald Welte4526da92020-03-05 23:08:10 +0100185 fn.apply();
186}
187
Harald Weltefe595e42020-04-21 22:56:47 +0200188private function f_concat_pad(integer tot_len, hexstring prefix, integer suffix) return hexstring {
189 var integer suffix_len := tot_len - lengthof(prefix);
190 var charstring suffix_ch := int2str(suffix);
191 var integer pad_len := suffix_len - lengthof(suffix_ch);
192
193 return prefix & int2hex(0, pad_len) & str2hex(suffix_ch);
194}
195
196function f_gen_imei(integer suffix) return hexstring {
197 return f_concat_pad(14, '49999'H, suffix);
198}
199
200function f_gen_imsi(integer suffix) return hexstring {
201 return f_concat_pad(15, '26242'H, suffix);
202}
203
204function f_gen_msisdn(integer suffix) return hexstring {
205 return f_concat_pad(12, '49123'H, suffix);
206}
207
Harald Weltef4001512020-04-26 21:48:34 +0200208private altstep as_DIA_CCR(DCC_NONE_CC_Request_Type req_type) runs on PGW_Session_CT {
209 var PDU_DIAMETER rx_dia;
210 [] DIAMETER.receive(tr_DIA_CCR(req_type := req_type)) -> value rx_dia {
211 var template (omit) AVP avp;
212 var octetstring sess_id;
213 var AVP_Unsigned32 req_num;
214
215 avp := f_DIAMETER_get_avp(rx_dia, c_AVP_Code_BASE_NONE_Session_Id);
216 sess_id := valueof(avp.avp_data.avp_BASE_NONE_Session_Id);
217
218 avp := f_DIAMETER_get_avp(rx_dia, c_AVP_Code_DCC_NONE_CC_Request_Number);
219 req_num := valueof(avp.avp_data.avp_DCC_NONE_CC_Request_Number);
220
221 DIAMETER.send(ts_DIA_CCA(rx_dia.hop_by_hop_id, rx_dia.end_to_end_id, sess_id,
222 req_type, req_num));
223 }
224 [] DIAMETER.receive(PDU_DIAMETER:?) -> value rx_dia {
225 setverdict(fail, "Received unexpected DIAMETER ", rx_dia);
226 self.stop;
227 }
228}
229
Harald Welte4526da92020-03-05 23:08:10 +0100230
231/* find TEID of given interface type (and optionally instance) */
232private function f_find_teid(FullyQualifiedTEID_List list,
233 template (present) integer if_type,
234 template (present) BIT4 instance := ?)
235return template (omit) FullyQualifiedTEID
236{
237 var integer i;
238 for (i := 0; i < lengthof(list); i := i+1) {
239 if (match(list[i].interfaceType, if_type) and
240 match(list[i].instance, instance)) {
241 return list[i];
242 }
243 }
244 return omit;
245}
246
247/* process one to-be-created bearer context */
248private function process_bctx_create(BearerContextGrouped bctx) runs on PGW_Session_CT
249{
250 /* FIXME: EPS Bearer ID */
251 /* FIXME: Cause */
252
253 /* find F-TEID of the P-GW U side */
254 var FullyQualifiedTEID rx_fteid;
255 rx_fteid := valueof(f_find_teid(bctx.bearerContextIEs.fullyQualifiedTEID, 5, '0010'B));
256 g_teid_remote := rx_fteid.tEID_GRE_Key;
257 if (rx_fteid.v4_Flag == '1'B) {
258 g_gtpu4_remote := rx_fteid.iPv4_Address;
259 }
260 if (rx_fteid.v6_Flag == '1'B) {
261 g_gtpu6_remote := rx_fteid.iPv6_Address;
262 }
263
264 var UECUPS_CreateTun uecups_create := {
265 tx_teid := oct2int(g_teid_remote),
266 rx_teid := oct2int(g_teid),
267 user_addr_type := IPV4,
268 user_addr := '00000000'O,
269 local_gtp_ep := valueof(ts_UECUPS_SockAddr(f_inet_addr(mp_local_hostname_u))),
270 remote_gtp_ep := valueof(ts_UECUPS_SockAddr(g_gtpu4_remote)),
271 tun_dev_name := g_pars.tun_dev_name,
272 tun_netns_name := g_pars.tun_netns_name
273 };
274
275 /* create tunnel in daemon */
276 if (isbound(g_ip4_addr)) {
277 uecups_create.user_addr := g_ip4_addr;
278 f_gtp2_create_tunnel(uecups_create);
279 }
280 if (isbound(g_ip6_addr)) {
281 uecups_create.user_addr_type := IPV6;
282 uecups_create.user_addr := g_ip6_addr;
283 f_gtp2_create_tunnel(uecups_create);
284 }
285}
286
287/* create a session on the PGW */
288private function f_create_session() runs on PGW_Session_CT {
289 var PDU_GTPCv2 rx;
290
291 /* allocate + register TEID-C on local side */
292 g_teic := f_gtp2_allocate_teid();
293 g_teid := g_teic;
294
295 var template (value) FullyQualifiedTEID fteid_c_ie, fteid_u_ie;
296 fteid_c_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPC, g_teic, 0,
297 f_inet_addr(mp_local_hostname_c), omit);
298 fteid_u_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPU, g_teid, 2,
299 f_inet_addr(mp_local_hostname_u), omit);
300 var template (value) PDU_GTPCv2 g2c :=
301 ts_GTP2C_CreateSessionReq(imsi := g_pars.imsi, msisdn := omit, rat_type := 6,
302 sender_fteid := fteid_c_ie,
303 apn := f_enc_dns_hostname(g_pars.apn),
304 pdn_type := g_pars.pdn_type, teid_list := { fteid_u_ie },
305 chg_car := '0000'O, bearer_id := 1);
306 /* open5gs up to 1.2.3 won't accept it without ULI, despite not mandatory */
307 var template (value) TAI tai := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0001'O };
308 var template (value) ECGI ecgi := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0'H, 23 };
309 g2c.gtpcv2_pdu.createSessionRequest.userLocationInfo := ts_GTP2C_UserLocInfo(tai := tai, ecgi := ecgi);
Vadim Yanitskiybada3c92022-01-20 18:59:07 +0600310 g2c.gtpcv2_pdu.createSessionRequest.servingNetwork := ts_GTP2C_ServingNetwork('001'H, '01F'H);
Harald Welte4526da92020-03-05 23:08:10 +0100311
312 GTP2.send(g2c);
Harald Weltef4001512020-04-26 21:48:34 +0200313 if (DIAMETER_PROC.checkstate("Connected")) {
314 as_DIA_CCR(INITIAL_REQUEST);
315 }
Harald Welte4526da92020-03-05 23:08:10 +0100316 alt {
317 [] GTP2.receive(tr_GTP2C_CreateSessionResp(d_teid:=g_teic, cause:='10'O)) -> value rx {
318 /* extract TEIDs */
319 var CreateSessionResponse resp := rx.gtpcv2_pdu.createSessionResponse;
320 g_teic_remote := resp.fullyQualifiedTEID[0].tEID_GRE_Key;
321
322 /* extract allocated address[es] */
323 var PDN_Address_and_Prefix paa := resp.pDN_AddressAllocation.pDN_Address_and_Prefix;
324 if (ischosen(paa.iPv4_Address)) {
325 g_ip4_addr := paa.iPv4_Address;
326 } else if (ischosen(paa.iPv6_Address)) {
327 g_ip6_addr := paa.iPv6_Address.iPv6_Address;
328 g_ip6_plen := paa.iPv6_Address.prefixLength;
329 } else if (ischosen(paa.iPv4_IPv6)) {
330 g_ip4_addr := paa.iPv4_IPv6.iPv4_Address;
331 g_ip6_addr := paa.iPv4_IPv6.iPv6_Address;
332 g_ip6_plen := paa.iPv4_IPv6.prefixLength;
333 }
334 var integer i;
335 for (i := 0; i < lengthof(resp.bearerContextGrouped); i := i+1) {
336 var BearerContextGrouped bctx := resp.bearerContextGrouped[i];
337 select (bctx.instance) {
338 case ('0000'B) { // created
339 process_bctx_create(bctx);
340 }
341 case ('0001'B) { // removed
342 setverdict(fail, "We don't expect removed bearer contexts yet");
343 }
344 }
345 }
346 }
347 [] GTP2.receive(tr_GTP2C_CreateSessionResp(d_teid:=g_teic, cause:=?)) -> value rx {
348 setverdict(fail, "Unexpected CreateSessionResp(cause=",
349 rx.gtpcv2_pdu.createSessionResponse.cause.causeValue, ")");
350 }
351 [] GTP2.receive {
352 setverdict(fail, "Unexpected GTPv2 while waiting for CreateSessionResp");
353 }
354 }
355
356}
357
358/* delete the session from the PGW */
359private function f_delete_session(template (omit) OCT1 tx_cause := omit,
360 template (present) OCT4 exp_teid,
361 template (present) OCT1 exp_cause) runs on PGW_Session_CT {
362 var template (value) FullyQualifiedTEID fteid_c_ie
363 fteid_c_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPC, g_teic, 0,
364 f_inet_addr(mp_local_hostname_c), omit);
365 var template PDU_GTPCv2 g2c :=
366 ts_GTP2C_DeleteSessionReq(d_teid := g_teic_remote, cause := tx_cause,
367 sender_fteid := fteid_c_ie,
368 teid_list := {}, bearer_id := 1);
369
370 GTP2.send(g2c);
Harald Weltef4001512020-04-26 21:48:34 +0200371 if (DIAMETER_PROC.checkstate("Connected")) {
372 as_DIA_CCR(TERMINATION_REQUEST);
373 }
Harald Welte4526da92020-03-05 23:08:10 +0100374 alt {
375 [] GTP2.receive(tr_GTP2C_DeleteSessionResp(d_teid := exp_teid, cause := exp_cause)) {
376 setverdict(pass);
377 }
378 [] GTP2.receive(tr_GTP2C_DeleteSessionResp(?, ?)) {
379 setverdict(fail, "Unexpected DeleteSessionResp");
380 }
381 [] GTP2.receive {
382 setverdict(fail, "Unexpected GTPv2 while waiting for DeleteSessionResp");
383 }
384 }
385
386 /* destroy tunnel in daemon */
387 if (isbound(g_teid)) {
388 var UECUPS_DestroyTun uecups_destroy := {
389 local_gtp_ep := valueof(ts_UECUPS_SockAddr(f_inet_addr(mp_local_hostname_u))),
390 rx_teid := oct2int(g_teid)
391 };
392 /* FIXME: what about IPv4/IPv6 differentiation? */
393 f_gtp2_destroy_tunnel(uecups_destroy);
394 }
395}
396
397/* start a program on the user plane side; return its PID */
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600398private function f_start_prog(charstring command, boolean redirect_output := true)
399runs on PGW_Session_CT return integer
Harald Welte4526da92020-03-05 23:08:10 +0100400{
401 var UECUPS_StartProgram sprog := {
402 command := command,
403 environment := {},
404 run_as_user := mp_run_prog_as_user,
405 tun_netns_name := g_pars.tun_netns_name
406 };
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600407
408 /* Redirect stdout/stderr to the user-specified location */
409 if (redirect_output) {
410 var charstring prefix := mp_run_prog_log_path & "/" & testcasename();
411 sprog.command := sprog.command & " 1>>" & prefix & ".prog.stdout";
412 sprog.command := sprog.command & " 2>>" & prefix & ".prog.stderr";
413 }
414
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600415 log("Starting a program: ", command);
Harald Welte4526da92020-03-05 23:08:10 +0100416 var UECUPS_StartProgramRes res := f_gtp2_start_program(sprog);
417 if (res.result != OK) {
418 setverdict(fail, "Unable to start program '", command, "'");
419 }
420 return res.pid;
421}
422
423/* wait for termination of a given PID with specified exit_code */
424private function f_wait_term(integer pid, template (present) integer exit_code := 0,
425 float tout := 10.0) runs on PGW_Session_CT
426{
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600427 var UECUPS_ProgramTermInd pti;
Harald Welte4526da92020-03-05 23:08:10 +0100428 timer T := tout;
429
430 T.start;
431 alt {
432 [] GTP2.receive(UECUPS_ProgramTermInd:{pid := pid, exit_code := exit_code}) {
433 setverdict(pass);
434 }
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600435 [] GTP2.receive(UECUPS_ProgramTermInd:?) -> value pti {
436 setverdict(fail, "Received unexpected ProgramTermInd := ", pti);
Harald Welte4526da92020-03-05 23:08:10 +0100437 }
438 [] T.timeout {
439 setverdict(fail, "timeout waiting for user-plane program termination");
440 }
441 }
442}
443
444/* execute a program and wait for result */
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600445private function f_start_prog_wait(charstring command,
446 template integer exit_code := 0,
447 float tout := 10.0,
448 boolean redirect_output := true)
449runs on PGW_Session_CT
Harald Welte4526da92020-03-05 23:08:10 +0100450{
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600451 var integer pid := f_start_prog(command, redirect_output);
Harald Welte4526da92020-03-05 23:08:10 +0100452 f_wait_term(pid, exit_code, tout);
453}
454
455/* execute ping command and wait for result */
456private function f_ping4(charstring host, integer interval := 1, integer count := 10) runs on PGW_Session_CT
457{
458 var charstring ping :="ping -c " & int2str(count) & " -i " & int2str(interval);
459 ping := ping & " -I " & f_inet_ntoa(g_ip4_addr);
460 ping := ping & " " & host;
Harald Welte8cfdc7c2020-04-21 22:48:34 +0200461 f_start_prog_wait(ping, tout := int2float(5 + interval*count));
Harald Welte4526da92020-03-05 23:08:10 +0100462}
463
464
465
466
467/* send echo request; expect response */
468testcase TC_tx_echo() runs on PGW_Test_CT {
469 timer T := 5.0;
470
471 f_init();
472
473 TEID0.send(ts_GTP2C_EchoReq(0));
474 T.start;
475 alt {
476 [] TEID0.receive(tr_GTP2C_EchoResp) {
477 setverdict(pass);
478 }
479 [] T.timeout {
480 setverdict(fail, "timeout waiting for Echo Response");
481 }
482 }
483}
484
485/* create a session, expect it to succeed */
486private function f_TC_createSession() runs on PGW_Session_CT {
487 f_create_session();
488 setverdict(pass);
489}
490testcase TC_createSession() runs on PGW_Test_CT {
491 var PGW_Session_CT vc_conn;
492 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun22"));
493 f_init();
494 vc_conn := f_start_handler(refers(f_TC_createSession), pars);
495 vc_conn.done;
496}
497
498/* create a session, then execute a ping command on the user plane */
499private function f_TC_createSession_ping4() runs on PGW_Session_CT {
500 f_create_session();
501 f_ping4(mp_ping_hostname);
502 setverdict(pass);
503}
504testcase TC_createSession_ping4() runs on PGW_Test_CT {
505 var PGW_Session_CT vc_conn;
506 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
507 f_init();
508 vc_conn := f_start_handler(refers(f_TC_createSession_ping4), pars);
509 vc_conn.done;
510}
Harald Weltefe595e42020-04-21 22:56:47 +0200511testcase TC_createSession_ping4_256() runs on PGW_Test_CT {
512 var PGW_Session_CT vc_conn[256];
513 var integer i;
514
515 f_init();
516
517 for (i := 0; i < sizeof(vc_conn); i:=i+1) {
518 var charstring tundev := "ping" & int2str(i);
519 var SessionPars pars := valueof(t_SessionPars(f_gen_imsi(i), tundev));
520 vc_conn[i] := f_start_handler(refers(f_TC_createSession_ping4), pars);
521 }
522
523 for (i := 0; i < lengthof(vc_conn); i:=i+1) {
524 vc_conn[i].done;
525 }
526}
527
Harald Welte4526da92020-03-05 23:08:10 +0100528
529/* create a session, then delete it again */
530private function f_TC_createSession_deleteSession() runs on PGW_Session_CT {
531 f_create_session();
532 f_delete_session(omit, g_teic, '10'O);
533 setverdict(pass);
534}
535testcase TC_createSession_deleteSession() runs on PGW_Test_CT {
536 var PGW_Session_CT vc_conn;
537 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
538 f_init();
539 vc_conn := f_start_handler(refers(f_TC_createSession_deleteSession), pars);
540 vc_conn.done;
541}
542
543/* send a DeleteSessionReq for an unknown/invalid TEID */
544private function f_TC_deleteSession_unknown() runs on PGW_Session_CT {
545 g_teic := f_gtp2_allocate_teid();
546 g_teic_remote := f_rnd_octstring(4);
547 f_delete_session(omit, '00000000'O, '40'O /* Context Unknown */);
548 setverdict(pass);
549}
550testcase TC_deleteSession_unknown() runs on PGW_Test_CT {
551 var PGW_Session_CT vc_conn;
552 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
553 f_init();
554 vc_conn := f_start_handler(refers(f_TC_deleteSession_unknown), pars);
555 vc_conn.done;
556}
557
558
559
560
561control {
562 execute( TC_tx_echo() );
563 execute( TC_createSession() );
564 execute( TC_createSession_ping4() );
Harald Weltefe595e42020-04-21 22:56:47 +0200565 execute( TC_createSession_ping4_256() );
Harald Welte4526da92020-03-05 23:08:10 +0100566 execute( TC_createSession_deleteSession() );
567 execute( TC_deleteSession_unknown() );
568}
569
570
571}