blob: e7398ed9c882b66e7efe149ee7e2cff2e0b16009 [file] [log] [blame]
Harald Welte9af6ddf2011-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 Welte1fa60c82009-02-09 18:13:26 +000021#include <string.h>
22#include <errno.h>
Maxd1f70ed2017-09-21 16:15:32 +020023#include <time.h>
Harald Welte25de9912009-04-30 15:53:07 +000024#include <netinet/in.h>
25
Neels Hofmeyrc0164792017-09-04 15:15:32 +020026#include <osmocom/bsc/gsm_data.h>
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +020027#include <osmocom/abis/e1_input.h>
Neels Hofmeyrc0164792017-09-04 15:15:32 +020028#include <osmocom/bsc/misdn.h>
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +020029#include <osmocom/abis/ipaccess.h>
Pablo Neira Ayuso136f4532011-03-22 16:47:59 +010030#include <osmocom/core/talloc.h>
Neels Hofmeyrc0164792017-09-04 15:15:32 +020031#include <osmocom/bsc/debug.h>
32#include <osmocom/bsc/abis_rsl.h>
Harald Welte1fa60c82009-02-09 18:13:26 +000033
34#define SAPI_L2ML 0
35#define SAPI_OML 62
36#define SAPI_RSL 0 /* 63 ? */
37
Neels Hofmeyr57e8a1f2016-09-25 15:18:39 +020038/* The e1_reconfig_*() functions below take the configuration present in the
Harald Welte42581822009-08-08 16:12:58 +020039 * bts/trx/ts data structures and ensure the E1 configuration reflects the
40 * timeslot/subslot/TEI configuration */
Harald Welte1fa60c82009-02-09 18:13:26 +000041
Harald Welte42581822009-08-08 16:12:58 +020042int e1_reconfig_ts(struct gsm_bts_trx_ts *ts)
43{
44 struct gsm_e1_subslot *e1_link = &ts->e1_link;
45 struct e1inp_line *line;
Harald Welte42581822009-08-08 16:12:58 +020046
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +020047 DEBUGP(DLMI, "e1_reconfig_ts(%u,%u,%u)\n", ts->trx->bts->nr, ts->trx->nr, ts->nr);
Harald Welte42581822009-08-08 16:12:58 +020048
Harald Welte889f16e2011-02-05 15:41:24 +010049 if (!e1_link->e1_ts) {
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +020050 LOGP(DLINP, LOGL_ERROR, "TS (%u/%u/%u) without E1 timeslot?\n",
Harald Welte889f16e2011-02-05 15:41:24 +010051 ts->nr, ts->trx->nr, ts->trx->bts->nr);
Harald Welte42581822009-08-08 16:12:58 +020052 return 0;
Harald Welte889f16e2011-02-05 15:41:24 +010053 }
Harald Welte42581822009-08-08 16:12:58 +020054
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +020055 line = e1inp_line_find(e1_link->e1_nr);
Harald Welte889f16e2011-02-05 15:41:24 +010056 if (!line) {
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +020057 LOGP(DLINP, LOGL_ERROR, "TS (%u/%u/%u) referring to "
Harald Welte889f16e2011-02-05 15:41:24 +010058 "non-existing E1 line %u\n", ts->nr, ts->trx->nr,
59 ts->trx->bts->nr, e1_link->e1_nr);
Harald Welte42581822009-08-08 16:12:58 +020060 return -ENOMEM;
Harald Welte889f16e2011-02-05 15:41:24 +010061 }
Harald Welte42581822009-08-08 16:12:58 +020062
Harald Welte42581822009-08-08 16:12:58 +020063 return 0;
64}
65
66int e1_reconfig_trx(struct gsm_bts_trx *trx)
67{
68 struct gsm_e1_subslot *e1_link = &trx->rsl_e1_link;
69 struct e1inp_ts *sign_ts;
70 struct e1inp_line *line;
71 struct e1inp_sign_link *rsl_link;
72 int i;
73
Harald Welte889f16e2011-02-05 15:41:24 +010074 if (!e1_link->e1_ts) {
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +020075 LOGP(DLINP, LOGL_ERROR, "TRX (%u/%u) RSL link without "
Harald Welte889f16e2011-02-05 15:41:24 +010076 "timeslot?\n", trx->bts->nr, trx->nr);
Harald Welte42581822009-08-08 16:12:58 +020077 return -EINVAL;
Harald Welte889f16e2011-02-05 15:41:24 +010078 }
Harald Welte42581822009-08-08 16:12:58 +020079
80 /* RSL Link */
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +020081 line = e1inp_line_find(e1_link->e1_nr);
Harald Welte889f16e2011-02-05 15:41:24 +010082 if (!line) {
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +020083 LOGP(DLINP, LOGL_ERROR, "TRX (%u/%u) RSL link referring "
Harald Welte889f16e2011-02-05 15:41:24 +010084 "to non-existing E1 line %u\n", trx->bts->nr,
85 trx->nr, e1_link->e1_nr);
Harald Welte42581822009-08-08 16:12:58 +020086 return -ENOMEM;
Harald Welte889f16e2011-02-05 15:41:24 +010087 }
Harald Welte42581822009-08-08 16:12:58 +020088 sign_ts = &line->ts[e1_link->e1_ts-1];
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +020089 e1inp_ts_config_sign(sign_ts, line);
Harald Welte15ccc772011-02-13 19:36:18 +010090 /* Ericsson RBS have a per-TRX OML link in parallel to RSL */
Harald Weltec834b702011-02-13 15:41:09 +010091 if (trx->bts->type == GSM_BTS_TYPE_RBS2000) {
Harald Welte15ccc772011-02-13 19:36:18 +010092 struct e1inp_sign_link *oml_link;
93 oml_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_OML, trx,
94 trx->rsl_tei, SAPI_OML);
95 if (!oml_link) {
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +020096 LOGP(DLINP, LOGL_ERROR, "TRX (%u/%u) OML link creation "
Harald Welte15ccc772011-02-13 19:36:18 +010097 "failed\n", trx->bts->nr, trx->nr);
98 return -ENOMEM;
99 }
100 if (trx->oml_link)
101 e1inp_sign_link_destroy(trx->oml_link);
102 trx->oml_link = oml_link;
Harald Weltec834b702011-02-13 15:41:09 +0100103 }
Harald Welte42581822009-08-08 16:12:58 +0200104 rsl_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_RSL,
105 trx, trx->rsl_tei, SAPI_RSL);
Harald Welte889f16e2011-02-05 15:41:24 +0100106 if (!rsl_link) {
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +0200107 LOGP(DLINP, LOGL_ERROR, "TRX (%u/%u) RSL link creation "
Harald Welte889f16e2011-02-05 15:41:24 +0100108 "failed\n", trx->bts->nr, trx->nr);
Harald Welte42581822009-08-08 16:12:58 +0200109 return -ENOMEM;
Harald Welte889f16e2011-02-05 15:41:24 +0100110 }
Harald Welte42581822009-08-08 16:12:58 +0200111 if (trx->rsl_link)
112 e1inp_sign_link_destroy(trx->rsl_link);
113 trx->rsl_link = rsl_link;
114
115 for (i = 0; i < TRX_NR_TS; i++)
116 e1_reconfig_ts(&trx->ts[i]);
117
118 return 0;
119}
120
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +0200121/* this is the generic callback for all ISDN-based BTS. */
122static int bts_isdn_sign_link(struct msgb *msg)
123{
124 int ret = -EINVAL;
125 struct e1inp_sign_link *link = msg->dst;
126 struct gsm_bts *bts;
127
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +0200128 switch (link->type) {
129 case E1INP_SIGN_OML:
130 bts = link->trx->bts;
131 ret = bts->model->oml_rcvmsg(msg);
132 break;
133 case E1INP_SIGN_RSL:
Stefan Sperling73acbca2018-02-12 14:28:52 +0100134 if (link->trx->mo.nm_state.administrative == NM_STATE_LOCKED) {
135 LOGP(DLMI, LOGL_ERROR, "(bts=%d/trx=%d) discarding RSL message received "
136 "in locked administrative state\n", link->trx->bts->nr, link->trx->nr);
137 msgb_free(msg);
138 break;
139 }
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +0200140 ret = abis_rsl_rcvmsg(msg);
141 break;
142 default:
143 LOGP(DLMI, LOGL_ERROR, "unknown link type %u\n", link->type);
Stefan Sperling73acbca2018-02-12 14:28:52 +0100144 msgb_free(msg);
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +0200145 break;
146 }
147 return ret;
148}
149
150struct e1inp_line_ops bts_isdn_e1inp_line_ops = {
151 .sign_link = bts_isdn_sign_link,
152};
153
Harald Welte42581822009-08-08 16:12:58 +0200154int e1_reconfig_bts(struct gsm_bts *bts)
155{
156 struct gsm_e1_subslot *e1_link = &bts->oml_e1_link;
157 struct e1inp_ts *sign_ts;
158 struct e1inp_line *line;
159 struct e1inp_sign_link *oml_link;
160 struct gsm_bts_trx *trx;
Maxd1f70ed2017-09-21 16:15:32 +0200161 struct timespec tp;
162 int rc;
Harald Welte42581822009-08-08 16:12:58 +0200163
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +0200164 DEBUGP(DLMI, "e1_reconfig_bts(%u)\n", bts->nr);
Harald Welte42581822009-08-08 16:12:58 +0200165
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +0200166 line = e1inp_line_find(e1_link->e1_nr);
167 if (!line) {
168 LOGP(DLINP, LOGL_ERROR, "BTS %u OML link referring to "
169 "non-existing E1 line %u\n", bts->nr, e1_link->e1_nr);
170 return -ENOMEM;
171 }
172
173 if (!bts->model->e1line_bind_ops) {
174 LOGP(DLINP, LOGL_ERROR, "no callback to bind E1 line operations\n");
175 return -EINVAL;
176 }
177 if (!line->ops)
178 bts->model->e1line_bind_ops(line);
179
180 /* skip signal link initialization, this is done later for these BTS. */
181 if (bts->type == GSM_BTS_TYPE_NANOBTS ||
Maxf9685c12017-03-23 12:01:07 +0100182 bts->type == GSM_BTS_TYPE_OSMOBTS)
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +0200183 return e1inp_line_update(line);
184
185 /* OML link */
Harald Welte889f16e2011-02-05 15:41:24 +0100186 if (!e1_link->e1_ts) {
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +0200187 LOGP(DLINP, LOGL_ERROR, "BTS %u OML link without timeslot?\n",
Harald Welte889f16e2011-02-05 15:41:24 +0100188 bts->nr);
Harald Welte42581822009-08-08 16:12:58 +0200189 return -EINVAL;
Harald Welte889f16e2011-02-05 15:41:24 +0100190 }
Harald Welte42581822009-08-08 16:12:58 +0200191
Harald Welte42581822009-08-08 16:12:58 +0200192 sign_ts = &line->ts[e1_link->e1_ts-1];
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +0200193 e1inp_ts_config_sign(sign_ts, line);
Harald Welte42581822009-08-08 16:12:58 +0200194 oml_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_OML,
195 bts->c0, bts->oml_tei, SAPI_OML);
Harald Welte889f16e2011-02-05 15:41:24 +0100196 if (!oml_link) {
Pablo Neira Ayusoed5cacb2011-08-17 22:44:07 +0200197 LOGP(DLINP, LOGL_ERROR, "BTS %u OML link creation failed\n",
Harald Welte889f16e2011-02-05 15:41:24 +0100198 bts->nr);
Harald Welte42581822009-08-08 16:12:58 +0200199 return -ENOMEM;
Harald Welte889f16e2011-02-05 15:41:24 +0100200 }
Harald Welte42581822009-08-08 16:12:58 +0200201 if (bts->oml_link)
202 e1inp_sign_link_destroy(bts->oml_link);
203 bts->oml_link = oml_link;
Maxd1f70ed2017-09-21 16:15:32 +0200204 rc = clock_gettime(CLOCK_MONOTONIC, &tp);
205 bts->uptime = (rc < 0) ? 0 : tp.tv_sec; /* we don't need sub-second precision for uptime */
Harald Welte42581822009-08-08 16:12:58 +0200206
207 llist_for_each_entry(trx, &bts->trx_list, list)
208 e1_reconfig_trx(trx);
209
210 /* notify E1 input something has changed */
211 return e1inp_line_update(line);
212}
213
214#if 0
Harald Welte1fa60c82009-02-09 18:13:26 +0000215/* do some compiled-in configuration for our BTS/E1 setup */
Holger Freytherb5c00f52009-04-22 22:08:07 +0000216int e1_config(struct gsm_bts *bts, int cardnr, int release_l2)
Harald Welte1fa60c82009-02-09 18:13:26 +0000217{
218 struct e1inp_line *line;
219 struct e1inp_ts *sign_ts;
220 struct e1inp_sign_link *oml_link, *rsl_link;
Harald Welte67b4c302009-07-29 16:42:16 +0200221 struct gsm_bts_trx *trx = bts->c0;
Harald Welte4e55d682009-08-04 14:13:35 +0200222 int base_ts;
223
224 switch (bts->nr) {
225 case 0:
226 /* First BTS uses E1 TS 01,02,03,04,05 */
Harald Welte63589be2009-08-06 17:38:10 +0200227 base_ts = HARDCODED_BTS0_TS - 1;
Harald Welte4e55d682009-08-04 14:13:35 +0200228 break;
229 case 1:
Harald Welte63589be2009-08-06 17:38:10 +0200230 /* Second BTS uses E1 TS 06,07,08,09,10 */
231 base_ts = HARDCODED_BTS1_TS - 1;
Harald Welte4e55d682009-08-04 14:13:35 +0200232 break;
233 case 2:
Harald Welte63589be2009-08-06 17:38:10 +0200234 /* Third BTS uses E1 TS 11,12,13,14,15 */
235 base_ts = HARDCODED_BTS2_TS - 1;
Harald Welte4e55d682009-08-04 14:13:35 +0200236 default:
237 return -EINVAL;
238 }
Harald Welte1fa60c82009-02-09 18:13:26 +0000239
Harald Welte470ec292009-06-26 20:25:23 +0200240 line = talloc_zero(tall_bsc_ctx, struct e1inp_line);
Harald Welte1fa60c82009-02-09 18:13:26 +0000241 if (!line)
242 return -ENOMEM;
Harald Welte1fa60c82009-02-09 18:13:26 +0000243
244 /* create E1 timeslots for signalling and TRAU frames */
Harald Welte4e55d682009-08-04 14:13:35 +0200245 e1inp_ts_config(&line->ts[base_ts+1-1], line, E1INP_TS_TYPE_SIGN);
246 e1inp_ts_config(&line->ts[base_ts+2-1], line, E1INP_TS_TYPE_TRAU);
247 e1inp_ts_config(&line->ts[base_ts+3-1], line, E1INP_TS_TYPE_TRAU);
Harald Welte1fa60c82009-02-09 18:13:26 +0000248
249 /* create signalling links for TS1 */
Harald Welte4e55d682009-08-04 14:13:35 +0200250 sign_ts = &line->ts[base_ts+1-1];
Harald Welte1fa60c82009-02-09 18:13:26 +0000251 oml_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_OML,
Harald Welte67b4c302009-07-29 16:42:16 +0200252 trx, TEI_OML, SAPI_OML);
Harald Welte1fa60c82009-02-09 18:13:26 +0000253 rsl_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_RSL,
Harald Welte67b4c302009-07-29 16:42:16 +0200254 trx, TEI_RSL, SAPI_RSL);
Harald Welte1fa60c82009-02-09 18:13:26 +0000255
256 /* create back-links from bts/trx */
257 bts->oml_link = oml_link;
Harald Welte67b4c302009-07-29 16:42:16 +0200258 trx->rsl_link = rsl_link;
Harald Welte1fa60c82009-02-09 18:13:26 +0000259
Harald Welte11fa29c2009-02-19 17:24:39 +0000260 /* enable subchannel demuxer on TS2 */
Harald Welte4e55d682009-08-04 14:13:35 +0200261 subch_demux_activate(&line->ts[base_ts+2-1].trau.demux, 1);
262 subch_demux_activate(&line->ts[base_ts+2-1].trau.demux, 2);
263 subch_demux_activate(&line->ts[base_ts+2-1].trau.demux, 3);
Harald Welte93d3f372009-02-18 03:27:14 +0000264
Harald Welte11fa29c2009-02-19 17:24:39 +0000265 /* enable subchannel demuxer on TS3 */
Harald Welte4e55d682009-08-04 14:13:35 +0200266 subch_demux_activate(&line->ts[base_ts+3-1].trau.demux, 0);
267 subch_demux_activate(&line->ts[base_ts+3-1].trau.demux, 1);
268 subch_demux_activate(&line->ts[base_ts+3-1].trau.demux, 2);
269 subch_demux_activate(&line->ts[base_ts+3-1].trau.demux, 3);
Harald Welte11fa29c2009-02-19 17:24:39 +0000270
Harald Welte67b4c302009-07-29 16:42:16 +0200271 trx = gsm_bts_trx_num(bts, 1);
272 if (trx) {
273 /* create E1 timeslots for TRAU frames of TRX1 */
Harald Welte4e55d682009-08-04 14:13:35 +0200274 e1inp_ts_config(&line->ts[base_ts+4-1], line, E1INP_TS_TYPE_TRAU);
275 e1inp_ts_config(&line->ts[base_ts+5-1], line, E1INP_TS_TYPE_TRAU);
Harald Welte1fa60c82009-02-09 18:13:26 +0000276
Harald Welte67b4c302009-07-29 16:42:16 +0200277 /* create RSL signalling link for TRX1 */
Harald Welte4e55d682009-08-04 14:13:35 +0200278 sign_ts = &line->ts[base_ts+1-1];
Harald Welte67b4c302009-07-29 16:42:16 +0200279 rsl_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_RSL,
280 trx, TEI_RSL+1, SAPI_RSL);
281 /* create back-links from trx */
282 trx->rsl_link = rsl_link;
283
284 /* enable subchannel demuxer on TS2 */
Harald Welte4e55d682009-08-04 14:13:35 +0200285 subch_demux_activate(&line->ts[base_ts+4-1].trau.demux, 0);
286 subch_demux_activate(&line->ts[base_ts+4-1].trau.demux, 1);
287 subch_demux_activate(&line->ts[base_ts+4-1].trau.demux, 2);
288 subch_demux_activate(&line->ts[base_ts+4-1].trau.demux, 3);
Harald Welte67b4c302009-07-29 16:42:16 +0200289
290 /* enable subchannel demuxer on TS3 */
Harald Welte4e55d682009-08-04 14:13:35 +0200291 subch_demux_activate(&line->ts[base_ts+5-1].trau.demux, 0);
292 subch_demux_activate(&line->ts[base_ts+5-1].trau.demux, 1);
293 subch_demux_activate(&line->ts[base_ts+5-1].trau.demux, 2);
294 subch_demux_activate(&line->ts[base_ts+5-1].trau.demux, 3);
Harald Welte67b4c302009-07-29 16:42:16 +0200295 }
Harald Welte1fa60c82009-02-09 18:13:26 +0000296
Holger Freytherb5c00f52009-04-22 22:08:07 +0000297 return mi_setup(cardnr, line, release_l2);
Harald Welte1fa60c82009-02-09 18:13:26 +0000298}
Harald Welte42581822009-08-08 16:12:58 +0200299#endif