/* * Copyright 2013-2020 Software Radio Systems Limited * * 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 "srsepc/hdr/mme/s1ap.h" #include "srslte/asn1/gtpc.h" #include "srslte/common/bcd_helpers.h" #include "srslte/common/liblte_security.h" #include #include // for printing uint64_t 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) { return; } s1ap::~s1ap() { return; } 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* nas_log, srslte::log_filter* s1ap_log) { 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_nas_log = nas_log; m_s1ap_log = s1ap_log; // Get pointer to the HSS m_hss = hss::get_instance(); // 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_s1ap_ctx_mngmt_proc = s1ap_ctx_mngmt_proc::get_instance(); // Context Management Procedures m_s1ap_ctx_mngmt_proc->init(); m_s1ap_paging = s1ap_paging::get_instance(); // Paging m_s1ap_paging->init(); // Get pointer to GTP-C class m_mme_gtpc = mme_gtpc::get_instance(); // Initialize S1-MME m_s1mme = enb_listen(); // Init PCAP m_pcap_enable = s1ap_args.pcap_enable; if (m_pcap_enable) { m_pcap.open(s1ap_args.pcap_filename.c_str()); } m_s1ap_log->info("S1AP Initialized\n"); return 0; } void s1ap::stop() { if (m_s1mme != -1) { close(m_s1mme); } std::map::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); srslte::out_stream("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::iterator ue_it = m_imsi_to_nas_ctx.begin(); while (ue_it != m_imsi_to_nas_ctx.end()) { m_s1ap_log->info("Deleting UE EMM context. IMSI: %015" PRIu64 "\n", ue_it->first); srslte::out_stream("Deleting UE EMM context. IMSI: %015" PRIu64 "\n", ue_it->first); delete ue_it->second; m_imsi_to_nas_ctx.erase(ue_it++); } // Cleanup message handlers s1ap_mngmt_proc::cleanup(); s1ap_nas_transport::cleanup(); s1ap_ctx_mngmt_proc::cleanup(); // PCAP if (m_pcap_enable) { m_pcap.close(); } 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) { srslte::out_stream("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); srslte::out_stream("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"); srslte::out_stream("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"); srslte::out_stream("Error in SCTP socket listen\n"); return -1; } return sock_fd; } bool s1ap::s1ap_tx_pdu(const asn1::s1ap::s1ap_pdu_c& pdu, struct sctp_sndrcvinfo* enb_sri) { m_s1ap_log->debug("Transmitting S1AP PDU. eNB SCTP association Id: %d\n", enb_sri->sinfo_assoc_id); srslte::unique_byte_buffer_t buf = srslte::allocate_unique_buffer(*m_pool); if (buf == nullptr) { m_s1ap_log->error("Fatal Error: Couldn't allocate buffer for S1AP PDU.\n"); return false; } asn1::bit_ref bref(buf->msg, buf->get_tailroom()); if (pdu.pack(bref) != asn1::SRSASN_SUCCESS) { m_s1ap_log->error("Could not pack S1AP PDU correctly.\n"); return false; } buf->N_bytes = bref.distance_bytes(); ssize_t n_sent = sctp_send(m_s1mme, buf->msg, buf->N_bytes, enb_sri, MSG_NOSIGNAL); if (n_sent == -1) { srslte::out_stream("Failed to send S1AP PDU. Error: %s\n", strerror(errno)); m_s1ap_log->error("Failed to send S1AP PDU. Error: %s \n", strerror(errno)); return false; } if (m_pcap_enable) { m_pcap.write_s1ap(buf->msg, buf->N_bytes); } return true; } void s1ap::handle_s1ap_rx_pdu(srslte::byte_buffer_t* pdu, struct sctp_sndrcvinfo* enb_sri) { // Save PCAP if (m_pcap_enable) { m_pcap.write_s1ap(pdu->msg, pdu->N_bytes); } // Get PDU type s1ap_pdu_t rx_pdu; asn1::cbit_ref bref(pdu->msg, pdu->N_bytes); if (rx_pdu.unpack(bref) != asn1::SRSASN_SUCCESS) { m_s1ap_log->error("Failed to unpack received PDU\n"); return; } switch (rx_pdu.type().value) { case s1ap_pdu_t::types_opts::init_msg: m_s1ap_log->info("Received Initiating PDU\n"); handle_initiating_message(rx_pdu.init_msg(), enb_sri); break; case s1ap_pdu_t::types_opts::successful_outcome: m_s1ap_log->info("Received Succeseful Outcome PDU\n"); handle_successful_outcome(rx_pdu.successful_outcome()); break; case s1ap_pdu_t::types_opts::unsuccessful_outcome: m_s1ap_log->info("Received Unsucceseful Outcome PDU\n"); // TODO handle_unsuccessfuloutcome(&rx_pdu.choice.unsuccessfulOutcome); break; default: m_s1ap_log->error("Unhandled PDU type %d\n", rx_pdu.type().value); } } void s1ap::handle_initiating_message(const asn1::s1ap::init_msg_s& msg, struct sctp_sndrcvinfo* enb_sri) { using init_msg_type_opts_t = asn1::s1ap::s1ap_elem_procs_o::init_msg_c::types_opts; switch (msg.value.type().value) { case init_msg_type_opts_t::s1_setup_request: m_s1ap_log->info("Received S1 Setup Request.\n"); m_s1ap_mngmt_proc->handle_s1_setup_request(msg.value.s1_setup_request(), enb_sri); break; case init_msg_type_opts_t::init_ue_msg: m_s1ap_log->info("Received Initial UE Message.\n"); m_s1ap_nas_transport->handle_initial_ue_message(msg.value.init_ue_msg(), enb_sri); break; case init_msg_type_opts_t::ul_nas_transport: m_s1ap_log->info("Received Uplink NAS Transport Message.\n"); m_s1ap_nas_transport->handle_uplink_nas_transport(msg.value.ul_nas_transport(), enb_sri); break; case init_msg_type_opts_t::ue_context_release_request: m_s1ap_log->info("Received UE Context Release Request Message.\n"); m_s1ap_ctx_mngmt_proc->handle_ue_context_release_request(msg.value.ue_context_release_request(), enb_sri); break; default: m_s1ap_log->error("Unhandled S1AP intiating message: %s\n", msg.value.type().to_string().c_str()); srslte::out_stream("Unhandled S1APintiating message: %s\n", msg.value.type().to_string().c_str()); } } void s1ap::handle_successful_outcome(const asn1::s1ap::successful_outcome_s& msg) { using successful_outcome_type_opts_t = asn1::s1ap::s1ap_elem_procs_o::successful_outcome_c::types_opts; switch (msg.value.type().value) { case successful_outcome_type_opts_t::init_context_setup_resp: m_s1ap_log->info("Received Initial Context Setup Response.\n"); m_s1ap_ctx_mngmt_proc->handle_initial_context_setup_response(msg.value.init_context_setup_resp()); break; case successful_outcome_type_opts_t::ue_context_release_complete: m_s1ap_log->info("Received UE Context Release Complete\n"); m_s1ap_ctx_mngmt_proc->handle_ue_context_release_complete(msg.value.ue_context_release_complete()); break; default: m_s1ap_log->error("Unhandled successful outcome message: %s\n", msg.value.type().to_string().c_str()); } } // 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 ue_set; enb_ctx_t* enb_ptr = new enb_ctx_t; *enb_ptr = enb_ctx; m_active_enbs.insert(std::pair(enb_ptr->enb_id, enb_ptr)); m_sctp_to_enb_id.insert(std::pair(enb_sri->sinfo_assoc_id, enb_ptr->enb_id)); m_enb_assoc_to_ue_ids.insert(std::pair >(enb_sri->sinfo_assoc_id, ue_set)); } enb_ctx_t* s1ap::find_enb_ctx(uint16_t enb_id) { std::map::iterator it = m_active_enbs.find(enb_id); if (it == m_active_enbs.end()) { return nullptr; } else { return it->second; } } void s1ap::delete_enb_ctx(int32_t assoc_id) { std::map::iterator it_assoc = m_sctp_to_enb_id.find(assoc_id); uint16_t enb_id = it_assoc->second; std::map::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); srslte::out_stream("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_nas_ctx_to_imsi_map(nas* nas_ctx) { std::map::iterator ctx_it = m_imsi_to_nas_ctx.find(nas_ctx->m_emm_ctx.imsi); if (ctx_it != m_imsi_to_nas_ctx.end()) { m_s1ap_log->error("UE Context already exists. IMSI %015" PRIu64 "\n", nas_ctx->m_emm_ctx.imsi); return false; } if (nas_ctx->m_ecm_ctx.mme_ue_s1ap_id != 0) { std::map::iterator ctx_it2 = m_mme_ue_s1ap_id_to_nas_ctx.find(nas_ctx->m_ecm_ctx.mme_ue_s1ap_id); if (ctx_it2 != m_mme_ue_s1ap_id_to_nas_ctx.end() && ctx_it2->second != nas_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_nas_ctx.insert(std::pair(nas_ctx->m_emm_ctx.imsi, nas_ctx)); m_s1ap_log->debug("Saved UE context corresponding to IMSI %015" PRIu64 "\n", nas_ctx->m_emm_ctx.imsi); return true; } bool s1ap::add_nas_ctx_to_mme_ue_s1ap_id_map(nas* nas_ctx) { if (nas_ctx->m_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.\n"); return false; } std::map::iterator ctx_it = m_mme_ue_s1ap_id_to_nas_ctx.find(nas_ctx->m_ecm_ctx.mme_ue_s1ap_id); if (ctx_it != m_mme_ue_s1ap_id_to_nas_ctx.end()) { m_s1ap_log->error("UE Context already exists. MME UE S1AP Id %015" PRIu64 "\n", nas_ctx->m_emm_ctx.imsi); return false; } if (nas_ctx->m_emm_ctx.imsi != 0) { std::map::iterator ctx_it2 = m_mme_ue_s1ap_id_to_nas_ctx.find(nas_ctx->m_ecm_ctx.mme_ue_s1ap_id); if (ctx_it2 != m_mme_ue_s1ap_id_to_nas_ctx.end() && ctx_it2->second != nas_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_nas_ctx.insert(std::pair(nas_ctx->m_ecm_ctx.mme_ue_s1ap_id, nas_ctx)); m_s1ap_log->debug("Saved UE context corresponding to MME UE S1AP Id %d\n", nas_ctx->m_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 >::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\n", enb_assoc); return false; } std::set::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\n", 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; } nas* s1ap::find_nas_ctx_from_mme_ue_s1ap_id(uint32_t mme_ue_s1ap_id) { std::map::iterator it = m_mme_ue_s1ap_id_to_nas_ctx.find(mme_ue_s1ap_id); if (it == m_mme_ue_s1ap_id_to_nas_ctx.end()) { return NULL; } else { return it->second; } } nas* s1ap::find_nas_ctx_from_imsi(uint64_t imsi) { std::map::iterator it = m_imsi_to_nas_ctx.find(imsi); if (it == m_imsi_to_nas_ctx.end()) { return NULL; } else { return it->second; } } void s1ap::release_ues_ecm_ctx_in_enb(int32_t enb_assoc) { srslte::out_stream("Releasing UEs context\n"); std::map >::iterator ues_in_enb = m_enb_assoc_to_ue_ids.find(enb_assoc); std::set::iterator ue_id = ues_in_enb->second.begin(); if (ue_id == ues_in_enb->second.end()) { srslte::out_stream("No UEs to be released\n"); } else { while (ue_id != ues_in_enb->second.end()) { std::map::iterator nas_ctx = m_mme_ue_s1ap_id_to_nas_ctx.find(*ue_id); emm_ctx_t* emm_ctx = &nas_ctx->second->m_emm_ctx; ecm_ctx_t* ecm_ctx = &nas_ctx->second->m_ecm_ctx; m_s1ap_log->info( "Releasing UE context. IMSI: %015" PRIu64 ", 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; } srslte::out_stream("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) { nas* nas_ctx = find_nas_ctx_from_mme_ue_s1ap_id(mme_ue_s1ap_id); if (nas_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; } ecm_ctx_t* ecm_ctx = &nas_ctx->m_ecm_ctx; // Delete UE within eNB UE set std::map::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 >::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_nas_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) { nas* nas_ctx = find_nas_ctx_from_imsi(imsi); if (nas_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 (nas_ctx->m_ecm_ctx.mme_ue_s1ap_id != 0) { release_ue_ecm_ctx(nas_ctx->m_ecm_ctx.mme_ue_s1ap_id); } // Delete UE context m_imsi_to_nas_ctx.erase(imsi); delete nas_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::iterator ue_ctx_it = m_imsi_to_nas_ctx.find(imsi); if (ue_ctx_it == m_imsi_to_nas_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->m_ecm_ctx.mme_ue_s1ap_id; std::map::iterator it = m_mme_ue_s1ap_id_to_nas_ctx.find(mme_ue_s1ap_id); if (it == m_mme_ue_s1ap_id_to_nas_ctx.end()) { m_s1ap_log->error("Could not activate EPS bearer: ECM context seems to be missing\n"); return; } ecm_ctx_t* ecm_ctx = &ue_ctx_it->second->m_ecm_ctx; esm_ctx_t* esm_ctx = &ue_ctx_it->second->m_esm_ctx[ebi]; if (esm_ctx->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, esm_ctx->state); srslte::out_stream( "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, esm_ctx->state); return; } esm_ctx->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(m_tmsi, imsi)); m_s1ap_log->debug("Allocated M-TMSI 0x%x to IMSI %015" PRIu64 ",\n", m_tmsi, imsi); return m_tmsi; } uint64_t s1ap::find_imsi_from_m_tmsi(uint32_t m_tmsi) { std::map::iterator it = m_tmsi_to_imsi.find(m_tmsi); if (it != m_tmsi_to_imsi.end()) { m_s1ap_log->debug("Found IMSI %015" PRIu64 " from M-TMSI 0x%x\n", it->second, m_tmsi); return it->second; } else { m_s1ap_log->debug("Could not find IMSI from M-TMSI 0x%x\n", m_tmsi); return 0; } } 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) { srslte::out_stream("%s - eNB Name: %s, eNB id: 0x%x\n", prefix.c_str(), enb_ctx.enb_name.c_str(), enb_ctx.enb_id); m_s1ap_log->info("%s - eNB Name: %s, eNB id: 0x%x\n", prefix.c_str(), enb_ctx.enb_name.c_str(), enb_ctx.enb_id); } else { srslte::out_stream("%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); srslte::out_stream("%s - MCC:%s, MNC:%s\n", prefix.c_str(), mcc_str.c_str(), mnc_str.c_str()); 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 0x%x\n", prefix.c_str(), enb_ctx.tacs[i], enb_ctx.bplmns[i][j]); srslte::out_stream("%s - TAC %d, B-PLMN 0x%x\n", prefix.c_str(), enb_ctx.tacs[i], enb_ctx.bplmns[i][j]); } } srslte::out_stream("%s - Paging DRX %s\n", prefix.c_str(), enb_ctx.drx.to_string().c_str()); return; } /* * Interfaces */ // GTP-C -> S1AP interface bool s1ap::send_paging(uint64_t imsi, uint16_t erab_to_setup) { m_s1ap_paging->send_paging(imsi, erab_to_setup); return true; } // GTP-C || NAS -> S1AP interface bool s1ap::send_initial_context_setup_request(uint64_t imsi, uint16_t erab_to_setup) { nas* nas_ctx = find_nas_ctx_from_imsi(imsi); if (nas_ctx == NULL) { m_s1ap_log->error("Error finding NAS context when sending initial context Setup Request\n"); return false; } m_s1ap_ctx_mngmt_proc->send_initial_context_setup_request(nas_ctx, erab_to_setup); return true; } // NAS -> S1AP interface bool s1ap::send_ue_context_release_command(uint32_t mme_ue_s1ap_id) { nas* nas_ctx = find_nas_ctx_from_mme_ue_s1ap_id(mme_ue_s1ap_id); if (nas_ctx == NULL) { m_s1ap_log->error("Error finding NAS context when sending UE Context Setup Release\n"); return false; } m_s1ap_ctx_mngmt_proc->send_ue_context_release_command(nas_ctx); return true; } bool s1ap::send_downlink_nas_transport(uint32_t enb_ue_s1ap_id, uint32_t mme_ue_s1ap_id, srslte::byte_buffer_t* nas_msg, struct sctp_sndrcvinfo enb_sri) { return m_s1ap_nas_transport->send_downlink_nas_transport(enb_ue_s1ap_id, mme_ue_s1ap_id, nas_msg, enb_sri); } bool s1ap::expire_nas_timer(enum nas_timer_type type, uint64_t imsi) { nas* nas_ctx = find_nas_ctx_from_imsi(imsi); if (nas_ctx == NULL) { m_s1ap_log->error("Error finding NAS context to handle timer\n"); return false; } bool err = nas_ctx->expire_timer(type); return err; } } // namespace srsepc