lte-enb.c 47.7 KB
Newer Older
1 2 3 4 5
/*
 * Licensed to the OpenAirInterface (OAI) Software Alliance under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The OpenAirInterface Software Alliance licenses this file to You under
6
 * the OAI Public License, Version 1.1  (the "License"); you may not use this file
7 8 9 10 11 12 13 14 15 16 17 18 19 20
 * except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.openairinterface.org/?page_id=698
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 *-------------------------------------------------------------------------------
 * For more information about the OpenAirInterface (OAI) Software Alliance:
 *      contact@openairinterface.org
 */
21 22 23 24 25 26 27 28 29 30 31

/*! \file lte-enb.c
 * \brief Top-level threads for eNodeB
 * \author R. Knopp, F. Kaltenberger, Navid Nikaein
 * \date 2012
 * \version 0.1
 * \company Eurecom
 * \email: knopp@eurecom.fr,florian.kaltenberger@eurecom.fr, navid.nikaein@eurecom.fr
 * \note
 * \warning
 */
32

Raymond Knopp's avatar
Raymond Knopp committed
33 34
#define _GNU_SOURCE
#include <pthread.h>
35

36

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

39 40
#include "rt_wrapper.h"

41
#include "assertions.h"
42

43 44 45

#include "PHY/types.h"

46 47 48 49 50 51
#include "PHY/INIT/phy_init.h"

#include "PHY/defs_eNB.h"
#include "SCHED/sched_eNB.h"
#include "PHY/LTE_TRANSPORT/transport_proto.h"

52 53 54 55 56 57 58
#undef MALLOC //there are two conflicting definitions, so we better make sure we don't use it at all
//#undef FRAME_LENGTH_COMPLEX_SAMPLES //there are two conflicting definitions, so we better make sure we don't use it at all

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

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

59
#include "PHY/LTE_TRANSPORT/if4_tools.h"
60
#include "PHY/LTE_TRANSPORT/if5_tools.h"
61

62 63 64 65 66 67 68 69
#include "PHY/phy_extern.h"


#include "LAYER2/MAC/mac.h"
#include "LAYER2/MAC/mac_extern.h"
#include "LAYER2/MAC/mac_proto.h"
#include "RRC/LTE/rrc_extern.h"
#include "PHY_INTERFACE/phy_interface.h"
70
#include "common/utils/LOG/log.h"
71 72 73
#include "UTIL/OTG/otg_tx.h"
#include "UTIL/OTG/otg_externs.h"
#include "UTIL/MATH/oml.h"
74
#include "common/utils/LOG/vcd_signal_dumper.h"
75 76
#include "UTIL/OPT/opt.h"
#include "enb_config.h"
77

78 79

#ifndef OPENAIR2
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
80
  #include "UTIL/OTG/otg_extern.h"
81 82 83
#endif

#if defined(ENABLE_ITTI)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
84 85 86 87 88 89
  #if defined(ENABLE_USE_MME)
    #include "s1ap_eNB.h"
    #ifdef PDCP_USE_NETLINK
      #include "SIMULATION/ETH_TRANSPORT/proto.h"
    #endif
  #endif
90 91
#endif

Rohit Gupta's avatar
Rohit Gupta committed
92 93
#include "T.h"

94 95 96 97 98 99 100 101 102 103
//#define DEBUG_THREADS 1

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

104 105
// Fix per CC openair rf/if device update
// extern openair0_device openair0;
106

107

108
#if defined(ENABLE_ITTI)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
109 110
  extern volatile int             start_eNB;
  extern volatile int             start_UE;
111 112 113
#endif
extern volatile int                    oai_exit;

Cedric Roux's avatar
Cedric Roux committed
114
extern int transmission_mode;
115

116 117
extern int oaisim_flag;

118 119 120
//uint16_t sf_ahead=4;
extern uint16_t sf_ahead;

121

122
//pthread_t                       main_eNB_thread;
123 124 125

time_stats_t softmodem_stats_mt; // main thread
time_stats_t softmodem_stats_hw; //  hw acquisition
126
time_stats_t softmodem_stats_rxtx_sf; // total tx time
127
time_stats_t nfapi_meas; // total tx time
128
time_stats_t softmodem_stats_rx_sf; // total rx time
129 130 131 132 133 134 135 136 137

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

140 141
extern double cpuf;

142

143
void init_eNB(int,int);
144
void stop_eNB(int nb_inst);
145

Wang Tsu-Han's avatar
Wang Tsu-Han committed
146
int wakeup_tx(PHY_VARS_eNB *eNB);
147
int wakeup_txfh(L1_rxtx_proc_t *proc,PHY_VARS_eNB *eNB);
148
void wakeup_prach_eNB(PHY_VARS_eNB *eNB,RU_t *ru,int frame,int subframe);
149
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
150
  void wakeup_prach_eNB_br(PHY_VARS_eNB *eNB,RU_t *ru,int frame,int subframe);
151
#endif
152

153 154 155 156 157 158
extern uint8_t nfapi_mode;
extern void oai_subframe_ind(uint16_t sfn, uint16_t sf);
extern void add_subframe(uint16_t *frameP, uint16_t *subframeP, int offset);

//#define TICK_TO_US(ts) (ts.diff)
#define TICK_TO_US(ts) (ts.trials==0?0:ts.diff/ts.trials)
Raymond Knopp's avatar
Raymond Knopp committed
159

160

161
static inline int rxtx(PHY_VARS_eNB *eNB,L1_rxtx_proc_t *proc, char *thread_name) {
Raymond Knopp's avatar
Raymond Knopp committed
162
  start_meas(&softmodem_stats_rxtx_sf);
163
  //L1_rxtx_proc_t *L1_proc_tx = &eNB->proc.L1_proc_tx;
164

165 166
  // *******************************************************************

167
#if defined(PRE_SCD_THREAD)
168
    RU_t *ru = RC.ru[0];
169
#endif
170

171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187
  if (nfapi_mode == 1) {
    // I am a PNF and I need to let nFAPI know that we have a (sub)frame tick
    uint16_t frame = proc->frame_rx;
    uint16_t subframe = proc->subframe_rx;
    //add_subframe(&frame, &subframe, 4);
    //oai_subframe_ind(proc->frame_tx, proc->subframe_tx);
    //LOG_D(PHY, "oai_subframe_ind(frame:%u, subframe:%d) - NOT CALLED ********\n", frame, subframe);
    start_meas(&nfapi_meas);
    oai_subframe_ind(frame, subframe);
    stop_meas(&nfapi_meas);

    if (eNB->UL_INFO.rx_ind.rx_indication_body.number_of_pdus||
        eNB->UL_INFO.harq_ind.harq_indication_body.number_of_harqs ||
        eNB->UL_INFO.crc_ind.crc_indication_body.number_of_crcs ||
        eNB->UL_INFO.rach_ind.rach_indication_body.number_of_preambles ||
        eNB->UL_INFO.cqi_ind.number_of_cqis
       ) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
188 189 190 191 192 193 194 195
      LOG_D(PHY, "UL_info[rx_ind:%05d:%d harqs:%05d:%d crcs:%05d:%d preambles:%05d:%d cqis:%d] RX:%04d%d TX:%04d%d num_pdcch_symbols:%d\n",
            NFAPI_SFNSF2DEC(eNB->UL_INFO.rx_ind.sfn_sf),   eNB->UL_INFO.rx_ind.rx_indication_body.number_of_pdus,
            NFAPI_SFNSF2DEC(eNB->UL_INFO.harq_ind.sfn_sf), eNB->UL_INFO.harq_ind.harq_indication_body.number_of_harqs,
            NFAPI_SFNSF2DEC(eNB->UL_INFO.crc_ind.sfn_sf),  eNB->UL_INFO.crc_ind.crc_indication_body.number_of_crcs,
            NFAPI_SFNSF2DEC(eNB->UL_INFO.rach_ind.sfn_sf), eNB->UL_INFO.rach_ind.rach_indication_body.number_of_preambles,
            eNB->UL_INFO.cqi_ind.number_of_cqis,
            proc->frame_rx, proc->subframe_rx,
            proc->frame_tx, proc->subframe_tx, eNB->pdcch_vars[proc->subframe_tx&1].num_pdcch_symbols);
196 197 198 199 200 201 202 203
    }
  }

  if (nfapi_mode == 1 && eNB->pdcch_vars[proc->subframe_tx&1].num_pdcch_symbols == 0) {
    LOG_E(PHY, "eNB->pdcch_vars[proc->subframe_tx&1].num_pdcch_symbols == 0");
    return 0;
  }

Raymond Knopp's avatar
Raymond Knopp committed
204 205
  // ****************************************
  // Common RX procedures subframe n
Cedric Roux's avatar
Cedric Roux committed
206 207
  T(T_ENB_PHY_DL_TICK, T_INT(eNB->Mod_id), T_INT(proc->frame_tx), T_INT(proc->subframe_tx));

208
  // if this is IF5 or 3GPP_eNB
209
  if (eNB && eNB->RU_list && eNB->RU_list[0] && eNB->RU_list[0]->function < NGFI_RAU_IF4p5) {
210
    wakeup_prach_eNB(eNB,NULL,proc->frame_rx,proc->subframe_rx);
211
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
212 213 214
    wakeup_prach_eNB_br(eNB,NULL,proc->frame_rx,proc->subframe_rx);
#endif
  }
Wang Tsu-Han's avatar
Wang Tsu-Han committed
215 216 217

  release_UE_in_freeList(eNB->Mod_id);

Raymond Knopp's avatar
Raymond Knopp committed
218
  // UE-specific RX processing for subframe n
219
  if (nfapi_mode == 0 || nfapi_mode == 1) {
220
    phy_procedures_eNB_uespec_RX(eNB, proc);
221
  }
222

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
223 224
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_ENB_DLSCH_ULSCH_SCHEDULER, 1 );

225 226 227 228
#if defined(PRE_SCD_THREAD)
    if (nfapi_mode == 2){
      new_dlsch_ue_select_tbl_in_use = dlsch_ue_select_tbl_in_use;
      dlsch_ue_select_tbl_in_use = !dlsch_ue_select_tbl_in_use;
229 230 231 232 233 234 235

// L2-emulator can work only one eNB.
//      memcpy(&pre_scd_eNB_UE_stats,&RC.mac[ru->eNB_list[0]->Mod_id]->UE_list.eNB_UE_stats, sizeof(eNB_UE_STATS)*MAX_NUM_CCs*NUMBER_OF_UE_MAX);
//      memcpy(&pre_scd_activeUE, &RC.mac[ru->eNB_list[0]->Mod_id]->UE_list.active, sizeof(boolean_t)*NUMBER_OF_UE_MAX);
      memcpy(&pre_scd_eNB_UE_stats,&RC.mac[0]->UE_list.eNB_UE_stats, sizeof(eNB_UE_STATS)*MAX_NUM_CCs*NUMBER_OF_UE_MAX);
      memcpy(&pre_scd_activeUE, &RC.mac[0]->UE_list.active, sizeof(boolean_t)*NUMBER_OF_UE_MAX);

236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259
      if (pthread_mutex_lock(&ru->proc.mutex_pre_scd)!= 0) {
          LOG_E( PHY, "[eNB] error locking proc mutex for eNB pre scd\n");
          exit_fun("error locking mutex_time");
      }

      ru->proc.instance_pre_scd++;

      if (ru->proc.instance_pre_scd == 0) {
          if (pthread_cond_signal(&ru->proc.cond_pre_scd) != 0) {
              LOG_E( PHY, "[eNB] ERROR pthread_cond_signal for eNB pre scd\n" );
              exit_fun( "ERROR pthread_cond_signal cond_pre_scd" );
          }
      }else{
          LOG_E( PHY, "[eNB] frame %d subframe %d rxtx busy instance_pre_scd %d\n",
                 proc->frame_rx,proc->subframe_rx,ru->proc.instance_pre_scd );
      }

      if (pthread_mutex_unlock(&ru->proc.mutex_pre_scd)!= 0) {
          LOG_E( PHY, "[eNB] error unlocking mutex_pre_scd mutex for eNB pre scd\n");
          exit_fun("error unlocking mutex_pre_scd");
      }
    }
#endif

260
  pthread_mutex_lock(&eNB->UL_INFO_mutex);
261 262 263 264 265
  eNB->UL_INFO.frame     = proc->frame_rx;
  eNB->UL_INFO.subframe  = proc->subframe_rx;
  eNB->UL_INFO.module_id = eNB->Mod_id;
  eNB->UL_INFO.CC_id     = eNB->CC_id;
  eNB->if_inst->UL_indication(&eNB->UL_INFO);
266
  pthread_mutex_unlock(&eNB->UL_INFO_mutex);
Wang Tsu-Han's avatar
Wang Tsu-Han committed
267 268
  /* this conflict resolution may be totally wrong, to be tested */
  /* CONFLICT RESOLUTION: BEGIN */
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
269 270
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_ENB_DLSCH_ULSCH_SCHEDULER, 0 );

271
  if(oai_exit) return(-1);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
272 273

  if(get_thread_parallel_conf() == PARALLEL_SINGLE_THREAD) {
Wang Tsu-Han's avatar
Wang Tsu-Han committed
274
#ifndef PHY_TX_THREAD
275
    phy_procedures_eNB_TX(eNB, proc, 1);
Wang Tsu-Han's avatar
Wang Tsu-Han committed
276
#endif
277
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
278

Wang Tsu-Han's avatar
Wang Tsu-Han committed
279 280 281
  /* CONFLICT RESOLUTION: what about this release_thread call, has it to be done? if yes, where? */
  //if (release_thread(&proc->mutex_rxtx,&proc->instance_cnt_rxtx,thread_name)<0) return(-1);
  /* CONFLICT RESOLUTION: END */
Raymond Knopp's avatar
Raymond Knopp committed
282
  stop_meas( &softmodem_stats_rxtx_sf );
283
  LOG_D(PHY,"%s() Exit proc[rx:%d%d tx:%d%d]\n", __FUNCTION__, proc->frame_rx, proc->subframe_rx, proc->frame_tx, proc->subframe_tx);
284
  LOG_D(PHY, "rxtx:%lld nfapi:%lld tx:%lld rx:%lld prach:%lld ofdm:%lld ",
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
285 286 287 288 289 290
        softmodem_stats_rxtx_sf.p_time, nfapi_meas.p_time,
        TICK_TO_US(eNB->phy_proc_tx),
        TICK_TO_US(eNB->phy_proc_rx),
        TICK_TO_US(eNB->rx_prach),
        TICK_TO_US(eNB->ofdm_mod_stats)
       );
291
  LOG_D(PHY,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
292 293 294 295 296 297 298 299
        "dlsch[enc:%lld mod:%lld scr:%lld rm:%lld t:%lld i:%lld] rx_dft:%lld ",
        TICK_TO_US(eNB->dlsch_encoding_stats),
        TICK_TO_US(eNB->dlsch_modulation_stats),
        TICK_TO_US(eNB->dlsch_scrambling_stats),
        TICK_TO_US(eNB->dlsch_rate_matching_stats),
        TICK_TO_US(eNB->dlsch_turbo_encoding_stats),
        TICK_TO_US(eNB->dlsch_interleaving_stats),
        TICK_TO_US(eNB->rx_dft_stats));
300
  LOG_D(PHY," ulsch[ch:%lld freq:%lld dec:%lld demod:%lld ru:%lld ",
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
301 302 303 304 305
        TICK_TO_US(eNB->ulsch_channel_estimation_stats),
        TICK_TO_US(eNB->ulsch_freq_offset_estimation_stats),
        TICK_TO_US(eNB->ulsch_decoding_stats),
        TICK_TO_US(eNB->ulsch_demodulation_stats),
        TICK_TO_US(eNB->ulsch_rate_unmatching_stats));
306
  LOG_D(PHY, "td:%lld dei:%lld dem:%lld llr:%lld tci:%lld ",
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
307 308 309 310 311 312 313 314 315 316 317 318 319
        TICK_TO_US(eNB->ulsch_turbo_decoding_stats),
        TICK_TO_US(eNB->ulsch_deinterleaving_stats),
        TICK_TO_US(eNB->ulsch_demultiplexing_stats),
        TICK_TO_US(eNB->ulsch_llr_stats),
        TICK_TO_US(eNB->ulsch_tc_init_stats));
  LOG_D(PHY, "tca:%lld tcb:%lld tcg:%lld tce:%lld l1:%lld l2:%lld]\n\n",
        TICK_TO_US(eNB->ulsch_tc_alpha_stats),
        TICK_TO_US(eNB->ulsch_tc_beta_stats),
        TICK_TO_US(eNB->ulsch_tc_gamma_stats),
        TICK_TO_US(eNB->ulsch_tc_ext_stats),
        TICK_TO_US(eNB->ulsch_tc_intl1_stats),
        TICK_TO_US(eNB->ulsch_tc_intl2_stats)
       );
Raymond Knopp's avatar
Raymond Knopp committed
320 321 322
  return(0);
}

323

Wang Tsu-Han's avatar
Wang Tsu-Han committed
324
static void* L1_thread_tx(void* param) {
325

326 327
  L1_proc_t *eNB_proc  = (L1_proc_t*)param;
  L1_rxtx_proc_t *proc = &eNB_proc->L1_proc_tx;
328 329
  PHY_VARS_eNB *eNB = RC.eNB[0][proc->CC_id];
  char thread_name[100];
Wang Tsu-Han's avatar
Wang Tsu-Han committed
330
  sprintf(thread_name,"TXnp4_%d\n",&eNB->proc.L1_proc == proc ? 0 : 1);
331
  thread_top_init(thread_name,1,470000,500000,500000);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
332

333
  //wait_sync("tx_thread");
334 335

  while (!oai_exit) {
Wang Tsu-Han's avatar
Wang Tsu-Han committed
336
    if (wait_on_condition(&proc->mutex,&proc->cond,&proc->instance_cnt,thread_name)<0) break;
337
    if (oai_exit) break;  
338 339 340 341 342 343

    // *****************************************
    // TX processing for subframe n+4
    // run PHY TX procedures the one after the other for all CCs to avoid race conditions
    // (may be relaxed in the future for performance reasons)
    // *****************************************
344 345 346 347
    VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_SUBFRAME_NUMBER_TX1_ENB,proc->subframe_tx);
    VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_SUBFRAME_NUMBER_RX1_ENB,proc->subframe_rx);
    VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_FRAME_NUMBER_TX1_ENB,proc->frame_tx);
    VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_FRAME_NUMBER_RX1_ENB,proc->frame_rx);
348
    phy_procedures_eNB_TX(eNB, proc, 1);
349

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
350

Wang Tsu-Han's avatar
Wang Tsu-Han committed
351
    pthread_mutex_lock( &proc->mutex );
352
    proc->instance_cnt = -1;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
353

354
    // the thread can now be woken up
Wang Tsu-Han's avatar
Wang Tsu-Han committed
355
    if (pthread_cond_signal(&proc->cond) != 0) {
356 357 358
      LOG_E( PHY, "[eNB] ERROR pthread_cond_signal for eNB TXnp4 thread\n");
      exit_fun( "ERROR pthread_cond_signal" );
    }
Wang Tsu-Han's avatar
Wang Tsu-Han committed
359
    pthread_mutex_unlock( &proc->mutex );
Wang Tsu-Han's avatar
Wang Tsu-Han committed
360
    wakeup_txfh(proc,eNB);
361 362 363 364 365
  }

  return 0;
}

366
/*!
367
 * \brief The RX UE-specific and TX thread of eNB.
368
 * \param param is a \ref L1_proc_t structure which contains the info what to process.
369 370
 * \returns a pointer to an int. The storage is not on the heap and must not be freed.
 */
371

Wang Tsu-Han's avatar
Wang Tsu-Han committed
372
static void* L1_thread( void* param ) {
373

374
  static int eNB_thread_rxtx_status;
375 376
  //L1_proc_t *eNB_proc  = (L1_proc_t*)param;
  L1_rxtx_proc_t *proc;
377 378 379

  // Working
  if(nfapi_mode ==2){
380
    proc = (L1_rxtx_proc_t*)param;
381 382
  }
  else{
383 384
    L1_proc_t *eNB_proc  = (L1_proc_t*)param;
    proc = &eNB_proc->L1_proc;
385 386
  }

387
  PHY_VARS_eNB *eNB = RC.eNB[0][proc->CC_id];
388

Raymond Knopp's avatar
Raymond Knopp committed
389
  char thread_name[100];
390 391
  cpu_set_t cpuset;
  CPU_ZERO(&cpuset);
392
  // set default return value
393
  eNB_thread_rxtx_status = 0;
394

Wang Tsu-Han's avatar
Wang Tsu-Han committed
395
  sprintf(thread_name,"RXn_TXnp4_%d\n",&eNB->proc.L1_proc == proc ? 0 : 1);
396 397 398
  thread_top_init(thread_name,1,470000,500000,500000);
  pthread_setname_np( pthread_self(),"rxtx processing");
  LOG_I(PHY,"thread rxtx created id=%ld\n", syscall(__NR_gettid));
399 400

  while (!oai_exit) {
401
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_RXTX0+(proc->subframe_rx&1), 0 );
402
    T(T_ENB_MASTER_TICK, T_INT(0), T_INT(proc->frame_rx), T_INT(proc->subframe_rx));
403

Wang Tsu-Han's avatar
Wang Tsu-Han committed
404
    if (wait_on_condition(&proc->mutex,&proc->cond,&proc->instance_cnt,thread_name)<0) break;
405

406
    VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_CPUID_ENB_THREAD_RXTX,sched_getcpu());
407
    VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_RXTX0+(proc->subframe_rx&1), 1 );
408 409 410 411
    VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_SUBFRAME_NUMBER_TX0_ENB,proc->subframe_tx);
    VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_SUBFRAME_NUMBER_RX0_ENB,proc->subframe_rx);
    VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_FRAME_NUMBER_TX0_ENB,proc->frame_tx);
    VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_FRAME_NUMBER_RX0_ENB,proc->frame_rx);
412

413 414
    if (oai_exit) break;

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
415
    if (eNB->CC_id==0) {
416
      if (rxtx(eNB,proc,thread_name) < 0) break;
417
    }
418

419
    if(get_thread_parallel_conf() == PARALLEL_RU_L1_SPLIT)              phy_procedures_eNB_TX(eNB, proc, 1);
Wang Tsu-Han's avatar
Wang Tsu-Han committed
420
    if (release_thread(&proc->mutex,&proc->instance_cnt,thread_name)<0) break;
Wang Tsu-Han's avatar
Wang Tsu-Han committed
421
    if (nfapi_mode!=2){
Wang Tsu-Han's avatar
Wang Tsu-Han committed
422
    	if(get_thread_parallel_conf() == PARALLEL_RU_L1_TRX_SPLIT)      wakeup_tx(eNB);
423
    	else if(get_thread_parallel_conf() == PARALLEL_RU_L1_SPLIT)     wakeup_txfh(proc,eNB);
424
    }
Raymond Knopp's avatar
Raymond Knopp committed
425
  } // while !oai_exit
426

427
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME( VCD_SIGNAL_DUMPER_FUNCTIONS_eNB_PROC_RXTX0+(proc->subframe_rx&1), 0 );
428
  LOG_D(PHY, " *** Exiting eNB thread RXn_TXnp4\n");
429 430
  eNB_thread_rxtx_status = 0;
  return &eNB_thread_rxtx_status;
431 432
}

433
void eNB_top(PHY_VARS_eNB *eNB, int frame_rx, int subframe_rx, char *string,RU_t *ru)
Cedric Roux's avatar
Cedric Roux committed
434
{
435 436
  L1_proc_t *proc           = &eNB->proc;
  L1_rxtx_proc_t *L1_proc = &proc->L1_proc;
Raymond Knopp's avatar
Raymond Knopp committed
437 438
  LTE_DL_FRAME_PARMS *fp = &ru->frame_parms;
  RU_proc_t *ru_proc=&ru->proc;
439 440
  proc->frame_rx    = frame_rx;
  proc->subframe_rx = subframe_rx;
Raymond Knopp's avatar
Raymond Knopp committed
441

Cedric Roux's avatar
Cedric Roux committed
442
  if (!oai_exit) {
443
    T(T_ENB_MASTER_TICK, T_INT(0), T_INT(proc->frame_rx), T_INT(proc->subframe_rx));
Raymond Knopp's avatar
Raymond Knopp committed
444

Wang Tsu-Han's avatar
Wang Tsu-Han committed
445 446 447 448 449
    L1_proc->timestamp_tx = ru_proc->timestamp_rx + (sf_ahead*fp->samples_per_tti);
    L1_proc->frame_rx     = ru_proc->frame_rx;
    L1_proc->subframe_rx  = ru_proc->subframe_rx;
    L1_proc->frame_tx     = (L1_proc->subframe_rx > (9-sf_ahead)) ? (L1_proc->frame_rx+1)&1023 : L1_proc->frame_rx;
    L1_proc->subframe_tx  = (L1_proc->subframe_rx + sf_ahead)%10;
Raymond Knopp's avatar
Raymond Knopp committed
450

Wang Tsu-Han's avatar
Wang Tsu-Han committed
451 452 453 454
    if (rxtx(eNB,L1_proc,string) < 0) LOG_E(PHY,"eNB %d CC_id %d failed during execution\n",eNB->Mod_id,eNB->CC_id);
    ru_proc->timestamp_tx = L1_proc->timestamp_tx;
    ru_proc->subframe_tx  = L1_proc->subframe_tx;
    ru_proc->frame_tx     = L1_proc->frame_tx;
455
  }
Raymond Knopp's avatar
Raymond Knopp committed
456 457
}

458
int wakeup_txfh(L1_rxtx_proc_t *proc,PHY_VARS_eNB *eNB) {
Wang Tsu-Han's avatar
Wang Tsu-Han committed
459 460 461 462 463

  RU_t *ru;
  RU_proc_t *ru_proc;


464 465 466
  struct timespec wait;
  wait.tv_sec=0;
  wait.tv_nsec=5000000L;
Wang Tsu-Han's avatar
Wang Tsu-Han committed
467

468
  if(wait_on_condition(&proc->mutex_RUs,&proc->cond_RUs,&proc->instance_cnt_RUs,"wakeup_txfh")<0) {
Wang Tsu-Han's avatar
Wang Tsu-Han committed
469
    LOG_E(PHY,"Frame %d, subframe %d: TX FH not ready\n", proc->frame_tx, proc->subframe_tx);
470 471
    return(-1);
  }
Wang Tsu-Han's avatar
Wang Tsu-Han committed
472 473 474
  pthread_mutex_lock(&eNB->proc.mutex_RU_tx);
  eNB->proc.RU_mask_tx = 0;
  pthread_mutex_unlock(&eNB->proc.mutex_RU_tx);
475
  if (release_thread(&proc->mutex_RUs,&proc->instance_cnt_RUs,"wakeup_txfh")<0) return(-1);
476

Wang Tsu-Han's avatar
Wang Tsu-Han committed
477 478 479 480 481 482 483 484 485 486 487 488 489
  for(int i=0; i<eNB->num_RU; i++)
  {
    ru      = eNB->RU_list[i];
    ru_proc = &ru->proc;
    if (ru_proc->instance_cnt_eNBs == 0) {
      LOG_E(PHY,"Frame %d, subframe %d: TX FH thread busy, dropping Frame %d, subframe %d\n", ru_proc->frame_tx, ru_proc->subframe_tx, proc->frame_rx, proc->subframe_rx);
      return(-1);
    }
    if (pthread_mutex_timedlock(&ru_proc->mutex_eNBs,&wait) != 0) {
      LOG_E( PHY, "[eNB] ERROR pthread_mutex_lock for eNB TX1 thread %d (IC %d)\n", ru_proc->subframe_rx&1,ru_proc->instance_cnt_eNBs );
      exit_fun( "error locking mutex_eNB" );
      return(-1);
    }
490

491
    ru_proc->instance_cnt_eNBs = 0;
Wang Tsu-Han's avatar
Wang Tsu-Han committed
492 493 494
    ru_proc->timestamp_tx = proc->timestamp_tx;
    ru_proc->subframe_tx  = proc->subframe_tx;
    ru_proc->frame_tx     = proc->frame_tx;
495

496
  
Wang Tsu-Han's avatar
Wang Tsu-Han committed
497 498 499 500 501 502
    // the thread can now be woken up
    if (pthread_cond_signal(&ru_proc->cond_eNBs) != 0) {
      LOG_E( PHY, "[eNB] ERROR pthread_cond_signal for eNB TXnp4 thread\n");
      exit_fun( "ERROR pthread_cond_signal" );
      return(-1);
    }
503
  
Wang Tsu-Han's avatar
Wang Tsu-Han committed
504 505
    pthread_mutex_unlock( &ru_proc->mutex_eNBs );
  }
506 507 508 509

  return(0);
}

Wang Tsu-Han's avatar
Wang Tsu-Han committed
510
int wakeup_tx(PHY_VARS_eNB *eNB) {
511

512
  L1_proc_t *proc=&eNB->proc;
513

514 515
  L1_rxtx_proc_t *L1_proc_tx = &proc->L1_proc_tx;
  L1_rxtx_proc_t *L1_proc    = &proc->L1_proc;
516 517 518 519 520 521

  
  struct timespec wait;
  wait.tv_sec=0;
  wait.tv_nsec=5000000L;
  
522
  
Wang Tsu-Han's avatar
Wang Tsu-Han committed
523
  if (pthread_mutex_timedlock(&L1_proc_tx->mutex,&wait) != 0) {
524 525
    LOG_E(PHY, "[SCHED][eNB] ERROR locking mutex for eNB L1_thread_tx\n");
    exit_fun("ERROR pthread_lock");
526 527
    return(-1);
  }
528 529
  while(L1_proc_tx->instance_cnt == 0){
    pthread_cond_wait(&L1_proc_tx->cond,&L1_proc_tx->mutex);
530 531
  }

532
  L1_proc_tx->instance_cnt = 0;
533 534

  
Wang Tsu-Han's avatar
Wang Tsu-Han committed
535 536 537 538 539
  L1_proc_tx->subframe_rx   = L1_proc->subframe_rx;
  L1_proc_tx->frame_rx      = L1_proc->frame_rx;
  L1_proc_tx->subframe_tx   = L1_proc->subframe_tx;
  L1_proc_tx->frame_tx      = L1_proc->frame_tx;
  L1_proc_tx->timestamp_tx  = L1_proc->timestamp_tx;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
540

541
  // the thread can now be woken up
Wang Tsu-Han's avatar
Wang Tsu-Han committed
542
  if (pthread_cond_signal(&L1_proc_tx->cond) != 0) {
543
    LOG_E( PHY, "[eNB] ERROR pthread_cond_signal for eNB TXnp4 thread\n");
544 545 546 547
    exit_fun( "ERROR pthread_cond_signal" );
    return(-1);
  }
  
Wang Tsu-Han's avatar
Wang Tsu-Han committed
548
  pthread_mutex_unlock( &L1_proc_tx->mutex);
549 550
  return(0);
}
Raymond Knopp's avatar
Raymond Knopp committed
551

552 553
int wakeup_rxtx(PHY_VARS_eNB *eNB,RU_t *ru) {

554
  L1_proc_t *proc=&eNB->proc;
555
  RU_proc_t *ru_proc=&ru->proc;
556

557
  L1_rxtx_proc_t *L1_proc=&proc->L1_proc;
558
  
559
  LTE_DL_FRAME_PARMS *fp = &eNB->frame_parms;
560 561
  int i;
  struct timespec wait;
562
  pthread_mutex_lock(&proc->mutex_RU);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
563 564

  for (i=0; i<eNB->num_RU; i++) {
565 566
    if (ru == eNB->RU_list[i]) {
      if ((proc->RU_mask&(1<<i)) > 0)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
567 568 569
        LOG_E(PHY,"eNB %d frame %d, subframe %d : previous information from RU %d (num_RU %d,mask %x) has not been served yet!\n",
              eNB->Mod_id,proc->frame_rx,proc->subframe_rx,ru->idx,eNB->num_RU,proc->RU_mask);

570 571 572
      proc->RU_mask |= (1<<i);
    }
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
573

574
  if (proc->RU_mask != (1<<eNB->num_RU)-1) {  // not all RUs have provided their information so return
575
    LOG_E(PHY,"Not all RUs have provided their info\n");
576 577
    pthread_mutex_unlock(&proc->mutex_RU);
    return(0);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
578
  } else { // all RUs have provided their information so continue on and wakeup eNB processing
579 580 581 582
    proc->RU_mask = 0;
    pthread_mutex_unlock(&proc->mutex_RU);
  }

583 584
  wait.tv_sec=0;
  wait.tv_nsec=5000000L;
585

Wang Tsu-Han's avatar
Wang Tsu-Han committed
586 587 588
    
  if (L1_proc->instance_cnt == 0) {
    LOG_E(PHY,"Frame %d, subframe %d: RXTX0 thread busy, dropping\n",L1_proc->frame_rx,L1_proc->subframe_rx);
589 590 591
    return(-1);
  }

Cedric Roux's avatar
Cedric Roux committed
592
  // wake up TX for subframe n+sf_ahead
593
  // lock the TX mutex and make sure the thread is ready
Wang Tsu-Han's avatar
Wang Tsu-Han committed
594 595
  if (pthread_mutex_timedlock(&L1_proc->mutex,&wait) != 0) {
    LOG_E( PHY, "[eNB] ERROR pthread_mutex_lock for eNB RXTX thread %d (IC %d)\n", L1_proc->subframe_rx&1,L1_proc->instance_cnt );
596 597 598
    exit_fun( "error locking mutex_rxtx" );
    return(-1);
  }
599

Wang Tsu-Han's avatar
Wang Tsu-Han committed
600
  ++L1_proc->instance_cnt;
601 602 603 604
  
  // We have just received and processed the common part of a subframe, say n. 
  // TS_rx is the last received timestamp (start of 1st slot), TS_tx is the desired 
  // transmitted timestamp of the next TX slot (first).
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
605
  // The last (TS_rx mod samples_per_frame) was n*samples_per_tti,
606 607
  // we want to generate subframe (n+sf_ahead), so TS_tx = TX_rx+sf_ahead*samples_per_tti,
  // and proc->subframe_tx = proc->subframe_rx+sf_ahead
Wang Tsu-Han's avatar
Wang Tsu-Han committed
608 609 610 611 612
  L1_proc->timestamp_tx = ru_proc->timestamp_rx + (sf_ahead*fp->samples_per_tti);
  L1_proc->frame_rx     = ru_proc->frame_rx;
  L1_proc->subframe_rx  = ru_proc->subframe_rx;
  L1_proc->frame_tx     = (L1_proc->subframe_rx > (9-sf_ahead)) ? (L1_proc->frame_rx+1)&1023 : L1_proc->frame_rx;
  L1_proc->subframe_tx  = (L1_proc->subframe_rx + sf_ahead)%10;
613

614
  // the thread can now be woken up
Wang Tsu-Han's avatar
Wang Tsu-Han committed
615
  if (pthread_cond_signal(&L1_proc->cond) != 0) {
616 617 618 619 620
    LOG_E( PHY, "[eNB] ERROR pthread_cond_signal for eNB RXn-TXnp4 thread\n");
    exit_fun( "ERROR pthread_cond_signal" );
    return(-1);
  }

621
  pthread_mutex_unlock( &L1_proc->mutex);
622 623 624
  return(0);
}

625
void wakeup_prach_eNB(PHY_VARS_eNB *eNB,RU_t *ru,int frame,int subframe) {
626

627
  L1_proc_t *proc = &eNB->proc;
628
  LTE_DL_FRAME_PARMS *fp=&eNB->frame_parms;
629
  int i;
630

631 632
  if (ru!=NULL) {
    pthread_mutex_lock(&proc->mutex_RU_PRACH);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
633 634

    for (i=0; i<eNB->num_RU; i++) {
635
      if (ru == eNB->RU_list[i]) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
636 637 638 639 640 641 642
        LOG_D(PHY,"frame %d, subframe %d: RU %d for eNB %d signals PRACH (mask %x, num_RU %d)\n",frame,subframe,i,eNB->Mod_id,proc->RU_mask_prach,eNB->num_RU);

        if ((proc->RU_mask_prach&(1<<i)) > 0)
          LOG_E(PHY,"eNB %d frame %d, subframe %d : previous information (PRACH) from RU %d (num_RU %d, mask %x) has not been served yet!\n",
                eNB->Mod_id,frame,subframe,ru->idx,eNB->num_RU,proc->RU_mask_prach);

        proc->RU_mask_prach |= (1<<i);
643 644
      }
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
645

646 647
    if (proc->RU_mask_prach != (1<<eNB->num_RU)-1) {  // not all RUs have provided their information so return
      pthread_mutex_unlock(&proc->mutex_RU_PRACH);
Raymond Knopp's avatar
Raymond Knopp committed
648
      return;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
649
    } else { // all RUs have provided their information so continue on and wakeup eNB processing
650 651 652 653
      proc->RU_mask_prach = 0;
      pthread_mutex_unlock(&proc->mutex_RU_PRACH);
    }
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
654

655
  // check if we have to detect PRACH first
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
656
  if (is_prach_subframe(fp,frame,subframe)>0) {
657
    LOG_D(PHY,"Triggering prach processing, frame %d, subframe %d\n",frame,subframe);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
658

659
    if (proc->instance_cnt_prach == 0) {
660
      LOG_W(PHY,"[eNB] Frame %d Subframe %d, dropping PRACH\n", frame,subframe);
661 662
      return;
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
663

664 665 666 667 668 669
    // wake up thread for PRACH RX
    if (pthread_mutex_lock(&proc->mutex_prach) != 0) {
      LOG_E( PHY, "[eNB] ERROR pthread_mutex_lock for eNB PRACH thread %d (IC %d)\n", proc->thread_index, proc->instance_cnt_prach);
      exit_fun( "error locking mutex_prach" );
      return;
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
670

671 672
    ++proc->instance_cnt_prach;
    // set timing for prach thread
673 674
    proc->frame_prach = frame;
    proc->subframe_prach = subframe;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
675

676 677 678 679 680 681
    // the thread can now be woken up
    if (pthread_cond_signal(&proc->cond_prach) != 0) {
      LOG_E( PHY, "[eNB] ERROR pthread_cond_signal for eNB PRACH thread %d\n", proc->thread_index);
      exit_fun( "ERROR pthread_cond_signal" );
      return;
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
682

683 684 685 686
    pthread_mutex_unlock( &proc->mutex_prach );
  }
}

687
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
688 689
void wakeup_prach_eNB_br(PHY_VARS_eNB *eNB,RU_t *ru,int frame,int subframe) {

690
  L1_proc_t *proc = &eNB->proc;
691 692 693 694 695
  LTE_DL_FRAME_PARMS *fp=&eNB->frame_parms;
  int i;

  if (ru!=NULL) {
    pthread_mutex_lock(&proc->mutex_RU_PRACH_br);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
696 697

    for (i=0; i<eNB->num_RU; i++) {
698
      if (ru == eNB->RU_list[i]) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
699 700 701 702 703 704 705
        LOG_D(PHY,"frame %d, subframe %d: RU %d for eNB %d signals PRACH BR (mask %x, num_RU %d)\n",frame,subframe,i,eNB->Mod_id,proc->RU_mask_prach_br,eNB->num_RU);

        if ((proc->RU_mask_prach_br&(1<<i)) > 0)
          LOG_E(PHY,"eNB %d frame %d, subframe %d : previous information (PRACH BR) from RU %d (num_RU %d, mask %x) has not been served yet!\n",
                eNB->Mod_id,frame,subframe,ru->idx,eNB->num_RU,proc->RU_mask_prach_br);

        proc->RU_mask_prach_br |= (1<<i);
706 707
      }
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
708

709 710
    if (proc->RU_mask_prach_br != (1<<eNB->num_RU)-1) {  // not all RUs have provided their information so return
      pthread_mutex_unlock(&proc->mutex_RU_PRACH_br);
Raymond Knopp's avatar
Raymond Knopp committed
711
      return;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
712
    } else { // all RUs have provided their information so continue on and wakeup eNB processing
713 714 715 716
      proc->RU_mask_prach_br = 0;
      pthread_mutex_unlock(&proc->mutex_RU_PRACH_br);
    }
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
717

718
  // check if we have to detect PRACH first
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
719
  if (is_prach_subframe(fp,frame,subframe)>0) {
720
    LOG_D(PHY,"Triggering prach br processing, frame %d, subframe %d\n",frame,subframe);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
721

722 723 724 725
    if (proc->instance_cnt_prach_br == 0) {
      LOG_W(PHY,"[eNB] Frame %d Subframe %d, dropping PRACH BR\n", frame,subframe);
      return;
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
726

727 728 729 730 731 732
    // wake up thread for PRACH RX
    if (pthread_mutex_lock(&proc->mutex_prach_br) != 0) {
      LOG_E( PHY, "[eNB] ERROR pthread_mutex_lock for eNB PRACH thread %d (IC %d)\n", proc->thread_index, proc->instance_cnt_prach_br);
      exit_fun( "error locking mutex_prach" );
      return;
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
733

734 735 736 737
    ++proc->instance_cnt_prach_br;
    // set timing for prach thread
    proc->frame_prach_br = frame;
    proc->subframe_prach_br = subframe;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
738

739 740 741 742 743 744
    // the thread can now be woken up
    if (pthread_cond_signal(&proc->cond_prach_br) != 0) {
      LOG_E( PHY, "[eNB] ERROR pthread_cond_signal for eNB PRACH BR thread %d\n", proc->thread_index);
      exit_fun( "ERROR pthread_cond_signal" );
      return;
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
745

746 747 748 749 750
    pthread_mutex_unlock( &proc->mutex_prach_br );
  }
}
#endif

Wang Tsu-Han's avatar
Wang Tsu-Han committed
751

752 753
/*!
 * \brief The prach receive thread of eNB.
754
 * \param param is a \ref L1_proc_t structure which contains the info what to process.
755 756
 * \returns a pointer to an int. The storage is not on the heap and must not be freed.
 */
757
static void *eNB_thread_prach( void *param ) {
758
  static int eNB_thread_prach_status;
759
  PHY_VARS_eNB *eNB= (PHY_VARS_eNB *)param;
760
  L1_proc_t *proc = &eNB->proc;
Raymond Knopp's avatar
Raymond Knopp committed
761

762 763
  // set default return value
  eNB_thread_prach_status = 0;
764
  thread_top_init("eNB_thread_prach",1,500000,1000000,20000000);
765

766
  //wait_sync("eNB_thread_prach");
767

768
  while (!oai_exit) {
Raymond Knopp's avatar
Raymond Knopp committed
769
    if (wait_on_condition(&proc->mutex_prach,&proc->cond_prach,&proc->instance_cnt_prach,"eNB_prach_thread") < 0) break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
770

771
    if (oai_exit) break;
772 773

    LOG_D(PHY,"Running eNB prach procedures\n");
774
    prach_procedures(eNB
775
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
776
                     ,0
777
#endif
778
                    );
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
779

Raymond Knopp's avatar
Raymond Knopp committed
780
    if (release_thread(&proc->mutex_prach,&proc->instance_cnt_prach,"eNB_prach_thread") < 0) break;
781
  }
782

783
  LOG_I(PHY, "Exiting eNB thread PRACH\n");
784 785
  eNB_thread_prach_status = 0;
  return &eNB_thread_prach_status;
786 787
}

788
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
789 790
/*!
 * \brief The prach receive thread of eNB for BL/CE UEs.
791
 * \param param is a \ref L1_proc_t structure which contains the info what to process.
792 793
 * \returns a pointer to an int. The storage is not on the heap and must not be freed.
 */
794
static void *eNB_thread_prach_br( void *param ) {
795 796
  static int eNB_thread_prach_status;
  PHY_VARS_eNB *eNB= (PHY_VARS_eNB *)param;
797
  L1_proc_t *proc = &eNB->proc;
798 799 800

  // set default return value
  eNB_thread_prach_status = 0;
801
  thread_top_init("eNB_thread_prach_br",1,500000,1000000,20000000);
802 803 804

  while (!oai_exit) {
    if (wait_on_condition(&proc->mutex_prach_br,&proc->cond_prach_br,&proc->instance_cnt_prach_br,"eNB_prach_thread_br") < 0) break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
805

806
    if (oai_exit) break;
807 808 809

    LOG_D(PHY,"Running eNB prach procedures for BL/CE UEs\n");
    prach_procedures(eNB,1);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
810

811 812 813 814 815 816 817 818 819
    if (release_thread(&proc->mutex_prach_br,&proc->instance_cnt_prach_br,"eNB_prach_thread_br") < 0) break;
  }

  LOG_I(PHY, "Exiting eNB thread PRACH BR\n");
  eNB_thread_prach_status = 0;
  return &eNB_thread_prach_status;
}

#endif
Raymond Knopp's avatar
Raymond Knopp committed
820

Raymond Knopp's avatar
Raymond Knopp committed
821

822

Wang Tsu-Han's avatar
Wang Tsu-Han committed
823
extern void init_td_thread(PHY_VARS_eNB *);
Eurecom's avatar
Eurecom committed
824
extern void init_te_thread(PHY_VARS_eNB *);
Wang Tsu-Han's avatar
Wang Tsu-Han committed
825 826
extern void kill_td_thread(PHY_VARS_eNB *);
extern void kill_te_thread(PHY_VARS_eNB *);
827

828 829
static void *process_stats_thread(void *param) {
  PHY_VARS_eNB     *eNB      = (PHY_VARS_eNB *)param;
830
  wait_sync("process_stats_thread");
831 832

  while (!oai_exit) {
Wang Tsu-Han's avatar
Wang Tsu-Han committed
833
    sleep(1);
834 835

    if (opp_enabled == 1) {
836 837
      if ( eNB->ulsch_decoding_stats.trials>0)
        print_meas(&eNB->ulsch_decoding_stats,"ulsch_decoding",NULL,NULL);
838

839
      if (eNB->dlsch_encoding_stats.trials >0) {
840 841 842 843 844 845 846 847 848 849 850
        print_meas(&eNB->dlsch_turbo_encoding_preperation_stats,"dlsch_coding_crc",NULL,NULL);
        print_meas(&eNB->dlsch_turbo_encoding_segmentation_stats,"dlsch_segmentation",NULL,NULL);
        print_meas(&eNB->dlsch_encoding_stats,"dlsch_encoding",NULL,NULL);
        print_meas(&eNB->dlsch_turbo_encoding_signal_stats,"coding_signal",NULL,NULL);
        print_meas(&eNB->dlsch_turbo_encoding_main_stats,"coding_main",NULL,NULL);
        print_meas(&eNB->dlsch_turbo_encoding_stats,"turbo_encoding",NULL,NULL);
        print_meas(&eNB->dlsch_interleaving_stats,"turbo_interleaving",NULL,NULL);
        print_meas(&eNB->dlsch_rate_matching_stats,"turbo_rate_matching",NULL,NULL);
        print_meas(&eNB->dlsch_turbo_encoding_waiting_stats,"coding_wait",NULL,NULL);
        print_meas(&eNB->dlsch_turbo_encoding_wakeup_stats0,"coding_worker_0",NULL,NULL);
        print_meas(&eNB->dlsch_turbo_encoding_wakeup_stats1,"coding_worker_1",NULL,NULL);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
851
      }
852

853 854
      print_meas(&eNB->dlsch_modulation_stats,"dlsch_modulation",NULL,NULL);
    }
855
  }
856

857 858 859
  return(NULL);
}

860

861
void init_eNB_proc(int inst) {
862
  /*int i=0;*/
863
  int CC_id;
864
  PHY_VARS_eNB *eNB;
865 866
  L1_proc_t *proc;
  L1_rxtx_proc_t *L1_proc, *L1_proc_tx;
Wang Tsu-Han's avatar
Wang Tsu-Han committed
867
  pthread_attr_t *attr0=NULL,*attr1=NULL,*attr_prach=NULL;
868
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
869 870
  pthread_attr_t *attr_prach_br=NULL;
#endif
871 872
  LOG_I(PHY,"%s(inst:%d) RC.nb_CC[inst]:%d \n",__FUNCTION__,inst,RC.nb_CC[inst]);

873
  for (CC_id=0; CC_id<RC.nb_CC[inst]; CC_id++) {
874
    eNB = RC.eNB[inst][CC_id];
laurent's avatar
laurent committed
875
#ifndef OCP_FRAMEWORK
876
    LOG_I(PHY,"Initializing eNB processes instance:%d CC_id %d \n",inst,CC_id);
laurent's avatar
laurent committed
877
#endif
878
    proc = &eNB->proc;
Raymond Knopp's avatar
Raymond Knopp committed
879

Wang Tsu-Han's avatar
Wang Tsu-Han committed
880 881 882 883
    L1_proc                        = &proc->L1_proc;
    L1_proc_tx                     = &proc->L1_proc_tx;
    L1_proc->instance_cnt          = -1;
    L1_proc_tx->instance_cnt       = -1;
884 885
    L1_proc->instance_cnt_RUs      = 0;
    L1_proc_tx->instance_cnt_RUs   = 0;
886
    proc->instance_cnt_prach       = -1;
887
    proc->instance_cnt_asynch_rxtx = -1;
Eurecom's avatar
Eurecom committed
888
    proc->instance_cnt_synch       = -1;
889
    proc->CC_id                    = CC_id;    
890

Wang Tsu-Han's avatar
Wang Tsu-Han committed
891 892
    proc->first_rx                 =1;
    proc->first_tx                 =1;
Wang Tsu-Han's avatar
Wang Tsu-Han committed
893
    proc->RU_mask_tx               = (1<<eNB->num_RU)-1;
Wang Tsu-Han's avatar
Wang Tsu-Han committed
894 895
    proc->RU_mask                  =0;
    proc->RU_mask_prach            =0;
Raymond Knopp's avatar
Raymond Knopp committed
896

897
    pthread_mutex_init( &eNB->UL_INFO_mutex, NULL);
Wang Tsu-Han's avatar
Wang Tsu-Han committed
898 899 900 901
    pthread_mutex_init( &L1_proc->mutex, NULL);
    pthread_mutex_init( &L1_proc_tx->mutex, NULL);
    pthread_cond_init( &L1_proc->cond, NULL);
    pthread_cond_init( &L1_proc_tx->cond, NULL);
902 903 904 905
    pthread_mutex_init( &L1_proc->mutex_RUs, NULL);
    pthread_mutex_init( &L1_proc_tx->mutex_RUs, NULL);
    pthread_cond_init( &L1_proc->cond_RUs, NULL);
    pthread_cond_init( &L1_proc_tx->cond_RUs, NULL);
Raymond Knopp's avatar
Raymond Knopp committed
906 907 908

    pthread_mutex_init( &proc->mutex_prach, NULL);
    pthread_mutex_init( &proc->mutex_asynch_rxtx, NULL);
909
    pthread_mutex_init( &proc->mutex_RU,NULL);
Wang Tsu-Han's avatar
Wang Tsu-Han committed
910
    pthread_mutex_init( &proc->mutex_RU_tx,NULL);
911
    pthread_mutex_init( &proc->mutex_RU_PRACH,NULL);
912
    pthread_cond_init( &proc->cond_prach, NULL);
913
    pthread_cond_init( &proc->cond_asynch_rxtx, NULL);
914 915
    pthread_attr_init( &proc->attr_prach);
    pthread_attr_init( &proc->attr_asynch_rxtx);
Wang Tsu-Han's avatar
Wang Tsu-Han committed
916 917
    pthread_attr_init( &L1_proc->attr);
    pthread_attr_init( &L1_proc_tx->attr);
918
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
919 920 921 922 923 924 925
    proc->instance_cnt_prach_br    = -1;
    proc->RU_mask_prach_br=0;
    pthread_mutex_init( &proc->mutex_prach_br, NULL);
    pthread_mutex_init( &proc->mutex_RU_PRACH_br,NULL);
    pthread_cond_init( &proc->cond_prach_br, NULL);
    pthread_attr_init( &proc->attr_prach_br);
#endif
926
#ifndef DEADLINE_SCHEDULER
Wang Tsu-Han's avatar
Wang Tsu-Han committed
927 928
    attr0       = &L1_proc->attr;
    attr1       = &L1_proc_tx->attr;
Raymond Knopp's avatar
Raymond Knopp committed
929
    attr_prach  = &proc->attr_prach;
930
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
931 932
    attr_prach_br  = &proc->attr_prach_br;
#endif
Raymond Knopp's avatar
Raymond Knopp committed
933
    //    attr_td     = &proc->attr_td;
934
    //    attr_te     = &proc->attr_te;
Raymond Knopp's avatar
Raymond Knopp committed
935
#endif
Wang Tsu-Han's avatar
Wang Tsu-Han committed
936

937
    if(get_thread_worker_conf() == WORKER_ENABLE) {
938
      init_te_thread(eNB);
Wang Tsu-Han's avatar
Wang Tsu-Han committed
939
      init_td_thread(eNB);
940
    }
Raymond Knopp's avatar
Raymond Knopp committed
941

942 943
    LOG_I(PHY,"eNB->single_thread_flag:%d\n", eNB->single_thread_flag);

944
    if ((get_thread_parallel_conf() == PARALLEL_RU_L1_SPLIT || get_thread_parallel_conf() == PARALLEL_RU_L1_TRX_SPLIT) && nfapi_mode!=2) {
Wang Tsu-Han's avatar
Wang Tsu-Han committed
945 946
      pthread_create( &L1_proc->pthread, attr0, L1_thread, proc );
      pthread_create( &L1_proc_tx->pthread, attr1, L1_thread_tx, proc);
947
    } else if (nfapi_mode == 2) { // this is neccesary in VNF or L2 FAPI simulator.
948 949 950
      // Original Code from Fujitsu w/ old structure/field name
      //pthread_create( &proc_rxtx[0].pthread_rxtx, attr0, eNB_thread_rxtx, &proc_rxtx[0] );
      //pthread_create( &proc_rxtx[1].pthread_rxtx, attr1, eNB_thread_rxtx, &proc_rxtx[1] );
951 952
      pthread_create( &L1_proc->pthread, attr0, L1_thread, L1_proc );
      pthread_create( &L1_proc_tx->pthread, attr1, L1_thread, L1_proc_tx);
953
    }
954

955
    pthread_create( &proc->pthread_prach, attr_prach, eNB_thread_prach, eNB );
956
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
957 958
    pthread_create( &proc->pthread_prach_br, attr_prach_br, eNB_thread_prach_br, eNB );
#endif
959 960 961

    AssertFatal(proc->instance_cnt_prach == -1,"instance_cnt_prach = %d\n",proc->instance_cnt_prach);

962
    if (opp_enabled == 1) pthread_create(&proc->process_stats_thread,NULL,process_stats_thread,(void *)eNB);
963
  }
964

965
  //for multiple CCs: setup master and slaves
966
  /*
967 968
     for (CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
     eNB = PHY_vars_eNB_g[inst][CC_id];
969

970 971
     if (eNB->node_timing == synch_to_ext_device) { //master
     eNB->proc.num_slaves = MAX_NUM_CCs-1;
972
     eNB->proc.slave_proc = (L1_proc_t**)malloc(eNB->proc.num_slaves*sizeof(L1_proc_t*));
973

974 975 976 977 978 979 980
     for (i=0; i< eNB->proc.num_slaves; i++) {
     if (i < CC_id)  eNB->proc.slave_proc[i] = &(PHY_vars_eNB_g[inst][i]->proc);
     if (i >= CC_id)  eNB->proc.slave_proc[i] = &(PHY_vars_eNB_g[inst][i+1]->proc);
     }
     }
     }
  */
981 982 983 984
  /* setup PHY proc TX sync mechanism */
  pthread_mutex_init(&sync_phy_proc.mutex_phy_proc_tx, NULL);
  pthread_cond_init(&sync_phy_proc.cond_phy_proc_tx, NULL);
  sync_phy_proc.phy_proc_CC_id = 0;
985 986
}

987

988

989 990 991
/*!
 * \brief Terminate eNB TX and RX threads.
 */
992
void kill_eNB_proc(int inst) {
993
  int *status;
994
  PHY_VARS_eNB *eNB;
995 996
  L1_proc_t *proc;
  L1_rxtx_proc_t *L1_proc, *L1_proc_tx;
997

998
  for (int CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
999
    eNB=RC.eNB[inst][CC_id];
1000
    
Wang Tsu-Han's avatar
Wang Tsu-Han committed
1001 1002 1003
    proc        = &eNB->proc;
    L1_proc     = &proc->L1_proc;
    L1_proc_tx  = &proc->L1_proc_tx;
1004

1005
    if(get_thread_worker_conf() == WORKER_ENABLE) {
1006 1007 1008
      kill_td_thread(eNB);
      kill_te_thread(eNB);
    }
1009

1010
    LOG_I(PHY, "Killing TX CC_id %d inst %d\n", CC_id, inst );
Wang Tsu-Han's avatar
Wang Tsu-Han committed
1011

1012 1013 1014 1015 1016 1017 1018 1019 1020 1021
    if ((get_thread_parallel_conf() == PARALLEL_RU_L1_SPLIT || get_thread_parallel_conf() == PARALLEL_RU_L1_TRX_SPLIT) && nfapi_mode!=2) {
      pthread_mutex_lock(&L1_proc->mutex);
      L1_proc->instance_cnt = 0;
      pthread_cond_signal(&L1_proc->cond);
      pthread_mutex_unlock(&L1_proc->mutex);

      pthread_mutex_lock(&L1_proc_tx->mutex);
      L1_proc_tx->instance_cnt = 0;
      pthread_cond_signal(&L1_proc_tx->cond);
      pthread_mutex_unlock(&L1_proc_tx->mutex);
1022
    }
Wang Tsu-Han's avatar
Wang Tsu-Han committed
1023

Robert Schmidt's avatar
Robert Schmidt committed
1024
    pthread_mutex_lock(&proc->mutex_prach);
Raymond Knopp's avatar
Raymond Knopp committed
1025 1026
    proc->instance_cnt_prach = 0;
    pthread_cond_signal( &proc->cond_prach );
Robert Schmidt's avatar
Robert Schmidt committed
1027
    pthread_mutex_unlock(&proc->mutex_prach);
1028
    pthread_cond_signal( &proc->cond_asynch_rxtx );
1029
    pthread_cond_broadcast(&sync_phy_proc.cond_phy_proc_tx);
1030
    LOG_D(PHY, "joining pthread_prach\n");
1031
    pthread_join( proc->pthread_prach, (void **)&status );
1032
    LOG_I(PHY, "Destroying prach mutex/cond\n");
1033
    pthread_mutex_destroy( &proc->mutex_prach );
1034
    pthread_cond_destroy( &proc->cond_prach );
1035
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
1036
    proc->instance_cnt_prach_br = 0;
1037
    pthread_cond_signal( &proc->cond_prach_br );
1038
    pthread_join( proc->pthread_prach_br, (void **)&status );
1039 1040
    pthread_mutex_destroy( &proc->mutex_prach_br );
    pthread_cond_destroy( &proc->cond_prach_br );
Wang Tsu-Han's avatar
Wang Tsu-Han committed
1041
#endif
1042
    LOG_I(PHY, "Destroying UL_INFO mutex\n");
1043
    pthread_mutex_destroy(&eNB->UL_INFO_mutex);
Wang Tsu-Han's avatar
Wang Tsu-Han committed
1044 1045 1046 1047 1048 1049

    if ((get_thread_parallel_conf() == PARALLEL_RU_L1_SPLIT || get_thread_parallel_conf() == PARALLEL_RU_L1_TRX_SPLIT) && nfapi_mode!=2) {
      LOG_I(PHY, "Joining L1_proc mutex/cond\n");
      pthread_join( L1_proc->pthread, (void**)&status );
      LOG_I(PHY, "Joining L1_proc_tx mutex/cond\n");
      pthread_join( L1_proc_tx->pthread, (void**)&status );
1050
    }
Wang Tsu-Han's avatar
Wang Tsu-Han committed
1051 1052 1053
    LOG_I(PHY, "Destroying L1_proc mutex/cond\n");
    pthread_mutex_destroy( &L1_proc->mutex );
    pthread_cond_destroy( &L1_proc->cond );
1054 1055
    pthread_mutex_destroy( &L1_proc->mutex_RUs );
    pthread_cond_destroy( &L1_proc->cond_RUs );
Wang Tsu-Han's avatar
Wang Tsu-Han committed
1056 1057 1058
    LOG_I(PHY, "Destroying L1_proc_tx mutex/cond\n");
    pthread_mutex_destroy( &L1_proc_tx->mutex );
    pthread_cond_destroy( &L1_proc_tx->cond );
1059 1060
    pthread_mutex_destroy( &L1_proc_tx->mutex_RUs );
    pthread_cond_destroy( &L1_proc_tx->cond_RUs );
Robert Schmidt's avatar
Robert Schmidt committed
1061 1062 1063

    pthread_attr_destroy(&proc->attr_prach);
    pthread_attr_destroy(&proc->attr_asynch_rxtx);
Wang Tsu-Han's avatar
Wang Tsu-Han committed
1064 1065
    pthread_attr_destroy(&L1_proc->attr);
    pthread_attr_destroy(&L1_proc_tx->attr);
1066
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
Robert Schmidt's avatar
Robert Schmidt committed
1067 1068 1069
    pthread_mutex_destroy(&proc->mutex_RU_PRACH_br);
    pthread_attr_destroy(&proc->attr_prach_br);
#endif
1070
  }
1071 1072
}

1073

1074 1075 1076 1077 1078 1079


void reset_opp_meas(void) {
  int sfn;
  reset_meas(&softmodem_stats_mt);
  reset_meas(&softmodem_stats_hw);
1080

1081
  for (sfn=0; sfn < 10; sfn++) {
1082
    reset_meas(&softmodem_stats_rxtx_sf);
1083
    reset_meas(&softmodem_stats_rx_sf);
1084 1085 1086
  }
}

1087

1088 1089 1090 1091
void print_opp_meas(void) {
  int sfn=0;
  print_meas(&softmodem_stats_mt, "Main ENB Thread", NULL, NULL);
  print_meas(&softmodem_stats_hw, "HW Acquisation", NULL, NULL);
1092

1093
  for (sfn=0; sfn < 10; sfn++) {
1094
    print_meas(&softmodem_stats_rxtx_sf,"[eNB][total_phy_proc_rxtx]",NULL, NULL);
1095
    print_meas(&softmodem_stats_rx_sf,"[eNB][total_phy_proc_rx]",NULL,NULL);
1096 1097
  }
}
1098

1099
void free_transport(PHY_VARS_eNB *eNB) {
1100 1101 1102 1103
  int i;
  int j;

  for (i=0; i<NUMBER_OF_UE_MAX; i++) {
1104
    LOG_D(PHY, "Freeing Transport Channel Buffers for DLSCH, UE %d\n",i);
1105 1106 1107

    for (j=0; j<2; j++) free_eNB_dlsch(eNB->dlsch[i][j]);

1108
    LOG_D(PHY, "Freeing Transport Channel Buffer for ULSCH, UE %d\n",i);
1109 1110
    free_eNB_ulsch(eNB->ulsch[1+i]);
  }
1111

1112 1113 1114
  free_eNB_ulsch(eNB->ulsch[0]);
}

1115 1116 1117 1118
void init_transport(PHY_VARS_eNB *eNB) {
  int i;
  int j;
  LTE_DL_FRAME_PARMS *fp = &eNB->frame_parms;
Cedric Roux's avatar
Cedric Roux committed
1119
  LOG_I(PHY, "Initialise transport\n");
1120

1121
  for (i=0; i<NUMBER_OF_UE_MAX; i++) {
Wang Tsu-Han's avatar
Wang Tsu-Han committed
1122
    LOG_D(PHY,"Allocating Transport Channel Buffers for DLSCH, UE %d\n",i);
1123

1124 1125
    for (j=0; j<2; j++) {
      eNB->dlsch[i][j] = new_eNB_dlsch(1,8,NSOFT,fp->N_RB_DL,0,fp);
1126

1127
      if (!eNB->dlsch[i][j]) {
1128
        LOG_E(PHY,"Can't get eNB dlsch structures for UE %d \n", i);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1129
        exit(-1);
1130
      } else {
1131 1132
        eNB->dlsch[i][j]->rnti=0;
        LOG_D(PHY,"dlsch[%d][%d] => %p rnti:%d\n",i,j,eNB->dlsch[i][j], eNB->dlsch[i][j]->rnti);
1133 1134
      }
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1135

Wang Tsu-Han's avatar
Wang Tsu-Han committed
1136
    LOG_D(PHY,"Allocating Transport Channel Buffer for ULSCH, UE %d\n",i);
1137
    eNB->ulsch[1+i] = new_eNB_ulsch(MAX_TURBO_ITERATIONS,fp->N_RB_UL, 0);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1138

1139 1140
    if (!eNB->ulsch[1+i]) {
      LOG_E(PHY,"Can't get eNB ulsch structures\n");
1141 1142
      exit(-1);
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1143

1144 1145 1146
    // this is the transmission mode for the signalling channels
    // this will be overwritten with the real transmission mode by the RRC once the UE is connected
    eNB->transmission_mode[i] = fp->nb_antenna_ports_eNB==1 ? 1 : 2;
1147
  }
1148

1149 1150
  // ULSCH for RA
  eNB->ulsch[0] = new_eNB_ulsch(MAX_TURBO_ITERATIONS, fp->N_RB_UL, 0);
1151

1152 1153 1154 1155
  if (!eNB->ulsch[0]) {
    LOG_E(PHY,"Can't get eNB ulsch structures\n");
    exit(-1);
  }
1156

1157 1158 1159 1160 1161 1162 1163
  eNB->dlsch_SI  = new_eNB_dlsch(1,8,NSOFT,fp->N_RB_DL, 0, fp);
  LOG_D(PHY,"eNB %d.%d : SI %p\n",eNB->Mod_id,eNB->CC_id,eNB->dlsch_SI);
  eNB->dlsch_ra  = new_eNB_dlsch(1,8,NSOFT,fp->N_RB_DL, 0, fp);
  LOG_D(PHY,"eNB %d.%d : RA %p\n",eNB->Mod_id,eNB->CC_id,eNB->dlsch_ra);
  eNB->dlsch_MCH = new_eNB_dlsch(1,8,NSOFT,fp->N_RB_DL, 0, fp);
  LOG_D(PHY,"eNB %d.%d : MCH %p\n",eNB->Mod_id,eNB->CC_id,eNB->dlsch_MCH);
  eNB->rx_total_gain_dB=130;
1164

1165 1166
  for(i=0; i<NUMBER_OF_UE_MAX; i++)
    eNB->mu_mimo_mode[i].dl_pow_off = 2;
1167

1168 1169 1170 1171 1172
  eNB->check_for_total_transmissions = 0;
  eNB->check_for_MUMIMO_transmissions = 0;
  eNB->FULL_MUMIMO_transmissions = 0;
  eNB->check_for_SUMIMO_transmissions = 0;
  fp->pucch_config_common.deltaPUCCH_Shift = 1;
1173
}
Raymond Knopp's avatar
Raymond Knopp committed
1174 1175

void init_eNB_afterRU(void) {
1176 1177
  int inst,CC_id,ru_id,i,aa;
  PHY_VARS_eNB *eNB;
1178
  LOG_I(PHY,"%s() RC.nb_inst:%d\n", __FUNCTION__, RC.nb_inst);
1179

1180
  for (inst=0; inst<RC.nb_inst; inst++) {
1181 1182
    LOG_I(PHY,"RC.nb_CC[inst]:%d\n", RC.nb_CC[inst]);

1183
    for (CC_id=0; CC_id<RC.nb_CC[inst]; CC_id++) {
1184
      LOG_I(PHY,"RC.nb_CC[inst:%d][CC_id:%d]:%p\n", inst, CC_id, RC.eNB[inst][CC_id]);
1185 1186
      eNB                                  =  RC.eNB[inst][CC_id];
      phy_init_lte_eNB(eNB,0,0);
1187

1188
      // map antennas and PRACH signals to eNB RX
1189
      if (0) AssertFatal(eNB->num_RU>0,"Number of RU attached to eNB %d is zero\n",eNB->Mod_id);
1190

1191
      LOG_I(PHY,"Mapping RX ports from %d RUs to eNB %d\n",eNB->num_RU,eNB->Mod_id);
1192
      eNB->frame_parms.nb_antennas_rx       = 0;
Cedric Roux's avatar
Cedric Roux committed
1193
      LOG_I(PHY,"Overwriting eNB->prach_vars.rxsigF[0]:%p\n", eNB->prach_vars.rxsigF[0]);
1194
      eNB->prach_vars.rxsigF[0] = (int16_t **)malloc16(64*sizeof(int16_t *));
1195
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
1196

1197
      for (int ce_level=0; ce_level<4; ce_level++) {
Cedric Roux's avatar
Cedric Roux committed
1198
        LOG_I(PHY,"Overwriting eNB->prach_vars_br.rxsigF.rxsigF[0]:%p\n", eNB->prach_vars_br.rxsigF[ce_level]);
1199
        eNB->prach_vars_br.rxsigF[ce_level] = (int16_t **)malloc16(64*sizeof(int16_t *));
1200 1201
      }

Raymond Knopp's avatar
Raymond Knopp committed
1202
#endif
1203 1204
      LOG_I(PHY,"eNB->num_RU:%d\n", eNB->num_RU);

1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216
      for (ru_id=0,aa=0; ru_id<eNB->num_RU; ru_id++) {
        eNB->frame_parms.nb_antennas_rx    += eNB->RU_list[ru_id]->nb_rx;
        AssertFatal(eNB->RU_list[ru_id]->common.rxdataF!=NULL,
                    "RU %d : common.rxdataF is NULL\n",
                    eNB->RU_list[ru_id]->idx);
        AssertFatal(eNB->RU_list[ru_id]->prach_rxsigF!=NULL,
                    "RU %d : prach_rxsigF is NULL\n",
                    eNB->RU_list[ru_id]->idx);

        for (i=0; i<eNB->RU_list[ru_id]->nb_rx; aa++,i++) {
          LOG_I(PHY,"Attaching RU %d antenna %d to eNB antenna %d\n",eNB->RU_list[ru_id]->idx,i,aa);
          eNB->prach_vars.rxsigF[0][aa]    =  eNB->RU_list[ru_id]->prach_rxsigF[i];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1217
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
1218

1219 1220
          for (int ce_level=0; ce_level<4; ce_level++)
            eNB->prach_vars_br.rxsigF[ce_level][aa] = eNB->RU_list[ru_id]->prach_rxsigF_br[ce_level][i];
1221

Raymond Knopp's avatar
Raymond Knopp committed
1222
#endif
1223
          eNB->common_vars.rxdataF[aa]     =  eNB->RU_list[ru_id]->common.rxdataF[i];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1224
        }
1225
      }
1226

Cedric Roux's avatar
Cedric Roux committed
1227 1228 1229 1230
      /* TODO: review this code, there is something wrong.
       * In monolithic mode, we come here with nb_antennas_rx == 0
       * (not tested in other modes).
       */
1231
      if (eNB->frame_parms.nb_antennas_rx < 1) {
1232 1233
        LOG_I(PHY, "%s() ************* DJP ***** eNB->frame_parms.nb_antennas_rx:%d - GOING TO HARD CODE TO 1", __FUNCTION__, eNB->frame_parms.nb_antennas_rx);
        eNB->frame_parms.nb_antennas_rx = 1;
1234
      } else {
Cedric Roux's avatar
Cedric Roux committed
1235
        //LOG_I(PHY," Delete code\n");
1236 1237
      }

1238
      if (eNB->frame_parms.nb_antennas_tx < 1) {
1239 1240
        LOG_I(PHY, "%s() ************* DJP ***** eNB->frame_parms.nb_antennas_tx:%d - GOING TO HARD CODE TO 1", __FUNCTION__, eNB->frame_parms.nb_antennas_tx);
        eNB->frame_parms.nb_antennas_tx = 1;
1241
      } else {
Cedric Roux's avatar
Cedric Roux committed
1242
        //LOG_I(PHY," Delete code\n");
1243 1244
      }

1245
      AssertFatal(eNB->frame_parms.nb_antennas_rx >0,
1246
                  "inst %d, CC_id %d : nb_antennas_rx %d\n",inst,CC_id,eNB->frame_parms.nb_antennas_rx);
1247
      LOG_I(PHY,"inst %d, CC_id %d : nb_antennas_rx %d\n",inst,CC_id,eNB->frame_parms.nb_antennas_rx);
1248
      init_transport(eNB);
Raymond Knopp's avatar
Raymond Knopp committed
1249
      //init_precoding_weights(RC.eNB[inst][CC_id]);
1250
    }
1251

1252 1253
    init_eNB_proc(inst);
  }
1254

1255
  for (ru_id=0; ru_id<RC.nb_RU; ru_id++) {
1256
    AssertFatal(RC.ru[ru_id]!=NULL,"ru_id %d is null\n",ru_id);
1257 1258
    RC.ru[ru_id]->wakeup_rxtx         = wakeup_rxtx;
    RC.ru[ru_id]->wakeup_prach_eNB    = wakeup_prach_eNB;
1259
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
1260
    RC.ru[ru_id]->wakeup_prach_eNB_br = wakeup_prach_eNB_br;
1261
#endif
1262
    RC.ru[ru_id]->eNB_top             = eNB_top;
1263 1264
  }
}
1265

1266
void init_eNB(int single_thread_flag,int wait_for_sync) {
1267
  int CC_id;
1268
  int inst;
1269
  PHY_VARS_eNB *eNB;
1270 1271
  LOG_I(PHY,"[lte-softmodem.c] eNB structure about to allocated RC.nb_L1_inst:%d RC.nb_L1_CC[0]:%d\n",RC.nb_L1_inst,RC.nb_L1_CC[0]);

1272
  if (RC.eNB == NULL) RC.eNB = (PHY_VARS_eNB ** *) malloc(RC.nb_L1_inst*sizeof(PHY_VARS_eNB **));
Raymond Knopp's avatar
Raymond Knopp committed
1273

1274
  LOG_I(PHY,"[lte-softmodem.c] eNB structure RC.eNB allocated\n");
1275

1276 1277
  for (inst=0; inst<RC.nb_L1_inst; inst++) {
    if (RC.eNB[inst] == NULL) RC.eNB[inst] = (PHY_VARS_eNB **) malloc(RC.nb_CC[inst]*sizeof(PHY_VARS_eNB *));
1278

1279 1280
    for (CC_id=0; CC_id<RC.nb_L1_CC[inst]; CC_id++) {
      if (RC.eNB[inst][CC_id] == NULL) RC.eNB[inst][CC_id] = (PHY_VARS_eNB *) malloc(sizeof(PHY_VARS_eNB));
1281

1282 1283 1284
      eNB                     = RC.eNB[inst][CC_id];
      eNB->abstraction_flag   = 0;
      eNB->single_thread_flag = single_thread_flag;
1285
      LOG_I(PHY,"Initializing eNB %d CC_id %d single_thread_flag:%d\n",inst,CC_id,single_thread_flag);
laurent's avatar
laurent committed
1286
#ifndef OCP_FRAMEWORK
1287
      LOG_I(PHY,"Initializing eNB %d CC_id %d\n",inst,CC_id);
laurent's avatar
laurent committed
1288
#endif
1289
      LOG_I(PHY,"Registering with MAC interface module\n");
1290 1291 1292
      AssertFatal((eNB->if_inst         = IF_Module_init(inst))!=NULL,"Cannot register interface");
      eNB->if_inst->schedule_response   = schedule_response;
      eNB->if_inst->PHY_config_req      = phy_config_request;
1293 1294
      memset((void *)&eNB->UL_INFO,0,sizeof(eNB->UL_INFO));
      memset((void *)&eNB->Sched_INFO,0,sizeof(eNB->Sched_INFO));
1295
      LOG_I(PHY,"Setting indication lists\n");
1296 1297 1298 1299
      eNB->UL_INFO.rx_ind.rx_indication_body.rx_pdu_list   = eNB->rx_pdu_list;
      eNB->UL_INFO.crc_ind.crc_indication_body.crc_pdu_list = eNB->crc_pdu_list;
      eNB->UL_INFO.sr_ind.sr_indication_body.sr_pdu_list = eNB->sr_pdu_list;
      eNB->UL_INFO.harq_ind.harq_indication_body.harq_pdu_list = eNB->harq_pdu_list;
1300 1301
      eNB->UL_INFO.cqi_ind.cqi_pdu_list = eNB->cqi_pdu_list;
      eNB->UL_INFO.cqi_ind.cqi_raw_pdu_list = eNB->cqi_raw_pdu_list;
1302
      eNB->prach_energy_counter = 0;
1303
    }
1304
  }
1305

1306
  LOG_I(PHY,"[lte-softmodem.c] eNB structure allocated\n");
1307 1308 1309
}


1310
void stop_eNB(int nb_inst) {
1311
  for (int inst=0; inst<nb_inst; inst++) {
1312
    LOG_I(PHY,"Killing eNB %d processing threads\n",inst);
1313 1314
    kill_eNB_proc(inst);
  }
1315
}