blob: 419470b635bfda0382cee65eb0e1673266cf94fa [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) */
281int gprs_rlcmac_add_paging(uint8_t chan_needed, uint8_t *identity_lv)
282{
283 struct gprs_rlcmac_bts *bts = gprs_rlcmac_bts;
284 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
Andreas Eversberge6228b32012-07-03 13:36:03 +0200455int write_immediate_assignment(bitvec * dest, uint8_t downlink, uint8_t ra,
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200456 uint32_t ref_fn, uint8_t ta, uint16_t arfcn, uint8_t ts, uint8_t tsc,
Andreas Eversberge6228b32012-07-03 13:36:03 +0200457 uint8_t tfi, uint8_t usf, uint32_t tlli,
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200458 uint8_t polling, uint32_t fn, uint8_t single_block, uint8_t alpha,
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200459 uint8_t gamma, int8_t ta_idx)
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400460{
461 unsigned wp = 0;
Andreas Eversberg7b045012012-07-05 07:38:49 +0200462 uint8_t plen;
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400463
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400464 bitvec_write_field(dest, wp,0x0,4); // Skip Indicator
465 bitvec_write_field(dest, wp,0x6,4); // Protocol Discriminator
466 bitvec_write_field(dest, wp,0x3F,8); // Immediate Assignment Message Type
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400467
468 // 10.5.2.25b Dedicated mode or TBF
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400469 bitvec_write_field(dest, wp,0x0,1); // spare
470 bitvec_write_field(dest, wp,0x0,1); // TMA : Two-message assignment: No meaning
471 bitvec_write_field(dest, wp,downlink,1); // Downlink : Downlink assignment to mobile in packet idle mode
472 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 +0400473
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400474 bitvec_write_field(dest, wp,0x0,4); // Page Mode
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400475
476 // GSM 04.08 10.5.2.25a Packet Channel Description
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400477 bitvec_write_field(dest, wp,0x1,5); // Channel type
Andreas Eversberg0aed6542012-06-23 10:33:16 +0200478 bitvec_write_field(dest, wp,ts,3); // TN
479 bitvec_write_field(dest, wp,tsc,3); // TSC
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400480 bitvec_write_field(dest, wp,0x0,3); // non-hopping RF channel configuraion
Andreas Eversberg0aed6542012-06-23 10:33:16 +0200481 bitvec_write_field(dest, wp,arfcn,10); // ARFCN
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400482
483 //10.5.2.30 Request Reference
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400484 bitvec_write_field(dest, wp,ra,8); // RA
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200485 bitvec_write_field(dest, wp,(ref_fn / (26 * 51)) % 32,5); // T1'
486 bitvec_write_field(dest, wp,ref_fn % 51,6); // T3
487 bitvec_write_field(dest, wp,ref_fn % 26,5); // T2
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400488
489 // 10.5.2.40 Timing Advance
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400490 bitvec_write_field(dest, wp,0x0,2); // spare
491 bitvec_write_field(dest, wp,ta,6); // Timing Advance value
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400492
493 // No mobile allocation in non-hopping systems.
494 // A zero-length LV. Just write L=0.
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400495 bitvec_write_field(dest, wp,0,8);
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400496
Andreas Eversberg7b045012012-07-05 07:38:49 +0200497 if ((wp % 8)) {
498 LOGP(DRLCMACUL, LOGL_ERROR, "Length of IMM.ASS without rest "
499 "octets is not multiple of 8 bits, PLEASE FIX!\n");
500 exit (0);
501 }
502 plen = wp / 8;
503
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400504 if (downlink)
505 {
506 // GSM 04.08 10.5.2.16 IA Rest Octets
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400507 bitvec_write_field(dest, wp, 3, 2); // "HH"
508 bitvec_write_field(dest, wp, 1, 2); // "01" Packet Downlink Assignment
509 bitvec_write_field(dest, wp,tlli,32); // TLLI
510 bitvec_write_field(dest, wp,0x1,1); // switch TFI : on
511 bitvec_write_field(dest, wp,tfi,5); // TFI
512 bitvec_write_field(dest, wp,0x0,1); // RLC acknowledged mode
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200513 if (alpha) {
514 bitvec_write_field(dest, wp,0x1,1); // ALPHA = present
515 bitvec_write_field(dest, wp,alpha,4); // ALPHA
516 } else {
517 bitvec_write_field(dest, wp,0x0,1); // ALPHA = not present
518 }
519 bitvec_write_field(dest, wp,gamma,5); // GAMMA power control parameter
Andreas Eversberge6228b32012-07-03 13:36:03 +0200520 bitvec_write_field(dest, wp,polling,1); // Polling Bit
521 bitvec_write_field(dest, wp,!polling,1); // TA_VALID ???
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200522 if (ta_idx < 0) {
523 bitvec_write_field(dest, wp,0x0,1); // switch TIMING_ADVANCE_INDEX = off
524 } else {
525 bitvec_write_field(dest, wp,0x1,1); // switch TIMING_ADVANCE_INDEX = on
526 bitvec_write_field(dest, wp,ta_idx,4); // TIMING_ADVANCE_INDEX
527 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200528 if (polling) {
529 bitvec_write_field(dest, wp,0x1,1); // TBF Starting TIME present
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200530 bitvec_write_field(dest, wp,(fn / (26 * 51)) % 32,5); // T1'
531 bitvec_write_field(dest, wp,fn % 51,6); // T3
532 bitvec_write_field(dest, wp,fn % 26,5); // T2
Andreas Eversberge6228b32012-07-03 13:36:03 +0200533 } else {
534 bitvec_write_field(dest, wp,0x0,1); // TBF Starting TIME present
535 }
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400536 bitvec_write_field(dest, wp,0x0,1); // P0 not present
Andreas Eversberge6228b32012-07-03 13:36:03 +0200537// bitvec_write_field(dest, wp,0x1,1); // P0 not present
538// bitvec_write_field(dest, wp,0xb,4);
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400539 }
540 else
541 {
Andreas Eversbergdfa563c2012-07-06 08:13:59 +0200542 struct gprs_rlcmac_bts *bts = gprs_rlcmac_bts;
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400543 // GMS 04.08 10.5.2.37b 10.5.2.16
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400544 bitvec_write_field(dest, wp, 3, 2); // "HH"
545 bitvec_write_field(dest, wp, 0, 2); // "0" Packet Uplink Assignment
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200546 if (single_block) {
547 bitvec_write_field(dest, wp, 0, 1); // Block Allocation : Single Block Allocation
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200548 if (alpha) {
549 bitvec_write_field(dest, wp,0x1,1); // ALPHA = present
550 bitvec_write_field(dest, wp,alpha,4); // ALPHA = present
551 } else
552 bitvec_write_field(dest, wp,0x0,1); // ALPHA = not present
553 bitvec_write_field(dest, wp,gamma,5); // GAMMA power control parameter
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200554 if (ta_idx < 0) {
555 bitvec_write_field(dest, wp,0x0,1); // switch TIMING_ADVANCE_INDEX = off
556 } else {
557 bitvec_write_field(dest, wp,0x1,1); // switch TIMING_ADVANCE_INDEX = on
558 bitvec_write_field(dest, wp,ta_idx,4); // TIMING_ADVANCE_INDEX
559 }
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200560 bitvec_write_field(dest, wp, 1, 1); // TBF_STARTING_TIME_FLAG
561 bitvec_write_field(dest, wp,(fn / (26 * 51)) % 32,5); // T1'
562 bitvec_write_field(dest, wp,fn % 51,6); // T3
563 bitvec_write_field(dest, wp,fn % 26,5); // T2
564 } else {
565 bitvec_write_field(dest, wp, 1, 1); // Block Allocation : Not Single Block Allocation
566 bitvec_write_field(dest, wp, tfi, 5); // TFI_ASSIGNMENT Temporary Flow Identity
567 bitvec_write_field(dest, wp, 0, 1); // POLLING
568 bitvec_write_field(dest, wp, 0, 1); // ALLOCATION_TYPE: dynamic
569 bitvec_write_field(dest, wp, usf, 3); // USF
570 bitvec_write_field(dest, wp, 0, 1); // USF_GRANULARITY
571 bitvec_write_field(dest, wp, 0, 1); // "0" power control: Not Present
Andreas Eversberg499ff412012-10-03 14:21:36 +0200572 bitvec_write_field(dest, wp, bts->initial_cs_ul-1, 2); // CHANNEL_CODING_COMMAND
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200573 bitvec_write_field(dest, wp, 1, 1); // TLLI_BLOCK_CHANNEL_CODING
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200574 if (alpha) {
575 bitvec_write_field(dest, wp,0x1,1); // ALPHA = present
576 bitvec_write_field(dest, wp,alpha,4); // ALPHA
577 } else
578 bitvec_write_field(dest, wp,0x0,1); // ALPHA = not present
579 bitvec_write_field(dest, wp,gamma,5); // GAMMA power control parameter
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200580 /* note: there is no choise for TAI and no starting time */
581 bitvec_write_field(dest, wp, 0, 1); // switch TIMING_ADVANCE_INDEX = off
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200582 bitvec_write_field(dest, wp, 0, 1); // TBF_STARTING_TIME_FLAG
583 }
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400584 }
585
Andreas Eversberg7b045012012-07-05 07:38:49 +0200586 return plen;
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400587}
588
Andreas Eversberge6228b32012-07-03 13:36:03 +0200589/* generate uplink assignment */
590void write_packet_uplink_assignment(bitvec * dest, uint8_t old_tfi,
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200591 uint8_t old_downlink, uint32_t tlli, uint8_t use_tlli,
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200592 struct gprs_rlcmac_tbf *tbf, uint8_t poll, uint8_t alpha,
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200593 uint8_t gamma, int8_t ta_idx)
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400594{
Andreas Eversberge6228b32012-07-03 13:36:03 +0200595 // TODO We should use our implementation of encode RLC/MAC Control messages.
Andreas Eversbergdfa563c2012-07-06 08:13:59 +0200596 struct gprs_rlcmac_bts *bts = gprs_rlcmac_bts;
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400597 unsigned wp = 0;
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200598 uint8_t ts;
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400599
Andreas Eversberge6228b32012-07-03 13:36:03 +0200600 bitvec_write_field(dest, wp,0x1,2); // Payload Type
601 bitvec_write_field(dest, wp,0x0,2); // Uplink block with TDMA framenumber (N+13)
602 bitvec_write_field(dest, wp,poll,1); // Suppl/Polling Bit
603 bitvec_write_field(dest, wp,0x0,3); // Uplink state flag
604 bitvec_write_field(dest, wp,0xa,6); // MESSAGE TYPE
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400605
Andreas Eversberge6228b32012-07-03 13:36:03 +0200606 bitvec_write_field(dest, wp,0x0,2); // Page Mode
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400607
Andreas Eversberge6228b32012-07-03 13:36:03 +0200608 bitvec_write_field(dest, wp,0x0,1); // switch PERSIST_LEVEL: off
609 if (use_tlli) {
610 bitvec_write_field(dest, wp,0x2,2); // switch TLLI : on
611 bitvec_write_field(dest, wp,tlli,32); // TLLI
612 } else {
613 bitvec_write_field(dest, wp,0x0,1); // switch TFI : on
614 bitvec_write_field(dest, wp,old_downlink,1); // 0=UPLINK TFI, 1=DL TFI
615 bitvec_write_field(dest, wp,old_tfi,5); // TFI
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400616 }
Ivan Kluchnikovb2b81002012-06-04 21:57:02 +0400617
Andreas Eversberge6228b32012-07-03 13:36:03 +0200618 bitvec_write_field(dest, wp,0x0,1); // Message escape
Andreas Eversberg499ff412012-10-03 14:21:36 +0200619 bitvec_write_field(dest, wp,bts->initial_cs_ul-1, 2); // CHANNEL_CODING_COMMAND
Andreas Eversbergdfa563c2012-07-06 08:13:59 +0200620 bitvec_write_field(dest, wp,0x1,1); // TLLI_BLOCK_CHANNEL_CODING
Andreas Eversberge6228b32012-07-03 13:36:03 +0200621 bitvec_write_field(dest, wp,0x1,1); // switch TIMING_ADVANCE_VALUE = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200622 bitvec_write_field(dest, wp,tbf->ta,6); // TIMING_ADVANCE_VALUE
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200623 if (ta_idx < 0) {
624 bitvec_write_field(dest, wp,0x0,1); // switch TIMING_ADVANCE_INDEX = off
625 } else {
626 bitvec_write_field(dest, wp,0x1,1); // switch TIMING_ADVANCE_INDEX = on
627 bitvec_write_field(dest, wp,ta_idx,4); // TIMING_ADVANCE_INDEX
628 }
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400629
Andreas Eversberge6228b32012-07-03 13:36:03 +0200630#if 1
631 bitvec_write_field(dest, wp,0x1,1); // Frequency Parameters information elements = present
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200632 bitvec_write_field(dest, wp,tbf->tsc,3); // Training Sequence Code (TSC)
Andreas Eversberge6228b32012-07-03 13:36:03 +0200633 bitvec_write_field(dest, wp,0x0,2); // ARFCN = present
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200634 bitvec_write_field(dest, wp,tbf->arfcn,10); // ARFCN
Andreas Eversberge6228b32012-07-03 13:36:03 +0200635#else
636 bitvec_write_field(dest, wp,0x0,1); // Frequency Parameters = off
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200637#endif
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400638
Andreas Eversberge6228b32012-07-03 13:36:03 +0200639 bitvec_write_field(dest, wp,0x1,2); // Dynamic Allocation
Ivan Kluchnikove3594232012-06-07 01:12:29 +0400640
Andreas Eversberge6228b32012-07-03 13:36:03 +0200641 bitvec_write_field(dest, wp,0x0,1); // Extended Dynamic Allocation = off
642 bitvec_write_field(dest, wp,0x0,1); // P0 = off
Ivan Kluchnikove3594232012-06-07 01:12:29 +0400643
Andreas Eversberge6228b32012-07-03 13:36:03 +0200644 bitvec_write_field(dest, wp,0x0,1); // USF_GRANULARITY
645 bitvec_write_field(dest, wp,0x1,1); // switch TFI : on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200646 bitvec_write_field(dest, wp,tbf->tfi,5);// TFI
Andreas Eversberge6228b32012-07-03 13:36:03 +0200647
648 bitvec_write_field(dest, wp,0x0,1); //
649 bitvec_write_field(dest, wp,0x0,1); // TBF Starting Time = off
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200650 if (alpha || gamma) {
651 bitvec_write_field(dest, wp,0x1,1); // Timeslot Allocation with Power Control
652 bitvec_write_field(dest, wp,alpha,4); // ALPHA
653 } else
654 bitvec_write_field(dest, wp,0x0,1); // Timeslot Allocation
Andreas Eversberge6228b32012-07-03 13:36:03 +0200655
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200656 for (ts = 0; ts < 8; ts++) {
657 if (tbf->pdch[ts]) {
Andreas Eversberge6228b32012-07-03 13:36:03 +0200658 bitvec_write_field(dest, wp,0x1,1); // USF_TN(i): on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200659 bitvec_write_field(dest, wp,tbf->dir.ul.usf[ts],3); // USF_TN(i)
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200660 if (alpha || gamma)
661 bitvec_write_field(dest, wp,gamma,5); // GAMMA power control parameter
Andreas Eversberge6228b32012-07-03 13:36:03 +0200662 } else
663 bitvec_write_field(dest, wp,0x0,1); // USF_TN(i): off
Ivan Kluchnikove3594232012-06-07 01:12:29 +0400664 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200665// bitvec_write_field(dest, wp,0x0,1); // Measurement Mapping struct not present
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400666}
667
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400668
Andreas Eversberge6228b32012-07-03 13:36:03 +0200669/* generate downlink assignment */
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400670void write_packet_downlink_assignment(RlcMacDownlink_t * block, uint8_t old_tfi,
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200671 uint8_t old_downlink, struct gprs_rlcmac_tbf *tbf, uint8_t poll,
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200672 uint8_t alpha, uint8_t gamma, int8_t ta_idx, uint8_t ta_ts)
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400673{
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400674 // Packet downlink assignment TS 44.060 11.2.7
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400675
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200676 uint8_t tn;
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400677
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400678 block->PAYLOAD_TYPE = 0x1; // RLC/MAC control block that does not include the optional octets of the RLC/MAC control header
679 block->RRBP = 0x0; // N+13
680 block->SP = poll; // RRBP field is valid
681 block->USF = 0x0; // Uplink state flag
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400682
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400683 block->u.Packet_Downlink_Assignment.MESSAGE_TYPE = 0x2; // Packet Downlink Assignment
684 block->u.Packet_Downlink_Assignment.PAGE_MODE = 0x0; // Normal Paging
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400685
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400686 block->u.Packet_Downlink_Assignment.Exist_PERSISTENCE_LEVEL = 0x0; // PERSISTENCE_LEVEL: off
Andreas Eversberge6228b32012-07-03 13:36:03 +0200687
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400688 block->u.Packet_Downlink_Assignment.ID.UnionType = 0x0; // TFI = on
689 block->u.Packet_Downlink_Assignment.ID.u.Global_TFI.UnionType = old_downlink; // 0=UPLINK TFI, 1=DL TFI
690 block->u.Packet_Downlink_Assignment.ID.u.Global_TFI.u.UPLINK_TFI = old_tfi; // TFI
Andreas Eversberge6228b32012-07-03 13:36:03 +0200691
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400692 block->u.Packet_Downlink_Assignment.MAC_MODE = 0x0; // Dynamic Allocation
693 block->u.Packet_Downlink_Assignment.RLC_MODE = 0x0; // RLC acknowledged mode
694 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 +0200695 block->u.Packet_Downlink_Assignment.TIMESLOT_ALLOCATION = 0; // timeslot(s)
696 for (tn = 0; tn < 8; tn++) {
697 if (tbf->pdch[tn])
698 block->u.Packet_Downlink_Assignment.TIMESLOT_ALLOCATION |= 0x80 >> tn; // timeslot(s)
699 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200700
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400701 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.Exist_TIMING_ADVANCE_VALUE = 0x1; // TIMING_ADVANCE_VALUE = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200702 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.TIMING_ADVANCE_VALUE = tbf->ta; // TIMING_ADVANCE_VALUE
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200703 if (ta_idx < 0) {
704 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.Exist_IndexAndtimeSlot = 0x0; // TIMING_ADVANCE_INDEX = off
705 } else {
706 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.Exist_IndexAndtimeSlot = 0x1; // TIMING_ADVANCE_INDEX = on
707 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.TIMING_ADVANCE_INDEX = ta_idx; // TIMING_ADVANCE_INDEX
708 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.TIMING_ADVANCE_TIMESLOT_NUMBER = ta_ts; // TIMING_ADVANCE_TS
709 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200710
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400711 block->u.Packet_Downlink_Assignment.Exist_P0_and_BTS_PWR_CTRL_MODE = 0x0; // POWER CONTROL = off
Andreas Eversberge6228b32012-07-03 13:36:03 +0200712
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400713 block->u.Packet_Downlink_Assignment.Exist_Frequency_Parameters = 0x1; // Frequency Parameters = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200714 block->u.Packet_Downlink_Assignment.Frequency_Parameters.TSC = tbf->tsc; // Training Sequence Code (TSC)
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400715 block->u.Packet_Downlink_Assignment.Frequency_Parameters.UnionType = 0x0; // ARFCN = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200716 block->u.Packet_Downlink_Assignment.Frequency_Parameters.u.ARFCN = tbf->arfcn; // ARFCN
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400717
718 block->u.Packet_Downlink_Assignment.Exist_DOWNLINK_TFI_ASSIGNMENT = 0x1; // DOWNLINK TFI ASSIGNMENT = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200719 block->u.Packet_Downlink_Assignment.DOWNLINK_TFI_ASSIGNMENT = tbf->tfi; // TFI
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400720
721 block->u.Packet_Downlink_Assignment.Exist_Power_Control_Parameters = 0x1; // Power Control Parameters = on
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200722 block->u.Packet_Downlink_Assignment.Power_Control_Parameters.ALPHA = alpha; // ALPHA
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400723
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200724 for (tn = 0; tn < 8; tn++)
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400725 {
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200726 if (tbf->pdch[tn])
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400727 {
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200728 block->u.Packet_Downlink_Assignment.Power_Control_Parameters.Slot[tn].Exist = 0x1; // Slot[i] = on
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200729 block->u.Packet_Downlink_Assignment.Power_Control_Parameters.Slot[tn].GAMMA_TN = gamma; // GAMMA_TN
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400730 }
731 else
732 {
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200733 block->u.Packet_Downlink_Assignment.Power_Control_Parameters.Slot[tn].Exist = 0x0; // Slot[i] = off
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400734 }
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400735 }
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400736
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400737 block->u.Packet_Downlink_Assignment.Exist_TBF_Starting_Time = 0x0; // TBF Starting TIME = off
738 block->u.Packet_Downlink_Assignment.Exist_Measurement_Mapping = 0x0; // Measurement_Mapping = off
739 block->u.Packet_Downlink_Assignment.Exist_AdditionsR99 = 0x0; // AdditionsR99 = off
Andreas Eversberge6228b32012-07-03 13:36:03 +0200740}
741
Andreas Eversberg8c3680d2012-10-08 12:30:56 +0200742/* generate paging request */
743int write_paging_request(bitvec * dest, uint8_t *ptmsi, uint16_t ptmsi_len)
744{
745 unsigned wp = 0;
746 int plen;
747
748 bitvec_write_field(dest, wp,0x0,4); // Skip Indicator
749 bitvec_write_field(dest, wp,0x6,4); // Protocol Discriminator
750 bitvec_write_field(dest, wp,0x21,8); // Paging Request Message Type
751
752 bitvec_write_field(dest, wp,0x0,4); // Page Mode
753 bitvec_write_field(dest, wp,0x0,4); // Channel Needed
754
755 // Mobile Identity
756 bitvec_write_field(dest, wp,ptmsi_len+1,8); // Mobile Identity length
757 bitvec_write_field(dest, wp,0xf,4); // unused
758 bitvec_write_field(dest, wp,0x4,4); // PTMSI type
759 for (int i = 0; i < ptmsi_len; i++)
760 {
761 bitvec_write_field(dest, wp,ptmsi[i],8); // PTMSI
762 }
763 if ((wp % 8)) {
764 LOGP(DRLCMACUL, LOGL_ERROR, "Length of PAG.REQ without rest "
765 "octets is not multiple of 8 bits, PLEASE FIX!\n");
766 exit (0);
767 }
768 plen = wp / 8;
769 bitvec_write_field(dest, wp,0x0,1); // "L" NLN(PCH) = off
770 bitvec_write_field(dest, wp,0x0,1); // "L" Priority1 = off
771 bitvec_write_field(dest, wp,0x1,1); // "L" Priority2 = off
772 bitvec_write_field(dest, wp,0x0,1); // "L" Group Call information = off
773 bitvec_write_field(dest, wp,0x0,1); // "H" Packet Page Indication 1 = packet paging procedure
774 bitvec_write_field(dest, wp,0x1,1); // "H" Packet Page Indication 2 = packet paging procedure
775
776 return plen;
777}
778
Andreas Eversberge6228b32012-07-03 13:36:03 +0200779/* generate uplink ack */
Holger Hans Peter Freytherae947fc2013-10-17 13:29:03 +0200780void write_packet_uplink_ack(struct gprs_rlcmac_bts *bts,
781 RlcMacDownlink_t * block, struct gprs_rlcmac_tbf *tbf,
Andreas Eversberge6228b32012-07-03 13:36:03 +0200782 uint8_t final)
783{
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400784 // Packet Uplink Ack/Nack TS 44.060 11.2.28
785
Andreas Eversberge6228b32012-07-03 13:36:03 +0200786 char show_v_n[65];
787
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400788 uint8_t rbb = 0;
Andreas Eversberge6228b32012-07-03 13:36:03 +0200789 uint16_t i, bbn;
790 uint16_t mod_sns_half = (tbf->sns >> 1) - 1;
791 char bit;
792
793 LOGP(DRLCMACUL, LOGL_DEBUG, "Sending Ack/Nack for TBF=%d "
794 "(final=%d)\n", tbf->tfi, final);
795
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400796 block->PAYLOAD_TYPE = 0x1; // RLC/MAC control block that does not include the optional octets of the RLC/MAC control header
797 block->RRBP = 0x0; // N+13
798 block->SP = final; // RRBP field is valid, if it is final ack
799 block->USF = 0x0; // Uplink state flag
Andreas Eversberge6228b32012-07-03 13:36:03 +0200800
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400801 block->u.Packet_Uplink_Ack_Nack.MESSAGE_TYPE = 0x9; // Packet Downlink Assignment
802 block->u.Packet_Uplink_Ack_Nack.PAGE_MODE = 0x0; // Normal Paging
803 block->u.Packet_Uplink_Ack_Nack.UPLINK_TFI = tbf->tfi; // Uplink TFI
Andreas Eversberge6228b32012-07-03 13:36:03 +0200804
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400805 block->u.Packet_Uplink_Ack_Nack.UnionType = 0x0; // PU_AckNack_GPRS = on
Andreas Eversberg499ff412012-10-03 14:21:36 +0200806 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 +0400807 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Ack_Nack_Description.FINAL_ACK_INDICATION = final; // FINAL ACK INDICATION
808 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 +0200809 // RECEIVE_BLOCK_BITMAP
810 for (i = 0, bbn = (tbf->dir.ul.v_r - 64) & mod_sns_half; i < 64;
811 i++, bbn = (bbn + 1) & mod_sns_half) {
812 bit = tbf->dir.ul.v_n[bbn];
813 if (bit == 0)
814 bit = ' ';
815 show_v_n[i] = bit;
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400816 if (bit == 'R')
817 rbb = (rbb << 1)|1;
818 else
819 rbb = (rbb << 1);
820 if((i%8) == 7)
821 {
822 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Ack_Nack_Description.RECEIVED_BLOCK_BITMAP[i/8] = rbb;
823 rbb = 0;
824 }
Ivan Kluchnikove3594232012-06-07 01:12:29 +0400825 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200826 show_v_n[64] = '\0';
827 LOGP(DRLCMACUL, LOGL_DEBUG, "- V(N): \"%s\" R=Received "
828 "N=Not-Received\n", show_v_n);
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400829
830 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.UnionType = 0x0; // Fixed Allocation Dummy = on
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400831 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.u.FixedAllocationDummy = 0x0; // Fixed Allocation Dummy
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400832 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Exist_AdditionsR99 = 0x0; // AdditionsR99 = off
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400833
834 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 +0400835 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 +0400836 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Common_Uplink_Ack_Nack_Data.Exist_Packet_Timing_Advance = 0x0;
837 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Common_Uplink_Ack_Nack_Data.Exist_Extension_Bits = 0x0;
838 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 +0400839}
840
Andreas Eversberg2b914642012-07-19 13:06:26 +0200841unsigned write_packet_paging_request(bitvec * dest)
842{
843 unsigned wp = 0;
844
845 bitvec_write_field(dest, wp,0x1,2); // Payload Type
846 bitvec_write_field(dest, wp,0x0,3); // No polling
847 bitvec_write_field(dest, wp,0x0,3); // Uplink state flag
848 bitvec_write_field(dest, wp,0x22,6); // MESSAGE TYPE
849
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200850 bitvec_write_field(dest, wp,0x0,2); // Page Mode
851
Andreas Eversberg2b914642012-07-19 13:06:26 +0200852 bitvec_write_field(dest, wp,0x0,1); // No PERSISTENCE_LEVEL
853 bitvec_write_field(dest, wp,0x0,1); // No NLN
854
855 return wp;
856}
857
858unsigned write_repeated_page_info(bitvec * dest, unsigned& wp, uint8_t len,
859 uint8_t *identity, uint8_t chan_needed)
860{
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200861 bitvec_write_field(dest, wp,0x1,1); // Repeated Page info exists
862
Andreas Eversberg2b914642012-07-19 13:06:26 +0200863 bitvec_write_field(dest, wp,0x1,1); // RR connection paging
864
865 if ((identity[0] & 0x07) == 4) {
866 bitvec_write_field(dest, wp,0x0,1); // TMSI
867 identity++;
868 len--;
869 } else {
870 bitvec_write_field(dest, wp,0x0,1); // MI
871 bitvec_write_field(dest, wp,len,4); // MI len
872 }
873 while (len) {
874 bitvec_write_field(dest, wp,*identity++,8); // MI data
875 len--;
876 }
877 bitvec_write_field(dest, wp,chan_needed,2); // CHANNEL_NEEDED
878 bitvec_write_field(dest, wp,0x0,1); // No eMLPP_PRIORITY
879
880 return wp;
881}
882
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400883/* Send Uplink unit-data to SGSN. */
Andreas Eversberg3e372d52012-07-06 09:28:15 +0200884int gprs_rlcmac_tx_ul_ud(gprs_rlcmac_tbf *tbf)
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400885{
Andreas Eversberg9a913462012-09-23 06:42:38 +0200886 uint8_t qos_profile[3];
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400887 struct msgb *llc_pdu;
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200888 unsigned msg_len = NS_HDR_LEN + BSSGP_HDR_LEN + tbf->llc_index;
Holger Hans Peter Freyther90692f92013-07-13 12:51:16 +0200889 struct bssgp_bvc_ctx *bctx = gprs_bssgp_pcu_current_bctx();
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400890
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200891 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 +0200892 if (!bctx) {
893 LOGP(DBSSGP, LOGL_ERROR, "No bctx\n");
894 return -EIO;
895 }
Ivan Kluchnikovc320d862012-03-18 15:04:48 +0400896
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400897 llc_pdu = msgb_alloc_headroom(msg_len, msg_len,"llc_pdu");
Andreas Eversberg9a913462012-09-23 06:42:38 +0200898 uint8_t *buf = msgb_push(llc_pdu, TL16V_GROSS_LEN(sizeof(uint8_t)*tbf->llc_index));
899 tl16v_put(buf, BSSGP_IE_LLC_PDU, sizeof(uint8_t)*tbf->llc_index, tbf->llc_frame);
900 qos_profile[0] = QOS_PROFILE >> 16;
901 qos_profile[1] = QOS_PROFILE >> 8;
902 qos_profile[2] = QOS_PROFILE;
903 bssgp_tx_ul_ud(bctx, tbf->tlli, qos_profile, llc_pdu);
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400904
Andreas Eversberg3e372d52012-07-06 09:28:15 +0200905 return 0;
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400906}
Andreas Eversberg8c3680d2012-10-08 12:30:56 +0200907
908int gprs_rlcmac_paging_request(uint8_t *ptmsi, uint16_t ptmsi_len,
909 const char *imsi)
910{
911 LOGP(DRLCMAC, LOGL_NOTICE, "TX: [PCU -> BTS] Paging Request (CCCH)\n");
912 bitvec *paging_request = bitvec_alloc(23);
913 bitvec_unhex(paging_request, "2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b");
914 int plen = write_paging_request(paging_request, ptmsi, ptmsi_len);
915 pcu_l1if_tx_pch(paging_request, plen, (char *)imsi);
916 bitvec_free(paging_request);
917
918 return 0;
919}
Andreas Eversberga004e6a2013-05-13 16:45:21 +0200920
921
922/*
923 * timing advance memory
924 */
925
926/* enable to debug timing advance memory */
927//#define DEBUG_TA
928
929static LLIST_HEAD(gprs_rlcmac_ta_list);
930static int gprs_rlcmac_ta_num = 0;
931
932struct gprs_rlcmac_ta {
933 struct llist_head list;
934 uint32_t tlli;
935 uint8_t ta;
936};
937
938/* remember timing advance of a given TLLI */
939int remember_timing_advance(uint32_t tlli, uint8_t ta)
940{
941 struct gprs_rlcmac_ta *ta_entry;
942
943 /* check for existing entry */
944 llist_for_each_entry(ta_entry, &gprs_rlcmac_ta_list, list) {
945 if (ta_entry->tlli == tlli) {
946#ifdef DEBUG_TA
947 fprintf(stderr, "update %08x %d\n", tlli, ta);
948#endif
949 ta_entry->ta = ta;
950 /* relink to end of list */
951 llist_del(&ta_entry->list);
952 llist_add_tail(&ta_entry->list, &gprs_rlcmac_ta_list);
953 return 0;
954 }
955 }
956
957#ifdef DEBUG_TA
958 fprintf(stderr, "remember %08x %d\n", tlli, ta);
959#endif
960 /* if list is full, remove oldest entry */
961 if (gprs_rlcmac_ta_num == 30) {
962 ta_entry = llist_entry(gprs_rlcmac_ta_list.next,
963 struct gprs_rlcmac_ta, list);
964 llist_del(&ta_entry->list);
965 talloc_free(ta_entry);
966 gprs_rlcmac_ta_num--;
967 }
968
969 /* create new TA entry */
970 ta_entry = talloc_zero(tall_pcu_ctx, struct gprs_rlcmac_ta);
971 if (!ta_entry)
972 return -ENOMEM;
973
974 ta_entry->tlli = tlli;
975 ta_entry->ta = ta;
976 llist_add_tail(&ta_entry->list, &gprs_rlcmac_ta_list);
977 gprs_rlcmac_ta_num++;
978
979 return 0;
980}
981
982int recall_timing_advance(uint32_t tlli)
983{
984 struct gprs_rlcmac_ta *ta_entry;
985 uint8_t ta;
986
987 llist_for_each_entry(ta_entry, &gprs_rlcmac_ta_list, list) {
988 if (ta_entry->tlli == tlli) {
989 ta = ta_entry->ta;
990#ifdef DEBUG_TA
991 fprintf(stderr, "recall %08x %d\n", tlli, ta);
992#endif
993 return ta;
994 }
995 }
996#ifdef DEBUG_TA
997 fprintf(stderr, "no entry for %08x\n", tlli);
998#endif
999
1000 return -EINVAL;
1001}
1002
1003int flush_timing_advance(void)
1004{
1005 struct gprs_rlcmac_ta *ta_entry;
1006 int count = 0;
1007
1008 while (!llist_empty(&gprs_rlcmac_ta_list)) {
1009 ta_entry = llist_entry(gprs_rlcmac_ta_list.next,
1010 struct gprs_rlcmac_ta, list);
1011#ifdef DEBUG_TA
1012 fprintf(stderr, "flush entry %08x %d\n", ta_entry->tlli,
1013 ta_entry->ta);
1014#endif
1015 llist_del(&ta_entry->list);
1016 talloc_free(ta_entry);
1017 count++;
1018 }
1019 gprs_rlcmac_ta_num = 0;
1020
1021 return count;
1022}
1023