blob: 2de2dda5275fbbdc11c52f748200b4ec31e87d12 [file] [log] [blame]
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +04001/* gprs_rlcmac.cpp
2 *
3 * Copyright (C) 2012 Ivan Klyuchnikov
Andreas Eversberg5dac2f02012-06-27 15:52:04 +02004 * Copyright (C) 2012 Andreas Eversberg <jolly@eversberg.eu>
Holger Hans Peter Freytherd6bd91e2013-08-24 21:16:55 +02005 * Copyright (C) 2013 by Holger Hans Peter Freyther
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +04006 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
20 */
21
22#include <gprs_bssgp_pcu.h>
23#include <pcu_l1_if.h>
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +040024#include <gprs_rlcmac.h>
Holger Hans Peter Freyther099535a2013-10-16 17:42:31 +020025#include <tbf.h>
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +040026
Andreas Eversberg53f47252012-07-15 07:10:10 +020027
Andreas Eversberg3b1332c2012-10-03 14:20:53 +020028struct gprs_rlcmac_cs gprs_rlcmac_cs[] = {
29/* frame length data block max payload */
30 { 0, 0, 0 },
31 { 23, 23, 20 }, /* CS-1 */
32 { 34, 33, 30 }, /* CS-2 */
33 { 40, 39, 36 }, /* CS-3 */
34 { 54, 53, 50 }, /* CS-4 */
35};
36
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +020037LLIST_HEAD(gprs_rlcmac_ul_tbfs);
38LLIST_HEAD(gprs_rlcmac_dl_tbfs);
Andreas Eversberg2b914642012-07-19 13:06:26 +020039llist_head *gprs_rlcmac_tbfs_lists[] = {
40 &gprs_rlcmac_ul_tbfs,
41 &gprs_rlcmac_dl_tbfs,
42 NULL
43};
Andreas Eversberg6681bb82012-07-25 08:48:44 +020044extern void *tall_pcu_ctx;
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +040045
Andreas Eversbergb83e2a72012-10-07 15:26:00 +020046#ifdef DEBUG_DIAGRAM
47struct timeval diagram_time = {0,0};
48struct timeval diagram_last_tv = {0,0};
49
50void debug_diagram(int diag, const char *format, ...)
51{
52 va_list ap;
53 char debug[128];
54 char line[1024];
55 struct gprs_rlcmac_tbf *tbf, *tbf_a[16];
56 int max_diag = -1, i;
57 uint64_t diff = 0;
58
59 va_start(ap, format);
60 vsnprintf(debug, sizeof(debug) - 1, format, ap);
61 debug[19] = ' ';
62 debug[20] = '\0';
63 va_end(ap);
64
65 memset(tbf_a, 0, sizeof(tbf_a));
66 llist_for_each_entry(tbf, &gprs_rlcmac_ul_tbfs, list) {
67 if (tbf->diag < 16) {
68 if (tbf->diag > max_diag)
69 max_diag = tbf->diag;
70 tbf_a[tbf->diag] = tbf;
71 }
72 }
73 llist_for_each_entry(tbf, &gprs_rlcmac_dl_tbfs, list) {
74 if (tbf->diag < 16) {
75 if (tbf->diag > max_diag)
76 max_diag = tbf->diag;
77 tbf_a[tbf->diag] = tbf;
78 }
79 }
80
81 if (diagram_last_tv.tv_sec) {
82 diff = (uint64_t)(diagram_time.tv_sec -
83 diagram_last_tv.tv_sec) * 1000;
84 diff += diagram_time.tv_usec / 1000;
85 diff -= diagram_last_tv.tv_usec / 1000;
86 }
87 memcpy(&diagram_last_tv, &diagram_time, sizeof(struct timeval));
88
89 if (diff > 0) {
90 if (diff > 99999)
91 strcpy(line, " ... : ");
92 else
93 sprintf(line, "%3d.%03d: ", (int)(diff / 1000),
94 (int)(diff % 1000));
95 for (i = 0; i <= max_diag; i++) {
96 if (tbf_a[i] == NULL) {
97 strcat(line, " ");
98 continue;
99 }
100 if (tbf_a[i]->diag_new) {
101 strcat(line, " | ");
102 continue;
103 }
104 strcat(line, " ");
105 }
106 puts(line);
107 }
108 strcpy(line, " : ");
109 for (i = 0; i <= max_diag; i++) {
110 if (tbf_a[i] == NULL) {
111 strcat(line, " ");
112 continue;
113 }
114 if (tbf_a[i]->diag != diag) {
115 strcat(line, " | ");
116 continue;
117 }
118 if (strlen(debug) < 19) {
119 strcat(line, " ");
120 memcpy(line + strlen(line) - 11 - strlen(debug) / 2,
121 debug, strlen(debug));
122 } else
123 strcat(line, debug);
124 tbf_a[i]->diag_new = 1;
125 }
126 puts(line);
127}
128#endif
129
Holger Hans Peter Freyther483f77a2013-10-16 16:33:00 +0200130/* FIXME: spread resources over multiple TRX. Also add option to use same
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200131 * TRX in case of existing TBF for TLLI in the other direction. */
Andreas Eversberg02d7cd22013-01-15 08:59:34 +0100132/* search for free TFI and return TFI, TRX */
Holger Hans Peter Freytherd6bd91e2013-08-24 21:16:55 +0200133int tfi_find_free(struct gprs_rlcmac_bts *bts, enum gprs_rlcmac_tbf_direction dir,
Holger Hans Peter Freytherb67a8a32013-07-28 18:55:14 +0200134 uint8_t *_trx, int8_t use_trx)
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400135{
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200136 struct gprs_rlcmac_pdch *pdch;
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200137 struct gprs_rlcmac_tbf **tbfp;
138 uint8_t trx_from, trx_to, trx, ts, tfi;
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400139
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200140 if (use_trx >= 0 && use_trx < 8)
141 trx_from = trx_to = use_trx;
142 else {
143 trx_from = 0;
144 trx_to = 7;
145 }
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200146
147 /* on TRX find first enabled TS */
148 for (trx = trx_from; trx <= trx_to; trx++) {
Andreas Eversberg02d7cd22013-01-15 08:59:34 +0100149 for (ts = 0; ts < 8; ts++) {
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200150 pdch = &bts->trx[trx].pdch[ts];
151 if (!pdch->enable)
152 continue;
153 break;
154 }
155 if (ts < 8)
156 break;
157 }
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200158 if (trx > trx_to) {
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200159 LOGP(DRLCMAC, LOGL_NOTICE, "No PDCH available.\n");
160 return -EINVAL;
161 }
162
163
164 LOGP(DRLCMAC, LOGL_DEBUG, "Searching for first unallocated TFI: "
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200165 "TRX=%d first TS=%d\n", trx, ts);
166 if (dir == GPRS_RLCMAC_UL_TBF)
Andreas Eversbergadb2f182012-08-07 17:06:08 +0200167 tbfp = bts->trx[trx].ul_tbf;
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200168 else
Andreas Eversbergadb2f182012-08-07 17:06:08 +0200169 tbfp = bts->trx[trx].dl_tbf;
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200170 for (tfi = 0; tfi < 32; tfi++) {
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200171 if (!tbfp[tfi])
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200172 break;
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400173 }
174
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200175 if (tfi < 32) {
176 LOGP(DRLCMAC, LOGL_DEBUG, " Found TFI=%d.\n", tfi);
177 *_trx = trx;
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200178 return tfi;
179 }
180 LOGP(DRLCMAC, LOGL_NOTICE, "No TFI available.\n");
181
182 return -1;
183}
184
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200185/* starting time for assigning single slot
186 * This offset must be a multiple of 13. */
187#define AGCH_START_OFFSET 52
188
189LLIST_HEAD(gprs_rlcmac_sbas);
190
Holger Hans Peter Freyther173a2402013-10-16 21:47:45 +0200191int sba_alloc(struct gprs_rlcmac_bts *bts,
192 uint8_t *_trx, uint8_t *_ts, uint32_t *_fn, uint8_t ta)
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200193{
194
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200195 struct gprs_rlcmac_pdch *pdch;
196 struct gprs_rlcmac_sba *sba;
197 uint8_t trx, ts;
198 uint32_t fn;
199
200 sba = talloc_zero(tall_pcu_ctx, struct gprs_rlcmac_sba);
201 if (!sba)
202 return -ENOMEM;
203
204 for (trx = 0; trx < 8; trx++) {
205 for (ts = 0; ts < 8; ts++) {
206 pdch = &bts->trx[trx].pdch[ts];
207 if (!pdch->enable)
208 continue;
209 break;
210 }
211 if (ts < 8)
212 break;
213 }
214 if (trx == 8) {
215 LOGP(DRLCMAC, LOGL_NOTICE, "No PDCH available.\n");
Holger Hans Peter Freytherdcc9c392013-10-16 21:51:37 +0200216 talloc_free(sba);
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200217 return -EINVAL;
218 }
219
220 fn = (pdch->last_rts_fn + AGCH_START_OFFSET) % 2715648;
221
222 sba->trx = trx;
223 sba->ts = ts;
224 sba->fn = fn;
225 sba->ta = ta;
226
227 llist_add(&sba->list, &gprs_rlcmac_sbas);
228
229 *_trx = trx;
230 *_ts = ts;
231 *_fn = fn;
232 return 0;
233}
234
235struct gprs_rlcmac_sba *sba_find(uint8_t trx, uint8_t ts, uint32_t fn)
236{
237 struct gprs_rlcmac_sba *sba;
238
239 llist_for_each_entry(sba, &gprs_rlcmac_sbas, list) {
240 if (sba->trx == trx && sba->ts == ts && sba->fn == fn)
241 return sba;
242 }
243
244 return NULL;
245}
246
Andreas Eversberge6228b32012-07-03 13:36:03 +0200247/* received RLC/MAC block from L1 */
Holger Hans Peter Freyther698b6122013-10-17 13:44:18 +0200248int gprs_rlcmac_rcv_block(struct gprs_rlcmac_bts *bts,
249 uint8_t trx, uint8_t ts, uint8_t *data, uint8_t len,
Andreas Eversberg570b44b2013-03-16 16:15:01 +0100250 uint32_t fn, int8_t rssi)
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400251{
Andreas Eversberge6228b32012-07-03 13:36:03 +0200252 unsigned payload = data[0] >> 6;
253 bitvec *block;
254 int rc = 0;
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400255
Andreas Eversberge6228b32012-07-03 13:36:03 +0200256 switch (payload) {
257 case GPRS_RLCMAC_DATA_BLOCK:
Holger Hans Peter Freyther698b6122013-10-17 13:44:18 +0200258 rc = gprs_rlcmac_rcv_data_block_acknowledged(bts, trx, ts, data,
Andreas Eversberg570b44b2013-03-16 16:15:01 +0100259 len, rssi);
Andreas Eversberge6228b32012-07-03 13:36:03 +0200260 break;
261 case GPRS_RLCMAC_CONTROL_BLOCK:
262 block = bitvec_alloc(len);
263 if (!block)
264 return -ENOMEM;
265 bitvec_unpack(block, data);
Holger Hans Peter Freyther8d7a6322013-10-17 15:23:49 +0200266 rc = gprs_rlcmac_rcv_control_block(bts, block, trx, ts, fn);
Andreas Eversberge6228b32012-07-03 13:36:03 +0200267 bitvec_free(block);
268 break;
269 case GPRS_RLCMAC_CONTROL_BLOCK_OPT:
270 LOGP(DRLCMAC, LOGL_NOTICE, "GPRS_RLCMAC_CONTROL_BLOCK_OPT block payload is not supported.\n");
Holger Hans Peter Freyther26da8362013-10-16 16:34:09 +0200271 break;
Andreas Eversberge6228b32012-07-03 13:36:03 +0200272 default:
Holger Hans Peter Freyther70862c92013-10-16 16:37:22 +0200273 LOGP(DRLCMAC, LOGL_NOTICE, "Unknown RLCMAC block payload(%u).\n", payload);
Andreas Eversberge6228b32012-07-03 13:36:03 +0200274 rc = -EINVAL;
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400275 }
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400276
Andreas Eversberge6228b32012-07-03 13:36:03 +0200277 return rc;
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400278}
279
Andreas Eversberg2b914642012-07-19 13:06:26 +0200280/* add paging to paging queue(s) */
Holger Hans Peter Freyther1af73f62013-10-17 15:51:39 +0200281int gprs_rlcmac_add_paging(struct gprs_rlcmac_bts *bts,
282 uint8_t chan_needed, uint8_t *identity_lv)
Andreas Eversberg2b914642012-07-19 13:06:26 +0200283{
Andreas Eversberg2b914642012-07-19 13:06:26 +0200284 uint8_t l, trx, ts, any_tbf = 0;
285 struct gprs_rlcmac_tbf *tbf;
286 struct gprs_rlcmac_paging *pag;
287 uint8_t slot_mask[8];
288 int8_t first_ts; /* must be signed */
289
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200290 LOGP(DRLCMAC, LOGL_INFO, "Add RR paging: chan-needed=%d MI=%s\n",
291 chan_needed, osmo_hexdump(identity_lv + 1, identity_lv[0]));
Andreas Eversberg2b914642012-07-19 13:06:26 +0200292
293 /* collect slots to page
294 * Mark slots for every TBF, but only mark one of it.
295 * Mark only the first slot found.
296 * Don't mark, if TBF uses a different slot that is already marked. */
297 memset(slot_mask, 0, sizeof(slot_mask));
298 for (l = 0; gprs_rlcmac_tbfs_lists[l]; l++) {
299 llist_for_each_entry(tbf, gprs_rlcmac_tbfs_lists[l], list) {
300 first_ts = -1;
301 for (ts = 0; ts < 8; ts++) {
302 if (tbf->pdch[ts]) {
303 /* remember the first slot found */
304 if (first_ts < 0)
305 first_ts = ts;
306 /* break, if we already marked a slot */
Holger Hans Peter Freyther96efa702013-09-29 07:44:39 +0200307 if ((slot_mask[tbf->trx_no] & (1 << ts)))
Andreas Eversberg2b914642012-07-19 13:06:26 +0200308 break;
309 }
310 }
311 /* mark first slot found, if none is marked already */
312 if (ts == 8 && first_ts >= 0) {
313 LOGP(DRLCMAC, LOGL_DEBUG, "- %s TBF=%d uses "
314 "TRX=%d TS=%d, so we mark\n",
315 (tbf->direction == GPRS_RLCMAC_UL_TBF)
316 ? "UL" : "DL",
Holger Hans Peter Freyther96efa702013-09-29 07:44:39 +0200317 tbf->tfi, tbf->trx_no, first_ts);
318 slot_mask[tbf->trx_no] |= (1 << first_ts);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200319 } else
320 LOGP(DRLCMAC, LOGL_DEBUG, "- %s TBF=%d uses "
321 "already marked TRX=%d TS=%d\n",
322 (tbf->direction == GPRS_RLCMAC_UL_TBF)
323 ? "UL" : "DL",
Holger Hans Peter Freyther96efa702013-09-29 07:44:39 +0200324 tbf->tfi, tbf->trx_no, ts);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200325 }
326 }
327
328 /* Now we have a list of marked slots. Every TBF uses at least one
329 * of these slots. */
330
331 /* schedule paging to all marked slots */
332 for (trx = 0; trx < 8; trx++) {
333 if (slot_mask[trx] == 0)
334 continue;
335 any_tbf = 1;
336 for (ts = 0; ts < 8; ts++) {
337 if ((slot_mask[trx] & (1 << ts))) {
338 /* schedule */
Andreas Eversberg6681bb82012-07-25 08:48:44 +0200339 pag = talloc_zero(tall_pcu_ctx,
Andreas Eversberg2b914642012-07-19 13:06:26 +0200340 struct gprs_rlcmac_paging);
341 if (!pag)
342 return -ENOMEM;
343 pag->chan_needed = chan_needed;
344 memcpy(pag->identity_lv, identity_lv,
345 identity_lv[0] + 1);
346 llist_add(&pag->list,
347 &bts->trx[trx].pdch[ts].paging_list);
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200348 LOGP(DRLCMAC, LOGL_INFO, "Paging on PACCH of "
349 "TRX=%d TS=%d\n", trx, ts);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200350 }
351 }
352 }
353
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200354 if (!any_tbf)
Andreas Eversberg2b914642012-07-19 13:06:26 +0200355 LOGP(DRLCMAC, LOGL_INFO, "No paging, because no TBF\n");
Andreas Eversberg2b914642012-07-19 13:06:26 +0200356
357 return 0;
358}
359
360struct gprs_rlcmac_paging *gprs_rlcmac_dequeue_paging(
361 struct gprs_rlcmac_pdch *pdch)
362{
363 struct gprs_rlcmac_paging *pag;
364
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200365 if (llist_empty(&pdch->paging_list))
366 return NULL;
Andreas Eversberg2b914642012-07-19 13:06:26 +0200367 pag = llist_entry(pdch->paging_list.next,
368 struct gprs_rlcmac_paging, list);
369 llist_del(&pag->list);
370
371 return pag;
372}
373
374struct msgb *gprs_rlcmac_send_packet_paging_request(
375 struct gprs_rlcmac_pdch *pdch)
376{
377 struct gprs_rlcmac_paging *pag;
378 struct msgb *msg;
379 unsigned wp = 0, len;
380
381 /* no paging, no message */
382 pag = gprs_rlcmac_dequeue_paging(pdch);
383 if (!pag)
384 return NULL;
385
386 LOGP(DRLCMAC, LOGL_DEBUG, "Scheduling paging\n");
387
388 /* alloc message */
389 msg = msgb_alloc(23, "pag ctrl block");
Andreas Eversberg5cae0872012-10-07 15:08:22 +0200390 if (!msg) {
391 talloc_free(pag);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200392 return NULL;
Andreas Eversberg5cae0872012-10-07 15:08:22 +0200393 }
Andreas Eversberg2b914642012-07-19 13:06:26 +0200394 bitvec *pag_vec = bitvec_alloc(23);
395 if (!pag_vec) {
396 msgb_free(msg);
Andreas Eversberg5cae0872012-10-07 15:08:22 +0200397 talloc_free(pag);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200398 return NULL;
399 }
400 wp = write_packet_paging_request(pag_vec);
401
402 /* loop until message is full */
403 while (pag) {
404 /* try to add paging */
405 if ((pag->identity_lv[1] & 0x07) == 4) {
406 /* TMSI */
407 LOGP(DRLCMAC, LOGL_DEBUG, "- TMSI=0x%08x\n",
408 ntohl(*((uint32_t *)(pag->identity_lv + 1))));
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200409 len = 1 + 1 + 1 + 32 + 2 + 1;
Andreas Eversberg2b914642012-07-19 13:06:26 +0200410 if (pag->identity_lv[0] != 5) {
411 LOGP(DRLCMAC, LOGL_ERROR, "TMSI paging with "
412 "MI != 5 octets!\n");
Andreas Eversberg5cae0872012-10-07 15:08:22 +0200413 goto continue_next;
Andreas Eversberg2b914642012-07-19 13:06:26 +0200414 }
415 } else {
416 /* MI */
417 LOGP(DRLCMAC, LOGL_DEBUG, "- MI=%s\n",
418 osmo_hexdump(pag->identity_lv + 1,
419 pag->identity_lv[0]));
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200420 len = 1 + 1 + 1 + 4 + (pag->identity_lv[0]<<3) + 2 + 1;
Andreas Eversberg2b914642012-07-19 13:06:26 +0200421 if (pag->identity_lv[0] > 8) {
422 LOGP(DRLCMAC, LOGL_ERROR, "Paging with "
423 "MI > 8 octets!\n");
Andreas Eversberg5cae0872012-10-07 15:08:22 +0200424 goto continue_next;
Andreas Eversberg2b914642012-07-19 13:06:26 +0200425 }
426 }
427 if (wp + len > 184) {
428 LOGP(DRLCMAC, LOGL_DEBUG, "- Does not fit, so schedule "
429 "next time\n");
430 /* put back paging record, because does not fit */
431 llist_add_tail(&pag->list, &pdch->paging_list);
432 break;
433 }
434 write_repeated_page_info(pag_vec, wp, pag->identity_lv[0],
435 pag->identity_lv + 1, pag->chan_needed);
436
Andreas Eversberg5cae0872012-10-07 15:08:22 +0200437continue_next:
438 talloc_free(pag);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200439 pag = gprs_rlcmac_dequeue_paging(pdch);
440 }
441
442 bitvec_pack(pag_vec, msgb_put(msg, 23));
Andreas Eversberg6681bb82012-07-25 08:48:44 +0200443 RlcMacDownlink_t * mac_control_block = (RlcMacDownlink_t *)talloc_zero(tall_pcu_ctx, RlcMacDownlink_t);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200444 LOGP(DRLCMAC, LOGL_DEBUG, "+++++++++++++++++++++++++ TX : Packet Paging Request +++++++++++++++++++++++++\n");
445 decode_gsm_rlcmac_downlink(pag_vec, mac_control_block);
446 LOGPC(DCSN1, LOGL_NOTICE, "\n");
447 LOGP(DRLCMAC, LOGL_DEBUG, "------------------------- TX : Packet Paging Request -------------------------\n");
448 bitvec_free(pag_vec);
Andreas Eversberg6681bb82012-07-25 08:48:44 +0200449 talloc_free(mac_control_block);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200450
451 return msg;
452}
453
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400454// GSM 04.08 9.1.18 Immediate assignment
Holger Hans Peter Freytheree7a5352013-10-17 15:41:36 +0200455int write_immediate_assignment(
456 struct gprs_rlcmac_bts *bts,
457 bitvec * dest, uint8_t downlink, uint8_t ra,
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200458 uint32_t ref_fn, uint8_t ta, uint16_t arfcn, uint8_t ts, uint8_t tsc,
Andreas Eversberge6228b32012-07-03 13:36:03 +0200459 uint8_t tfi, uint8_t usf, uint32_t tlli,
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200460 uint8_t polling, uint32_t fn, uint8_t single_block, uint8_t alpha,
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200461 uint8_t gamma, int8_t ta_idx)
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400462{
463 unsigned wp = 0;
Andreas Eversberg7b045012012-07-05 07:38:49 +0200464 uint8_t plen;
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400465
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400466 bitvec_write_field(dest, wp,0x0,4); // Skip Indicator
467 bitvec_write_field(dest, wp,0x6,4); // Protocol Discriminator
468 bitvec_write_field(dest, wp,0x3F,8); // Immediate Assignment Message Type
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400469
470 // 10.5.2.25b Dedicated mode or TBF
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400471 bitvec_write_field(dest, wp,0x0,1); // spare
472 bitvec_write_field(dest, wp,0x0,1); // TMA : Two-message assignment: No meaning
473 bitvec_write_field(dest, wp,downlink,1); // Downlink : Downlink assignment to mobile in packet idle mode
474 bitvec_write_field(dest, wp,0x1,1); // T/D : TBF or dedicated mode: this message assigns a Temporary Block Flow (TBF).
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400475
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400476 bitvec_write_field(dest, wp,0x0,4); // Page Mode
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400477
478 // GSM 04.08 10.5.2.25a Packet Channel Description
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400479 bitvec_write_field(dest, wp,0x1,5); // Channel type
Andreas Eversberg0aed6542012-06-23 10:33:16 +0200480 bitvec_write_field(dest, wp,ts,3); // TN
481 bitvec_write_field(dest, wp,tsc,3); // TSC
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400482 bitvec_write_field(dest, wp,0x0,3); // non-hopping RF channel configuraion
Andreas Eversberg0aed6542012-06-23 10:33:16 +0200483 bitvec_write_field(dest, wp,arfcn,10); // ARFCN
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400484
485 //10.5.2.30 Request Reference
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400486 bitvec_write_field(dest, wp,ra,8); // RA
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200487 bitvec_write_field(dest, wp,(ref_fn / (26 * 51)) % 32,5); // T1'
488 bitvec_write_field(dest, wp,ref_fn % 51,6); // T3
489 bitvec_write_field(dest, wp,ref_fn % 26,5); // T2
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400490
491 // 10.5.2.40 Timing Advance
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400492 bitvec_write_field(dest, wp,0x0,2); // spare
493 bitvec_write_field(dest, wp,ta,6); // Timing Advance value
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400494
495 // No mobile allocation in non-hopping systems.
496 // A zero-length LV. Just write L=0.
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400497 bitvec_write_field(dest, wp,0,8);
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400498
Andreas Eversberg7b045012012-07-05 07:38:49 +0200499 if ((wp % 8)) {
500 LOGP(DRLCMACUL, LOGL_ERROR, "Length of IMM.ASS without rest "
501 "octets is not multiple of 8 bits, PLEASE FIX!\n");
502 exit (0);
503 }
504 plen = wp / 8;
505
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400506 if (downlink)
507 {
508 // GSM 04.08 10.5.2.16 IA Rest Octets
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400509 bitvec_write_field(dest, wp, 3, 2); // "HH"
510 bitvec_write_field(dest, wp, 1, 2); // "01" Packet Downlink Assignment
511 bitvec_write_field(dest, wp,tlli,32); // TLLI
512 bitvec_write_field(dest, wp,0x1,1); // switch TFI : on
513 bitvec_write_field(dest, wp,tfi,5); // TFI
514 bitvec_write_field(dest, wp,0x0,1); // RLC acknowledged mode
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200515 if (alpha) {
516 bitvec_write_field(dest, wp,0x1,1); // ALPHA = present
517 bitvec_write_field(dest, wp,alpha,4); // ALPHA
518 } else {
519 bitvec_write_field(dest, wp,0x0,1); // ALPHA = not present
520 }
521 bitvec_write_field(dest, wp,gamma,5); // GAMMA power control parameter
Andreas Eversberge6228b32012-07-03 13:36:03 +0200522 bitvec_write_field(dest, wp,polling,1); // Polling Bit
523 bitvec_write_field(dest, wp,!polling,1); // TA_VALID ???
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200524 if (ta_idx < 0) {
525 bitvec_write_field(dest, wp,0x0,1); // switch TIMING_ADVANCE_INDEX = off
526 } else {
527 bitvec_write_field(dest, wp,0x1,1); // switch TIMING_ADVANCE_INDEX = on
528 bitvec_write_field(dest, wp,ta_idx,4); // TIMING_ADVANCE_INDEX
529 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200530 if (polling) {
531 bitvec_write_field(dest, wp,0x1,1); // TBF Starting TIME present
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200532 bitvec_write_field(dest, wp,(fn / (26 * 51)) % 32,5); // T1'
533 bitvec_write_field(dest, wp,fn % 51,6); // T3
534 bitvec_write_field(dest, wp,fn % 26,5); // T2
Andreas Eversberge6228b32012-07-03 13:36:03 +0200535 } else {
536 bitvec_write_field(dest, wp,0x0,1); // TBF Starting TIME present
537 }
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400538 bitvec_write_field(dest, wp,0x0,1); // P0 not present
Andreas Eversberge6228b32012-07-03 13:36:03 +0200539// bitvec_write_field(dest, wp,0x1,1); // P0 not present
540// bitvec_write_field(dest, wp,0xb,4);
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400541 }
542 else
543 {
544 // GMS 04.08 10.5.2.37b 10.5.2.16
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400545 bitvec_write_field(dest, wp, 3, 2); // "HH"
546 bitvec_write_field(dest, wp, 0, 2); // "0" Packet Uplink Assignment
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200547 if (single_block) {
548 bitvec_write_field(dest, wp, 0, 1); // Block Allocation : Single Block Allocation
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200549 if (alpha) {
550 bitvec_write_field(dest, wp,0x1,1); // ALPHA = present
551 bitvec_write_field(dest, wp,alpha,4); // ALPHA = present
552 } else
553 bitvec_write_field(dest, wp,0x0,1); // ALPHA = not present
554 bitvec_write_field(dest, wp,gamma,5); // GAMMA power control parameter
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200555 if (ta_idx < 0) {
556 bitvec_write_field(dest, wp,0x0,1); // switch TIMING_ADVANCE_INDEX = off
557 } else {
558 bitvec_write_field(dest, wp,0x1,1); // switch TIMING_ADVANCE_INDEX = on
559 bitvec_write_field(dest, wp,ta_idx,4); // TIMING_ADVANCE_INDEX
560 }
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200561 bitvec_write_field(dest, wp, 1, 1); // TBF_STARTING_TIME_FLAG
562 bitvec_write_field(dest, wp,(fn / (26 * 51)) % 32,5); // T1'
563 bitvec_write_field(dest, wp,fn % 51,6); // T3
564 bitvec_write_field(dest, wp,fn % 26,5); // T2
565 } else {
566 bitvec_write_field(dest, wp, 1, 1); // Block Allocation : Not Single Block Allocation
567 bitvec_write_field(dest, wp, tfi, 5); // TFI_ASSIGNMENT Temporary Flow Identity
568 bitvec_write_field(dest, wp, 0, 1); // POLLING
569 bitvec_write_field(dest, wp, 0, 1); // ALLOCATION_TYPE: dynamic
570 bitvec_write_field(dest, wp, usf, 3); // USF
571 bitvec_write_field(dest, wp, 0, 1); // USF_GRANULARITY
572 bitvec_write_field(dest, wp, 0, 1); // "0" power control: Not Present
Andreas Eversberg499ff412012-10-03 14:21:36 +0200573 bitvec_write_field(dest, wp, bts->initial_cs_ul-1, 2); // CHANNEL_CODING_COMMAND
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200574 bitvec_write_field(dest, wp, 1, 1); // TLLI_BLOCK_CHANNEL_CODING
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200575 if (alpha) {
576 bitvec_write_field(dest, wp,0x1,1); // ALPHA = present
577 bitvec_write_field(dest, wp,alpha,4); // ALPHA
578 } else
579 bitvec_write_field(dest, wp,0x0,1); // ALPHA = not present
580 bitvec_write_field(dest, wp,gamma,5); // GAMMA power control parameter
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200581 /* note: there is no choise for TAI and no starting time */
582 bitvec_write_field(dest, wp, 0, 1); // switch TIMING_ADVANCE_INDEX = off
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200583 bitvec_write_field(dest, wp, 0, 1); // TBF_STARTING_TIME_FLAG
584 }
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400585 }
586
Andreas Eversberg7b045012012-07-05 07:38:49 +0200587 return plen;
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400588}
589
Andreas Eversberge6228b32012-07-03 13:36:03 +0200590/* generate uplink assignment */
Holger Hans Peter Freyther485860c2013-10-17 15:38:23 +0200591void write_packet_uplink_assignment(
592 struct gprs_rlcmac_bts *bts,
593 bitvec * dest, uint8_t old_tfi,
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200594 uint8_t old_downlink, uint32_t tlli, uint8_t use_tlli,
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200595 struct gprs_rlcmac_tbf *tbf, uint8_t poll, uint8_t alpha,
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200596 uint8_t gamma, int8_t ta_idx)
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400597{
Andreas Eversberge6228b32012-07-03 13:36:03 +0200598 // TODO We should use our implementation of encode RLC/MAC Control messages.
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400599 unsigned wp = 0;
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200600 uint8_t ts;
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400601
Andreas Eversberge6228b32012-07-03 13:36:03 +0200602 bitvec_write_field(dest, wp,0x1,2); // Payload Type
603 bitvec_write_field(dest, wp,0x0,2); // Uplink block with TDMA framenumber (N+13)
604 bitvec_write_field(dest, wp,poll,1); // Suppl/Polling Bit
605 bitvec_write_field(dest, wp,0x0,3); // Uplink state flag
606 bitvec_write_field(dest, wp,0xa,6); // MESSAGE TYPE
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400607
Andreas Eversberge6228b32012-07-03 13:36:03 +0200608 bitvec_write_field(dest, wp,0x0,2); // Page Mode
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400609
Andreas Eversberge6228b32012-07-03 13:36:03 +0200610 bitvec_write_field(dest, wp,0x0,1); // switch PERSIST_LEVEL: off
611 if (use_tlli) {
612 bitvec_write_field(dest, wp,0x2,2); // switch TLLI : on
613 bitvec_write_field(dest, wp,tlli,32); // TLLI
614 } else {
615 bitvec_write_field(dest, wp,0x0,1); // switch TFI : on
616 bitvec_write_field(dest, wp,old_downlink,1); // 0=UPLINK TFI, 1=DL TFI
617 bitvec_write_field(dest, wp,old_tfi,5); // TFI
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400618 }
Ivan Kluchnikovb2b81002012-06-04 21:57:02 +0400619
Andreas Eversberge6228b32012-07-03 13:36:03 +0200620 bitvec_write_field(dest, wp,0x0,1); // Message escape
Andreas Eversberg499ff412012-10-03 14:21:36 +0200621 bitvec_write_field(dest, wp,bts->initial_cs_ul-1, 2); // CHANNEL_CODING_COMMAND
Andreas Eversbergdfa563c2012-07-06 08:13:59 +0200622 bitvec_write_field(dest, wp,0x1,1); // TLLI_BLOCK_CHANNEL_CODING
Andreas Eversberge6228b32012-07-03 13:36:03 +0200623 bitvec_write_field(dest, wp,0x1,1); // switch TIMING_ADVANCE_VALUE = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200624 bitvec_write_field(dest, wp,tbf->ta,6); // TIMING_ADVANCE_VALUE
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200625 if (ta_idx < 0) {
626 bitvec_write_field(dest, wp,0x0,1); // switch TIMING_ADVANCE_INDEX = off
627 } else {
628 bitvec_write_field(dest, wp,0x1,1); // switch TIMING_ADVANCE_INDEX = on
629 bitvec_write_field(dest, wp,ta_idx,4); // TIMING_ADVANCE_INDEX
630 }
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400631
Andreas Eversberge6228b32012-07-03 13:36:03 +0200632#if 1
633 bitvec_write_field(dest, wp,0x1,1); // Frequency Parameters information elements = present
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200634 bitvec_write_field(dest, wp,tbf->tsc,3); // Training Sequence Code (TSC)
Andreas Eversberge6228b32012-07-03 13:36:03 +0200635 bitvec_write_field(dest, wp,0x0,2); // ARFCN = present
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200636 bitvec_write_field(dest, wp,tbf->arfcn,10); // ARFCN
Andreas Eversberge6228b32012-07-03 13:36:03 +0200637#else
638 bitvec_write_field(dest, wp,0x0,1); // Frequency Parameters = off
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200639#endif
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400640
Andreas Eversberge6228b32012-07-03 13:36:03 +0200641 bitvec_write_field(dest, wp,0x1,2); // Dynamic Allocation
Ivan Kluchnikove3594232012-06-07 01:12:29 +0400642
Andreas Eversberge6228b32012-07-03 13:36:03 +0200643 bitvec_write_field(dest, wp,0x0,1); // Extended Dynamic Allocation = off
644 bitvec_write_field(dest, wp,0x0,1); // P0 = off
Ivan Kluchnikove3594232012-06-07 01:12:29 +0400645
Andreas Eversberge6228b32012-07-03 13:36:03 +0200646 bitvec_write_field(dest, wp,0x0,1); // USF_GRANULARITY
647 bitvec_write_field(dest, wp,0x1,1); // switch TFI : on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200648 bitvec_write_field(dest, wp,tbf->tfi,5);// TFI
Andreas Eversberge6228b32012-07-03 13:36:03 +0200649
650 bitvec_write_field(dest, wp,0x0,1); //
651 bitvec_write_field(dest, wp,0x0,1); // TBF Starting Time = off
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200652 if (alpha || gamma) {
653 bitvec_write_field(dest, wp,0x1,1); // Timeslot Allocation with Power Control
654 bitvec_write_field(dest, wp,alpha,4); // ALPHA
655 } else
656 bitvec_write_field(dest, wp,0x0,1); // Timeslot Allocation
Andreas Eversberge6228b32012-07-03 13:36:03 +0200657
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200658 for (ts = 0; ts < 8; ts++) {
659 if (tbf->pdch[ts]) {
Andreas Eversberge6228b32012-07-03 13:36:03 +0200660 bitvec_write_field(dest, wp,0x1,1); // USF_TN(i): on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200661 bitvec_write_field(dest, wp,tbf->dir.ul.usf[ts],3); // USF_TN(i)
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200662 if (alpha || gamma)
663 bitvec_write_field(dest, wp,gamma,5); // GAMMA power control parameter
Andreas Eversberge6228b32012-07-03 13:36:03 +0200664 } else
665 bitvec_write_field(dest, wp,0x0,1); // USF_TN(i): off
Ivan Kluchnikove3594232012-06-07 01:12:29 +0400666 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200667// bitvec_write_field(dest, wp,0x0,1); // Measurement Mapping struct not present
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400668}
669
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400670
Andreas Eversberge6228b32012-07-03 13:36:03 +0200671/* generate downlink assignment */
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400672void write_packet_downlink_assignment(RlcMacDownlink_t * block, uint8_t old_tfi,
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200673 uint8_t old_downlink, struct gprs_rlcmac_tbf *tbf, uint8_t poll,
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200674 uint8_t alpha, uint8_t gamma, int8_t ta_idx, uint8_t ta_ts)
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400675{
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400676 // Packet downlink assignment TS 44.060 11.2.7
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400677
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200678 uint8_t tn;
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400679
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400680 block->PAYLOAD_TYPE = 0x1; // RLC/MAC control block that does not include the optional octets of the RLC/MAC control header
681 block->RRBP = 0x0; // N+13
682 block->SP = poll; // RRBP field is valid
683 block->USF = 0x0; // Uplink state flag
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400684
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400685 block->u.Packet_Downlink_Assignment.MESSAGE_TYPE = 0x2; // Packet Downlink Assignment
686 block->u.Packet_Downlink_Assignment.PAGE_MODE = 0x0; // Normal Paging
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400687
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400688 block->u.Packet_Downlink_Assignment.Exist_PERSISTENCE_LEVEL = 0x0; // PERSISTENCE_LEVEL: off
Andreas Eversberge6228b32012-07-03 13:36:03 +0200689
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400690 block->u.Packet_Downlink_Assignment.ID.UnionType = 0x0; // TFI = on
691 block->u.Packet_Downlink_Assignment.ID.u.Global_TFI.UnionType = old_downlink; // 0=UPLINK TFI, 1=DL TFI
692 block->u.Packet_Downlink_Assignment.ID.u.Global_TFI.u.UPLINK_TFI = old_tfi; // TFI
Andreas Eversberge6228b32012-07-03 13:36:03 +0200693
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400694 block->u.Packet_Downlink_Assignment.MAC_MODE = 0x0; // Dynamic Allocation
695 block->u.Packet_Downlink_Assignment.RLC_MODE = 0x0; // RLC acknowledged mode
696 block->u.Packet_Downlink_Assignment.CONTROL_ACK = old_downlink; // NW establishes no new DL TBF for the MS with running timer T3192
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200697 block->u.Packet_Downlink_Assignment.TIMESLOT_ALLOCATION = 0; // timeslot(s)
698 for (tn = 0; tn < 8; tn++) {
699 if (tbf->pdch[tn])
700 block->u.Packet_Downlink_Assignment.TIMESLOT_ALLOCATION |= 0x80 >> tn; // timeslot(s)
701 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200702
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400703 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.Exist_TIMING_ADVANCE_VALUE = 0x1; // TIMING_ADVANCE_VALUE = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200704 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.TIMING_ADVANCE_VALUE = tbf->ta; // TIMING_ADVANCE_VALUE
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200705 if (ta_idx < 0) {
706 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.Exist_IndexAndtimeSlot = 0x0; // TIMING_ADVANCE_INDEX = off
707 } else {
708 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.Exist_IndexAndtimeSlot = 0x1; // TIMING_ADVANCE_INDEX = on
709 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.TIMING_ADVANCE_INDEX = ta_idx; // TIMING_ADVANCE_INDEX
710 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.TIMING_ADVANCE_TIMESLOT_NUMBER = ta_ts; // TIMING_ADVANCE_TS
711 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200712
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400713 block->u.Packet_Downlink_Assignment.Exist_P0_and_BTS_PWR_CTRL_MODE = 0x0; // POWER CONTROL = off
Andreas Eversberge6228b32012-07-03 13:36:03 +0200714
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400715 block->u.Packet_Downlink_Assignment.Exist_Frequency_Parameters = 0x1; // Frequency Parameters = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200716 block->u.Packet_Downlink_Assignment.Frequency_Parameters.TSC = tbf->tsc; // Training Sequence Code (TSC)
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400717 block->u.Packet_Downlink_Assignment.Frequency_Parameters.UnionType = 0x0; // ARFCN = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200718 block->u.Packet_Downlink_Assignment.Frequency_Parameters.u.ARFCN = tbf->arfcn; // ARFCN
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400719
720 block->u.Packet_Downlink_Assignment.Exist_DOWNLINK_TFI_ASSIGNMENT = 0x1; // DOWNLINK TFI ASSIGNMENT = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200721 block->u.Packet_Downlink_Assignment.DOWNLINK_TFI_ASSIGNMENT = tbf->tfi; // TFI
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400722
723 block->u.Packet_Downlink_Assignment.Exist_Power_Control_Parameters = 0x1; // Power Control Parameters = on
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200724 block->u.Packet_Downlink_Assignment.Power_Control_Parameters.ALPHA = alpha; // ALPHA
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400725
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200726 for (tn = 0; tn < 8; tn++)
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400727 {
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200728 if (tbf->pdch[tn])
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400729 {
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200730 block->u.Packet_Downlink_Assignment.Power_Control_Parameters.Slot[tn].Exist = 0x1; // Slot[i] = on
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200731 block->u.Packet_Downlink_Assignment.Power_Control_Parameters.Slot[tn].GAMMA_TN = gamma; // GAMMA_TN
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400732 }
733 else
734 {
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200735 block->u.Packet_Downlink_Assignment.Power_Control_Parameters.Slot[tn].Exist = 0x0; // Slot[i] = off
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400736 }
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400737 }
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400738
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400739 block->u.Packet_Downlink_Assignment.Exist_TBF_Starting_Time = 0x0; // TBF Starting TIME = off
740 block->u.Packet_Downlink_Assignment.Exist_Measurement_Mapping = 0x0; // Measurement_Mapping = off
741 block->u.Packet_Downlink_Assignment.Exist_AdditionsR99 = 0x0; // AdditionsR99 = off
Andreas Eversberge6228b32012-07-03 13:36:03 +0200742}
743
Andreas Eversberg8c3680d2012-10-08 12:30:56 +0200744/* generate paging request */
745int write_paging_request(bitvec * dest, uint8_t *ptmsi, uint16_t ptmsi_len)
746{
747 unsigned wp = 0;
748 int plen;
749
750 bitvec_write_field(dest, wp,0x0,4); // Skip Indicator
751 bitvec_write_field(dest, wp,0x6,4); // Protocol Discriminator
752 bitvec_write_field(dest, wp,0x21,8); // Paging Request Message Type
753
754 bitvec_write_field(dest, wp,0x0,4); // Page Mode
755 bitvec_write_field(dest, wp,0x0,4); // Channel Needed
756
757 // Mobile Identity
758 bitvec_write_field(dest, wp,ptmsi_len+1,8); // Mobile Identity length
759 bitvec_write_field(dest, wp,0xf,4); // unused
760 bitvec_write_field(dest, wp,0x4,4); // PTMSI type
761 for (int i = 0; i < ptmsi_len; i++)
762 {
763 bitvec_write_field(dest, wp,ptmsi[i],8); // PTMSI
764 }
765 if ((wp % 8)) {
766 LOGP(DRLCMACUL, LOGL_ERROR, "Length of PAG.REQ without rest "
767 "octets is not multiple of 8 bits, PLEASE FIX!\n");
768 exit (0);
769 }
770 plen = wp / 8;
771 bitvec_write_field(dest, wp,0x0,1); // "L" NLN(PCH) = off
772 bitvec_write_field(dest, wp,0x0,1); // "L" Priority1 = off
773 bitvec_write_field(dest, wp,0x1,1); // "L" Priority2 = off
774 bitvec_write_field(dest, wp,0x0,1); // "L" Group Call information = off
775 bitvec_write_field(dest, wp,0x0,1); // "H" Packet Page Indication 1 = packet paging procedure
776 bitvec_write_field(dest, wp,0x1,1); // "H" Packet Page Indication 2 = packet paging procedure
777
778 return plen;
779}
780
Andreas Eversberge6228b32012-07-03 13:36:03 +0200781/* generate uplink ack */
Holger Hans Peter Freytherae947fc2013-10-17 13:29:03 +0200782void write_packet_uplink_ack(struct gprs_rlcmac_bts *bts,
783 RlcMacDownlink_t * block, struct gprs_rlcmac_tbf *tbf,
Andreas Eversberge6228b32012-07-03 13:36:03 +0200784 uint8_t final)
785{
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400786 // Packet Uplink Ack/Nack TS 44.060 11.2.28
787
Andreas Eversberge6228b32012-07-03 13:36:03 +0200788 char show_v_n[65];
789
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400790 uint8_t rbb = 0;
Andreas Eversberge6228b32012-07-03 13:36:03 +0200791 uint16_t i, bbn;
792 uint16_t mod_sns_half = (tbf->sns >> 1) - 1;
793 char bit;
794
795 LOGP(DRLCMACUL, LOGL_DEBUG, "Sending Ack/Nack for TBF=%d "
796 "(final=%d)\n", tbf->tfi, final);
797
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400798 block->PAYLOAD_TYPE = 0x1; // RLC/MAC control block that does not include the optional octets of the RLC/MAC control header
799 block->RRBP = 0x0; // N+13
800 block->SP = final; // RRBP field is valid, if it is final ack
801 block->USF = 0x0; // Uplink state flag
Andreas Eversberge6228b32012-07-03 13:36:03 +0200802
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400803 block->u.Packet_Uplink_Ack_Nack.MESSAGE_TYPE = 0x9; // Packet Downlink Assignment
804 block->u.Packet_Uplink_Ack_Nack.PAGE_MODE = 0x0; // Normal Paging
805 block->u.Packet_Uplink_Ack_Nack.UPLINK_TFI = tbf->tfi; // Uplink TFI
Andreas Eversberge6228b32012-07-03 13:36:03 +0200806
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400807 block->u.Packet_Uplink_Ack_Nack.UnionType = 0x0; // PU_AckNack_GPRS = on
Andreas Eversberg499ff412012-10-03 14:21:36 +0200808 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.CHANNEL_CODING_COMMAND = bts->initial_cs_ul - 1; // CS1
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400809 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Ack_Nack_Description.FINAL_ACK_INDICATION = final; // FINAL ACK INDICATION
810 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Ack_Nack_Description.STARTING_SEQUENCE_NUMBER = tbf->dir.ul.v_r; // STARTING_SEQUENCE_NUMBER
Andreas Eversberge6228b32012-07-03 13:36:03 +0200811 // RECEIVE_BLOCK_BITMAP
812 for (i = 0, bbn = (tbf->dir.ul.v_r - 64) & mod_sns_half; i < 64;
813 i++, bbn = (bbn + 1) & mod_sns_half) {
814 bit = tbf->dir.ul.v_n[bbn];
815 if (bit == 0)
816 bit = ' ';
817 show_v_n[i] = bit;
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400818 if (bit == 'R')
819 rbb = (rbb << 1)|1;
820 else
821 rbb = (rbb << 1);
822 if((i%8) == 7)
823 {
824 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Ack_Nack_Description.RECEIVED_BLOCK_BITMAP[i/8] = rbb;
825 rbb = 0;
826 }
Ivan Kluchnikove3594232012-06-07 01:12:29 +0400827 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200828 show_v_n[64] = '\0';
829 LOGP(DRLCMACUL, LOGL_DEBUG, "- V(N): \"%s\" R=Received "
830 "N=Not-Received\n", show_v_n);
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400831
832 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.UnionType = 0x0; // Fixed Allocation Dummy = on
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400833 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.u.FixedAllocationDummy = 0x0; // Fixed Allocation Dummy
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400834 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Exist_AdditionsR99 = 0x0; // AdditionsR99 = off
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400835
836 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Common_Uplink_Ack_Nack_Data.Exist_CONTENTION_RESOLUTION_TLLI = 0x1;
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400837 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Common_Uplink_Ack_Nack_Data.CONTENTION_RESOLUTION_TLLI = tbf->tlli;
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400838 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Common_Uplink_Ack_Nack_Data.Exist_Packet_Timing_Advance = 0x0;
839 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Common_Uplink_Ack_Nack_Data.Exist_Extension_Bits = 0x0;
840 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Common_Uplink_Ack_Nack_Data.Exist_Power_Control_Parameters = 0x0;
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400841}
842
Andreas Eversberg2b914642012-07-19 13:06:26 +0200843unsigned write_packet_paging_request(bitvec * dest)
844{
845 unsigned wp = 0;
846
847 bitvec_write_field(dest, wp,0x1,2); // Payload Type
848 bitvec_write_field(dest, wp,0x0,3); // No polling
849 bitvec_write_field(dest, wp,0x0,3); // Uplink state flag
850 bitvec_write_field(dest, wp,0x22,6); // MESSAGE TYPE
851
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200852 bitvec_write_field(dest, wp,0x0,2); // Page Mode
853
Andreas Eversberg2b914642012-07-19 13:06:26 +0200854 bitvec_write_field(dest, wp,0x0,1); // No PERSISTENCE_LEVEL
855 bitvec_write_field(dest, wp,0x0,1); // No NLN
856
857 return wp;
858}
859
860unsigned write_repeated_page_info(bitvec * dest, unsigned& wp, uint8_t len,
861 uint8_t *identity, uint8_t chan_needed)
862{
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200863 bitvec_write_field(dest, wp,0x1,1); // Repeated Page info exists
864
Andreas Eversberg2b914642012-07-19 13:06:26 +0200865 bitvec_write_field(dest, wp,0x1,1); // RR connection paging
866
867 if ((identity[0] & 0x07) == 4) {
868 bitvec_write_field(dest, wp,0x0,1); // TMSI
869 identity++;
870 len--;
871 } else {
872 bitvec_write_field(dest, wp,0x0,1); // MI
873 bitvec_write_field(dest, wp,len,4); // MI len
874 }
875 while (len) {
876 bitvec_write_field(dest, wp,*identity++,8); // MI data
877 len--;
878 }
879 bitvec_write_field(dest, wp,chan_needed,2); // CHANNEL_NEEDED
880 bitvec_write_field(dest, wp,0x0,1); // No eMLPP_PRIORITY
881
882 return wp;
883}
884
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400885/* Send Uplink unit-data to SGSN. */
Andreas Eversberg3e372d52012-07-06 09:28:15 +0200886int gprs_rlcmac_tx_ul_ud(gprs_rlcmac_tbf *tbf)
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400887{
Andreas Eversberg9a913462012-09-23 06:42:38 +0200888 uint8_t qos_profile[3];
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400889 struct msgb *llc_pdu;
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200890 unsigned msg_len = NS_HDR_LEN + BSSGP_HDR_LEN + tbf->llc_index;
Holger Hans Peter Freyther90692f92013-07-13 12:51:16 +0200891 struct bssgp_bvc_ctx *bctx = gprs_bssgp_pcu_current_bctx();
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400892
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200893 LOGP(DBSSGP, LOGL_INFO, "LLC [PCU -> SGSN] TFI: %u TLLI: 0x%08x len=%d\n", tbf->tfi, tbf->tlli, tbf->llc_index);
Andreas Eversberg3e372d52012-07-06 09:28:15 +0200894 if (!bctx) {
895 LOGP(DBSSGP, LOGL_ERROR, "No bctx\n");
896 return -EIO;
897 }
Ivan Kluchnikovc320d862012-03-18 15:04:48 +0400898
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400899 llc_pdu = msgb_alloc_headroom(msg_len, msg_len,"llc_pdu");
Andreas Eversberg9a913462012-09-23 06:42:38 +0200900 uint8_t *buf = msgb_push(llc_pdu, TL16V_GROSS_LEN(sizeof(uint8_t)*tbf->llc_index));
901 tl16v_put(buf, BSSGP_IE_LLC_PDU, sizeof(uint8_t)*tbf->llc_index, tbf->llc_frame);
902 qos_profile[0] = QOS_PROFILE >> 16;
903 qos_profile[1] = QOS_PROFILE >> 8;
904 qos_profile[2] = QOS_PROFILE;
905 bssgp_tx_ul_ud(bctx, tbf->tlli, qos_profile, llc_pdu);
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400906
Andreas Eversberg3e372d52012-07-06 09:28:15 +0200907 return 0;
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400908}
Andreas Eversberg8c3680d2012-10-08 12:30:56 +0200909
910int gprs_rlcmac_paging_request(uint8_t *ptmsi, uint16_t ptmsi_len,
911 const char *imsi)
912{
913 LOGP(DRLCMAC, LOGL_NOTICE, "TX: [PCU -> BTS] Paging Request (CCCH)\n");
914 bitvec *paging_request = bitvec_alloc(23);
915 bitvec_unhex(paging_request, "2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b");
916 int plen = write_paging_request(paging_request, ptmsi, ptmsi_len);
917 pcu_l1if_tx_pch(paging_request, plen, (char *)imsi);
918 bitvec_free(paging_request);
919
920 return 0;
921}
Andreas Eversberga004e6a2013-05-13 16:45:21 +0200922
923
924/*
925 * timing advance memory
926 */
927
928/* enable to debug timing advance memory */
929//#define DEBUG_TA
930
931static LLIST_HEAD(gprs_rlcmac_ta_list);
932static int gprs_rlcmac_ta_num = 0;
933
934struct gprs_rlcmac_ta {
935 struct llist_head list;
936 uint32_t tlli;
937 uint8_t ta;
938};
939
940/* remember timing advance of a given TLLI */
941int remember_timing_advance(uint32_t tlli, uint8_t ta)
942{
943 struct gprs_rlcmac_ta *ta_entry;
944
945 /* check for existing entry */
946 llist_for_each_entry(ta_entry, &gprs_rlcmac_ta_list, list) {
947 if (ta_entry->tlli == tlli) {
948#ifdef DEBUG_TA
949 fprintf(stderr, "update %08x %d\n", tlli, ta);
950#endif
951 ta_entry->ta = ta;
952 /* relink to end of list */
953 llist_del(&ta_entry->list);
954 llist_add_tail(&ta_entry->list, &gprs_rlcmac_ta_list);
955 return 0;
956 }
957 }
958
959#ifdef DEBUG_TA
960 fprintf(stderr, "remember %08x %d\n", tlli, ta);
961#endif
962 /* if list is full, remove oldest entry */
963 if (gprs_rlcmac_ta_num == 30) {
964 ta_entry = llist_entry(gprs_rlcmac_ta_list.next,
965 struct gprs_rlcmac_ta, list);
966 llist_del(&ta_entry->list);
967 talloc_free(ta_entry);
968 gprs_rlcmac_ta_num--;
969 }
970
971 /* create new TA entry */
972 ta_entry = talloc_zero(tall_pcu_ctx, struct gprs_rlcmac_ta);
973 if (!ta_entry)
974 return -ENOMEM;
975
976 ta_entry->tlli = tlli;
977 ta_entry->ta = ta;
978 llist_add_tail(&ta_entry->list, &gprs_rlcmac_ta_list);
979 gprs_rlcmac_ta_num++;
980
981 return 0;
982}
983
984int recall_timing_advance(uint32_t tlli)
985{
986 struct gprs_rlcmac_ta *ta_entry;
987 uint8_t ta;
988
989 llist_for_each_entry(ta_entry, &gprs_rlcmac_ta_list, list) {
990 if (ta_entry->tlli == tlli) {
991 ta = ta_entry->ta;
992#ifdef DEBUG_TA
993 fprintf(stderr, "recall %08x %d\n", tlli, ta);
994#endif
995 return ta;
996 }
997 }
998#ifdef DEBUG_TA
999 fprintf(stderr, "no entry for %08x\n", tlli);
1000#endif
1001
1002 return -EINVAL;
1003}
1004
1005int flush_timing_advance(void)
1006{
1007 struct gprs_rlcmac_ta *ta_entry;
1008 int count = 0;
1009
1010 while (!llist_empty(&gprs_rlcmac_ta_list)) {
1011 ta_entry = llist_entry(gprs_rlcmac_ta_list.next,
1012 struct gprs_rlcmac_ta, list);
1013#ifdef DEBUG_TA
1014 fprintf(stderr, "flush entry %08x %d\n", ta_entry->tlli,
1015 ta_entry->ta);
1016#endif
1017 llist_del(&ta_entry->list);
1018 talloc_free(ta_entry);
1019 count++;
1020 }
1021 gprs_rlcmac_ta_num = 0;
1022
1023 return count;
1024}
1025