Add element-level optimizations

This commit is contained in:
Gabriel Tofvesson 2020-03-13 14:25:28 +01:00
parent 127dd571e4
commit da1ea8895d
3 changed files with 27 additions and 7 deletions

View File

@ -78,7 +78,6 @@ board_make_links (struct board *board)
for (board_pos y = 0; y < 9; ++y) for (board_pos y = 0; y < 9; ++y)
for (board_pos x = 0; x < 9; ++x) for (board_pos x = 0; x < 9; ++x)
{ {
struct board_element *elem = BOARD_ELEM (board, x, y);
unsigned pos = ELEM_POS (x, y); unsigned pos = ELEM_POS (x, y);
unsigned link_count = -1; unsigned link_count = -1;
@ -586,7 +585,6 @@ board_place (
{ {
if (board_meta_can_set (board, x, y, value)) if (board_meta_can_set (board, x, y, value))
{ {
struct board_element *elem = BOARD_ELEM (board, x, y);
unsigned pos = ELEM_POS (x, y); unsigned pos = ELEM_POS (x, y);
/* Unmark all adjacent elements */ /* Unmark all adjacent elements */

View File

@ -203,6 +203,10 @@ board_unmark (
element_value value element_value value
); );
/**
* Removes a marking of a potential value of an element
*/
bool bool
elem_unmark ( elem_unmark (
struct board_element *elem, struct board_element *elem,
@ -249,7 +253,7 @@ board_is_marked (
/** /**
* get whether or not a given element is marked with a particular value * Get whether or not a given element is marked with a particular value
*/ */
bool bool
elem_is_marked ( elem_is_marked (

26
main.c
View File

@ -319,6 +319,13 @@ first_potential_value (struct board_element *element, struct board *board, bool
/** /**
* Reduce away all elements on board with complexity=1 until none remain * Reduce away all elements on board with complexity=1 until none remain
*/ */
#ifdef NOVERB
bool
simplify (
struct boards_table *board_specs,
unsigned long long depth
)
#else
bool bool
simplify ( simplify (
struct boards_table *board_specs, struct boards_table *board_specs,
@ -326,10 +333,12 @@ simplify (
unsigned long long *counter, unsigned long long *counter,
unsigned verbosity unsigned verbosity
) )
#endif
{ {
/* Get current table */ /* Get current table */
struct board *board = board_specs->board_specs[depth]; struct board *board = board_specs->board_specs[depth];
#ifndef NOVERB
if (verbosity > 0) if (verbosity > 0)
{ {
if (((*counter) & (0xFFFF >> (4 * (4 - verbosity)))) == 0) if (((*counter) & (0xFFFF >> (4 * (4 - verbosity)))) == 0)
@ -340,6 +349,7 @@ simplify (
} }
*counter += 1; *counter += 1;
} }
#endif
bool error; bool error;
@ -378,14 +388,14 @@ simplify (
struct board_element *elem = BOARD_ELEM (board, x, y); struct board_element *elem = BOARD_ELEM (board, x, y);
/* Find a simplest element on the board */ /* Find a simplest element on the board */
if ( if (
! board_has_value (board, x, y) && ! elem->has_value &&
elem->complexity == board->complexity elem->complexity == board->complexity
) )
for (element_value value = 0; value < 9; ++value) for (element_value value = 0; value < 9; ++value)
{ {
/* Try speculative placement of each potential value and recurse */ /* Try speculative placement of each potential value and recurse */
tables_ensure_depth (board_specs, depth + 1); tables_ensure_depth (board_specs, depth + 1);
if ((elem->potential & (1 << value)) != 0) if (elem_is_marked (elem, value))
{ {
struct board *board_spec = struct board *board_spec =
board_place_speculative ( board_place_speculative (
@ -399,19 +409,27 @@ simplify (
/* If speculative placement failed, try another value */ /* If speculative placement failed, try another value */
if (board_spec == NULL) if (board_spec == NULL)
{ {
if (! board_unmark (board, x, y, value)) if (! elem_unmark (elem, value))
return false; return false;
continue; continue;
} }
/* Found solution */ /* Found solution */
if ( if (
#ifdef NOVERB
simplify (
board_specs,
depth + 1
) &&
#else
simplify ( simplify (
board_specs, board_specs,
depth + 1, depth + 1,
counter, counter,
verbosity verbosity
) && ) &&
#endif
board_spec->complexity == 0) board_spec->complexity == 0)
{ {
board_copy (board_spec, board); board_copy (board_spec, board);
@ -419,7 +437,7 @@ simplify (
y = 9; y = 9;
value = 9; value = 9;
} }
else if (! board_unmark (board, x, y, value)) else if (! elem_unmark (elem, value))
return false; return false;
} }
} }