/* * 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 "sbi-path.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 "pfcp-path.h" #include "ngap-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, 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; 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_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 smf_nsmf_handle_create_sm_context(sess, stream, sbi_message); break; 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)); 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_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) != true) { OGS_FSM_TRAN(s, smf_gsm_state_session_will_release); } 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]) { 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); break; } smf_npcf_smpolicycontrol_handle_create( sess, stream, state, sbi_message); } 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); } smf_npcf_smpolicycontrol_handle_delete( sess, stream, state, sbi_message); 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); 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); 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_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; 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; case OGS_NAS_5GS_PDU_SESSION_RELEASE_COMPLETE: ogs_assert(true == ogs_sbi_send_http_status_no_content(stream)); /* * Race condition for PDU session release complete * - CLIENT : /nsmf-pdusession/v1/sm-contexts/{smContextRef}/modify * - SERVER : /namf-callback/v1/{supi}/sm-context-status/{psi}) * * ogs_sbi_send_http_status_no_content(stream); * 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)); 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_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 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_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: 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_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: ogs_error("Unknown event %s", smf_event_get_name(e)); break; } }