blob: a26df03753f69b4eefcaf70f42c890bcae815e22 [file] [log] [blame]
Harald Welte3cfa4502015-12-25 15:33:41 +01001/* RANAP interface for a core-network node */
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 <unistd.h>
22#include <errno.h>
23#include <string.h>
24
25#include <osmocom/core/utils.h>
26#include <osmocom/core/logging.h>
27
28#include "ranap_common.h"
29#include "ranap_ies_defs.h"
30
31#include "cn_ranap_common.h"
32
33static int cn_ranap_rx_initiating_msg_co(void *ctx, RANAP_InitiatingMessage_t *imsg,
34 ranap_message *message)
35{
36 int rc = 0;
37
38 message->procedureCode = imsg->procedureCode;
39 message->criticality = imsg->criticality;
40
41 DEBUGP(DRANAP, "Rx CO IM (%s)\n",
42 get_value_string(ranap_procedure_code_vals, imsg->procedureCode));
43
44 switch (imsg->procedureCode) {
45 case RANAP_ProcedureCode_id_InitialUE_Message:
46 rc = ranap_decode_initialue_messageies(&message->msg.initialUE_MessageIEs, &imsg->value);
47 break;
48 case RANAP_ProcedureCode_id_DirectTransfer:
49 rc = ranap_decode_directtransferies(&message->msg.directTransferIEs, &imsg->value);
50 break;
51 case RANAP_ProcedureCode_id_RAB_ReleaseRequest:
52 /* RNC requests the release of RAB */
53 rc = ranap_decode_rab_releaserequesties(&message->msg.raB_ReleaseRequestIEs, &imsg->value);
54 break;
55 case RANAP_ProcedureCode_id_Iu_ReleaseRequest:
56 /* RNC requests the release of Iu */
57 rc = ranap_decode_iu_releaserequesties(&message->msg.iu_ReleaseRequestIEs, &imsg->value);
58 break;
59 case RANAP_ProcedureCode_id_ErrorIndication:
60 rc = ranap_decode_errorindicationies(&message->msg.errorIndicationIEs, &imsg->value);
61 break;
62 case RANAP_ProcedureCode_id_RAB_ModifyRequest:
63 rc = ranap_decode_rab_modifyrequesties(&message->msg.raB_ModifyRequestIEs, &imsg->value);
64 break;
65 default:
66 LOGP(DRANAP, LOGL_NOTICE, "Received suspicious RANAP "
67 "Procedure %s (CO, IM) from RNC, ignoring\n",
68 get_value_string(ranap_procedure_code_vals, imsg->procedureCode));
69 rc = -1;
70 break;
71 }
72
73 return rc;
74}
75
76static int cn_ranap_rx_successful_msg_co(void *ctx, RANAP_SuccessfulOutcome_t *imsg,
77 ranap_message *message)
78{
79 int rc = 0;
80
81 message->procedureCode = imsg->procedureCode;
82 message->criticality = imsg->criticality;
83
84 DEBUGP(DRANAP, "Rx CL SO (%s)\n",
85 get_value_string(ranap_procedure_code_vals, imsg->procedureCode));
86
87 switch (imsg->procedureCode) {
88 case RANAP_ProcedureCode_id_RAB_Assignment:
89 /* RAB assignment response */
90 rc = ranap_decode_rab_assignmentresponseies(&message->msg.raB_AssignmentResponseIEs, &imsg->value);
91 break;
92 case RANAP_ProcedureCode_id_SecurityModeControl:
93 /* Security Mode Complete */
94 rc = ranap_decode_securitymodecompleteies(&message->msg.securityModeCompleteIEs, &imsg->value);
95 break;
96 case RANAP_ProcedureCode_id_Iu_Release:
97 /* Iu release Complete; confirmation of CN-initiated release */
98 rc = ranap_decode_iu_releasecompleteies(&message->msg.iu_ReleaseCompleteIEs, &imsg->value);
99 break;
100 default:
101 LOGP(DRANAP, LOGL_NOTICE, "Received suspicious RANAP "
102 "Procedure %s (SO) from RNC, ignoring\n",
103 get_value_string(ranap_procedure_code_vals, imsg->procedureCode));
104 rc = -1;
105 break;
106 }
107
108 return rc;
109}
110
111static int _cn_ranap_rx_co(void *ctx, RANAP_RANAP_PDU_t *pdu, ranap_message *message)
112{
113 int rc = 0;
114
115 switch (pdu->present) {
116 case RANAP_RANAP_PDU_PR_initiatingMessage:
117 rc = cn_ranap_rx_initiating_msg_co(ctx, &pdu->choice.initiatingMessage, message);
118 break;
119 case RANAP_RANAP_PDU_PR_successfulOutcome:
120 rc = cn_ranap_rx_successful_msg_co(ctx, &pdu->choice.successfulOutcome, message);
121 break;
122 case RANAP_RANAP_PDU_PR_unsuccessfulOutcome:
123 LOGP(DRANAP, LOGL_NOTICE, "Received unsupported RANAP "
124 "unsuccessful outcome procedure %s (CO) from RNC, ignoring\n",
125 get_value_string(ranap_procedure_code_vals,
126 pdu->choice.unsuccessfulOutcome.procedureCode));
127 rc = -1;
128 break;
129 default:
130 LOGP(DRANAP, LOGL_NOTICE, "Received suspicious RANAP "
131 "presence %s (CO) from RNC, ignoring\n",
132 get_value_string(ranap_presence_vals, pdu->present));
133 rc = -1;
134 break;
135 }
136
137 return rc;
138}
139
140/* receive a connection-oriented RANAP message and call
141 * cn_ranap_handle_co() with the resulting ranap_message struct */
142int cn_ranap_rx_co(void *ctx, uint8_t *data, size_t len)
143{
144 RANAP_RANAP_PDU_t *pdu = NULL;
145 ranap_message message;
146 asn_dec_rval_t dec_ret;
147 int rc;
148
149 memset(&message, 0, sizeof(message));
150
151 memset(pdu, 0, sizeof(*pdu));
152 dec_ret = aper_decode(NULL,&asn_DEF_RANAP_RANAP_PDU, (void **) &pdu,
153 data, len, 0, 0);
154 if (dec_ret.code != RC_OK) {
155 LOGP(DRANAP, LOGL_ERROR, "Error in RANAP ASN.1 decode\n");
156 return rc;
157 }
158
159 message.direction = pdu->present;
160
161 rc = _cn_ranap_rx_co(ctx, pdu, &message);
162
163 if (rc == 0)
164 cn_ranap_handle_co(ctx, &message);
165
166 ASN_STRUCT_FREE(asn_DEF_RANAP_RANAP_PDU, pdu);
167
168 return rc;
169}
170
171static int cn_ranap_rx_initiating_msg_cl(void *ctx, RANAP_InitiatingMessage_t *imsg,
172 ranap_message *message)
173{
174 int rc;
175
176 message->procedureCode = imsg->procedureCode;
177 message->criticality = imsg->criticality;
178
179 DEBUGP(DRANAP, "Rx CL IM (%s)\n",
180 get_value_string(ranap_procedure_code_vals, imsg->procedureCode));
181
182 switch (imsg->procedureCode) {
183 case RANAP_ProcedureCode_id_Reset:
184 /* Reset request */
185 rc = ranap_decode_reseties(&message->msg.resetIEs, &imsg->value);
186 break;
187 case RANAP_ProcedureCode_id_OverloadControl: /* Overload ind */
188 rc = ranap_decode_overloadies(&message->msg.overloadIEs, &imsg->value);
189 break;
190 case RANAP_ProcedureCode_id_ErrorIndication: /* Error ind */
191 rc = ranap_decode_errorindicationies(&message->msg.errorIndicationIEs, &imsg->value);
192 break;
193 case RANAP_ProcedureCode_id_ResetResource: /* request */
194 rc = ranap_decode_resetresourceies(&message->msg.resetResourceIEs, &imsg->value);
195 break;
196 case RANAP_ProcedureCode_id_InformationTransfer:
197 rc = ranap_decode_informationtransferindicationies(&message->msg.informationTransferIndicationIEs, &imsg->value);
198 break;
199 case RANAP_ProcedureCode_id_DirectInformationTransfer:
200 rc = ranap_decode_directinformationtransferies(&message->msg.directInformationTransferIEs, &imsg->value);
201 break;
202 case RANAP_ProcedureCode_id_UplinkInformationExchange:
203 rc = ranap_decode_uplinkinformationexchangerequesties(&message->msg.uplinkInformationExchangeRequestIEs, &imsg->value);
204 break;
205 default:
206 LOGP(DRANAP, LOGL_NOTICE, "Received suspicious RANAP "
207 "Procedure %s (CL, IM) from RNC, ignoring\n",
208 get_value_string(ranap_procedure_code_vals, imsg->procedureCode));
209 break;
210 }
211}
212
213static int cn_ranap_rx_successful_msg_cl(void *ctx, RANAP_SuccessfulOutcome_t *imsg,
214 ranap_message *message)
215{
216 int rc;
217
218 message->procedureCode = imsg->procedureCode;
219 message->criticality = imsg->criticality;
220
221 DEBUGP(DRANAP, "Rx CL SO (%s)\n",
222 get_value_string(ranap_procedure_code_vals, imsg->procedureCode));
223
224 switch (imsg->procedureCode) {
225 case RANAP_ProcedureCode_id_Reset: /* Reset acknowledge */
226 rc = ranap_decode_resetacknowledgeies(&message->msg.resetAcknowledgeIEs, &imsg->value);
227 break;
228 case RANAP_ProcedureCode_id_ResetResource: /* response */
229 rc = ranap_decode_resetresourceacknowledgeies(&message->msg.resetResourceAcknowledgeIEs, &imsg->value);
230 case RANAP_ProcedureCode_id_InformationTransfer:
231 rc = ranap_decode_resetresourceacknowledgeies(&message->msg.resetResourceAcknowledgeIEs, &imsg->value);
232 break;
233 case RANAP_ProcedureCode_id_DirectInformationTransfer:
234 rc = ranap_decode_informationtransferconfirmationies(&message->msg.informationTransferConfirmationIEs, &imsg->value);
235 break;
236 case RANAP_ProcedureCode_id_UplinkInformationExchange:
237 rc = ranap_decode_uplinkinformationexchangeresponseies(&message->msg.uplinkInformationExchangeResponseIEs, &imsg->value);
238 break;
239 default:
240 LOGP(DRANAP, LOGL_NOTICE, "Received suspicious RANAP "
241 "Procedure %s (CL, SO) from RNC, ignoring\n",
242 get_value_string(ranap_procedure_code_vals, imsg->procedureCode));
243 break;
244 }
245}
246
247static int _cn_ranap_rx_cl(void *ctx, RANAP_RANAP_PDU_t *pdu, ranap_message *message)
248{
249 int rc;
250
251 switch (pdu->present) {
252 case RANAP_RANAP_PDU_PR_initiatingMessage:
253 rc = cn_ranap_rx_initiating_msg_cl(ctx, &pdu->choice.initiatingMessage,
254 message);
255 break;
256 case RANAP_RANAP_PDU_PR_successfulOutcome:
257 rc = cn_ranap_rx_successful_msg_cl(ctx, &pdu->choice.successfulOutcome,
258 message);
259 break;
260 case RANAP_RANAP_PDU_PR_unsuccessfulOutcome:
261 LOGP(DRANAP, LOGL_NOTICE, "Received unsupported RANAP "
262 "unsuccessful outcome procedure %s (CL) from RNC, ignoring\n",
263 get_value_string(ranap_procedure_code_vals,
264 pdu->choice.unsuccessfulOutcome.procedureCode));
265 break;
266 default:
267 LOGP(DRANAP, LOGL_NOTICE, "Received suspicious RANAP "
268 "presence %s (CL) from RNC, ignoring\n",
269 get_value_string(ranap_presence_vals, pdu->present));
270 break;
271 }
272}
273
274/* receive a connection-oriented RANAP message and call
275 * cn_ranap_handle_co() with the resulting ranap_message struct */
276int cn_ranap_rx_cl(void *ctx, uint8_t *data, size_t len)
277{
278 RANAP_RANAP_PDU_t *pdu = NULL;
279 ranap_message message;
280 asn_dec_rval_t dec_ret;
281 int rc;
282
283 memset(&message, 0, sizeof(message));
284
285 memset(pdu, 0, sizeof(*pdu));
286 dec_ret = aper_decode(NULL,&asn_DEF_RANAP_RANAP_PDU, (void **) &pdu,
287 data, len, 0, 0);
288 if (dec_ret.code != RC_OK) {
289 LOGP(DRANAP, LOGL_ERROR, "Error in RANAP ASN.1 decode\n");
290 return rc;
291 }
292
293 message.direction = pdu->present;
294
295 rc = _cn_ranap_rx_cl(ctx, pdu, &message);
296
297 if (rc == 0)
298 cn_ranap_handle_cl(ctx, &message);
299
300 ASN_STRUCT_FREE(asn_DEF_RANAP_RANAP_PDU, pdu);
301
302 return rc;
303}