blob: 281f81677c776b4c7975ebfcd61b2c46ea57e396 [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
Harald Welte4526da92020-03-05 23:08:10 +0100135/* configuration data for a given Session */
136type record SessionPars {
137 hexstring imsi,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200138 hexstring msisdn optional,
Harald Welte4526da92020-03-05 23:08:10 +0100139 // serving network
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100140 GTP2C_RAT_Type rat_type,
Harald Welte4526da92020-03-05 23:08:10 +0100141 // flags?
142 charstring apn,
143 /* Apn subscribed or non-subscribed */
144 boolean selection_mode,
145 BIT3 pdn_type,
146 /* PAA */
147 /* Max APN Restriction */
148 /* APN-AMBR */
149 octetstring pco optional,
150 octetstring epco optional,
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100151
152 /* TEI (Control) local side */
153 OCT4 teic_local,
154 /* TEI (Control) remote side */
155 OCT4 teic_remote optional,
Harald Welte4526da92020-03-05 23:08:10 +0100156 /* Bearer Contexts to be created */
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100157 BearerConfig bearer optional,
Harald Welte4526da92020-03-05 23:08:10 +0100158
159 charstring tun_dev_name,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200160 charstring tun_netns_name optional,
161
162 /* In seconds. 0 => disabled, !0 => grant over CC-Time period */
163 integer gy_validity_time
Harald Welte4526da92020-03-05 23:08:10 +0100164}
165
166template (value) SessionPars
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100167t_SessionPars(hexstring imsi,
168 charstring tundev,
169 template (omit) hexstring msisdn := '1234'H,
170 GTP2C_RAT_Type rat_type := GTP2C_RAT_EUTRAN,
171 charstring apn := "internet",
172 boolean selection_mode := false,
173 BIT3 pdn_type := '001'B) := {
Harald Welte4526da92020-03-05 23:08:10 +0100174 imsi := imsi,
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100175 msisdn := msisdn,
Harald Welte4526da92020-03-05 23:08:10 +0100176 rat_type := rat_type,
177 apn := apn,
178 selection_mode := selection_mode,
179 pdn_type := pdn_type,
180 pco := omit,
181 epco := omit,
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100182 teic_local := '00000000'O,
183 teic_remote := omit,
184 bearer := {
185 ebi := 5,
186 teid_local := omit,
187 teid_remote := omit
188 },
Harald Welte4526da92020-03-05 23:08:10 +0100189 tun_dev_name := tundev,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200190 tun_netns_name := tundev,
191 gy_validity_time := 0
Harald Welte4526da92020-03-05 23:08:10 +0100192}
193
Harald Welte4526da92020-03-05 23:08:10 +0100194
195type function void_fn() runs on PGW_Session_CT;
196
Harald Weltef4001512020-04-26 21:48:34 +0200197friend function DiameterForwardUnitdataCallback(PDU_DIAMETER msg)
198runs on DIAMETER_Emulation_CT return template PDU_DIAMETER {
199 DIAMETER_UNIT.send(msg);
200 return omit;
201}
202
203friend function f_init_diameter(charstring id) runs on PGW_Test_CT {
204 var DIAMETEROps ops := {
205 create_cb := refers(DIAMETER_Emulation.ExpectedCreateCallback),
Vadim Yanitskiyb46f01e2021-12-06 03:23:13 +0300206 unitdata_cb := refers(DiameterForwardUnitdataCallback),
207 raw := false /* handler mode (IMSI based routing) */
Harald Weltef4001512020-04-26 21:48:34 +0200208 };
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200209 var DIAMETER_conn_parameters pars;
210
211 /* Gx setup: */
212 pars := {
Harald Weltef4001512020-04-26 21:48:34 +0200213 remote_ip := mp_pgw_hostname,
214 remote_sctp_port := -1,
215 local_ip := mp_pcrf_local_ip,
216 local_sctp_port := mp_pcrf_local_port,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200217 origin_host := "pcrf." & mp_diam_realm,
218 origin_realm := mp_diam_realm,
Pau Espin Pedrol33b47492022-03-08 17:43:01 +0100219 auth_app_id := omit,
Harald Weltef4001512020-04-26 21:48:34 +0200220 vendor_app_id := c_DIAMETER_3GPP_Gx_AID
221 };
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200222 vc_Gx := DIAMETER_Emulation_CT.create(id);
223 map(vc_Gx:DIAMETER, system:DIAMETER_CODEC_PT);
224 connect(vc_Gx:DIAMETER_UNIT, self:Gx_UNIT);
225 connect(vc_Gx:DIAMETER_PROC, self:Gx_PROC);
226 vc_Gx.start(DIAMETER_Emulation.main(ops, pars, id));
Harald Weltef4001512020-04-26 21:48:34 +0200227
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200228 /* Gy setup: */
229 pars := {
230 remote_ip := mp_pgw_hostname,
231 remote_sctp_port := -1,
232 local_ip := mp_ocs_local_ip,
233 local_sctp_port := mp_ocs_local_port,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200234 origin_host := "ocs." & mp_diam_realm,
235 origin_realm := mp_diam_realm,
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200236 auth_app_id := c_DIAMETER_CREDIT_CONTROL_AID,
237 vendor_app_id := omit
238 };
239 vc_Gy := DIAMETER_Emulation_CT.create(id);
240 map(vc_Gy:DIAMETER, system:DIAMETER_CODEC_PT);
241 connect(vc_Gy:DIAMETER_UNIT, self:Gy_UNIT);
242 connect(vc_Gy:DIAMETER_PROC, self:Gy_PROC);
243 vc_Gy.start(DIAMETER_Emulation.main(ops, pars, id));
244
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100245 /* S6b setup: */
246 pars := {
247 remote_ip := mp_pgw_hostname,
248 remote_sctp_port := -1,
249 local_ip := mp_aaa_local_ip,
250 local_sctp_port := mp_aaa_local_port,
251 origin_host := "aaa." & mp_diam_realm,
252 origin_realm := mp_diam_realm,
253 auth_app_id := c_DIAMETER_3GPP_S6b_AID,
254 vendor_app_id := c_DIAMETER_3GPP_S6b_AID
255 };
256 vc_S6b := DIAMETER_Emulation_CT.create(id);
257 map(vc_S6b:DIAMETER, system:DIAMETER_CODEC_PT);
258 connect(vc_S6b:DIAMETER_UNIT, self:S6b_UNIT);
259 connect(vc_S6b:DIAMETER_PROC, self:S6b_PROC);
260 vc_S6b.start(DIAMETER_Emulation.main(ops, pars, id));
261
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200262 f_diameter_wait_capability(Gx_UNIT);
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200263 f_diameter_wait_capability(Gy_UNIT);
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100264 f_diameter_wait_capability(S6b_UNIT);
Pau Espin Pedrolbb5f45f2022-04-11 12:36:55 +0200265 /* Give some time for our emulation to get out of SUSPECT list of SUT (3 watchdong ping-pongs):
266 * RFC6733 sec 5.1
267 * RFC3539 sec 3.4.1 [5]
268 * https://github.com/freeDiameter/freeDiameter/blob/master/libfdcore/p_psm.c#L49
269 */
270 f_sleep(1.0);
Harald Weltef4001512020-04-26 21:48:34 +0200271}
272
Pau Espin Pedrol08880f12022-04-07 19:05:15 +0200273private function f_init(float guard_timeout := 60.0) runs on PGW_Test_CT {
274 T_guard.start(guard_timeout);
275 activate(as_Tguard());
276
Harald Welte4526da92020-03-05 23:08:10 +0100277 var Gtp2EmulationCfg cfg := {
278 gtpc_bind_ip := mp_local_hostname_c,
279 gtpc_bind_port := GTP2C_PORT,
280 gtpc_remote_ip := mp_pgw_hostname,
281 gtpc_remote_port := GTP2C_PORT,
Pau Espin Pedrol297333a2024-03-05 11:25:50 +0100282 gtpu_bind_ip := omit, /* using gtpu daemon */
283 gtpu_bind_port := omit, /* using gtpu daemon */
Harald Welte4526da92020-03-05 23:08:10 +0100284 sgw_role := true,
285 use_gtpu_daemon := true
286 };
287
288 vc_GTP2 := GTPv2_Emulation_CT.create("GTP2_EM");
289 map(vc_GTP2:GTP2C, system:GTP2C);
290 connect(vc_GTP2:TEID0, self:TEID0);
Pau Espin Pedrolbddba3a2024-01-09 12:17:01 +0100291 connect(vc_GTP2:CLIENT, self:GTP2);
292 connect(vc_GTP2:CLIENT_PROC, self:GTP2_PROC);
Harald Welte4526da92020-03-05 23:08:10 +0100293 vc_GTP2.start(GTPv2_Emulation.main(cfg));
Harald Weltef4001512020-04-26 21:48:34 +0200294
295 if (mp_pcrf_local_ip != "") {
296 f_init_diameter(testcasename());
297 }
Harald Welte4526da92020-03-05 23:08:10 +0100298}
299
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200300function f_start_handler(void_fn fn, template (omit) SessionPars pars_tmpl := omit)
Harald Welte4526da92020-03-05 23:08:10 +0100301runs on PGW_Test_CT return PGW_Session_CT {
302 var charstring id := testcasename();
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100303 var DIAMETER_ConnHdlr_CT vc_conn_gx, vc_conn_gy, vc_conn_s6b;
Harald Welte4526da92020-03-05 23:08:10 +0100304 var PGW_Session_CT vc_conn;
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200305 var SessionPars pars;
306
307 if (isvalue(pars_tmpl)) {
308 pars := valueof(pars_tmpl);
309 } else {
310 /*TODO: set default values */
311 }
312
Harald Welte4526da92020-03-05 23:08:10 +0100313 vc_conn := PGW_Session_CT.create(id);
314 connect(vc_conn:GTP2, vc_GTP2:CLIENT);
315 connect(vc_conn:GTP2_PROC, vc_GTP2:CLIENT_PROC);
Harald Weltef4001512020-04-26 21:48:34 +0200316
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200317 if (isbound(vc_Gx)) {
318 vc_conn_gx := DIAMETER_ConnHdlr_CT.create(id);
319 connect(vc_conn_gx:DIAMETER, vc_Gx:DIAMETER_CLIENT);
320 connect(vc_conn_gx:DIAMETER_PROC, vc_Gx:DIAMETER_PROC);
321 connect(vc_conn:Gx, vc_conn_gx:DIAMETER_CLIENT);
322 vc_conn_gx.start(f_diam_connhldr_ct_main(pars.imsi));
Harald Weltef4001512020-04-26 21:48:34 +0200323 }
324
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200325 if (isbound(vc_Gy)) {
326 vc_conn_gy := DIAMETER_ConnHdlr_CT.create(id);
327 connect(vc_conn_gy:DIAMETER, vc_Gy:DIAMETER_CLIENT);
328 connect(vc_conn_gy:DIAMETER_PROC, vc_Gy:DIAMETER_PROC);
329 connect(vc_conn:Gy, vc_conn_gy:DIAMETER_CLIENT);
330 vc_conn_gy.start(f_diam_connhldr_ct_main(pars.imsi));
331 }
332
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100333 if (isbound(vc_S6b)) {
334 vc_conn_s6b := DIAMETER_ConnHdlr_CT.create(id);
335 connect(vc_conn_s6b:DIAMETER, vc_S6b:DIAMETER_CLIENT);
336 connect(vc_conn_s6b:DIAMETER_PROC, vc_S6b:DIAMETER_PROC);
337 connect(vc_conn:S6b, vc_conn_s6b:DIAMETER_CLIENT);
338 vc_conn_s6b.start(f_diam_connhldr_ct_main(pars.imsi));
339 }
340
Harald Welte4526da92020-03-05 23:08:10 +0100341 vc_conn.start(f_handler_init(fn, pars));
342 return vc_conn;
343}
344
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200345private function f_handler_init(void_fn fn, SessionPars pars)
Harald Welte4526da92020-03-05 23:08:10 +0100346runs on PGW_Session_CT {
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200347 g_pars := valueof(pars);
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100348 /* allocate + register TEID-C on local side */
349 g_pars.teic_local := f_gtp2_allocate_teid();
350 g_pars.bearer.teid_local := g_pars.teic_local;
Harald Welte4526da92020-03-05 23:08:10 +0100351 fn.apply();
352}
353
Harald Weltefe595e42020-04-21 22:56:47 +0200354private function f_concat_pad(integer tot_len, hexstring prefix, integer suffix) return hexstring {
355 var integer suffix_len := tot_len - lengthof(prefix);
356 var charstring suffix_ch := int2str(suffix);
357 var integer pad_len := suffix_len - lengthof(suffix_ch);
358
359 return prefix & int2hex(0, pad_len) & str2hex(suffix_ch);
360}
361
362function f_gen_imei(integer suffix) return hexstring {
363 return f_concat_pad(14, '49999'H, suffix);
364}
365
366function f_gen_imsi(integer suffix) return hexstring {
367 return f_concat_pad(15, '26242'H, suffix);
368}
369
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100370/* S6b emulation (AAA-Server) */
371private altstep as_DIA_S6b_AAR() runs on PGW_Session_CT {
372 var PDU_DIAMETER rx_dia;
373 [] S6b.receive(tr_DIA_S6b_AAR()) -> value rx_dia {
374 var template (omit) AVP avp;
375 var octetstring sess_id;
376
377 avp := f_DIAMETER_get_avp(rx_dia, c_AVP_Code_BASE_NONE_Session_Id);
378 sess_id := valueof(avp.avp_data.avp_BASE_NONE_Session_Id);
379
380 S6b.send(ts_DIA_S6b_AAA(sess_id, "aaa." & mp_diam_realm,
381 mp_diam_realm, mp_diam_realm,
382 rx_dia.hop_by_hop_id, rx_dia.end_to_end_id));
383 setverdict(pass);
384 }
385 [] S6b.receive(PDU_DIAMETER:?) -> value rx_dia {
386 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
387 log2str("Received unexpected DIAMETER ", rx_dia));
388 }
389}
390
391/* Gx emulation (PCRF)*/
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200392private altstep as_DIA_Gx_CCR(DCC_NONE_CC_Request_Type req_type) runs on PGW_Session_CT {
Harald Weltef4001512020-04-26 21:48:34 +0200393 var PDU_DIAMETER rx_dia;
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200394 [] Gx.receive(tr_DIA_Gx_CCR(req_type := req_type)) -> value rx_dia {
Harald Weltef4001512020-04-26 21:48:34 +0200395 var template (omit) AVP avp;
396 var octetstring sess_id;
397 var AVP_Unsigned32 req_num;
398
399 avp := f_DIAMETER_get_avp(rx_dia, c_AVP_Code_BASE_NONE_Session_Id);
400 sess_id := valueof(avp.avp_data.avp_BASE_NONE_Session_Id);
401
402 avp := f_DIAMETER_get_avp(rx_dia, c_AVP_Code_DCC_NONE_CC_Request_Number);
403 req_num := valueof(avp.avp_data.avp_DCC_NONE_CC_Request_Number);
404
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200405 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 +0200406 req_type, req_num));
407 }
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200408 [] Gx.receive(PDU_DIAMETER:?) -> value rx_dia {
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200409 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
410 log2str("Received unexpected DIAMETER ", rx_dia));
411 }
412}
413
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100414/* Gy emulation (OCS) */
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200415function f_tr_DIA_Gy_CCR(DCC_NONE_CC_Request_Type req_type)
416runs on PGW_Session_CT return template (present) PDU_DIAMETER
417{
418 var template (present) PDU_DIAMETER tpl;
419 var charstring smf_origin_host := "smf." & mp_diam_realm;
420 var template (present) octetstring imsi := ?;
421 var template (present) octetstring msisdn := ?;
422 var template (present) octetstring rat_type := ?;
423 var template (present) OCT4 charging_char := ?;
424 var template (present) OCT1 nsapi := ?;
425 imsi := char2oct(f_dec_TBCD(imsi_hex2oct(g_pars.imsi)));
426 //msisdn := char2oct(f_dec_TBCD(substr(ctx_val.msisdn, 1, lengthof(ctx_val.msisdn) -1)));
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100427 rat_type := int2oct(enum2int(g_pars.rat_type), 1);
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100428 charging_char := char2oct(oct2str('0000'O)); // f_s5s8_create_session() uses hardcoded chg_car := '0000'O
429 nsapi := int2oct(g_pars.bearer.ebi, 1);
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200430 select (req_type) {
431 case (INITIAL_REQUEST) {
432 tpl := tr_DIAMETER(flags:='11000000'B, cmd_code:=Credit_Control,
433 avps := superset(
434 tr_AVP_SessionId,
435 tr_AVP_OriginHost(smf_origin_host),
436 tr_AVP_OriginRealm(mp_diam_realm),
437 tr_AVP_DestinationRealm(mp_diam_realm),
438 tr_AVP_AuthAppId(int2oct(c_DIAMETER_CREDIT_CONTROL_AID, 4)),
439 tr_AVP_ServiceContextId,
440 tr_AVP_CcReqType(req_type),
441 tr_AVP_CcReqNum(?),
442 tr_AVP_EventTimestamp(?),
443 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_IMSI), tr_AVP_SubcrIdData(imsi)}),
444 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_E164), tr_AVP_SubcrIdData(msisdn)}),
445 tr_AVP_RequestedAction(DIRECT_DEBITING),
446 tr_AVP_3GPP_AoCRequestType,
447 tr_AVP_MultipleServicesIndicator,
448 tr_AVP_Multiple_Services_Credit_Control(content := superset(
449 tr_AVP_Requested_Service_Unit,
Pau Espin Pedrol6477d732022-06-03 12:04:40 +0200450 tr_AVP_PCC_3GPP_QoS_Information,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200451 tr_AVP_GI_3GPP_RatType(rat_type)
452 )),
453 tr_AVP_3GPP_ServiceInformation(content := superset(
454 tr_AVP_3GPP_PSInformation(content := superset(
455 tr_AVP_3GPP_ChargingId,
456 tr_AVP_3GPP_PDPType((IPv4,IPv6,IPv4v6)),
457 tr_AVP_3GPP_PDPAddress(tr_AVP_Address((IP,IP6), ?)),
458 tr_AVP_3GPP_SGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pcrf_local_ip))),
459 tr_AVP_3GPP_GGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pgw_hostname))),
460 tr_AVP_3GPP_CalledStationId,
461 tr_AVP_3GPP_SelectionMode,
462 tr_AVP_3GPP_ChargingCharacteristics(charging_char),
463 tr_AVP_3GPP_SGSNMCCMNC,
464 tr_AVP_3GPP_NSAPI(nsapi),
465 /*We don't yet send MS_Tz in CreateSessionReq:
466 tr_AVP_3GPP_MS_TimeZone,*/
467 tr_AVP_3GPP_ULI/*,
468 We don't yet send IMEI in CreateSessionReq:
469 tr_AVP_UserEquipmentInfo({
470 tr_AVP_UserEquipmentInfoType(IMEISV),
471 tr_AVP_UserEquipmentInfoValue(imeisv)
472 })*/
473 ))
474 ))
475 ));
476 }
477 case (UPDATE_REQUEST) {
478 tpl := tr_DIAMETER(flags:='11000000'B, cmd_code:=Credit_Control,
479 avps := superset(
480 tr_AVP_SessionId,
481 tr_AVP_OriginHost(smf_origin_host),
482 tr_AVP_OriginRealm(mp_diam_realm),
483 tr_AVP_DestinationRealm(mp_diam_realm),
484 tr_AVP_AuthAppId(int2oct(c_DIAMETER_CREDIT_CONTROL_AID, 4)),
485 tr_AVP_ServiceContextId,
486 tr_AVP_CcReqType(req_type),
487 tr_AVP_CcReqNum(?),
488 tr_AVP_DestinationHost(?),
489 tr_AVP_EventTimestamp(?),
490 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_IMSI), tr_AVP_SubcrIdData(imsi)}),
491 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_E164), tr_AVP_SubcrIdData(msisdn)}),
492 tr_AVP_RequestedAction(DIRECT_DEBITING),
493 tr_AVP_3GPP_AoCRequestType,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200494 tr_AVP_Multiple_Services_Credit_Control(content := superset(
495 tr_AVP_Requested_Service_Unit,
496 tr_AVP_Used_Service_Unit,
Pau Espin Pedrolcba0f6d2022-05-24 13:49:46 +0200497 /* tr_AVP_3GPP_Reporting_Reason, can be sometimes inside UsedServiceUnit */
Pau Espin Pedrol6477d732022-06-03 12:04:40 +0200498 tr_AVP_PCC_3GPP_QoS_Information,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200499 tr_AVP_GI_3GPP_RatType(rat_type)
500 )),
501 tr_AVP_3GPP_ServiceInformation(content := superset(
502 tr_AVP_3GPP_PSInformation(content := superset(
503 tr_AVP_3GPP_ChargingId,
504 /* tr_AVP_3GPP_PDPType, Only in INIT */
505 tr_AVP_3GPP_PDPAddress(tr_AVP_Address((IP,IP6), ?)),
506 tr_AVP_3GPP_SGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pcrf_local_ip))),
507 tr_AVP_3GPP_GGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pgw_hostname))),
508 tr_AVP_3GPP_CalledStationId,
509 tr_AVP_3GPP_SelectionMode,
510 tr_AVP_3GPP_ChargingCharacteristics(charging_char),
511 tr_AVP_3GPP_SGSNMCCMNC,
512 tr_AVP_3GPP_NSAPI(nsapi),
513 /*We don't yet send MS_Tz in CreateSessionReq:
514 tr_AVP_3GPP_MS_TimeZone,*/
515 tr_AVP_3GPP_ULI/*,
516 We don't yet send IMEI in CreateSessionReq:
517 tr_AVP_UserEquipmentInfo({
518 tr_AVP_UserEquipmentInfoType(IMEISV),
519 tr_AVP_UserEquipmentInfoValue(imeisv)
520 })*/
521 ))
522 ))
523 ));
524 }
525 case (TERMINATION_REQUEST) {
526 tpl := tr_DIAMETER(flags:='11000000'B, cmd_code:=Credit_Control,
527 avps := superset(
528 tr_AVP_SessionId,
529 tr_AVP_OriginHost(smf_origin_host),
530 tr_AVP_OriginRealm(mp_diam_realm),
531 tr_AVP_DestinationRealm(mp_diam_realm),
532 tr_AVP_AuthAppId(int2oct(c_DIAMETER_CREDIT_CONTROL_AID, 4)),
533 tr_AVP_ServiceContextId,
534 tr_AVP_CcReqType(req_type),
535 tr_AVP_CcReqNum(?),
536 tr_AVP_DestinationHost(?),
537 tr_AVP_EventTimestamp(?),
538 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_IMSI), tr_AVP_SubcrIdData(imsi)}),
539 tr_AVP_SubcrId({tr_AVP_SubcrIdType(END_USER_E164), tr_AVP_SubcrIdData(msisdn)}),
540 tr_AVP_TerminationCause(?),
541 tr_AVP_RequestedAction(DIRECT_DEBITING),
542 tr_AVP_3GPP_AoCRequestType,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200543 tr_AVP_Multiple_Services_Credit_Control(content := superset(
544 /* tr_AVP_Requested_Service_Unit, Only in INIT and UPDATE */
545 tr_AVP_Used_Service_Unit,
Pau Espin Pedrolcba0f6d2022-05-24 13:49:46 +0200546 tr_AVP_3GPP_Reporting_Reason(FINAL),
Pau Espin Pedrol6477d732022-06-03 12:04:40 +0200547 tr_AVP_PCC_3GPP_QoS_Information,
Pau Espin Pedrole343a882022-05-18 18:51:11 +0200548 tr_AVP_GI_3GPP_RatType(rat_type)
549 )),
550 tr_AVP_3GPP_ServiceInformation(content := superset(
551 tr_AVP_3GPP_PSInformation(content := superset(
552 tr_AVP_3GPP_ChargingId,
553 /* tr_AVP_3GPP_PDPType, Only in INIT */
554 tr_AVP_3GPP_PDPAddress(tr_AVP_Address((IP,IP6), ?)),
555 tr_AVP_3GPP_SGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pcrf_local_ip))),
556 tr_AVP_3GPP_GGSNAddress(tr_AVP_Address(IP, f_inet_addr(mp_pgw_hostname))),
557 tr_AVP_3GPP_CalledStationId,
558 tr_AVP_3GPP_SelectionMode,
559 tr_AVP_3GPP_ChargingCharacteristics(charging_char),
560 tr_AVP_3GPP_SGSNMCCMNC,
561 tr_AVP_3GPP_NSAPI(nsapi),
562 /*We don't yet send MS_Tz in CreateSessionReq:
563 tr_AVP_3GPP_MS_TimeZone,*/
564 tr_AVP_3GPP_ULI/*,
565 We don't yet send IMEI in CreateSessionReq:
566 tr_AVP_UserEquipmentInfo({
567 tr_AVP_UserEquipmentInfoType(IMEISV),
568 tr_AVP_UserEquipmentInfoValue(imeisv)
569 })*/
570 ))
571 ))
572 ));
573 }
574 }
575 return tpl;
576}
577
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200578private altstep as_DIA_Gy_CCR(DCC_NONE_CC_Request_Type req_type)
579runs on PGW_Session_CT {
580 [] Gy.receive(f_tr_DIA_Gy_CCR(req_type := req_type)) -> value g_rx_gy {
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200581 var template (value) PDU_DIAMETER tx_dia;
582 var template (omit) AVP avp;
583 var octetstring sess_id;
584 var AVP_Unsigned32 req_num;
585
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200586 avp := f_DIAMETER_get_avp(g_rx_gy, c_AVP_Code_BASE_NONE_Session_Id);
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200587 sess_id := valueof(avp.avp_data.avp_BASE_NONE_Session_Id);
588
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200589 avp := f_DIAMETER_get_avp(g_rx_gy, c_AVP_Code_DCC_NONE_CC_Request_Number);
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200590 req_num := valueof(avp.avp_data.avp_DCC_NONE_CC_Request_Number);
591 if (g_pars.gy_validity_time > 0) {
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200592 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 +0200593 req_type, req_num, g_pars.gy_validity_time);
594 } else {
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200595 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 +0200596 req_type, req_num);
597 }
598 Gy.send(tx_dia);
599 }
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200600 [] Gy.receive(PDU_DIAMETER:?) -> value g_rx_gy {
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200601 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +0200602 log2str("Received unexpected DIAMETER Gy", g_rx_gy));
Harald Weltef4001512020-04-26 21:48:34 +0200603 }
604}
605
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100606/* GTPv2C */
607private function is_s2b_iface() runs on PGW_Session_CT return boolean {
608 return (g_pars.rat_type == GTP2C_RAT_WLAN or g_pars.rat_type == GTP2C_RAT_Virtual);
609}
Harald Welte4526da92020-03-05 23:08:10 +0100610
611/* find TEID of given interface type (and optionally instance) */
612private function f_find_teid(FullyQualifiedTEID_List list,
613 template (present) integer if_type,
614 template (present) BIT4 instance := ?)
615return template (omit) FullyQualifiedTEID
616{
617 var integer i;
618 for (i := 0; i < lengthof(list); i := i+1) {
619 if (match(list[i].interfaceType, if_type) and
620 match(list[i].instance, instance)) {
621 return list[i];
622 }
623 }
624 return omit;
625}
626
627/* process one to-be-created bearer context */
628private function process_bctx_create(BearerContextGrouped bctx) runs on PGW_Session_CT
629{
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100630 g_pars.bearer.ebi := bctx.bearerContextIEs.ePS_Bearer_ID.ePS_Bearer_ID_Value;
Harald Welte4526da92020-03-05 23:08:10 +0100631 /* FIXME: Cause */
632
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100633 var integer exp_fteid_if_type;
634 var BIT4 exp_fteid_instance;
635 if (is_s2b_iface()) {
636 exp_fteid_if_type := 33;
637 exp_fteid_instance := '0100'B;
638 } else {
639 exp_fteid_if_type := 5;
640 exp_fteid_instance := '0010'B;
641 }
642
Harald Welte4526da92020-03-05 23:08:10 +0100643 /* find F-TEID of the P-GW U side */
644 var FullyQualifiedTEID rx_fteid;
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100645 rx_fteid := valueof(f_find_teid(bctx.bearerContextIEs.fullyQualifiedTEID,
646 exp_fteid_if_type, exp_fteid_instance));
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100647 g_pars.bearer.teid_remote := rx_fteid.tEID_GRE_Key;
Harald Welte4526da92020-03-05 23:08:10 +0100648 if (rx_fteid.v4_Flag == '1'B) {
649 g_gtpu4_remote := rx_fteid.iPv4_Address;
650 }
651 if (rx_fteid.v6_Flag == '1'B) {
652 g_gtpu6_remote := rx_fteid.iPv6_Address;
653 }
654
655 var UECUPS_CreateTun uecups_create := {
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100656 tx_teid := oct2int(g_pars.bearer.teid_remote),
657 rx_teid := oct2int(g_pars.bearer.teid_local),
Harald Welte4526da92020-03-05 23:08:10 +0100658 user_addr_type := IPV4,
659 user_addr := '00000000'O,
660 local_gtp_ep := valueof(ts_UECUPS_SockAddr(f_inet_addr(mp_local_hostname_u))),
661 remote_gtp_ep := valueof(ts_UECUPS_SockAddr(g_gtpu4_remote)),
662 tun_dev_name := g_pars.tun_dev_name,
663 tun_netns_name := g_pars.tun_netns_name
664 };
665
666 /* create tunnel in daemon */
667 if (isbound(g_ip4_addr)) {
668 uecups_create.user_addr := g_ip4_addr;
669 f_gtp2_create_tunnel(uecups_create);
670 }
671 if (isbound(g_ip6_addr)) {
672 uecups_create.user_addr_type := IPV6;
673 uecups_create.user_addr := g_ip6_addr;
674 f_gtp2_create_tunnel(uecups_create);
675 }
676}
677
678/* create a session on the PGW */
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100679private function f_create_session(template (value) FullyQualifiedTEID fteid_c_ie,
680 template (value) FullyQualifiedTEID fteid_u_ie,
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +0100681 template (omit) UserLocationInfo uli_ie := omit,
682 template (omit) APCO apco := omit,
683 template APCO exp_apco := *) runs on PGW_Session_CT {
Harald Welte4526da92020-03-05 23:08:10 +0100684 var PDU_GTPCv2 rx;
685
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100686 /* Defaults used for s5/s8: */
687 var boolean do_s6b := false;
688 var template APN_Restriction apn_restriction := ?;
689 /* Change behavior when on S2b: */
690 if (is_s2b_iface()) {
691 do_s6b := true;
692 apn_restriction := omit;
693 }
694
Harald Welte4526da92020-03-05 23:08:10 +0100695 var template (value) PDU_GTPCv2 g2c :=
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100696 ts_GTP2C_CreateSessionReq(imsi := g_pars.imsi,
697 msisdn := g_pars.msisdn,
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100698 rat_type := enum2int(g_pars.rat_type),
Harald Welte4526da92020-03-05 23:08:10 +0100699 sender_fteid := fteid_c_ie,
700 apn := f_enc_dns_hostname(g_pars.apn),
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100701 pdn_type := g_pars.pdn_type,
702 teid_list := { fteid_u_ie },
703 chg_car := '0000'O,
704 bearer_id := g_pars.bearer.ebi,
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +0100705 uli := uli_ie,
706 apco := apco);
Vadim Yanitskiybada3c92022-01-20 18:59:07 +0600707 g2c.gtpcv2_pdu.createSessionRequest.servingNetwork := ts_GTP2C_ServingNetwork('001'H, '01F'H);
Harald Welte4526da92020-03-05 23:08:10 +0100708
709 GTP2.send(g2c);
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100710 if (do_s6b and S6b.checkstate("Connected")) {
711 as_DIA_S6b_AAR();
712 }
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200713 if (Gx.checkstate("Connected")) {
714 as_DIA_Gx_CCR(INITIAL_REQUEST);
Harald Weltef4001512020-04-26 21:48:34 +0200715 }
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100716 /* FIXME: When on S2b interface, SMF is not using the Gy interface, unknown reason. */
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200717 if (Gy.checkstate("Connected")) {
718 as_DIA_Gy_CCR(INITIAL_REQUEST);
719 }
Harald Welte4526da92020-03-05 23:08:10 +0100720 alt {
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100721 [] GTP2.receive(tr_GTP2C_CreateSessionResp(d_teid := g_pars.teic_local,
722 cause := Request_accepted,
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +0100723 apn_restriction := apn_restriction,
724 exp_apco := exp_apco)) -> value rx {
Harald Welte4526da92020-03-05 23:08:10 +0100725 /* extract TEIDs */
726 var CreateSessionResponse resp := rx.gtpcv2_pdu.createSessionResponse;
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100727 g_pars.teic_remote := resp.fullyQualifiedTEID[0].tEID_GRE_Key;
Harald Welte4526da92020-03-05 23:08:10 +0100728
729 /* extract allocated address[es] */
730 var PDN_Address_and_Prefix paa := resp.pDN_AddressAllocation.pDN_Address_and_Prefix;
731 if (ischosen(paa.iPv4_Address)) {
732 g_ip4_addr := paa.iPv4_Address;
733 } else if (ischosen(paa.iPv6_Address)) {
734 g_ip6_addr := paa.iPv6_Address.iPv6_Address;
735 g_ip6_plen := paa.iPv6_Address.prefixLength;
736 } else if (ischosen(paa.iPv4_IPv6)) {
737 g_ip4_addr := paa.iPv4_IPv6.iPv4_Address;
738 g_ip6_addr := paa.iPv4_IPv6.iPv6_Address;
739 g_ip6_plen := paa.iPv4_IPv6.prefixLength;
740 }
741 var integer i;
742 for (i := 0; i < lengthof(resp.bearerContextGrouped); i := i+1) {
743 var BearerContextGrouped bctx := resp.bearerContextGrouped[i];
744 select (bctx.instance) {
745 case ('0000'B) { // created
746 process_bctx_create(bctx);
747 }
748 case ('0001'B) { // removed
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100749 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
750 "We don't expect removed bearer contexts yet");
Harald Welte4526da92020-03-05 23:08:10 +0100751 }
752 }
753 }
754 }
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100755 [] GTP2.receive(tr_GTP2C_CreateSessionResp(d_teid:=g_pars.teic_local, cause:=?)) -> value rx {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100756 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
757 log2str("Unexpected CreateSessionResp(cause=",
758 rx.gtpcv2_pdu.createSessionResponse.cause.causeValue, ")"));
Harald Welte4526da92020-03-05 23:08:10 +0100759 }
760 [] GTP2.receive {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100761 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
762 "Unexpected GTPv2 while waiting for CreateSessionResp");
Harald Welte4526da92020-03-05 23:08:10 +0100763 }
764 }
765
766}
767
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100768/* create a session on the PGW on a S5/S8 interface (from SGW )*/
769private function f_s5s8_create_session() runs on PGW_Session_CT {
770 var template (value) FullyQualifiedTEID fteid_c_ie, fteid_u_ie;
771 var template (value) UserLocationInfo uli_ie;
772
773 fteid_c_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPC, g_pars.teic_local, 0,
774 f_inet_addr(mp_local_hostname_c), omit);
775 fteid_u_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPU, g_pars.bearer.teid_local, 2,
776 f_inet_addr(mp_local_hostname_u), omit);
777
778
779 /* open5gs up to 1.2.3 won't accept it without ULI, despite not mandatory */
780 var template (value) TAI tai := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0001'O };
781 var template (value) ECGI ecgi := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0'H, 23 };
782 uli_ie := ts_GTP2C_UserLocInfo(tai := tai, ecgi := ecgi);
783
784 f_create_session(fteid_c_ie, fteid_u_ie, uli_ie);
785
786}
787
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100788/* create a session on the PGW on a S2b interface (from ePDG)*/
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +0100789private function f_s2b_create_session(template (omit) APCO apco := omit,
790 template APCO exp_apco := *) runs on PGW_Session_CT {
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100791 var template (value) FullyQualifiedTEID fteid_c_ie, fteid_u_ie;
792 var template (value) UserLocationInfo uli_ie;
793
794 fteid_c_ie := ts_GTP2C_FTEID(FTEID_IF_S2b_ePDG_GTPC, g_pars.teic_local, 0,
795 f_inet_addr(mp_local_hostname_c), omit);
796 fteid_u_ie := ts_GTP2C_FTEID(FTEID_IF_S2bU_ePDG_GTPU, g_pars.bearer.teid_local, 5,
797 f_inet_addr(mp_local_hostname_u), omit);
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +0100798
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +0100799 var template (value) TAI tai := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0001'O };
800 var template (value) ECGI ecgi := { '0'H, '0'H, '1'H, 'F'H, '0'H, '1'H, '0'H, 23 };
801 uli_ie := ts_GTP2C_UserLocInfo(tai := tai, ecgi := ecgi);
802
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +0100803 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 +0100804
805}
806
Harald Welte4526da92020-03-05 23:08:10 +0100807/* delete the session from the PGW */
Pau Espin Pedrol1344e472023-10-23 19:54:29 +0200808private function f_delete_session(template (omit) GTP2C_Cause tx_cause := omit,
Harald Welte4526da92020-03-05 23:08:10 +0100809 template (present) OCT4 exp_teid,
Pau Espin Pedrol1344e472023-10-23 19:54:29 +0200810 template (present) GTP2C_Cause exp_cause,
Pau Espin Pedrol5b0327b2022-04-11 13:02:36 +0200811 boolean expect_diameter := true) runs on PGW_Session_CT {
Harald Welte4526da92020-03-05 23:08:10 +0100812 var template (value) FullyQualifiedTEID fteid_c_ie
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100813 fteid_c_ie := ts_GTP2C_FTEID(FTEID_IF_S5S8_SGW_GTPC, g_pars.teic_local, 0,
Harald Welte4526da92020-03-05 23:08:10 +0100814 f_inet_addr(mp_local_hostname_c), omit);
815 var template PDU_GTPCv2 g2c :=
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100816 ts_GTP2C_DeleteSessionReq(d_teid := g_pars.teic_remote, cause := tx_cause,
Harald Welte4526da92020-03-05 23:08:10 +0100817 sender_fteid := fteid_c_ie,
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100818 teid_list := {}, bearer_id := g_pars.bearer.ebi);
Harald Welte4526da92020-03-05 23:08:10 +0100819
820 GTP2.send(g2c);
Pau Espin Pedrol8b3123f2022-04-20 17:17:05 +0200821 if (Gx.checkstate("Connected") and expect_diameter) {
822 as_DIA_Gx_CCR(TERMINATION_REQUEST);
Harald Weltef4001512020-04-26 21:48:34 +0200823 }
Pau Espin Pedrol251d0642022-04-20 18:16:45 +0200824 if (Gy.checkstate("Connected") and expect_diameter) {
825 as_DIA_Gy_CCR(TERMINATION_REQUEST);
826 }
Harald Welte4526da92020-03-05 23:08:10 +0100827 alt {
828 [] GTP2.receive(tr_GTP2C_DeleteSessionResp(d_teid := exp_teid, cause := exp_cause)) {
829 setverdict(pass);
830 }
831 [] GTP2.receive(tr_GTP2C_DeleteSessionResp(?, ?)) {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100832 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
833 "Unexpected DeleteSessionResp");
Harald Welte4526da92020-03-05 23:08:10 +0100834 }
835 [] GTP2.receive {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100836 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
837 "Unexpected GTPv2 while waiting for DeleteSessionResp");
Harald Welte4526da92020-03-05 23:08:10 +0100838 }
839 }
840
841 /* destroy tunnel in daemon */
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100842 if (isbound(g_pars.bearer.teid_local)) {
Harald Welte4526da92020-03-05 23:08:10 +0100843 var UECUPS_DestroyTun uecups_destroy := {
844 local_gtp_ep := valueof(ts_UECUPS_SockAddr(f_inet_addr(mp_local_hostname_u))),
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100845 rx_teid := oct2int(g_pars.bearer.teid_local)
Harald Welte4526da92020-03-05 23:08:10 +0100846 };
847 /* FIXME: what about IPv4/IPv6 differentiation? */
848 f_gtp2_destroy_tunnel(uecups_destroy);
849 }
850}
851
852/* start a program on the user plane side; return its PID */
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600853private function f_start_prog(charstring command, boolean redirect_output := true)
854runs on PGW_Session_CT return integer
Harald Welte4526da92020-03-05 23:08:10 +0100855{
856 var UECUPS_StartProgram sprog := {
857 command := command,
858 environment := {},
859 run_as_user := mp_run_prog_as_user,
860 tun_netns_name := g_pars.tun_netns_name
861 };
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100862 g_start_prog_count := g_start_prog_count + 1;
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600863
864 /* Redirect stdout/stderr to the user-specified location */
865 if (redirect_output) {
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100866 var charstring id := testcasename() & "-" & hex2str(g_pars.imsi) & "-" & int2str(g_start_prog_count);
867 var charstring prefix := mp_run_prog_log_path & "/" & id;
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600868 sprog.command := sprog.command & " 1>>" & prefix & ".prog.stdout";
869 sprog.command := sprog.command & " 2>>" & prefix & ".prog.stderr";
870 }
871
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600872 log("Starting a program: ", command);
Harald Welte4526da92020-03-05 23:08:10 +0100873 var UECUPS_StartProgramRes res := f_gtp2_start_program(sprog);
874 if (res.result != OK) {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100875 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
876 log2str("Unable to start program '", command, "'"));
Harald Welte4526da92020-03-05 23:08:10 +0100877 }
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100878 log("Started program '", command, "' with PID ", res.pid);
Harald Welte4526da92020-03-05 23:08:10 +0100879 return res.pid;
880}
881
882/* wait for termination of a given PID with specified exit_code */
883private function f_wait_term(integer pid, template (present) integer exit_code := 0,
884 float tout := 10.0) runs on PGW_Session_CT
885{
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600886 var UECUPS_ProgramTermInd pti;
Harald Welte4526da92020-03-05 23:08:10 +0100887 timer T := tout;
888
889 T.start;
890 alt {
891 [] GTP2.receive(UECUPS_ProgramTermInd:{pid := pid, exit_code := exit_code}) {
892 setverdict(pass);
893 }
Vadim Yanitskiy36ab7972022-02-01 20:46:05 +0600894 [] GTP2.receive(UECUPS_ProgramTermInd:?) -> value pti {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100895 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
896 log2str("Received unexpected ProgramTermInd := ", pti));
Harald Welte4526da92020-03-05 23:08:10 +0100897 }
898 [] T.timeout {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100899 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail,
Pau Espin Pedrol967d8712023-02-28 16:09:32 +0100900 log2str("timeout (", tout, " seconds) waiting for user-plane program PID ", pid, " termination"));
Harald Welte4526da92020-03-05 23:08:10 +0100901 }
902 }
903}
904
905/* execute a program and wait for result */
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600906private function f_start_prog_wait(charstring command,
907 template integer exit_code := 0,
908 float tout := 10.0,
909 boolean redirect_output := true)
910runs on PGW_Session_CT
Harald Welte4526da92020-03-05 23:08:10 +0100911{
Vadim Yanitskiy284c68e2022-02-02 22:30:37 +0600912 var integer pid := f_start_prog(command, redirect_output);
Harald Welte4526da92020-03-05 23:08:10 +0100913 f_wait_term(pid, exit_code, tout);
914}
915
916/* execute ping command and wait for result */
917private function f_ping4(charstring host, integer interval := 1, integer count := 10) runs on PGW_Session_CT
918{
919 var charstring ping :="ping -c " & int2str(count) & " -i " & int2str(interval);
Pau Espin Pedrol55980c42023-02-28 12:40:52 +0100920
921 if (not isbound(g_ip4_addr)) {
922 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail, "f_ping4(): g_ip4_addr is unset!");
923 }
Harald Welte4526da92020-03-05 23:08:10 +0100924 ping := ping & " -I " & f_inet_ntoa(g_ip4_addr);
925 ping := ping & " " & host;
Harald Welte8cfdc7c2020-04-21 22:48:34 +0200926 f_start_prog_wait(ping, tout := int2float(5 + interval*count));
Harald Welte4526da92020-03-05 23:08:10 +0100927}
928
929
930
931
932/* send echo request; expect response */
933testcase TC_tx_echo() runs on PGW_Test_CT {
934 timer T := 5.0;
935
936 f_init();
937
Pau Espin Pedrolbddba3a2024-01-09 12:17:01 +0100938 GTP2.send(ts_GTP2C_EchoReq(0));
Harald Welte4526da92020-03-05 23:08:10 +0100939 T.start;
940 alt {
Pau Espin Pedrolbddba3a2024-01-09 12:17:01 +0100941 [] GTP2.receive(tr_GTP2C_EchoResp) {
Harald Welte4526da92020-03-05 23:08:10 +0100942 setverdict(pass);
943 }
944 [] T.timeout {
Pau Espin Pedrolf97f6e22023-02-28 14:13:42 +0100945 Misc_Helpers.f_shutdown(__BFILE__, __LINE__, fail, "timeout waiting for Echo Response");
Harald Welte4526da92020-03-05 23:08:10 +0100946 }
947 }
948}
949
950/* create a session, expect it to succeed */
951private function f_TC_createSession() runs on PGW_Session_CT {
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100952 f_s5s8_create_session();
Harald Welte4526da92020-03-05 23:08:10 +0100953 setverdict(pass);
954}
955testcase TC_createSession() runs on PGW_Test_CT {
956 var PGW_Session_CT vc_conn;
957 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun22"));
958 f_init();
959 vc_conn := f_start_handler(refers(f_TC_createSession), pars);
960 vc_conn.done;
961}
962
963/* create a session, then execute a ping command on the user plane */
964private function f_TC_createSession_ping4() runs on PGW_Session_CT {
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100965 f_s5s8_create_session();
Harald Welte4526da92020-03-05 23:08:10 +0100966 f_ping4(mp_ping_hostname);
967 setverdict(pass);
968}
969testcase TC_createSession_ping4() runs on PGW_Test_CT {
970 var PGW_Session_CT vc_conn;
971 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
972 f_init();
973 vc_conn := f_start_handler(refers(f_TC_createSession_ping4), pars);
974 vc_conn.done;
975}
Harald Weltefe595e42020-04-21 22:56:47 +0200976testcase TC_createSession_ping4_256() runs on PGW_Test_CT {
977 var PGW_Session_CT vc_conn[256];
978 var integer i;
979
980 f_init();
981
982 for (i := 0; i < sizeof(vc_conn); i:=i+1) {
983 var charstring tundev := "ping" & int2str(i);
984 var SessionPars pars := valueof(t_SessionPars(f_gen_imsi(i), tundev));
985 vc_conn[i] := f_start_handler(refers(f_TC_createSession_ping4), pars);
986 }
987
988 for (i := 0; i < lengthof(vc_conn); i:=i+1) {
989 vc_conn[i].done;
990 }
991}
992
Harald Welte4526da92020-03-05 23:08:10 +0100993
994/* create a session, then delete it again */
995private function f_TC_createSession_deleteSession() runs on PGW_Session_CT {
Pau Espin Pedrol355a1332024-02-23 17:44:14 +0100996 f_s5s8_create_session();
997 f_delete_session(omit, g_pars.teic_local, Request_accepted);
Harald Welte4526da92020-03-05 23:08:10 +0100998 setverdict(pass);
999}
1000testcase TC_createSession_deleteSession() runs on PGW_Test_CT {
1001 var PGW_Session_CT vc_conn;
1002 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
1003 f_init();
1004 vc_conn := f_start_handler(refers(f_TC_createSession_deleteSession), pars);
1005 vc_conn.done;
1006}
1007
1008/* send a DeleteSessionReq for an unknown/invalid TEID */
1009private function f_TC_deleteSession_unknown() runs on PGW_Session_CT {
Pau Espin Pedrol355a1332024-02-23 17:44:14 +01001010 g_pars.teic_remote := f_rnd_octstring(4);
Pau Espin Pedrol1344e472023-10-23 19:54:29 +02001011 f_delete_session(omit, '00000000'O, Context_Not_Found, false);
Harald Welte4526da92020-03-05 23:08:10 +01001012 setverdict(pass);
1013}
1014testcase TC_deleteSession_unknown() runs on PGW_Test_CT {
1015 var PGW_Session_CT vc_conn;
1016 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
1017 f_init();
1018 vc_conn := f_start_handler(refers(f_TC_deleteSession_unknown), pars);
1019 vc_conn.done;
1020}
1021
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001022/* Test charging over Gy interface. */
1023private function f_TC_gy_charging_cc_time() runs on PGW_Session_CT {
1024 var default d;
Harald Welte4526da92020-03-05 23:08:10 +01001025
Pau Espin Pedrol355a1332024-02-23 17:44:14 +01001026 f_s5s8_create_session();
Harald Welte4526da92020-03-05 23:08:10 +01001027
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001028 /* We should receive an update even if no traffic is sent: */
1029 as_DIA_Gy_CCR(UPDATE_REQUEST);
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +02001030 f_validate_gy_cc_report(g_rx_gy, VALIDITY_TIME, (3..4), 0, 0);
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001031
1032 d := activate(as_DIA_Gy_CCR(UPDATE_REQUEST));
1033 f_ping4(mp_ping_hostname);
1034 /* Let the CCA reach the GGSN */
1035 f_sleep(0.5);
1036 deactivate(d);
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +02001037 f_validate_gy_cc_report(g_rx_gy, VALIDITY_TIME, (3..4), (28..1000), (28..1000));
1038
1039 as_DIA_Gy_CCR(UPDATE_REQUEST);
1040 f_validate_gy_cc_report(g_rx_gy, VALIDITY_TIME, (3..4), ?, ?);
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001041
Pau Espin Pedrol355a1332024-02-23 17:44:14 +01001042 f_delete_session(omit, g_pars.teic_local, Request_accepted);
Pau Espin Pedrol2fde1ef2022-05-20 18:53:09 +02001043 f_validate_gy_cc_report(g_rx_gy, FINAL, (0..1), 0, 0);
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001044 setverdict(pass);
1045}
1046testcase TC_gy_charging_cc_time() runs on PGW_Test_CT {
1047 var PGW_Session_CT vc_conn;
1048 var SessionPars pars := valueof(t_SessionPars('001010123456789'H, "tun23"));
1049 pars.gy_validity_time := 3; /* Grant access for 3 seconds, needs to be re-validated afterwards */
1050 f_init();
1051 vc_conn := f_start_handler(refers(f_TC_gy_charging_cc_time), pars);
1052 vc_conn.done;
1053}
Harald Welte4526da92020-03-05 23:08:10 +01001054
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +01001055/* create a session, expect it to succeed */
1056private function f_TC_s2b_createSession_v4_noapco() runs on PGW_Session_CT {
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +01001057 var template (omit) APCO apco := omit;
1058 var template APCO exp_apco := omit;
1059 f_s2b_create_session(apco, exp_apco);
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +01001060 setverdict(pass);
1061}
1062testcase TC_s2b_createSession_v4_noapco() runs on PGW_Test_CT {
1063 var PGW_Session_CT vc_conn;
1064 var SessionPars pars := valueof(t_SessionPars('001010123456789'H,
1065 "tun22",
1066 rat_type := GTP2C_RAT_WLAN));
1067 f_init();
1068 vc_conn := f_start_handler(refers(f_TC_s2b_createSession_v4_noapco), pars);
1069 vc_conn.done;
1070}
1071
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +01001072private function f_TC_s2b_createSession_v4_apco() runs on PGW_Session_CT {
1073 var template (omit) APCO apco := ts_GTP2C_APCO('0000'B,
1074 {ts_GTP2C_PCO_P_DNS_IPv4(''O),
1075 ts_GTP2C_PCO_P_PCSCF_IPv4(''O)});
1076 var template APCO exp_apco := tr_GTP2C_APCO('0000'B,
1077 {tr_GTP2C_PCO_P_DNS_IPv4(?),
1078 *, /* open5gs-smfd can contain several DNS servers */
1079 tr_GTP2C_PCO_P_PCSCF_IPv4(?)});
1080 f_s2b_create_session(apco, exp_apco);
1081 setverdict(pass);
1082}
1083testcase TC_s2b_createSession_v4_apco() runs on PGW_Test_CT {
1084 var PGW_Session_CT vc_conn;
1085 var SessionPars pars := valueof(t_SessionPars('001010123456789'H,
1086 "tun22",
1087 rat_type := GTP2C_RAT_WLAN));
1088 f_init();
1089 vc_conn := f_start_handler(refers(f_TC_s2b_createSession_v4_apco), pars);
1090 vc_conn.done;
1091}
1092
Harald Welte4526da92020-03-05 23:08:10 +01001093control {
1094 execute( TC_tx_echo() );
1095 execute( TC_createSession() );
1096 execute( TC_createSession_ping4() );
Harald Weltefe595e42020-04-21 22:56:47 +02001097 execute( TC_createSession_ping4_256() );
Harald Welte4526da92020-03-05 23:08:10 +01001098 execute( TC_createSession_deleteSession() );
1099 execute( TC_deleteSession_unknown() );
Pau Espin Pedrol431f3462022-04-20 18:45:35 +02001100 execute( TC_gy_charging_cc_time() );
Pau Espin Pedrol518e24b2024-02-23 17:45:08 +01001101 execute( TC_s2b_createSession_v4_noapco() );
Pau Espin Pedrol4ecc8082024-02-26 15:28:12 +01001102 execute( TC_s2b_createSession_v4_apco() );
Harald Welte4526da92020-03-05 23:08:10 +01001103}
1104
1105
1106}