Start implementing metadata analysis
This commit is contained in:
parent
98de45e928
commit
83e2d0f3e3
259
board.c
259
board.c
@ -20,6 +20,20 @@
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Raises an error if the given condition is not met
|
||||
* The error will be the given reason
|
||||
*/
|
||||
#ifdef OPTIMIZE
|
||||
#define ASSERT(cond, reason)
|
||||
#else
|
||||
#define ASSERT(cond, reason) \
|
||||
{ \
|
||||
if (! cond) ERROR ((reason)); \
|
||||
}
|
||||
#endif
|
||||
|
||||
|
||||
/**
|
||||
* Check if a given xy-pair is in bounds of a Sudoku board
|
||||
*/
|
||||
@ -51,11 +65,139 @@ is_valid_value (element_value value)
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
meta_init (struct metadata *meta)
|
||||
{
|
||||
memset (meta, 0, sizeof (struct metadata));
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
board_init (struct board *board)
|
||||
{
|
||||
memset (board, 0, sizeof board->elements);
|
||||
board->complexity = 9;
|
||||
|
||||
for (unsigned i = 0; i < 9; ++i)
|
||||
{
|
||||
meta_init (&board->meta_quad[i]);
|
||||
meta_init (&board->meta_row[i]);
|
||||
meta_init (&board->meta_col[i]);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
bool
|
||||
meta_has_value (const struct metadata *meta, element_value value)
|
||||
{
|
||||
return ((meta->marked >> value) & 1) == 1;
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
meta_set_value (struct metadata *meta, element_value value)
|
||||
{
|
||||
meta->marked |= 1 << value;
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
meta_clear_values (struct metadata *meta)
|
||||
{
|
||||
meta->marked = 0;
|
||||
}
|
||||
|
||||
|
||||
static inline void
|
||||
meta_mark (struct metadata *meta, element_value value, unsigned index)
|
||||
{
|
||||
meta_set_value (meta, value);
|
||||
|
||||
unsigned char count = meta->unique[value].count;
|
||||
if (count == 0)
|
||||
{
|
||||
meta->unique[value].count = 1;
|
||||
meta->unique[value].index = index;
|
||||
}
|
||||
else
|
||||
{
|
||||
meta->unique[value].count = 2;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
board_meta_quad_refresh (struct board *board, board_pos qx, board_pos qy)
|
||||
{
|
||||
struct metadata *meta = BOARD_QUAD (board, qx, qy);
|
||||
board_pos quad_base_x = qx * 3;
|
||||
board_pos quad_base_y = qy * 3;
|
||||
|
||||
meta_clear_values (meta);
|
||||
|
||||
for (board_pos off_y = 0; off_y < 3; ++off_y)
|
||||
for (board_pos off_x = 0; off_x < 3; ++off_x)
|
||||
{
|
||||
struct board_element *elem =
|
||||
BOARD_ELEM (board, quad_base_x + off_x, quad_base_y + off_y);
|
||||
|
||||
if (elem->has_value)
|
||||
meta_mark (meta, elem->value, (off_y * 3) + off_x);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
board_meta_row_refresh (struct board *board, board_pos y)
|
||||
{
|
||||
struct metadata *meta = BOARD_ROW (board, y);
|
||||
|
||||
meta_clear_values (meta);
|
||||
|
||||
for (board_pos x = 0; x < 9; ++x)
|
||||
{
|
||||
struct board_element *elem = BOARD_ELEM (board, x, y);
|
||||
|
||||
if (elem->has_value)
|
||||
meta_mark (meta, elem->value, x);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
board_meta_col_refresh (struct board *board, board_pos x)
|
||||
{
|
||||
struct metadata *meta = BOARD_COL (board, x);
|
||||
|
||||
meta_clear_values (meta);
|
||||
|
||||
for (board_pos y = 0; y < 9; ++y)
|
||||
{
|
||||
struct board_element *elem = BOARD_ELEM (board, x, y);
|
||||
|
||||
if (elem->has_value)
|
||||
meta_mark (meta, elem->value, y);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
bool
|
||||
board_meta_can_set (
|
||||
const struct board *board,
|
||||
board_pos x,
|
||||
board_pos y,
|
||||
element_value value
|
||||
)
|
||||
{
|
||||
if (is_in_bounds (x, y) && is_valid_value (value))
|
||||
{
|
||||
return ! (
|
||||
meta_has_value (BOARD_ROW (board, y), value) ||
|
||||
meta_has_value (BOARD_COL (board, x), value) ||
|
||||
meta_has_value (BOARD_QUAD (board, TO_QUAD (x), TO_QUAD (y)), value)
|
||||
);
|
||||
}
|
||||
else ERROR("Invalid parameters to function board_meta_can_set()");
|
||||
}
|
||||
|
||||
|
||||
@ -69,6 +211,11 @@ board_set (
|
||||
{
|
||||
if (is_in_bounds (x, y) && is_valid_value (value))
|
||||
{
|
||||
ASSERT (
|
||||
board_meta_can_set (board, x, y, value),
|
||||
"Attempt to set impossible value on board"
|
||||
);
|
||||
|
||||
struct board_element *elem = BOARD_ELEM (board, x, y);
|
||||
elem->has_value = true;
|
||||
elem->value = value;
|
||||
@ -87,6 +234,11 @@ board_mark (
|
||||
{
|
||||
if (is_in_bounds (x, y) && is_valid_value (value))
|
||||
{
|
||||
ASSERT (
|
||||
! board_has_value (board, x, y),
|
||||
"Attempt to mark element with value"
|
||||
);
|
||||
|
||||
struct board_element *elem = BOARD_ELEM (board, x, y);
|
||||
if (! board_is_marked (board, x, y, value))
|
||||
{
|
||||
@ -108,15 +260,18 @@ board_unmark (
|
||||
{
|
||||
if (is_in_bounds (x, y) && is_valid_value (value))
|
||||
{
|
||||
ASSERT (
|
||||
! board_has_value (board, x, y),
|
||||
"Attempt to mark element with value"
|
||||
);
|
||||
|
||||
struct board_element *elem = BOARD_ELEM (board, x, y);
|
||||
|
||||
if (board_is_marked (board, x, y, value))
|
||||
{
|
||||
/* Shift bit to correct place and then invert first 9 bits */
|
||||
elem->potential &= (1 << value) ^ 0x1FF;
|
||||
elem->potential ^= (1 << value);
|
||||
--(elem->complexity);
|
||||
if (elem->complexity < board->complexity)
|
||||
board->complexity = elem->complexity;
|
||||
}
|
||||
}
|
||||
else ERROR("Invalid parameters to function board_unmark()");
|
||||
@ -169,47 +324,6 @@ board_is_marked (
|
||||
}
|
||||
|
||||
|
||||
bool
|
||||
board_can_place_value (
|
||||
const struct board *board,
|
||||
board_pos x,
|
||||
board_pos y,
|
||||
element_value value
|
||||
)
|
||||
{
|
||||
if (is_in_bounds (x, y) && is_valid_value (value))
|
||||
{
|
||||
/* Check for x-axis */
|
||||
for (board_pos compare_x = 0; compare_x < 9; ++compare_x)
|
||||
{
|
||||
if (compare_x == x)
|
||||
continue;
|
||||
else if (
|
||||
board_has_value (board, compare_x, y) &&
|
||||
board_get_value (board, compare_x, y) == value
|
||||
)
|
||||
return false;
|
||||
}
|
||||
|
||||
/* Check for y-axis */
|
||||
for (board_pos compare_y = 0; compare_y < 9; ++compare_y)
|
||||
{
|
||||
if (compare_y == y)
|
||||
continue;
|
||||
else if (
|
||||
board_has_value (board, x, compare_y) &&
|
||||
board_get_value (board, x, compare_y) == value
|
||||
)
|
||||
return false;
|
||||
}
|
||||
|
||||
/* No matches */
|
||||
return true;
|
||||
}
|
||||
else ERROR("Invalid parameters to function board_can_place_value()");
|
||||
}
|
||||
|
||||
|
||||
bool
|
||||
board_is_valid (struct board *board)
|
||||
{
|
||||
@ -217,7 +331,7 @@ board_is_valid (struct board *board)
|
||||
for (board_pos x = 0; x < 9; ++x)
|
||||
if (
|
||||
board_has_value (board, x, y) &&
|
||||
! board_can_place_value (
|
||||
! board_meta_can_set (
|
||||
board,
|
||||
x,
|
||||
y,
|
||||
@ -245,14 +359,18 @@ board_update_marks (
|
||||
elem->complexity = 0;
|
||||
|
||||
/* Check x-axis */
|
||||
for (board_pos check_x = 0; check_x < 9; ++check_x)
|
||||
if (check_x != x && board_has_value (board, check_x, y))
|
||||
elem->potential |= (1 << BOARD_ELEM (board, check_x, y)->value);
|
||||
elem->potential |= BOARD_QUAD (board, TO_QUAD (x), TO_QUAD (y))->marked;
|
||||
elem->potential |= BOARD_ROW (board, y)->marked;
|
||||
elem->potential |= BOARD_COL (board, x)->marked;
|
||||
|
||||
/* Check y-axis */
|
||||
for (board_pos check_y = 0; check_y < 9; ++check_y)
|
||||
if (check_y != y && board_has_value (board, x, check_y))
|
||||
elem->potential |= (1 << BOARD_ELEM (board, x, check_y)->value);
|
||||
// for (board_pos check_x = 0; check_x < 9; ++check_x)
|
||||
// if (check_x != x && board_has_value (board, check_x, y))
|
||||
// elem->potential |= (1 << BOARD_ELEM (board, check_x, y)->value);
|
||||
|
||||
// /* Check y-axis */
|
||||
// for (board_pos check_y = 0; check_y < 9; ++check_y)
|
||||
// if (check_y != y && board_has_value (board, x, check_y))
|
||||
// elem->potential |= (1 << BOARD_ELEM (board, x, check_y)->value);
|
||||
|
||||
/* Invert matches */
|
||||
elem->potential ^= 0x1FF;
|
||||
@ -265,10 +383,6 @@ board_update_marks (
|
||||
++(elem->complexity);
|
||||
potential >>= 1;
|
||||
}
|
||||
|
||||
/* Store complexity if element is the simplest */
|
||||
if (elem->complexity < board->complexity)
|
||||
board->complexity = elem->complexity;
|
||||
}
|
||||
else ERROR("Invalid parameters to function board_update_marks()");
|
||||
}
|
||||
@ -463,7 +577,6 @@ board_update_marks_by_quad (
|
||||
changed |= true;
|
||||
result.unique->potential = 1 << value;
|
||||
result.unique->complexity = 1;
|
||||
board->complexity = 1;
|
||||
}
|
||||
}
|
||||
return changed;
|
||||
@ -583,10 +696,7 @@ board_update_marks_by_rows (struct board *board)
|
||||
{
|
||||
struct board_element *elem = BOARD_ELEM (board, x ,y);
|
||||
elem->complexity = 1;
|
||||
board->complexity = 1;
|
||||
changed |= elem->potential != (elem->potential = (1 << value));
|
||||
|
||||
abort();
|
||||
}
|
||||
}
|
||||
|
||||
@ -618,7 +728,6 @@ board_update_marks_by_cols (struct board *board)
|
||||
{
|
||||
struct board_element *elem = BOARD_ELEM (board, x ,y);
|
||||
elem->complexity = 1;
|
||||
board->complexity = 1;
|
||||
changed |= elem->potential != (elem->potential = (1 << value));
|
||||
}
|
||||
}
|
||||
@ -639,21 +748,22 @@ board_update_all_marks (struct board *board)
|
||||
if (! board_has_value (board, x, y))
|
||||
board_update_marks (board, x, y);
|
||||
|
||||
return;
|
||||
bool changed;
|
||||
do
|
||||
{
|
||||
changed = false;
|
||||
|
||||
/* Update marks by exclusion analysis */
|
||||
// for (board_pos y = 0; y < 9; ++y)
|
||||
// for (board_pos x = 0; x < 9; ++x)
|
||||
// if (! board_has_value (board, x, y))
|
||||
// changed |= board_update_marks_by_excl (board, x, y);
|
||||
for (board_pos y = 0; y < 9; ++y)
|
||||
for (board_pos x = 0; x < 9; ++x)
|
||||
if (! board_has_value (board, x, y))
|
||||
changed |= board_update_marks_by_excl (board, x, y);
|
||||
|
||||
// /* Update marks by quad potential analysis */
|
||||
// for (board_pos y = 0; y < 9; y += 3)
|
||||
// for (board_pos x = 0; x < 9; x += 3)
|
||||
// changed |= board_update_marks_by_quad (board, x, y);
|
||||
/* Update marks by quad potential analysis */
|
||||
for (board_pos y = 0; y < 9; y += 3)
|
||||
for (board_pos x = 0; x < 9; x += 3)
|
||||
changed |= board_update_marks_by_quad (board, x, y);
|
||||
|
||||
changed |= board_update_marks_by_rows (board);
|
||||
changed |= board_update_marks_by_cols (board);
|
||||
@ -672,7 +782,7 @@ board_place (
|
||||
{
|
||||
if (is_in_bounds (x, y) && is_valid_value (value))
|
||||
{
|
||||
if (board_can_place_value (board, x, y, value))
|
||||
if (board_meta_can_set (board, x, y, value))
|
||||
{
|
||||
/* Unmark x-axis */
|
||||
for (board_pos unmark_x = 0; unmark_x < 9; ++unmark_x)
|
||||
@ -702,7 +812,12 @@ board_place (
|
||||
board_unmark (board, target_x, target_y, value);
|
||||
}
|
||||
|
||||
/* Set value */
|
||||
/* Update metadata */
|
||||
meta_set_value (BOARD_QUAD (board, quad_x, quad_y), value);
|
||||
meta_set_value (BOARD_ROW (board, y), value);
|
||||
meta_set_value (BOARD_COL (board, x), value);
|
||||
|
||||
/* Set value */
|
||||
board_set (board, x, y, value);
|
||||
|
||||
/* Update board complexity */
|
||||
@ -726,7 +841,7 @@ board_place_speculative (
|
||||
if (is_in_bounds (x, y) && is_valid_value (value))
|
||||
{
|
||||
/* Ensure value can be placed*/
|
||||
if (board_can_place_value (board, x, y, value))
|
||||
if (board_meta_can_set (board, x, y, value))
|
||||
{
|
||||
/* Create duplicate and place value */
|
||||
board_copy (board, board_duplicate);
|
||||
|
115
board.h
115
board.h
@ -11,6 +11,22 @@
|
||||
*/
|
||||
#define BOARD_ELEM(board_ptr, x, y) (&(board_ptr)->elements[((y) * 9) + (x)])
|
||||
|
||||
/**
|
||||
* Get a `struct metadata`-entry from a specified location on the quadrand grid
|
||||
*/
|
||||
#define BOARD_QUAD(board_ptr, qx, qy) (&(board_ptr)->meta_quad[(qy) * 3 + (qx)])
|
||||
|
||||
/**
|
||||
* Get a `struct metadata`-entry from a specified row value
|
||||
*/
|
||||
#define BOARD_ROW(board_ptr, y) (&(board_ptr)->meta_row[(y)])
|
||||
|
||||
/**
|
||||
* Get a `struct metadata`-entry from a specified column value
|
||||
*/
|
||||
#define BOARD_COL(board_ptr, x) (&(board_ptr)->meta_col[(x)])
|
||||
|
||||
|
||||
/**
|
||||
* Convert any valid board position to a quadrant base position (lowest index)
|
||||
*/
|
||||
@ -38,24 +54,104 @@ struct board_element {
|
||||
};
|
||||
};
|
||||
|
||||
/**
|
||||
* Board region metadata
|
||||
*/
|
||||
struct metadata {
|
||||
unsigned short marked : 9; /* Which values have been marked */
|
||||
struct { /* Unique potentials */
|
||||
unsigned char count : 2; /* Whether or not a potential is unique */
|
||||
unsigned char index : 3; /* Metadata index. Context-specific */
|
||||
} unique[9];
|
||||
};
|
||||
|
||||
/**
|
||||
* Board structure representing the state of a Sudoku game
|
||||
* Complexity describes
|
||||
* Complexity describes how many possible values an element can legally have
|
||||
*
|
||||
* TODO: Replace elements with packed structure
|
||||
*/
|
||||
struct board {
|
||||
/* Immediate data*/
|
||||
struct board_element elements[81]; /* Game board */
|
||||
unsigned char complexity; /* Complexity of simplest element */
|
||||
unsigned char complexity : 3; /* Complexity of simplest element */
|
||||
|
||||
/* Metadata */
|
||||
struct metadata meta_quad [9]; /* Quadrant metadata */
|
||||
struct metadata meta_row [9]; /* Row metadata */
|
||||
struct metadata meta_col [9]; /* Column metadata */
|
||||
};
|
||||
|
||||
|
||||
/**
|
||||
* Initialize a board to a blank state with 0 complexity
|
||||
* Initialize metadata to a blank state
|
||||
*/
|
||||
void
|
||||
meta_init (struct metadata *meta);
|
||||
|
||||
|
||||
/**
|
||||
* Initialize a board to a blank state with maximum complexity
|
||||
*/
|
||||
void
|
||||
board_init (struct board *board);
|
||||
|
||||
|
||||
/**
|
||||
* Check if a metadata structure has marked a given value
|
||||
*/
|
||||
bool
|
||||
meta_has_value (const struct metadata *meta, element_value value);
|
||||
|
||||
|
||||
/**
|
||||
* Set value as marked for the given metadata structure
|
||||
*/
|
||||
void
|
||||
meta_set_value (struct metadata *meta, element_value value);
|
||||
|
||||
|
||||
/**
|
||||
* Clear all marked values for the given metadata structure
|
||||
*/
|
||||
void
|
||||
meta_clear_values (struct metadata *meta);
|
||||
|
||||
|
||||
/**
|
||||
* Refresh metadata for a given quadrant
|
||||
*/
|
||||
void
|
||||
board_meta_quad_refresh (struct board *board, board_pos qx, board_pos qy);
|
||||
|
||||
|
||||
/**
|
||||
* Refresh metadata for a given row
|
||||
*/
|
||||
void
|
||||
board_meta_row_refresh (struct board *board, board_pos y);
|
||||
|
||||
|
||||
/**
|
||||
* Refresh metadata for a given column
|
||||
*/
|
||||
void
|
||||
board_meta_col_refresh (struct board *board, board_pos x);
|
||||
|
||||
|
||||
/**
|
||||
* Check if a value can be set at a given position on the board based on
|
||||
* analysing the metadata structures of `board`
|
||||
*/
|
||||
bool
|
||||
board_meta_can_set (
|
||||
const struct board *board,
|
||||
board_pos x,
|
||||
board_pos y,
|
||||
element_value value
|
||||
);
|
||||
|
||||
|
||||
/**
|
||||
* Set the value of an element on the board
|
||||
*
|
||||
@ -135,19 +231,6 @@ board_is_marked (
|
||||
);
|
||||
|
||||
|
||||
/**
|
||||
* Checks if placing the given value at the given position on the board would
|
||||
* leave the board in a valid state
|
||||
*/
|
||||
bool
|
||||
board_can_place_value (
|
||||
const struct board *board,
|
||||
board_pos x,
|
||||
board_pos y,
|
||||
element_value value
|
||||
);
|
||||
|
||||
|
||||
/**
|
||||
* Checks if there are any pairs of board elements that share a value and
|
||||
* also share either a row or a column
|
||||
|
17
main.c
17
main.c
@ -292,9 +292,14 @@ simplify (
|
||||
struct board *board = board_specs->board_specs[depth];
|
||||
|
||||
bool error;
|
||||
|
||||
bool placed = false;
|
||||
unsigned count = 0;
|
||||
|
||||
/* Reduce using low-complexity computation */
|
||||
while (board->complexity == 1)
|
||||
{
|
||||
placed = false;
|
||||
for (board_pos y = 0; y < 9; ++y)
|
||||
for (board_pos x = 0; x < 9; ++x)
|
||||
if (! board_has_value (board, x, y))
|
||||
@ -305,9 +310,19 @@ simplify (
|
||||
element_value value = first_potential_value (elem, board, &error);
|
||||
if (error) return false;
|
||||
|
||||
board_place (board, x, y, value);
|
||||
placed = true;
|
||||
++count;
|
||||
|
||||
if (! board_place (board, x, y, value))
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (! placed)
|
||||
{
|
||||
print_board (board_specs->board_specs[depth], board_specs->board_specs[0], 21, 0);
|
||||
printf ("Count: %u\n", count);
|
||||
abort ();
|
||||
}
|
||||
}
|
||||
|
||||
/* Attempt to reduce with speculative placement */
|
||||
|
Loading…
x
Reference in New Issue
Block a user