You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

293 lines
8.0 KiB
C++

/**
*
* \section COPYRIGHT
*
* Copyright 2013-2021 Software Radio Systems Limited
*
* By using this file, you agree to the terms and conditions set
* forth in the LICENSE file which can be found at the top level of
* the distribution.
*
*/
#include "srsepc/hdr/mbms-gw/mbms-gw.h"
#include "srsran/common/standard_streams.h"
#include "srsran/upper/gtpu.h"
#include <algorithm>
#include <fcntl.h>
#include <iostream>
#include <linux/if.h>
#include <linux/if_tun.h>
#include <linux/ip.h>
#include <netinet/in.h>
#include <netinet/udp.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
namespace srsepc {
mbms_gw* mbms_gw::m_instance = NULL;
pthread_mutex_t mbms_gw_instance_mutex = PTHREAD_MUTEX_INITIALIZER;
const uint16_t MBMS_GW_BUFFER_SIZE = 2500;
mbms_gw::mbms_gw() : m_running(false), m_sgi_mb_up(false), thread("MBMS_GW")
{
return;
}
mbms_gw::~mbms_gw()
{
return;
}
mbms_gw* mbms_gw::get_instance(void)
{
pthread_mutex_lock(&mbms_gw_instance_mutex);
if (NULL == m_instance) {
m_instance = new mbms_gw();
}
pthread_mutex_unlock(&mbms_gw_instance_mutex);
return (m_instance);
}
void mbms_gw::cleanup(void)
{
pthread_mutex_lock(&mbms_gw_instance_mutex);
if (NULL != m_instance) {
delete m_instance;
m_instance = NULL;
}
pthread_mutex_unlock(&mbms_gw_instance_mutex);
}
int mbms_gw::init(mbms_gw_args_t* args)
{
int err;
err = init_sgi_mb_if(args);
if (err != SRSRAN_SUCCESS) {
srsran::console("Error initializing SGi-MB.\n");
m_logger.error("Error initializing SGi-MB.");
return SRSRAN_ERROR_CANT_START;
}
err = init_m1_u(args);
if (err != SRSRAN_SUCCESS) {
srsran::console("Error initializing SGi-MB.\n");
m_logger.error("Error initializing SGi-MB.");
return SRSRAN_ERROR_CANT_START;
}
m_logger.info("MBMS GW Initiated");
srsran::console("MBMS GW Initiated\n");
return SRSRAN_SUCCESS;
}
void mbms_gw::stop()
{
if (m_running) {
if (m_sgi_mb_up) {
close(m_sgi_mb_if);
m_logger.info("Closed SGi-MB interface");
}
m_running = false;
thread_cancel();
wait_thread_finish();
}
return;
}
int mbms_gw::init_sgi_mb_if(mbms_gw_args_t* args)
{
struct ifreq ifr;
if (m_sgi_mb_up) {
return SRSRAN_ERROR_ALREADY_STARTED;
}
// Construct the TUN device
m_sgi_mb_if = open("/dev/net/tun", O_RDWR);
m_logger.info("TUN file descriptor = %d", m_sgi_mb_if);
if (m_sgi_mb_if < 0) {
m_logger.error("Failed to open TUN device: %s", strerror(errno));
return SRSRAN_ERROR_CANT_START;
}
memset(&ifr, 0, sizeof(ifr));
ifr.ifr_flags = IFF_TUN | IFF_NO_PI;
strncpy(ifr.ifr_ifrn.ifrn_name,
args->sgi_mb_if_name.c_str(),
std::min(args->sgi_mb_if_name.length(), (size_t)IFNAMSIZ - 1));
ifr.ifr_ifrn.ifrn_name[IFNAMSIZ - 1] = '\0';
if (ioctl(m_sgi_mb_if, TUNSETIFF, &ifr) < 0) {
m_logger.error("Failed to set TUN device name: %s", strerror(errno));
close(m_sgi_mb_if);
return SRSRAN_ERROR_CANT_START;
} else {
m_logger.debug("Set TUN device name: %s", args->sgi_mb_if_name.c_str());
}
// Bring up the interface
int sgi_mb_sock = socket(AF_INET, SOCK_DGRAM, 0);
if (sgi_mb_sock < 0) {
m_logger.error("Failed to bring up socket: %s", strerror(errno));
close(m_sgi_mb_if);
return SRSRAN_ERROR_CANT_START;
}
if (ioctl(sgi_mb_sock, SIOCGIFFLAGS, &ifr) < 0) {
m_logger.error("Failed to bring up interface: %s", strerror(errno));
close(m_sgi_mb_if);
close(sgi_mb_sock);
return SRSRAN_ERROR_CANT_START;
}
ifr.ifr_flags |= IFF_UP | IFF_RUNNING;
if (ioctl(sgi_mb_sock, SIOCSIFFLAGS, &ifr) < 0) {
m_logger.error("Failed to set socket flags: %s", strerror(errno));
close(sgi_mb_sock);
close(m_sgi_mb_if);
return SRSRAN_ERROR_CANT_START;
}
// Set IP of the interface
struct sockaddr_in* addr = (struct sockaddr_in*)&ifr.ifr_addr;
addr->sin_family = AF_INET;
addr->sin_addr.s_addr = inet_addr(args->sgi_mb_if_addr.c_str());
addr->sin_port = 0;
if (ioctl(sgi_mb_sock, SIOCSIFADDR, &ifr) < 0) {
m_logger.error(
"Failed to set TUN interface IP. Address: %s, Error: %s", args->sgi_mb_if_addr.c_str(), strerror(errno));
close(m_sgi_mb_if);
close(sgi_mb_sock);
return SRSRAN_ERROR_CANT_START;
}
ifr.ifr_netmask.sa_family = AF_INET;
((struct sockaddr_in*)&ifr.ifr_netmask)->sin_addr.s_addr = inet_addr(args->sgi_mb_if_mask.c_str());
if (ioctl(sgi_mb_sock, SIOCSIFNETMASK, &ifr) < 0) {
m_logger.error("Failed to set TUN interface Netmask. Error: %s", strerror(errno));
close(m_sgi_mb_if);
close(sgi_mb_sock);
return SRSRAN_ERROR_CANT_START;
}
m_sgi_mb_up = true;
close(sgi_mb_sock);
return SRSRAN_SUCCESS;
}
int mbms_gw::init_m1_u(mbms_gw_args_t* args)
{
int addrlen;
struct sockaddr_in addr;
m_m1u = socket(AF_INET, SOCK_DGRAM, 0);
if (m_m1u < 0) {
m_logger.error("Failed to open socket: %s", strerror(errno));
return SRSRAN_ERROR_CANT_START;
}
m_m1u_up = true;
/* set no loopback */
char loopch = 0;
if (setsockopt(m_m1u, IPPROTO_IP, IP_MULTICAST_LOOP, (char*)&loopch, sizeof(char)) < 0) {
m_logger.error("Failed to disable loopback: %s", strerror(errno));
return SRSRAN_ERROR_CANT_START;
} else {
m_logger.debug("Loopback disabled");
}
/* Set local interface for outbound multicast packets*/
/* The IP must be associated with a local multicast capable interface */
struct in_addr local_if;
local_if.s_addr = inet_addr(args->m1u_multi_if.c_str());
if (setsockopt(m_m1u, IPPROTO_IP, IP_MULTICAST_IF, (char*)&local_if, sizeof(struct in_addr)) < 0) {
m_logger.error("Error %s setting multicast interface %s.", strerror(errno), args->m1u_multi_if.c_str());
return SRSRAN_ERROR_CANT_START;
} else {
printf("Multicast interface specified. Address: %s\n", args->m1u_multi_if.c_str());
}
/*Set Multicast TTL*/
if (setsockopt(m_m1u, IPPROTO_IP, IP_MULTICAST_TTL, &args->m1u_multi_ttl, sizeof(args->m1u_multi_ttl)) < 0) {
perror("Error setting multicast ttl.\n");
return SRSRAN_ERROR_CANT_START;
}
bzero(&m_m1u_multi_addr, sizeof(m_m1u_multi_addr));
m_m1u_multi_addr.sin_family = AF_INET;
m_m1u_multi_addr.sin_port = htons(GTPU_RX_PORT + 1);
m_m1u_multi_addr.sin_addr.s_addr = inet_addr(args->m1u_multi_addr.c_str());
m_logger.info("Initialized M1-U");
return SRSRAN_SUCCESS;
}
void mbms_gw::run_thread()
{
// Mark the thread as running
m_running = true;
srsran::unique_byte_buffer_t msg = srsran::make_byte_buffer();
if (msg == nullptr) {
m_logger.error("Couldn't allocate PDU in %s().", __FUNCTION__);
return;
}
uint8_t seq = 0;
while (m_running) {
msg->clear();
int n;
do {
n = read(m_sgi_mb_if, msg->msg, SRSRAN_MAX_BUFFER_SIZE_BYTES);
} while (n == -1 && errno == EAGAIN);
if (n < 0) {
m_logger.error("Error reading from TUN interface. Error: %s", strerror(errno));
} else {
msg->N_bytes = n;
handle_sgi_md_pdu(msg.get());
}
}
return;
}
void mbms_gw::handle_sgi_md_pdu(srsran::byte_buffer_t* msg)
{
uint8_t version;
srsran::gtpu_header_t header;
// Setup GTP-U header
header.flags = GTPU_FLAGS_VERSION_V1 | GTPU_FLAGS_GTP_PROTOCOL;
header.message_type = GTPU_MSG_DATA_PDU;
header.length = msg->N_bytes;
header.teid = 0xAAAA; // TODO Harcoded TEID for now
// Sanity Check IP packet
if (msg->N_bytes < 20) {
m_logger.error("IPv4 min len: %d, drop msg len %d", 20, msg->N_bytes);
return;
}
// IP Headers
struct iphdr* iph = (struct iphdr*)msg->msg;
if (iph->version != 4) {
m_logger.info("IPv6 not supported yet.");
return;
}
// Write GTP-U header into packet
if (!srsran::gtpu_write_header(&header, msg, m_logger)) {
srsran::console("Error writing GTP-U header on PDU\n");
}
int n = sendto(m_m1u, msg->msg, msg->N_bytes, 0, (sockaddr*)&m_m1u_multi_addr, sizeof(struct sockaddr));
if (n < 0) {
srsran::console("Error writing to M1-U socket.\n");
} else {
m_logger.debug("Sent %d Bytes", msg->N_bytes);
}
}
} // namespace srsepc