/** * Copyright 2013-2022 Software Radio Systems Limited * * This file is part of srsRAN. * * srsRAN 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. * * srsRAN 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/. * */ #ifndef SRSEPC_NAS_H #define SRSEPC_NAS_H #include "srsran/asn1/gtpc_ies.h" #include "srsran/asn1/liblte_mme.h" #include "srsran/common/buffer_pool.h" #include "srsran/common/security.h" #include "srsran/interfaces/epc_interfaces.h" #include "srsran/srslog/srslog.h" #include namespace srsepc { static const uint8_t MAX_ERABS_PER_UE = 16; // MME EMM states (3GPP 24.301 v10.0.0, section 5.1.3.4) typedef enum { EMM_STATE_DEREGISTERED = 0, EMM_STATE_COMMON_PROCEDURE_INITIATED, EMM_STATE_REGISTERED, EMM_STATE_DEREGISTERED_INITIATED, EMM_STATE_N_ITEMS, } emm_state_t; static const char emm_state_text[EMM_STATE_N_ITEMS][100] = {"DEREGISTERED", "COMMON PROCEDURE INITIATED", "REGISTERED", "DEREGISTERED INITIATED"}; // MME ECM states (3GPP 23.401 v10.0.0, section 4.6.3) typedef enum { ECM_STATE_IDLE = 0, ECM_STATE_CONNECTED, ECM_STATE_N_ITEMS, } ecm_state_t; static const char ecm_state_text[ECM_STATE_N_ITEMS][100] = {"IDLE", "CONNECTED"}; /* // MME ESM states (3GPP 23.401 v10.0.0, section 4.6.3) typedef enum { ESM_BEARER_CONTEXT_INACTIVE = 0, ESM_BEARER_CONTEXT_ACTIVE_PENDING, ESM_BEARER_CONTEXT_ACTIVE, ESM_BEARER_CONTEXT_INACTIVE_PENDING, ESM_BEARER_CONTEXT_MODIFY_PENDING, ESM_BEARER_PROCEDURE_TRANSACTION_INACTIVE, ESM_BEARER_PROCEDURE_TRANSACTION_PENDING, ESM_STATE_N_ITEMS, } esm_state_t; static const char esm_state_text[ESM_STATE_N_ITEMS][100] = {"CONTEXT INACTIVE", "CONTEXT ACTIVE PENDING", "CONTEXT ACTIVE", "CONTEXT_INACTIVE_PENDING", "CONTEXT_MODIFY_PENDING", "PROCEDURE_TRANSACTION_INACTIVE" "PROCEDURE_TRANSACTION_PENDING"}; */ typedef enum { ERAB_DEACTIVATED, ERAB_CTX_REQUESTED, ERAB_CTX_SETUP, ERAB_ACTIVE } esm_state_t; /* * EMM, ECM, ESM and EPS Security context definitions */ typedef struct { uint64_t imsi; emm_state_t state; uint8_t procedure_transaction_id; uint8_t attach_type; struct in_addr ue_ip; srsran::gtpc_f_teid_ie sgw_ctrl_fteid; } emm_ctx_t; typedef struct { ecm_state_t state; uint32_t enb_ue_s1ap_id; uint32_t mme_ue_s1ap_id; struct sctp_sndrcvinfo enb_sri; bool eit; } ecm_ctx_t; typedef struct { uint8_t erab_id; esm_state_t state; uint8_t qci; srsran::gtpc_f_teid_ie enb_fteid; srsran::gtpc_f_teid_ie sgw_s1u_fteid; srsran::gtpc_pdn_address_allocation_ie pdn_addr_alloc; } esm_ctx_t; typedef struct { uint8_t eksi; uint8_t k_asme[32]; uint8_t autn[16]; uint8_t rand[16]; uint8_t xres[16]; // minimum 6, maximum 16 uint32_t dl_nas_count; uint32_t ul_nas_count; srsran::CIPHERING_ALGORITHM_ID_ENUM cipher_algo; srsran::INTEGRITY_ALGORITHM_ID_ENUM integ_algo; uint8_t k_nas_enc[32]; uint8_t k_nas_int[32]; uint8_t k_enb[32]; LIBLTE_MME_UE_NETWORK_CAPABILITY_STRUCT ue_network_cap; bool ms_network_cap_present; LIBLTE_MME_MS_NETWORK_CAPABILITY_STRUCT ms_network_cap; LIBLTE_MME_EPS_MOBILE_ID_GUTI_STRUCT guti; } sec_ctx_t; /* * NAS Initialization Arguments */ typedef struct { uint16_t mcc; uint16_t mnc; uint8_t mme_code; uint16_t mme_group; uint16_t tac; uint16_t paging_timer; std::string apn; std::string dns; std::string full_net_name; std::string short_net_name; srsran::CIPHERING_ALGORITHM_ID_ENUM cipher_algo; srsran::INTEGRITY_ALGORITHM_ID_ENUM integ_algo; bool request_imeisv; } nas_init_t; typedef struct { s1ap_interface_nas* s1ap; gtpc_interface_nas* gtpc; hss_interface_nas* hss; mme_interface_nas* mme; } nas_if_t; class nas { public: nas(const nas_init_t& args, const nas_if_t& itf); void reset(); /*********************** * Initial UE messages * ***********************/ // Attach request messages static bool handle_attach_request(uint32_t enb_ue_s1ap_id, struct sctp_sndrcvinfo* enb_sri, srsran::byte_buffer_t* nas_rx, const nas_init_t& args, const nas_if_t& itf); static bool handle_imsi_attach_request_unknown_ue(uint32_t enb_ue_s1ap_id, struct sctp_sndrcvinfo* enb_sri, const LIBLTE_MME_ATTACH_REQUEST_MSG_STRUCT& attach_req, const LIBLTE_MME_PDN_CONNECTIVITY_REQUEST_MSG_STRUCT& pdn_con_req, const nas_init_t& args, const nas_if_t& itf); static bool handle_imsi_attach_request_known_ue(nas* nas_ctx, uint32_t enb_ue_s1ap_id, struct sctp_sndrcvinfo* enb_sri, const LIBLTE_MME_ATTACH_REQUEST_MSG_STRUCT& attach_req, const LIBLTE_MME_PDN_CONNECTIVITY_REQUEST_MSG_STRUCT& pdn_con_req, srsran::byte_buffer_t* nas_rx, const nas_init_t& args, const nas_if_t& itf); static bool handle_guti_attach_request_unknown_ue(uint32_t enb_ue_s1ap_id, struct sctp_sndrcvinfo* enb_sri, const LIBLTE_MME_ATTACH_REQUEST_MSG_STRUCT& attach_req, const LIBLTE_MME_PDN_CONNECTIVITY_REQUEST_MSG_STRUCT& pdn_con_req, const nas_init_t& args, const nas_if_t& itf); static bool handle_guti_attach_request_known_ue(nas* nas_ctx, uint32_t enb_ue_s1ap_id, struct sctp_sndrcvinfo* enb_sri, const LIBLTE_MME_ATTACH_REQUEST_MSG_STRUCT& attach_req, const LIBLTE_MME_PDN_CONNECTIVITY_REQUEST_MSG_STRUCT& pdn_con_req, srsran::byte_buffer_t* nas_rx, const nas_init_t& args, const nas_if_t& itf); // Service request messages static bool handle_service_request(uint32_t m_tmsi, uint32_t enb_ue_s1ap_id, struct sctp_sndrcvinfo* enb_sri, srsran::byte_buffer_t* nas_rx, const nas_init_t& args, const nas_if_t& itf); // Dettach request messages static bool handle_detach_request(uint32_t m_tmsi, uint32_t enb_ue_s1ap_id, struct sctp_sndrcvinfo* enb_sri, srsran::byte_buffer_t* nas_rx, const nas_init_t& args, const nas_if_t& itf); // Tracking area update request messages static bool handle_tracking_area_update_request(uint32_t m_tmsi, uint32_t enb_ue_s1ap_id, struct sctp_sndrcvinfo* enb_sri, srsran::byte_buffer_t* nas_rx, const nas_init_t& args, const nas_if_t& itf); /* Uplink NAS messages handling */ bool handle_attach_request(srsran::byte_buffer_t* nas_rx); bool handle_pdn_connectivity_request(srsran::byte_buffer_t* nas_rx); bool handle_authentication_response(srsran::byte_buffer_t* nas_rx); bool handle_security_mode_complete(srsran::byte_buffer_t* nas_rx); bool handle_attach_complete(srsran::byte_buffer_t* nas_rx); bool handle_esm_information_response(srsran::byte_buffer_t* nas_rx); bool handle_identity_response(srsran::byte_buffer_t* nas_rx); bool handle_tracking_area_update_request(srsran::byte_buffer_t* nas_rx); bool handle_authentication_failure(srsran::byte_buffer_t* nas_rx); bool handle_detach_request(srsran::byte_buffer_t* nas_rx); /* Downlink NAS messages packing */ bool pack_authentication_request(srsran::byte_buffer_t* nas_buffer); bool pack_authentication_reject(srsran::byte_buffer_t* nas_buffer); bool pack_security_mode_command(srsran::byte_buffer_t* nas_buffer); bool pack_esm_information_request(srsran::byte_buffer_t* nas_buffer); bool pack_identity_request(srsran::byte_buffer_t* nas_buffer); bool pack_emm_information(srsran::byte_buffer_t* nas_buffer); bool pack_service_reject(srsran::byte_buffer_t* nas_buffer, uint8_t emm_cause); bool pack_tracking_area_update_reject(srsran::byte_buffer_t* nas_buffer, uint8_t emm_cause); bool pack_attach_accept(srsran::byte_buffer_t* nas_buffer); /* Security functions */ bool integrity_check(srsran::byte_buffer_t* pdu, bool warn_failure = true); bool short_integrity_check(srsran::byte_buffer_t* pdu); void integrity_generate(srsran::byte_buffer_t* pdu, uint8_t* mac); void cipher_decrypt(srsran::byte_buffer_t* pdu); void cipher_encrypt(srsran::byte_buffer_t* pdu); /*Timer functions*/ bool start_timer(enum nas_timer_type type); bool expire_timer(enum nas_timer_type type); /* UE Context */ emm_ctx_t m_emm_ctx = {}; ecm_ctx_t m_ecm_ctx = {}; esm_ctx_t m_esm_ctx[MAX_ERABS_PER_UE] = {}; sec_ctx_t m_sec_ctx = {}; private: srslog::basic_logger& m_logger = srslog::fetch_basic_logger("NAS"); gtpc_interface_nas* m_gtpc = nullptr; s1ap_interface_nas* m_s1ap = nullptr; hss_interface_nas* m_hss = nullptr; mme_interface_nas* m_mme = nullptr; uint16_t m_mcc = 0; uint16_t m_mnc = 0; uint16_t m_mme_group = 0; uint16_t m_mme_code = 0; uint16_t m_tac = 0; std::string m_apn; std::string m_dns; std::string m_full_net_name; std::string m_short_net_name; bool m_request_imeisv = false; // Timers timeout values uint16_t m_t3413 = 0; // Timer functions bool start_t3413(); bool expire_t3413(); }; } // namespace srsepc #endif // SRSEPC_NAS_H