ofdm_mod.c 10.6 KB
Newer Older
1 2 3 4 5
/*
 * Licensed to the OpenAirInterface (OAI) Software Alliance under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The OpenAirInterface Software Alliance licenses this file to You under
6
 * the OAI Public License, Version 1.1  (the "License"); you may not use this file
7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
 * 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
 */

22 23 24 25 26 27 28 29 30 31
/*
* @defgroup _PHY_MODULATION_
* @ingroup _physical_layer_ref_implementation_
* @{
\section _phy_modulation_ OFDM Modulation Blocks
This section deals with basic functions for OFDM Modulation.


*/

32 33
#include "PHY/defs_eNB.h"
#include "PHY/impl_defs_top.h"
knopp's avatar
knopp committed
34
#include "UTIL/LOG/log.h"
35
#include "UTIL/LOG/vcd_signal_dumper.h"
36
#include "modulation_common.h"
knopp's avatar
knopp committed
37
#include "PHY/LTE_TRANSPORT/transport_common_proto.h"
38 39 40
//#define DEBUG_OFDM_MOD


41 42
void normal_prefix_mod(int32_t *txdataF,int32_t *txdata,uint8_t nsymb,LTE_DL_FRAME_PARMS *frame_parms)
{
43 44


45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60
  
  PHY_ofdm_mod(txdataF,        // input
	       txdata,         // output
	       frame_parms->ofdm_symbol_size,                
	       1,                 // number of symbols
	       frame_parms->nb_prefix_samples0,               // number of prefix samples
	       CYCLIC_PREFIX);
  PHY_ofdm_mod(txdataF+frame_parms->ofdm_symbol_size,        // input
	       txdata+OFDM_SYMBOL_SIZE_COMPLEX_SAMPLES0,         // output
	       frame_parms->ofdm_symbol_size,                
	       nsymb-1,
	       frame_parms->nb_prefix_samples,               // number of prefix samples
	       CYCLIC_PREFIX);
  

  
61 62 63
}

void PHY_ofdm_mod(int *input,                       /// pointer to complex input
64
                  int *output,                      /// pointer to complex output
65
                  int fftsize,            /// FFT_SIZE
66 67 68 69 70
                  unsigned char nb_symbols,         /// number of OFDM symbols
                  unsigned short nb_prefix_samples,  /// cyclic prefix length
                  Extension_t etype                /// type of extension
                 )
{
71

72
  short temp[2048*4] __attribute__((aligned(32)));
73 74 75 76 77 78 79 80
  unsigned short i,j;
  short k;

  volatile int *output_ptr=(int*)0;

  int *temp_ptr=(int*)0;
  void (*idft)(int16_t *,int16_t *, int);

81 82
  switch (fftsize) {
  case 128:
83 84
    idft = idft128;
    break;
85

86
  case 256:
87 88
    idft = idft256;
    break;
89

90
  case 512:
91 92
    idft = idft512;
    break;
93

94
  case 1024:
95 96
    idft = idft1024;
    break;
97

98 99 100 101 102
  case 1536:
    idft = idft1536;
    break;

  case 2048:
103 104
    idft = idft2048;
    break;
105

106 107 108 109 110 111
  default:
    idft = idft512;
    break;
  }

#ifdef DEBUG_OFDM_MOD
112 113
  printf("[PHY] OFDM mod (size %d,prefix %d) Symbols %d, input %p, output %p\n",
      fftsize,nb_prefix_samples,nb_symbols,input,output);
114 115 116
#endif


117 118

  for (i=0; i<nb_symbols; i++) {
119 120

#ifdef DEBUG_OFDM_MOD
121
    printf("[PHY] symbol %d/%d offset %d (%p,%p -> %p)\n",i,nb_symbols,i*fftsize+(i*nb_prefix_samples),input,&input[i*fftsize],&output[(i*fftsize) + ((i)*nb_prefix_samples)]);
122 123
#endif

124 125
#ifndef __AVX2__
    // handle 128-bit alignment for 128-bit SIMD (SSE4,NEON,AltiVEC)
Florian Kaltenberger's avatar
Florian Kaltenberger committed
126
    idft((int16_t *)&input[i*fftsize],
127
         (fftsize==128) ? (int16_t *)temp : (int16_t *)&output[(i*fftsize) + ((1+i)*nb_prefix_samples)],
128
         1);
129 130
#else
    // on AVX2 need 256-bit alignment
Florian Kaltenberger's avatar
Florian Kaltenberger committed
131
    idft((int16_t *)&input[i*fftsize],
132
         (int16_t *)temp,
133
         1);
134

135
#endif
136

137 138
    // Copy to frame buffer with Cyclic Extension
    // Note:  will have to adjust for synchronization offset!
139

140 141
    switch (etype) {
    case CYCLIC_PREFIX:
142
      output_ptr = &output[(i*fftsize) + ((1+i)*nb_prefix_samples)];
143
      temp_ptr = (int *)temp;
144

145 146 147

      //      msg("Doing cyclic prefix method\n");

148
#ifndef __AVX2__
149
      if (fftsize==128) 
150 151
#endif
      {
152
        /*for (j=0; j<fftsize ; j++) {
153
          output_ptr[j] = temp_ptr[j];
154 155
        }*/
        memcpy((void*)output_ptr,(void*)temp_ptr,fftsize<<2);
156
      }
157

158
      j=fftsize;
159 160 161

      for (k=-1; k>=-nb_prefix_samples; k--) {
        output_ptr[k] = output_ptr[--j];
162
      }
163

164
      break;
165

166
    case CYCLIC_SUFFIX:
167 168


169
      output_ptr = &output[(i*fftsize)+ (i*nb_prefix_samples)];
170

171
      temp_ptr = (int *)temp;
172

173 174
      //      msg("Doing cyclic suffix method\n");

175
      for (j=0; j<fftsize ; j++) {
176
        output_ptr[j] = temp_ptr[2*j];
177
      }
178 179 180


      for (j=0; j<nb_prefix_samples; j++)
181
        output_ptr[fftsize+j] = output_ptr[j];
182

183 184 185 186 187 188 189 190 191
      break;

    case ZEROS:

      break;

    case NONE:

      //      msg("NO EXTENSION!\n");
192
      output_ptr = &output[fftsize];
193 194

      temp_ptr = (int *)temp;
195

196
      for (j=0; j<fftsize ; j++) {
197
        output_ptr[j] = temp_ptr[2*j];
198 199 200 201 202 203 204 205 206 207 208


      }

      break;

    default:
      break;

    }

209 210


211
  }
212

213

214 215
}

knopp's avatar
knopp committed
216

217
void do_OFDM_mod(int32_t **txdataF, int32_t **txdata, uint32_t frame,uint16_t next_slot, LTE_DL_FRAME_PARMS *frame_parms)
218
{
knopp's avatar
knopp committed
219 220 221 222 223

  int aa, slot_offset, slot_offset_F;

  slot_offset_F = (next_slot)*(frame_parms->ofdm_symbol_size)*((frame_parms->Ncp==1) ? 6 : 7);
  slot_offset = (next_slot)*(frame_parms->samples_per_tti>>1);
224

knopp's avatar
knopp committed
225
  for (aa=0; aa<frame_parms->nb_antennas_tx; aa++) {
226
    if (is_pmch_subframe(frame,next_slot>>1,frame_parms)) {
knopp's avatar
knopp committed
227
      if ((next_slot%2)==0) {
228 229 230
        LOG_D(PHY,"Frame %d, subframe %d: Doing MBSFN modulation (slot_offset %d)\n",frame,next_slot>>1,slot_offset);
        PHY_ofdm_mod(&txdataF[aa][slot_offset_F],        // input
                     &txdata[aa][slot_offset],         // output
231
                     frame_parms->ofdm_symbol_size,                
232 233 234 235 236 237 238
                     12,                 // number of symbols
                     frame_parms->ofdm_symbol_size>>2,               // number of prefix samples
                     CYCLIC_PREFIX);

        if (frame_parms->Ncp == EXTENDED)
          PHY_ofdm_mod(&txdataF[aa][slot_offset_F],        // input
                       &txdata[aa][slot_offset],         // output
239
                       frame_parms->ofdm_symbol_size,                
240 241 242 243 244 245 246 247 248 249
                       2,                 // number of symbols
                       frame_parms->nb_prefix_samples,               // number of prefix samples
                       CYCLIC_PREFIX);
        else {
          LOG_D(PHY,"Frame %d, subframe %d: Doing PDCCH modulation\n",frame,next_slot>>1);
          normal_prefix_mod(&txdataF[aa][slot_offset_F],
                            &txdata[aa][slot_offset],
                            2,
                            frame_parms);
        }
knopp's avatar
knopp committed
250
      }
251
    } else {
knopp's avatar
knopp committed
252
      if (frame_parms->Ncp == EXTENDED)
253 254
        PHY_ofdm_mod(&txdataF[aa][slot_offset_F],        // input
                     &txdata[aa][slot_offset],         // output
255
                     frame_parms->ofdm_symbol_size,                
256 257 258
                     6,                 // number of symbols
                     frame_parms->nb_prefix_samples,               // number of prefix samples
                     CYCLIC_PREFIX);
knopp's avatar
knopp committed
259
      else {
260 261 262 263
        normal_prefix_mod(&txdataF[aa][slot_offset_F],
                          &txdata[aa][slot_offset],
                          7,
                          frame_parms);
knopp's avatar
knopp committed
264
      }
265
    }
knopp's avatar
knopp committed
266
  }
267

knopp's avatar
knopp committed
268 269
}

270
/*
271
// OFDM modulation for each symbol
272
void do_OFDM_mod_symbol(LTE_eNB_COMMON *eNB_common_vars, RU_COMMON *common, uint16_t next_slot, LTE_DL_FRAME_PARMS *frame_parms,int do_precoding)
273 274
{

275
  int aa, l, slot_offset, slot_offsetF;
276 277 278
  int32_t **txdataF    = eNB_common_vars->txdataF;
  int32_t **txdataF_BF = common->txdataF_BF;
  int32_t **txdata     = common->txdata;
279

280 281
  slot_offset  = (next_slot)*(frame_parms->samples_per_tti>>1);
  slot_offsetF = (next_slot)*(frame_parms->ofdm_symbol_size)*((frame_parms->Ncp==EXTENDED) ? 6 : 7);
282
  //printf("Thread %d starting ... aa %d (%llu)\n",omp_get_thread_num(),aa,rdtsc());
283 284 285 286
  for (l=0; l<frame_parms->symbols_per_tti>>1; l++) {
  
    for (aa=0; aa<frame_parms->nb_antennas_tx; aa++) {

287 288
      //printf("do_OFDM_mod_l, slot=%d, l=%d, NUMBER_OF_OFDM_CARRIERS=%d,OFDM_SYMBOL_SIZE_COMPLEX_SAMPLES=%d\n",next_slot, l,NUMBER_OF_OFDM_CARRIERS,OFDM_SYMBOL_SIZE_COMPLEX_SAMPLES);
      VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_ENB_BEAM_PRECODING,1);
289
      if (do_precoding==1) beam_precoding(txdataF,txdataF_BF,frame_parms,eNB_common_vars->beam_weights,next_slot,l,aa);
290 291
      VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_ENB_BEAM_PRECODING,0);

292 293
      //PMCH case not implemented... 

294 295
      if (frame_parms->Ncp == EXTENDED)
        PHY_ofdm_mod((do_precoding == 1)?txdataF_BF[aa]:&txdataF[aa][slot_offsetF+l*frame_parms->ofdm_symbol_size],         // input
296 297 298 299 300 301 302
                     &txdata[aa][slot_offset+l*OFDM_SYMBOL_SIZE_COMPLEX_SAMPLES],            // output
                     frame_parms->ofdm_symbol_size,       
                     1,                                   // number of symbols
                     frame_parms->nb_prefix_samples,      // number of prefix samples
                     CYCLIC_PREFIX);
      else {
        if (l==0) {
303
          PHY_ofdm_mod((do_precoding==1)?txdataF_BF[aa]:&txdataF[aa][slot_offsetF+l*frame_parms->ofdm_symbol_size],        // input
304 305 306 307 308 309 310
                       &txdata[aa][slot_offset],           // output
                       frame_parms->ofdm_symbol_size,      
                       1,                                  // number of symbols
                       frame_parms->nb_prefix_samples0,    // number of prefix samples
                       CYCLIC_PREFIX);
           
        } else {
311
	  PHY_ofdm_mod((do_precoding==1)?txdataF_BF[aa]:&txdataF[aa][slot_offsetF+l*frame_parms->ofdm_symbol_size],        // input
312 313 314 315 316
                       &txdata[aa][slot_offset+OFDM_SYMBOL_SIZE_COMPLEX_SAMPLES0+(l-1)*OFDM_SYMBOL_SIZE_COMPLEX_SAMPLES],           // output
                       frame_parms->ofdm_symbol_size,      
                       1,                                  // number of symbols
                       frame_parms->nb_prefix_samples,     // number of prefix samples
                       CYCLIC_PREFIX);
317

318 319
          //printf("txdata[%d][%d]=%d\n",aa,slot_offset+OFDM_SYMBOL_SIZE_COMPLEX_SAMPLES0+(l-1)*OFDM_SYMBOL_SIZE_COMPLEX_SAMPLES,txdata[aa][slot_offset+OFDM_SYMBOL_SIZE_COMPLEX_SAMPLES0+(l-1)*OFDM_SYMBOL_SIZE_COMPLEX_SAMPLES]);
 
320 321 322 323 324 325
        }
      }
    }
  }

}
326
*/