-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathperm_utils.hpp
309 lines (251 loc) · 8.82 KB
/
perm_utils.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
#ifndef _PERM_UTILS_HPP_
#define _PERM_UTILS_HPP_
/** Utility methods for dealing with PermLib's permutation types.
*
* @author Aaron Moss
*/
/* Copyright: Aaron Moss, 2012, [email protected] */
/* This file is part of Basil.
Basil is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as
published by the Free Software Foundation, either version 3 of
the License, or (at your option) any later version.
Basil is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with Basil. If not, see <http://www.gnu.org/licenses/>. */
#include <set>
#include <sstream>
#include <vector>
#include "basil.hpp"
#include "lrs/cobasis.hpp"
#include "permlib/permlib_api.h"
namespace basil {
/** a representation of a permutation cycle */
typedef std::vector<permlib::dom_int> permutation_cycle;
/** a list of the cycles of a permutation */
typedef std::vector<permutation_cycle> permutation_cycle_list;
/** Extracts the list of cycles from a permutation
* @param p The permutation to extract the cycles from
* @return A representation fo p as a list of disjoint cycles
*/
static permutation_cycle_list cycle_list(permutation const& p) {
typedef permlib::dom_int dom_int;
permutation_cycle_list cycles;
/* set of group elements that have been added to their cycle */
std::set<dom_int> worked;
for (dom_int x = 0; x < p.size(); ++x) {
dom_int px, startX;
startX = x; /* the start of the cycle */
px = p/x; /* the permutation applied to x */
/* if x has already been found, or is not moved, ignore */
if (worked.count(x) || x == px) {
continue;
}
/*start the cycle */
permutation_cycle c;
worked.insert(x);
c.push_back(x);
/* while the cycle is not complete, follow it */
while ( p/px != startX ) {
worked.insert(px);
c.push_back(px);
px = p/px;
}
/* finish the cycle */
worked.insert(px);
c.push_back(px);
cycles.push_back(c);
}
return cycles;
}
/** Reconstitutes a permutation from a list of cycles.
* @param n The number of elements the permutation acts on
* @param l The list of cycles defining the permutation
*/
static permutation perm(uind n, permutation_cycle_list const& l) {
typedef permlib::dom_int dom_int;
dom_int p_i[n];
for (uind i = 0; i < n; ++i) p_i[i] = i;
for (permutation_cycle_list::const_iterator iter = l.begin();
iter != l.end(); ++iter) {
permutation_cycle::const_iterator eIter = iter->begin();
dom_int startX = *eIter;
dom_int last = startX;
++eIter;
while ( eIter != iter->end() ) {
dom_int next = *eIter;
p_i[last] = next;
last = next;
++eIter;
}
p_i[last] = startX;
}
return permutation(p_i, p_i+n);
}
/** Converts a permutation to a string in a cycle representation compatible
* with its input format. Derived from standard printing code in PermLib
* @param p The perumation to print
* @return A representation of p as comma-delimited cycles of whitespace-
* delimited elements, where the cycles are ordered by minimum
* element
*/
static string in_str(permutation const& p) {
std::ostringstream o;
permutation_cycle_list l = cycle_list(p);
bool isFirst = true;
for (permutation_cycle_list::iterator iter = l.begin();
iter != l.end(); ++iter) {
if ( isFirst ) isFirst = false; else o << " ,";
for (permutation_cycle::iterator iter2 = iter->begin();
iter2 != iter->end(); ++iter2) {
o << " " << (*iter2)+1;
}
}
return o.str();
}
/** Gets the strong generating set for a permutation group.
* @param g The permutation group
* @return The strong generating set of generators for the group, as a
* list of pointers to permutations.
*/
static permutation_list strong_gen_set(permutation_group const& g) {
return g.S;
}
/** Gets the strong generating set for a permutation group.
* @param g The permutation group
* @return The strong generating set of generators for the group, as a
* list of pointers to permutations.
*/
static permutation_list& strong_gen_set(permutation_group& g) {
return g.S;
}
/** Gets a small set of generators for a permutation group.
* @param g The permutation group
* @return A hopefully small list of generators for the group, as a list
* of pointers to permutations.
*/
static permutation_list small_gen_set(permutation_group& g) {
typedef permutation_list::iterator perm_iter;
perm_iter iter = g.S.begin();
permutation_list gens;
if ( iter == g.S.end() ) return gens;
/* target order */
uint64_t ord = g.order();
/* first, sort through and find out which generators are essential */
permutation_list gsgs(g.S);
permutation_list opts;
iter = gsgs.begin();
perm_iter next_iter = iter; ++next_iter;
permutation_ptr p;
permutation_group_ptr gn;
while ( iter != gsgs.end() ) {
/* remove a generator from the generator list */
p = *iter;
gsgs.erase(iter);
/* construct a new group without it */
gn = permlib::construct(g.n, gsgs.begin(), gsgs.end());
/* test group order, keeping any permutation that lowers it */
if ( gn->order() < ord ) gens.push_back(p); else opts.push_back(p);
/* put permutation back in list regardless */
gsgs.insert(next_iter, p);
iter = next_iter; ++next_iter;
}
/* new group generated iteratively from the generators of g, initially
* constructed to include all the essential generators */
gn = permlib::construct(g.n, gens.begin(), gens.end());
iter = opts.begin();
/* while there are more possible generators, and the new group is not
* of full order, add more generators */
while ( iter != opts.end() && gn->order() < ord ) {
/* if the current generator is not a member of the new group,
* add it to the generators */
if ( ! gn->sifts(**iter) ) {
gens.push_back(*iter);
gn = permlib::construct(g.n, gens.begin(), gens.end());
}
++iter;
}
return gens;
}
/** @return a copy of a permutation group such that no state is shared
* between the original and the copy
*/
static permutation_group deepCopy(permutation_group& g) {
typedef permutation_list::iterator perm_iter;
/* New list of generators */
permutation_list gens;
/* Copy old generators */
for (perm_iter it = g.S.begin(); it != g.S.end(); ++it) {
//permutation_cycle_list l = cycle_list(**it);
//permutation_ptr p = boost::make_shared<permutation>(perm(g.n, l));
permutation_ptr p = *it;
gens.push_back(p);
}
/* Group from new generators */
return *permlib::construct(g.n, gens.begin(), gens.end());
}
/** Converts an index set into an index list.
* @param s The set to convert
* @return an index list containing the elements of s in increasing order
*/
static index_list as_list(index_set const& s) {
index_list r(s.count());
ind i = 0;
for (lrs::index_set_iter iter = lrs::begin(s);
iter != lrs::end(s); ++iter) {
r[i++] = *iter;
}
return r;
}
/** Converts an index list into an index set.
* @param l The list to convert
* @param n The upper bound (inclusive) for the range of values
* represented by l
* @return an index set containing all the elements of l
*/
static index_set as_set(index_list const& l, ind n) {
index_set r(n+1);
for (index_list::const_iterator iter = l.begin();
iter != l.end(); ++iter) {
r.set(*iter);
}
return r;
}
/** Applies a permutation element-wise to an index set.
* @param p The permutation to apply
* @param s The set to permute
* @return an index set corresponding element-wise to the elements of s
* acted on by p
*/
static index_set apply(permutation const& p, index_set const& s) {
index_set r(s.size());
/* For each index in the set */
for (lrs::index_set_iter iter = lrs::begin(s);
iter != lrs::end(s); ++iter) {
/* set the permutation image of that index into the return */
r.set((p / ((*iter)-1)) + 1);
}
return r;
}
/** Applies a permutation element-wise to an index list.
* @param p The permutation to apply
* @param l The list to permute
* @return an index list corresponding element-wise to the elements of l
* acted on by p
*/
static index_list apply(permutation const& p, index_list const& l) {
index_list r(l.size());
/* For each index in the list */
ind i = 0;
for (index_list::const_iterator iter = l.begin();
iter != l.end(); ++iter) {
/* set the permutation image of that index into the return */
r[i++] = (p / ((*iter)-1)) + 1;
}
return r;
}
} /* namespace basil */
#endif /* _PERM_UTILS_HPP_ */