/**
*
* \ section COPYRIGHT
*
* Copyright 2013 - 2017 Software Radio Systems Limited
*
* \ section LICENSE
*
* This file is part of srsLTE .
*
* srsLTE 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 .
*
* srsLTE 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 Affero General Public License for more details .
*
* A copy of the GNU Affero General Public License can be found in
* the LICENSE file in the top - level directory of this distribution
* and at http : //www.gnu.org/licenses/.
*
*/
# include <iostream>
# include <cmath>
# include <inttypes.h> // for printing uint64_t
# include "srslte/common/bcd_helpers.h"
# include "srsepc/hdr/mme/s1ap.h"
# include "srslte/asn1/gtpc.h"
# include "srslte/common/liblte_security.h"
namespace srsepc {
s1ap * s1ap : : m_instance = NULL ;
pthread_mutex_t s1ap_instance_mutex = PTHREAD_MUTEX_INITIALIZER ;
s1ap : : s1ap ( ) :
m_s1mme ( - 1 ) ,
m_next_mme_ue_s1ap_id ( 1 ) ,
m_mme_gtpc ( NULL ) ,
m_pool ( NULL )
{
}
s1ap : : ~ s1ap ( )
{
}
s1ap *
s1ap : : get_instance ( void )
{
pthread_mutex_lock ( & s1ap_instance_mutex ) ;
if ( m_instance = = NULL ) {
m_instance = new s1ap ( ) ;
}
pthread_mutex_unlock ( & s1ap_instance_mutex ) ;
return ( m_instance ) ;
}
void
s1ap : : cleanup ( void )
{
pthread_mutex_lock ( & s1ap_instance_mutex ) ;
if ( NULL ! = m_instance ) {
delete m_instance ;
m_instance = NULL ;
}
pthread_mutex_unlock ( & s1ap_instance_mutex ) ;
}
int
s1ap : : init ( s1ap_args_t s1ap_args , srslte : : log_filter * s1ap_log , hss_interface_s1ap * hss_ )
{
m_pool = srslte : : byte_buffer_pool : : get_instance ( ) ;
m_s1ap_args = s1ap_args ;
srslte : : s1ap_mccmnc_to_plmn ( s1ap_args . mcc , s1ap_args . mnc , & m_plmn ) ;
m_next_m_tmsi = rand ( ) ;
//Init log
m_s1ap_log = s1ap_log ;
//Get pointer to the HSS
m_hss = hss_ ;
//Init message handlers
m_s1ap_mngmt_proc = s1ap_mngmt_proc : : get_instance ( ) ; //Managment procedures
m_s1ap_mngmt_proc - > init ( ) ;
m_s1ap_nas_transport = s1ap_nas_transport : : get_instance ( ) ; //NAS Transport procedures
m_s1ap_nas_transport - > init ( m_hss ) ;
m_s1ap_ctx_mngmt_proc = s1ap_ctx_mngmt_proc : : get_instance ( ) ; //Context Management Procedures
m_s1ap_ctx_mngmt_proc - > init ( ) ;
//Get pointer to GTP-C class
m_mme_gtpc = mme_gtpc : : get_instance ( ) ;
//Initialize S1-MME
m_s1mme = enb_listen ( ) ;
m_s1ap_log - > info ( " S1AP Initialized \n " ) ;
return 0 ;
}
void
s1ap : : stop ( )
{
if ( m_s1mme ! = - 1 ) {
close ( m_s1mme ) ;
}
std : : map < uint16_t , enb_ctx_t * > : : iterator enb_it = m_active_enbs . begin ( ) ;
while ( enb_it ! = m_active_enbs . end ( ) )
{
m_s1ap_log - > info ( " Deleting eNB context. eNB Id: 0x%x \n " , enb_it - > second - > enb_id ) ;
m_s1ap_log - > console ( " Deleting eNB context. eNB Id: 0x%x \n " , enb_it - > second - > enb_id ) ;
delete enb_it - > second ;
m_active_enbs . erase ( enb_it + + ) ;
}
std : : map < uint64_t , ue_ctx_t * > : : iterator ue_it = m_imsi_to_ue_ctx . begin ( ) ;
while ( ue_it ! = m_imsi_to_ue_ctx . end ( ) )
{
m_s1ap_log - > info ( " Deleting UE EMM context. IMSI: %015lu \n " , ue_it - > first ) ;
m_s1ap_log - > console ( " Deleting UE EMM context. IMSI: %015lu \n " , ue_it - > first ) ;
delete ue_it - > second ;
m_imsi_to_ue_ctx . erase ( ue_it + + ) ;
}
//Cleanup message handlers
s1ap_mngmt_proc : : cleanup ( ) ;
s1ap_nas_transport : : cleanup ( ) ;
s1ap_ctx_mngmt_proc : : cleanup ( ) ;
return ;
}
int
s1ap : : get_s1_mme ( )
{
return m_s1mme ;
}
uint32_t
s1ap : : get_next_mme_ue_s1ap_id ( )
{
return m_next_mme_ue_s1ap_id + + ;
}
int
s1ap : : enb_listen ( )
{
/*This function sets up the SCTP socket for eNBs to connect to*/
int sock_fd , err ;
struct sockaddr_in s1mme_addr ;
struct sctp_event_subscribe evnts ;
m_s1ap_log - > info ( " S1-MME Initializing \n " ) ;
sock_fd = socket ( AF_INET , SOCK_SEQPACKET , IPPROTO_SCTP ) ;
if ( sock_fd = = - 1 ) {
m_s1ap_log - > console ( " Could not create SCTP socket \n " ) ;
return - 1 ;
}
//Sets the data_io_event to be able to use sendrecv_info
//Subscribes to the SCTP_SHUTDOWN event, to handle graceful shutdown
bzero ( & evnts , sizeof ( evnts ) ) ;
evnts . sctp_data_io_event = 1 ;
evnts . sctp_shutdown_event = 1 ;
if ( setsockopt ( sock_fd , IPPROTO_SCTP , SCTP_EVENTS , & evnts , sizeof ( evnts ) ) ) {
close ( sock_fd ) ;
m_s1ap_log - > console ( " Subscribing to sctp_data_io_events failed \n " ) ;
return - 1 ;
}
//S1-MME bind
bzero ( & s1mme_addr , sizeof ( s1mme_addr ) ) ;
s1mme_addr . sin_family = AF_INET ;
inet_pton ( AF_INET , m_s1ap_args . mme_bind_addr . c_str ( ) , & ( s1mme_addr . sin_addr ) ) ;
s1mme_addr . sin_port = htons ( S1MME_PORT ) ;
err = bind ( sock_fd , ( struct sockaddr * ) & s1mme_addr , sizeof ( s1mme_addr ) ) ;
if ( err ! = 0 ) {
close ( sock_fd ) ;
m_s1ap_log - > error ( " Error binding SCTP socket \n " ) ;
m_s1ap_log - > console ( " Error binding SCTP socket \n " ) ;
return - 1 ;
}
//Listen for connections
err = listen ( sock_fd , SOMAXCONN ) ;
if ( err ! = 0 ) {
close ( sock_fd ) ;
m_s1ap_log - > error ( " Error in SCTP socket listen \n " ) ;
m_s1ap_log - > console ( " Error in SCTP socket listen \n " ) ;
return - 1 ;
}
return sock_fd ;
}
bool
s1ap : : handle_s1ap_rx_pdu ( srslte : : byte_buffer_t * pdu , struct sctp_sndrcvinfo * enb_sri )
{
LIBLTE_S1AP_S1AP_PDU_STRUCT rx_pdu ;
if ( liblte_s1ap_unpack_s1ap_pdu ( ( LIBLTE_BYTE_MSG_STRUCT * ) pdu , & rx_pdu ) ! = LIBLTE_SUCCESS ) {
m_s1ap_log - > error ( " Failed to unpack received PDU \n " ) ;
return false ;
}
switch ( rx_pdu . choice_type ) {
case LIBLTE_S1AP_S1AP_PDU_CHOICE_INITIATINGMESSAGE :
m_s1ap_log - > info ( " Received initiating PDU \n " ) ;
return handle_initiating_message ( & rx_pdu . choice . initiatingMessage , enb_sri ) ;
break ;
case LIBLTE_S1AP_S1AP_PDU_CHOICE_SUCCESSFULOUTCOME :
m_s1ap_log - > info ( " Received Succeseful Outcome PDU \n " ) ;
return handle_successful_outcome ( & rx_pdu . choice . successfulOutcome ) ;
break ;
case LIBLTE_S1AP_S1AP_PDU_CHOICE_UNSUCCESSFULOUTCOME :
m_s1ap_log - > info ( " Received Unsucceseful Outcome PDU \n " ) ;
return true ; //TODO handle_unsuccessfuloutcome(&rx_pdu.choice.unsuccessfulOutcome);
break ;
default :
m_s1ap_log - > error ( " Unhandled PDU type %d \n " , rx_pdu . choice_type ) ;
return false ;
}
return true ;
}
bool
s1ap : : handle_initiating_message ( LIBLTE_S1AP_INITIATINGMESSAGE_STRUCT * msg , struct sctp_sndrcvinfo * enb_sri )
{
bool reply_flag = false ;
srslte : : byte_buffer_t * reply_buffer = m_pool - > allocate ( ) ;
switch ( msg - > choice_type ) {
case LIBLTE_S1AP_INITIATINGMESSAGE_CHOICE_S1SETUPREQUEST :
m_s1ap_log - > info ( " Received S1 Setup Request. \n " ) ;
m_s1ap_mngmt_proc - > handle_s1_setup_request ( & msg - > choice . S1SetupRequest , enb_sri , reply_buffer , & reply_flag ) ;
break ;
case LIBLTE_S1AP_INITIATINGMESSAGE_CHOICE_INITIALUEMESSAGE :
m_s1ap_log - > info ( " Received Initial UE Message. \n " ) ;
m_s1ap_nas_transport - > handle_initial_ue_message ( & msg - > choice . InitialUEMessage , enb_sri , reply_buffer , & reply_flag ) ;
break ;
case LIBLTE_S1AP_INITIATINGMESSAGE_CHOICE_UPLINKNASTRANSPORT :
m_s1ap_log - > info ( " Received Uplink NAS Transport Message. \n " ) ;
m_s1ap_nas_transport - > handle_uplink_nas_transport ( & msg - > choice . UplinkNASTransport , enb_sri , reply_buffer , & reply_flag ) ;
break ;
case LIBLTE_S1AP_INITIATINGMESSAGE_CHOICE_UECONTEXTRELEASEREQUEST :
m_s1ap_log - > info ( " Received UE Context Release Request Message. \n " ) ;
m_s1ap_ctx_mngmt_proc - > handle_ue_context_release_request ( & msg - > choice . UEContextReleaseRequest , enb_sri , reply_buffer , & reply_flag ) ;
break ;
default :
m_s1ap_log - > error ( " Unhandled S1AP intiating message: %s \n " , liblte_s1ap_initiatingmessage_choice_text [ msg - > choice_type ] ) ;
m_s1ap_log - > console ( " Unhandled S1APintiating message: %s \n " , liblte_s1ap_initiatingmessage_choice_text [ msg - > choice_type ] ) ;
}
//Send Reply to eNB
if ( reply_flag = = true )
{
ssize_t n_sent = sctp_send ( m_s1mme , reply_buffer - > msg , reply_buffer - > N_bytes , enb_sri , 0 ) ;
if ( n_sent = = - 1 )
{
m_s1ap_log - > console ( " Failed to send S1AP Initiating Reply. \n " ) ;
m_s1ap_log - > error ( " Failed to send S1AP Initiating Reply. \n " ) ;
m_pool - > deallocate ( reply_buffer ) ;
return false ;
}
}
m_pool - > deallocate ( reply_buffer ) ;
return true ;
}
bool
s1ap : : handle_successful_outcome ( LIBLTE_S1AP_SUCCESSFULOUTCOME_STRUCT * msg )
{
switch ( msg - > choice_type ) {
case LIBLTE_S1AP_SUCCESSFULOUTCOME_CHOICE_INITIALCONTEXTSETUPRESPONSE :
m_s1ap_log - > info ( " Received Initial Context Setup Response. \n " ) ;
return m_s1ap_ctx_mngmt_proc - > handle_initial_context_setup_response ( & msg - > choice . InitialContextSetupResponse ) ;
case LIBLTE_S1AP_SUCCESSFULOUTCOME_CHOICE_UECONTEXTRELEASECOMPLETE :
m_s1ap_log - > info ( " Received UE Context Release Complete \n " ) ;
return m_s1ap_ctx_mngmt_proc - > handle_ue_context_release_complete ( & msg - > choice . UEContextReleaseComplete ) ;
default :
m_s1ap_log - > error ( " Unhandled successful outcome message: %s \n " , liblte_s1ap_successfuloutcome_choice_text [ msg - > choice_type ] ) ;
}
return true ;
}
//eNB Context Managment
void
s1ap : : add_new_enb_ctx ( const enb_ctx_t & enb_ctx , const struct sctp_sndrcvinfo * enb_sri )
{
m_s1ap_log - > info ( " Adding new eNB context. eNB ID %d \n " , enb_ctx . enb_id ) ;
std : : set < uint32_t > ue_set ;
enb_ctx_t * enb_ptr = new enb_ctx_t ;
memcpy ( enb_ptr , & enb_ctx , sizeof ( enb_ctx_t ) ) ;
m_active_enbs . insert ( std : : pair < uint16_t , enb_ctx_t * > ( enb_ptr - > enb_id , enb_ptr ) ) ;
m_sctp_to_enb_id . insert ( std : : pair < int32_t , uint16_t > ( enb_sri - > sinfo_assoc_id , enb_ptr - > enb_id ) ) ;
m_enb_assoc_to_ue_ids . insert ( std : : pair < int32_t , std : : set < uint32_t > > ( enb_sri - > sinfo_assoc_id , ue_set ) ) ;
return ;
}
enb_ctx_t *
s1ap : : find_enb_ctx ( uint16_t enb_id )
{
std : : map < uint16_t , enb_ctx_t * > : : iterator it = m_active_enbs . find ( enb_id ) ;
if ( it = = m_active_enbs . end ( ) )
{
return NULL ;
}
else
{
return it - > second ;
}
}
void
s1ap : : delete_enb_ctx ( int32_t assoc_id )
{
std : : map < int32_t , uint16_t > : : iterator it_assoc = m_sctp_to_enb_id . find ( assoc_id ) ;
uint16_t enb_id = it_assoc - > second ;
std : : map < uint16_t , enb_ctx_t * > : : iterator it_ctx = m_active_enbs . find ( enb_id ) ;
if ( it_ctx = = m_active_enbs . end ( ) | | it_assoc = = m_sctp_to_enb_id . end ( ) )
{
m_s1ap_log - > error ( " Could not find eNB to delete. Association: %d \n " , assoc_id ) ;
return ;
}
m_s1ap_log - > info ( " Deleting eNB context. eNB Id: 0x%x \n " , enb_id ) ;
m_s1ap_log - > console ( " Deleting eNB context. eNB Id: 0x%x \n " , enb_id ) ;
//Delete connected UEs ctx
release_ues_ecm_ctx_in_enb ( assoc_id ) ;
//Delete eNB
delete it_ctx - > second ;
m_active_enbs . erase ( it_ctx ) ;
m_sctp_to_enb_id . erase ( it_assoc ) ;
return ;
}
//UE Context Management
bool
s1ap : : add_ue_ctx_to_imsi_map ( ue_ctx_t * ue_ctx )
{
std : : map < uint64_t , ue_ctx_t * > : : iterator ctx_it = m_imsi_to_ue_ctx . find ( ue_ctx - > emm_ctx . imsi ) ;
if ( ctx_it ! = m_imsi_to_ue_ctx . end ( ) )
{
m_s1ap_log - > error ( " UE Context already exists. IMSI %015lu " , ue_ctx - > emm_ctx . imsi ) ;
return false ;
}
if ( ue_ctx - > ecm_ctx . mme_ue_s1ap_id ! = 0 )
{
std : : map < uint32_t , ue_ctx_t * > : : iterator ctx_it2 = m_mme_ue_s1ap_id_to_ue_ctx . find ( ue_ctx - > ecm_ctx . mme_ue_s1ap_id ) ;
if ( ctx_it2 ! = m_mme_ue_s1ap_id_to_ue_ctx . end ( ) & & ctx_it2 - > second ! = ue_ctx )
{
m_s1ap_log - > error ( " Context identified with IMSI does not match context identified by MME UE S1AP Id. \n " ) ;
return false ;
}
}
m_imsi_to_ue_ctx . insert ( std : : pair < uint64_t , ue_ctx_t * > ( ue_ctx - > emm_ctx . imsi , ue_ctx ) ) ;
m_s1ap_log - > debug ( " Saved UE context corresponding to IMSI %015lu \n " , ue_ctx - > emm_ctx . imsi ) ;
return true ;
}
bool
s1ap : : add_ue_ctx_to_mme_ue_s1ap_id_map ( ue_ctx_t * ue_ctx )
{
if ( ue_ctx - > ecm_ctx . mme_ue_s1ap_id = = 0 )
{
m_s1ap_log - > error ( " Could not add UE context to MME UE S1AP map. MME UE S1AP ID 0 is not valid. " ) ;
return false ;
}
std : : map < uint32_t , ue_ctx_t * > : : iterator ctx_it = m_mme_ue_s1ap_id_to_ue_ctx . find ( ue_ctx - > ecm_ctx . mme_ue_s1ap_id ) ;
if ( ctx_it ! = m_mme_ue_s1ap_id_to_ue_ctx . end ( ) )
{
m_s1ap_log - > error ( " UE Context already exists. MME UE S1AP Id %015lu " , ue_ctx - > emm_ctx . imsi ) ;
return false ;
}
if ( ue_ctx - > ecm_ctx . imsi ! = 0 )
{
std : : map < uint32_t , ue_ctx_t * > : : iterator ctx_it2 = m_mme_ue_s1ap_id_to_ue_ctx . find ( ue_ctx - > ecm_ctx . mme_ue_s1ap_id ) ;
if ( ctx_it2 ! = m_mme_ue_s1ap_id_to_ue_ctx . end ( ) & & ctx_it2 - > second ! = ue_ctx )
{
m_s1ap_log - > error ( " Context identified with MME UE S1AP Id does not match context identified by IMSI. \n " ) ;
return false ;
}
}
m_mme_ue_s1ap_id_to_ue_ctx . insert ( std : : pair < uint32_t , ue_ctx_t * > ( ue_ctx - > ecm_ctx . mme_ue_s1ap_id , ue_ctx ) ) ;
m_s1ap_log - > debug ( " Saved UE context corresponding to MME UE S1AP Id %d \n " , ue_ctx - > ecm_ctx . mme_ue_s1ap_id ) ;
return true ;
}
bool
s1ap : : add_ue_to_enb_set ( int32_t enb_assoc , uint32_t mme_ue_s1ap_id )
{
std : : map < int32_t , std : : set < uint32_t > > : : iterator ues_in_enb = m_enb_assoc_to_ue_ids . find ( enb_assoc ) ;
if ( ues_in_enb = = m_enb_assoc_to_ue_ids . end ( ) )
{
m_s1ap_log - > error ( " Could not find eNB from eNB SCTP association %d " , enb_assoc ) ;
return false ;
}
std : : set < uint32_t > : : iterator ue_id = ues_in_enb - > second . find ( mme_ue_s1ap_id ) ;
if ( ue_id ! = ues_in_enb - > second . end ( ) )
{
m_s1ap_log - > error ( " UE with MME UE S1AP Id already exists %d " , mme_ue_s1ap_id ) ;
return false ;
}
ues_in_enb - > second . insert ( mme_ue_s1ap_id ) ;
m_s1ap_log - > debug ( " Added UE with MME-UE S1AP Id %d to eNB with association %d \n " , mme_ue_s1ap_id , enb_assoc ) ;
return true ;
}
ue_ctx_t *
s1ap : : find_ue_ctx_from_mme_ue_s1ap_id ( uint32_t mme_ue_s1ap_id )
{
std : : map < uint32_t , ue_ctx_t * > : : iterator it = m_mme_ue_s1ap_id_to_ue_ctx . find ( mme_ue_s1ap_id ) ;
if ( it = = m_mme_ue_s1ap_id_to_ue_ctx . end ( ) )
{
return NULL ;
}
else
{
return it - > second ;
}
}
ue_ctx_t *
s1ap : : find_ue_ctx_from_imsi ( uint64_t imsi )
{
std : : map < uint64_t , ue_ctx_t * > : : iterator it = m_imsi_to_ue_ctx . find ( imsi ) ;
if ( it = = m_imsi_to_ue_ctx . end ( ) )
{
return NULL ;
}
else
{
return it - > second ;
}
}
void
s1ap : : release_ues_ecm_ctx_in_enb ( int32_t enb_assoc )
{
m_s1ap_log - > console ( " Releasing UEs context \n " ) ;
//delete UEs ctx
std : : map < int32_t , std : : set < uint32_t > > : : iterator ues_in_enb = m_enb_assoc_to_ue_ids . find ( enb_assoc ) ;
std : : set < uint32_t > : : iterator ue_id = ues_in_enb - > second . begin ( ) ;
if ( ue_id = = ues_in_enb - > second . end ( ) )
{
m_s1ap_log - > console ( " No UEs to be released \n " ) ;
} else {
while ( ue_id ! = ues_in_enb - > second . end ( ) )
{
std : : map < uint32_t , ue_ctx_t * > : : iterator ue_ctx = m_mme_ue_s1ap_id_to_ue_ctx . find ( * ue_id ) ;
ue_emm_ctx_t * emm_ctx = & ue_ctx - > second - > emm_ctx ;
ue_ecm_ctx_t * ecm_ctx = & ue_ctx - > second - > ecm_ctx ;
m_s1ap_log - > info ( " Releasing UE context. IMSI: %015lu, UE-MME S1AP Id: %d \n " , emm_ctx - > imsi , ecm_ctx - > mme_ue_s1ap_id ) ;
if ( emm_ctx - > state = = EMM_STATE_REGISTERED )
{
m_mme_gtpc - > send_delete_session_request ( emm_ctx - > imsi ) ;
emm_ctx - > state = EMM_STATE_DEREGISTERED ;
}
m_s1ap_log - > console ( " Releasing UE ECM context. UE-MME S1AP Id: %d \n " , ecm_ctx - > mme_ue_s1ap_id ) ;
ecm_ctx - > state = ECM_STATE_IDLE ;
ecm_ctx - > mme_ue_s1ap_id = 0 ;
ecm_ctx - > enb_ue_s1ap_id = 0 ;
ues_in_enb - > second . erase ( ue_id + + ) ;
}
}
}
bool
s1ap : : release_ue_ecm_ctx ( uint32_t mme_ue_s1ap_id )
{
ue_ctx_t * ue_ctx = find_ue_ctx_from_mme_ue_s1ap_id ( mme_ue_s1ap_id ) ;
if ( ue_ctx = = NULL )
{
m_s1ap_log - > error ( " Cannot release UE ECM context, UE not found. MME-UE S1AP Id: %d \n " , mme_ue_s1ap_id ) ;
return false ;
}
ue_ecm_ctx_t * ecm_ctx = & ue_ctx - > ecm_ctx ;
//Delete UE within eNB UE set
std : : map < int32_t , uint16_t > : : iterator it = m_sctp_to_enb_id . find ( ecm_ctx - > enb_sri . sinfo_assoc_id ) ;
if ( it = = m_sctp_to_enb_id . end ( ) )
{
m_s1ap_log - > error ( " Could not find eNB for UE release request. \n " ) ;
return false ;
}
uint16_t enb_id = it - > second ;
std : : map < int32_t , std : : set < uint32_t > > : : iterator ue_set = m_enb_assoc_to_ue_ids . find ( ecm_ctx - > enb_sri . sinfo_assoc_id ) ;
if ( ue_set = = m_enb_assoc_to_ue_ids . end ( ) )
{
m_s1ap_log - > error ( " Could not find the eNB's UEs. \n " ) ;
return false ;
}
ue_set - > second . erase ( mme_ue_s1ap_id ) ;
//Release UE ECM context
m_mme_ue_s1ap_id_to_ue_ctx . erase ( mme_ue_s1ap_id ) ;
ecm_ctx - > state = ECM_STATE_IDLE ;
ecm_ctx - > mme_ue_s1ap_id = 0 ;
ecm_ctx - > enb_ue_s1ap_id = 0 ;
m_s1ap_log - > info ( " Released UE ECM Context. \n " ) ;
return true ;
}
bool
s1ap : : delete_ue_ctx ( uint64_t imsi )
{
ue_ctx_t * ue_ctx = find_ue_ctx_from_imsi ( imsi ) ;
if ( ue_ctx = = NULL )
{
m_s1ap_log - > info ( " Cannot delete UE context, UE not found. IMSI: % " PRIu64 " \n " , imsi ) ;
return false ;
}
//Make sure to release ECM ctx
if ( ue_ctx - > ecm_ctx . mme_ue_s1ap_id ! = 0 )
{
release_ue_ecm_ctx ( ue_ctx - > ecm_ctx . mme_ue_s1ap_id ) ;
}
//Delete UE context
m_imsi_to_ue_ctx . erase ( imsi ) ;
delete ue_ctx ;
m_s1ap_log - > info ( " Deleted UE Context. \n " ) ;
return true ;
}
//UE Bearer Managment
void
s1ap : : activate_eps_bearer ( uint64_t imsi , uint8_t ebi )
{
std : : map < uint64_t , ue_ctx_t * > : : iterator ue_ctx_it = m_imsi_to_ue_ctx . find ( imsi ) ;
if ( ue_ctx_it = = m_imsi_to_ue_ctx . end ( ) )
{
m_s1ap_log - > error ( " Could not activate EPS bearer: Could not find UE context \n " ) ;
return ;
}
//Make sure NAS is active
uint32_t mme_ue_s1ap_id = ue_ctx_it - > second - > ecm_ctx . mme_ue_s1ap_id ;
std : : map < uint32_t , ue_ctx_t * > : : iterator it = m_mme_ue_s1ap_id_to_ue_ctx . find ( mme_ue_s1ap_id ) ;
if ( it = = m_mme_ue_s1ap_id_to_ue_ctx . end ( ) )
{
m_s1ap_log - > error ( " Could not activate EPS bearer: ECM context seems to be missing \n " ) ;
return ;
}
ue_ecm_ctx_t * ecm_ctx = & ue_ctx_it - > second - > ecm_ctx ;
if ( ecm_ctx - > erabs_ctx [ ebi ] . state ! = ERAB_CTX_SETUP )
{
m_s1ap_log - > error ( " Could not be activate EPS Bearer, bearer in wrong state: MME S1AP Id %d, EPS Bearer id %d, state %d \n " , mme_ue_s1ap_id , ebi , ecm_ctx - > erabs_ctx [ ebi ] . state ) ;
m_s1ap_log - > console ( " Could not be activate EPS Bearer, bearer in wrong state: MME S1AP Id %d, EPS Bearer id %d, state %d \n " , mme_ue_s1ap_id , ebi , ecm_ctx - > erabs_ctx [ ebi ] . state ) ;
return ;
}
ecm_ctx - > erabs_ctx [ ebi ] . state = ERAB_ACTIVE ;
ecm_ctx - > state = ECM_STATE_CONNECTED ;
m_s1ap_log - > info ( " Activated EPS Bearer: Bearer id %d \n " , ebi ) ;
return ;
}
uint32_t
s1ap : : allocate_m_tmsi ( uint64_t imsi )
{
uint32_t m_tmsi = m_next_m_tmsi ;
m_next_m_tmsi = ( m_next_m_tmsi + 1 ) % UINT32_MAX ;
m_tmsi_to_imsi . insert ( std : : pair < uint32_t , uint64_t > ( m_tmsi , imsi ) ) ;
m_s1ap_log - > debug ( " Allocated M-TMSI 0x%x to IMSI %015lu, \n " , m_tmsi , imsi ) ;
return m_tmsi ;
}
void
s1ap : : print_enb_ctx_info ( const std : : string & prefix , const enb_ctx_t & enb_ctx )
{
std : : string mnc_str , mcc_str ;
if ( enb_ctx . enb_name_present )
{
m_s1ap_log - > console ( " %s - eNB Name: %s, eNB id: 0x%x \n " , prefix . c_str ( ) , enb_ctx . enb_name , enb_ctx . enb_id ) ;
m_s1ap_log - > info ( " %s - eNB Name: %s, eNB id: 0x%x \n " , prefix . c_str ( ) , enb_ctx . enb_name , enb_ctx . enb_id ) ;
}
else
{
m_s1ap_log - > console ( " %s - eNB Id 0x%x \n " , prefix . c_str ( ) , enb_ctx . enb_id ) ;
m_s1ap_log - > info ( " %s - eNB Id 0x%x \n " , prefix . c_str ( ) , enb_ctx . enb_id ) ;
}
srslte : : mcc_to_string ( enb_ctx . mcc , & mcc_str ) ;
srslte : : mnc_to_string ( enb_ctx . mnc , & mnc_str ) ;
m_s1ap_log - > info ( " %s - MCC:%s, MNC:%s, PLMN: %d \n " , prefix . c_str ( ) , mcc_str . c_str ( ) , mnc_str . c_str ( ) , enb_ctx . plmn ) ;
m_s1ap_log - > console ( " %s - MCC:%s, MNC:%s, PLMN: %d \n " , prefix . c_str ( ) , mcc_str . c_str ( ) , mnc_str . c_str ( ) , enb_ctx . plmn ) ;
for ( int i = 0 ; i < enb_ctx . nof_supported_ta ; i + + )
{
for ( int j = 0 ; i < enb_ctx . nof_supported_ta ; i + + )
{
m_s1ap_log - > info ( " %s - TAC %d, B-PLMN %d \n " , prefix . c_str ( ) , enb_ctx . tac [ i ] , enb_ctx . bplmns [ i ] [ j ] ) ;
m_s1ap_log - > console ( " %s - TAC %d, B-PLMN %d \n " , prefix . c_str ( ) , enb_ctx . tac [ i ] , enb_ctx . bplmns [ i ] [ j ] ) ;
}
}
m_s1ap_log - > console ( " %s - Paging DRX %d \n " , prefix . c_str ( ) , enb_ctx . drx ) ;
return ;
}
} //namespace srsepc