blob: 9fdf3794dcff74a16247cc54a65bcb74ca1d4e50 [file] [log] [blame]
Neels Hofmeyr17518fe2017-06-20 04:35:06 +02001/*
2 * (C) 2009,2010 by Holger Hans Peter Freyther <zecke@selfish.org>
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +02003 * (C) 2009,2010 by On-Waves
4 * All Rights Reserved
5 *
Harald Weltee08da972017-11-13 01:00:26 +09006 * SPDX-License-Identifier: GPL-2.0+
7 *
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +02008 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License along
19 * with this program; if not, write to the Free Software Foundation, Inc.,
20 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
21 *
22 */
23
Alexander Chemeris22630e62020-05-13 00:44:04 +030024#include <string.h>
25
Harald Welte95871da2017-05-15 12:11:36 +020026#include <osmocom/core/byteswap.h>
Philipp Maier2908dbf2020-06-05 14:16:11 +020027#include <osmocom/core/endian.h>
Pablo Neira Ayuso83419342011-03-22 16:36:13 +010028#include <osmocom/gsm/gsm0808.h>
Philipp Maierfa896ab2017-03-27 16:55:32 +020029#include <osmocom/gsm/gsm0808_utils.h>
Pablo Neira Ayuso83419342011-03-22 16:36:13 +010030#include <osmocom/gsm/protocol/gsm_08_08.h>
31#include <osmocom/gsm/gsm48.h>
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020032
Harald Welte96e2a002017-06-12 21:44:18 +020033/*! \addtogroup gsm0808
34 * @{
Harald Welte37b61652017-10-16 18:46:03 +020035 * \file gsm0808.c
36 * Helper functions regarding the TS 08.08 / 48.008 A interface, primarily
37 * message generation/encoding.
Harald Welte96e2a002017-06-12 21:44:18 +020038 */
39
Alexander Chemeris22630e62020-05-13 00:44:04 +030040/*! Char buffer to return strings from functions */
41static __thread char str_buff[512];
42
Neels Hofmeyrc4fce142018-02-20 13:47:08 +010043/*! Create "Complete L3 Info" for AoIP, legacy implementation.
44 * Instead use gsm0808_create_layer3_aoip2(), which is capable of three-digit MNC with leading zeros.
Harald Welte96e2a002017-06-12 21:44:18 +020045 * \param[in] msg_l3 msgb containing Layer 3 Message
46 * \param[in] nc Mobile Network Code
47 * \param[in] cc Mobile Country Code
48 * \param[in] lac Location Area Code
49 * \param[in] _ci Cell Identity
50 * \param[in] scl Speech Codec List
51 * \returns callee-allocated msgb with Complete L3 Info message */
Philipp Maierfa896ab2017-03-27 16:55:32 +020052struct msgb *gsm0808_create_layer3_aoip(const struct msgb *msg_l3, uint16_t nc,
53 uint16_t cc, int lac, uint16_t _ci,
54 const struct gsm0808_speech_codec_list
55 *scl)
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020056{
Neels Hofmeyrc4fce142018-02-20 13:47:08 +010057 struct osmo_cell_global_id cgi = {
58 .lai = {
59 .plmn = {
60 .mcc = cc,
61 .mnc = nc,
62 },
63 .lac = lac,
64 },
65 .cell_identity = _ci,
66 };
67 return gsm0808_create_layer3_2(msg_l3, &cgi, scl);
68}
69
70/*! Create "Complete L3 Info" for AoIP.
71 * \param[in] msg_l3 msgb containing Layer 3 Message -- not modified by this call.
72 * \param[in] cell MCC, MNC, LAC, CI to identify the cell.
73 * \param[in] scl Speech Codec List, optional.
74 * \returns newly allocated msgb with Complete L3 Info message */
75struct msgb *gsm0808_create_layer3_2(const struct msgb *msg_l3, const struct osmo_cell_global_id *cell,
76 const struct gsm0808_speech_codec_list *scl)
77{
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020078 struct msgb* msg;
Harald Welte65c2d362012-01-21 14:26:01 +010079 struct {
80 uint8_t ident;
81 struct gsm48_loc_area_id lai;
82 uint16_t ci;
83 } __attribute__ ((packed)) lai_ci;
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020084
85 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
86 "bssmap cmpl l3");
87 if (!msg)
88 return NULL;
89
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020090 /* create layer 3 header */
Harald Welte65c2d362012-01-21 14:26:01 +010091 msgb_v_put(msg, BSS_MAP_MSG_COMPLETE_LAYER_3);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020092
93 /* create the cell header */
Harald Welte65c2d362012-01-21 14:26:01 +010094 lai_ci.ident = CELL_IDENT_WHOLE_GLOBAL;
Neels Hofmeyrc4fce142018-02-20 13:47:08 +010095 gsm48_generate_lai2(&lai_ci.lai, &cell->lai);
96 lai_ci.ci = osmo_htons(cell->cell_identity);
Harald Welte65c2d362012-01-21 14:26:01 +010097 msgb_tlv_put(msg, GSM0808_IE_CELL_IDENTIFIER, sizeof(lai_ci),
98 (uint8_t *) &lai_ci);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020099
100 /* copy the layer3 data */
Harald Welte65c2d362012-01-21 14:26:01 +0100101 msgb_tlv_put(msg, GSM0808_IE_LAYER_3_INFORMATION,
102 msgb_l3len(msg_l3), msg_l3->l3h);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200103
Philipp Maierfa896ab2017-03-27 16:55:32 +0200104 /* AoIP: add Codec List (BSS Supported) 3.2.2.103 */
105 if (scl)
106 gsm0808_enc_speech_codec_list(msg, scl);
107
Harald Welte65c2d362012-01-21 14:26:01 +0100108 /* push the bssmap header */
109 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200110
111 return msg;
112}
113
Neels Hofmeyrc4fce142018-02-20 13:47:08 +0100114/*! Create "Complete L3 Info" for A, legacy implementation.
115 * Instead use gsm0808_create_layer3_2() with the scl parameter passed as NULL,
116 * which is capable of three-digit MNC with leading zeros.
Harald Welte96e2a002017-06-12 21:44:18 +0200117 * \param[in] msg_l3 msgb containing Layer 3 Message
118 * \param[in] nc Mobile Network Code
119 * \param[in] cc Mobile Country Code
120 * \param[in] lac Location Area Code
121 * \param[in] _ci Cell Identity
122 * \returns callee-allocated msgb with Complete L3 Info message */
Philipp Maierfa896ab2017-03-27 16:55:32 +0200123struct msgb *gsm0808_create_layer3(struct msgb *msg_l3, uint16_t nc,
124 uint16_t cc, int lac, uint16_t _ci)
125{
Neels Hofmeyr4eeb8082018-03-23 01:47:14 +0100126 struct osmo_cell_global_id cgi = {
127 .lai = {
128 .plmn = {
129 .mcc = cc,
130 .mnc = nc,
131 },
132 .lac = lac,
133 },
134 .cell_identity = _ci,
135 };
136 return gsm0808_create_layer3_2(msg_l3, &cgi, NULL);
Philipp Maierfa896ab2017-03-27 16:55:32 +0200137}
138
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200139/*! Create BSSMAP RESET message
Harald Welte96e2a002017-06-12 21:44:18 +0200140 * \returns callee-allocated msgb with BSSMAP Reset message */
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200141struct msgb *gsm0808_create_reset(void)
142{
Harald Welte65c2d362012-01-21 14:26:01 +0100143 uint8_t cause = GSM0808_CAUSE_EQUIPMENT_FAILURE;
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200144 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
145 "bssmap: reset");
146 if (!msg)
147 return NULL;
148
Harald Welte65c2d362012-01-21 14:26:01 +0100149 msgb_v_put(msg, BSS_MAP_MSG_RESET);
Philipp Maier4f4905f2018-11-30 13:36:12 +0100150 gsm0808_enc_cause(msg, cause);
Harald Welte65c2d362012-01-21 14:26:01 +0100151 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
152
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200153 return msg;
154}
155
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200156/*! Create BSSMAP RESET ACK message
Harald Welte96e2a002017-06-12 21:44:18 +0200157 * \returns callee-allocated msgb with BSSMAP Reset ACK message */
Harald Weltea62fe312013-06-19 15:14:37 +0200158struct msgb *gsm0808_create_reset_ack(void)
159{
160 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
161 "bssmap: reset ack");
162 if (!msg)
163 return NULL;
164
165 msgb_v_put(msg, BSS_MAP_MSG_RESET_ACKNOWLEDGE);
166 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
167
168 return msg;
169}
170
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200171/*! Create BSSMAP Clear Complete message
Harald Welte96e2a002017-06-12 21:44:18 +0200172 * \returns callee-allocated msgb with BSSMAP Clear Complete message */
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200173struct msgb *gsm0808_create_clear_complete(void)
174{
175 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
176 "bssmap: clear complete");
Harald Welte65c2d362012-01-21 14:26:01 +0100177 uint8_t val = BSS_MAP_MSG_CLEAR_COMPLETE;
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200178 if (!msg)
179 return NULL;
180
Harald Welte65c2d362012-01-21 14:26:01 +0100181 msg->l3h = msg->data;
182 msgb_tlv_put(msg, BSSAP_MSG_BSS_MANAGEMENT, 1, &val);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200183
184 return msg;
185}
186
Harald Welted1365e12019-02-18 13:44:30 +0100187/*! Create BSSMAP Clear Command message with BSSAP header *before* l3h and BSSMAP in l3h.
188 * This is quite different from most (all?) other gsm0808_create_* which have l3h
189 * point to the BSSAP header. However, we have to keep this for backwards compatibility.
190 * Use gsm0808_create_clear_command2() for a 'modern' implementation.
Philipp Maier1a146c82018-10-30 09:36:49 +0100191 * \param[in] cause TS 08.08 cause value
Harald Welte96e2a002017-06-12 21:44:18 +0200192 * \returns callee-allocated msgb with BSSMAP Clear Command message */
Philipp Maier1a146c82018-10-30 09:36:49 +0100193struct msgb *gsm0808_create_clear_command(uint8_t cause)
Holger Hans Peter Freythera3f05d82010-10-27 11:49:24 +0200194{
195 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
196 "bssmap: clear command");
197 if (!msg)
198 return NULL;
199
Holger Hans Peter Freytheraeebe392010-10-27 12:36:05 +0200200 msg->l3h = msgb_tv_put(msg, BSSAP_MSG_BSS_MANAGEMENT, 4);
201 msgb_v_put(msg, BSS_MAP_MSG_CLEAR_CMD);
Philipp Maier4f4905f2018-11-30 13:36:12 +0100202 gsm0808_enc_cause(msg, cause);
Harald Welte65c2d362012-01-21 14:26:01 +0100203
Holger Hans Peter Freythera3f05d82010-10-27 11:49:24 +0200204 return msg;
205}
206
Philipp Maier74c4c4e2019-02-04 16:42:28 +0100207/*! Create BSSMAP Clear Command message.
208 * \param[in] cause TS 08.08 cause value.
209 * \param[in] csfb_ind indicate that the call was established in an CSFB context.
210 * \returns callee-allocated msgb with BSSMAP Clear Command message. */
211struct msgb *gsm0808_create_clear_command2(uint8_t cause, bool csfb_ind)
212{
Harald Welte10ba47d2019-02-18 12:36:54 +0100213 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
214 "bssmap: clear command");
Philipp Maier74c4c4e2019-02-04 16:42:28 +0100215 if (!msg)
216 return NULL;
217
Harald Welte10ba47d2019-02-18 12:36:54 +0100218 msgb_v_put(msg, BSS_MAP_MSG_CLEAR_CMD);
219 gsm0808_enc_cause(msg, cause);
220
Philipp Maier74c4c4e2019-02-04 16:42:28 +0100221 if (csfb_ind)
222 msgb_v_put(msg, GSM0808_IE_CSFB_INDICATION);
223
Harald Welte10ba47d2019-02-18 12:36:54 +0100224 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
225
Philipp Maier74c4c4e2019-02-04 16:42:28 +0100226 return msg;
227}
228
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200229/*! Create BSSMAP Cipher Mode Command message
Harald Welte96e2a002017-06-12 21:44:18 +0200230 * \param[in] ei Mandatory Encryption Information
231 * \param[in] cipher_response_mode optional 1-byte Cipher Response Mode
232 * \returns callee-allocated msgb with BSSMAP Cipher Mode Command message */
Philipp Maierb478dd32017-03-29 15:50:05 +0200233struct msgb *gsm0808_create_cipher(const struct gsm0808_encrypt_info *ei,
234 const uint8_t *cipher_response_mode)
235{
236 /* See also: 3GPP TS 48.008 3.2.1.30 CIPHER MODE COMMAND */
237 struct msgb *msg;
238
239 /* Mandatory emelent! */
240 OSMO_ASSERT(ei);
241
242 msg =
243 msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
244 "cipher-mode-command");
245 if (!msg)
246 return NULL;
247
248 /* Message Type 3.2.2.1 */
249 msgb_v_put(msg, BSS_MAP_MSG_CIPHER_MODE_CMD);
250
251 /* Encryption Information 3.2.2.10 */
252 gsm0808_enc_encrypt_info(msg, ei);
253
254 /* Cipher Response Mode 3.2.2.34 */
255 if (cipher_response_mode)
256 msgb_tv_put(msg, GSM0808_IE_CIPHER_RESPONSE_MODE,
257 *cipher_response_mode);
258
259 /* pre-pend the header */
260 msg->l3h =
261 msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
262
263 return msg;
264}
265
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200266/*! Create BSSMAP Cipher Mode Complete message
Harald Welte96e2a002017-06-12 21:44:18 +0200267 * \param[in] layer3 L3 Message to be included
268 * \param[in] alg_id Chosen Encrpytion Algorithm
269 * \returns callee-allocated msgb with BSSMAP Cipher Mode Complete message */
Holger Hans Peter Freyther81716d52010-04-17 06:16:35 +0200270struct msgb *gsm0808_create_cipher_complete(struct msgb *layer3, uint8_t alg_id)
271{
272 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
273 "cipher-complete");
274 if (!msg)
275 return NULL;
276
277 /* send response with BSS override for A5/1... cheating */
Harald Welte65c2d362012-01-21 14:26:01 +0100278 msgb_v_put(msg, BSS_MAP_MSG_CIPHER_MODE_COMPLETE);
Holger Hans Peter Freyther81716d52010-04-17 06:16:35 +0200279
280 /* include layer3 in case we have at least two octets */
281 if (layer3 && msgb_l3len(layer3) > 2) {
Harald Welte65c2d362012-01-21 14:26:01 +0100282 msg->l4h = msgb_tlv_put(msg, GSM0808_IE_LAYER_3_MESSAGE_CONTENTS,
283 msgb_l3len(layer3), layer3->l3h);
Holger Hans Peter Freyther81716d52010-04-17 06:16:35 +0200284 }
285
286 /* and the optional BSS message */
Harald Welte65c2d362012-01-21 14:26:01 +0100287 msgb_tv_put(msg, GSM0808_IE_CHOSEN_ENCR_ALG, alg_id);
Holger Hans Peter Freyther81716d52010-04-17 06:16:35 +0200288
Harald Welte65c2d362012-01-21 14:26:01 +0100289 /* pre-pend the header */
290 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
291
Holger Hans Peter Freyther81716d52010-04-17 06:16:35 +0200292 return msg;
293}
294
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200295/*! Create BSSMAP Cipher Mode Reject message
Maxaa934632018-11-07 13:16:54 +0100296 * \param[in] cause 3GPP TS 08.08 §3.2.2.5 cause value
Harald Welte96e2a002017-06-12 21:44:18 +0200297 * \returns callee-allocated msgb with BSSMAP Cipher Mode Reject message */
Maxaa934632018-11-07 13:16:54 +0100298struct msgb *gsm0808_create_cipher_reject(enum gsm0808_cause cause)
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200299{
300 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
Maxaa934632018-11-07 13:16:54 +0100301 "bssmap: cipher mode reject");
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200302 if (!msg)
303 return NULL;
304
Harald Welte62e40852017-12-17 20:50:34 +0100305 msgb_v_put(msg, BSS_MAP_MSG_CIPHER_MODE_REJECT);
Maxaa934632018-11-07 13:16:54 +0100306
Philipp Maier4f4905f2018-11-30 13:36:12 +0100307 gsm0808_enc_cause(msg, cause);
Harald Welte65c2d362012-01-21 14:26:01 +0100308
309 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200310
311 return msg;
312}
313
Maxed651d22018-11-07 15:25:05 +0100314/*! Create BSSMAP Cipher Mode Reject message
315 * \param[in] class 3GPP TS 08.08 §3.2.2.5 cause's class
316 * \param[in] ext 3GPP TS 08.08 §3.2.2.5 cause value (national application extension)
317 * \returns callee-allocated msgb with BSSMAP Cipher Mode Reject message */
318struct msgb *gsm0808_create_cipher_reject_ext(enum gsm0808_cause_class class, uint8_t ext)
319{
Philipp Maier4f4905f2018-11-30 13:36:12 +0100320 uint16_t cause;
Maxed651d22018-11-07 15:25:05 +0100321 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
322 "bssmap: cipher mode reject");
323 if (!msg)
324 return NULL;
325
Philipp Maier4f4905f2018-11-30 13:36:12 +0100326 /* Set cause code class in the upper byte */
327 cause = 0x80 | (class << 4);
328 cause = cause << 8;
329
330 /* Set cause code extension in the lower byte */
331 cause |= ext;
Maxed651d22018-11-07 15:25:05 +0100332
333 msgb_v_put(msg, BSS_MAP_MSG_CIPHER_MODE_REJECT);
334
Philipp Maier4f4905f2018-11-30 13:36:12 +0100335 gsm0808_enc_cause(msg, cause);
Maxed651d22018-11-07 15:25:05 +0100336
337 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
338
339 return msg;
340}
341
Harald Welte64e807c2018-05-29 21:00:56 +0200342/*! Create BSSMAP LCLS CONNECT CONTROL message (TS 48.008 3.2.1.91).
343 * \param[in] config LCLS Configuration
344 * \param[in] control LCLS Connection Status Control
345 * \returns callee-allocated msgb with BSSMAP LCLS NOTIFICATION */
Max45f89c92018-12-19 19:35:26 +0100346struct msgb *gsm0808_create_lcls_conn_ctrl(enum gsm0808_lcls_config config,
347 enum gsm0808_lcls_control control)
Harald Welte64e807c2018-05-29 21:00:56 +0200348{
Max45f89c92018-12-19 19:35:26 +0100349 struct msgb *msg;
350
351 /* According to NOTE 1 in §3.2.1.91 at least one of the parameters is required */
352 if (config == GSM0808_LCLS_CFG_NA && control == GSM0808_LCLS_CSC_NA)
353 return NULL;
354
355 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "bssmap: LCLS CONN CTRL");
Harald Welte64e807c2018-05-29 21:00:56 +0200356 if (!msg)
357 return NULL;
358
359 msgb_v_put(msg, BSS_MAP_MSG_LCLS_CONNECT_CTRL);
Max45f89c92018-12-19 19:35:26 +0100360 if (config != GSM0808_LCLS_CFG_NA)
361 msgb_tv_put(msg, GSM0808_IE_LCLS_CONFIG, config);
362 if (control != GSM0808_LCLS_CSC_NA)
363 msgb_tv_put(msg, GSM0808_IE_LCLS_CONFIG, control);
Harald Welte64e807c2018-05-29 21:00:56 +0200364 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
365
366 return msg;
367}
368
369/*! Create BSSMAP LCLS CONNECT CONTROL ACK message (TS 48.008 3.2.1.92).
370 * \param[in] status LCLS BSS Status
371 * \returns callee-allocated msgb with BSSMAP LCLS NOTIFICATION */
372struct msgb *gsm0808_create_lcls_conn_ctrl_ack(enum gsm0808_lcls_status status)
373{
374 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
375 "bssmap: LCLS CONN CTRL ACK");
376 if (!msg)
377 return NULL;
378
379 msgb_v_put(msg, BSS_MAP_MSG_LCLS_CONNECT_CTRL_ACK);
380 msgb_tv_put(msg, GSM0808_IE_LCLS_BSS_STATUS, status);
381 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
382
383 return msg;
384}
385
386/*! Create BSSMAP LCLS NOTIFICATION message (TS 48.008 3.2.1.93).
387 * \param[in] status LCLS BSS Status
388 * \param[in] break_req Include the LCLS BREAK REQ IE (true) or not (false)
389 * \returns callee-allocated msgb with BSSMAP LCLS NOTIFICATION */
390struct msgb *gsm0808_create_lcls_notification(enum gsm0808_lcls_status status, bool break_req)
391{
392 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
393 "bssmap: LCLS NOTIFICATION");
394 if (!msg)
395 return NULL;
396
397 msgb_v_put(msg, BSS_MAP_MSG_LCLS_NOTIFICATION);
398 msgb_tv_put(msg, GSM0808_IE_LCLS_BSS_STATUS, status);
399 if (break_req)
400 msgb_v_put(msg, GSM0808_IE_LCLS_BREAK_REQ);
401 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
402
403 return msg;
404}
405
Neels Hofmeyr2c79d552018-09-13 05:36:32 +0200406/*! Create BSSMAP Classmark Request message
407 * \returns callee-allocated msgb with BSSMAP Classmark Request message */
408struct msgb *gsm0808_create_classmark_request()
409{
410 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
411 "classmark-request");
412 if (!msg)
413 return NULL;
414
415 msgb_v_put(msg, BSS_MAP_MSG_CLASSMARK_RQST);
416 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
417 return msg;
418}
Harald Welte64e807c2018-05-29 21:00:56 +0200419
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200420/*! Create BSSMAP Classmark Update message
Harald Welte96e2a002017-06-12 21:44:18 +0200421 * \param[in] cm2 Classmark 2
422 * \param[in] cm2_len length (in octets) of \a cm2
423 * \param[in] cm3 Classmark 3
424 * \param[in] cm3_len length (in octets) of \a cm3
425 * \returns callee-allocated msgb with BSSMAP Classmark Update message */
Harald Welte07b625d2012-01-23 10:02:58 +0100426struct msgb *gsm0808_create_classmark_update(const uint8_t *cm2, uint8_t cm2_len,
427 const uint8_t *cm3, uint8_t cm3_len)
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200428{
429 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
430 "classmark-update");
431 if (!msg)
432 return NULL;
433
Harald Welte65c2d362012-01-21 14:26:01 +0100434 msgb_v_put(msg, BSS_MAP_MSG_CLASSMARK_UPDATE);
Harald Welte07b625d2012-01-23 10:02:58 +0100435 msgb_tlv_put(msg, GSM0808_IE_CLASSMARK_INFORMATION_T2, cm2_len, cm2);
436 if (cm3)
437 msgb_tlv_put(msg, GSM0808_IE_CLASSMARK_INFORMATION_T3,
438 cm3_len, cm3);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200439
Harald Welte65c2d362012-01-21 14:26:01 +0100440 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
441
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200442 return msg;
443}
444
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200445/*! Create BSSMAP SAPI N Reject message
Harald Welte96e2a002017-06-12 21:44:18 +0200446 * \param[in] link_id Link Identifier
447 * \returns callee-allocated msgb with BSSMAP SAPI N Reject message */
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200448struct msgb *gsm0808_create_sapi_reject(uint8_t link_id)
449{
450 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
451 "bssmap: sapi 'n' reject");
452 if (!msg)
453 return NULL;
454
Harald Welte65c2d362012-01-21 14:26:01 +0100455 msgb_v_put(msg, BSS_MAP_MSG_SAPI_N_REJECT);
Alexander Chemerisa5b1b862020-05-12 01:03:08 +0300456 msgb_tv_put(msg, GSM0808_IE_DLCI, link_id);
457 gsm0808_enc_cause(msg, GSM0808_CAUSE_BSS_NOT_EQUIPPED);
Harald Welte65c2d362012-01-21 14:26:01 +0100458
459 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200460
461 return msg;
462}
463
Max52074322018-11-30 10:44:07 +0100464/*! Create BSSMAP Assignment Request message, 3GPP TS 48.008 §3.2.1.1.
465 * This is identical to gsm0808_create_ass(), but adds KC and LCLS IEs.
Harald Welte96e2a002017-06-12 21:44:18 +0200466 * \param[in] ct Channel Type
467 * \param[in] cic Circuit Identity Code (Classic A only)
468 * \param[in] ss Socket Address of MSC-side RTP socket (AoIP only)
469 * \param[in] scl Speech Codec List (AoIP only)
Max49c06682018-11-21 22:10:26 +0100470 * \param[in] ci Call Identifier (Optional), §3.2.2.105
Max52074322018-11-30 10:44:07 +0100471 * \param[in] kc Kc128 ciphering key (Optional, A5/4), §3.2.2.109
472 * \param[in] lcls Optional LCLS parameters
Harald Welte96e2a002017-06-12 21:44:18 +0200473 * \returns callee-allocated msgb with BSSMAP Assignment Request message */
Max52074322018-11-30 10:44:07 +0100474struct msgb *gsm0808_create_ass2(const struct gsm0808_channel_type *ct,
475 const uint16_t *cic,
476 const struct sockaddr_storage *ss,
477 const struct gsm0808_speech_codec_list *scl,
478 const uint32_t *ci,
479 const uint8_t *kc, const struct osmo_lcls *lcls)
Philipp Maierc6144a22017-03-29 17:53:43 +0200480{
481 /* See also: 3GPP TS 48.008 3.2.1.1 ASSIGNMENT REQUEST */
482 struct msgb *msg;
483 uint16_t cic_sw;
484 uint32_t ci_sw;
485
486 /* Mandatory emelent! */
487 OSMO_ASSERT(ct);
488
489 msg =
490 msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
491 "bssmap: ass req");
492 if (!msg)
493 return NULL;
494
495 /* Message Type 3.2.2.1 */
496 msgb_v_put(msg, BSS_MAP_MSG_ASSIGMENT_RQST);
497
498 /* Channel Type 3.2.2.11 */
499 gsm0808_enc_channel_type(msg, ct);
500
501 /* Circuit Identity Code 3.2.2.2 */
502 if (cic) {
Harald Welte95871da2017-05-15 12:11:36 +0200503 cic_sw = osmo_htons(*cic);
Philipp Maierc6144a22017-03-29 17:53:43 +0200504 msgb_tv_fixed_put(msg, GSM0808_IE_CIRCUIT_IDENTITY_CODE,
505 sizeof(cic_sw), (uint8_t *) & cic_sw);
506 }
507
508 /* AoIP: AoIP Transport Layer Address (MGW) 3.2.2.102 */
509 if (ss) {
510 gsm0808_enc_aoip_trasp_addr(msg, ss);
511 }
512
513 /* AoIP: Codec List (MSC Preferred) 3.2.2.103 */
514 if (scl)
515 gsm0808_enc_speech_codec_list(msg, scl);
516
517 /* AoIP: Call Identifier 3.2.2.105 */
518 if (ci) {
Philipp Maier2908dbf2020-06-05 14:16:11 +0200519 /* NOTE: 3GPP TS 48.008, section 3.2.2.105 specifies that
520 the least significant byte should be transmitted first.
521 On x86, this would mean that the endieness is already
522 correct, however a platform independed implementation
523 is required: */
524#ifndef OSMO_IS_LITTLE_ENDIAN
525 ci_sw = osmo_swab32(*ci);
526#else
527 ci_sw = *ci;
528#endif
Philipp Maierc6144a22017-03-29 17:53:43 +0200529 msgb_tv_fixed_put(msg, GSM0808_IE_CALL_ID, sizeof(ci_sw),
530 (uint8_t *) & ci_sw);
531 }
532
Max52074322018-11-30 10:44:07 +0100533 if (kc)
534 msgb_tv_fixed_put(msg, GSM0808_IE_KC_128, 16, kc);
535
Max47022152018-12-19 18:51:00 +0100536 if (lcls)
537 gsm0808_enc_lcls(msg, lcls);
Max52074322018-11-30 10:44:07 +0100538
Philipp Maierc6144a22017-03-29 17:53:43 +0200539 /* push the bssmap header */
540 msg->l3h =
541 msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
542
543 return msg;
544}
545
Max52074322018-11-30 10:44:07 +0100546/*! Create BSSMAP Assignment Request message, 3GPP TS 48.008 §3.2.1.1.
547 * \param[in] ct Channel Type
548 * \param[in] cic Circuit Identity Code (Classic A only)
549 * \param[in] ss Socket Address of MSC-side RTP socket (AoIP only)
550 * \param[in] scl Speech Codec List (AoIP only)
551 * \param[in] ci Call Identifier (Optional), §3.2.2.105
552 * \returns callee-allocated msgb with BSSMAP Assignment Request message */
553struct msgb *gsm0808_create_ass(const struct gsm0808_channel_type *ct,
554 const uint16_t *cic,
555 const struct sockaddr_storage *ss,
556 const struct gsm0808_speech_codec_list *scl,
557 const uint32_t *ci)
558{
559 return gsm0808_create_ass2(ct, cic, ss, scl, ci, NULL, NULL);
560}
561
Max414c8f52019-01-08 14:44:24 +0100562/*! Create BSSMAP Assignment Completed message as per 3GPP TS 48.008 §3.2.1.2
Harald Welte96e2a002017-06-12 21:44:18 +0200563 * \param[in] rr_cause GSM 04.08 RR Cause value
564 * \param[in] chosen_channel Chosen Channel
565 * \param[in] encr_alg_id Encryption Algorithm ID
566 * \param[in] speech_mode Speech Mode
567 * \param[in] ss Socket Address of BSS-side RTP socket
568 * \param[in] sc Speech Codec (current)
569 * \param[in] scl Speech Codec List (permitted)
Max414c8f52019-01-08 14:44:24 +0100570 * \param[in] lcls_bss_status §3.2.2.119 LCLS-BSS-Status, optional
Harald Welte96e2a002017-06-12 21:44:18 +0200571 * \returns callee-allocated msgb with BSSMAP Assignment Complete message */
Max414c8f52019-01-08 14:44:24 +0100572struct msgb *gsm0808_create_ass_compl2(uint8_t rr_cause, uint8_t chosen_channel,
Philipp Maierfa896ab2017-03-27 16:55:32 +0200573 uint8_t encr_alg_id, uint8_t speech_mode,
574 const struct sockaddr_storage *ss,
575 const struct gsm0808_speech_codec *sc,
Max414c8f52019-01-08 14:44:24 +0100576 const struct gsm0808_speech_codec_list *scl,
577 enum gsm0808_lcls_status lcls_bss_status)
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200578{
Harald Welte65c2d362012-01-21 14:26:01 +0100579 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
580 "bssmap: ass compl");
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200581 if (!msg)
582 return NULL;
583
Harald Welte65c2d362012-01-21 14:26:01 +0100584 msgb_v_put(msg, BSS_MAP_MSG_ASSIGMENT_COMPLETE);
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200585
586 /* write 3.2.2.22 */
Harald Welte65c2d362012-01-21 14:26:01 +0100587 msgb_tv_put(msg, GSM0808_IE_RR_CAUSE, rr_cause);
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200588
589 /* write cirtcuit identity code 3.2.2.2 */
590 /* write cell identifier 3.2.2.17 */
591 /* write chosen channel 3.2.2.33 when BTS picked it */
Harald Welte65c2d362012-01-21 14:26:01 +0100592 msgb_tv_put(msg, GSM0808_IE_CHOSEN_CHANNEL, chosen_channel);
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200593
594 /* write chosen encryption algorithm 3.2.2.44 */
Harald Welte65c2d362012-01-21 14:26:01 +0100595 msgb_tv_put(msg, GSM0808_IE_CHOSEN_ENCR_ALG, encr_alg_id);
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200596
597 /* write circuit pool 3.2.2.45 */
598 /* write speech version chosen: 3.2.2.51 when BTS picked it */
Harald Welte65c2d362012-01-21 14:26:01 +0100599 if (speech_mode != 0)
600 msgb_tv_put(msg, GSM0808_IE_SPEECH_VERSION, speech_mode);
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200601
Philipp Maierfa896ab2017-03-27 16:55:32 +0200602 /* AoIP: AoIP Transport Layer Address (BSS) 3.2.2.102 */
603 if (ss)
604 gsm0808_enc_aoip_trasp_addr(msg, ss);
605
606 /* AoIP: Speech Codec (Chosen) 3.2.2.104 */
607 if (sc)
608 gsm0808_enc_speech_codec(msg, sc);
609
610 /* AoIP: add Codec List (BSS Supported) 3.2.2.103 */
611 if (scl)
612 gsm0808_enc_speech_codec_list(msg, scl);
613
Max414c8f52019-01-08 14:44:24 +0100614 /* FIXME: write LSA identifier 3.2.2.15 - see 3GPP TS 43.073 */
615
616 /* LCLS-BSS-Status 3.2.2.119 */
617 if (lcls_bss_status != GSM0808_LCLS_STS_NA)
618 msgb_tv_put(msg, GSM0808_IE_LCLS_BSS_STATUS, lcls_bss_status);
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200619
Harald Welte65c2d362012-01-21 14:26:01 +0100620 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200621
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200622 return msg;
623}
624
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200625/*! Create BSSMAP Assignment Completed message
Harald Welte96e2a002017-06-12 21:44:18 +0200626 * \param[in] rr_cause GSM 04.08 RR Cause value
627 * \param[in] chosen_channel Chosen Channel
628 * \param[in] encr_alg_id Encryption Algorithm ID
629 * \param[in] speech_mode Speech Mode
Max414c8f52019-01-08 14:44:24 +0100630 * \param[in] ss Socket Address of BSS-side RTP socket
631 * \param[in] sc Speech Codec (current)
632 * \param[in] scl Speech Codec List (permitted)
633 * \returns callee-allocated msgb with BSSMAP Assignment Complete message */
634struct msgb *gsm0808_create_ass_compl(uint8_t rr_cause, uint8_t chosen_channel,
635 uint8_t encr_alg_id, uint8_t speech_mode,
636 const struct sockaddr_storage *ss,
637 const struct gsm0808_speech_codec *sc,
638 const struct gsm0808_speech_codec_list *scl)
639{
640 return gsm0808_create_ass_compl2(rr_cause, chosen_channel, encr_alg_id, speech_mode,
641 ss, sc, scl, GSM0808_LCLS_STS_NA);
642}
643
644/*! Create BSSMAP Assignment Completed message
645 * \param[in] rr_cause GSM 04.08 RR Cause value
646 * \param[in] chosen_channel Chosen Channel
647 * \param[in] encr_alg_id Encryption Algorithm ID
648 * \param[in] speech_mode Speech Mode
Harald Welte96e2a002017-06-12 21:44:18 +0200649 * \returns callee-allocated msgb with BSSMAP Assignment Complete message */
Philipp Maierfa896ab2017-03-27 16:55:32 +0200650struct msgb *gsm0808_create_assignment_completed(uint8_t rr_cause,
651 uint8_t chosen_channel,
652 uint8_t encr_alg_id,
653 uint8_t speech_mode)
654{
Max414c8f52019-01-08 14:44:24 +0100655 return gsm0808_create_ass_compl2(rr_cause, chosen_channel, encr_alg_id,
656 speech_mode, NULL, NULL, NULL, GSM0808_LCLS_STS_NA);
Philipp Maierfa896ab2017-03-27 16:55:32 +0200657}
658
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200659/*! Create BSSMAP Assignment Failure message
Harald Welte96e2a002017-06-12 21:44:18 +0200660 * \param[in] cause BSSMAP Cause value
661 * \param[in] rr_cause GSM 04.08 RR Cause value
662 * \param[in] scl Optional Speech Cdec List (AoIP)
663 * \returns callee-allocated msgb with BSSMAP Assignment Failure message */
Philipp Maierfa896ab2017-03-27 16:55:32 +0200664struct msgb *gsm0808_create_ass_fail(uint8_t cause, const uint8_t *rr_cause,
665 const struct gsm0808_speech_codec_list
666 *scl)
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200667{
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200668 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
669 "bssmap: ass fail");
670 if (!msg)
671 return NULL;
672
Harald Welte65c2d362012-01-21 14:26:01 +0100673 msgb_v_put(msg, BSS_MAP_MSG_ASSIGMENT_FAILURE);
Philipp Maier4f4905f2018-11-30 13:36:12 +0100674 gsm0808_enc_cause(msg, cause);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200675
676 /* RR cause 3.2.2.22 */
Harald Welte65c2d362012-01-21 14:26:01 +0100677 if (rr_cause)
678 msgb_tv_put(msg, GSM0808_IE_RR_CAUSE, *rr_cause);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200679
680 /* Circuit pool 3.22.45 */
681 /* Circuit pool list 3.2.2.46 */
682
Philipp Maierfa896ab2017-03-27 16:55:32 +0200683 /* AoIP: add Codec List (BSS Supported) 3.2.2.103 */
684 if (scl)
685 gsm0808_enc_speech_codec_list(msg, scl);
686
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200687 /* update the size */
Harald Welte65c2d362012-01-21 14:26:01 +0100688 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
689
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200690 return msg;
691}
Holger Hans Peter Freyther7daa01c2010-04-17 05:14:36 +0200692
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200693/*! Create BSSMAP Assignment Failure message
Harald Welte96e2a002017-06-12 21:44:18 +0200694 * \param[in] cause BSSMAP Cause value
695 * \param[in] rr_cause GSM 04.08 RR Cause value
696 * \returns callee-allocated msgb with BSSMAP Assignment Failure message */
Philipp Maierfa896ab2017-03-27 16:55:32 +0200697struct msgb *gsm0808_create_assignment_failure(uint8_t cause,
698 uint8_t *rr_cause)
699{
700 return gsm0808_create_ass_fail(cause, rr_cause, NULL);
701}
702
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200703/*! Create BSSMAP Clear Request message
Harald Welte96e2a002017-06-12 21:44:18 +0200704 * \param[in] cause BSSMAP Cause value
705 * \returns callee-allocated msgb with BSSMAP Clear Request message */
Holger Hans Peter Freytheraf270a42010-11-04 12:42:50 +0100706struct msgb *gsm0808_create_clear_rqst(uint8_t cause)
707{
708 struct msgb *msg;
709
710 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
711 "bssmap: clear rqst");
712 if (!msg)
713 return NULL;
714
Harald Welte65c2d362012-01-21 14:26:01 +0100715 msgb_v_put(msg, BSS_MAP_MSG_CLEAR_RQST);
Philipp Maier4f4905f2018-11-30 13:36:12 +0100716 gsm0808_enc_cause(msg, cause);
Harald Welte65c2d362012-01-21 14:26:01 +0100717 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
Holger Hans Peter Freytheraf270a42010-11-04 12:42:50 +0100718
Holger Hans Peter Freytheraf270a42010-11-04 12:42:50 +0100719 return msg;
720}
721
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200722/*! Create BSSMAP PAGING message
Harald Welte96e2a002017-06-12 21:44:18 +0200723 * \param[in] imsi Mandatory paged IMSI in string representation
724 * \param[in] tmsi Optional paged TMSI
Stefan Sperling3953f412018-08-28 15:06:30 +0200725 * \param[in] cil Mandatory Cell Identity List (where to page)
Harald Welte96e2a002017-06-12 21:44:18 +0200726 * \param[in] chan_needed Channel Type needed
727 * \returns callee-allocated msgb with BSSMAP PAGING message */
Stefan Sperling11a4d9d2018-02-15 18:28:04 +0100728struct msgb *gsm0808_create_paging2(const char *imsi, const uint32_t *tmsi,
729 const struct gsm0808_cell_id_list2 *cil,
730 const uint8_t *chan_needed)
Philipp Maier3d48ec02017-03-29 17:37:55 +0200731{
732 struct msgb *msg;
733 uint8_t mid_buf[GSM48_MI_SIZE + 2];
734 int mid_len;
735 uint32_t tmsi_sw;
736
Stefan Sperling11a4d9d2018-02-15 18:28:04 +0100737 /* Mandatory elements! */
Philipp Maier3d48ec02017-03-29 17:37:55 +0200738 OSMO_ASSERT(imsi);
739 OSMO_ASSERT(cil);
740
741 /* Malformed IMSI */
742 OSMO_ASSERT(strlen(imsi) <= GSM48_MI_SIZE);
743
744 msg =
745 msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "paging");
746 if (!msg)
747 return NULL;
748
749 /* Message Type 3.2.2.1 */
750 msgb_v_put(msg, BSS_MAP_MSG_PAGING);
751
Stefan Sperling3953f412018-08-28 15:06:30 +0200752 /* mandatory IMSI 3.2.2.6 */
Philipp Maier3d48ec02017-03-29 17:37:55 +0200753 mid_len = gsm48_generate_mid_from_imsi(mid_buf, imsi);
754 msgb_tlv_put(msg, GSM0808_IE_IMSI, mid_len - 2, mid_buf + 2);
755
756 /* TMSI 3.2.2.7 */
757 if (tmsi) {
Harald Welte95871da2017-05-15 12:11:36 +0200758 tmsi_sw = osmo_htonl(*tmsi);
Philipp Maier3d48ec02017-03-29 17:37:55 +0200759 msgb_tlv_put(msg, GSM0808_IE_TMSI, sizeof(*tmsi),
760 (uint8_t *) & tmsi_sw);
761 }
762
Stefan Sperling3953f412018-08-28 15:06:30 +0200763 /* mandatory Cell Identifier List 3.2.2.27 */
764 gsm0808_enc_cell_id_list2(msg, cil);
Philipp Maier3d48ec02017-03-29 17:37:55 +0200765
766 /* Channel Needed 3.2.2.36 */
767 if (chan_needed) {
768 msgb_tv_put(msg, GSM0808_IE_CHANNEL_NEEDED,
769 (*chan_needed) & 0x03);
770 }
771
772 /* pre-pend the header */
773 msg->l3h =
774 msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
775
776 return msg;
777}
778
Stefan Sperling11a4d9d2018-02-15 18:28:04 +0100779/*! DEPRECATED: Use gsm0808_create_paging2 instead.
780 * Create BSSMAP PAGING message.
781 * \param[in] imsi Mandatory paged IMSI in string representation
782 * \param[in] tmsi Optional paged TMSI
783 * \param[in] cil Cell Identity List (where to page)
784 * \param[in] chan_needed Channel Type needed
785 * \returns callee-allocated msgb with BSSMAP PAGING message */
786struct msgb *gsm0808_create_paging(const char *imsi, const uint32_t *tmsi,
787 const struct gsm0808_cell_id_list *cil,
788 const uint8_t *chan_needed)
789{
790 struct gsm0808_cell_id_list2 cil2 = {};
791
792 /* Mandatory emelents! */
793 OSMO_ASSERT(cil);
794
795 if (cil->id_list_len > GSM0808_CELL_ID_LIST2_MAXLEN)
796 return NULL;
797
798 cil2.id_discr = cil->id_discr;
799 memcpy(cil2.id_list, cil->id_list_lac, cil->id_list_len * sizeof(cil2.id_list[0].lac));
800 cil2.id_list_len = cil->id_list_len;
801
802 return gsm0808_create_paging2(imsi, tmsi, &cil2, chan_needed);
803}
804
Neels Hofmeyr70aba3f2018-03-13 03:40:53 +0100805static uint8_t put_old_bss_to_new_bss_information(struct msgb *msg,
806 const struct gsm0808_old_bss_to_new_bss_info *i)
807{
808 uint8_t *old_tail;
809 uint8_t *tlv_len;
810
811 msgb_put_u8(msg, GSM0808_IE_OLD_BSS_TO_NEW_BSS_INFORMATION);
812 tlv_len = msgb_put(msg, 1);
813 old_tail = msg->tail;
814
815 if (i->extra_information_present) {
816 uint8_t val = 0;
817 if (i->extra_information.prec)
818 val |= 1 << 0;
819 if (i->extra_information.lcs)
820 val |= 1 << 1;
821 if (i->extra_information.ue_prob)
822 val |= 1 << 2;
823 msgb_tlv_put(msg, GSM0808_FE_IE_EXTRA_INFORMATION, 1, &val);
824 }
825
826 if (i->current_channel_type_2_present) {
827 uint8_t val[2] = {
828 i->current_channel_type_2.mode,
829 i->current_channel_type_2.field,
830 };
831 msgb_tlv_put(msg, GSM0808_FE_IE_CURRENT_CHANNEL_TYPE_2, 2, val);
832 }
833
834 *tlv_len = (uint8_t) (msg->tail - old_tail);
835 return *tlv_len + 2;
836}
837
838/*! Create BSSMAP HANDOVER REQUIRED message.
839 * \param[in] params All information to be encoded.
Neels Hofmeyr302aafc2019-04-10 19:23:45 +0200840 * \returns newly allocated msgb with BSSMAP HANDOVER REQUIRED message. */
Neels Hofmeyr70aba3f2018-03-13 03:40:53 +0100841struct msgb *gsm0808_create_handover_required(const struct gsm0808_handover_required *params)
842{
843 struct msgb *msg;
844
845 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-REQUIRED");
846 if (!msg)
847 return NULL;
848
849 /* Message Type, 3.2.2.1 */
850 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_REQUIRED);
851
852 /* Cause, 3.2.2.5 */
Philipp Maier4f4905f2018-11-30 13:36:12 +0100853 gsm0808_enc_cause(msg, params->cause);
Neels Hofmeyr70aba3f2018-03-13 03:40:53 +0100854
855 /* Cell Identifier List, 3.2.2.27 */
856 gsm0808_enc_cell_id_list2(msg, &params->cil);
857
858 /* Current Channel Type 1, 3.2.2.49 */
859 if (params->current_channel_type_1_present)
860 msgb_tv_fixed_put(msg, GSM0808_IE_CURRENT_CHANNEL_TYPE_1, 1, &params->current_channel_type_1);
861
862 /* Speech Version (Used), 3.2.2.51 */
863 if (params->speech_version_used_present)
Neels Hofmeyr302aafc2019-04-10 19:23:45 +0200864 msgb_tv_put(msg, GSM0808_IE_SPEECH_VERSION, params->speech_version_used);
Neels Hofmeyr70aba3f2018-03-13 03:40:53 +0100865
866 if (params->old_bss_to_new_bss_info_present)
867 put_old_bss_to_new_bss_information(msg, &params->old_bss_to_new_bss_info);
868
869 /* pre-pend the header */
870 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
871
872 return msg;
873}
874
Neels Hofmeyrf7e9c512019-03-06 04:25:38 +0100875/*! Create BSSMAP HANDOVER REQUIRED REJECT message.
876 * \returns newly allocated msgb with BSSMAP HANDOVER REQUIRED REJECT message. */
877struct msgb *gsm0808_create_handover_required_reject(const struct gsm0808_handover_required_reject *params)
878{
879 struct msgb *msg;
880
881 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-REQUIRED-REJECT");
882 if (!msg)
883 return NULL;
884
885 /* Message Type, 3.2.2.1 */
886 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_REQUIRED_REJECT);
887
888 /* Cause, 3.2.2.5 */
889 gsm0808_enc_cause(msg, params->cause);
890
891 /* prepend the header */
892 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
893
894 return msg;
895}
896
897/*! Create BSSMAP HANDOVER REQUEST message, 3GPP TS 48.008 3.2.1.8.
898 * Sent from the MSC to the potential new target cell during inter-BSC handover, or to the target MSC during inter-MSC
899 * handover.
900 */
901struct msgb *gsm0808_create_handover_request(const struct gsm0808_handover_request *params)
902{
903 struct msgb *msg;
904
905 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-REQUEST");
906 if (!msg)
907 return NULL;
908
909 /* Message Type, 3.2.2.1 */
910 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_RQST);
911
912 /* Channel Type 3.2.2.11 */
913 gsm0808_enc_channel_type(msg, &params->channel_type);
914
915 /* Encryption Information 3.2.2.10 */
916 gsm0808_enc_encrypt_info(msg, &params->encryption_information);
917
918 /* Classmark Information 1 3.2.2.30 or Classmark Information 2 3.2.2.19 (Classmark 2 wins) */
919 if (params->classmark_information.classmark2_len) {
920 msgb_tlv_put(msg, GSM0808_IE_CLASSMARK_INFORMATION_T2,
921 params->classmark_information.classmark2_len,
922 (const uint8_t*)&params->classmark_information.classmark2);
923 } else if (params->classmark_information.classmark1_set) {
924 msgb_tlv_put(msg, GSM0808_IE_CLASSMARK_INFORMATION_TYPE_1,
925 sizeof(params->classmark_information.classmark1),
926 (const uint8_t*)&params->classmark_information.classmark1);
927 }
928 /* (Classmark 3 possibly follows below) */
929
930 /* Cell Identifier (Serving) , 3.2.2.17 */
931 gsm0808_enc_cell_id(msg, &params->cell_identifier_serving);
932
933 /* Cell Identifier (Target) , 3.2.2.17 */
934 gsm0808_enc_cell_id(msg, &params->cell_identifier_target);
935
936 /* Cause, 3.2.2.5 */
937 gsm0808_enc_cause(msg, params->cause);
938
939 /* Classmark Information 3 3.2.2.20 */
940 if (params->classmark_information.classmark3_len) {
941 msgb_tlv_put(msg, GSM0808_IE_CLASSMARK_INFORMATION_T3,
942 params->classmark_information.classmark3_len,
943 (const uint8_t*)&params->classmark_information.classmark3);
944 }
945
946 /* Current Channel type 1 3.2.2.49 */
947 if (params->current_channel_type_1_present)
948 msgb_tv_fixed_put(msg, GSM0808_IE_CURRENT_CHANNEL_TYPE_1, 1, &params->current_channel_type_1);
949
950 /* Speech Version (Used), 3.2.2.51 */
951 if (params->speech_version_used) {
952 msgb_tv_put(msg, GSM0808_IE_SPEECH_VERSION, params->speech_version_used);
953 }
954
955 /* Chosen Encryption Algorithm (Serving) 3.2.2.44 */
956 if (params->chosen_encryption_algorithm_serving)
957 msgb_tv_put(msg, GSM0808_IE_CHOSEN_ENCR_ALG, params->chosen_encryption_algorithm_serving);
958
959 /* Old BSS to New BSS Information 3.2.2.58 */
960 if (params->old_bss_to_new_bss_info_raw && params->old_bss_to_new_bss_info_raw_len) {
961 msgb_tlv_put(msg, GSM0808_IE_OLD_BSS_TO_NEW_BSS_INFORMATION,
962 params->old_bss_to_new_bss_info_raw_len,
963 params->old_bss_to_new_bss_info_raw);
964 } else if (params->old_bss_to_new_bss_info_present) {
965 put_old_bss_to_new_bss_information(msg, &params->old_bss_to_new_bss_info);
966 }
967
968 /* IMSI 3.2.2.6 */
969 if (params->imsi) {
970 uint8_t mid_buf[GSM48_MI_SIZE + 2];
971 int mid_len = gsm48_generate_mid_from_imsi(mid_buf, params->imsi);
972 msgb_tlv_put(msg, GSM0808_IE_IMSI, mid_len - 2, mid_buf + 2);
973 }
974
975 if (params->aoip_transport_layer)
976 gsm0808_enc_aoip_trasp_addr(msg, params->aoip_transport_layer);
977
978 if (params->codec_list_msc_preferred)
979 gsm0808_enc_speech_codec_list(msg, params->codec_list_msc_preferred);
980
981 if (params->call_id_present) {
982 uint8_t val[4];
983 osmo_store32le(params->call_id, val);
984 msgb_tv_fixed_put(msg, GSM0808_IE_CALL_ID, 4, val);
985 }
986
987 if (params->global_call_reference && params->global_call_reference_len) {
988 msgb_tlv_put(msg, GSM0808_IE_GLOBAL_CALL_REF,
989 params->global_call_reference_len, params->global_call_reference);
990 }
991
992 /* prepend header with final length */
993 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
994
995 return msg;
996}
997
Neels Hofmeyr60f31532018-04-16 22:42:09 +0200998/*! Create BSSMAP HANDOVER REQUEST ACKNOWLEDGE message, 3GPP TS 48.008 3.2.1.10.
999 * Sent from the MT BSC back to the MSC when it has allocated an lchan to handover to.
1000 * l3_info is the RR Handover Command that the MO BSC sends to the MS to move over. */
Neels Hofmeyr73b943e2019-03-14 04:10:25 +01001001struct msgb *gsm0808_create_handover_request_ack2(const struct gsm0808_handover_request_ack *params)
Neels Hofmeyrb662b362018-04-16 22:31:15 +02001002{
1003 struct msgb *msg;
1004
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001005 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-REQUEST-ACK");
Neels Hofmeyrb662b362018-04-16 22:31:15 +02001006 if (!msg)
1007 return NULL;
1008
1009 /* Message Type, 3.2.2.1 */
1010 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_RQST_ACKNOWLEDGE);
1011
Neels Hofmeyr73b943e2019-03-14 04:10:25 +01001012 /* Layer 3 Information, 3.2.2.24 -- it is actually mandatory, but rather compose a nonstandard message than
1013 * segfault or return NULL without a log message. */
1014 if (params->l3_info && params->l3_info_len)
1015 msgb_tlv_put(msg, GSM0808_IE_LAYER_3_INFORMATION, params->l3_info_len, params->l3_info);
Neels Hofmeyrb662b362018-04-16 22:31:15 +02001016
Neels Hofmeyr73b943e2019-03-14 04:10:25 +01001017 if (params->chosen_channel_present)
1018 msgb_tv_put(msg, GSM0808_IE_CHOSEN_CHANNEL, params->chosen_channel);
1019 if (params->chosen_encr_alg)
1020 msgb_tv_put(msg, GSM0808_IE_CHOSEN_ENCR_ALG, params->chosen_encr_alg);
1021
1022 if (params->chosen_speech_version != 0)
1023 msgb_tv_put(msg, GSM0808_IE_SPEECH_VERSION, params->chosen_speech_version);
1024
1025 if (params->aoip_transport_layer)
1026 gsm0808_enc_aoip_trasp_addr(msg, params->aoip_transport_layer);
1027
1028 /* AoIP: Speech Codec (Chosen) 3.2.2.104 */
1029 if (params->speech_codec_chosen_present)
1030 gsm0808_enc_speech_codec(msg, &params->speech_codec_chosen);
Neels Hofmeyrb662b362018-04-16 22:31:15 +02001031
Neels Hofmeyr43c266f2018-08-28 01:08:38 +02001032 /* prepend header with final length */
1033 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1034
Neels Hofmeyrb662b362018-04-16 22:31:15 +02001035 return msg;
1036}
1037
Neels Hofmeyr73b943e2019-03-14 04:10:25 +01001038/*! Same as gsm0808_create_handover_request_ack2() but with less parameters.
1039 * In particular, this lacks the AoIP Transport Layer address. */
1040struct msgb *gsm0808_create_handover_request_ack(const uint8_t *l3_info, uint8_t l3_info_len,
1041 uint8_t chosen_channel, uint8_t chosen_encr_alg,
1042 uint8_t chosen_speech_version)
1043{
1044 struct gsm0808_handover_request_ack params = {
1045 .l3_info = l3_info,
1046 .l3_info_len = l3_info_len,
1047 .chosen_channel = chosen_channel,
1048 .chosen_encr_alg = chosen_encr_alg,
1049 .chosen_speech_version = chosen_speech_version,
1050 };
1051
1052 return gsm0808_create_handover_request_ack2(&params);
1053}
1054
Neels Hofmeyrf7e9c512019-03-06 04:25:38 +01001055/*! Create BSSMAP HANDOVER COMMAND message, 3GPP TS 48.008 3.2.1.11.
1056 * Sent from the MSC to the old BSS to transmit the RR Handover Command received from the new BSS. */
1057struct msgb *gsm0808_create_handover_command(const struct gsm0808_handover_command *params)
1058{
1059 struct msgb *msg;
1060
1061 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-COMMAND");
1062 if (!msg)
1063 return NULL;
1064
1065 /* Message Type, 3.2.2.1 */
1066 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_CMD);
1067
1068 msgb_tlv_put(msg, GSM0808_IE_LAYER_3_INFORMATION, params->l3_info_len, params->l3_info);
1069
1070 if (params->cell_identifier.id_discr != CELL_IDENT_NO_CELL)
1071 gsm0808_enc_cell_id(msg, &params->cell_identifier);
1072
1073 if (params->new_bss_to_old_bss_info_raw
1074 && params->new_bss_to_old_bss_info_raw_len)
1075 msgb_tlv_put(msg, GSM0808_IE_NEW_BSS_TO_OLD_BSS_INFO, params->new_bss_to_old_bss_info_raw_len,
1076 params->new_bss_to_old_bss_info_raw);
1077
1078 /* prepend header with final length */
1079 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1080
1081 return msg;
1082}
1083
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001084/*! Create BSSMAP HANDOVER DETECT message, 3GPP TS 48.008 3.2.1.40.
1085 * Sent from the MT BSC back to the MSC when the MS has sent a handover RACH request and the MT BSC has
1086 * received the Handover Detect message. */
1087struct msgb *gsm0808_create_handover_detect()
1088{
1089 struct msgb *msg;
1090
1091 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-DETECT");
1092 if (!msg)
1093 return NULL;
1094
1095 /* Message Type, 3.2.2.1 */
1096 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_DETECT);
1097
Neels Hofmeyr43c266f2018-08-28 01:08:38 +02001098 /* prepend header with final length */
1099 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1100
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001101 return msg;
1102}
1103
Neels Hofmeyrf7e9c512019-03-06 04:25:38 +01001104/*! Create BSSMAP HANDOVER SUCCEEDED message, 3GPP TS 48.008 3.2.1.13.
1105 * Sent from the MSC back to the old BSS to notify that the MS has successfully accessed the new BSS. */
1106struct msgb *gsm0808_create_handover_succeeded()
1107{
1108 struct msgb *msg;
1109
1110 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-DETECT");
1111 if (!msg)
1112 return NULL;
1113
1114 /* Message Type, 3.2.2.1 */
1115 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_SUCCEEDED);
1116
1117 /* prepend header with final length */
1118 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1119
1120 return msg;
1121}
1122
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001123/*! Create BSSMAP HANDOVER COMPLETE message, 3GPP TS 48.008 3.2.1.12.
1124 * Sent from the MT BSC back to the MSC when the MS has fully settled into the new lchan. */
1125struct msgb *gsm0808_create_handover_complete(const struct gsm0808_handover_complete *params)
1126{
1127 struct msgb *msg;
1128
1129 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-COMPLETE");
1130 if (!msg)
1131 return NULL;
1132
1133 /* Message Type, 3.2.2.1 */
1134 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_COMPLETE);
1135
1136 /* RR Cause, 3.2.2.22 */
1137 if (params->rr_cause_present)
1138 msgb_tlv_put(msg, GSM0808_IE_RR_CAUSE, 1, &params->rr_cause);
1139
1140 /* AoIP: Speech Codec (Chosen) 3.2.2.104 */
1141 if (params->speech_codec_chosen_present)
1142 gsm0808_enc_speech_codec(msg, &params->speech_codec_chosen);
1143
1144 /* AoIP: add Codec List (BSS Supported) 3.2.2.103 */
1145 if (params->codec_list_bss_supported.len)
1146 gsm0808_enc_speech_codec_list(msg, &params->codec_list_bss_supported);
1147
1148 /* Chosen Encryption Algorithm 3.2.2.44 */
1149 if (params->chosen_encr_alg_present)
1150 msgb_tv_put(msg, GSM0808_IE_CHOSEN_ENCR_ALG, params->chosen_encr_alg);
1151
1152 /* LCLS-BSS-Status 3.2.2.119 */
1153 if (params->lcls_bss_status_present)
1154 msgb_tv_put(msg, GSM0808_IE_LCLS_BSS_STATUS, params->lcls_bss_status);
1155
Neels Hofmeyr43c266f2018-08-28 01:08:38 +02001156 /* prepend header with final length */
1157 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1158
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001159 return msg;
1160}
1161
1162/*! Create BSSMAP HANDOVER FAILURE message, 3GPP TS 48.008 3.2.1.16.
1163 * Sent from the MT BSC back to the MSC when the handover has failed. */
1164struct msgb *gsm0808_create_handover_failure(const struct gsm0808_handover_failure *params)
1165{
1166 struct msgb *msg;
1167
1168 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-FAILURE");
1169 if (!msg)
1170 return NULL;
1171
1172 /* Message Type, 3.2.2.1 */
1173 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_FAILURE);
1174
1175 /* Cause, 3.2.2.5 */
Philipp Maier4f4905f2018-11-30 13:36:12 +01001176 gsm0808_enc_cause(msg, params->cause);
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001177
1178 /* RR Cause, 3.2.2.22 */
1179 if (params->rr_cause_present)
1180 msgb_tlv_put(msg, GSM0808_IE_RR_CAUSE, 1, &params->rr_cause);
1181
1182 /* AoIP: add Codec List (BSS Supported) 3.2.2.103 */
1183 if (params->codec_list_bss_supported.len)
1184 gsm0808_enc_speech_codec_list(msg, &params->codec_list_bss_supported);
1185
Neels Hofmeyr43c266f2018-08-28 01:08:38 +02001186 /* prepend header with final length */
1187 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1188
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001189 return msg;
1190}
1191
Philipp Maier225bdf42018-10-30 14:56:59 +01001192/*! Create BSSMAP HANDOVER PERFORMED message, 3GPP TS 48.008 3.2.1.25.
1193 * \param[in] params All information to be encoded.
1194 * \returns callee-allocated msgb with BSSMAP HANDOVER PERFORMED message */
1195struct msgb *gsm0808_create_handover_performed(const struct gsm0808_handover_performed *params)
1196{
1197 struct msgb *msg;
1198
1199 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-PERFORMED");
1200 if (!msg)
1201 return NULL;
1202
1203 /* Message Type, 3.2.2.1 */
1204 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_PERFORMED);
1205
1206 /* Cause, 3.2.2.5 */
Philipp Maier4f4905f2018-11-30 13:36:12 +01001207 gsm0808_enc_cause(msg, params->cause);
Philipp Maier225bdf42018-10-30 14:56:59 +01001208
1209 /* Cell Identifier, 3.2.2.17 */
1210 gsm0808_enc_cell_id(msg, &params->cell_id);
1211
1212 /* Chosen Channel 3.2.2.33 */
1213 if (params->chosen_channel_present)
1214 msgb_tv_put(msg, GSM0808_IE_CHOSEN_CHANNEL, params->chosen_channel);
1215
1216 /* Chosen Encryption Algorithm 3.2.2.44 */
1217 if (params->chosen_encr_alg_present)
1218 msgb_tv_put(msg, GSM0808_IE_CHOSEN_ENCR_ALG, params->chosen_encr_alg);
1219
1220 /* Speech Version (chosen) 3.2.2.51 */
1221 if (params->speech_version_chosen_present)
1222 msgb_tv_put(msg, GSM0808_IE_SPEECH_VERSION, params->speech_version_chosen);
1223
1224 /* AoIP: Speech Codec (chosen) 3.2.2.104 */
1225 if (params->speech_codec_chosen_present)
1226 gsm0808_enc_speech_codec(msg, &params->speech_codec_chosen);
1227
1228 /* LCLS-BSS-Status 3.2.2.119 */
1229 if (params->lcls_bss_status_present)
1230 msgb_tv_put(msg, GSM0808_IE_LCLS_BSS_STATUS, params->lcls_bss_status);
1231
1232 /* prepend header with final length */
1233 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1234
1235 return msg;
1236}
1237
Neels Hofmeyr87e45502017-06-20 00:17:59 +02001238/*! Prepend a DTAP header to given Message Buffer
Harald Welte96e2a002017-06-12 21:44:18 +02001239 * \param[in] msgb Message Buffer
1240 * \param[in] link_id Link Identifier */
Holger Hans Peter Freyther9a3dec02010-05-16 08:15:40 +08001241void gsm0808_prepend_dtap_header(struct msgb *msg, uint8_t link_id)
1242{
1243 uint8_t *hh = msgb_push(msg, 3);
1244 hh[0] = BSSAP_MSG_DTAP;
1245 hh[1] = link_id;
1246 hh[2] = msg->len - 3;
1247}
1248
Neels Hofmeyr87e45502017-06-20 00:17:59 +02001249/*! Create BSSMAP DTAP message
Harald Welte96e2a002017-06-12 21:44:18 +02001250 * \param[in] msg_l3 Messge Buffer containing Layer3 message
1251 * \param[in] link_id Link Identifier
1252 * \returns callee-allocated msgb with BSSMAP DTAP message */
Holger Hans Peter Freytherc25c6682010-11-04 12:26:06 +01001253struct msgb *gsm0808_create_dtap(struct msgb *msg_l3, uint8_t link_id)
1254{
1255 struct dtap_header *header;
1256 uint8_t *data;
1257 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
1258 "dtap");
1259 if (!msg)
1260 return NULL;
1261
1262 /* DTAP header */
1263 msg->l3h = msgb_put(msg, sizeof(*header));
1264 header = (struct dtap_header *) &msg->l3h[0];
1265 header->type = BSSAP_MSG_DTAP;
1266 header->link_id = link_id;
1267 header->length = msgb_l3len(msg_l3);
1268
1269 /* Payload */
1270 data = msgb_put(msg, header->length);
1271 memcpy(data, msg_l3->l3h, header->length);
1272
1273 return msg;
1274}
1275
Harald Welte92107df2014-06-21 23:16:20 +02001276/* As per 3GPP TS 48.008 version 11.7.0 Release 11 */
Holger Hans Peter Freyther7daa01c2010-04-17 05:14:36 +02001277static const struct tlv_definition bss_att_tlvdef = {
1278 .def = {
Harald Welte92107df2014-06-21 23:16:20 +02001279 [GSM0808_IE_CIRCUIT_IDENTITY_CODE] = { TLV_TYPE_FIXED, 2 },
1280 [GSM0808_IE_CONNECTION_RELEASE_RQSTED] = { TLV_TYPE_TV },
1281 [GSM0808_IE_RESOURCE_AVAILABLE] = { TLV_TYPE_FIXED, 21 },
1282 [GSM0808_IE_CAUSE] = { TLV_TYPE_TLV },
Holger Hans Peter Freyther7daa01c2010-04-17 05:14:36 +02001283 [GSM0808_IE_IMSI] = { TLV_TYPE_TLV },
1284 [GSM0808_IE_TMSI] = { TLV_TYPE_TLV },
Harald Welte92107df2014-06-21 23:16:20 +02001285 [GSM0808_IE_NUMBER_OF_MSS] = { TLV_TYPE_TV },
Dmitri Soloviev29099422013-07-11 09:25:37 +02001286 [GSM0808_IE_LAYER_3_HEADER_INFORMATION] = { TLV_TYPE_TLV },
Harald Welte92107df2014-06-21 23:16:20 +02001287 [GSM0808_IE_ENCRYPTION_INFORMATION] = { TLV_TYPE_TLV },
1288 [GSM0808_IE_CHANNEL_TYPE] = { TLV_TYPE_TLV },
1289 [GSM0808_IE_PERIODICITY] = { TLV_TYPE_TV },
1290 [GSM0808_IE_EXTENDED_RESOURCE_INDICATOR]= { TLV_TYPE_TV },
1291 [GSM0808_IE_TOTAL_RESOURCE_ACCESSIBLE] = { TLV_TYPE_FIXED, 4 },
1292 [GSM0808_IE_LSA_IDENTIFIER] = { TLV_TYPE_TLV },
1293 [GSM0808_IE_LSA_IDENTIFIER_LIST] = { TLV_TYPE_TLV },
Holger Hans Peter Freyther715e9452014-08-21 14:17:45 +02001294 [GSM0808_IE_LSA_INFORMATION] = { TLV_TYPE_TLV },
Harald Welte92107df2014-06-21 23:16:20 +02001295 [GSM0808_IE_CELL_IDENTIFIER] = { TLV_TYPE_TLV },
1296 [GSM0808_IE_PRIORITY] = { TLV_TYPE_TLV },
1297 [GSM0808_IE_CLASSMARK_INFORMATION_T2] = { TLV_TYPE_TLV },
1298 [GSM0808_IE_CLASSMARK_INFORMATION_T3] = { TLV_TYPE_TLV },
1299 [GSM0808_IE_INTERFERENCE_BAND_TO_USE] = { TLV_TYPE_TV },
1300 [GSM0808_IE_RR_CAUSE] = { TLV_TYPE_TV },
Harald Welte92107df2014-06-21 23:16:20 +02001301 [GSM0808_IE_LAYER_3_INFORMATION] = { TLV_TYPE_TLV },
1302 [GSM0808_IE_DLCI] = { TLV_TYPE_TV },
1303 [GSM0808_IE_DOWNLINK_DTX_FLAG] = { TLV_TYPE_TV },
1304 [GSM0808_IE_CELL_IDENTIFIER_LIST] = { TLV_TYPE_TLV },
1305 [GSM0808_IE_CELL_ID_LIST_SEGMENT] = { TLV_TYPE_TLV },
1306 [GSM0808_IE_CELL_ID_LIST_SEG_EST_CELLS] = { TLV_TYPE_TLV },
1307 [GSM0808_IE_CELL_ID_LIST_SEG_CELLS_TBE] = { TLV_TYPE_TLV },
1308 [GSM0808_IE_CELL_ID_LIST_SEG_REL_CELLS] = { TLV_TYPE_TLV },
1309 [GSM0808_IE_CELL_ID_LIST_SEG_NE_CELLS] = { TLV_TYPE_TLV },
1310 [GSM0808_IE_RESPONSE_RQST] = { TLV_TYPE_T },
1311 [GSM0808_IE_RESOURCE_INDICATION_METHOD] = { TLV_TYPE_TV },
1312 [GSM0808_IE_CLASSMARK_INFORMATION_TYPE_1] = { TLV_TYPE_TV },
1313 [GSM0808_IE_CIRCUIT_IDENTITY_CODE_LIST] = { TLV_TYPE_TLV },
1314 [GSM0808_IE_DIAGNOSTIC] = { TLV_TYPE_TLV },
1315 [GSM0808_IE_CHOSEN_CHANNEL] = { TLV_TYPE_TV },
1316 [GSM0808_IE_CIPHER_RESPONSE_MODE] = { TLV_TYPE_TV },
1317 [GSM0808_IE_LAYER_3_MESSAGE_CONTENTS] = { TLV_TYPE_TLV },
1318 [GSM0808_IE_CHANNEL_NEEDED] = { TLV_TYPE_TV },
1319 [GSM0808_IE_TRACE_TYPE] = { TLV_TYPE_TV },
1320 [GSM0808_IE_TRIGGERID] = { TLV_TYPE_TLV },
1321 [GSM0808_IE_TRACE_REFERENCE] = { TLV_TYPE_TV },
1322 [GSM0808_IE_TRANSACTIONID] = { TLV_TYPE_TLV },
1323 [GSM0808_IE_MOBILE_IDENTITY] = { TLV_TYPE_TLV },
1324 [GSM0808_IE_OMCID] = { TLV_TYPE_TLV },
1325 [GSM0808_IE_FORWARD_INDICATOR] = { TLV_TYPE_TV },
Holger Hans Peter Freytherc2b7f922010-08-04 18:50:43 +08001326 [GSM0808_IE_CHOSEN_ENCR_ALG] = { TLV_TYPE_TV },
Harald Welte92107df2014-06-21 23:16:20 +02001327 [GSM0808_IE_CIRCUIT_POOL] = { TLV_TYPE_TV },
1328 [GSM0808_IE_CIRCUIT_POOL_LIST] = { TLV_TYPE_TLV },
1329 [GSM0808_IE_TIME_INDICATION] = { TLV_TYPE_TV },
1330 [GSM0808_IE_RESOURCE_SITUATION] = { TLV_TYPE_TLV },
1331 [GSM0808_IE_CURRENT_CHANNEL_TYPE_1] = { TLV_TYPE_TV },
1332 [GSM0808_IE_QUEUEING_INDICATOR] = { TLV_TYPE_TV },
1333 [GSM0808_IE_SPEECH_VERSION] = { TLV_TYPE_TV },
1334 [GSM0808_IE_ASSIGNMENT_REQUIREMENT] = { TLV_TYPE_TV },
1335 [GSM0808_IE_TALKER_FLAG] = { TLV_TYPE_T },
1336 [GSM0808_IE_GROUP_CALL_REFERENCE] = { TLV_TYPE_TLV },
1337 [GSM0808_IE_EMLPP_PRIORITY] = { TLV_TYPE_TV },
1338 [GSM0808_IE_CONFIG_EVO_INDI] = { TLV_TYPE_TV },
1339 [GSM0808_IE_OLD_BSS_TO_NEW_BSS_INFORMATION] = { TLV_TYPE_TLV },
1340 [GSM0808_IE_LCS_QOS] = { TLV_TYPE_TLV },
1341 [GSM0808_IE_LSA_ACCESS_CTRL_SUPPR] = { TLV_TYPE_TV },
1342 [GSM0808_IE_LCS_PRIORITY] = { TLV_TYPE_TLV },
1343 [GSM0808_IE_LOCATION_TYPE] = { TLV_TYPE_TLV },
1344 [GSM0808_IE_LOCATION_ESTIMATE] = { TLV_TYPE_TLV },
1345 [GSM0808_IE_POSITIONING_DATA] = { TLV_TYPE_TLV },
1346 [GSM0808_IE_LCS_CAUSE] = { TLV_TYPE_TLV },
1347 [GSM0808_IE_APDU] = { TLV_TYPE_TLV },
1348 [GSM0808_IE_NETWORK_ELEMENT_IDENTITY] = { TLV_TYPE_TLV },
1349 [GSM0808_IE_GPS_ASSISTANCE_DATA] = { TLV_TYPE_TLV },
1350 [GSM0808_IE_DECIPHERING_KEYS] = { TLV_TYPE_TLV },
1351 [GSM0808_IE_RETURN_ERROR_RQST] = { TLV_TYPE_TLV },
1352 [GSM0808_IE_RETURN_ERROR_CAUSE] = { TLV_TYPE_TLV },
1353 [GSM0808_IE_SEGMENTATION] = { TLV_TYPE_TLV },
1354 [GSM0808_IE_SERVICE_HANDOVER] = { TLV_TYPE_TLV },
1355 [GSM0808_IE_SOURCE_RNC_TO_TARGET_RNC_TRANSPARENT_UMTS] = { TLV_TYPE_TLV },
1356 [GSM0808_IE_SOURCE_RNC_TO_TARGET_RNC_TRANSPARENT_CDMA2000] = { TLV_TYPE_TLV },
1357 [GSM0808_IE_GERAN_CLASSMARK] = { TLV_TYPE_TLV },
1358 [GSM0808_IE_GERAN_BSC_CONTAINER] = { TLV_TYPE_TLV },
1359 [GSM0808_IE_NEW_BSS_TO_OLD_BSS_INFO] = { TLV_TYPE_TLV },
1360 [GSM0800_IE_INTER_SYSTEM_INFO] = { TLV_TYPE_TLV },
1361 [GSM0808_IE_SNA_ACCESS_INFO] = { TLV_TYPE_TLV },
1362 [GSM0808_IE_VSTK_RAND_INFO] = { TLV_TYPE_TLV },
1363 [GSM0808_IE_PAGING_INFO] = { TLV_TYPE_TV },
1364 [GSM0808_IE_IMEI] = { TLV_TYPE_TLV },
1365 [GSM0808_IE_VELOCITY_ESTIMATE] = { TLV_TYPE_TLV },
1366 [GSM0808_IE_VGCS_FEATURE_FLAGS] = { TLV_TYPE_TLV },
1367 [GSM0808_IE_TALKER_PRIORITY] = { TLV_TYPE_TV },
1368 [GSM0808_IE_EMERGENCY_SET_INDICATION] = { TLV_TYPE_T },
1369 [GSM0808_IE_TALKER_IDENTITY] = { TLV_TYPE_TLV },
1370 [GSM0808_IE_SMS_TO_VGCS] = { TLV_TYPE_TLV },
1371 [GSM0808_IE_VGCS_TALKER_MODE] = { TLV_TYPE_TLV },
1372 [GSM0808_IE_VGCS_VBS_CELL_STATUS] = { TLV_TYPE_TLV },
1373 [GSM0808_IE_GANSS_ASSISTANCE_DATA] = { TLV_TYPE_TLV },
1374 [GSM0808_IE_GANSS_POSITIONING_DATA] = { TLV_TYPE_TLV },
1375 [GSM0808_IE_GANSS_LOCATION_TYPE] = { TLV_TYPE_TLV },
1376 [GSM0808_IE_APP_DATA] = { TLV_TYPE_TLV },
1377 [GSM0808_IE_DATA_IDENTITY] = { TLV_TYPE_TLV },
1378 [GSM0808_IE_APP_DATA_INFO] = { TLV_TYPE_TLV },
1379 [GSM0808_IE_MSISDN] = { TLV_TYPE_TLV },
1380 [GSM0808_IE_AOIP_TRASP_ADDR] = { TLV_TYPE_TLV },
1381 [GSM0808_IE_SPEECH_CODEC_LIST] = { TLV_TYPE_TLV },
1382 [GSM0808_IE_SPEECH_CODEC] = { TLV_TYPE_TLV },
1383 [GSM0808_IE_CALL_ID] = { TLV_TYPE_FIXED, 4 },
1384 [GSM0808_IE_CALL_ID_LIST] = { TLV_TYPE_TLV },
1385 [GSM0808_IE_A_IF_SEL_FOR_RESET] = { TLV_TYPE_TV },
1386 [GSM0808_IE_KC_128] = { TLV_TYPE_FIXED, 16 },
1387 [GSM0808_IE_CSG_IDENTIFIER] = { TLV_TYPE_TLV },
1388 [GSM0808_IE_REDIR_ATTEMPT_FLAG] = { TLV_TYPE_T },
1389 [GSM0808_IE_REROUTE_REJ_CAUSE] = { TLV_TYPE_TV },
1390 [GSM0808_IE_SEND_SEQ_NUM] = { TLV_TYPE_TV },
1391 [GSM0808_IE_REROUTE_COMPL_OUTCOME] = { TLV_TYPE_TV },
1392 [GSM0808_IE_GLOBAL_CALL_REF] = { TLV_TYPE_TLV },
1393 [GSM0808_IE_LCLS_CONFIG] = { TLV_TYPE_TV },
1394 [GSM0808_IE_LCLS_CONN_STATUS_CTRL] = { TLV_TYPE_TV },
1395 [GSM0808_IE_LCLS_CORR_NOT_NEEDED] = { TLV_TYPE_TV },
1396 [GSM0808_IE_LCLS_BSS_STATUS] = { TLV_TYPE_TV },
1397 [GSM0808_IE_LCLS_BREAK_REQ] = { TLV_TYPE_TV },
1398 [GSM0808_IE_CSFB_INDICATION] = { TLV_TYPE_T },
1399 [GSM0808_IE_CS_TO_PS_SRVCC] = { TLV_TYPE_T },
1400 [GSM0808_IE_SRC_ENB_TO_TGT_ENB_TRANSP] = { TLV_TYPE_TLV },
1401 [GSM0808_IE_CS_TO_PS_SRVCC_IND] = { TLV_TYPE_T },
1402 [GSM0808_IE_CN_TO_MS_TRANSP_INFO] = { TLV_TYPE_TLV },
1403 [GSM0808_IE_SELECTED_PLMN_ID] = { TLV_TYPE_FIXED, 3 },
1404 [GSM0808_IE_LAST_USED_EUTRAN_PLMN_ID] = { TLV_TYPE_FIXED, 3 },
Pau Espin Pedrol18506c82019-04-16 15:47:59 +02001405
1406 /* Osmocom extensions */
1407 [GSM0808_IE_OSMO_OSMUX_SUPPORT] = { TLV_TYPE_T },
1408 [GSM0808_IE_OSMO_OSMUX_CID] = { TLV_TYPE_TV },
Holger Hans Peter Freyther7daa01c2010-04-17 05:14:36 +02001409 },
1410};
1411
Harald Weltef4d45ab2011-07-16 12:13:00 +02001412const struct tlv_definition *gsm0808_att_tlvdef(void)
Holger Hans Peter Freyther7daa01c2010-04-17 05:14:36 +02001413{
1414 return &bss_att_tlvdef;
1415}
Harald Welte9b837e62011-07-11 17:43:19 +02001416
Pau Espin Pedrol392f6072019-11-27 12:07:04 +01001417const struct value_string gsm0406_dlci_sapi_names[] = {
1418 { DLCI_SAPI_RR_MM_CC, "RR/MM/CC" },
1419 { DLCI_SAPI_SMS, "SMS" },
1420 { 0, NULL }
1421};
1422
Harald Welte9b837e62011-07-11 17:43:19 +02001423static const struct value_string gsm0808_msgt_names[] = {
1424 { BSS_MAP_MSG_ASSIGMENT_RQST, "ASSIGNMENT REQ" },
1425 { BSS_MAP_MSG_ASSIGMENT_COMPLETE, "ASSIGNMENT COMPL" },
1426 { BSS_MAP_MSG_ASSIGMENT_FAILURE, "ASSIGNMENT FAIL" },
Harald Welte92107df2014-06-21 23:16:20 +02001427 { BSS_MAP_MSG_CHAN_MOD_RQST, "CHANNEL MODIFY REQUEST" },
Harald Welte9b837e62011-07-11 17:43:19 +02001428
1429 { BSS_MAP_MSG_HANDOVER_RQST, "HANDOVER REQ" },
1430 { BSS_MAP_MSG_HANDOVER_REQUIRED, "HANDOVER REQUIRED" },
1431 { BSS_MAP_MSG_HANDOVER_RQST_ACKNOWLEDGE,"HANDOVER REQ ACK" },
1432 { BSS_MAP_MSG_HANDOVER_CMD, "HANDOVER CMD" },
1433 { BSS_MAP_MSG_HANDOVER_COMPLETE, "HANDOVER COMPLETE" },
1434 { BSS_MAP_MSG_HANDOVER_SUCCEEDED, "HANDOVER SUCCESS" },
1435 { BSS_MAP_MSG_HANDOVER_FAILURE, "HANDOVER FAILURE" },
1436 { BSS_MAP_MSG_HANDOVER_PERFORMED, "HANDOVER PERFORMED" },
1437 { BSS_MAP_MSG_HANDOVER_CANDIDATE_ENQUIRE, "HANDOVER CAND ENQ" },
1438 { BSS_MAP_MSG_HANDOVER_CANDIDATE_RESPONSE, "HANDOVER CAND RESP" },
1439 { BSS_MAP_MSG_HANDOVER_REQUIRED_REJECT, "HANDOVER REQ REJ" },
1440 { BSS_MAP_MSG_HANDOVER_DETECT, "HANDOVER DETECT" },
Harald Welte92107df2014-06-21 23:16:20 +02001441 { BSS_MAP_MSG_INT_HANDOVER_REQUIRED, "INT HANDOVER REQ" },
1442 { BSS_MAP_MSG_INT_HANDOVER_REQUIRED_REJ,"INT HANDOVER REQ REJ" },
1443 { BSS_MAP_MSG_INT_HANDOVER_CMD, "INT HANDOVER CMD" },
1444 { BSS_MAP_MSG_INT_HANDOVER_ENQUIRY, "INT HANDOVER ENQ" },
Harald Welte9b837e62011-07-11 17:43:19 +02001445
1446 { BSS_MAP_MSG_CLEAR_CMD, "CLEAR COMMAND" },
1447 { BSS_MAP_MSG_CLEAR_COMPLETE, "CLEAR COMPLETE" },
1448 { BSS_MAP_MSG_CLEAR_RQST, "CLEAR REQUEST" },
1449 { BSS_MAP_MSG_SAPI_N_REJECT, "SAPI N REJECT" },
1450 { BSS_MAP_MSG_CONFUSION, "CONFUSION" },
1451
1452 { BSS_MAP_MSG_SUSPEND, "SUSPEND" },
1453 { BSS_MAP_MSG_RESUME, "RESUME" },
1454 { BSS_MAP_MSG_CONNECTION_ORIENTED_INFORMATION, "CONN ORIENT INFO" },
1455 { BSS_MAP_MSG_PERFORM_LOCATION_RQST, "PERFORM LOC REQ" },
1456 { BSS_MAP_MSG_LSA_INFORMATION, "LSA INFORMATION" },
1457 { BSS_MAP_MSG_PERFORM_LOCATION_RESPONSE, "PERFORM LOC RESP" },
1458 { BSS_MAP_MSG_PERFORM_LOCATION_ABORT, "PERFORM LOC ABORT" },
1459 { BSS_MAP_MSG_COMMON_ID, "COMMON ID" },
Harald Welte92107df2014-06-21 23:16:20 +02001460 { BSS_MAP_MSG_REROUTE_CMD, "REROUTE COMMAND" },
1461 { BSS_MAP_MSG_REROUTE_COMPLETE, "REROUTE COMPLETE" },
Harald Welte9b837e62011-07-11 17:43:19 +02001462
1463 { BSS_MAP_MSG_RESET, "RESET" },
1464 { BSS_MAP_MSG_RESET_ACKNOWLEDGE, "RESET ACK" },
1465 { BSS_MAP_MSG_OVERLOAD, "OVERLOAD" },
1466 { BSS_MAP_MSG_RESET_CIRCUIT, "RESET CIRCUIT" },
1467 { BSS_MAP_MSG_RESET_CIRCUIT_ACKNOWLEDGE, "RESET CIRCUIT ACK" },
1468 { BSS_MAP_MSG_MSC_INVOKE_TRACE, "MSC INVOKE TRACE" },
1469 { BSS_MAP_MSG_BSS_INVOKE_TRACE, "BSS INVOKE TRACE" },
1470 { BSS_MAP_MSG_CONNECTIONLESS_INFORMATION, "CONNLESS INFO" },
Harald Welte92107df2014-06-21 23:16:20 +02001471 { BSS_MAP_MSG_RESET_IP_RSRC, "RESET IP RESOURCE" },
1472 { BSS_MAP_MSG_RESET_IP_RSRC_ACK, "RESET IP RESOURCE ACK" },
Harald Welte9b837e62011-07-11 17:43:19 +02001473
1474 { BSS_MAP_MSG_BLOCK, "BLOCK" },
1475 { BSS_MAP_MSG_BLOCKING_ACKNOWLEDGE, "BLOCK ACK" },
1476 { BSS_MAP_MSG_UNBLOCK, "UNBLOCK" },
1477 { BSS_MAP_MSG_UNBLOCKING_ACKNOWLEDGE, "UNBLOCK ACK" },
1478 { BSS_MAP_MSG_CIRCUIT_GROUP_BLOCK, "CIRC GROUP BLOCK" },
1479 { BSS_MAP_MSG_CIRCUIT_GROUP_BLOCKING_ACKNOWLEDGE, "CIRC GORUP BLOCK ACK" },
1480 { BSS_MAP_MSG_CIRCUIT_GROUP_UNBLOCK, "CIRC GROUP UNBLOCK" },
1481 { BSS_MAP_MSG_CIRCUIT_GROUP_UNBLOCKING_ACKNOWLEDGE, "CIRC GROUP UNBLOCK ACK" },
1482 { BSS_MAP_MSG_UNEQUIPPED_CIRCUIT, "UNEQUIPPED CIRCUIT" },
1483 { BSS_MAP_MSG_CHANGE_CIRCUIT, "CHANGE CIRCUIT" },
1484 { BSS_MAP_MSG_CHANGE_CIRCUIT_ACKNOWLEDGE, "CHANGE CIRCUIT ACK" },
1485
1486 { BSS_MAP_MSG_RESOURCE_RQST, "RESOURCE REQ" },
1487 { BSS_MAP_MSG_RESOURCE_INDICATION, "RESOURCE IND" },
1488 { BSS_MAP_MSG_PAGING, "PAGING" },
1489 { BSS_MAP_MSG_CIPHER_MODE_CMD, "CIPHER MODE CMD" },
1490 { BSS_MAP_MSG_CLASSMARK_UPDATE, "CLASSMARK UPDATE" },
1491 { BSS_MAP_MSG_CIPHER_MODE_COMPLETE, "CIPHER MODE COMPLETE" },
1492 { BSS_MAP_MSG_QUEUING_INDICATION, "QUEUING INDICATION" },
1493 { BSS_MAP_MSG_COMPLETE_LAYER_3, "COMPLETE LAYER 3" },
1494 { BSS_MAP_MSG_CLASSMARK_RQST, "CLASSMARK REQ" },
1495 { BSS_MAP_MSG_CIPHER_MODE_REJECT, "CIPHER MODE REJECT" },
1496 { BSS_MAP_MSG_LOAD_INDICATION, "LOAD IND" },
1497
Harald Welte92107df2014-06-21 23:16:20 +02001498 { BSS_MAP_MSG_VGCS_VBS_SETUP, "VGCS/VBS SETUP" },
1499 { BSS_MAP_MSG_VGCS_VBS_SETUP_ACK, "VGCS/VBS SETUP ACK" },
1500 { BSS_MAP_MSG_VGCS_VBS_SETUP_REFUSE, "VGCS/VBS SETUP REFUSE" },
1501 { BSS_MAP_MSG_VGCS_VBS_ASSIGNMENT_RQST, "VGCS/VBS ASSIGN REQ" },
1502 { BSS_MAP_MSG_VGCS_VBS_ASSIGNMENT_RESULT, "VGCS/VBS ASSIGN RES" },
1503 { BSS_MAP_MSG_VGCS_VBS_ASSIGNMENT_FAILURE, "VGCS/VBS ASSIGN FAIL" },
1504 { BSS_MAP_MSG_VGCS_VBS_QUEUING_INDICATION, "VGCS/VBS QUEUING IND" },
1505 { BSS_MAP_MSG_UPLINK_RQST, "UPLINK REQ" },
1506 { BSS_MAP_MSG_UPLINK_RQST_ACKNOWLEDGE, "UPLINK REQ ACK" },
1507 { BSS_MAP_MSG_UPLINK_RQST_CONFIRMATION, "UPLINK REQ CONF" },
1508 { BSS_MAP_MSG_UPLINK_RELEASE_INDICATION,"UPLINK REL IND" },
1509 { BSS_MAP_MSG_UPLINK_REJECT_CMD, "UPLINK REJ CMD" },
1510 { BSS_MAP_MSG_UPLINK_RELEASE_CMD, "UPLINK REL CMD" },
1511 { BSS_MAP_MSG_UPLINK_SEIZED_CMD, "UPLINK SEIZED CMD" },
1512 { BSS_MAP_MSG_VGCS_ADDL_INFO, "VGCS ADDL INFO" },
1513 { BSS_MAP_MSG_NOTIFICATION_DATA, "NOTIF DATA" },
1514 { BSS_MAP_MSG_UPLINK_APP_DATA, "UPLINK APP DATA" },
1515
1516 { BSS_MAP_MSG_LCLS_CONNECT_CTRL, "LCLS-CONNECT-CONTROL" },
1517 { BSS_MAP_MSG_LCLS_CONNECT_CTRL_ACK, "CLS-CONNECT-CONTROL-ACK" },
1518 { BSS_MAP_MSG_LCLS_NOTIFICATION, "LCLS-NOTIFICATION" },
Harald Welte9b837e62011-07-11 17:43:19 +02001519
1520 { 0, NULL }
1521};
1522
Neels Hofmeyr87e45502017-06-20 00:17:59 +02001523/*! Return string name of BSSMAP Message Type */
Harald Welte9b837e62011-07-11 17:43:19 +02001524const char *gsm0808_bssmap_name(uint8_t msg_type)
1525{
1526 return get_value_string(gsm0808_msgt_names, msg_type);
1527}
1528
1529static const struct value_string gsm0808_bssap_names[] = {
1530 { BSSAP_MSG_BSS_MANAGEMENT, "MANAGEMENT" },
1531 { BSSAP_MSG_DTAP, "DTAP" },
Neels Hofmeyr90fdb082017-03-01 14:59:44 +01001532 { 0, NULL }
Harald Welte9b837e62011-07-11 17:43:19 +02001533};
1534
Neels Hofmeyr87e45502017-06-20 00:17:59 +02001535/*! Return string name of BSSAP Message Type */
Harald Welte9b837e62011-07-11 17:43:19 +02001536const char *gsm0808_bssap_name(uint8_t msg_type)
1537{
1538 return get_value_string(gsm0808_bssap_names, msg_type);
1539}
Harald Welte96e2a002017-06-12 21:44:18 +02001540
Neels Hofmeyrffad5742018-01-12 05:34:03 +01001541const struct value_string gsm0808_speech_codec_type_names[] = {
1542 { GSM0808_SCT_FR1, "FR1" },
1543 { GSM0808_SCT_FR2, "FR2" },
1544 { GSM0808_SCT_FR3, "FR3" },
1545 { GSM0808_SCT_FR4, "FR4" },
1546 { GSM0808_SCT_FR5, "FR5" },
1547 { GSM0808_SCT_HR1, "HR1" },
1548 { GSM0808_SCT_HR3, "HR3" },
1549 { GSM0808_SCT_HR4, "HR4" },
1550 { GSM0808_SCT_HR6, "HR6" },
1551 { GSM0808_SCT_CSD, "CSD" },
1552 { 0, NULL }
1553};
1554
Philipp Maiercdd05812018-07-12 18:21:07 +02001555const struct value_string gsm0808_permitted_speech_names[] = {
1556 { GSM0808_PERM_FR1, "FR1" },
1557 { GSM0808_PERM_FR2, "FR2" },
1558 { GSM0808_PERM_FR3, "FR3" },
1559 { GSM0808_PERM_FR4, "FR4" },
1560 { GSM0808_PERM_FR5, "FR5" },
1561 { GSM0808_PERM_HR1, "HR1" },
1562 { GSM0808_PERM_HR2, "HR2" },
1563 { GSM0808_PERM_HR3, "HR3" },
1564 { GSM0808_PERM_HR4, "HR4" },
1565 { GSM0808_PERM_HR6, "HR6" },
1566 { 0, NULL }
1567};
1568
Pau Espin Pedrolf2cda622018-07-06 17:16:41 +02001569const struct value_string gsm0808_chosen_enc_alg_names[] = {
1570 { GSM0808_ALG_ID_A5_0, "A5/0" },
1571 { GSM0808_ALG_ID_A5_1, "A5/1" },
1572 { GSM0808_ALG_ID_A5_2, "A5/2" },
1573 { GSM0808_ALG_ID_A5_3, "A5/3" },
1574 { GSM0808_ALG_ID_A5_4, "A5/4" },
1575 { GSM0808_ALG_ID_A5_5, "A5/5" },
1576 { GSM0808_ALG_ID_A5_6, "A5/6" },
1577 { GSM0808_ALG_ID_A5_7, "A5/7" },
1578 { 0, NULL }
1579};
1580
Philipp Maierdbb76592018-03-29 12:55:26 +02001581static const struct value_string gsm0808_cause_names[] = {
1582 { GSM0808_CAUSE_RADIO_INTERFACE_MESSAGE_FAILURE, "RADIO INTERFACE MESSAGE FAILURE" },
1583 { GSM0808_CAUSE_RADIO_INTERFACE_FAILURE, "RADIO INTERFACE FAILURE" },
1584 { GSM0808_CAUSE_UPLINK_QUALITY, "UPLINK QUALITY" },
1585 { GSM0808_CAUSE_UPLINK_STRENGTH, "UPLINK STRENGTH" },
1586 { GSM0808_CAUSE_DOWNLINK_QUALITY, "DOWNLINK QUALITY" },
1587 { GSM0808_CAUSE_DOWNLINK_STRENGTH, "DOWNLINK STRENGTH" },
1588 { GSM0808_CAUSE_DISTANCE, "DISTANCE" },
1589 { GSM0808_CAUSE_O_AND_M_INTERVENTION, "O AND M INTERVENTION" },
1590 { GSM0808_CAUSE_RESPONSE_TO_MSC_INVOCATION, "RESPONSE TO MSC INVOCATION" },
1591 { GSM0808_CAUSE_CALL_CONTROL, "CALL CONTROL" },
1592 { GSM0808_CAUSE_RADIO_INTERFACE_FAILURE_REVERSION, "RADIO INTERFACE FAILURE REVERSION" },
1593 { GSM0808_CAUSE_HANDOVER_SUCCESSFUL, "HANDOVER SUCCESSFUL" },
1594 { GSM0808_CAUSE_BETTER_CELL, "BETTER CELL" },
1595 { GSM0808_CAUSE_DIRECTED_RETRY, "DIRECTED RETRY" },
1596 { GSM0808_CAUSE_JOINED_GROUP_CALL_CHANNEL, "JOINED GROUP CALL CHANNEL" },
1597 { GSM0808_CAUSE_TRAFFIC, "TRAFFIC" },
1598 { GSM0808_CAUSE_REDUCE_LOAD_IN_SERVING_CELL, "REDUCE LOAD IN SERVING CELL" },
1599 { GSM0808_CAUSE_TRAFFIC_LOAD_IN_TGT_HIGHER_THAN_IN_SRC_CELL, "TRAFFIC LOAD IN TGT HIGHER THAN IN SRC CELL" },
1600 { GSM0808_CAUSE_RELOCATION_TRIGGERED, "RELOCATION TRIGGERED" },
Thorsten Alteholz0062a5f2018-05-15 15:28:55 +02001601 { GSM0808_CAUSE_REQUESTED_OPT_NOT_AUTHORISED, "REQUESTED OPT NOT AUTHORISED" },
Philipp Maierdbb76592018-03-29 12:55:26 +02001602 { GSM0808_CAUSE_ALT_CHAN_CONFIG_REQUESTED, "ALT CHAN CONFIG REQUESTED" },
1603 { GSM0808_CAUSE_RESP_TO_INT_HO_ENQ_MSG, "RESP TO INT HO ENQ MSG" },
1604 { GSM0808_CAUSE_INT_HO_ENQUIRY_REJECT, "INT HO ENQUIRY REJECT" },
1605 { GSM0808_CAUSE_REDUNDANCY_LEVEL_NOT_ADEQUATE, "REDUNDANCY LEVEL NOT ADEQUATE" },
1606 { GSM0808_CAUSE_EQUIPMENT_FAILURE, "EQUIPMENT FAILURE" },
1607 { GSM0808_CAUSE_NO_RADIO_RESOURCE_AVAILABLE, "NO RADIO RESOURCE AVAILABLE" },
1608 { GSM0808_CAUSE_RQSTED_TERRESTRIAL_RESOURCE_UNAVAILABLE, "RQSTED TERRESTRIAL RESOURCE UNAVAILABLE" },
1609 { GSM0808_CAUSE_CCCH_OVERLOAD, "CCCH OVERLOAD" },
1610 { GSM0808_CAUSE_PROCESSOR_OVERLOAD, "PROCESSOR OVERLOAD" },
1611 { GSM0808_CAUSE_BSS_NOT_EQUIPPED, "BSS NOT EQUIPPED" },
1612 { GSM0808_CAUSE_MS_NOT_EQUIPPED, "MS NOT EQUIPPED" },
1613 { GSM0808_CAUSE_INVALID_CELL, "INVALID CELL" },
1614 { GSM0808_CAUSE_TRAFFIC_LOAD, "TRAFFIC LOAD" },
1615 { GSM0808_CAUSE_PREEMPTION, "PREEMPTION" },
1616 { GSM0808_CAUSE_DTM_HO_SGSN_FAILURE, "DTM HO SGSN FAILURE" },
1617 { GSM0808_CAUSE_DTM_HO_PS_ALLOC_FAILURE, "DTM HO PS ALLOC FAILURE" },
1618 { GSM0808_CAUSE_RQSTED_TRANSCODING_RATE_ADAPTION_UNAVAILABLE, "RQSTED TRANSCODING RATE ADAPTION UNAVAILABLE" },
1619 { GSM0808_CAUSE_CIRCUIT_POOL_MISMATCH, "CIRCUIT POOL MISMATCH" },
1620 { GSM0808_CAUSE_SWITCH_CIRCUIT_POOL, "SWITCH CIRCUIT POOL" },
1621 { GSM0808_CAUSE_RQSTED_SPEECH_VERSION_UNAVAILABLE, "RQSTED SPEECH VERSION UNAVAILABLE" },
1622 { GSM0808_CAUSE_LSA_NOT_ALLOWED, "LSA NOT ALLOWED" },
1623 { GSM0808_CAUSE_REQ_CODEC_TYPE_OR_CONFIG_UNAVAIL, "REQ CODEC TYPE OR CONFIG UNAVAIL" },
1624 { GSM0808_CAUSE_REQ_A_IF_TYPE_UNAVAIL, "REQ A IF TYPE UNAVAIL" },
1625 { GSM0808_CAUSE_INVALID_CSG_CELL, "INVALID CSG CELL" },
1626 { GSM0808_CAUSE_REQ_REDUND_LEVEL_NOT_AVAIL, "REQ REDUND LEVEL NOT AVAIL" },
1627 { GSM0808_CAUSE_CIPHERING_ALGORITHM_NOT_SUPPORTED, "CIPHERING ALGORITHM NOT SUPPORTED" },
1628 { GSM0808_CAUSE_GERAN_IU_MODE_FAILURE, "GERAN IU MODE FAILURE" },
1629 { GSM0808_CAUSE_INC_RELOC_NOT_SUPP_DT_PUESBINE_FEATURE, "INC RELOC NOT SUPP DT PUESBINE FEATURE" },
1630 { GSM0808_CAUSE_ACCESS_RESTRICTED_DUE_TO_SHARED_NETWORKS, "ACCESS RESTRICTED DUE TO SHARED NETWORKS" },
1631 { GSM0808_CAUSE_REQ_CODEC_TYPE_OR_CONFIG_NOT_SUPP, "REQ CODEC TYPE OR CONFIG NOT SUPP" },
1632 { GSM0808_CAUSE_REQ_A_IF_TYPE_NOT_SUPP, "REQ A IF TYPE NOT SUPP" },
1633 { GSM0808_CAUSE_REQ_REDUND_LVL_NOT_SUPP, "REQ REDUND LVL NOT SUPP" },
1634 { GSM0808_CAUSE_TERRESTRIAL_CIRCUIT_ALREADY_ALLOCATED, "TERRESTRIAL CIRCUIT ALREADY ALLOCATED" },
1635 { GSM0808_CAUSE_INVALID_MESSAGE_CONTENTS, "INVALID MESSAGE CONTENTS" },
1636 { GSM0808_CAUSE_INFORMATION_ELEMENT_OR_FIELD_MISSING, "INFORMATION ELEMENT OR FIELD MISSING" },
1637 { GSM0808_CAUSE_INCORRECT_VALUE, "INCORRECT VALUE" },
1638 { GSM0808_CAUSE_UNKNOWN_MESSAGE_TYPE, "UNKNOWN MESSAGE TYPE" },
1639 { GSM0808_CAUSE_UNKNOWN_INFORMATION_ELEMENT, "UNKNOWN INFORMATION ELEMENT" },
1640 { GSM0808_CAUSE_DTM_HO_INVALID_PS_IND, "DTM HO INVALID PS IND" },
1641 { GSM0808_CAUSE_CALL_ID_ALREADY_ALLOC, "CALL ID ALREADY ALLOC" },
1642 { GSM0808_CAUSE_PROTOCOL_ERROR_BETWEEN_BSS_AND_MSC, "PROTOCOL ERROR BETWEEN BSS AND MSC" },
Thorsten Alteholz0062a5f2018-05-15 15:28:55 +02001643 { GSM0808_CAUSE_VGCS_VBS_CALL_NON_EXISTENT, "VGCS VBS CALL NON EXISTENT" },
Philipp Maierdbb76592018-03-29 12:55:26 +02001644 { GSM0808_CAUSE_DTM_HO_TIMER_EXPIRY, "DTM HO TIMER EXPIRY" },
1645 { 0, NULL }
1646};
1647
Maxaa934632018-11-07 13:16:54 +01001648static const struct value_string gsm0808_cause_class_names[] = {
1649 { GSM0808_CAUSE_CLASS_NORM0, "Normal event" },
1650 { GSM0808_CAUSE_CLASS_NORM1, "Normal event" },
1651 { GSM0808_CAUSE_CLASS_RES_UNAVAIL, "Resource unavailable" },
1652 { GSM0808_CAUSE_CLASS_SRV_OPT_NA, "Service or option not available" },
1653 { GSM0808_CAUSE_CLASS_SRV_OPT_NIMPL, "Service or option not implemented" },
1654 { GSM0808_CAUSE_CLASS_INVAL, "Invalid message" },
1655 { GSM0808_CAUSE_CLASS_PERR, "Protocol error" },
1656 { GSM0808_CAUSE_CLASS_INTW, "Interworking" },
1657 { 0, NULL }
1658};
1659
1660/*! Return string name of BSSMAP Cause Class name */
1661const char *gsm0808_cause_class_name(enum gsm0808_cause_class class)
1662{
1663 return get_value_string(gsm0808_cause_class_names, class);
1664}
1665
Philipp Maierdbb76592018-03-29 12:55:26 +02001666/*! Return string name of BSSMAP Cause name */
Maxaa934632018-11-07 13:16:54 +01001667const char *gsm0808_cause_name(enum gsm0808_cause cause)
Philipp Maierdbb76592018-03-29 12:55:26 +02001668{
1669 return get_value_string(gsm0808_cause_names, cause);
1670}
1671
Alexander Chemerisfdfe25b2020-05-12 23:21:56 +03001672enum gsm0808_cause gsm0808_get_cause(const struct tlv_parsed *tp)
1673{
1674 const uint8_t *buf = TLVP_VAL_MINLEN(tp, GSM0808_IE_CAUSE, 1);
1675
1676 if (!buf)
1677 return -EBADMSG;
1678
1679 if (TLVP_LEN(tp, GSM0808_IE_CAUSE) > 1) {
1680 if (!gsm0808_cause_ext(buf[0]))
1681 return -EINVAL;
1682 return buf[1];
1683 }
1684
1685 return buf[0];
1686}
1687
Alexander Chemeris22630e62020-05-13 00:44:04 +03001688const char *gsm0808_diagnostics_octet_location_str(uint8_t pointer)
1689{
1690 switch (pointer) {
1691 case 0:
1692 return "Error location not determined";
1693 case 1:
1694 return "The first octet of the message received (i.e. the message type) was found erroneous (unknown)";
1695 case 0xfd:
1696 return "The first octet of the BSSAP header (Discrimination) was found erroneous";
1697 case 0xfe:
1698 return "(DTAP only) The DLCI (second) octet of the BSSAP header was found erroneous";
1699 case 0xff:
1700 return "The last octet of the BSSAP header (length indicator) was found erroneous";
1701 default:
1702 snprintf(str_buff, sizeof(str_buff), "The %d octet of the message received was found erroneous", pointer);
1703 return str_buff;
1704 }
1705}
1706
1707const char *gsm0808_diagnostics_bit_location_str(uint8_t bit_pointer)
1708{
1709 if (bit_pointer == 0) {
1710 return "No particular part of the octet is indicated";
1711 } else if (bit_pointer > 8) {
1712 return "Reserved value";
1713 }
1714
1715 snprintf(str_buff, sizeof(str_buff),
1716 "An error was provoked by the field whose most significant bit is in bit position %d",
1717 bit_pointer);
1718 return str_buff;
1719}
1720
Harald Welteebd362d2018-06-02 14:11:19 +02001721const struct value_string gsm0808_lcls_config_names[] = {
1722 { GSM0808_LCLS_CFG_BOTH_WAY, "Connect both-way" },
1723 { GSM0808_LCLS_CFG_BOTH_WAY_AND_BICAST_UL,
1724 "Connect both-way, bi-cast UL to CN" },
1725 { GSM0808_LCLS_CFG_BOTH_WAY_AND_SEND_DL,
1726 "Connect both-way, send access DL from CN" },
1727 { GSM0808_LCLS_CFG_BOTH_WAY_AND_SEND_DL_BLOCK_LOCAL_DL,
1728 "Connect both-way, send access DL from CN, block local DL" },
1729 { GSM0808_LCLS_CFG_BOTH_WAY_AND_BICAST_UL_SEND_DL,
1730 "Connect both-way, bi-cast UL to CN, send access DL from CN" },
1731 { GSM0808_LCLS_CFG_BOTH_WAY_AND_BICAST_UL_SEND_DL_BLOCK_LOCAL_DL,
1732 "Connect both-way, bi-cast UL to CN, send access DL from CN, block local DL" },
Max961db7c2018-11-08 11:40:23 +01001733 { GSM0808_LCLS_CFG_NA, "Not available" },
Harald Welteebd362d2018-06-02 14:11:19 +02001734 { 0, NULL }
1735};
1736
1737const struct value_string gsm0808_lcls_control_names[] = {
1738 { GSM0808_LCLS_CSC_CONNECT, "Connect" },
1739 { GSM0808_LCLS_CSC_DO_NOT_CONNECT, "Do not connect" },
1740 { GSM0808_LCLS_CSC_RELEASE_LCLS, "Release LCLS" },
1741 { GSM0808_LCLS_CSC_BICAST_UL_AT_HANDOVER, "Bi-cast UL at Handover" },
1742 { GSM0808_LCLS_CSC_BICAST_UL_AND_RECV_DL_AT_HANDOVER, "Bi-cast UL and receive DL at Handover" },
Max961db7c2018-11-08 11:40:23 +01001743 { GSM0808_LCLS_CSC_NA, "Not available" },
Harald Welteebd362d2018-06-02 14:11:19 +02001744 { 0, NULL }
1745};
1746
1747const struct value_string gsm0808_lcls_status_names[] = {
1748 { GSM0808_LCLS_STS_NOT_YET_LS, "Call not yet locally switched" },
1749 { GSM0808_LCLS_STS_NOT_POSSIBLE_LS, "Call not possible to be locally switched" },
1750 { GSM0808_LCLS_STS_NO_LONGER_LS, "Call is no longer locally switched" },
1751 { GSM0808_LCLS_STS_REQ_LCLS_NOT_SUPP, "Requested LCLS configuration is not supported" },
1752 { GSM0808_LCLS_STS_LOCALLY_SWITCHED, "Call is locally switched with requested LCLS config" },
Max961db7c2018-11-08 11:40:23 +01001753 { GSM0808_LCLS_STS_NA, "Not available" },
Harald Welteebd362d2018-06-02 14:11:19 +02001754 { 0, NULL }
1755};
1756
Harald Welte96e2a002017-06-12 21:44:18 +02001757/*! @} */