|
|
|
/**
|
|
|
|
*
|
|
|
|
* \section COPYRIGHT
|
|
|
|
*
|
|
|
|
* Copyright 2013-2020 Software Radio Systems Limited
|
|
|
|
*
|
|
|
|
* By using this file, you agree to the terms and conditions set
|
|
|
|
* forth in the LICENSE file which can be found at the top level of
|
|
|
|
* the distribution.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef SRSLTE_PDCP_ENTITY_NR_H
|
|
|
|
#define SRSLTE_PDCP_ENTITY_NR_H
|
|
|
|
|
|
|
|
#include "pdcp_entity_base.h"
|
|
|
|
#include "srslte/common/buffer_pool.h"
|
|
|
|
#include "srslte/common/common.h"
|
|
|
|
#include "srslte/common/interfaces_common.h"
|
|
|
|
#include "srslte/common/log.h"
|
|
|
|
#include "srslte/common/security.h"
|
|
|
|
#include "srslte/common/task_scheduler.h"
|
|
|
|
#include "srslte/common/threads.h"
|
|
|
|
#include "srslte/interfaces/ue_gw_interfaces.h"
|
|
|
|
#include "srslte/interfaces/ue_interfaces.h"
|
|
|
|
#include "srslte/interfaces/ue_rlc_interfaces.h"
|
|
|
|
#include <map>
|
|
|
|
|
|
|
|
namespace srslte {
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* NR PDCP Entity
|
|
|
|
* PDCP entity for 5G NR
|
|
|
|
***************************************************************************/
|
|
|
|
class pdcp_entity_nr final : public pdcp_entity_base
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
pdcp_entity_nr(srsue::rlc_interface_pdcp* rlc_,
|
|
|
|
srsue::rrc_interface_pdcp* rrc_,
|
|
|
|
srsue::gw_interface_pdcp* gw_,
|
|
|
|
srslte::task_sched_handle task_sched_,
|
|
|
|
srslog::basic_logger& logger,
|
|
|
|
uint32_t lcid,
|
|
|
|
pdcp_config_t cfg_);
|
|
|
|
~pdcp_entity_nr() final;
|
|
|
|
void reset() final;
|
|
|
|
void reestablish() final;
|
|
|
|
|
|
|
|
// RRC interface
|
|
|
|
void write_sdu(unique_byte_buffer_t sdu, int sn = -1) final;
|
|
|
|
|
|
|
|
// RLC interface
|
|
|
|
void write_pdu(unique_byte_buffer_t pdu) final;
|
|
|
|
void notify_delivery(const pdcp_sn_vector_t& pdcp_sns) final;
|
|
|
|
void notify_failure(const pdcp_sn_vector_t& pdcp_sns) final;
|
|
|
|
|
|
|
|
// State variable setters (should be used only for testing)
|
|
|
|
void set_tx_next(uint32_t tx_next_) { tx_next = tx_next_; }
|
|
|
|
void set_rx_next(uint32_t rx_next_) { rx_next = rx_next_; }
|
|
|
|
void set_rx_deliv(uint32_t rx_deliv_) { rx_deliv = rx_deliv_; }
|
|
|
|
void set_rx_reord(uint32_t rx_reord_) { rx_reord = rx_reord_; }
|
|
|
|
|
|
|
|
void get_bearer_state(pdcp_lte_state_t* state) override;
|
|
|
|
void set_bearer_state(const pdcp_lte_state_t& state) override;
|
|
|
|
|
|
|
|
void send_status_report() override {}
|
|
|
|
pdcp_bearer_metrics_t get_metrics() override;
|
|
|
|
void reset_metrics() override;
|
|
|
|
|
|
|
|
std::map<uint32_t, srslte::unique_byte_buffer_t> get_buffered_pdus() override { return {}; }
|
|
|
|
|
|
|
|
// State variable getters (useful for testing)
|
|
|
|
uint32_t nof_discard_timers() { return discard_timers_map.size(); }
|
|
|
|
|
|
|
|
private:
|
|
|
|
srsue::rlc_interface_pdcp* rlc = nullptr;
|
|
|
|
srsue::rrc_interface_pdcp* rrc = nullptr;
|
|
|
|
srsue::gw_interface_pdcp* gw = nullptr;
|
|
|
|
|
|
|
|
// State variables: 3GPP TS 38.323 v15.2.0, section 7.1
|
|
|
|
uint32_t tx_next = 0; // COUNT value of next SDU to be transmitted.
|
|
|
|
uint32_t rx_next = 0; // COUNT value of next SDU expected to be received.
|
|
|
|
uint32_t rx_deliv = 0; // COUNT value of first SDU not delivered to upper layers, but still waited for.
|
|
|
|
uint32_t rx_reord = 0; // COUNT value following the COUNT value of PDCP Data PDU which triggered t-Reordering.
|
|
|
|
|
|
|
|
// Constants: 3GPP TS 38.323 v15.2.0, section 7.2
|
|
|
|
uint32_t window_size = 0;
|
|
|
|
|
|
|
|
// Reordering Queue / Timers
|
|
|
|
std::map<uint32_t, unique_byte_buffer_t> reorder_queue;
|
|
|
|
timer_handler::unique_timer reordering_timer;
|
|
|
|
|
|
|
|
// Pass to Upper Layers Helper function
|
|
|
|
void deliver_all_consecutive_counts();
|
|
|
|
void pass_to_upper_layers(unique_byte_buffer_t pdu);
|
|
|
|
|
|
|
|
// Reodering callback (t-Reordering)
|
|
|
|
class reordering_callback;
|
|
|
|
std::unique_ptr<reordering_callback> reordering_fnc;
|
|
|
|
|
|
|
|
// Discard callback (discardTimer)
|
|
|
|
class discard_callback;
|
|
|
|
std::map<uint32_t, timer_handler::unique_timer> discard_timers_map;
|
|
|
|
|
|
|
|
// COUNT overflow protection
|
|
|
|
bool tx_overflow = false;
|
|
|
|
bool rx_overflow = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Timer callbacks
|
|
|
|
*/
|
|
|
|
// Reordering callback (t-Reordering)
|
|
|
|
class pdcp_entity_nr::reordering_callback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
reordering_callback(pdcp_entity_nr* parent_) { parent = parent_; };
|
|
|
|
void operator()(uint32_t timer_id);
|
|
|
|
|
|
|
|
private:
|
|
|
|
pdcp_entity_nr* parent;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Discard callback (discardTimer)
|
|
|
|
class pdcp_entity_nr::discard_callback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
discard_callback(pdcp_entity_nr* parent_, uint32_t sn_)
|
|
|
|
{
|
|
|
|
parent = parent_;
|
|
|
|
discard_sn = sn_;
|
|
|
|
};
|
|
|
|
void operator()(uint32_t timer_id);
|
|
|
|
|
|
|
|
private:
|
|
|
|
pdcp_entity_nr* parent;
|
|
|
|
uint32_t discard_sn;
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Helpers
|
|
|
|
*/
|
|
|
|
inline void pdcp_entity_nr::pass_to_upper_layers(unique_byte_buffer_t sdu)
|
|
|
|
{
|
|
|
|
if (is_srb()) {
|
|
|
|
rrc->write_pdu(lcid, std::move(sdu));
|
|
|
|
} else {
|
|
|
|
gw->write_pdu(lcid, std::move(sdu));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace srslte
|
|
|
|
#endif // SRSLTE_PDCP_ENTITY_NR_H
|