eNB_scheduler_dlsch.c 90.5 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

/*! \file eNB_scheduler_dlsch.c
 * \brief procedures related to eNB for the DLSCH transport channel
24
 * \author  Navid Nikaein and Raymond Knopp
25
 * \date 2010 - 2014
26
 * \email: navid.nikaein@eurecom.fr
27
 * \version 1.0
28 29 30 31
 * @ingroup _mac

 */

32
#define _GNU_SOURCE
33

34 35 36
#include "LAYER2/MAC/mac.h"
#include "LAYER2/MAC/mac_proto.h"
#include "LAYER2/MAC/mac_extern.h"
37 38
#include "common/utils/LOG/log.h"
#include "common/utils/LOG/vcd_signal_dumper.h"
39 40 41
#include "UTIL/OPT/opt.h"
#include "OCG.h"
#include "OCG_extern.h"
42
#include "PHY/LTE_TRANSPORT/transport_common_proto.h"
43

44
#include "RRC/LTE/rrc_extern.h"
45 46 47 48 49
#include "RRC/L2_INTERFACE/openair_rrc_L2_interface.h"

//#include "LAYER2/MAC/pre_processor.c"
#include "pdcp.h"

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
50
#include "SIMULATION/TOOLS/sim.h" // for taus
51 52

#include "assertions.h"
53

54
#if defined(ENABLE_ITTI)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
55
  #include "intertask_interface.h"
56 57
#endif

58 59
#include <dlfcn.h>

Cedric Roux's avatar
Cedric Roux committed
60 61
#include "T.h"

62
#define ENABLE_MAC_PAYLOAD_DEBUG
knopp's avatar
knopp committed
63
//#define DEBUG_eNB_SCHEDULER 1
64

65
#include "common/ran_context.h"
66
extern RAN_CONTEXT_t RC;
67
extern uint8_t nfapi_mode;
68

69 70
//------------------------------------------------------------------------------
void
71
add_ue_dlsch_info(module_id_t module_idP,
72 73
                  int CC_id,
                  int UE_id, sub_frame_t subframeP, UE_DLSCH_STATUS status)
74
//------------------------------------------------------------------------------
75
{
76
  //LOG_D(MAC, "%s(module_idP:%d, CC_id:%d, UE_id:%d, subframeP:%d, status:%d) serving_num:%d rnti:%x\n", __FUNCTION__, module_idP, CC_id, UE_id, subframeP, status, eNB_dlsch_info[module_idP][CC_id][UE_id].serving_num, UE_RNTI(module_idP,UE_id));
knopp's avatar
knopp committed
77
  eNB_dlsch_info[module_idP][CC_id][UE_id].rnti =
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
78
    UE_RNTI(module_idP, UE_id);
knopp's avatar
knopp committed
79 80 81 82
  //  eNB_dlsch_info[module_idP][CC_id][ue_mod_idP].weight           = weight;
  eNB_dlsch_info[module_idP][CC_id][UE_id].subframe = subframeP;
  eNB_dlsch_info[module_idP][CC_id][UE_id].status = status;
  eNB_dlsch_info[module_idP][CC_id][UE_id].serving_num++;
83 84
}

85 86
//------------------------------------------------------------------------------
int
87
schedule_next_dlue(module_id_t module_idP, int CC_id,
88
                   sub_frame_t subframeP)
89
//------------------------------------------------------------------------------
90
{
knopp's avatar
knopp committed
91 92
  int next_ue;
  UE_list_t *UE_list = &RC.mac[module_idP]->UE_list;
93

knopp's avatar
knopp committed
94 95 96
  for (next_ue = UE_list->head; next_ue >= 0;
       next_ue = UE_list->next[next_ue]) {
    if (eNB_dlsch_info[module_idP][CC_id][next_ue].status ==
97
        S_DL_WAITING) {
knopp's avatar
knopp committed
98
      return next_ue;
99
    }
knopp's avatar
knopp committed
100
  }
101

knopp's avatar
knopp committed
102 103 104 105
  for (next_ue = UE_list->head; next_ue >= 0;
       next_ue = UE_list->next[next_ue]) {
    if (eNB_dlsch_info[module_idP][CC_id][next_ue].status == S_DL_BUFFERED) {
      eNB_dlsch_info[module_idP][CC_id][next_ue].status = S_DL_WAITING;
106
    }
knopp's avatar
knopp committed
107
  }
knopp's avatar
knopp committed
108

109
  return (-1);        //next_ue;
110 111
}

112
//------------------------------------------------------------------------------
113
int
114
generate_dlsch_header(unsigned char *mac_header,
115 116 117 118 119 120 121 122
                      unsigned char num_sdus,
                      unsigned short *sdu_lengths,
                      unsigned char *sdu_lcids,
                      unsigned char drx_cmd,
                      unsigned short timing_advance_cmd,
                      unsigned char *ue_cont_res_id,
                      unsigned char short_padding,
                      unsigned short post_padding)
123
//------------------------------------------------------------------------------
124
{
knopp's avatar
knopp committed
125 126 127 128
  SCH_SUBHEADER_FIXED *mac_header_ptr = (SCH_SUBHEADER_FIXED *) mac_header;
  uint8_t first_element = 0, last_size = 0, i;
  uint8_t mac_header_control_elements[16], *ce_ptr;
  ce_ptr = &mac_header_control_elements[0];
129

knopp's avatar
knopp committed
130
  // compute header components
131

knopp's avatar
knopp committed
132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147
  if ((short_padding == 1) || (short_padding == 2)) {
    mac_header_ptr->R = 0;
    mac_header_ptr->E = 0;
    mac_header_ptr->LCID = SHORT_PADDING;
    first_element = 1;
    last_size = 1;
  }

  if (short_padding == 2) {
    mac_header_ptr->E = 1;
    mac_header_ptr++;
    mac_header_ptr->R = 0;
    mac_header_ptr->E = 0;
    mac_header_ptr->LCID = SHORT_PADDING;
    last_size = 1;
  }
148

knopp's avatar
knopp committed
149 150 151 152 153 154
  if (drx_cmd != 255) {
    if (first_element > 0) {
      mac_header_ptr->E = 1;
      mac_header_ptr++;
    } else {
      first_element = 1;
155
    }
156

knopp's avatar
knopp committed
157 158 159 160 161
    mac_header_ptr->R = 0;
    mac_header_ptr->E = 0;
    mac_header_ptr->LCID = DRX_CMD;
    last_size = 1;
  }
162

knopp's avatar
knopp committed
163 164 165 166 167 168
  if (timing_advance_cmd != 31) {
    if (first_element > 0) {
      mac_header_ptr->E = 1;
      mac_header_ptr++;
    } else {
      first_element = 1;
knopp's avatar
knopp committed
169
    }
170

knopp's avatar
knopp committed
171 172 173 174 175 176 177
    mac_header_ptr->R = 0;
    mac_header_ptr->E = 0;
    mac_header_ptr->LCID = TIMING_ADV_CMD;
    last_size = 1;
    //    msg("last_size %d,mac_header_ptr %p\n",last_size,mac_header_ptr);
    ((TIMING_ADVANCE_CMD *) ce_ptr)->R = 0;
    AssertFatal(timing_advance_cmd < 64,
178 179
                "timing_advance_cmd %d > 63\n", timing_advance_cmd);
    ((TIMING_ADVANCE_CMD *) ce_ptr)->TA = timing_advance_cmd;    //(timing_advance_cmd+31)&0x3f;
knopp's avatar
knopp committed
180
    LOG_D(MAC, "timing advance =%d (%d)\n", timing_advance_cmd,
181
          ((TIMING_ADVANCE_CMD *) ce_ptr)->TA);
knopp's avatar
knopp committed
182 183 184
    ce_ptr += sizeof(TIMING_ADVANCE_CMD);
    //msg("offset %d\n",ce_ptr-mac_header_control_elements);
  }
185

knopp's avatar
knopp committed
186 187 188 189
  if (ue_cont_res_id) {
    if (first_element > 0) {
      mac_header_ptr->E = 1;
      /*
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
190 191 192 193
      printf("[eNB][MAC] last subheader : %x (R%d,E%d,LCID%d)\n",*(unsigned char*)mac_header_ptr,
      ((SCH_SUBHEADER_FIXED *)mac_header_ptr)->R,
      ((SCH_SUBHEADER_FIXED *)mac_header_ptr)->E,
      ((SCH_SUBHEADER_FIXED *)mac_header_ptr)->LCID);
knopp's avatar
knopp committed
194 195 196 197
      */
      mac_header_ptr++;
    } else {
      first_element = 1;
knopp's avatar
knopp committed
198
    }
199

knopp's avatar
knopp committed
200 201 202 203 204
    mac_header_ptr->R = 0;
    mac_header_ptr->E = 0;
    mac_header_ptr->LCID = UE_CONT_RES;
    last_size = 1;
    LOG_T(MAC,
205 206 207
          "[eNB ][RAPROC] Generate contention resolution msg: %x.%x.%x.%x.%x.%x\n",
          ue_cont_res_id[0], ue_cont_res_id[1], ue_cont_res_id[2],
          ue_cont_res_id[3], ue_cont_res_id[4], ue_cont_res_id[5]);
knopp's avatar
knopp committed
208 209 210 211
    memcpy(ce_ptr, ue_cont_res_id, 6);
    ce_ptr += 6;
    // msg("(cont_res) : offset %d\n",ce_ptr-mac_header_control_elements);
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
212

knopp's avatar
knopp committed
213 214 215 216
  //msg("last_size %d,mac_header_ptr %p\n",last_size,mac_header_ptr);

  for (i = 0; i < num_sdus; i++) {
    LOG_T(MAC, "[eNB] Generate DLSCH header num sdu %d len sdu %d\n",
217
          num_sdus, sdu_lengths[i]);
knopp's avatar
knopp committed
218 219 220 221

    if (first_element > 0) {
      mac_header_ptr->E = 1;
      /*msg("last subheader : %x (R%d,E%d,LCID%d)\n",*(unsigned char*)mac_header_ptr,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
222 223 224
      ((SCH_SUBHEADER_FIXED *)mac_header_ptr)->R,
      ((SCH_SUBHEADER_FIXED *)mac_header_ptr)->E,
      ((SCH_SUBHEADER_FIXED *)mac_header_ptr)->LCID);
knopp's avatar
knopp committed
225 226 227 228 229
      */
      mac_header_ptr += last_size;
      //msg("last_size %d,mac_header_ptr %p\n",last_size,mac_header_ptr);
    } else {
      first_element = 1;
knopp's avatar
knopp committed
230
    }
231

knopp's avatar
knopp committed
232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247
    if (sdu_lengths[i] < 128) {
      ((SCH_SUBHEADER_SHORT *) mac_header_ptr)->R = 0;
      ((SCH_SUBHEADER_SHORT *) mac_header_ptr)->E = 0;
      ((SCH_SUBHEADER_SHORT *) mac_header_ptr)->F = 0;
      ((SCH_SUBHEADER_SHORT *) mac_header_ptr)->LCID = sdu_lcids[i];
      ((SCH_SUBHEADER_SHORT *) mac_header_ptr)->L = (unsigned char) sdu_lengths[i];
      last_size = 2;
    } else {
      ((SCH_SUBHEADER_LONG *) mac_header_ptr)->R = 0;
      ((SCH_SUBHEADER_LONG *) mac_header_ptr)->E = 0;
      ((SCH_SUBHEADER_LONG *) mac_header_ptr)->F = 1;
      ((SCH_SUBHEADER_LONG *) mac_header_ptr)->LCID = sdu_lcids[i];
      ((SCH_SUBHEADER_LONG *) mac_header_ptr)->L_MSB = ((unsigned short) sdu_lengths[i] >> 8) & 0x7f;
      ((SCH_SUBHEADER_LONG *) mac_header_ptr)->L_LSB = (unsigned short) sdu_lengths[i] & 0xff;
      ((SCH_SUBHEADER_LONG *) mac_header_ptr)->padding = 0x00;
      last_size = 3;
248
#ifdef DEBUG_HEADER_PARSING
knopp's avatar
knopp committed
249
      LOG_D(MAC,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
250 251 252 253
            "[eNB] generate long sdu, size %x (MSB %x, LSB %x)\n",
            sdu_lengths[i],
            ((SCH_SUBHEADER_LONG *) mac_header_ptr)->L_MSB,
            ((SCH_SUBHEADER_LONG *) mac_header_ptr)->L_LSB);
254
#endif
knopp's avatar
knopp committed
255
    }
knopp's avatar
knopp committed
256
  }
257

knopp's avatar
knopp committed
258
  /*
259

knopp's avatar
knopp committed
260
    printf("last_size %d,mac_header_ptr %p\n",last_size,mac_header_ptr);
261

knopp's avatar
knopp committed
262 263 264 265
    printf("last subheader : %x (R%d,E%d,LCID%d)\n",*(unsigned char*)mac_header_ptr,
    ((SCH_SUBHEADER_FIXED *)mac_header_ptr)->R,
    ((SCH_SUBHEADER_FIXED *)mac_header_ptr)->E,
    ((SCH_SUBHEADER_FIXED *)mac_header_ptr)->LCID);
266 267


knopp's avatar
knopp committed
268 269 270 271 272
    if (((SCH_SUBHEADER_FIXED*)mac_header_ptr)->LCID < UE_CONT_RES) {
    if (((SCH_SUBHEADER_SHORT*)mac_header_ptr)->F == 0)
    printf("F = 0, sdu len (L field) %d\n",(((SCH_SUBHEADER_SHORT*)mac_header_ptr)->L));
    else
    printf("F = 1, sdu len (L field) %d\n",(((SCH_SUBHEADER_LONG*)mac_header_ptr)->L));
273
    }
knopp's avatar
knopp committed
274
  */
275
  if (post_padding > 0) {    // we have lots of padding at the end of the packet
knopp's avatar
knopp committed
276 277 278 279 280 281 282
    mac_header_ptr->E = 1;
    mac_header_ptr += last_size;
    // add a padding element
    mac_header_ptr->R = 0;
    mac_header_ptr->E = 0;
    mac_header_ptr->LCID = SHORT_PADDING;
    mac_header_ptr++;
283
  } else {            // no end of packet padding
knopp's avatar
knopp committed
284 285 286
    // last SDU subhead is of fixed type (sdu length implicitly to be computed at UE)
    mac_header_ptr++;
  }
287

knopp's avatar
knopp committed
288
  //msg("After subheaders %d\n",(uint8_t*)mac_header_ptr - mac_header);
289

knopp's avatar
knopp committed
290 291 292
  if ((ce_ptr - mac_header_control_elements) > 0) {
    // printf("Copying %d bytes for control elements\n",ce_ptr-mac_header_control_elements);
    memcpy((void *) mac_header_ptr, mac_header_control_elements,
293
           ce_ptr - mac_header_control_elements);
knopp's avatar
knopp committed
294
    mac_header_ptr +=
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
295
      (unsigned char) (ce_ptr - mac_header_control_elements);
knopp's avatar
knopp committed
296 297
  }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
298
  //msg("After CEs %d\n",(uint8_t*)mac_header_ptr - mac_header);
knopp's avatar
knopp committed
299
  return ((unsigned char *) mac_header_ptr - mac_header);
300 301
}

302 303
//------------------------------------------------------------------------------
void
304
set_ul_DAI(int module_idP, int UE_idP, int CC_idP, int frameP,
305
           int subframeP)
306
//------------------------------------------------------------------------------
307
{
knopp's avatar
knopp committed
308 309 310 311
  eNB_MAC_INST *eNB = RC.mac[module_idP];
  UE_list_t *UE_list = &eNB->UE_list;
  unsigned char DAI;
  COMMON_channels_t *cc = &eNB->common_channels[CC_idP];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
312

313
  if (cc->tdd_Config != NULL) {    //TDD
knopp's avatar
knopp committed
314 315
    DAI = (UE_list->UE_template[CC_idP][UE_idP].DAI - 1) & 3;
    LOG_D(MAC,
316 317
          "[eNB %d] CC_id %d Frame %d, subframe %d: DAI %d for UE %d\n",
          module_idP, CC_idP, frameP, subframeP, DAI, UE_idP);
knopp's avatar
knopp committed
318 319 320
    // Save DAI for Format 0 DCI

    switch (cc->tdd_Config->subframeAssignment) {
321
      case 0:
322
        //      if ((subframeP==0)||(subframeP==1)||(subframeP==5)||(subframeP==6))
323
        break;
324

325 326
      case 1:
        switch (subframeP) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
327 328 329 330
          case 0:
          case 1:
            UE_list->UE_template[CC_idP][UE_idP].DAI_ul[7] = DAI;
            break;
331

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
332 333 334
          case 4:
            UE_list->UE_template[CC_idP][UE_idP].DAI_ul[8] = DAI;
            break;
335

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
336 337 338 339
          case 5:
          case 6:
            UE_list->UE_template[CC_idP][UE_idP].DAI_ul[2] = DAI;
            break;
340

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
341 342 343
          case 9:
            UE_list->UE_template[CC_idP][UE_idP].DAI_ul[3] = DAI;
            break;
344
        }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
345

346
        break;
347

348 349 350 351
      case 2:
        //      if ((subframeP==3)||(subframeP==8))
        //  UE_list->UE_template[CC_idP][UE_idP].DAI_ul = DAI;
        break;
352

353
      case 3:
354

355 356 357 358 359 360 361 362 363 364
        //if ((subframeP==6)||(subframeP==8)||(subframeP==0)) {
        //  LOG_D(MAC,"schedule_ue_spec: setting UL DAI to %d for subframeP %d => %d\n",DAI,subframeP, ((subframeP+8)%10)>>1);
        //  UE_list->UE_template[CC_idP][UE_idP].DAI_ul[((subframeP+8)%10)>>1] = DAI;
        //}
        switch (subframeP) {
          case 5:
          case 6:
          case 1:
            UE_list->UE_template[CC_idP][UE_idP].DAI_ul[2] = DAI;
            break;
365

366 367 368 369
          case 7:
          case 8:
            UE_list->UE_template[CC_idP][UE_idP].DAI_ul[3] = DAI;
            break;
knopp's avatar
knopp committed
370

371 372 373 374
          case 9:
          case 0:
            UE_list->UE_template[CC_idP][UE_idP].DAI_ul[4] = DAI;
            break;
knopp's avatar
knopp committed
375

376 377 378
          default:
            break;
        }
knopp's avatar
knopp committed
379

380
        break;
knopp's avatar
knopp committed
381

382 383 384 385
      case 4:
        //      if ((subframeP==8)||(subframeP==9))
        //  UE_list->UE_template[CC_idP][UE_idP].DAI_ul = DAI;
        break;
knopp's avatar
knopp committed
386

387 388 389 390
      case 5:
        //      if (subframeP==8)
        //  UE_list->UE_template[CC_idP][UE_idP].DAI_ul = DAI;
        break;
knopp's avatar
knopp committed
391

392 393 394 395
      case 6:
        //      if ((subframeP==1)||(subframeP==4)||(subframeP==6)||(subframeP==9))
        //  UE_list->UE_template[CC_idP][UE_idP].DAI_ul = DAI;
        break;
knopp's avatar
knopp committed
396

397 398
      default:
        break;
knopp's avatar
knopp committed
399
    }
knopp's avatar
knopp committed
400
  }
knopp's avatar
knopp committed
401 402
}

403 404
//------------------------------------------------------------------------------
void
405
schedule_dlsch(module_id_t module_idP, frame_t frameP, sub_frame_t subframeP, int *mbsfn_flag) {
406
  int i = 0;
407
  slice_info_t *sli = &RC.mac[module_idP]->slice_info;
408
  memset(sli->rballoc_sub, 0, sizeof(sli->rballoc_sub));
409

410
  for (i = 0; i < sli->n_dl; i++) {
411
    // Run each enabled slice-specific schedulers one by one
412
    sli->dl[i].sched_cb(module_idP, i, frameP, subframeP, mbsfn_flag/*, dl_info*/);
413 414
  }
}
415

416 417
// changes to pre-processor for eMTC

418 419
//------------------------------------------------------------------------------
void
420
schedule_ue_spec(module_id_t module_idP, int slice_idxP,
421
                 frame_t frameP, sub_frame_t subframeP, int *mbsfn_flag)
422
//------------------------------------------------------------------------------
423
{
424
  int CC_id;
knopp's avatar
knopp committed
425
  int UE_id;
426
  int aggregation;
knopp's avatar
knopp committed
427
  mac_rlc_status_resp_t rlc_status;
428 429 430 431 432 433
  int ta_len = 0;
  unsigned char sdu_lcids[NB_RB_MAX];
  int lcid, offset, num_sdus = 0;
  int nb_rb, nb_rb_temp, nb_available_rb;
  uint16_t sdu_lengths[NB_RB_MAX];
  int TBS, j, rnti, padding = 0, post_padding = 0;
knopp's avatar
knopp committed
434
  unsigned char dlsch_buffer[MAX_DLSCH_PAYLOAD_BYTES];
435 436
  int round = 0;
  int harq_pid = 0;
knopp's avatar
knopp committed
437
  eNB_UE_STATS *eNB_UE_stats = NULL;
438
  int sdu_length_total = 0;
knopp's avatar
knopp committed
439 440 441 442 443
  eNB_MAC_INST *eNB = RC.mac[module_idP];
  COMMON_channels_t *cc = eNB->common_channels;
  UE_list_t *UE_list = &eNB->UE_list;
  int continue_flag = 0;
  int32_t normalized_rx_power, target_rx_power;
444
  int tpc = 1;
knopp's avatar
knopp committed
445 446 447
  UE_sched_ctrl *ue_sched_ctl;
  int mcs;
  int i;
448 449 450 451
  int min_rb_unit[NFAPI_CC_MAX];
  int N_RB_DL[NFAPI_CC_MAX];
  int total_nb_available_rb[NFAPI_CC_MAX];
  int N_RBG[NFAPI_CC_MAX];
knopp's avatar
knopp committed
452 453 454 455
  nfapi_dl_config_request_body_t *dl_req;
  nfapi_dl_config_request_pdu_t *dl_config_pdu;
  int tdd_sfa;
  int ta_update;
456 457
  int header_length_last;
  int header_length_total;
458
  rrc_eNB_ue_context_t *ue_contextP = NULL;
knopp's avatar
knopp committed
459
  start_meas(&eNB->schedule_dlsch);
460
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_SCHEDULE_DLSCH, VCD_FUNCTION_IN);
461

knopp's avatar
knopp committed
462 463 464
  // for TDD: check that we have to act here, otherwise return
  if (cc[0].tdd_Config) {
    tdd_sfa = cc[0].tdd_Config->subframeAssignment;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
465

knopp's avatar
knopp committed
466
    switch (subframeP) {
467 468 469
      case 0:
        // always continue
        break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
470

471 472 473
      case 1:
        return;
        break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
474

475 476 477
      case 2:
        return;
        break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
478

479 480 481
      case 3:
        if ((tdd_sfa != 2) && (tdd_sfa != 5))
          return;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
482

483
        break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
484

485 486 487 488
      case 4:
        if ((tdd_sfa != 1) && (tdd_sfa != 2) && (tdd_sfa != 4)
            && (tdd_sfa != 5))
          return;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
489

490
        break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
491

492 493
      case 5:
        break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
494

495 496 497 498
      case 6:
      case 7:
        if ((tdd_sfa != 3) && (tdd_sfa != 4) && (tdd_sfa != 5))
          return;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
499

500
        break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
501

502 503 504 505
      case 8:
        if ((tdd_sfa != 2) && (tdd_sfa != 3) && (tdd_sfa != 4)
            && (tdd_sfa != 5))
          return;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
506

507
        break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
508

509 510 511
      case 9:
        if (tdd_sfa == 0)
          return;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
512

513
        break;
514
    }
knopp's avatar
knopp committed
515
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
516

knopp's avatar
knopp committed
517 518
  //weight = get_ue_weight(module_idP,UE_id);
  aggregation = 2;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
519

520
  for (CC_id = 0; CC_id < RC.nb_mac_CC[module_idP]; CC_id++) {
knopp's avatar
knopp committed
521 522 523 524
    N_RB_DL[CC_id] = to_prb(cc[CC_id].mib->message.dl_Bandwidth);
    min_rb_unit[CC_id] = get_min_rb_unit(module_idP, CC_id);
    // get number of PRBs less those used by common channels
    total_nb_available_rb[CC_id] = N_RB_DL[CC_id];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
525

knopp's avatar
knopp committed
526 527
    for (i = 0; i < N_RB_DL[CC_id]; i++)
      if (cc[CC_id].vrb_map[i] != 0)
528
        total_nb_available_rb[CC_id]--;
knopp's avatar
knopp committed
529 530 531 532 533 534 535 536 537

    N_RBG[CC_id] = to_rbg(cc[CC_id].mib->message.dl_Bandwidth);
    // store the global enb stats:
    eNB->eNB_stats[CC_id].num_dlactive_UEs = UE_list->num_UEs;
    eNB->eNB_stats[CC_id].available_prbs = total_nb_available_rb[CC_id];
    eNB->eNB_stats[CC_id].total_available_prbs += total_nb_available_rb[CC_id];
    eNB->eNB_stats[CC_id].dlsch_bytes_tx = 0;
    eNB->eNB_stats[CC_id].dlsch_pdus_tx = 0;
  }
538

539 540 541
  // CALLING Pre_Processor for downlink scheduling
  // (Returns estimation of RBs required by each UE and the allocation on sub-band)
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_DLSCH_PREPROCESSOR, VCD_FUNCTION_IN);
knopp's avatar
knopp committed
542
  start_meas(&eNB->schedule_dlsch_preprocessor);
543
  dlsch_scheduler_pre_processor(module_idP,
544
                                slice_idxP,
545 546
                                frameP,
                                subframeP,
547 548
                                mbsfn_flag,
                                eNB->slice_info.rballoc_sub);
knopp's avatar
knopp committed
549
  stop_meas(&eNB->schedule_dlsch_preprocessor);
550
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_DLSCH_PREPROCESSOR, VCD_FUNCTION_OUT);
551

552
  //RC.mac[module_idP]->slice_info.slice_counter--;
553
  // Do the multiplexing and actual allocation only when all slices have been pre-processed.
554
  //if (RC.mac[module_idP]->slice_info.slice_counter > 0) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
555 556 557
  //stop_meas(&eNB->schedule_dlsch);
  //VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_SCHEDULE_DLSCH, VCD_FUNCTION_OUT);
  //return;
558
  //}
559

560
  if (RC.mac[module_idP]->slice_info.interslice_share_active) {
561
    dlsch_scheduler_interslice_multiplexing(module_idP, frameP, subframeP, eNB->slice_info.rballoc_sub);
562 563 564
    /* the interslice multiplexing re-sorts the UE_list for the slices it tries
     * to multiplex, so we need to sort it for the current slice again */
    sort_UEs(module_idP, slice_idxP, frameP, subframeP);
565
  }
566

567
  for (CC_id = 0; CC_id < RC.nb_mac_CC[module_idP]; CC_id++) {
knopp's avatar
knopp committed
568 569
    LOG_D(MAC, "doing schedule_ue_spec for CC_id %d\n", CC_id);
    dl_req = &eNB->DL_req[CC_id].dl_config_request_body;
570

knopp's avatar
knopp committed
571
    if (mbsfn_flag[CC_id] > 0)
knopp's avatar
knopp committed
572
      continue;
573

574
    for (UE_id = UE_list->head; UE_id >= 0; UE_id = UE_list->next[UE_id]) {
Robert Schmidt's avatar
Robert Schmidt committed
575
      LOG_D(MAC, "doing schedule_ue_spec for CC_id %d UE %d\n", CC_id, UE_id);
576
      continue_flag = 0; // reset the flag to allow allocation for the remaining UEs
knopp's avatar
knopp committed
577
      rnti = UE_RNTI(module_idP, UE_id);
578
      eNB_UE_stats = &UE_list->eNB_UE_stats[CC_id][UE_id];
579
      ue_sched_ctl = &UE_list->UE_sched_ctrl[UE_id];
580

knopp's avatar
knopp committed
581
      if (rnti == NOT_A_RNTI) {
582 583
        LOG_D(MAC, "Cannot find rnti for UE_id %d (num_UEs %d)\n", UE_id, UE_list->num_UEs);
        continue_flag = 1;
584
      }
585

knopp's avatar
knopp committed
586
      if (eNB_UE_stats == NULL) {
587 588
        LOG_D(MAC, "[eNB] Cannot find eNB_UE_stats\n");
        continue_flag = 1;
589
      }
590

Robert Schmidt's avatar
Robert Schmidt committed
591 592 593 594 595
      if (!ue_dl_slice_membership(module_idP, UE_id, slice_idxP)) {
        LOG_D(MAC, "UE%d is not part of slice %d ID %d\n",
              UE_id, slice_idxP, RC.mac[module_idP]->slice_info.dl[slice_idxP].id);
        /* prevent execution of add_ue_dlsch_info(), it is done by the other
         * slice */
596
        continue;
597
      }
598

knopp's avatar
knopp committed
599
      if (continue_flag != 1) {
600 601 602 603 604 605 606 607
        switch (get_tmode(module_idP, CC_id, UE_id)) {
          case 1:
          case 2:
          case 7:
            aggregation = get_aggregation(get_bw_index(module_idP, CC_id),
                                          ue_sched_ctl->dl_cqi[CC_id],
                                          format1);
            break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
608

609 610 611 612 613
          case 3:
            aggregation = get_aggregation(get_bw_index(module_idP, CC_id),
                                          ue_sched_ctl->dl_cqi[CC_id],
                                          format2A);
            break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
614

615 616 617 618
          default:
            LOG_W(MAC, "Unsupported transmission mode %d\n", get_tmode(module_idP, CC_id, UE_id));
            aggregation = 2;
        }
knopp's avatar
knopp committed
619
      }
620

knopp's avatar
knopp committed
621
      /* if (continue_flag != 1 */
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
622 623 624 625 626 627 628
      if ((ue_sched_ctl->pre_nb_available_rbs[CC_id] == 0) || // no RBs allocated
          CCE_allocation_infeasible(module_idP, CC_id, 1, subframeP,
                                    aggregation, rnti)) {
        LOG_D(MAC,
              "[eNB %d] Frame %d : no RB allocated for UE %d on CC_id %d: continue \n",
              module_idP, frameP, UE_id, CC_id);
        continue_flag = 1;  //to next user (there might be rbs availiable for other UEs in TM5
knopp's avatar
knopp committed
629
      }
knopp's avatar
knopp committed
630

631 632 633 634 635 636 637 638 639
      // If TDD
      if (cc[CC_id].tdd_Config != NULL) {    //TDD
        set_ue_dai(subframeP,
                   UE_id,
                   CC_id,
                   cc[CC_id].tdd_Config->subframeAssignment,
                   UE_list);
        // update UL DAI after DLSCH scheduling
        set_ul_DAI(module_idP, UE_id, CC_id, frameP, subframeP);
knopp's avatar
knopp committed
640
      }
641

knopp's avatar
knopp committed
642
      if (continue_flag == 1) {
643 644
        add_ue_dlsch_info(module_idP, CC_id, UE_id, subframeP, S_DL_NONE);
        continue;
knopp's avatar
knopp committed
645
      }
Cedric Roux's avatar
Cedric Roux committed
646

knopp's avatar
knopp committed
647
      nb_available_rb = ue_sched_ctl->pre_nb_available_rbs[CC_id];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
648
      harq_pid = frame_subframe2_dl_harq_pid(cc->tdd_Config,frameP,subframeP);
knopp's avatar
knopp committed
649
      round = ue_sched_ctl->round[CC_id][harq_pid];
650
      UE_list->eNB_UE_stats[CC_id][UE_id].crnti = rnti;
knopp's avatar
knopp committed
651
      UE_list->eNB_UE_stats[CC_id][UE_id].rrc_status = mac_eNB_get_rrc_status(module_idP, rnti);
652
      UE_list->eNB_UE_stats[CC_id][UE_id].harq_pid = harq_pid;
knopp's avatar
knopp committed
653
      UE_list->eNB_UE_stats[CC_id][UE_id].harq_round = round;
654

Robert Schmidt's avatar
Robert Schmidt committed
655 656
      if (UE_list->eNB_UE_stats[CC_id][UE_id].rrc_status < RRC_CONNECTED) {
        LOG_D(MAC, "UE %d is not in RRC_CONNECTED\n", UE_id);
657
        continue;
Robert Schmidt's avatar
Robert Schmidt committed
658
      }
659

660
      header_length_total = 0;
knopp's avatar
knopp committed
661 662
      sdu_length_total = 0;
      num_sdus = 0;
663

knopp's avatar
knopp committed
664
      /*
665 666 667
      DevCheck(((eNB_UE_stats->dl_cqi < MIN_CQI_VALUE) ||
                (eNB_UE_stats->dl_cqi > MAX_CQI_VALUE)),
                eNB_UE_stats->dl_cqi, MIN_CQI_VALUE, MAX_CQI_VALUE);
knopp's avatar
knopp committed
668 669
      */
      if (nfapi_mode) {
670 671
        eNB_UE_stats->dlsch_mcs1 = 10; // cqi_to_mcs[ue_sched_ctl->dl_cqi[CC_id]];
      } else { // this operation is also done in the preprocessor
672
        eNB_UE_stats->dlsch_mcs1 = cmin(eNB_UE_stats->dlsch_mcs1,
673
                                        eNB->slice_info.dl[slice_idxP].maxmcs);  // cmin(eNB_UE_stats->dlsch_mcs1, openair_daq_vars.target_ue_dl_mcs);
knopp's avatar
knopp committed
674
      }
675

676 677
      // Store stats
      // UE_list->eNB_UE_stats[CC_id][UE_id].dl_cqi= eNB_UE_stats->dl_cqi;
678

679
      // Initializing the rb allocation indicator for each UE
knopp's avatar
knopp committed
680
      for (j = 0; j < N_RBG[CC_id]; j++) {
681
        UE_list->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j] = 0;
knopp's avatar
knopp committed
682 683 684
      }

      LOG_D(MAC,
685 686 687 688 689
            "[eNB %d] Frame %d: Scheduling UE %d on CC_id %d (rnti %x, harq_pid %d, round %d, rb %d, cqi %d, mcs %d, rrc %d)\n",
            module_idP, frameP, UE_id, CC_id, rnti, harq_pid, round,
            nb_available_rb, ue_sched_ctl->dl_cqi[CC_id],
            eNB_UE_stats->dlsch_mcs1,
            UE_list->eNB_UE_stats[CC_id][UE_id].rrc_status);
knopp's avatar
knopp committed
690 691 692

      /* process retransmission  */
      if (round != 8) {
693 694 695
        // get freq_allocation
        nb_rb = UE_list->UE_template[CC_id][UE_id].nb_rb[harq_pid];
        TBS = get_TBS_DL(UE_list->UE_template[CC_id][UE_id].oldmcs1[harq_pid], nb_rb);
knopp's avatar
knopp committed
696

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
697 698 699 700 701 702 703 704 705 706
        if (nb_rb <= nb_available_rb) {
          if (cc[CC_id].tdd_Config != NULL) {
            UE_list->UE_template[CC_id][UE_id].DAI++;
            update_ul_dci(module_idP, CC_id, rnti,
                          UE_list->UE_template[CC_id][UE_id].DAI, subframeP);
            LOG_D(MAC,
                  "DAI update: CC_id %d subframeP %d: UE %d, DAI %d\n",
                  CC_id, subframeP, UE_id,
                  UE_list->UE_template[CC_id][UE_id].DAI);
          }
knopp's avatar
knopp committed
707

708 709 710 711 712 713 714 715 716 717 718 719
          if (nb_rb == ue_sched_ctl->pre_nb_available_rbs[CC_id]) {
            for (j = 0; j < N_RBG[CC_id]; ++j) { // for indicating the rballoc for each sub-band
              UE_list->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j] = ue_sched_ctl->rballoc_sub_UE[CC_id][j];
            }
          } else {
            nb_rb_temp = nb_rb;
            j = 0;

            while ((nb_rb_temp > 0) && (j < N_RBG[CC_id])) {
              if (ue_sched_ctl->rballoc_sub_UE[CC_id][j] == 1) {
                if (UE_list->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j])
                  printf("WARN: rballoc_subband not free for retrans?\n");
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
720

721 722 723 724 725 726 727 728
                UE_list->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j] = ue_sched_ctl->rballoc_sub_UE[CC_id][j];

                if ((j == N_RBG[CC_id] - 1) && ((N_RB_DL[CC_id] == 25) || (N_RB_DL[CC_id] == 50))) {
                  nb_rb_temp = nb_rb_temp - min_rb_unit[CC_id] + 1;
                } else {
                  nb_rb_temp = nb_rb_temp - min_rb_unit[CC_id];
                }
              }
729

730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756
              j = j + 1;
            }
          }

          nb_available_rb -= nb_rb;
          /*
          eNB->mu_mimo_mode[UE_id].pre_nb_available_rbs = nb_rb;
          eNB->mu_mimo_mode[UE_id].dl_pow_off = ue_sched_ctl->dl_pow_off[CC_id];

          for(j = 0; j < N_RBG[CC_id]; ++j) {
            eNB->mu_mimo_mode[UE_id].rballoc_sub[j] = UE_list->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j];
          }
          */

          switch (get_tmode(module_idP, CC_id, UE_id)) {
            case 1:
            case 2:
            case 7:
            default:
              LOG_D(MAC, "retransmission DL_REQ: rnti:%x\n", rnti);
              dl_config_pdu = &dl_req->dl_config_pdu_list[dl_req->number_pdu];
              memset((void *) dl_config_pdu, 0, sizeof(nfapi_dl_config_request_pdu_t));
              dl_config_pdu->pdu_type = NFAPI_DL_CONFIG_DCI_DL_PDU_TYPE;
              dl_config_pdu->pdu_size = (uint8_t) (2 + sizeof(nfapi_dl_config_dci_dl_pdu));
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.tl.tag = NFAPI_DL_CONFIG_REQUEST_DCI_DL_PDU_REL8_TAG;
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.dci_format = NFAPI_DL_DCI_FORMAT_1;
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.aggregation_level =
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
757 758 759
                get_aggregation(get_bw_index(module_idP, CC_id),
                                ue_sched_ctl->dl_cqi[CC_id],
                                format1);
760 761 762 763 764 765 766 767 768 769 770 771
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.rnti = rnti;
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.rnti_type = 1; // CRNTI: see Table 4-10 from SCF082 - nFAPI specifications
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.transmission_power = 6000; // equal to RS power
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.harq_process = harq_pid;
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.tpc = 1; // Don't adjust power when retransmitting
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.new_data_indicator_1 = UE_list->UE_template[CC_id][UE_id].oldNDI[harq_pid];
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.mcs_1 = UE_list->UE_template[CC_id][UE_id].oldmcs1[harq_pid];
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.redundancy_version_1 = round & 3;

              // TDD
              if (cc[CC_id].tdd_Config != NULL) {
                dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.downlink_assignment_index =
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
772
                  (UE_list->UE_template[CC_id][UE_id].DAI - 1) & 3;
773 774 775 776 777 778 779 780 781 782
                LOG_D(MAC,
                      "[eNB %d] Retransmission CC_id %d : harq_pid %d, round %d, dai %d, mcs %d\n",
                      module_idP, CC_id, harq_pid, round,
                      (UE_list->UE_template[CC_id][UE_id].DAI - 1),
                      UE_list->UE_template[CC_id][UE_id].oldmcs1[harq_pid]);
              } else {
                LOG_D(MAC,
                      "[eNB %d] Retransmission CC_id %d : harq_pid %d, round %d, mcs %d\n",
                      module_idP, CC_id, harq_pid, round,
                      UE_list->UE_template[CC_id][UE_id].oldmcs1[harq_pid]);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
783
              }
784

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
785 786 787 788 789 790 791
              if (!CCE_allocation_infeasible(module_idP, CC_id, 1, subframeP,
                                             dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.aggregation_level, rnti)) {
                dl_req->number_dci++;
                dl_req->number_pdu++;
                dl_req->tl.tag = NFAPI_DL_CONFIG_REQUEST_BODY_TAG;
                eNB->DL_req[CC_id].sfn_sf = frameP<<4 | subframeP;
                eNB->DL_req[CC_id].header.message_id = NFAPI_DL_CONFIG_REQUEST;
792 793 794 795 796 797 798 799 800 801 802 803
                fill_nfapi_dlsch_config(eNB, dl_req, TBS, -1,
                                        /* retransmission, no pdu_index */
                                        rnti, 0, // type 0 allocation from 7.1.6 in 36.213
                                        0,    // virtual_resource_block_assignment_flag, unused here
                                        0,    // resource_block_coding, to be filled in later
                                        getQm(UE_list->UE_template[CC_id][UE_id].oldmcs1[harq_pid]),
                                        round & 3, // redundancy version
                                        1,    // transport blocks
                                        0,    // transport block to codeword swap flag
                                        cc[CC_id].p_eNB == 1 ? 0 : 1,    // transmission_scheme
                                        1,    // number of layers
                                        1,    // number of subbands
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
804
                                        //                      uint8_t codebook_index,
805 806 807 808 809 810 811 812
                                        4,    // UE category capacity
                                        UE_list->UE_template[CC_id][UE_id].physicalConfigDedicated->pdsch_ConfigDedicated->p_a,
                                        0,    // delta_power_offset for TM5
                                        0,    // ngap
                                        0,    // nprb
                                        cc[CC_id].p_eNB == 1 ? 1 : 2,    // transmission mode
                                        0,    //number of PRBs treated as one subband, not used here
                                        0    // number of beamforming vectors, not used here
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
813
                                       );
814 815 816 817 818 819 820 821 822 823 824 825
                LOG_D(MAC,
                      "Filled NFAPI configuration for DCI/DLSCH %d, retransmission round %d\n",
                      eNB->pdu_index[CC_id], round);
                program_dlsch_acknak(module_idP, CC_id, UE_id, frameP, subframeP,
                                     dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.cce_idx);
                // No TX request for retransmission (check if null request for FAPI)
              } else {
                LOG_W(MAC,
                      "Frame %d, Subframe %d: Dropping DLSCH allocation for UE %d\%x, infeasible CCE allocation\n",
                      frameP, subframeP, UE_id, rnti);
              }
          }
knopp's avatar
knopp committed
826

827
          add_ue_dlsch_info(module_idP, CC_id, UE_id, subframeP, S_DL_SCHEDULED);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843
          //eNB_UE_stats->dlsch_trials[round]++;
          UE_list->eNB_UE_stats[CC_id][UE_id].num_retransmission += 1;
          UE_list->eNB_UE_stats[CC_id][UE_id].rbs_used_retx = nb_rb;
          UE_list->eNB_UE_stats[CC_id][UE_id].total_rbs_used_retx += nb_rb;
          UE_list->eNB_UE_stats[CC_id][UE_id].dlsch_mcs1 = eNB_UE_stats->dlsch_mcs1;
          UE_list->eNB_UE_stats[CC_id][UE_id].dlsch_mcs2 = eNB_UE_stats->dlsch_mcs1;
        } else {
          LOG_D(MAC,
                "[eNB %d] Frame %d CC_id %d : don't schedule UE %d, its retransmission takes more resources than we have\n",
                module_idP, frameP, CC_id, UE_id);
        }
      } else {    /* This is a potentially new SDU opportunity */
        rlc_status.bytes_in_buffer = 0;
        // Now check RLC information to compute number of required RBs
        // get maximum TBS size for RLC request
        TBS = get_TBS_DL(eNB_UE_stats->dlsch_mcs1, nb_available_rb);
844

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
845
        // add the length for  all the control elements (timing adv, drx, etc) : header + payload
846

847 848
        if (ue_sched_ctl->ta_timer == 0) {
          ta_update = ue_sched_ctl->ta_update;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
849

850 851 852
          /* if we send TA then set timer to not send it for a while */
          if (ta_update != 31)
            ue_sched_ctl->ta_timer = 20;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
853

854 855 856 857 858
          /* reset ta_update */
          ue_sched_ctl->ta_update = 31;
        } else {
          ta_update = 31;
        }
859

860
        ta_len = (ta_update != 31) ? 2 : 0;
knopp's avatar
knopp committed
861

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
862 863 864
        // RLC data on DCCH
        if (TBS - ta_len - header_length_total - sdu_length_total - 3 > 0) {
          rlc_status = mac_rlc_status_ind(module_idP, rnti, module_idP, frameP, subframeP, ENB_FLAG_YES, MBMS_FLAG_NO, DCCH,
865
                                          TBS - ta_len - header_length_total - sdu_length_total - 3
866
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
867
                                          ,0, 0
868
#endif
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
869
                                         );
870
          sdu_lengths[0] = 0;
knopp's avatar
knopp committed
871

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
872 873 874 875 876
          if (rlc_status.bytes_in_buffer > 0) {
            LOG_D(MAC, "[eNB %d] SFN/SF %d.%d, DL-DCCH->DLSCH CC_id %d, Requesting %d bytes from RLC (RRC message)\n",
                  module_idP, frameP, subframeP, CC_id,
                  TBS - ta_len - header_length_total - sdu_length_total - 3);
            sdu_lengths[0] = mac_rlc_data_req(module_idP, rnti, module_idP, frameP, ENB_FLAG_YES, MBMS_FLAG_NO, DCCH,
877
                                              TBS, //not used
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
878
                                              (char *)&dlsch_buffer[0]
879
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
880
                                              ,0, 0
881
#endif
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
882
                                             );
883
            pthread_mutex_lock(&rrc_release_freelist);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
884 885

            if((rrc_release_info.num_UEs > 0) && (rlc_am_mui.rrc_mui_num > 0)) {
886
              uint16_t release_total = 0;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
887 888 889

              for(uint16_t release_num = 0; release_num < NUMBER_OF_UE_MAX; release_num++) {
                if(rrc_release_info.RRC_release_ctrl[release_num].flag > 0) {
Wu Jing's avatar
Wu Jing committed
890
                  release_total++;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
891
                } else {
Wu Jing's avatar
Wu Jing committed
892 893 894
                  continue;
                }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
895 896 897 898
                if(rrc_release_info.RRC_release_ctrl[release_num].flag == 1) {
                  if(rrc_release_info.RRC_release_ctrl[release_num].rnti == rnti) {
                    for(uint16_t mui_num = 0; mui_num < rlc_am_mui.rrc_mui_num; mui_num++) {
                      if(rrc_release_info.RRC_release_ctrl[release_num].rrc_eNB_mui == rlc_am_mui.rrc_mui[mui_num]) {
Wu Jing's avatar
Wu Jing committed
899 900 901
                        rrc_release_info.RRC_release_ctrl[release_num].flag = 3;
                        LOG_D(MAC,"DLSCH Release send:index %d rnti %x mui %d mui_num %d flag 1->3\n",release_num,rnti,rlc_am_mui.rrc_mui[mui_num],mui_num);
                        break;
902 903
                      }
                    }
904 905
                  }
                }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
906 907 908 909 910

                if(rrc_release_info.RRC_release_ctrl[release_num].flag == 2) {
                  if(rrc_release_info.RRC_release_ctrl[release_num].rnti == rnti) {
                    for(uint16_t mui_num = 0; mui_num < rlc_am_mui.rrc_mui_num; mui_num++) {
                      if(rrc_release_info.RRC_release_ctrl[release_num].rrc_eNB_mui == rlc_am_mui.rrc_mui[mui_num]) {
911 912 913
                        rrc_release_info.RRC_release_ctrl[release_num].flag = 4;
                        LOG_D(MAC,"DLSCH Release send:index %d rnti %x mui %d mui_num %d flag 2->4\n",release_num,rnti,rlc_am_mui.rrc_mui[mui_num],mui_num);
                        break;
Wu Jing's avatar
Wu Jing committed
914
                      }
915
                    }
Wu Jing's avatar
Wu Jing committed
916
                  }
917
                }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
918

919 920 921 922
                if(release_total >= rrc_release_info.num_UEs)
                  break;
              }
            }
Xu Bo's avatar
Xu Bo committed
923

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
924
            pthread_mutex_unlock(&rrc_release_freelist);
Xu Bo's avatar
Xu Bo committed
925
            RA_t *ra = &eNB->common_channels[CC_id].ra[0];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
926

Xu Bo's avatar
Xu Bo committed
927
            for (uint8_t ra_ii = 0; ra_ii < NB_RA_PROC_MAX; ra_ii++) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
928 929 930
              if((ra[ra_ii].rnti == rnti) && (ra[ra_ii].state == MSGCRNTI)) {
                for(uint16_t mui_num = 0; mui_num < rlc_am_mui.rrc_mui_num; mui_num++) {
                  if(ra[ra_ii].crnti_rrc_mui == rlc_am_mui.rrc_mui[mui_num]) {
Wu Jing's avatar
Wu Jing committed
931 932 933 934 935
                    ra[ra_ii].crnti_harq_pid = harq_pid;
                    ra[ra_ii].state = MSGCRNTI_ACK;
                    break;
                  }
                }
Xu Bo's avatar
Xu Bo committed
936 937
              }
            }
938

939 940 941 942
            T(T_ENB_MAC_UE_DL_SDU, T_INT(module_idP),
              T_INT(CC_id), T_INT(rnti), T_INT(frameP),
              T_INT(subframeP), T_INT(harq_pid), T_INT(DCCH),
              T_INT(sdu_lengths[0]));
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
943 944 945 946
            LOG_D(MAC, "[eNB %d][DCCH] CC_id %d Got %d bytes from RLC\n",
                  module_idP, CC_id, sdu_lengths[0]);
            sdu_length_total = sdu_lengths[0];
            sdu_lcids[0] = DCCH;
947 948
            UE_list->eNB_UE_stats[CC_id][UE_id].lcid_sdu[0] = DCCH;
            UE_list->eNB_UE_stats[CC_id][UE_id].sdu_length_tx[DCCH] = sdu_lengths[0];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
949 950
            UE_list->eNB_UE_stats[CC_id][UE_id].num_pdu_tx[DCCH] += 1;
            UE_list->eNB_UE_stats[CC_id][UE_id].num_bytes_tx[DCCH] += sdu_lengths[0];
951 952
            header_length_last = 1 + 1 + (sdu_lengths[0] >= 128);
            header_length_total += header_length_last;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
953
            num_sdus = 1;
954
#ifdef DEBUG_eNB_SCHEDULER
955
            LOG_T(MAC,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
956 957
                  "[eNB %d][DCCH] CC_id %d Got %d bytes :",
                  module_idP, CC_id, sdu_lengths[0]);
958

959 960 961
            for (j = 0; j < sdu_lengths[0]; ++j) {
              LOG_T(MAC, "%x ", dlsch_buffer[j]);
            }
962

963
            LOG_T(MAC, "\n");
964
#endif
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
965 966
          }
        }
967

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
968 969 970
        // RLC data on DCCH1
        if (TBS - ta_len - header_length_total - sdu_length_total - 3 > 0) {
          rlc_status = mac_rlc_status_ind(module_idP, rnti, module_idP, frameP, subframeP, ENB_FLAG_YES, MBMS_FLAG_NO, DCCH + 1,
971
                                          TBS - ta_len - header_length_total - sdu_length_total - 3
972
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
973
                                          ,0, 0
974 975
#endif
                                         );
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
976 977 978 979 980 981 982 983 984 985
          // DCCH SDU
          sdu_lengths[num_sdus] = 0;

          if (rlc_status.bytes_in_buffer > 0) {
            LOG_D(MAC, "[eNB %d], Frame %d, DCCH1->DLSCH, CC_id %d, Requesting %d bytes from RLC (RRC message)\n",
                  module_idP, frameP, CC_id,
                  TBS - ta_len - header_length_total - sdu_length_total - 3);
            sdu_lengths[num_sdus] += mac_rlc_data_req(module_idP, rnti, module_idP, frameP, ENB_FLAG_YES, MBMS_FLAG_NO, DCCH + 1,
                                     TBS, //not used
                                     (char *)&dlsch_buffer[sdu_length_total]
986
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
987
                                     ,0, 0
988
#endif
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
989
                                                     );
990 991 992 993
            T(T_ENB_MAC_UE_DL_SDU, T_INT(module_idP),
              T_INT(CC_id), T_INT(rnti), T_INT(frameP),
              T_INT(subframeP), T_INT(harq_pid),
              T_INT(DCCH + 1), T_INT(sdu_lengths[num_sdus]));
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
994 995
            sdu_lcids[num_sdus] = DCCH1;
            sdu_length_total += sdu_lengths[num_sdus];
996 997
            UE_list->eNB_UE_stats[CC_id][UE_id].lcid_sdu[num_sdus] = DCCH1;
            UE_list->eNB_UE_stats[CC_id][UE_id].sdu_length_tx[DCCH1] = sdu_lengths[num_sdus];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
998 999
            UE_list->eNB_UE_stats[CC_id][UE_id].num_pdu_tx[DCCH1] += 1;
            UE_list->eNB_UE_stats[CC_id][UE_id].num_bytes_tx[DCCH1] += sdu_lengths[num_sdus];
1000 1001
            header_length_last = 1 + 1 + (sdu_lengths[num_sdus] >= 128);
            header_length_total += header_length_last;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1002
            num_sdus++;
nikaeinn's avatar
nikaeinn committed
1003
#ifdef DEBUG_eNB_SCHEDULER
1004
            LOG_T(MAC,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1005 1006
                  "[eNB %d][DCCH1] CC_id %d Got %d bytes :",
                  module_idP, CC_id, sdu_lengths[num_sdus]);
1007

1008 1009 1010
            for (j = 0; j < sdu_lengths[num_sdus]; ++j) {
              LOG_T(MAC, "%x ", dlsch_buffer[j]);
            }
1011

1012
            LOG_T(MAC, "\n");
nikaeinn's avatar
nikaeinn committed
1013
#endif
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1014 1015
          }
        }
1016

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049
        // TODO: lcid has to be sorted before the actual allocation (similar struct as ue_list).
        for (lcid = NB_RB_MAX - 1; lcid >= DTCH; lcid--) {
          // TODO: check if the lcid is active
          LOG_D(MAC, "[eNB %d], Frame %d, DTCH%d->DLSCH, Checking RLC status (tbs %d, len %d)\n",
                module_idP,
                frameP,
                lcid,
                TBS,
                TBS - ta_len - header_length_total - sdu_length_total - 3);

          if (TBS - ta_len - header_length_total - sdu_length_total - 3 > 0) {
            rlc_status = mac_rlc_status_ind(module_idP,
                                            rnti,
                                            module_idP,
                                            frameP,
                                            subframeP,
                                            ENB_FLAG_YES,
                                            MBMS_FLAG_NO,
                                            lcid,
                                            TBS - ta_len - header_length_total - sdu_length_total - 3
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
                                            , 0, 0
#endif
                                           );

            if (rlc_status.bytes_in_buffer > 0) {
              LOG_D(MAC, "[eNB %d][USER-PLANE DEFAULT DRB] Frame %d : DTCH->DLSCH, Requesting %d bytes from RLC (lcid %d total hdr len %d)\n",
                    module_idP,
                    frameP,
                    TBS - ta_len - header_length_total - sdu_length_total - 3,
                    lcid,
                    header_length_total);
              sdu_lengths[num_sdus] = mac_rlc_data_req(module_idP,
1050 1051 1052 1053 1054 1055
                                      rnti,
                                      module_idP,
                                      frameP,
                                      ENB_FLAG_YES,
                                      MBMS_FLAG_NO,
                                      lcid,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1056 1057
                                      TBS, //not used
                                      (char *)&dlsch_buffer[sdu_length_total]
1058
#if (LTE_RRC_VERSION >= MAKE_VERSION(14, 0, 0))
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1059
                                      , 0, 0
1060
#endif
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086
                                                      );
              T(T_ENB_MAC_UE_DL_SDU,
                T_INT(module_idP),
                T_INT(CC_id),
                T_INT(rnti),
                T_INT(frameP),
                T_INT(subframeP),
                T_INT(harq_pid),
                T_INT(lcid),
                T_INT(sdu_lengths[num_sdus]));
              LOG_D(MAC, "[eNB %d][USER-PLANE DEFAULT DRB] Got %d bytes for DTCH %d \n",
                    module_idP,
                    sdu_lengths[num_sdus],
                    lcid);
              sdu_lcids[num_sdus] = lcid;
              sdu_length_total += sdu_lengths[num_sdus];
              UE_list->eNB_UE_stats[CC_id][UE_id].num_pdu_tx[lcid]++;
              UE_list->eNB_UE_stats[CC_id][UE_id].lcid_sdu[num_sdus] = lcid;
              UE_list->eNB_UE_stats[CC_id][UE_id].sdu_length_tx[lcid] = sdu_lengths[num_sdus];
              UE_list->eNB_UE_stats[CC_id][UE_id].num_bytes_tx[lcid] += sdu_lengths[num_sdus];
              header_length_last = 1 + 1 + (sdu_lengths[num_sdus] >= 128);
              header_length_total += header_length_last;
              num_sdus++;
              UE_list->UE_sched_ctrl[UE_id].uplane_inactivity_timer = 0;
              // reset RRC inactivity timer after uplane activity
              ue_contextP = rrc_eNB_get_ue_context(RC.rrc[module_idP], rnti);
1087 1088 1089 1090 1091 1092 1093 1094
              if (ue_contextP != NULL) {
                ue_contextP->ue_context.ue_rrc_inactivity_timer = 1;
              } else {
                LOG_E(MAC, "[eNB %d] CC_id %d Couldn't find the context associated to UE (RNTI %d) and reset RRC inactivity timer\n",
                      module_idP,
                      CC_id,
                      rnti);
              }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1095 1096 1097 1098 1099
            } // end if (rlc_status.bytes_in_buffer > 0)
          } else {  // no TBS left
            break;  // break for (lcid = NB_RB_MAX - 1; lcid >= DTCH; lcid--)
          }
        }
1100 1101 1102 1103 1104 1105 1106

        /* last header does not have length field */
        if (header_length_total) {
          header_length_total -= header_length_last;
          header_length_total++;
        }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1107 1108 1109 1110 1111
        // there is at least one SDU or TA command
        // if (num_sdus > 0 ){
        if (ta_len + sdu_length_total + header_length_total > 0) {
          // Now compute number of required RBs for total sdu length
          // Assume RAH format 2
1112
          mcs = eNB_UE_stats->dlsch_mcs1;
knopp's avatar
knopp committed
1113

1114 1115 1116 1117 1118
          if (mcs == 0) {
            nb_rb = 4;    // don't let the TBS get too small
          } else {
            nb_rb = min_rb_unit[CC_id];
          }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1119

1120
          TBS = get_TBS_DL(mcs, nb_rb);
knopp's avatar
knopp committed
1121

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1122 1123
          while (TBS < sdu_length_total + header_length_total + ta_len) {
            nb_rb += min_rb_unit[CC_id];  //
knopp's avatar
knopp committed
1124

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1125 1126 1127 1128 1129 1130
            if (nb_rb > nb_available_rb) {  // if we've gone beyond the maximum number of RBs
              // (can happen if N_RB_DL is odd)
              TBS = get_TBS_DL(eNB_UE_stats->dlsch_mcs1, nb_available_rb);
              nb_rb = nb_available_rb;
              break;
            }
knopp's avatar
knopp committed
1131

1132 1133
            TBS = get_TBS_DL(eNB_UE_stats->dlsch_mcs1, nb_rb);
          }
knopp's avatar
knopp committed
1134

1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152
          if (nb_rb == ue_sched_ctl->pre_nb_available_rbs[CC_id]) {
            for (j = 0; j < N_RBG[CC_id]; ++j) {    // for indicating the rballoc for each sub-band
              UE_list->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j] = ue_sched_ctl->rballoc_sub_UE[CC_id][j];
            }
          } else {
            nb_rb_temp = nb_rb;
            j = 0;

            while ((nb_rb_temp > 0) && (j < N_RBG[CC_id])) {
              if (ue_sched_ctl->rballoc_sub_UE[CC_id][j] == 1) {
                UE_list->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j] = ue_sched_ctl->rballoc_sub_UE[CC_id][j];

                if ((j == N_RBG[CC_id] - 1) && ((N_RB_DL[CC_id] == 25) || (N_RB_DL[CC_id] == 50))) {
                  nb_rb_temp = nb_rb_temp - min_rb_unit[CC_id] + 1;
                } else {
                  nb_rb_temp = nb_rb_temp - min_rb_unit[CC_id];
                }
              }
knopp's avatar
knopp committed
1153

1154 1155 1156
              j = j + 1;
            }
          }
knopp's avatar
knopp committed
1157

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172
          // decrease mcs until TBS falls below required length
          while ((TBS > sdu_length_total + header_length_total + ta_len) && (mcs > 0)) {
            mcs--;
            TBS = get_TBS_DL(mcs, nb_rb);
          }

          // if we have decreased too much or we don't have enough RBs, increase MCS
          while ((TBS < sdu_length_total + header_length_total + ta_len)
                 && (((ue_sched_ctl->dl_pow_off[CC_id] > 0)
                      && (mcs < 28))
                     || ((ue_sched_ctl->dl_pow_off[CC_id] == 0)
                         && (mcs <= 15)))) {
            mcs++;
            TBS = get_TBS_DL(mcs, nb_rb);
          }
knopp's avatar
knopp committed
1173

1174 1175 1176
          LOG_D(MAC,
                "dlsch_mcs before and after the rate matching = (%d, %d)\n",
                eNB_UE_stats->dlsch_mcs1, mcs);
1177
#ifdef DEBUG_eNB_SCHEDULER
1178
          LOG_D(MAC,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1179 1180
                "[eNB %d] CC_id %d Generated DLSCH header (mcs %d, TBS %d, nb_rb %d)\n",
                module_idP, CC_id, mcs, TBS, nb_rb);
1181 1182
          // msg("[MAC][eNB ] Reminder of DLSCH with random data %d %d %d %d \n",
          //  TBS, sdu_length_total, offset, TBS-sdu_length_total-offset);
1183
#endif
1184

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1185 1186 1187 1188 1189 1190 1191
          if (TBS - header_length_total - sdu_length_total - ta_len <= 2) {
            padding = TBS - header_length_total - sdu_length_total - ta_len;
            post_padding = 0;
          } else {
            padding = 0;
            post_padding = 1;
          }
knopp's avatar
knopp committed
1192

1193 1194 1195 1196 1197 1198 1199
          offset = generate_dlsch_header((unsigned char *) UE_list->DLSCH_pdu[CC_id][0][UE_id].payload[0],
                                         num_sdus,    //num_sdus
                                         sdu_lengths,    //
                                         sdu_lcids, 255,    // no drx
                                         ta_update,    // timing advance
                                         NULL,    // contention res id
                                         padding, post_padding);
knopp's avatar
knopp committed
1200

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212
          //#ifdef DEBUG_eNB_SCHEDULER
          if (ta_update != 31) {
            LOG_D(MAC,
                  "[eNB %d][DLSCH] Frame %d Generate header for UE_id %d on CC_id %d: sdu_length_total %d, num_sdus %d, sdu_lengths[0] %d, sdu_lcids[0] %d => payload offset %d,timing advance value : %d, padding %d,post_padding %d,(mcs %d, TBS %d, nb_rb %d),header_length %d\n",
                  module_idP, frameP, UE_id, CC_id,
                  sdu_length_total, num_sdus, sdu_lengths[0],
                  sdu_lcids[0], offset, ta_update, padding,
                  post_padding, mcs, TBS, nb_rb,
                  header_length_total);