blob: 20f5286f8eddb435a9d50c91814e7ed31465562e [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"
Pau Espin Pedrol9d1cdb12019-09-25 17:47:02 +020023#include "tbf_ul.h"
Pau Espin Pedrol2182e622021-01-14 16:48:38 +010024#include "tbf_dl.h"
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +020025#include "bts.h"
Pau Espin Pedrol2182e622021-01-14 16:48:38 +010026#include "gprs_ms.h"
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +020027
28#include <string.h>
29#include <stdio.h>
30
31extern "C" {
Max9f460712018-01-23 20:57:08 +010032#include "mslot_class.h"
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +020033#include <osmocom/core/application.h>
34#include <osmocom/core/msgb.h>
35#include <osmocom/core/talloc.h>
36#include <osmocom/core/utils.h>
37}
38
39/* globals used by the code */
40void *tall_pcu_ctx;
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +020041int16_t spoof_mnc = 0, spoof_mcc = 0;
Neels Hofmeyrbdc55fa2018-02-21 00:39:07 +010042bool spoof_mnc_3_digits = false;
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +020043
Daniel Willmann48aa0b02014-07-16 18:54:10 +020044static gprs_rlcmac_tbf *tbf_alloc(struct gprs_rlcmac_bts *bts,
Jacob Erlbecke2e004e2015-06-18 17:16:26 +020045 GprsMs *ms, gprs_rlcmac_tbf_direction dir,
Pau Espin Pedrol322456e2020-05-08 18:15:59 +020046 uint8_t use_trx, bool single_slot)
Daniel Willmann48aa0b02014-07-16 18:54:10 +020047{
Pau Espin Pedrol322456e2020-05-08 18:15:59 +020048 OSMO_ASSERT(ms != NULL);
Pau Espin Pedrol17402a52020-05-08 17:44:33 +020049
Daniel Willmann48aa0b02014-07-16 18:54:10 +020050 if (dir == GPRS_RLCMAC_UL_TBF)
Pau Espin Pedrol17402a52020-05-08 17:44:33 +020051 return tbf_alloc_ul_tbf(bts, ms, use_trx, single_slot);
Daniel Willmann48aa0b02014-07-16 18:54:10 +020052 else
Pau Espin Pedrol322456e2020-05-08 18:15:59 +020053 return tbf_alloc_dl_tbf(bts, ms, use_trx, single_slot);
Daniel Willmann48aa0b02014-07-16 18:54:10 +020054}
55
Pau Espin Pedrol2182e622021-01-14 16:48:38 +010056static void check_tfi_usage(struct gprs_rlcmac_bts *bts)
Jacob Erlbeck61205a72015-07-09 11:35:50 +020057{
58 int pdch_no;
Jacob Erlbeck61205a72015-07-09 11:35:50 +020059
60 struct gprs_rlcmac_tbf *tfi_usage[8][8][2][32] = {{{{NULL}}}};
Pau Espin Pedrol2182e622021-01-14 16:48:38 +010061 struct llist_head *tbf_lists[2] = {
62 &bts->ul_tbfs,
63 &bts->dl_tbfs
Jacob Erlbeck61205a72015-07-09 11:35:50 +020064 };
65
Pau Espin Pedrol2182e622021-01-14 16:48:38 +010066 struct llist_item *pos;
Jacob Erlbeck61205a72015-07-09 11:35:50 +020067 gprs_rlcmac_tbf *tbf;
Jacob Erlbeck61205a72015-07-09 11:35:50 +020068 unsigned list_idx;
69 struct gprs_rlcmac_tbf **tbf_var;
70
71 for (list_idx = 0; list_idx < ARRAY_SIZE(tbf_lists); list_idx += 1)
72 {
73
Pau Espin Pedrol2182e622021-01-14 16:48:38 +010074 llist_for_each_entry(pos, tbf_lists[list_idx], list) {
75 tbf = (struct gprs_rlcmac_tbf *)pos->entry;
Jacob Erlbeck61205a72015-07-09 11:35:50 +020076 for (pdch_no = 0; pdch_no < 8; pdch_no += 1) {
77 struct gprs_rlcmac_pdch *pdch = tbf->pdch[pdch_no];
78 if (pdch == NULL)
79 continue;
80
81 tbf_var = &tfi_usage
82 [tbf->trx->trx_no]
83 [pdch_no]
84 [tbf->direction]
85 [tbf->tfi()];
86
87 OSMO_ASSERT(*tbf_var == NULL);
88 if (tbf->direction == GPRS_RLCMAC_DL_TBF) {
89 OSMO_ASSERT(pdch->dl_tbf_by_tfi(
90 tbf->tfi()) == tbf);
Pau Espin Pedrol2182e622021-01-14 16:48:38 +010091 OSMO_ASSERT(bts_dl_tbf_by_tfi(bts,
Jacob Erlbeck61205a72015-07-09 11:35:50 +020092 tbf->tfi(),
Jacob Erlbeck3a10dbd2015-07-10 19:52:37 +020093 tbf->trx->trx_no,
94 pdch_no) == tbf);
Jacob Erlbeck61205a72015-07-09 11:35:50 +020095 } else {
96 OSMO_ASSERT(pdch->ul_tbf_by_tfi(
97 tbf->tfi()) == tbf);
Pau Espin Pedrol2182e622021-01-14 16:48:38 +010098 OSMO_ASSERT(bts_ul_tbf_by_tfi(bts,
Jacob Erlbeck61205a72015-07-09 11:35:50 +020099 tbf->tfi(),
Jacob Erlbeck3a10dbd2015-07-10 19:52:37 +0200100 tbf->trx->trx_no,
101 pdch_no) == tbf);
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200102 }
103 *tbf_var = tbf;
Jacob Erlbeck47a57f62015-07-08 12:53:16 +0200104 OSMO_ASSERT(pdch->assigned_tfi(tbf->direction) &
105 (1 << tbf->tfi()));
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200106 }
107 }
108 }
109}
110
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200111static void test_alloc_a(gprs_rlcmac_tbf_direction dir,
112 uint8_t slots, const int count)
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200113{
114 int tfi;
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200115 int i;
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200116 uint8_t used_trx, tmp_trx;
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100117 struct gprs_rlcmac_bts *bts = bts_alloc(the_pcu);
Pau Espin Pedrol322456e2020-05-08 18:15:59 +0200118 GprsMs *ms;
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200119 struct gprs_rlcmac_tbf *tbfs[32*8+1] = { 0, };
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200120
121 printf("Testing alloc_a direction(%d)\n", dir);
122
Pau Espin Pedrolac3fd122021-01-13 18:54:38 +0100123 the_pcu->alloc_algorithm = alloc_algorithm_a;
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200124
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +0200125 struct gprs_rlcmac_trx *trx = &bts->trx[0];
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200126 for (i = 0; i < 8; i += 1)
127 if (slots & (1 << i))
128 trx->pdch[i].enable();
129
130 OSMO_ASSERT(count >= 0 && count <= (int)ARRAY_SIZE(tbfs));
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200131
132 /**
133 * Currently alloc_a will only allocate from the first
134 * PDCH and all possible usf's. We run out of usf's before
135 * we are out of tfi's. Observe this and make sure that at
136 * least this part is working okay.
137 */
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200138 for (i = 0; i < (int)ARRAY_SIZE(tbfs); ++i) {
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100139 ms = bts_alloc_ms(bts, 0, 0);
Pau Espin Pedrol322456e2020-05-08 18:15:59 +0200140 tbfs[i] = tbf_alloc(bts, ms, dir, -1, 0);
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200141 if (tbfs[i] == NULL)
142 break;
143
144 used_trx = tbfs[i]->trx->trx_no;
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100145 tfi = bts_tfi_find_free(bts, dir, &tmp_trx, used_trx);
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200146 OSMO_ASSERT(tbfs[i]->tfi() != tfi);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200147 }
148
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100149 check_tfi_usage(bts);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200150
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200151 OSMO_ASSERT(i == count);
152
153 for (i = 0; i < count; ++i)
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200154 if (tbfs[i])
155 tbf_free(tbfs[i]);
156
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100157 ms = bts_alloc_ms(bts, 0, 0);
Pau Espin Pedrol322456e2020-05-08 18:15:59 +0200158 tbfs[0] = tbf_alloc(bts, ms, dir, -1, 0);
Holger Hans Peter Freytherf3f1bde2016-02-22 15:14:01 +0100159 OSMO_ASSERT(tbfs[0]);
160 tbf_free(tbfs[0]);
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100161 talloc_free(bts);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200162}
163
164static void test_alloc_a()
165{
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200166 /* slots 2 - 3 */
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200167 test_alloc_a(GPRS_RLCMAC_DL_TBF, 0x0c, 32*2);
Jacob Erlbeckec478752015-06-19 16:35:38 +0200168 test_alloc_a(GPRS_RLCMAC_UL_TBF, 0x0c, 14);
Jacob Erlbeckfa464bb2015-06-29 12:45:11 +0200169
170 /* slots 1 - 5 */
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200171 test_alloc_a(GPRS_RLCMAC_DL_TBF, 0x1e, 32*4);
Jacob Erlbeckec478752015-06-19 16:35:38 +0200172 test_alloc_a(GPRS_RLCMAC_UL_TBF, 0x1e, 28);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200173}
174
Max2ecf0fd2017-11-21 18:13:31 +0100175static void dump_assignment(struct gprs_rlcmac_tbf *tbf, const char *dir, bool verbose)
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100176{
Max2ecf0fd2017-11-21 18:13:31 +0100177 if (!verbose)
178 return;
179
Jacob Erlbeck1f332942015-05-04 08:21:17 +0200180 for (size_t i = 0; i < ARRAY_SIZE(tbf->pdch); ++i)
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100181 if (tbf->pdch[i])
Neels Hofmeyrd34646a2017-02-08 17:07:40 +0100182 printf("PDCH[%zu] is used for %s\n", i, dir);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100183 printf("PDCH[%d] is control_ts for %s\n", tbf->control_ts, dir);
184 printf("PDCH[%d] is first common for %s\n", tbf->first_common_ts, dir);
185}
186
Max2ecf0fd2017-11-21 18:13:31 +0100187#define ENABLE_PDCH(ts_no, enable_flag, trx) \
188 if (enable_flag) \
189 trx->pdch[ts_no].enable();
190
191static inline void enable_ts_on_bts(struct gprs_rlcmac_bts *bts,
192 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 +0100193{
Max2ecf0fd2017-11-21 18:13:31 +0100194 struct gprs_rlcmac_trx *trx = &bts->trx[0];
195
196 ENABLE_PDCH(0, ts0, trx);
197 ENABLE_PDCH(1, ts1, trx);
198 ENABLE_PDCH(2, ts2, trx);
199 ENABLE_PDCH(3, ts3, trx);
200 ENABLE_PDCH(4, ts4, trx);
201 ENABLE_PDCH(5, ts5, trx);
202 ENABLE_PDCH(6, ts6, trx);
203 ENABLE_PDCH(7, ts7, trx);
204}
205
206static inline bool test_alloc_b_ul_dl(bool ts0, bool ts1, bool ts2, bool ts3, bool ts4, bool ts5, bool ts6, bool ts7,
207 uint8_t ms_class, bool verbose)
208{
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100209 struct gprs_rlcmac_bts *bts = bts_alloc(the_pcu);
Pau Espin Pedrol17402a52020-05-08 17:44:33 +0200210 GprsMs *ms;
Max2ecf0fd2017-11-21 18:13:31 +0100211 gprs_rlcmac_ul_tbf *ul_tbf;
212 gprs_rlcmac_dl_tbf *dl_tbf;
213
214 if (verbose)
215 printf("Testing UL then DL assignment.\n");
216
Pau Espin Pedrolac3fd122021-01-13 18:54:38 +0100217 the_pcu->alloc_algorithm = alloc_algorithm_b;
Max2ecf0fd2017-11-21 18:13:31 +0100218
219 enable_ts_on_bts(bts, ts0, ts1, ts2, ts3, ts4, ts5, ts6, ts7);
220
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100221 ms = bts_alloc_ms(bts, ms_class, 0);
Pau Espin Pedrol1e009472021-01-11 20:40:19 +0100222 /* Avoid delaying free to avoid tons of to-be-freed ms objects queuing */
223 ms_set_timeout(ms, 0);
Pau Espin Pedrol17402a52020-05-08 17:44:33 +0200224 ul_tbf = tbf_alloc_ul_tbf(bts, ms, -1, true);
Max2ecf0fd2017-11-21 18:13:31 +0100225 if (!ul_tbf)
226 return false;
227
228 OSMO_ASSERT(ul_tbf->ms());
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100229 OSMO_ASSERT(ms_current_trx(ul_tbf->ms()));
Max2ecf0fd2017-11-21 18:13:31 +0100230
231 dump_assignment(ul_tbf, "UL", verbose);
232
233 /* assume final ack has not been sent */
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100234 dl_tbf = tbf_alloc_dl_tbf(bts, ms, ms_current_trx(ms)->trx_no, false);
Max2ecf0fd2017-11-21 18:13:31 +0100235 if (!dl_tbf)
236 return false;
237
238 dump_assignment(dl_tbf, "DL", verbose);
239
240 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
241
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100242 check_tfi_usage(bts);
Max2ecf0fd2017-11-21 18:13:31 +0100243
244 tbf_free(dl_tbf);
245 tbf_free(ul_tbf);
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100246 talloc_free(bts);
Max2ecf0fd2017-11-21 18:13:31 +0100247 return true;
248}
249
250static inline bool test_alloc_b_dl_ul(bool ts0, bool ts1, bool ts2, bool ts3, bool ts4, bool ts5, bool ts6, bool ts7,
251 uint8_t ms_class, bool verbose)
252{
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100253 struct gprs_rlcmac_bts *bts = bts_alloc(the_pcu);
Pau Espin Pedrol17402a52020-05-08 17:44:33 +0200254 GprsMs *ms;
Max2ecf0fd2017-11-21 18:13:31 +0100255 gprs_rlcmac_ul_tbf *ul_tbf;
256 gprs_rlcmac_dl_tbf *dl_tbf;
257
258 if (verbose)
259 printf("Testing DL then UL assignment followed by update\n");
260
Pau Espin Pedrolac3fd122021-01-13 18:54:38 +0100261 the_pcu->alloc_algorithm = alloc_algorithm_b;
Max2ecf0fd2017-11-21 18:13:31 +0100262
263 enable_ts_on_bts(bts, ts0, ts1, ts2, ts3, ts4, ts5, ts6, ts7);
264
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100265 ms = bts_alloc_ms(bts, ms_class, 0);
Pau Espin Pedrol1e009472021-01-11 20:40:19 +0100266 /* Avoid delaying free to avoid tons of to-be-freed ms objects queuing */
267 ms_set_timeout(ms, 0);
Pau Espin Pedrol322456e2020-05-08 18:15:59 +0200268 dl_tbf = tbf_alloc_dl_tbf(bts, ms, -1, true);
Max2ecf0fd2017-11-21 18:13:31 +0100269 if (!dl_tbf)
270 return false;
271
272 dl_tbf->update_ms(0x23, GPRS_RLCMAC_DL_TBF);
Pau Espin Pedrol322456e2020-05-08 18:15:59 +0200273 OSMO_ASSERT(dl_tbf->ms() == ms);
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100274 OSMO_ASSERT(ms_current_trx(dl_tbf->ms()));
Max2ecf0fd2017-11-21 18:13:31 +0100275
276 dump_assignment(dl_tbf, "DL", verbose);
277
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100278 ul_tbf = tbf_alloc_ul_tbf(bts, ms, ms_current_trx(ms)->trx_no, false);
Max2ecf0fd2017-11-21 18:13:31 +0100279 if (!ul_tbf)
280 return false;
281
282 ul_tbf->update_ms(0x23, GPRS_RLCMAC_UL_TBF);
283 ul_tbf->m_contention_resolution_done = 1;
284
285 dump_assignment(ul_tbf, "UL", verbose);
286
287 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
288
289 /* now update the dl_tbf */
290 dl_tbf->update();
291 dump_assignment(dl_tbf, "DL", verbose);
292 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
293
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100294 check_tfi_usage(bts);
Max2ecf0fd2017-11-21 18:13:31 +0100295
296 tbf_free(dl_tbf);
297 tbf_free(ul_tbf);
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100298 talloc_free(bts);
Max2ecf0fd2017-11-21 18:13:31 +0100299 return true;
300}
301
302static inline bool test_alloc_b_jolly(uint8_t ms_class)
303{
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100304 struct gprs_rlcmac_bts *bts = bts_alloc(the_pcu);
Pau Espin Pedrol17402a52020-05-08 17:44:33 +0200305 GprsMs *ms;
Max2ecf0fd2017-11-21 18:13:31 +0100306 int tfi;
307 uint8_t trx_no;
308 gprs_rlcmac_tbf *ul_tbf, *dl_tbf;
309
310 printf("Testing jolly example\n");
311
Pau Espin Pedrolac3fd122021-01-13 18:54:38 +0100312 the_pcu->alloc_algorithm = alloc_algorithm_b;
Max2ecf0fd2017-11-21 18:13:31 +0100313
314 enable_ts_on_bts(bts, false, true, true, true, true, false, false, false);
315
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100316 tfi = bts_tfi_find_free(bts, GPRS_RLCMAC_UL_TBF, &trx_no, -1);
Max2ecf0fd2017-11-21 18:13:31 +0100317 OSMO_ASSERT(tfi >= 0);
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100318 ms = bts_alloc_ms(bts, ms_class, 0);
Pau Espin Pedrol1e009472021-01-11 20:40:19 +0100319 /* Avoid delaying free to avoid tons of to-be-freed ms objects queuing */
320 ms_set_timeout(ms, 0);
Pau Espin Pedrol17402a52020-05-08 17:44:33 +0200321 ul_tbf = tbf_alloc_ul_tbf(bts, ms, -1, false);
Max2ecf0fd2017-11-21 18:13:31 +0100322 if (!ul_tbf)
323 return false;
324
Pau Espin Pedrol322456e2020-05-08 18:15:59 +0200325 OSMO_ASSERT(ul_tbf->ms() == ms);
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100326 OSMO_ASSERT(ms_current_trx(ul_tbf->ms()));
327 trx_no = ms_current_trx(ms)->trx_no;
Max2ecf0fd2017-11-21 18:13:31 +0100328 dump_assignment(ul_tbf, "UL", true);
329
330 /* assume final ack has not been sent */
Pau Espin Pedrol322456e2020-05-08 18:15:59 +0200331 dl_tbf = tbf_alloc_dl_tbf(bts, ms, trx_no, false);
Max2ecf0fd2017-11-21 18:13:31 +0100332 if (!dl_tbf)
333 return false;
334
335 dump_assignment(dl_tbf, "DL", true);
336
337 OSMO_ASSERT(dl_tbf->first_common_ts == ul_tbf->first_common_ts);
338
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100339 check_tfi_usage(bts);
Max2ecf0fd2017-11-21 18:13:31 +0100340
341 tbf_free(dl_tbf);
342 tbf_free(ul_tbf);
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100343 talloc_free(bts);
Max2ecf0fd2017-11-21 18:13:31 +0100344 return true;
345}
346
347static void test_alloc_b_for_ms(uint8_t ms_class)
348{
349 bool rc;
350
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100351 printf("Going to test multislot assignment MS_CLASS=%d\n", ms_class);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100352 /*
353 * PDCH is on TS 6,7,8 and we start with a UL allocation and
354 * then follow two DL allocations (once single, once normal).
355 *
356 * Uplink assigned and still available..
357 */
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100358
Max2ecf0fd2017-11-21 18:13:31 +0100359 rc = test_alloc_b_ul_dl(false, false, false, false, false, true, true, true, ms_class, true);
360 if (!rc)
361 return;
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100362
363 /**
364 * Test with the other order.. first DL and then UL
365 */
Max2ecf0fd2017-11-21 18:13:31 +0100366 rc = test_alloc_b_dl_ul(false, false, false, false, false, true, true, true, ms_class, true);
367 if (!rc)
368 return;
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100369
370 /* Andreas osmocom-pcu example */
Max2ecf0fd2017-11-21 18:13:31 +0100371 test_alloc_b_jolly(ms_class);
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100372}
373
Max2ecf0fd2017-11-21 18:13:31 +0100374static 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 +0100375{
Max2ecf0fd2017-11-21 18:13:31 +0100376 bool rc;
377
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100378 /* we can test the allocation failures differently */
379 if (!ts0 && !ts1 && !ts2 && !ts3 && !ts4 && !ts5 && !ts6 && !ts7)
380 return;
381
382 printf("Mass test: TS0(%c%c%c%c%c%c%c%c)TS7 MS_Class=%d\n",
383 ts0 ? 'O' : 'x',
384 ts1 ? 'O' : 'x',
385 ts2 ? 'O' : 'x',
386 ts3 ? 'O' : 'x',
387 ts4 ? 'O' : 'x',
388 ts5 ? 'O' : 'x',
389 ts6 ? 'O' : 'x',
390 ts7 ? 'O' : 'x', ms_class);
391 fflush(stdout);
392
Max2ecf0fd2017-11-21 18:13:31 +0100393 rc = test_alloc_b_ul_dl(ts0, ts1, ts2, ts3, ts4, ts5, ts6, ts7, ms_class, false);
394 if (!rc)
395 return;
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100396
397 /**
398 * Test with the other order.. first DL and then UL
399 */
Max2ecf0fd2017-11-21 18:13:31 +0100400 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 +0100401}
402
403static void test_all_alloc_b()
404{
405 /* it is a bit crazy... */
406 for (uint8_t ts0 = 0; ts0 < 2; ++ts0)
407 for (uint8_t ts1 = 0; ts1 < 2; ++ts1)
408 for (uint8_t ts2 = 0; ts2 < 2; ++ts2)
409 for (uint8_t ts3 = 0; ts3 < 2; ++ts3)
410 for (uint8_t ts4 = 0; ts4 < 2; ++ts4)
411 for (uint8_t ts5 = 0; ts5 < 2; ++ts5)
412 for (uint8_t ts6 = 0; ts6 < 2; ++ts6)
413 for (uint8_t ts7 = 0; ts7 < 2; ++ts7)
Max9f460712018-01-23 20:57:08 +0100414 for (int ms_class = 0; ms_class < mslot_class_max(); ++ms_class)
Max2ecf0fd2017-11-21 18:13:31 +0100415 test_alloc_mass(ts0, ts1, ts2, ts3, ts4, ts5, ts6, ts7, ms_class);
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100416}
417
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100418static void test_alloc_b()
419{
Max9f460712018-01-23 20:57:08 +0100420 for (int i = 0; i < mslot_class_max(); ++i)
Max2ecf0fd2017-11-21 18:13:31 +0100421 test_alloc_b_for_ms(i);
Holger Hans Peter Freytherf3eec042013-12-26 10:19:18 +0100422
423 test_all_alloc_b();
Holger Hans Peter Freytherc7b998c2013-12-25 19:25:10 +0100424}
425
Maxe9fe0e32017-09-28 15:56:05 +0200426typedef int (*algo_t)(struct gprs_rlcmac_bts *bts, struct GprsMs *ms, struct gprs_rlcmac_tbf *tbf, bool single,
427 int8_t use_trx);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200428
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200429static char get_dir_char(uint8_t mask, uint8_t tx, uint8_t rx, uint8_t busy)
Jacob Erlbecke5655642015-06-29 12:19:52 +0200430{
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200431 int offs = busy ? 32 : 0;
432 return (mask & tx & rx) ? 'C' + offs :
433 (mask & tx) ? 'U' + offs :
434 (mask & rx) ? 'D' + offs :
Jacob Erlbecke5655642015-06-29 12:19:52 +0200435 '.';
436}
437
438enum test_mode {
439 TEST_MODE_UL_ONLY,
440 TEST_MODE_DL_ONLY,
441 TEST_MODE_UL_AND_DL,
442 TEST_MODE_DL_AND_UL,
443 TEST_MODE_DL_AFTER_UL,
444 TEST_MODE_UL_AFTER_DL,
445};
446
Maxc59ef122017-11-27 13:21:41 +0100447static inline char *test_mode_descr(enum test_mode t)
448{
449 switch (t) {
450 case TEST_MODE_UL_ONLY: return (char*)"UL only";
451 case TEST_MODE_DL_ONLY: return (char*)"DL only";
452 case TEST_MODE_UL_AND_DL: return (char*)"UL and DL";
453 case TEST_MODE_DL_AND_UL: return (char*)"DL and UL";
454 case TEST_MODE_DL_AFTER_UL: return (char*)"DL after UL";
455 case TEST_MODE_UL_AFTER_DL: return (char*)"UL after DL";
456 default: return NULL;
457 }
458}
459
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100460static GprsMs *alloc_tbfs(struct gprs_rlcmac_bts *bts, struct GprsMs *old_ms, enum test_mode mode)
Jacob Erlbecke5655642015-06-29 12:19:52 +0200461{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100462 struct GprsMs *ms, *new_ms;
Jacob Erlbeck5879c642015-07-10 10:41:36 +0200463 uint8_t trx_no = -1;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200464
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100465 OSMO_ASSERT(old_ms != NULL);
Pau Espin Pedrol17402a52020-05-08 17:44:33 +0200466
Jacob Erlbecke5655642015-06-29 12:19:52 +0200467 gprs_rlcmac_tbf *tbf = NULL;
468
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100469 if (ms_current_trx(old_ms))
470 trx_no = ms_current_trx(old_ms)->trx_no;
Jacob Erlbeck5879c642015-07-10 10:41:36 +0200471
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100472 ms_ref(old_ms);
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200473
Jacob Erlbecke5655642015-06-29 12:19:52 +0200474 /* Allocate what is needed first */
475 switch (mode) {
476 case TEST_MODE_UL_ONLY:
477 case TEST_MODE_DL_AFTER_UL:
478 case TEST_MODE_UL_AND_DL:
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100479 if (ms_ul_tbf(old_ms))
480 tbf_free(ms_ul_tbf(old_ms));
481 tbf = tbf_alloc_ul_tbf(bts, old_ms, trx_no, false);
482 if (tbf == NULL) {
483 ms_unref(old_ms);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200484 return NULL;
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100485 }
Jacob Erlbecke5655642015-06-29 12:19:52 +0200486 break;
487 case TEST_MODE_DL_ONLY:
488 case TEST_MODE_UL_AFTER_DL:
489 case TEST_MODE_DL_AND_UL:
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100490 if (ms_dl_tbf(old_ms))
491 tbf_free(ms_dl_tbf(old_ms));
492 tbf = tbf_alloc_dl_tbf(bts, old_ms, trx_no, false);
493 if (tbf == NULL) {
494 ms_unref(old_ms);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200495 return NULL;
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100496 }
Jacob Erlbecke5655642015-06-29 12:19:52 +0200497 }
498
499 OSMO_ASSERT(tbf);
500 OSMO_ASSERT(tbf->ms());
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100501 OSMO_ASSERT(old_ms == tbf->ms());
Jacob Erlbecke5655642015-06-29 12:19:52 +0200502 ms = tbf->ms();
503
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100504 ms_ref(ms);
505 new_ms = ms;
Jacob Erlbeck14376a72015-07-07 11:31:28 +0200506 /* Continue with what is needed next */
507 switch (mode) {
508 case TEST_MODE_UL_ONLY:
509 case TEST_MODE_DL_ONLY:
510 /* We are done */
511 break;
512
513 case TEST_MODE_DL_AFTER_UL:
514 case TEST_MODE_UL_AND_DL:
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100515 new_ms = alloc_tbfs(bts, ms, TEST_MODE_DL_ONLY);
Jacob Erlbeck14376a72015-07-07 11:31:28 +0200516 break;
517
518 case TEST_MODE_UL_AFTER_DL:
519 case TEST_MODE_DL_AND_UL:
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100520 new_ms = alloc_tbfs(bts, ms, TEST_MODE_UL_ONLY);
Jacob Erlbeck14376a72015-07-07 11:31:28 +0200521 break;
522 }
523
Jacob Erlbecke5655642015-06-29 12:19:52 +0200524 /* Optionally delete the TBF */
525 switch (mode) {
526 case TEST_MODE_DL_AFTER_UL:
527 case TEST_MODE_UL_AFTER_DL:
528 tbf_free(tbf);
Jacob Erlbeck0f352a62015-07-16 18:23:33 +0200529 tbf = NULL;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200530 break;
531
532 default:
533 break;
534 }
535
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100536 if (!new_ms && tbf)
Jacob Erlbeck0f352a62015-07-16 18:23:33 +0200537 tbf_free(tbf);
538
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100539 ms_unref(old_ms);
540 ms_unref(ms);
541 return new_ms;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200542}
543
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100544static unsigned alloc_many_tbfs(struct gprs_rlcmac_bts *bts, unsigned min_class,
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200545 unsigned max_class, enum test_mode mode)
Jacob Erlbecke5655642015-06-29 12:19:52 +0200546{
Jacob Erlbecke5655642015-06-29 12:19:52 +0200547 unsigned counter;
548 unsigned ms_class = min_class;
549
Jacob Erlbecke5655642015-06-29 12:19:52 +0200550 for (counter = 0; 1; counter += 1) {
551 gprs_rlcmac_tbf *ul_tbf, *dl_tbf;
552 uint8_t ul_slots = 0;
553 uint8_t dl_slots = 0;
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200554 uint8_t busy_slots = 0;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200555 unsigned i;
556 int tfi = -1;
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200557 int tfi2;
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200558 uint8_t trx_no2;
559 struct gprs_rlcmac_trx *trx;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200560 GprsMs *ms;
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200561 enum gprs_rlcmac_tbf_direction dir;
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200562 uint32_t tlli = counter + 0xc0000000;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200563
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100564 ms = bts_ms_by_tlli(bts, tlli, GSM_RESERVED_TMSI);
Pau Espin Pedrol17402a52020-05-08 17:44:33 +0200565 if (!ms)
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100566 ms = bts_alloc_ms(bts, 0, 0);
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100567 ms_set_ms_class(ms, ms_class);
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100568 ms = alloc_tbfs(bts, ms, mode);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200569 if (!ms)
570 break;
571
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100572 ms_set_tlli(ms, tlli);
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200573
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100574 ul_tbf = ms_ul_tbf(ms);
575 dl_tbf = ms_dl_tbf(ms);
576 trx = ms_current_trx(ms);
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200577
578 OSMO_ASSERT(ul_tbf || dl_tbf);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200579
580 if (ul_tbf) {
581 ul_slots = 1 << ul_tbf->first_common_ts;
582 tfi = ul_tbf->tfi();
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200583 dir = GPRS_RLCMAC_UL_TBF;
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200584 } else {
Jacob Erlbecke5655642015-06-29 12:19:52 +0200585 ul_slots = 1 << dl_tbf->first_common_ts;
586 tfi = dl_tbf->tfi();
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200587 dir = GPRS_RLCMAC_DL_TBF;
Jacob Erlbecke5655642015-06-29 12:19:52 +0200588 }
589
590 for (i = 0; dl_tbf && i < ARRAY_SIZE(dl_tbf->pdch); i += 1)
591 if (dl_tbf->pdch[i])
592 dl_slots |= 1 << i;
593
Max5b0df1f2017-09-11 10:38:59 +0200594 for (i = 0; ul_tbf && i < ARRAY_SIZE(ul_tbf->pdch); i += 1)
595 if (ul_tbf->pdch[i])
596 ul_slots |= 1 << i;
597
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200598 for (i = 0; trx && i < ARRAY_SIZE(trx->pdch); i += 1) {
599 struct gprs_rlcmac_pdch *pdch = &trx->pdch[i];
600
601 if (ul_tbf && dl_tbf)
602 continue;
603
604 if (ul_tbf &&
Maxd000d802017-09-20 17:55:28 +0200605 pdch->assigned_tfi(GPRS_RLCMAC_DL_TBF) != NO_FREE_TFI)
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200606 continue;
607
608 if (dl_tbf &&
Maxd000d802017-09-20 17:55:28 +0200609 pdch->assigned_tfi(GPRS_RLCMAC_UL_TBF) != NO_FREE_TFI)
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200610 continue;
611
612 busy_slots |= 1 << i;
613 }
614
Maxd000d802017-09-20 17:55:28 +0200615 printf(" TBF[%d] class %d reserves " OSMO_BIT_SPEC "\n",
Jacob Erlbecke5655642015-06-29 12:19:52 +0200616 tfi, ms_class,
Jacob Erlbeckbf904222015-07-16 18:19:09 +0200617 get_dir_char(0x01, ul_slots, dl_slots, busy_slots),
618 get_dir_char(0x02, ul_slots, dl_slots, busy_slots),
619 get_dir_char(0x04, ul_slots, dl_slots, busy_slots),
620 get_dir_char(0x08, ul_slots, dl_slots, busy_slots),
621 get_dir_char(0x10, ul_slots, dl_slots, busy_slots),
622 get_dir_char(0x20, ul_slots, dl_slots, busy_slots),
623 get_dir_char(0x40, ul_slots, dl_slots, busy_slots),
624 get_dir_char(0x80, ul_slots, dl_slots, busy_slots));
Jacob Erlbecke5655642015-06-29 12:19:52 +0200625
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200626 if (tfi >= 0) {
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100627 OSMO_ASSERT(ms_current_trx(ms));
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100628 tfi2 = bts_tfi_find_free(bts, dir, &trx_no2,
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100629 ms_current_trx(ms)->trx_no);
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200630 OSMO_ASSERT(tfi != tfi2);
Jacob Erlbeck7b3675b2015-07-16 18:28:22 +0200631 OSMO_ASSERT(tfi2 < 0 ||
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100632 trx_no2 == ms_current_trx(ms)->trx_no);
Jacob Erlbeck61205a72015-07-09 11:35:50 +0200633 }
634
Jacob Erlbecke5655642015-06-29 12:19:52 +0200635 ms_class += 1;
636 if (ms_class > max_class)
637 ms_class = min_class;
638 }
639
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200640 return counter;
641}
642
643static void test_successive_allocation(algo_t algo, unsigned min_class,
644 unsigned max_class, enum test_mode mode,
645 unsigned expect_num, const char *text)
646{
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100647 struct gprs_rlcmac_bts *bts = bts_alloc(the_pcu);
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200648 struct gprs_rlcmac_trx *trx;
649 unsigned counter;
650
Maxc59ef122017-11-27 13:21:41 +0100651 printf("Going to test assignment with many TBF, algorithm %s class %u..%u (%s)\n",
652 text, min_class, max_class, test_mode_descr(mode));
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200653
Pau Espin Pedrolac3fd122021-01-13 18:54:38 +0100654 the_pcu->alloc_algorithm = algo;
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200655
656 trx = &bts->trx[0];
657 trx->pdch[3].enable();
658 trx->pdch[4].enable();
659 trx->pdch[5].enable();
660 trx->pdch[6].enable();
661 trx->pdch[7].enable();
662
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100663 counter = alloc_many_tbfs(bts, min_class, max_class, mode);
Jacob Erlbeck69c9bfa2015-07-13 14:38:18 +0200664
Maxc59ef122017-11-27 13:21:41 +0100665 printf(" Successfully allocated %u UL TBFs, algorithm %s class %u..%u (%s)\n",
666 counter, text, min_class, max_class, test_mode_descr(mode));
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200667 if (counter != expect_num)
Maxc59ef122017-11-27 13:21:41 +0100668 fprintf(stderr, " Expected %u TBFs (got %u), algorithm %s class %u..%u (%s)\n",
669 expect_num, counter, text, min_class, max_class, test_mode_descr(mode));
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200670
Jacob Erlbeckec478752015-06-19 16:35:38 +0200671 OSMO_ASSERT(counter == expect_num);
Jacob Erlbecke0853cd2015-07-10 12:25:25 +0200672
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100673 check_tfi_usage(bts);
674 talloc_free(bts);
Jacob Erlbecke5655642015-06-29 12:19:52 +0200675}
676
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200677static void test_many_connections(algo_t algo, unsigned expect_num,
678 const char *text)
679{
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100680 struct gprs_rlcmac_bts *bts = bts_alloc(the_pcu);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200681 struct gprs_rlcmac_trx *trx;
682 int counter1, counter2 = -1;
683 unsigned i;
684 enum test_mode mode_seq[] = {
685 TEST_MODE_DL_AFTER_UL,
686 TEST_MODE_UL_ONLY,
687 TEST_MODE_DL_AFTER_UL,
688 TEST_MODE_DL_ONLY,
689 };
690
Maxc59ef122017-11-27 13:21:41 +0100691 printf("Going to test assignment with many connections, algorithm %s\n", text);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200692
Pau Espin Pedrolac3fd122021-01-13 18:54:38 +0100693 the_pcu->alloc_algorithm = algo;
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200694
695 trx = &bts->trx[0];
696 trx->pdch[3].enable();
697 trx->pdch[4].enable();
698 trx->pdch[5].enable();
699 trx->pdch[6].enable();
700 trx->pdch[7].enable();
701
702 for (i = 0; i < ARRAY_SIZE(mode_seq); i += 1) {
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100703 counter1 = alloc_many_tbfs(bts, 1, mslot_class_max(), mode_seq[i]);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200704 fprintf(stderr, " Allocated %d TBFs (previously %d)\n",
705 counter1, counter2);
706
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100707 check_tfi_usage(bts);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200708
709 /* This will stop earlier due to USF shortage */
710 if (mode_seq[i] == TEST_MODE_UL_ONLY)
711 continue;
712
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200713 if (counter2 >= 0) {
714 if (counter1 < counter2)
715 fprintf(stderr, " Expected %d >= %d in %s\n",
716 counter1, counter2, text);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200717 OSMO_ASSERT(counter1 >= counter2);
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200718 }
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200719
720 counter2 = counter1;
721 }
722
723 printf(" Successfully allocated %d TBFs\n", counter1);
Jacob Erlbeck88fb6132015-07-16 15:01:38 +0200724 if (counter1 != (int)expect_num)
Maxc59ef122017-11-27 13:21:41 +0100725 fprintf(stderr, " Expected %d TBFs (got %d) for algorithm %s\n", expect_num, counter1, text);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200726
727 OSMO_ASSERT(expect_num == (unsigned)counter1);
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100728 talloc_free(bts);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200729}
730
Maxc59ef122017-11-27 13:21:41 +0100731static 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 +0200732{
Maxc59ef122017-11-27 13:21:41 +0100733 test_successive_allocation(alloc_algorithm_a, 1, 1, mode, exp_A, "A");
734 test_successive_allocation(alloc_algorithm_b, 10, 10, mode, exp_B, "B");
735 test_successive_allocation(alloc_algorithm_dynamic, 10, 10, mode, exp_dyn, "dynamic");
Jacob Erlbecke5655642015-06-29 12:19:52 +0200736}
737
Maxc59ef122017-11-27 13:21:41 +0100738static void test_successive_allocations()
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200739{
Maxc59ef122017-11-27 13:21:41 +0100740 test_successive_allocation(alloc_algorithm_a, 1, 1, TEST_MODE_UL_AND_DL, 35, "A");
741 test_successive_allocation(alloc_algorithm_b, 10, 10, TEST_MODE_UL_AND_DL, 32, "B");
742 test_successive_allocation(alloc_algorithm_b, 12, 12, TEST_MODE_UL_AND_DL, 32, "B");
743
Max9f460712018-01-23 20:57:08 +0100744 test_successive_allocation(alloc_algorithm_b, 1, 12, TEST_MODE_UL_AND_DL, 32, "B");
Max01bd0cc2018-01-23 20:58:49 +0100745 test_successive_allocation(alloc_algorithm_b, 1, mslot_class_max(), TEST_MODE_UL_AND_DL, 32, "B");
Max9f460712018-01-23 20:57:08 +0100746 test_successive_allocation(alloc_algorithm_dynamic, 1, mslot_class_max(), TEST_MODE_UL_AND_DL, 35, "dynamic");
Maxc59ef122017-11-27 13:21:41 +0100747
748 test_a_b_dyn(TEST_MODE_DL_AND_UL, 35, 32, 32);
749 test_a_b_dyn(TEST_MODE_DL_AFTER_UL, 160, 32, 95);
750 test_a_b_dyn(TEST_MODE_UL_AFTER_DL, 35, 32, 35);
751 test_a_b_dyn(TEST_MODE_UL_ONLY, 35, 32, 35);
752 test_a_b_dyn(TEST_MODE_DL_ONLY, 160, 32, 101);
Jacob Erlbecka8c2aaf2015-07-13 14:50:08 +0200753}
754
Aravind Sirsikare26ee012016-09-06 18:15:45 +0530755static void test_2_consecutive_dl_tbfs()
756{
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100757 struct gprs_rlcmac_bts *bts = bts_alloc(the_pcu);
Pau Espin Pedrol322456e2020-05-08 18:15:59 +0200758 GprsMs *ms;
Aravind Sirsikare26ee012016-09-06 18:15:45 +0530759 struct gprs_rlcmac_trx *trx;
760 uint8_t ms_class = 11;
761 uint8_t egprs_ms_class = 11;
762 gprs_rlcmac_tbf *dl_tbf1, *dl_tbf2;
763 uint8_t numTs1 = 0, numTs2 = 0;
764
765 printf("Testing DL TS allocation for Multi UEs\n");
766
Pau Espin Pedrolac3fd122021-01-13 18:54:38 +0100767 the_pcu->alloc_algorithm = alloc_algorithm_b;
Aravind Sirsikare26ee012016-09-06 18:15:45 +0530768
769 trx = &bts->trx[0];
770 trx->pdch[4].enable();
771 trx->pdch[5].enable();
772 trx->pdch[6].enable();
773 trx->pdch[7].enable();
774
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100775 ms = bts_alloc_ms(bts, ms_class, egprs_ms_class);
Pau Espin Pedrol322456e2020-05-08 18:15:59 +0200776 dl_tbf1 = tbf_alloc_dl_tbf(bts, ms, 0, false);
Aravind Sirsikare26ee012016-09-06 18:15:45 +0530777 OSMO_ASSERT(dl_tbf1);
778
779 for (int i = 0; i < 8; i++) {
780 if (dl_tbf1->pdch[i])
781 numTs1++;
782 }
783 OSMO_ASSERT(numTs1 == 4);
784 printf("TBF1: numTs(%d)\n", numTs1);
785
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100786 ms = bts_alloc_ms(bts, ms_class, egprs_ms_class);
Pau Espin Pedrol322456e2020-05-08 18:15:59 +0200787 dl_tbf2 = tbf_alloc_dl_tbf(bts, ms, 0, false);
Aravind Sirsikare26ee012016-09-06 18:15:45 +0530788 OSMO_ASSERT(dl_tbf2);
789
790 for (int i = 0; i < 8; i++) {
791 if (dl_tbf2->pdch[i])
792 numTs2++;
793 }
794
795 /*
796 * TODO: currently 2nd DL TBF gets 3 TS
797 * This behaviour will be fixed in subsequent patch
798 */
799 printf("TBF2: numTs(%d)\n", numTs2);
800 OSMO_ASSERT(numTs2 == 3);
801
802 tbf_free(dl_tbf1);
803 tbf_free(dl_tbf2);
Pau Espin Pedrol2182e622021-01-14 16:48:38 +0100804 talloc_free(bts);
Aravind Sirsikare26ee012016-09-06 18:15:45 +0530805}
806
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200807int main(int argc, char **argv)
808{
809 tall_pcu_ctx = talloc_named_const(NULL, 1, "moiji-mobile AllocTest context");
810 if (!tall_pcu_ctx)
811 abort();
812
Neels Hofmeyr78ce5912017-02-08 17:07:31 +0100813 msgb_talloc_ctx_init(tall_pcu_ctx, 0);
Neels Hofmeyr42f2d612018-04-01 16:54:40 +0200814 osmo_init_logging2(tall_pcu_ctx, &gprs_log_info);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200815 log_set_use_color(osmo_stderr_target, 0);
816 log_set_print_filename(osmo_stderr_target, 0);
Philipp Maierde0e5582020-03-25 12:23:52 +0100817 log_set_category_filter(osmo_stderr_target, DTBF, 1, LOGL_INFO);
Jacob Erlbeck9ec49e22015-06-29 13:00:20 +0200818 if (getenv("LOGL_DEBUG"))
819 log_set_log_level(osmo_stderr_target, LOGL_DEBUG);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200820
Pau Espin Pedrolac3fd122021-01-13 18:54:38 +0100821 the_pcu = gprs_pcu_alloc(tall_pcu_ctx);
822
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200823 test_alloc_a();
Holger Hans Peter Freyther4af30532013-12-25 19:16:55 +0100824 test_alloc_b();
Maxc59ef122017-11-27 13:21:41 +0100825 test_successive_allocations();
826 test_many_connections(alloc_algorithm_a, 160, "A");
Max01bd0cc2018-01-23 20:58:49 +0100827 test_many_connections(alloc_algorithm_b, 32, "B");
Maxc59ef122017-11-27 13:21:41 +0100828 test_many_connections(alloc_algorithm_dynamic, 160, "dynamic");
Aravind Sirsikare26ee012016-09-06 18:15:45 +0530829 test_2_consecutive_dl_tbfs();
Pau Espin Pedrolac3fd122021-01-13 18:54:38 +0100830
831 talloc_free(the_pcu);
Holger Hans Peter Freytherbfdd5f22013-10-16 17:29:31 +0200832 return EXIT_SUCCESS;
833}
834
835/*
836 * stubs that should not be reached
837 */
838extern "C" {
839void l1if_pdch_req() { abort(); }
840void l1if_connect_pdch() { abort(); }
841void l1if_close_pdch() { abort(); }
842void l1if_open_pdch() { abort(); }
843}