2019-07-11 12:53:54 +00:00
|
|
|
/*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
2019-06-01 09:52:38 +00:00
|
|
|
#include "s1ap-path.h"
|
2019-06-11 13:10:47 +00:00
|
|
|
#include "s1ap-build.h"
|
|
|
|
#include "esm-build.h"
|
|
|
|
#include "emm-build.h"
|
|
|
|
#include "nas-path.h"
|
|
|
|
#include "mme-event.h"
|
2019-07-20 12:28:36 +00:00
|
|
|
#include "mme-timer.h"
|
2019-06-11 13:10:47 +00:00
|
|
|
#include "mme-sm.h"
|
2017-08-10 08:02:53 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_to_enb(mme_ue_t *mme_ue, ogs_pkbuf_t *pkbuf)
|
2017-08-10 08:02:53 +00:00
|
|
|
{
|
2017-09-10 14:03:24 +00:00
|
|
|
enb_ue_t *enb_ue = NULL;
|
2017-08-10 08:02:53 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-10 14:03:24 +00:00
|
|
|
enb_ue = mme_ue->enb_ue;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(enb_ue);
|
2017-08-10 08:02:53 +00:00
|
|
|
|
2018-05-13 09:02:24 +00:00
|
|
|
return s1ap_send_to_enb_ue(enb_ue, pkbuf);
|
2017-08-10 08:02:53 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_emm_to_esm(
|
2017-09-07 06:56:31 +00:00
|
|
|
mme_ue_t *mme_ue, nas_esm_message_container_t *esm_message_container)
|
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_pkbuf_t *esmbuf = NULL;
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
|
|
|
ogs_assert(esm_message_container);
|
|
|
|
ogs_assert(esm_message_container->length);
|
2017-09-07 06:56:31 +00:00
|
|
|
|
|
|
|
/* The Packet Buffer(pkbuf_t) for NAS message MUST make a HEADROOM.
|
|
|
|
* When calculating AES_CMAC, we need to use the headroom of the packet. */
|
2019-04-27 14:54:30 +00:00
|
|
|
esmbuf = ogs_pkbuf_alloc(NULL, NAS_HEADROOM+esm_message_container->length);
|
|
|
|
ogs_pkbuf_reserve(esmbuf, NAS_HEADROOM);
|
|
|
|
ogs_pkbuf_put_data(esmbuf,
|
2017-09-07 06:56:31 +00:00
|
|
|
esm_message_container->buffer, esm_message_container->length);
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(s1ap_send_to_esm(mme_ue, esmbuf) == OGS_OK);
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2017-09-07 06:56:31 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_to_downlink_nas_transport(mme_ue_t *mme_ue, ogs_pkbuf_t *pkbuf)
|
2017-08-10 08:02:53 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *s1apbuf = NULL;
|
2017-08-10 08:02:53 +00:00
|
|
|
enb_ue_t *enb_ue = NULL;
|
|
|
|
|
2019-06-23 11:17:44 +00:00
|
|
|
ogs_assert(pkbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-08-10 08:02:53 +00:00
|
|
|
enb_ue = mme_ue->enb_ue;
|
2019-06-23 11:17:44 +00:00
|
|
|
if (!enb_ue) {
|
|
|
|
ogs_warn("S1 context has already been removed");
|
|
|
|
ogs_pkbuf_free(pkbuf);
|
2017-08-10 08:02:53 +00:00
|
|
|
|
2019-06-23 11:17:44 +00:00
|
|
|
} else {
|
|
|
|
rv = s1ap_build_downlink_nas_transport(&s1apbuf, enb_ue, pkbuf);
|
|
|
|
ogs_assert(rv == OGS_OK && s1apbuf);
|
2017-08-10 08:02:53 +00:00
|
|
|
|
2019-06-23 11:17:44 +00:00
|
|
|
rv = nas_send_to_enb(mme_ue, s1apbuf);
|
|
|
|
ogs_assert(rv == OGS_OK);
|
|
|
|
}
|
2017-09-10 14:03:24 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2017-08-10 08:02:53 +00:00
|
|
|
}
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_attach_accept(mme_ue_t *mme_ue)
|
2017-09-07 06:56:31 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
2017-09-07 06:56:31 +00:00
|
|
|
mme_sess_t *sess = NULL;
|
2017-09-11 05:01:11 +00:00
|
|
|
mme_bearer_t *bearer = NULL;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_pkbuf_t *s1apbuf = NULL;
|
|
|
|
ogs_pkbuf_t *esmbuf = NULL, *emmbuf = NULL;
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-07 06:56:31 +00:00
|
|
|
sess = mme_sess_first(mme_ue);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(sess);
|
|
|
|
ogs_assert(mme_sess_next(sess) == NULL);
|
2017-09-11 05:01:11 +00:00
|
|
|
bearer = mme_default_bearer_in_sess(sess);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(bearer);
|
|
|
|
ogs_assert(mme_bearer_next(bearer) == NULL);
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-07-20 13:30:53 +00:00
|
|
|
if (mme_ue->t3450.pkbuf) {
|
|
|
|
s1apbuf = mme_ue->t3450.pkbuf;
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-07-20 13:30:53 +00:00
|
|
|
} else {
|
|
|
|
rv = esm_build_activate_default_bearer_context_request(&esmbuf, sess);
|
|
|
|
ogs_assert(rv == OGS_OK && esmbuf);
|
|
|
|
|
|
|
|
rv = emm_build_attach_accept(&emmbuf, mme_ue, esmbuf);
|
|
|
|
ogs_assert(rv == OGS_OK && emmbuf);
|
|
|
|
|
|
|
|
rv = s1ap_build_initial_context_setup_request(&s1apbuf, mme_ue, emmbuf);
|
|
|
|
ogs_assert(rv == OGS_OK && s1apbuf);
|
|
|
|
}
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-07-20 13:30:53 +00:00
|
|
|
mme_ue->t3450.pkbuf = ogs_pkbuf_copy(s1apbuf);
|
|
|
|
ogs_timer_start(mme_ue->t3450.timer,
|
|
|
|
mme_timer_cfg(MME_TIMER_T3450)->duration);
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2017-09-10 14:03:24 +00:00
|
|
|
rv = nas_send_to_enb(mme_ue, s1apbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2017-09-07 06:56:31 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_attach_reject(mme_ue_t *mme_ue,
|
2018-01-19 07:29:21 +00:00
|
|
|
nas_emm_cause_t emm_cause, nas_esm_cause_t esm_cause)
|
2017-09-07 06:56:31 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
2017-09-07 06:56:31 +00:00
|
|
|
mme_sess_t *sess = NULL;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_pkbuf_t *esmbuf = NULL, *emmbuf = NULL;
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_debug("[EMM] Attach reject");
|
|
|
|
ogs_debug(" IMSI[%s] Cause[%d]", mme_ue->imsi_bcd, emm_cause);
|
2018-01-22 14:14:20 +00:00
|
|
|
|
2017-09-07 06:56:31 +00:00
|
|
|
sess = mme_sess_first(mme_ue);
|
2019-07-11 12:53:54 +00:00
|
|
|
if (sess) {
|
2017-09-08 09:50:27 +00:00
|
|
|
rv = esm_build_pdn_connectivity_reject(&esmbuf, sess, esm_cause);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && esmbuf);
|
2017-09-07 06:56:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
rv = emm_build_attach_reject(&emmbuf, emm_cause, esmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && emmbuf);
|
2017-09-11 09:41:14 +00:00
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, emmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2017-09-11 09:41:14 +00:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_identity_request(mme_ue_t *mme_ue)
|
2018-01-22 07:35:12 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *emmbuf = NULL;
|
2018-01-22 07:35:12 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2018-01-22 07:35:12 +00:00
|
|
|
|
2019-07-20 12:49:47 +00:00
|
|
|
ogs_debug("[EMM] Identity request");
|
|
|
|
|
|
|
|
if (mme_ue->t3470.pkbuf) {
|
|
|
|
emmbuf = mme_ue->t3470.pkbuf;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
rv = emm_build_identity_request(&emmbuf, mme_ue);
|
|
|
|
ogs_assert(rv == OGS_OK && emmbuf);
|
|
|
|
}
|
|
|
|
|
|
|
|
mme_ue->t3470.pkbuf = ogs_pkbuf_copy(emmbuf);
|
|
|
|
ogs_timer_start(mme_ue->t3470.timer,
|
|
|
|
mme_timer_cfg(MME_TIMER_T3470)->duration);
|
2018-01-22 07:35:12 +00:00
|
|
|
|
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, emmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2018-01-22 07:35:12 +00:00
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_authentication_request(
|
2017-09-11 09:41:14 +00:00
|
|
|
mme_ue_t *mme_ue, e_utran_vector_t *e_utran_vector)
|
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *emmbuf = NULL;
|
2017-09-11 09:41:14 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-11 09:41:14 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_debug("[EMM] Authentication request");
|
|
|
|
ogs_debug(" IMSI[%s]", mme_ue->imsi_bcd);
|
2018-01-22 14:14:20 +00:00
|
|
|
|
2019-07-20 12:28:36 +00:00
|
|
|
if (mme_ue->t3460.pkbuf) {
|
|
|
|
emmbuf = mme_ue->t3460.pkbuf;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
ogs_assert(e_utran_vector);
|
|
|
|
rv = emm_build_authentication_request(&emmbuf, e_utran_vector);
|
|
|
|
ogs_assert(rv == OGS_OK && emmbuf);
|
|
|
|
}
|
|
|
|
|
|
|
|
mme_ue->t3460.pkbuf = ogs_pkbuf_copy(emmbuf);
|
|
|
|
ogs_timer_start(mme_ue->t3460.timer,
|
|
|
|
mme_timer_cfg(MME_TIMER_T3460)->duration);
|
|
|
|
|
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, emmbuf);
|
|
|
|
ogs_assert(rv == OGS_OK);
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
int nas_send_security_mode_command(mme_ue_t *mme_ue)
|
|
|
|
{
|
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *emmbuf = NULL;
|
|
|
|
|
|
|
|
ogs_assert(mme_ue);
|
|
|
|
|
|
|
|
ogs_debug("[EMM] Security mode command");
|
|
|
|
ogs_debug(" IMSI[%s]", mme_ue->imsi_bcd);
|
|
|
|
|
|
|
|
if (mme_ue->t3460.pkbuf) {
|
|
|
|
emmbuf = mme_ue->t3460.pkbuf;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
rv = emm_build_security_mode_command(&emmbuf, mme_ue);
|
|
|
|
ogs_assert(rv == OGS_OK && emmbuf);
|
|
|
|
}
|
|
|
|
|
|
|
|
mme_ue->t3460.pkbuf = ogs_pkbuf_copy(emmbuf);
|
|
|
|
ogs_timer_start(mme_ue->t3460.timer,
|
|
|
|
mme_timer_cfg(MME_TIMER_T3460)->duration);
|
2017-09-11 09:41:14 +00:00
|
|
|
|
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, emmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-11 09:41:14 +00:00
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_authentication_reject(mme_ue_t *mme_ue)
|
2017-09-11 09:41:14 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *emmbuf = NULL;
|
2017-09-11 09:41:14 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-11 09:41:14 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_debug("[EMM] Authentication reject");
|
|
|
|
ogs_debug(" IMSI[%s]", mme_ue->imsi_bcd);
|
2018-01-22 14:14:20 +00:00
|
|
|
|
2017-09-11 09:41:14 +00:00
|
|
|
rv = emm_build_authentication_reject(&emmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && emmbuf);
|
2017-09-11 09:41:14 +00:00
|
|
|
|
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, emmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2017-09-07 06:56:31 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_detach_accept(mme_ue_t *mme_ue)
|
2017-09-10 14:03:24 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
2017-09-10 14:03:24 +00:00
|
|
|
enb_ue_t *enb_ue = NULL;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_pkbuf_t *emmbuf = NULL;
|
2017-09-10 14:03:24 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-10 14:03:24 +00:00
|
|
|
enb_ue = mme_ue->enb_ue;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(enb_ue);
|
2017-09-10 14:03:24 +00:00
|
|
|
|
|
|
|
/* reply with detach accept */
|
2019-07-11 12:53:54 +00:00
|
|
|
if (mme_ue->nas_eps.detach.switch_off == 0) {
|
2017-09-10 14:03:24 +00:00
|
|
|
rv = emm_build_detach_accept(&emmbuf, mme_ue);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && emmbuf);
|
2017-09-10 14:03:24 +00:00
|
|
|
|
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, emmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-10 14:03:24 +00:00
|
|
|
}
|
|
|
|
|
2018-01-21 11:31:28 +00:00
|
|
|
rv = s1ap_send_ue_context_release_command(enb_ue,
|
2018-03-05 14:01:07 +00:00
|
|
|
S1AP_Cause_PR_nas, S1AP_CauseNas_detach,
|
2018-02-04 08:05:30 +00:00
|
|
|
S1AP_UE_CTX_REL_S1_NORMAL_RELEASE, 0);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-10 14:03:24 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2017-09-10 14:03:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_pdn_connectivity_reject(
|
2017-09-08 09:50:27 +00:00
|
|
|
mme_sess_t *sess, nas_esm_cause_t esm_cause)
|
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
2017-09-08 09:50:27 +00:00
|
|
|
mme_ue_t *mme_ue;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_pkbuf_t *esmbuf = NULL;
|
2017-09-08 09:50:27 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(sess);
|
2017-09-08 09:50:27 +00:00
|
|
|
mme_ue = sess->mme_ue;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-08 09:50:27 +00:00
|
|
|
|
2019-05-26 03:57:29 +00:00
|
|
|
if (OGS_FSM_CHECK(&mme_ue->sm, emm_state_registered)) {
|
|
|
|
rv = esm_build_pdn_connectivity_reject(&esmbuf, sess, esm_cause);
|
|
|
|
ogs_assert(rv == OGS_OK && esmbuf);
|
2017-09-08 09:50:27 +00:00
|
|
|
|
2019-05-26 03:57:29 +00:00
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, esmbuf);
|
|
|
|
ogs_assert(rv == OGS_OK);
|
|
|
|
} else {
|
|
|
|
/* During the UE-attach process, we'll send Attach-Reject
|
|
|
|
* with pyggybacking PDN-connectivity-Reject */
|
|
|
|
rv = nas_send_attach_reject(mme_ue,
|
|
|
|
EMM_CAUSE_EPS_SERVICES_AND_NON_EPS_SERVICES_NOT_ALLOWED, esm_cause);
|
|
|
|
ogs_assert(rv == OGS_OK);
|
|
|
|
}
|
2017-09-08 09:50:27 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2017-09-08 09:50:27 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_esm_information_request(mme_bearer_t *bearer)
|
2017-09-23 15:59:51 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
2017-09-23 15:59:51 +00:00
|
|
|
mme_ue_t *mme_ue = NULL;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_pkbuf_t *esmbuf = NULL;
|
2017-09-23 15:59:51 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(bearer);
|
2017-09-23 15:59:51 +00:00
|
|
|
mme_ue = bearer->mme_ue;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-23 15:59:51 +00:00
|
|
|
|
2019-07-20 14:06:54 +00:00
|
|
|
if (bearer->t3489.pkbuf) {
|
|
|
|
esmbuf = bearer->t3489.pkbuf;
|
|
|
|
} else {
|
|
|
|
rv = esm_build_information_request(&esmbuf, bearer);
|
|
|
|
ogs_assert(rv == OGS_OK && esmbuf);
|
|
|
|
}
|
|
|
|
|
|
|
|
bearer->t3489.pkbuf = ogs_pkbuf_copy(esmbuf);
|
|
|
|
ogs_timer_start(bearer->t3489.timer,
|
|
|
|
mme_timer_cfg(MME_TIMER_T3489)->duration);
|
2017-09-23 15:59:51 +00:00
|
|
|
|
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, esmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-23 15:59:51 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2017-09-23 15:59:51 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_activate_default_bearer_context_request(mme_bearer_t *bearer)
|
2017-09-10 14:03:24 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *s1apbuf = NULL;
|
|
|
|
ogs_pkbuf_t *esmbuf = NULL;
|
2017-09-10 14:03:24 +00:00
|
|
|
mme_sess_t *sess = NULL;
|
|
|
|
mme_ue_t *mme_ue = NULL;
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(bearer);
|
2017-09-10 14:03:24 +00:00
|
|
|
sess = bearer->sess;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(sess);
|
2017-09-10 14:03:24 +00:00
|
|
|
mme_ue = bearer->mme_ue;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-10 14:03:24 +00:00
|
|
|
|
|
|
|
rv = esm_build_activate_default_bearer_context_request(&esmbuf, sess);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && esmbuf);
|
2017-09-10 14:03:24 +00:00
|
|
|
|
|
|
|
rv = s1ap_build_e_rab_setup_request(&s1apbuf, bearer, esmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && s1apbuf);
|
2017-09-10 14:03:24 +00:00
|
|
|
|
|
|
|
rv = nas_send_to_enb(mme_ue, s1apbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-10 14:03:24 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2017-09-10 14:03:24 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_activate_dedicated_bearer_context_request(
|
2017-09-10 14:03:24 +00:00
|
|
|
mme_bearer_t *bearer)
|
2017-09-07 06:56:31 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *s1apbuf = NULL;
|
|
|
|
ogs_pkbuf_t *esmbuf = NULL;
|
2017-09-10 14:03:24 +00:00
|
|
|
mme_ue_t *mme_ue = NULL;
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(bearer);
|
2017-09-10 14:03:24 +00:00
|
|
|
mme_ue = bearer->mme_ue;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2017-09-07 12:28:12 +00:00
|
|
|
rv = esm_build_activate_dedicated_bearer_context_request(&esmbuf, bearer);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && esmbuf);
|
2017-09-07 06:56:31 +00:00
|
|
|
|
|
|
|
rv = s1ap_build_e_rab_setup_request(&s1apbuf, bearer, esmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && s1apbuf);
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2017-09-10 14:03:24 +00:00
|
|
|
rv = nas_send_to_enb(mme_ue, s1apbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-07 06:56:31 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2017-09-07 06:56:31 +00:00
|
|
|
}
|
2017-09-07 14:41:05 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_activate_all_dedicated_bearers(mme_bearer_t *default_bearer)
|
2018-02-01 12:38:59 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
2018-02-01 12:38:59 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(default_bearer);
|
2018-02-01 12:38:59 +00:00
|
|
|
|
|
|
|
mme_bearer_t *dedicated_bearer = mme_bearer_next(default_bearer);
|
2019-07-11 12:53:54 +00:00
|
|
|
while (dedicated_bearer) {
|
2018-02-01 12:38:59 +00:00
|
|
|
rv = nas_send_activate_dedicated_bearer_context_request(
|
|
|
|
dedicated_bearer);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2018-02-01 12:38:59 +00:00
|
|
|
|
|
|
|
dedicated_bearer = mme_bearer_next(dedicated_bearer);
|
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2018-02-01 12:38:59 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_modify_bearer_context_request(
|
2018-01-17 06:41:45 +00:00
|
|
|
mme_bearer_t *bearer, int qos_presence, int tft_presence)
|
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *s1apbuf = NULL;
|
|
|
|
ogs_pkbuf_t *esmbuf = NULL;
|
2018-01-17 06:41:45 +00:00
|
|
|
mme_ue_t *mme_ue = NULL;
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(bearer);
|
2018-01-17 06:41:45 +00:00
|
|
|
mme_ue = bearer->mme_ue;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2018-01-17 06:41:45 +00:00
|
|
|
|
|
|
|
rv = esm_build_modify_bearer_context_request(
|
|
|
|
&esmbuf, bearer, qos_presence, tft_presence);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && esmbuf);
|
2018-01-17 06:41:45 +00:00
|
|
|
|
2019-07-11 12:53:54 +00:00
|
|
|
if (qos_presence == 1) {
|
2018-01-17 06:41:45 +00:00
|
|
|
rv = s1ap_build_e_rab_modify_request(&s1apbuf, bearer, esmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && s1apbuf);
|
2018-01-17 06:41:45 +00:00
|
|
|
|
|
|
|
rv = nas_send_to_enb(mme_ue, s1apbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2019-07-11 12:53:54 +00:00
|
|
|
} else {
|
2018-01-17 06:41:45 +00:00
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, esmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2018-01-17 06:41:45 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2018-01-17 06:41:45 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_deactivate_bearer_context_request(mme_bearer_t *bearer)
|
2017-09-07 14:41:05 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *s1apbuf = NULL;
|
|
|
|
ogs_pkbuf_t *esmbuf = NULL;
|
2017-09-10 14:03:24 +00:00
|
|
|
mme_ue_t *mme_ue = NULL;
|
2017-09-07 14:41:05 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(bearer);
|
2017-09-10 14:03:24 +00:00
|
|
|
mme_ue = bearer->mme_ue;
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-07 14:41:05 +00:00
|
|
|
|
|
|
|
rv = esm_build_deactivate_bearer_context_request(
|
|
|
|
&esmbuf, bearer, ESM_CAUSE_REGULAR_DEACTIVATION);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && esmbuf);
|
2017-09-07 14:41:05 +00:00
|
|
|
|
2018-01-21 11:31:28 +00:00
|
|
|
rv = s1ap_build_e_rab_release_command(&s1apbuf, bearer, esmbuf,
|
2018-03-05 14:01:07 +00:00
|
|
|
S1AP_Cause_PR_nas, S1AP_CauseNas_normal_release);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && s1apbuf);
|
2017-09-07 14:41:05 +00:00
|
|
|
|
2017-09-10 14:03:24 +00:00
|
|
|
rv = nas_send_to_enb(mme_ue, s1apbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-07 14:41:05 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2017-09-07 14:41:05 +00:00
|
|
|
}
|
2017-09-08 07:46:37 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_tau_accept(
|
2018-03-05 14:01:07 +00:00
|
|
|
mme_ue_t *mme_ue, S1AP_ProcedureCode_t procedureCode)
|
2017-09-08 07:46:37 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *emmbuf = NULL;
|
2017-09-08 07:46:37 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-08 07:46:37 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_debug("[EMM] Tracking area update accept");
|
|
|
|
ogs_debug(" IMSI[%s]", mme_ue->imsi_bcd);
|
2018-02-03 02:48:15 +00:00
|
|
|
|
2018-02-02 04:46:49 +00:00
|
|
|
rv = emm_build_tau_accept(&emmbuf, mme_ue);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2018-02-02 04:46:49 +00:00
|
|
|
|
2019-07-11 12:53:54 +00:00
|
|
|
if (procedureCode == S1AP_ProcedureCode_id_InitialContextSetup) {
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_pkbuf_t *s1apbuf = NULL;
|
2018-02-04 07:01:09 +00:00
|
|
|
rv = s1ap_build_initial_context_setup_request(&s1apbuf, mme_ue, emmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK && s1apbuf);
|
2017-09-10 14:03:24 +00:00
|
|
|
|
2018-02-04 07:01:09 +00:00
|
|
|
rv = nas_send_to_enb(mme_ue, s1apbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2019-07-11 12:53:54 +00:00
|
|
|
} else if (procedureCode == S1AP_ProcedureCode_id_downlinkNASTransport) {
|
2018-02-04 07:01:09 +00:00
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, emmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2019-07-11 12:53:54 +00:00
|
|
|
} else
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert_if_reached();
|
2018-02-04 07:01:09 +00:00
|
|
|
|
|
|
|
return rv;
|
2017-09-08 07:46:37 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_tau_reject(mme_ue_t *mme_ue, nas_emm_cause_t emm_cause)
|
2017-09-08 07:46:37 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *emmbuf = NULL;
|
2017-09-08 07:46:37 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-08 07:46:37 +00:00
|
|
|
|
|
|
|
/* Build TAU reject */
|
2018-01-19 07:29:21 +00:00
|
|
|
rv = emm_build_tau_reject(&emmbuf, emm_cause, mme_ue);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-08 22:56:42 +00:00
|
|
|
|
2017-09-10 14:03:24 +00:00
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, emmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-08 22:56:42 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2017-09-08 22:56:42 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
int nas_send_service_reject(mme_ue_t *mme_ue, nas_emm_cause_t emm_cause)
|
2017-09-08 22:56:42 +00:00
|
|
|
{
|
2019-04-27 14:54:30 +00:00
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *emmbuf = NULL;
|
2017-09-08 22:56:42 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(mme_ue);
|
2017-09-08 22:56:42 +00:00
|
|
|
|
2017-09-11 05:18:18 +00:00
|
|
|
/* Build Service Reject */
|
2018-01-19 07:29:21 +00:00
|
|
|
rv = emm_build_service_reject(&emmbuf, emm_cause, mme_ue);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-08 07:46:37 +00:00
|
|
|
|
2017-09-10 14:03:24 +00:00
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, emmbuf);
|
2019-04-27 14:54:30 +00:00
|
|
|
ogs_assert(rv == OGS_OK);
|
2017-09-08 07:46:37 +00:00
|
|
|
|
2019-04-27 14:54:30 +00:00
|
|
|
return OGS_OK;
|
2017-09-08 07:46:37 +00:00
|
|
|
}
|
2019-07-11 09:14:32 +00:00
|
|
|
|
|
|
|
int nas_send_cs_service_notification(mme_ue_t *mme_ue)
|
|
|
|
{
|
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *emmbuf = NULL;
|
|
|
|
|
|
|
|
ogs_assert(mme_ue);
|
|
|
|
|
|
|
|
ogs_debug("[EMM] CS Service Notification");
|
|
|
|
ogs_debug(" IMSI[%s]", mme_ue->imsi_bcd);
|
|
|
|
|
|
|
|
rv = emm_build_cs_service_notification(&emmbuf, mme_ue);
|
|
|
|
ogs_assert(rv == OGS_OK && emmbuf);
|
|
|
|
|
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, emmbuf);
|
|
|
|
ogs_assert(rv == OGS_OK);
|
|
|
|
|
|
|
|
return OGS_OK;
|
|
|
|
}
|
2019-07-13 13:52:50 +00:00
|
|
|
|
|
|
|
int nas_send_downlink_nas_transport(
|
|
|
|
mme_ue_t *mme_ue, uint8_t *buffer, uint8_t length)
|
|
|
|
{
|
|
|
|
int rv;
|
|
|
|
ogs_pkbuf_t *emmbuf = NULL;
|
|
|
|
|
|
|
|
ogs_assert(mme_ue);
|
|
|
|
ogs_assert(buffer);
|
|
|
|
ogs_assert(length);
|
|
|
|
|
|
|
|
ogs_debug("[EMM] Downlink NAS transport");
|
|
|
|
ogs_debug(" IMSI[%s]", mme_ue->imsi_bcd);
|
|
|
|
|
|
|
|
rv = emm_build_downlink_nas_transport(&emmbuf, mme_ue, buffer, length);
|
|
|
|
ogs_assert(rv == OGS_OK && emmbuf);
|
|
|
|
|
|
|
|
rv = nas_send_to_downlink_nas_transport(mme_ue, emmbuf);
|
|
|
|
ogs_assert(rv == OGS_OK);
|
|
|
|
|
|
|
|
return OGS_OK;
|
|
|
|
}
|