|
|
|
@ -23,7 +23,101 @@
|
|
|
|
|
|
|
|
|
|
namespace srsenb {
|
|
|
|
|
|
|
|
|
|
cc_buffer_handler::cc_buffer_handler()
|
|
|
|
|
cc_used_buffers_map::cc_used_buffers_map(srslte::pdu_queue& shared_pdu_queue_) :
|
|
|
|
|
pdu_map(), shared_pdu_queue(&shared_pdu_queue_), logger(&srslog::fetch_basic_logger("MAC"))
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
bool cc_used_buffers_map::push_pdu(tti_point tti, uint32_t len)
|
|
|
|
|
{
|
|
|
|
|
if (not has_tti(tti)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
auto& pdu_pair = pdu_map[tti.to_uint()];
|
|
|
|
|
if (len > 0) {
|
|
|
|
|
shared_pdu_queue->push(pdu_pair.second, len);
|
|
|
|
|
} else {
|
|
|
|
|
logger->error("Error pushing PDU: null length");
|
|
|
|
|
}
|
|
|
|
|
// clear entry in map
|
|
|
|
|
pdu_pair.first = tti_point();
|
|
|
|
|
pdu_pair.second = nullptr;
|
|
|
|
|
return len > 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t* cc_used_buffers_map::request_pdu(tti_point tti, uint32_t len)
|
|
|
|
|
{
|
|
|
|
|
if (pdu_map[tti.to_uint()].second != nullptr) {
|
|
|
|
|
logger->error("UE buffers: buffer for tti=%d already allocated", tti.to_uint());
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t* pdu = shared_pdu_queue->request(len);
|
|
|
|
|
if (pdu == nullptr) {
|
|
|
|
|
logger->error("UE buffers: Requesting buffer from pool");
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pdu_map[tti.to_uint()].first = tti;
|
|
|
|
|
pdu_map[tti.to_uint()].second = pdu;
|
|
|
|
|
return pdu;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void cc_used_buffers_map::clear_old_pdus(tti_point current_tti)
|
|
|
|
|
{
|
|
|
|
|
static const uint32_t old_tti_threshold = SRSLTE_FDD_NOF_HARQ + 4;
|
|
|
|
|
|
|
|
|
|
tti_point max_tti{current_tti - old_tti_threshold};
|
|
|
|
|
for (auto& pdu_pair : pdu_map) {
|
|
|
|
|
if (pdu_pair.second != nullptr and pdu_pair.first < max_tti) {
|
|
|
|
|
logger->warning("UE buffers: Removing old buffer tti=%d, interval=%d",
|
|
|
|
|
pdu_pair.first.to_uint(),
|
|
|
|
|
current_tti - pdu_pair.first);
|
|
|
|
|
remove_pdu(pdu_pair.first);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void cc_used_buffers_map::remove_pdu(tti_point tti)
|
|
|
|
|
{
|
|
|
|
|
auto& pdu_pair = pdu_map[tti.to_uint()];
|
|
|
|
|
assert(pdu_pair.second != nullptr && "cannot remove inexistent PDU");
|
|
|
|
|
// return pdus back to the queue
|
|
|
|
|
shared_pdu_queue->deallocate(pdu_pair.second);
|
|
|
|
|
// clear entry in map
|
|
|
|
|
pdu_pair.first = tti_point();
|
|
|
|
|
pdu_pair.second = nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool cc_used_buffers_map::try_deallocate_pdu(tti_point tti)
|
|
|
|
|
{
|
|
|
|
|
if (pdu_map[tti.to_uint()].second == nullptr) {
|
|
|
|
|
remove_pdu(tti);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void cc_used_buffers_map::clear()
|
|
|
|
|
{
|
|
|
|
|
for (auto& pdu : pdu_map) {
|
|
|
|
|
remove_pdu(pdu.first);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint8_t*& cc_used_buffers_map::operator[](tti_point tti)
|
|
|
|
|
{
|
|
|
|
|
assert(has_tti(tti) && "Trying to access buffer that does not exist");
|
|
|
|
|
return pdu_map[tti.to_uint()].second;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool cc_used_buffers_map::has_tti(tti_point tti) const
|
|
|
|
|
{
|
|
|
|
|
return pdu_map[tti.to_uint()].first == tti and pdu_map[tti.to_uint()].second != nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
////////////////
|
|
|
|
|
|
|
|
|
|
cc_buffer_handler::cc_buffer_handler(srslte::pdu_queue& shared_pdu_queue_) : rx_used_buffers(shared_pdu_queue_)
|
|
|
|
|
{
|
|
|
|
|
for (auto& harq_buffers : tx_payload_buffer) {
|
|
|
|
|
for (srslte::unique_byte_buffer_t& tb_buffer : harq_buffers) {
|
|
|
|
@ -114,9 +208,12 @@ ue::ue(uint16_t rnti_,
|
|
|
|
|
pdus(logger),
|
|
|
|
|
nof_rx_harq_proc(nof_rx_harq_proc_),
|
|
|
|
|
nof_tx_harq_proc(nof_tx_harq_proc_),
|
|
|
|
|
ta_fsm(this),
|
|
|
|
|
cc_buffers(nof_cells_)
|
|
|
|
|
ta_fsm(this)
|
|
|
|
|
{
|
|
|
|
|
cc_buffers.reserve(nof_cells_);
|
|
|
|
|
for (size_t i = 0; i < nof_cells_; ++i) {
|
|
|
|
|
cc_buffers.emplace_back(pdus);
|
|
|
|
|
}
|
|
|
|
|
pdus.init(this);
|
|
|
|
|
|
|
|
|
|
// Allocate buffer for PCell
|
|
|
|
@ -125,15 +222,10 @@ ue::ue(uint16_t rnti_,
|
|
|
|
|
|
|
|
|
|
ue::~ue()
|
|
|
|
|
{
|
|
|
|
|
{
|
|
|
|
|
std::unique_lock<std::mutex> lock(rx_buffers_mutex);
|
|
|
|
|
for (auto& cc : cc_buffers) {
|
|
|
|
|
for (auto& q : cc.get_rx_used_buffers()) {
|
|
|
|
|
pdus.deallocate(q.second);
|
|
|
|
|
}
|
|
|
|
|
cc.get_rx_used_buffers().clear();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ue::reset()
|
|
|
|
@ -182,17 +274,7 @@ uint8_t* ue::request_buffer(uint32_t tti, uint32_t ue_cc_idx, const uint32_t len
|
|
|
|
|
std::unique_lock<std::mutex> lock(rx_buffers_mutex);
|
|
|
|
|
uint8_t* pdu = nullptr;
|
|
|
|
|
if (len > 0) {
|
|
|
|
|
// Deallocate oldest buffer if we didn't deallocate it
|
|
|
|
|
if (!cc_buffers.at(ue_cc_idx).get_rx_used_buffers().count(tti)) {
|
|
|
|
|
pdu = pdus.request(len);
|
|
|
|
|
if (pdu) {
|
|
|
|
|
cc_buffers.at(ue_cc_idx).get_rx_used_buffers().emplace(tti, pdu);
|
|
|
|
|
} else {
|
|
|
|
|
logger.error("UE buffers: Requesting buffer from pool");
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
logger.error("UE buffers: buffer for tti=%d already allocated", tti);
|
|
|
|
|
}
|
|
|
|
|
pdu = cc_buffers.at(ue_cc_idx).get_rx_used_buffers().request_pdu(tti_point(tti), len);
|
|
|
|
|
} else {
|
|
|
|
|
logger.error("UE buffers: Requesting buffer for zero bytes");
|
|
|
|
|
}
|
|
|
|
@ -205,20 +287,7 @@ void ue::clear_old_buffers(uint32_t tti)
|
|
|
|
|
|
|
|
|
|
// remove old buffers
|
|
|
|
|
for (auto& cc : cc_buffers) {
|
|
|
|
|
auto& rx_buffer_cc = cc.get_rx_used_buffers();
|
|
|
|
|
for (auto it = rx_buffer_cc.begin(); it != rx_buffer_cc.end();) {
|
|
|
|
|
if (srslte_tti_interval(tti, it->first) > 20 && srslte_tti_interval(tti, it->first) < 500) {
|
|
|
|
|
logger.warning("UE buffers: Removing old buffer tti=%d, rnti=%d, now is %d, interval=%d",
|
|
|
|
|
it->first,
|
|
|
|
|
rnti,
|
|
|
|
|
tti,
|
|
|
|
|
srslte_tti_interval(tti, it->first));
|
|
|
|
|
pdus.deallocate(it->second);
|
|
|
|
|
it = rx_buffer_cc.erase(it);
|
|
|
|
|
} else {
|
|
|
|
|
++it;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
cc.get_rx_used_buffers().clear_old_pdus(tti_point{tti});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -353,11 +422,7 @@ void ue::process_pdu(uint8_t* pdu, uint32_t nof_bytes, srslte::pdu_queue::channe
|
|
|
|
|
void ue::deallocate_pdu(uint32_t tti, uint32_t ue_cc_idx)
|
|
|
|
|
{
|
|
|
|
|
std::unique_lock<std::mutex> lock(rx_buffers_mutex);
|
|
|
|
|
|
|
|
|
|
if (cc_buffers.at(ue_cc_idx).get_rx_used_buffers().count(tti)) {
|
|
|
|
|
pdus.deallocate(cc_buffers.at(ue_cc_idx).get_rx_used_buffers().at(tti));
|
|
|
|
|
cc_buffers.at(ue_cc_idx).get_rx_used_buffers().erase(tti);
|
|
|
|
|
} else {
|
|
|
|
|
if (not cc_buffers.at(ue_cc_idx).get_rx_used_buffers().try_deallocate_pdu(tti_point(tti))) {
|
|
|
|
|
logger.warning("UE buffers: Null RX PDU pointer in deallocate_pdu for rnti=0x%x pid=%d cc_idx=%d",
|
|
|
|
|
rnti,
|
|
|
|
|
tti % nof_rx_harq_proc,
|
|
|
|
@ -368,18 +433,9 @@ void ue::deallocate_pdu(uint32_t tti, uint32_t ue_cc_idx)
|
|
|
|
|
void ue::push_pdu(uint32_t tti, uint32_t ue_cc_idx, uint32_t len)
|
|
|
|
|
{
|
|
|
|
|
std::unique_lock<std::mutex> lock(rx_buffers_mutex);
|
|
|
|
|
if (cc_buffers.at(ue_cc_idx).get_rx_used_buffers().count(tti)) {
|
|
|
|
|
if (len > 0) {
|
|
|
|
|
pdus.push(cc_buffers.at(ue_cc_idx).get_rx_used_buffers().at(tti), len);
|
|
|
|
|
} else {
|
|
|
|
|
logger.error("Error pushing PDU: null length");
|
|
|
|
|
}
|
|
|
|
|
cc_buffers.at(ue_cc_idx).get_rx_used_buffers().erase(tti);
|
|
|
|
|
} else {
|
|
|
|
|
logger.warning("UE buffers: Null RX PDU pointer in push_pdu for rnti=0x%x pid=%d cc_idx=%d",
|
|
|
|
|
rnti,
|
|
|
|
|
tti % nof_rx_harq_proc,
|
|
|
|
|
ue_cc_idx);
|
|
|
|
|
if (not cc_buffers.at(ue_cc_idx).get_rx_used_buffers().push_pdu(tti_point(tti), len)) {
|
|
|
|
|
logger.warning(
|
|
|
|
|
"UE buffers: Failed to push RX PDU for rnti=0x%x pid=%d cc_idx=%d", rnti, tti % nof_rx_harq_proc, ue_cc_idx);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|