|
|
|
/*
|
|
|
|
* Copyright 2013-2019 Software Radio Systems Limited
|
|
|
|
*
|
|
|
|
* This file is part of srsLTE.
|
|
|
|
*
|
|
|
|
* srsLTE is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License as
|
|
|
|
* published by the Free Software Foundation, either version 3 of
|
|
|
|
* the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* srsLTE is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* A copy of the GNU Affero General Public License can be found in
|
|
|
|
* the LICENSE file in the top-level directory of this distribution
|
|
|
|
* and at http://www.gnu.org/licenses/.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "srslte/upper/rlc_tm.h"
|
|
|
|
|
|
|
|
namespace srslte {
|
|
|
|
|
|
|
|
rlc_tm::rlc_tm(srslte::log* log_,
|
|
|
|
uint32_t lcid_,
|
|
|
|
srsue::pdcp_interface_rlc* pdcp_,
|
|
|
|
srsue::rrc_interface_rlc* rrc_,
|
|
|
|
srslte::timer_handler* timers_,
|
|
|
|
uint32_t queue_len_) :
|
|
|
|
ul_queue(queue_len_),
|
|
|
|
log(log_),
|
|
|
|
pdcp(pdcp_),
|
|
|
|
rrc(rrc_),
|
|
|
|
lcid(lcid_)
|
|
|
|
{
|
|
|
|
pool = byte_buffer_pool::get_instance();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Warning: must call stop() to properly deallocate all buffers
|
|
|
|
rlc_tm::~rlc_tm()
|
|
|
|
{
|
|
|
|
pool = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool rlc_tm::configure(rlc_config_t cnfg)
|
|
|
|
{
|
|
|
|
log->error("Attempted to configure TM RLC entity\n");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rlc_tm::empty_queue()
|
|
|
|
{
|
|
|
|
// Drop all messages in TX queue
|
|
|
|
unique_byte_buffer_t buf;
|
|
|
|
while (ul_queue.try_read(&buf)) {
|
|
|
|
}
|
|
|
|
ul_queue.reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
void rlc_tm::reestablish()
|
|
|
|
{
|
|
|
|
stop();
|
|
|
|
tx_enabled = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rlc_tm::stop()
|
|
|
|
{
|
|
|
|
tx_enabled = false;
|
|
|
|
empty_queue();
|
|
|
|
}
|
|
|
|
|
|
|
|
rlc_mode_t rlc_tm::get_mode()
|
|
|
|
{
|
|
|
|
return rlc_mode_t::tm;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t rlc_tm::get_bearer()
|
|
|
|
{
|
|
|
|
return lcid;
|
|
|
|
}
|
|
|
|
|
|
|
|
// PDCP interface
|
|
|
|
void rlc_tm::write_sdu(unique_byte_buffer_t sdu, bool blocking)
|
|
|
|
{
|
|
|
|
if (!tx_enabled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (sdu) {
|
|
|
|
if (blocking) {
|
|
|
|
log->info_hex(sdu->msg,
|
|
|
|
sdu->N_bytes,
|
|
|
|
"%s Tx SDU, queue size=%d, bytes=%d",
|
|
|
|
rrc->get_rb_name(lcid).c_str(),
|
|
|
|
ul_queue.size(),
|
|
|
|
ul_queue.size_bytes());
|
|
|
|
ul_queue.write(std::move(sdu));
|
|
|
|
} else {
|
|
|
|
uint8_t* msg_ptr = sdu->msg;
|
|
|
|
uint32_t nof_bytes = sdu->N_bytes;
|
|
|
|
std::pair<bool, unique_byte_buffer_t> ret = ul_queue.try_write(std::move(sdu));
|
|
|
|
if (ret.first) {
|
|
|
|
log->info_hex(msg_ptr,
|
|
|
|
nof_bytes,
|
|
|
|
"%s Tx SDU, queue size=%d, bytes=%d",
|
|
|
|
rrc->get_rb_name(lcid).c_str(),
|
|
|
|
ul_queue.size(),
|
|
|
|
ul_queue.size_bytes());
|
|
|
|
} else {
|
|
|
|
log->info_hex(ret.second->msg,
|
|
|
|
ret.second->N_bytes,
|
|
|
|
"[Dropped SDU] %s Tx SDU, queue size=%d, bytes=%d",
|
|
|
|
rrc->get_rb_name(lcid).c_str(),
|
|
|
|
ul_queue.size(),
|
|
|
|
ul_queue.size_bytes());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log->warning("NULL SDU pointer in write_sdu()\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void rlc_tm::discard_sdu(uint32_t discard_sn)
|
|
|
|
{
|
|
|
|
if (!tx_enabled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
log->warning("SDU discard not implemented on RLC TM\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
// MAC interface
|
|
|
|
bool rlc_tm::has_data()
|
|
|
|
{
|
|
|
|
return not ul_queue.is_empty();
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t rlc_tm::get_buffer_state()
|
|
|
|
{
|
|
|
|
return ul_queue.size_bytes();
|
|
|
|
}
|
|
|
|
|
|
|
|
rlc_bearer_metrics_t rlc_tm::get_metrics()
|
|
|
|
{
|
|
|
|
return metrics;
|
|
|
|
}
|
|
|
|
|
|
|
|
void rlc_tm::reset_metrics()
|
|
|
|
{
|
|
|
|
metrics = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
int rlc_tm::read_pdu(uint8_t* payload, uint32_t nof_bytes)
|
|
|
|
{
|
|
|
|
uint32_t pdu_size = ul_queue.size_tail_bytes();
|
|
|
|
if (pdu_size > nof_bytes) {
|
|
|
|
log->error(
|
|
|
|
"TX %s PDU size larger than MAC opportunity (%d > %d)\n", rrc->get_rb_name(lcid).c_str(), pdu_size, nof_bytes);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
unique_byte_buffer_t buf;
|
|
|
|
if (ul_queue.try_read(&buf)) {
|
|
|
|
pdu_size = buf->N_bytes;
|
|
|
|
memcpy(payload, buf->msg, buf->N_bytes);
|
|
|
|
log->debug("%s Complete SDU scheduled for tx. Stack latency: %ld us\n",
|
|
|
|
rrc->get_rb_name(lcid).c_str(),
|
|
|
|
buf->get_latency_us());
|
|
|
|
log->info_hex(payload,
|
|
|
|
pdu_size,
|
|
|
|
"TX %s, %s PDU, queue size=%d, bytes=%d",
|
|
|
|
rrc->get_rb_name(lcid).c_str(),
|
|
|
|
srslte::to_string(rlc_mode_t::tm).c_str(),
|
|
|
|
ul_queue.size(),
|
|
|
|
ul_queue.size_bytes());
|
|
|
|
|
|
|
|
metrics.num_tx_bytes += pdu_size;
|
|
|
|
return pdu_size;
|
|
|
|
} else {
|
|
|
|
log->warning("Queue empty while trying to read\n");
|
|
|
|
if (ul_queue.size_bytes() > 0) {
|
|
|
|
log->warning("Corrupted queue: empty but size_bytes > 0. Resetting queue\n");
|
|
|
|
ul_queue.reset();
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void rlc_tm::write_pdu(uint8_t* payload, uint32_t nof_bytes)
|
|
|
|
{
|
|
|
|
unique_byte_buffer_t buf = allocate_unique_buffer(*pool);
|
|
|
|
if (buf) {
|
|
|
|
memcpy(buf->msg, payload, nof_bytes);
|
|
|
|
buf->N_bytes = nof_bytes;
|
|
|
|
buf->set_timestamp();
|
|
|
|
metrics.num_rx_bytes += nof_bytes;
|
|
|
|
if (rrc->get_rb_name(lcid) == "SRB0") {
|
|
|
|
rrc->write_pdu(lcid, std::move(buf));
|
|
|
|
} else {
|
|
|
|
pdcp->write_pdu(lcid, std::move(buf));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log->error("Fatal Error: Couldn't allocate buffer in rlc_tm::write_pdu().\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace srslte
|