mirror of https://github.com/pvnis/srsRAN_4G.git
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
254 lines
7.2 KiB
C++
254 lines
7.2 KiB
C++
/**
|
|
*
|
|
* \section COPYRIGHT
|
|
*
|
|
* Copyright 2013-2020 Software Radio Systems Limited
|
|
*
|
|
* By using this file, you agree to the terms and conditions set
|
|
* forth in the LICENSE file which can be found at the top level of
|
|
* the distribution.
|
|
*
|
|
*/
|
|
|
|
#include "srsue/test/ttcn3/hdr/ttcn3_ue.h"
|
|
#include "lte_ttcn3_phy.h"
|
|
#include "srsue/hdr/stack/ue_stack_lte.h"
|
|
#include <sstream>
|
|
|
|
extern "C" {
|
|
SRSLTE_API char* srslte_get_build_info();
|
|
SRSLTE_API char* srslte_get_build_mode();
|
|
}
|
|
|
|
ttcn3_ue::ttcn3_ue() : tft_matcher(&log) {}
|
|
|
|
int ttcn3_ue::init(all_args_t args, srslte::logger* logger_, syssim_interface_phy* syssim_, const std::string tc_name_)
|
|
{
|
|
logger = logger_;
|
|
|
|
// Init UE log
|
|
log.init("UE ", logger);
|
|
log.set_level(srslte::LOG_LEVEL_INFO);
|
|
log.info("Built in %s mode using %s.\n", srslte_get_build_mode(), srslte_get_build_info());
|
|
|
|
// Patch args
|
|
args.stack.nas.force_imsi_attach = true;
|
|
args.stack.nas.eia = "1,2,3";
|
|
args.stack.nas.eea = "0,1,2,3";
|
|
|
|
// Configure default parameters
|
|
args.stack.usim.algo = "xor";
|
|
args.stack.usim.imei = "356092040793011";
|
|
args.stack.usim.imsi = "001010123456789"; // Anritsu test USIM
|
|
args.stack.usim.k = "000102030405060708090A0B0C0D0E0F"; // fixed as per TS 34.108 Sec. 8.2
|
|
|
|
args.stack.rrc.feature_group = 0xe6041000;
|
|
args.stack.rrc.ue_category_str = SRSLTE_UE_CATEGORY_DEFAULT;
|
|
args.stack.rrc.ue_category = strtol(args.stack.rrc.ue_category_str.c_str(), nullptr, 10);
|
|
args.stack.rrc.nof_supported_bands = 1;
|
|
args.stack.rrc.supported_bands[0] = 7;
|
|
args.stack.rrc.release = 8;
|
|
args.stack.rrc.mbms_service_id = -1;
|
|
|
|
args.phy.dl_earfcn = "3400";
|
|
args.rf.type = "none";
|
|
args.stack.type = "lte";
|
|
args.phy.type = "lte_ttcn3";
|
|
|
|
// Instantiate layers and stack together our UE
|
|
if (args.stack.type == "lte") {
|
|
stack = std::unique_ptr<ue_stack_lte>(new ue_stack_lte());
|
|
if (!stack) {
|
|
srslte::console("Error creating LTE stack instance.\n");
|
|
return SRSLTE_ERROR;
|
|
}
|
|
|
|
phy = std::unique_ptr<srsue::lte_ttcn3_phy>(new srsue::lte_ttcn3_phy(logger));
|
|
if (!phy) {
|
|
srslte::console("Error creating LTE PHY instance.\n");
|
|
return SRSLTE_ERROR;
|
|
}
|
|
} else {
|
|
srslte::console("Invalid stack type %s. Supported values are [lte].\n", args.stack.type.c_str());
|
|
return SRSLTE_ERROR;
|
|
}
|
|
|
|
// init layers
|
|
if (phy->init(args.phy, stack.get(), syssim_)) {
|
|
srslte::console("Error initializing PHY.\n");
|
|
return SRSLTE_ERROR;
|
|
}
|
|
|
|
if (stack->init(args.stack, logger, phy.get(), this)) {
|
|
srslte::console("Error initializing stack.\n");
|
|
return SRSLTE_ERROR;
|
|
}
|
|
|
|
return SRSLTE_SUCCESS;
|
|
}
|
|
|
|
ttcn3_ue::~ttcn3_ue()
|
|
{
|
|
stop();
|
|
}
|
|
|
|
void ttcn3_ue::stop()
|
|
{
|
|
if (stack) {
|
|
stack->stop();
|
|
}
|
|
|
|
if (phy) {
|
|
phy->stop();
|
|
}
|
|
}
|
|
|
|
bool ttcn3_ue::switch_on()
|
|
{
|
|
return stack->switch_on();
|
|
}
|
|
|
|
bool ttcn3_ue::switch_off()
|
|
{
|
|
return stack->switch_off();
|
|
}
|
|
|
|
bool ttcn3_ue::enable_data()
|
|
{
|
|
return stack->enable_data();
|
|
}
|
|
|
|
bool ttcn3_ue::disable_data()
|
|
{
|
|
return stack->disable_data();
|
|
}
|
|
|
|
// The interface for SYSSIM
|
|
void ttcn3_ue::set_cell_map(lte_ttcn3_phy::cell_list_t phy_cell_map)
|
|
{
|
|
phy->set_cell_map(phy_cell_map);
|
|
}
|
|
|
|
void ttcn3_ue::new_grant_ul(const srsue::mac_interface_phy_lte::mac_grant_ul_t grant)
|
|
{
|
|
phy->new_grant_ul(grant);
|
|
}
|
|
|
|
void ttcn3_ue::new_tb(const srsue::mac_interface_phy_lte::mac_grant_dl_t mac_grant, const uint8_t* data)
|
|
{
|
|
phy->new_tb(mac_grant, data);
|
|
}
|
|
|
|
void ttcn3_ue::set_current_tti(uint32_t tti)
|
|
{
|
|
phy->set_current_tti(tti);
|
|
stack->run_tti(tti, 1);
|
|
}
|
|
|
|
uint16_t ttcn3_ue::get_dl_sched_rnti(uint32_t tti)
|
|
{
|
|
return stack->get_dl_sched_rnti(tti);
|
|
}
|
|
|
|
// GW interface
|
|
void ttcn3_ue::add_mch_port(uint32_t lcid, uint32_t port) {}
|
|
void ttcn3_ue::write_pdu(uint32_t lcid, srslte::unique_byte_buffer_t pdu)
|
|
{
|
|
log.debug_hex(pdu->msg, pdu->N_bytes, "Rx PDU (%d B) on lcid=%d\n", pdu->N_bytes, lcid);
|
|
switch (test_loop_mode) {
|
|
case TEST_LOOP_INACTIVE:
|
|
log.warning("Test loop inactive. Dropping PDU.\n");
|
|
break;
|
|
case TEST_LOOP_MODE_A_ACTIVE:
|
|
log.error("Test loop mode A not implemented. Dropping PDU.\n");
|
|
break;
|
|
case TEST_LOOP_MODE_B_ACTIVE:
|
|
// Section 5.4.4 in TS 36.509
|
|
if (pdu_delay_timer.is_running()) {
|
|
pdu_queue[lcid].push(std::move(pdu));
|
|
} else {
|
|
if (pdu_delay_timer.is_set()) {
|
|
pdu_queue[lcid].push(std::move(pdu));
|
|
pdu_delay_timer.run(); // timer is already set
|
|
} else {
|
|
loop_back_pdu_with_tft(lcid, std::move(pdu));
|
|
}
|
|
}
|
|
break;
|
|
case TEST_LOOP_MODE_C_ACTIVE:
|
|
log.error("Test loop mode C not implemented. Dropping PDU.\n");
|
|
break;
|
|
}
|
|
}
|
|
void ttcn3_ue::write_pdu_mch(uint32_t lcid, srslte::unique_byte_buffer_t pdu) {}
|
|
int ttcn3_ue::setup_if_addr(uint32_t lcid, uint8_t pdn_type, uint32_t ip_addr, uint8_t* ipv6_if_id, char* err_str)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int ttcn3_ue::apply_traffic_flow_template(const uint8_t& eps_bearer_id,
|
|
const uint8_t& lcid,
|
|
const LIBLTE_MME_TRAFFIC_FLOW_TEMPLATE_STRUCT* tft)
|
|
{
|
|
return tft_matcher.apply_traffic_flow_template(eps_bearer_id, lcid, tft);
|
|
}
|
|
|
|
void ttcn3_ue::set_test_loop_mode(const test_loop_mode_state_t mode, const uint32_t ip_pdu_delay_ms_)
|
|
{
|
|
test_loop_mode = mode;
|
|
switch (test_loop_mode) {
|
|
case TEST_LOOP_INACTIVE:
|
|
// deactivate timer
|
|
log.info("Deactivating Test Loop Mode\n");
|
|
pdu_delay_timer.release();
|
|
break;
|
|
case TEST_LOOP_MODE_A_ACTIVE:
|
|
log.error("Test loop mode A not implemented\n");
|
|
break;
|
|
case TEST_LOOP_MODE_B_ACTIVE:
|
|
log.info("Activating Test loop mode B with %d ms PDU delay\n", ip_pdu_delay_ms_);
|
|
pdu_delay_timer = stack->get_task_sched().get_unique_timer();
|
|
if (ip_pdu_delay_ms_ > 0) {
|
|
pdu_delay_timer.set(ip_pdu_delay_ms_, [this](uint32_t tid) { timer_expired(tid); });
|
|
}
|
|
break;
|
|
case TEST_LOOP_MODE_C_ACTIVE:
|
|
log.error("Test loop mode A not implemented\n");
|
|
break;
|
|
}
|
|
}
|
|
|
|
void ttcn3_ue::timer_expired(uint32_t timeout_id)
|
|
{
|
|
if (timeout_id == pdu_delay_timer.id()) {
|
|
log.info("Testmode B PDU delay timer expired\n");
|
|
send_queued_data();
|
|
}
|
|
}
|
|
void ttcn3_ue::send_queued_data()
|
|
{
|
|
if (!stack->is_rrc_connected()) {
|
|
log.info("RRC not connected, requesting NAS attach\n");
|
|
if (not stack->start_service_request()) {
|
|
log.warning("Could not reestablish the connection\n");
|
|
}
|
|
stack->get_task_sched().defer_callback(500, [&]() { send_queued_data(); });
|
|
return;
|
|
}
|
|
|
|
for (auto& bearer_pdu_queue : pdu_queue) {
|
|
log.info("Delivering %zd buffered PDUs for LCID=%d\n", bearer_pdu_queue.second.size(), bearer_pdu_queue.first);
|
|
while (not bearer_pdu_queue.second.empty()) {
|
|
srslte::unique_byte_buffer_t pdu;
|
|
bearer_pdu_queue.second.try_pop(&pdu);
|
|
loop_back_pdu_with_tft(bearer_pdu_queue.first, std::move(pdu));
|
|
}
|
|
}
|
|
}
|
|
|
|
void ttcn3_ue::loop_back_pdu_with_tft(uint32_t input_lcid, srslte::unique_byte_buffer_t pdu)
|
|
{
|
|
log.info_hex(pdu->msg, pdu->N_bytes, "Rx PDU (%d B) on lcid=%d, looping back\n", pdu->N_bytes, input_lcid);
|
|
stack->write_sdu(input_lcid, std::move(pdu));
|
|
}
|