Skip to content
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

[RFC] Handle *some* out-of-spec behaviour explicitly #4563

Draft
wants to merge 11 commits into
base: master
Choose a base branch
from
9 changes: 7 additions & 2 deletions src/endgame.h
Original file line number Diff line number Diff line change
Expand Up @@ -109,9 +109,14 @@ namespace Endgames {
template<EndgameCode E, typename T = eg_type<E>>
void add(const std::string& code) {

Position pos;
StateInfo st;
map<T>()[Position().set(code, WHITE, &st).material_key()] = Ptr<T>(new Endgame<E>(WHITE));
map<T>()[Position().set(code, BLACK, &st).material_key()] = Ptr<T>(new Endgame<E>(BLACK));

pos.set(code, WHITE, &st);
map<T>()[pos.material_key()] = Ptr<T>(new Endgame<E>(WHITE));

pos.set(code, BLACK, &st);
map<T>()[pos.material_key()] = Ptr<T>(new Endgame<E>(BLACK));
}

template<typename T>
Expand Down
250 changes: 203 additions & 47 deletions src/position.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -154,10 +154,23 @@ void Position::init() {


/// Position::set() initializes the position object with the given FEN string.
/// This function is not very robust - make sure that input FENs are correct,
/// this is assumed to be the responsibility of the GUI.

Position& Position::set(const string& fenStr, bool isChess960, StateInfo* si, Thread* th) {
/// Some validation is performed and positions that are invalid, or not supported,
/// or potentially not supported, by Stockfish, will be rejected and an error returned.
/// The state of the Position after an error is undefined. In such case Position::set must be
/// called again with a valid position before the Position object can be used.
///
/// Stockfish informally requires that the position passed to Position::set is reachable from
/// chess (or any chess960) starting position.
/// Validation, however, will only reject the following kinds of positions (provided the FEN itself is valid in the first place):
/// - any side has less or more than 1 king
/// - opponent's king is in check
/// - any side has more than 16 pieces
/// - any side has more than 8 pawns
/// - any side has material configuration that is unattainable through pawn promotions
/// - rule50 counter is above 150 (75-move rule, though note that any non-mate position with rule50>100 is considered a draw)
/// - fullmove counter exceeds the maximum theoretical game length

std::optional<PositionSetError> Position::set(const string& fenStr, bool isChess960, StateInfo* si, Thread* th) {
/*
A FEN string defines a particular position using only the ASCII character set.

Expand Down Expand Up @@ -187,15 +200,13 @@ Position& Position::set(const string& fenStr, bool isChess960, StateInfo* si, Th

5) Halfmove clock. This is the number of halfmoves since the last pawn advance
or capture. This is used to determine if a draw can be claimed under the
fifty-move rule.
fifty-move rule. This field is optional. Default: 0.

6) Fullmove number. The number of the full move. It starts at 1, and is
incremented after Black's move.
incremented after Black's move. This field is optional. Default: 1.
*/

unsigned char col, row, token;
size_t idx;
Square sq = SQ_A8;
unsigned char token;
std::istringstream ss(fenStr);

std::memset(this, 0, sizeof(Position));
Expand All @@ -204,89 +215,208 @@ Position& Position::set(const string& fenStr, bool isChess960, StateInfo* si, Th

ss >> std::noskipws;

int numPieces = 0;
File file = FILE_A;
Rank rank = RANK_8;

// 1. Piece placement
while ((ss >> token) && !isspace(token))
for (;;)
{
if (isdigit(token))
sq += (token - '0') * EAST; // Advance the given number of files
if (!(ss >> token))
return PositionSetError("Invalid FEN. Unexpected end of stream.");

if (isspace(token))
break;

if (isdigit(token))
{
const int diff = token - '0';
if (diff < 1 || diff > 8)
return PositionSetError("Invalid FEN. Invalid number of squares to skip.");
file = File(file + diff);
if (file > FILE_NB)
return PositionSetError("Invalid FEN. Invalid file reached.");
}
else if (token == '/')
sq += 2 * SOUTH;
{
if (file != FILE_NB)
return PositionSetError("Invalid FEN. Trying to end rank when not at the end of it.");

--rank;
file = FILE_A;

else if ((idx = PieceToChar.find(token)) != string::npos) {
if (rank < RANK_1)
return PositionSetError("Invalid FEN. Invalid rank reached.");
}
else
{
const size_t idx = PieceToChar.find(token);
if (idx == string::npos)
return PositionSetError(std::string("Invalid FEN. Invalid piece: ") + std::string(1, token));
if (++numPieces > 32)
return PositionSetError("Invalid FEN. More than 32 pieces on the board.");

const Square sq = make_square(file, rank);
put_piece(Piece(idx), sq);
++sq;

++file;
if (file > FILE_NB)
return PositionSetError("Invalid FEN. Invalid file reached.");
}
}

if (rank != RANK_1 || file != FILE_NB)
return PositionSetError("Invalid FEN. Board state encoding ended but cursor not at end.");

{
const int wPawns = count<PAWN>(WHITE);
const int bPawns = count<PAWN>(BLACK);
if (wPawns > 8)
return PositionSetError("Invalid FEN. WHITE has more than 8 pawns.");
if (bPawns > 8)
return PositionSetError("Invalid FEN. BLACK has more than 8 pawns.");

const int wAdditionalKnights = std::max((int)count<KNIGHT>(WHITE) - 2, 0);
const int bAdditionalKnights = std::max((int)count<KNIGHT>(BLACK) - 2, 0);
const int wAdditionalBishops = std::max((int)count<BISHOP>(WHITE) - 2, 0);
Copy link
Contributor

@robertnurnberg robertnurnberg Jun 9, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Tiny nit: one could rather check difference of number of DSBs and LSBs to 1, for both white and black. (not sure how easy the number of dark squared bishops can be obtained, for example)

This would help catch a board with two white LSBs, no white DSB and 8 white pawns, say.

const int bAdditionalBishops = std::max((int)count<BISHOP>(BLACK) - 2, 0);
const int wAdditionalRooks = std::max((int)count<ROOK>(WHITE) - 2, 0);
const int bAdditionalRooks = std::max((int)count<ROOK>(BLACK) - 2, 0);
const int wAdditionalQueens = std::max((int)count<QUEEN>(WHITE) - 1, 0);
const int bAdditionalQueens = std::max((int)count<QUEEN>(BLACK) - 1, 0);
if (wAdditionalKnights + wAdditionalBishops + wAdditionalRooks + wAdditionalQueens > 8 - wPawns)
Copy link
Contributor

@robertnurnberg robertnurnberg Jun 9, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If one defines 8 - wPawns as wMaxPromotions, the latter could be set to zero if all opposing pawns are still on their original rank. (plus some other cases, where they form an impenetrable wall, that could never have been overcome by an opposing pawn). But these kind of complicated checks may already go beyond the original scope of this PR.

return PositionSetError("Invalid FEN. Invalid piece configuration for WHITE.");
if (bAdditionalKnights + bAdditionalBishops + bAdditionalRooks + bAdditionalQueens > 8 - bPawns)
return PositionSetError("Invalid FEN. Invalid piece configuration for BLACK.");
}

ss >> std::ws;

// 2. Active color
ss >> token;
if (!(ss >> token))
return PositionSetError("Invalid FEN. Unexpected end of stream.");
if (token != 'w' && token != 'b')
return PositionSetError(std::string("Invalid FEN. Invalid side to move: ") + std::string(1, token));
sideToMove = (token == 'w' ? WHITE : BLACK);
ss >> token;

ss >> std::ws;

// 3. Castling availability. Compatible with 3 standards: Normal FEN standard,
// Shredder-FEN that uses the letters of the columns on which the rooks began
// the game instead of KQkq and also X-FEN standard that, in case of Chess960,
// if an inner rook is associated with the castling right, the castling tag is
// replaced by the file letter of the involved rook, as for the Shredder-FEN.
while ((ss >> token) && !isspace(token))
int num_castling_rights = 0;
for (;;)
{
if (!(ss >> token))
return PositionSetError("Invalid FEN. Unexpected end of stream.");

if (isspace(token))
break;

if (num_castling_rights == 0 && token == '-')
break;

if (++num_castling_rights > 4)
return PositionSetError("Invalid FEN. Maximum of 4 castling rights can be specified.");

Square rsq;
Color c = islower(token) ? BLACK : WHITE;
Piece rook = make_piece(c, ROOK);

token = char(toupper(token));

if (token == 'K')
for (rsq = relative_square(c, SQ_H1); piece_on(rsq) != rook; --rsq) {}

for (rsq = relative_square(c, SQ_H1); piece_on(rsq) != rook && file_of(rsq) >= FILE_A; --rsq) {}
else if (token == 'Q')
for (rsq = relative_square(c, SQ_A1); piece_on(rsq) != rook; ++rsq) {}

for (rsq = relative_square(c, SQ_A1); piece_on(rsq) != rook && file_of(rsq) <= FILE_H; ++rsq) {}
else if (token >= 'A' && token <= 'H')
rsq = make_square(File(token - 'A'), relative_rank(c, RANK_1));

else
continue;
return PositionSetError(std::string("Invalid FEN. Expected castling rights. Got: ") + std::string(1, token));

if (piece_on(rsq) != rook)
return PositionSetError("Invalid FEN. Trying to set castling rights without required rook.");

set_castling_right(c, rsq);
}

// 4. En passant square.
// Ignore if square is invalid or not on side to move relative rank 6.
bool enpassant = false;
ss >> std::ws;

if ( ((ss >> col) && (col >= 'a' && col <= 'h'))
&& ((ss >> row) && (row == (sideToMove == WHITE ? '6' : '3'))))
// 4. En passant square. Faux ep-square is ignored. Otherwise invalid ep-square is an error.
bool enpassant = false;
unsigned char col, row;
if (!(ss >> col))
return PositionSetError("Invalid FEN. Unexpected end of stream.");
if (col != '-')
{
st->epSquare = make_square(File(col - 'a'), Rank(row - '1'));

// En passant square will be considered only if
// a) side to move have a pawn threatening epSquare
// b) there is an enemy pawn in front of epSquare
// c) there is no piece on epSquare or behind epSquare
enpassant = pawn_attacks_bb(~sideToMove, st->epSquare) & pieces(sideToMove, PAWN)
&& (pieces(~sideToMove, PAWN) & (st->epSquare + pawn_push(~sideToMove)))
&& !(pieces() & (st->epSquare | (st->epSquare + pawn_push(sideToMove))));
if (!(ss >> row))
return PositionSetError("Invalid FEN. Unexpected end of stream.");

if ( (col >= 'a' && col <= 'h')
&& (row == (sideToMove == WHITE ? '6' : '3')))
{
st->epSquare = make_square(File(col - 'a'), Rank(row - '1'));

// En passant square will be considered only if
// a) side to move have a pawn threatening epSquare
// b) there is an enemy pawn in front of epSquare
// c) there is no piece on epSquare or behind epSquare
enpassant = pawn_attacks_bb(~sideToMove, st->epSquare) & pieces(sideToMove, PAWN)
&& (pieces(~sideToMove, PAWN) & (st->epSquare + pawn_push(~sideToMove)))
&& !(pieces() & (st->epSquare | (st->epSquare + pawn_push(sideToMove))));
}
else
return PositionSetError("Invalid FEN. Invalid en-passant square.");
}

if (!enpassant)
st->epSquare = SQ_NONE;

// 5-6. Halfmove clock and fullmove number
ss >> std::skipws >> st->rule50 >> gamePly;
ss >> std::skipws;

// 5-6. Halfmove clock and fullmove number. Either none or both must be present.
// If they are not present then values are the same as for startpos.
if (ss >> st->rule50)
{
if (!(ss >> gamePly))
return PositionSetError("Invalid FEN. Unexpected end of stream or invalid numbers.");
}
else
{
st->rule50 = 0;
gamePly = 1;
}

// Technically, positions with rule50==100 are correct, just no moves can be made further.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is quite unnecessary IMO, rule50 count doesn't really affect gameplay, and user can easily artifically bypass it anyways.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

While the 50-move rule and 75-move rule do apply in over-the-board play, they do not apply for certain purposes in endgame theory. In particular, endgame tablebases will allow one to step through a forced with that takes far longer, and this would (presulably) result in a larger count here. Instead of refusing to import the FEN, I think it would be better for Stockfish to clamp rule50 to 150.

// However, due to human stuff, we want to *support* rule50 up to 150.
constexpr int MaxRule50Fullmoves = 75;
if (st->rule50 < 0 || st->rule50 > MaxRule50Fullmoves * 2)
return PositionSetError("Invalid FEN. Rule50 counter outside of range, got: " + std::to_string(st->rule50));

// https://chess.stackexchange.com/questions/4113/longest-chess-game-possible-maximum-moves
constexpr int MaxMoves = MaxRule50Fullmoves - 1
+ 32 * MaxRule50Fullmoves
+ 6 * 8 * MaxRule50Fullmoves
+ 7 * MaxRule50Fullmoves
+ 30 * MaxRule50Fullmoves;
if (gamePly < 1 || gamePly > MaxMoves)
return PositionSetError("Invalid FEN. Full-move counter outside of range, got: " + std::to_string(gamePly));

// Convert from fullmove starting from 1 to gamePly starting from 0,
// handle also common incorrect FEN with fullmove = 0.
gamePly = std::max(2 * (gamePly - 1), 0) + (sideToMove == BLACK);

chess960 = isChess960;
thisThread = th;
set_state();

assert(pos_is_ok());
if (!pos_is_ok())
return PositionSetError("Invalid FEN.");

return *this;
set_state();

return std::nullopt;
}


Expand Down Expand Up @@ -373,7 +503,7 @@ void Position::set_state() const {
/// the given endgame code string like "KBPKN". It is mainly a helper to
/// get the material key out of an endgame code.

Position& Position::set(const string& code, Color c, StateInfo* si) {
std::optional<PositionSetError> Position::set(const string& code, Color c, StateInfo* si) {

assert(code[0] == 'K');

Expand Down Expand Up @@ -1292,42 +1422,65 @@ bool Position::pos_is_ok() const {

constexpr bool Fast = true; // Quick (default) or full check?

if ( pieceCount[W_KING] != 1
|| pieceCount[B_KING] != 1)
{
assert(0 && "pos_is_ok: King count");
return false;
}

if ( (sideToMove != WHITE && sideToMove != BLACK)
|| piece_on(square<KING>(WHITE)) != W_KING
|| piece_on(square<KING>(BLACK)) != B_KING
|| ( ep_square() != SQ_NONE
&& relative_rank(sideToMove, ep_square()) != RANK_6))
{
assert(0 && "pos_is_ok: Default");
return false;
}

if (Fast)
return true;

if ( pieceCount[W_KING] != 1
|| pieceCount[B_KING] != 1
|| attackers_to(square<KING>(~sideToMove)) & pieces(sideToMove))
if (attackers_to(square<KING>(~sideToMove)) & pieces(sideToMove))
{
assert(0 && "pos_is_ok: Kings");
return false;
}

if ( (pieces(PAWN) & (Rank1BB | Rank8BB))
|| pieceCount[W_PAWN] > 8
|| pieceCount[B_PAWN] > 8)
{
assert(0 && "pos_is_ok: Pawns");
return false;
}

if ( (pieces(WHITE) & pieces(BLACK))
|| (pieces(WHITE) | pieces(BLACK)) != pieces()
|| popcount(pieces(WHITE)) > 16
|| popcount(pieces(BLACK)) > 16)
{
assert(0 && "pos_is_ok: Bitboards");
return false;
}

for (PieceType p1 = PAWN; p1 <= KING; ++p1)
for (PieceType p2 = PAWN; p2 <= KING; ++p2)
if (p1 != p2 && (pieces(p1) & pieces(p2)))
{
assert(0 && "pos_is_ok: Bitboards");
return false;
}


for (Piece pc : Pieces)
if ( pieceCount[pc] != popcount(pieces(color_of(pc), type_of(pc)))
|| pieceCount[pc] != std::count(board, board + SQUARE_NB, pc))
{
assert(0 && "pos_is_ok: Pieces");
return false;
}

for (Color c : { WHITE, BLACK })
for (CastlingRights cr : {c & KING_SIDE, c & QUEEN_SIDE})
Expand All @@ -1338,7 +1491,10 @@ bool Position::pos_is_ok() const {
if ( piece_on(castlingRookSquare[cr]) != make_piece(c, ROOK)
|| castlingRightsMask[castlingRookSquare[cr]] != cr
|| (castlingRightsMask[square<KING>(c)] & cr) != cr)
{
assert(0 && "pos_is_ok: Castling");
return false;
}
}

return true;
Expand Down
Loading