blob: f8c40aca9e7f3c53dd68cc603e39c5f5fa2e3cb0 [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 */
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200248int gprs_rlcmac_rcv_block(uint8_t trx, uint8_t ts, uint8_t *data, uint8_t len,
Andreas Eversberg570b44b2013-03-16 16:15:01 +0100249 uint32_t fn, int8_t rssi)
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400250{
Andreas Eversberge6228b32012-07-03 13:36:03 +0200251 unsigned payload = data[0] >> 6;
252 bitvec *block;
253 int rc = 0;
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400254
Andreas Eversberge6228b32012-07-03 13:36:03 +0200255 switch (payload) {
256 case GPRS_RLCMAC_DATA_BLOCK:
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200257 rc = gprs_rlcmac_rcv_data_block_acknowledged(trx, ts, data,
Andreas Eversberg570b44b2013-03-16 16:15:01 +0100258 len, rssi);
Andreas Eversberge6228b32012-07-03 13:36:03 +0200259 break;
260 case GPRS_RLCMAC_CONTROL_BLOCK:
261 block = bitvec_alloc(len);
262 if (!block)
263 return -ENOMEM;
264 bitvec_unpack(block, data);
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200265 rc = gprs_rlcmac_rcv_control_block(block, trx, ts, fn);
Andreas Eversberge6228b32012-07-03 13:36:03 +0200266 bitvec_free(block);
267 break;
268 case GPRS_RLCMAC_CONTROL_BLOCK_OPT:
269 LOGP(DRLCMAC, LOGL_NOTICE, "GPRS_RLCMAC_CONTROL_BLOCK_OPT block payload is not supported.\n");
Holger Hans Peter Freyther26da8362013-10-16 16:34:09 +0200270 break;
Andreas Eversberge6228b32012-07-03 13:36:03 +0200271 default:
Holger Hans Peter Freyther70862c92013-10-16 16:37:22 +0200272 LOGP(DRLCMAC, LOGL_NOTICE, "Unknown RLCMAC block payload(%u).\n", payload);
Andreas Eversberge6228b32012-07-03 13:36:03 +0200273 rc = -EINVAL;
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400274 }
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400275
Andreas Eversberge6228b32012-07-03 13:36:03 +0200276 return rc;
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400277}
278
Andreas Eversberg2b914642012-07-19 13:06:26 +0200279/* add paging to paging queue(s) */
280int gprs_rlcmac_add_paging(uint8_t chan_needed, uint8_t *identity_lv)
281{
282 struct gprs_rlcmac_bts *bts = gprs_rlcmac_bts;
283 uint8_t l, trx, ts, any_tbf = 0;
284 struct gprs_rlcmac_tbf *tbf;
285 struct gprs_rlcmac_paging *pag;
286 uint8_t slot_mask[8];
287 int8_t first_ts; /* must be signed */
288
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200289 LOGP(DRLCMAC, LOGL_INFO, "Add RR paging: chan-needed=%d MI=%s\n",
290 chan_needed, osmo_hexdump(identity_lv + 1, identity_lv[0]));
Andreas Eversberg2b914642012-07-19 13:06:26 +0200291
292 /* collect slots to page
293 * Mark slots for every TBF, but only mark one of it.
294 * Mark only the first slot found.
295 * Don't mark, if TBF uses a different slot that is already marked. */
296 memset(slot_mask, 0, sizeof(slot_mask));
297 for (l = 0; gprs_rlcmac_tbfs_lists[l]; l++) {
298 llist_for_each_entry(tbf, gprs_rlcmac_tbfs_lists[l], list) {
299 first_ts = -1;
300 for (ts = 0; ts < 8; ts++) {
301 if (tbf->pdch[ts]) {
302 /* remember the first slot found */
303 if (first_ts < 0)
304 first_ts = ts;
305 /* break, if we already marked a slot */
Holger Hans Peter Freyther96efa702013-09-29 07:44:39 +0200306 if ((slot_mask[tbf->trx_no] & (1 << ts)))
Andreas Eversberg2b914642012-07-19 13:06:26 +0200307 break;
308 }
309 }
310 /* mark first slot found, if none is marked already */
311 if (ts == 8 && first_ts >= 0) {
312 LOGP(DRLCMAC, LOGL_DEBUG, "- %s TBF=%d uses "
313 "TRX=%d TS=%d, so we mark\n",
314 (tbf->direction == GPRS_RLCMAC_UL_TBF)
315 ? "UL" : "DL",
Holger Hans Peter Freyther96efa702013-09-29 07:44:39 +0200316 tbf->tfi, tbf->trx_no, first_ts);
317 slot_mask[tbf->trx_no] |= (1 << first_ts);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200318 } else
319 LOGP(DRLCMAC, LOGL_DEBUG, "- %s TBF=%d uses "
320 "already marked TRX=%d TS=%d\n",
321 (tbf->direction == GPRS_RLCMAC_UL_TBF)
322 ? "UL" : "DL",
Holger Hans Peter Freyther96efa702013-09-29 07:44:39 +0200323 tbf->tfi, tbf->trx_no, ts);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200324 }
325 }
326
327 /* Now we have a list of marked slots. Every TBF uses at least one
328 * of these slots. */
329
330 /* schedule paging to all marked slots */
331 for (trx = 0; trx < 8; trx++) {
332 if (slot_mask[trx] == 0)
333 continue;
334 any_tbf = 1;
335 for (ts = 0; ts < 8; ts++) {
336 if ((slot_mask[trx] & (1 << ts))) {
337 /* schedule */
Andreas Eversberg6681bb82012-07-25 08:48:44 +0200338 pag = talloc_zero(tall_pcu_ctx,
Andreas Eversberg2b914642012-07-19 13:06:26 +0200339 struct gprs_rlcmac_paging);
340 if (!pag)
341 return -ENOMEM;
342 pag->chan_needed = chan_needed;
343 memcpy(pag->identity_lv, identity_lv,
344 identity_lv[0] + 1);
345 llist_add(&pag->list,
346 &bts->trx[trx].pdch[ts].paging_list);
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200347 LOGP(DRLCMAC, LOGL_INFO, "Paging on PACCH of "
348 "TRX=%d TS=%d\n", trx, ts);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200349 }
350 }
351 }
352
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200353 if (!any_tbf)
Andreas Eversberg2b914642012-07-19 13:06:26 +0200354 LOGP(DRLCMAC, LOGL_INFO, "No paging, because no TBF\n");
Andreas Eversberg2b914642012-07-19 13:06:26 +0200355
356 return 0;
357}
358
359struct gprs_rlcmac_paging *gprs_rlcmac_dequeue_paging(
360 struct gprs_rlcmac_pdch *pdch)
361{
362 struct gprs_rlcmac_paging *pag;
363
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200364 if (llist_empty(&pdch->paging_list))
365 return NULL;
Andreas Eversberg2b914642012-07-19 13:06:26 +0200366 pag = llist_entry(pdch->paging_list.next,
367 struct gprs_rlcmac_paging, list);
368 llist_del(&pag->list);
369
370 return pag;
371}
372
373struct msgb *gprs_rlcmac_send_packet_paging_request(
374 struct gprs_rlcmac_pdch *pdch)
375{
376 struct gprs_rlcmac_paging *pag;
377 struct msgb *msg;
378 unsigned wp = 0, len;
379
380 /* no paging, no message */
381 pag = gprs_rlcmac_dequeue_paging(pdch);
382 if (!pag)
383 return NULL;
384
385 LOGP(DRLCMAC, LOGL_DEBUG, "Scheduling paging\n");
386
387 /* alloc message */
388 msg = msgb_alloc(23, "pag ctrl block");
Andreas Eversberg5cae0872012-10-07 15:08:22 +0200389 if (!msg) {
390 talloc_free(pag);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200391 return NULL;
Andreas Eversberg5cae0872012-10-07 15:08:22 +0200392 }
Andreas Eversberg2b914642012-07-19 13:06:26 +0200393 bitvec *pag_vec = bitvec_alloc(23);
394 if (!pag_vec) {
395 msgb_free(msg);
Andreas Eversberg5cae0872012-10-07 15:08:22 +0200396 talloc_free(pag);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200397 return NULL;
398 }
399 wp = write_packet_paging_request(pag_vec);
400
401 /* loop until message is full */
402 while (pag) {
403 /* try to add paging */
404 if ((pag->identity_lv[1] & 0x07) == 4) {
405 /* TMSI */
406 LOGP(DRLCMAC, LOGL_DEBUG, "- TMSI=0x%08x\n",
407 ntohl(*((uint32_t *)(pag->identity_lv + 1))));
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200408 len = 1 + 1 + 1 + 32 + 2 + 1;
Andreas Eversberg2b914642012-07-19 13:06:26 +0200409 if (pag->identity_lv[0] != 5) {
410 LOGP(DRLCMAC, LOGL_ERROR, "TMSI paging with "
411 "MI != 5 octets!\n");
Andreas Eversberg5cae0872012-10-07 15:08:22 +0200412 goto continue_next;
Andreas Eversberg2b914642012-07-19 13:06:26 +0200413 }
414 } else {
415 /* MI */
416 LOGP(DRLCMAC, LOGL_DEBUG, "- MI=%s\n",
417 osmo_hexdump(pag->identity_lv + 1,
418 pag->identity_lv[0]));
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200419 len = 1 + 1 + 1 + 4 + (pag->identity_lv[0]<<3) + 2 + 1;
Andreas Eversberg2b914642012-07-19 13:06:26 +0200420 if (pag->identity_lv[0] > 8) {
421 LOGP(DRLCMAC, LOGL_ERROR, "Paging with "
422 "MI > 8 octets!\n");
Andreas Eversberg5cae0872012-10-07 15:08:22 +0200423 goto continue_next;
Andreas Eversberg2b914642012-07-19 13:06:26 +0200424 }
425 }
426 if (wp + len > 184) {
427 LOGP(DRLCMAC, LOGL_DEBUG, "- Does not fit, so schedule "
428 "next time\n");
429 /* put back paging record, because does not fit */
430 llist_add_tail(&pag->list, &pdch->paging_list);
431 break;
432 }
433 write_repeated_page_info(pag_vec, wp, pag->identity_lv[0],
434 pag->identity_lv + 1, pag->chan_needed);
435
Andreas Eversberg5cae0872012-10-07 15:08:22 +0200436continue_next:
437 talloc_free(pag);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200438 pag = gprs_rlcmac_dequeue_paging(pdch);
439 }
440
441 bitvec_pack(pag_vec, msgb_put(msg, 23));
Andreas Eversberg6681bb82012-07-25 08:48:44 +0200442 RlcMacDownlink_t * mac_control_block = (RlcMacDownlink_t *)talloc_zero(tall_pcu_ctx, RlcMacDownlink_t);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200443 LOGP(DRLCMAC, LOGL_DEBUG, "+++++++++++++++++++++++++ TX : Packet Paging Request +++++++++++++++++++++++++\n");
444 decode_gsm_rlcmac_downlink(pag_vec, mac_control_block);
445 LOGPC(DCSN1, LOGL_NOTICE, "\n");
446 LOGP(DRLCMAC, LOGL_DEBUG, "------------------------- TX : Packet Paging Request -------------------------\n");
447 bitvec_free(pag_vec);
Andreas Eversberg6681bb82012-07-25 08:48:44 +0200448 talloc_free(mac_control_block);
Andreas Eversberg2b914642012-07-19 13:06:26 +0200449
450 return msg;
451}
452
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400453// GSM 04.08 9.1.18 Immediate assignment
Andreas Eversberge6228b32012-07-03 13:36:03 +0200454int write_immediate_assignment(bitvec * dest, uint8_t downlink, uint8_t ra,
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200455 uint32_t ref_fn, uint8_t ta, uint16_t arfcn, uint8_t ts, uint8_t tsc,
Andreas Eversberge6228b32012-07-03 13:36:03 +0200456 uint8_t tfi, uint8_t usf, uint32_t tlli,
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200457 uint8_t polling, uint32_t fn, uint8_t single_block, uint8_t alpha,
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200458 uint8_t gamma, int8_t ta_idx)
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400459{
460 unsigned wp = 0;
Andreas Eversberg7b045012012-07-05 07:38:49 +0200461 uint8_t plen;
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400462
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400463 bitvec_write_field(dest, wp,0x0,4); // Skip Indicator
464 bitvec_write_field(dest, wp,0x6,4); // Protocol Discriminator
465 bitvec_write_field(dest, wp,0x3F,8); // Immediate Assignment Message Type
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400466
467 // 10.5.2.25b Dedicated mode or TBF
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400468 bitvec_write_field(dest, wp,0x0,1); // spare
469 bitvec_write_field(dest, wp,0x0,1); // TMA : Two-message assignment: No meaning
470 bitvec_write_field(dest, wp,downlink,1); // Downlink : Downlink assignment to mobile in packet idle mode
471 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 +0400472
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400473 bitvec_write_field(dest, wp,0x0,4); // Page Mode
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400474
475 // GSM 04.08 10.5.2.25a Packet Channel Description
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400476 bitvec_write_field(dest, wp,0x1,5); // Channel type
Andreas Eversberg0aed6542012-06-23 10:33:16 +0200477 bitvec_write_field(dest, wp,ts,3); // TN
478 bitvec_write_field(dest, wp,tsc,3); // TSC
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400479 bitvec_write_field(dest, wp,0x0,3); // non-hopping RF channel configuraion
Andreas Eversberg0aed6542012-06-23 10:33:16 +0200480 bitvec_write_field(dest, wp,arfcn,10); // ARFCN
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400481
482 //10.5.2.30 Request Reference
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400483 bitvec_write_field(dest, wp,ra,8); // RA
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200484 bitvec_write_field(dest, wp,(ref_fn / (26 * 51)) % 32,5); // T1'
485 bitvec_write_field(dest, wp,ref_fn % 51,6); // T3
486 bitvec_write_field(dest, wp,ref_fn % 26,5); // T2
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400487
488 // 10.5.2.40 Timing Advance
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400489 bitvec_write_field(dest, wp,0x0,2); // spare
490 bitvec_write_field(dest, wp,ta,6); // Timing Advance value
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400491
492 // No mobile allocation in non-hopping systems.
493 // A zero-length LV. Just write L=0.
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400494 bitvec_write_field(dest, wp,0,8);
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400495
Andreas Eversberg7b045012012-07-05 07:38:49 +0200496 if ((wp % 8)) {
497 LOGP(DRLCMACUL, LOGL_ERROR, "Length of IMM.ASS without rest "
498 "octets is not multiple of 8 bits, PLEASE FIX!\n");
499 exit (0);
500 }
501 plen = wp / 8;
502
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400503 if (downlink)
504 {
505 // GSM 04.08 10.5.2.16 IA Rest Octets
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400506 bitvec_write_field(dest, wp, 3, 2); // "HH"
507 bitvec_write_field(dest, wp, 1, 2); // "01" Packet Downlink Assignment
508 bitvec_write_field(dest, wp,tlli,32); // TLLI
509 bitvec_write_field(dest, wp,0x1,1); // switch TFI : on
510 bitvec_write_field(dest, wp,tfi,5); // TFI
511 bitvec_write_field(dest, wp,0x0,1); // RLC acknowledged mode
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200512 if (alpha) {
513 bitvec_write_field(dest, wp,0x1,1); // ALPHA = present
514 bitvec_write_field(dest, wp,alpha,4); // ALPHA
515 } else {
516 bitvec_write_field(dest, wp,0x0,1); // ALPHA = not present
517 }
518 bitvec_write_field(dest, wp,gamma,5); // GAMMA power control parameter
Andreas Eversberge6228b32012-07-03 13:36:03 +0200519 bitvec_write_field(dest, wp,polling,1); // Polling Bit
520 bitvec_write_field(dest, wp,!polling,1); // TA_VALID ???
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200521 if (ta_idx < 0) {
522 bitvec_write_field(dest, wp,0x0,1); // switch TIMING_ADVANCE_INDEX = off
523 } else {
524 bitvec_write_field(dest, wp,0x1,1); // switch TIMING_ADVANCE_INDEX = on
525 bitvec_write_field(dest, wp,ta_idx,4); // TIMING_ADVANCE_INDEX
526 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200527 if (polling) {
528 bitvec_write_field(dest, wp,0x1,1); // TBF Starting TIME present
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200529 bitvec_write_field(dest, wp,(fn / (26 * 51)) % 32,5); // T1'
530 bitvec_write_field(dest, wp,fn % 51,6); // T3
531 bitvec_write_field(dest, wp,fn % 26,5); // T2
Andreas Eversberge6228b32012-07-03 13:36:03 +0200532 } else {
533 bitvec_write_field(dest, wp,0x0,1); // TBF Starting TIME present
534 }
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400535 bitvec_write_field(dest, wp,0x0,1); // P0 not present
Andreas Eversberge6228b32012-07-03 13:36:03 +0200536// bitvec_write_field(dest, wp,0x1,1); // P0 not present
537// bitvec_write_field(dest, wp,0xb,4);
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400538 }
539 else
540 {
Andreas Eversbergdfa563c2012-07-06 08:13:59 +0200541 struct gprs_rlcmac_bts *bts = gprs_rlcmac_bts;
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400542 // GMS 04.08 10.5.2.37b 10.5.2.16
Ivan Kluchnikov835f91e2012-04-30 18:00:36 +0400543 bitvec_write_field(dest, wp, 3, 2); // "HH"
544 bitvec_write_field(dest, wp, 0, 2); // "0" Packet Uplink Assignment
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200545 if (single_block) {
546 bitvec_write_field(dest, wp, 0, 1); // Block Allocation : Single Block Allocation
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200547 if (alpha) {
548 bitvec_write_field(dest, wp,0x1,1); // ALPHA = present
549 bitvec_write_field(dest, wp,alpha,4); // ALPHA = present
550 } else
551 bitvec_write_field(dest, wp,0x0,1); // ALPHA = not present
552 bitvec_write_field(dest, wp,gamma,5); // GAMMA power control parameter
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200553 if (ta_idx < 0) {
554 bitvec_write_field(dest, wp,0x0,1); // switch TIMING_ADVANCE_INDEX = off
555 } else {
556 bitvec_write_field(dest, wp,0x1,1); // switch TIMING_ADVANCE_INDEX = on
557 bitvec_write_field(dest, wp,ta_idx,4); // TIMING_ADVANCE_INDEX
558 }
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200559 bitvec_write_field(dest, wp, 1, 1); // TBF_STARTING_TIME_FLAG
560 bitvec_write_field(dest, wp,(fn / (26 * 51)) % 32,5); // T1'
561 bitvec_write_field(dest, wp,fn % 51,6); // T3
562 bitvec_write_field(dest, wp,fn % 26,5); // T2
563 } else {
564 bitvec_write_field(dest, wp, 1, 1); // Block Allocation : Not Single Block Allocation
565 bitvec_write_field(dest, wp, tfi, 5); // TFI_ASSIGNMENT Temporary Flow Identity
566 bitvec_write_field(dest, wp, 0, 1); // POLLING
567 bitvec_write_field(dest, wp, 0, 1); // ALLOCATION_TYPE: dynamic
568 bitvec_write_field(dest, wp, usf, 3); // USF
569 bitvec_write_field(dest, wp, 0, 1); // USF_GRANULARITY
570 bitvec_write_field(dest, wp, 0, 1); // "0" power control: Not Present
Andreas Eversberg499ff412012-10-03 14:21:36 +0200571 bitvec_write_field(dest, wp, bts->initial_cs_ul-1, 2); // CHANNEL_CODING_COMMAND
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200572 bitvec_write_field(dest, wp, 1, 1); // TLLI_BLOCK_CHANNEL_CODING
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200573 if (alpha) {
574 bitvec_write_field(dest, wp,0x1,1); // ALPHA = present
575 bitvec_write_field(dest, wp,alpha,4); // ALPHA
576 } else
577 bitvec_write_field(dest, wp,0x0,1); // ALPHA = not present
578 bitvec_write_field(dest, wp,gamma,5); // GAMMA power control parameter
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200579 /* note: there is no choise for TAI and no starting time */
580 bitvec_write_field(dest, wp, 0, 1); // switch TIMING_ADVANCE_INDEX = off
Andreas Eversberg07e97cf2012-08-07 16:00:56 +0200581 bitvec_write_field(dest, wp, 0, 1); // TBF_STARTING_TIME_FLAG
582 }
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400583 }
584
Andreas Eversberg7b045012012-07-05 07:38:49 +0200585 return plen;
Ivan Kluchnikov5310d452012-04-17 22:00:31 +0400586}
587
Andreas Eversberge6228b32012-07-03 13:36:03 +0200588/* generate uplink assignment */
589void write_packet_uplink_assignment(bitvec * dest, uint8_t old_tfi,
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200590 uint8_t old_downlink, uint32_t tlli, uint8_t use_tlli,
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200591 struct gprs_rlcmac_tbf *tbf, uint8_t poll, uint8_t alpha,
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200592 uint8_t gamma, int8_t ta_idx)
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400593{
Andreas Eversberge6228b32012-07-03 13:36:03 +0200594 // TODO We should use our implementation of encode RLC/MAC Control messages.
Andreas Eversbergdfa563c2012-07-06 08:13:59 +0200595 struct gprs_rlcmac_bts *bts = gprs_rlcmac_bts;
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400596 unsigned wp = 0;
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200597 uint8_t ts;
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400598
Andreas Eversberge6228b32012-07-03 13:36:03 +0200599 bitvec_write_field(dest, wp,0x1,2); // Payload Type
600 bitvec_write_field(dest, wp,0x0,2); // Uplink block with TDMA framenumber (N+13)
601 bitvec_write_field(dest, wp,poll,1); // Suppl/Polling Bit
602 bitvec_write_field(dest, wp,0x0,3); // Uplink state flag
603 bitvec_write_field(dest, wp,0xa,6); // MESSAGE TYPE
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400604
Andreas Eversberge6228b32012-07-03 13:36:03 +0200605 bitvec_write_field(dest, wp,0x0,2); // Page Mode
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400606
Andreas Eversberge6228b32012-07-03 13:36:03 +0200607 bitvec_write_field(dest, wp,0x0,1); // switch PERSIST_LEVEL: off
608 if (use_tlli) {
609 bitvec_write_field(dest, wp,0x2,2); // switch TLLI : on
610 bitvec_write_field(dest, wp,tlli,32); // TLLI
611 } else {
612 bitvec_write_field(dest, wp,0x0,1); // switch TFI : on
613 bitvec_write_field(dest, wp,old_downlink,1); // 0=UPLINK TFI, 1=DL TFI
614 bitvec_write_field(dest, wp,old_tfi,5); // TFI
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400615 }
Ivan Kluchnikovb2b81002012-06-04 21:57:02 +0400616
Andreas Eversberge6228b32012-07-03 13:36:03 +0200617 bitvec_write_field(dest, wp,0x0,1); // Message escape
Andreas Eversberg499ff412012-10-03 14:21:36 +0200618 bitvec_write_field(dest, wp,bts->initial_cs_ul-1, 2); // CHANNEL_CODING_COMMAND
Andreas Eversbergdfa563c2012-07-06 08:13:59 +0200619 bitvec_write_field(dest, wp,0x1,1); // TLLI_BLOCK_CHANNEL_CODING
Andreas Eversberge6228b32012-07-03 13:36:03 +0200620 bitvec_write_field(dest, wp,0x1,1); // switch TIMING_ADVANCE_VALUE = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200621 bitvec_write_field(dest, wp,tbf->ta,6); // TIMING_ADVANCE_VALUE
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200622 if (ta_idx < 0) {
623 bitvec_write_field(dest, wp,0x0,1); // switch TIMING_ADVANCE_INDEX = off
624 } else {
625 bitvec_write_field(dest, wp,0x1,1); // switch TIMING_ADVANCE_INDEX = on
626 bitvec_write_field(dest, wp,ta_idx,4); // TIMING_ADVANCE_INDEX
627 }
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400628
Andreas Eversberge6228b32012-07-03 13:36:03 +0200629#if 1
630 bitvec_write_field(dest, wp,0x1,1); // Frequency Parameters information elements = present
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200631 bitvec_write_field(dest, wp,tbf->tsc,3); // Training Sequence Code (TSC)
Andreas Eversberge6228b32012-07-03 13:36:03 +0200632 bitvec_write_field(dest, wp,0x0,2); // ARFCN = present
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200633 bitvec_write_field(dest, wp,tbf->arfcn,10); // ARFCN
Andreas Eversberge6228b32012-07-03 13:36:03 +0200634#else
635 bitvec_write_field(dest, wp,0x0,1); // Frequency Parameters = off
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200636#endif
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400637
Andreas Eversberge6228b32012-07-03 13:36:03 +0200638 bitvec_write_field(dest, wp,0x1,2); // Dynamic Allocation
Ivan Kluchnikove3594232012-06-07 01:12:29 +0400639
Andreas Eversberge6228b32012-07-03 13:36:03 +0200640 bitvec_write_field(dest, wp,0x0,1); // Extended Dynamic Allocation = off
641 bitvec_write_field(dest, wp,0x0,1); // P0 = off
Ivan Kluchnikove3594232012-06-07 01:12:29 +0400642
Andreas Eversberge6228b32012-07-03 13:36:03 +0200643 bitvec_write_field(dest, wp,0x0,1); // USF_GRANULARITY
644 bitvec_write_field(dest, wp,0x1,1); // switch TFI : on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200645 bitvec_write_field(dest, wp,tbf->tfi,5);// TFI
Andreas Eversberge6228b32012-07-03 13:36:03 +0200646
647 bitvec_write_field(dest, wp,0x0,1); //
648 bitvec_write_field(dest, wp,0x0,1); // TBF Starting Time = off
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200649 if (alpha || gamma) {
650 bitvec_write_field(dest, wp,0x1,1); // Timeslot Allocation with Power Control
651 bitvec_write_field(dest, wp,alpha,4); // ALPHA
652 } else
653 bitvec_write_field(dest, wp,0x0,1); // Timeslot Allocation
Andreas Eversberge6228b32012-07-03 13:36:03 +0200654
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200655 for (ts = 0; ts < 8; ts++) {
656 if (tbf->pdch[ts]) {
Andreas Eversberge6228b32012-07-03 13:36:03 +0200657 bitvec_write_field(dest, wp,0x1,1); // USF_TN(i): on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200658 bitvec_write_field(dest, wp,tbf->dir.ul.usf[ts],3); // USF_TN(i)
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200659 if (alpha || gamma)
660 bitvec_write_field(dest, wp,gamma,5); // GAMMA power control parameter
Andreas Eversberge6228b32012-07-03 13:36:03 +0200661 } else
662 bitvec_write_field(dest, wp,0x0,1); // USF_TN(i): off
Ivan Kluchnikove3594232012-06-07 01:12:29 +0400663 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200664// bitvec_write_field(dest, wp,0x0,1); // Measurement Mapping struct not present
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400665}
666
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400667
Andreas Eversberge6228b32012-07-03 13:36:03 +0200668/* generate downlink assignment */
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400669void write_packet_downlink_assignment(RlcMacDownlink_t * block, uint8_t old_tfi,
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200670 uint8_t old_downlink, struct gprs_rlcmac_tbf *tbf, uint8_t poll,
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200671 uint8_t alpha, uint8_t gamma, int8_t ta_idx, uint8_t ta_ts)
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400672{
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400673 // Packet downlink assignment TS 44.060 11.2.7
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400674
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200675 uint8_t tn;
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400676
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400677 block->PAYLOAD_TYPE = 0x1; // RLC/MAC control block that does not include the optional octets of the RLC/MAC control header
678 block->RRBP = 0x0; // N+13
679 block->SP = poll; // RRBP field is valid
680 block->USF = 0x0; // Uplink state flag
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400681
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400682 block->u.Packet_Downlink_Assignment.MESSAGE_TYPE = 0x2; // Packet Downlink Assignment
683 block->u.Packet_Downlink_Assignment.PAGE_MODE = 0x0; // Normal Paging
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400684
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400685 block->u.Packet_Downlink_Assignment.Exist_PERSISTENCE_LEVEL = 0x0; // PERSISTENCE_LEVEL: off
Andreas Eversberge6228b32012-07-03 13:36:03 +0200686
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400687 block->u.Packet_Downlink_Assignment.ID.UnionType = 0x0; // TFI = on
688 block->u.Packet_Downlink_Assignment.ID.u.Global_TFI.UnionType = old_downlink; // 0=UPLINK TFI, 1=DL TFI
689 block->u.Packet_Downlink_Assignment.ID.u.Global_TFI.u.UPLINK_TFI = old_tfi; // TFI
Andreas Eversberge6228b32012-07-03 13:36:03 +0200690
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400691 block->u.Packet_Downlink_Assignment.MAC_MODE = 0x0; // Dynamic Allocation
692 block->u.Packet_Downlink_Assignment.RLC_MODE = 0x0; // RLC acknowledged mode
693 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 +0200694 block->u.Packet_Downlink_Assignment.TIMESLOT_ALLOCATION = 0; // timeslot(s)
695 for (tn = 0; tn < 8; tn++) {
696 if (tbf->pdch[tn])
697 block->u.Packet_Downlink_Assignment.TIMESLOT_ALLOCATION |= 0x80 >> tn; // timeslot(s)
698 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200699
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400700 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.Exist_TIMING_ADVANCE_VALUE = 0x1; // TIMING_ADVANCE_VALUE = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200701 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.TIMING_ADVANCE_VALUE = tbf->ta; // TIMING_ADVANCE_VALUE
Andreas Eversbergafdd9e12012-10-07 17:08:55 +0200702 if (ta_idx < 0) {
703 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.Exist_IndexAndtimeSlot = 0x0; // TIMING_ADVANCE_INDEX = off
704 } else {
705 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.Exist_IndexAndtimeSlot = 0x1; // TIMING_ADVANCE_INDEX = on
706 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.TIMING_ADVANCE_INDEX = ta_idx; // TIMING_ADVANCE_INDEX
707 block->u.Packet_Downlink_Assignment.Packet_Timing_Advance.TIMING_ADVANCE_TIMESLOT_NUMBER = ta_ts; // TIMING_ADVANCE_TS
708 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200709
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400710 block->u.Packet_Downlink_Assignment.Exist_P0_and_BTS_PWR_CTRL_MODE = 0x0; // POWER CONTROL = off
Andreas Eversberge6228b32012-07-03 13:36:03 +0200711
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400712 block->u.Packet_Downlink_Assignment.Exist_Frequency_Parameters = 0x1; // Frequency Parameters = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200713 block->u.Packet_Downlink_Assignment.Frequency_Parameters.TSC = tbf->tsc; // Training Sequence Code (TSC)
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400714 block->u.Packet_Downlink_Assignment.Frequency_Parameters.UnionType = 0x0; // ARFCN = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200715 block->u.Packet_Downlink_Assignment.Frequency_Parameters.u.ARFCN = tbf->arfcn; // ARFCN
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400716
717 block->u.Packet_Downlink_Assignment.Exist_DOWNLINK_TFI_ASSIGNMENT = 0x1; // DOWNLINK TFI ASSIGNMENT = on
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200718 block->u.Packet_Downlink_Assignment.DOWNLINK_TFI_ASSIGNMENT = tbf->tfi; // TFI
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400719
720 block->u.Packet_Downlink_Assignment.Exist_Power_Control_Parameters = 0x1; // Power Control Parameters = on
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200721 block->u.Packet_Downlink_Assignment.Power_Control_Parameters.ALPHA = alpha; // ALPHA
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400722
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200723 for (tn = 0; tn < 8; tn++)
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400724 {
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200725 if (tbf->pdch[tn])
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400726 {
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200727 block->u.Packet_Downlink_Assignment.Power_Control_Parameters.Slot[tn].Exist = 0x1; // Slot[i] = on
Andreas Eversbergaafcbbb2012-09-27 09:20:45 +0200728 block->u.Packet_Downlink_Assignment.Power_Control_Parameters.Slot[tn].GAMMA_TN = gamma; // GAMMA_TN
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400729 }
730 else
731 {
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200732 block->u.Packet_Downlink_Assignment.Power_Control_Parameters.Slot[tn].Exist = 0x0; // Slot[i] = off
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400733 }
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400734 }
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400735
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400736 block->u.Packet_Downlink_Assignment.Exist_TBF_Starting_Time = 0x0; // TBF Starting TIME = off
737 block->u.Packet_Downlink_Assignment.Exist_Measurement_Mapping = 0x0; // Measurement_Mapping = off
738 block->u.Packet_Downlink_Assignment.Exist_AdditionsR99 = 0x0; // AdditionsR99 = off
Andreas Eversberge6228b32012-07-03 13:36:03 +0200739}
740
Andreas Eversberg8c3680d2012-10-08 12:30:56 +0200741/* generate paging request */
742int write_paging_request(bitvec * dest, uint8_t *ptmsi, uint16_t ptmsi_len)
743{
744 unsigned wp = 0;
745 int plen;
746
747 bitvec_write_field(dest, wp,0x0,4); // Skip Indicator
748 bitvec_write_field(dest, wp,0x6,4); // Protocol Discriminator
749 bitvec_write_field(dest, wp,0x21,8); // Paging Request Message Type
750
751 bitvec_write_field(dest, wp,0x0,4); // Page Mode
752 bitvec_write_field(dest, wp,0x0,4); // Channel Needed
753
754 // Mobile Identity
755 bitvec_write_field(dest, wp,ptmsi_len+1,8); // Mobile Identity length
756 bitvec_write_field(dest, wp,0xf,4); // unused
757 bitvec_write_field(dest, wp,0x4,4); // PTMSI type
758 for (int i = 0; i < ptmsi_len; i++)
759 {
760 bitvec_write_field(dest, wp,ptmsi[i],8); // PTMSI
761 }
762 if ((wp % 8)) {
763 LOGP(DRLCMACUL, LOGL_ERROR, "Length of PAG.REQ without rest "
764 "octets is not multiple of 8 bits, PLEASE FIX!\n");
765 exit (0);
766 }
767 plen = wp / 8;
768 bitvec_write_field(dest, wp,0x0,1); // "L" NLN(PCH) = off
769 bitvec_write_field(dest, wp,0x0,1); // "L" Priority1 = off
770 bitvec_write_field(dest, wp,0x1,1); // "L" Priority2 = off
771 bitvec_write_field(dest, wp,0x0,1); // "L" Group Call information = off
772 bitvec_write_field(dest, wp,0x0,1); // "H" Packet Page Indication 1 = packet paging procedure
773 bitvec_write_field(dest, wp,0x1,1); // "H" Packet Page Indication 2 = packet paging procedure
774
775 return plen;
776}
777
Andreas Eversberge6228b32012-07-03 13:36:03 +0200778/* generate uplink ack */
Holger Hans Peter Freytherae947fc2013-10-17 13:29:03 +0200779void write_packet_uplink_ack(struct gprs_rlcmac_bts *bts,
780 RlcMacDownlink_t * block, struct gprs_rlcmac_tbf *tbf,
Andreas Eversberge6228b32012-07-03 13:36:03 +0200781 uint8_t final)
782{
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400783 // Packet Uplink Ack/Nack TS 44.060 11.2.28
784
Andreas Eversberge6228b32012-07-03 13:36:03 +0200785 char show_v_n[65];
786
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400787 uint8_t rbb = 0;
Andreas Eversberge6228b32012-07-03 13:36:03 +0200788 uint16_t i, bbn;
789 uint16_t mod_sns_half = (tbf->sns >> 1) - 1;
790 char bit;
791
792 LOGP(DRLCMACUL, LOGL_DEBUG, "Sending Ack/Nack for TBF=%d "
793 "(final=%d)\n", tbf->tfi, final);
794
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400795 block->PAYLOAD_TYPE = 0x1; // RLC/MAC control block that does not include the optional octets of the RLC/MAC control header
796 block->RRBP = 0x0; // N+13
797 block->SP = final; // RRBP field is valid, if it is final ack
798 block->USF = 0x0; // Uplink state flag
Andreas Eversberge6228b32012-07-03 13:36:03 +0200799
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400800 block->u.Packet_Uplink_Ack_Nack.MESSAGE_TYPE = 0x9; // Packet Downlink Assignment
801 block->u.Packet_Uplink_Ack_Nack.PAGE_MODE = 0x0; // Normal Paging
802 block->u.Packet_Uplink_Ack_Nack.UPLINK_TFI = tbf->tfi; // Uplink TFI
Andreas Eversberge6228b32012-07-03 13:36:03 +0200803
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400804 block->u.Packet_Uplink_Ack_Nack.UnionType = 0x0; // PU_AckNack_GPRS = on
Andreas Eversberg499ff412012-10-03 14:21:36 +0200805 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 +0400806 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Ack_Nack_Description.FINAL_ACK_INDICATION = final; // FINAL ACK INDICATION
807 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 +0200808 // RECEIVE_BLOCK_BITMAP
809 for (i = 0, bbn = (tbf->dir.ul.v_r - 64) & mod_sns_half; i < 64;
810 i++, bbn = (bbn + 1) & mod_sns_half) {
811 bit = tbf->dir.ul.v_n[bbn];
812 if (bit == 0)
813 bit = ' ';
814 show_v_n[i] = bit;
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400815 if (bit == 'R')
816 rbb = (rbb << 1)|1;
817 else
818 rbb = (rbb << 1);
819 if((i%8) == 7)
820 {
821 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Ack_Nack_Description.RECEIVED_BLOCK_BITMAP[i/8] = rbb;
822 rbb = 0;
823 }
Ivan Kluchnikove3594232012-06-07 01:12:29 +0400824 }
Andreas Eversberge6228b32012-07-03 13:36:03 +0200825 show_v_n[64] = '\0';
826 LOGP(DRLCMACUL, LOGL_DEBUG, "- V(N): \"%s\" R=Received "
827 "N=Not-Received\n", show_v_n);
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400828
829 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.UnionType = 0x0; // Fixed Allocation Dummy = on
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400830 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.u.FixedAllocationDummy = 0x0; // Fixed Allocation Dummy
Ivan Kluchnikovef7f28c2012-07-12 14:49:15 +0400831 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Exist_AdditionsR99 = 0x0; // AdditionsR99 = off
Ivan Kluchnikov34460b82012-06-27 18:41:04 +0400832
833 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 +0400834 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 +0400835 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Common_Uplink_Ack_Nack_Data.Exist_Packet_Timing_Advance = 0x0;
836 block->u.Packet_Uplink_Ack_Nack.u.PU_AckNack_GPRS_Struct.Common_Uplink_Ack_Nack_Data.Exist_Extension_Bits = 0x0;
837 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 +0400838}
839
Andreas Eversberg2b914642012-07-19 13:06:26 +0200840unsigned write_packet_paging_request(bitvec * dest)
841{
842 unsigned wp = 0;
843
844 bitvec_write_field(dest, wp,0x1,2); // Payload Type
845 bitvec_write_field(dest, wp,0x0,3); // No polling
846 bitvec_write_field(dest, wp,0x0,3); // Uplink state flag
847 bitvec_write_field(dest, wp,0x22,6); // MESSAGE TYPE
848
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200849 bitvec_write_field(dest, wp,0x0,2); // Page Mode
850
Andreas Eversberg2b914642012-07-19 13:06:26 +0200851 bitvec_write_field(dest, wp,0x0,1); // No PERSISTENCE_LEVEL
852 bitvec_write_field(dest, wp,0x0,1); // No NLN
853
854 return wp;
855}
856
857unsigned write_repeated_page_info(bitvec * dest, unsigned& wp, uint8_t len,
858 uint8_t *identity, uint8_t chan_needed)
859{
Andreas Eversberg3b7461c2012-07-20 11:19:59 +0200860 bitvec_write_field(dest, wp,0x1,1); // Repeated Page info exists
861
Andreas Eversberg2b914642012-07-19 13:06:26 +0200862 bitvec_write_field(dest, wp,0x1,1); // RR connection paging
863
864 if ((identity[0] & 0x07) == 4) {
865 bitvec_write_field(dest, wp,0x0,1); // TMSI
866 identity++;
867 len--;
868 } else {
869 bitvec_write_field(dest, wp,0x0,1); // MI
870 bitvec_write_field(dest, wp,len,4); // MI len
871 }
872 while (len) {
873 bitvec_write_field(dest, wp,*identity++,8); // MI data
874 len--;
875 }
876 bitvec_write_field(dest, wp,chan_needed,2); // CHANNEL_NEEDED
877 bitvec_write_field(dest, wp,0x0,1); // No eMLPP_PRIORITY
878
879 return wp;
880}
881
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400882/* Send Uplink unit-data to SGSN. */
Andreas Eversberg3e372d52012-07-06 09:28:15 +0200883int gprs_rlcmac_tx_ul_ud(gprs_rlcmac_tbf *tbf)
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400884{
Andreas Eversberg9a913462012-09-23 06:42:38 +0200885 uint8_t qos_profile[3];
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400886 struct msgb *llc_pdu;
Andreas Eversberg5dac2f02012-06-27 15:52:04 +0200887 unsigned msg_len = NS_HDR_LEN + BSSGP_HDR_LEN + tbf->llc_index;
Holger Hans Peter Freyther90692f92013-07-13 12:51:16 +0200888 struct bssgp_bvc_ctx *bctx = gprs_bssgp_pcu_current_bctx();
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400889
Andreas Eversbergb0c7ea72012-07-13 14:46:03 +0200890 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 +0200891 if (!bctx) {
892 LOGP(DBSSGP, LOGL_ERROR, "No bctx\n");
893 return -EIO;
894 }
Ivan Kluchnikovc320d862012-03-18 15:04:48 +0400895
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400896 llc_pdu = msgb_alloc_headroom(msg_len, msg_len,"llc_pdu");
Andreas Eversberg9a913462012-09-23 06:42:38 +0200897 uint8_t *buf = msgb_push(llc_pdu, TL16V_GROSS_LEN(sizeof(uint8_t)*tbf->llc_index));
898 tl16v_put(buf, BSSGP_IE_LLC_PDU, sizeof(uint8_t)*tbf->llc_index, tbf->llc_frame);
899 qos_profile[0] = QOS_PROFILE >> 16;
900 qos_profile[1] = QOS_PROFILE >> 8;
901 qos_profile[2] = QOS_PROFILE;
902 bssgp_tx_ul_ud(bctx, tbf->tlli, qos_profile, llc_pdu);
Ivan Kluchnikov8ee60512012-03-05 19:24:57 +0400903
Andreas Eversberg3e372d52012-07-06 09:28:15 +0200904 return 0;
Ivan Kluchnikovc7e7f682012-06-29 22:53:15 +0400905}
Andreas Eversberg8c3680d2012-10-08 12:30:56 +0200906
907int gprs_rlcmac_paging_request(uint8_t *ptmsi, uint16_t ptmsi_len,
908 const char *imsi)
909{
910 LOGP(DRLCMAC, LOGL_NOTICE, "TX: [PCU -> BTS] Paging Request (CCCH)\n");
911 bitvec *paging_request = bitvec_alloc(23);
912 bitvec_unhex(paging_request, "2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b2b");
913 int plen = write_paging_request(paging_request, ptmsi, ptmsi_len);
914 pcu_l1if_tx_pch(paging_request, plen, (char *)imsi);
915 bitvec_free(paging_request);
916
917 return 0;
918}
Andreas Eversberga004e6a2013-05-13 16:45:21 +0200919
920
921/*
922 * timing advance memory
923 */
924
925/* enable to debug timing advance memory */
926//#define DEBUG_TA
927
928static LLIST_HEAD(gprs_rlcmac_ta_list);
929static int gprs_rlcmac_ta_num = 0;
930
931struct gprs_rlcmac_ta {
932 struct llist_head list;
933 uint32_t tlli;
934 uint8_t ta;
935};
936
937/* remember timing advance of a given TLLI */
938int remember_timing_advance(uint32_t tlli, uint8_t ta)
939{
940 struct gprs_rlcmac_ta *ta_entry;
941
942 /* check for existing entry */
943 llist_for_each_entry(ta_entry, &gprs_rlcmac_ta_list, list) {
944 if (ta_entry->tlli == tlli) {
945#ifdef DEBUG_TA
946 fprintf(stderr, "update %08x %d\n", tlli, ta);
947#endif
948 ta_entry->ta = ta;
949 /* relink to end of list */
950 llist_del(&ta_entry->list);
951 llist_add_tail(&ta_entry->list, &gprs_rlcmac_ta_list);
952 return 0;
953 }
954 }
955
956#ifdef DEBUG_TA
957 fprintf(stderr, "remember %08x %d\n", tlli, ta);
958#endif
959 /* if list is full, remove oldest entry */
960 if (gprs_rlcmac_ta_num == 30) {
961 ta_entry = llist_entry(gprs_rlcmac_ta_list.next,
962 struct gprs_rlcmac_ta, list);
963 llist_del(&ta_entry->list);
964 talloc_free(ta_entry);
965 gprs_rlcmac_ta_num--;
966 }
967
968 /* create new TA entry */
969 ta_entry = talloc_zero(tall_pcu_ctx, struct gprs_rlcmac_ta);
970 if (!ta_entry)
971 return -ENOMEM;
972
973 ta_entry->tlli = tlli;
974 ta_entry->ta = ta;
975 llist_add_tail(&ta_entry->list, &gprs_rlcmac_ta_list);
976 gprs_rlcmac_ta_num++;
977
978 return 0;
979}
980
981int recall_timing_advance(uint32_t tlli)
982{
983 struct gprs_rlcmac_ta *ta_entry;
984 uint8_t ta;
985
986 llist_for_each_entry(ta_entry, &gprs_rlcmac_ta_list, list) {
987 if (ta_entry->tlli == tlli) {
988 ta = ta_entry->ta;
989#ifdef DEBUG_TA
990 fprintf(stderr, "recall %08x %d\n", tlli, ta);
991#endif
992 return ta;
993 }
994 }
995#ifdef DEBUG_TA
996 fprintf(stderr, "no entry for %08x\n", tlli);
997#endif
998
999 return -EINVAL;
1000}
1001
1002int flush_timing_advance(void)
1003{
1004 struct gprs_rlcmac_ta *ta_entry;
1005 int count = 0;
1006
1007 while (!llist_empty(&gprs_rlcmac_ta_list)) {
1008 ta_entry = llist_entry(gprs_rlcmac_ta_list.next,
1009 struct gprs_rlcmac_ta, list);
1010#ifdef DEBUG_TA
1011 fprintf(stderr, "flush entry %08x %d\n", ta_entry->tlli,
1012 ta_entry->ta);
1013#endif
1014 llist_del(&ta_entry->list);
1015 talloc_free(ta_entry);
1016 count++;
1017 }
1018 gprs_rlcmac_ta_num = 0;
1019
1020 return count;
1021}
1022