rlc, nr: avoid multi increments of SDU's retx_count within one status message

master
Robert Falkenberg 3 years ago
parent 511ad9ed25
commit 0fb6420e8b

@ -303,8 +303,8 @@ private:
struct rlc_amd_retx_t { struct rlc_amd_retx_t {
uint32_t sn; uint32_t sn;
bool is_segment; bool is_segment;
uint32_t so_start; uint32_t so_start; // offset to first byte of this segment
uint32_t so_end; uint32_t so_end; // offset to first byte beyond the end of this segment
uint32_t current_so; uint32_t current_so;
}; };

@ -18,6 +18,7 @@
#include "srsran/rlc/rlc_am_nr_packing.h" #include "srsran/rlc/rlc_am_nr_packing.h"
#include "srsran/srslog/event_trace.h" #include "srsran/srslog/event_trace.h"
#include <iostream> #include <iostream>
#include <set>
#define RLC_AM_NR_WINDOW_SIZE 2048 #define RLC_AM_NR_WINDOW_SIZE 2048
@ -721,7 +722,8 @@ void rlc_am_nr_tx::handle_control_pdu(uint8_t* payload, uint32_t nof_bytes)
} }
notify_info_vec.clear(); notify_info_vec.clear();
// Process N_acks // Process N_nacks
std::set<uint32_t> retx_sn_set; // Set of PDU SNs added for retransmission (no duplicates)
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++) {
// TODO: Possibly loop NACK range // TODO: Possibly loop NACK range
if (st.tx_next_ack <= status.nacks[nack_idx].nack_sn && status.nacks[nack_idx].nack_sn <= st.tx_next) { if (st.tx_next_ack <= status.nacks[nack_idx].nack_sn && status.nacks[nack_idx].nack_sn <= st.tx_next) {
@ -729,7 +731,6 @@ void rlc_am_nr_tx::handle_control_pdu(uint8_t* payload, uint32_t nof_bytes)
uint32_t nack_sn = nack.nack_sn; uint32_t nack_sn = nack.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];
bool retx_enqueued = false;
if (nack.has_so) { if (nack.has_so) {
// NACK'ing missing bytes in SDU segment. // NACK'ing missing bytes in SDU segment.
@ -748,7 +749,9 @@ void rlc_am_nr_tx::handle_control_pdu(uint8_t* payload, uint32_t nof_bytes)
retx.so_start = segm->so; retx.so_start = segm->so;
retx.current_so = segm->so; retx.current_so = segm->so;
retx.so_end = segm->so + segm->payload_len; retx.so_end = segm->so + segm->payload_len;
retx_enqueued = true; retx_sn_set.insert(nack_sn);
RlcInfo(
"Schedule PDU segment SN=%d, so_start=%d, so_end=%d for retx", retx.sn, retx.so_start, retx.so_end);
} }
} }
} else { } else {
@ -761,14 +764,20 @@ void rlc_am_nr_tx::handle_control_pdu(uint8_t* payload, uint32_t nof_bytes)
retx.so_start = 0; retx.so_start = 0;
retx.current_so = 0; retx.current_so = 0;
retx.so_end = pdu.sdu_buf->N_bytes; retx.so_end = pdu.sdu_buf->N_bytes;
retx_enqueued = true; retx_sn_set.insert(nack_sn);
RlcInfo("Schedule PDU SN=%d for retx", retx.sn);
} }
} }
} // TX window containts NACK SN
} // NACK SN within expected range
} // NACK loop
if (retx_enqueued) { // Process retx_count and inform upper layers if needed
for (auto retx_sn_it = retx_sn_set.begin(); retx_sn_it != retx_sn_set.end(); retx_sn_it++) {
auto& pdu = tx_window[*retx_sn_it];
// Increment retx_count // Increment retx_count
if (pdu.retx_count == RETX_COUNT_NOT_STARTED) { if (pdu.retx_count == RETX_COUNT_NOT_STARTED) {
// Set retx_count = 0 on first RE-transmission (38.322 Sec. 5.3.2) // Set retx_count = 0 on first RE-transmission of associated SDU (38.322 Sec. 5.3.2)
pdu.retx_count = 0; pdu.retx_count = 0;
} else { } else {
// Increment otherwise // Increment otherwise
@ -776,12 +785,7 @@ void rlc_am_nr_tx::handle_control_pdu(uint8_t* payload, uint32_t nof_bytes)
} }
// Inform upper layers if needed // Inform upper layers if needed
check_sn_reached_max_retx(nack_sn); check_sn_reached_max_retx(*retx_sn_it);
RlcInfo("Schedule SN=%d for retx", nack_sn);
}
}
}
} }
/** /**

@ -981,38 +981,75 @@ int max_retx_lost_segments_test()
rlc1.write_sdu(std::move(sdu_bufs[i])); rlc1.write_sdu(std::move(sdu_bufs[i]));
} }
// Read 4 PDUs from RLC1 (max 15 byte each) // Read 2*2=4 PDUs from RLC1 and limit to 15 byte to force segmentation in two parts:
// Segment 1: 2 byte header + 13 byte payload; space fully used
// Segment 2: 4 byte header + 7 byte payload; space not fully used, 4 bytes left over
const uint32_t n_pdus = 4; const uint32_t n_pdus = 4;
byte_buffer_t pdu_bufs[n_pdus]; byte_buffer_t pdu_bufs[n_pdus];
for (uint32_t i = 0; i < n_pdus; i++) { for (uint32_t i = 0; i < n_pdus; i++) {
len = rlc1.read_pdu(pdu_bufs[i].msg, 15); // 2 byte header + 13 byte payload len = rlc1.read_pdu(pdu_bufs[i].msg, 15);
pdu_bufs[i].N_bytes = len; pdu_bufs[i].N_bytes = len;
} }
TESTASSERT(0 == rlc1.get_buffer_state()); TESTASSERT(0 == rlc1.get_buffer_state());
// Fake status PDU that ack SN=1 and nack SN=0 // Fake status PDU that ack SN=1 and nack {SN=0 segment 0, SN=0 segment 1}
rlc_am_nr_status_pdu_t fake_status = {}; rlc_am_nr_status_pdu_t status_lost_both_segments = {};
fake_status.ack_sn = 2; // delivered up to SN=1 status_lost_both_segments.ack_sn = 2; // delivered up to SN=1
fake_status.N_nack = 1; // one SN was lost status_lost_both_segments.N_nack = 2; // two segments lost
fake_status.nacks[0].nack_sn = 0; // it was SN=0 that was lost status_lost_both_segments.nacks[0].nack_sn = 0; // it was SN=0 that was lost
status_lost_both_segments.nacks[0].has_so = true; // this NACKs a segment
status_lost_both_segments.nacks[0].so_start = 0; // segment starts at (and includes) byte 0
status_lost_both_segments.nacks[0].so_end = 12; // segment ends at (and includes) byte 12
status_lost_both_segments.nacks[1].nack_sn = 0; // it was SN=0 that was lost
status_lost_both_segments.nacks[1].has_so = true; // this NACKs a segment
status_lost_both_segments.nacks[1].so_start = 13; // segment starts at (and includes) byte 13
status_lost_both_segments.nacks[1].so_end = 19; // segment ends at (and includes) byte 19
// pack into PDU // pack into PDU
byte_buffer_t status_pdu; byte_buffer_t status_pdu_lost_both_segments;
rlc_am_nr_write_status_pdu(fake_status, rlc_cfg.am_nr.tx_sn_field_length, &status_pdu); rlc_am_nr_write_status_pdu(
status_lost_both_segments, rlc_cfg.am_nr.tx_sn_field_length, &status_pdu_lost_both_segments);
// Fake status PDU that ack SN=1 and nack {SN=0 segment 2}
rlc_am_nr_status_pdu_t status_lost_second_segment = {};
status_lost_second_segment.ack_sn = 2; // delivered up to SN=1
status_lost_second_segment.N_nack = 1; // one SN was lost
status_lost_second_segment.nacks[0].nack_sn = 0; // it was SN=0 that was lost
status_lost_second_segment.nacks[0].has_so = true; // this NACKs a segment
status_lost_second_segment.nacks[0].so_start = 13; // segment starts at (and includes) byte 13
status_lost_second_segment.nacks[0].so_end = 19; // segment ends at (and includes) byte 19
// pack into PDU
byte_buffer_t status_pdu_lost_second_segment;
rlc_am_nr_write_status_pdu(
status_lost_second_segment, rlc_cfg.am_nr.tx_sn_field_length, &status_pdu_lost_second_segment);
// Exceed the number of tolerated retransmissions by one additional retransmission // Exceed the number of tolerated retransmissions by one additional retransmission
// to trigger notification of the higher protocol layers. Note that the initial transmission // to trigger notification of the higher protocol layers. Note that the initial transmission
// (before starting retransmissions) does not count. See TS 38.322 Sec. 5.3.2 // (before starting retransmissions) does not count. See TS 38.322 Sec. 5.3.2
for (uint32_t retx_count = 0; retx_count < rlc_cfg.am_nr.max_retx_thresh + 1; ++retx_count) { for (uint32_t retx_count = 0; retx_count < rlc_cfg.am_nr.max_retx_thresh + 1; ++retx_count) {
byte_buffer_t pdu_buf;
// we've not yet reached max attempts // we've not yet reached max attempts
TESTASSERT(tester.max_retx_triggered == false); TESTASSERT(tester.max_retx_triggered == false);
// Write status PDU to RLC1 if (retx_count < rlc_cfg.am_nr.max_retx_thresh / 2) {
rlc1.write_pdu(status_pdu.msg, status_pdu.N_bytes); // Send NACK for segment 1 and segment 2
// Although two segments, this must count as one retransmission,
// because both segments NACK the same SDU in the same status message.
rlc1.write_pdu(status_pdu_lost_both_segments.msg, status_pdu_lost_both_segments.N_bytes);
byte_buffer_t pdu_buf; // read the retransmitted PDUs
len = rlc1.read_pdu(pdu_buf.msg, 3); // 2 byte header + 1 byte payload len = rlc1.read_pdu(pdu_buf.msg, 15); // 2 byte header + 13 byte payload
len = rlc1.read_pdu(pdu_buf.msg, 15); // 4 byte header + 7 byte payload
} else {
// Send NACK for segment 2 (assume at least segment 1 was finally received)
rlc1.write_pdu(status_pdu_lost_second_segment.msg, status_pdu_lost_second_segment.N_bytes);
// read the retransmitted PDUs
len = rlc1.read_pdu(pdu_buf.msg, 15); // 4 byte header + 7 byte payload
}
} }
// Now maxRetx should have been triggered // Now maxRetx should have been triggered

Loading…
Cancel
Save