blob: fa7be47d22d878ab2f019c84be60ce955f6e755b [file] [log] [blame]
Harald Welte4526da92020-03-05 23:08:10 +01001module PGW_Tests {
2
Pau Espin Pedrole343a882022-05-18 18:51:11 +02003import from TCCEncoding_Functions all;
4
Harald Welte4526da92020-03-05 23:08:10 +01005import from General_Types all;
6import from Osmocom_Types all;
7import from Native_Functions all;
Pau Espin Pedrol251d0642022-04-20 18:16:45 +02008import from Misc_Helpers all;
Harald Welte4526da92020-03-05 23:08:10 +01009
10import from GTPv2_Types all;
11import from GTPv2_Templates all;
12import from GTPv2_Emulation all;
13
14import from UECUPS_Types all;
15
16import from DNS_Helpers all;
17
Harald Weltef4001512020-04-26 21:48:34 +020018
19import from DIAMETER_Types all;
20import from DIAMETER_Templates all;
Pau Espin Pedrol117a94f2023-12-21 16:10:12 +010021import from DIAMETER_ts29_272_Templates all;
Harald Weltef4001512020-04-26 21:48:34 +020022import from DIAMETER_Emulation all;
23
24
Harald Welte4526da92020-03-05 23:08:10 +010025modulepar {
Vadim Yanitskiy589972f2022-01-20 19:38:16 +060026 charstring mp_pgw_hostname := "127.0.0.4";
Harald Welte4526da92020-03-05 23:08:10 +010027 charstring mp_local_hostname_c := "127.0.0.1";
28 charstring mp_local_hostname_u := "127.0.0.1";
Harald Weltef4001512020-04-26 21:48:34 +020029
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +060030 charstring mp_run_prog_log_path := "/tmp";
Harald Welte4526da92020-03-05 23:08:10 +010031 charstring mp_run_prog_as_user := "laforge";
32 charstring mp_ping_hostname := "10.45.0.1";
Harald Weltef4001512020-04-26 21:48:34 +020033
Vadim Yanitskiy589972f2022-01-20 19:38:16 +060034 charstring mp_pcrf_local_ip := "127.0.0.9";
Harald Weltef4001512020-04-26 21:48:34 +020035 integer mp_pcrf_local_port := 3868;
Pau Espin Pedrol251d0642022-04-20 18:16:45 +020036
37 charstring mp_ocs_local_ip := "127.0.0.9";
38 integer mp_ocs_local_port := 3869;
Pau Espin Pedrole343a882022-05-18 18:51:11 +020039
40 charstring mp_diam_realm := "localdomain";
Harald Welte4526da92020-03-05 23:08:10 +010041}
42
43/* main component, we typically have one per testcase */
Pau Espin Pedrolbddba3a2024-01-09 12:17:01 +010044type component PGW_Test_CT extends GTP2_ConnHdlr {
Harald Welte4526da92020-03-05 23:08:10 +010045 var GTPv2_Emulation_CT vc_GTP2;
46 port GTP2EM_PT TEID0;
Harald Weltef4001512020-04-26 21:48:34 +020047
48 /* emulated PCRF */
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020049 var DIAMETER_Emulation_CT vc_Gx;
50 port DIAMETER_PT Gx_UNIT;
51 port DIAMETEREM_PROC_PT Gx_PROC;
Pau Espin Pedrol251d0642022-04-20 18:16:45 +020052 /* emulated OCS */
53 var DIAMETER_Emulation_CT vc_Gy;
54 port DIAMETER_PT Gy_UNIT;
55 port DIAMETEREM_PROC_PT Gy_PROC;
Pau Espin Pedrol08880f12022-04-07 19:05:15 +020056 /* global test case guard timer (actual timeout value is set in f_init()) */
57 timer T_guard;
58}
59
60/* global altstep for global guard timer; */
61altstep as_Tguard() runs on PGW_Test_CT {
62 [] T_guard.timeout {
63 setverdict(fail, "Timeout of T_guard");
64 mtc.stop;
65 }
Harald Welte4526da92020-03-05 23:08:10 +010066}
67
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020068type component DIAMETER_ConnHdlr_CT extends DIAMETER_ConnHdlr {
69 port DIAMETER_Conn_PT DIAMETER_CLIENT;
70}
71
72function f_diam_connhldr_ct_main(hexstring imsi) runs on DIAMETER_ConnHdlr_CT {
73 var PDU_DIAMETER msg;
74
75 if (DIAMETER_PROC.checkstate("Connected")) {
Pau Espin Pedroldb017f42023-08-25 19:22:25 +020076 f_diameter_expect_imsi(imsi);
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020077 }
78
79 while (true) {
80 alt {
81 [] DIAMETER_CLIENT.receive(PDU_DIAMETER:?) -> value msg {
82 DIAMETER.send(msg);
83 }
84 [] DIAMETER.receive(PDU_DIAMETER:?) -> value msg {
85 DIAMETER_CLIENT.send(msg);
86 }
87 }
88 }
89}
90
91
Harald Welte4526da92020-03-05 23:08:10 +010092/* per-session component; we typically have 1..N per testcase */
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020093type component PGW_Session_CT extends GTP2_ConnHdlr {
Harald Welte4526da92020-03-05 23:08:10 +010094 var SessionPars g_pars;
95
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020096 port DIAMETER_Conn_PT Gx;
Pau Espin Pedrol251d0642022-04-20 18:16:45 +020097 port DIAMETER_Conn_PT Gy;
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020098
Harald Welte4526da92020-03-05 23:08:10 +010099 /* TEI (Data) local side */
100 var OCT4 g_teid;
101 /* TEI (Control) local side */
102 var OCT4 g_teic;
103 /* TEI (Data) remote side */
104 var OCT4 g_teid_remote;
105 /* TEI (Control) remote side */
106 var OCT4 g_teic_remote;
107 /* GTP-U IPv4 address remote sie */
108 var OCT4 g_gtpu4_remote;
109 var OCT16 g_gtpu6_remote;
110
111 /* Address allocation */
112 var OCT4 g_ip4_addr;
113 var OCT16 g_ip6_addr;
114 var integer g_ip6_plen;
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200115
116 /* Store last received Gy message */
117 var PDU_DIAMETER g_rx_gy;
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100118
119 /* number of programs started, used as identifier. */
120 var integer g_start_prog_count := 0;
Harald Welte4526da92020-03-05 23:08:10 +0100121}
122
123/* configuration data for a given Session */
124type record SessionPars {
125 hexstring imsi,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200126 hexstring msisdn optional,
Harald Welte4526da92020-03-05 23:08:10 +0100127 // serving network
128 integer rat_type,
129 // flags?
130 charstring apn,
131 /* Apn subscribed or non-subscribed */
132 boolean selection_mode,
133 BIT3 pdn_type,
134 /* PAA */
135 /* Max APN Restriction */
136 /* APN-AMBR */
137 octetstring pco optional,
138 octetstring epco optional,
139 /* Bearer Contexts to be created */
140
141 charstring tun_dev_name,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200142 charstring tun_netns_name optional,
143
144 /* In seconds. 0 => disabled, !0 => grant over CC-Time period */
145 integer gy_validity_time
Harald Welte4526da92020-03-05 23:08:10 +0100146}
147
148template (value) SessionPars
149t_SessionPars(hexstring imsi, charstring tundev, integer rat_type := 6, charstring apn := "internet",
150 boolean selection_mode := false, BIT3 pdn_type := '001'B) := {
151 imsi := imsi,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200152 msisdn := '1234'H,
Harald Welte4526da92020-03-05 23:08:10 +0100153 rat_type := rat_type,
154 apn := apn,
155 selection_mode := selection_mode,
156 pdn_type := pdn_type,
157 pco := omit,
158 epco := omit,
159 tun_dev_name := tundev,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200160 tun_netns_name := tundev,
161 gy_validity_time := 0
Harald Welte4526da92020-03-05 23:08:10 +0100162}
163
164type record BearerConfig {
165 integer eps_bearer_id
166}
167
168type function void_fn() runs on PGW_Session_CT;
169
Harald Weltef4001512020-04-26 21:48:34 +0200170friend function DiameterForwardUnitdataCallback(PDU_DIAMETER msg)
171runs on DIAMETER_Emulation_CT return template PDU_DIAMETER {
172 DIAMETER_UNIT.send(msg);
173 return omit;
174}
175
176friend function f_init_diameter(charstring id) runs on PGW_Test_CT {
177 var DIAMETEROps ops := {
178 create_cb := refers(DIAMETER_Emulation.ExpectedCreateCallback),
Vadim Yanitskiyb46f01e2021-12-06 03:23:13 +0300179 unitdata_cb := refers(DiameterForwardUnitdataCallback),
180 raw := false /* handler mode (IMSI based routing) */
Harald Weltef4001512020-04-26 21:48:34 +0200181 };
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200182 var DIAMETER_conn_parameters pars;
183
184 /* Gx setup: */
185 pars := {
Harald Weltef4001512020-04-26 21:48:34 +0200186 remote_ip := mp_pgw_hostname,
187 remote_sctp_port := -1,
188 local_ip := mp_pcrf_local_ip,
189 local_sctp_port := mp_pcrf_local_port,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200190 origin_host := "pcrf." & mp_diam_realm,
191 origin_realm := mp_diam_realm,
Pau Espin Pedrol33b47492022-03-08 17:43:01 +0100192 auth_app_id := omit,
Harald Weltef4001512020-04-26 21:48:34 +0200193 vendor_app_id := c_DIAMETER_3GPP_Gx_AID
194 };
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200195 vc_Gx := DIAMETER_Emulation_CT.create(id);
196 map(vc_Gx:DIAMETER, system:DIAMETER_CODEC_PT);
197 connect(vc_Gx:DIAMETER_UNIT, self:Gx_UNIT);
198 connect(vc_Gx:DIAMETER_PROC, self:Gx_PROC);
199 vc_Gx.start(DIAMETER_Emulation.main(ops, pars, id));
Harald Weltef4001512020-04-26 21:48:34 +0200200
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200201 /* Gy setup: */
202 pars := {
203 remote_ip := mp_pgw_hostname,
204 remote_sctp_port := -1,
205 local_ip := mp_ocs_local_ip,
206 local_sctp_port := mp_ocs_local_port,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200207 origin_host := "ocs." & mp_diam_realm,
208 origin_realm := mp_diam_realm,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200209 auth_app_id := c_DIAMETER_CREDIT_CONTROL_AID,
210 vendor_app_id := omit
211 };
212 vc_Gy := DIAMETER_Emulation_CT.create(id);
213 map(vc_Gy:DIAMETER, system:DIAMETER_CODEC_PT);
214 connect(vc_Gy:DIAMETER_UNIT, self:Gy_UNIT);
215 connect(vc_Gy:DIAMETER_PROC, self:Gy_PROC);
216 vc_Gy.start(DIAMETER_Emulation.main(ops, pars, id));
217
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200218 f_diameter_wait_capability(Gx_UNIT);
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200219 f_diameter_wait_capability(Gy_UNIT);
Pau Espin Pedrolbb5f45f2022-04-11 12:36:55 +0200220 /* Give some time for our emulation to get out of SUSPECT list of SUT (3 watchdong ping-pongs):
221 * RFC6733 sec 5.1
222 * RFC3539 sec 3.4.1 [5]
223 * https://github.com/freeDiameter/freeDiameter/blob/master/libfdcore/p_psm.c#L49
224 */
225 f_sleep(1.0);
Harald Weltef4001512020-04-26 21:48:34 +0200226}
227
Pau Espin Pedrol08880f12022-04-07 19:05:15 +0200228private function f_init(float guard_timeout := 60.0) runs on PGW_Test_CT {
229 T_guard.start(guard_timeout);
230 activate(as_Tguard());
231
Harald Welte4526da92020-03-05 23:08:10 +0100232 var Gtp2EmulationCfg cfg := {
233 gtpc_bind_ip := mp_local_hostname_c,
234 gtpc_bind_port := GTP2C_PORT,
235 gtpc_remote_ip := mp_pgw_hostname,
236 gtpc_remote_port := GTP2C_PORT,
237 sgw_role := true,
238 use_gtpu_daemon := true
239 };
240
241 vc_GTP2 := GTPv2_Emulation_CT.create("GTP2_EM");
242 map(vc_GTP2:GTP2C, system:GTP2C);
243 connect(vc_GTP2:TEID0, self:TEID0);
Pau Espin Pedrolbddba3a2024-01-09 12:17:01 +0100244 connect(vc_GTP2:CLIENT, self:GTP2);
245 connect(vc_GTP2:CLIENT_PROC, self:GTP2_PROC);
Harald Welte4526da92020-03-05 23:08:10 +0100246 vc_GTP2.start(GTPv2_Emulation.main(cfg));
Harald Weltef4001512020-04-26 21:48:34 +0200247
248 if (mp_pcrf_local_ip != "") {
249 f_init_diameter(testcasename());
250 }
Harald Welte4526da92020-03-05 23:08:10 +0100251}
252
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200253function f_start_handler(void_fn fn, template (omit) SessionPars pars_tmpl := omit)
Harald Welte4526da92020-03-05 23:08:10 +0100254runs on PGW_Test_CT return PGW_Session_CT {
255 var charstring id := testcasename();
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200256 var DIAMETER_ConnHdlr_CT vc_conn_gx, vc_conn_gy;
Harald Welte4526da92020-03-05 23:08:10 +0100257 var PGW_Session_CT vc_conn;
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200258 var SessionPars pars;
259
260 if (isvalue(pars_tmpl)) {
261 pars := valueof(pars_tmpl);
262 } else {
263 /*TODO: set default values */
264 }
265
Harald Welte4526da92020-03-05 23:08:10 +0100266 vc_conn := PGW_Session_CT.create(id);
267 connect(vc_conn:GTP2, vc_GTP2:CLIENT);
268 connect(vc_conn:GTP2_PROC, vc_GTP2:CLIENT_PROC);
Harald Weltef4001512020-04-26 21:48:34 +0200269
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200270 if (isbound(vc_Gx)) {
271 vc_conn_gx := DIAMETER_ConnHdlr_CT.create(id);
272 connect(vc_conn_gx:DIAMETER, vc_Gx:DIAMETER_CLIENT);
273 connect(vc_conn_gx:DIAMETER_PROC, vc_Gx:DIAMETER_PROC);
274 connect(vc_conn:Gx, vc_conn_gx:DIAMETER_CLIENT);
275 vc_conn_gx.start(f_diam_connhldr_ct_main(pars.imsi));
Harald Weltef4001512020-04-26 21:48:34 +0200276 }
277
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200278 if (isbound(vc_Gy)) {
279 vc_conn_gy := DIAMETER_ConnHdlr_CT.create(id);
280 connect(vc_conn_gy:DIAMETER, vc_Gy:DIAMETER_CLIENT);
281 connect(vc_conn_gy:DIAMETER_PROC, vc_Gy:DIAMETER_PROC);
282 connect(vc_conn:Gy, vc_conn_gy:DIAMETER_CLIENT);
283 vc_conn_gy.start(f_diam_connhldr_ct_main(pars.imsi));
284 }
285
Harald Welte4526da92020-03-05 23:08:10 +0100286 vc_conn.start(f_handler_init(fn, pars));
287 return vc_conn;
288}
289
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200290private function f_handler_init(void_fn fn, SessionPars pars)
Harald Welte4526da92020-03-05 23:08:10 +0100291runs on PGW_Session_CT {
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200292 g_pars := valueof(pars);
Harald Welte4526da92020-03-05 23:08:10 +0100293 fn.apply();
294}
295
Harald Weltefe595e42020-04-21 22:56:47 +0200296private function f_concat_pad(integer tot_len, hexstring prefix, integer suffix) return hexstring {
297 var integer suffix_len := tot_len - lengthof(prefix);
298 var charstring suffix_ch := int2str(suffix);
299 var integer pad_len := suffix_len - lengthof(suffix_ch);
300
301 return prefix & int2hex(0, pad_len) & str2hex(suffix_ch);
302}
303
304function f_gen_imei(integer suffix) return hexstring {
305 return f_concat_pad(14, '49999'H, suffix);
306}
307
308function f_gen_imsi(integer suffix) return hexstring {
309 return f_concat_pad(15, '26242'H, suffix);
310}
311
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200312private altstep as_DIA_Gx_CCR(DCC_NONE_CC_Request_Type req_type) runs on PGW_Session_CT {
Harald Weltef4001512020-04-26 21:48:34 +0200313 var PDU_DIAMETER rx_dia;
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200314 [] Gx.receive(tr_DIA_Gx_CCR(req_type := req_type)) -> value rx_dia {
Harald Weltef4001512020-04-26 21:48:34 +0200315 var template (omit) AVP avp;
316 var octetstring sess_id;
317 var AVP_Unsigned32 req_num;
318
319 avp := f_DIAMETER_get_avp(rx_dia, c_AVP_Code_BASE_NONE_Session_Id);
320 sess_id := valueof(avp.avp_data.avp_BASE_NONE_Session_Id);
321
322 avp := f_DIAMETER_get_avp(rx_dia, c_AVP_Code_DCC_NONE_CC_Request_Number);
323 req_num := valueof(avp.avp_data.avp_DCC_NONE_CC_Request_Number);
324
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200325 Gx.send(ts_DIA_Gx_CCA(rx_dia.hop_by_hop_id, rx_dia.end_to_end_id, sess_id,
Harald Weltef4001512020-04-26 21:48:34 +0200326 req_type, req_num));
327 }
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200328 [] Gx.receive(PDU_DIAMETER:?) -> value rx_dia {
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200329 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
330 log2str("Received unexpected DIAMETER ", rx_dia));
331 }
332}
333
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200334function f_tr_DIA_Gy_CCR(DCC_NONE_CC_Request_Type req_type)
335runs on PGW_Session_CT return template (present) PDU_DIAMETER
336{
337 var template (present) PDU_DIAMETER tpl;
338 var charstring smf_origin_host := "smf." & mp_diam_realm;
339 var template (present) octetstring imsi := ?;
340 var template (present) octetstring msisdn := ?;
341 var template (present) octetstring rat_type := ?;
342 var template (present) OCT4 charging_char := ?;
343 var template (present) OCT1 nsapi := ?;
344 imsi := char2oct(f_dec_TBCD(imsi_hex2oct(g_pars.imsi)));
345 //msisdn := char2oct(f_dec_TBCD(substr(ctx_val.msisdn, 1, lengthof(ctx_val.msisdn) -1)));
346 rat_type := int2oct(g_pars.rat_type, 1);
347 charging_char := char2oct(oct2str('0000'O)); // f_create_session() uses hardcoded chg_car := '0000'O
348 nsapi := '01'O; // f_create_session() uses hardcoded bearer_id := 1
349 select (req_type) {
350 case (INITIAL_REQUEST) {
351 tpl := tr_DIAMETER(flags:='11000000'B, cmd_code:=Credit_Control,
352 avps := superset(
353 tr_AVP_SessionId,
354 tr_AVP_OriginHost(smf_origin_host),
355 tr_AVP_OriginRealm(mp_diam_realm),
356 tr_AVP_DestinationRealm(mp_diam_realm),
357 tr_AVP_AuthAppId(int2oct(c_DIAMETER_CREDIT_CONTROL_AID, 4)),
358 tr_AVP_ServiceContextId,
359 tr_AVP_CcReqType(req_type),
360 tr_AVP_CcReqNum(?),
361 tr_AVP_EventTimestamp(?),
362 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_IMSI), tr_AVP_SubcrIdData(imsi)}),
363 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_E164), tr_AVP_SubcrIdData(msisdn)}),
364 tr_AVP_RequestedAction(DIRECT_DEBITING),
365 tr_AVP_3GPP_AoCRequestType,
366 tr_AVP_MultipleServicesIndicator,
367 tr_AVP_Multiple_Services_Credit_Control(content := superset(
368 tr_AVP_Requested_Service_Unit,
Pau Espin Pedrol6477d732022-06-03 12:04:40 +0200369 tr_AVP_PCC_3GPP_QoS_Information,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200370 tr_AVP_GI_3GPP_RatType(rat_type)
371 )),
372 tr_AVP_3GPP_ServiceInformation(content := superset(
373 tr_AVP_3GPP_PSInformation(content := superset(
374 tr_AVP_3GPP_ChargingId,
375 tr_AVP_3GPP_PDPType((IPv4,IPv6,IPv4v6)),
376 tr_AVP_3GPP_PDPAddress(tr_AVP_Address((IP,IP6), ?)),
377 tr_AVP_3GPP_SGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pcrf_local_ip))),
378 tr_AVP_3GPP_GGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pgw_hostname))),
379 tr_AVP_3GPP_CalledStationId,
380 tr_AVP_3GPP_SelectionMode,
381 tr_AVP_3GPP_ChargingCharacteristics(charging_char),
382 tr_AVP_3GPP_SGSNMCCMNC,
383 tr_AVP_3GPP_NSAPI(nsapi),
384 /*We don't yet send MS_Tz in CreateSessionReq:
385 tr_AVP_3GPP_MS_TimeZone,*/
386 tr_AVP_3GPP_ULI/*,
387 We don't yet send IMEI in CreateSessionReq:
388 tr_AVP_UserEquipmentInfo({
389 tr_AVP_UserEquipmentInfoType(IMEISV),
390 tr_AVP_UserEquipmentInfoValue(imeisv)
391 })*/
392 ))
393 ))
394 ));
395 }
396 case (UPDATE_REQUEST) {
397 tpl := tr_DIAMETER(flags:='11000000'B, cmd_code:=Credit_Control,
398 avps := superset(
399 tr_AVP_SessionId,
400 tr_AVP_OriginHost(smf_origin_host),
401 tr_AVP_OriginRealm(mp_diam_realm),
402 tr_AVP_DestinationRealm(mp_diam_realm),
403 tr_AVP_AuthAppId(int2oct(c_DIAMETER_CREDIT_CONTROL_AID, 4)),
404 tr_AVP_ServiceContextId,
405 tr_AVP_CcReqType(req_type),
406 tr_AVP_CcReqNum(?),
407 tr_AVP_DestinationHost(?),
408 tr_AVP_EventTimestamp(?),
409 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_IMSI), tr_AVP_SubcrIdData(imsi)}),
410 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_E164), tr_AVP_SubcrIdData(msisdn)}),
411 tr_AVP_RequestedAction(DIRECT_DEBITING),
412 tr_AVP_3GPP_AoCRequestType,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200413 tr_AVP_Multiple_Services_Credit_Control(content := superset(
414 tr_AVP_Requested_Service_Unit,
415 tr_AVP_Used_Service_Unit,
Pau Espin Pedrolcba0f6d2022-05-24 13:49:46 +0200416 /* tr_AVP_3GPP_Reporting_Reason, can be sometimes inside UsedServiceUnit */
Pau Espin Pedrol6477d732022-06-03 12:04:40 +0200417 tr_AVP_PCC_3GPP_QoS_Information,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200418 tr_AVP_GI_3GPP_RatType(rat_type)
419 )),
420 tr_AVP_3GPP_ServiceInformation(content := superset(
421 tr_AVP_3GPP_PSInformation(content := superset(
422 tr_AVP_3GPP_ChargingId,
423 /* tr_AVP_3GPP_PDPType, Only in INIT */
424 tr_AVP_3GPP_PDPAddress(tr_AVP_Address((IP,IP6), ?)),
425 tr_AVP_3GPP_SGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pcrf_local_ip))),
426 tr_AVP_3GPP_GGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pgw_hostname))),
427 tr_AVP_3GPP_CalledStationId,
428 tr_AVP_3GPP_SelectionMode,
429 tr_AVP_3GPP_ChargingCharacteristics(charging_char),
430 tr_AVP_3GPP_SGSNMCCMNC,
431 tr_AVP_3GPP_NSAPI(nsapi),
432 /*We don't yet send MS_Tz in CreateSessionReq:
433 tr_AVP_3GPP_MS_TimeZone,*/
434 tr_AVP_3GPP_ULI/*,
435 We don't yet send IMEI in CreateSessionReq:
436 tr_AVP_UserEquipmentInfo({
437 tr_AVP_UserEquipmentInfoType(IMEISV),
438 tr_AVP_UserEquipmentInfoValue(imeisv)
439 })*/
440 ))
441 ))
442 ));
443 }
444 case (TERMINATION_REQUEST) {
445 tpl := tr_DIAMETER(flags:='11000000'B, cmd_code:=Credit_Control,
446 avps := superset(
447 tr_AVP_SessionId,
448 tr_AVP_OriginHost(smf_origin_host),
449 tr_AVP_OriginRealm(mp_diam_realm),
450 tr_AVP_DestinationRealm(mp_diam_realm),
451 tr_AVP_AuthAppId(int2oct(c_DIAMETER_CREDIT_CONTROL_AID, 4)),
452 tr_AVP_ServiceContextId,
453 tr_AVP_CcReqType(req_type),
454 tr_AVP_CcReqNum(?),
455 tr_AVP_DestinationHost(?),
456 tr_AVP_EventTimestamp(?),
457 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_IMSI), tr_AVP_SubcrIdData(imsi)}),
458 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_E164), tr_AVP_SubcrIdData(msisdn)}),
459 tr_AVP_TerminationCause(?),
460 tr_AVP_RequestedAction(DIRECT_DEBITING),
461 tr_AVP_3GPP_AoCRequestType,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200462 tr_AVP_Multiple_Services_Credit_Control(content := superset(
463 /* tr_AVP_Requested_Service_Unit, Only in INIT and UPDATE */
464 tr_AVP_Used_Service_Unit,
Pau Espin Pedrolcba0f6d2022-05-24 13:49:46 +0200465 tr_AVP_3GPP_Reporting_Reason(FINAL),
Pau Espin Pedrol6477d732022-06-03 12:04:40 +0200466 tr_AVP_PCC_3GPP_QoS_Information,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200467 tr_AVP_GI_3GPP_RatType(rat_type)
468 )),
469 tr_AVP_3GPP_ServiceInformation(content := superset(
470 tr_AVP_3GPP_PSInformation(content := superset(
471 tr_AVP_3GPP_ChargingId,
472 /* tr_AVP_3GPP_PDPType, Only in INIT */
473 tr_AVP_3GPP_PDPAddress(tr_AVP_Address((IP,IP6), ?)),
474 tr_AVP_3GPP_SGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pcrf_local_ip))),
475 tr_AVP_3GPP_GGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pgw_hostname))),
476 tr_AVP_3GPP_CalledStationId,
477 tr_AVP_3GPP_SelectionMode,
478 tr_AVP_3GPP_ChargingCharacteristics(charging_char),
479 tr_AVP_3GPP_SGSNMCCMNC,
480 tr_AVP_3GPP_NSAPI(nsapi),
481 /*We don't yet send MS_Tz in CreateSessionReq:
482 tr_AVP_3GPP_MS_TimeZone,*/
483 tr_AVP_3GPP_ULI/*,
484 We don't yet send IMEI in CreateSessionReq:
485 tr_AVP_UserEquipmentInfo({
486 tr_AVP_UserEquipmentInfoType(IMEISV),
487 tr_AVP_UserEquipmentInfoValue(imeisv)
488 })*/
489 ))
490 ))
491 ));
492 }
493 }
494 return tpl;
495}
496
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200497private altstep as_DIA_Gy_CCR(DCC_NONE_CC_Request_Type req_type)
498runs on PGW_Session_CT {
499 [] Gy.receive(f_tr_DIA_Gy_CCR(req_type := req_type)) -> value g_rx_gy {
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200500 var template (value) PDU_DIAMETER tx_dia;
501 var template (omit) AVP avp;
502 var octetstring sess_id;
503 var AVP_Unsigned32 req_num;
504
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200505 avp := f_DIAMETER_get_avp(g_rx_gy, c_AVP_Code_BASE_NONE_Session_Id);
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200506 sess_id := valueof(avp.avp_data.avp_BASE_NONE_Session_Id);
507
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200508 avp := f_DIAMETER_get_avp(g_rx_gy, c_AVP_Code_DCC_NONE_CC_Request_Number);
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200509 req_num := valueof(avp.avp_data.avp_DCC_NONE_CC_Request_Number);
510 if (g_pars.gy_validity_time > 0) {
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200511 tx_dia := ts_DIA_Gy_CCA_ValidityTime(g_rx_gy.hop_by_hop_id, g_rx_gy.end_to_end_id, sess_id,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200512 req_type, req_num, g_pars.gy_validity_time);
513 } else {
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200514 tx_dia := ts_DIA_Gy_CCA(g_rx_gy.hop_by_hop_id, g_rx_gy.end_to_end_id, sess_id,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200515 req_type, req_num);
516 }
517 Gy.send(tx_dia);
518 }
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200519 [] Gy.receive(PDU_DIAMETER:?) -> value g_rx_gy {
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200520 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200521 log2str("Received unexpected DIAMETER Gy", g_rx_gy));
Harald Weltef4001512020-04-26 21:48:34 +0200522 }
523}
524
Harald Welte4526da92020-03-05 23:08:10 +0100525
526/* find TEID of given interface type (and optionally instance) */
527private function f_find_teid(FullyQualifiedTEID_List list,
528 template (present) integer if_type,
529 template (present) BIT4 instance := ?)
530return template (omit) FullyQualifiedTEID
531{
532 var integer i;
533 for (i := 0; i < lengthof(list); i := i+1) {
534 if (match(list[i].interfaceType, if_type) and
535 match(list[i].instance, instance)) {
536 return list[i];
537 }
538 }
539 return omit;
540}
541
542/* process one to-be-created bearer context */
543private function process_bctx_create(BearerContextGrouped bctx) runs on PGW_Session_CT
544{
545 /* FIXME: EPS Bearer ID */
546 /* FIXME: Cause */
547
548 /* find F-TEID of the P-GW U side */
549 var FullyQualifiedTEID rx_fteid;
550 rx_fteid := valueof(f_find_teid(bctx.bearerContextIEs.fullyQualifiedTEID, 5, '0010'B));
551 g_teid_remote := rx_fteid.tEID_GRE_Key;
552 if (rx_fteid.v4_Flag == '1'B) {
553 g_gtpu4_remote := rx_fteid.iPv4_Address;
554 }
555 if (rx_fteid.v6_Flag == '1'B) {
556 g_gtpu6_remote := rx_fteid.iPv6_Address;
557 }
558
559 var UECUPS_CreateTun uecups_create := {
560 tx_teid := oct2int(g_teid_remote),
561 rx_teid := oct2int(g_teid),
562 user_addr_type := IPV4,
563 user_addr := '00000000'O,
564 local_gtp_ep := valueof(ts_UECUPS_SockAddr(f_inet_addr(mp_local_hostname_u))),
565 remote_gtp_ep := valueof(ts_UECUPS_SockAddr(g_gtpu4_remote)),
566 tun_dev_name := g_pars.tun_dev_name,
567 tun_netns_name := g_pars.tun_netns_name
568 };
569
570 /* create tunnel in daemon */
571 if (isbound(g_ip4_addr)) {
572 uecups_create.user_addr := g_ip4_addr;
573 f_gtp2_create_tunnel(uecups_create);
574 }
575 if (isbound(g_ip6_addr)) {
576 uecups_create.user_addr_type := IPV6;
577 uecups_create.user_addr := g_ip6_addr;
578 f_gtp2_create_tunnel(uecups_create);
579 }
580}
581
582/* create a session on the PGW */
583private function f_create_session() runs on PGW_Session_CT {
584 var PDU_GTPCv2 rx;
585
586 /* allocate + register TEID-C on local side */
587 g_teic := f_gtp2_allocate_teid();
588 g_teid := g_teic;
589
590 var template (value) FullyQualifiedTEID fteid_c_ie, fteid_u_ie;
591 fteid_c_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPC, g_teic, 0,
592 f_inet_addr(mp_local_hostname_c), omit);
593 fteid_u_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPU, g_teid, 2,
594 f_inet_addr(mp_local_hostname_u), omit);
595 var template (value) PDU_GTPCv2 g2c :=
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200596 ts_GTP2C_CreateSessionReq(imsi := g_pars.imsi, msisdn := g_pars.msisdn, rat_type := 6,
Harald Welte4526da92020-03-05 23:08:10 +0100597 sender_fteid := fteid_c_ie,
598 apn := f_enc_dns_hostname(g_pars.apn),
599 pdn_type := g_pars.pdn_type, teid_list := { fteid_u_ie },
600 chg_car := '0000'O, bearer_id := 1);
601 /* open5gs up to 1.2.3 won't accept it without ULI, despite not mandatory */
602 var template (value) TAI tai := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0001'O };
603 var template (value) ECGI ecgi := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0'H, 23 };
604 g2c.gtpcv2_pdu.createSessionRequest.userLocationInfo := ts_GTP2C_UserLocInfo(tai := tai, ecgi := ecgi);
Vadim Yanitskiybada3c92022-01-20 18:59:07 +0600605 g2c.gtpcv2_pdu.createSessionRequest.servingNetwork := ts_GTP2C_ServingNetwork('001'H, '01F'H);
Harald Welte4526da92020-03-05 23:08:10 +0100606
607 GTP2.send(g2c);
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200608 if (Gx.checkstate("Connected")) {
609 as_DIA_Gx_CCR(INITIAL_REQUEST);
Harald Weltef4001512020-04-26 21:48:34 +0200610 }
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200611 if (Gy.checkstate("Connected")) {
612 as_DIA_Gy_CCR(INITIAL_REQUEST);
613 }
Harald Welte4526da92020-03-05 23:08:10 +0100614 alt {
Pau Espin Pedrol1344e472023-10-23 19:54:29 +0200615 [] GTP2.receive(tr_GTP2C_CreateSessionResp(d_teid:=g_teic, cause:=Request_accepted)) -> value rx {
Harald Welte4526da92020-03-05 23:08:10 +0100616 /* extract TEIDs */
617 var CreateSessionResponse resp := rx.gtpcv2_pdu.createSessionResponse;
618 g_teic_remote := resp.fullyQualifiedTEID[0].tEID_GRE_Key;
619
620 /* extract allocated address[es] */
621 var PDN_Address_and_Prefix paa := resp.pDN_AddressAllocation.pDN_Address_and_Prefix;
622 if (ischosen(paa.iPv4_Address)) {
623 g_ip4_addr := paa.iPv4_Address;
624 } else if (ischosen(paa.iPv6_Address)) {
625 g_ip6_addr := paa.iPv6_Address.iPv6_Address;
626 g_ip6_plen := paa.iPv6_Address.prefixLength;
627 } else if (ischosen(paa.iPv4_IPv6)) {
628 g_ip4_addr := paa.iPv4_IPv6.iPv4_Address;
629 g_ip6_addr := paa.iPv4_IPv6.iPv6_Address;
630 g_ip6_plen := paa.iPv4_IPv6.prefixLength;
631 }
632 var integer i;
633 for (i := 0; i < lengthof(resp.bearerContextGrouped); i := i+1) {
634 var BearerContextGrouped bctx := resp.bearerContextGrouped[i];
635 select (bctx.instance) {
636 case ('0000'B) { // created
637 process_bctx_create(bctx);
638 }
639 case ('0001'B) { // removed
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100640 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
641 "We don't expect removed bearer contexts yet");
Harald Welte4526da92020-03-05 23:08:10 +0100642 }
643 }
644 }
645 }
646 [] GTP2.receive(tr_GTP2C_CreateSessionResp(d_teid:=g_teic, cause:=?)) -> value rx {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100647 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
648 log2str("Unexpected CreateSessionResp(cause=",
649 rx.gtpcv2_pdu.createSessionResponse.cause.causeValue, ")"));
Harald Welte4526da92020-03-05 23:08:10 +0100650 }
651 [] GTP2.receive {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100652 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
653 "Unexpected GTPv2 while waiting for CreateSessionResp");
Harald Welte4526da92020-03-05 23:08:10 +0100654 }
655 }
656
657}
658
659/* delete the session from the PGW */
Pau Espin Pedrol1344e472023-10-23 19:54:29 +0200660private function f_delete_session(template (omit) GTP2C_Cause tx_cause := omit,
Harald Welte4526da92020-03-05 23:08:10 +0100661 template (present) OCT4 exp_teid,
Pau Espin Pedrol1344e472023-10-23 19:54:29 +0200662 template (present) GTP2C_Cause exp_cause,
Pau Espin Pedrol5b0327b2022-04-11 13:02:36 +0200663 boolean expect_diameter := true) runs on PGW_Session_CT {
Harald Welte4526da92020-03-05 23:08:10 +0100664 var template (value) FullyQualifiedTEID fteid_c_ie
665 fteid_c_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPC, g_teic, 0,
666 f_inet_addr(mp_local_hostname_c), omit);
667 var template PDU_GTPCv2 g2c :=
668 ts_GTP2C_DeleteSessionReq(d_teid := g_teic_remote, cause := tx_cause,
669 sender_fteid := fteid_c_ie,
670 teid_list := {}, bearer_id := 1);
671
672 GTP2.send(g2c);
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200673 if (Gx.checkstate("Connected") and expect_diameter) {
674 as_DIA_Gx_CCR(TERMINATION_REQUEST);
Harald Weltef4001512020-04-26 21:48:34 +0200675 }
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200676 if (Gy.checkstate("Connected") and expect_diameter) {
677 as_DIA_Gy_CCR(TERMINATION_REQUEST);
678 }
Harald Welte4526da92020-03-05 23:08:10 +0100679 alt {
680 [] GTP2.receive(tr_GTP2C_DeleteSessionResp(d_teid := exp_teid, cause := exp_cause)) {
681 setverdict(pass);
682 }
683 [] GTP2.receive(tr_GTP2C_DeleteSessionResp(?, ?)) {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100684 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
685 "Unexpected DeleteSessionResp");
Harald Welte4526da92020-03-05 23:08:10 +0100686 }
687 [] GTP2.receive {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100688 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
689 "Unexpected GTPv2 while waiting for DeleteSessionResp");
Harald Welte4526da92020-03-05 23:08:10 +0100690 }
691 }
692
693 /* destroy tunnel in daemon */
694 if (isbound(g_teid)) {
695 var UECUPS_DestroyTun uecups_destroy := {
696 local_gtp_ep := valueof(ts_UECUPS_SockAddr(f_inet_addr(mp_local_hostname_u))),
697 rx_teid := oct2int(g_teid)
698 };
699 /* FIXME: what about IPv4/IPv6 differentiation? */
700 f_gtp2_destroy_tunnel(uecups_destroy);
701 }
702}
703
704/* start a program on the user plane side; return its PID */
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600705private function f_start_prog(charstring command, boolean redirect_output := true)
706runs on PGW_Session_CT return integer
Harald Welte4526da92020-03-05 23:08:10 +0100707{
708 var UECUPS_StartProgram sprog := {
709 command := command,
710 environment := {},
711 run_as_user := mp_run_prog_as_user,
712 tun_netns_name := g_pars.tun_netns_name
713 };
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100714 g_start_prog_count := g_start_prog_count + 1;
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600715
716 /* Redirect stdout/stderr to the user-specified location */
717 if (redirect_output) {
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100718 var charstring id := testcasename() & "-" & hex2str(g_pars.imsi) & "-" & int2str(g_start_prog_count);
719 var charstring prefix := mp_run_prog_log_path & "/" & id;
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600720 sprog.command := sprog.command & " 1>>" & prefix & ".prog.stdout";
721 sprog.command := sprog.command & " 2>>" & prefix & ".prog.stderr";
722 }
723
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600724 log("Starting a program: ", command);
Harald Welte4526da92020-03-05 23:08:10 +0100725 var UECUPS_StartProgramRes res := f_gtp2_start_program(sprog);
726 if (res.result != OK) {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100727 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
728 log2str("Unable to start program '", command, "'"));
Harald Welte4526da92020-03-05 23:08:10 +0100729 }
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100730 log("Started program '", command, "' with PID ", res.pid);
Harald Welte4526da92020-03-05 23:08:10 +0100731 return res.pid;
732}
733
734/* wait for termination of a given PID with specified exit_code */
735private function f_wait_term(integer pid, template (present) integer exit_code := 0,
736 float tout := 10.0) runs on PGW_Session_CT
737{
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600738 var UECUPS_ProgramTermInd pti;
Harald Welte4526da92020-03-05 23:08:10 +0100739 timer T := tout;
740
741 T.start;
742 alt {
743 [] GTP2.receive(UECUPS_ProgramTermInd:{pid := pid, exit_code := exit_code}) {
744 setverdict(pass);
745 }
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600746 [] GTP2.receive(UECUPS_ProgramTermInd:?) -> value pti {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100747 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
748 log2str("Received unexpected ProgramTermInd := ", pti));
Harald Welte4526da92020-03-05 23:08:10 +0100749 }
750 [] T.timeout {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100751 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100752 log2str("timeout (", tout, " seconds) waiting for user-plane program PID ", pid, " termination"));
Harald Welte4526da92020-03-05 23:08:10 +0100753 }
754 }
755}
756
757/* execute a program and wait for result */
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600758private function f_start_prog_wait(charstring command,
759 template integer exit_code := 0,
760 float tout := 10.0,
761 boolean redirect_output := true)
762runs on PGW_Session_CT
Harald Welte4526da92020-03-05 23:08:10 +0100763{
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600764 var integer pid := f_start_prog(command, redirect_output);
Harald Welte4526da92020-03-05 23:08:10 +0100765 f_wait_term(pid, exit_code, tout);
766}
767
768/* execute ping command and wait for result */
769private function f_ping4(charstring host, integer interval := 1, integer count := 10) runs on PGW_Session_CT
770{
771 var charstring ping :="ping -c " & int2str(count) & " -i " & int2str(interval);
Pau Espin Pedrol55980c42023-02-28 12:40:52 +0100772
773 if (not isbound(g_ip4_addr)) {
774 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail, "f_ping4(): g_ip4_addr is unset!");
775 }
Harald Welte4526da92020-03-05 23:08:10 +0100776 ping := ping & " -I " & f_inet_ntoa(g_ip4_addr);
777 ping := ping & " " & host;
Harald Welte8cfdc7c2020-04-21 22:48:34 +0200778 f_start_prog_wait(ping, tout := int2float(5 + interval*count));
Harald Welte4526da92020-03-05 23:08:10 +0100779}
780
781
782
783
784/* send echo request; expect response */
785testcase TC_tx_echo() runs on PGW_Test_CT {
786 timer T := 5.0;
787
788 f_init();
789
Pau Espin Pedrolbddba3a2024-01-09 12:17:01 +0100790 GTP2.send(ts_GTP2C_EchoReq(0));
Harald Welte4526da92020-03-05 23:08:10 +0100791 T.start;
792 alt {
Pau Espin Pedrolbddba3a2024-01-09 12:17:01 +0100793 [] GTP2.receive(tr_GTP2C_EchoResp) {
Harald Welte4526da92020-03-05 23:08:10 +0100794 setverdict(pass);
795 }
796 [] T.timeout {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100797 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail, "timeout waiting for Echo Response");
Harald Welte4526da92020-03-05 23:08:10 +0100798 }
799 }
800}
801
802/* create a session, expect it to succeed */
803private function f_TC_createSession() runs on PGW_Session_CT {
804 f_create_session();
805 setverdict(pass);
806}
807testcase TC_createSession() runs on PGW_Test_CT {
808 var PGW_Session_CT vc_conn;
809 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun22"));
810 f_init();
811 vc_conn := f_start_handler(refers(f_TC_createSession), pars);
812 vc_conn.done;
813}
814
815/* create a session, then execute a ping command on the user plane */
816private function f_TC_createSession_ping4() runs on PGW_Session_CT {
817 f_create_session();
818 f_ping4(mp_ping_hostname);
819 setverdict(pass);
820}
821testcase TC_createSession_ping4() runs on PGW_Test_CT {
822 var PGW_Session_CT vc_conn;
823 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
824 f_init();
825 vc_conn := f_start_handler(refers(f_TC_createSession_ping4), pars);
826 vc_conn.done;
827}
Harald Weltefe595e42020-04-21 22:56:47 +0200828testcase TC_createSession_ping4_256() runs on PGW_Test_CT {
829 var PGW_Session_CT vc_conn[256];
830 var integer i;
831
832 f_init();
833
834 for (i := 0; i < sizeof(vc_conn); i:=i+1) {
835 var charstring tundev := "ping" & int2str(i);
836 var SessionPars pars := valueof(t_SessionPars(f_gen_imsi(i), tundev));
837 vc_conn[i] := f_start_handler(refers(f_TC_createSession_ping4), pars);
838 }
839
840 for (i := 0; i < lengthof(vc_conn); i:=i+1) {
841 vc_conn[i].done;
842 }
843}
844
Harald Welte4526da92020-03-05 23:08:10 +0100845
846/* create a session, then delete it again */
847private function f_TC_createSession_deleteSession() runs on PGW_Session_CT {
848 f_create_session();
Pau Espin Pedrol1344e472023-10-23 19:54:29 +0200849 f_delete_session(omit, g_teic, Request_accepted);
Harald Welte4526da92020-03-05 23:08:10 +0100850 setverdict(pass);
851}
852testcase TC_createSession_deleteSession() runs on PGW_Test_CT {
853 var PGW_Session_CT vc_conn;
854 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
855 f_init();
856 vc_conn := f_start_handler(refers(f_TC_createSession_deleteSession), pars);
857 vc_conn.done;
858}
859
860/* send a DeleteSessionReq for an unknown/invalid TEID */
861private function f_TC_deleteSession_unknown() runs on PGW_Session_CT {
862 g_teic := f_gtp2_allocate_teid();
863 g_teic_remote := f_rnd_octstring(4);
Pau Espin Pedrol1344e472023-10-23 19:54:29 +0200864 f_delete_session(omit, '00000000'O, Context_Not_Found, false);
Harald Welte4526da92020-03-05 23:08:10 +0100865 setverdict(pass);
866}
867testcase TC_deleteSession_unknown() runs on PGW_Test_CT {
868 var PGW_Session_CT vc_conn;
869 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
870 f_init();
871 vc_conn := f_start_handler(refers(f_TC_deleteSession_unknown), pars);
872 vc_conn.done;
873}
874
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200875/* Test charging over Gy interface. */
876private function f_TC_gy_charging_cc_time() runs on PGW_Session_CT {
877 var default d;
Harald Welte4526da92020-03-05 23:08:10 +0100878
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200879 f_create_session();
Harald Welte4526da92020-03-05 23:08:10 +0100880
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200881 /* We should receive an update even if no traffic is sent: */
882 as_DIA_Gy_CCR(UPDATE_REQUEST);
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200883 f_validate_gy_cc_report(g_rx_gy, VALIDITY_TIME, (3..4), 0, 0);
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200884
885 d := activate(as_DIA_Gy_CCR(UPDATE_REQUEST));
886 f_ping4(mp_ping_hostname);
887 /* Let the CCA reach the GGSN */
888 f_sleep(0.5);
889 deactivate(d);
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200890 f_validate_gy_cc_report(g_rx_gy, VALIDITY_TIME, (3..4), (28..1000), (28..1000));
891
892 as_DIA_Gy_CCR(UPDATE_REQUEST);
893 f_validate_gy_cc_report(g_rx_gy, VALIDITY_TIME, (3..4), ?, ?);
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200894
Pau Espin Pedrol1344e472023-10-23 19:54:29 +0200895 f_delete_session(omit, g_teic, Request_accepted);
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200896 f_validate_gy_cc_report(g_rx_gy, FINAL, (0..1), 0, 0);
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200897 setverdict(pass);
898}
899testcase TC_gy_charging_cc_time() runs on PGW_Test_CT {
900 var PGW_Session_CT vc_conn;
901 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
902 pars.gy_validity_time := 3; /* Grant access for 3 seconds, needs to be re-validated afterwards */
903 f_init();
904 vc_conn := f_start_handler(refers(f_TC_gy_charging_cc_time), pars);
905 vc_conn.done;
906}
Harald Welte4526da92020-03-05 23:08:10 +0100907
908control {
909 execute( TC_tx_echo() );
910 execute( TC_createSession() );
911 execute( TC_createSession_ping4() );
Harald Weltefe595e42020-04-21 22:56:47 +0200912 execute( TC_createSession_ping4_256() );
Harald Welte4526da92020-03-05 23:08:10 +0100913 execute( TC_createSession_deleteSession() );
914 execute( TC_deleteSession_unknown() );
Pau Espin Pedrol431f3462022-04-20 18:45:35 +0200915 execute( TC_gy_charging_cc_time() );
Harald Welte4526da92020-03-05 23:08:10 +0100916}
917
918
919}