aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--Cargo.toml2
-rw-r--r--benches/negamax.rs21
-rw-r--r--benches/search.rs20
-rw-r--r--src/board/engine.rs346
-rw-r--r--src/board/mod.rs42
-rw-r--r--src/main.rs43
-rw-r--r--src/square.rs32
7 files changed, 408 insertions, 98 deletions
diff --git a/Cargo.toml b/Cargo.toml
index f8eb87c..62a6499 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -14,5 +14,5 @@ name = "perft"
harness = false
[[bench]]
-name = "negamax"
+name = "search"
harness = false
diff --git a/benches/negamax.rs b/benches/negamax.rs
deleted file mode 100644
index 0c952d6..0000000
--- a/benches/negamax.rs
+++ /dev/null
@@ -1,21 +0,0 @@
-use std::{time::Instant, f32::INFINITY};
-use chessnost::board::Board;
-
-fn main() {
- let fen = String::from("r3k2r/p1ppqpb1/bn2pnp1/3PN3/1p2P3/2N2Q1p/PPPBBPPP/R3K2R w KQkq - 0 1");
- let mut board = Board::from_FEN(fen);
- board.ply = 0;
-
- let depth = 4;
- let start = Instant::now();
- let (score, pv) = board.negamax_search(-INFINITY, INFINITY, depth);
- println!("Negamax search (depth = {}) finished in {:?}: {:?}", depth, start.elapsed(), score);
- board.print();
- for mov in pv {
- println!("{:?}", mov);
- println!("Score for {:?}: {}", board.color(), board.evaluate(None));
- board.make_move(mov);
- board.print();
- }
- println!("Score for {:?}: {}", board.color(), board.evaluate(None));
-}
diff --git a/benches/search.rs b/benches/search.rs
new file mode 100644
index 0000000..41027f9
--- /dev/null
+++ b/benches/search.rs
@@ -0,0 +1,20 @@
+use std::{time::{Instant, Duration}, f32::INFINITY};
+use chessnost::board::Board;
+
+fn main() {
+ let fen = String::from("r3k2r/p1ppqpb1/bn2pnp1/3PN3/1p2P3/2N2Q1p/PPPBBPPP/R3K2R w KQkq - ");
+ let mut board = Board::from_FEN(fen);
+
+ let start = Instant::now();
+ let (score, pv) = board.iterative_deepening(6, Duration::from_secs(15));
+
+ println!("Finished in {:?}: score={:?} {:?}", start.elapsed(), score, pv);
+ board.print();
+ for mov in pv {
+ println!("Score for {:?} is now: {}", board.color(), board.quiscence(-INFINITY, INFINITY));
+ println!("{:?}", mov);
+ board.make_move(mov);
+ board.print();
+ }
+ println!("Score for {:?} is now: {}", board.color(), board.quiscence(-INFINITY, INFINITY));
+}
diff --git a/src/board/engine.rs b/src/board/engine.rs
index 8852c1f..6a0db61 100644
--- a/src/board/engine.rs
+++ b/src/board/engine.rs
@@ -5,6 +5,8 @@ use super::ttable::{NodeType, TranspositionTableItem};
static A_FILE: Bitboard = 0x0101010101010101;
+const VALUE_WIN: f32 = 20_000.0;
+
#[derive(Debug, Default, PartialEq)]
pub struct PerftResult {
leaf_nodes: u64,
@@ -81,41 +83,39 @@ impl Board {
}
/// Compute material advantage relative to the current player
- pub fn material_advantage(&self) -> f32 {
- let mut eval = 0f32;
- for (piece_index, bitboard) in self.pieces.iter().enumerate() {
+ pub fn material(&self, color: Color) -> f32 {
+ let mut material = 0f32;
+ for (piece_index, bitboard) in self.pieces_by_color(color).iter().enumerate() {
let piece_type = PieceType::from(piece_index);
- let sign = if Color::from_piece(piece_type) == self.color() {
- 1.
- } else {
- -1.
- };
-
- eval += sign * match piece_type {
+ material += match piece_type {
PieceType::Pawn => {
- serialize_bitboard(*bitboard).iter().fold(0., |acc, square| {
- acc + match (*square).rank() {
- 6 => 3.,
- 5 => 2.,
- _ => 1.,
+ match color {
+ Color::White => {
+ serialize_bitboard(*bitboard).iter().fold(0., |acc, square| {
+ acc + match (*square).rank() {
+ 6 => 3.,
+ 5 => 2.,
+ _ => 1.,
+ }
+ })
+ },
+ Color::Black => {
+ serialize_bitboard(*bitboard).iter().fold(0., |acc, square| {
+ acc + match (*square).rank() {
+ 1 => 3.,
+ 2 => 2.,
+ _ => 1.,
+ }
+ })
}
- })
- }
- PieceType::PawnBlack => {
- serialize_bitboard(*bitboard).iter().fold(0., |acc, square| {
- acc + match (*square).rank() {
- 1 => 3.,
- 2 => 2.,
- _ => 1.,
- }
- })
+ }
}
_ => {
piece_type.static_eval() * pop_count(*bitboard) as f32
}
};
}
- eval
+ material
}
/// Returns sum of the doubled, blocked and isolated pawns
@@ -154,20 +154,50 @@ impl Board {
result
}
+ /// Returns the weighted sum of distances from attacking pieces to a king
+ /// The higher this value, the safer is the king
+ pub fn king_tropism(&self, color: Color) -> f32 {
+ let mut result = 0.0;
+
+ let king_square = bitscan(match color {
+ Color::White => self.pieces[PieceType::King as usize],
+ Color::Black => self.pieces[PieceType::KingBlack as usize],
+ });
+
+ for (piece_type, bitboard) in self.pieces_by_color(color.flip()).iter().enumerate() {
+ if piece_type != PieceType::King as usize && piece_type != PieceType::Pawn as usize {
+ for square in serialize_bitboard(*bitboard) {
+ let distance =
+ (king_square.rank() as f32 - square.rank() as f32).abs() +
+ (king_square.file() as f32 - square.file() as f32).abs();
+
+ result += distance / PieceType::from(piece_type).static_eval();
+ }
+ }
+ }
+ result
+ }
+
/// Evaluate a position relative to the current player
pub fn evaluate(&self, precomputed_mobility: Option<f32>) -> f32 {
- let opponent_mobility = self.mobility(self.color().flip());
+ let color = self.color();
+ let opponent_color = color.flip();
+
+ let opponent_mobility = self.mobility(opponent_color);
let player_mobility = match precomputed_mobility {
Some(m) => m,
- None => self.mobility(self.color()),
+ None => self.mobility(color),
};
let mobility_advantage = player_mobility - opponent_mobility as f32;
- let material_advantage = self.material_advantage();
+ let opponent_material = self.material(opponent_color);
+ let material_advantage = self.material(color) - opponent_material;
- let pawn_structure_penalty = self.pawn_structure_penalty(self.color()) - self.pawn_structure_penalty(self.color().flip());
+ let pawn_structure_penalty = self.pawn_structure_penalty(color) - self.pawn_structure_penalty(opponent_color);
- material_advantage + 0.1 * mobility_advantage - 0.5 * pawn_structure_penalty
+ let king_tropism_penalty = self.king_tropism(color) - self.king_tropism(opponent_color);
+
+ material_advantage + 0.1 * mobility_advantage - 0.4 * pawn_structure_penalty + king_tropism_penalty * opponent_material / 150.0
}
/// Evaluate move for move ordering, prioritizing efficient captures
@@ -187,11 +217,26 @@ impl Board {
0.0
}
- pub fn order_moves(&self, moves: &mut Vec<Move>) {
- moves.sort_unstable_by(|a, b| {
- let a_eval = self.eval_move(*a);
- let b_eval = self.eval_move(*b);
- if a_eval == 0.0 && b_eval == 0.0 {
+ pub fn hash_move(&self) -> Option<Move> {
+ match self.transposition_table[(self.hash % TTABLE_SIZE) as usize] {
+ Some(item) => {
+ if item.hash == self.hash {
+ return Some(item.best_move)
+ }
+ None
+ }
+ None => None
+ }
+ }
+
+ pub fn order_moves(&self, moves: Vec<Move>) -> Vec<Move> {
+ let mut moves_with_eval: Vec<(Move, f32)> = moves
+ .iter()
+ .map(|m| (*m, self.eval_move(*m)))
+ .collect();
+
+ moves_with_eval.sort_unstable_by(|(a, a_eval), (b, b_eval)| {
+ if *a_eval == 0.0 && *b_eval == 0.0 {
// Prioritize equal captures over non-captures
if a.is_tactical() && !b.is_tactical() {
return Ordering::Less
@@ -200,8 +245,10 @@ impl Board {
return Ordering::Greater
}
}
- a_eval.total_cmp(&b_eval).reverse()
+ a_eval.total_cmp(b_eval).reverse()
});
+
+ moves_with_eval.iter_mut().map(|(m, _)| *m).collect()
}
pub fn negamax_search(&mut self, mut alpha: f32, beta: f32, depth_left: u8, parent_killers: &mut Vec<Move>, deadline: Instant) -> (f32, Vec<Move>) {
@@ -210,7 +257,12 @@ impl Board {
let color = self.color();
let mut moves = self.generate_pseudolegal_moves(color);
- self.order_moves(&mut moves);
+ moves = self.order_moves(moves);
+
+ match self.hash_move() {
+ Some(mov) => moves.insert(0, mov),
+ None => {},
+ }
let loosing_capture_index = match moves.iter().position(|m| {
m.is_tactical() && self.eval_move(*m) < 0.0
@@ -219,7 +271,7 @@ impl Board {
None => 0,
};
- // Insert killer moves after winning and equal captures
+ // Insert killer moves (from previous siblings) after winning and equal captures
for mov in &mut *parent_killers {
// Validate that killer piece still exists
if mov.source.to_bitboard() & self.color_occupancy(color) > 0 {
@@ -227,19 +279,11 @@ impl Board {
}
}
- match self.transposition_table[(self.hash % TTABLE_SIZE) as usize] {
- Some(item) => {
- if item.hash == self.hash {
- moves.insert(0, item.best_move);
- }
- }
- None => {},
- }
-
if depth_left == 0 {
return (self.quiscence(alpha, beta), principal_variation);
}
+ let mut legal_move_found = false;
for mov in moves {
let ep_target_before = self.ep_target.clone();
let castling_rights_before = self.castling_rights.clone();
@@ -247,6 +291,7 @@ impl Board {
let captured_piece = self.make_move(mov);
if !self.is_king_in_check(color) {
+ legal_move_found = true;
let (mut score, mut subtree_pv) = self.negamax_search(-beta, -alpha, depth_left - 1, &mut killer_moves, deadline);
score *= -1.;
self.unmake_move(mov, captured_piece, ep_target_before, castling_rights_before, hash_before);
@@ -261,8 +306,6 @@ impl Board {
});
if mov.kind == MoveKind::Quiet {
- // println!("Killer {:?} found at depth {}", mov, depth_left);
- // self.print();
match parent_killers.iter().find(|m| **m == mov) {
None => parent_killers.push(mov),
Some(..) => {},
@@ -299,25 +342,26 @@ impl Board {
// Could not finish in time, return what we have so far
if Instant::now() > deadline {
- println!("Returning early!");
return (alpha, principal_variation)
}
}
+ if !legal_move_found {
+ if self.is_king_in_check(color) {
+ return (-VALUE_WIN, principal_variation);
+ }
+ }
+
(alpha, principal_variation)
}
pub fn quiscence(&mut self, mut alpha: f32, beta: f32) -> f32 {
let color = self.color();
let mut moves = self.generate_pseudolegal_moves(color);
- self.order_moves(&mut moves);
+ moves = self.order_moves(moves);
- match self.transposition_table[(self.hash % TTABLE_SIZE) as usize] {
- Some(item) => {
- if item.hash == self.hash {
- moves.insert(0, item.best_move);
- }
- }
+ match self.hash_move() {
+ Some(mov) => moves.insert(0, mov),
None => {},
}
@@ -363,30 +407,42 @@ impl Board {
let mut depth = 1;
let mut alpha = -INFINITY;
let mut beta = INFINITY;
+ let window_size = 0.5;
+ let mut gradual_widening_counter = 0;
let mut root_killers: Vec<Move> = Vec::new();
- let window_size = 0.25;
while depth <= max_depth {
+ println!("\nSearching depth({}) in the window {:?}", depth, (alpha, beta));
let search_result = self.negamax_search(alpha, beta, depth, &mut root_killers, deadline);
- println!("Finished depth({}) {:?} [{:?} left]", depth, search_result, deadline - Instant::now());
+
+ if search_result.0.abs() >= VALUE_WIN {
+ return search_result
+ }
if Instant::now() > deadline {
+ println!("Aborting...");
break;
}
+
+ println!("Finished depth({}) {:?} [{:?} left]", depth, search_result, deadline - Instant::now());
+
if search_result.1.len() > 0 {
depth += 1;
+ gradual_widening_counter = 0;
alpha = search_result.0 - window_size;
beta = search_result.0 + window_size;
} else if search_result.0 <= alpha { // Alpha-cutoff
println!("Alpha cutoff {} <= {:?}", search_result.0, (alpha, beta));
+ gradual_widening_counter += 1;
beta = alpha;
- alpha = search_result.0 - window_size * 4.0;
+ alpha = search_result.0 - window_size * 2.0f32.powi(gradual_widening_counter);
continue;
} else if search_result.0 >= beta { // Beta-cutoff
println!("Beta cutoff {:?} <= {}", (alpha, beta), search_result.0);
+ gradual_widening_counter += 1;
alpha = beta;
- beta = search_result.0 + window_size * 4.0;
+ beta = search_result.0 + window_size * 2.0f32.powi(gradual_widening_counter);
continue;
} else {
panic!("Can this ever be possible? (probably not)");
@@ -405,7 +461,9 @@ impl Board {
#[cfg(test)]
mod tests {
- use crate::board::{Board, engine::PerftResult};
+ use std::time::Duration;
+ use crate::{board::{Board, engine::PerftResult, Color}, square::Square, moves::{Move, MoveKind}};
+ use super::VALUE_WIN;
#[test]
fn perft() {
@@ -441,9 +499,167 @@ mod tests {
}
#[test]
- fn material_advantage() {
+ fn material() {
let board = Board::new();
- assert_eq!(board.material_advantage(), 0.0, "Material advantage should be 0 at starting position");
+ assert_eq!(board.material(Color::Black), board.material(Color::White));
+
+ }
+
+ #[test]
+ fn checkmate() {
+ let fen = String::from("2kr1b1r/pp1npppp/2p1bn2/7q/5B2/2NB1Q1P/PPP1N1P1/2KR3R w - - 0 1");
+ let mut board = Board::from_FEN(fen);
+ let (score, pv) = board.iterative_deepening(8, Duration::from_secs(15));
+
+ assert_eq!(score, VALUE_WIN);
+ assert_eq!(pv, vec![
+ Move { source: Square::F3, target: Square::C6, kind: MoveKind::Capture },
+ Move { source: Square::B7, target: Square::C6, kind: MoveKind::Capture },
+ Move { source: Square::D3, target: Square::A6, kind: MoveKind::Quiet },
+ ]);
+ }
+
+ #[test]
+ fn stupid_knight_sac() {
+ let fen = String::from("r3k1r1/pp3ppp/1q6/2ppPn2/6P1/1PPP1P2/P1N3KP/R2QR3 b - - 0 18");
+ let mut board = Board::from_FEN(fen);
+ board.ply += 1; // TODO: remove me when FEN parsing includes side to move
+
+ let (_, pv) = board.iterative_deepening(6, Duration::from_secs(60));
+ assert_eq!(
+ pv[0],
+ Move { source: Square::F5, target: Square::H4, kind: MoveKind::Quiet },
+ "You should save this poor knight from danger!"
+ );
+ }
+ #[test]
+ fn weird_bishop_sac() {
+ let fen = String::from("r1b1k1nr/p4pp1/1pp1p3/4n2p/1b1qP3/1B1P3N/PPPBQPPP/RN2K2R w KQkq - 7 10");
+ let mut board = Board::from_FEN(fen);
+
+ let (_, pv) = board.iterative_deepening(5, Duration::from_secs(60));
+ assert_eq!(
+ pv[0],
+ Move { source: Square::C2, target: Square::C3, kind: MoveKind::Quiet },
+ "You should fork this bastard!"
+ );
}
+
+ mod evaluation {
+ use crate::{moves::{Move, MoveKind}, square::Square};
+
+ use super::*;
+
+ #[test]
+ fn initial_eval() {
+ let board = Board::new();
+ assert_eq!(board.evaluate(None), 0.0);
+ }
+
+ #[test]
+ fn king_tropism() {
+ let mut board = Board::new();
+ board.make_move(Move { source: Square::D1, target: Square::F5, kind: MoveKind::Quiet });
+ let score = board.evaluate(None);
+ board.print();
+ println!("Score {}", score);
+
+ assert!(score < 0.0);
+ assert!(score > -1.0);
+ }
+
+ #[test]
+ fn white_winning() {
+ let fen = String::from("8/5pk1/6p1/R4b1p/3P4/1P2N3/P1r2PPP/R5K1 b - - 1 27");
+ let board = Board::from_FEN(fen);
+ let score = board.evaluate(None);
+ board.print();
+ println!("Score {}", score);
+
+ assert!(score > 7.0);
+ }
+
+ #[test]
+ fn black_winning() {
+ let fen = String::from("8/p7/1k4K1/8/4P3/8/PP5r/8 b - - 1 38");
+ let board = Board::from_FEN(fen);
+ let score = board.evaluate(None);
+ board.print();
+ println!("Score {}", score);
+
+ assert!(score < -3.0);
+ }
+
+ #[test]
+ fn encourage_center_pawns() {
+ let score1 = {
+ let fen = String::from("rnbqkbnr/pppp1ppp/8/4p3/4P3/8/PPPP1PPP/RNBQKBNR w KQkq - 0 2");
+ let board = Board::from_FEN(fen);
+ let score = board.evaluate(None);
+ board.print();
+ println!("Score {}", score);
+ score
+ };
+
+ let score2 = {
+ let fen = String::from("rnbqkbnr/pppp1ppp/8/4p3/2P5/8/PP1PPPPP/RNBQKBNR w KQkq - 0 2");
+ let board = Board::from_FEN(fen);
+ let score = board.evaluate(None);
+ board.print();
+ println!("Score {}", score);
+ score
+ };
+
+ assert!(score1 > score2);
+ }
+
+ #[test]
+ fn discourage_edge_knights() {
+ let score1 = {
+ let fen = String::from("r1bqkbnr/pppp1ppp/2n5/4p3/4P3/5N2/PPPP1PPP/RNBQKB1R w KQkq - 2 3");
+ let board = Board::from_FEN(fen);
+ let score = board.evaluate(None);
+ board.print();
+ println!("Score {}", score);
+ score
+ };
+
+ let score2 = {
+ let fen = String::from("r1bqkbnr/pppp1ppp/2n5/4p3/4P3/7N/PPPP1PPP/RNBQKB1R w KQkq - 2 3");
+ let board = Board::from_FEN(fen);
+ let score = board.evaluate(None);
+ board.print();
+ println!("Score {}", score);
+ score
+ };
+
+ assert!(score1 > score2);
+ }
+
+ #[test]
+ fn mirrored_evaluation() {
+ let score1 = {
+ let fen = String::from("r3k2r/Pppp1ppp/1b3nbN/nP6/BBP1P3/q4N2/Pp1P2PP/R2Q1RK1 w kq - 0 1");
+ let board = Board::from_FEN(fen);
+ let score = board.evaluate(None);
+ board.print();
+ println!("Score {}", score);
+ score
+ };
+
+ let score2 = {
+ let fen = String::from("r2q1rk1/pP1p2pp/Q4n2/bbp1p3/Np6/1B3NBn/pPPP1PPP/R3K2R b KQ - 0 1 ");
+ let mut board = Board::from_FEN(fen);
+ board.ply += 1; // TODO: remove me when FEN parsing includes side to move
+ let score = board.evaluate(None);
+ board.print();
+ println!("Score {}", score);
+ score
+ };
+
+ assert_eq!(score1.abs(), score2.abs());
+ }
+ }
+
}
diff --git a/src/board/mod.rs b/src/board/mod.rs
index 06b61bb..8f5fbde 100644
--- a/src/board/mod.rs
+++ b/src/board/mod.rs
@@ -1,3 +1,5 @@
+use std::io::{stdin, stdout, Write};
+
use rand::{rngs::StdRng,SeedableRng,Rng};
use crate::{bitboard::{Bitboard, serialize_bitboard, bitscan, pop_count}, moves::{Move, MoveKind}, attacks::Attacks, square::Square};
@@ -31,7 +33,6 @@ pub struct Board {
attacks: Attacks,
}
-
#[derive(Debug, Clone, Copy, PartialEq, Eq, num_enum::FromPrimitive)]
#[repr(usize)]
pub enum PieceType {
@@ -58,11 +59,11 @@ impl PieceType {
// Return the price of the peice
pub fn static_eval(&self) -> f32 {
match self.without_color() {
- PieceType::Pawn => 1.,
- PieceType::Bishop => 3.,
- PieceType::Knight => 3.,
- PieceType::Rook => 4.5,
- PieceType::Queen => 9.,
+ PieceType::Pawn => 1.0,
+ PieceType::Bishop => 3.3,
+ PieceType::Knight => 3.2,
+ PieceType::Rook => 5.0,
+ PieceType::Queen => 9.0,
PieceType::King => 0.,
_ => panic!("Piece should be without color"),
}
@@ -146,6 +147,32 @@ impl Board {
Self::from_FEN(default_fen)
}
+ pub fn read_move(&self) -> Result<Move, String> {
+ print!("\nEnter a move: ");
+ stdout().flush().unwrap();
+ let mut s = String::new();
+ stdin().read_line(&mut s).unwrap();
+ let chars = &mut s.chars();
+
+ let source = match Square::from_notation(chars) {
+ Ok(s) => s,
+ Err(e) => return Err(e),
+ };
+ let target = match Square::from_notation(chars) {
+ Ok(s) => s,
+ Err(e) => return Err(e),
+ };
+
+ let moves = self.generate_pseudolegal_moves(self.color());
+
+ let mov = match moves.iter().find(|m| m.source == source && m.target == target) {
+ Some(m) => *m,
+ None => return Err(String::from("Move is not valid")),
+ };
+
+ Ok(mov)
+ }
+
/// Color to move at this ply
pub fn color(&self) -> Color {
Color::from(self.ply as u8 % 2)
@@ -493,7 +520,8 @@ impl Board {
}
PieceType::Queen => {
for source in serialize_bitboard(*piece) {
- mobility += pop_count(self.attacks.queen(self.occupancy, source) & (empty | opponent_occupancy)) as f32;
+ // Scale down mobility because we don't want our queen to be rushing too much
+ mobility += pop_count(self.attacks.queen(self.occupancy, source) & (empty | opponent_occupancy)) as f32 / 3.0;
}
}
incorrect_type => panic!("Incorrect piece type: {:?}", incorrect_type),
diff --git a/src/main.rs b/src/main.rs
index 3e8e461..13af134 100644
--- a/src/main.rs
+++ b/src/main.rs
@@ -3,17 +3,52 @@ use std::{f32::INFINITY, time::{Duration, Instant}};
use chessnost::board::Board;
fn main() {
+ let increment = Duration::from_secs(0);
+ let mut time_left = Duration::from_secs(60 * 10);
+ let manual_decrement = Duration::from_secs(7); // Time to sync moves with the game
+
let mut board = Board::new();
+ board.print();
+
loop {
- let start = Instant::now();
- let (score, pv) = board.iterative_deepening(Duration::from_secs(4));
- println!("Finished in {:?}: score={:?}", start.elapsed(), score);
+ time_left += increment;
+ time_left -= manual_decrement;
+
+ let mov = match board.read_move() {
+ Ok(m) => m,
+ Err(e) => {
+ println!("Error: {}", e);
+ continue;
+ }
+ };
+ print!("{:?}", mov);
+ board.make_move(mov);
+ board.print();
+
+ let allowed_move_duration = time_left / 20;
+ let max_depth = 6 + (board.ply as i32 / 15) as u8;
+ println!("~{:?} left, allocating {:?} for a move", time_left, allowed_move_duration);
+
+ let move_start = Instant::now();
+ let (score, pv) = board.iterative_deepening(max_depth, allowed_move_duration);
+ let elapsed = move_start.elapsed();
+
+ if time_left >= elapsed {
+ time_left -= elapsed;
+ } else {
+ println!("You are probably out of time. I will keep going anyway...");
+ }
+
+ println!("Finished in {:?}: score={:?}", elapsed, score);
let mov = pv[0];
println!("{:?}", mov);
board.make_move(mov);
board.print();
- println!("Score for {:?} is now: {} (material advantage={})", board.color(), board.quiscence(-INFINITY, INFINITY), board.material_advantage());
+ println!("Score for {:?} is now: {}", board.color(), board.quiscence(-INFINITY, INFINITY));
+
+ println!("\nPondering for 3 seconds...");
+ board.iterative_deepening(max_depth - 1, Duration::from_secs(3));
}
}
diff --git a/src/square.rs b/src/square.rs
index efa682c..e518a92 100644
--- a/src/square.rs
+++ b/src/square.rs
@@ -1,3 +1,5 @@
+use std::str::Chars;
+
use crate::bitboard::Bitboard;
/// Aliases to board square indexes
@@ -49,6 +51,36 @@ impl Square {
pub fn east_one(&self) -> Self {
Self::from(*self as u8 + 1)
}
+
+ pub fn from_notation(chars: &mut Chars) -> Result<Self, String> {
+ let file = match chars.next() {
+ Some(ch) => {
+ match ch {
+ 'a' => 0,
+ 'b' => 1,
+ 'c' => 2,
+ 'd' => 3,
+ 'e' => 4,
+ 'f' => 5,
+ 'g' => 6,
+ 'h' => 7,
+ _ => return Err(String::from("Incorrect file!"))
+ }
+ },
+ None => return Err(String::from("Missing file"))
+ };
+ let rank = match chars.next() {
+ Some(ch) => {
+ match ch.to_digit(10) {
+ Some(digit) => digit - 1,
+ None => return Err(String::from("Incorrect rank"))
+ }
+ }
+ None => return Err(String::from("Missing rank"))
+ };
+
+ Ok(Self::from_coords(rank as u8, file))
+ }
}