/** * Copyright 2013-2021 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 SRSENB_TEST_HELPERS_H #define SRSENB_TEST_HELPERS_H #include "srsenb/src/enb_cfg_parser.h" #include "srsenb/test/common/dummy_classes.h" #include "srsran/adt/span.h" using namespace srsenb; using namespace asn1::rrc; namespace argparse { extern std::string repository_dir; extern srslog::basic_levels log_level; inline void usage(char* prog) { printf("Usage: %s [v] -i repository_dir\n", prog); printf("\t-v [set srsran_verbose to debug, default none]\n"); } inline void parse_args(int argc, char** argv) { int opt; while ((opt = getopt(argc, argv, "i")) != -1) { switch (opt) { case 'i': repository_dir = argv[optind]; break; case 'v': log_level = srslog::basic_levels::debug; break; default: usage(argv[0]); exit(-1); } } if (repository_dir.empty()) { usage(argv[0]); exit(-1); } } } // namespace argparse namespace test_dummies { class s1ap_mobility_dummy : public s1ap_dummy { public: struct ho_req_data { uint16_t rnti; uint32_t target_eci; srsran::plmn_id_t target_plmn; srsran::unique_byte_buffer_t rrc_container; } last_ho_required = {}; struct enb_status_transfer_info { bool status_present; uint16_t rnti; std::vector bearer_list; } last_enb_status = {}; struct ho_req_ack { uint16_t rnti; srsran::unique_byte_buffer_t ho_cmd_pdu; std::vector admitted_bearers; std::vector not_admitted_bearers; } last_ho_req_ack; bool send_ho_required(uint16_t rnti, uint32_t target_eci, srsran::plmn_id_t target_plmn, srsran::span fwd_erabs, srsran::unique_byte_buffer_t rrc_container, bool has_direct_fwd_path) final { last_ho_required = ho_req_data{rnti, target_eci, target_plmn, std::move(rrc_container)}; return true; } bool send_enb_status_transfer_proc(uint16_t rnti, std::vector& bearer_status_list) override { last_enb_status = {true, rnti, bearer_status_list}; return true; } bool send_ho_req_ack(const asn1::s1ap::ho_request_s& msg, uint16_t rnti, uint32_t enb_cc_idx, srsran::unique_byte_buffer_t ho_cmd, srsran::span admitted_bearers, srsran::const_span not_admitted_bearers) override { last_ho_req_ack.rnti = rnti; last_ho_req_ack.ho_cmd_pdu = std::move(ho_cmd); last_ho_req_ack.admitted_bearers.assign(admitted_bearers.begin(), admitted_bearers.end()); last_ho_req_ack.not_admitted_bearers.assign(not_admitted_bearers.begin(), not_admitted_bearers.end()); return true; } void user_mod(uint16_t old_rnti, uint16_t new_rnti) override {} }; class pdcp_mobility_dummy : public pdcp_dummy { public: struct last_sdu_t { uint16_t rnti; uint32_t lcid; srsran::unique_byte_buffer_t sdu; } last_sdu; struct lcid_cfg_t { bool enable_integrity = false; bool enable_encryption = false; srsran::pdcp_lte_state_t state{}; srsran::as_security_config_t sec_cfg{}; }; std::map > bearers; void write_sdu(uint16_t rnti, uint32_t lcid, srsran::unique_byte_buffer_t sdu, int pdcp_sn) override { last_sdu.rnti = rnti; last_sdu.lcid = lcid; last_sdu.sdu = std::move(sdu); } bool set_bearer_state(uint16_t rnti, uint32_t lcid, const srsran::pdcp_lte_state_t& state) override { bearers[rnti][lcid].state = state; return true; } void enable_integrity(uint16_t rnti, uint32_t lcid) override { bearers[rnti][lcid].enable_integrity = true; } void enable_encryption(uint16_t rnti, uint32_t lcid) override { bearers[rnti][lcid].enable_encryption = true; } void config_security(uint16_t rnti, uint32_t lcid, srsran::as_security_config_t sec_cfg_) override { bearers[rnti][lcid].sec_cfg = sec_cfg_; } }; class rlc_mobility_dummy : public rlc_dummy { public: struct ue_ctxt { int nof_pdcp_sdus = 0, reest_sdu_counter = 0; uint32_t last_lcid = 0; srsran::unique_byte_buffer_t last_sdu; }; std::map ue_db; void test_reset_all() { for (auto& u : ue_db) { u.second = {}; } } void write_sdu(uint16_t rnti, uint32_t lcid, srsran::unique_byte_buffer_t sdu) override { ue_db[rnti].nof_pdcp_sdus++; ue_db[rnti].reest_sdu_counter++; ue_db[rnti].last_lcid = lcid; ue_db[rnti].last_sdu = std::move(sdu); } void reestablish(uint16_t rnti) final { ue_db[rnti].reest_sdu_counter = 0; } }; class mac_mobility_dummy : public mac_dummy { public: int ue_cfg(uint16_t rnti, const sched_interface::ue_cfg_t* cfg) override { ue_db[rnti] = *cfg; return 0; } int ue_set_crnti(uint16_t temp_crnti, uint16_t crnti, const sched_interface::ue_cfg_t& cfg) override { ue_db[crnti] = cfg; return 0; } std::map ue_db; }; class phy_mobility_dummy : public phy_dummy { public: void set_config(uint16_t rnti, const phy_rrc_cfg_list_t& dedicated_list) override { phy_cfg_set = true; last_cfg = dedicated_list; } bool phy_cfg_set = false; phy_rrc_cfg_list_t last_cfg; }; } // namespace test_dummies namespace test_helpers { int parse_default_cfg(rrc_cfg_t* rrc_cfg, srsenb::all_args_t& args); int parse_default_cfg_phy(rrc_cfg_t* rrc_cfg, phy_cfg_t* phy_cfg, srsenb::all_args_t& args); template bool unpack_asn1(ASN1Type& asn1obj, srsran::const_byte_span pdu) { asn1::cbit_ref bref{pdu.data(), (uint32_t)pdu.size()}; if (asn1obj.unpack(bref) != asn1::SRSASN_SUCCESS) { srslog::fetch_basic_logger("TEST").error("Failed to unpack ASN1 type"); return false; } return true; } int bring_rrc_to_reconf_state(srsenb::rrc& rrc, srsran::timer_handler& timers, uint16_t rnti); } // namespace test_helpers namespace srsenb { meas_cell_cfg_t generate_cell1(); report_cfg_eutra_s generate_rep1(); bool is_cell_cfg_equal(const meas_cell_cfg_t& cfg, const cells_to_add_mod_s& cell); } // namespace srsenb #endif // SRSENB_TEST_HELPERS_H