diff options
| -rw-r--r-- | src/board/io.rs | 9 | ||||
| -rw-r--r-- | src/board/mod.rs | 43 | ||||
| -rw-r--r-- | src/board/zobrist.rs | 79 | 
3 files changed, 88 insertions, 43 deletions
| diff --git a/src/board/io.rs b/src/board/io.rs index cb73c52..b5cd95e 100644 --- a/src/board/io.rs +++ b/src/board/io.rs @@ -1,9 +1,8 @@  use std::io::{stdin, stdout, Write}; -use rand::{rngs::StdRng,SeedableRng,Rng};  use crate::{bitboard::Bitboard, attacks::Attacks, moves::Move, square::Square}; -use super::{Board, PieceType, ttable::TTABLE_SIZE}; +use super::{Board, PieceType, ttable::TTABLE_SIZE, zobrist::Zobrist};  const PIECE_CHARS: [&str; 12] = [      "♟︎", "♞", "♝", "♜", "♛", "♚", @@ -91,8 +90,6 @@ impl IO for Board {              }          } -        let mut rng = StdRng::seed_from_u64(228); -        let zobrist_seed = [(); 781].map(|_| rng.gen());          let mut board = Self {              piece_sets, @@ -103,10 +100,10 @@ impl IO for Board {              ep_target: None, // TODO: parse from FEN              hash: 0,              transposition_table: vec![None; TTABLE_SIZE as usize], -            zobrist_seed, +            zobrist_seed: Board::seed(),          };          board.update_occupancy(); -        board.update_zobrist_hash(); +        board.compute_hash();          board      } diff --git a/src/board/mod.rs b/src/board/mod.rs index f7016d3..854ce0e 100644 --- a/src/board/mod.rs +++ b/src/board/mod.rs @@ -1,10 +1,12 @@  use crate::{bitboard::{Bitboard, BitboardFns}, moves::{Move, MoveKind}, attacks::Attacks, square::Square, board::io::IO}; -use self::ttable::{TranspositionTable, TTABLE_SIZE}; +use self::{ttable::{TranspositionTable, TTABLE_SIZE}, zobrist::ZobristSeed};  pub mod io; +mod zobrist;  mod engine;  mod ttable; +#[derive(Debug, Clone, Copy)]  pub enum CastlingSide {      King,      Queen, @@ -26,7 +28,7 @@ pub struct Board {      // TODO: remove      transposition_table: TranspositionTable, -    zobrist_seed: [u64; 781], +    zobrist_seed: ZobristSeed,      attacks: Attacks,  } @@ -99,39 +101,6 @@ impl Board {          self.pieces_by_color(color).iter().fold(0, |acc, bitboard| acc | bitboard)      } -    /// Compute and store zobrist hash of the current position -    /// https://www.chessprogramming.org/Zobrist_Hashing -    fn update_zobrist_hash(&mut self) { -        self.hash = 0; - -        if self.color() == Color::Black { -            self.hash ^= match self.zobrist_seed.last() { -                Some(x) => x, -                None => panic!("Something is wrong with zobrist seed list"), -            }; -        } - -        for (piece_id, bitboard) in self.piece_sets.iter().enumerate() { -            for square in bitboard.serialize() { -                self.hash ^= self.zobrist_seed[piece_id * 64 + square as usize]; -            } -        } - -        for color in 0..2 { -            for castle_side in 0..2 { -                if self.castling_rights[color][castle_side] { -                    self.hash ^= self.zobrist_seed[(12 * 64) + color * 2 + castle_side]; -                } -            } -        } - -        match self.ep_target { -            Some(square) => { -                self.hash ^= self.zobrist_seed[(12 * 64 + 4) + square.file() as usize]; -            }, -            None => {}, -        } -    }      fn ep_bitboard(&self) -> Bitboard { @@ -731,7 +700,7 @@ impl Color {  #[cfg(test)]  mod tests {      use super::*; -    use crate::{bitboard::BitboardFns, square::Square}; +    use crate::{bitboard::BitboardFns, square::Square, board::zobrist::Zobrist};      #[test]      fn square_enum() { @@ -813,7 +782,7 @@ mod tests {          board.print();          let hash = board.hash; -        board.update_zobrist_hash(); +        board.compute_hash();          assert_eq!(hash, board.hash, "Hash should be correctly updated after move");          assert!(board.piece_sets[PieceType::PawnBlack as usize] & Square::F7.to_bitboard() == 0); diff --git a/src/board/zobrist.rs b/src/board/zobrist.rs new file mode 100644 index 0000000..e282074 --- /dev/null +++ b/src/board/zobrist.rs @@ -0,0 +1,79 @@ +use rand::{rngs::StdRng,SeedableRng,Rng}; + +use crate::{square::Square, bitboard::BitboardFns}; + +use super::{Board, Color, PieceType, CastlingSide}; + + +const TOTAL_PIECES:          usize = 12; +const TOTAL_SQUARES:         usize = 64; +const TOTAL_FILES:           usize = 8; +const TOTAL_CASTLING_RIGHTS: usize = 4; +const ZOBRIST_SEED_SIZE:     usize = TOTAL_SQUARES * TOTAL_PIECES + TOTAL_CASTLING_RIGHTS + TOTAL_FILES + 1; + +pub type ZobristSeed = [u64; ZOBRIST_SEED_SIZE]; + +pub trait Zobrist { +    fn seed() -> ZobristSeed; + +    /// Compute store zobrist hash of the current position +    /// https://www.chessprogramming.org/Zobrist_Hashing +    fn compute_hash(&mut self) -> (); + +    fn zobrist_toggle_piece(&mut self, piece_type: PieceType, square: Square) -> (); +    fn zobrist_toggle_castling_right(&mut self, color: Color, side: CastlingSide) -> (); +    fn zobrist_toggle_ep_square(&mut self, ep_square: Square) -> (); +    fn zobrist_toggle_color(&mut self) -> (); +} + +impl Zobrist for Board { +    fn seed() -> ZobristSeed { +        let mut rng = StdRng::seed_from_u64(228); +        [(); ZOBRIST_SEED_SIZE].map(|_| rng.gen()) +    } + +    fn compute_hash(&mut self) -> () { +        self.hash = 0; + +        for piece_id in 0..self.piece_sets.len() { +            for square in self.piece_sets[piece_id].serialize() { +                self.zobrist_toggle_piece(PieceType::from(piece_id), square); +            } +        } +        for color in [Color::White, Color::Black] { +            for castle_side in [CastlingSide::King, CastlingSide::Queen] { +                if self.castling_rights[color as usize][castle_side as usize] { +                    self.zobrist_toggle_castling_right(color, castle_side) +                } +            } +        } + +        match self.ep_target { +            Some(square) => self.zobrist_toggle_ep_square(square), +            None => {}, +        } + +        if self.color() == Color::Black { +            self.zobrist_toggle_color(); +        } +    } + +    fn zobrist_toggle_piece(&mut self, piece_type: PieceType, square: Square) -> () { +        self.hash ^= self.zobrist_seed[piece_type as usize * TOTAL_SQUARES + square as usize]; +    } + +    fn zobrist_toggle_castling_right(&mut self, color: Color, side: CastlingSide) -> () { +        self.hash ^= self.zobrist_seed[(TOTAL_PIECES * TOTAL_SQUARES) + color as usize * 2 + side as usize]; +    } + +    fn zobrist_toggle_ep_square(&mut self, ep_square: Square) -> () { +        self.hash ^= self.zobrist_seed[(TOTAL_PIECES * TOTAL_SQUARES + TOTAL_CASTLING_RIGHTS) + ep_square.file() as usize]; +    } + +    fn zobrist_toggle_color(&mut self) -> () { +        self.hash ^= match self.zobrist_seed.last() { +            Some(x) => x, +            None => panic!("Something is wrong with zobrist seed list"), +        }; +    } +} | 
