blob: ae0d16c83dffad4046a3322c6cde010edb8ea9b8 [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 Welteffa7c0a2015-12-23 00:03:41 +010024#include <osmocom/netif/stream.h>
Harald Welteba43de42015-08-29 20:33:16 +020025
Harald Welteb3dae302015-08-30 12:20:09 +020026#include <unistd.h>
Harald Welteee77cff2015-08-30 16:57:53 +020027#include <errno.h>
Harald Welteb3dae302015-08-30 12:20:09 +020028#include <string.h>
29
Harald Welte30afef32015-08-30 12:28:29 +020030#include "asn1helpers.h"
Neels Hofmeyr83457922016-08-26 23:56:44 +020031#include <osmocom/hnbap/hnbap_common.h>
Neels Hofmeyr96979af2016-01-05 15:19:44 +010032#include <osmocom/ranap/iu_helpers.h>
Harald Welte30afef32015-08-30 12:28:29 +020033
Neels Hofmeyrdf63de22016-08-18 13:13:55 +020034#include <osmocom/iuh/hnbgw.h>
Neels Hofmeyr83457922016-08-26 23:56:44 +020035#include <osmocom/hnbap/hnbap_ies_defs.h>
Harald Welteba43de42015-08-29 20:33:16 +020036
Harald Welteee77cff2015-08-30 16:57:53 +020037#define IU_MSG_NUM_IES 32
38#define IU_MSG_NUM_EXT_IES 32
39
Harald Welte27f9c4a2015-08-30 22:47:18 +020040static int hnbgw_hnbap_tx(struct hnb_context *ctx, struct msgb *msg)
Harald Welteee77cff2015-08-30 16:57:53 +020041{
Harald Welte7b54e322015-09-07 22:41:45 +020042 if (!msg)
43 return -EINVAL;
44
Harald Welteffa7c0a2015-12-23 00:03:41 +010045 msgb_sctp_ppid(msg) = IUH_PPI_HNBAP;
Daniel Willmann6480cad2016-01-06 18:06:26 +010046 osmo_stream_srv_send(ctx->conn, msg);
47
48 return 0;
Harald Welte27f9c4a2015-08-30 22:47:18 +020049}
Harald Welteee77cff2015-08-30 16:57:53 +020050
Harald Welte27f9c4a2015-08-30 22:47:18 +020051static int hnbgw_tx_hnb_register_acc(struct hnb_context *ctx)
52{
53 HNBRegisterAccept_t accept_out;
54 struct msgb *msg;
55 int rc;
56
57 /* Single required response IE: RNC-ID */
58 HNBRegisterAcceptIEs_t accept = {
59 .rnc_id = ctx->gw->config.rnc_id
60 };
61
62 /* encode the Information Elements */
Harald Welte2204f9d2015-09-07 21:10:50 +020063 memset(&accept_out, 0, sizeof(accept_out));
Harald Welte27f9c4a2015-08-30 22:47:18 +020064 rc = hnbap_encode_hnbregisteraccepties(&accept_out, &accept);
65 if (rc < 0) {
66 return rc;
Harald Welteee77cff2015-08-30 16:57:53 +020067 }
68
Harald Welte27f9c4a2015-08-30 22:47:18 +020069 /* generate a successfull outcome PDU */
70 msg = hnbap_generate_successful_outcome(ProcedureCode_id_HNBRegister,
71 Criticality_reject,
72 &asn_DEF_HNBRegisterAccept,
73 &accept_out);
Harald Welteee77cff2015-08-30 16:57:53 +020074
Daniel Willmann59d17d82015-12-17 17:56:56 +010075 ASN_STRUCT_FREE_CONTENTS_ONLY(asn_DEF_HNBRegisterAccept, &accept_out);
76
Harald Welte27f9c4a2015-08-30 22:47:18 +020077 return hnbgw_hnbap_tx(ctx, msg);
Harald Welteee77cff2015-08-30 16:57:53 +020078}
79
80
Harald Welte27f9c4a2015-08-30 22:47:18 +020081static int hnbgw_tx_ue_register_acc(struct ue_context *ue)
Harald Welteba43de42015-08-29 20:33:16 +020082{
Harald Welte27f9c4a2015-08-30 22:47:18 +020083 UERegisterAccept_t accept_out;
84 UERegisterAcceptIEs_t accept;
85 struct msgb *msg;
Harald Welte10dfc5a2015-09-11 01:34:45 +020086 uint8_t encoded_imsi[10];
Daniel Willmann92247312015-12-09 19:04:33 +010087 uint32_t ctx_id;
Harald Welte10dfc5a2015-09-11 01:34:45 +020088 size_t encoded_imsi_len;
Harald Welte27f9c4a2015-08-30 22:47:18 +020089 int rc;
Harald Welteba43de42015-08-29 20:33:16 +020090
Harald Welte056984f2016-01-03 16:31:31 +010091 encoded_imsi_len = ranap_imsi_encode(encoded_imsi,
Harald Welte1d2c39d2015-09-11 17:49:37 +020092 sizeof(encoded_imsi), ue->imsi);
Harald Welte10dfc5a2015-09-11 01:34:45 +020093
Harald Welte2204f9d2015-09-07 21:10:50 +020094 memset(&accept, 0, sizeof(accept));
Harald Welte10dfc5a2015-09-11 01:34:45 +020095 accept.uE_Identity.present = UE_Identity_PR_iMSI;
Harald Welte1d2c39d2015-09-11 17:49:37 +020096 OCTET_STRING_fromBuf(&accept.uE_Identity.choice.iMSI,
97 (const char *)encoded_imsi, encoded_imsi_len);
Daniel Willmann92247312015-12-09 19:04:33 +010098 asn1_u24_to_bitstring(&accept.context_ID, &ctx_id, ue->context_id);
Harald Welteba43de42015-08-29 20:33:16 +020099
Harald Welte2204f9d2015-09-07 21:10:50 +0200100 memset(&accept_out, 0, sizeof(accept_out));
Harald Welte27f9c4a2015-08-30 22:47:18 +0200101 rc = hnbap_encode_ueregisteraccepties(&accept_out, &accept);
102 if (rc < 0) {
103 return rc;
Harald Welteba43de42015-08-29 20:33:16 +0200104 }
Harald Welte27f9c4a2015-08-30 22:47:18 +0200105
106 msg = hnbap_generate_successful_outcome(ProcedureCode_id_UERegister,
107 Criticality_reject,
108 &asn_DEF_UERegisterAccept,
109 &accept_out);
Daniel Willmann541e4292015-12-22 16:25:29 +0100110
111 ASN_STRUCT_FREE_CONTENTS_ONLY(asn_DEF_OCTET_STRING, &accept.uE_Identity.choice.iMSI);
112 ASN_STRUCT_FREE_CONTENTS_ONLY(asn_DEF_UERegisterAccept, &accept_out);
113
Harald Welte27f9c4a2015-08-30 22:47:18 +0200114 return hnbgw_hnbap_tx(ue->hnb, msg);
Harald Welteba43de42015-08-29 20:33:16 +0200115}
116
Neels Hofmeyr1a0bb512016-04-25 14:55:35 +0200117static int hnbgw_tx_ue_register_rej_tmsi(struct hnb_context *hnb, UE_Identity_t *ue_id)
118{
119 UERegisterReject_t reject_out;
120 UERegisterRejectIEs_t reject;
121 struct msgb *msg;
122 int rc;
123
124 memset(&reject, 0, sizeof(reject));
125 reject.uE_Identity.present = ue_id->present;
126
Neels Hofmeyrc94ed092016-09-26 01:27:55 +0200127 /* Copy the identity over to the reject message */
128 switch (ue_id->present) {
129 case UE_Identity_PR_tMSILAI:
130 LOGP(DHNBAP, LOGL_DEBUG, "REJ UE_Id tMSI %d %s\n",
131 ue_id->choice.tMSILAI.tMSI.size,
132 osmo_hexdump(ue_id->choice.tMSILAI.tMSI.buf,
133 ue_id->choice.tMSILAI.tMSI.size));
134
135 LOGP(DHNBAP, LOGL_DEBUG, "REJ UE_Id pLMNID %d %s\n",
136 ue_id->choice.tMSILAI.lAI.pLMNID.size,
137 osmo_hexdump(ue_id->choice.tMSILAI.lAI.pLMNID.buf,
138 ue_id->choice.tMSILAI.lAI.pLMNID.size));
139
140 LOGP(DHNBAP, LOGL_DEBUG, "REJ UE_Id lAC %d %s\n",
141 ue_id->choice.tMSILAI.lAI.lAC.size,
142 osmo_hexdump(ue_id->choice.tMSILAI.lAI.lAC.buf,
143 ue_id->choice.tMSILAI.lAI.lAC.size));
144
145 BIT_STRING_fromBuf(&reject.uE_Identity.choice.tMSILAI.tMSI,
146 ue_id->choice.tMSILAI.tMSI.buf,
147 ue_id->choice.tMSILAI.tMSI.size * 8
148 - ue_id->choice.tMSILAI.tMSI.bits_unused);
149 OCTET_STRING_fromBuf(&reject.uE_Identity.choice.tMSILAI.lAI.pLMNID,
150 ue_id->choice.tMSILAI.lAI.pLMNID.buf,
151 ue_id->choice.tMSILAI.lAI.pLMNID.size);
152 OCTET_STRING_fromBuf(&reject.uE_Identity.choice.tMSILAI.lAI.lAC,
153 ue_id->choice.tMSILAI.lAI.lAC.buf,
154 ue_id->choice.tMSILAI.lAI.lAC.size);
155 break;
156
157 case UE_Identity_PR_pTMSIRAI:
158 LOGP(DHNBAP, LOGL_DEBUG, "REJ UE_Id pTMSI %d %s\n",
159 ue_id->choice.pTMSIRAI.pTMSI.size,
160 osmo_hexdump(ue_id->choice.pTMSIRAI.pTMSI.buf,
161 ue_id->choice.pTMSIRAI.pTMSI.size));
162
163 LOGP(DHNBAP, LOGL_DEBUG, "REJ UE_Id pLMNID %d %s\n",
164 ue_id->choice.pTMSIRAI.rAI.lAI.pLMNID.size,
165 osmo_hexdump(ue_id->choice.pTMSIRAI.rAI.lAI.pLMNID.buf,
166 ue_id->choice.pTMSIRAI.rAI.lAI.pLMNID.size));
167
168 LOGP(DHNBAP, LOGL_DEBUG, "REJ UE_Id lAC %d %s\n",
169 ue_id->choice.pTMSIRAI.rAI.lAI.lAC.size,
170 osmo_hexdump(ue_id->choice.pTMSIRAI.rAI.lAI.lAC.buf,
171 ue_id->choice.pTMSIRAI.rAI.lAI.lAC.size));
172
173 LOGP(DHNBAP, LOGL_DEBUG, "REJ UE_Id rAC %d %s\n",
174 ue_id->choice.pTMSIRAI.rAI.rAC.size,
175 osmo_hexdump(ue_id->choice.pTMSIRAI.rAI.rAC.buf,
176 ue_id->choice.pTMSIRAI.rAI.rAC.size));
177
178 BIT_STRING_fromBuf(&reject.uE_Identity.choice.pTMSIRAI.pTMSI,
179 ue_id->choice.pTMSIRAI.pTMSI.buf,
180 ue_id->choice.pTMSIRAI.pTMSI.size * 8
181 - ue_id->choice.pTMSIRAI.pTMSI.bits_unused);
182 OCTET_STRING_fromBuf(&reject.uE_Identity.choice.pTMSIRAI.rAI.lAI.pLMNID,
183 ue_id->choice.pTMSIRAI.rAI.lAI.pLMNID.buf,
184 ue_id->choice.pTMSIRAI.rAI.lAI.pLMNID.size);
185 OCTET_STRING_fromBuf(&reject.uE_Identity.choice.pTMSIRAI.rAI.lAI.lAC,
186 ue_id->choice.pTMSIRAI.rAI.lAI.lAC.buf,
187 ue_id->choice.pTMSIRAI.rAI.lAI.lAC.size);
188 OCTET_STRING_fromBuf(&reject.uE_Identity.choice.pTMSIRAI.rAI.rAC,
189 ue_id->choice.pTMSIRAI.rAI.rAC.buf,
190 ue_id->choice.pTMSIRAI.rAI.rAC.size);
191 break;
192
193 default:
194 LOGP(DHNBAP, LOGL_ERROR, "Cannot compose UE Register Reject:"
195 " unsupported UE ID (present=%d)\n", ue_id->present);
Neels Hofmeyr1a0bb512016-04-25 14:55:35 +0200196 return -1;
197 }
198
Neels Hofmeyrc94ed092016-09-26 01:27:55 +0200199 LOGP(DHNBAP, LOGL_ERROR, "Rejecting UE Register Request:"
200 " TMSI identity registration is switched off\n");
Neels Hofmeyr1a0bb512016-04-25 14:55:35 +0200201
202 reject.cause.present = Cause_PR_radioNetwork;
203 reject.cause.choice.radioNetwork = CauseRadioNetwork_invalid_UE_identity;
204
205 memset(&reject_out, 0, sizeof(reject_out));
206 rc = hnbap_encode_ueregisterrejecties(&reject_out, &reject);
Neels Hofmeyrc94ed092016-09-26 01:27:55 +0200207 if (rc < 0)
Neels Hofmeyr1a0bb512016-04-25 14:55:35 +0200208 return rc;
Neels Hofmeyr1a0bb512016-04-25 14:55:35 +0200209
210 msg = hnbap_generate_unsuccessful_outcome(ProcedureCode_id_UERegister,
211 Criticality_reject,
212 &asn_DEF_UERegisterReject,
213 &reject_out);
214
Neels Hofmeyrc94ed092016-09-26 01:27:55 +0200215 /* Free copied identity IEs */
216 switch (ue_id->present) {
217 case UE_Identity_PR_tMSILAI:
218 ASN_STRUCT_FREE_CONTENTS_ONLY(asn_DEF_BIT_STRING,
219 &reject.uE_Identity.choice.tMSILAI.tMSI);
220 ASN_STRUCT_FREE_CONTENTS_ONLY(asn_DEF_OCTET_STRING,
221 &reject.uE_Identity.choice.tMSILAI.lAI.pLMNID);
222 ASN_STRUCT_FREE_CONTENTS_ONLY(asn_DEF_OCTET_STRING,
223 &reject.uE_Identity.choice.tMSILAI.lAI.lAC);
224 break;
225
226 case UE_Identity_PR_pTMSIRAI:
227 ASN_STRUCT_FREE_CONTENTS_ONLY(asn_DEF_BIT_STRING,
228 &reject.uE_Identity.choice.pTMSIRAI.pTMSI);
229 ASN_STRUCT_FREE_CONTENTS_ONLY(asn_DEF_OCTET_STRING,
230 &reject.uE_Identity.choice.pTMSIRAI.rAI.lAI.pLMNID);
231 ASN_STRUCT_FREE_CONTENTS_ONLY(asn_DEF_OCTET_STRING,
232 &reject.uE_Identity.choice.pTMSIRAI.rAI.lAI.lAC);
233 ASN_STRUCT_FREE_CONTENTS_ONLY(asn_DEF_OCTET_STRING,
234 &reject.uE_Identity.choice.pTMSIRAI.rAI.rAC);
235 break;
236
237 default:
238 /* should never happen after above switch() */
239 break;
240 }
241
Neels Hofmeyr1a0bb512016-04-25 14:55:35 +0200242 ASN_STRUCT_FREE_CONTENTS_ONLY(asn_DEF_UERegisterReject, &reject_out);
243
244 return hnbgw_hnbap_tx(hnb, msg);
245}
246
Daniel Willmannb433b972016-01-06 18:08:21 +0100247static int hnbgw_rx_hnb_deregister(struct hnb_context *ctx, ANY_t *in)
248{
249 HNBDe_RegisterIEs_t ies;
250 int rc;
251
252 rc = hnbap_decode_hnbde_registeries(&ies, in);
253 if (rc < 0)
254 return rc;
255
Neels Hofmeyr9246cc92016-04-25 14:47:26 +0200256 DEBUGP(DHNBAP, "HNB-DE-REGISTER cause=%ld\n",
Daniel Willmannb433b972016-01-06 18:08:21 +0100257 ies.cause);
258
Daniel Willmannd10002c2016-01-07 12:27:41 +0100259 hnbap_free_hnbde_registeries(&ies);
Daniel Willmannb433b972016-01-06 18:08:21 +0100260 hnb_context_release(ctx);
261
262 return 0;
263}
264
Harald Welte27f9c4a2015-08-30 22:47:18 +0200265static int hnbgw_rx_hnb_register_req(struct hnb_context *ctx, ANY_t *in)
Harald Welteba43de42015-08-29 20:33:16 +0200266{
Harald Welte27f9c4a2015-08-30 22:47:18 +0200267 HNBRegisterRequestIEs_t ies;
268 int rc;
Harald Welteba43de42015-08-29 20:33:16 +0200269
Harald Welte27f9c4a2015-08-30 22:47:18 +0200270 rc = hnbap_decode_hnbregisterrequesties(&ies, in);
271 if (rc < 0)
272 return rc;
Harald Welteba43de42015-08-29 20:33:16 +0200273
Harald Weltea2e6a7a2015-08-29 21:47:39 +0200274 /* copy all identity parameters from the message to ctx */
Harald Welte27f9c4a2015-08-30 22:47:18 +0200275 asn1_strncpy(ctx->identity_info, &ies.hnB_Identity.hNB_Identity_Info,
276 sizeof(ctx->identity_info));
277 ctx->id.lac = asn1str_to_u16(&ies.lac);
278 ctx->id.sac = asn1str_to_u16(&ies.sac);
279 ctx->id.rac = asn1str_to_u8(&ies.rac);
Daniel Willmannd6a45b42015-12-08 13:55:17 +0100280 ctx->id.cid = asn1bitstr_to_u28(&ies.cellIdentity);
Harald Welteb3dae302015-08-30 12:20:09 +0200281 //ctx->id.mcc FIXME
282 //ctx->id.mnc FIXME
Harald Weltea2e6a7a2015-08-29 21:47:39 +0200283
Daniel Willmannbded9842015-12-17 11:51:17 +0100284 DEBUGP(DHNBAP, "HNB-REGISTER-REQ from %s\n", ctx->identity_info);
Harald Welteee77cff2015-08-30 16:57:53 +0200285
Harald Welte27f9c4a2015-08-30 22:47:18 +0200286 /* Send HNBRegisterAccept */
Daniel Willmannd10002c2016-01-07 12:27:41 +0100287 rc = hnbgw_tx_hnb_register_acc(ctx);
288 hnbap_free_hnbregisterrequesties(&ies);
289 return rc;
Harald Welteba43de42015-08-29 20:33:16 +0200290}
291
Harald Welte27f9c4a2015-08-30 22:47:18 +0200292static int hnbgw_rx_ue_register_req(struct hnb_context *ctx, ANY_t *in)
Harald Welteba43de42015-08-29 20:33:16 +0200293{
Harald Welte27f9c4a2015-08-30 22:47:18 +0200294 UERegisterRequestIEs_t ies;
295 struct ue_context *ue;
Harald Welte10dfc5a2015-09-11 01:34:45 +0200296 char imsi[16];
Harald Welte27f9c4a2015-08-30 22:47:18 +0200297 int rc;
Harald Welteba43de42015-08-29 20:33:16 +0200298
Harald Welte27f9c4a2015-08-30 22:47:18 +0200299 rc = hnbap_decode_ueregisterrequesties(&ies, in);
300 if (rc < 0)
301 return rc;
Harald Welteba43de42015-08-29 20:33:16 +0200302
Harald Welte10dfc5a2015-09-11 01:34:45 +0200303 switch (ies.uE_Identity.present) {
304 case UE_Identity_PR_iMSI:
Harald Welte056984f2016-01-03 16:31:31 +0100305 ranap_bcd_decode(imsi, sizeof(imsi), ies.uE_Identity.choice.iMSI.buf,
Harald Welte10dfc5a2015-09-11 01:34:45 +0200306 ies.uE_Identity.choice.iMSI.size);
307 break;
308 case UE_Identity_PR_iMSIDS41:
Harald Welte056984f2016-01-03 16:31:31 +0100309 ranap_bcd_decode(imsi, sizeof(imsi), ies.uE_Identity.choice.iMSIDS41.buf,
Harald Welte10dfc5a2015-09-11 01:34:45 +0200310 ies.uE_Identity.choice.iMSIDS41.size);
311 break;
312 case UE_Identity_PR_iMSIESN:
Harald Welte056984f2016-01-03 16:31:31 +0100313 ranap_bcd_decode(imsi, sizeof(imsi), ies.uE_Identity.choice.iMSIESN.iMSIDS41.buf,
Harald Welte10dfc5a2015-09-11 01:34:45 +0200314 ies.uE_Identity.choice.iMSIESN.iMSIDS41.size);
315 break;
316 default:
Neels Hofmeyr9246cc92016-04-25 14:47:26 +0200317 LOGP(DHNBAP, LOGL_NOTICE, "UE-REGISTER-REQ without IMSI\n");
318 /* TODO: this is probably a TMSI registration. Store TMSIs
319 * and look them up to accept UE Registration. */
Neels Hofmeyr1a0bb512016-04-25 14:55:35 +0200320 rc = hnbgw_tx_ue_register_rej_tmsi(ctx, &ies.uE_Identity);
Neels Hofmeyrb248c8c2016-04-25 14:51:27 +0200321 hnbap_free_ueregisterrequesties(&ies);
Neels Hofmeyr1a0bb512016-04-25 14:55:35 +0200322 return rc;
Harald Welte10dfc5a2015-09-11 01:34:45 +0200323 }
Harald Welteb534e5c2015-09-11 00:15:16 +0200324
Harald Welte706213a2015-12-25 10:33:09 +0100325 DEBUGP(DHNBAP, "UE-REGISTER-REQ ID_type=%d imsi=%s cause=%ld\n",
Harald Welte10dfc5a2015-09-11 01:34:45 +0200326 ies.uE_Identity.present, imsi, ies.registration_Cause);
327
Harald Weltec4338de2015-12-24 00:40:52 +0100328 ue = ue_context_by_imsi(ctx->gw, imsi);
Harald Welte10dfc5a2015-09-11 01:34:45 +0200329 if (!ue)
330 ue = ue_context_alloc(ctx, imsi);
Harald Welte27f9c4a2015-08-30 22:47:18 +0200331
Daniel Willmannd10002c2016-01-07 12:27:41 +0100332 hnbap_free_ueregisterrequesties(&ies);
Harald Welte27f9c4a2015-08-30 22:47:18 +0200333 /* Send UERegisterAccept */
334 return hnbgw_tx_ue_register_acc(ue);
Harald Welteba43de42015-08-29 20:33:16 +0200335}
336
Daniel Willmannefceb182015-12-15 20:30:12 +0100337static int hnbgw_rx_ue_deregister(struct hnb_context *ctx, ANY_t *in)
338{
339 UEDe_RegisterIEs_t ies;
340 struct ue_context *ue;
341 int rc;
342 uint32_t ctxid;
343
344 rc = hnbap_decode_uede_registeries(&ies, in);
345 if (rc < 0)
346 return rc;
347
348 ctxid = asn1bitstr_to_u24(&ies.context_ID);
349
Harald Welte706213a2015-12-25 10:33:09 +0100350 DEBUGP(DHNBAP, "UE-DE-REGISTER context=%ld cause=%s\n",
Harald Welte2963ee22015-12-25 10:32:37 +0100351 ctxid, hnbap_cause_str(&ies.cause));
Daniel Willmannefceb182015-12-15 20:30:12 +0100352
Harald Weltec4338de2015-12-24 00:40:52 +0100353 ue = ue_context_by_id(ctx->gw, ctxid);
Daniel Willmannefceb182015-12-15 20:30:12 +0100354 if (ue)
355 ue_context_free(ue);
356
Daniel Willmannd10002c2016-01-07 12:27:41 +0100357 hnbap_free_uede_registeries(&ies);
Daniel Willmannefceb182015-12-15 20:30:12 +0100358 return 0;
359}
360
Harald Welte3af1db82015-09-11 17:03:16 +0200361static int hnbgw_rx_err_ind(struct hnb_context *hnb, ANY_t *in)
362{
363 ErrorIndicationIEs_t ies;
364 int rc;
365
Harald Welte1d2c39d2015-09-11 17:49:37 +0200366 rc = hnbap_decode_errorindicationies(&ies, in);
Harald Welte3af1db82015-09-11 17:03:16 +0200367 if (rc < 0)
368 return rc;
369
Daniel Willmannbded9842015-12-17 11:51:17 +0100370 LOGP(DHNBAP, LOGL_NOTICE, "HNBAP ERROR.ind, cause: %s\n",
Harald Welte3af1db82015-09-11 17:03:16 +0200371 hnbap_cause_str(&ies.cause));
372
Daniel Willmannd10002c2016-01-07 12:27:41 +0100373 hnbap_free_errorindicationies(&ies);
Harald Welte3af1db82015-09-11 17:03:16 +0200374 return 0;
375}
376
Harald Welte27f9c4a2015-08-30 22:47:18 +0200377static int hnbgw_rx_initiating_msg(struct hnb_context *hnb, InitiatingMessage_t *imsg)
Harald Welteba43de42015-08-29 20:33:16 +0200378{
379 int rc;
380
Harald Welteb3dae302015-08-30 12:20:09 +0200381 switch (imsg->procedureCode) {
Harald Welte27f9c4a2015-08-30 22:47:18 +0200382 case ProcedureCode_id_HNBRegister: /* 8.2 */
383 rc = hnbgw_rx_hnb_register_req(hnb, &imsg->value);
Harald Welteba43de42015-08-29 20:33:16 +0200384 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200385 case ProcedureCode_id_HNBDe_Register: /* 8.3 */
Daniel Willmannb433b972016-01-06 18:08:21 +0100386 rc = hnbgw_rx_hnb_deregister(hnb, &imsg->value);
Harald Welteba43de42015-08-29 20:33:16 +0200387 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200388 case ProcedureCode_id_UERegister: /* 8.4 */
389 rc = hnbgw_rx_ue_register_req(hnb, &imsg->value);
Harald Welteba43de42015-08-29 20:33:16 +0200390 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200391 case ProcedureCode_id_UEDe_Register: /* 8.5 */
Daniel Willmannefceb182015-12-15 20:30:12 +0100392 rc = hnbgw_rx_ue_deregister(hnb, &imsg->value);
Harald Welteba43de42015-08-29 20:33:16 +0200393 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200394 case ProcedureCode_id_ErrorIndication: /* 8.6 */
Harald Welte3af1db82015-09-11 17:03:16 +0200395 rc = hnbgw_rx_err_ind(hnb, &imsg->value);
396 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200397 case ProcedureCode_id_TNLUpdate: /* 8.9 */
398 case ProcedureCode_id_HNBConfigTransfer: /* 8.10 */
399 case ProcedureCode_id_RelocationComplete: /* 8.11 */
400 case ProcedureCode_id_U_RNTIQuery: /* 8.12 */
401 case ProcedureCode_id_privateMessage:
Daniel Willmannbded9842015-12-17 11:51:17 +0100402 LOGP(DHNBAP, LOGL_NOTICE, "Unimplemented HNBAP Procedure %ld\n",
Harald Welte3af1db82015-09-11 17:03:16 +0200403 imsg->procedureCode);
Harald Welteba43de42015-08-29 20:33:16 +0200404 break;
405 default:
Daniel Willmannbded9842015-12-17 11:51:17 +0100406 LOGP(DHNBAP, LOGL_NOTICE, "Unknown HNBAP Procedure %ld\n",
Harald Welte3af1db82015-09-11 17:03:16 +0200407 imsg->procedureCode);
Harald Welteba43de42015-08-29 20:33:16 +0200408 break;
409 }
410}
411
Harald Welte27f9c4a2015-08-30 22:47:18 +0200412static int hnbgw_rx_successful_outcome_msg(struct hnb_context *hnb, SuccessfulOutcome_t *msg)
Harald Welteba43de42015-08-29 20:33:16 +0200413{
414
415}
416
Harald Welte27f9c4a2015-08-30 22:47:18 +0200417static int hnbgw_rx_unsuccessful_outcome_msg(struct hnb_context *hnb, UnsuccessfulOutcome_t *msg)
Harald Welteba43de42015-08-29 20:33:16 +0200418{
419
420}
421
422
Harald Welte27f9c4a2015-08-30 22:47:18 +0200423static int _hnbgw_hnbap_rx(struct hnb_context *hnb, HNBAP_PDU_t *pdu)
Harald Welteba43de42015-08-29 20:33:16 +0200424{
Daniel Willmann5f810f42015-12-17 17:57:51 +0100425 int rc = 0;
Harald Welteb3dae302015-08-30 12:20:09 +0200426
Harald Welteba43de42015-08-29 20:33:16 +0200427 /* it's a bit odd that we can't dispatch on procedure code, but
428 * that's not possible */
Harald Welte27f9c4a2015-08-30 22:47:18 +0200429 switch (pdu->present) {
430 case HNBAP_PDU_PR_initiatingMessage:
431 rc = hnbgw_rx_initiating_msg(hnb, &pdu->choice.initiatingMessage);
Harald Welteba43de42015-08-29 20:33:16 +0200432 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200433 case HNBAP_PDU_PR_successfulOutcome:
434 rc = hnbgw_rx_successful_outcome_msg(hnb, &pdu->choice.successfulOutcome);
Harald Welteba43de42015-08-29 20:33:16 +0200435 break;
Harald Welte27f9c4a2015-08-30 22:47:18 +0200436 case HNBAP_PDU_PR_unsuccessfulOutcome:
437 rc = hnbgw_rx_unsuccessful_outcome_msg(hnb, &pdu->choice.unsuccessfulOutcome);
Harald Welteba43de42015-08-29 20:33:16 +0200438 break;
439 default:
Daniel Willmannbded9842015-12-17 11:51:17 +0100440 LOGP(DHNBAP, LOGL_NOTICE, "Unknown HNBAP Presence %u\n",
Harald Welte3af1db82015-09-11 17:03:16 +0200441 pdu->present);
Daniel Willmann5f810f42015-12-17 17:57:51 +0100442 rc = -1;
Harald Welteba43de42015-08-29 20:33:16 +0200443 }
Daniel Willmann5f810f42015-12-17 17:57:51 +0100444
445 return rc;
Harald Welteba43de42015-08-29 20:33:16 +0200446}
447
Harald Weltea2e6a7a2015-08-29 21:47:39 +0200448int hnbgw_hnbap_rx(struct hnb_context *hnb, struct msgb *msg)
Harald Welteba43de42015-08-29 20:33:16 +0200449{
Harald Welte27f9c4a2015-08-30 22:47:18 +0200450 HNBAP_PDU_t _pdu, *pdu = &_pdu;
451 asn_dec_rval_t dec_ret;
Harald Welteee77cff2015-08-30 16:57:53 +0200452 int rc;
453
454 /* decode and handle to _hnbgw_hnbap_rx() */
455
Harald Welte2204f9d2015-09-07 21:10:50 +0200456 memset(pdu, 0, sizeof(*pdu));
Harald Welte27f9c4a2015-08-30 22:47:18 +0200457 dec_ret = aper_decode(NULL, &asn_DEF_HNBAP_PDU, (void **) &pdu,
458 msg->data, msgb_length(msg), 0, 0);
459 if (dec_ret.code != RC_OK) {
Daniel Willmannbded9842015-12-17 11:51:17 +0100460 LOGP(DHNBAP, LOGL_ERROR, "Error in ASN.1 decode\n");
Harald Welteee77cff2015-08-30 16:57:53 +0200461 return rc;
462 }
463
464 rc = _hnbgw_hnbap_rx(hnb, pdu);
Harald Welteee77cff2015-08-30 16:57:53 +0200465
Daniel Willmann59d17d82015-12-17 17:56:56 +0100466 ASN_STRUCT_FREE_CONTENTS_ONLY(asn_DEF_HNBAP_PDU, pdu);
467
Harald Welteee77cff2015-08-30 16:57:53 +0200468 return rc;
Harald Welteba43de42015-08-29 20:33:16 +0200469}
470
471
472int hnbgw_hnbap_init(void)
473{
474
475}