From 786d29962bd624040172e4facc912e2f0a222035 Mon Sep 17 00:00:00 2001 From: frankist Date: Mon, 23 Apr 2018 17:28:29 +0100 Subject: [PATCH] refactoring and improvement of time-domain RR scheduling (#181) * fixed the scheduler not filling the mask for multiple UEs * fixed a bug related with the reset of the allocations * The RR now saves the last UE to be assigned --- srsenb/hdr/mac/scheduler_metric.h | 13 ++- srsenb/hdr/mac/scheduler_ue.h | 3 +- srsenb/src/mac/scheduler_metric.cc | 132 ++++++++++++++--------------- srsenb/src/mac/scheduler_ue.cc | 2 +- 4 files changed, 73 insertions(+), 77 deletions(-) diff --git a/srsenb/hdr/mac/scheduler_metric.h b/srsenb/hdr/mac/scheduler_metric.h index 11e77b455..4685f932a 100644 --- a/srsenb/hdr/mac/scheduler_metric.h +++ b/srsenb/hdr/mac/scheduler_metric.h @@ -42,18 +42,17 @@ private: bool new_allocation(uint32_t nof_rbg, uint32_t* rbgmask); void update_allocation(uint32_t new_mask); - bool allocation_is_valid(uint32_t mask); - - + bool allocation_is_valid(uint32_t mask); + dl_harq_proc* apply_user_allocation(sched_ue *user); + uint32_t get_required_rbg(sched_ue *user, uint32_t tti); uint32_t count_rbg(uint32_t mask); uint32_t calc_rbg_mask(bool mask[25]); bool used_rb[MAX_RBG]; - uint32_t nof_users_with_data; - - uint32_t current_tti; + + uint32_t current_tti; uint32_t total_rb; uint32_t used_rb_mask; uint32_t nof_ctrl_symbols; @@ -72,8 +71,8 @@ private: bool new_allocation(uint32_t L, ul_harq_proc::ul_alloc_t *alloc); bool allocation_is_valid(ul_harq_proc::ul_alloc_t alloc); + ul_harq_proc* apply_user_allocation(sched_ue *user); - uint32_t nof_users_with_data; bool used_rb[MAX_PRB]; uint32_t current_tti; diff --git a/srsenb/hdr/mac/scheduler_ue.h b/srsenb/hdr/mac/scheduler_ue.h index 90db552a6..aaa4b49b7 100644 --- a/srsenb/hdr/mac/scheduler_ue.h +++ b/srsenb/hdr/mac/scheduler_ue.h @@ -40,7 +40,8 @@ class sched_ue { public: // used by sched_metric - uint32_t ue_idx; + dl_harq_proc* dl_next_alloc; + ul_harq_proc* ul_next_alloc; bool has_pucch; diff --git a/srsenb/src/mac/scheduler_metric.cc b/srsenb/src/mac/scheduler_metric.cc index a15faebbd..f6ec18555 100644 --- a/srsenb/src/mac/scheduler_metric.cc +++ b/srsenb/src/mac/scheduler_metric.cc @@ -78,7 +78,6 @@ uint32_t dl_metric_rr::get_required_rbg(sched_ue *user, uint32_t tti) void dl_metric_rr::new_tti(std::map &ue_db, uint32_t start_rb, uint32_t nof_rb, uint32_t nof_ctrl_symbols_, uint32_t tti) { - total_rb = start_rb+nof_rb; for (uint32_t i=0;i &ue_db, uint32_t start_rb } available_rb = nof_rb; used_rb_mask = calc_rbg_mask(used_rb); - current_tti = tti; - nof_ctrl_symbols = nof_ctrl_symbols_; - - nof_users_with_data = 0; - for(std::map::iterator iter=ue_db.begin(); iter!=ue_db.end(); ++iter) { - sched_ue *user = (sched_ue*) &iter->second; - if (user->get_pending_dl_new_data(current_tti) || user->get_pending_dl_harq(current_tti)) { - user->ue_idx = nof_users_with_data; - nof_users_with_data++; + current_tti = tti; + nof_ctrl_symbols = nof_ctrl_symbols_; + + if(ue_db.size()==0) + return; + + // give priority in a time-domain RR basis + uint32_t priority_idx = current_tti % ue_db.size(); + std::map::iterator iter = ue_db.begin(); + std::advance(iter,priority_idx); + for(uint32_t ue_count = 0 ; ue_count < ue_db.size() ; ++iter, ++ue_count) { + if(iter==ue_db.end()) { + iter = ue_db.begin(); // wrap around } + sched_ue *user = (sched_ue*) &iter->second; + user->dl_next_alloc = apply_user_allocation(user); } } @@ -136,25 +141,11 @@ bool dl_metric_rr::allocation_is_valid(uint32_t mask) return (mask & used_rb_mask); } -dl_harq_proc* dl_metric_rr::get_user_allocation(sched_ue *user) -{ - uint32_t pending_data = user->get_pending_dl_new_data(current_tti); +dl_harq_proc* dl_metric_rr::apply_user_allocation(sched_ue *user) { + uint32_t pending_data = user->get_pending_dl_new_data(current_tti); dl_harq_proc *h = user->get_pending_dl_harq(current_tti); - // Time-domain RR scheduling -#if ASYNC_DL_SCHED - if (pending_data || h) { -#else - if (pending_data || (h && !h->is_empty())) { -#endif - if (nof_users_with_data) { - if ((current_tti%nof_users_with_data) != user->ue_idx) { - return NULL; - } - } - } - - // Schedule retx if we have space + // Schedule retx if we have space #if ASYNC_DL_SCHED if (h) { #else @@ -164,38 +155,45 @@ dl_harq_proc* dl_metric_rr::get_user_allocation(sched_ue *user) // If can schedule the same mask, do it if (!allocation_is_valid(retx_mask)) { update_allocation(retx_mask); - return h; + return h; } - // If not, try to find another mask in the current tti + + // If not, try to find another mask in the current tti uint32_t nof_rbg = count_rbg(retx_mask); if (nof_rbg < available_rb) { if (new_allocation(nof_rbg, &retx_mask)) { update_allocation(retx_mask); h->set_rbgmask(retx_mask); - return h; + return h; } } - } + } // If could not schedule the reTx, or there wasn't any pending retx, find an empty PID #if ASYNC_DL_SCHED - h = user->get_empty_dl_harq(); + h = user->get_empty_dl_harq(); if (h) { #else - if (h && h->is_empty()) { + if (h && h->is_empty()) { #endif // Allocate resources based on pending data if (pending_data) { uint32_t pending_rb = user->get_required_prb_dl(pending_data, nof_ctrl_symbols); - uint32_t newtx_mask = 0; + uint32_t newtx_mask = 0; new_allocation(pending_rb, &newtx_mask); if (newtx_mask) { update_allocation(newtx_mask); h->set_rbgmask(newtx_mask); - return h; + return h; } - } + } } - return NULL; + + return NULL; +} + +dl_harq_proc* dl_metric_rr::get_user_allocation(sched_ue *user) +{ + return user->dl_next_alloc; } @@ -219,15 +217,20 @@ void ul_metric_rr::new_tti(std::map &ue_db, uint32_t nof_rb_, available_rb = nof_rb_; bzero(used_rb, nof_rb*sizeof(bool)); - nof_users_with_data = 0; - for(std::map::iterator iter=ue_db.begin(); iter!=ue_db.end(); ++iter) { - sched_ue *user = (sched_ue*) &iter->second; - if (user->get_pending_ul_new_data(current_tti) || !user->get_ul_harq(current_tti)->is_empty(0)) { - user->ue_idx = nof_users_with_data; - nof_users_with_data++; + if(ue_db.size()==0) + return; + + // give priority in a time-domain RR basis + uint32_t priority_idx = (current_tti+ue_db.size()/2) % ue_db.size(); // make DL and UL interleaved + std::map::iterator iter = ue_db.begin(); + std::advance(iter,priority_idx); + for(uint32_t ue_count = 0 ; ue_count < ue_db.size() ; ++iter, ++ue_count) { + if(iter==ue_db.end()) { + iter = ue_db.begin(); // wrap around } + sched_ue *user = (sched_ue*) &iter->second; + user->ul_next_alloc = apply_user_allocation(user); } - } bool ul_metric_rr::allocation_is_valid(ul_harq_proc::ul_alloc_t alloc) @@ -288,56 +291,49 @@ void ul_metric_rr::update_allocation(ul_harq_proc::ul_alloc_t alloc) available_rb -= alloc.L; } -ul_harq_proc* ul_metric_rr::get_user_allocation(sched_ue *user) -{ +ul_harq_proc* ul_metric_rr::apply_user_allocation(sched_ue *user) { // Time-domain RR scheduling - uint32_t pending_data = user->get_pending_ul_new_data(current_tti); + uint32_t pending_data = user->get_pending_ul_new_data(current_tti); ul_harq_proc *h = user->get_ul_harq(current_tti); - - if (pending_data || !h->is_empty(0)) { - if (nof_users_with_data) { - if ((current_tti%nof_users_with_data) != user->ue_idx) { - return NULL; - } - } - } - // Schedule retx if we have space - + // Schedule retx if we have space if (!h->is_empty(0)) { - ul_harq_proc::ul_alloc_t alloc = h->get_alloc(); - + // If can schedule the same mask, do it if (allocation_is_valid(alloc)) { update_allocation(alloc); return h; } - + // If not, try to find another mask in the current tti if (new_allocation(alloc.L, &alloc)) { update_allocation(alloc); h->set_alloc(alloc); return h; } - } - // If could not schedule the reTx, or there wasn't any pending retx, find an empty PID + } + + // If could not schedule the reTx, or there wasn't any pending retx, find an empty PID if (h->is_empty(0)) { - // Allocate resources based on pending data + // Allocate resources based on pending data if (pending_data) { uint32_t pending_rb = user->get_required_prb_ul(pending_data); - ul_harq_proc::ul_alloc_t alloc; + ul_harq_proc::ul_alloc_t alloc; new_allocation(pending_rb, &alloc); if (alloc.L) { update_allocation(alloc); h->set_alloc(alloc); - return h; + return h; } - } + } } - return NULL; + return NULL; } - +ul_harq_proc* ul_metric_rr::get_user_allocation(sched_ue *user) +{ + return user->ul_next_alloc; +} } diff --git a/srsenb/src/mac/scheduler_ue.cc b/srsenb/src/mac/scheduler_ue.cc index 6cb4462e8..f9245ac9b 100644 --- a/srsenb/src/mac/scheduler_ue.cc +++ b/srsenb/src/mac/scheduler_ue.cc @@ -49,7 +49,7 @@ namespace srsenb { * *******************************************************/ -sched_ue::sched_ue() : ue_idx(0), has_pucch(false), power_headroom(0), rnti(0), max_mcs_dl(0), max_mcs_ul(0), +sched_ue::sched_ue() : dl_next_alloc(NULL), ul_next_alloc(NULL), has_pucch(false), power_headroom(0), rnti(0), max_mcs_dl(0), max_mcs_ul(0), fixed_mcs_ul(0), fixed_mcs_dl(0), phy_config_dedicated_enabled(false) { log_h = NULL;