blob: f06c403f410c2439fb7c7280ca9538e63177bfbb [file] [log] [blame]
Neels Hofmeyr17518fe2017-06-20 04:35:06 +02001/*! \file gprs_bssgp_bss.c
2 * GPRS BSSGP protocol implementation as per 3GPP TS 08.18. */
3/*
4 * (C) 2009-2017 by Harald Welte <laforge@gnumonks.org>
Harald Welte85fc3142011-11-25 08:58:40 +01005 *
6 * All Rights Reserved
7 *
Harald Weltee08da972017-11-13 01:00:26 +09008 * SPDX-License-Identifier: GPL-2.0+
9 *
Harald Welte85fc3142011-11-25 08:58:40 +010010 * This program is free software; you can redistribute it and/or modify
Harald Welte7fa89c22014-10-26 20:33:09 +010011 * 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
Harald Welte85fc3142011-11-25 08:58:40 +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 Welte7fa89c22014-10-26 20:33:09 +010018 * GNU General Public License for more details.
Harald Welte85fc3142011-11-25 08:58:40 +010019 *
Harald Welte7fa89c22014-10-26 20:33:09 +010020 * You should have received a copy of the GNU General Public License
Harald Welte85fc3142011-11-25 08:58:40 +010021 * along with this program. If not, see <http://www.gnu.org/licenses/>.
22 *
23 */
24
25#include <errno.h>
26#include <stdint.h>
27
Harald Welte85fc3142011-11-25 08:58:40 +010028#include <osmocom/core/msgb.h>
Harald Weltebfe62e52017-05-15 12:48:30 +020029#include <osmocom/core/byteswap.h>
Harald Welte85fc3142011-11-25 08:58:40 +010030#include <osmocom/core/rate_ctr.h>
31#include <osmocom/gsm/tlv.h>
32#include <osmocom/core/talloc.h>
Harald Welte73952e32012-06-16 14:59:56 +080033#include <osmocom/gprs/gprs_bssgp.h>
Harald Welte30fabdf2012-09-10 08:54:35 +020034#include <osmocom/gprs/gprs_bssgp_bss.h>
Harald Welte73952e32012-06-16 14:59:56 +080035#include <osmocom/gprs/gprs_ns.h>
Harald Welte85fc3142011-11-25 08:58:40 +010036
Harald Weltecca49632012-06-16 17:45:59 +080037#include "common_vty.h"
Harald Welte85fc3142011-11-25 08:58:40 +010038
Jacob Erlbeckc1cb75e2015-06-18 13:21:30 +020039#define GSM_IMSI_LENGTH 17
40
Harald Welte85fc3142011-11-25 08:58:40 +010041uint8_t *bssgp_msgb_tlli_put(struct msgb *msg, uint32_t tlli)
42{
Harald Weltebfe62e52017-05-15 12:48:30 +020043 uint32_t _tlli = osmo_htonl(tlli);
Harald Welte85fc3142011-11-25 08:58:40 +010044 return msgb_tvlv_put(msg, BSSGP_IE_TLLI, 4, (uint8_t *) &_tlli);
45}
46
Maxf1ad60e2018-01-05 14:19:33 +010047uint8_t *bssgp_msgb_ra_put(struct msgb *msg, const struct gprs_ra_id *ra_id)
48{
49 struct gsm48_ra_id ra;
50
51 gsm48_encode_ra(&ra, ra_id);
52 return msgb_tvlv_put(msg, BSSGP_IE_ROUTEING_AREA, sizeof(ra), (const uint8_t *)&ra);
53}
54
Neels Hofmeyr87e45502017-06-20 00:17:59 +020055/*! GMM-SUSPEND.req (Chapter 10.3.6) */
Harald Welte85fc3142011-11-25 08:58:40 +010056int bssgp_tx_suspend(uint16_t nsei, uint32_t tlli,
57 const struct gprs_ra_id *ra_id)
58{
59 struct msgb *msg = bssgp_msgb_alloc();
60 struct bssgp_normal_hdr *bgph =
61 (struct bssgp_normal_hdr *) msgb_put(msg, sizeof(*bgph));
Harald Welte85fc3142011-11-25 08:58:40 +010062
63 LOGP(DBSSGP, LOGL_NOTICE, "BSSGP (BVCI=0) Tx SUSPEND (TLLI=0x%04x)\n",
64 tlli);
65 msgb_nsei(msg) = nsei;
66 msgb_bvci(msg) = 0; /* Signalling */
67 bgph->pdu_type = BSSGP_PDUT_SUSPEND;
68
69 bssgp_msgb_tlli_put(msg, tlli);
Maxf1ad60e2018-01-05 14:19:33 +010070 bssgp_msgb_ra_put(msg, ra_id);
Harald Welte85fc3142011-11-25 08:58:40 +010071
72 return gprs_ns_sendmsg(bssgp_nsi, msg);
73}
74
Neels Hofmeyr87e45502017-06-20 00:17:59 +020075/*! GMM-RESUME.req (Chapter 10.3.9) */
Harald Welte85fc3142011-11-25 08:58:40 +010076int bssgp_tx_resume(uint16_t nsei, uint32_t tlli,
77 const struct gprs_ra_id *ra_id, uint8_t suspend_ref)
78{
79 struct msgb *msg = bssgp_msgb_alloc();
80 struct bssgp_normal_hdr *bgph =
81 (struct bssgp_normal_hdr *) msgb_put(msg, sizeof(*bgph));
Harald Welte85fc3142011-11-25 08:58:40 +010082
83 LOGP(DBSSGP, LOGL_NOTICE, "BSSGP (BVCI=0) Tx RESUME (TLLI=0x%04x)\n",
84 tlli);
85 msgb_nsei(msg) = nsei;
86 msgb_bvci(msg) = 0; /* Signalling */
87 bgph->pdu_type = BSSGP_PDUT_RESUME;
88
89 bssgp_msgb_tlli_put(msg, tlli);
Maxf1ad60e2018-01-05 14:19:33 +010090 bssgp_msgb_ra_put(msg, ra_id);
Harald Welte85fc3142011-11-25 08:58:40 +010091
92 msgb_tvlv_put(msg, BSSGP_IE_SUSPEND_REF_NR, 1, &suspend_ref);
93
94 return gprs_ns_sendmsg(bssgp_nsi, msg);
95}
96
Neels Hofmeyr87e45502017-06-20 00:17:59 +020097/*! Transmit RA-CAPABILITY-UPDATE (10.3.3) */
Harald Welte85fc3142011-11-25 08:58:40 +010098int bssgp_tx_ra_capa_upd(struct bssgp_bvc_ctx *bctx, uint32_t tlli, uint8_t tag)
99{
100 struct msgb *msg = bssgp_msgb_alloc();
101 struct bssgp_normal_hdr *bgph =
102 (struct bssgp_normal_hdr *) msgb_put(msg, sizeof(*bgph));
103
104 LOGP(DBSSGP, LOGL_NOTICE, "BSSGP (BVCI=%u) Tx RA-CAPA-UPD (TLLI=0x%04x)\n",
105 bctx->bvci, tlli);
106
107 /* set NSEI and BVCI in msgb cb */
108 msgb_nsei(msg) = bctx->nsei;
109 msgb_bvci(msg) = bctx->bvci;
110
111 bgph->pdu_type = BSSGP_PDUT_RA_CAPA_UDPATE;
112 bssgp_msgb_tlli_put(msg, tlli);
113
114 msgb_tvlv_put(msg, BSSGP_IE_TAG, 1, &tag);
115
116 return gprs_ns_sendmsg(bssgp_nsi, msg);
117}
118
119/* first common part of RADIO-STATUS */
120static struct msgb *common_tx_radio_status(struct bssgp_bvc_ctx *bctx)
121{
122 struct msgb *msg = bssgp_msgb_alloc();
123 struct bssgp_normal_hdr *bgph =
124 (struct bssgp_normal_hdr *) msgb_put(msg, sizeof(*bgph));
125
126 LOGP(DBSSGP, LOGL_NOTICE, "BSSGP (BVCI=%u) Tx RADIO-STATUS ",
127 bctx->bvci);
128
129 /* set NSEI and BVCI in msgb cb */
130 msgb_nsei(msg) = bctx->nsei;
131 msgb_bvci(msg) = bctx->bvci;
132
133 bgph->pdu_type = BSSGP_PDUT_RADIO_STATUS;
134
135 return msg;
136}
137
138/* second common part of RADIO-STATUS */
139static int common_tx_radio_status2(struct msgb *msg, uint8_t cause)
140{
141 msgb_tvlv_put(msg, BSSGP_IE_CAUSE, 1, &cause);
Max2c34ab42016-03-17 15:42:26 +0100142 LOGPC(DBSSGP, LOGL_NOTICE, "CAUSE=%s\n", bssgp_cause_str(cause));
Harald Welte85fc3142011-11-25 08:58:40 +0100143
144 return gprs_ns_sendmsg(bssgp_nsi, msg);
145}
146
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200147/*! Transmit RADIO-STATUS for TLLI (10.3.5) */
Harald Welte85fc3142011-11-25 08:58:40 +0100148int bssgp_tx_radio_status_tlli(struct bssgp_bvc_ctx *bctx, uint8_t cause,
149 uint32_t tlli)
150{
151 struct msgb *msg = common_tx_radio_status(bctx);
152
153 if (!msg)
154 return -ENOMEM;
155 bssgp_msgb_tlli_put(msg, tlli);
156 LOGPC(DBSSGP, LOGL_NOTICE, "TLLI=0x%08x ", tlli);
157
158 return common_tx_radio_status2(msg, cause);
159}
160
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200161/*! Transmit RADIO-STATUS for TMSI (10.3.5) */
Harald Welte85fc3142011-11-25 08:58:40 +0100162int bssgp_tx_radio_status_tmsi(struct bssgp_bvc_ctx *bctx, uint8_t cause,
163 uint32_t tmsi)
164{
165 struct msgb *msg = common_tx_radio_status(bctx);
Harald Weltebfe62e52017-05-15 12:48:30 +0200166 uint32_t _tmsi = osmo_htonl(tmsi);
Harald Welte85fc3142011-11-25 08:58:40 +0100167
168 if (!msg)
169 return -ENOMEM;
170 msgb_tvlv_put(msg, BSSGP_IE_TMSI, 4, (uint8_t *)&_tmsi);
171 LOGPC(DBSSGP, LOGL_NOTICE, "TMSI=0x%08x ", tmsi);
172
173 return common_tx_radio_status2(msg, cause);
174}
175
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200176/*! Transmit RADIO-STATUS for IMSI (10.3.5) */
Harald Welte85fc3142011-11-25 08:58:40 +0100177int bssgp_tx_radio_status_imsi(struct bssgp_bvc_ctx *bctx, uint8_t cause,
178 const char *imsi)
179{
180 struct msgb *msg = common_tx_radio_status(bctx);
Harald Welte1c3bae12019-01-20 10:37:49 +0100181 uint8_t mi[GSM48_MID_MAX_SIZE];
Harald Welte85fc3142011-11-25 08:58:40 +0100182 int imsi_len = gsm48_generate_mid_from_imsi(mi, imsi);
183
184 if (!msg)
185 return -ENOMEM;
Harald Welte2033be82019-01-20 13:45:31 +0100186/* gsm48_generate_mid_from_imsi() is guaranteed to never return more than 11,
187 * but somehow gcc (8.2) is not smart enough to figure this out and claims that
188 * the memcpy in msgb_tvlv_put() below will cause and out-of-bounds access up to
189 * mi[131], which is wrong */
190#pragma GCC diagnostic push
191#pragma GCC diagnostic ignored "-Warray-bounds"
Harald Welte85fc3142011-11-25 08:58:40 +0100192 /* strip the MI type and length values (2 bytes) */
193 if (imsi_len > 2)
194 msgb_tvlv_put(msg, BSSGP_IE_IMSI, imsi_len-2, mi+2);
Harald Welte2033be82019-01-20 13:45:31 +0100195#pragma GCC diagnostic pop
Harald Welte85fc3142011-11-25 08:58:40 +0100196 LOGPC(DBSSGP, LOGL_NOTICE, "IMSI=%s ", imsi);
197
198 return common_tx_radio_status2(msg, cause);
199}
200
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200201/*! Transmit FLUSH-LL-ACK (Chapter 10.4.2) */
Harald Welte85fc3142011-11-25 08:58:40 +0100202int bssgp_tx_flush_ll_ack(struct bssgp_bvc_ctx *bctx, uint32_t tlli,
203 uint8_t action, uint16_t bvci_new,
204 uint32_t num_octets)
205{
206 struct msgb *msg = bssgp_msgb_alloc();
207 struct bssgp_normal_hdr *bgph =
208 (struct bssgp_normal_hdr *) msgb_put(msg, sizeof(*bgph));
Harald Weltebfe62e52017-05-15 12:48:30 +0200209 uint16_t _bvci_new = osmo_htons(bvci_new);
210 uint32_t _oct_aff = osmo_htonl(num_octets & 0xFFFFFF);
Harald Welte85fc3142011-11-25 08:58:40 +0100211
212 msgb_nsei(msg) = bctx->nsei;
213 msgb_bvci(msg) = 0; /* Signalling */
214 bgph->pdu_type = BSSGP_PDUT_FLUSH_LL_ACK;
215
216 bssgp_msgb_tlli_put(msg, tlli);
217 msgb_tvlv_put(msg, BSSGP_IE_FLUSH_ACTION, 1, &action);
218 if (action == 1) /* transferred */
219 msgb_tvlv_put(msg, BSSGP_IE_BVCI, 2, (uint8_t *) &_bvci_new);
220 msgb_tvlv_put(msg, BSSGP_IE_NUM_OCT_AFF, 3, (uint8_t *) &_oct_aff);
221
222 return gprs_ns_sendmsg(bssgp_nsi, msg);
223}
224
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200225/*! Transmit LLC-DISCARDED (Chapter 10.4.3) */
Harald Welte85fc3142011-11-25 08:58:40 +0100226int bssgp_tx_llc_discarded(struct bssgp_bvc_ctx *bctx, uint32_t tlli,
227 uint8_t num_frames, uint32_t num_octets)
228{
229 struct msgb *msg = bssgp_msgb_alloc();
230 struct bssgp_normal_hdr *bgph =
231 (struct bssgp_normal_hdr *) msgb_put(msg, sizeof(*bgph));
Harald Weltebfe62e52017-05-15 12:48:30 +0200232 uint16_t _bvci = osmo_htons(bctx->bvci);
233 uint32_t _oct_aff = osmo_htonl(num_octets & 0xFFFFFF);
Harald Welte85fc3142011-11-25 08:58:40 +0100234
235 LOGP(DBSSGP, LOGL_NOTICE, "BSSGP (BVCI=%u) Tx LLC-DISCARDED "
236 "TLLI=0x%04x, FRAMES=%u, OCTETS=%u\n", bctx->bvci, tlli,
237 num_frames, num_octets);
238 msgb_nsei(msg) = bctx->nsei;
239 msgb_bvci(msg) = 0; /* Signalling */
240 bgph->pdu_type = BSSGP_PDUT_LLC_DISCARD;
241
242 bssgp_msgb_tlli_put(msg, tlli);
243
244 msgb_tvlv_put(msg, BSSGP_IE_LLC_FRAMES_DISCARDED, 1, &num_frames);
245 msgb_tvlv_put(msg, BSSGP_IE_BVCI, 2, (uint8_t *) &_bvci);
Andreas Eversbergaa5d0e82012-07-21 13:33:39 +0200246 msgb_tvlv_put(msg, BSSGP_IE_NUM_OCT_AFF, 3, ((uint8_t *) &_oct_aff) + 1);
Harald Welte85fc3142011-11-25 08:58:40 +0100247
248 return gprs_ns_sendmsg(bssgp_nsi, msg);
249}
250
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200251/*! Transmit a BVC-BLOCK message (Chapter 10.4.8) */
Harald Welte85fc3142011-11-25 08:58:40 +0100252int bssgp_tx_bvc_block(struct bssgp_bvc_ctx *bctx, uint8_t cause)
253{
254 struct msgb *msg = bssgp_msgb_alloc();
255 struct bssgp_normal_hdr *bgph =
256 (struct bssgp_normal_hdr *) msgb_put(msg, sizeof(*bgph));
Harald Weltebfe62e52017-05-15 12:48:30 +0200257 uint16_t _bvci = osmo_htons(bctx->bvci);
Harald Welte85fc3142011-11-25 08:58:40 +0100258
259 LOGP(DBSSGP, LOGL_NOTICE, "BSSGP (BVCI=%u) Tx BVC-BLOCK "
Max2c34ab42016-03-17 15:42:26 +0100260 "CAUSE=%s\n", bctx->bvci, bssgp_cause_str(cause));
Harald Welte85fc3142011-11-25 08:58:40 +0100261
262 msgb_nsei(msg) = bctx->nsei;
263 msgb_bvci(msg) = 0; /* Signalling */
264 bgph->pdu_type = BSSGP_PDUT_BVC_BLOCK;
265
266 msgb_tvlv_put(msg, BSSGP_IE_BVCI, 2, (uint8_t *) &_bvci);
267 msgb_tvlv_put(msg, BSSGP_IE_CAUSE, 1, &cause);
268
269 return gprs_ns_sendmsg(bssgp_nsi, msg);
270}
271
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200272/*! Transmit a BVC-UNBLOCK message (Chapter 10.4.10) */
Harald Welte85fc3142011-11-25 08:58:40 +0100273int bssgp_tx_bvc_unblock(struct bssgp_bvc_ctx *bctx)
274{
275 struct msgb *msg = bssgp_msgb_alloc();
276 struct bssgp_normal_hdr *bgph =
277 (struct bssgp_normal_hdr *) msgb_put(msg, sizeof(*bgph));
Harald Weltebfe62e52017-05-15 12:48:30 +0200278 uint16_t _bvci = osmo_htons(bctx->bvci);
Harald Welte85fc3142011-11-25 08:58:40 +0100279
Harald Welte6a469e12019-02-19 11:07:50 +0100280 LOGP(DBSSGP, LOGL_NOTICE, "BSSGP (BVCI=%u) Tx BVC-UNBLOCK\n", bctx->bvci);
Harald Welte85fc3142011-11-25 08:58:40 +0100281
282 msgb_nsei(msg) = bctx->nsei;
283 msgb_bvci(msg) = 0; /* Signalling */
284 bgph->pdu_type = BSSGP_PDUT_BVC_UNBLOCK;
285
286 msgb_tvlv_put(msg, BSSGP_IE_BVCI, 2, (uint8_t *) &_bvci);
287
288 return gprs_ns_sendmsg(bssgp_nsi, msg);
289}
290
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200291/*! Transmit a BVC-RESET message (Chapter 10.4.12) */
Harald Welte85fc3142011-11-25 08:58:40 +0100292int bssgp_tx_bvc_reset(struct bssgp_bvc_ctx *bctx, uint16_t bvci, uint8_t cause)
293{
294 struct msgb *msg = bssgp_msgb_alloc();
295 struct bssgp_normal_hdr *bgph =
296 (struct bssgp_normal_hdr *) msgb_put(msg, sizeof(*bgph));
Harald Weltebfe62e52017-05-15 12:48:30 +0200297 uint16_t _bvci = osmo_htons(bvci);
Harald Welte85fc3142011-11-25 08:58:40 +0100298
299 LOGP(DBSSGP, LOGL_NOTICE, "BSSGP (BVCI=%u) Tx BVC-RESET "
Max2c34ab42016-03-17 15:42:26 +0100300 "CAUSE=%s\n", bvci, bssgp_cause_str(cause));
Harald Welte85fc3142011-11-25 08:58:40 +0100301
302 msgb_nsei(msg) = bctx->nsei;
303 msgb_bvci(msg) = 0; /* Signalling */
304 bgph->pdu_type = BSSGP_PDUT_BVC_RESET;
305
306 msgb_tvlv_put(msg, BSSGP_IE_BVCI, 2, (uint8_t *) &_bvci);
307 msgb_tvlv_put(msg, BSSGP_IE_CAUSE, 1, &cause);
308 if (bvci != BVCI_PTM) {
309 uint8_t bssgp_cid[8];
310 bssgp_create_cell_id(bssgp_cid, &bctx->ra_id, bctx->cell_id);
311 msgb_tvlv_put(msg, BSSGP_IE_CELL_ID, sizeof(bssgp_cid), bssgp_cid);
312 }
313 /* Optional: Feature Bitmap */
314
315 return gprs_ns_sendmsg(bssgp_nsi, msg);
316}
317
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200318/*! Transmit a FLOW_CONTROL-BVC (Chapter 10.4.4)
Harald Weltee92866b2012-09-10 08:56:04 +0200319 * \param[in] bctx BVC Context
Andreas Eversbergb8d18f32012-09-30 16:59:33 +0200320 * \param[in] tag Additional tag to identify acknowledge
Harald Weltee92866b2012-09-10 08:56:04 +0200321 * \param[in] bucket_size Maximum bucket size in octets
322 * \param[in] bucket_leak_rate Bucket leak rate in octets/sec
323 * \param[in] bmax_default_ms Maximum bucket size default for MS
324 * \param[in] r_default_ms Bucket leak rate default for MS in octets/sec
325 * \param[in] bucket_full_ratio Ratio (in percent) of queue filling
326 * \param[in] queue_delay_ms Average queuing delay in milliseconds
327 */
Andreas Eversbergf44ed8c2012-09-23 06:05:20 +0200328int bssgp_tx_fc_bvc(struct bssgp_bvc_ctx *bctx, uint8_t tag,
Harald Weltee92866b2012-09-10 08:56:04 +0200329 uint32_t bucket_size, uint32_t bucket_leak_rate,
Jacob Erlbeck9385d1e2015-05-06 09:29:32 +0200330 uint32_t bmax_default_ms, uint32_t r_default_ms,
Harald Weltee92866b2012-09-10 08:56:04 +0200331 uint8_t *bucket_full_ratio, uint32_t *queue_delay_ms)
332{
333 struct msgb *msg;
334 struct bssgp_normal_hdr *bgph;
Andreas Eversbergf44ed8c2012-09-23 06:05:20 +0200335 uint16_t e_bucket_size, e_leak_rate, e_bmax_default_ms, e_r_default_ms;
Harald Weltee92866b2012-09-10 08:56:04 +0200336 uint16_t e_queue_delay = 0; /* to make gcc happy */
337
338 if ((bucket_size / 100) > 0xffff)
339 return -EINVAL;
Harald Weltebfe62e52017-05-15 12:48:30 +0200340 e_bucket_size = osmo_htons(bucket_size / 100);
Harald Weltee92866b2012-09-10 08:56:04 +0200341
342 if ((bucket_leak_rate * 8 / 100) > 0xffff)
343 return -EINVAL;
Harald Weltebfe62e52017-05-15 12:48:30 +0200344 e_leak_rate = osmo_htons((bucket_leak_rate * 8) / 100);
Harald Weltee92866b2012-09-10 08:56:04 +0200345
Andreas Eversbergf44ed8c2012-09-23 06:05:20 +0200346 if ((bmax_default_ms / 100) > 0xffff)
347 return -EINVAL;
Harald Weltebfe62e52017-05-15 12:48:30 +0200348 e_bmax_default_ms = osmo_htons(bmax_default_ms / 100);
Andreas Eversbergf44ed8c2012-09-23 06:05:20 +0200349
Harald Weltee92866b2012-09-10 08:56:04 +0200350 if ((r_default_ms * 8 / 100) > 0xffff)
351 return -EINVAL;
Harald Weltebfe62e52017-05-15 12:48:30 +0200352 e_r_default_ms = osmo_htons((r_default_ms * 8) / 100);
Harald Weltee92866b2012-09-10 08:56:04 +0200353
354 if (queue_delay_ms) {
355 if ((*queue_delay_ms / 10) > 60000)
356 return -EINVAL;
357 else if (*queue_delay_ms == 0xFFFFFFFF)
358 e_queue_delay = 0xFFFF;
359 else
Harald Weltebfe62e52017-05-15 12:48:30 +0200360 e_queue_delay = osmo_htons(*queue_delay_ms / 10);
Harald Weltee92866b2012-09-10 08:56:04 +0200361 }
362
363 msg = bssgp_msgb_alloc();
364 bgph = (struct bssgp_normal_hdr *) msgb_put(msg, sizeof(*bgph));
365 msgb_nsei(msg) = bctx->nsei;
Andreas Eversbergf44ed8c2012-09-23 06:05:20 +0200366 msgb_bvci(msg) = bctx->bvci;
Harald Weltee92866b2012-09-10 08:56:04 +0200367 bgph->pdu_type = BSSGP_PDUT_FLOW_CONTROL_BVC;
368
369 msgb_tvlv_put(msg, BSSGP_IE_TAG, sizeof(tag), (uint8_t *)&tag);
370 msgb_tvlv_put(msg, BSSGP_IE_BVC_BUCKET_SIZE,
Andreas Eversbergf44ed8c2012-09-23 06:05:20 +0200371 sizeof(e_bucket_size), (uint8_t *) &e_bucket_size);
Harald Weltee92866b2012-09-10 08:56:04 +0200372 msgb_tvlv_put(msg, BSSGP_IE_BUCKET_LEAK_RATE,
373 sizeof(e_leak_rate), (uint8_t *) &e_leak_rate);
374 msgb_tvlv_put(msg, BSSGP_IE_BMAX_DEFAULT_MS,
Andreas Eversbergf44ed8c2012-09-23 06:05:20 +0200375 sizeof(e_bmax_default_ms),
376 (uint8_t *) &e_bmax_default_ms);
Harald Weltee92866b2012-09-10 08:56:04 +0200377 msgb_tvlv_put(msg, BSSGP_IE_R_DEFAULT_MS,
378 sizeof(e_r_default_ms), (uint8_t *) &e_r_default_ms);
379 if (bucket_full_ratio)
380 msgb_tvlv_put(msg, BSSGP_IE_BUCKET_FULL_RATIO,
381 1, bucket_full_ratio);
382 if (queue_delay_ms)
383 msgb_tvlv_put(msg, BSSGP_IE_BVC_MEASUREMENT,
384 sizeof(e_queue_delay),
385 (uint8_t *) &e_queue_delay);
386
387 return gprs_ns_sendmsg(bssgp_nsi, msg);
388}
389
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200390/*! Transmit a FLOW_CONTROL-MS (Chapter 10.4.6)
Harald Weltee92866b2012-09-10 08:56:04 +0200391 * \param[in] bctx BVC Context
Andreas Eversbergb8d18f32012-09-30 16:59:33 +0200392 * \param[in] tlli TLLI to identify MS
393 * \param[in] tag Additional tag to identify acknowledge
Harald Weltee92866b2012-09-10 08:56:04 +0200394 * \param[in] ms_bucket_size Maximum bucket size in octets
395 * \param[in] bucket_leak_rate Bucket leak rate in octets/sec
396 * \param[in] bucket_full_ratio Ratio (in percent) of queue filling
397 */
Andreas Eversbergf44ed8c2012-09-23 06:05:20 +0200398int bssgp_tx_fc_ms(struct bssgp_bvc_ctx *bctx, uint32_t tlli, uint8_t tag,
Harald Weltee92866b2012-09-10 08:56:04 +0200399 uint32_t ms_bucket_size, uint32_t bucket_leak_rate,
400 uint8_t *bucket_full_ratio)
401{
402 struct msgb *msg;
403 struct bssgp_normal_hdr *bgph;
404 uint16_t e_bucket_size, e_leak_rate;
405
406 if ((ms_bucket_size / 100) > 0xffff)
407 return -EINVAL;
408 e_bucket_size = ms_bucket_size / 100;
409
410 if ((bucket_leak_rate * 8 / 100) > 0xffff)
411 return -EINVAL;
412 e_leak_rate = (bucket_leak_rate * 8) / 100;
413
414 msg = bssgp_msgb_alloc();
415 bgph = (struct bssgp_normal_hdr *) msgb_put(msg, sizeof(*bgph));
416 msgb_nsei(msg) = bctx->nsei;
Andreas Eversbergf44ed8c2012-09-23 06:05:20 +0200417 msgb_bvci(msg) = bctx->bvci;
Harald Weltee92866b2012-09-10 08:56:04 +0200418 bgph->pdu_type = BSSGP_PDUT_FLOW_CONTROL_MS;
419
Maxe29ec852018-01-05 14:30:22 +0100420 bssgp_msgb_tlli_put(msg, tlli);
Harald Weltee92866b2012-09-10 08:56:04 +0200421 msgb_tvlv_put(msg, BSSGP_IE_TAG, sizeof(tag), (uint8_t *)&tag);
422 msgb_tvlv_put(msg, BSSGP_IE_MS_BUCKET_SIZE,
423 sizeof(e_bucket_size), (uint8_t *) &e_bucket_size);
424 msgb_tvlv_put(msg, BSSGP_IE_BUCKET_LEAK_RATE,
425 sizeof(e_leak_rate), (uint8_t *) &e_leak_rate);
426 if (bucket_full_ratio)
427 msgb_tvlv_put(msg, BSSGP_IE_BUCKET_FULL_RATIO,
428 1, bucket_full_ratio);
429
430 return gprs_ns_sendmsg(bssgp_nsi, msg);
431}
Harald Welte85fc3142011-11-25 08:58:40 +0100432
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200433/*! RL-UL-UNITDATA.req (Chapter 10.2.2)
Andreas Eversbergb8d18f32012-09-30 16:59:33 +0200434 * \param[in] bctx BVC Context
435 * \param[in] tlli TLLI to identify MS
436 * \param[in] qos_profile Pointer to three octests of QoS profile
437 * \param[in] llc_pdu msgb pointer containing UL Unitdata IE payload
438 */
Harald Welte85fc3142011-11-25 08:58:40 +0100439int bssgp_tx_ul_ud(struct bssgp_bvc_ctx *bctx, uint32_t tlli,
440 const uint8_t *qos_profile, struct msgb *llc_pdu)
441{
442 struct msgb *msg = llc_pdu;
443 uint8_t bssgp_cid[8];
Andreas Eversbergf44ed8c2012-09-23 06:05:20 +0200444 uint8_t bssgp_align[3] = {0, 0, 0};
Harald Welte85fc3142011-11-25 08:58:40 +0100445 struct bssgp_ud_hdr *budh;
Andreas Eversbergf44ed8c2012-09-23 06:05:20 +0200446 int align = sizeof(*budh);
Harald Welte85fc3142011-11-25 08:58:40 +0100447
448 /* FIXME: Optional LSA Identifier List, PFI */
449
450 /* Cell Identifier */
451 bssgp_create_cell_id(bssgp_cid, &bctx->ra_id, bctx->cell_id);
Andreas Eversbergf44ed8c2012-09-23 06:05:20 +0200452 align += 2; /* add T+L */
453 align += sizeof(bssgp_cid);
454
455 /* First push alignment IE */
456 align += 2; /* add T+L */
457 align = (4 - align) & 3; /* how many octest are required to align? */
458 msgb_tvlv_push(msg, BSSGP_IE_ALIGNMENT, align, bssgp_align);
459
460 /* Push other IEs */
Harald Welte85fc3142011-11-25 08:58:40 +0100461 msgb_tvlv_push(msg, BSSGP_IE_CELL_ID, sizeof(bssgp_cid), bssgp_cid);
462
463 /* User Data Header */
464 budh = (struct bssgp_ud_hdr *) msgb_push(msg, sizeof(*budh));
Harald Weltebfe62e52017-05-15 12:48:30 +0200465 budh->tlli = osmo_htonl(tlli);
Harald Welte85fc3142011-11-25 08:58:40 +0100466 memcpy(budh->qos_profile, qos_profile, 3);
467 budh->pdu_type = BSSGP_PDUT_UL_UNITDATA;
468
469 /* set NSEI and BVCI in msgb cb */
470 msgb_nsei(msg) = bctx->nsei;
471 msgb_bvci(msg) = bctx->bvci;
472
473 rate_ctr_inc(&bctx->ctrg->ctr[BSSGP_CTR_PKTS_OUT]);
474 rate_ctr_add(&bctx->ctrg->ctr[BSSGP_CTR_BYTES_OUT], msg->len);
475
476 return gprs_ns_sendmsg(bssgp_nsi, msg);
477}
478
479/* Parse a single GMM-PAGING.req to a given NSEI/NS-BVCI */
Harald Weltede4599c2012-06-17 13:04:02 +0800480int bssgp_rx_paging(struct bssgp_paging_info *pinfo,
481 struct msgb *msg)
Harald Welte85fc3142011-11-25 08:58:40 +0100482{
483 struct bssgp_normal_hdr *bgph =
484 (struct bssgp_normal_hdr *) msgb_bssgph(msg);
485 struct tlv_parsed tp;
486 uint8_t ra[6];
487 int rc, data_len;
488
489 memset(ra, 0, sizeof(ra));
490
491 data_len = msgb_bssgp_len(msg) - sizeof(*bgph);
492 rc = bssgp_tlv_parse(&tp, bgph->data, data_len);
493 if (rc < 0)
494 goto err_mand_ie;
495
496 switch (bgph->pdu_type) {
497 case BSSGP_PDUT_PAGING_PS:
498 pinfo->mode = BSSGP_PAGING_PS;
499 break;
500 case BSSGP_PDUT_PAGING_CS:
501 pinfo->mode = BSSGP_PAGING_CS;
502 break;
503 default:
504 return -EINVAL;
505 }
506
507 /* IMSI */
508 if (!TLVP_PRESENT(&tp, BSSGP_IE_IMSI))
509 goto err_mand_ie;
510 if (!pinfo->imsi)
Jacob Erlbeckc1cb75e2015-06-18 13:21:30 +0200511 pinfo->imsi = talloc_zero_size(pinfo, GSM_IMSI_LENGTH);
512 gsm48_mi_to_string(pinfo->imsi, GSM_IMSI_LENGTH,
Harald Welte85fc3142011-11-25 08:58:40 +0100513 TLVP_VAL(&tp, BSSGP_IE_IMSI),
514 TLVP_LEN(&tp, BSSGP_IE_IMSI));
515
516 /* DRX Parameters */
517 if (!TLVP_PRESENT(&tp, BSSGP_IE_DRX_PARAMS))
518 goto err_mand_ie;
Harald Weltebfe62e52017-05-15 12:48:30 +0200519 pinfo->drx_params = tlvp_val16be(&tp, BSSGP_IE_DRX_PARAMS);
Harald Welte85fc3142011-11-25 08:58:40 +0100520
521 /* Scope */
522 if (TLVP_PRESENT(&tp, BSSGP_IE_BSS_AREA_ID)) {
523 pinfo->scope = BSSGP_PAGING_BSS_AREA;
524 } else if (TLVP_PRESENT(&tp, BSSGP_IE_LOCATION_AREA)) {
525 pinfo->scope = BSSGP_PAGING_LOCATION_AREA;
526 memcpy(ra, TLVP_VAL(&tp, BSSGP_IE_LOCATION_AREA),
527 TLVP_LEN(&tp, BSSGP_IE_LOCATION_AREA));
528 gsm48_parse_ra(&pinfo->raid, ra);
529 } else if (TLVP_PRESENT(&tp, BSSGP_IE_ROUTEING_AREA)) {
530 pinfo->scope = BSSGP_PAGING_ROUTEING_AREA;
531 memcpy(ra, TLVP_VAL(&tp, BSSGP_IE_ROUTEING_AREA),
532 TLVP_LEN(&tp, BSSGP_IE_ROUTEING_AREA));
533 gsm48_parse_ra(&pinfo->raid, ra);
534 } else if (TLVP_PRESENT(&tp, BSSGP_IE_BVCI)) {
535 pinfo->scope = BSSGP_PAGING_BVCI;
Harald Weltebfe62e52017-05-15 12:48:30 +0200536 pinfo->bvci = tlvp_val16be(&tp, BSSGP_IE_BVCI);
Harald Welte85fc3142011-11-25 08:58:40 +0100537 } else
538 return -EINVAL;
539
540 /* QoS profile mandatory for PS */
541 if (pinfo->mode == BSSGP_PAGING_PS) {
542 if (!TLVP_PRESENT(&tp, BSSGP_IE_QOS_PROFILE))
543 goto err_cond_ie;
544 if (TLVP_LEN(&tp, BSSGP_IE_QOS_PROFILE) < 3)
545 goto err;
546
547 memcpy(&pinfo->qos, TLVP_VAL(&tp, BSSGP_IE_QOS_PROFILE),
548 3);
549 }
550
551 /* Optional (P-)TMSI */
552 if (TLVP_PRESENT(&tp, BSSGP_IE_TMSI) &&
Harald Welte6176b6e2016-11-11 15:10:33 +0100553 TLVP_LEN(&tp, BSSGP_IE_TMSI) >= 4) {
Harald Welte85fc3142011-11-25 08:58:40 +0100554 if (!pinfo->ptmsi)
555 pinfo->ptmsi = talloc_zero_size(pinfo, sizeof(uint32_t));
Harald Weltebfe62e52017-05-15 12:48:30 +0200556 *(pinfo->ptmsi) = osmo_load32be(TLVP_VAL(&tp, BSSGP_IE_TMSI));
Harald Welte6176b6e2016-11-11 15:10:33 +0100557 }
Harald Welte85fc3142011-11-25 08:58:40 +0100558
559 return 0;
560
561err_mand_ie:
562err_cond_ie:
563err:
564 /* FIXME */
565 return 0;
566}