Track UL buffers per TTI instead of per PID and remove old ones periodically

master
Ismael Gomez 4 years ago
parent c75b5057fa
commit 44e411be2b

@ -76,6 +76,7 @@ public:
void process_pdu(uint8_t* pdu, uint32_t nof_bytes, srslte::pdu_queue::channel_t channel) override; void process_pdu(uint8_t* pdu, uint32_t nof_bytes, srslte::pdu_queue::channel_t channel) override;
void push_pdu(uint32_t tti, uint32_t ue_cc_idx, uint32_t len); void push_pdu(uint32_t tti, uint32_t ue_cc_idx, uint32_t len);
void deallocate_pdu(uint32_t tti, uint32_t ue_cc_idx); void deallocate_pdu(uint32_t tti, uint32_t ue_cc_idx);
void clear_old_buffers(uint32_t tti);
void metrics_read(mac_ue_metrics_t* metrics_); void metrics_read(mac_ue_metrics_t* metrics_);
void metrics_rx(bool crc, uint32_t tbs); void metrics_rx(bool crc, uint32_t tbs);
@ -123,8 +124,7 @@ private:
std::vector<std::array<std::array<srslte::unique_byte_buffer_t, SRSLTE_MAX_TB>, SRSLTE_FDD_NOF_HARQ> > std::vector<std::array<std::array<srslte::unique_byte_buffer_t, SRSLTE_MAX_TB>, SRSLTE_FDD_NOF_HARQ> >
tx_payload_buffer; tx_payload_buffer;
// Save 2 buffers per HARQ process std::vector<std::map<uint32_t, uint8_t*> > rx_used_buffers;
std::vector<srslte::circular_array<uint8_t*, 2 * SRSLTE_FDD_NOF_HARQ> > rx_used_buffers;
srslte::block_queue<uint32_t> pending_ta_commands; srslte::block_queue<uint32_t> pending_ta_commands;
ta ta_fsm; ta ta_fsm;

@ -940,6 +940,10 @@ int mac::get_ul_sched(uint32_t tti_tx_ul, ul_sched_list_t& ul_sched_res_list)
} }
phy_ul_sched_res->nof_phich = sched_result.nof_phich_elems; phy_ul_sched_res->nof_phich = sched_result.nof_phich_elems;
} }
// clear old buffers from all users
for (auto& u : ue_db) {
u.second->clear_old_buffers(tti_tx_ul);
}
return SRSLTE_SUCCESS; return SRSLTE_SUCCESS;
} }

@ -77,6 +77,12 @@ ue::~ue()
srslte_softbuffer_tx_free(&buffer); srslte_softbuffer_tx_free(&buffer);
} }
} }
for (auto& rx_buffers_cc : rx_used_buffers) {
for (auto& q : rx_buffers_cc) {
pdus.deallocate(q.second);
}
rx_buffers_cc.clear();
}
} }
void ue::reset() void ue::reset()
@ -95,15 +101,6 @@ void ue::reset()
srslte_softbuffer_tx_reset(&buffer); srslte_softbuffer_tx_reset(&buffer);
} }
} }
for (auto& rx_buffers_cc : rx_used_buffers) {
for (auto& ptr : rx_buffers_cc) {
if (ptr) {
pdus.deallocate(ptr);
ptr = nullptr;
}
}
}
} }
/** /**
@ -175,25 +172,39 @@ uint8_t* ue::request_buffer(uint32_t tti, uint32_t cc_idx, const uint32_t len)
{ {
uint8_t* pdu = nullptr; uint8_t* pdu = nullptr;
if (len > 0) { if (len > 0) {
// Deallocate oldest buffer if we didn't deallocate it
if (!rx_used_buffers.at(cc_idx).count(tti)) {
pdu = pdus.request(len); pdu = pdus.request(len);
if (pdu) { if (pdu) {
// Deallocate oldest buffer if we didn't deallocate it rx_used_buffers.at(cc_idx).emplace(tti, pdu);
if (rx_used_buffers.at(cc_idx)[tti] != nullptr) { } else {
pdus.deallocate(rx_used_buffers.at(cc_idx)[tti]); Error("UE buffers: Requesting buffer from pool\n");
rx_used_buffers.at(cc_idx)[tti] = nullptr;
log_h->warning("buffers: RX PDU of rnti=0x%x and pid=%d wasn't deallocated\n", rnti, tti % nof_rx_harq_proc);
} }
rx_used_buffers.at(cc_idx)[tti] = pdu;
log_h->info("RX PDU saved for pid=%d\n", tti % nof_rx_harq_proc);
} else { } else {
log_h->error("buffers: Requesting buffer from pool\n"); Error("UE buffers: buffer for tti=%d already allocated\n", tti);
} }
} else { } else {
printf("buffers: Requesting buffer for zero bytes\n"); Error("UE buffers: Requesting buffer for zero bytes\n");
} }
return pdu; return pdu;
} }
void ue::clear_old_buffers(uint32_t tti)
{
// remove old buffers
for (auto& rx_buffer_cc : rx_used_buffers) {
for (auto it = rx_buffer_cc.begin(); it != rx_buffer_cc.end();) {
if (srslte_tti_interval(tti, it->first) > 20) {
Warning("UE buffers: Removing old buffer tti=%d, rnti=%d, now is %d\n", it->first, rnti, tti);
pdus.deallocate(it->second);
it = rx_buffer_cc.erase(it);
} else {
++it;
}
}
}
}
bool ue::process_pdus() bool ue::process_pdus()
{ {
return pdus.process_pdus(); return pdus.process_pdus();
@ -318,11 +329,11 @@ void ue::process_pdu(uint8_t* pdu, uint32_t nof_bytes, srslte::pdu_queue::channe
void ue::deallocate_pdu(uint32_t tti, uint32_t cc_idx) void ue::deallocate_pdu(uint32_t tti, uint32_t cc_idx)
{ {
if (rx_used_buffers.at(cc_idx)[tti] != nullptr) { if (rx_used_buffers.at(cc_idx).count(tti)) {
pdus.deallocate(rx_used_buffers.at(cc_idx)[tti]); pdus.deallocate(rx_used_buffers.at(cc_idx).at(tti));
rx_used_buffers.at(cc_idx)[tti] = nullptr; rx_used_buffers.at(cc_idx).erase(tti);
} else { } else {
Warning("buffers: Null RX PDU pointer in deallocate_pdu for rnti=0x%x pid=%d cc_idx=%d\n", Warning("UE buffers: Null RX PDU pointer in deallocate_pdu for rnti=0x%x pid=%d cc_idx=%d\n",
rnti, rnti,
tti % nof_rx_harq_proc, tti % nof_rx_harq_proc,
cc_idx); cc_idx);
@ -331,15 +342,15 @@ void ue::deallocate_pdu(uint32_t tti, uint32_t cc_idx)
void ue::push_pdu(uint32_t tti, uint32_t cc_idx, uint32_t len) void ue::push_pdu(uint32_t tti, uint32_t cc_idx, uint32_t len)
{ {
if (rx_used_buffers.at(cc_idx)[tti] != nullptr) { if (rx_used_buffers.at(cc_idx).count(tti)) {
if (len > 0) { if (len > 0) {
pdus.push(rx_used_buffers.at(cc_idx)[tti], len); pdus.push(rx_used_buffers.at(cc_idx).at(tti), len);
} else { } else {
Error("Error pushing PDU: null length\n"); Error("Error pushing PDU: null length\n");
} }
rx_used_buffers.at(cc_idx)[tti] = nullptr; rx_used_buffers.at(cc_idx).erase(tti);
} else { } else {
Warning("buffers: Null RX PDU pointer in push_pdu for rnti=0x%x pid=%d cc_idx=%d\n", Warning("UE buffers: Null RX PDU pointer in push_pdu for rnti=0x%x pid=%d cc_idx=%d\n",
rnti, rnti,
tti % nof_rx_harq_proc, tti % nof_rx_harq_proc,
cc_idx); cc_idx);

Loading…
Cancel
Save