nr_dci.c 10.1 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
/*
 * 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 PHY/NR_TRANSPORT/nr_dci.c
23
* \brief Implements DCI encoding/decoding and PDCCH TX/RX procedures (38.212/38.213/38.214). V15.2.0 2018-06.
24
25
26
27
28
29
30
31
32
33
34
* \author Guy De Souza
* \date 2018
* \version 0.1
* \company Eurecom
* \email: desouza@eurecom.fr
* \note
* \warning
*/

#include "nr_dci.h"

35
36
37
38
39
#define DEBUG_PDCCH_DMRS
#define DEBUG_DCI

extern short nr_mod_table[NR_MOD_TABLE_SIZE_SHORT];

yilmazt's avatar
yilmazt committed
40
uint16_t nr_get_dci_size(nfapi_nr_dci_format_e format,
Guy De Souza's avatar
Guy De Souza committed
41
                        nfapi_nr_rnti_type_e rnti_type,
Guy De Souza's avatar
Guy De Souza committed
42
                        NR_BWP_PARMS* bwp,
Guy De Souza's avatar
Guy De Souza committed
43
                        nfapi_nr_config_request_t* config)
44
{
yilmazt's avatar
yilmazt committed
45
  uint16_t size = 0;
46
47
48
  uint16_t N_RB = bwp->N_RB;

  switch(format) {
Guy De Souza's avatar
Guy De Souza committed
49
/*Only sizes for 0_0 and 1_0 are correct at the moment*/
Guy De Souza's avatar
Guy De Souza committed
50
    case NFAPI_NR_UL_DCI_FORMAT_0_0:
Guy De Souza's avatar
Guy De Souza committed
51
52
      /// fixed: Format identifier 1, Hop flag 1, MCS 5, NDI 1, RV 2, HARQ PID 4, PUSCH TPC 2 Time Domain assgnmt 4 --20
      size += 20;
53
      size += (uint8_t)ceil( log2( (N_RB*(N_RB+1))>>1 ) ); // Freq domain assignment -- hopping scenario to be updated
Guy De Souza's avatar
Guy De Souza committed
54
      size += nr_get_dci_size(NFAPI_NR_DL_DCI_FORMAT_1_0, rnti_type, bwp, config) - size; // Padding to match 1_0 size
Guy De Souza's avatar
Guy De Souza committed
55
      // UL/SUL indicator assumed to be 0
56
57
      break;

Guy De Souza's avatar
Guy De Souza committed
58
    case NFAPI_NR_UL_DCI_FORMAT_0_1:
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
      /// fixed: Format identifier 1, MCS 5, NDI 1, RV 2, HARQ PID 4, PUSCH TPC 2, SRS request 2 --17
      size += 17;
      // Carrier indicator
      // UL/SUL indicator
      // BWP Indicator
      // Freq domain assignment
      // Time domain assignment
      // VRB to PRB mapping
      // Frequency Hopping flag
      // 1st DAI
      // 2nd DAI
      // SRS resource indicator
      // Precoding info and number of layers
      // Antenna ports
      // CSI request
      // CBGTI
      // PTRS - DMRS association
      // beta offset indicator
      // DMRS sequence init
      break;

Guy De Souza's avatar
Guy De Souza committed
80
    case NFAPI_NR_DL_DCI_FORMAT_1_0:
Guy De Souza's avatar
Guy De Souza committed
81
82
      /// fixed: Format identifier 1, VRB2PRB 1, MCS 5, NDI 1, RV 2, HARQ PID 4, DAI 2, PUCCH TPC 2, PUCCH RInd 3, PDSCH to HARQ TInd 3 Time Domain assgnmt 4 -- 28
      size += 28;
83
      size += (uint8_t)ceil( log2( (N_RB*(N_RB+1))>>1 ) ); // Freq domain assignment
Guy De Souza's avatar
Guy De Souza committed
84

85
86
      break;

Guy De Souza's avatar
Guy De Souza committed
87
    case NFAPI_NR_DL_DCI_FORMAT_1_1:
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
      // Carrier indicator
      size += 1; // Format identifier
      // BWP Indicator
      // Freq domain assignment
      // Time domain assignment
      // VRB to PRB mapping
      // PRB bundling size indicator
      // Rate matching indicator
      // ZP CSI-RS trigger
        /// TB1- MCS 5, NDI 1, RV 2
      size += 8;
      // TB2
      size += 4 ;  // HARQ PID
      // DAI
      size += 2; // TPC PUCCH
      size += 3; // PUCCH resource indicator
      size += 3; // PDSCH to HARQ timing indicator
      // Antenna ports
      // Tx Config Indication
      size += 2; // SRS request
      // CBGTI
      // CBGFI
      size += 1; // DMRS sequence init
      
      break;

Guy De Souza's avatar
Guy De Souza committed
114
    case NFAPI_NR_DL_DCI_FORMAT_2_0:
115
116
      break;

Guy De Souza's avatar
Guy De Souza committed
117
    case NFAPI_NR_DL_DCI_FORMAT_2_1:
118
119
      break;

Guy De Souza's avatar
Guy De Souza committed
120
    case NFAPI_NR_DL_DCI_FORMAT_2_2:
121
122
      break;

Guy De Souza's avatar
Guy De Souza committed
123
    case NFAPI_NR_DL_DCI_FORMAT_2_3:
124
125
126
127
128
129
130
131
132
      break;

  default:
    AssertFatal(1==0, "Invalid NR DCI format %d\n", format);
  }

  return size;
}

133
134
135
136
void nr_pdcch_scrambling(uint32_t *in,
                         uint8_t size,
                         uint32_t Nid,
                         uint32_t n_RNTI,
137
138
139
140
141
142
143
144
                         uint32_t* out) {

  uint8_t reset;
  uint32_t x1, x2, s=0;

  reset = 1;
  x2 = (n_RNTI<<16) + Nid;

145
  for (int i=0; i<size; i++) {
146
147
148
149
150
    if ((i&0x1f)==0) {
      s = lte_gold_generic(&x1, &x2, reset);
      reset = 0;
    }
    *out ^= (((*in)>>i)&1) ^ ((s>>i)&1);
151
  }
152
153
154

}

155
uint8_t nr_generate_dci_top(NR_gNB_PDCCH pdcch_vars,
156
							t_nrPolar_paramsPtr *nrPolar_params,
157
                            uint32_t *gold_pdcch_dmrs,
Guy De Souza's avatar
Guy De Souza committed
158
159
                            int32_t** txdataF,
                            int16_t amp,
160
                            NR_DL_FRAME_PARMS frame_parms,
161
                            nfapi_nr_config_request_t config)
162
{
163

Guy De Souza's avatar
Guy De Souza committed
164
  uint16_t mod_dmrs[NR_MAX_PDCCH_DMRS_LENGTH>>1];
Guy De Souza's avatar
Guy De Souza committed
165
  uint8_t idx=0;
166
  uint16_t a;
167
  int k,l,k_prime,dci_idx, dmrs_idx;
168
  nr_cce_t cce;
Guy De Souza's avatar
Guy De Souza committed
169

170
171
172
  /*First iteration: single DCI*/
  NR_gNB_DCI_ALLOC_t dci_alloc = pdcch_vars.dci_alloc[0];
  nfapi_nr_dl_config_pdcch_parameters_rel15_t pdcch_params = dci_alloc.pdcch_params;
173

Guy De Souza's avatar
Guy De Souza committed
174
  /// DMRS QPSK modulation
175
176
    /*There is a need to shift from which index the pregenerated DMRS sequence is used
     * see 38211 r15.2.0 section 7.4.1.3.2: assumption is the reference point for k refers to the DMRS sequence*/
177
  if (pdcch_params.config_type == NFAPI_NR_CSET_CONFIG_PDCCH_CONFIG)
178
    gold_pdcch_dmrs += ((int)floor(frame_parms.ssb_start_subcarrier/NR_NB_SC_PER_RB)+pdcch_params.rb_offset)*3/32;
179

180
181
182
183
  for (int i=0; i<NR_MAX_PDCCH_DMRS_LENGTH>>1; i++) {
    idx = ((((gold_pdcch_dmrs[(i<<1)>>5])>>((i<<1)&0x1f))&1)<<1) ^ (((gold_pdcch_dmrs[((i<<1)+1)>>5])>>(((i<<1)+1)&0x1f))&1);
    mod_dmrs[i<<1] = nr_mod_table[(NR_MOD_TABLE_QPSK_OFFSET + idx)<<1];
    mod_dmrs[(i<<1)+1] = nr_mod_table[((NR_MOD_TABLE_QPSK_OFFSET + idx)<<1) + 1];
Guy De Souza's avatar
Guy De Souza committed
184
#ifdef DEBUG_PDCCH_DMRS
185
186
  printf("i %d idx %d gold seq %d b0-b1 %d-%d mod_dmrs %d %d\n", i, idx, gold_pdcch_dmrs[(i<<1)>>5], (((gold_pdcch_dmrs[(i<<1)>>5])>>((i<<1)&0x1f))&1),
  (((gold_pdcch_dmrs[((i<<1)+1)>>5])>>(((i<<1)+1)&0x1f))&1), mod_dmrs[(i<<1)], mod_dmrs[(i<<1)+1]);
187
188
189
#endif
  }

190
  /// DCI payload processing
191
192
193
194
195
196
197
198
199
200
201
202
203
  //channel coding
  uint8_t *encoderInput = malloc(sizeof(uint8_t) * dci_alloc.size);
  nr_bit2byte(dci_alloc.dci_pdu, dci_alloc.size, encoderInput);

  nr_polar_init(&nrPolar_params, NR_POLAR_DCI_MESSAGE_TYPE, dci_alloc.size, pdcch_params.aggregation_level);
  t_nrPolar_paramsPtr currentPtr = nr_polar_params(nrPolar_params,
		  	  	  	  	  	  	  	  	  	  	   NR_POLAR_DCI_MESSAGE_TYPE,
												   dci_alloc.size);

  uint8_t *encoderOutput = malloc(sizeof(uint8_t) * currentPtr->encoderLength);
  polar_encoder(encoderInput, encoderOutput, currentPtr);
  uint32_t encoded_payload[4];
  nr_byte2bit(encoderOutput,currentPtr->encoderLength,encoded_payload);
204
  
205
    // scrambling
206
  uint32_t scrambled_payload[4];
Guy De Souza's avatar
Guy De Souza committed
207
208
  uint32_t Nid = (pdcch_params.search_space_type == NFAPI_NR_SEARCH_SPACE_TYPE_UE_SPECIFIC)? pdcch_params.scrambling_id : config.sch_config.physical_cell_id.value;
  uint32_t n_RNTI = (pdcch_params.search_space_type == NFAPI_NR_SEARCH_SPACE_TYPE_UE_SPECIFIC)? pdcch_params.rnti : 0;
209
  nr_pdcch_scrambling(encoded_payload, dci_alloc.size, Nid, n_RNTI, scrambled_payload);
210
211

    // QPSK modulation
212
213
214
215
216
  uint32_t mod_dci[NR_MAX_DCI_SIZE>>1];
  for (int i=0; i<dci_alloc.size>>1; i++) {
    idx = (((scrambled_payload[i<<1]>>(i<<1))&1)<<1) ^ ((scrambled_payload[(i<<1)+1]>>((i<<1)+1))&1);
    mod_dci[i<<1] = nr_mod_table[(NR_MOD_TABLE_QPSK_OFFSET + idx)<<1];
    mod_dci[(i<<1)+1] = nr_mod_table[((NR_MOD_TABLE_QPSK_OFFSET + idx)<<1) + 1];
Guy De Souza's avatar
Guy De Souza committed
217
#ifdef DEBUG_DCI
Guy De Souza's avatar
Guy De Souza committed
218
  printf("i %d idx %d b0-b1 %d-%d mod_dci %d %d\n", i, idx, (((scrambled_payload[(i<<1)>>5])>>((i<<1)&0x1f))&1),
Guy De Souza's avatar
Guy De Souza committed
219
220
  (((scrambled_payload[((i<<1)+1)>>5])>>(((i<<1)+1)&0x1f))&1), mod_dci[(i<<1)], mod_dci[(i<<1)+1]);
#endif
221
222
223
224
225
226
227
228
  }

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

   /*The coreset is initialised
    * in frequency: the first subcarrier is obtained by adding the first CRB overlapping the SSB and the rb_offset
    * in time: by its first slot and its first symbol*/
229
230
  uint8_t cset_start_sc = frame_parms.first_carrier_offset + ((int)floor(frame_parms.ssb_start_subcarrier/NR_NB_SC_PER_RB)+pdcch_params.rb_offset)*NR_NB_SC_PER_RB;
  uint8_t cset_start_symb = pdcch_params.first_slot*frame_parms.symbols_per_slot + pdcch_params.first_symbol;
231
232
  dci_idx = 0;
  dmrs_idx = 0;
233
234
235
236
237
238

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

239
    if (pdcch_params.precoder_granularity == NFAPI_NR_CSET_SAME_AS_REG_BUNDLE) {
240
241

      for (int cce_idx=0; cce_idx<dci_alloc.L; cce_idx++){
242
        cce = dci_alloc.cce_list[cce_idx];
243
          for (int reg_idx=0; reg_idx<NR_NB_REG_PER_CCE; reg_idx++) {
244
            k = cset_start_sc + cce.reg_list[reg_idx].start_sc_idx;
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
            l = cset_start_symb + cce.reg_list[reg_idx].symb_idx;
            k_prime = 0;
            for (int m=0; m<NR_NB_SC_PER_RB; m++) {
              if ( m == (k_prime<<2)+1) { // DMRS
                ((int16_t*)txdataF[aa])[(l*frame_parms.ofdm_symbol_size + k)<<1] = (a * mod_dmrs[dmrs_idx<<1]) >> 15;
                ((int16_t*)txdataF[aa])[((l*frame_parms.ofdm_symbol_size + k)<<1) + 1] = (a * mod_dmrs[(dmrs_idx<<1) + 1]) >> 15;
                k_prime++;
                dmrs_idx++;
              }
              else { // DCI payload
                ((int16_t*)txdataF[aa])[(l*frame_parms.ofdm_symbol_size + k)<<1] = (a * mod_dci[dci_idx<<1]) >> 15;
                ((int16_t*)txdataF[aa])[((l*frame_parms.ofdm_symbol_size + k)<<1) + 1] = (a * mod_dci[(dci_idx<<1) + 1]) >> 15;
                dci_idx++;
              }
              k++;
              if (k >= frame_parms.ofdm_symbol_size)
                k -= frame_parms.ofdm_symbol_size;
            }
        }
264
265
266
      }
    }

267
    else { //NFAPI_NR_CSET_ALL_CONTIGUOUS_RBS
268
269
    }

270
  }
271

272
273
  return 0;
}