From 06185b5b7f5d9f1d5d57627f86e286879be92e6e Mon Sep 17 00:00:00 2001 From: eug-vs Date: Mon, 23 Jan 2023 07:46:17 +0300 Subject: refactor: update occupancy in an optimal way --- src/board.rs | 35 ++++++++++++++++++++++++----------- 1 file changed, 24 insertions(+), 11 deletions(-) diff --git a/src/board.rs b/src/board.rs index ef8dde7..54efab9 100644 --- a/src/board.rs +++ b/src/board.rs @@ -201,6 +201,9 @@ impl Board { /// *Blindlessly* apply a move without any validation /// Move should be validated beforehand pub fn make_move(&mut self, mov: Move) -> Option { + let move_source_bb = mov.source.to_bitboard(); + let move_target_bb = mov.target.to_bitboard(); + // Remove existing piece (if any) from target square let captured_piece = match self.pieces .iter() @@ -208,7 +211,7 @@ impl Board { .find(|(piece_type, bitboard)| *bitboard & mov.target.to_bitboard() > 0) { Some((target_piece, _)) => { - self.pieces[target_piece] ^= mov.target.to_bitboard(); + self.pieces[target_piece] ^= move_target_bb; Some(PieceType::from(target_piece)) }, None => None, @@ -221,13 +224,15 @@ impl Board { .find(|(piece_type, bitboard)| *bitboard & mov.source.to_bitboard() > 0) { Some((source_piece, _)) => { - self.pieces[source_piece] ^= mov.source.to_bitboard(); - self.pieces[source_piece] |= mov.target.to_bitboard(); + self.pieces[source_piece] ^= move_source_bb; + self.occupancy ^= move_source_bb; + + self.pieces[source_piece] |= move_target_bb; + self.occupancy |= move_target_bb; }, None => panic!("Move is malformed: source piece not found"), }; - self.update_occupancy(); self.ply += 1; captured_piece @@ -235,15 +240,21 @@ impl Board { /// Completely reverse make_move as if it never happened pub fn unmake_move(&mut self, mov: Move, captured_piece: Option) { - // Remove source piece from target square + let move_source_bb = mov.source.to_bitboard(); + let move_target_bb = mov.target.to_bitboard(); + + // Move a piece from target square back to source square match self.pieces .iter() .enumerate() .find(|(piece_type, bitboard)| *bitboard & mov.target.to_bitboard() > 0) { Some((source_piece, _)) => { - self.pieces[source_piece] ^= mov.target.to_bitboard(); - self.pieces[source_piece] |= mov.source.to_bitboard(); + self.pieces[source_piece] ^= move_target_bb; + self.occupancy ^= move_target_bb; + + self.pieces[source_piece] |= move_source_bb; + self.occupancy |= move_source_bb; }, None => panic!("Trying to unmake move which was not made: no piece was found on target square"), }; @@ -251,12 +262,12 @@ impl Board { // Return captured piece to target square match captured_piece { Some(target_piece) => { - self.pieces[target_piece as usize] |= mov.target.to_bitboard(); + self.pieces[target_piece as usize] |= move_target_bb; + self.occupancy |= move_target_bb; }, None => {} } - self.update_occupancy(); self.ply -= 1; } } @@ -332,13 +343,14 @@ mod tests { fn test_make_move() { let fen = String::from("q1b2k2/5p1p/4p1pb/pPPp4/3N4/3nPB2/P2QKnR1/1R6 w - - 0 25"); let mut board = Board::from_FEN(fen); + let initial_board = board.clone(); board.print(); let black_move = Move { source: Square::F7, target: Square::F5 }; println!("\n{:?}", black_move); match board.make_move(black_move) { - Some(..) => panic!("No piece should have been captured"), + Some(..) => panic!("No piece should be captured"), None => {}, }; @@ -353,7 +365,7 @@ mod tests { match board.make_move(white_move) { Some(captured) => assert!(captured == PieceType::PawnBlack), - None => panic!("A piece should have been captured"), + None => panic!("A piece should be captured"), }; board.print(); @@ -361,6 +373,7 @@ mod tests { assert!(board.pieces[PieceType::PawnBlack as usize] & Square::A5.to_bitboard() == 0, "Target piece should be captured"); assert!(board.pieces[PieceType::Queen as usize] & Square::D2.to_bitboard() == 0); assert!(board.pieces[PieceType::Queen as usize] & Square::A5.to_bitboard() > 0); + assert_ne!(board.occupancy, initial_board.occupancy, "Occupancy should change after make_move"); assert!(board.ply == 2); } -- cgit v1.2.3