blob: fbb46f6b508c1df7de4f3c7d7a3392caa1db61b4 [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>
37
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +020038 #include "coding_scheme.h"
Jacob Erlbeckd9e10242015-05-28 15:43:53 +020039}
40
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020041#include <stdint.h>
42#include <stddef.h>
Pau Espin Pedrolf2dad592020-08-18 20:26:25 +020043#include <inttypes.h>
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020044
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020045struct BTS;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +020046struct gprs_rlcmac_trx;
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020047
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020048class GprsMs {
49public:
50 struct Callback {
51 virtual void ms_idle(class GprsMs *) = 0;
52 virtual void ms_active(class GprsMs *) = 0;
53 };
54
55 class Guard {
56 public:
57 Guard(GprsMs *ms);
58 ~Guard();
59
Jacob Erlbeckb2439bb2015-07-13 14:23:32 +020060 bool is_idle() const;
61
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020062 private:
63 GprsMs * const m_ms;
64 };
65
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020066 GprsMs(BTS *bts, uint32_t tlli);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020067 ~GprsMs();
68
69 void set_callback(Callback *cb) {m_cb = cb;}
70
Pau Espin Pedrol528820d2020-10-26 12:40:11 +010071 void merge_and_clear_ms(GprsMs *old_ms);
Jacob Erlbeck2b349b52015-08-18 11:55:03 +020072
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020073 gprs_rlcmac_ul_tbf *ul_tbf() const {return m_ul_tbf;}
74 gprs_rlcmac_dl_tbf *dl_tbf() const {return m_dl_tbf;}
Jacob Erlbeckac89a552015-06-29 14:18:46 +020075 gprs_rlcmac_tbf *tbf(enum gprs_rlcmac_tbf_direction dir) const;
Jacob Erlbeck93990462015-05-15 15:50:43 +020076 uint32_t tlli() const;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020077 void set_tlli(uint32_t tlli);
Jacob Erlbeck93990462015-05-15 15:50:43 +020078 bool confirm_tlli(uint32_t tlli);
79 bool check_tlli(uint32_t tlli);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020080
Jacob Erlbeckac289052015-08-14 12:50:54 +020081 void reset();
Maxa4de02d2019-03-13 16:35:09 +010082 enum mcs_kind mode() const;
83 void set_mode(enum mcs_kind mode);
Jacob Erlbeckac289052015-08-14 12:50:54 +020084
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +020085 const char *imsi() const;
86 void set_imsi(const char *imsi);
87
Jacob Erlbeck9200ce62015-05-22 17:48:04 +020088 uint8_t ta() const;
89 void set_ta(uint8_t ta);
Jacob Erlbeckbefc7602015-06-02 12:33:30 +020090 uint8_t ms_class() const;
Jacob Erlbeckc3c58042015-09-28 17:55:32 +020091 uint8_t egprs_ms_class() const;
Jacob Erlbeckbefc7602015-06-02 12:33:30 +020092 void set_ms_class(uint8_t ms_class);
Jacob Erlbeckc3c58042015-09-28 17:55:32 +020093 void set_egprs_ms_class(uint8_t ms_class);
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +020094 void set_current_cs_dl(enum CodingScheme scheme);
Jacob Erlbeck9200ce62015-05-22 17:48:04 +020095
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +020096 enum CodingScheme current_cs_ul() const;
97 enum CodingScheme current_cs_dl() const;
98 enum CodingScheme max_cs_ul() const;
99 enum CodingScheme max_cs_dl() const;
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200100
Jacob Erlbeck699b8dc2015-06-29 14:05:55 +0200101 int first_common_ts() const;
Jacob Erlbeck617c7122015-06-30 09:18:30 +0200102 uint8_t dl_slots() const;
103 uint8_t ul_slots() const;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200104 uint8_t reserved_dl_slots() const;
105 uint8_t reserved_ul_slots() const;
Jacob Erlbeck7c72aca2016-01-22 17:06:14 +0100106 uint8_t current_pacch_slots() const;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200107 gprs_rlcmac_trx *current_trx() const;
108 void set_reserved_slots(gprs_rlcmac_trx *trx,
109 uint8_t ul_slots, uint8_t dl_slots);
Jacob Erlbeck699b8dc2015-06-29 14:05:55 +0200110
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200111 gprs_llc_queue *llc_queue();
112 const gprs_llc_queue *llc_queue() const;
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200113 gprs_codel *codel_state() const;
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200114
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200115 void set_timeout(unsigned secs);
116
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200117 void attach_tbf(gprs_rlcmac_tbf *tbf);
118 void attach_ul_tbf(gprs_rlcmac_ul_tbf *tbf);
119 void attach_dl_tbf(gprs_rlcmac_dl_tbf *tbf);
120
121 void detach_tbf(gprs_rlcmac_tbf *tbf);
122
Jacob Erlbeck1751c622015-06-04 12:12:32 +0200123 void update_error_rate(gprs_rlcmac_tbf *tbf, int percent);
124
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200125 bool is_idle() const;
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200126 bool need_dl_tbf() const;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200127
128 void* operator new(size_t num);
129 void operator delete(void* p);
130
Jacob Erlbeck53670862015-05-12 17:54:33 +0200131 LListHead<GprsMs>& list() {return this->m_list;}
132 const LListHead<GprsMs>& list() const {return this->m_list;}
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200133 const LListHead<gprs_rlcmac_tbf>& old_tbfs() const {return m_old_tbfs;}
Jacob Erlbeck53670862015-05-12 17:54:33 +0200134
Jacob Erlbecke4bcb622015-06-08 11:26:38 +0200135 void update_l1_meas(const pcu_l1_meas *meas);
136 const pcu_l1_meas* l1_meas() const {return &m_l1_meas;};
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200137 unsigned nack_rate_dl() const;
sivasankarida7250a2016-12-16 12:57:18 +0530138 unsigned dl_ctrl_msg() const;
139 void update_dl_ctrl_msg();
Jacob Erlbecke4bcb622015-06-08 11:26:38 +0200140
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200141 /* internal use */
142 static void timeout(void *priv_);
143
Oliver Smithcfb63212019-09-05 17:13:33 +0200144 bool app_info_pending;
145
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200146protected:
Pau Espin Pedrol528820d2020-10-26 12:40:11 +0100147 void merge_old_ms(GprsMs *old_ms);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200148 void update_status();
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200149 GprsMs *ref();
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200150 void unref();
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200151 void start_timer();
152 void stop_timer();
Jacob Erlbeckcb728902016-01-05 15:33:03 +0100153 void update_cs_ul(const pcu_l1_meas*);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200154
155private:
Jacob Erlbeck17214bb2015-06-02 14:06:12 +0200156 BTS *m_bts;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200157 Callback * m_cb;
158 gprs_rlcmac_ul_tbf *m_ul_tbf;
159 gprs_rlcmac_dl_tbf *m_dl_tbf;
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200160 LListHead<gprs_rlcmac_tbf> m_old_tbfs;
161
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200162 uint32_t m_tlli;
Jacob Erlbeck93990462015-05-15 15:50:43 +0200163 uint32_t m_new_ul_tlli;
164 uint32_t m_new_dl_tlli;
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200165
166 /* store IMSI for look-up and PCH retransmission */
Pau Espin Pedrol43f0bce2020-06-26 13:09:44 +0200167 char m_imsi[OSMO_IMSI_BUF_SIZE];
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200168 uint8_t m_ta;
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200169 uint8_t m_ms_class;
Jacob Erlbeckc3c58042015-09-28 17:55:32 +0200170 uint8_t m_egprs_ms_class;
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200171 /* current coding scheme */
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +0200172 enum CodingScheme m_current_cs_ul;
173 enum CodingScheme m_current_cs_dl;
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200174
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200175 gprs_llc_queue m_llc_queue;
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200176
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200177 bool m_is_idle;
178 int m_ref;
Jacob Erlbeck53670862015-05-12 17:54:33 +0200179 LListHead<GprsMs> m_list;
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200180 struct osmo_timer_list m_timer;
181 unsigned m_delay;
Jacob Erlbeck8158ea72015-06-04 17:46:33 +0200182
183 int64_t m_last_cs_not_low;
Jacob Erlbecke4bcb622015-06-08 11:26:38 +0200184
185 pcu_l1_meas m_l1_meas;
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200186 unsigned m_nack_rate_dl;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200187 uint8_t m_reserved_dl_slots;
188 uint8_t m_reserved_ul_slots;
189 gprs_rlcmac_trx *m_current_trx;
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200190
191 struct gprs_codel *m_codel_state;
Maxa4de02d2019-03-13 16:35:09 +0100192 enum mcs_kind m_mode;
sivasankarida7250a2016-12-16 12:57:18 +0530193
194 unsigned m_dl_ctrl_msg;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200195};
Jacob Erlbeck93990462015-05-15 15:50:43 +0200196
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200197inline bool GprsMs::is_idle() const
198{
199 return !m_ul_tbf && !m_dl_tbf && !m_ref && llist_empty(&m_old_tbfs);
200}
201
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200202inline bool GprsMs::need_dl_tbf() const
203{
204 if (dl_tbf() != NULL && dl_tbf()->state_is_not(GPRS_RLCMAC_WAIT_RELEASE))
205 return false;
206
207 return llc_queue()->size() > 0;
208}
209
Jacob Erlbeck93990462015-05-15 15:50:43 +0200210inline uint32_t GprsMs::tlli() const
211{
Jacob Erlbeck0e50ce62015-05-21 16:58:22 +0200212 return m_new_ul_tlli ? m_new_ul_tlli :
213 m_tlli ? m_tlli :
214 m_new_dl_tlli;
Jacob Erlbeck93990462015-05-15 15:50:43 +0200215}
216
217inline bool GprsMs::check_tlli(uint32_t tlli)
218{
219 return tlli != 0 &&
220 (tlli == m_tlli || tlli == m_new_ul_tlli || tlli == m_new_dl_tlli);
221}
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200222
223inline const char *GprsMs::imsi() const
224{
225 return m_imsi;
226}
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200227
228inline uint8_t GprsMs::ta() const
229{
230 return m_ta;
231}
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200232
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200233inline uint8_t GprsMs::ms_class() const
234{
235 return m_ms_class;
236}
237
Jacob Erlbeckc3c58042015-09-28 17:55:32 +0200238inline uint8_t GprsMs::egprs_ms_class() const
239{
240 return m_egprs_ms_class;
241}
242
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +0200243inline enum CodingScheme GprsMs::current_cs_ul() const
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200244{
245 return m_current_cs_ul;
246}
247
Maxa4de02d2019-03-13 16:35:09 +0100248inline enum mcs_kind GprsMs::mode() const
Jacob Erlbeckcb728902016-01-05 15:33:03 +0100249{
250 return m_mode;
251}
252
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200253inline void GprsMs::set_timeout(unsigned secs)
254{
255 m_delay = secs;
256}
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200257
258inline gprs_llc_queue *GprsMs::llc_queue()
259{
260 return &m_llc_queue;
261}
262
263inline const gprs_llc_queue *GprsMs::llc_queue() const
264{
265 return &m_llc_queue;
266}
267
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200268inline gprs_codel *GprsMs::codel_state() const
269{
270 return m_codel_state;
271}
272
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200273inline unsigned GprsMs::nack_rate_dl() const
274{
275 return m_nack_rate_dl;
276}
277
sivasankarida7250a2016-12-16 12:57:18 +0530278inline unsigned GprsMs::dl_ctrl_msg() const
279{
280 return m_dl_ctrl_msg;
281}
282
283inline void GprsMs::update_dl_ctrl_msg()
284{
285 m_dl_ctrl_msg++;
286}
287
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200288inline uint8_t GprsMs::reserved_dl_slots() const
289{
290 return m_reserved_dl_slots;
291}
292
293inline uint8_t GprsMs::reserved_ul_slots() const
294{
295 return m_reserved_ul_slots;
296}
297
298inline gprs_rlcmac_trx *GprsMs::current_trx() const
299{
300 return m_current_trx;
301}
Pau Espin Pedrolf2dad592020-08-18 20:26:25 +0200302
303#define LOGPMS(ms, category, level, fmt, args...) \
304 LOGP(category, level, "MS(TLLI=0x%08x, IMSI=%s, TA=%" PRIu8 ", %" PRIu8 "/%" PRIu8 ",%s%s) " fmt, \
305 (ms)->tlli(), (ms)->imsi(), (ms)->ta(), (ms)->ms_class(), (ms)->egprs_ms_class(), \
306 (ms)->ul_tbf() ? " UL": "", \
307 (ms)->dl_tbf() ? " DL": "", \
308 ## args)