-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathdlx_sudoku.c
166 lines (146 loc) · 3.33 KB
/
dlx_sudoku.c
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
#include "dlx_sudoku.h"
int select_col0_by_row(int row)
{
return row / SUDOKU_RANK;
}
int select_col1_by_row(int row)
{
return SUDOKU_RANK * SUDOKU_RANK
+ row / (SUDOKU_RANK * SUDOKU_RANK) * SUDOKU_RANK
+ row % SUDOKU_RANK;
}
int select_col2_by_row(int row)
{
return SUDOKU_RANK * SUDOKU_RANK * 2
+ row % (SUDOKU_RANK * SUDOKU_RANK);
}
int select_col3_by_row(int row)
{
return SUDOKU_RANK * SUDOKU_RANK * 3
+ (row / (SUDOKU_RANK * SUDOKU_RANK * 3)) * SUDOKU_RANK * 3
+ (row % (SUDOKU_RANK * SUDOKU_RANK * 3))
% (SUDOKU_RANK * SUDOKU_RANK) / (SUDOKU_RANK * 3)
* SUDOKU_RANK
+ row % SUDOKU_RANK;
}
struct dlx_node *find_row_node(struct dlx_head *h, int row)
{
struct dlx_node *node;
struct dlx_node *col;
int col_id;
col_id = select_col0_by_row(row);
for (col = h->h.rx; col != &h->h; col = col->rx) {
if (col->colx->id == col_id) {
for (node = col->dx; node != col; node = node->dx) {
if (node->row_id == row) {
return node;
}
}
}
}
return NULL;
}
struct sudoku_dsr * str2sudoku(struct sudoku_dsr *sudoku, int w, int h,
char *str, size_t str_lenth)
{
size_t i;
int n = 0;
assert(sudoku != NULL && w > 0 && h > 0);
sudoku->w = w;
sudoku->h = h;
for (i = 0; i < str_lenth; i++) {
if (str[i] >= '1' && str[i] <= '9') {
*(sudoku->data + n) = str[i] - '0';
} else if (str[i] != '*'
&& str[i] != '.'
&& str[i] != '0'
&& str[i] != 'x'
&& str[i] != 'X') {
continue;
} else {
*(sudoku->data + n) = 0;
}
if (++n > w * h) {
break;
}
}
return sudoku;
}
int cell2row(int r, int c, int val)
{
assert(val > 0 && val < 10 && r >= 0 && r < 9 && c >= 0 && c < 9);
return r * SUDOKU_RANK * SUDOKU_RANK + c * SUDOKU_RANK + val - 1;
}
struct sudoku_cell *row2cell(int row, struct sudoku_cell *cell)
{
cell->r = row / (SUDOKU_RANK * SUDOKU_RANK);
cell->c = (row / SUDOKU_RANK) % SUDOKU_RANK;
cell->val = row % SUDOKU_RANK + 1;
return cell;
}
void set_sudoku_cell(struct sudoku_dsr *sudoku, const struct sudoku_cell *cell)
{
int r, c;
r = cell->r;
c = cell->c;
sudoku->data[SUDOKU_RANK * r + c] = cell->val;
}
void set_sudoku_cell_via_row(struct sudoku_dsr *sudoku, int row)
{
struct sudoku_cell cell;
row2cell(row, &cell);
set_sudoku_cell(sudoku, &cell);
}
int set_dlx_h_sudoku(struct dlx_head *h, const struct sudoku_dsr *sudoku,
struct dlx_node **save_node)
{
int i, j;
int val;
int row;
int n = 0;
assert(h && sudoku);
for (i = 0; i < sudoku->h; i++) {
for (j = 0; j < sudoku->w; j++) {
val = *(sudoku->data + i * sudoku->w + j);
if (val > 0) {
row = cell2row(i, j, val);
*(save_node + n) = find_row_node(h, row);
if (dlx_select_row(*(save_node + n)) < 0) {
return -1;
}
n++;
}
}
}
return 0;
}
void print_sudoku_str(const struct sudoku_dsr *sudoku)
{
int i, j;
for (i = 0; i < SUDOKU_RANK; i++) {
for (j = 0; j < SUDOKU_RANK; j++) {
printf("%d", *(sudoku->data + i * SUDOKU_RANK + j));
}
}
printf("\n");
}
void print_sudoku(const struct sudoku_dsr *sudoku)
{
int i, j;
for (i = 0; i < SUDOKU_RANK; i++) {
if (i % 3 == 0) {
printf("+-------+-------+-------+\n");
}
for (j = 0; j < SUDOKU_RANK; j++) {
if (j % 3 == 0) {
printf("| ");
}
printf("%d ", *(sudoku->data + i * SUDOKU_RANK + j));
if (j == SUDOKU_RANK - 1) {
printf("|");
}
}
printf("\n");
}
printf("+-------+-------+-------+\n");
}