blob: 577c9bdf71df448c4f1267bf390ed11aaa413f0c [file] [log] [blame]
Harald Welte34b5a952019-05-27 11:54:11 +02001/* LAPDm definitions according to 3GPP TS 44.006
2 * (C) 2017-2018 by Harald Welte <laforge@gnumonks.org>
3 * All rights reserved.
4 *
5 * Released under the terms of GNU General Public License, Version 2 or
6 * (at your option) any later version.
7 *
8 * SPDX-License-Identifier: GPL-2.0-or-later
9 */
10
Harald Weltef6543322017-07-16 07:35:10 +020011module LAPDm_Types {
12
13 import from General_Types all;
14 import from Osmocom_Types all;
15
16 type uint3_t LapdmSapi;
17 type BIT2 LapdmSBits;
18 type BIT3 LapdmUBits;
19 type BIT2 LapdmU2Bits;
20
Harald Welte0472ab42018-03-12 15:02:26 +010021 /* 44.006 6.3.2 */
22 const boolean cr_MO_CMD := false;
23 const boolean cr_MO_RSP := true;
24 const boolean cr_MT_CMD := true;
25 const boolean cr_MT_RSP := false;
26
Harald Weltef6543322017-07-16 07:35:10 +020027 /* TS 44.006 Figure 4 */
28 type record LapdmAddressField {
29 BIT1 spare,
Harald Weltec2a5c072017-07-17 20:58:32 +020030 uint2_t lpd,
Harald Weltef6543322017-07-16 07:35:10 +020031 LapdmSapi sapi,
32 boolean c_r,
33 boolean ea
Harald Weltec2a5c072017-07-17 20:58:32 +020034 } with { variant "FIELDORDER(msb)" };
Harald Weltef6543322017-07-16 07:35:10 +020035
Harald Welte9e4725d2017-07-16 23:18:09 +020036 template LapdmAddressField tr_LapdmAddr(template LapdmSapi sapi, template boolean c_r) := {
Harald Weltef6543322017-07-16 07:35:10 +020037 spare := '0'B,
38 lpd := 0,
39 sapi := sapi,
40 c_r := c_r,
41 ea := true
42 };
43
Harald Welted879bd92018-03-12 15:01:23 +010044 template (value) LapdmAddressField ts_LapdmAddr(LapdmSapi sapi, boolean c_r) := {
45 spare := '0'B,
46 lpd := 0,
47 sapi := sapi,
48 c_r := c_r,
49 ea := true
50 };
51
Harald Weltef6543322017-07-16 07:35:10 +020052 type record LapdmCtrlI {
Harald Welte1bbe0b72018-05-10 18:28:15 +020053 uint3_t n_r,
Harald Weltec2a5c072017-07-17 20:58:32 +020054 boolean p,
Harald Welte1bbe0b72018-05-10 18:28:15 +020055 uint3_t n_s,
56 BIT1 spare ('0'B)
57 } with { variant "FIELDORDER(msb)" };
Harald Weltef6543322017-07-16 07:35:10 +020058
59 type record LapdmCtrlS {
Harald Welte1bbe0b72018-05-10 18:28:15 +020060 uint3_t n_r,
Harald Weltec2a5c072017-07-17 20:58:32 +020061 boolean p_f,
Harald Welte1bbe0b72018-05-10 18:28:15 +020062 LapdmSBits s,
63 BIT2 spare ('01'B)
64 } with { variant "FIELDORDER(msb)" };
Harald Weltef6543322017-07-16 07:35:10 +020065
66 type record LapdmCtrlU {
Harald Welte1bbe0b72018-05-10 18:28:15 +020067 LapdmUBits u,
Harald Weltec2a5c072017-07-17 20:58:32 +020068 boolean p_f,
Harald Welte1bbe0b72018-05-10 18:28:15 +020069 LapdmU2Bits u2,
70 BIT2 spare ('11'B)
71 } with { variant "FIELDORDER(msb)" };
Harald Weltef6543322017-07-16 07:35:10 +020072
73 /* TS 44.006 Table 3 */
74 type union LapdmCtrl {
Harald Weltef6543322017-07-16 07:35:10 +020075 LapdmCtrlS s,
Harald Weltec2a5c072017-07-17 20:58:32 +020076 LapdmCtrlU u,
77 LapdmCtrlI i,
78 uint8_t other
79 } with { variant "TAG(u, spare = '11'B;
Harald Weltef6543322017-07-16 07:35:10 +020080 s, spare = '01'B;
Harald Weltec2a5c072017-07-17 20:58:32 +020081 i, spare = '0'B;
82 other, OTHERWISE)" };
83 /* )" }; */
84
85 /* TS 44.006 Table 4 */
Harald Weltef6543322017-07-16 07:35:10 +020086
Harald Welted879bd92018-03-12 15:01:23 +010087 template LapdmCtrl tr_LapdmCtrlS := {
Harald Welte1bbe0b72018-05-10 18:28:15 +020088 s := { n_r := ?, p_f := ?, s := ?, spare := '01'B }
Harald Weltef6543322017-07-16 07:35:10 +020089 };
90
Harald Welted879bd92018-03-12 15:01:23 +010091 template LapdmCtrl tr_LapdmCtrlU := {
Harald Welte1bbe0b72018-05-10 18:28:15 +020092 u := { u := ?, p_f := ?, u2 := ?, spare := '11'B }
Harald Weltef6543322017-07-16 07:35:10 +020093 };
94
95 /* TS 44.006 Table 4 */
Harald Welted879bd92018-03-12 15:01:23 +010096 template LapdmCtrl tr_LapdmCtrlI(template uint3_t nr, template uint3_t ns, template boolean p) := {
Harald Welte1bbe0b72018-05-10 18:28:15 +020097 i := { n_r := nr, p := p, n_s := ns, spare := '0'B }
98 };
99 template (value) LapdmCtrl ts_LapdmCtrlI(uint3_t nr, uint3_t ns, boolean p) := {
100 i := { n_r := nr, p := p, n_s := ns, spare := '0'B }
Harald Weltef6543322017-07-16 07:35:10 +0200101 };
102
Harald Welte1bbe0b72018-05-10 18:28:15 +0200103
Harald Welted879bd92018-03-12 15:01:23 +0100104 template LapdmCtrl tr_LapdmCtrlRR(template uint3_t nr, template boolean pf) modifies tr_LapdmCtrlS := {
Harald Welte1bbe0b72018-05-10 18:28:15 +0200105 s := { n_r := nr, p_f := pf, s := '00'B }
106 };
107 template (value) LapdmCtrl ts_LapdmCtrlRR(uint3_t nr, boolean pf) := {
Harald Welte2f2b2b72019-05-31 22:24:57 +0200108 s := { n_r := nr, p_f := pf, s := '00'B, spare := '01'B }
Harald Weltef6543322017-07-16 07:35:10 +0200109 };
110
Harald Welted879bd92018-03-12 15:01:23 +0100111 template LapdmCtrl tr_LapdmCtrlRNR(template uint3_t nr, template boolean pf) modifies tr_LapdmCtrlS := {
Harald Welte1bbe0b72018-05-10 18:28:15 +0200112 s := { n_r := nr, p_f := pf, s := '01'B }
Harald Weltef6543322017-07-16 07:35:10 +0200113 };
114
Harald Welted879bd92018-03-12 15:01:23 +0100115 template LapdmCtrl tr_LapdmCtrlREJ(template uint3_t nr, template boolean pf) modifies tr_LapdmCtrlS := {
Harald Welte1bbe0b72018-05-10 18:28:15 +0200116 s := { n_r := nr, p_f := pf, s := '10'B }
Harald Weltef6543322017-07-16 07:35:10 +0200117 };
Harald Welte2f2b2b72019-05-31 22:24:57 +0200118 template (value) LapdmCtrl ts_LapdmCtrlREJ(uint3_t nr, boolean pf) := {
119 s := { n_r := nr, p_f := pf, s := '10'B, spare := '01'B }
120 };
121
Harald Weltef6543322017-07-16 07:35:10 +0200122
Harald Welte1bbe0b72018-05-10 18:28:15 +0200123 template LapdmCtrl tr_LapdmCtrlSABM(template boolean p) := {
124 u := { u := '001'B, p_f := p, u2 := '11'B, spare := '11'B }
Harald Weltef6543322017-07-16 07:35:10 +0200125 };
Harald Welted879bd92018-03-12 15:01:23 +0100126 template (value) LapdmCtrl ts_LapdmCtrlSABM(boolean p) := {
Harald Welte1bbe0b72018-05-10 18:28:15 +0200127 u := { u := '001'B, p_f := p, u2 := '11'B, spare := '11'B }
Harald Welted879bd92018-03-12 15:01:23 +0100128 };
Harald Weltef6543322017-07-16 07:35:10 +0200129
Harald Welte1bbe0b72018-05-10 18:28:15 +0200130 template LapdmCtrl tr_LapdmCtrlDM(template boolean f) := {
131 u := { u := '000'B, p_f := f, u2 := '11'B, spare := '11'B }
Harald Weltef6543322017-07-16 07:35:10 +0200132 };
Harald Welte2f2b2b72019-05-31 22:24:57 +0200133 template (value) LapdmCtrl ts_LapdmCtrlDM(boolean f) := {
134 u := { u := '000'B, p_f := f, u2 := '11'B, spare := '11'B }
135 };
Harald Weltef6543322017-07-16 07:35:10 +0200136
Harald Welte1bbe0b72018-05-10 18:28:15 +0200137 template LapdmCtrl tr_LapdmCtrlUI(template boolean p := false) := {
138 u := { u := '000'B, p_f := p, u2 := '00'B, spare := '11'B }
Harald Weltef6543322017-07-16 07:35:10 +0200139 };
Harald Welte1bbe0b72018-05-10 18:28:15 +0200140 template (value) LapdmCtrl ts_LapdmCtrlUI(boolean p := false) := {
141 u := { u := '000'B, p_f := p, u2 := '00'B, spare := '11'B }
Harald Welted879bd92018-03-12 15:01:23 +0100142 };
Harald Weltef6543322017-07-16 07:35:10 +0200143
Harald Welte1bbe0b72018-05-10 18:28:15 +0200144 template LapdmCtrl tr_LapdmCtrlDISC(template boolean p) := {
145 u := { u := '010'B, p_f := p, u2 := '00'B, spare := '11'B }
Harald Weltef6543322017-07-16 07:35:10 +0200146 };
Harald Welted879bd92018-03-12 15:01:23 +0100147 template LapdmCtrl ts_LapdmCtrlDISC(boolean p) := {
Harald Welte1bbe0b72018-05-10 18:28:15 +0200148 u := { u := '010'B, p_f := p, u2 := '00'B, spare := '11'B }
Harald Welted879bd92018-03-12 15:01:23 +0100149 };
Harald Weltef6543322017-07-16 07:35:10 +0200150
Harald Welted879bd92018-03-12 15:01:23 +0100151 template LapdmCtrl tr_LapdmCtrlUA(template boolean f) modifies tr_LapdmCtrlU := {
Harald Welte1bbe0b72018-05-10 18:28:15 +0200152 u := { u := '011'B, p_f := f, u2 := '00'B, spare := '11'B }
Harald Weltef6543322017-07-16 07:35:10 +0200153 };
Harald Welted879bd92018-03-12 15:01:23 +0100154 template (value) LapdmCtrl ts_LapdmCtrlUA(boolean f) := {
Harald Welte1bbe0b72018-05-10 18:28:15 +0200155 u := { u := '011'B, p_f := f, u2 := '00'B, spare := '11'B }
Harald Welted879bd92018-03-12 15:01:23 +0100156 };
157
158
Harald Weltef6543322017-07-16 07:35:10 +0200159
Harald Welte1bd7c322017-07-17 20:59:46 +0200160 external function dec_LapdmAddressField(in octetstring stream) return LapdmAddressField
161 with { extension "prototype(convert) decode(RAW)" };
162
163 external function dec_LapdmCtrl(in octetstring stream) return LapdmCtrl
164 with { extension "prototype(convert) decode(RAW)" };
165
166 external function dec_LapdmCtrlU(in octetstring stream) return LapdmCtrlU
167 with { extension "prototype(convert) decode(RAW)" };
168
Harald Weltef6543322017-07-16 07:35:10 +0200169 /* Formats B, Bter and B4 are used on DCCHs for frames containing an information field:
170 /* - format Bter is used on request of higher layers if and only if short L2 header type 1 is
171 * supported and a UI command is to be transmitted on SAPI 0 */
172 /* - format B4 is used for UI frames transmitted by the network on SACCH; */
173 /* - format B is applied in all other cases. */
174 /* Format Bbis is used only on BCCH, PCH, NCH, and AGCH.
175
176 /* Format B */
Harald Welted879bd92018-03-12 15:01:23 +0100177 type record LapdmFrameAB {
Harald Weltef6543322017-07-16 07:35:10 +0200178 LapdmAddressField addr,
179 LapdmCtrl ctrl,
Harald Welted879bd92018-03-12 15:01:23 +0100180 uint6_t len,
Harald Welted4ba7ff2017-07-17 21:00:48 +0200181 boolean m,
Harald Welted879bd92018-03-12 15:01:23 +0100182 uint1_t el,
183 octetstring payload /* zero-length in Frame A */
Harald Welted4ba7ff2017-07-17 21:00:48 +0200184 } with { variant (len) "LENGTHTO(payload)"
185 variant "FIELDORDER(msb)" };
Harald Weltef6543322017-07-16 07:35:10 +0200186
Harald Welted879bd92018-03-12 15:01:23 +0100187 external function enc_LapdmFrameAB(in LapdmFrameAB si) return octetstring
Harald Weltef6543322017-07-16 07:35:10 +0200188 with { extension "prototype(convert) encode(RAW)" };
Harald Welted879bd92018-03-12 15:01:23 +0100189 external function dec_LapdmFrameAB(in octetstring stream) return LapdmFrameAB
Harald Weltef6543322017-07-16 07:35:10 +0200190 with { extension "prototype(convert) decode(RAW)" };
191
Harald Weltef6543322017-07-16 07:35:10 +0200192 /* Format B4 */
193 type record LapdmFrameB4 {
194 LapdmAddressField addr,
195 LapdmCtrl ctrl,
196 octetstring payload
197 } with { variant "" };
198
199 external function enc_LapdmFrameB4(in LapdmFrameB4 si) return octetstring
200 with { extension "prototype(convert) encode(RAW)" };
201 external function dec_LapdmFrameB4(in octetstring stream) return LapdmFrameB4
202 with { extension "prototype(convert) decode(RAW)" };
203
Harald Weltec2a5c072017-07-17 20:58:32 +0200204 type record LapdmFrameBbis {
205 octetstring payload
206 } with { variant "" };
207
208 external function enc_LapdmFrameBbis(in LapdmFrameBbis si) return octetstring
209 with { extension "prototype(convert) encode(RAW)" };
210 external function dec_LapdmFrameBbis(in octetstring stream) return LapdmFrameBbis
211 with { extension "prototype(convert) decode(RAW)" };
212
213 type union LapdmFrame {
Harald Welted879bd92018-03-12 15:01:23 +0100214 LapdmFrameAB ab,
Harald Weltec2a5c072017-07-17 20:58:32 +0200215 LapdmFrameBbis bbis,
216 LapdmFrameB4 b4
217 } with { variant "" };
218
219 external function enc_LapdmFrame(in LapdmFrame si) return octetstring
220 with { extension "prototype(convert) encode(RAW)" };
221 /* automatic decoding to the generic LapdmFrame will not work, you have to call one of the
222 * type-specific decoder routines above */
223
Harald Welte0472ab42018-03-12 15:02:26 +0100224 /* SABM frame with L3 payload */
225 template (value) LapdmFrame ts_LAPDm_SABM(LapdmSapi sapi, boolean c_r, boolean p,
226 octetstring l3) := {
227 ab := {
228 addr := ts_LapdmAddr(sapi, c_r),
229 ctrl := ts_LapdmCtrlSABM(p),
230 len := 0, /* overwritten in encoder */
231 m := false,
232 el := 1,
233 payload := l3
234 }
235 }
236 template LapdmFrame tr_LAPDm_SABM(template LapdmSapi sapi, template boolean c_r,
237 template boolean p, template octetstring l3) := {
238 ab := {
239 addr := tr_LapdmAddr(sapi, c_r),
240 ctrl := tr_LapdmCtrlSABM(p),
241 len := ?,
242 m := false,
243 el := 1,
244 payload := l3
245 }
246 }
247
248 template (value) LapdmFrame ts_LAPDm_UA(LapdmSapi sapi, boolean c_r, boolean f,
249 octetstring l3) := {
250 ab := {
251 addr := ts_LapdmAddr(sapi, c_r),
252 ctrl := ts_LapdmCtrlUA(f),
253 len := 0, /* overwritten in encoder */
254 m := false,
255 el := 1,
256 payload := l3
257 }
258 }
259 template LapdmFrame tr_LAPDm_UA(template LapdmSapi sapi, template boolean c_r,
260 template boolean f, template octetstring l3) := {
261 ab := {
262 addr := tr_LapdmAddr(sapi, c_r),
263 ctrl := tr_LapdmCtrlUA(f),
264 len := ?,
265 m := false,
266 el := 1,
267 payload := l3
268 }
269 }
270
Harald Welte2f2b2b72019-05-31 22:24:57 +0200271 template (value) LapdmFrame ts_LAPDm_DM(LapdmSapi sapi, boolean c_r, boolean f) := {
272 ab := {
273 addr := ts_LapdmAddr(sapi, c_r),
274 ctrl := ts_LapdmCtrlDM(f),
275 len := 0, /* overwritten in encoder */
276 m := false,
277 el := 1,
278 payload := ''O
279 }
280 }
281 template LapdmFrame tr_LAPDm_DM(template LapdmSapi sapi, template boolean c_r,
282 template boolean f) := {
283 ab := {
284 addr := tr_LapdmAddr(sapi, c_r),
285 ctrl := tr_LapdmCtrlDM(f),
286 len := ?,
287 m := false,
288 el := 1,
289 payload := ''O
290 }
291 }
292
Harald Welte0472ab42018-03-12 15:02:26 +0100293 template LapdmFrame ts_LAPDm_DISC(LapdmSapi sapi, boolean c_r, boolean p) := {
294 ab := {
295 addr := ts_LapdmAddr(sapi, c_r),
296 ctrl := ts_LapdmCtrlDISC(p),
297 len := 0,
298 m := false,
299 el := 1,
300 payload := ''O
301 }
302 }
303 template LapdmFrame tr_LAPDm_DISC(template LapdmSapi sapi, template boolean c_r,
304 template boolean p) := {
305 ab := {
306 addr := tr_LapdmAddr(sapi, c_r),
307 ctrl := tr_LapdmCtrlDISC(p),
308 len := ?,
309 m := false,
310 el := 1,
311 payload := ''O
312 }
313 }
314
Harald Weltee613f962018-04-18 22:38:16 +0200315 template LapdmFrame ts_LAPDm_UI(LapdmSapi sapi, boolean c_r, octetstring l3) := {
Harald Welte0472ab42018-03-12 15:02:26 +0100316 ab := {
317 addr := ts_LapdmAddr(sapi, c_r),
Harald Weltee613f962018-04-18 22:38:16 +0200318 ctrl := ts_LapdmCtrlUI,
Harald Welte0472ab42018-03-12 15:02:26 +0100319 len := 0,
320 m := false,
321 el := 1,
322 payload := l3
323 }
324 }
325 template LapdmFrame tr_LAPDm_UI(template LapdmSapi sapi, template boolean c_r,
Harald Weltee613f962018-04-18 22:38:16 +0200326 template octetstring l3) := {
Harald Welte0472ab42018-03-12 15:02:26 +0100327 ab := {
328 addr := tr_LapdmAddr(sapi, c_r),
Harald Weltee613f962018-04-18 22:38:16 +0200329 ctrl := tr_LapdmCtrlUI,
Harald Welte0472ab42018-03-12 15:02:26 +0100330 len := ?,
331 m := false,
332 el := 1,
333 payload := l3
334 }
335 }
336
Harald Weltee613f962018-04-18 22:38:16 +0200337 template LapdmFrame ts_LAPDm_B4_UI(LapdmSapi sapi, boolean c_r, octetstring l3) := {
Harald Welte0472ab42018-03-12 15:02:26 +0100338 b4 := {
339 addr := ts_LapdmAddr(sapi, c_r),
Harald Weltee613f962018-04-18 22:38:16 +0200340 ctrl := ts_LapdmCtrlUI,
Harald Welte0472ab42018-03-12 15:02:26 +0100341 payload := l3
342 }
343 }
344 template LapdmFrame tr_LAPDm_B4_UI(template LapdmSapi sapi, template boolean c_r,
Harald Weltee613f962018-04-18 22:38:16 +0200345 template octetstring l3) := {
Harald Welte0472ab42018-03-12 15:02:26 +0100346 b4 := {
347 addr := tr_LapdmAddr(sapi, c_r),
Harald Weltee613f962018-04-18 22:38:16 +0200348 ctrl := tr_LapdmCtrlUI,
Harald Welte0472ab42018-03-12 15:02:26 +0100349 payload := l3
350 }
351 }
352
Harald Weltee613f962018-04-18 22:38:16 +0200353 template LapdmFrame tr_LAPDm_I(template LapdmSapi sapi, template boolean c_r,
354 template boolean p, template uint3_t nr,
355 template uint3_t ns, template octetstring l3) := {
356 ab := {
357 addr := tr_LapdmAddr(sapi, c_r),
358 ctrl := tr_LapdmCtrlI(nr, ns, p),
359 len := ?,
360 m := false,
361 el := 1,
362 payload := l3
363 }
364 }
365 template (value) LapdmFrame ts_LAPDm_I(LapdmSapi sapi, boolean c_r, boolean p, uint3_t nr,
366 uint3_t ns, octetstring l3) := {
367 ab := {
368 addr := ts_LapdmAddr(sapi, c_r),
369 ctrl := ts_LapdmCtrlI(nr, ns, p),
370 len := 0,
371 m := false,
372 el := 1,
373 payload := l3
374 }
375 }
376
377 template LapdmFrame tr_LAPDm_RR(template LapdmSapi sapi, template boolean c_r,
378 template boolean p, template uint3_t nr) := {
379 ab := {
380 addr := tr_LapdmAddr(sapi, c_r),
381 ctrl := tr_LapdmCtrlRR(nr, p),
382 len := 0,
383 m := false,
384 el := 1,
385 payload := ''O
386 }
387 }
388 template (value) LapdmFrame ts_LAPDm_RR(LapdmSapi sapi, boolean c_r,
389 boolean p, uint3_t nr) := {
390 ab := {
391 addr := ts_LapdmAddr(sapi, c_r),
392 ctrl := ts_LapdmCtrlRR(nr, p),
393 len := 0,
394 m := false,
395 el := 1,
396 payload := ''O
397 }
398 }
399
Harald Welte2f2b2b72019-05-31 22:24:57 +0200400 template LapdmFrame tr_LAPDm_REJ(template LapdmSapi sapi, template boolean c_r,
401 template boolean p, template uint3_t nr) := {
402 ab := {
403 addr := tr_LapdmAddr(sapi, c_r),
404 ctrl := tr_LapdmCtrlREJ(nr, p),
405 len := 0,
406 m := false,
407 el := 1,
408 payload := ''O
409 }
410 }
411 template (value) LapdmFrame ts_LAPDm_REJ(LapdmSapi sapi, boolean c_r,
412 boolean p, uint3_t nr) := {
413 ab := {
414 addr := ts_LapdmAddr(sapi, c_r),
415 ctrl := ts_LapdmCtrlREJ(nr, p),
416 len := 0,
417 m := false,
418 el := 1,
419 payload := ''O
420 }
421 }
422
Harald Welte0472ab42018-03-12 15:02:26 +0100423
424
Harald Weltec2a5c072017-07-17 20:58:32 +0200425} with { encode "RAW"; /*variant "FIELDORDER(msb)" */}