-
Notifications
You must be signed in to change notification settings - Fork 4
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add doc comments & tests. Some refactoring #28
Changes from 2 commits
78968f1
92a53fc
2ed61bc
1a96943
10f783c
5776c78
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -25,6 +25,7 @@ impl Position { | |
} | ||
av | ||
} | ||
/// Generate moves. | ||
fn generate_all(&self, av: &mut ArrayVec<Move, MAX_LEGAL_MOVES>) { | ||
let target = !self.player_bitboard(self.side_to_move()); | ||
self.generate_for_fu(av, &target); | ||
|
@@ -39,52 +40,52 @@ impl Position { | |
self.generate_for_ry(av, &target); | ||
self.generate_drop(av, &(!self.occupied_bitboard() & !Bitboard::empty())); | ||
} | ||
/// Generate moves to evade check, optimized using AttackInfo. | ||
fn generate_evasions(&self, av: &mut ArrayVec<Move, MAX_LEGAL_MOVES>) { | ||
let c = self.side_to_move(); | ||
if let Some(king) = self.king_position(c) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Unwrapped |
||
let mut checkers_attacks = Bitboard::empty(); | ||
let mut checkers_count = 0; | ||
for ch in self.checkers() { | ||
if let Some(p) = self.piece_at(ch) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Unwrapped |
||
let pk = p.piece_kind(); | ||
// 龍が斜め位置から王手している場合のみ、他の駒の裏に逃がれることができる可能性がある | ||
if pk == PieceKind::ProRook | ||
&& ch.file() != king.file() | ||
&& ch.rank() != king.rank() | ||
{ | ||
checkers_attacks |= ATTACK_TABLE.hi.attack(ch, &self.occupied_bitboard()); | ||
} else { | ||
checkers_attacks |= ATTACK_TABLE.pseudo_attack(pk, ch, c.flip()); | ||
} | ||
} | ||
checkers_count += 1; | ||
} | ||
for to in ATTACK_TABLE.ou.attack(king, c) & !self.player_bitboard(c) & !checkers_attacks | ||
let king = self.king_position(c).unwrap(); | ||
let mut checkers_attacks = Bitboard::empty(); | ||
let mut checkers_count = 0; | ||
for ch in self.checkers() { | ||
let pk = self.piece_at(ch).unwrap().piece_kind(); | ||
// 龍が斜め位置から王手している場合のみ、他の駒の裏に逃がれることができる可能性がある | ||
if pk == PieceKind::ProRook | ||
&& ch.file() != king.file() | ||
&& ch.rank() != king.rank() | ||
{ | ||
av.push(Move::Normal { | ||
from: king, | ||
to, | ||
promote: false, | ||
}); | ||
} | ||
// 両王手の場合は玉が逃げるしかない | ||
if checkers_count > 1 { | ||
return; | ||
} | ||
if let Some(ch) = self.checkers().into_iter().next() { | ||
let target_drop = BETWEEN_TABLE[ch.array_index()][king.array_index()]; | ||
let target_move = target_drop | self.checkers(); | ||
self.generate_for_fu(av, &target_move); | ||
self.generate_for_ky(av, &target_move); | ||
self.generate_for_ke(av, &target_move); | ||
self.generate_for_gi(av, &target_move); | ||
self.generate_for_ka(av, &target_move); | ||
self.generate_for_hi(av, &target_move); | ||
self.generate_for_ki(av, &target_move); | ||
self.generate_for_um(av, &target_move); | ||
self.generate_for_ry(av, &target_move); | ||
self.generate_drop(av, &target_drop); | ||
checkers_attacks |= ATTACK_TABLE.hi.attack(ch, &self.occupied_bitboard()); | ||
} else { | ||
checkers_attacks |= ATTACK_TABLE.pseudo_attack(pk, ch, c.flip()); | ||
} | ||
checkers_count += 1; | ||
} | ||
for to in ATTACK_TABLE.ou.attack(king, c) & !self.player_bitboard(c) & !checkers_attacks | ||
{ | ||
av.push(Move::Normal { | ||
from: king, | ||
to, | ||
promote: false, | ||
}); | ||
} | ||
// 両王手の場合は玉が逃げるしかない | ||
if checkers_count > 1 { | ||
return; | ||
} | ||
let ch = self.checkers().into_iter().next().unwrap(); | ||
let target_drop = BETWEEN_TABLE[ch.array_index()][king.array_index()]; | ||
let target_move = target_drop | self.checkers(); | ||
self.generate_for_fu(av, &target_move); | ||
self.generate_for_ky(av, &target_move); | ||
self.generate_for_ke(av, &target_move); | ||
self.generate_for_gi(av, &target_move); | ||
self.generate_for_ka(av, &target_move); | ||
self.generate_for_hi(av, &target_move); | ||
self.generate_for_ki(av, &target_move); | ||
self.generate_for_um(av, &target_move); | ||
self.generate_for_ry(av, &target_move); | ||
if !target_drop.is_empty() { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'm guessing |
||
// No need to exclude occupied bitboard: Existence of cells between attacker and king is given. | ||
self.generate_drop(av, &target_drop); | ||
} | ||
} | ||
fn generate_for_fu(&self, av: &mut ArrayVec<Move, MAX_LEGAL_MOVES>, target: &Bitboard) { | ||
|
@@ -231,6 +232,7 @@ impl Position { | |
} | ||
} | ||
} | ||
// Generate moves of pieces which moves like KI | ||
fn generate_for_ki(&self, av: &mut ArrayVec<Move, MAX_LEGAL_MOVES>, target: &Bitboard) { | ||
let c = self.side_to_move(); | ||
for from in (self.piece_kind_bitboard(PieceKind::Gold) | ||
|
@@ -322,15 +324,16 @@ impl Position { | |
} | ||
} | ||
} | ||
// Checks if the move isn't illegal: king's suicidal moves and moving pinned piece away. | ||
fn is_legal(&self, m: Move) -> bool { | ||
if let Some(from) = m.from() { | ||
let c = self.side_to_move(); | ||
let king = [Piece::B_K, Piece::W_K][c.array_index()]; | ||
// 玉が相手の攻撃範囲内に動いてしまう指し手は除外 | ||
if self.piece_at(from) == Some(king) | ||
&& !self | ||
.attackers_to(c.flip(), m.to(), &self.occupied_bitboard()) | ||
.is_empty() | ||
.attackers_to(c.flip(), m.to(), &self.occupied_bitboard()) | ||
.is_empty() | ||
{ | ||
return false; | ||
} | ||
|
@@ -390,6 +393,7 @@ impl Position { | |
| (ATTACK_TABLE.ki.attack(to, opp) & (self.piece_kind_bitboard(PieceKind::Gold) | self.piece_kind_bitboard(PieceKind::ProPawn) | self.piece_kind_bitboard(PieceKind::ProLance) | self.piece_kind_bitboard(PieceKind::ProKnight) | self.piece_kind_bitboard(PieceKind::ProSilver) | self.piece_kind_bitboard(PieceKind::ProBishop) | self.piece_kind_bitboard(PieceKind::King))) | ||
) & self.player_bitboard(c) | ||
} | ||
/// Attackers except for king, lance & pawn, which are not applicable to evade check by pawn | ||
#[rustfmt::skip] | ||
fn attackers_to_except_klp(&self, c: Color, to: Square) -> Bitboard { | ||
let opp = c.flip(); | ||
|
@@ -405,6 +409,7 @@ impl Position { | |
|
||
#[cfg(test)] | ||
mod tests { | ||
use shogi_core::consts::square::SQ_2I; | ||
use super::*; | ||
use shogi_core::PartialPosition; | ||
use shogi_usi_parser::FromUsi; | ||
|
@@ -433,7 +438,7 @@ mod tests { | |
PartialPosition::from_usi( | ||
"sfen lnsgkg1nl/1r5s1/pppppp1pp/6p2/9/2P6/PP1PPPPPP/7R1/LNSGKGSNL b Bb 1", | ||
) | ||
.expect("failed to parse"), | ||
.expect("failed to parse"), | ||
); | ||
assert_eq!( | ||
43, | ||
|
@@ -466,6 +471,31 @@ mod tests { | |
assert_eq!(593, pos.legal_moves().len()); | ||
} | ||
|
||
#[test] | ||
fn evasion_moves() { | ||
// TODO: add more cases | ||
// Behind RY | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Cover the line 57, described as
|
||
// P1 * * * * * * * * * | ||
// P2 * * * * * * * * * | ||
// P3 * * * * * * * * * | ||
// P4 * * * * * * * * * | ||
// P5 * * * * * * * * * | ||
// P6 * * * * * * * -FU * | ||
// P7 * * * * * * * -RY * | ||
// P8 * * * * * * +OU+KE * | ||
// P9 * * * * -OU * +GI * * | ||
// P+00FU | ||
// P-00AL | ||
// + | ||
let pos = Position::new( | ||
PartialPosition::from_usi("sfen 9/9/9/9/9/7p1/7+r1/6KN1/4k1S2 b Pr2b4g3s3n4l16p 1") | ||
.expect("failed to parse"), | ||
); | ||
let moves = pos.legal_moves(); | ||
assert_eq!(1, moves.len()); | ||
assert_eq!(SQ_2I, moves[0].to()); | ||
} | ||
|
||
#[test] | ||
fn pawn_drop() { | ||
{ | ||
|
@@ -485,7 +515,7 @@ mod tests { | |
PartialPosition::from_usi( | ||
"sfen lnsgkgsnl/1r5s1/pppppppp1/9/8L/9/PPPPPPPP1/1B5S1/LNSGKGSN1 w Pp 1", | ||
) | ||
.expect("failed to parse"), | ||
.expect("failed to parse"), | ||
); | ||
let drop_moves = pos | ||
.legal_moves() | ||
|
@@ -518,7 +548,7 @@ mod tests { | |
PartialPosition::from_usi( | ||
"sfen lnsgkgsn1/1r5s1/pppppppp1/9/8l/9/PPPPPPPP1/1B5S1/LNSGKGSN1 b Ppl 1", | ||
) | ||
.expect("failed to parse"), | ||
.expect("failed to parse"), | ||
); | ||
let drop_moves = pos | ||
.legal_moves() | ||
|
@@ -642,7 +672,7 @@ mod tests { | |
PartialPosition::from_usi( | ||
"sfen 6B2/7np/8k/7P1/7G1/9/9/9/9 b P2rb3g4s3n4l15p 1", | ||
) | ||
.expect("failed to parse"), | ||
.expect("failed to parse"), | ||
), | ||
Square::SQ_1D, | ||
true, | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Wanted to make sure where the positives are