|
|
@ -606,9 +606,7 @@ static uint32_t encode_ri_ack(uint8_t data[2], uint32_t data_len, srslte_uci_bit
|
|
|
|
/* Decode UCI HARQ/ACK bits as described in 5.2.2.6 of 36.212
|
|
|
|
/* Decode UCI HARQ/ACK bits as described in 5.2.2.6 of 36.212
|
|
|
|
* Currently only supporting 1-bit HARQ
|
|
|
|
* Currently only supporting 1-bit HARQ
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
#ifndef MIMO_ENB
|
|
|
|
static int32_t decode_ri_ack_1bit(int16_t *q_bits, uint8_t *c_seq, srslte_uci_bit_t *pos)
|
|
|
|
|
|
|
|
|
|
|
|
static int32_t decode_ri_ack(int16_t *q_bits, uint8_t *c_seq, srslte_uci_bit_t *pos)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
uint32_t p0 = pos[0].position;
|
|
|
|
uint32_t p0 = pos[0].position;
|
|
|
|
uint32_t p1 = pos[1].position;
|
|
|
|
uint32_t p1 = pos[1].position;
|
|
|
@ -618,33 +616,8 @@ static int32_t decode_ri_ack(int16_t *q_bits, uint8_t *c_seq, srslte_uci_bit_t *
|
|
|
|
|
|
|
|
|
|
|
|
return -(q0+q1);
|
|
|
|
return -(q0+q1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int srslte_uci_decode_ack(srslte_pusch_cfg_t *cfg, int16_t *q_bits, uint8_t *c_seq,
|
|
|
|
|
|
|
|
float beta, uint32_t H_prime_total,
|
|
|
|
|
|
|
|
uint32_t O_cqi, srslte_uci_bit_t *ack_bits, uint8_t acks[2], uint32_t nof_acks)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
int32_t rx_ack = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (beta < 0) {
|
|
|
|
|
|
|
|
fprintf(stderr, "Error beta is reserved\n");
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t Qprime = Q_prime_ri_ack(cfg, 1, O_cqi, beta);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Use the same interleaver function to get the HARQ bit position
|
|
|
|
|
|
|
|
for (uint32_t i=0;i<Qprime;i++) {
|
|
|
|
|
|
|
|
uci_ulsch_interleave_ack_gen(i, cfg->grant.Qm, H_prime_total, cfg->nbits.nof_symb, cfg->cp, &ack_bits[cfg->grant.Qm*i]);
|
|
|
|
|
|
|
|
rx_ack += (int32_t) decode_ri_ack(q_bits, c_seq, &ack_bits[cfg->grant.Qm*i]);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (acks) {
|
|
|
|
|
|
|
|
acks[0] = rx_ack>0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return (int) Qprime;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void decode_ri_ack(int16_t *q_bits, uint8_t *c_seq, srslte_uci_bit_t *pos, uint32_t Qm, int32_t data[3])
|
|
|
|
static void decode_ri_ack_2bits(int16_t *q_bits, uint8_t *c_seq, srslte_uci_bit_t *pos, uint32_t Qm, int32_t data[3])
|
|
|
|
{
|
|
|
|
{
|
|
|
|
uint32_t p0 = pos[Qm * 0 + 0].position;
|
|
|
|
uint32_t p0 = pos[Qm * 0 + 0].position;
|
|
|
|
uint32_t p1 = pos[Qm * 0 + 1].position;
|
|
|
|
uint32_t p1 = pos[Qm * 0 + 1].position;
|
|
|
@ -665,120 +638,91 @@ static void decode_ri_ack(int16_t *q_bits, uint8_t *c_seq, srslte_uci_bit_t *pos
|
|
|
|
data[2] -= q2 + q5;
|
|
|
|
data[2] -= q2 + q5;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int srslte_uci_decode_ack(srslte_pusch_cfg_t *cfg, int16_t *q_bits, uint8_t *c_seq,
|
|
|
|
/* Encode UCI ACK/RI bits as described in 5.2.2.6 of 36.212
|
|
|
|
float beta, uint32_t H_prime_total,
|
|
|
|
* Currently only supporting 1-bit RI
|
|
|
|
uint32_t O_cqi, srslte_uci_bit_t *ack_bits, uint8_t acks[2], uint32_t nof_acks)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
int32_t acks_sum[3] = {0, 0, 0};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (beta < 0) {
|
|
|
|
|
|
|
|
fprintf(stderr, "Error beta is reserved\n");
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t Qprime = Q_prime_ri_ack(cfg, nof_acks, O_cqi, beta);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Use the same interleaver function to get the HARQ bit position
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < Qprime; i++) {
|
|
|
|
|
|
|
|
uci_ulsch_interleave_ack_gen(i, cfg->grant.Qm, H_prime_total, cfg->nbits.nof_symb, cfg->cp, &ack_bits[cfg->grant.Qm*i]);
|
|
|
|
|
|
|
|
if ((i % 3 == 0) && i > 0) {
|
|
|
|
|
|
|
|
decode_ri_ack(q_bits, &c_seq[0], &ack_bits[cfg->grant.Qm*(i-3)], cfg->grant.Qm, acks_sum);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (nof_acks == 1 && acks) {
|
|
|
|
|
|
|
|
acks[0] = (uint8_t)(acks_sum[0] + acks_sum[1] + acks_sum[2] > 0);
|
|
|
|
|
|
|
|
} else if (acks) {
|
|
|
|
|
|
|
|
acks[0] = (uint8_t)(acks_sum[0] > 0);
|
|
|
|
|
|
|
|
acks[1] = (uint8_t)(acks_sum[1] > 0);
|
|
|
|
|
|
|
|
// TODO: Do something with acks_sum[2]
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return (int) Qprime;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Encode UCI HARQ/ACK bits as described in 5.2.2.6 of 36.212
|
|
|
|
|
|
|
|
* Currently only supporting 1-bit HARQ
|
|
|
|
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
int srslte_uci_encode_ack(srslte_pusch_cfg_t *cfg, uint8_t acks[2], uint32_t nof_acks,
|
|
|
|
int srslte_uci_encode_ack_ri(srslte_pusch_cfg_t *cfg,
|
|
|
|
uint32_t O_cqi, float beta, uint32_t H_prime_total,
|
|
|
|
uint8_t *data, uint32_t data_len,
|
|
|
|
srslte_uci_bit_t *ack_bits)
|
|
|
|
uint32_t O_cqi, float beta, uint32_t H_prime_total,
|
|
|
|
{
|
|
|
|
srslte_uci_bit_t *bits, bool ack_ri) {
|
|
|
|
if (beta < 0) {
|
|
|
|
if (beta < 0) {
|
|
|
|
fprintf(stderr, "Error beta is reserved\n");
|
|
|
|
fprintf(stderr, "Error beta is reserved\n");
|
|
|
|
return -1;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t Qprime = Q_prime_ri_ack(cfg, data_len, O_cqi, beta);
|
|
|
|
uint32_t Qprime = Q_prime_ri_ack(cfg, nof_acks, O_cqi, beta);
|
|
|
|
|
|
|
|
srslte_uci_bit_type_t q_encoded_bits[18];
|
|
|
|
srslte_uci_bit_type_t q_encoded_bits[18];
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t nof_encoded_bits = encode_ri_ack(acks, nof_acks, q_encoded_bits, cfg->grant.Qm);
|
|
|
|
uint32_t nof_encoded_bits = encode_ri_ack(data, data_len, q_encoded_bits, cfg->grant.Qm);
|
|
|
|
|
|
|
|
|
|
|
|
for (uint32_t i=0;i<Qprime;i++) {
|
|
|
|
for (uint32_t i = 0; i < Qprime; i++) {
|
|
|
|
uci_ulsch_interleave_ack_gen(i, cfg->grant.Qm, H_prime_total, cfg->nbits.nof_symb, cfg->cp, &ack_bits[cfg->grant.Qm*i]);
|
|
|
|
if (ack_ri) {
|
|
|
|
uci_ulsch_interleave_put(&q_encoded_bits[(i*cfg->grant.Qm)%nof_encoded_bits], cfg->grant.Qm, &ack_bits[cfg->grant.Qm*i]);
|
|
|
|
uci_ulsch_interleave_ri_gen(i,
|
|
|
|
|
|
|
|
cfg->grant.Qm,
|
|
|
|
|
|
|
|
H_prime_total,
|
|
|
|
|
|
|
|
cfg->nbits.nof_symb,
|
|
|
|
|
|
|
|
cfg->cp,
|
|
|
|
|
|
|
|
&bits[cfg->grant.Qm * i]);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
uci_ulsch_interleave_ack_gen(i,
|
|
|
|
|
|
|
|
cfg->grant.Qm,
|
|
|
|
|
|
|
|
H_prime_total,
|
|
|
|
|
|
|
|
cfg->nbits.nof_symb,
|
|
|
|
|
|
|
|
cfg->cp,
|
|
|
|
|
|
|
|
&bits[cfg->grant.Qm * i]);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
uci_ulsch_interleave_put(&q_encoded_bits[(i * cfg->grant.Qm) % nof_encoded_bits],
|
|
|
|
|
|
|
|
cfg->grant.Qm,
|
|
|
|
|
|
|
|
&bits[cfg->grant.Qm * i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return (int) Qprime;
|
|
|
|
return (int) Qprime;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Encode UCI RI bits as described in 5.2.2.6 of 36.212
|
|
|
|
/* Decode UCI ACK/RI bits as described in 5.2.2.6 of 36.212
|
|
|
|
* Currently only supporting 1-bit RI
|
|
|
|
* Currently only supporting 1-bit RI
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
int srslte_uci_decode_ri(srslte_pusch_cfg_t *cfg, int16_t *q_bits, uint8_t *c_seq,
|
|
|
|
int srslte_uci_decode_ack_ri(srslte_pusch_cfg_t *cfg, int16_t *q_bits, uint8_t *c_seq,
|
|
|
|
float beta, uint32_t H_prime_total,
|
|
|
|
float beta, uint32_t H_prime_total,
|
|
|
|
uint32_t O_cqi, srslte_uci_bit_t *ri_bits, uint8_t *data)
|
|
|
|
uint32_t O_cqi, srslte_uci_bit_t *ack_ri_bits, uint8_t data[2], uint32_t nof_bits, bool is_ri)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
int32_t ri_sum[3] = {0, 0, 0};
|
|
|
|
int32_t sum[3] = {0, 0, 0};
|
|
|
|
|
|
|
|
|
|
|
|
if (beta < 0) {
|
|
|
|
if (beta < 0) {
|
|
|
|
fprintf(stderr, "Error beta is reserved\n");
|
|
|
|
fprintf(stderr, "Error beta is reserved\n");
|
|
|
|
return -1;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t Qprime = Q_prime_ri_ack(cfg, 1, O_cqi, beta);
|
|
|
|
uint32_t Qprime = Q_prime_ri_ack(cfg, nof_bits, O_cqi, beta);
|
|
|
|
|
|
|
|
|
|
|
|
// Use the same interleaver function to get the HARQ bit position
|
|
|
|
|
|
|
|
for (uint32_t i=0;i<Qprime;i++) {
|
|
|
|
|
|
|
|
uci_ulsch_interleave_ri_gen(i, cfg->grant.Qm, H_prime_total, cfg->nbits.nof_symb, cfg->cp, &ri_bits[cfg->grant.Qm*i]);
|
|
|
|
|
|
|
|
if ((i % 3 == 0) && i > 0) {
|
|
|
|
|
|
|
|
//decode_ri_ack(q_bits, &c_seq[0], &ri_bits[cfg->grant.Qm*(i-3)], cfg->grant.Qm, ri_sum);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (data) {
|
|
|
|
|
|
|
|
*data = (uint8_t) ((ri_sum[0] + ri_sum[1] + ri_sum[2]) > 0);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return (int) Qprime;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < Qprime; i++) {
|
|
|
|
|
|
|
|
if (is_ri) {
|
|
|
|
|
|
|
|
uci_ulsch_interleave_ri_gen(i,
|
|
|
|
|
|
|
|
cfg->grant.Qm,
|
|
|
|
|
|
|
|
H_prime_total,
|
|
|
|
|
|
|
|
cfg->nbits.nof_symb,
|
|
|
|
|
|
|
|
cfg->cp,
|
|
|
|
|
|
|
|
&ack_ri_bits[cfg->grant.Qm * i]);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
uci_ulsch_interleave_ack_gen(i,
|
|
|
|
|
|
|
|
cfg->grant.Qm,
|
|
|
|
|
|
|
|
H_prime_total,
|
|
|
|
|
|
|
|
cfg->nbits.nof_symb,
|
|
|
|
|
|
|
|
cfg->cp,
|
|
|
|
|
|
|
|
&ack_ri_bits[cfg->grant.Qm * i]);
|
|
|
|
|
|
|
|
|
|
|
|
/* Encode UCI RI bits as described in 5.2.2.6 of 36.212
|
|
|
|
}
|
|
|
|
* Currently only supporting 1-bit RI
|
|
|
|
if (nof_bits == 2 && (i % 3 == 0) && i > 0) {
|
|
|
|
*/
|
|
|
|
decode_ri_ack_2bits(q_bits, &c_seq[0], &ack_ri_bits[cfg->grant.Qm * (i - 3)], cfg->grant.Qm, sum);
|
|
|
|
int srslte_uci_encode_ri(srslte_pusch_cfg_t *cfg,
|
|
|
|
} else if (nof_bits == 1) {
|
|
|
|
uint8_t ri,
|
|
|
|
sum[0] += (int32_t) decode_ri_ack_1bit(q_bits, c_seq, &ack_ri_bits[cfg->grant.Qm * i]);
|
|
|
|
uint32_t O_cqi, float beta, uint32_t H_prime_total,
|
|
|
|
}
|
|
|
|
srslte_uci_bit_t *ri_bits)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
// FIXME: It supports RI of 1 bit only
|
|
|
|
|
|
|
|
uint8_t data[2] = {ri, 0};
|
|
|
|
|
|
|
|
if (beta < 0) {
|
|
|
|
|
|
|
|
fprintf(stderr, "Error beta is reserved\n");
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
uint32_t Qprime = Q_prime_ri_ack(cfg, 1, O_cqi, beta);
|
|
|
|
|
|
|
|
srslte_uci_bit_type_t q_encoded_bits[18];
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t nof_encoded_bits = encode_ri_ack(data, 1, q_encoded_bits, cfg->grant.Qm);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for (uint32_t i=0;i<Qprime;i++) {
|
|
|
|
data[0] = (uint8_t) (sum[0] > 0);
|
|
|
|
uci_ulsch_interleave_ri_gen(i, cfg->grant.Qm, H_prime_total, cfg->nbits.nof_symb, cfg->cp, &ri_bits[cfg->grant.Qm*i]);
|
|
|
|
if (nof_bits == 2) {
|
|
|
|
uci_ulsch_interleave_put(&q_encoded_bits[(i*cfg->grant.Qm)%nof_encoded_bits], cfg->grant.Qm, &ri_bits[cfg->grant.Qm*i]);
|
|
|
|
data[1] = (uint8_t) (sum[1] > 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return (int) Qprime;
|
|
|
|
return (int) Qprime;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|