GNU Radio Manual and C++ API Reference  3.7.6.1
The Free & Open Software Radio Ecosystem
volk_16i_permute_and_scalar_add.h
Go to the documentation of this file.
1 /* -*- c++ -*- */
2 /*
3  * Copyright 2014 Free Software Foundation, Inc.
4  *
5  * This file is part of GNU Radio
6  *
7  * GNU Radio is free software; you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation; either version 3, or (at your option)
10  * any later version.
11  *
12  * GNU Radio is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with GNU Radio; see the file COPYING. If not, write to
19  * the Free Software Foundation, Inc., 51 Franklin Street,
20  * Boston, MA 02110-1301, USA.
21  */
22 
23 #ifndef INCLUDED_volk_16i_permute_and_scalar_add_a_H
24 #define INCLUDED_volk_16i_permute_and_scalar_add_a_H
25 
26 
27 #include<inttypes.h>
28 #include<stdio.h>
29 
30 
31 
32 
33 #ifdef LV_HAVE_SSE2
34 
35 #include<xmmintrin.h>
36 #include<emmintrin.h>
37 
38 static inline void volk_16i_permute_and_scalar_add_a_sse2(short* target, short* src0, short* permute_indexes, short* cntl0, short* cntl1, short* cntl2, short* cntl3, short* scalars, unsigned int num_points) {
39 
40  const unsigned int num_bytes = num_points*2;
41 
42  __m128i xmm0, xmm1, xmm2, xmm3, xmm4, xmm5, xmm6, xmm7;
43 
44  __m128i *p_target, *p_cntl0, *p_cntl1, *p_cntl2, *p_cntl3, *p_scalars;
45 
46  short* p_permute_indexes = permute_indexes;
47 
48  p_target = (__m128i*)target;
49  p_cntl0 = (__m128i*)cntl0;
50  p_cntl1 = (__m128i*)cntl1;
51  p_cntl2 = (__m128i*)cntl2;
52  p_cntl3 = (__m128i*)cntl3;
53  p_scalars = (__m128i*)scalars;
54 
55  int i = 0;
56 
57  int bound = (num_bytes >> 4);
58  int leftovers = (num_bytes >> 1) & 7;
59 
60  xmm0 = _mm_load_si128(p_scalars);
61 
62  xmm1 = _mm_shufflelo_epi16(xmm0, 0);
63  xmm2 = _mm_shufflelo_epi16(xmm0, 0x55);
64  xmm3 = _mm_shufflelo_epi16(xmm0, 0xaa);
65  xmm4 = _mm_shufflelo_epi16(xmm0, 0xff);
66 
67  xmm1 = _mm_shuffle_epi32(xmm1, 0x00);
68  xmm2 = _mm_shuffle_epi32(xmm2, 0x00);
69  xmm3 = _mm_shuffle_epi32(xmm3, 0x00);
70  xmm4 = _mm_shuffle_epi32(xmm4, 0x00);
71 
72 
73  for(; i < bound; ++i) {
74  xmm0 = _mm_setzero_si128();
75  xmm5 = _mm_setzero_si128();
76  xmm6 = _mm_setzero_si128();
77  xmm7 = _mm_setzero_si128();
78 
79  xmm0 = _mm_insert_epi16(xmm0, src0[p_permute_indexes[0]], 0);
80  xmm5 = _mm_insert_epi16(xmm5, src0[p_permute_indexes[1]], 1);
81  xmm6 = _mm_insert_epi16(xmm6, src0[p_permute_indexes[2]], 2);
82  xmm7 = _mm_insert_epi16(xmm7, src0[p_permute_indexes[3]], 3);
83  xmm0 = _mm_insert_epi16(xmm0, src0[p_permute_indexes[4]], 4);
84  xmm5 = _mm_insert_epi16(xmm5, src0[p_permute_indexes[5]], 5);
85  xmm6 = _mm_insert_epi16(xmm6, src0[p_permute_indexes[6]], 6);
86  xmm7 = _mm_insert_epi16(xmm7, src0[p_permute_indexes[7]], 7);
87 
88  xmm0 = _mm_add_epi16(xmm0, xmm5);
89  xmm6 = _mm_add_epi16(xmm6, xmm7);
90 
91  p_permute_indexes += 8;
92 
93  xmm0 = _mm_add_epi16(xmm0, xmm6);
94 
95  xmm5 = _mm_load_si128(p_cntl0);
96  xmm6 = _mm_load_si128(p_cntl1);
97  xmm7 = _mm_load_si128(p_cntl2);
98 
99  xmm5 = _mm_and_si128(xmm5, xmm1);
100  xmm6 = _mm_and_si128(xmm6, xmm2);
101  xmm7 = _mm_and_si128(xmm7, xmm3);
102 
103  xmm0 = _mm_add_epi16(xmm0, xmm5);
104 
105  xmm5 = _mm_load_si128(p_cntl3);
106 
107  xmm6 = _mm_add_epi16(xmm6, xmm7);
108 
109  p_cntl0 += 1;
110 
111  xmm5 = _mm_and_si128(xmm5, xmm4);
112 
113  xmm0 = _mm_add_epi16(xmm0, xmm6);
114 
115  p_cntl1 += 1;
116  p_cntl2 += 1;
117 
118  xmm0 = _mm_add_epi16(xmm0, xmm5);
119 
120  p_cntl3 += 1;
121 
122  _mm_store_si128(p_target, xmm0);
123 
124  p_target += 1;
125  }
126 
127 
128 
129 
130 
131  for(i = bound * 8; i < (bound * 8) + leftovers; ++i) {
132  target[i] = src0[permute_indexes[i]]
133  + (cntl0[i] & scalars[0])
134  + (cntl1[i] & scalars[1])
135  + (cntl2[i] & scalars[2])
136  + (cntl3[i] & scalars[3]);
137  }
138 }
139 #endif /*LV_HAVE_SSEs*/
140 
141 
142 #ifdef LV_HAVE_GENERIC
143 static inline void volk_16i_permute_and_scalar_add_generic(short* target, short* src0, short* permute_indexes, short* cntl0, short* cntl1, short* cntl2, short* cntl3, short* scalars, unsigned int num_points) {
144 
145  const unsigned int num_bytes = num_points*2;
146 
147  int i = 0;
148 
149  int bound = num_bytes >> 1;
150 
151  for(i = 0; i < bound; ++i) {
152  target[i] = src0[permute_indexes[i]]
153  + (cntl0[i] & scalars[0])
154  + (cntl1[i] & scalars[1])
155  + (cntl2[i] & scalars[2])
156  + (cntl3[i] & scalars[3]);
157 
158  }
159 }
160 
161 #endif /*LV_HAVE_GENERIC*/
162 
163 
164 #endif /*INCLUDED_volk_16i_permute_and_scalar_add_a_H*/