blob: c44e9df51280157d188ab10f4ae797b014b31389 [file] [log] [blame]
Harald Welte09538f82019-08-01 09:50:25 +02001module BSC_Tests_CBSP {
2
3/* CBSP Integration Tests for OsmoBSC
4 * (C) 2019 by Harald Welte <laforge@gnumonks.org>
5 * All rights reserved.
6 *
7 * Released under the terms of GNU General Public License, Version 2 or
8 * (at your option) any later version.
9 *
10 * SPDX-License-Identifier: GPL-2.0-or-later
11 *
12 * This test suite tests OsmoBSC while emulating the CBC (Cell Broadcast Centre)
13 */
14
15import from General_Types all;
16import from Osmocom_Types all;
17import from GSM_Types all;
18import from IPL4asp_Types all;
19import from BSSAP_Types all;
20import from BSSMAP_Templates all;
21
22import from BSC_Tests all;
23
24import from IPA_Emulation all;
25import from IPA_CodecPort all;
26import from IPA_Types all;
27
Harald Welte187f7a92019-09-05 11:15:20 +020028import from MobileL3_Types all;
29import from MobileL3_RRM_Types all;
30import from L3_Templates all;
31
Harald Welte09538f82019-08-01 09:50:25 +020032import from RSL_Types all;
33import from RSL_Emulation all;
34
35import from CBSP_Types all;
36import from CBSP_Templates all;
37import from CBSP_Adapter all;
38import from CBSP_CodecPort all;
39
40modulepar {
41 charstring mp_cbc_ip := "0.0.0.0";
42 integer mp_cbc_port := 48049;
43 integer mp_bsc_cbsp_port := 48050;
44
Harald Welte49a61e62020-03-30 20:13:32 +020045 /* BTS 0: 001-01-1-0 with CBCH
46 * BTS 1: 001-01-1-1 with CBCH
47 * BTS 2: 001-01-2-1 with CBCH
48 * BTS 3: 001-01-2-3 without CBCH */
49 GsmCgiAbstract mp_cgi_bts0 := { '001'H, '01'H, 1, 0 };
50 GsmCgiAbstract mp_cgi_bts1 := { '001'H, '01'H, 1, 1 };
51 GsmCgiAbstract mp_cgi_bts2 := { '001'H, '01'H, 2, 1 };
52 GsmCgiAbstract mp_cgi_bts3 := { '001'H, '01'H, 2, 3 };
Harald Welte09538f82019-08-01 09:50:25 +020053}
54
55private type record GsmCgiAbstract {
56 GsmMcc mcc,
57 GsmMnc mnc,
58 GsmLac lac,
59 GsmCellId ci
60};
61private template (value) BSSMAP_FIELD_CellIdentification_CGI bssmap_cgi(GsmCgiAbstract cgi) :=
62 ts_BSSMAP_CI_CGI(cgi.mcc, cgi.mnc, cgi.lac, cgi.ci);
63private template (value) BSSMAP_FIELD_CellIdentification_LAC_CI bssmap_lac_ci(GsmCgiAbstract cgi) :=
64 ts_BSSMAP_CI_LAC_CI(cgi.lac, cgi.ci);
65private template (value) BSSMAP_FIELD_CellIdentification_LAI bssmap_lai(GsmCgiAbstract cgi) :=
66 ts_BSSMAP_CI_LAI(cgi.mcc, cgi.mnc, cgi.lac);
67private template (value) OCT2 bssmap_lac(GsmCgiAbstract cgi) := ts_BSSMAP_CI_LAC(cgi.lac);
68private template (value) OCT2 bssmap_ci(GsmCgiAbstract cgi) := ts_BSSMAP_CI_CI(cgi.ci);
69
70type component cbsp_test_CT extends test_CT, CBSP_Adapter_CT {
71}
72
73private altstep as_IgnRSL(template RSL_Message tr) runs on cbsp_test_CT {
Vadim Yanitskiyca5c5202020-05-25 22:03:28 +070074[] IPA_RSL[0].receive(tr_ASP_RSL_UD(tr)) { repeat; }
75[] IPA_RSL[1].receive(tr_ASP_RSL_UD(tr)) { repeat; }
76[] IPA_RSL[2].receive(tr_ASP_RSL_UD(tr)) { repeat; }
Harald Welte09538f82019-08-01 09:50:25 +020077}
78
79private altstep as_FailRSL() runs on cbsp_test_CT {
80var template RSL_Message tr := (tr_RSL_SMSCB_CMD);
81var ASP_RSL_Unitdata rx;
Vadim Yanitskiyca5c5202020-05-25 22:03:28 +070082[] IPA_RSL[0].receive(tr_ASP_RSL_UD(tr)) -> value rx {
Harald Welte09538f82019-08-01 09:50:25 +020083 setverdict(fail, "Received unexpected RSL ", rx);
84 mtc.stop;
85 }
Vadim Yanitskiyca5c5202020-05-25 22:03:28 +070086[] IPA_RSL[1].receive(tr_ASP_RSL_UD(tr)) -> value rx {
Harald Welte09538f82019-08-01 09:50:25 +020087 setverdict(fail, "Received unexpected RSL ", rx);
88 mtc.stop;
89 }
Vadim Yanitskiyca5c5202020-05-25 22:03:28 +070090[] IPA_RSL[2].receive(tr_ASP_RSL_UD(tr)) -> value rx {
Harald Welte09538f82019-08-01 09:50:25 +020091 setverdict(fail, "Received unexpected RSL ", rx);
92 mtc.stop;
93 }
94}
95
Neels Hofmeyr579bfa82020-07-29 00:33:59 +020096private function f_init(float guard_timeout := 30.0) runs on cbsp_test_CT {
97 BSC_Tests.f_init(guard_timeout := guard_timeout);
Harald Welte09538f82019-08-01 09:50:25 +020098 activate(as_IgnRSL((tr_RSL_BCCH_INFO, tr_RSL_SACCH_FILL,
99 tr_RSL_NO_BCCH_INFO, tr_RSL_NO_SACCH_FILL,
100 tr_RSL_MsgTypeD(?))));
101 activate(as_FailRSL());
102}
103private function f_cbsp_init_client() runs on cbsp_test_CT {
104 f_init();
105 CBSP_Adapter.f_connect(mp_bsc_ip, mp_bsc_cbsp_port, "", -1);
106 f_cbsp_init_tail();
107}
108
Neels Hofmeyr579bfa82020-07-29 00:33:59 +0200109private function f_cbsp_init_server(float guard_timeout := 30.0) runs on cbsp_test_CT {
Harald Welte09538f82019-08-01 09:50:25 +0200110 var ASP_Event asp_evt;
111 timer T := 10.0;
112
Neels Hofmeyr579bfa82020-07-29 00:33:59 +0200113 f_init(guard_timeout := guard_timeout);
Harald Welte09538f82019-08-01 09:50:25 +0200114 CBSP_Adapter.f_bind(mp_cbc_ip, mp_cbc_port);
115
116 T.start;
117 alt {
118 [] CBSP[0].receive(ASP_Event:{connOpened:=?}) -> value asp_evt {
119 g_cbsp_conn_id[0] := asp_evt.connOpened.connId;
120 }
121 [] T.timeout {
122 setverdict(fail, "Timeout waiting for incoming connection to CBSP Port");
Neels Hofmeyree3b9272020-07-28 21:20:15 +0200123 mtc.stop;
Harald Welte09538f82019-08-01 09:50:25 +0200124 }
125 }
126 f_cbsp_init_tail();
127}
128private function f_cbsp_init_tail() runs on cbsp_test_CT {
129 interleave {
130 [] CBSP[0].receive(tr_CBSP_Recv(?, tr_CBSP_RESTART(?, CBSP_BC_MSGT_CBS, CBSP_RI_DATA_LOST)));
131 /* should we also expect a restart for emergency related messages? */
132 //[] CBSP[0].receive(tr_CBSP_Recv(?, tr_CBSP_RESTART(?, CBSP_BC_MSGT_EMERG, CBSP_RI_DATA_LOST)));
133 }
134}
135
136function f_gen_page() return CBSP_IE {
137 var integer len := f_rnd_int(82);
138 var octetstring payload := f_rnd_octstring(len);
139 return valueof(ts_CbspMsgContent(payload, len));
140}
141
142function f_cbsp_reset_bss(integer idx) runs on CBSP_Adapter_CT {
143 var template (value) CBSP_PDU tx;
144 timer T := 3.0;
145 tx := ts_CBSP_RESET(cell_list := ts_BSSMAP_CIL_BSS);
146 CBSP[idx].clear;
147 CBSP[idx].send(ts_CBSP_Send(g_cbsp_conn_id[idx], tx));
148 T.start;
149 alt {
150 [] CBSP[idx].receive(tr_CBSP_Recv(?, tr_CBSP_RESET_COMPL(cell_list := ts_BSSMAP_CIL_BSS)));
151 [] CBSP[idx].receive {
152 setverdict(fail, "received unexpected CBSP");
153 mtc.stop;
154 }
155 [] T.timeout {
156 setverdict(fail, "timeout waiting for RESET COMPLETE");
157 mtc.stop;
158 }
159 }
160}
161
162/* send a WRITE CBS to the BSC; expect either COMPLETE or FAILURE in response*/
Harald Welte187f7a92019-09-05 11:15:20 +0200163function f_cbsp_write_emerg(uint16_t msg_id, uint16_t ser_no,
164 template (value) BSSMAP_FIELD_CellIdentificationList cell_list := ts_BSSMAP_CIL_BSS,
165 template (value) uint8_t emerg_ind := 1,
166 template (value) uint16_t warn_type := oct2int('0780'O),
167 template (value) uint16_t warn_per := 5,
168 template BSSMAP_FIELD_CellIdentificationList success_list := ?,
169 template CBSP_FailureListItems fail_list := omit) runs on cbsp_test_CT {
170 var template (value) CBSP_PDU tx;
171 var template CBSP_PDU rx;
172 var CBSP_IEs pages := {f_gen_page()};
173
174 tx := ts_CBSP_WRITE_EMERG(msg_id, ser_no, cell_list, emerg_ind, warn_type, warn_per);
175 CBSP[0].send(ts_CBSP_Send(g_cbsp_conn_id[0], tx));
176 if (istemplatekind(fail_list, "omit")) {
177 rx := tr_CBSP_WRITE_CBS_COMPL(msg_id, ser_no, success_list, omit);
178 } else {
179 rx := tr_CBSP_WRITE_CBS_FAIL(msg_id, ser_no, fail_list, *, success_list, omit);
180 }
181 alt {
182 [] CBSP[0].receive(tr_CBSP_Recv(g_cbsp_conn_id[0], rx)) {
183 setverdict(pass);
184 }
185 [] CBSP[0].receive(tr_CBSP_Recv(g_cbsp_conn_id[0], ?)) {
186 setverdict(fail, "Received unexpected CBSP");
Neels Hofmeyree3b9272020-07-28 21:20:15 +0200187 mtc.stop;
Harald Welte187f7a92019-09-05 11:15:20 +0200188 }
189 }
190}
191
192/* send a WRITE CBS to the BSC; expect either COMPLETE or FAILURE in response*/
Harald Welte09538f82019-08-01 09:50:25 +0200193function f_cbsp_write(uint16_t msg_id, uint16_t ser_no,
194 template (value) BSSMAP_FIELD_CellIdentificationList cell_list := ts_BSSMAP_CIL_BSS,
195 template (value) CBSP_Category category := CBSP_CATEG_NORMAL,
196 uint16_t rep_period := 10, uint16_t num_bcast_req := 1,
197 uint8_t dcs := 0, uint8_t channel_ind := 0, CBSP_IEs content,
198 template BSSMAP_FIELD_CellIdentificationList success_list := ?,
199 template CBSP_FailureListItems fail_list := omit) runs on cbsp_test_CT {
200 var template (value) CBSP_PDU tx;
201 var template CBSP_PDU rx;
Harald Welte09538f82019-08-01 09:50:25 +0200202
203 tx := ts_CBSP_WRITE_CBS(msg_id, ser_no, cell_list, channel_ind, category,
204 rep_period, num_bcast_req, dcs, content);
205 CBSP[0].send(ts_CBSP_Send(g_cbsp_conn_id[0], tx));
206 if (istemplatekind(fail_list, "omit")) {
207 rx := tr_CBSP_WRITE_CBS_COMPL(msg_id, ser_no, success_list, channel_ind);
208 } else {
209 rx := tr_CBSP_WRITE_CBS_FAIL(msg_id, ser_no, fail_list, *, success_list, channel_ind);
210 }
211 alt {
212 [] CBSP[0].receive(tr_CBSP_Recv(g_cbsp_conn_id[0], rx)) {
213 setverdict(pass);
214 }
215 [] CBSP[0].receive(tr_CBSP_Recv(g_cbsp_conn_id[0], ?)) {
216 setverdict(fail, "Received unexpected CBSP");
Neels Hofmeyree3b9272020-07-28 21:20:15 +0200217 mtc.stop;
Harald Welte09538f82019-08-01 09:50:25 +0200218 }
219 }
220}
221
222/* send a REPLACE CBS to the BSC; expect either COMPLETE or FAILURE in response*/
223function f_cbsp_replace(uint16_t msg_id, uint16_t new_ser_no, uint16_t old_ser_no,
224 template (value) BSSMAP_FIELD_CellIdentificationList cell_list := ts_BSSMAP_CIL_BSS,
225 template (value) CBSP_Category category := CBSP_CATEG_NORMAL,
226 uint16_t rep_period := 10, uint16_t num_bcast_req := 1,
227 uint8_t dcs := 0, uint8_t channel_ind := 0, CBSP_IEs content,
228 template BSSMAP_FIELD_CellIdentificationList success_list := ?,
229 template CBSP_FailureListItems fail_list := omit) runs on cbsp_test_CT {
230 var template (value) CBSP_PDU tx;
231 var template CBSP_PDU rx;
Harald Welte09538f82019-08-01 09:50:25 +0200232
233 tx := ts_CBSP_REPLACE_CBS(msg_id, new_ser_no, old_ser_no, cell_list, channel_ind, category,
234 rep_period, num_bcast_req, dcs, content);
235 CBSP[0].send(ts_CBSP_Send(g_cbsp_conn_id[0], tx));
236 if (istemplatekind(fail_list, "omit")) {
237 rx := tr_CBSP_REPLACE_CBS_COMPL(msg_id, new_ser_no, old_ser_no, ?, success_list,
238 channel_ind);
239 } else {
240 rx := tr_CBSP_REPLACE_CBS_FAIL(msg_id, new_ser_no, old_ser_no, fail_list, *, success_list,
241 channel_ind);
242 }
243 alt {
244 [] CBSP[0].receive(tr_CBSP_Recv(g_cbsp_conn_id[0], rx)) {
245 setverdict(pass);
246 }
247 [] CBSP[0].receive(tr_CBSP_Recv(g_cbsp_conn_id[0], ?)) {
248 setverdict(fail, "Received unexpected CBSP");
Neels Hofmeyree3b9272020-07-28 21:20:15 +0200249 mtc.stop;
Harald Welte09538f82019-08-01 09:50:25 +0200250 }
251 }
252}
253/* send a KILL CBS to the BSC; expect either COMPLETE or FAILURE in response*/
254function f_cbsp_kill(uint16_t msg_id, uint16_t ser_no, template (omit) uint8_t channel_ind := 0,
255 template (value) BSSMAP_FIELD_CellIdentificationList cell_list := ts_BSSMAP_CIL_BSS,
256 template BSSMAP_FIELD_CellIdentificationList success_list := ?,
257 template CBSP_FailureListItems fail_list := omit) runs on cbsp_test_CT
258{
259 var template (value) CBSP_PDU tx;
260 var template CBSP_PDU rx;
261
262 tx := ts_CBSP_KILL(msg_id, ser_no, cell_list, channel_ind);
263 CBSP[0].send(ts_CBSP_Send(g_cbsp_conn_id[0], tx));
264 if (istemplatekind(fail_list, "omit")) {
265 rx := tr_CBSP_KILL_COMPL(msg_id, ser_no, compl_list:=*, cell_list:=success_list,
266 channel_ind:=channel_ind);
267 } else {
268 rx := tr_CBSP_KILL_FAIL(msg_id, ser_no, fail_list, compl_list:=*, cell_list:=success_list,
269 channel_ind:=channel_ind);
270 }
271 alt {
272 [] CBSP[0].receive(tr_CBSP_Recv(g_cbsp_conn_id[0], rx)) {
273 setverdict(pass);
274 }
275 [] CBSP[0].receive(tr_CBSP_Recv(g_cbsp_conn_id[0], ?)) {
276 setverdict(fail, "Received unexpected CBSP");
Neels Hofmeyree3b9272020-07-28 21:20:15 +0200277 mtc.stop;
Harald Welte09538f82019-08-01 09:50:25 +0200278 }
279 }
280}
281
Harald Welte09538f82019-08-01 09:50:25 +0200282template (present) RSL_IE_CbCommandType
283tr_RslCbCmdType(template (present) uint2_t lblock := ?, template (present) RSL_CbCommand cmd := ?) := {
284 command := cmd,
285 default_bcast_null := ?,
286 spare := ?,
287 last_block := lblock
288}
289
290/* build a RSL_Message receive template from a CBSP page */
291private function f_page2rsl(CBSP_IE page, uint16_t msg_id, uint16_t ser_no, boolean ext_cbch := false)
292return template (present) RSL_Message
293{
294 var template RSL_Message tr;
295 var integer lblock := page.body.msg_content.user_len / 22;
296 var octetstring payload;
297 if (page.body.msg_content.user_len mod 22 > 0) {
298 lblock := lblock + 1;
299 }
300 payload := int2oct(ser_no, 2) & int2oct(msg_id, 2) & '0011'O & page.body.msg_content.val;
301 tr := tr_RSL_SMSCB_CMD(tr_RslCbCmdType(lblock), f_pad_oct(payload, 88, '00'O));
302 if (ext_cbch) {
303 tr.ies[3] := tr_RSL_IE(RSL_IE_Body:{smscb_chan_ind := 1});
304 tr.ies[4] := *;
305 }
306 return tr;
307}
308
309/***********************************************************************
310 * Test Cases
311 ***********************************************************************/
312
313/* Test if BSC (server) accepts connections from CBC (client) */
314testcase TC_cbsp_bsc_server() runs on cbsp_test_CT {
315 f_cbsp_init_client();
316 setverdict(pass);
317}
318
319/* Test if BSC (client) is connecting to CBC (server) */
320testcase TC_cbsp_bsc_client() runs on cbsp_test_CT {
321 f_cbsp_init_server();
322 setverdict(pass);
323}
324
325/* Test if a BSS-global RESET is executed successfully */
326testcase TC_cbsp_reset_bss() runs on cbsp_test_CT {
327 f_cbsp_init_server();
328
329 f_cbsp_reset_bss(0);
330 setverdict(pass);
331}
332
333testcase TC_cbsp_write() runs on cbsp_test_CT {
334 var template (value) CBSP_PDU tx;
335 var CBSP_IEs pages := {f_gen_page()};
336 f_cbsp_init_server();
337
338 tx := ts_CBSP_WRITE_CBS(msg_id:=23, new_ser_nr:=42, cell_list:=ts_BSSMAP_CIL_BSS,
339 channel_ind:=0, category:=CBSP_CATEG_NORMAL,
340 rep_period:=10, num_bcast_req:=1, dcs := 0,
341 content:=pages);
342
343 CBSP[0].send(ts_CBSP_Send(g_cbsp_conn_id[0], tx));
344 f_sleep(10.0);
345}
346
347/* Write to entire BSS; three cells succeed; one fails (no CBCH) */
348testcase TC_cbsp_write_bss() runs on cbsp_test_CT {
349 var CBSP_IEs pages := {f_gen_page()};
350 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
351 cell_list := ts_BSSMAP_CIL_BSS;
352 f_cbsp_init_server();
353 f_cbsp_write(1, 1001, cell_list, content:=pages,
354 success_list:=tr_BSSMAP_CIL_CGI({?,?,?}), fail_list:={?});
355
356 var template RSL_Message tr := f_page2rsl(pages[0], 1, 1001);
357 interleave {
Vadim Yanitskiyca5c5202020-05-25 22:03:28 +0700358 [] IPA_RSL[0].receive(tr_ASP_RSL_UD(tr)) {}
359 [] IPA_RSL[1].receive(tr_ASP_RSL_UD(tr)) {}
360 [] IPA_RSL[2].receive(tr_ASP_RSL_UD(tr)) {}
Harald Welte09538f82019-08-01 09:50:25 +0200361 }
362}
363
364/* Write to single BTS supporting CBCH: success */
365testcase TC_cbsp_write_bts_cgi() runs on cbsp_test_CT {
366 var CBSP_IEs pages := {f_gen_page()};
367 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
368 cell_list := ts_BSSMAP_CIL_CGI({bssmap_cgi(mp_cgi_bts0)});
369 f_cbsp_init_server();
370 f_cbsp_write(2, 1002, cell_list, content:=pages,
371 success_list:=cell_list, fail_list:=omit);
372 var template RSL_Message tr := f_page2rsl(pages[0], 1, 1001);
Vadim Yanitskiyca5c5202020-05-25 22:03:28 +0700373 IPA_RSL[0].receive(tr_ASP_RSL_UD(tr));
Harald Welte09538f82019-08-01 09:50:25 +0200374 f_sleep(5.0);
375}
376
377/* Write to single BTS not supporting CBCH: failure */
378testcase TC_cbsp_write_bts_no_cbch() runs on cbsp_test_CT {
379 var CBSP_IEs pages := {f_gen_page()};
380 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
381 cell_list := ts_BSSMAP_CIL_CGI({bssmap_cgi(mp_cgi_bts3)});
382 f_cbsp_init_server();
383 f_cbsp_write(3, 1003, cell_list, content:=pages,
384 success_list:=omit, fail_list:={?});
385 f_sleep(5.0);
386}
387
388/* Write to single non-existant BTS */
389testcase TC_cbsp_write_unknown_bts() runs on cbsp_test_CT {
390 var CBSP_IEs pages := {f_gen_page()};
391 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
392 cell_list := ts_BSSMAP_CIL_CGI({ts_BSSMAP_CI_CGI(mp_cgi_bts0.mcc, mp_cgi_bts1.mnc, 22222, 33333)});
393 f_cbsp_init_server();
394 f_cbsp_write(4, 1004, cell_list, content:=pages,
395 success_list:=omit, fail_list:={?});
396 f_sleep(5.0);
397}
398
399/* Write to single BTS using LAC+CI */
400testcase TC_cbsp_write_lac_ci() runs on cbsp_test_CT {
401 var CBSP_IEs pages := {f_gen_page()};
402 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
403 cell_list := ts_BSSMAP_CIL_LAC_CI({bssmap_lac_ci(mp_cgi_bts0)});
404 f_cbsp_init_server();
405 f_cbsp_write(5, 1005, cell_list, content:=pages,
406 success_list:=?, fail_list:=omit);
Vadim Yanitskiyca5c5202020-05-25 22:03:28 +0700407 IPA_RSL[0].receive(tr_ASP_RSL_UD(f_page2rsl(pages[0], 5, 1005)));
Harald Welte09538f82019-08-01 09:50:25 +0200408 f_sleep(5.0);
409}
410
411/* Write to single BTS using CI */
412testcase TC_cbsp_write_ci() runs on cbsp_test_CT {
413 var CBSP_IEs pages := {f_gen_page()};
414 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
415 cell_list := ts_BSSMAP_CIL_CI({bssmap_ci(mp_cgi_bts0)});
416 f_cbsp_init_server();
417 f_cbsp_write(6, 1006, cell_list, content:=pages,
418 success_list:=?, fail_list:=omit);
Vadim Yanitskiyca5c5202020-05-25 22:03:28 +0700419 IPA_RSL[0].receive(tr_ASP_RSL_UD(f_page2rsl(pages[0], 6, 1006)));
Harald Welte09538f82019-08-01 09:50:25 +0200420 f_sleep(5.0);
421}
422
423/* Write to single BTS using LAI */
424testcase TC_cbsp_write_lai() runs on cbsp_test_CT {
425 var CBSP_IEs pages := {f_gen_page()};
426 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
427 cell_list := ts_BSSMAP_CIL_LAI({bssmap_lai(mp_cgi_bts0)});
428 f_cbsp_init_server();
429 f_cbsp_write(7, 1007, cell_list, content:=pages,
430 success_list:=?, fail_list:=omit);
Vadim Yanitskiyca5c5202020-05-25 22:03:28 +0700431 IPA_RSL[0].receive(tr_ASP_RSL_UD(f_page2rsl(pages[0], 7, 1007)));
Harald Welte09538f82019-08-01 09:50:25 +0200432 f_sleep(5.0);
433}
434
435/* Write to two BTS using LAC */
436testcase TC_cbsp_write_lac() runs on cbsp_test_CT {
437 var CBSP_IEs pages := {f_gen_page()};
438 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
439 cell_list := ts_BSSMAP_CIL_LAC({bssmap_lac(mp_cgi_bts0)});
440 f_cbsp_init_server();
441 f_cbsp_write(8, 1008, cell_list, content:=pages,
442 success_list:=?, fail_list:=omit);
443 var template RSL_Message tr := f_page2rsl(pages[0], 8, 1008);
444 interleave {
Vadim Yanitskiyca5c5202020-05-25 22:03:28 +0700445 [] IPA_RSL[0].receive(tr_ASP_RSL_UD(tr));
446 [] IPA_RSL[1].receive(tr_ASP_RSL_UD(tr));
Harald Welte09538f82019-08-01 09:50:25 +0200447 }
448 f_sleep(5.0);
449}
450
451/* Write a message, then replace it */
452testcase TC_cbsp_write_then_replace() runs on cbsp_test_CT {
453 var CBSP_IEs pages := {f_gen_page()};
454 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
455 cell_list := ts_BSSMAP_CIL_LAC_CI({bssmap_lac_ci(mp_cgi_bts0)});
456 f_cbsp_init_server();
457 f_cbsp_write(9, 1009, cell_list, num_bcast_req:=10, content:=pages,
458 success_list:=?, fail_list:=omit);
459 f_cbsp_replace(9, 2009, 1009, cell_list, content:=pages,
460 success_list:=?, fail_list:=omit);
461}
462
463/* Replace a message that doesn't exist: failure */
464testcase TC_cbsp_replace_nonexist() runs on cbsp_test_CT {
465 var CBSP_IEs pages := {f_gen_page()};
466 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
467 cell_list := ts_BSSMAP_CIL_LAC_CI({bssmap_lac_ci(mp_cgi_bts0)});
468 f_cbsp_init_server();
469 f_cbsp_replace(10, 2010, 1010, cell_list, content:=pages,
470 success_list:=omit, fail_list:=?);
471}
472
473/* Write more messages than can be scheduled */
474testcase TC_cbsp_write_too_many() runs on cbsp_test_CT {
475 /* repeating three pages at an interval of 1 is impossible */
476 var CBSP_IEs pages := {f_gen_page(), f_gen_page(), f_gen_page()};
477 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
478 cell_list := ts_BSSMAP_CIL_LAC_CI({bssmap_lac_ci(mp_cgi_bts0)});
479 f_cbsp_init_server();
480 f_cbsp_write(11, 1011, cell_list, rep_period:=1, content:=pages,
481 success_list:=omit, fail_list:=?);
482}
483
484/* Kill message that doesn't exist: failure */
485testcase TC_cbsp_kill_nonexist() runs on cbsp_test_CT {
486 var CBSP_IEs pages := {f_gen_page()};
487 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
488 cell_list := ts_BSSMAP_CIL_LAC_CI({bssmap_lac_ci(mp_cgi_bts0)});
489 f_cbsp_init_server();
490 f_cbsp_kill(12, 1012, 0, cell_list, success_list:=omit, fail_list:=?);
491}
492/* Write a message, then kill it */
493testcase TC_cbsp_write_then_kill() runs on cbsp_test_CT {
494 var CBSP_IEs pages := {f_gen_page()};
495 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
496 cell_list := ts_BSSMAP_CIL_LAC_CI({bssmap_lac_ci(mp_cgi_bts0)});
497 f_cbsp_init_server();
498 f_cbsp_write(13, 1013, cell_list, content:=pages, success_list:=?, fail_list:=omit);
499 f_cbsp_kill(13, 1013, 0, cell_list, success_list:=?, fail_list:=omit);
500}
501
502/* Write a message, then reset all messages */
503testcase TC_cbsp_write_then_reset() runs on cbsp_test_CT {
504 var CBSP_IEs pages := {f_gen_page()};
505 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
506 cell_list := ts_BSSMAP_CIL_LAC_CI({bssmap_lac_ci(mp_cgi_bts0)});
507 f_cbsp_init_server();
508 f_cbsp_write(14, 1014, cell_list, content:=pages, success_list:=?, fail_list:=omit);
509 f_cbsp_reset_bss(0);
510}
511
Harald Welte187f7a92019-09-05 11:15:20 +0200512private const octetstring c_ETWS_sec_default :=
513 '00000000000000000000000000000000000000000000000000'O &
514 '00000000000000000000000000000000000000000000000000'O;
515function f_gen_etws_pn(uint16_t ser_nr, uint16_t msg_id, OCT2 msg_type := '0780'O,
516 octetstring sec_inf := c_ETWS_sec_default) return octetstring {
517 return int2oct(ser_nr, 2) & int2oct(msg_id, 2) & msg_type & sec_inf;
518}
519
520/* Write ETWS PN to single BTS; verify it arrives on DCHAN */
521testcase TC_cbsp_emerg_write_bts_cgi_dchan() runs on cbsp_test_CT {
522 var CBSP_IEs pages := {f_gen_page()};
523 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
524 var ASP_RSL_Unitdata rx_rsl_ud;
525
526 cell_list := ts_BSSMAP_CIL_CGI({bssmap_cgi(mp_cgi_bts0)});
527 f_cbsp_init_server();
528
529 /* first establish a dedicated channel */
530 var DchanTuple dt := f_est_dchan('23'O, 23, '00010203040506'O);
531
532 /* then send ETWS PN */
533 f_cbsp_write_emerg(15, 1015, cell_list);
534 var template (present) octetstring tr_apdu := f_gen_etws_pn(1015, 15);
535 timer T := 5.0;
536 T.start;
537 alt {
Vadim Yanitskiy9b4e3562020-05-25 21:40:52 +0700538 [] IPA_RSL[0].receive(tr_ASP_RSL_UD(tr_RSL_DATA_REQ(dt.rsl_chan_nr, ?, ?))) -> value rx_rsl_ud {
Harald Welte187f7a92019-09-05 11:15:20 +0200539 var RSL_IE_Body l3_ie;
540 if (f_rsl_find_ie(rx_rsl_ud.rsl, RSL_IE_L3_INFO, l3_ie) == false) {
541 setverdict(fail, "RSL DATA REQ without L3?");
542 mtc.stop;
543 }
544 var PDU_ML3_NW_MS l3 := dec_PDU_ML3_NW_MS(l3_ie.l3_info.payload);
545 var template (present) APDU_Flags_V tr_flags := {
546 lastSeg := '0'B,
547 firstSeg := '0'B,
548 cR := '0'B,
549 spare := '0'B
550 };
551 if (match(l3, tr_RR_APP_INFO('0001'B, tr_apdu, tr_flags))) {
552 setverdict(pass);
553 }
554 }
555 [] IPA_RSL[0].receive { repeat; }
556 [] T.timeout {
557 setverdict(fail, "Waiting for APP INFO");
558 }
559 }
560}
561
Harald Welte0b5e0f92019-09-07 08:30:25 +0200562/* Write ETWS PN to single BTS; verify it arrives on CCHAN */
563testcase TC_cbsp_emerg_write_bts_cgi_cchan() runs on cbsp_test_CT {
564 var CBSP_IEs pages := {f_gen_page()};
565 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
566 var ASP_RSL_Unitdata rx_rsl_ud;
567
568 cell_list := ts_BSSMAP_CIL_CGI({bssmap_cgi(mp_cgi_bts0)});
569 f_cbsp_init_server();
570
571 f_cbsp_write_emerg(16, 1016, cell_list);
572 var template (present) octetstring tr_apdu := f_gen_etws_pn(1016, 16);
573 timer T := 5.0;
574 T.start;
575 alt {
Vadim Yanitskiy9b4e3562020-05-25 21:40:52 +0700576 [] IPA_RSL[0].receive(tr_ASP_RSL_UD(tr_RSL_OSMO_ETWS_CMD(t_RslChanNr_PCH_AGCH(0), tr_apdu))) {
Harald Welte0b5e0f92019-09-07 08:30:25 +0200577 setverdict(pass);
578 }
Vadim Yanitskiy9b4e3562020-05-25 21:40:52 +0700579 [] IPA_RSL[0].receive(tr_ASP_RSL_UD(tr_RSL_OSMO_ETWS_CMD(?,?))) {
Harald Welte0b5e0f92019-09-07 08:30:25 +0200580 setverdict(fail, "Received unexpected OSMO_ETWS_CMD");
Neels Hofmeyree3b9272020-07-28 21:20:15 +0200581 mtc.stop;
Harald Welte0b5e0f92019-09-07 08:30:25 +0200582 }
583 [] IPA_RSL[0].receive { repeat; }
584 [] T.timeout {
585 setverdict(fail, "Timeout waiting for RSL_OSMO_ETWS_CMD");
Neels Hofmeyree3b9272020-07-28 21:20:15 +0200586 mtc.stop;
Harald Welte0b5e0f92019-09-07 08:30:25 +0200587 }
588 }
589}
590
591/* Write ETWS PN to single BTS; verify it arrives on CCHAN */
592testcase TC_cbsp_emerg_write_bts_cgi_cchan_disable() runs on cbsp_test_CT {
593 var CBSP_IEs pages := {f_gen_page()};
594 var template (value) BSSMAP_FIELD_CellIdentificationList cell_list;
595 var ASP_RSL_Unitdata rx_rsl_ud;
596
597 cell_list := ts_BSSMAP_CIL_CGI({bssmap_cgi(mp_cgi_bts0)});
598 f_cbsp_init_server();
599
600 f_cbsp_write_emerg(16, 1016, cell_list);
601
602 /* first expect the PN to be enabled */
603 var template (present) octetstring tr_apdu := f_gen_etws_pn(1016, 16);
604 timer T := 5.0;
605 T.start;
606 alt {
Vadim Yanitskiy9b4e3562020-05-25 21:40:52 +0700607 [] IPA_RSL[0].receive(tr_ASP_RSL_UD(tr_RSL_OSMO_ETWS_CMD(t_RslChanNr_PCH_AGCH(0), tr_apdu))) {
Harald Welte0b5e0f92019-09-07 08:30:25 +0200608 setverdict(pass);
609 }
Vadim Yanitskiy9b4e3562020-05-25 21:40:52 +0700610 [] IPA_RSL[0].receive(tr_ASP_RSL_UD(tr_RSL_OSMO_ETWS_CMD(?,?))) {
Harald Welte0b5e0f92019-09-07 08:30:25 +0200611 setverdict(fail, "Received unexpected OSMO_ETWS_CMD");
Neels Hofmeyree3b9272020-07-28 21:20:15 +0200612 mtc.stop;
Harald Welte0b5e0f92019-09-07 08:30:25 +0200613 }
614 [] IPA_RSL[0].receive { repeat; }
615 [] T.timeout {
616 setverdict(fail, "Timeout waiting for RSL_OSMO_ETWS_CMD (enable)");
Neels Hofmeyree3b9272020-07-28 21:20:15 +0200617 mtc.stop;
Harald Welte0b5e0f92019-09-07 08:30:25 +0200618 }
619 }
620
621 /* then expect it to be disabled after the warning period (5s) */
622 T.start;
623 alt {
Vadim Yanitskiy9b4e3562020-05-25 21:40:52 +0700624 [] IPA_RSL[0].receive(tr_ASP_RSL_UD(tr_RSL_OSMO_ETWS_CMD(t_RslChanNr_PCH_AGCH(0), ''O))) {
Harald Welte0b5e0f92019-09-07 08:30:25 +0200625 setverdict(pass);
626 }
Vadim Yanitskiy9b4e3562020-05-25 21:40:52 +0700627 [] IPA_RSL[0].receive(tr_ASP_RSL_UD(tr_RSL_OSMO_ETWS_CMD(?,?))) {
Harald Welte0b5e0f92019-09-07 08:30:25 +0200628 setverdict(fail, "Received unexpected OSMO_ETWS_CMD");
Neels Hofmeyree3b9272020-07-28 21:20:15 +0200629 mtc.stop;
Harald Welte0b5e0f92019-09-07 08:30:25 +0200630 }
631 [] IPA_RSL[0].receive { repeat; }
632 [] T.timeout {
633 setverdict(fail, "Timeout waiting for RSL_OSMO_ETWS_CMD (disable)");
Neels Hofmeyree3b9272020-07-28 21:20:15 +0200634 mtc.stop;
Harald Welte0b5e0f92019-09-07 08:30:25 +0200635 }
636 }
637}
638
639
Harald Welte187f7a92019-09-05 11:15:20 +0200640
Harald Welte09538f82019-08-01 09:50:25 +0200641control {
642 execute( TC_cbsp_bsc_server() );
643 execute( TC_cbsp_bsc_client() );
644 execute( TC_cbsp_reset_bss() );
645
646 /* test various different types of Cell Identities */
647 execute( TC_cbsp_write_bss() );
648 execute( TC_cbsp_write_bts_cgi() );
649 execute( TC_cbsp_write_bts_no_cbch() );
650 execute( TC_cbsp_write_unknown_bts() );
651 execute( TC_cbsp_write_lac_ci() );
652 execute( TC_cbsp_write_ci() );
653 execute( TC_cbsp_write_lai() );
654 execute( TC_cbsp_write_lac() );
655
656 execute( TC_cbsp_write_then_replace() );
657 execute( TC_cbsp_replace_nonexist() );
658 execute( TC_cbsp_write_too_many() );
659 execute( TC_cbsp_kill_nonexist() );
660 execute( TC_cbsp_write_then_kill() );
661 execute( TC_cbsp_write_then_reset() );
Harald Welte187f7a92019-09-05 11:15:20 +0200662
663 execute( TC_cbsp_emerg_write_bts_cgi_dchan() );
Harald Welte0b5e0f92019-09-07 08:30:25 +0200664 execute( TC_cbsp_emerg_write_bts_cgi_cchan() );
665 execute( TC_cbsp_emerg_write_bts_cgi_cchan_disable() );
Harald Welte09538f82019-08-01 09:50:25 +0200666}
667
668
669}