blob: 3e3715cfd02ce872ac2b636d88b039d85a30e312 [file] [log] [blame]
Harald Welte77847ad2015-10-06 22:07:04 +02001/* hnb-gw specific code for HNBAP */
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
Harald Welteba43de42015-08-29 20:33:16 +020021#include <osmocom/core/msgb.h>
Harald Welteb3dae302015-08-30 12:20:09 +020022#include <osmocom/core/utils.h>
Harald Welte10dfc5a2015-09-11 01:34:45 +020023#include <osmocom/gsm/gsm48.h>
Harald Welteba43de42015-08-29 20:33:16 +020024
Harald Welteb3dae302015-08-30 12:20:09 +020025#include <unistd.h>
Harald Welteee77cff2015-08-30 16:57:53 +020026#include <errno.h>
Harald Welteb3dae302015-08-30 12:20:09 +020027#include <string.h>
28
Harald Welte30afef32015-08-30 12:28:29 +020029#include "asn1helpers.h"
Harald Welte1d2c39d2015-09-11 17:49:37 +020030#include "iu_helpers.h"
Harald Welte30afef32015-08-30 12:28:29 +020031
Harald Welteba43de42015-08-29 20:33:16 +020032#include "hnbgw.h"
Harald Welte27f9c4a2015-08-30 22:47:18 +020033#include "hnbap_common.h"
34#include "hnbap_ies_defs.h"
Harald Welteba43de42015-08-29 20:33:16 +020035
Harald Welteee77cff2015-08-30 16:57:53 +020036#define IU_MSG_NUM_IES 32
37#define IU_MSG_NUM_EXT_IES 32
38
Harald Welte27f9c4a2015-08-30 22:47:18 +020039static int hnbgw_hnbap_tx(struct hnb_context *ctx, struct msgb *msg)
Harald Welteee77cff2015-08-30 16:57:53 +020040{
Harald Welte7b54e322015-09-07 22:41:45 +020041 if (!msg)
42 return -EINVAL;
43
Harald Welte3f712562015-09-07 21:53:25 +020044 msgb_ppid(msg) = IUH_PPI_HNBAP;
45 return osmo_wqueue_enqueue(&ctx->wqueue, msg);
Harald Welte27f9c4a2015-08-30 22:47:18 +020046}
Harald Welteee77cff2015-08-30 16:57:53 +020047
Harald Welte27f9c4a2015-08-30 22:47:18 +020048static int hnbgw_tx_hnb_register_acc(struct hnb_context *ctx)
49{
50 HNBRegisterAccept_t accept_out;
51 struct msgb *msg;
52 int rc;
53
54 /* Single required response IE: RNC-ID */
55 HNBRegisterAcceptIEs_t accept = {
56 .rnc_id = ctx->gw->config.rnc_id
57 };
58
59 /* encode the Information Elements */
Harald Welte2204f9d2015-09-07 21:10:50 +020060 memset(&accept_out, 0, sizeof(accept_out));
Harald Welte27f9c4a2015-08-30 22:47:18 +020061 rc = hnbap_encode_hnbregisteraccepties(&accept_out, &accept);
62 if (rc < 0) {
63 return rc;
Harald Welteee77cff2015-08-30 16:57:53 +020064 }
65
Harald Welte27f9c4a2015-08-30 22:47:18 +020066 /* generate a successfull outcome PDU */
67 msg = hnbap_generate_successful_outcome(ProcedureCode_id_HNBRegister,
68 Criticality_reject,
69 &asn_DEF_HNBRegisterAccept,
70 &accept_out);
Harald Welteee77cff2015-08-30 16:57:53 +020071
Harald Welte27f9c4a2015-08-30 22:47:18 +020072 return hnbgw_hnbap_tx(ctx, msg);
Harald Welteee77cff2015-08-30 16:57:53 +020073}
74
75
Harald Welte27f9c4a2015-08-30 22:47:18 +020076static int hnbgw_tx_ue_register_acc(struct ue_context *ue)
Harald Welteba43de42015-08-29 20:33:16 +020077{
Harald Welte27f9c4a2015-08-30 22:47:18 +020078 UERegisterAccept_t accept_out;
79 UERegisterAcceptIEs_t accept;
80 struct msgb *msg;
Harald Welte10dfc5a2015-09-11 01:34:45 +020081 uint8_t encoded_imsi[10];
82 size_t encoded_imsi_len;
Harald Welte27f9c4a2015-08-30 22:47:18 +020083 int rc;
Harald Welteba43de42015-08-29 20:33:16 +020084
Harald Welte1d2c39d2015-09-11 17:49:37 +020085 encoded_imsi_len = encode_iu_imsi(encoded_imsi,
86 sizeof(encoded_imsi), ue->imsi);
Harald Welte10dfc5a2015-09-11 01:34:45 +020087
Harald Welte2204f9d2015-09-07 21:10:50 +020088 memset(&accept, 0, sizeof(accept));
Harald Welte10dfc5a2015-09-11 01:34:45 +020089 accept.uE_Identity.present = UE_Identity_PR_iMSI;
Harald Welte1d2c39d2015-09-11 17:49:37 +020090 OCTET_STRING_fromBuf(&accept.uE_Identity.choice.iMSI,
91 (const char *)encoded_imsi, encoded_imsi_len);
Harald Welte27f9c4a2015-08-30 22:47:18 +020092 asn1_u32_to_bitstring(&accept.context_ID, &ue->context_id);
Harald Welte62868802015-09-18 01:23:55 +020093 accept.context_ID.size = 3; /* 24bit field */
Harald Welteba43de42015-08-29 20:33:16 +020094
Harald Welte2204f9d2015-09-07 21:10:50 +020095 memset(&accept_out, 0, sizeof(accept_out));
Harald Welte27f9c4a2015-08-30 22:47:18 +020096 rc = hnbap_encode_ueregisteraccepties(&accept_out, &accept);
97 if (rc < 0) {
98 return rc;
Harald Welteba43de42015-08-29 20:33:16 +020099 }
Harald Welte27f9c4a2015-08-30 22:47:18 +0200100
101 msg = hnbap_generate_successful_outcome(ProcedureCode_id_UERegister,
102 Criticality_reject,
103 &asn_DEF_UERegisterAccept,
104 &accept_out);
105 return hnbgw_hnbap_tx(ue->hnb, msg);
Harald Welteba43de42015-08-29 20:33:16 +0200106}
107
Harald Welte27f9c4a2015-08-30 22:47:18 +0200108static int hnbgw_rx_hnb_register_req(struct hnb_context *ctx, ANY_t *in)
Harald Welteba43de42015-08-29 20:33:16 +0200109{
Harald Welte27f9c4a2015-08-30 22:47:18 +0200110 HNBRegisterRequestIEs_t ies;
111 int rc;
Harald Welteba43de42015-08-29 20:33:16 +0200112
Harald Welte27f9c4a2015-08-30 22:47:18 +0200113 rc = hnbap_decode_hnbregisterrequesties(&ies, in);
114 if (rc < 0)
115 return rc;
Harald Welteba43de42015-08-29 20:33:16 +0200116
Harald Weltea2e6a7a2015-08-29 21:47:39 +0200117 /* copy all identity parameters from the message to ctx */
Harald Welte27f9c4a2015-08-30 22:47:18 +0200118 asn1_strncpy(ctx->identity_info, &ies.hnB_Identity.hNB_Identity_Info,
119 sizeof(ctx->identity_info));
120 ctx->id.lac = asn1str_to_u16(&ies.lac);
121 ctx->id.sac = asn1str_to_u16(&ies.sac);
122 ctx->id.rac = asn1str_to_u8(&ies.rac);
123 ctx->id.cid = asn1bitstr_to_u32(&ies.cellIdentity);
Harald Welteb3dae302015-08-30 12:20:09 +0200124 //ctx->id.mcc FIXME
125 //ctx->id.mnc FIXME
Harald Weltea2e6a7a2015-08-29 21:47:39 +0200126
Harald Welteee77cff2015-08-30 16:57:53 +0200127 DEBUGP(DMAIN, "HNB-REGISTER-REQ from %s\n", ctx->identity_info);
128
Harald Welte27f9c4a2015-08-30 22:47:18 +0200129 /* Send HNBRegisterAccept */
130 return hnbgw_tx_hnb_register_acc(ctx);
Harald Welteba43de42015-08-29 20:33:16 +0200131}
132
Harald Welte27f9c4a2015-08-30 22:47:18 +0200133static int hnbgw_rx_ue_register_req(struct hnb_context *ctx, ANY_t *in)
Harald Welteba43de42015-08-29 20:33:16 +0200134{
Harald Welte27f9c4a2015-08-30 22:47:18 +0200135 UERegisterRequestIEs_t ies;
136 struct ue_context *ue;
Harald Welte10dfc5a2015-09-11 01:34:45 +0200137 char imsi[16];
Harald Welte27f9c4a2015-08-30 22:47:18 +0200138 int rc;
Harald Welteba43de42015-08-29 20:33:16 +0200139
Harald Welte27f9c4a2015-08-30 22:47:18 +0200140 rc = hnbap_decode_ueregisterrequesties(&ies, in);
141 if (rc < 0)
142 return rc;
Harald Welteba43de42015-08-29 20:33:16 +0200143
Harald Welte10dfc5a2015-09-11 01:34:45 +0200144 switch (ies.uE_Identity.present) {
145 case UE_Identity_PR_iMSI:
146 decode_iu_bcd(imsi, sizeof(imsi), ies.uE_Identity.choice.iMSI.buf,
147 ies.uE_Identity.choice.iMSI.size);
148 break;
149 case UE_Identity_PR_iMSIDS41:
150 decode_iu_bcd(imsi, sizeof(imsi), ies.uE_Identity.choice.iMSIDS41.buf,
151 ies.uE_Identity.choice.iMSIDS41.size);
152 break;
153 case UE_Identity_PR_iMSIESN:
154 decode_iu_bcd(imsi, sizeof(imsi), ies.uE_Identity.choice.iMSIESN.iMSIDS41.buf,
155 ies.uE_Identity.choice.iMSIESN.iMSIDS41.size);
156 break;
157 default:
Harald Welte1d2c39d2015-09-11 17:49:37 +0200158 LOGP(DMAIN, LOGL_NOTICE, "UE-REGISTER-REQ without IMSI?!?\n");
Harald Welte10dfc5a2015-09-11 01:34:45 +0200159 return -1;
160 }
Harald Welteb534e5c2015-09-11 00:15:16 +0200161
Harald Welte10dfc5a2015-09-11 01:34:45 +0200162 DEBUGP(DMAIN, "UE-REGSITER-REQ ID_type=%d imsi=%s cause=%ld\n",
163 ies.uE_Identity.present, imsi, ies.registration_Cause);
164
165 ue = ue_context_by_imsi(imsi);
166 if (!ue)
167 ue = ue_context_alloc(ctx, imsi);
Harald Welte27f9c4a2015-08-30 22:47:18 +0200168
Harald Welte27f9c4a2015-08-30 22:47:18 +0200169 /* Send UERegisterAccept */
170 return hnbgw_tx_ue_register_acc(ue);
Harald Welteba43de42015-08-29 20:33:16 +0200171}
172
Harald Welte3af1db82015-09-11 17:03:16 +0200173static int hnbgw_rx_err_ind(struct hnb_context *hnb, ANY_t *in)
174{
175 ErrorIndicationIEs_t ies;
176 int rc;
177
Harald Welte1d2c39d2015-09-11 17:49:37 +0200178 rc = hnbap_decode_errorindicationies(&ies, in);
Harald Welte3af1db82015-09-11 17:03:16 +0200179 if (rc < 0)
180 return rc;
181
182 LOGP(DMAIN, LOGL_NOTICE, "HNBAP ERROR.ind, cause: %s\n",
183 hnbap_cause_str(&ies.cause));
184
185 return 0;
186}
187
Harald Welte27f9c4a2015-08-30 22:47:18 +0200188static int hnbgw_rx_initiating_msg(struct hnb_context *hnb, InitiatingMessage_t *imsg)
Harald Welteba43de42015-08-29 20:33:16 +0200189{
190 int rc;
191
Harald Welteb3dae302015-08-30 12:20:09 +0200192 switch (imsg->procedureCode) {
Harald Welte27f9c4a2015-08-30 22:47:18 +0200193 case ProcedureCode_id_HNBRegister: /* 8.2 */
194 rc = hnbgw_rx_hnb_register_req(hnb, &imsg->value);
Harald Welteba43de42015-08-29 20:33:16 +0200195 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200196 case ProcedureCode_id_HNBDe_Register: /* 8.3 */
Harald Welteba43de42015-08-29 20:33:16 +0200197 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200198 case ProcedureCode_id_UERegister: /* 8.4 */
199 rc = hnbgw_rx_ue_register_req(hnb, &imsg->value);
Harald Welteba43de42015-08-29 20:33:16 +0200200 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200201 case ProcedureCode_id_UEDe_Register: /* 8.5 */
Harald Welteba43de42015-08-29 20:33:16 +0200202 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200203 case ProcedureCode_id_ErrorIndication: /* 8.6 */
Harald Welte3af1db82015-09-11 17:03:16 +0200204 rc = hnbgw_rx_err_ind(hnb, &imsg->value);
205 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200206 case ProcedureCode_id_TNLUpdate: /* 8.9 */
207 case ProcedureCode_id_HNBConfigTransfer: /* 8.10 */
208 case ProcedureCode_id_RelocationComplete: /* 8.11 */
209 case ProcedureCode_id_U_RNTIQuery: /* 8.12 */
210 case ProcedureCode_id_privateMessage:
Harald Welte1d2c39d2015-09-11 17:49:37 +0200211 LOGP(DMAIN, LOGL_NOTICE, "Unimplemented HNBAP Procedure %ld\n",
Harald Welte3af1db82015-09-11 17:03:16 +0200212 imsg->procedureCode);
Harald Welteba43de42015-08-29 20:33:16 +0200213 break;
214 default:
Harald Welte1d2c39d2015-09-11 17:49:37 +0200215 LOGP(DMAIN, LOGL_NOTICE, "Unknown HNBAP Procedure %ld\n",
Harald Welte3af1db82015-09-11 17:03:16 +0200216 imsg->procedureCode);
Harald Welteba43de42015-08-29 20:33:16 +0200217 break;
218 }
219}
220
Harald Welte27f9c4a2015-08-30 22:47:18 +0200221static int hnbgw_rx_successful_outcome_msg(struct hnb_context *hnb, SuccessfulOutcome_t *msg)
Harald Welteba43de42015-08-29 20:33:16 +0200222{
223
224}
225
Harald Welte27f9c4a2015-08-30 22:47:18 +0200226static int hnbgw_rx_unsuccessful_outcome_msg(struct hnb_context *hnb, UnsuccessfulOutcome_t *msg)
Harald Welteba43de42015-08-29 20:33:16 +0200227{
228
229}
230
231
Harald Welte27f9c4a2015-08-30 22:47:18 +0200232static int _hnbgw_hnbap_rx(struct hnb_context *hnb, HNBAP_PDU_t *pdu)
Harald Welteba43de42015-08-29 20:33:16 +0200233{
Harald Welteb3dae302015-08-30 12:20:09 +0200234 int rc;
235
Harald Welteba43de42015-08-29 20:33:16 +0200236 /* it's a bit odd that we can't dispatch on procedure code, but
237 * that's not possible */
Harald Welte27f9c4a2015-08-30 22:47:18 +0200238 switch (pdu->present) {
239 case HNBAP_PDU_PR_initiatingMessage:
240 rc = hnbgw_rx_initiating_msg(hnb, &pdu->choice.initiatingMessage);
Harald Welteba43de42015-08-29 20:33:16 +0200241 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200242 case HNBAP_PDU_PR_successfulOutcome:
243 rc = hnbgw_rx_successful_outcome_msg(hnb, &pdu->choice.successfulOutcome);
Harald Welteba43de42015-08-29 20:33:16 +0200244 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200245 case HNBAP_PDU_PR_unsuccessfulOutcome:
246 rc = hnbgw_rx_unsuccessful_outcome_msg(hnb, &pdu->choice.unsuccessfulOutcome);
Harald Welteba43de42015-08-29 20:33:16 +0200247 break;
248 default:
Harald Welte3af1db82015-09-11 17:03:16 +0200249 LOGP(DMAIN, LOGL_NOTICE, "Unknown HNBAP Presence %u\n",
250 pdu->present);
Harald Welteba43de42015-08-29 20:33:16 +0200251 return -1;
252 }
253}
254
Harald Weltea2e6a7a2015-08-29 21:47:39 +0200255int hnbgw_hnbap_rx(struct hnb_context *hnb, struct msgb *msg)
Harald Welteba43de42015-08-29 20:33:16 +0200256{
Harald Welte27f9c4a2015-08-30 22:47:18 +0200257 HNBAP_PDU_t _pdu, *pdu = &_pdu;
258 asn_dec_rval_t dec_ret;
Harald Welteee77cff2015-08-30 16:57:53 +0200259 int rc;
260
261 /* decode and handle to _hnbgw_hnbap_rx() */
262
Harald Welte2204f9d2015-09-07 21:10:50 +0200263 memset(pdu, 0, sizeof(*pdu));
Harald Welte27f9c4a2015-08-30 22:47:18 +0200264 dec_ret = aper_decode(NULL, &asn_DEF_HNBAP_PDU, (void **) &pdu,
265 msg->data, msgb_length(msg), 0, 0);
266 if (dec_ret.code != RC_OK) {
267 LOGP(DMAIN, LOGL_ERROR, "Error in ASN.1 decode\n");
Harald Welteee77cff2015-08-30 16:57:53 +0200268 return rc;
269 }
270
271 rc = _hnbgw_hnbap_rx(hnb, pdu);
Harald Welteee77cff2015-08-30 16:57:53 +0200272
273 return rc;
Harald Welteba43de42015-08-29 20:33:16 +0200274}
275
276
277int hnbgw_hnbap_init(void)
278{
279
280}