rlc_am_nr: fix buffer status calculation for retxs

master
Pedro Alvarez 3 years ago
parent 63657ddfe6
commit 473a45aae0

@ -86,6 +86,7 @@ private:
using rlc_amd_tx_pdu_nr = rlc_amd_tx_pdu<rlc_am_nr_pdu_header_t>; using rlc_amd_tx_pdu_nr = rlc_amd_tx_pdu<rlc_am_nr_pdu_header_t>;
rlc_ringbuffer_t<rlc_amd_tx_pdu_nr, RLC_AM_WINDOW_SIZE> tx_window; rlc_ringbuffer_t<rlc_amd_tx_pdu_nr, RLC_AM_WINDOW_SIZE> tx_window;
pdu_retx_queue<RLC_AM_WINDOW_SIZE> retx_queue;
}; };
// Receiver sub-class // Receiver sub-class

@ -76,7 +76,7 @@ uint32_t rlc_am_nr_tx::read_pdu(uint8_t* payload, uint32_t nof_bytes)
// logger.debug("tx_window size - %zu PDUs", tx_window.size()); // logger.debug("tx_window size - %zu PDUs", tx_window.size());
// Tx STATUS if requested // Tx STATUS if requested
if (do_status() /*&& not status_prohibit_timer.is_running()*/) { if (do_status()) {
unique_byte_buffer_t tx_pdu = srsran::make_byte_buffer(); unique_byte_buffer_t tx_pdu = srsran::make_byte_buffer();
build_status_pdu(tx_pdu.get(), nof_bytes); build_status_pdu(tx_pdu.get(), nof_bytes);
memcpy(payload, tx_pdu->msg, tx_pdu->N_bytes); memcpy(payload, tx_pdu->msg, tx_pdu->N_bytes);
@ -131,7 +131,13 @@ uint32_t rlc_am_nr_tx::read_pdu(uint8_t* payload, uint32_t nof_bytes)
hdr.sn = st.tx_next; hdr.sn = st.tx_next;
log_rlc_am_nr_pdu_header_to_string(logger->info, hdr); log_rlc_am_nr_pdu_header_to_string(logger->info, hdr);
uint32_t len = rlc_am_nr_write_data_pdu_header(hdr, tx_sdu.get()); // insert newly assigned SN into window and use reference for in-place operations
// NOTE: from now on, we can't return from this function anymore before increasing vt_s
rlc_amd_tx_pdu_nr& tx_pdu = tx_window.add_pdu(hdr.sn);
tx_pdu.buf = srsran::make_byte_buffer();
memcpy(tx_pdu.buf->msg, tx_sdu->msg, tx_sdu->N_bytes);
tx_pdu.buf->N_bytes = tx_sdu->N_bytes;
uint32_t len = rlc_am_nr_write_data_pdu_header(hdr, tx_pdu.buf.get());
if (len > nof_bytes) { if (len > nof_bytes) {
logger->error("Error writing AMD PDU header"); logger->error("Error writing AMD PDU header");
} }
@ -139,10 +145,10 @@ uint32_t rlc_am_nr_tx::read_pdu(uint8_t* payload, uint32_t nof_bytes)
// Update TX Next // Update TX Next
st.tx_next = (st.tx_next + 1) % MOD; st.tx_next = (st.tx_next + 1) % MOD;
memcpy(payload, tx_sdu->msg, tx_sdu->N_bytes); memcpy(payload, tx_pdu.buf->msg, tx_pdu.buf->N_bytes);
logger->debug("Wrote RLC PDU - %d bytes", tx_sdu->N_bytes); logger->debug("Wrote RLC PDU - %d bytes", tx_sdu->N_bytes);
return tx_sdu->N_bytes; return tx_pdu.buf->N_bytes;
} }
uint32_t rlc_am_nr_tx::build_status_pdu(byte_buffer_t* payload, uint32_t nof_bytes) uint32_t rlc_am_nr_tx::build_status_pdu(byte_buffer_t* payload, uint32_t nof_bytes)
@ -176,8 +182,6 @@ void rlc_am_nr_tx::handle_control_pdu(uint8_t* payload, uint32_t nof_bytes)
rlc_am_nr_read_status_pdu(payload, nof_bytes, rlc_am_nr_sn_size_t::size12bits, &status); rlc_am_nr_read_status_pdu(payload, nof_bytes, rlc_am_nr_sn_size_t::size12bits, &status);
log_rlc_am_nr_status_pdu_to_string(logger->info, "%s Rx Status PDU: %s", &status, parent->rb_name); log_rlc_am_nr_status_pdu_to_string(logger->info, "%s Rx Status PDU: %s", &status, parent->rb_name);
// Local variables for handling Status PDU will be updated with lock // Local variables for handling Status PDU will be updated with lock
uint32_t ack_sn = status.ack_sn;
uint32_t n_nacks = status.N_nack;
/* /*
* - if the SN of the corresponding RLC SDU falls within the range * - if the SN of the corresponding RLC SDU falls within the range
* TX_Next_Ack <= SN < = the highest SN of the AMD PDU among the AMD PDUs submitted to lower layer: * TX_Next_Ack <= SN < = the highest SN of the AMD PDU among the AMD PDUs submitted to lower layer:
@ -185,9 +189,8 @@ void rlc_am_nr_tx::handle_control_pdu(uint8_t* payload, uint32_t nof_bytes)
* retransmission. * retransmission.
*/ */
// Process N_acks // Process N_acks
/*
for (uint32_t nack_idx = 0; nack_idx < status.N_nack; nack_idx++) { for (uint32_t nack_idx = 0; nack_idx < status.N_nack; nack_idx++) {
if (st.tx_next_ack <= status.nacks[nack_idx].nack_sn && status.nacks[nack_idx].nack_sn <= tx_window.end()) { if (st.tx_next_ack <= status.nacks[nack_idx].nack_sn && status.nacks[nack_idx].nack_sn <= st.tx_next) {
uint32_t nack_sn = status.nacks[nack_idx].nack_sn; uint32_t nack_sn = status.nacks[nack_idx].nack_sn;
if (tx_window.has_sn(nack_sn)) { if (tx_window.has_sn(nack_sn)) {
auto& pdu = tx_window[nack_sn]; auto& pdu = tx_window[nack_sn];
@ -196,7 +199,7 @@ void rlc_am_nr_tx::handle_control_pdu(uint8_t* payload, uint32_t nof_bytes)
if (not retx_queue.has_sn(nack_sn)) { if (not retx_queue.has_sn(nack_sn)) {
// increment Retx counter and inform upper layers if needed // increment Retx counter and inform upper layers if needed
pdu.retx_count++; pdu.retx_count++;
check_sn_reached_max_retx(i); // check_sn_reached_max_retx(nack_sn);
rlc_amd_retx_t& retx = retx_queue.push(); rlc_amd_retx_t& retx = retx_queue.push();
srsran_expect(tx_window[nack_sn].rlc_sn == nack_sn, srsran_expect(tx_window[nack_sn].rlc_sn == nack_sn,
@ -210,7 +213,7 @@ void rlc_am_nr_tx::handle_control_pdu(uint8_t* payload, uint32_t nof_bytes)
} }
} }
} }
}*/ }
/** /**
* Section 5.3.3.3: Reception of a STATUS report * Section 5.3.3.3: Reception of a STATUS report
@ -234,14 +237,7 @@ void rlc_am_nr_tx::get_buffer_state(uint32_t& n_bytes_new, uint32_t& n_bytes_pri
logger->debug("Buffer state requested, %s", rb_name); logger->debug("Buffer state requested, %s", rb_name);
std::lock_guard<std::mutex> lock(mutex); std::lock_guard<std::mutex> lock(mutex);
/* logger->debug("%s Buffer state - do_status=%s", rb_name, do_status() ? "yes" : "no");
logger.debug("%s Buffer state - do_status=%s, status_prohibit_running=%s (%d/%d)",
rb_name,
do_status() ? "yes" : "no",
status_prohibit_timer.is_running() ? "yes" : "no",
status_prohibit_timer.time_elapsed(),
status_prohibit_timer.duration());
*/
// Bytes needed for status report // Bytes needed for status report
if (do_status()) { if (do_status()) {
@ -250,7 +246,25 @@ void rlc_am_nr_tx::get_buffer_state(uint32_t& n_bytes_new, uint32_t& n_bytes_pri
} }
// Bytes needed for retx // Bytes needed for retx
// TODO if (not retx_queue.empty()) {
rlc_amd_retx_t& retx = retx_queue.front();
logger->debug("%s Buffer state - retx - SN=%d, Segment: %s, %d:%d",
parent->rb_name,
retx.sn,
retx.is_segment ? "true" : "false",
retx.so_start,
retx.so_end);
if (tx_window.has_sn(retx.sn)) {
int req_bytes = retx.so_end - retx.so_start;
if (req_bytes <= 0) {
logger->error("In get_buffer_state(): Removing retx.sn=%d from queue", retx.sn);
retx_queue.pop();
} else {
n_bytes_prio += req_bytes;
logger->debug("Buffer state - retx: %d bytes", n_bytes_prio);
}
}
}
// Bytes needed for tx SDUs // Bytes needed for tx SDUs
uint32_t n_sdus = tx_sdu_queue.get_n_sdus(); uint32_t n_sdus = tx_sdu_queue.get_n_sdus();

@ -193,7 +193,7 @@ int lost_pdu_test()
rlc1.write_pdu(status_buf.msg, status_buf.N_bytes); rlc1.write_pdu(status_buf.msg, status_buf.N_bytes);
// Check there is an Retx of SN=3 // Check there is an Retx of SN=3
TESTASSERT(0 == rlc1.get_buffer_state()); TESTASSERT(3 == rlc1.get_buffer_state());
} }
// Check statistics // Check statistics

Loading…
Cancel
Save