blob: e24696701ca317a7fb9d6f556ce0fbb0f41e9510 [file] [log] [blame]
Harald Welte37b61652017-10-16 18:46:03 +02001/*! \addtogroup gsm0808
2 * @{
3 * \file gsm0808_utils.h */
Neels Hofmeyr17518fe2017-06-20 04:35:06 +02004/*
Harald Weltee08da972017-11-13 01:00:26 +09005 * (C) 2016 by sysmocom - s.f.m.c. GmbH, Author: Philipp Maier
Philipp Maier22401432017-03-24 17:59:26 +01006 * All Rights Reserved
7 *
Harald Weltee08da972017-11-13 01:00:26 +09008 * SPDX-License-Identifier: GPL-2.0+
Philipp Maier22401432017-03-24 17:59:26 +01009 *
10 * This program is free software; you can redistribute it and/or modify
Harald Weltee08da972017-11-13 01:00:26 +090011 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
Philipp Maier22401432017-03-24 17:59:26 +010013 * (at your option) any later version.
14 *
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Harald Weltee08da972017-11-13 01:00:26 +090018 * GNU General Public License for more details.
Philipp Maier22401432017-03-24 17:59:26 +010019 *
Harald Weltee08da972017-11-13 01:00:26 +090020 * You should have received a copy of the GNU General Public License
Philipp Maier22401432017-03-24 17:59:26 +010021 * along with this program. If not, see <http://www.gnu.org/licenses/>.
22 *
23 */
24#pragma once
25
Harald Welte20725b92017-05-15 12:50:04 +020026struct sockaddr_storage;
Philipp Maier22401432017-03-24 17:59:26 +010027
Philipp Maier6f725d62017-03-24 18:03:17 +010028#include <osmocom/gsm/protocol/gsm_08_08.h>
Neels Hofmeyrafacc2b2018-04-16 22:41:51 +020029#include <osmocom/gsm/protocol/gsm_04_08.h>
Max969fb2e2018-12-10 11:01:10 +010030#include <osmocom/gsm/gsm29205.h>
Stefan Sperling11a4d9d2018-02-15 18:28:04 +010031#include <osmocom/gsm/gsm23003.h>
Neels Hofmeyrafacc2b2018-04-16 22:41:51 +020032#include <osmocom/gsm/gsm_utils.h>
Maxadc41722018-11-19 10:44:55 +010033#include <osmocom/gsm/tlv.h>
Stefan Sperling11a4d9d2018-02-15 18:28:04 +010034
35 /*! (225-1)/2 is the maximum number of elements in a cell identifier list. */
36#define GSM0808_CELL_ID_LIST2_MAXLEN 127
37
Neels Hofmeyr250e7f72018-04-13 03:30:14 +020038/*! Instead of this, use either struct gsm0808_cell_id or gsm0808_cell_id_list2.
39 * All elements contain parsed representations of the data in the corresponding IE, in host-byte order.
40 */
41union gsm0808_cell_id_u {
42 struct osmo_cell_global_id global;
43 struct osmo_lac_and_ci_id lac_and_ci;
44 uint16_t ci;
45 struct osmo_location_area_id lai_and_lac;
46 uint16_t lac;
47};
48
49/*! Parsed representation of Cell Identifier IE (3GPP TS 48.008 3.2.2.17) */
50struct gsm0808_cell_id {
51 enum CELL_IDENT id_discr;
52 union gsm0808_cell_id_u id;
53};
54
55/*! Parsed representation of a Cell Identifier List IE (3GPP TS 48.008 3.2.2.27). */
Stefan Sperling11a4d9d2018-02-15 18:28:04 +010056struct gsm0808_cell_id_list2 {
57 enum CELL_IDENT id_discr;
Neels Hofmeyr250e7f72018-04-13 03:30:14 +020058 union gsm0808_cell_id_u id_list[GSM0808_CELL_ID_LIST2_MAXLEN];
Stefan Sperling11a4d9d2018-02-15 18:28:04 +010059 unsigned int id_list_len;
60};
Philipp Maier6f725d62017-03-24 18:03:17 +010061
Max73878242018-11-09 12:35:03 +010062/*! LCLS-related parameters from 3GPP TS 48.008 */
63struct osmo_lcls {
Max14f24c62018-11-29 19:30:07 +010064 enum gsm0808_lcls_config config; /**< §3.2.2.116 Configuration */
65 enum gsm0808_lcls_control control; /**< §3.2.2.117 Connection Status Control */
Max3b901252019-01-15 14:15:11 +010066 struct osmo_gcr_parsed gcr; /**< §3.2.2.115 Global Call Reference */
67 bool gcr_available;
Max14f24c62018-11-29 19:30:07 +010068 bool corr_needed; /**< §3.2.2.118 Correlation-Not-Needed */
Max73878242018-11-09 12:35:03 +010069};
70
Max5ec0cf52019-01-15 16:37:09 +010071char *osmo_lcls_dump(const struct osmo_lcls *lcls);
Harald Welte4a62eda2019-03-18 18:27:00 +010072char *osmo_lcls_dump_buf(char *buf, size_t buf_len, const struct osmo_lcls *lcls);
Max5ec0cf52019-01-15 16:37:09 +010073char *osmo_gcr_dump(const struct osmo_lcls *lcls);
Harald Welte4a62eda2019-03-18 18:27:00 +010074char *osmo_gcr_dump_buf(char *buf, size_t buf_len, const struct osmo_lcls *lcls);
Max5ec0cf52019-01-15 16:37:09 +010075
Neels Hofmeyra4399c82018-04-17 02:26:10 +020076extern const struct value_string gsm0808_cell_id_discr_names[];
77static inline const char *gsm0808_cell_id_discr_name(enum CELL_IDENT id_discr)
78{ return get_value_string(gsm0808_cell_id_discr_names, id_discr); }
79
80const char *gsm0808_cell_id_name(const struct gsm0808_cell_id *cid);
81const char *gsm0808_cell_id_name2(const struct gsm0808_cell_id *cid);
82const char *gsm0808_cell_id_list_name(const struct gsm0808_cell_id_list2 *cil);
83int gsm0808_cell_id_list_name_buf(char *buf, size_t buflen, const struct gsm0808_cell_id_list2 *cil);
84int gsm0808_cell_id_u_name(char *buf, size_t buflen,
85 enum CELL_IDENT id_discr, const union gsm0808_cell_id_u *u);
Neels Hofmeyrd01ef752018-09-21 15:57:26 +020086bool gsm0808_cell_ids_match(const struct gsm0808_cell_id *id1, const struct gsm0808_cell_id *id2, bool exact_match);
87int gsm0808_cell_id_matches_list(const struct gsm0808_cell_id *id, const struct gsm0808_cell_id_list2 *list,
88 unsigned int match_nr, bool exact_match);
Neels Hofmeyr3a504532019-02-10 22:28:27 +010089void gsm0808_cell_id_from_cgi(struct gsm0808_cell_id *cid, enum CELL_IDENT id_discr,
90 const struct osmo_cell_global_id *cgi);
91int gsm0808_cell_id_to_cgi(struct osmo_cell_global_id *cgi, const struct gsm0808_cell_id *cid);
Neels Hofmeyra4399c82018-04-17 02:26:10 +020092
Philipp Maier4f4905f2018-11-30 13:36:12 +010093uint8_t gsm0808_enc_cause(struct msgb *msg, uint16_t cause);
Philipp Maier22401432017-03-24 17:59:26 +010094uint8_t gsm0808_enc_aoip_trasp_addr(struct msgb *msg,
95 const struct sockaddr_storage *ss);
Philipp Maier22401432017-03-24 17:59:26 +010096int gsm0808_dec_aoip_trasp_addr(struct sockaddr_storage *ss,
97 const uint8_t *elem, uint8_t len);
Max969fb2e2018-12-10 11:01:10 +010098
Max47022152018-12-19 18:51:00 +010099uint8_t gsm0808_enc_lcls(struct msgb *msg, const struct osmo_lcls *lcls);
100int gsm0808_dec_lcls(struct osmo_lcls *lcls, const struct tlv_parsed *tp);
Max969fb2e2018-12-10 11:01:10 +0100101
Philipp Maier6f725d62017-03-24 18:03:17 +0100102uint8_t gsm0808_enc_speech_codec(struct msgb *msg,
103 const struct gsm0808_speech_codec *sc);
Philipp Maier6f725d62017-03-24 18:03:17 +0100104int gsm0808_dec_speech_codec(struct gsm0808_speech_codec *sc,
105 const uint8_t *elem, uint8_t len);
Philipp Maier6f725d62017-03-24 18:03:17 +0100106uint8_t gsm0808_enc_speech_codec_list(struct msgb *msg,
107 const struct gsm0808_speech_codec_list
108 *scl);
Philipp Maier6f725d62017-03-24 18:03:17 +0100109int gsm0808_dec_speech_codec_list(struct gsm0808_speech_codec_list *scl,
110 const uint8_t *elem, uint8_t len);
Philipp Maiere0c65302017-03-28 17:05:40 +0200111uint8_t gsm0808_enc_channel_type(struct msgb *msg,
112 const struct gsm0808_channel_type *ct);
Philipp Maiere0c65302017-03-28 17:05:40 +0200113int gsm0808_dec_channel_type(struct gsm0808_channel_type *ct,
114 const uint8_t *elem, uint8_t len);
Philipp Maier14e76b92017-03-28 18:36:52 +0200115uint8_t gsm0808_enc_encrypt_info(struct msgb *msg,
116 const struct gsm0808_encrypt_info *ei);
Philipp Maier14e76b92017-03-28 18:36:52 +0200117int gsm0808_dec_encrypt_info(struct gsm0808_encrypt_info *ei,
118 const uint8_t *elem, uint8_t len);
Stefan Sperling11a4d9d2018-02-15 18:28:04 +0100119uint8_t gsm0808_enc_cell_id_list2(struct msgb *msg, const struct gsm0808_cell_id_list2 *cil);
Philipp Maier783047e2017-03-29 11:35:50 +0200120uint8_t gsm0808_enc_cell_id_list(struct msgb *msg,
Stefan Sperling11a4d9d2018-02-15 18:28:04 +0100121 const struct gsm0808_cell_id_list *cil)
122 OSMO_DEPRECATED("use gsm0808_enc_cell_id_list2 instead");
123int gsm0808_dec_cell_id_list2(struct gsm0808_cell_id_list2 *cil, const uint8_t *elem, uint8_t len);
Philipp Maier783047e2017-03-29 11:35:50 +0200124int gsm0808_dec_cell_id_list(struct gsm0808_cell_id_list *cil,
Stefan Sperling11a4d9d2018-02-15 18:28:04 +0100125 const uint8_t *elem, uint8_t len)
126 OSMO_DEPRECATED("use gsm0808_dec_cell_id_list2 instead");
Neels Hofmeyr74663d92018-03-23 01:46:42 +0100127int gsm0808_cell_id_list_add(struct gsm0808_cell_id_list2 *dst, const struct gsm0808_cell_id_list2 *src);
Neels Hofmeyr38e58412018-05-25 16:56:35 +0200128void gsm0808_cell_id_to_list(struct gsm0808_cell_id_list2 *dst, const struct gsm0808_cell_id *src);
Neels Hofmeyr250e7f72018-04-13 03:30:14 +0200129uint8_t gsm0808_enc_cell_id(struct msgb *msg, const struct gsm0808_cell_id *ci);
130int gsm0808_dec_cell_id(struct gsm0808_cell_id *ci, const uint8_t *elem, uint8_t len);
Neels Hofmeyr9a391e22017-06-23 02:34:08 +0200131int gsm0808_chan_type_to_speech_codec(uint8_t perm_spch);
Philipp Maier884ba0f2017-06-02 13:49:16 +0200132int gsm0808_speech_codec_from_chan_type(struct gsm0808_speech_codec *sc,
133 uint8_t perm_spch);
Philipp Maier369015c2018-09-21 09:07:20 +0200134uint16_t gsm0808_sc_cfg_from_gsm48_mr_cfg(const struct gsm48_multi_rate_conf *cfg, bool fr);
Philipp Maier3713af82019-02-27 16:48:25 +0100135int gsm48_mr_cfg_from_gsm0808_sc_cfg(struct gsm48_multi_rate_conf *cfg, uint16_t s15_s0);
Harald Welte37b61652017-10-16 18:46:03 +0200136
Maxaa934632018-11-07 13:16:54 +0100137/*! \returns 3GPP TS 08.08 §3.2.2.5 Class of a given Cause */
138static inline enum gsm0808_cause_class gsm0808_cause_class(enum gsm0808_cause cause)
139{
140 return (cause << 1) >> 4;
141}
142
143/*! \returns true if 3GPP TS 08.08 §3.2.2.5 Class has extended bit set */
144static inline bool gsm0808_cause_ext(enum gsm0808_cause cause)
145{
146 /* check that cause looks like 1XXX0000 where XXX represent class */
147 return (cause & 0x80) && !(cause & 0x0F);
148}
149
Maxed651d22018-11-07 15:25:05 +0100150int gsm0808_get_cipher_reject_cause(const struct tlv_parsed *tp);
151
Maxaa934632018-11-07 13:16:54 +0100152/*! \returns 3GPP TS 48.008 3.2.2.49 Current Channel Type 1 from enum gsm_chan_t. */
Neels Hofmeyrafacc2b2018-04-16 22:41:51 +0200153static inline uint8_t gsm0808_current_channel_type_1(enum gsm_chan_t type)
154{
155 switch (type) {
156 default:
157 return 0;
158 case GSM_LCHAN_SDCCH:
159 return 0x01;
160 case GSM_LCHAN_TCH_F:
161 return 0x18;
162 case GSM_LCHAN_TCH_H:
163 return 0x19;
164 }
165}
166
167/*! Return 3GPP TS 48.008 3.2.2.51 Speech Version aka permitted speech version indication in 3.2.2.11
168 * Channel Type. */
169static inline enum gsm0808_permitted_speech gsm0808_permitted_speech(enum gsm_chan_t type,
170 enum gsm48_chan_mode mode)
171{
172 switch (mode) {
173 case GSM48_CMODE_SPEECH_V1:
174 switch (type) {
175 case GSM_LCHAN_TCH_F:
176 return GSM0808_PERM_FR1;
177 case GSM_LCHAN_TCH_H:
178 return GSM0808_PERM_HR1;
179 default:
180 return 0;
181 }
182 case GSM48_CMODE_SPEECH_EFR:
183 switch (type) {
184 case GSM_LCHAN_TCH_F:
185 return GSM0808_PERM_FR2;
186 case GSM_LCHAN_TCH_H:
187 return GSM0808_PERM_HR2;
188 default:
189 return 0;
190 }
191 case GSM48_CMODE_SPEECH_AMR:
192 switch (type) {
193 case GSM_LCHAN_TCH_F:
Neels Hofmeyr94e42652018-06-07 16:54:11 +0200194 return GSM0808_PERM_FR3;
Neels Hofmeyrafacc2b2018-04-16 22:41:51 +0200195 case GSM_LCHAN_TCH_H:
196 return GSM0808_PERM_HR3;
197 default:
198 return 0;
199 }
200 default:
201 return 0;
202 }
203}
204
205/*! Return 3GPP TS 48.008 3.2.2.33 Chosen Channel. */
206static inline uint8_t gsm0808_chosen_channel(enum gsm_chan_t type, enum gsm48_chan_mode mode)
207{
208 uint8_t channel_mode = 0, channel = 0;
209
210 switch (mode) {
211 case GSM48_CMODE_SPEECH_V1:
212 case GSM48_CMODE_SPEECH_EFR:
213 case GSM48_CMODE_SPEECH_AMR:
214 channel_mode = 0x9;
215 break;
216 case GSM48_CMODE_SIGN:
217 channel_mode = 0x8;
218 break;
219 case GSM48_CMODE_DATA_14k5:
220 channel_mode = 0xe;
221 break;
222 case GSM48_CMODE_DATA_12k0:
223 channel_mode = 0xb;
224 break;
225 case GSM48_CMODE_DATA_6k0:
226 channel_mode = 0xc;
227 break;
228 case GSM48_CMODE_DATA_3k6:
229 channel_mode = 0xd;
230 break;
231 default:
232 return 0;
233 }
234
235 switch (type) {
236 case GSM_LCHAN_NONE:
237 channel = 0x0;
238 break;
239 case GSM_LCHAN_SDCCH:
240 channel = 0x1;
241 break;
242 case GSM_LCHAN_TCH_F:
243 channel = 0x8;
244 break;
245 case GSM_LCHAN_TCH_H:
246 channel = 0x9;
247 break;
248 default:
249 return 0;
250 }
251
252 return channel_mode << 4 | channel;
253}
254
255const char *gsm0808_channel_type_name(const struct gsm0808_channel_type *ct);
Harald Welte4a62eda2019-03-18 18:27:00 +0100256char *gsm0808_channel_type_name_buf(char *buf, size_t buf_len, const struct gsm0808_channel_type *ct);
Neels Hofmeyrafacc2b2018-04-16 22:41:51 +0200257
Harald Welte37b61652017-10-16 18:46:03 +0200258/*! @} */