-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsolve.go
289 lines (234 loc) · 8.01 KB
/
solve.go
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
package sudoku
import (
"math/rand"
"sync"
)
func (self *mutableGridImpl) Solve() bool {
//Special case; check if it's already solved.
//TODO: removing this causes Solve, when called on an already solved grid, to sometimes fail. Why is that?
if self.Solved() {
return true
}
//TODO: Optimization: we only need one, so we can bail as soon as we find a single one.
solutions := self.Solutions()
if len(solutions) == 0 {
return false
}
self.LoadSDK(solutions[0].DataString())
return true
}
func (self *gridImpl) NumSolutions() int {
return len(self.Solutions())
}
func (self *mutableGridImpl) NumSolutions() int {
return len(self.Solutions())
}
func (self *gridImpl) HasSolution() bool {
return len(nOrFewerSolutions(self, 1)) > 0
}
func (self *mutableGridImpl) HasSolution() bool {
return len(cachedNOrFewerSolutions(self, 1)) > 0
}
func (self *gridImpl) HasMultipleSolutions() bool {
return len(nOrFewerSolutions(self, 2)) >= 2
}
func (self *mutableGridImpl) HasMultipleSolutions() bool {
return len(cachedNOrFewerSolutions(self, 2)) >= 2
}
func (self *gridImpl) Solutions() []Grid {
return nOrFewerSolutions(self, 0)
}
func (self *mutableGridImpl) Solutions() (solutions []Grid) {
return cachedNOrFewerSolutions(self, 0)
}
func cachedNOrFewerSolutions(self *mutableGridImpl, max int) []Grid {
//TODO: investigate how useful this actually is for mutableGridImpl, and
//also test whether it makes sense to make one for gridImpl.
self.cachedSolutionsLock.RLock()
hasNoCachedSolutions := self.cachedSolutions == nil
cachedSolutionsLen := self.cachedSolutionsRequestedLength
//If these two values are OK we'll just return the value from the cache.
result := self.cachedSolutions
self.cachedSolutionsLock.RUnlock()
if hasNoCachedSolutions || (max == 0 && cachedSolutionsLen != 0) || (max > 0 && cachedSolutionsLen < max && cachedSolutionsLen > 0) {
//Nope, we need to recalculate solutions.
solutions := nOrFewerSolutions(self, max)
result = solutions
self.cachedSolutionsLock.Lock()
self.cachedSolutions = solutions
self.cachedSolutionsRequestedLength = max
self.cachedSolutionsLock.Unlock()
}
return result
}
//The actual workhorse of solutions generating. 0 means "as many as you can
//find". It might return more than you asked for, if it already had more
//results than requested sitting around.
func nOrFewerSolutions(grid Grid, max int) []Grid {
queueDone := make(chan bool, 1)
queue := newSyncedFiniteQueue(0, DIM*DIM, queueDone)
queue.In <- grid.Copy()
//In the past this wasn't buffered, but then when we finished early other items would try to go into it
//and block, which prevented them from looping back up and getting the signal to shut down.
//Since there's only a known number of threads, we'll make sure they all ahve a place to leave their work
//without blocking so they can get the signal to shut down.
incomingSolutions := make(chan Grid, _NUM_SOLVER_THREADS)
//Using a pattern for closing fan in style receivers from http://blog.golang.org/pipelines
var wg sync.WaitGroup
var solutions []Grid
//TODO: figure out a way to kill all of these threads when necessary.
//TODO: don't use a constant here, use someting around the lines of
//numCPU
for i := 0; i < _NUM_SOLVER_THREADS; i++ {
go func() {
//Sovler thread.
firstRun := true
defer wg.Done()
for {
grid, ok := <-queue.Out
if !ok {
return
}
result := searchGridSolutions(grid.(Grid), queue, firstRun, max)
if result != nil {
incomingSolutions <- result
}
queue.ItemDone <- true
firstRun = false
}
}()
}
wg.Add(_NUM_SOLVER_THREADS)
go func() {
wg.Wait()
close(incomingSolutions)
}()
OuterLoop:
for {
select {
case solution := <-incomingSolutions:
//incomingSolutions must have been closed because no more work to do.
if solution == nil {
break OuterLoop
}
//Add it to results
solutions = append(solutions, solution)
if max > 0 && len(solutions) >= max {
break OuterLoop
}
case <-queueDone:
//Well, that's as good as it's going to get.
break OuterLoop
}
}
//In some cases that previous select would have something incoming on
//incomingsolutions, as well as on queueDone, and queueDone would have
//just so happened to have won. Check for one last remaining item
//coming in from incomingSolutions. Technically it's possible (how?)
//to have multiple items waiting on incomingSolutions, so read as many
//as we can get without blocking.(Not checking for this was the reason
//for bug #134.)
DoneReading:
for {
select {
case solution := <-incomingSolutions:
if solution != nil {
solutions = append(solutions, solution)
}
default:
//Nope, guess there wasn't one left.
break DoneReading
}
}
//There might be some things waiting to go into incomingSolutions here, but because it has a slot
//for every thread to be buffered, it's OK, we can just stop now.
//TODO: the grids waiting in the queue will never have their .Done called. This isn't a big deal--GC should reclaim them--
//but we won't have as many that we could reuse.
queue.Exit <- true
return solutions
}
func searchGridSolutions(grid Grid, queue *syncedFiniteQueue, isFirstRun bool, numSoughtSolutions int) Grid {
//This will only be called by Solutions.
//We will return ourselves if we are a solution, and if not we will return nil.
//If there are any sub children, we will send them to counter before we're done.
if grid.Invalid() {
return nil
}
grid = withSimpleCellsFilled(grid)
//Have any cells noticed they were invalid while solving forward?
if grid.basicInvalid() {
return nil
}
if grid.Solved() {
return grid
}
//Well, looks like we're going to have to branch.
rankedObject := grid.queue().NewGetter().Get()
if rankedObject == nil {
panic("Queue didn't have any cells.")
}
cell, ok := rankedObject.(Cell)
if !ok {
panic("We got back a non-cell from the grid's queue")
}
unshuffledPossibilities := cell.Possibilities()
possibilities := make([]int, len(unshuffledPossibilities))
for i, j := range rand.Perm(len(unshuffledPossibilities)) {
possibilities[i] = unshuffledPossibilities[j]
}
var result Grid
for i, num := range possibilities {
modification := newCellModification(cell.Reference())
modification.Number = num
copy := grid.CopyWithModifications(GridModification{
modification,
})
//As an optimization for cases where there are many solutions, we'll just continue a DFS until we barf then unroll back up.
//It doesn't appear to slow things down in the general case
if i == 0 && !isFirstRun {
result = searchGridSolutions(copy, queue, false, numSoughtSolutions)
if result != nil && numSoughtSolutions == 1 {
//No need to spin off other branches, just return up.
return result
}
} else {
queue.In <- copy
}
}
return result
}
//Returns a copy of Grid that has filled in all of the cells it can without
//branching or doing any advanced techniques that require anything more than a
//single cell's possibles list.
func withSimpleCellsFilled(grid Grid) Grid {
//TODO: it's weird that this isn't a method (it is just to save a little
//tiny bit of wiring up the same impl to both underlying structs)
//We fetch all of the cells that have a single possibility, then create a
//copy with all of those filled. Then we repeat, because filling those
//cells may have set other cells to now only have one possibility. Repeat
//until no more cells with one possibility found.
changesMade := true
for changesMade {
changesMade = false
getter := grid.queue().NewGetter()
obj := getter.GetSmallerThan(2)
var modifications GridModification
for obj != nil && !grid.basicInvalid() {
cell, ok := obj.(Cell)
if !ok {
panic("Expected a Cell in the queue, found something else")
}
changesMade = true
modification := &CellModification{
Cell: cell.Reference(),
Number: cell.implicitNumber(),
}
modifications = append(modifications, modification)
obj = getter.GetSmallerThan(2)
}
if modifications != nil {
grid = grid.CopyWithModifications(modifications)
}
}
return grid
}