blob: d2174d7b5900e5d3b1c4fd4df680d6bcf4e6ff83 [file] [log] [blame]
Pau Espin Pedrole8d7d162020-04-29 19:07:36 +02001/* TITAN REW encode/decode definitions for 3GPP TS 44.060 RLC/MAC Blocks */
2
3/* (C) 2017-2018 Harald Welte <laforge@gnumonks.org>
4 * (C) 2020 by sysmocom s.f.m.c. GmbH <info@sysmocom.de>
5 * All rights reserved.
6 *
7 * Released under the terms of GNU General Public License, Version 2 or
8 * (at your option) any later version.
9 *
10 * SPDX-License-Identifier: GPL-2.0-or-later
11 */
12
13module RLCMAC_Templates {
14 import from General_Types all;
15 import from Osmocom_Types all;
16 import from GSM_Types all;
17 import from RLCMAC_CSN1_Types all;
18 import from RLCMAC_CSN1_Templates all;
19 import from RLCMAC_Types all;
20
21 /* TS 44.060 10.4.5 */
22 function f_rrbp_fn_delay(MacRrbp rrbp) return uint32_t {
23 select (rrbp) {
24 case (RRBP_Nplus13_mod_2715648) { return 13; }
25 case (RRBP_Nplus17_or_18_mod_2715648) { return 17; }
26 case (RRBP_Nplus21_or_22_mod_2715648) { return 21; }
27 case (RRBP_Nplus26_mod_2715648) { return 26; }
28 }
29 return 0;
30 }
31
Pau Espin Pedrol27d6af52020-04-30 20:13:32 +020032 function f_rlcmac_mcs2headertype(CodingScheme mcs) return EgprsHeaderType {
33 select (mcs) {
34 case (MCS_0) { return RLCMAC_HDR_TYPE_3; }
35 case (MCS_1) { return RLCMAC_HDR_TYPE_3; }
36 case (MCS_2) { return RLCMAC_HDR_TYPE_3; }
37 case (MCS_3) { return RLCMAC_HDR_TYPE_3; }
38 case (MCS_4) { return RLCMAC_HDR_TYPE_3; }
39 case (MCS_5) { return RLCMAC_HDR_TYPE_2; }
40 case (MCS_6) { return RLCMAC_HDR_TYPE_2; }
41 case (MCS_7) { return RLCMAC_HDR_TYPE_1; }
42 case (MCS_8) { return RLCMAC_HDR_TYPE_1; }
43 case (MCS_9) { return RLCMAC_HDR_TYPE_1; }
44 }
45 return RLCMAC_HDR_TYPE_3;
46 }
47
Pau Espin Pedrole8d7d162020-04-29 19:07:36 +020048 function f_rlcmac_cs_mcs2block_len(CodingScheme cs_mcs) return uint32_t {
49 select (cs_mcs) {
50 case (CS_1) { return 23; }
51 case (CS_2) { return 34; }
52 case (CS_3) { return 40; }
53 case (CS_4) { return 54; }
54 case (MCS_1) { return 27; }
55 case (MCS_2) { return 33; }
56 case (MCS_3) { return 42; }
57 case (MCS_4) { return 49; }
58 case (MCS_5) { return 61; }
59 case (MCS_6) { return 79; }
60 case (MCS_7) { return 119; }
61 case (MCS_8) { return 143; }
62 case (MCS_9) { return 155; }
63 }
64 return 0;
65 }
66
67 function f_rlcmac_block_len2cs_mcs(uint32_t len) return CodingScheme {
68 select (len) {
69 case (23) { return CS_1; }
70 case (34) { return CS_2; }
71 case (40) { return CS_3; }
72 case (54) { return CS_4; }
73 case (27) { return MCS_1; }
74 case (33) { return MCS_2; }
75 case (42) { return MCS_3; }
76 case (49) { return MCS_4; }
77 case (60) { return MCS_5; }
78 case (61) { return MCS_5; }
79 case (78) { return MCS_6; }
80 case (79) { return MCS_6; }
81 case (118) { return MCS_7; }
82 case (119) { return MCS_7; }
83 case (142) { return MCS_8; }
84 case (143) { return MCS_8; }
85 case (154) { return MCS_9; }
86 case (155) { return MCS_9; }
87 }
88 return CS_1;
89 }
90
Pau Espin Pedrol27d6af52020-04-30 20:13:32 +020091 /* Coding and Puncturing Scheme indicator field for Header type 1 in EGPRS TBF or EC TBF or downlink EGPRS2 TBF */
92 function f_rlcmac_cps_htype1_to_mcs(uint3_t cps) return CodingScheme {
93 var CodingSchemeArray egprs_Header_type1_coding_puncturing_scheme_to_mcs := {
94 MCS_9 /* 0x00, "(MCS-9/P1 ; MCS-9/P1)" */,
95 MCS_9 /* 0x01, "(MCS-9/P1 ; MCS-9/P2)" */,
96 MCS_9 /* 0x02, "(MCS-9/P1 ; MCS-9/P3)" */,
97 MCS_0 /* 0x03, "reserved" */,
98 MCS_9 /* 0x04, "(MCS-9/P2 ; MCS-9/P1)" */,
99 MCS_9 /* 0x05, "(MCS-9/P2 ; MCS-9/P2)" */,
100 MCS_9 /* 0x06, "(MCS-9/P2 ; MCS-9/P3)" */,
101 MCS_0 /* 0x07, "reserved" */,
102 MCS_9 /* 0x08, "(MCS-9/P3 ; MCS-9/P1)" */,
103 MCS_9 /* 0x09, "(MCS-9/P3 ; MCS-9/P2)" */,
104 MCS_9 /* 0x0A, "(MCS-9/P3 ; MCS-9/P3)" */,
105 MCS_8 /* 0x0B, "(MCS-8/P1 ; MCS-8/P1)" */,
106 MCS_8 /* 0x0C, "(MCS-8/P1 ; MCS-8/P2)" */,
107 MCS_8 /* 0x0D, "(MCS-8/P1 ; MCS-8/P3)" */,
108 MCS_8 /* 0x0E, "(MCS-8/P2 ; MCS-8/P1)" */,
109 MCS_8 /* 0x0F, "(MCS-8/P2 ; MCS-8/P2)" */,
110 MCS_8 /* 0x10, "(MCS-8/P2 ; MCS-8/P3)" */,
111 MCS_8 /* 0x11, "(MCS-8/P3 ; MCS-8/P1)" */,
112 MCS_8 /* 0x12, "(MCS-8/P3 ; MCS-8/P2)" */,
113 MCS_8 /* 0x13, "(MCS-8/P3 ; MCS-8/P3)" */,
114 MCS_7 /* 0x14, "(MCS-7/P1 ; MCS-7/P1)" */,
115 MCS_7 /* 0x15, "(MCS-7/P1 ; MCS-7/P2)" */,
116 MCS_7 /* 0x16, "(MCS-7/P1 ; MCS-7/P3)" */,
117 MCS_7 /* 0x17, "(MCS-7/P2 ; MCS-7/P1)" */,
118 MCS_7 /* 0x18, "(MCS-7/P2 ; MCS-7/P2)" */,
119 MCS_7 /* 0x19, "(MCS-7/P2 ; MCS-7/P3)" */,
120 MCS_7 /* 0x1A, "(MCS-7/P3 ; MCS-7/P1)" */,
121 MCS_7 /* 0x1B, "(MCS-7/P3 ; MCS-7/P2)" */,
122 MCS_7 /* 0x1C, "(MCS-7/P3 ; MCS-7/P3)" */,
123 MCS_0 /* 0x1D, "reserved" */,
124 MCS_0 /* 0x1E, "reserved" */,
125 MCS_0 /* 0x1F, "reserved" */
126 };
127 return egprs_Header_type1_coding_puncturing_scheme_to_mcs[cps];
128 }
129
130 /* Coding and Puncturing Scheme indicator field for Header type 2 in (EC-)EGPRS TBF or uplink EGPRS2-A TBF */
131 function f_rlcmac_cps_htype2_to_mcs(uint3_t cps) return CodingScheme {
132 var CodingSchemeArray egprs_Header_type2_coding_puncturing_scheme_to_mcs := {
133 MCS_6 /* {0x00, "MCS-6/P1"} */,
134 MCS_6 /* {0x01, "MCS-6/P2"} */,
135 MCS_6 /* {0x02, "MCS-6/P1 with 6 octet padding"} */,
136 MCS_6 /* {0x03, "MCS-6/P2 with 6 octet padding "} */,
137 MCS_5 /* {0x04, "MCS-5/P1"} */,
138 MCS_5 /* {0x05, "MCS-5/P2"} */,
139 MCS_5 /* {0x06, "MCS-6/P1 with 10 octet padding "} */,
140 MCS_5 /* {0x07, "MCS-6/P2 with 10 octet padding "} */
141 };
142 return egprs_Header_type2_coding_puncturing_scheme_to_mcs[cps];
143 }
144
145 /* Coding and Puncturing Scheme indicator field for Header type 3 */
146 function f_rlcmac_cps_htype3_to_mcs(uint3_t cps) return CodingScheme {
147 var CodingSchemeArray egprs_Header_type3_coding_puncturing_scheme_to_mcs := {
148 MCS_4 /* {0x00, "MCS-4/P1"} */,
149 MCS_4 /* {0x01, "MCS-4/P2"} */,
150 MCS_4 /* {0x02, "MCS-4/P3"} */,
151 MCS_3 /* {0x03, "MCS-3/P1"} */,
152 MCS_3 /* {0x04, "MCS-3/P2"} */,
153 MCS_3 /* {0x05, "MCS-3/P3"} */,
154 MCS_3 /* {0x06, "MCS-3/P1 with padding"} */,
155 MCS_3 /* {0x07, "MCS-3/P2 with padding"} */,
156 MCS_3 /* {0x08, "MCS-3/P3 with padding"} */,
157 MCS_2 /* {0x09, "MCS-2/P1"} */,
158 MCS_2 /* {0x0A, "MCS-2/P2"} */,
159 MCS_1 /* {0x0B, "MCS-1/P1"} */,
160 MCS_1 /* {0x0C, "MCS-1/P2"} */,
161 MCS_2 /* {0x0D, "MCS-2/P1 with padding"} */,
162 MCS_2 /* {0x0E, "MCS-2/P2 with padding"} */,
163 MCS_0 /* {0x0F, "MCS-0"} */
164 };
165 return egprs_Header_type3_coding_puncturing_scheme_to_mcs[cps];
166 }
167
168 function f_rlcmac_cps_htype_to_mcs(uint3_t cps, EgprsHeaderType htype) return CodingScheme {
169 select (htype) {
170 case (RLCMAC_HDR_TYPE_1) { return f_rlcmac_cps_htype1_to_mcs(cps); }
171 case (RLCMAC_HDR_TYPE_2) { return f_rlcmac_cps_htype2_to_mcs(cps); }
172 case (RLCMAC_HDR_TYPE_3) { return f_rlcmac_cps_htype3_to_mcs(cps); }
173 }
174 //TODO: return error here.
175 return CS_1;
176 }
177
178 function f_rlcmac_mcs_to_cps_htype1(CodingScheme mcs, uint2_t part, boolean with_padding) return uint5_t {
179 //TODO: implement similar to f_rlcmac_mcs_to_cps_htype3()
180 //TODO: return error here.
181 return 0;
182 }
183
184 function f_rlcmac_mcs_to_cps_htype2(CodingScheme mcs, uint2_t part, boolean with_padding) return uint5_t {
185 //TODO: implement similar to f_rlcmac_mcs_to_cps_htype3()
186 //TODO: return error here.
187 return 0;
188 }
189
190 function f_rlcmac_mcs_to_cps_htype3(CodingScheme mcs, uint2_t part, boolean with_padding) return uint5_t {
191 select (mcs) {
192 case (MCS_4) {
193 select (part) {
194 case (1) { return 0; /* {0x00, "MCS-4/P1"} */ }
195 case (2) { return 1; /* {0x01, "MCS-4/P2"} */ }
196 case (3) { return 2; /* {0x01, "MCS-4/P2"} */ }
197 }
198 }
199 case (MCS_3) {
200 if (not with_padding) {
201 select (part) {
202 case (1) { return 3; /* {0x03, "MCS-3/P1"} */ }
203 case (2) { return 4; /* {0x04, "MCS-3/P2"} */ }
204 case (3) { return 5; /* {0x05, "MCS-3/P3"} */ }
205 }
206 } else {
207 select (part) {
208 case (1) { return 6; /* {0x06, "MCS-3/P1 with padding"} */ }
209 case (2) { return 7; /* {0x07, "MCS-3/P2 with padding"} */ }
210 case (3) { return 8; /* {0x08, "MCS-3/P3 with padding"} */ }
211 }
212 }
213 }
214 case (MCS_2) {
215 if (not with_padding) {
216 select (part) {
217 case (1) { return 9; /* {0x09, "MCS-2/P1"} */ }
218 case (2) { return 10; /* {0x0A, "MCS-2/P2"} */ }
219 }
220 } else {
221 select (part) {
222 case (1) { return 13; /* {0x0D, "MCS-2/P1 with padding"} */ }
223 case (2) { return 14; /* {0x0E, "MCS-2/P2 with padding"} */}
224 }
225 }
226 }
227 case (MCS_1) {
228 select (part) {
229 case (1) { return 11; /* {0x0B, "MCS-1/P1"} */ }
230 case (2) { return 12; /* {0x0C, "MCS-1/P2"} */ }
231 }
232 }
233 case (MCS_0) { return 15; /* {0x0F, "MCS-0"} */ }
234 }
235 //TODO: return error here.
236 return 0;
237 }
238
239 function f_rlcmac_mcs_to_cps(CodingScheme mcs, uint2_t part, boolean with_padding := false) return uint5_t {
240
241 var EgprsHeaderType htype := f_rlcmac_mcs2headertype(mcs);
242 select (htype) {
243 case (RLCMAC_HDR_TYPE_1) { return f_rlcmac_mcs_to_cps_htype1(mcs, part, with_padding); }
244 case (RLCMAC_HDR_TYPE_2) { return f_rlcmac_mcs_to_cps_htype2(mcs, part, with_padding); }
245 case (RLCMAC_HDR_TYPE_3) { return f_rlcmac_mcs_to_cps_htype3(mcs, part, with_padding); }
246 }
247 //TODO: return error here.
248 return 0;
249 }
250
Pau Espin Pedrole8d7d162020-04-29 19:07:36 +0200251 template (value) RlcmacUlBlock ts_RLC_UL_CTRL_ACK(RlcmacUlCtrlMsg ctrl,
252 MacPayloadType pt := MAC_PT_RLCMAC_NO_OPT,
253 boolean retry := false) := {
254 ctrl := {
255 mac_hdr := {
256 payload_type := pt,
257 spare := '00000'B,
258 retry := retry
259 },
260 payload := ctrl
261 }
262 }
263
264 /* Send Template for Downlink ACK/NACK */
265 template RlcmacUlBlock ts_RLCMAC_DL_ACK_NACK(template uint5_t tfi, AckNackDescription andesc, boolean retry := false) := {
266 ctrl := {
267 mac_hdr := {
268 payload_type := MAC_PT_RLCMAC_NO_OPT,
269 spare := '00000'B,
270 retry := retry
271 },
272 payload := {
273 msg_type := PACKET_DL_ACK_NACK,
274 u := {
275 dl_ack_nack := {
276 dl_tfi := tfi,
277 ack_nack_desc := andesc,
278 chreq_desc_presence := '0'B,
279 chreq_desc := omit,
280 ch_qual_rep := c_ChQualRep_default
281 }
282 }
283 }
284 }
285 }
286
287 /* Template for uplink Data block */
288 template RlcmacUlBlock t_RLCMAC_UL_DATA(template uint5_t tfi, template uint4_t cv, template uint7_t bsn,
289 template LlcBlocks blocks := {}, template boolean stall := false) := {
290 data := {
291 mac_hdr := {
292 payload_type := MAC_PT_RLC_DATA,
293 countdown := cv,
294 stall_ind := false,
295 retry := false,
296 spare := '0'B,
297 pfi_ind := false,
298 tfi := tfi,
299 tlli_ind := false,
300 bsn := bsn,
301 e := false
302 },
303 tlli := omit,
304 pfi := omit,
305 blocks := blocks
306 }
307 }
308 template RlcmacUlBlock t_RLCMAC_UL_DATA_TLLI(template uint5_t tfi, template uint4_t cv, template uint7_t bsn,
309 template LlcBlocks blocks := {}, template boolean stall := false, template GprsTlli tlli) := {
310 data := {
311 mac_hdr := {
312 payload_type := MAC_PT_RLC_DATA,
313 countdown := cv,
314 stall_ind := false,
315 retry := false,
316 spare := '0'B,
317 pfi_ind := false,
318 tfi := tfi,
319 tlli_ind := true,
320 bsn := bsn,
321 e := false
322 },
323 tlli := tlli,
324 pfi := omit,
325 blocks := blocks
326 }
327 }
328
329 template DlMacHeader t_RLCMAC_DlMacH(template MacPayloadType pt, template MacRrbp rrbp, template
330uint3_t usf) := {
331 payload_type := pt,
332 rrbp := rrbp,
333 rrbp_valid := ispresent(rrbp),
334 usf := usf
335 }
336
337 template RlcmacDlBlock tr_RLCMAC_DUMMY_CTRL(template uint3_t usf := ?, template PageMode page_mode := ?) := {
338 ctrl := {
339 mac_hdr := {
340 payload_type := (MAC_PT_RLCMAC_NO_OPT, MAC_PT_RLCMAC_OPT),
341 rrbp:= ?,
342 rrbp_valid := ?,
343 usf := usf
344 },
345 opt := *,
346 payload := {
347 msg_type := PACKET_DL_DUMMY_CTRL,
348 u := {
349 dl_dummy := {
350 page_mode := page_mode,
351 persistence_levels_present := ?,
352 persistence_levels := *
353 }
354 }
355 }
356 }
357 }
358
359 template RlcmacDlBlock tr_RLCMAC_DL_PACKET_ASS(template uint3_t usf := ?) := {
360 ctrl := {
361 mac_hdr := {
362 payload_type := (MAC_PT_RLCMAC_NO_OPT, MAC_PT_RLCMAC_OPT),
363 rrbp:= ?,
364 rrbp_valid := ?,
365 usf := usf
366 },
367 opt := *,
368 payload := {
369 msg_type := PACKET_DL_ASSIGNMENT,
370 u := {
371 dl_assignment := {
372 page_mode := ?,
373 pres1 := ?,
374 persistence_levels := *,
375 tfi_or_tlli := ?
376 }
377 }
378 }
379 }
380 }
381
382 template RlcmacDlBlock tr_RLCMAC_UL_PACKET_ASS(template uint3_t usf := ?) := {
383 ctrl := {
384 mac_hdr := {
385 payload_type := (MAC_PT_RLCMAC_NO_OPT, MAC_PT_RLCMAC_OPT),
386 rrbp:= ?,
387 rrbp_valid := ?,
388 usf := usf
389 },
390 opt := *,
391 payload := {
392 msg_type := PACKET_UL_ASSIGNMENT,
393 u := {
394 ul_assignment := {
395 page_mode := ?,
396 persistence_levels_present := ?,
397 persistence_levels := *,
398 identity := ?,
399 is_egprs := ?, /* msg escape */
400 gprs := *
401 }
402 }
403 }
404 }
405 }
406
407
408 /* Receive Template for Uplink ACK/NACK */
409 template RlcmacDlBlock tr_RLCMAC_UL_ACK_NACK(template uint5_t ul_tfi, template GprsTlli tlli := ?) := {
410 ctrl := {
411 mac_hdr := {
412 payload_type := (MAC_PT_RLCMAC_NO_OPT, MAC_PT_RLCMAC_OPT),
413 rrbp:= ?,
414 rrbp_valid := ?,
415 usf := ?
416 },
417 opt := *,
418 payload := {
419 msg_type := PACKET_UL_ACK_NACK,
420 u := {
421 ul_ack_nack := {
422 page_mode := ?,
423 msg_excape := ?,
424 uplink_tfi := ul_tfi,
425 is_egprs := '0'B,
426 gprs := {
427 ch_coding_cmd := ?,
428 ack_nack_desc := ?,
429 cont_res_tlli_present := ?,
430 cont_res_tlli := tlli,
431 pkt_ta_present := ?,
432 pkt_ta := *,
433 pwr_ctrl_present := ?,
434 pwr_ctrl := *
435 }
436 }
437 }
438 }
439 }
440 }
441
442 template RlcmacDlBlock tr_RLCMAC_PACKET_PAG_REQ(template uint3_t usf := ?) := {
443 ctrl := {
444 mac_hdr := {
445 payload_type := MAC_PT_RLCMAC_NO_OPT,
446 rrbp:= ?,
447 rrbp_valid := ?,
448 usf := usf
449 },
450 opt := *,
451 payload := {
452 msg_type := PACKET_PAGING_REQUEST,
453 u := {
454 paging := {
455 page_mode := ?,
456 persistence_levels_present := ?,
457 persistence_levels := *,
458 nln_present := ?,
459 nln := *,
460 repeated_pageinfo_present := ?,
461 repeated_pageinfo := *
462 }
463 }
464 }
465 }
466 }
467
468 template RlcmacDlBlock tr_RLCMAC_DATA_RRBP := {
469 data := {
470 mac_hdr := {
471 mac_hdr := {
472 payload_type := MAC_PT_RLC_DATA,
473 rrbp := ?,
474 rrbp_valid := true,
475 usf := ?
476 },
477 hdr_ext := ?
478 },
479 blocks := ?
480 }
481 }
482
483 template RlcmacDlBlock tr_RLCMAC_DATA_EGPRS := {
484 data_egprs := {
485 mac_hdr := ?,
486 fbi := ?,
487 e := ?,
488 blocks := ?
489 }
490 }
491
492 /* Template for Uplink MAC Control Header */
493 template UlMacCtrlHeader t_RLCMAC_UlMacCtrlH(template MacPayloadType pt, template boolean retry := false) := {
494 payload_type := pt,
495 spare := '00000'B,
496 retry := retry
497 }
498
499 /* Template for Uplink Control ACK */
500 template RlcmacUlBlock ts_RLCMAC_CTRL_ACK(GprsTlli tlli, CtrlAck ack := MS_RCVD_TWO_RLC_SAME_RTI_DIFF_RBSN) := {
501 ctrl := {
502 mac_hdr := t_RLCMAC_UlMacCtrlH(MAC_PT_RLCMAC_NO_OPT),
503 payload := {
504 msg_type := PACKET_CONTROL_ACK,
505 u := {
506 ctrl_ack := {
507 tlli := tlli,
508 ctrl_ack := ack
509 }
510 }
511 }
512 }
513 }
514
515 /* Template for a LlcBlock (part of a LLC frame inside RlcMac?lDataBlock */
516 template LlcBlock t_RLCMAC_LLCBLOCK(octetstring data, boolean more := false, boolean e := true) := {
517 /* let encoder figure out the header */
518 hdr := omit,
519 payload := data
520 }
521
522} with { encode "RAW"; variant "FIELDORDER(msb)" }