|
|
@ -24,102 +24,65 @@ using dl_sched_t = sched_nr_interface::dl_sched_t;
|
|
|
|
static const srsran::phy_cfg_nr_t default_phy_cfg =
|
|
|
|
static const srsran::phy_cfg_nr_t default_phy_cfg =
|
|
|
|
srsran::phy_cfg_nr_default_t{srsran::phy_cfg_nr_default_t::reference_cfg_t{}};
|
|
|
|
srsran::phy_cfg_nr_default_t{srsran::phy_cfg_nr_default_t::reference_cfg_t{}};
|
|
|
|
|
|
|
|
|
|
|
|
struct task_job_manager {
|
|
|
|
class sched_nr_tester : public sched_nr_sim_parallel
|
|
|
|
std::mutex mutex;
|
|
|
|
|
|
|
|
int res_count = 0;
|
|
|
|
|
|
|
|
int pdsch_count = 0;
|
|
|
|
|
|
|
|
srslog::basic_logger& test_logger = srslog::fetch_basic_logger("TEST");
|
|
|
|
|
|
|
|
struct slot_guard {
|
|
|
|
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
std::condition_variable cvar;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
srsran::bounded_vector<slot_guard, 10> slot_counter{};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
explicit task_job_manager(int max_concurrent_slots = 4) : slot_counter(max_concurrent_slots) {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void start_slot(slot_point slot, int nof_sectors)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(mutex);
|
|
|
|
public:
|
|
|
|
auto& sl = slot_counter[slot.to_uint() % slot_counter.size()];
|
|
|
|
using sched_nr_sim_parallel::sched_nr_sim_parallel;
|
|
|
|
while (sl.count > 0) {
|
|
|
|
|
|
|
|
sl.cvar.wait(lock);
|
|
|
|
void process_cc_result(const cc_result_t& cc_result) override
|
|
|
|
}
|
|
|
|
|
|
|
|
sl.count = nof_sectors;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void finish_cc(slot_point slot,
|
|
|
|
|
|
|
|
const sched_nr_interface::dl_sched_res_t& dl_res,
|
|
|
|
|
|
|
|
const sched_nr_interface::ul_sched_t& ul_res)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
std::unique_lock<std::mutex> lock(mutex);
|
|
|
|
|
|
|
|
TESTASSERT(dl_res.dl_sched.pdcch_dl.size() <= 1);
|
|
|
|
|
|
|
|
res_count++;
|
|
|
|
|
|
|
|
pdsch_count += dl_res.dl_sched.pdcch_dl.size();
|
|
|
|
|
|
|
|
auto& sl = slot_counter[slot.to_uint() % slot_counter.size()];
|
|
|
|
|
|
|
|
if (--sl.count == 0) {
|
|
|
|
|
|
|
|
sl.cvar.notify_one();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void wait_task_finish()
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(mutex);
|
|
|
|
tot_latency_sched_ns += cc_result.sched_latency_ns.count();
|
|
|
|
for (auto& sl : slot_counter) {
|
|
|
|
result_count++;
|
|
|
|
while (sl.count > 0) {
|
|
|
|
pdsch_count += cc_result.dl_res.dl_sched.pdcch_dl.size();
|
|
|
|
sl.cvar.wait(lock);
|
|
|
|
|
|
|
|
}
|
|
|
|
TESTASSERT(cc_result.dl_res.dl_sched.pdcch_dl.size() <= 1);
|
|
|
|
sl.count = 1;
|
|
|
|
if (srsran_duplex_nr_is_dl(&cell_params[cc_result.cc].cfg.duplex, 0, current_slot_tx.slot_idx())) {
|
|
|
|
|
|
|
|
TESTASSERT(cc_result.dl_res.dl_sched.pdcch_dl.size() == 1 or not cc_result.dl_res.dl_sched.ssb.empty());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void print_results() const
|
|
|
|
void print_results() const
|
|
|
|
{
|
|
|
|
{
|
|
|
|
test_logger.info("TESTER: %f PDSCH/{slot,cc} were allocated", pdsch_count / (double)res_count);
|
|
|
|
test_logger.info("TESTER: %f PDSCH/{slot,cc} were allocated", pdsch_count / (double)result_count);
|
|
|
|
srslog::flush();
|
|
|
|
srslog::flush();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
srslog::basic_logger& test_logger = srslog::fetch_basic_logger("TEST");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint64_t tot_latency_sched_ns = 0;
|
|
|
|
|
|
|
|
uint32_t result_count = 0;
|
|
|
|
|
|
|
|
uint32_t pdsch_count = 0;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
void sched_nr_cfg_serialized_test()
|
|
|
|
void sched_nr_cfg_serialized_test()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
uint32_t max_nof_ttis = 1000, nof_sectors = 4;
|
|
|
|
uint32_t max_nof_ttis = 1000, nof_sectors = 4;
|
|
|
|
task_job_manager tasks;
|
|
|
|
uint16_t rnti = 0x4601;
|
|
|
|
|
|
|
|
|
|
|
|
sched_nr_interface::sched_args_t cfg;
|
|
|
|
sched_nr_interface::sched_args_t cfg;
|
|
|
|
cfg.auto_refill_buffer = true;
|
|
|
|
cfg.auto_refill_buffer = true;
|
|
|
|
|
|
|
|
|
|
|
|
std::vector<sched_nr_interface::cell_cfg_t> cells_cfg = get_default_cells_cfg(nof_sectors);
|
|
|
|
std::vector<sched_nr_interface::cell_cfg_t> cells_cfg = get_default_cells_cfg(nof_sectors);
|
|
|
|
|
|
|
|
|
|
|
|
sched_nr_sim_base sched_tester(cfg, cells_cfg, "Serialized Test");
|
|
|
|
sched_nr_tester tester(cfg, cells_cfg, "Serialized Test");
|
|
|
|
|
|
|
|
|
|
|
|
sched_nr_interface::ue_cfg_t uecfg = get_default_ue_cfg(nof_sectors);
|
|
|
|
sched_nr_interface::ue_cfg_t uecfg = get_default_ue_cfg(nof_sectors);
|
|
|
|
sched_tester.add_user(0x46, uecfg, slot_point{0, 0}, 0);
|
|
|
|
tester.add_user(rnti, uecfg, slot_point{0, 0}, 0);
|
|
|
|
|
|
|
|
|
|
|
|
std::vector<long> count_per_cc(nof_sectors, 0);
|
|
|
|
|
|
|
|
for (uint32_t nof_slots = 0; nof_slots < max_nof_ttis; ++nof_slots) {
|
|
|
|
for (uint32_t nof_slots = 0; nof_slots < max_nof_ttis; ++nof_slots) {
|
|
|
|
slot_point slot_rx(0, nof_slots % 10240);
|
|
|
|
slot_point slot_rx(0, nof_slots % 10240);
|
|
|
|
slot_point slot_tx = slot_rx + TX_ENB_DELAY;
|
|
|
|
slot_point slot_tx = slot_rx + TX_ENB_DELAY;
|
|
|
|
tasks.start_slot(slot_rx, nof_sectors);
|
|
|
|
tester.new_slot(slot_tx);
|
|
|
|
sched_tester.new_slot(slot_tx);
|
|
|
|
for (uint32_t cc = 0; cc != cells_cfg.size(); ++cc) {
|
|
|
|
for (uint32_t cc = 0; cc < cells_cfg.size(); ++cc) {
|
|
|
|
tester.generate_cc_result(cc);
|
|
|
|
sched_nr_interface::dl_sched_res_t dl_res;
|
|
|
|
|
|
|
|
sched_nr_interface::ul_sched_t ul_res;
|
|
|
|
|
|
|
|
auto tp1 = std::chrono::steady_clock::now();
|
|
|
|
|
|
|
|
TESTASSERT(sched_tester.get_sched()->run_slot(slot_tx, cc, dl_res) == SRSRAN_SUCCESS);
|
|
|
|
|
|
|
|
TESTASSERT(sched_tester.get_sched()->get_ul_sched(slot_tx, cc, ul_res) == SRSRAN_SUCCESS);
|
|
|
|
|
|
|
|
auto tp2 = std::chrono::steady_clock::now();
|
|
|
|
|
|
|
|
count_per_cc[cc] += std::chrono::duration_cast<std::chrono::nanoseconds>(tp2 - tp1).count();
|
|
|
|
|
|
|
|
sched_nr_cc_output_res_t out{slot_tx, cc, &dl_res, &ul_res};
|
|
|
|
|
|
|
|
sched_tester.update(out);
|
|
|
|
|
|
|
|
tasks.finish_cc(slot_rx, dl_res, ul_res);
|
|
|
|
|
|
|
|
TESTASSERT(not srsran_duplex_nr_is_dl(&cells_cfg[cc].duplex, 0, (slot_tx).slot_idx()) or
|
|
|
|
|
|
|
|
(dl_res.dl_sched.pdcch_dl.size() == 1 or not dl_res.dl_sched.ssb.empty()));
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
tasks.print_results();
|
|
|
|
tester.stop();
|
|
|
|
|
|
|
|
tester.print_results();
|
|
|
|
// TESTASSERT(tasks.pdsch_count == (int)(max_nof_ttis * nof_sectors * 0.6));
|
|
|
|
// TESTASSERT(tasks.pdsch_count == (int)(max_nof_ttis * nof_sectors * 0.6));
|
|
|
|
|
|
|
|
|
|
|
|
double final_avg_usec = 0;
|
|
|
|
double final_avg_usec = tester.tot_latency_sched_ns;
|
|
|
|
for (uint32_t cc = 0; cc < cells_cfg.size(); ++cc) {
|
|
|
|
|
|
|
|
final_avg_usec += count_per_cc[cc];
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
final_avg_usec = final_avg_usec / 1000.0 / max_nof_ttis;
|
|
|
|
final_avg_usec = final_avg_usec / 1000.0 / max_nof_ttis;
|
|
|
|
printf("Total time taken per slot: %f usec\n", final_avg_usec);
|
|
|
|
printf("Total time taken per slot: %f usec\n", final_avg_usec);
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -128,50 +91,46 @@ void sched_nr_cfg_parallel_cc_test()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
uint32_t nof_sectors = 4;
|
|
|
|
uint32_t nof_sectors = 4;
|
|
|
|
uint32_t max_nof_ttis = 1000;
|
|
|
|
uint32_t max_nof_ttis = 1000;
|
|
|
|
task_job_manager tasks;
|
|
|
|
uint16_t rnti = 0x4601;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Initiate CC Workers
|
|
|
|
|
|
|
|
std::vector<std::unique_ptr<srsran::task_worker> > cc_workers;
|
|
|
|
|
|
|
|
cc_workers.reserve(nof_sectors - 1);
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < nof_sectors - 1; ++i) {
|
|
|
|
|
|
|
|
fmt::memory_buffer fmtbuf;
|
|
|
|
|
|
|
|
fmt::format_to(fmtbuf, "worker{}", i);
|
|
|
|
|
|
|
|
cc_workers.emplace_back(new srsran::task_worker{to_string(fmtbuf), 10});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sched_nr_interface::sched_args_t cfg;
|
|
|
|
sched_nr_interface::sched_args_t cfg;
|
|
|
|
cfg.auto_refill_buffer = true;
|
|
|
|
cfg.auto_refill_buffer = true;
|
|
|
|
|
|
|
|
|
|
|
|
std::vector<sched_nr_interface::cell_cfg_t> cells_cfg = get_default_cells_cfg(nof_sectors);
|
|
|
|
std::vector<sched_nr_interface::cell_cfg_t> cells_cfg = get_default_cells_cfg(nof_sectors);
|
|
|
|
|
|
|
|
|
|
|
|
sched_nr_sim_base sched_tester(cfg, cells_cfg, "Parallel CC Test");
|
|
|
|
sched_nr_tester tester(cfg, cells_cfg, "Parallel CC Test");
|
|
|
|
|
|
|
|
|
|
|
|
sched_nr_interface::ue_cfg_t uecfg = get_default_ue_cfg(cells_cfg.size());
|
|
|
|
sched_nr_interface::ue_cfg_t uecfg = get_default_ue_cfg(cells_cfg.size());
|
|
|
|
sched_tester.add_user(0x46, uecfg, slot_point{0, 0}, 0);
|
|
|
|
tester.add_user(rnti, uecfg, slot_point{0, 0}, 0);
|
|
|
|
|
|
|
|
|
|
|
|
std::array<std::atomic<long>, SRSRAN_MAX_CARRIERS> nano_count{};
|
|
|
|
|
|
|
|
for (uint32_t nof_slots = 0; nof_slots < max_nof_ttis; ++nof_slots) {
|
|
|
|
for (uint32_t nof_slots = 0; nof_slots < max_nof_ttis; ++nof_slots) {
|
|
|
|
slot_point slot_rx(0, nof_slots % 10240);
|
|
|
|
slot_point slot_rx(0, nof_slots % 10240);
|
|
|
|
slot_point slot_tx = slot_rx + TX_ENB_DELAY;
|
|
|
|
slot_point slot_tx = slot_rx + TX_ENB_DELAY;
|
|
|
|
tasks.start_slot(slot_tx, nof_sectors);
|
|
|
|
tester.new_slot(slot_tx);
|
|
|
|
sched_tester.new_slot(slot_tx);
|
|
|
|
// Run scheduler in parallel for {0, cc-2} and in the same thread for last cc
|
|
|
|
for (uint32_t cc = 0; cc < cells_cfg.size(); ++cc) {
|
|
|
|
for (uint32_t cc = 0; cc != cells_cfg.size() - 1; ++cc) {
|
|
|
|
srsran::get_background_workers().push_task([cc, slot_tx, &tasks, &sched_tester, &nano_count]() {
|
|
|
|
cc_workers[cc]->push_task([cc, &tester]() { tester.generate_cc_result(cc); });
|
|
|
|
sched_nr_interface::dl_sched_res_t dl_res;
|
|
|
|
|
|
|
|
sched_nr_interface::ul_sched_t ul_res;
|
|
|
|
|
|
|
|
auto tp1 = std::chrono::steady_clock::now();
|
|
|
|
|
|
|
|
TESTASSERT(sched_tester.get_sched()->run_slot(slot_tx, cc, dl_res) == SRSRAN_SUCCESS);
|
|
|
|
|
|
|
|
TESTASSERT(sched_tester.get_sched()->get_ul_sched(slot_tx, cc, ul_res) == SRSRAN_SUCCESS);
|
|
|
|
|
|
|
|
auto tp2 = std::chrono::steady_clock::now();
|
|
|
|
|
|
|
|
nano_count[cc].fetch_add(std::chrono::duration_cast<std::chrono::nanoseconds>(tp2 - tp1).count(),
|
|
|
|
|
|
|
|
std::memory_order_relaxed);
|
|
|
|
|
|
|
|
sched_nr_cc_output_res_t out{slot_tx, cc, &dl_res, &ul_res};
|
|
|
|
|
|
|
|
sched_tester.update(out);
|
|
|
|
|
|
|
|
tasks.finish_cc(slot_tx, dl_res, ul_res);
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tester.generate_cc_result(cells_cfg.size() - 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
tasks.wait_task_finish();
|
|
|
|
// Wait for all jobs to finish
|
|
|
|
|
|
|
|
tester.stop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
tester.print_results();
|
|
|
|
|
|
|
|
|
|
|
|
tasks.print_results();
|
|
|
|
|
|
|
|
// TESTASSERT(tasks.pdsch_count == (int)(max_nof_ttis * nof_sectors * 0.6));
|
|
|
|
// TESTASSERT(tasks.pdsch_count == (int)(max_nof_ttis * nof_sectors * 0.6));
|
|
|
|
|
|
|
|
|
|
|
|
double final_avg_usec = 0;
|
|
|
|
double final_avg_usec = tester.tot_latency_sched_ns;
|
|
|
|
for (uint32_t i = 0; i < nof_sectors; ++i) {
|
|
|
|
|
|
|
|
final_avg_usec += nano_count[i];
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
final_avg_usec = final_avg_usec / 1000.0 / max_nof_ttis / nof_sectors;
|
|
|
|
final_avg_usec = final_avg_usec / 1000.0 / max_nof_ttis / nof_sectors;
|
|
|
|
printf("Total time taken per slot [usec]: %f\n", final_avg_usec);
|
|
|
|
printf("Total time taken per slot [usec]: %f\n", final_avg_usec);
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -181,17 +140,15 @@ void sched_nr_cfg_parallel_cc_test()
|
|
|
|
int main()
|
|
|
|
int main()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
auto& test_logger = srslog::fetch_basic_logger("TEST");
|
|
|
|
auto& test_logger = srslog::fetch_basic_logger("TEST");
|
|
|
|
test_logger.set_level(srslog::basic_levels::info);
|
|
|
|
test_logger.set_level(srslog::basic_levels::error);
|
|
|
|
auto& mac_logger = srslog::fetch_basic_logger("MAC");
|
|
|
|
auto& mac_nr_logger = srslog::fetch_basic_logger("MAC-NR");
|
|
|
|
mac_logger.set_level(srslog::basic_levels::info);
|
|
|
|
mac_nr_logger.set_level(srslog::basic_levels::error);
|
|
|
|
auto& pool_logger = srslog::fetch_basic_logger("POOL");
|
|
|
|
auto& pool_logger = srslog::fetch_basic_logger("POOL");
|
|
|
|
pool_logger.set_level(srslog::basic_levels::info);
|
|
|
|
pool_logger.set_level(srslog::basic_levels::debug);
|
|
|
|
|
|
|
|
|
|
|
|
// Start the log backend.
|
|
|
|
// Start the log backend.
|
|
|
|
srslog::init();
|
|
|
|
srslog::init();
|
|
|
|
|
|
|
|
|
|
|
|
srsran::get_background_workers().set_nof_workers(6);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
srsenb::sched_nr_cfg_serialized_test();
|
|
|
|
srsenb::sched_nr_cfg_serialized_test();
|
|
|
|
srsenb::sched_nr_cfg_parallel_cc_test();
|
|
|
|
srsenb::sched_nr_cfg_parallel_cc_test();
|
|
|
|
}
|
|
|
|
}
|
|
|
|