open5gs/src/amf/nudm-handler.c

234 lines
8.9 KiB
C

/*
* Copyright (C) 2019 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 "nudm-handler.h"
#include "sbi-path.h"
#include "nas-path.h"
int amf_nudm_sdm_handle_provisioned(
amf_ue_t *amf_ue, ogs_sbi_message_t *recvmsg)
{
int i;
ogs_assert(amf_ue);
ogs_assert(recvmsg);
SWITCH(recvmsg->h.resource.component[1])
CASE(OGS_SBI_RESOURCE_NAME_AM_DATA)
if (recvmsg->AccessAndMobilitySubscriptionData) {
OpenAPI_list_t *gpsiList =
recvmsg->AccessAndMobilitySubscriptionData->gpsis;
OpenAPI_ambr_rm_t *SubscribedUeAmbr =
recvmsg->AccessAndMobilitySubscriptionData->subscribed_ue_ambr;
OpenAPI_nssai_t *NSSAI =
recvmsg->AccessAndMobilitySubscriptionData->nssai;
OpenAPI_lnode_t *node = NULL;
/* Clear MSISDN */
for (i = 0; i < amf_ue->num_of_msisdn; i++) {
ogs_assert(amf_ue->msisdn[i]);
ogs_free(amf_ue->msisdn[i]);
}
amf_ue->num_of_msisdn = 0;
if (gpsiList) {
OpenAPI_list_for_each(gpsiList, node) {
if (node->data) {
char *gpsi = NULL;
gpsi = ogs_id_get_type(node->data);
ogs_assert(gpsi);
if (strncmp(gpsi, OGS_ID_GPSI_TYPE_MSISDN,
strlen(OGS_ID_GPSI_TYPE_MSISDN)) != 0) {
ogs_error("Unknown GPSI Type [%s]", gpsi);
} else {
amf_ue->msisdn[amf_ue->num_of_msisdn] =
ogs_id_get_value(node->data);
ogs_assert(amf_ue->msisdn[amf_ue->num_of_msisdn]);
amf_ue->num_of_msisdn++;
}
ogs_free(gpsi);
}
}
}
/* Clear Subscribed-UE-AMBR */
amf_ue->ue_ambr.uplink = 0;
amf_ue->ue_ambr.downlink = 0;
if (SubscribedUeAmbr) {
amf_ue->ue_ambr.uplink =
ogs_sbi_bitrate_from_string(SubscribedUeAmbr->uplink);
amf_ue->ue_ambr.downlink =
ogs_sbi_bitrate_from_string(SubscribedUeAmbr->downlink);
}
if (NSSAI) {
OpenAPI_list_t *DefaultSingleNssaiList = NULL;
OpenAPI_list_t *SingleNssaiList = NULL;
/* Clear SubscribedInfo */
amf_clear_subscribed_info(amf_ue);
DefaultSingleNssaiList = NSSAI->default_single_nssais;
if (DefaultSingleNssaiList) {
OpenAPI_list_for_each(DefaultSingleNssaiList, node) {
OpenAPI_snssai_t *Snssai = node->data;
ogs_slice_data_t *slice =
&amf_ue->slice[amf_ue->num_of_slice];
if (Snssai) {
slice->s_nssai.sst = Snssai->sst;
slice->s_nssai.sd =
ogs_s_nssai_sd_from_string(Snssai->sd);
}
/* DEFAULT S-NSSAI */
slice->default_indicator = true;
amf_ue->num_of_slice++;
}
SingleNssaiList = NSSAI->single_nssais;
if (SingleNssaiList) {
OpenAPI_list_for_each(SingleNssaiList, node) {
OpenAPI_snssai_t *Snssai = node->data;
ogs_slice_data_t *slice =
&amf_ue->slice[amf_ue->num_of_slice];
if (Snssai) {
slice->s_nssai.sst = Snssai->sst;
slice->s_nssai.sd =
ogs_s_nssai_sd_from_string(Snssai->sd);
}
/* Non default S-NSSAI */
slice->default_indicator = false;
amf_ue->num_of_slice++;
}
}
}
}
}
if (amf_update_allowed_nssai(amf_ue) == false) {
ogs_error("No Allowed-NSSAI");
return OGS_ERROR;
}
ogs_assert(true ==
amf_ue_sbi_discover_and_send(
OGS_SBI_SERVICE_TYPE_NUDM_SDM, NULL,
amf_nudm_sdm_build_get,
amf_ue, (char *)OGS_SBI_RESOURCE_NAME_SMF_SELECT_DATA));
break;
CASE(OGS_SBI_RESOURCE_NAME_SMF_SELECT_DATA)
if (recvmsg->SmfSelectionSubscriptionData) {
OpenAPI_list_t *SubscribedSnssaiInfoList = NULL;
OpenAPI_map_t *SubscribedSnssaiInfoMap = NULL;
OpenAPI_snssai_info_t *SubscribedSnssaiInfo = NULL;
OpenAPI_list_t *DnnInfoList = NULL;
OpenAPI_dnn_info_t *DnnInfo = NULL;
OpenAPI_lnode_t *node = NULL, *node2 = NULL;
SubscribedSnssaiInfoList = recvmsg->
SmfSelectionSubscriptionData->subscribed_snssai_infos;
if (SubscribedSnssaiInfoList) {
OpenAPI_list_for_each(SubscribedSnssaiInfoList, node) {
SubscribedSnssaiInfoMap = node->data;
if (SubscribedSnssaiInfoMap &&
SubscribedSnssaiInfoMap->key) {
ogs_slice_data_t *slice = NULL;
ogs_s_nssai_t s_nssai;
bool rc = ogs_sbi_s_nssai_from_string(
&s_nssai, SubscribedSnssaiInfoMap->key);
if (rc == false) {
ogs_error("Invalid S-NSSAI format [%s]",
SubscribedSnssaiInfoMap->key);
continue;
}
slice = ogs_slice_find_by_s_nssai(
amf_ue->slice, amf_ue->num_of_slice,
&s_nssai);
if (!slice) {
ogs_error("Cannt find S-NSSAI[SST:%d SD:0x%x]",
s_nssai.sst, s_nssai.sd.v);
continue;
}
SubscribedSnssaiInfo = SubscribedSnssaiInfoMap->value;
if (SubscribedSnssaiInfo) {
DnnInfoList = SubscribedSnssaiInfo->dnn_infos;
if (DnnInfoList) {
OpenAPI_list_for_each(DnnInfoList, node2) {
DnnInfo = node2->data;
if (DnnInfo) {
ogs_session_t *session =
&slice->session
[slice->num_of_session];
session->name =
ogs_strdup(DnnInfo->dnn);
ogs_assert(session->name);
if (DnnInfo->is_default_dnn_indicator ==
true) {
session->default_dnn_indicator =
DnnInfo->default_dnn_indicator;
}
slice->num_of_session++;
}
}
}
}
}
}
}
}
ogs_assert(true ==
amf_ue_sbi_discover_and_send(
OGS_SBI_SERVICE_TYPE_NUDM_SDM, NULL,
amf_nudm_sdm_build_get,
amf_ue, (char *)OGS_SBI_RESOURCE_NAME_UE_CONTEXT_IN_SMF_DATA));
break;
CASE(OGS_SBI_RESOURCE_NAME_UE_CONTEXT_IN_SMF_DATA)
ogs_assert(true ==
amf_ue_sbi_discover_and_send(
OGS_SBI_SERVICE_TYPE_NPCF_AM_POLICY_CONTROL, NULL,
amf_npcf_am_policy_control_build_create, amf_ue, NULL));
break;
DEFAULT
END
return OGS_OK;
}