blob: 1aa3f636366e2055480bc22a9c7cd95400376e72 [file] [log] [blame]
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +02001/* AllocTest.cpp
2 *
3 * Copyright (C) 2013 by Holger Hans Peter Freyther
4 *
5 * This program is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU General Public License
7 * as published by the Free Software Foundation; either version 2
8 * of the License, or (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
18 */
19
20#include "gprs_rlcmac.h"
21#include "gprs_debug.h"
22#include "tbf.h"
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +020023#include "bts.h"
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +020024
25#include <string.h>
26#include <stdio.h>
27
28extern "C" {
Max9f460712018-01-23 20:57:08 +010029#include "mslot_class.h"
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +020030#include <osmocom/core/application.h>
31#include <osmocom/core/msgb.h>
32#include <osmocom/core/talloc.h>
33#include <osmocom/core/utils.h>
34}
35
36/* globals used by the code */
37void *tall_pcu_ctx;
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +020038int16_t spoof_mnc = 0, spoof_mcc = 0;
39
Daniel Willmann48aa0b02014-07-16 18:54:10 +020040static gprs_rlcmac_tbf *tbf_alloc(struct gprs_rlcmac_bts *bts,
Jacob Erlbecke2e004e2015-06-18 17:16:26 +020041 GprsMs *ms, gprs_rlcmac_tbf_direction dir,
Jacob Erlbeck5879c642015-07-10 10:41:36 +020042 uint8_t use_trx,
Maxe9fe0e32017-09-28 15:56:05 +020043 uint8_t ms_class, uint8_t egprs_ms_class, bool single_slot)
Daniel Willmann48aa0b02014-07-16 18:54:10 +020044{
45 if (dir == GPRS_RLCMAC_UL_TBF)
Jacob Erlbeck86b6f052015-11-27 15:17:34 +010046 return tbf_alloc_ul_tbf(bts, ms, use_trx,
47 ms_class, egprs_ms_class, single_slot);
Daniel Willmann48aa0b02014-07-16 18:54:10 +020048 else
Jacob Erlbeck86b6f052015-11-27 15:17:34 +010049 return tbf_alloc_dl_tbf(bts, ms, use_trx,
50 ms_class, egprs_ms_class, single_slot);
Daniel Willmann48aa0b02014-07-16 18:54:10 +020051}
52
Jacob Erlbeck61205a72015-07-09 11:35:50 +020053static void check_tfi_usage(BTS *the_bts)
54{
55 int pdch_no;
Jacob Erlbeck61205a72015-07-09 11:35:50 +020056
57 struct gprs_rlcmac_tbf *tfi_usage[8][8][2][32] = {{{{NULL}}}};
Jacob Erlbecked2dbf62015-12-28 19:15:40 +010058 LListHead<gprs_rlcmac_tbf> *tbf_lists[2] = {
59 &the_bts->ul_tbfs(),
60 &the_bts->dl_tbfs()
Jacob Erlbeck61205a72015-07-09 11:35:50 +020061 };
62
Jacob Erlbecked2dbf62015-12-28 19:15:40 +010063 LListHead<gprs_rlcmac_tbf> *pos;
Jacob Erlbeck61205a72015-07-09 11:35:50 +020064 gprs_rlcmac_tbf *tbf;
Jacob Erlbeck61205a72015-07-09 11:35:50 +020065 unsigned list_idx;
66 struct gprs_rlcmac_tbf **tbf_var;
67
68 for (list_idx = 0; list_idx < ARRAY_SIZE(tbf_lists); list_idx += 1)
69 {
70
Jacob Erlbecked2dbf62015-12-28 19:15:40 +010071 llist_for_each(pos, tbf_lists[list_idx]) {
72 tbf = pos->entry();
Jacob Erlbeck61205a72015-07-09 11:35:50 +020073 for (pdch_no = 0; pdch_no < 8; pdch_no += 1) {
74 struct gprs_rlcmac_pdch *pdch = tbf->pdch[pdch_no];
75 if (pdch == NULL)
76 continue;
77
78 tbf_var = &tfi_usage
79 [tbf->trx->trx_no]
80 [pdch_no]
81 [tbf->direction]
82 [tbf->tfi()];
83
84 OSMO_ASSERT(*tbf_var == NULL);
85 if (tbf->direction == GPRS_RLCMAC_DL_TBF) {
86 OSMO_ASSERT(pdch->dl_tbf_by_tfi(
87 tbf->tfi()) == tbf);
Jacob Erlbeck61205a72015-07-09 11:35:50 +020088 OSMO_ASSERT(the_bts->dl_tbf_by_tfi(
89 tbf->tfi(),
Jacob Erlbeck3a10dbd2015-07-10 19:52:37 +020090 tbf->trx->trx_no,
91 pdch_no) == tbf);
Jacob Erlbeck61205a72015-07-09 11:35:50 +020092 } else {
93 OSMO_ASSERT(pdch->ul_tbf_by_tfi(
94 tbf->tfi()) == tbf);
Jacob Erlbeck61205a72015-07-09 11:35:50 +020095 OSMO_ASSERT(the_bts->ul_tbf_by_tfi(
96 tbf->tfi(),
Jacob Erlbeck3a10dbd2015-07-10 19:52:37 +020097 tbf->trx->trx_no,
98 pdch_no) == tbf);
Jacob Erlbeck61205a72015-07-09 11:35:50 +020099 }
100 *tbf_var = tbf;
Jacob Erlbeck47a57f62015-07-08 12:53:16 +0200101 OSMO_ASSERT(pdch->assigned_tfi(tbf->direction) &
102 (1 << tbf->tfi()));
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200103 }
104 }
105 }
106}
107
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200108static void test_alloc_a(gprs_rlcmac_tbf_direction dir,
109 uint8_t slots, const int count)
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200110{
111 int tfi;
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200112 int i;
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200113 uint8_t used_trx, tmp_trx;
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +0200114 BTS the_bts;
115 struct gprs_rlcmac_bts *bts;
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200116 struct gprs_rlcmac_tbf *tbfs[32*8+1] = { 0, };
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200117
118 printf("Testing alloc_a direction(%d)\n", dir);
119
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +0200120 bts = the_bts.bts_data();
121 bts->alloc_algorithm = alloc_algorithm_a;
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200122
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +0200123 struct gprs_rlcmac_trx *trx = &bts->trx[0];
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200124 for (i = 0; i < 8; i += 1)
125 if (slots & (1 << i))
126 trx->pdch[i].enable();
127
128 OSMO_ASSERT(count >= 0 && count <= (int)ARRAY_SIZE(tbfs));
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200129
130 /**
131 * Currently alloc_a will only allocate from the first
132 * PDCH and all possible usf's. We run out of usf's before
133 * we are out of tfi's. Observe this and make sure that at
134 * least this part is working okay.
135 */
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200136 for (i = 0; i < (int)ARRAY_SIZE(tbfs); ++i) {
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100137 tbfs[i] = tbf_alloc(bts, NULL, dir, -1, 0, 0, 0);
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200138 if (tbfs[i] == NULL)
139 break;
140
141 used_trx = tbfs[i]->trx->trx_no;
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200142 tfi = the_bts.tfi_find_free(dir, &tmp_trx, used_trx);
143 OSMO_ASSERT(tbfs[i]->tfi() != tfi);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200144 }
145
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200146 check_tfi_usage(&the_bts);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200147
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200148 OSMO_ASSERT(i == count);
149
150 for (i = 0; i < count; ++i)
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200151 if (tbfs[i])
152 tbf_free(tbfs[i]);
153
Holger Hans Peter Freytherf3f1bde2016-02-22 15:14:01 +0100154 tbfs[0] = tbf_alloc(bts, NULL, dir, -1, 0, 0, 0);
155 OSMO_ASSERT(tbfs[0]);
156 tbf_free(tbfs[0]);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200157}
158
159static void test_alloc_a()
160{
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200161 /* slots 2 - 3 */
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200162 test_alloc_a(GPRS_RLCMAC_DL_TBF, 0x0c, 32*2);
Jacob Erlbeckec478752015-06-19 16:35:38 +0200163 test_alloc_a(GPRS_RLCMAC_UL_TBF, 0x0c, 14);
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200164
165 /* slots 1 - 5 */
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200166 test_alloc_a(GPRS_RLCMAC_DL_TBF, 0x1e, 32*4);
Jacob Erlbeckec478752015-06-19 16:35:38 +0200167 test_alloc_a(GPRS_RLCMAC_UL_TBF, 0x1e, 28);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200168}
169
Max2ecf0fd2017-11-21 18:13:31 +0100170static void dump_assignment(struct gprs_rlcmac_tbf *tbf, const char *dir, bool verbose)
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100171{
Max2ecf0fd2017-11-21 18:13:31 +0100172 if (!verbose)
173 return;
174
Jacob Erlbeck1f332942015-05-04 08:21:17 +0200175 for (size_t i = 0; i < ARRAY_SIZE(tbf->pdch); ++i)
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100176 if (tbf->pdch[i])
Neels Hofmeyrd34646a2017-02-08 17:07:40 +0100177 printf("PDCH[%zu] is used for %s\n", i, dir);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100178 printf("PDCH[%d] is control_ts for %s\n", tbf->control_ts, dir);
179 printf("PDCH[%d] is first common for %s\n", tbf->first_common_ts, dir);
180}
181
Max2ecf0fd2017-11-21 18:13:31 +0100182#define ENABLE_PDCH(ts_no, enable_flag, trx) \
183 if (enable_flag) \
184 trx->pdch[ts_no].enable();
185
186static inline void enable_ts_on_bts(struct gprs_rlcmac_bts *bts,
187 bool ts0, bool ts1, bool ts2, bool ts3, bool ts4, bool ts5, bool ts6, bool ts7)
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100188{
Max2ecf0fd2017-11-21 18:13:31 +0100189 struct gprs_rlcmac_trx *trx = &bts->trx[0];
190
191 ENABLE_PDCH(0, ts0, trx);
192 ENABLE_PDCH(1, ts1, trx);
193 ENABLE_PDCH(2, ts2, trx);
194 ENABLE_PDCH(3, ts3, trx);
195 ENABLE_PDCH(4, ts4, trx);
196 ENABLE_PDCH(5, ts5, trx);
197 ENABLE_PDCH(6, ts6, trx);
198 ENABLE_PDCH(7, ts7, trx);
199}
200
201static inline bool test_alloc_b_ul_dl(bool ts0, bool ts1, bool ts2, bool ts3, bool ts4, bool ts5, bool ts6, bool ts7,
202 uint8_t ms_class, bool verbose)
203{
204 BTS the_bts;
205 struct gprs_rlcmac_bts *bts = the_bts.bts_data();
206 gprs_rlcmac_ul_tbf *ul_tbf;
207 gprs_rlcmac_dl_tbf *dl_tbf;
208
209 if (verbose)
210 printf("Testing UL then DL assignment.\n");
211
212 bts->alloc_algorithm = alloc_algorithm_b;
213
214 enable_ts_on_bts(bts, ts0, ts1, ts2, ts3, ts4, ts5, ts6, ts7);
215
216 ul_tbf = tbf_alloc_ul_tbf(bts, NULL, -1, ms_class, 0, 1);
217 if (!ul_tbf)
218 return false;
219
220 OSMO_ASSERT(ul_tbf->ms());
221 OSMO_ASSERT(ul_tbf->ms()->current_trx());
222
223 dump_assignment(ul_tbf, "UL", verbose);
224
225 /* assume final ack has not been sent */
226 dl_tbf = tbf_alloc_dl_tbf(bts, ul_tbf->ms(), ul_tbf->ms()->current_trx()->trx_no, ms_class, 0, 0);
227 if (!dl_tbf)
228 return false;
229
230 dump_assignment(dl_tbf, "DL", verbose);
231
232 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
233
234 check_tfi_usage(&the_bts);
235
236 tbf_free(dl_tbf);
237 tbf_free(ul_tbf);
238
239 return true;
240}
241
242static inline bool test_alloc_b_dl_ul(bool ts0, bool ts1, bool ts2, bool ts3, bool ts4, bool ts5, bool ts6, bool ts7,
243 uint8_t ms_class, bool verbose)
244{
245 BTS the_bts;
246 struct gprs_rlcmac_bts *bts = the_bts.bts_data();
247 gprs_rlcmac_ul_tbf *ul_tbf;
248 gprs_rlcmac_dl_tbf *dl_tbf;
249
250 if (verbose)
251 printf("Testing DL then UL assignment followed by update\n");
252
253 bts->alloc_algorithm = alloc_algorithm_b;
254
255 enable_ts_on_bts(bts, ts0, ts1, ts2, ts3, ts4, ts5, ts6, ts7);
256
257 dl_tbf = tbf_alloc_dl_tbf(bts, NULL, -1, ms_class, 0, 1);
258 if (!dl_tbf)
259 return false;
260
261 dl_tbf->update_ms(0x23, GPRS_RLCMAC_DL_TBF);
262
263 OSMO_ASSERT(dl_tbf->ms());
264 OSMO_ASSERT(dl_tbf->ms()->current_trx());
265
266 dump_assignment(dl_tbf, "DL", verbose);
267
268 ul_tbf = tbf_alloc_ul_tbf(bts, dl_tbf->ms(), dl_tbf->ms()->current_trx()->trx_no, ms_class, 0, 0);
269 if (!ul_tbf)
270 return false;
271
272 ul_tbf->update_ms(0x23, GPRS_RLCMAC_UL_TBF);
273 ul_tbf->m_contention_resolution_done = 1;
274
275 dump_assignment(ul_tbf, "UL", verbose);
276
277 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
278
279 /* now update the dl_tbf */
280 dl_tbf->update();
281 dump_assignment(dl_tbf, "DL", verbose);
282 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
283
284 check_tfi_usage(&the_bts);
285
286 tbf_free(dl_tbf);
287 tbf_free(ul_tbf);
288
289 return true;
290}
291
292static inline bool test_alloc_b_jolly(uint8_t ms_class)
293{
294 BTS the_bts;
295 struct gprs_rlcmac_bts *bts = the_bts.bts_data();
296 int tfi;
297 uint8_t trx_no;
298 gprs_rlcmac_tbf *ul_tbf, *dl_tbf;
299
300 printf("Testing jolly example\n");
301
302 bts->alloc_algorithm = alloc_algorithm_b;
303
304 enable_ts_on_bts(bts, false, true, true, true, true, false, false, false);
305
306 tfi = the_bts.tfi_find_free(GPRS_RLCMAC_UL_TBF, &trx_no, -1);
307 OSMO_ASSERT(tfi >= 0);
308 ul_tbf = tbf_alloc_ul_tbf(bts, NULL, .1, ms_class, 0, 0);
309 if (!ul_tbf)
310 return false;
311
312 OSMO_ASSERT(ul_tbf->ms());
313 OSMO_ASSERT(ul_tbf->ms()->current_trx());
314 trx_no = ul_tbf->ms()->current_trx()->trx_no;
315 dump_assignment(ul_tbf, "UL", true);
316
317 /* assume final ack has not been sent */
318 dl_tbf = tbf_alloc_dl_tbf(bts, ul_tbf->ms(), trx_no, ms_class, 0, 0);
319 if (!dl_tbf)
320 return false;
321
322 dump_assignment(dl_tbf, "DL", true);
323
324 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
325
326 check_tfi_usage(&the_bts);
327
328 tbf_free(dl_tbf);
329 tbf_free(ul_tbf);
330
331 return true;
332}
333
334static void test_alloc_b_for_ms(uint8_t ms_class)
335{
336 bool rc;
337
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100338 printf("Going to test multislot assignment MS_CLASS=%d\n", ms_class);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100339 /*
340 * PDCH is on TS 6,7,8 and we start with a UL allocation and
341 * then follow two DL allocations (once single, once normal).
342 *
343 * Uplink assigned and still available..
344 */
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100345
Max2ecf0fd2017-11-21 18:13:31 +0100346 rc = test_alloc_b_ul_dl(false, false, false, false, false, true, true, true, ms_class, true);
347 if (!rc)
348 return;
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100349
350 /**
351 * Test with the other order.. first DL and then UL
352 */
Max2ecf0fd2017-11-21 18:13:31 +0100353 rc = test_alloc_b_dl_ul(false, false, false, false, false, true, true, true, ms_class, true);
354 if (!rc)
355 return;
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100356
357 /* Andreas osmocom-pcu example */
Max2ecf0fd2017-11-21 18:13:31 +0100358 test_alloc_b_jolly(ms_class);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100359}
360
Max2ecf0fd2017-11-21 18:13:31 +0100361static void test_alloc_mass(bool ts0, bool ts1, bool ts2, bool ts3, bool ts4, bool ts5, bool ts6, bool ts7, int ms_class)
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100362{
Max2ecf0fd2017-11-21 18:13:31 +0100363 bool rc;
364
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100365 /* we can test the allocation failures differently */
366 if (!ts0 && !ts1 && !ts2 && !ts3 && !ts4 && !ts5 && !ts6 && !ts7)
367 return;
368
369 printf("Mass test: TS0(%c%c%c%c%c%c%c%c)TS7 MS_Class=%d\n",
370 ts0 ? 'O' : 'x',
371 ts1 ? 'O' : 'x',
372 ts2 ? 'O' : 'x',
373 ts3 ? 'O' : 'x',
374 ts4 ? 'O' : 'x',
375 ts5 ? 'O' : 'x',
376 ts6 ? 'O' : 'x',
377 ts7 ? 'O' : 'x', ms_class);
378 fflush(stdout);
379
Max2ecf0fd2017-11-21 18:13:31 +0100380 rc = test_alloc_b_ul_dl(ts0, ts1, ts2, ts3, ts4, ts5, ts6, ts7, ms_class, false);
381 if (!rc)
382 return;
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100383
384 /**
385 * Test with the other order.. first DL and then UL
386 */
Max2ecf0fd2017-11-21 18:13:31 +0100387 test_alloc_b_dl_ul(ts0, ts1, ts2, ts3, ts4, ts5, ts6, ts7, ms_class, false);
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100388}
389
390static void test_all_alloc_b()
391{
392 /* it is a bit crazy... */
393 for (uint8_t ts0 = 0; ts0 < 2; ++ts0)
394 for (uint8_t ts1 = 0; ts1 < 2; ++ts1)
395 for (uint8_t ts2 = 0; ts2 < 2; ++ts2)
396 for (uint8_t ts3 = 0; ts3 < 2; ++ts3)
397 for (uint8_t ts4 = 0; ts4 < 2; ++ts4)
398 for (uint8_t ts5 = 0; ts5 < 2; ++ts5)
399 for (uint8_t ts6 = 0; ts6 < 2; ++ts6)
400 for (uint8_t ts7 = 0; ts7 < 2; ++ts7)
Max9f460712018-01-23 20:57:08 +0100401 for (int ms_class = 0; ms_class < mslot_class_max(); ++ms_class)
Max2ecf0fd2017-11-21 18:13:31 +0100402 test_alloc_mass(ts0, ts1, ts2, ts3, ts4, ts5, ts6, ts7, ms_class);
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100403}
404
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100405static void test_alloc_b()
406{
Max9f460712018-01-23 20:57:08 +0100407 for (int i = 0; i < mslot_class_max(); ++i)
Max2ecf0fd2017-11-21 18:13:31 +0100408 test_alloc_b_for_ms(i);
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100409
410 test_all_alloc_b();
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100411}
412
Maxe9fe0e32017-09-28 15:56:05 +0200413typedef int (*algo_t)(struct gprs_rlcmac_bts *bts, struct GprsMs *ms, struct gprs_rlcmac_tbf *tbf, bool single,
414 int8_t use_trx);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200415
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200416static char get_dir_char(uint8_t mask, uint8_t tx, uint8_t rx, uint8_t busy)
Jacob Erlbecke5655642015-06-29 12:19:52 +0200417{
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200418 int offs = busy ? 32 : 0;
419 return (mask & tx & rx) ? 'C' + offs :
420 (mask & tx) ? 'U' + offs :
421 (mask & rx) ? 'D' + offs :
Jacob Erlbecke5655642015-06-29 12:19:52 +0200422 '.';
423}
424
425enum test_mode {
426 TEST_MODE_UL_ONLY,
427 TEST_MODE_DL_ONLY,
428 TEST_MODE_UL_AND_DL,
429 TEST_MODE_DL_AND_UL,
430 TEST_MODE_DL_AFTER_UL,
431 TEST_MODE_UL_AFTER_DL,
432};
433
Maxc59ef122017-11-27 13:21:41 +0100434static inline char *test_mode_descr(enum test_mode t)
435{
436 switch (t) {
437 case TEST_MODE_UL_ONLY: return (char*)"UL only";
438 case TEST_MODE_DL_ONLY: return (char*)"DL only";
439 case TEST_MODE_UL_AND_DL: return (char*)"UL and DL";
440 case TEST_MODE_DL_AND_UL: return (char*)"DL and UL";
441 case TEST_MODE_DL_AFTER_UL: return (char*)"DL after UL";
442 case TEST_MODE_UL_AFTER_DL: return (char*)"UL after DL";
443 default: return NULL;
444 }
445}
446
Jacob Erlbecke5655642015-06-29 12:19:52 +0200447static GprsMs *alloc_tbfs(BTS *the_bts, GprsMs *ms, unsigned ms_class,
448 enum test_mode mode)
449{
450 struct gprs_rlcmac_bts *bts;
Jacob Erlbeck5879c642015-07-10 10:41:36 +0200451 uint8_t trx_no = -1;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200452
453 bts = the_bts->bts_data();
454
455 gprs_rlcmac_tbf *tbf = NULL;
456
Jacob Erlbeck5879c642015-07-10 10:41:36 +0200457 if (ms && ms->current_trx())
458 trx_no = ms->current_trx()->trx_no;
459
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200460 GprsMs::Guard guard1(ms);
461
Jacob Erlbecke5655642015-06-29 12:19:52 +0200462 /* Allocate what is needed first */
463 switch (mode) {
464 case TEST_MODE_UL_ONLY:
465 case TEST_MODE_DL_AFTER_UL:
466 case TEST_MODE_UL_AND_DL:
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200467 if (ms && ms->ul_tbf())
468 tbf_free(ms->ul_tbf());
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100469 tbf = tbf_alloc_ul_tbf(bts, ms, trx_no, ms_class, 0, 0);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200470 if (tbf == NULL)
471 return NULL;
472 break;
473 case TEST_MODE_DL_ONLY:
474 case TEST_MODE_UL_AFTER_DL:
475 case TEST_MODE_DL_AND_UL:
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200476 if (ms && ms->dl_tbf())
477 tbf_free(ms->dl_tbf());
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100478 tbf = tbf_alloc_dl_tbf(bts, ms, trx_no, ms_class, 0, 0);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200479 if (tbf == NULL)
480 return NULL;
481 }
482
483 OSMO_ASSERT(tbf);
484 OSMO_ASSERT(tbf->ms());
485 OSMO_ASSERT(ms == NULL || ms == tbf->ms());
486 ms = tbf->ms();
487
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200488 GprsMs::Guard guard2(ms);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200489
Jacob Erlbeck14376a72015-07-07 11:31:28 +0200490 /* Continue with what is needed next */
491 switch (mode) {
492 case TEST_MODE_UL_ONLY:
493 case TEST_MODE_DL_ONLY:
494 /* We are done */
495 break;
496
497 case TEST_MODE_DL_AFTER_UL:
498 case TEST_MODE_UL_AND_DL:
499 ms = alloc_tbfs(the_bts, ms, ms_class, TEST_MODE_DL_ONLY);
500 break;
501
502 case TEST_MODE_UL_AFTER_DL:
503 case TEST_MODE_DL_AND_UL:
504 ms = alloc_tbfs(the_bts, ms, ms_class, TEST_MODE_UL_ONLY);
505 break;
506 }
507
Jacob Erlbecke5655642015-06-29 12:19:52 +0200508 /* Optionally delete the TBF */
509 switch (mode) {
510 case TEST_MODE_DL_AFTER_UL:
511 case TEST_MODE_UL_AFTER_DL:
512 tbf_free(tbf);
Jacob Erlbeck0f352a62015-07-16 18:23:33 +0200513 tbf = NULL;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200514 break;
515
516 default:
517 break;
518 }
519
Jacob Erlbeck0f352a62015-07-16 18:23:33 +0200520 if (!ms && tbf)
521 tbf_free(tbf);
522
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200523 return guard2.is_idle() ? NULL : ms;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200524}
525
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200526static unsigned alloc_many_tbfs(BTS *the_bts, unsigned min_class,
527 unsigned max_class, enum test_mode mode)
Jacob Erlbecke5655642015-06-29 12:19:52 +0200528{
Jacob Erlbecke5655642015-06-29 12:19:52 +0200529 unsigned counter;
530 unsigned ms_class = min_class;
531
Jacob Erlbecke5655642015-06-29 12:19:52 +0200532 for (counter = 0; 1; counter += 1) {
533 gprs_rlcmac_tbf *ul_tbf, *dl_tbf;
534 uint8_t ul_slots = 0;
535 uint8_t dl_slots = 0;
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200536 uint8_t busy_slots = 0;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200537 unsigned i;
538 int tfi = -1;
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200539 int tfi2;
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200540 uint8_t trx_no2;
541 struct gprs_rlcmac_trx *trx;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200542 GprsMs *ms;
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200543 enum gprs_rlcmac_tbf_direction dir;
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200544 uint32_t tlli = counter + 0xc0000000;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200545
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200546 ms = the_bts->ms_by_tlli(tlli);
547
548 ms = alloc_tbfs(the_bts, ms, ms_class, mode);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200549 if (!ms)
550 break;
551
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200552 ms->set_tlli(tlli);
553
Jacob Erlbecke5655642015-06-29 12:19:52 +0200554 ul_tbf = ms->ul_tbf();
555 dl_tbf = ms->dl_tbf();
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200556 trx = ms->current_trx();
557
558 OSMO_ASSERT(ul_tbf || dl_tbf);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200559
560 if (ul_tbf) {
561 ul_slots = 1 << ul_tbf->first_common_ts;
562 tfi = ul_tbf->tfi();
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200563 dir = GPRS_RLCMAC_UL_TBF;
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200564 } else {
Jacob Erlbecke5655642015-06-29 12:19:52 +0200565 ul_slots = 1 << dl_tbf->first_common_ts;
566 tfi = dl_tbf->tfi();
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200567 dir = GPRS_RLCMAC_DL_TBF;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200568 }
569
570 for (i = 0; dl_tbf && i < ARRAY_SIZE(dl_tbf->pdch); i += 1)
571 if (dl_tbf->pdch[i])
572 dl_slots |= 1 << i;
573
Max5b0df1f2017-09-11 10:38:59 +0200574 for (i = 0; ul_tbf && i < ARRAY_SIZE(ul_tbf->pdch); i += 1)
575 if (ul_tbf->pdch[i])
576 ul_slots |= 1 << i;
577
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200578 for (i = 0; trx && i < ARRAY_SIZE(trx->pdch); i += 1) {
579 struct gprs_rlcmac_pdch *pdch = &trx->pdch[i];
580
581 if (ul_tbf && dl_tbf)
582 continue;
583
584 if (ul_tbf &&
585 pdch->assigned_tfi(GPRS_RLCMAC_DL_TBF) != 0xffffffff)
586 continue;
587
588 if (dl_tbf &&
589 pdch->assigned_tfi(GPRS_RLCMAC_UL_TBF) != 0xffffffff)
590 continue;
591
592 busy_slots |= 1 << i;
593 }
594
Jacob Erlbecke5655642015-06-29 12:19:52 +0200595 printf(" TBF[%d] class %d reserves %c%c%c%c%c%c%c%c\n",
596 tfi, ms_class,
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200597 get_dir_char(0x01, ul_slots, dl_slots, busy_slots),
598 get_dir_char(0x02, ul_slots, dl_slots, busy_slots),
599 get_dir_char(0x04, ul_slots, dl_slots, busy_slots),
600 get_dir_char(0x08, ul_slots, dl_slots, busy_slots),
601 get_dir_char(0x10, ul_slots, dl_slots, busy_slots),
602 get_dir_char(0x20, ul_slots, dl_slots, busy_slots),
603 get_dir_char(0x40, ul_slots, dl_slots, busy_slots),
604 get_dir_char(0x80, ul_slots, dl_slots, busy_slots));
Jacob Erlbecke5655642015-06-29 12:19:52 +0200605
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200606 if (tfi >= 0) {
607 OSMO_ASSERT(ms->current_trx());
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200608 tfi2 = the_bts->tfi_find_free(dir, &trx_no2,
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200609 ms->current_trx()->trx_no);
610 OSMO_ASSERT(tfi != tfi2);
Jacob Erlbeck7b3675b2015-07-16 18:28:22 +0200611 OSMO_ASSERT(tfi2 < 0 ||
612 trx_no2 == ms->current_trx()->trx_no);
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200613 }
614
Jacob Erlbecke5655642015-06-29 12:19:52 +0200615 ms_class += 1;
616 if (ms_class > max_class)
617 ms_class = min_class;
618 }
619
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200620 return counter;
621}
622
623static void test_successive_allocation(algo_t algo, unsigned min_class,
624 unsigned max_class, enum test_mode mode,
625 unsigned expect_num, const char *text)
626{
627 BTS the_bts;
628 struct gprs_rlcmac_bts *bts;
629 struct gprs_rlcmac_trx *trx;
630 unsigned counter;
631
Maxc59ef122017-11-27 13:21:41 +0100632 printf("Going to test assignment with many TBF, algorithm %s class %u..%u (%s)\n",
633 text, min_class, max_class, test_mode_descr(mode));
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200634
635 bts = the_bts.bts_data();
636 bts->alloc_algorithm = algo;
637
638 trx = &bts->trx[0];
639 trx->pdch[3].enable();
640 trx->pdch[4].enable();
641 trx->pdch[5].enable();
642 trx->pdch[6].enable();
643 trx->pdch[7].enable();
644
645 counter = alloc_many_tbfs(&the_bts, min_class, max_class, mode);
646
Maxc59ef122017-11-27 13:21:41 +0100647 printf(" Successfully allocated %u UL TBFs, algorithm %s class %u..%u (%s)\n",
648 counter, text, min_class, max_class, test_mode_descr(mode));
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200649 if (counter != expect_num)
Maxc59ef122017-11-27 13:21:41 +0100650 fprintf(stderr, " Expected %u TBFs (got %u), algorithm %s class %u..%u (%s)\n",
651 expect_num, counter, text, min_class, max_class, test_mode_descr(mode));
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200652
Jacob Erlbeckec478752015-06-19 16:35:38 +0200653 OSMO_ASSERT(counter == expect_num);
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200654
655 check_tfi_usage(&the_bts);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200656}
657
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200658static void test_many_connections(algo_t algo, unsigned expect_num,
659 const char *text)
660{
661 BTS the_bts;
662 struct gprs_rlcmac_bts *bts;
663 struct gprs_rlcmac_trx *trx;
664 int counter1, counter2 = -1;
665 unsigned i;
666 enum test_mode mode_seq[] = {
667 TEST_MODE_DL_AFTER_UL,
668 TEST_MODE_UL_ONLY,
669 TEST_MODE_DL_AFTER_UL,
670 TEST_MODE_DL_ONLY,
671 };
672
Maxc59ef122017-11-27 13:21:41 +0100673 printf("Going to test assignment with many connections, algorithm %s\n", text);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200674
675 bts = the_bts.bts_data();
676 bts->alloc_algorithm = algo;
677
678 trx = &bts->trx[0];
679 trx->pdch[3].enable();
680 trx->pdch[4].enable();
681 trx->pdch[5].enable();
682 trx->pdch[6].enable();
683 trx->pdch[7].enable();
684
685 for (i = 0; i < ARRAY_SIZE(mode_seq); i += 1) {
Max9f460712018-01-23 20:57:08 +0100686 counter1 = alloc_many_tbfs(&the_bts, 1, mslot_class_max(), mode_seq[i]);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200687 fprintf(stderr, " Allocated %d TBFs (previously %d)\n",
688 counter1, counter2);
689
690 check_tfi_usage(&the_bts);
691
692 /* This will stop earlier due to USF shortage */
693 if (mode_seq[i] == TEST_MODE_UL_ONLY)
694 continue;
695
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200696 if (counter2 >= 0) {
697 if (counter1 < counter2)
698 fprintf(stderr, " Expected %d >= %d in %s\n",
699 counter1, counter2, text);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200700 OSMO_ASSERT(counter1 >= counter2);
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200701 }
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200702
703 counter2 = counter1;
704 }
705
706 printf(" Successfully allocated %d TBFs\n", counter1);
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200707 if (counter1 != (int)expect_num)
Maxc59ef122017-11-27 13:21:41 +0100708 fprintf(stderr, " Expected %d TBFs (got %d) for algorithm %s\n", expect_num, counter1, text);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200709
710 OSMO_ASSERT(expect_num == (unsigned)counter1);
711}
712
Maxc59ef122017-11-27 13:21:41 +0100713static inline void test_a_b_dyn(enum test_mode mode, uint8_t exp_A, uint8_t exp_B, uint8_t exp_dyn)
Jacob Erlbecke5655642015-06-29 12:19:52 +0200714{
Maxc59ef122017-11-27 13:21:41 +0100715 test_successive_allocation(alloc_algorithm_a, 1, 1, mode, exp_A, "A");
716 test_successive_allocation(alloc_algorithm_b, 10, 10, mode, exp_B, "B");
717 test_successive_allocation(alloc_algorithm_dynamic, 10, 10, mode, exp_dyn, "dynamic");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200718}
719
Maxc59ef122017-11-27 13:21:41 +0100720static void test_successive_allocations()
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200721{
Maxc59ef122017-11-27 13:21:41 +0100722 test_successive_allocation(alloc_algorithm_a, 1, 1, TEST_MODE_UL_AND_DL, 35, "A");
723 test_successive_allocation(alloc_algorithm_b, 10, 10, TEST_MODE_UL_AND_DL, 32, "B");
724 test_successive_allocation(alloc_algorithm_b, 12, 12, TEST_MODE_UL_AND_DL, 32, "B");
725
Max9f460712018-01-23 20:57:08 +0100726 test_successive_allocation(alloc_algorithm_b, 1, 12, TEST_MODE_UL_AND_DL, 32, "B");
Max01bd0cc2018-01-23 20:58:49 +0100727 test_successive_allocation(alloc_algorithm_b, 1, mslot_class_max(), TEST_MODE_UL_AND_DL, 32, "B");
Max9f460712018-01-23 20:57:08 +0100728 test_successive_allocation(alloc_algorithm_dynamic, 1, mslot_class_max(), TEST_MODE_UL_AND_DL, 35, "dynamic");
Maxc59ef122017-11-27 13:21:41 +0100729
730 test_a_b_dyn(TEST_MODE_DL_AND_UL, 35, 32, 32);
731 test_a_b_dyn(TEST_MODE_DL_AFTER_UL, 160, 32, 95);
732 test_a_b_dyn(TEST_MODE_UL_AFTER_DL, 35, 32, 35);
733 test_a_b_dyn(TEST_MODE_UL_ONLY, 35, 32, 35);
734 test_a_b_dyn(TEST_MODE_DL_ONLY, 160, 32, 101);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200735}
736
Aravind Sirsikare26ee012016-09-06 18:15:45 +0530737static void test_2_consecutive_dl_tbfs()
738{
739 BTS the_bts;
740 struct gprs_rlcmac_bts *bts;
741 struct gprs_rlcmac_trx *trx;
742 uint8_t ms_class = 11;
743 uint8_t egprs_ms_class = 11;
744 gprs_rlcmac_tbf *dl_tbf1, *dl_tbf2;
745 uint8_t numTs1 = 0, numTs2 = 0;
746
747 printf("Testing DL TS allocation for Multi UEs\n");
748
749 bts = the_bts.bts_data();
750 bts->alloc_algorithm = alloc_algorithm_b;
751
752 trx = &bts->trx[0];
753 trx->pdch[4].enable();
754 trx->pdch[5].enable();
755 trx->pdch[6].enable();
756 trx->pdch[7].enable();
757
758 dl_tbf1 = tbf_alloc_dl_tbf(bts, NULL, 0, ms_class, egprs_ms_class, 0);
759 OSMO_ASSERT(dl_tbf1);
760
761 for (int i = 0; i < 8; i++) {
762 if (dl_tbf1->pdch[i])
763 numTs1++;
764 }
765 OSMO_ASSERT(numTs1 == 4);
766 printf("TBF1: numTs(%d)\n", numTs1);
767
768 dl_tbf2 = tbf_alloc_dl_tbf(bts, NULL, 0, ms_class, egprs_ms_class, 0);
769 OSMO_ASSERT(dl_tbf2);
770
771 for (int i = 0; i < 8; i++) {
772 if (dl_tbf2->pdch[i])
773 numTs2++;
774 }
775
776 /*
777 * TODO: currently 2nd DL TBF gets 3 TS
778 * This behaviour will be fixed in subsequent patch
779 */
780 printf("TBF2: numTs(%d)\n", numTs2);
781 OSMO_ASSERT(numTs2 == 3);
782
783 tbf_free(dl_tbf1);
784 tbf_free(dl_tbf2);
785}
786
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200787int main(int argc, char **argv)
788{
789 tall_pcu_ctx = talloc_named_const(NULL, 1, "moiji-mobile AllocTest context");
790 if (!tall_pcu_ctx)
791 abort();
792
Neels Hofmeyr78ce5912017-02-08 17:07:31 +0100793 msgb_talloc_ctx_init(tall_pcu_ctx, 0);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200794 osmo_init_logging(&gprs_log_info);
795 log_set_use_color(osmo_stderr_target, 0);
796 log_set_print_filename(osmo_stderr_target, 0);
Jacob Erlbeck9ec49e22015-06-29 13:00:20 +0200797 if (getenv("LOGL_DEBUG"))
798 log_set_log_level(osmo_stderr_target, LOGL_DEBUG);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200799
800 test_alloc_a();
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100801 test_alloc_b();
Maxc59ef122017-11-27 13:21:41 +0100802 test_successive_allocations();
803 test_many_connections(alloc_algorithm_a, 160, "A");
Max01bd0cc2018-01-23 20:58:49 +0100804 test_many_connections(alloc_algorithm_b, 32, "B");
Maxc59ef122017-11-27 13:21:41 +0100805 test_many_connections(alloc_algorithm_dynamic, 160, "dynamic");
Aravind Sirsikare26ee012016-09-06 18:15:45 +0530806 test_2_consecutive_dl_tbfs();
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200807 return EXIT_SUCCESS;
808}
809
810/*
811 * stubs that should not be reached
812 */
813extern "C" {
814void l1if_pdch_req() { abort(); }
815void l1if_connect_pdch() { abort(); }
816void l1if_close_pdch() { abort(); }
817void l1if_open_pdch() { abort(); }
818}