From 96815e3a5f0ebb7e6f8be90e4c1d4addbd9e06ce Mon Sep 17 00:00:00 2001 From: Andre Puschmann Date: Tue, 2 Oct 2018 17:00:34 +0200 Subject: [PATCH] fix RLC AM handling of bigger MAC grants - fix concatenation and header reconstruction --- lib/include/srslte/upper/rlc_am.h | 6 +- lib/src/upper/rlc_am.cc | 171 +++++++++++++++++++++--------- 2 files changed, 125 insertions(+), 52 deletions(-) diff --git a/lib/include/srslte/upper/rlc_am.h b/lib/include/srslte/upper/rlc_am.h index ac04c991a..0e18a7e7f 100644 --- a/lib/include/srslte/upper/rlc_am.h +++ b/lib/include/srslte/upper/rlc_am.h @@ -225,7 +225,8 @@ private: void timer_expired(uint32_t timeout_id); // Functions needed by Tx subclass to query rx state - int get_status(rlc_status_pdu_t* status); + int get_status_pdu_length(); + int get_status_pdu(rlc_status_pdu_t* status, const uint32_t nof_bytes); bool get_do_status(); void reset_status(); // called when status PDU has been sent @@ -320,7 +321,8 @@ uint32_t rlc_am_packed_length(rlc_amd_retx_t retx); bool rlc_am_is_control_pdu(byte_buffer_t *pdu); bool rlc_am_is_control_pdu(uint8_t *payload); bool rlc_am_is_pdu_segment(uint8_t *payload); -std::string rlc_am_to_string(rlc_status_pdu_t *status); +std::string rlc_am_status_pdu_to_string(rlc_status_pdu_t *status); +std::string rlc_amd_pdu_header_to_string(const rlc_amd_pdu_header_t &header); bool rlc_am_start_aligned(const uint8_t fi); bool rlc_am_end_aligned(const uint8_t fi); bool rlc_am_is_unaligned(const uint8_t fi); diff --git a/lib/src/upper/rlc_am.cc b/lib/src/upper/rlc_am.cc index 2b032ac9f..7c2e18b0f 100644 --- a/lib/src/upper/rlc_am.cc +++ b/lib/src/upper/rlc_am.cc @@ -322,7 +322,7 @@ uint32_t rlc_am::rlc_am_tx::get_buffer_state() // Bytes needed for status report if (do_status() && not status_prohibited) { - n_bytes = parent->rx.get_status(&tx_status); + n_bytes = parent->rx.get_status_pdu_length(); log->debug("%s Buffer state - status report: %d bytes\n", RB_NAME, n_bytes); goto unlock_and_return; } @@ -378,7 +378,7 @@ uint32_t rlc_am::rlc_am_tx::get_total_buffer_state() // Bytes needed for status report if(do_status() && not status_prohibited) { - n_bytes += parent->rx.get_status(&tx_status); + n_bytes += parent->rx.get_status_pdu_length(); log->debug("%s Buffer state - total status report: %d bytes\n", RB_NAME, n_bytes); } @@ -582,10 +582,11 @@ bool rlc_am::rlc_am_tx::poll_required() int rlc_am::rlc_am_tx::build_status_pdu(uint8_t *payload, uint32_t nof_bytes) { - int pdu_len = parent->rx.get_status(&tx_status); + int pdu_len = parent->rx.get_status_pdu(&tx_status, nof_bytes); + log->debug("%s\n", rlc_am_status_pdu_to_string(&tx_status).c_str()); if (pdu_len > 0 && nof_bytes >= static_cast(pdu_len)) { log->info("%s Tx status PDU - %s\n", - RB_NAME, rlc_am_to_string(&tx_status).c_str()); + RB_NAME, rlc_am_status_pdu_to_string(&tx_status).c_str()); parent->rx.reset_status(); @@ -760,6 +761,10 @@ int rlc_am::rlc_am_tx::build_segment(uint8_t *payload, uint32_t nof_bytes, rlc_a upper += old_header.li[i]; head_len = rlc_am_packed_length(&new_header); + + // Accomodate some extra space for for LIs if old header contained segments too + head_len += old_header.N_li; + pdu_space = nof_bytes-head_len; if(pdu_space < (retx.so_end-retx.so_start)) { retx.so_end = retx.so_start + pdu_space; @@ -779,19 +784,17 @@ int rlc_am::rlc_am_tx::build_segment(uint8_t *payload, uint32_t nof_bytes, rlc_a if (upper == retx.so_end) { new_header.fi &= RLC_FI_FIELD_NOT_START_ALIGNED; // segment end is aligned with this SDU } - new_header.li[new_header.N_li++] = li; + new_header.li[new_header.N_li] = li; + + // only increment N_li if more SDU (segments) are being added + if (retx.so_end > upper) { + new_header.N_li++; + } } lower += old_header.li[i]; } - // Make sure LI is not deleted in case the SDU boundary is crossed - // FIXME: fix if N_li > 1 - if (new_header.N_li == 1 && retx.so_start + new_header.li[0] < retx.so_end && retx.so_end <= retx.so_start + pdu_space) { - // This segment crosses a SDU boundary - new_header.N_li++; - } - // Update retx_queue if(tx_window[retx.sn].buf->N_bytes == retx.so_end) { retx_queue.pop_front(); @@ -804,9 +807,6 @@ int rlc_am::rlc_am_tx::build_segment(uint8_t *payload, uint32_t nof_bytes, rlc_a } else { retx_queue.front().is_segment = true; retx_queue.front().so_start = retx.so_end; - if (new_header.N_li > 0) { - new_header.N_li--; - } } // Write header and pdu @@ -816,9 +816,6 @@ int rlc_am::rlc_am_tx::build_segment(uint8_t *payload, uint32_t nof_bytes, rlc_a uint32_t len = retx.so_end - retx.so_start; memcpy(ptr, data, len); - log->info("%s Retx PDU segment scheduled for tx. SN: %d, SO: %d\n", - RB_NAME, retx.sn, retx.so_start); - debug_state(); int pdu_len = (ptr-payload) + len; if (pdu_len > static_cast(nof_bytes)) { @@ -827,6 +824,10 @@ int rlc_am::rlc_am_tx::build_segment(uint8_t *payload, uint32_t nof_bytes, rlc_a log->debug("%s Retx PDU segment length error. Header len: %ld, Payload len: %d, N_li: %d\n", RB_NAME, (ptr-payload), len, new_header.N_li); } + + log->info_hex(payload, pdu_len, "%s Retx PDU segment of SN=%d (%d B), SO: %d, N_li: %d\n", + RB_NAME, retx.sn, pdu_len, retx.so_start, new_header.N_li); + return pdu_len; } @@ -921,7 +922,8 @@ int rlc_am::rlc_am_tx::build_data_pdu(uint8_t *payload, uint32_t nof_bytes) // Pull SDUs from queue while (pdu_space > head_len + 1 && tx_sdu_queue.size() > 0) { if (last_li > 0) { - header.li[header.N_li++] = last_li; + header.li[header.N_li] = last_li; + header.N_li++; } head_len = rlc_am_packed_length(&header); if (head_len >= pdu_space) { @@ -993,10 +995,11 @@ int rlc_am::rlc_am_tx::build_data_pdu(uint8_t *payload, uint32_t nof_bytes) uint8_t *ptr = payload; rlc_am_write_data_pdu_header(&header, &ptr); memcpy(ptr, pdu->msg, pdu->N_bytes); - log->info_hex(payload, pdu->N_bytes, "%s PDU scheduled for tx. SN: %d (%d B)\n", RB_NAME, header.sn, pdu->N_bytes); - + int total_len = (ptr-payload) + pdu->N_bytes; + log->info_hex(payload, total_len, "%s Tx PDU SN=%d (%d B)\n", RB_NAME, header.sn, total_len); + log->debug("%s\n", rlc_amd_pdu_header_to_string(header).c_str()); debug_state(); - return (ptr-payload) + pdu->N_bytes; + return total_len; } void rlc_am::rlc_am_tx::handle_control_pdu(uint8_t *payload, uint32_t nof_bytes) @@ -1008,10 +1011,10 @@ void rlc_am::rlc_am_tx::handle_control_pdu(uint8_t *payload, uint32_t nof_bytes) rlc_status_pdu_t status; rlc_am_read_status_pdu(payload, nof_bytes, &status); - log->info("%s Rx Status PDU: %s\n", RB_NAME, rlc_am_to_string(&status).c_str()); + log->info("%s Rx Status PDU: %s\n", RB_NAME, rlc_am_status_pdu_to_string(&status).c_str()); if (poll_retx_timer != NULL) { - poll_retx_timer->reset(); + poll_retx_timer->stop(); } // flush retx queue to avoid unordered SNs, we expect the Rx to request lost PDUs again @@ -1305,11 +1308,11 @@ void rlc_am::rlc_am_rx::handle_data_pdu(uint8_t *payload, uint32_t nof_bytes, rl { std::map::iterator it; - log->info_hex(payload, nof_bytes, "%s Rx data PDU SN: %d (%d B), %s", + log->info_hex(payload, nof_bytes, "%s Rx data PDU SN=%d (%d B)", RB_NAME, header.sn, - nof_bytes, - rlc_fi_field_text[header.fi]); + nof_bytes); + log->debug("%s\n", rlc_amd_pdu_header_to_string(header).c_str()); if(!inside_rx_window(header.sn)) { if(header.p) { @@ -1376,9 +1379,7 @@ void rlc_am::rlc_am_rx::handle_data_pdu(uint8_t *payload, uint32_t nof_bytes, rl poll_received = true; // 36.322 v10 Section 5.2.3 - if(RX_MOD_BASE(header.sn) < RX_MOD_BASE(vr_ms) || - RX_MOD_BASE(header.sn) >= RX_MOD_BASE(vr_mr)) - { + if (RX_MOD_BASE(header.sn) < RX_MOD_BASE(vr_ms) || RX_MOD_BASE(header.sn) >= RX_MOD_BASE(vr_mr)) { do_status = true; } // else delay for reordering timer @@ -1391,7 +1392,7 @@ void rlc_am::rlc_am_rx::handle_data_pdu(uint8_t *payload, uint32_t nof_bytes, rl if (reordering_timer != NULL) { if (reordering_timer->is_running()) { if(vr_x == vr_r || (!inside_rx_window(vr_x) && vr_x != vr_mr)) { - reordering_timer->reset(); + reordering_timer->stop(); } } @@ -1412,8 +1413,9 @@ void rlc_am::rlc_am_rx::handle_data_pdu_segment(uint8_t *payload, uint32_t nof_b { std::map::iterator it; - log->info_hex(payload, nof_bytes, "%s Rx data PDU segment. SN: %d, SO: %d", - RB_NAME, header.sn, header.so); + log->info_hex(payload, nof_bytes, "%s Rx data PDU segment of SN=%d (%d B), SO=%d, N_li=%d", + RB_NAME, header.sn, nof_bytes, header.so, header.N_li); + log->debug("%s\n", rlc_amd_pdu_header_to_string(header).c_str()); // Check inside rx window if(!inside_rx_window(header.sn)) { @@ -1444,9 +1446,9 @@ void rlc_am::rlc_am_rx::handle_data_pdu_segment(uint8_t *payload, uint32_t nof_b // Check if we already have a segment from the same PDU it = rx_segments.find(header.sn); - if(rx_segments.end() != it) { + if (rx_segments.end() != it) { - if(header.p) { + if (header.p) { log->info("%s Status packet requested through polling bit\n", RB_NAME); do_status = true; } @@ -1518,6 +1520,9 @@ void rlc_am::rlc_am_rx::reassemble_rx_sdus() for(uint32_t i=0; idebug_hex(rx_window[vr_r].buf->msg, len, "Handling segment %d/%d of length %d B of SN=%d\n", i+1, rx_window[vr_r].header.N_li, len, vr_r); + // sanity check to avoid zero-size SDUs if (len == 0) { break; @@ -1557,11 +1562,12 @@ void rlc_am::rlc_am_rx::reassemble_rx_sdus() // Handle last segment len = rx_window[vr_r].buf->N_bytes; + log->debug_hex(rx_window[vr_r].buf->msg, len, "Handling last segment of length %d B of SN=%d\n", len, vr_r); if (rx_sdu->get_tailroom() >= len) { memcpy(&rx_sdu->msg[rx_sdu->N_bytes], rx_window[vr_r].buf->msg, len); rx_sdu->N_bytes += rx_window[vr_r].buf->N_bytes; } else { - log->error("Cannot fit RLC PDU in SDU buffer, dropping both.\n"); + log->error("Cannot fit RLC PDU in SDU buffer, dropping both. Erasing SN=%d.\n", vr_r); pool->deallocate(rx_sdu); pool->deallocate(rx_window[vr_r].buf); rx_window.erase(vr_r); @@ -1585,6 +1591,19 @@ void rlc_am::rlc_am_rx::reassemble_rx_sdus() exit: // Move the rx_window + log->debug("Erasing SN=%d.\n", vr_r); + // also erase any segments of this SN + std::map::iterator it; + it = rx_segments.find(vr_r); + if(rx_segments.end() != it) { + log->debug("Erasing segments of SN=%d\n", vr_r); + std::list::iterator segit; + for(segit = it->second.segments.begin(); segit != it->second.segments.end(); ++segit) { + log->debug(" Erasing segment of SN=%d SO=%d Len=%d N_li=%d\n", segit->header.sn, segit->header.so, segit->buf->N_bytes, segit->header.N_li); + pool->deallocate(segit->buf); + } + it->second.segments.clear(); + } pool->deallocate(rx_window[vr_r].buf); rx_window.erase(vr_r); vr_r = (vr_r + 1)%MOD; @@ -1670,8 +1689,8 @@ void rlc_am::rlc_am_rx::timer_expired(uint32_t timeout_id) pthread_mutex_unlock(&mutex); } -// Called from Tx object (packs status PDU and returns length of it) -int rlc_am::rlc_am_rx::get_status(rlc_status_pdu_t* status) +// Called from Tx object to pack status PDU that doesn't exceed a given size +int rlc_am::rlc_am_rx::get_status_pdu(rlc_status_pdu_t* status, const uint32_t max_pdu_size) { pthread_mutex_lock(&mutex); status->N_nack = 0; @@ -1679,7 +1698,8 @@ int rlc_am::rlc_am_rx::get_status(rlc_status_pdu_t* status) // We don't use segment NACKs - just NACK the full PDU uint32_t i = vr_r; - while (RX_MOD_BASE(i) < RX_MOD_BASE(vr_ms) && status->N_nack < RLC_AM_WINDOW_SIZE) { + while (RX_MOD_BASE(i) < RX_MOD_BASE(vr_ms) && status->N_nack < RLC_AM_WINDOW_SIZE && rlc_am_packed_length(status) <= max_pdu_size-2) { + status->ack_sn = i; if(rx_window.find(i) == rx_window.end()) { status->nacks[status->N_nack].nack_sn = i; status->N_nack++; @@ -1690,6 +1710,22 @@ int rlc_am::rlc_am_rx::get_status(rlc_status_pdu_t* status) return rlc_am_packed_length(status); } +// Called from Tx object to obtain length of the full status PDU +int rlc_am::rlc_am_rx::get_status_pdu_length() +{ + pthread_mutex_lock(&mutex); + rlc_status_pdu_t status; + uint32_t i = vr_r; + while (RX_MOD_BASE(i) < RX_MOD_BASE(vr_ms) && status.N_nack < RLC_AM_WINDOW_SIZE) { + if(rx_window.find(i) == rx_window.end()) { + status.N_nack++; + } + i = (i + 1)%MOD; + } + pthread_mutex_unlock(&mutex); + return rlc_am_packed_length(&status); +} + void rlc_am::rlc_am_rx::print_rx_segments() { std::map::iterator it; @@ -1758,33 +1794,46 @@ bool rlc_am::rlc_am_rx::add_segment_and_check(rlc_amd_rx_pdu_segments_t *pdu, rl header.fi |= (pdu->segments.front().header.fi & RLC_FI_FIELD_NOT_START_ALIGNED); header.fi |= (pdu->segments.back().header.fi & RLC_FI_FIELD_NOT_END_ALIGNED); + log->debug("Starting header reconstruction of %zd segments\n", pdu->segments.size()); + // Reconstruct li fields uint16_t count = 0; uint16_t carryover = 0; for(it = pdu->segments.begin(); it != pdu->segments.end(); it++) { - if(it->header.N_li > 0) { - header.li[header.N_li++] = it->header.li[0] + carryover; - count += it->header.li[0]; - for(uint32_t i=1; iheader.N_li; i++) { - header.li[header.N_li++] = it->header.li[i]; - count += it->header.li[i]; + log->debug(" Handling %d PDU segments\n", it->header.N_li); + for(uint32_t i=0; iheader.N_li; i++) { + header.li[header.N_li] = it->header.li[i]; + if (i == 0) { + header.li[header.N_li] += carryover; } + log->debug(" - adding segment %d/%d (%d B, SO=%d, carryover=%d, count=%d)\n", i+1, it->header.N_li, header.li[header.N_li], header.so, carryover, count); + header.N_li++; + count += it->header.li[i]; + carryover = 0; } - // accumulate segment sizes until end aligned PDU is received - if (rlc_am_not_start_aligned(it->header.fi)) { + if (count <= it->buf->N_bytes) { carryover += it->buf->N_bytes - count; + log->debug("Incremented carryover (it->buf->N_bytes=%d, count=%d). New carryover=%d\n", it->buf->N_bytes, count, carryover); } else { - carryover = it->buf->N_bytes - count; + // Next segment would be too long, recalculate carryover + header.N_li--; + carryover = it->buf->N_bytes - (count - header.li[header.N_li]); + log->debug("Recalculated carryover=%d (it->buf->N_bytes=%d, count=%d, header.li[header.N_li]=%d)\n", carryover, it->buf->N_bytes, count, header.li[header.N_li]); } + tmpit = it; if(rlc_am_end_aligned(it->header.fi) && ++tmpit != pdu->segments.end()) { - header.li[header.N_li++] = carryover; + log->debug("Header is end-aligned, overwrite header.li[%d]=%d\n", header.N_li, carryover); + header.li[header.N_li] = carryover; + header.N_li++; carryover = 0; } count = 0; } + log->debug("Finished header reconstruction of %zd segments\n", pdu->segments.size()); + // Copy data byte_buffer_t *full_pdu = pool_allocate_blocking; if (full_pdu == NULL) { @@ -2091,7 +2140,7 @@ bool rlc_am_is_pdu_segment(uint8_t *payload) return ((*(payload) >> 6) & 0x01) == 1; } -std::string rlc_am_to_string(rlc_status_pdu_t *status) +std::string rlc_am_status_pdu_to_string(rlc_status_pdu_t *status) { std::stringstream ss; ss << "ACK_SN = " << status->ack_sn; @@ -2112,6 +2161,28 @@ std::string rlc_am_to_string(rlc_status_pdu_t *status) return ss.str(); } +std::string rlc_amd_pdu_header_to_string(const rlc_amd_pdu_header_t &header) +{ + std::stringstream ss; + ss << "[" << rlc_dc_field_text[header.dc]; + ss << ", RF=" << (header.rf ? "1":"0"); + ss << ", P=" << (header.p ? "1":"0"); + ss << ", FI=" << (header.fi ? "1":"0"); + ss << ", SN=" << header.sn; + ss << ", LSF=" << (header.lsf ? "1":"0"); + ss << ", SO=" << header.so; + ss << ", N_li=" << header.N_li; + if (header.N_li > 0) { + ss << " ("; + for (uint32_t i = 0; i < header.N_li; i++) { + ss << header.li[i] << ", "; + } + ss << ")"; + } + ss << "]"; + return ss.str(); +} + bool rlc_am_start_aligned(const uint8_t fi) { return (fi == RLC_FI_FIELD_START_AND_END_ALIGNED || fi == RLC_FI_FIELD_NOT_END_ALIGNED);