@ -69,6 +69,7 @@ proc_outcome_t nas::plmn_search_proc::step()
if ( ret . is_success ( ) ) {
if ( ret . is_success ( ) ) {
return proc_outcome_t : : success ;
return proc_outcome_t : : success ;
}
}
nas_ptr - > enter_emm_deregistered ( ) ;
return proc_outcome_t : : error ;
return proc_outcome_t : : error ;
}
}
return proc_outcome_t : : yield ;
return proc_outcome_t : : yield ;
@ -114,7 +115,7 @@ proc_outcome_t nas::plmn_search_proc::trigger_event(const plmn_search_complete_t
nas_ptr - > rrc - > plmn_select ( nas_ptr - > current_plmn ) ;
nas_ptr - > rrc - > plmn_select ( nas_ptr - > current_plmn ) ;
if ( not nas_ptr - > rrc_connector . launch ( nas_ptr , nullptr ) ) {
if ( not nas_ptr - > rrc_connector . launch ( nas_ptr , srslte : : establishment_cause_t : : mo_data , nullptr ) ) {
Error ( " Unable to initiate RRC connection. \n " ) ;
Error ( " Unable to initiate RRC connection. \n " ) ;
return proc_outcome_t : : error ;
return proc_outcome_t : : error ;
}
}
@ -123,7 +124,8 @@ proc_outcome_t nas::plmn_search_proc::trigger_event(const plmn_search_complete_t
return proc_outcome_t : : yield ;
return proc_outcome_t : : yield ;
}
}
proc_outcome_t nas : : rrc_connect_proc : : init ( nas * nas_ptr_ , srslte : : unique_byte_buffer_t pdu )
proc_outcome_t
nas : : rrc_connect_proc : : init ( nas * nas_ptr_ , srslte : : establishment_cause_t cause_ , srslte : : unique_byte_buffer_t pdu )
{
{
nas_ptr = nas_ptr_ ;
nas_ptr = nas_ptr_ ;
@ -155,14 +157,8 @@ proc_outcome_t nas::rrc_connect_proc::init(nas* nas_ptr_, srslte::unique_byte_bu
nas_ptr - > rrc - > set_ue_identity ( s_tmsi ) ;
nas_ptr - > rrc - > set_ue_identity ( s_tmsi ) ;
}
}
establishment_cause_t establish_cause = establishment_cause_t : : mo_sig ;
if ( nas_ptr - > state = = EMM_STATE_REGISTERED ) {
// FIXME: only need to use MT_ACCESS for establishment after paging
establish_cause = establishment_cause_t : : mt_access ;
}
state = state_t : : conn_req ;
state = state_t : : conn_req ;
if ( not nas_ptr - > start_connection_request ( establish_ cause, std : : move ( pdu ) ) ) {
if ( not nas_ptr - > start_connection_request ( cause_ , std : : move ( pdu ) ) ) {
return proc_outcome_t : : error ;
return proc_outcome_t : : error ;
}
}
@ -303,7 +299,6 @@ void nas::start_attach_request(srslte::proc_state_t* result)
nas_log - > info ( " Attach Request \n " ) ;
nas_log - > info ( " Attach Request \n " ) ;
switch ( state ) {
switch ( state ) {
case EMM_STATE_DEREGISTERED :
case EMM_STATE_DEREGISTERED :
// Search PLMN is not selected
// Search PLMN is not selected
if ( ! plmn_is_selected ) {
if ( ! plmn_is_selected ) {
nas_log - > info ( " No PLMN selected. Starting PLMN Search... \n " ) ;
nas_log - > info ( " No PLMN selected. Starting PLMN Search... \n " ) ;
@ -318,6 +313,10 @@ void nas::start_attach_request(srslte::proc_state_t* result)
plmn_search_proc p = plmn_searcher . pop ( ) ;
plmn_search_proc p = plmn_searcher . pop ( ) ;
nas_log - > info ( " Attach Request from PLMN Search %s \n " , p . is_success ( ) ? " finished successfully " : " failed " ) ;
nas_log - > info ( " Attach Request from PLMN Search %s \n " , p . is_success ( ) ? " finished successfully " : " failed " ) ;
* result = p . is_success ( ) ? proc_state_t : : success : proc_state_t : : error ;
* result = p . is_success ( ) ? proc_state_t : : success : proc_state_t : : error ;
// stay in this state if attach failed
if ( not p . is_success ( ) ) {
enter_emm_deregistered ( ) ;
}
return proc_outcome_t : : success ;
return proc_outcome_t : : success ;
} ) ;
} ) ;
} else {
} else {
@ -331,7 +330,7 @@ void nas::start_attach_request(srslte::proc_state_t* result)
* result = proc_state_t : : success ;
* result = proc_state_t : : success ;
} else {
} else {
nas_log - > info ( " NAS is already registered but RRC disconnected. Connecting now... \n " ) ;
nas_log - > info ( " NAS is already registered but RRC disconnected. Connecting now... \n " ) ;
if ( not rrc_connector . launch ( this , nullptr ) ) {
if ( not rrc_connector . launch ( this , srslte : : establishment_cause_t : : mo_data , nullptr ) ) {
nas_log - > error ( " Cannot initiate concurrent rrc connection procedures \n " ) ;
nas_log - > error ( " Cannot initiate concurrent rrc connection procedures \n " ) ;
* result = proc_state_t : : error ;
* result = proc_state_t : : error ;
return ;
return ;
@ -363,7 +362,8 @@ void nas::plmn_search_completed(rrc_interface_nas::found_plmn_t found_plmns[rrc_
plmn_searcher . trigger_event ( plmn_search_proc : : plmn_search_complete_t ( found_plmns , nof_plmns ) ) ;
plmn_searcher . trigger_event ( plmn_search_proc : : plmn_search_complete_t ( found_plmns , nof_plmns ) ) ;
}
}
bool nas : : detach_request ( ) {
bool nas : : detach_request ( const bool switch_off )
{
// attempt detach for 5s
// attempt detach for 5s
nas_log - > info ( " Detach Request \n " ) ;
nas_log - > info ( " Detach Request \n " ) ;
@ -373,9 +373,7 @@ bool nas::detach_request() {
break ;
break ;
case EMM_STATE_REGISTERED :
case EMM_STATE_REGISTERED :
// send detach request
// send detach request
send_detach_request ( true ) ;
send_detach_request ( switch_off ) ;
plmn_is_selected = false ;
state = EMM_STATE_DEREGISTERED ;
break ;
break ;
case EMM_STATE_DEREGISTERED_INITIATED :
case EMM_STATE_DEREGISTERED_INITIATED :
// do nothing ..
// do nothing ..
@ -386,7 +384,18 @@ bool nas::detach_request() {
return false ;
return false ;
}
}
void nas : : leave_connected ( )
void nas : : enter_emm_deregistered ( )
{
// Deactivate EPS bearer according to Sec. 5.5.2.2.2
nas_log - > info ( " Clearing EPS bearer context. \n " ) ;
eps_bearer . clear ( ) ;
plmn_is_selected = false ;
state = EMM_STATE_DEREGISTERED ;
}
void nas : : left_rrc_connected ( )
{
{
return ;
return ;
}
}
@ -404,7 +413,7 @@ void nas::paging(s_tmsi_t* ue_identity)
nas_log - > error ( " Cannot initiate concurrent RRC connection establishment procedures \n " ) ;
nas_log - > error ( " Cannot initiate concurrent RRC connection establishment procedures \n " ) ;
return ;
return ;
}
}
if ( not rrc_connector . launch ( this , nullptr ) ) {
if ( not rrc_connector . launch ( this , srslte : : establishment_cause_t : : mt_access , nullptr ) ) {
nas_log - > error ( " Could not launch RRC Connect() \n " ) ;
nas_log - > error ( " Could not launch RRC Connect() \n " ) ;
return ;
return ;
}
}
@ -417,7 +426,6 @@ void nas::paging(s_tmsi_t* ue_identity)
rrc - > paging_completed ( success ) ;
rrc - > paging_completed ( success ) ;
return proc_outcome_t : : success ;
return proc_outcome_t : : success ;
} ) ;
} ) ;
} else {
} else {
nas_log - > warning ( " Received paging while in state %s \n " , emm_state_text [ state ] ) ;
nas_log - > warning ( " Received paging while in state %s \n " , emm_state_text [ state ] ) ;
}
}
@ -1034,7 +1042,7 @@ void nas::parse_attach_accept(uint32_t lcid, unique_byte_buffer_t pdu)
} else {
} else {
nas_log - > info ( " Not handling attach type %u \n " , attach_accept . eps_attach_result ) ;
nas_log - > info ( " Not handling attach type %u \n " , attach_accept . eps_attach_result ) ;
state = EMM_STATE_DEREGISTERED ;
enter_emm_deregistered( ) ;
}
}
ctxt . rx_count + + ;
ctxt . rx_count + + ;
@ -1048,7 +1056,7 @@ void nas::parse_attach_reject(uint32_t lcid, unique_byte_buffer_t pdu)
liblte_mme_unpack_attach_reject_msg ( ( LIBLTE_BYTE_MSG_STRUCT * ) pdu . get ( ) , & attach_rej ) ;
liblte_mme_unpack_attach_reject_msg ( ( LIBLTE_BYTE_MSG_STRUCT * ) pdu . get ( ) , & attach_rej ) ;
nas_log - > warning ( " Received Attach Reject. Cause= %02X \n " , attach_rej . emm_cause ) ;
nas_log - > warning ( " Received Attach Reject. Cause= %02X \n " , attach_rej . emm_cause ) ;
nas_log - > console ( " Received Attach Reject. Cause= %02X \n " , attach_rej . emm_cause ) ;
nas_log - > console ( " Received Attach Reject. Cause= %02X \n " , attach_rej . emm_cause ) ;
state = EMM_STATE_DEREGISTERED ;
enter_emm_deregistered( ) ;
// FIXME: Command RRC to release?
// FIXME: Command RRC to release?
}
}
@ -1101,7 +1109,7 @@ void nas::parse_authentication_request(uint32_t lcid, unique_byte_buffer_t pdu,
void nas : : parse_authentication_reject ( uint32_t lcid , unique_byte_buffer_t pdu )
void nas : : parse_authentication_reject ( uint32_t lcid , unique_byte_buffer_t pdu )
{
{
nas_log - > warning ( " Received Authentication Reject \n " ) ;
nas_log - > warning ( " Received Authentication Reject \n " ) ;
state = EMM_STATE_DEREGISTERED ;
enter_emm_deregistered( ) ;
// FIXME: Command RRC to release?
// FIXME: Command RRC to release?
}
}
@ -1256,8 +1264,7 @@ void nas::parse_service_reject(uint32_t lcid, unique_byte_buffer_t pdu)
// FIXME: handle NAS backoff-timers correctly
// FIXME: handle NAS backoff-timers correctly
// Mark state as EMM-DEREGISTERED
enter_emm_deregistered ( ) ;
state = EMM_STATE_DEREGISTERED ;
// Reset security context
// Reset security context
ctxt = { } ;
ctxt = { } ;
@ -1297,13 +1304,19 @@ void nas::parse_detach_request(uint32_t lcid, unique_byte_buffer_t pdu)
liblte_mme_unpack_detach_request_msg ( ( LIBLTE_BYTE_MSG_STRUCT * ) pdu . get ( ) , & detach_request ) ;
liblte_mme_unpack_detach_request_msg ( ( LIBLTE_BYTE_MSG_STRUCT * ) pdu . get ( ) , & detach_request ) ;
ctxt . rx_count + + ;
ctxt . rx_count + + ;
if ( state = = EMM_STATE_REGISTERED ) {
switch ( state ) {
nas_log - > info ( " Received Detach request (type=%d) \n " , detach_request . detach_type . type_of_detach ) ;
case EMM_STATE_DEREGISTERED_INITIATED :
state = EMM_STATE_DEREGISTERED ;
nas_log - > info ( " Received detach from network while performing UE initiated detach. Aborting UE detach. \n " ) ;
// send accept
case EMM_STATE_REGISTERED :
send_detach_accept ( ) ;
nas_log - > info ( " Received detach request (type=%d) \n " , detach_request . detach_type . type_of_detach ) ;
} else {
nas_log - > warning ( " Received detach request in invalid state (state=%d) \n " , state ) ;
// send accept and leave state
send_detach_accept ( ) ;
enter_emm_deregistered ( ) ;
break ;
default :
nas_log - > warning ( " Received detach request in invalid state (%s) \n " , emm_state_text [ state ] ) ;
break ;
}
}
}
}
@ -1651,7 +1664,7 @@ void nas::send_detach_request(bool switch_off)
detach_request . detach_type . type_of_detach = LIBLTE_MME_SO_FLAG_SWITCH_OFF ;
detach_request . detach_type . type_of_detach = LIBLTE_MME_SO_FLAG_SWITCH_OFF ;
} else {
} else {
detach_request . detach_type . switch_off = 0 ;
detach_request . detach_type . switch_off = 0 ;
detach_request . detach_type . type_of_detach = LIBLTE_MME_ SO_FLAG_NORMAL _DETACH;
detach_request . detach_type . type_of_detach = LIBLTE_MME_ TOD_UL_EPS _DETACH;
}
}
// GUTI or IMSI detach
// GUTI or IMSI detach
@ -1701,17 +1714,17 @@ void nas::send_detach_request(bool switch_off)
}
}
if ( switch_off ) {
if ( switch_off ) {
// Deactivate EPS bearer according to Sec. 5.5.2.2.2
enter_emm_deregistered ( ) ;
nas_log - > info ( " Clearing EPS bearer context. \n " ) ;
} else {
eps_bearer . clear ( ) ;
// we are expecting a response from the core
state = EMM_STATE_DEREGISTERED_INITIATED ;
}
}
nas_log - > info ( " Sending detach request \n " ) ;
nas_log - > info ( " Sending detach request \n " ) ;
if ( rrc - > is_connected ( ) ) {
if ( rrc - > is_connected ( ) ) {
rrc - > write_sdu ( std : : move ( pdu ) ) ;
rrc - > write_sdu ( std : : move ( pdu ) ) ;
} else {
} else {
if ( not rrc_connector . launch ( this , establishment_cause_t : : mo_sig , std : : move ( pdu ) ) ) {
if ( not rrc_connector . launch ( this , std : : move ( pdu ) ) ) {
nas_log - > error ( " Failed to initiate RRC Connection Request \n " ) ;
nas_log - > error ( " Failed to initiate RRC Connection Request \n " ) ;
}
}
callbacks . defer_proc ( rrc_connector ) ;
callbacks . defer_proc ( rrc_connector ) ;