blob: f9b63f26044cea0c99856935229848cd591b1c11 [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
23struct gprs_rlcmac_tbf;
24struct gprs_rlcmac_dl_tbf;
25struct gprs_rlcmac_ul_tbf;
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +020026struct gprs_codel;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020027
Jacob Erlbeck53670862015-05-12 17:54:33 +020028#include "cxx_linuxlist.h"
Jacob Erlbeck489a2b32015-05-28 19:07:01 +020029#include "llc.h"
Jacob Erlbeckac89a552015-06-29 14:18:46 +020030#include "tbf.h"
Jacob Erlbecke4bcb622015-06-08 11:26:38 +020031#include "pcu_l1_if.h"
Jacob Erlbeckd9e10242015-05-28 15:43:53 +020032
33extern "C" {
34 #include <osmocom/core/timer.h>
35}
36
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020037#include <stdint.h>
38#include <stddef.h>
39
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020040struct BTS;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +020041struct gprs_rlcmac_trx;
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020042
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020043class GprsMs {
44public:
45 struct Callback {
46 virtual void ms_idle(class GprsMs *) = 0;
47 virtual void ms_active(class GprsMs *) = 0;
48 };
49
50 class Guard {
51 public:
52 Guard(GprsMs *ms);
53 ~Guard();
54
Jacob Erlbeckb2439bb2015-07-13 14:23:32 +020055 bool is_idle() const;
56
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020057 private:
58 GprsMs * const m_ms;
59 };
60
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020061 GprsMs(BTS *bts, uint32_t tlli);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020062 ~GprsMs();
63
64 void set_callback(Callback *cb) {m_cb = cb;}
65
Jacob Erlbeck2b349b52015-08-18 11:55:03 +020066 void merge_old_ms(GprsMs *old_ms);
67
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020068 gprs_rlcmac_ul_tbf *ul_tbf() const {return m_ul_tbf;}
69 gprs_rlcmac_dl_tbf *dl_tbf() const {return m_dl_tbf;}
Jacob Erlbeckac89a552015-06-29 14:18:46 +020070 gprs_rlcmac_tbf *tbf(enum gprs_rlcmac_tbf_direction dir) const;
Jacob Erlbeck93990462015-05-15 15:50:43 +020071 uint32_t tlli() const;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020072 void set_tlli(uint32_t tlli);
Jacob Erlbeck93990462015-05-15 15:50:43 +020073 bool confirm_tlli(uint32_t tlli);
74 bool check_tlli(uint32_t tlli);
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020075
Jacob Erlbeckac289052015-08-14 12:50:54 +020076 void reset();
77
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +020078 const char *imsi() const;
79 void set_imsi(const char *imsi);
80
Jacob Erlbeck9200ce62015-05-22 17:48:04 +020081 uint8_t ta() const;
82 void set_ta(uint8_t ta);
Jacob Erlbeckbefc7602015-06-02 12:33:30 +020083 uint8_t ms_class() const;
Jacob Erlbeckc3c58042015-09-28 17:55:32 +020084 uint8_t egprs_ms_class() const;
Jacob Erlbeckbefc7602015-06-02 12:33:30 +020085 void set_ms_class(uint8_t ms_class);
Jacob Erlbeckc3c58042015-09-28 17:55:32 +020086 void set_egprs_ms_class(uint8_t ms_class);
Jacob Erlbeck9200ce62015-05-22 17:48:04 +020087
Jacob Erlbecka700dd92015-06-02 16:00:41 +020088 uint8_t current_cs_ul() const;
89 uint8_t current_cs_dl() const;
90
Jacob Erlbeck699b8dc2015-06-29 14:05:55 +020091 int first_common_ts() const;
Jacob Erlbeck617c7122015-06-30 09:18:30 +020092 uint8_t dl_slots() const;
93 uint8_t ul_slots() const;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +020094 uint8_t reserved_dl_slots() const;
95 uint8_t reserved_ul_slots() const;
96 gprs_rlcmac_trx *current_trx() const;
97 void set_reserved_slots(gprs_rlcmac_trx *trx,
98 uint8_t ul_slots, uint8_t dl_slots);
Jacob Erlbeck699b8dc2015-06-29 14:05:55 +020099
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200100 gprs_llc_queue *llc_queue();
101 const gprs_llc_queue *llc_queue() const;
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200102 gprs_codel *codel_state() const;
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200103
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200104 void set_timeout(unsigned secs);
105
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200106 void attach_tbf(gprs_rlcmac_tbf *tbf);
107 void attach_ul_tbf(gprs_rlcmac_ul_tbf *tbf);
108 void attach_dl_tbf(gprs_rlcmac_dl_tbf *tbf);
109
110 void detach_tbf(gprs_rlcmac_tbf *tbf);
111
Jacob Erlbeck1751c622015-06-04 12:12:32 +0200112 void update_error_rate(gprs_rlcmac_tbf *tbf, int percent);
113
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200114 bool is_idle() const;
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200115 bool need_dl_tbf() const;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200116
117 void* operator new(size_t num);
118 void operator delete(void* p);
119
Jacob Erlbeck53670862015-05-12 17:54:33 +0200120 LListHead<GprsMs>& list() {return this->m_list;}
121 const LListHead<GprsMs>& list() const {return this->m_list;}
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200122 const LListHead<gprs_rlcmac_tbf>& old_tbfs() const {return m_old_tbfs;}
Jacob Erlbeck53670862015-05-12 17:54:33 +0200123
Jacob Erlbecke4bcb622015-06-08 11:26:38 +0200124 void update_l1_meas(const pcu_l1_meas *meas);
125 const pcu_l1_meas* l1_meas() const {return &m_l1_meas;};
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200126 unsigned nack_rate_dl() const;
Jacob Erlbecke4bcb622015-06-08 11:26:38 +0200127
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200128 /* internal use */
129 static void timeout(void *priv_);
130
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200131protected:
132 void update_status();
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200133 GprsMs *ref();
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200134 void unref();
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200135 void start_timer();
136 void stop_timer();
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200137
138private:
Jacob Erlbeck17214bb2015-06-02 14:06:12 +0200139 BTS *m_bts;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200140 Callback * m_cb;
141 gprs_rlcmac_ul_tbf *m_ul_tbf;
142 gprs_rlcmac_dl_tbf *m_dl_tbf;
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200143 LListHead<gprs_rlcmac_tbf> m_old_tbfs;
144
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200145 uint32_t m_tlli;
Jacob Erlbeck93990462015-05-15 15:50:43 +0200146 uint32_t m_new_ul_tlli;
147 uint32_t m_new_dl_tlli;
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200148
149 /* store IMSI for look-up and PCH retransmission */
150 char m_imsi[16];
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200151 uint8_t m_ta;
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200152 uint8_t m_ms_class;
Jacob Erlbeckc3c58042015-09-28 17:55:32 +0200153 uint8_t m_egprs_ms_class;
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200154 /* current coding scheme */
155 uint8_t m_current_cs_ul;
156 uint8_t m_current_cs_dl;
157
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200158 gprs_llc_queue m_llc_queue;
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200159
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200160 bool m_is_idle;
161 int m_ref;
Jacob Erlbeck53670862015-05-12 17:54:33 +0200162 LListHead<GprsMs> m_list;
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200163 struct osmo_timer_list m_timer;
164 unsigned m_delay;
Jacob Erlbeck8158ea72015-06-04 17:46:33 +0200165
166 int64_t m_last_cs_not_low;
Jacob Erlbecke4bcb622015-06-08 11:26:38 +0200167
168 pcu_l1_meas m_l1_meas;
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200169 unsigned m_nack_rate_dl;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200170 uint8_t m_reserved_dl_slots;
171 uint8_t m_reserved_ul_slots;
172 gprs_rlcmac_trx *m_current_trx;
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200173
174 struct gprs_codel *m_codel_state;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +0200175};
Jacob Erlbeck93990462015-05-15 15:50:43 +0200176
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200177inline bool GprsMs::is_idle() const
178{
179 return !m_ul_tbf && !m_dl_tbf && !m_ref && llist_empty(&m_old_tbfs);
180}
181
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200182inline bool GprsMs::need_dl_tbf() const
183{
184 if (dl_tbf() != NULL && dl_tbf()->state_is_not(GPRS_RLCMAC_WAIT_RELEASE))
185 return false;
186
187 return llc_queue()->size() > 0;
188}
189
Jacob Erlbeck93990462015-05-15 15:50:43 +0200190inline uint32_t GprsMs::tlli() const
191{
Jacob Erlbeck0e50ce62015-05-21 16:58:22 +0200192 return m_new_ul_tlli ? m_new_ul_tlli :
193 m_tlli ? m_tlli :
194 m_new_dl_tlli;
Jacob Erlbeck93990462015-05-15 15:50:43 +0200195}
196
197inline bool GprsMs::check_tlli(uint32_t tlli)
198{
199 return tlli != 0 &&
200 (tlli == m_tlli || tlli == m_new_ul_tlli || tlli == m_new_dl_tlli);
201}
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200202
203inline const char *GprsMs::imsi() const
204{
205 return m_imsi;
206}
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200207
208inline uint8_t GprsMs::ta() const
209{
210 return m_ta;
211}
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200212
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200213inline uint8_t GprsMs::ms_class() const
214{
215 return m_ms_class;
216}
217
Jacob Erlbeckc3c58042015-09-28 17:55:32 +0200218inline uint8_t GprsMs::egprs_ms_class() const
219{
220 return m_egprs_ms_class;
221}
222
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200223inline uint8_t GprsMs::current_cs_ul() const
224{
225 return m_current_cs_ul;
226}
227
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200228inline void GprsMs::set_timeout(unsigned secs)
229{
230 m_delay = secs;
231}
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200232
233inline gprs_llc_queue *GprsMs::llc_queue()
234{
235 return &m_llc_queue;
236}
237
238inline const gprs_llc_queue *GprsMs::llc_queue() const
239{
240 return &m_llc_queue;
241}
242
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200243inline gprs_codel *GprsMs::codel_state() const
244{
245 return m_codel_state;
246}
247
Jacob Erlbeck04a10862015-06-12 16:01:56 +0200248inline unsigned GprsMs::nack_rate_dl() const
249{
250 return m_nack_rate_dl;
251}
252
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200253inline uint8_t GprsMs::reserved_dl_slots() const
254{
255 return m_reserved_dl_slots;
256}
257
258inline uint8_t GprsMs::reserved_ul_slots() const
259{
260 return m_reserved_ul_slots;
261}
262
263inline gprs_rlcmac_trx *GprsMs::current_trx() const
264{
265 return m_current_trx;
266}