blob: 0c5b47a096f2b0bd3587d29b8635355514196bb4 [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>
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +020028#include <osmocom/core/timer.h>
29}
Holger Hans Peter Freytherb78adcd2013-10-17 20:12:37 +020030
31#include "poll_controller.h"
Holger Hans Peter Freythercedf8902013-10-19 20:47:12 +020032#include "sba.h"
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +020033#include "tbf.h"
Jacob Erlbecke43460b2015-05-13 13:33:12 +020034#include "gprs_ms_storage.h"
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +020035#endif
36
37#include <stdint.h>
38
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +020039struct BTS;
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +020040
41/*
42 * PDCH instance
43 */
44struct gprs_rlcmac_pdch {
Holger Hans Peter Freyther17b0d832013-10-19 17:37:48 +020045#ifdef __cplusplus
Holger Hans Peter Freyther24e98d02013-10-19 18:15:44 +020046 struct gprs_rlcmac_paging *dequeue_paging();
47 struct msgb *packet_paging_request();
48
Holger Hans Peter Freytherf0984892013-10-19 18:28:59 +020049 void add_paging(struct gprs_rlcmac_paging *pag);
Holger Hans Peter Freyther24e98d02013-10-19 18:15:44 +020050
Holger Hans Peter Freyther09ef27a2013-10-20 16:37:05 +020051 void free_resources();
Holger Hans Peter Freyther17b0d832013-10-19 17:37:48 +020052
53 bool is_enabled() const;
54
55 void enable();
56 void disable();
Holger Hans Peter Freyther9ae367f2013-10-26 16:42:38 +020057
58 /* dispatching of messages */
Jacob Erlbeck20f6fd12015-06-08 11:05:45 +020059 int rcv_block(uint8_t *data, uint8_t len, uint32_t fn,
60 struct pcu_l1_meas *meas);
Holger Hans Peter Freyther65be4802013-10-26 18:39:36 +020061
62 gprs_rlcmac_bts *bts_data() const;
63 BTS *bts() const;
Holger Hans Peter Freyther05f8efc2013-10-26 18:53:16 +020064 uint8_t trx_no() const;
Daniel Willmann1e0c6102014-06-04 14:56:09 +020065
Daniel Willmannfe6e2e42014-07-10 17:44:06 +020066 struct gprs_rlcmac_ul_tbf *ul_tbf_by_tfi(uint8_t tfi);
67 struct gprs_rlcmac_dl_tbf *dl_tbf_by_tfi(uint8_t tfi);
Holger Hans Peter Freyther17b0d832013-10-19 17:37:48 +020068#endif
69
70 uint8_t m_is_enabled; /* TS is enabled */
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +020071 uint8_t tsc; /* TSC of this slot */
72 uint8_t next_ul_tfi; /* next uplink TBF/TFI to schedule (0..31) */
73 uint8_t next_dl_tfi; /* next downlink TBF/TFI to schedule (0..31) */
Holger Hans Peter Freyther705653b2013-11-26 16:39:28 +010074 uint8_t next_ctrl_prio; /* next kind of ctrl message to schedule */
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +020075 struct llist_head paging_list; /* list of paging messages */
76 uint32_t last_rts_fn; /* store last frame number of RTS */
Holger Hans Peter Freyther4ed1dae2013-10-20 10:14:03 +020077
78 /* back pointers */
79 struct gprs_rlcmac_trx *trx;
80 uint8_t ts_no;
Holger Hans Peter Freytherd11290b2013-10-26 17:32:04 +020081
82#ifdef __cplusplus
83private:
Jacob Erlbeck20f6fd12015-06-08 11:05:45 +020084 int rcv_data_block_acknowledged(uint8_t *data, uint8_t len,
85 struct pcu_l1_meas *meas);
Holger Hans Peter Freyther05f8efc2013-10-26 18:53:16 +020086 int rcv_control_block(bitvec *rlc_block, uint32_t fn);
Holger Hans Peter Freytherd11290b2013-10-26 17:32:04 +020087
Holger Hans Peter Freythere1a075a2013-10-27 13:35:30 +010088 void rcv_control_ack(Packet_Control_Acknowledgement_t *, uint32_t fn);
Holger Hans Peter Freyther5da20142013-10-27 11:41:01 +010089 void rcv_control_dl_ack_nack(Packet_Downlink_Ack_Nack_t *, uint32_t fn);
Holger Hans Peter Freyther7a344712013-10-27 11:37:53 +010090 void rcv_resource_request(Packet_Resource_Request_t *t, uint32_t fn);
Holger Hans Peter Freythercb5c49b2013-10-27 11:19:13 +010091 void rcv_measurement_report(Packet_Measurement_Report_t *t, uint32_t fn);
Daniel Willmann2207c5e2014-07-10 17:44:06 +020092 gprs_rlcmac_tbf *tbf_from_list_by_tfi(struct llist_head *tbf_list, uint8_t tfi,
93 enum gprs_rlcmac_tbf_direction dir);
Holger Hans Peter Freytherd11290b2013-10-26 17:32:04 +020094#endif
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +020095};
96
97struct gprs_rlcmac_trx {
98 void *fl1h;
99 uint16_t arfcn;
100 struct gprs_rlcmac_pdch pdch[8];
101 struct gprs_rlcmac_tbf *ul_tbf[32]; /* array of UL TBF, by UL TFI */
102 struct gprs_rlcmac_tbf *dl_tbf[32]; /* array of DL TBF, by DL TFI */
Holger Hans Peter Freyther4ed1dae2013-10-20 10:14:03 +0200103
104 /* back pointers */
105 struct BTS *bts;
106 uint8_t trx_no;
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200107};
108
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200109/**
110 * This is the data from C. As soon as our minimal compiler is gcc 4.7
111 * we can start to compile pcu_vty.c with c++ and remove the split.
112 */
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200113struct gprs_rlcmac_bts {
114 uint8_t bsic;
115 uint8_t fc_interval;
Jacob Erlbeck0288cdb2015-05-06 10:47:30 +0200116 uint16_t fc_bucket_time;
Jacob Erlbeck87d73412015-04-21 12:56:48 +0200117 uint32_t fc_bvc_bucket_size;
118 uint32_t fc_bvc_leak_rate;
119 uint32_t fc_ms_bucket_size;
120 uint32_t fc_ms_leak_rate;
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200121 uint8_t cs1;
122 uint8_t cs2;
123 uint8_t cs3;
124 uint8_t cs4;
125 uint8_t initial_cs_dl, initial_cs_ul;
Jacob Erlbeckb33e6752015-06-04 19:04:30 +0200126 uint8_t max_cs_dl, max_cs_ul;
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200127 uint8_t force_cs; /* 0=use from BTS 1=use from VTY */
128 uint16_t force_llc_lifetime; /* overrides lifetime from SGSN */
Jacob Erlbeck0c1c8772015-03-20 12:02:42 +0100129 uint32_t llc_discard_csec;
Jacob Erlbeckd0261b72015-04-02 13:58:09 +0200130 uint32_t llc_idle_ack_csec;
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200131 uint8_t t3142;
132 uint8_t t3169;
133 uint8_t t3191;
134 uint16_t t3193_msec;
135 uint8_t t3195;
136 uint8_t n3101;
137 uint8_t n3103;
138 uint8_t n3105;
139 struct gprs_rlcmac_trx trx[8];
140 int (*alloc_algorithm)(struct gprs_rlcmac_bts *bts,
141 struct gprs_rlcmac_tbf *old_tbf,
142 struct gprs_rlcmac_tbf *tbf, uint32_t cust, uint8_t single);
143 uint32_t alloc_algorithm_curst; /* options to customize algorithm */
144 uint8_t force_two_phase;
145 uint8_t alpha, gamma;
Jacob Erlbeck3bed5d12015-03-19 11:22:38 +0100146 uint32_t dl_tbf_idle_msec; /* hold time for idle DL TBFs */
Jacob Erlbecka098c192015-05-28 16:11:19 +0200147 uint32_t ms_idle_sec;
Jacob Erlbeck1751c622015-06-04 12:12:32 +0200148 uint8_t cs_adj_enabled;
149 uint8_t cs_adj_upper_limit;
150 uint8_t cs_adj_lower_limit;
Jacob Erlbeck94cde132015-06-09 09:44:36 +0200151 struct {int16_t low; int16_t high;} cs_lqual_ranges[4];
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200152
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +0200153 /* TBF handling, make private or move into TBFController */
154 /* list of uplink TBFs */
155 struct llist_head ul_tbfs;
156 /* list of downlink TBFs */
157 struct llist_head dl_tbfs;
158
159
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200160 /**
161 * Point back to the C++ object. This is used during the transition
162 * period.
163 */
164 struct BTS *bts;
Holger Hans Peter Freyther67ed34e2013-10-17 17:01:54 +0200165};
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200166
167#ifdef __cplusplus
168/**
169 * I represent a GSM BTS. I have one or more TRX, I know the current
170 * GSM time and I have controllers that help with allocating resources
171 * on my TRXs.
172 */
173struct BTS {
174public:
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100175 enum {
176 CTR_TBF_DL_ALLOCATED,
177 CTR_TBF_DL_FREED,
178 CTR_TBF_UL_ALLOCATED,
179 CTR_TBF_UL_FREED,
Holger Hans Peter Freytheraa35ba72013-11-13 15:02:50 +0100180 CTR_TBF_REUSED,
Holger Hans Peter Freythere9429b52013-11-13 19:36:57 +0100181 CTR_RLC_SENT,
182 CTR_RLC_RESENT,
Holger Hans Peter Freytheref93bdb2013-11-24 00:01:50 +0100183 CTR_RLC_RESTARTED,
Holger Hans Peter Freytherc70aae42013-11-19 17:09:37 +0100184 CTR_RLC_STALLED,
Holger Hans Peter Freyther092478f2013-11-23 01:01:19 +0100185 CTR_RLC_NACKED,
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100186 CTR_DECODE_ERRORS,
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100187 CTR_SBA_ALLOCATED,
188 CTR_SBA_FREED,
189 CTR_SBA_TIMEDOUT,
Holger Hans Peter Freyther19977872013-10-27 10:34:31 +0100190 CTR_LLC_FRAME_TIMEDOUT,
191 CTR_LLC_FRAME_DROPPED,
Holger Hans Peter Freytherb3d5ee22013-11-13 16:43:26 +0100192 CTR_LLC_FRAME_SCHED,
Holger Hans Peter Freytherc1ae2262013-10-27 10:50:35 +0100193 CTR_RACH_REQUESTS,
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100194 };
195
Jacob Erlbeck502bd1f2015-03-20 14:26:05 +0100196 enum {
197 TIMER_T3190_MSEC = 5000,
198 };
199
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200200 BTS();
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100201 ~BTS();
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200202
203 static BTS* main_bts();
204
205 struct gprs_rlcmac_bts *bts_data();
Holger Hans Peter Freythercedf8902013-10-19 20:47:12 +0200206 SBAController *sba();
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200207
Holger Hans Peter Freyther9b30c7f2013-10-17 19:59:56 +0200208 /** TODO: change the number to unsigned */
209 void set_current_frame_number(int frame_number);
210 int current_frame_number() const;
211
Holger Hans Peter Freytherf0984892013-10-19 18:28:59 +0200212 /** add paging to paging queue(s) */
213 int add_paging(uint8_t chan_needed, uint8_t *identity_lv);
214
Daniel Willmannfe6e2e42014-07-10 17:44:06 +0200215 gprs_rlcmac_dl_tbf *dl_tbf_by_poll_fn(uint32_t fn, uint8_t trx, uint8_t ts);
216 gprs_rlcmac_ul_tbf *ul_tbf_by_poll_fn(uint32_t fn, uint8_t trx, uint8_t ts);
217 gprs_rlcmac_dl_tbf *dl_tbf_by_tfi(uint8_t tfi, uint8_t trx);
218 gprs_rlcmac_ul_tbf *ul_tbf_by_tfi(uint8_t tfi, uint8_t trx);
Holger Hans Peter Freyther34bd8bd2013-10-19 21:10:38 +0200219
Holger Hans Peter Freyther70ddde62013-10-26 19:17:58 +0200220 int tfi_find_free(enum gprs_rlcmac_tbf_direction dir, uint8_t *_trx, int8_t use_trx);
221
Holger Hans Peter Freyther40cfaa62013-10-26 19:49:16 +0200222 int rcv_imm_ass_cnf(const uint8_t *data, uint32_t fn);
Holger Hans Peter Freyther02beed52013-10-26 20:56:20 +0200223 int rcv_rach(uint8_t ra, uint32_t Fn, int16_t qta);
Holger Hans Peter Freyther40cfaa62013-10-26 19:49:16 +0200224
Jacob Erlbeck71e55112015-05-21 11:10:15 +0200225 void trigger_dl_ass(gprs_rlcmac_dl_tbf *tbf, gprs_rlcmac_tbf *old_tbf);
Holger Hans Peter Freytherd9262b32013-10-26 20:12:59 +0200226 void snd_dl_ass(gprs_rlcmac_tbf *tbf, uint8_t poll, const char *imsi);
227
Jacob Erlbecke43460b2015-05-13 13:33:12 +0200228 GprsMsStorage &ms_store();
229 GprsMs *ms_by_tlli(uint32_t tlli, uint32_t old_tlli = 0);
230
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100231 /*
232 * Statistics
233 */
234 void tbf_dl_created();
235 void tbf_dl_freed();
236 void tbf_ul_created();
237 void tbf_ul_freed();
Holger Hans Peter Freytheraa35ba72013-11-13 15:02:50 +0100238 void tbf_reused();
Holger Hans Peter Freythere9429b52013-11-13 19:36:57 +0100239 void rlc_sent();
240 void rlc_resent();
Holger Hans Peter Freytheref93bdb2013-11-24 00:01:50 +0100241 void rlc_restarted();
Holger Hans Peter Freytherc70aae42013-11-19 17:09:37 +0100242 void rlc_stalled();
Holger Hans Peter Freyther092478f2013-11-23 01:01:19 +0100243 void rlc_nacked();
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100244 void decode_error();
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100245 void sba_allocated();
246 void sba_freed();
247 void sba_timedout();
Holger Hans Peter Freytherb3d5ee22013-11-13 16:43:26 +0100248 void llc_timedout_frame();
249 void llc_dropped_frame();
250 void llc_frame_sched();
Holger Hans Peter Freytherc1ae2262013-10-27 10:50:35 +0100251 void rach_frame();
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100252
253 /*
254 * Below for C interface for the VTY
255 */
256 struct rate_ctr_group *rate_counters() const;
257
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200258private:
Holger Hans Peter Freyther9b30c7f2013-10-17 19:59:56 +0200259 int m_cur_fn;
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200260 struct gprs_rlcmac_bts m_bts;
Holger Hans Peter Freytherb78adcd2013-10-17 20:12:37 +0200261 PollController m_pollController;
Holger Hans Peter Freythercedf8902013-10-19 20:47:12 +0200262 SBAController m_sba;
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100263 struct rate_ctr_group *m_ratectrs;
Daniel Willmann54044b02014-07-02 17:58:15 +0200264 gprs_rlcmac_tbf *tbf_by_tfi(uint8_t tfi, uint8_t trx, enum gprs_rlcmac_tbf_direction dir);
Holger Hans Peter Freytherb78adcd2013-10-17 20:12:37 +0200265
Jacob Erlbecke43460b2015-05-13 13:33:12 +0200266 GprsMsStorage m_ms_store;
267
Holger Hans Peter Freytherb78adcd2013-10-17 20:12:37 +0200268private:
269 /* disable copying to avoid slicing */
270 BTS(const BTS&);
271 BTS& operator=(const BTS&);
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200272};
Holger Hans Peter Freyther9b30c7f2013-10-17 19:59:56 +0200273
274inline int BTS::current_frame_number() const
275{
276 return m_cur_fn;
277}
Holger Hans Peter Freyther111614a2013-10-19 20:04:57 +0200278
Holger Hans Peter Freythercedf8902013-10-19 20:47:12 +0200279inline SBAController *BTS::sba()
280{
281 return &m_sba;
282}
Holger Hans Peter Freyther65be4802013-10-26 18:39:36 +0200283
Jacob Erlbecke43460b2015-05-13 13:33:12 +0200284inline GprsMsStorage &BTS::ms_store()
285{
286 return m_ms_store;
287}
288
289inline GprsMs *BTS::ms_by_tlli(uint32_t tlli, uint32_t old_tlli)
290{
291 return ms_store().get_ms(tlli, old_tlli);
292}
293
Holger Hans Peter Freyther65be4802013-10-26 18:39:36 +0200294inline BTS *gprs_rlcmac_pdch::bts() const
295{
296 return trx->bts;
297}
298
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100299inline struct rate_ctr_group *BTS::rate_counters() const
300{
301 return m_ratectrs;
302}
303
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100304#define CREATE_COUNT_INLINE(func_name, ctr_name) \
305 inline void BTS::func_name() {\
306 rate_ctr_inc(&m_ratectrs->ctr[ctr_name]); \
307 }
Holger Hans Peter Freyther15877642013-10-27 09:50:15 +0100308
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100309CREATE_COUNT_INLINE(tbf_dl_created, CTR_TBF_DL_ALLOCATED)
310CREATE_COUNT_INLINE(tbf_dl_freed, CTR_TBF_DL_FREED)
311CREATE_COUNT_INLINE(tbf_ul_created, CTR_TBF_UL_ALLOCATED)
312CREATE_COUNT_INLINE(tbf_ul_freed, CTR_TBF_UL_FREED)
Holger Hans Peter Freytheraa35ba72013-11-13 15:02:50 +0100313CREATE_COUNT_INLINE(tbf_reused, CTR_TBF_REUSED)
Holger Hans Peter Freythere9429b52013-11-13 19:36:57 +0100314CREATE_COUNT_INLINE(rlc_sent, CTR_RLC_SENT)
315CREATE_COUNT_INLINE(rlc_resent, CTR_RLC_RESENT)
Holger Hans Peter Freytheref93bdb2013-11-24 00:01:50 +0100316CREATE_COUNT_INLINE(rlc_restarted, CTR_RLC_RESTARTED)
Holger Hans Peter Freytherc70aae42013-11-19 17:09:37 +0100317CREATE_COUNT_INLINE(rlc_stalled, CTR_RLC_STALLED)
Holger Hans Peter Freyther092478f2013-11-23 01:01:19 +0100318CREATE_COUNT_INLINE(rlc_nacked, CTR_RLC_NACKED)
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100319CREATE_COUNT_INLINE(decode_error, CTR_DECODE_ERRORS)
320CREATE_COUNT_INLINE(sba_allocated, CTR_SBA_ALLOCATED)
321CREATE_COUNT_INLINE(sba_freed, CTR_SBA_FREED)
322CREATE_COUNT_INLINE(sba_timedout, CTR_SBA_TIMEDOUT)
Holger Hans Peter Freytherb3d5ee22013-11-13 16:43:26 +0100323CREATE_COUNT_INLINE(llc_timedout_frame, CTR_LLC_FRAME_TIMEDOUT);
324CREATE_COUNT_INLINE(llc_dropped_frame, CTR_LLC_FRAME_DROPPED);
325CREATE_COUNT_INLINE(llc_frame_sched, CTR_LLC_FRAME_SCHED);
Holger Hans Peter Freytherc1ae2262013-10-27 10:50:35 +0100326CREATE_COUNT_INLINE(rach_frame, CTR_RACH_REQUESTS);
Holger Hans Peter Freyther15877642013-10-27 09:50:15 +0100327
Holger Hans Peter Freyther93e048f2013-10-27 10:00:47 +0100328#undef CREATE_COUNT_INLINE
Holger Hans Peter Freyther15877642013-10-27 09:50:15 +0100329
Holger Hans Peter Freyther15877642013-10-27 09:50:15 +0100330
Holger Hans Peter Freyther65be4802013-10-26 18:39:36 +0200331inline gprs_rlcmac_bts *gprs_rlcmac_pdch::bts_data() const
332{
333 return trx->bts->bts_data();
334}
Holger Hans Peter Freyther05f8efc2013-10-26 18:53:16 +0200335
336inline uint8_t gprs_rlcmac_pdch::trx_no() const
337{
338 return trx->trx_no;
339}
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200340#endif
341
342#ifdef __cplusplus
343extern "C" {
344#endif
345 struct gprs_rlcmac_bts *bts_main_data();
Holger Hans Peter Freytherf5372982013-10-27 09:02:31 +0100346 struct rate_ctr_group *bts_main_data_stats();
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200347#ifdef __cplusplus
348}
Holger Hans Peter Freyther17b0d832013-10-19 17:37:48 +0200349
350inline bool gprs_rlcmac_pdch::is_enabled() const
351{
352 return m_is_enabled;
353}
Holger Hans Peter Freytherb6acfda2013-10-17 19:41:11 +0200354#endif