eNB_scheduler_fairRR.c 142 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35
/*
 * 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
 * the OAI Public License, Version 1.1  (the "License"); you may not use this file
 * 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
 */

/*! \file eNB_scheduler_fairRR.h
 * \brief eNB scheduler fair round robin
 * \author Masayuki Harada
 * \date 2018
 * \email masayuki.harada@jp.fujitsu.com
 * \version 1.0
 * @ingroup _mac
 */

#define _GNU_SOURCE
#include <stdlib.h>

#include "assertions.h"

36
#include "PHY/phy_extern.h"
37 38
#include "PHY/LTE_TRANSPORT/transport_common_proto.h"
#include "SIMULATION/TOOLS/sim.h"
39 40
#include "LAYER2/MAC/mac_proto.h"
#include "LAYER2/MAC/mac_extern.h"
41
#include "LAYER2/MAC/eNB_scheduler_fairRR.h"
42
#include "common/utils/LOG/log.h"
frtabu's avatar
frtabu committed
43
#include "nfapi/oai_integration/vendor_ext.h"
44
#include "common/utils/LOG/vcd_signal_dumper.h"
45 46 47 48 49 50 51 52
#include "UTIL/OPT/opt.h"
#include "OCG.h"
#include "OCG_extern.h"
#include "RRC/L2_INTERFACE/openair_rrc_L2_interface.h"
#include "rlc.h"

#include "T.h"

frtabu's avatar
frtabu committed
53

54
#ifdef PHY_TX_THREAD
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
55 56
  extern volatile int16_t phy_tx_txdataF_end;
  extern int oai_exit;
57
#endif
58 59 60
extern uint16_t sfnsf_add_subframe(uint16_t frameP, uint16_t subframeP, int offset);
extern void add_subframe(uint16_t *frameP, uint16_t *subframeP, int offset);

61 62 63 64 65 66
/* internal vars */
DLSCH_UE_SELECT dlsch_ue_select[MAX_NUM_CCs];
int last_dlsch_ue_id[MAX_NUM_CCs] = {-1};
int last_ulsch_ue_id[MAX_NUM_CCs] = {-1};

#if defined(PRE_SCD_THREAD)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
67 68 69 70 71
  uint16_t pre_nb_rbs_required[2][MAX_NUM_CCs][NUMBER_OF_UE_MAX];
  uint8_t dlsch_ue_select_tbl_in_use;
  uint8_t new_dlsch_ue_select_tbl_in_use;
  boolean_t pre_scd_activeUE[NUMBER_OF_UE_MAX];
  eNB_UE_STATS pre_scd_eNB_UE_stats[MAX_NUM_CCs][NUMBER_OF_UE_MAX];
72 73 74 75 76 77
#endif

#define DEBUG_eNB_SCHEDULER 1
#define DEBUG_HEADER_PARSING 1
//#define DEBUG_PACKET_TRACE 1

78 79 80 81 82 83 84 85 86
void set_dl_ue_select_msg2(int CC_idP, uint16_t nb_rb, int UE_id, rnti_t rnti) {
  dlsch_ue_select[CC_idP].list[dlsch_ue_select[CC_idP].ue_num].ue_priority = SCH_DL_MSG2;
  dlsch_ue_select[CC_idP].list[dlsch_ue_select[CC_idP].ue_num].nb_rb = nb_rb;
  dlsch_ue_select[CC_idP].list[dlsch_ue_select[CC_idP].ue_num].UE_id = UE_id;
  dlsch_ue_select[CC_idP].list[dlsch_ue_select[CC_idP].ue_num].rnti = rnti;
  dlsch_ue_select[CC_idP].ue_num++;
}
void set_dl_ue_select_msg4(int CC_idP, uint16_t nb_rb, int UE_id, rnti_t rnti) {
  dlsch_ue_select[CC_idP].list[dlsch_ue_select[CC_idP].ue_num].ue_priority = SCH_DL_MSG4;
87 88 89 90 91 92
  dlsch_ue_select[CC_idP].list[dlsch_ue_select[CC_idP].ue_num].nb_rb = nb_rb;
  dlsch_ue_select[CC_idP].list[dlsch_ue_select[CC_idP].ue_num].UE_id = UE_id;
  dlsch_ue_select[CC_idP].list[dlsch_ue_select[CC_idP].ue_num].rnti = rnti;
  dlsch_ue_select[CC_idP].ue_num++;
}
#if defined(PRE_SCD_THREAD)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
93
inline uint16_t search_rbs_required(uint16_t mcs, uint16_t TBS,uint16_t NB_RB, uint16_t step_size) {
94 95
  uint16_t nb_rb,i_TBS,tmp_TBS;
  i_TBS=get_I_TBS(mcs);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
96 97

  for(nb_rb=step_size; nb_rb<NB_RB; nb_rb+=step_size) {
98
    tmp_TBS = TBStable[i_TBS][nb_rb-1]>>3;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
99

100 101
    if(TBS<tmp_TBS)return(nb_rb);
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
102

103 104 105 106 107 108
  return NB_RB;
}
void pre_scd_nb_rbs_required(    module_id_t     module_idP,
                                 frame_t         frameP,
                                 sub_frame_t     subframeP,
                                 int             min_rb_unit[MAX_NUM_CCs],
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
109 110 111 112 113 114 115 116
                                 uint16_t        nb_rbs_required[MAX_NUM_CCs][NUMBER_OF_UE_MAX]) {
  int                          CC_id=0,UE_id, lc_id, N_RB_DL;
  UE_TEMPLATE                  UE_template;
  eNB_UE_STATS                 *eNB_UE_stats;
  rnti_t                       rnti;
  mac_rlc_status_resp_t        rlc_status;
  uint16_t                     step_size=2;
  N_RB_DL = to_prb(RC.mac[module_idP]->common_channels[CC_id].mib->message.dl_Bandwidth);
117

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
118
  if(N_RB_DL==50) step_size=3;
119

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
120
  if(N_RB_DL==100) step_size=4;
121

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
122
  memset(nb_rbs_required, 0, sizeof(uint16_t)*MAX_NUM_CCs*NUMBER_OF_UE_MAX);
123
  UE_info_t *UE_info = &RC.mac[module_idP]->UE_info;
124

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
125 126 127
  for (UE_id = 0; UE_id <NUMBER_OF_UE_MAX; UE_id++) {
    if (pre_scd_activeUE[UE_id] != TRUE)
      continue;
128

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
129 130 131 132
    // store dlsch buffer
    // clear logical channel interface variables
    UE_template.dl_buffer_total = 0;
    rnti = UE_RNTI(module_idP, UE_id);
133

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
134 135 136
    for (lc_id = DCCH; lc_id <= DTCH; lc_id++) {
      rlc_status =
        mac_rlc_status_ind(module_idP, rnti, module_idP, frameP, subframeP,
137
                           ENB_FLAG_YES, MBMS_FLAG_NO, lc_id, 0, 0
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
138 139 140
                          );
      UE_template.dl_buffer_total += rlc_status.bytes_in_buffer; //storing the total dlsch buffer
    }
141

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
142 143 144 145 146
    // end of store dlsch buffer
    // assgin rbs required
    // Calculate the number of RBs required by each UE on the basis of logical channel's buffer
    //update CQI information across component carriers
    eNB_UE_stats = &pre_scd_eNB_UE_stats[CC_id][UE_id];
147
    eNB_UE_stats->dlsch_mcs1 = cqi_to_mcs[UE_info->UE_sched_ctrl[UE_id].dl_cqi[CC_id]];
148

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
149 150
    if (UE_template.dl_buffer_total > 0) {
      nb_rbs_required[CC_id][UE_id] = search_rbs_required(eNB_UE_stats->dlsch_mcs1, UE_template.dl_buffer_total, N_RB_DL, step_size);
151
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
152
  }
153 154 155
}
#endif

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
156
int cc_id_end(uint8_t *cc_id_flag ) {
157
  int end_flag = 1;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
158 159

  for (int CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
160 161 162 163 164
    if (cc_id_flag[CC_id]==0) {
      end_flag = 0;
      break;
    }
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
165

166 167 168 169
  return end_flag;
}

void dlsch_scheduler_pre_ue_select_fairRR(
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
170 171 172 173
  module_id_t     module_idP,
  frame_t         frameP,
  sub_frame_t     subframeP,
  int            *mbsfn_flag,
laurent's avatar
laurent committed
174
  uint16_t        nb_rbs_required[MAX_NUM_CCs][MAX_MOBILES_PER_ENB],
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
175
  DLSCH_UE_SELECT dlsch_ue_select[MAX_NUM_CCs]) {
176 177
  eNB_MAC_INST                   *eNB      = RC.mac[module_idP];
  COMMON_channels_t              *cc       = eNB->common_channels;
178
  UE_info_t                      *UE_info  = &eNB->UE_info;
179
  UE_sched_ctrl_t                  *ue_sched_ctl;
180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195
  uint8_t                        CC_id;
  int                            UE_id;
  unsigned char                  round             = 0;
  unsigned char                  harq_pid          = 0;
  rnti_t                         rnti;
  uint16_t                       i;
  unsigned char                  aggregation;
  int                            format_flag;
  nfapi_dl_config_request_body_t *DL_req;
  nfapi_dl_config_request_pdu_t  *dl_config_pdu;
  uint16_t                       dlsch_ue_max_num[MAX_NUM_CCs] = {0};
  uint16_t                       saved_dlsch_dci[MAX_NUM_CCs] = {0};
  uint8_t                        end_flag[MAX_NUM_CCs] = {0};

  // Initialization
  for (CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++) {
196
    dlsch_ue_max_num[CC_id] = (uint16_t)RC.rrc[module_idP]->configuration.radioresourceconfig[CC_id].ue_multiple_max;
197 198 199 200 201 202 203 204 205 206 207 208
    // save origin DL PDU number
    DL_req          = &eNB->DL_req[CC_id].dl_config_request_body;
    saved_dlsch_dci[CC_id] = DL_req->number_pdu;
  }

  // Insert DLSCH(retransmission) UE into selected UE list
  for (CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++) {
    if (mbsfn_flag[CC_id]>0) {
      continue;
    }

    DL_req          = &eNB->DL_req[CC_id].dl_config_request_body;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
209

210
    for (UE_id = 0; UE_id < NUMBER_OF_UE_MAX; UE_id++) {
211
      if (UE_info->active[UE_id] == FALSE) {
212 213 214 215
        continue;
      }

      rnti = UE_RNTI(module_idP, UE_id);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
216

217 218 219 220
      if (rnti == NOT_A_RNTI) {
        continue;
      }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
221
      if(mac_eNB_get_rrc_status(module_idP,rnti) < RRC_CONNECTED) {
222 223 224
        continue;
      }

225
      ue_sched_ctl = &UE_info->UE_sched_ctrl[UE_id];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
226
      harq_pid = frame_subframe2_dl_harq_pid(cc[CC_id].tdd_Config,frameP,subframeP);
227
      round = ue_sched_ctl->round[CC_id][harq_pid];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
228

229
      if (round != 8) {  // retransmission
230
        if(UE_info->UE_template[CC_id][UE_id].nb_rb[harq_pid] == 0) {
231 232
          continue;
        }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
233

234 235 236 237 238
        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),
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
239 240
                                          ue_sched_ctl->dl_cqi[CC_id],
                                          format1);
241
            break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
242

243 244
          case 3:
            aggregation = get_aggregation(get_bw_index(module_idP,CC_id),
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
245 246
                                          ue_sched_ctl->dl_cqi[CC_id],
                                          format2A);
247
            break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
248

249 250 251 252 253
          default:
            LOG_W(MAC,"Unsupported transmission mode %d\n", get_tmode(module_idP,CC_id,UE_id));
            aggregation = 2;
            break;
        }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
254

255
        format_flag = 1;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
256

257
        if (!CCE_allocation_infeasible(module_idP,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
258 259 260 261 262
                                       CC_id,
                                       format_flag,
                                       subframeP,
                                       aggregation,
                                       rnti)) {
263 264 265 266 267 268
          dl_config_pdu = &DL_req->dl_config_pdu_list[DL_req->number_pdu];
          dl_config_pdu->pdu_type                                     = NFAPI_DL_CONFIG_DCI_DL_PDU_TYPE;
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.rnti              = rnti;
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.rnti_type         = (format_flag == 0)?2:1;
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.aggregation_level = aggregation;
          DL_req->number_pdu++;
269
          nb_rbs_required[CC_id][UE_id] = UE_info->UE_template[CC_id][UE_id].nb_rb[harq_pid];
270 271 272 273 274 275
          // Insert DLSCH(retransmission) UE into selected UE list
          dlsch_ue_select[CC_id].list[dlsch_ue_select[CC_id].ue_num].UE_id = UE_id;
          dlsch_ue_select[CC_id].list[dlsch_ue_select[CC_id].ue_num].ue_priority = SCH_DL_RETRANS;
          dlsch_ue_select[CC_id].list[dlsch_ue_select[CC_id].ue_num].rnti = rnti;
          dlsch_ue_select[CC_id].list[dlsch_ue_select[CC_id].ue_num].nb_rb = nb_rbs_required[CC_id][UE_id];
          dlsch_ue_select[CC_id].ue_num++;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
276

277 278 279 280 281 282 283
          if (dlsch_ue_select[CC_id].ue_num == dlsch_ue_max_num[CC_id]) {
            end_flag[CC_id] = 1;
            break;
          }
        } else {
          if (cc[CC_id].tdd_Config != NULL) { //TDD
            set_ue_dai (subframeP,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
284 285 286
                        UE_id,
                        CC_id,
                        cc[CC_id].tdd_Config->subframeAssignment,
287
                        UE_info);
288 289 290 291 292 293 294 295
            // update UL DAI after DLSCH scheduling
            set_ul_DAI(module_idP,UE_id,CC_id,frameP,subframeP);
          }

          add_ue_dlsch_info(module_idP,
                            CC_id,
                            UE_id,
                            subframeP,
Stefan's avatar
Stefan committed
296 297
                            S_DL_NONE,
                            rnti);
298 299 300 301 302 303
          end_flag[CC_id] = 1;
          break;
        }
      }
    }
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
304 305

  if(cc_id_end(end_flag) == 1) {
306 307 308 309
    for (CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++) {
      DL_req          = &eNB->DL_req[CC_id].dl_config_request_body;
      DL_req->number_pdu = saved_dlsch_dci[CC_id];
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
310

311 312 313 314 315
    return;
  }

  // Insert DLSCH(first transmission) UE into selected UE list (UE_id > last_dlsch_ue_id[CC_id])
  for (CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
316 317 318 319 320 321 322 323 324 325 326
    if (mbsfn_flag[CC_id]>0) {
      continue;
    }

    DL_req          = &eNB->DL_req[CC_id].dl_config_request_body;

    for (UE_id = (last_dlsch_ue_id[CC_id]+1); UE_id <NUMBER_OF_UE_MAX; UE_id++) {
      if(end_flag[CC_id] == 1) {
        break;
      }

327
      if (UE_info->active[UE_id] == FALSE) {
328 329 330
        continue;
      }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
331 332 333 334 335 336 337 338 339
      rnti = UE_RNTI(module_idP,UE_id);

      if (rnti == NOT_A_RNTI)
        continue;

      if(mac_eNB_get_rrc_status(module_idP,rnti) < RRC_CONNECTED) {
        continue;
      }

340
      ue_sched_ctl = &UE_info->UE_sched_ctrl[UE_id];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
341 342 343

      for(i = 0; i<dlsch_ue_select[CC_id].ue_num; i++) {
        if(dlsch_ue_select[CC_id].list[i].UE_id == UE_id) {
344 345
          break;
        }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
346
      }
347

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
348 349
      if(i < dlsch_ue_select[CC_id].ue_num)
        continue;
350

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
351 352
      harq_pid = frame_subframe2_dl_harq_pid(cc[CC_id].tdd_Config,frameP,subframeP);
      round = ue_sched_ctl->round[CC_id][harq_pid];
353

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
354 355
      if (round == 8) {
        if (nb_rbs_required[CC_id][UE_id] == 0) {
356 357 358
          continue;
        }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377
        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;

          case 3:
            aggregation = get_aggregation(get_bw_index(module_idP,CC_id),
                                          ue_sched_ctl->dl_cqi[CC_id],
                                          format2A);
            break;

          default:
            LOG_W(MAC,"Unsupported transmission mode %d\n", get_tmode(module_idP,CC_id,UE_id));
            aggregation = 2;
            break;
378 379
        }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
380
        format_flag = 1;
381

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399
        if (!CCE_allocation_infeasible(module_idP,
                                       CC_id,
                                       format_flag,
                                       subframeP,
                                       aggregation,
                                       rnti)) {
          dl_config_pdu = &DL_req->dl_config_pdu_list[DL_req->number_pdu];
          dl_config_pdu->pdu_type                                     = NFAPI_DL_CONFIG_DCI_DL_PDU_TYPE;
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.rnti              = rnti;
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.rnti_type         = (format_flag == 0)?2:1;
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.aggregation_level = aggregation;
          DL_req->number_pdu++;
          // Insert DLSCH(first transmission) UE into selected selected UE list
          dlsch_ue_select[CC_id].list[dlsch_ue_select[CC_id].ue_num].ue_priority = SCH_DL_FIRST;
          dlsch_ue_select[CC_id].list[dlsch_ue_select[CC_id].ue_num].nb_rb = nb_rbs_required[CC_id][UE_id];
          dlsch_ue_select[CC_id].list[dlsch_ue_select[CC_id].ue_num].UE_id = UE_id;
          dlsch_ue_select[CC_id].list[dlsch_ue_select[CC_id].ue_num].rnti = rnti;
          dlsch_ue_select[CC_id].ue_num++;
400 401

          if (dlsch_ue_select[CC_id].ue_num == dlsch_ue_max_num[CC_id]) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
402 403 404 405 406 407 408 409 410
            end_flag[CC_id] = 1;
            break;
          }
        } else {
          if (cc[CC_id].tdd_Config != NULL) { //TDD
            set_ue_dai (subframeP,
                        UE_id,
                        CC_id,
                        cc[CC_id].tdd_Config->subframeAssignment,
411
                        UE_info);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
412 413 414 415 416
            // update UL DAI after DLSCH scheduling
            set_ul_DAI(module_idP,UE_id,CC_id,frameP,subframeP);
          }

          add_ue_dlsch_info(module_idP,
417 418 419
                            CC_id,
                            UE_id,
                            subframeP,
Stefan's avatar
Stefan committed
420 421
                            S_DL_NONE,
                            rnti);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
422 423 424
          end_flag[CC_id] = 1;
          break;
        }
425 426 427
      }
    }
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
428 429

  if(cc_id_end(end_flag) == 1) {
430 431 432 433
    for (CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++) {
      DL_req          = &eNB->DL_req[CC_id].dl_config_request_body;
      DL_req->number_pdu = saved_dlsch_dci[CC_id];
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
434

435 436 437 438 439
    return;
  }

  // Insert DLSCH(first transmission) UE into selected UE list (UE_id <= last_dlsch_ue_id[CC_id])
  for (CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
440 441 442 443 444 445 446 447 448 449 450
    if (mbsfn_flag[CC_id]>0) {
      continue;
    }

    DL_req          = &eNB->DL_req[CC_id].dl_config_request_body;

    for (UE_id = 0; UE_id <= last_dlsch_ue_id[CC_id]; UE_id++) {
      if(end_flag[CC_id] == 1) {
        break;
      }

451
      if (UE_info->active[UE_id] == FALSE) {
452 453 454
        continue;
      }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
455 456 457 458 459 460 461 462 463
      rnti = UE_RNTI(module_idP,UE_id);

      if (rnti == NOT_A_RNTI)
        continue;

      if(mac_eNB_get_rrc_status(module_idP,rnti) < RRC_CONNECTED) {
        continue;
      }

464
      ue_sched_ctl = &UE_info->UE_sched_ctrl[UE_id];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
465 466 467

      for(i = 0; i<dlsch_ue_select[CC_id].ue_num; i++) {
        if(dlsch_ue_select[CC_id].list[i].UE_id == UE_id) {
468 469
          break;
        }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
470
      }
471

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
472 473
      if(i < dlsch_ue_select[CC_id].ue_num)
        continue;
474

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
475 476
      harq_pid = frame_subframe2_dl_harq_pid(cc[CC_id].tdd_Config,frameP,subframeP);
      round = ue_sched_ctl->round[CC_id][harq_pid];
477

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
478 479
      if (round == 8) {
        if (nb_rbs_required[CC_id][UE_id] == 0) {
480 481 482
          continue;
        }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501
        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;

          case 3:
            aggregation = get_aggregation(get_bw_index(module_idP,CC_id),
                                          ue_sched_ctl->dl_cqi[CC_id],
                                          format2A);
            break;

          default:
            LOG_W(MAC,"Unsupported transmission mode %d\n", get_tmode(module_idP,CC_id,UE_id));
            aggregation = 2;
            break;
502 503
        }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534
        format_flag = 1;

        if (!CCE_allocation_infeasible(module_idP,
                                       CC_id,
                                       format_flag,
                                       subframeP,
                                       aggregation,
                                       rnti)) {
          dl_config_pdu = &DL_req->dl_config_pdu_list[DL_req->number_pdu];
          dl_config_pdu->pdu_type                                     = NFAPI_DL_CONFIG_DCI_DL_PDU_TYPE;
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.rnti              = rnti;
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.rnti_type         = (format_flag == 0)?2:1;
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.aggregation_level = aggregation;
          DL_req->number_pdu++;
          // Insert DLSCH(first transmission) UE into selected selected UE list
          dlsch_ue_select[CC_id].list[dlsch_ue_select[CC_id].ue_num].ue_priority = SCH_DL_FIRST;
          dlsch_ue_select[CC_id].list[dlsch_ue_select[CC_id].ue_num].nb_rb = nb_rbs_required[CC_id][UE_id];
          dlsch_ue_select[CC_id].list[dlsch_ue_select[CC_id].ue_num].UE_id = UE_id;
          dlsch_ue_select[CC_id].list[dlsch_ue_select[CC_id].ue_num].rnti = rnti;
          dlsch_ue_select[CC_id].ue_num++;

          if (dlsch_ue_select[CC_id].ue_num == dlsch_ue_max_num[CC_id]) {
            end_flag[CC_id] = 1;
            break;
          }
        } else {
          if (cc[CC_id].tdd_Config != NULL) { //TDD
            set_ue_dai (subframeP,
                        UE_id,
                        CC_id,
                        cc[CC_id].tdd_Config->subframeAssignment,
535
                        UE_info);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
536 537 538
            // update UL DAI after DLSCH scheduling
            set_ul_DAI(module_idP,UE_id,CC_id,frameP,subframeP);
          }
539 540 541 542 543

          add_ue_dlsch_info(module_idP,
                            CC_id,
                            UE_id,
                            subframeP,
Stefan's avatar
Stefan committed
544 545
                            S_DL_NONE,
                            rnti);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
546 547 548
          end_flag[CC_id] = 1;
          break;
        }
549 550 551 552 553 554 555 556
      }
    }
  }

  for (CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++) {
    DL_req          = &eNB->DL_req[CC_id].dl_config_request_body;
    DL_req->number_pdu = saved_dlsch_dci[CC_id];
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
557

558 559 560
  return;
}

561 562 563 564 565 566 567 568 569 570 571
void dlsch_scheduler_pre_processor_reset_fairRR(
    module_id_t module_idP,
    frame_t frameP,
    sub_frame_t subframeP,
    int min_rb_unit[NFAPI_CC_MAX],
    uint16_t nb_rbs_required[NFAPI_CC_MAX][MAX_MOBILES_PER_ENB],
    uint8_t rballoc_sub[NFAPI_CC_MAX][N_RBG_MAX],
    uint8_t MIMO_mode_indicator[NFAPI_CC_MAX][N_RBG_MAX]) {
  int UE_id;
  uint8_t CC_id;
  int i, j;
572
  UE_info_t *UE_info;
573 574 575 576 577 578 579 580 581 582 583 584 585 586 587
  UE_sched_ctrl_t *ue_sched_ctl;
  int N_RB_DL, RBGsize, RBGsize_last;
  int N_RBG[NFAPI_CC_MAX];
  rnti_t rnti;
  uint8_t *vrb_map;
  COMMON_channels_t *cc;

  //
  for (CC_id = 0; CC_id < RC.nb_mac_CC[module_idP]; CC_id++) {
    // initialize harq_pid and round
    cc = &RC.mac[module_idP]->common_channels[CC_id];
    N_RBG[CC_id] = to_rbg(cc->mib->message.dl_Bandwidth);
    min_rb_unit[CC_id] = get_min_rb_unit(module_idP, CC_id);

    for (UE_id = 0; UE_id < MAX_MOBILES_PER_ENB; ++UE_id) {
588 589
      UE_info = &RC.mac[module_idP]->UE_info;
      ue_sched_ctl = &UE_info->UE_sched_ctrl[UE_id];
590 591 592 593 594
      rnti = UE_RNTI(module_idP, UE_id);

      if (rnti == NOT_A_RNTI)
        continue;

595
      if (UE_info->active[UE_id] != TRUE)
596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728
        continue;

      LOG_D(MAC, "Running preprocessor for UE %d (%x)\n", UE_id, rnti);

      // initialize harq_pid and round
      if (ue_sched_ctl->ta_timer)
        ue_sched_ctl->ta_timer--;

      /*
         eNB_UE_stats *eNB_UE_stats;

         if (eNB_UE_stats == NULL)
         return;


         mac_xface->get_ue_active_harq_pid(module_idP,CC_id,rnti,
         frameP,subframeP,
         &ue_sched_ctl->harq_pid[CC_id],
         &ue_sched_ctl->round[CC_id],
         openair_harq_DL);


         if (ue_sched_ctl->ta_timer == 0) {

         // WE SHOULD PROTECT the eNB_UE_stats with a mutex here ...

         ue_sched_ctl->ta_timer = 20;  // wait 20 subframes before taking TA measurement from PHY
         switch (N_RB_DL) {
         case 6:
         ue_sched_ctl->ta_update = eNB_UE_stats->timing_advance_update;
         break;

         case 15:
         ue_sched_ctl->ta_update = eNB_UE_stats->timing_advance_update/2;
         break;

         case 25:
         ue_sched_ctl->ta_update = eNB_UE_stats->timing_advance_update/4;
         break;

         case 50:
         ue_sched_ctl->ta_update = eNB_UE_stats->timing_advance_update/8;
         break;

         case 75:
         ue_sched_ctl->ta_update = eNB_UE_stats->timing_advance_update/12;
         break;

         case 100:
         ue_sched_ctl->ta_update = eNB_UE_stats->timing_advance_update/16;
         break;
         }
         // clear the update in case PHY does not have a new measurement after timer expiry
         eNB_UE_stats->timing_advance_update =  0;
         }
         else {
         ue_sched_ctl->ta_timer--;
         ue_sched_ctl->ta_update =0; // don't trigger a timing advance command
         }


         if (UE_id==0) {
         VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_UE0_TIMING_ADVANCE,ue_sched_ctl->ta_update);
         }
       */
      nb_rbs_required[CC_id][UE_id] = 0;
      ue_sched_ctl->pre_nb_available_rbs[CC_id] = 0;
      ue_sched_ctl->dl_pow_off[CC_id] = 2;

      for (i = 0; i < N_RBG[CC_id]; i++) {
        ue_sched_ctl->rballoc_sub_UE[CC_id][i] = 0;
      }
    }

    N_RB_DL = to_prb(RC.mac[module_idP]->common_channels[CC_id].mib->message.dl_Bandwidth);

    switch (N_RB_DL) {
      case 6:
        RBGsize = 1;
        RBGsize_last = 1;
        break;

      case 15:
        RBGsize = 2;
        RBGsize_last = 1;
        break;

      case 25:
        RBGsize = 2;
        RBGsize_last = 1;
        break;

      case 50:
        RBGsize = 3;
        RBGsize_last = 2;
        break;

      case 75:
        RBGsize = 4;
        RBGsize_last = 3;
        break;

      case 100:
        RBGsize = 4;
        RBGsize_last = 4;
        break;

      default:
        AssertFatal(1 == 0, "unsupported RBs (%d)\n", N_RB_DL);
    }

    vrb_map = RC.mac[module_idP]->common_channels[CC_id].vrb_map;

    // Initialize Subbands according to VRB map
    for (i = 0; i < N_RBG[CC_id]; i++) {
      int rb_size = i == N_RBG[CC_id] - 1 ? RBGsize_last : RBGsize;

      // for SI-RNTI,RA-RNTI and P-RNTI allocations
      for (j = 0; j < rb_size; j++) {
        if (vrb_map[j + (i*RBGsize)] != 0) {
          rballoc_sub[CC_id][i] = 1;
          LOG_D(MAC, "Frame %d, subframe %d : vrb %d allocated\n", frameP, subframeP, j + (i*RBGsize));
          break;
        }
      }

      //LOG_D(MAC, "Frame %d Subframe %d CC_id %d RBG %i : rb_alloc %d\n",
      //frameP, subframeP, CC_id, i, rballoc_sub[CC_id][i]);
      MIMO_mode_indicator[CC_id][i] = 2;
    }
  }
}

729 730 731 732 733 734 735 736
// This function returns the estimated number of RBs required by each UE for downlink scheduling
void assign_rbs_required_fairRR(
    module_id_t Mod_id,
    frame_t frameP,
    sub_frame_t subframe,
    uint16_t nb_rbs_required[NFAPI_CC_MAX][MAX_MOBILES_PER_ENB]) {
  uint16_t TBS = 0;
  int UE_id, n, i, j, CC_id, pCCid, tmp;
737
  UE_info_t *UE_info = &RC.mac[Mod_id]->UE_info;
738 739 740 741 742
  eNB_UE_STATS *eNB_UE_stats, *eNB_UE_stats_i, *eNB_UE_stats_j;
  int N_RB_DL;

  // clear rb allocations across all CC_id
  for (UE_id = 0; UE_id < MAX_MOBILES_PER_ENB; UE_id++) {
743
    if (UE_info->active[UE_id] != TRUE)
744 745 746 747 748
      continue;

    pCCid = UE_PCCID(Mod_id, UE_id);

    // update CQI information across component carriers
749 750 751 752
    for (n = 0; n < UE_info->numactiveCCs[UE_id]; n++) {
      CC_id = UE_info->ordered_CCids[n][UE_id];
      eNB_UE_stats = &UE_info->eNB_UE_stats[CC_id][UE_id];
      eNB_UE_stats->dlsch_mcs1 = cqi_to_mcs[UE_info->UE_sched_ctrl[UE_id].dl_cqi[CC_id]];
753 754 755
    }

    // provide the list of CCs sorted according to MCS
756
    for (i = 0; i < UE_info->numactiveCCs[UE_id]; ++i) {
757
      eNB_UE_stats_i =
758
          &UE_info->eNB_UE_stats[UE_info->ordered_CCids[i][UE_id]][UE_id];
759

760
      for (j = i + 1; j < UE_info->numactiveCCs[UE_id]; j++) {
761 762
        DevAssert(j < NFAPI_CC_MAX);
        eNB_UE_stats_j =
763
            &UE_info->eNB_UE_stats[UE_info->ordered_CCids[j][UE_id]][UE_id];
764 765

        if (eNB_UE_stats_j->dlsch_mcs1 > eNB_UE_stats_i->dlsch_mcs1) {
766 767 768
          tmp = UE_info->ordered_CCids[i][UE_id];
          UE_info->ordered_CCids[i][UE_id] = UE_info->ordered_CCids[j][UE_id];
          UE_info->ordered_CCids[j][UE_id] = tmp;
769 770 771 772
        }
      }
    }

773
    if (UE_info->UE_template[pCCid][UE_id].dl_buffer_total > 0) {
774 775
      LOG_D(MAC, "[preprocessor] assign RB for UE %d\n", UE_id);

776 777 778
      for (i = 0; i < UE_info->numactiveCCs[UE_id]; i++) {
        CC_id = UE_info->ordered_CCids[i][UE_id];
        eNB_UE_stats = &UE_info->eNB_UE_stats[CC_id][UE_id];
779 780 781 782 783 784 785
        const int min_rb_unit = get_min_rb_unit(Mod_id, CC_id);

        if (eNB_UE_stats->dlsch_mcs1 == 0) {
          nb_rbs_required[CC_id][UE_id] = 4; // don't let the TBS get too small
        } else {
          nb_rbs_required[CC_id][UE_id] = min_rb_unit;
        }
786

787 788 789 790 791 792
        TBS = get_TBS_DL(eNB_UE_stats->dlsch_mcs1, nb_rbs_required[CC_id][UE_id]);
        LOG_D(MAC,
              "[preprocessor] start RB assignement for UE %d CC_id %d dl "
              "buffer %d (RB unit %d, MCS %d, TBS %d) \n",
              UE_id,
              CC_id,
793
              UE_info->UE_template[pCCid][UE_id].dl_buffer_total,
794 795 796 797 798 799
              nb_rbs_required[CC_id][UE_id],
              eNB_UE_stats->dlsch_mcs1,
              TBS);
        N_RB_DL = to_prb(RC.mac[Mod_id]->common_channels[CC_id].mib->message.dl_Bandwidth);

        /* calculating required number of RBs for each UE */
800
        while (TBS < UE_info->UE_template[pCCid][UE_id].dl_buffer_total) {
801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837
          nb_rbs_required[CC_id][UE_id] += min_rb_unit;

          if (nb_rbs_required[CC_id][UE_id] > N_RB_DL) {
            TBS = get_TBS_DL(eNB_UE_stats->dlsch_mcs1, N_RB_DL);
            nb_rbs_required[CC_id][UE_id] = N_RB_DL;
            break;
          }

          TBS = get_TBS_DL(eNB_UE_stats->dlsch_mcs1,
                           nb_rbs_required[CC_id][UE_id]);
        } // end of while

        LOG_D(MAC,
              "[eNB %d] Frame %d: UE %d on CC %d: RB unit %d,  nb_required RB "
              "%d (TBS %d, mcs %d)\n",
              Mod_id,
              frameP,
              UE_id,
              CC_id,
              min_rb_unit,
              nb_rbs_required[CC_id][UE_id],
              TBS,
              eNB_UE_stats->dlsch_mcs1);
      }
    }
  }
}

void dlsch_scheduler_pre_processor_allocate_fairRR(
    module_id_t Mod_id,
    int UE_id,
    uint8_t CC_id,
    int N_RBG,
    uint16_t nb_rbs_required[NFAPI_CC_MAX][MAX_MOBILES_PER_ENB],
    uint16_t nb_rbs_remaining[NFAPI_CC_MAX][MAX_MOBILES_PER_ENB],
    uint8_t rballoc_sub[NFAPI_CC_MAX][N_RBG_MAX]) {
  int i;
838 839
  UE_info_t *UE_info = &RC.mac[Mod_id]->UE_info;
  UE_sched_ctrl_t *ue_sched_ctl = &UE_info->UE_sched_ctrl[UE_id];
840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885
  int N_RB_DL =
      to_prb(RC.mac[Mod_id]->common_channels[CC_id].mib->message.dl_Bandwidth);
  const int min_rb_unit = get_min_rb_unit(Mod_id, CC_id);

  for (i = 0; i < N_RBG; i++) {
    if (rballoc_sub[CC_id][i] != 0)
      continue;

    if (ue_sched_ctl->rballoc_sub_UE[CC_id][i] != 0)
      continue;

    if (nb_rbs_remaining[CC_id][UE_id] <= 0)
      continue;

    if (ue_sched_ctl->pre_nb_available_rbs[CC_id]
        >= nb_rbs_required[CC_id][UE_id])
      continue;

    if (ue_sched_ctl->dl_pow_off[CC_id] == 0)
      continue;

    if ((i == N_RBG - 1) && ((N_RB_DL == 25) || (N_RB_DL == 50))) {
      // Allocating last, smaller RBG
      if (nb_rbs_remaining[CC_id][UE_id] >= min_rb_unit - 1) {
        rballoc_sub[CC_id][i] = 1;
        ue_sched_ctl->rballoc_sub_UE[CC_id][i] = 1;

        nb_rbs_remaining[CC_id][UE_id] =
            nb_rbs_remaining[CC_id][UE_id] - min_rb_unit + 1;
        ue_sched_ctl->pre_nb_available_rbs[CC_id] =
            ue_sched_ctl->pre_nb_available_rbs[CC_id] + min_rb_unit - 1;
      }
    } else {
      // Allocating a standard-sized RBG
      if (nb_rbs_remaining[CC_id][UE_id] >= min_rb_unit) {
        rballoc_sub[CC_id][i] = 1;
        ue_sched_ctl->rballoc_sub_UE[CC_id][i] = 1;

        nb_rbs_remaining[CC_id][UE_id] =
            nb_rbs_remaining[CC_id][UE_id] - min_rb_unit;
        ue_sched_ctl->pre_nb_available_rbs[CC_id] =
            ue_sched_ctl->pre_nb_available_rbs[CC_id] + min_rb_unit;
      }
    }
  }
}
886 887 888

// This function assigns pre-available RBS to each UE in specified sub-bands before scheduling is done
void dlsch_scheduler_pre_processor_fairRR (module_id_t   Mod_id,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
889 890 891 892
    frame_t       frameP,
    sub_frame_t   subframeP,
    int           N_RBG[MAX_NUM_CCs],
    int           *mbsfn_flag) {
893 894 895 896
  unsigned char rballoc_sub[MAX_NUM_CCs][N_RBG_MAX],harq_pid=0,Round=0;
  uint16_t                temp_total_rbs_count;
  unsigned char           temp_total_ue_count;
  unsigned char MIMO_mode_indicator[MAX_NUM_CCs][N_RBG_MAX];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
897
  int                     UE_id, i;
898
  uint16_t                j;
laurent's avatar
laurent committed
899 900
  uint16_t                nb_rbs_required[MAX_NUM_CCs][MAX_MOBILES_PER_ENB];
  uint16_t                nb_rbs_required_remaining[MAX_NUM_CCs][MAX_MOBILES_PER_ENB];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
901
  //  uint16_t                nb_rbs_required_remaining_1[MAX_NUM_CCs][NUMBER_OF_UE_MAX];
902 903 904
  uint16_t                average_rbs_per_user[MAX_NUM_CCs] = {0};
  rnti_t             rnti;
  int                min_rb_unit[MAX_NUM_CCs];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
905
  //  uint16_t r1=0;
906
  uint8_t CC_id;
907
  UE_info_t *UE_info = &RC.mac[Mod_id]->UE_info;
wujing's avatar
wujing committed
908
  int N_RB_DL;
909
  UE_sched_ctrl_t *ue_sched_ctl;
910 911 912
  //  int rrc_status           = RRC_IDLE;
  COMMON_channels_t *cc;
#ifdef TM5
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
913 914 915 916 917 918 919
  int harq_pid1 = 0;
  int round1 = 0, round2 = 0;
  int UE_id2;
  uint16_t i1, i2, i3;
  rnti_t rnti1, rnti2;
  LTE_eNB_UE_stats *eNB_UE_stats1 = NULL;
  LTE_eNB_UE_stats *eNB_UE_stats2 = NULL;
920
  UE_sched_ctrl_t *ue_sched_ctl1, *ue_sched_ctl2;
921
#endif
922
  memset(rballoc_sub[0],0,(MAX_NUM_CCs)*(N_RBG_MAX)*sizeof(unsigned char));
923
  memset(min_rb_unit,0,sizeof(min_rb_unit));
924
  memset(MIMO_mode_indicator[0], 0, MAX_NUM_CCs*N_RBG_MAX*sizeof(unsigned char));
925

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
926 927 928
  for (CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
    if (mbsfn_flag[CC_id] > 0)  // If this CC is allocated for MBSFN skip it here
      continue;
929

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
930
    min_rb_unit[CC_id] = get_min_rb_unit(Mod_id, CC_id);
931

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
932
    for (i = 0; i < NUMBER_OF_UE_MAX; i++) {
933
      if (UE_info->active[i] != TRUE)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
934
        continue;
935

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
936 937
      UE_id = i;
      // Initialize scheduling information for all active UEs
938 939 940 941 942 943 944 945
      dlsch_scheduler_pre_processor_reset_fairRR(
          Mod_id,
          frameP,
          subframeP,
          min_rb_unit,
          (uint16_t(*)[MAX_MOBILES_PER_ENB])nb_rbs_required,
          rballoc_sub,
          MIMO_mode_indicator);
946
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
947
  }
948 949

#if (!defined(PRE_SCD_THREAD))
950 951
  // Store the DLSCH buffer for each logical channel and for PCCID (assume 0)
  store_dlsch_buffer(Mod_id, 0, frameP, subframeP);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
952
  // Calculate the number of RBs required by each UE on the basis of logical channel's buffer
953
  assign_rbs_required_fairRR(Mod_id, frameP, subframeP, nb_rbs_required);
954
#else
laurent's avatar
laurent committed
955
  memcpy(nb_rbs_required, pre_nb_rbs_required[dlsch_ue_select_tbl_in_use], sizeof(uint16_t)*MAX_NUM_CCs*MAX_MOBILES_PER_ENB);
956 957 958 959 960 961 962
#endif
  dlsch_scheduler_pre_ue_select_fairRR(Mod_id,frameP,subframeP, mbsfn_flag,nb_rbs_required,dlsch_ue_select);

  for (CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++) {
    average_rbs_per_user[CC_id] = 0;
    cc = &RC.mac[Mod_id]->common_channels[CC_id];
    // Get total available RBS count and total UE count
wujing's avatar
wujing committed
963 964
    N_RB_DL = to_prb(cc->mib->message.dl_Bandwidth);
    temp_total_rbs_count = 0;
965 966 967

    for(uint8_t rbg_i = 0; rbg_i < N_RBG[CC_id]; rbg_i++ ) {
      if(rballoc_sub[CC_id][rbg_i] == 0) {
wujing's avatar
wujing committed
968
        if((rbg_i == N_RBG[CC_id] -1) &&
969
            ((N_RB_DL == 25) || (N_RB_DL == 50))) {
wujing's avatar
wujing committed
970
          temp_total_rbs_count += (min_rb_unit[CC_id] -1);
971
        } else {
wujing's avatar
wujing committed
972 973 974 975
          temp_total_rbs_count += min_rb_unit[CC_id];
        }
      }
    }
976

977 978 979
    temp_total_ue_count = dlsch_ue_select[CC_id].ue_num;

    for (i = 0; i < dlsch_ue_select[CC_id].ue_num; i++) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
980 981 982
      if(dlsch_ue_select[CC_id].list[i].ue_priority == SCH_DL_MSG2) {
        temp_total_ue_count--;
        continue;
983
      }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
984 985 986 987

      if(dlsch_ue_select[CC_id].list[i].ue_priority == SCH_DL_MSG4) {
        temp_total_ue_count--;
        continue;
988
      }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
989

990 991 992
      UE_id = dlsch_ue_select[CC_id].list[i].UE_id;
      nb_rbs_required[CC_id][UE_id] = dlsch_ue_select[CC_id].list[i].nb_rb;
      average_rbs_per_user[CC_id] = (uint16_t)round((double)temp_total_rbs_count/(double)temp_total_ue_count);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
993 994

      if( average_rbs_per_user[CC_id] < min_rb_unit[CC_id] ) {
995 996 997 998 999 1000 1001
        temp_total_ue_count--;
        dlsch_ue_select[CC_id].ue_num--;
        i--;
        continue;
      }

      rnti = dlsch_ue_select[CC_id].list[i].rnti;
1002
      ue_sched_ctl = &UE_info->UE_sched_ctrl[UE_id];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1003
      harq_pid = frame_subframe2_dl_harq_pid(cc->tdd_Config,frameP,subframeP);
1004 1005 1006 1007 1008 1009 1010 1011 1012 1013
      Round    = ue_sched_ctl->round[CC_id][harq_pid];

      //if (mac_eNB_get_rrc_status(Mod_id, rnti) < RRC_RECONFIGURED || round > 0) {
      if (mac_eNB_get_rrc_status(Mod_id, rnti) < RRC_RECONFIGURED || Round != 8) {  // FIXME
        nb_rbs_required_remaining[CC_id][UE_id] = dlsch_ue_select[CC_id].list[i].nb_rb;
      } else {
        nb_rbs_required_remaining[CC_id][UE_id] = cmin(average_rbs_per_user[CC_id], dlsch_ue_select[CC_id].list[i].nb_rb);
      }

      LOG_T(MAC,"calling dlsch_scheduler_pre_processor_allocate .. \n ");
1014 1015 1016 1017 1018 1019 1020 1021
      dlsch_scheduler_pre_processor_allocate_fairRR(
          Mod_id,
          UE_id,
          CC_id,
          N_RBG[CC_id],
          (uint16_t(*)[NUMBER_OF_UE_MAX])nb_rbs_required,
          (uint16_t(*)[NUMBER_OF_UE_MAX])nb_rbs_required_remaining,
          rballoc_sub);
1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033
      temp_total_rbs_count -= ue_sched_ctl->pre_nb_available_rbs[CC_id];
      temp_total_ue_count--;

      if (ue_sched_ctl->pre_nb_available_rbs[CC_id] == 0) {
        dlsch_ue_select[CC_id].ue_num = i;
        break;
      }

      if (temp_total_rbs_count == 0) {
        dlsch_ue_select[CC_id].ue_num = i+1;
        break;
      }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1034 1035 1036 1037 1038

      LOG_D(MAC,
            "DLSCH UE Select: frame %d subframe %d pre_nb_available_rbs %d(i %d UE_id %d nb_rbs_required %d nb_rbs_required_remaining %d average_rbs_per_user %d (temp_total rbs_count %d ue_num %d) available_prbs %d)\n",
            frameP,subframeP,ue_sched_ctl->pre_nb_available_rbs[CC_id],i,UE_id,nb_rbs_required[CC_id][UE_id],nb_rbs_required_remaining[CC_id][UE_id],
            average_rbs_per_user[CC_id],temp_total_rbs_count,temp_total_ue_count,RC.mac[Mod_id]->eNB_stats[CC_id].available_prbs);
1039 1040 1041 1042 1043 1044 1045 1046
#ifdef TM5
      // TODO: data channel TM5: to be re-visited
#endif
    }
  }

#ifdef TM5

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060
  // This has to be revisited!!!!
  for (CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++) {
    i1 = 0;
    i2 = 0;
    i3 = 0;

    for (j = 0; j < N_RBG[CC_id]; j++) {
      if (MIMO_mode_indicator[CC_id][j] == 2) {
        i1 = i1 + 1;
      } else if (MIMO_mode_indicator[CC_id][j] == 1) {
        i2 = i2 + 1;
      } else if (MIMO_mode_indicator[CC_id][j] == 0) {
        i3 = i3 + 1;
      }
1061 1062
    }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1063 1064 1065 1066
    if ((i1 < N_RBG[CC_id]) && (i2 > 0) && (i3 == 0)) {
      PHY_vars_eNB_g[Mod_id][CC_id]->check_for_SUMIMO_transmissions =
        PHY_vars_eNB_g[Mod_id][CC_id]->
        check_for_SUMIMO_transmissions + 1;
1067 1068
    }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1069 1070 1071 1072 1073
    if (i3 == N_RBG[CC_id] && i1 == 0 && i2 == 0) {
      PHY_vars_eNB_g[Mod_id][CC_id]->FULL_MUMIMO_transmissions =
        PHY_vars_eNB_g[Mod_id][CC_id]->FULL_MUMIMO_transmissions +
        1;
    }
1074

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1075 1076 1077 1078
    if((i1 < N_RBG[CC_id]) && (i3 > 0)) {
      PHY_vars_eNB_g[Mod_id][CC_id]->
      check_for_MUMIMO_transmissions + 1;
    }
1079

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1080 1081 1082 1083
    PHY_vars_eNB_g[Mod_id][CC_id]->check_for_total_transmissions =
      PHY_vars_eNB_g[Mod_id][CC_id]->check_for_total_transmissions +
      1;
  }
1084

1085
#endif
1086

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097
  for (CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++) {
    for (i = 0; i < dlsch_ue_select[CC_id].ue_num; i++) {
      if(dlsch_ue_select[CC_id].list[i].ue_priority == SCH_DL_MSG2) {
        continue;
      }

      if(dlsch_ue_select[CC_id].list[i].ue_priority == SCH_DL_MSG4) {
        continue;
      }

      UE_id = dlsch_ue_select[CC_id].list[i].UE_id;
1098
      ue_sched_ctl = &RC.mac[Mod_id]->UE_info.UE_sched_ctrl[UE_id];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121
      //PHY_vars_eNB_g[Mod_id]->mu_mimo_mode[UE_id].dl_pow_off = dl_pow_off[UE_id];

      if (ue_sched_ctl->pre_nb_available_rbs[CC_id] > 0) {
        LOG_D(MAC,
              "******************DL Scheduling Information for UE%d ************************\n",
              UE_id);
        LOG_D(MAC, "dl power offset UE%d = %d \n", UE_id,
              ue_sched_ctl->dl_pow_off[CC_id]);
        LOG_D(MAC,
              "***********RB Alloc for every subband for UE%d ***********\n",
              UE_id);

        for (j = 0; j < N_RBG[CC_id]; j++) {
          //PHY_vars_eNB_g[Mod_id]->mu_mimo_mode[UE_id].rballoc_sub[i] = rballoc_sub_UE[CC_id][UE_id][i];
          LOG_D(MAC, "RB Alloc for UE%d and Subband%d = %d\n",
                UE_id, j,
                ue_sched_ctl->rballoc_sub_UE[CC_id][j]);
        }

        //PHY_vars_eNB_g[Mod_id]->mu_mimo_mode[UE_id].pre_nb_available_rbs = pre_nb_available_rbs[CC_id][UE_id];
        LOG_D(MAC, "Total RBs allocated for UE%d = %d\n", UE_id,
              ue_sched_ctl->pre_nb_available_rbs[CC_id]);
      }
1122
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1123 1124
  }
}
1125

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141

//------------------------------------------------------------------------------
void
schedule_ue_spec_fairRR(module_id_t module_idP,
                        frame_t frameP, sub_frame_t subframeP, int *mbsfn_flag)
//------------------------------------------------------------------------------
{
  uint8_t CC_id;
  int UE_id;
  //    unsigned char aggregation;
  mac_rlc_status_resp_t rlc_status;
  unsigned char header_len_dcch = 0, header_len_dcch_tmp = 0;
  unsigned char header_len_dtch = 0, header_len_dtch_tmp = 0, header_len_dtch_last = 0;
  unsigned char ta_len = 0;
  unsigned char sdu_lcids[NB_RB_MAX], lcid, offset, num_sdus = 0;
  uint16_t nb_rb, nb_rb_temp, nb_available_rb;
Stefan's avatar
Stefan committed
1142 1143
  uint16_t TBS, j, sdu_lengths[NB_RB_MAX], padding = 0, post_padding = 0;
  rnti_t rnti = 0;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1144 1145 1146 1147 1148 1149 1150
  unsigned char dlsch_buffer[MAX_DLSCH_PAYLOAD_BYTES];
  unsigned char round = 0;
  unsigned char harq_pid = 0;
  eNB_UE_STATS *eNB_UE_stats = NULL;
  uint16_t sdu_length_total = 0;
  eNB_MAC_INST *eNB = RC.mac[module_idP];
  COMMON_channels_t *cc = eNB->common_channels;
1151
  UE_info_t *UE_info = &eNB->UE_info;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1152
  // int continue_flag = 0;
Cedric Roux's avatar
Cedric Roux committed
1153
  int32_t snr, target_snr;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1154 1155
  int32_t tpc = 1;
  static int32_t tpc_accumulated = 0;
1156
  UE_sched_ctrl_t *ue_sched_ctl;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169
  int mcs;
  int i;
  int min_rb_unit[MAX_NUM_CCs];
  int N_RB_DL[MAX_NUM_CCs];
  int total_nb_available_rb[MAX_NUM_CCs];
  int N_RBG[MAX_NUM_CCs];
  nfapi_dl_config_request_body_t *dl_req;
  nfapi_dl_config_request_pdu_t *dl_config_pdu;
  int tdd_sfa;
  int ta_update;
#ifdef DEBUG_eNB_SCHEDULER
  int k;
#endif
1170 1171 1172 1173 1174 1175

  if(is_pmch_subframe(frameP,subframeP,&RC.eNB[module_idP][0]->frame_parms)){
       //LOG_E(MAC,"fairRR Frame[%d] SF:%d This SF should not be allocated\n",frameP,subframeP);
       return;
  }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1176 1177 1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231
  start_meas(&eNB->schedule_dlsch);
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME
  (VCD_SIGNAL_DUMPER_FUNCTIONS_SCHEDULE_DLSCH, VCD_FUNCTION_IN);

  // for TDD: check that we have to act here, otherwise return
  if (cc[0].tdd_Config) {
    tdd_sfa = cc[0].tdd_Config->subframeAssignment;

    switch (subframeP) {
      case 0:
        // always continue
        break;

      case 1:
        return;
        break;

      case 2:
        return;
        break;

      case 3:
        if ((tdd_sfa != 2) && (tdd_sfa != 5))
          return;

        break;

      case 4:
        if ((tdd_sfa != 1) && (tdd_sfa != 2) && (tdd_sfa != 4)
            && (tdd_sfa != 5))
          return;

        break;

      case 5:
        break;

      case 6:
      case 7:
        if ((tdd_sfa != 3)&& (tdd_sfa != 4) && (tdd_sfa != 5))
          return;

        break;

      case 8:
        if ((tdd_sfa != 2) && (tdd_sfa != 3) && (tdd_sfa != 4)
            && (tdd_sfa != 5))
          return;

        break;

      case 9:
        if (tdd_sfa == 0)
          return;

        break;
1232
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1233
  }
1234

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1235 1236 1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248
  //weight = get_ue_weight(module_idP,UE_id);
  //    aggregation = 2;
  for (CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++) {
    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];

    for (i = 0; i < N_RB_DL[CC_id]; i++)
      if (cc[CC_id].vrb_map[i] != 0)
        total_nb_available_rb[CC_id]--;

    N_RBG[CC_id] = to_rbg(cc[CC_id].mib->message.dl_Bandwidth);
    // store the global enb stats:
1249
    eNB->eNB_stats[CC_id].num_dlactive_UEs = UE_info->num_UEs;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1250 1251 1252 1253 1254 1255 1256
    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;
  }
1257

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1258
  /// CALLING Pre_Processor for downlink scheduling (Returns estimation of RBs required by each UE and the allocation on sub-band)
1259 1260 1261
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_DLSCH_PREPROCESSOR,VCD_FUNCTION_IN);
  start_meas(&eNB->schedule_dlsch_preprocessor);
  dlsch_scheduler_pre_processor_fairRR(module_idP,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1262 1263 1264 1265
                                       frameP,
                                       subframeP,
                                       N_RBG,
                                       mbsfn_flag);
1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276
  stop_meas(&eNB->schedule_dlsch_preprocessor);
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_DLSCH_PREPROCESSOR,VCD_FUNCTION_OUT);

  for (CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
    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;

    if (mbsfn_flag[CC_id]>0)
      continue;

    for (i = 0; i < dlsch_ue_select[CC_id].ue_num; i++) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1277
      if(dlsch_ue_select[CC_id].list[i].ue_priority == SCH_DL_MSG2) {
1278 1279
        continue;
      }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1280 1281

      if(dlsch_ue_select[CC_id].list[i].ue_priority == SCH_DL_MSG4) {
1282 1283
        continue;
      }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1284

1285 1286
      UE_id = dlsch_ue_select[CC_id].list[i].UE_id;
      rnti = UE_RNTI(module_idP,UE_id);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1287

1288
      if (rnti==NOT_A_RNTI) {
1289
        LOG_E(MAC,"Cannot find rnti for UE_id %d (num_UEs %d)\n",UE_id,UE_info->num_UEs);
1290 1291 1292
        continue;
      }

1293 1294
      eNB_UE_stats = &UE_info->eNB_UE_stats[CC_id][UE_id];
      ue_sched_ctl = &UE_info->UE_sched_ctrl[UE_id];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308 1309 1310 1311 1312 1313 1314 1315

      /*
            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;
            case 3:
              aggregation = get_aggregation(get_bw_index(module_idP,CC_id),
                                            ue_sched_ctl->dl_cqi[CC_id],
                                            format2A);
              break;
            default:
              LOG_W(MAC,"Unsupported transmission mode %d\n", get_tmode(module_idP,CC_id,UE_id));
              aggregation = 2;
              break;
            }
      */
1316 1317 1318 1319
      if (cc[CC_id].tdd_Config != NULL) { //TDD
        set_ue_dai (subframeP,
                    UE_id,
                    CC_id,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1320
                    cc[CC_id].tdd_Config->subframeAssignment,
1321
                    UE_info);
1322 1323 1324 1325
        // update UL DAI after DLSCH scheduling
        set_ul_DAI(module_idP,UE_id,CC_id,frameP,subframeP);
      }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1326 1327 1328
      nb_available_rb = ue_sched_ctl->pre_nb_available_rbs[CC_id];
      harq_pid = frame_subframe2_dl_harq_pid(cc->tdd_Config,frameP,subframeP);
      round = ue_sched_ctl->round[CC_id][harq_pid];
1329 1330 1331 1332
      UE_info->eNB_UE_stats[CC_id][UE_id].crnti = rnti;
      UE_info->eNB_UE_stats[CC_id][UE_id].rrc_status = mac_eNB_get_rrc_status(module_idP, rnti);
      UE_info->eNB_UE_stats[CC_id][UE_id].harq_pid = harq_pid;
      UE_info->eNB_UE_stats[CC_id][UE_id].harq_round = round;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1333

1334 1335
      if (UE_info->eNB_UE_stats[CC_id][UE_id].rrc_status < RRC_RECONFIGURED) {
        UE_info->UE_sched_ctrl[UE_id].uplane_inactivity_timer = 0;
1336 1337
      }

1338
      if (UE_info->eNB_UE_stats[CC_id][UE_id].rrc_status < RRC_CONNECTED)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1339
        continue;
1340

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1341 1342
      sdu_length_total = 0;
      num_sdus = 0;
1343

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1344 1345 1346 1347
      /*
         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);
       */
frtabu's avatar
frtabu committed
1348
      if (NFAPI_MODE != NFAPI_MONOLITHIC) {
1349
        eNB_UE_stats->dlsch_mcs1 = cqi_to_mcs[ue_sched_ctl->dl_cqi[CC_id]];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1350 1351 1352 1353
      } else {
        eNB_UE_stats->dlsch_mcs1 = cqi_to_mcs[ue_sched_ctl->dl_cqi[CC_id]];
      }

1354
      //eNB_UE_stats->dlsch_mcs1 = cmin(eNB_UE_stats->dlsch_mcs1, openair_daq_vars.target_ue_dl_mcs);
1355

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1356
      // store stats
1357
      //UE_info->eNB_UE_stats[CC_id][UE_id].dl_cqi= eNB_UE_stats->dl_cqi;
1358

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1359 1360
      // initializing the rb allocation indicator for each UE
      for (j = 0; j < N_RBG[CC_id]; j++) {
1361
        UE_info->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j] = 0;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1362
      }
1363

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1364 1365 1366 1367 1368
      LOG_D(MAC,
            "[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,
1369
            UE_info->eNB_UE_stats[CC_id][UE_id].rrc_status);
1370

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1371
      /* process retransmission  */
1372

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1373 1374
      if (round != 8) {
        // get freq_allocation
1375
        nb_rb = UE_info->UE_template[CC_id][UE_id].nb_rb[harq_pid];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1376
        TBS =
1377
          get_TBS_DL(UE_info->UE_template[CC_id][UE_id].oldmcs1[harq_pid],
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1378
                     nb_rb);
1379

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1380 1381
        if (nb_rb <= nb_available_rb) {
          if (cc[CC_id].tdd_Config != NULL) {
1382
            UE_info->UE_template[CC_id][UE_id].DAI++;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1383
            update_ul_dci(module_idP, CC_id, rnti,
1384
                          UE_info->UE_template[CC_id][UE_id].DAI,subframeP);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1385 1386 1387
            LOG_D(MAC,
                  "DAI update: CC_id %d subframeP %d: UE %d, DAI %d\n",
                  CC_id, subframeP, UE_id,
1388
                  UE_info->UE_template[CC_id][UE_id].DAI);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1389 1390 1391 1392
          }

          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
1393
              UE_info->UE_template[CC_id][UE_id].
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1394 1395
              rballoc_subband[harq_pid][j] =
                ue_sched_ctl->rballoc_sub_UE[CC_id][j];
1396
            }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1397 1398 1399 1400 1401 1402 1403
          } 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) {
1404 1405
                if (UE_info->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j])
                  LOG_W(MAC, "WARN: rballoc_subband not free for retrans?\n");
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1406

1407
                UE_info->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j] = ue_sched_ctl->rballoc_sub_UE[CC_id][j];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1408 1409 1410 1411 1412 1413 1414 1415 1416 1417 1418 1419 1420 1421

                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];
                }
              }

              j = j + 1;
1422
            }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1423 1424 1425 1426 1427 1428 1429 1430
          }

          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++) {
1431
             eNB->mu_mimo_mode[UE_id].rballoc_sub[j] = UE_info->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459 1460 1461 1462 1463 1464 1465 1466 1467
             }
           */

          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 =
                get_aggregation(get_bw_index
                                (module_idP, CC_id),
                                ue_sched_ctl->dl_cqi[CC_id],
                                format1);
              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;  // dont adjust power when retransmitting
              dl_config_pdu->dci_dl_pdu.
1468 1469 1470
              dci_dl_pdu_rel8.new_data_indicator_1 = UE_info->UE_template[CC_id][UE_id].oldNDI[harq_pid];
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.mcs_1 = UE_info->UE_template[CC_id][UE_id].oldmcs1[harq_pid];
              dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.redundancy_version_1 = round & 3;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1471 1472 1473

              if (cc[CC_id].tdd_Config != NULL) { //TDD
                dl_config_pdu->dci_dl_pdu.
1474
                dci_dl_pdu_rel8.downlink_assignment_index = (UE_info->UE_template[CC_id][UE_id].DAI - 1) & 3;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1475 1476 1477
                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,
1478 1479
                      UE_info->UE_template[CC_id][UE_id].DAI - 1,
                      UE_info-> UE_template[CC_id][UE_id].oldmcs1[harq_pid]);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1480 1481 1482 1483
              } 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,
1484
                      UE_info->UE_template[CC_id][UE_id].oldmcs1[harq_pid]);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1485 1486 1487 1488 1489 1490 1491 1492 1493 1494 1495 1496 1497 1498 1499 1500
              }

              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;
                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
1501
                                        getQm(UE_info->UE_template[CC_id][UE_id].oldmcs1[harq_pid]), round & 3, // redundancy version
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1502 1503 1504 1505 1506 1507 1508
                                        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
                                        //                      uint8_t codebook_index,
                                        4,  // UE category capacity
1509
                                        UE_info->UE_template[CC_id][UE_id].physicalConfigDedicated->pdsch_ConfigDedicated->p_a, 0,  // delta_power_offset for TM5
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1510 1511 1512 1513 1514 1515 1516 1517 1518 1519 1520 1521 1522 1523 1524 1525 1526 1527 1528 1529 1530 1531 1532
                                        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
                                       );
                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);
              }
          }

          add_ue_dlsch_info(module_idP,
1533 1534
                            CC_id,
                            UE_id,
Stefan's avatar
Stefan committed
1535 1536 1537
                            subframeP,
                            S_DL_SCHEDULED,
                            rnti);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1538
          //eNB_UE_stats->dlsch_trials[round]++;
1539 1540 1541 1542 1543
          UE_info->eNB_UE_stats[CC_id][UE_id].num_retransmission += 1;
          UE_info->eNB_UE_stats[CC_id][UE_id].rbs_used_retx = nb_rb;
          UE_info->eNB_UE_stats[CC_id][UE_id].total_rbs_used_retx += nb_rb;
          UE_info->eNB_UE_stats[CC_id][UE_id].dlsch_mcs1 = eNB_UE_stats->dlsch_mcs1;
          UE_info->eNB_UE_stats[CC_id][UE_id].dlsch_mcs2 = eNB_UE_stats->dlsch_mcs1;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1544 1545 1546 1547 1548 1549 1550 1551 1552 1553 1554 1555 1556 1557 1558 1559 1560 1561 1562 1563 1564 1565 1566 1567 1568 1569 1570 1571 1572 1573 1574
        } 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);
        // check first for RLC data on DCCH
        // add the length for  all the control elements (timing adv, drx, etc) : header + payload

        if (ue_sched_ctl->ta_timer == 0) {
          ta_update = ue_sched_ctl->ta_update;

          /* if we send TA then set timer to not send it for a while */
          if (ta_update != 31)
            ue_sched_ctl->ta_timer = 20;

          /* reset ta_update */
          ue_sched_ctl->ta_update = 31;
        } else {
          ta_update = 31;
        }

        ta_len = (ta_update != 31) ? 2 : 0;
        header_len_dcch = 2;  // 2 bytes DCCH SDU subheader

        if (TBS - ta_len - header_len_dcch > 0) {
1575
          rlc_status = mac_rlc_status_ind(module_idP, rnti, module_idP, frameP, subframeP, ENB_FLAG_YES, MBMS_FLAG_NO, DCCH, 0, 0); // transport block set size
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1576 1577 1578 1579 1580 1581 1582
          sdu_lengths[0] = 0;

          if (rlc_status.bytes_in_buffer > 0) { // There is DCCH to transmit
            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 - header_len_dcch);
1583 1584
            sdu_lengths[0] = mac_rlc_data_req(module_idP, rnti, module_idP, frameP, ENB_FLAG_YES, MBMS_FLAG_NO, DCCH,
                                              TBS - ta_len - header_len_dcch,
1585
                                              (char *) &dlsch_buffer[0],0, 0
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1586 1587
                                             );

1588
            if((rrc_release_info.num_UEs > 0) && (rlc_am_mui.rrc_mui_num > 0)) {
1589 1590 1591
              while(pthread_mutex_trylock(&rrc_release_freelist)) {
                /* spin... */
              }
1592

1593
              uint16_t release_total = 0;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1594 1595 1596

              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) {
1597
                  release_total++;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1598
                } else {
1599 1600 1601
                  continue;
                }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1602 1603 1604 1605
                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]) {
1606 1607 1608
                        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;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1609 1610
                      }
                    }
1611 1612
                  }
                }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1613 1614 1615 1616 1617 1618 1619 1620

                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]) {
                        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;
1621 1622 1623 1624
                      }
                    }
                  }
                }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1625

1626 1627 1628
                if(release_total >= rrc_release_info.num_UEs)
                  break;
              }
1629

1630
              pthread_mutex_unlock(&rrc_release_freelist);
1631 1632 1633
            }

            RA_t *ra = &eNB->common_channels[CC_id].ra[0];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1634

1635
            for (uint8_t ra_ii = 0; ra_ii < NB_RA_PROC_MAX; ra_ii++) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1636 1637 1638
              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]) {
1639 1640 1641 1642 1643 1644 1645
                    ra[ra_ii].crnti_harq_pid = harq_pid;
                    ra[ra_ii].state = MSGCRNTI_ACK;
                    break;
                  }
                }
              }
            }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1646 1647 1648 1649 1650 1651 1652 1653 1654 1655

            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]));
            LOG_D(MAC,
                  "[eNB %d][DCCH] CC_id %d frame %d subframe %d UE_id %d/%x Got %d bytes bytes_in_buffer %d from release_num %d\n",
                  module_idP, CC_id, frameP, subframeP, UE_id, rnti, sdu_lengths[0],rlc_status.bytes_in_buffer,rrc_release_info.num_UEs);
            sdu_length_total = sdu_lengths[0];
            sdu_lcids[0] = DCCH;
1656 1657
            UE_info->eNB_UE_stats[CC_id][UE_id].num_pdu_tx[DCCH] += 1;
            UE_info->eNB_UE_stats[CC_id][UE_id].num_bytes_tx[DCCH] += sdu_lengths[0];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1658
            num_sdus = 1;
1659
#ifdef DEBUG_eNB_SCHEDULER
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1660 1661 1662
            LOG_T(MAC,
                  "[eNB %d][DCCH] CC_id %d Got %d bytes :",
                  module_idP, CC_id, sdu_lengths[0]);
1663

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1664 1665 1666
            for (k = 0; k < sdu_lengths[0]; k++) {
              LOG_T(MAC, "%x ", dlsch_buffer[k]);
            }
1667

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1668
            LOG_T(MAC, "\n");
1669
#endif
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1670 1671 1672 1673 1674 1675 1676 1677
          } else {
            header_len_dcch = 0;
            sdu_length_total = 0;
          }
        }

        // check for DCCH1 and update header information (assume 2 byte sub-header)
        if (TBS - ta_len - header_len_dcch - sdu_length_total > 0) {
1678
          rlc_status = mac_rlc_status_ind(module_idP, rnti, module_idP, frameP, subframeP, ENB_FLAG_YES, MBMS_FLAG_NO, DCCH + 1, 0, 0
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1679 1680 1681 1682 1683 1684 1685 1686 1687
                                         ); // transport block set size less allocations for timing advance and
          // 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 - header_len_dcch - sdu_length_total);
1688 1689
            sdu_lengths[num_sdus] += mac_rlc_data_req(module_idP, rnti, module_idP, frameP, ENB_FLAG_YES, MBMS_FLAG_NO, DCCH + 1,
                                                      TBS - ta_len - header_len_dcch - sdu_length_total,
1690
                                     (char *) &dlsch_buffer[sdu_length_total],0, 0
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1691 1692 1693 1694 1695 1696 1697 1698
                                                     );
            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]));
            sdu_lcids[num_sdus] = DCCH1;
            sdu_length_total += sdu_lengths[num_sdus];
            header_len_dcch += 2;
1699 1700
            UE_info->eNB_UE_stats[CC_id][UE_id].num_pdu_tx[DCCH1] += 1;
            UE_info->eNB_UE_stats[CC_id][UE_id].num_bytes_tx[DCCH1] += sdu_lengths[num_sdus];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1701
            num_sdus++;
1702
#ifdef DEBUG_eNB_SCHEDULER
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1703 1704 1705
            LOG_T(MAC,
                  "[eNB %d][DCCH1] CC_id %d Got %d bytes :",
                  module_idP, CC_id, sdu_lengths[num_sdus]);
1706

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1707 1708 1709
            for (k = 0; k < sdu_lengths[num_sdus]; k++) {
              LOG_T(MAC, "%x ", dlsch_buffer[k]);
            }
1710

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1711
            LOG_T(MAC, "\n");
1712
#endif
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1713 1714
          }
        }
1715

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1716 1717 1718
        // assume the max dtch header size, and adjust it later
        header_len_dtch = 0;
        header_len_dtch_last = 0; // the header length of the last mac sdu
1719

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1720 1721
        // lcid has to be sorted before the actual allocation (similar struct as ue_list).
        /* TODO limited lcid for performance */
1722
        for (lcid = DTCH; lcid >= DTCH; lcid--) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1723 1724 1725 1726 1727 1728 1729 1730 1731 1732 1733 1734 1735 1736 1737 1738 1739 1740 1741
          // TBD: check if the lcid is active
          header_len_dtch += 3;
          header_len_dtch_last = 3;
          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_len_dcch - sdu_length_total - header_len_dtch);

          if (TBS - ta_len - header_len_dcch - sdu_length_total - header_len_dtch > 0) {  // NN: > 2 ?
            rlc_status = mac_rlc_status_ind(module_idP,
                                            rnti,
                                            module_idP,
                                            frameP,
                                            subframeP,
                                            ENB_FLAG_YES,
                                            MBMS_FLAG_NO,
                                            lcid,
1742
                                            0, 0
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1743 1744 1745 1746 1747 1748 1749 1750 1751 1752 1753 1754 1755 1756 1757 1758
                                           );

            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 - header_len_dcch - sdu_length_total - header_len_dtch,
                    lcid,
                    header_len_dtch);
              sdu_lengths[num_sdus] = mac_rlc_data_req(module_idP,
                                      rnti,
                                      module_idP,
                                      frameP,
                                      ENB_FLAG_YES,
                                      MBMS_FLAG_NO,
                                      lcid,
1759
                                      TBS - ta_len - header_len_dcch - sdu_length_total - header_len_dtch,
1760
                                      (char *)&dlsch_buffer[sdu_length_total], 0, 0
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1761 1762 1763 1764 1765 1766 1767 1768 1769 1770 1771 1772 1773 1774 1775 1776
                                                      );
              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];
1777 1778
              UE_info->eNB_UE_stats[CC_id][UE_id].num_pdu_tx[lcid] += 1;
              UE_info->eNB_UE_stats[CC_id][UE_id].num_bytes_tx[lcid] += sdu_lengths[num_sdus];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1779 1780 1781 1782 1783 1784 1785

              if (sdu_lengths[num_sdus] < 128) {
                header_len_dtch--;
                header_len_dtch_last--;
              }

              num_sdus++;
1786
              UE_info->UE_sched_ctrl[UE_id].uplane_inactivity_timer = 0;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1787 1788 1789 1790 1791 1792 1793
            } else { // no data for this LCID
              header_len_dtch -= 3;
            }
          } else {  // no TBS left
            header_len_dtch -= 3;
            break;
          }
1794
        }
1795

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1796 1797 1798 1799 1800 1801 1802 1803 1804 1805 1806 1807 1808 1809 1810 1811 1812 1813 1814 1815 1816 1817 1818 1819 1820 1821 1822 1823 1824 1825 1826 1827 1828 1829 1830 1831 1832 1833 1834 1835 1836 1837 1838
        if (header_len_dtch == 0)
          header_len_dtch_last = 0;

        // there is at least one SDU
        // if (num_sdus > 0 ){
        if ((sdu_length_total + header_len_dcch +
             header_len_dtch) > 0) {
          // Now compute number of required RBs for total sdu length
          // Assume RAH format 2
          // adjust  header lengths
          header_len_dcch_tmp = header_len_dcch;
          header_len_dtch_tmp = header_len_dtch;

          if (header_len_dtch == 0) {
            header_len_dcch = (header_len_dcch > 0) ? 1 : 0;  //header_len_dcch;  // remove length field
          } else {
            header_len_dtch_last -= 1;  // now use it to find how many bytes has to be removed for the last MAC SDU
            header_len_dtch = (header_len_dtch > 0) ? header_len_dtch - header_len_dtch_last : header_len_dtch; // remove length field for the last SDU
          }

          mcs = eNB_UE_stats->dlsch_mcs1;
          nb_rb = min_rb_unit[CC_id];
          TBS = get_TBS_DL(mcs, nb_rb);

          while (TBS <
                 (sdu_length_total + header_len_dcch +
                  header_len_dtch + ta_len)) {
            nb_rb += min_rb_unit[CC_id];  //

            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;
            }

            TBS = get_TBS_DL(eNB_UE_stats->dlsch_mcs1, nb_rb);
          }

          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
1839
              UE_info->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j] = ue_sched_ctl->rballoc_sub_UE[CC_id][j];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1840 1841 1842 1843 1844 1845 1846 1847
            }
          } 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) {
1848
                UE_info->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][j] = ue_sched_ctl->rballoc_sub_UE[CC_id][j];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1849 1850 1851 1852 1853 1854 1855 1856 1857 1858 1859 1860 1861 1862 1863 1864 1865 1866 1867 1868 1869 1870 1871 1872 1873 1874 1875 1876 1877 1878 1879 1880 1881 1882 1883 1884

                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];
                }
              }

              j = j + 1;
            }
          }

          // decrease mcs until TBS falls below required length
          while ((TBS >
                  (sdu_length_total + header_len_dcch +
                   header_len_dtch + 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_len_dcch +
                   header_len_dtch + 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);
          }
1885

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1886 1887 1888
          LOG_D(MAC,
                "dlsch_mcs before and after the rate matching = (%d, %d)\n",
                eNB_UE_stats->dlsch_mcs1, mcs);
1889
#ifdef DEBUG_eNB_SCHEDULER
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1890 1891 1892 1893 1894
          LOG_D(MAC,
                "[eNB %d] CC_id %d Generated DLSCH header (mcs %d, TBS %d, nb_rb %d)\n",
                module_idP, CC_id, mcs, TBS, nb_rb);
          // msg("[MAC][eNB ] Reminder of DLSCH with random data %d %d %d %d \n",
          //  TBS, sdu_length_total, offset, TBS-sdu_length_total-offset);
1895 1896
#endif

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1897 1898 1899 1900 1901 1902 1903 1904 1905 1906 1907 1908 1909 1910 1911 1912 1913 1914
          if ((TBS - header_len_dcch - header_len_dtch -
               sdu_length_total - ta_len) <= 2) {
            padding =
              (TBS - header_len_dcch - header_len_dtch -
               sdu_length_total - ta_len);
            post_padding = 0;
          } else {
            padding = 0;

            // adjust the header len
            if (header_len_dtch == 0) {
              header_len_dcch = header_len_dcch_tmp;
            } else {  //if (( header_len_dcch==0)&&((header_len_dtch==1)||(header_len_dtch==2)))
              header_len_dtch = header_len_dtch_tmp;
            }

            post_padding = TBS - sdu_length_total - header_len_dcch - header_len_dtch - ta_len; // 1 is for the postpadding header
          }
1915 1916

#ifdef PHY_TX_THREAD
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1917 1918 1919 1920 1921 1922 1923 1924 1925
          struct timespec time_req, time_rem;
          time_req.tv_sec = 0;
          time_req.tv_nsec = 10000;

          while((!oai_exit)&&(phy_tx_txdataF_end == 0)) {
            nanosleep(&time_req,&time_rem);
            continue;
          }

1926
#endif
1927
          offset = generate_dlsch_header((unsigned char *) UE_info->DLSCH_pdu[CC_id][0][UE_id].payload[0], num_sdus,  //num_sdus
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1928 1929 1930 1931 1932 1933 1934 1935 1936 1937 1938 1939 1940 1941 1942 1943
                                         sdu_lengths, //
                                         sdu_lcids, 255,  // no drx
                                         ta_update, // timing advance
                                         NULL,  // contention res id
                                         padding, post_padding);

          //#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_dcch %d, header_dtch %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_len_dcch, header_len_dtch);
          }
1944

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1945
          //#endif
1946
#ifdef DEBUG_eNB_SCHEDULER
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1947
          LOG_T(MAC, "[eNB %d] First 16 bytes of DLSCH : \n",module_idP );
1948

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1949 1950 1951
          for (k = 0; k < 16; k++) {
            LOG_T(MAC, "%x.", dlsch_buffer[k]);
          }
1952

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1953
          LOG_T(MAC, "\n");
1954 1955
#endif
          // cycle through SDUs and place in dlsch_buffer
1956
          memcpy(&UE_info->DLSCH_pdu[CC_id][0][UE_id].payload[0][offset],dlsch_buffer,sdu_length_total);
1957 1958 1959 1960
          // memcpy(RC.mac[0].DLSCH_pdu[0][0].payload[0][offset],dcch_buffer,sdu_lengths[0]);

          // fill remainder of DLSCH with random data
          for (j=0; j<(TBS-sdu_length_total-offset); j++) {
1961
            UE_info->DLSCH_pdu[CC_id][0][UE_id].payload[0][offset+sdu_length_total+j] = (char)(taus()&0xff);
1962 1963
          }

1964
          trace_pdu(DIRECTION_DOWNLINK, (uint8_t *)UE_info->DLSCH_pdu[CC_id][0][UE_id].payload[0],
laurent's avatar
laurent committed
1965 1966
                    TBS, module_idP, WS_RA_RNTI, UE_RNTI(module_idP, UE_id),
                    eNB->frame, eNB->subframe,0,0);
1967
          T(T_ENB_MAC_UE_DL_PDU_WITH_DATA, T_INT(module_idP), T_INT(CC_id), T_INT(rnti), T_INT(frameP), T_INT(subframeP),
1968 1969
            T_INT(harq_pid), T_BUFFER(UE_info->DLSCH_pdu[CC_id][0][UE_id].payload[0], TBS));
          UE_info->UE_template[CC_id][UE_id].nb_rb[harq_pid] = nb_rb;
1970 1971 1972 1973
          add_ue_dlsch_info(module_idP,
                            CC_id,
                            UE_id,
                            subframeP,
Stefan's avatar
Stefan committed
1974 1975
                            S_DL_SCHEDULED,
                            rnti);
1976 1977 1978
          // store stats
          eNB->eNB_stats[CC_id].dlsch_bytes_tx+=sdu_length_total;
          eNB->eNB_stats[CC_id].dlsch_pdus_tx+=1;
1979 1980 1981 1982 1983 1984 1985 1986 1987
          UE_info->eNB_UE_stats[CC_id][UE_id].rbs_used = nb_rb;
          UE_info->eNB_UE_stats[CC_id][UE_id].total_rbs_used += nb_rb;
          UE_info->eNB_UE_stats[CC_id][UE_id].dlsch_mcs1=eNB_UE_stats->dlsch_mcs1;
          UE_info->eNB_UE_stats[CC_id][UE_id].dlsch_mcs2=mcs;
          UE_info->eNB_UE_stats[CC_id][UE_id].TBS = TBS;
          UE_info->eNB_UE_stats[CC_id][UE_id].overhead_bytes= TBS- sdu_length_total;
          UE_info->eNB_UE_stats[CC_id][UE_id].total_sdu_bytes+= sdu_length_total;
          UE_info->eNB_UE_stats[CC_id][UE_id].total_pdu_bytes+= TBS;
          UE_info->eNB_UE_stats[CC_id][UE_id].total_num_pdus+=1;
1988 1989

          if (cc[CC_id].tdd_Config != NULL) { // TDD
1990 1991
            UE_info->UE_template[CC_id][UE_id].DAI++;
            update_ul_dci(module_idP,CC_id,rnti,UE_info->UE_template[CC_id][UE_id].DAI,subframeP);
1992 1993
          }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
1994
          // do PUCCH power control
Cedric Roux's avatar
Cedric Roux committed
1995
          // this is the snr
1996
          eNB_UE_stats =  &UE_info->eNB_UE_stats[CC_id][UE_id];
1997
          /* Unit is not dBm, it's special from nfapi */
Cedric Roux's avatar
Cedric Roux committed
1998 1999
          snr = (5 * ue_sched_ctl->pucch1_snr[CC_id] - 640) / 10;
          target_snr = eNB->puCch10xSnr / 10;
2000
          // this assumes accumulated tpc
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2001
          // make sure that we are only sending a tpc update once a frame, otherwise the control loop will freak out
2002
          int32_t framex10psubframe = UE_info->UE_template[CC_id][UE_id].pucch_tpc_tx_frame*10+UE_info->UE_template[CC_id][UE_id].pucch_tpc_tx_subframe;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2003

2004
          if (((framex10psubframe+10)<=(frameP*10+subframeP)) || //normal case
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2005 2006 2007
              ((framex10psubframe>(frameP*10+subframeP)) && (((10240-framex10psubframe+frameP*10+subframeP)>=10)))) //frame wrap-around
            if (ue_sched_ctl->pucch1_cqi_update[CC_id] == 1) {
              ue_sched_ctl->pucch1_cqi_update[CC_id] = 0;
2008 2009
              UE_info->UE_template[CC_id][UE_id].pucch_tpc_tx_frame=frameP;
              UE_info->UE_template[CC_id][UE_id].pucch_tpc_tx_subframe=subframeP;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2010

Cedric Roux's avatar
Cedric Roux committed
2011
              if (snr > target_snr + 4) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2012 2013
                tpc = 0; //-1
                tpc_accumulated--;
Cedric Roux's avatar
Cedric Roux committed
2014
              } else if (snr < target_snr - 4) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2015 2016 2017 2018 2019 2020
                tpc = 2; //+1
                tpc_accumulated++;
              } else {
                tpc = 1; //0
              }

Cedric Roux's avatar
Cedric Roux committed
2021
              LOG_D(MAC,"[eNB %d] DLSCH scheduler: frame %d, subframe %d, harq_pid %d, tpc %d, accumulated %d, snr/target snr %d/%d\n",
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2022
                    module_idP,frameP, subframeP,harq_pid,tpc,
Cedric Roux's avatar
Cedric Roux committed
2023
                    tpc_accumulated,snr,target_snr);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2024 2025 2026 2027 2028 2029 2030
            } // Po_PUCCH has been updated
            else {
              tpc = 1; //0
            } // time to do TPC update
          else {
            tpc = 1; //0
          }
2031

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2032 2033 2034 2035 2036 2037
          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.dci_format                  = NFAPI_DL_DCI_FORMAT_1;
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.aggregation_level           = get_aggregation(get_bw_index(module_idP,CC_id),ue_sched_ctl->dl_cqi[CC_id],format1);
2038
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.tl.tag                      = NFAPI_DL_CONFIG_REQUEST_DCI_DL_PDU_REL8_TAG;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2039 2040 2041 2042 2043
          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                         = tpc; // dont adjust power when retransmitting
2044
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.new_data_indicator_1        = 1-UE_info->UE_template[CC_id][UE_id].oldNDI[harq_pid];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2045 2046 2047 2048 2049 2050 2051
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.mcs_1                       = mcs;
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.redundancy_version_1        = 0;
          //deactivate second codeword
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.mcs_2                       = 0;
          dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.redundancy_version_2        = 1;

          if (cc[CC_id].tdd_Config != NULL) { //TDD
2052
            dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.downlink_assignment_index = (UE_info->UE_template[CC_id][UE_id].DAI-1)&3;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2053 2054
            LOG_D(MAC,"[eNB %d] Initial transmission CC_id %d : harq_pid %d, dai %d, mcs %d\n",
                  module_idP,CC_id,harq_pid,
2055
                  (UE_info->UE_template[CC_id][UE_id].DAI-1),
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2056 2057 2058 2059 2060
                  mcs);
          } else {
            LOG_D(MAC,"[eNB %d] Initial transmission CC_id %d : harq_pid %d, mcs %d\n",
                  module_idP,CC_id,harq_pid,mcs);
          }
2061

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2062 2063 2064 2065 2066 2067 2068 2069 2070 2071 2072 2073
          LOG_D(MAC,"Checking feasibility pdu %d (new sdu)\n",dl_req->number_pdu);

          if (!CCE_allocation_infeasible(module_idP,CC_id,1,subframeP,dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.aggregation_level,rnti)) {
            ue_sched_ctl->round[CC_id][harq_pid] = 0;
            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;
            // Toggle NDI for next time
            LOG_D(MAC,"CC_id %d Frame %d, subframeP %d: Toggling Format1 NDI for UE %d (rnti %x/%d) oldNDI %d\n",
                  CC_id, frameP,subframeP,UE_id,
2074 2075 2076 2077 2078 2079
                  rnti,harq_pid,UE_info->UE_template[CC_id][UE_id].oldNDI[harq_pid]);
            UE_info->UE_template[CC_id][UE_id].oldNDI[harq_pid]=1-UE_info->UE_template[CC_id][UE_id].oldNDI[harq_pid];
            UE_info->UE_template[CC_id][UE_id].oldmcs1[harq_pid] = mcs;
            UE_info->UE_template[CC_id][UE_id].oldmcs2[harq_pid] = 0;
            AssertFatal(UE_info->UE_template[CC_id][UE_id].physicalConfigDedicated!=NULL,"physicalConfigDedicated is NULL\n");
            AssertFatal(UE_info->UE_template[CC_id][UE_id].physicalConfigDedicated->pdsch_ConfigDedicated!=NULL,"physicalConfigDedicated->pdsch_ConfigDedicated is NULL\n");
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2080 2081 2082 2083 2084 2085 2086 2087 2088 2089 2090 2091 2092 2093 2094 2095
            fill_nfapi_dlsch_config(eNB,dl_req,
                                    TBS,
                                    eNB->pdu_index[CC_id],
                                    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(mcs),
                                    0, // 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
                                    //           uint8_t codebook_index,
                                    4, // UE category capacity
2096
                                    UE_info->UE_template[CC_id][UE_id].physicalConfigDedicated->pdsch_ConfigDedicated->p_a,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2097 2098 2099 2100 2101 2102 2103 2104 2105 2106 2107
                                    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
                                   );
            eNB->TX_req[CC_id].sfn_sf = fill_nfapi_tx_req(&eNB->TX_req[CC_id].tx_request_body,
                                        (frameP*10)+subframeP,
                                        TBS,
                                        eNB->pdu_index[CC_id],
2108
                                        eNB->UE_info.DLSCH_pdu[CC_id][0][(unsigned char)UE_id].payload[0]);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2109 2110 2111 2112 2113 2114 2115 2116 2117
            LOG_D(MAC,"Filled NFAPI configuration for DCI/DLSCH/TXREQ %d, new SDU\n",eNB->pdu_index[CC_id]);
            eNB->pdu_index[CC_id]++;
            program_dlsch_acknak(module_idP,CC_id,UE_id,frameP,subframeP,dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.cce_idx);
            last_dlsch_ue_id[CC_id] = UE_id;
          } else {
            LOG_W(MAC,"Frame %d, Subframe %d: Dropping DLSCH allocation for UE %d/%x, infeasible CCE allocations\n",
                  frameP,subframeP,UE_id,rnti);
          }
        } else {  // There is no data from RLC or MAC header, so don't schedule
2118 2119 2120
        }
      }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2121
      if (cc[CC_id].tdd_Config != NULL) { // TDD
2122 2123 2124 2125 2126 2127 2128 2129 2130 2131 2132 2133 2134
        set_ul_DAI(module_idP,UE_id,CC_id,frameP,subframeP);
      }
    } // UE_id loop
  }  // CC_id loop

  fill_DLSCH_dci_fairRR(module_idP,frameP,subframeP,mbsfn_flag);
  stop_meas(&eNB->schedule_dlsch);
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_SCHEDULE_DLSCH,VCD_FUNCTION_OUT);
}

//------------------------------------------------------------------------------
void
fill_DLSCH_dci_fairRR(
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2135 2136 2137 2138
  module_id_t module_idP,
  frame_t frameP,
  sub_frame_t subframeP,
  int *mbsfn_flagP)
2139 2140 2141 2142 2143 2144 2145 2146 2147 2148 2149 2150 2151 2152
//------------------------------------------------------------------------------
{
  // loop over all allocated UEs and compute frequency allocations for PDSCH
  int   UE_id = -1;
  uint8_t            /* first_rb, */ nb_rb=3;
  rnti_t        rnti;
  //unsigned char *vrb_map;
  uint8_t            rballoc_sub[25];
  //uint8_t number_of_subbands=13;
  //unsigned char round;
  unsigned char     harq_pid;
  int               i;
  int               CC_id;
  eNB_MAC_INST      *eNB  =RC.mac[module_idP];
2153
  UE_info_t         *UE_info = &eNB->UE_info;
2154 2155 2156 2157 2158 2159 2160 2161 2162 2163 2164 2165 2166 2167 2168 2169 2170 2171 2172
  int               N_RBG;
  int               N_RB_DL;
  COMMON_channels_t *cc;
  int               j;
  start_meas(&eNB->fill_DLSCH_dci);
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_FILL_DLSCH_DCI,VCD_FUNCTION_IN);

  for (CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
    LOG_D(MAC,"Doing fill DCI for CC_id %d\n",CC_id);

    if (mbsfn_flagP[CC_id]>0)
      continue;

    cc              = &eNB->common_channels[CC_id];
    N_RBG           = to_rbg(cc->mib->message.dl_Bandwidth);
    N_RB_DL         = to_prb(cc->mib->message.dl_Bandwidth);

    // UE specific DCIs
    for (j = 0; j < dlsch_ue_select[CC_id].ue_num; j++) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2173
      if(dlsch_ue_select[CC_id].list[j].ue_priority == SCH_DL_MSG2) {
2174 2175
        continue;
      }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2176 2177

      if(dlsch_ue_select[CC_id].list[j].ue_priority == SCH_DL_MSG4) {
2178 2179
        continue;
      }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2180

2181 2182 2183 2184 2185 2186 2187
      UE_id = dlsch_ue_select[CC_id].list[j].UE_id;
      LOG_T(MAC,"CC_id %d, UE_id: %d => status %d\n",CC_id,UE_id,eNB_dlsch_info[module_idP][CC_id][UE_id].status);

      if (eNB_dlsch_info[module_idP][CC_id][UE_id].status == S_DL_SCHEDULED) {
        // clear scheduling flag
        eNB_dlsch_info[module_idP][CC_id][UE_id].status = S_DL_WAITING;
        rnti = UE_RNTI(module_idP,UE_id);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2188
        harq_pid = frame_subframe2_dl_harq_pid(cc->tdd_Config,frameP,subframeP);
2189
        nb_rb = UE_info->UE_template[CC_id][UE_id].nb_rb[harq_pid];
2190 2191 2192

        /// Synchronizing rballoc with rballoc_sub
        for(i=0; i<N_RBG; i++) {
2193
          rballoc_sub[i] = UE_info->UE_template[CC_id][UE_id].rballoc_subband[harq_pid][i];
2194 2195
        }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2196 2197 2198 2199 2200 2201 2202 2203 2204 2205 2206 2207 2208 2209 2210 2211 2212
        nfapi_dl_config_request_t      *DL_req         = &RC.mac[module_idP]->DL_req[0];
        nfapi_dl_config_request_pdu_t *dl_config_pdu;

        for (i=0; i<DL_req[CC_id].dl_config_request_body.number_pdu; i++) {
          dl_config_pdu                    = &DL_req[CC_id].dl_config_request_body.dl_config_pdu_list[i];

          if ((dl_config_pdu->pdu_type == NFAPI_DL_CONFIG_DCI_DL_PDU_TYPE)&&
              (dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.rnti == rnti) &&
              (dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.dci_format != 1)) {
            dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.resource_block_coding    = allocate_prbs_sub(nb_rb,N_RB_DL,N_RBG,rballoc_sub);
            dl_config_pdu->dci_dl_pdu.dci_dl_pdu_rel8.resource_allocation_type = 0;
          } else if ((dl_config_pdu->pdu_type == NFAPI_DL_CONFIG_DLSCH_PDU_TYPE)&&
                     (dl_config_pdu->dlsch_pdu.dlsch_pdu_rel8.rnti == rnti) &&
                     (dl_config_pdu->dlsch_pdu.dlsch_pdu_rel8.resource_allocation_type==0)) {
            dl_config_pdu->dlsch_pdu.dlsch_pdu_rel8.resource_block_coding    = allocate_prbs_sub(nb_rb,N_RB_DL,N_RBG,rballoc_sub);
          }
        }
2213 2214
      }
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2215
  }
2216

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2217 2218 2219
  stop_meas(&eNB->fill_DLSCH_dci);
  VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME
  (VCD_SIGNAL_DUMPER_FUNCTIONS_FILL_DLSCH_DCI, VCD_FUNCTION_OUT);
2220 2221 2222 2223 2224
}



void ulsch_scheduler_pre_ue_select_fairRR(
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2225 2226 2227 2228 2229
  module_id_t       module_idP,
  frame_t           frameP,
  sub_frame_t       subframeP,
  sub_frame_t       sched_subframeP,
  ULSCH_UE_SELECT   ulsch_ue_select[MAX_NUM_CCs]) {
2230 2231 2232 2233 2234 2235 2236 2237 2238 2239
  eNB_MAC_INST *eNB=RC.mac[module_idP];
  COMMON_channels_t *cc;
  int CC_id,UE_id;
  int ret;
  uint16_t i;
  uint8_t ue_first_num[MAX_NUM_CCs];
  uint8_t first_ue_total[MAX_NUM_CCs][20];
  uint8_t first_ue_id[MAX_NUM_CCs][20];
  uint8_t ul_inactivity_num[MAX_NUM_CCs];
  uint8_t ul_inactivity_id[MAX_NUM_CCs][20];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2240
  //  LTE_DL_FRAME_PARMS *frame_parms;
2241 2242 2243
  uint8_t ulsch_ue_max_num[MAX_NUM_CCs];
  uint16_t saved_ulsch_dci[MAX_NUM_CCs];
  rnti_t rnti;
2244
  UE_sched_ctrl_t *UE_sched_ctl = NULL;
2245 2246
  uint8_t cc_id_flag[MAX_NUM_CCs];
  uint8_t harq_pid = 0,round = 0;
2247
  UE_info_t *UE_info= &eNB->UE_info;
2248
  uint8_t                        aggregation;
2249 2250 2251
  int                            format_flag;
  nfapi_hi_dci0_request_body_t   *HI_DCI0_req;
  nfapi_hi_dci0_request_pdu_t    *hi_dci0_pdu;
2252
  int rrc_status;
2253 2254

  for ( CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++ ) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2255 2256 2257
    //save ulsch dci number
    saved_ulsch_dci[CC_id] = eNB->HI_DCI0_req[CC_id][subframeP].hi_dci0_request_body.number_of_dci;
    // maximum multiplicity number
2258
    ulsch_ue_max_num[CC_id] =RC.rrc[module_idP]->configuration.radioresourceconfig[CC_id].ue_multiple_max;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2259 2260 2261
    cc_id_flag[CC_id] = 0;
    ue_first_num[CC_id] = 0;
    ul_inactivity_num[CC_id] = 0;
2262
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2263

2264 2265
  // UE round >0
  for ( UE_id = 0; UE_id < NUMBER_OF_UE_MAX; UE_id++ ) {
2266
    if (UE_info->active[UE_id] == FALSE)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2267
      continue;
2268

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2269 2270 2271 2272 2273 2274 2275
    rnti = UE_RNTI(module_idP,UE_id);

    if (rnti ==NOT_A_RNTI)
      continue;

    CC_id = UE_PCCID(module_idP,UE_id);

2276
    if (UE_info->UE_template[CC_id][UE_id].configured == FALSE)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2277 2278
      continue;

2279
    if (UE_info->UE_sched_ctrl[UE_id].ul_out_of_sync == 1)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2280 2281 2282 2283 2284 2285 2286 2287 2288 2289 2290
      continue;

    // UL DCI
    HI_DCI0_req   = &eNB->HI_DCI0_req[CC_id][subframeP].hi_dci0_request_body;

    if ( (ulsch_ue_select[CC_id].ue_num >= ulsch_ue_max_num[CC_id]) || (cc_id_flag[CC_id] == 1) ) {
      cc_id_flag[CC_id] = 1;
      HI_DCI0_req->number_of_dci = saved_ulsch_dci[CC_id];
      ret = cc_id_end(cc_id_flag);

      if ( ret == 0 ) {
2291
        continue;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2292
      }
2293

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2294 2295 2296 2297 2298 2299 2300 2301 2302
      if ( ret == 1 ) {
        return;
      }
    }

    cc = &eNB->common_channels[CC_id];
    //harq_pid
    harq_pid = subframe2harqpid(cc,(frameP+(sched_subframeP<subframeP ? 1 : 0)),sched_subframeP);
    //round
2303
    round = UE_info->UE_sched_ctrl[UE_id].round_UL[CC_id][harq_pid];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2304 2305 2306 2307

    if ( round > 0 ) {
      hi_dci0_pdu   = &HI_DCI0_req->hi_dci0_pdu_list[HI_DCI0_req->number_of_dci+HI_DCI0_req->number_of_hi];
      format_flag = 2;
2308
      aggregation=get_aggregation(get_bw_index(module_idP,CC_id),UE_info->UE_sched_ctrl[UE_id].dl_cqi[CC_id],format0);
2309

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2310 2311 2312 2313 2314 2315 2316 2317 2318
      if (CCE_allocation_infeasible(module_idP,CC_id,format_flag,subframeP,aggregation,rnti) == 1) {
        cc_id_flag[CC_id] = 1;
        continue;
      } else {
        hi_dci0_pdu->pdu_type                               = NFAPI_HI_DCI0_DCI_PDU_TYPE;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.rnti              = rnti;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.aggregation_level = aggregation;
        HI_DCI0_req->number_of_dci++;
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].ue_priority = SCH_UL_RETRANS;
2319 2320
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].start_rb = eNB->UE_info.UE_template[CC_id][UE_id].first_rb_ul[harq_pid];
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].nb_rb = eNB->UE_info.UE_template[CC_id][UE_id].nb_rb_ul[harq_pid];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2321 2322
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].UE_id = UE_id;
        ulsch_ue_select[CC_id].ue_num++;
2323
        continue;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2324 2325
      }
    }
2326

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2327
    //
2328
    int bytes_to_schedule = UE_info->UE_template[CC_id][UE_id].estimated_ul_buffer - UE_info->UE_template[CC_id][UE_id].scheduled_ul_bytes;
2329

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2330 2331 2332 2333 2334 2335 2336
    if (bytes_to_schedule < 0) bytes_to_schedule = 0;

    if ( UE_id > last_ulsch_ue_id[CC_id] && ((ulsch_ue_select[CC_id].ue_num+ue_first_num[CC_id]) < ulsch_ue_max_num[CC_id]) ) {
      if ( bytes_to_schedule > 0 ) {
        first_ue_id[CC_id][ue_first_num[CC_id]]= UE_id;
        first_ue_total[CC_id][ue_first_num[CC_id]] = bytes_to_schedule;
        ue_first_num[CC_id]++;
2337
        continue;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2338
      }
2339

2340
      if ( UE_info->UE_template[CC_id][UE_id].ul_SR > 0 ) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2341 2342 2343 2344
        first_ue_id[CC_id][ue_first_num[CC_id]]= UE_id;
        first_ue_total[CC_id] [ue_first_num[CC_id]] = 0;
        ue_first_num[CC_id]++;
        continue;
2345 2346
      }

2347
      UE_sched_ctl = &UE_info->UE_sched_ctrl[UE_id];
2348
      rrc_status = mac_eNB_get_rrc_status(module_idP, rnti);
2349

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2350
      if ( ((UE_sched_ctl->ul_inactivity_timer>20)&&(UE_sched_ctl->ul_scheduled==0))  ||
2351 2352
           ((UE_sched_ctl->ul_inactivity_timer>10)&&(UE_sched_ctl->ul_scheduled==0)&&(rrc_status < RRC_CONNECTED)) ||
           ((UE_sched_ctl->cqi_req_timer>300)&&((rrc_status >= RRC_CONNECTED))) ) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2353 2354 2355 2356
        first_ue_id[CC_id][ue_first_num[CC_id]]= UE_id;
        first_ue_total[CC_id] [ue_first_num[CC_id]] = 0;
        ue_first_num[CC_id]++;
        continue;
2357
      }
2358

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2359
      /*if ( (ulsch_ue_select[CC_id].ue_num+ul_inactivity_num[CC_id] ) < ulsch_ue_max_num[CC_id] ) {
2360
          UE_sched_ctl = &UE_info->UE_sched_ctrl[UE_id];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2361 2362 2363 2364 2365 2366 2367
          uint8_t ul_period = 0;
          if (cc->tdd_Config) {
            ul_period = 50;
          } else {
            ul_period = 20;
          }
          if ( ((UE_sched_ctl->ul_inactivity_timer>ul_period)&&(UE_sched_ctl->ul_scheduled==0))  ||
2368
            ((UE_sched_ctl->ul_inactivity_timer>10)&&(UE_sched_ctl->ul_scheduled==0)&&(mac_eNB_get_rrc_status(module_idP,UE_RNTI(module_idP,UE_id)) < RRC_CONNECTED))) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2369 2370
          ul_inactivity_id[CC_id][ul_inactivity_num[CC_id]]= UE_id;
          ul_inactivity_num[CC_id] ++;
2371 2372
          continue;
        }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2373 2374
      }*/
    }
2375 2376 2377 2378
  }

  for ( CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++ ) {
    HI_DCI0_req   = &eNB->HI_DCI0_req[CC_id][subframeP].hi_dci0_request_body;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2379

2380 2381 2382 2383 2384 2385 2386 2387 2388 2389
    for ( int temp = 0; temp < ue_first_num[CC_id]; temp++ ) {
      if ( (ulsch_ue_select[CC_id].ue_num >= ulsch_ue_max_num[CC_id]) || (cc_id_flag[CC_id] == 1) ) {
        cc_id_flag[CC_id] = 1;
        HI_DCI0_req->number_of_dci = saved_ulsch_dci[CC_id];
        break;
      }

      hi_dci0_pdu   = &HI_DCI0_req->hi_dci0_pdu_list[HI_DCI0_req->number_of_dci+HI_DCI0_req->number_of_hi];
      format_flag = 2;
      rnti = UE_RNTI(module_idP,first_ue_id[CC_id][temp]);
2390
      aggregation=get_aggregation(get_bw_index(module_idP,CC_id),UE_info->UE_sched_ctrl[first_ue_id[CC_id][temp]].dl_cqi[CC_id],format0);
2391

2392
      if (CCE_allocation_infeasible(module_idP,CC_id,format_flag,subframeP,aggregation,rnti) == 1) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2393 2394
        cc_id_flag[CC_id] = 1;
        break;
2395
      } else {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2396 2397 2398 2399 2400 2401 2402 2403
        hi_dci0_pdu->pdu_type                               = NFAPI_HI_DCI0_DCI_PDU_TYPE;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.rnti              = rnti;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.aggregation_level = aggregation;
        HI_DCI0_req->number_of_dci++;
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].ue_priority = SCH_UL_FIRST;
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].ul_total_buffer = first_ue_total[CC_id][temp];
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].UE_id = first_ue_id[CC_id][temp];
        ulsch_ue_select[CC_id].ue_num++;
2404 2405 2406 2407 2408
      }
    }
  }

  for ( UE_id = 0; UE_id < NUMBER_OF_UE_MAX; UE_id++ ) {
2409
    if (UE_info->active[UE_id] == FALSE)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2410
      continue;
2411 2412

    rnti = UE_RNTI(module_idP,UE_id);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2413

2414
    if (rnti ==NOT_A_RNTI)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2415
      continue;
2416 2417 2418 2419

    CC_id = UE_PCCID(module_idP,UE_id);

    if (UE_id > last_ulsch_ue_id[CC_id])
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2420
      continue;
2421

2422
    if (UE_info->UE_template[CC_id][UE_id].configured == FALSE)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2423
      continue;
2424

2425
    if (UE_info->UE_sched_ctrl[UE_id].ul_out_of_sync == 1)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2426
      continue;
2427 2428

    if ( (ulsch_ue_select[CC_id].ue_num >= ulsch_ue_max_num[CC_id]) || (cc_id_flag[CC_id] == 1) ) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2429 2430 2431 2432 2433 2434 2435 2436 2437 2438 2439
      cc_id_flag[CC_id] = 1;
      HI_DCI0_req->number_of_dci = saved_ulsch_dci[CC_id];
      ret = cc_id_end(cc_id_flag);

      if ( ret == 0 ) {
        continue;
      }

      if ( ret == 1 ) {
        return;
      }
2440 2441
    }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2442 2443 2444
    for(i = 0; i<ulsch_ue_select[CC_id].ue_num; i++) {
      if(ulsch_ue_select[CC_id].list[i].UE_id == UE_id) {
        break;
2445 2446
      }
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2447

2448 2449 2450 2451 2452
    if(i < ulsch_ue_select[CC_id].ue_num)
      continue;

    HI_DCI0_req   = &eNB->HI_DCI0_req[CC_id][subframeP].hi_dci0_request_body;
    //SR BSR
2453 2454
    UE_sched_ctl = &UE_info->UE_sched_ctrl[UE_id];
    int bytes_to_schedule = UE_info->UE_template[CC_id][UE_id].estimated_ul_buffer - UE_info->UE_template[CC_id][UE_id].scheduled_ul_bytes;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2455

2456
    if (bytes_to_schedule < 0) bytes_to_schedule = 0;
2457
    rrc_status = mac_eNB_get_rrc_status(module_idP, rnti);
2458

2459
    if ( (bytes_to_schedule > 0) || (UE_info->UE_template[CC_id][UE_id].ul_SR > 0) ||
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2460
         ((UE_sched_ctl->ul_inactivity_timer>20)&&(UE_sched_ctl->ul_scheduled==0))  ||
2461 2462
         ((UE_sched_ctl->ul_inactivity_timer>10)&&(UE_sched_ctl->ul_scheduled==0)&&(rrc_status < RRC_CONNECTED)) ||
         ((UE_sched_ctl->cqi_req_timer>300)&&((rrc_status >= RRC_CONNECTED))) ) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2463 2464
      hi_dci0_pdu   = &HI_DCI0_req->hi_dci0_pdu_list[HI_DCI0_req->number_of_dci+HI_DCI0_req->number_of_hi];
      format_flag = 2;
2465
      aggregation=get_aggregation(get_bw_index(module_idP,CC_id),UE_info->UE_sched_ctrl[UE_id].dl_cqi[CC_id],format0);
2466

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2467 2468 2469 2470 2471 2472 2473 2474 2475 2476 2477 2478
      if (CCE_allocation_infeasible(module_idP,CC_id,format_flag,subframeP,aggregation,rnti) == 1) {
        cc_id_flag[CC_id] = 1;
        continue;
      } else {
        hi_dci0_pdu->pdu_type                               = NFAPI_HI_DCI0_DCI_PDU_TYPE;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.rnti              = rnti;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.aggregation_level = aggregation;
        HI_DCI0_req->number_of_dci++;
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].ue_priority = SCH_UL_FIRST;

        if(bytes_to_schedule > 0)
          ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].ul_total_buffer = bytes_to_schedule;
2479
        else if(UE_info->UE_template[CC_id][UE_id].ul_SR > 0)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2480 2481 2482 2483 2484 2485
          ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].ul_total_buffer = 0;

        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].UE_id = UE_id;
        ulsch_ue_select[CC_id].ue_num++;
        continue;
      }
2486
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2487

2488
    //inactivity UE
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2489
    /*    if ( (ulsch_ue_select[CC_id].ue_num+ul_inactivity_num[CC_id]) < ulsch_ue_max_num[CC_id] ) {
2490
            UE_sched_ctl = &UE_info->UE_sched_ctrl[UE_id];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2491 2492 2493 2494 2495 2496 2497 2498 2499 2500 2501 2502 2503
            uint8_t ul_period = 0;
            if (cc->tdd_Config) {
              ul_period = 50;
            } else {
              ul_period = 20;
            }
            if ( ((UE_sched_ctl->ul_inactivity_timer>ul_period)&&(UE_sched_ctl->ul_scheduled==0))  ||
                ((UE_sched_ctl->ul_inactivity_timer>10)&&(UE_sched_ctl->ul_scheduled==0)&&(mac_eNB_get_rrc_status(module_idP,UE_RNTI(module_idP,UE_id)) < RRC_CONNECTED))) {
              ul_inactivity_id[CC_id][ul_inactivity_num[CC_id]]= UE_id;
              ul_inactivity_num[CC_id]++;
              continue;
            }
        }*/
2504 2505 2506 2507
  }

  for ( CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++ ) {
    HI_DCI0_req   = &eNB->HI_DCI0_req[CC_id][subframeP].hi_dci0_request_body;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2508

2509 2510 2511 2512 2513 2514 2515 2516 2517 2518
    for ( int temp = 0; temp < ul_inactivity_num[CC_id]; temp++ ) {
      if ( (ulsch_ue_select[CC_id].ue_num >= ulsch_ue_max_num[CC_id]) || (cc_id_flag[CC_id] == 1) ) {
        HI_DCI0_req   = &eNB->HI_DCI0_req[CC_id][subframeP].hi_dci0_request_body;
        cc_id_flag[CC_id] = 1;
        break;
      }

      hi_dci0_pdu   = &HI_DCI0_req->hi_dci0_pdu_list[HI_DCI0_req->number_of_dci+HI_DCI0_req->number_of_hi];
      format_flag = 2;
      rnti = UE_RNTI(module_idP,ul_inactivity_id[CC_id][temp]);
2519
      aggregation=get_aggregation(get_bw_index(module_idP,CC_id),UE_info->UE_sched_ctrl[ul_inactivity_id[CC_id][temp]].dl_cqi[CC_id],format0);
2520

2521
      if (CCE_allocation_infeasible(module_idP,CC_id,format_flag,subframeP,aggregation,rnti) == 1) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2522 2523
        cc_id_flag[CC_id] = 1;
        continue;
2524
      } else {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2525 2526 2527 2528 2529 2530 2531 2532
        hi_dci0_pdu->pdu_type                               = NFAPI_HI_DCI0_DCI_PDU_TYPE;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.rnti              = rnti;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.aggregation_level = aggregation;
        HI_DCI0_req->number_of_dci++;
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].ue_priority = SCH_UL_INACTIVE;
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].ul_total_buffer = 0;
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].UE_id = ul_inactivity_id[CC_id][temp];
        ulsch_ue_select[CC_id].ue_num++;
2533 2534
      }
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2535

2536 2537
    HI_DCI0_req->number_of_dci = saved_ulsch_dci[CC_id];
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2538

2539 2540 2541
  return;
}

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2542
uint8_t find_rb_table_index(uint8_t average_rbs) {
2543
  int i;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2544

2545 2546 2547 2548 2549
  for ( i = 0; i < 34; i++ ) {
    if ( rb_table[i] > average_rbs ) {
      return (i-1);
    }
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2550

2551 2552 2553 2554
  return i;
}

void ulsch_scheduler_pre_processor_fairRR(module_id_t module_idP,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2555 2556 2557 2558
    frame_t frameP,
    sub_frame_t subframeP,
    sub_frame_t sched_subframeP,
    ULSCH_UE_SELECT ulsch_ue_select[MAX_NUM_CCs]) {
2559 2560
  int                CC_id,ulsch_ue_num;
  eNB_MAC_INST       *eNB = RC.mac[module_idP];
2561
  UE_info_t          *UE_info= &eNB->UE_info;
2562 2563 2564 2565 2566 2567 2568 2569 2570 2571 2572 2573 2574 2575 2576 2577 2578 2579 2580 2581 2582
  UE_TEMPLATE        *UE_template = NULL;
  LTE_DL_FRAME_PARMS *frame_parms = NULL;
  uint8_t            ue_num_temp;
  uint8_t            total_rbs=0;
  uint8_t            average_rbs;
  uint16_t           first_rb[MAX_NUM_CCs];
  uint8_t            mcs;
  uint8_t            rb_table_index;
  uint8_t            num_pucch_rb;
  uint32_t           tbs;
  int16_t            tx_power;
  int                UE_id;
  COMMON_channels_t *cc;
  LOG_D(MAC,"In ulsch_preprocessor: ulsch ue select\n");
  //ue select
  ulsch_scheduler_pre_ue_select_fairRR(module_idP,frameP,subframeP,sched_subframeP,ulsch_ue_select);

  // MCS and RB assgin
  for ( CC_id = 0; CC_id < MAX_NUM_CCs; CC_id++ ) {
    cc = &RC.mac[module_idP]->common_channels[CC_id];
    frame_parms = &(RC.eNB[module_idP][CC_id]->frame_parms);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2583

2584 2585
    if (cc->tdd_Config) { //TDD
      if (frame_parms->N_RB_UL == 25) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2586
        num_pucch_rb = 1;
2587
      } else if (frame_parms->N_RB_UL == 50) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2588
        num_pucch_rb = 2;
2589
      } else {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2590
        num_pucch_rb = 3;
2591 2592 2593
      }
    } else {//FDD
      if (frame_parms->N_RB_UL == 25) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2594
        num_pucch_rb = 1;
2595
      } else {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2596
        num_pucch_rb = 2;
2597 2598 2599 2600 2601 2602
      }
    }

    first_rb[CC_id] = num_pucch_rb;
    ue_num_temp       = ulsch_ue_select[CC_id].ue_num;

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2603
    for ( ulsch_ue_num = 0; ulsch_ue_num < ulsch_ue_select[CC_id].ue_num; ulsch_ue_num++ ) {
2604 2605 2606 2607 2608 2609 2610 2611 2612 2613 2614 2615 2616 2617 2618
      UE_id = ulsch_ue_select[CC_id].list[ulsch_ue_num].UE_id;

      if (ulsch_ue_select[CC_id].list[ulsch_ue_num].ue_priority == SCH_UL_MSG3) {
        first_rb[CC_id] ++;
        ue_num_temp--;
        continue;
      }

      if (ulsch_ue_select[CC_id].list[ulsch_ue_num].ue_priority == SCH_UL_PRACH) {
        first_rb[CC_id] = ulsch_ue_select[CC_id].list[ulsch_ue_num].start_rb+ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb;
        ue_num_temp--;
        continue;
      }

      if (first_rb[CC_id] >= frame_parms->N_RB_UL-num_pucch_rb ) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2619 2620 2621
        LOG_W(MAC,"[eNB %d] frame %d subframe %d, UE %d/%x CC %d: dropping, not enough RBs\n",
              module_idP,frameP,subframeP,UE_id,UE_RNTI(CC_id,UE_id),CC_id);
        break;
2622 2623
      }

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2624
      total_rbs = frame_parms->N_RB_UL-num_pucch_rb-first_rb[CC_id];
2625
      average_rbs = (int)round((double)total_rbs/(double)ue_num_temp);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2626

2627 2628 2629 2630 2631 2632
      if ( average_rbs < 3 ) {
        ue_num_temp--;
        ulsch_ue_num--;
        ulsch_ue_select[CC_id].ue_num--;
        continue;
      }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2633

2634 2635
      if ( ulsch_ue_select[CC_id].list[ulsch_ue_num].ue_priority == SCH_UL_RETRANS ) {
        if ( ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb <= average_rbs ) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2636 2637 2638
          // assigne RBS(nb_rb)
          ulsch_ue_select[CC_id].list[ulsch_ue_num].start_rb = first_rb[CC_id];
          first_rb[CC_id] = first_rb[CC_id] + ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb;
2639
        }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2640

2641 2642
        if ( ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb > average_rbs ) {
          if ( ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb <= total_rbs ) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2643 2644 2645
            // assigne RBS(average_rbs)
            ulsch_ue_select[CC_id].list[ulsch_ue_num].start_rb = first_rb[CC_id];
            first_rb[CC_id] = first_rb[CC_id] + ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb;
2646
          } else {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2647 2648 2649 2650 2651 2652 2653 2654 2655 2656
            // assigne RBS(remain rbs)
            ulsch_ue_select[CC_id].list[ulsch_ue_num].start_rb = first_rb[CC_id];
            rb_table_index = 2;

            while(rb_table[rb_table_index] <= total_rbs) {
              rb_table_index++;
            }

            ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb = rb_table[rb_table_index-1];
            first_rb[CC_id] = first_rb[CC_id] + rb_table[rb_table_index-1];
2657 2658
          }
        }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2659
      } else {
2660
        UE_template = &UE_info->UE_template[CC_id][UE_id];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2661

2662
        if ( UE_info->UE_sched_ctrl[UE_id].phr_received == 1 ) {
2663 2664 2665 2666
          mcs = 20;
        } else {
          mcs = 10;
        }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2667

2668
        if ( ulsch_ue_select[CC_id].list[ulsch_ue_num].ue_priority  == SCH_UL_FIRST ) {
2669
          int bytes_to_schedule = UE_template->estimated_ul_buffer - UE_template->scheduled_ul_bytes;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2670

2671 2672
          if (bytes_to_schedule < 0) bytes_to_schedule = 0;

2673 2674 2675 2676 2677
          if ( ulsch_ue_select[CC_id].list[ulsch_ue_num].ul_total_buffer > 0 ) {
            rb_table_index = 2;
            tbs = get_TBS_UL(mcs,rb_table[rb_table_index]);
            tx_power= estimate_ue_tx_power(tbs*8,rb_table[rb_table_index],0,frame_parms->Ncp,0);

2678
            while ( (((UE_template->phr_info - tx_power) < 0 ) || (tbs > bytes_to_schedule)) && (mcs > 3) ) {
2679 2680 2681 2682 2683
              mcs--;
              tbs = get_TBS_UL(mcs,rb_table[rb_table_index]);
              tx_power= estimate_ue_tx_power(tbs*8,rb_table[rb_table_index],0,frame_parms->Ncp,0);
            }

2684
            while ( (tbs < bytes_to_schedule) && (rb_table[rb_table_index]<(frame_parms->N_RB_UL-num_pucch_rb-first_rb[CC_id])) &&
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2685
                    ((UE_template->phr_info - tx_power) > 0) && (rb_table_index < 32 )) {
2686
              rb_table_index++;
2687 2688
              tbs = get_TBS_UL(mcs,rb_table[rb_table_index]);
              tx_power= estimate_ue_tx_power(tbs*8,rb_table[rb_table_index],0,frame_parms->Ncp,0);
2689 2690 2691 2692 2693
            }

            if ( rb_table[rb_table_index]<3 ) {
              rb_table_index=2;
            }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2694

2695 2696 2697
            if ( rb_table[rb_table_index] <= average_rbs ) {
              // assigne RBS( nb_rb)
              first_rb[CC_id] = first_rb[CC_id] + rb_table[rb_table_index];
2698 2699 2700
              UE_info->UE_template[CC_id][UE_id].pre_allocated_nb_rb_ul[0] = rb_table[rb_table_index];
              UE_info->UE_template[CC_id][UE_id].pre_allocated_rb_table_index_ul = rb_table_index;
              UE_info->UE_template[CC_id][UE_id].pre_assigned_mcs_ul = mcs;
2701
            }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2702

2703 2704 2705
            if ( rb_table[rb_table_index] > average_rbs ) {
              // assigne RBS(average_rbs)
              rb_table_index = find_rb_table_index(average_rbs);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2706 2707 2708 2709 2710

              if (rb_table_index>=34) {
                LOG_W(MAC,"[eNB %d] frame %d subframe %d, UE %d/%x CC %d: average RBs %d > 100\n",
                      module_idP,frameP,subframeP,UE_id,UE_RNTI(CC_id,UE_id),CC_id,average_rbs);
                break;
2711
              }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2712

2713
              first_rb[CC_id] = first_rb[CC_id] + rb_table[rb_table_index];
2714 2715 2716
              UE_info->UE_template[CC_id][UE_id].pre_allocated_nb_rb_ul[0] = rb_table[rb_table_index];
              UE_info->UE_template[CC_id][UE_id].pre_allocated_rb_table_index_ul = rb_table_index;
              UE_info->UE_template[CC_id][UE_id].pre_assigned_mcs_ul = mcs;
2717
            }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2718
          } else {
2719
            if (mac_eNB_get_rrc_status(module_idP,UE_RNTI(module_idP, UE_id)) < RRC_CONNECTED) {
2720 2721
              // assigne RBS( 6 RBs)
              first_rb[CC_id] = first_rb[CC_id] + 6;
2722 2723 2724
              UE_info->UE_template[CC_id][UE_id].pre_allocated_nb_rb_ul[0] = 6;
              UE_info->UE_template[CC_id][UE_id].pre_allocated_rb_table_index_ul = 5;
              UE_info->UE_template[CC_id][UE_id].pre_assigned_mcs_ul = 10;
2725
            } else {
Haruki NAOI's avatar
Haruki NAOI committed
2726 2727
              // assigne RBS( 3 RBs)
              first_rb[CC_id] = first_rb[CC_id] + 3;
2728 2729 2730
              UE_info->UE_template[CC_id][UE_id].pre_allocated_nb_rb_ul[0] = 3;
              UE_info->UE_template[CC_id][UE_id].pre_allocated_rb_table_index_ul = 2;
              UE_info->UE_template[CC_id][UE_id].pre_assigned_mcs_ul = 10;
Haruki NAOI's avatar
Haruki NAOI committed
2731
            }
2732
          }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2733
        } else if ( ulsch_ue_select[CC_id].list[ulsch_ue_num].ue_priority  == SCH_UL_INACTIVE ) {
2734 2735
          // assigne RBS( 3 RBs)
          first_rb[CC_id] = first_rb[CC_id] + 3;
2736 2737 2738
          UE_info->UE_template[CC_id][UE_id].pre_allocated_nb_rb_ul[0] = 3;
          UE_info->UE_template[CC_id][UE_id].pre_allocated_rb_table_index_ul = 2;
          UE_info->UE_template[CC_id][UE_id].pre_assigned_mcs_ul = 10;
2739 2740
        }
      }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2741

2742 2743 2744 2745 2746 2747 2748 2749
      ue_num_temp--;
    }
  }
}


void
schedule_ulsch_fairRR(module_id_t module_idP, frame_t frameP,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2750
                      sub_frame_t subframeP) {
2751 2752 2753 2754 2755 2756 2757 2758 2759
  uint16_t i;
  int CC_id;
  eNB_MAC_INST *mac = RC.mac[module_idP];
  COMMON_channels_t *cc;
  int sched_frame=frameP;
  start_meas(&mac->schedule_ulsch);
  int sched_subframe = (subframeP+4)%10;
  cc = &mac->common_channels[0];
  int tdd_sfa;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2760

2761 2762 2763
  // for TDD: check subframes where we have to act and return if nothing should be done now
  if (cc->tdd_Config) {
    tdd_sfa = cc->tdd_Config->subframeAssignment;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2764

2765
    switch (subframeP) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2766 2767 2768 2769 2770 2771 2772 2773 2774 2775 2776 2777 2778 2779 2780 2781 2782 2783 2784 2785 2786 2787 2788 2789 2790 2791 2792 2793 2794 2795 2796 2797 2798 2799 2800 2801 2802 2803 2804 2805 2806 2807 2808 2809 2810 2811 2812 2813 2814 2815 2816 2817 2818 2819 2820 2821 2822 2823 2824 2825
      case 0:
        if ((tdd_sfa == 0)||
            (tdd_sfa == 3)) sched_subframe = 4;
        else if (tdd_sfa==6) sched_subframe = 7;
        else return;

        break;

      case 1:
        if ((tdd_sfa==0)||
            (tdd_sfa==1)) sched_subframe = 7;
        else if (tdd_sfa==6) sched_subframe = 8;
        else return;

        break;

      case 2: // Don't schedule UL in subframe 2 for TDD
        return;

      case 3:
        if (tdd_sfa==2) sched_subframe = 7;
        else return;

        break;

      case 4:
        if (tdd_sfa==1) sched_subframe = 8;
        else return;

        break;

      case 5:
        if (tdd_sfa==0)      sched_subframe = 9;
        else if (tdd_sfa==6) sched_subframe = 2;
        else return;

        break;

      case 6:
        if (tdd_sfa==0 || tdd_sfa==1)      sched_subframe = 2;
        else if (tdd_sfa==6) sched_subframe = 3;
        else return;

        break;

      case 7:
        return;

      case 8:
        if ((tdd_sfa>=2) && (tdd_sfa<=5)) sched_subframe=2;
        else return;

        break;

      case 9:
        if ((tdd_sfa==1) || (tdd_sfa==3) || (tdd_sfa==4)) sched_subframe=3;
        else if (tdd_sfa==6) sched_subframe=4;
        else return;

        break;
2826 2827
    }
  }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2828

2829
  if (sched_subframe < subframeP) sched_frame++;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2830

2831 2832 2833 2834 2835 2836 2837
  ULSCH_UE_SELECT ulsch_ue_select[MAX_NUM_CCs];
  memset(ulsch_ue_select, 0, sizeof(ulsch_ue_select));
  LTE_DL_FRAME_PARMS *frame_parms ;

  for (CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
    cc = &mac->common_channels[CC_id];
    frame_parms= &RC.eNB[module_idP][CC_id]->frame_parms;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2838

2839 2840 2841 2842 2843
    // output of scheduling, the UE numbers in RBs, where it is in the code???
    // check if RA (Msg3) is active in this subframeP, if so skip the PRBs used for Msg3
    // Msg3 is using 1 PRB so we need to increase first_rb accordingly
    // not sure about the break (can there be more than 1 active RA procedure?)
    for (i=0; i<NB_RA_PROC_MAX; i++) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2844
      if ((cc->ra[i].state == WAITMSG3) &&(cc->ra[i].Msg3_subframe == sched_subframe)) {
2845
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].ue_priority = SCH_UL_MSG3;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2846

2847
        if (cc->tdd_Config == NULL) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2848 2849 2850 2851 2852
          if(frame_parms->N_RB_UL == 25) {
            ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].start_rb = 1;
          } else {
            ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].start_rb = 2;
          }
2853
        } else {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2854
          switch(frame_parms->N_RB_UL) {
2855 2856 2857 2858
            case 25:
            default:
              ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].start_rb = 1;
              break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2859

2860 2861 2862
            case 50:
              ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].start_rb = 2;
              break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2863

2864 2865 2866
            case 100:
              ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].start_rb = 3;
              break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2867
          }
2868
        }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2869

2870 2871 2872 2873 2874 2875 2876 2877 2878 2879 2880
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].nb_rb = 1;
        ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].UE_id = -1;
        ulsch_ue_select[CC_id].ue_num++;
        break;
      }
    }

    //PRACH
    if (is_prach_subframe(frame_parms,sched_frame,sched_subframe)==1) {
      ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].ue_priority = SCH_UL_PRACH;
      ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].start_rb = get_prach_prb_offset(
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2881 2882 2883 2884
            frame_parms,
            frame_parms->prach_config_common.prach_ConfigInfo.prach_ConfigIndex,
            frame_parms->prach_config_common.prach_ConfigInfo.prach_FreqOffset,
            0,//tdd_mapindex
2885
            frameP); //Nf --> shouldn't it be sched_frame ???
2886 2887 2888 2889 2890 2891 2892 2893 2894 2895 2896
      ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].nb_rb = 6;
      ulsch_ue_select[CC_id].list[ulsch_ue_select[CC_id].ue_num].UE_id = -1;
      ulsch_ue_select[CC_id].ue_num++;
    }
  }

  schedule_ulsch_rnti_fairRR(module_idP, frameP, subframeP, sched_subframe,ulsch_ue_select);
  stop_meas(&mac->schedule_ulsch);
}

void schedule_ulsch_rnti_fairRR(module_id_t   module_idP,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2897 2898 2899 2900
                                frame_t       frameP,
                                sub_frame_t   subframeP,
                                unsigned char sched_subframeP,
                                ULSCH_UE_SELECT  ulsch_ue_select[MAX_NUM_CCs]) {
2901
  int16_t           UE_id;
2902
  uint8_t           aggregation;
2903 2904 2905 2906 2907 2908
  uint16_t          first_rb[MAX_NUM_CCs];
  uint8_t           ULSCH_first_end;
  rnti_t            rnti           = -1;
  uint8_t           round          = 0;
  uint8_t           harq_pid       = 0;
  uint8_t           status         = 0;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2909
  uint8_t           rb_table_index = -1;
2910
  uint32_t          cqi_req,cshift,ndi,tpc;
Cedric Roux's avatar
Cedric Roux committed
2911 2912
  int32_t           snr;
  int32_t           target_snr=0;
2913 2914 2915 2916 2917
  static int32_t    tpc_accumulated=0;
  int               CC_id,ulsch_ue_num;
  int               N_RB_UL;
  eNB_MAC_INST      *eNB = RC.mac[module_idP];
  COMMON_channels_t *cc;
2918
  UE_info_t         *UE_info=&eNB->UE_info;
2919
  UE_TEMPLATE       *UE_template;
2920
  UE_sched_ctrl_t     *UE_sched_ctrl;
2921 2922 2923 2924
  int               sched_frame=frameP;
  int               rvidx_tab[4] = {0,2,3,1};
  uint16_t          ul_req_index;
  uint8_t           dlsch_flag;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2925

2926 2927 2928 2929 2930 2931 2932 2933
  if (sched_subframeP < subframeP) sched_frame++;

  nfapi_hi_dci0_request_body_t   *hi_dci0_req;
  nfapi_hi_dci0_request_pdu_t    *hi_dci0_pdu;
  nfapi_ul_config_request_body_t *ul_req_tmp;
  nfapi_ul_config_ulsch_harq_information *ulsch_harq_information;
  LOG_D(MAC,"entering ulsch preprocesor\n");
  ulsch_scheduler_pre_processor_fairRR(module_idP,
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2934 2935 2936 2937
                                       frameP,
                                       subframeP,
                                       sched_subframeP,
                                       ulsch_ue_select);
2938 2939 2940 2941
  LOG_D(MAC,"exiting ulsch preprocesor\n");

  // loop over all active UEs
  for (CC_id=0; CC_id<MAX_NUM_CCs; CC_id++) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2942 2943 2944 2945
    hi_dci0_req = &eNB->HI_DCI0_req[CC_id][subframeP].hi_dci0_request_body;
    eNB->HI_DCI0_req[CC_id][subframeP].sfn_sf = (frameP<<4)+subframeP;
    ul_req_tmp = &eNB->UL_req_tmp[CC_id][sched_subframeP].ul_config_request_body;
    nfapi_ul_config_request_t *ul_req  = &eNB->UL_req_tmp[CC_id][sched_subframeP];
2946 2947 2948 2949
    ULSCH_first_end = 0;
    cc  = &eNB->common_channels[CC_id];
    // This is the actual CC_id in the list
    N_RB_UL      = to_prb(cc->mib->message.dl_Bandwidth);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2950

2951 2952
    //leave out first RB for PUCCH
    if (cc->tdd_Config == NULL) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2953 2954 2955 2956 2957 2958 2959
      if(N_RB_UL == 25) {
        first_rb[CC_id] = 1;
      } else {
        first_rb[CC_id] = 2;
      }
    } else {
      switch(N_RB_UL) {
2960 2961
        case 25:
        default:
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2962
          first_rb[CC_id] = 1;
2963
          break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2964

2965
        case 50:
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2966
          first_rb[CC_id] = 2;
2967
          break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2968

2969
        case 100:
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2970
          first_rb[CC_id] = 3;
2971
          break;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2972
      }
2973
    }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2974

2975 2976
    for ( ulsch_ue_num = 0; ulsch_ue_num < ulsch_ue_select[CC_id].ue_num; ulsch_ue_num++ ) {
      UE_id = ulsch_ue_select[CC_id].list[ulsch_ue_num].UE_id;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2977

2978
      /* be sure that there are some free RBs */
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2979 2980 2981 2982 2983 2984 2985 2986 2987 2988 2989 2990
      if (cc->tdd_Config == NULL) {
        if(N_RB_UL == 25) {
          if (first_rb[CC_id] >= N_RB_UL-1) {
            LOG_W(MAC,"[eNB %d] frame %d subframe %d, UE %d/%x CC %d: dropping, not enough RBs\n",
                  module_idP,frameP,subframeP,UE_id,rnti,CC_id);
            break;
          }
        } else {
          if (first_rb[CC_id] >= N_RB_UL-2) {
            LOG_W(MAC,"[eNB %d] frame %d subframe %d, UE %d/%x CC %d: dropping, not enough RBs\n",
                  module_idP,frameP,subframeP,UE_id,rnti,CC_id);
            break;
2991
          }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2992
        }
2993
      } else {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
2994 2995 2996 2997 2998
        if(N_RB_UL == 25) {
          if (first_rb[CC_id] >= N_RB_UL-1) {
            LOG_W(MAC,"[eNB %d] frame %d subframe %d, UE %d/%x CC %d N_RB_UL %d first_rb %d: dropping, not enough RBs\n",
                  module_idP,frameP,subframeP,UE_id,rnti,CC_id, N_RB_UL, first_rb[CC_id]);
            break;
2999
          }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3000 3001 3002 3003 3004 3005 3006 3007 3008 3009 3010 3011 3012
        } else if(N_RB_UL == 50) {
          if (first_rb[CC_id] >= N_RB_UL-2) {
            LOG_W(MAC,"[eNB %d] frame %d subframe %d, UE %d/%x CC %d N_RB_UL %d first_rb %d: dropping, not enough RBs\n",
                  module_idP,frameP,subframeP,UE_id,rnti,CC_id, N_RB_UL, first_rb[CC_id]);
            break;
          }
        } else if(N_RB_UL == 100) {
          if (first_rb[CC_id] >= N_RB_UL-3) {
            LOG_W(MAC,"[eNB %d] frame %d subframe %d, UE %d/%x CC %d N_RB_UL %d first_rb %d: dropping, not enough RBs\n",
                  module_idP,frameP,subframeP,UE_id,rnti,CC_id, N_RB_UL, first_rb[CC_id]);
            break;
          }
        }
3013
      }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3014

3015 3016 3017 3018 3019
      //MSG3
      if (ulsch_ue_select[CC_id].list[ulsch_ue_num].ue_priority == SCH_UL_MSG3) {
        first_rb[CC_id] ++;
        continue;
      }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3020

3021 3022 3023 3024 3025 3026
      //PRACH
      if (ulsch_ue_select[CC_id].list[ulsch_ue_num].ue_priority == SCH_UL_PRACH) {
        first_rb[CC_id] = ulsch_ue_select[CC_id].list[ulsch_ue_num].start_rb+ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb;
        continue;
      }

3027 3028
      UE_template   = &UE_info->UE_template[CC_id][UE_id];
      UE_sched_ctrl = &UE_info->UE_sched_ctrl[UE_id];
3029 3030
      harq_pid      = subframe2harqpid(cc,sched_frame,sched_subframeP);
      rnti = UE_RNTI(CC_id,UE_id);
3031
      aggregation=get_aggregation(get_bw_index(module_idP,CC_id),UE_sched_ctrl[UE_id].dl_cqi[CC_id],format0);
3032 3033
      LOG_D(MAC,"[eNB %d] frame %d subframe %d,Checking PUSCH %d for UE %d/%x CC %d : aggregation level %d, N_RB_UL %d\n",
            module_idP,frameP,subframeP,harq_pid,UE_id,rnti,CC_id, aggregation,N_RB_UL);
3034
      int bytes_to_schedule = UE_template->estimated_ul_buffer - UE_template->scheduled_ul_bytes;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3035

3036 3037 3038
      if (bytes_to_schedule < 0) bytes_to_schedule = 0;

      RC.eNB[module_idP][CC_id]->pusch_stats_BO[UE_id][(frameP*10)+subframeP] = bytes_to_schedule;
3039 3040
      VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_UE0_BO,RC.eNB[module_idP][CC_id]->pusch_stats_BO[UE_id][(frameP*10)+subframeP]);
      status = mac_eNB_get_rrc_status(module_idP,rnti);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3041

3042 3043
      if (status < RRC_CONNECTED)
        cqi_req = 0;
3044 3045 3046 3047 3048 3049 3050
      else if (UE_sched_ctrl->cqi_received == 1){
        LOG_D(MAC,"Clearing CQI request timer\n");
        UE_sched_ctrl->cqi_req_flag = 0;
        UE_sched_ctrl->cqi_received = 0;
        UE_sched_ctrl->cqi_req_timer = 0;
        cqi_req = 0;
      }else if (UE_sched_ctrl->cqi_req_timer>30) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3051 3052 3053 3054 3055 3056
        cqi_req = 1;

        // To be safe , do not ask CQI in special SFs:36.213/7.2.3 CQI definition
        if (cc->tdd_Config) {
          switch (cc->tdd_Config->subframeAssignment) {
            case 1:
3057
              if( subframeP == 1 || subframeP == 6 ) cqi_req=0;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3058 3059 3060 3061

              break;

            case 3:
3062
              if( subframeP == 1 ) cqi_req=0;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3063 3064 3065 3066 3067 3068 3069 3070 3071 3072 3073 3074 3075

              break;

            default:
              LOG_E(MAC," TDD config not supported\n");
              break;
          }
        }

        if(cqi_req == 1) {
          UE_sched_ctrl->cqi_req_flag |= 1 << sched_subframeP;
        }
      } else
3076 3077 3078 3079 3080
        cqi_req = 0;

      //power control
      //compute the expected ULSCH RX power (for the stats)
      // this is the normalized RX power and this should be constant (regardless of mcs
Cedric Roux's avatar
Cedric Roux committed
3081 3082
      snr = (5 * UE_sched_ctrl->pusch_snr[CC_id] - 640) / 10;
      target_snr = eNB->puSch10xSnr / 10;
3083 3084 3085
      // this assumes accumulated tpc
      // make sure that we are only sending a tpc update once a frame, otherwise the control loop will freak out
      int32_t framex10psubframe = UE_template->pusch_tpc_tx_frame*10+UE_template->pusch_tpc_tx_subframe;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3086

3087
      if (((framex10psubframe+10)<=(frameP*10+subframeP)) || //normal case
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3088 3089 3090 3091
          ((framex10psubframe>(frameP*10+subframeP)) && (((10240-framex10psubframe+frameP*10+subframeP)>=10)))) { //frame wrap-around
        UE_template->pusch_tpc_tx_frame=frameP;
        UE_template->pusch_tpc_tx_subframe=subframeP;

Cedric Roux's avatar
Cedric Roux committed
3092
        if (snr > target_snr + 4) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3093 3094
          tpc = 0; //-1
          tpc_accumulated--;
Cedric Roux's avatar
Cedric Roux committed
3095
        } else if (snr < target_snr - 4) {
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3096 3097
          tpc = 2; //+1
          tpc_accumulated++;
3098 3099 3100
        } else {
          tpc = 1; //0
        }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3101 3102 3103 3104 3105
      } else {
        tpc = 1; //0
      }

      if (tpc!=1) {
laurent's avatar
laurent committed
3106
        LOG_D(MAC,"[eNB %d] ULSCH schedulerRR: frame %d, subframe %d, harq_pid %d, tpc %d, accumulated %d, snr/target snr %d/%d\n",
3107
              module_idP,frameP,subframeP,harq_pid,tpc,
Cedric Roux's avatar
Cedric Roux committed
3108
              tpc_accumulated,snr,target_snr);
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3109 3110 3111 3112 3113 3114 3115 3116 3117 3118 3119 3120
      }

      // new transmission
      if ((ulsch_ue_select[CC_id].list[ulsch_ue_num].ue_priority == SCH_UL_FIRST) ||
          (ulsch_ue_select[CC_id].list[ulsch_ue_num].ue_priority == SCH_UL_INACTIVE)) {
        LOG_D(MAC,"[eNB %d][PUSCH %d] Frame %d subframe %d Scheduling UE %d/%x (SR %d,UL_inactivity timer %d,UL_failure timer %d,cqi_req_timer %d)\n",
              module_idP,harq_pid,frameP,subframeP,UE_id,rnti,UE_template->ul_SR,
              UE_sched_ctrl->ul_inactivity_timer,
              UE_sched_ctrl->ul_failure_timer,
              UE_sched_ctrl->cqi_req_timer);
        ndi = 1-UE_template->oldNDI_UL[harq_pid];
        UE_template->oldNDI_UL[harq_pid]=ndi;
3121 3122 3123
        UE_info->eNB_UE_stats[CC_id][UE_id].snr = snr;
        UE_info->eNB_UE_stats[CC_id][UE_id].target_snr = target_snr;
        UE_info->eNB_UE_stats[CC_id][UE_id].ulsch_mcs1=UE_template->pre_assigned_mcs_ul;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3124 3125 3126 3127 3128 3129 3130
        UE_template->mcs_UL[harq_pid] = UE_template->pre_assigned_mcs_ul;//cmin (UE_template->pre_assigned_mcs_ul, openair_daq_vars.target_ue_ul_mcs); // adjust, based on user-defined MCS

        if (UE_template->pre_allocated_rb_table_index_ul >=0) {
          rb_table_index=UE_template->pre_allocated_rb_table_index_ul;
        } else {
          UE_template->mcs_UL[harq_pid]=10;//cmin (10, openair_daq_vars.target_ue_ul_mcs);
          rb_table_index=5; // for PHR
3131
        }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3132

3133
        UE_info->eNB_UE_stats[CC_id][UE_id].ulsch_mcs2=UE_template->mcs_UL[harq_pid];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3134
        UE_template->TBS_UL[harq_pid] = get_TBS_UL(UE_template->mcs_UL[harq_pid],rb_table[rb_table_index]);
3135 3136
        UE_info->eNB_UE_stats[CC_id][UE_id].total_rbs_used_rx+=rb_table[rb_table_index];
        UE_info->eNB_UE_stats[CC_id][UE_id].ulsch_TBS=UE_template->TBS_UL[harq_pid];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3137 3138 3139 3140 3141 3142 3143 3144 3145 3146 3147 3148 3149 3150
        T(T_ENB_MAC_UE_UL_SCHEDULE, T_INT(module_idP), T_INT(CC_id), T_INT(rnti), T_INT(frameP),
          T_INT(subframeP), T_INT(harq_pid), T_INT(UE_template->mcs_UL[harq_pid]), T_INT(first_rb[CC_id]), T_INT(rb_table[rb_table_index]),
          T_INT(UE_template->TBS_UL[harq_pid]), T_INT(ndi));

        if (mac_eNB_get_rrc_status(module_idP,rnti) < RRC_CONNECTED)
          LOG_D(MAC,"[eNB %d][PUSCH %d/%x] CC_id %d Frame %d subframeP %d Scheduled UE %d (mcs %d, first rb %d, nb_rb %d, rb_table_index %d, TBS %d, harq_pid %d)\n",
                module_idP,harq_pid,rnti,CC_id,frameP,subframeP,UE_id,UE_template->mcs_UL[harq_pid],
                first_rb[CC_id],rb_table[rb_table_index],
                rb_table_index,UE_template->TBS_UL[harq_pid],harq_pid);

        // bad indices : 20 (40 PRB), 21 (45 PRB), 22 (48 PRB)
        //store for possible retransmission
        UE_template->nb_rb_ul[harq_pid]    = rb_table[rb_table_index];
        UE_template->first_rb_ul[harq_pid] = first_rb[CC_id];
3151
        UE_template->cqi_req[harq_pid] = cqi_req;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3152 3153
        UE_sched_ctrl->ul_scheduled |= (1<<harq_pid);

3154
        if (UE_id == UE_info->list.head)
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3155 3156 3157 3158 3159 3160 3161 3162 3163 3164 3165 3166 3167 3168 3169 3170 3171
          VCD_SIGNAL_DUMPER_DUMP_VARIABLE_BY_NAME(VCD_SIGNAL_DUMPER_VARIABLES_UE0_SCHEDULED,UE_sched_ctrl->ul_scheduled);

        // adjust total UL buffer status by TBS, wait for UL sdus to do final update
        /*LOG_D(MAC,"[eNB %d] CC_id %d UE %d/%x : adjusting ul_total_buffer, old %d, TBS %d\n", module_idP,CC_id,UE_id,rnti,UE_template->ul_total_buffer,UE_template->TBS_UL[harq_pid]);
        if (UE_template->ul_total_buffer > UE_template->TBS_UL[harq_pid])
          UE_template->ul_total_buffer -= UE_template->TBS_UL[harq_pid];
        else
          UE_template->ul_total_buffer = 0;
        LOG_D(MAC,"ul_total_buffer, new %d\n", UE_template->ul_total_buffer);*/
        // Cyclic shift for DM RS
        cshift = 0;// values from 0 to 7 can be used for mapping the cyclic shift (36.211 , Table 5.5.2.1.1-1)
        // save it for a potential retransmission
        UE_template->cshift[harq_pid] = cshift;
        hi_dci0_pdu                                                         = &hi_dci0_req->hi_dci0_pdu_list[hi_dci0_req->number_of_dci+hi_dci0_req->number_of_hi];
        memset((void *)hi_dci0_pdu,0,sizeof(nfapi_hi_dci0_request_pdu_t));
        hi_dci0_pdu->pdu_type                                               = NFAPI_HI_DCI0_DCI_PDU_TYPE;
        hi_dci0_pdu->pdu_size                                               = 2+sizeof(nfapi_hi_dci0_dci_pdu);
3172
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.tl.tag                            = NFAPI_HI_DCI0_REQUEST_DCI_PDU_REL8_TAG;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3173 3174 3175 3176 3177 3178 3179 3180 3181 3182 3183 3184 3185 3186 3187 3188 3189 3190 3191 3192 3193 3194 3195 3196 3197 3198 3199 3200 3201 3202 3203
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.dci_format                        = NFAPI_UL_DCI_FORMAT_0;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.aggregation_level                 = aggregation;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.rnti                              = rnti;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.transmission_power                = 6000;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.resource_block_start              = first_rb[CC_id];
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.number_of_resource_block          = rb_table[rb_table_index];
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.mcs_1                             = UE_template->mcs_UL[harq_pid];
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.cyclic_shift_2_for_drms           = cshift;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.frequency_hopping_enabled_flag    = 0;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.new_data_indication_1             = ndi;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.tpc                               = tpc;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.cqi_csi_request                   = cqi_req;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.dl_assignment_index               = UE_template->DAI_ul[sched_subframeP];
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.harq_pid                          = harq_pid;
        hi_dci0_req->number_of_dci++;
        hi_dci0_req->sfnsf = sfnsf_add_subframe(sched_frame, sched_subframeP, 0); //(frameP, subframeP, 4)
        hi_dci0_req->tl.tag = NFAPI_HI_DCI0_REQUEST_BODY_TAG;
        nfapi_hi_dci0_request_t        *nfapi_hi_dci0_req = &eNB->HI_DCI0_req[CC_id][subframeP];
        nfapi_hi_dci0_req->sfn_sf = frameP<<4|subframeP; // sfnsf_add_subframe(sched_frame, sched_subframeP, 0); // sunday!
        nfapi_hi_dci0_req->header.message_id = NFAPI_HI_DCI0_REQUEST;
        LOG_D(MAC,"[PUSCH %d] Frame %d, Subframe %d: Adding UL CONFIG.Request for UE %d/%x, ulsch_frame %d, ulsch_subframe %d\n",
              harq_pid,frameP,subframeP,UE_id,rnti,sched_frame,sched_subframeP);
        ul_req_index = 0;
        dlsch_flag = 0;

        for(ul_req_index = 0; ul_req_index < ul_req_tmp->number_of_pdus; ul_req_index++) {
          if((ul_req_tmp->ul_config_pdu_list[ul_req_index].pdu_type == NFAPI_UL_CONFIG_UCI_HARQ_PDU_TYPE) &&
              (ul_req_tmp->ul_config_pdu_list[ul_req_index].uci_harq_pdu.ue_information.ue_information_rel8.rnti == rnti)) {
            dlsch_flag = 1;
            LOG_D(MAC,"Frame %d, Subframe %d:rnti %x ul_req_index %d Switched UCI HARQ to ULSCH HARQ(first)\n",frameP,subframeP,rnti,ul_req_index);
            break;
3204
          }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3205 3206 3207 3208 3209 3210 3211 3212 3213 3214 3215 3216 3217 3218 3219 3220 3221 3222 3223 3224 3225 3226 3227 3228 3229 3230 3231 3232 3233 3234 3235 3236 3237 3238 3239 3240 3241 3242 3243 3244 3245 3246 3247 3248 3249 3250 3251 3252 3253
        }

        // Add UL_config PDUs
        fill_nfapi_ulsch_config_request_rel8(&ul_req_tmp->ul_config_pdu_list[ul_req_index],
                                             cqi_req,
                                             cc,
                                             UE_template->physicalConfigDedicated,
                                             get_tmode(module_idP,CC_id,UE_id),
                                             eNB->ul_handle,
                                             rnti,
                                             first_rb[CC_id], // resource_block_start
                                             rb_table[rb_table_index], // number_of_resource_blocks
                                             UE_template->mcs_UL[harq_pid],
                                             cshift, // cyclic_shift_2_for_drms
                                             0, // frequency_hopping_enabled_flag
                                             0, // frequency_hopping_bits
                                             ndi, // new_data_indication
                                             0, // redundancy_version
                                             harq_pid, // harq_process_number
                                             0, // ul_tx_mode
                                             0, // current_tx_nb
                                             0, // n_srs
                                             get_TBS_UL(UE_template->mcs_UL[harq_pid],
                                                 rb_table[rb_table_index])
                                            );

        if (UE_template->rach_resource_type>0) { // This is a BL/CE UE allocation
          fill_nfapi_ulsch_config_request_emtc(&ul_req_tmp->ul_config_pdu_list[ul_req_index],
                                               UE_template->rach_resource_type>2 ? 2 : 1,
                                               1, //total_number_of_repetitions
                                               1, //repetition_number
                                               (frameP*10)+subframeP);
        }

        if(dlsch_flag == 1) {
          if(cqi_req == 1) {
            ul_req_tmp->ul_config_pdu_list[ul_req_index].pdu_type = NFAPI_UL_CONFIG_ULSCH_CQI_HARQ_RI_PDU_TYPE;
            ulsch_harq_information = &ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_cqi_harq_ri_pdu.harq_information;
            ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_cqi_harq_ri_pdu.initial_transmission_parameters.initial_transmission_parameters_rel8.tl.tag=
              NFAPI_UL_CONFIG_REQUEST_INITIAL_TRANSMISSION_PARAMETERS_REL8_TAG;
            ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_cqi_harq_ri_pdu.initial_transmission_parameters.initial_transmission_parameters_rel8.n_srs_initial = 0;    // last symbol not punctured
            ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_cqi_harq_ri_pdu.initial_transmission_parameters.initial_transmission_parameters_rel8.initial_number_of_resource_blocks = rb_table[rb_table_index];
          } else {
            ul_req_tmp->ul_config_pdu_list[ul_req_index].pdu_type = NFAPI_UL_CONFIG_ULSCH_HARQ_PDU_TYPE;
            ulsch_harq_information = &ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_harq_pdu.harq_information;
            ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_harq_pdu.initial_transmission_parameters.initial_transmission_parameters_rel8.tl.tag =
              NFAPI_UL_CONFIG_REQUEST_INITIAL_TRANSMISSION_PARAMETERS_REL8_TAG;
            ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_harq_pdu.initial_transmission_parameters.initial_transmission_parameters_rel8.n_srs_initial = 0;  // last symbol not punctured
            ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_harq_pdu.initial_transmission_parameters.initial_transmission_parameters_rel8.initial_number_of_resource_blocks = rb_table[rb_table_index];
3254
          }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3255 3256 3257 3258 3259 3260 3261 3262 3263 3264 3265 3266 3267 3268 3269 3270 3271 3272 3273 3274 3275 3276 3277 3278 3279 3280 3281 3282 3283 3284 3285 3286 3287 3288 3289 3290 3291

          fill_nfapi_ulsch_harq_information(module_idP, CC_id,rnti, ulsch_harq_information,subframeP);
        } else {
          ul_req_tmp->number_of_pdus++;
        }

        eNB->ul_handle++;
        ul_req->header.message_id = NFAPI_UL_CONFIG_REQUEST;
        ul_req_tmp->tl.tag = NFAPI_UL_CONFIG_REQUEST_BODY_TAG;
        uint16_t ul_sched_frame = sched_frame;
        uint16_t ul_sched_subframeP = sched_subframeP;
        add_subframe(&ul_sched_frame, &ul_sched_subframeP, 2);
        ul_req->sfn_sf = ul_sched_frame<<4|ul_sched_subframeP;
        add_ue_ulsch_info(module_idP,
                          CC_id,
                          UE_id,
                          subframeP,
                          S_UL_SCHEDULED);
        LOG_D(MAC,"[eNB %d] CC_id %d Frame %d, subframeP %d: Generated ULSCH DCI for next UE_id %d, format 0\n", module_idP,CC_id,frameP,subframeP,UE_id);
        // increment first rb for next UE allocation
        first_rb[CC_id]+=rb_table[rb_table_index];

        if(ulsch_ue_select[CC_id].list[ulsch_ue_num].ue_priority == SCH_UL_FIRST) {
          UE_template->scheduled_ul_bytes += get_TBS_UL(UE_template->mcs_UL[harq_pid],rb_table[rb_table_index]);
          UE_template->ul_SR = 0;
        }

        if((ulsch_ue_select[CC_id].list[ulsch_ue_num].ue_priority == SCH_UL_INACTIVE) && (ULSCH_first_end == 0)) {
          ULSCH_first_end = 1;
          last_ulsch_ue_id[CC_id] = ulsch_ue_select[CC_id].list[ulsch_ue_num-1].UE_id;
        }

        if((ulsch_ue_num == ulsch_ue_select[CC_id].ue_num-1) && (ULSCH_first_end == 0)) {
          ULSCH_first_end = 1;
          last_ulsch_ue_id[CC_id] = ulsch_ue_select[CC_id].list[ulsch_ue_num].UE_id;
        }
      } else if (ulsch_ue_select[CC_id].list[ulsch_ue_num].ue_priority == SCH_UL_RETRANS) { // round > 0 => retransmission
3292
        round = UE_sched_ctrl->round_UL[CC_id][harq_pid];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3293
        T(T_ENB_MAC_UE_UL_SCHEDULE_RETRANSMISSION, T_INT(module_idP), T_INT(CC_id), T_INT(rnti), T_INT(frameP),
3294
          T_INT(subframeP), T_INT(harq_pid), T_INT(UE_template->mcs_UL[harq_pid]), T_INT(ulsch_ue_select[CC_id].list[ulsch_ue_num].start_rb), T_INT(ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb),
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3295
          T_INT(round));
3296 3297
        UE_info->eNB_UE_stats[CC_id][UE_id].snr = snr;
        UE_info->eNB_UE_stats[CC_id][UE_id].target_snr = target_snr;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3298 3299 3300 3301 3302 3303 3304 3305 3306 3307 3308
        uint8_t mcs_rv = 0;

        if(rvidx_tab[round&3]==1) {
          mcs_rv = 29;
        } else if(rvidx_tab[round&3]==2) {
          mcs_rv = 30;
        } else if(rvidx_tab[round&3]==3) {
          mcs_rv = 31;
        }

        UE_template->TBS_UL[harq_pid] = get_TBS_UL(UE_template->mcs_UL[harq_pid],ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb);
3309
        UE_info->eNB_UE_stats[CC_id][UE_id].ulsch_TBS=UE_template->TBS_UL[harq_pid];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3310 3311

        if (mac_eNB_get_rrc_status(module_idP,rnti) < RRC_CONNECTED)
3312 3313 3314
          LOG_D(MAC,"[eNB %d][PUSCH %d/%x] CC_id %d Frame %d subframeP %d Scheduled UE %d (mcs %d, first rb %d, nb_rb %d, TBS %d, harq_pid %d)\n",
                module_idP,harq_pid,rnti,CC_id,frameP,subframeP,UE_id,mcs_rv,first_rb[CC_id],ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb,UE_template->TBS_UL[harq_pid],harq_pid);

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3315 3316 3317 3318
        // bad indices : 20 (40 PRB), 21 (45 PRB), 22 (48 PRB)
        //store for possible retransmission
        UE_template->nb_rb_ul[harq_pid]    = ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb;
        UE_template->first_rb_ul[harq_pid] = ulsch_ue_select[CC_id].list[ulsch_ue_num].start_rb;
3319
        cqi_req = UE_template->cqi_req[harq_pid];
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3320 3321 3322 3323 3324 3325 3326
        UE_sched_ctrl->ul_scheduled |= (1<<harq_pid);
        // Cyclic shift for DM RS
        cshift = 0;// values from 0 to 7 can be used for mapping the cyclic shift (36.211 , Table 5.5.2.1.1-1)
        hi_dci0_pdu                                                         = &hi_dci0_req->hi_dci0_pdu_list[hi_dci0_req->number_of_dci+hi_dci0_req->number_of_hi];
        memset((void *)hi_dci0_pdu,0,sizeof(nfapi_hi_dci0_request_pdu_t));
        hi_dci0_pdu->pdu_type                                               = NFAPI_HI_DCI0_DCI_PDU_TYPE;
        hi_dci0_pdu->pdu_size                                               = 2+sizeof(nfapi_hi_dci0_dci_pdu);
3327
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.tl.tag                            = NFAPI_HI_DCI0_REQUEST_DCI_PDU_REL8_TAG;
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3328 3329 3330 3331 3332 3333 3334 3335 3336 3337 3338 3339 3340 3341 3342 3343 3344 3345 3346 3347 3348 3349 3350 3351 3352 3353 3354 3355 3356 3357 3358 3359 3360 3361 3362 3363 3364 3365 3366 3367 3368 3369 3370 3371 3372 3373 3374 3375 3376 3377 3378
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.dci_format                        = NFAPI_UL_DCI_FORMAT_0;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.aggregation_level                 = aggregation;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.rnti                              = rnti;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.transmission_power                = 6000;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.resource_block_start              = ulsch_ue_select[CC_id].list[ulsch_ue_num].start_rb;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.number_of_resource_block          = ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.mcs_1                             = mcs_rv;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.cyclic_shift_2_for_drms           = cshift;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.frequency_hopping_enabled_flag    = 0;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.new_data_indication_1             = UE_template->oldNDI_UL[harq_pid];
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.tpc                               = tpc;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.cqi_csi_request                   = cqi_req;
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.dl_assignment_index               = UE_template->DAI_ul[sched_subframeP];
        hi_dci0_pdu->dci_pdu.dci_pdu_rel8.harq_pid                          = harq_pid;
        hi_dci0_req->number_of_dci++;
        // Add UL_config PDUs
        LOG_D(MAC,"[PUSCH %d] Frame %d, Subframe %d: Adding UL CONFIG.Request for UE %d/%x, ulsch_frame %d, ulsch_subframe %d\n",
              harq_pid,frameP,subframeP,UE_id,rnti,sched_frame,sched_subframeP);
        ul_req_index = 0;
        dlsch_flag = 0;

        for(ul_req_index = 0; ul_req_index < ul_req_tmp->number_of_pdus; ul_req_index++) {
          if((ul_req_tmp->ul_config_pdu_list[ul_req_index].pdu_type == NFAPI_UL_CONFIG_UCI_HARQ_PDU_TYPE) &&
              (ul_req_tmp->ul_config_pdu_list[ul_req_index].uci_harq_pdu.ue_information.ue_information_rel8.rnti == rnti)) {
            dlsch_flag = 1;
            LOG_D(MAC,"Frame %d, Subframe %d:rnti %x ul_req_index %d Switched UCI HARQ to ULSCH HARQ(phich)\n",frameP,subframeP,rnti,ul_req_index);
            break;
          }
        }

        fill_nfapi_ulsch_config_request_rel8(&ul_req_tmp->ul_config_pdu_list[ul_req_index],
                                             cqi_req,
                                             cc,
                                             UE_template->physicalConfigDedicated,
                                             get_tmode(module_idP,CC_id,UE_id),
                                             eNB->ul_handle,
                                             rnti,
                                             ulsch_ue_select[CC_id].list[ulsch_ue_num].start_rb, // resource_block_start
                                             ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb, // number_of_resource_blocks
                                             UE_template->mcs_UL[harq_pid],
                                             cshift, // cyclic_shift_2_for_drms
                                             0, // frequency_hopping_enabled_flag
                                             0, // frequency_hopping_bits
                                             UE_template->oldNDI_UL[harq_pid], // new_data_indication
                                             rvidx_tab[round&3], // redundancy_version
                                             harq_pid, // harq_process_number
                                             0, // ul_tx_mode
                                             0, // current_tx_nb
                                             0, // n_srs
                                             UE_template->TBS_UL[harq_pid]
                                            );
3379

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3380 3381 3382 3383 3384 3385 3386
        if (UE_template->rach_resource_type>0) { // This is a BL/CE UE allocation
          fill_nfapi_ulsch_config_request_emtc(&ul_req_tmp->ul_config_pdu_list[ul_req_index],
                                               UE_template->rach_resource_type>2 ? 2 : 1,
                                               1, //total_number_of_repetitions
                                               1, //repetition_number
                                               (frameP*10)+subframeP);
        }
3387

Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3388 3389 3390 3391 3392 3393 3394 3395 3396 3397 3398 3399 3400 3401 3402 3403 3404
        if(dlsch_flag == 1) {
          if(cqi_req == 1) {
            ul_req_tmp->ul_config_pdu_list[ul_req_index].pdu_type = NFAPI_UL_CONFIG_ULSCH_CQI_HARQ_RI_PDU_TYPE;
            ulsch_harq_information = &ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_cqi_harq_ri_pdu.harq_information;
            ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_cqi_harq_ri_pdu.initial_transmission_parameters.initial_transmission_parameters_rel8.tl.tag=
              NFAPI_UL_CONFIG_REQUEST_INITIAL_TRANSMISSION_PARAMETERS_REL8_TAG;
            ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_cqi_harq_ri_pdu.initial_transmission_parameters.initial_transmission_parameters_rel8.n_srs_initial = 0;    // last symbol not punctured
            ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_cqi_harq_ri_pdu.initial_transmission_parameters.initial_transmission_parameters_rel8.initial_number_of_resource_blocks =
              ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb;
          } else {
            ul_req_tmp->ul_config_pdu_list[ul_req_index].pdu_type = NFAPI_UL_CONFIG_ULSCH_HARQ_PDU_TYPE;
            ulsch_harq_information = &ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_harq_pdu.harq_information;
            ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_harq_pdu.initial_transmission_parameters.initial_transmission_parameters_rel8.tl.tag =
              NFAPI_UL_CONFIG_REQUEST_INITIAL_TRANSMISSION_PARAMETERS_REL8_TAG;
            ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_harq_pdu.initial_transmission_parameters.initial_transmission_parameters_rel8.n_srs_initial = 0;  // last symbol not punctured
            ul_req_tmp->ul_config_pdu_list[ul_req_index].ulsch_harq_pdu.initial_transmission_parameters.initial_transmission_parameters_rel8.initial_number_of_resource_blocks =
              ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb;
3405
          }
Louis Adrien Dufrene's avatar
Louis Adrien Dufrene committed
3406 3407 3408 3409 3410 3411 3412 3413 3414 3415 3416 3417 3418 3419

          fill_nfapi_ulsch_harq_information(module_idP, CC_id,rnti, ulsch_harq_information,subframeP);
        } else {
          ul_req_tmp->number_of_pdus++;
        }

        eNB->ul_handle++;
        ul_req->header.message_id = NFAPI_UL_CONFIG_REQUEST;
        ul_req_tmp->tl.tag = NFAPI_UL_CONFIG_REQUEST_BODY_TAG;
        ul_req->sfn_sf = sched_frame<<4|sched_subframeP;
        LOG_D(MAC,"[eNB %d] CC_id %d Frame %d, subframeP %d: Generated ULSCH DCI for next UE_id %d, format 0(round >0)\n", module_idP,CC_id,frameP,subframeP,UE_id);
        // increment first rb for next UE allocation
        first_rb[CC_id]+=ulsch_ue_select[CC_id].list[ulsch_ue_num].nb_rb;
      }
3420 3421 3422
    } // loop over UE_id
  } // loop of CC_id
}