blob: 688708e709e98cc76f661b7832cc18630caa7a1f [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
Max1187a772018-01-26 13:31:42 +010025#include <gprs_coding_scheme.h>
Jacob Erlbeck53670862015-05-12 17:54:33 +020026#include "cxx_linuxlist.h"
Jacob Erlbeck489a2b32015-05-28 19:07:01 +020027#include "llc.h"
Jacob Erlbeckac89a552015-06-29 14:18:46 +020028#include "tbf.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"
Max1187a772018-01-26 13:31:42 +010031#include <gprs_coding_scheme.h>
Jacob Erlbeckd9e10242015-05-28 15:43:53 +020032
33extern "C" {
34 #include <osmocom/core/timer.h>
Max1187a772018-01-26 13:31:42 +010035 #include <osmocom/core/linuxlist.h>
Jacob Erlbeckd9e10242015-05-28 15:43:53 +020036}
37
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020038#include <stdint.h>
39#include <stddef.h>
40
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020041struct BTS;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +020042struct gprs_rlcmac_trx;
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020043
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020044class GprsMs {
45public:
46 struct Callback {
47 virtual void ms_idle(class GprsMs *) = 0;
48 virtual void ms_active(class GprsMs *) = 0;
49 };
50
51 class Guard {
52 public:
53 Guard(GprsMs *ms);
54 ~Guard();
55
Jacob Erlbeckb2439bb2015-07-13 14:23:32 +020056 bool is_idle() const;
57
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020058 private:
59 GprsMs * const m_ms;
60 };
61
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020062 GprsMs(BTS *bts, uint32_t tlli);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020063 ~GprsMs();
64
65 void set_callback(Callback *cb) {m_cb = cb;}
66
Jacob Erlbeck2b349b52015-08-18 11:55:03 +020067 void merge_old_ms(GprsMs *old_ms);
68
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020069 gprs_rlcmac_ul_tbf *ul_tbf() const {return m_ul_tbf;}
70 gprs_rlcmac_dl_tbf *dl_tbf() const {return m_dl_tbf;}
Jacob Erlbeckac89a552015-06-29 14:18:46 +020071 gprs_rlcmac_tbf *tbf(enum gprs_rlcmac_tbf_direction dir) const;
Jacob Erlbeck93990462015-05-15 15:50:43 +020072 uint32_t tlli() const;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020073 void set_tlli(uint32_t tlli);
Jacob Erlbeck93990462015-05-15 15:50:43 +020074 bool confirm_tlli(uint32_t tlli);
75 bool check_tlli(uint32_t tlli);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020076
Jacob Erlbeckac289052015-08-14 12:50:54 +020077 void reset();
Maxa4de02d2019-03-13 16:35:09 +010078 enum mcs_kind mode() const;
79 void set_mode(enum mcs_kind mode);
Jacob Erlbeckac289052015-08-14 12:50:54 +020080
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +020081 const char *imsi() const;
82 void set_imsi(const char *imsi);
83
Jacob Erlbeck9200ce62015-05-22 17:48:04 +020084 uint8_t ta() const;
85 void set_ta(uint8_t ta);
Jacob Erlbeckbefc7602015-06-02 12:33:30 +020086 uint8_t ms_class() const;
Jacob Erlbeckc3c58042015-09-28 17:55:32 +020087 uint8_t egprs_ms_class() const;
Jacob Erlbeckbefc7602015-06-02 12:33:30 +020088 void set_ms_class(uint8_t ms_class);
Jacob Erlbeckc3c58042015-09-28 17:55:32 +020089 void set_egprs_ms_class(uint8_t ms_class);
Maxbea2edb2019-03-06 17:04:59 +010090 void set_current_cs_dl(CodingScheme scheme);
Jacob Erlbeck9200ce62015-05-22 17:48:04 +020091
Jacob Erlbeckcb728902016-01-05 15:33:03 +010092 GprsCodingScheme current_cs_ul() const;
93 GprsCodingScheme current_cs_dl() const;
94 GprsCodingScheme max_cs_ul() const;
95 GprsCodingScheme max_cs_dl() const;
Jacob Erlbecka700dd92015-06-02 16:00:41 +020096
Jacob Erlbeck699b8dc2015-06-29 14:05:55 +020097 int first_common_ts() const;
Jacob Erlbeck617c7122015-06-30 09:18:30 +020098 uint8_t dl_slots() const;
99 uint8_t ul_slots() const;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200100 uint8_t reserved_dl_slots() const;
101 uint8_t reserved_ul_slots() const;
Jacob Erlbeck7c72aca2016-01-22 17:06:14 +0100102 uint8_t current_pacch_slots() const;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200103 gprs_rlcmac_trx *current_trx() const;
104 void set_reserved_slots(gprs_rlcmac_trx *trx,
105 uint8_t ul_slots, uint8_t dl_slots);
Jacob Erlbeck699b8dc2015-06-29 14:05:55 +0200106
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200107 gprs_llc_queue *llc_queue();
108 const gprs_llc_queue *llc_queue() const;
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200109 gprs_codel *codel_state() const;
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200110
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200111 void set_timeout(unsigned secs);
112
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200113 void attach_tbf(gprs_rlcmac_tbf *tbf);
114 void attach_ul_tbf(gprs_rlcmac_ul_tbf *tbf);
115 void attach_dl_tbf(gprs_rlcmac_dl_tbf *tbf);
116
117 void detach_tbf(gprs_rlcmac_tbf *tbf);
118
Jacob Erlbeck1751c622015-06-04 12:12:32 +0200119 void update_error_rate(gprs_rlcmac_tbf *tbf, int percent);
120
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200121 bool is_idle() const;
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200122 bool need_dl_tbf() const;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200123
124 void* operator new(size_t num);
125 void operator delete(void* p);
126
Jacob Erlbeck53670862015-05-12 17:54:33 +0200127 LListHead<GprsMs>& list() {return this->m_list;}
128 const LListHead<GprsMs>& list() const {return this->m_list;}
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200129 const LListHead<gprs_rlcmac_tbf>& old_tbfs() const {return m_old_tbfs;}
Jacob Erlbeck53670862015-05-12 17:54:33 +0200130
Jacob Erlbecke4bcb622015-06-08 11:26:38 +0200131 void update_l1_meas(const pcu_l1_meas *meas);
132 const pcu_l1_meas* l1_meas() const {return &m_l1_meas;};
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200133 unsigned nack_rate_dl() const;
sivasankarida7250a2016-12-16 12:57:18 +0530134 unsigned dl_ctrl_msg() const;
135 void update_dl_ctrl_msg();
Jacob Erlbecke4bcb622015-06-08 11:26:38 +0200136
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200137 /* internal use */
138 static void timeout(void *priv_);
139
Oliver Smithcfb63212019-09-05 17:13:33 +0200140 bool app_info_pending;
141
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200142protected:
143 void update_status();
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200144 GprsMs *ref();
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200145 void unref();
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200146 void start_timer();
147 void stop_timer();
Jacob Erlbeckcb728902016-01-05 15:33:03 +0100148 void update_cs_ul(const pcu_l1_meas*);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200149
150private:
Jacob Erlbeck17214bb2015-06-02 14:06:12 +0200151 BTS *m_bts;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200152 Callback * m_cb;
153 gprs_rlcmac_ul_tbf *m_ul_tbf;
154 gprs_rlcmac_dl_tbf *m_dl_tbf;
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200155 LListHead<gprs_rlcmac_tbf> m_old_tbfs;
156
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200157 uint32_t m_tlli;
Jacob Erlbeck93990462015-05-15 15:50:43 +0200158 uint32_t m_new_ul_tlli;
159 uint32_t m_new_dl_tlli;
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200160
161 /* store IMSI for look-up and PCH retransmission */
162 char m_imsi[16];
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200163 uint8_t m_ta;
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200164 uint8_t m_ms_class;
Jacob Erlbeckc3c58042015-09-28 17:55:32 +0200165 uint8_t m_egprs_ms_class;
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200166 /* current coding scheme */
Jacob Erlbeckcb728902016-01-05 15:33:03 +0100167 GprsCodingScheme m_current_cs_ul;
168 GprsCodingScheme m_current_cs_dl;
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200169
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200170 gprs_llc_queue m_llc_queue;
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200171
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200172 bool m_is_idle;
173 int m_ref;
Jacob Erlbeck53670862015-05-12 17:54:33 +0200174 LListHead<GprsMs> m_list;
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200175 struct osmo_timer_list m_timer;
176 unsigned m_delay;
Jacob Erlbeck8158ea72015-06-04 17:46:33 +0200177
178 int64_t m_last_cs_not_low;
Jacob Erlbecke4bcb622015-06-08 11:26:38 +0200179
180 pcu_l1_meas m_l1_meas;
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200181 unsigned m_nack_rate_dl;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200182 uint8_t m_reserved_dl_slots;
183 uint8_t m_reserved_ul_slots;
184 gprs_rlcmac_trx *m_current_trx;
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200185
186 struct gprs_codel *m_codel_state;
Maxa4de02d2019-03-13 16:35:09 +0100187 enum mcs_kind m_mode;
sivasankarida7250a2016-12-16 12:57:18 +0530188
189 unsigned m_dl_ctrl_msg;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200190};
Jacob Erlbeck93990462015-05-15 15:50:43 +0200191
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200192inline bool GprsMs::is_idle() const
193{
194 return !m_ul_tbf && !m_dl_tbf && !m_ref && llist_empty(&m_old_tbfs);
195}
196
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200197inline bool GprsMs::need_dl_tbf() const
198{
199 if (dl_tbf() != NULL && dl_tbf()->state_is_not(GPRS_RLCMAC_WAIT_RELEASE))
200 return false;
201
202 return llc_queue()->size() > 0;
203}
204
Jacob Erlbeck93990462015-05-15 15:50:43 +0200205inline uint32_t GprsMs::tlli() const
206{
Jacob Erlbeck0e50ce62015-05-21 16:58:22 +0200207 return m_new_ul_tlli ? m_new_ul_tlli :
208 m_tlli ? m_tlli :
209 m_new_dl_tlli;
Jacob Erlbeck93990462015-05-15 15:50:43 +0200210}
211
212inline bool GprsMs::check_tlli(uint32_t tlli)
213{
214 return tlli != 0 &&
215 (tlli == m_tlli || tlli == m_new_ul_tlli || tlli == m_new_dl_tlli);
216}
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200217
218inline const char *GprsMs::imsi() const
219{
220 return m_imsi;
221}
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200222
223inline uint8_t GprsMs::ta() const
224{
225 return m_ta;
226}
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200227
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200228inline uint8_t GprsMs::ms_class() const
229{
230 return m_ms_class;
231}
232
Jacob Erlbeckc3c58042015-09-28 17:55:32 +0200233inline uint8_t GprsMs::egprs_ms_class() const
234{
235 return m_egprs_ms_class;
236}
237
Jacob Erlbeckcb728902016-01-05 15:33:03 +0100238inline GprsCodingScheme GprsMs::current_cs_ul() const
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200239{
240 return m_current_cs_ul;
241}
242
Maxa4de02d2019-03-13 16:35:09 +0100243inline enum mcs_kind GprsMs::mode() const
Jacob Erlbeckcb728902016-01-05 15:33:03 +0100244{
245 return m_mode;
246}
247
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200248inline void GprsMs::set_timeout(unsigned secs)
249{
250 m_delay = secs;
251}
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200252
253inline gprs_llc_queue *GprsMs::llc_queue()
254{
255 return &m_llc_queue;
256}
257
258inline const gprs_llc_queue *GprsMs::llc_queue() const
259{
260 return &m_llc_queue;
261}
262
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200263inline gprs_codel *GprsMs::codel_state() const
264{
265 return m_codel_state;
266}
267
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200268inline unsigned GprsMs::nack_rate_dl() const
269{
270 return m_nack_rate_dl;
271}
272
sivasankarida7250a2016-12-16 12:57:18 +0530273inline unsigned GprsMs::dl_ctrl_msg() const
274{
275 return m_dl_ctrl_msg;
276}
277
278inline void GprsMs::update_dl_ctrl_msg()
279{
280 m_dl_ctrl_msg++;
281}
282
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200283inline uint8_t GprsMs::reserved_dl_slots() const
284{
285 return m_reserved_dl_slots;
286}
287
288inline uint8_t GprsMs::reserved_ul_slots() const
289{
290 return m_reserved_ul_slots;
291}
292
293inline gprs_rlcmac_trx *GprsMs::current_trx() const
294{
295 return m_current_trx;
296}