blob: bd6ebba9d685b5ae144a1255db75e2f21d2c3cd4 [file] [log] [blame]
Jonathan Santos03fd8d02011-05-25 13:54:02 -04001/* 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
21#include <string.h>
22#include <errno.h>
23
24#include <netinet/in.h>
25
26#include <openbsc/gsm_data.h>
27#include <openbsc/e1_input.h>
28#include <openbsc/trau_frame.h>
29#include <openbsc/trau_mux.h>
30#include <openbsc/misdn.h>
31#include <openbsc/ipaccess.h>
Jonathan Santos5a45b152011-08-17 15:33:57 -040032#include <osmocom/core/talloc.h>
Jonathan Santos03fd8d02011-05-25 13:54:02 -040033#include <openbsc/debug.h>
34
35#define SAPI_L2ML 0
36#define SAPI_OML 62
37#define SAPI_RSL 0 /* 63 ? */
38
39/* The e1_reconfig_*() functions below tale the configuration present in the
40 * bts/trx/ts data structures and ensure the E1 configuration reflects the
41 * timeslot/subslot/TEI configuration */
42
43int e1_reconfig_ts(struct gsm_bts_trx_ts *ts)
44{
45 struct gsm_e1_subslot *e1_link = &ts->e1_link;
46 struct e1inp_line *line;
47 struct e1inp_ts *e1_ts;
48
49 DEBUGP(DMI, "e1_reconfig_ts(%u,%u,%u)\n", ts->trx->bts->nr, ts->trx->nr, ts->nr);
50
51 if (!e1_link->e1_ts) {
52 LOGP(DINP, LOGL_ERROR, "TS (%u/%u/%u) without E1 timeslot?\n",
53 ts->nr, ts->trx->nr, ts->trx->bts->nr);
54 return 0;
55 }
56
57 line = e1inp_line_get(e1_link->e1_nr);
58 if (!line) {
59 LOGP(DINP, LOGL_ERROR, "TS (%u/%u/%u) referring to "
60 "non-existing E1 line %u\n", ts->nr, ts->trx->nr,
61 ts->trx->bts->nr, e1_link->e1_nr);
62 return -ENOMEM;
63 }
64
65 switch (ts->pchan) {
66 case GSM_PCHAN_TCH_F:
67 case GSM_PCHAN_TCH_H:
68 e1_ts = &line->ts[e1_link->e1_ts-1];
69 e1inp_ts_config(e1_ts, line, E1INP_TS_TYPE_TRAU);
70 subch_demux_activate(&e1_ts->trau.demux, e1_link->e1_ts_ss);
71 break;
72 default:
73 break;
74 }
75
76 return 0;
77}
78
79int e1_reconfig_trx(struct gsm_bts_trx *trx)
80{
81 struct gsm_e1_subslot *e1_link = &trx->rsl_e1_link;
82 struct e1inp_ts *sign_ts;
83 struct e1inp_line *line;
84 struct e1inp_sign_link *rsl_link;
85 int i;
86
87 if (!e1_link->e1_ts) {
88 LOGP(DINP, LOGL_ERROR, "TRX (%u/%u) RSL link without "
89 "timeslot?\n", trx->bts->nr, trx->nr);
90 return -EINVAL;
91 }
92
93 /* RSL Link */
94 line = e1inp_line_get(e1_link->e1_nr);
95 if (!line) {
96 LOGP(DINP, LOGL_ERROR, "TRX (%u/%u) RSL link referring "
97 "to non-existing E1 line %u\n", trx->bts->nr,
98 trx->nr, e1_link->e1_nr);
99 return -ENOMEM;
100 }
101 sign_ts = &line->ts[e1_link->e1_ts-1];
102 e1inp_ts_config(sign_ts, line, E1INP_TS_TYPE_SIGN);
103 /* Ericsson RBS have a per-TRX OML link in parallel to RSL */
104 if (trx->bts->type == GSM_BTS_TYPE_RBS2000) {
105 struct e1inp_sign_link *oml_link;
106 oml_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_OML, trx,
107 trx->rsl_tei, SAPI_OML);
108 if (!oml_link) {
109 LOGP(DINP, LOGL_ERROR, "TRX (%u/%u) OML link creation "
110 "failed\n", trx->bts->nr, trx->nr);
111 return -ENOMEM;
112 }
113 if (trx->oml_link)
114 e1inp_sign_link_destroy(trx->oml_link);
115 trx->oml_link = oml_link;
116 }
117 rsl_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_RSL,
118 trx, trx->rsl_tei, SAPI_RSL);
119 if (!rsl_link) {
120 LOGP(DINP, LOGL_ERROR, "TRX (%u/%u) RSL link creation "
121 "failed\n", trx->bts->nr, trx->nr);
122 return -ENOMEM;
123 }
124 if (trx->rsl_link)
125 e1inp_sign_link_destroy(trx->rsl_link);
126 trx->rsl_link = rsl_link;
127
128 for (i = 0; i < TRX_NR_TS; i++)
129 e1_reconfig_ts(&trx->ts[i]);
130
131 return 0;
132}
133
134int e1_reconfig_bts(struct gsm_bts *bts)
135{
136 struct gsm_e1_subslot *e1_link = &bts->oml_e1_link;
137 struct e1inp_ts *sign_ts;
138 struct e1inp_line *line;
139 struct e1inp_sign_link *oml_link;
140 struct gsm_bts_trx *trx;
141
142 DEBUGP(DMI, "e1_reconfig_bts(%u)\n", bts->nr);
143
144 if (!e1_link->e1_ts) {
145 LOGP(DINP, LOGL_ERROR, "BTS %u OML link without timeslot?\n",
146 bts->nr);
147 return -EINVAL;
148 }
149
150 /* OML link */
151 line = e1inp_line_get(e1_link->e1_nr);
152 if (!line) {
153 LOGP(DINP, LOGL_ERROR, "BTS %u OML link referring to "
154 "non-existing E1 line %u\n", bts->nr, e1_link->e1_nr);
155 return -ENOMEM;
156 }
157 sign_ts = &line->ts[e1_link->e1_ts-1];
158 e1inp_ts_config(sign_ts, line, E1INP_TS_TYPE_SIGN);
159 oml_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_OML,
160 bts->c0, bts->oml_tei, SAPI_OML);
161 if (!oml_link) {
162 LOGP(DINP, LOGL_ERROR, "BTS %u OML link creation failed\n",
163 bts->nr);
164 return -ENOMEM;
165 }
166 if (bts->oml_link)
167 e1inp_sign_link_destroy(bts->oml_link);
168 bts->oml_link = oml_link;
169
170 llist_for_each_entry(trx, &bts->trx_list, list)
171 e1_reconfig_trx(trx);
172
173 /* notify E1 input something has changed */
174 return e1inp_line_update(line);
175}
176
177#if 0
178/* do some compiled-in configuration for our BTS/E1 setup */
179int e1_config(struct gsm_bts *bts, int cardnr, int release_l2)
180{
181 struct e1inp_line *line;
182 struct e1inp_ts *sign_ts;
183 struct e1inp_sign_link *oml_link, *rsl_link;
184 struct gsm_bts_trx *trx = bts->c0;
185 int base_ts;
186
187 switch (bts->nr) {
188 case 0:
189 /* First BTS uses E1 TS 01,02,03,04,05 */
190 base_ts = HARDCODED_BTS0_TS - 1;
191 break;
192 case 1:
193 /* Second BTS uses E1 TS 06,07,08,09,10 */
194 base_ts = HARDCODED_BTS1_TS - 1;
195 break;
196 case 2:
197 /* Third BTS uses E1 TS 11,12,13,14,15 */
198 base_ts = HARDCODED_BTS2_TS - 1;
199 default:
200 return -EINVAL;
201 }
202
203 line = talloc_zero(tall_bsc_ctx, struct e1inp_line);
204 if (!line)
205 return -ENOMEM;
206
207 /* create E1 timeslots for signalling and TRAU frames */
208 e1inp_ts_config(&line->ts[base_ts+1-1], line, E1INP_TS_TYPE_SIGN);
209 e1inp_ts_config(&line->ts[base_ts+2-1], line, E1INP_TS_TYPE_TRAU);
210 e1inp_ts_config(&line->ts[base_ts+3-1], line, E1INP_TS_TYPE_TRAU);
211
212 /* create signalling links for TS1 */
213 sign_ts = &line->ts[base_ts+1-1];
214 oml_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_OML,
215 trx, TEI_OML, SAPI_OML);
216 rsl_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_RSL,
217 trx, TEI_RSL, SAPI_RSL);
218
219 /* create back-links from bts/trx */
220 bts->oml_link = oml_link;
221 trx->rsl_link = rsl_link;
222
223 /* enable subchannel demuxer on TS2 */
224 subch_demux_activate(&line->ts[base_ts+2-1].trau.demux, 1);
225 subch_demux_activate(&line->ts[base_ts+2-1].trau.demux, 2);
226 subch_demux_activate(&line->ts[base_ts+2-1].trau.demux, 3);
227
228 /* enable subchannel demuxer on TS3 */
229 subch_demux_activate(&line->ts[base_ts+3-1].trau.demux, 0);
230 subch_demux_activate(&line->ts[base_ts+3-1].trau.demux, 1);
231 subch_demux_activate(&line->ts[base_ts+3-1].trau.demux, 2);
232 subch_demux_activate(&line->ts[base_ts+3-1].trau.demux, 3);
233
234 trx = gsm_bts_trx_num(bts, 1);
235 if (trx) {
236 /* create E1 timeslots for TRAU frames of TRX1 */
237 e1inp_ts_config(&line->ts[base_ts+4-1], line, E1INP_TS_TYPE_TRAU);
238 e1inp_ts_config(&line->ts[base_ts+5-1], line, E1INP_TS_TYPE_TRAU);
239
240 /* create RSL signalling link for TRX1 */
241 sign_ts = &line->ts[base_ts+1-1];
242 rsl_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_RSL,
243 trx, TEI_RSL+1, SAPI_RSL);
244 /* create back-links from trx */
245 trx->rsl_link = rsl_link;
246
247 /* enable subchannel demuxer on TS2 */
248 subch_demux_activate(&line->ts[base_ts+4-1].trau.demux, 0);
249 subch_demux_activate(&line->ts[base_ts+4-1].trau.demux, 1);
250 subch_demux_activate(&line->ts[base_ts+4-1].trau.demux, 2);
251 subch_demux_activate(&line->ts[base_ts+4-1].trau.demux, 3);
252
253 /* enable subchannel demuxer on TS3 */
254 subch_demux_activate(&line->ts[base_ts+5-1].trau.demux, 0);
255 subch_demux_activate(&line->ts[base_ts+5-1].trau.demux, 1);
256 subch_demux_activate(&line->ts[base_ts+5-1].trau.demux, 2);
257 subch_demux_activate(&line->ts[base_ts+5-1].trau.demux, 3);
258 }
259
260 return mi_setup(cardnr, line, release_l2);
261}
262#endif
263
264/* configure pseudo E1 line in ip.access style and connect to BTS */
265int ia_config_connect(struct gsm_bts *bts, struct sockaddr_in *sin)
266{
267 struct e1inp_line *line;
268 struct e1inp_ts *sign_ts, *rsl_ts;
269 struct e1inp_sign_link *oml_link, *rsl_link;
270
271 line = talloc_zero(tall_bsc_ctx, struct e1inp_line);
272 if (!line)
273 return -ENOMEM;
274
275 /* create E1 timeslots for signalling and TRAU frames */
276 e1inp_ts_config(&line->ts[1-1], line, E1INP_TS_TYPE_SIGN);
277 e1inp_ts_config(&line->ts[2-1], line, E1INP_TS_TYPE_SIGN);
278
279 /* create signalling links for TS1 */
280 sign_ts = &line->ts[1-1];
281 rsl_ts = &line->ts[2-1];
282 oml_link = e1inp_sign_link_create(sign_ts, E1INP_SIGN_OML,
283 bts->c0, 0xff, 0);
284 rsl_link = e1inp_sign_link_create(rsl_ts, E1INP_SIGN_RSL,
285 bts->c0, 0, 0);
286
287 /* create back-links from bts/trx */
288 bts->oml_link = oml_link;
289 bts->c0->rsl_link = rsl_link;
290
291 /* default port at BTS for incoming connections is 3006 */
292 if (sin->sin_port == 0)
293 sin->sin_port = htons(3006);
294
295 return ipaccess_connect(line, sin);
296}