nr_ulsch_ue.c 17.7 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
                               uint8_t frame,
97 98
                               uint8_t slot,
                               uint8_t thread_id,
99
                               int gNB_id) {
100

101
  uint32_t available_bits, TBS;
102
  uint8_t mod_order, cwd_index, num_of_codewords, l;
103 104 105 106 107
  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;
108
  int8_t Wf[2], Wt[2], l_prime[2], delta;
109
  uint16_t rnti, n_dmrs, code_rate, number_dmrs_symbols, nb_rb, k, k_temp;
110
  uint8_t dmrs_type, nb_dmrs_re_per_rb, number_of_symbols, mcs, Nl;
111 112
  int ap, start_symbol, Nid_cell, i;
  int sample_offsetF, N_RE_prime, N_PRB_oh;
113
  uint16_t n_rnti, ul_dmrs_symb_pos;
114
  uint8_t data_existing =0;
adk's avatar
adk committed
115 116
  uint8_t L_ptrs, K_ptrs; // PTRS parameters
  uint16_t beta_ptrs; // PTRS parameter related to power control
117 118 119 120

  NR_UE_ULSCH_t *ulsch_ue;
  NR_UL_UE_HARQ_t *harq_process_ul_ue;
  NR_DL_FRAME_PARMS *frame_parms = &UE->frame_parms;
121
  NR_UE_PUSCH *pusch_ue = UE->pusch_vars[thread_id][gNB_id];
122
  uint8_t ulsch_input_buffer[MAX_ULSCH_PAYLOAD_BYTES];
adk's avatar
adk committed
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
  number_dmrs_symbols = 0;
129
  uint8_t mapping_type = UE->pusch_config.pusch_TimeDomainResourceAllocation[0]->mappingType;
130

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

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

136 137
    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;
138
    number_of_symbols = harq_process_ul_ue->pusch_pdu.nr_of_symbols;
139
    dmrs_type         = harq_process_ul_ue->pusch_pdu.dmrs_config_type;
140

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

143
      if((ul_dmrs_symb_pos >> (mapping_type ? (i - start_symbol) : i)) & 0x01)
144 145 146
        number_dmrs_symbols += 1;

    }
147

148
    rnti                  = harq_process_ul_ue->pusch_pdu.rnti;
149
    ulsch_ue->Nid_cell    = Nid_cell;
150
    nb_dmrs_re_per_rb     = ((dmrs_type == pusch_dmrs_type1) ? 6:4);
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
                                             harq_process_ul_ue->pusch_pdu.nrOfLayers);
170

171
    TBS = harq_process_ul_ue->pusch_pdu.pusch_data.tb_size;
172 173
    uint8_t access_mode = SCHEDULED_ACCESS;

174 175
    //-----------------------------------------------------//
    // to be removed later when MAC is ready
176

177
    if (harq_process_ul_ue != NULL){
178
      data_existing = 0;
179 180

    	if (IS_SOFTMODEM_NOS1){
181 182
        data_existing = nr_ue_get_sdu(UE->Mod_id, UE->CC_id, frame,
                                      slot, 0, ulsch_input_buffer, TBS/8, &access_mode);
183
    		//IP traffic to be transmitted
184 185 186
        if(data_existing){
          //harq_process_ul_ue->a = (unsigned char*)calloc(TBS/8, sizeof(unsigned char));
          memcpy(harq_process_ul_ue->a, ulsch_input_buffer, TBS/8);
187 188

    			#ifdef DEBUG_MAC_PDU
189 190
            LOG_I(PHY, "Printing MAC PDU to be encoded, TBS is: %d \n", TBS/8);
            for (i = 0; i < TBS / 8; i++) {
191 192 193 194 195
    					printf("0x%02x",harq_process_ul_ue->a[i]);
    				}
    				printf("\n");
				#endif
    		}
196 197
      }
      //Random traffic to be transmitted if there is no IP traffic available for this Tx opportunity
198
      if (!IS_SOFTMODEM_NOS1 || !data_existing) {
199 200 201 202 203 204 205 206
        //Use zeros for the header bytes in noS1 mode, in order to make sure that the LCID is not valid
        //and block this traffic from being forwarded to the upper layers at the gNB
        uint16_t payload_offset = 5;
        LOG_D(PHY, "Random data to be tranmsitted: \n");
        //Give the header bytes some dummy value in order to block the random packet at the MAC layer of the receiver
        for (i = 0; i<payload_offset; i++)
          harq_process_ul_ue->a[i] = 0;

207
        for (i = payload_offset; i < TBS / 8; i++) {
208 209
          harq_process_ul_ue->a[i] = (unsigned char) rand();
          //printf(" input encoder a[%d]=0x%02x\n",i,harq_process_ul_ue->a[i]);
210
        }
211
      }
212 213 214 215 216 217
    } else {
      LOG_E(PHY, "[phy_procedures_nrUE_TX] harq_process_ul_ue is NULL !!\n");
      return;
    }

    //-----------------------------------------------------//
218

219 220 221
    /////////////////////////ULSCH coding/////////////////////////
    ///////////

222 223
    unsigned int G = nr_get_G(nb_rb, number_of_symbols,
                              nb_dmrs_re_per_rb, number_dmrs_symbols, mod_order, Nl);
224

225
    nr_ulsch_encoding(ulsch_ue, frame_parms, harq_pid, G);
226 227 228 229 230 231 232

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

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

233 234
    available_bits = nr_get_G(nb_rb,
                              number_of_symbols,
235 236
                              nb_dmrs_re_per_rb,
                              number_dmrs_symbols,
237 238 239 240 241 242 243 244
                              mod_order,
                              1);

    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,
245
                                 rnti,
246 247 248 249 250 251 252 253 254 255 256 257 258 259
                                 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);

260

261
    // pusch_transform_precoding(ulsch_ue, frame_parms, harq_pid);
262 263 264 265 266

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


267
  //}
268 269 270 271 272
  }

  /////////////////////////DMRS Modulation/////////////////////////
  ///////////
  pusch_dmrs = UE->nr_gold_pusch_dmrs[slot];
273
  n_dmrs = (nb_rb*nb_dmrs_re_per_rb*number_dmrs_symbols);
274 275 276 277
  int16_t mod_dmrs[n_dmrs<<1];
  ///////////
  ////////////////////////////////////////////////////////////////////////

adk's avatar
adk committed
278 279 280 281

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

282
  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
283 284 285 286
  K_ptrs = 0; // just to avoid a warning

  if (UE->ptrs_configured == 1) {

287
    K_ptrs = harq_process_ul_ue->pusch_pdu.pusch_ptrs.ptrs_freq_density;
288
    L_ptrs = 1<<harq_process_ul_ue->pusch_pdu.pusch_ptrs.ptrs_time_density;
adk's avatar
adk committed
289 290 291 292 293 294

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

    ulsch_ue->ptrs_symbols = 0;

    set_ptrs_symb_idx(&ulsch_ue->ptrs_symbols,
295
                      number_of_symbols,
adk's avatar
adk committed
296 297
                      start_symbol,
                      L_ptrs,
298
                      ul_dmrs_symb_pos);
adk's avatar
adk committed
299 300 301 302 303
  }

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

304 305 306 307 308
  /////////////////////////ULSCH layer mapping/////////////////////////
  ///////////

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

309
  nr_ue_layer_mapping(UE->ulsch[thread_id][gNB_id],
310
                   Nl,
311 312 313
                   available_bits/mod_order,
                   tx_layers);

314 315 316 317 318 319 320 321 322 323 324 325 326
  ///////////
  ////////////////////////////////////////////////////////////////////////


  //////////////////////// 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;

327
  for (l = start_symbol; l < start_symbol + number_of_symbols; l++) {
328

329
    if((ul_dmrs_symb_pos >> ((mapping_type)?l-start_symbol:l)) & 0x01)
330 331 332
      is_dmrs = 1;
    else
      is_dmrs = 0;
333 334

    if (is_dmrs == 1)
335
      nb_re_dmrs_per_rb = nb_dmrs_re_per_rb;
336 337 338
    else
      nb_re_dmrs_per_rb = 0;
    
339
    nb_re_pusch = nb_rb * (NR_NB_SC_PER_RB - nb_re_dmrs_per_rb);
340 341 342 343 344 345 346 347

    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
348 349 350 351

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

352 353


354 355 356 357 358
  /////////////////////////ULSCH RE mapping/////////////////////////
  ///////////

  txdataF = UE->common_vars.txdataF;

359
  start_rb = harq_process_ul_ue->pusch_pdu.rb_start;
360 361 362 363 364
  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;

365
  for (ap=0; ap< Nl; ap++) {
366 367 368 369 370

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

373
    uint8_t k_prime=0;
adk's avatar
adk committed
374
    uint8_t is_dmrs, is_ptrs;
375
    uint8_t l_ref;
adk's avatar
adk committed
376
    uint16_t m=0, n=0, dmrs_idx=0, ptrs_idx = 0;
377

378
    for (l=start_symbol; l<start_symbol+number_of_symbols; l++) {
379 380

      k = start_sc;
Khalid Ahmed's avatar
Khalid Ahmed committed
381
      n = 0;
382
      dmrs_idx = 0;
383
      l_ref = (mapping_type) ? l-start_symbol : l;
384

385
      for (i=0; i< nb_rb*NR_NB_SC_PER_RB; i++) {
386 387 388

        sample_offsetF = l*frame_parms->ofdm_symbol_size + k;

Khalid Ahmed's avatar
Khalid Ahmed committed
389
        is_dmrs = 0;
adk's avatar
adk committed
390
        is_ptrs = 0;
Khalid Ahmed's avatar
Khalid Ahmed committed
391

392
        if((ul_dmrs_symb_pos >> l_ref) & 0x01) {
393 394 395
          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
396

adk's avatar
adk committed
397
        if (UE->ptrs_configured == 1){
398 399 400 401 402 403

          if(k < start_sc)
            k_temp = k + frame_parms->ofdm_symbol_size;
          else
            k_temp = k;

adk's avatar
adk committed
404
          is_ptrs = is_ptrs_symbol(l,
405
                                   k_temp,
406 407
                                   rnti,
                                   nb_rb,
adk's avatar
adk committed
408 409 410 411
                                   ap,
                                   K_ptrs,
                                   ulsch_ue->ptrs_symbols,
                                   start_sc,
412
                                   dmrs_type,
413
                                   ptrs_Uplink_Config->resourceElementOffset);
adk's avatar
adk committed
414 415
        }

Khalid Ahmed's avatar
Khalid Ahmed committed
416
        if (is_dmrs == 1) {
417

adk's avatar
adk committed
418 419 420
          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
          }
421 422 423 424 425

          ((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
426
            printf("dmrs_idx %d\t l %d \t k %d \t k_prime %d \t n %d \t dmrs: %d %d\n",
427 428 429 430
            dmrs_idx, l, k, k_prime, n, ((int16_t*)txdataF[ap])[(sample_offsetF)<<1],
            ((int16_t*)txdataF[ap])[((sample_offsetF)<<1) + 1]);
          #endif

431

432 433 434 435 436
          dmrs_idx++;
          k_prime++;
          k_prime&=1;
          n+=(k_prime)?0:1;

adk's avatar
adk committed
437 438 439
        }  else if (is_ptrs == 1) {

          if (k == start_sc){
440
            nr_modulation(pusch_dmrs[l][0], nb_rb, DMRS_MOD_ORDER, mod_ptrs);
adk's avatar
adk committed
441 442 443 444 445 446 447 448
          }

          ((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++;

          } else {
449

450 451
          ((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];
452 453 454 455 456 457 458 459 460 461 462 463 464 465 466

          #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

          m++;
        }

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

468 469
  //}

470 471 472
  ///////////
  ////////////////////////////////////////////////////////////////////////

473
  LOG_D(PHY, "Is data existing ?: %d \n", data_existing);
474 475 476 477
}


uint8_t nr_ue_pusch_common_procedures(PHY_VARS_NR_UE *UE,
478
                                      uint8_t harq_pid,
479
                                      uint8_t slot,
480 481
                                      uint8_t thread_id,
                                      uint8_t gNB_id,
482 483 484 485 486
                                      NR_DL_FRAME_PARMS *frame_parms) {

  int tx_offset, ap;
  int32_t **txdata;
  int32_t **txdataF;
487
  int timing_advance;
488
  uint8_t Nl = UE->ulsch[thread_id][gNB_id][0]->harq_processes[harq_pid]->pusch_pdu.nrOfLayers; // cw 0
489 490 491 492

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

493 494 495 496 497 498
#if defined(EXMIMO) || defined(OAI_USRP) || defined(OAI_BLADERF) || defined(OAI_LMSSDR)  || defined(OAI_ADRV9371_ZC706)
  timing_advance = UE->timing_advance;
#else
  timing_advance = 0;
#endif

499
  tx_offset = frame_parms->get_samples_slot_timestamp(slot,frame_parms,0) - timing_advance;
500 501 502 503

  if (tx_offset < 0)
    tx_offset += frame_parms->samples_per_frame;

504 505 506 507 508 509 510
  // 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));
  }*/


511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528
  txdata = UE->common_vars.txdata;
  txdataF = UE->common_vars.txdataF;

  for (ap = 0; ap < Nl; ap++) {
      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);
      }
    }
529

530 531 532 533
  ///////////
  ////////////////////////////////////////////////////
  return 0;
}