lte-softmodem.c 28.2 KB
Newer Older
1 2 3 4 5
/*
 * 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
6
 * the OAI Public License, Version 1.1  (the "License"); you may not use this file
7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
 * 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
 */

22
/*! \file lte-softmodem.c
23
 * \brief Top-level threads for eNodeB
24
 * \author R. Knopp, F. Kaltenberger, Navid Nikaein
knopp's avatar
knopp committed
25 26 27
 * \date 2012
 * \version 0.1
 * \company Eurecom
28
 * \email: knopp@eurecom.fr,florian.kaltenberger@eurecom.fr, navid.nikaein@eurecom.fr
knopp's avatar
knopp committed
29 30 31
 * \note
 * \warning
 */
32

33

Cedric Roux's avatar
Cedric Roux committed
34 35 36
#define _GNU_SOURCE             /* See feature_test_macros(7) */
#include <sched.h>

37
#include "rt_wrapper.h"
38

39

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

42
#include "assertions.h"
43
#include "msc.h"
44 45

#include "PHY/types.h"
46

47
#include "PHY/defs_eNB.h"
48
#include "common/ran_context.h"
49
#include "common/config/config_userapi.h"
oai's avatar
oai committed
50
#include "common/utils/load_module_shlib.h"
51
#undef MALLOC //there are two conflicting definitions, so we better make sure we don't use it at all
knopp's avatar
knopp committed
52
//#undef FRAME_LENGTH_COMPLEX_SAMPLES //there are two conflicting definitions, so we better make sure we don't use it at all
53

knopp's avatar
knopp committed
54
#include "../../ARCH/COMMON/common_lib.h"
55
#include "../../ARCH/ETHERNET/USERSPACE/LIB/if_defs.h"
knopp's avatar
knopp committed
56

knopp's avatar
knopp committed
57
//#undef FRAME_LENGTH_COMPLEX_SAMPLES //there are two conflicting definitions, so we better make sure we don't use it at all
58

59 60 61
#include "PHY/phy_vars.h"
#include "SCHED/sched_common_vars.h"
#include "LAYER2/MAC/mac_vars.h"
62

63 64 65 66
#include "LAYER2/MAC/mac.h"
#include "LAYER2/MAC/mac_proto.h"
#include "RRC/LTE/rrc_vars.h"
#include "PHY_INTERFACE/phy_interface_vars.h"
67 68 69 70 71

#ifdef SMBV
#include "PHY/TOOLS/smbv.h"
unsigned short config_frames[4] = {2,9,11,13};
#endif
72
#include "common/utils/LOG/log.h"
nikaeinn's avatar
nikaeinn committed
73
#include "UTIL/OTG/otg_tx.h"
74
#include "UTIL/OTG/otg_externs.h"
75
#include "UTIL/MATH/oml.h"
76
#include "common/utils/LOG/vcd_signal_dumper.h"
77
#include "UTIL/OPT/opt.h"
78
#include "enb_config.h"
nikaeinn's avatar
nikaeinn committed
79
//#include "PHY/TOOLS/time_meas.h"
80

Florian Kaltenberger's avatar
Florian Kaltenberger committed
81
#ifndef OPENAIR2
82
  #include "UTIL/OTG/otg_vars.h"
Florian Kaltenberger's avatar
Florian Kaltenberger committed
83 84
#endif

85
#if defined(ENABLE_ITTI)
86
  #include "create_tasks.h"
87 88
#endif

knopp's avatar
knopp committed
89 90
#include "PHY/INIT/phy_init.h"

91 92
#include "system.h"

93
#ifdef XFORMS
94 95
  #include "PHY/TOOLS/lte_phy_scope.h"
  #include "stats.h"
96
#endif
97
#include "lte-softmodem.h"
oai's avatar
oai committed
98
#include "NB_IoT_interface.h"
99
#ifdef XFORMS
100 101 102 103 104 105 106 107
  // current status is that every UE has a DL scope for a SINGLE eNB (eNB_id=0)
  // at eNB 0, an UL scope for every UE
  FD_lte_phy_scope_ue  *form_ue[NUMBER_OF_UE_MAX];
  FD_lte_phy_scope_enb *form_enb[MAX_NUM_CCs][NUMBER_OF_UE_MAX];
  FD_stats_form                  *form_stats=NULL,*form_stats_l2=NULL;
  char title[255];
  unsigned char                   scope_enb_num_ue = 2;
  static pthread_t                forms_thread; //xforms
108 109
#endif //XFORMS

frtabu's avatar
frtabu committed
110
#ifndef ENABLE_USE_MME
111
  #define EPC_MODE_ENABLED 0
frtabu's avatar
frtabu committed
112 113
#endif

114 115 116 117 118 119
pthread_cond_t nfapi_sync_cond;
pthread_mutex_t nfapi_sync_mutex;
int nfapi_sync_var=-1; //!< protected by mutex \ref nfapi_sync_mutex

uint8_t nfapi_mode = 0; // Default to monolithic mode

120 121
uint16_t sf_ahead=4;

knopp's avatar
knopp committed
122 123
pthread_cond_t sync_cond;
pthread_mutex_t sync_mutex;
124
int sync_var=-1; //!< protected by mutex \ref sync_mutex.
125
int config_sync_var=-1;
126

127 128
uint16_t runtime_phy_rx[29][6]; // SISO [MCS 0-28][RBs 0-5 : 6, 15, 25, 50, 75, 100]
uint16_t runtime_phy_tx[29][6]; // SISO [MCS 0-28][RBs 0-5 : 6, 15, 25, 50, 75, 100]
knopp's avatar
knopp committed
129

130

knopp's avatar
knopp committed
131
volatile int             oai_exit = 0;
132

133
uint32_t                 downlink_frequency[MAX_NUM_CCs][4];
134
int32_t                  uplink_frequency_offset[MAX_NUM_CCs][4];
gauthier's avatar
gauthier committed
135

136
int UE_scan = 1;
137
int UE_scan_carrier = 0;
138
runmode_t mode = normal_txrx;
knopp's avatar
knopp committed
139
int simL1flag;
140 141
FILE *input_fd=NULL;

142

143 144
#if MAX_NUM_CCs == 1
rx_gain_t                rx_gain_mode[MAX_NUM_CCs][4] = {{max_gain,max_gain,max_gain,max_gain}};
145 146
double tx_gain[MAX_NUM_CCs][4] = {{20,0,0,0}};
double rx_gain[MAX_NUM_CCs][4] = {{110,0,0,0}};
147 148 149 150
#else
rx_gain_t                rx_gain_mode[MAX_NUM_CCs][4] = {{max_gain,max_gain,max_gain,max_gain},{max_gain,max_gain,max_gain,max_gain}};
double tx_gain[MAX_NUM_CCs][4] = {{20,0,0,0},{20,0,0,0}};
double rx_gain[MAX_NUM_CCs][4] = {{110,0,0,0},{20,0,0,0}};
gauthier's avatar
gauthier committed
151
#endif
knopp's avatar
knopp committed
152

153
double rx_gain_off = 0.0;
154

knopp's avatar
knopp committed
155
double sample_rate=30.72e6;
156
double bw = 10.0e6;
157

Florian Kaltenberger's avatar
Florian Kaltenberger committed
158

159 160 161
uint8_t dci_Format = 0;
uint8_t agregation_Level =0xFF;

162 163
uint8_t nb_antenna_tx = 1;
uint8_t nb_antenna_rx = 1;
164

knopp's avatar
knopp committed
165 166 167
char ref[128] = "internal";
char channels[128] = "0";

168
int                      rx_input_level_dBm;
169

170
#ifdef XFORMS
171
  extern int                      otg_enabled;
172
#else
173
  int                             otg_enabled;
174
#endif
knopp's avatar
knopp committed
175
//int                             number_of_cards =   1;
176

177

178 179
uint8_t exit_missed_slots=1;
uint64_t num_missed_slots=0; // counter for the number of missed slots
180

181

182 183
extern void reset_opp_meas(void);
extern void print_opp_meas(void);
184

185 186 187

extern void init_eNB_afterRU(void);

188
int transmission_mode=1;
189
int emulate_rf = 0;
190
int numerology = 0;
191

192
THREAD_STRUCT thread_struct;
193 194 195
/* struct for ethernet specific parameters given in eNB conf file */
eth_params_t *eth_params;

196 197
double cpuf;

knopp's avatar
knopp committed
198 199


200 201 202
/* forward declarations */
void set_default_frame_parms(LTE_DL_FRAME_PARMS *frame_parms[MAX_NUM_CCs]);

203 204 205 206 207
/*---------------------BMC: timespec helpers -----------------------------*/

struct timespec min_diff_time = { .tv_sec = 0, .tv_nsec = 0 };
struct timespec max_diff_time = { .tv_sec = 0, .tv_nsec = 0 };

208
struct timespec clock_difftime(struct timespec start, struct timespec end) {
209
  struct timespec temp;
210

211 212 213 214 215 216 217
  if ((end.tv_nsec-start.tv_nsec)<0) {
    temp.tv_sec = end.tv_sec-start.tv_sec-1;
    temp.tv_nsec = 1000000000+end.tv_nsec-start.tv_nsec;
  } else {
    temp.tv_sec = end.tv_sec-start.tv_sec;
    temp.tv_nsec = end.tv_nsec-start.tv_nsec;
  }
218

219
  return temp;
220 221
}

222
void print_difftimes(void) {
223
#ifdef DEBUG
224
  printf("difftimes min = %lu ns ; max = %lu ns\n", min_diff_time.tv_nsec, max_diff_time.tv_nsec);
225
#else
226
  LOG_I(HW,"difftimes min = %lu ns ; max = %lu ns\n", min_diff_time.tv_nsec, max_diff_time.tv_nsec);
227 228 229
#endif
}

230
void update_difftimes(struct timespec start, struct timespec end) {
231 232 233
  struct timespec diff_time = { .tv_sec = 0, .tv_nsec = 0 };
  int             changed = 0;
  diff_time = clock_difftime(start, end);
234

235 236 237 238
  if ((min_diff_time.tv_nsec == 0) || (diff_time.tv_nsec < min_diff_time.tv_nsec)) {
    min_diff_time.tv_nsec = diff_time.tv_nsec;
    changed = 1;
  }
239

240 241 242 243
  if ((max_diff_time.tv_nsec == 0) || (diff_time.tv_nsec > max_diff_time.tv_nsec)) {
    max_diff_time.tv_nsec = diff_time.tv_nsec;
    changed = 1;
  }
244

245
#if 1
246

247
  if (changed) print_difftimes();
248

249 250 251 252 253
#endif
}

/*------------------------------------------------------------------------*/

254
unsigned int build_rflocal(int txi, int txq, int rxi, int rxq) {
knopp's avatar
knopp committed
255
  return (txi + (txq<<6) + (rxi<<12) + (rxq<<18));
256
}
257
unsigned int build_rfdc(int dcoff_i_rxfe, int dcoff_q_rxfe) {
knopp's avatar
knopp committed
258
  return (dcoff_i_rxfe + (dcoff_q_rxfe<<8));
259 260
}

261

262
void signal_handler(int sig) {
263 264 265 266 267 268 269 270 271 272
  void *array[10];
  size_t size;

  if (sig==SIGSEGV) {
    // get void*'s for all entries on the stack
    size = backtrace(array, 10);
    // print out all the frames to stderr
    fprintf(stderr, "Error: signal %d:\n", sig);
    backtrace_symbols_fd(array, size, 2);
    exit(-1);
273
  } else {
274 275
    printf("Linux signal %s...\n",strsignal(sig));
    exit_function(__FILE__, __FUNCTION__, __LINE__,"softmodem starting exit procedure\n");
276 277
  }
}
278

Raymond Knopp's avatar
Raymond Knopp committed
279

280
void exit_function(const char *file, const char *function, const int line, const char *s) {
281
  int ru_id;
Raymond Knopp's avatar
Raymond Knopp committed
282

283
  if (s != NULL) {
284
    printf("%s:%d %s() Exiting OAI softmodem: %s\n",file,line, function, s);
285 286 287
  }

  oai_exit = 1;
288

289 290
  if (RC.ru == NULL)
    exit(-1); // likely init not completed, prevent crash or hang, exit now...
291

292 293 294 295
  for (ru_id=0; ru_id<RC.nb_RU; ru_id++) {
    if (RC.ru[ru_id] && RC.ru[ru_id]->rfdevice.trx_end_func) {
      RC.ru[ru_id]->rfdevice.trx_end_func(&RC.ru[ru_id]->rfdevice);
      RC.ru[ru_id]->rfdevice.trx_end_func = NULL;
296 297
    }

298 299 300 301 302
    if (RC.ru[ru_id] && RC.ru[ru_id]->ifdevice.trx_end_func) {
      RC.ru[ru_id]->ifdevice.trx_end_func(&RC.ru[ru_id]->ifdevice);
      RC.ru[ru_id]->ifdevice.trx_end_func = NULL;
    }
  }
303

304
  sleep(1); //allow lte-softmodem threads to exit first
305
#if defined(ENABLE_ITTI)
306
  itti_terminate_tasks (TASK_UNKNOWN);
307
#endif
308
  exit(1);
309
}
310

311
#ifdef XFORMS
312

313

314
void reset_stats(FL_OBJECT *button, long arg) {
315
  int i,j,k;
316
  PHY_VARS_eNB *phy_vars_eNB = RC.eNB[0][0];
317 318 319

  for (i=0; i<NUMBER_OF_UE_MAX; i++) {
    for (k=0; k<8; k++) { //harq_processes
320
      for (j=0; j<phy_vars_eNB->dlsch[i][0]->Mlimit; j++) {
321 322 323
        phy_vars_eNB->UE_stats[i].dlsch_NAK[k][j]=0;
        phy_vars_eNB->UE_stats[i].dlsch_ACK[k][j]=0;
        phy_vars_eNB->UE_stats[i].dlsch_trials[k][j]=0;
324
      }
325

326 327 328
      phy_vars_eNB->UE_stats[i].dlsch_l2_errors[k]=0;
      phy_vars_eNB->UE_stats[i].ulsch_errors[k]=0;
      phy_vars_eNB->UE_stats[i].ulsch_consecutive_errors=0;
329 330 331
      phy_vars_eNB->UE_stats[i].dlsch_sliding_cnt=0;
      phy_vars_eNB->UE_stats[i].dlsch_NAK_round0=0;
      phy_vars_eNB->UE_stats[i].dlsch_mcs_offset=0;
332 333 334 335
    }
  }
}

336
static void *scope_thread(void *arg) {
337
# ifdef ENABLE_XFORMS_WRITE_STATS
338
  FILE *eNB_stats;
339
# endif
knopp's avatar
knopp committed
340
  struct sched_param sched_param;
341
  int UE_id, CC_id;
342
  int ue_cnt=0;
343
  sched_param.sched_priority = sched_get_priority_min(SCHED_FIFO)+1;
knopp's avatar
knopp committed
344 345
  sched_setscheduler(0, SCHED_FIFO,&sched_param);
  printf("Scope thread has priority %d\n",sched_param.sched_priority);
346
# ifdef ENABLE_XFORMS_WRITE_STATS
347
  eNB_stats = fopen("eNB_stats.txt", "w");
348
#endif
349

knopp's avatar
knopp committed
350
  while (!oai_exit) {
351 352 353 354 355 356 357 358 359 360 361 362
    ue_cnt=0;

    for(UE_id=0; UE_id<NUMBER_OF_UE_MAX; UE_id++) {
      for(CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
        if ((ue_cnt<scope_enb_num_ue)) {
          phy_scope_eNB(form_enb[CC_id][ue_cnt],
                        RC.eNB[0][CC_id],
                        UE_id);
          ue_cnt++;
        }
      }
    }
363

Florian Kaltenberger's avatar
Florian Kaltenberger committed
364
    sleep(1);
knopp's avatar
knopp committed
365
  }
366

367
  //  printf("%s",stats_buffer);
368
# ifdef ENABLE_XFORMS_WRITE_STATS
369

370 371 372 373 374
  if (eNB_stats) {
    rewind (eNB_stats);
    fwrite (stats_buffer, 1, len, eNB_stats);
    fclose (eNB_stats);
  }
375

376
# endif
377
  pthread_exit((void *)arg);
378 379 380
}
#endif

381

382
static void get_options(void) {
383
  CONFIG_SETRTFLAG(CONFIG_NOEXITONHELP);
384
  get_common_options();
385
  CONFIG_CLEARRTFLAG(CONFIG_NOEXITONHELP);
oai's avatar
oai committed
386

387
  if ( !(CONFIG_ISFLAGSET(CONFIG_ABORT)) ) {
388 389 390 391 392 393 394 395 396 397 398 399 400 401 402
    memset((void *)&RC,0,sizeof(RC));
    /* Read RC configuration file */
    RCConfig();
    NB_eNB_INST = RC.nb_inst;
    printf("Configuration: nb_rrc_inst %d, nb_L1_inst %d, nb_ru %d\n",NB_eNB_INST,RC.nb_L1_inst,RC.nb_RU);

    if (!SOFTMODEM_NONBIOT) {
      load_NB_IoT();
      printf("               nb_nbiot_rrc_inst %d, nb_nbiot_L1_inst %d, nb_nbiot_macrlc_inst %d\n",
             RC.nb_nb_iot_rrc_inst, RC.nb_nb_iot_L1_inst, RC.nb_nb_iot_macrlc_inst);
    } else {
      printf("All Nb-IoT instances disabled\n");
      RC.nb_nb_iot_rrc_inst=RC.nb_nb_iot_L1_inst=RC.nb_nb_iot_macrlc_inst=0;
    }
  }
403 404 405
}


406

407

408

knopp's avatar
knopp committed
409
void set_default_frame_parms(LTE_DL_FRAME_PARMS *frame_parms[MAX_NUM_CCs]) {
knopp's avatar
knopp committed
410 411 412
  int CC_id;

  for (CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
413
    frame_parms[CC_id] = (LTE_DL_FRAME_PARMS *) malloc(sizeof(LTE_DL_FRAME_PARMS));
knopp's avatar
knopp committed
414 415 416 417 418 419 420 421 422 423
    /* Set some default values that may be overwritten while reading options */
    frame_parms[CC_id]->frame_type          = FDD;
    frame_parms[CC_id]->tdd_config          = 3;
    frame_parms[CC_id]->tdd_config_S        = 0;
    frame_parms[CC_id]->N_RB_DL             = 100;
    frame_parms[CC_id]->N_RB_UL             = 100;
    frame_parms[CC_id]->Ncp                 = NORMAL;
    frame_parms[CC_id]->Ncp_UL              = NORMAL;
    frame_parms[CC_id]->Nid_cell            = 0;
    frame_parms[CC_id]->num_MBSFN_config    = 0;
424
    frame_parms[CC_id]->nb_antenna_ports_eNB  = 1;
knopp's avatar
knopp committed
425 426 427 428 429 430 431 432 433 434 435 436 437 438 439
    frame_parms[CC_id]->nb_antennas_tx      = 1;
    frame_parms[CC_id]->nb_antennas_rx      = 1;
    frame_parms[CC_id]->nushift             = 0;
    frame_parms[CC_id]->phich_config_common.phich_resource = oneSixth;
    frame_parms[CC_id]->phich_config_common.phich_duration = normal;
    // UL RS Config
    frame_parms[CC_id]->pusch_config_common.ul_ReferenceSignalsPUSCH.cyclicShift = 0;//n_DMRS1 set to 0
    frame_parms[CC_id]->pusch_config_common.ul_ReferenceSignalsPUSCH.groupHoppingEnabled = 0;
    frame_parms[CC_id]->pusch_config_common.ul_ReferenceSignalsPUSCH.sequenceHoppingEnabled = 0;
    frame_parms[CC_id]->pusch_config_common.ul_ReferenceSignalsPUSCH.groupAssignmentPUSCH = 0;
    frame_parms[CC_id]->prach_config_common.rootSequenceIndex=22;
    frame_parms[CC_id]->prach_config_common.prach_ConfigInfo.zeroCorrelationZoneConfig=1;
    frame_parms[CC_id]->prach_config_common.prach_ConfigInfo.prach_ConfigIndex=0;
    frame_parms[CC_id]->prach_config_common.prach_ConfigInfo.highSpeedFlag=0;
    frame_parms[CC_id]->prach_config_common.prach_ConfigInfo.prach_FreqOffset=0;
440 441 442 443
    //    downlink_frequency[CC_id][0] = 2680000000; // Use float to avoid issue with frequency over 2^31.
    //    downlink_frequency[CC_id][1] = downlink_frequency[CC_id][0];
    //    downlink_frequency[CC_id][2] = downlink_frequency[CC_id][0];
    //    downlink_frequency[CC_id][3] = downlink_frequency[CC_id][0];
knopp's avatar
knopp committed
444
    //printf("Downlink for CC_id %d frequency set to %u\n", CC_id, downlink_frequency[CC_id][0]);
445
    frame_parms[CC_id]->dl_CarrierFreq=downlink_frequency[CC_id][0];
knopp's avatar
knopp committed
446 447 448
  }
}

449
void wait_RUs(void) {
450
  LOG_I(PHY,"Waiting for RUs to be configured ... RC.ru_mask:%02lx\n", RC.ru_mask);
451 452
  // wait for all RUs to be configured over fronthaul
  pthread_mutex_lock(&RC.ru_mutex);
453

454 455
  while (RC.ru_mask>0) {
    pthread_cond_wait(&RC.ru_cond,&RC.ru_mutex);
456
    printf("RC.ru_mask:%02lx\n", RC.ru_mask);
457 458
  }

459
  pthread_mutex_unlock(&RC.ru_mutex);
460 461 462
  LOG_I(PHY,"RUs configured\n");
}

463
void wait_eNBs(void) {
464 465 466 467
  int i,j;
  int waiting=1;

  while (waiting==1) {
Cedric Roux's avatar
Cedric Roux committed
468 469
    printf("Waiting for eNB L1 instances to all get configured ... sleeping 50ms (nb_L1_inst %d)\n",RC.nb_L1_inst);
    usleep(50*1000);
470
    waiting=0;
471

472
    for (i=0; i<RC.nb_L1_inst; i++) {
473 474
      printf("RC.nb_L1_CC[%d]:%d\n", i, RC.nb_L1_CC[i]);

475 476 477 478 479
      for (j=0; j<RC.nb_L1_CC[i]; j++) {
        if (RC.eNB[i][j]->configured==0) {
          waiting=1;
          break;
        }
480 481
      }
    }
482
  }
483

484 485
  printf("eNB L1 are configured\n");
}
486

487
#if defined(ENABLE_ITTI)
488 489
/*
 * helper function to terminate a certain ITTI task
490
 */
491
void terminate_task(module_id_t mod_id, task_id_t from, task_id_t to) {
492
  LOG_I(ENB_APP, "sending TERMINATE_MESSAGE from task %s (%d) to task %s (%d)\n",
493
        itti_get_task_name(from), from, itti_get_task_name(to), to);
494
  MessageDef *msg;
495 496
  msg = itti_alloc_new_message (from, TERMINATE_MESSAGE);
  itti_send_msg_to_task (to, ENB_MODULE_ID_TO_INSTANCE(mod_id), msg);
497 498
}

499
extern void  free_transport(PHY_VARS_eNB *);
500
extern void  phy_free_RU(RU_t *);
501

502
int stop_L1L2(module_id_t enb_id) {
503 504
  LOG_W(ENB_APP, "stopping lte-softmodem\n");

505
  if (!RC.ru) {
506
    LOG_UI(ENB_APP, "no RU configured\n");
507 508 509
    return -1;
  }

510
  /* these tasks need to pick up new configuration */
511
  terminate_task(enb_id, TASK_ENB_APP, TASK_RRC_ENB);
512
  oai_exit = 1;
513
  LOG_I(ENB_APP, "calling kill_RU_proc() for instance %d\n", enb_id);
514
  kill_RU_proc(RC.ru[enb_id]);
515
  LOG_I(ENB_APP, "calling kill_eNB_proc() for instance %d\n", enb_id);
516
  kill_eNB_proc(enb_id);
517
  oai_exit = 0;
518

519 520 521 522
  for (int cc_id = 0; cc_id < RC.nb_CC[enb_id]; cc_id++) {
    free_transport(RC.eNB[enb_id][cc_id]);
    phy_free_lte_eNB(RC.eNB[enb_id][cc_id]);
  }
523

524
  phy_free_RU(RC.ru[enb_id]);
525
  free_lte_top();
526 527 528 529
  return 0;
}

/*
530
 * Restart the lte-softmodem after it has been soft-stopped with stop_L1L2()
531
 */
532
int restart_L1L2(module_id_t enb_id) {
533 534 535
  RU_t *ru = RC.ru[enb_id];
  int cc_id;
  MessageDef *msg_p = NULL;
536 537
  LOG_W(ENB_APP, "restarting lte-softmodem\n");
  /* block threads */
538
  pthread_mutex_lock(&sync_mutex);
539
  sync_var = -1;
540
  pthread_mutex_unlock(&sync_mutex);
541

542 543
  for (cc_id = 0; cc_id < RC.nb_L1_CC[enb_id]; cc_id++) {
    RC.eNB[enb_id][cc_id]->configured = 0;
544 545
  }

546 547 548 549 550
  RC.ru_mask |= (1 << ru->idx);
  /* copy the changed frame parameters to the RU */
  /* TODO this should be done for all RUs associated to this eNB */
  memcpy(&ru->frame_parms, &RC.eNB[enb_id][0]->frame_parms, sizeof(LTE_DL_FRAME_PARMS));
  set_function_spec_param(RC.ru[enb_id]);
551 552 553
  /* reset the list of connected UEs in the MAC, since in this process with
   * loose all UEs (have to reconnect) */
  init_UE_list(&RC.mac[enb_id]->UE_list);
554
  LOG_I(ENB_APP, "attempting to create ITTI tasks\n");
555

556 557 558 559 560 561
  if (itti_create_task (TASK_RRC_ENB, rrc_enb_task, NULL) < 0) {
    LOG_E(RRC, "Create task for RRC eNB failed\n");
    return -1;
  } else {
    LOG_I(RRC, "Re-created task for RRC eNB successfully\n");
  }
562

563 564 565 566 567
  /* pass a reconfiguration request which will configure everything down to
   * RC.eNB[i][j]->frame_parms, too */
  msg_p = itti_alloc_new_message(TASK_ENB_APP, RRC_CONFIGURATION_REQ);
  RRC_CONFIGURATION_REQ(msg_p) = RC.rrc[enb_id]->configuration;
  itti_send_msg_to_task(TASK_RRC_ENB, ENB_MODULE_ID_TO_INSTANCE(enb_id), msg_p);
568 569
  /* TODO XForms might need to be restarted, but it is currently (09/02/18)
   * broken, so we cannot test it */
570 571 572 573 574 575
  wait_eNBs();
  init_RU_proc(ru);
  ru->rf_map.card = 0;
  ru->rf_map.chain = 0; /* CC_id + chain_offset;*/
  wait_RUs();
  init_eNB_afterRU();
576 577 578 579 580 581
  printf("Sending sync to all threads\n");
  pthread_mutex_lock(&sync_mutex);
  sync_var=0;
  pthread_cond_broadcast(&sync_cond);
  pthread_mutex_unlock(&sync_mutex);
  return 0;
582
}
583
#endif
584

585
static  void wait_nfapi_init(char *thread_name) {
586 587
  printf( "waiting for NFAPI PNF connection and population of global structure (%s)\n",thread_name);
  pthread_mutex_lock( &nfapi_sync_mutex );
588

589 590
  while (nfapi_sync_var<0)
    pthread_cond_wait( &nfapi_sync_cond, &nfapi_sync_mutex );
591

592 593 594 595
  pthread_mutex_unlock(&nfapi_sync_mutex);
  printf( "NFAPI: got sync (%s)\n", thread_name);
}

596
int main( int argc, char **argv ) {
597
  int i;
598
#if defined (XFORMS)
knopp's avatar
knopp committed
599
  void *status;
600
#endif
knopp's avatar
knopp committed
601
  int CC_id;
602
  int ru_id;
knopp's avatar
knopp committed
603
#if defined (XFORMS)
604 605
  int ret;
#endif
606

607
  if ( load_configmodule(argc,argv,0) == NULL) {
608
    exit_fun("[SOFTMODEM] Error, configuration module init failed\n");
609
  }
610

611
  mode = normal_txrx;
knopp's avatar
knopp committed
612
  set_latency_target();
613
  logInit();
614
  printf("Reading in command-line options\n");
615 616
  CONFIG_SETRTFLAG(CONFIG_NOCHECKUNKOPT);
  get_options ();
617

618
  if (CONFIG_ISFLAGSET(CONFIG_ABORT) ) {
619 620
    fprintf(stderr,"Getting configuration failed\n");
    exit(-1);
621
  }
knopp's avatar
knopp committed
622

623
#if T_TRACER
624
  T_Config_Init();
625
#endif
626
  ret=config_check_unknown_cmdlineopt(NULL);
627

628
  if (ret != 0) {
629 630
    LOG_E(ENB_APP, "%i unknown options in command line\n",ret);
    exit_fun("");
631
  }
632

633
  CONFIG_CLEARRTFLAG(CONFIG_NOCHECKUNKOPT);
634 635
  //randominit (0);
  set_taus_seed (0);
636
  printf("configuring for RAU/RRU\n");
637

638
  if (opp_enabled ==1) {
639
    reset_opp_meas();
640
  }
641

642
  cpuf=get_cpu_freq_GHz();
643
#if defined(ENABLE_ITTI)
644
  printf("ITTI init, useMME: %i\n" ,EPC_MODE_ENABLED);
Cedric Roux's avatar
Cedric Roux committed
645
  itti_init(TASK_MAX, THREAD_MAX, MESSAGES_ID_MAX, tasks_info, messages_info);
646

647
  // initialize mscgen log after ITTI
648
  if (get_softmodem_params()->start_msc) {
649
    load_module_shlib("msc",NULL,0,&msc_interface);
650
  }
651

652
  MSC_INIT(MSC_E_UTRAN, THREAD_MAX+TASK_MAX);
653
#endif
654

655
  if (opt_type != OPT_NONE) {
Thomas Laurent's avatar
Thomas Laurent committed
656
    if (init_opt(in_path, in_ip) == -1)
657 658
      LOG_E(OPT,"failed to run OPT \n");
  }
659

660
#ifdef PDCP_USE_NETLINK
661
  printf("PDCP netlink\n");
knopp's avatar
knopp committed
662
  netlink_init();
663 664 665
#if defined(PDCP_USE_NETLINK_QUEUES)
  pdcp_netlink_init();
#endif
666
#endif
knopp's avatar
knopp committed
667 668 669
  // to make a graceful exit when ctrl-c is pressed
  signal(SIGSEGV, signal_handler);
  signal(SIGINT, signal_handler);
670 671
  signal(SIGTERM, signal_handler);
  signal(SIGABRT, signal_handler);
knopp's avatar
knopp committed
672
  check_clock();
673 674 675 676
#ifndef PACKAGE_VERSION
#  define PACKAGE_VERSION "UNKNOWN-EXPERIMENTAL"
#endif
  LOG_I(HW, "Version: %s\n", PACKAGE_VERSION);
677
  printf("Runtime table\n");
678
  fill_modeled_runtime_table(runtime_phy_rx,runtime_phy_tx);
laurent's avatar
laurent committed
679
#ifndef DEADLINE_SCHEDULER
680
  printf("NO deadline scheduler\n");
681
  /* Currently we set affinity for UHD to CPU 0 for eNB/UE and only if number of CPUS >2 */
682 683 684 685
  cpu_set_t cpuset;
  int s;
  char cpu_affinity[1024];
  CPU_ZERO(&cpuset);
686
#ifdef CPU_AFFINITY
687

688 689 690
  if (get_nprocs() > 2) {
    CPU_SET(0, &cpuset);
    s = pthread_setaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset);
691

692
    if (s != 0) {
693 694
      perror( "pthread_setaffinity_np");
      exit_fun("Error setting processor affinity");
695
    }
696

697 698
    LOG_I(HW, "Setting the affinity of main function to CPU 0, for device library to use CPU 0 only!\n");
  }
699

700
#endif
701 702
  /* Check the actual affinity mask assigned to the thread */
  s = pthread_getaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset);
703

704 705 706 707
  if (s != 0) {
    perror( "pthread_getaffinity_np");
    exit_fun("Error getting processor affinity ");
  }
708

709
  memset(cpu_affinity, 0 , sizeof(cpu_affinity));
710

711 712 713 714 715
  for (int j = 0; j < CPU_SETSIZE; j++) {
    if (CPU_ISSET(j, &cpuset)) {
      char temp[1024];
      sprintf(temp, " CPU_%d ", j);
      strcat(cpu_affinity, temp);
716
    }
717
  }
718

719
  LOG_I(HW, "CPU Affinity of main() function is... %s\n", cpu_affinity);
720
#endif
721
#if defined(ENABLE_ITTI)
722

723
  if (RC.nb_inst > 0)  {
knopp's avatar
knopp committed
724
    // don't create if node doesn't connect to RRC/S1/GTP
725 726 727 728 729
    if (create_tasks(1) < 0) {
      printf("cannot create ITTI tasks\n");
      exit(-1); // need a softer mode
    }

730
    printf("ITTI tasks created\n");
731
  } else {
732 733
    printf("No ITTI, Initializing L1\n");
    RCconfig_L1();
734
  }
735

736
#endif
737
  /* Start the agent. If it is turned off in the configuration, it won't start */
738
  RCconfig_flexran();
739

740 741
  for (i = 0; i < RC.nb_L1_inst; i++) {
    flexran_agent_start(i);
742
  }
743 744 745

  // init UE_PF_PO and mutex lock
  pthread_mutex_init(&ue_pf_po_mutex, NULL);
knopp's avatar
knopp committed
746
  memset (&UE_PF_PO[0][0], 0, sizeof(UE_PF_PO_t)*MAX_MOBILES_PER_ENB*MAX_NUM_CCs);
knopp's avatar
knopp committed
747
  mlockall(MCL_CURRENT | MCL_FUTURE);
knopp's avatar
knopp committed
748 749
  pthread_cond_init(&sync_cond,NULL);
  pthread_mutex_init(&sync_mutex, NULL);
750
#ifdef XFORMS
751
  int UE_id;
752 753
  printf("XFORMS\n");

754
  if (get_softmodem_params()->do_forms==1) {
knopp's avatar
knopp committed
755
    fl_initialize (&argc, argv, NULL, 0, 0);
756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776
    form_stats_l2 = create_form_stats_form();
    fl_show_form (form_stats_l2->stats_form, FL_PLACE_HOTSPOT, FL_FULLBORDER, "l2 stats");
    form_stats = create_form_stats_form();
    fl_show_form (form_stats->stats_form, FL_PLACE_HOTSPOT, FL_FULLBORDER, "stats");

    for(UE_id=0; UE_id<scope_enb_num_ue; UE_id++) {
      for(CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
        form_enb[CC_id][UE_id] = create_lte_phy_scope_enb();
        sprintf (title, "LTE UL SCOPE eNB for CC_id %d, UE %d",CC_id,UE_id);
        fl_show_form (form_enb[CC_id][UE_id]->lte_phy_scope_enb, FL_PLACE_HOTSPOT, FL_FULLBORDER, title);

        if (otg_enabled) {
          fl_set_button(form_enb[CC_id][UE_id]->button_0,1);
          fl_set_object_label(form_enb[CC_id][UE_id]->button_0,"DL Traffic ON");
        } else {
          fl_set_button(form_enb[CC_id][UE_id]->button_0,0);
          fl_set_object_label(form_enb[CC_id][UE_id]->button_0,"DL Traffic OFF");
        }
      } // CC_id
    } // UE_id

knopp's avatar
knopp committed
777
    ret = pthread_create(&forms_thread, NULL, scope_thread, NULL);
778

779 780
    if (ret == 0)
      pthread_setname_np( forms_thread, "xforms" );
781

knopp's avatar
knopp committed
782 783
    printf("Scope thread created, ret=%d\n",ret);
  }
784

785
#endif
786
  rt_sleep_ns(10*100000000ULL);
Cedric Roux's avatar
Cedric Roux committed
787

788
  if (nfapi_mode) {
789 790 791 792 793
    printf("NFAPI*** - mutex and cond created - will block shortly for completion of PNF connection\n");
    pthread_cond_init(&sync_cond,NULL);
    pthread_mutex_init(&sync_mutex, NULL);
  }

794 795 796 797 798 799
  const char *nfapi_mode_str = "<UNKNOWN>";

  switch(nfapi_mode) {
    case 0:
      nfapi_mode_str = "MONOLITHIC";
      break;
800

801 802 803
    case 1:
      nfapi_mode_str = "PNF";
      break;
804

805 806 807
    case 2:
      nfapi_mode_str = "VNF";
      break;
808

809 810 811 812
    default:
      nfapi_mode_str = "<UNKNOWN NFAPI MODE>";
      break;
  }
813

814 815 816 817 818 819
  printf("NFAPI MODE:%s\n", nfapi_mode_str);

  if (nfapi_mode==2) // VNF
    wait_nfapi_init("main?");

  printf("START MAIN THREADS\n");
820
  // start the main threads
821 822 823 824 825 826 827 828 829
  number_of_cards = 1;
  printf("RC.nb_L1_inst:%d\n", RC.nb_L1_inst);

  if (RC.nb_L1_inst > 0) {
    printf("Initializing eNB threads single_thread_flag:%d wait_for_sync:%d\n", get_softmodem_params()->single_thread_flag,get_softmodem_params()->wait_for_sync);
    init_eNB(get_softmodem_params()->single_thread_flag,get_softmodem_params()->wait_for_sync);
    //      for (inst=0;inst<RC.nb_L1_inst;inst++)
    //  for (CC_id=0;CC_id<RC.nb_L1_CC[inst];CC_id++) phy_init_lte_eNB(RC.eNB[inst][CC_id],0,0);
  }
830

831 832 833
  printf("wait_eNBs()\n");
  wait_eNBs();
  printf("About to Init RU threads RC.nb_RU:%d\n", RC.nb_RU);
834

835 836 837
  if (RC.nb_RU >0) {
    printf("Initializing RU threads\n");
    init_RU(get_softmodem_params()->rf_config_file);
838

839 840 841
    for (ru_id=0; ru_id<RC.nb_RU; ru_id++) {
      RC.ru[ru_id]->rf_map.card=0;
      RC.ru[ru_id]->rf_map.chain=CC_id+(get_softmodem_params()->chain_offset);
842
    }
843
  }
844

845
  config_sync_var=0;
846

847 848 849
  if (nfapi_mode==1) { // PNF
    wait_nfapi_init("main?");
  }
850

851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867
  printf("wait RUs\n");
  fflush(stdout);
  fflush(stderr);
  wait_RUs();
  printf("ALL RUs READY!\n");
  printf("RC.nb_RU:%d\n", RC.nb_RU);
  // once all RUs are ready intiailize the rest of the eNBs ((dependence on final RU parameters after configuration)
  printf("ALL RUs ready - init eNBs\n");

  if (nfapi_mode != 1 && nfapi_mode != 2) {
    printf("Not NFAPI mode - call init_eNB_afterRU()\n");
    init_eNB_afterRU();
  } else {
    printf("NFAPI mode - DO NOT call init_eNB_afterRU()\n");
  }

  printf("ALL RUs ready - ALL eNBs ready\n");
868
  // connect the TX/RX buffers
869
  sleep(1); /* wait for thread activation */
870
  printf("Sending sync to all threads\n");
knopp's avatar
knopp committed
871
  pthread_mutex_lock(&sync_mutex);
872
  sync_var=0;
knopp's avatar
knopp committed
873 874
  pthread_cond_broadcast(&sync_cond);
  pthread_mutex_unlock(&sync_mutex);
875
  ret=config_check_unknown_cmdlineopt(CONFIG_CHECKALLSECTIONS);
876

877
  if (ret != 0) {
878 879
    LOG_E(ENB_APP, "%i unknown options in command line (invalid section name)\n",ret);
    exit_fun("");
880 881
  }

882
  // wait for end of program
knopp's avatar
knopp committed
883 884
  printf("TYPE <CTRL-C> TO TERMINATE\n");
  //getchar();
885
#if defined(ENABLE_ITTI)
knopp's avatar
knopp committed
886 887
  printf("Entering ITTI signals handler\n");
  itti_wait_tasks_end();
888
  printf("Returned from ITTI signal handler\n");
889
  oai_exit=1;
890
  printf("oai_exit=%d\n",oai_exit);
891
#else
892

knopp's avatar
knopp committed
893
  while (oai_exit==0)
894
    rt_sleep_ns(100000000ULL);
895

896
  printf("Terminating application - oai_exit=%d\n",oai_exit);
897
#endif
knopp's avatar
knopp committed
898
  // stop threads
899
#ifdef XFORMS
knopp's avatar
knopp committed
900
  printf("waiting for XFORMS thread\n");
901

902
  if (get_softmodem_params()->do_forms==1) {
903 904 905
    pthread_join(forms_thread,&status);
    fl_hide_form(form_stats->stats_form);
    fl_free_form(form_stats->stats_form);
906 907
    fl_hide_form(form_stats_l2->stats_form);
    fl_free_form(form_stats_l2->stats_form);
908

909 910 911 912
    for(UE_id=0; UE_id<scope_enb_num_ue; UE_id++) {
      for(CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
        fl_hide_form(form_enb[CC_id][UE_id]->lte_phy_scope_enb);
        fl_free_form(form_enb[CC_id][UE_id]->lte_phy_scope_enb);
knopp's avatar
knopp committed
913
      }
914
    }
915 916
  }

917
#endif
knopp's avatar
knopp committed
918
  printf("stopping MODEM threads\n");
919 920
  stop_eNB(NB_eNB_INST);
  stop_RU(RC.nb_RU);
921

922 923 924 925 926 927
  /* release memory used by the RU/eNB threads (incomplete), after all
   * threads have been stopped (they partially use the same memory) */
  for (int inst = 0; inst < NB_eNB_INST; inst++) {
    for (int cc_id = 0; cc_id < RC.nb_CC[inst]; cc_id++) {
      free_transport(RC.eNB[inst][cc_id]);
      phy_free_lte_eNB(RC.eNB[inst][cc_id]);
928
    }
929
  }
930

931 932 933
  for (int inst = 0; inst < RC.nb_RU; inst++) {
    phy_free_RU(RC.ru[inst]);
  }
934

935
  free_lte_top();
936 937 938
  printf("About to call end_configmodule() from %s() %s:%d\n", __FUNCTION__, __FILE__, __LINE__);
  end_configmodule();
  printf("Called end_configmodule() from %s() %s:%d\n", __FUNCTION__, __FILE__, __LINE__);
knopp's avatar
knopp committed
939 940
  pthread_cond_destroy(&sync_cond);
  pthread_mutex_destroy(&sync_mutex);
941 942
  pthread_cond_destroy(&nfapi_sync_cond);
  pthread_mutex_destroy(&nfapi_sync_mutex);
943
  pthread_mutex_destroy(&ue_pf_po_mutex);
944

945 946 947 948 949
  for(ru_id=0; ru_id<RC.nb_RU; ru_id++) {
    if (RC.ru[ru_id]->rfdevice.trx_end_func) {
      RC.ru[ru_id]->rfdevice.trx_end_func(&RC.ru[ru_id]->rfdevice);
      RC.ru[ru_id]->rfdevice.trx_end_func = NULL;
    }
950

951 952 953
    if (RC.ru[ru_id]->ifdevice.trx_end_func) {
      RC.ru[ru_id]->ifdevice.trx_end_func(&RC.ru[ru_id]->ifdevice);
      RC.ru[ru_id]->ifdevice.trx_end_func = NULL;
knopp's avatar
knopp committed
954
    }
955 956
  }

957
  if (opt_enabled == 1)
958
    terminate_opt();
959

960
  logClean();
961
  printf("Bye.\n");
knopp's avatar
knopp committed
962 963
  return 0;
}