open5gs/src/mme/mme_gtp_path.c

432 lines
9.9 KiB
C
Raw Normal View History

2018-01-04 11:38:22 +00:00
#include "gtp/gtp_node.h"
#include "gtp/gtp_path.h"
#include "gtp/gtp_xact.h"
2017-11-30 11:13:15 +00:00
2017-04-04 01:49:19 +00:00
#include "mme_event.h"
2017-08-31 02:05:00 +00:00
#include "mme_gtp_path.h"
2017-09-07 06:56:31 +00:00
#include "mme_s11_build.h"
#include "mme_sm.h"
2017-03-23 14:05:40 +00:00
2019-04-27 14:54:30 +00:00
static void _gtpv2_c_recv_cb(short when, ogs_socket_t fd, void *data)
2017-03-24 04:19:36 +00:00
{
2019-04-27 14:54:30 +00:00
int rv;
mme_event_t *e = NULL;
ogs_pkbuf_t *pkbuf = NULL;
2017-03-26 15:48:33 +00:00
2019-04-27 14:54:30 +00:00
ogs_assert(fd != INVALID_SOCKET);
2017-03-24 04:19:36 +00:00
2019-04-27 14:54:30 +00:00
rv = gtp_recv(fd, &pkbuf);
if (rv != OGS_OK)
2017-03-24 04:19:36 +00:00
{
2019-04-27 14:54:30 +00:00
ogs_log_message(OGS_LOG_WARN, ogs_socket_errno, "gtp_recv() failed");
return;
2017-03-24 04:19:36 +00:00
}
2019-04-27 14:54:30 +00:00
e = mme_event_new(MME_EVT_S11_MESSAGE);
ogs_assert(e);
e->pkbuf = pkbuf;
rv = ogs_queue_push(mme_self()->queue, e);
if (rv != OGS_OK) {
ogs_error("ogs_queue_push() failed:%d", (int)rv);
ogs_pkbuf_free(e->pkbuf);
mme_event_free(e);
}
2017-03-24 04:19:36 +00:00
}
2019-04-27 14:54:30 +00:00
static ogs_sockaddr_t *pgw_addr_find_by_family(ogs_list_t *list, int family)
2017-03-23 14:05:40 +00:00
{
mme_pgw_t *pgw = NULL;
2019-04-27 14:54:30 +00:00
ogs_assert(list);
2017-12-01 13:04:51 +00:00
2019-04-27 14:54:30 +00:00
ogs_list_for_each(list, pgw)
2017-12-03 12:34:39 +00:00
{
2019-04-27 14:54:30 +00:00
ogs_assert(pgw->gnode);
ogs_sockaddr_t *addr = pgw->gnode->sa_list;
2017-12-03 12:34:39 +00:00
while(addr)
{
2017-12-07 04:27:17 +00:00
if (addr->c_sa_family == family)
2017-12-03 12:34:39 +00:00
{
2017-12-07 04:27:17 +00:00
return addr;
2017-12-03 12:34:39 +00:00
}
addr = addr->next;
}
}
2017-12-07 04:27:17 +00:00
return NULL;
}
2017-12-03 12:34:39 +00:00
2019-04-27 14:54:30 +00:00
int mme_gtp_open()
2017-12-07 04:27:17 +00:00
{
2019-04-27 14:54:30 +00:00
int rv;
ogs_socknode_t *snode = NULL;
ogs_sock_t *sock = NULL;
mme_sgw_t *sgw = NULL;
2017-12-07 04:27:17 +00:00
2019-04-27 14:54:30 +00:00
ogs_list_for_each(&mme_self()->gtpc_list, snode)
{
rv = gtp_server(snode);
ogs_assert(rv == OGS_OK);
sock = snode->sock;
ogs_assert(sock);
snode->poll = ogs_pollset_add(mme_self()->pollset,
OGS_POLLIN, sock->fd, _gtpv2_c_recv_cb, NULL);
ogs_assert(snode->poll);
}
ogs_list_for_each(&mme_self()->gtpc_list6, snode)
{
rv = gtp_server(snode);
ogs_assert(rv == OGS_OK);
sock = snode->sock;
ogs_assert(sock);
snode->poll = ogs_pollset_add(mme_self()->pollset,
OGS_POLLIN, sock->fd, _gtpv2_c_recv_cb, NULL);
ogs_assert(snode->poll);
}
2017-12-07 04:27:17 +00:00
2018-01-09 07:37:05 +00:00
mme_self()->gtpc_sock = gtp_local_sock_first(&mme_self()->gtpc_list);
mme_self()->gtpc_sock6 = gtp_local_sock_first(&mme_self()->gtpc_list6);
2017-12-07 04:27:17 +00:00
mme_self()->gtpc_addr = gtp_local_addr_first(&mme_self()->gtpc_list);
mme_self()->gtpc_addr6 = gtp_local_addr_first(&mme_self()->gtpc_list6);
2019-04-27 14:54:30 +00:00
ogs_assert(mme_self()->gtpc_addr || mme_self()->gtpc_addr6);
2017-12-07 04:27:17 +00:00
mme_self()->pgw_addr = pgw_addr_find_by_family(
2017-12-07 04:27:17 +00:00
&mme_self()->pgw_list, AF_INET);
mme_self()->pgw_addr6 = pgw_addr_find_by_family(
2017-12-07 04:27:17 +00:00
&mme_self()->pgw_list, AF_INET6);
2019-04-27 14:54:30 +00:00
ogs_assert(mme_self()->pgw_addr || mme_self()->pgw_addr6);
2017-12-03 12:34:39 +00:00
2019-04-27 14:54:30 +00:00
ogs_list_for_each(&mme_self()->sgw_list, sgw)
2018-01-09 07:37:05 +00:00
{
2019-04-27 14:54:30 +00:00
rv = gtp_connect(
mme_self()->gtpc_sock, mme_self()->gtpc_sock6, sgw->gnode);
ogs_assert(rv == OGS_OK);
2018-01-09 07:37:05 +00:00
}
2019-04-27 14:54:30 +00:00
return OGS_OK;
2017-12-02 05:17:32 +00:00
}
2019-04-27 14:54:30 +00:00
int mme_gtp_close()
2017-03-23 14:05:40 +00:00
{
2019-04-27 14:54:30 +00:00
ogs_socknode_t *snode = NULL;
ogs_list_for_each(&mme_self()->gtpc_list, snode)
{
ogs_pollset_remove(snode->poll);
ogs_sock_destroy(snode->sock);
}
ogs_list_for_each(&mme_self()->gtpc_list6, snode)
{
ogs_pollset_remove(snode->poll);
ogs_sock_destroy(snode->sock);
}
2017-12-01 13:04:51 +00:00
2019-04-27 14:54:30 +00:00
return OGS_OK;
2017-03-23 14:05:40 +00:00
}
2017-09-07 06:56:31 +00:00
2019-04-27 14:54:30 +00:00
int mme_gtp_send_create_session_request(mme_sess_t *sess)
2017-09-08 07:46:37 +00:00
{
2019-04-27 14:54:30 +00:00
int rv;
2017-09-08 07:46:37 +00:00
gtp_header_t h;
2019-04-27 14:54:30 +00:00
ogs_pkbuf_t *pkbuf = NULL;
2017-09-08 07:46:37 +00:00
gtp_xact_t *xact = NULL;
mme_ue_t *mme_ue = NULL;
mme_ue = sess->mme_ue;
2019-04-27 14:54:30 +00:00
ogs_assert(mme_ue);
2017-09-08 07:46:37 +00:00
memset(&h, 0, sizeof(gtp_header_t));
h.type = GTP_CREATE_SESSION_REQUEST_TYPE;
h.teid = mme_ue->sgw_s11_teid;
rv = mme_s11_build_create_session_request(&pkbuf, h.type, sess);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2017-09-08 07:46:37 +00:00
2017-12-06 08:58:38 +00:00
xact = gtp_xact_local_create(mme_ue->gnode, &h, pkbuf);
2019-04-27 14:54:30 +00:00
ogs_assert(xact);
2017-09-08 07:46:37 +00:00
rv = gtp_xact_commit(xact);
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-04-27 14:54:30 +00:00
int mme_gtp_send_modify_bearer_request(
2017-09-14 08:29:34 +00:00
mme_bearer_t *bearer, int uli_presence)
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
gtp_xact_t *xact = NULL;
mme_ue_t *mme_ue = NULL;
gtp_header_t h;
2019-04-27 14:54:30 +00:00
ogs_pkbuf_t *pkbuf = NULL;
2017-09-07 06:56:31 +00:00
2019-04-27 14:54:30 +00:00
ogs_assert(bearer);
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
memset(&h, 0, sizeof(gtp_header_t));
h.type = GTP_MODIFY_BEARER_REQUEST_TYPE;
h.teid = mme_ue->sgw_s11_teid;
2017-09-14 08:29:34 +00:00
rv = mme_s11_build_modify_bearer_request(
&pkbuf, h.type, bearer, uli_presence);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2017-09-07 06:56:31 +00:00
2017-12-06 08:58:38 +00:00
xact = gtp_xact_local_create(mme_ue->gnode, &h, pkbuf);
2019-04-27 14:54:30 +00:00
ogs_assert(xact);
2017-09-07 06:56:31 +00:00
rv = gtp_xact_commit(xact);
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 mme_gtp_send_delete_session_request(mme_sess_t *sess)
2017-09-07 13:22:42 +00:00
{
2019-04-27 14:54:30 +00:00
int rv;
ogs_pkbuf_t *s11buf = NULL;
2017-09-07 13:22:42 +00:00
gtp_header_t h;
gtp_xact_t *xact = NULL;
mme_ue_t *mme_ue = NULL;
2019-04-27 14:54:30 +00:00
ogs_assert(sess);
2017-09-07 13:22:42 +00:00
mme_ue = sess->mme_ue;
2019-04-27 14:54:30 +00:00
ogs_assert(mme_ue);
2017-09-07 13:22:42 +00:00
memset(&h, 0, sizeof(gtp_header_t));
h.type = GTP_DELETE_SESSION_REQUEST_TYPE;
h.teid = mme_ue->sgw_s11_teid;
2017-09-14 08:29:34 +00:00
rv = mme_s11_build_delete_session_request(&s11buf, h.type, sess);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2017-09-07 13:22:42 +00:00
2017-12-06 08:58:38 +00:00
xact = gtp_xact_local_create(mme_ue->gnode, &h, s11buf);
2019-04-27 14:54:30 +00:00
ogs_assert(xact);
2017-09-07 13:22:42 +00:00
GTP_XACT_STORE_SESSION(xact, sess);
rv = gtp_xact_commit(xact);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2017-09-07 13:22:42 +00:00
2019-04-27 14:54:30 +00:00
return OGS_OK;
2017-09-07 13:22:42 +00:00
}
2019-04-27 14:54:30 +00:00
int mme_gtp_send_delete_all_sessions(mme_ue_t *mme_ue)
2017-09-07 14:51:06 +00:00
{
2019-04-27 14:54:30 +00:00
int rv;
2017-09-07 14:51:06 +00:00
mme_sess_t *sess = NULL, *next_sess = NULL;
2019-04-27 14:54:30 +00:00
ogs_assert(mme_ue);
2017-09-07 14:51:06 +00:00
sess = mme_sess_first(mme_ue);
while (sess != NULL)
{
next_sess = mme_sess_next(sess);
if (MME_HAVE_SGW_S1U_PATH(sess))
{
mme_bearer_t *bearer = mme_default_bearer_in_sess(sess);
2019-04-27 14:54:30 +00:00
ogs_assert(bearer);
2019-04-27 14:54:30 +00:00
if (bearer &&
OGS_FSM_CHECK(&bearer->sm, esm_state_pdn_will_disconnect))
{
2019-04-27 14:54:30 +00:00
ogs_warn("PDN will disconnect[EBI:%d]", bearer->ebi);
}
else
{
rv = mme_gtp_send_delete_session_request(sess);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
}
2017-09-07 14:51:06 +00:00
}
else
{
mme_sess_remove(sess);
}
sess = next_sess;
}
2019-04-27 14:54:30 +00:00
return OGS_OK;
2017-09-07 14:51:06 +00:00
}
2019-04-27 14:54:30 +00:00
int mme_gtp_send_create_bearer_response(mme_bearer_t *bearer)
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
gtp_xact_t *xact = NULL;
mme_ue_t *mme_ue = NULL;
gtp_header_t h;
2019-04-27 14:54:30 +00:00
ogs_pkbuf_t *pkbuf = NULL;
2017-09-07 06:56:31 +00:00
2019-04-27 14:54:30 +00:00
ogs_assert(bearer);
2017-09-07 06:56:31 +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
xact = bearer->xact;
2019-04-27 14:54:30 +00:00
ogs_assert(xact);
2017-09-07 06:56:31 +00:00
memset(&h, 0, sizeof(gtp_header_t));
h.type = GTP_CREATE_BEARER_RESPONSE_TYPE;
h.teid = mme_ue->sgw_s11_teid;
2017-09-14 08:29:34 +00:00
rv = mme_s11_build_create_bearer_response(&pkbuf, h.type, bearer);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2017-09-07 06:56:31 +00:00
rv = gtp_xact_update_tx(xact, &h, pkbuf);
2019-04-27 14:54:30 +00:00
ogs_assert(xact);
2017-09-07 06:56:31 +00:00
rv = gtp_xact_commit(xact);
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 10:44:08 +00:00
2019-04-27 14:54:30 +00:00
int mme_gtp_send_update_bearer_response(mme_bearer_t *bearer)
2018-01-17 07:01:11 +00:00
{
2019-04-27 14:54:30 +00:00
int rv;
2018-01-17 07:01:11 +00:00
gtp_xact_t *xact = NULL;
mme_ue_t *mme_ue = NULL;
gtp_header_t h;
2019-04-27 14:54:30 +00:00
ogs_pkbuf_t *pkbuf = NULL;
2018-01-17 07:01:11 +00:00
2019-04-27 14:54:30 +00:00
ogs_assert(bearer);
2018-01-17 07:01:11 +00:00
mme_ue = bearer->mme_ue;
2019-04-27 14:54:30 +00:00
ogs_assert(mme_ue);
2018-01-17 07:01:11 +00:00
xact = bearer->xact;
2019-04-27 14:54:30 +00:00
ogs_assert(xact);
2018-01-17 07:01:11 +00:00
memset(&h, 0, sizeof(gtp_header_t));
h.type = GTP_UPDATE_BEARER_RESPONSE_TYPE;
h.teid = mme_ue->sgw_s11_teid;
rv = mme_s11_build_update_bearer_response(&pkbuf, h.type, bearer);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2018-01-17 07:01:11 +00:00
rv = gtp_xact_update_tx(xact, &h, pkbuf);
2019-04-27 14:54:30 +00:00
ogs_assert(xact);
2018-01-17 07:01:11 +00:00
rv = gtp_xact_commit(xact);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2018-01-17 07:01:11 +00:00
2019-04-27 14:54:30 +00:00
return OGS_OK;
2018-01-17 07:01:11 +00:00
}
2019-04-27 14:54:30 +00:00
int mme_gtp_send_delete_bearer_response(mme_bearer_t *bearer)
2018-01-16 13:14:59 +00:00
{
2019-04-27 14:54:30 +00:00
int rv;
2018-01-16 13:14:59 +00:00
gtp_xact_t *xact = NULL;
mme_ue_t *mme_ue = NULL;
gtp_header_t h;
2019-04-27 14:54:30 +00:00
ogs_pkbuf_t *pkbuf = NULL;
2018-01-16 13:14:59 +00:00
2019-04-27 14:54:30 +00:00
ogs_assert(bearer);
2018-01-16 13:14:59 +00:00
mme_ue = bearer->mme_ue;
2019-04-27 14:54:30 +00:00
ogs_assert(mme_ue);
2018-01-16 13:14:59 +00:00
xact = bearer->xact;
2019-04-27 14:54:30 +00:00
ogs_assert(xact);
2018-01-16 13:14:59 +00:00
memset(&h, 0, sizeof(gtp_header_t));
h.type = GTP_DELETE_BEARER_RESPONSE_TYPE;
h.teid = mme_ue->sgw_s11_teid;
rv = mme_s11_build_delete_bearer_response(&pkbuf, h.type, bearer);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2018-01-16 13:14:59 +00:00
rv = gtp_xact_update_tx(xact, &h, pkbuf);
2019-04-27 14:54:30 +00:00
ogs_assert(xact);
2018-01-16 13:14:59 +00:00
rv = gtp_xact_commit(xact);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2018-01-16 13:14:59 +00:00
2019-04-27 14:54:30 +00:00
return OGS_OK;
2018-01-16 13:14:59 +00:00
}
2019-04-27 14:54:30 +00:00
int mme_gtp_send_release_access_bearers_request(mme_ue_t *mme_ue)
2017-09-07 10:44:08 +00:00
{
2019-04-27 14:54:30 +00:00
int rv;
2017-09-07 10:44:08 +00:00
gtp_header_t h;
2019-04-27 14:54:30 +00:00
ogs_pkbuf_t *pkbuf = NULL;
2017-09-07 10:44:08 +00:00
gtp_xact_t *xact = NULL;
2019-04-27 14:54:30 +00:00
ogs_assert(mme_ue);
2017-09-07 10:44:08 +00:00
memset(&h, 0, sizeof(gtp_header_t));
h.type = GTP_RELEASE_ACCESS_BEARERS_REQUEST_TYPE;
h.teid = mme_ue->sgw_s11_teid;
rv = mme_s11_build_release_access_bearers_request(&pkbuf, h.type);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2017-09-07 10:44:08 +00:00
2017-12-06 08:58:38 +00:00
xact = gtp_xact_local_create(mme_ue->gnode, &h, pkbuf);
2019-04-27 14:54:30 +00:00
ogs_assert(xact);
2017-09-07 10:44:08 +00:00
rv = gtp_xact_commit(xact);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2017-09-07 10:44:08 +00:00
2019-04-27 14:54:30 +00:00
return OGS_OK;
2017-09-07 10:44:08 +00:00
}
2019-04-27 14:54:30 +00:00
int mme_gtp_send_create_indirect_data_forwarding_tunnel_request(
mme_ue_t *mme_ue)
{
2019-04-27 14:54:30 +00:00
int rv;
gtp_header_t h;
2019-04-27 14:54:30 +00:00
ogs_pkbuf_t *pkbuf = NULL;
gtp_xact_t *xact = NULL;
2019-04-27 14:54:30 +00:00
ogs_assert(mme_ue);
memset(&h, 0, sizeof(gtp_header_t));
h.type = GTP_CREATE_INDIRECT_DATA_FORWARDING_TUNNEL_REQUEST_TYPE;
h.teid = mme_ue->sgw_s11_teid;
rv = mme_s11_build_create_indirect_data_forwarding_tunnel_request(
&pkbuf, h.type, mme_ue);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2017-12-06 08:58:38 +00:00
xact = gtp_xact_local_create(mme_ue->gnode, &h, pkbuf);
2019-04-27 14:54:30 +00:00
ogs_assert(xact);
rv = gtp_xact_commit(xact);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2019-04-27 14:54:30 +00:00
return OGS_OK;
}
2017-09-14 05:18:47 +00:00
2019-04-27 14:54:30 +00:00
int mme_gtp_send_delete_indirect_data_forwarding_tunnel_request(
2017-09-14 05:18:47 +00:00
mme_ue_t *mme_ue)
{
2019-04-27 14:54:30 +00:00
int rv;
2017-09-14 05:18:47 +00:00
gtp_header_t h;
2019-04-27 14:54:30 +00:00
ogs_pkbuf_t *pkbuf = NULL;
2017-09-14 05:18:47 +00:00
gtp_xact_t *xact = NULL;
2019-04-27 14:54:30 +00:00
ogs_assert(mme_ue);
2017-09-14 05:18:47 +00:00
memset(&h, 0, sizeof(gtp_header_t));
h.type = GTP_DELETE_INDIRECT_DATA_FORWARDING_TUNNEL_REQUEST_TYPE;
h.teid = mme_ue->sgw_s11_teid;
2019-04-27 14:54:30 +00:00
pkbuf = ogs_pkbuf_alloc(NULL, TLV_MAX_HEADROOM);
ogs_pkbuf_reserve(pkbuf, TLV_MAX_HEADROOM);
2017-09-14 05:18:47 +00:00
2017-12-06 08:58:38 +00:00
xact = gtp_xact_local_create(mme_ue->gnode, &h, pkbuf);
2019-04-27 14:54:30 +00:00
ogs_assert(xact);
2017-09-14 05:18:47 +00:00
rv = gtp_xact_commit(xact);
2019-04-27 14:54:30 +00:00
ogs_assert(rv == OGS_OK);
2017-09-14 05:18:47 +00:00
2019-04-27 14:54:30 +00:00
return OGS_OK;
2017-09-14 05:18:47 +00:00
}