blob: ade3f3bad2b2dc5c5c5c5bfcc51b059cec3337b7 [file] [log] [blame]
Jacob Erlbecke04e0b02015-05-06 18:30:48 +02001/* gprs_ms.h
2 *
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +01003 * Copyright (C) 2015-2020 by Sysmocom s.f.m.c. GmbH
Jacob Erlbecke04e0b02015-05-06 18:30:48 +02004 * Author: Jacob Erlbeck <jerlbeck@sysmocom.de>
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
19 */
20
21#pragma once
22
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +020023struct gprs_codel;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020024
Jacob Erlbeck489a2b32015-05-28 19:07:01 +020025#include "llc.h"
Jacob Erlbeckac89a552015-06-29 14:18:46 +020026#include "tbf.h"
Pau Espin Pedrol442198c2020-10-23 22:30:04 +020027#include "tbf_ul.h"
Pau Espin Pedrol9d1cdb12019-09-25 17:47:02 +020028#include "tbf_dl.h"
Jacob Erlbecke4bcb622015-06-08 11:26:38 +020029#include "pcu_l1_if.h"
Jacob Erlbeckd9e10242015-05-28 15:43:53 +020030
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010031#ifdef __cplusplus
Jacob Erlbeckd9e10242015-05-28 15:43:53 +020032extern "C" {
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010033#endif
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +020034
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010035#include <osmocom/core/timer.h>
36#include <osmocom/core/linuxlist.h>
Pau Espin Pedrol43f0bce2020-06-26 13:09:44 +020037
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010038#include <osmocom/gsm/protocol/gsm_23_003.h>
39#include <osmocom/gsm/gsm48.h>
40
41#include "coding_scheme.h"
Jacob Erlbeckd9e10242015-05-28 15:43:53 +020042
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020043#include <stdint.h>
44#include <stddef.h>
Pau Espin Pedrolf2dad592020-08-18 20:26:25 +020045#include <inttypes.h>
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020046
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020047struct BTS;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +020048struct gprs_rlcmac_trx;
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010049struct GprsMs;
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020050
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010051struct gpr_ms_callback {
52 void (*ms_idle)(struct GprsMs *);
53 void (*ms_active)(struct GprsMs *);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020054};
Jacob Erlbeck93990462015-05-15 15:50:43 +020055
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010056struct GprsMs {
57 struct llist_head list; /* list of all GprsMs */
58 struct gpr_ms_callback cb;
59 bool app_info_pending;
60
61 struct BTS *bts;
62 struct gprs_rlcmac_ul_tbf *ul_tbf;
63 struct gprs_rlcmac_dl_tbf *dl_tbf;
64 struct llist_head old_tbfs; /* list of gprs_rlcmac_tbf */
65
66 uint32_t tlli;
67 uint32_t new_ul_tlli;
68 uint32_t new_dl_tlli;
69
70 /* store IMSI for look-up and PCH retransmission */
71 char imsi[OSMO_IMSI_BUF_SIZE];
72 uint8_t ta;
73 uint8_t ms_class;
74 uint8_t egprs_ms_class;
75 /* current coding scheme */
76 enum CodingScheme current_cs_ul;
77 enum CodingScheme current_cs_dl;
78
79 struct gprs_llc_queue llc_queue;
80
81 bool is_idle;
82 int ref;
83 struct osmo_timer_list timer;
84 unsigned delay;
85
86 int64_t last_cs_not_low;
87
88 struct pcu_l1_meas l1_meas;
89 unsigned nack_rate_dl;
90 uint8_t reserved_dl_slots;
91 uint8_t reserved_ul_slots;
92 struct gprs_rlcmac_trx *current_trx;
93
94 struct gprs_codel *codel_state;
95 enum mcs_kind mode;
96
97 unsigned dl_ctrl_msg;
98};
99
100struct GprsMs *ms_alloc(struct BTS *bts, uint32_t tlli);
101
102int ms_first_common_ts(const struct GprsMs *ms);
103void ms_set_reserved_slots(struct GprsMs *ms, struct gprs_rlcmac_trx *trx,
104 uint8_t ul_slots, uint8_t dl_slots);
105struct GprsMs *ms_ref(struct GprsMs *ms);
106void ms_unref(struct GprsMs *ms);
107void ms_set_mode(struct GprsMs *ms, enum mcs_kind mode);
108void ms_set_ms_class(struct GprsMs *ms, uint8_t ms_class_);
109void ms_set_egprs_ms_class(struct GprsMs *ms, uint8_t ms_class_);
110void ms_set_ta(struct GprsMs *ms, uint8_t ta_);
111
112enum CodingScheme ms_current_cs_dl(const struct GprsMs *ms);
113enum CodingScheme ms_max_cs_ul(const struct GprsMs *ms);
114enum CodingScheme ms_max_cs_dl(const struct GprsMs *ms);
115void ms_set_current_cs_dl(struct GprsMs *ms, enum CodingScheme scheme);
116
117void ms_update_error_rate(struct GprsMs *ms, struct gprs_rlcmac_tbf *tbf, int error_rate);
118uint8_t ms_current_pacch_slots(const struct GprsMs *ms);
119
120void ms_merge_and_clear_ms(struct GprsMs *ms, struct GprsMs *old_ms);
121
122void ms_attach_tbf(struct GprsMs *ms, struct gprs_rlcmac_tbf *tbf);
123void ms_detach_tbf(struct GprsMs *ms, struct gprs_rlcmac_tbf *tbf);
124
125void ms_set_tlli(struct GprsMs *ms, uint32_t tlli);
126bool ms_confirm_tlli(struct GprsMs *ms, uint32_t tlli);
127void ms_set_imsi(struct GprsMs *ms, const char *imsi);
128
129void ms_update_l1_meas(struct GprsMs *ms, const struct pcu_l1_meas *meas);
130
131struct gprs_rlcmac_tbf *ms_tbf(const struct GprsMs *ms, enum gprs_rlcmac_tbf_direction dir);
132static inline struct gprs_rlcmac_ul_tbf *ms_ul_tbf(const struct GprsMs *ms) {return ms->ul_tbf;}
133static inline struct gprs_rlcmac_dl_tbf *ms_dl_tbf(const struct GprsMs *ms) {return ms->dl_tbf;}
134
135
136void ms_set_callback(struct GprsMs *ms, struct gpr_ms_callback *cb);
137
138static inline bool ms_is_idle(const struct GprsMs *ms)
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200139{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100140 return !ms->ul_tbf && !ms->dl_tbf && !ms->ref && llist_empty(&ms->old_tbfs);
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200141}
142
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100143static inline struct gprs_llc_queue *ms_llc_queue(struct GprsMs *ms)
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200144{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100145 return &ms->llc_queue;
146}
147
148static inline bool ms_need_dl_tbf(struct GprsMs *ms)
149{
150 if (ms_dl_tbf(ms) != NULL &&
151 tbf_state((const struct gprs_rlcmac_tbf *)ms_dl_tbf(ms)) != GPRS_RLCMAC_WAIT_RELEASE)
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200152 return false;
153
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100154 return llc_queue_size(ms_llc_queue(ms)) > 0;
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200155}
156
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100157static inline uint32_t ms_tlli(const struct GprsMs *ms)
Jacob Erlbeck93990462015-05-15 15:50:43 +0200158{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100159 if (ms->new_ul_tlli != GSM_RESERVED_TMSI)
160 return ms->new_ul_tlli;
161 if (ms->tlli != GSM_RESERVED_TMSI)
162 return ms->tlli;
Vadim Yanitskiycb988942020-11-08 13:27:35 +0700163
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100164 return ms->new_dl_tlli;
Jacob Erlbeck93990462015-05-15 15:50:43 +0200165}
166
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100167static inline bool ms_check_tlli(struct GprsMs *ms, uint32_t tlli)
Jacob Erlbeck93990462015-05-15 15:50:43 +0200168{
Vadim Yanitskiycb988942020-11-08 13:27:35 +0700169 return tlli != GSM_RESERVED_TMSI &&
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100170 (tlli == ms->tlli || tlli == ms->new_ul_tlli || tlli == ms->new_dl_tlli);
Jacob Erlbeck93990462015-05-15 15:50:43 +0200171}
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200172
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100173static inline const char *ms_imsi(const struct GprsMs *ms)
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200174{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100175 return ms->imsi;
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200176}
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200177
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100178static inline uint8_t ms_ta(const struct GprsMs *ms)
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200179{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100180 return ms->ta;
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200181}
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200182
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100183static inline uint8_t ms_ms_class(const struct GprsMs *ms)
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200184{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100185 return ms->ms_class;
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200186}
187
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100188static inline uint8_t ms_egprs_ms_class(const struct GprsMs *ms)
Jacob Erlbeckc3c58042015-09-28 17:55:32 +0200189{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100190 return ms->egprs_ms_class;
Jacob Erlbeckc3c58042015-09-28 17:55:32 +0200191}
192
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100193static inline enum CodingScheme ms_current_cs_ul(const struct GprsMs *ms)
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200194{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100195 return ms->current_cs_ul;
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200196}
197
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100198static inline enum mcs_kind ms_mode(const struct GprsMs *ms)
Jacob Erlbeckcb728902016-01-05 15:33:03 +0100199{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100200 return ms->mode;
Jacob Erlbeckcb728902016-01-05 15:33:03 +0100201}
202
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100203static inline void ms_set_timeout(struct GprsMs *ms, unsigned secs)
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200204{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100205 ms->delay = secs;
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200206}
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200207
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100208static inline struct gprs_codel *ms_codel_state(const struct GprsMs *ms)
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200209{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100210 return ms->codel_state;
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200211}
212
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100213static inline unsigned ms_nack_rate_dl(const struct GprsMs *ms)
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200214{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100215 return ms->nack_rate_dl;
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200216}
217
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100218static inline unsigned ms_dl_ctrl_msg(const struct GprsMs *ms)
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200219{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100220 return ms->dl_ctrl_msg;
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200221}
222
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100223static inline void ms_update_dl_ctrl_msg(struct GprsMs *ms)
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200224{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100225 ms->dl_ctrl_msg++;
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200226}
227
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100228static inline uint8_t ms_reserved_dl_slots(const struct GprsMs *ms)
sivasankarida7250a2016-12-16 12:57:18 +0530229{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100230 return ms->reserved_dl_slots;
sivasankarida7250a2016-12-16 12:57:18 +0530231}
232
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100233static inline uint8_t ms_reserved_ul_slots(const struct GprsMs *ms)
sivasankarida7250a2016-12-16 12:57:18 +0530234{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100235 return ms->reserved_ul_slots;
sivasankarida7250a2016-12-16 12:57:18 +0530236}
237
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100238static inline struct gprs_rlcmac_trx *ms_current_trx(const struct GprsMs *ms)
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200239{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100240 return ms->current_trx;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200241}
Pau Espin Pedrolf2dad592020-08-18 20:26:25 +0200242
243#define LOGPMS(ms, category, level, fmt, args...) \
244 LOGP(category, level, "MS(TLLI=0x%08x, IMSI=%s, TA=%" PRIu8 ", %" PRIu8 "/%" PRIu8 ",%s%s) " fmt, \
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100245 ms_tlli(ms), ms_imsi(ms), ms_ta(ms), ms_ms_class(ms), ms_egprs_ms_class(ms), \
246 ms_ul_tbf(ms) ? " UL": "", \
247 ms_dl_tbf(ms) ? " DL": "", \
Pau Espin Pedrolf2dad592020-08-18 20:26:25 +0200248 ## args)
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100249
250#ifdef __cplusplus
251}
252#endif