lte-softmodem.c 89.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
openair0_config_t openair0_cfg[MAX_CARDS];
Raymond Knopp's avatar
 
Raymond Knopp committed
150

Raymond Knopp's avatar
 
Raymond Knopp committed
151 152
int32_t *rxdata;
int32_t *txdata;
153 154
int setup_ue_buffers(PHY_VARS_UE **phy_vars_ue, openair0_config_t *openair0_cfg, openair0_rf_map rf_map[MAX_NUM_CCs]);
int setup_eNB_buffers(PHY_VARS_eNB **phy_vars_eNB, openair0_config_t *openair0_cfg, openair0_rf_map rf_map[MAX_NUM_CCs]);
155 156

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

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

170 171
static int                      thread0;
static int                      thread1;
Raymond Knopp's avatar
 
Raymond Knopp committed
172 173 174
#ifdef USRP
static SEM *sync_sem; // to sync rx & tx streaming
#endif
175 176
//static int sync_thread;
#else
177 178 179 180
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
181 182 183 184
#ifdef USRP
pthread_cond_t sync_cond;
pthread_mutex_t sync_mutex;
#endif
185 186
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
187 188 189 190
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
191

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

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

202
/*
Raymond Knopp's avatar
 
Raymond Knopp committed
203 204 205
  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;
206 207 208 209
*/
//extern unsigned int bigphys_top;
//extern unsigned int mem_base;

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

212

213
#if defined(ENABLE_ITTI)
214 215
static volatile int             start_eNB = 0;
static volatile int             start_UE = 0;
216
#endif
217 218 219 220 221 222
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
223

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

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

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

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

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

Raymond Knopp's avatar
 
Raymond Knopp committed
247
#ifndef USRP
248 249 250 251
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;
252

Raymond Knopp's avatar
 
Raymond Knopp committed
253

Raymond Knopp's avatar
 
Raymond Knopp committed
254
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
255
static unsigned int             rxg_max[4] =    {133,133,133,133};
Raymond Knopp's avatar
 
Raymond Knopp committed
256 257
//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
258
static int                      tx_max_power =  0;
Raymond Knopp's avatar
 
Raymond Knopp committed
259 260 261
char ref[128] = "internal";
char channels[128] = "0";

Raymond Knopp's avatar
 
Raymond Knopp committed
262 263 264
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
265 266 267 268 269 270 271 272 273

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
274
/*
Raymond Knopp's avatar
 
Raymond Knopp committed
275 276 277
  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};
278
*/
Raymond Knopp's avatar
 
Raymond Knopp committed
279 280
//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
281 282
//{8254617, 8254617, 8254617, 8254617}; //eNB khalifa
//{8255067,8254810,8257340,8257340}; // eNB PETRONAS
283

Raymond Knopp's avatar
 
Raymond Knopp committed
284 285 286 287 288
//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};
289 290 291

static runmode_t                mode;
static int                      rx_input_level_dBm;
292
static int                      online_log_messages=0;
293
#ifdef XFORMS
294
extern int                      otg_enabled;
295
static char                     do_forms=0;
296
#else
297
int                             otg_enabled;
298
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
299
//int                             number_of_cards =   1;
Raymond Knopp's avatar
 
Raymond Knopp committed
300
#ifndef USRP
301 302
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
303
#endif
304

Raymond Knopp's avatar
 
Raymond Knopp committed
305
static LTE_DL_FRAME_PARMS      *frame_parms[MAX_NUM_CCs];
306

Raymond Knopp's avatar
 
Raymond Knopp committed
307
int multi_thread=1;
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

Raymond Knopp's avatar
 
Raymond Knopp committed
724
void do_OFDM_mod_rt(int subframe,PHY_VARS_eNB *phy_vars_eNB) {
Raymond Knopp's avatar
 
Raymond Knopp committed
725 726 727 728

  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
    
Raymond Knopp's avatar
 
Raymond Knopp committed
876
    do_OFDM_mod_rt(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
void init_eNB_proc(void) {
Raymond Knopp's avatar
 
Raymond Knopp committed
1030 1031

  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
void kill_eNB_proc(void) {
Raymond Knopp's avatar
 
Raymond Knopp committed
1090 1091

  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 1165
  //  int tx_offset;

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



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

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

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

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

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

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

Raymond Knopp's avatar
 
Raymond Knopp committed
1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213
#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
1214 1215

#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
1216 1217
      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
1218 1219
      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
1220 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
      //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
1248
      vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_DAQ_MBOX, *DAQ_MBOX);
Raymond Knopp's avatar
 
Raymond Knopp committed
1249 1250 1251 1252 1253 1254 1255 1256 1257 1258
      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
1259
	vcd_signal_dumper_dump_variable_by_name(VCD_SIGNAL_DUMPER_VARIABLES_DAQ_MBOX, *DAQ_MBOX);
Raymond Knopp's avatar
 
Raymond Knopp committed
1260 1261 1262 1263 1264 1265 1266 1267 1268 1269 1270
	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
1271
	  diff = 150-mbox_current+mbox_target;
Raymond Knopp's avatar
 
Raymond Knopp committed
1272
	else if ((mbox_current<15) && (mbox_target>=135))
Raymond Knopp's avatar
 
Raymond Knopp committed
1273
	  diff = -150+mbox_target-mbox_current;
Raymond Knopp's avatar
 
Raymond Knopp committed
1274
	else
Raymond Knopp's avatar
 
Raymond Knopp committed
1275
	  diff = mbox_target - mbox_current;
Raymond Knopp's avatar
 
Raymond Knopp committed
1276
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1277 1278

#else  // USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
1279 1280
      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
1281 1282
      while (rx_cnt < sf_bounds[hw_subframe]) {

Raymond Knopp's avatar
 
Raymond Knopp committed
1283 1284 1285
	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
1286 1287 1288
	/*
	// Grab 1/4 of RX buffer and get timestamp

Raymond Knopp's avatar
 
Raymond Knopp committed
1289 1290 1291 1292 1293
	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
1294
	  oai_exit=1;
Raymond Knopp's avatar
 
Raymond Knopp committed
1295 1296 1297 1298 1299 1300 1301 1302 1303 1304

	*/

	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
1305 1306
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_TRX_READ,0);

Raymond Knopp's avatar
 
Raymond Knopp committed
1307
	// Transmit TX buffer based on timestamp from RX
Raymond Knopp's avatar
 
Raymond Knopp committed
1308
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_TRX_WRITE,1);
Raymond Knopp's avatar
 
Raymond Knopp committed
1309 1310 1311 1312 1313
	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
1314
	vcd_signal_dumper_dump_function_by_name(VCD_SIGNAL_DUMPER_FUNCTIONS_TRX_WRITE,0);
Raymond Knopp's avatar
 
Raymond Knopp committed
1315
	/*
Raymond Knopp's avatar
 
Raymond Knopp committed
1316 1317 1318 1319 1320 1321 1322 1323 1324
	// 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
1325
	*/
Raymond Knopp's avatar
 
Raymond Knopp committed
1326

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

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

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

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

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

Raymond Knopp's avatar
 
Raymond Knopp committed
1344 1345 1346 1347 1348 1349 1350 1351 1352 1353 1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368
	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
1369 1370
	  if ((slot&1) == 0) {
	    LOG_I(PHY,"[eNB] Single thread slot %d\n",slot);
1371 1372
	      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++) {
Raymond Knopp's avatar
 
Raymond Knopp committed
1373
		do_OFDM_mod_rt((2+(slot>>1))%10,PHY_vars_eNB_g[0][CC_id]);
1374
	      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1375
	  }
Raymond Knopp's avatar
 
Raymond Knopp committed
1376 1377
	}
	else { // multi-thread > 0
Raymond Knopp's avatar
 
Raymond Knopp committed
1378
	  if ((slot&1) == 1) {
Raymond Knopp's avatar
 
Raymond Knopp committed
1379
	    sf = ((slot>>1)+1)%10;
Raymond Knopp's avatar
 
Raymond Knopp committed
1380 1381 1382 1383
	    //		    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
1384 1385
	      }
	      else {
Raymond Knopp's avatar
 
Raymond Knopp committed
1386 1387 1388 1389 1390 1391 1392 1393 1394 1395 1396
		//		      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
1397 1398
		}
	      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1399 1400 1401 1402
	      
	      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
1403
	      else {
Raymond Knopp's avatar
 
Raymond Knopp committed
1404 1405 1406 1407 1408 1409 1410 1411 1412 1413 1414 1415
		//		      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
1416
	      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1417
	      
Raymond Knopp's avatar
 
Raymond Knopp committed
1418
	    }
Raymond Knopp's avatar
 
Raymond Knopp committed
1419 1420 1421
	  }
	}
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1422 1423 1424 1425 1426 1427 1428
#ifndef RTAI
      //pthread_mutex_lock(&tti_mutex);
#endif




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


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

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

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

Raymond Knopp's avatar
 
Raymond Knopp committed
1476 1477


Raymond Knopp's avatar
 
Raymond Knopp committed
1478
#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
1479 1480
/* 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) {
1481
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
1482 1483 1484 1485 1486 1487 1488 1489 1490 1491 1492 1493 1494
  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
1495
  //exmimo_config_t *p_exmimo_config = openair0_exmimo_pci[card].exmimo_config_ptr;;
Raymond Knopp's avatar
 
Raymond Knopp committed
1496 1497
#endif

1498

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

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

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

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

Raymond Knopp's avatar
 
Raymond Knopp committed
1515 1516 1517 1518 1519 1520 1521 1522 1523 1524 1525 1526 1527
  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
1528
      
Raymond Knopp's avatar
 
Raymond Knopp committed
1529 1530
    if (is_synchronized) {
      //this is the mbox counter that indicates the start of the frame
Raymond Knopp's avatar
 
Raymond Knopp committed
1531
      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
1532 1533 1534 1535 1536 1537 1538 1539 1540 1541 1542 1543 1544
      //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;
1545
	
Raymond Knopp's avatar
 
Raymond Knopp committed
1546 1547 1548 1549 1550 1551 1552 1553
      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
1554
	}
Raymond Knopp's avatar
 
Raymond Knopp committed
1555 1556 1557 1558 1559 1560 1561 1562 1563 1564 1565 1566 1567 1568 1569 1570 1571 1572 1573 1574 1575 1576 1577 1578
	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);
1579
	  
Raymond Knopp's avatar
 
Raymond Knopp committed
1580 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
	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
1607
      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
1608 1609 1610 1611 1612 1613 1614 1615 1616 1617 1618
	
    }
    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
1619
      if (initial_sync(PHY_vars_UE_g[0][0],mode)==0) {
Raymond Knopp's avatar
 
Raymond Knopp committed
1620
	/*
Raymond Knopp's avatar
 
Raymond Knopp committed
1621 1622 1623 1624 1625
	  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
1626
	*/
Raymond Knopp's avatar
 
Raymond Knopp committed
1627 1628 1629 1630 1631 1632 1633 1634
	//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
1635
	}
Raymond Knopp's avatar
 
Raymond Knopp committed
1636 1637 1638 1639 1640 1641
	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
1642
	  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
1643 1644 1645 1646 1647 1648 1649 1650 1651 1652 1653 1654 1655 1656 1657 1658 1659 1660
	  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
1661 1662 1663 1664 1665
	  for (card=0;card<MAX_CARDS;card++) {
	    for (i=0; i<openair0_cfg[card].rx_num_channels; i++) {
	      openair0_cfg[card].rx_freq[i] = carrier_freq[card][i]+openair_daq_vars.freq_offset;
	      openair0_cfg[card].tx_freq[i] = carrier_freq[card][i]+openair_daq_vars.freq_offset;
	    }
Raymond Knopp's avatar
 
Raymond Knopp committed
1666
	  }
1667
	  openair0_dump_config(&openair0_cfg[0],UE_flag);
1668
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1669 1670
	  rt_sleep_ns(FRAME_PERIOD);
	}
Raymond Knopp's avatar
 
Raymond Knopp committed
1671
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1672
    }
1673
    
Raymond Knopp's avatar
 
Raymond Knopp committed
1674 1675 1676 1677 1678 1679 1680 1681 1682 1683 1684 1685 1686 1687
    /*
      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);
1688 1689

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

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

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

1703
#endif
1704

1705

Raymond Knopp's avatar
 
Raymond Knopp committed
1706
static void get_options (int argc, char **argv) {
1707
  int c;
Raymond Knopp's avatar
 
Raymond Knopp committed
1708 1709
  //  char                          line[1000];
  //  int                           l;
1710
  int k;//i,j,k;
Raymond Knopp's avatar
 
Raymond Knopp committed
1711 1712 1713
#ifdef USRP
  int clock_src;
#endif
1714 1715
  int CC_id;

Raymond Knopp's avatar
 
Raymond Knopp committed
1716 1717 1718 1719 1720 1721 1722 1723 1724 1725 1726 1727 1728 1729 1730 1731 1732 1733 1734 1735 1736 1737 1738 1739
  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}};
  
1740
  while ((c = getopt_long (argc, argv, "C:dK:g:G:qO:SUVRMr:s:",long_options,NULL)) != -1) {
Raymond Knopp's avatar
 
Raymond Knopp committed
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 1772
    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
1773
      multi_thread=0;
Raymond Knopp's avatar
 
Raymond Knopp committed
1774 1775
      break;
    case 'C':
1776 1777 1778 1779 1780 1781 1782 1783 1784 1785 1786
      for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
      downlink_frequency[CC_id][0] = atof(optarg); // Use float to avoid issue with frequency over 2^31.
      downlink_frequency[CC_id][1] = downlink_frequency[CC_id][0];
      downlink_frequency[CC_id][2] = downlink_frequency[CC_id][0];
      downlink_frequency[CC_id][3] = downlink_frequency[CC_id][0];
      carrier_freq[CC_id][0] = downlink_frequency[CC_id][0];
      carrier_freq[CC_id][1] = downlink_frequency[CC_id][1];
      carrier_freq[CC_id][2] = downlink_frequency[CC_id][2];
      carrier_freq[CC_id][3] = downlink_frequency[CC_id][3];
      }
      printf("Downlink for CC_id %d frequency set to %u\n", CC_id, downlink_frequency[CC_id][0]);
Raymond Knopp's avatar
 
Raymond Knopp committed
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
      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':
Raymond Knopp's avatar
 
Raymond Knopp committed
1821 1822 1823 1824 1825 1826 1827 1828 1829 1830 1831 1832 1833 1834 1835 1836 1837 1838 1839 1840 1841 1842
      for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
	switch(atoi(optarg)) {
	case 6:
	  frame_parms[CC_id]->N_RB_DL=6;
	  frame_parms[CC_id]->N_RB_UL=6;
	  break;
	case 25:
	  frame_parms[CC_id]->N_RB_DL=25;
	  frame_parms[CC_id]->N_RB_UL=25;
	  break;
	case 50:
	  frame_parms[CC_id]->N_RB_DL=50;
	  frame_parms[CC_id]->N_RB_UL=50;
	  break;
	case 100:
	  frame_parms[CC_id]->N_RB_DL=100;
	  frame_parms[CC_id]->N_RB_UL=100;
	  break;
	default:
	  printf("Unknown N_RB_DL %d, switching to 25\n",atoi(optarg));
	  break;
	}
Raymond Knopp's avatar
 
Raymond Knopp committed
1843
      }
1844
      break;
Raymond Knopp's avatar
 
Raymond Knopp committed
1845 1846
    case 's':
#ifdef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
1847 1848 1849

      clock_src = atoi(optarg);
      if (clock_src == 0) {
1850
	//	char ref[128] = "internal";
Raymond Knopp's avatar
 
Raymond Knopp committed
1851 1852 1853
	//strncpy(uhd_ref, ref, strlen(ref)+1);
      }
      else if (clock_src == 1) {
1854
	//char ref[128] = "external";
Raymond Knopp's avatar
 
Raymond Knopp committed
1855 1856
	//strncpy(uhd_ref, ref, strlen(ref)+1);
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
1857 1858 1859 1860
#else
      printf("Note: -s not defined for ExpressMIMO2\n");
#endif
      break;
1861 1862 1863 1864 1865 1866
    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
1867 1868
    default:
      break;
1869 1870
    }
  }
Raymond Knopp's avatar
 
Raymond Knopp committed
1871 1872 1873 1874 1875 1876 1877 1878 1879 1880 1881 1882 1883 1884
  
  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 */
1885
    for (i=0; i < enb_properties->number; i++) {
Raymond Knopp's avatar
 
Raymond Knopp committed
1886 1887 1888 1889 1890 1891 1892
      for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
	frame_parms[CC_id]->frame_type =       enb_properties->properties[i]->frame_type[CC_id];
	frame_parms[CC_id]->tdd_config =       enb_properties->properties[i]->tdd_config[CC_id];
	frame_parms[CC_id]->tdd_config_S =     enb_properties->properties[i]->tdd_config_s[CC_id];
	frame_parms[CC_id]->Ncp =              enb_properties->properties[i]->prefix_type[CC_id];
	
	//for (j=0; j < enb_properties->properties[i]->nb_cc; j++ ){ 
1893
	frame_parms[CC_id]->Nid_cell          =  enb_properties->properties[i]->Nid_cell[CC_id];
Raymond Knopp's avatar
 
Raymond Knopp committed
1894
	frame_parms[CC_id]->N_RB_DL          =  enb_properties->properties[i]->N_RB_DL[CC_id];
1895
	//} // j
Raymond Knopp's avatar
 
Raymond Knopp committed
1896 1897
      }

1898 1899 1900 1901 1902 1903 1904 1905 1906 1907 1908 1909 1910 1911 1912 1913
      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;
    
    
1914 1915 1916 1917 1918 1919 1920 1921 1922 1923
      // adjust the log 
      for (CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
	for (k = 0 ; k < 4; k++) {
	  downlink_frequency[CC_id][k] =       enb_properties->properties[i]->downlink_frequency[CC_id];
	  uplink_frequency_offset[CC_id][k] =  enb_properties->properties[i]->uplink_frequency_offset[CC_id];
	}
	printf("Downlink frequency/ uplink offset of CC_id %d set to %llu/%d\n", CC_id, 
	       enb_properties->properties[i]->downlink_frequency[CC_id],
	       enb_properties->properties[i]->uplink_frequency_offset[CC_id]);
      } // CC_id 
1924
    }// i
Raymond Knopp's avatar
 
Raymond Knopp committed
1925 1926
  }
}
1927

Raymond Knopp's avatar
 
Raymond Knopp committed
1928
int main(int argc, char **argv) {
1929
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
1930
  // RT_TASK *task;
Raymond Knopp's avatar
 
Raymond Knopp committed
1931
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
1932
  int *eNB_thread_status_p;
Raymond Knopp's avatar
 
Raymond Knopp committed
1933
  //  int *eNB_thread_status_rx[10],*eNB_thread_status_tx[10];
1934
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1935
  int i,j,aa;
1936
#if defined (XFORMS) || defined (EMOS) || (! defined (RTAI))
Raymond Knopp's avatar
 
Raymond Knopp committed
1937
  void *status;
1938
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
1939
  
Raymond Knopp's avatar
 
Raymond Knopp committed
1940
  int CC_id;
Raymond Knopp's avatar
 
Raymond Knopp committed
1941 1942
  uint16_t Nid_cell = 0;
  uint8_t  cooperation_flag=0, transmission_mode=1, abstraction_flag=0;
1943
#ifndef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
1944
  uint8_t beta_ACK=0,beta_RI=0,beta_CQI=2;
1945 1946 1947
#endif

#ifdef ENABLE_TCXO
Raymond Knopp's avatar
 
Raymond Knopp committed
1948
  unsigned int tcxo = 114;
1949 1950
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
1951
  //  int amp;
Raymond Knopp's avatar
 
Raymond Knopp committed
1952 1953
  // uint8_t prach_fmt;
  // int N_ZC;
1954

Raymond Knopp's avatar
 
Raymond Knopp committed
1955
  //  int ret, ant;
Raymond Knopp's avatar
 
Raymond Knopp committed
1956
  int ant_offset=0;
1957 1958 1959
#ifdef XFORMS
  int ret;
#endif
1960
#if defined (EMOS) || (! defined (RTAI))
Raymond Knopp's avatar
 
Raymond Knopp committed
1961
  int error_code;
1962
#endif
1963 1964

  memset(&openair0_cfg[0],0,sizeof(openair0_config_t)*MAX_CARDS);
1965

Raymond Knopp's avatar
 
Raymond Knopp committed
1966
  set_latency_target();
1967

Raymond Knopp's avatar
 
Raymond Knopp committed
1968
  mode = normal_txrx;
1969

Raymond Knopp's avatar
 
Raymond Knopp committed
1970 1971 1972 1973 1974 1975 1976 1977 1978 1979 1980 1981
  for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
    frame_parms[CC_id] = (LTE_DL_FRAME_PARMS*) malloc(sizeof(LTE_DL_FRAME_PARMS));
    /* Set some default values that may be overwritten while reading options */
    frame_parms[CC_id]->frame_type         = TDD; /* TDD */
    frame_parms[CC_id]->tdd_config          = 3;
    frame_parms[CC_id]->tdd_config_S        = 0;
    frame_parms[CC_id]->N_RB_DL             = 25;
    frame_parms[CC_id]->N_RB_UL             = 25;
    frame_parms[CC_id]->Ncp                = NORMAL;
    frame_parms[CC_id]->Ncp_UL              = NORMAL;
    frame_parms[CC_id]->Nid_cell            = Nid_cell;
  }
1982

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

Raymond Knopp's avatar
 
Raymond Knopp committed
1985 1986
  //randominit (0);
  set_taus_seed (0);
winckel's avatar
winckel committed
1987

Raymond Knopp's avatar
 
Raymond Knopp committed
1988 1989 1990
  // initialize the log (see log.h for details)
  logInit();

1991
  set_glog(glog_level, glog_verbosity);
Raymond Knopp's avatar
 
Raymond Knopp committed
1992 1993 1994
  if (UE_flag==1)
    {
      printf("configuring for UE\n");
1995

Raymond Knopp's avatar
 
Raymond Knopp committed
1996
      set_comp_log(HW,      LOG_INFO,  LOG_HIGH, 1);
1997
#ifdef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
1998
      set_comp_log(PHY,     LOG_INFO,   LOG_HIGH, 1);
1999
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
2000
      set_comp_log(PHY,     LOG_INFO,   LOG_HIGH, 1);
2001
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2002 2003 2004 2005 2006
      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);
2007
#if defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
2008
      set_comp_log(EMU,     LOG_INFO,   LOG_MED, 1);
2009
# if defined(ENABLE_USE_MME)
Raymond Knopp's avatar
 
Raymond Knopp committed
2010
      set_comp_log(NAS,     LOG_INFO,   LOG_HIGH, 1);
2011 2012
# endif
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2013 2014 2015 2016
    }
  else
    {
      printf("configuring for eNB\n");
2017

2018
      set_comp_log(HW,      hw_log_level, hw_log_verbosity, 1);
2019
#ifdef OPENAIR2
2020
      set_comp_log(PHY,     phy_log_level,   phy_log_verbosity, 1);
2021
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
2022
      set_comp_log(PHY,     LOG_INFO,   LOG_HIGH, 1);
2023
#endif
2024 2025 2026 2027
      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);
2028
#if defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
2029
      set_comp_log(EMU,     LOG_INFO,   LOG_MED, 1);
2030
# if defined(ENABLE_USE_MME)
Raymond Knopp's avatar
 
Raymond Knopp committed
2031 2032 2033 2034
      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);
2035
# endif
2036
#if defined(ENABLE_SECURITY)
Raymond Knopp's avatar
 
Raymond Knopp committed
2037
      set_comp_log(OSA,    LOG_DEBUG,   LOG_HIGH, 1);
2038
#endif
2039
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2040
      set_comp_log(ENB_APP, LOG_INFO, LOG_HIGH, 1);
2041
      set_comp_log(OTG,     LOG_INFO,   LOG_HIGH, 1);
Raymond Knopp's avatar
 
Raymond Knopp committed
2042 2043 2044
      if (online_log_messages == 1) { 
	set_component_filelog(RRC);
	set_component_filelog(PDCP);
Raymond Knopp's avatar
 
Raymond Knopp committed
2045 2046
      }
    }
2047

Raymond Knopp's avatar
 
Raymond Knopp committed
2048 2049 2050 2051 2052 2053 2054
  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");
  }

2055
#if defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
2056 2057 2058 2059 2060 2061
  if (UE_flag == 1) {
    log_set_instance_type (LOG_INSTANCE_UE);
  }
  else {
    log_set_instance_type (LOG_INSTANCE_ENB);
  }
2062

Raymond Knopp's avatar
 
Raymond Knopp committed
2063
  itti_init(TASK_MAX, THREAD_MAX, MESSAGES_ID_MAX, tasks_info, messages_info, messages_definition_xml, itti_dump_file);
2064 2065
#endif

2066
#ifdef NAS_NETLINK
Raymond Knopp's avatar
 
Raymond Knopp committed
2067
  netlink_init();
2068 2069
#endif

2070
#if !defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
2071 2072 2073
  // to make a graceful exit when ctrl-c is pressed
  signal(SIGSEGV, signal_handler);
  signal(SIGINT, signal_handler);
2074
#endif
2075 2076

#ifndef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2077 2078 2079 2080
  check_clock();
#endif

  // init the parameters
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
  for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
    frame_parms[CC_id]->nushift            = 0;
    if (UE_flag==0)
      {
	switch (transmission_mode) {
	case 1: 
	  frame_parms[CC_id]->nb_antennas_tx     = 1;
	  frame_parms[CC_id]->nb_antennas_rx     = 1;
	  break;
	case 2:
	case 5:
	case 6:
	  frame_parms[CC_id]->nb_antennas_tx     = 2;
	  frame_parms[CC_id]->nb_antennas_rx     = 2;
	  break;
	default:
	  printf("Unsupported transmission mode %d\n",transmission_mode);
	  exit(-1);
	}
Raymond Knopp's avatar
 
Raymond Knopp committed
2100
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
2101 2102 2103 2104 2105 2106 2107 2108 2109 2110 2111 2112 2113 2114
    else
      { //UE_flag==1
	frame_parms[CC_id]->nb_antennas_tx     = 1;
	frame_parms[CC_id]->nb_antennas_rx     = 1;
      }
    frame_parms[CC_id]->nb_antennas_tx_eNB = (transmission_mode == 1) ? 1 : 2; //initial value overwritten by initial sync later
    frame_parms[CC_id]->mode1_flag         = (transmission_mode == 1) ? 1 : 0;
    frame_parms[CC_id]->phich_config_common.phich_resource = oneSixth;
    frame_parms[CC_id]->phich_config_common.phich_duration = normal;
    // UL RS Config
    frame_parms[CC_id]->pusch_config_common.ul_ReferenceSignalsPUSCH.cyclicShift = 0;//n_DMRS1 set to 0
    frame_parms[CC_id]->pusch_config_common.ul_ReferenceSignalsPUSCH.groupHoppingEnabled = 0;
    frame_parms[CC_id]->pusch_config_common.ul_ReferenceSignalsPUSCH.sequenceHoppingEnabled = 0;
    frame_parms[CC_id]->pusch_config_common.ul_ReferenceSignalsPUSCH.groupAssignmentPUSCH = 0;
2115
    init_ul_hopping(frame_parms[CC_id]);
Raymond Knopp's avatar
 
Raymond Knopp committed
2116

2117
    init_frame_parms(frame_parms[CC_id],1);
Raymond Knopp's avatar
 
Raymond Knopp committed
2118 2119 2120 2121 2122 2123 2124 2125 2126 2127 2128 2129 2130
  }

  phy_init_top(frame_parms[0]);
  phy_init_lte_top(frame_parms[0]);

  for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
    //init prach for openair1 test
    frame_parms[CC_id]->prach_config_common.rootSequenceIndex=22; 
    frame_parms[CC_id]->prach_config_common.prach_ConfigInfo.zeroCorrelationZoneConfig=1;
    frame_parms[CC_id]->prach_config_common.prach_ConfigInfo.prach_ConfigIndex=0; 
    frame_parms[CC_id]->prach_config_common.prach_ConfigInfo.highSpeedFlag=0;
    frame_parms[CC_id]->prach_config_common.prach_ConfigInfo.prach_FreqOffset=0;
    // prach_fmt = get_prach_fmt(frame_parms->prach_config_common.prach_ConfigInfo.prach_ConfigIndex, frame_parms->frame_type);
Raymond Knopp's avatar
 
Raymond Knopp committed
2131
  // N_ZC = (prach_fmt <4)?839:139;
Raymond Knopp's avatar
 
Raymond Knopp committed
2132
  }
Raymond Knopp's avatar
 
Raymond Knopp committed
2133 2134

  if (UE_flag==1) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2135 2136 2137
    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++) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2138
      PHY_vars_UE_g[CC_id][0] = init_lte_UE(frame_parms[CC_id], UE_id,abstraction_flag,transmission_mode);
Raymond Knopp's avatar
 
Raymond Knopp committed
2139
      
Raymond Knopp's avatar
 
Raymond Knopp committed
2140 2141 2142 2143 2144 2145 2146 2147 2148 2149
#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;
      }
2150
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2151 2152 2153 2154 2155 2156
      
      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;
2157
#ifndef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
2158
      PHY_vars_UE_g[CC_id][0]->lte_ue_pdcch_vars[0]->crnti = 0x1235;
2159
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2160

Raymond Knopp's avatar
 
Raymond Knopp committed
2161
    for (i=0;i<4;i++) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2162 2163 2164
      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
2165
    }
2166
    
Raymond Knopp's avatar
 
Raymond Knopp committed
2167 2168
    if ((mode == normal_txrx) || (mode == rx_calib_ue) || (mode == no_L2_connect) || (mode == debug_prach)) {
      for (i=0;i<4;i++)
2169
	openair0_cfg[CC_id].rxg_mode[i] =  max_gain;
Raymond Knopp's avatar
 
Raymond Knopp committed
2170
      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
2171 2172 2173
    }
    else if ((mode == rx_calib_ue_med)) {
      for (i=0;i<4;i++)
2174
	openair0_cfg[CC_id].rxg_mode[i] =  med_gain;
Raymond Knopp's avatar
 
Raymond Knopp committed
2175
      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
2176 2177 2178
    }
    else if ((mode == rx_calib_ue_byp)) {
      for (i=0;i<4;i++)
2179
	openair0_cfg[CC_id].rxg_mode[i] =  byp_gain;
Raymond Knopp's avatar
 
Raymond Knopp committed
2180
      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
2181 2182
    }
    
Raymond Knopp's avatar
 
Raymond Knopp committed
2183 2184 2185 2186 2187 2188 2189 2190 2191 2192 2193 2194
    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
2195 2196 2197 2198 2199
    
    //  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
2200 2201 2202
      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++) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2203
	PHY_vars_eNB_g[0][CC_id] = init_lte_eNB(frame_parms[CC_id],eNB_id,Nid_cell,cooperation_flag,transmission_mode,abstraction_flag);
Raymond Knopp's avatar
 
Raymond Knopp committed
2204 2205
      
       
2206
#ifndef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
2207 2208 2209 2210 2211 2212 2213 2214 2215
	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
2216
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2217 2218 2219 2220
      
	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
2221

Raymond Knopp's avatar
 
Raymond Knopp committed
2222
	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
2223

2224 2225 2226
	// set eNB to max gain
	for (i=0;i<4;i++)
	  openair0_cfg[CC_id].rxg_mode[i] =  max_gain;
Raymond Knopp's avatar
 
Raymond Knopp committed
2227
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
2228 2229
      NB_eNB_INST=1;
      NB_INST=1;
Raymond Knopp's avatar
 
Raymond Knopp committed
2230

Raymond Knopp's avatar
 
Raymond Knopp committed
2231
      openair_daq_vars.ue_dl_rb_alloc=0x1fff;
2232
      openair_daq_vars.target_ue_dl_mcs=20;
Raymond Knopp's avatar
 
Raymond Knopp committed
2233 2234
      openair_daq_vars.ue_ul_nb_rb=6;
      openair_daq_vars.target_ue_ul_mcs=6;
Raymond Knopp's avatar
 
Raymond Knopp committed
2235 2236

    }
2237

2238

2239

Raymond Knopp's avatar
 
Raymond Knopp committed
2240
  dump_frame_parms(frame_parms[0]);
2241

Raymond Knopp's avatar
 
Raymond Knopp committed
2242
  if(frame_parms[0]->N_RB_DL == 100) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2243 2244 2245 2246 2247 2248 2249 2250
    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
2251
    tx_delay = 9;
Raymond Knopp's avatar
 
Raymond Knopp committed
2252 2253
#endif
  }
Raymond Knopp's avatar
 
Raymond Knopp committed
2254
  else if(frame_parms[0]->N_RB_DL == 50){
2255
    sample_rate = 15.36e6;
Raymond Knopp's avatar
 
Raymond Knopp committed
2256 2257 2258 2259 2260 2261 2262 2263 2264
#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
  }
Raymond Knopp's avatar
 
Raymond Knopp committed
2265
  else if (frame_parms[0]->N_RB_DL == 25) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2266 2267 2268 2269 2270 2271 2272
    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
2273
    tx_delay = 8;
Raymond Knopp's avatar
 
Raymond Knopp committed
2274 2275 2276
#endif
  }
  
2277

2278 2279 2280 2281 2282 2283 2284
  for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
    openair0_cfg[CC_id].sample_rate = sample_rate;
    openair0_cfg[CC_id].tx_bw = bw;
    openair0_cfg[CC_id].rx_bw = bw;
    for (i=0;i<4;i++) {
      openair0_cfg[CC_id].tx_gain[i] = tx_gain;
      openair0_cfg[CC_id].rx_gain[i] = rx_gain;
Raymond Knopp's avatar
 
Raymond Knopp committed
2285 2286
    }
  }
2287 2288

  if (openair0_device_init(&openair0, &openair0_cfg[0]) <0) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2289 2290 2291
    printf("Exiting, cannot initialize device\n");
    exit(-1);
  }
2292 2293 2294 2295


  mac_xface = malloc(sizeof(MAC_xface));

2296
#ifdef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
2297
  int eMBMS_active=0;
2298

2299
  l2_init(frame_parms[0],eMBMS_active,
Raymond Knopp's avatar
 
Raymond Knopp committed
2300 2301 2302 2303 2304 2305
	  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);
2306 2307
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2308
  mac_xface->macphy_exit = &exit_fun;
2309

winckel's avatar
winckel committed
2310
#if defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
2311 2312 2313 2314
  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
2315 2316
#endif

2317
#ifdef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
2318 2319 2320 2321 2322 2323 2324 2325 2326 2327 2328
  //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
2329
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2330 2331 2332
  }
  init_predef_traffic(UE_flag ? 1 : 0, UE_flag ? 0 : 1);
  //  }
2333 2334
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2335
  //  number_of_cards = openair0_num_detected_cards;
2336

Raymond Knopp's avatar
 
Raymond Knopp committed
2337
  openair_daq_vars.timing_advance = 0;
2338

2339 2340 2341 2342
  openair0_rf_map rf_map[MAX_NUM_CCs];
  rf_map[0].card=0;
  rf_map[0].chain=1;

Raymond Knopp's avatar
 
Raymond Knopp committed
2343 2344
  // connect the TX/RX buffers
  if (UE_flag==1) {
2345 2346 2347
    setup_ue_buffers(PHY_vars_UE_g[0],&openair0_cfg[0],rf_map);
    printf("Setting UE buffer to all-RX\n");
    // Set LSBs for antenna switch (ExpressMIMO)
Raymond Knopp's avatar
 
Raymond Knopp committed
2348
    for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2349
      for (i=0; i<frame_parms[CC_id]->samples_per_tti*10; i++)
2350
	for (aa=0; aa<frame_parms[CC_id]->nb_antennas_tx; aa++)
2351
	  PHY_vars_UE_g[0][CC_id]->lte_ue_common_vars.txdata[aa][i] = 0x00010001;
Raymond Knopp's avatar
 
Raymond Knopp committed
2352
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2353 2354 2355
    //p_exmimo_config->framing.tdd_config = TXRXSWITCH_TESTRX;
  }
  else {
2356 2357 2358 2359 2360 2361
    if (setup_eNB_buffers(PHY_vars_eNB_g[0],&openair0_cfg[0],rf_map)!=0) {
      printf("Error setting up eNB buffer\n");
      exit(-1);
    }
    printf("Setting eNB buffer to all-RX\n");
    // Set LSBs for antenna switch (ExpressMIMO)
Raymond Knopp's avatar
 
Raymond Knopp committed
2362
    for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
Raymond Knopp's avatar
 
Raymond Knopp committed
2363 2364
      for (i=0; i<frame_parms[CC_id]->samples_per_tti*10; i++)
	for (aa=0; aa<frame_parms[CC_id]->nb_antennas_tx; aa++)
2365
	  PHY_vars_eNB_g[0][CC_id]->lte_eNB_common_vars.txdata[0][aa][i] = 0x00010001;
Raymond Knopp's avatar
 
Raymond Knopp committed
2366
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2367
  }
Raymond Knopp's avatar
 
Raymond Knopp committed
2368
#ifndef USRP
2369
  openair0_dump_config(&openair0_cfg[0],UE_flag);
Raymond Knopp's avatar
 
Raymond Knopp committed
2370 2371
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2372
  /*  
Raymond Knopp's avatar
 
Raymond Knopp committed
2373 2374
      for (ant=0;ant<4;ant++)
      p_exmimo_config->rf.do_autocal[ant] = 0;
Raymond Knopp's avatar
 
Raymond Knopp committed
2375
  */
2376 2377

#ifdef EMOS
Raymond Knopp's avatar
 
Raymond Knopp committed
2378 2379 2380 2381 2382 2383 2384 2385 2386
  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);
2387 2388
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2389
  mlockall(MCL_CURRENT | MCL_FUTURE);
2390 2391

#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2392 2393 2394 2395 2396 2397 2398 2399 2400 2401 2402 2403 2404 2405 2406 2407 2408 2409 2410 2411 2412
  // 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
2413 2414 2415 2416 2417 2418 2419 2420 2421 2422
#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
2423 2424 2425
#endif


Raymond Knopp's avatar
 
Raymond Knopp committed
2426 2427

  // this starts the DMA transfers
Raymond Knopp's avatar
 
Raymond Knopp committed
2428
#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
2429 2430
  if (UE_flag!=1)
    openair0_start_rt_acquisition(card);
Raymond Knopp's avatar
 
Raymond Knopp committed
2431
#endif
2432 2433

#ifdef XFORMS
Raymond Knopp's avatar
 
Raymond Knopp committed
2434 2435 2436 2437 2438 2439 2440 2441 2442 2443 2444 2445
  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);
2446
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
2447 2448 2449 2450 2451 2452 2453
    }
    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
2454 2455
	}
	else {
Raymond Knopp's avatar
 
Raymond Knopp committed
2456 2457
	  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
2458
	}
2459 2460
      }
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2461 2462 2463 2464 2465 2466 2467 2468 2469 2470 2471 2472 2473 2474
    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);
  }
2475 2476 2477
#endif

#ifdef EMOS
Raymond Knopp's avatar
 
Raymond Knopp committed
2478 2479
  ret = pthread_create(&thread3, NULL, emos_thread, NULL);
  printf("EMOS thread created, ret=%d\n",ret);
2480 2481
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2482
  rt_sleep_ns(10*FRAME_PERIOD);
2483 2484

#ifndef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2485 2486 2487 2488 2489 2490
  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);
2491 2492
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2493 2494 2495
  // start the main thread
  if (UE_flag == 1) {
#ifndef USRP
2496
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2497
    thread1 = rt_thread_create(UE_thread, NULL, 100000000);
2498
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
2499 2500 2501 2502 2503 2504 2505 2506
    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");
    }
2507 2508
#endif
#ifdef DLSCH_THREAD
Raymond Knopp's avatar
 
Raymond Knopp committed
2509 2510 2511
    init_rx_pdsch_thread();
    rt_sleep_ns(FRAME_PERIOD/10);
    init_dlsch_threads();
2512
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2513
    printf("UE threads created\n");
Raymond Knopp's avatar
 
Raymond Knopp committed
2514
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
2515
    printf("UE functionality not yet supported on USRP");
2516
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2517 2518
  }
  else {
2519

Raymond Knopp's avatar
 
Raymond Knopp committed
2520 2521 2522 2523
    if (multi_thread>0) {
      init_eNB_proc();
      LOG_D(HW,"[lte-softmodem.c] eNB threads created\n");
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2524
    printf("Creating eNB_thread \n");
Raymond Knopp's avatar
 
Raymond Knopp committed
2525 2526 2527 2528 2529 2530 2531 2532 2533 2534
#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
2535
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2536 2537
#endif
  }
2538

Raymond Knopp's avatar
 
Raymond Knopp committed
2539 2540 2541
  // Sleep to allow all threads to setup
  sleep(5);

Raymond Knopp's avatar
 
Raymond Knopp committed
2542 2543
#ifdef USRP
  openair0.trx_start_func(&openair0);
Raymond Knopp's avatar
 
Raymond Knopp committed
2544
  //  printf("returning from usrp start streaming: %llu\n",get_usrp_time(&openair0));
Raymond Knopp's avatar
 
Raymond Knopp committed
2545 2546 2547 2548 2549 2550 2551 2552
#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
2553 2554 2555
  // wait for end of program
  printf("TYPE <CTRL-C> TO TERMINATE\n");
  //getchar();
2556 2557

#if defined(ENABLE_ITTI)
Raymond Knopp's avatar
 
Raymond Knopp committed
2558 2559
  printf("Entering ITTI signals handler\n");
  itti_wait_tasks_end();
2560
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
2561 2562
  while (oai_exit==0)
    rt_sleep_ns(FRAME_PERIOD);
2563
#endif
2564

Raymond Knopp's avatar
 
Raymond Knopp committed
2565
  // stop threads
2566
#ifdef XFORMS
Raymond Knopp's avatar
 
Raymond Knopp committed
2567 2568 2569 2570 2571 2572 2573 2574 2575 2576 2577 2578 2579 2580
  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
2581
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
2582
    }
2583 2584
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2585 2586 2587 2588
  printf("stopping MODEM threads\n");
  // cleanup
  if (UE_flag == 1) {
#ifndef USRP
2589
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2590
    rt_thread_join(thread1); 
2591
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
2592
    pthread_join(thread1,&status); 
2593 2594
#endif
#ifdef DLSCH_THREAD
Raymond Knopp's avatar
 
Raymond Knopp committed
2595 2596
    cleanup_dlsch_threads();
    cleanup_rx_pdsch_thread();
2597
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2598 2599 2600
#endif
  }
  else {
2601
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2602
    rt_thread_join(thread0); 
2603
#else
Raymond Knopp's avatar
 
Raymond Knopp committed
2604
#ifdef DEBUG_THREADS
Raymond Knopp's avatar
 
Raymond Knopp committed
2605
    printf("Joining eNB_thread ...");
Raymond Knopp's avatar
 
Raymond Knopp committed
2606
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2607
    pthread_join(thread0,(void**)&eNB_thread_status_p); 
Raymond Knopp's avatar
 
Raymond Knopp committed
2608
#ifdef DEBUG_THREADS
Raymond Knopp's avatar
 
Raymond Knopp committed
2609
    printf("status %d\n",*eNB_thread_status_p);
Raymond Knopp's avatar
 
Raymond Knopp committed
2610
#endif
2611
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2612

Raymond Knopp's avatar
 
Raymond Knopp committed
2613 2614 2615
    if (multi_thread>0) {
      printf("Killing eNB processing threads\n");
      kill_eNB_proc();
Raymond Knopp's avatar
 
Raymond Knopp committed
2616

Raymond Knopp's avatar
 
Raymond Knopp committed
2617
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2618
  }
2619

2620
#ifdef OPENAIR2
Raymond Knopp's avatar
 
Raymond Knopp committed
2621
  //cleanup_pdcp_thread();
2622 2623
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2624
#ifdef USRP
2625
#ifdef RTAI
Raymond Knopp's avatar
 
Raymond Knopp committed
2626
  rt_sem_delete(sync_sem);
Raymond Knopp's avatar
 
Raymond Knopp committed
2627
  stop_rt_timer();
Raymond Knopp's avatar
 
Raymond Knopp committed
2628 2629 2630
#else
  pthread_cond_destroy(&sync_cond);
  pthread_mutex_destroy(&sync_mutex);
Raymond Knopp's avatar
 
Raymond Knopp committed
2631
#endif
2632 2633
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2634
#ifndef USRP
Raymond Knopp's avatar
 
Raymond Knopp committed
2635 2636 2637 2638
  printf("stopping card\n");
  openair0_stop(card);
  printf("closing openair0_lib\n");
  openair0_close();
Raymond Knopp's avatar
 
Raymond Knopp committed
2639
#endif
2640 2641

#ifdef EMOS
Raymond Knopp's avatar
 
Raymond Knopp committed
2642 2643 2644
  printf("waiting for EMOS thread\n");
  pthread_cancel(thread3);
  pthread_join(thread3,&status);
2645 2646 2647
#endif

#ifdef EMOS
Raymond Knopp's avatar
 
Raymond Knopp committed
2648 2649
  error_code = rtf_destroy(CHANSOUNDER_FIFO_MINOR);
  printf("[OPENAIR][SCHED][CLEANUP] EMOS FIFO closed, error_code %d\n", error_code);
2650 2651
#endif

Raymond Knopp's avatar
 
Raymond Knopp committed
2652 2653
  if (ouput_vcd)
    vcd_signal_dumper_close();
2654

Raymond Knopp's avatar
 
Raymond Knopp committed
2655
  logClean();
2656

Raymond Knopp's avatar
 
Raymond Knopp committed
2657 2658
  return 0;
}
2659

2660 2661
int setup_ue_buffers(PHY_VARS_UE **phy_vars_ue, openair0_config_t *openair0_cfg, openair0_rf_map rf_map[MAX_NUM_CCs])
{
Raymond Knopp's avatar
 
Raymond Knopp committed
2662
#ifndef USRP
2663 2664 2665 2666 2667
  int i, CC_id;
  LTE_DL_FRAME_PARMS *frame_parms;
  for (CC_id=0;CC_id<MAX_NUM_CCs;CC_id++) {
    if (phy_vars_ue[CC_id]) {
      frame_parms = &(phy_vars_ue[CC_id]->lte_frame_parms); 
Raymond Knopp's avatar
 
Raymond Knopp committed
2668
    }
2669 2670 2671
    else {
      printf("phy_vars_eNB[%d] not initialized\n", CC_id);
      return(-1);
Raymond Knopp's avatar
 
Raymond Knopp committed
2672
    }
2673

2674 2675 2676 2677 2678 2679 2680 2681 2682 2683 2684 2685 2686 2687 2688 2689
    openair0_cfg[CC_id].tx_num_channels = 0;
    openair0_cfg[CC_id].rx_num_channels = 0;

    // replace RX signal buffers with mmaped HW versions
    for (i=0;i<frame_parms->nb_antennas_rx;i++) {
      printf("Mapping eNB CC_id %d, rx_ant %d, freq %u on card %d, chain %d\n",CC_id,i,downlink_frequency[CC_id][i],rf_map[CC_id].card,rf_map[CC_id].chain+i);
      free(phy_vars_ue[CC_id]->lte_ue_common_vars.rxdata[i]);
      phy_vars_ue[CC_id]->lte_ue_common_vars.rxdata[i] = (int32_t*) openair0_exmimo_pci[rf_map[CC_id].card].adc_head[rf_map[CC_id].chain+i];
      if (openair0_cfg[rf_map[CC_id].card].rx_freq[rf_map[CC_id].chain+i]) {
	printf("Error with rf_map! A channel has already been allocated!\n");
	return(-1);
      }
      else {
	openair0_cfg[rf_map[CC_id].card].rx_freq[rf_map[CC_id].chain+i] = downlink_frequency[CC_id][i];
	openair0_cfg[rf_map[CC_id].card].rx_num_channels++;
      }
2690

2691
      printf("rxdata[%d] @ %p\n",i,phy_vars_ue[CC_id]->lte_ue_common_vars.rxdata[i]);
Raymond Knopp's avatar
 
Raymond Knopp committed
2692
    }
2693 2694 2695 2696 2697 2698 2699 2700 2701 2702 2703 2704
    for (i=0;i<frame_parms->nb_antennas_tx;i++) {
      printf("Mapping eNB CC_id %d, tx_ant %d, freq %u on card %d, chain %d\n",CC_id,i,downlink_frequency[CC_id][i],rf_map[CC_id].card,rf_map[CC_id].chain+i);
      free(phy_vars_ue[CC_id]->lte_ue_common_vars.txdata[i]);
      phy_vars_ue[CC_id]->lte_ue_common_vars.txdata[i] = (int32_t*) openair0_exmimo_pci[rf_map[CC_id].card].dac_head[rf_map[CC_id].chain+i];
      if (openair0_cfg[rf_map[CC_id].card].tx_freq[rf_map[CC_id].chain+i]) {
	printf("Error with rf_map! A channel has already been allocated!\n");
	return(-1);
      }
      else {
	openair0_cfg[rf_map[CC_id].card].tx_freq[rf_map[CC_id].chain+i] = downlink_frequency[CC_id][i]+uplink_frequency_offset[CC_id][i];
	openair0_cfg[rf_map[CC_id].card].tx_num_channels++;
      }
2705

2706
      printf("txdata[%d] @ %p\n",i,phy_vars_ue[CC_id]->lte_ue_common_vars.txdata[i]);
2707 2708
    }
  }
2709
  return(0);
Raymond Knopp's avatar
 
Raymond Knopp committed
2710 2711
#else
  printf("USRP not supported for UE yet!");
2712
  return(-1);
Raymond Knopp's avatar
 
Raymond Knopp committed
2713
#endif
Raymond Knopp's avatar
 
Raymond Knopp committed
2714
}
2715

2716 2717 2718 2719 2720
/* this function maps the phy_vars_eNB tx and rx buffers to the available rf chains. 
   Each rf chain is is addressed by the card number and the chain on the card. The 
   rf_map specifies for each CC, on which rf chain the mapping should start. Multiple 
   antennas are mapped to successive RF chains on the same card. */
int setup_eNB_buffers(PHY_VARS_eNB **phy_vars_eNB, openair0_config_t *openair0_cfg, openair0_rf_map rf_map[MAX_NUM_CCs]) {
2721

2722
  int i, CC_id;
Raymond Knopp's avatar
 
Raymond Knopp committed
2723 2724
#ifdef USRP
  uint16_t N_TA_offset = 0;
Raymond Knopp's avatar
 
Raymond Knopp committed
2725 2726
#else
  int j;
Raymond Knopp's avatar
 
Raymond Knopp committed
2727
#endif
2728
  LTE_DL_FRAME_PARMS *frame_parms;
2729

2730 2731 2732
  for (CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
    if (phy_vars_eNB[CC_id]) {
      frame_parms = &(phy_vars_eNB[CC_id]->lte_frame_parms); 
Raymond Knopp's avatar
 
Raymond Knopp committed
2733
    }
2734 2735 2736
    else {
      printf("phy_vars_eNB[%d] not initialized\n", CC_id);
      return(-1);
Raymond Knopp's avatar
 
Raymond Knopp committed
2737
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2738

Raymond Knopp's avatar
 
Raymond Knopp committed
2739
#ifdef USRP
2740 2741
    if (frame_parms->frame_type == TDD) {
      if (frame_parms->N_RB_DL == 100)
Raymond Knopp's avatar
 
Raymond Knopp committed
2742
	N_TA_offset = 624;
2743
      else if (frame_parms->N_RB_DL == 50)
Raymond Knopp's avatar
 
Raymond Knopp committed
2744
	N_TA_offset = 624/2;
2745
      else if (frame_parms->N_RB_DL == 25)
Raymond Knopp's avatar
 
Raymond Knopp committed
2746 2747
	N_TA_offset = 624/4;
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2748
#endif
2749 2750 2751 2752 2753 2754 2755
   

    openair0_cfg[CC_id].tx_num_channels = 0;
    openair0_cfg[CC_id].rx_num_channels = 0;
   

  
Raymond Knopp's avatar
 
Raymond Knopp committed
2756
    // replace RX signal buffers with mmaped HW versions
Raymond Knopp's avatar
 
Raymond Knopp committed
2757
#ifndef USRP
2758 2759 2760 2761 2762 2763 2764 2765 2766 2767 2768 2769 2770
    for (i=0;i<frame_parms->nb_antennas_rx;i++) {
      printf("Mapping eNB CC_id %d, rx_ant %d, freq %u on card %d, chain %d\n",CC_id,i,downlink_frequency[CC_id][i]+uplink_frequency_offset[CC_id][i],rf_map[CC_id].card,rf_map[CC_id].chain+i);
      free(phy_vars_eNB[CC_id]->lte_eNB_common_vars.rxdata[0][i]);
      phy_vars_eNB[CC_id]->lte_eNB_common_vars.rxdata[0][i] = (int32_t*) openair0_exmimo_pci[rf_map[CC_id].card].adc_head[rf_map[CC_id].chain+i];
      if (openair0_cfg[rf_map[CC_id].card].rx_freq[rf_map[CC_id].chain+i]) {
	printf("Error with rf_map! A channel has already been allocated!\n");
	return(-1);
      }
      else {
	openair0_cfg[rf_map[CC_id].card].rx_freq[rf_map[CC_id].chain+i] = downlink_frequency[CC_id][i]+uplink_frequency_offset[CC_id][i];
	openair0_cfg[rf_map[CC_id].card].rx_num_channels++;
      }
      printf("rxdata[%d] @ %p\n",i,phy_vars_eNB[CC_id]->lte_eNB_common_vars.rxdata[0][i]);
Raymond Knopp's avatar
 
Raymond Knopp committed
2771
      for (j=0;j<16;j++) {
2772 2773
	printf("rxbuffer %d: %x\n",j,phy_vars_eNB[CC_id]->lte_eNB_common_vars.rxdata[0][i][j]);
	phy_vars_eNB[CC_id]->lte_eNB_common_vars.rxdata[0][i][j] = 16-j;
2774
      }
Raymond Knopp's avatar
 
Raymond Knopp committed
2775
    }
2776 2777 2778 2779 2780 2781 2782 2783 2784 2785 2786 2787 2788 2789
    for (i=0;i<frame_parms->nb_antennas_tx;i++) {
      printf("Mapping eNB CC_id %d, tx_ant %d, freq %u on card %d, chain %d\n",CC_id,i,downlink_frequency[CC_id][i],rf_map[CC_id].card,rf_map[CC_id].chain+i);
      free(phy_vars_eNB[CC_id]->lte_eNB_common_vars.txdata[0][i]);
      phy_vars_eNB[CC_id]->lte_eNB_common_vars.txdata[0][i] = (int32_t*) openair0_exmimo_pci[rf_map[CC_id].card].dac_head[rf_map[CC_id].chain+i];
      if (openair0_cfg[rf_map[CC_id].card].tx_freq[rf_map[CC_id].chain+i]) {
	printf("Error with rf_map! A channel has already been allocated!\n");
	return(-1);
      }
      else {
	openair0_cfg[rf_map[CC_id].card].tx_freq[rf_map[CC_id].chain+i] = downlink_frequency[CC_id][i];
	openair0_cfg[rf_map[CC_id].card].tx_num_channels++;
      }
      
      printf("txdata[%d] @ %p\n",i,phy_vars_eNB[CC_id]->lte_eNB_common_vars.txdata[0][i]);
Raymond Knopp's avatar
 
Raymond Knopp committed
2790
      for (j=0;j<16;j++) {
2791 2792
	printf("txbuffer %d: %x\n",j,phy_vars_eNB[CC_id]->lte_eNB_common_vars.txdata[0][i][j]);
	phy_vars_eNB[CC_id]->lte_eNB_common_vars.txdata[0][i][j] = 16-j;
2793
      }
2794
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
2795
#else // USRP
2796 2797 2798 2799 2800 2801
    for (i=0;i<frame_parms->nb_antennas_rx;i++) {
      free(phy_vars_eNB[CC_id]->lte_eNB_common_vars.rxdata[0][i]);
      rxdata = (int32_t*)malloc16(samples_per_frame*sizeof(int32_t));
      phy_vars_eNB[CC_id]->lte_eNB_common_vars.rxdata[0][i] = rxdata-N_TA_offset; // N_TA offset for TDD
      memset(rxdata, 0, samples_per_frame*sizeof(int32_t));
      printf("rxdata[%d] @ %p (%p)\n", i, phy_vars_eNB[CC_id]->lte_eNB_common_vars.rxdata[0][i],rxdata);
Raymond Knopp's avatar
 
Raymond Knopp committed
2802
    }
2803 2804 2805 2806 2807 2808
    for (i=0;i<frame_parms->nb_antennas_tx;i++) {
      free(phy_vars_eNB[CC_id]->lte_eNB_common_vars.txdata[0][i]);
      txdata = (int32_t*)malloc16(samples_per_frame*sizeof(int32_t));
      phy_vars_eNB[CC_id]->lte_eNB_common_vars.txdata[0][i] = txdata;
      memset(txdata, 0, samples_per_frame*sizeof(int32_t));
      printf("txdata[%d] @ %p\n", i, phy_vars_eNB[CC_id]->lte_eNB_common_vars.txdata[0][i]);
Raymond Knopp's avatar
 
Raymond Knopp committed
2809 2810
    }
#endif
2811
  }
2812
  return(0);
Raymond Knopp's avatar
 
Raymond Knopp committed
2813
}
2814