forked from google/or-tools
-
Notifications
You must be signed in to change notification settings - Fork 9
/
symmetry.h
165 lines (141 loc) · 6.85 KB
/
symmetry.h
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
// Copyright 2010-2024 Google LLC
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#ifndef OR_TOOLS_SAT_SYMMETRY_H_
#define OR_TOOLS_SAT_SYMMETRY_H_
#include <memory>
#include <vector>
#include "absl/types/span.h"
#include "ortools/algorithms/sparse_permutation.h"
#include "ortools/base/strong_vector.h"
#include "ortools/sat/sat_base.h"
#include "ortools/util/stats.h"
namespace operations_research {
namespace sat {
// This class implements more or less the strategy described in the paper:
// Devriendt J., Bogaerts B., De Cat B., Denecker M., Mears C. "Symmetry
// propagation: Improved Dynamic Symmetry Breaking in SAT", 2012,
// IEEE 24th International Conference on Tools with Artificial Intelligence.
//
// Basically, each time a literal is propagated, this class tries to detect
// if another literal could also be propagated by symmetry. Note that this uses
// a heuristic in order to be efficient and that it is not exhaustive in the
// sense that it doesn't detect all possible propagations.
//
// Algorithm details:
//
// Given the current solver trail (i.e. the assigned literals and their
// assignment order) the idea is to compute (as efficiently as possible) for
// each permutation added to this class what is called the first (under the
// trail assignment order) non-symmetric literal. A literal 'l' is said to be
// non-symmetric under a given assignment and for a given permutation 'p' if
// 'l' is assigned to true but not 'p(l)'.
//
// If a first non-symmetric literal 'l' for a permutation 'p' is not a decision,
// then:
// - Because it is not a decision, 'l' has been implied by a reason formed by
// literals assigned to true at lower trail indices.
// - Because this is the first non-symmetric literal for 'p', the permuted
// reason only contains literal that are also assigned to true.
// - Because of this, 'p(l)' is also implied by the current assignment.
// Of course, this assume that p is a symmetry of the full problem.
// Note that if it is already assigned to false, then we have a conflict.
//
// TODO(user): Implement the optimizations mentioned in the paper?
// TODO(user): Instrument and see if the code can be optimized.
class SymmetryPropagator : public SatPropagator {
public:
SymmetryPropagator();
// This type is neither copyable nor movable.
SymmetryPropagator(const SymmetryPropagator&) = delete;
SymmetryPropagator& operator=(const SymmetryPropagator&) = delete;
~SymmetryPropagator() override;
bool Propagate(Trail* trail) final;
void Untrail(const Trail& trail, int trail_index) final;
absl::Span<const Literal> Reason(const Trail& trail, int trail_index,
int64_t conflict_id) const final;
// Adds a new permutation to this symmetry propagator. The ownership is
// transferred. This must be an integer permutation such that:
// - Its domain is [0, 2 * num_variables) and corresponds to the index
// representation of the literals over num_variables variables.
// - It must be compatible with the negation, for any literal l; not(p(l))
// must be the same as p(not(l)), where p(x) represents the image of x by
// the permutation.
//
// Remark: Any permutation which is a symmetry of the main SAT problem can be
// added here. However, since the number of permutations is usually not
// manageable, a good alternative is to only add the generators of the
// permutation group. It is also important to add permutations with a support
// as small as possible.
//
// TODO(user): Currently this can only be called before PropagateNext() is
// called (DCHECKed). Not sure if we need more incrementality though.
void AddSymmetry(std::unique_ptr<SparsePermutation> permutation);
int num_permutations() const { return permutations_.size(); }
// Visible for testing.
//
// Permutes a list of literals from input into output using the permutation
// with given index. This uses tmp_literal_mapping_ and has a complexity in
// O(permutation_support + input_size).
void Permute(int index, absl::Span<const Literal> input,
std::vector<Literal>* output) const;
private:
// Propagates the literal at propagation_trail_index_ from the trail.
bool PropagateNext(Trail* trail);
// The permutations.
// The index of a permutation is its position in this vector.
std::vector<std::unique_ptr<SparsePermutation>> permutations_;
// Reverse mapping (source literal) -> list of (permutation_index, image).
struct ImageInfo {
ImageInfo(int p, Literal i) : permutation_index(p), image(i) {}
int permutation_index;
Literal image;
};
util_intops::StrongVector<LiteralIndex, std::vector<ImageInfo>> images_;
// For each permutation p, we maintain the list of all assigned literals
// affected by p whose trail index is < propagation_trail_index_; sorted by
// trail index. Next to each such literal, we also store:
struct AssignedLiteralInfo {
AssignedLiteralInfo(Literal l, Literal i, int index)
: literal(l), image(i), first_non_symmetric_info_index_so_far(index) {}
// The literal in question (assigned to true and in the support of p).
Literal literal;
// The image by p of the literal above.
Literal image;
// Previous AssignedLiteralInfos are considered 'symmetric' iff both their
// 'literal' and 'image' were assigned to true at the time the current
// AssignedLiteralInfo's literal was assigned (i.e. earlier in the trail).
int first_non_symmetric_info_index_so_far;
};
std::vector<std::vector<AssignedLiteralInfo>> permutation_trails_;
// Adds an AssignedLiteralInfo to the given permutation trail.
// Returns false if there is a non-symmetric literal in this trail with its
// image not already assigned to true by the solver.
bool Enqueue(const Trail& trail, Literal literal, Literal image,
std::vector<AssignedLiteralInfo>* p_trail);
// The identity permutation over all the literals.
// This is temporary modified to encode a sparse permutation and then always
// restored to the identity.
mutable util_intops::StrongVector<LiteralIndex, Literal> tmp_literal_mapping_;
// Symmetry reason indexed by trail_index.
struct ReasonInfo {
int source_trail_index;
int symmetry_index;
};
std::vector<ReasonInfo> reasons_;
mutable StatsGroup stats_;
int num_propagations_;
int num_conflicts_;
};
} // namespace sat
} // namespace operations_research
#endif // OR_TOOLS_SAT_SYMMETRY_H_