|
|
@ -205,14 +205,16 @@ private:
|
|
|
|
*ack = false;
|
|
|
|
*ack = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
set_harq_feedback(*ack);
|
|
|
|
harq_feedback = *ack;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t max_retx;
|
|
|
|
// Reset HARQ process if TB has changed
|
|
|
|
if (is_msg3) {
|
|
|
|
if (harq_feedback && has_grant() && grant) {
|
|
|
|
max_retx = harq_entity->params->max_harq_msg3_tx;
|
|
|
|
if (grant->n_bytes[0] != cur_grant.n_bytes[0] && cur_grant.n_bytes[0] > 0) {
|
|
|
|
} else {
|
|
|
|
Debug("UL %d: Reset due to change of grant size last_grant=%d, new_grant=%d\n",
|
|
|
|
max_retx = harq_entity->params->max_harq_tx;
|
|
|
|
pid, cur_grant.n_bytes[0], grant->n_bytes[0]);
|
|
|
|
|
|
|
|
reset();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Receive and route HARQ feedbacks
|
|
|
|
// Receive and route HARQ feedbacks
|
|
|
@ -222,6 +224,7 @@ private:
|
|
|
|
grant->is_from_rar)
|
|
|
|
grant->is_from_rar)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// New transmission
|
|
|
|
// New transmission
|
|
|
|
|
|
|
|
reset();
|
|
|
|
|
|
|
|
|
|
|
|
// Uplink grant in a RAR
|
|
|
|
// Uplink grant in a RAR
|
|
|
|
if (grant->is_from_rar) {
|
|
|
|
if (grant->is_from_rar) {
|
|
|
@ -245,26 +248,14 @@ private:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (has_grant()) {
|
|
|
|
} else if (has_grant()) {
|
|
|
|
// Adaptive Re-TX
|
|
|
|
// Adaptive Re-TX
|
|
|
|
if (current_tx_nb >= max_retx) {
|
|
|
|
|
|
|
|
Info("UL %d: Maximum number of ReTX reached (%d). Discarting TB.\n", pid, max_retx);
|
|
|
|
|
|
|
|
reset();
|
|
|
|
|
|
|
|
action->expect_ack = false;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
generate_retx(tti_tx, grant, action);
|
|
|
|
generate_retx(tti_tx, grant, action);
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
Warning("UL %d: Received retransmission but no previous grant available for this PID.\n", pid);
|
|
|
|
Warning("UL %d: Received retransmission but no previous grant available for this PID.\n", pid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (has_grant()) {
|
|
|
|
} else if (has_grant()) {
|
|
|
|
// Non-Adaptive Re-Tx
|
|
|
|
// Non-Adaptive Re-Tx
|
|
|
|
if (current_tx_nb >= max_retx) {
|
|
|
|
|
|
|
|
Info("UL %d: Maximum number of ReTX reached (%d). Discarting TB.\n", pid, max_retx);
|
|
|
|
|
|
|
|
reset();
|
|
|
|
|
|
|
|
action->expect_ack = false;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
generate_retx(tti_tx, action);
|
|
|
|
generate_retx(tti_tx, action);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (harq_entity->pcap && grant) {
|
|
|
|
if (harq_entity->pcap && grant) {
|
|
|
|
if (grant->is_from_rar) {
|
|
|
|
if (grant->is_from_rar) {
|
|
|
|
grant->rnti = harq_entity->rntis->temp_rnti;
|
|
|
|
grant->rnti = harq_entity->rntis->temp_rnti;
|
|
|
@ -273,18 +264,6 @@ private:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void set_harq_feedback(bool ack)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
harq_feedback = ack;
|
|
|
|
|
|
|
|
// UL packet successfully delivered
|
|
|
|
|
|
|
|
if (ack) {
|
|
|
|
|
|
|
|
Info("UL %d: HARQ = ACK for UL transmission. Discarting TB.\n", pid);
|
|
|
|
|
|
|
|
reset();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
Info("UL %d: HARQ = NACK for UL transmission\n", pid);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t get_rv()
|
|
|
|
uint32_t get_rv()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
int rv_of_irv[4] = {0, 2, 3, 1};
|
|
|
|
int rv_of_irv[4] = {0, 2, 3, 1};
|
|
|
@ -326,25 +305,44 @@ private:
|
|
|
|
void generate_retx(uint32_t tti_tx, Tgrant *grant,
|
|
|
|
void generate_retx(uint32_t tti_tx, Tgrant *grant,
|
|
|
|
Taction *action)
|
|
|
|
Taction *action)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
uint32_t max_retx;
|
|
|
|
|
|
|
|
if (is_msg3) {
|
|
|
|
|
|
|
|
max_retx = harq_entity->params->max_harq_msg3_tx;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
max_retx = harq_entity->params->max_harq_tx;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (current_tx_nb >= max_retx) {
|
|
|
|
|
|
|
|
Info("UL %d: Maximum number of ReTX reached (%d). Discarting TB.\n", pid, max_retx);
|
|
|
|
|
|
|
|
reset();
|
|
|
|
|
|
|
|
action->expect_ack = false;
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int irv_of_rv[4] = {0, 3, 1, 2};
|
|
|
|
int irv_of_rv[4] = {0, 3, 1, 2};
|
|
|
|
if (grant) {
|
|
|
|
|
|
|
|
// HARQ entity requests an adaptive transmission
|
|
|
|
// HARQ entity requests an adaptive transmission
|
|
|
|
|
|
|
|
if (grant) {
|
|
|
|
if (grant->rv) {
|
|
|
|
if (grant->rv) {
|
|
|
|
current_irv = irv_of_rv[grant->rv[0]%4];
|
|
|
|
current_irv = irv_of_rv[grant->rv[0]%4];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
memcpy(&cur_grant, grant, sizeof(Tgrant));
|
|
|
|
memcpy(&cur_grant, grant, sizeof(Tgrant));
|
|
|
|
harq_feedback = false;
|
|
|
|
harq_feedback = false;
|
|
|
|
Info("UL %d: Adaptive retx=%d, RV=%d, TBS=%d\n",
|
|
|
|
|
|
|
|
pid, current_tx_nb, get_rv(), grant->n_bytes[0]);
|
|
|
|
Info("UL %d: Adaptive retx=%d, RV=%d, TBS=%d, HI=%s\n",
|
|
|
|
|
|
|
|
pid, current_tx_nb, get_rv(), grant->n_bytes[0], harq_feedback?"ACK":"NACK");
|
|
|
|
|
|
|
|
|
|
|
|
generate_tx(tti_tx, action);
|
|
|
|
generate_tx(tti_tx, action);
|
|
|
|
} else {
|
|
|
|
|
|
|
|
Info("UL %d: Non-Adaptive retx=%d, RV=%d, TBS=%d\n",
|
|
|
|
|
|
|
|
pid, current_tx_nb, get_rv(), cur_grant.n_bytes[0]);
|
|
|
|
|
|
|
|
// HARQ entity requests a non-adaptive transmission
|
|
|
|
// HARQ entity requests a non-adaptive transmission
|
|
|
|
if (!harq_feedback) {
|
|
|
|
} else if (!harq_feedback) {
|
|
|
|
|
|
|
|
// Non-adaptive retx are only sent if HI=NACK. If HI=ACK but no grant was received do not reset PID
|
|
|
|
|
|
|
|
Info("UL %d: Non-Adaptive retx=%d, RV=%d, TBS=%d, HI=%s\n",
|
|
|
|
|
|
|
|
pid, current_tx_nb, get_rv(), cur_grant.n_bytes[0], harq_feedback?"ACK":"NACK");
|
|
|
|
|
|
|
|
|
|
|
|
generate_tx(tti_tx, action);
|
|
|
|
generate_tx(tti_tx, action);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// On every Msg3 retransmission, restart mac-ContentionResolutionTimer as defined in Section 5.1.5
|
|
|
|
// On every Msg3 retransmission, restart mac-ContentionResolutionTimer as defined in Section 5.1.5
|
|
|
|
if (is_msg3) {
|
|
|
|
if (is_msg3) {
|
|
|
|