|
|
|
/**
|
|
|
|
*
|
|
|
|
* \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 "srslte/common/test_common.h"
|
|
|
|
#include "srslte/test/ue_test_interfaces.h"
|
|
|
|
#include "srsue/hdr/stack/rrc/phy_controller.h"
|
|
|
|
|
|
|
|
namespace srsue {
|
|
|
|
|
|
|
|
struct cell_search_result_test {
|
|
|
|
cell_search_result_test(phy_controller* phy_ctrl_) : phy_ctrl(phy_ctrl_) {}
|
|
|
|
|
|
|
|
void trigger(const phy_controller::cell_srch_res& result_)
|
|
|
|
{
|
|
|
|
trigger_called = true;
|
|
|
|
result = result_;
|
|
|
|
|
|
|
|
if (phy_ctrl->current_state_name() == "searching_cell" or phy_ctrl->is_trigger_locked()) {
|
|
|
|
phy_ctrl->get_logger().error(
|
|
|
|
"When caller is signalled with cell search result, cell search state cannot be active");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool trigger_called = false;
|
|
|
|
phy_controller::cell_srch_res result = {};
|
|
|
|
phy_controller* phy_ctrl = nullptr;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct cell_select_result_test {
|
|
|
|
cell_select_result_test(phy_controller* phy_ctrl_) : phy_ctrl(phy_ctrl_) {}
|
|
|
|
|
|
|
|
void trigger(const bool& result_)
|
|
|
|
{
|
|
|
|
result = result_ ? 1 : 0;
|
|
|
|
if (phy_ctrl->current_state_name() == "selecting_cell" or phy_ctrl->is_trigger_locked()) {
|
|
|
|
phy_ctrl->get_logger().error(
|
|
|
|
"When caller is signalled with cell select result, cell select state cannot be active");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// start a new cell selection right away
|
|
|
|
if (counter++ < 1) {
|
|
|
|
phy_cell_t new_cell = {};
|
|
|
|
new_cell.pci = 3;
|
|
|
|
new_cell.earfcn = 3400;
|
|
|
|
phy_ctrl->start_cell_select(new_cell, *this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int result = -1, counter = 0;
|
|
|
|
phy_controller* phy_ctrl = nullptr;
|
|
|
|
};
|
|
|
|
|
|
|
|
int test_phy_ctrl_fsm()
|
|
|
|
{
|
|
|
|
srslte::task_scheduler task_sched;
|
|
|
|
phy_dummy_interface phy;
|
|
|
|
phy_controller phy_ctrl{&phy, &task_sched};
|
|
|
|
cell_search_result_test csearch_tester{&phy_ctrl};
|
|
|
|
cell_select_result_test csel_tester{&phy_ctrl};
|
|
|
|
|
|
|
|
TESTASSERT(not phy_ctrl.is_in_sync());
|
|
|
|
TESTASSERT(not phy_ctrl.cell_is_camping());
|
|
|
|
|
|
|
|
// TEST: Sync event changes phy controller state
|
|
|
|
phy_ctrl.in_sync();
|
|
|
|
TESTASSERT(phy_ctrl.is_in_sync());
|
|
|
|
phy_ctrl.out_sync();
|
|
|
|
TESTASSERT(not phy_ctrl.is_in_sync());
|
|
|
|
phy_ctrl.in_sync();
|
|
|
|
TESTASSERT(phy_ctrl.is_in_sync());
|
|
|
|
|
|
|
|
// TEST: Correct initiation of Cell Search state
|
|
|
|
TESTASSERT(phy_ctrl.start_cell_search(csearch_tester));
|
|
|
|
TESTASSERT(not phy_ctrl.is_in_sync());
|
|
|
|
|
|
|
|
// TEST: Cell Search only listens to a cell search result event
|
|
|
|
phy_ctrl.in_sync();
|
|
|
|
TESTASSERT(not phy_ctrl.is_in_sync());
|
|
|
|
phy_ctrl.out_sync();
|
|
|
|
TESTASSERT(not phy_ctrl.is_in_sync());
|
|
|
|
TESTASSERT(phy_ctrl.current_state_name() == "searching_cell");
|
|
|
|
rrc_interface_phy_lte::cell_search_ret_t cs_ret;
|
|
|
|
cs_ret.found = rrc_interface_phy_lte::cell_search_ret_t::CELL_FOUND;
|
|
|
|
phy_cell_t found_cell;
|
|
|
|
found_cell.pci = 1;
|
|
|
|
found_cell.earfcn = 2;
|
|
|
|
phy_ctrl.cell_search_completed(cs_ret, found_cell);
|
|
|
|
TESTASSERT(phy_ctrl.current_state_name() != "searching_cell");
|
|
|
|
|
|
|
|
// TEST: Check propagation of cell search result to caller
|
|
|
|
task_sched.run_pending_tasks();
|
|
|
|
TESTASSERT(csearch_tester.trigger_called);
|
|
|
|
TESTASSERT(csearch_tester.result.cs_ret.found == cs_ret.found);
|
|
|
|
TESTASSERT(csearch_tester.result.found_cell.pci == found_cell.pci);
|
|
|
|
TESTASSERT(csearch_tester.result.found_cell.earfcn == found_cell.earfcn);
|
|
|
|
phy_ctrl.in_sync();
|
|
|
|
TESTASSERT(phy_ctrl.is_in_sync());
|
|
|
|
phy_ctrl.out_sync();
|
|
|
|
|
|
|
|
// TEST: Correct initiation of Cell Select state
|
|
|
|
phy_ctrl.start_cell_select(found_cell, csel_tester);
|
|
|
|
TESTASSERT(not phy_ctrl.is_in_sync());
|
|
|
|
TESTASSERT(phy_ctrl.current_state_name() == "selecting_cell");
|
|
|
|
|
|
|
|
// TEST: Cell Selection state ignores events other than the cell selection result
|
|
|
|
// Cell selection observer is called
|
|
|
|
// New cell selection is started right away without affecting normal operation (e.g. observer should be called)
|
|
|
|
phy_ctrl.in_sync();
|
|
|
|
TESTASSERT(not phy_ctrl.is_in_sync());
|
|
|
|
phy_ctrl.cell_selection_completed(true);
|
|
|
|
// Note: Still in cell selection, but now waiting for the first in_sync
|
|
|
|
TESTASSERT(phy_ctrl.current_state_name() == "selecting_cell");
|
|
|
|
TESTASSERT(not phy_ctrl.is_in_sync());
|
|
|
|
TESTASSERT(csel_tester.result < 0);
|
|
|
|
phy_ctrl.in_sync();
|
|
|
|
task_sched.run_pending_tasks();
|
|
|
|
// observer is called that starts new cell selection
|
|
|
|
TESTASSERT(not phy_ctrl.is_in_sync());
|
|
|
|
TESTASSERT(phy_ctrl.current_state_name() == "selecting_cell");
|
|
|
|
TESTASSERT(csel_tester.result == 1);
|
|
|
|
csel_tester.result = 0;
|
|
|
|
phy_ctrl.cell_selection_completed(true);
|
|
|
|
TESTASSERT(phy_ctrl.current_state_name() == "selecting_cell");
|
|
|
|
phy_ctrl.in_sync();
|
|
|
|
task_sched.run_pending_tasks();
|
|
|
|
TESTASSERT(phy_ctrl.is_in_sync());
|
|
|
|
TESTASSERT(phy_ctrl.current_state_name() != "selecting_cell");
|
|
|
|
TESTASSERT(csel_tester.result == 1);
|
|
|
|
|
|
|
|
// TEST: Cell Selection with timeout being reached
|
|
|
|
csel_tester.result = -1;
|
|
|
|
TESTASSERT(phy_ctrl.start_cell_select(found_cell, csel_tester));
|
|
|
|
TESTASSERT(not phy_ctrl.is_in_sync());
|
|
|
|
phy_ctrl.cell_selection_completed(true);
|
|
|
|
TESTASSERT(phy_ctrl.current_state_name() == "selecting_cell");
|
|
|
|
TESTASSERT(csel_tester.result < 0);
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < phy_controller::wait_sync_timeout_ms; ++i) {
|
|
|
|
TESTASSERT(phy_ctrl.current_state_name() == "selecting_cell");
|
|
|
|
task_sched.tic();
|
|
|
|
task_sched.run_pending_tasks();
|
|
|
|
}
|
|
|
|
TESTASSERT(phy_ctrl.current_state_name() != "selecting_cell");
|
|
|
|
|
|
|
|
// TEST: Propagation of cell selection result to caller
|
|
|
|
task_sched.run_pending_tasks();
|
|
|
|
TESTASSERT(csel_tester.result == 0);
|
|
|
|
|
|
|
|
phy_ctrl.start_cell_select(found_cell, csel_tester);
|
|
|
|
|
|
|
|
return SRSLTE_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
class phy_test_dummy : public phy_dummy_interface
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
bool success_on_cell_select_init = true;
|
|
|
|
|
|
|
|
bool cell_select(phy_cell_t cell) override { return success_on_cell_select_init; }
|
|
|
|
};
|
|
|
|
|
|
|
|
/// TEST: Check if controller handles the case when PHY fails to init cell selection
|
|
|
|
int test_phy_cell_select_init_error_handling()
|
|
|
|
{
|
|
|
|
srslte::task_scheduler task_sched;
|
|
|
|
phy_test_dummy phy;
|
|
|
|
phy_controller phy_ctrl{&phy, &task_sched};
|
|
|
|
phy_cell_t found_cell{};
|
|
|
|
found_cell.pci = 1;
|
|
|
|
found_cell.earfcn = 2;
|
|
|
|
int test_result = -1;
|
|
|
|
phy.success_on_cell_select_init = false;
|
|
|
|
|
|
|
|
phy_ctrl.start_cell_select(found_cell, [&test_result](bool csel_result) { test_result = csel_result ? 1 : 0; });
|
|
|
|
|
|
|
|
return test_result == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace srsue
|
|
|
|
|
|
|
|
int main()
|
|
|
|
{
|
|
|
|
auto& test_logger = srslog::fetch_basic_logger("TEST", false);
|
|
|
|
test_logger.set_level(srslog::basic_levels::info);
|
|
|
|
test_logger.set_hex_dump_max_size(-1);
|
|
|
|
auto& RRC_logger = srslog::fetch_basic_logger("RRC", false);
|
|
|
|
RRC_logger.set_level(srslog::basic_levels::info);
|
|
|
|
RRC_logger.set_hex_dump_max_size(-1);
|
|
|
|
srslog::init();
|
|
|
|
|
|
|
|
TESTASSERT(srsue::test_phy_ctrl_fsm() == SRSLTE_SUCCESS);
|
|
|
|
TESTASSERT(srsue::test_phy_cell_select_init_error_handling() == SRSLTE_SUCCESS);
|
|
|
|
test_logger.info("Finished RRC PHY controller test successfully");
|
|
|
|
}
|