|
|
@ -566,17 +566,23 @@ int srslte_prach_detect(srslte_prach_t* p,
|
|
|
|
{
|
|
|
|
{
|
|
|
|
return srslte_prach_detect_offset(p, freq_offset, signal, sig_len, indices, NULL, NULL, n_indices);
|
|
|
|
return srslte_prach_detect_offset(p, freq_offset, signal, sig_len, indices, NULL, NULL, n_indices);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// this function subtracts the detected prach preamble from the signal so as to allow for lower power prach signals to
|
|
|
|
|
|
|
|
/// be detected more easily in the subsequent searches
|
|
|
|
void srslte_prach_cancellation (srslte_prach_t* p, cf_t *signal, uint32_t begin, int sig_len, srslte_prach_cancellation_t prach_cancel)
|
|
|
|
void srslte_prach_cancellation (srslte_prach_t* p, cf_t *signal, uint32_t begin, int sig_len, srslte_prach_cancellation_t prach_cancel)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
cf_t sub[sig_len];
|
|
|
|
cf_t sub[sig_len];
|
|
|
|
memcpy(sub,&p->td_signals[p->root_seqs_idx[prach_cancel.idx]][p->N_cp], sig_len*sizeof(cf_t));
|
|
|
|
memcpy(sub,&p->td_signals[p->root_seqs_idx[prach_cancel.idx]][p->N_cp], sig_len*sizeof(cf_t));
|
|
|
|
srslte_vec_sc_prod_cfc(sub, prach_cancel.factor, sub, p->N_seq);
|
|
|
|
srslte_vec_sc_prod_cfc(sub, prach_cancel.factor, sub, p->N_seq);
|
|
|
|
int offset = (int) (prach_cancel.offset*sig_len*DELTA_F_RA);
|
|
|
|
int offset = (int) (prach_cancel.offset*sig_len*DELTA_F_RA);
|
|
|
|
|
|
|
|
srslte_vec_sc_prod_ccc(sub, cexpf(_Complex_I * prach_cancel.phase), sub, sig_len);
|
|
|
|
srslte_vec_sub_ccc(&signal[offset], sub, &signal[offset], sig_len);
|
|
|
|
srslte_vec_sub_ccc(&signal[offset], sub, &signal[offset], sig_len);
|
|
|
|
srslte_dft_run(&p->fft, signal, p->signal_fft);
|
|
|
|
srslte_dft_run(&p->fft, signal, p->signal_fft);
|
|
|
|
memcpy(p->prach_bins, &p->signal_fft[begin], p->N_zc * sizeof(cf_t));
|
|
|
|
memcpy(p->prach_bins, &p->signal_fft[begin], p->N_zc * sizeof(cf_t));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// this function checks if we have already detected and stored this particular PRACH index and if so, doesnt store it
|
|
|
|
|
|
|
|
// again in the detected prachs array
|
|
|
|
bool srslte_prach_have_stored(srslte_prach_t* p,int current_idx, uint32_t* indices, uint32_t n_indices) {
|
|
|
|
bool srslte_prach_have_stored(srslte_prach_t* p,int current_idx, uint32_t* indices, uint32_t n_indices) {
|
|
|
|
for(int i = 0; i < n_indices; i++) {
|
|
|
|
for(int i = 0; i < n_indices; i++) {
|
|
|
|
if (indices[i] == current_idx) {
|
|
|
|
if (indices[i] == current_idx) {
|
|
|
@ -597,41 +603,21 @@ float srslte_prach_set_offset(srslte_prach_t* p, int n_win) {
|
|
|
|
return corr * p->peak_offsets[n_win] / (DELTA_F_RA * p->N_zc);
|
|
|
|
return corr * p->peak_offsets[n_win] / (DELTA_F_RA * p->N_zc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int srslte_prach_detect_offset(srslte_prach_t* p,
|
|
|
|
// This function carries out the main processing on the incomming PRACH signal
|
|
|
|
uint32_t freq_offset,
|
|
|
|
int srslte_prach_process(srslte_prach_t* p,
|
|
|
|
cf_t* signal,
|
|
|
|
cf_t* signal,
|
|
|
|
uint32_t sig_len,
|
|
|
|
|
|
|
|
uint32_t* indices,
|
|
|
|
uint32_t* indices,
|
|
|
|
float* t_offsets,
|
|
|
|
float* t_offsets,
|
|
|
|
float* peak_to_avg,
|
|
|
|
float* peak_to_avg,
|
|
|
|
uint32_t* n_indices)
|
|
|
|
uint32_t* n_indices,
|
|
|
|
|
|
|
|
int cancellation_idx,
|
|
|
|
|
|
|
|
srslte_prach_cancellation_t prach_cancel,
|
|
|
|
|
|
|
|
uint32_t begin,
|
|
|
|
|
|
|
|
uint32_t sig_len)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
int ret = SRSLTE_ERROR;
|
|
|
|
|
|
|
|
if (p != NULL && signal != NULL && sig_len > 0 && indices != NULL) {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (sig_len < p->N_ifft_prach) {
|
|
|
|
|
|
|
|
ERROR("srslte_prach_detect: Signal length is %d and should be %d\n", sig_len, p->N_ifft_prach);
|
|
|
|
|
|
|
|
return SRSLTE_ERROR_INVALID_INPUTS;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
int cancellation_idx = -2;
|
|
|
|
|
|
|
|
srslte_prach_cancellation_t prach_cancel;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// FFT incoming signal
|
|
|
|
|
|
|
|
srslte_dft_run(&p->fft, signal, p->signal_fft);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*n_indices = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Extract bins of interest
|
|
|
|
|
|
|
|
uint32_t N_rb_ul = srslte_nof_prb(p->N_ifft_ul);
|
|
|
|
|
|
|
|
uint32_t k_0 = freq_offset * N_RB_SC - N_rb_ul * N_RB_SC / 2 + p->N_ifft_ul / 2;
|
|
|
|
|
|
|
|
uint32_t K = DELTA_F / DELTA_F_RA;
|
|
|
|
|
|
|
|
uint32_t begin = PHI + (K * k_0) + (K / 2);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
memcpy(p->prach_bins, &p->signal_fft[begin], p->N_zc * sizeof(cf_t));
|
|
|
|
|
|
|
|
int loops = (p->successive_cancellation)?p->num_ra_preambles:1;
|
|
|
|
|
|
|
|
for (int l = 0; l < loops; l++) {
|
|
|
|
|
|
|
|
float max_to_cancel = 0;
|
|
|
|
float max_to_cancel = 0;
|
|
|
|
cancellation_idx = -1;
|
|
|
|
cancellation_idx = -1;
|
|
|
|
|
|
|
|
int max_idx = 0;
|
|
|
|
for (int i = 0; i < p->num_ra_preambles; i++) {
|
|
|
|
for (int i = 0; i < p->num_ra_preambles; i++) {
|
|
|
|
cf_t* root_spec = p->dft_seqs[p->root_seqs_idx[i]];
|
|
|
|
cf_t* root_spec = p->dft_seqs[p->root_seqs_idx[i]];
|
|
|
|
|
|
|
|
|
|
|
@ -666,6 +652,7 @@ int srslte_prach_detect_offset(srslte_prach_t* p,
|
|
|
|
p->peak_offsets[j] = k - start;
|
|
|
|
p->peak_offsets[j] = k - start;
|
|
|
|
if (p->peak_values[j] > max_peak) {
|
|
|
|
if (p->peak_values[j] > max_peak) {
|
|
|
|
max_peak = p->peak_values[j];
|
|
|
|
max_peak = p->peak_values[j];
|
|
|
|
|
|
|
|
max_idx = k;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -679,20 +666,24 @@ int srslte_prach_detect_offset(srslte_prach_t* p,
|
|
|
|
cancellation_idx = (i * n_wins) + j;
|
|
|
|
cancellation_idx = (i * n_wins) + j;
|
|
|
|
max_to_cancel = max_peak;
|
|
|
|
max_to_cancel = max_peak;
|
|
|
|
prach_cancel.idx = cancellation_idx;
|
|
|
|
prach_cancel.idx = cancellation_idx;
|
|
|
|
prach_cancel.offset = srslte_prach_set_offset(p,j);
|
|
|
|
prach_cancel.offset = srslte_prach_set_offset(p, j);
|
|
|
|
prach_cancel.factor = sqrt((max_peak/2)/((sig_len/2)*p->N_zc*p->N_zc));
|
|
|
|
prach_cancel.factor = sqrt((max_peak / 2) / ((sig_len / 2) * p->N_zc * p->N_zc));
|
|
|
|
|
|
|
|
prach_cancel.phase = cargf(p->corr_spec[max_idx]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (srslte_prach_have_stored(p,((i * n_wins) + j),indices, *n_indices)) {
|
|
|
|
if (srslte_prach_have_stored(p, ((i * n_wins) + j), indices, *n_indices)) {
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
srslte_vec_fprint_c(stdout, p->corr_spec, 10) printf("max_idx %d\n", max_idx);
|
|
|
|
|
|
|
|
printf("cargf(p->corr_spec[max_idx]) %f\n", cargf(p->corr_spec[max_idx]));
|
|
|
|
indices[*n_indices] = (i * n_wins) + j;
|
|
|
|
indices[*n_indices] = (i * n_wins) + j;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (peak_to_avg) {
|
|
|
|
if (peak_to_avg) {
|
|
|
|
peak_to_avg[*n_indices] = p->peak_values[j] / corr_ave;
|
|
|
|
peak_to_avg[*n_indices] = p->peak_values[j] / corr_ave;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (t_offsets) {
|
|
|
|
if (t_offsets) {
|
|
|
|
t_offsets[*n_indices] = srslte_prach_set_offset(p,j);
|
|
|
|
t_offsets[*n_indices] = srslte_prach_set_offset(p, j);
|
|
|
|
|
|
|
|
printf("t_offsets[*n_indices] %f\n", t_offsets[*n_indices]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(*n_indices)++;
|
|
|
|
(*n_indices)++;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -702,6 +693,48 @@ int srslte_prach_detect_offset(srslte_prach_t* p,
|
|
|
|
if (cancellation_idx != -1) {
|
|
|
|
if (cancellation_idx != -1) {
|
|
|
|
srslte_prach_cancellation(p, signal, begin, sig_len, prach_cancel);
|
|
|
|
srslte_prach_cancellation(p, signal, begin, sig_len, prach_cancel);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int srslte_prach_detect_offset(srslte_prach_t* p,
|
|
|
|
|
|
|
|
uint32_t freq_offset,
|
|
|
|
|
|
|
|
cf_t* signal,
|
|
|
|
|
|
|
|
uint32_t sig_len,
|
|
|
|
|
|
|
|
uint32_t* indices,
|
|
|
|
|
|
|
|
float* t_offsets,
|
|
|
|
|
|
|
|
float* peak_to_avg,
|
|
|
|
|
|
|
|
uint32_t* n_indices)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
int ret = SRSLTE_ERROR;
|
|
|
|
|
|
|
|
if (p != NULL && signal != NULL && sig_len > 0 && indices != NULL) {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (sig_len < p->N_ifft_prach) {
|
|
|
|
|
|
|
|
ERROR("srslte_prach_detect: Signal length is %d and should be %d\n", sig_len, p->N_ifft_prach);
|
|
|
|
|
|
|
|
return SRSLTE_ERROR_INVALID_INPUTS;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
int cancellation_idx = -2;
|
|
|
|
|
|
|
|
srslte_prach_cancellation_t prach_cancel = {};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// FFT incoming signal
|
|
|
|
|
|
|
|
srslte_dft_run(&p->fft, signal, p->signal_fft);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*n_indices = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Extract bins of interest
|
|
|
|
|
|
|
|
uint32_t N_rb_ul = srslte_nof_prb(p->N_ifft_ul);
|
|
|
|
|
|
|
|
uint32_t k_0 = freq_offset * N_RB_SC - N_rb_ul * N_RB_SC / 2 + p->N_ifft_ul / 2;
|
|
|
|
|
|
|
|
uint32_t K = DELTA_F / DELTA_F_RA;
|
|
|
|
|
|
|
|
uint32_t begin = PHI + (K * k_0) + (K / 2);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
memcpy(p->prach_bins, &p->signal_fft[begin], p->N_zc * sizeof(cf_t));
|
|
|
|
|
|
|
|
int loops = (p->successive_cancellation) ? p->num_ra_preambles : 1;
|
|
|
|
|
|
|
|
// if successive cancellation is enabled, we perform the entire search process p->num_ra_preambles times, removing
|
|
|
|
|
|
|
|
// the highest power PRACH preamble each time.
|
|
|
|
|
|
|
|
for (int l = 0; l < loops; l++) {
|
|
|
|
|
|
|
|
if (srslte_prach_process(
|
|
|
|
|
|
|
|
p, signal, indices, t_offsets, peak_to_avg, n_indices, cancellation_idx, prach_cancel, begin, sig_len)) {
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|