blob: d82b009c703f479fef0d2d3e7986672998d22526 [file] [log] [blame]
Harald Welte0e3e88e2011-01-01 15:25:50 +01001/* OpenBSC E1 Input code */
2
3/* (C) 2008-2010 by Harald Welte <laforge@gnumonks.org>
4 * All Rights Reserved
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU Affero General Public License as published by
8 * the Free Software Foundation; either version 3 of the License, or
9 * (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 Affero General Public License for more details.
15 *
16 * You should have received a copy of the GNU Affero General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 *
19 */
20
Harald Welte59b04682009-06-10 05:40:52 +080021#include <string.h>
22#include <errno.h>
23
24#include <netinet/in.h>
25
26#include <openbsc/gsm_data.h>
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +020027#include <osmocom/abis/e1_input.h>
28#include <osmocom/abis/trau_frame.h>
Harald Welte59b04682009-06-10 05:40:52 +080029#include <openbsc/trau_mux.h>
30#include <openbsc/misdn.h>
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +020031#include <osmocom/abis/ipaccess.h>
Pablo Neira Ayusodd5fff42011-03-22 16:47:59 +010032#include <osmocom/core/talloc.h>
Harald Weltecf2ec4a2009-12-17 23:10:46 +010033#include <openbsc/debug.h>
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +020034#include <openbsc/abis_rsl.h>
Harald Welte59b04682009-06-10 05:40:52 +080035
36#define SAPI_L2ML 0
37#define SAPI_OML 62
38#define SAPI_RSL 0 /* 63 ? */
39
Harald Welte62868882009-08-08 16:12:58 +020040/* The e1_reconfig_*() functions below tale the configuration present in the
41 * bts/trx/ts data structures and ensure the E1 configuration reflects the
42 * timeslot/subslot/TEI configuration */
Harald Welte59b04682009-06-10 05:40:52 +080043
Harald Welte62868882009-08-08 16:12:58 +020044int e1_reconfig_ts(struct gsm_bts_trx_ts *ts)
45{
46 struct gsm_e1_subslot *e1_link = &ts->e1_link;
47 struct e1inp_line *line;
48 struct e1inp_ts *e1_ts;
49
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +020050 DEBUGP(DLMI, "e1_reconfig_ts(%u,%u,%u)\n", ts->trx->bts->nr, ts->trx->nr, ts->nr);
Harald Welte62868882009-08-08 16:12:58 +020051
Harald Weltea8497822011-02-05 15:41:24 +010052 if (!e1_link->e1_ts) {
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +020053 LOGP(DLINP, LOGL_ERROR, "TS (%u/%u/%u) without E1 timeslot?\n",
Harald Weltea8497822011-02-05 15:41:24 +010054 ts->nr, ts->trx->nr, ts->trx->bts->nr);
Harald Welte62868882009-08-08 16:12:58 +020055 return 0;
Harald Weltea8497822011-02-05 15:41:24 +010056 }
Harald Welte62868882009-08-08 16:12:58 +020057
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +020058 line = e1inp_line_find(e1_link->e1_nr);
Harald Weltea8497822011-02-05 15:41:24 +010059 if (!line) {
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +020060 LOGP(DLINP, LOGL_ERROR, "TS (%u/%u/%u) referring to "
Harald Weltea8497822011-02-05 15:41:24 +010061 "non-existing E1 line %u\n", ts->nr, ts->trx->nr,
62 ts->trx->bts->nr, e1_link->e1_nr);
Harald Welte62868882009-08-08 16:12:58 +020063 return -ENOMEM;
Harald Weltea8497822011-02-05 15:41:24 +010064 }
Harald Welte62868882009-08-08 16:12:58 +020065
66 switch (ts->pchan) {
67 case GSM_PCHAN_TCH_F:
68 case GSM_PCHAN_TCH_H:
69 e1_ts = &line->ts[e1_link->e1_ts-1];
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +020070 e1inp_ts_config_trau(e1_ts, line, subch_cb);
Harald Welte62868882009-08-08 16:12:58 +020071 subch_demux_activate(&e1_ts->trau.demux, e1_link->e1_ts_ss);
72 break;
73 default:
74 break;
75 }
76
77 return 0;
78}
79
80int e1_reconfig_trx(struct gsm_bts_trx *trx)
81{
82 struct gsm_e1_subslot *e1_link = &trx->rsl_e1_link;
83 struct e1inp_ts *sign_ts;
84 struct e1inp_line *line;
85 struct e1inp_sign_link *rsl_link;
86 int i;
87
Harald Weltea8497822011-02-05 15:41:24 +010088 if (!e1_link->e1_ts) {
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +020089 LOGP(DLINP, LOGL_ERROR, "TRX (%u/%u) RSL link without "
Harald Weltea8497822011-02-05 15:41:24 +010090 "timeslot?\n", trx->bts->nr, trx->nr);
Harald Welte62868882009-08-08 16:12:58 +020091 return -EINVAL;
Harald Weltea8497822011-02-05 15:41:24 +010092 }
Harald Welte62868882009-08-08 16:12:58 +020093
94 /* RSL Link */
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +020095 line = e1inp_line_find(e1_link->e1_nr);
Harald Weltea8497822011-02-05 15:41:24 +010096 if (!line) {
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +020097 LOGP(DLINP, LOGL_ERROR, "TRX (%u/%u) RSL link referring "
Harald Weltea8497822011-02-05 15:41:24 +010098 "to non-existing E1 line %u\n", trx->bts->nr,
99 trx->nr, e1_link->e1_nr);
Harald Welte62868882009-08-08 16:12:58 +0200100 return -ENOMEM;
Harald Weltea8497822011-02-05 15:41:24 +0100101 }
Harald Welte62868882009-08-08 16:12:58 +0200102 sign_ts = &line->ts[e1_link->e1_ts-1];
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +0200103 e1inp_ts_config_sign(sign_ts, line);
Harald Weltecaeb9bc2011-02-13 19:36:18 +0100104 /* Ericsson RBS have a per-TRX OML link in parallel to RSL */
Harald Welte6e4fd612011-02-13 15:41:09 +0100105 if (trx->bts->type == GSM_BTS_TYPE_RBS2000) {
Harald Weltecaeb9bc2011-02-13 19:36:18 +0100106 struct e1inp_sign_link *oml_link;
107 oml_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_OML, trx,
108 trx->rsl_tei, SAPI_OML);
109 if (!oml_link) {
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +0200110 LOGP(DLINP, LOGL_ERROR, "TRX (%u/%u) OML link creation "
Harald Weltecaeb9bc2011-02-13 19:36:18 +0100111 "failed\n", trx->bts->nr, trx->nr);
112 return -ENOMEM;
113 }
114 if (trx->oml_link)
115 e1inp_sign_link_destroy(trx->oml_link);
116 trx->oml_link = oml_link;
Harald Welte6e4fd612011-02-13 15:41:09 +0100117 }
Harald Welte62868882009-08-08 16:12:58 +0200118 rsl_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_RSL,
119 trx, trx->rsl_tei, SAPI_RSL);
Harald Weltea8497822011-02-05 15:41:24 +0100120 if (!rsl_link) {
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +0200121 LOGP(DLINP, LOGL_ERROR, "TRX (%u/%u) RSL link creation "
Harald Weltea8497822011-02-05 15:41:24 +0100122 "failed\n", trx->bts->nr, trx->nr);
Harald Welte62868882009-08-08 16:12:58 +0200123 return -ENOMEM;
Harald Weltea8497822011-02-05 15:41:24 +0100124 }
Harald Welte62868882009-08-08 16:12:58 +0200125 if (trx->rsl_link)
126 e1inp_sign_link_destroy(trx->rsl_link);
127 trx->rsl_link = rsl_link;
128
129 for (i = 0; i < TRX_NR_TS; i++)
130 e1_reconfig_ts(&trx->ts[i]);
131
132 return 0;
133}
134
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +0200135/* this is the generic callback for all ISDN-based BTS. */
136static int bts_isdn_sign_link(struct msgb *msg)
137{
138 int ret = -EINVAL;
139 struct e1inp_sign_link *link = msg->dst;
140 struct gsm_bts *bts;
141
142 log_set_context(BSC_CTX_BTS, link->trx->bts);
143 switch (link->type) {
144 case E1INP_SIGN_OML:
145 bts = link->trx->bts;
146 ret = bts->model->oml_rcvmsg(msg);
147 break;
148 case E1INP_SIGN_RSL:
149 ret = abis_rsl_rcvmsg(msg);
150 break;
151 default:
152 LOGP(DLMI, LOGL_ERROR, "unknown link type %u\n", link->type);
153 break;
154 }
155 return ret;
156}
157
158struct e1inp_line_ops bts_isdn_e1inp_line_ops = {
159 .sign_link = bts_isdn_sign_link,
160};
161
Harald Welte62868882009-08-08 16:12:58 +0200162int e1_reconfig_bts(struct gsm_bts *bts)
163{
164 struct gsm_e1_subslot *e1_link = &bts->oml_e1_link;
165 struct e1inp_ts *sign_ts;
166 struct e1inp_line *line;
167 struct e1inp_sign_link *oml_link;
168 struct gsm_bts_trx *trx;
Harald Welte62868882009-08-08 16:12:58 +0200169
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +0200170 DEBUGP(DLMI, "e1_reconfig_bts(%u)\n", bts->nr);
Harald Welte62868882009-08-08 16:12:58 +0200171
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +0200172 line = e1inp_line_find(e1_link->e1_nr);
173 if (!line) {
174 LOGP(DLINP, LOGL_ERROR, "BTS %u OML link referring to "
175 "non-existing E1 line %u\n", bts->nr, e1_link->e1_nr);
176 return -ENOMEM;
177 }
178
179 if (!bts->model->e1line_bind_ops) {
180 LOGP(DLINP, LOGL_ERROR, "no callback to bind E1 line operations\n");
181 return -EINVAL;
182 }
183 if (!line->ops)
184 bts->model->e1line_bind_ops(line);
185
186 /* skip signal link initialization, this is done later for these BTS. */
187 if (bts->type == GSM_BTS_TYPE_NANOBTS ||
Holger Hans Peter Freyther1b0af512013-07-03 16:06:20 +0200188 bts->type == GSM_BTS_TYPE_OSMO_SYSMO)
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +0200189 return e1inp_line_update(line);
190
191 /* OML link */
Harald Weltea8497822011-02-05 15:41:24 +0100192 if (!e1_link->e1_ts) {
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +0200193 LOGP(DLINP, LOGL_ERROR, "BTS %u OML link without timeslot?\n",
Harald Weltea8497822011-02-05 15:41:24 +0100194 bts->nr);
Harald Welte62868882009-08-08 16:12:58 +0200195 return -EINVAL;
Harald Weltea8497822011-02-05 15:41:24 +0100196 }
Harald Welte62868882009-08-08 16:12:58 +0200197
Harald Welte62868882009-08-08 16:12:58 +0200198 sign_ts = &line->ts[e1_link->e1_ts-1];
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +0200199 e1inp_ts_config_sign(sign_ts, line);
Harald Welte62868882009-08-08 16:12:58 +0200200 oml_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_OML,
201 bts->c0, bts->oml_tei, SAPI_OML);
Harald Weltea8497822011-02-05 15:41:24 +0100202 if (!oml_link) {
Pablo Neira Ayuso42e41df2011-08-17 22:44:07 +0200203 LOGP(DLINP, LOGL_ERROR, "BTS %u OML link creation failed\n",
Harald Weltea8497822011-02-05 15:41:24 +0100204 bts->nr);
Harald Welte62868882009-08-08 16:12:58 +0200205 return -ENOMEM;
Harald Weltea8497822011-02-05 15:41:24 +0100206 }
Harald Welte62868882009-08-08 16:12:58 +0200207 if (bts->oml_link)
208 e1inp_sign_link_destroy(bts->oml_link);
209 bts->oml_link = oml_link;
210
211 llist_for_each_entry(trx, &bts->trx_list, list)
212 e1_reconfig_trx(trx);
213
214 /* notify E1 input something has changed */
215 return e1inp_line_update(line);
216}
217
218#if 0
Harald Welte59b04682009-06-10 05:40:52 +0800219/* do some compiled-in configuration for our BTS/E1 setup */
220int e1_config(struct gsm_bts *bts, int cardnr, int release_l2)
221{
222 struct e1inp_line *line;
223 struct e1inp_ts *sign_ts;
224 struct e1inp_sign_link *oml_link, *rsl_link;
Harald Welte25b70c52009-07-29 16:42:16 +0200225 struct gsm_bts_trx *trx = bts->c0;
Harald Welted1b1cda2009-08-04 14:13:35 +0200226 int base_ts;
227
228 switch (bts->nr) {
229 case 0:
230 /* First BTS uses E1 TS 01,02,03,04,05 */
Harald Welte2d73d4e2009-08-06 17:38:10 +0200231 base_ts = HARDCODED_BTS0_TS - 1;
Harald Welted1b1cda2009-08-04 14:13:35 +0200232 break;
233 case 1:
Harald Welte2d73d4e2009-08-06 17:38:10 +0200234 /* Second BTS uses E1 TS 06,07,08,09,10 */
235 base_ts = HARDCODED_BTS1_TS - 1;
Harald Welted1b1cda2009-08-04 14:13:35 +0200236 break;
237 case 2:
Harald Welte2d73d4e2009-08-06 17:38:10 +0200238 /* Third BTS uses E1 TS 11,12,13,14,15 */
239 base_ts = HARDCODED_BTS2_TS - 1;
Harald Welted1b1cda2009-08-04 14:13:35 +0200240 default:
241 return -EINVAL;
242 }
Harald Welte59b04682009-06-10 05:40:52 +0800243
Harald Welte857e00d2009-06-26 20:25:23 +0200244 line = talloc_zero(tall_bsc_ctx, struct e1inp_line);
Harald Welte59b04682009-06-10 05:40:52 +0800245 if (!line)
246 return -ENOMEM;
Harald Welte59b04682009-06-10 05:40:52 +0800247
248 /* create E1 timeslots for signalling and TRAU frames */
Harald Welted1b1cda2009-08-04 14:13:35 +0200249 e1inp_ts_config(&line->ts[base_ts+1-1], line, E1INP_TS_TYPE_SIGN);
250 e1inp_ts_config(&line->ts[base_ts+2-1], line, E1INP_TS_TYPE_TRAU);
251 e1inp_ts_config(&line->ts[base_ts+3-1], line, E1INP_TS_TYPE_TRAU);
Harald Welte59b04682009-06-10 05:40:52 +0800252
253 /* create signalling links for TS1 */
Harald Welted1b1cda2009-08-04 14:13:35 +0200254 sign_ts = &line->ts[base_ts+1-1];
Harald Welte59b04682009-06-10 05:40:52 +0800255 oml_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_OML,
Harald Welte25b70c52009-07-29 16:42:16 +0200256 trx, TEI_OML, SAPI_OML);
Harald Welte59b04682009-06-10 05:40:52 +0800257 rsl_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_RSL,
Harald Welte25b70c52009-07-29 16:42:16 +0200258 trx, TEI_RSL, SAPI_RSL);
Harald Welte59b04682009-06-10 05:40:52 +0800259
260 /* create back-links from bts/trx */
261 bts->oml_link = oml_link;
Harald Welte25b70c52009-07-29 16:42:16 +0200262 trx->rsl_link = rsl_link;
Harald Welte59b04682009-06-10 05:40:52 +0800263
264 /* enable subchannel demuxer on TS2 */
Harald Welted1b1cda2009-08-04 14:13:35 +0200265 subch_demux_activate(&line->ts[base_ts+2-1].trau.demux, 1);
266 subch_demux_activate(&line->ts[base_ts+2-1].trau.demux, 2);
267 subch_demux_activate(&line->ts[base_ts+2-1].trau.demux, 3);
Harald Welte59b04682009-06-10 05:40:52 +0800268
269 /* enable subchannel demuxer on TS3 */
Harald Welted1b1cda2009-08-04 14:13:35 +0200270 subch_demux_activate(&line->ts[base_ts+3-1].trau.demux, 0);
271 subch_demux_activate(&line->ts[base_ts+3-1].trau.demux, 1);
272 subch_demux_activate(&line->ts[base_ts+3-1].trau.demux, 2);
273 subch_demux_activate(&line->ts[base_ts+3-1].trau.demux, 3);
Harald Welte59b04682009-06-10 05:40:52 +0800274
Harald Welte25b70c52009-07-29 16:42:16 +0200275 trx = gsm_bts_trx_num(bts, 1);
276 if (trx) {
277 /* create E1 timeslots for TRAU frames of TRX1 */
Harald Welted1b1cda2009-08-04 14:13:35 +0200278 e1inp_ts_config(&line->ts[base_ts+4-1], line, E1INP_TS_TYPE_TRAU);
279 e1inp_ts_config(&line->ts[base_ts+5-1], line, E1INP_TS_TYPE_TRAU);
Harald Welte59b04682009-06-10 05:40:52 +0800280
Harald Welte25b70c52009-07-29 16:42:16 +0200281 /* create RSL signalling link for TRX1 */
Harald Welted1b1cda2009-08-04 14:13:35 +0200282 sign_ts = &line->ts[base_ts+1-1];
Harald Welte25b70c52009-07-29 16:42:16 +0200283 rsl_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_RSL,
284 trx, TEI_RSL+1, SAPI_RSL);
285 /* create back-links from trx */
286 trx->rsl_link = rsl_link;
287
288 /* enable subchannel demuxer on TS2 */
Harald Welted1b1cda2009-08-04 14:13:35 +0200289 subch_demux_activate(&line->ts[base_ts+4-1].trau.demux, 0);
290 subch_demux_activate(&line->ts[base_ts+4-1].trau.demux, 1);
291 subch_demux_activate(&line->ts[base_ts+4-1].trau.demux, 2);
292 subch_demux_activate(&line->ts[base_ts+4-1].trau.demux, 3);
Harald Welte25b70c52009-07-29 16:42:16 +0200293
294 /* enable subchannel demuxer on TS3 */
Harald Welted1b1cda2009-08-04 14:13:35 +0200295 subch_demux_activate(&line->ts[base_ts+5-1].trau.demux, 0);
296 subch_demux_activate(&line->ts[base_ts+5-1].trau.demux, 1);
297 subch_demux_activate(&line->ts[base_ts+5-1].trau.demux, 2);
298 subch_demux_activate(&line->ts[base_ts+5-1].trau.demux, 3);
Harald Welte25b70c52009-07-29 16:42:16 +0200299 }
Harald Welte59b04682009-06-10 05:40:52 +0800300
301 return mi_setup(cardnr, line, release_l2);
302}
Harald Welte62868882009-08-08 16:12:58 +0200303#endif