Implement speculative value-placement
This commit is contained in:
parent
749ad4a260
commit
91dcf29b5f
@ -44,8 +44,6 @@ aforementioned specifications.
|
|||||||
|
|
||||||
## TODO
|
## TODO
|
||||||
|
|
||||||
* Speculative value placement
|
|
||||||
|
|
||||||
* Optimizations
|
* Optimizations
|
||||||
|
|
||||||
* Per-quadrant bit-field of populated values
|
* Per-quadrant bit-field of populated values
|
||||||
|
73
main.c
73
main.c
@ -7,6 +7,12 @@
|
|||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
bool
|
||||||
|
simplify (struct board *board, unsigned long long depth);
|
||||||
|
|
||||||
|
|
||||||
struct board_file
|
struct board_file
|
||||||
{
|
{
|
||||||
int fd;
|
int fd;
|
||||||
@ -141,14 +147,16 @@ print_board (struct board *board)
|
|||||||
* Compute first potential value of a given element
|
* Compute first potential value of a given element
|
||||||
*/
|
*/
|
||||||
element_value
|
element_value
|
||||||
first_potential_value (struct board_element *element, struct board *board)
|
first_potential_value (struct board_element *element, struct board *board, bool *error)
|
||||||
{
|
{
|
||||||
unsigned short potential = element->potential;
|
unsigned short potential = element->potential;
|
||||||
if (potential == 0)
|
if (potential == 0)
|
||||||
{
|
{
|
||||||
print_board (board);
|
*error = true;
|
||||||
abort(); /* Error */
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
*error = false;
|
||||||
|
|
||||||
element_value value = 0;
|
element_value value = 0;
|
||||||
while (potential != 0)
|
while (potential != 0)
|
||||||
@ -167,9 +175,11 @@ first_potential_value (struct board_element *element, struct board *board)
|
|||||||
/**
|
/**
|
||||||
* Reduce away all elements on board with complexity=1 until none remain
|
* Reduce away all elements on board with complexity=1 until none remain
|
||||||
*/
|
*/
|
||||||
void
|
bool
|
||||||
simplify (struct board *board)
|
simplify (struct board *board, unsigned long long depth)
|
||||||
{
|
{
|
||||||
|
bool error;
|
||||||
|
/* Reduce using low-complexity computation */
|
||||||
while (board->complexity == 1)
|
while (board->complexity == 1)
|
||||||
{
|
{
|
||||||
for (board_pos y = 0; y < 9; ++y)
|
for (board_pos y = 0; y < 9; ++y)
|
||||||
@ -179,10 +189,59 @@ simplify (struct board *board)
|
|||||||
struct board_element *elem = BOARD_ELEM (board, x, y);
|
struct board_element *elem = BOARD_ELEM (board, x, y);
|
||||||
if (elem->complexity == 1)
|
if (elem->complexity == 1)
|
||||||
{
|
{
|
||||||
board_place (board, x, y, first_potential_value (elem, board));
|
element_value value = first_potential_value (elem, board, &error);
|
||||||
|
if (error) return false;
|
||||||
|
|
||||||
|
board_place (board, x, y, value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Attempt to reduce with speculative placement */
|
||||||
|
if (board->complexity > 1)
|
||||||
|
{
|
||||||
|
for (board_pos y = 0; y < 9; ++y)
|
||||||
|
for (board_pos x = 0; x < 9; ++x)
|
||||||
|
{
|
||||||
|
struct board_element *elem = BOARD_ELEM (board, x, y);
|
||||||
|
/* Find a simplest element on the board*/
|
||||||
|
if (
|
||||||
|
! board_has_value (board, x, y) &&
|
||||||
|
elem->complexity == board->complexity
|
||||||
|
)
|
||||||
|
for (element_value value = 0; value < 9; ++value)
|
||||||
|
{
|
||||||
|
/* Try speculative placement of each potential value and recurse */
|
||||||
|
if ((elem->potential & (1 << value)) != 0)
|
||||||
|
{
|
||||||
|
/* printf ("Speculating (%u, %u) = %u (depth = %llu)\n", x, y, value, depth); */
|
||||||
|
struct board *board_spec =
|
||||||
|
board_place_speculative (board, x, y, value);
|
||||||
|
|
||||||
|
/* If speculative placement failed, try another value */
|
||||||
|
if (board_spec == NULL)
|
||||||
|
{
|
||||||
|
/* printf ("Speculation failed catastrophically\n"); */
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Found solution */
|
||||||
|
if (simplify (board_spec, depth + 1) && board_spec->complexity == 0)
|
||||||
|
{
|
||||||
|
board_copy (board_spec, board);
|
||||||
|
x = 9;
|
||||||
|
y = 9;
|
||||||
|
value = 9;
|
||||||
|
}
|
||||||
|
/* else printf ("Speculation failed\n"); */
|
||||||
|
|
||||||
|
/* board_copy leaves resposibility of freeing to caller */
|
||||||
|
free (board_spec);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -210,7 +269,7 @@ main (int argc, char **argv, char **env)
|
|||||||
puts ("Board is invalid!");
|
puts ("Board is invalid!");
|
||||||
|
|
||||||
puts("\nReducing...");
|
puts("\nReducing...");
|
||||||
simplify (&board);
|
simplify (&board, 1);
|
||||||
|
|
||||||
print_board (&board);
|
print_board (&board);
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user