/* * Copyright (C) 2019,2020 by Sukchan Lee * * 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 . */ #include "binding.h" #include "sbi-path.h" #include "gn-handler.h" #include "gx-handler.h" #include "gy-handler.h" #include "n4-handler.h" #include "s5c-handler.h" #include "nnrf-handler.h" #include "nsmf-handler.h" #include "nudm-handler.h" #include "npcf-handler.h" #include "namf-handler.h" #include "gsm-handler.h" #include "ngap-handler.h" #include "gtp-path.h" #include "pfcp-path.h" #include "ngap-path.h" #include "fd-path.h" static uint8_t gtp_cause_from_diameter(uint8_t gtp_version, const uint32_t dia_err, const uint32_t *dia_exp_err) { switch (gtp_version) { case 1: switch (dia_err) { case OGS_DIAM_UNKNOWN_SESSION_ID: return OGS_GTP1_CAUSE_APN_ACCESS_DENIED; } break; case 2: switch (dia_err) { case OGS_DIAM_UNKNOWN_SESSION_ID: return OGS_GTP2_CAUSE_APN_ACCESS_DENIED_NO_SUBSCRIPTION; } break; } ogs_error("Unexpected Diameter Result Code %d/%d, defaulting to severe " "network failure", dia_err, dia_exp_err ? *dia_exp_err : -1); switch (gtp_version) { case 1: return OGS_GTP1_CAUSE_USER_AUTHENTICATION_FAILED; case 2: default: return OGS_GTP2_CAUSE_UE_NOT_AUTHORISED_BY_OCS_OR_EXTERNAL_AAA_SERVER; } } static void send_gtp_create_err_msg(const smf_sess_t *sess, ogs_gtp_xact_t *gtp_xact, uint8_t gtp_cause) { if (gtp_xact->gtp_version == 1) ogs_gtp1_send_error_message(gtp_xact, sess->sgw_s5c_teid, OGS_GTP1_CREATE_PDP_CONTEXT_RESPONSE_TYPE, gtp_cause); else ogs_gtp2_send_error_message(gtp_xact, sess->sgw_s5c_teid, OGS_GTP2_CREATE_SESSION_RESPONSE_TYPE, gtp_cause); } static void send_gtp_delete_err_msg(const smf_sess_t *sess, ogs_gtp_xact_t *gtp_xact, uint8_t gtp_cause) { if (gtp_xact->gtp_version == 1) ogs_gtp1_send_error_message(gtp_xact, sess->sgw_s5c_teid, OGS_GTP1_DELETE_PDP_CONTEXT_RESPONSE_TYPE, gtp_cause); else ogs_gtp2_send_error_message(gtp_xact, sess->sgw_s5c_teid, OGS_GTP2_DELETE_SESSION_RESPONSE_TYPE, gtp_cause); } static bool send_ccr_init_req_gx_gy(smf_sess_t *sess, smf_event_t *e) { int use_gy = smf_use_gy_iface(); if (use_gy == -1) { ogs_error("No Gy Diameter Peer"); /* TODO: drop Gx connection here, * possibly move to another "releasing" state! */ uint8_t gtp_cause = (e->gtp_xact->gtp_version == 1) ? OGS_GTP1_CAUSE_NO_RESOURCES_AVAILABLE : OGS_GTP2_CAUSE_UE_NOT_AUTHORISED_BY_OCS_OR_EXTERNAL_AAA_SERVER; send_gtp_create_err_msg(sess, e->gtp_xact, gtp_cause); return false; } sess->sm_data.gx_ccr_init_in_flight = true; smf_gx_send_ccr(sess, e->gtp_xact, OGS_DIAM_GX_CC_REQUEST_TYPE_INITIAL_REQUEST); if (use_gy == 1) { /* Gy is available, * set up session for the bearer before accepting it towards the UE */ sess->sm_data.gy_ccr_init_in_flight = true; smf_gy_send_ccr(sess, e->gtp_xact, OGS_DIAM_GY_CC_REQUEST_TYPE_INITIAL_REQUEST); } return true; } static bool send_ccr_termination_req_gx_gy_s6b(smf_sess_t *sess, smf_event_t *e) { /* TODO: we should take into account here whether "sess" has an active Gy session created, not whether one was supposedly created as per policy */ int use_gy = smf_use_gy_iface(); if (use_gy == -1) { ogs_error("No Gy Diameter Peer"); /* TODO: drop Gx connection here, * possibly move to another "releasing" state! */ uint8_t gtp_cause = (e->gtp_xact->gtp_version == 1) ? OGS_GTP1_CAUSE_NO_RESOURCES_AVAILABLE : OGS_GTP2_CAUSE_UE_NOT_AUTHORISED_BY_OCS_OR_EXTERNAL_AAA_SERVER; send_gtp_delete_err_msg(sess, e->gtp_xact, gtp_cause); return false; } if (sess->gtp_rat_type == OGS_GTP2_RAT_TYPE_WLAN) { sess->sm_data.s6b_str_in_flight = true; smf_s6b_send_str(sess, e->gtp_xact, OGS_DIAM_TERMINATION_CAUSE_DIAMETER_LOGOUT); } sess->sm_data.gx_ccr_term_in_flight = true; smf_gx_send_ccr(sess, e->gtp_xact, OGS_DIAM_GX_CC_REQUEST_TYPE_TERMINATION_REQUEST); if (use_gy == 1) { /* Gy is available, * set up session for the bearer before accepting it towards the UE */ sess->sm_data.gy_ccr_term_in_flight = true; smf_gy_send_ccr(sess, e->gtp_xact, OGS_DIAM_GY_CC_REQUEST_TYPE_TERMINATION_REQUEST); } return true; } static bool send_sbi_message_from_delete_trigger( smf_sess_t *sess, ogs_sbi_stream_t *stream, int trigger) { ogs_sbi_message_t sendmsg; ogs_sbi_response_t *response = NULL; if (trigger == OGS_PFCP_DELETE_TRIGGER_LOCAL_INITIATED) { /* Nothing */ } else if (trigger == OGS_PFCP_DELETE_TRIGGER_UE_REQUESTED) { ogs_pkbuf_t *n1smbuf = NULL, *n2smbuf = NULL; n1smbuf = gsm_build_pdu_session_release_command( sess, OGS_5GSM_CAUSE_REGULAR_DEACTIVATION); ogs_assert(n1smbuf); n2smbuf = ngap_build_pdu_session_resource_release_command_transfer( sess, SMF_NGAP_STATE_DELETE_TRIGGER_UE_REQUESTED, NGAP_Cause_PR_nas, NGAP_CauseNas_normal_release); ogs_assert(n2smbuf); ogs_assert(stream); smf_sbi_send_sm_context_updated_data_n1_n2_message(sess, stream, n1smbuf, OpenAPI_n2_sm_info_type_PDU_RES_REL_CMD, n2smbuf); } else if (trigger == OGS_PFCP_DELETE_TRIGGER_AMF_UPDATE_SM_CONTEXT || trigger == OGS_PFCP_DELETE_TRIGGER_AMF_RELEASE_SM_CONTEXT) { memset(&sendmsg, 0, sizeof(sendmsg)); response = ogs_sbi_build_response( &sendmsg, OGS_SBI_HTTP_STATUS_NO_CONTENT); ogs_assert(response); ogs_assert(stream); ogs_assert(true == ogs_sbi_server_send_response(stream, response)); } else if (trigger == OGS_PFCP_DELETE_TRIGGER_PCF_INITIATED) { smf_n1_n2_message_transfer_param_t param; memset(¶m, 0, sizeof(param)); param.state = SMF_NETWORK_REQUESTED_PDU_SESSION_RELEASE; param.n2smbuf = ngap_build_pdu_session_resource_release_command_transfer( sess, SMF_NGAP_STATE_DELETE_TRIGGER_PCF_INITIATED, NGAP_Cause_PR_nas, NGAP_CauseNas_normal_release); ogs_assert(param.n2smbuf); param.skip_ind = true; smf_namf_comm_send_n1_n2_message_transfer(sess, ¶m); } else { ogs_fatal("Unknown trigger [%d]", trigger); ogs_assert_if_reached(); } return true; } void smf_gsm_state_initial(ogs_fsm_t *s, smf_event_t *e) { int rv; char *strerror = NULL; smf_ue_t *smf_ue = NULL; smf_sess_t *sess = NULL; ogs_gtp1_message_t *gtp1_message = NULL; ogs_gtp2_message_t *gtp2_message = NULL; uint8_t gtp1_cause, gtp2_cause; ogs_nas_5gs_message_t *nas_message = NULL; ogs_sbi_stream_t *stream = NULL; 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: /* reset state: */ sess->sm_data.gx_ccr_init_in_flight = false; sess->sm_data.gy_ccr_init_in_flight = false; sess->sm_data.gx_cca_init_err = ER_DIAMETER_SUCCESS; sess->sm_data.gy_cca_init_err = ER_DIAMETER_SUCCESS; break; case OGS_FSM_EXIT_SIG: break; case SMF_EVT_GN_MESSAGE: gtp1_message = e->gtp1_message; ogs_assert(gtp1_message); switch(gtp1_message->h.type) { case OGS_GTP1_CREATE_PDP_CONTEXT_REQUEST_TYPE: gtp1_cause = smf_gn_handle_create_pdp_context_request(sess, e->gtp_xact, &e->gtp1_message->create_pdp_context_request); if (gtp1_cause != OGS_GTP1_CAUSE_REQUEST_ACCEPTED) { send_gtp_create_err_msg(sess, e->gtp_xact, gtp1_cause); return; } if (send_ccr_init_req_gx_gy(sess, e) == true) OGS_FSM_TRAN(s, smf_gsm_state_wait_epc_auth_initial); } break; case SMF_EVT_S5C_MESSAGE: gtp2_message = e->gtp2_message; ogs_assert(gtp2_message); switch(gtp2_message->h.type) { case OGS_GTP2_CREATE_SESSION_REQUEST_TYPE: gtp2_cause = smf_s5c_handle_create_session_request(sess, e->gtp_xact, &e->gtp2_message->create_session_request); if (gtp2_cause != OGS_GTP2_CAUSE_REQUEST_ACCEPTED) { send_gtp_create_err_msg(sess, e->gtp_xact, gtp2_cause); return; } switch (sess->gtp_rat_type) { case OGS_GTP2_RAT_TYPE_EUTRAN: if (send_ccr_init_req_gx_gy(sess, e) == true) OGS_FSM_TRAN(s, smf_gsm_state_wait_epc_auth_initial); break; case OGS_GTP2_RAT_TYPE_WLAN: smf_s6b_send_aar(sess, e->gtp_xact); OGS_FSM_TRAN(s, smf_gsm_state_wait_epc_auth_initial); break; default: ogs_error("Unknown RAT Type [%d]", sess->gtp_rat_type); ogs_assert_if_reached(); } break; default: ogs_error("Not implmeneted(type:%d)", gtp2_message->h.type); } break; case SMF_EVT_SBI_SERVER: sbi_message = e->sbi.message; ogs_assert(sbi_message); stream = e->sbi.data; ogs_assert(stream); 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) CASE(OGS_SBI_RESOURCE_NAME_RELEASE) ogs_error("Invalid resource name [%s]", sbi_message->h.resource.component[2]); ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, sbi_message, "Invalid resource name [%s]", sbi_message->h.resource.component[2])); OGS_FSM_TRAN(s, smf_gsm_state_exception); break; DEFAULT if (smf_nsmf_handle_create_sm_context( sess, stream, sbi_message) == false) { ogs_error("smf_nsmf_handle_create_sm_context() failed"); OGS_FSM_TRAN(s, smf_gsm_state_exception); } END break; DEFAULT ogs_error("Invalid API name [%s]", sbi_message->h.service.name); ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, sbi_message, "Invalid API name", sbi_message->h.service.name)); OGS_FSM_TRAN(s, smf_gsm_state_exception); 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); switch (nas_message->gsm.h.message_type) { case OGS_NAS_5GS_PDU_SESSION_ESTABLISHMENT_REQUEST: rv = gsm_handle_pdu_session_establishment_request(sess, stream, &nas_message->gsm.pdu_session_establishment_request); if (rv != OGS_OK) { ogs_error("[%s:%d] Cannot handle NAS message", smf_ue->supi, sess->psi); OGS_FSM_TRAN(s, smf_gsm_state_exception); break; } OGS_FSM_TRAN(s, smf_gsm_state_wait_5gc_sm_policy_association); break; default: strerror = ogs_msprintf("Unknown message [%d]", nas_message->gsm.h.message_type); ogs_assert(strerror); ogs_error("%s", strerror); ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, NULL, strerror, NULL)); ogs_free(strerror); OGS_FSM_TRAN(s, smf_gsm_state_exception); } break; default: ogs_error("No handler for event %s", smf_event_get_name(e)); break; } } void smf_gsm_state_wait_epc_auth_initial(ogs_fsm_t *s, smf_event_t *e) { smf_sess_t *sess = NULL; ogs_diam_gy_message_t *gy_message = NULL; ogs_diam_gx_message_t *gx_message = NULL; uint32_t diam_err; ogs_assert(s); ogs_assert(e); smf_sm_debug(e); sess = e->sess; ogs_assert(sess); switch (e->id) { case SMF_EVT_GX_MESSAGE: gx_message = e->gx_message; ogs_assert(gx_message); switch(gx_message->cmd_code) { case OGS_DIAM_GX_CMD_CODE_CREDIT_CONTROL: switch(gx_message->cc_request_type) { case OGS_DIAM_GX_CC_REQUEST_TYPE_INITIAL_REQUEST: ogs_assert(e->gtp_xact); diam_err = smf_gx_handle_cca_initial_request(sess, gx_message, e->gtp_xact); sess->sm_data.gx_ccr_init_in_flight = false; sess->sm_data.gx_cca_init_err = diam_err; goto test_can_proceed; } break; } break; case SMF_EVT_GY_MESSAGE: gy_message = e->gy_message; ogs_assert(gy_message); switch(gy_message->cmd_code) { case OGS_DIAM_GY_CMD_CODE_CREDIT_CONTROL: switch(gy_message->cc_request_type) { case OGS_DIAM_GY_CC_REQUEST_TYPE_INITIAL_REQUEST: ogs_assert(e->gtp_xact); diam_err = smf_gy_handle_cca_initial_request(sess, gy_message, e->gtp_xact); sess->sm_data.gy_ccr_init_in_flight = false; sess->sm_data.gy_cca_init_err = diam_err; goto test_can_proceed; } break; } break; } return; test_can_proceed: /* First wait for both Gx and Gy requests to be done: */ if (!sess->sm_data.gx_ccr_init_in_flight && !sess->sm_data.gy_ccr_init_in_flight) { diam_err = ER_DIAMETER_SUCCESS; if (sess->sm_data.gx_cca_init_err != ER_DIAMETER_SUCCESS) diam_err = sess->sm_data.gx_cca_init_err; if (sess->sm_data.gy_cca_init_err != ER_DIAMETER_SUCCESS) diam_err = sess->sm_data.gy_cca_init_err; if (diam_err == ER_DIAMETER_SUCCESS) { OGS_FSM_TRAN(s, &smf_gsm_state_wait_pfcp_establishment); ogs_assert(OGS_OK == smf_epc_pfcp_send_session_establishment_request( sess, e->gtp_xact)); } else { /* FIXME: tear down Gx/Gy session * if its sm_data.*init_err == ER_DIAMETER_SUCCESS */ uint8_t gtp_cause = gtp_cause_from_diameter( e->gtp_xact->gtp_version, diam_err, NULL); send_gtp_create_err_msg(sess, e->gtp_xact, gtp_cause); } } } void smf_gsm_state_wait_5gc_sm_policy_association(ogs_fsm_t *s, smf_event_t *e) { char *strerror = NULL; smf_ue_t *smf_ue = NULL; smf_sess_t *sess = NULL; ogs_sbi_stream_t *stream = NULL; ogs_sbi_message_t *sbi_message = NULL; int state = 0; 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_CLIENT: sbi_message = e->sbi.message; ogs_assert(sbi_message); sess = e->sess; ogs_assert(sess); smf_ue = sess->smf_ue; ogs_assert(smf_ue); SWITCH(sbi_message->h.service.name) CASE(OGS_SBI_SERVICE_NAME_NUDM_SDM) stream = e->sbi.data; ogs_assert(stream); SWITCH(sbi_message->h.resource.component[1]) CASE(OGS_SBI_RESOURCE_NAME_SM_DATA) if (sbi_message->res_status != OGS_SBI_HTTP_STATUS_OK) { 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_assert(true == ogs_sbi_server_send_error( stream, sbi_message->res_status, sbi_message, strerror, NULL)); ogs_free(strerror); OGS_FSM_TRAN(s, smf_gsm_state_exception); break; } if (smf_nudm_sdm_handle_get( sess, stream, sbi_message) == false) { ogs_error("smf_nudm_sdm_handle_get() failed"); OGS_FSM_TRAN(s, smf_gsm_state_exception); } break; DEFAULT strerror = ogs_msprintf("[%s:%d] Invalid resource name [%s]", smf_ue->supi, sess->psi, sbi_message->h.resource.component[1]); ogs_assert(strerror); ogs_error("%s", strerror); ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, sbi_message, strerror, NULL)); ogs_free(strerror); OGS_FSM_TRAN(s, smf_gsm_state_exception); END break; CASE(OGS_SBI_SERVICE_NAME_NPCF_SMPOLICYCONTROL) stream = e->sbi.data; state = e->sbi.state; SWITCH(sbi_message->h.resource.component[0]) CASE(OGS_SBI_RESOURCE_NAME_SM_POLICIES) if (sbi_message->h.resource.component[1]) { strerror = ogs_msprintf("[%s:%d] " "Unknown resource name [%s]", smf_ue->supi, sess->psi, sbi_message->h.resource.component[1]); ogs_assert(strerror); ogs_error("%s", strerror); if (stream) ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, sbi_message, strerror, NULL)); ogs_free(strerror); OGS_FSM_TRAN(s, smf_gsm_state_exception); } else { ogs_assert(stream); 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_assert(true == ogs_sbi_server_send_error(stream, sbi_message->res_status, sbi_message, strerror, NULL)); ogs_free(strerror); OGS_FSM_TRAN(s, smf_gsm_state_exception); break; } if (smf_npcf_smpolicycontrol_handle_create( sess, stream, state, sbi_message) == true) { OGS_FSM_TRAN(s, &smf_gsm_state_wait_pfcp_establishment); } else { ogs_error( "smf_npcf_smpolicycontrol_handle_create() failed"); OGS_FSM_TRAN(s, smf_gsm_state_exception); } } 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_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, sbi_message, strerror, NULL)); ogs_free(strerror); OGS_FSM_TRAN(s, smf_gsm_state_exception); END break; DEFAULT ogs_error("[%s:%d] Invalid API name [%s]", smf_ue->supi, sess->psi, sbi_message->h.service.name); ogs_assert_if_reached(); END break; default: ogs_error("No handler for event %s", smf_event_get_name(e)); break; } } void smf_gsm_state_wait_pfcp_establishment(ogs_fsm_t *s, smf_event_t *e) { smf_sess_t *sess = NULL; uint8_t pfcp_cause, gtp_cause; smf_n1_n2_message_transfer_param_t param; ogs_pfcp_xact_t *pfcp_xact = NULL; ogs_pfcp_message_t *pfcp_message = NULL; ogs_assert(s); ogs_assert(e); smf_sm_debug(e); sess = e->sess; ogs_assert(sess); switch (e->id) { case SMF_EVT_N4_MESSAGE: pfcp_xact = e->pfcp_xact; ogs_assert(pfcp_xact); pfcp_message = e->pfcp_message; ogs_assert(pfcp_message); switch (pfcp_message->h.type) { case OGS_PFCP_SESSION_ESTABLISHMENT_RESPONSE_TYPE: if (pfcp_xact->epc) { ogs_gtp_xact_t *gtp_xact = pfcp_xact->assoc_xact; ogs_assert(gtp_xact); pfcp_cause = smf_epc_n4_handle_session_establishment_response( sess, pfcp_xact, &pfcp_message->pfcp_session_establishment_response); if (pfcp_cause != OGS_PFCP_CAUSE_REQUEST_ACCEPTED) { /* FIXME: tear down Gy and Gx */ gtp_cause = gtp_cause_from_pfcp( pfcp_cause, gtp_xact->gtp_version); send_gtp_create_err_msg(sess, e->gtp_xact, gtp_cause); return; } switch (gtp_xact->gtp_version) { case 1: ogs_assert(OGS_OK == smf_gtp1_send_create_pdp_context_response( sess, gtp_xact)); break; case 2: ogs_assert(OGS_OK == smf_gtp2_send_create_session_response( sess, gtp_xact)); break; } if (sess->gtp_rat_type == OGS_GTP2_RAT_TYPE_WLAN) { /* * TS23.214 * 6.3.1.7 Procedures with modification of bearer * p50 * 2. ... * For "PGW/MME initiated bearer deactivation procedure", * PGW-C shall indicate PGW-U to stop counting and stop * forwarding downlink packets for the affected bearer(s). */ ogs_assert(OGS_OK == smf_epc_pfcp_send_deactivation(sess, OGS_GTP2_CAUSE_RAT_CHANGED_FROM_3GPP_TO_NON_3GPP)); } smf_bearer_binding(sess); } else { pfcp_cause = smf_5gc_n4_handle_session_establishment_response( sess, pfcp_xact, &pfcp_message->pfcp_session_establishment_response); if (pfcp_cause != OGS_PFCP_CAUSE_REQUEST_ACCEPTED) return; memset(¶m, 0, sizeof(param)); param.state = SMF_UE_REQUESTED_PDU_SESSION_ESTABLISHMENT; param.n1smbuf = gsm_build_pdu_session_establishment_accept(sess); ogs_assert(param.n1smbuf); param.n2smbuf = ngap_build_pdu_session_resource_setup_request_transfer( sess); ogs_assert(param.n2smbuf); smf_namf_comm_send_n1_n2_message_transfer(sess, ¶m); } OGS_FSM_TRAN(s, &smf_gsm_state_operational); break; default: ogs_error("cannot handle PFCP message type[%d]", pfcp_message->h.type); } } } void smf_gsm_state_operational(ogs_fsm_t *s, smf_event_t *e) { int rv, ngap_state; char *strerror = NULL; smf_ue_t *smf_ue = NULL; smf_sess_t *sess = NULL; ogs_pkbuf_t *pkbuf = NULL; ogs_nas_5gs_message_t *nas_message = NULL; ogs_sbi_stream_t *stream = NULL; ogs_sbi_message_t *sbi_message = NULL; ogs_gtp1_message_t *gtp1_message = NULL; ogs_gtp2_message_t *gtp2_message = NULL; uint8_t gtp1_cause, gtp2_cause; bool release; 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_GN_MESSAGE: gtp1_message = e->gtp1_message; ogs_assert(gtp1_message); switch(gtp1_message->h.type) { case OGS_GTP1_DELETE_PDP_CONTEXT_REQUEST_TYPE: gtp1_cause = smf_gn_handle_delete_pdp_context_request(sess, e->gtp_xact, >p1_message->delete_pdp_context_request); if (gtp1_cause != OGS_GTP1_CAUSE_REQUEST_ACCEPTED) { ogs_gtp1_send_error_message(e->gtp_xact, sess->sgw_s5c_teid, OGS_GTP1_DELETE_PDP_CONTEXT_RESPONSE_TYPE, gtp1_cause); return; } OGS_FSM_TRAN(s, &smf_gsm_state_wait_pfcp_deletion); } break; case SMF_EVT_S5C_MESSAGE: gtp2_message = e->gtp2_message; ogs_assert(gtp2_message); switch(gtp2_message->h.type) { case OGS_GTP2_DELETE_SESSION_REQUEST_TYPE: gtp2_cause = smf_s5c_handle_delete_session_request( sess, e->gtp_xact, >p2_message->delete_session_request); if (gtp2_cause != OGS_GTP2_CAUSE_REQUEST_ACCEPTED) { ogs_gtp2_send_error_message(e->gtp_xact, sess->sgw_s5c_teid, OGS_GTP2_DELETE_SESSION_RESPONSE_TYPE, gtp2_cause); return; } OGS_FSM_TRAN(s, &smf_gsm_state_wait_pfcp_deletion); break; case OGS_GTP2_DELETE_BEARER_RESPONSE_TYPE: release = smf_s5c_handle_delete_bearer_response( sess, e->gtp_xact, &e->gtp2_message->delete_bearer_response); if (release) { e->gtp_xact = NULL; OGS_FSM_TRAN(s, &smf_gsm_state_wait_pfcp_deletion); } break; default: ogs_error("Not implmeneted(type:%d)", gtp2_message->h.type); } break; case SMF_EVT_SBI_SERVER: sbi_message = e->sbi.message; ogs_assert(sbi_message); stream = e->sbi.data; ogs_assert(stream); 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); break; CASE(OGS_SBI_RESOURCE_NAME_RELEASE) smf_nsmf_handle_release_sm_context(sess, stream, sbi_message); break; DEFAULT ogs_error("Invalid resource name [%s]", sbi_message->h.resource.component[2]); ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, sbi_message, "Invalid resource name [%s]", sbi_message->h.resource.component[2])); OGS_FSM_TRAN(s, smf_gsm_state_exception); END break; DEFAULT ogs_error("Invalid API name [%s]", sbi_message->h.service.name); ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, sbi_message, "Invalid API name", sbi_message->h.service.name)); OGS_FSM_TRAN(s, smf_gsm_state_exception); END break; case SMF_EVT_SBI_CLIENT: sbi_message = e->sbi.message; ogs_assert(sbi_message); sess = e->sess; ogs_assert(sess); smf_ue = sess->smf_ue; ogs_assert(smf_ue); SWITCH(sbi_message->h.service.name) CASE(OGS_SBI_SERVICE_NAME_NPCF_SMPOLICYCONTROL) stream = e->sbi.data; SWITCH(sbi_message->h.resource.component[0]) CASE(OGS_SBI_RESOURCE_NAME_SM_POLICIES) if (!sbi_message->h.resource.component[1]) { ogs_assert(stream); 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_assert(true == ogs_sbi_server_send_error( stream, sbi_message->res_status, sbi_message, strerror, NULL)); ogs_free(strerror); OGS_FSM_TRAN(s, smf_gsm_state_exception); break; } else { SWITCH(sbi_message->h.resource.component[2]) CASE(OGS_SBI_RESOURCE_NAME_DELETE) if (sbi_message->res_status != OGS_SBI_HTTP_STATUS_NO_CONTENT) { 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_free(strerror); OGS_FSM_TRAN(s, smf_gsm_state_exception); break; } OGS_FSM_TRAN(&sess->sm, &smf_gsm_state_wait_pfcp_deletion); break; DEFAULT strerror = ogs_msprintf("[%s:%d] " "Unknown resource name [%s]", smf_ue->supi, sess->psi, sbi_message->h.resource.component[2]); ogs_assert(strerror); ogs_error("%s", strerror); if (stream) ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, sbi_message, strerror, NULL)); ogs_free(strerror); OGS_FSM_TRAN(s, smf_gsm_state_exception); END } 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_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, sbi_message, strerror, NULL)); ogs_free(strerror); OGS_FSM_TRAN(s, smf_gsm_state_exception); END break; CASE(OGS_SBI_SERVICE_NAME_NAMF_COMM) SWITCH(sbi_message->h.resource.component[0]) CASE(OGS_SBI_RESOURCE_NAME_UE_CONTEXTS) smf_namf_comm_handle_n1_n2_message_transfer( sess, e->sbi.state, sbi_message); break; DEFAULT ogs_error("[%s:%d] Invalid resource name [%s]", smf_ue->supi, sess->psi, sbi_message->h.resource.component[0]); ogs_assert_if_reached(); END break; DEFAULT ogs_error("[%s:%d] Invalid API name [%s]", smf_ue->supi, sess->psi, sbi_message->h.service.name); ogs_assert_if_reached(); 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); switch (nas_message->gsm.h.message_type) { case OGS_NAS_5GS_PDU_SESSION_MODIFICATION_REQUEST: rv = gsm_handle_pdu_session_modification_request(sess, stream, &nas_message->gsm.pdu_session_modification_request); if (rv != OGS_OK) { ogs_error("[%s:%d] Cannot handle NAS message", smf_ue->supi, sess->psi); OGS_FSM_TRAN(s, smf_gsm_state_exception); } break; case OGS_NAS_5GS_PDU_SESSION_MODIFICATION_COMPLETE: ogs_assert(true == ogs_sbi_send_http_status_no_content(stream)); break; case OGS_NAS_5GS_PDU_SESSION_RELEASE_REQUEST: if (sess->policy_association_id) { smf_npcf_smpolicycontrol_param_t param; memset(¶m, 0, sizeof(param)); param.ran_nas_release.gsm_cause = OGS_5GSM_CAUSE_REGULAR_DEACTIVATION; param.ran_nas_release.ngap_cause.group = NGAP_Cause_PR_nas; param.ran_nas_release.ngap_cause.value = NGAP_CauseNas_normal_release; ogs_assert(true == smf_sbi_discover_and_send(OpenAPI_nf_type_PCF, sess, stream, OGS_PFCP_DELETE_TRIGGER_UE_REQUESTED, ¶m, smf_npcf_smpolicycontrol_build_delete)); } else { ogs_error("[%s:%d] No PolicyAssociationId", smf_ue->supi, sess->psi); OGS_FSM_TRAN(s, smf_gsm_state_exception); } break; default: strerror = ogs_msprintf("Unknown message [%d]", nas_message->gsm.h.message_type); ogs_assert(strerror); ogs_error("%s", strerror); ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, NULL, strerror, NULL)); ogs_free(strerror); } 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); 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); 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_SETUP_FAIL: rv = ngap_handle_pdu_session_resource_setup_unsuccessful_transfer( sess, stream, pkbuf); 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); 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_REL_RSP: ngap_state = sess->ngap_state.pdu_session_resource_release; if (ngap_state == SMF_NGAP_STATE_NONE) { strerror = ogs_msprintf( "[%s:%d] No PDUSessionResourceReleaseRequest", smf_ue->supi, sess->psi); ogs_assert(strerror); ogs_error("%s", strerror); ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, NULL, strerror, NULL)); ogs_free(strerror); OGS_FSM_TRAN(s, smf_gsm_state_exception); } else if (ngap_state == SMF_NGAP_STATE_ERROR_INDICATION_RECEIVED_FROM_5G_AN) { smf_n1_n2_message_transfer_param_t param; ogs_assert(true == ogs_sbi_send_http_status_no_content(stream)); memset(¶m, 0, sizeof(param)); param.state = SMF_NETWORK_TRIGGERED_SERVICE_REQUEST; param.n2smbuf = ngap_build_pdu_session_resource_setup_request_transfer( sess); ogs_assert(param.n2smbuf); param.n1n2_failure_txf_notif_uri = true; smf_namf_comm_send_n1_n2_message_transfer(sess, ¶m); } else { ogs_fatal("Invalid state [%d]", ngap_state); ogs_assert_if_reached(); } break; case OpenAPI_n2_sm_info_type_PATH_SWITCH_REQ: rv = ngap_handle_path_switch_request_transfer(sess, stream, pkbuf); 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_HANDOVER_REQUIRED: rv = ngap_handle_handover_required_transfer(sess, stream, pkbuf); 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_HANDOVER_REQ_ACK: rv = ngap_handle_handover_request_ack(sess, stream, pkbuf); 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; default: ogs_error("Unknown message[%d]", e->ngap.type); } break; default: ogs_error("Unknown event [%s]", smf_event_get_name(e)); } } void smf_gsm_state_wait_pfcp_deletion(ogs_fsm_t *s, smf_event_t *e) { int status; smf_sess_t *sess = NULL; ogs_sbi_stream_t *stream = NULL; ogs_pfcp_xact_t *pfcp_xact = NULL; ogs_pfcp_message_t *pfcp_message = NULL; uint8_t pfcp_cause, gtp_cause; ogs_gtp_xact_t *gtp_xact = 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: /* Since `pfcp_xact->epc` is not avaiable, * we'll use `sess->epc` */ if (sess->epc) { /* EPC */ ogs_assert(OGS_OK == smf_epc_pfcp_send_session_deletion_request(sess, e->gtp_xact)); } else { /* 5GC */ stream = e->sbi.data; ogs_assert(stream); ogs_assert(OGS_OK == smf_5gc_pfcp_send_session_deletion_request( sess, stream, e->sbi.state)); } break; case OGS_FSM_EXIT_SIG: break; case SMF_EVT_GN_MESSAGE: case SMF_EVT_S5C_MESSAGE: break; /* ignore */ case SMF_EVT_N4_MESSAGE: pfcp_xact = e->pfcp_xact; ogs_assert(pfcp_xact); pfcp_message = e->pfcp_message; ogs_assert(pfcp_message); switch (pfcp_message->h.type) { case OGS_PFCP_SESSION_DELETION_RESPONSE_TYPE: if (pfcp_xact->epc) { gtp_xact = pfcp_xact->assoc_xact; pfcp_cause = smf_epc_n4_handle_session_deletion_response( sess, pfcp_xact, &pfcp_message->pfcp_session_deletion_response); if (pfcp_cause != OGS_PFCP_CAUSE_REQUEST_ACCEPTED) { /* FIXME: tear down Gy and Gx */ ogs_assert(gtp_xact); gtp_cause = gtp_cause_from_pfcp( pfcp_cause, gtp_xact->gtp_version); send_gtp_delete_err_msg(sess, gtp_xact, gtp_cause); break; } e->gtp_xact = gtp_xact; if (send_ccr_termination_req_gx_gy_s6b(sess, e) == true) OGS_FSM_TRAN(s, &smf_gsm_state_wait_epc_auth_release); /* else: free session? */ } else { int trigger; stream = pfcp_xact->assoc_stream; trigger = pfcp_xact->delete_trigger; ogs_assert(trigger); ogs_pfcp_xact_commit(pfcp_xact); status = smf_5gc_n4_handle_session_deletion_response( sess, stream, trigger, &pfcp_message->pfcp_session_deletion_response); if (status != OGS_SBI_HTTP_STATUS_OK) { ogs_error( "[%d] smf_5gc_n4_handle_session_deletion_response() " "failed", trigger); OGS_FSM_TRAN(s, smf_gsm_state_session_will_release); break; } if (send_sbi_message_from_delete_trigger( sess, stream, trigger) == true) { if (trigger == OGS_PFCP_DELETE_TRIGGER_LOCAL_INITIATED) { ogs_warn("OLD Session Released"); OGS_FSM_TRAN(s, smf_gsm_state_session_will_release); } else if ( trigger == OGS_PFCP_DELETE_TRIGGER_UE_REQUESTED || trigger == OGS_PFCP_DELETE_TRIGGER_PCF_INITIATED) { OGS_FSM_TRAN(s, smf_gsm_state_wait_5gc_n1_n2_release); } else if (trigger == OGS_PFCP_DELETE_TRIGGER_AMF_UPDATE_SM_CONTEXT || trigger == OGS_PFCP_DELETE_TRIGGER_AMF_RELEASE_SM_CONTEXT) { OGS_FSM_TRAN(s, smf_gsm_state_session_will_release); } else { ogs_fatal("Unknown trigger [%d]", trigger); ogs_assert_if_reached(); } } } break; default: ogs_error("cannot handle PFCP message type[%d]", pfcp_message->h.type); } } } void smf_gsm_state_wait_epc_auth_release(ogs_fsm_t *s, smf_event_t *e) { smf_sess_t *sess = NULL; ogs_diam_gx_message_t *gx_message = NULL; ogs_diam_gy_message_t *gy_message = NULL; ogs_diam_s6b_message_t *s6b_message = NULL; uint32_t diam_err; ogs_assert(s); ogs_assert(e); smf_sm_debug(e); sess = e->sess; ogs_assert(sess); switch (e->id) { case OGS_FSM_ENTRY_SIG: /* reset state: */ sess->sm_data.gx_cca_term_err = ER_DIAMETER_SUCCESS; sess->sm_data.gy_cca_term_err = ER_DIAMETER_SUCCESS; sess->sm_data.s6b_sta_err = ER_DIAMETER_SUCCESS; break; case SMF_EVT_GN_MESSAGE: case SMF_EVT_S5C_MESSAGE: case SMF_EVT_N4_MESSAGE: break; /* ignore */ case SMF_EVT_GX_MESSAGE: gx_message = e->gx_message; ogs_assert(gx_message); switch(gx_message->cmd_code) { case OGS_DIAM_GX_CMD_CODE_CREDIT_CONTROL: switch(gx_message->cc_request_type) { case OGS_DIAM_GX_CC_REQUEST_TYPE_TERMINATION_REQUEST: diam_err = smf_gx_handle_cca_termination_request(sess, gx_message, e->gtp_xact); sess->sm_data.gx_ccr_term_in_flight = false; sess->sm_data.gx_cca_term_err = diam_err; goto test_can_proceed; } break; } break; case SMF_EVT_GY_MESSAGE: gy_message = e->gy_message; ogs_assert(gy_message); switch(gy_message->cmd_code) { case OGS_DIAM_GY_CMD_CODE_CREDIT_CONTROL: switch(gy_message->cc_request_type) { case OGS_DIAM_GY_CC_REQUEST_TYPE_TERMINATION_REQUEST: ogs_assert(e->gtp_xact); diam_err = smf_gy_handle_cca_termination_request(sess, gy_message, e->gtp_xact); sess->sm_data.gy_ccr_term_in_flight = false; sess->sm_data.gy_cca_term_err = diam_err; goto test_can_proceed; } break; } break; case SMF_EVT_S6B_MESSAGE: s6b_message = e->s6b_message; ogs_assert(s6b_message); switch(s6b_message->cmd_code) { case OGS_DIAM_S6B_CMD_SESSION_TERMINATION: sess->sm_data.s6b_str_in_flight = false; /* TODO: validate error code from message below: */ sess->sm_data.s6b_sta_err = ER_DIAMETER_SUCCESS; goto test_can_proceed; } break; } return; test_can_proceed: /* First wait for both Gx and Gy requests to be done: */ if (!sess->sm_data.gx_ccr_term_in_flight && !sess->sm_data.gy_ccr_term_in_flight && !sess->sm_data.s6b_str_in_flight) { diam_err = ER_DIAMETER_SUCCESS; if (sess->sm_data.gx_cca_term_err != ER_DIAMETER_SUCCESS) diam_err = sess->sm_data.gx_cca_term_err; if (sess->sm_data.gy_cca_term_err != ER_DIAMETER_SUCCESS) diam_err = sess->sm_data.gy_cca_term_err; if (sess->sm_data.s6b_sta_err != ER_DIAMETER_SUCCESS) diam_err = sess->sm_data.s6b_sta_err; /* Initiated by peer request, let's answer: */ if (e->gtp_xact) { if (diam_err == ER_DIAMETER_SUCCESS) { /* * 1. MME sends Delete Session Request to SGW/SMF. * 2. SMF sends Delete Session Response to SGW/MME. */ switch (e->gtp_xact->gtp_version) { case 1: ogs_assert(OGS_OK == smf_gtp1_send_delete_pdp_context_response( sess, e->gtp_xact)); break; case 2: ogs_assert(OGS_OK == smf_gtp2_send_delete_session_response( sess, e->gtp_xact)); break; } } else { uint8_t gtp_cause = gtp_cause_from_diameter( e->gtp_xact->gtp_version, diam_err, NULL); send_gtp_delete_err_msg(sess, e->gtp_xact, gtp_cause); } } OGS_FSM_TRAN(s, &smf_gsm_state_session_will_release); } } void smf_gsm_state_wait_5gc_n1_n2_release(ogs_fsm_t *s, smf_event_t *e) { int ngap_state; char *strerror = NULL; smf_ue_t *smf_ue = NULL; smf_sess_t *sess = NULL; ogs_pkbuf_t *pkbuf = NULL; ogs_nas_5gs_message_t *nas_message = NULL; ogs_sbi_stream_t *stream = NULL; 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); 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); break; DEFAULT ogs_error("Invalid resource name [%s]", sbi_message->h.resource.component[2]); ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, sbi_message, "Invalid resource name [%s]", sbi_message->h.resource.component[2])); OGS_FSM_TRAN(s, smf_gsm_state_exception); END break; DEFAULT ogs_error("Invalid API name [%s]", sbi_message->h.service.name); ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, sbi_message, "Invalid API name", sbi_message->h.service.name)); OGS_FSM_TRAN(s, smf_gsm_state_exception); END break; case SMF_EVT_SBI_CLIENT: sbi_message = e->sbi.message; ogs_assert(sbi_message); sess = e->sess; ogs_assert(sess); smf_ue = sess->smf_ue; ogs_assert(smf_ue); SWITCH(sbi_message->h.service.name) CASE(OGS_SBI_SERVICE_NAME_NAMF_COMM) SWITCH(sbi_message->h.resource.component[0]) CASE(OGS_SBI_RESOURCE_NAME_UE_CONTEXTS) smf_namf_comm_handle_n1_n2_message_transfer( sess, e->sbi.state, sbi_message); break; DEFAULT ogs_error("[%s:%d] Invalid resource name [%s]", smf_ue->supi, sess->psi, sbi_message->h.resource.component[0]); ogs_assert_if_reached(); END break; DEFAULT ogs_error("[%s:%d] Invalid API name [%s]", smf_ue->supi, sess->psi, sbi_message->h.service.name); ogs_assert_if_reached(); END 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); pkbuf = e->pkbuf; ogs_assert(pkbuf); ogs_assert(e->ngap.type); switch (e->ngap.type) { case OpenAPI_n2_sm_info_type_PDU_RES_REL_RSP: ngap_state = sess->ngap_state.pdu_session_resource_release; if (ngap_state == SMF_NGAP_STATE_NONE) { strerror = ogs_msprintf( "[%s:%d] No PDUSessionResourceReleaseRequest", smf_ue->supi, sess->psi); ogs_assert(strerror); ogs_error("%s", strerror); ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, NULL, strerror, NULL)); ogs_free(strerror); OGS_FSM_TRAN(s, smf_gsm_state_exception); } else if ( ngap_state == SMF_NGAP_STATE_DELETE_TRIGGER_UE_REQUESTED || ngap_state == SMF_NGAP_STATE_DELETE_TRIGGER_PCF_INITIATED) { ogs_assert(true == ogs_sbi_send_http_status_no_content(stream)); } else { ogs_fatal("Invalid state [%d]", ngap_state); ogs_assert_if_reached(); } break; default: ogs_error("Unknown message[%d]", e->ngap.type); } 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); switch (nas_message->gsm.h.message_type) { case OGS_NAS_5GS_PDU_SESSION_RELEASE_COMPLETE: ogs_assert(true == ogs_sbi_send_http_status_no_content(stream)); ogs_assert(true == smf_sbi_send_sm_context_status_notify(sess)); OGS_FSM_TRAN(s, &smf_gsm_state_session_will_release); break; default: strerror = ogs_msprintf("Unknown message [%d]", nas_message->gsm.h.message_type); ogs_assert(strerror); ogs_error("%s", strerror); ogs_assert(true == ogs_sbi_server_send_error(stream, OGS_SBI_HTTP_STATUS_BAD_REQUEST, NULL, strerror, NULL)); ogs_free(strerror); } break; } } void smf_gsm_state_session_will_release(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: SMF_SESS_CLEAR(sess); break; case OGS_FSM_EXIT_SIG: break; default: ogs_error("Unknown event %s", smf_event_get_name(e)); break; } } void smf_gsm_state_exception(ogs_fsm_t *s, smf_event_t *e) { smf_ue_t *smf_ue = NULL; smf_sess_t *sess = NULL; ogs_assert(s); ogs_assert(e); smf_sm_debug(e); sess = e->sess; ogs_assert(sess); smf_ue = sess->smf_ue; ogs_assert(smf_ue); switch (e->id) { case OGS_FSM_ENTRY_SIG: ogs_error("[%s:%d] State machine exception", smf_ue->supi, sess->psi); SMF_SESS_CLEAR(sess); break; case OGS_FSM_EXIT_SIG: break; default: ogs_error("Unknown event %s", smf_event_get_name(e)); break; } } void smf_gsm_state_final(ogs_fsm_t *s, smf_event_t *e) { }