blob: 23468c3efbd1ebca293a7d357530370e79f35834 [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>
Pablo Neira Ayuso83419342011-03-22 16:36:13 +010027#include <osmocom/gsm/gsm0808.h>
Philipp Maierfa896ab2017-03-27 16:55:32 +020028#include <osmocom/gsm/gsm0808_utils.h>
Pablo Neira Ayuso83419342011-03-22 16:36:13 +010029#include <osmocom/gsm/protocol/gsm_08_08.h>
30#include <osmocom/gsm/gsm48.h>
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020031
Harald Welte96e2a002017-06-12 21:44:18 +020032/*! \addtogroup gsm0808
33 * @{
Harald Welte37b61652017-10-16 18:46:03 +020034 * \file gsm0808.c
35 * Helper functions regarding the TS 08.08 / 48.008 A interface, primarily
36 * message generation/encoding.
Harald Welte96e2a002017-06-12 21:44:18 +020037 */
38
Alexander Chemeris22630e62020-05-13 00:44:04 +030039/*! Char buffer to return strings from functions */
40static __thread char str_buff[512];
41
Neels Hofmeyrc4fce142018-02-20 13:47:08 +010042/*! Create "Complete L3 Info" for AoIP, legacy implementation.
43 * Instead use gsm0808_create_layer3_aoip2(), which is capable of three-digit MNC with leading zeros.
Harald Welte96e2a002017-06-12 21:44:18 +020044 * \param[in] msg_l3 msgb containing Layer 3 Message
45 * \param[in] nc Mobile Network Code
46 * \param[in] cc Mobile Country Code
47 * \param[in] lac Location Area Code
48 * \param[in] _ci Cell Identity
49 * \param[in] scl Speech Codec List
50 * \returns callee-allocated msgb with Complete L3 Info message */
Philipp Maierfa896ab2017-03-27 16:55:32 +020051struct msgb *gsm0808_create_layer3_aoip(const struct msgb *msg_l3, uint16_t nc,
52 uint16_t cc, int lac, uint16_t _ci,
53 const struct gsm0808_speech_codec_list
54 *scl)
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020055{
Neels Hofmeyrc4fce142018-02-20 13:47:08 +010056 struct osmo_cell_global_id cgi = {
57 .lai = {
58 .plmn = {
59 .mcc = cc,
60 .mnc = nc,
61 },
62 .lac = lac,
63 },
64 .cell_identity = _ci,
65 };
66 return gsm0808_create_layer3_2(msg_l3, &cgi, scl);
67}
68
69/*! Create "Complete L3 Info" for AoIP.
70 * \param[in] msg_l3 msgb containing Layer 3 Message -- not modified by this call.
71 * \param[in] cell MCC, MNC, LAC, CI to identify the cell.
72 * \param[in] scl Speech Codec List, optional.
73 * \returns newly allocated msgb with Complete L3 Info message */
74struct msgb *gsm0808_create_layer3_2(const struct msgb *msg_l3, const struct osmo_cell_global_id *cell,
75 const struct gsm0808_speech_codec_list *scl)
76{
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020077 struct msgb* msg;
Harald Welte65c2d362012-01-21 14:26:01 +010078 struct {
79 uint8_t ident;
80 struct gsm48_loc_area_id lai;
81 uint16_t ci;
82 } __attribute__ ((packed)) lai_ci;
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020083
84 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
85 "bssmap cmpl l3");
86 if (!msg)
87 return NULL;
88
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020089 /* create layer 3 header */
Harald Welte65c2d362012-01-21 14:26:01 +010090 msgb_v_put(msg, BSS_MAP_MSG_COMPLETE_LAYER_3);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020091
92 /* create the cell header */
Harald Welte65c2d362012-01-21 14:26:01 +010093 lai_ci.ident = CELL_IDENT_WHOLE_GLOBAL;
Neels Hofmeyrc4fce142018-02-20 13:47:08 +010094 gsm48_generate_lai2(&lai_ci.lai, &cell->lai);
95 lai_ci.ci = osmo_htons(cell->cell_identity);
Harald Welte65c2d362012-01-21 14:26:01 +010096 msgb_tlv_put(msg, GSM0808_IE_CELL_IDENTIFIER, sizeof(lai_ci),
97 (uint8_t *) &lai_ci);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +020098
99 /* copy the layer3 data */
Harald Welte65c2d362012-01-21 14:26:01 +0100100 msgb_tlv_put(msg, GSM0808_IE_LAYER_3_INFORMATION,
101 msgb_l3len(msg_l3), msg_l3->l3h);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200102
Philipp Maierfa896ab2017-03-27 16:55:32 +0200103 /* AoIP: add Codec List (BSS Supported) 3.2.2.103 */
104 if (scl)
105 gsm0808_enc_speech_codec_list(msg, scl);
106
Harald Welte65c2d362012-01-21 14:26:01 +0100107 /* push the bssmap header */
108 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200109
110 return msg;
111}
112
Neels Hofmeyrc4fce142018-02-20 13:47:08 +0100113/*! Create "Complete L3 Info" for A, legacy implementation.
114 * Instead use gsm0808_create_layer3_2() with the scl parameter passed as NULL,
115 * which is capable of three-digit MNC with leading zeros.
Harald Welte96e2a002017-06-12 21:44:18 +0200116 * \param[in] msg_l3 msgb containing Layer 3 Message
117 * \param[in] nc Mobile Network Code
118 * \param[in] cc Mobile Country Code
119 * \param[in] lac Location Area Code
120 * \param[in] _ci Cell Identity
121 * \returns callee-allocated msgb with Complete L3 Info message */
Philipp Maierfa896ab2017-03-27 16:55:32 +0200122struct msgb *gsm0808_create_layer3(struct msgb *msg_l3, uint16_t nc,
123 uint16_t cc, int lac, uint16_t _ci)
124{
Neels Hofmeyr4eeb8082018-03-23 01:47:14 +0100125 struct osmo_cell_global_id cgi = {
126 .lai = {
127 .plmn = {
128 .mcc = cc,
129 .mnc = nc,
130 },
131 .lac = lac,
132 },
133 .cell_identity = _ci,
134 };
135 return gsm0808_create_layer3_2(msg_l3, &cgi, NULL);
Philipp Maierfa896ab2017-03-27 16:55:32 +0200136}
137
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200138/*! Create BSSMAP RESET message
Harald Welte96e2a002017-06-12 21:44:18 +0200139 * \returns callee-allocated msgb with BSSMAP Reset message */
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200140struct msgb *gsm0808_create_reset(void)
141{
Harald Welte65c2d362012-01-21 14:26:01 +0100142 uint8_t cause = GSM0808_CAUSE_EQUIPMENT_FAILURE;
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200143 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
144 "bssmap: reset");
145 if (!msg)
146 return NULL;
147
Harald Welte65c2d362012-01-21 14:26:01 +0100148 msgb_v_put(msg, BSS_MAP_MSG_RESET);
Philipp Maier4f4905f2018-11-30 13:36:12 +0100149 gsm0808_enc_cause(msg, cause);
Harald Welte65c2d362012-01-21 14:26:01 +0100150 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
151
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200152 return msg;
153}
154
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200155/*! Create BSSMAP RESET ACK message
Harald Welte96e2a002017-06-12 21:44:18 +0200156 * \returns callee-allocated msgb with BSSMAP Reset ACK message */
Harald Weltea62fe312013-06-19 15:14:37 +0200157struct msgb *gsm0808_create_reset_ack(void)
158{
159 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
160 "bssmap: reset ack");
161 if (!msg)
162 return NULL;
163
164 msgb_v_put(msg, BSS_MAP_MSG_RESET_ACKNOWLEDGE);
165 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
166
167 return msg;
168}
169
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200170/*! Create BSSMAP Clear Complete message
Harald Welte96e2a002017-06-12 21:44:18 +0200171 * \returns callee-allocated msgb with BSSMAP Clear Complete message */
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200172struct msgb *gsm0808_create_clear_complete(void)
173{
174 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
175 "bssmap: clear complete");
Harald Welte65c2d362012-01-21 14:26:01 +0100176 uint8_t val = BSS_MAP_MSG_CLEAR_COMPLETE;
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200177 if (!msg)
178 return NULL;
179
Harald Welte65c2d362012-01-21 14:26:01 +0100180 msg->l3h = msg->data;
181 msgb_tlv_put(msg, BSSAP_MSG_BSS_MANAGEMENT, 1, &val);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200182
183 return msg;
184}
185
Harald Welted1365e12019-02-18 13:44:30 +0100186/*! Create BSSMAP Clear Command message with BSSAP header *before* l3h and BSSMAP in l3h.
187 * This is quite different from most (all?) other gsm0808_create_* which have l3h
188 * point to the BSSAP header. However, we have to keep this for backwards compatibility.
189 * Use gsm0808_create_clear_command2() for a 'modern' implementation.
Philipp Maier1a146c82018-10-30 09:36:49 +0100190 * \param[in] cause TS 08.08 cause value
Harald Welte96e2a002017-06-12 21:44:18 +0200191 * \returns callee-allocated msgb with BSSMAP Clear Command message */
Philipp Maier1a146c82018-10-30 09:36:49 +0100192struct msgb *gsm0808_create_clear_command(uint8_t cause)
Holger Hans Peter Freythera3f05d82010-10-27 11:49:24 +0200193{
194 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
195 "bssmap: clear command");
196 if (!msg)
197 return NULL;
198
Holger Hans Peter Freytheraeebe392010-10-27 12:36:05 +0200199 msg->l3h = msgb_tv_put(msg, BSSAP_MSG_BSS_MANAGEMENT, 4);
200 msgb_v_put(msg, BSS_MAP_MSG_CLEAR_CMD);
Philipp Maier4f4905f2018-11-30 13:36:12 +0100201 gsm0808_enc_cause(msg, cause);
Harald Welte65c2d362012-01-21 14:26:01 +0100202
Holger Hans Peter Freythera3f05d82010-10-27 11:49:24 +0200203 return msg;
204}
205
Philipp Maier74c4c4e2019-02-04 16:42:28 +0100206/*! Create BSSMAP Clear Command message.
207 * \param[in] cause TS 08.08 cause value.
208 * \param[in] csfb_ind indicate that the call was established in an CSFB context.
209 * \returns callee-allocated msgb with BSSMAP Clear Command message. */
210struct msgb *gsm0808_create_clear_command2(uint8_t cause, bool csfb_ind)
211{
Harald Welte10ba47d2019-02-18 12:36:54 +0100212 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
213 "bssmap: clear command");
Philipp Maier74c4c4e2019-02-04 16:42:28 +0100214 if (!msg)
215 return NULL;
216
Harald Welte10ba47d2019-02-18 12:36:54 +0100217 msgb_v_put(msg, BSS_MAP_MSG_CLEAR_CMD);
218 gsm0808_enc_cause(msg, cause);
219
Philipp Maier74c4c4e2019-02-04 16:42:28 +0100220 if (csfb_ind)
221 msgb_v_put(msg, GSM0808_IE_CSFB_INDICATION);
222
Harald Welte10ba47d2019-02-18 12:36:54 +0100223 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
224
Philipp Maier74c4c4e2019-02-04 16:42:28 +0100225 return msg;
226}
227
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200228/*! Create BSSMAP Cipher Mode Command message
Harald Welte96e2a002017-06-12 21:44:18 +0200229 * \param[in] ei Mandatory Encryption Information
230 * \param[in] cipher_response_mode optional 1-byte Cipher Response Mode
231 * \returns callee-allocated msgb with BSSMAP Cipher Mode Command message */
Philipp Maierb478dd32017-03-29 15:50:05 +0200232struct msgb *gsm0808_create_cipher(const struct gsm0808_encrypt_info *ei,
233 const uint8_t *cipher_response_mode)
234{
235 /* See also: 3GPP TS 48.008 3.2.1.30 CIPHER MODE COMMAND */
236 struct msgb *msg;
237
238 /* Mandatory emelent! */
239 OSMO_ASSERT(ei);
240
241 msg =
242 msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
243 "cipher-mode-command");
244 if (!msg)
245 return NULL;
246
247 /* Message Type 3.2.2.1 */
248 msgb_v_put(msg, BSS_MAP_MSG_CIPHER_MODE_CMD);
249
250 /* Encryption Information 3.2.2.10 */
251 gsm0808_enc_encrypt_info(msg, ei);
252
253 /* Cipher Response Mode 3.2.2.34 */
254 if (cipher_response_mode)
255 msgb_tv_put(msg, GSM0808_IE_CIPHER_RESPONSE_MODE,
256 *cipher_response_mode);
257
258 /* pre-pend the header */
259 msg->l3h =
260 msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
261
262 return msg;
263}
264
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200265/*! Create BSSMAP Cipher Mode Complete message
Harald Welte96e2a002017-06-12 21:44:18 +0200266 * \param[in] layer3 L3 Message to be included
267 * \param[in] alg_id Chosen Encrpytion Algorithm
268 * \returns callee-allocated msgb with BSSMAP Cipher Mode Complete message */
Holger Hans Peter Freyther81716d52010-04-17 06:16:35 +0200269struct msgb *gsm0808_create_cipher_complete(struct msgb *layer3, uint8_t alg_id)
270{
271 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
272 "cipher-complete");
273 if (!msg)
274 return NULL;
275
276 /* send response with BSS override for A5/1... cheating */
Harald Welte65c2d362012-01-21 14:26:01 +0100277 msgb_v_put(msg, BSS_MAP_MSG_CIPHER_MODE_COMPLETE);
Holger Hans Peter Freyther81716d52010-04-17 06:16:35 +0200278
279 /* include layer3 in case we have at least two octets */
280 if (layer3 && msgb_l3len(layer3) > 2) {
Harald Welte65c2d362012-01-21 14:26:01 +0100281 msg->l4h = msgb_tlv_put(msg, GSM0808_IE_LAYER_3_MESSAGE_CONTENTS,
282 msgb_l3len(layer3), layer3->l3h);
Holger Hans Peter Freyther81716d52010-04-17 06:16:35 +0200283 }
284
285 /* and the optional BSS message */
Harald Welte65c2d362012-01-21 14:26:01 +0100286 msgb_tv_put(msg, GSM0808_IE_CHOSEN_ENCR_ALG, alg_id);
Holger Hans Peter Freyther81716d52010-04-17 06:16:35 +0200287
Harald Welte65c2d362012-01-21 14:26:01 +0100288 /* pre-pend the header */
289 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
290
Holger Hans Peter Freyther81716d52010-04-17 06:16:35 +0200291 return msg;
292}
293
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200294/*! Create BSSMAP Cipher Mode Reject message
Maxaa934632018-11-07 13:16:54 +0100295 * \param[in] cause 3GPP TS 08.08 §3.2.2.5 cause value
Harald Welte96e2a002017-06-12 21:44:18 +0200296 * \returns callee-allocated msgb with BSSMAP Cipher Mode Reject message */
Maxaa934632018-11-07 13:16:54 +0100297struct msgb *gsm0808_create_cipher_reject(enum gsm0808_cause cause)
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200298{
299 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
Maxaa934632018-11-07 13:16:54 +0100300 "bssmap: cipher mode reject");
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200301 if (!msg)
302 return NULL;
303
Harald Welte62e40852017-12-17 20:50:34 +0100304 msgb_v_put(msg, BSS_MAP_MSG_CIPHER_MODE_REJECT);
Maxaa934632018-11-07 13:16:54 +0100305
Philipp Maier4f4905f2018-11-30 13:36:12 +0100306 gsm0808_enc_cause(msg, cause);
Harald Welte65c2d362012-01-21 14:26:01 +0100307
308 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200309
310 return msg;
311}
312
Maxed651d22018-11-07 15:25:05 +0100313/*! Create BSSMAP Cipher Mode Reject message
314 * \param[in] class 3GPP TS 08.08 §3.2.2.5 cause's class
315 * \param[in] ext 3GPP TS 08.08 §3.2.2.5 cause value (national application extension)
316 * \returns callee-allocated msgb with BSSMAP Cipher Mode Reject message */
317struct msgb *gsm0808_create_cipher_reject_ext(enum gsm0808_cause_class class, uint8_t ext)
318{
Philipp Maier4f4905f2018-11-30 13:36:12 +0100319 uint16_t cause;
Maxed651d22018-11-07 15:25:05 +0100320 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
321 "bssmap: cipher mode reject");
322 if (!msg)
323 return NULL;
324
Philipp Maier4f4905f2018-11-30 13:36:12 +0100325 /* Set cause code class in the upper byte */
326 cause = 0x80 | (class << 4);
327 cause = cause << 8;
328
329 /* Set cause code extension in the lower byte */
330 cause |= ext;
Maxed651d22018-11-07 15:25:05 +0100331
332 msgb_v_put(msg, BSS_MAP_MSG_CIPHER_MODE_REJECT);
333
Philipp Maier4f4905f2018-11-30 13:36:12 +0100334 gsm0808_enc_cause(msg, cause);
Maxed651d22018-11-07 15:25:05 +0100335
336 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
337
338 return msg;
339}
340
Harald Welte64e807c2018-05-29 21:00:56 +0200341/*! Create BSSMAP LCLS CONNECT CONTROL message (TS 48.008 3.2.1.91).
342 * \param[in] config LCLS Configuration
343 * \param[in] control LCLS Connection Status Control
344 * \returns callee-allocated msgb with BSSMAP LCLS NOTIFICATION */
Max45f89c92018-12-19 19:35:26 +0100345struct msgb *gsm0808_create_lcls_conn_ctrl(enum gsm0808_lcls_config config,
346 enum gsm0808_lcls_control control)
Harald Welte64e807c2018-05-29 21:00:56 +0200347{
Max45f89c92018-12-19 19:35:26 +0100348 struct msgb *msg;
349
350 /* According to NOTE 1 in §3.2.1.91 at least one of the parameters is required */
351 if (config == GSM0808_LCLS_CFG_NA && control == GSM0808_LCLS_CSC_NA)
352 return NULL;
353
354 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "bssmap: LCLS CONN CTRL");
Harald Welte64e807c2018-05-29 21:00:56 +0200355 if (!msg)
356 return NULL;
357
358 msgb_v_put(msg, BSS_MAP_MSG_LCLS_CONNECT_CTRL);
Max45f89c92018-12-19 19:35:26 +0100359 if (config != GSM0808_LCLS_CFG_NA)
360 msgb_tv_put(msg, GSM0808_IE_LCLS_CONFIG, config);
361 if (control != GSM0808_LCLS_CSC_NA)
362 msgb_tv_put(msg, GSM0808_IE_LCLS_CONFIG, control);
Harald Welte64e807c2018-05-29 21:00:56 +0200363 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
364
365 return msg;
366}
367
368/*! Create BSSMAP LCLS CONNECT CONTROL ACK message (TS 48.008 3.2.1.92).
369 * \param[in] status LCLS BSS Status
370 * \returns callee-allocated msgb with BSSMAP LCLS NOTIFICATION */
371struct msgb *gsm0808_create_lcls_conn_ctrl_ack(enum gsm0808_lcls_status status)
372{
373 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
374 "bssmap: LCLS CONN CTRL ACK");
375 if (!msg)
376 return NULL;
377
378 msgb_v_put(msg, BSS_MAP_MSG_LCLS_CONNECT_CTRL_ACK);
379 msgb_tv_put(msg, GSM0808_IE_LCLS_BSS_STATUS, status);
380 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
381
382 return msg;
383}
384
385/*! Create BSSMAP LCLS NOTIFICATION message (TS 48.008 3.2.1.93).
386 * \param[in] status LCLS BSS Status
387 * \param[in] break_req Include the LCLS BREAK REQ IE (true) or not (false)
388 * \returns callee-allocated msgb with BSSMAP LCLS NOTIFICATION */
389struct msgb *gsm0808_create_lcls_notification(enum gsm0808_lcls_status status, bool break_req)
390{
391 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
392 "bssmap: LCLS NOTIFICATION");
393 if (!msg)
394 return NULL;
395
396 msgb_v_put(msg, BSS_MAP_MSG_LCLS_NOTIFICATION);
397 msgb_tv_put(msg, GSM0808_IE_LCLS_BSS_STATUS, status);
398 if (break_req)
399 msgb_v_put(msg, GSM0808_IE_LCLS_BREAK_REQ);
400 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
401
402 return msg;
403}
404
Neels Hofmeyr2c79d552018-09-13 05:36:32 +0200405/*! Create BSSMAP Classmark Request message
406 * \returns callee-allocated msgb with BSSMAP Classmark Request message */
407struct msgb *gsm0808_create_classmark_request()
408{
409 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
410 "classmark-request");
411 if (!msg)
412 return NULL;
413
414 msgb_v_put(msg, BSS_MAP_MSG_CLASSMARK_RQST);
415 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
416 return msg;
417}
Harald Welte64e807c2018-05-29 21:00:56 +0200418
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200419/*! Create BSSMAP Classmark Update message
Harald Welte96e2a002017-06-12 21:44:18 +0200420 * \param[in] cm2 Classmark 2
421 * \param[in] cm2_len length (in octets) of \a cm2
422 * \param[in] cm3 Classmark 3
423 * \param[in] cm3_len length (in octets) of \a cm3
424 * \returns callee-allocated msgb with BSSMAP Classmark Update message */
Harald Welte07b625d2012-01-23 10:02:58 +0100425struct msgb *gsm0808_create_classmark_update(const uint8_t *cm2, uint8_t cm2_len,
426 const uint8_t *cm3, uint8_t cm3_len)
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200427{
428 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
429 "classmark-update");
430 if (!msg)
431 return NULL;
432
Harald Welte65c2d362012-01-21 14:26:01 +0100433 msgb_v_put(msg, BSS_MAP_MSG_CLASSMARK_UPDATE);
Harald Welte07b625d2012-01-23 10:02:58 +0100434 msgb_tlv_put(msg, GSM0808_IE_CLASSMARK_INFORMATION_T2, cm2_len, cm2);
435 if (cm3)
436 msgb_tlv_put(msg, GSM0808_IE_CLASSMARK_INFORMATION_T3,
437 cm3_len, cm3);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200438
Harald Welte65c2d362012-01-21 14:26:01 +0100439 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
440
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200441 return msg;
442}
443
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200444/*! Create BSSMAP SAPI N Reject message
Harald Welte96e2a002017-06-12 21:44:18 +0200445 * \param[in] link_id Link Identifier
446 * \returns callee-allocated msgb with BSSMAP SAPI N Reject message */
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200447struct msgb *gsm0808_create_sapi_reject(uint8_t link_id)
448{
449 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
450 "bssmap: sapi 'n' reject");
451 if (!msg)
452 return NULL;
453
Harald Welte65c2d362012-01-21 14:26:01 +0100454 msgb_v_put(msg, BSS_MAP_MSG_SAPI_N_REJECT);
Alexander Chemerisa5b1b862020-05-12 01:03:08 +0300455 msgb_tv_put(msg, GSM0808_IE_DLCI, link_id);
456 gsm0808_enc_cause(msg, GSM0808_CAUSE_BSS_NOT_EQUIPPED);
Harald Welte65c2d362012-01-21 14:26:01 +0100457
458 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200459
460 return msg;
461}
462
Max52074322018-11-30 10:44:07 +0100463/*! Create BSSMAP Assignment Request message, 3GPP TS 48.008 §3.2.1.1.
464 * This is identical to gsm0808_create_ass(), but adds KC and LCLS IEs.
Harald Welte96e2a002017-06-12 21:44:18 +0200465 * \param[in] ct Channel Type
466 * \param[in] cic Circuit Identity Code (Classic A only)
467 * \param[in] ss Socket Address of MSC-side RTP socket (AoIP only)
468 * \param[in] scl Speech Codec List (AoIP only)
Max49c06682018-11-21 22:10:26 +0100469 * \param[in] ci Call Identifier (Optional), §3.2.2.105
Max52074322018-11-30 10:44:07 +0100470 * \param[in] kc Kc128 ciphering key (Optional, A5/4), §3.2.2.109
471 * \param[in] lcls Optional LCLS parameters
Harald Welte96e2a002017-06-12 21:44:18 +0200472 * \returns callee-allocated msgb with BSSMAP Assignment Request message */
Max52074322018-11-30 10:44:07 +0100473struct msgb *gsm0808_create_ass2(const struct gsm0808_channel_type *ct,
474 const uint16_t *cic,
475 const struct sockaddr_storage *ss,
476 const struct gsm0808_speech_codec_list *scl,
477 const uint32_t *ci,
478 const uint8_t *kc, const struct osmo_lcls *lcls)
Philipp Maierc6144a22017-03-29 17:53:43 +0200479{
480 /* See also: 3GPP TS 48.008 3.2.1.1 ASSIGNMENT REQUEST */
481 struct msgb *msg;
482 uint16_t cic_sw;
483 uint32_t ci_sw;
484
485 /* Mandatory emelent! */
486 OSMO_ASSERT(ct);
487
488 msg =
489 msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
490 "bssmap: ass req");
491 if (!msg)
492 return NULL;
493
494 /* Message Type 3.2.2.1 */
495 msgb_v_put(msg, BSS_MAP_MSG_ASSIGMENT_RQST);
496
497 /* Channel Type 3.2.2.11 */
498 gsm0808_enc_channel_type(msg, ct);
499
500 /* Circuit Identity Code 3.2.2.2 */
501 if (cic) {
Harald Welte95871da2017-05-15 12:11:36 +0200502 cic_sw = osmo_htons(*cic);
Philipp Maierc6144a22017-03-29 17:53:43 +0200503 msgb_tv_fixed_put(msg, GSM0808_IE_CIRCUIT_IDENTITY_CODE,
504 sizeof(cic_sw), (uint8_t *) & cic_sw);
505 }
506
507 /* AoIP: AoIP Transport Layer Address (MGW) 3.2.2.102 */
508 if (ss) {
509 gsm0808_enc_aoip_trasp_addr(msg, ss);
510 }
511
512 /* AoIP: Codec List (MSC Preferred) 3.2.2.103 */
513 if (scl)
514 gsm0808_enc_speech_codec_list(msg, scl);
515
516 /* AoIP: Call Identifier 3.2.2.105 */
517 if (ci) {
Harald Welte95871da2017-05-15 12:11:36 +0200518 ci_sw = osmo_htonl(*ci);
Philipp Maierc6144a22017-03-29 17:53:43 +0200519 msgb_tv_fixed_put(msg, GSM0808_IE_CALL_ID, sizeof(ci_sw),
520 (uint8_t *) & ci_sw);
521 }
522
Max52074322018-11-30 10:44:07 +0100523 if (kc)
524 msgb_tv_fixed_put(msg, GSM0808_IE_KC_128, 16, kc);
525
Max47022152018-12-19 18:51:00 +0100526 if (lcls)
527 gsm0808_enc_lcls(msg, lcls);
Max52074322018-11-30 10:44:07 +0100528
Philipp Maierc6144a22017-03-29 17:53:43 +0200529 /* push the bssmap header */
530 msg->l3h =
531 msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
532
533 return msg;
534}
535
Max52074322018-11-30 10:44:07 +0100536/*! Create BSSMAP Assignment Request message, 3GPP TS 48.008 §3.2.1.1.
537 * \param[in] ct Channel Type
538 * \param[in] cic Circuit Identity Code (Classic A only)
539 * \param[in] ss Socket Address of MSC-side RTP socket (AoIP only)
540 * \param[in] scl Speech Codec List (AoIP only)
541 * \param[in] ci Call Identifier (Optional), §3.2.2.105
542 * \returns callee-allocated msgb with BSSMAP Assignment Request message */
543struct msgb *gsm0808_create_ass(const struct gsm0808_channel_type *ct,
544 const uint16_t *cic,
545 const struct sockaddr_storage *ss,
546 const struct gsm0808_speech_codec_list *scl,
547 const uint32_t *ci)
548{
549 return gsm0808_create_ass2(ct, cic, ss, scl, ci, NULL, NULL);
550}
551
Max414c8f52019-01-08 14:44:24 +0100552/*! Create BSSMAP Assignment Completed message as per 3GPP TS 48.008 §3.2.1.2
Harald Welte96e2a002017-06-12 21:44:18 +0200553 * \param[in] rr_cause GSM 04.08 RR Cause value
554 * \param[in] chosen_channel Chosen Channel
555 * \param[in] encr_alg_id Encryption Algorithm ID
556 * \param[in] speech_mode Speech Mode
557 * \param[in] ss Socket Address of BSS-side RTP socket
558 * \param[in] sc Speech Codec (current)
559 * \param[in] scl Speech Codec List (permitted)
Max414c8f52019-01-08 14:44:24 +0100560 * \param[in] lcls_bss_status §3.2.2.119 LCLS-BSS-Status, optional
Harald Welte96e2a002017-06-12 21:44:18 +0200561 * \returns callee-allocated msgb with BSSMAP Assignment Complete message */
Max414c8f52019-01-08 14:44:24 +0100562struct msgb *gsm0808_create_ass_compl2(uint8_t rr_cause, uint8_t chosen_channel,
Philipp Maierfa896ab2017-03-27 16:55:32 +0200563 uint8_t encr_alg_id, uint8_t speech_mode,
564 const struct sockaddr_storage *ss,
565 const struct gsm0808_speech_codec *sc,
Max414c8f52019-01-08 14:44:24 +0100566 const struct gsm0808_speech_codec_list *scl,
567 enum gsm0808_lcls_status lcls_bss_status)
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200568{
Harald Welte65c2d362012-01-21 14:26:01 +0100569 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
570 "bssmap: ass compl");
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200571 if (!msg)
572 return NULL;
573
Harald Welte65c2d362012-01-21 14:26:01 +0100574 msgb_v_put(msg, BSS_MAP_MSG_ASSIGMENT_COMPLETE);
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200575
576 /* write 3.2.2.22 */
Harald Welte65c2d362012-01-21 14:26:01 +0100577 msgb_tv_put(msg, GSM0808_IE_RR_CAUSE, rr_cause);
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200578
579 /* write cirtcuit identity code 3.2.2.2 */
580 /* write cell identifier 3.2.2.17 */
581 /* write chosen channel 3.2.2.33 when BTS picked it */
Harald Welte65c2d362012-01-21 14:26:01 +0100582 msgb_tv_put(msg, GSM0808_IE_CHOSEN_CHANNEL, chosen_channel);
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200583
584 /* write chosen encryption algorithm 3.2.2.44 */
Harald Welte65c2d362012-01-21 14:26:01 +0100585 msgb_tv_put(msg, GSM0808_IE_CHOSEN_ENCR_ALG, encr_alg_id);
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200586
587 /* write circuit pool 3.2.2.45 */
588 /* write speech version chosen: 3.2.2.51 when BTS picked it */
Harald Welte65c2d362012-01-21 14:26:01 +0100589 if (speech_mode != 0)
590 msgb_tv_put(msg, GSM0808_IE_SPEECH_VERSION, speech_mode);
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200591
Philipp Maierfa896ab2017-03-27 16:55:32 +0200592 /* AoIP: AoIP Transport Layer Address (BSS) 3.2.2.102 */
593 if (ss)
594 gsm0808_enc_aoip_trasp_addr(msg, ss);
595
596 /* AoIP: Speech Codec (Chosen) 3.2.2.104 */
597 if (sc)
598 gsm0808_enc_speech_codec(msg, sc);
599
600 /* AoIP: add Codec List (BSS Supported) 3.2.2.103 */
601 if (scl)
602 gsm0808_enc_speech_codec_list(msg, scl);
603
Max414c8f52019-01-08 14:44:24 +0100604 /* FIXME: write LSA identifier 3.2.2.15 - see 3GPP TS 43.073 */
605
606 /* LCLS-BSS-Status 3.2.2.119 */
607 if (lcls_bss_status != GSM0808_LCLS_STS_NA)
608 msgb_tv_put(msg, GSM0808_IE_LCLS_BSS_STATUS, lcls_bss_status);
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200609
Harald Welte65c2d362012-01-21 14:26:01 +0100610 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200611
Holger Hans Peter Freytherba6172a2010-04-17 06:21:49 +0200612 return msg;
613}
614
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200615/*! Create BSSMAP Assignment Completed message
Harald Welte96e2a002017-06-12 21:44:18 +0200616 * \param[in] rr_cause GSM 04.08 RR Cause value
617 * \param[in] chosen_channel Chosen Channel
618 * \param[in] encr_alg_id Encryption Algorithm ID
619 * \param[in] speech_mode Speech Mode
Max414c8f52019-01-08 14:44:24 +0100620 * \param[in] ss Socket Address of BSS-side RTP socket
621 * \param[in] sc Speech Codec (current)
622 * \param[in] scl Speech Codec List (permitted)
623 * \returns callee-allocated msgb with BSSMAP Assignment Complete message */
624struct msgb *gsm0808_create_ass_compl(uint8_t rr_cause, uint8_t chosen_channel,
625 uint8_t encr_alg_id, uint8_t speech_mode,
626 const struct sockaddr_storage *ss,
627 const struct gsm0808_speech_codec *sc,
628 const struct gsm0808_speech_codec_list *scl)
629{
630 return gsm0808_create_ass_compl2(rr_cause, chosen_channel, encr_alg_id, speech_mode,
631 ss, sc, scl, GSM0808_LCLS_STS_NA);
632}
633
634/*! Create BSSMAP Assignment Completed message
635 * \param[in] rr_cause GSM 04.08 RR Cause value
636 * \param[in] chosen_channel Chosen Channel
637 * \param[in] encr_alg_id Encryption Algorithm ID
638 * \param[in] speech_mode Speech Mode
Harald Welte96e2a002017-06-12 21:44:18 +0200639 * \returns callee-allocated msgb with BSSMAP Assignment Complete message */
Philipp Maierfa896ab2017-03-27 16:55:32 +0200640struct msgb *gsm0808_create_assignment_completed(uint8_t rr_cause,
641 uint8_t chosen_channel,
642 uint8_t encr_alg_id,
643 uint8_t speech_mode)
644{
Max414c8f52019-01-08 14:44:24 +0100645 return gsm0808_create_ass_compl2(rr_cause, chosen_channel, encr_alg_id,
646 speech_mode, NULL, NULL, NULL, GSM0808_LCLS_STS_NA);
Philipp Maierfa896ab2017-03-27 16:55:32 +0200647}
648
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200649/*! Create BSSMAP Assignment Failure message
Harald Welte96e2a002017-06-12 21:44:18 +0200650 * \param[in] cause BSSMAP Cause value
651 * \param[in] rr_cause GSM 04.08 RR Cause value
652 * \param[in] scl Optional Speech Cdec List (AoIP)
653 * \returns callee-allocated msgb with BSSMAP Assignment Failure message */
Philipp Maierfa896ab2017-03-27 16:55:32 +0200654struct msgb *gsm0808_create_ass_fail(uint8_t cause, const uint8_t *rr_cause,
655 const struct gsm0808_speech_codec_list
656 *scl)
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200657{
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200658 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
659 "bssmap: ass fail");
660 if (!msg)
661 return NULL;
662
Harald Welte65c2d362012-01-21 14:26:01 +0100663 msgb_v_put(msg, BSS_MAP_MSG_ASSIGMENT_FAILURE);
Philipp Maier4f4905f2018-11-30 13:36:12 +0100664 gsm0808_enc_cause(msg, cause);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200665
666 /* RR cause 3.2.2.22 */
Harald Welte65c2d362012-01-21 14:26:01 +0100667 if (rr_cause)
668 msgb_tv_put(msg, GSM0808_IE_RR_CAUSE, *rr_cause);
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200669
670 /* Circuit pool 3.22.45 */
671 /* Circuit pool list 3.2.2.46 */
672
Philipp Maierfa896ab2017-03-27 16:55:32 +0200673 /* AoIP: add Codec List (BSS Supported) 3.2.2.103 */
674 if (scl)
675 gsm0808_enc_speech_codec_list(msg, scl);
676
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200677 /* update the size */
Harald Welte65c2d362012-01-21 14:26:01 +0100678 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
679
Holger Hans Peter Freyther280cd512010-04-15 10:10:39 +0200680 return msg;
681}
Holger Hans Peter Freyther7daa01c2010-04-17 05:14:36 +0200682
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200683/*! Create BSSMAP Assignment Failure message
Harald Welte96e2a002017-06-12 21:44:18 +0200684 * \param[in] cause BSSMAP Cause value
685 * \param[in] rr_cause GSM 04.08 RR Cause value
686 * \returns callee-allocated msgb with BSSMAP Assignment Failure message */
Philipp Maierfa896ab2017-03-27 16:55:32 +0200687struct msgb *gsm0808_create_assignment_failure(uint8_t cause,
688 uint8_t *rr_cause)
689{
690 return gsm0808_create_ass_fail(cause, rr_cause, NULL);
691}
692
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200693/*! Create BSSMAP Clear Request message
Harald Welte96e2a002017-06-12 21:44:18 +0200694 * \param[in] cause BSSMAP Cause value
695 * \returns callee-allocated msgb with BSSMAP Clear Request message */
Holger Hans Peter Freytheraf270a42010-11-04 12:42:50 +0100696struct msgb *gsm0808_create_clear_rqst(uint8_t cause)
697{
698 struct msgb *msg;
699
700 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
701 "bssmap: clear rqst");
702 if (!msg)
703 return NULL;
704
Harald Welte65c2d362012-01-21 14:26:01 +0100705 msgb_v_put(msg, BSS_MAP_MSG_CLEAR_RQST);
Philipp Maier4f4905f2018-11-30 13:36:12 +0100706 gsm0808_enc_cause(msg, cause);
Harald Welte65c2d362012-01-21 14:26:01 +0100707 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
Holger Hans Peter Freytheraf270a42010-11-04 12:42:50 +0100708
Holger Hans Peter Freytheraf270a42010-11-04 12:42:50 +0100709 return msg;
710}
711
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200712/*! Create BSSMAP PAGING message
Harald Welte96e2a002017-06-12 21:44:18 +0200713 * \param[in] imsi Mandatory paged IMSI in string representation
714 * \param[in] tmsi Optional paged TMSI
Stefan Sperling3953f412018-08-28 15:06:30 +0200715 * \param[in] cil Mandatory Cell Identity List (where to page)
Harald Welte96e2a002017-06-12 21:44:18 +0200716 * \param[in] chan_needed Channel Type needed
717 * \returns callee-allocated msgb with BSSMAP PAGING message */
Stefan Sperling11a4d9d2018-02-15 18:28:04 +0100718struct msgb *gsm0808_create_paging2(const char *imsi, const uint32_t *tmsi,
719 const struct gsm0808_cell_id_list2 *cil,
720 const uint8_t *chan_needed)
Philipp Maier3d48ec02017-03-29 17:37:55 +0200721{
722 struct msgb *msg;
723 uint8_t mid_buf[GSM48_MI_SIZE + 2];
724 int mid_len;
725 uint32_t tmsi_sw;
726
Stefan Sperling11a4d9d2018-02-15 18:28:04 +0100727 /* Mandatory elements! */
Philipp Maier3d48ec02017-03-29 17:37:55 +0200728 OSMO_ASSERT(imsi);
729 OSMO_ASSERT(cil);
730
731 /* Malformed IMSI */
732 OSMO_ASSERT(strlen(imsi) <= GSM48_MI_SIZE);
733
734 msg =
735 msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "paging");
736 if (!msg)
737 return NULL;
738
739 /* Message Type 3.2.2.1 */
740 msgb_v_put(msg, BSS_MAP_MSG_PAGING);
741
Stefan Sperling3953f412018-08-28 15:06:30 +0200742 /* mandatory IMSI 3.2.2.6 */
Philipp Maier3d48ec02017-03-29 17:37:55 +0200743 mid_len = gsm48_generate_mid_from_imsi(mid_buf, imsi);
744 msgb_tlv_put(msg, GSM0808_IE_IMSI, mid_len - 2, mid_buf + 2);
745
746 /* TMSI 3.2.2.7 */
747 if (tmsi) {
Harald Welte95871da2017-05-15 12:11:36 +0200748 tmsi_sw = osmo_htonl(*tmsi);
Philipp Maier3d48ec02017-03-29 17:37:55 +0200749 msgb_tlv_put(msg, GSM0808_IE_TMSI, sizeof(*tmsi),
750 (uint8_t *) & tmsi_sw);
751 }
752
Stefan Sperling3953f412018-08-28 15:06:30 +0200753 /* mandatory Cell Identifier List 3.2.2.27 */
754 gsm0808_enc_cell_id_list2(msg, cil);
Philipp Maier3d48ec02017-03-29 17:37:55 +0200755
756 /* Channel Needed 3.2.2.36 */
757 if (chan_needed) {
758 msgb_tv_put(msg, GSM0808_IE_CHANNEL_NEEDED,
759 (*chan_needed) & 0x03);
760 }
761
762 /* pre-pend the header */
763 msg->l3h =
764 msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
765
766 return msg;
767}
768
Stefan Sperling11a4d9d2018-02-15 18:28:04 +0100769/*! DEPRECATED: Use gsm0808_create_paging2 instead.
770 * Create BSSMAP PAGING message.
771 * \param[in] imsi Mandatory paged IMSI in string representation
772 * \param[in] tmsi Optional paged TMSI
773 * \param[in] cil Cell Identity List (where to page)
774 * \param[in] chan_needed Channel Type needed
775 * \returns callee-allocated msgb with BSSMAP PAGING message */
776struct msgb *gsm0808_create_paging(const char *imsi, const uint32_t *tmsi,
777 const struct gsm0808_cell_id_list *cil,
778 const uint8_t *chan_needed)
779{
780 struct gsm0808_cell_id_list2 cil2 = {};
781
782 /* Mandatory emelents! */
783 OSMO_ASSERT(cil);
784
785 if (cil->id_list_len > GSM0808_CELL_ID_LIST2_MAXLEN)
786 return NULL;
787
788 cil2.id_discr = cil->id_discr;
789 memcpy(cil2.id_list, cil->id_list_lac, cil->id_list_len * sizeof(cil2.id_list[0].lac));
790 cil2.id_list_len = cil->id_list_len;
791
792 return gsm0808_create_paging2(imsi, tmsi, &cil2, chan_needed);
793}
794
Neels Hofmeyr70aba3f2018-03-13 03:40:53 +0100795static uint8_t put_old_bss_to_new_bss_information(struct msgb *msg,
796 const struct gsm0808_old_bss_to_new_bss_info *i)
797{
798 uint8_t *old_tail;
799 uint8_t *tlv_len;
800
801 msgb_put_u8(msg, GSM0808_IE_OLD_BSS_TO_NEW_BSS_INFORMATION);
802 tlv_len = msgb_put(msg, 1);
803 old_tail = msg->tail;
804
805 if (i->extra_information_present) {
806 uint8_t val = 0;
807 if (i->extra_information.prec)
808 val |= 1 << 0;
809 if (i->extra_information.lcs)
810 val |= 1 << 1;
811 if (i->extra_information.ue_prob)
812 val |= 1 << 2;
813 msgb_tlv_put(msg, GSM0808_FE_IE_EXTRA_INFORMATION, 1, &val);
814 }
815
816 if (i->current_channel_type_2_present) {
817 uint8_t val[2] = {
818 i->current_channel_type_2.mode,
819 i->current_channel_type_2.field,
820 };
821 msgb_tlv_put(msg, GSM0808_FE_IE_CURRENT_CHANNEL_TYPE_2, 2, val);
822 }
823
824 *tlv_len = (uint8_t) (msg->tail - old_tail);
825 return *tlv_len + 2;
826}
827
828/*! Create BSSMAP HANDOVER REQUIRED message.
829 * \param[in] params All information to be encoded.
Neels Hofmeyr302aafc2019-04-10 19:23:45 +0200830 * \returns newly allocated msgb with BSSMAP HANDOVER REQUIRED message. */
Neels Hofmeyr70aba3f2018-03-13 03:40:53 +0100831struct msgb *gsm0808_create_handover_required(const struct gsm0808_handover_required *params)
832{
833 struct msgb *msg;
834
835 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-REQUIRED");
836 if (!msg)
837 return NULL;
838
839 /* Message Type, 3.2.2.1 */
840 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_REQUIRED);
841
842 /* Cause, 3.2.2.5 */
Philipp Maier4f4905f2018-11-30 13:36:12 +0100843 gsm0808_enc_cause(msg, params->cause);
Neels Hofmeyr70aba3f2018-03-13 03:40:53 +0100844
845 /* Cell Identifier List, 3.2.2.27 */
846 gsm0808_enc_cell_id_list2(msg, &params->cil);
847
848 /* Current Channel Type 1, 3.2.2.49 */
849 if (params->current_channel_type_1_present)
850 msgb_tv_fixed_put(msg, GSM0808_IE_CURRENT_CHANNEL_TYPE_1, 1, &params->current_channel_type_1);
851
852 /* Speech Version (Used), 3.2.2.51 */
853 if (params->speech_version_used_present)
Neels Hofmeyr302aafc2019-04-10 19:23:45 +0200854 msgb_tv_put(msg, GSM0808_IE_SPEECH_VERSION, params->speech_version_used);
Neels Hofmeyr70aba3f2018-03-13 03:40:53 +0100855
856 if (params->old_bss_to_new_bss_info_present)
857 put_old_bss_to_new_bss_information(msg, &params->old_bss_to_new_bss_info);
858
859 /* pre-pend the header */
860 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
861
862 return msg;
863}
864
Neels Hofmeyrf7e9c512019-03-06 04:25:38 +0100865/*! Create BSSMAP HANDOVER REQUIRED REJECT message.
866 * \returns newly allocated msgb with BSSMAP HANDOVER REQUIRED REJECT message. */
867struct msgb *gsm0808_create_handover_required_reject(const struct gsm0808_handover_required_reject *params)
868{
869 struct msgb *msg;
870
871 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-REQUIRED-REJECT");
872 if (!msg)
873 return NULL;
874
875 /* Message Type, 3.2.2.1 */
876 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_REQUIRED_REJECT);
877
878 /* Cause, 3.2.2.5 */
879 gsm0808_enc_cause(msg, params->cause);
880
881 /* prepend the header */
882 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
883
884 return msg;
885}
886
887/*! Create BSSMAP HANDOVER REQUEST message, 3GPP TS 48.008 3.2.1.8.
888 * Sent from the MSC to the potential new target cell during inter-BSC handover, or to the target MSC during inter-MSC
889 * handover.
890 */
891struct msgb *gsm0808_create_handover_request(const struct gsm0808_handover_request *params)
892{
893 struct msgb *msg;
894
895 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-REQUEST");
896 if (!msg)
897 return NULL;
898
899 /* Message Type, 3.2.2.1 */
900 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_RQST);
901
902 /* Channel Type 3.2.2.11 */
903 gsm0808_enc_channel_type(msg, &params->channel_type);
904
905 /* Encryption Information 3.2.2.10 */
906 gsm0808_enc_encrypt_info(msg, &params->encryption_information);
907
908 /* Classmark Information 1 3.2.2.30 or Classmark Information 2 3.2.2.19 (Classmark 2 wins) */
909 if (params->classmark_information.classmark2_len) {
910 msgb_tlv_put(msg, GSM0808_IE_CLASSMARK_INFORMATION_T2,
911 params->classmark_information.classmark2_len,
912 (const uint8_t*)&params->classmark_information.classmark2);
913 } else if (params->classmark_information.classmark1_set) {
914 msgb_tlv_put(msg, GSM0808_IE_CLASSMARK_INFORMATION_TYPE_1,
915 sizeof(params->classmark_information.classmark1),
916 (const uint8_t*)&params->classmark_information.classmark1);
917 }
918 /* (Classmark 3 possibly follows below) */
919
920 /* Cell Identifier (Serving) , 3.2.2.17 */
921 gsm0808_enc_cell_id(msg, &params->cell_identifier_serving);
922
923 /* Cell Identifier (Target) , 3.2.2.17 */
924 gsm0808_enc_cell_id(msg, &params->cell_identifier_target);
925
926 /* Cause, 3.2.2.5 */
927 gsm0808_enc_cause(msg, params->cause);
928
929 /* Classmark Information 3 3.2.2.20 */
930 if (params->classmark_information.classmark3_len) {
931 msgb_tlv_put(msg, GSM0808_IE_CLASSMARK_INFORMATION_T3,
932 params->classmark_information.classmark3_len,
933 (const uint8_t*)&params->classmark_information.classmark3);
934 }
935
936 /* Current Channel type 1 3.2.2.49 */
937 if (params->current_channel_type_1_present)
938 msgb_tv_fixed_put(msg, GSM0808_IE_CURRENT_CHANNEL_TYPE_1, 1, &params->current_channel_type_1);
939
940 /* Speech Version (Used), 3.2.2.51 */
941 if (params->speech_version_used) {
942 msgb_tv_put(msg, GSM0808_IE_SPEECH_VERSION, params->speech_version_used);
943 }
944
945 /* Chosen Encryption Algorithm (Serving) 3.2.2.44 */
946 if (params->chosen_encryption_algorithm_serving)
947 msgb_tv_put(msg, GSM0808_IE_CHOSEN_ENCR_ALG, params->chosen_encryption_algorithm_serving);
948
949 /* Old BSS to New BSS Information 3.2.2.58 */
950 if (params->old_bss_to_new_bss_info_raw && params->old_bss_to_new_bss_info_raw_len) {
951 msgb_tlv_put(msg, GSM0808_IE_OLD_BSS_TO_NEW_BSS_INFORMATION,
952 params->old_bss_to_new_bss_info_raw_len,
953 params->old_bss_to_new_bss_info_raw);
954 } else if (params->old_bss_to_new_bss_info_present) {
955 put_old_bss_to_new_bss_information(msg, &params->old_bss_to_new_bss_info);
956 }
957
958 /* IMSI 3.2.2.6 */
959 if (params->imsi) {
960 uint8_t mid_buf[GSM48_MI_SIZE + 2];
961 int mid_len = gsm48_generate_mid_from_imsi(mid_buf, params->imsi);
962 msgb_tlv_put(msg, GSM0808_IE_IMSI, mid_len - 2, mid_buf + 2);
963 }
964
965 if (params->aoip_transport_layer)
966 gsm0808_enc_aoip_trasp_addr(msg, params->aoip_transport_layer);
967
968 if (params->codec_list_msc_preferred)
969 gsm0808_enc_speech_codec_list(msg, params->codec_list_msc_preferred);
970
971 if (params->call_id_present) {
972 uint8_t val[4];
973 osmo_store32le(params->call_id, val);
974 msgb_tv_fixed_put(msg, GSM0808_IE_CALL_ID, 4, val);
975 }
976
977 if (params->global_call_reference && params->global_call_reference_len) {
978 msgb_tlv_put(msg, GSM0808_IE_GLOBAL_CALL_REF,
979 params->global_call_reference_len, params->global_call_reference);
980 }
981
982 /* prepend header with final length */
983 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
984
985 return msg;
986}
987
Neels Hofmeyr60f31532018-04-16 22:42:09 +0200988/*! Create BSSMAP HANDOVER REQUEST ACKNOWLEDGE message, 3GPP TS 48.008 3.2.1.10.
989 * Sent from the MT BSC back to the MSC when it has allocated an lchan to handover to.
990 * 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 +0100991struct msgb *gsm0808_create_handover_request_ack2(const struct gsm0808_handover_request_ack *params)
Neels Hofmeyrb662b362018-04-16 22:31:15 +0200992{
993 struct msgb *msg;
994
Neels Hofmeyr60f31532018-04-16 22:42:09 +0200995 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-REQUEST-ACK");
Neels Hofmeyrb662b362018-04-16 22:31:15 +0200996 if (!msg)
997 return NULL;
998
999 /* Message Type, 3.2.2.1 */
1000 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_RQST_ACKNOWLEDGE);
1001
Neels Hofmeyr73b943e2019-03-14 04:10:25 +01001002 /* Layer 3 Information, 3.2.2.24 -- it is actually mandatory, but rather compose a nonstandard message than
1003 * segfault or return NULL without a log message. */
1004 if (params->l3_info && params->l3_info_len)
1005 msgb_tlv_put(msg, GSM0808_IE_LAYER_3_INFORMATION, params->l3_info_len, params->l3_info);
Neels Hofmeyrb662b362018-04-16 22:31:15 +02001006
Neels Hofmeyr73b943e2019-03-14 04:10:25 +01001007 if (params->chosen_channel_present)
1008 msgb_tv_put(msg, GSM0808_IE_CHOSEN_CHANNEL, params->chosen_channel);
1009 if (params->chosen_encr_alg)
1010 msgb_tv_put(msg, GSM0808_IE_CHOSEN_ENCR_ALG, params->chosen_encr_alg);
1011
1012 if (params->chosen_speech_version != 0)
1013 msgb_tv_put(msg, GSM0808_IE_SPEECH_VERSION, params->chosen_speech_version);
1014
1015 if (params->aoip_transport_layer)
1016 gsm0808_enc_aoip_trasp_addr(msg, params->aoip_transport_layer);
1017
1018 /* AoIP: Speech Codec (Chosen) 3.2.2.104 */
1019 if (params->speech_codec_chosen_present)
1020 gsm0808_enc_speech_codec(msg, &params->speech_codec_chosen);
Neels Hofmeyrb662b362018-04-16 22:31:15 +02001021
Neels Hofmeyr43c266f2018-08-28 01:08:38 +02001022 /* prepend header with final length */
1023 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1024
Neels Hofmeyrb662b362018-04-16 22:31:15 +02001025 return msg;
1026}
1027
Neels Hofmeyr73b943e2019-03-14 04:10:25 +01001028/*! Same as gsm0808_create_handover_request_ack2() but with less parameters.
1029 * In particular, this lacks the AoIP Transport Layer address. */
1030struct msgb *gsm0808_create_handover_request_ack(const uint8_t *l3_info, uint8_t l3_info_len,
1031 uint8_t chosen_channel, uint8_t chosen_encr_alg,
1032 uint8_t chosen_speech_version)
1033{
1034 struct gsm0808_handover_request_ack params = {
1035 .l3_info = l3_info,
1036 .l3_info_len = l3_info_len,
1037 .chosen_channel = chosen_channel,
1038 .chosen_encr_alg = chosen_encr_alg,
1039 .chosen_speech_version = chosen_speech_version,
1040 };
1041
1042 return gsm0808_create_handover_request_ack2(&params);
1043}
1044
Neels Hofmeyrf7e9c512019-03-06 04:25:38 +01001045/*! Create BSSMAP HANDOVER COMMAND message, 3GPP TS 48.008 3.2.1.11.
1046 * Sent from the MSC to the old BSS to transmit the RR Handover Command received from the new BSS. */
1047struct msgb *gsm0808_create_handover_command(const struct gsm0808_handover_command *params)
1048{
1049 struct msgb *msg;
1050
1051 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-COMMAND");
1052 if (!msg)
1053 return NULL;
1054
1055 /* Message Type, 3.2.2.1 */
1056 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_CMD);
1057
1058 msgb_tlv_put(msg, GSM0808_IE_LAYER_3_INFORMATION, params->l3_info_len, params->l3_info);
1059
1060 if (params->cell_identifier.id_discr != CELL_IDENT_NO_CELL)
1061 gsm0808_enc_cell_id(msg, &params->cell_identifier);
1062
1063 if (params->new_bss_to_old_bss_info_raw
1064 && params->new_bss_to_old_bss_info_raw_len)
1065 msgb_tlv_put(msg, GSM0808_IE_NEW_BSS_TO_OLD_BSS_INFO, params->new_bss_to_old_bss_info_raw_len,
1066 params->new_bss_to_old_bss_info_raw);
1067
1068 /* prepend header with final length */
1069 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1070
1071 return msg;
1072}
1073
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001074/*! Create BSSMAP HANDOVER DETECT message, 3GPP TS 48.008 3.2.1.40.
1075 * Sent from the MT BSC back to the MSC when the MS has sent a handover RACH request and the MT BSC has
1076 * received the Handover Detect message. */
1077struct msgb *gsm0808_create_handover_detect()
1078{
1079 struct msgb *msg;
1080
1081 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-DETECT");
1082 if (!msg)
1083 return NULL;
1084
1085 /* Message Type, 3.2.2.1 */
1086 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_DETECT);
1087
Neels Hofmeyr43c266f2018-08-28 01:08:38 +02001088 /* prepend header with final length */
1089 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1090
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001091 return msg;
1092}
1093
Neels Hofmeyrf7e9c512019-03-06 04:25:38 +01001094/*! Create BSSMAP HANDOVER SUCCEEDED message, 3GPP TS 48.008 3.2.1.13.
1095 * Sent from the MSC back to the old BSS to notify that the MS has successfully accessed the new BSS. */
1096struct msgb *gsm0808_create_handover_succeeded()
1097{
1098 struct msgb *msg;
1099
1100 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-DETECT");
1101 if (!msg)
1102 return NULL;
1103
1104 /* Message Type, 3.2.2.1 */
1105 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_SUCCEEDED);
1106
1107 /* prepend header with final length */
1108 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1109
1110 return msg;
1111}
1112
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001113/*! Create BSSMAP HANDOVER COMPLETE message, 3GPP TS 48.008 3.2.1.12.
1114 * Sent from the MT BSC back to the MSC when the MS has fully settled into the new lchan. */
1115struct msgb *gsm0808_create_handover_complete(const struct gsm0808_handover_complete *params)
1116{
1117 struct msgb *msg;
1118
1119 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-COMPLETE");
1120 if (!msg)
1121 return NULL;
1122
1123 /* Message Type, 3.2.2.1 */
1124 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_COMPLETE);
1125
1126 /* RR Cause, 3.2.2.22 */
1127 if (params->rr_cause_present)
1128 msgb_tlv_put(msg, GSM0808_IE_RR_CAUSE, 1, &params->rr_cause);
1129
1130 /* AoIP: Speech Codec (Chosen) 3.2.2.104 */
1131 if (params->speech_codec_chosen_present)
1132 gsm0808_enc_speech_codec(msg, &params->speech_codec_chosen);
1133
1134 /* AoIP: add Codec List (BSS Supported) 3.2.2.103 */
1135 if (params->codec_list_bss_supported.len)
1136 gsm0808_enc_speech_codec_list(msg, &params->codec_list_bss_supported);
1137
1138 /* Chosen Encryption Algorithm 3.2.2.44 */
1139 if (params->chosen_encr_alg_present)
1140 msgb_tv_put(msg, GSM0808_IE_CHOSEN_ENCR_ALG, params->chosen_encr_alg);
1141
1142 /* LCLS-BSS-Status 3.2.2.119 */
1143 if (params->lcls_bss_status_present)
1144 msgb_tv_put(msg, GSM0808_IE_LCLS_BSS_STATUS, params->lcls_bss_status);
1145
Neels Hofmeyr43c266f2018-08-28 01:08:38 +02001146 /* prepend header with final length */
1147 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1148
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001149 return msg;
1150}
1151
1152/*! Create BSSMAP HANDOVER FAILURE message, 3GPP TS 48.008 3.2.1.16.
1153 * Sent from the MT BSC back to the MSC when the handover has failed. */
1154struct msgb *gsm0808_create_handover_failure(const struct gsm0808_handover_failure *params)
1155{
1156 struct msgb *msg;
1157
1158 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-FAILURE");
1159 if (!msg)
1160 return NULL;
1161
1162 /* Message Type, 3.2.2.1 */
1163 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_FAILURE);
1164
1165 /* Cause, 3.2.2.5 */
Philipp Maier4f4905f2018-11-30 13:36:12 +01001166 gsm0808_enc_cause(msg, params->cause);
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001167
1168 /* RR Cause, 3.2.2.22 */
1169 if (params->rr_cause_present)
1170 msgb_tlv_put(msg, GSM0808_IE_RR_CAUSE, 1, &params->rr_cause);
1171
1172 /* AoIP: add Codec List (BSS Supported) 3.2.2.103 */
1173 if (params->codec_list_bss_supported.len)
1174 gsm0808_enc_speech_codec_list(msg, &params->codec_list_bss_supported);
1175
Neels Hofmeyr43c266f2018-08-28 01:08:38 +02001176 /* prepend header with final length */
1177 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1178
Neels Hofmeyr60f31532018-04-16 22:42:09 +02001179 return msg;
1180}
1181
Philipp Maier225bdf42018-10-30 14:56:59 +01001182/*! Create BSSMAP HANDOVER PERFORMED message, 3GPP TS 48.008 3.2.1.25.
1183 * \param[in] params All information to be encoded.
1184 * \returns callee-allocated msgb with BSSMAP HANDOVER PERFORMED message */
1185struct msgb *gsm0808_create_handover_performed(const struct gsm0808_handover_performed *params)
1186{
1187 struct msgb *msg;
1188
1189 msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM, "BSSMAP-HANDOVER-PERFORMED");
1190 if (!msg)
1191 return NULL;
1192
1193 /* Message Type, 3.2.2.1 */
1194 msgb_v_put(msg, BSS_MAP_MSG_HANDOVER_PERFORMED);
1195
1196 /* Cause, 3.2.2.5 */
Philipp Maier4f4905f2018-11-30 13:36:12 +01001197 gsm0808_enc_cause(msg, params->cause);
Philipp Maier225bdf42018-10-30 14:56:59 +01001198
1199 /* Cell Identifier, 3.2.2.17 */
1200 gsm0808_enc_cell_id(msg, &params->cell_id);
1201
1202 /* Chosen Channel 3.2.2.33 */
1203 if (params->chosen_channel_present)
1204 msgb_tv_put(msg, GSM0808_IE_CHOSEN_CHANNEL, params->chosen_channel);
1205
1206 /* Chosen Encryption Algorithm 3.2.2.44 */
1207 if (params->chosen_encr_alg_present)
1208 msgb_tv_put(msg, GSM0808_IE_CHOSEN_ENCR_ALG, params->chosen_encr_alg);
1209
1210 /* Speech Version (chosen) 3.2.2.51 */
1211 if (params->speech_version_chosen_present)
1212 msgb_tv_put(msg, GSM0808_IE_SPEECH_VERSION, params->speech_version_chosen);
1213
1214 /* AoIP: Speech Codec (chosen) 3.2.2.104 */
1215 if (params->speech_codec_chosen_present)
1216 gsm0808_enc_speech_codec(msg, &params->speech_codec_chosen);
1217
1218 /* LCLS-BSS-Status 3.2.2.119 */
1219 if (params->lcls_bss_status_present)
1220 msgb_tv_put(msg, GSM0808_IE_LCLS_BSS_STATUS, params->lcls_bss_status);
1221
1222 /* prepend header with final length */
1223 msg->l3h = msgb_tv_push(msg, BSSAP_MSG_BSS_MANAGEMENT, msgb_length(msg));
1224
1225 return msg;
1226}
1227
Neels Hofmeyr87e45502017-06-20 00:17:59 +02001228/*! Prepend a DTAP header to given Message Buffer
Harald Welte96e2a002017-06-12 21:44:18 +02001229 * \param[in] msgb Message Buffer
1230 * \param[in] link_id Link Identifier */
Holger Hans Peter Freyther9a3dec02010-05-16 08:15:40 +08001231void gsm0808_prepend_dtap_header(struct msgb *msg, uint8_t link_id)
1232{
1233 uint8_t *hh = msgb_push(msg, 3);
1234 hh[0] = BSSAP_MSG_DTAP;
1235 hh[1] = link_id;
1236 hh[2] = msg->len - 3;
1237}
1238
Neels Hofmeyr87e45502017-06-20 00:17:59 +02001239/*! Create BSSMAP DTAP message
Harald Welte96e2a002017-06-12 21:44:18 +02001240 * \param[in] msg_l3 Messge Buffer containing Layer3 message
1241 * \param[in] link_id Link Identifier
1242 * \returns callee-allocated msgb with BSSMAP DTAP message */
Holger Hans Peter Freytherc25c6682010-11-04 12:26:06 +01001243struct msgb *gsm0808_create_dtap(struct msgb *msg_l3, uint8_t link_id)
1244{
1245 struct dtap_header *header;
1246 uint8_t *data;
1247 struct msgb *msg = msgb_alloc_headroom(BSSMAP_MSG_SIZE, BSSMAP_MSG_HEADROOM,
1248 "dtap");
1249 if (!msg)
1250 return NULL;
1251
1252 /* DTAP header */
1253 msg->l3h = msgb_put(msg, sizeof(*header));
1254 header = (struct dtap_header *) &msg->l3h[0];
1255 header->type = BSSAP_MSG_DTAP;
1256 header->link_id = link_id;
1257 header->length = msgb_l3len(msg_l3);
1258
1259 /* Payload */
1260 data = msgb_put(msg, header->length);
1261 memcpy(data, msg_l3->l3h, header->length);
1262
1263 return msg;
1264}
1265
Harald Welte92107df2014-06-21 23:16:20 +02001266/* As per 3GPP TS 48.008 version 11.7.0 Release 11 */
Holger Hans Peter Freyther7daa01c2010-04-17 05:14:36 +02001267static const struct tlv_definition bss_att_tlvdef = {
1268 .def = {
Harald Welte92107df2014-06-21 23:16:20 +02001269 [GSM0808_IE_CIRCUIT_IDENTITY_CODE] = { TLV_TYPE_FIXED, 2 },
1270 [GSM0808_IE_CONNECTION_RELEASE_RQSTED] = { TLV_TYPE_TV },
1271 [GSM0808_IE_RESOURCE_AVAILABLE] = { TLV_TYPE_FIXED, 21 },
1272 [GSM0808_IE_CAUSE] = { TLV_TYPE_TLV },
Holger Hans Peter Freyther7daa01c2010-04-17 05:14:36 +02001273 [GSM0808_IE_IMSI] = { TLV_TYPE_TLV },
1274 [GSM0808_IE_TMSI] = { TLV_TYPE_TLV },
Harald Welte92107df2014-06-21 23:16:20 +02001275 [GSM0808_IE_NUMBER_OF_MSS] = { TLV_TYPE_TV },
Dmitri Soloviev29099422013-07-11 09:25:37 +02001276 [GSM0808_IE_LAYER_3_HEADER_INFORMATION] = { TLV_TYPE_TLV },
Harald Welte92107df2014-06-21 23:16:20 +02001277 [GSM0808_IE_ENCRYPTION_INFORMATION] = { TLV_TYPE_TLV },
1278 [GSM0808_IE_CHANNEL_TYPE] = { TLV_TYPE_TLV },
1279 [GSM0808_IE_PERIODICITY] = { TLV_TYPE_TV },
1280 [GSM0808_IE_EXTENDED_RESOURCE_INDICATOR]= { TLV_TYPE_TV },
1281 [GSM0808_IE_TOTAL_RESOURCE_ACCESSIBLE] = { TLV_TYPE_FIXED, 4 },
1282 [GSM0808_IE_LSA_IDENTIFIER] = { TLV_TYPE_TLV },
1283 [GSM0808_IE_LSA_IDENTIFIER_LIST] = { TLV_TYPE_TLV },
Holger Hans Peter Freyther715e9452014-08-21 14:17:45 +02001284 [GSM0808_IE_LSA_INFORMATION] = { TLV_TYPE_TLV },
Harald Welte92107df2014-06-21 23:16:20 +02001285 [GSM0808_IE_CELL_IDENTIFIER] = { TLV_TYPE_TLV },
1286 [GSM0808_IE_PRIORITY] = { TLV_TYPE_TLV },
1287 [GSM0808_IE_CLASSMARK_INFORMATION_T2] = { TLV_TYPE_TLV },
1288 [GSM0808_IE_CLASSMARK_INFORMATION_T3] = { TLV_TYPE_TLV },
1289 [GSM0808_IE_INTERFERENCE_BAND_TO_USE] = { TLV_TYPE_TV },
1290 [GSM0808_IE_RR_CAUSE] = { TLV_TYPE_TV },
Harald Welte92107df2014-06-21 23:16:20 +02001291 [GSM0808_IE_LAYER_3_INFORMATION] = { TLV_TYPE_TLV },
1292 [GSM0808_IE_DLCI] = { TLV_TYPE_TV },
1293 [GSM0808_IE_DOWNLINK_DTX_FLAG] = { TLV_TYPE_TV },
1294 [GSM0808_IE_CELL_IDENTIFIER_LIST] = { TLV_TYPE_TLV },
1295 [GSM0808_IE_CELL_ID_LIST_SEGMENT] = { TLV_TYPE_TLV },
1296 [GSM0808_IE_CELL_ID_LIST_SEG_EST_CELLS] = { TLV_TYPE_TLV },
1297 [GSM0808_IE_CELL_ID_LIST_SEG_CELLS_TBE] = { TLV_TYPE_TLV },
1298 [GSM0808_IE_CELL_ID_LIST_SEG_REL_CELLS] = { TLV_TYPE_TLV },
1299 [GSM0808_IE_CELL_ID_LIST_SEG_NE_CELLS] = { TLV_TYPE_TLV },
1300 [GSM0808_IE_RESPONSE_RQST] = { TLV_TYPE_T },
1301 [GSM0808_IE_RESOURCE_INDICATION_METHOD] = { TLV_TYPE_TV },
1302 [GSM0808_IE_CLASSMARK_INFORMATION_TYPE_1] = { TLV_TYPE_TV },
1303 [GSM0808_IE_CIRCUIT_IDENTITY_CODE_LIST] = { TLV_TYPE_TLV },
1304 [GSM0808_IE_DIAGNOSTIC] = { TLV_TYPE_TLV },
1305 [GSM0808_IE_CHOSEN_CHANNEL] = { TLV_TYPE_TV },
1306 [GSM0808_IE_CIPHER_RESPONSE_MODE] = { TLV_TYPE_TV },
1307 [GSM0808_IE_LAYER_3_MESSAGE_CONTENTS] = { TLV_TYPE_TLV },
1308 [GSM0808_IE_CHANNEL_NEEDED] = { TLV_TYPE_TV },
1309 [GSM0808_IE_TRACE_TYPE] = { TLV_TYPE_TV },
1310 [GSM0808_IE_TRIGGERID] = { TLV_TYPE_TLV },
1311 [GSM0808_IE_TRACE_REFERENCE] = { TLV_TYPE_TV },
1312 [GSM0808_IE_TRANSACTIONID] = { TLV_TYPE_TLV },
1313 [GSM0808_IE_MOBILE_IDENTITY] = { TLV_TYPE_TLV },
1314 [GSM0808_IE_OMCID] = { TLV_TYPE_TLV },
1315 [GSM0808_IE_FORWARD_INDICATOR] = { TLV_TYPE_TV },
Holger Hans Peter Freytherc2b7f922010-08-04 18:50:43 +08001316 [GSM0808_IE_CHOSEN_ENCR_ALG] = { TLV_TYPE_TV },
Harald Welte92107df2014-06-21 23:16:20 +02001317 [GSM0808_IE_CIRCUIT_POOL] = { TLV_TYPE_TV },
1318 [GSM0808_IE_CIRCUIT_POOL_LIST] = { TLV_TYPE_TLV },
1319 [GSM0808_IE_TIME_INDICATION] = { TLV_TYPE_TV },
1320 [GSM0808_IE_RESOURCE_SITUATION] = { TLV_TYPE_TLV },
1321 [GSM0808_IE_CURRENT_CHANNEL_TYPE_1] = { TLV_TYPE_TV },
1322 [GSM0808_IE_QUEUEING_INDICATOR] = { TLV_TYPE_TV },
1323 [GSM0808_IE_SPEECH_VERSION] = { TLV_TYPE_TV },
1324 [GSM0808_IE_ASSIGNMENT_REQUIREMENT] = { TLV_TYPE_TV },
1325 [GSM0808_IE_TALKER_FLAG] = { TLV_TYPE_T },
1326 [GSM0808_IE_GROUP_CALL_REFERENCE] = { TLV_TYPE_TLV },
1327 [GSM0808_IE_EMLPP_PRIORITY] = { TLV_TYPE_TV },
1328 [GSM0808_IE_CONFIG_EVO_INDI] = { TLV_TYPE_TV },
1329 [GSM0808_IE_OLD_BSS_TO_NEW_BSS_INFORMATION] = { TLV_TYPE_TLV },
1330 [GSM0808_IE_LCS_QOS] = { TLV_TYPE_TLV },
1331 [GSM0808_IE_LSA_ACCESS_CTRL_SUPPR] = { TLV_TYPE_TV },
1332 [GSM0808_IE_LCS_PRIORITY] = { TLV_TYPE_TLV },
1333 [GSM0808_IE_LOCATION_TYPE] = { TLV_TYPE_TLV },
1334 [GSM0808_IE_LOCATION_ESTIMATE] = { TLV_TYPE_TLV },
1335 [GSM0808_IE_POSITIONING_DATA] = { TLV_TYPE_TLV },
1336 [GSM0808_IE_LCS_CAUSE] = { TLV_TYPE_TLV },
1337 [GSM0808_IE_APDU] = { TLV_TYPE_TLV },
1338 [GSM0808_IE_NETWORK_ELEMENT_IDENTITY] = { TLV_TYPE_TLV },
1339 [GSM0808_IE_GPS_ASSISTANCE_DATA] = { TLV_TYPE_TLV },
1340 [GSM0808_IE_DECIPHERING_KEYS] = { TLV_TYPE_TLV },
1341 [GSM0808_IE_RETURN_ERROR_RQST] = { TLV_TYPE_TLV },
1342 [GSM0808_IE_RETURN_ERROR_CAUSE] = { TLV_TYPE_TLV },
1343 [GSM0808_IE_SEGMENTATION] = { TLV_TYPE_TLV },
1344 [GSM0808_IE_SERVICE_HANDOVER] = { TLV_TYPE_TLV },
1345 [GSM0808_IE_SOURCE_RNC_TO_TARGET_RNC_TRANSPARENT_UMTS] = { TLV_TYPE_TLV },
1346 [GSM0808_IE_SOURCE_RNC_TO_TARGET_RNC_TRANSPARENT_CDMA2000] = { TLV_TYPE_TLV },
1347 [GSM0808_IE_GERAN_CLASSMARK] = { TLV_TYPE_TLV },
1348 [GSM0808_IE_GERAN_BSC_CONTAINER] = { TLV_TYPE_TLV },
1349 [GSM0808_IE_NEW_BSS_TO_OLD_BSS_INFO] = { TLV_TYPE_TLV },
1350 [GSM0800_IE_INTER_SYSTEM_INFO] = { TLV_TYPE_TLV },
1351 [GSM0808_IE_SNA_ACCESS_INFO] = { TLV_TYPE_TLV },
1352 [GSM0808_IE_VSTK_RAND_INFO] = { TLV_TYPE_TLV },
1353 [GSM0808_IE_PAGING_INFO] = { TLV_TYPE_TV },
1354 [GSM0808_IE_IMEI] = { TLV_TYPE_TLV },
1355 [GSM0808_IE_VELOCITY_ESTIMATE] = { TLV_TYPE_TLV },
1356 [GSM0808_IE_VGCS_FEATURE_FLAGS] = { TLV_TYPE_TLV },
1357 [GSM0808_IE_TALKER_PRIORITY] = { TLV_TYPE_TV },
1358 [GSM0808_IE_EMERGENCY_SET_INDICATION] = { TLV_TYPE_T },
1359 [GSM0808_IE_TALKER_IDENTITY] = { TLV_TYPE_TLV },
1360 [GSM0808_IE_SMS_TO_VGCS] = { TLV_TYPE_TLV },
1361 [GSM0808_IE_VGCS_TALKER_MODE] = { TLV_TYPE_TLV },
1362 [GSM0808_IE_VGCS_VBS_CELL_STATUS] = { TLV_TYPE_TLV },
1363 [GSM0808_IE_GANSS_ASSISTANCE_DATA] = { TLV_TYPE_TLV },
1364 [GSM0808_IE_GANSS_POSITIONING_DATA] = { TLV_TYPE_TLV },
1365 [GSM0808_IE_GANSS_LOCATION_TYPE] = { TLV_TYPE_TLV },
1366 [GSM0808_IE_APP_DATA] = { TLV_TYPE_TLV },
1367 [GSM0808_IE_DATA_IDENTITY] = { TLV_TYPE_TLV },
1368 [GSM0808_IE_APP_DATA_INFO] = { TLV_TYPE_TLV },
1369 [GSM0808_IE_MSISDN] = { TLV_TYPE_TLV },
1370 [GSM0808_IE_AOIP_TRASP_ADDR] = { TLV_TYPE_TLV },
1371 [GSM0808_IE_SPEECH_CODEC_LIST] = { TLV_TYPE_TLV },
1372 [GSM0808_IE_SPEECH_CODEC] = { TLV_TYPE_TLV },
1373 [GSM0808_IE_CALL_ID] = { TLV_TYPE_FIXED, 4 },
1374 [GSM0808_IE_CALL_ID_LIST] = { TLV_TYPE_TLV },
1375 [GSM0808_IE_A_IF_SEL_FOR_RESET] = { TLV_TYPE_TV },
1376 [GSM0808_IE_KC_128] = { TLV_TYPE_FIXED, 16 },
1377 [GSM0808_IE_CSG_IDENTIFIER] = { TLV_TYPE_TLV },
1378 [GSM0808_IE_REDIR_ATTEMPT_FLAG] = { TLV_TYPE_T },
1379 [GSM0808_IE_REROUTE_REJ_CAUSE] = { TLV_TYPE_TV },
1380 [GSM0808_IE_SEND_SEQ_NUM] = { TLV_TYPE_TV },
1381 [GSM0808_IE_REROUTE_COMPL_OUTCOME] = { TLV_TYPE_TV },
1382 [GSM0808_IE_GLOBAL_CALL_REF] = { TLV_TYPE_TLV },
1383 [GSM0808_IE_LCLS_CONFIG] = { TLV_TYPE_TV },
1384 [GSM0808_IE_LCLS_CONN_STATUS_CTRL] = { TLV_TYPE_TV },
1385 [GSM0808_IE_LCLS_CORR_NOT_NEEDED] = { TLV_TYPE_TV },
1386 [GSM0808_IE_LCLS_BSS_STATUS] = { TLV_TYPE_TV },
1387 [GSM0808_IE_LCLS_BREAK_REQ] = { TLV_TYPE_TV },
1388 [GSM0808_IE_CSFB_INDICATION] = { TLV_TYPE_T },
1389 [GSM0808_IE_CS_TO_PS_SRVCC] = { TLV_TYPE_T },
1390 [GSM0808_IE_SRC_ENB_TO_TGT_ENB_TRANSP] = { TLV_TYPE_TLV },
1391 [GSM0808_IE_CS_TO_PS_SRVCC_IND] = { TLV_TYPE_T },
1392 [GSM0808_IE_CN_TO_MS_TRANSP_INFO] = { TLV_TYPE_TLV },
1393 [GSM0808_IE_SELECTED_PLMN_ID] = { TLV_TYPE_FIXED, 3 },
1394 [GSM0808_IE_LAST_USED_EUTRAN_PLMN_ID] = { TLV_TYPE_FIXED, 3 },
Pau Espin Pedrol18506c82019-04-16 15:47:59 +02001395
1396 /* Osmocom extensions */
1397 [GSM0808_IE_OSMO_OSMUX_SUPPORT] = { TLV_TYPE_T },
1398 [GSM0808_IE_OSMO_OSMUX_CID] = { TLV_TYPE_TV },
Holger Hans Peter Freyther7daa01c2010-04-17 05:14:36 +02001399 },
1400};
1401
Harald Weltef4d45ab2011-07-16 12:13:00 +02001402const struct tlv_definition *gsm0808_att_tlvdef(void)
Holger Hans Peter Freyther7daa01c2010-04-17 05:14:36 +02001403{
1404 return &bss_att_tlvdef;
1405}
Harald Welte9b837e62011-07-11 17:43:19 +02001406
Pau Espin Pedrol392f6072019-11-27 12:07:04 +01001407const struct value_string gsm0406_dlci_sapi_names[] = {
1408 { DLCI_SAPI_RR_MM_CC, "RR/MM/CC" },
1409 { DLCI_SAPI_SMS, "SMS" },
1410 { 0, NULL }
1411};
1412
Harald Welte9b837e62011-07-11 17:43:19 +02001413static const struct value_string gsm0808_msgt_names[] = {
1414 { BSS_MAP_MSG_ASSIGMENT_RQST, "ASSIGNMENT REQ" },
1415 { BSS_MAP_MSG_ASSIGMENT_COMPLETE, "ASSIGNMENT COMPL" },
1416 { BSS_MAP_MSG_ASSIGMENT_FAILURE, "ASSIGNMENT FAIL" },
Harald Welte92107df2014-06-21 23:16:20 +02001417 { BSS_MAP_MSG_CHAN_MOD_RQST, "CHANNEL MODIFY REQUEST" },
Harald Welte9b837e62011-07-11 17:43:19 +02001418
1419 { BSS_MAP_MSG_HANDOVER_RQST, "HANDOVER REQ" },
1420 { BSS_MAP_MSG_HANDOVER_REQUIRED, "HANDOVER REQUIRED" },
1421 { BSS_MAP_MSG_HANDOVER_RQST_ACKNOWLEDGE,"HANDOVER REQ ACK" },
1422 { BSS_MAP_MSG_HANDOVER_CMD, "HANDOVER CMD" },
1423 { BSS_MAP_MSG_HANDOVER_COMPLETE, "HANDOVER COMPLETE" },
1424 { BSS_MAP_MSG_HANDOVER_SUCCEEDED, "HANDOVER SUCCESS" },
1425 { BSS_MAP_MSG_HANDOVER_FAILURE, "HANDOVER FAILURE" },
1426 { BSS_MAP_MSG_HANDOVER_PERFORMED, "HANDOVER PERFORMED" },
1427 { BSS_MAP_MSG_HANDOVER_CANDIDATE_ENQUIRE, "HANDOVER CAND ENQ" },
1428 { BSS_MAP_MSG_HANDOVER_CANDIDATE_RESPONSE, "HANDOVER CAND RESP" },
1429 { BSS_MAP_MSG_HANDOVER_REQUIRED_REJECT, "HANDOVER REQ REJ" },
1430 { BSS_MAP_MSG_HANDOVER_DETECT, "HANDOVER DETECT" },
Harald Welte92107df2014-06-21 23:16:20 +02001431 { BSS_MAP_MSG_INT_HANDOVER_REQUIRED, "INT HANDOVER REQ" },
1432 { BSS_MAP_MSG_INT_HANDOVER_REQUIRED_REJ,"INT HANDOVER REQ REJ" },
1433 { BSS_MAP_MSG_INT_HANDOVER_CMD, "INT HANDOVER CMD" },
1434 { BSS_MAP_MSG_INT_HANDOVER_ENQUIRY, "INT HANDOVER ENQ" },
Harald Welte9b837e62011-07-11 17:43:19 +02001435
1436 { BSS_MAP_MSG_CLEAR_CMD, "CLEAR COMMAND" },
1437 { BSS_MAP_MSG_CLEAR_COMPLETE, "CLEAR COMPLETE" },
1438 { BSS_MAP_MSG_CLEAR_RQST, "CLEAR REQUEST" },
1439 { BSS_MAP_MSG_SAPI_N_REJECT, "SAPI N REJECT" },
1440 { BSS_MAP_MSG_CONFUSION, "CONFUSION" },
1441
1442 { BSS_MAP_MSG_SUSPEND, "SUSPEND" },
1443 { BSS_MAP_MSG_RESUME, "RESUME" },
1444 { BSS_MAP_MSG_CONNECTION_ORIENTED_INFORMATION, "CONN ORIENT INFO" },
1445 { BSS_MAP_MSG_PERFORM_LOCATION_RQST, "PERFORM LOC REQ" },
1446 { BSS_MAP_MSG_LSA_INFORMATION, "LSA INFORMATION" },
1447 { BSS_MAP_MSG_PERFORM_LOCATION_RESPONSE, "PERFORM LOC RESP" },
1448 { BSS_MAP_MSG_PERFORM_LOCATION_ABORT, "PERFORM LOC ABORT" },
1449 { BSS_MAP_MSG_COMMON_ID, "COMMON ID" },
Harald Welte92107df2014-06-21 23:16:20 +02001450 { BSS_MAP_MSG_REROUTE_CMD, "REROUTE COMMAND" },
1451 { BSS_MAP_MSG_REROUTE_COMPLETE, "REROUTE COMPLETE" },
Harald Welte9b837e62011-07-11 17:43:19 +02001452
1453 { BSS_MAP_MSG_RESET, "RESET" },
1454 { BSS_MAP_MSG_RESET_ACKNOWLEDGE, "RESET ACK" },
1455 { BSS_MAP_MSG_OVERLOAD, "OVERLOAD" },
1456 { BSS_MAP_MSG_RESET_CIRCUIT, "RESET CIRCUIT" },
1457 { BSS_MAP_MSG_RESET_CIRCUIT_ACKNOWLEDGE, "RESET CIRCUIT ACK" },
1458 { BSS_MAP_MSG_MSC_INVOKE_TRACE, "MSC INVOKE TRACE" },
1459 { BSS_MAP_MSG_BSS_INVOKE_TRACE, "BSS INVOKE TRACE" },
1460 { BSS_MAP_MSG_CONNECTIONLESS_INFORMATION, "CONNLESS INFO" },
Harald Welte92107df2014-06-21 23:16:20 +02001461 { BSS_MAP_MSG_RESET_IP_RSRC, "RESET IP RESOURCE" },
1462 { BSS_MAP_MSG_RESET_IP_RSRC_ACK, "RESET IP RESOURCE ACK" },
Harald Welte9b837e62011-07-11 17:43:19 +02001463
1464 { BSS_MAP_MSG_BLOCK, "BLOCK" },
1465 { BSS_MAP_MSG_BLOCKING_ACKNOWLEDGE, "BLOCK ACK" },
1466 { BSS_MAP_MSG_UNBLOCK, "UNBLOCK" },
1467 { BSS_MAP_MSG_UNBLOCKING_ACKNOWLEDGE, "UNBLOCK ACK" },
1468 { BSS_MAP_MSG_CIRCUIT_GROUP_BLOCK, "CIRC GROUP BLOCK" },
1469 { BSS_MAP_MSG_CIRCUIT_GROUP_BLOCKING_ACKNOWLEDGE, "CIRC GORUP BLOCK ACK" },
1470 { BSS_MAP_MSG_CIRCUIT_GROUP_UNBLOCK, "CIRC GROUP UNBLOCK" },
1471 { BSS_MAP_MSG_CIRCUIT_GROUP_UNBLOCKING_ACKNOWLEDGE, "CIRC GROUP UNBLOCK ACK" },
1472 { BSS_MAP_MSG_UNEQUIPPED_CIRCUIT, "UNEQUIPPED CIRCUIT" },
1473 { BSS_MAP_MSG_CHANGE_CIRCUIT, "CHANGE CIRCUIT" },
1474 { BSS_MAP_MSG_CHANGE_CIRCUIT_ACKNOWLEDGE, "CHANGE CIRCUIT ACK" },
1475
1476 { BSS_MAP_MSG_RESOURCE_RQST, "RESOURCE REQ" },
1477 { BSS_MAP_MSG_RESOURCE_INDICATION, "RESOURCE IND" },
1478 { BSS_MAP_MSG_PAGING, "PAGING" },
1479 { BSS_MAP_MSG_CIPHER_MODE_CMD, "CIPHER MODE CMD" },
1480 { BSS_MAP_MSG_CLASSMARK_UPDATE, "CLASSMARK UPDATE" },
1481 { BSS_MAP_MSG_CIPHER_MODE_COMPLETE, "CIPHER MODE COMPLETE" },
1482 { BSS_MAP_MSG_QUEUING_INDICATION, "QUEUING INDICATION" },
1483 { BSS_MAP_MSG_COMPLETE_LAYER_3, "COMPLETE LAYER 3" },
1484 { BSS_MAP_MSG_CLASSMARK_RQST, "CLASSMARK REQ" },
1485 { BSS_MAP_MSG_CIPHER_MODE_REJECT, "CIPHER MODE REJECT" },
1486 { BSS_MAP_MSG_LOAD_INDICATION, "LOAD IND" },
1487
Harald Welte92107df2014-06-21 23:16:20 +02001488 { BSS_MAP_MSG_VGCS_VBS_SETUP, "VGCS/VBS SETUP" },
1489 { BSS_MAP_MSG_VGCS_VBS_SETUP_ACK, "VGCS/VBS SETUP ACK" },
1490 { BSS_MAP_MSG_VGCS_VBS_SETUP_REFUSE, "VGCS/VBS SETUP REFUSE" },
1491 { BSS_MAP_MSG_VGCS_VBS_ASSIGNMENT_RQST, "VGCS/VBS ASSIGN REQ" },
1492 { BSS_MAP_MSG_VGCS_VBS_ASSIGNMENT_RESULT, "VGCS/VBS ASSIGN RES" },
1493 { BSS_MAP_MSG_VGCS_VBS_ASSIGNMENT_FAILURE, "VGCS/VBS ASSIGN FAIL" },
1494 { BSS_MAP_MSG_VGCS_VBS_QUEUING_INDICATION, "VGCS/VBS QUEUING IND" },
1495 { BSS_MAP_MSG_UPLINK_RQST, "UPLINK REQ" },
1496 { BSS_MAP_MSG_UPLINK_RQST_ACKNOWLEDGE, "UPLINK REQ ACK" },
1497 { BSS_MAP_MSG_UPLINK_RQST_CONFIRMATION, "UPLINK REQ CONF" },
1498 { BSS_MAP_MSG_UPLINK_RELEASE_INDICATION,"UPLINK REL IND" },
1499 { BSS_MAP_MSG_UPLINK_REJECT_CMD, "UPLINK REJ CMD" },
1500 { BSS_MAP_MSG_UPLINK_RELEASE_CMD, "UPLINK REL CMD" },
1501 { BSS_MAP_MSG_UPLINK_SEIZED_CMD, "UPLINK SEIZED CMD" },
1502 { BSS_MAP_MSG_VGCS_ADDL_INFO, "VGCS ADDL INFO" },
1503 { BSS_MAP_MSG_NOTIFICATION_DATA, "NOTIF DATA" },
1504 { BSS_MAP_MSG_UPLINK_APP_DATA, "UPLINK APP DATA" },
1505
1506 { BSS_MAP_MSG_LCLS_CONNECT_CTRL, "LCLS-CONNECT-CONTROL" },
1507 { BSS_MAP_MSG_LCLS_CONNECT_CTRL_ACK, "CLS-CONNECT-CONTROL-ACK" },
1508 { BSS_MAP_MSG_LCLS_NOTIFICATION, "LCLS-NOTIFICATION" },
Harald Welte9b837e62011-07-11 17:43:19 +02001509
1510 { 0, NULL }
1511};
1512
Neels Hofmeyr87e45502017-06-20 00:17:59 +02001513/*! Return string name of BSSMAP Message Type */
Harald Welte9b837e62011-07-11 17:43:19 +02001514const char *gsm0808_bssmap_name(uint8_t msg_type)
1515{
1516 return get_value_string(gsm0808_msgt_names, msg_type);
1517}
1518
1519static const struct value_string gsm0808_bssap_names[] = {
1520 { BSSAP_MSG_BSS_MANAGEMENT, "MANAGEMENT" },
1521 { BSSAP_MSG_DTAP, "DTAP" },
Neels Hofmeyr90fdb082017-03-01 14:59:44 +01001522 { 0, NULL }
Harald Welte9b837e62011-07-11 17:43:19 +02001523};
1524
Neels Hofmeyr87e45502017-06-20 00:17:59 +02001525/*! Return string name of BSSAP Message Type */
Harald Welte9b837e62011-07-11 17:43:19 +02001526const char *gsm0808_bssap_name(uint8_t msg_type)
1527{
1528 return get_value_string(gsm0808_bssap_names, msg_type);
1529}
Harald Welte96e2a002017-06-12 21:44:18 +02001530
Neels Hofmeyrffad5742018-01-12 05:34:03 +01001531const struct value_string gsm0808_speech_codec_type_names[] = {
1532 { GSM0808_SCT_FR1, "FR1" },
1533 { GSM0808_SCT_FR2, "FR2" },
1534 { GSM0808_SCT_FR3, "FR3" },
1535 { GSM0808_SCT_FR4, "FR4" },
1536 { GSM0808_SCT_FR5, "FR5" },
1537 { GSM0808_SCT_HR1, "HR1" },
1538 { GSM0808_SCT_HR3, "HR3" },
1539 { GSM0808_SCT_HR4, "HR4" },
1540 { GSM0808_SCT_HR6, "HR6" },
1541 { GSM0808_SCT_CSD, "CSD" },
1542 { 0, NULL }
1543};
1544
Philipp Maiercdd05812018-07-12 18:21:07 +02001545const struct value_string gsm0808_permitted_speech_names[] = {
1546 { GSM0808_PERM_FR1, "FR1" },
1547 { GSM0808_PERM_FR2, "FR2" },
1548 { GSM0808_PERM_FR3, "FR3" },
1549 { GSM0808_PERM_FR4, "FR4" },
1550 { GSM0808_PERM_FR5, "FR5" },
1551 { GSM0808_PERM_HR1, "HR1" },
1552 { GSM0808_PERM_HR2, "HR2" },
1553 { GSM0808_PERM_HR3, "HR3" },
1554 { GSM0808_PERM_HR4, "HR4" },
1555 { GSM0808_PERM_HR6, "HR6" },
1556 { 0, NULL }
1557};
1558
Pau Espin Pedrolf2cda622018-07-06 17:16:41 +02001559const struct value_string gsm0808_chosen_enc_alg_names[] = {
1560 { GSM0808_ALG_ID_A5_0, "A5/0" },
1561 { GSM0808_ALG_ID_A5_1, "A5/1" },
1562 { GSM0808_ALG_ID_A5_2, "A5/2" },
1563 { GSM0808_ALG_ID_A5_3, "A5/3" },
1564 { GSM0808_ALG_ID_A5_4, "A5/4" },
1565 { GSM0808_ALG_ID_A5_5, "A5/5" },
1566 { GSM0808_ALG_ID_A5_6, "A5/6" },
1567 { GSM0808_ALG_ID_A5_7, "A5/7" },
1568 { 0, NULL }
1569};
1570
Philipp Maierdbb76592018-03-29 12:55:26 +02001571static const struct value_string gsm0808_cause_names[] = {
1572 { GSM0808_CAUSE_RADIO_INTERFACE_MESSAGE_FAILURE, "RADIO INTERFACE MESSAGE FAILURE" },
1573 { GSM0808_CAUSE_RADIO_INTERFACE_FAILURE, "RADIO INTERFACE FAILURE" },
1574 { GSM0808_CAUSE_UPLINK_QUALITY, "UPLINK QUALITY" },
1575 { GSM0808_CAUSE_UPLINK_STRENGTH, "UPLINK STRENGTH" },
1576 { GSM0808_CAUSE_DOWNLINK_QUALITY, "DOWNLINK QUALITY" },
1577 { GSM0808_CAUSE_DOWNLINK_STRENGTH, "DOWNLINK STRENGTH" },
1578 { GSM0808_CAUSE_DISTANCE, "DISTANCE" },
1579 { GSM0808_CAUSE_O_AND_M_INTERVENTION, "O AND M INTERVENTION" },
1580 { GSM0808_CAUSE_RESPONSE_TO_MSC_INVOCATION, "RESPONSE TO MSC INVOCATION" },
1581 { GSM0808_CAUSE_CALL_CONTROL, "CALL CONTROL" },
1582 { GSM0808_CAUSE_RADIO_INTERFACE_FAILURE_REVERSION, "RADIO INTERFACE FAILURE REVERSION" },
1583 { GSM0808_CAUSE_HANDOVER_SUCCESSFUL, "HANDOVER SUCCESSFUL" },
1584 { GSM0808_CAUSE_BETTER_CELL, "BETTER CELL" },
1585 { GSM0808_CAUSE_DIRECTED_RETRY, "DIRECTED RETRY" },
1586 { GSM0808_CAUSE_JOINED_GROUP_CALL_CHANNEL, "JOINED GROUP CALL CHANNEL" },
1587 { GSM0808_CAUSE_TRAFFIC, "TRAFFIC" },
1588 { GSM0808_CAUSE_REDUCE_LOAD_IN_SERVING_CELL, "REDUCE LOAD IN SERVING CELL" },
1589 { GSM0808_CAUSE_TRAFFIC_LOAD_IN_TGT_HIGHER_THAN_IN_SRC_CELL, "TRAFFIC LOAD IN TGT HIGHER THAN IN SRC CELL" },
1590 { GSM0808_CAUSE_RELOCATION_TRIGGERED, "RELOCATION TRIGGERED" },
Thorsten Alteholz0062a5f2018-05-15 15:28:55 +02001591 { GSM0808_CAUSE_REQUESTED_OPT_NOT_AUTHORISED, "REQUESTED OPT NOT AUTHORISED" },
Philipp Maierdbb76592018-03-29 12:55:26 +02001592 { GSM0808_CAUSE_ALT_CHAN_CONFIG_REQUESTED, "ALT CHAN CONFIG REQUESTED" },
1593 { GSM0808_CAUSE_RESP_TO_INT_HO_ENQ_MSG, "RESP TO INT HO ENQ MSG" },
1594 { GSM0808_CAUSE_INT_HO_ENQUIRY_REJECT, "INT HO ENQUIRY REJECT" },
1595 { GSM0808_CAUSE_REDUNDANCY_LEVEL_NOT_ADEQUATE, "REDUNDANCY LEVEL NOT ADEQUATE" },
1596 { GSM0808_CAUSE_EQUIPMENT_FAILURE, "EQUIPMENT FAILURE" },
1597 { GSM0808_CAUSE_NO_RADIO_RESOURCE_AVAILABLE, "NO RADIO RESOURCE AVAILABLE" },
1598 { GSM0808_CAUSE_RQSTED_TERRESTRIAL_RESOURCE_UNAVAILABLE, "RQSTED TERRESTRIAL RESOURCE UNAVAILABLE" },
1599 { GSM0808_CAUSE_CCCH_OVERLOAD, "CCCH OVERLOAD" },
1600 { GSM0808_CAUSE_PROCESSOR_OVERLOAD, "PROCESSOR OVERLOAD" },
1601 { GSM0808_CAUSE_BSS_NOT_EQUIPPED, "BSS NOT EQUIPPED" },
1602 { GSM0808_CAUSE_MS_NOT_EQUIPPED, "MS NOT EQUIPPED" },
1603 { GSM0808_CAUSE_INVALID_CELL, "INVALID CELL" },
1604 { GSM0808_CAUSE_TRAFFIC_LOAD, "TRAFFIC LOAD" },
1605 { GSM0808_CAUSE_PREEMPTION, "PREEMPTION" },
1606 { GSM0808_CAUSE_DTM_HO_SGSN_FAILURE, "DTM HO SGSN FAILURE" },
1607 { GSM0808_CAUSE_DTM_HO_PS_ALLOC_FAILURE, "DTM HO PS ALLOC FAILURE" },
1608 { GSM0808_CAUSE_RQSTED_TRANSCODING_RATE_ADAPTION_UNAVAILABLE, "RQSTED TRANSCODING RATE ADAPTION UNAVAILABLE" },
1609 { GSM0808_CAUSE_CIRCUIT_POOL_MISMATCH, "CIRCUIT POOL MISMATCH" },
1610 { GSM0808_CAUSE_SWITCH_CIRCUIT_POOL, "SWITCH CIRCUIT POOL" },
1611 { GSM0808_CAUSE_RQSTED_SPEECH_VERSION_UNAVAILABLE, "RQSTED SPEECH VERSION UNAVAILABLE" },
1612 { GSM0808_CAUSE_LSA_NOT_ALLOWED, "LSA NOT ALLOWED" },
1613 { GSM0808_CAUSE_REQ_CODEC_TYPE_OR_CONFIG_UNAVAIL, "REQ CODEC TYPE OR CONFIG UNAVAIL" },
1614 { GSM0808_CAUSE_REQ_A_IF_TYPE_UNAVAIL, "REQ A IF TYPE UNAVAIL" },
1615 { GSM0808_CAUSE_INVALID_CSG_CELL, "INVALID CSG CELL" },
1616 { GSM0808_CAUSE_REQ_REDUND_LEVEL_NOT_AVAIL, "REQ REDUND LEVEL NOT AVAIL" },
1617 { GSM0808_CAUSE_CIPHERING_ALGORITHM_NOT_SUPPORTED, "CIPHERING ALGORITHM NOT SUPPORTED" },
1618 { GSM0808_CAUSE_GERAN_IU_MODE_FAILURE, "GERAN IU MODE FAILURE" },
1619 { GSM0808_CAUSE_INC_RELOC_NOT_SUPP_DT_PUESBINE_FEATURE, "INC RELOC NOT SUPP DT PUESBINE FEATURE" },
1620 { GSM0808_CAUSE_ACCESS_RESTRICTED_DUE_TO_SHARED_NETWORKS, "ACCESS RESTRICTED DUE TO SHARED NETWORKS" },
1621 { GSM0808_CAUSE_REQ_CODEC_TYPE_OR_CONFIG_NOT_SUPP, "REQ CODEC TYPE OR CONFIG NOT SUPP" },
1622 { GSM0808_CAUSE_REQ_A_IF_TYPE_NOT_SUPP, "REQ A IF TYPE NOT SUPP" },
1623 { GSM0808_CAUSE_REQ_REDUND_LVL_NOT_SUPP, "REQ REDUND LVL NOT SUPP" },
1624 { GSM0808_CAUSE_TERRESTRIAL_CIRCUIT_ALREADY_ALLOCATED, "TERRESTRIAL CIRCUIT ALREADY ALLOCATED" },
1625 { GSM0808_CAUSE_INVALID_MESSAGE_CONTENTS, "INVALID MESSAGE CONTENTS" },
1626 { GSM0808_CAUSE_INFORMATION_ELEMENT_OR_FIELD_MISSING, "INFORMATION ELEMENT OR FIELD MISSING" },
1627 { GSM0808_CAUSE_INCORRECT_VALUE, "INCORRECT VALUE" },
1628 { GSM0808_CAUSE_UNKNOWN_MESSAGE_TYPE, "UNKNOWN MESSAGE TYPE" },
1629 { GSM0808_CAUSE_UNKNOWN_INFORMATION_ELEMENT, "UNKNOWN INFORMATION ELEMENT" },
1630 { GSM0808_CAUSE_DTM_HO_INVALID_PS_IND, "DTM HO INVALID PS IND" },
1631 { GSM0808_CAUSE_CALL_ID_ALREADY_ALLOC, "CALL ID ALREADY ALLOC" },
1632 { GSM0808_CAUSE_PROTOCOL_ERROR_BETWEEN_BSS_AND_MSC, "PROTOCOL ERROR BETWEEN BSS AND MSC" },
Thorsten Alteholz0062a5f2018-05-15 15:28:55 +02001633 { GSM0808_CAUSE_VGCS_VBS_CALL_NON_EXISTENT, "VGCS VBS CALL NON EXISTENT" },
Philipp Maierdbb76592018-03-29 12:55:26 +02001634 { GSM0808_CAUSE_DTM_HO_TIMER_EXPIRY, "DTM HO TIMER EXPIRY" },
1635 { 0, NULL }
1636};
1637
Maxaa934632018-11-07 13:16:54 +01001638static const struct value_string gsm0808_cause_class_names[] = {
1639 { GSM0808_CAUSE_CLASS_NORM0, "Normal event" },
1640 { GSM0808_CAUSE_CLASS_NORM1, "Normal event" },
1641 { GSM0808_CAUSE_CLASS_RES_UNAVAIL, "Resource unavailable" },
1642 { GSM0808_CAUSE_CLASS_SRV_OPT_NA, "Service or option not available" },
1643 { GSM0808_CAUSE_CLASS_SRV_OPT_NIMPL, "Service or option not implemented" },
1644 { GSM0808_CAUSE_CLASS_INVAL, "Invalid message" },
1645 { GSM0808_CAUSE_CLASS_PERR, "Protocol error" },
1646 { GSM0808_CAUSE_CLASS_INTW, "Interworking" },
1647 { 0, NULL }
1648};
1649
1650/*! Return string name of BSSMAP Cause Class name */
1651const char *gsm0808_cause_class_name(enum gsm0808_cause_class class)
1652{
1653 return get_value_string(gsm0808_cause_class_names, class);
1654}
1655
Philipp Maierdbb76592018-03-29 12:55:26 +02001656/*! Return string name of BSSMAP Cause name */
Maxaa934632018-11-07 13:16:54 +01001657const char *gsm0808_cause_name(enum gsm0808_cause cause)
Philipp Maierdbb76592018-03-29 12:55:26 +02001658{
1659 return get_value_string(gsm0808_cause_names, cause);
1660}
1661
Alexander Chemerisfdfe25b2020-05-12 23:21:56 +03001662enum gsm0808_cause gsm0808_get_cause(const struct tlv_parsed *tp)
1663{
1664 const uint8_t *buf = TLVP_VAL_MINLEN(tp, GSM0808_IE_CAUSE, 1);
1665
1666 if (!buf)
1667 return -EBADMSG;
1668
1669 if (TLVP_LEN(tp, GSM0808_IE_CAUSE) > 1) {
1670 if (!gsm0808_cause_ext(buf[0]))
1671 return -EINVAL;
1672 return buf[1];
1673 }
1674
1675 return buf[0];
1676}
1677
Alexander Chemeris22630e62020-05-13 00:44:04 +03001678const char *gsm0808_diagnostics_octet_location_str(uint8_t pointer)
1679{
1680 switch (pointer) {
1681 case 0:
1682 return "Error location not determined";
1683 case 1:
1684 return "The first octet of the message received (i.e. the message type) was found erroneous (unknown)";
1685 case 0xfd:
1686 return "The first octet of the BSSAP header (Discrimination) was found erroneous";
1687 case 0xfe:
1688 return "(DTAP only) The DLCI (second) octet of the BSSAP header was found erroneous";
1689 case 0xff:
1690 return "The last octet of the BSSAP header (length indicator) was found erroneous";
1691 default:
1692 snprintf(str_buff, sizeof(str_buff), "The %d octet of the message received was found erroneous", pointer);
1693 return str_buff;
1694 }
1695}
1696
1697const char *gsm0808_diagnostics_bit_location_str(uint8_t bit_pointer)
1698{
1699 if (bit_pointer == 0) {
1700 return "No particular part of the octet is indicated";
1701 } else if (bit_pointer > 8) {
1702 return "Reserved value";
1703 }
1704
1705 snprintf(str_buff, sizeof(str_buff),
1706 "An error was provoked by the field whose most significant bit is in bit position %d",
1707 bit_pointer);
1708 return str_buff;
1709}
1710
Harald Welteebd362d2018-06-02 14:11:19 +02001711const struct value_string gsm0808_lcls_config_names[] = {
1712 { GSM0808_LCLS_CFG_BOTH_WAY, "Connect both-way" },
1713 { GSM0808_LCLS_CFG_BOTH_WAY_AND_BICAST_UL,
1714 "Connect both-way, bi-cast UL to CN" },
1715 { GSM0808_LCLS_CFG_BOTH_WAY_AND_SEND_DL,
1716 "Connect both-way, send access DL from CN" },
1717 { GSM0808_LCLS_CFG_BOTH_WAY_AND_SEND_DL_BLOCK_LOCAL_DL,
1718 "Connect both-way, send access DL from CN, block local DL" },
1719 { GSM0808_LCLS_CFG_BOTH_WAY_AND_BICAST_UL_SEND_DL,
1720 "Connect both-way, bi-cast UL to CN, send access DL from CN" },
1721 { GSM0808_LCLS_CFG_BOTH_WAY_AND_BICAST_UL_SEND_DL_BLOCK_LOCAL_DL,
1722 "Connect both-way, bi-cast UL to CN, send access DL from CN, block local DL" },
Max961db7c2018-11-08 11:40:23 +01001723 { GSM0808_LCLS_CFG_NA, "Not available" },
Harald Welteebd362d2018-06-02 14:11:19 +02001724 { 0, NULL }
1725};
1726
1727const struct value_string gsm0808_lcls_control_names[] = {
1728 { GSM0808_LCLS_CSC_CONNECT, "Connect" },
1729 { GSM0808_LCLS_CSC_DO_NOT_CONNECT, "Do not connect" },
1730 { GSM0808_LCLS_CSC_RELEASE_LCLS, "Release LCLS" },
1731 { GSM0808_LCLS_CSC_BICAST_UL_AT_HANDOVER, "Bi-cast UL at Handover" },
1732 { GSM0808_LCLS_CSC_BICAST_UL_AND_RECV_DL_AT_HANDOVER, "Bi-cast UL and receive DL at Handover" },
Max961db7c2018-11-08 11:40:23 +01001733 { GSM0808_LCLS_CSC_NA, "Not available" },
Harald Welteebd362d2018-06-02 14:11:19 +02001734 { 0, NULL }
1735};
1736
1737const struct value_string gsm0808_lcls_status_names[] = {
1738 { GSM0808_LCLS_STS_NOT_YET_LS, "Call not yet locally switched" },
1739 { GSM0808_LCLS_STS_NOT_POSSIBLE_LS, "Call not possible to be locally switched" },
1740 { GSM0808_LCLS_STS_NO_LONGER_LS, "Call is no longer locally switched" },
1741 { GSM0808_LCLS_STS_REQ_LCLS_NOT_SUPP, "Requested LCLS configuration is not supported" },
1742 { GSM0808_LCLS_STS_LOCALLY_SWITCHED, "Call is locally switched with requested LCLS config" },
Max961db7c2018-11-08 11:40:23 +01001743 { GSM0808_LCLS_STS_NA, "Not available" },
Harald Welteebd362d2018-06-02 14:11:19 +02001744 { 0, NULL }
1745};
1746
Harald Welte96e2a002017-06-12 21:44:18 +02001747/*! @} */