blob: c2cc31638f91bf4fd21a54554f849a7a1714c08b [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);
Holger Hans Peter Freyther40cfaa62013-10-26 19:49:16 +0200304
Maxfc8afc22019-02-18 18:52:38 +0100305 void snd_dl_ass(gprs_rlcmac_tbf *tbf, bool poll, const char *imsi);
Holger Hans Peter Freytherd9262b32013-10-26 20:12:59 +0200306
Jacob Erlbecke43460b2015-05-13 13:33:12 +0200307 GprsMsStorage &ms_store();
308 GprsMs *ms_by_tlli(uint32_t tlli, uint32_t old_tlli = 0);
Jacob Erlbeck076f5c72015-08-21 18:00:54 +0200309 GprsMs *ms_by_imsi(const char *imsi);
Jacob Erlbeck86b6f052015-11-27 15:17:34 +0100310 GprsMs *ms_alloc(uint8_t ms_class, uint8_t egprs_ms_class = 0);
Jacob Erlbecke43460b2015-05-13 13:33:12 +0200311
Harald Welte717cdf52017-07-21 21:56:23 +0200312 void send_gsmtap(enum pcu_gsmtap_category categ, bool uplink, uint8_t trx_no,
313 uint8_t ts_no, uint8_t channel, uint32_t fn,
314 const uint8_t *data, unsigned int len);
315
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100316 /*
317 * Statistics
318 */
319 void tbf_dl_created();
320 void tbf_dl_freed();
Jacob Erlbeck0316dc62016-01-21 20:12:04 +0100321 void tbf_dl_aborted();
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100322 void tbf_ul_created();
323 void tbf_ul_freed();
Jacob Erlbeck0316dc62016-01-21 20:12:04 +0100324 void tbf_ul_aborted();
Holger Hans Peter Freytheraa35ba72013-11-13 15:02:50 +0100325 void tbf_reused();
Jacob Erlbeck5979fe92015-07-14 14:02:41 +0200326 void tbf_alloc_algo_a();
327 void tbf_alloc_algo_b();
Jacob Erlbeck9b3d7e02016-01-19 10:44:42 +0100328 void tbf_failed_egprs_only();
Holger Hans Peter Freythere9429b52013-11-13 19:36:57 +0100329 void rlc_sent();
330 void rlc_resent();
Holger Hans Peter Freytheref93bdb2013-11-24 00:01:50 +0100331 void rlc_restarted();
Holger Hans Peter Freytherc70aae42013-11-19 17:09:37 +0100332 void rlc_stalled();
Holger Hans Peter Freyther092478f2013-11-23 01:01:19 +0100333 void rlc_nacked();
sivasankari168911b2016-11-25 19:53:36 +0530334 void rlc_final_block_resent();
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200335 void rlc_ass_timedout();
Jacob Erlbeck7c8d39a2015-09-07 14:04:56 +0200336 void rlc_ass_failed();
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200337 void rlc_ack_timedout();
Jacob Erlbeck7c8d39a2015-09-07 14:04:56 +0200338 void rlc_ack_failed();
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200339 void rlc_rel_timedout();
Jacob Erlbeckbe4a08b2015-08-25 15:19:31 +0200340 void rlc_late_block();
Alexander Couzens2cb15472016-05-18 16:43:10 +0200341 void rlc_sent_dummy();
Alexander Couzensf929e622016-05-22 00:02:56 +0200342 void rlc_sent_control();
Alexander Couzens7fdbf892016-05-21 19:45:23 +0200343 void rlc_dl_bytes(int bytes);
344 void rlc_dl_payload_bytes(int bytes);
345 void rlc_ul_bytes(int bytes);
346 void rlc_ul_payload_bytes(int bytes);
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100347 void decode_error();
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100348 void sba_allocated();
349 void sba_freed();
350 void sba_timedout();
Holger Hans Peter Freytherb3d5ee22013-11-13 16:43:26 +0100351 void llc_timedout_frame();
352 void llc_dropped_frame();
353 void llc_frame_sched();
Alexander Couzens7fdbf892016-05-21 19:45:23 +0200354 void llc_dl_bytes(int bytes);
355 void llc_ul_bytes(int bytes);
Holger Hans Peter Freytherc1ae2262013-10-27 10:50:35 +0100356 void rach_frame();
sivasankari67b89ca2016-12-29 16:25:30 +0530357 void rach_frame_11bit();
sivasankarida7250a2016-12-16 12:57:18 +0530358 void spb_uplink_first_segment();
359 void spb_uplink_second_segment();
360 void spb_downlink_first_segment();
361 void spb_downlink_second_segment();
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530362 void immediate_assignment_ul_tbf();
sivasankari168911b2016-11-25 19:53:36 +0530363 void immediate_assignment_reject();
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530364 void immediate_assignment_dl_tbf();
sivasankari168911b2016-11-25 19:53:36 +0530365 void channel_request_description();
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530366 void pkt_ul_assignment();
sivasankari168911b2016-11-25 19:53:36 +0530367 void pkt_access_reject();
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530368 void pkt_dl_assignemnt();
369 void rlc_rcvd_control();
370 void pua_poll_timedout();
371 void pua_poll_failed();
372 void pda_poll_timedout();
373 void pda_poll_failed();
374 void pkt_ul_ack_nack_poll_timedout();
375 void pkt_ul_ack_nack_poll_failed();
376 void pkt_dl_ack_nack_poll_timedout();
377 void pkt_dl_ack_nack_poll_failed();
378 void gprs_dl_cs1();
379 void gprs_dl_cs2();
380 void gprs_dl_cs3();
381 void gprs_dl_cs4();
382 void egprs_dl_mcs1();
383 void egprs_dl_mcs2();
384 void egprs_dl_mcs3();
385 void egprs_dl_mcs4();
386 void egprs_dl_mcs5();
387 void egprs_dl_mcs6();
388 void egprs_dl_mcs7();
389 void egprs_dl_mcs8();
390 void egprs_dl_mcs9();
391 void gprs_ul_cs1();
392 void gprs_ul_cs2();
393 void gprs_ul_cs3();
394 void gprs_ul_cs4();
395 void egprs_ul_mcs1();
396 void egprs_ul_mcs2();
397 void egprs_ul_mcs3();
398 void egprs_ul_mcs4();
399 void egprs_ul_mcs5();
400 void egprs_ul_mcs6();
401 void egprs_ul_mcs7();
402 void egprs_ul_mcs8();
403 void egprs_ul_mcs9();
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100404
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100405 void ms_present(int32_t n);
406 int32_t ms_present_get();
407
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100408 /*
409 * Below for C interface for the VTY
410 */
411 struct rate_ctr_group *rate_counters() const;
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100412 struct osmo_stat_item_group *stat_items() const;
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100413
Jacob Erlbecked2dbf62015-12-28 19:15:40 +0100414 LListHead<gprs_rlcmac_tbf>& ul_tbfs();
415 LListHead<gprs_rlcmac_tbf>& dl_tbfs();
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200416private:
Holger Hans Peter Freyther9b30c7f2013-10-17 19:59:56 +0200417 int m_cur_fn;
Jacob Erlbeck60f77032015-08-24 14:35:14 +0200418 int m_cur_blk_fn;
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200419 struct gprs_rlcmac_bts m_bts;
Holger Hans Peter Freytherb78adcd2013-10-17 20:12:37 +0200420 PollController m_pollController;
Holger Hans Peter Freythercedf8902013-10-19 20:47:12 +0200421 SBAController m_sba;
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100422 struct rate_ctr_group *m_ratectrs;
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100423 struct osmo_stat_item_group *m_statg;
Holger Hans Peter Freytherb78adcd2013-10-17 20:12:37 +0200424
Jacob Erlbecke43460b2015-05-13 13:33:12 +0200425 GprsMsStorage m_ms_store;
426
Jacob Erlbecked2dbf62015-12-28 19:15:40 +0100427 /* list of uplink TBFs */
428 LListHead<gprs_rlcmac_tbf> m_ul_tbfs;
429 /* list of downlink TBFs */
430 LListHead<gprs_rlcmac_tbf> m_dl_tbfs;
431
Holger Hans Peter Freytherb78adcd2013-10-17 20:12:37 +0200432 /* disable copying to avoid slicing */
433 BTS(const BTS&);
434 BTS& operator=(const BTS&);
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200435};
Holger Hans Peter Freyther9b30c7f2013-10-17 19:59:56 +0200436
437inline int BTS::current_frame_number() const
438{
439 return m_cur_fn;
440}
Holger Hans Peter Freyther111614a2013-10-19 20:04:57 +0200441
Holger Hans Peter Freythercedf8902013-10-19 20:47:12 +0200442inline SBAController *BTS::sba()
443{
444 return &m_sba;
445}
Holger Hans Peter Freyther65be4802013-10-26 18:39:36 +0200446
Jacob Erlbecke43460b2015-05-13 13:33:12 +0200447inline GprsMsStorage &BTS::ms_store()
448{
449 return m_ms_store;
450}
451
452inline GprsMs *BTS::ms_by_tlli(uint32_t tlli, uint32_t old_tlli)
453{
454 return ms_store().get_ms(tlli, old_tlli);
455}
456
Jacob Erlbeck076f5c72015-08-21 18:00:54 +0200457inline GprsMs *BTS::ms_by_imsi(const char *imsi)
458{
459 return ms_store().get_ms(0, 0, imsi);
460}
461
Jacob Erlbecked2dbf62015-12-28 19:15:40 +0100462inline LListHead<gprs_rlcmac_tbf>& BTS::ul_tbfs()
463{
464 return m_ul_tbfs;
465}
466
467inline LListHead<gprs_rlcmac_tbf>& BTS::dl_tbfs()
468{
469 return m_dl_tbfs;
470}
471
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100472inline struct rate_ctr_group *BTS::rate_counters() const
473{
474 return m_ratectrs;
475}
476
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100477inline struct osmo_stat_item_group *BTS::stat_items() const
478{
479 return m_statg;
480}
481
Alexander Couzens7fdbf892016-05-21 19:45:23 +0200482#define CREATE_COUNT_ADD_INLINE(func_name, ctr_name) \
483 inline void BTS::func_name(int inc) {\
484 rate_ctr_add(&m_ratectrs->ctr[ctr_name], inc); \
485 }
486
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100487#define CREATE_COUNT_INLINE(func_name, ctr_name) \
488 inline void BTS::func_name() {\
489 rate_ctr_inc(&m_ratectrs->ctr[ctr_name]); \
490 }
Holger Hans Peter Freyther15877642013-10-27 09:50:15 +0100491
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100492CREATE_COUNT_INLINE(tbf_dl_created, CTR_TBF_DL_ALLOCATED)
493CREATE_COUNT_INLINE(tbf_dl_freed, CTR_TBF_DL_FREED)
Jacob Erlbeck0316dc62016-01-21 20:12:04 +0100494CREATE_COUNT_INLINE(tbf_dl_aborted, CTR_TBF_DL_ABORTED)
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100495CREATE_COUNT_INLINE(tbf_ul_created, CTR_TBF_UL_ALLOCATED)
496CREATE_COUNT_INLINE(tbf_ul_freed, CTR_TBF_UL_FREED)
Jacob Erlbeck0316dc62016-01-21 20:12:04 +0100497CREATE_COUNT_INLINE(tbf_ul_aborted, CTR_TBF_UL_ABORTED)
Holger Hans Peter Freytheraa35ba72013-11-13 15:02:50 +0100498CREATE_COUNT_INLINE(tbf_reused, CTR_TBF_REUSED)
Jacob Erlbeck5979fe92015-07-14 14:02:41 +0200499CREATE_COUNT_INLINE(tbf_alloc_algo_a, CTR_TBF_ALLOC_ALGO_A)
500CREATE_COUNT_INLINE(tbf_alloc_algo_b, CTR_TBF_ALLOC_ALGO_B)
Jacob Erlbeck9b3d7e02016-01-19 10:44:42 +0100501CREATE_COUNT_INLINE(tbf_failed_egprs_only, CTR_TBF_FAILED_EGPRS_ONLY)
Holger Hans Peter Freythere9429b52013-11-13 19:36:57 +0100502CREATE_COUNT_INLINE(rlc_sent, CTR_RLC_SENT)
503CREATE_COUNT_INLINE(rlc_resent, CTR_RLC_RESENT)
Holger Hans Peter Freytheref93bdb2013-11-24 00:01:50 +0100504CREATE_COUNT_INLINE(rlc_restarted, CTR_RLC_RESTARTED)
Holger Hans Peter Freytherc70aae42013-11-19 17:09:37 +0100505CREATE_COUNT_INLINE(rlc_stalled, CTR_RLC_STALLED)
Holger Hans Peter Freyther092478f2013-11-23 01:01:19 +0100506CREATE_COUNT_INLINE(rlc_nacked, CTR_RLC_NACKED)
sivasankari168911b2016-11-25 19:53:36 +0530507CREATE_COUNT_INLINE(rlc_final_block_resent, CTR_RLC_FINAL_BLOCK_RESENT);
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200508CREATE_COUNT_INLINE(rlc_ass_timedout, CTR_RLC_ASS_TIMEDOUT);
Jacob Erlbeck7c8d39a2015-09-07 14:04:56 +0200509CREATE_COUNT_INLINE(rlc_ass_failed, CTR_RLC_ASS_FAILED);
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200510CREATE_COUNT_INLINE(rlc_ack_timedout, CTR_RLC_ACK_TIMEDOUT);
Jacob Erlbeck7c8d39a2015-09-07 14:04:56 +0200511CREATE_COUNT_INLINE(rlc_ack_failed, CTR_RLC_ACK_FAILED);
Jacob Erlbeckc91c18e2015-07-07 09:33:29 +0200512CREATE_COUNT_INLINE(rlc_rel_timedout, CTR_RLC_REL_TIMEDOUT);
Jacob Erlbeckbe4a08b2015-08-25 15:19:31 +0200513CREATE_COUNT_INLINE(rlc_late_block, CTR_RLC_LATE_BLOCK);
Alexander Couzens2cb15472016-05-18 16:43:10 +0200514CREATE_COUNT_INLINE(rlc_sent_dummy, CTR_RLC_SENT_DUMMY);
Alexander Couzensf929e622016-05-22 00:02:56 +0200515CREATE_COUNT_INLINE(rlc_sent_control, CTR_RLC_SENT_CONTROL);
Alexander Couzens7fdbf892016-05-21 19:45:23 +0200516CREATE_COUNT_ADD_INLINE(rlc_dl_bytes, CTR_RLC_DL_BYTES);
517CREATE_COUNT_ADD_INLINE(rlc_dl_payload_bytes, CTR_RLC_DL_PAYLOAD_BYTES);
518CREATE_COUNT_ADD_INLINE(rlc_ul_bytes, CTR_RLC_UL_BYTES);
519CREATE_COUNT_ADD_INLINE(rlc_ul_payload_bytes, CTR_RLC_UL_PAYLOAD_BYTES);
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100520CREATE_COUNT_INLINE(decode_error, CTR_DECODE_ERRORS)
521CREATE_COUNT_INLINE(sba_allocated, CTR_SBA_ALLOCATED)
522CREATE_COUNT_INLINE(sba_freed, CTR_SBA_FREED)
523CREATE_COUNT_INLINE(sba_timedout, CTR_SBA_TIMEDOUT)
Holger Hans Peter Freytherb3d5ee22013-11-13 16:43:26 +0100524CREATE_COUNT_INLINE(llc_timedout_frame, CTR_LLC_FRAME_TIMEDOUT);
525CREATE_COUNT_INLINE(llc_dropped_frame, CTR_LLC_FRAME_DROPPED);
526CREATE_COUNT_INLINE(llc_frame_sched, CTR_LLC_FRAME_SCHED);
Alexander Couzens7fdbf892016-05-21 19:45:23 +0200527CREATE_COUNT_ADD_INLINE(llc_dl_bytes, CTR_LLC_DL_BYTES);
528CREATE_COUNT_ADD_INLINE(llc_ul_bytes, CTR_LLC_UL_BYTES);
Holger Hans Peter Freytherc1ae2262013-10-27 10:50:35 +0100529CREATE_COUNT_INLINE(rach_frame, CTR_RACH_REQUESTS);
sivasankari67b89ca2016-12-29 16:25:30 +0530530CREATE_COUNT_INLINE(rach_frame_11bit, CTR_11BIT_RACH_REQUESTS);
sivasankarida7250a2016-12-16 12:57:18 +0530531CREATE_COUNT_INLINE(spb_uplink_first_segment, CTR_SPB_UL_FIRST_SEGMENT);
532CREATE_COUNT_INLINE(spb_uplink_second_segment, CTR_SPB_UL_SECOND_SEGMENT);
533CREATE_COUNT_INLINE(spb_downlink_first_segment, CTR_SPB_DL_FIRST_SEGMENT);
534CREATE_COUNT_INLINE(spb_downlink_second_segment, CTR_SPB_DL_SECOND_SEGMENT);
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530535CREATE_COUNT_INLINE(immediate_assignment_ul_tbf, CTR_IMMEDIATE_ASSIGN_UL_TBF);
sivasankari168911b2016-11-25 19:53:36 +0530536CREATE_COUNT_INLINE(immediate_assignment_reject, CTR_IMMEDIATE_ASSIGN_REJ);
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530537CREATE_COUNT_INLINE(immediate_assignment_dl_tbf, CTR_IMMEDIATE_ASSIGN_DL_TBF);
sivasankari168911b2016-11-25 19:53:36 +0530538CREATE_COUNT_INLINE(channel_request_description, CTR_CHANNEL_REQUEST_DESCRIPTION);
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530539CREATE_COUNT_INLINE(pkt_ul_assignment, CTR_PKT_UL_ASSIGNMENT);
sivasankari168911b2016-11-25 19:53:36 +0530540CREATE_COUNT_INLINE(pkt_access_reject, CTR_PKT_ACCESS_REJ);
Mrinal Mishraf86307e2016-11-10 18:16:30 +0530541CREATE_COUNT_INLINE(pkt_dl_assignemnt, CTR_PKT_DL_ASSIGNMENT);
542CREATE_COUNT_INLINE(rlc_rcvd_control, CTR_RLC_RECV_CONTROL);
543CREATE_COUNT_INLINE(pua_poll_timedout, CTR_PUA_POLL_TIMEDOUT);
544CREATE_COUNT_INLINE(pua_poll_failed, CTR_PUA_POLL_FAILED);
545CREATE_COUNT_INLINE(pda_poll_timedout, CTR_PDA_POLL_TIMEDOUT);
546CREATE_COUNT_INLINE(pda_poll_failed, CTR_PDA_POLL_FAILED);
547CREATE_COUNT_INLINE(pkt_ul_ack_nack_poll_timedout, CTR_PUAN_POLL_TIMEDOUT);
548CREATE_COUNT_INLINE(pkt_ul_ack_nack_poll_failed, CTR_PUAN_POLL_FAILED);
549CREATE_COUNT_INLINE(pkt_dl_ack_nack_poll_timedout, CTR_PDAN_POLL_TIMEDOUT);
550CREATE_COUNT_INLINE(pkt_dl_ack_nack_poll_failed, CTR_PDAN_POLL_FAILED);
551CREATE_COUNT_INLINE(gprs_dl_cs1, CTR_GPRS_DL_CS1);
552CREATE_COUNT_INLINE(gprs_dl_cs2, CTR_GPRS_DL_CS2);
553CREATE_COUNT_INLINE(gprs_dl_cs3, CTR_GPRS_DL_CS3);
554CREATE_COUNT_INLINE(gprs_dl_cs4, CTR_GPRS_DL_CS4);
555CREATE_COUNT_INLINE(egprs_dl_mcs1, CTR_EGPRS_DL_MCS1);
556CREATE_COUNT_INLINE(egprs_dl_mcs2, CTR_EGPRS_DL_MCS2);
557CREATE_COUNT_INLINE(egprs_dl_mcs3, CTR_EGPRS_DL_MCS3);
558CREATE_COUNT_INLINE(egprs_dl_mcs4, CTR_EGPRS_DL_MCS4);
559CREATE_COUNT_INLINE(egprs_dl_mcs5, CTR_EGPRS_DL_MCS5);
560CREATE_COUNT_INLINE(egprs_dl_mcs6, CTR_EGPRS_DL_MCS6);
561CREATE_COUNT_INLINE(egprs_dl_mcs7, CTR_EGPRS_DL_MCS7);
562CREATE_COUNT_INLINE(egprs_dl_mcs8, CTR_EGPRS_DL_MCS8);
563CREATE_COUNT_INLINE(egprs_dl_mcs9, CTR_EGPRS_DL_MCS9);
564CREATE_COUNT_INLINE(gprs_ul_cs1, CTR_GPRS_UL_CS1);
565CREATE_COUNT_INLINE(gprs_ul_cs2, CTR_GPRS_UL_CS2);
566CREATE_COUNT_INLINE(gprs_ul_cs3, CTR_GPRS_UL_CS3);
567CREATE_COUNT_INLINE(gprs_ul_cs4, CTR_GPRS_UL_CS4);
568CREATE_COUNT_INLINE(egprs_ul_mcs1, CTR_EGPRS_UL_MCS1);
569CREATE_COUNT_INLINE(egprs_ul_mcs2, CTR_EGPRS_UL_MCS2);
570CREATE_COUNT_INLINE(egprs_ul_mcs3, CTR_EGPRS_UL_MCS3);
571CREATE_COUNT_INLINE(egprs_ul_mcs4, CTR_EGPRS_UL_MCS4);
572CREATE_COUNT_INLINE(egprs_ul_mcs5, CTR_EGPRS_UL_MCS5);
573CREATE_COUNT_INLINE(egprs_ul_mcs6, CTR_EGPRS_UL_MCS6);
574CREATE_COUNT_INLINE(egprs_ul_mcs7, CTR_EGPRS_UL_MCS7);
575CREATE_COUNT_INLINE(egprs_ul_mcs8, CTR_EGPRS_UL_MCS8);
576CREATE_COUNT_INLINE(egprs_ul_mcs9, CTR_EGPRS_UL_MCS9);
Holger Hans Peter Freyther15877642013-10-27 09:50:15 +0100577
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100578#undef CREATE_COUNT_INLINE
Holger Hans Peter Freyther15877642013-10-27 09:50:15 +0100579
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100580#define CREATE_STAT_INLINE(func_name, func_name_get, stat_name) \
581 inline void BTS::func_name(int32_t val) {\
582 osmo_stat_item_set(m_statg->items[stat_name], val); \
583 } \
584 inline int32_t BTS::func_name_get() {\
585 return osmo_stat_item_get_last(m_statg->items[stat_name]); \
586 }
587
588CREATE_STAT_INLINE(ms_present, ms_present_get, STAT_MS_PRESENT);
589
590#undef CREATE_STAT_INLINE
Holger Hans Peter Freyther15877642013-10-27 09:50:15 +0100591
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200592#endif
593
594#ifdef __cplusplus
595extern "C" {
596#endif
Maxd5ffeb52019-03-18 15:48:38 +0100597 void bts_cleanup();
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200598 struct gprs_rlcmac_bts *bts_main_data();
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100599 struct rate_ctr_group *bts_main_data_stats();
Jacob Erlbeckf5898a02015-11-27 19:05:13 +0100600 struct osmo_stat_item_group *bts_main_data_stat_items();
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200601#ifdef __cplusplus
602}
Holger Hans Peter Freyther17b0d832013-10-19 17:37:48 +0200603
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200604#endif