2 * This program is free software; you can redistribute it and/or modify
3 * it under the terms of the GNU General Public License as published by
4 * the Free Software Foundation; either version 2 of the License, or
5 * (at your option) any later version.
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
12 * You should have received a copy of the GNU General Public License
13 * along with this program; if not, write to the Free Software
14 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
26 /* forward declarations of functions */
27 static int zwsearch_nonull(int beta
, int depth
, int ply
);
29 /* zero window search */
31 zwsearch(int beta
, int depth
, int ply
, int check
)
43 assert(beta
> -MATE_VALUE
&& beta
<= +MATE_VALUE
);
44 assert(depth
>= 0 && depth
<= MAX_PLY
);
45 assert(board_is_legal(&brd
));
50 if (reps() || evaluate_draw())
53 /* if we are too deep */
54 if (ply
>= MAX_PLY
- 1)
55 return evaluate(beta
- 1, beta
);
57 /* search for captures */
58 if (depth
<= 0 && !check
)
59 return quiesce(beta
- 1, beta
, 0, ply
, 0);
62 futility_prune
= FALSE
;
66 /* check transposition table */
67 if (e
.main_hash_enabled
) {
68 i
= hashtable_get(brd
.wtm
, depth
, ply
, &hash_moves
[ply
], &score
);
70 #ifdef STATISTIC_COUNTERS
71 counters
.transposition
++;
91 /* limited razoring at pre pre frontier */
92 if (e
.razoring
&& depth
== 3 && !check
&& (MATERIAL_SCORE
+ 875) < beta
) {
93 #ifdef STATISTIC_COUNTERS
99 /* NULL move pruning */
100 if (e
.null_pruning
&& do_null
&& !check
&& (depth
>= 2) && \
101 (MATERIAL_SCORE
>= beta
- 875) && \
102 !SCORE_IS_MATE(beta
) && (popcount(side_boards
[brd
.wtm
]) > 4)) {
104 int null_reduction
= 3;
107 brd
.move
= 0; /* NULL move */
110 memcpy(&game_history
.board
[game_history
.count
++], &brd
, \
111 sizeof(struct board_t
));
114 brd
.hash_key
^= hash_ep
[_FILE(brd
.ep
)];
115 brd
.hash_key
^= hash_side
;
121 /* search reduced depth */
122 if (depth
- null_reduction
- 1 > 0) {
123 score
= -zwsearch_nonull(1 - beta
, depth
- null_reduction
- 1, \
128 score
= -quiesce(-beta
, 1 - beta
, 0, ply
+ 1, 1);
136 #ifdef STATISTIC_COUNTERS
137 counters
.null_move_prunings
++;
143 /* futility pruning according to Ernst A. Heinz */
144 if (e
.futility_pruning
&& depth
< 3 && !check
) {
149 score
= evaluate(beta
- 1, beta
) + 475;
151 futility_max
= score
;
152 futility_prune
= TRUE
;
156 score
= evaluate(beta
- 1, beta
) + 175;
158 futility_max
= score
;
159 futility_prune
= TRUE
;
166 int futil_count
= 3 + (1 << (6 * depth
/ 8));
168 for (i
= 0; i
< moves
.count
; i
++) {
169 /* get move with largest score */
170 sort_moves(&moves
, i
, ply
);
171 if (!make_move(moves
.move
[i
], FALSE
))
174 is_check
= IS_CHECKED(brd
.wtm
);
176 if (futility_prune
&& !is_check
&& moves_count
&& !MOVE_IS_TACTICAL(moves
.move
[i
])) {
178 #ifdef STATISTIC_COUNTERS
179 counters
.futility_prunings
++;
186 /* examine if we can reduce or extend move */
187 extend
= moves_count
> NOLMR_MOVES
? \
188 get_extensions(moves
.move
[i
], check
, is_check
, depth
, ply
) : \
190 if (extend
< 0 && moves_count
>= futil_count
) extend
--;
192 score
= -zwsearch(1 - beta
, depth
+ extend
- 1, ply
+ 1, is_check
);
196 if (score
>= beta
&& extend
< 0) {
197 /* research with original depth */
198 score
= -zwsearch(1 - beta
, depth
- 1, ply
+ 1, is_check
);
206 #ifdef STATISTIC_COUNTERS
207 counters
.failed_high_total
++;
208 if (moves_count
== 1)
209 counters
.failed_high_first
++;
211 /* update history only for non tactical move */
212 if (!MOVE_IS_TACTICAL(moves
.move
[i
]))
213 history_store(moves
.move
[i
], depth
, ply
, beta
);
214 if (e
.main_hash_enabled
)
215 /* save in hashtable */
216 hashtable_put(brd
.wtm
, depth
, ply
, LOWER
, moves
.move
[i
],
223 beta
= check
? -MATE_VALUE
+ ply
+ 1 : 1;
225 if (e
.main_hash_enabled
)
226 /* save in hashtable */
227 hashtable_put(brd
.wtm
, depth
, ply
, UPPER
, 0, beta
- 1);
232 /* zero window search with some restrictions */
234 zwsearch_nonull(int beta
, int depth
, int ply
)
240 struct moves_t moves
;
242 assert(beta
> -MATE_VALUE
&& beta
<= +MATE_VALUE
);
243 assert(depth
>= 1 && depth
<= MAX_PLY
);
244 assert(!IS_CHECKED(brd
.wtm
));
245 assert(board_is_legal(&brd
));
254 for (i
= 0; i
< moves
.count
; i
++) {
255 /* get move with largest score */
256 sort_moves(&moves
, i
, ply
);
257 if (!make_move(moves
.move
[i
], FALSE
))
262 is_check
= IS_CHECKED(brd
.wtm
);
263 score
= -zwsearch(1 - beta
, depth
- 1, ply
+ 1, is_check
);
270 #ifdef STATISTIC_COUNTERS
271 counters
.failed_high_total
++;
272 if (moves_count
== 1)
273 counters
.failed_high_first
++;
280 /* FIXME: what should I return? */