blob: 8333d781ba57b5bba5eacff5edd03db508b3c3ea [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
21#include <osmocom/core/msgb.h>
22#include <osmocom/core/utils.h>
23#include <osmocom/core/timer.h>
24
25#include <osmocom/sigtran/protocol/sua.h>
26#include <osmocom/sigtran/sua.h>
27#include <osmocom/sigtran/sccp_sap.h>
28
29#include "hnbgw.h"
30#include "hnbgw_rua.h"
31#include "ranap_ies_defs.h"
32#include "ranap_msg_factory.h"
33#include "context_map.h"
34#include "sccp_helpers.h"
35
36#define SCCP_SSN_RANAP 143
37
38/***********************************************************************
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
61 return sccp_tx_unitdata_msg(cnlink->sua_link, &cnlink->local_addr,
62 &cnlink->remote_addr, msg);
63}
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 */
107 return rc;
108}
109
110static int cn_ranap_rx_reset_ack(struct hnbgw_cnlink *cnlink,
111 RANAP_SuccessfulOutcome_t *omsg)
112{
113 RANAP_ResetAcknowledgeIEs_t ies;
114 int rc;
115
116 rc = ranap_decode_resetacknowledgeies(&ies, &omsg->value);
117
118 hnbgw_cnlink_change_state(cnlink, CNLINK_S_EST_ACTIVE);
119
120 return rc;
121}
122
123static int cn_ranap_rx_paging_cmd(struct hnbgw_cnlink *cnlink,
Harald Weltebc4560c2015-12-24 08:46:58 +0100124 RANAP_InitiatingMessage_t *imsg,
125 const uint8_t *data, unsigned int len)
Harald Weltec4338de2015-12-24 00:40:52 +0100126{
Harald Weltebc4560c2015-12-24 08:46:58 +0100127 struct hnb_gw *gw;
128 struct hnb_context *hnb;
Harald Weltec4338de2015-12-24 00:40:52 +0100129 RANAP_PagingIEs_t ies;
Harald Weltebc4560c2015-12-24 08:46:58 +0100130 int rc = 0;
Harald Weltec4338de2015-12-24 00:40:52 +0100131
132 rc = ranap_decode_pagingies(&ies, &imsg->value);
133
Harald Weltebc4560c2015-12-24 08:46:58 +0100134 /* FIXME: determine which HNBs to send this Paging command,
135 * rather than broadcasting to all HNBs */
136 llist_for_each_entry(hnb, &gw->hnb_list, list) {
137 rc = rua_tx_udt(hnb, data, len);
138 }
139 return 0;
Harald Weltec4338de2015-12-24 00:40:52 +0100140}
141
142static int cn_ranap_rx_initiating_msg(struct hnbgw_cnlink *cnlink,
Harald Weltebc4560c2015-12-24 08:46:58 +0100143 RANAP_InitiatingMessage_t *imsg,
144 const uint8_t *data, unsigned int len)
Harald Weltec4338de2015-12-24 00:40:52 +0100145{
146 int rc;
147
148 switch (imsg->procedureCode) {
149 case RANAP_ProcedureCode_id_Reset:
150 return cn_ranap_rx_reset_cmd(cnlink, imsg);
151 case RANAP_ProcedureCode_id_Paging:
Harald Weltebc4560c2015-12-24 08:46:58 +0100152 return cn_ranap_rx_paging_cmd(cnlink, imsg, data, len);
Harald Weltec4338de2015-12-24 00:40:52 +0100153 case RANAP_ProcedureCode_id_OverloadControl: /* Overload ind */
154 break;
155 case RANAP_ProcedureCode_id_ErrorIndication: /* Error ind */
156 break;
157 case RANAP_ProcedureCode_id_ResetResource: /* request */
158 case RANAP_ProcedureCode_id_InformationTransfer:
159 case RANAP_ProcedureCode_id_DirectInformationTransfer:
160 case RANAP_ProcedureCode_id_UplinkInformationExchange:
161 LOGP(DRANAP, LOGL_NOTICE, "Received unsupported RANAP "
162 "Procedure %u from CN, ignoring\n", imsg->procedureCode);
163 break;
164 default:
165 LOGP(DRANAP, LOGL_NOTICE, "Received suspicious RANAP "
166 "Procedure %u from CN, ignoring\n", imsg->procedureCode);
167 break;
168 }
169 return 0;
170}
171
172static int cn_ranap_rx_successful_msg(struct hnbgw_cnlink *cnlink,
173 RANAP_SuccessfulOutcome_t *omsg)
174{
175 int rc;
176
177 switch (omsg->procedureCode) {
178 case RANAP_ProcedureCode_id_Reset: /* Reset acknowledge */
179 return cn_ranap_rx_reset_ack(cnlink, omsg);
180 case RANAP_ProcedureCode_id_ResetResource: /* response */
181 case RANAP_ProcedureCode_id_InformationTransfer:
182 case RANAP_ProcedureCode_id_DirectInformationTransfer:
183 case RANAP_ProcedureCode_id_UplinkInformationExchange:
184 LOGP(DRANAP, LOGL_NOTICE, "Received unsupported RANAP "
185 "Procedure %u from CN, ignoring\n", omsg->procedureCode);
186 break;
187 default:
188 LOGP(DRANAP, LOGL_NOTICE, "Received suspicious RANAP "
189 "Procedure %u from CN, ignoring\n", omsg->procedureCode);
190 break;
191 }
192 return 0;
193}
194
195
Harald Weltebc4560c2015-12-24 08:46:58 +0100196static int _cn_ranap_rx(struct hnbgw_cnlink *cnlink, RANAP_RANAP_PDU_t *pdu,
197 const uint8_t *data, unsigned int len)
Harald Weltec4338de2015-12-24 00:40:52 +0100198{
199 int rc;
200
201 switch (pdu->present) {
202 case RANAP_RANAP_PDU_PR_initiatingMessage:
Harald Weltebc4560c2015-12-24 08:46:58 +0100203 rc = cn_ranap_rx_initiating_msg(cnlink, &pdu->choice.initiatingMessage,
204 data, len);
Harald Weltec4338de2015-12-24 00:40:52 +0100205 break;
206 case RANAP_RANAP_PDU_PR_successfulOutcome:
207 rc = cn_ranap_rx_successful_msg(cnlink, &pdu->choice.successfulOutcome);
208 break;
209 case RANAP_RANAP_PDU_PR_unsuccessfulOutcome:
210 LOGP(DRANAP, LOGL_NOTICE, "Received unsupported RANAP "
211 "unsuccessful outcome procedure %u from CN, ignoring\n",
212 pdu->choice.unsuccessfulOutcome.procedureCode);
213 break;
214 default:
215 LOGP(DRANAP, LOGL_NOTICE, "Received suspicious RANAP "
216 "presence %u from CN, ignoring\n", pdu->present);
217 break;
218 }
219}
220
221static int handle_cn_ranap(struct hnbgw_cnlink *cnlink, const uint8_t *data,
222 unsigned int len)
223{
224 RANAP_RANAP_PDU_t _pdu, *pdu = &_pdu;
225 asn_dec_rval_t dec_ret;
226 int rc;
227
228 memset(pdu, 0, sizeof(*pdu));
229 dec_ret = aper_decode(NULL,&asn_DEF_RANAP_RANAP_PDU, (void **) &pdu,
230 data, len, 0, 0);
231 if (dec_ret.code != RC_OK) {
232 LOGP(DRANAP, LOGL_ERROR, "Error in RANAP ASN.1 decode\n");
233 return rc;
234 }
235
Harald Weltebc4560c2015-12-24 08:46:58 +0100236 rc = _cn_ranap_rx(cnlink, pdu, data, len);
Harald Weltec4338de2015-12-24 00:40:52 +0100237
238 return rc;
239}
240
241
242static int handle_cn_unitdata(struct hnbgw_cnlink *cnlink,
243 const struct osmo_scu_unitdata_param *param,
244 struct osmo_prim_hdr *oph)
245{
246 if (param->called_addr.ssn != SCCP_SSN_RANAP) {
247 LOGP(DMAIN, LOGL_NOTICE, "N-UNITDATA.ind for unknown SSN %u\n",
248 param->called_addr.ssn);
249 return -1;
250 }
251
252 return handle_cn_ranap(cnlink, msgb_l2(oph->msg), msgb_l2len(oph->msg));
253}
254
255static int handle_cn_conn_conf(struct hnbgw_cnlink *cnlink,
256 const struct osmo_scu_connect_param *param,
257 struct osmo_prim_hdr *oph)
258{
259 /* we don't actually need to do anything, as RUA towards the HNB
260 * doesn't seem to know any confirmations to its CONNECT
261 * operation */
262
263 return 0;
264}
265
266static int handle_cn_data_ind(struct hnbgw_cnlink *cnlink,
267 const struct osmo_scu_data_param *param,
268 struct osmo_prim_hdr *oph)
269{
270 struct hnbgw_context_map *map;
271
272 /* connection-oriented data is always passed transparently
273 * towards the specific HNB, via a RUA connection identified by
274 * conn_id */
275
276 map = context_map_by_cn(cnlink, param->conn_id);
277 if (!map) {
278 /* FIXME: Return an error / released primitive */
279 return 0;
280 }
281
282 return rua_tx_dt(map->hnb_ctx, map->cn_link->is_ps, map->rua_ctx_id,
283 msgb_l2(oph->msg), msgb_l2len(oph->msg));
284}
285
286static int handle_cn_disc_ind(struct hnbgw_cnlink *cnlink,
287 const struct osmo_scu_disconn_param *param,
288 struct osmo_prim_hdr *oph)
289{
290 struct hnbgw_context_map *map;
291
292 RUA_Cause_t rua_cause = {
293 .present = RUA_Cause_PR_NOTHING,
294 /* FIXME: Convert incoming SCCP cause to RUA cause */
295 };
296
297 /* we need to notify the HNB associated with this connection via
298 * a RUA DISCONNECT */
299
300 map = context_map_by_cn(cnlink, param->conn_id);
301 if (!map) {
302 /* FIXME: Return an error / released primitive */
303 return 0;
304 }
305
306 return rua_tx_disc(map->hnb_ctx, map->cn_link->is_ps, map->rua_ctx_id,
307 &rua_cause, msgb_l2(oph->msg), msgb_l2len(oph->msg));
308}
309
310/* Entry point for primitives coming up from SCCP User SAP */
311static int sccp_sap_up(struct osmo_prim_hdr *oph, void *slink)
312{
313 struct osmo_scu_prim *prim = (struct osmo_scu_prim *) oph;
314 int rc;
315
316 LOGP(DMAIN, LOGL_DEBUG, "sccp_sap_up(%s)\n", osmo_scu_prim_name(oph));
317
318 switch (OSMO_PRIM_HDR(oph)) {
319 case OSMO_PRIM(OSMO_SCU_PRIM_N_UNITDATA, PRIM_OP_INDICATION):
320 rc = handle_cn_unitdata(slink, &prim->u.unitdata, oph);
321 break;
322 case OSMO_PRIM(OSMO_SCU_PRIM_N_CONNECT, PRIM_OP_CONFIRM):
323 rc = handle_cn_conn_conf(slink, &prim->u.connect, oph);
324 break;
325 case OSMO_PRIM(OSMO_SCU_PRIM_N_DATA, PRIM_OP_INDICATION):
326 rc = handle_cn_data_ind(slink, &prim->u.data, oph);
327 break;
328 case OSMO_PRIM(OSMO_SCU_PRIM_N_DISCONNECT, PRIM_OP_INDICATION):
329 rc = handle_cn_disc_ind(slink, &prim->u.disconnect, oph);
330 break;
331 defualt:
332 LOGP(DMAIN, LOGL_ERROR,
333 "Received unknown prim %u from SCCP USER SAP\n",
334 OSMO_PRIM_HDR(oph));
335 break;
336 }
337
338 msgb_free(oph->msg);
339
340 return 0;
341}
342
343
344struct hnbgw_cnlink *hnbgw_cnlink_init(struct hnb_gw *gw, const char *host, uint16_t port)
345{
346 struct hnbgw_cnlink *cnlink = talloc_zero(gw, struct hnbgw_cnlink);
347 int rc;
348
349 INIT_LLIST_HEAD(&cnlink->map_list);
350 cnlink->T_RafC.cb = cnlink_trafc_cb;
351 cnlink->T_RafC.data = cnlink;
352 sccp_make_addr_pc_ssn(&cnlink->local_addr, 2, SCCP_SSN_RANAP);
353 sccp_make_addr_pc_ssn(&cnlink->remote_addr, 1, SCCP_SSN_RANAP);
354
355 cnlink->sua_user = osmo_sua_user_create(cnlink, sccp_sap_up);
356 if (!cnlink->sua_user) {
357 LOGP(DMAIN, LOGL_ERROR, "Failed to init SUA\n");
358 goto out_free;
359 }
360 rc = osmo_sua_client_connect(cnlink->sua_user, host, port);
361 if (rc < 0) {
362 LOGP(DMAIN, LOGL_ERROR, "Failed to connect SUA\n");
363 goto out_user;
364 }
365 cnlink->sua_link = osmo_sua_client_get_link(cnlink->sua_user);
366 if (!cnlink->sua_link) {
367 LOGP(DMAIN, LOGL_ERROR, "Failed to get SUA link\n");
368 goto out_disconnect;
369 }
370
371 llist_add_tail(&cnlink->list, &gw->cn_list);
372
373 return cnlink;
374
375out_disconnect:
376 /* FIXME */
377out_user:
378 osmo_sua_user_destroy(cnlink->sua_user);
379out_free:
380 talloc_free(cnlink);
381}