nr_pbch.c 15.5 KB
Newer Older
Guy De Souza's avatar
Guy De Souza committed
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.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
 */

Guy De Souza's avatar
Guy De Souza committed
22
/*! \file PHY/NR_TRANSPORT/nr_pbch.c
Guy De Souza's avatar
Guy De Souza committed
23
* \brief Top-level routines for generating the PBCH/BCH physical/transport channel V15.1 03/2018
24
* \author Guy De Souza
25
* \thanks Special Thanks to Son Dang for helpful contributions and testing
Guy De Souza's avatar
Guy De Souza committed
26
* \date 2018
Guy De Souza's avatar
Guy De Souza committed
27
28
* \version 0.1
* \company Eurecom
Guy De Souza's avatar
Guy De Souza committed
29
* \email: desouza@eurecom.fr
Guy De Souza's avatar
Guy De Souza committed
30
31
32
* \note
* \warning
*/
Guy De Souza's avatar
Guy De Souza committed
33

34
#include "PHY/defs_gNB.h"
Guy De Souza's avatar
Guy De Souza committed
35
#include "PHY/NR_TRANSPORT/nr_transport.h"
Guy De Souza's avatar
Guy De Souza committed
36
#include "PHY/LTE_REFSIG/lte_refsig.h"
Guy De Souza's avatar
Guy De Souza committed
37
38
#include "PHY/sse_intrin.h"

Guy De Souza's avatar
Guy De Souza committed
39
//#define DEBUG_PBCH
knopp's avatar
knopp committed
40
//#define DEBUG_PBCH_ENCODING
41
//#define DEBUG_PBCH_DMRS
Guy De Souza's avatar
Guy De Souza committed
42

43
#include "PHY/NR_REFSIG/nr_mod_table.h"
Guy De Souza's avatar
Guy De Souza committed
44

Guy De Souza's avatar
Guy De Souza committed
45
46
47
uint8_t nr_pbch_payload_interleaving_pattern[32] = {16, 23, 18, 17, 8, 30, 10, 6, 24, 7, 0, 5, 3, 2, 1, 4,
                                                9, 11, 12, 13, 14, 15, 19, 20, 21, 22, 25, 26, 27, 28, 29, 31};

Guy De Souza's avatar
Guy De Souza committed
48
49
50
51
int nr_generate_pbch_dmrs(uint32_t *gold_pbch_dmrs,
                          int32_t **txdataF,
                          int16_t amp,
                          uint8_t ssb_start_symbol,
ChenWeiTai's avatar
ChenWeiTai committed
52
                          nfapi_nr_config_request_t* config,
Guy De Souza's avatar
Guy De Souza committed
53
54
                          NR_DL_FRAME_PARMS *frame_parms)
{
55
56
  int k,l;
  int16_t a;
57
58
  int16_t mod_dmrs[NR_PBCH_DMRS_LENGTH<<1];
  uint8_t idx=0;
Guy De Souza's avatar
Guy De Souza committed
59
  uint8_t nushift = config->sch_config.physical_cell_id.value &3;
Guy De Souza's avatar
Guy De Souza committed
60

61
  LOG_D(PHY, "PBCH DMRS mapping started at symbol %d shift %d\n", ssb_start_symbol+1, nushift);
Guy De Souza's avatar
Guy De Souza committed
62

63
  /// QPSK modulation
64
  for (int m=0; m<NR_PBCH_DMRS_LENGTH; m++) {
65
66
67
    idx = ((((gold_pbch_dmrs[(m<<1)>>5])>>((m<<1)&0x1f))&1)<<1) ^ (((gold_pbch_dmrs[((m<<1)+1)>>5])>>(((m<<1)+1)&0x1f))&1);
    mod_dmrs[m<<1] = nr_mod_table[(NR_MOD_TABLE_QPSK_OFFSET + idx)<<1];
    mod_dmrs[(m<<1)+1] = nr_mod_table[((NR_MOD_TABLE_QPSK_OFFSET + idx)<<1) + 1];
Guy De Souza's avatar
Guy De Souza committed
68
#ifdef DEBUG_PBCH_DMRS
69
70
  printf("m %d idx %d gold seq %d b0-b1 %d-%d mod_dmrs %d %d\n", m, idx, gold_pbch_dmrs[(m<<1)>>5], (((gold_pbch_dmrs[(m<<1)>>5])>>((m<<1)&0x1f))&1),
  (((gold_pbch_dmrs[((m<<1)+1)>>5])>>(((m<<1)+1)&0x1f))&1), mod_dmrs[(m<<1)], mod_dmrs[(m<<1)+1]);
Guy De Souza's avatar
Guy De Souza committed
71
72
73
74
75
76
#endif
  }

  /// Resource mapping
  a = (config->rf_config.tx_antenna_ports.value == 1) ? amp : (amp*ONE_OVER_SQRT2_Q15)>>15;

77
  for (int aa = 0; aa < config->rf_config.tx_antenna_ports.value; aa++)
Guy De Souza's avatar
Guy De Souza committed
78
79
  {

80
81
82
    // PBCH DMRS are mapped  within the SSB block on every fourth subcarrier starting from nushift of symbols 1, 2, 3
      ///symbol 1  [0+nushift:4:236+nushift] -- 60 mod symbols
    k = frame_parms->first_carrier_offset + frame_parms->ssb_start_subcarrier + nushift;
Guy De Souza's avatar
Guy De Souza committed
83
84
    l = ssb_start_symbol + 1;

85
    for (int m = 0; m < 60; m++) {
Guy De Souza's avatar
Guy De Souza committed
86
#ifdef DEBUG_PBCH_DMRS
Guy De Souza's avatar
Guy De Souza committed
87
  printf("m %d at k %d of l %d\n", m, k, l);
Guy De Souza's avatar
Guy De Souza committed
88
#endif
Guy De Souza's avatar
Guy De Souza committed
89
      ((int16_t*)txdataF[aa])[(l*frame_parms->ofdm_symbol_size + k)<<1] = (a * mod_dmrs[m<<1]) >> 15;
Guy De Souza's avatar
Guy De Souza committed
90
      ((int16_t*)txdataF[aa])[((l*frame_parms->ofdm_symbol_size + k)<<1) + 1] = (a * mod_dmrs[(m<<1) + 1]) >> 15;
knopp's avatar
knopp committed
91
92
93
94
95
#ifdef DEBUG_PBCH_DMRS
      printf("(%d,%d)\n",
	((int16_t*)txdataF[aa])[(l*frame_parms->ofdm_symbol_size + k)<<1],
	((int16_t*)txdataF[aa])[((l*frame_parms->ofdm_symbol_size + k)<<1)+1]);
#endif
Guy De Souza's avatar
Guy De Souza committed
96
97
98
99
100
101
      k+=4;

      if (k >= frame_parms->ofdm_symbol_size)
        k-=frame_parms->ofdm_symbol_size;
    }

102
103
      ///symbol 2  [0+u:4:44+nushift ; 192+nu:4:236+nushift] -- 24 mod symbols
    k = frame_parms->first_carrier_offset + frame_parms->ssb_start_subcarrier + nushift;
Guy De Souza's avatar
Guy De Souza committed
104
105
    l++;

106
    for (int m = 60; m < 84; m++) {
Guy De Souza's avatar
Guy De Souza committed
107
#ifdef DEBUG_PBCH_DMRS
Guy De Souza's avatar
Guy De Souza committed
108
  printf("m %d at k %d of l %d\n", m, k, l);
Guy De Souza's avatar
Guy De Souza committed
109
#endif
Guy De Souza's avatar
Guy De Souza committed
110
      ((int16_t*)txdataF[aa])[(l*frame_parms->ofdm_symbol_size + k)<<1] = (a * mod_dmrs[m<<1]) >> 15;
Guy De Souza's avatar
Guy De Souza committed
111
      ((int16_t*)txdataF[aa])[((l*frame_parms->ofdm_symbol_size + k)<<1) + 1] = (a * mod_dmrs[(m<<1) + 1]) >> 15;
knopp's avatar
knopp committed
112
113
114
115
116
#ifdef DEBUG_PBCH_DMRS
      printf("(%d,%d)\n",
	((int16_t*)txdataF[aa])[(l*frame_parms->ofdm_symbol_size + k)<<1],
	((int16_t*)txdataF[aa])[((l*frame_parms->ofdm_symbol_size + k)<<1)+1]);
#endif
Guy De Souza's avatar
Guy De Souza committed
117
      k+=(m==71)?148:4; // Jump from 44+nu to 192+nu
Guy De Souza's avatar
Guy De Souza committed
118
119
120
121

      if (k >= frame_parms->ofdm_symbol_size)
        k-=frame_parms->ofdm_symbol_size;
    }
Guy De Souza's avatar
Guy De Souza committed
122

123
124
      ///symbol 3  [0+nushift:4:236+nushift] -- 60 mod symbols
    k = frame_parms->first_carrier_offset + frame_parms->ssb_start_subcarrier + nushift;
Guy De Souza's avatar
Guy De Souza committed
125
126
    l++;

127
    for (int m = 84; m < NR_PBCH_DMRS_LENGTH; m++) {
Guy De Souza's avatar
Guy De Souza committed
128
#ifdef DEBUG_PBCH_DMRS
Guy De Souza's avatar
Guy De Souza committed
129
  printf("m %d at k %d of l %d\n", m, k, l);
Guy De Souza's avatar
Guy De Souza committed
130
#endif
Guy De Souza's avatar
Guy De Souza committed
131
      ((int16_t*)txdataF[aa])[(l*frame_parms->ofdm_symbol_size + k)<<1] = (a * mod_dmrs[m<<1]) >> 15;
Guy De Souza's avatar
Guy De Souza committed
132
      ((int16_t*)txdataF[aa])[((l*frame_parms->ofdm_symbol_size + k)<<1) + 1] = (a * mod_dmrs[(m<<1) + 1]) >> 15;
knopp's avatar
knopp committed
133
134
135
136
137
#ifdef DEBUG_PBCH_DMRS
      printf("(%d,%d)\n",
	((int16_t*)txdataF[aa])[(l*frame_parms->ofdm_symbol_size + k)<<1],
	((int16_t*)txdataF[aa])[((l*frame_parms->ofdm_symbol_size + k)<<1)+1]);
#endif
Guy De Souza's avatar
Guy De Souza committed
138
139
140
141
142
143
      k+=4;

      if (k >= frame_parms->ofdm_symbol_size)
        k-=frame_parms->ofdm_symbol_size;
    }

Guy De Souza's avatar
Guy De Souza committed
144
  }
Guy De Souza's avatar
Guy De Souza committed
145
146


Guy De Souza's avatar
Guy De Souza committed
147
148
#ifdef DEBUG_PBCH_DMRS
  write_output("txdataF_pbch_dmrs.m", "txdataF_pbch_dmrs", txdataF[0], frame_parms->samples_per_frame_wCP>>1, 1, 1);
Guy De Souza's avatar
Guy De Souza committed
149
#endif
Guy De Souza's avatar
Guy De Souza committed
150
151
152
  return 0;
}

Guy De Souza's avatar
Guy De Souza committed
153
154
void nr_pbch_scrambling(NR_gNB_PBCH *pbch,
                        uint32_t Nid,
Guy De Souza's avatar
Guy De Souza committed
155
                        uint8_t nushift,
Guy De Souza's avatar
Guy De Souza committed
156
                        uint16_t M,
Guy De Souza's avatar
Guy De Souza committed
157
                        uint16_t length,
Guy De Souza's avatar
Guy De Souza committed
158
159
                        uint8_t encoded,
                        uint32_t unscrambling_mask) {
Guy De Souza's avatar
Guy De Souza committed
160
  uint8_t reset, offset;
161
  uint32_t x1, x2, s=0;
Hongzhi's avatar
Hongzhi committed
162
  uint32_t *pbch_e = pbch->pbch_e;
163
164
165
  reset = 1;
  // x1 is set in lte_gold_generic
  x2 = Nid;
Guy De Souza's avatar
Guy De Souza committed
166

167
  // The Gold sequence is shifted by nushift* M, so we skip (nushift*M /32) double words
knopp's avatar
knopp committed
168
  for (int i=0; i<(uint16_t)ceil(((float)nushift*M)/32); i++) {
Guy De Souza's avatar
Guy De Souza committed
169
    s = lte_gold_generic(&x1, &x2, reset);
170
171
    reset = 0;
  }
Guy De Souza's avatar
Guy De Souza committed
172

Guy De Souza's avatar
Guy De Souza committed
173
  // Scrambling is now done with offset (nushift*M)%32
Guy De Souza's avatar
Guy De Souza committed
174
  offset = (nushift*M)&0x1f;
Guy De Souza's avatar
Guy De Souza committed
175
#ifdef DEBUG_PBCH_ENCODING
Hongzhi's avatar
Hongzhi committed
176
  printf("Scrambling params: nushift %d M %d length %d encoded %d offset %d\n", nushift, M, length, encoded, offset);
Guy De Souza's avatar
Guy De Souza committed
177
178
#endif
#ifdef DEBUG_PBCH_ENCODING
Guy De Souza's avatar
Guy De Souza committed
179
  printf("s: %04x\t", s);
Guy De Souza's avatar
Guy De Souza committed
180
#endif
Guy De Souza's avatar
Guy De Souza committed
181
182
183
184
185
  int k = 0;

  if (!encoded) {
    /// 1st Scrambling
    for (int i = 0; i < length; ++i) {
186
187
      if ((unscrambling_mask>>i)&1)
        pbch->pbch_a_prime ^= ((pbch->pbch_a_interleaved>>i)&1)<<i;
Guy De Souza's avatar
Guy De Souza committed
188
189
190
191
192
      else {
        if (((k+offset)&0x1f)==0) {
          s = lte_gold_generic(&x1, &x2, reset);
          reset = 0;
        }
193
        pbch->pbch_a_prime ^= (((pbch->pbch_a_interleaved>>i)&1) ^ ((s>>((k+offset)&0x1f))&1))<<i;
Guy De Souza's avatar
Guy De Souza committed
194
195
196
197
198
199
200
201
202
203
        k++;                  /// k increase only when payload bit is not special bit
      }
    }
  } else {
    /// 2nd Scrambling
    for (int i = 0; i < length; ++i) {
      if (((i+offset)&0x1f)==0) {
        s = lte_gold_generic(&x1, &x2, reset);
        reset = 0;
      }
Guy De Souza's avatar
Guy De Souza committed
204

Guy De Souza's avatar
Guy De Souza committed
205
      pbch_e[i>>5] ^= (((s>>((i+offset)&0x1f))&1)<<(i&0x1f));
Guy De Souza's avatar
Guy De Souza committed
206
    }
207
208
209
  }
}

Guy De Souza's avatar
Guy De Souza committed
210

Guy De Souza's avatar
Guy De Souza committed
211
void nr_init_pbch_interleaver(uint8_t *interleaver) {
Guy De Souza's avatar
Guy De Souza committed
212
213
214
  uint8_t j_sfn=0, j_hrf=10, j_ssb=11, j_other=14;
  memset((void*)interleaver,0, NR_POLAR_PBCH_PAYLOAD_BITS);

Guy De Souza's avatar
Guy De Souza committed
215
  for (uint8_t i=0; i<NR_POLAR_PBCH_PAYLOAD_BITS; i++)
Guy De Souza's avatar
Guy De Souza committed
216
    if (!i) // choice bit:1
217
     *(interleaver+i) = *(nr_pbch_payload_interleaving_pattern+j_other++);
Guy De Souza's avatar
Guy De Souza committed
218
    else if (i<7) //Sfn bits:6
219
      *(interleaver+i) = *(nr_pbch_payload_interleaving_pattern+j_sfn++);
Guy De Souza's avatar
Guy De Souza committed
220
    else if (i<24) // other:17
221
      *(interleaver+i) = *(nr_pbch_payload_interleaving_pattern+j_other++);
Guy De Souza's avatar
Guy De Souza committed
222
    else if (i<28) // Sfn:4
223
      *(interleaver+i) = *(nr_pbch_payload_interleaving_pattern+j_sfn++);
Guy De Souza's avatar
Guy De Souza committed
224
    else if (i==28) // Hrf bit:1
225
      *(interleaver+i) = *(nr_pbch_payload_interleaving_pattern+j_hrf);
Guy De Souza's avatar
Guy De Souza committed
226
    else // Ssb bits:3
227
      *(interleaver+i) = *(nr_pbch_payload_interleaving_pattern+j_ssb++);
Guy De Souza's avatar
Guy De Souza committed
228

229
}
Guy De Souza's avatar
Guy De Souza committed
230

Guy De Souza's avatar
Guy De Souza committed
231
int nr_generate_pbch(NR_gNB_PBCH *pbch,
Guy De Souza's avatar
Guy De Souza committed
232
                     t_nrPolar_paramsPtr polar_params,
Guy De Souza's avatar
Guy De Souza committed
233
                     uint8_t *pbch_pdu,
Guy De Souza's avatar
Guy De Souza committed
234
                     uint8_t *interleaver,
Guy De Souza's avatar
Guy De Souza committed
235
236
237
                     int32_t **txdataF,
                     int16_t amp,
                     uint8_t ssb_start_symbol,
238
                     uint8_t n_hf,
Guy De Souza's avatar
Guy De Souza committed
239
240
                     uint8_t Lmax,
                     uint8_t ssb_index,
241
                     int sfn,
ChenWeiTai's avatar
ChenWeiTai committed
242
                     nfapi_nr_config_request_t* config,
Guy De Souza's avatar
Guy De Souza committed
243
244
245
                     NR_DL_FRAME_PARMS *frame_parms)
{

Guy De Souza's avatar
Guy De Souza committed
246
  int k,l,m;
247
  int16_t a;
Guy De Souza's avatar
Guy De Souza committed
248
249
250
251
  int16_t mod_pbch_e[NR_POLAR_PBCH_E];
  uint8_t idx=0;
  uint16_t M;
  uint8_t nushift;
Guy De Souza's avatar
Guy De Souza committed
252
  uint32_t unscrambling_mask;
Guy De Souza's avatar
Guy De Souza committed
253

Guy De Souza's avatar
Guy De Souza committed
254
  LOG_I(PHY, "PBCH generation started\n");
Guy De Souza's avatar
Guy De Souza committed
255

256
  ///Payload generation
Guy De Souza's avatar
Guy De Souza committed
257
258
259
260

  memset((void*)pbch, 0, sizeof(NR_gNB_PBCH));
  pbch->pbch_a=0;
  for (int i=0; i<NR_PBCH_PDU_BITS; i++)
261
    pbch->pbch_a |= ((pbch_pdu[2-(i>>3)]>>(7-(i&7)))&1)<<i;
Guy De Souza's avatar
Guy De Souza committed
262
#ifdef DEBUG_PBCH_ENCODING
Guy De Souza's avatar
Guy De Souza committed
263
264
265
  for (int i=0; i<3; i++)
    printf("pbch_pdu[%d]: 0x%02x\n", i, pbch_pdu[i]);
  printf("PBCH payload = 0x%08x\n",pbch->pbch_a);
Guy De Souza's avatar
Guy De Souza committed
266
#endif
267

Guy De Souza's avatar
Guy De Souza committed
268
    // Extra byte generation
269
  for (int i=0; i<4; i++)
270
    pbch->pbch_a |= ((sfn>>(3-i))&1)<<(24+i); // resp. 4th, 3rd, 2nd ans 1st lsb of sfn
271

272
  pbch->pbch_a |= n_hf<<28; // half frame index bit
273

Guy De Souza's avatar
Guy De Souza committed
274
275
  if (Lmax == 64)
    for (int i=0; i<3; i++)
276
      pbch->pbch_a |= ((ssb_index>>(5-i))&1)<<(29+i); // resp. 6th, 5th and 4th bits of ssb_index
Guy De Souza's avatar
Guy De Souza committed
277
  else
278
    pbch->pbch_a |= ((config->sch_config.ssb_subcarrier_offset.value>>4)&1)<<29; //MSB of k_SSB (bit index 4)
Guy De Souza's avatar
Guy De Souza committed
279

Guy De Souza's avatar
Guy De Souza committed
280
  LOG_I(PHY,"After extra byte: pbch_a = 0x%08x\n",pbch->pbch_a);
Guy De Souza's avatar
Guy De Souza committed
281

Guy De Souza's avatar
Guy De Souza committed
282
283
    // Payload interleaving
  
Guy De Souza's avatar
Guy De Souza committed
284
  for (int i=0; i<NR_POLAR_PBCH_PAYLOAD_BITS; i++) {
Guy De Souza's avatar
Guy De Souza committed
285
    pbch->pbch_a_interleaved |= ((pbch->pbch_a>>i)&1)<<(*(interleaver+i));
Guy De Souza's avatar
Guy De Souza committed
286
#ifdef DEBUG_PBCH_ENCODING
Guy De Souza's avatar
Guy De Souza committed
287
  printf("i %d out 0x%08x ilv %d (in>>i)&1) %d\n", i, pbch->pbch_a_interleaved, *(interleaver+i), (pbch->pbch_a>>i)&1);
Guy De Souza's avatar
Guy De Souza committed
288
289
#endif
  }
Guy De Souza's avatar
Guy De Souza committed
290

Guy De Souza's avatar
Guy De Souza committed
291
292
#ifdef DEBUG_PBCH_ENCODING
  printf("Interleaving:\n");
knopp's avatar
knopp committed
293
  printf("pbch_a_interleaved: 0x%08x\n", pbch->pbch_a_interleaved);
Guy De Souza's avatar
Guy De Souza committed
294
#endif
295
296

    // Scrambling
297
  unscrambling_mask = (Lmax ==64)? 0x100006D:0x1000041;
Guy De Souza's avatar
Guy De Souza committed
298
299
  M = (Lmax == 64)? (NR_POLAR_PBCH_PAYLOAD_BITS - 6) : (NR_POLAR_PBCH_PAYLOAD_BITS - 3);
  nushift = (((sfn>>2)&1)<<1) ^ ((sfn>>1)&1);
knopp's avatar
knopp committed
300
  pbch->pbch_a_prime = 0;
Guy De Souza's avatar
Guy De Souza committed
301
  nr_pbch_scrambling(pbch, (uint32_t)config->sch_config.physical_cell_id.value, nushift, M, NR_POLAR_PBCH_PAYLOAD_BITS, 0, unscrambling_mask);
Guy De Souza's avatar
Guy De Souza committed
302
#ifdef DEBUG_PBCH_ENCODING
knopp's avatar
knopp committed
303
304
  printf("Payload scrambling: nushift %d M %d sfn3 %d sfn2 %d\n", nushift, M, (sfn>>2)&1, (sfn>>1)&1);
  printf("pbch_a_prime: 0x%08x\n", pbch->pbch_a_prime);
Guy De Souza's avatar
Guy De Souza committed
305
#endif
306
307

  /// CRC, coding and rate matching
knopp's avatar
knopp committed
308
  polar_encoder_fast (&pbch->pbch_a_prime, (uint32_t*)pbch->pbch_e, 0, polar_params);
Guy De Souza's avatar
Guy De Souza committed
309
#ifdef DEBUG_PBCH_ENCODING
Guy De Souza's avatar
Guy De Souza committed
310
  printf("Channel coding:\n");
Hongzhi's avatar
Hongzhi committed
311
  for (int i=0; i<NR_POLAR_PBCH_E_DWORD; i++)
Guy De Souza's avatar
Guy De Souza committed
312
    printf("pbch_e[%d]: 0x%08x\t", i, pbch->pbch_e[i]);
Guy De Souza's avatar
Guy De Souza committed
313
  printf("\n");
Guy De Souza's avatar
Guy De Souza committed
314
#endif
315

Guy De Souza's avatar
Guy De Souza committed
316
317
  /// Scrambling
  M =  NR_POLAR_PBCH_E;
Guy De Souza's avatar
Guy De Souza committed
318
  nushift = (Lmax==4)? ssb_index&3 : ssb_index&7;
Guy De Souza's avatar
Guy De Souza committed
319
  nr_pbch_scrambling(pbch, (uint32_t)config->sch_config.physical_cell_id.value, nushift, M, NR_POLAR_PBCH_E, 1, 0);
Guy De Souza's avatar
Guy De Souza committed
320
321
#ifdef DEBUG_PBCH_ENCODING
  printf("Scrambling:\n");
Hongzhi's avatar
Hongzhi committed
322
  for (int i=0; i<NR_POLAR_PBCH_E_DWORD; i++)
Guy De Souza's avatar
Guy De Souza committed
323
    printf("pbch_e[%d]: 0x%08x\t", i, pbch->pbch_e[i]);
Guy De Souza's avatar
Guy De Souza committed
324
  printf("\n");
Guy De Souza's avatar
Guy De Souza committed
325
#endif
326
327

  /// QPSK modulation
Guy De Souza's avatar
Guy De Souza committed
328
  for (int i=0; i<NR_POLAR_PBCH_E>>1; i++){
Guy De Souza's avatar
Guy De Souza committed
329
    idx = (((pbch->pbch_e[(i<<1)>>5]>>((i<<1)&0x1f))&1)<<1) ^ ((pbch->pbch_e[((i<<1)+1)>>5]>>(((i<<1)+1)&0x1f))&1);
Guy De Souza's avatar
Guy De Souza committed
330
331
    mod_pbch_e[i<<1] = nr_mod_table[(NR_MOD_TABLE_QPSK_OFFSET + idx)<<1];
    mod_pbch_e[(i<<1)+1] = nr_mod_table[((NR_MOD_TABLE_QPSK_OFFSET + idx)<<1)+1];
332
333

#ifdef DEBUG_PBCH
Guy De Souza's avatar
Guy De Souza committed
334
  printf("i %d idx %d  mod_pbch %d %d\n", i, idx, mod_pbch_e[2*i], mod_pbch_e[2*i+1]);
335
336
337
338
#endif
  }

  /// Resource mapping
Guy De Souza's avatar
Guy De Souza committed
339
  nushift = config->sch_config.physical_cell_id.value &3;
Guy De Souza's avatar
Guy De Souza committed
340
  a = (config->rf_config.tx_antenna_ports.value == 1) ? amp : (amp*ONE_OVER_SQRT2_Q15)>>15;
341
342
343
344

  for (int aa = 0; aa < config->rf_config.tx_antenna_ports.value; aa++)
  {

Guy De Souza's avatar
Guy De Souza committed
345
346
347
    // PBCH modulated symbols are mapped  within the SSB block on symbols 1, 2, 3 excluding the subcarriers used for the PBCH DMRS
      ///symbol 1  [0:239] -- 180 mod symbols
    k = frame_parms->first_carrier_offset + frame_parms->ssb_start_subcarrier;
348
    l = ssb_start_symbol + 1;
Guy De Souza's avatar
Guy De Souza committed
349
350
351
    m = 0;

    for (int ssb_sc_idx = 0; ssb_sc_idx < 240; ssb_sc_idx++) {
352

Guy De Souza's avatar
Guy De Souza committed
353
354
      if ((ssb_sc_idx&3) == nushift) {  //skip DMRS
        k++;
Guy De Souza's avatar
Guy De Souza committed
355
        continue;
Guy De Souza's avatar
Guy De Souza committed
356
357
      }
      else {
Guy De Souza's avatar
Guy De Souza committed
358
359
360
#ifdef DEBUG_PBCH
  printf("m %d ssb_sc_idx %d at k %d of l %d\n", m, ssb_sc_idx, k, l);
#endif
Guy De Souza's avatar
Guy De Souza committed
361
362
363
        ((int16_t*)txdataF[aa])[(l*frame_parms->ofdm_symbol_size + k)<<1] = (a * mod_pbch_e[m<<1]) >> 15;
        ((int16_t*)txdataF[aa])[((l*frame_parms->ofdm_symbol_size + k)<<1) + 1] = (a * mod_pbch_e[(m<<1) + 1]) >> 15;
        k++;
Guy De Souza's avatar
Guy De Souza committed
364
        m++;
Guy De Souza's avatar
Guy De Souza committed
365
      }
366
367
368
369
370

      if (k >= frame_parms->ofdm_symbol_size)
        k-=frame_parms->ofdm_symbol_size;
    }

Guy De Souza's avatar
Guy De Souza committed
371
372
      ///symbol 2  [0:47 ; 192:239] -- 72 mod symbols
    k = frame_parms->first_carrier_offset + frame_parms->ssb_start_subcarrier;
373
    l++;
Guy De Souza's avatar
Guy De Souza committed
374
375
376
    m=180;

    for (int ssb_sc_idx = 0; ssb_sc_idx < 48; ssb_sc_idx++) {
377

Guy De Souza's avatar
Guy De Souza committed
378
379
380
381
382
      if ((ssb_sc_idx&3) == nushift) {
        k++;
        continue;
      }
      else {
383
#ifdef DEBUG_PBCH
Guy De Souza's avatar
Guy De Souza committed
384
  printf("m %d ssb_sc_idx %d at k %d of l %d\n", m, ssb_sc_idx, k, l);
385
#endif
Guy De Souza's avatar
Guy De Souza committed
386
387
388
389
390
391
392
393
394
395
        ((int16_t*)txdataF[aa])[(l*frame_parms->ofdm_symbol_size + k)<<1] = (a * mod_pbch_e[m<<1]) >> 15;
        ((int16_t*)txdataF[aa])[((l*frame_parms->ofdm_symbol_size + k)<<1) + 1] = (a * mod_pbch_e[(m<<1) + 1]) >> 15;
        k++;
        m++;
      }

      if (k >= frame_parms->ofdm_symbol_size)
        k-=frame_parms->ofdm_symbol_size;
    }

Guy De Souza's avatar
Guy De Souza committed
396
397
398
399
    k += 144;
    if (k >= frame_parms->ofdm_symbol_size)
      k-=frame_parms->ofdm_symbol_size;

Guy De Souza's avatar
Guy De Souza committed
400
401
402
403
    m=216;

    for (int ssb_sc_idx = 192; ssb_sc_idx < 240; ssb_sc_idx++) {

Guy De Souza's avatar
Guy De Souza committed
404
      if ((ssb_sc_idx&3) == nushift) {
Guy De Souza's avatar
Guy De Souza committed
405
406
        k++;
        continue;
Guy De Souza's avatar
Guy De Souza committed
407
408
      }
      else {
Guy De Souza's avatar
Guy De Souza committed
409
410
411
#ifdef DEBUG_PBCH
  printf("m %d ssb_sc_idx %d at k %d of l %d\n", m, ssb_sc_idx, k, l);
#endif
Guy De Souza's avatar
Guy De Souza committed
412
413
        ((int16_t*)txdataF[aa])[(l*frame_parms->ofdm_symbol_size + k)<<1] = (a * mod_pbch_e[m<<1]) >> 15;
        ((int16_t*)txdataF[aa])[((l*frame_parms->ofdm_symbol_size + k)<<1) + 1] = (a * mod_pbch_e[(m<<1) + 1]) >> 15;
Guy De Souza's avatar
Guy De Souza committed
414
415
        k++;
        m++;
Guy De Souza's avatar
Guy De Souza committed
416
      }
417
418
419
420
421

      if (k >= frame_parms->ofdm_symbol_size)
        k-=frame_parms->ofdm_symbol_size;
    }

Guy De Souza's avatar
Guy De Souza committed
422
423
      ///symbol 3  [0:239] -- 180 mod symbols
    k = frame_parms->first_carrier_offset + frame_parms->ssb_start_subcarrier;
424
    l++;
Guy De Souza's avatar
Guy De Souza committed
425
426
427
    m=252;

    for (int ssb_sc_idx = 0; ssb_sc_idx < 240; ssb_sc_idx++) {
428

Guy De Souza's avatar
Guy De Souza committed
429
430
      if ((ssb_sc_idx&3) == nushift) {
        k++;
Guy De Souza's avatar
Guy De Souza committed
431
        continue;
Guy De Souza's avatar
Guy De Souza committed
432
433
      }
      else {
Guy De Souza's avatar
Guy De Souza committed
434
435
436
#ifdef DEBUG_PBCH
  printf("m %d ssb_sc_idx %d at k %d of l %d\n", m, ssb_sc_idx, k, l);
#endif
Guy De Souza's avatar
Guy De Souza committed
437
438
439
        ((int16_t*)txdataF[aa])[(l*frame_parms->ofdm_symbol_size + k)<<1] = (a * mod_pbch_e[m<<1]) >> 15;
        ((int16_t*)txdataF[aa])[((l*frame_parms->ofdm_symbol_size + k)<<1) + 1] = (a * mod_pbch_e[(m<<1) + 1]) >> 15;
        k++;
Guy De Souza's avatar
Guy De Souza committed
440
        m++;
Guy De Souza's avatar
Guy De Souza committed
441
      }
442
443
444
445
446
447
448
449
450

      if (k >= frame_parms->ofdm_symbol_size)
        k-=frame_parms->ofdm_symbol_size;
    }

  }


#ifdef DEBUG_PBCH
Guy De Souza's avatar
Guy De Souza committed
451
  write_output("txdataF_pbch.m", "txdataF_pbch", txdataF[0], frame_parms->samples_per_frame_wCP>>1, 1, 1);
452
453
#endif

Guy De Souza's avatar
Guy De Souza committed
454

Guy De Souza's avatar
Guy De Souza committed
455
  return 0;
Guy De Souza's avatar
Guy De Souza committed
456
}