aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/attacks.rs91
-rw-r--r--src/board.rs35
2 files changed, 74 insertions, 52 deletions
diff --git a/src/attacks.rs b/src/attacks.rs
index 141ba44..a8daec9 100644
--- a/src/attacks.rs
+++ b/src/attacks.rs
@@ -1,9 +1,10 @@
-use crate::bitboard::{Bitboard, ls1b, pop_count};
+use crate::bitboard::{Bitboard, ls1b};
static NOT_A_FILE: Bitboard = 0xFEFEFEFEFEFEFEFE;
static NOT_B_FILE: Bitboard = 0xFDFDFDFDFDFDFDFD;
static NOT_G_FILE: Bitboard = 0xBFBFBFBFBFBFBFBF;
static NOT_H_FILE: Bitboard = 0x7F7F7F7F7F7F7F7F;
+static B_FILE: Bitboard = 0x0202020202020202;
/// An array where N-th item is an attack bitboard
/// of a piece on N-th square
@@ -26,20 +27,33 @@ pub struct Attacks {
pub knight: AttackTable,
pub king: AttackTable,
pub first_rank_attacks: FirstRankAttacks,
+
+ // pub nort_ray: AttackTable,
+ // pub noea_ray: AttackTable,
+ // pub east_ray: AttackTable,
+ // pub soea_ray: AttackTable,
+ // pub sout_ray: AttackTable,
+ // pub sowe_ray: AttackTable,
+ // pub west_ray: AttackTable,
+ // pub nowe_ray: AttackTable,
}
+#[allow(unused)]
impl Attacks {
pub fn new() -> Self {
- let mut knight = [0; 64];
- let mut king = [0; 64];
- let mut first_rank_attacks = [[0; 8]; 256];
- Self::precompute_knight_attacks(&mut knight);
- Self::precompute_king_attacks(&mut king);
- Self::precompute_first_rank_attacks(&mut first_rank_attacks);
- Self { knight, king, first_rank_attacks }
+ let knight = Self::precompute_knight_attacks();
+ let king = Self::precompute_king_attacks();
+ let first_rank_attacks = Self::precompute_first_rank_attacks();
+
+ Self {
+ knight,
+ king,
+ first_rank_attacks,
+ }
}
- fn precompute_knight_attacks(attacks: &mut AttackTable) {
+ fn precompute_knight_attacks() -> AttackTable {
+ let mut attacks = [0; 64];
for index in 0..64 {
let position = 1u64 << index;
attacks[index] =
@@ -52,9 +66,11 @@ impl Attacks {
((position & NOT_H_FILE) >> 15) |
((position & NOT_A_FILE) >> 17);
}
+ attacks
}
- fn precompute_king_attacks(attacks: &mut AttackTable) {
+ fn precompute_king_attacks() -> AttackTable {
+ let mut attacks = [0; 64];
for index in 0..64 {
let position = 1u64 << index;
attacks[index] =
@@ -67,9 +83,13 @@ impl Attacks {
(position << 8) |
(position >> 8);
}
+ attacks
}
- fn precompute_first_rank_attacks(attacks: &mut FirstRankAttacks) {
+
+ fn precompute_first_rank_attacks() -> FirstRankAttacks {
+ let mut attacks = [[0; 8]; 256];
+
/// Really slow implementation of Most Significant One Bit which is fine to use when pre-computing
fn log_base_2(bb: Bitboard) -> u64 {
(bb as f64).log2() as u64
@@ -99,6 +119,7 @@ impl Attacks {
}
}
}
+ attacks
}
}
@@ -110,8 +131,8 @@ mod tests {
#[test]
fn test_knight_attacks() {
- let attacks = Attacks::new();
- let e4_attacks = attacks.knight[Square::E4 as usize];
+ let attacks = Attacks::precompute_knight_attacks();
+ let e4_attacks = attacks[Square::E4 as usize];
assert_ne!(e4_attacks & 1 << Square::G5 as usize, 0);
assert_ne!(e4_attacks & 1 << Square::G3 as usize, 0);
@@ -126,37 +147,37 @@ mod tests {
assert_eq!(e4_attacks & 1 << Square::D4 as usize, 0);
assert_eq!(e4_attacks & 1 << Square::A1 as usize, 0);
- assert_eq!(pop_count(attacks.knight[Square::G1 as usize]), 3);
- assert_eq!(pop_count(attacks.knight[Square::H8 as usize]), 2);
+ assert_eq!(pop_count(attacks[Square::G1 as usize]), 3);
+ assert_eq!(pop_count(attacks[Square::H8 as usize]), 2);
}
#[test]
fn test_king_attacks() {
- let attacks = Attacks::new();
- assert_eq!(pop_count(attacks.king[Square::E4 as usize]), 8);
-
- assert_eq!(pop_count(attacks.king[Square::A1 as usize]), 3);
- assert_eq!(pop_count(attacks.king[Square::A8 as usize]), 3);
- assert_eq!(pop_count(attacks.king[Square::H1 as usize]), 3);
- assert_eq!(pop_count(attacks.king[Square::H8 as usize]), 3);
-
- assert_eq!(pop_count(attacks.king[Square::E1 as usize]), 5);
- assert_eq!(pop_count(attacks.king[Square::E8 as usize]), 5);
- assert_eq!(pop_count(attacks.king[Square::A4 as usize]), 5);
- assert_eq!(pop_count(attacks.king[Square::H4 as usize]), 5);
+ let attacks = Attacks::precompute_king_attacks();
+ assert_eq!(pop_count(attacks[Square::E4 as usize]), 8);
+
+ assert_eq!(pop_count(attacks[Square::A1 as usize]), 3);
+ assert_eq!(pop_count(attacks[Square::A8 as usize]), 3);
+ assert_eq!(pop_count(attacks[Square::H1 as usize]), 3);
+ assert_eq!(pop_count(attacks[Square::H8 as usize]), 3);
+
+ assert_eq!(pop_count(attacks[Square::E1 as usize]), 5);
+ assert_eq!(pop_count(attacks[Square::E8 as usize]), 5);
+ assert_eq!(pop_count(attacks[Square::A4 as usize]), 5);
+ assert_eq!(pop_count(attacks[Square::H4 as usize]), 5);
}
#[test]
fn test_first_rank_attacks() {
- let attacks = Attacks::new();
+ let attacks = Attacks::precompute_first_rank_attacks();
// HGFEDCBA HGFEDCBA
- assert_eq!(attacks.first_rank_attacks[0b00010000][4], 0b11101111, "If rook is the only one on the file, it should be able to attack all rank");
- assert_eq!(attacks.first_rank_attacks[0b00010001][4], 0b11101111, "If only other piece is on A rank, rook should be able to attack all rank");
- assert_eq!(attacks.first_rank_attacks[0b10010000][4], 0b11101111, "If only other piece is on H rank, rook should be able to attack all rank");
- assert_eq!(attacks.first_rank_attacks[0b10010001][4], 0b11101111, "If only other pieces are on A and H ranks, rook should be able to attack all rank");
- assert_eq!(attacks.first_rank_attacks[0b00010100][4], 0b11101100);
- assert_eq!(attacks.first_rank_attacks[0b01010100][4], 0b01101100);
- assert_eq!(attacks.first_rank_attacks[0b01010010][4], 0b01101110);
+ assert_eq!(attacks[0b00010000][4], 0b11101111, "If rook is the only one on the file, it should be able to attack all rank");
+ assert_eq!(attacks[0b00010001][4], 0b11101111, "If only other piece is on A rank, rook should be able to attack all rank");
+ assert_eq!(attacks[0b10010000][4], 0b11101111, "If only other piece is on H rank, rook should be able to attack all rank");
+ assert_eq!(attacks[0b10010001][4], 0b11101111, "If only other pieces are on A and H ranks, rook should be able to attack all rank");
+ assert_eq!(attacks[0b00010100][4], 0b11101100);
+ assert_eq!(attacks[0b01010100][4], 0b01101100);
+ assert_eq!(attacks[0b01010010][4], 0b01101110);
}
}
diff --git a/src/board.rs b/src/board.rs
index 13f86ec..86c42c7 100644
--- a/src/board.rs
+++ b/src/board.rs
@@ -32,6 +32,7 @@ const PIECE_CHARS: [&str; 12] = [
];
+#[allow(unused)]
impl Board {
#[allow(non_snake_case)]
pub fn from_FEN(fen: String) -> Self {
@@ -128,7 +129,7 @@ pub enum Square {
#[cfg(test)]
mod tests {
use super::*;
- use crate::bitboard;
+ use crate::bitboard::{pop_count, bitscan};
#[test]
fn test_square_enum() {
@@ -142,22 +143,22 @@ mod tests {
let fen = String::from("rnbqkbnr/pppppppp/8/8/8/8/PPPPPPPP/RNBQKBNR w KQkq - 0 1");
let board = Board::from_FEN(fen);
- assert_eq!(bitboard::pop_count(board.pieces[PieceTypes::Pawn as usize]), 8);
- assert_eq!(bitboard::pop_count(board.pieces[PieceTypes::Knight as usize]), 2);
- assert_eq!(bitboard::pop_count(board.pieces[PieceTypes::Bishop as usize]), 2);
- assert_eq!(bitboard::pop_count(board.pieces[PieceTypes::Rook as usize]), 2);
- assert_eq!(bitboard::pop_count(board.pieces[PieceTypes::Queen as usize]), 1);
- assert_eq!(bitboard::pop_count(board.pieces[PieceTypes::King as usize]), 1);
-
- assert_eq!(bitboard::pop_count(board.pieces[PieceTypes::PawnBlack as usize]), 8);
- assert_eq!(bitboard::pop_count(board.pieces[PieceTypes::KnightBlack as usize]), 2);
- assert_eq!(bitboard::pop_count(board.pieces[PieceTypes::BishopBlack as usize]), 2);
- assert_eq!(bitboard::pop_count(board.pieces[PieceTypes::RookBlack as usize]), 2);
- assert_eq!(bitboard::pop_count(board.pieces[PieceTypes::QueenBlack as usize]), 1);
- assert_eq!(bitboard::pop_count(board.pieces[PieceTypes::KingBlack as usize]), 1);
-
- assert_eq!(bitboard::bitscan(board.pieces[PieceTypes::King as usize]), Square::E1 as u8);
- assert_eq!(bitboard::bitscan(board.pieces[PieceTypes::QueenBlack as usize]), Square::D8 as u8);
+ assert_eq!(pop_count(board.pieces[PieceTypes::Pawn as usize]), 8);
+ assert_eq!(pop_count(board.pieces[PieceTypes::Knight as usize]), 2);
+ assert_eq!(pop_count(board.pieces[PieceTypes::Bishop as usize]), 2);
+ assert_eq!(pop_count(board.pieces[PieceTypes::Rook as usize]), 2);
+ assert_eq!(pop_count(board.pieces[PieceTypes::Queen as usize]), 1);
+ assert_eq!(pop_count(board.pieces[PieceTypes::King as usize]), 1);
+
+ assert_eq!(pop_count(board.pieces[PieceTypes::PawnBlack as usize]), 8);
+ assert_eq!(pop_count(board.pieces[PieceTypes::KnightBlack as usize]), 2);
+ assert_eq!(pop_count(board.pieces[PieceTypes::BishopBlack as usize]), 2);
+ assert_eq!(pop_count(board.pieces[PieceTypes::RookBlack as usize]), 2);
+ assert_eq!(pop_count(board.pieces[PieceTypes::QueenBlack as usize]), 1);
+ assert_eq!(pop_count(board.pieces[PieceTypes::KingBlack as usize]), 1);
+
+ assert_eq!(bitscan(board.pieces[PieceTypes::King as usize]), Square::E1 as u8);
+ assert_eq!(bitscan(board.pieces[PieceTypes::QueenBlack as usize]), Square::D8 as u8);
board.print();
}