2024-10-25 22:30:03 -04:00
|
|
|
/*
|
|
|
|
|
|
|
|
This file is part of chess_inator.
|
|
|
|
|
|
|
|
chess_inator is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
chess_inator is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License along with chess_inator. If not, see https://www.gnu.org/licenses/.
|
|
|
|
|
|
|
|
Copyright © 2024 dogeystamp <dogeystamp@disroot.org>
|
|
|
|
*/
|
|
|
|
|
2024-09-22 21:08:40 -04:00
|
|
|
#![deny(rust_2018_idioms)]
|
|
|
|
|
2024-10-20 11:41:10 -04:00
|
|
|
use std::fmt::Display;
|
2024-10-20 12:06:00 -04:00
|
|
|
use std::str::FromStr;
|
2024-10-20 11:37:44 -04:00
|
|
|
|
2024-09-27 21:20:14 -04:00
|
|
|
pub mod fen;
|
2024-09-29 10:43:45 -04:00
|
|
|
pub mod movegen;
|
2024-09-27 21:20:14 -04:00
|
|
|
|
2024-10-25 20:26:41 -04:00
|
|
|
use crate::fen::{FromFen, ToFen, START_POSITION};
|
2024-10-25 11:59:53 -04:00
|
|
|
|
2024-09-22 21:08:40 -04:00
|
|
|
const BOARD_WIDTH: usize = 8;
|
|
|
|
const BOARD_HEIGHT: usize = 8;
|
|
|
|
const N_SQUARES: usize = BOARD_WIDTH * BOARD_HEIGHT;
|
|
|
|
|
2024-09-27 20:59:38 -04:00
|
|
|
#[derive(Debug, Copy, Clone, Default, PartialEq, Eq)]
|
2024-10-25 21:34:33 -04:00
|
|
|
pub enum Color {
|
2024-09-24 22:01:49 -04:00
|
|
|
#[default]
|
2024-09-29 10:43:45 -04:00
|
|
|
White = 0,
|
|
|
|
Black = 1,
|
2024-09-22 21:08:40 -04:00
|
|
|
}
|
|
|
|
const N_COLORS: usize = 2;
|
|
|
|
|
2024-09-29 10:43:45 -04:00
|
|
|
impl Color {
|
|
|
|
/// Return opposite color (does not assign).
|
|
|
|
pub fn flip(self) -> Self {
|
|
|
|
match self {
|
|
|
|
Color::White => Color::Black,
|
|
|
|
Color::Black => Color::White,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-29 12:32:11 -04:00
|
|
|
impl From<Color> for char {
|
|
|
|
fn from(value: Color) -> Self {
|
|
|
|
match value {
|
|
|
|
Color::White => 'w',
|
|
|
|
Color::Black => 'b',
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-04 20:03:19 -04:00
|
|
|
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
|
2024-09-22 21:08:40 -04:00
|
|
|
enum Piece {
|
|
|
|
Rook,
|
|
|
|
Bishop,
|
|
|
|
Knight,
|
|
|
|
King,
|
|
|
|
Queen,
|
|
|
|
Pawn,
|
|
|
|
}
|
|
|
|
const N_PIECES: usize = 6;
|
|
|
|
|
2024-10-25 21:34:33 -04:00
|
|
|
pub struct PieceErr;
|
2024-09-22 21:08:40 -04:00
|
|
|
|
|
|
|
/// Color and piece.
|
2024-10-20 19:12:18 -04:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
2024-10-25 21:34:33 -04:00
|
|
|
pub struct ColPiece {
|
2024-09-22 21:08:40 -04:00
|
|
|
pc: Piece,
|
|
|
|
col: Color,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<char> for ColPiece {
|
|
|
|
type Error = PieceErr;
|
|
|
|
|
|
|
|
fn try_from(value: char) -> Result<Self, Self::Error> {
|
2024-09-24 22:01:49 -04:00
|
|
|
let col = if value.is_ascii_uppercase() {
|
|
|
|
Color::White
|
|
|
|
} else {
|
|
|
|
Color::Black
|
|
|
|
};
|
2024-09-22 21:08:40 -04:00
|
|
|
let mut lower = value;
|
|
|
|
lower.make_ascii_lowercase();
|
2024-09-24 22:01:49 -04:00
|
|
|
Ok(ColPiece {
|
|
|
|
pc: Piece::try_from(lower)?,
|
|
|
|
col,
|
|
|
|
})
|
2024-09-22 21:08:40 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<ColPiece> for char {
|
|
|
|
fn from(value: ColPiece) -> Self {
|
|
|
|
let lower = char::from(value.pc);
|
|
|
|
match value.col {
|
|
|
|
Color::White => lower.to_ascii_uppercase(),
|
|
|
|
Color::Black => lower,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<ColPiece> for Color {
|
|
|
|
fn from(value: ColPiece) -> Self {
|
|
|
|
value.col
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<ColPiece> for Piece {
|
|
|
|
fn from(value: ColPiece) -> Self {
|
|
|
|
value.pc
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ColPiece {
|
|
|
|
/// Convert option of piece to character.
|
|
|
|
pub fn opt_to_char(opt: Option<Self>) -> char {
|
|
|
|
match opt {
|
|
|
|
Some(pc) => pc.into(),
|
|
|
|
None => '.',
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-25 22:02:30 -04:00
|
|
|
type SquareIdx = u8;
|
|
|
|
|
2024-09-22 21:08:40 -04:00
|
|
|
/// Square index newtype.
|
|
|
|
///
|
|
|
|
/// A1 is (0, 0) -> 0, A2 is (0, 1) -> 2, and H8 is (7, 7) -> 63.
|
2024-10-22 16:31:30 -04:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
2024-10-25 22:02:30 -04:00
|
|
|
pub struct Square(SquareIdx);
|
2024-09-22 21:08:40 -04:00
|
|
|
|
2024-09-29 12:32:11 -04:00
|
|
|
#[derive(Debug)]
|
2024-10-04 23:02:00 -04:00
|
|
|
pub enum SquareError {
|
2024-09-22 21:08:40 -04:00
|
|
|
OutOfBounds,
|
2024-09-29 13:47:02 -04:00
|
|
|
InvalidCharacter(char),
|
2024-09-22 21:08:40 -04:00
|
|
|
}
|
|
|
|
|
2024-10-25 22:02:30 -04:00
|
|
|
impl TryFrom<SquareIdx> for Square {
|
2024-09-29 13:47:02 -04:00
|
|
|
type Error = SquareError;
|
2024-09-22 21:08:40 -04:00
|
|
|
|
2024-10-25 22:02:30 -04:00
|
|
|
fn try_from(value: SquareIdx) -> Result<Self, Self::Error> {
|
|
|
|
if (0..N_SQUARES).contains(&value.into()) {
|
2024-09-29 10:43:45 -04:00
|
|
|
Ok(Square(value))
|
2024-09-22 21:08:40 -04:00
|
|
|
} else {
|
2024-09-29 13:47:02 -04:00
|
|
|
Err(SquareError::OutOfBounds)
|
2024-09-22 21:08:40 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-10-20 11:37:44 -04:00
|
|
|
|
|
|
|
macro_rules! sq_try_from {
|
|
|
|
($T: ty) => {
|
|
|
|
impl TryFrom<$T> for Square {
|
|
|
|
type Error = SquareError;
|
|
|
|
|
|
|
|
fn try_from(value: $T) -> Result<Self, Self::Error> {
|
|
|
|
if let Ok(upper_bound) = <$T>::try_from(N_SQUARES) {
|
|
|
|
if (0..upper_bound).contains(&value) {
|
2024-10-25 22:02:30 -04:00
|
|
|
return Ok(Square(value as SquareIdx));
|
2024-10-20 11:37:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(SquareError::OutOfBounds)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-10-25 22:02:30 -04:00
|
|
|
sq_try_from!(i8);
|
2024-10-20 11:37:44 -04:00
|
|
|
sq_try_from!(i32);
|
|
|
|
sq_try_from!(isize);
|
2024-10-25 22:02:30 -04:00
|
|
|
sq_try_from!(usize);
|
2024-10-20 11:37:44 -04:00
|
|
|
|
2024-10-25 22:02:30 -04:00
|
|
|
impl From<Square> for SquareIdx {
|
2024-09-29 10:43:45 -04:00
|
|
|
fn from(value: Square) -> Self {
|
2024-09-22 21:08:40 -04:00
|
|
|
value.0
|
|
|
|
}
|
|
|
|
}
|
2024-10-20 12:06:00 -04:00
|
|
|
|
2024-10-25 22:02:30 -04:00
|
|
|
impl From<Square> for usize {
|
|
|
|
fn from(value: Square) -> Self {
|
|
|
|
value.0.into()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-20 12:06:00 -04:00
|
|
|
macro_rules! from_row_col_generic {
|
|
|
|
($T: ty, $r: ident, $c: ident) => {
|
|
|
|
if !(0..(BOARD_HEIGHT as $T)).contains(&$r) || !(0..(BOARD_WIDTH as $T)).contains(&$c) {
|
|
|
|
Err(SquareError::OutOfBounds)
|
|
|
|
} else {
|
|
|
|
let ret = (BOARD_WIDTH as $T) * $r + $c;
|
|
|
|
ret.try_into()
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-09-29 10:43:45 -04:00
|
|
|
impl Square {
|
2024-09-29 13:47:02 -04:00
|
|
|
fn from_row_col(r: usize, c: usize) -> Result<Self, SquareError> {
|
2024-09-22 21:08:40 -04:00
|
|
|
//! Get index of square based on row and column.
|
2024-10-20 12:06:00 -04:00
|
|
|
from_row_col_generic!(usize, r, c)
|
2024-09-22 21:08:40 -04:00
|
|
|
}
|
2024-10-20 11:37:44 -04:00
|
|
|
fn from_row_col_signed(r: isize, c: isize) -> Result<Self, SquareError> {
|
2024-10-20 12:06:00 -04:00
|
|
|
from_row_col_generic!(isize, r, c)
|
2024-10-20 11:37:44 -04:00
|
|
|
}
|
2024-09-29 12:32:11 -04:00
|
|
|
fn to_row_col(self) -> (usize, usize) {
|
|
|
|
//! Get row, column from index
|
2024-10-25 22:02:30 -04:00
|
|
|
let div = usize::from(self.0) / BOARD_WIDTH;
|
|
|
|
let rem = usize::from(self.0) % BOARD_WIDTH;
|
|
|
|
debug_assert!(div <= 7);
|
|
|
|
debug_assert!(rem <= 7);
|
2024-09-29 12:32:11 -04:00
|
|
|
(div, rem)
|
|
|
|
}
|
2024-10-20 15:40:34 -04:00
|
|
|
fn to_row_col_signed(self) -> (isize, isize) {
|
|
|
|
//! Get row, column (signed) from index
|
|
|
|
let (r, c) = self.to_row_col();
|
|
|
|
(r.try_into().unwrap(), c.try_into().unwrap())
|
|
|
|
}
|
2024-10-20 19:12:18 -04:00
|
|
|
|
|
|
|
/// Vertically mirror a square.
|
|
|
|
fn mirror_vert(&self) -> Self {
|
|
|
|
let (r, c) = self.to_row_col();
|
|
|
|
let (nr, nc) = (BOARD_HEIGHT - 1 - r, c);
|
|
|
|
Square::from_row_col(nr, nc)
|
|
|
|
.unwrap_or_else(|e| panic!("mirrored square should be valid: nr {nr} nc {nc}: {e:?}"))
|
|
|
|
}
|
2024-10-20 11:41:10 -04:00
|
|
|
}
|
2024-09-29 12:32:11 -04:00
|
|
|
|
2024-10-20 11:41:10 -04:00
|
|
|
impl Display for Square {
|
2024-09-29 12:32:11 -04:00
|
|
|
/// Convert square to typical human-readable form (e.g. `e4`).
|
2024-10-20 11:41:10 -04:00
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
2024-09-29 12:32:11 -04:00
|
|
|
let letters = ['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h'];
|
|
|
|
let (row, col) = self.to_row_col();
|
|
|
|
let rank = (row + 1).to_string();
|
|
|
|
let file = letters[col];
|
2024-10-20 11:41:10 -04:00
|
|
|
write!(f, "{}{}", file, rank)
|
2024-09-29 12:32:11 -04:00
|
|
|
}
|
2024-10-20 11:37:44 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
impl FromStr for Square {
|
|
|
|
type Err = SquareError;
|
2024-09-29 13:47:02 -04:00
|
|
|
|
|
|
|
/// Convert typical human-readable form (e.g. `e4`) to square index.
|
2024-10-20 11:37:44 -04:00
|
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
|
|
let bytes = s.as_bytes();
|
2024-09-29 13:47:02 -04:00
|
|
|
let col = match bytes[0] as char {
|
|
|
|
'a' => 0,
|
|
|
|
'b' => 1,
|
|
|
|
'c' => 2,
|
|
|
|
'd' => 3,
|
|
|
|
'e' => 4,
|
|
|
|
'f' => 5,
|
|
|
|
'g' => 6,
|
|
|
|
'h' => 7,
|
2024-10-06 21:28:11 -04:00
|
|
|
_ => return Err(SquareError::InvalidCharacter(bytes[0] as char)),
|
2024-09-29 13:47:02 -04:00
|
|
|
};
|
|
|
|
if let Some(row) = (bytes[1] as char).to_digit(10) {
|
|
|
|
Square::from_row_col(row as usize - 1, col as usize)
|
|
|
|
} else {
|
|
|
|
Err(SquareError::InvalidCharacter(bytes[1] as char))
|
|
|
|
}
|
|
|
|
}
|
2024-09-29 12:32:11 -04:00
|
|
|
}
|
|
|
|
|
2024-09-22 21:08:40 -04:00
|
|
|
impl TryFrom<char> for Piece {
|
|
|
|
type Error = PieceErr;
|
|
|
|
|
|
|
|
fn try_from(s: char) -> Result<Self, Self::Error> {
|
|
|
|
match s {
|
|
|
|
'r' => Ok(Piece::Rook),
|
|
|
|
'b' => Ok(Piece::Bishop),
|
|
|
|
'n' => Ok(Piece::Knight),
|
|
|
|
'k' => Ok(Piece::King),
|
|
|
|
'q' => Ok(Piece::Queen),
|
|
|
|
'p' => Ok(Piece::Pawn),
|
|
|
|
_ => Err(PieceErr),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Piece> for char {
|
|
|
|
fn from(value: Piece) -> Self {
|
|
|
|
match value {
|
|
|
|
Piece::Rook => 'r',
|
|
|
|
Piece::Bishop => 'b',
|
|
|
|
Piece::Knight => 'n',
|
|
|
|
Piece::King => 'k',
|
|
|
|
Piece::Queen => 'q',
|
|
|
|
Piece::Pawn => 'p',
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-20 19:12:18 -04:00
|
|
|
#[derive(Default, Debug, Clone, Copy, PartialEq, Eq)]
|
2024-10-25 21:34:33 -04:00
|
|
|
pub struct Bitboard(u64);
|
2024-09-22 21:08:40 -04:00
|
|
|
|
|
|
|
impl Bitboard {
|
2024-10-22 16:31:30 -04:00
|
|
|
pub fn on_sq(&mut self, idx: Square) {
|
|
|
|
//! Set a square on.
|
2024-09-22 21:08:40 -04:00
|
|
|
self.0 |= 1 << usize::from(idx);
|
|
|
|
}
|
|
|
|
|
2024-10-22 16:31:30 -04:00
|
|
|
pub fn off_sq(&mut self, idx: Square) {
|
|
|
|
//! Set a square off.
|
2024-09-22 21:08:40 -04:00
|
|
|
self.0 &= !(1 << usize::from(idx));
|
|
|
|
}
|
2024-10-18 17:12:06 -04:00
|
|
|
|
2024-10-22 16:31:30 -04:00
|
|
|
pub fn get_sq(&self, idx: Square) -> bool {
|
|
|
|
//! Read the value at a square.
|
|
|
|
(self.0 & 1 << usize::from(idx)) == 1
|
|
|
|
}
|
|
|
|
|
2024-10-18 17:12:06 -04:00
|
|
|
pub fn is_empty(&self) -> bool {
|
|
|
|
self.0 == 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl IntoIterator for Bitboard {
|
|
|
|
type Item = Square;
|
|
|
|
|
|
|
|
type IntoIter = BitboardIterator;
|
|
|
|
|
|
|
|
fn into_iter(self) -> Self::IntoIter {
|
|
|
|
BitboardIterator { remaining: self }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-25 21:34:33 -04:00
|
|
|
pub struct BitboardIterator {
|
2024-10-18 17:12:06 -04:00
|
|
|
remaining: Bitboard,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Iterator for BitboardIterator {
|
|
|
|
type Item = Square;
|
|
|
|
|
|
|
|
fn next(&mut self) -> Option<Self::Item> {
|
|
|
|
if self.remaining.is_empty() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
let next_idx = self.remaining.0.trailing_zeros() as usize;
|
2024-10-25 22:02:30 -04:00
|
|
|
let sq = Square(next_idx.try_into().unwrap());
|
2024-10-22 16:31:30 -04:00
|
|
|
self.remaining.off_sq(sq);
|
2024-10-18 17:12:06 -04:00
|
|
|
Some(sq)
|
|
|
|
}
|
|
|
|
}
|
2024-09-22 21:08:40 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Array form board.
|
|
|
|
///
|
|
|
|
/// Complements bitboards, notably for "what piece is at this square?" queries.
|
2024-10-20 19:12:18 -04:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
2024-09-22 21:08:40 -04:00
|
|
|
struct Mailbox([Option<ColPiece>; N_SQUARES]);
|
|
|
|
|
|
|
|
impl Default for Mailbox {
|
|
|
|
fn default() -> Self {
|
|
|
|
Mailbox([None; N_SQUARES])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Mailbox {
|
|
|
|
/// Get mutable reference to square at index.
|
2024-09-29 10:43:45 -04:00
|
|
|
fn sq_mut(&mut self, idx: Square) -> &mut Option<ColPiece> {
|
2024-09-22 21:08:40 -04:00
|
|
|
&mut self.0[usize::from(idx)]
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get non-mutable reference to square at index.
|
2024-09-29 10:43:45 -04:00
|
|
|
fn sq(&self, idx: Square) -> &Option<ColPiece> {
|
2024-09-22 21:08:40 -04:00
|
|
|
&self.0[usize::from(idx)]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Piece bitboards and state for one player.
|
|
|
|
///
|
|
|
|
/// Default is all empty.
|
2024-10-20 19:12:18 -04:00
|
|
|
#[derive(Default, Debug, Clone, Copy, PartialEq, Eq)]
|
2024-09-22 21:08:40 -04:00
|
|
|
struct Player {
|
|
|
|
/// Bitboards for individual pieces. Piece -> locations.
|
|
|
|
bit: [Bitboard; N_PIECES],
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Player {
|
2024-10-20 11:37:44 -04:00
|
|
|
/// Get board (non-mutable) for a specific piece.
|
|
|
|
fn board(&self, pc: Piece) -> &Bitboard {
|
|
|
|
&self.bit[pc as usize]
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get board (mutable) for a specific piece.
|
|
|
|
fn board_mut(&mut self, pc: Piece) -> &mut Bitboard {
|
2024-09-22 21:08:40 -04:00
|
|
|
&mut self.bit[pc as usize]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-24 22:01:49 -04:00
|
|
|
/// Castling rights for one player
|
2024-09-29 10:43:45 -04:00
|
|
|
#[derive(Debug, Default, PartialEq, Eq, Clone, Copy)]
|
2024-09-29 12:32:11 -04:00
|
|
|
pub struct CastlePlayer {
|
2024-09-24 22:01:49 -04:00
|
|
|
/// Kingside
|
|
|
|
k: bool,
|
|
|
|
/// Queenside
|
|
|
|
q: bool,
|
|
|
|
}
|
|
|
|
|
2024-09-29 12:32:11 -04:00
|
|
|
/// Castling rights for both players
|
|
|
|
#[derive(Debug, Default, PartialEq, Eq, Clone, Copy)]
|
|
|
|
pub struct CastleRights([CastlePlayer; N_COLORS]);
|
|
|
|
|
|
|
|
impl ToString for CastleRights {
|
|
|
|
/// Convert to FEN castling rights format.
|
|
|
|
fn to_string(&self) -> String {
|
|
|
|
let mut ret = String::with_capacity(4);
|
|
|
|
for (val, ch) in [
|
|
|
|
(self.0[Color::White as usize].k, 'K'),
|
|
|
|
(self.0[Color::White as usize].q, 'Q'),
|
|
|
|
(self.0[Color::Black as usize].k, 'k'),
|
|
|
|
(self.0[Color::Black as usize].q, 'q'),
|
|
|
|
] {
|
|
|
|
if val {
|
|
|
|
ret.push(ch)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ret.is_empty() {
|
|
|
|
ret.push('-')
|
|
|
|
}
|
|
|
|
ret
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-29 10:43:45 -04:00
|
|
|
/// Immutable game state, unique to a position.
|
2024-09-22 21:08:40 -04:00
|
|
|
///
|
|
|
|
/// Default is empty.
|
2024-10-20 19:12:18 -04:00
|
|
|
#[derive(Debug, Default, Clone, Copy, PartialEq, Eq)]
|
2024-10-25 11:59:53 -04:00
|
|
|
pub struct Board {
|
2024-09-22 21:08:40 -04:00
|
|
|
/// Player bitboards
|
|
|
|
players: [Player; N_COLORS],
|
|
|
|
|
|
|
|
/// Mailbox (array) board. Location -> piece.
|
|
|
|
mail: Mailbox,
|
2024-09-24 22:01:49 -04:00
|
|
|
|
|
|
|
/// En-passant square.
|
|
|
|
///
|
|
|
|
/// (If a pawn moves twice, this is one square in front of the start position.)
|
2024-09-29 10:43:45 -04:00
|
|
|
ep_square: Option<Square>,
|
2024-09-24 22:01:49 -04:00
|
|
|
|
2024-09-27 20:59:38 -04:00
|
|
|
/// Castling rights
|
2024-09-29 12:32:11 -04:00
|
|
|
castle: CastleRights,
|
2024-09-24 22:01:49 -04:00
|
|
|
|
|
|
|
/// Plies since last irreversible (capture, pawn) move
|
|
|
|
half_moves: usize,
|
|
|
|
|
|
|
|
/// Full move counter (incremented after each black turn)
|
|
|
|
full_moves: usize,
|
|
|
|
|
|
|
|
/// Whose turn it is
|
|
|
|
turn: Color,
|
2024-09-22 21:08:40 -04:00
|
|
|
}
|
|
|
|
|
2024-10-25 11:59:53 -04:00
|
|
|
impl Board {
|
|
|
|
/// Default chess position.
|
|
|
|
pub fn starting_pos() -> Self {
|
|
|
|
Board::from_fen(START_POSITION).unwrap()
|
|
|
|
}
|
2024-10-06 21:28:11 -04:00
|
|
|
|
2024-09-22 21:08:40 -04:00
|
|
|
/// Get mutable reference to a player.
|
|
|
|
fn pl_mut(&mut self, col: Color) -> &mut Player {
|
|
|
|
&mut self.players[col as usize]
|
|
|
|
}
|
|
|
|
|
2024-10-20 13:09:04 -04:00
|
|
|
/// Get immutable reference to a player.
|
|
|
|
fn pl(&self, col: Color) -> &Player {
|
|
|
|
&self.players[col as usize]
|
|
|
|
}
|
|
|
|
|
2024-10-20 14:53:55 -04:00
|
|
|
/// Get immutable reference to castling rights.
|
2024-10-25 21:34:33 -04:00
|
|
|
pub fn pl_castle(&self, col: Color) -> &CastlePlayer {
|
2024-10-20 14:53:55 -04:00
|
|
|
&self.castle.0[col as usize]
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get mutable reference to castling rights.
|
|
|
|
fn pl_castle_mut(&mut self, col: Color) -> &mut CastlePlayer {
|
|
|
|
&mut self.castle.0[col as usize]
|
|
|
|
}
|
|
|
|
|
2024-10-20 19:12:18 -04:00
|
|
|
/// Get iterator over all squares.
|
2024-10-25 21:34:33 -04:00
|
|
|
pub fn squares() -> impl Iterator<Item = Square> {
|
2024-10-20 19:12:18 -04:00
|
|
|
(0..N_SQUARES).map(Square::try_from).map(|x| x.unwrap())
|
|
|
|
}
|
|
|
|
|
2024-10-25 11:59:53 -04:00
|
|
|
/// Create a new piece in a location, and pop any existing piece in the destination.
|
2024-10-25 21:34:33 -04:00
|
|
|
pub fn set_piece(&mut self, idx: Square, pc: ColPiece) -> Option<ColPiece> {
|
2024-10-25 11:59:53 -04:00
|
|
|
let dest_pc = self.del_piece(idx);
|
2024-09-22 21:08:40 -04:00
|
|
|
let pl = self.pl_mut(pc.col);
|
2024-10-22 16:31:30 -04:00
|
|
|
pl.board_mut(pc.into()).on_sq(idx);
|
2024-09-22 21:08:40 -04:00
|
|
|
*self.mail.sq_mut(idx) = Some(pc);
|
2024-10-25 11:59:53 -04:00
|
|
|
dest_pc
|
2024-09-22 21:08:40 -04:00
|
|
|
}
|
|
|
|
|
2024-10-25 11:59:53 -04:00
|
|
|
/// Set the piece (or no piece) in a square, and return ("pop") the existing piece.
|
2024-10-25 21:34:33 -04:00
|
|
|
pub fn set_square(&mut self, idx: Square, pc: Option<ColPiece>) -> Option<ColPiece> {
|
2024-10-20 19:12:18 -04:00
|
|
|
match pc {
|
2024-10-25 20:26:41 -04:00
|
|
|
Some(pc) => self.set_piece(idx, pc),
|
|
|
|
None => self.del_piece(idx),
|
2024-10-20 19:12:18 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-18 16:24:15 -04:00
|
|
|
/// Delete the piece in a location, and return ("pop") that piece.
|
2024-10-25 21:34:33 -04:00
|
|
|
pub fn del_piece(&mut self, idx: Square) -> Option<ColPiece> {
|
2024-09-22 21:08:40 -04:00
|
|
|
if let Some(pc) = *self.mail.sq_mut(idx) {
|
|
|
|
let pl = self.pl_mut(pc.col);
|
2024-10-22 16:31:30 -04:00
|
|
|
pl.board_mut(pc.into()).off_sq(idx);
|
2024-09-22 21:08:40 -04:00
|
|
|
*self.mail.sq_mut(idx) = None;
|
2024-10-25 11:59:53 -04:00
|
|
|
Some(pc)
|
2024-10-06 21:28:11 -04:00
|
|
|
} else {
|
2024-10-25 11:59:53 -04:00
|
|
|
None
|
2024-09-22 21:08:40 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-18 16:24:15 -04:00
|
|
|
fn move_piece(&mut self, src: Square, dest: Square) {
|
2024-10-25 20:26:41 -04:00
|
|
|
let pc = self.del_piece(src).unwrap_or_else(|| {
|
|
|
|
panic!(
|
|
|
|
"move ({src} -> {dest}) should have piece at source (pos '{}')",
|
|
|
|
self.to_fen()
|
|
|
|
)
|
|
|
|
});
|
2024-10-18 16:24:15 -04:00
|
|
|
self.set_piece(dest, pc);
|
|
|
|
}
|
|
|
|
|
2024-09-22 21:08:40 -04:00
|
|
|
/// Get the piece at a location.
|
2024-10-25 21:34:33 -04:00
|
|
|
pub fn get_piece(&self, idx: Square) -> Option<ColPiece> {
|
2024-09-22 21:08:40 -04:00
|
|
|
*self.mail.sq(idx)
|
|
|
|
}
|
|
|
|
|
2024-10-20 19:12:18 -04:00
|
|
|
/// Mirrors the position so that black and white are switched.
|
|
|
|
///
|
|
|
|
/// Mainly to avoid duplication in tests.
|
2024-10-25 21:34:33 -04:00
|
|
|
pub fn flip_colors(&self) -> Self {
|
2024-10-20 19:12:18 -04:00
|
|
|
let mut new_board = Self {
|
|
|
|
turn: self.turn.flip(),
|
|
|
|
half_moves: self.half_moves,
|
|
|
|
full_moves: self.full_moves,
|
|
|
|
players: Default::default(),
|
|
|
|
mail: Default::default(),
|
|
|
|
ep_square: self.ep_square.map(|sq| sq.mirror_vert()),
|
2024-10-21 14:43:18 -04:00
|
|
|
castle: CastleRights(self.castle.0),
|
2024-10-20 19:12:18 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
new_board.castle.0.reverse();
|
|
|
|
|
2024-10-25 11:59:53 -04:00
|
|
|
for sq in Board::squares() {
|
2024-10-20 19:12:18 -04:00
|
|
|
let opt_pc = self.get_piece(sq.mirror_vert()).map(|pc| ColPiece {
|
|
|
|
col: pc.col.flip(),
|
|
|
|
pc: pc.pc,
|
|
|
|
});
|
2024-10-25 11:59:53 -04:00
|
|
|
new_board.set_square(sq, opt_pc);
|
2024-10-20 19:12:18 -04:00
|
|
|
}
|
|
|
|
new_board
|
|
|
|
}
|
|
|
|
|
2024-10-25 21:25:38 -04:00
|
|
|
/// Is a given player in check?
|
2024-10-25 21:34:33 -04:00
|
|
|
pub fn is_check(&self, pl: Color) -> bool {
|
2024-10-25 21:25:38 -04:00
|
|
|
for src in self.pl(pl).board(Piece::King).into_iter() {
|
|
|
|
macro_rules! detect_checker {
|
|
|
|
($dirs: ident, $pc: pat, $keep_going: expr) => {
|
|
|
|
for dir in $dirs.into_iter() {
|
|
|
|
let (mut r, mut c) = src.to_row_col_signed();
|
|
|
|
loop {
|
|
|
|
let (nr, nc) = (r + dir.0, c + dir.1);
|
|
|
|
if let Ok(sq) = Square::from_row_col_signed(nr, nc) {
|
|
|
|
if let Some(pc) = self.get_piece(sq) {
|
|
|
|
if matches!(pc.pc, $pc) && pc.col != pl {
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!($keep_going)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
r = nr;
|
|
|
|
c = nc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
let dirs_white_pawn = [(-1, 1), (-1, -1)];
|
|
|
|
let dirs_black_pawn = [(1, 1), (1, -1)];
|
|
|
|
|
|
|
|
use Piece::*;
|
|
|
|
|
|
|
|
use movegen::{DIRS_DIAG, DIRS_KNIGHT, DIRS_STAR, DIRS_STRAIGHT};
|
|
|
|
|
|
|
|
detect_checker!(DIRS_DIAG, Bishop | Queen, true);
|
|
|
|
detect_checker!(DIRS_STRAIGHT, Rook | Queen, true);
|
|
|
|
detect_checker!(DIRS_STAR, King, false);
|
|
|
|
detect_checker!(DIRS_KNIGHT, Knight, false);
|
|
|
|
match pl {
|
|
|
|
Color::White => detect_checker!(dirs_black_pawn, Pawn, false),
|
|
|
|
Color::Black => detect_checker!(dirs_white_pawn, Pawn, false),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
2024-09-27 20:59:38 -04:00
|
|
|
/// Maximum amount of moves in the counter to parse before giving up
|
|
|
|
const MAX_MOVES: usize = 9_999;
|
2024-09-22 21:08:40 -04:00
|
|
|
}
|
|
|
|
|
2024-10-25 11:59:53 -04:00
|
|
|
impl core::fmt::Display for Board {
|
2024-09-22 21:08:40 -04:00
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
|
|
let mut str = String::with_capacity(N_SQUARES + BOARD_HEIGHT);
|
|
|
|
for row in (0..BOARD_HEIGHT).rev() {
|
|
|
|
for col in 0..BOARD_WIDTH {
|
2024-09-29 10:43:45 -04:00
|
|
|
let idx = Square::from_row_col(row, col).or(Err(std::fmt::Error))?;
|
2024-09-22 21:08:40 -04:00
|
|
|
let pc = self.get_piece(idx);
|
|
|
|
str.push(ColPiece::opt_to_char(pc));
|
|
|
|
}
|
|
|
|
str += "\n";
|
2024-09-24 22:01:49 -04:00
|
|
|
}
|
2024-09-22 21:08:40 -04:00
|
|
|
write!(f, "{}", str)
|
|
|
|
}
|
|
|
|
}
|
2024-10-18 17:12:06 -04:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
2024-10-22 12:51:22 -04:00
|
|
|
use fen::FromFen;
|
|
|
|
|
2024-10-20 11:37:44 -04:00
|
|
|
#[test]
|
|
|
|
fn test_square_casts() {
|
|
|
|
let fail_cases = [-1, 64, 0x7FFFFFFF, 257, 256, 128, 65, -3, !0x7FFFFFFF];
|
|
|
|
for tc in fail_cases {
|
|
|
|
macro_rules! try_type {
|
|
|
|
($T: ty) => {
|
|
|
|
if let Ok(conv) = <$T>::try_from(tc) {
|
2024-10-20 12:06:00 -04:00
|
|
|
assert!(matches!(
|
|
|
|
Square::try_from(conv),
|
|
|
|
Err(SquareError::OutOfBounds)
|
|
|
|
))
|
2024-10-20 11:37:44 -04:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
try_type!(i32);
|
|
|
|
try_type!(i8);
|
|
|
|
try_type!(isize);
|
2024-10-25 22:02:30 -04:00
|
|
|
try_type!(u8);
|
2024-10-20 11:37:44 -04:00
|
|
|
}
|
|
|
|
|
2024-10-25 22:02:30 -04:00
|
|
|
let good_cases = 0..SquareIdx::try_from(N_SQUARES).unwrap();
|
2024-10-20 11:37:44 -04:00
|
|
|
for tc in good_cases {
|
|
|
|
macro_rules! try_type {
|
|
|
|
($T: ty) => {
|
|
|
|
let conv = <$T>::try_from(tc).unwrap();
|
|
|
|
let res = Square::try_from(conv).unwrap();
|
|
|
|
assert_eq!(res.0, tc);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
try_type!(i32);
|
|
|
|
try_type!(i8);
|
|
|
|
try_type!(isize);
|
2024-10-25 22:02:30 -04:00
|
|
|
try_type!(u8);
|
2024-10-20 11:37:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-10-18 17:12:06 -04:00
|
|
|
#[test]
|
|
|
|
fn test_to_from_algebraic() {
|
|
|
|
let test_cases = [("a1", 0), ("a8", 56), ("h1", 7), ("h8", 63)];
|
|
|
|
for (sqr, idx) in test_cases {
|
2024-10-20 11:41:10 -04:00
|
|
|
assert_eq!(Square::try_from(idx).unwrap().to_string(), sqr);
|
2024-10-18 17:12:06 -04:00
|
|
|
assert_eq!(
|
2024-10-20 11:37:44 -04:00
|
|
|
sqr.parse::<Square>().unwrap(),
|
2024-10-18 17:12:06 -04:00
|
|
|
Square::try_from(idx).unwrap()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_bitboard_iteration() {
|
2024-10-25 22:02:30 -04:00
|
|
|
let indices = [0, 5, 17, 24, 34, 39, 42, 45, 49, 50, 63];
|
2024-10-18 17:12:06 -04:00
|
|
|
|
|
|
|
let mut bitboard = Bitboard::default();
|
|
|
|
|
|
|
|
let squares = indices.map(Square);
|
|
|
|
for sq in squares {
|
2024-10-22 16:31:30 -04:00
|
|
|
bitboard.on_sq(sq);
|
2024-10-18 17:12:06 -04:00
|
|
|
}
|
|
|
|
// ensure that iteration does not consume the board
|
|
|
|
for _ in 0..=1 {
|
|
|
|
for (i, sq) in bitboard.into_iter().enumerate() {
|
|
|
|
assert_eq!(squares[i], sq)
|
|
|
|
}
|
|
|
|
}
|
2024-10-20 13:09:04 -04:00
|
|
|
|
2024-10-25 11:59:53 -04:00
|
|
|
let board = Board::from_fen("8/4p3/1q1Q1p2/4p3/1p1r4/8/8/8 w - - 0 1").unwrap();
|
2024-10-20 13:09:26 -04:00
|
|
|
let white_queens = board
|
|
|
|
.pl(Color::White)
|
|
|
|
.board(Piece::Queen)
|
|
|
|
.into_iter()
|
|
|
|
.collect::<Vec<Square>>();
|
2024-10-20 13:09:04 -04:00
|
|
|
assert_eq!(white_queens, vec![Square::from_str("d6").unwrap()])
|
2024-10-18 17:12:06 -04:00
|
|
|
}
|
2024-10-20 19:12:18 -04:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_square_mirror() {
|
|
|
|
for (sq, expect) in [("a1", "a8"), ("h1", "h8"), ("d4", "d5")] {
|
|
|
|
let sq = sq.parse::<Square>().unwrap();
|
|
|
|
let expect = expect.parse::<Square>().unwrap();
|
|
|
|
assert_eq!(sq.mirror_vert(), expect);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_flip_colors() {
|
|
|
|
let test_cases = [
|
2024-10-21 14:43:18 -04:00
|
|
|
(
|
|
|
|
"2kqrbnp/8/8/8/8/8/8/2KQRBNP w - - 0 1",
|
|
|
|
"2kqrbnp/8/8/8/8/8/8/2KQRBNP b - - 0 1",
|
|
|
|
),
|
|
|
|
(
|
|
|
|
"2kqrbnp/8/8/8/8/8/6N1/2KQRB1P w - a1 0 1",
|
|
|
|
"2kqrb1p/6n1/8/8/8/8/8/2KQRBNP b - a8 0 1",
|
|
|
|
),
|
|
|
|
(
|
|
|
|
"r3k2r/8/8/8/8/8/8/R3K2R w Kq - 0 1",
|
|
|
|
"r3k2r/8/8/8/8/8/8/R3K2R b Qk - 0 1",
|
|
|
|
),
|
2024-10-20 19:12:18 -04:00
|
|
|
];
|
|
|
|
for (tc, expect) in test_cases {
|
2024-10-25 11:59:53 -04:00
|
|
|
let tc = Board::from_fen(tc).unwrap();
|
|
|
|
let expect = Board::from_fen(expect).unwrap();
|
2024-10-20 19:12:18 -04:00
|
|
|
assert_eq!(tc.flip_colors(), expect);
|
|
|
|
}
|
|
|
|
}
|
2024-10-18 17:12:06 -04:00
|
|
|
}
|