nr_ulsch_ue.c 16.2 KB
Newer Older
Khalid Ahmed's avatar
Khalid Ahmed committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32
/*
 * Licensed to the OpenAirInterface (OAI) Software Alliance under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The OpenAirInterface Software Alliance licenses this file to You under
 * the OAI Public License, Version 1.1  (the "License"); you may not use this file
 * except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.openairinterface.org/?page_id=698
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 *-------------------------------------------------------------------------------
 * For more information about the OpenAirInterface (OAI) Software Alliance:
 *      contact@openairinterface.org
 */

/*! \file PHY/NR_UE_TRANSPORT/nr_ulsch.c
* \brief Top-level routines for transmission of the PUSCH TS 38.211 v 15.4.0
* \author Khalid Ahmed
* \date 2019
* \version 0.1
* \company Fraunhofer IIS
* \email: khalid.ahmed@iis.fraunhofer.de
* \note
* \warning
*/
#include <stdint.h>
33
#include "PHY/NR_REFSIG/dmrs_nr.h"
adk's avatar
adk committed
34
#include "PHY/NR_REFSIG/ptrs_nr.h"
35
#include "PHY/NR_UE_TRANSPORT/nr_transport_ue.h"
36 37
#include "PHY/NR_UE_TRANSPORT/nr_transport_proto_ue.h"
#include "PHY/MODULATION/nr_modulation.h"
38
#include "PHY/MODULATION/modulation_common.h"
Khalid Ahmed's avatar
Khalid Ahmed committed
39
#include "common/utils/assertions.h"
40
#include "common/utils/LOG/vcd_signal_dumper.h"
Khalid Ahmed's avatar
Khalid Ahmed committed
41
#include "PHY/NR_TRANSPORT/nr_transport_common_proto.h"
42
#include "PHY/NR_TRANSPORT/nr_sch_dmrs.h"
Khalid Ahmed's avatar
Khalid Ahmed committed
43
#include "PHY/defs_nr_common.h"
44
#include "PHY/TOOLS/tools_defs.h"
45
#include "executables/nr-softmodem.h"
46
#include "executables/softmodem-common.h"
matzakos's avatar
matzakos committed
47
#include "LAYER2/NR_MAC_UE/mac_proto.h"
48 49

//#define DEBUG_SCFDMA
50
//#define DEBUG_PUSCH_MAPPING
51
//#define DEBUG_MAC_PDU
52

53
//extern int32_t uplink_counter;
54

Khalid Ahmed's avatar
Khalid Ahmed committed
55
void nr_pusch_codeword_scrambling(uint8_t *in,
56
                         uint32_t size,
Khalid Ahmed's avatar
Khalid Ahmed committed
57 58 59 60 61
                         uint32_t Nid,
                         uint32_t n_RNTI,
                         uint32_t* out) {

  uint8_t reset, b_idx;
Khalid Ahmed's avatar
Khalid Ahmed committed
62
  uint32_t x1, x2, s=0, temp_out;
Khalid Ahmed's avatar
Khalid Ahmed committed
63 64 65 66 67 68 69 70 71 72 73 74 75 76

  reset = 1;
  x2 = (n_RNTI<<15) + Nid;

  for (int i=0; i<size; i++) {
    b_idx = i&0x1f;
    if (b_idx==0) {
      s = lte_gold_generic(&x1, &x2, reset);
      reset = 0;
      if (i)
        out++;
    }
    if (in[i]==NR_PUSCH_x)
      *out ^= 1<<b_idx;
Khalid Ahmed's avatar
Khalid Ahmed committed
77 78 79 80 81 82 83 84 85 86
    else if (in[i]==NR_PUSCH_y){
      if (b_idx!=0)
        *out ^= (*out & (1<<(b_idx-1)))<<1;
      else{

        temp_out = *(out-1);
        *out ^= temp_out>>31;

      }
    }
Khalid Ahmed's avatar
Khalid Ahmed committed
87 88 89 90 91
    else
      *out ^= (((in[i])&1) ^ ((s>>b_idx)&1))<<b_idx;
    //printf("i %d b_idx %d in %d s 0x%08x out 0x%08x\n", i, b_idx, in[i], s, *out);
  }

92
}
93

94
void nr_ue_ulsch_procedures(PHY_VARS_NR_UE *UE,
95
                               unsigned char harq_pid,
96
                               uint32_t frame,
97 98
                               uint8_t slot,
                               uint8_t thread_id,
99
                               int gNB_id) {
100

101 102 103
  LOG_D(PHY,"nr_ue_ulsch_procedures hard_id %d %d.%d\n",harq_pid,frame,slot);

  uint32_t available_bits;
104
  uint8_t mod_order, cwd_index, num_of_codewords, l;
105 106 107 108 109
  uint32_t scrambled_output[NR_MAX_NB_CODEWORDS][NR_MAX_PDSCH_ENCODED_LENGTH>>5];
  uint32_t ***pusch_dmrs;
  int16_t **tx_layers;
  int32_t **txdataF;
  uint16_t start_sc, start_rb;
110
  int8_t Wf[2], Wt[2], l_prime[2], delta;
111
  uint16_t rnti, n_dmrs, code_rate, number_dmrs_symbols, nb_rb, k;
112
  uint8_t dmrs_type, nb_dmrs_re_per_rb, number_of_symbols, mcs, Nl;
113 114
  int ap, start_symbol, Nid_cell, i;
  int sample_offsetF, N_RE_prime, N_PRB_oh;
115
  uint16_t ul_dmrs_symb_pos;
adk's avatar
adk committed
116 117
  uint8_t L_ptrs, K_ptrs; // PTRS parameters
  uint16_t beta_ptrs; // PTRS parameter related to power control
118 119

  NR_UE_ULSCH_t *ulsch_ue;
laurent's avatar
laurent committed
120
  NR_UL_UE_HARQ_t *harq_process_ul_ue=NULL;
121
  NR_DL_FRAME_PARMS *frame_parms = &UE->frame_parms;
122
  NR_UE_PUSCH *pusch_ue = UE->pusch_vars[thread_id][gNB_id];
123
  // ptrs_UplinkConfig_t *ptrs_Uplink_Config = &UE->pusch_config.dmrs_UplinkConfig.ptrs_UplinkConfig;
124 125

  num_of_codewords = 1; // tmp assumption
126 127
  Nid_cell = 0;
  N_PRB_oh = 0; // higher layer (RRC) parameter xOverhead in PUSCH-ServingCellConfig
128 129
  number_dmrs_symbols = 0;

130 131
  for (cwd_index = 0;cwd_index < num_of_codewords; cwd_index++) {

132
    ulsch_ue = UE->ulsch[thread_id][gNB_id][cwd_index];
133 134
    harq_process_ul_ue = ulsch_ue->harq_processes[harq_pid];

135 136
    start_symbol      = harq_process_ul_ue->pusch_pdu.start_symbol_index;
    ul_dmrs_symb_pos  = harq_process_ul_ue->pusch_pdu.ul_dmrs_symb_pos;
137
    number_of_symbols = harq_process_ul_ue->pusch_pdu.nr_of_symbols;
138
    dmrs_type         = harq_process_ul_ue->pusch_pdu.dmrs_config_type;
139

140
    for (i = start_symbol; i < start_symbol + number_of_symbols; i++) {
141

142
      if((ul_dmrs_symb_pos >> i) & 0x01)
143 144 145
        number_dmrs_symbols += 1;

    }
146

147
    rnti                  = harq_process_ul_ue->pusch_pdu.rnti;
148
    ulsch_ue->Nid_cell    = Nid_cell;
149 150

    nb_dmrs_re_per_rb = ((dmrs_type == pusch_dmrs_type1) ? 6:4)*harq_process_ul_ue->pusch_pdu.num_dmrs_cdm_grps_no_data;
151

152
    N_RE_prime = NR_NB_SC_PER_RB*number_of_symbols - nb_dmrs_re_per_rb*number_dmrs_symbols - N_PRB_oh;
153

154
    nb_rb = harq_process_ul_ue->pusch_pdu.rb_size;
155

156
    harq_process_ul_ue->num_of_mod_symbols = N_RE_prime*nb_rb*num_of_codewords;
157

158 159 160 161 162 163
    mcs            = harq_process_ul_ue->pusch_pdu.mcs_index;
    Nl             = harq_process_ul_ue->pusch_pdu.nrOfLayers;
    mod_order      = nr_get_Qm_ul(mcs, 0);
    code_rate      = nr_get_code_rate_ul(mcs, 0);

    harq_process_ul_ue->pusch_pdu.pusch_data.tb_size = nr_compute_tbs(mod_order,
164
                                             code_rate,
165 166
                                             nb_rb,
                                             number_of_symbols,
167
                                             nb_dmrs_re_per_rb*number_dmrs_symbols,
168
                                             0,
169
                                             0,
170
                                             harq_process_ul_ue->pusch_pdu.nrOfLayers);
171

172 173 174
    /////////////////////////ULSCH coding/////////////////////////
    ///////////

175 176
    unsigned int G = nr_get_G(nb_rb, number_of_symbols,
                              nb_dmrs_re_per_rb, number_dmrs_symbols, mod_order, Nl);
177
    nr_ulsch_encoding(ulsch_ue, frame_parms, harq_pid, G);
178 179 180 181 182 183 184

    ///////////
    ////////////////////////////////////////////////////////////////////

    /////////////////////////ULSCH scrambling/////////////////////////
    ///////////

185
    available_bits = G;
186 187 188 189 190 191

    memset(scrambled_output[cwd_index], 0, ((available_bits>>5)+1)*sizeof(uint32_t));

    nr_pusch_codeword_scrambling(ulsch_ue->g,
                                 available_bits,
                                 ulsch_ue->Nid_cell,
192
                                 rnti,
193 194 195 196 197 198 199 200 201 202 203 204 205 206
                                 scrambled_output[cwd_index]); // assume one codeword for the moment


    /////////////
    //////////////////////////////////////////////////////////////////////////

    /////////////////////////ULSCH modulation/////////////////////////
    ///////////

    nr_modulation(scrambled_output[cwd_index], // assume one codeword for the moment
                  available_bits,
                  mod_order,
                  (int16_t *)ulsch_ue->d_mod);

207

208
    // pusch_transform_precoding(ulsch_ue, frame_parms, harq_pid);
209 210 211 212 213

    ///////////
    ////////////////////////////////////////////////////////////////////////


214
  //}
215 216 217 218 219
  }

  /////////////////////////DMRS Modulation/////////////////////////
  ///////////
  pusch_dmrs = UE->nr_gold_pusch_dmrs[slot];
220 221
  n_dmrs = (nb_rb*((dmrs_type == pusch_dmrs_type1) ? 6:4)*number_dmrs_symbols);
  int16_t mod_dmrs[n_dmrs<<1] __attribute((aligned(16)));
222 223 224
  ///////////
  ////////////////////////////////////////////////////////////////////////

adk's avatar
adk committed
225 226 227 228

  /////////////////////////PTRS parameters' initialization/////////////////////////
  ///////////

229
  int16_t mod_ptrs[(nb_rb/2)*(NR_SYMBOLS_PER_SLOT-1)*2]; // assume maximum number of PTRS per pusch allocation
adk's avatar
adk committed
230 231
  K_ptrs = 0; // just to avoid a warning

232
  if (harq_process_ul_ue->pusch_pdu.pdu_bit_map & PUSCH_PDU_BITMAP_PUSCH_PTRS) {
adk's avatar
adk committed
233

234
    K_ptrs = (harq_process_ul_ue->pusch_pdu.pusch_ptrs.ptrs_freq_density)?4:2;
235
    L_ptrs = 1<<harq_process_ul_ue->pusch_pdu.pusch_ptrs.ptrs_time_density;
adk's avatar
adk committed
236 237 238 239 240 241

    beta_ptrs = 1; // temp value until power control is implemented

    ulsch_ue->ptrs_symbols = 0;

    set_ptrs_symb_idx(&ulsch_ue->ptrs_symbols,
242
                      number_of_symbols,
adk's avatar
adk committed
243 244
                      start_symbol,
                      L_ptrs,
245
                      ul_dmrs_symb_pos);
adk's avatar
adk committed
246 247 248 249 250
  }

  ///////////
  ////////////////////////////////////////////////////////////////////////////////

251 252 253 254 255
  /////////////////////////ULSCH layer mapping/////////////////////////
  ///////////

  tx_layers = (int16_t **)pusch_ue->txdataF_layers;

256
  nr_ue_layer_mapping(UE->ulsch[thread_id][gNB_id],
257 258 259
                      Nl,
                      available_bits/mod_order,
                      tx_layers);
260

261 262 263 264 265 266 267 268 269 270 271 272 273
  ///////////
  ////////////////////////////////////////////////////////////////////////


  //////////////////////// ULSCH transform precoding ////////////////////////
  ///////////

  l_prime[0] = 0; // single symbol ap 0

#ifdef NR_SC_FDMA
  uint32_t nb_re_pusch, nb_re_dmrs_per_rb;
  uint32_t y_offset = 0;

274
  for (l = start_symbol; l < start_symbol + number_of_symbols; l++) {
275

276
    if((ul_dmrs_symb_pos >> l) & 0x01)
277 278 279
      is_dmrs = 1;
    else
      is_dmrs = 0;
280 281

    if (is_dmrs == 1)
282
      nb_re_dmrs_per_rb = nb_dmrs_re_per_rb;
283 284 285
    else
      nb_re_dmrs_per_rb = 0;
    
286
    nb_re_pusch = nb_rb * (NR_NB_SC_PER_RB - nb_re_dmrs_per_rb);
287 288 289 290 291 292 293 294

    nr_dft(&ulsch_ue->y[y_offset], &((int32_t*)tx_layers[0])[y_offset], nb_re_pusch);

    y_offset = y_offset + nb_re_pusch;
  }
#else
  memcpy(ulsch_ue->y, tx_layers[0], (available_bits/mod_order)*sizeof(int32_t));
#endif
295 296 297 298

  ///////////
  ////////////////////////////////////////////////////////////////////////

299 300


301 302 303 304 305
  /////////////////////////ULSCH RE mapping/////////////////////////
  ///////////

  txdataF = UE->common_vars.txdataF;

306
  start_rb = harq_process_ul_ue->pusch_pdu.rb_start;
307 308 309 310 311
  start_sc = frame_parms->first_carrier_offset + start_rb*NR_NB_SC_PER_RB;

  if (start_sc >= frame_parms->ofdm_symbol_size)
    start_sc -= frame_parms->ofdm_symbol_size;

312
  for (ap=0; ap< Nl; ap++) {
313 314 315 316 317

    // DMRS params for this ap
    get_Wt(Wt, ap, dmrs_type);
    get_Wf(Wf, ap, dmrs_type);
    delta = get_delta(ap, dmrs_type);
318
    
319

320
    uint8_t k_prime=0;
321
    uint8_t is_dmrs, is_ptrs, is_dmrs_symbol;
adk's avatar
adk committed
322
    uint16_t m=0, n=0, dmrs_idx=0, ptrs_idx = 0;
323

324
    for (l=start_symbol; l<start_symbol+number_of_symbols; l++) {
325 326

      k = start_sc;
Khalid Ahmed's avatar
Khalid Ahmed committed
327
      n = 0;
328
      dmrs_idx = 0;
329

330
      for (i=0; i< nb_rb*NR_NB_SC_PER_RB; i++) {
331 332 333

        sample_offsetF = l*frame_parms->ofdm_symbol_size + k;

Khalid Ahmed's avatar
Khalid Ahmed committed
334
        is_dmrs = 0;
adk's avatar
adk committed
335
        is_ptrs = 0;
336
        is_dmrs_symbol = 0;
Khalid Ahmed's avatar
Khalid Ahmed committed
337

338
        if((ul_dmrs_symb_pos >> l) & 0x01) {
339
          is_dmrs_symbol = 1;
340 341 342
          if (k == ((start_sc+get_dmrs_freq_idx_ul(n, k_prime, delta, dmrs_type))%frame_parms->ofdm_symbol_size))
            is_dmrs = 1;
        }
Khalid Ahmed's avatar
Khalid Ahmed committed
343

344
        if (harq_process_ul_ue->pusch_pdu.pdu_bit_map & PUSCH_PDU_BITMAP_PUSCH_PTRS){
345

346 347 348 349 350 351 352 353 354 355
          if(is_ptrs_symbol(l, ulsch_ue->ptrs_symbols))
            is_ptrs = is_ptrs_subcarrier(k,
                                         rnti,
                                         ap,
                                         dmrs_type,
                                         K_ptrs,
                                         nb_rb,
                                         harq_process_ul_ue->pusch_pdu.pusch_ptrs.ptrs_ports_list[0].ptrs_re_offset,
                                         start_sc,
                                         frame_parms->ofdm_symbol_size);
adk's avatar
adk committed
356 357
        }

Khalid Ahmed's avatar
Khalid Ahmed committed
358
        if (is_dmrs == 1) {
359

adk's avatar
adk committed
360 361 362
          if (k == start_sc){
            nr_modulation(pusch_dmrs[l][0], n_dmrs*2, DMRS_MOD_ORDER, mod_dmrs); // currently only codeword 0 is modulated. Qm = 2 as DMRS is QPSK modulated
          }
363 364 365 366 367

          ((int16_t*)txdataF[ap])[(sample_offsetF)<<1] = (Wt[l_prime[0]]*Wf[k_prime]*AMP*mod_dmrs[dmrs_idx<<1]) >> 15;
          ((int16_t*)txdataF[ap])[((sample_offsetF)<<1) + 1] = (Wt[l_prime[0]]*Wf[k_prime]*AMP*mod_dmrs[(dmrs_idx<<1) + 1]) >> 15;

          #ifdef DEBUG_PUSCH_MAPPING
368
            printf("dmrs_idx %d\t l %d \t k %d \t k_prime %d \t n %d \t dmrs: %d %d\n",
369 370 371 372
            dmrs_idx, l, k, k_prime, n, ((int16_t*)txdataF[ap])[(sample_offsetF)<<1],
            ((int16_t*)txdataF[ap])[((sample_offsetF)<<1) + 1]);
          #endif

373

374 375 376 377 378
          dmrs_idx++;
          k_prime++;
          k_prime&=1;
          n+=(k_prime)?0:1;

adk's avatar
adk committed
379 380 381
        }  else if (is_ptrs == 1) {

          if (k == start_sc){
382
            nr_modulation(pusch_dmrs[l][0], nb_rb, DMRS_MOD_ORDER, mod_ptrs);
adk's avatar
adk committed
383 384 385 386 387 388 389
          }

          ((int16_t*)txdataF[ap])[(sample_offsetF)<<1] = (beta_ptrs*AMP*mod_ptrs[ptrs_idx<<1]) >> 15;
          ((int16_t*)txdataF[ap])[((sample_offsetF)<<1) + 1] = (beta_ptrs*AMP*mod_ptrs[(ptrs_idx<<1) + 1]) >> 15;

          ptrs_idx++;

390
        } else if (!is_dmrs_symbol || allowed_xlsch_re_in_dmrs_symbol(k,start_sc,harq_process_ul_ue->pusch_pdu.num_dmrs_cdm_grps_no_data,dmrs_type)) {
391

392 393
          ((int16_t*)txdataF[ap])[(sample_offsetF)<<1]       = ((int16_t *) ulsch_ue->y)[m<<1];
          ((int16_t*)txdataF[ap])[((sample_offsetF)<<1) + 1] = ((int16_t *) ulsch_ue->y)[(m<<1) + 1];
394

395 396 397 398 399
        #ifdef DEBUG_PUSCH_MAPPING
          printf("m %d\t l %d \t k %d \t txdataF: %d %d\n",
          m, l, k, ((int16_t*)txdataF[ap])[(sample_offsetF)<<1],
          ((int16_t*)txdataF[ap])[((sample_offsetF)<<1) + 1]);
        #endif
400 401

          m++;
402 403 404 405 406 407

        } else {

          ((int16_t*)txdataF[ap])[(sample_offsetF)<<1]       = 0;
          ((int16_t*)txdataF[ap])[((sample_offsetF)<<1) + 1] = 0;

408 409 410 411 412 413 414
        }

        if (++k >= frame_parms->ofdm_symbol_size)
          k -= frame_parms->ofdm_symbol_size;
      }
    }
  }
415

416
  //}
417 418 419
  NR_UL_UE_HARQ_t *harq_process_ulsch=NULL;
  harq_process_ulsch = UE->ulsch[thread_id][gNB_id][0]->harq_processes[harq_pid];
  harq_process_ulsch->status = SCH_IDLE;
420

421 422 423 424 425 426 427 428
  ///////////
  ////////////////////////////////////////////////////////////////////////

}


uint8_t nr_ue_pusch_common_procedures(PHY_VARS_NR_UE *UE,
                                      uint8_t slot,
429 430
                                      NR_DL_FRAME_PARMS *frame_parms,
                                      uint8_t Nl) {
431 432 433 434 435 436 437 438

  int tx_offset, ap;
  int32_t **txdata;
  int32_t **txdataF;

  /////////////////////////IFFT///////////////////////
  ///////////

439
  tx_offset = frame_parms->get_samples_slot_timestamp(slot, frame_parms, 0);
440

441 442 443 444 445 446 447
  // clear the transmit data array for the current subframe
  /*for (int aa=0; aa<UE->frame_parms.nb_antennas_tx; aa++) {
	  memset(&UE->common_vars.txdata[aa][tx_offset],0,UE->frame_parms.samples_per_slot*sizeof(int32_t));
	  //memset(&UE->common_vars.txdataF[aa][tx_offset],0,UE->frame_parms.samples_per_slot*sizeof(int32_t));
  }*/


448 449 450
  txdata = UE->common_vars.txdata;
  txdataF = UE->common_vars.txdataF;

451 452 453 454 455 456 457 458 459 460 461 462 463
  int symb_offset = (slot%frame_parms->slots_per_subframe)*frame_parms->symbols_per_slot;
  for(ap = 0; ap < Nl; ap++) {
    for (int s=0;s<NR_NUMBER_OF_SYMBOLS_PER_SLOT;s++){
      LOG_D(PHY,"rotating txdataF symbol %d (%d) => (%d.%d)\n",
	    s,s+symb_offset,frame_parms->symbol_rotation[2*(s+symb_offset)],frame_parms->symbol_rotation[1+(2*(s+symb_offset))]);
      rotate_cpx_vector((int16_t *)&txdataF[ap][frame_parms->ofdm_symbol_size*s],
			&frame_parms->symbol_rotation[2*(s+symb_offset)],
			(int16_t *)&txdataF[ap][frame_parms->ofdm_symbol_size*s],
			frame_parms->ofdm_symbol_size,
			15);
    }
  }

464
  for (ap = 0; ap < Nl; ap++) {
465 466 467 468 469 470 471 472 473 474 475 476
    if (frame_parms->Ncp == 1) { // extended cyclic prefix
      PHY_ofdm_mod(txdataF[ap],
                   &txdata[ap][tx_offset],
                   frame_parms->ofdm_symbol_size,
                   12,
                   frame_parms->nb_prefix_samples,
                   CYCLIC_PREFIX);
    } else { // normal cyclic prefix
      nr_normal_prefix_mod(txdataF[ap],
                           &txdata[ap][tx_offset],
                           14,
                           frame_parms);
477
    }
478
  }
479

480 481 482 483
  ///////////
  ////////////////////////////////////////////////////
  return 0;
}