blob: 09b2726671253da01439bae86cf38d3a22af3d49 [file] [log] [blame]
Harald Weltec4338de2015-12-24 00:40:52 +01001/* IuCS/IuPS Core Network interface of HNB-GW */
2
3/* (C) 2015 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
Neels Hofmeyr37017f52016-04-15 22:47:21 +020021#include <arpa/inet.h>
22
Harald Weltec4338de2015-12-24 00:40:52 +010023#include <osmocom/core/msgb.h>
24#include <osmocom/core/utils.h>
25#include <osmocom/core/timer.h>
26
27#include <osmocom/sigtran/protocol/sua.h>
28#include <osmocom/sigtran/sua.h>
29#include <osmocom/sigtran/sccp_sap.h>
Neels Hofmeyr3da86082016-03-30 12:36:15 +020030#include <osmocom/sigtran/sccp_helpers.h>
Harald Weltec4338de2015-12-24 00:40:52 +010031
32#include "hnbgw.h"
33#include "hnbgw_rua.h"
Neels Hofmeyr96979af2016-01-05 15:19:44 +010034#include <osmocom/ranap/ranap_ies_defs.h>
35#include <osmocom/ranap/ranap_msg_factory.h>
Harald Weltec4338de2015-12-24 00:40:52 +010036#include "context_map.h"
Harald Weltec4338de2015-12-24 00:40:52 +010037
Harald Weltec4338de2015-12-24 00:40:52 +010038/***********************************************************************
39 * Outbound RANAP RESET to CN
40 ***********************************************************************/
41
42int hnbgw_cnlink_change_state(struct hnbgw_cnlink *cnlink, enum hnbgw_cnlink_state state);
43
44static int transmit_rst(struct hnbgw_cnlink *cnlink)
45{
46 struct msgb *msg;
47 struct msgb *msgprim;
48 RANAP_CN_DomainIndicator_t domain;
49 RANAP_Cause_t cause = {
50 .present = RANAP_Cause_PR_transmissionNetwork,
51 .choice. transmissionNetwork = RANAP_CauseTransmissionNetwork_signalling_transport_resource_failure,
52 };
53
54 if (cnlink->is_ps)
55 domain = RANAP_CN_DomainIndicator_ps_domain;
56 else
57 domain = RANAP_CN_DomainIndicator_cs_domain;
58
59 msg = ranap_new_msg_reset(domain, &cause);
60
Neels Hofmeyr576f6422016-03-30 13:32:40 +020061 return osmo_sccp_tx_unitdata_msg(cnlink->sua_link, &cnlink->local_addr,
62 &cnlink->remote_addr, msg);
Harald Weltec4338de2015-12-24 00:40:52 +010063}
64
65/* Timer callback once T_RafC expires */
66static void cnlink_trafc_cb(void *data)
67{
68 struct hnbgw_cnlink *cnlink = data;
69
70 transmit_rst(cnlink);
71 hnbgw_cnlink_change_state(cnlink, CNLINK_S_EST_RST_TX_WAIT_ACK);
72 /* The spec states that we should abandon after a configurable
73 * number of times. We decide to simply continue trying */
74}
75
76/* change the state of a CN Link */
77int hnbgw_cnlink_change_state(struct hnbgw_cnlink *cnlink, enum hnbgw_cnlink_state state)
78{
79 switch (state) {
80 case CNLINK_S_NULL:
81 case CNLINK_S_EST_PEND:
82 break;
83 case CNLINK_S_EST_CONF:
84 cnlink_trafc_cb(cnlink);
85 break;
86 case CNLINK_S_EST_RST_TX_WAIT_ACK:
87 osmo_timer_schedule(&cnlink->T_RafC, 5, 0);
88 break;
89 case CNLINK_S_EST_ACTIVE:
90 osmo_timer_del(&cnlink->T_RafC);
91 break;
92 }
93}
94
95/***********************************************************************
96 * Incoming primitives from SCCP User SAP
97 ***********************************************************************/
98
99static int cn_ranap_rx_reset_cmd(struct hnbgw_cnlink *cnlink,
100 RANAP_InitiatingMessage_t *imsg)
101{
102 RANAP_ResetIEs_t ies;
103 int rc;
104
105 rc = ranap_decode_reseties(&ies, &imsg->value);
106 /* FIXME: reset resources and return reset ack */
Daniel Willmann11e912a2016-01-07 13:19:30 +0100107
108 ranap_free_reseties(&ies);
Harald Weltec4338de2015-12-24 00:40:52 +0100109 return rc;
110}
111
112static int cn_ranap_rx_reset_ack(struct hnbgw_cnlink *cnlink,
113 RANAP_SuccessfulOutcome_t *omsg)
114{
115 RANAP_ResetAcknowledgeIEs_t ies;
116 int rc;
117
118 rc = ranap_decode_resetacknowledgeies(&ies, &omsg->value);
119
120 hnbgw_cnlink_change_state(cnlink, CNLINK_S_EST_ACTIVE);
121
Daniel Willmann11e912a2016-01-07 13:19:30 +0100122 ranap_free_resetacknowledgeies(&ies);
Harald Weltec4338de2015-12-24 00:40:52 +0100123 return rc;
124}
125
126static int cn_ranap_rx_paging_cmd(struct hnbgw_cnlink *cnlink,
Harald Weltebc4560c2015-12-24 08:46:58 +0100127 RANAP_InitiatingMessage_t *imsg,
128 const uint8_t *data, unsigned int len)
Harald Weltec4338de2015-12-24 00:40:52 +0100129{
Neels Hofmeyr2b01f3a2016-04-19 17:57:03 +0200130 struct hnb_gw *gw = cnlink->gw;
Harald Weltebc4560c2015-12-24 08:46:58 +0100131 struct hnb_context *hnb;
Harald Weltec4338de2015-12-24 00:40:52 +0100132 RANAP_PagingIEs_t ies;
Harald Weltebc4560c2015-12-24 08:46:58 +0100133 int rc = 0;
Harald Weltec4338de2015-12-24 00:40:52 +0100134
135 rc = ranap_decode_pagingies(&ies, &imsg->value);
136
Harald Weltebc4560c2015-12-24 08:46:58 +0100137 /* FIXME: determine which HNBs to send this Paging command,
138 * rather than broadcasting to all HNBs */
139 llist_for_each_entry(hnb, &gw->hnb_list, list) {
140 rc = rua_tx_udt(hnb, data, len);
141 }
Daniel Willmann11e912a2016-01-07 13:19:30 +0100142
143 ranap_free_pagingies(&ies);
Harald Weltebc4560c2015-12-24 08:46:58 +0100144 return 0;
Harald Weltec4338de2015-12-24 00:40:52 +0100145}
146
147static int cn_ranap_rx_initiating_msg(struct hnbgw_cnlink *cnlink,
Harald Weltebc4560c2015-12-24 08:46:58 +0100148 RANAP_InitiatingMessage_t *imsg,
149 const uint8_t *data, unsigned int len)
Harald Weltec4338de2015-12-24 00:40:52 +0100150{
151 int rc;
152
153 switch (imsg->procedureCode) {
154 case RANAP_ProcedureCode_id_Reset:
155 return cn_ranap_rx_reset_cmd(cnlink, imsg);
156 case RANAP_ProcedureCode_id_Paging:
Harald Weltebc4560c2015-12-24 08:46:58 +0100157 return cn_ranap_rx_paging_cmd(cnlink, imsg, data, len);
Harald Weltec4338de2015-12-24 00:40:52 +0100158 case RANAP_ProcedureCode_id_OverloadControl: /* Overload ind */
159 break;
160 case RANAP_ProcedureCode_id_ErrorIndication: /* Error ind */
161 break;
162 case RANAP_ProcedureCode_id_ResetResource: /* request */
163 case RANAP_ProcedureCode_id_InformationTransfer:
164 case RANAP_ProcedureCode_id_DirectInformationTransfer:
165 case RANAP_ProcedureCode_id_UplinkInformationExchange:
166 LOGP(DRANAP, LOGL_NOTICE, "Received unsupported RANAP "
167 "Procedure %u from CN, ignoring\n", imsg->procedureCode);
168 break;
169 default:
170 LOGP(DRANAP, LOGL_NOTICE, "Received suspicious RANAP "
171 "Procedure %u from CN, ignoring\n", imsg->procedureCode);
172 break;
173 }
174 return 0;
175}
176
177static int cn_ranap_rx_successful_msg(struct hnbgw_cnlink *cnlink,
178 RANAP_SuccessfulOutcome_t *omsg)
179{
180 int rc;
181
182 switch (omsg->procedureCode) {
183 case RANAP_ProcedureCode_id_Reset: /* Reset acknowledge */
184 return cn_ranap_rx_reset_ack(cnlink, omsg);
185 case RANAP_ProcedureCode_id_ResetResource: /* response */
186 case RANAP_ProcedureCode_id_InformationTransfer:
187 case RANAP_ProcedureCode_id_DirectInformationTransfer:
188 case RANAP_ProcedureCode_id_UplinkInformationExchange:
189 LOGP(DRANAP, LOGL_NOTICE, "Received unsupported RANAP "
190 "Procedure %u from CN, ignoring\n", omsg->procedureCode);
191 break;
192 default:
193 LOGP(DRANAP, LOGL_NOTICE, "Received suspicious RANAP "
194 "Procedure %u from CN, ignoring\n", omsg->procedureCode);
195 break;
196 }
197 return 0;
198}
199
200
Harald Weltebc4560c2015-12-24 08:46:58 +0100201static int _cn_ranap_rx(struct hnbgw_cnlink *cnlink, RANAP_RANAP_PDU_t *pdu,
202 const uint8_t *data, unsigned int len)
Harald Weltec4338de2015-12-24 00:40:52 +0100203{
204 int rc;
205
206 switch (pdu->present) {
207 case RANAP_RANAP_PDU_PR_initiatingMessage:
Harald Weltebc4560c2015-12-24 08:46:58 +0100208 rc = cn_ranap_rx_initiating_msg(cnlink, &pdu->choice.initiatingMessage,
209 data, len);
Harald Weltec4338de2015-12-24 00:40:52 +0100210 break;
211 case RANAP_RANAP_PDU_PR_successfulOutcome:
212 rc = cn_ranap_rx_successful_msg(cnlink, &pdu->choice.successfulOutcome);
213 break;
214 case RANAP_RANAP_PDU_PR_unsuccessfulOutcome:
215 LOGP(DRANAP, LOGL_NOTICE, "Received unsupported RANAP "
216 "unsuccessful outcome procedure %u from CN, ignoring\n",
217 pdu->choice.unsuccessfulOutcome.procedureCode);
218 break;
219 default:
220 LOGP(DRANAP, LOGL_NOTICE, "Received suspicious RANAP "
221 "presence %u from CN, ignoring\n", pdu->present);
222 break;
223 }
224}
225
226static int handle_cn_ranap(struct hnbgw_cnlink *cnlink, const uint8_t *data,
227 unsigned int len)
228{
229 RANAP_RANAP_PDU_t _pdu, *pdu = &_pdu;
230 asn_dec_rval_t dec_ret;
231 int rc;
232
233 memset(pdu, 0, sizeof(*pdu));
234 dec_ret = aper_decode(NULL,&asn_DEF_RANAP_RANAP_PDU, (void **) &pdu,
235 data, len, 0, 0);
236 if (dec_ret.code != RC_OK) {
237 LOGP(DRANAP, LOGL_ERROR, "Error in RANAP ASN.1 decode\n");
238 return rc;
239 }
240
Harald Weltebc4560c2015-12-24 08:46:58 +0100241 rc = _cn_ranap_rx(cnlink, pdu, data, len);
Harald Weltec4338de2015-12-24 00:40:52 +0100242
243 return rc;
244}
245
246
247static int handle_cn_unitdata(struct hnbgw_cnlink *cnlink,
248 const struct osmo_scu_unitdata_param *param,
249 struct osmo_prim_hdr *oph)
250{
Harald Welte8c572fe2015-12-26 08:42:07 +0100251 if (param->called_addr.ssn != OSMO_SCCP_SSN_RANAP) {
Harald Weltec4338de2015-12-24 00:40:52 +0100252 LOGP(DMAIN, LOGL_NOTICE, "N-UNITDATA.ind for unknown SSN %u\n",
253 param->called_addr.ssn);
254 return -1;
255 }
256
257 return handle_cn_ranap(cnlink, msgb_l2(oph->msg), msgb_l2len(oph->msg));
258}
259
Neels Hofmeyr65037672016-04-19 18:00:54 +0200260static int handle_cn_conn_conf(struct hnbgw_cnlink *cnlink,
Neels Hofmeyr630483b2016-04-19 18:01:25 +0200261 const struct osmo_scu_connect_param *param,
262 struct osmo_prim_hdr *oph)
Harald Weltec4338de2015-12-24 00:40:52 +0100263{
264 /* we don't actually need to do anything, as RUA towards the HNB
265 * doesn't seem to know any confirmations to its CONNECT
266 * operation */
267
Neels Hofmeyr0ff24432016-04-04 18:33:33 +0200268 LOGP(DMAIN, LOGL_DEBUG, "handle_cn_conn_conf() conn_id=%d\n",
269 param->conn_id);
270 LOGP(DMAIN, LOGL_DEBUG, "handle_cn_conn_conf() called_addr=%s\n",
271 inet_ntoa(param->called_addr.ip.v4));
272 LOGP(DMAIN, LOGL_DEBUG, "handle_cn_conn_conf() calling_addr=%s\n",
273 inet_ntoa(param->calling_addr.ip.v4));
274 LOGP(DMAIN, LOGL_DEBUG, "handle_cn_conn_conf() responding_addr=%s\n",
275 inet_ntoa(param->responding_addr.ip.v4));
276
Harald Weltec4338de2015-12-24 00:40:52 +0100277 return 0;
278}
279
Neels Hofmeyr65037672016-04-19 18:00:54 +0200280static int handle_cn_data_ind(struct hnbgw_cnlink *cnlink,
Harald Weltec4338de2015-12-24 00:40:52 +0100281 const struct osmo_scu_data_param *param,
282 struct osmo_prim_hdr *oph)
283{
284 struct hnbgw_context_map *map;
285
286 /* connection-oriented data is always passed transparently
287 * towards the specific HNB, via a RUA connection identified by
288 * conn_id */
289
290 map = context_map_by_cn(cnlink, param->conn_id);
291 if (!map) {
292 /* FIXME: Return an error / released primitive */
293 return 0;
294 }
295
296 return rua_tx_dt(map->hnb_ctx, map->cn_link->is_ps, map->rua_ctx_id,
297 msgb_l2(oph->msg), msgb_l2len(oph->msg));
298}
299
Neels Hofmeyr65037672016-04-19 18:00:54 +0200300static int handle_cn_disc_ind(struct hnbgw_cnlink *cnlink,
Harald Weltec4338de2015-12-24 00:40:52 +0100301 const struct osmo_scu_disconn_param *param,
302 struct osmo_prim_hdr *oph)
303{
304 struct hnbgw_context_map *map;
305
Neels Hofmeyr02be4e32016-04-04 19:29:35 +0200306 LOGP(DMAIN, LOGL_DEBUG, "handle_cn_disc_ind() conn_id=%d originator=%d\n",
307 param->conn_id, param->originator);
308 LOGP(DMAIN, LOGL_DEBUG, "handle_cn_disc_ind() responding_addr=%s\n",
309 inet_ntoa(param->responding_addr.ip.v4));
310
Harald Weltec4338de2015-12-24 00:40:52 +0100311 RUA_Cause_t rua_cause = {
312 .present = RUA_Cause_PR_NOTHING,
313 /* FIXME: Convert incoming SCCP cause to RUA cause */
314 };
315
316 /* we need to notify the HNB associated with this connection via
317 * a RUA DISCONNECT */
318
319 map = context_map_by_cn(cnlink, param->conn_id);
320 if (!map) {
321 /* FIXME: Return an error / released primitive */
322 return 0;
323 }
324
325 return rua_tx_disc(map->hnb_ctx, map->cn_link->is_ps, map->rua_ctx_id,
326 &rua_cause, msgb_l2(oph->msg), msgb_l2len(oph->msg));
327}
328
329/* Entry point for primitives coming up from SCCP User SAP */
Neels Hofmeyra1bf4f32016-07-07 15:36:07 +0200330static int sccp_sap_up(struct osmo_prim_hdr *oph, void *ctx)
Harald Weltec4338de2015-12-24 00:40:52 +0100331{
Neels Hofmeyra1bf4f32016-07-07 15:36:07 +0200332 struct osmo_sccp_link *slink = ctx;
Neels Hofmeyr65037672016-04-19 18:00:54 +0200333 struct hnbgw_cnlink *cnlink = osmo_sua_link_get_user_priv(slink);
Harald Weltec4338de2015-12-24 00:40:52 +0100334 struct osmo_scu_prim *prim = (struct osmo_scu_prim *) oph;
335 int rc;
336
337 LOGP(DMAIN, LOGL_DEBUG, "sccp_sap_up(%s)\n", osmo_scu_prim_name(oph));
338
339 switch (OSMO_PRIM_HDR(oph)) {
340 case OSMO_PRIM(OSMO_SCU_PRIM_N_UNITDATA, PRIM_OP_INDICATION):
Neels Hofmeyr65037672016-04-19 18:00:54 +0200341 rc = handle_cn_unitdata(cnlink, &prim->u.unitdata, oph);
Harald Weltec4338de2015-12-24 00:40:52 +0100342 break;
343 case OSMO_PRIM(OSMO_SCU_PRIM_N_CONNECT, PRIM_OP_CONFIRM):
Neels Hofmeyr65037672016-04-19 18:00:54 +0200344 rc = handle_cn_conn_conf(cnlink, &prim->u.connect, oph);
Harald Weltec4338de2015-12-24 00:40:52 +0100345 break;
346 case OSMO_PRIM(OSMO_SCU_PRIM_N_DATA, PRIM_OP_INDICATION):
Neels Hofmeyr65037672016-04-19 18:00:54 +0200347 rc = handle_cn_data_ind(cnlink, &prim->u.data, oph);
Harald Weltec4338de2015-12-24 00:40:52 +0100348 break;
349 case OSMO_PRIM(OSMO_SCU_PRIM_N_DISCONNECT, PRIM_OP_INDICATION):
Neels Hofmeyr65037672016-04-19 18:00:54 +0200350 rc = handle_cn_disc_ind(cnlink, &prim->u.disconnect, oph);
Harald Weltec4338de2015-12-24 00:40:52 +0100351 break;
352 defualt:
353 LOGP(DMAIN, LOGL_ERROR,
354 "Received unknown prim %u from SCCP USER SAP\n",
355 OSMO_PRIM_HDR(oph));
356 break;
357 }
358
359 msgb_free(oph->msg);
360
361 return 0;
362}
363
364
Neels Hofmeyr1befe6b2016-04-04 18:05:36 +0200365/* Set up a link towards the core network for the circuit switched (is_ps == 0)
366 * or packet switched (is_ps != 0) domain. */
Daniel Willmann4deab942016-01-14 15:35:11 +0100367struct hnbgw_cnlink *hnbgw_cnlink_init(struct hnb_gw *gw, const char *host, uint16_t port, int is_ps)
Harald Weltec4338de2015-12-24 00:40:52 +0100368{
369 struct hnbgw_cnlink *cnlink = talloc_zero(gw, struct hnbgw_cnlink);
370 int rc;
371
Neels Hofmeyr2b01f3a2016-04-19 17:57:03 +0200372 cnlink->gw = gw;
Harald Weltec4338de2015-12-24 00:40:52 +0100373 INIT_LLIST_HEAD(&cnlink->map_list);
374 cnlink->T_RafC.cb = cnlink_trafc_cb;
375 cnlink->T_RafC.data = cnlink;
Harald Welte552fdf12015-12-26 23:39:30 +0100376 cnlink->next_conn_id = 1000;
Daniel Willmann4deab942016-01-14 15:35:11 +0100377 cnlink->is_ps = is_ps;
Neels Hofmeyr576f6422016-03-30 13:32:40 +0200378 osmo_sccp_make_addr_pc_ssn(&cnlink->local_addr, 2,
379 OSMO_SCCP_SSN_RANAP);
380 osmo_sccp_make_addr_pc_ssn(&cnlink->remote_addr, 1,
381 OSMO_SCCP_SSN_RANAP);
Harald Weltec4338de2015-12-24 00:40:52 +0100382
Neels Hofmeyrd045ceb2016-04-19 18:04:56 +0200383 LOGP(DRUA, LOGL_DEBUG, "New hnbgw_cnlink %p (gw %p): %s %d %s\n",
384 cnlink, cnlink->gw, host, port, is_ps? "PS" : "CS");
385
Harald Welte2ebe42f2015-12-26 23:38:38 +0100386 cnlink->sua_user = osmo_sua_user_create(cnlink, sccp_sap_up, cnlink);
Harald Weltec4338de2015-12-24 00:40:52 +0100387 if (!cnlink->sua_user) {
388 LOGP(DMAIN, LOGL_ERROR, "Failed to init SUA\n");
389 goto out_free;
390 }
391 rc = osmo_sua_client_connect(cnlink->sua_user, host, port);
392 if (rc < 0) {
393 LOGP(DMAIN, LOGL_ERROR, "Failed to connect SUA\n");
394 goto out_user;
395 }
396 cnlink->sua_link = osmo_sua_client_get_link(cnlink->sua_user);
397 if (!cnlink->sua_link) {
398 LOGP(DMAIN, LOGL_ERROR, "Failed to get SUA link\n");
399 goto out_disconnect;
400 }
401
402 llist_add_tail(&cnlink->list, &gw->cn_list);
403
404 return cnlink;
405
406out_disconnect:
407 /* FIXME */
408out_user:
409 osmo_sua_user_destroy(cnlink->sua_user);
410out_free:
411 talloc_free(cnlink);
412}