stub: proper time management
the issue here is that something is wrong and it keeps blundering
This commit is contained in:
parent
1158c817a7
commit
64edde9bad
191
src/coordination.rs
Normal file
191
src/coordination.rs
Normal file
@ -0,0 +1,191 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
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>
|
||||||
|
*/
|
||||||
|
|
||||||
|
//! Threading, state, and flow of information management.
|
||||||
|
//!
|
||||||
|
//! This file contains types and helper utilities; see main for actual implementation.
|
||||||
|
|
||||||
|
use crate::prelude::*;
|
||||||
|
|
||||||
|
/// State machine states.
|
||||||
|
#[derive(Clone, Copy, Debug)]
|
||||||
|
pub enum UCIMode {
|
||||||
|
/// It is engine's turn; engine is thinking about a move.
|
||||||
|
Think,
|
||||||
|
/// It is the opponent's turn; engine is thinking about a move.
|
||||||
|
Ponder,
|
||||||
|
/// The engine is not doing anything.
|
||||||
|
Idle,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// State machine transitions.
|
||||||
|
#[derive(Clone, Copy, Debug)]
|
||||||
|
pub enum UCIModeTransition {
|
||||||
|
/// Engine produces a best move result. Thinking to Idle.
|
||||||
|
Bestmove,
|
||||||
|
/// Engine is stopped via a UCI `stop` command. Thinking/Ponder to Idle.
|
||||||
|
Stop,
|
||||||
|
/// Engine is asked for a best move through a UCI `go`. Idle -> Thinking.
|
||||||
|
Go,
|
||||||
|
/// Engine starts pondering on the opponent's time. Idle -> Ponder.
|
||||||
|
GoPonder,
|
||||||
|
/// While engine ponders, the opponent plays a different move than expected. Ponder -> Thinking
|
||||||
|
///
|
||||||
|
/// In UCI, this means that a new `position` command is sent.
|
||||||
|
PonderMiss,
|
||||||
|
/// While engine ponders, the opponent plays the expected move (`ponderhit`). Ponder -> Thinking
|
||||||
|
PonderHit,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl UCIModeTransition {
|
||||||
|
/// The state that a transition goes to.
|
||||||
|
const fn dest_mode(&self) -> UCIMode {
|
||||||
|
use UCIMode::*;
|
||||||
|
use UCIModeTransition::*;
|
||||||
|
match self {
|
||||||
|
Bestmove => Idle,
|
||||||
|
Stop => Idle,
|
||||||
|
Go => Think,
|
||||||
|
GoPonder => Ponder,
|
||||||
|
PonderMiss => Think,
|
||||||
|
PonderHit => Think,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// State machine for engine's UCI modes.
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct UCIModeMachine {
|
||||||
|
pub mode: UCIMode,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct InvalidTransitionError {
|
||||||
|
/// Original state.
|
||||||
|
pub from: UCIMode,
|
||||||
|
/// Desired destination state.
|
||||||
|
pub to: UCIMode,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for UCIModeMachine {
|
||||||
|
fn default() -> Self {
|
||||||
|
UCIModeMachine {
|
||||||
|
mode: UCIMode::Idle,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl UCIModeMachine {
|
||||||
|
/// Change state (checked to prevent invalid transitions.)
|
||||||
|
pub fn transition(&mut self, t: UCIModeTransition) -> Result<(), InvalidTransitionError> {
|
||||||
|
macro_rules! illegal {
|
||||||
|
() => {
|
||||||
|
return Err(InvalidTransitionError {
|
||||||
|
from: self.mode,
|
||||||
|
to: t.dest_mode(),
|
||||||
|
})
|
||||||
|
};
|
||||||
|
}
|
||||||
|
macro_rules! legal {
|
||||||
|
() => {{
|
||||||
|
self.mode = t.dest_mode();
|
||||||
|
return Ok(());
|
||||||
|
}};
|
||||||
|
}
|
||||||
|
|
||||||
|
use UCIModeTransition::*;
|
||||||
|
|
||||||
|
match t {
|
||||||
|
Bestmove => match self.mode {
|
||||||
|
UCIMode::Think => legal!(),
|
||||||
|
_ => illegal!(),
|
||||||
|
},
|
||||||
|
Stop => match self.mode {
|
||||||
|
UCIMode::Ponder | UCIMode::Think => legal!(),
|
||||||
|
_ => illegal!(),
|
||||||
|
},
|
||||||
|
Go | GoPonder => match self.mode {
|
||||||
|
UCIMode::Idle => legal!(),
|
||||||
|
_ => illegal!(),
|
||||||
|
},
|
||||||
|
PonderMiss => match self.mode {
|
||||||
|
UCIMode::Ponder => legal!(),
|
||||||
|
_ => illegal!(),
|
||||||
|
},
|
||||||
|
PonderHit => match self.mode {
|
||||||
|
UCIMode::Ponder => legal!(),
|
||||||
|
_ => illegal!(),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test_state_machine {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
/// Non-exhaustive test of state machine.
|
||||||
|
#[test]
|
||||||
|
fn test_transitions() {
|
||||||
|
let mut machine = UCIModeMachine {
|
||||||
|
mode: UCIMode::Idle,
|
||||||
|
};
|
||||||
|
assert!(matches!(machine.transition(UCIModeTransition::Go), Ok(())));
|
||||||
|
assert!(matches!(machine.mode, UCIMode::Think));
|
||||||
|
assert!(matches!(
|
||||||
|
machine.transition(UCIModeTransition::Stop),
|
||||||
|
Ok(())
|
||||||
|
));
|
||||||
|
assert!(matches!(machine.mode, UCIMode::Idle));
|
||||||
|
assert!(matches!(machine.transition(UCIModeTransition::Go), Ok(())));
|
||||||
|
assert!(matches!(
|
||||||
|
machine.transition(UCIModeTransition::Bestmove),
|
||||||
|
Ok(())
|
||||||
|
));
|
||||||
|
assert!(matches!(machine.mode, UCIMode::Idle));
|
||||||
|
assert!(matches!(
|
||||||
|
machine.transition(UCIModeTransition::Bestmove),
|
||||||
|
Err(_)
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Message (engine->main) to communicate the best move.
|
||||||
|
pub struct MsgBestmove {
|
||||||
|
/// Best line (reversed stack; last element is best current move)
|
||||||
|
pub pv: Vec<Move>,
|
||||||
|
/// Evaluation of the position
|
||||||
|
pub eval: SearchEval,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Interface messages that may be received by main's channel.
|
||||||
|
pub enum MsgToMain {
|
||||||
|
StdinLine(String),
|
||||||
|
Bestmove(MsgBestmove),
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct GoMessage {
|
||||||
|
pub board: Board,
|
||||||
|
pub config: SearchConfig,
|
||||||
|
pub time_lims: TimeLimits,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Main -> Engine thread channel message.
|
||||||
|
pub enum MsgToEngine {
|
||||||
|
/// `go` command. Also sends board position and engine configuration to avoid state
|
||||||
|
/// synchronization issues (i.e. avoid sending position after a go command, and not before).
|
||||||
|
Go(Box<GoMessage>),
|
||||||
|
/// Hard stop command. Halt search immediately.
|
||||||
|
Stop,
|
||||||
|
/// Ask the engine to wipe its state (notably transposition table).
|
||||||
|
NewGame,
|
||||||
|
}
|
@ -17,6 +17,7 @@ use std::fmt::Display;
|
|||||||
use std::ops::{Index, IndexMut};
|
use std::ops::{Index, IndexMut};
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
|
||||||
|
pub mod coordination;
|
||||||
pub mod eval;
|
pub mod eval;
|
||||||
pub mod fen;
|
pub mod fen;
|
||||||
mod hash;
|
mod hash;
|
||||||
@ -432,7 +433,7 @@ impl Display for CastleRights {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Immutable game state, unique to a position.
|
/// Game state, describes a position.
|
||||||
///
|
///
|
||||||
/// Default is empty.
|
/// Default is empty.
|
||||||
#[derive(Debug, Default, Clone, Copy, PartialEq, Eq)]
|
#[derive(Debug, Default, Clone, Copy, PartialEq, Eq)]
|
||||||
|
452
src/main.rs
452
src/main.rs
@ -12,164 +12,34 @@ Copyright © 2024 dogeystamp <dogeystamp@disroot.org>
|
|||||||
|
|
||||||
//! Main UCI engine binary.
|
//! Main UCI engine binary.
|
||||||
//!
|
//!
|
||||||
//! This runs three threads, main, engine, and stdin. The main thread coordinates everything, and
|
//! # Architecture
|
||||||
//! performs UCI parsing/communication. `stdin` is read on a different thread, in order to avoid
|
|
||||||
//! blocking on it. The engine thread is where the actual computation happens. It communicates
|
|
||||||
//! state (best move, evaluations, board state and configuration) with the main thread.
|
|
||||||
//!
|
//!
|
||||||
//! The main thread has a single rx (receive) channel. This is so that it can wait for either the
|
//! This runs three threads, Main, Engine, and Stdin. Main coordinates everything, and performs UCI
|
||||||
//! engine to finish a computation, or for stdin to receive a UCI command. This way, the engine is
|
//! parsing/communication. Stdin is read on a different thread, in order to avoid blocking on it.
|
||||||
//! always listening, even when it is thinking.
|
//! The Engine is where the actual computation happens. It communicates state (best move, evaluations,
|
||||||
|
//! board state and configuration) with Main.
|
||||||
|
//!
|
||||||
|
//! Main has a single rx (receive) channel. This is so that it can wait for either the Engine to
|
||||||
|
//! finish a computation, or for Stdin to receive a UCI command. This way, the overall engine
|
||||||
|
//! program is always listening, even when it is thinking.
|
||||||
|
//!
|
||||||
|
//! For every go command, Main sends data, notably the current position and engine configuration,
|
||||||
|
//! to the Engine. The current position and config are re-sent every time because Main is where the
|
||||||
|
//! opponent's move, as well as any configuration options, are read and parsed. Meanwhile, internal
|
||||||
|
//! data, like the transposition table, is owned by the Engine thread.
|
||||||
|
//!
|
||||||
|
//! # Notes
|
||||||
|
//!
|
||||||
|
//! - The naming scheme for channels here is `tx_main`, `rx_main` for "transmit to Main" and
|
||||||
|
//! "receive at Main" respectively. These names would be used for one channel.
|
||||||
|
|
||||||
use chess_inator::prelude::*;
|
use chess_inator::prelude::*;
|
||||||
use std::cmp::min;
|
use std::cmp::min;
|
||||||
use std::io;
|
use std::io;
|
||||||
use std::sync::mpsc::{channel, Sender};
|
use std::process::exit;
|
||||||
|
use std::sync::mpsc::{channel, Receiver, Sender};
|
||||||
use std::thread;
|
use std::thread;
|
||||||
use std::time::Duration;
|
use std::time::{Duration, Instant};
|
||||||
|
|
||||||
/// State machine states.
|
|
||||||
#[derive(Clone, Copy, Debug)]
|
|
||||||
enum UCIMode {
|
|
||||||
/// It is engine's turn; engine is thinking about a move.
|
|
||||||
Think,
|
|
||||||
/// It is the opponent's turn; engine is thinking about a move.
|
|
||||||
Ponder,
|
|
||||||
/// The engine is not doing anything.
|
|
||||||
Idle,
|
|
||||||
}
|
|
||||||
|
|
||||||
/// State machine transitions.
|
|
||||||
#[derive(Clone, Copy, Debug)]
|
|
||||||
enum UCIModeTransition {
|
|
||||||
/// Engine produces a best move result. Thinking to Idle.
|
|
||||||
Bestmove,
|
|
||||||
/// Engine is stopped via a UCI `stop` command. Thinking/Ponder to Idle.
|
|
||||||
Stop,
|
|
||||||
/// Engine is asked for a best move through a UCI `go`. Idle -> Thinking.
|
|
||||||
Go,
|
|
||||||
/// Engine starts pondering on the opponent's time. Idle -> Ponder.
|
|
||||||
GoPonder,
|
|
||||||
/// While engine ponders, the opponent plays a different move than expected. Ponder -> Thinking
|
|
||||||
///
|
|
||||||
/// In UCI, this means that a new `position` command is sent.
|
|
||||||
PonderMiss,
|
|
||||||
/// While engine ponders, the opponent plays the expected move (`ponderhit`). Ponder -> Thinking
|
|
||||||
PonderHit,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl UCIModeTransition {
|
|
||||||
/// The state that a transition goes to.
|
|
||||||
const fn dest_mode(&self) -> UCIMode {
|
|
||||||
use UCIMode::*;
|
|
||||||
use UCIModeTransition::*;
|
|
||||||
match self {
|
|
||||||
Bestmove => Idle,
|
|
||||||
Stop => Idle,
|
|
||||||
Go => Think,
|
|
||||||
GoPonder => Ponder,
|
|
||||||
PonderMiss => Think,
|
|
||||||
PonderHit => Think,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// State machine for engine's UCI modes.
|
|
||||||
#[derive(Debug)]
|
|
||||||
struct UCIModeMachine {
|
|
||||||
mode: UCIMode,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
struct InvalidTransitionError {
|
|
||||||
/// Original state.
|
|
||||||
from: UCIMode,
|
|
||||||
/// Desired destination state.
|
|
||||||
to: UCIMode,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Default for UCIModeMachine {
|
|
||||||
fn default() -> Self {
|
|
||||||
UCIModeMachine {
|
|
||||||
mode: UCIMode::Idle,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl UCIModeMachine {
|
|
||||||
/// Change state (checked to prevent invalid transitions.)
|
|
||||||
fn transition(&mut self, t: UCIModeTransition) -> Result<(), InvalidTransitionError> {
|
|
||||||
macro_rules! illegal {
|
|
||||||
() => {
|
|
||||||
return Err(InvalidTransitionError {
|
|
||||||
from: self.mode,
|
|
||||||
to: t.dest_mode(),
|
|
||||||
})
|
|
||||||
};
|
|
||||||
}
|
|
||||||
macro_rules! legal {
|
|
||||||
() => {{
|
|
||||||
self.mode = t.dest_mode();
|
|
||||||
return Ok(());
|
|
||||||
}};
|
|
||||||
}
|
|
||||||
|
|
||||||
use UCIModeTransition::*;
|
|
||||||
|
|
||||||
match t {
|
|
||||||
Bestmove => match self.mode {
|
|
||||||
UCIMode::Think => legal!(),
|
|
||||||
_ => illegal!(),
|
|
||||||
},
|
|
||||||
Stop => match self.mode {
|
|
||||||
UCIMode::Ponder | UCIMode::Think => legal!(),
|
|
||||||
_ => illegal!(),
|
|
||||||
},
|
|
||||||
Go | GoPonder => match self.mode {
|
|
||||||
UCIMode::Idle => legal!(),
|
|
||||||
_ => illegal!(),
|
|
||||||
},
|
|
||||||
PonderMiss => match self.mode {
|
|
||||||
UCIMode::Ponder => legal!(),
|
|
||||||
_ => illegal!(),
|
|
||||||
},
|
|
||||||
PonderHit => match self.mode {
|
|
||||||
UCIMode::Ponder => legal!(),
|
|
||||||
_ => illegal!(),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(test)]
|
|
||||||
mod test_state_machine {
|
|
||||||
use super::*;
|
|
||||||
|
|
||||||
/// Non-exhaustive test of state machine.
|
|
||||||
#[test]
|
|
||||||
fn test_transitions() {
|
|
||||||
let mut machine = UCIModeMachine {
|
|
||||||
mode: UCIMode::Idle,
|
|
||||||
};
|
|
||||||
assert!(matches!(machine.transition(UCIModeTransition::Go), Ok(())));
|
|
||||||
assert!(matches!(machine.mode, UCIMode::Think));
|
|
||||||
assert!(matches!(
|
|
||||||
machine.transition(UCIModeTransition::Stop),
|
|
||||||
Ok(())
|
|
||||||
));
|
|
||||||
assert!(matches!(machine.mode, UCIMode::Idle));
|
|
||||||
assert!(matches!(machine.transition(UCIModeTransition::Go), Ok(())));
|
|
||||||
assert!(matches!(
|
|
||||||
machine.transition(UCIModeTransition::Bestmove),
|
|
||||||
Ok(())
|
|
||||||
));
|
|
||||||
assert!(matches!(machine.mode, UCIMode::Idle));
|
|
||||||
assert!(matches!(
|
|
||||||
machine.transition(UCIModeTransition::Bestmove),
|
|
||||||
Err(_)
|
|
||||||
));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// UCI protocol says to ignore any unknown words.
|
/// UCI protocol says to ignore any unknown words.
|
||||||
///
|
///
|
||||||
@ -206,7 +76,7 @@ fn cmd_position_moves(mut tokens: std::str::SplitWhitespace<'_>, mut board: Boar
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Sets the position.
|
/// Sets the position.
|
||||||
fn cmd_position(mut tokens: std::str::SplitWhitespace<'_>) -> Board {
|
fn cmd_position(mut tokens: std::str::SplitWhitespace<'_>, state: &mut MainState) {
|
||||||
while let Some(token) = tokens.next() {
|
while let Some(token) = tokens.next() {
|
||||||
match token {
|
match token {
|
||||||
"fen" => {
|
"fen" => {
|
||||||
@ -223,95 +93,81 @@ fn cmd_position(mut tokens: std::str::SplitWhitespace<'_>) -> Board {
|
|||||||
.unwrap_or_else(|e| panic!("failed to parse fen '{fen}': {e:?}"));
|
.unwrap_or_else(|e| panic!("failed to parse fen '{fen}': {e:?}"));
|
||||||
let board = cmd_position_moves(tokens, board);
|
let board = cmd_position_moves(tokens, board);
|
||||||
|
|
||||||
return board;
|
state.board = board;
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
"startpos" => {
|
"startpos" => {
|
||||||
let board = Board::starting_pos();
|
let board = Board::starting_pos();
|
||||||
let board = cmd_position_moves(tokens, board);
|
let board = cmd_position_moves(tokens, board);
|
||||||
|
|
||||||
return board;
|
state.board = board;
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
_ => ignore!(),
|
_ => ignore!(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
panic!("position command was empty")
|
eprintln!("cmd_position: position command was empty")
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Play the game.
|
/// Play the game.
|
||||||
fn cmd_go(
|
fn cmd_go(mut tokens: std::str::SplitWhitespace<'_>, state: &mut MainState) {
|
||||||
mut tokens: std::str::SplitWhitespace<'_>,
|
// hard timeout
|
||||||
board: &mut Board,
|
let mut hard_ms = 15_000;
|
||||||
cache: &mut TranspositionTable,
|
// soft timeout
|
||||||
) {
|
let mut soft_ms = 1_650;
|
||||||
// interface-to-engine
|
|
||||||
let (tx1, rx) = channel();
|
|
||||||
let tx2 = tx1.clone();
|
|
||||||
|
|
||||||
// can expect a 1sec soft timeout to result in more time than that of thinking
|
macro_rules! set_time {
|
||||||
let mut timeout = 1650;
|
() => {
|
||||||
|
if state.board.get_turn() == Color::White {
|
||||||
|
if let Some(time) = tokens.next() {
|
||||||
|
if let Ok(time) = time.parse::<u64>() {
|
||||||
|
let factor = if (time > 20_000) { 10 } else { 40 };
|
||||||
|
hard_ms = min(time / factor, hard_ms);
|
||||||
|
|
||||||
|
soft_ms = min(time / 50, soft_ms);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
while let Some(token) = tokens.next() {
|
while let Some(token) = tokens.next() {
|
||||||
match token {
|
match token {
|
||||||
"wtime" => {
|
"wtime" => {
|
||||||
if board.get_turn() == Color::White {
|
set_time!()
|
||||||
if let Some(time) = tokens.next() {
|
|
||||||
if let Ok(time) = time.parse::<u64>() {
|
|
||||||
timeout = min(time / 50, timeout);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
"btime" => {
|
"btime" => {
|
||||||
if board.get_turn() == Color::Black {
|
set_time!()
|
||||||
if let Some(time) = tokens.next() {
|
|
||||||
if let Ok(time) = time.parse::<u64>() {
|
|
||||||
timeout = min(time / 50, timeout);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
_ => ignore!(),
|
_ => ignore!(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// timeout
|
let hard_limit = Instant::now() + Duration::from_millis(hard_ms);
|
||||||
thread::spawn(move || {
|
let soft_limit = Instant::now() + Duration::from_millis(soft_ms);
|
||||||
thread::sleep(Duration::from_millis(timeout));
|
|
||||||
let _ = tx2.send(InterfaceMsg::Stop);
|
|
||||||
});
|
|
||||||
|
|
||||||
let mut engine_state = EngineState::new(SearchConfig::default(), rx, cache);
|
state
|
||||||
let (line, eval) = best_line(board, &mut engine_state);
|
.tx_engine
|
||||||
|
.send(MsgToEngine::Go(Box::new(GoMessage {
|
||||||
let chosen = line.last().copied();
|
board: state.board,
|
||||||
println!(
|
config: state.config,
|
||||||
"info pv{}",
|
time_lims: TimeLimits {
|
||||||
line.iter()
|
hard: None,
|
||||||
.rev()
|
soft: Some(soft_limit),
|
||||||
.map(|mv| mv.to_uci_algebraic())
|
},
|
||||||
.fold(String::new(), |a, b| a + " " + &b)
|
})))
|
||||||
);
|
.unwrap();
|
||||||
match eval {
|
|
||||||
SearchEval::Checkmate(n) => println!("info score mate {}", n / 2),
|
|
||||||
SearchEval::Centipawns(eval) => {
|
|
||||||
println!("info score cp {}", eval,)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
match chosen {
|
|
||||||
Some(mv) => println!("bestmove {}", mv.to_uci_algebraic()),
|
|
||||||
None => println!("bestmove 0000"),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Print static evaluation of the position.
|
/// Print static evaluation of the position.
|
||||||
fn cmd_eval(mut _tokens: std::str::SplitWhitespace<'_>, board: &mut Board) {
|
fn cmd_eval(mut _tokens: std::str::SplitWhitespace<'_>, state: &mut MainState) {
|
||||||
let res = eval_metrics(board);
|
let res = eval_metrics(&state.board);
|
||||||
println!("STATIC EVAL (negative black, positive white):\n- pst: {}\n- king distance: {} ({} distance)\n- phase: {}\n- total: {}", res.pst_eval, res.king_distance_eval, res.king_distance, res.phase, res.total_eval);
|
println!("STATIC EVAL (negative black, positive white):\n- pst: {}\n- king distance: {} ({} distance)\n- phase: {}\n- total: {}", res.pst_eval, res.king_distance_eval, res.king_distance, res.phase, res.total_eval);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Root UCI parser.
|
/// Root UCI parser.
|
||||||
fn cmd_root(mut tokens: std::str::SplitWhitespace<'_>, board: &mut Board, cache: &mut TranspositionTable) {
|
fn cmd_root(mut tokens: std::str::SplitWhitespace<'_>, state: &mut MainState) {
|
||||||
while let Some(token) = tokens.next() {
|
while let Some(token) = tokens.next() {
|
||||||
match token {
|
match token {
|
||||||
"uci" => {
|
"uci" => {
|
||||||
@ -321,21 +177,33 @@ fn cmd_root(mut tokens: std::str::SplitWhitespace<'_>, board: &mut Board, cache:
|
|||||||
println!("readyok");
|
println!("readyok");
|
||||||
}
|
}
|
||||||
"ucinewgame" => {
|
"ucinewgame" => {
|
||||||
*board = Board::starting_pos();
|
if matches!(state.uci_mode.mode, UCIMode::Idle) {
|
||||||
*cache = TranspositionTable::new(24);
|
state.tx_engine.send(MsgToEngine::NewGame).unwrap();
|
||||||
|
state.board = Board::starting_pos();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"quit" => {
|
"quit" => {
|
||||||
return;
|
exit(0);
|
||||||
}
|
}
|
||||||
"position" => {
|
"position" => {
|
||||||
*board = cmd_position(tokens);
|
if matches!(state.uci_mode.mode, UCIMode::Idle) {
|
||||||
|
cmd_position(tokens, state);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"go" => {
|
"go" => {
|
||||||
cmd_go(tokens, board, cache);
|
if state.uci_mode.transition(UCIModeTransition::Go).is_ok() {
|
||||||
|
cmd_go(tokens, state);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"stop" => {
|
||||||
|
// actually setting state to stop happens when bestmove is received
|
||||||
|
if matches!(state.uci_mode.mode, UCIMode::Think | UCIMode::Ponder) {
|
||||||
|
state.tx_engine.send(MsgToEngine::Stop).unwrap();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// non-standard command.
|
// non-standard command.
|
||||||
"eval" => {
|
"eval" => {
|
||||||
cmd_eval(tokens, board);
|
cmd_eval(tokens, state);
|
||||||
}
|
}
|
||||||
_ => ignore!(),
|
_ => ignore!(),
|
||||||
}
|
}
|
||||||
@ -344,51 +212,147 @@ fn cmd_root(mut tokens: std::str::SplitWhitespace<'_>, board: &mut Board, cache:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Message (engine->main) to communicate the best move.
|
/// Format a bestmove.
|
||||||
struct MsgBestmove {
|
fn outp_bestmove(bestmove: MsgBestmove) {
|
||||||
/// Best line (reversed stack; last element is best current move)
|
let chosen = bestmove.pv.last().copied();
|
||||||
pv: Vec<Move>,
|
println!(
|
||||||
/// Evaluation of the position
|
"info pv{}",
|
||||||
eval: SearchEval,
|
bestmove
|
||||||
|
.pv
|
||||||
|
.iter()
|
||||||
|
.rev()
|
||||||
|
.map(|mv| mv.to_uci_algebraic())
|
||||||
|
.fold(String::new(), |a, b| a + " " + &b)
|
||||||
|
);
|
||||||
|
match bestmove.eval {
|
||||||
|
SearchEval::Checkmate(n) => println!("info score mate {}", n / 2),
|
||||||
|
SearchEval::Centipawns(eval) => {
|
||||||
|
println!("info score cp {}", eval,)
|
||||||
|
}
|
||||||
|
SearchEval::Stopped => {
|
||||||
|
println!("info string ERROR: stopped search")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
match chosen {
|
||||||
|
Some(mv) => println!("bestmove {}", mv.to_uci_algebraic()),
|
||||||
|
None => println!("bestmove 0000"),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Interface messages that may be received by main's channel.
|
/// The "Stdin" thread to read stdin while avoiding blocking
|
||||||
enum MsgToMain {
|
|
||||||
StdinLine(String),
|
|
||||||
Bestmove(MsgBestmove),
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Read stdin line-by-line in a non-blocking way (in another thread)
|
|
||||||
///
|
///
|
||||||
/// # Arguments
|
/// # Arguments
|
||||||
/// - `tx`: channel write end to send lines to
|
/// - `tx_main`: channel write end to send lines to
|
||||||
fn task_stdin_reader(tx: Sender<MsgToMain>) {
|
fn task_stdin_reader(tx_main: Sender<MsgToMain>) {
|
||||||
thread::spawn(move || {
|
thread::spawn(move || {
|
||||||
let stdin = io::stdin();
|
let stdin = io::stdin();
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
let mut line = String::new();
|
let mut line = String::new();
|
||||||
stdin.read_line(&mut line).unwrap();
|
stdin.read_line(&mut line).unwrap();
|
||||||
tx.send(MsgToMain::StdinLine(line)).unwrap();
|
tx_main.send(MsgToMain::StdinLine(line)).unwrap();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
/// The "Engine" thread that does all the computation.
|
||||||
let mut board = Board::starting_pos();
|
fn task_engine(tx_main: Sender<MsgToMain>, rx_engine: Receiver<MsgToEngine>) {
|
||||||
let mut transposition_table = TranspositionTable::new(24);
|
thread::spawn(move || {
|
||||||
|
let mut state = EngineState::new(
|
||||||
|
SearchConfig::default(),
|
||||||
|
rx_engine,
|
||||||
|
TranspositionTable::new(0),
|
||||||
|
TimeLimits::default(),
|
||||||
|
);
|
||||||
|
|
||||||
let (tx, rx) = channel();
|
loop {
|
||||||
task_stdin_reader(tx.clone());
|
let msg = state.rx_engine.recv().unwrap();
|
||||||
|
match msg {
|
||||||
loop {
|
MsgToEngine::Go(msg_box) => {
|
||||||
let msg = rx.recv().unwrap();
|
let mut board = msg_box.board;
|
||||||
match msg {
|
state.config = msg_box.config;
|
||||||
MsgToMain::StdinLine(line) => {
|
state.time_lims = msg_box.time_lims;
|
||||||
let tokens = line.split_whitespace();
|
let (pv, eval) = best_line(&mut board, &mut state);
|
||||||
cmd_root(tokens, &mut board, &mut transposition_table);
|
tx_main
|
||||||
|
.send(MsgToMain::Bestmove(MsgBestmove { pv, eval }))
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
MsgToEngine::Stop => {
|
||||||
|
// Main keeps track of state, so this should not happen.
|
||||||
|
panic!("Received stop while idle.");
|
||||||
|
}
|
||||||
|
MsgToEngine::NewGame => {
|
||||||
|
state.cache = TranspositionTable::new(state.config.transposition_size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/// State contained within the main thread.
|
||||||
|
///
|
||||||
|
/// This struct helps pass around this thread state.
|
||||||
|
struct MainState {
|
||||||
|
/// Channel to send messages to Engine.
|
||||||
|
tx_engine: Sender<MsgToEngine>,
|
||||||
|
/// Channel to receive messages from Engine and Stdin.
|
||||||
|
rx_main: Receiver<MsgToMain>,
|
||||||
|
/// Chessboard.
|
||||||
|
board: Board,
|
||||||
|
/// Engine configuration settings.
|
||||||
|
config: SearchConfig,
|
||||||
|
/// UCI mode state machine
|
||||||
|
uci_mode: UCIModeMachine,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl MainState {
|
||||||
|
fn new(
|
||||||
|
tx_engine: Sender<MsgToEngine>,
|
||||||
|
rx_main: Receiver<MsgToMain>,
|
||||||
|
board: Board,
|
||||||
|
config: SearchConfig,
|
||||||
|
uci_mode: UCIModeMachine,
|
||||||
|
) -> Self {
|
||||||
|
Self {
|
||||||
|
tx_engine,
|
||||||
|
rx_main,
|
||||||
|
board,
|
||||||
|
config,
|
||||||
|
uci_mode,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The "Main" thread.
|
||||||
|
fn main() {
|
||||||
|
let (tx_main, rx_main) = channel();
|
||||||
|
task_stdin_reader(tx_main.clone());
|
||||||
|
|
||||||
|
let (tx_engine, rx_engine) = channel();
|
||||||
|
task_engine(tx_main, rx_engine);
|
||||||
|
|
||||||
|
let mut state = MainState::new(
|
||||||
|
tx_engine,
|
||||||
|
rx_main,
|
||||||
|
Board::starting_pos(),
|
||||||
|
SearchConfig::default(),
|
||||||
|
UCIModeMachine::default(),
|
||||||
|
);
|
||||||
|
|
||||||
|
loop {
|
||||||
|
let msg = state.rx_main.recv().unwrap();
|
||||||
|
match msg {
|
||||||
|
MsgToMain::StdinLine(line) => {
|
||||||
|
let tokens = line.split_whitespace();
|
||||||
|
cmd_root(tokens, &mut state);
|
||||||
|
}
|
||||||
|
MsgToMain::Bestmove(msg_bestmove) => {
|
||||||
|
state
|
||||||
|
.uci_mode
|
||||||
|
.transition(UCIModeTransition::Bestmove)
|
||||||
|
.unwrap();
|
||||||
|
outp_bestmove(msg_bestmove);
|
||||||
}
|
}
|
||||||
MsgToMain::Bestmove(msg_bestmove) => todo!(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,5 +16,6 @@ Copyright © 2024 dogeystamp <dogeystamp@disroot.org>
|
|||||||
pub use crate::eval::{eval_metrics, EvalMetrics};
|
pub use crate::eval::{eval_metrics, EvalMetrics};
|
||||||
pub use crate::fen::{FromFen, ToFen};
|
pub use crate::fen::{FromFen, ToFen};
|
||||||
pub use crate::movegen::{FromUCIAlgebraic, Move, MoveGen, ToUCIAlgebraic};
|
pub use crate::movegen::{FromUCIAlgebraic, Move, MoveGen, ToUCIAlgebraic};
|
||||||
pub use crate::search::{best_line, best_move, InterfaceMsg, SearchEval, TranspositionTable, EngineState, SearchConfig};
|
pub use crate::search::{best_line, best_move, SearchEval, TranspositionTable, EngineState, SearchConfig, TimeLimits};
|
||||||
pub use crate::{Board, Color, BOARD_HEIGHT, BOARD_WIDTH, N_COLORS, N_PIECES, N_SQUARES};
|
pub use crate::{Board, Color, BOARD_HEIGHT, BOARD_WIDTH, N_COLORS, N_PIECES, N_SQUARES};
|
||||||
|
pub use crate::coordination::{UCIMode, UCIModeTransition, UCIModeMachine, MsgBestmove, MsgToMain, MsgToEngine, GoMessage};
|
||||||
|
232
src/search.rs
232
src/search.rs
@ -13,12 +13,14 @@ Copyright © 2024 dogeystamp <dogeystamp@disroot.org>
|
|||||||
|
|
||||||
//! Game-tree search.
|
//! Game-tree search.
|
||||||
|
|
||||||
|
use crate::coordination::MsgToEngine;
|
||||||
use crate::eval::{Eval, EvalInt};
|
use crate::eval::{Eval, EvalInt};
|
||||||
use crate::hash::ZobristTable;
|
use crate::hash::ZobristTable;
|
||||||
use crate::movegen::{Move, MoveGen};
|
use crate::movegen::{Move, MoveGen};
|
||||||
use crate::{Board, Piece};
|
use crate::{Board, Piece};
|
||||||
use std::cmp::max;
|
use std::cmp::max;
|
||||||
use std::sync::mpsc;
|
use std::sync::mpsc;
|
||||||
|
use std::time::Instant;
|
||||||
|
|
||||||
// min can't be represented as positive
|
// min can't be represented as positive
|
||||||
const EVAL_WORST: EvalInt = -(EvalInt::MAX);
|
const EVAL_WORST: EvalInt = -(EvalInt::MAX);
|
||||||
@ -43,6 +45,8 @@ pub enum SearchEval {
|
|||||||
Checkmate(i8),
|
Checkmate(i8),
|
||||||
/// Centipawn score.
|
/// Centipawn score.
|
||||||
Centipawns(EvalInt),
|
Centipawns(EvalInt),
|
||||||
|
/// Search was hard-stopped.
|
||||||
|
Stopped,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SearchEval {
|
impl SearchEval {
|
||||||
@ -58,6 +62,7 @@ impl SearchEval {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
SearchEval::Centipawns(eval) => Self::Centipawns(-eval),
|
SearchEval::Centipawns(eval) => Self::Centipawns(-eval),
|
||||||
|
SearchEval::Stopped => SearchEval::Stopped,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -74,6 +79,7 @@ impl From<SearchEval> for EvalInt {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
SearchEval::Centipawns(eval) => eval,
|
SearchEval::Centipawns(eval) => eval,
|
||||||
|
SearchEval::Stopped => 0,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -96,11 +102,13 @@ impl PartialOrd for SearchEval {
|
|||||||
#[derive(Clone, Copy, Debug)]
|
#[derive(Clone, Copy, Debug)]
|
||||||
pub struct SearchConfig {
|
pub struct SearchConfig {
|
||||||
/// Enable alpha-beta pruning.
|
/// Enable alpha-beta pruning.
|
||||||
alpha_beta_on: bool,
|
pub alpha_beta_on: bool,
|
||||||
/// Limit regular search depth
|
/// Limit regular search depth
|
||||||
depth: usize,
|
pub depth: usize,
|
||||||
/// Enable transposition table.
|
/// Enable transposition table.
|
||||||
enable_trans_table: bool,
|
pub enable_trans_table: bool,
|
||||||
|
/// Transposition table size (2^n where this is n)
|
||||||
|
pub transposition_size: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for SearchConfig {
|
impl Default for SearchConfig {
|
||||||
@ -110,6 +118,7 @@ impl Default for SearchConfig {
|
|||||||
// try to make this even to be more conservative and avoid horizon problem
|
// try to make this even to be more conservative and avoid horizon problem
|
||||||
depth: 10,
|
depth: 10,
|
||||||
enable_trans_table: true,
|
enable_trans_table: true,
|
||||||
|
transposition_size: 24,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -155,11 +164,34 @@ fn move_priority(board: &mut Board, mv: &Move) -> EvalInt {
|
|||||||
/// The best line (in reverse move order), and its corresponding absolute eval for the current player.
|
/// The best line (in reverse move order), and its corresponding absolute eval for the current player.
|
||||||
fn minmax(
|
fn minmax(
|
||||||
board: &mut Board,
|
board: &mut Board,
|
||||||
engine_state: &mut EngineState<'_>,
|
state: &mut EngineState,
|
||||||
depth: usize,
|
depth: usize,
|
||||||
alpha: Option<EvalInt>,
|
alpha: Option<EvalInt>,
|
||||||
beta: Option<EvalInt>,
|
beta: Option<EvalInt>,
|
||||||
) -> (Vec<Move>, SearchEval) {
|
) -> (Vec<Move>, SearchEval) {
|
||||||
|
if false {
|
||||||
|
if state.node_count % 2048 == 1 {
|
||||||
|
// respect the hard stop if given
|
||||||
|
match state.rx_engine.try_recv() {
|
||||||
|
Ok(msg) => match msg {
|
||||||
|
MsgToEngine::Go(_) => panic!("received go while thinking"),
|
||||||
|
MsgToEngine::Stop => return (Vec::new(), SearchEval::Stopped),
|
||||||
|
MsgToEngine::NewGame => panic!("received newgame while thinking"),
|
||||||
|
},
|
||||||
|
Err(e) => match e {
|
||||||
|
mpsc::TryRecvError::Empty => {}
|
||||||
|
mpsc::TryRecvError::Disconnected => panic!("thread Main stopped"),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(hard) = state.time_lims.hard {
|
||||||
|
if Instant::now() > hard {
|
||||||
|
return (Vec::new(), SearchEval::Stopped);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// default to worst, then gradually improve
|
// default to worst, then gradually improve
|
||||||
let mut alpha = alpha.unwrap_or(EVAL_WORST);
|
let mut alpha = alpha.unwrap_or(EVAL_WORST);
|
||||||
// our best is their worst
|
// our best is their worst
|
||||||
@ -179,8 +211,8 @@ fn minmax(
|
|||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
// get transposition table entry
|
// get transposition table entry
|
||||||
if engine_state.config.enable_trans_table {
|
if state.config.enable_trans_table {
|
||||||
if let Some(entry) = &engine_state.cache[board.zobrist] {
|
if let Some(entry) = &state.cache[board.zobrist] {
|
||||||
// the entry has a deeper knowledge than we do, so follow its best move exactly instead of
|
// the entry has a deeper knowledge than we do, so follow its best move exactly instead of
|
||||||
// just prioritizing what it thinks is best
|
// just prioritizing what it thinks is best
|
||||||
if entry.depth >= depth {
|
if entry.depth >= depth {
|
||||||
@ -209,8 +241,13 @@ fn minmax(
|
|||||||
|
|
||||||
for (_priority, mv) in mvs {
|
for (_priority, mv) in mvs {
|
||||||
let anti_mv = mv.make(board);
|
let anti_mv = mv.make(board);
|
||||||
let (continuation, score) =
|
let (continuation, score) = minmax(board, state, depth - 1, Some(-beta), Some(-alpha));
|
||||||
minmax(board, engine_state, depth - 1, Some(-beta), Some(-alpha));
|
|
||||||
|
// propagate hard stops
|
||||||
|
if matches!(score, SearchEval::Stopped) {
|
||||||
|
return (Vec::new(), SearchEval::Stopped);
|
||||||
|
}
|
||||||
|
|
||||||
let abs_score = score.increment();
|
let abs_score = score.increment();
|
||||||
if abs_score > abs_best {
|
if abs_score > abs_best {
|
||||||
abs_best = abs_score;
|
abs_best = abs_score;
|
||||||
@ -219,7 +256,7 @@ fn minmax(
|
|||||||
}
|
}
|
||||||
alpha = max(alpha, abs_best.into());
|
alpha = max(alpha, abs_best.into());
|
||||||
anti_mv.unmake(board);
|
anti_mv.unmake(board);
|
||||||
if alpha >= beta && engine_state.config.alpha_beta_on {
|
if alpha >= beta && state.config.alpha_beta_on {
|
||||||
// alpha-beta prune.
|
// alpha-beta prune.
|
||||||
//
|
//
|
||||||
// Beta represents the best eval that the other player can get in sibling branches
|
// Beta represents the best eval that the other player can get in sibling branches
|
||||||
@ -232,8 +269,8 @@ fn minmax(
|
|||||||
|
|
||||||
if let Some(best_move) = best_move {
|
if let Some(best_move) = best_move {
|
||||||
best_continuation.push(best_move);
|
best_continuation.push(best_move);
|
||||||
if engine_state.config.enable_trans_table {
|
if state.config.enable_trans_table {
|
||||||
engine_state.cache[board.zobrist] = Some(TranspositionEntry {
|
state.cache[board.zobrist] = Some(TranspositionEntry {
|
||||||
best_move,
|
best_move,
|
||||||
eval: abs_best,
|
eval: abs_best,
|
||||||
depth,
|
depth,
|
||||||
@ -241,16 +278,11 @@ fn minmax(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
state.node_count += 1;
|
||||||
|
|
||||||
(best_continuation, abs_best)
|
(best_continuation, abs_best)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Messages from the interface to the search thread.
|
|
||||||
pub enum InterfaceMsg {
|
|
||||||
Stop,
|
|
||||||
}
|
|
||||||
|
|
||||||
type InterfaceRx = mpsc::Receiver<InterfaceMsg>;
|
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug)]
|
#[derive(Clone, Copy, Debug)]
|
||||||
pub struct TranspositionEntry {
|
pub struct TranspositionEntry {
|
||||||
/// best move found last time
|
/// best move found last time
|
||||||
@ -264,131 +296,89 @@ pub struct TranspositionEntry {
|
|||||||
pub type TranspositionTable = ZobristTable<TranspositionEntry>;
|
pub type TranspositionTable = ZobristTable<TranspositionEntry>;
|
||||||
|
|
||||||
/// Iteratively deepen search until it is stopped.
|
/// Iteratively deepen search until it is stopped.
|
||||||
fn iter_deep(board: &mut Board, engine_state: &mut EngineState<'_>) -> (Vec<Move>, SearchEval) {
|
fn iter_deep(board: &mut Board, state: &mut EngineState) -> (Vec<Move>, SearchEval) {
|
||||||
// don't interrupt a depth 1 search so that there's at least a move to be played
|
// keep two previous lines (in case current one is halted)
|
||||||
let (mut prev_line, mut prev_eval) = minmax(board, engine_state, 1, None, None);
|
// 1 is the most recent
|
||||||
for depth in 2..=engine_state.config.depth {
|
let (mut line1, mut eval1) = minmax(board, state, 1, None, None);
|
||||||
let (line, eval) = minmax(board, engine_state, depth, None, None);
|
let (mut line2, mut eval2) = (line1.clone(), eval1);
|
||||||
|
|
||||||
match engine_state.interface.try_recv() {
|
macro_rules! ret_best {
|
||||||
Ok(msg) => match msg {
|
($depth: expr) => {
|
||||||
InterfaceMsg::Stop => {
|
if $depth & 1 == 1 && (EvalInt::from(eval1) - EvalInt::from(eval2) > 300) {
|
||||||
if depth & 1 == 1 && (EvalInt::from(eval) - EvalInt::from(prev_eval) > 300) {
|
// be skeptical if we move last and we suddenly earn a lot of
|
||||||
// be skeptical if we move last and we suddenly earn a lot of
|
// centipawns. this may be a sign of horizon problem
|
||||||
// centipawns. this may be a sign of horizon problem
|
return (line2, eval2);
|
||||||
return (prev_line, prev_eval);
|
} else {
|
||||||
} else {
|
return (line1, eval1);
|
||||||
return (line, eval);
|
}
|
||||||
}
|
};
|
||||||
}
|
|
||||||
},
|
|
||||||
Err(e) => match e {
|
|
||||||
mpsc::TryRecvError::Empty => {}
|
|
||||||
mpsc::TryRecvError::Disconnected => panic!("interface thread stopped"),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
(prev_line, prev_eval) = (line, eval);
|
|
||||||
}
|
}
|
||||||
(prev_line, prev_eval)
|
|
||||||
|
for depth in 2..=state.config.depth {
|
||||||
|
let (line, eval) = minmax(board, state, depth, None, None);
|
||||||
|
if matches!(eval, SearchEval::Stopped) {
|
||||||
|
ret_best!(depth - 1)
|
||||||
|
} else {
|
||||||
|
(line2, eval2) = (line1, eval1);
|
||||||
|
(line1, eval1) = (line, eval);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(soft_lim) = state.time_lims.soft {
|
||||||
|
if Instant::now() > soft_lim {
|
||||||
|
ret_best!(depth)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
(line1, eval1)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Helper type to avoid retyping the same arguments into every function prototype
|
/// Deadlines for the engine to think of a move.
|
||||||
pub struct EngineState<'a> {
|
#[derive(Default)]
|
||||||
/// Configuration
|
pub struct TimeLimits {
|
||||||
config: SearchConfig,
|
/// The engine must respect this time limit. It will abort if this deadline is passed.
|
||||||
/// Channel that can talk to the main thread
|
pub hard: Option<Instant>,
|
||||||
interface: InterfaceRx,
|
pub soft: Option<Instant>,
|
||||||
cache: &'a mut TranspositionTable,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> EngineState<'a> {
|
/// Helper type to avoid retyping the same arguments into every function prototype.
|
||||||
|
///
|
||||||
|
/// This should be owned outside the actual thinking part so that the engine can remember state
|
||||||
|
/// between moves.
|
||||||
|
pub struct EngineState {
|
||||||
|
pub config: SearchConfig,
|
||||||
|
/// Main -> Engine channel receiver
|
||||||
|
pub rx_engine: mpsc::Receiver<MsgToEngine>,
|
||||||
|
pub cache: TranspositionTable,
|
||||||
|
/// Nodes traversed (i.e. number of times minmax called)
|
||||||
|
node_count: usize,
|
||||||
|
pub time_lims: TimeLimits,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl EngineState {
|
||||||
pub fn new(
|
pub fn new(
|
||||||
config: SearchConfig,
|
config: SearchConfig,
|
||||||
interface: InterfaceRx,
|
interface: mpsc::Receiver<MsgToEngine>,
|
||||||
cache: &'a mut TranspositionTable,
|
cache: TranspositionTable,
|
||||||
|
time_lims: TimeLimits,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Self {
|
Self {
|
||||||
config,
|
config,
|
||||||
interface,
|
rx_engine: interface,
|
||||||
cache,
|
cache,
|
||||||
|
node_count: 0,
|
||||||
|
time_lims,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Find the best line (in reverse order) and its evaluation.
|
/// Find the best line (in reverse order) and its evaluation.
|
||||||
pub fn best_line(board: &mut Board, engine_state: &mut EngineState<'_>) -> (Vec<Move>, SearchEval) {
|
pub fn best_line(board: &mut Board, engine_state: &mut EngineState) -> (Vec<Move>, SearchEval) {
|
||||||
let (line, eval) = iter_deep(board, engine_state);
|
let (line, eval) = iter_deep(board, engine_state);
|
||||||
(line, eval)
|
(line, eval)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Find the best move.
|
/// Find the best move.
|
||||||
pub fn best_move(board: &mut Board, engine_state: &mut EngineState<'_>) -> Option<Move> {
|
pub fn best_move(board: &mut Board, engine_state: &mut EngineState) -> Option<Move> {
|
||||||
let (line, _eval) = best_line(board, engine_state);
|
let (line, _eval) = best_line(board, engine_state);
|
||||||
line.last().copied()
|
line.last().copied()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
|
||||||
mod tests {
|
|
||||||
use super::*;
|
|
||||||
use crate::fen::{FromFen, ToFen};
|
|
||||||
use crate::movegen::ToUCIAlgebraic;
|
|
||||||
|
|
||||||
/// Theoretically, alpha-beta pruning should not affect the result of minmax.
|
|
||||||
#[test]
|
|
||||||
fn alpha_beta_same_result() {
|
|
||||||
let test_cases = [
|
|
||||||
"r2q1rk1/1bp1pp1p/p2p2p1/1p1P2P1/2n1P3/3Q1P2/PbPBN2P/3RKB1R b K - 5 15",
|
|
||||||
"r1b1k2r/p1qpppbp/1p4pn/2B3N1/1PP1P3/2P5/P4PPP/RN1QR1K1 w kq - 0 14",
|
|
||||||
];
|
|
||||||
for fen in test_cases {
|
|
||||||
let mut board = Board::from_fen(fen).unwrap();
|
|
||||||
let (_tx, _rx) = mpsc::channel();
|
|
||||||
let mut _cache = ZobristTable::new(0);
|
|
||||||
let mut engine_state = EngineState::new(
|
|
||||||
SearchConfig {
|
|
||||||
alpha_beta_on: false,
|
|
||||||
depth: 3,
|
|
||||||
enable_trans_table: false,
|
|
||||||
},
|
|
||||||
_rx,
|
|
||||||
&mut _cache,
|
|
||||||
);
|
|
||||||
|
|
||||||
let mv_no_prune = best_move(
|
|
||||||
&mut board,
|
|
||||||
&mut engine_state,
|
|
||||||
)
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
assert_eq!(board.to_fen(), fen);
|
|
||||||
|
|
||||||
let (_tx, _rx) = mpsc::channel();
|
|
||||||
let mut engine_state = EngineState::new(
|
|
||||||
SearchConfig {
|
|
||||||
alpha_beta_on: true,
|
|
||||||
depth: 3,
|
|
||||||
enable_trans_table: false,
|
|
||||||
},
|
|
||||||
_rx,
|
|
||||||
&mut _cache,
|
|
||||||
);
|
|
||||||
|
|
||||||
let mv_with_prune = best_move(
|
|
||||||
&mut board,
|
|
||||||
&mut engine_state,
|
|
||||||
)
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
assert_eq!(board.to_fen(), fen);
|
|
||||||
|
|
||||||
println!(
|
|
||||||
"without ab prune got {}, otherwise {}, fen {}",
|
|
||||||
mv_no_prune.to_uci_algebraic(),
|
|
||||||
mv_with_prune.to_uci_algebraic(),
|
|
||||||
fen
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_eq!(mv_no_prune, mv_with_prune);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
Loading…
Reference in New Issue
Block a user