open5gs/lib/pfcp/conv.c

401 lines
11 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 "ogs-pfcp.h"
int ogs_pfcp_sockaddr_to_node_id(
ogs_sockaddr_t *addr, ogs_sockaddr_t *addr6, int prefer_ipv4,
ogs_pfcp_node_id_t *node_id, int *len)
{
const int hdr_len = 1;
char *hostname = NULL;
ogs_assert(node_id);
memset(node_id, 0, sizeof *node_id);
if (addr) {
hostname = ogs_gethostname(addr);
if (hostname) {
node_id->type = OGS_PFCP_NODE_ID_FQDN;
*len = ogs_fqdn_build(node_id->fqdn,
hostname, strlen(hostname)) + hdr_len;
return OGS_OK;
}
}
if (addr6) {
hostname = ogs_gethostname(addr6);
if (hostname) {
node_id->type = OGS_PFCP_NODE_ID_FQDN;
*len = ogs_fqdn_build(node_id->fqdn,
hostname, strlen(hostname)) + hdr_len;
return OGS_OK;
}
}
if (prefer_ipv4 && addr) {
node_id->type = OGS_PFCP_NODE_ID_IPV4;
node_id->addr = addr->sin.sin_addr.s_addr;
*len = OGS_IPV4_LEN + hdr_len;
} else if (addr6) {
node_id->type = OGS_PFCP_NODE_ID_IPV6;
memcpy(node_id->addr6, addr6->sin6.sin6_addr.s6_addr, OGS_IPV6_LEN);
*len = OGS_IPV6_LEN + hdr_len;
} else if (addr) {
node_id->type = OGS_PFCP_NODE_ID_IPV4;
node_id->addr = addr->sin.sin_addr.s_addr;
*len = OGS_IPV4_LEN + hdr_len;
} else {
ogs_assert_if_reached();
return OGS_ERROR;
}
return OGS_OK;
}
int ogs_pfcp_f_seid_to_sockaddr(
ogs_pfcp_f_seid_t *f_seid, uint16_t port, ogs_sockaddr_t **list)
{
ogs_sockaddr_t *addr = NULL, *addr6 = NULL;
ogs_assert(f_seid);
ogs_assert(list);
addr = ogs_calloc(1, sizeof(ogs_sockaddr_t));
ogs_assert(addr);
addr->ogs_sa_family = AF_INET;
addr->ogs_sin_port = htobe16(port);
addr6 = ogs_calloc(1, sizeof(ogs_sockaddr_t));
ogs_assert(addr6);
addr6->ogs_sa_family = AF_INET6;
addr6->ogs_sin_port = htobe16(port);
if (f_seid->ipv4 && f_seid->ipv6) {
addr->next = addr6;
addr->sin.sin_addr.s_addr = f_seid->both.addr;
memcpy(addr6->sin6.sin6_addr.s6_addr, f_seid->both.addr6, OGS_IPV6_LEN);
*list = addr;
} else if (f_seid->ipv4) {
addr->sin.sin_addr.s_addr = f_seid->addr;
ogs_free(addr6);
*list = addr;
} else if (f_seid->ipv6) {
memcpy(addr6->sin6.sin6_addr.s6_addr, f_seid->addr6, OGS_IPV6_LEN);
ogs_free(addr);
*list = addr6;
} else {
ogs_free(addr);
ogs_free(addr6);
ogs_assert_if_reached();
}
return OGS_OK;
}
int ogs_pfcp_sockaddr_to_f_seid(
ogs_sockaddr_t *addr, ogs_sockaddr_t *addr6,
ogs_pfcp_f_seid_t *f_seid, int *len)
{
const int hdr_len = 9;
ogs_assert(f_seid);
memset(f_seid, 0, sizeof *f_seid);
if (addr && addr6) {
f_seid->ipv4 = 1;
f_seid->both.addr = addr->sin.sin_addr.s_addr;
f_seid->ipv6 = 1;
memcpy(f_seid->both.addr6, addr6->sin6.sin6_addr.s6_addr, OGS_IPV6_LEN);
*len = OGS_IPV4V6_LEN + hdr_len;
} else if (addr) {
f_seid->ipv4 = 1;
f_seid->ipv6 = 0;
f_seid->addr = addr->sin.sin_addr.s_addr;
*len = OGS_IPV4_LEN + hdr_len;
} else if (addr6) {
f_seid->ipv4 = 0;
f_seid->ipv6 = 1;
memcpy(f_seid->addr6, addr6->sin6.sin6_addr.s6_addr, OGS_IPV6_LEN);
*len = OGS_IPV6_LEN + hdr_len;
} else
ogs_assert_if_reached();
return OGS_OK;
}
int ogs_pfcp_f_seid_to_ip(ogs_pfcp_f_seid_t *f_seid, ogs_ip_t *ip)
{
ogs_assert(ip);
ogs_assert(f_seid);
memset(ip, 0, sizeof *ip);
ip->ipv4 = f_seid->ipv4;
ip->ipv6 = f_seid->ipv6;
if (ip->ipv4 && ip->ipv6) {
ip->addr = f_seid->both.addr;
memcpy(ip->addr6, f_seid->both.addr6, OGS_IPV6_LEN);
ip->len = OGS_IPV4V6_LEN;
} else if (ip->ipv4) {
ip->addr = f_seid->addr;
ip->len = OGS_IPV4_LEN;
} else if (ip->ipv6) {
memcpy(ip->addr6, f_seid->addr6, OGS_IPV6_LEN);
ip->len = OGS_IPV6_LEN;
} else
ogs_assert_if_reached();
return OGS_OK;
}
int ogs_pfcp_sockaddr_to_f_teid(
ogs_sockaddr_t *addr, ogs_sockaddr_t *addr6,
ogs_pfcp_f_teid_t *f_teid, int *len)
{
const int hdr_len = 5;
ogs_assert(addr || addr6);
ogs_assert(f_teid);
memset(f_teid, 0, sizeof *f_teid);
if (addr && addr6) {
f_teid->ipv4 = 1;
f_teid->both.addr = addr->sin.sin_addr.s_addr;
f_teid->ipv6 = 1;
memcpy(f_teid->both.addr6, addr6->sin6.sin6_addr.s6_addr, OGS_IPV6_LEN);
*len = OGS_IPV4V6_LEN + hdr_len;
} else if (addr) {
f_teid->ipv4 = 1;
f_teid->ipv6 = 0;
f_teid->addr = addr->sin.sin_addr.s_addr;
*len = OGS_IPV4_LEN + hdr_len;
} else if (addr6) {
f_teid->ipv4 = 0;
f_teid->ipv6 = 1;
memcpy(f_teid->addr6, addr6->sin6.sin6_addr.s6_addr, OGS_IPV6_LEN);
*len = OGS_IPV6_LEN + hdr_len;
} else
ogs_assert_if_reached();
return OGS_OK;
}
int ogs_pfcp_f_teid_to_sockaddr(
ogs_pfcp_f_teid_t *f_teid, int f_teid_len,
ogs_sockaddr_t **addr, ogs_sockaddr_t **addr6)
{
ogs_assert(addr && addr6);
ogs_assert(f_teid);
ogs_assert(f_teid_len);
*addr = NULL;
*addr6 = NULL;
if (f_teid->ipv4 && f_teid->ipv6) {
*addr = ogs_calloc(1, sizeof(**addr));
ogs_assert(*addr);
(*addr)->sin.sin_addr.s_addr = f_teid->both.addr;
(*addr)->ogs_sa_family = AF_INET;
*addr6 = ogs_calloc(1, sizeof(**addr6));
ogs_assert(*addr6);
memcpy((*addr6)->sin6.sin6_addr.s6_addr,
f_teid->both.addr6, OGS_IPV6_LEN);
(*addr6)->ogs_sa_family = AF_INET6;
} else if (f_teid->ipv4) {
*addr = ogs_calloc(1, sizeof(**addr));
ogs_assert(*addr);
(*addr)->sin.sin_addr.s_addr = f_teid->addr;
(*addr)->ogs_sa_family = AF_INET;
} else if (f_teid->ipv6) {
*addr6 = ogs_calloc(1, sizeof(**addr6));
ogs_assert(*addr6);
memcpy((*addr6)->sin6.sin6_addr.s6_addr, f_teid->addr6, OGS_IPV6_LEN);
(*addr6)->ogs_sa_family = AF_INET6;
}
return OGS_OK;
}
int ogs_pfcp_f_teid_to_ip(ogs_pfcp_f_teid_t *f_teid, ogs_ip_t *ip)
{
ogs_assert(ip);
ogs_assert(f_teid);
memset(ip, 0, sizeof *ip);
ip->ipv4 = f_teid->ipv4;
ip->ipv6 = f_teid->ipv6;
if (ip->ipv4 && ip->ipv6) {
ip->addr = f_teid->both.addr;
memcpy(ip->addr6, f_teid->both.addr6, OGS_IPV6_LEN);
ip->len = OGS_IPV4V6_LEN;
} else if (ip->ipv4) {
ip->addr = f_teid->addr;
ip->len = OGS_IPV4_LEN;
} else if (ip->ipv6) {
memcpy(ip->addr6, f_teid->addr6, OGS_IPV6_LEN);
ip->len = OGS_IPV6_LEN;
} else
ogs_assert_if_reached();
return OGS_OK;
}
int ogs_pfcp_user_plane_ip_resource_info_to_f_teid(
ogs_user_plane_ip_resource_info_t *info,
ogs_pfcp_f_teid_t *f_teid, int *len)
{
const int hdr_len = 5;
ogs_assert(info);
ogs_assert(f_teid);
memset(f_teid, 0, sizeof *f_teid);
if (info->v4 && info->v6) {
f_teid->ipv4 = 1;
f_teid->both.addr = info->addr;
f_teid->ipv6 = 1;
memcpy(f_teid->both.addr6, info->addr6, OGS_IPV6_LEN);
*len = OGS_IPV4V6_LEN + hdr_len;
} else if (info->v4) {
f_teid->ipv4 = 1;
f_teid->ipv6 = 0;
f_teid->addr = info->addr;
*len = OGS_IPV4_LEN + hdr_len;
} else if (info->v6) {
f_teid->ipv4 = 0;
f_teid->ipv6 = 1;
memcpy(f_teid->addr6, info->addr6, OGS_IPV6_LEN);
*len = OGS_IPV6_LEN + hdr_len;
} else
ogs_assert_if_reached();
return OGS_OK;
}
int ogs_pfcp_paa_to_ue_ip_addr(
ogs_paa_t *paa, ogs_pfcp_ue_ip_addr_t *addr, int *len)
{
const int hdr_len = 1;
ogs_assert(paa);
ogs_assert(addr);
memset(addr, 0, sizeof *addr);
if (paa->session_type == OGS_PDU_SESSION_TYPE_IPV4V6) {
addr->ipv4 = 1;
addr->both.addr = paa->both.addr;
addr->ipv6 = 1;
memcpy(addr->both.addr6, paa->both.addr6, OGS_IPV6_LEN);
*len = OGS_IPV4V6_LEN + hdr_len;
} else if (paa->session_type == OGS_PDU_SESSION_TYPE_IPV4) {
addr->ipv4 = 1;
addr->ipv6 = 0;
addr->addr = paa->addr;
*len = OGS_IPV4_LEN + hdr_len;
} else if (paa->session_type == OGS_PDU_SESSION_TYPE_IPV6) {
addr->ipv4 = 0;
addr->ipv6 = 1;
memcpy(addr->addr6, paa->addr6, OGS_IPV6_LEN);
*len = OGS_IPV6_LEN + hdr_len;
} else
ogs_assert_if_reached();
return OGS_OK;
}
int ogs_pfcp_ip_to_outer_header_creation(ogs_ip_t *ip,
ogs_pfcp_outer_header_creation_t *outer_header_creation, int *len)
{
const int hdr_len = 6;
ogs_assert(ip);
ogs_assert(outer_header_creation);
memset(outer_header_creation, 0, sizeof *outer_header_creation);
if (ip->ipv4 && ip->ipv6) {
outer_header_creation->gtpu4 = 1;
outer_header_creation->both.addr = ip->addr;
outer_header_creation->gtpu6 = 1;
memcpy(outer_header_creation->both.addr6, ip->addr6, OGS_IPV6_LEN);
*len = OGS_IPV4V6_LEN + hdr_len;
} else if (ip->ipv4) {
outer_header_creation->gtpu4 = 1;
outer_header_creation->gtpu6 = 0;
outer_header_creation->addr = ip->addr;
*len = OGS_IPV4_LEN + hdr_len;
} else if (ip->ipv6) {
outer_header_creation->gtpu4 = 0;
outer_header_creation->gtpu6 = 1;
memcpy(outer_header_creation->addr6, ip->addr6, OGS_IPV6_LEN);
*len = OGS_IPV6_LEN + hdr_len;
} else
ogs_assert_if_reached();
return OGS_OK;
}
int ogs_pfcp_outer_header_creation_to_ip(
ogs_pfcp_outer_header_creation_t *outer_header_creation, ogs_ip_t *ip)
{
ogs_assert(outer_header_creation);
ogs_assert(ip);
memset(ip, 0, sizeof *ip);
if ((outer_header_creation->gtpu4 ||
outer_header_creation->ip4 ||
outer_header_creation->udp4) &&
(outer_header_creation->gtpu6 ||
outer_header_creation->ip6 ||
outer_header_creation->udp6)) {
ip->ipv4 = 1; ip->ipv6 = 1;
ip->len = OGS_IPV4V6_LEN;
ip->addr = outer_header_creation->both.addr;
memcpy(ip->addr6, outer_header_creation->both.addr6, OGS_IPV6_LEN);
} else if (outer_header_creation->gtpu4 ||
outer_header_creation->ip4 ||
outer_header_creation->udp4) {
ip->ipv4 = 1;
ip->len = OGS_IPV4_LEN;
ip->addr = outer_header_creation->addr;
} else if (outer_header_creation->gtpu6 ||
outer_header_creation->ip6 ||
outer_header_creation->udp6) {
ip->ipv6 = 1;
ip->len = OGS_IPV6_LEN;
memcpy(ip->addr6, outer_header_creation->addr6, OGS_IPV6_LEN);
} else
return OGS_ERROR;
return OGS_OK;
}