mirror of https://github.com/PentHertz/srsLTE.git
Started to fill in the necessary information for the create session response.
This commit is contained in:
parent
2172662388
commit
2e6050bf38
|
@ -42,6 +42,7 @@ public:
|
||||||
|
|
||||||
void init();
|
void init();
|
||||||
void send_create_session_request(uint64_t imsi, struct srslte::gtpc_create_session_response *cs_resp);
|
void send_create_session_request(uint64_t imsi, struct srslte::gtpc_create_session_response *cs_resp);
|
||||||
|
uint64_t get_new_ctrl_teid();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
|
@ -53,6 +54,8 @@ private:
|
||||||
|
|
||||||
spgw* m_spgw;
|
spgw* m_spgw;
|
||||||
in_addr_t m_mme_gtpc_ip;
|
in_addr_t m_mme_gtpc_ip;
|
||||||
|
|
||||||
|
uint64_t m_next_ctrl_teid;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,6 +39,7 @@
|
||||||
#include "srslte/common/log_filter.h"
|
#include "srslte/common/log_filter.h"
|
||||||
#include "srslte/common/buffer_pool.h"
|
#include "srslte/common/buffer_pool.h"
|
||||||
#include "srslte/common/threads.h"
|
#include "srslte/common/threads.h"
|
||||||
|
#include "srslte/asn1/gtpc.h"
|
||||||
|
|
||||||
namespace srsepc{
|
namespace srsepc{
|
||||||
|
|
||||||
|
@ -50,6 +51,15 @@ typedef struct {
|
||||||
} spgw_args_t;
|
} spgw_args_t;
|
||||||
|
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint64_t imsi;
|
||||||
|
in_addr_t ue_ipv4;
|
||||||
|
struct gtpc_f_teid_ie uplink_ctrl;
|
||||||
|
struct gtpc_f_teid_ie uplink_user;
|
||||||
|
struct gtpc_f_teid_ie downlink_ctrl;
|
||||||
|
struct gtpc_f_teid_ie downlink_user;
|
||||||
|
} spgw_ue_ctxr;
|
||||||
|
|
||||||
class spgw:
|
class spgw:
|
||||||
public thread
|
public thread
|
||||||
{
|
{
|
||||||
|
@ -60,6 +70,8 @@ public:
|
||||||
void stop();
|
void stop();
|
||||||
void run_thread();
|
void run_thread();
|
||||||
|
|
||||||
|
void handle_create_session_request(struct srslte::gtpc_create_session_request *cs_req, struct srslte::gtpc_create_session_response *cs_resp);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
spgw();
|
spgw();
|
||||||
|
@ -81,7 +93,7 @@ private:
|
||||||
|
|
||||||
/*Logs*/
|
/*Logs*/
|
||||||
srslte::log_filter *m_spgw_log;
|
srslte::log_filter *m_spgw_log;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace srsepc
|
} // namespace srsepc
|
||||||
|
|
|
@ -34,6 +34,7 @@ mme_gtpc* mme_gtpc::m_instance = NULL;
|
||||||
boost::mutex mme_gtpc_instance_mutex;
|
boost::mutex mme_gtpc_instance_mutex;
|
||||||
|
|
||||||
mme_gtpc::mme_gtpc()
|
mme_gtpc::mme_gtpc()
|
||||||
|
:m_next_ctrl_teid(1)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -69,12 +70,20 @@ mme_gtpc::init()
|
||||||
m_spgw = spgw::get_instance();
|
m_spgw = spgw::get_instance();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uint64_t
|
||||||
|
mme_gtpc::get_new_ctrl_teid()
|
||||||
|
{
|
||||||
|
return m_next_ctrl_teid++; //FIXME Use a Id pool?
|
||||||
|
}
|
||||||
void
|
void
|
||||||
mme_gtpc::send_create_session_request(uint64_t imsi, struct srslte::gtpc_create_session_response *cs_resp)
|
mme_gtpc::send_create_session_request(uint64_t imsi, struct srslte::gtpc_create_session_response *cs_resp)
|
||||||
{
|
{
|
||||||
struct srslte::gtpc_pdu cs_req_pdu;
|
struct srslte::gtpc_pdu cs_req_pdu;
|
||||||
struct srslte::gtpc_create_session_request *cs_req = &cs_req_pdu.choice.create_session_request;
|
struct srslte::gtpc_create_session_request *cs_req = &cs_req_pdu.choice.create_session_request;
|
||||||
|
|
||||||
|
//Initialize GTP-C message to zero
|
||||||
|
bzero(&cs_req_pdu, sizeof(struct srslte::gtpc_pdu));
|
||||||
|
|
||||||
//Setup GTP-C Header. FIXME: Length, sequence and other fields need to be added.
|
//Setup GTP-C Header. FIXME: Length, sequence and other fields need to be added.
|
||||||
cs_req_pdu.header.piggyback = false;
|
cs_req_pdu.header.piggyback = false;
|
||||||
cs_req_pdu.header.teid_present = true;
|
cs_req_pdu.header.teid_present = true;
|
||||||
|
@ -93,7 +102,7 @@ mme_gtpc::send_create_session_request(uint64_t imsi, struct srslte::gtpc_create_
|
||||||
//Save RX Control TEID
|
//Save RX Control TEID
|
||||||
//create_rx_control_teid(cs_req->sender_f_teid);
|
//create_rx_control_teid(cs_req->sender_f_teid);
|
||||||
|
|
||||||
//spgw->handle_create_session_request(&cs_req, cs_resp);
|
m_spgw->handle_create_session_request(cs_req, cs_resp);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
} //namespace srsepc
|
} //namespace srsepc
|
||||||
|
|
|
@ -35,6 +35,7 @@
|
||||||
#include <linux/if_tun.h>
|
#include <linux/if_tun.h>
|
||||||
#include "spgw/spgw.h"
|
#include "spgw/spgw.h"
|
||||||
|
|
||||||
|
|
||||||
namespace srsepc{
|
namespace srsepc{
|
||||||
|
|
||||||
spgw* spgw::m_instance = NULL;
|
spgw* spgw::m_instance = NULL;
|
||||||
|
@ -128,6 +129,106 @@ spgw::stop()
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
srslte::error_t
|
||||||
|
spgw::init_sgi_if(spgw_args_t *args)
|
||||||
|
{
|
||||||
|
char dev[IFNAMSIZ] = "srs_spgw_sgi";
|
||||||
|
struct ifreq ifr;
|
||||||
|
|
||||||
|
if(m_sgi_up)
|
||||||
|
{
|
||||||
|
return(srslte::ERROR_ALREADY_STARTED);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Construct the TUN device
|
||||||
|
m_sgi_if = open("/dev/net/tun", O_RDWR);
|
||||||
|
m_spgw_log->info("TUN file descriptor = %d\n", m_sgi_if);
|
||||||
|
if(m_sgi_if < 0)
|
||||||
|
{
|
||||||
|
m_spgw_log->error("Failed to open TUN device: %s\n", strerror(errno));
|
||||||
|
return(srslte::ERROR_CANT_START);
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(&ifr, 0, sizeof(ifr));
|
||||||
|
ifr.ifr_flags = IFF_TUN | IFF_NO_PI;
|
||||||
|
strncpy(ifr.ifr_ifrn.ifrn_name, dev, IFNAMSIZ);
|
||||||
|
if(ioctl(m_sgi_if, TUNSETIFF, &ifr) < 0)
|
||||||
|
{
|
||||||
|
m_spgw_log->error("Failed to set TUN device name: %s\n", strerror(errno));
|
||||||
|
close(m_sgi_if);
|
||||||
|
return(srslte::ERROR_CANT_START);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Bring up the interface
|
||||||
|
m_sgi_sock = socket(AF_INET, SOCK_DGRAM, 0);
|
||||||
|
|
||||||
|
if(ioctl(m_sgi_sock, SIOCGIFFLAGS, &ifr) < 0)
|
||||||
|
{
|
||||||
|
m_spgw_log->error("Failed to bring up socket: %s\n", strerror(errno));
|
||||||
|
close(m_sgi_if);
|
||||||
|
return(srslte::ERROR_CANT_START);
|
||||||
|
}
|
||||||
|
ifr.ifr_flags |= IFF_UP | IFF_RUNNING;
|
||||||
|
if(ioctl(m_sgi_sock, SIOCSIFFLAGS, &ifr) < 0)
|
||||||
|
{
|
||||||
|
m_spgw_log->error("Failed to set socket flags: %s\n", strerror(errno));
|
||||||
|
close(m_sgi_if);
|
||||||
|
return(srslte::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_if_addr.c_str());
|
||||||
|
addr->sin_port = 0;
|
||||||
|
|
||||||
|
if (ioctl(m_sgi_sock, SIOCSIFADDR, &ifr) < 0) {
|
||||||
|
m_spgw_log->error("Failed to set TUN interface IP. Address: %s, Error: %s\n", args->sgi_if_addr.c_str(), strerror(errno));
|
||||||
|
close(m_sgi_if);
|
||||||
|
close(m_sgi_sock);
|
||||||
|
return srslte::ERROR_CANT_START;
|
||||||
|
}
|
||||||
|
|
||||||
|
ifr.ifr_netmask.sa_family = AF_INET;
|
||||||
|
((struct sockaddr_in *)&ifr.ifr_netmask)->sin_addr.s_addr = inet_addr("255.255.255.0");
|
||||||
|
if (ioctl(m_sgi_sock, SIOCSIFNETMASK, &ifr) < 0) {
|
||||||
|
m_spgw_log->error("Failed to set TUN interface Netmask. Error: %s\n", strerror(errno));
|
||||||
|
close(m_sgi_if);
|
||||||
|
close(m_sgi_sock);
|
||||||
|
return srslte::ERROR_CANT_START;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_sgi_up = true;
|
||||||
|
return(srslte::ERROR_NONE);
|
||||||
|
}
|
||||||
|
|
||||||
|
srslte::error_t
|
||||||
|
spgw::init_s1u(spgw_args_t *args)
|
||||||
|
{
|
||||||
|
//Open S1-U socket
|
||||||
|
m_s1u = socket(AF_INET,SOCK_DGRAM,0);
|
||||||
|
if (m_s1u == -1)
|
||||||
|
{
|
||||||
|
m_spgw_log->error("Failed to open socket: %s\n", strerror(errno));
|
||||||
|
return srslte::ERROR_CANT_START;
|
||||||
|
}
|
||||||
|
m_s1u_up = true;
|
||||||
|
|
||||||
|
//Bind the socket
|
||||||
|
struct sockaddr_in servaddr;
|
||||||
|
servaddr.sin_family = AF_INET;
|
||||||
|
servaddr.sin_addr.s_addr=inet_addr(args->gtpu_bind_addr.c_str());
|
||||||
|
servaddr.sin_port=htons(GTPU_RX_PORT);
|
||||||
|
|
||||||
|
if (bind(m_s1u,(struct sockaddr *)&servaddr,sizeof(struct sockaddr_in))) {
|
||||||
|
m_spgw_log->error("Failed to bind socket: %s\n", strerror(errno));
|
||||||
|
return srslte::ERROR_CANT_START;
|
||||||
|
}
|
||||||
|
m_spgw_log->info("S1-U socket = %d\n", m_s1u);
|
||||||
|
return srslte::ERROR_NONE;
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
spgw::run_thread()
|
spgw::run_thread()
|
||||||
{
|
{
|
||||||
|
@ -183,104 +284,42 @@ spgw::run_thread()
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
srslte::error_t
|
|
||||||
spgw::init_sgi_if(spgw_args_t *args)
|
|
||||||
|
|
||||||
|
void
|
||||||
|
spgw::handle_create_session_request(struct srslte::gtpc_create_session_request *cs_req, struct srslte::gtpc_create_session_response *cs_resp)
|
||||||
{
|
{
|
||||||
char dev[IFNAMSIZ] = "srs_spgw_sgi";
|
//Setup uplink control TEID
|
||||||
struct ifreq ifr;
|
uint64_t spgw_uplink_ctrl_teid = get_new_ctrl_teid();
|
||||||
|
//Setup uplink user TEID
|
||||||
|
uint64_t spgw_uplink_user_teid = get_new_user_teid();
|
||||||
|
//Allocate UE IP
|
||||||
|
in_addr_t ue_ip = get_new_ue_ipv4();
|
||||||
|
|
||||||
if(m_sgi_up)
|
//Save the UE context //TODO!!!
|
||||||
{
|
|
||||||
return(srslte::ERROR_ALREADY_STARTED);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
//Create session response message
|
||||||
// Construct the TUN device
|
//Initialize to zero\\
|
||||||
m_sgi_if = open("/dev/net/tun", O_RDWR);
|
bzero(cs_resp,sizeof(struct srslte::gtpc_create_session_response));
|
||||||
m_spgw_log->info("TUN file descriptor = %d\n", m_sgi_if);
|
//Setup Cause\\
|
||||||
if(m_sgi_if < 0)
|
cs_resp->cause = ;
|
||||||
{
|
//Setup sender F-TEID (ctrl)\\
|
||||||
m_spgw_log->error("Failed to open TUN device: %s\n", strerror(errno));
|
cs_resp->sender_f_teid.teid_present = true;
|
||||||
return(srslte::ERROR_CANT_START);
|
cs_resp->sender_f_teid.teid = spgw_uplink_ctrl_teid;
|
||||||
}
|
cs_resp->sender_f_teid.ipv4 = m_gtpu_bind_addr;//FIXME This is not relevant, as the GTP-C is not transmitted over sockets yet.
|
||||||
|
//Bearer context created\\
|
||||||
|
cs_resp->eps_bearer_context_created.ebi = 5;
|
||||||
|
cs_resp->eps_bearer_context_created.cause = ;
|
||||||
|
cs_resp->eps_bearer_context_created.s1_u_sgw_f_teid_present=true;
|
||||||
|
cs_resp->eps_bearer_context_created.s1_u_sgw_f_teid.teid = spgw_uplink_user_teid;
|
||||||
|
//Fill in the PDA\\
|
||||||
|
cs_resp->pda_present = true;
|
||||||
|
cs_resp->pda.pdn_type = srslte::GTPC_IPV4;
|
||||||
|
cs_resp->ipv4_present = true;
|
||||||
|
cs_resp->ipv4 = ue_ip;
|
||||||
|
|
||||||
memset(&ifr, 0, sizeof(ifr));
|
return;
|
||||||
ifr.ifr_flags = IFF_TUN | IFF_NO_PI;
|
|
||||||
strncpy(ifr.ifr_ifrn.ifrn_name, dev, IFNAMSIZ);
|
|
||||||
if(ioctl(m_sgi_if, TUNSETIFF, &ifr) < 0)
|
|
||||||
{
|
|
||||||
m_spgw_log->error("Failed to set TUN device name: %s\n", strerror(errno));
|
|
||||||
close(m_sgi_if);
|
|
||||||
return(srslte::ERROR_CANT_START);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Bring up the interface
|
|
||||||
m_sgi_sock = socket(AF_INET, SOCK_DGRAM, 0);
|
|
||||||
|
|
||||||
if(ioctl(m_sgi_sock, SIOCGIFFLAGS, &ifr) < 0)
|
|
||||||
{
|
|
||||||
m_spgw_log->error("Failed to bring up socket: %s\n", strerror(errno));
|
|
||||||
close(m_sgi_if);
|
|
||||||
return(srslte::ERROR_CANT_START);
|
|
||||||
}
|
|
||||||
ifr.ifr_flags |= IFF_UP | IFF_RUNNING;
|
|
||||||
if(ioctl(m_sgi_sock, SIOCSIFFLAGS, &ifr) < 0)
|
|
||||||
{
|
|
||||||
m_spgw_log->error("Failed to set socket flags: %s\n", strerror(errno));
|
|
||||||
close(m_sgi_if);
|
|
||||||
return(srslte::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_if_addr.c_str());
|
|
||||||
addr->sin_port = 0;
|
|
||||||
|
|
||||||
if (ioctl(m_sgi_sock, SIOCSIFADDR, &ifr) < 0) {
|
|
||||||
m_spgw_log->error("Failed to set TUN interface IP. Address: %s, Error: %s\n", args->sgi_if_addr.c_str(), strerror(errno));
|
|
||||||
close(m_sgi_if);
|
|
||||||
close(m_sgi_sock);
|
|
||||||
return srslte::ERROR_CANT_START;
|
|
||||||
}
|
|
||||||
|
|
||||||
ifr.ifr_netmask.sa_family = AF_INET;
|
|
||||||
((struct sockaddr_in *)&ifr.ifr_netmask)->sin_addr.s_addr = inet_addr("255.255.255.0");
|
|
||||||
if (ioctl(m_sgi_sock, SIOCSIFNETMASK, &ifr) < 0) {
|
|
||||||
m_spgw_log->error("Failed to set TUN interface Netmask. Error: %s\n", strerror(errno));
|
|
||||||
close(m_sgi_if);
|
|
||||||
close(m_sgi_sock);
|
|
||||||
return srslte::ERROR_CANT_START;
|
|
||||||
}
|
|
||||||
|
|
||||||
m_sgi_up = true;
|
|
||||||
return(srslte::ERROR_NONE);
|
|
||||||
}
|
|
||||||
|
|
||||||
srslte::error_t
|
|
||||||
spgw::init_s1u(spgw_args_t *args)
|
|
||||||
{
|
|
||||||
//Open S1-U socket
|
|
||||||
m_s1u = socket(AF_INET,SOCK_DGRAM,0);
|
|
||||||
if (m_s1u == -1)
|
|
||||||
{
|
|
||||||
m_spgw_log->error("Failed to open socket: %s\n", strerror(errno));
|
|
||||||
return srslte::ERROR_CANT_START;
|
|
||||||
}
|
|
||||||
m_s1u_up = true;
|
|
||||||
|
|
||||||
//Bind the socket
|
|
||||||
struct sockaddr_in servaddr;
|
|
||||||
servaddr.sin_family = AF_INET;
|
|
||||||
servaddr.sin_addr.s_addr=inet_addr(args->gtpu_bind_addr.c_str());
|
|
||||||
servaddr.sin_port=htons(GTPU_RX_PORT);
|
|
||||||
|
|
||||||
if (bind(m_s1u,(struct sockaddr *)&servaddr,sizeof(struct sockaddr_in))) {
|
|
||||||
m_spgw_log->error("Failed to bind socket: %s\n", strerror(errno));
|
|
||||||
return srslte::ERROR_CANT_START;
|
|
||||||
}
|
|
||||||
m_spgw_log->info("S1-U socket = %d\n", m_s1u);
|
|
||||||
return srslte::ERROR_NONE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} //namespace srsepc
|
} //namespace srsepc
|
||||||
|
|
Loading…
Reference in New Issue