blob: 8b8940bb36f65d853ef0e918f3248b61cd2241e7 [file] [log] [blame]
Jacob Erlbecke04e0b02015-05-06 18:30:48 +02001/* gprs_ms.h
2 *
3 * Copyright (C) 2015 by Sysmocom s.f.m.c. GmbH
4 * 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 Erlbeck53670862015-05-12 17:54:33 +020025#include "cxx_linuxlist.h"
Jacob Erlbeck489a2b32015-05-28 19:07:01 +020026#include "llc.h"
Jacob Erlbeckac89a552015-06-29 14:18:46 +020027#include "tbf.h"
Pau Espin Pedrol442198c2020-10-23 22:30:04 +020028#include "tbf_ul.h"
Pau Espin Pedrol9d1cdb12019-09-25 17:47:02 +020029#include "tbf_dl.h"
Jacob Erlbecke4bcb622015-06-08 11:26:38 +020030#include "pcu_l1_if.h"
Jacob Erlbeckd9e10242015-05-28 15:43:53 +020031
32extern "C" {
33 #include <osmocom/core/timer.h>
Max1187a772018-01-26 13:31:42 +010034 #include <osmocom/core/linuxlist.h>
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +020035
Pau Espin Pedrol43f0bce2020-06-26 13:09:44 +020036 #include <osmocom/gsm/protocol/gsm_23_003.h>
Vadim Yanitskiycb988942020-11-08 13:27:35 +070037 #include <osmocom/gsm/gsm48.h>
Pau Espin Pedrol43f0bce2020-06-26 13:09:44 +020038
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +020039 #include "coding_scheme.h"
Jacob Erlbeckd9e10242015-05-28 15:43:53 +020040}
41
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020042#include <stdint.h>
43#include <stddef.h>
Pau Espin Pedrolf2dad592020-08-18 20:26:25 +020044#include <inttypes.h>
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020045
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020046struct BTS;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +020047struct gprs_rlcmac_trx;
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020048
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020049class GprsMs {
50public:
51 struct Callback {
52 virtual void ms_idle(class GprsMs *) = 0;
53 virtual void ms_active(class GprsMs *) = 0;
54 };
55
56 class Guard {
57 public:
58 Guard(GprsMs *ms);
59 ~Guard();
60
Jacob Erlbeckb2439bb2015-07-13 14:23:32 +020061 bool is_idle() const;
62
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020063 private:
64 GprsMs * const m_ms;
65 };
66
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020067 GprsMs(BTS *bts, uint32_t tlli);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020068 ~GprsMs();
69
70 void set_callback(Callback *cb) {m_cb = cb;}
71
Pau Espin Pedrol528820d2020-10-26 12:40:11 +010072 void merge_and_clear_ms(GprsMs *old_ms);
Jacob Erlbeck2b349b52015-08-18 11:55:03 +020073
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020074 gprs_rlcmac_ul_tbf *ul_tbf() const {return m_ul_tbf;}
75 gprs_rlcmac_dl_tbf *dl_tbf() const {return m_dl_tbf;}
Jacob Erlbeckac89a552015-06-29 14:18:46 +020076 gprs_rlcmac_tbf *tbf(enum gprs_rlcmac_tbf_direction dir) const;
Jacob Erlbeck93990462015-05-15 15:50:43 +020077 uint32_t tlli() const;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020078 void set_tlli(uint32_t tlli);
Jacob Erlbeck93990462015-05-15 15:50:43 +020079 bool confirm_tlli(uint32_t tlli);
80 bool check_tlli(uint32_t tlli);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020081
Jacob Erlbeckac289052015-08-14 12:50:54 +020082 void reset();
Maxa4de02d2019-03-13 16:35:09 +010083 enum mcs_kind mode() const;
84 void set_mode(enum mcs_kind mode);
Jacob Erlbeckac289052015-08-14 12:50:54 +020085
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +020086 const char *imsi() const;
87 void set_imsi(const char *imsi);
88
Jacob Erlbeck9200ce62015-05-22 17:48:04 +020089 uint8_t ta() const;
90 void set_ta(uint8_t ta);
Jacob Erlbeckbefc7602015-06-02 12:33:30 +020091 uint8_t ms_class() const;
Jacob Erlbeckc3c58042015-09-28 17:55:32 +020092 uint8_t egprs_ms_class() const;
Jacob Erlbeckbefc7602015-06-02 12:33:30 +020093 void set_ms_class(uint8_t ms_class);
Jacob Erlbeckc3c58042015-09-28 17:55:32 +020094 void set_egprs_ms_class(uint8_t ms_class);
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +020095 void set_current_cs_dl(enum CodingScheme scheme);
Jacob Erlbeck9200ce62015-05-22 17:48:04 +020096
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +020097 enum CodingScheme current_cs_ul() const;
98 enum CodingScheme current_cs_dl() const;
99 enum CodingScheme max_cs_ul() const;
100 enum CodingScheme max_cs_dl() const;
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200101
Jacob Erlbeck699b8dc2015-06-29 14:05:55 +0200102 int first_common_ts() const;
Jacob Erlbeck617c7122015-06-30 09:18:30 +0200103 uint8_t dl_slots() const;
104 uint8_t ul_slots() const;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200105 uint8_t reserved_dl_slots() const;
106 uint8_t reserved_ul_slots() const;
Jacob Erlbeck7c72aca2016-01-22 17:06:14 +0100107 uint8_t current_pacch_slots() const;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200108 gprs_rlcmac_trx *current_trx() const;
109 void set_reserved_slots(gprs_rlcmac_trx *trx,
110 uint8_t ul_slots, uint8_t dl_slots);
Jacob Erlbeck699b8dc2015-06-29 14:05:55 +0200111
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200112 gprs_llc_queue *llc_queue();
113 const gprs_llc_queue *llc_queue() const;
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200114 gprs_codel *codel_state() const;
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200115
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200116 void set_timeout(unsigned secs);
117
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200118 void attach_tbf(gprs_rlcmac_tbf *tbf);
119 void attach_ul_tbf(gprs_rlcmac_ul_tbf *tbf);
120 void attach_dl_tbf(gprs_rlcmac_dl_tbf *tbf);
121
122 void detach_tbf(gprs_rlcmac_tbf *tbf);
123
Jacob Erlbeck1751c622015-06-04 12:12:32 +0200124 void update_error_rate(gprs_rlcmac_tbf *tbf, int percent);
125
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200126 bool is_idle() const;
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200127 bool need_dl_tbf() const;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200128
129 void* operator new(size_t num);
130 void operator delete(void* p);
131
Jacob Erlbeck53670862015-05-12 17:54:33 +0200132 LListHead<GprsMs>& list() {return this->m_list;}
133 const LListHead<GprsMs>& list() const {return this->m_list;}
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200134 const LListHead<gprs_rlcmac_tbf>& old_tbfs() const {return m_old_tbfs;}
Jacob Erlbeck53670862015-05-12 17:54:33 +0200135
Jacob Erlbecke4bcb622015-06-08 11:26:38 +0200136 void update_l1_meas(const pcu_l1_meas *meas);
137 const pcu_l1_meas* l1_meas() const {return &m_l1_meas;};
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200138 unsigned nack_rate_dl() const;
sivasankarida7250a2016-12-16 12:57:18 +0530139 unsigned dl_ctrl_msg() const;
140 void update_dl_ctrl_msg();
Jacob Erlbecke4bcb622015-06-08 11:26:38 +0200141
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200142 /* internal use */
143 static void timeout(void *priv_);
144
Oliver Smithcfb63212019-09-05 17:13:33 +0200145 bool app_info_pending;
146
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200147protected:
Pau Espin Pedrol528820d2020-10-26 12:40:11 +0100148 void merge_old_ms(GprsMs *old_ms);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200149 void update_status();
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200150 GprsMs *ref();
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200151 void unref();
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200152 void start_timer();
153 void stop_timer();
Jacob Erlbeckcb728902016-01-05 15:33:03 +0100154 void update_cs_ul(const pcu_l1_meas*);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200155
156private:
Jacob Erlbeck17214bb2015-06-02 14:06:12 +0200157 BTS *m_bts;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200158 Callback * m_cb;
159 gprs_rlcmac_ul_tbf *m_ul_tbf;
160 gprs_rlcmac_dl_tbf *m_dl_tbf;
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200161 LListHead<gprs_rlcmac_tbf> m_old_tbfs;
162
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200163 uint32_t m_tlli;
Jacob Erlbeck93990462015-05-15 15:50:43 +0200164 uint32_t m_new_ul_tlli;
165 uint32_t m_new_dl_tlli;
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200166
167 /* store IMSI for look-up and PCH retransmission */
Pau Espin Pedrol43f0bce2020-06-26 13:09:44 +0200168 char m_imsi[OSMO_IMSI_BUF_SIZE];
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200169 uint8_t m_ta;
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200170 uint8_t m_ms_class;
Jacob Erlbeckc3c58042015-09-28 17:55:32 +0200171 uint8_t m_egprs_ms_class;
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200172 /* current coding scheme */
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +0200173 enum CodingScheme m_current_cs_ul;
174 enum CodingScheme m_current_cs_dl;
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200175
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200176 gprs_llc_queue m_llc_queue;
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200177
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200178 bool m_is_idle;
179 int m_ref;
Jacob Erlbeck53670862015-05-12 17:54:33 +0200180 LListHead<GprsMs> m_list;
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200181 struct osmo_timer_list m_timer;
182 unsigned m_delay;
Jacob Erlbeck8158ea72015-06-04 17:46:33 +0200183
184 int64_t m_last_cs_not_low;
Jacob Erlbecke4bcb622015-06-08 11:26:38 +0200185
186 pcu_l1_meas m_l1_meas;
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200187 unsigned m_nack_rate_dl;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200188 uint8_t m_reserved_dl_slots;
189 uint8_t m_reserved_ul_slots;
190 gprs_rlcmac_trx *m_current_trx;
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200191
192 struct gprs_codel *m_codel_state;
Maxa4de02d2019-03-13 16:35:09 +0100193 enum mcs_kind m_mode;
sivasankarida7250a2016-12-16 12:57:18 +0530194
195 unsigned m_dl_ctrl_msg;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200196};
Jacob Erlbeck93990462015-05-15 15:50:43 +0200197
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200198inline bool GprsMs::is_idle() const
199{
200 return !m_ul_tbf && !m_dl_tbf && !m_ref && llist_empty(&m_old_tbfs);
201}
202
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200203inline bool GprsMs::need_dl_tbf() const
204{
205 if (dl_tbf() != NULL && dl_tbf()->state_is_not(GPRS_RLCMAC_WAIT_RELEASE))
206 return false;
207
208 return llc_queue()->size() > 0;
209}
210
Jacob Erlbeck93990462015-05-15 15:50:43 +0200211inline uint32_t GprsMs::tlli() const
212{
Vadim Yanitskiycb988942020-11-08 13:27:35 +0700213 if (m_new_ul_tlli != GSM_RESERVED_TMSI)
214 return m_new_ul_tlli;
215 if (m_tlli != GSM_RESERVED_TMSI)
216 return m_tlli;
217
218 return m_new_dl_tlli;
Jacob Erlbeck93990462015-05-15 15:50:43 +0200219}
220
221inline bool GprsMs::check_tlli(uint32_t tlli)
222{
Vadim Yanitskiycb988942020-11-08 13:27:35 +0700223 return tlli != GSM_RESERVED_TMSI &&
Jacob Erlbeck93990462015-05-15 15:50:43 +0200224 (tlli == m_tlli || tlli == m_new_ul_tlli || tlli == m_new_dl_tlli);
225}
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200226
227inline const char *GprsMs::imsi() const
228{
229 return m_imsi;
230}
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200231
232inline uint8_t GprsMs::ta() const
233{
234 return m_ta;
235}
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200236
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200237inline uint8_t GprsMs::ms_class() const
238{
239 return m_ms_class;
240}
241
Jacob Erlbeckc3c58042015-09-28 17:55:32 +0200242inline uint8_t GprsMs::egprs_ms_class() const
243{
244 return m_egprs_ms_class;
245}
246
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +0200247inline enum CodingScheme GprsMs::current_cs_ul() const
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200248{
249 return m_current_cs_ul;
250}
251
Maxa4de02d2019-03-13 16:35:09 +0100252inline enum mcs_kind GprsMs::mode() const
Jacob Erlbeckcb728902016-01-05 15:33:03 +0100253{
254 return m_mode;
255}
256
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200257inline void GprsMs::set_timeout(unsigned secs)
258{
259 m_delay = secs;
260}
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200261
262inline gprs_llc_queue *GprsMs::llc_queue()
263{
264 return &m_llc_queue;
265}
266
267inline const gprs_llc_queue *GprsMs::llc_queue() const
268{
269 return &m_llc_queue;
270}
271
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200272inline gprs_codel *GprsMs::codel_state() const
273{
274 return m_codel_state;
275}
276
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200277inline unsigned GprsMs::nack_rate_dl() const
278{
279 return m_nack_rate_dl;
280}
281
sivasankarida7250a2016-12-16 12:57:18 +0530282inline unsigned GprsMs::dl_ctrl_msg() const
283{
284 return m_dl_ctrl_msg;
285}
286
287inline void GprsMs::update_dl_ctrl_msg()
288{
289 m_dl_ctrl_msg++;
290}
291
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200292inline uint8_t GprsMs::reserved_dl_slots() const
293{
294 return m_reserved_dl_slots;
295}
296
297inline uint8_t GprsMs::reserved_ul_slots() const
298{
299 return m_reserved_ul_slots;
300}
301
302inline gprs_rlcmac_trx *GprsMs::current_trx() const
303{
304 return m_current_trx;
305}
Pau Espin Pedrolf2dad592020-08-18 20:26:25 +0200306
307#define LOGPMS(ms, category, level, fmt, args...) \
308 LOGP(category, level, "MS(TLLI=0x%08x, IMSI=%s, TA=%" PRIu8 ", %" PRIu8 "/%" PRIu8 ",%s%s) " fmt, \
309 (ms)->tlli(), (ms)->imsi(), (ms)->ta(), (ms)->ms_class(), (ms)->egprs_ms_class(), \
310 (ms)->ul_tbf() ? " UL": "", \
311 (ms)->dl_tbf() ? " DL": "", \
312 ## args)