blob: a88f4776c0de32a68663311ab7c0e9d47003670b [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" {
29#include <osmocom/core/application.h>
30#include <osmocom/core/msgb.h>
31#include <osmocom/core/talloc.h>
32#include <osmocom/core/utils.h>
33}
34
35/* globals used by the code */
36void *tall_pcu_ctx;
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +020037int16_t spoof_mnc = 0, spoof_mcc = 0;
38
Daniel Willmann48aa0b02014-07-16 18:54:10 +020039static gprs_rlcmac_tbf *tbf_alloc(struct gprs_rlcmac_bts *bts,
Jacob Erlbecke2e004e2015-06-18 17:16:26 +020040 GprsMs *ms, gprs_rlcmac_tbf_direction dir,
Jacob Erlbeck5879c642015-07-10 10:41:36 +020041 uint8_t use_trx,
Jacob Erlbeck86b6f052015-11-27 15:17:34 +010042 uint8_t ms_class, uint8_t egprs_ms_class, uint8_t single_slot)
Daniel Willmann48aa0b02014-07-16 18:54:10 +020043{
44 if (dir == GPRS_RLCMAC_UL_TBF)
Jacob Erlbeck86b6f052015-11-27 15:17:34 +010045 return tbf_alloc_ul_tbf(bts, ms, use_trx,
46 ms_class, egprs_ms_class, single_slot);
Daniel Willmann48aa0b02014-07-16 18:54:10 +020047 else
Jacob Erlbeck86b6f052015-11-27 15:17:34 +010048 return tbf_alloc_dl_tbf(bts, ms, use_trx,
49 ms_class, egprs_ms_class, single_slot);
Daniel Willmann48aa0b02014-07-16 18:54:10 +020050}
51
Jacob Erlbeck61205a72015-07-09 11:35:50 +020052static void check_tfi_usage(BTS *the_bts)
53{
54 int pdch_no;
Jacob Erlbeck61205a72015-07-09 11:35:50 +020055
56 struct gprs_rlcmac_tbf *tfi_usage[8][8][2][32] = {{{{NULL}}}};
Jacob Erlbecked2dbf62015-12-28 19:15:40 +010057 LListHead<gprs_rlcmac_tbf> *tbf_lists[2] = {
58 &the_bts->ul_tbfs(),
59 &the_bts->dl_tbfs()
Jacob Erlbeck61205a72015-07-09 11:35:50 +020060 };
61
Jacob Erlbecked2dbf62015-12-28 19:15:40 +010062 LListHead<gprs_rlcmac_tbf> *pos;
Jacob Erlbeck61205a72015-07-09 11:35:50 +020063 gprs_rlcmac_tbf *tbf;
Jacob Erlbeck61205a72015-07-09 11:35:50 +020064 unsigned list_idx;
65 struct gprs_rlcmac_tbf **tbf_var;
66
67 for (list_idx = 0; list_idx < ARRAY_SIZE(tbf_lists); list_idx += 1)
68 {
69
Jacob Erlbecked2dbf62015-12-28 19:15:40 +010070 llist_for_each(pos, tbf_lists[list_idx]) {
71 tbf = pos->entry();
Jacob Erlbeck61205a72015-07-09 11:35:50 +020072 for (pdch_no = 0; pdch_no < 8; pdch_no += 1) {
73 struct gprs_rlcmac_pdch *pdch = tbf->pdch[pdch_no];
74 if (pdch == NULL)
75 continue;
76
77 tbf_var = &tfi_usage
78 [tbf->trx->trx_no]
79 [pdch_no]
80 [tbf->direction]
81 [tbf->tfi()];
82
83 OSMO_ASSERT(*tbf_var == NULL);
84 if (tbf->direction == GPRS_RLCMAC_DL_TBF) {
85 OSMO_ASSERT(pdch->dl_tbf_by_tfi(
86 tbf->tfi()) == tbf);
Jacob Erlbeck61205a72015-07-09 11:35:50 +020087 OSMO_ASSERT(the_bts->dl_tbf_by_tfi(
88 tbf->tfi(),
Jacob Erlbeck3a10dbd2015-07-10 19:52:37 +020089 tbf->trx->trx_no,
90 pdch_no) == tbf);
Jacob Erlbeck61205a72015-07-09 11:35:50 +020091 } else {
92 OSMO_ASSERT(pdch->ul_tbf_by_tfi(
93 tbf->tfi()) == tbf);
Jacob Erlbeck61205a72015-07-09 11:35:50 +020094 OSMO_ASSERT(the_bts->ul_tbf_by_tfi(
95 tbf->tfi(),
Jacob Erlbeck3a10dbd2015-07-10 19:52:37 +020096 tbf->trx->trx_no,
97 pdch_no) == tbf);
Jacob Erlbeck61205a72015-07-09 11:35:50 +020098 }
99 *tbf_var = tbf;
Jacob Erlbeck47a57f62015-07-08 12:53:16 +0200100 OSMO_ASSERT(pdch->assigned_tfi(tbf->direction) &
101 (1 << tbf->tfi()));
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200102 }
103 }
104 }
105}
106
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200107static void test_alloc_a(gprs_rlcmac_tbf_direction dir,
108 uint8_t slots, const int count)
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200109{
110 int tfi;
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200111 int i;
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200112 uint8_t used_trx, tmp_trx;
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +0200113 BTS the_bts;
114 struct gprs_rlcmac_bts *bts;
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200115 struct gprs_rlcmac_tbf *tbfs[32*8+1] = { 0, };
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200116
117 printf("Testing alloc_a direction(%d)\n", dir);
118
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +0200119 bts = the_bts.bts_data();
120 bts->alloc_algorithm = alloc_algorithm_a;
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200121
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +0200122 struct gprs_rlcmac_trx *trx = &bts->trx[0];
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200123 for (i = 0; i < 8; i += 1)
124 if (slots & (1 << i))
125 trx->pdch[i].enable();
126
127 OSMO_ASSERT(count >= 0 && count <= (int)ARRAY_SIZE(tbfs));
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200128
129 /**
130 * Currently alloc_a will only allocate from the first
131 * PDCH and all possible usf's. We run out of usf's before
132 * we are out of tfi's. Observe this and make sure that at
133 * least this part is working okay.
134 */
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200135 for (i = 0; i < (int)ARRAY_SIZE(tbfs); ++i) {
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100136 tbfs[i] = tbf_alloc(bts, NULL, dir, -1, 0, 0, 0);
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200137 if (tbfs[i] == NULL)
138 break;
139
140 used_trx = tbfs[i]->trx->trx_no;
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200141 tfi = the_bts.tfi_find_free(dir, &tmp_trx, used_trx);
142 OSMO_ASSERT(tbfs[i]->tfi() != tfi);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200143 }
144
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200145 check_tfi_usage(&the_bts);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200146
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200147 OSMO_ASSERT(i == count);
148
149 for (i = 0; i < count; ++i)
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200150 if (tbfs[i])
151 tbf_free(tbfs[i]);
152
Holger Hans Peter Freytherf3f1bde2016-02-22 15:14:01 +0100153 tbfs[0] = tbf_alloc(bts, NULL, dir, -1, 0, 0, 0);
154 OSMO_ASSERT(tbfs[0]);
155 tbf_free(tbfs[0]);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200156}
157
158static void test_alloc_a()
159{
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200160 /* slots 2 - 3 */
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200161 test_alloc_a(GPRS_RLCMAC_DL_TBF, 0x0c, 32*2);
Jacob Erlbeckec478752015-06-19 16:35:38 +0200162 test_alloc_a(GPRS_RLCMAC_UL_TBF, 0x0c, 14);
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200163
164 /* slots 1 - 5 */
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200165 test_alloc_a(GPRS_RLCMAC_DL_TBF, 0x1e, 32*4);
Jacob Erlbeckec478752015-06-19 16:35:38 +0200166 test_alloc_a(GPRS_RLCMAC_UL_TBF, 0x1e, 28);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200167}
168
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100169static void dump_assignment(struct gprs_rlcmac_tbf *tbf, const char *dir)
170{
Jacob Erlbeck1f332942015-05-04 08:21:17 +0200171 for (size_t i = 0; i < ARRAY_SIZE(tbf->pdch); ++i)
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100172 if (tbf->pdch[i])
Neels Hofmeyrd34646a2017-02-08 17:07:40 +0100173 printf("PDCH[%zu] is used for %s\n", i, dir);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100174 printf("PDCH[%d] is control_ts for %s\n", tbf->control_ts, dir);
175 printf("PDCH[%d] is first common for %s\n", tbf->first_common_ts, dir);
176}
177
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100178static void test_alloc_b(int ms_class)
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100179{
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100180 printf("Going to test multislot assignment MS_CLASS=%d\n", ms_class);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100181 /*
182 * PDCH is on TS 6,7,8 and we start with a UL allocation and
183 * then follow two DL allocations (once single, once normal).
184 *
185 * Uplink assigned and still available..
186 */
187 {
188 BTS the_bts;
189 struct gprs_rlcmac_bts *bts;
190 struct gprs_rlcmac_trx *trx;
Jacob Erlbeck1f332942015-05-04 08:21:17 +0200191 uint8_t trx_no;
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100192
193 gprs_rlcmac_tbf *ul_tbf, *dl_tbf;
194
195 printf("Testing UL then DL assignment.\n");
196
197 bts = the_bts.bts_data();
198 bts->alloc_algorithm = alloc_algorithm_b;
199
200 trx = &bts->trx[0];
201 trx->pdch[5].enable();
202 trx->pdch[6].enable();
203 trx->pdch[7].enable();
204
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100205 ul_tbf = tbf_alloc_ul_tbf(bts, NULL, -1, ms_class, 0, 1);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100206 OSMO_ASSERT(ul_tbf);
Jacob Erlbeck5879c642015-07-10 10:41:36 +0200207 OSMO_ASSERT(ul_tbf->ms());
208 OSMO_ASSERT(ul_tbf->ms()->current_trx());
209 trx_no = ul_tbf->ms()->current_trx()->trx_no;
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100210 dump_assignment(ul_tbf, "UL");
211
212 /* assume final ack has not been sent */
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100213 dl_tbf = tbf_alloc_dl_tbf(bts, ul_tbf->ms(), trx_no, ms_class, 0, 0);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100214 OSMO_ASSERT(dl_tbf);
215 dump_assignment(dl_tbf, "DL");
216
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100217 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
218
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200219 check_tfi_usage(&the_bts);
220
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100221 tbf_free(dl_tbf);
222 tbf_free(ul_tbf);
223 }
224
225 /**
226 * Test with the other order.. first DL and then UL
227 */
228 {
229 BTS the_bts;
230 struct gprs_rlcmac_bts *bts;
231 struct gprs_rlcmac_trx *trx;
Jacob Erlbeck1f332942015-05-04 08:21:17 +0200232 uint8_t trx_no;
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100233
Daniel Willmann351a5732014-08-07 12:57:44 +0200234 gprs_rlcmac_ul_tbf *ul_tbf;
235 gprs_rlcmac_dl_tbf *dl_tbf;
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100236
237 printf("Testing DL then UL assignment followed by update\n");
238
239 bts = the_bts.bts_data();
240 bts->alloc_algorithm = alloc_algorithm_b;
241
242 trx = &bts->trx[0];
243 trx->pdch[5].enable();
244 trx->pdch[6].enable();
245 trx->pdch[7].enable();
246
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100247 dl_tbf = tbf_alloc_dl_tbf(bts, NULL, -1, ms_class, 0, 1);
Jacob Erlbeck767193e2015-05-20 12:06:46 +0200248 dl_tbf->update_ms(0x23, GPRS_RLCMAC_DL_TBF);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100249 OSMO_ASSERT(dl_tbf);
Jacob Erlbeck5879c642015-07-10 10:41:36 +0200250 OSMO_ASSERT(dl_tbf->ms());
251 OSMO_ASSERT(dl_tbf->ms()->current_trx());
252 trx_no = dl_tbf->ms()->current_trx()->trx_no;
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100253 dump_assignment(dl_tbf, "DL");
254
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100255 ul_tbf = tbf_alloc_ul_tbf(bts, dl_tbf->ms(), trx_no, ms_class, 0, 0);
Jacob Erlbeck767193e2015-05-20 12:06:46 +0200256 ul_tbf->update_ms(0x23, GPRS_RLCMAC_UL_TBF);
Daniel Willmann7e994e32014-08-07 15:49:21 +0200257 ul_tbf->m_contention_resolution_done = 1;
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100258 OSMO_ASSERT(ul_tbf);
259 dump_assignment(ul_tbf, "UL");
260
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100261 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
262
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100263 /* now update the dl_tbf */
264 dl_tbf->update();
265 dump_assignment(dl_tbf, "DL");
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100266 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100267
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200268 check_tfi_usage(&the_bts);
269
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100270 tbf_free(dl_tbf);
271 tbf_free(ul_tbf);
272 }
273
274 /* Andreas osmocom-pcu example */
275 {
276 BTS the_bts;
277 struct gprs_rlcmac_bts *bts;
278 struct gprs_rlcmac_trx *trx;
279 int tfi;
Jacob Erlbeck1f332942015-05-04 08:21:17 +0200280 uint8_t trx_no;
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100281
282 gprs_rlcmac_tbf *ul_tbf, *dl_tbf;
283
284 printf("Testing jolly example\n");
285
286 bts = the_bts.bts_data();
287 bts->alloc_algorithm = alloc_algorithm_b;
288
289 trx = &bts->trx[0];
290 trx->pdch[1].enable();
291 trx->pdch[2].enable();
292 trx->pdch[3].enable();
293 trx->pdch[4].enable();
294
295 tfi = the_bts.tfi_find_free(GPRS_RLCMAC_UL_TBF, &trx_no, -1);
296 OSMO_ASSERT(tfi >= 0);
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100297 ul_tbf = tbf_alloc_ul_tbf(bts, NULL, .1, ms_class, 0, 0);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100298 OSMO_ASSERT(ul_tbf);
Jacob Erlbeck5879c642015-07-10 10:41:36 +0200299 OSMO_ASSERT(ul_tbf->ms());
300 OSMO_ASSERT(ul_tbf->ms()->current_trx());
301 trx_no = ul_tbf->ms()->current_trx()->trx_no;
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100302 dump_assignment(ul_tbf, "UL");
303
304 /* assume final ack has not been sent */
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100305 dl_tbf = tbf_alloc_dl_tbf(bts, ul_tbf->ms(), trx_no, ms_class, 0, 0);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100306 OSMO_ASSERT(dl_tbf);
307 dump_assignment(dl_tbf, "DL");
308
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100309 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
310
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200311 check_tfi_usage(&the_bts);
312
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100313 tbf_free(dl_tbf);
314 tbf_free(ul_tbf);
315 }
316}
317
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100318#define ENABLE_PDCH(ts_no, enable_flag, trx) \
319 if (enable_flag) \
320 trx->pdch[ts_no].enable();
321
322static void test_alloc_b(bool ts0, bool ts1, bool ts2, bool ts3, bool ts4, bool ts5, bool ts6, bool ts7, int ms_class)
323{
324 /* we can test the allocation failures differently */
325 if (!ts0 && !ts1 && !ts2 && !ts3 && !ts4 && !ts5 && !ts6 && !ts7)
326 return;
327
328 printf("Mass test: TS0(%c%c%c%c%c%c%c%c)TS7 MS_Class=%d\n",
329 ts0 ? 'O' : 'x',
330 ts1 ? 'O' : 'x',
331 ts2 ? 'O' : 'x',
332 ts3 ? 'O' : 'x',
333 ts4 ? 'O' : 'x',
334 ts5 ? 'O' : 'x',
335 ts6 ? 'O' : 'x',
336 ts7 ? 'O' : 'x', ms_class);
337 fflush(stdout);
338
339 {
340 BTS the_bts;
341 struct gprs_rlcmac_bts *bts;
342 struct gprs_rlcmac_trx *trx;
Jacob Erlbeck1f332942015-05-04 08:21:17 +0200343 uint8_t trx_no;
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100344
345 gprs_rlcmac_tbf *ul_tbf, *dl_tbf;
346
347 bts = the_bts.bts_data();
348 bts->alloc_algorithm = alloc_algorithm_b;
349
350 trx = &bts->trx[0];
351 ENABLE_PDCH(0, ts0, trx);
352 ENABLE_PDCH(1, ts1, trx);
353 ENABLE_PDCH(2, ts2, trx);
354 ENABLE_PDCH(3, ts3, trx);
355 ENABLE_PDCH(4, ts4, trx);
356 ENABLE_PDCH(5, ts5, trx);
357 ENABLE_PDCH(6, ts6, trx);
358 ENABLE_PDCH(7, ts7, trx);
359
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100360 ul_tbf = tbf_alloc_ul_tbf(bts, NULL, -1, ms_class, 0, 1);
Jacob Erlbeck5879c642015-07-10 10:41:36 +0200361 OSMO_ASSERT(ul_tbf->ms());
362 OSMO_ASSERT(ul_tbf->ms()->current_trx());
363 trx_no = ul_tbf->ms()->current_trx()->trx_no;
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100364 OSMO_ASSERT(ul_tbf);
365
366 /* assume final ack has not been sent */
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100367 dl_tbf = tbf_alloc_dl_tbf(bts, ul_tbf->ms(), trx_no, ms_class, 0, 0);
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100368 OSMO_ASSERT(dl_tbf);
369
370 /* verify that both are on the same ts */
371 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
372
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200373 check_tfi_usage(&the_bts);
374
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100375 tbf_free(dl_tbf);
376 tbf_free(ul_tbf);
377 }
378
379 /**
380 * Test with the other order.. first DL and then UL
381 */
382 {
383 BTS the_bts;
384 struct gprs_rlcmac_bts *bts;
385 struct gprs_rlcmac_trx *trx;
Jacob Erlbeck1f332942015-05-04 08:21:17 +0200386 uint8_t trx_no;
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100387
Daniel Willmann351a5732014-08-07 12:57:44 +0200388 gprs_rlcmac_ul_tbf *ul_tbf;
389 gprs_rlcmac_dl_tbf *dl_tbf;
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100390
391 bts = the_bts.bts_data();
392 bts->alloc_algorithm = alloc_algorithm_b;
393
394 trx = &bts->trx[0];
395 ENABLE_PDCH(0, ts0, trx);
396 ENABLE_PDCH(1, ts1, trx);
397 ENABLE_PDCH(2, ts2, trx);
398 ENABLE_PDCH(3, ts3, trx);
399 ENABLE_PDCH(4, ts4, trx);
400 ENABLE_PDCH(5, ts5, trx);
401 ENABLE_PDCH(6, ts6, trx);
402 ENABLE_PDCH(7, ts7, trx);
403
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100404 dl_tbf = tbf_alloc_dl_tbf(bts, NULL, -1, ms_class, 0, 1);
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100405 OSMO_ASSERT(dl_tbf);
Jacob Erlbeck5879c642015-07-10 10:41:36 +0200406 OSMO_ASSERT(dl_tbf->ms());
407 OSMO_ASSERT(dl_tbf->ms()->current_trx());
408 trx_no = dl_tbf->ms()->current_trx()->trx_no;
Jacob Erlbeck767193e2015-05-20 12:06:46 +0200409 dl_tbf->update_ms(0x23, GPRS_RLCMAC_DL_TBF);
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100410
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100411 ul_tbf = tbf_alloc_ul_tbf(bts, dl_tbf->ms(), trx_no, ms_class, 0, 0);
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100412 OSMO_ASSERT(ul_tbf);
Jacob Erlbeck767193e2015-05-20 12:06:46 +0200413 ul_tbf->update_ms(0x23, GPRS_RLCMAC_UL_TBF);
Daniel Willmann7e994e32014-08-07 15:49:21 +0200414 ul_tbf->m_contention_resolution_done = 1;
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100415
416 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
417
418 /* now update the dl_tbf */
419 dl_tbf->update();
420 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
421
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200422 OSMO_ASSERT(ul_tbf->ms_class() == ms_class);
423 OSMO_ASSERT(dl_tbf->ms_class() == ms_class);
424
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200425 check_tfi_usage(&the_bts);
426
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100427 tbf_free(dl_tbf);
428 tbf_free(ul_tbf);
429 }
430}
431
432static void test_all_alloc_b()
433{
434 /* it is a bit crazy... */
435 for (uint8_t ts0 = 0; ts0 < 2; ++ts0)
436 for (uint8_t ts1 = 0; ts1 < 2; ++ts1)
437 for (uint8_t ts2 = 0; ts2 < 2; ++ts2)
438 for (uint8_t ts3 = 0; ts3 < 2; ++ts3)
439 for (uint8_t ts4 = 0; ts4 < 2; ++ts4)
440 for (uint8_t ts5 = 0; ts5 < 2; ++ts5)
441 for (uint8_t ts6 = 0; ts6 < 2; ++ts6)
442 for (uint8_t ts7 = 0; ts7 < 2; ++ts7)
443 for (int ms_class = 0; ms_class < 30; ++ms_class)
444 test_alloc_b(ts0, ts1, ts2, ts3, ts4, ts5, ts6, ts7, ms_class);
445}
446
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100447static void test_alloc_b()
448{
449 for (int i = 0; i < 30; ++i)
450 test_alloc_b(i);
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100451
452 test_all_alloc_b();
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100453}
454
Jacob Erlbecke5655642015-06-29 12:19:52 +0200455typedef int (*algo_t)(struct gprs_rlcmac_bts *bts,
456 struct GprsMs *ms,
Jacob Erlbeck5879c642015-07-10 10:41:36 +0200457 struct gprs_rlcmac_tbf *tbf, uint32_t cust, uint8_t single,
458 int use_trx);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200459
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200460static char get_dir_char(uint8_t mask, uint8_t tx, uint8_t rx, uint8_t busy)
Jacob Erlbecke5655642015-06-29 12:19:52 +0200461{
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200462 int offs = busy ? 32 : 0;
463 return (mask & tx & rx) ? 'C' + offs :
464 (mask & tx) ? 'U' + offs :
465 (mask & rx) ? 'D' + offs :
Jacob Erlbecke5655642015-06-29 12:19:52 +0200466 '.';
467}
468
469enum test_mode {
470 TEST_MODE_UL_ONLY,
471 TEST_MODE_DL_ONLY,
472 TEST_MODE_UL_AND_DL,
473 TEST_MODE_DL_AND_UL,
474 TEST_MODE_DL_AFTER_UL,
475 TEST_MODE_UL_AFTER_DL,
476};
477
478static GprsMs *alloc_tbfs(BTS *the_bts, GprsMs *ms, unsigned ms_class,
479 enum test_mode mode)
480{
481 struct gprs_rlcmac_bts *bts;
Jacob Erlbeck5879c642015-07-10 10:41:36 +0200482 uint8_t trx_no = -1;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200483
484 bts = the_bts->bts_data();
485
486 gprs_rlcmac_tbf *tbf = NULL;
487
Jacob Erlbeck5879c642015-07-10 10:41:36 +0200488 if (ms && ms->current_trx())
489 trx_no = ms->current_trx()->trx_no;
490
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200491 GprsMs::Guard guard1(ms);
492
Jacob Erlbecke5655642015-06-29 12:19:52 +0200493 /* Allocate what is needed first */
494 switch (mode) {
495 case TEST_MODE_UL_ONLY:
496 case TEST_MODE_DL_AFTER_UL:
497 case TEST_MODE_UL_AND_DL:
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200498 if (ms && ms->ul_tbf())
499 tbf_free(ms->ul_tbf());
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100500 tbf = tbf_alloc_ul_tbf(bts, ms, trx_no, ms_class, 0, 0);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200501 if (tbf == NULL)
502 return NULL;
503 break;
504 case TEST_MODE_DL_ONLY:
505 case TEST_MODE_UL_AFTER_DL:
506 case TEST_MODE_DL_AND_UL:
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200507 if (ms && ms->dl_tbf())
508 tbf_free(ms->dl_tbf());
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100509 tbf = tbf_alloc_dl_tbf(bts, ms, trx_no, ms_class, 0, 0);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200510 if (tbf == NULL)
511 return NULL;
512 }
513
514 OSMO_ASSERT(tbf);
515 OSMO_ASSERT(tbf->ms());
516 OSMO_ASSERT(ms == NULL || ms == tbf->ms());
517 ms = tbf->ms();
518
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200519 GprsMs::Guard guard2(ms);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200520
Jacob Erlbeck14376a72015-07-07 11:31:28 +0200521 /* Continue with what is needed next */
522 switch (mode) {
523 case TEST_MODE_UL_ONLY:
524 case TEST_MODE_DL_ONLY:
525 /* We are done */
526 break;
527
528 case TEST_MODE_DL_AFTER_UL:
529 case TEST_MODE_UL_AND_DL:
530 ms = alloc_tbfs(the_bts, ms, ms_class, TEST_MODE_DL_ONLY);
531 break;
532
533 case TEST_MODE_UL_AFTER_DL:
534 case TEST_MODE_DL_AND_UL:
535 ms = alloc_tbfs(the_bts, ms, ms_class, TEST_MODE_UL_ONLY);
536 break;
537 }
538
Jacob Erlbecke5655642015-06-29 12:19:52 +0200539 /* Optionally delete the TBF */
540 switch (mode) {
541 case TEST_MODE_DL_AFTER_UL:
542 case TEST_MODE_UL_AFTER_DL:
543 tbf_free(tbf);
Jacob Erlbeck0f352a62015-07-16 18:23:33 +0200544 tbf = NULL;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200545 break;
546
547 default:
548 break;
549 }
550
Jacob Erlbeck0f352a62015-07-16 18:23:33 +0200551 if (!ms && tbf)
552 tbf_free(tbf);
553
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200554 return guard2.is_idle() ? NULL : ms;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200555}
556
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200557static unsigned alloc_many_tbfs(BTS *the_bts, unsigned min_class,
558 unsigned max_class, enum test_mode mode)
Jacob Erlbecke5655642015-06-29 12:19:52 +0200559{
Jacob Erlbecke5655642015-06-29 12:19:52 +0200560 unsigned counter;
561 unsigned ms_class = min_class;
562
Jacob Erlbecke5655642015-06-29 12:19:52 +0200563 for (counter = 0; 1; counter += 1) {
564 gprs_rlcmac_tbf *ul_tbf, *dl_tbf;
565 uint8_t ul_slots = 0;
566 uint8_t dl_slots = 0;
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200567 uint8_t busy_slots = 0;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200568 unsigned i;
569 int tfi = -1;
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200570 int tfi2;
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200571 uint8_t trx_no2;
572 struct gprs_rlcmac_trx *trx;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200573 GprsMs *ms;
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200574 enum gprs_rlcmac_tbf_direction dir;
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200575 uint32_t tlli = counter + 0xc0000000;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200576
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200577 ms = the_bts->ms_by_tlli(tlli);
578
579 ms = alloc_tbfs(the_bts, ms, ms_class, mode);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200580 if (!ms)
581 break;
582
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200583 ms->set_tlli(tlli);
584
Jacob Erlbecke5655642015-06-29 12:19:52 +0200585 ul_tbf = ms->ul_tbf();
586 dl_tbf = ms->dl_tbf();
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200587 trx = ms->current_trx();
588
589 OSMO_ASSERT(ul_tbf || dl_tbf);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200590
591 if (ul_tbf) {
592 ul_slots = 1 << ul_tbf->first_common_ts;
593 tfi = ul_tbf->tfi();
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200594 dir = GPRS_RLCMAC_UL_TBF;
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200595 } else {
Jacob Erlbecke5655642015-06-29 12:19:52 +0200596 ul_slots = 1 << dl_tbf->first_common_ts;
597 tfi = dl_tbf->tfi();
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200598 dir = GPRS_RLCMAC_DL_TBF;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200599 }
600
601 for (i = 0; dl_tbf && i < ARRAY_SIZE(dl_tbf->pdch); i += 1)
602 if (dl_tbf->pdch[i])
603 dl_slots |= 1 << i;
604
Max5b0df1f2017-09-11 10:38:59 +0200605 for (i = 0; ul_tbf && i < ARRAY_SIZE(ul_tbf->pdch); i += 1)
606 if (ul_tbf->pdch[i])
607 ul_slots |= 1 << i;
608
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200609 for (i = 0; trx && i < ARRAY_SIZE(trx->pdch); i += 1) {
610 struct gprs_rlcmac_pdch *pdch = &trx->pdch[i];
611
612 if (ul_tbf && dl_tbf)
613 continue;
614
615 if (ul_tbf &&
616 pdch->assigned_tfi(GPRS_RLCMAC_DL_TBF) != 0xffffffff)
617 continue;
618
619 if (dl_tbf &&
620 pdch->assigned_tfi(GPRS_RLCMAC_UL_TBF) != 0xffffffff)
621 continue;
622
623 busy_slots |= 1 << i;
624 }
625
Jacob Erlbecke5655642015-06-29 12:19:52 +0200626 printf(" TBF[%d] class %d reserves %c%c%c%c%c%c%c%c\n",
627 tfi, ms_class,
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200628 get_dir_char(0x01, ul_slots, dl_slots, busy_slots),
629 get_dir_char(0x02, ul_slots, dl_slots, busy_slots),
630 get_dir_char(0x04, ul_slots, dl_slots, busy_slots),
631 get_dir_char(0x08, ul_slots, dl_slots, busy_slots),
632 get_dir_char(0x10, ul_slots, dl_slots, busy_slots),
633 get_dir_char(0x20, ul_slots, dl_slots, busy_slots),
634 get_dir_char(0x40, ul_slots, dl_slots, busy_slots),
635 get_dir_char(0x80, ul_slots, dl_slots, busy_slots));
Jacob Erlbecke5655642015-06-29 12:19:52 +0200636
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200637 if (tfi >= 0) {
638 OSMO_ASSERT(ms->current_trx());
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200639 tfi2 = the_bts->tfi_find_free(dir, &trx_no2,
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200640 ms->current_trx()->trx_no);
641 OSMO_ASSERT(tfi != tfi2);
Jacob Erlbeck7b3675b2015-07-16 18:28:22 +0200642 OSMO_ASSERT(tfi2 < 0 ||
643 trx_no2 == ms->current_trx()->trx_no);
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200644 }
645
Jacob Erlbecke5655642015-06-29 12:19:52 +0200646 ms_class += 1;
647 if (ms_class > max_class)
648 ms_class = min_class;
649 }
650
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200651 return counter;
652}
653
654static void test_successive_allocation(algo_t algo, unsigned min_class,
655 unsigned max_class, enum test_mode mode,
656 unsigned expect_num, const char *text)
657{
658 BTS the_bts;
659 struct gprs_rlcmac_bts *bts;
660 struct gprs_rlcmac_trx *trx;
661 unsigned counter;
662
663 printf("Going to test assignment with many TBF, %s\n", text);
664
665 bts = the_bts.bts_data();
666 bts->alloc_algorithm = algo;
667
668 trx = &bts->trx[0];
669 trx->pdch[3].enable();
670 trx->pdch[4].enable();
671 trx->pdch[5].enable();
672 trx->pdch[6].enable();
673 trx->pdch[7].enable();
674
675 counter = alloc_many_tbfs(&the_bts, min_class, max_class, mode);
676
Jacob Erlbecke5655642015-06-29 12:19:52 +0200677 printf(" Successfully allocated %d UL TBFs\n", counter);
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200678 if (counter != expect_num)
679 fprintf(stderr, " Expected %d TBFs for %s\n", expect_num, text);
680
Jacob Erlbeckec478752015-06-19 16:35:38 +0200681 OSMO_ASSERT(counter == expect_num);
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200682
683 check_tfi_usage(&the_bts);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200684}
685
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200686static void test_many_connections(algo_t algo, unsigned expect_num,
687 const char *text)
688{
689 BTS the_bts;
690 struct gprs_rlcmac_bts *bts;
691 struct gprs_rlcmac_trx *trx;
692 int counter1, counter2 = -1;
693 unsigned i;
694 enum test_mode mode_seq[] = {
695 TEST_MODE_DL_AFTER_UL,
696 TEST_MODE_UL_ONLY,
697 TEST_MODE_DL_AFTER_UL,
698 TEST_MODE_DL_ONLY,
699 };
700
701 printf("Going to test assignment with many connections, %s\n", text);
702
703 bts = the_bts.bts_data();
704 bts->alloc_algorithm = algo;
705
706 trx = &bts->trx[0];
707 trx->pdch[3].enable();
708 trx->pdch[4].enable();
709 trx->pdch[5].enable();
710 trx->pdch[6].enable();
711 trx->pdch[7].enable();
712
713 for (i = 0; i < ARRAY_SIZE(mode_seq); i += 1) {
714 counter1 = alloc_many_tbfs(&the_bts, 1, 29, mode_seq[i]);
715 fprintf(stderr, " Allocated %d TBFs (previously %d)\n",
716 counter1, counter2);
717
718 check_tfi_usage(&the_bts);
719
720 /* This will stop earlier due to USF shortage */
721 if (mode_seq[i] == TEST_MODE_UL_ONLY)
722 continue;
723
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200724 if (counter2 >= 0) {
725 if (counter1 < counter2)
726 fprintf(stderr, " Expected %d >= %d in %s\n",
727 counter1, counter2, text);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200728 OSMO_ASSERT(counter1 >= counter2);
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200729 }
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200730
731 counter2 = counter1;
732 }
733
734 printf(" Successfully allocated %d TBFs\n", counter1);
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200735 if (counter1 != (int)expect_num)
736 fprintf(stderr, " Expected %d TBFs for %s\n", expect_num, text);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200737
738 OSMO_ASSERT(expect_num == (unsigned)counter1);
739}
740
Jacob Erlbecke5655642015-06-29 12:19:52 +0200741static void test_successive_allocation()
742{
743 test_successive_allocation(alloc_algorithm_a, 1, 1, TEST_MODE_UL_AND_DL,
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200744 35, "algorithm A (UL and DL)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200745 test_successive_allocation(alloc_algorithm_b, 10, 10, TEST_MODE_UL_AND_DL,
Jacob Erlbeck16c6ecc2015-06-30 13:40:18 +0200746 32, "algorithm B class 10 (UL and DL)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200747 test_successive_allocation(alloc_algorithm_b, 12, 12, TEST_MODE_UL_AND_DL,
Jacob Erlbeck16c6ecc2015-06-30 13:40:18 +0200748 32, "algorithm B class 12 (UL and DL)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200749 test_successive_allocation(alloc_algorithm_b, 1, 12, TEST_MODE_UL_AND_DL,
Jacob Erlbecked46afd2015-07-01 12:19:40 +0200750 32, "algorithm B class 1-12 (UL and DL)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200751 test_successive_allocation(alloc_algorithm_b, 1, 29, TEST_MODE_UL_AND_DL,
Jacob Erlbeck16c6ecc2015-06-30 13:40:18 +0200752 32, "algorithm B class 1-29 (UL and DL)");
Jacob Erlbeck400ec022015-07-14 13:31:48 +0200753 test_successive_allocation(alloc_algorithm_dynamic, 1, 29, TEST_MODE_UL_AND_DL,
754 35, "algorithm dynamic class 1-29 (UL and DL)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200755
756 test_successive_allocation(alloc_algorithm_a, 1, 1, TEST_MODE_DL_AND_UL,
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200757 35, "algorithm A (DL and UL)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200758 test_successive_allocation(alloc_algorithm_b, 10, 10, TEST_MODE_DL_AND_UL,
Jacob Erlbeck16c6ecc2015-06-30 13:40:18 +0200759 32, "algorithm B class 10 (DL and UL)");
Jacob Erlbeck400ec022015-07-14 13:31:48 +0200760 test_successive_allocation(alloc_algorithm_dynamic, 10, 10, TEST_MODE_DL_AND_UL,
761 32, "algorithm dynamic class 10 (DL and UL)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200762
763 test_successive_allocation(alloc_algorithm_a, 1, 1, TEST_MODE_DL_AFTER_UL,
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200764 160, "algorithm A (DL after UL)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200765 test_successive_allocation(alloc_algorithm_b, 10, 10, TEST_MODE_DL_AFTER_UL,
Jacob Erlbeckec478752015-06-19 16:35:38 +0200766 32, "algorithm B class 10 (DL after UL)");
Jacob Erlbeck400ec022015-07-14 13:31:48 +0200767 test_successive_allocation(alloc_algorithm_dynamic, 10, 10, TEST_MODE_DL_AFTER_UL,
Jacob Erlbeck77da3552015-07-16 18:33:46 +0200768 95, "algorithm dynamic class 10 (DL after UL)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200769
770 test_successive_allocation(alloc_algorithm_a, 1, 1, TEST_MODE_UL_AFTER_DL,
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200771 35, "algorithm A (UL after DL)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200772 test_successive_allocation(alloc_algorithm_b, 10, 10, TEST_MODE_UL_AFTER_DL,
Jacob Erlbeck16c6ecc2015-06-30 13:40:18 +0200773 32, "algorithm B class 10 (UL after DL)");
Jacob Erlbeck400ec022015-07-14 13:31:48 +0200774 test_successive_allocation(alloc_algorithm_dynamic, 10, 10, TEST_MODE_UL_AFTER_DL,
775 35, "algorithm dynamic class 10 (UL after DL)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200776
777 test_successive_allocation(alloc_algorithm_a, 1, 1, TEST_MODE_UL_ONLY,
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200778 35, "algorithm A (UL only)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200779 test_successive_allocation(alloc_algorithm_b, 10, 10, TEST_MODE_UL_ONLY,
Jacob Erlbeck16c6ecc2015-06-30 13:40:18 +0200780 32, "algorithm B class 10 (UL only)");
Jacob Erlbeck400ec022015-07-14 13:31:48 +0200781 test_successive_allocation(alloc_algorithm_dynamic, 10, 10, TEST_MODE_UL_ONLY,
782 35, "algorithm dynamic class 10 (UL only)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200783
784 test_successive_allocation(alloc_algorithm_a, 1, 1, TEST_MODE_DL_ONLY,
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200785 160, "algorithm A (DL ONLY)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200786 test_successive_allocation(alloc_algorithm_b, 10, 10, TEST_MODE_DL_ONLY,
Jacob Erlbeckec478752015-06-19 16:35:38 +0200787 32, "algorithm B class 10 (DL ONLY)");
Jacob Erlbeck400ec022015-07-14 13:31:48 +0200788 test_successive_allocation(alloc_algorithm_dynamic, 10, 10, TEST_MODE_DL_ONLY,
789 101, "algorithm dynamic class 10 (DL ONLY)");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200790}
791
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200792static void test_many_connections()
793{
794 test_many_connections(alloc_algorithm_a, 160, "algorithm A");
795 test_many_connections(alloc_algorithm_b, 32, "algorithm B");
Jacob Erlbeck77da3552015-07-16 18:33:46 +0200796 test_many_connections(alloc_algorithm_dynamic, 160, "algorithm dynamic");
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200797}
798
Aravind Sirsikare26ee012016-09-06 18:15:45 +0530799static void test_2_consecutive_dl_tbfs()
800{
801 BTS the_bts;
802 struct gprs_rlcmac_bts *bts;
803 struct gprs_rlcmac_trx *trx;
804 uint8_t ms_class = 11;
805 uint8_t egprs_ms_class = 11;
806 gprs_rlcmac_tbf *dl_tbf1, *dl_tbf2;
807 uint8_t numTs1 = 0, numTs2 = 0;
808
809 printf("Testing DL TS allocation for Multi UEs\n");
810
811 bts = the_bts.bts_data();
812 bts->alloc_algorithm = alloc_algorithm_b;
813
814 trx = &bts->trx[0];
815 trx->pdch[4].enable();
816 trx->pdch[5].enable();
817 trx->pdch[6].enable();
818 trx->pdch[7].enable();
819
820 dl_tbf1 = tbf_alloc_dl_tbf(bts, NULL, 0, ms_class, egprs_ms_class, 0);
821 OSMO_ASSERT(dl_tbf1);
822
823 for (int i = 0; i < 8; i++) {
824 if (dl_tbf1->pdch[i])
825 numTs1++;
826 }
827 OSMO_ASSERT(numTs1 == 4);
828 printf("TBF1: numTs(%d)\n", numTs1);
829
830 dl_tbf2 = tbf_alloc_dl_tbf(bts, NULL, 0, ms_class, egprs_ms_class, 0);
831 OSMO_ASSERT(dl_tbf2);
832
833 for (int i = 0; i < 8; i++) {
834 if (dl_tbf2->pdch[i])
835 numTs2++;
836 }
837
838 /*
839 * TODO: currently 2nd DL TBF gets 3 TS
840 * This behaviour will be fixed in subsequent patch
841 */
842 printf("TBF2: numTs(%d)\n", numTs2);
843 OSMO_ASSERT(numTs2 == 3);
844
845 tbf_free(dl_tbf1);
846 tbf_free(dl_tbf2);
847}
848
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200849int main(int argc, char **argv)
850{
851 tall_pcu_ctx = talloc_named_const(NULL, 1, "moiji-mobile AllocTest context");
852 if (!tall_pcu_ctx)
853 abort();
854
Neels Hofmeyr78ce5912017-02-08 17:07:31 +0100855 msgb_talloc_ctx_init(tall_pcu_ctx, 0);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200856 osmo_init_logging(&gprs_log_info);
857 log_set_use_color(osmo_stderr_target, 0);
858 log_set_print_filename(osmo_stderr_target, 0);
Jacob Erlbeck9ec49e22015-06-29 13:00:20 +0200859 if (getenv("LOGL_DEBUG"))
860 log_set_log_level(osmo_stderr_target, LOGL_DEBUG);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200861
862 test_alloc_a();
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100863 test_alloc_b();
Jacob Erlbecke5655642015-06-29 12:19:52 +0200864 test_successive_allocation();
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200865 test_many_connections();
Aravind Sirsikare26ee012016-09-06 18:15:45 +0530866 test_2_consecutive_dl_tbfs();
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200867 return EXIT_SUCCESS;
868}
869
870/*
871 * stubs that should not be reached
872 */
873extern "C" {
874void l1if_pdch_req() { abort(); }
875void l1if_connect_pdch() { abort(); }
876void l1if_close_pdch() { abort(); }
877void l1if_open_pdch() { abort(); }
878}