open5gs/src/smf/gsm-sm.c

348 lines
11 KiB
C
Raw Normal View History

2020-06-17 05:22:28 +00:00
/*
* Copyright (C) 2019,2020 by Sukchan Lee <acetcom@gmail.com>
*
* This file is part of Open5GS.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
#include "sbi-path.h"
#include "nnrf-handler.h"
#include "nsmf-handler.h"
#include "nudm-handler.h"
2020-12-11 19:03:20 +00:00
#include "npcf-handler.h"
2021-01-01 02:07:08 +00:00
#include "namf-handler.h"
2020-06-17 05:22:28 +00:00
#include "gsm-handler.h"
#include "ngap-handler.h"
#include "pfcp-path.h"
void smf_gsm_state_initial(ogs_fsm_t *s, smf_event_t *e)
{
ogs_assert(s);
OGS_FSM_TRAN(s, &smf_gsm_state_operational);
}
void smf_gsm_state_final(ogs_fsm_t *s, smf_event_t *e)
{
}
void smf_gsm_state_operational(ogs_fsm_t *s, smf_event_t *e)
{
int rv;
2020-07-02 05:50:23 +00:00
char *strerror = NULL;
smf_ue_t *smf_ue = NULL;
2020-06-17 05:22:28 +00:00
smf_sess_t *sess = NULL;
ogs_pkbuf_t *pkbuf = NULL;
ogs_nas_5gs_message_t *nas_message = NULL;
ogs_sbi_stream_t *stream = NULL;
2020-06-17 05:22:28 +00:00
ogs_sbi_message_t *sbi_message = NULL;
ogs_assert(s);
ogs_assert(e);
smf_sm_debug(e);
sess = e->sess;
ogs_assert(sess);
switch (e->id) {
case OGS_FSM_ENTRY_SIG:
break;
case OGS_FSM_EXIT_SIG:
break;
case SMF_EVT_SBI_SERVER:
sbi_message = e->sbi.message;
ogs_assert(sbi_message);
stream = e->sbi.data;
ogs_assert(stream);
2020-06-17 05:22:28 +00:00
SWITCH(sbi_message->h.service.name)
CASE(OGS_SBI_SERVICE_NAME_NSMF_PDUSESSION)
SWITCH(sbi_message->h.resource.component[2])
CASE(OGS_SBI_RESOURCE_NAME_MODIFY)
smf_nsmf_handle_update_sm_context(sess, stream, sbi_message);
2020-06-17 05:22:28 +00:00
break;
CASE(OGS_SBI_RESOURCE_NAME_RELEASE)
smf_nsmf_handle_release_sm_context(sess, stream, sbi_message);
2020-06-17 05:22:28 +00:00
break;
DEFAULT
smf_nsmf_handle_create_sm_context(sess, stream, sbi_message);
2020-06-17 05:22:28 +00:00
break;
END
break;
DEFAULT
ogs_error("Invalid API name [%s]", sbi_message->h.service.name);
ogs_sbi_server_send_error(stream,
2020-06-17 05:22:28 +00:00
OGS_SBI_HTTP_STATUS_BAD_REQUEST, sbi_message,
"Invalid API name", sbi_message->h.service.name);
END
break;
case SMF_EVT_SBI_CLIENT:
sbi_message = e->sbi.message;
ogs_assert(sbi_message);
sess = e->sess;
2020-06-17 05:22:28 +00:00
ogs_assert(sess);
smf_ue = sess->smf_ue;
ogs_assert(smf_ue);
2020-06-17 05:22:28 +00:00
SWITCH(sbi_message->h.service.name)
CASE(OGS_SBI_SERVICE_NAME_NUDM_SDM)
2021-01-01 02:07:08 +00:00
stream = e->sbi.data;
ogs_assert(stream);
2020-06-17 05:22:28 +00:00
SWITCH(sbi_message->h.resource.component[1])
CASE(OGS_SBI_RESOURCE_NAME_SM_DATA)
if (sbi_message->res_status != OGS_SBI_HTTP_STATUS_OK) {
2020-12-11 19:03:20 +00:00
strerror = ogs_msprintf("[%s:%d] HTTP response error [%d]",
smf_ue->supi, sess->psi, sbi_message->res_status);
ogs_assert(strerror);
ogs_error("%s", strerror);
ogs_sbi_server_send_error(stream, sbi_message->res_status,
sbi_message, strerror, NULL);
ogs_free(strerror);
2020-06-17 05:22:28 +00:00
break;
}
if (smf_nudm_sdm_handle_get(
sess, stream, sbi_message) != true) {
ogs_sbi_server_send_error(stream,
2020-06-17 05:22:28 +00:00
OGS_SBI_HTTP_STATUS_INTERNAL_SERVER_ERROR,
sbi_message, "HTTP response error", smf_ue->supi);
2020-06-17 05:22:28 +00:00
}
break;
DEFAULT
2020-12-11 19:03:20 +00:00
strerror = ogs_msprintf("[%s:%d] Invalid resource name [%s]",
smf_ue->supi, sess->psi,
2020-06-17 05:22:28 +00:00
sbi_message->h.resource.component[1]);
2020-12-11 19:03:20 +00:00
ogs_assert(strerror);
ogs_error("%s", strerror);
ogs_sbi_server_send_error(stream,
2020-12-11 19:03:20 +00:00
OGS_SBI_HTTP_STATUS_BAD_REQUEST,
sbi_message, strerror, NULL);
ogs_free(strerror);
END
break;
CASE(OGS_SBI_SERVICE_NAME_NPCF_SMPOLICYCONTROL)
2021-01-01 02:07:08 +00:00
stream = e->sbi.data;
ogs_assert(stream);
2020-12-11 19:03:20 +00:00
SWITCH(sbi_message->h.resource.component[0])
CASE(OGS_SBI_RESOURCE_NAME_SM_POLICIES)
if (sbi_message->res_status != OGS_SBI_HTTP_STATUS_CREATED) {
strerror = ogs_msprintf("[%s:%d] HTTP response error [%d]",
smf_ue->supi, sess->psi, sbi_message->res_status);
ogs_assert(strerror);
ogs_error("%s", strerror);
ogs_sbi_server_send_error(stream, sbi_message->res_status,
sbi_message, strerror, NULL);
ogs_free(strerror);
break;
}
smf_npcf_smpolicycontrol_handle_create(
sess, stream, sbi_message);
break;
DEFAULT
strerror = ogs_msprintf("[%s:%d] Invalid resource name [%s]",
smf_ue->supi, sess->psi,
sbi_message->h.resource.component[0]);
ogs_assert(strerror);
ogs_error("%s", strerror);
ogs_sbi_server_send_error(stream,
OGS_SBI_HTTP_STATUS_BAD_REQUEST,
sbi_message, strerror, NULL);
ogs_free(strerror);
2020-06-17 05:22:28 +00:00
END
break;
CASE(OGS_SBI_SERVICE_NAME_NAMF_COMM)
SWITCH(sbi_message->h.resource.component[0])
CASE(OGS_SBI_RESOURCE_NAME_UE_CONTEXTS)
if (sbi_message->res_status != OGS_SBI_HTTP_STATUS_OK &&
sbi_message->res_status != OGS_SBI_HTTP_STATUS_ACCEPTED) {
2020-12-11 19:03:20 +00:00
ogs_error("[%s:%d] HTTP response error [%d]",
smf_ue->supi, sess->psi, sbi_message->res_status);
2020-06-17 05:22:28 +00:00
break;
}
2021-01-01 02:07:08 +00:00
smf_namf_comm_handler_n1_n2_message_transfer(
sess, e->sbi.state, sbi_message);
2020-06-17 05:22:28 +00:00
break;
DEFAULT
2020-12-11 19:03:20 +00:00
ogs_error("[%s:%d] Invalid resource name [%s]",
smf_ue->supi, sess->psi,
2020-06-17 05:22:28 +00:00
sbi_message->h.resource.component[0]);
ogs_assert_if_reached();
END
break;
DEFAULT
2020-12-11 19:03:20 +00:00
ogs_error("[%s:%d] Invalid API name [%s]",
smf_ue->supi, sess->psi, sbi_message->h.service.name);
2020-06-17 05:22:28 +00:00
END
break;
case SMF_EVT_5GSM_MESSAGE:
nas_message = e->nas.message;
ogs_assert(nas_message);
sess = e->sess;
ogs_assert(sess);
stream = e->sbi.data;
ogs_assert(stream);
smf_ue = sess->smf_ue;
ogs_assert(smf_ue);
2020-06-17 05:22:28 +00:00
switch (nas_message->gsm.h.message_type) {
case OGS_NAS_5GS_PDU_SESSION_ESTABLISHMENT_REQUEST:
rv = gsm_handle_pdu_session_establishment_request(sess, stream,
2020-07-31 02:10:20 +00:00
&nas_message->gsm.pdu_session_establishment_request);
2020-06-17 05:22:28 +00:00
if (rv != OGS_OK) {
ogs_error("[%s:%d] Cannot handle NAS message",
smf_ue->supi, sess->psi);
2020-06-17 05:22:28 +00:00
OGS_FSM_TRAN(s, smf_gsm_state_exception);
}
break;
2020-07-02 05:50:23 +00:00
2021-01-01 02:07:08 +00:00
case OGS_NAS_5GS_PDU_SESSION_MODIFICATION_COMPLETE:
smf_sbi_send_response(stream, OGS_SBI_HTTP_STATUS_NO_CONTENT);
break;
2020-07-02 05:50:23 +00:00
case OGS_NAS_5GS_PDU_SESSION_RELEASE_REQUEST:
smf_5gc_pfcp_send_session_deletion_request(
sess, stream, OGS_PFCP_DELETE_TRIGGER_UE_REQUESTED);
2020-07-02 05:50:23 +00:00
break;
case OGS_NAS_5GS_PDU_SESSION_RELEASE_COMPLETE:
smf_sbi_send_response(stream, OGS_SBI_HTTP_STATUS_NO_CONTENT);
2020-07-02 05:50:23 +00:00
2020-07-30 03:26:12 +00:00
/*
* Race condition for PDU session release complete
* - CLIENT : /nsmf-pdusession/v1/sm-contexts/{smContextRef}/modify
* - SERVER : /namf-callback/v1/{supi}/sm-context-status/{psi})
*
* smf_sbi_send_response(stream, OGS_SBI_HTTP_STATUS_NO_CONTENT);
2020-07-30 03:26:12 +00:00
* smf_sbi_send_sm_context_status_notify(sess);
*
* When executed as above,
* NOTIFY transmits first, and Modify's Response transmits later.
*
* Use the Release Timer to send Notify
* later than Modify's Response.
*/
ogs_timer_start(sess->t_release_holding, ogs_time_from_msec(1));
2020-07-02 05:50:23 +00:00
break;
2020-06-17 05:22:28 +00:00
default:
2020-07-02 05:50:23 +00:00
strerror = ogs_msprintf("Unknown message [%d]",
nas_message->gsm.h.message_type);
ogs_assert(strerror);
2020-12-11 19:03:20 +00:00
2020-07-02 05:50:23 +00:00
ogs_error("%s", strerror);
ogs_sbi_server_send_error(stream,
2020-07-02 05:50:23 +00:00
OGS_SBI_HTTP_STATUS_BAD_REQUEST, NULL, strerror, NULL);
ogs_free(strerror);
2020-06-17 05:22:28 +00:00
break;
}
break;
case SMF_EVT_NGAP_MESSAGE:
sess = e->sess;
ogs_assert(sess);
stream = e->sbi.data;
ogs_assert(stream);
smf_ue = sess->smf_ue;
ogs_assert(smf_ue);
2020-06-17 05:22:28 +00:00
pkbuf = e->pkbuf;
ogs_assert(pkbuf);
ogs_assert(e->ngap.type);
switch (e->ngap.type) {
case OpenAPI_n2_sm_info_type_PDU_RES_SETUP_RSP:
rv = ngap_handle_pdu_session_resource_setup_response_transfer(
sess, stream, pkbuf);
2021-01-01 02:07:08 +00:00
if (rv != OGS_OK) {
ogs_error("[%s:%d] Cannot handle NGAP message",
smf_ue->supi, sess->psi);
OGS_FSM_TRAN(s, smf_gsm_state_exception);
}
break;
case OpenAPI_n2_sm_info_type_PDU_RES_MOD_RSP:
rv = ngap_handle_pdu_session_resource_modify_response_transfer(
sess, stream, pkbuf);
2020-06-17 05:22:28 +00:00
if (rv != OGS_OK) {
ogs_error("[%s:%d] Cannot handle NGAP message",
smf_ue->supi, sess->psi);
2020-06-17 05:22:28 +00:00
OGS_FSM_TRAN(s, smf_gsm_state_exception);
}
break;
2020-07-02 05:50:23 +00:00
case OpenAPI_n2_sm_info_type_PDU_RES_REL_RSP:
smf_sbi_send_response(stream, OGS_SBI_HTTP_STATUS_NO_CONTENT);
2020-07-02 05:50:23 +00:00
break;
2020-06-17 05:22:28 +00:00
default:
ogs_error("Unknown message[%d]", e->ngap.type);
}
break;
default:
ogs_error("Unknown event [%s]", smf_event_get_name(e));
2020-06-17 05:22:28 +00:00
break;
}
}
void smf_gsm_state_exception(ogs_fsm_t *s, smf_event_t *e)
{
smf_sess_t *sess = NULL;
ogs_assert(s);
ogs_assert(e);
smf_sm_debug(e);
sess = e->sess;
ogs_assert(sess);
switch (e->id) {
case OGS_FSM_ENTRY_SIG:
break;
case OGS_FSM_EXIT_SIG:
break;
default:
2020-07-20 01:42:58 +00:00
ogs_error("Unknown event %s", smf_event_get_name(e));
2020-06-17 05:22:28 +00:00
break;
}
}