annotate gcc/vec-perm-indices.h @ 131:84e7813d76e9

gcc-8.2
author mir3636
date Thu, 25 Oct 2018 07:37:49 +0900
parents
children 1830386684a0
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
131
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
1 /* A representation of vector permutation indices.
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
2 Copyright (C) 2017-2018 Free Software Foundation, Inc.
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
3
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
4 This file is part of GCC.
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
5
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
6 GCC is free software; you can redistribute it and/or modify it under
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
7 the terms of the GNU General Public License as published by the Free
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
8 Software Foundation; either version 3, or (at your option) any later
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
9 version.
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
10
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
11 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
12 WARRANTY; without even the implied warranty of MERCHANTABILITY or
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
13 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
14 for more details.
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
15
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
16 You should have received a copy of the GNU General Public License
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
17 along with GCC; see the file COPYING3. If not see
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
18 <http://www.gnu.org/licenses/>. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
19
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
20 #ifndef GCC_VEC_PERN_INDICES_H
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
21 #define GCC_VEC_PERN_INDICES_H 1
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
22
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
23 #include "int-vector-builder.h"
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
24
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
25 /* A vector_builder for building constant permutation vectors.
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
26 The elements do not need to be clamped to a particular range
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
27 of input elements. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
28 typedef int_vector_builder<poly_int64> vec_perm_builder;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
29
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
30 /* This class represents a constant permutation vector, such as that used
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
31 as the final operand to a VEC_PERM_EXPR.
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
32
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
33 Permutation vectors select indices modulo the number of input elements,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
34 and the class canonicalizes each permutation vector for a particular
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
35 number of input vectors and for a particular number of elements per
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
36 input. For example, the gimple statements:
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
37
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
38 _1 = VEC_PERM_EXPR <a, a, { 0, 2, 4, 6, 0, 2, 4, 6 }>;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
39 _2 = VEC_PERM_EXPR <a, a, { 0, 2, 4, 6, 8, 10, 12, 14 }>;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
40 _3 = VEC_PERM_EXPR <a, a, { 0, 2, 20, 22, 24, 2, 4, 14 }>;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
41
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
42 effectively have only a single vector input "a". If "a" has 8
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
43 elements, the indices select elements modulo 8, which makes all three
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
44 VEC_PERM_EXPRs equivalent. The canonical form is for the indices to be
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
45 in the range [0, number of input elements - 1], so the class treats the
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
46 second and third permutation vectors as though they had been the first.
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
47
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
48 The class copes with cases in which the input and output vectors have
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
49 different numbers of elements. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
50 class vec_perm_indices
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
51 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
52 typedef poly_int64 element_type;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
53
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
54 public:
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
55 vec_perm_indices ();
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
56 vec_perm_indices (const vec_perm_builder &, unsigned int, poly_uint64);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
57
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
58 void new_vector (const vec_perm_builder &, unsigned int, poly_uint64);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
59 void new_expanded_vector (const vec_perm_indices &, unsigned int);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
60 void rotate_inputs (int delta);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
61
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
62 /* Return the underlying vector encoding. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
63 const vec_perm_builder &encoding () const { return m_encoding; }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
64
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
65 /* Return the number of output elements. This is called length ()
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
66 so that we present a more vec-like interface. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
67 poly_uint64 length () const { return m_encoding.full_nelts (); }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
68
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
69 /* Return the number of input vectors being permuted. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
70 unsigned int ninputs () const { return m_ninputs; }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
71
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
72 /* Return the number of elements in each input vector. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
73 poly_uint64 nelts_per_input () const { return m_nelts_per_input; }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
74
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
75 /* Return the total number of input elements. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
76 poly_uint64 input_nelts () const { return m_ninputs * m_nelts_per_input; }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
77
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
78 element_type clamp (element_type) const;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
79 element_type operator[] (unsigned int i) const;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
80 bool series_p (unsigned int, unsigned int, element_type, element_type) const;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
81 bool all_in_range_p (element_type, element_type) const;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
82 bool all_from_input_p (unsigned int) const;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
83
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
84 private:
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
85 vec_perm_indices (const vec_perm_indices &);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
86
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
87 vec_perm_builder m_encoding;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
88 unsigned int m_ninputs;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
89 poly_uint64 m_nelts_per_input;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
90 };
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
91
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
92 bool tree_to_vec_perm_builder (vec_perm_builder *, tree);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
93 tree vec_perm_indices_to_tree (tree, const vec_perm_indices &);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
94 rtx vec_perm_indices_to_rtx (machine_mode, const vec_perm_indices &);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
95
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
96 inline
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
97 vec_perm_indices::vec_perm_indices ()
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
98 : m_ninputs (0),
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
99 m_nelts_per_input (0)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
100 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
101 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
102
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
103 /* Construct a permutation vector that selects between NINPUTS vector
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
104 inputs that have NELTS_PER_INPUT elements each. Take the elements of
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
105 the new vector from ELEMENTS, clamping each one to be in range. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
106
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
107 inline
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
108 vec_perm_indices::vec_perm_indices (const vec_perm_builder &elements,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
109 unsigned int ninputs,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
110 poly_uint64 nelts_per_input)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
111 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
112 new_vector (elements, ninputs, nelts_per_input);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
113 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
114
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
115 /* Return the canonical value for permutation vector element ELT,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
116 taking into account the current number of input elements. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
117
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
118 inline vec_perm_indices::element_type
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
119 vec_perm_indices::clamp (element_type elt) const
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
120 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
121 element_type limit = input_nelts (), elem_within_input;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
122 HOST_WIDE_INT input;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
123 if (!can_div_trunc_p (elt, limit, &input, &elem_within_input))
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
124 return elt;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
125
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
126 /* Treat negative elements as counting from the end. This only matters
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
127 if the vector size is not a power of 2. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
128 if (known_lt (elem_within_input, 0))
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
129 return elem_within_input + limit;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
130
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
131 return elem_within_input;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
132 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
133
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
134 /* Return the value of vector element I, which might or might not be
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
135 explicitly encoded. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
136
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
137 inline vec_perm_indices::element_type
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
138 vec_perm_indices::operator[] (unsigned int i) const
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
139 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
140 return clamp (m_encoding.elt (i));
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
141 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
142
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
143 /* Return true if the permutation vector only selects elements from
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
144 input I. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
145
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
146 inline bool
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
147 vec_perm_indices::all_from_input_p (unsigned int i) const
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
148 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
149 return all_in_range_p (i * m_nelts_per_input, m_nelts_per_input);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
150 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
151
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
152 #endif