GNU Radio Manual and C++ API Reference  3.7.5.1
The Free & Open Software Radio Ecosystem
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Groups Pages
volk_32fc_x2_multiply_conjugate_32fc.h
Go to the documentation of this file.
1 #ifndef INCLUDED_volk_32fc_x2_multiply_conjugate_32fc_u_H
2 #define INCLUDED_volk_32fc_x2_multiply_conjugate_32fc_u_H
3 
4 #include <inttypes.h>
5 #include <stdio.h>
6 #include <volk/volk_complex.h>
7 #include <float.h>
8 
9 #ifdef LV_HAVE_SSE3
10 #include <pmmintrin.h>
11  /*!
12  \brief Multiplies vector a by the conjugate of vector b and stores the results in the third vector
13  \param cVector The vector where the results will be stored
14  \param aVector First vector to be multiplied
15  \param bVector Second vector that is conjugated before being multiplied
16  \param num_points The number of complex values in aVector and bVector to be multiplied together and stored into cVector
17  */
18 static inline void volk_32fc_x2_multiply_conjugate_32fc_u_sse3(lv_32fc_t* cVector, const lv_32fc_t* aVector, const lv_32fc_t* bVector, unsigned int num_points){
19  unsigned int number = 0;
20  const unsigned int halfPoints = num_points / 2;
21 
22  __m128 x, y, yl, yh, z, tmp1, tmp2;
23  lv_32fc_t* c = cVector;
24  const lv_32fc_t* a = aVector;
25  const lv_32fc_t* b = bVector;
26 
27  __m128 conjugator = _mm_setr_ps(0, -0.f, 0, -0.f);
28 
29  for(;number < halfPoints; number++){
30 
31  x = _mm_loadu_ps((float*)a); // Load the ar + ai, br + bi as ar,ai,br,bi
32  y = _mm_loadu_ps((float*)b); // Load the cr + ci, dr + di as cr,ci,dr,di
33 
34  y = _mm_xor_ps(y, conjugator); // conjugate y
35 
36  yl = _mm_moveldup_ps(y); // Load yl with cr,cr,dr,dr
37  yh = _mm_movehdup_ps(y); // Load yh with ci,ci,di,di
38 
39  tmp1 = _mm_mul_ps(x,yl); // tmp1 = ar*cr,ai*cr,br*dr,bi*dr
40 
41  x = _mm_shuffle_ps(x,x,0xB1); // Re-arrange x to be ai,ar,bi,br
42 
43  tmp2 = _mm_mul_ps(x,yh); // tmp2 = ai*ci,ar*ci,bi*di,br*di
44 
45  z = _mm_addsub_ps(tmp1,tmp2); // ar*cr-ai*ci, ai*cr+ar*ci, br*dr-bi*di, bi*dr+br*di
46 
47  _mm_storeu_ps((float*)c,z); // Store the results back into the C container
48 
49  a += 2;
50  b += 2;
51  c += 2;
52  }
53 
54  if((num_points % 2) != 0) {
55  *c = (*a) * lv_conj(*b);
56  }
57 }
58 #endif /* LV_HAVE_SSE */
59 
60 #ifdef LV_HAVE_GENERIC
61  /*!
62  \brief Multiplies vector a by the conjugate of vector b and stores the results in the third vector
63  \param cVector The vector where the results will be stored
64  \param aVector First vector to be multiplied
65  \param bVector Second vector that is conjugated before being multiplied
66  \param num_points The number of complex values in aVector and bVector to be multiplied together and stored into cVector
67  */
68 static inline void volk_32fc_x2_multiply_conjugate_32fc_generic(lv_32fc_t* cVector, const lv_32fc_t* aVector, const lv_32fc_t* bVector, unsigned int num_points){
69  lv_32fc_t* cPtr = cVector;
70  const lv_32fc_t* aPtr = aVector;
71  const lv_32fc_t* bPtr= bVector;
72  unsigned int number = 0;
73 
74  for(number = 0; number < num_points; number++){
75  *cPtr++ = (*aPtr++) * lv_conj(*bPtr++);
76  }
77 }
78 #endif /* LV_HAVE_GENERIC */
79 
80 
81 #endif /* INCLUDED_volk_32fc_x2_multiply_conjugate_32fc_u_H */
82 #ifndef INCLUDED_volk_32fc_x2_multiply_conjugate_32fc_a_H
83 #define INCLUDED_volk_32fc_x2_multiply_conjugate_32fc_a_H
84 
85 #include <inttypes.h>
86 #include <stdio.h>
87 #include <volk/volk_complex.h>
88 #include <float.h>
89 
90 #ifdef LV_HAVE_SSE3
91 #include <pmmintrin.h>
92  /*!
93  \brief Multiplies vector a by the conjugate of vector b and stores the results in the third vector
94  \param cVector The vector where the results will be stored
95  \param aVector First vector to be multiplied
96  \param bVector Second vector that is conjugated before being multiplied
97  \param num_points The number of complex values in aVector and bVector to be multiplied together and stored into cVector
98  */
99 static inline void volk_32fc_x2_multiply_conjugate_32fc_a_sse3(lv_32fc_t* cVector, const lv_32fc_t* aVector, const lv_32fc_t* bVector, unsigned int num_points){
100  unsigned int number = 0;
101  const unsigned int halfPoints = num_points / 2;
102 
103  __m128 x, y, yl, yh, z, tmp1, tmp2;
104  lv_32fc_t* c = cVector;
105  const lv_32fc_t* a = aVector;
106  const lv_32fc_t* b = bVector;
107 
108  __m128 conjugator = _mm_setr_ps(0, -0.f, 0, -0.f);
109 
110  for(;number < halfPoints; number++){
111 
112  x = _mm_load_ps((float*)a); // Load the ar + ai, br + bi as ar,ai,br,bi
113  y = _mm_load_ps((float*)b); // Load the cr + ci, dr + di as cr,ci,dr,di
114 
115  y = _mm_xor_ps(y, conjugator); // conjugate y
116 
117  yl = _mm_moveldup_ps(y); // Load yl with cr,cr,dr,dr
118  yh = _mm_movehdup_ps(y); // Load yh with ci,ci,di,di
119 
120  tmp1 = _mm_mul_ps(x,yl); // tmp1 = ar*cr,ai*cr,br*dr,bi*dr
121 
122  x = _mm_shuffle_ps(x,x,0xB1); // Re-arrange x to be ai,ar,bi,br
123 
124  tmp2 = _mm_mul_ps(x,yh); // tmp2 = ai*ci,ar*ci,bi*di,br*di
125 
126  z = _mm_addsub_ps(tmp1,tmp2); // ar*cr-ai*ci, ai*cr+ar*ci, br*dr-bi*di, bi*dr+br*di
127 
128  _mm_store_ps((float*)c,z); // Store the results back into the C container
129 
130  a += 2;
131  b += 2;
132  c += 2;
133  }
134 
135  if((num_points % 2) != 0) {
136  *c = (*a) * lv_conj(*b);
137  }
138 }
139 #endif /* LV_HAVE_SSE */
140 
141 #ifdef LV_HAVE_NEON
142 #include <arm_neon.h>
143  /*!
144  \brief Multiplies vector a by the conjugate of vector b and stores the results in the third vector
145  \param cVector The vector where the results will be stored
146  \param aVector First vector to be multiplied
147  \param bVector Second vector that is conjugated before being multiplied
148  \param num_points The number of complex values in aVector and bVector to be multiplied together and stored into cVector
149  */
150 static inline void volk_32fc_x2_multiply_conjugate_32fc_neon(lv_32fc_t* cVector, const lv_32fc_t* aVector, const lv_32fc_t* bVector, unsigned int num_points){
151 
152  lv_32fc_t *a_ptr = (lv_32fc_t*) aVector;
153  lv_32fc_t *b_ptr = (lv_32fc_t*) bVector;
154  unsigned int quarter_points = num_points / 4;
155  float32x4x2_t a_val, b_val, c_val;
156  float32x4x2_t tmp_real, tmp_imag;
157  unsigned int number = 0;
158 
159  for(number = 0; number < quarter_points; ++number) {
160  a_val = vld2q_f32((float*)a_ptr); // a0r|a1r|a2r|a3r || a0i|a1i|a2i|a3i
161  b_val = vld2q_f32((float*)b_ptr); // b0r|b1r|b2r|b3r || b0i|b1i|b2i|b3i
162  b_val.val[1] = vnegq_f32(b_val.val[1]);
163  __builtin_prefetch(a_ptr+4);
164  __builtin_prefetch(b_ptr+4);
165 
166  // multiply the real*real and imag*imag to get real result
167  // a0r*b0r|a1r*b1r|a2r*b2r|a3r*b3r
168  tmp_real.val[0] = vmulq_f32(a_val.val[0], b_val.val[0]);
169  // a0i*b0i|a1i*b1i|a2i*b2i|a3i*b3i
170  tmp_real.val[1] = vmulq_f32(a_val.val[1], b_val.val[1]);
171 
172  // Multiply cross terms to get the imaginary result
173  // a0r*b0i|a1r*b1i|a2r*b2i|a3r*b3i
174  tmp_imag.val[0] = vmulq_f32(a_val.val[0], b_val.val[1]);
175  // a0i*b0r|a1i*b1r|a2i*b2r|a3i*b3r
176  tmp_imag.val[1] = vmulq_f32(a_val.val[1], b_val.val[0]);
177 
178  // store the results
179  c_val.val[0] = vsubq_f32(tmp_real.val[0], tmp_real.val[1]);
180  c_val.val[1] = vaddq_f32(tmp_imag.val[0], tmp_imag.val[1]);
181  vst2q_f32((float*)cVector, c_val);
182 
183  a_ptr += 4;
184  b_ptr += 4;
185  cVector += 4;
186  }
187 
188  for(number = quarter_points*4; number < num_points; number++){
189  *cVector++ = (*a_ptr++) * conj(*b_ptr++);
190  }
191 }
192 #endif /* LV_HAVE_NEON */
193 
194 #ifdef LV_HAVE_GENERIC
195  /*!
196  \brief Multiplies vector a by the conjugate of vector b and stores the results in the third vector
197  \param cVector The vector where the results will be stored
198  \param aVector First vector to be multiplied
199  \param bVector Second vector that is conjugated before being multiplied
200  \param num_points The number of complex values in aVector and bVector to be multiplied together and stored into cVector
201  */
202 static inline void volk_32fc_x2_multiply_conjugate_32fc_a_generic(lv_32fc_t* cVector, const lv_32fc_t* aVector, const lv_32fc_t* bVector, unsigned int num_points){
203  lv_32fc_t* cPtr = cVector;
204  const lv_32fc_t* aPtr = aVector;
205  const lv_32fc_t* bPtr= bVector;
206  unsigned int number = 0;
207 
208  for(number = 0; number < num_points; number++){
209  *cPtr++ = (*aPtr++) * lv_conj(*bPtr++);
210  }
211 }
212 #endif /* LV_HAVE_GENERIC */
213 
214 
215 #endif /* INCLUDED_volk_32fc_x2_multiply_conjugate_32fc_a_H */
#define lv_conj(x)
Definition: volk_complex.h:80
float complex lv_32fc_t
Definition: volk_complex.h:56