blob: 08cd1a0b8e2c231f5ebb01feecec5c7fddb8095e [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 Pedrol867b1302024-01-24 16:14:28 +010021import from DIAMETER_ts29_212_Templates all;
Pau Espin Pedrol117a94f2023-12-21 16:10:12 +010022import from DIAMETER_ts29_272_Templates all;
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +010023import from DIAMETER_ts29_273_Templates all;
Pau Espin Pedrol867b1302024-01-24 16:14:28 +010024import from DIAMETER_ts32_299_Templates all;
Harald Weltef4001512020-04-26 21:48:34 +020025import from DIAMETER_Emulation all;
26
27
Harald Welte4526da92020-03-05 23:08:10 +010028modulepar {
Vadim Yanitskiy589972f2022-01-20 19:38:16 +060029 charstring mp_pgw_hostname := "127.0.0.4";
Harald Welte4526da92020-03-05 23:08:10 +010030 charstring mp_local_hostname_c := "127.0.0.1";
31 charstring mp_local_hostname_u := "127.0.0.1";
Harald Weltef4001512020-04-26 21:48:34 +020032
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +060033 charstring mp_run_prog_log_path := "/tmp";
Harald Welte4526da92020-03-05 23:08:10 +010034 charstring mp_run_prog_as_user := "laforge";
35 charstring mp_ping_hostname := "10.45.0.1";
Harald Weltef4001512020-04-26 21:48:34 +020036
Vadim Yanitskiy589972f2022-01-20 19:38:16 +060037 charstring mp_pcrf_local_ip := "127.0.0.9";
Harald Weltef4001512020-04-26 21:48:34 +020038 integer mp_pcrf_local_port := 3868;
Pau Espin Pedrol251d0642022-04-20 18:16:45 +020039
40 charstring mp_ocs_local_ip := "127.0.0.9";
41 integer mp_ocs_local_port := 3869;
Pau Espin Pedrole343a882022-05-18 18:51:11 +020042
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +010043 charstring mp_aaa_local_ip := "127.0.0.9";
44 integer mp_aaa_local_port := 3870;
45
Pau Espin Pedrole343a882022-05-18 18:51:11 +020046 charstring mp_diam_realm := "localdomain";
Harald Welte4526da92020-03-05 23:08:10 +010047}
48
49/* main component, we typically have one per testcase */
Pau Espin Pedrolbddba3a2024-01-09 12:17:01 +010050type component PGW_Test_CT extends GTP2_ConnHdlr {
Harald Welte4526da92020-03-05 23:08:10 +010051 var GTPv2_Emulation_CT vc_GTP2;
52 port GTP2EM_PT TEID0;
Harald Weltef4001512020-04-26 21:48:34 +020053
54 /* emulated PCRF */
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020055 var DIAMETER_Emulation_CT vc_Gx;
56 port DIAMETER_PT Gx_UNIT;
57 port DIAMETEREM_PROC_PT Gx_PROC;
Pau Espin Pedrol251d0642022-04-20 18:16:45 +020058 /* emulated OCS */
59 var DIAMETER_Emulation_CT vc_Gy;
60 port DIAMETER_PT Gy_UNIT;
61 port DIAMETEREM_PROC_PT Gy_PROC;
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +010062 /* emulated AAA-Server */
63 var DIAMETER_Emulation_CT vc_S6b;
64 port DIAMETER_PT S6b_UNIT;
65 port DIAMETEREM_PROC_PT S6b_PROC;
Pau Espin Pedrol08880f12022-04-07 19:05:15 +020066 /* global test case guard timer (actual timeout value is set in f_init()) */
67 timer T_guard;
68}
69
70/* global altstep for global guard timer; */
71altstep as_Tguard() runs on PGW_Test_CT {
72 [] T_guard.timeout {
73 setverdict(fail, "Timeout of T_guard");
74 mtc.stop;
75 }
Harald Welte4526da92020-03-05 23:08:10 +010076}
77
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020078type component DIAMETER_ConnHdlr_CT extends DIAMETER_ConnHdlr {
79 port DIAMETER_Conn_PT DIAMETER_CLIENT;
80}
81
82function f_diam_connhldr_ct_main(hexstring imsi) runs on DIAMETER_ConnHdlr_CT {
83 var PDU_DIAMETER msg;
84
85 if (DIAMETER_PROC.checkstate("Connected")) {
Pau Espin Pedroldb017f42023-08-25 19:22:25 +020086 f_diameter_expect_imsi(imsi);
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +020087 }
88
89 while (true) {
90 alt {
91 [] DIAMETER_CLIENT.receive(PDU_DIAMETER:?) -> value msg {
92 DIAMETER.send(msg);
93 }
94 [] DIAMETER.receive(PDU_DIAMETER:?) -> value msg {
95 DIAMETER_CLIENT.send(msg);
96 }
97 }
98 }
99}
100
101
Harald Welte4526da92020-03-05 23:08:10 +0100102/* per-session component; we typically have 1..N per testcase */
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200103type component PGW_Session_CT extends GTP2_ConnHdlr {
Harald Welte4526da92020-03-05 23:08:10 +0100104 var SessionPars g_pars;
105
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200106 port DIAMETER_Conn_PT Gx;
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200107 port DIAMETER_Conn_PT Gy;
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100108 port DIAMETER_Conn_PT S6b;
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200109
Harald Welte4526da92020-03-05 23:08:10 +0100110 /* GTP-U IPv4 address remote sie */
111 var OCT4 g_gtpu4_remote;
112 var OCT16 g_gtpu6_remote;
113
114 /* Address allocation */
115 var OCT4 g_ip4_addr;
116 var OCT16 g_ip6_addr;
117 var integer g_ip6_plen;
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200118
119 /* Store last received Gy message */
120 var PDU_DIAMETER g_rx_gy;
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100121
122 /* number of programs started, used as identifier. */
123 var integer g_start_prog_count := 0;
Harald Welte4526da92020-03-05 23:08:10 +0100124}
125
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100126type record BearerConfig {
127 /* EPS Bearer ID */
128 uint4_t ebi optional,
129 /* TEI (Data) local side */
130 OCT4 teid_local optional,
131 /* TEI (Data) remote side */
132 OCT4 teid_remote optional
133};
134
Pau Espin Pedrol06fba9c2024-03-18 20:12:12 +0100135type record SessionParsGy {
136 /* In seconds. 0 => disabled, !0 => grant over CC-Time period */
137 integer validity_time,
138
139 /* Result-Code to use when sending Gy CCA, usually DIAMETER_SUCCESS */
140 DIAMETER_Resultcode cca_res_code
141};
142
Harald Welte4526da92020-03-05 23:08:10 +0100143/* configuration data for a given Session */
144type record SessionPars {
145 hexstring imsi,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200146 hexstring msisdn optional,
Harald Welte4526da92020-03-05 23:08:10 +0100147 // serving network
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100148 GTP2C_RAT_Type rat_type,
Harald Welte4526da92020-03-05 23:08:10 +0100149 // flags?
150 charstring apn,
151 /* Apn subscribed or non-subscribed */
152 boolean selection_mode,
153 BIT3 pdn_type,
154 /* PAA */
155 /* Max APN Restriction */
156 /* APN-AMBR */
157 octetstring pco optional,
158 octetstring epco optional,
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100159
160 /* TEI (Control) local side */
161 OCT4 teic_local,
162 /* TEI (Control) remote side */
163 OCT4 teic_remote optional,
Harald Welte4526da92020-03-05 23:08:10 +0100164 /* Bearer Contexts to be created */
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100165 BearerConfig bearer optional,
Harald Welte4526da92020-03-05 23:08:10 +0100166
167 charstring tun_dev_name,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200168 charstring tun_netns_name optional,
169
Pau Espin Pedrol06fba9c2024-03-18 20:12:12 +0100170 SessionParsGy gy
Harald Welte4526da92020-03-05 23:08:10 +0100171}
172
173template (value) SessionPars
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100174t_SessionPars(hexstring imsi,
175 charstring tundev,
176 template (omit) hexstring msisdn := '1234'H,
177 GTP2C_RAT_Type rat_type := GTP2C_RAT_EUTRAN,
178 charstring apn := "internet",
179 boolean selection_mode := false,
180 BIT3 pdn_type := '001'B) := {
Harald Welte4526da92020-03-05 23:08:10 +0100181 imsi := imsi,
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100182 msisdn := msisdn,
Harald Welte4526da92020-03-05 23:08:10 +0100183 rat_type := rat_type,
184 apn := apn,
185 selection_mode := selection_mode,
186 pdn_type := pdn_type,
187 pco := omit,
188 epco := omit,
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100189 teic_local := '00000000'O,
190 teic_remote := omit,
191 bearer := {
192 ebi := 5,
193 teid_local := omit,
194 teid_remote := omit
195 },
Harald Welte4526da92020-03-05 23:08:10 +0100196 tun_dev_name := tundev,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200197 tun_netns_name := tundev,
Pau Espin Pedrol06fba9c2024-03-18 20:12:12 +0100198 gy := {
199 validity_time := 0,
200 cca_res_code := DIAMETER_SUCCESS
201 }
Harald Welte4526da92020-03-05 23:08:10 +0100202}
203
Harald Welte4526da92020-03-05 23:08:10 +0100204
205type function void_fn() runs on PGW_Session_CT;
206
Harald Weltef4001512020-04-26 21:48:34 +0200207friend function DiameterForwardUnitdataCallback(PDU_DIAMETER msg)
208runs on DIAMETER_Emulation_CT return template PDU_DIAMETER {
209 DIAMETER_UNIT.send(msg);
210 return omit;
211}
212
213friend function f_init_diameter(charstring id) runs on PGW_Test_CT {
214 var DIAMETEROps ops := {
215 create_cb := refers(DIAMETER_Emulation.ExpectedCreateCallback),
Vadim Yanitskiyb46f01e2021-12-06 03:23:13 +0300216 unitdata_cb := refers(DiameterForwardUnitdataCallback),
217 raw := false /* handler mode (IMSI based routing) */
Harald Weltef4001512020-04-26 21:48:34 +0200218 };
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200219 var DIAMETER_conn_parameters pars;
220
221 /* Gx setup: */
222 pars := {
Harald Weltef4001512020-04-26 21:48:34 +0200223 remote_ip := mp_pgw_hostname,
224 remote_sctp_port := -1,
225 local_ip := mp_pcrf_local_ip,
226 local_sctp_port := mp_pcrf_local_port,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200227 origin_host := "pcrf." & mp_diam_realm,
228 origin_realm := mp_diam_realm,
Pau Espin Pedrol33b47492022-03-08 17:43:01 +0100229 auth_app_id := omit,
Harald Weltef4001512020-04-26 21:48:34 +0200230 vendor_app_id := c_DIAMETER_3GPP_Gx_AID
231 };
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200232 vc_Gx := DIAMETER_Emulation_CT.create(id);
233 map(vc_Gx:DIAMETER, system:DIAMETER_CODEC_PT);
234 connect(vc_Gx:DIAMETER_UNIT, self:Gx_UNIT);
235 connect(vc_Gx:DIAMETER_PROC, self:Gx_PROC);
236 vc_Gx.start(DIAMETER_Emulation.main(ops, pars, id));
Harald Weltef4001512020-04-26 21:48:34 +0200237
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200238 /* Gy setup: */
239 pars := {
240 remote_ip := mp_pgw_hostname,
241 remote_sctp_port := -1,
242 local_ip := mp_ocs_local_ip,
243 local_sctp_port := mp_ocs_local_port,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200244 origin_host := "ocs." & mp_diam_realm,
245 origin_realm := mp_diam_realm,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200246 auth_app_id := c_DIAMETER_CREDIT_CONTROL_AID,
247 vendor_app_id := omit
248 };
249 vc_Gy := DIAMETER_Emulation_CT.create(id);
250 map(vc_Gy:DIAMETER, system:DIAMETER_CODEC_PT);
251 connect(vc_Gy:DIAMETER_UNIT, self:Gy_UNIT);
252 connect(vc_Gy:DIAMETER_PROC, self:Gy_PROC);
253 vc_Gy.start(DIAMETER_Emulation.main(ops, pars, id));
254
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100255 /* S6b setup: */
256 pars := {
257 remote_ip := mp_pgw_hostname,
258 remote_sctp_port := -1,
259 local_ip := mp_aaa_local_ip,
260 local_sctp_port := mp_aaa_local_port,
261 origin_host := "aaa." & mp_diam_realm,
262 origin_realm := mp_diam_realm,
263 auth_app_id := c_DIAMETER_3GPP_S6b_AID,
264 vendor_app_id := c_DIAMETER_3GPP_S6b_AID
265 };
266 vc_S6b := DIAMETER_Emulation_CT.create(id);
267 map(vc_S6b:DIAMETER, system:DIAMETER_CODEC_PT);
268 connect(vc_S6b:DIAMETER_UNIT, self:S6b_UNIT);
269 connect(vc_S6b:DIAMETER_PROC, self:S6b_PROC);
270 vc_S6b.start(DIAMETER_Emulation.main(ops, pars, id));
271
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200272 f_diameter_wait_capability(Gx_UNIT);
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200273 f_diameter_wait_capability(Gy_UNIT);
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100274 f_diameter_wait_capability(S6b_UNIT);
Pau Espin Pedrolbb5f45f2022-04-11 12:36:55 +0200275 /* Give some time for our emulation to get out of SUSPECT list of SUT (3 watchdong ping-pongs):
276 * RFC6733 sec 5.1
277 * RFC3539 sec 3.4.1 [5]
278 * https://github.com/freeDiameter/freeDiameter/blob/master/libfdcore/p_psm.c#L49
279 */
280 f_sleep(1.0);
Harald Weltef4001512020-04-26 21:48:34 +0200281}
282
Pau Espin Pedrol08880f12022-04-07 19:05:15 +0200283private function f_init(float guard_timeout := 60.0) runs on PGW_Test_CT {
284 T_guard.start(guard_timeout);
285 activate(as_Tguard());
286
Harald Welte4526da92020-03-05 23:08:10 +0100287 var Gtp2EmulationCfg cfg := {
288 gtpc_bind_ip := mp_local_hostname_c,
289 gtpc_bind_port := GTP2C_PORT,
290 gtpc_remote_ip := mp_pgw_hostname,
291 gtpc_remote_port := GTP2C_PORT,
Pau Espin Pedrol297333a2024-03-05 11:25:50 +0100292 gtpu_bind_ip := omit, /* using gtpu daemon */
293 gtpu_bind_port := omit, /* using gtpu daemon */
Harald Welte4526da92020-03-05 23:08:10 +0100294 sgw_role := true,
295 use_gtpu_daemon := true
296 };
297
298 vc_GTP2 := GTPv2_Emulation_CT.create("GTP2_EM");
299 map(vc_GTP2:GTP2C, system:GTP2C);
300 connect(vc_GTP2:TEID0, self:TEID0);
Pau Espin Pedrolbddba3a2024-01-09 12:17:01 +0100301 connect(vc_GTP2:CLIENT, self:GTP2);
302 connect(vc_GTP2:CLIENT_PROC, self:GTP2_PROC);
Harald Welte4526da92020-03-05 23:08:10 +0100303 vc_GTP2.start(GTPv2_Emulation.main(cfg));
Harald Weltef4001512020-04-26 21:48:34 +0200304
305 if (mp_pcrf_local_ip != "") {
306 f_init_diameter(testcasename());
307 }
Harald Welte4526da92020-03-05 23:08:10 +0100308}
309
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200310function f_start_handler(void_fn fn, template (omit) SessionPars pars_tmpl := omit)
Harald Welte4526da92020-03-05 23:08:10 +0100311runs on PGW_Test_CT return PGW_Session_CT {
312 var charstring id := testcasename();
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100313 var DIAMETER_ConnHdlr_CT vc_conn_gx, vc_conn_gy, vc_conn_s6b;
Harald Welte4526da92020-03-05 23:08:10 +0100314 var PGW_Session_CT vc_conn;
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200315 var SessionPars pars;
316
317 if (isvalue(pars_tmpl)) {
318 pars := valueof(pars_tmpl);
319 } else {
320 /*TODO: set default values */
321 }
322
Harald Welte4526da92020-03-05 23:08:10 +0100323 vc_conn := PGW_Session_CT.create(id);
324 connect(vc_conn:GTP2, vc_GTP2:CLIENT);
325 connect(vc_conn:GTP2_PROC, vc_GTP2:CLIENT_PROC);
Harald Weltef4001512020-04-26 21:48:34 +0200326
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200327 if (isbound(vc_Gx)) {
328 vc_conn_gx := DIAMETER_ConnHdlr_CT.create(id);
329 connect(vc_conn_gx:DIAMETER, vc_Gx:DIAMETER_CLIENT);
330 connect(vc_conn_gx:DIAMETER_PROC, vc_Gx:DIAMETER_PROC);
331 connect(vc_conn:Gx, vc_conn_gx:DIAMETER_CLIENT);
332 vc_conn_gx.start(f_diam_connhldr_ct_main(pars.imsi));
Harald Weltef4001512020-04-26 21:48:34 +0200333 }
334
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200335 if (isbound(vc_Gy)) {
336 vc_conn_gy := DIAMETER_ConnHdlr_CT.create(id);
337 connect(vc_conn_gy:DIAMETER, vc_Gy:DIAMETER_CLIENT);
338 connect(vc_conn_gy:DIAMETER_PROC, vc_Gy:DIAMETER_PROC);
339 connect(vc_conn:Gy, vc_conn_gy:DIAMETER_CLIENT);
340 vc_conn_gy.start(f_diam_connhldr_ct_main(pars.imsi));
341 }
342
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100343 if (isbound(vc_S6b)) {
344 vc_conn_s6b := DIAMETER_ConnHdlr_CT.create(id);
345 connect(vc_conn_s6b:DIAMETER, vc_S6b:DIAMETER_CLIENT);
346 connect(vc_conn_s6b:DIAMETER_PROC, vc_S6b:DIAMETER_PROC);
347 connect(vc_conn:S6b, vc_conn_s6b:DIAMETER_CLIENT);
348 vc_conn_s6b.start(f_diam_connhldr_ct_main(pars.imsi));
349 }
350
Harald Welte4526da92020-03-05 23:08:10 +0100351 vc_conn.start(f_handler_init(fn, pars));
352 return vc_conn;
353}
354
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200355private function f_handler_init(void_fn fn, SessionPars pars)
Harald Welte4526da92020-03-05 23:08:10 +0100356runs on PGW_Session_CT {
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200357 g_pars := valueof(pars);
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100358 /* allocate + register TEID-C on local side */
359 g_pars.teic_local := f_gtp2_allocate_teid();
360 g_pars.bearer.teid_local := g_pars.teic_local;
Harald Welte4526da92020-03-05 23:08:10 +0100361 fn.apply();
362}
363
Harald Weltefe595e42020-04-21 22:56:47 +0200364private function f_concat_pad(integer tot_len, hexstring prefix, integer suffix) return hexstring {
365 var integer suffix_len := tot_len - lengthof(prefix);
366 var charstring suffix_ch := int2str(suffix);
367 var integer pad_len := suffix_len - lengthof(suffix_ch);
368
369 return prefix & int2hex(0, pad_len) & str2hex(suffix_ch);
370}
371
372function f_gen_imei(integer suffix) return hexstring {
373 return f_concat_pad(14, '49999'H, suffix);
374}
375
376function f_gen_imsi(integer suffix) return hexstring {
377 return f_concat_pad(15, '26242'H, suffix);
378}
379
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100380/* S6b emulation (AAA-Server) */
381private altstep as_DIA_S6b_AAR() runs on PGW_Session_CT {
382 var PDU_DIAMETER rx_dia;
383 [] S6b.receive(tr_DIA_S6b_AAR()) -> value rx_dia {
384 var template (omit) AVP avp;
385 var octetstring sess_id;
386
387 avp := f_DIAMETER_get_avp(rx_dia, c_AVP_Code_BASE_NONE_Session_Id);
388 sess_id := valueof(avp.avp_data.avp_BASE_NONE_Session_Id);
389
390 S6b.send(ts_DIA_S6b_AAA(sess_id, "aaa." & mp_diam_realm,
391 mp_diam_realm, mp_diam_realm,
392 rx_dia.hop_by_hop_id, rx_dia.end_to_end_id));
393 setverdict(pass);
394 }
395 [] S6b.receive(PDU_DIAMETER:?) -> value rx_dia {
396 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
397 log2str("Received unexpected DIAMETER ", rx_dia));
398 }
399}
400
401/* Gx emulation (PCRF)*/
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200402private altstep as_DIA_Gx_CCR(DCC_NONE_CC_Request_Type req_type) runs on PGW_Session_CT {
Harald Weltef4001512020-04-26 21:48:34 +0200403 var PDU_DIAMETER rx_dia;
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200404 [] Gx.receive(tr_DIA_Gx_CCR(req_type := req_type)) -> value rx_dia {
Harald Weltef4001512020-04-26 21:48:34 +0200405 var template (omit) AVP avp;
406 var octetstring sess_id;
407 var AVP_Unsigned32 req_num;
408
409 avp := f_DIAMETER_get_avp(rx_dia, c_AVP_Code_BASE_NONE_Session_Id);
410 sess_id := valueof(avp.avp_data.avp_BASE_NONE_Session_Id);
411
412 avp := f_DIAMETER_get_avp(rx_dia, c_AVP_Code_DCC_NONE_CC_Request_Number);
413 req_num := valueof(avp.avp_data.avp_DCC_NONE_CC_Request_Number);
414
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200415 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 +0200416 req_type, req_num));
417 }
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200418 [] Gx.receive(PDU_DIAMETER:?) -> value rx_dia {
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200419 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
420 log2str("Received unexpected DIAMETER ", rx_dia));
421 }
422}
423
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100424/* Gy emulation (OCS) */
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200425function f_tr_DIA_Gy_CCR(DCC_NONE_CC_Request_Type req_type)
426runs on PGW_Session_CT return template (present) PDU_DIAMETER
427{
428 var template (present) PDU_DIAMETER tpl;
429 var charstring smf_origin_host := "smf." & mp_diam_realm;
430 var template (present) octetstring imsi := ?;
431 var template (present) octetstring msisdn := ?;
432 var template (present) octetstring rat_type := ?;
433 var template (present) OCT4 charging_char := ?;
434 var template (present) OCT1 nsapi := ?;
435 imsi := char2oct(f_dec_TBCD(imsi_hex2oct(g_pars.imsi)));
436 //msisdn := char2oct(f_dec_TBCD(substr(ctx_val.msisdn, 1, lengthof(ctx_val.msisdn) -1)));
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100437 rat_type := int2oct(enum2int(g_pars.rat_type), 1);
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100438 charging_char := char2oct(oct2str('0000'O)); // f_s5s8_create_session() uses hardcoded chg_car := '0000'O
439 nsapi := int2oct(g_pars.bearer.ebi, 1);
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200440 select (req_type) {
441 case (INITIAL_REQUEST) {
442 tpl := tr_DIAMETER(flags:='11000000'B, cmd_code:=Credit_Control,
443 avps := superset(
444 tr_AVP_SessionId,
445 tr_AVP_OriginHost(smf_origin_host),
446 tr_AVP_OriginRealm(mp_diam_realm),
447 tr_AVP_DestinationRealm(mp_diam_realm),
448 tr_AVP_AuthAppId(int2oct(c_DIAMETER_CREDIT_CONTROL_AID, 4)),
449 tr_AVP_ServiceContextId,
450 tr_AVP_CcReqType(req_type),
451 tr_AVP_CcReqNum(?),
452 tr_AVP_EventTimestamp(?),
453 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_IMSI), tr_AVP_SubcrIdData(imsi)}),
454 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_E164), tr_AVP_SubcrIdData(msisdn)}),
455 tr_AVP_RequestedAction(DIRECT_DEBITING),
456 tr_AVP_3GPP_AoCRequestType,
457 tr_AVP_MultipleServicesIndicator,
458 tr_AVP_Multiple_Services_Credit_Control(content := superset(
459 tr_AVP_Requested_Service_Unit,
Pau Espin Pedrol97f95472024-03-25 12:17:43 +0100460 tr_AVP_PCC_3GPP_QoS_Information
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200461 )),
462 tr_AVP_3GPP_ServiceInformation(content := superset(
463 tr_AVP_3GPP_PSInformation(content := superset(
464 tr_AVP_3GPP_ChargingId,
465 tr_AVP_3GPP_PDPType((IPv4,IPv6,IPv4v6)),
466 tr_AVP_3GPP_PDPAddress(tr_AVP_Address((IP,IP6), ?)),
467 tr_AVP_3GPP_SGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pcrf_local_ip))),
468 tr_AVP_3GPP_GGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pgw_hostname))),
469 tr_AVP_3GPP_CalledStationId,
470 tr_AVP_3GPP_SelectionMode,
471 tr_AVP_3GPP_ChargingCharacteristics(charging_char),
472 tr_AVP_3GPP_SGSNMCCMNC,
473 tr_AVP_3GPP_NSAPI(nsapi),
474 /*We don't yet send MS_Tz in CreateSessionReq:
475 tr_AVP_3GPP_MS_TimeZone,*/
Pau Espin Pedrol97f95472024-03-25 12:17:43 +0100476 tr_AVP_3GPP_ULI,
477 tr_AVP_GI_3GPP_RatType(rat_type)/*,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200478 We don't yet send IMEI in CreateSessionReq:
479 tr_AVP_UserEquipmentInfo({
480 tr_AVP_UserEquipmentInfoType(IMEISV),
481 tr_AVP_UserEquipmentInfoValue(imeisv)
482 })*/
483 ))
484 ))
485 ));
486 }
487 case (UPDATE_REQUEST) {
488 tpl := tr_DIAMETER(flags:='11000000'B, cmd_code:=Credit_Control,
489 avps := superset(
490 tr_AVP_SessionId,
491 tr_AVP_OriginHost(smf_origin_host),
492 tr_AVP_OriginRealm(mp_diam_realm),
493 tr_AVP_DestinationRealm(mp_diam_realm),
494 tr_AVP_AuthAppId(int2oct(c_DIAMETER_CREDIT_CONTROL_AID, 4)),
495 tr_AVP_ServiceContextId,
496 tr_AVP_CcReqType(req_type),
497 tr_AVP_CcReqNum(?),
498 tr_AVP_DestinationHost(?),
499 tr_AVP_EventTimestamp(?),
500 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_IMSI), tr_AVP_SubcrIdData(imsi)}),
501 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_E164), tr_AVP_SubcrIdData(msisdn)}),
502 tr_AVP_RequestedAction(DIRECT_DEBITING),
503 tr_AVP_3GPP_AoCRequestType,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200504 tr_AVP_Multiple_Services_Credit_Control(content := superset(
505 tr_AVP_Requested_Service_Unit,
506 tr_AVP_Used_Service_Unit,
Pau Espin Pedrolcba0f6d2022-05-24 13:49:46 +0200507 /* tr_AVP_3GPP_Reporting_Reason, can be sometimes inside UsedServiceUnit */
Pau Espin Pedrol97f95472024-03-25 12:17:43 +0100508 tr_AVP_PCC_3GPP_QoS_Information
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200509 )),
510 tr_AVP_3GPP_ServiceInformation(content := superset(
511 tr_AVP_3GPP_PSInformation(content := superset(
512 tr_AVP_3GPP_ChargingId,
513 /* tr_AVP_3GPP_PDPType, Only in INIT */
514 tr_AVP_3GPP_PDPAddress(tr_AVP_Address((IP,IP6), ?)),
515 tr_AVP_3GPP_SGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pcrf_local_ip))),
516 tr_AVP_3GPP_GGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pgw_hostname))),
517 tr_AVP_3GPP_CalledStationId,
518 tr_AVP_3GPP_SelectionMode,
519 tr_AVP_3GPP_ChargingCharacteristics(charging_char),
520 tr_AVP_3GPP_SGSNMCCMNC,
521 tr_AVP_3GPP_NSAPI(nsapi),
522 /*We don't yet send MS_Tz in CreateSessionReq:
523 tr_AVP_3GPP_MS_TimeZone,*/
Pau Espin Pedrol97f95472024-03-25 12:17:43 +0100524 tr_AVP_3GPP_ULI,
525 tr_AVP_GI_3GPP_RatType(rat_type)/*,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200526 We don't yet send IMEI in CreateSessionReq:
527 tr_AVP_UserEquipmentInfo({
528 tr_AVP_UserEquipmentInfoType(IMEISV),
529 tr_AVP_UserEquipmentInfoValue(imeisv)
530 })*/
531 ))
532 ))
533 ));
534 }
535 case (TERMINATION_REQUEST) {
536 tpl := tr_DIAMETER(flags:='11000000'B, cmd_code:=Credit_Control,
537 avps := superset(
538 tr_AVP_SessionId,
539 tr_AVP_OriginHost(smf_origin_host),
540 tr_AVP_OriginRealm(mp_diam_realm),
541 tr_AVP_DestinationRealm(mp_diam_realm),
542 tr_AVP_AuthAppId(int2oct(c_DIAMETER_CREDIT_CONTROL_AID, 4)),
543 tr_AVP_ServiceContextId,
544 tr_AVP_CcReqType(req_type),
545 tr_AVP_CcReqNum(?),
546 tr_AVP_DestinationHost(?),
547 tr_AVP_EventTimestamp(?),
548 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_IMSI), tr_AVP_SubcrIdData(imsi)}),
549 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_E164), tr_AVP_SubcrIdData(msisdn)}),
550 tr_AVP_TerminationCause(?),
551 tr_AVP_RequestedAction(DIRECT_DEBITING),
552 tr_AVP_3GPP_AoCRequestType,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200553 tr_AVP_Multiple_Services_Credit_Control(content := superset(
554 /* tr_AVP_Requested_Service_Unit, Only in INIT and UPDATE */
555 tr_AVP_Used_Service_Unit,
Pau Espin Pedrolcba0f6d2022-05-24 13:49:46 +0200556 tr_AVP_3GPP_Reporting_Reason(FINAL),
Pau Espin Pedrol97f95472024-03-25 12:17:43 +0100557 tr_AVP_PCC_3GPP_QoS_Information
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200558 )),
559 tr_AVP_3GPP_ServiceInformation(content := superset(
560 tr_AVP_3GPP_PSInformation(content := superset(
561 tr_AVP_3GPP_ChargingId,
562 /* tr_AVP_3GPP_PDPType, Only in INIT */
563 tr_AVP_3GPP_PDPAddress(tr_AVP_Address((IP,IP6), ?)),
564 tr_AVP_3GPP_SGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pcrf_local_ip))),
565 tr_AVP_3GPP_GGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pgw_hostname))),
566 tr_AVP_3GPP_CalledStationId,
567 tr_AVP_3GPP_SelectionMode,
568 tr_AVP_3GPP_ChargingCharacteristics(charging_char),
569 tr_AVP_3GPP_SGSNMCCMNC,
570 tr_AVP_3GPP_NSAPI(nsapi),
571 /*We don't yet send MS_Tz in CreateSessionReq:
572 tr_AVP_3GPP_MS_TimeZone,*/
Pau Espin Pedrol97f95472024-03-25 12:17:43 +0100573 tr_AVP_3GPP_ULI,
574 tr_AVP_GI_3GPP_RatType(rat_type)/*,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200575 We don't yet send IMEI in CreateSessionReq:
576 tr_AVP_UserEquipmentInfo({
577 tr_AVP_UserEquipmentInfoType(IMEISV),
578 tr_AVP_UserEquipmentInfoValue(imeisv)
579 })*/
580 ))
581 ))
582 ));
583 }
584 }
585 return tpl;
586}
587
Pau Espin Pedrol06fba9c2024-03-18 20:12:12 +0100588function f_ts_DIA_Gy_CCA(PDU_DIAMETER rx_gy)
589runs on PGW_Session_CT return template (value) PDU_DIAMETER
590{
591 var template (value) PDU_DIAMETER tx_dia;
592 var template (omit) AVP avp;
593 var octetstring sess_id;
594 var AVP_Unsigned32 req_num;
595 var DCC_NONE_CC_Request_Type req_type;
596 var DIAMETER_Resultcode cca_res_code_success := DIAMETER_SUCCESS;
597
598 avp := f_DIAMETER_get_avp(rx_gy, c_AVP_Code_BASE_NONE_Session_Id);
599 sess_id := valueof(avp.avp_data.avp_BASE_NONE_Session_Id);
600
601 avp := f_DIAMETER_get_avp(rx_gy, c_AVP_Code_DCC_NONE_CC_Request_Type);
602 req_type := valueof(avp.avp_data.avp_DCC_NONE_CC_Request_Type);
603
604 avp := f_DIAMETER_get_avp(rx_gy, c_AVP_Code_DCC_NONE_CC_Request_Number);
605 req_num := valueof(avp.avp_data.avp_DCC_NONE_CC_Request_Number);
606
607 if (g_pars.gy.cca_res_code != cca_res_code_success) {
608 tx_dia := ts_DIA_Gy_CCA(rx_gy.hop_by_hop_id, rx_gy.end_to_end_id, sess_id,
609 g_pars.gy.cca_res_code, req_type, req_num);
610 } else if (g_pars.gy.validity_time > 0) {
611 tx_dia := ts_DIA_Gy_CCA_ValidityTime(rx_gy.hop_by_hop_id, rx_gy.end_to_end_id, sess_id,
612 req_type, req_num, g_pars.gy.validity_time);
613 } else {
614 tx_dia := ts_DIA_Gy_CCA(rx_gy.hop_by_hop_id, rx_gy.end_to_end_id, sess_id,
615 g_pars.gy.cca_res_code, req_type, req_num);
616 }
617 return tx_dia;
618}
619
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200620private altstep as_DIA_Gy_CCR(DCC_NONE_CC_Request_Type req_type)
621runs on PGW_Session_CT {
622 [] Gy.receive(f_tr_DIA_Gy_CCR(req_type := req_type)) -> value g_rx_gy {
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200623 var template (value) PDU_DIAMETER tx_dia;
Pau Espin Pedrol06fba9c2024-03-18 20:12:12 +0100624 tx_dia := f_ts_DIA_Gy_CCA(g_rx_gy);
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200625 Gy.send(tx_dia);
626 }
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200627 [] Gy.receive(PDU_DIAMETER:?) -> value g_rx_gy {
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200628 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200629 log2str("Received unexpected DIAMETER Gy", g_rx_gy));
Harald Weltef4001512020-04-26 21:48:34 +0200630 }
631}
632
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100633/* GTPv2C */
634private function is_s2b_iface() runs on PGW_Session_CT return boolean {
635 return (g_pars.rat_type == GTP2C_RAT_WLAN or g_pars.rat_type == GTP2C_RAT_Virtual);
636}
Harald Welte4526da92020-03-05 23:08:10 +0100637
638/* find TEID of given interface type (and optionally instance) */
639private function f_find_teid(FullyQualifiedTEID_List list,
640 template (present) integer if_type,
641 template (present) BIT4 instance := ?)
642return template (omit) FullyQualifiedTEID
643{
644 var integer i;
645 for (i := 0; i < lengthof(list); i := i+1) {
646 if (match(list[i].interfaceType, if_type) and
647 match(list[i].instance, instance)) {
648 return list[i];
649 }
650 }
651 return omit;
652}
653
654/* process one to-be-created bearer context */
655private function process_bctx_create(BearerContextGrouped bctx) runs on PGW_Session_CT
656{
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100657 g_pars.bearer.ebi := bctx.bearerContextIEs.ePS_Bearer_ID.ePS_Bearer_ID_Value;
Harald Welte4526da92020-03-05 23:08:10 +0100658 /* FIXME: Cause */
659
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100660 var integer exp_fteid_if_type;
661 var BIT4 exp_fteid_instance;
662 if (is_s2b_iface()) {
663 exp_fteid_if_type := 33;
664 exp_fteid_instance := '0100'B;
665 } else {
666 exp_fteid_if_type := 5;
667 exp_fteid_instance := '0010'B;
668 }
669
Harald Welte4526da92020-03-05 23:08:10 +0100670 /* find F-TEID of the P-GW U side */
671 var FullyQualifiedTEID rx_fteid;
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100672 rx_fteid := valueof(f_find_teid(bctx.bearerContextIEs.fullyQualifiedTEID,
673 exp_fteid_if_type, exp_fteid_instance));
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100674 g_pars.bearer.teid_remote := rx_fteid.tEID_GRE_Key;
Harald Welte4526da92020-03-05 23:08:10 +0100675 if (rx_fteid.v4_Flag == '1'B) {
676 g_gtpu4_remote := rx_fteid.iPv4_Address;
677 }
678 if (rx_fteid.v6_Flag == '1'B) {
679 g_gtpu6_remote := rx_fteid.iPv6_Address;
680 }
681
682 var UECUPS_CreateTun uecups_create := {
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100683 tx_teid := oct2int(g_pars.bearer.teid_remote),
684 rx_teid := oct2int(g_pars.bearer.teid_local),
Harald Welte4526da92020-03-05 23:08:10 +0100685 user_addr_type := IPV4,
686 user_addr := '00000000'O,
687 local_gtp_ep := valueof(ts_UECUPS_SockAddr(f_inet_addr(mp_local_hostname_u))),
688 remote_gtp_ep := valueof(ts_UECUPS_SockAddr(g_gtpu4_remote)),
689 tun_dev_name := g_pars.tun_dev_name,
690 tun_netns_name := g_pars.tun_netns_name
691 };
692
693 /* create tunnel in daemon */
694 if (isbound(g_ip4_addr)) {
695 uecups_create.user_addr := g_ip4_addr;
696 f_gtp2_create_tunnel(uecups_create);
697 }
698 if (isbound(g_ip6_addr)) {
699 uecups_create.user_addr_type := IPV6;
700 uecups_create.user_addr := g_ip6_addr;
701 f_gtp2_create_tunnel(uecups_create);
702 }
703}
704
705/* create a session on the PGW */
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100706private function f_create_session(template (value) FullyQualifiedTEID fteid_c_ie,
707 template (value) FullyQualifiedTEID fteid_u_ie,
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +0100708 template (omit) UserLocationInfo uli_ie := omit,
709 template (omit) APCO apco := omit,
710 template APCO exp_apco := *) runs on PGW_Session_CT {
Harald Welte4526da92020-03-05 23:08:10 +0100711 var PDU_GTPCv2 rx;
712
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100713 /* Defaults used for s5/s8: */
714 var boolean do_s6b := false;
715 var template APN_Restriction apn_restriction := ?;
716 /* Change behavior when on S2b: */
717 if (is_s2b_iface()) {
718 do_s6b := true;
719 apn_restriction := omit;
720 }
721
Harald Welte4526da92020-03-05 23:08:10 +0100722 var template (value) PDU_GTPCv2 g2c :=
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100723 ts_GTP2C_CreateSessionReq(imsi := g_pars.imsi,
724 msisdn := g_pars.msisdn,
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100725 rat_type := enum2int(g_pars.rat_type),
Harald Welte4526da92020-03-05 23:08:10 +0100726 sender_fteid := fteid_c_ie,
727 apn := f_enc_dns_hostname(g_pars.apn),
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100728 pdn_type := g_pars.pdn_type,
729 teid_list := { fteid_u_ie },
730 chg_car := '0000'O,
731 bearer_id := g_pars.bearer.ebi,
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +0100732 uli := uli_ie,
733 apco := apco);
Vadim Yanitskiybada3c92022-01-20 18:59:07 +0600734 g2c.gtpcv2_pdu.createSessionRequest.servingNetwork := ts_GTP2C_ServingNetwork('001'H, '01F'H);
Harald Welte4526da92020-03-05 23:08:10 +0100735
736 GTP2.send(g2c);
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100737 if (do_s6b and S6b.checkstate("Connected")) {
738 as_DIA_S6b_AAR();
739 }
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200740 if (Gx.checkstate("Connected")) {
741 as_DIA_Gx_CCR(INITIAL_REQUEST);
Harald Weltef4001512020-04-26 21:48:34 +0200742 }
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100743 /* FIXME: When on S2b interface, SMF is not using the Gy interface, unknown reason. */
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200744 if (Gy.checkstate("Connected")) {
745 as_DIA_Gy_CCR(INITIAL_REQUEST);
746 }
Harald Welte4526da92020-03-05 23:08:10 +0100747 alt {
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100748 [] GTP2.receive(tr_GTP2C_CreateSessionResp(d_teid := g_pars.teic_local,
749 cause := Request_accepted,
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +0100750 apn_restriction := apn_restriction,
751 exp_apco := exp_apco)) -> value rx {
Harald Welte4526da92020-03-05 23:08:10 +0100752 /* extract TEIDs */
753 var CreateSessionResponse resp := rx.gtpcv2_pdu.createSessionResponse;
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100754 g_pars.teic_remote := resp.fullyQualifiedTEID[0].tEID_GRE_Key;
Harald Welte4526da92020-03-05 23:08:10 +0100755
756 /* extract allocated address[es] */
757 var PDN_Address_and_Prefix paa := resp.pDN_AddressAllocation.pDN_Address_and_Prefix;
758 if (ischosen(paa.iPv4_Address)) {
759 g_ip4_addr := paa.iPv4_Address;
760 } else if (ischosen(paa.iPv6_Address)) {
761 g_ip6_addr := paa.iPv6_Address.iPv6_Address;
762 g_ip6_plen := paa.iPv6_Address.prefixLength;
763 } else if (ischosen(paa.iPv4_IPv6)) {
764 g_ip4_addr := paa.iPv4_IPv6.iPv4_Address;
765 g_ip6_addr := paa.iPv4_IPv6.iPv6_Address;
766 g_ip6_plen := paa.iPv4_IPv6.prefixLength;
767 }
768 var integer i;
769 for (i := 0; i < lengthof(resp.bearerContextGrouped); i := i+1) {
770 var BearerContextGrouped bctx := resp.bearerContextGrouped[i];
771 select (bctx.instance) {
772 case ('0000'B) { // created
773 process_bctx_create(bctx);
774 }
775 case ('0001'B) { // removed
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100776 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
777 "We don't expect removed bearer contexts yet");
Harald Welte4526da92020-03-05 23:08:10 +0100778 }
779 }
780 }
781 }
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100782 [] GTP2.receive(tr_GTP2C_CreateSessionResp(d_teid:=g_pars.teic_local, cause:=?)) -> value rx {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100783 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
784 log2str("Unexpected CreateSessionResp(cause=",
785 rx.gtpcv2_pdu.createSessionResponse.cause.causeValue, ")"));
Harald Welte4526da92020-03-05 23:08:10 +0100786 }
787 [] GTP2.receive {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100788 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
789 "Unexpected GTPv2 while waiting for CreateSessionResp");
Harald Welte4526da92020-03-05 23:08:10 +0100790 }
791 }
792
793}
794
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100795/* create a session on the PGW on a S5/S8 interface (from SGW )*/
796private function f_s5s8_create_session() runs on PGW_Session_CT {
797 var template (value) FullyQualifiedTEID fteid_c_ie, fteid_u_ie;
798 var template (value) UserLocationInfo uli_ie;
799
800 fteid_c_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPC, g_pars.teic_local, 0,
801 f_inet_addr(mp_local_hostname_c), omit);
802 fteid_u_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPU, g_pars.bearer.teid_local, 2,
803 f_inet_addr(mp_local_hostname_u), omit);
804
805
806 /* open5gs up to 1.2.3 won't accept it without ULI, despite not mandatory */
807 var template (value) TAI tai := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0001'O };
808 var template (value) ECGI ecgi := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0'H, 23 };
809 uli_ie := ts_GTP2C_UserLocInfo(tai := tai, ecgi := ecgi);
810
811 f_create_session(fteid_c_ie, fteid_u_ie, uli_ie);
812
813}
814
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100815/* create a session on the PGW on a S2b interface (from ePDG)*/
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +0100816private function f_s2b_create_session(template (omit) APCO apco := omit,
817 template APCO exp_apco := *) runs on PGW_Session_CT {
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100818 var template (value) FullyQualifiedTEID fteid_c_ie, fteid_u_ie;
819 var template (value) UserLocationInfo uli_ie;
820
821 fteid_c_ie := ts_GTP2C_FTEID(FTEID_IF_S2b_ePDG_GTPC, g_pars.teic_local, 0,
822 f_inet_addr(mp_local_hostname_c), omit);
823 fteid_u_ie := ts_GTP2C_FTEID(FTEID_IF_S2bU_ePDG_GTPU, g_pars.bearer.teid_local, 5,
824 f_inet_addr(mp_local_hostname_u), omit);
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +0100825
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100826 var template (value) TAI tai := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0001'O };
827 var template (value) ECGI ecgi := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0'H, 23 };
828 uli_ie := ts_GTP2C_UserLocInfo(tai := tai, ecgi := ecgi);
829
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +0100830 f_create_session(fteid_c_ie, fteid_u_ie, uli_ie := uli_ie, apco := apco, exp_apco := exp_apco);
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100831
832}
833
Harald Welte4526da92020-03-05 23:08:10 +0100834/* delete the session from the PGW */
Pau Espin Pedrol1344e472023-10-23 19:54:29 +0200835private function f_delete_session(template (omit) GTP2C_Cause tx_cause := omit,
Pau Espin Pedrol1344e472023-10-23 19:54:29 +0200836 template (present) GTP2C_Cause exp_cause,
Pau Espin Pedrol5b0327b2022-04-11 13:02:36 +0200837 boolean expect_diameter := true) runs on PGW_Session_CT {
Harald Welte4526da92020-03-05 23:08:10 +0100838 var template (value) FullyQualifiedTEID fteid_c_ie
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100839 fteid_c_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPC, g_pars.teic_local, 0,
Harald Welte4526da92020-03-05 23:08:10 +0100840 f_inet_addr(mp_local_hostname_c), omit);
841 var template PDU_GTPCv2 g2c :=
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100842 ts_GTP2C_DeleteSessionReq(d_teid := g_pars.teic_remote, cause := tx_cause,
Harald Welte4526da92020-03-05 23:08:10 +0100843 sender_fteid := fteid_c_ie,
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100844 teid_list := {}, bearer_id := g_pars.bearer.ebi);
Harald Welte4526da92020-03-05 23:08:10 +0100845
846 GTP2.send(g2c);
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200847 if (Gx.checkstate("Connected") and expect_diameter) {
848 as_DIA_Gx_CCR(TERMINATION_REQUEST);
Harald Weltef4001512020-04-26 21:48:34 +0200849 }
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200850 if (Gy.checkstate("Connected") and expect_diameter) {
851 as_DIA_Gy_CCR(TERMINATION_REQUEST);
852 }
Harald Welte4526da92020-03-05 23:08:10 +0100853 alt {
Pau Espin Pedrol44a47632024-04-06 19:34:18 +0200854 [] GTP2.receive(tr_GTP2C_DeleteSessionResp(d_teid := g_pars.teic_local, cause := exp_cause)) {
Harald Welte4526da92020-03-05 23:08:10 +0100855 setverdict(pass);
856 }
857 [] GTP2.receive(tr_GTP2C_DeleteSessionResp(?, ?)) {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100858 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
859 "Unexpected DeleteSessionResp");
Harald Welte4526da92020-03-05 23:08:10 +0100860 }
861 [] GTP2.receive {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100862 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
863 "Unexpected GTPv2 while waiting for DeleteSessionResp");
Harald Welte4526da92020-03-05 23:08:10 +0100864 }
865 }
866
867 /* destroy tunnel in daemon */
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100868 if (isbound(g_pars.bearer.teid_local)) {
Harald Welte4526da92020-03-05 23:08:10 +0100869 var UECUPS_DestroyTun uecups_destroy := {
870 local_gtp_ep := valueof(ts_UECUPS_SockAddr(f_inet_addr(mp_local_hostname_u))),
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100871 rx_teid := oct2int(g_pars.bearer.teid_local)
Harald Welte4526da92020-03-05 23:08:10 +0100872 };
873 /* FIXME: what about IPv4/IPv6 differentiation? */
874 f_gtp2_destroy_tunnel(uecups_destroy);
875 }
876}
877
878/* start a program on the user plane side; return its PID */
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600879private function f_start_prog(charstring command, boolean redirect_output := true)
880runs on PGW_Session_CT return integer
Harald Welte4526da92020-03-05 23:08:10 +0100881{
882 var UECUPS_StartProgram sprog := {
883 command := command,
884 environment := {},
885 run_as_user := mp_run_prog_as_user,
886 tun_netns_name := g_pars.tun_netns_name
887 };
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100888 g_start_prog_count := g_start_prog_count + 1;
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600889
890 /* Redirect stdout/stderr to the user-specified location */
891 if (redirect_output) {
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100892 var charstring id := testcasename() & "-" & hex2str(g_pars.imsi) & "-" & int2str(g_start_prog_count);
893 var charstring prefix := mp_run_prog_log_path & "/" & id;
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600894 sprog.command := sprog.command & " 1>>" & prefix & ".prog.stdout";
895 sprog.command := sprog.command & " 2>>" & prefix & ".prog.stderr";
896 }
897
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600898 log("Starting a program: ", command);
Harald Welte4526da92020-03-05 23:08:10 +0100899 var UECUPS_StartProgramRes res := f_gtp2_start_program(sprog);
900 if (res.result != OK) {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100901 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
902 log2str("Unable to start program '", command, "'"));
Harald Welte4526da92020-03-05 23:08:10 +0100903 }
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100904 log("Started program '", command, "' with PID ", res.pid);
Harald Welte4526da92020-03-05 23:08:10 +0100905 return res.pid;
906}
907
908/* wait for termination of a given PID with specified exit_code */
909private function f_wait_term(integer pid, template (present) integer exit_code := 0,
910 float tout := 10.0) runs on PGW_Session_CT
911{
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600912 var UECUPS_ProgramTermInd pti;
Harald Welte4526da92020-03-05 23:08:10 +0100913 timer T := tout;
914
915 T.start;
916 alt {
917 [] GTP2.receive(UECUPS_ProgramTermInd:{pid := pid, exit_code := exit_code}) {
918 setverdict(pass);
919 }
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600920 [] GTP2.receive(UECUPS_ProgramTermInd:?) -> value pti {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100921 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
922 log2str("Received unexpected ProgramTermInd := ", pti));
Harald Welte4526da92020-03-05 23:08:10 +0100923 }
924 [] T.timeout {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100925 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100926 log2str("timeout (", tout, " seconds) waiting for user-plane program PID ", pid, " termination"));
Harald Welte4526da92020-03-05 23:08:10 +0100927 }
928 }
929}
930
931/* execute a program and wait for result */
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600932private function f_start_prog_wait(charstring command,
933 template integer exit_code := 0,
934 float tout := 10.0,
935 boolean redirect_output := true)
936runs on PGW_Session_CT
Harald Welte4526da92020-03-05 23:08:10 +0100937{
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600938 var integer pid := f_start_prog(command, redirect_output);
Harald Welte4526da92020-03-05 23:08:10 +0100939 f_wait_term(pid, exit_code, tout);
940}
941
942/* execute ping command and wait for result */
943private function f_ping4(charstring host, integer interval := 1, integer count := 10) runs on PGW_Session_CT
944{
945 var charstring ping :="ping -c " & int2str(count) & " -i " & int2str(interval);
Pau Espin Pedrol55980c42023-02-28 12:40:52 +0100946
947 if (not isbound(g_ip4_addr)) {
948 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail, "f_ping4(): g_ip4_addr is unset!");
949 }
Harald Welte4526da92020-03-05 23:08:10 +0100950 ping := ping & " -I " & f_inet_ntoa(g_ip4_addr);
951 ping := ping & " " & host;
Harald Welte8cfdc7c2020-04-21 22:48:34 +0200952 f_start_prog_wait(ping, tout := int2float(5 + interval*count));
Harald Welte4526da92020-03-05 23:08:10 +0100953}
954
955
956
957
958/* send echo request; expect response */
959testcase TC_tx_echo() runs on PGW_Test_CT {
960 timer T := 5.0;
961
962 f_init();
963
Pau Espin Pedrolbddba3a2024-01-09 12:17:01 +0100964 GTP2.send(ts_GTP2C_EchoReq(0));
Harald Welte4526da92020-03-05 23:08:10 +0100965 T.start;
966 alt {
Pau Espin Pedrolbddba3a2024-01-09 12:17:01 +0100967 [] GTP2.receive(tr_GTP2C_EchoResp) {
Harald Welte4526da92020-03-05 23:08:10 +0100968 setverdict(pass);
969 }
970 [] T.timeout {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100971 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail, "timeout waiting for Echo Response");
Harald Welte4526da92020-03-05 23:08:10 +0100972 }
973 }
974}
975
976/* create a session, expect it to succeed */
977private function f_TC_createSession() runs on PGW_Session_CT {
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100978 f_s5s8_create_session();
Harald Welte4526da92020-03-05 23:08:10 +0100979 setverdict(pass);
980}
981testcase TC_createSession() runs on PGW_Test_CT {
982 var PGW_Session_CT vc_conn;
983 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun22"));
984 f_init();
985 vc_conn := f_start_handler(refers(f_TC_createSession), pars);
986 vc_conn.done;
987}
988
989/* create a session, then execute a ping command on the user plane */
990private function f_TC_createSession_ping4() runs on PGW_Session_CT {
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100991 f_s5s8_create_session();
Harald Welte4526da92020-03-05 23:08:10 +0100992 f_ping4(mp_ping_hostname);
993 setverdict(pass);
994}
995testcase TC_createSession_ping4() runs on PGW_Test_CT {
996 var PGW_Session_CT vc_conn;
997 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
998 f_init();
999 vc_conn := f_start_handler(refers(f_TC_createSession_ping4), pars);
1000 vc_conn.done;
1001}
Harald Weltefe595e42020-04-21 22:56:47 +02001002testcase TC_createSession_ping4_256() runs on PGW_Test_CT {
1003 var PGW_Session_CT vc_conn[256];
1004 var integer i;
1005
1006 f_init();
1007
1008 for (i := 0; i < sizeof(vc_conn); i:=i+1) {
1009 var charstring tundev := "ping" & int2str(i);
1010 var SessionPars pars := valueof(t_SessionPars(f_gen_imsi(i), tundev));
1011 vc_conn[i] := f_start_handler(refers(f_TC_createSession_ping4), pars);
1012 }
1013
1014 for (i := 0; i < lengthof(vc_conn); i:=i+1) {
1015 vc_conn[i].done;
1016 }
1017}
1018
Harald Welte4526da92020-03-05 23:08:10 +01001019
1020/* create a session, then delete it again */
1021private function f_TC_createSession_deleteSession() runs on PGW_Session_CT {
Pau Espin Pedrol355a1332024-02-23 17:44:14 +01001022 f_s5s8_create_session();
Pau Espin Pedrol44a47632024-04-06 19:34:18 +02001023 f_delete_session(omit, Request_accepted);
Harald Welte4526da92020-03-05 23:08:10 +01001024 setverdict(pass);
1025}
1026testcase TC_createSession_deleteSession() runs on PGW_Test_CT {
1027 var PGW_Session_CT vc_conn;
1028 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
1029 f_init();
1030 vc_conn := f_start_handler(refers(f_TC_createSession_deleteSession), pars);
1031 vc_conn.done;
1032}
1033
1034/* send a DeleteSessionReq for an unknown/invalid TEID */
1035private function f_TC_deleteSession_unknown() runs on PGW_Session_CT {
Pau Espin Pedrol355a1332024-02-23 17:44:14 +01001036 g_pars.teic_remote := f_rnd_octstring(4);
Pau Espin Pedrol44a47632024-04-06 19:34:18 +02001037 f_delete_session(omit, Context_Not_Found, false);
Harald Welte4526da92020-03-05 23:08:10 +01001038 setverdict(pass);
1039}
1040testcase TC_deleteSession_unknown() runs on PGW_Test_CT {
1041 var PGW_Session_CT vc_conn;
1042 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
1043 f_init();
1044 vc_conn := f_start_handler(refers(f_TC_deleteSession_unknown), pars);
1045 vc_conn.done;
1046}
1047
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001048/* Test charging over Gy interface. */
1049private function f_TC_gy_charging_cc_time() runs on PGW_Session_CT {
1050 var default d;
Harald Welte4526da92020-03-05 23:08:10 +01001051
Pau Espin Pedrol355a1332024-02-23 17:44:14 +01001052 f_s5s8_create_session();
Harald Welte4526da92020-03-05 23:08:10 +01001053
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001054 /* We should receive an update even if no traffic is sent: */
1055 as_DIA_Gy_CCR(UPDATE_REQUEST);
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +02001056 f_validate_gy_cc_report(g_rx_gy, VALIDITY_TIME, (3..4), 0, 0);
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001057
1058 d := activate(as_DIA_Gy_CCR(UPDATE_REQUEST));
1059 f_ping4(mp_ping_hostname);
Pau Espin Pedrol819a3b82024-03-18 20:35:46 +01001060 /* Let the CCA reach the PGW */
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001061 f_sleep(0.5);
1062 deactivate(d);
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +02001063 f_validate_gy_cc_report(g_rx_gy, VALIDITY_TIME, (3..4), (28..1000), (28..1000));
1064
1065 as_DIA_Gy_CCR(UPDATE_REQUEST);
1066 f_validate_gy_cc_report(g_rx_gy, VALIDITY_TIME, (3..4), ?, ?);
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001067
Pau Espin Pedrol44a47632024-04-06 19:34:18 +02001068 f_delete_session(omit, Request_accepted);
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +02001069 f_validate_gy_cc_report(g_rx_gy, FINAL, (0..1), 0, 0);
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001070 setverdict(pass);
1071}
1072testcase TC_gy_charging_cc_time() runs on PGW_Test_CT {
1073 var PGW_Session_CT vc_conn;
1074 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
Pau Espin Pedrol06fba9c2024-03-18 20:12:12 +01001075 pars.gy.validity_time := 3; /* Grant access for 3 seconds, needs to be re-validated afterwards */
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001076 f_init();
1077 vc_conn := f_start_handler(refers(f_TC_gy_charging_cc_time), pars);
1078 vc_conn.done;
1079}
Harald Welte4526da92020-03-05 23:08:10 +01001080
Pau Espin Pedrol0a228ea2024-03-18 20:53:47 +01001081/* Test Gy CCR rejected with CCA Result-Code DIAMETER_AUTHORIZATION_REJECTED (5003) */
1082private function f_TC_gy_ccr_update_rejected() runs on PGW_Session_CT {
1083 var default d;
1084
1085 f_s5s8_create_session();
1086
1087 /* We should receive an update even if no traffic is sent: */
1088 as_DIA_Gy_CCR(UPDATE_REQUEST);
1089
1090 /* Answer next CCR[Update] with CCA Reject: */
1091 g_pars.gy.cca_res_code := DIAMETER_AUTHORIZATION_REJECTED;
1092 as_DIA_Gy_CCR(UPDATE_REQUEST);
1093
1094 /* Expect PGW to tear down the connection as a result: */
1095 as_DIA_Gx_CCR(TERMINATION_REQUEST);
1096 as_DIA_Gy_CCR(TERMINATION_REQUEST);
1097
1098 setverdict(pass);
1099}
1100testcase TC_gy_ccr_update_rejected() runs on PGW_Test_CT {
1101 var PGW_Session_CT vc_conn;
1102 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
1103 pars.gy.validity_time := 3; /* Grant access for 3 seconds, needs to be re-validated afterwards */
1104 f_init();
1105 vc_conn := f_start_handler(refers(f_TC_gy_ccr_update_rejected), pars);
1106 vc_conn.done;
1107}
1108
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +01001109/* create a session, expect it to succeed */
1110private function f_TC_s2b_createSession_v4_noapco() runs on PGW_Session_CT {
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +01001111 var template (omit) APCO apco := omit;
1112 var template APCO exp_apco := omit;
1113 f_s2b_create_session(apco, exp_apco);
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +01001114 setverdict(pass);
1115}
1116testcase TC_s2b_createSession_v4_noapco() runs on PGW_Test_CT {
1117 var PGW_Session_CT vc_conn;
1118 var SessionPars pars := valueof(t_SessionPars('001010123456789'H,
1119 "tun22",
1120 rat_type := GTP2C_RAT_WLAN));
1121 f_init();
1122 vc_conn := f_start_handler(refers(f_TC_s2b_createSession_v4_noapco), pars);
1123 vc_conn.done;
1124}
1125
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +01001126private function f_TC_s2b_createSession_v4_apco() runs on PGW_Session_CT {
1127 var template (omit) APCO apco := ts_GTP2C_APCO('0000'B,
1128 {ts_GTP2C_PCO_P_DNS_IPv4(''O),
1129 ts_GTP2C_PCO_P_PCSCF_IPv4(''O)});
1130 var template APCO exp_apco := tr_GTP2C_APCO('0000'B,
1131 {tr_GTP2C_PCO_P_DNS_IPv4(?),
1132 *, /* open5gs-smfd can contain several DNS servers */
1133 tr_GTP2C_PCO_P_PCSCF_IPv4(?)});
1134 f_s2b_create_session(apco, exp_apco);
1135 setverdict(pass);
1136}
1137testcase TC_s2b_createSession_v4_apco() runs on PGW_Test_CT {
1138 var PGW_Session_CT vc_conn;
1139 var SessionPars pars := valueof(t_SessionPars('001010123456789'H,
1140 "tun22",
1141 rat_type := GTP2C_RAT_WLAN));
1142 f_init();
1143 vc_conn := f_start_handler(refers(f_TC_s2b_createSession_v4_apco), pars);
1144 vc_conn.done;
1145}
1146
Harald Welte4526da92020-03-05 23:08:10 +01001147control {
1148 execute( TC_tx_echo() );
1149 execute( TC_createSession() );
1150 execute( TC_createSession_ping4() );
Harald Weltefe595e42020-04-21 22:56:47 +02001151 execute( TC_createSession_ping4_256() );
Harald Welte4526da92020-03-05 23:08:10 +01001152 execute( TC_createSession_deleteSession() );
1153 execute( TC_deleteSession_unknown() );
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001154 execute( TC_gy_charging_cc_time() );
Pau Espin Pedrol0a228ea2024-03-18 20:53:47 +01001155 execute( TC_gy_ccr_update_rejected() );
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +01001156 execute( TC_s2b_createSession_v4_noapco() );
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +01001157 execute( TC_s2b_createSession_v4_apco() );
Harald Welte4526da92020-03-05 23:08:10 +01001158}
1159
1160
1161}