lte-enb.c 45.3 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 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70
/*******************************************************************************
    OpenAirInterface
    Copyright(c) 1999 - 2014 Eurecom

    OpenAirInterface is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.


    OpenAirInterface is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with OpenAirInterface.The full GNU General Public License is
    included in this distribution in the file called "COPYING". If not,
    see <http://www.gnu.org/licenses/>.

   Contact Information
   OpenAirInterface Admin: openair_admin@eurecom.fr
   OpenAirInterface Tech : openair_tech@eurecom.fr
   OpenAirInterface Dev  : openair4g-devel@lists.eurecom.fr

   Address      : Eurecom, Campus SophiaTech, 450 Route des Chappes, CS 50193 - 06904 Biot Sophia Antipolis cedex, FRANCE

*******************************************************************************/

/*! \file lte-enb.c
 * \brief Top-level threads for eNodeB
 * \author R. Knopp, F. Kaltenberger, Navid Nikaein
 * \date 2012
 * \version 0.1
 * \company Eurecom
 * \email: knopp@eurecom.fr,florian.kaltenberger@eurecom.fr, navid.nikaein@eurecom.fr
 * \note
 * \warning
 */
#define _GNU_SOURCE
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/mman.h>
#include <sched.h>
#include <linux/sched.h>
#include <signal.h>
#include <execinfo.h>
#include <getopt.h>
#include <sys/sysinfo.h>
#include "rt_wrapper.h"

#undef MALLOC //there are two conflicting definitions, so we better make sure we don't use it at all

#include "assertions.h"
#include "msc.h"

#include "PHY/types.h"

#include "PHY/defs.h"
#undef MALLOC //there are two conflicting definitions, so we better make sure we don't use it at all
//#undef FRAME_LENGTH_COMPLEX_SAMPLES //there are two conflicting definitions, so we better make sure we don't use it at all

#include "../../ARCH/COMMON/common_lib.h"

//#undef FRAME_LENGTH_COMPLEX_SAMPLES //there are two conflicting definitions, so we better make sure we don't use it at all

71 72
#include "PHY/LTE_TRANSPORT/if4_tools.h"

73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124
#include "PHY/extern.h"
#include "SCHED/extern.h"
#include "LAYER2/MAC/extern.h"

#include "../../SIMU/USER/init_lte.h"

#include "LAYER2/MAC/defs.h"
#include "LAYER2/MAC/extern.h"
#include "LAYER2/MAC/proto.h"
#include "RRC/LITE/extern.h"
#include "PHY_INTERFACE/extern.h"

#ifdef SMBV
#include "PHY/TOOLS/smbv.h"
unsigned short config_frames[4] = {2,9,11,13};
#endif
#include "UTIL/LOG/log_extern.h"
#include "UTIL/OTG/otg_tx.h"
#include "UTIL/OTG/otg_externs.h"
#include "UTIL/MATH/oml.h"
#include "UTIL/LOG/vcd_signal_dumper.h"
#include "UTIL/OPT/opt.h"
#include "enb_config.h"
//#include "PHY/TOOLS/time_meas.h"

#ifndef OPENAIR2
#include "UTIL/OTG/otg_extern.h"
#endif

#if defined(ENABLE_ITTI)
# if defined(ENABLE_USE_MME)
#   include "s1ap_eNB.h"
#ifdef PDCP_USE_NETLINK
#   include "SIMULATION/ETH_TRANSPORT/proto.h"
#endif
# endif
#endif





//#define DEBUG_THREADS 1

//#define USRP_DEBUG 1
struct timing_info_t {
  //unsigned int frame, hw_slot, last_slot, next_slot;
  RTIME time_min, time_max, time_avg, time_last, time_now;
  //unsigned int mbox0, mbox1, mbox2, mbox_target;
  unsigned int n_samples;
} timing_info;

125 126
// Fix per CC openair rf/if device update
// extern openair0_device openair0;
127 128 129 130 131 132 133 134 135 136 137 138 139

#if defined(ENABLE_ITTI)
extern volatile int             start_eNB;
extern volatile int             start_UE;
#endif
extern volatile int                    oai_exit;

extern openair0_config_t openair0_cfg[MAX_CARDS];

extern pthread_cond_t sync_cond;
extern pthread_mutex_t sync_mutex;
extern int sync_var;

140
//pthread_t                       main_eNB_thread;
141 142 143

time_stats_t softmodem_stats_mt; // main thread
time_stats_t softmodem_stats_hw; //  hw acquisition
knopp's avatar
knopp committed
144
time_stats_t softmodem_stats_rxtx_sf; // total tx time
145
time_stats_t softmodem_stats_rx_sf; // total rx time
146 147 148 149 150 151 152 153 154 155 156 157 158
int32_t **rxdata;
int32_t **txdata;

static int                      time_offset[4] = {0,0,0,0};

/* mutex, cond and variable to serialize phy proc TX calls
 * (this mechanism may be relaxed in the future for better
 * performances)
 */
static struct {
  pthread_mutex_t  mutex_phy_proc_tx;
  pthread_cond_t   cond_phy_proc_tx;
  volatile uint8_t phy_proc_CC_id;
159
} sync_phy_proc;
160 161 162 163


void exit_fun(const char* s);

164
void init_eNB(eNB_func_t node_function);
165 166 167 168 169 170 171
void stop_eNB(void);

void do_OFDM_mod_rt(int subframe,PHY_VARS_eNB *phy_vars_eNB)
{

  unsigned int aa,slot_offset, slot_offset_F;
  int dummy_tx_b[7680*4] __attribute__((aligned(32)));
knopp's avatar
knopp committed
172
  int i,j, tx_offset;
173 174
  int slot_sizeF = (phy_vars_eNB->frame_parms.ofdm_symbol_size)*
                   ((phy_vars_eNB->frame_parms.Ncp==1) ? 6 : 7);
knopp's avatar
knopp committed
175 176
  int len,len2;
  int16_t *txdata;
177 178 179

  slot_offset_F = (subframe<<1)*slot_sizeF;

180
  slot_offset = subframe*phy_vars_eNB->frame_parms.samples_per_tti;
181

182 183
  if ((subframe_select(&phy_vars_eNB->frame_parms,subframe)==SF_DL)||
      ((subframe_select(&phy_vars_eNB->frame_parms,subframe)==SF_S))) {
184 185 186
    //    LOG_D(HW,"Frame %d: Generating slot %d\n",frame,next_slot);


187 188 189
    for (aa=0; aa<phy_vars_eNB->frame_parms.nb_antennas_tx; aa++) {
      if (phy_vars_eNB->frame_parms.Ncp == EXTENDED) {
        PHY_ofdm_mod(&phy_vars_eNB->common_vars.txdataF[0][aa][slot_offset_F],
190
                     dummy_tx_b,
191
                     phy_vars_eNB->frame_parms.ofdm_symbol_size,
192
                     6,
193
                     phy_vars_eNB->frame_parms.nb_prefix_samples,
194
                     CYCLIC_PREFIX);
195 196 197
        PHY_ofdm_mod(&phy_vars_eNB->common_vars.txdataF[0][aa][slot_offset_F+slot_sizeF],
                     dummy_tx_b+(phy_vars_eNB->frame_parms.samples_per_tti>>1),
                     phy_vars_eNB->frame_parms.ofdm_symbol_size,
198
                     6,
199
                     phy_vars_eNB->frame_parms.nb_prefix_samples,
200 201
                     CYCLIC_PREFIX);
      } else {
202
        normal_prefix_mod(&phy_vars_eNB->common_vars.txdataF[0][aa][slot_offset_F],
203 204
                          dummy_tx_b,
                          7,
205
                          &(phy_vars_eNB->frame_parms));
206
	// if S-subframe generate first slot only
207 208 209
	if (subframe_select(&phy_vars_eNB->frame_parms,subframe) == SF_DL)
	  normal_prefix_mod(&phy_vars_eNB->common_vars.txdataF[0][aa][slot_offset_F+slot_sizeF],
			    dummy_tx_b+(phy_vars_eNB->frame_parms.samples_per_tti>>1),
210
			    7,
211
			    &(phy_vars_eNB->frame_parms));
212 213 214
      }

      // if S-subframe generate first slot only
215 216
      if (subframe_select(&phy_vars_eNB->frame_parms,subframe) == SF_S)
	len = phy_vars_eNB->frame_parms.samples_per_tti>>1;
217
      else
218
	len = phy_vars_eNB->frame_parms.samples_per_tti;
219 220 221 222 223 224 225
      /*
      for (i=0;i<len;i+=4) {
	dummy_tx_b[i] = 0x100;
	dummy_tx_b[i+1] = 0x01000000;
	dummy_tx_b[i+2] = 0xff00;
	dummy_tx_b[i+3] = 0xff000000;
	}*/
knopp's avatar
knopp committed
226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241
      
      if (slot_offset+time_offset[aa]<0) {
	txdata = (int16_t*)&phy_vars_eNB->common_vars.txdata[0][aa][(LTE_NUMBER_OF_SUBFRAMES_PER_FRAME*phy_vars_eNB->frame_parms.samples_per_tti)+tx_offset];
        len2 = -(slot_offset+time_offset[aa]);
	len2 = (len2>len) ? len : len2;
	for (i=0; i<(len2<<1); i++) {
	  txdata[i] = ((int16_t*)dummy_tx_b)[i]<<openair0_cfg[0].iq_txshift;
	}
	if (len2<len) {
	  txdata = (int16_t*)&phy_vars_eNB->common_vars.txdata[0][aa][0];
	  for (j=0; i<(len<<1); i++,j++) {
	    txdata[j++] = ((int16_t*)dummy_tx_b)[i]<<openair0_cfg[0].iq_txshift;
	  }
	}
      }
      else if ((slot_offset+time_offset[aa]+len)>(LTE_NUMBER_OF_SUBFRAMES_PER_FRAME*phy_vars_eNB->frame_parms.samples_per_tti)) {
242

knopp's avatar
knopp committed
243 244 245 246 247 248 249 250 251 252 253 254 255 256
	tx_offset = (int)slot_offset+time_offset[aa];
	txdata = (int16_t*)&phy_vars_eNB->common_vars.txdata[0][aa][tx_offset];
	len2 = -tx_offset+LTE_NUMBER_OF_SUBFRAMES_PER_FRAME*phy_vars_eNB->frame_parms.samples_per_tti;
	for (i=0; i<(len2<<1); i++) {
	  txdata[i] = ((int16_t*)dummy_tx_b)[i]<<openair0_cfg[0].iq_txshift;
	}
	txdata = (int16_t*)&phy_vars_eNB->common_vars.txdata[0][aa][0];
	for (j=0; i<(len<<1); i++,j++) {
	  txdata[j++] = ((int16_t*)dummy_tx_b)[i]<<openair0_cfg[0].iq_txshift;
	}
      }
      else {
	tx_offset = (int)slot_offset+time_offset[aa];
	txdata = (int16_t*)&phy_vars_eNB->common_vars.txdata[0][aa][tx_offset];
257

knopp's avatar
knopp committed
258 259 260 261 262
	for (i=0; i<(len<<1); i++) {
	  txdata[i] = ((int16_t*)dummy_tx_b)[i]<<openair0_cfg[0].iq_txshift;
	}
      }
      
263 264

     // if S-subframe switch to RX in second subframe
knopp's avatar
knopp committed
265
      /*
266
     if (subframe_select(&phy_vars_eNB->frame_parms,subframe) == SF_S) {
267
       for (i=0; i<len; i++) {
268
	 phy_vars_eNB->common_vars.txdata[0][aa][tx_offset++] = 0x00010001;
269 270
       }
     }
knopp's avatar
knopp committed
271
      */
272 273 274 275
     if ((((phy_vars_eNB->frame_parms.tdd_config==0) ||
	  (phy_vars_eNB->frame_parms.tdd_config==1) ||
	  (phy_vars_eNB->frame_parms.tdd_config==2) ||
	  (phy_vars_eNB->frame_parms.tdd_config==6)) && 
276 277 278 279 280 281
	  (subframe==0)) || (subframe==5)) {
       // turn on tx switch N_TA_offset before
       //LOG_D(HW,"subframe %d, time to switch to tx (N_TA_offset %d, slot_offset %d) \n",subframe,phy_vars_eNB->N_TA_offset,slot_offset);
       for (i=0; i<phy_vars_eNB->N_TA_offset; i++) {
	 tx_offset = (int)slot_offset+time_offset[aa]+i-phy_vars_eNB->N_TA_offset/2;
	 if (tx_offset<0)
282
	   tx_offset += LTE_NUMBER_OF_SUBFRAMES_PER_FRAME*phy_vars_eNB->frame_parms.samples_per_tti;
283
	 
284 285
	 if (tx_offset>=(LTE_NUMBER_OF_SUBFRAMES_PER_FRAME*phy_vars_eNB->frame_parms.samples_per_tti))
	   tx_offset -= LTE_NUMBER_OF_SUBFRAMES_PER_FRAME*phy_vars_eNB->frame_parms.samples_per_tti;
286
	 
287
	 phy_vars_eNB->common_vars.txdata[0][aa][tx_offset] = 0x00000000;
288 289 290 291 292 293 294
       }
     }
    }
  }
}

/*!
knopp's avatar
knopp committed
295
 * \brief The RX UE-specific and TX thread of eNB.
296 297 298
 * \param param is a \ref eNB_proc_t structure which contains the info what to process.
 * \returns a pointer to an int. The storage is not on the heap and must not be freed.
 */
knopp's avatar
knopp committed
299
static void* eNB_thread_rxtx( void* param )
300
{
knopp's avatar
knopp committed
301
  static int eNB_thread_rxtx_status;
302

knopp's avatar
knopp committed
303
  eNB_rxtx_proc_t *proc = (eNB_rxtx_proc_t*)param;
304 305
  FILE  *tx_time_file = NULL;
  char tx_time_name[101];
knopp's avatar
knopp committed
306
  void *txp[PHY_vars_eNB_g[0][0]->frame_parms.nb_antennas_tx]; 
307 308

  if (opp_enabled == 1) {
309
    snprintf(tx_time_name, 100,"/tmp/%s_tx_time_thread_sf", "eNB");
310 311 312
    tx_time_file = fopen(tx_time_name,"w");
  }
  // set default return value
knopp's avatar
knopp committed
313
  eNB_thread_rxtx_status = 0;
314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335

  MSC_START_USE();

#ifdef LOWLATENCY
  struct sched_attr attr;
  unsigned int flags = 0;
  uint64_t runtime  = 850000 ;  
  uint64_t deadline = 1   *  1000000 ; // each tx thread will finish within 1ms
  uint64_t period   = 1   * 10000000; // each tx thread has a period of 10ms from the starting point

  attr.size = sizeof(attr);
  attr.sched_flags = 0;
  attr.sched_nice = 0;
  attr.sched_priority = 0;

  attr.sched_policy   = SCHED_DEADLINE;
  attr.sched_runtime  = runtime;
  attr.sched_deadline = deadline;
  attr.sched_period   = period; 

  if (sched_setattr(0, &attr, flags) < 0 ) {
    perror("[SCHED] eNB tx thread: sched_setattr failed\n");
knopp's avatar
knopp committed
336
    return &eNB_thread_rxtx_status;
337 338
  }

knopp's avatar
knopp committed
339
  LOG_I( HW, "[SCHED] eNB RXn-TXnp4 deadline thread (TID %ld) started on CPU %d\n", gettid(), sched_getcpu() );
340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401

#else //LOW_LATENCY
  int policy, s, j;
  struct sched_param sparam;
  char cpu_affinity[1024];
  cpu_set_t cpuset;

  /* Set affinity mask to include CPUs 1 to MAX_CPUS */
  /* CPU 0 is reserved for UHD threads */
  /* CPU 1 is reserved for all TX threads */
  /* Enable CPU Affinity only if number of CPUs >2 */
  CPU_ZERO(&cpuset);

#ifdef CPU_AFFINITY
  if (get_nprocs() > 2)
  {
    for (j = 1; j < get_nprocs(); j++)
        CPU_SET(j, &cpuset);
    s = pthread_setaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset);
    if (s != 0)
    {
      perror( "pthread_setaffinity_np");
      exit_fun("Error setting processor affinity");
    }
  }
#endif //CPU_AFFINITY

  /* Check the actual affinity mask assigned to the thread */

  s = pthread_getaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset);
  if (s != 0)
  {
    perror( "pthread_getaffinity_np");
    exit_fun("Error getting processor affinity ");
  }
  memset(cpu_affinity,0,sizeof(cpu_affinity));
  for (j = 0; j < CPU_SETSIZE; j++)
     if (CPU_ISSET(j, &cpuset))
     {  
        char temp[1024];
        sprintf (temp, " CPU_%d", j);
        strcat(cpu_affinity, temp);
     }

  memset(&sparam, 0 , sizeof (sparam));
  sparam.sched_priority = sched_get_priority_max(SCHED_FIFO)-1;
  policy = SCHED_FIFO ; 
  
  s = pthread_setschedparam(pthread_self(), policy, &sparam);
  if (s != 0)
     {
     perror("pthread_setschedparam : ");
     exit_fun("Error setting thread priority");
     }
  s = pthread_getschedparam(pthread_self(), &policy, &sparam);
  if (s != 0)
   {
     perror("pthread_getschedparam : ");
     exit_fun("Error getting thread priority");

   }

402
 LOG_I( HW, "[SCHED][eNB] TX thread started on CPU %d TID %ld, sched_policy = %s , priority = %d, CPU Affinity=%s \n",sched_getcpu(),gettid(),
403 404 405 406 407 408 409 410 411 412 413 414
                   (policy == SCHED_FIFO)  ? "SCHED_FIFO" :
                   (policy == SCHED_RR)    ? "SCHED_RR" :
                   (policy == SCHED_OTHER) ? "SCHED_OTHER" :
                   "???",
                   sparam.sched_priority, cpu_affinity );


#endif //LOW_LATENCY


  mlockall(MCL_CURRENT | MCL_FUTURE);

415

416 417
  while (!oai_exit) {

knopp's avatar
knopp committed
418
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_RXTX0+(proc->subframe_rx&1), 0 );
419

knopp's avatar
knopp committed
420 421
    if (pthread_mutex_lock(&proc->mutex_rxtx) != 0) {
      LOG_E( PHY, "[SCHED][eNB] error locking mutex for eNB RXn-TXnp4\n");
422 423 424 425
      exit_fun("nothing to add");
      break;
    }

knopp's avatar
knopp committed
426
    while (proc->instance_cnt_rxtx < 0) {
427
      // most of the time the thread is waiting here
knopp's avatar
knopp committed
428 429
      // proc->instance_cnt_rxtx is -1
      pthread_cond_wait( &proc->cond_rxtx, &proc->mutex_rxtx ); // this unlocks mutex_rxtx while waiting and then locks it again
430 431
    }

knopp's avatar
knopp committed
432
    if (pthread_mutex_unlock(&proc->mutex_rxtx) != 0) {
433
      LOG_E(PHY,"[SCHED][eNB] error unlocking mutex for eNB TX\n");
434 435 436 437
      exit_fun("nothing to add");
      break;
    }

knopp's avatar
knopp committed
438 439
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_RXTX0+(proc->subframe_rx&1), 1 );
    start_meas( &softmodem_stats_rxtx_sf );
440
  
441 442
    if (oai_exit) break;

knopp's avatar
knopp committed
443 444 445 446 447 448 449 450 451
    // UE-specific RX processing for subframe n
    if (PHY_vars_eNB_g[0][proc->CC_id]->node_function != NGFI_RRU_IF4) {
      VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_PROCEDURES_ENB_RX_UESPEC, 1 );
      // this is the ue-specific processing for the subframe and can be multi-threaded later
      phy_procedures_eNB_uespec_RX(PHY_vars_eNB_g[0][proc->CC_id], proc, 0, no_relay );
      VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_PROCEDURES_ENB_RX_UESPEC, 0 );
    }

    // TX processing for subframe n+4
452 453 454 455
    if (((PHY_vars_eNB_g[0][proc->CC_id]->frame_parms.frame_type == TDD) &&
         ((subframe_select(&PHY_vars_eNB_g[0][proc->CC_id]->frame_parms,proc->subframe_tx) == SF_DL) ||
          (subframe_select(&PHY_vars_eNB_g[0][proc->CC_id]->frame_parms,proc->subframe_tx) == SF_S))) ||
        (PHY_vars_eNB_g[0][proc->CC_id]->frame_parms.frame_type == FDD)) {
456 457 458
      /* run PHY TX procedures the one after the other for all CCs to avoid race conditions
       * (may be relaxed in the future for performance reasons)
       */
knopp's avatar
knopp committed
459
      
460 461
      if (pthread_mutex_lock(&sync_phy_proc.mutex_phy_proc_tx) != 0) {
        LOG_E(PHY, "[SCHED][eNB] error locking PHY proc mutex for eNB TX\n");
462 463 464
        exit_fun("nothing to add");
        break;
      }
knopp's avatar
knopp committed
465
      // wait for our turn or oai_exit
466 467 468
      while (sync_phy_proc.phy_proc_CC_id != proc->CC_id && !oai_exit) {
        pthread_cond_wait(&sync_phy_proc.cond_phy_proc_tx,
                          &sync_phy_proc.mutex_phy_proc_tx);
469 470
      }

471 472
      if (pthread_mutex_unlock(&sync_phy_proc.mutex_phy_proc_tx) != 0) {
        LOG_E(PHY, "[SCHED][eNB] error unlocking PHY proc mutex for eNB TX\n");
473 474 475
        exit_fun("nothing to add");
      }

knopp's avatar
knopp committed
476
      
knopp's avatar
knopp committed
477 478 479

      VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME( VCD_SIGNAL_DUMPER_VARIABLES_FRAME_NUMBER_TX_ENB, proc->frame_tx );
      VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME( VCD_SIGNAL_DUMPER_VARIABLES_SUBFRAME_NUMBER_TX_ENB, proc->subframe_tx );
480 481
      if (oai_exit)
        break;
482
      if (PHY_vars_eNB_g[0][proc->CC_id]->node_function != NGFI_RRU_IF4) { 
knopp's avatar
knopp committed
483
	phy_procedures_eNB_TX(PHY_vars_eNB_g[0][proc->CC_id], proc, 0, no_relay, NULL );
484 485
	
	/* we're done, let the next one proceed */
486 487
	if (pthread_mutex_lock(&sync_phy_proc.mutex_phy_proc_tx) != 0) {
	  LOG_E(PHY, "[SCHED][eNB] error locking PHY proc mutex for eNB TX proc\n");
488 489 490
	  exit_fun("nothing to add");
	  break;
	}
491 492 493 494 495
	sync_phy_proc.phy_proc_CC_id++;
	sync_phy_proc.phy_proc_CC_id %= MAX_NUM_CCs;
	pthread_cond_broadcast(&sync_phy_proc.cond_phy_proc_tx);
	if (pthread_mutex_unlock(&sync_phy_proc.mutex_phy_proc_tx) != 0) {
	  LOG_E(PHY, "[SCHED][eNB] error unlocking PHY proc mutex for eNB TX proc\n");
496 497 498
	  exit_fun("nothing to add");
	  break;
	}
499 500 501 502 503
      } else {

        /// **** recv_IF4 of txdataF from RCC **** ///        
        //recv_IF4( eNB, proc, packet_type, symbol_number);        
        
504 505 506
      }
    }

507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540
  // eNodeB_3GPP and RRU create txdata and write to RF device
  if (PHY_vars_eNB_g[0][proc->CC_id]->node_function != NGFI_RCC_IF4) {
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_ENB_SFGEN , 1 );
    do_OFDM_mod_rt( proc->subframe_tx, PHY_vars_eNB_g[0][proc->CC_id] );
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_ENB_SFGEN , 0 );
  /*
    short *txdata = (short*)&PHY_vars_eNB_g[0][proc->CC_id]->common_vars.txdata[0][0][proc->subframe_tx*PHY_vars_eNB_g[0][proc->CC_id]->frame_parms.samples_per_tti];
    int i;
    for (i=0;i<PHY_vars_eNB_g[0][proc->CC_id]->frame_parms.samples_per_tti*2;i+=8) {
    txdata[i] = 2047;
    txdata[i+1] = 0;
    txdata[i+2] = 0;
    txdata[i+3] = 2047;
    txdata[i+4] = -2047;
    txdata[i+5] = 0;
    txdata[i+6] = 0;
    txdata[i+7] = -2047;      }
  */


    // Transmit TX buffer based on timestamp from RX
  
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_TRX_WRITE, 1 );
    // prepare tx buffer pointers
    int i;
    for (i=0; i<PHY_vars_eNB_g[0][0]->frame_parms.nb_antennas_tx; i++)
  txp[i] = (void*)&PHY_vars_eNB_g[0][0]->common_vars.txdata[0][i][proc->subframe_tx*PHY_vars_eNB_g[0][0]->frame_parms.samples_per_tti];
    // if symb_written < spp ==> error 
    PHY_vars_eNB_g[0][proc->CC_id]->rfdevice.trx_write_func(&PHY_vars_eNB_g[0][proc->CC_id]->rfdevice,
          (proc->timestamp_tx-openair0_cfg[0].tx_sample_advance),
          txp,
          PHY_vars_eNB_g[0][0]->frame_parms.samples_per_tti,
          PHY_vars_eNB_g[0][0]->frame_parms.nb_antennas_tx,
          1);
knopp's avatar
knopp committed
541
    
knopp's avatar
knopp committed
542

knopp's avatar
knopp committed
543

544
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_TRX_WRITE, 0 );
knopp's avatar
knopp committed
545

546 547 548 549 550 551 552 553
    VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME( VCD_SIGNAL_DUMPER_VARIABLES_TRX_TST, (proc->timestamp_tx-openair0_cfg[0].tx_sample_advance)&0xffffffff );

  } else {  // RCC sends the txdataF using send_IF4 function

    /// **** send_IF4 of txdataF to RRU **** ///    
    //send_IF4(PHY_vars_eNB_g[0][proc->CC_id], proc);
    
  }
554

knopp's avatar
knopp committed
555
    if (pthread_mutex_lock(&proc->mutex_rxtx) != 0) {
556
      LOG_E( PHY, "[SCHED][eNB] error locking mutex for eNB TX proc\n");
557 558 559 560
      exit_fun("nothing to add");
      break;
    }

knopp's avatar
knopp committed
561
    proc->instance_cnt_rxtx--;
562

knopp's avatar
knopp committed
563
    if (pthread_mutex_unlock(&proc->mutex_rxtx) != 0) {
564
      LOG_E( PHY, "[SCHED][eNB] error unlocking mutex for eNB TX proc\n");
565 566 567 568
      exit_fun("nothing to add");
      break;
    }

knopp's avatar
knopp committed
569
    stop_meas( &softmodem_stats_rxtx_sf );
570 571
#ifdef LOWLATENCY
    if (opp_enabled){
knopp's avatar
knopp committed
572 573
      if(softmodem_stats_rxtx_sf.diff_now/(cpuf) > attr.sched_runtime){
	VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME( VCD_SIGNAL_DUMPER_VARIABLES_RUNTIME_TX_ENB, (softmodem_stats_rxtx_sf.diff_now/cpuf - attr.sched_runtime)/1000000.0);
574 575 576
      }
    }
#endif 
knopp's avatar
knopp committed
577
    print_meas_now(&softmodem_stats_rxtx_sf,"eNB_TX_SF",tx_time_file);
578 579 580 581 582

  }



knopp's avatar
knopp committed
583
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_RXTX0+(proc->subframe_rx&1), 0 );
584 585

#ifdef DEBUG_THREADS
586
  printf( "Exiting eNB thread TX\n");
587 588
#endif

knopp's avatar
knopp committed
589 590
  eNB_thread_rxtx_status = 0;
  return &eNB_thread_rxtx_status;
591 592
}

593 594 595
#if defined(ENABLE_ITTI)
static void wait_system_ready (char *message, volatile int *start_flag)

596 597 598 599 600 601 602 603 604 605 606 607
  /* Wait for eNB application initialization to be complete (eNB registration to MME) */
{
  static char *indicator[] = {".    ", "..   ", "...  ", ".... ", ".....",
			      " ....", "  ...", "   ..", "    .", "     "
  };
  int i = 0;
  
  while ((!oai_exit) && (*start_flag == 0)) {
    LOG_N(EMU, message, indicator[i]);
    fflush(stdout);
    i = (i + 1) % (sizeof(indicator) / sizeof(indicator[0]));
    usleep(200000);
608
  }
609 610
  
  LOG_D(EMU,"\n");
611
}
612

613
#endif
614 615

/*!
knopp's avatar
knopp committed
616
 * \brief The RX common thread of eNB.
617 618 619
 * \param param is a \ref eNB_proc_t structure which contains the info what to process.
 * \returns a pointer to an int. The storage is not on the heap and must not be freed.
 */
knopp's avatar
knopp committed
620
static void* eNB_thread_rx_common( void* param )
621
{
622
  static int eNB_thread_rx_status;
623 624

  eNB_proc_t *proc = (eNB_proc_t*)param;
625 626
  PHY_VARS_eNB *eNB = PHY_vars_eNB_g[0][proc->CC_id];
  LTE_DL_FRAME_PARMS *fp = &eNB->frame_parms;
627 628 629 630 631 632

  FILE  *rx_time_file = NULL;
  char rx_time_name[101];
  //int i;

  if (opp_enabled == 1){
633
    snprintf(rx_time_name, 100,"/tmp/%s_rx_time_thread_sf", "eNB");
634 635 636
    rx_time_file = fopen(rx_time_name,"w");
  }
  // set default return value
637
  eNB_thread_rx_status = 0;
638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659

  MSC_START_USE();

#ifdef LOWLATENCY
  struct sched_attr attr;
  unsigned int flags = 0;
  uint64_t runtime  = 870000 ;
  uint64_t deadline = 1   *  1000000;
  uint64_t period   = 1   * 10000000; // each rx thread has a period of 10ms from the starting point
 
  attr.size = sizeof(attr);
  attr.sched_flags = 0;
  attr.sched_nice = 0;
  attr.sched_priority = 0;

  attr.sched_policy = SCHED_DEADLINE;
  attr.sched_runtime  = runtime;
  attr.sched_deadline = deadline;
  attr.sched_period   = period; 

  if (sched_setattr(0, &attr, flags) < 0 ) {
    perror("[SCHED] eNB RX sched_setattr failed\n");
660
    return &eNB_thread_rx_status;
661 662
  }

663
  LOG_I( HW, "[SCHED] eNB RX deadline thread (TID %ld) started on CPU %d\n", gettid(), sched_getcpu() );
664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709
#else // LOW_LATENCY
  int policy, s, j;
  struct sched_param sparam;
  char cpu_affinity[1024];
  cpu_set_t cpuset;

  /* Set affinity mask to include CPUs 1 to MAX_CPUS */
  /* CPU 0 is reserved for UHD */
  /* CPU 1 is reserved for all TX threads */
  /* CPU 2..MAX_CPUS is reserved for all RX threads */
  /* Set CPU Affinity only if number of CPUs >2 */
  CPU_ZERO(&cpuset);
#ifdef CPU_AFFINITY
  if (get_nprocs() >2)
  {
    for (j = 1; j < get_nprocs(); j++)
       CPU_SET(j, &cpuset);
  
    s = pthread_setaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset);
    if (s != 0)
    {
      perror( "pthread_setaffinity_np");  
      exit_fun (" Error setting processor affinity :");
    }
  }
#endif //CPU_AFFINITY
  /* Check the actual affinity mask assigned to the thread */

  s = pthread_getaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset);
  if (s != 0)
  {
     perror ("pthread_getaffinity_np");
     exit_fun (" Error getting processor affinity :");
  }
  memset(cpu_affinity,0, sizeof(cpu_affinity));

  for (j = 0; j < CPU_SETSIZE; j++)
     if (CPU_ISSET(j, &cpuset))
     {  
        char temp[1024];
        sprintf (temp, " CPU_%d", j);
        strcat(cpu_affinity, temp);
     }


  memset(&sparam, 0 , sizeof (sparam)); 
knopp's avatar
knopp committed
710
  sparam.sched_priority = sched_get_priority_max(SCHED_FIFO);
711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729

  policy = SCHED_FIFO ; 
  s = pthread_setschedparam(pthread_self(), policy, &sparam);
  if (s != 0)
     {
     perror("pthread_setschedparam : ");
     exit_fun("Error setting thread priority");
     }

  memset(&sparam, 0 , sizeof (sparam));

  s = pthread_getschedparam(pthread_self(), &policy, &sparam);
  if (s != 0)
   {
     perror("pthread_getschedparam");
     exit_fun("Error getting thread priority");
   }


730
  LOG_I( HW, "[SCHED][eNB] RX thread started on CPU %d TID %ld, sched_policy = %s, priority = %d, CPU Affinity = %s\n", sched_getcpu(),gettid(),
731 732 733 734 735 736 737 738 739 740 741 742
	 (policy == SCHED_FIFO)  ? "SCHED_FIFO" :
	 (policy == SCHED_RR)    ? "SCHED_RR" :
	 (policy == SCHED_OTHER) ? "SCHED_OTHER" :
	 "???",
	 sparam.sched_priority, cpu_affinity);
  
  
#endif // LOWLATENCY

 mlockall(MCL_CURRENT | MCL_FUTURE);


743
 // wait for top-level synchronization and do one acquisition to get timestamp for setting frame/subframe of TX and RX threads
knopp's avatar
knopp committed
744
 printf( "waiting for sync (eNB_thread_rx_common)\n");
745 746 747 748 749 750 751 752 753 754 755 756
 pthread_mutex_lock( &sync_mutex );

 while (sync_var<0)
   pthread_cond_wait( &sync_cond, &sync_mutex );
 
 pthread_mutex_unlock(&sync_mutex);
 
 printf( "got sync (eNB_thread)\n" );
 
#if defined(ENABLE_ITTI)
  wait_system_ready ("Waiting for eNB application to be ready %s\r", &start_eNB);
#endif 
757 758
  
  // Start RF device for this CC
759
  if (eNB->node_function != NGFI_RCC_IF4) {
760 761 762 763 764
    if (eNB->rfdevice.trx_start_func(&eNB->rfdevice) != 0 ) 
      LOG_E(HW,"Could not start the RF device\n");
  }
    
  // Start IF device for this CC
765
  if (eNB->node_function != eNodeB_3GPP) {
766 767 768 769
    if (eNB->ifdevice.trx_start_func(&eNB->ifdevice) != 0 ) 
      LOG_E(HW,"Could not start the IF device\n");
  }

770
 // This is a forever while loop, it loops over subframes which are scheduled by incoming samples from HW devices
771 772 773
 while (!oai_exit) {
   
   
knopp's avatar
knopp committed
774
   VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_RX, 0 );
knopp's avatar
knopp committed
775
   VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_PROCEDURES_ENB_RX_COMMON, 0 );
776
   start_meas( &softmodem_stats_rx_sf );
777 778 779
   
   if (oai_exit) break;
   
780 781
   if ((((fp->frame_type == TDD )&&(subframe_select(fp,proc->subframe_rx)==SF_UL)) ||
	(fp->frame_type == FDD))) {
knopp's avatar
knopp committed
782
     VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_PROCEDURES_ENB_RX_COMMON, 1 );
knopp's avatar
knopp committed
783
     // this spawns the prach inside and updates the frame and subframe counters
784
     phy_procedures_eNB_common_RX(eNB, 0);
785
     
knopp's avatar
knopp committed
786
     
knopp's avatar
knopp committed
787
     VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_PROCEDURES_ENB_RX_COMMON, 0 );
788
   }
knopp's avatar
knopp committed
789

knopp's avatar
knopp committed
790 791 792
   // choose even or odd thread for RXn-TXnp4 processing 
   eNB_rxtx_proc_t *proc_rxtx = &proc->proc_rxtx[proc->subframe_rx&1];

knopp's avatar
knopp committed
793 794
   // wake up TX for subframe n+4
   // lock the TX mutex and make sure the thread is ready
knopp's avatar
knopp committed
795 796 797
   if (pthread_mutex_lock(&proc_rxtx->mutex_rxtx) != 0) {
     LOG_E( PHY, "[eNB] ERROR pthread_mutex_lock for eNB TX thread %d (IC %d)\n", proc_rxtx->instance_cnt_rxtx );
     exit_fun( "error locking mutex_rxtx" );
knopp's avatar
knopp committed
798
     break;
799
   }
knopp's avatar
knopp committed
800
   int cnt_rxtx = ++proc_rxtx->instance_cnt_rxtx;
knopp's avatar
knopp committed
801 802 803
   // We have just received and processed the common part of a subframe, say n. 
   // TS_rx is the last received timestamp (start of 1st slot), TS_tx is the desired 
   // transmitted timestamp of the next TX slot (first).
knopp's avatar
knopp committed
804
   // The last (TS_rx mod samples_pexr_frame) was n*samples_per_tti, 
knopp's avatar
knopp committed
805 806
   // we want to generate subframe (n+3), so TS_tx = TX_rx+3*samples_per_tti,
   // and proc->subframe_tx = proc->subframe_rx+3
knopp's avatar
knopp committed
807 808 809 810 811
   proc_rxtx->timestamp_tx = proc->timestamp_rx + (4*fp->samples_per_tti);
   proc_rxtx->frame_rx     = proc->frame_rx;
   proc_rxtx->subframe_rx  = proc->subframe_rx;
   proc_rxtx->frame_tx     = (proc_rxtx->subframe_rx > 5) ? (proc_rxtx->frame_rx+1)&1023 : proc_rxtx->frame_rx;
   proc_rxtx->subframe_tx  = (proc_rxtx->subframe_rx + 4)%10;
knopp's avatar
knopp committed
812
   
knopp's avatar
knopp committed
813
   pthread_mutex_unlock( &proc_rxtx->mutex_rxtx );
knopp's avatar
knopp committed
814
   
knopp's avatar
knopp committed
815
   if (cnt_rxtx == 0){
knopp's avatar
knopp committed
816
     // the thread was presumably waiting where it should and can now be woken up
knopp's avatar
knopp committed
817 818
     if (pthread_cond_signal(&proc_rxtx->cond_rxtx) != 0) {
       LOG_E( PHY, "[eNB] ERROR pthread_cond_signal for eNB RXn-TXnp4 thread\n");
knopp's avatar
knopp committed
819 820 821 822
       exit_fun( "ERROR pthread_cond_signal" );
       break;
     }
   } else {
knopp's avatar
knopp committed
823
     LOG_W( PHY,"[eNB] Frame %d, eNB RXn-TXnp4 thread busy!! (cnt_rxtx %i)\n", proc_rxtx->frame_tx, cnt_rxtx );
knopp's avatar
knopp committed
824 825 826
     exit_fun( "TX thread busy" );
     break;
   }       
827
   
knopp's avatar
knopp committed
828
   stop_meas( &softmodem_stats_rxtx_sf );
829 830
#ifdef LOWLATENCY
   if (opp_enabled){
knopp's avatar
knopp committed
831 832
     if(softmodem_stats_rxtx_sf.diff_now/(cpuf) > attr.sched_runtime){
       VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME( VCD_SIGNAL_DUMPER_VARIABLES_RUNTIME_RXTX_ENB, (softmodem_stats_rxtx_sf.diff_now/cpuf - attr.sched_runtime)/1000000.0);
833 834 835
     }
   }
#endif // LOWLATENCY  
836
   print_meas_now(&softmodem_stats_rx_sf,"eNB_RX_SF", rx_time_file);
knopp's avatar
knopp committed
837
   VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_RXTX0+(proc->subframe_rx&1), 0 );
knopp's avatar
knopp committed
838 839 840
   

 }
knopp's avatar
knopp committed
841 842
 
 
843
#ifdef DEBUG_THREADS
knopp's avatar
knopp committed
844
 printf( "Exiting eNB thread RXn-TXnp4\n");
845
#endif
knopp's avatar
knopp committed
846 847 848
 
 eNB_thread_rx_status = 0;
 return &eNB_thread_rx_status;
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 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938
}



/*!
 * \brief The prach receive thread of eNB.
 * \param param is a \ref eNB_proc_t structure which contains the info what to process.
 * \returns a pointer to an int. The storage is not on the heap and must not be freed.
 */
static void* eNB_thread_prach( void* param )
{
  static int eNB_thread_prach_status;

  eNB_proc_t *proc = (eNB_proc_t*)param;
  PHY_VARS_eNB *eNB= PHY_vars_eNB_g[0][proc->CC_id];
  // set default return value
  eNB_thread_prach_status = 0;

  MSC_START_USE();

    
#ifdef LOWLATENCY
  struct sched_attr attr;
  unsigned int flags = 0;
  uint64_t runtime  = 870000 ;
  uint64_t deadline = 1   *  1000000;
  uint64_t period   = 1   * 10000000; // each prach thread has a period of 10ms from the starting point
 
  attr.size = sizeof(attr);
  attr.sched_flags = 0;
  attr.sched_nice = 0;
  attr.sched_priority = 0;

  attr.sched_policy = SCHED_DEADLINE;
  attr.sched_runtime  = runtime;
  attr.sched_deadline = deadline;
  attr.sched_period   = period; 

  if (sched_setattr(0, &attr, flags) < 0 ) {
    perror("[SCHED] eNB PRACH sched_setattr failed\n");
    return &eNB_thread_prach_status;
  }

  LOG_I( HW, "[SCHED] eNB PRACH deadline thread (TID %ld) started on CPU %d\n", 0, gettid(), sched_getcpu() );
#else // LOW_LATENCY
  int policy, s, j;
  struct sched_param sparam;
  char cpu_affinity[1024];
  cpu_set_t cpuset;

  /* Set affinity mask to include CPUs 1 to MAX_CPUS */
  /* CPU 0 is reserved for UHD */
  /* CPU 1 is reserved for all TX threads */
  /* CPU 2..MAX_CPUS is reserved for all RX threads */
  /* Set CPU Affinity only if number of CPUs >2 */
  CPU_ZERO(&cpuset);
#ifdef CPU_AFFINITY
  if (get_nprocs() >2)
  {
    for (j = 1; j < get_nprocs(); j++)
       CPU_SET(j, &cpuset);
  
    s = pthread_setaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset);
    if (s != 0)
    {
      perror( "pthread_setaffinity_np");  
      exit_fun (" Error setting processor affinity :");
    }
  }
#endif //CPU_AFFINITY
  /* Check the actual affinity mask assigned to the thread */

  s = pthread_getaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset);
  if (s != 0)
  {
     perror ("pthread_getaffinity_np");
     exit_fun (" Error getting processor affinity :");
  }
  memset(cpu_affinity,0, sizeof(cpu_affinity));

  for (j = 0; j < CPU_SETSIZE; j++)
     if (CPU_ISSET(j, &cpuset))
     {  
        char temp[1024];
        sprintf (temp, " CPU_%d", j);
        strcat(cpu_affinity, temp);
     }


  memset(&sparam, 0 , sizeof (sparam)); 
knopp's avatar
knopp committed
939
  sparam.sched_priority = sched_get_priority_max(SCHED_FIFO)-2;
940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955

  policy = SCHED_FIFO ; 
  s = pthread_setschedparam(pthread_self(), policy, &sparam);
  if (s != 0)
     {
     perror("pthread_setschedparam : ");
     exit_fun("Error setting thread priority");
     }

  memset(&sparam, 0 , sizeof (sparam));

  s = pthread_getschedparam(pthread_self(), &policy, &sparam);
  if (s != 0)
   {
     perror("pthread_getschedparam");
     exit_fun("Error getting thread priority");
956
   }
957 958


knopp's avatar
knopp committed
959
  LOG_I( HW, "[SCHED][eNB] PRACH thread started on CPU %d TID %ld, sched_policy = %s, priority = %d, CPU Affinity = %s\n", sched_getcpu(),gettid(),
960 961 962 963 964 965 966 967 968 969 970 971 972 973
	 (policy == SCHED_FIFO)  ? "SCHED_FIFO" :
	 (policy == SCHED_RR)    ? "SCHED_RR" :
	 (policy == SCHED_OTHER) ? "SCHED_OTHER" :
	 "???",
	 sparam.sched_priority, cpu_affinity);
  
  
#endif // LOWLATENCY

 mlockall(MCL_CURRENT | MCL_FUTURE);


 while (!oai_exit) {
 
974
   
975 976 977
   if (oai_exit) break;
        
   if (pthread_mutex_lock(&proc->mutex_prach) != 0) {
978
     LOG_E( PHY, "[SCHED][eNB] error locking mutex for eNB PRACH\n");
979 980 981
     exit_fun( "error locking mutex" );
     break;
   }
982 983 984

   while (proc->instance_cnt_prach < 0) {
     // most of the time the thread is waiting here
knopp's avatar
knopp committed
985 986
     // proc->instance_cnt_prach is -1
     pthread_cond_wait( &proc->cond_prach, &proc->mutex_prach ); // this unlocks mutex_rxtx while waiting and then locks it again
987 988 989
   }

   if (pthread_mutex_unlock(&proc->mutex_prach) != 0) {
990
     LOG_E( PHY, "[SCHED][eNB] error unlocking mutex for eNB PRACH\n");
991 992 993 994
     exit_fun( "error unlocking mutex" );
     break;
   }
   
995
   VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_ENB_PRACH_RX,1);
996
   prach_procedures(eNB,0);
997 998 999
   VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_PHY_ENB_PRACH_RX,0);
    
   if (pthread_mutex_lock(&proc->mutex_prach) != 0) {
1000
     LOG_E( PHY, "[SCHED][eNB] error locking mutex for eNB PRACH proc %d\n");
1001 1002 1003
     exit_fun( "error locking mutex" );
     break;
   }
1004
   
1005
   proc->instance_cnt_prach--;
1006
   
1007
   if (pthread_mutex_unlock(&proc->mutex_prach) != 0) {
1008
     LOG_E( PHY, "[SCHED][eNB] error unlocking mutex for eNB RX proc %d\n");
1009 1010
     exit_fun( "error unlocking mutex" );
     break;
1011
   }
1012
 }
1013 1014

#ifdef DEBUG_THREADS
1015
  printf( "Exiting eNB thread PRACH\n");
1016 1017
#endif

1018 1019
  eNB_thread_prach_status = 0;
  return &eNB_thread_prach_status;
1020 1021 1022 1023 1024 1025 1026 1027 1028
}




void init_eNB_proc(void)
{
  int i;
  int CC_id;
1029 1030
  PHY_VARS_eNB *eNB;
  eNB_proc_t *proc;
knopp's avatar
knopp committed
1031
  eNB_rxtx_proc_t *proc_rxtx;
1032 1033

  for (CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
1034 1035
    eNB = PHY_vars_eNB_g[0][CC_id];

1036 1037
    
    proc = &eNB->proc;
knopp's avatar
knopp committed
1038
    proc_rxtx = proc->proc_rxtx;
1039
#ifndef LOWLATENCY 
1040 1041 1042
    /*  
	pthread_attr_init( &attr_eNB_proc_tx[CC_id][i] );
	if (pthread_attr_setstacksize( &attr_eNB_proc_tx[CC_id][i], 64 *PTHREAD_STACK_MIN ) != 0)
1043
        perror("[ENB_PROC_TX] setting thread stack size failed\n");
1044 1045 1046
	
	pthread_attr_init( &attr_eNB_proc_rx[CC_id][i] );
	if (pthread_attr_setstacksize( &attr_eNB_proc_rx[CC_id][i], 64 * PTHREAD_STACK_MIN ) != 0)
1047
        perror("[ENB_PROC_RX] setting thread stack size failed\n");
1048 1049
    */
    // set the kernel scheduling policy and priority
knopp's avatar
knopp committed
1050 1051 1052 1053 1054 1055
    proc_rxtx[0].sched_param_rxtx.sched_priority = sched_get_priority_max(SCHED_FIFO)-1; //OPENAIR_THREAD_PRIORITY;
    pthread_attr_setschedparam  (&proc_rxtx[0].attr_rxtx, &proc_rxtx[0].sched_param_rxtx);
    pthread_attr_setschedpolicy (&proc_rxtx[0].attr_rxtx, SCHED_FIFO);
    proc_rxtx[1].sched_param_rxtx.sched_priority = sched_get_priority_max(SCHED_FIFO)-1; //OPENAIR_THREAD_PRIORITY;
    pthread_attr_setschedparam  (&proc_rxtx[1].attr_rxtx, &proc_rxtx[1].sched_param_rxtx);
    pthread_attr_setschedpolicy (&proc_rxtx[1].attr_rxtx, SCHED_FIFO);
1056
    
knopp's avatar
knopp committed
1057
    proc->sched_param_rx.sched_priority = sched_get_priority_max(SCHED_FIFO); //OPENAIR_THREAD_PRIORITY;
1058 1059 1060 1061 1062 1063 1064 1065
    pthread_attr_setschedparam  (&proc->attr_rx, &proc->sched_param_rx);
    pthread_attr_setschedpolicy (&proc->attr_rx, SCHED_FIFO);
    
    proc->sched_param_prach.sched_priority = sched_get_priority_max(SCHED_FIFO)-1; //OPENAIR_THREAD_PRIORITY;
    pthread_attr_setschedparam  (&proc->attr_prach, &proc->sched_param_prach);
    pthread_attr_setschedpolicy (&proc->attr_prach, SCHED_FIFO);
    
    printf("Setting OS scheduler to SCHED_FIFO for eNB [cc%d][thread%d] \n",CC_id, i);
1066
#endif
knopp's avatar
knopp committed
1067 1068
    proc_rxtx[0].instance_cnt_rxtx = -1;
    proc_rxtx[1].instance_cnt_rxtx = -1;
1069 1070
    proc->instance_cnt_prach = -1;
    proc->CC_id = CC_id;
1071 1072 1073

    proc->first_rx=1;

knopp's avatar
knopp committed
1074 1075 1076 1077 1078
    pthread_mutex_init( &proc_rxtx[0].mutex_rxtx, NULL);
    pthread_mutex_init( &proc_rxtx[1].mutex_rxtx, NULL);
    pthread_mutex_init( &proc->mutex_prach, NULL);
    pthread_cond_init( &proc_rxtx[0].cond_rxtx, NULL);
    pthread_cond_init( &proc_rxtx[1].cond_rxtx, NULL);
1079
    pthread_cond_init( &proc->cond_prach, NULL);
1080
#ifndef LOWLATENCY
knopp's avatar
knopp committed
1081 1082 1083
    pthread_create( &proc_rxtx[0].pthread_rxtx, &proc_rxtx[0].attr_rxtx, eNB_thread_rxtx, &proc_rxtx[0] );
    pthread_create( &proc_rxtx[1].pthread_rxtx, &proc_rxtx[1].attr_rxtx, eNB_thread_rxtx, &proc_rxtx[1] );
    pthread_create( &proc->pthread_rx, &proc->attr_rx, eNB_thread_rx_common, &eNB->proc );
1084
    pthread_create( &proc->pthread_prach, &proc->attr_prach, eNB_thread_prach, &eNB->proc );
1085
#else 
knopp's avatar
knopp committed
1086 1087 1088
    pthread_create( &proc_rxtx[0].pthread_rxtx, NULL, eNB_thread_rxtx, &eNB->proc_rxtx[0] );
    pthread_create( &proc_rxtx[1].pthread_rxtx, NULL, eNB_thread_rxtx, &eNB->proc_rxtx[1] );
    pthread_create( &proc->pthread_rx, NULL, eNB_thread_rx_common, &eNB->proc );
1089
    pthread_create( &proc->pthread_prach, NULL, eNB_thread_prach, &eNB->proc );
1090
#endif
1091
    char name[16];
knopp's avatar
knopp committed
1092 1093 1094 1095
    snprintf( name, sizeof(name), "RXTX0 %d", i );
    pthread_setname_np( proc_rxtx[0].pthread_rxtx, name );
    snprintf( name, sizeof(name), "RXTX1 %d", i );
    pthread_setname_np( proc_rxtx[1].pthread_rxtx, name );
1096 1097
    snprintf( name, sizeof(name), "RX %d", i );
    pthread_setname_np( proc->pthread_rx, name );
1098
  }
1099 1100
  
  
1101
  /* setup PHY proc TX sync mechanism */
1102 1103 1104
  pthread_mutex_init(&sync_phy_proc.mutex_phy_proc_tx, NULL);
  pthread_cond_init(&sync_phy_proc.cond_phy_proc_tx, NULL);
  sync_phy_proc.phy_proc_CC_id = 0;
1105 1106
}

1107

1108 1109 1110 1111 1112 1113
/*!
 * \brief Terminate eNB TX and RX threads.
 */
void kill_eNB_proc(void)
{
  int *status;
1114 1115
  PHY_VARS_eNB *eNB;
  eNB_proc_t *proc;
knopp's avatar
knopp committed
1116
  eNB_rxtx_proc_t *proc_rxtx;
1117 1118
  for (int CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
    eNB=PHY_vars_eNB_g[0][CC_id];
1119 1120
    
    proc = &eNB->proc;
knopp's avatar
knopp committed
1121
    proc_rxtx = &proc->proc_rxtx[0];
1122
    
1123
#ifdef DEBUG_THREADS
1124
    printf( "Killing TX CC_id %d thread %d\n", CC_id, i );
1125
#endif
1126
    
knopp's avatar
knopp committed
1127 1128 1129 1130
    proc_rxtx[0].instance_cnt_rxtx = 0; // FIXME data race!
    proc_rxtx[1].instance_cnt_rxtx = 0; // FIXME data race!
    pthread_cond_signal( &proc_rxtx[0].cond_rxtx );    
    pthread_cond_signal( &proc_rxtx[0].cond_rxtx );
1131 1132
    pthread_cond_broadcast(&sync_phy_proc.cond_phy_proc_tx);
    
1133
#ifdef DEBUG_THREADS
knopp's avatar
knopp committed
1134
    printf( "Joining eNB TX CC_id %d thread\n", CC_id);
1135
#endif
knopp's avatar
knopp committed
1136 1137 1138
    int result,i;
    for (i=0;i<1;i++) {
      pthread_join( proc_rxtx[i].pthread_rxtx, (void**)&status );
1139
    
1140
#ifdef DEBUG_THREADS
1141
    
knopp's avatar
knopp committed
1142 1143
      if (result != 0) {
	printf( "Error joining thread.\n" );
1144
      } else {
knopp's avatar
knopp committed
1145 1146 1147 1148 1149
	if (status) {
	  printf( "status %d\n", *status );
	} else {
	  printf( "The thread was killed. No status available.\n" );
	}
1150
      }
1151
    
1152
#else
knopp's avatar
knopp committed
1153
      UNUSED(result);
1154
#endif
knopp's avatar
knopp committed
1155 1156 1157 1158 1159 1160

      pthread_mutex_destroy( &proc_rxtx[i].mutex_rxtx );
      pthread_cond_destroy( &proc_rxtx[i].cond_rxtx );


    }
1161
#ifdef DEBUG_THREADS
knopp's avatar
knopp committed
1162
    printf( "Killing RX CC_id %d thread\n", CC_id);
1163
#endif
1164
    
1165
#ifdef DEBUG_THREADS