|
|
@ -33,40 +33,19 @@
|
|
|
|
|
|
|
|
|
|
|
|
namespace srslte {
|
|
|
|
namespace srslte {
|
|
|
|
|
|
|
|
|
|
|
|
rlc_um::rlc_um(uint32_t queue_len) : tx_sdu_queue(queue_len)
|
|
|
|
rlc_um::rlc_um(uint32_t queue_len)
|
|
|
|
|
|
|
|
:lcid(0)
|
|
|
|
|
|
|
|
,tx(queue_len)
|
|
|
|
|
|
|
|
,rrc(NULL)
|
|
|
|
|
|
|
|
,log(NULL)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
log = NULL;
|
|
|
|
|
|
|
|
pdcp = NULL;
|
|
|
|
|
|
|
|
rrc = NULL;
|
|
|
|
|
|
|
|
reordering_timer = NULL;
|
|
|
|
|
|
|
|
lcid = 0;
|
|
|
|
|
|
|
|
reordering_timer_id = 0;
|
|
|
|
|
|
|
|
bzero(&cfg, sizeof(srslte_rlc_um_config_t));
|
|
|
|
bzero(&cfg, sizeof(srslte_rlc_um_config_t));
|
|
|
|
|
|
|
|
|
|
|
|
tx_sdu = NULL;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rx_sdu = NULL;
|
|
|
|
|
|
|
|
pool = byte_buffer_pool::get_instance();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
pthread_mutex_init(&mutex, NULL);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
vt_us = 0;
|
|
|
|
|
|
|
|
vr_ur = 0;
|
|
|
|
|
|
|
|
vr_ux = 0;
|
|
|
|
|
|
|
|
vr_uh = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
vr_ur_in_rx_sdu = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mac_timers = NULL;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
pdu_lost = false;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Warning: must call stop() to properly deallocate all buffers
|
|
|
|
// Warning: must call stop() to properly deallocate all buffers
|
|
|
|
rlc_um::~rlc_um()
|
|
|
|
rlc_um::~rlc_um()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
pthread_mutex_destroy(&mutex);
|
|
|
|
stop();
|
|
|
|
pool = NULL;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::init(srslte::log *log_,
|
|
|
|
void rlc_um::init(srslte::log *log_,
|
|
|
@ -75,106 +54,81 @@ void rlc_um::init(srslte::log *log_,
|
|
|
|
srsue::rrc_interface_rlc *rrc_,
|
|
|
|
srsue::rrc_interface_rlc *rrc_,
|
|
|
|
srslte::mac_interface_timers *mac_timers_)
|
|
|
|
srslte::mac_interface_timers *mac_timers_)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
log = log_;
|
|
|
|
tx.init(log_);
|
|
|
|
|
|
|
|
rx.init(log_, lcid_, pdcp_, rrc_, mac_timers_);
|
|
|
|
lcid = lcid_;
|
|
|
|
lcid = lcid_;
|
|
|
|
pdcp = pdcp_;
|
|
|
|
rrc = rrc_; // needed to determine bearer name during configuration
|
|
|
|
rrc = rrc_;
|
|
|
|
log = log_;
|
|
|
|
mac_timers = mac_timers_;
|
|
|
|
|
|
|
|
reordering_timer_id = mac_timers->timer_get_unique_id();
|
|
|
|
|
|
|
|
reordering_timer = mac_timers->timer_get(reordering_timer_id);
|
|
|
|
|
|
|
|
tx_enabled = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::configure(srslte_rlc_config_t cnfg_)
|
|
|
|
|
|
|
|
|
|
|
|
bool rlc_um::configure(srslte_rlc_config_t cnfg_)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
cfg = cnfg_.um;
|
|
|
|
// determine bearer name and configure Rx/Tx objects
|
|
|
|
if(cnfg_.um.is_mrb){
|
|
|
|
rb_name = get_rb_name(rrc, lcid, cnfg_.um.is_mrb);
|
|
|
|
tx_sdu_queue.resize(512);
|
|
|
|
|
|
|
|
|
|
|
|
if (not rx.configure(cnfg_, rb_name)) {
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
switch(cnfg_.rlc_mode)
|
|
|
|
|
|
|
|
{
|
|
|
|
if (not tx.configure(cnfg_, rb_name)) {
|
|
|
|
case LIBLTE_RRC_RLC_MODE_UM_BI:
|
|
|
|
return false;
|
|
|
|
log->warning("%s configured in %s mode: "
|
|
|
|
}
|
|
|
|
"t_reordering=%d ms, rx_sn_field_length=%u bits, tx_sn_field_length=%u bits\n",
|
|
|
|
|
|
|
|
rb_name().c_str(), liblte_rrc_rlc_mode_text[cnfg_.rlc_mode],
|
|
|
|
log->warning("%s configured in %s mode: t_reordering=%d ms, rx_sn_field_length=%u bits, tx_sn_field_length=%u bits\n",
|
|
|
|
|
|
|
|
rb_name.c_str(), rlc_mode_text[cnfg_.rlc_mode],
|
|
|
|
cfg.t_reordering, rlc_umd_sn_size_num[cfg.rx_sn_field_length], rlc_umd_sn_size_num[cfg.rx_sn_field_length]);
|
|
|
|
cfg.t_reordering, rlc_umd_sn_size_num[cfg.rx_sn_field_length], rlc_umd_sn_size_num[cfg.rx_sn_field_length]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LIBLTE_RRC_RLC_MODE_UM_UNI_UL:
|
|
|
|
|
|
|
|
log->warning("%s configured in %s mode: tx_sn_field_length=%u bits\n",
|
|
|
|
|
|
|
|
rrc->get_rb_name(lcid).c_str(), liblte_rrc_rlc_mode_text[cnfg_.rlc_mode],
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rlc_umd_sn_size_num[cfg.rx_sn_field_length]);
|
|
|
|
// store config
|
|
|
|
break;
|
|
|
|
cfg = cnfg_.um;
|
|
|
|
case LIBLTE_RRC_RLC_MODE_UM_UNI_DL:
|
|
|
|
|
|
|
|
log->warning("%s configured in %s mode: "
|
|
|
|
return true;
|
|
|
|
"t_reordering=%d ms, rx_sn_field_length=%u bits\n",
|
|
|
|
}
|
|
|
|
rb_name().c_str(), liblte_rrc_rlc_mode_text[cnfg_.rlc_mode],
|
|
|
|
|
|
|
|
cfg.t_reordering, rlc_umd_sn_size_num[cfg.rx_sn_field_length]);
|
|
|
|
|
|
|
|
break;
|
|
|
|
bool rlc_um::rlc_um_rx::configure(srslte_rlc_config_t cnfg_, std::string rb_name_)
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
log->error("RLC configuration mode not recognized\n");
|
|
|
|
cfg = cnfg_.um;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (cfg.rx_mod == 0) {
|
|
|
|
|
|
|
|
log->error("Error configuring %s RLC UM: rx_mod==0\n", get_rb_name());
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rb_name = rb_name_;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rx_enabled = true;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::empty_queue() {
|
|
|
|
void rlc_um::empty_queue() {
|
|
|
|
// Drop all messages in TX SDU queue
|
|
|
|
// Drop all messages in TX SDU queue
|
|
|
|
byte_buffer_t *buf;
|
|
|
|
tx.empty_queue();
|
|
|
|
while(tx_sdu_queue.try_read(&buf)) {
|
|
|
|
|
|
|
|
pool->deallocate(buf);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
tx_sdu_queue.reset();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool rlc_um::is_mrb()
|
|
|
|
bool rlc_um::is_mrb()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
return cfg.is_mrb;
|
|
|
|
return cfg.is_mrb;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::reestablish() {
|
|
|
|
|
|
|
|
stop();
|
|
|
|
|
|
|
|
tx_enabled = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::stop()
|
|
|
|
void rlc_um::reestablish()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// Empty tx_sdu_queue before locking the mutex
|
|
|
|
tx.reestablish(); // calls stop and enables tx again
|
|
|
|
tx_enabled = false;
|
|
|
|
rx.reestablish(); // nothing else needed
|
|
|
|
empty_queue();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
|
|
|
|
vt_us = 0;
|
|
|
|
|
|
|
|
vr_ur = 0;
|
|
|
|
|
|
|
|
vr_ux = 0;
|
|
|
|
|
|
|
|
vr_uh = 0;
|
|
|
|
|
|
|
|
pdu_lost = false;
|
|
|
|
|
|
|
|
if(rx_sdu) {
|
|
|
|
|
|
|
|
pool->deallocate(rx_sdu);
|
|
|
|
|
|
|
|
rx_sdu = NULL;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if(tx_sdu) {
|
|
|
|
|
|
|
|
pool->deallocate(tx_sdu);
|
|
|
|
|
|
|
|
tx_sdu = NULL;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if(reordering_timer) {
|
|
|
|
|
|
|
|
reordering_timer->stop();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Drop all messages in RX window
|
|
|
|
void rlc_um::stop()
|
|
|
|
std::map<uint32_t, rlc_umd_pdu_t>::iterator it;
|
|
|
|
{
|
|
|
|
for(it = rx_window.begin(); it != rx_window.end(); it++) {
|
|
|
|
tx.stop();
|
|
|
|
pool->deallocate(it->second.buf);
|
|
|
|
rx.stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
rx_window.clear();
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (mac_timers && reordering_timer) {
|
|
|
|
|
|
|
|
mac_timers->timer_release_id(reordering_timer_id);
|
|
|
|
|
|
|
|
reordering_timer = NULL;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rlc_mode_t rlc_um::get_mode()
|
|
|
|
rlc_mode_t rlc_um::get_mode()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
return RLC_MODE_UM;
|
|
|
|
return RLC_MODE_UM;
|
|
|
@ -188,147 +142,247 @@ uint32_t rlc_um::get_bearer()
|
|
|
|
/****************************************************************************
|
|
|
|
/****************************************************************************
|
|
|
|
* PDCP interface
|
|
|
|
* PDCP interface
|
|
|
|
***************************************************************************/
|
|
|
|
***************************************************************************/
|
|
|
|
void rlc_um::write_sdu(byte_buffer_t *sdu)
|
|
|
|
void rlc_um::write_sdu(byte_buffer_t *sdu, bool blocking)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (!tx_enabled) {
|
|
|
|
if (blocking) {
|
|
|
|
byte_buffer_pool::get_instance()->deallocate(sdu);
|
|
|
|
tx.write_sdu(sdu);
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sdu) {
|
|
|
|
|
|
|
|
tx_sdu_queue.write(sdu);
|
|
|
|
|
|
|
|
log->info_hex(sdu->msg, sdu->N_bytes, "%s Tx SDU (%d B ,tx_sdu_queue_len=%d)", rrc->get_rb_name(lcid).c_str(), sdu->N_bytes, tx_sdu_queue.size());
|
|
|
|
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
log->warning("NULL SDU pointer in write_sdu()\n");
|
|
|
|
tx.try_write_sdu(sdu);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::write_sdu_nb(byte_buffer_t *sdu)
|
|
|
|
/****************************************************************************
|
|
|
|
|
|
|
|
* MAC interface
|
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t rlc_um::get_buffer_state()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (!tx_enabled) {
|
|
|
|
return tx.get_buffer_size_bytes();
|
|
|
|
byte_buffer_pool::get_instance()->deallocate(sdu);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
uint32_t rlc_um::get_total_buffer_state()
|
|
|
|
if (sdu) {
|
|
|
|
{
|
|
|
|
if (tx_sdu_queue.try_write(sdu)) {
|
|
|
|
return get_buffer_state();
|
|
|
|
log->info_hex(sdu->msg, sdu->N_bytes, "%s Tx SDU (%d B,tx_sdu_queue_len=%d)", rrc->get_rb_name(lcid).c_str(), sdu->N_bytes, tx_sdu_queue.size());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
|
|
|
|
log->debug_hex(sdu->msg, sdu->N_bytes, "[Dropped SDU] %s Tx SDU (%d B,tx_sdu_queue_len=%d)", rrc->get_rb_name(lcid).c_str(), sdu->N_bytes, tx_sdu_queue.size());
|
|
|
|
int rlc_um::read_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
pool->deallocate(sdu);
|
|
|
|
{
|
|
|
|
}
|
|
|
|
return tx.build_data_pdu(payload, nof_bytes);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::write_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
rx.handle_data_pdu(payload, nof_bytes);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t rlc_um::get_num_tx_bytes()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
return tx.get_num_tx_bytes();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t rlc_um::get_num_rx_bytes()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
return rx.get_num_rx_bytes();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::reset_metrics()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
tx.reset_metrics();
|
|
|
|
|
|
|
|
rx.reset_metrics();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
|
|
|
* Helper functions
|
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
std::string rlc_um::get_rb_name(srsue::rrc_interface_rlc *rrc, uint32_t lcid, bool is_mrb)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if(is_mrb) {
|
|
|
|
|
|
|
|
std::stringstream ss;
|
|
|
|
|
|
|
|
ss << "MRB" << lcid;
|
|
|
|
|
|
|
|
return ss.str();
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
log->warning("NULL SDU pointer in write_sdu()\n");
|
|
|
|
return rrc->get_rb_name(lcid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
/****************************************************************************
|
|
|
|
* MAC interface
|
|
|
|
* Tx subclass implementation
|
|
|
|
***************************************************************************/
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t rlc_um::get_buffer_state()
|
|
|
|
rlc_um::rlc_um_tx::rlc_um_tx(uint32_t queue_len)
|
|
|
|
|
|
|
|
:tx_sdu_queue(queue_len)
|
|
|
|
|
|
|
|
,pool(byte_buffer_pool::get_instance())
|
|
|
|
|
|
|
|
,log(NULL)
|
|
|
|
|
|
|
|
,tx_sdu(NULL)
|
|
|
|
|
|
|
|
,vt_us(0)
|
|
|
|
|
|
|
|
,tx_enabled(false)
|
|
|
|
|
|
|
|
,num_tx_bytes(0)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// Bytes needed for tx SDUs
|
|
|
|
pthread_mutex_init(&mutex, NULL);
|
|
|
|
uint32_t n_sdus = tx_sdu_queue.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t n_bytes = tx_sdu_queue.size_bytes();
|
|
|
|
|
|
|
|
if(tx_sdu)
|
|
|
|
rlc_um::rlc_um_tx::~rlc_um_tx()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
n_sdus++;
|
|
|
|
pthread_mutex_destroy(&mutex);
|
|
|
|
n_bytes += tx_sdu->N_bytes;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_tx::init(srslte::log *log_)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
log = log_;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool rlc_um::rlc_um_tx::configure(srslte_rlc_config_t cnfg_, std::string rb_name_)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
cfg = cnfg_.um;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (cfg.tx_mod == 0) {
|
|
|
|
|
|
|
|
log->error("Error configuring %s RLC UM: tx_mod==0\n", get_rb_name());
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Room needed for header extensions? (integer rounding)
|
|
|
|
if(cfg.is_mrb){
|
|
|
|
if(n_sdus > 1)
|
|
|
|
tx_sdu_queue.resize(512);
|
|
|
|
n_bytes += ((n_sdus-1)*1.5)+0.5;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Room needed for fixed header?
|
|
|
|
rb_name = rb_name_;
|
|
|
|
if(n_bytes > 0)
|
|
|
|
tx_enabled = true;
|
|
|
|
n_bytes += (cfg.is_mrb)?2:3;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return n_bytes;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t rlc_um::get_total_buffer_state()
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_tx::stop()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
return get_buffer_state();
|
|
|
|
tx_enabled = false;
|
|
|
|
|
|
|
|
empty_queue();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int rlc_um::read_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_tx::reestablish()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
stop();
|
|
|
|
log->debug("MAC opportunity - %d bytes\n", nof_bytes);
|
|
|
|
tx_enabled = true;
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
|
|
|
|
r = build_data_pdu(payload, nof_bytes);
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
|
|
|
|
return r;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::write_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_tx::empty_queue()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
handle_data_pdu(payload, nof_bytes);
|
|
|
|
|
|
|
|
|
|
|
|
// deallocate all SDUs in transmit queue
|
|
|
|
|
|
|
|
while(tx_sdu_queue.size() > 0) {
|
|
|
|
|
|
|
|
byte_buffer_t *buf;
|
|
|
|
|
|
|
|
tx_sdu_queue.read(&buf);
|
|
|
|
|
|
|
|
pool->deallocate(buf);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// deallocate SDU that is currently processed
|
|
|
|
|
|
|
|
if(tx_sdu) {
|
|
|
|
|
|
|
|
pool->deallocate(tx_sdu);
|
|
|
|
|
|
|
|
tx_sdu = NULL;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
|
|
|
* Timeout callback interface
|
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::timer_expired(uint32_t timeout_id)
|
|
|
|
uint32_t rlc_um::rlc_um_tx::get_num_tx_bytes()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
return num_tx_bytes;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_tx::reset_metrics()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if(reordering_timer_id == timeout_id)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
|
|
|
|
num_tx_bytes = 0;
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 36.322 v10 Section 5.1.2.2.4
|
|
|
|
|
|
|
|
log->info("%s reordering timeout expiry - updating vr_ur and reassembling\n",
|
|
|
|
|
|
|
|
rb_name().c_str());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
log->warning("Lost PDU SN: %d\n", vr_ur);
|
|
|
|
uint32_t rlc_um::rlc_um_tx::get_buffer_size_bytes()
|
|
|
|
pdu_lost = true;
|
|
|
|
{
|
|
|
|
rx_sdu->reset();
|
|
|
|
// Bytes needed for tx SDUs
|
|
|
|
while(RX_MOD_BASE(vr_ur) < RX_MOD_BASE(vr_ux))
|
|
|
|
uint32_t n_sdus = tx_sdu_queue.size();
|
|
|
|
{
|
|
|
|
uint32_t n_bytes = tx_sdu_queue.size_bytes();
|
|
|
|
vr_ur = (vr_ur + 1)%cfg.rx_mod;
|
|
|
|
if(tx_sdu) {
|
|
|
|
log->debug("Entering Reassemble from timeout id=%d\n", timeout_id);
|
|
|
|
n_sdus++;
|
|
|
|
reassemble_rx_sdus();
|
|
|
|
n_bytes += tx_sdu->N_bytes;
|
|
|
|
log->debug("Finished reassemble from timeout id=%d\n", timeout_id);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
reordering_timer->stop();
|
|
|
|
|
|
|
|
if(RX_MOD_BASE(vr_uh) > RX_MOD_BASE(vr_ur))
|
|
|
|
// Room needed for header extensions? (integer rounding)
|
|
|
|
{
|
|
|
|
if(n_sdus > 1) {
|
|
|
|
reordering_timer->set(this, cfg.t_reordering);
|
|
|
|
n_bytes += ((n_sdus-1)*1.5)+0.5;
|
|
|
|
reordering_timer->run();
|
|
|
|
|
|
|
|
vr_ux = vr_uh;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
debug_state();
|
|
|
|
// Room needed for fixed header?
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
if(n_bytes > 0)
|
|
|
|
|
|
|
|
n_bytes += (cfg.is_mrb)?2:3;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return n_bytes;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_tx::write_sdu(byte_buffer_t *sdu)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (!tx_enabled) {
|
|
|
|
|
|
|
|
byte_buffer_pool::get_instance()->deallocate(sdu);
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (sdu) {
|
|
|
|
|
|
|
|
tx_sdu_queue.write(sdu);
|
|
|
|
|
|
|
|
log->info_hex(sdu->msg, sdu->N_bytes, "%s Tx SDU (%d B, tx_sdu_queue_len=%d)", get_rb_name(), sdu->N_bytes, tx_sdu_queue.size());
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
log->warning("NULL SDU pointer in write_sdu()\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool rlc_um::reordering_timeout_running()
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_tx::try_write_sdu(byte_buffer_t *sdu)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
return reordering_timer->is_running();
|
|
|
|
if (!tx_enabled) {
|
|
|
|
|
|
|
|
byte_buffer_pool::get_instance()->deallocate(sdu);
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (sdu) {
|
|
|
|
|
|
|
|
if (tx_sdu_queue.try_write(sdu)) {
|
|
|
|
|
|
|
|
log->info_hex(sdu->msg, sdu->N_bytes, "%s Tx SDU (%d B, tx_sdu_queue_len=%d)", get_rb_name(), sdu->N_bytes, tx_sdu_queue.size());
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
log->info_hex(sdu->msg, sdu->N_bytes, "[Dropped SDU] %s Tx SDU (%d B, tx_sdu_queue_len=%d)", get_rb_name(), sdu->N_bytes, tx_sdu_queue.size());
|
|
|
|
|
|
|
|
pool->deallocate(sdu);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
log->warning("NULL SDU pointer in write_sdu()\n");
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
|
|
|
* Helpers
|
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int rlc_um::build_data_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
int rlc_um::rlc_um_tx::build_data_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if(!tx_sdu && tx_sdu_queue.size() == 0)
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
{
|
|
|
|
log->debug("MAC opportunity - %d bytes\n", nof_bytes);
|
|
|
|
|
|
|
|
if(!tx_sdu && tx_sdu_queue.size() == 0) {
|
|
|
|
log->info("No data available to be sent\n");
|
|
|
|
log->info("No data available to be sent\n");
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
return 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
byte_buffer_t *pdu = pool_allocate;
|
|
|
|
byte_buffer_t *pdu = pool_allocate;
|
|
|
|
if(!pdu || pdu->N_bytes != 0)
|
|
|
|
if(!pdu || pdu->N_bytes != 0) {
|
|
|
|
{
|
|
|
|
|
|
|
|
log->error("Failed to allocate PDU buffer\n");
|
|
|
|
log->error("Failed to allocate PDU buffer\n");
|
|
|
|
return -1;
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
rlc_umd_pdu_header_t header;
|
|
|
|
rlc_umd_pdu_header_t header;
|
|
|
|
header.fi = RLC_FI_FIELD_START_AND_END_ALIGNED;
|
|
|
|
header.fi = RLC_FI_FIELD_START_AND_END_ALIGNED;
|
|
|
|
header.sn = vt_us;
|
|
|
|
header.sn = vt_us;
|
|
|
@ -346,17 +400,17 @@ int rlc_um::build_data_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
pool->deallocate(pdu);
|
|
|
|
pool->deallocate(pdu);
|
|
|
|
log->warning("%s Cannot build a PDU - %d bytes available, %d bytes required for header\n",
|
|
|
|
log->warning("%s Cannot build a PDU - %d bytes available, %d bytes required for header\n",
|
|
|
|
rb_name().c_str(), nof_bytes, head_len);
|
|
|
|
get_rb_name(), nof_bytes, head_len);
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
return 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Check for SDU segment
|
|
|
|
// Check for SDU segment
|
|
|
|
if(tx_sdu)
|
|
|
|
if(tx_sdu) {
|
|
|
|
{
|
|
|
|
|
|
|
|
uint32_t space = pdu_space-head_len;
|
|
|
|
uint32_t space = pdu_space-head_len;
|
|
|
|
to_move = space >= tx_sdu->N_bytes ? tx_sdu->N_bytes : space;
|
|
|
|
to_move = space >= tx_sdu->N_bytes ? tx_sdu->N_bytes : space;
|
|
|
|
log->debug("%s adding remainder of SDU segment - %d bytes of %d remaining\n",
|
|
|
|
log->debug("%s adding remainder of SDU segment - %d bytes of %d remaining\n",
|
|
|
|
rb_name().c_str(), to_move, tx_sdu->N_bytes);
|
|
|
|
get_rb_name(), to_move, tx_sdu->N_bytes);
|
|
|
|
memcpy(pdu_ptr, tx_sdu->msg, to_move);
|
|
|
|
memcpy(pdu_ptr, tx_sdu->msg, to_move);
|
|
|
|
last_li = to_move;
|
|
|
|
last_li = to_move;
|
|
|
|
pdu_ptr += to_move;
|
|
|
|
pdu_ptr += to_move;
|
|
|
@ -366,7 +420,7 @@ int rlc_um::build_data_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
if(tx_sdu->N_bytes == 0)
|
|
|
|
if(tx_sdu->N_bytes == 0)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
log->debug("%s Complete SDU scheduled for tx. Stack latency: %ld us\n",
|
|
|
|
log->debug("%s Complete SDU scheduled for tx. Stack latency: %ld us\n",
|
|
|
|
rrc->get_rb_name(lcid).c_str(), tx_sdu->get_latency_us());
|
|
|
|
get_rb_name(), tx_sdu->get_latency_us());
|
|
|
|
|
|
|
|
|
|
|
|
pool->deallocate(tx_sdu);
|
|
|
|
pool->deallocate(tx_sdu);
|
|
|
|
tx_sdu = NULL;
|
|
|
|
tx_sdu = NULL;
|
|
|
@ -376,8 +430,7 @@ int rlc_um::build_data_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Pull SDUs from queue
|
|
|
|
// Pull SDUs from queue
|
|
|
|
while(pdu_space > head_len + 1 && tx_sdu_queue.size() > 0)
|
|
|
|
while(pdu_space > head_len + 1 && tx_sdu_queue.size() > 0) {
|
|
|
|
{
|
|
|
|
|
|
|
|
log->debug("pdu_space=%d, head_len=%d\n", pdu_space, head_len);
|
|
|
|
log->debug("pdu_space=%d, head_len=%d\n", pdu_space, head_len);
|
|
|
|
if(last_li > 0)
|
|
|
|
if(last_li > 0)
|
|
|
|
header.li[header.N_li++] = last_li;
|
|
|
|
header.li[header.N_li++] = last_li;
|
|
|
@ -386,17 +439,16 @@ int rlc_um::build_data_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
uint32_t space = pdu_space-head_len;
|
|
|
|
uint32_t space = pdu_space-head_len;
|
|
|
|
to_move = space >= tx_sdu->N_bytes ? tx_sdu->N_bytes : space;
|
|
|
|
to_move = space >= tx_sdu->N_bytes ? tx_sdu->N_bytes : space;
|
|
|
|
log->debug("%s adding new SDU segment - %d bytes of %d remaining\n",
|
|
|
|
log->debug("%s adding new SDU segment - %d bytes of %d remaining\n",
|
|
|
|
rb_name().c_str(), to_move, tx_sdu->N_bytes);
|
|
|
|
get_rb_name(), to_move, tx_sdu->N_bytes);
|
|
|
|
memcpy(pdu_ptr, tx_sdu->msg, to_move);
|
|
|
|
memcpy(pdu_ptr, tx_sdu->msg, to_move);
|
|
|
|
last_li = to_move;
|
|
|
|
last_li = to_move;
|
|
|
|
pdu_ptr += to_move;
|
|
|
|
pdu_ptr += to_move;
|
|
|
|
pdu->N_bytes += to_move;
|
|
|
|
pdu->N_bytes += to_move;
|
|
|
|
tx_sdu->N_bytes -= to_move;
|
|
|
|
tx_sdu->N_bytes -= to_move;
|
|
|
|
tx_sdu->msg += to_move;
|
|
|
|
tx_sdu->msg += to_move;
|
|
|
|
if(tx_sdu->N_bytes == 0)
|
|
|
|
if(tx_sdu->N_bytes == 0) {
|
|
|
|
{
|
|
|
|
|
|
|
|
log->debug("%s Complete SDU scheduled for tx. Stack latency: %ld us\n",
|
|
|
|
log->debug("%s Complete SDU scheduled for tx. Stack latency: %ld us\n",
|
|
|
|
rrc->get_rb_name(lcid).c_str(), tx_sdu->get_latency_us());
|
|
|
|
get_rb_name(), tx_sdu->get_latency_us());
|
|
|
|
|
|
|
|
|
|
|
|
pool->deallocate(tx_sdu);
|
|
|
|
pool->deallocate(tx_sdu);
|
|
|
|
tx_sdu = NULL;
|
|
|
|
tx_sdu = NULL;
|
|
|
@ -404,70 +456,177 @@ int rlc_um::build_data_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
pdu_space -= to_move;
|
|
|
|
pdu_space -= to_move;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if(tx_sdu)
|
|
|
|
if(tx_sdu) {
|
|
|
|
header.fi |= RLC_FI_FIELD_NOT_END_ALIGNED; // Last byte does not correspond to last byte of SDU
|
|
|
|
header.fi |= RLC_FI_FIELD_NOT_END_ALIGNED; // Last byte does not correspond to last byte of SDU
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Set SN
|
|
|
|
// Set SN
|
|
|
|
|
|
|
|
|
|
|
|
header.sn = vt_us;
|
|
|
|
header.sn = vt_us;
|
|
|
|
vt_us = (vt_us + 1)%cfg.tx_mod;
|
|
|
|
vt_us = (vt_us + 1)%cfg.tx_mod;
|
|
|
|
|
|
|
|
|
|
|
|
// Add header and TX
|
|
|
|
// Add header and TX
|
|
|
|
log->debug("%s packing PDU with length %d\n", rb_name().c_str(), pdu->N_bytes);
|
|
|
|
log->debug("%s packing PDU with length %d\n", get_rb_name(), pdu->N_bytes);
|
|
|
|
rlc_um_write_data_pdu_header(&header, pdu);
|
|
|
|
rlc_um_write_data_pdu_header(&header, pdu);
|
|
|
|
memcpy(payload, pdu->msg, pdu->N_bytes);
|
|
|
|
memcpy(payload, pdu->msg, pdu->N_bytes);
|
|
|
|
uint32_t ret = pdu->N_bytes;
|
|
|
|
uint32_t ret = pdu->N_bytes;
|
|
|
|
|
|
|
|
|
|
|
|
log->debug("%s returning length %d\n", rrc->get_rb_name(lcid).c_str(), pdu->N_bytes);
|
|
|
|
log->debug("%s returning length %d\n", get_rb_name(), pdu->N_bytes);
|
|
|
|
pool->deallocate(pdu);
|
|
|
|
pool->deallocate(pdu);
|
|
|
|
|
|
|
|
|
|
|
|
debug_state();
|
|
|
|
debug_state();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
num_tx_bytes += ret;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
return ret;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::handle_data_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
|
|
|
|
|
|
|
|
const char* rlc_um::rlc_um_tx::get_rb_name()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
return rb_name.c_str();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_tx::debug_state()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
log->debug("%s vt_us = %d\n", get_rb_name(), vt_us);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
|
|
|
* Rx subclass implementation
|
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rlc_um::rlc_um_rx::rlc_um_rx()
|
|
|
|
|
|
|
|
:reordering_timer(NULL)
|
|
|
|
|
|
|
|
,reordering_timer_id(0)
|
|
|
|
|
|
|
|
,pool(byte_buffer_pool::get_instance())
|
|
|
|
|
|
|
|
,log(NULL)
|
|
|
|
|
|
|
|
,pdcp(NULL)
|
|
|
|
|
|
|
|
,rrc(NULL)
|
|
|
|
|
|
|
|
,rx_sdu(NULL)
|
|
|
|
|
|
|
|
,vr_ur(0)
|
|
|
|
|
|
|
|
,vr_ux (0)
|
|
|
|
|
|
|
|
,vr_uh(0)
|
|
|
|
|
|
|
|
,vr_ur_in_rx_sdu(0)
|
|
|
|
|
|
|
|
,pdu_lost(false)
|
|
|
|
|
|
|
|
,mac_timers(NULL)
|
|
|
|
|
|
|
|
,lcid(0)
|
|
|
|
|
|
|
|
,num_rx_bytes(0)
|
|
|
|
|
|
|
|
,rx_enabled(false)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
pthread_mutex_init(&mutex, NULL);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rlc_um::rlc_um_rx::~rlc_um_rx()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
pthread_mutex_destroy(&mutex);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_rx::init(srslte::log *log_, uint32_t lcid_, srsue::pdcp_interface_rlc *pdcp_, srsue::rrc_interface_rlc *rrc_, srslte::mac_interface_timers *mac_timers_)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
log = log_;
|
|
|
|
|
|
|
|
lcid = lcid_;
|
|
|
|
|
|
|
|
pdcp = pdcp_;
|
|
|
|
|
|
|
|
rrc = rrc_;
|
|
|
|
|
|
|
|
mac_timers = mac_timers_;
|
|
|
|
|
|
|
|
reordering_timer_id = mac_timers->timer_get_unique_id();
|
|
|
|
|
|
|
|
reordering_timer = mac_timers->timer_get(reordering_timer_id);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_rx::reestablish()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
stop();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_rx::stop()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
|
|
|
|
if(reordering_timer) {
|
|
|
|
|
|
|
|
reordering_timer->stop();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
vr_ur = 0;
|
|
|
|
|
|
|
|
vr_ux = 0;
|
|
|
|
|
|
|
|
vr_uh = 0;
|
|
|
|
|
|
|
|
pdu_lost = false;
|
|
|
|
|
|
|
|
rx_enabled = false;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if(rx_sdu) {
|
|
|
|
|
|
|
|
pool->deallocate(rx_sdu);
|
|
|
|
|
|
|
|
rx_sdu = NULL;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (mac_timers && reordering_timer) {
|
|
|
|
|
|
|
|
mac_timers->timer_release_id(reordering_timer_id);
|
|
|
|
|
|
|
|
reordering_timer = NULL;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Drop all messages in RX window
|
|
|
|
|
|
|
|
std::map<uint32_t, rlc_umd_pdu_t>::iterator it;
|
|
|
|
|
|
|
|
for(it = rx_window.begin(); it != rx_window.end(); it++) {
|
|
|
|
|
|
|
|
pool->deallocate(it->second.buf);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
rx_window.clear();
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_rx::handle_data_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
rlc_umd_pdu_t pdu;
|
|
|
|
|
|
|
|
int header_len = 0;
|
|
|
|
std::map<uint32_t, rlc_umd_pdu_t>::iterator it;
|
|
|
|
std::map<uint32_t, rlc_umd_pdu_t>::iterator it;
|
|
|
|
rlc_umd_pdu_header_t header;
|
|
|
|
rlc_umd_pdu_header_t header;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!rx_enabled) {
|
|
|
|
|
|
|
|
goto unlock_and_exit;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
num_rx_bytes += nof_bytes;
|
|
|
|
|
|
|
|
|
|
|
|
rlc_um_read_data_pdu_header(payload, nof_bytes, cfg.rx_sn_field_length, &header);
|
|
|
|
rlc_um_read_data_pdu_header(payload, nof_bytes, cfg.rx_sn_field_length, &header);
|
|
|
|
|
|
|
|
|
|
|
|
log->info_hex(payload, nof_bytes, "RX %s Rx data PDU SN: %d",
|
|
|
|
log->info_hex(payload, nof_bytes, "RX %s Rx data PDU SN: %d", get_rb_name(), header.sn);
|
|
|
|
rb_name().c_str(), header.sn);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if(RX_MOD_BASE(header.sn) >= RX_MOD_BASE(vr_uh-cfg.rx_window_size) &&
|
|
|
|
if(RX_MOD_BASE(header.sn) >= RX_MOD_BASE(vr_uh-cfg.rx_window_size) &&
|
|
|
|
RX_MOD_BASE(header.sn) < RX_MOD_BASE(vr_ur))
|
|
|
|
RX_MOD_BASE(header.sn) < RX_MOD_BASE(vr_ur))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
log->info("%s SN: %d outside rx window [%d:%d] - discarding\n",
|
|
|
|
log->info("%s SN: %d outside rx window [%d:%d] - discarding\n",
|
|
|
|
rb_name().c_str(), header.sn, vr_ur, vr_uh);
|
|
|
|
get_rb_name(), header.sn, vr_ur, vr_uh);
|
|
|
|
return;
|
|
|
|
goto unlock_and_exit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
it = rx_window.find(header.sn);
|
|
|
|
it = rx_window.find(header.sn);
|
|
|
|
if(rx_window.end() != it)
|
|
|
|
if(rx_window.end() != it)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
log->info("%s Discarding duplicate SN: %d\n",
|
|
|
|
log->info("%s Discarding duplicate SN: %d\n", get_rb_name(), header.sn);
|
|
|
|
rb_name().c_str(), header.sn);
|
|
|
|
goto unlock_and_exit;
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Write to rx window
|
|
|
|
// Write to rx window
|
|
|
|
rlc_umd_pdu_t pdu;
|
|
|
|
|
|
|
|
pdu.buf = pool_allocate;
|
|
|
|
pdu.buf = pool_allocate;
|
|
|
|
if (!pdu.buf) {
|
|
|
|
if (!pdu.buf) {
|
|
|
|
log->error("Discarting packet: no space in buffer pool\n");
|
|
|
|
log->error("Discarting packet: no space in buffer pool\n");
|
|
|
|
return;
|
|
|
|
goto unlock_and_exit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
memcpy(pdu.buf->msg, payload, nof_bytes);
|
|
|
|
memcpy(pdu.buf->msg, payload, nof_bytes);
|
|
|
|
pdu.buf->N_bytes = nof_bytes;
|
|
|
|
pdu.buf->N_bytes = nof_bytes;
|
|
|
|
//Strip header from PDU
|
|
|
|
//Strip header from PDU
|
|
|
|
int header_len = rlc_um_packed_length(&header);
|
|
|
|
header_len = rlc_um_packed_length(&header);
|
|
|
|
pdu.buf->msg += header_len;
|
|
|
|
pdu.buf->msg += header_len;
|
|
|
|
pdu.buf->N_bytes -= header_len;
|
|
|
|
pdu.buf->N_bytes -= header_len;
|
|
|
|
pdu.header = header;
|
|
|
|
pdu.header = header;
|
|
|
|
rx_window[header.sn] = pdu;
|
|
|
|
rx_window[header.sn] = pdu;
|
|
|
|
|
|
|
|
|
|
|
|
// Update vr_uh
|
|
|
|
// Update vr_uh
|
|
|
|
if(!inside_reordering_window(header.sn))
|
|
|
|
if(!inside_reordering_window(header.sn)) {
|
|
|
|
vr_uh = (header.sn + 1)%cfg.rx_mod;
|
|
|
|
vr_uh = (header.sn + 1)%cfg.rx_mod;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Reassemble and deliver SDUs, while updating vr_ur
|
|
|
|
// Reassemble and deliver SDUs, while updating vr_ur
|
|
|
|
log->debug("Entering Reassemble from received PDU\n");
|
|
|
|
log->debug("Entering Reassemble from received PDU\n");
|
|
|
@ -475,18 +634,15 @@ void rlc_um::handle_data_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
log->debug("Finished reassemble from received PDU\n");
|
|
|
|
log->debug("Finished reassemble from received PDU\n");
|
|
|
|
|
|
|
|
|
|
|
|
// Update reordering variables and timers
|
|
|
|
// Update reordering variables and timers
|
|
|
|
if(reordering_timer->is_running())
|
|
|
|
if(reordering_timer->is_running()) {
|
|
|
|
{
|
|
|
|
|
|
|
|
if(RX_MOD_BASE(vr_ux) <= RX_MOD_BASE(vr_ur) ||
|
|
|
|
if(RX_MOD_BASE(vr_ux) <= RX_MOD_BASE(vr_ur) ||
|
|
|
|
(!inside_reordering_window(vr_ux) && vr_ux != vr_uh))
|
|
|
|
(!inside_reordering_window(vr_ux) && vr_ux != vr_uh))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
reordering_timer->stop();
|
|
|
|
reordering_timer->stop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(!reordering_timer->is_running())
|
|
|
|
if(!reordering_timer->is_running()) {
|
|
|
|
{
|
|
|
|
if(RX_MOD_BASE(vr_uh) > RX_MOD_BASE(vr_ur)) {
|
|
|
|
if(RX_MOD_BASE(vr_uh) > RX_MOD_BASE(vr_ur))
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
reordering_timer->set(this, cfg.t_reordering);
|
|
|
|
reordering_timer->set(this, cfg.t_reordering);
|
|
|
|
reordering_timer->run();
|
|
|
|
reordering_timer->run();
|
|
|
|
vr_ux = vr_uh;
|
|
|
|
vr_ux = vr_uh;
|
|
|
@ -494,9 +650,14 @@ void rlc_um::handle_data_pdu(uint8_t *payload, uint32_t nof_bytes)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
debug_state();
|
|
|
|
debug_state();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
unlock_and_exit:
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::reassemble_rx_sdus()
|
|
|
|
|
|
|
|
|
|
|
|
// No locking required as only called from within handle_data_pdu and timer_expired which lock
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_rx::reassemble_rx_sdus()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if(!rx_sdu) {
|
|
|
|
if(!rx_sdu) {
|
|
|
|
rx_sdu = pool_allocate;
|
|
|
|
rx_sdu = pool_allocate;
|
|
|
@ -536,7 +697,7 @@ void rlc_um::reassemble_rx_sdus()
|
|
|
|
log->warning("Dropping remainder of lost PDU (lower edge middle segments, vr_ur=%d, vr_ur_in_rx_sdu=%d)\n", vr_ur, vr_ur_in_rx_sdu);
|
|
|
|
log->warning("Dropping remainder of lost PDU (lower edge middle segments, vr_ur=%d, vr_ur_in_rx_sdu=%d)\n", vr_ur, vr_ur_in_rx_sdu);
|
|
|
|
rx_sdu->reset();
|
|
|
|
rx_sdu->reset();
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
log->info_hex(rx_sdu->msg, rx_sdu->N_bytes, "%s Rx SDU vr_ur=%d, i=%d (lower edge middle segments)", rb_name().c_str(), vr_ur, i);
|
|
|
|
log->info_hex(rx_sdu->msg, rx_sdu->N_bytes, "%s Rx SDU vr_ur=%d, i=%d (lower edge middle segments)", get_rb_name(), vr_ur, i);
|
|
|
|
rx_sdu->set_timestamp();
|
|
|
|
rx_sdu->set_timestamp();
|
|
|
|
if(cfg.is_mrb){
|
|
|
|
if(cfg.is_mrb){
|
|
|
|
pdcp->write_pdu_mch(lcid, rx_sdu);
|
|
|
|
pdcp->write_pdu_mch(lcid, rx_sdu);
|
|
|
@ -553,7 +714,6 @@ void rlc_um::reassemble_rx_sdus()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Handle last segment
|
|
|
|
// Handle last segment
|
|
|
|
|
|
|
|
|
|
|
|
if (rx_sdu->N_bytes > 0 || rlc_um_start_aligned(rx_window[vr_ur].header.fi)) {
|
|
|
|
if (rx_sdu->N_bytes > 0 || rlc_um_start_aligned(rx_window[vr_ur].header.fi)) {
|
|
|
|
log->debug("Writing last segment in SDU buffer. Lower edge vr_ur=%d, Buffer size=%d, segment size=%d\n",
|
|
|
|
log->debug("Writing last segment in SDU buffer. Lower edge vr_ur=%d, Buffer size=%d, segment size=%d\n",
|
|
|
|
vr_ur, rx_sdu->N_bytes, rx_window[vr_ur].buf->N_bytes);
|
|
|
|
vr_ur, rx_sdu->N_bytes, rx_window[vr_ur].buf->N_bytes);
|
|
|
@ -567,7 +727,7 @@ void rlc_um::reassemble_rx_sdus()
|
|
|
|
log->warning("Dropping remainder of lost PDU (lower edge last segments)\n");
|
|
|
|
log->warning("Dropping remainder of lost PDU (lower edge last segments)\n");
|
|
|
|
rx_sdu->reset();
|
|
|
|
rx_sdu->reset();
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
log->info_hex(rx_sdu->msg, rx_sdu->N_bytes, "%s Rx SDU vr_ur=%d (lower edge last segments)", rrc->get_rb_name(lcid).c_str(), vr_ur);
|
|
|
|
log->info_hex(rx_sdu->msg, rx_sdu->N_bytes, "%s Rx SDU vr_ur=%d (lower edge last segments)", get_rb_name(), vr_ur);
|
|
|
|
rx_sdu->set_timestamp();
|
|
|
|
rx_sdu->set_timestamp();
|
|
|
|
if(cfg.is_mrb){
|
|
|
|
if(cfg.is_mrb){
|
|
|
|
pdcp->write_pdu_mch(lcid, rx_sdu);
|
|
|
|
pdcp->write_pdu_mch(lcid, rx_sdu);
|
|
|
@ -592,13 +752,10 @@ void rlc_um::reassemble_rx_sdus()
|
|
|
|
vr_ur = (vr_ur + 1)%cfg.rx_mod;
|
|
|
|
vr_ur = (vr_ur + 1)%cfg.rx_mod;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Now update vr_ur until we reach an SN we haven't yet received
|
|
|
|
// Now update vr_ur until we reach an SN we haven't yet received
|
|
|
|
while(rx_window.end() != rx_window.find(vr_ur))
|
|
|
|
while(rx_window.end() != rx_window.find(vr_ur)) {
|
|
|
|
{
|
|
|
|
|
|
|
|
// Handle any SDU segments
|
|
|
|
// Handle any SDU segments
|
|
|
|
for(uint32_t i=0; i<rx_window[vr_ur].header.N_li; i++)
|
|
|
|
for(uint32_t i=0; i<rx_window[vr_ur].header.N_li; i++) {
|
|
|
|
{
|
|
|
|
|
|
|
|
int len = rx_window[vr_ur].header.li[i];
|
|
|
|
int len = rx_window[vr_ur].header.li[i];
|
|
|
|
|
|
|
|
|
|
|
|
// Check if the first part of the PDU is a middle or end segment
|
|
|
|
// Check if the first part of the PDU is a middle or end segment
|
|
|
@ -628,7 +785,7 @@ void rlc_um::reassemble_rx_sdus()
|
|
|
|
log->warning("Dropping remainder of lost PDU (update vr_ur middle segments, vr_ur=%d, vr_ur_in_rx_sdu=%d)\n", vr_ur, vr_ur_in_rx_sdu);
|
|
|
|
log->warning("Dropping remainder of lost PDU (update vr_ur middle segments, vr_ur=%d, vr_ur_in_rx_sdu=%d)\n", vr_ur, vr_ur_in_rx_sdu);
|
|
|
|
rx_sdu->reset();
|
|
|
|
rx_sdu->reset();
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
log->info_hex(rx_sdu->msg, rx_sdu->N_bytes, "%s Rx SDU vr_ur=%d, i=%d, (update vr_ur middle segments)", rb_name().c_str(), vr_ur, i);
|
|
|
|
log->info_hex(rx_sdu->msg, rx_sdu->N_bytes, "%s Rx SDU vr_ur=%d, i=%d, (update vr_ur middle segments)", get_rb_name(), vr_ur, i);
|
|
|
|
rx_sdu->set_timestamp();
|
|
|
|
rx_sdu->set_timestamp();
|
|
|
|
if(cfg.is_mrb){
|
|
|
|
if(cfg.is_mrb){
|
|
|
|
pdcp->write_pdu_mch(lcid, rx_sdu);
|
|
|
|
pdcp->write_pdu_mch(lcid, rx_sdu);
|
|
|
@ -665,13 +822,12 @@ void rlc_um::reassemble_rx_sdus()
|
|
|
|
rx_sdu->N_bytes, rx_window[vr_ur].buf->N_bytes, vr_ur);
|
|
|
|
rx_sdu->N_bytes, rx_window[vr_ur].buf->N_bytes, vr_ur);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
vr_ur_in_rx_sdu = vr_ur;
|
|
|
|
vr_ur_in_rx_sdu = vr_ur;
|
|
|
|
if(rlc_um_end_aligned(rx_window[vr_ur].header.fi))
|
|
|
|
if(rlc_um_end_aligned(rx_window[vr_ur].header.fi)) {
|
|
|
|
{
|
|
|
|
|
|
|
|
if(pdu_lost && !rlc_um_start_aligned(rx_window[vr_ur].header.fi)) {
|
|
|
|
if(pdu_lost && !rlc_um_start_aligned(rx_window[vr_ur].header.fi)) {
|
|
|
|
log->warning("Dropping remainder of lost PDU (update vr_ur last segments)\n");
|
|
|
|
log->warning("Dropping remainder of lost PDU (update vr_ur last segments)\n");
|
|
|
|
rx_sdu->reset();
|
|
|
|
rx_sdu->reset();
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
log->info_hex(rx_sdu->msg, rx_sdu->N_bytes, "%s Rx SDU vr_ur=%d (update vr_ur last segments)", rb_name().c_str(), vr_ur);
|
|
|
|
log->info_hex(rx_sdu->msg, rx_sdu->N_bytes, "%s Rx SDU vr_ur=%d (update vr_ur last segments)", get_rb_name(), vr_ur);
|
|
|
|
rx_sdu->set_timestamp();
|
|
|
|
rx_sdu->set_timestamp();
|
|
|
|
if(cfg.is_mrb){
|
|
|
|
if(cfg.is_mrb){
|
|
|
|
pdcp->write_pdu_mch(lcid, rx_sdu);
|
|
|
|
pdcp->write_pdu_mch(lcid, rx_sdu);
|
|
|
@ -687,7 +843,7 @@ void rlc_um::reassemble_rx_sdus()
|
|
|
|
pdu_lost = false;
|
|
|
|
pdu_lost = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
clean_up_rx_window:
|
|
|
|
clean_up_rx_window:
|
|
|
|
|
|
|
|
|
|
|
|
// Clean up rx_window
|
|
|
|
// Clean up rx_window
|
|
|
|
pool->deallocate(rx_window[vr_ur].buf);
|
|
|
|
pool->deallocate(rx_window[vr_ur].buf);
|
|
|
@ -697,7 +853,8 @@ clean_up_rx_window:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool rlc_um::inside_reordering_window(uint16_t sn)
|
|
|
|
// Only called when lock is hold
|
|
|
|
|
|
|
|
bool rlc_um::rlc_um_rx::inside_reordering_window(uint16_t sn)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if(cfg.rx_window_size == 0) {
|
|
|
|
if(cfg.rx_window_size == 0) {
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
@ -711,23 +868,78 @@ bool rlc_um::inside_reordering_window(uint16_t sn)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::debug_state()
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t rlc_um::rlc_um_rx::get_num_rx_bytes()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
log->debug("%s vt_us = %d, vr_ur = %d, vr_ux = %d, vr_uh = %d \n",
|
|
|
|
return num_rx_bytes;
|
|
|
|
rb_name().c_str(), vt_us, vr_ur, vr_ux, vr_uh);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_rx::reset_metrics()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
|
|
|
|
num_rx_bytes = 0;
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::string rlc_um::rb_name() {
|
|
|
|
|
|
|
|
if(cfg.is_mrb) {
|
|
|
|
/****************************************************************************
|
|
|
|
std::stringstream ss;
|
|
|
|
* Timeout callback interface
|
|
|
|
ss << "MRB" << lcid;
|
|
|
|
***************************************************************************/
|
|
|
|
return ss.str();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
void rlc_um::rlc_um_rx::timer_expired(uint32_t timeout_id)
|
|
|
|
return rrc->get_rb_name(lcid);
|
|
|
|
{
|
|
|
|
|
|
|
|
if(reordering_timer_id == timeout_id)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
pthread_mutex_lock(&mutex);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 36.322 v10 Section 5.1.2.2.4
|
|
|
|
|
|
|
|
log->info("%s reordering timeout expiry - updating vr_ur and reassembling\n",
|
|
|
|
|
|
|
|
get_rb_name());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
log->warning("Lost PDU SN: %d\n", vr_ur);
|
|
|
|
|
|
|
|
pdu_lost = true;
|
|
|
|
|
|
|
|
rx_sdu->reset();
|
|
|
|
|
|
|
|
while(RX_MOD_BASE(vr_ur) < RX_MOD_BASE(vr_ux))
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
vr_ur = (vr_ur + 1)%cfg.rx_mod;
|
|
|
|
|
|
|
|
log->debug("Entering Reassemble from timeout id=%d\n", timeout_id);
|
|
|
|
|
|
|
|
reassemble_rx_sdus();
|
|
|
|
|
|
|
|
log->debug("Finished reassemble from timeout id=%d\n", timeout_id);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
reordering_timer->stop();
|
|
|
|
|
|
|
|
if(RX_MOD_BASE(vr_uh) > RX_MOD_BASE(vr_ur))
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
reordering_timer->set(this, cfg.t_reordering);
|
|
|
|
|
|
|
|
reordering_timer->run();
|
|
|
|
|
|
|
|
vr_ux = vr_uh;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
debug_state();
|
|
|
|
|
|
|
|
pthread_mutex_unlock(&mutex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool rlc_um::rlc_um_rx::reordering_timeout_running()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
return reordering_timer->is_running();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
|
|
|
|
* Helper functions
|
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void rlc_um::rlc_um_rx::debug_state()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
log->debug("%s vr_ur = %d, vr_ux = %d, vr_uh = %d\n", get_rb_name(), vr_ur, vr_ux, vr_uh);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const char* rlc_um::rlc_um_rx::get_rb_name()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
return rb_name.c_str();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
/****************************************************************************
|
|
|
|
* Header pack/unpack helper functions
|
|
|
|
* Header pack/unpack helper functions
|
|
|
|
* Ref: 3GPP TS 36.322 v10.0.0 Section 6.2.1
|
|
|
|
* Ref: 3GPP TS 36.322 v10.0.0 Section 6.2.1
|
|
|
|