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

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


*/

#include "PHY/defs.h"
Raymond Knopp's avatar
 
Raymond Knopp committed
33
#include "UTIL/LOG/log.h"
34
#include "UTIL/LOG/vcd_signal_dumper.h"
35

36 37 38
//#define DEBUG_OFDM_MOD


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


43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58
  
  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);
  

  
59 60 61
}

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

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

  volatile int *output_ptr=(int*)0;

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

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

84
  case 256:
85 86
    idft = idft256;
    break;
87

88
  case 512:
89 90
    idft = idft512;
    break;
91

92
  case 1024:
93 94
    idft = idft1024;
    break;
95

96 97 98 99 100
  case 1536:
    idft = idft1536;
    break;

  case 2048:
101 102
    idft = idft2048;
    break;
103

104 105 106 107 108 109
  default:
    idft = idft512;
    break;
  }

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


115 116

  for (i=0; i<nb_symbols; i++) {
117 118

#ifdef DEBUG_OFDM_MOD
119
    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)]);
120 121
#endif

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

133
#endif
134

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

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

143 144 145

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

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

156
      j=fftsize;
157 158 159

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

162
      break;
163

164
    case CYCLIC_SUFFIX:
165 166


167
      output_ptr = &output[(i*fftsize)+ (i*nb_prefix_samples)];
168

169
      temp_ptr = (int *)temp;
170

171 172
      //      msg("Doing cyclic suffix method\n");

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


      for (j=0; j<nb_prefix_samples; j++)
179
        output_ptr[fftsize+j] = output_ptr[j];
180

181 182 183 184 185 186 187 188 189
      break;

    case ZEROS:

      break;

    case NONE:

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

      temp_ptr = (int *)temp;
193

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


      }

      break;

    default:
      break;

    }

207 208


209
  }
210

211

212 213
}

Raymond Knopp's avatar
 
Raymond Knopp committed
214

215
void do_OFDM_mod(int32_t **txdataF, int32_t **txdata, uint32_t frame,uint16_t next_slot, LTE_DL_FRAME_PARMS *frame_parms)
216
{
Raymond Knopp's avatar
 
Raymond Knopp committed
217 218 219 220 221

  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);
222

Raymond Knopp's avatar
 
Raymond Knopp committed
223
  for (aa=0; aa<frame_parms->nb_antennas_tx; aa++) {
224
    if (is_pmch_subframe(frame,next_slot>>1,frame_parms)) {
Raymond Knopp's avatar
 
Raymond Knopp committed
225
      if ((next_slot%2)==0) {
226 227 228
        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
229
                     frame_parms->ofdm_symbol_size,                
230 231 232 233 234 235 236
                     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
237
                       frame_parms->ofdm_symbol_size,                
238 239 240 241 242 243 244 245 246 247
                       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);
        }
Raymond Knopp's avatar
 
Raymond Knopp committed
248
      }
249
    } else {
Raymond Knopp's avatar
 
Raymond Knopp committed
250
      if (frame_parms->Ncp == EXTENDED)
251 252
        PHY_ofdm_mod(&txdataF[aa][slot_offset_F],        // input
                     &txdata[aa][slot_offset],         // output
253
                     frame_parms->ofdm_symbol_size,                
254 255 256
                     6,                 // number of symbols
                     frame_parms->nb_prefix_samples,               // number of prefix samples
                     CYCLIC_PREFIX);
Raymond Knopp's avatar
 
Raymond Knopp committed
257
      else {
258 259 260 261
        normal_prefix_mod(&txdataF[aa][slot_offset_F],
                          &txdata[aa][slot_offset],
                          7,
                          frame_parms);
Raymond Knopp's avatar
 
Raymond Knopp committed
262
      }
263
    }
Raymond Knopp's avatar
 
Raymond Knopp committed
264
  }
265

Raymond Knopp's avatar
 
Raymond Knopp committed
266 267
}

268
/*
269
// OFDM modulation for each symbol
270
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)
271 272
{

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

278 279
  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);
280
  //printf("Thread %d starting ... aa %d (%llu)\n",omp_get_thread_num(),aa,rdtsc());
281 282 283 284
  for (l=0; l<frame_parms->symbols_per_tti>>1; l++) {
  
    for (aa=0; aa<frame_parms->nb_antennas_tx; aa++) {

285 286
      //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);
287
      if (do_precoding==1) beam_precoding(txdataF,txdataF_BF,frame_parms,eNB_common_vars->beam_weights,next_slot,l,aa);
288 289
      VCD_SIGNAL_DUMPER_DUMP_FUNCTION_BY_NAME(VCD_SIGNAL_DUMPER_FUNCTIONS_ENB_BEAM_PRECODING,0);

290 291
      //PMCH case not implemented... 

292 293
      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
294 295 296 297 298 299 300
                     &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) {
301
          PHY_ofdm_mod((do_precoding==1)?txdataF_BF[aa]:&txdataF[aa][slot_offsetF+l*frame_parms->ofdm_symbol_size],        // input
302 303 304 305 306 307 308
                       &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 {
309
	  PHY_ofdm_mod((do_precoding==1)?txdataF_BF[aa]:&txdataF[aa][slot_offsetF+l*frame_parms->ofdm_symbol_size],        // input
310 311 312 313 314
                       &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);
315

316 317
          //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]);
 
318 319 320 321 322 323
        }
      }
    }
  }

}
324
*/