eNB_scheduler_mch.c 28.7 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
/*
 * 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.0  (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
 */
21 22

/*! \file eNB_scheduler_mch.c
nikaeinn's avatar
nikaeinn committed
23
 * \brief procedures related to eNB for the MCH transport channel
24 25 26
 * \author  Navid Nikaein and Raymond Knopp
 * \date 2012 - 2014
 * \email: navid.nikaein@eurecom.fr
nikaeinn's avatar
nikaeinn committed
27
 * \version 1.0
28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
 * @ingroup _mac

 */

#include "assertions.h"
#include "PHY/defs.h"
#include "PHY/extern.h"

#include "SCHED/defs.h"
#include "SCHED/extern.h"

#include "LAYER2/MAC/defs.h"
#include "LAYER2/MAC/proto.h"
#include "LAYER2/MAC/extern.h"
#include "UTIL/LOG/log.h"
#include "UTIL/LOG/vcd_signal_dumper.h"
#include "UTIL/OPT/opt.h"
#include "OCG.h"
#include "OCG_extern.h"

#include "RRC/LITE/extern.h"
#include "RRC/L2_INTERFACE/openair_rrc_L2_interface.h"

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

#if defined(ENABLE_ITTI)
# include "intertask_interface.h"
#endif

58 59
#include "SIMULATION/TOOLS/defs.h" // for taus

60 61 62 63 64
#define ENABLE_MAC_PAYLOAD_DEBUG
#define DEBUG_eNB_SCHEDULER 1


#ifdef Rel10
65 66
int8_t get_mbsfn_sf_alloction (module_id_t module_idP, uint8_t CC_id, uint8_t mbsfn_sync_area)
{
67
  // currently there is one-to-one mapping between sf allocation pattern and sync area
68
  if (mbsfn_sync_area > MAX_MBSFN_AREA) {
69
    LOG_W(MAC,"[eNB %d] CC_id %d MBSFN synchronization area %d out of range\n ", module_idP, CC_id, mbsfn_sync_area);
70
    return -1;
71
  } else if (eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[mbsfn_sync_area] != NULL) {
72
    return mbsfn_sync_area;
73
  } else {
74
    LOG_W(MAC,"[eNB %d] CC_id %d MBSFN Subframe Config pattern %d not found \n ", module_idP, CC_id, mbsfn_sync_area);
75
    return -1;
76 77 78
  }
}

79 80
int schedule_MBMS(module_id_t module_idP, uint8_t CC_id, frame_t frameP, sub_frame_t subframeP)
{
81 82 83 84

  int mcch_flag=0,mtch_flag=0, msi_flag=0;
  int mbsfn_period =0;// 1<<(eNB_mac_inst[module_idP].mbsfn_SubframeConfig[0]->radioframeAllocationPeriod);
  int mcch_period = 0;//32<<(eNB_mac_inst[module_idP].mbsfn_AreaInfo[0]->mcch_Config_r9.mcch_RepetitionPeriod_r9);
85
  int mch_scheduling_period = 8<<(eNB_mac_inst[module_idP].common_channels[CC_id].pmch_Config[0]->mch_SchedulingPeriod_r9);
86 87 88 89 90 91 92 93 94 95 96 97
  unsigned char mcch_sdu_length;
  unsigned char header_len_mcch=0,header_len_msi=0,header_len_mtch=0, header_len_mtch_temp=0, header_len_mcch_temp=0, header_len_msi_temp=0;
  int ii=0, msi_pos=0;
  int mcch_mcs = -1;
  uint16_t TBS,j,padding=0,post_padding=0;
  mac_rlc_status_resp_t rlc_status;
  int num_mtch;
  int msi_length,i,k;
  unsigned char sdu_lcids[11], num_sdus=0, offset=0;
  uint16_t sdu_lengths[11], sdu_length_total=0;
  unsigned char mch_buffer[MAX_DLSCH_PAYLOAD_BYTES]; // check the max value, this is for dlsch only

98
  eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.Pdu_size=0;
99 100

  for (i=0;
101 102 103
       i< eNB_mac_inst[module_idP].common_channels[CC_id].num_active_mbsfn_area;
       i++ ) {
    // assume, that there is always a mapping
104
    if ((j=get_mbsfn_sf_alloction(module_idP,CC_id,i)) == -1) {
105
      return 0;
106
    }
107

108 109 110 111
    mbsfn_period = 1<<(eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->radioframeAllocationPeriod);
    mcch_period = 32<<(eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.mcch_RepetitionPeriod_r9);
    msi_pos=0;
    ii=0;
112 113
    LOG_D(MAC,"[eNB %d] CC_id %d Frame %d subframeP %d : Checking MBSFN Sync Area %d/%d with SF allocation %d/%d for MCCH and MTCH (mbsfn period %d, mcch period %d)\n",
          module_idP, CC_id, frameP, subframeP,i,eNB_mac_inst[module_idP].common_channels[CC_id].num_active_mbsfn_area,
114
          j,eNB_mac_inst[module_idP].common_channels[CC_id].num_sf_allocation_pattern,mbsfn_period,mcch_period);
115 116


117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145
    switch (eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.signallingMCS_r9) {
    case 0:
      mcch_mcs = 2;
      break;

    case 1:
      mcch_mcs = 7;
      break;

    case 2:
      mcch_mcs = 13;
      break;

    case 3:
      mcch_mcs = 19;
      break;
    }

    // 1st: Check the MBSFN subframes from SIB2 info (SF allocation pattern i, max 8 non-overlapping patterns exist)
    if (frameP %  mbsfn_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->radioframeAllocationOffset) { // MBSFN frameP
      if (eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.present == MBSFN_SubframeConfig__subframeAllocation_PR_oneFrame) { // one-frameP format

        //  Find the first subframeP in this MCH to transmit MSI
        if (frameP % mch_scheduling_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->radioframeAllocationOffset ) {
          while (ii == 0) {
            ii = eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0] & (0x80>>msi_pos);
            msi_pos++;
          }

146 147
          LOG_D(MAC,"[eNB %d] CC_id %d Frame %d subframeP %d : sync area %d sf allocation pattern %d sf alloc %x msi pos is %d \n",
                module_idP, CC_id, frameP, subframeP,i,j,
148 149 150 151 152 153
                eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0], msi_pos);
        }

        // Check if the subframeP is for MSI, MCCH or MTCHs and Set the correspoding flag to 1
        switch (subframeP) {
        case 1:
154
          if (mac_xface->frame_parms->frame_type == FDD) {
155
            if ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0] & MBSFN_FDD_SF1) == MBSFN_FDD_SF1) {
156
              if (msi_pos == 1) {
157
                msi_flag = 1;
158
              }
159 160

              if ( (frameP % mcch_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.mcch_Offset_r9) &&
161
                   ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.sf_AllocInfo_r9.buf[0] & MBSFN_FDD_SF1) == MBSFN_FDD_SF1) ) {
162
                mcch_flag = 1;
163
              }
164 165 166 167 168 169 170 171

              mtch_flag = 1;
            }
          }

          break;

        case 2:
172
          if (mac_xface->frame_parms->frame_type == FDD) {
173
            if ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0] & MBSFN_FDD_SF2) == MBSFN_FDD_SF2) {
174
              if (msi_pos == 2) {
175
                msi_flag = 1;
176
              }
177 178

              if ( (frameP % mcch_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.mcch_Offset_r9) &&
179
                   ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.sf_AllocInfo_r9.buf[0] & MBSFN_FDD_SF2) == MBSFN_FDD_SF2) ) {
180
                mcch_flag = 1;
181
              }
182

183 184
              mtch_flag = 1;
            }
185
          }
186 187 188 189

          break;

        case 3:
190
          if (mac_xface->frame_parms->frame_type == TDD) { // TDD
191
            if ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0] & MBSFN_TDD_SF3) == MBSFN_TDD_SF3) {
192
              if (msi_pos == 1) {
193
                msi_flag = 1;
194
              }
195 196

              if ( (frameP % mcch_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.mcch_Offset_r9) &&
197
                   ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.sf_AllocInfo_r9.buf[0] & MBSFN_TDD_SF3) == MBSFN_TDD_SF3) ) {
198
                mcch_flag = 1;
199
              }
200 201 202 203 204

              mtch_flag = 1;
            }
          } else { // FDD
            if ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0] & MBSFN_FDD_SF3) == MBSFN_FDD_SF3) {
205
              if (msi_pos == 3) {
206
                msi_flag = 1;
207
              }
208 209

              if ( (frameP % mcch_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.mcch_Offset_r9) &&
210
                   ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.sf_AllocInfo_r9.buf[0] & MBSFN_FDD_SF3) == MBSFN_FDD_SF3) ) {
211
                mcch_flag = 1;
212
              }
213 214 215 216 217 218 219 220

              mtch_flag = 1;
            }
          }

          break;

        case 4:
221
          if (mac_xface->frame_parms->frame_type == TDD) {
222
            if ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0] & MBSFN_TDD_SF4) == MBSFN_TDD_SF4) {
223
              if (msi_pos == 2) {
224
                msi_flag = 1;
225
              }
226 227

              if ( (frameP % mcch_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.mcch_Offset_r9) &&
228
                   ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.sf_AllocInfo_r9.buf[0] & MBSFN_TDD_SF4) == MBSFN_TDD_SF4) ) {
229
                mcch_flag = 1;
230
              }
231 232 233 234 235 236 237 238

              mtch_flag = 1;
            }
          }

          break;

        case 6:
239
          if (mac_xface->frame_parms->frame_type == FDD) {
240
            if ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0] & MBSFN_FDD_SF6) == MBSFN_FDD_SF6) {
241
              if (msi_pos == 4) {
242
                msi_flag = 1;
243
              }
244 245

              if ( (frameP % mcch_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.mcch_Offset_r9) &&
246
                   ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.sf_AllocInfo_r9.buf[0] & MBSFN_FDD_SF6) == MBSFN_FDD_SF6) ) {
247
                mcch_flag = 1;
248
              }
249 250 251 252 253 254 255 256

              mtch_flag = 1;
            }
          }

          break;

        case 7:
257
          if (mac_xface->frame_parms->frame_type == TDD) { // TDD
258
            if ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0] & MBSFN_TDD_SF7) == MBSFN_TDD_SF7) {
259
              if (msi_pos == 3) {
260
                msi_flag = 1;
261
              }
262 263

              if ( (frameP % mcch_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.mcch_Offset_r9) &&
264
                   ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.sf_AllocInfo_r9.buf[0] & MBSFN_TDD_SF7) == MBSFN_TDD_SF7) ) {
265
                mcch_flag = 1;
266
              }
267 268 269 270 271

              mtch_flag = 1;
            }
          } else { // FDD
            if ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0] & MBSFN_FDD_SF7) == MBSFN_FDD_SF7) {
272
              if (msi_pos == 5) {
273
                msi_flag = 1;
274
              }
275 276

              if ( (frameP % mcch_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.mcch_Offset_r9) &&
277
                   ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.sf_AllocInfo_r9.buf[0] & MBSFN_FDD_SF7) == MBSFN_FDD_SF7) ) {
278
                mcch_flag = 1;
279
              }
280 281 282 283 284 285 286 287

              mtch_flag = 1;
            }
          }

          break;

        case 8:
288
          if (mac_xface->frame_parms->frame_type == TDD) { //TDD
289
            if ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0] & MBSFN_TDD_SF8) == MBSFN_TDD_SF8) {
290
              if (msi_pos == 4) {
291
                msi_flag = 1;
292
              }
293 294

              if ( (frameP % mcch_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.mcch_Offset_r9) &&
295
                   ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.sf_AllocInfo_r9.buf[0] & MBSFN_TDD_SF8) == MBSFN_TDD_SF8) ) {
296
                mcch_flag = 1;
297
              }
298 299 300 301 302

              mtch_flag = 1;
            }
          } else { // FDD
            if ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0] & MBSFN_FDD_SF8) == MBSFN_FDD_SF8) {
303
              if (msi_pos == 6) {
304
                msi_flag = 1;
305
              }
306 307

              if ( (frameP % mcch_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.mcch_Offset_r9) &&
308
                   ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.sf_AllocInfo_r9.buf[0] & MBSFN_FDD_SF8) == MBSFN_FDD_SF8) ) {
309
                mcch_flag = 1;
310
              }
311 312 313 314 315 316 317 318

              mtch_flag = 1;
            }
          }

          break;

        case 9:
319
          if (mac_xface->frame_parms->frame_type == TDD) {
320
            if ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_SubframeConfig[j]->subframeAllocation.choice.oneFrame.buf[0] & MBSFN_TDD_SF9) == MBSFN_TDD_SF9) {
321
              if (msi_pos == 5) {
322
                msi_flag = 1;
323
              }
324 325

              if ( (frameP % mcch_period == eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.mcch_Offset_r9) &&
326
                   ((eNB_mac_inst[module_idP].common_channels[CC_id].mbsfn_AreaInfo[i]->mcch_Config_r9.sf_AllocInfo_r9.buf[0] & MBSFN_TDD_SF9) == MBSFN_TDD_SF9) ) {
327
                mcch_flag = 1;
328
              }
329 330 331

              mtch_flag = 1;
            }
332
          }
333 334 335 336 337 338

          break;
        }// end switch

        // sf allocation is non-overlapping
        if ((msi_flag==1) || (mcch_flag==1) || (mtch_flag==1)) {
339 340
          LOG_D(MAC,"[eNB %d] CC_id %d Frame %d Subframe %d: sync area %d SF alloc %d: msi flag %d, mcch flag %d, mtch flag %d\n",
                module_idP, CC_id, frameP, subframeP,i,j,msi_flag,mcch_flag,mtch_flag);
341 342 343
          break;
        }
      } else { // four-frameP format
344
      }
345
    }
346
  } // end of for loop
347

348 349 350
  eNB_mac_inst[module_idP].common_channels[CC_id].msi_active=0;
  eNB_mac_inst[module_idP].common_channels[CC_id].mcch_active=0;
  eNB_mac_inst[module_idP].common_channels[CC_id].mtch_active=0;
351

352 353
  // Calculate the mcs
  if ((msi_flag==1) || (mcch_flag==1)) {
354 355 356
    eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.mcs = mcch_mcs;
  } else if (mtch_flag == 1) { // only MTCH in this subframeP
    eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.mcs = eNB_mac_inst[module_idP].common_channels[CC_id].pmch_Config[0]->dataMCS_r9;
357 358 359 360 361 362 363
  }


  // 2nd: Create MSI, get MCCH from RRC and MTCHs from RLC

  // there is MSI (MCH Scheduling Info)
  if (msi_flag == 1) {
364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391
    // Create MSI here
    uint16_t msi_control_element[29], *msi_ptr;

    msi_ptr = &msi_control_element[0];
    ((MSI_ELEMENT *) msi_ptr)->lcid = MCCH_LCHANID; //MCCH

    if (mcch_flag==1) {
      ((MSI_ELEMENT *) msi_ptr)->stop_sf_MSB = 0;
      ((MSI_ELEMENT *) msi_ptr)->stop_sf_LSB = 0;
    } else {                  // no mcch for this MSP
      ((MSI_ELEMENT *) msi_ptr)->stop_sf_MSB = 0x7;// stop value is 2047
      ((MSI_ELEMENT *) msi_ptr)->stop_sf_LSB = 0xff;
    }

    msi_ptr+= sizeof(MSI_ELEMENT);

    //Header for MTCHs
    num_mtch = eNB_mac_inst[module_idP].common_channels[CC_id].mbms_SessionList[0]->list.count;

    for (k=0; k<num_mtch; k++) { // loop for all session in this MCH (MCH[0]) at this moment
      ((MSI_ELEMENT *) msi_ptr)->lcid = eNB_mac_inst[module_idP].common_channels[CC_id].mbms_SessionList[0]->list.array[k]->logicalChannelIdentity_r9;//mtch_lcid;
      ((MSI_ELEMENT *) msi_ptr)->stop_sf_MSB = 0; // last subframeP of this mtch (only one mtch now)
      ((MSI_ELEMENT *) msi_ptr)->stop_sf_LSB = 0xB;
      msi_ptr+=sizeof(MSI_ELEMENT);
    }

    msi_length = msi_ptr-msi_control_element;

392
    if (msi_length<128) {
393
      header_len_msi = 2;
394
    } else {
395
      header_len_msi = 3;
396
    }
397

398 399
    LOG_D(MAC,"[eNB %d] CC_id %d Frame %d : MSI->MCH, length of MSI is %d bytes \n",
          module_idP,CC_id,frameP,msi_length);
400 401 402 403 404 405 406 407 408 409 410
    //LOG_D(MAC,"Scheduler: MSI is transmitted in this subframeP \n" );

    //   LOG_D(MAC,"Scheduler: MSI length is %d bytes\n",msi_length);
    // Store MSI data to mch_buffer[0]
    memcpy((char *)&mch_buffer[sdu_length_total],
           msi_control_element,
           msi_length);

    sdu_lcids[num_sdus] = MCH_SCHDL_INFO;
    sdu_lengths[num_sdus] = msi_length;
    sdu_length_total += sdu_lengths[num_sdus];
411
    LOG_I(MAC,"[eNB %d] CC_id %d Create %d bytes for MSI\n", module_idP, CC_id, sdu_lengths[num_sdus]);
412 413
    num_sdus++;
    eNB_mac_inst[module_idP].common_channels[CC_id].msi_active=1;
414 415 416 417
  }

  // there is MCCH
  if (mcch_flag == 1) {
418 419
    LOG_D(MAC,"[eNB %d] CC_id %d Frame %d Subframe %d: Schedule MCCH MESSAGE (area %d, sfAlloc %d)\n",
          module_idP, CC_id, frameP, subframeP, i, j);
420

421
    mcch_sdu_length = mac_rrc_data_req(module_idP,
422
                                       CC_id,
423 424 425 426 427 428
                                       frameP,
                                       MCCH,1,
                                       &eNB_mac_inst[module_idP].common_channels[CC_id].MCCH_pdu.payload[0],
                                       1,// this is eNB
                                       module_idP, // index
                                       i); // this is the mbsfn sync area index
429

430
    if (mcch_sdu_length > 0) {
431 432
      LOG_D(MAC,"[eNB %d] CC_id %d Frame %d subframeP %d : MCCH->MCH, Received %d bytes from RRC \n",
            module_idP,CC_id,frameP,subframeP,mcch_sdu_length);
433

434 435
      header_len_mcch = 2;

436
      if (mac_xface->frame_parms->frame_type == TDD) {
437 438
        LOG_D(MAC,"[eNB %d] CC_id %d Frame %d subframeP %d: Scheduling MCCH->MCH (TDD) for MCCH message %d bytes (mcs %d )\n",
              module_idP, CC_id,
439 440 441 442
              frameP,subframeP,
              mcch_sdu_length,
              mcch_mcs);
      } else {
443 444
        LOG_I(MAC,"[eNB %d] CC_id %d Frame %d subframeP %d: Scheduling MCCH->MCH (FDD) for MCCH message %d bytes (mcs %d)\n",
              module_idP, CC_id,
445 446 447
              frameP, subframeP,
              mcch_sdu_length,
              mcch_mcs);
448
      }
449 450 451 452 453 454 455 456 457

      eNB_mac_inst[module_idP].common_channels[CC_id].mcch_active=1;

      memcpy((char *)&mch_buffer[sdu_length_total],
             &eNB_mac_inst[module_idP].common_channels[CC_id].MCCH_pdu.payload[0],
             mcch_sdu_length);
      sdu_lcids[num_sdus] = MCCH_LCHANID;
      sdu_lengths[num_sdus] = mcch_sdu_length;

458
      if (sdu_lengths[num_sdus]>128) {
459
        header_len_mcch = 3;
460
      }
461 462

      sdu_length_total += sdu_lengths[num_sdus];
463
      LOG_D(MAC,"[eNB %d] CC_id %d Got %d bytes for MCCH from RRC \n",module_idP,CC_id,sdu_lengths[num_sdus]);
464 465
      num_sdus++;
    }
466
  }
467

468
  TBS = mac_xface->get_TBS_DL(eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.mcs, mac_xface->frame_parms->N_RB_DL);
469 470 471 472 473
#ifdef Rel10
  // do not let mcch and mtch multiplexing when relaying is active
  // for sync area 1, so not transmit data
  //if ((i == 0) && ((eNB_mac_inst[module_idP].MBMS_flag != multicast_relay) || (eNB_mac_inst[module_idP].mcch_active==0))) {
#endif
474

475 476
  // there is MTCHs, loop if there are more than 1
  if (mtch_flag == 1) {
477 478
    // Calculate TBS
    /* if ((msi_flag==1) || (mcch_flag==1)) {
479
     TBS = mac_xface->get_TBS(mcch_mcs, mac_xface->frame_parms->N_RB_DL);
480 481
     }
     else { // only MTCH in this subframeP
482
     TBS = mac_xface->get_TBS(eNB_mac_inst[module_idP].pmch_Config[0]->dataMCS_r9, mac_xface->frame_parms->N_RB_DL);
483
     }
484 485

    // get MTCH data from RLC (like for DTCH)
486
    LOG_D(MAC,"[eNB %d] CC_id %d Frame %d subframe %d: Schedule MTCH (area %d, sfAlloc %d)\n",Mod_id,CC_id,frame,subframe,i,j);
487 488

    header_len_mtch = 3;
489 490
    LOG_D(MAC,"[eNB %d], CC_id %d, Frame %d, MTCH->MCH, Checking RLC status (rab %d, tbs %d, len %d)\n",
    Mod_id,CC_id,frame,MTCH,TBS,
491
    TBS-header_len_mcch-header_len_msi-sdu_length_total-header_len_mtch);
492 493

    rlc_status = mac_rlc_status_ind(Mod_id,frame,1,RLC_MBMS_YES,MTCH+ (maxDRB + 3) * MAX_MOBILES_PER_RG,
494
          TBS-header_len_mcch-header_len_msi-sdu_length_total-header_len_mtch);
495 496
    printf("frame %d, subframe %d,  rlc_status.bytes_in_buffer is %d\n",frame,subframe, rlc_status.bytes_in_buffer);

497
     */
498

499
    // get MTCH data from RLC (like for DTCH)
500
    LOG_D(MAC,"[eNB %d] CC_id %d Frame %d subframeP %d: Schedule MTCH (area %d, sfAlloc %d)\n",module_idP,CC_id,frameP,subframeP,i,j);
501

502
    header_len_mtch = 3;
503 504
    LOG_D(MAC,"[eNB %d], CC_id %d, Frame %d, MTCH->MCH, Checking RLC status (rab %d, tbs %d, len %d)\n",
          module_idP,CC_id,frameP,MTCH,TBS,
505 506
          TBS-header_len_mcch-header_len_msi-sdu_length_total-header_len_mtch);

507
    rlc_status = mac_rlc_status_ind(module_idP,0,frameP,subframeP,module_idP,ENB_FLAG_YES,MBMS_FLAG_YES,MTCH,
508 509 510 511 512
                                    TBS-header_len_mcch-header_len_msi-sdu_length_total-header_len_mtch);
    LOG_D(MAC,"e-MBMS log channel %u frameP %d, subframeP %d,  rlc_status.bytes_in_buffer is %d\n",
          MTCH,frameP,subframeP, rlc_status.bytes_in_buffer);

    if (rlc_status.bytes_in_buffer >0) {
513 514
      LOG_I(MAC,"[eNB %d][MBMS USER-PLANE], CC_id %d, Frame %d, MTCH->MCH, Requesting %d bytes from RLC (header len mtch %d)\n",
            module_idP,CC_id,frameP,TBS-header_len_mcch-header_len_msi-sdu_length_total-header_len_mtch,header_len_mtch);
515 516 517 518

      sdu_lengths[num_sdus] = mac_rlc_data_req(
                                module_idP,
                                0,
519
				module_idP,
520 521 522 523
                                frameP,
                                ENB_FLAG_YES,
                                MBMS_FLAG_YES,
                                MTCH,
524
								0,	//not used
525 526
                                (char*)&mch_buffer[sdu_length_total]);
      //sdu_lengths[num_sdus] = mac_rlc_data_req(module_idP,frameP, MBMS_FLAG_NO,  MTCH+(MAX_NUM_RB*(NUMBER_OF_UE_MAX+1)), (char*)&mch_buffer[sdu_length_total]);
527
      LOG_I(MAC,"[eNB %d][MBMS USER-PLANE] CC_id %d Got %d bytes for MTCH %d\n",module_idP,CC_id,sdu_lengths[num_sdus],MTCH);
528 529 530
      eNB_mac_inst[module_idP].common_channels[CC_id].mtch_active=1;
      sdu_lcids[num_sdus] = MTCH;
      sdu_length_total += sdu_lengths[num_sdus];
531

532
      if (sdu_lengths[num_sdus] < 128) {
533
        header_len_mtch = 2;
534
      }
535

536 537 538 539
      num_sdus++;
    } else {
      header_len_mtch = 0;
    }
540
  }
541

542 543 544
#ifdef Rel10
  //  }
#endif
545

546 547
  // FINAL STEP: Prepare and multiplexe MSI, MCCH and MTCHs
  if ((sdu_length_total + header_len_msi + header_len_mcch + header_len_mtch) >0) {
548 549 550 551 552 553 554 555 556 557 558 559
    // Adjust the last subheader
    /*                                 if ((msi_flag==1) || (mcch_flag==1)) {
                                       eNB_mac_inst[module_idP].MCH_pdu.mcs = mcch_mcs;
                                        }
                                      else if (mtch_flag == 1) { // only MTCH in this subframeP
                                     eNB_mac_inst[module_idP].MCH_pdu.mcs = eNB_mac_inst[module_idP].pmch_Config[0]->dataMCS_r9;
                                        }
     */
    header_len_mtch_temp = header_len_mtch;
    header_len_mcch_temp = header_len_mcch;
    header_len_msi_temp = header_len_msi;

560
    if (header_len_mtch>0) {
561
      header_len_mtch=1;         // remove Length field in the  subheader for the last PDU
562
    } else if (header_len_mcch>0) {
563
      header_len_mcch=1;
564 565 566
    } else {
      header_len_msi=1;
    }
567 568 569 570 571

    // Calculate the padding
    if ((TBS - header_len_mtch - header_len_mcch - header_len_msi - sdu_length_total) < 0) {
      LOG_E(MAC,"Error in building MAC PDU, TBS %d < PDU %d \n",
            TBS, header_len_mtch + header_len_mcch + header_len_msi + sdu_length_total);
572
      return 0;
573 574 575 576 577 578
    } else if ((TBS - header_len_mtch - header_len_mcch - header_len_msi - sdu_length_total) <= 2) {
      padding = (TBS - header_len_mtch - header_len_mcch - header_len_msi - sdu_length_total);
      post_padding = 0;
    } else { // using post_padding, give back the Length field of subheader  for the last PDU
      padding = 0;

579
      if (header_len_mtch>0) {
580
        header_len_mtch = header_len_mtch_temp;
581
      } else if (header_len_mcch>0) {
582
        header_len_mcch = header_len_mcch_temp;
583 584 585
      } else {
        header_len_msi = header_len_msi_temp;
      }
586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607

      post_padding = TBS - sdu_length_total - header_len_msi - header_len_mcch - header_len_mtch;
    }

    // Generate the MAC Header for MCH
    // here we use the function for DLSCH because DLSCH & MCH have the same Header structure
    offset = generate_dlsch_header((unsigned char*)eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.payload,
                                   num_sdus,
                                   sdu_lengths,
                                   sdu_lcids,
                                   255,    // no drx
                                   0,  // no timing advance
                                   NULL,  // no contention res id
                                   padding,
                                   post_padding);

    eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.Pdu_size=TBS;
    eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.sync_area=i;
    eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.msi_active= eNB_mac_inst[module_idP].common_channels[CC_id].msi_active;
    eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.mcch_active= eNB_mac_inst[module_idP].common_channels[CC_id].mcch_active;
    eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.mtch_active= eNB_mac_inst[module_idP].common_channels[CC_id].mtch_active;
    LOG_D(MAC," MCS for this sf is %d (mcch active %d, mtch active %d)\n", eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.mcs,
608
          eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.mcch_active,eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.mtch_active );
609
    LOG_I(MAC,
610 611
          "[eNB %d][MBMS USER-PLANE ] CC_id %d Generate header : sdu_length_total %d, num_sdus %d, sdu_lengths[0] %d, sdu_lcids[0] %d => payload offset %d,padding %d,post_padding %d (mcs %d, TBS %d), header MTCH %d, header MCCH %d, header MSI %d\n",
          module_idP,CC_id,sdu_length_total,num_sdus,sdu_lengths[0],sdu_lcids[0],offset,padding,post_padding,eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.mcs,TBS,
612
          header_len_mtch, header_len_mcch, header_len_msi);
613 614 615 616
    // copy SDU to mch_pdu after the MAC Header
    memcpy(&eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.payload[offset],mch_buffer,sdu_length_total);

    // filling remainder of MCH with random data if necessery
617
    for (j=0; j<(TBS-sdu_length_total-offset); j++) {
618
      eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.payload[offset+sdu_length_total+j] = (char)(taus()&0xff);
619
    }
620

621
    /* Tracing of PDU is done on UE side */
622
    if (opt_enabled ==1 ) {
623
      trace_pdu(1, (uint8_t *)eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.payload,
624
                TBS, module_idP, 6, 0xffff,  // M_RNTI = 6 in wirehsark
Wilson's avatar
Wilson committed
625
                eNB_mac_inst[module_idP].frame, eNB_mac_inst[module_idP].subframe,0,0);
626 627
      LOG_D(OPT,"[eNB %d][MCH] CC_id %d Frame %d : MAC PDU with size %d\n",
            module_idP, CC_id, frameP, TBS);
628
    }
629

630 631 632 633 634 635 636 637 638 639 640 641 642 643 644
    /*
    for (j=0;j<sdu_length_total;j++)
    printf("%2x.",eNB_mac_inst[module_idP].MCH_pdu.payload[j+offset]);
    printf(" \n");*/
    return 1;
  } else {
    eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.Pdu_size=0;
    eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.sync_area=0;
    eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.msi_active=0;
    eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.mcch_active=0;
    eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu.mtch_active=0;
    // for testing purpose, fill with random data
    //for (j=0;j<(TBS-sdu_length_total-offset);j++)
    //  eNB_mac_inst[module_idP].MCH_pdu.payload[offset+sdu_length_total+j] = (char)(taus()&0xff);
    return 0;
645
  }
646

647 648 649 650 651 652 653 654 655 656
  //this is for testing
  /*
  if (mtch_flag == 1) {
  //  LOG_D(MAC,"DUY: mch_buffer length so far is : %ld\n", &mch_buffer[sdu_length_total]-&mch_buffer[0]);
  return 1;
  }
  else
  return 0;
   */
}
657 658

MCH_PDU *get_mch_sdu(module_id_t module_idP, int CC_id, frame_t frameP, sub_frame_t subframeP)
659
{
660 661
  //  eNB_mac_inst[module_idP].MCH_pdu.mcs=0;
  //LOG_D(MAC," MCH_pdu.mcs is %d\n", eNB_mac_inst[module_idP].MCH_pdu.mcs);
662
//#warning "MCH pdu should take the CC_id index"
663
  return(&eNB_mac_inst[module_idP].common_channels[CC_id].MCH_pdu);
664 665 666
}

#endif