blob: 54a29cb7c536eaa0799e6f8ccd3fc7d24b4f674f [file] [log] [blame]
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +02001/* bts.h
2 *
3 * Copyright (C) 2012 Ivan Klyuchnikov
4 * Copyright (C) 2013 by Holger Hans Peter Freyther
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
23
24#ifdef __cplusplus
25extern "C" {
26#include <osmocom/core/linuxlist.h>
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +010027#include <osmocom/core/rate_ctr.h>
Jacob Erlbeckf5898a02015-11-27 19:05:13 +010028#include <osmocom/core/stat_item.h>
Pau Espin Pedrol28f160e2019-09-05 14:48:35 +020029#include <osmocom/core/tdef.h>
bhargava959d1de2016-08-17 15:17:21 +053030#include <osmocom/gsm/l1sap.h>
Max84bf0fa2017-09-01 11:02:40 +020031#include <osmocom/gsm/protocol/gsm_04_08.h>
Maxc5407c72018-02-05 16:11:36 +010032 #include <mslot_class.h>
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +020033}
Holger Hans Peter Freytherb78adcd2013-10-17 20:12:37 +020034
Max1187a772018-01-26 13:31:42 +010035#include <gsm_rlcmac.h>
Holger Hans Peter Freytherb78adcd2013-10-17 20:12:37 +020036#include "poll_controller.h"
Holger Hans Peter Freythercedf8902013-10-19 20:47:12 +020037#include "sba.h"
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +020038#include "tbf.h"
Jacob Erlbecke43460b2015-05-13 13:33:12 +020039#include "gprs_ms_storage.h"
Jacob Erlbeck9e862e12015-12-14 15:19:12 +010040#include "gprs_coding_scheme.h"
Max1187a772018-01-26 13:31:42 +010041#include <cxx_linuxlist.h>
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +020042#endif
43
Daniel Willmannfb3fd092019-03-06 17:09:43 +010044#include <pdch.h>
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +020045#include <stdint.h>
46
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +020047#define LLC_CODEL_DISABLE 0
48#define LLC_CODEL_USE_DEFAULT (-1)
Max8119ecd2019-03-06 19:03:01 +010049
50#define MAX_EDGE_MCS 9
51#define MAX_GPRS_CS 4
Aravind Sirsikar99ab0a82016-06-15 17:46:41 +053052
Harald Welte717cdf52017-07-21 21:56:23 +020053/* see bts->gsmtap_categ_mask */
54enum pcu_gsmtap_category {
55 PCU_GSMTAP_C_DL_UNKNOWN = 0, /* unknown or undecodable downlink blocks */
56 PCU_GSMTAP_C_DL_DUMMY = 1, /* downlink dummy blocks */
57 PCU_GSMTAP_C_DL_CTRL = 2, /* downlink control blocks */
58 PCU_GSMTAP_C_DL_DATA_GPRS = 3, /* downlink GPRS data blocks */
59 PCU_GSMTAP_C_DL_DATA_EGPRS = 4, /* downlink EGPRS data blocks */
Harald Weltebc219d52017-07-29 13:42:27 +020060 PCU_GSMTAP_C_DL_PTCCH = 5, /* downlink PTCCH blocks */
Pau Espin Pedrol58543702019-09-30 19:07:16 +020061 PCU_GSMTAP_C_DL_AGCH = 6, /* downlink AGCH blocks */
62 PCU_GSMTAP_C_DL_PCH = 7, /* downlink PCH blocks */
Harald Welte717cdf52017-07-21 21:56:23 +020063
64 PCU_GSMTAP_C_UL_UNKNOWN = 15, /* unknown or undecodable uplink blocks */
65 PCU_GSMTAP_C_UL_DUMMY = 16, /* uplink dummy blocks */
66 PCU_GSMTAP_C_UL_CTRL = 17, /* uplink control blocks */
67 PCU_GSMTAP_C_UL_DATA_GPRS = 18, /* uplink GPRS data blocks */
68 PCU_GSMTAP_C_UL_DATA_EGPRS = 19, /* uplink EGPRS data blocks */
Pau Espin Pedrol1ec211f2019-09-30 18:33:14 +020069 PCU_GSMTAP_C_UL_RACH = 20, /* uplink RACH bursts */
Harald Welte717cdf52017-07-21 21:56:23 +020070};
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +020071
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +020072struct BTS;
Jacob Erlbecke2e004e2015-06-18 17:16:26 +020073struct GprsMs;
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +020074
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +020075struct gprs_rlcmac_trx {
76 void *fl1h;
77 uint16_t arfcn;
78 struct gprs_rlcmac_pdch pdch[8];
Holger Hans Peter Freyther4ed1dae2013-10-20 10:14:03 +020079
80 /* back pointers */
81 struct BTS *bts;
82 uint8_t trx_no;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +020083
84#ifdef __cplusplus
85 void reserve_slots(enum gprs_rlcmac_tbf_direction dir, uint8_t slots);
86 void unreserve_slots(enum gprs_rlcmac_tbf_direction dir, uint8_t slots);
87#endif
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +020088};
89
Maxd71e8b32016-09-19 16:17:06 +020090#ifdef __cplusplus
91extern "C" {
92#endif
Minh-Quang Nguyen1bcfa9a2017-11-01 14:41:37 -040093void bts_update_tbf_ta(const char *p, uint32_t fn, uint8_t trx_no, uint8_t ts, int8_t ta, bool is_rach);
Maxd71e8b32016-09-19 16:17:06 +020094#ifdef __cplusplus
95}
96#endif
97
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +020098/**
99 * This is the data from C. As soon as our minimal compiler is gcc 4.7
100 * we can start to compile pcu_vty.c with c++ and remove the split.
101 */
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200102struct gprs_rlcmac_bts {
Stefan Sperling5b22fb72018-02-14 19:46:33 +0100103 bool active;
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200104 uint8_t bsic;
105 uint8_t fc_interval;
Jacob Erlbeck0288cdb2015-05-06 10:47:30 +0200106 uint16_t fc_bucket_time;
Jacob Erlbeck87d73412015-04-21 12:56:48 +0200107 uint32_t fc_bvc_bucket_size;
108 uint32_t fc_bvc_leak_rate;
109 uint32_t fc_ms_bucket_size;
110 uint32_t fc_ms_leak_rate;
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200111 uint8_t cs1;
112 uint8_t cs2;
113 uint8_t cs3;
114 uint8_t cs4;
115 uint8_t initial_cs_dl, initial_cs_ul;
Jacob Erlbeck96ccea82016-01-07 15:00:43 +0100116 uint8_t initial_mcs_dl, initial_mcs_ul;
Jacob Erlbeckb33e6752015-06-04 19:04:30 +0200117 uint8_t max_cs_dl, max_cs_ul;
Jacob Erlbeck0d058052016-01-07 11:48:28 +0100118 uint8_t max_mcs_dl, max_mcs_ul;
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200119 uint8_t force_cs; /* 0=use from BTS 1=use from VTY */
120 uint16_t force_llc_lifetime; /* overrides lifetime from SGSN */
Jacob Erlbeck0c1c8772015-03-20 12:02:42 +0100121 uint32_t llc_discard_csec;
Jacob Erlbeckd0261b72015-04-02 13:58:09 +0200122 uint32_t llc_idle_ack_csec;
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +0200123 uint32_t llc_codel_interval_msec; /* 0=disabled, -1=use default interval */
Pau Espin Pedrol28f160e2019-09-05 14:48:35 +0200124 /* Timer defintions */
125 struct osmo_tdef *T_defs_bts; /* timers controlled by BTS, received through PCUIF */
126 struct osmo_tdef *T_defs_pcu; /* timers controlled by PCU */
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200127 uint8_t n3101;
128 uint8_t n3103;
129 uint8_t n3105;
Max2efdf692016-02-22 11:39:30 +0100130 struct gsmtap_inst *gsmtap;
Harald Welte717cdf52017-07-21 21:56:23 +0200131 uint32_t gsmtap_categ_mask;
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200132 struct gprs_rlcmac_trx trx[8];
Maxe9fe0e32017-09-28 15:56:05 +0200133 int (*alloc_algorithm)(struct gprs_rlcmac_bts *bts, struct GprsMs *ms, struct gprs_rlcmac_tbf *tbf,
134 bool single, int8_t use_tbf);
135
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200136 uint8_t force_two_phase;
137 uint8_t alpha, gamma;
Jacob Erlbeck953c7892015-09-28 18:12:57 +0200138 uint8_t egprs_enabled;
Oliver Smith45fdc442019-09-05 15:12:20 +0200139 bool dl_tbf_preemptive_retransmission;
Max84bf0fa2017-09-01 11:02:40 +0200140 uint8_t si13[GSM_MACBLOCK_LEN];
141 bool si13_is_set;
Aravind Sirsikare6cadb42016-08-16 16:32:59 +0530142 /* 0 to support resegmentation in DL, 1 for no reseg */
143 uint8_t dl_arq_type;
144
Jacob Erlbeck1751c622015-06-04 12:12:32 +0200145 uint8_t cs_adj_enabled;
146 uint8_t cs_adj_upper_limit;
147 uint8_t cs_adj_lower_limit;
Aravind Sirsikar99ab0a82016-06-15 17:46:41 +0530148 struct {int16_t low; int16_t high; } cs_lqual_ranges[MAX_GPRS_CS];
Max8119ecd2019-03-06 19:03:01 +0100149 struct {int16_t low; int16_t high; } mcs_lqual_ranges[MAX_EDGE_MCS];
Jacob Erlbeck70b96aa2015-06-12 10:52:34 +0200150 uint16_t cs_downgrade_threshold; /* downgrade if less packets left (DL) */
Jacob Erlbeck36df7742016-01-19 15:53:30 +0100151 uint16_t ws_base;
152 uint16_t ws_pdch; /* increase WS by this value per PDCH */
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200153
Jacob Erlbeck77da3552015-07-16 18:33:46 +0200154 /* State for dynamic algorithm selection */
155 int multislot_disabled;
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +0200156
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200157 /**
158 * Point back to the C++ object. This is used during the transition
159 * period.
160 */
161 struct BTS *bts;
Pau Espin Pedrolc4178e52017-08-08 15:03:50 +0200162
163 /* Path to be used for the pcu-bts socket */
164 char *pcu_sock_path;
Harald Welte57d35152018-07-05 03:11:17 +0200165
166 /* Are we talking Gb with IP-SNS (true) or classic Gb? */
167 bool gb_dialect_sns;
Oliver Smithcfb63212019-09-05 17:13:33 +0200168
169 /* Packet Application Information (3GPP TS 44.060 11.2.47, usually ETWS primary message). We don't need to store
170 * more than one message, because they get sent so rarely. */
171 struct msgb *app_info;
172 uint32_t app_info_pending; /* Count of MS with active TBF, to which we did not send app_info yet */
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200173};
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200174
175#ifdef __cplusplus
176/**
177 * I represent a GSM BTS. I have one or more TRX, I know the current
178 * GSM time and I have controllers that help with allocating resources
179 * on my TRXs.
180 */
181struct BTS {
182public:
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100183 enum {
184 CTR_TBF_DL_ALLOCATED,
185 CTR_TBF_DL_FREED,
Jacob Erlbeck0316dc62016-01-21 20:12:04 +0100186 CTR_TBF_DL_ABORTED,
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100187 CTR_TBF_UL_ALLOCATED,
188 CTR_TBF_UL_FREED,
Jacob Erlbeck0316dc62016-01-21 20:12:04 +0100189 CTR_TBF_UL_ABORTED,
Holger Hans Peter Freytheraa35ba72013-11-13 15:02:50 +0100190 CTR_TBF_REUSED,
Jacob Erlbeck5979fe92015-07-14 14:02:41 +0200191 CTR_TBF_ALLOC_ALGO_A,
192 CTR_TBF_ALLOC_ALGO_B,
Jacob Erlbeck9b3d7e02016-01-19 10:44:42 +0100193 CTR_TBF_FAILED_EGPRS_ONLY,
Holger Hans Peter Freythere9429b52013-11-13 19:36:57 +0100194 CTR_RLC_SENT,
195 CTR_RLC_RESENT,
Holger Hans Peter Freytheref93bdb2013-11-24 00:01:50 +0100196 CTR_RLC_RESTARTED,
Holger Hans Peter Freytherc70aae42013-11-19 17:09:37 +0100197 CTR_RLC_STALLED,
Holger Hans Peter Freyther092478f2013-11-23 01:01:19 +0100198 CTR_RLC_NACKED,
sivasankari168911b2016-11-25 19:53:36 +0530199 CTR_RLC_FINAL_BLOCK_RESENT,
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200200 CTR_RLC_ASS_TIMEDOUT,
Jacob Erlbeck7c8d39a2015-09-07 14:04:56 +0200201 CTR_RLC_ASS_FAILED,
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200202 CTR_RLC_ACK_TIMEDOUT,
Jacob Erlbeck7c8d39a2015-09-07 14:04:56 +0200203 CTR_RLC_ACK_FAILED,
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200204 CTR_RLC_REL_TIMEDOUT,
Jacob Erlbeckbe4a08b2015-08-25 15:19:31 +0200205 CTR_RLC_LATE_BLOCK,
Alexander Couzens2cb15472016-05-18 16:43:10 +0200206 CTR_RLC_SENT_DUMMY,
Alexander Couzensf929e622016-05-22 00:02:56 +0200207 CTR_RLC_SENT_CONTROL,
Alexander Couzens7fdbf892016-05-21 19:45:23 +0200208 CTR_RLC_DL_BYTES,
209 CTR_RLC_DL_PAYLOAD_BYTES,
210 CTR_RLC_UL_BYTES,
211 CTR_RLC_UL_PAYLOAD_BYTES,
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100212 CTR_DECODE_ERRORS,
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100213 CTR_SBA_ALLOCATED,
214 CTR_SBA_FREED,
215 CTR_SBA_TIMEDOUT,
Holger Hans Peter Freyther19977872013-10-27 10:34:31 +0100216 CTR_LLC_FRAME_TIMEDOUT,
217 CTR_LLC_FRAME_DROPPED,
Holger Hans Peter Freytherb3d5ee22013-11-13 16:43:26 +0100218 CTR_LLC_FRAME_SCHED,
Alexander Couzens7fdbf892016-05-21 19:45:23 +0200219 CTR_LLC_DL_BYTES,
220 CTR_LLC_UL_BYTES,
Holger Hans Peter Freytherc1ae2262013-10-27 10:50:35 +0100221 CTR_RACH_REQUESTS,
sivasankari67b89ca2016-12-29 16:25:30 +0530222 CTR_11BIT_RACH_REQUESTS,
sivasankarida7250a2016-12-16 12:57:18 +0530223 CTR_SPB_UL_FIRST_SEGMENT,
224 CTR_SPB_UL_SECOND_SEGMENT,
225 CTR_SPB_DL_FIRST_SEGMENT,
226 CTR_SPB_DL_SECOND_SEGMENT,
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530227 CTR_IMMEDIATE_ASSIGN_UL_TBF,
sivasankari168911b2016-11-25 19:53:36 +0530228 CTR_IMMEDIATE_ASSIGN_REJ,
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530229 CTR_IMMEDIATE_ASSIGN_DL_TBF,
sivasankari168911b2016-11-25 19:53:36 +0530230 CTR_CHANNEL_REQUEST_DESCRIPTION,
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530231 CTR_PKT_UL_ASSIGNMENT,
sivasankari168911b2016-11-25 19:53:36 +0530232 CTR_PKT_ACCESS_REJ,
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530233 CTR_PKT_DL_ASSIGNMENT,
234 CTR_RLC_RECV_CONTROL,
235 CTR_PUA_POLL_TIMEDOUT,
236 CTR_PUA_POLL_FAILED,
237 CTR_PDA_POLL_TIMEDOUT,
238 CTR_PDA_POLL_FAILED,
239 CTR_PUAN_POLL_TIMEDOUT,
240 CTR_PUAN_POLL_FAILED,
241 CTR_PDAN_POLL_TIMEDOUT,
242 CTR_PDAN_POLL_FAILED,
243 CTR_GPRS_DL_CS1,
244 CTR_GPRS_DL_CS2,
245 CTR_GPRS_DL_CS3,
246 CTR_GPRS_DL_CS4,
247 CTR_EGPRS_DL_MCS1,
248 CTR_EGPRS_DL_MCS2,
249 CTR_EGPRS_DL_MCS3,
250 CTR_EGPRS_DL_MCS4,
251 CTR_EGPRS_DL_MCS5,
252 CTR_EGPRS_DL_MCS6,
253 CTR_EGPRS_DL_MCS7,
254 CTR_EGPRS_DL_MCS8,
255 CTR_EGPRS_DL_MCS9,
256 CTR_GPRS_UL_CS1,
257 CTR_GPRS_UL_CS2,
258 CTR_GPRS_UL_CS3,
259 CTR_GPRS_UL_CS4,
260 CTR_EGPRS_UL_MCS1,
261 CTR_EGPRS_UL_MCS2,
262 CTR_EGPRS_UL_MCS3,
263 CTR_EGPRS_UL_MCS4,
264 CTR_EGPRS_UL_MCS5,
265 CTR_EGPRS_UL_MCS6,
266 CTR_EGPRS_UL_MCS7,
267 CTR_EGPRS_UL_MCS8,
268 CTR_EGPRS_UL_MCS9,
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100269 };
270
Jacob Erlbeck502bd1f2015-03-20 14:26:05 +0100271 enum {
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100272 STAT_MS_PRESENT,
273 };
274
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200275 BTS();
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100276 ~BTS();
Maxd5ffeb52019-03-18 15:48:38 +0100277 void cleanup();
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200278
279 static BTS* main_bts();
280
281 struct gprs_rlcmac_bts *bts_data();
Holger Hans Peter Freythercedf8902013-10-19 20:47:12 +0200282 SBAController *sba();
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200283
Holger Hans Peter Freyther9b30c7f2013-10-17 19:59:56 +0200284 /** TODO: change the number to unsigned */
285 void set_current_frame_number(int frame_number);
Jacob Erlbeckaf75ce82015-08-26 13:22:28 +0200286 void set_current_block_frame_number(int frame_number, unsigned max_delay);
Holger Hans Peter Freyther9b30c7f2013-10-17 19:59:56 +0200287 int current_frame_number() const;
288
Holger Hans Peter Freytherf0984892013-10-19 18:28:59 +0200289 /** add paging to paging queue(s) */
290 int add_paging(uint8_t chan_needed, uint8_t *identity_lv);
291
Daniel Willmannfe6e2e42014-07-10 17:44:06 +0200292 gprs_rlcmac_dl_tbf *dl_tbf_by_poll_fn(uint32_t fn, uint8_t trx, uint8_t ts);
293 gprs_rlcmac_ul_tbf *ul_tbf_by_poll_fn(uint32_t fn, uint8_t trx, uint8_t ts);
Jacob Erlbeck3a10dbd2015-07-10 19:52:37 +0200294 gprs_rlcmac_dl_tbf *dl_tbf_by_tfi(uint8_t tfi, uint8_t trx, uint8_t ts);
295 gprs_rlcmac_ul_tbf *ul_tbf_by_tfi(uint8_t tfi, uint8_t trx, uint8_t ts);
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +0200296
Maxa76a7d02018-01-26 11:09:16 +0100297 int tfi_find_free(enum gprs_rlcmac_tbf_direction dir, uint8_t *_trx, int8_t use_trx) const;
Holger Hans Peter Freyther70ddde62013-10-26 19:17:58 +0200298
Holger Hans Peter Freyther40cfaa62013-10-26 19:49:16 +0200299 int rcv_imm_ass_cnf(const uint8_t *data, uint32_t fn);
Philipp Maier1275a3f2017-02-21 19:35:23 +0100300
Max5dd8d1b2017-03-08 12:06:42 +0100301 uint32_t rfn_to_fn(int32_t rfn);
Max58818582018-01-05 15:04:50 +0100302 int rcv_rach(uint16_t ra, uint32_t Fn, int16_t qta, bool is_11bit,
bhargava959d1de2016-08-17 15:17:21 +0530303 enum ph_burst_type burst_type);
Vadim Yanitskiyffebd242019-10-05 23:45:31 +0700304 int rcv_ptcch_rach(uint8_t trx_nr, uint8_t ts_nr, uint32_t fn, int16_t qta);
Holger Hans Peter Freyther40cfaa62013-10-26 19:49:16 +0200305
Maxfc8afc22019-02-18 18:52:38 +0100306 void snd_dl_ass(gprs_rlcmac_tbf *tbf, bool poll, const char *imsi);
Holger Hans Peter Freytherd9262b32013-10-26 20:12:59 +0200307
Jacob Erlbecke43460b2015-05-13 13:33:12 +0200308 GprsMsStorage &ms_store();
309 GprsMs *ms_by_tlli(uint32_t tlli, uint32_t old_tlli = 0);
Jacob Erlbeck076f5c72015-08-21 18:00:54 +0200310 GprsMs *ms_by_imsi(const char *imsi);
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100311 GprsMs *ms_alloc(uint8_t ms_class, uint8_t egprs_ms_class = 0);
Jacob Erlbecke43460b2015-05-13 13:33:12 +0200312
Harald Welte717cdf52017-07-21 21:56:23 +0200313 void send_gsmtap(enum pcu_gsmtap_category categ, bool uplink, uint8_t trx_no,
314 uint8_t ts_no, uint8_t channel, uint32_t fn,
315 const uint8_t *data, unsigned int len);
316
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100317 /*
318 * Statistics
319 */
320 void tbf_dl_created();
321 void tbf_dl_freed();
Jacob Erlbeck0316dc62016-01-21 20:12:04 +0100322 void tbf_dl_aborted();
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100323 void tbf_ul_created();
324 void tbf_ul_freed();
Jacob Erlbeck0316dc62016-01-21 20:12:04 +0100325 void tbf_ul_aborted();
Holger Hans Peter Freytheraa35ba72013-11-13 15:02:50 +0100326 void tbf_reused();
Jacob Erlbeck5979fe92015-07-14 14:02:41 +0200327 void tbf_alloc_algo_a();
328 void tbf_alloc_algo_b();
Jacob Erlbeck9b3d7e02016-01-19 10:44:42 +0100329 void tbf_failed_egprs_only();
Holger Hans Peter Freythere9429b52013-11-13 19:36:57 +0100330 void rlc_sent();
331 void rlc_resent();
Holger Hans Peter Freytheref93bdb2013-11-24 00:01:50 +0100332 void rlc_restarted();
Holger Hans Peter Freytherc70aae42013-11-19 17:09:37 +0100333 void rlc_stalled();
Holger Hans Peter Freyther092478f2013-11-23 01:01:19 +0100334 void rlc_nacked();
sivasankari168911b2016-11-25 19:53:36 +0530335 void rlc_final_block_resent();
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200336 void rlc_ass_timedout();
Jacob Erlbeck7c8d39a2015-09-07 14:04:56 +0200337 void rlc_ass_failed();
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200338 void rlc_ack_timedout();
Jacob Erlbeck7c8d39a2015-09-07 14:04:56 +0200339 void rlc_ack_failed();
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200340 void rlc_rel_timedout();
Jacob Erlbeckbe4a08b2015-08-25 15:19:31 +0200341 void rlc_late_block();
Alexander Couzens2cb15472016-05-18 16:43:10 +0200342 void rlc_sent_dummy();
Alexander Couzensf929e622016-05-22 00:02:56 +0200343 void rlc_sent_control();
Alexander Couzens7fdbf892016-05-21 19:45:23 +0200344 void rlc_dl_bytes(int bytes);
345 void rlc_dl_payload_bytes(int bytes);
346 void rlc_ul_bytes(int bytes);
347 void rlc_ul_payload_bytes(int bytes);
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100348 void decode_error();
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100349 void sba_allocated();
350 void sba_freed();
351 void sba_timedout();
Holger Hans Peter Freytherb3d5ee22013-11-13 16:43:26 +0100352 void llc_timedout_frame();
353 void llc_dropped_frame();
354 void llc_frame_sched();
Alexander Couzens7fdbf892016-05-21 19:45:23 +0200355 void llc_dl_bytes(int bytes);
356 void llc_ul_bytes(int bytes);
Holger Hans Peter Freytherc1ae2262013-10-27 10:50:35 +0100357 void rach_frame();
sivasankari67b89ca2016-12-29 16:25:30 +0530358 void rach_frame_11bit();
sivasankarida7250a2016-12-16 12:57:18 +0530359 void spb_uplink_first_segment();
360 void spb_uplink_second_segment();
361 void spb_downlink_first_segment();
362 void spb_downlink_second_segment();
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530363 void immediate_assignment_ul_tbf();
sivasankari168911b2016-11-25 19:53:36 +0530364 void immediate_assignment_reject();
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530365 void immediate_assignment_dl_tbf();
sivasankari168911b2016-11-25 19:53:36 +0530366 void channel_request_description();
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530367 void pkt_ul_assignment();
sivasankari168911b2016-11-25 19:53:36 +0530368 void pkt_access_reject();
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530369 void pkt_dl_assignemnt();
370 void rlc_rcvd_control();
371 void pua_poll_timedout();
372 void pua_poll_failed();
373 void pda_poll_timedout();
374 void pda_poll_failed();
375 void pkt_ul_ack_nack_poll_timedout();
376 void pkt_ul_ack_nack_poll_failed();
377 void pkt_dl_ack_nack_poll_timedout();
378 void pkt_dl_ack_nack_poll_failed();
379 void gprs_dl_cs1();
380 void gprs_dl_cs2();
381 void gprs_dl_cs3();
382 void gprs_dl_cs4();
383 void egprs_dl_mcs1();
384 void egprs_dl_mcs2();
385 void egprs_dl_mcs3();
386 void egprs_dl_mcs4();
387 void egprs_dl_mcs5();
388 void egprs_dl_mcs6();
389 void egprs_dl_mcs7();
390 void egprs_dl_mcs8();
391 void egprs_dl_mcs9();
392 void gprs_ul_cs1();
393 void gprs_ul_cs2();
394 void gprs_ul_cs3();
395 void gprs_ul_cs4();
396 void egprs_ul_mcs1();
397 void egprs_ul_mcs2();
398 void egprs_ul_mcs3();
399 void egprs_ul_mcs4();
400 void egprs_ul_mcs5();
401 void egprs_ul_mcs6();
402 void egprs_ul_mcs7();
403 void egprs_ul_mcs8();
404 void egprs_ul_mcs9();
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100405
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100406 void ms_present(int32_t n);
407 int32_t ms_present_get();
408
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100409 /*
410 * Below for C interface for the VTY
411 */
412 struct rate_ctr_group *rate_counters() const;
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100413 struct osmo_stat_item_group *stat_items() const;
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100414
Jacob Erlbecked2dbf62015-12-28 19:15:40 +0100415 LListHead<gprs_rlcmac_tbf>& ul_tbfs();
416 LListHead<gprs_rlcmac_tbf>& dl_tbfs();
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200417private:
Holger Hans Peter Freyther9b30c7f2013-10-17 19:59:56 +0200418 int m_cur_fn;
Jacob Erlbeck60f77032015-08-24 14:35:14 +0200419 int m_cur_blk_fn;
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200420 struct gprs_rlcmac_bts m_bts;
Holger Hans Peter Freytherb78adcd2013-10-17 20:12:37 +0200421 PollController m_pollController;
Holger Hans Peter Freythercedf8902013-10-19 20:47:12 +0200422 SBAController m_sba;
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100423 struct rate_ctr_group *m_ratectrs;
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100424 struct osmo_stat_item_group *m_statg;
Holger Hans Peter Freytherb78adcd2013-10-17 20:12:37 +0200425
Jacob Erlbecke43460b2015-05-13 13:33:12 +0200426 GprsMsStorage m_ms_store;
427
Jacob Erlbecked2dbf62015-12-28 19:15:40 +0100428 /* list of uplink TBFs */
429 LListHead<gprs_rlcmac_tbf> m_ul_tbfs;
430 /* list of downlink TBFs */
431 LListHead<gprs_rlcmac_tbf> m_dl_tbfs;
432
Holger Hans Peter Freytherb78adcd2013-10-17 20:12:37 +0200433 /* disable copying to avoid slicing */
434 BTS(const BTS&);
435 BTS& operator=(const BTS&);
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200436};
Holger Hans Peter Freyther9b30c7f2013-10-17 19:59:56 +0200437
438inline int BTS::current_frame_number() const
439{
440 return m_cur_fn;
441}
Holger Hans Peter Freyther111614a2013-10-19 20:04:57 +0200442
Holger Hans Peter Freythercedf8902013-10-19 20:47:12 +0200443inline SBAController *BTS::sba()
444{
445 return &m_sba;
446}
Holger Hans Peter Freyther65be4802013-10-26 18:39:36 +0200447
Jacob Erlbecke43460b2015-05-13 13:33:12 +0200448inline GprsMsStorage &BTS::ms_store()
449{
450 return m_ms_store;
451}
452
453inline GprsMs *BTS::ms_by_tlli(uint32_t tlli, uint32_t old_tlli)
454{
455 return ms_store().get_ms(tlli, old_tlli);
456}
457
Jacob Erlbeck076f5c72015-08-21 18:00:54 +0200458inline GprsMs *BTS::ms_by_imsi(const char *imsi)
459{
460 return ms_store().get_ms(0, 0, imsi);
461}
462
Jacob Erlbecked2dbf62015-12-28 19:15:40 +0100463inline LListHead<gprs_rlcmac_tbf>& BTS::ul_tbfs()
464{
465 return m_ul_tbfs;
466}
467
468inline LListHead<gprs_rlcmac_tbf>& BTS::dl_tbfs()
469{
470 return m_dl_tbfs;
471}
472
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100473inline struct rate_ctr_group *BTS::rate_counters() const
474{
475 return m_ratectrs;
476}
477
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100478inline struct osmo_stat_item_group *BTS::stat_items() const
479{
480 return m_statg;
481}
482
Alexander Couzens7fdbf892016-05-21 19:45:23 +0200483#define CREATE_COUNT_ADD_INLINE(func_name, ctr_name) \
484 inline void BTS::func_name(int inc) {\
485 rate_ctr_add(&m_ratectrs->ctr[ctr_name], inc); \
486 }
487
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100488#define CREATE_COUNT_INLINE(func_name, ctr_name) \
489 inline void BTS::func_name() {\
490 rate_ctr_inc(&m_ratectrs->ctr[ctr_name]); \
491 }
Holger Hans Peter Freyther15877642013-10-27 09:50:15 +0100492
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100493CREATE_COUNT_INLINE(tbf_dl_created, CTR_TBF_DL_ALLOCATED)
494CREATE_COUNT_INLINE(tbf_dl_freed, CTR_TBF_DL_FREED)
Jacob Erlbeck0316dc62016-01-21 20:12:04 +0100495CREATE_COUNT_INLINE(tbf_dl_aborted, CTR_TBF_DL_ABORTED)
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100496CREATE_COUNT_INLINE(tbf_ul_created, CTR_TBF_UL_ALLOCATED)
497CREATE_COUNT_INLINE(tbf_ul_freed, CTR_TBF_UL_FREED)
Jacob Erlbeck0316dc62016-01-21 20:12:04 +0100498CREATE_COUNT_INLINE(tbf_ul_aborted, CTR_TBF_UL_ABORTED)
Holger Hans Peter Freytheraa35ba72013-11-13 15:02:50 +0100499CREATE_COUNT_INLINE(tbf_reused, CTR_TBF_REUSED)
Jacob Erlbeck5979fe92015-07-14 14:02:41 +0200500CREATE_COUNT_INLINE(tbf_alloc_algo_a, CTR_TBF_ALLOC_ALGO_A)
501CREATE_COUNT_INLINE(tbf_alloc_algo_b, CTR_TBF_ALLOC_ALGO_B)
Jacob Erlbeck9b3d7e02016-01-19 10:44:42 +0100502CREATE_COUNT_INLINE(tbf_failed_egprs_only, CTR_TBF_FAILED_EGPRS_ONLY)
Holger Hans Peter Freythere9429b52013-11-13 19:36:57 +0100503CREATE_COUNT_INLINE(rlc_sent, CTR_RLC_SENT)
504CREATE_COUNT_INLINE(rlc_resent, CTR_RLC_RESENT)
Holger Hans Peter Freytheref93bdb2013-11-24 00:01:50 +0100505CREATE_COUNT_INLINE(rlc_restarted, CTR_RLC_RESTARTED)
Holger Hans Peter Freytherc70aae42013-11-19 17:09:37 +0100506CREATE_COUNT_INLINE(rlc_stalled, CTR_RLC_STALLED)
Holger Hans Peter Freyther092478f2013-11-23 01:01:19 +0100507CREATE_COUNT_INLINE(rlc_nacked, CTR_RLC_NACKED)
sivasankari168911b2016-11-25 19:53:36 +0530508CREATE_COUNT_INLINE(rlc_final_block_resent, CTR_RLC_FINAL_BLOCK_RESENT);
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200509CREATE_COUNT_INLINE(rlc_ass_timedout, CTR_RLC_ASS_TIMEDOUT);
Jacob Erlbeck7c8d39a2015-09-07 14:04:56 +0200510CREATE_COUNT_INLINE(rlc_ass_failed, CTR_RLC_ASS_FAILED);
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200511CREATE_COUNT_INLINE(rlc_ack_timedout, CTR_RLC_ACK_TIMEDOUT);
Jacob Erlbeck7c8d39a2015-09-07 14:04:56 +0200512CREATE_COUNT_INLINE(rlc_ack_failed, CTR_RLC_ACK_FAILED);
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200513CREATE_COUNT_INLINE(rlc_rel_timedout, CTR_RLC_REL_TIMEDOUT);
Jacob Erlbeckbe4a08b2015-08-25 15:19:31 +0200514CREATE_COUNT_INLINE(rlc_late_block, CTR_RLC_LATE_BLOCK);
Alexander Couzens2cb15472016-05-18 16:43:10 +0200515CREATE_COUNT_INLINE(rlc_sent_dummy, CTR_RLC_SENT_DUMMY);
Alexander Couzensf929e622016-05-22 00:02:56 +0200516CREATE_COUNT_INLINE(rlc_sent_control, CTR_RLC_SENT_CONTROL);
Alexander Couzens7fdbf892016-05-21 19:45:23 +0200517CREATE_COUNT_ADD_INLINE(rlc_dl_bytes, CTR_RLC_DL_BYTES);
518CREATE_COUNT_ADD_INLINE(rlc_dl_payload_bytes, CTR_RLC_DL_PAYLOAD_BYTES);
519CREATE_COUNT_ADD_INLINE(rlc_ul_bytes, CTR_RLC_UL_BYTES);
520CREATE_COUNT_ADD_INLINE(rlc_ul_payload_bytes, CTR_RLC_UL_PAYLOAD_BYTES);
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100521CREATE_COUNT_INLINE(decode_error, CTR_DECODE_ERRORS)
522CREATE_COUNT_INLINE(sba_allocated, CTR_SBA_ALLOCATED)
523CREATE_COUNT_INLINE(sba_freed, CTR_SBA_FREED)
524CREATE_COUNT_INLINE(sba_timedout, CTR_SBA_TIMEDOUT)
Holger Hans Peter Freytherb3d5ee22013-11-13 16:43:26 +0100525CREATE_COUNT_INLINE(llc_timedout_frame, CTR_LLC_FRAME_TIMEDOUT);
526CREATE_COUNT_INLINE(llc_dropped_frame, CTR_LLC_FRAME_DROPPED);
527CREATE_COUNT_INLINE(llc_frame_sched, CTR_LLC_FRAME_SCHED);
Alexander Couzens7fdbf892016-05-21 19:45:23 +0200528CREATE_COUNT_ADD_INLINE(llc_dl_bytes, CTR_LLC_DL_BYTES);
529CREATE_COUNT_ADD_INLINE(llc_ul_bytes, CTR_LLC_UL_BYTES);
Holger Hans Peter Freytherc1ae2262013-10-27 10:50:35 +0100530CREATE_COUNT_INLINE(rach_frame, CTR_RACH_REQUESTS);
sivasankari67b89ca2016-12-29 16:25:30 +0530531CREATE_COUNT_INLINE(rach_frame_11bit, CTR_11BIT_RACH_REQUESTS);
sivasankarida7250a2016-12-16 12:57:18 +0530532CREATE_COUNT_INLINE(spb_uplink_first_segment, CTR_SPB_UL_FIRST_SEGMENT);
533CREATE_COUNT_INLINE(spb_uplink_second_segment, CTR_SPB_UL_SECOND_SEGMENT);
534CREATE_COUNT_INLINE(spb_downlink_first_segment, CTR_SPB_DL_FIRST_SEGMENT);
535CREATE_COUNT_INLINE(spb_downlink_second_segment, CTR_SPB_DL_SECOND_SEGMENT);
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530536CREATE_COUNT_INLINE(immediate_assignment_ul_tbf, CTR_IMMEDIATE_ASSIGN_UL_TBF);
sivasankari168911b2016-11-25 19:53:36 +0530537CREATE_COUNT_INLINE(immediate_assignment_reject, CTR_IMMEDIATE_ASSIGN_REJ);
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530538CREATE_COUNT_INLINE(immediate_assignment_dl_tbf, CTR_IMMEDIATE_ASSIGN_DL_TBF);
sivasankari168911b2016-11-25 19:53:36 +0530539CREATE_COUNT_INLINE(channel_request_description, CTR_CHANNEL_REQUEST_DESCRIPTION);
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530540CREATE_COUNT_INLINE(pkt_ul_assignment, CTR_PKT_UL_ASSIGNMENT);
sivasankari168911b2016-11-25 19:53:36 +0530541CREATE_COUNT_INLINE(pkt_access_reject, CTR_PKT_ACCESS_REJ);
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530542CREATE_COUNT_INLINE(pkt_dl_assignemnt, CTR_PKT_DL_ASSIGNMENT);
543CREATE_COUNT_INLINE(rlc_rcvd_control, CTR_RLC_RECV_CONTROL);
544CREATE_COUNT_INLINE(pua_poll_timedout, CTR_PUA_POLL_TIMEDOUT);
545CREATE_COUNT_INLINE(pua_poll_failed, CTR_PUA_POLL_FAILED);
546CREATE_COUNT_INLINE(pda_poll_timedout, CTR_PDA_POLL_TIMEDOUT);
547CREATE_COUNT_INLINE(pda_poll_failed, CTR_PDA_POLL_FAILED);
548CREATE_COUNT_INLINE(pkt_ul_ack_nack_poll_timedout, CTR_PUAN_POLL_TIMEDOUT);
549CREATE_COUNT_INLINE(pkt_ul_ack_nack_poll_failed, CTR_PUAN_POLL_FAILED);
550CREATE_COUNT_INLINE(pkt_dl_ack_nack_poll_timedout, CTR_PDAN_POLL_TIMEDOUT);
551CREATE_COUNT_INLINE(pkt_dl_ack_nack_poll_failed, CTR_PDAN_POLL_FAILED);
552CREATE_COUNT_INLINE(gprs_dl_cs1, CTR_GPRS_DL_CS1);
553CREATE_COUNT_INLINE(gprs_dl_cs2, CTR_GPRS_DL_CS2);
554CREATE_COUNT_INLINE(gprs_dl_cs3, CTR_GPRS_DL_CS3);
555CREATE_COUNT_INLINE(gprs_dl_cs4, CTR_GPRS_DL_CS4);
556CREATE_COUNT_INLINE(egprs_dl_mcs1, CTR_EGPRS_DL_MCS1);
557CREATE_COUNT_INLINE(egprs_dl_mcs2, CTR_EGPRS_DL_MCS2);
558CREATE_COUNT_INLINE(egprs_dl_mcs3, CTR_EGPRS_DL_MCS3);
559CREATE_COUNT_INLINE(egprs_dl_mcs4, CTR_EGPRS_DL_MCS4);
560CREATE_COUNT_INLINE(egprs_dl_mcs5, CTR_EGPRS_DL_MCS5);
561CREATE_COUNT_INLINE(egprs_dl_mcs6, CTR_EGPRS_DL_MCS6);
562CREATE_COUNT_INLINE(egprs_dl_mcs7, CTR_EGPRS_DL_MCS7);
563CREATE_COUNT_INLINE(egprs_dl_mcs8, CTR_EGPRS_DL_MCS8);
564CREATE_COUNT_INLINE(egprs_dl_mcs9, CTR_EGPRS_DL_MCS9);
565CREATE_COUNT_INLINE(gprs_ul_cs1, CTR_GPRS_UL_CS1);
566CREATE_COUNT_INLINE(gprs_ul_cs2, CTR_GPRS_UL_CS2);
567CREATE_COUNT_INLINE(gprs_ul_cs3, CTR_GPRS_UL_CS3);
568CREATE_COUNT_INLINE(gprs_ul_cs4, CTR_GPRS_UL_CS4);
569CREATE_COUNT_INLINE(egprs_ul_mcs1, CTR_EGPRS_UL_MCS1);
570CREATE_COUNT_INLINE(egprs_ul_mcs2, CTR_EGPRS_UL_MCS2);
571CREATE_COUNT_INLINE(egprs_ul_mcs3, CTR_EGPRS_UL_MCS3);
572CREATE_COUNT_INLINE(egprs_ul_mcs4, CTR_EGPRS_UL_MCS4);
573CREATE_COUNT_INLINE(egprs_ul_mcs5, CTR_EGPRS_UL_MCS5);
574CREATE_COUNT_INLINE(egprs_ul_mcs6, CTR_EGPRS_UL_MCS6);
575CREATE_COUNT_INLINE(egprs_ul_mcs7, CTR_EGPRS_UL_MCS7);
576CREATE_COUNT_INLINE(egprs_ul_mcs8, CTR_EGPRS_UL_MCS8);
577CREATE_COUNT_INLINE(egprs_ul_mcs9, CTR_EGPRS_UL_MCS9);
Holger Hans Peter Freyther15877642013-10-27 09:50:15 +0100578
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100579#undef CREATE_COUNT_INLINE
Holger Hans Peter Freyther15877642013-10-27 09:50:15 +0100580
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100581#define CREATE_STAT_INLINE(func_name, func_name_get, stat_name) \
582 inline void BTS::func_name(int32_t val) {\
583 osmo_stat_item_set(m_statg->items[stat_name], val); \
584 } \
585 inline int32_t BTS::func_name_get() {\
586 return osmo_stat_item_get_last(m_statg->items[stat_name]); \
587 }
588
589CREATE_STAT_INLINE(ms_present, ms_present_get, STAT_MS_PRESENT);
590
591#undef CREATE_STAT_INLINE
Holger Hans Peter Freyther15877642013-10-27 09:50:15 +0100592
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200593#endif
594
595#ifdef __cplusplus
596extern "C" {
597#endif
Maxd5ffeb52019-03-18 15:48:38 +0100598 void bts_cleanup();
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200599 struct gprs_rlcmac_bts *bts_main_data();
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100600 struct rate_ctr_group *bts_main_data_stats();
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100601 struct osmo_stat_item_group *bts_main_data_stat_items();
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200602#ifdef __cplusplus
603}
Holger Hans Peter Freyther17b0d832013-10-19 17:37:48 +0200604
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200605#endif