/** * Sudoku board header file * * Created by Gabriel Tofvesson */ #include #define ELEM_POS(x, y) (((y) * 9) + (x)) /** * Get a `struct board_element`-entry from a specified location on the board */ #define BOARD_ELEM(board_ptr, x, y) (&(board_ptr)->elements[ELEM_POS (x, y)]) /** * Get a `struct metadata`-entry from a specified location on the quadrand grid */ #define BOARD_QUAD(board_ptr, x, y) (&(board_ptr)->meta_quad[TO_QUAD ((y)) + ((x) / 3)]) /** * 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) */ #define TO_QUAD(pos) (((pos) / 3) * 3) struct board_element; typedef unsigned short int board_pos; typedef unsigned char element_value; /** * Simple definition of a board element. Vaild values range: 0-8 * Complexity describes how many possible values would be valid for an instance * of a board element. For example, if it could be either 1 or 5, it would have * a complexity of 2. If it could hold 4, 5 or 8, it would have a complexity * of 3 */ struct board_element { bool has_value : 1; /* Whether element has a decided value */ element_value value : 4; /* Value of element */ unsigned short potential : 9; /* Bitfield of possible values*/ unsigned char complexity : 4; /* Complexity tracker */ }; /** * 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 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 : 4; /* 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 */ struct board_element *links [81][20];/* All "connected", "adjacent" elements */ }; /** * Initialize metadata to a blank state */ void meta_init (struct metadata *meta); /* * Just generate board element adjacency links */ void board_make_links (struct board *board); /** * 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 * * NOTE: This marks an element as having a decided value */ void board_set ( struct board *board, board_pos x, board_pos y, element_value value ); /** * Mark a potential value of an element on the board * * NOTE: Marking an element with a decided value is undefined */ void board_mark ( struct board *board, board_pos x, board_pos y, element_value value ); /** * Removes a marking of a potential value of an element on the board * * NOTE: Unmarking an element with a decied value is undefined */ bool board_unmark ( struct board *board, board_pos x, board_pos y, element_value value ); /** * Removes a marking of a potential value of an element */ bool elem_unmark ( struct board_element *elem, element_value value ); /** * Get whether or not an element has a decided value */ bool board_has_value ( const struct board *board, board_pos x, board_pos y ); /** * Get definite value of a board element * * NOTE: Getting the definite value of an element without a value is undefined */ element_value board_get_value ( const struct board *board, board_pos x, board_pos y ); /** * Get whether or not a board element is marked with a particular value * * NOTE: Getting the mark state of an element with a value is undefined */ bool board_is_marked ( const struct board *board, board_pos x, board_pos y, element_value value ); /** * Get whether or not a given element is marked with a particular value */ bool elem_is_marked ( const struct board_element *elem, 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 */ bool board_is_valid (struct board *board); /** * Check row and column of given position and mark potential values that could * be set at that position and still leave the board in a valid state */ void board_update_marks ( struct board *board, board_pos x, board_pos y ); /** * Refreshes marks of all board elements without a value */ void board_update_all_marks (struct board *board); /** * Attempt to set value at the given position * If value can be placed, this updates all elements without a value on the * same row or column that do not have a value so that their markers reflect * the addition of this value and returns true * If value cannot be placed, just return false */ bool board_place ( struct board *board, board_pos x, board_pos y, element_value value ); /** * Place a speculative value. This allocates a duplicate board with the element * placed at the given location, if possible, while leaving the given board * untouched * * NOTE: If element cannot be placed, this returns NULL */ struct board * board_place_speculative ( const struct board *board, struct board *board_duplicate, board_pos x, board_pos y, element_value value ); /** * Recomputes board complexity by searching all elements on board for the * lowest complexity */ bool board_refresh_complexity (struct board *board); /** * Copy a board layout to another board */ void board_copy (const struct board *board_from, struct board *board_to);