|
|
|
@ -14,16 +14,42 @@
|
|
|
|
|
|
|
|
|
|
namespace srsenb {
|
|
|
|
|
|
|
|
|
|
rbgmask_t find_available_dl_rbgs(uint32_t max_rbgs, const rbgmask_t& current_mask)
|
|
|
|
|
/*********************************
|
|
|
|
|
* Common UL/DL Helper methods
|
|
|
|
|
********************************/
|
|
|
|
|
|
|
|
|
|
template <size_t N>
|
|
|
|
|
srslte::interval<uint32_t> find_contiguous_interval(const srslte::bounded_bitset<N, true>& in_mask, uint32_t max_size)
|
|
|
|
|
{
|
|
|
|
|
if (max_rbgs == 0 or current_mask.all()) {
|
|
|
|
|
return rbgmask_t{};
|
|
|
|
|
srslte::interval<uint32_t> interv, max_interv;
|
|
|
|
|
|
|
|
|
|
for (uint32_t n = 0; n < in_mask.size() and interv.length() < max_size; n++) {
|
|
|
|
|
if (not in_mask.test(n) and interv.empty()) {
|
|
|
|
|
// new interval
|
|
|
|
|
interv.set(n, n + 1);
|
|
|
|
|
} else if (not in_mask.test(n)) {
|
|
|
|
|
// extend current interval
|
|
|
|
|
interv.resize_by(1);
|
|
|
|
|
} else if (not interv.empty()) {
|
|
|
|
|
// reset interval
|
|
|
|
|
max_interv = interv.length() > max_interv.length() ? interv : max_interv;
|
|
|
|
|
interv = {};
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return interv.length() > max_interv.length() ? interv : max_interv;
|
|
|
|
|
}
|
|
|
|
|
// 1's for free rbgs
|
|
|
|
|
rbgmask_t localmask = ~(current_mask);
|
|
|
|
|
|
|
|
|
|
/****************************
|
|
|
|
|
* DL Helper methods
|
|
|
|
|
***************************/
|
|
|
|
|
|
|
|
|
|
rbgmask_t find_available_rb_mask(const rbgmask_t& in_mask, uint32_t max_size)
|
|
|
|
|
{
|
|
|
|
|
// 1's for free RBs
|
|
|
|
|
rbgmask_t localmask = ~(in_mask);
|
|
|
|
|
|
|
|
|
|
uint32_t i = 0, nof_alloc = 0;
|
|
|
|
|
for (; i < localmask.size() and nof_alloc < max_rbgs; ++i) {
|
|
|
|
|
for (; i < localmask.size() and nof_alloc < max_size; ++i) {
|
|
|
|
|
if (localmask.test(i)) {
|
|
|
|
|
nof_alloc++;
|
|
|
|
|
}
|
|
|
|
@ -32,42 +58,17 @@ rbgmask_t find_available_dl_rbgs(uint32_t max_rbgs, const rbgmask_t& current_mas
|
|
|
|
|
return localmask;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
prb_interval find_contiguous_ul_prbs(uint32_t L, const prbmask_t& current_mask)
|
|
|
|
|
rbgmask_t compute_user_rbgmask_greedy(uint32_t max_nof_rbgs, bool is_contiguous, const rbgmask_t& current_mask)
|
|
|
|
|
{
|
|
|
|
|
prb_interval prb_interv, prb_interv2;
|
|
|
|
|
for (uint32_t n = 0; n < current_mask.size() and prb_interv.length() < L; n++) {
|
|
|
|
|
if (not current_mask.test(n) and prb_interv.length() == 0) {
|
|
|
|
|
// new interval
|
|
|
|
|
prb_interv.set(n, n + 1);
|
|
|
|
|
} else if (not current_mask.test(n)) {
|
|
|
|
|
// extend current interval
|
|
|
|
|
prb_interv.resize_by(1);
|
|
|
|
|
} else if (prb_interv.length() > 0) {
|
|
|
|
|
// reset interval
|
|
|
|
|
prb_interv2 = prb_interv.length() > prb_interv2.length() ? prb_interv : prb_interv2;
|
|
|
|
|
prb_interv = {};
|
|
|
|
|
}
|
|
|
|
|
// Allocate enough RBs that accommodate pending data
|
|
|
|
|
rbgmask_t newtx_mask(current_mask.size());
|
|
|
|
|
if (is_contiguous) {
|
|
|
|
|
rbg_interval interv = find_contiguous_interval(current_mask, max_nof_rbgs);
|
|
|
|
|
newtx_mask.fill(interv.start(), interv.stop());
|
|
|
|
|
} else {
|
|
|
|
|
newtx_mask = find_available_rb_mask(current_mask, max_nof_rbgs);
|
|
|
|
|
}
|
|
|
|
|
prb_interv = prb_interv2.length() > prb_interv.length() ? prb_interv2 : prb_interv;
|
|
|
|
|
if (prb_interv.empty()) {
|
|
|
|
|
return prb_interv;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Make sure L is allowed by SC-FDMA modulation
|
|
|
|
|
prb_interv2 = prb_interv;
|
|
|
|
|
while (not srslte_dft_precoding_valid_prb(prb_interv.length()) and prb_interv.stop() < current_mask.size() and
|
|
|
|
|
not current_mask.test(prb_interv.stop())) {
|
|
|
|
|
prb_interv.resize_by(1);
|
|
|
|
|
}
|
|
|
|
|
if (not srslte_dft_precoding_valid_prb(prb_interv.length())) {
|
|
|
|
|
// if length increase failed, try to decrease
|
|
|
|
|
prb_interv = prb_interv2;
|
|
|
|
|
prb_interv.resize_by(-1);
|
|
|
|
|
while (not srslte_dft_precoding_valid_prb(prb_interv.length()) and not prb_interv.empty()) {
|
|
|
|
|
prb_interv.resize_by(-1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return prb_interv;
|
|
|
|
|
return newtx_mask;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int get_ue_cc_idx_if_pdsch_enabled(const sched_ue& user, sf_sched* tti_sched)
|
|
|
|
@ -104,6 +105,87 @@ const dl_harq_proc* get_dl_newtx_harq(sched_ue& user, sf_sched* tti_sched)
|
|
|
|
|
return user.get_empty_dl_harq(tti_sched->get_tti_tx_dl(), tti_sched->get_enb_cc_idx());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
alloc_outcome_t try_dl_retx_alloc(sf_sched& tti_sched, sched_ue& ue, const dl_harq_proc& h)
|
|
|
|
|
{
|
|
|
|
|
// Try to reuse the same mask
|
|
|
|
|
rbgmask_t retx_mask = h.get_rbgmask();
|
|
|
|
|
alloc_outcome_t code = tti_sched.alloc_dl_user(&ue, retx_mask, h.get_id());
|
|
|
|
|
if (code == alloc_outcome_t::SUCCESS or code == alloc_outcome_t::DCI_COLLISION) {
|
|
|
|
|
return code;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If previous mask does not fit, find another with exact same number of rbgs
|
|
|
|
|
size_t nof_rbg = retx_mask.count();
|
|
|
|
|
bool is_contiguous_alloc = ue.get_dci_format() == SRSLTE_DCI_FORMAT1A;
|
|
|
|
|
retx_mask = compute_user_rbgmask_greedy(nof_rbg, is_contiguous_alloc, tti_sched.get_dl_mask());
|
|
|
|
|
if (retx_mask.count() == nof_rbg) {
|
|
|
|
|
return tti_sched.alloc_dl_user(&ue, retx_mask, h.get_id());
|
|
|
|
|
}
|
|
|
|
|
return alloc_outcome_t::RB_COLLISION;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
alloc_outcome_t
|
|
|
|
|
try_dl_newtx_alloc_greedy(sf_sched& tti_sched, sched_ue& ue, const dl_harq_proc& h, rbgmask_t* result_mask)
|
|
|
|
|
{
|
|
|
|
|
if (result_mask != nullptr) {
|
|
|
|
|
*result_mask = {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If all RBGs are occupied, the next steps can be shortcut
|
|
|
|
|
const rbgmask_t& current_mask = tti_sched.get_dl_mask();
|
|
|
|
|
if (current_mask.all()) {
|
|
|
|
|
return alloc_outcome_t::RB_COLLISION;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If there is no data to transmit, no need to allocate
|
|
|
|
|
rbg_interval req_rbgs = ue.get_required_dl_rbgs(tti_sched.get_enb_cc_idx());
|
|
|
|
|
if (req_rbgs.stop() == 0) {
|
|
|
|
|
return alloc_outcome_t::NO_DATA;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Find RBG mask that accommodates pending data
|
|
|
|
|
bool is_contiguous_alloc = ue.get_dci_format() == SRSLTE_DCI_FORMAT1A;
|
|
|
|
|
rbgmask_t newtxmask = compute_user_rbgmask_greedy(req_rbgs.stop(), is_contiguous_alloc, current_mask);
|
|
|
|
|
if (newtxmask.none() or newtxmask.count() < req_rbgs.start()) {
|
|
|
|
|
return alloc_outcome_t::RB_COLLISION;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// empty RBGs were found. Attempt allocation
|
|
|
|
|
alloc_outcome_t ret = tti_sched.alloc_dl_user(&ue, newtxmask, h.get_id());
|
|
|
|
|
if (ret == alloc_outcome_t::SUCCESS and result_mask != nullptr) {
|
|
|
|
|
*result_mask = newtxmask;
|
|
|
|
|
}
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*****************
|
|
|
|
|
* UL Helpers
|
|
|
|
|
****************/
|
|
|
|
|
|
|
|
|
|
prb_interval find_contiguous_ul_prbs(uint32_t L, const prbmask_t& current_mask)
|
|
|
|
|
{
|
|
|
|
|
prb_interval prb_interv = find_contiguous_interval(current_mask, L);
|
|
|
|
|
if (prb_interv.empty()) {
|
|
|
|
|
return prb_interv;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Make sure L is allowed by SC-FDMA modulation
|
|
|
|
|
prb_interval prb_interv2 = prb_interv;
|
|
|
|
|
while (not srslte_dft_precoding_valid_prb(prb_interv.length()) and prb_interv.stop() < current_mask.size() and
|
|
|
|
|
not current_mask.test(prb_interv.stop())) {
|
|
|
|
|
prb_interv.resize_by(1);
|
|
|
|
|
}
|
|
|
|
|
if (not srslte_dft_precoding_valid_prb(prb_interv.length())) {
|
|
|
|
|
// if length increase failed, try to decrease
|
|
|
|
|
prb_interv = prb_interv2;
|
|
|
|
|
prb_interv.resize_by(-1);
|
|
|
|
|
while (not srslte_dft_precoding_valid_prb(prb_interv.length()) and not prb_interv.empty()) {
|
|
|
|
|
prb_interv.resize_by(-1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return prb_interv;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int get_ue_cc_idx_if_pusch_enabled(const sched_ue& user, sf_sched* tti_sched, bool needs_pdcch)
|
|
|
|
|
{
|
|
|
|
|
// Do not allocate a user multiple times in the same tti
|
|
|
|
@ -139,24 +221,6 @@ const ul_harq_proc* get_ul_newtx_harq(sched_ue& user, sf_sched* tti_sched)
|
|
|
|
|
return h->is_empty() ? h : nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
alloc_outcome_t try_dl_retx_alloc(sf_sched& tti_sched, sched_ue& ue, const dl_harq_proc& h)
|
|
|
|
|
{
|
|
|
|
|
// Try to reuse the same mask
|
|
|
|
|
rbgmask_t retx_mask = h.get_rbgmask();
|
|
|
|
|
alloc_outcome_t code = tti_sched.alloc_dl_user(&ue, retx_mask, h.get_id());
|
|
|
|
|
if (code == alloc_outcome_t::SUCCESS or code == alloc_outcome_t::DCI_COLLISION) {
|
|
|
|
|
return code;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If previous mask does not fit, find another with exact same number of rbgs
|
|
|
|
|
size_t nof_rbg = retx_mask.count();
|
|
|
|
|
retx_mask = find_available_dl_rbgs(nof_rbg, tti_sched.get_dl_mask());
|
|
|
|
|
if (retx_mask.count() == nof_rbg) {
|
|
|
|
|
return tti_sched.alloc_dl_user(&ue, retx_mask, h.get_id());
|
|
|
|
|
}
|
|
|
|
|
return alloc_outcome_t::RB_COLLISION;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
alloc_outcome_t try_ul_retx_alloc(sf_sched& tti_sched, sched_ue& ue, const ul_harq_proc& h)
|
|
|
|
|
{
|
|
|
|
|
// If can schedule the same mask, do it
|
|
|
|
|