lte-softmodem.c 85.1 KB
Newer Older
1
/*******************************************************************************
2 3
    OpenAirInterface 
    Copyright(c) 1999 - 2014 Eurecom
4

5 6 7 8
    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.
9 10


11 12 13 14
    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.
15

16 17 18 19
    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/>.
20

21 22 23 24 25
   Contact Information
   OpenAirInterface Admin: openair_admin@eurecom.fr
   OpenAirInterface Tech : openair_tech@eurecom.fr
   OpenAirInterface Dev  : openair4g-devel@eurecom.fr
  
ghaddab's avatar
ghaddab committed
26
   Address      : Eurecom, Campus SophiaTech, 450 Route des Chappes, CS 50193 - 06904 Biot Sophia Antipolis cedex, FRANCE
27

28
 *******************************************************************************/
29 30

/*! \file lte-softmodem.c
Raymond Knopp's avatar
 
Raymond Knopp committed
31
 * \brief main program to control HW and scheduling
32
 * \author R. Knopp, F. Kaltenberger, Navid Nikaein
Raymond Knopp's avatar
 
Raymond Knopp committed
33 34 35
 * \date 2012
 * \version 0.1
 * \company Eurecom
36
 * \email: knopp@eurecom.fr,florian.kaltenberger@eurecom.fr, navid.nikaein@eurecom.fr
Raymond Knopp's avatar
 
Raymond Knopp committed
37 38 39
 * \note
 * \warning
 */
Raymond Knopp's avatar
 
Raymond Knopp committed
40
#define _GNU_SOURCE
41 42 43 44 45 46 47 48 49 50 51 52 53 54 55
#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 <sys/stat.h>
#include <fcntl.h>
#include <sched.h>
#include <signal.h>
#include <execinfo.h>
#include <getopt.h>

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

Raymond Knopp's avatar
 
Raymond Knopp committed
58
#ifdef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
59
static int hw_subframe;
Raymond Knopp's avatar
 
Raymond Knopp committed
60
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
61

62
#include "assertions.h"
63 64 65 66 67 68

#ifdef EMOS
#include <gps.h>
#endif

#include "PHY/types.h"
69

70
#include "PHY/defs.h"
71 72 73
#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

Raymond Knopp's avatar
 
Raymond Knopp committed
74
#ifndef USRP
75
#include "openair0_lib.h"
Raymond Knopp's avatar
 
Raymond Knopp committed
76 77 78 79
#else
#include "../../ARCH/COMMON/common_lib.h"
#endif

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

#include "PHY/vars.h"
#include "MAC_INTERFACE/vars.h"
84
//#include "SCHED/defs.h"
85 86 87 88 89 90 91 92 93 94
#include "SCHED/vars.h"
#include "LAYER2/MAC/vars.h"

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

#ifdef EMOS
#include "SCHED/phy_procedures_emos.h"
#endif

#ifdef OPENAIR2
95
#include "otg_tx.h"
96 97
#include "LAYER2/MAC/defs.h"
#include "LAYER2/MAC/vars.h"
98
#include "LAYER2/MAC/proto.h"
99 100 101 102 103 104 105 106 107 108 109 110 111 112 113
#ifndef CELLULAR
#include "RRC/LITE/vars.h"
#endif
#include "PHY_INTERFACE/vars.h"
#endif

#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.h"
#include "UTIL/OTG/otg_vars.h"
#include "UTIL/MATH/oml.h"
#include "UTIL/LOG/vcd_signal_dumper.h"
114
#include "enb_config.h"
115

116 117
#if defined(ENABLE_ITTI)
# include "intertask_interface_init.h"
118
# include "create_tasks.h"
119 120 121
# if defined(ENABLE_USE_MME)
#   include "s1ap_eNB.h"
# endif
122 123
#endif

124 125 126
#ifdef XFORMS
#include "PHY/TOOLS/lte_phy_scope.h"
#include "stats.h"
127 128 129 130 131
#endif

#define FRAME_PERIOD    100000000ULL
#define DAQ_PERIOD      66667ULL

Raymond Knopp's avatar
 
Raymond Knopp committed
132 133
#define DEBUG_THREADS 1

134 135 136 137 138 139 140
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;

141 142
extern int16_t* sync_corr_ue0;
extern int16_t prach_ifft[4][1024*2];
143 144 145

int init_dlsch_threads(void);
void cleanup_dlsch_threads(void);
146
int32_t init_rx_pdsch_thread(void);
147
void cleanup_rx_pdsch_thread(void);
Raymond Knopp's avatar
 
Raymond Knopp committed
148 149


Raymond Knopp's avatar
 
Raymond Knopp committed
150 151
int32_t *rxdata;
int32_t *txdata;
152 153 154 155
void setup_ue_buffers(PHY_VARS_UE *phy_vars_ue, LTE_DL_FRAME_PARMS *frame_parms, int carrier);
void setup_eNB_buffers(PHY_VARS_eNB *phy_vars_eNB, LTE_DL_FRAME_PARMS *frame_parms, int carrier);

#ifdef XFORMS
156 157 158 159
// 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[NUMBER_OF_UE_MAX];
160
FD_stats_form                  *form_stats=NULL;
161
char title[255];
162
unsigned char                   scope_enb_num_ue = 1;
163 164 165
#endif //XFORMS

#ifdef RTAI
166
static SEM                     *mutex;
167 168
//static CND *cond;

169 170
static int                      thread0;
static int                      thread1;
Raymond Knopp's avatar
 
Raymond Knopp committed
171 172 173
#ifdef USRP
static SEM *sync_sem; // to sync rx & tx streaming
#endif
174 175
//static int sync_thread;
#else
176 177 178 179
pthread_t                       thread0;
pthread_t                       thread1;
pthread_attr_t                  attr_dlsch_threads;
struct sched_param              sched_param_dlsch;
Raymond Knopp's avatar
 
Raymond Knopp committed
180 181 182 183
#ifdef USRP
pthread_cond_t sync_cond;
pthread_mutex_t sync_mutex;
#endif
184 185
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
186 187 188 189
pthread_attr_t                  attr_eNB_proc_tx[MAX_NUM_CCs][10];
pthread_attr_t                  attr_eNB_proc_rx[MAX_NUM_CCs][10];
struct sched_param              sched_param_eNB_proc_tx[MAX_NUM_CCs][10];
struct sched_param              sched_param_eNB_proc_rx[MAX_NUM_CCs][10];
Raymond Knopp's avatar
 
Raymond Knopp committed
190

191 192 193 194 195 196
#ifdef XFORMS
static pthread_t                thread2; //xforms
#endif
#ifdef EMOS
static pthread_t                thread3; //emos
#endif
197

Raymond Knopp's avatar
 
Raymond Knopp committed
198 199 200
openair0_device openair0;
openair0_timestamp timestamp;

201
/*
Raymond Knopp's avatar
 
Raymond Knopp committed
202 203 204
  static int instance_cnt=-1; //0 means worker is busy, -1 means its free
  int instance_cnt_ptr_kern,*instance_cnt_ptr_user;
  int pci_interface_ptr_kern;
205 206 207 208
*/
//extern unsigned int bigphys_top;
//extern unsigned int mem_base;

209
int                             card = 0;
Raymond Knopp's avatar
 
Raymond Knopp committed
210

211

212
#if defined(ENABLE_ITTI)
213 214
static volatile int             start_eNB = 0;
static volatile int             start_UE = 0;
215
#endif
216 217 218 219 220 221
volatile int                    oai_exit = 0;

//static int                      time_offset[4] = {-138,-138,-138,-138};
//static int                      time_offset[4] = {-145,-145,-145,-145};
static int                      time_offset[4] = {0,0,0,0};

Raymond Knopp's avatar
 
Raymond Knopp committed
222

223
static char                     UE_flag=0;
224
static uint8_t                       eNB_id=0,UE_id=0;
225

226
uint32_t                             carrier_freq[4] =           {1907600000,1907600000,1907600000,1907600000}; /* For UE! */
227
static uint32_t          downlink_frequency[4] =     {1907600000,1907600000,1907600000,1907600000};
228
static int32_t                      uplink_frequency_offset[4]= {-120000000,-120000000,-120000000,-120000000};
229 230
static char                    *conf_config_file_name = NULL;

231
#if defined(ENABLE_ITTI)
232
static char                    *itti_dump_file = NULL;
Raymond Knopp's avatar
 
Raymond Knopp committed
233 234
#endif

Lionel Gauthier's avatar
Lionel Gauthier committed
235 236
#ifndef USRP
double tx_gain = 20;
Raymond Knopp's avatar
 
Raymond Knopp committed
237
double rx_gain = 10;
Lionel Gauthier's avatar
Lionel Gauthier committed
238
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
239 240
double tx_gain = 120;
double rx_gain = 30;
Lionel Gauthier's avatar
Lionel Gauthier committed
241
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
242

Raymond Knopp's avatar
 
Raymond Knopp committed
243 244
double sample_rate=30.72e6;
double bw = 14e6;
245

Raymond Knopp's avatar
 
Raymond Knopp committed
246
#ifndef USRP
247 248 249 250
static unsigned int             rxg_max[4] =    {133,133,133,133};
static unsigned int             rxg_med[4] =    {127,127,127,127};
static unsigned int             rxg_byp[4] =    {120,120,120,120};
static int                      tx_max_power =  0;
251

Raymond Knopp's avatar
 
Raymond Knopp committed
252

Raymond Knopp's avatar
 
Raymond Knopp committed
253
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
254
static unsigned int             rxg_max[4] =    {133,133,133,133};
Raymond Knopp's avatar
 
Raymond Knopp committed
255 256
//static unsigned int            rxg_med[4] =    {127,127,127,127};
//static unsigned int            rxg_byp[4] =    {120,120,120,120};
Raymond Knopp's avatar
 
Raymond Knopp committed
257
static int                      tx_max_power =  0;
Raymond Knopp's avatar
 
Raymond Knopp committed
258 259 260
char ref[128] = "internal";
char channels[128] = "0";

Raymond Knopp's avatar
 
Raymond Knopp committed
261 262 263
unsigned int samples_per_frame = 307200;
unsigned int samples_per_packets = 2048; // samples got every recv or send
unsigned int tx_forward_nsamps;
Raymond Knopp's avatar
 
Raymond Knopp committed
264 265 266 267 268 269 270 271 272

int sf_bounds_5[10] = {8, 15, 23, 30, 38, 45, 53, 60, 68, 75};
int sf_bounds_10[10] = {8, 15, 23, 30, 38, 45, 53, 60, 68, 75};
int sf_bounds_20[10] = {15, 30, 45, 60, 75, 90, 105, 120, 135, 150};
int *sf_bounds;
int max_cnt;
int tx_delay;

#endif
273
/*
Raymond Knopp's avatar
 
Raymond Knopp committed
274 275 276
  uint32_t rf_mode_max[4]     = {55759,55759,55759,55759};
  uint32_t rf_mode_med[4]     = {39375,39375,39375,39375};
  uint32_t rf_mode_byp[4]     = {22991,22991,22991,22991};
277
*/
Raymond Knopp's avatar
 
Raymond Knopp committed
278 279
//static uint32_t                      rf_mode[4] =        {MY_RF_MODE,0,0,0};
//static uint32_t                      rf_local[4] =       {8255000,8255000,8255000,8255000}; // UE zepto
Raymond Knopp's avatar
 
Raymond Knopp committed
280 281
//{8254617, 8254617, 8254617, 8254617}; //eNB khalifa
//{8255067,8254810,8257340,8257340}; // eNB PETRONAS
282

Raymond Knopp's avatar
 
Raymond Knopp committed
283 284 285 286 287
//static uint32_t                      rf_vcocal[4] =      {910,910,910,910};
//static uint32_t                      rf_vcocal_850[4] =  {2015, 2015, 2015, 2015};
//static uint32_t                      rf_rxdc[4] =        {32896,32896,32896,32896};
//static uint32_t                      rxgain[4] =         {20,20,20,20};
//static uint32_t                      txgain[4] =         {20,20,20,20};
288 289 290

static runmode_t                mode;
static int                      rx_input_level_dBm;
291
static int                      online_log_messages=0;
292
#ifdef XFORMS
293
extern int                      otg_enabled;
294
static char                     do_forms=0;
295
#else
296
int                             otg_enabled;
297
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
298
//int                             number_of_cards =   1;
Raymond Knopp's avatar
 
Raymond Knopp committed
299
#ifndef USRP
300 301
static int                      mbox_bounds[20] =   {8,16,24,30,38,46,54,60,68,76,84,90,98,106,114,120,128,136,144, 0}; ///boundaries of slots in terms ob mbox counter rounded up to even numbers
//static int                      mbox_bounds[20] =   {6,14,22,28,36,44,52,58,66,74,82,88,96,104,112,118,126,134,142, 148}; ///boundaries of slots in terms ob mbox counter rounded up to even numbers
Raymond Knopp's avatar
 
Raymond Knopp committed
302
#endif
303

304
static LTE_DL_FRAME_PARMS      *frame_parms;
305

Raymond Knopp's avatar
 
Raymond Knopp committed
306
int multi_thread=1;
Raymond Knopp's avatar
 
Raymond Knopp committed
307
int N_RB_DL=25;
Raymond Knopp's avatar
 
Raymond Knopp committed
308

309 310 311 312 313 314 315 316 317 318 319 320 321 322 323
int16_t           glog_level=LOG_DEBUG;
int16_t           glog_verbosity=LOG_MED;
int16_t           hw_log_level=LOG_DEBUG;
int16_t           hw_log_verbosity=LOG_MED;
int16_t           phy_log_level=LOG_DEBUG;
int16_t           phy_log_verbosity=LOG_MED;
int16_t           mac_log_level=LOG_DEBUG;
int16_t           mac_log_verbosity=LOG_MED;
int16_t           rlc_log_level=LOG_DEBUG;
int16_t           rlc_log_verbosity=LOG_MED;
int16_t           pdcp_log_level=LOG_DEBUG;
int16_t           pdcp_log_verbosity=LOG_MED;
int16_t           rrc_log_level=LOG_DEBUG;
int16_t           rrc_log_verbosity=LOG_MED;

324
unsigned int build_rflocal(int txi, int txq, int rxi, int rxq)
325
{
Raymond Knopp's avatar
 
Raymond Knopp committed
326
  return (txi + (txq<<6) + (rxi<<12) + (rxq<<18));
327 328 329
}
unsigned int build_rfdc(int dcoff_i_rxfe, int dcoff_q_rxfe)
{
Raymond Knopp's avatar
 
Raymond Knopp committed
330
  return (dcoff_i_rxfe + (dcoff_q_rxfe<<8));
331 332
}

333
#if !defined(ENABLE_ITTI)
334 335 336 337 338 339 340 341 342 343 344 345 346 347 348
void signal_handler(int sig)
{
  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);
  }
  else {
349
    oai_exit = 1;
350 351
  }
}
352
#endif
353 354 355

void exit_fun(const char* s)
{
356
  if (s != NULL) {
Lionel Gauthier's avatar
Lionel Gauthier committed
357
    printf("%s %s() Exiting: %s\n",__FILE__, __FUNCTION__, s);
358 359 360 361 362
  }

  oai_exit = 1;

#if defined(ENABLE_ITTI)
363
  itti_terminate_tasks (TASK_UNKNOWN);
364
#endif
365 366 367 368 369 370

  //rt_sleep_ns(FRAME_PERIOD);

  //exit (-1);
}

Raymond Knopp's avatar
 
Raymond Knopp committed
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
static int latency_target_fd = -1;
static int32_t latency_target_value = 0;
/* Latency trick - taken from cyclictest.c 
* if the file /dev/cpu_dma_latency exists,
* open it and write a zero into it. This will tell
* the power management system not to transition to
* a high cstate (in fact, the system acts like idle=poll)
* When the fd to /dev/cpu_dma_latency is closed, the behavior
* goes back to the system default.
*
* Documentation/power/pm_qos_interface.txt
*/
static void set_latency_target(void)
{
  struct stat s;
  int ret;
  if (stat("/dev/cpu_dma_latency", &s) == 0) {
    latency_target_fd = open("/dev/cpu_dma_latency", O_RDWR);
    if (latency_target_fd == -1)
      return;
    ret = write(latency_target_fd, &latency_target_value, 4);
    if (ret == 0) {
      printf("# error setting cpu_dma_latency to %d!: %s\n", latency_target_value, strerror(errno));
      close(latency_target_fd);
      return;
    }
    printf("# /dev/cpu_dma_latency set to %dus\n", latency_target_value);
  }
}

401
#ifdef XFORMS
402
static void *scope_thread(void *arg) {
Raymond Knopp's avatar
 
Raymond Knopp committed
403
  char stats_buffer[16384];
404
# ifdef ENABLE_XFORMS_WRITE_STATS
Raymond Knopp's avatar
 
Raymond Knopp committed
405 406
  FILE *UE_stats, *eNB_stats;
  int len = 0;
407
# endif
Raymond Knopp's avatar
 
Raymond Knopp committed
408
  struct sched_param sched_param;
409

Raymond Knopp's avatar
 
Raymond Knopp committed
410 411
  sched_param.sched_priority = sched_get_priority_min(SCHED_FIFO)+1; 
  sched_setscheduler(0, SCHED_FIFO,&sched_param);
412

Raymond Knopp's avatar
 
Raymond Knopp committed
413
  printf("Scope thread has priority %d\n",sched_param.sched_priority);
414
    
Raymond Knopp's avatar
 
Raymond Knopp committed
415 416 417 418 419 420
  /*
    if (UE_flag==1) 
    UE_stats  = fopen("UE_stats.txt", "w");
    else 
    eNB_stats = fopen("eNB_stats.txt", "w");
  */
421
    
Raymond Knopp's avatar
 
Raymond Knopp committed
422 423
  while (!oai_exit) {
    if (UE_flag==1) {
424
# ifdef ENABLE_XFORMS_WRITE_STATS
Raymond Knopp's avatar
 
Raymond Knopp committed
425
      len =
426
# endif
427
	dump_ue_stats (PHY_vars_UE_g[0][0], stats_buffer, 0, mode,rx_input_level_dBm);
Raymond Knopp's avatar
 
Raymond Knopp committed
428
      fl_set_object_label(form_stats->stats_text, stats_buffer);
429
# ifdef ENABLE_XFORMS_WRITE_STATS
Raymond Knopp's avatar
 
Raymond Knopp committed
430 431
      rewind (UE_stats);
      fwrite (stats_buffer, 1, len, UE_stats);
432
# endif
Raymond Knopp's avatar
 
Raymond Knopp committed
433
      phy_scope_UE(form_ue[UE_id], 
434
		   PHY_vars_UE_g[UE_id][0],
Raymond Knopp's avatar
 
Raymond Knopp committed
435 436
		   eNB_id,
		   UE_id,7);
437
            
Raymond Knopp's avatar
 
Raymond Knopp committed
438
    } else {
439
# ifdef ENABLE_XFORMS_WRITE_STATS
Raymond Knopp's avatar
 
Raymond Knopp committed
440
      len =
441
# endif
Raymond Knopp's avatar
 
Raymond Knopp committed
442
	dump_eNB_stats (PHY_vars_eNB_g[0][0], stats_buffer, 0);
Raymond Knopp's avatar
 
Raymond Knopp committed
443
      fl_set_object_label(form_stats->stats_text, stats_buffer);
444
# ifdef ENABLE_XFORMS_WRITE_STATS
Raymond Knopp's avatar
 
Raymond Knopp committed
445 446
      rewind (eNB_stats);
      fwrite (stats_buffer, 1, len, eNB_stats);
447
# endif
Raymond Knopp's avatar
 
Raymond Knopp committed
448 449
      for(UE_id=0;UE_id<scope_enb_num_ue;UE_id++) {
	phy_scope_eNB(form_enb[UE_id], 
Raymond Knopp's avatar
 
Raymond Knopp committed
450
		      PHY_vars_eNB_g[eNB_id][0],
Raymond Knopp's avatar
 
Raymond Knopp committed
451 452
		      UE_id);
      }
453 454
              
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
455 456 457
    //printf("doing forms\n");
    usleep(100000); // 100 ms
  }
458
    
459
# ifdef ENABLE_XFORMS_WRITE_STATS
Raymond Knopp's avatar
 
Raymond Knopp committed
460 461
  fclose (UE_stats);
  fclose (eNB_stats);
462
# endif
463
    
Raymond Knopp's avatar
 
Raymond Knopp committed
464
  pthread_exit((void*)arg);
465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489
}
#endif

#ifdef EMOS
#define NO_ESTIMATES_DISK 100 //No. of estimates that are aquired before dumped to disk

void *emos_thread (void *arg)
{
  char c;
  char *fifo2file_buffer, *fifo2file_ptr;

  int fifo, counter=0, bytes;

  FILE  *dumpfile_id;
  char  dumpfile_name[1024];
  time_t starttime_tmp;
  struct tm starttime;
  
  int channel_buffer_size;
  
  time_t timer;
  struct tm *now;

  struct gps_data_t *gps_data = NULL;
  struct gps_fix_t dummy_gps_data;
490 491 492 493 494 495 496

  struct sched_param sched_param;
  
  sched_param.sched_priority = sched_get_priority_max(SCHED_FIFO)-1; 
  sched_setscheduler(0, SCHED_FIFO,&sched_param);
  
  printf("EMOS thread has priority %d\n",sched_param.sched_priority);
497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518
 
  timer = time(NULL);
  now = localtime(&timer);

  memset(&dummy_gps_data,1,sizeof(struct gps_fix_t));
  
  gps_data = gps_open("127.0.0.1","2947");
  if (gps_data == NULL) 
    {
      printf("[EMOS] Could not open GPS\n");
      //exit(-1);
    }
#if GPSD_API_MAJOR_VERSION>=4
  else if (gps_stream(gps_data, WATCH_ENABLE,NULL) != 0)
#else
  else if (gps_query(gps_data, "w+x") != 0)
#endif
    {
      //sprintf(tmptxt,"Error sending command to GPS, gps_data = %x", gps_data);
      printf("[EMOS] Error sending command to GPS\n");
      //exit(-1);
    }
519 520
  else 
    printf("[EMOS] Opened GPS, gps_data=%p\n");
521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565
  
  if (UE_flag==0)
    channel_buffer_size = sizeof(fifo_dump_emos_eNB);
  else
    channel_buffer_size = sizeof(fifo_dump_emos_UE);

  // allocate memory for NO_FRAMES_DISK channes estimations
  fifo2file_buffer = malloc(NO_ESTIMATES_DISK*channel_buffer_size);
  fifo2file_ptr = fifo2file_buffer;

  if (fifo2file_buffer == NULL)
    {
      printf("[EMOS] Cound not allocate memory for fifo2file_buffer\n");
      exit(EXIT_FAILURE);
    }

  if ((fifo = open(CHANSOUNDER_FIFO_DEV, O_RDONLY)) < 0)
    {
      fprintf(stderr, "[EMOS] Error opening the fifo\n");
      exit(EXIT_FAILURE);
    }


  time(&starttime_tmp);
  localtime_r(&starttime_tmp,&starttime);
  snprintf(dumpfile_name,1024,"/tmp/%s_data_%d%02d%02d_%02d%02d%02d.EMOS",
	   (UE_flag==0) ? "eNB" : "UE",
	   1900+starttime.tm_year, starttime.tm_mon+1, starttime.tm_mday, starttime.tm_hour, starttime.tm_min, starttime.tm_sec);

  dumpfile_id = fopen(dumpfile_name,"w");
  if (dumpfile_id == NULL)
    {
      fprintf(stderr, "[EMOS] Error opening dumpfile %s\n",dumpfile_name);
      exit(EXIT_FAILURE);
    }


  printf("[EMOS] starting dump, channel_buffer_size=%d ...\n",channel_buffer_size);
  while (!oai_exit)
    {
      bytes = rtf_read_timed(fifo, fifo2file_ptr, channel_buffer_size,100);
      if (bytes==0)
	continue;

      /*
Raymond Knopp's avatar
 
Raymond Knopp committed
566
	if (UE_flag==0)
567
	printf("eNB: count %d, frame %d, read: %d bytes from the fifo\n",counter, ((fifo_dump_emos_eNB*)fifo2file_ptr)->frame_tx,bytes);
Raymond Knopp's avatar
 
Raymond Knopp committed
568
	else
569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629
	printf("UE: count %d, frame %d, read: %d bytes from the fifo\n",counter, ((fifo_dump_emos_UE*)fifo2file_ptr)->frame_rx,bytes);
      */

      fifo2file_ptr += channel_buffer_size;
      counter ++;

      if (counter == NO_ESTIMATES_DISK)
        {
          //reset stuff
          fifo2file_ptr = fifo2file_buffer;
          counter = 0;

          //flush buffer to disk
	  if (UE_flag==0)
	    printf("[EMOS] eNB: count %d, frame %d, flushing buffer to disk\n",
		   counter, ((fifo_dump_emos_eNB*)fifo2file_ptr)->frame_tx);
	  else
	    printf("[EMOS] UE: count %d, frame %d, flushing buffer to disk\n",
		   counter, ((fifo_dump_emos_UE*)fifo2file_ptr)->frame_rx);


          if (fwrite(fifo2file_buffer, sizeof(char), NO_ESTIMATES_DISK*channel_buffer_size, dumpfile_id) != NO_ESTIMATES_DISK*channel_buffer_size)
            {
              fprintf(stderr, "[EMOS] Error writing to dumpfile\n");
              exit(EXIT_FAILURE);
            }
	  if (gps_data)
	    {
	      if (gps_poll(gps_data) != 0) {
		printf("[EMOS] problem polling data from gps\n");
	      }
	      else {
		printf("[EMOS] lat %g, lon %g\n",gps_data->fix.latitude,gps_data->fix.longitude);
	      }
	      if (fwrite(&(gps_data->fix), sizeof(char), sizeof(struct gps_fix_t), dumpfile_id) != sizeof(struct gps_fix_t))
		{
		  printf("[EMOS] Error writing to dumpfile, stopping recording\n");
		  exit(EXIT_FAILURE);
		}
	    }
	  else
	    {
	      printf("[EMOS] WARNING: No GPS data available, storing dummy packet\n");
	      if (fwrite(&(dummy_gps_data), sizeof(char), sizeof(struct gps_fix_t), dumpfile_id) != sizeof(struct gps_fix_t))
		{
		  printf("[EMOS] Error writing to dumpfile, stopping recording\n");
		  exit(EXIT_FAILURE);
		}
	    } 
        }
    }
  
  free(fifo2file_buffer);
  fclose(dumpfile_id);
  close(fifo);
  
  pthread_exit((void*) arg);

}
#endif

630
#if defined(ENABLE_ITTI)
631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648
static void wait_system_ready (char *message, volatile int *start_flag)
{
  /* 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]);
      i = (i + 1) % (sizeof(indicator) / sizeof(indicator[0]));
      usleep(200000);
    }
    LOG_D(EMU,"\n");
  }
}
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
649
#if defined(ENABLE_ITTI)
650
void *l2l1_task(void *arg)
651
{
Raymond Knopp's avatar
 
Raymond Knopp committed
652 653 654 655 656
  MessageDef *message_p = NULL;
  int         result;

  itti_set_task_real_time(TASK_L2L1);
  itti_mark_task_ready(TASK_L2L1);
657

Raymond Knopp's avatar
 
Raymond Knopp committed
658 659
  if (UE_flag == 0) {
    /* Wait for the initialize message */
660
    do {
Raymond Knopp's avatar
 
Raymond Knopp committed
661 662 663 664
      if (message_p != NULL) {
	result = itti_free (ITTI_MSG_ORIGIN_ID(message_p), message_p);
	AssertFatal (result == EXIT_SUCCESS, "Failed to free memory (%d)!\n", result);
      }
665 666 667
      itti_receive_msg (TASK_L2L1, &message_p);

      switch (ITTI_MSG_ID(message_p)) {
Raymond Knopp's avatar
 
Raymond Knopp committed
668 669 670 671 672 673 674 675 676 677 678 679 680 681
      case INITIALIZE_MESSAGE:
	/* Start eNB thread */
	LOG_D(EMU, "L2L1 TASK received %s\n", ITTI_MSG_NAME(message_p));
	start_eNB = 1;
	break;

      case TERMINATE_MESSAGE:
	oai_exit=1;
	itti_exit_task ();
	break;

      default:
	LOG_E(EMU, "Received unexpected message %s\n", ITTI_MSG_NAME(message_p));
	break;
682
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704
    } while (ITTI_MSG_ID(message_p) != INITIALIZE_MESSAGE);
    result = itti_free (ITTI_MSG_ORIGIN_ID(message_p), message_p);
    AssertFatal (result == EXIT_SUCCESS, "Failed to free memory (%d)!\n", result);
  }

  do {
    // Wait for a message
    itti_receive_msg (TASK_L2L1, &message_p);

    switch (ITTI_MSG_ID(message_p)) {
    case TERMINATE_MESSAGE:
      oai_exit=1;
      itti_exit_task ();
      break;

    case ACTIVATE_MESSAGE:
      start_UE = 1;
      break;

    case DEACTIVATE_MESSAGE:
      start_UE = 0;
      break;
705

Raymond Knopp's avatar
 
Raymond Knopp committed
706 707 708 709 710 711 712 713 714 715 716 717
    case MESSAGE_TEST:
      LOG_I(EMU, "Received %s\n", ITTI_MSG_NAME(message_p));
      break;

    default:
      LOG_E(EMU, "Received unexpected message %s\n", ITTI_MSG_NAME(message_p));
      break;
    }

    result = itti_free (ITTI_MSG_ORIGIN_ID(message_p), message_p);
    AssertFatal (result == EXIT_SUCCESS, "Failed to free memory (%d)!\n", result);
  } while(1);
718

Raymond Knopp's avatar
 
Raymond Knopp committed
719
  return NULL;
720 721 722
}
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
723 724 725 726 727 728

void do_OFDM_mod(int subframe,PHY_VARS_eNB *phy_vars_eNB) {

  unsigned int aa,slot_offset, slot_offset_F;
  int dummy_tx_b[7680*4] __attribute__((aligned(16)));
  int i, tx_offset;
Raymond Knopp's avatar
 
Raymond Knopp committed
729
  int slot_sizeF = (phy_vars_eNB->lte_frame_parms.ofdm_symbol_size)*
Raymond Knopp's avatar
 
Raymond Knopp committed
730
    ((phy_vars_eNB->lte_frame_parms.Ncp==1) ? 6 : 7);
Raymond Knopp's avatar
 
Raymond Knopp committed
731 732 733

  slot_offset_F = (subframe<<1)*slot_sizeF;
    
Raymond Knopp's avatar
 
Raymond Knopp committed
734 735
  slot_offset = subframe*phy_vars_eNB->lte_frame_parms.samples_per_tti;

Raymond Knopp's avatar
 
Raymond Knopp committed
736 737
  if ((subframe_select(&phy_vars_eNB->lte_frame_parms,subframe)==SF_DL)||
      ((subframe_select(&phy_vars_eNB->lte_frame_parms,subframe)==SF_S))) {
Raymond Knopp's avatar
 
Raymond Knopp committed
738
    //	  LOG_D(HW,"Frame %d: Generating slot %d\n",frame,next_slot);
Raymond Knopp's avatar
 
Raymond Knopp committed
739

Raymond Knopp's avatar
 
Raymond Knopp committed
740
    
Raymond Knopp's avatar
 
Raymond Knopp committed
741 742 743 744 745 746 747 748 749 750
    for (aa=0; aa<phy_vars_eNB->lte_frame_parms.nb_antennas_tx; aa++) {
      if (phy_vars_eNB->lte_frame_parms.Ncp == EXTENDED){ 
	PHY_ofdm_mod(&phy_vars_eNB->lte_eNB_common_vars.txdataF[0][aa][slot_offset_F],
		     dummy_tx_b,
		     phy_vars_eNB->lte_frame_parms.log2_symbol_size,
		     6,
		     phy_vars_eNB->lte_frame_parms.nb_prefix_samples,
		     phy_vars_eNB->lte_frame_parms.twiddle_ifft,
		     phy_vars_eNB->lte_frame_parms.rev,
		     CYCLIC_PREFIX);
Raymond Knopp's avatar
 
Raymond Knopp committed
751 752 753 754 755 756 757 758
	PHY_ofdm_mod(&phy_vars_eNB->lte_eNB_common_vars.txdataF[0][aa][slot_offset_F+slot_sizeF],
		     dummy_tx_b+(phy_vars_eNB->lte_frame_parms.samples_per_tti>>1),
		     phy_vars_eNB->lte_frame_parms.log2_symbol_size,
		     6,
		     phy_vars_eNB->lte_frame_parms.nb_prefix_samples,
		     phy_vars_eNB->lte_frame_parms.twiddle_ifft,
		     phy_vars_eNB->lte_frame_parms.rev,
		     CYCLIC_PREFIX);
Raymond Knopp's avatar
 
Raymond Knopp committed
759 760 761 762 763 764
      }
      else {
	normal_prefix_mod(&phy_vars_eNB->lte_eNB_common_vars.txdataF[0][aa][slot_offset_F],
			  dummy_tx_b,
			  7,
			  &(phy_vars_eNB->lte_frame_parms));
Raymond Knopp's avatar
 
Raymond Knopp committed
765 766 767 768
	normal_prefix_mod(&phy_vars_eNB->lte_eNB_common_vars.txdataF[0][aa][slot_offset_F+slot_sizeF],
			  dummy_tx_b+(phy_vars_eNB->lte_frame_parms.samples_per_tti>>1),
			  7,
			  &(phy_vars_eNB->lte_frame_parms));
Raymond Knopp's avatar
 
Raymond Knopp committed
769
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
770 771

      for (i=0; i<phy_vars_eNB->lte_frame_parms.samples_per_tti; i++) {
Raymond Knopp's avatar
 
Raymond Knopp committed
772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788
	tx_offset = (int)slot_offset+time_offset[aa]+i;
	if (tx_offset<0)
	  tx_offset += LTE_NUMBER_OF_SUBFRAMES_PER_FRAME*phy_vars_eNB->lte_frame_parms.samples_per_tti;
	if (tx_offset>=(LTE_NUMBER_OF_SUBFRAMES_PER_FRAME*phy_vars_eNB->lte_frame_parms.samples_per_tti))
	  tx_offset -= LTE_NUMBER_OF_SUBFRAMES_PER_FRAME*phy_vars_eNB->lte_frame_parms.samples_per_tti;
	((short*)&phy_vars_eNB->lte_eNB_common_vars.txdata[0][aa][tx_offset])[0]=
	  ((short*)dummy_tx_b)[2*i]<<4;
	((short*)&phy_vars_eNB->lte_eNB_common_vars.txdata[0][aa][tx_offset])[1]=
	  ((short*)dummy_tx_b)[2*i+1]<<4;
      }
    }
  }
}


int eNB_thread_tx_status[10];
static void * eNB_thread_tx(void *param) {
Raymond Knopp's avatar
 
Raymond Knopp committed
789 790 791

  //unsigned long cpuid;
  eNB_proc_t *proc = (eNB_proc_t*)param;
Raymond Knopp's avatar
 
Raymond Knopp committed
792
  //  RTIME time_in,time_out;
Raymond Knopp's avatar
 
Raymond Knopp committed
793 794
#ifdef RTAI
  RT_TASK *task;
795
  char task_name[8];
Raymond Knopp's avatar
 
Raymond Knopp committed
796
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
797 798

 
Raymond Knopp's avatar
 
Raymond Knopp committed
799

Raymond Knopp's avatar
 
Raymond Knopp committed
800 801 802 803 804
#if defined(ENABLE_ITTI)
  /* Wait for eNB application initialization to be complete (eNB registration to MME) */
  wait_system_ready ("Waiting for eNB application to be ready %s\r", &start_eNB);
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
805
#ifdef RTAI
806
  sprintf(task_name,"eNTX%d",proc->subframe);
Raymond Knopp's avatar
 
Raymond Knopp committed
807 808 809
  task = rt_task_init_schmod(nam2num(task_name), 0, 0, 0, SCHED_FIFO, 0xF);

  if (task==NULL) {
Raymond Knopp's avatar
 
Raymond Knopp committed
810
    LOG_E(PHY,"[SCHED][eNB] Problem starting eNB_proc_TX thread_index %d (%s)!!!!\n",proc->subframe,task_name);
Raymond Knopp's avatar
 
Raymond Knopp committed
811 812 813
    return 0;
  }
  else {
814
    LOG_I(PHY,"[SCHED][eNB] eNB TX thread %d started with id %p\n",
Raymond Knopp's avatar
 
Raymond Knopp committed
815
	  proc->subframe,
816
	  task);
Raymond Knopp's avatar
 
Raymond Knopp committed
817
  }
Raymond Knopp's avatar
 
Raymond Knopp committed
818
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
819 820 821
  //  LOG_I(PHY,
	printf("[SCHED][eNB] eNB TX thread %d started on CPU %d\n",
	       proc->subframe,sched_getcpu());
Raymond Knopp's avatar
 
Raymond Knopp committed
822 823 824 825 826 827 828 829 830 831 832 833
#endif

  mlockall(MCL_CURRENT | MCL_FUTURE);

  //rt_set_runnable_on_cpuid(task,1);
  //cpuid = rtai_cpuid();

#ifdef HARD_RT
  rt_make_hard_real_time();
#endif

  while (!oai_exit){
Raymond Knopp's avatar
 
Raymond Knopp committed
834
    
Raymond Knopp's avatar
 
Raymond Knopp committed
835
    vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_TX0+(2*proc->subframe),0);
Raymond Knopp's avatar
 
Raymond Knopp committed
836 837
    
    
Raymond Knopp's avatar
 
Raymond Knopp committed
838
    //    LOG_I(PHY,"Locking mutex for eNB proc %d (IC %d,mutex %p)\n",proc->subframe,proc->instance_cnt,&proc->mutex);
Raymond Knopp's avatar
 
Raymond Knopp committed
839 840
    //printf("Locking mutex for eNB proc %d (subframe_tx %d))\n",proc->subframe,subframe_tx);

Raymond Knopp's avatar
 
Raymond Knopp committed
841 842
    if (pthread_mutex_lock(&proc->mutex_tx) != 0) {
      LOG_E(PHY,"[SCHED][eNB] error locking mutex for eNB TX proc %d\n",proc->subframe);
Raymond Knopp's avatar
 
Raymond Knopp committed
843 844
    }
    else {
Raymond Knopp's avatar
 
Raymond Knopp committed
845
      
Raymond Knopp's avatar
 
Raymond Knopp committed
846 847
      while (proc->instance_cnt_tx < 0) {
	//	LOG_I(PHY,"Waiting and unlocking mutex for eNB proc %d (IC %d,lock %d)\n",proc->subframe,proc->instance_cnt,pthread_mutex_trylock(&proc->mutex));
Raymond Knopp's avatar
 
Raymond Knopp committed
848
	//printf("Waiting and unlocking mutex for eNB proc %d (subframe_tx %d)\n",proc->subframe,subframe_tx);
Raymond Knopp's avatar
 
Raymond Knopp committed
849
	
Raymond Knopp's avatar
 
Raymond Knopp committed
850
	pthread_cond_wait(&proc->cond_tx,&proc->mutex_tx);
Raymond Knopp's avatar
 
Raymond Knopp committed
851
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
852 853 854
      //      LOG_I(PHY,"Waking up and unlocking mutex for eNB proc %d\n",proc->subframe);
      if (pthread_mutex_unlock(&proc->mutex_tx) != 0) {	
	LOG_E(PHY,"[SCHED][eNB] error unlocking mutex for eNB TX proc %d\n",proc->subframe);
Raymond Knopp's avatar
 
Raymond Knopp committed
855 856
      }
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
857 858 859
    vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_TX0+(2*proc->subframe),1);    
    vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_FRAME_NUMBER_ENB, proc->frame_tx);
    vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_SLOT_NUMBER_ENB, proc->subframe*2);
Raymond Knopp's avatar
 
Raymond Knopp committed
860
    
Raymond Knopp's avatar
 
Raymond Knopp committed
861
    if (oai_exit) break;
Raymond Knopp's avatar
 
Raymond Knopp committed
862

Raymond Knopp's avatar
 
Raymond Knopp committed
863 864 865 866 867 868 869
    
    if ((((PHY_vars_eNB_g[0][proc->CC_id]->lte_frame_parms.frame_type == TDD)&&
	  (subframe_select(&PHY_vars_eNB_g[0][proc->CC_id]->lte_frame_parms,proc->subframe_tx)==SF_DL))||
	 (PHY_vars_eNB_g[0][proc->CC_id]->lte_frame_parms.frame_type == FDD))) {
      
      phy_procedures_eNB_TX(proc->subframe,PHY_vars_eNB_g[0][proc->CC_id],0,no_relay,NULL);
      
Raymond Knopp's avatar
 
Raymond Knopp committed
870
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
871 872
    if ((subframe_select(&PHY_vars_eNB_g[0][proc->CC_id]->lte_frame_parms,proc->subframe_tx)==SF_S)) {
      phy_procedures_eNB_TX(proc->subframe,PHY_vars_eNB_g[0][proc->CC_id],0,no_relay,NULL);
Raymond Knopp's avatar
 
Raymond Knopp committed
873
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
874
    
Raymond Knopp's avatar
 
Raymond Knopp committed
875 876
    
    do_OFDM_mod(proc->subframe_tx,PHY_vars_eNB_g[0][proc->CC_id]);  
Raymond Knopp's avatar
 
Raymond Knopp committed
877
    
Raymond Knopp's avatar
 
Raymond Knopp committed
878 879
    if (pthread_mutex_lock(&proc->mutex_tx) != 0) {
      printf("[openair][SCHED][eNB] error locking mutex for eNB TX proc %d\n",proc->subframe);
Raymond Knopp's avatar
 
Raymond Knopp committed
880
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
881 882
    else {
      proc->instance_cnt_tx--;
Raymond Knopp's avatar
 
Raymond Knopp committed
883
      
Raymond Knopp's avatar
 
Raymond Knopp committed
884 885 886 887
      if (pthread_mutex_unlock(&proc->mutex_tx) != 0) {	
	printf("[openair][SCHED][eNB] error unlocking mutex for eNB TX proc %d\n",proc->subframe);
      }
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
888
    
Raymond Knopp's avatar
 
Raymond Knopp committed
889 890 891
    proc->frame_tx++;
    if (proc->frame_tx==1024)
      proc->frame_tx=0;
Raymond Knopp's avatar
 
Raymond Knopp committed
892
  }    
Raymond Knopp's avatar
 
Raymond Knopp committed
893 894 895 896
  vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_TX0+(2*proc->subframe),0);        
#ifdef HARD_RT
  rt_make_soft_real_time();
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
897
  
Raymond Knopp's avatar
 
Raymond Knopp committed
898 899 900 901 902 903 904 905 906 907
#ifdef DEBUG_THREADS
  printf("Exiting eNB thread TX %d\n",proc->subframe);
#endif
  // clean task
#ifdef RTAI
  rt_task_delete(task);
#else
  eNB_thread_tx_status[proc->subframe]=0;
  pthread_exit(&eNB_thread_tx_status[proc->subframe]);
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
908
  
Raymond Knopp's avatar
 
Raymond Knopp committed
909 910 911 912 913 914 915 916 917 918 919 920 921
#ifdef DEBUG_THREADS
  printf("Exiting eNB TX thread %d\n",proc->subframe);
#endif
}

int eNB_thread_rx_status[10];
static void * eNB_thread_rx(void *param) {

  //unsigned long cpuid;
  eNB_proc_t *proc = (eNB_proc_t*)param;
  //  RTIME time_in,time_out;
#ifdef RTAI
  RT_TASK *task;
922
  char task_name[8];
Raymond Knopp's avatar
 
Raymond Knopp committed
923 924 925 926 927 928 929 930
#endif

#if defined(ENABLE_ITTI)
  /* Wait for eNB application initialization to be complete (eNB registration to MME) */
  wait_system_ready ("Waiting for eNB application to be ready %s\r", &start_eNB);
#endif

#ifdef RTAI
931
  sprintf(task_name,"eNRX%d",proc->subframe);
Raymond Knopp's avatar
 
Raymond Knopp committed
932 933 934 935 936 937 938
  task = rt_task_init_schmod(nam2num(task_name), 0, 0, 0, SCHED_FIFO, 0xF);

  if (task==NULL) {
    LOG_E(PHY,"[SCHED][eNB] Problem starting eNB_proc_RX thread_index %d (%s)!!!!\n",proc->subframe,task_name);
    return 0;
  }
  else {
939
    LOG_I(PHY,"[SCHED][eNB] eNB RX thread %d started with id %p\n", /*  on CPU %d*/
Raymond Knopp's avatar
 
Raymond Knopp committed
940
	  proc->subframe,
941
	  task); /*,rtai_cpuid()*/
Raymond Knopp's avatar
 
Raymond Knopp committed
942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971
  }
#else
  LOG_I(PHY,"[SCHED][eNB] eNB RX thread %d started on CPU %d\n",
	proc->subframe,sched_getcpu());
#endif

  mlockall(MCL_CURRENT | MCL_FUTURE);

  //rt_set_runnable_on_cpuid(task,1);
  //cpuid = rtai_cpuid();

#ifdef HARD_RT
  rt_make_hard_real_time();
#endif

  while (!oai_exit){

    vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_RX0+(2*proc->subframe),0);


    //    LOG_I(PHY,"Locking mutex for eNB proc %d (IC %d,mutex %p)\n",proc->subframe,proc->instance_cnt,&proc->mutex);
    if (pthread_mutex_lock(&proc->mutex_rx) != 0) {
      LOG_E(PHY,"[SCHED][eNB] error locking mutex for eNB RX proc %d\n",proc->subframe);
    }
    else {
        
      while (proc->instance_cnt_rx < 0) {
	//	LOG_I(PHY,"Waiting and unlocking mutex for eNB proc %d (IC %d,lock %d)\n",proc->subframe,proc->instance_cnt,pthread_mutex_trylock(&proc->mutex));

	pthread_cond_wait(&proc->cond_rx,&proc->mutex_rx);
Raymond Knopp's avatar
 
Raymond Knopp committed
972
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
973 974 975
      //      LOG_I(PHY,"Waking up and unlocking mutex for eNB proc %d\n",proc->subframe);
      if (pthread_mutex_unlock(&proc->mutex_rx) != 0) {	
	LOG_E(PHY,"[SCHED][eNB] error unlocking mutex for eNB RX proc %d\n",proc->subframe);
Raymond Knopp's avatar
 
Raymond Knopp committed
976 977
      }
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
978 979 980 981
    vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_RX0+(2*proc->subframe),1);    

    if (oai_exit) break;
    
Raymond Knopp's avatar
 
Raymond Knopp committed
982 983 984
    if ((((PHY_vars_eNB_g[0][proc->CC_id]->lte_frame_parms.frame_type == TDD )&&(subframe_select(&PHY_vars_eNB_g[0][proc->CC_id]->lte_frame_parms,proc->subframe_rx)==SF_UL)) ||
	 (PHY_vars_eNB_g[0][proc->CC_id]->lte_frame_parms.frame_type == FDD))){
      phy_procedures_eNB_RX(proc->subframe,PHY_vars_eNB_g[0][proc->CC_id],0,no_relay);
Raymond Knopp's avatar
 
Raymond Knopp committed
985
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
986 987
    if ((subframe_select(&PHY_vars_eNB_g[0][proc->CC_id]->lte_frame_parms,proc->subframe_rx)==SF_S)){
      phy_procedures_eNB_S_RX(proc->subframe,PHY_vars_eNB_g[0][proc->CC_id],0,no_relay);
Raymond Knopp's avatar
 
Raymond Knopp committed
988 989 990 991
    }
      
    if (pthread_mutex_lock(&proc->mutex_rx) != 0) {
      printf("[openair][SCHED][eNB] error locking mutex for eNB RX proc %d\n",proc->subframe);
Raymond Knopp's avatar
 
Raymond Knopp committed
992 993
    }
    else {
Raymond Knopp's avatar
 
Raymond Knopp committed
994
      proc->instance_cnt_rx--;
Raymond Knopp's avatar
 
Raymond Knopp committed
995
      
Raymond Knopp's avatar
 
Raymond Knopp committed
996 997
      if (pthread_mutex_unlock(&proc->mutex_rx) != 0) {	
	printf("[openair][SCHED][eNB] error unlocking mutex for eNB RX proc %d\n",proc->subframe);
Raymond Knopp's avatar
 
Raymond Knopp committed
998 999
      }
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
1000 1001 1002 1003

    proc->frame_rx++;
    if (proc->frame_rx==1024)
      proc->frame_rx=0;
Raymond Knopp's avatar
 
Raymond Knopp committed
1004
    
Raymond Knopp's avatar
 
Raymond Knopp committed
1005 1006
  }
  vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_RX0+(2*proc->subframe),0);        
Raymond Knopp's avatar
 
Raymond Knopp committed
1007
#ifdef HARD_RT
Raymond Knopp's avatar
 
Raymond Knopp committed
1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019
  rt_make_soft_real_time();
#endif

#ifdef DEBUG_THREADS
  printf("Exiting eNB thread RX %d\n",proc->subframe);
#endif
  // clean task
#ifdef RTAI
  rt_task_delete(task);
#else
  eNB_thread_rx_status[proc->subframe]=0;
  pthread_exit(&eNB_thread_rx_status[proc->subframe]);
Raymond Knopp's avatar
 
Raymond Knopp committed
1020 1021
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
1022 1023 1024
#ifdef DEBUG_THREADS
  printf("Exiting eNB RX thread %d\n",proc->subframe);
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1025 1026 1027 1028 1029 1030 1031
}



void init_eNB_proc() {

  int i;
Raymond Knopp's avatar
 
Raymond Knopp committed
1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061
  int CC_id;

  for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
    for (i=0;i<10;i++) {
      pthread_attr_init (&attr_eNB_proc_tx[CC_id][i]);
      pthread_attr_setstacksize(&attr_eNB_proc_tx[CC_id][i],OPENAIR_THREAD_STACK_SIZE);
      //attr_dlsch_threads.priority = 1;
      sched_param_eNB_proc_tx[CC_id][i].sched_priority = sched_get_priority_max(SCHED_FIFO)-1; //OPENAIR_THREAD_PRIORITY;
      pthread_attr_setschedparam  (&attr_eNB_proc_tx[CC_id][i], &sched_param_eNB_proc_tx[CC_id][i]);
      pthread_attr_setschedpolicy (&attr_eNB_proc_tx[CC_id][i], SCHED_FIFO);
      
      pthread_attr_init (&attr_eNB_proc_rx[CC_id][i]);
      pthread_attr_setstacksize(&attr_eNB_proc_rx[CC_id][i],OPENAIR_THREAD_STACK_SIZE);
      //attr_dlsch_threads.priority = 1;
      sched_param_eNB_proc_rx[CC_id][i].sched_priority = sched_get_priority_max(SCHED_FIFO)-1; //OPENAIR_THREAD_PRIORITY;
      pthread_attr_setschedparam  (&attr_eNB_proc_rx[CC_id][i], &sched_param_eNB_proc_rx[CC_id][i]);
      pthread_attr_setschedpolicy (&attr_eNB_proc_rx[CC_id][i], SCHED_FIFO);
      
	PHY_vars_eNB_g[0][CC_id]->proc[i].instance_cnt_tx=-1;
	PHY_vars_eNB_g[0][CC_id]->proc[i].instance_cnt_rx=-1;
	PHY_vars_eNB_g[0][CC_id]->proc[i].subframe=i;
	PHY_vars_eNB_g[0][CC_id]->proc[i].CC_id = CC_id;
	pthread_mutex_init(&PHY_vars_eNB_g[0][CC_id]->proc[i].mutex_tx,NULL);
	pthread_mutex_init(&PHY_vars_eNB_g[0][CC_id]->proc[i].mutex_rx,NULL);
	pthread_cond_init(&PHY_vars_eNB_g[0][CC_id]->proc[i].cond_tx,NULL);
	pthread_cond_init(&PHY_vars_eNB_g[0][CC_id]->proc[i].cond_rx,NULL);
	pthread_create(&PHY_vars_eNB_g[0][CC_id]->proc[i].pthread_tx,NULL,eNB_thread_tx,(void*)&PHY_vars_eNB_g[0][CC_id]->proc[i]);
	pthread_create(&PHY_vars_eNB_g[0][CC_id]->proc[i].pthread_rx,NULL,eNB_thread_rx,(void*)&PHY_vars_eNB_g[0][CC_id]->proc[i]);
	PHY_vars_eNB_g[0][CC_id]->proc[i].frame_tx = 0;
	PHY_vars_eNB_g[0][CC_id]->proc[i].frame_rx = 0;
Raymond Knopp's avatar
 
Raymond Knopp committed
1062
#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
1063 1064
	PHY_vars_eNB_g[0][CC_id]->proc[i].subframe_rx = (i+9)%10;
	PHY_vars_eNB_g[0][CC_id]->proc[i].subframe_tx = (i+1)%10;
Raymond Knopp's avatar
 
Raymond Knopp committed
1065
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
1066 1067
	PHY_vars_eNB_g[0][CC_id]->proc[i].subframe_rx = i;
	PHY_vars_eNB_g[0][CC_id]->proc[i].subframe_tx = (i+2)%10;
Raymond Knopp's avatar
 
Raymond Knopp committed
1068
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1069 1070 1071
    }
  
  
Raymond Knopp's avatar
 
Raymond Knopp committed
1072 1073
#ifndef USRP
  // TX processes subframe + 1, RX subframe -1
Raymond Knopp's avatar
 
Raymond Knopp committed
1074
  // Note this inialization is because the first process awoken for frame 0 is number 1 and so processes 9 and 0 have to start with frame 1
Raymond Knopp's avatar
 
Raymond Knopp committed
1075 1076 1077 1078
    
  //PHY_vars_eNB_g[0][CC_id]->proc[0].frame_rx = 1023;
    PHY_vars_eNB_g[0][CC_id]->proc[9].frame_tx = 1;
    PHY_vars_eNB_g[0][CC_id]->proc[0].frame_tx = 1;
Raymond Knopp's avatar
 
Raymond Knopp committed
1079
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
1080 1081 1082 1083 1084
    // TX processes subframe +2, RX subframe
    // Note this inialization is because the first process awoken for frame 0 is number 1 and so processes 8,9 and 0 have to start with frame 1
    PHY_vars_eNB_g[0][CC_id]->proc[8].frame_tx = 1;
    PHY_vars_eNB_g[0][CC_id]->proc[9].frame_tx = 1;
    PHY_vars_eNB_g[0][CC_id]->proc[0].frame_tx = 1;
Raymond Knopp's avatar
 
Raymond Knopp committed
1085
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1086 1087
  }
}
Raymond Knopp's avatar
 
Raymond Knopp committed
1088 1089 1090 1091

void kill_eNB_proc() {

  int i;
Raymond Knopp's avatar
 
Raymond Knopp committed
1092
  int *status_tx,*status_rx;
Raymond Knopp's avatar
 
Raymond Knopp committed
1093
  int CC_id;
Raymond Knopp's avatar
 
Raymond Knopp committed
1094

Raymond Knopp's avatar
 
Raymond Knopp committed
1095 1096 1097
  for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) 
    for (i=0;i<10;i++) {
      
Raymond Knopp's avatar
 
Raymond Knopp committed
1098
#ifdef DEBUG_THREADS
Raymond Knopp's avatar
 
Raymond Knopp committed
1099
      printf("Killing TX CC_id %d thread %d\n",CC_id,i);
Raymond Knopp's avatar
 
Raymond Knopp committed
1100
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1101 1102
      PHY_vars_eNB_g[0][CC_id]->proc[i].instance_cnt_tx=0; 
      pthread_cond_signal(&PHY_vars_eNB_g[0][CC_id]->proc[i].cond_tx);
Raymond Knopp's avatar
 
Raymond Knopp committed
1103
#ifdef DEBUG_THREADS
Raymond Knopp's avatar
 
Raymond Knopp committed
1104
      printf("Joining eNB TX CC_id %d thread %d...",CC_id,i);
Raymond Knopp's avatar
 
Raymond Knopp committed
1105
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1106
      pthread_join(PHY_vars_eNB_g[0][CC_id]->proc[i].pthread_tx,(void**)status_tx);
Raymond Knopp's avatar
 
Raymond Knopp committed
1107
#ifdef DEBUG_THREADS
Raymond Knopp's avatar
 
Raymond Knopp committed
1108
      if (status_tx) printf("status %d...",*status_tx);
Raymond Knopp's avatar
 
Raymond Knopp committed
1109 1110
#endif
#ifdef DEBUG_THREADS
Raymond Knopp's avatar
 
Raymond Knopp committed
1111
      printf("Killing RX CC_id %d thread %d\n",CC_id,i);
Raymond Knopp's avatar
 
Raymond Knopp committed
1112
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1113 1114
      PHY_vars_eNB_g[0][CC_id]->proc[i].instance_cnt_rx=0; 
      pthread_cond_signal(&PHY_vars_eNB_g[0][CC_id]->proc[i].cond_rx);
Raymond Knopp's avatar
 
Raymond Knopp committed
1115
#ifdef DEBUG_THREADS
Raymond Knopp's avatar
 
Raymond Knopp committed
1116
      printf("Joining eNB RX CC_id %d thread %d...",CC_id,i);
Raymond Knopp's avatar
 
Raymond Knopp committed
1117
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1118
      pthread_join(PHY_vars_eNB_g[0][CC_id]->proc[i].pthread_rx,(void**)status_rx);
Raymond Knopp's avatar
 
Raymond Knopp committed
1119
#ifdef DEBUG_THREADS 
Raymond Knopp's avatar
 
Raymond Knopp committed
1120
      if (status_rx) printf("status %d...",*status_rx);
Raymond Knopp's avatar
 
Raymond Knopp committed
1121
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1122 1123 1124 1125 1126
      pthread_mutex_destroy(&PHY_vars_eNB_g[0][CC_id]->proc[i].mutex_tx);
      pthread_mutex_destroy(&PHY_vars_eNB_g[0][CC_id]->proc[i].mutex_rx);
      pthread_cond_destroy(&PHY_vars_eNB_g[0][CC_id]->proc[i].cond_tx);
      pthread_cond_destroy(&PHY_vars_eNB_g[0][CC_id]->proc[i].cond_rx);
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
1127 1128 1129
}


Raymond Knopp's avatar
 
Raymond Knopp committed
1130 1131


Raymond Knopp's avatar
 
Raymond Knopp committed
1132 1133
  
/* This is the main eNB thread. */
Raymond Knopp's avatar
 
Raymond Knopp committed
1134 1135
int eNB_thread_status;

Raymond Knopp's avatar
 
Raymond Knopp committed
1136

Raymond Knopp's avatar
 
Raymond Knopp committed
1137

1138 1139 1140 1141 1142
static void *eNB_thread(void *arg)
{
#ifdef RTAI
  RT_TASK *task;
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1143 1144 1145 1146 1147
#ifndef USRP
  unsigned char slot=0;
#else
  unsigned char slot=1;
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1148
  int frame=0;
Raymond Knopp's avatar
 
Raymond Knopp committed
1149
  int CC_id;
Raymond Knopp's avatar
 
Raymond Knopp committed
1150

1151
  RTIME time_in, time_diff;
Raymond Knopp's avatar
 
Raymond Knopp committed
1152

Raymond Knopp's avatar
 
Raymond Knopp committed
1153
  int sf;
Raymond Knopp's avatar
 
Raymond Knopp committed
1154
#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
1155
  volatile unsigned int *DAQ_MBOX = openair0_daq_cnt();
Raymond Knopp's avatar
 
Raymond Knopp committed
1156 1157
  int mbox_target=0,mbox_current=0;
  int hw_slot,delay_cnt;
Raymond Knopp's avatar
 
Raymond Knopp committed
1158 1159 1160
  int diff;
  int ret;

Raymond Knopp's avatar
 
Raymond Knopp committed
1161
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
1162 1163
  unsigned int rx_cnt = 0;
  unsigned int tx_cnt = tx_delay;
Raymond Knopp's avatar
 
Raymond Knopp committed
1164
  //  int tx_offset;
Raymond Knopp's avatar
 
Raymond Knopp committed
1165
  int CC_id;
Raymond Knopp's avatar
 
Raymond Knopp committed
1166

Raymond Knopp's avatar
 
Raymond Knopp committed
1167
  hw_subframe = 0;
Raymond Knopp's avatar
 
Raymond Knopp committed
1168 1169 1170



Raymond Knopp's avatar
 
Raymond Knopp committed
1171
#endif
1172 1173
#if defined(ENABLE_ITTI)
  /* Wait for eNB application initialization to be complete (eNB registration to MME) */
1174
  wait_system_ready ("Waiting for eNB application to be ready %s\r", &start_eNB);
1175 1176
#endif

1177
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
1178
  task = rt_task_init_schmod(nam2num("TASK0"), 0, 0, 0, SCHED_FIFO, 0xF);
1179 1180 1181 1182
#endif

  if (!oai_exit) {
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
1183
    printf("[SCHED][eNB] Started eNB thread (id %p)\n",task);
Raymond Knopp's avatar
 
Raymond Knopp committed
1184
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
1185 1186
    printf("[SCHED][eNB] Started eNB thread on CPU %d\n",
	   sched_getcpu());
1187 1188 1189
#endif

#ifdef HARD_RT
1190
    rt_make_hard_real_time();
1191 1192
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
1193
    printf("eNB_thread: mlockall in ...\n");
1194
    mlockall(MCL_CURRENT | MCL_FUTURE);
Raymond Knopp's avatar
 
Raymond Knopp committed
1195
    printf("eNB_thread: mlockall out ...\n");
1196

1197 1198 1199 1200
    timing_info.time_min = 100000000ULL;
    timing_info.time_max = 0;
    timing_info.time_avg = 0;
    timing_info.n_samples = 0;
1201

Raymond Knopp's avatar
 
Raymond Knopp committed
1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214
#ifdef USRP
    printf("waiting for USRP sync \n");
#ifdef RTAI
    rt_sem_wait(sync_sem);
#else
    //pthread_mutex_lock(&sync_mutex);
    pthread_cond_wait(&sync_cond, &sync_mutex);
    //pthread_mutex_unlock(&sync_mutex);
#endif
    //    printf("starting eNB thread @ %llu\n",get_usrp_time(&openair0));
#endif

  while (!oai_exit) {
Raymond Knopp's avatar
 
Raymond Knopp committed
1215 1216

#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
1217 1218
      hw_slot = (((((volatile unsigned int *)DAQ_MBOX)[0]+1)%150)<<1)/15;
      //        LOG_D(HW,"eNB frame %d, time %llu: slot %d, hw_slot %d (mbox %d)\n",frame,rt_get_time_ns(),slot,hw_slot,((unsigned int *)DAQ_MBOX)[0]);
Raymond Knopp's avatar
 
Raymond Knopp committed
1219 1220
      vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_HW_SUBFRAME, hw_slot>>1);
      vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_HW_FRAME, frame);
Raymond Knopp's avatar
 
Raymond Knopp committed
1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248
      //this is the mbox counter where we should be
      mbox_target = mbox_bounds[slot];
      //this is the mbox counter where we are
      mbox_current = ((volatile unsigned int *)DAQ_MBOX)[0];
      //this is the time we need to sleep in order to synchronize with the hw (in multiples of DAQ_PERIOD)
      if ((mbox_current>=135) && (mbox_target<15)) //handle the frame wrap-arround
	diff = 150-mbox_current+mbox_target;
      else if ((mbox_current<15) && (mbox_target>=135))
	diff = -150+mbox_target-mbox_current;
      else
	diff = mbox_target - mbox_current;
      
      if (((slot%2==0) && (diff < (-14))) || ((slot%2==1) && (diff < (-7)))) {
	// at the eNB, even slots have double as much time since most of the processing is done here and almost nothing in odd slots
	LOG_D(HW,"eNB Frame %d, time %llu: missed slot, proceeding with next one (slot %d, hw_slot %d, diff %d)\n",frame, rt_get_time_ns(), slot, hw_slot, diff);
	slot++;
	if (frame > 0) {
	  exit_fun("[HW][eNB] missed slot");
	}
	if (slot==20){
	  slot=0;
	  frame++;
	}
	continue;
      }
      if (diff>8)
	LOG_D(HW,"eNB Frame %d, time %llu: skipped slot, waiting for hw to catch up (slot %d, hw_slot %d, mbox_current %d, mbox_target %d, diff %d)\n",frame, rt_get_time_ns(), slot, hw_slot, mbox_current, mbox_target, diff);
      
Raymond Knopp's avatar
 
Raymond Knopp committed
1249
      vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_DAQ_MBOX, *DAQ_MBOX);
Raymond Knopp's avatar
 
Raymond Knopp committed
1250 1251 1252 1253 1254 1255 1256 1257 1258 1259
      vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_DIFF, diff);
      
      delay_cnt = 0;
      while ((diff>0) && (!oai_exit)) {
	time_in = rt_get_time_ns();
	//LOG_D(HW,"eNB Frame %d delaycnt %d : hw_slot %d (%d), slot %d, (slot+1)*15=%d, diff %d, time %llu\n",frame,delay_cnt,hw_slot,((unsigned int *)DAQ_MBOX)[0],slot,(((slot+1)*15)>>1),diff,time_in);
	//LOG_D(HW,"eNB Frame %d, time %llu: sleeping for %llu (slot %d, hw_slot %d, diff %d, mbox %d, delay_cnt %d)\n", frame, time_in, diff*DAQ_PERIOD,slot,hw_slot,diff,((volatile unsigned int *)DAQ_MBOX)[0],delay_cnt);
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_RT_SLEEP,1);
	ret = rt_sleep_ns(diff*DAQ_PERIOD);
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_RT_SLEEP,0);
Raymond Knopp's avatar
 
Raymond Knopp committed
1260
	vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_DAQ_MBOX, *DAQ_MBOX);
Raymond Knopp's avatar
 
Raymond Knopp committed
1261 1262 1263 1264 1265 1266 1267 1268 1269 1270 1271
	if (ret)
	  LOG_D(HW,"eNB Frame %d, time %llu: rt_sleep_ns returned %d\n",frame, time_in);
	hw_slot = (((((volatile unsigned int *)DAQ_MBOX)[0]+1)%150)<<1)/15;
	//LOG_D(HW,"eNB Frame %d : hw_slot %d, time %llu\n",frame,hw_slot,rt_get_time_ns());
	delay_cnt++;
	if (delay_cnt == 10) {
	  LOG_D(HW,"eNB Frame %d: HW stopped ... \n",frame);
	  exit_fun("[HW][eNB] HW stopped");
	}
	mbox_current = ((volatile unsigned int *)DAQ_MBOX)[0];
	if ((mbox_current>=135) && (mbox_target<15)) //handle the frame wrap-arround
Raymond Knopp's avatar
 
Raymond Knopp committed
1272
	  diff = 150-mbox_current+mbox_target;
Raymond Knopp's avatar
 
Raymond Knopp committed
1273
	else if ((mbox_current<15) && (mbox_target>=135))
Raymond Knopp's avatar
 
Raymond Knopp committed
1274
	  diff = -150+mbox_target-mbox_current;
Raymond Knopp's avatar
 
Raymond Knopp committed
1275
	else
Raymond Knopp's avatar
 
Raymond Knopp committed
1276
	  diff = mbox_target - mbox_current;
Raymond Knopp's avatar
 
Raymond Knopp committed
1277
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1278 1279

#else  // USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
1280 1281
      vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_HW_SUBFRAME, hw_subframe);
      vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_HW_FRAME, frame);
Raymond Knopp's avatar
 
Raymond Knopp committed
1282 1283
      while (rx_cnt < sf_bounds[hw_subframe]) {

Raymond Knopp's avatar
 
Raymond Knopp committed
1284 1285 1286
	openair0_timestamp time0,time1;
	unsigned int rxs;
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_TRX_READ,1);
Raymond Knopp's avatar
 
Raymond Knopp committed
1287 1288 1289
	/*
	// Grab 1/4 of RX buffer and get timestamp

Raymond Knopp's avatar
 
Raymond Knopp committed
1290 1291 1292 1293 1294
	rxs = openair0.trx_read_func(&openair0, 
				     &timestamp, 
				     &rxdata[rx_cnt*samples_per_packets], 
				     (samples_per_packets>>2));
	if (rxs != (samples_per_packets>>2))
Raymond Knopp's avatar
 
Raymond Knopp committed
1295
	  oai_exit=1;
Raymond Knopp's avatar
 
Raymond Knopp committed
1296 1297 1298 1299 1300 1301 1302 1303 1304 1305

	*/

	rxs = openair0.trx_read_func(&openair0, 
				     &timestamp, 
				     &rxdata[rx_cnt*samples_per_packets], 
				     samples_per_packets);
	if (rxs != samples_per_packets)
	  oai_exit=1;

Raymond Knopp's avatar
 
Raymond Knopp committed
1306 1307
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_TRX_READ,0);

Raymond Knopp's avatar
 
Raymond Knopp committed
1308
	// Transmit TX buffer based on timestamp from RX
Raymond Knopp's avatar
 
Raymond Knopp committed
1309
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_TRX_WRITE,1);
Raymond Knopp's avatar
 
Raymond Knopp committed
1310 1311 1312 1313 1314
	openair0.trx_write_func(&openair0, 
				(timestamp+samples_per_packets*tx_delay-tx_forward_nsamps), 
				&txdata[tx_cnt*samples_per_packets], 
				samples_per_packets, 
				1);
Raymond Knopp's avatar
 
Raymond Knopp committed
1315
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_TRX_WRITE,0);
Raymond Knopp's avatar
 
Raymond Knopp committed
1316
	/*
Raymond Knopp's avatar
 
Raymond Knopp committed
1317 1318 1319 1320 1321 1322 1323 1324 1325
	// Grab remaining 3/4 of RX buffer
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_TRX_READ,1);
	rxs = openair0.trx_read_func(&openair0, 
				     &timestamp, 
				     &rxdata[(rx_cnt*samples_per_packets)+(samples_per_packets>>2)], 
				     3*((samples_per_packets>>2)));
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_TRX_READ,0);
	if (rxs != (3*(samples_per_packets>>2)))
	  oai_exit=1;
Raymond Knopp's avatar
 
Raymond Knopp committed
1326
	*/
Raymond Knopp's avatar
 
Raymond Knopp committed
1327

Raymond Knopp's avatar
 
Raymond Knopp committed
1328 1329
	rx_cnt++;
	tx_cnt++;
Raymond Knopp's avatar
 
Raymond Knopp committed
1330 1331 1332

	if(tx_cnt == max_cnt)
	  tx_cnt = 0;
Raymond Knopp's avatar
 
Raymond Knopp committed
1333 1334
      }

Raymond Knopp's avatar
 
Raymond Knopp committed
1335 1336
      if(rx_cnt == max_cnt)
	rx_cnt = 0; 
Raymond Knopp's avatar
 
Raymond Knopp committed
1337
      
Raymond Knopp's avatar
 
Raymond Knopp committed
1338 1339 1340

#endif // USRP
     
Raymond Knopp's avatar
 
Raymond Knopp committed
1341
      if (oai_exit) break;
1342

Raymond Knopp's avatar
 
Raymond Knopp committed
1343
      if (frame>99)  {
Raymond Knopp's avatar
 
Raymond Knopp committed
1344

Raymond Knopp's avatar
 
Raymond Knopp committed
1345 1346 1347 1348 1349 1350 1351 1352 1353 1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368 1369
	timing_info.time_last = timing_info.time_now;
	timing_info.time_now = rt_get_time_ns();
	
	if (timing_info.n_samples>0) {
	  time_diff = timing_info.time_now - timing_info.time_last;
	  if (time_diff < timing_info.time_min)
	    timing_info.time_min = time_diff;
	  if (time_diff > timing_info.time_max)
	    timing_info.time_max = time_diff;
	  timing_info.time_avg += time_diff;
	}
	
	timing_info.n_samples++;
	/*
	  if ((timing_info.n_samples%2000)==0) {
	  LOG_D(HW,"frame %d (%d), slot %d, hw_slot %d: diff=%llu, min=%llu, max=%llu, avg=%llu (n_samples %d)\n",
	  frame, PHY_vars_eNB_g[0]->frame, slot, hw_slot,time_diff,
	  timing_info.time_min,timing_info.time_max,timing_info.time_avg/timing_info.n_samples,timing_info.n_samples);
	  timing_info.n_samples = 0;
	  timing_info.time_avg = 0;
	  }
	*/
	//}
	
	if (multi_thread == 0) {
Raymond Knopp's avatar
 
Raymond Knopp committed
1370 1371
	  if ((slot&1) == 0) {
	    LOG_I(PHY,"[eNB] Single thread slot %d\n",slot);
1372 1373 1374 1375
	      phy_procedures_eNB_lte ((2+(slot>>1))%10, PHY_vars_eNB_g[0], 0, no_relay,NULL);
	      for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
		do_OFDM_mod((2+(slot>>1))%10,PHY_vars_eNB_g[0][CC_id]);
	      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1376
	  }
Raymond Knopp's avatar
 
Raymond Knopp committed
1377 1378
	}
	else { // multi-thread > 0
Raymond Knopp's avatar
 
Raymond Knopp committed
1379
	  if ((slot&1) == 1) {
Raymond Knopp's avatar
 
Raymond Knopp committed
1380
	    sf = ((slot>>1)+1)%10;
Raymond Knopp's avatar
 
Raymond Knopp committed
1381 1382 1383 1384
	    //		    LOG_I(PHY,"[eNB] Multithread slot %d (IC %d)\n",slot,PHY_vars_eNB_g[0][CC_id]->proc[sf].instance_cnt);
	    for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
	      if (pthread_mutex_lock(&PHY_vars_eNB_g[0][CC_id]->proc[sf].mutex_tx) != 0) {
		LOG_E(PHY,"[eNB] ERROR pthread_mutex_lock for eNB TX thread %d (IC %d)\n",sf,PHY_vars_eNB_g[0][CC_id]->proc[sf].instance_cnt_tx);   
Raymond Knopp's avatar
 
Raymond Knopp committed
1385 1386
	      }
	      else {
Raymond Knopp's avatar
 
Raymond Knopp committed
1387 1388 1389 1390 1391 1392 1393 1394 1395 1396 1397
		//		      LOG_I(PHY,"[eNB] Waking up eNB process %d (IC %d)\n",sf,PHY_vars_eNB_g[0][CC_id]->proc[sf].instance_cnt); 
		PHY_vars_eNB_g[0][CC_id]->proc[sf].instance_cnt_tx++;
		pthread_mutex_unlock(&PHY_vars_eNB_g[0][CC_id]->proc[sf].mutex_tx);
		if (PHY_vars_eNB_g[0][CC_id]->proc[sf].instance_cnt_tx == 0) {
		  if (pthread_cond_signal(&PHY_vars_eNB_g[0][CC_id]->proc[sf].cond_tx) != 0) {
		    LOG_E(PHY,"[eNB] ERROR pthread_cond_signal for eNB TX thread %d\n",sf);
		  }
		}
		else {
		  LOG_W(PHY,"[eNB] Frame %d, eNB TX thread %d busy!!\n",PHY_vars_eNB_g[0][CC_id]->proc[sf].frame_tx,sf);
		  oai_exit=1;
Raymond Knopp's avatar
 
Raymond Knopp committed
1398 1399
		}
	      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1400 1401 1402 1403
	      
	      if (pthread_mutex_lock(&PHY_vars_eNB_g[0][CC_id]->proc[sf].mutex_rx) != 0) {
		LOG_E(PHY,"[eNB] ERROR pthread_mutex_lock for eNB RX thread %d (IC %d)\n",sf,PHY_vars_eNB_g[0][CC_id]->proc[sf].instance_cnt_rx);   
	      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1404
	      else {
Raymond Knopp's avatar
 
Raymond Knopp committed
1405 1406 1407 1408 1409 1410 1411 1412 1413 1414 1415 1416
		//		      LOG_I(PHY,"[eNB] Waking up eNB process %d (IC %d)\n",sf,PHY_vars_eNB_g[0][CC_id]->proc[sf].instance_cnt); 
		PHY_vars_eNB_g[0][CC_id]->proc[sf].instance_cnt_rx++;
		pthread_mutex_unlock(&PHY_vars_eNB_g[0][CC_id]->proc[sf].mutex_rx);
		if (PHY_vars_eNB_g[0][CC_id]->proc[sf].instance_cnt_rx == 0) {
		  if (pthread_cond_signal(&PHY_vars_eNB_g[0][CC_id]->proc[sf].cond_rx) != 0) {
		    LOG_E(PHY,"[eNB] ERROR pthread_cond_signal for eNB RX thread %d\n",sf);
		  }
		}
		else {
		  LOG_W(PHY,"[eNB] Frame %d, eNB RX thread %d busy!!\n",PHY_vars_eNB_g[0][CC_id]->proc[sf].frame_rx,sf);
		  oai_exit=1;
		}
Raymond Knopp's avatar
 
Raymond Knopp committed
1417
	      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1418
	      
Raymond Knopp's avatar
 
Raymond Knopp committed
1419
	    }
Raymond Knopp's avatar
 
Raymond Knopp committed
1420 1421 1422
	  }
	}
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1423 1424 1425 1426 1427 1428 1429
#ifndef RTAI
      //pthread_mutex_lock(&tti_mutex);
#endif




Raymond Knopp's avatar
 
Raymond Knopp committed
1430
#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
1431
      slot++;
Raymond Knopp's avatar
 
Raymond Knopp committed
1432 1433 1434 1435
      if (slot == 20) {
	frame++;
	slot = 0;
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1436
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
1437 1438 1439 1440
      hw_subframe++;
      slot+=2;
      if(hw_subframe==10) {
        hw_subframe = 0;
Raymond Knopp's avatar
 
Raymond Knopp committed
1441
	frame++;
Raymond Knopp's avatar
 
Raymond Knopp committed
1442
	slot = 1;
Raymond Knopp's avatar
 
Raymond Knopp committed
1443
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1444 1445 1446
 #endif     


1447
#if defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
1448
      itti_update_lte_time(frame, slot);
1449
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1450 1451 1452 1453 1454 1455
    }
  }
#ifdef DEBUG_THREADS
  printf("eNB_thread: finished, ran %d times.\n",frame);
#endif
  
1456
#ifdef HARD_RT
Raymond Knopp's avatar
 
Raymond Knopp committed
1457
  rt_make_soft_real_time();
1458 1459
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
1460 1461 1462 1463 1464

#ifdef DEBUG_THREADS
  printf("Exiting eNB_thread ...");
#endif
  // clean task
1465
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
1466 1467 1468 1469
  rt_task_delete(task);
#else
  eNB_thread_status = 0;
  pthread_exit(&eNB_thread_status);
1470
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1471 1472 1473 1474 1475
#ifdef DEBUG_THREADS
  printf("eNB_thread deleted. returning\n");
#endif
  return 0;
}
1476

Raymond Knopp's avatar
 
Raymond Knopp committed
1477 1478


Raymond Knopp's avatar
 
Raymond Knopp committed
1479
#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
1480 1481
/* This is the main UE thread. Initially it is doing a periodic get_frame. One synchronized it gets woken up by the kernel driver using the RTAI message mechanism (rt_send and rt_receive). */
static void *UE_thread(void *arg) {
1482
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
1483 1484 1485 1486 1487 1488 1489 1490 1491 1492 1493 1494 1495 1496 1497 1498
  RT_TASK *task;
#endif
  // RTIME in, out, diff;
  int slot=0,frame=0,hw_slot,last_slot, next_slot;
  // unsigned int aa;
  static int is_synchronized = 0;
  int delay_cnt;
  RTIME time_in;
  int hw_slot_offset=0,rx_offset_mbox=0,mbox_target=0,mbox_current=0;
  int diff2;
  int i, ret;
  volatile unsigned int *DAQ_MBOX = openair0_daq_cnt();
#ifndef USRP
  exmimo_config_t *p_exmimo_config = openair0_exmimo_pci[card].exmimo_config_ptr;;
#endif

1499

1500
#if defined(ENABLE_ITTI) && defined(ENABLE_USE_MME)
Raymond Knopp's avatar
 
Raymond Knopp committed
1501 1502
  /* Wait for NAS UE to start cell selection */
  wait_system_ready ("Waiting for UE to be activated by UserProcess %s\r", &start_UE);
1503 1504
#endif

1505
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
1506 1507
  task = rt_task_init_schmod(nam2num("TASK0"), 0, 0, 0, SCHED_FIFO, 0xF);
  LOG_D(HW,"Started UE thread (id %p)\n",task);
1508 1509 1510
#endif

#ifdef HARD_RT
Raymond Knopp's avatar
 
Raymond Knopp committed
1511
  rt_make_hard_real_time();
1512 1513
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
1514
  mlockall(MCL_CURRENT | MCL_FUTURE);
1515

Raymond Knopp's avatar
 
Raymond Knopp committed
1516 1517 1518 1519 1520 1521 1522 1523 1524 1525 1526 1527 1528
  openair_daq_vars.freq_offset = 0; //-7500;
  /*
    if (mode == rx_calib_ue) {
    openair_daq_vars.freq_offset = -7500;
    for (i=0; i<4; i++) {
    p_exmimo_config->rf.rf_freq_rx[i] = p_exmimo_config->rf.rf_freq_rx[i]+openair_daq_vars.freq_offset;
    p_exmimo_config->rf.rf_freq_tx[i] = p_exmimo_config->rf.rf_freq_rx[i]+openair_daq_vars.freq_offset;
    }
    openair0_dump_config(card);
    }
  */
  while (!oai_exit)  {
    hw_slot = (((((volatile unsigned int *)DAQ_MBOX)[0]+1)%150)<<1)/15; //the slot the hw is about to store
1529
      
Raymond Knopp's avatar
 
Raymond Knopp committed
1530 1531
    if (is_synchronized) {
      //this is the mbox counter that indicates the start of the frame
Raymond Knopp's avatar
 
Raymond Knopp committed
1532
      rx_offset_mbox = (PHY_vars_UE_g[0][0]->rx_offset * 150) / (10*PHY_vars_UE_g[0][0]->lte_frame_parms.samples_per_tti);
Raymond Knopp's avatar
 
Raymond Knopp committed
1533 1534 1535 1536 1537 1538 1539 1540 1541 1542 1543 1544 1545
      //this is the mbox counter where we should be
      mbox_target = (((((slot+1)%20)*15+1)>>1) + rx_offset_mbox + 1)%150;
      // round up to the next multiple of two (mbox counter from express MIMO gives only even numbers)
      mbox_target = ((mbox_target+1)-((mbox_target-1)%2))%150;
      //this is the mbox counter where we are
      mbox_current = ((volatile unsigned int *)DAQ_MBOX)[0];
      //this is the time we need to sleep in order to synchronize with the hw (in multiples of DAQ_PERIOD)
      if ((mbox_current>=120) && (mbox_target<30)) //handle the frame wrap-arround
	diff2 = 150-mbox_current+mbox_target;
      else if ((mbox_current<30) && (mbox_target>=120))
	diff2 = -150+mbox_target-mbox_current;
      else
	diff2 = mbox_target - mbox_current;
1546
	
Raymond Knopp's avatar
 
Raymond Knopp committed
1547 1548 1549 1550 1551 1552 1553 1554
      if (diff2 <(-7)) {
	LOG_D(HW,"UE Frame %d: missed slot, proceeding with next one (slot %d, hw_slot %d, diff %d)\n",frame, slot, hw_slot, diff2);
	if (frame>0)
	  exit_fun("[HW][UE] missed slot");
	slot++;
	if (slot==20) {
	  slot=0;
	  frame++;
Raymond Knopp's avatar
 
Raymond Knopp committed
1555
	}
Raymond Knopp's avatar
 
Raymond Knopp committed
1556 1557 1558 1559 1560 1561 1562 1563 1564 1565 1566 1567 1568 1569 1570 1571 1572 1573 1574 1575 1576 1577 1578 1579
	continue;
      }
      if (diff2>8)
	LOG_D(HW,"UE Frame %d: skipped slot, waiting for hw to catch up (slot %d, hw_slot %d, mbox_current %d, mbox_target %d, diff %d)\n",frame, slot, hw_slot, mbox_current, mbox_target, diff2);
	
      /*
	if (frame%100==0)
	LOG_D(HW,"frame %d (%d), slot %d, hw_slot %d, rx_offset_mbox %d, mbox_target %d, mbox_current %d, diff %d\n",frame, PHY_vars_UE_g[0]->frame, slot,hw_slot,rx_offset_mbox,mbox_target,mbox_current,diff2);
      */
	
      vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_DAQ_MBOX, *DAQ_MBOX);
      vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_DIFF, diff2);
	
      delay_cnt = 0;
      while ((diff2>0) && (!oai_exit) && (is_synchronized) )  {
	time_in = rt_get_time_ns();
	//LOG_D(HW,"eNB Frame %d delaycnt %d : hw_slot %d (%d), slot %d (%d), diff %d, time %llu\n",frame,delay_cnt,hw_slot,((volatile unsigned int *)DAQ_MBOX)[0],slot,mbox_target,diff2,time_in);
	vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_DAQ_MBOX, *DAQ_MBOX);
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_RT_SLEEP,1);
	ret = rt_sleep_ns(diff2*DAQ_PERIOD);
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_RT_SLEEP,0);
	vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_DAQ_MBOX, *DAQ_MBOX);
	if (ret)
	  LOG_D(HW,"eNB Frame %d, time %llu: rt_sleep_ns returned %d\n",frame, time_in);
1580
	  
Raymond Knopp's avatar
 
Raymond Knopp committed
1581 1582 1583 1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595 1596 1597 1598 1599 1600 1601 1602 1603 1604 1605 1606 1607
	hw_slot = (((((volatile unsigned int *)DAQ_MBOX)[0]+1)%150)<<1)/15;
	//LOG_D(HW,"eNB Frame %d : hw_slot %d, time %llu\n",frame,hw_slot,rt_get_time_ns());
	delay_cnt++;
	if (delay_cnt == 30)  {
	  LOG_D(HW,"UE frame %d: HW stopped ... \n",frame);
	  exit_fun("[HW][UE] HW stopped");
	}
	mbox_current = ((volatile unsigned int *)DAQ_MBOX)[0];
	if ((mbox_current>=135) && (mbox_target<15)) //handle the frame wrap-arround
	  diff2 = 150-mbox_current+mbox_target;
	else if ((mbox_current<15) && (mbox_target>=135))
	  diff2 = -150+mbox_target-mbox_current;
	else
	  diff2 = mbox_target - mbox_current;
	  
	vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_DAQ_MBOX, *DAQ_MBOX);
	vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_DIFF, diff2);
      }
	
    }
      
    last_slot = (slot)%LTE_SLOTS_PER_FRAME;
    if (last_slot <0)
      last_slot+=LTE_SLOTS_PER_FRAME;
    next_slot = (slot+3)%LTE_SLOTS_PER_FRAME;
      
    if (is_synchronized)  {
Raymond Knopp's avatar
 
Raymond Knopp committed
1608
      phy_procedures_UE_lte (last_slot, next_slot, PHY_vars_UE_g[0][0], 0, 0,mode,0,NULL);
Raymond Knopp's avatar
 
Raymond Knopp committed
1609 1610 1611 1612 1613 1614 1615 1616 1617 1618 1619
	
    }
    else {  // we are not yet synchronized
      hw_slot_offset = 0;
	
      slot = 0;
      openair0_get_frame(card);
      //          LOG_D(HW,"after get_frame\n");
      //          rt_sleep_ns(FRAME_PERIOD);
      //          LOG_D(HW,"after sleep\n");
	
Raymond Knopp's avatar
 
Raymond Knopp committed
1620
      if (initial_sync(PHY_vars_UE_g[0][0],mode)==0) {
Raymond Knopp's avatar
 
Raymond Knopp committed
1621
	/*
Raymond Knopp's avatar
 
Raymond Knopp committed
1622 1623 1624 1625 1626
	  lte_adjust_synch(&PHY_vars_UE_g[0]->lte_frame_parms,
	  PHY_vars_UE_g[0],
	  0,
	  1,
	  16384);
Raymond Knopp's avatar
 
Raymond Knopp committed
1627
	*/
Raymond Knopp's avatar
 
Raymond Knopp committed
1628 1629 1630 1631 1632 1633 1634 1635
	//for better visualization afterwards
	/*
	  for (aa=0; aa<PHY_vars_UE_g[0]->lte_frame_parms.nb_antennas_rx; aa++)
	  memset(PHY_vars_UE_g[0]->lte_ue_common_vars.rxdata[aa],0,
	  PHY_vars_UE_g[0]->lte_frame_parms.samples_per_tti*LTE_NUMBER_OF_SUBFRAMES_PER_FRAME*sizeof(int));
	*/
	if (mode == rx_calib_ue) {
	  exit_fun("[HW][UE] UE in RX calibration mode");
Raymond Knopp's avatar
 
Raymond Knopp committed
1636
	}
Raymond Knopp's avatar
 
Raymond Knopp committed
1637 1638 1639 1640 1641 1642
	else {
	  is_synchronized = 1;
	  //start the DMA transfers
	  //LOG_D(HW,"Before openair0_start_rt_acquisition \n");
	  openair0_start_rt_acquisition(card);
	    
Raymond Knopp's avatar
 
Raymond Knopp committed
1643
	  hw_slot_offset = (PHY_vars_UE_g[0][0]->rx_offset<<1) / PHY_vars_UE_g[0][0]->lte_frame_parms.samples_per_tti;
Raymond Knopp's avatar
 
Raymond Knopp committed
1644 1645 1646 1647 1648 1649 1650 1651 1652 1653 1654 1655 1656 1657 1658 1659 1660 1661 1662 1663 1664 1665 1666 1667
	  LOG_D(HW,"Got synch: hw_slot_offset %d\n",hw_slot_offset);
	}
      }
      else {
	if (openair_daq_vars.freq_offset >= 0) {
	  openair_daq_vars.freq_offset += 100;
	  openair_daq_vars.freq_offset *= -1;
	}
	else {
	  openair_daq_vars.freq_offset *= -1;
	}
	if (abs(openair_daq_vars.freq_offset) > 7500) {
	  LOG_I(PHY,"[initial_sync] No cell synchronization found, abondoning\n");
	  mac_xface->macphy_exit("No cell synchronization found, abondoning");
	}
	else {
	  LOG_I(PHY,"[initial_sync] trying carrier off %d Hz\n",openair_daq_vars.freq_offset);
#ifndef USRP
	  for (i=0; i<4; i++) {
	    if (p_exmimo_config->rf.rf_freq_rx[i])
	      p_exmimo_config->rf.rf_freq_rx[i] = carrier_freq[i]+openair_daq_vars.freq_offset;
	    if (p_exmimo_config->rf.rf_freq_tx[i])
	      p_exmimo_config->rf.rf_freq_tx[i] = carrier_freq[i]+openair_daq_vars.freq_offset;
	  }
1668
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1669 1670 1671
	  openair0_dump_config(card);
	  rt_sleep_ns(FRAME_PERIOD);
	}
Raymond Knopp's avatar
 
Raymond Knopp committed
1672
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1673 1674 1675 1676 1677 1678 1679 1680 1681 1682 1683 1684 1685 1686 1687 1688
    }
      
    /*
      if ((slot%2000)<10)
      LOG_D(HW,"fun0: doing very hard work\n");
    */
    slot++;
    if (slot==20) {
      slot=0;
      frame++;
    }
#if defined(ENABLE_ITTI)
    itti_update_lte_time(frame, slot);
#endif
  }
  LOG_D(HW,"UE_thread: finished, ran %d times.\n",frame);
1689 1690

#ifdef HARD_RT
Raymond Knopp's avatar
 
Raymond Knopp committed
1691
  rt_make_soft_real_time();
1692 1693
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
1694
  // clean task
1695
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
1696
  rt_task_delete(task);
1697
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1698 1699 1700
  LOG_D(HW,"Task deleted. returning\n");
  return 0;
}
1701

Raymond Knopp's avatar
 
Raymond Knopp committed
1702
#else  // This is for USRP or ETHERNET targets
1703

1704
#endif
1705

1706

Raymond Knopp's avatar
 
Raymond Knopp committed
1707 1708
static void get_options (int argc, char **argv) {
  int                           c;
Raymond Knopp's avatar
 
Raymond Knopp committed
1709 1710
  //  char                          line[1000];
  //  int                           l;
1711
  int i,j,k;
Raymond Knopp's avatar
 
Raymond Knopp committed
1712 1713 1714
#ifdef USRP
  int clock_src;
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1715 1716 1717 1718 1719 1720 1721 1722 1723 1724 1725 1726 1727 1728 1729 1730 1731 1732 1733 1734 1735 1736 1737 1738
  const Enb_properties_array_t *enb_properties;
  
  enum long_option_e {
    LONG_OPTION_START = 0x100, /* Start after regular single char options */
    
    LONG_OPTION_ULSCH_MAX_CONSECUTIVE_ERRORS,
    LONG_OPTION_CALIB_UE_RX,
    LONG_OPTION_CALIB_UE_RX_MED,
    LONG_OPTION_CALIB_UE_RX_BYP,
    
    LONG_OPTION_DEBUG_UE_PRACH,
    
    LONG_OPTION_NO_L2_CONNECT,
  };
  
  static const struct option long_options[] = {
    {"ulsch-max-errors",required_argument,  NULL, LONG_OPTION_ULSCH_MAX_CONSECUTIVE_ERRORS},
    {"calib-ue-rx",     required_argument,  NULL, LONG_OPTION_CALIB_UE_RX},
    {"calib-ue-rx-med", required_argument,  NULL, LONG_OPTION_CALIB_UE_RX_MED},
    {"calib-ue-rx-byp", required_argument,  NULL, LONG_OPTION_CALIB_UE_RX_BYP},
    {"debug-ue-prach",  no_argument,        NULL, LONG_OPTION_DEBUG_UE_PRACH},
    {"no-L2-connect",   no_argument,        NULL, LONG_OPTION_NO_L2_CONNECT},
    {NULL, 0, NULL, 0}};
  
1739
  while ((c = getopt_long (argc, argv, "C:dK:g:G:qO:SUVRMr:s:",long_options,NULL)) != -1) {
Raymond Knopp's avatar
 
Raymond Knopp committed
1740 1741 1742 1743 1744 1745 1746 1747 1748 1749 1750 1751 1752 1753 1754 1755 1756 1757 1758 1759 1760 1761 1762 1763 1764 1765 1766 1767 1768 1769 1770 1771
    switch (c) {
    case LONG_OPTION_ULSCH_MAX_CONSECUTIVE_ERRORS:
      ULSCH_max_consecutive_errors = atoi(optarg);
      printf("Set ULSCH_max_consecutive_errors = %d\n",ULSCH_max_consecutive_errors);
      break;
      
    case LONG_OPTION_CALIB_UE_RX:
      mode = rx_calib_ue;
      rx_input_level_dBm = atoi(optarg);
      printf("Running with UE calibration on (LNA max), input level %d dBm\n",rx_input_level_dBm);
      break;
      
    case LONG_OPTION_CALIB_UE_RX_MED:
      mode = rx_calib_ue_med;
      rx_input_level_dBm = atoi(optarg);
      printf("Running with UE calibration on (LNA med), input level %d dBm\n",rx_input_level_dBm);
      break;
      
    case LONG_OPTION_CALIB_UE_RX_BYP:
      mode = rx_calib_ue_byp;
      rx_input_level_dBm = atoi(optarg);
      printf("Running with UE calibration on (LNA byp), input level %d dBm\n",rx_input_level_dBm);
      break;
      
    case LONG_OPTION_DEBUG_UE_PRACH:
      mode = debug_prach;
      break;
      
    case LONG_OPTION_NO_L2_CONNECT:
      mode = no_L2_connect;
      break;
    case 'M':
Raymond Knopp's avatar
 
Raymond Knopp committed
1772
      multi_thread=0;
Raymond Knopp's avatar
 
Raymond Knopp committed
1773 1774 1775 1776 1777 1778 1779 1780 1781 1782 1783 1784 1785 1786 1787 1788 1789 1790 1791 1792 1793 1794 1795 1796 1797 1798 1799 1800 1801 1802 1803 1804 1805 1806 1807 1808 1809 1810 1811 1812 1813 1814 1815 1816 1817 1818 1819 1820 1821 1822 1823 1824 1825 1826 1827 1828 1829 1830 1831 1832 1833
      break;
    case 'C':
      downlink_frequency[0] = atof(optarg); // Use float to avoid issue with frequency over 2^31.
      downlink_frequency[1] = downlink_frequency[0];
      downlink_frequency[2] = downlink_frequency[0];
      downlink_frequency[3] = downlink_frequency[0];
      carrier_freq[0] = downlink_frequency[0];
      carrier_freq[1] = downlink_frequency[1];
      carrier_freq[2] = downlink_frequency[2];
      carrier_freq[3] = downlink_frequency[3];
      printf("Downlink frequency set to %u\n", downlink_frequency[0]);
      break;
      
    case 'd':
#ifdef XFORMS
      do_forms=1;
#endif
      break;
      
    case 'K':
#if defined(ENABLE_ITTI)
      itti_dump_file = strdup(optarg);
#else
      printf("-K option is disabled when ENABLE_ITTI is not defined\n");
#endif
      break;
      
    case 'O':
      conf_config_file_name = optarg;
      break;
      
    case 'U':
      UE_flag = 1;
      break;
      
    case 'V':
      ouput_vcd = 1;
      break;
    case  'q': 
      opp_enabled = 1;
      break;
    case  'R' :
      online_log_messages =1;
      break;
    case 'r':
      switch(atoi(optarg)) {
      case 6:
	N_RB_DL=6;
	break;
      case 25:
	N_RB_DL=25;
	break;
      case 50:
	N_RB_DL=50;
	break;
      case 100:
	N_RB_DL=100;
	break;
      default:
	printf("Unknown N_RB_DL %d, switching to 25\n",atoi(optarg));
	break;
Raymond Knopp's avatar
 
Raymond Knopp committed
1834
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1835 1836
    case 's':
#ifdef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
1837 1838 1839 1840 1841 1842 1843 1844 1845 1846

      clock_src = atoi(optarg);
      if (clock_src == 0) {
	char ref[128] = "internal";
	//strncpy(uhd_ref, ref, strlen(ref)+1);
      }
      else if (clock_src == 1) {
	char ref[128] = "external";
	//strncpy(uhd_ref, ref, strlen(ref)+1);
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1847 1848 1849 1850
#else
      printf("Note: -s not defined for ExpressMIMO2\n");
#endif
      break;
1851 1852 1853 1854 1855 1856
    case 'g':
      glog_level=atoi(optarg); // value between 1 - 9 
      break;
    case 'G':
      glog_verbosity=atoi(optarg);// value from 0, 0x5, 0x15, 0x35, 0x75
      break;
Raymond Knopp's avatar
 
Raymond Knopp committed
1857 1858
    default:
      break;
1859 1860
    }
  }
Raymond Knopp's avatar
 
Raymond Knopp committed
1861 1862 1863 1864 1865 1866 1867 1868 1869 1870 1871 1872 1873 1874
  
  if ((UE_flag == 0) && (conf_config_file_name != NULL)) {
    int i;
    
    NB_eNB_INST = 1;
    
    /* Read eNB configuration file */
    enb_properties = enb_config_init(conf_config_file_name);
    
    AssertFatal (NB_eNB_INST <= enb_properties->number,
		 "Number of eNB is greater than eNB defined in configuration file %s (%d/%d)!",
		 conf_config_file_name, NB_eNB_INST, enb_properties->number);
    
    /* Update some simulation parameters */
1875 1876 1877 1878 1879 1880 1881
    for (i=0; i < enb_properties->number; i++) {
      
      frame_parms->frame_type =       enb_properties->properties[i]->frame_type;
      frame_parms->tdd_config =       enb_properties->properties[i]->tdd_config;
      frame_parms->tdd_config_S =     enb_properties->properties[i]->tdd_config_s;

      for (j=0; j < enb_properties->properties[i]->nb_cc; j++ ){ 
1882 1883
	frame_parms->Nid_cell          =  enb_properties->properties[i]->cell_id[j];
	frame_parms->N_RB_DL          =  enb_properties->properties[i]->N_RB_DL[j];
1884 1885 1886 1887 1888 1889 1890 1891 1892 1893 1894 1895 1896 1897 1898 1899 1900 1901 1902 1903 1904 1905 1906 1907 1908 1909
      } // j
    
      glog_level                     = enb_properties->properties[i]->glog_level;
      glog_verbosity                 = enb_properties->properties[i]->glog_verbosity;
      hw_log_level                   = enb_properties->properties[i]->hw_log_level;
      hw_log_verbosity               = enb_properties->properties[i]->hw_log_verbosity ;
      phy_log_level                  = enb_properties->properties[i]->phy_log_level;
      phy_log_verbosity              = enb_properties->properties[i]->phy_log_verbosity;
      mac_log_level                  = enb_properties->properties[i]->mac_log_level;
      mac_log_verbosity              = enb_properties->properties[i]->mac_log_verbosity;
      rlc_log_level                  = enb_properties->properties[i]->rlc_log_level;
      rlc_log_verbosity              = enb_properties->properties[i]->rlc_log_verbosity;
      pdcp_log_level                 = enb_properties->properties[i]->pdcp_log_level;
      pdcp_log_verbosity             = enb_properties->properties[i]->pdcp_log_verbosity;
      rrc_log_level                  = enb_properties->properties[i]->rrc_log_level;
      rrc_log_verbosity              = enb_properties->properties[i]->rrc_log_verbosity;
    
    
    // adjust the log 

      for (k = 0 ; k < (sizeof(downlink_frequency) / sizeof (downlink_frequency[0])); k++) {
	downlink_frequency[k] =       enb_properties->properties[i]->downlink_frequency;
	printf("Downlink frequency set to %u\n", downlink_frequency[k]);
	uplink_frequency_offset[k] =  enb_properties->properties[i]->uplink_frequency_offset;
      } // k 
    }// i
Raymond Knopp's avatar
 
Raymond Knopp committed
1910 1911
  }
}
1912

Raymond Knopp's avatar
 
Raymond Knopp committed
1913
int main(int argc, char **argv) {
1914
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
1915
  // RT_TASK *task;
Raymond Knopp's avatar
 
Raymond Knopp committed
1916
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
1917
  int *eNB_thread_status_p;
Raymond Knopp's avatar
 
Raymond Knopp committed
1918
  //  int *eNB_thread_status_rx[10],*eNB_thread_status_tx[10];
1919
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1920
  int i,j,aa;
1921
#if defined (XFORMS) || defined (EMOS) || (! defined (RTAI))
Raymond Knopp's avatar
 
Raymond Knopp committed
1922
  void *status;
1923
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1924
  
Raymond Knopp's avatar
 
Raymond Knopp committed
1925
  int CC_id;
Raymond Knopp's avatar
 
Raymond Knopp committed
1926 1927
  uint16_t Nid_cell = 0;
  uint8_t  cooperation_flag=0, transmission_mode=1, abstraction_flag=0;
1928
#ifndef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
1929
  uint8_t beta_ACK=0,beta_RI=0,beta_CQI=2;
1930 1931 1932
#endif

#ifdef ENABLE_TCXO
Raymond Knopp's avatar
 
Raymond Knopp committed
1933
  unsigned int tcxo = 114;
1934 1935
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
1936
  //  int amp;
Raymond Knopp's avatar
 
Raymond Knopp committed
1937 1938
  // uint8_t prach_fmt;
  // int N_ZC;
1939

Raymond Knopp's avatar
 
Raymond Knopp committed
1940
  //  int ret, ant;
Raymond Knopp's avatar
 
Raymond Knopp committed
1941
  int ant_offset=0;
1942 1943 1944
#ifdef XFORMS
  int ret;
#endif
1945
#if defined (EMOS) || (! defined (RTAI))
Raymond Knopp's avatar
 
Raymond Knopp committed
1946
  int error_code;
1947
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1948
  openair0_config_t openair0_cfg;
1949

Raymond Knopp's avatar
 
Raymond Knopp committed
1950
  set_latency_target();
1951

Raymond Knopp's avatar
 
Raymond Knopp committed
1952
  mode = normal_txrx;
1953

Raymond Knopp's avatar
 
Raymond Knopp committed
1954 1955 1956 1957 1958
  frame_parms = (LTE_DL_FRAME_PARMS*) malloc(sizeof(LTE_DL_FRAME_PARMS));
  /* Set some default values that may be overwritten while reading options */
  frame_parms->frame_type         = TDD; /* TDD */
  frame_parms->tdd_config         = 3;
  frame_parms->tdd_config_S       = 0;
1959

Raymond Knopp's avatar
 
Raymond Knopp committed
1960
  get_options (argc, argv); //Command-line options
winckel's avatar
winckel committed
1961

Raymond Knopp's avatar
 
Raymond Knopp committed
1962 1963
  //randominit (0);
  set_taus_seed (0);
winckel's avatar
winckel committed
1964

Raymond Knopp's avatar
 
Raymond Knopp committed
1965 1966 1967
  // initialize the log (see log.h for details)
  logInit();

1968
  set_glog(glog_level, glog_verbosity);
Raymond Knopp's avatar
 
Raymond Knopp committed
1969 1970 1971
  if (UE_flag==1)
    {
      printf("configuring for UE\n");
1972

Raymond Knopp's avatar
 
Raymond Knopp committed
1973
      set_comp_log(HW,      LOG_INFO,  LOG_HIGH, 1);
1974
#ifdef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
1975
      set_comp_log(PHY,     LOG_INFO,   LOG_HIGH, 1);
1976
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
1977
      set_comp_log(PHY,     LOG_INFO,   LOG_HIGH, 1);
1978
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1979 1980 1981 1982 1983
      set_comp_log(MAC,     LOG_INFO,   LOG_HIGH, 1);
      set_comp_log(RLC,     LOG_INFO,   LOG_HIGH, 1);
      set_comp_log(PDCP,    LOG_INFO,   LOG_HIGH, 1);
      set_comp_log(OTG,     LOG_INFO,   LOG_HIGH, 1);
      set_comp_log(RRC,     LOG_INFO,   LOG_HIGH, 1);
1984
#if defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
1985
      set_comp_log(EMU,     LOG_INFO,   LOG_MED, 1);
1986
# if defined(ENABLE_USE_MME)
Raymond Knopp's avatar
 
Raymond Knopp committed
1987
      set_comp_log(NAS,     LOG_INFO,   LOG_HIGH, 1);
1988 1989
# endif
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1990 1991 1992 1993
    }
  else
    {
      printf("configuring for eNB\n");
1994

1995
      set_comp_log(HW,      hw_log_level, hw_log_verbosity, 1);
1996
#ifdef OPENAIR2
1997
      set_comp_log(PHY,     phy_log_level,   phy_log_verbosity, 1);
1998
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
1999
      set_comp_log(PHY,     LOG_INFO,   LOG_HIGH, 1);
2000
#endif
2001 2002 2003 2004
      set_comp_log(MAC,     mac_log_level,  mac_log_verbosity, 1);
      set_comp_log(RLC,     rlc_log_level,   rlc_log_verbosity, 1);
      set_comp_log(PDCP,    pdcp_log_level,  pdcp_log_verbosity, 1);
      set_comp_log(RRC,     rrc_log_level,  rrc_log_verbosity, 1);
2005
#if defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
2006
      set_comp_log(EMU,     LOG_INFO,   LOG_MED, 1);
2007
# if defined(ENABLE_USE_MME)
Raymond Knopp's avatar
 
Raymond Knopp committed
2008 2009 2010 2011
      set_comp_log(UDP_,    LOG_DEBUG,   LOG_HIGH, 1);
      set_comp_log(GTPU,    LOG_DEBUG,   LOG_HIGH, 1);
      set_comp_log(S1AP,    LOG_DEBUG,   LOG_HIGH, 1);
      set_comp_log(SCTP,    LOG_INFO,   LOG_HIGH, 1);
2012
# endif
2013
#if defined(ENABLE_SECURITY)
Raymond Knopp's avatar
 
Raymond Knopp committed
2014
      set_comp_log(OSA,    LOG_DEBUG,   LOG_HIGH, 1);
2015
#endif
2016
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2017
      set_comp_log(ENB_APP, LOG_INFO, LOG_HIGH, 1);
2018
      set_comp_log(OTG,     LOG_INFO,   LOG_HIGH, 1);
Raymond Knopp's avatar
 
Raymond Knopp committed
2019 2020 2021
      if (online_log_messages == 1) { 
	set_component_filelog(RRC);
	set_component_filelog(PDCP);
Raymond Knopp's avatar
 
Raymond Knopp committed
2022 2023
      }
    }
2024

Raymond Knopp's avatar
 
Raymond Knopp committed
2025 2026 2027 2028 2029 2030 2031
  if (ouput_vcd) {
    if (UE_flag==1)
      vcd_signal_dumper_init("/tmp/openair_dump_UE.vcd");
    else
      vcd_signal_dumper_init("/tmp/openair_dump_eNB.vcd");
  }

2032
#if defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
2033 2034 2035 2036 2037 2038
  if (UE_flag == 1) {
    log_set_instance_type (LOG_INSTANCE_UE);
  }
  else {
    log_set_instance_type (LOG_INSTANCE_ENB);
  }
2039

Raymond Knopp's avatar
 
Raymond Knopp committed
2040
  itti_init(TASK_MAX, THREAD_MAX, MESSAGES_ID_MAX, tasks_info, messages_info, messages_definition_xml, itti_dump_file);
2041 2042
#endif

2043
#ifdef NAS_NETLINK
Raymond Knopp's avatar
 
Raymond Knopp committed
2044
  netlink_init();
2045 2046
#endif

2047
#if !defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
2048 2049 2050
  // to make a graceful exit when ctrl-c is pressed
  signal(SIGSEGV, signal_handler);
  signal(SIGINT, signal_handler);
2051
#endif
2052 2053

#ifndef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2054 2055 2056 2057 2058 2059 2060 2061 2062 2063 2064 2065 2066 2067
  check_clock();
#endif

  // init the parameters
  frame_parms->N_RB_DL            = N_RB_DL;
  frame_parms->N_RB_UL            = N_RB_DL;
  frame_parms->Ncp                = NORMAL;
  frame_parms->Ncp_UL             = NORMAL;
  frame_parms->Nid_cell           = Nid_cell;
  frame_parms->nushift            = 0;
  if (UE_flag==0)
    {
      switch (transmission_mode) {
      case 1: 
Raymond Knopp's avatar
 
Raymond Knopp committed
2068 2069
	frame_parms->nb_antennas_tx     = 1;
	frame_parms->nb_antennas_rx     = 1;
Raymond Knopp's avatar
 
Raymond Knopp committed
2070 2071 2072 2073 2074 2075 2076 2077 2078 2079
	break;
      case 2:
      case 5:
      case 6:
	frame_parms->nb_antennas_tx     = 2;
	frame_parms->nb_antennas_rx     = 2;
	break;
      default:
	printf("Unsupported transmission mode %d\n",transmission_mode);
	exit(-1);
Raymond Knopp's avatar
 
Raymond Knopp committed
2080
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
2081 2082 2083 2084 2085 2086 2087 2088 2089 2090 2091 2092 2093 2094 2095 2096 2097 2098 2099 2100 2101 2102 2103 2104 2105 2106 2107 2108 2109 2110 2111 2112
    }
  else
    { //UE_flag==1
      frame_parms->nb_antennas_tx     = 1;
      frame_parms->nb_antennas_rx     = 1;
    }
  frame_parms->nb_antennas_tx_eNB = (transmission_mode == 1) ? 1 : 2; //initial value overwritten by initial sync later
  frame_parms->mode1_flag         = (transmission_mode == 1) ? 1 : 0;
  frame_parms->phich_config_common.phich_resource = oneSixth;
  frame_parms->phich_config_common.phich_duration = normal;
  // UL RS Config
  frame_parms->pusch_config_common.ul_ReferenceSignalsPUSCH.cyclicShift = 0;//n_DMRS1 set to 0
  frame_parms->pusch_config_common.ul_ReferenceSignalsPUSCH.groupHoppingEnabled = 0;
  frame_parms->pusch_config_common.ul_ReferenceSignalsPUSCH.sequenceHoppingEnabled = 0;
  frame_parms->pusch_config_common.ul_ReferenceSignalsPUSCH.groupAssignmentPUSCH = 0;
  init_ul_hopping(frame_parms);

  init_frame_parms(frame_parms,1);

  phy_init_top(frame_parms);
  phy_init_lte_top(frame_parms);

  //init prach for openair1 test
  frame_parms->prach_config_common.rootSequenceIndex=22; 
  frame_parms->prach_config_common.prach_ConfigInfo.zeroCorrelationZoneConfig=1;
  frame_parms->prach_config_common.prach_ConfigInfo.prach_ConfigIndex=0; 
  frame_parms->prach_config_common.prach_ConfigInfo.highSpeedFlag=0;
  frame_parms->prach_config_common.prach_ConfigInfo.prach_FreqOffset=0;
  // prach_fmt = get_prach_fmt(frame_parms->prach_config_common.prach_ConfigInfo.prach_ConfigIndex, frame_parms->frame_type);
  // N_ZC = (prach_fmt <4)?839:139;

  if (UE_flag==1) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2113 2114 2115 2116
    PHY_vars_UE_g = malloc(sizeof(PHY_VARS_UE**));
    PHY_vars_UE_g[0] = malloc(sizeof(PHY_VARS_UE*)*MAX_NUM_CCs);
    for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
      PHY_vars_UE_g[CC_id][0] = init_lte_UE(frame_parms, UE_id,abstraction_flag,transmission_mode);
Raymond Knopp's avatar
 
Raymond Knopp committed
2117
      
Raymond Knopp's avatar
 
Raymond Knopp committed
2118 2119 2120 2121 2122 2123 2124 2125 2126 2127
#ifndef OPENAIR2
      for (i=0;i<NUMBER_OF_eNB_MAX;i++) {
	PHY_vars_UE_g[CC_id][0]->pusch_config_dedicated[i].betaOffset_ACK_Index = beta_ACK;
	PHY_vars_UE_g[CC_id][0]->pusch_config_dedicated[i].betaOffset_RI_Index  = beta_RI;
	PHY_vars_UE_g[CC_id][0]->pusch_config_dedicated[i].betaOffset_CQI_Index = beta_CQI;
	
	PHY_vars_UE_g[CC_id][0]->scheduling_request_config[i].sr_PUCCH_ResourceIndex = UE_id;
	PHY_vars_UE_g[CC_id][0]->scheduling_request_config[i].sr_ConfigIndex = 7+(UE_id%3);
	PHY_vars_UE_g[CC_id][0]->scheduling_request_config[i].dsr_TransMax = sr_n4;
      }
2128
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2129 2130 2131 2132 2133 2134
      
      compute_prach_seq(&PHY_vars_UE_g[CC_id][0]->lte_frame_parms.prach_config_common,
			PHY_vars_UE_g[CC_id][0]->lte_frame_parms.frame_type,
			PHY_vars_UE_g[CC_id][0]->X_u);
      
      PHY_vars_UE_g[CC_id][0]->lte_ue_pdcch_vars[0]->crnti = 0x1234;
2135
#ifndef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
2136
      PHY_vars_UE_g[CC_id][0]->lte_ue_pdcch_vars[0]->crnti = 0x1235;
2137
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2138

Raymond Knopp's avatar
 
Raymond Knopp committed
2139
    for (i=0;i<4;i++) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2140 2141 2142
      PHY_vars_UE_g[CC_id][0]->rx_gain_max[i] = rxg_max[i];
      //      PHY_vars_UE_g[CC_id][0]->rx_gain_med[i] = rxg_med[i];
      //      PHY_vars_UE_g[CC_id][0]->rx_gain_byp[i] = rxg_byp[i];
Raymond Knopp's avatar
 
Raymond Knopp committed
2143
    }
2144
    
Raymond Knopp's avatar
 
Raymond Knopp committed
2145 2146 2147
    if ((mode == normal_txrx) || (mode == rx_calib_ue) || (mode == no_L2_connect) || (mode == debug_prach)) {
      for (i=0;i<4;i++)
	openair0_cfg.rxg_mode[i] =  max_gain;
Raymond Knopp's avatar
 
Raymond Knopp committed
2148
      PHY_vars_UE_g[CC_id][0]->rx_total_gain_dB =  PHY_vars_UE_g[CC_id][0]->rx_gain_max[0] + (int)rx_gain - 30; //-30 because it was calibrated with a 30dB gain
Raymond Knopp's avatar
 
Raymond Knopp committed
2149 2150 2151 2152
    }
    else if ((mode == rx_calib_ue_med)) {
      for (i=0;i<4;i++)
	openair0_cfg.rxg_mode[i] =  med_gain;
Raymond Knopp's avatar
 
Raymond Knopp committed
2153
      PHY_vars_UE_g[CC_id][0]->rx_total_gain_dB =  PHY_vars_UE_g[CC_id][0]->rx_gain_med[0]  + (int)rx_gain - 30; //-30 because it was calibrated with a 30dB gain;
Raymond Knopp's avatar
 
Raymond Knopp committed
2154 2155 2156 2157
    }
    else if ((mode == rx_calib_ue_byp)) {
      for (i=0;i<4;i++)
	openair0_cfg.rxg_mode[i] =  byp_gain;
Raymond Knopp's avatar
 
Raymond Knopp committed
2158
      PHY_vars_UE_g[CC_id][0]->rx_total_gain_dB =  PHY_vars_UE_g[CC_id][0]->rx_gain_byp[0]  + (int)rx_gain - 30; //-30 because it was calibrated with a 30dB gain;
Raymond Knopp's avatar
 
Raymond Knopp committed
2159 2160
    }
    
Raymond Knopp's avatar
 
Raymond Knopp committed
2161 2162 2163 2164 2165 2166 2167 2168 2169 2170 2171 2172
    PHY_vars_UE_g[CC_id][0]->tx_power_max_dBm = tx_max_power;
    }
    NB_UE_INST=1;
    NB_INST=1;
    
    openair_daq_vars.manual_timing_advance = 0;
    //openair_daq_vars.timing_advance = TIMING_ADVANCE_HW;
    openair_daq_vars.rx_gain_mode = DAQ_AGC_ON;
    openair_daq_vars.auto_freq_correction = 0;
    openair_daq_vars.use_ia_receiver = 0;
    

Raymond Knopp's avatar
 
Raymond Knopp committed
2173 2174 2175 2176 2177
    
    //  printf("tx_max_power = %d -> amp %d\n",tx_max_power,get_tx_amp(tx_max_power,tx_max_power));
  }
  else
    { //this is eNB
Raymond Knopp's avatar
 
Raymond Knopp committed
2178 2179 2180 2181 2182 2183
      PHY_vars_eNB_g = malloc(sizeof(PHY_VARS_eNB**));
      PHY_vars_eNB_g[0] = malloc(sizeof(PHY_VARS_eNB*));
      for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
	PHY_vars_eNB_g[0][CC_id] = init_lte_eNB(frame_parms,eNB_id,Nid_cell,cooperation_flag,transmission_mode,abstraction_flag);
      
       
2184
#ifndef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
2185 2186 2187 2188 2189 2190 2191 2192 2193
	for (i=0;i<NUMBER_OF_UE_MAX;i++) {
	  PHY_vars_eNB_g[0][CC_id]->pusch_config_dedicated[i].betaOffset_ACK_Index = beta_ACK;
	  PHY_vars_eNB_g[0][CC_id]->pusch_config_dedicated[i].betaOffset_RI_Index  = beta_RI;
	  PHY_vars_eNB_g[0][CC_id]->pusch_config_dedicated[i].betaOffset_CQI_Index = beta_CQI;
	  
	  PHY_vars_eNB_g[0][CC_id]->scheduling_request_config[i].sr_PUCCH_ResourceIndex = i;
	  PHY_vars_eNB_g[0][CC_id]->scheduling_request_config[i].sr_ConfigIndex = 7+(i%3);
	  PHY_vars_eNB_g[0][CC_id]->scheduling_request_config[i].dsr_TransMax = sr_n4;
	}
Raymond Knopp's avatar
 
Raymond Knopp committed
2194
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2195 2196 2197 2198
      
	compute_prach_seq(&PHY_vars_eNB_g[0][CC_id]->lte_frame_parms.prach_config_common,
			  PHY_vars_eNB_g[0][CC_id]->lte_frame_parms.frame_type,
			  PHY_vars_eNB_g[0][CC_id]->X_u);
Raymond Knopp's avatar
 
Raymond Knopp committed
2199

Raymond Knopp's avatar
 
Raymond Knopp committed
2200
	PHY_vars_eNB_g[0][CC_id]->rx_total_gain_eNB_dB =  rxg_max[0] + (int)rx_gain - 30; //was measured at rxgain=30;
Raymond Knopp's avatar
 
Raymond Knopp committed
2201

Raymond Knopp's avatar
 
Raymond Knopp committed
2202
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
2203 2204
      NB_eNB_INST=1;
      NB_INST=1;
Raymond Knopp's avatar
 
Raymond Knopp committed
2205

Raymond Knopp's avatar
 
Raymond Knopp committed
2206
      openair_daq_vars.ue_dl_rb_alloc=0x1fff;
2207
      openair_daq_vars.target_ue_dl_mcs=20;
Raymond Knopp's avatar
 
Raymond Knopp committed
2208 2209
      openair_daq_vars.ue_ul_nb_rb=6;
      openair_daq_vars.target_ue_ul_mcs=6;
Raymond Knopp's avatar
 
Raymond Knopp committed
2210

Raymond Knopp's avatar
 
Raymond Knopp committed
2211 2212 2213
      // set eNB to max gain
      for (i=0;i<4;i++)
	openair0_cfg.rxg_mode[i] =  max_gain;
Raymond Knopp's avatar
 
Raymond Knopp committed
2214
    }
2215

2216

2217

Raymond Knopp's avatar
 
Raymond Knopp committed
2218
  dump_frame_parms(frame_parms);
2219

Raymond Knopp's avatar
 
Raymond Knopp committed
2220 2221 2222 2223 2224 2225 2226 2227 2228
  if(frame_parms->N_RB_DL == 100) {
    sample_rate = 30.72e6;
#ifdef USRP
    samples_per_packets = 2048;
    samples_per_frame = 307200;
    // from usrp_time_offset
    tx_forward_nsamps = 175;
    sf_bounds = sf_bounds_20;
    max_cnt = 150;
Raymond Knopp's avatar
 
Raymond Knopp committed
2229
    tx_delay = 9;
Raymond Knopp's avatar
 
Raymond Knopp committed
2230 2231 2232 2233 2234 2235 2236 2237 2238 2239 2240 2241 2242 2243 2244 2245 2246 2247 2248 2249 2250
#endif
  }
  else if(frame_parms->N_RB_DL == 50){
    sample_rate = 15.36e6;
#ifdef USRP
    samples_per_packets = 2048;
    samples_per_frame = 153600;
    tx_forward_nsamps = 95;
    sf_bounds = sf_bounds_10;
    max_cnt = 75;
    tx_delay = 4;
#endif
  }
  else if (frame_parms->N_RB_DL == 25) {
    sample_rate = 7.68e6;
#ifdef USRP
    samples_per_packets = 1024;
    samples_per_frame = 76800;
    tx_forward_nsamps = 70;
    sf_bounds = sf_bounds_5;
    max_cnt = 75;
Raymond Knopp's avatar
 
Raymond Knopp committed
2251
    tx_delay = 8;
Raymond Knopp's avatar
 
Raymond Knopp committed
2252 2253 2254 2255
#endif
  }
  
  mac_xface = malloc(sizeof(MAC_xface));
2256 2257


Raymond Knopp's avatar
 
Raymond Knopp committed
2258 2259 2260 2261 2262
  openair0_cfg.sample_rate = sample_rate;
  openair0_cfg.tx_num_channels = frame_parms->nb_antennas_tx;
  openair0_cfg.rx_num_channels = frame_parms->nb_antennas_rx;
  
  for (i=0;i<4;i++) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2263
    if (UE_flag==1) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2264 2265
      openair0_cfg.tx_freq[i] = downlink_frequency[i]+uplink_frequency_offset[i];
      openair0_cfg.rx_freq[i] = downlink_frequency[i];
Raymond Knopp's avatar
 
Raymond Knopp committed
2266
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2267 2268 2269 2270 2271 2272 2273 2274 2275 2276 2277
    else {
      openair0_cfg.rx_freq[i] = downlink_frequency[i]+uplink_frequency_offset[i];
      openair0_cfg.tx_freq[i] = downlink_frequency[i];
    }
  }
  openair0_cfg.tx_bw = bw;
  openair0_cfg.rx_bw = bw;
  for (i=0;i<4;i++) {
    openair0_cfg.tx_gain[i] = tx_gain;
    openair0_cfg.rx_gain[i] = rx_gain;
  }
2278
  
Raymond Knopp's avatar
 
Raymond Knopp committed
2279 2280 2281 2282
  if (openair0_device_init(&openair0, &openair0_cfg) <0) {
    printf("Exiting, cannot initialize device\n");
    exit(-1);
  }
2283
#ifdef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
2284
  int eMBMS_active=0;
2285

Raymond Knopp's avatar
 
Raymond Knopp committed
2286 2287 2288 2289 2290 2291 2292
  l2_init(frame_parms,eMBMS_active,
	  0,// cba_group_active
	  0); // HO flag
  if (UE_flag == 1)
    mac_xface->dl_phy_sync_success (0, 0, 0, 1);
  else
    mac_xface->mrbch_phy_sync_failure (0, 0, 0);
2293 2294
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2295
  mac_xface->macphy_exit = &exit_fun;
2296

winckel's avatar
winckel committed
2297
#if defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
2298 2299 2300 2301
  if (create_tasks(UE_flag ? 0 : 1, UE_flag ? 1 : 0) < 0) {
    exit(-1); // need a softer mode
  }
  printf("ITTI tasks created\n");
winckel's avatar
winckel committed
2302 2303
#endif

2304
#ifdef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
2305 2306 2307 2308 2309 2310 2311 2312 2313 2314 2315
  //if (otg_enabled) {
  init_all_otg(0);
  g_otg->seed = 0;
  init_seeds(g_otg->seed);
  g_otg->num_nodes = 2;
  for (i=0; i<g_otg->num_nodes; i++){
    for (j=0; j<g_otg->num_nodes; j++){
      g_otg->application_idx[i][j] = 1;
      //g_otg->packet_gen_type=SUBSTRACT_STRING;
      g_otg->aggregation_level[i][j][0]=1;
      g_otg->application_type[i][j][0] = BCBR; //MCBR, BCBR
2316
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2317 2318 2319
  }
  init_predef_traffic(UE_flag ? 1 : 0, UE_flag ? 0 : 1);
  //  }
2320 2321
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2322
  //  number_of_cards = openair0_num_detected_cards;
2323

Raymond Knopp's avatar
 
Raymond Knopp committed
2324
  openair_daq_vars.timing_advance = 0;
2325

Raymond Knopp's avatar
 
Raymond Knopp committed
2326 2327
  // connect the TX/RX buffers
  if (UE_flag==1) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2328 2329 2330 2331 2332 2333 2334 2335
    for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
      setup_ue_buffers(PHY_vars_UE_g[CC_id][0],frame_parms,ant_offset);
      printf("Setting UE buffer to all-RX\n");
      // Set LSBs for antenna switch (ExpressMIMO)
      for (i=0; i<frame_parms->samples_per_tti*10; i++)
	for (aa=0; aa<frame_parms->nb_antennas_tx; aa++)
	  PHY_vars_UE_g[CC_id][0]->lte_ue_common_vars.txdata[aa][i] = 0x00010001;
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2336 2337 2338
    //p_exmimo_config->framing.tdd_config = TXRXSWITCH_TESTRX;
  }
  else {
Raymond Knopp's avatar
 
Raymond Knopp committed
2339 2340 2341 2342 2343 2344 2345 2346
    for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
      setup_eNB_buffers(PHY_vars_eNB_g[CC_id][0],frame_parms,ant_offset);
      printf("Setting eNB buffer to all-RX\n");
      // Set LSBs for antenna switch (ExpressMIMO)
      for (i=0; i<frame_parms->samples_per_tti*10; i++)
	for (aa=0; aa<frame_parms->nb_antennas_tx; aa++)
	  PHY_vars_eNB_g[CC_id][0]->lte_eNB_common_vars.txdata[0][aa][i] = 0x00010001;
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2347
  }
Raymond Knopp's avatar
 
Raymond Knopp committed
2348
#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
2349
  openair0_dump_config(card);
Raymond Knopp's avatar
 
Raymond Knopp committed
2350 2351
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2352
  /*  
Raymond Knopp's avatar
 
Raymond Knopp committed
2353 2354
      for (ant=0;ant<4;ant++)
      p_exmimo_config->rf.do_autocal[ant] = 0;
Raymond Knopp's avatar
 
Raymond Knopp committed
2355
  */
2356 2357

#ifdef EMOS
Raymond Knopp's avatar
 
Raymond Knopp committed
2358 2359 2360 2361 2362 2363 2364 2365 2366
  error_code = rtf_create(CHANSOUNDER_FIFO_MINOR,CHANSOUNDER_FIFO_SIZE);
  if (error_code==0)
    printf("[OPENAIR][SCHED][INIT] Created EMOS FIFO %d\n",CHANSOUNDER_FIFO_MINOR);
  else if (error_code==ENODEV)
    printf("[OPENAIR][SCHED][INIT] Problem: EMOS FIFO %d is greater than or equal to RTF_NO\n",CHANSOUNDER_FIFO_MINOR);
  else if (error_code==ENOMEM)
    printf("[OPENAIR][SCHED][INIT] Problem: cannot allocate memory for EMOS FIFO %d\n",CHANSOUNDER_FIFO_MINOR);
  else 
    printf("[OPENAIR][SCHED][INIT] Problem creating EMOS FIFO %d, error_code %d\n",CHANSOUNDER_FIFO_MINOR,error_code);
2367 2368
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2369
  mlockall(MCL_CURRENT | MCL_FUTURE);
2370 2371

#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2372 2373 2374 2375 2376 2377 2378 2379 2380 2381 2382 2383 2384 2385 2386 2387 2388 2389 2390 2391 2392
  // make main thread LXRT soft realtime
  /* task = */ rt_task_init_schmod(nam2num("MYTASK"), 9, 0, 0, SCHED_FIFO, 0xF);

  // start realtime timer and scheduler
  //rt_set_oneshot_mode();
  rt_set_periodic_mode();
  start_rt_timer(0);

  //now = rt_get_time() + 10*PERIOD;
  //rt_task_make_periodic(task, now, PERIOD);

  printf("Init mutex\n");
  //mutex = rt_get_adr(nam2num("MUTEX"));
  mutex = rt_sem_init(nam2num("MUTEX"), 1);
  if (mutex==0)
    {
      printf("Error init mutex\n");
      exit(-1);
    }
  else
    printf("mutex=%p\n",mutex);
Raymond Knopp's avatar
 
Raymond Knopp committed
2393 2394 2395 2396 2397 2398 2399 2400 2401 2402
#ifdef USRP
  sync_sem = rt_typed_sem_init(nam2num("syncsem"), 0, BIN_SEM|FIFO_Q);
  if(sync_sem == 0)
    printf("error init sync semphore\n");
#endif
#else
#ifdef USRP
  pthread_cond_init(&sync_cond,NULL);
  pthread_mutex_init(&sync_mutex, NULL);
#endif
2403 2404 2405
#endif


Raymond Knopp's avatar
 
Raymond Knopp committed
2406 2407

  // this starts the DMA transfers
Raymond Knopp's avatar
 
Raymond Knopp committed
2408
#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
2409 2410
  if (UE_flag!=1)
    openair0_start_rt_acquisition(card);
Raymond Knopp's avatar
 
Raymond Knopp committed
2411
#endif
2412 2413

#ifdef XFORMS
Raymond Knopp's avatar
 
Raymond Knopp committed
2414 2415 2416 2417 2418 2419 2420 2421 2422 2423 2424 2425
  if (do_forms==1) {
    fl_initialize (&argc, argv, NULL, 0, 0);
    form_stats = create_form_stats_form();
    if (UE_flag==1) {
      form_ue[UE_id] = create_lte_phy_scope_ue();
      sprintf (title, "LTE DL SCOPE UE");
      fl_show_form (form_ue[UE_id]->lte_phy_scope_ue, FL_PLACE_HOTSPOT, FL_FULLBORDER, title);
    } else {
      for(UE_id=0;UE_id<scope_enb_num_ue;UE_id++) {
	form_enb[UE_id] = create_lte_phy_scope_enb();
	sprintf (title, "UE%d LTE UL SCOPE eNB",UE_id+1);
	fl_show_form (form_enb[UE_id]->lte_phy_scope_enb, FL_PLACE_HOTSPOT, FL_FULLBORDER, title);
2426
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
2427 2428 2429 2430 2431 2432 2433
    }
    fl_show_form (form_stats->stats_form, FL_PLACE_HOTSPOT, FL_FULLBORDER, "stats");
    if (UE_flag==0) {
      for (UE_id=0;UE_id<scope_enb_num_ue;UE_id++) {
	if (otg_enabled) {
	  fl_set_button(form_enb[UE_id]->button_0,1);
	  fl_set_object_label(form_enb[UE_id]->button_0,"DL Traffic ON");
Raymond Knopp's avatar
 
Raymond Knopp committed
2434 2435
	}
	else {
Raymond Knopp's avatar
 
Raymond Knopp committed
2436 2437
	  fl_set_button(form_enb[UE_id]->button_0,0);
	  fl_set_object_label(form_enb[UE_id]->button_0,"DL Traffic OFF");
Raymond Knopp's avatar
 
Raymond Knopp committed
2438
	}
2439 2440
      }
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2441 2442 2443 2444 2445 2446 2447 2448 2449 2450 2451 2452 2453 2454
    else {
      if (openair_daq_vars.use_ia_receiver) {
	fl_set_button(form_ue[UE_id]->button_0,1);
	fl_set_object_label(form_ue[UE_id]->button_0, "IA Receiver ON");
      }
      else {
	fl_set_button(form_ue[UE_id]->button_0,0);
	fl_set_object_label(form_ue[UE_id]->button_0, "IA Receiver OFF");
      }
    }

    ret = pthread_create(&thread2, NULL, scope_thread, NULL);
    printf("Scope thread created, ret=%d\n",ret);
  }
2455 2456 2457
#endif

#ifdef EMOS
Raymond Knopp's avatar
 
Raymond Knopp committed
2458 2459
  ret = pthread_create(&thread3, NULL, emos_thread, NULL);
  printf("EMOS thread created, ret=%d\n",ret);
2460 2461
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2462
  rt_sleep_ns(10*FRAME_PERIOD);
2463 2464

#ifndef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2465 2466 2467 2468 2469 2470
  pthread_attr_init (&attr_dlsch_threads);
  pthread_attr_setstacksize(&attr_dlsch_threads,OPENAIR_THREAD_STACK_SIZE);
  //attr_dlsch_threads.priority = 1;
  sched_param_dlsch.sched_priority = 90;//sched_get_priority_max(SCHED_FIFO); //OPENAIR_THREAD_PRIORITY;
  pthread_attr_setschedparam  (&attr_dlsch_threads, &sched_param_dlsch);
  pthread_attr_setschedpolicy (&attr_dlsch_threads, SCHED_FIFO);
2471 2472
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2473 2474 2475
  // start the main thread
  if (UE_flag == 1) {
#ifndef USRP
2476
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2477
    thread1 = rt_thread_create(UE_thread, NULL, 100000000);
2478
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
2479 2480 2481 2482 2483 2484 2485 2486
    error_code = pthread_create(&thread1, &attr_dlsch_threads, UE_thread, NULL);
    if (error_code!= 0) {
      LOG_D(HW,"[lte-softmodem.c] Could not allocate UE_thread, error %d\n",error_code);
      return(error_code);
    }
    else {
      LOG_D(HW,"[lte-softmodem.c] Allocate UE_thread successful\n");
    }
2487 2488
#endif
#ifdef DLSCH_THREAD
Raymond Knopp's avatar
 
Raymond Knopp committed
2489 2490 2491
    init_rx_pdsch_thread();
    rt_sleep_ns(FRAME_PERIOD/10);
    init_dlsch_threads();
2492
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2493
    printf("UE threads created\n");
Raymond Knopp's avatar
 
Raymond Knopp committed
2494
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
2495
    printf("UE functionality not yet supported on USRP");
2496
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2497 2498
  }
  else {
2499

Raymond Knopp's avatar
 
Raymond Knopp committed
2500 2501 2502 2503
    if (multi_thread>0) {
      init_eNB_proc();
      LOG_D(HW,"[lte-softmodem.c] eNB threads created\n");
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2504
    printf("Creating eNB_thread \n");
Raymond Knopp's avatar
 
Raymond Knopp committed
2505 2506 2507 2508 2509 2510 2511 2512 2513 2514
#ifdef RTAI
    thread0 = rt_thread_create(eNB_thread, NULL, 100000000);
#else
    error_code = pthread_create(&thread0, &attr_dlsch_threads, eNB_thread, NULL);
    if (error_code!= 0) {
      LOG_D(HW,"[lte-softmodem.c] Could not allocate eNB_thread, error %d\n",error_code);
      return(error_code);
    }
    else {
      LOG_D(HW,"[lte-softmodem.c] Allocate eNB_thread successful\n");
Raymond Knopp's avatar
 
Raymond Knopp committed
2515
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2516 2517
#endif
  }
2518

Raymond Knopp's avatar
 
Raymond Knopp committed
2519 2520 2521
  // Sleep to allow all threads to setup
  sleep(5);

Raymond Knopp's avatar
 
Raymond Knopp committed
2522 2523
#ifdef USRP
  openair0.trx_start_func(&openair0);
Raymond Knopp's avatar
 
Raymond Knopp committed
2524
  //  printf("returning from usrp start streaming: %llu\n",get_usrp_time(&openair0));
Raymond Knopp's avatar
 
Raymond Knopp committed
2525 2526 2527 2528 2529 2530 2531 2532
#ifdef RTAI
  rt_sem_signal(sync_sem);
#else
  //pthread_mutex_lock(&sync_mutex);
  pthread_cond_signal(&sync_cond);
  //pthread_mutex_unlock(&sync_mutex);
#endif
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2533 2534 2535
  // wait for end of program
  printf("TYPE <CTRL-C> TO TERMINATE\n");
  //getchar();
2536 2537

#if defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
2538 2539
  printf("Entering ITTI signals handler\n");
  itti_wait_tasks_end();
2540
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
2541 2542
  while (oai_exit==0)
    rt_sleep_ns(FRAME_PERIOD);
2543
#endif
2544

Raymond Knopp's avatar
 
Raymond Knopp committed
2545
  // stop threads
2546
#ifdef XFORMS
Raymond Knopp's avatar
 
Raymond Knopp committed
2547 2548 2549 2550 2551 2552 2553 2554 2555 2556 2557 2558 2559 2560
  printf("waiting for XFORMS thread\n");
  if (do_forms==1)
    {
      pthread_join(thread2,&status);
      fl_hide_form(form_stats->stats_form);
      fl_free_form(form_stats->stats_form);
      if (UE_flag==1) {
	fl_hide_form(form_ue[UE_id]->lte_phy_scope_ue);
	fl_free_form(form_ue[UE_id]->lte_phy_scope_ue);
      } else {
	for(UE_id=0;UE_id<scope_enb_num_ue;UE_id++) {
	  fl_hide_form(form_enb[UE_id]->lte_phy_scope_enb);
	  fl_free_form(form_enb[UE_id]->lte_phy_scope_enb);
	}
Raymond Knopp's avatar
 
Raymond Knopp committed
2561
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
2562
    }
2563 2564
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2565 2566 2567 2568
  printf("stopping MODEM threads\n");
  // cleanup
  if (UE_flag == 1) {
#ifndef USRP
2569
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2570
    rt_thread_join(thread1); 
2571
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
2572
    pthread_join(thread1,&status); 
2573 2574
#endif
#ifdef DLSCH_THREAD
Raymond Knopp's avatar
 
Raymond Knopp committed
2575 2576
    cleanup_dlsch_threads();
    cleanup_rx_pdsch_thread();
2577
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2578 2579 2580
#endif
  }
  else {
2581
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2582
    rt_thread_join(thread0); 
2583
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
2584
#ifdef DEBUG_THREADS
Raymond Knopp's avatar
 
Raymond Knopp committed
2585
    printf("Joining eNB_thread ...");
Raymond Knopp's avatar
 
Raymond Knopp committed
2586
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2587
    pthread_join(thread0,(void**)&eNB_thread_status_p); 
Raymond Knopp's avatar
 
Raymond Knopp committed
2588
#ifdef DEBUG_THREADS
Raymond Knopp's avatar
 
Raymond Knopp committed
2589
    printf("status %d\n",*eNB_thread_status_p);
Raymond Knopp's avatar
 
Raymond Knopp committed
2590
#endif
2591
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2592

Raymond Knopp's avatar
 
Raymond Knopp committed
2593 2594 2595
    if (multi_thread>0) {
      printf("Killing eNB processing threads\n");
      kill_eNB_proc();
Raymond Knopp's avatar
 
Raymond Knopp committed
2596

Raymond Knopp's avatar
 
Raymond Knopp committed
2597
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2598
  }
2599

2600
#ifdef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
2601
  //cleanup_pdcp_thread();
2602 2603
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2604
#ifdef USRP
2605
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2606
  rt_sem_delete(sync_sem);
Raymond Knopp's avatar
 
Raymond Knopp committed
2607
  stop_rt_timer();
Raymond Knopp's avatar
 
Raymond Knopp committed
2608 2609 2610
#else
  pthread_cond_destroy(&sync_cond);
  pthread_mutex_destroy(&sync_mutex);
Raymond Knopp's avatar
 
Raymond Knopp committed
2611
#endif
2612 2613
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2614
#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
2615 2616 2617 2618
  printf("stopping card\n");
  openair0_stop(card);
  printf("closing openair0_lib\n");
  openair0_close();
Raymond Knopp's avatar
 
Raymond Knopp committed
2619
#endif
2620 2621

#ifdef EMOS
Raymond Knopp's avatar
 
Raymond Knopp committed
2622 2623 2624
  printf("waiting for EMOS thread\n");
  pthread_cancel(thread3);
  pthread_join(thread3,&status);
2625 2626 2627
#endif

#ifdef EMOS
Raymond Knopp's avatar
 
Raymond Knopp committed
2628 2629
  error_code = rtf_destroy(CHANSOUNDER_FIFO_MINOR);
  printf("[OPENAIR][SCHED][CLEANUP] EMOS FIFO closed, error_code %d\n", error_code);
2630 2631
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2632 2633
  if (ouput_vcd)
    vcd_signal_dumper_close();
2634

Raymond Knopp's avatar
 
Raymond Knopp committed
2635
  logClean();
2636

Raymond Knopp's avatar
 
Raymond Knopp committed
2637 2638
  return 0;
}
2639

Raymond Knopp's avatar
 
Raymond Knopp committed
2640
void setup_ue_buffers(PHY_VARS_UE *phy_vars_ue, LTE_DL_FRAME_PARMS *frame_parms, int carrier) {
2641

Raymond Knopp's avatar
 
Raymond Knopp committed
2642
  int i;
Raymond Knopp's avatar
 
Raymond Knopp committed
2643
#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
2644 2645 2646 2647 2648
  if (phy_vars_ue) {
    if ((frame_parms->nb_antennas_rx>1) && (carrier>0)) {
      printf("RX antennas > 1 and carrier > 0 not possible\n");
      exit(-1);
    }
2649

Raymond Knopp's avatar
 
Raymond Knopp committed
2650 2651 2652 2653
    if ((frame_parms->nb_antennas_tx>1) && (carrier>0)) {
      printf("TX antennas > 1 and carrier > 0 not possible\n");
      exit(-1);
    }
2654
    
Raymond Knopp's avatar
 
Raymond Knopp committed
2655 2656 2657 2658
    // replace RX signal buffers with mmaped HW versions
    for (i=0;i<frame_parms->nb_antennas_rx;i++) {
      free(phy_vars_ue->lte_ue_common_vars.rxdata[i]);
      phy_vars_ue->lte_ue_common_vars.rxdata[i] = (int32_t*) openair0_exmimo_pci[card].adc_head[i+carrier];
2659 2660


Raymond Knopp's avatar
 
Raymond Knopp committed
2661 2662 2663 2664 2665
      printf("rxdata[%d] @ %p\n",i,phy_vars_ue->lte_ue_common_vars.rxdata[i]);
    }
    for (i=0;i<frame_parms->nb_antennas_tx;i++) {
      free(phy_vars_ue->lte_ue_common_vars.txdata[i]);
      phy_vars_ue->lte_ue_common_vars.txdata[i] = (int32_t*) openair0_exmimo_pci[card].dac_head[i+carrier];
2666

Raymond Knopp's avatar
 
Raymond Knopp committed
2667
      printf("txdata[%d] @ %p\n",i,phy_vars_ue->lte_ue_common_vars.txdata[i]);
2668 2669
    }
  }
Raymond Knopp's avatar
 
Raymond Knopp committed
2670 2671 2672
#else
  printf("USRP not supported for UE yet!");
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2673
}
2674

Raymond Knopp's avatar
 
Raymond Knopp committed
2675
void setup_eNB_buffers(PHY_VARS_eNB *phy_vars_eNB, LTE_DL_FRAME_PARMS *frame_parms, int carrier) {
2676

Raymond Knopp's avatar
 
Raymond Knopp committed
2677
  int i;
Raymond Knopp's avatar
 
Raymond Knopp committed
2678 2679
#ifdef USRP
  uint16_t N_TA_offset = 0;
Raymond Knopp's avatar
 
Raymond Knopp committed
2680 2681
#else
  int j;
Raymond Knopp's avatar
 
Raymond Knopp committed
2682
#endif
2683

Raymond Knopp's avatar
 
Raymond Knopp committed
2684 2685 2686 2687 2688
  if (phy_vars_eNB) {
    if ((frame_parms->nb_antennas_rx>1) && (carrier>0)) {
      printf("RX antennas > 1 and carrier > 0 not possible\n");
      exit(-1);
    }
2689

Raymond Knopp's avatar
 
Raymond Knopp committed
2690 2691 2692 2693
    if ((frame_parms->nb_antennas_tx>1) && (carrier>0)) {
      printf("TX antennas > 1 and carrier > 0 not possible\n");
      exit(-1);
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2694

Raymond Knopp's avatar
 
Raymond Knopp committed
2695
#ifdef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
2696 2697 2698 2699 2700 2701 2702 2703
    if (frame_parms->frame_type == TDD) {
      if (phy_vars_eNB->lte_frame_parms.N_RB_DL == 100)
	N_TA_offset = 624;
      else if (phy_vars_eNB->lte_frame_parms.N_RB_DL == 50)
	N_TA_offset = 624/2;
      else if (phy_vars_eNB->lte_frame_parms.N_RB_DL == 25)
	N_TA_offset = 624/4;
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2704
#endif
2705
    
Raymond Knopp's avatar
 
Raymond Knopp committed
2706
    // replace RX signal buffers with mmaped HW versions
Raymond Knopp's avatar
 
Raymond Knopp committed
2707
#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
2708 2709 2710 2711 2712 2713 2714 2715
    for (i=0;i<frame_parms->nb_antennas_rx;i++) {
      free(phy_vars_eNB->lte_eNB_common_vars.rxdata[0][i]);
      phy_vars_eNB->lte_eNB_common_vars.rxdata[0][i] = (int32_t*) openair0_exmimo_pci[card].adc_head[i+carrier];

      printf("rxdata[%d] @ %p\n",i,phy_vars_eNB->lte_eNB_common_vars.rxdata[0][i]);
      for (j=0;j<16;j++) {
	printf("rxbuffer %d: %x\n",j,phy_vars_eNB->lte_eNB_common_vars.rxdata[0][i][j]);
	phy_vars_eNB->lte_eNB_common_vars.rxdata[0][i][j] = 16-j;
2716
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
2717 2718 2719 2720 2721 2722 2723 2724 2725
    }
    for (i=0;i<frame_parms->nb_antennas_tx;i++) {
      free(phy_vars_eNB->lte_eNB_common_vars.txdata[0][i]);
      phy_vars_eNB->lte_eNB_common_vars.txdata[0][i] = (int32_t*) openair0_exmimo_pci[card].dac_head[i+carrier];

      printf("txdata[%d] @ %p\n",i,phy_vars_eNB->lte_eNB_common_vars.txdata[0][i]);
      for (j=0;j<16;j++) {
	printf("txbuffer %d: %x\n",j,phy_vars_eNB->lte_eNB_common_vars.txdata[0][i][j]);
	phy_vars_eNB->lte_eNB_common_vars.txdata[0][i][j] = 16-j;
2726
      }
2727
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2728 2729 2730
#else // USRP
    for (i=0;i<frame_parms->nb_antennas_rx;i++) {
        free(phy_vars_eNB->lte_eNB_common_vars.rxdata[0][i]);
Raymond Knopp's avatar
 
Raymond Knopp committed
2731
        rxdata = (int32_t*)malloc16(samples_per_frame*sizeof(int32_t));
Raymond Knopp's avatar
 
Raymond Knopp committed
2732
        phy_vars_eNB->lte_eNB_common_vars.rxdata[0][i] = rxdata-N_TA_offset; // N_TA offset for TDD
Raymond Knopp's avatar
 
Raymond Knopp committed
2733
        memset(rxdata, 0, samples_per_frame*sizeof(int32_t));
Raymond Knopp's avatar
 
Raymond Knopp committed
2734
        printf("rxdata[%d] @ %p (%p)\n", i, phy_vars_eNB->lte_eNB_common_vars.rxdata[0][i],rxdata);
Raymond Knopp's avatar
 
Raymond Knopp committed
2735 2736 2737
    }
    for (i=0;i<frame_parms->nb_antennas_tx;i++) {
        free(phy_vars_eNB->lte_eNB_common_vars.txdata[0][i]);
Raymond Knopp's avatar
 
Raymond Knopp committed
2738
        txdata = (int32_t*)malloc16(samples_per_frame*sizeof(int32_t));
Raymond Knopp's avatar
 
Raymond Knopp committed
2739
        phy_vars_eNB->lte_eNB_common_vars.txdata[0][i] = txdata;
Raymond Knopp's avatar
 
Raymond Knopp committed
2740
        memset(txdata, 0, samples_per_frame*sizeof(int32_t));
Raymond Knopp's avatar
 
Raymond Knopp committed
2741 2742 2743
        printf("txdata[%d] @ %p\n", i, phy_vars_eNB->lte_eNB_common_vars.txdata[0][i]);
    }
#endif
2744
  }
Raymond Knopp's avatar
 
Raymond Knopp committed
2745
}