nr_ulsch_coding.c 16.4 KB
Newer Older
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 33 34 35 36 37 38
/*
 * 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_coding.c
* \brief Top-level routines for coding the ULSCH transport channel as described in 38.212 V15.4 2018-12
* \author Khalid Ahmed
* \date 2019
* \version 0.1
* \company Fraunhofer IIS
* \email: khalid.ahmed@iis.fraunhofer.de
* \note
* \warning
*/

#include "PHY/defs_UE.h"
#include "PHY/phy_extern_ue.h"
#include "PHY/NR_UE_TRANSPORT/nr_transport_proto_ue.h"
#include "PHY/CODING/coding_defs.h"
#include "PHY/CODING/coding_extern.h"
#include "PHY/CODING/lte_interleaver_inline.h"
39
#include "PHY/CODING/nrLDPC_extern.h"
40 41
#include "PHY/NR_UE_TRANSPORT/nr_transport_ue.h"
#include "common/utils/LOG/vcd_signal_dumper.h"
42
#include "LAYER2/NR_MAC_gNB/mac_proto.h"
43 44


45
void free_nr_ue_ulsch(NR_UE_ULSCH_t **ulschptr,unsigned char N_RB_UL)
46 47

{
yilmazt's avatar
yilmazt committed
48
  int i, r;
49
  NR_UE_ULSCH_t *ulsch = *ulschptr;
50 51 52 53 54 55

  if (ulsch) {
#ifdef DEBUG_ULSCH_FREE
    printf("Freeing ulsch %p\n",ulsch);
#endif

56 57 58 59
  uint16_t a_segments = MAX_NUM_NR_ULSCH_SEGMENTS;  //number of segments to be allocated

  if (N_RB_UL != 273) {
    a_segments = a_segments*N_RB_UL;
60
    a_segments = a_segments/273 +1;
61 62 63
  }  


64 65
    for (i=0; i<NR_MAX_ULSCH_HARQ_PROCESSES; i++) {
      if (ulsch->harq_processes[i]) {
66 67

        if (ulsch->harq_processes[i]->a) {
laurent's avatar
laurent committed
68
          free16(ulsch->harq_processes[i]->a,a_segments*1056);
69 70
          ulsch->harq_processes[i]->a = NULL;
        }
71
        if (ulsch->harq_processes[i]->b) {
laurent's avatar
laurent committed
72
          free16(ulsch->harq_processes[i]->b,a_segments*1056);
73 74
          ulsch->harq_processes[i]->b = NULL;
        }
75 76 77 78 79 80 81 82 83
        if (ulsch->harq_processes[i]->e) {
          free16(ulsch->harq_processes[i]->e,14*N_RB_UL*12*8);
          ulsch->harq_processes[i]->e = NULL;
        }
        if (ulsch->harq_processes[i]->f) {
          free16(ulsch->harq_processes[i]->f,14*N_RB_UL*12*8);
          ulsch->harq_processes[i]->f = NULL;
        }
        for (r=0; r<a_segments; r++) {
84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100
          if (ulsch->harq_processes[i]->c[r]) {
            free16(ulsch->harq_processes[i]->c[r],((r==0)?8:0) + 3+768);
            ulsch->harq_processes[i]->c[r] = NULL;
          }

          if (ulsch->harq_processes[i]->d[r]) {
            free16(ulsch->harq_processes[i]->d[r],68*384);
            ulsch->harq_processes[i]->d[r] = NULL;
          }

        }

        free16(ulsch->harq_processes[i],sizeof(NR_UL_UE_HARQ_t));
        ulsch->harq_processes[i] = NULL;
      }
    }
    free16(ulsch,sizeof(NR_UE_ULSCH_t));
101
    *ulschptr = NULL;
102 103 104 105
  }

}

Ahmed's avatar
Ahmed committed
106

107
NR_UE_ULSCH_t *new_nr_ue_ulsch(uint16_t N_RB_UL,
yilmazt's avatar
yilmazt committed
108 109
                               int number_of_harq_pids,
                               uint8_t abstraction_flag)
110 111
{
  NR_UE_ULSCH_t *ulsch;
112
  unsigned char exit_flag = 0,i,r;
113
  uint16_t a_segments = MAX_NUM_NR_ULSCH_SEGMENTS;  //number of segments to be allocated
114

Francesco Mani's avatar
Francesco Mani committed
115 116
  if (N_RB_UL != 273) {
    a_segments = a_segments*N_RB_UL;
117
    a_segments = a_segments/273 +1;
118
  }  
119

120
  uint16_t ulsch_bytes = a_segments*1056;  // allocated bytes per segment
121 122 123 124 125 126 127 128 129

  ulsch = (NR_UE_ULSCH_t *)malloc16(sizeof(NR_UE_ULSCH_t));

  if (ulsch) {
    memset(ulsch,0,sizeof(NR_UE_ULSCH_t));

    ulsch->number_harq_processes_for_pusch = NR_MAX_ULSCH_HARQ_PROCESSES;
    ulsch->Mlimit = 4; // maximum harq retransmissions

130 131 132
    //for (i=0; i<10; i++)
      //ulsch->harq_ids[i] = 0;

133 134 135 136 137 138 139
    for (i=0; i<number_of_harq_pids; i++) {

      ulsch->harq_processes[i] = (NR_UL_UE_HARQ_t *)malloc16(sizeof(NR_UL_UE_HARQ_t));

      //      printf("ulsch->harq_processes[%d] %p\n",i,ulsch->harq_processes[i]);
      if (ulsch->harq_processes[i]) {
        memset(ulsch->harq_processes[i], 0, sizeof(NR_UL_UE_HARQ_t));
140 141
        ulsch->harq_processes[i]->b = (uint8_t*)malloc16(ulsch_bytes);
        ulsch->harq_processes[i]->a = (unsigned char*)malloc16(ulsch_bytes);
142 143

        if (ulsch->harq_processes[i]->a) {
144
          bzero(ulsch->harq_processes[i]->a,ulsch_bytes);
145 146 147 148 149 150
        } else {
          printf("Can't allocate PDU\n");
          exit_flag=1;
        }

        if (ulsch->harq_processes[i]->b)
151
          bzero(ulsch->harq_processes[i]->b,ulsch_bytes);
152 153 154 155 156 157
        else {
          LOG_E(PHY,"Can't get b\n");
          exit_flag=1;
        }

        if (abstraction_flag==0) {
158
          for (r=0; r<a_segments; r++) {
159 160 161
            // account for filler in first segment and CRCs for multiple segment case
            ulsch->harq_processes[i]->c[r] = (uint8_t*)malloc16(8448);
            ulsch->harq_processes[i]->d[r] = (uint8_t*)malloc16(68*384); //max size for coded output
162

163 164 165 166 167 168 169 170 171 172 173 174 175
            if (ulsch->harq_processes[i]->c[r]) {
              bzero(ulsch->harq_processes[i]->c[r],8448);
            } else {
              printf("Can't get c\n");
              exit_flag=2;
            }
            if (ulsch->harq_processes[i]->d[r]) {
              bzero(ulsch->harq_processes[i]->d[r],(68*384));
            } else {
              printf("Can't get d\n");
              exit_flag=2;
            }
          }
176 177 178 179 180 181 182 183 184 185 186 187 188 189
          ulsch->harq_processes[i]->e = (uint8_t*)malloc16(14*N_RB_UL*12*8);
          if (ulsch->harq_processes[i]->e) {
            bzero(ulsch->harq_processes[i]->e,14*N_RB_UL*12*8);
          } else {
            printf("Can't get e\n");
            exit_flag=1;
          }
          ulsch->harq_processes[i]->f = (uint8_t*)malloc16(14*N_RB_UL*12*8);
          if (ulsch->harq_processes[i]->f) {
            bzero(ulsch->harq_processes[i]->f,14*N_RB_UL*12*8);
          } else {
            printf("Can't get f\n");
            exit_flag=1;
          }
190 191 192 193 194 195 196 197 198 199
        }

        ulsch->harq_processes[i]->subframe_scheduling_flag = 0;
        ulsch->harq_processes[i]->first_tx = 1;
      } else {
        LOG_E(PHY,"Can't get harq_p %d\n",i);
        exit_flag=3;
      }
    }

200 201 202 203
    if (exit_flag==0) {
      for (i=0; i<number_of_harq_pids; i++) {
        ulsch->harq_processes[i]->round=0;
      }
204 205

      return(ulsch);
206
    }
207 208 209
  }

  LOG_E(PHY,"new_ue_ulsch exit flag, size of  %d ,   %zu\n",exit_flag, sizeof(LTE_UE_ULSCH_t));
210
  free_nr_ue_ulsch(&ulsch,N_RB_UL);
211 212 213
  return(NULL);


214 215 216 217
}


int nr_ulsch_encoding(NR_UE_ULSCH_t *ulsch,
yilmazt's avatar
yilmazt committed
218
                      NR_DL_FRAME_PARMS* frame_parms,
219 220
                      uint8_t harq_pid,
                      unsigned int G)
221 222 223 224
{
/////////////////////////parameters and variables declaration/////////////////////////
///////////

225
  unsigned int crc;
226 227
  NR_UL_UE_HARQ_t *harq_process; 
  uint16_t nb_rb ;
Sakthivel Velumani's avatar
Sakthivel Velumani committed
228 229 230
  uint32_t A, F;
  static uint32_t Z = 0;
  uint32_t *pz = &Z; 
231
  uint8_t mod_order; 
232 233
  uint16_t Kr,r;
  uint32_t r_offset;
234
  uint8_t BG;
Francesco Mani's avatar
Francesco Mani committed
235
  uint32_t E,Kb;
236 237
  uint8_t Ilbrm; 
  uint32_t Tbslbrm; 
Sakthivel Velumani's avatar
Sakthivel Velumani committed
238
  uint16_t R;
239
  float Coderate;
240 241 242 243 244 245 246 247 248 249

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


/////////////////////////parameters and variables initialization/////////////////////////
///////////

  crc = 1;
  harq_process = ulsch->harq_processes[harq_pid];
250 251
  nb_rb = harq_process->pusch_pdu.rb_size;
  A = harq_process->pusch_pdu.pusch_data.tb_size;
252
  pz = &Z;
253 254
  mod_order = nr_get_Qm_ul(harq_process->pusch_pdu.mcs_index, harq_process->pusch_pdu.mcs_table);
  R = nr_get_code_rate_ul(harq_process->pusch_pdu.mcs_index, harq_process->pusch_pdu.mcs_table);
255 256 257
  Kr=0;
  r_offset=0;
  BG = 1;
258
  F=0;
259 260
  Ilbrm = 0;
  Tbslbrm = 950984; //max tbs
261
  Coderate = 0.0;
262
  harq_process->round = nr_rv_round_map_ue[harq_process->pusch_pdu.pusch_data.rv_index];
263 264 265 266

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

Rakesh's avatar
Rakesh committed
267
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_NR_UE_ULSCH_ENCODING, VCD_FUNCTION_IN);
268

269
  LOG_D(PHY,"ulsch coding nb_rb %d, Nl = %d\n", nb_rb, harq_process->pusch_pdu.nrOfLayers);
270
  LOG_D(PHY,"ulsch coding A %d G %d mod_order %d\n", A,G, mod_order);
271 272 273 274 275 276 277 278 279 280 281

  //  if (harq_process->Ndi == 1) {  // this is a new packet
  if (harq_process->round == 0) {  // this is a new packet
#ifdef DEBUG_DLSCH_CODING
  printf("encoding thinks this is a new packet \n");
#endif

///////////////////////// a---->| add CRC |---->b /////////////////////////
///////////
    /*
    int i;
282
    printf("ulsch (tx): \n");
283 284 285 286 287
    for (i=0;i<(A>>3);i++)
      printf("%02x.",a[i]);
    printf("\n");
    */

288 289 290 291 292 293 294 295
    if (A > 3824) {
      // Add 24-bit crc (polynomial A) to payload
      crc = crc24a(harq_process->a,A)>>8;
      harq_process->a[A>>3] = ((uint8_t*)&crc)[2];
      harq_process->a[1+(A>>3)] = ((uint8_t*)&crc)[1];
      harq_process->a[2+(A>>3)] = ((uint8_t*)&crc)[0];
      //printf("CRC %x (A %d)\n",crc,A);
      //printf("a0 %d a1 %d a2 %d\n", a[A>>3], a[1+(A>>3)], a[2+(A>>3)]);
296

297
      harq_process->B = A+24;
298

299
      AssertFatal((A/8)+4 <= MAX_NR_ULSCH_PAYLOAD_BYTES,"A %d is too big (A/8+4 = %d > %d)\n",A,(A/8)+4,MAX_NR_ULSCH_PAYLOAD_BYTES);
300

301 302 303 304 305 306 307 308 309 310 311 312 313
      memcpy(harq_process->b,harq_process->a,(A/8)+4);
    }
    else {
      // Add 16-bit crc (polynomial A) to payload
      crc = crc16(harq_process->a,A)>>16;
      harq_process->a[A>>3] = ((uint8_t*)&crc)[1];
      harq_process->a[1+(A>>3)] = ((uint8_t*)&crc)[0];
      //printf("CRC %x (A %d)\n",crc,A);
      //printf("a0 %d a1 %d \n", a[A>>3], a[1+(A>>3)]);

      harq_process->B = A+16;

      AssertFatal((A/8)+3 <= MAX_NR_ULSCH_PAYLOAD_BYTES,"A %d is too big (A/8+3 = %d > %d)\n",A,(A/8)+3,MAX_NR_ULSCH_PAYLOAD_BYTES);
314

315 316
      memcpy(harq_process->b,harq_process->a,(A/8)+3);  // using 3 bytes to mimic the case of 24 bit crc
    }
317 318 319 320 321 322
///////////
///////////////////////////////////////////////////////////////////////////

///////////////////////// b---->| block segmentation |---->c /////////////////////////
///////////

Sakthivel Velumani's avatar
Sakthivel Velumani committed
323 324 325 326
    if (R<1024)
      Coderate = (float) R /(float) 1024;
    else
      Coderate = (float) R /(float) 2048;
327 328 329 330 331 332 333

    if ((A <=292) || ((A<=3824) && (Coderate <= 0.6667)) || Coderate <= 0.25){
      BG = 2;
    }
    else{
      BG = 1;
    }
334

335
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_NR_SEGMENTATION, VCD_FUNCTION_IN);
Francesco Mani's avatar
Francesco Mani committed
336 337 338 339 340 341 342 343
    Kb=nr_segmentation(harq_process->b,
                       harq_process->c,
                       harq_process->B,
                       &harq_process->C,
                       &harq_process->K,
                       pz,
                       &harq_process->F,
                       BG);
344
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_NR_SEGMENTATION, VCD_FUNCTION_OUT);
345 346

    F = harq_process->F;
347
    Kr = harq_process->K;
348 349
#ifdef DEBUG_DLSCH_CODING
    uint16_t Kr_bytes;
350
    Kr_bytes = Kr>>3;
351
#endif
352 353 354 355

///////////////////////// c---->| LDCP coding |---->d /////////////////////////
///////////

356
    //printf("segment Z %d k %d Kr %d BG %d\n", *pz,harq_process->K,Kr,BG);
357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377

    //start_meas(te_stats);
    for (r=0; r<harq_process->C; r++) {
      //channel_input[r] = &harq_process->d[r][0];
#ifdef DEBUG_DLSCH_CODING
      printf("Encoder: B %d F %d \n",harq_process->B, harq_process->F);
      printf("start ldpc encoder segment %d/%d\n",r,harq_process->C);
      printf("input %d %d %d %d %d \n", harq_process->c[r][0], harq_process->c[r][1], harq_process->c[r][2],harq_process->c[r][3], harq_process->c[r][4]);
      for (int cnt =0 ; cnt < 22*(*pz)/8; cnt ++){
      printf("%d ", harq_process->c[r][cnt]);
      }
      printf("\n");

#endif
      //ldpc_encoder_orig((unsigned char*)harq_process->c[r],harq_process->d[r],Kr,BG,0);
      //ldpc_encoder_optim((unsigned char*)harq_process->c[r],(unsigned char*)&harq_process->d[r][0],Kr,BG,NULL,NULL,NULL,NULL);
    }

    //for (int i=0;i<68*384;i++)
      //        printf("channel_input[%d]=%d\n",i,channel_input[i]);

378 379 380 381 382 383
    int temp_opp = 0;

    if (opp_enabled) {
      opp_enabled = 0;
      temp_opp = 1;
    }
384 385 386 387 388 389 390


    /*printf("output %d %d %d %d %d \n", harq_process->d[0][0], harq_process->d[0][1], harq_process->d[r][2],harq_process->d[0][3], harq_process->d[0][4]);
      for (int cnt =0 ; cnt < 66*(*pz); cnt ++){
      printf("%d \n",  harq_process->d[0][cnt]);
      }
      printf("\n");*/
391 392 393
    encoder_implemparams_t impp;
    impp.n_segments=harq_process->C;
    impp.macro_num=0;
394 395 396 397
    impp.tinput  = NULL;
    impp.tprep   = NULL;
    impp.tparity = NULL;
    impp.toutput = NULL;
398

Rakesh's avatar
Rakesh committed
399 400
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_LDPC_ENCODER_OPTIM, VCD_FUNCTION_IN);
    
401
    nrLDPC_encoder(harq_process->c,harq_process->d,*pz,Kb,Kr,BG,&impp);
Rakesh's avatar
Rakesh committed
402 403
    
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_LDPC_ENCODER_OPTIM, VCD_FUNCTION_OUT);
404 405 406 407

    //stop_meas(te_stats);
    //printf("end ldpc encoder -- output\n");
#ifdef DEBUG_DLSCH_CODING
408 409
    write_output("ulsch_enc_input0.m","enc_in0",&harq_process->c[0][0],Kr_bytes,1,4);
    write_output("ulsch_enc_output0.m","enc0",&harq_process->d[0][0],(3*8*Kr_bytes)+12,1,4);
410 411
#endif

412 413
    if (temp_opp) opp_enabled = 1;

414 415 416 417
///////////
///////////////////////////////////////////////////////////////////////////////

  }
Sakthivel Velumani's avatar
Sakthivel Velumani committed
418 419
  F = harq_process->F;
  Kr = harq_process->K;
420 421

  for (r=0; r<harq_process->C; r++) { // looping over C segments
422 423 424 425 426 427 428 429 430

    if (harq_process->F>0) {
            for (int k=(Kr-F-2*(*pz)); k<Kr-2*(*pz); k++) {
              harq_process->d[r][k] = NR_NULL;
              //if (k<(Kr-F+8))
              //printf("r %d filler bits [%d] = %d \n", r,k, harq_process->d[r][k]);
            }
    }

431

432 433 434 435 436 437
    LOG_D(PHY,"Rate Matching, Code segment %d (coded bits (G) %u, unpunctured/repeated bits per code segment %d, mod_order %d, nb_rb %d, rvidx %d)...\n",
	  r,
	  G,
	  Kr*3,
	  mod_order,nb_rb,
	  harq_process->pusch_pdu.pusch_data.rv_index);
438

439
    //start_meas(rm_stats);
440 441 442
///////////////////////// d---->| Rate matching bit selection |---->e /////////////////////////
///////////

443
    E = nr_get_E(G, harq_process->C, mod_order, harq_process->pusch_pdu.nrOfLayers, r);
444

445
    Tbslbrm = nr_compute_tbslbrm(0,nb_rb,harq_process->pusch_pdu.nrOfLayers,harq_process->C);
446

Rakesh's avatar
Rakesh committed
447
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_NR_RATE_MATCHING_LDPC, VCD_FUNCTION_IN);
448 449 450 451 452 453 454
    nr_rate_matching_ldpc(Ilbrm,
                          Tbslbrm,
                          BG,
                          *pz,
                          harq_process->d[r],
                          harq_process->e+r_offset,
                          harq_process->C,
455 456
			  F,
                          Kr-F-2*(*pz),
457
                          harq_process->pusch_pdu.pusch_data.rv_index,
458
                          E);
Rakesh's avatar
Rakesh committed
459
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_NR_RATE_MATCHING_LDPC, VCD_FUNCTION_OUT);
460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477



#ifdef DEBUG_DLSCH_CODING
    for (int i =0; i<16; i++)
      printf("output ratematching e[%d]= %d r_offset %d\n", i,harq_process->e[i+r_offset], r_offset);
#endif

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

    
///////////////////////// e---->| Rate matching bit interleaving |---->f /////////////////////////
///////////

    //stop_meas(rm_stats);

    //start_meas(i_stats);
Rakesh's avatar
Rakesh committed
478 479 480
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_NR_INTERLEAVING_LDPC, VCD_FUNCTION_IN);
    
    nr_interleaving_ldpc(E,
481 482 483
            mod_order,
            harq_process->e+r_offset,
            harq_process->f+r_offset);
Rakesh's avatar
Rakesh committed
484 485
    
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_NR_INTERLEAVING_LDPC, VCD_FUNCTION_OUT);
486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503
    //stop_meas(i_stats);


#ifdef DEBUG_DLSCH_CODING
    for (int i =0; i<16; i++)
      printf("output interleaving f[%d]= %d r_offset %d\n", i,harq_process->f[i+r_offset], r_offset);

    if (r==harq_process->C-1)
      write_output("enc_output.m","enc",harq_process->f,G,1,4);
#endif

    r_offset += E;

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

  }

504
  memcpy(ulsch->g,harq_process->f,G); // g is the concatenated code block
505

Rakesh's avatar
Rakesh committed
506
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_NR_UE_ULSCH_ENCODING, VCD_FUNCTION_OUT);
507 508 509

  return(0);
}