phy_procedures_lte_eNb.c 62 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
/*
 * 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.0  (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
 */
21 22 23

/*! \file phy_procedures_lte_eNB.c
 * \brief Implementation of eNB procedures from 36.213 LTE specifications
24
 * \author R. Knopp, F. Kaltenberger, N. Nikaein, X. Foukas
25 26 27
 * \date 2011
 * \version 0.1
 * \company Eurecom
28
 * \email: knopp@eurecom.fr,florian.kaltenberger@eurecom.fr,navid.nikaein@eurecom.fr, x.foukas@sms.ed.ac.uk
29 30 31 32 33 34 35 36
 * \note
 * \warning
 */

#include "PHY/defs.h"
#include "PHY/extern.h"
#include "SCHED/defs.h"
#include "SCHED/extern.h"
37
#include "nfapi_interface.h"
38
#include "fapi_l1.h"
39 40 41
#include "UTIL/LOG/log.h"
#include "UTIL/LOG/vcd_signal_dumper.h"

42 43
#include "T.h"

44
#include "assertions.h"
gauthier's avatar
gauthier committed
45
#include "msc.h"
46

47 48
#include <time.h>

49
#if defined(ENABLE_ITTI)
gauthier's avatar
For RAL  
gauthier committed
50
#   include "intertask_interface.h"
51 52
#endif

53

knopp's avatar
 
knopp committed
54

55

56
void pmch_procedures(PHY_VARS_eNB *eNB,eNB_rxtx_proc_t *proc,PHY_VARS_RN *rn,relaying_type_t r_type) {
57 58


Cedric Roux's avatar
Cedric Roux committed
59
#if defined(Rel10) || defined(Rel14)
60 61
  MCH_PDU *mch_pduP;
  MCH_PDU  mch_pdu;
62
  //  uint8_t sync_area=255;
63
#endif
64

65
  int subframe = proc->subframe_tx;
66

67 68 69 70 71 72
  // This is DL-Cell spec pilots in Control region
  generate_pilots_slot(eNB,
		       eNB->common_vars.txdataF,
		       AMP,
		       subframe<<1,1);

73
  
Cedric Roux's avatar
Cedric Roux committed
74
#if defined(Rel10) || defined(Rel14)
75 76
  // if mcch is active, send regardless of the node type: eNB or RN
  // when mcch is active, MAC sched does not allow MCCH and MTCH multiplexing
77
  /*
78 79
  mch_pduP = mac_xface->get_mch_sdu(eNB->Mod_id,
				    eNB->CC_id,
80
				    proc->frame_tx,
81
				    subframe);
82
  */
83 84 85
  switch (r_type) {
  case no_relay:
    if ((mch_pduP->Pdu_size > 0) && (mch_pduP->sync_area == 0)) // TEST: only transmit mcch for sync area 0
knopp's avatar
knopp committed
86
      LOG_D(PHY,"[eNB%"PRIu8"] Frame %d subframe %d : Got MCH pdu for MBSFN (MCS %"PRIu8", TBS %d) \n",
87 88
	    eNB->Mod_id,proc->frame_tx,subframe,mch_pduP->mcs,
	    eNB->dlsch_MCH->harq_processes[0]->TBS>>3);
89 90
    else {
      LOG_D(PHY,"[DeNB %"PRIu8"] Frame %d subframe %d : Do not transmit MCH pdu for MBSFN sync area %"PRIu8" (%s)\n",
91
	    eNB->Mod_id,proc->frame_tx,subframe,mch_pduP->sync_area,
92 93
	    (mch_pduP->Pdu_size == 0)? "Empty MCH PDU":"Let RN transmit for the moment");
      mch_pduP = NULL;
gauthier's avatar
For RAL  
gauthier committed
94
    }
95
    
knopp's avatar
 
knopp committed
96
    break;
97 98 99
    
  case multicast_relay:
    if ((mch_pduP->Pdu_size > 0) && ((mch_pduP->mcch_active == 1) || mch_pduP->msi_active==1)) {
knopp's avatar
knopp committed
100
      LOG_D(PHY,"[RN %"PRIu8"] Frame %d subframe %d: Got the MCH PDU for MBSFN  sync area %"PRIu8" (MCS %"PRIu8", TBS %"PRIu16")\n",
101 102 103 104 105 106 107 108 109
	    rn->Mod_id,rn->frame, subframe,
	    mch_pduP->sync_area,mch_pduP->mcs,mch_pduP->Pdu_size);
    } else if (rn->mch_avtive[subframe%5] == 1) { // SF2 -> SF7, SF3 -> SF8
      mch_pduP= &mch_pdu;
      memcpy(&mch_pduP->payload, // could be a simple copy
	     rn->dlsch_rn_MCH[subframe%5]->harq_processes[0]->b,
	     rn->dlsch_rn_MCH[subframe%5]->harq_processes[0]->TBS>>3);
      mch_pduP->Pdu_size = (uint16_t) (rn->dlsch_rn_MCH[subframe%5]->harq_processes[0]->TBS>>3);
      mch_pduP->mcs = rn->dlsch_rn_MCH[subframe%5]->harq_processes[0]->mcs;
knopp's avatar
knopp committed
110
      LOG_D(PHY,"[RN %"PRIu8"] Frame %d subframe %d: Forward the MCH PDU for MBSFN received on SF %d sync area %"PRIu8" (MCS %"PRIu8", TBS %"PRIu16")\n",
111 112
	    rn->Mod_id,rn->frame, subframe,subframe%5,
	    rn->sync_area[subframe%5],mch_pduP->mcs,mch_pduP->Pdu_size);
knopp's avatar
 
knopp committed
113
    } else {
114
      mch_pduP=NULL;
knopp's avatar
 
knopp committed
115
    }
116 117
    
    rn->mch_avtive[subframe]=0;
knopp's avatar
 
knopp committed
118
    break;
119
    
120
  default:
121
    LOG_W(PHY,"[eNB %"PRIu8"] Frame %d subframe %d: unknown relaying type %d \n",
122
	  eNB->Mod_id,proc->frame_tx,subframe,r_type);
123 124 125 126 127
    mch_pduP=NULL;
    break;
  }// switch
  
  if (mch_pduP) {
128
    fill_eNB_dlsch_MCH(eNB,mch_pduP->mcs,1,0);
129
    // Generate PMCH
130
    generate_mch(eNB,proc,(uint8_t*)mch_pduP->payload);
131
  } else {
132
    LOG_D(PHY,"[eNB/RN] Frame %d subframe %d: MCH not generated \n",proc->frame_tx,subframe);
knopp's avatar
 
knopp committed
133
  }
134
  
135
#endif
136
}
137

138
void common_signal_procedures (PHY_VARS_eNB *eNB,int frame, int subframe) {
139

140
  LTE_DL_FRAME_PARMS *fp=&eNB->frame_parms;
141
  int **txdataF = eNB->common_vars.txdataF;
142
  uint8_t *pbch_pdu=&eNB->pbch_pdu[0];
143

144 145
  LOG_D(PHY,"common_signal_procedures: frame %d, subframe %d\n",frame,subframe); 

146
  // generate Cell-Specific Reference Signals for both slots
147 148 149 150 151 152 153
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_ENB_RS_TX,1);
  generate_pilots_slot(eNB,
		       txdataF,
		       AMP,
		       subframe<<1,0);
  // check that 2nd slot is for DL
  if (subframe_select(fp,subframe) == SF_DL)
154 155 156
    generate_pilots_slot(eNB,
			 txdataF,
			 AMP,
157 158 159 160
			 (subframe<<1)+1,0);
  
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_ENB_RS_TX,0);
      
161

162
  // First half of PSS/SSS (FDD, slot 0)
knopp's avatar
 
knopp committed
163
  if (subframe == 0) {
164
    if (fp->frame_type == FDD) {
165 166 167 168 169 170 171 172 173 174 175
      generate_pss(txdataF,
		   AMP,
		   fp,
		   (fp->Ncp==NORMAL) ? 6 : 5,
		   0);
      generate_sss(txdataF,
		   AMP,
		   fp,
		   (fp->Ncp==NORMAL) ? 5 : 4,
		   0);
      
176
    }
177
    
178 179


180 181 182
      
    /// First half of SSS (TDD, slot 1)
    
183
    if (fp->frame_type == TDD) {
184 185 186 187 188 189
      generate_sss(txdataF,
		   AMP,
		   fp,
		   (fp->Ncp==NORMAL) ? 6 : 5,
		   1);
    }
190

191 192
    // generate PBCH (Physical Broadcast CHannel) info

193
    /// generate PBCH
194
    if ((frame&3)==0) {
195 196
      //AssertFatal(eNB->pbch_configured==1,"PBCH was not configured by MAC\n");
      if (eNB->pbch_configured!=1) return;
197
      eNB->pbch_configured=0;
198
    }
199 200 201 202 203 204 205
    generate_pbch(&eNB->pbch,
		  txdataF,
		  AMP,
		  fp,
		  pbch_pdu,
		  frame&3);
  
206
  }
207
  else if ((subframe == 1) &&
208
	   (fp->frame_type == TDD)){
209 210 211 212 213
    generate_pss(txdataF,
		 AMP,
		 fp,
		 2,
		 2);
214
  }
215 216 217
  
  // Second half of PSS/SSS (FDD, slot 10)
  else if ((subframe == 5) && 
218
	   (fp->frame_type == FDD)) {
219 220 221 222 223 224 225 226 227 228
    generate_pss(txdataF,
		 AMP,
		 &eNB->frame_parms,
		 (fp->Ncp==NORMAL) ? 6 : 5,
		 10);
    generate_sss(txdataF,
		 AMP,
		 &eNB->frame_parms,
		 (fp->Ncp==NORMAL) ? 5 : 4,
		 10);
229

230
  }
231

232 233
  //  Second-half of SSS (TDD, slot 11)
  else if ((subframe == 5) &&
234
	   (fp->frame_type == TDD)) {
235 236 237 238 239
    generate_sss(txdataF,
		 AMP,
		 fp,
		 (fp->Ncp==NORMAL) ? 6 : 5,
		 11);
knopp's avatar
 
knopp committed
240
  }
knopp's avatar
 
knopp committed
241

242 243
  // Second half of PSS (TDD, slot 12)
  else if ((subframe == 6) &&
244
	   (fp->frame_type == TDD)) { 
245 246 247 248 249 250
    generate_pss(txdataF,
		 AMP,
		 fp,
		 2,
		 12);
  }
knopp's avatar
 
knopp committed
251

252
}
knopp's avatar
 
knopp committed
253

254 255


256 257 258 259 260 261
void pdsch_procedures(PHY_VARS_eNB *eNB,
		      eNB_rxtx_proc_t *proc,
		      int harq_pid,
		      LTE_eNB_DLSCH_t *dlsch, 
		      LTE_eNB_DLSCH_t *dlsch1,
		      LTE_eNB_UE_stats *ue_stats,
262
		      int ra_flag) {
263

264 265
  int frame=proc->frame_tx;
  int subframe=proc->subframe_tx;
266 267
  LTE_DL_eNB_HARQ_t *dlsch_harq=dlsch->harq_processes[harq_pid];
  int input_buffer_length = dlsch_harq->TBS/8;
268
  LTE_DL_FRAME_PARMS *fp=&eNB->frame_parms;
269

270 271
  //if (frame < 200) {
  if (1){
272

273
    LOG_D(PHY,
274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293
	  "[eNB %"PRIu8"][PDSCH %"PRIx16"/%"PRIu8"] Frame %d, subframe %d: Generating PDSCH/DLSCH with input size = %"PRIu16", pdsch_start %d, G %d, nb_rb %"PRIu16", rb0 %x, rb1 %x, TBS %"PRIu16", pmi_alloc %"PRIx64", rv %"PRIu8" (round %"PRIu8")\n",
	  eNB->Mod_id, dlsch->rnti,harq_pid,
	  frame, subframe, input_buffer_length, dlsch_harq->pdsch_start,
	  get_G(fp,
		dlsch_harq->nb_rb,
		dlsch_harq->rb_alloc,
		dlsch_harq->Qm,
		dlsch_harq->Nl,
		dlsch_harq->pdsch_start,
		frame,
		subframe,
		dlsch_harq->mimo_mode==TM7?7:0),
	  dlsch_harq->nb_rb,
	  dlsch_harq->rb_alloc[0],
	  dlsch_harq->rb_alloc[1],
	  dlsch_harq->TBS,
	  pmi2hex_2Ar1(dlsch_harq->pmi_alloc),
	  dlsch_harq->rvidx,
	  dlsch_harq->round);
  }    
294
#if defined(MESSAGE_CHART_GENERATOR_PHY)
295 296 297
  MSC_LOG_TX_MESSAGE(
		     MSC_PHY_ENB,MSC_PHY_UE,
		     NULL,0,
298
		     "%05u:%02u PDSCH/DLSCH input size = %"PRIu16", G %d, nb_rb %"PRIu16", TBS %"PRIu16", pmi_alloc %"PRIx16", rv %"PRIu8" (round %"PRIu8")",
299 300 301 302 303
		     frame, subframe,
		     input_buffer_length,
		     get_G(fp,
			   dlsch_harq->nb_rb,
			   dlsch_harq->rb_alloc,
304
			   dlsch_harq->Qm,
305
			   dlsch_harq->Nl,
306
			   dlsch_harq->pdsch_start,
307 308 309
			   frame,
			   subframe,
			   dlsch_harq->mimo_mode==TM7?7:0),
310
		     dlsch_harq->nb_rb,
311
		     dlsch_harq->TBS,
312 313 314 315
		     pmi2hex_2Ar1(dlsch_harq->pmi_alloc),
		     dlsch_harq->rvidx,
		     dlsch_harq->round);
#endif
316 317
  
  
318
  if (ue_stats) ue_stats->dlsch_sliding_cnt++;
319
  
320 321 322
  if (dlsch_harq->round == 0) {
    if (ue_stats)
      ue_stats->dlsch_trials[harq_pid][0]++;
323 324 325 326 327 328 329 330
  } else {
    ue_stats->dlsch_trials[harq_pid][dlsch_harq->round]++;
#ifdef DEBUG_PHY_PROC
#ifdef DEBUG_DLSCH
    LOG_D(PHY,"[eNB] This DLSCH is a retransmission\n");
#endif
#endif
  }
331 332


333
  LOG_D(PHY,"Generating DLSCH/PDSCH %d\n",ra_flag);
334
  // 36-212 
335
  start_meas(&eNB->dlsch_encoding_stats);
336
  AssertFatal(dlsch_harq->pdu!=NULL,"dlsch_harq->pdu == NULL (rnti %x)\n",dlsch->rnti);
337 338
  eNB->te(eNB,
	  dlsch_harq->pdu,
339
	  dlsch_harq->pdsch_start,
340 341 342 343 344 345 346 347 348 349 350 351 352 353 354
	  dlsch,
	  frame,subframe,
	  &eNB->dlsch_rate_matching_stats,
	  &eNB->dlsch_turbo_encoding_stats,
	  &eNB->dlsch_interleaving_stats);
  stop_meas(&eNB->dlsch_encoding_stats);
  // 36-211
  start_meas(&eNB->dlsch_scrambling_stats);
  dlsch_scrambling(fp,
		   0,
		   dlsch,
		   harq_pid,
		   get_G(fp,
			 dlsch_harq->nb_rb,
			 dlsch_harq->rb_alloc,
355
			 dlsch_harq->Qm,
356
			 dlsch_harq->Nl,
357
			 dlsch_harq->pdsch_start,
358 359 360
			 frame,subframe,
			 0),
		   0,
361
		   frame,
362 363 364 365 366 367 368 369 370 371
		   subframe<<1);
  stop_meas(&eNB->dlsch_scrambling_stats);
  
  start_meas(&eNB->dlsch_modulation_stats);
  
  
  dlsch_modulation(eNB,
		   eNB->common_vars.txdataF,
		   AMP,
		   subframe,
372
		   dlsch_harq->pdsch_start,
373 374 375 376
		   dlsch,
		   dlsch1);
  
  stop_meas(&eNB->dlsch_modulation_stats);
377

378 379
  dlsch->active = 0;
}
380 381


382

383
void phy_procedures_eNB_TX(PHY_VARS_eNB *eNB,
knopp's avatar
knopp committed
384
			   eNB_rxtx_proc_t *proc,
385
                           relaying_type_t r_type,
386 387
			   PHY_VARS_RN *rn,
			   int do_meas)
388 389
{
  UNUSED(rn);
390 391
  int frame=proc->frame_tx;
  int subframe=proc->subframe_tx;
392
  uint32_t i,aa;
393
  uint8_t harq_pid;
394
  int8_t UE_id=0;
395
  uint8_t num_pdcch_symbols=0;
396
  uint8_t num_dci=0;
397 398
  uint8_t ul_subframe;
  uint32_t ul_frame;
399
  LTE_DL_FRAME_PARMS *fp=&eNB->frame_parms;
400
  LTE_UL_eNB_HARQ_t *ulsch_harq;
401

402
  int offset = eNB->CC_id;//proc == &eNB->proc.proc_rxtx[0] ? 0 : 1;
403

404
  if ((fp->frame_type == TDD) && (subframe_select(fp,subframe)==SF_UL)) return;
405

Raymond Knopp's avatar
Raymond Knopp committed
406
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_PROCEDURES_ENB_TX+offset,1);
407
  if (do_meas==1) start_meas(&eNB->phy_proc_tx);
408

Cedric Roux's avatar
Cedric Roux committed
409
  T(T_ENB_PHY_DL_TICK, T_INT(eNB->Mod_id), T_INT(frame), T_INT(subframe));
410

411
  // clear the transmit data array for the current subframe
412 413 414
  for (aa=0; aa<fp->nb_antenna_ports_eNB; aa++) {      
    memset(&eNB->common_vars.txdataF[aa][subframe*fp->ofdm_symbol_size*(fp->symbols_per_tti)],
	   0,fp->ofdm_symbol_size*(fp->symbols_per_tti)*sizeof(int32_t));
knopp's avatar
 
knopp committed
415
  }
416
  
417

418
  if (is_pmch_subframe(frame,subframe,fp)) {
419
    pmch_procedures(eNB,proc,rn,r_type);
420 421 422
  }
  else {
    // this is not a pmch subframe, so generate PSS/SSS/PBCH
423
    common_signal_procedures(eNB,proc->frame_tx, proc->subframe_tx);
424
  }
425

knopp's avatar
 
knopp committed
426
  // clear existing ulsch dci allocations before applying info from MAC  (this is table
427
  ul_subframe = pdcch_alloc2ul_subframe(fp,subframe);
428
  ul_frame = pdcch_alloc2ul_frame(fp,frame,subframe);
429 430 431



knopp's avatar
 
knopp committed
432
  // clear previous allocation information for all UEs
433
  for (i=0; i<NUMBER_OF_UE_MAX; i++) {
434 435
    if (eNB->dlsch[i][0])
      eNB->dlsch[i][0]->subframe_tx[subframe] = 0;
knopp's avatar
 
knopp committed
436
  }
437

Cedric Roux's avatar
Cedric Roux committed
438 439
  /* save old HARQ information needed for PHICH generation */
  for (i=0; i<NUMBER_OF_UE_MAX; i++) {
440
    harq_pid = subframe2harq_pid(fp,ul_frame,ul_subframe);
Cedric Roux's avatar
Cedric Roux committed
441
    if (eNB->ulsch[i]) {
442 443
      ulsch_harq = eNB->ulsch[i]->harq_processes[harq_pid];

Cedric Roux's avatar
Cedric Roux committed
444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463
      /* Store first_rb and n_DMRS for correct PHICH generation below.
       * For PHICH generation we need "old" values of last scheduling
       * for this HARQ process. 'generate_eNB_dlsch_params' below will
       * overwrite first_rb and n_DMRS and 'generate_phich_top', done
       * after 'generate_eNB_dlsch_params', would use the "new" values
       * instead of the "old" ones.
       *
       * This has been tested for FDD only, may be wrong for TDD.
       *
       * TODO: maybe we should restructure the code to be sure it
       *       is done correctly. The main concern is if the code
       *       changes and first_rb and n_DMRS are modified before
       *       we reach here, then the PHICH processing will be wrong,
       *       using wrong first_rb and n_DMRS values to compute
       *       ngroup_PHICH and nseq_PHICH.
       *
       * TODO: check if that works with TDD.
       */
      if ((subframe_select(fp,ul_subframe)==SF_UL) ||
          (fp->frame_type == FDD)) {
464 465
        ulsch_harq->previous_first_rb = ulsch_harq->first_rb;
        ulsch_harq->previous_n_DMRS   = ulsch_harq->n_DMRS;
Cedric Roux's avatar
Cedric Roux committed
466 467 468 469
      }
    }
  }

470

471 472 473 474 475
  //  num_pdcch_symbols = DCI_pdu->num_pdcch_symbols;
  num_pdcch_symbols = eNB->pdcch_vars[subframe&1].num_pdcch_symbols;
  num_dci           = eNB->pdcch_vars[subframe&1].num_dci;
  //  LOG_D(PHY,"num_pdcch_symbols %"PRIu8",(dci common %"PRIu8", dci uespec %"PRIu8"\n",num_pdcch_symbols,
  //        DCI_pdu->Num_common_dci,DCI_pdu->Num_ue_spec_dci);
476
  LOG_D(PHY,"num_pdcch_symbols %"PRIu8",(number dci %"PRIu8"\n",num_pdcch_symbols,
477 478
	num_dci);
  VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_DCI_INFO,num_pdcch_symbols);
479

480

481 482
  VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_DCI_INFO,(frame*10)+subframe);

483
  if (num_dci > 0)
484
    LOG_D(PHY,"[eNB %"PRIu8"] Frame %d, subframe %d: Calling generate_dci_top (pdcch) (num_dci %"PRIu8") num_pdcch_symbols:%d\n",eNB->Mod_id,frame, subframe, num_dci, num_pdcch_symbols);
485
    
486 487 488 489 490 491 492 493 494
  generate_dci_top(num_pdcch_symbols,
		   num_dci,
		   &eNB->pdcch_vars[subframe&1].dci_alloc[0],
		   0,
		   AMP,
		   fp,
		   eNB->common_vars.txdataF,
		   subframe);
  
495

gauthier's avatar
gauthier committed
496
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_ENB_PDCCH_TX,0);
497

knopp's avatar
 
knopp committed
498
  // Now scan UE specific DLSCH
499
  LTE_eNB_DLSCH_t *dlsch0,*dlsch1;
500
  for (UE_id=0; UE_id<NUMBER_OF_UE_MAX; UE_id++)
501
    {
502 503 504 505 506 507
      dlsch0 = eNB->dlsch[(uint8_t)UE_id][0]; 
      dlsch1 = eNB->dlsch[(uint8_t)UE_id][1]; 

      if ((dlsch0)&&
	  (dlsch0->rnti>0)&&
	  (dlsch0->active == 1)) {
508

509
	// get harq_pid
510
	harq_pid = dlsch0->harq_ids[subframe];
511 512 513 514 515
	AssertFatal(harq_pid>=0,"harq_pid is negative\n");
	// generate pdsch
	pdsch_procedures(eNB,
			 proc,
			 harq_pid,
516 517
			 dlsch0,
			 dlsch1,
518
			 &eNB->UE_stats[(uint32_t)UE_id],
519
			 0);
520 521


knopp's avatar
 
knopp committed
522
      }
523

524 525 526
      else if ((dlsch0)&&
	       (dlsch0->rnti>0)&&
	       (dlsch0->active == 0)) {
527

528
	// clear subframe TX flag since UE is not scheduled for PDSCH in this subframe (so that we don't look for PUCCH later)
529
	dlsch0->subframe_tx[subframe]=0;
knopp's avatar
 
knopp committed
530
      }
knopp's avatar
 
knopp committed
531
    }
532

533 534


535 536 537
  generate_phich_top(eNB,
		     proc,
		     AMP);
538

Raymond Knopp's avatar
Raymond Knopp committed
539
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_PROCEDURES_ENB_TX+offset,0);
540
  if (do_meas==1) stop_meas(&eNB->phy_proc_tx);
541
  
542 543
}

544 545 546 547 548
void prach_procedures(PHY_VARS_eNB *eNB,
#ifdef Rel14
		      int br_flag
#endif
		      ) {
549

550
  uint16_t max_preamble[4],max_preamble_energy[4],max_preamble_delay[4];
gauthier's avatar
gauthier committed
551
  uint16_t i;
552 553 554 555 556
  int frame,subframe;
#ifdef Rel14
  if (br_flag==1) {
    subframe = eNB->proc.subframe_prach_br;
    frame = eNB->proc.frame_prach_br;
557 558 559
    pthread_mutex_lock(&eNB->UL_INFO_mutex);
    eNB->UL_INFO.rach_ind_br.number_of_preambles=0;
    pthread_mutex_unlock(&eNB->UL_INFO_mutex);
560 561 562 563
  }
  else
#endif
    {
564 565 566
      pthread_mutex_lock(&eNB->UL_INFO_mutex);
      eNB->UL_INFO.rach_ind.number_of_preambles=0;
      pthread_mutex_unlock(&eNB->UL_INFO_mutex);
567 568 569
      subframe = eNB->proc.subframe_prach;
      frame = eNB->proc.frame_prach;
    }
570 571 572
  RU_t *ru;
  int aa=0;
  int ru_aa;
573

574
 
575
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_ENB_PRACH_RX,1);
576

577 578


579 580
  for (i=0;i<eNB->num_RU;i++) {
    ru=eNB->RU_list[i];
581
    for (ru_aa=0,aa=0;ru_aa<ru->nb_rx;ru_aa++,aa++) {
582
      eNB->prach_vars.rxsigF[0][aa] = eNB->RU_list[i]->prach_rxsigF[ru_aa];
583 584 585 586
#ifdef Rel14
      int ce_level;

      if (br_flag==1)
587
	for (ce_level=0;ce_level<4;ce_level++) eNB->prach_vars_br.rxsigF[ce_level][aa] = eNB->RU_list[i]->prach_rxsigF_br[ce_level][ru_aa];
588 589
#endif
    }
590
  }
591

592 593
  rx_prach(eNB,
	   eNB->RU_list[0],
594 595 596
	   &max_preamble[0],
	   &max_preamble_energy[0],
	   &max_preamble_delay[0],
597
	   frame,
598 599 600 601 602
	   0
#ifdef Rel14
	   ,br_flag
#endif
	   );
603

604
  //#ifdef DEBUG_PHY_PROC
knopp's avatar
knopp committed
605
  LOG_D(PHY,"[RAPROC] Frame %d, subframe %d : Most likely preamble %d, energy %d dB delay %d\n",
606
        frame,subframe,
607
	max_preamble[0],
608
        max_preamble_energy[0]/10,
609
        max_preamble_delay[0]);
610
  //q#endif
611

612 613
#ifdef Rel14
  if (br_flag==1) {
knopp's avatar
 
knopp committed
614

615 616 617 618 619
    int prach_mask;
      
    prach_mask = is_prach_subframe(&eNB->frame_parms,eNB->proc.frame_prach_br,eNB->proc.subframe_prach_br);
    
    eNB->UL_INFO.rach_ind_br.preamble_list                              = eNB->preamble_list_br;
620 621 622
    int ind=0;
    int ce_level=0;
    /* Save for later, it doesn't work    
623
    for (int ind=0,ce_level=0;ce_level<4;ce_level++) {
624 625 626
      
      if ((eNB->frame_parms.prach_emtc_config_common.prach_ConfigInfo.prach_CElevel_enable[ce_level]==1)&&
	  (prach_mask&(1<<(1+ce_level)) > 0) && // prach is active and CE level has finished its repetitions
627 628
	  (eNB->prach_vars_br.repetition_number[ce_level]==
	   eNB->frame_parms.prach_emtc_config_common.prach_ConfigInfo.prach_numRepetitionPerPreambleAttempt[ce_level])) {
629
    */ 
630 631 632 633 634 635 636 637 638 639
    if (eNB->frame_parms.prach_emtc_config_common.prach_ConfigInfo.prach_CElevel_enable[0]==1){ 
      if (max_preamble_energy[0] > 350) {
	eNB->UL_INFO.rach_ind_br.number_of_preambles++;
	
	eNB->preamble_list_br[ind].preamble_rel8.timing_advance        = max_preamble_delay[ind];//
	eNB->preamble_list_br[ind].preamble_rel8.preamble              = max_preamble[ind];
	// note: fid is implicitly 0 here, this is the rule for eMTC RA-RNTI from 36.321, Section 5.1.4
	eNB->preamble_list_br[ind].preamble_rel8.rnti                  = 1+subframe+(eNB->prach_vars_br.first_frame[ce_level]%40);  
	eNB->preamble_list_br[ind].instance_length                     = 0; //don't know exactly what this is
	eNB->preamble_list_br[ind].preamble_rel13.rach_resource_type   = 1+ce_level;  // CE Level
knopp's avatar
knopp committed
640
	LOG_D(PHY,"Filling NFAPI indication for RACH %d CELevel %d (mask %x) : TA %d, Preamble %d, rnti %x, rach_resource_type %d\n",
641 642 643 644 645 646 647 648 649
	      ind,
	      ce_level,
	      prach_mask,
	      eNB->preamble_list_br[ind].preamble_rel8.timing_advance,
	      eNB->preamble_list_br[ind].preamble_rel8.preamble,
	      eNB->preamble_list_br[ind].preamble_rel8.rnti,
	      eNB->preamble_list_br[ind].preamble_rel13.rach_resource_type);
      }
      /*
650 651
	ind++;
      }
652 653
      } */// ce_level
    }
654 655 656
  }
  else
#endif
Cedric Roux's avatar
Cedric Roux committed
657

658
    {
659
      if (max_preamble_energy[0] > 350) {
660 661 662 663 664 665 666 667 668 669

	LOG_D(PHY,"[eNB %d/%d][RAPROC] Frame %d, subframe %d Initiating RA procedure with preamble %d, energy %d.%d dB, delay %d\n",
	      eNB->Mod_id,
	      eNB->CC_id,
	      frame,
	      subframe,
	      max_preamble[0],
	      max_preamble_energy[0]/10,
	      max_preamble_energy[0]%10,
	      max_preamble_delay[0]);
670
	
671 672 673
	    T(T_ENB_PHY_INITIATE_RA_PROCEDURE, T_INT(eNB->Mod_id), T_INT(frame), T_INT(subframe), 0,
	      T_INT(max_preamble[0]), T_INT(max_preamble_energy[0]), T_INT(max_preamble_delay[0]));
	    
674 675 676 677 678 679 680 681 682 683 684 685 686
	    
	    
	    pthread_mutex_lock(&eNB->UL_INFO_mutex);
	    
	    eNB->UL_INFO.rach_ind.number_of_preambles                 = 1;
	    eNB->UL_INFO.rach_ind.preamble_list                       = eNB->preamble_list;
	    
	    eNB->preamble_list[0].preamble_rel8.timing_advance        = max_preamble_delay[0];
	    eNB->preamble_list[0].preamble_rel8.preamble              = max_preamble[0];
	    eNB->preamble_list[0].preamble_rel8.rnti                  = 1+subframe;  // note: fid is implicitly 0 here
	    eNB->preamble_list[0].preamble_rel13.rach_resource_type   = 0;
	    eNB->preamble_list[0].instance_length                     = 0; //don't know exactly what this is
	    
knopp's avatar
knopp committed
687
	    LOG_D(PHY,"Filling NFAPI indication for RACH : TA %d, Preamble %d, rnti %x, rach_resource_type %d\n",
688 689 690 691 692 693
		  eNB->preamble_list[0].preamble_rel8.timing_advance,
		  eNB->preamble_list[0].preamble_rel8.preamble,
		  eNB->preamble_list[0].preamble_rel8.rnti,
		  eNB->preamble_list[0].preamble_rel13.rach_resource_type);	    
	    pthread_mutex_unlock(&eNB->UL_INFO_mutex);
      
694
      } // max_preamble_energy > 350
695
    } // else br_flag
696
      /*
697
	mac_xface->initiate_ra_proc(eNB->Mod_id,
698 699 700 701 702 703 704 705 706 707 708 709
	eNB->CC_id,
	frame,
	preamble_max,
	preamble_delay_list[preamble_max]*update_TA/update_TA2,
	0,subframe,0);*/
      
    
    /*  } else {
    MSC_LOG_EVENT(MSC_PHY_ENB, "0 RA Failed add user, too many");
    LOG_I(PHY,"[eNB %d][RAPROC] frame %d, subframe %d: Unable to add user, max user count reached\n",
	  eNB->Mod_id,frame, subframe);
	  }*/
710

711
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_ENB_PRACH_RX,0);
712 713
}

714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759
void srs_procedures(PHY_VARS_eNB *eNB,eNB_rxtx_proc_t *proc) {

  LTE_DL_FRAME_PARMS *fp = &eNB->frame_parms;
  const int subframe = proc->subframe_rx;
  const int frame = proc->frame_rx;

  int i;

  if (is_srs_occasion_common(fp,frame,subframe)) { 
  
  // Do SRS processing 
  // check if there is SRS and we have to use shortened format
  // TODO: check for exceptions in transmission of SRS together with ACK/NACK
    for (i=0;i<NUMBER_OF_UE_MAX;i++) {

      if (eNB->soundingrs_ul_config_dedicated[i].active==1) {

      
	if (lte_srs_channel_estimation(fp,
				       &eNB->common_vars,
				       &eNB->srs_vars[i],
				       &eNB->soundingrs_ul_config_dedicated[i],
				       subframe,
				       0/*eNB_id*/)) {
	  LOG_E(PHY,"problem processing SRS\n");
	}
	eNB->soundingrs_ul_config_dedicated[i].active=0;
      }
    }
  }
}

void fill_sr_indication(PHY_VARS_eNB *eNB,uint16_t rnti,int frame,int subframe) {

  pthread_mutex_lock(&eNB->UL_INFO_mutex);
  nfapi_sr_indication_pdu_t *pdu =   &eNB->UL_INFO.sr_ind.sr_pdu_list[eNB->UL_INFO.sr_ind.number_of_srs];

  pdu->instance_length                                = 0; // don't know what to do with this
  //  pdu->rx_ue_information.handle                       = handle;
  pdu->rx_ue_information.rnti                         = rnti;

  eNB->UL_INFO.sr_ind.number_of_srs++;
  pthread_mutex_unlock(&eNB->UL_INFO_mutex);
}

void uci_procedures(PHY_VARS_eNB *eNB,eNB_rxtx_proc_t *proc)
Cedric Roux's avatar
Cedric Roux committed
760
{
761
  LTE_DL_FRAME_PARMS *fp=&eNB->frame_parms;
762 763
  uint8_t SR_payload = 0,pucch_b0b1[4][2]= {{0,0},{0,0},{0,0},{0,0}},harq_ack[4]={0,0,0,0};
  int32_t metric[4]={0,0,0,0},metric_SR=0,max_metric;
764
  ANFBmode_t bundling_flag;
765 766
#ifdef DEBUG_PHY_PROC
  uint8_t pucch_sel = 0;
767
  PUCCH_FMT_t format;
768
#endif
769 770
  const int subframe = proc->subframe_rx;
  const int frame = proc->frame_rx;
771 772 773 774
  int i;
  LTE_eNB_UCI *uci;
  uint16_t tdd_multiplexing_mask=0;
  int res;
775

776
  for (i=0;i<NUMBER_OF_UE_MAX;i++) {
Cedric Roux's avatar
Cedric Roux committed
777

778 779 780 781
    uci = &eNB->uci_vars[i];
    if ((uci->active == 1) &&
	(uci->frame == frame) &&
	(uci->subframe == subframe)) {
Cedric Roux's avatar
Cedric Roux committed
782

knopp's avatar
knopp committed
783 784 785
      // Null out PUCCH PRBs for noise measurement
      switch(fp->N_RB_UL) {
      case 6:
Cedric Roux's avatar
Cedric Roux committed
786 787
        eNB->rb_mask_ul[0] |= (0x1 | (1<<5)); //position 5
        break;
knopp's avatar
knopp committed
788
      case 15:
Cedric Roux's avatar
Cedric Roux committed
789 790
        eNB->rb_mask_ul[0] |= (0x1 | (1<<14)); // position 14
        break;
knopp's avatar
knopp committed
791
      case 25:
Cedric Roux's avatar
Cedric Roux committed
792 793
        eNB->rb_mask_ul[0] |= (0x1 | (1<<24)); // position 24
        break;
knopp's avatar
knopp committed
794
      case 50:
Cedric Roux's avatar
Cedric Roux committed
795 796 797
        eNB->rb_mask_ul[0] |= 0x1;
        eNB->rb_mask_ul[1] |= (1<<17); // position 49 (49-32)
        break;
knopp's avatar
knopp committed
798
      case 75:
Cedric Roux's avatar
Cedric Roux committed
799 800 801
        eNB->rb_mask_ul[0] |= 0x1;
        eNB->rb_mask_ul[2] |= (1<<10); // position 74 (74-64)
        break;
knopp's avatar
knopp committed
802
      case 100:
Cedric Roux's avatar
Cedric Roux committed
803 804 805
        eNB->rb_mask_ul[0] |= 0x1;
        eNB->rb_mask_ul[3] |= (1<<3); // position 99 (99-96)
        break;
knopp's avatar
knopp committed
806
      default:
Cedric Roux's avatar
Cedric Roux committed
807 808
        LOG_E(PHY,"Unknown number for N_RB_UL %d\n",fp->N_RB_UL);
        break;
knopp's avatar
knopp committed
809
      }
810

811 812 813 814 815 816 817 818
      switch (uci->type) {
      case SR:
      case HARQ_SR:
		
	metric_SR = rx_pucch(eNB,
			      uci->pucch_fmt,
			      i,
			      uci->n_pucch_1_0_sr[0],
819
			      0, // n2_pucch
820
			      uci->srs_active, // shortened format
821 822 823 824
			      &SR_payload,
			      frame,
			      subframe,
			      PUCCH1_THRES);
knopp's avatar
knopp committed
825
	LOG_D(PHY,"[eNB %d][SR %x] Frame %d subframe %d Checking SR is %d (SR n1pucch is %d)\n",
826
	      eNB->Mod_id,
827
	      uci->rnti,
828 829 830
	      frame,
	      subframe,
	      SR_payload,
831 832 833 834 835 836 837 838 839 840 841 842
	      uci->n_pucch_1_0_sr[0]);
	if (uci->type == SR) {
	  if (SR_payload == 1) {
	    fill_sr_indication(eNB,uci->rnti,frame,subframe);
	    return;
	  }
	  else {
	    return;
	  }
	}
      case HARQ:
	if (fp->frame_type == FDD) {
knopp's avatar
knopp committed
843
	  LOG_D(PHY,"Frame %d Subframe %d Demodulating PUCCH (UCI %d) for ACK/NAK (uci->pucch_fmt %d,uci->type %d.uci->frame %d, uci->subframe %d): n1_pucch0 %d SR_payload %d\n",
844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880
		frame,subframe,i,
		uci->pucch_fmt,uci->type,
		uci->frame,uci->subframe,uci->n_pucch_1[0][0],
		SR_payload);
	  
	  metric[0] = rx_pucch(eNB,
			       uci->pucch_fmt,
			       i,
			       uci->n_pucch_1[0][0],
			       0, //n2_pucch
			       uci->srs_active, // shortened format
			       pucch_b0b1[0],
			       frame,
			       subframe,
			       PUCCH1a_THRES);
	  
	  
	  /* cancel SR detection if reception on n1_pucch0 is better than on SR PUCCH resource index, otherwise send it up to MAC */
	  if (uci-><