BarbChess - Building a chess engine


I've been playing chess recently, although I'm not very good at it. Platforms like [lichess](https://lichess.org/) or [chess.com](https://www.chess.com/home) really managed to turn one of the oldest game ever made into a modern competitive online game, with matchmaking, ranking, tools, learning contents, and also AI for training, if the player does not want to face human players directly. Chess.com in this regards is very interesting, as the user can face many AI that have specific traits in their play style.

These incredible platforms, plus the amazing Sebastian Lague's video on chess programming that has stayed in my mind for many months, motivated me to create my own chess engine, and on top of this, a custom AI that I would be able to tweak around many play styles.

It took me some time to realize that creating a chess engine stable enough to generate correctly all legal moves while being performant to calculate these 3 or 4 turns in advance, is already quite challenging, so I will leave the AI part for another time.

Structure

BarbChessEngine

The whole code is divided in two repositories. One is the engine, named BarbChessEngine: it contains the board representation and the overall analysis on it. It has 3 main capabilities:

  • Create a board from a FEN.
FEN stands for _Forsyth-Edwards Notation_, and is a standard notation for describing a particular board position of a chess game. It contains not only all the piece positions, but also which color is expected to play, which castles are available, and also if an en passant move is currently available or not.

  • Give all the legal moves playable for a board. Moves are exposed in the form of FlaggedMove, which is a structure that exposes not only the movement being done (start and end coordinate), but also additional information like if the move is a castle or not, if it is eating something, etc.


  • Execute a move, changing the board accordingly. This move should obviously be chosen among the previously generated FlaggedMove.


This project has been made in C#, and the source can be found here.

BarbChess

BarbChess is a Unity application that leverages BarbChessEngine by providing a render of the board. It also contains the implementation of different kind of players, which concretely are entities expected to choose a move to play in order to continue the game. Currently it handles:

  • A human player, which chooses moves by actually moving pieces on the board.
  • A random AI, which simply chooses a random move among the legal ones.
  • BarbAI, the custom AI, which is suppose to be at least smarter than the random one.

BarbChess source code can be found here.

Board representation

A good board representation is fundamental for a chess engine. For this aspect, I felt very in touch with the way Sebastian Lague's did it in his chess programming video.

A chess piece is represented by a single integer, and its binary representation contains the piece type in the weakest 3 digits, and the piece color with the strongest ones. If we want to represent no piece, then its a zero.


Then, the board itself is nothing more than a 64 integer array. Each square can then be represented from two ways: a classic chess coordinate with a file and a rank is convenient from a user point of view, but mostly we can use the index in the array to ease calculation. Thus, we can also create an array of offsets which represents which number one should add in order to move to a specific direction.


At this stage, we have enough elements to easily fill this board from a FEN, by writing a simple translator, and assigning each piece to its corresponding square.

Making moves

The next piece of work is to be able to apply moves to the board. This is not really a complicated task: a chess move is really nothing more than moving a piece from a start position to an end position, removing from the board a piece which may already be here.

There are some things to consider though:

  • After each move, some information must be computed from the new board and the move that has been done.
    • Castles possibilites should be reconsidered after a move implying a rook or a king.
    • Pawns that moved 2 squares should be recognized, becauses they defined an en-passant square which can be used for the next move.
  • Using these information, executing castles and en-passant require obviously more work than regular moves.

Besides the possibility of making a move, it will be very important later on to be able to unmake a move. This combination make/unmake must be perfectly stable, as it is a critical path for an AI board evaluation.

I first wanted to store a board state before each move, so that on unmake, I'll be able to simply retrieve the old board state, instead of doubling the logic which could easily lead to mistakes. Clearly it is much faster to simply apply some small changes in the board array rather than copying and storing board states among the game. On the other hand, my implementation of the board is stateful: if an AI wish to explore the board after making a move, the caller is expected to unmake it.

Generate legal moves

The idea is simple: considering a board, give all moves that the player expected to play (the player who has the trait) can legally do. Because in chess, we have the moves that can be done computed from the specific rules of each pieces (those are called pseudo-legals), but in a modern chess engine, moves that lead to a king being eaten on the next turn are illegal, and thus not playable. Legal moves are then all moves a player can do without exposing his king to an immediate loss, and if no legal moves are available, then it's a mat (or a pat but let's not consider it for now).

Pseudo-legal moves

Generating pseudo-legal moves is pretty straightforward, you just need to apply the chess rules for each piece in the board. I won't go into much details here, but I wanted to highlight two cases that are amazingly handheld in Sebastian Lague's video (again, what a great work):

Straight lines

Taking our offset array back, to generate moves on straight lines (which are important for the bishops, the rooks, and the queen which combine the two), we can simply start from the piece position, take the offsets we want, apply it to our start position, and loop until we find another piece.



piece_position = input
start_direction = 0 // 0 for rook or queen, 4 for bishop
end _direction = 0 // 4 for rook, 7 for bishop or queen
from i = start_direction to end_direction do
    create_move_to(board[piece_position + offsets[i]])

You can consider computing in advance for all squares in the board how many squares exist for every direction. This way you don't have to manage board limits anymore.

Precomputed knight moves

Knights are really easy and convenient: regardless the state of the board, a knight on a specific square always have the exact same set of pseudo-legal moves. That mean we can compute these moves in advance for all squares in the board! Easy work :)

Legal moves by exploration

A very simple and straightforward way of isolating legal moves from pseudo-legals is simply to play the move. If the opponent has a pseudo-legal move that ends on the opposite king square, then you know the move you just did is not legal.

From this board, trait to black:

Kd8, Kd7, Ke7, Kf8, Kf7 are the pseudo-legal moves. To compute legal moves, we play each of these moves, computing all opposite pseudo legal moves afterwards. We then realize that Kd8 leads to Rxd8, and Kd7 to Rxd7, so we remove Kd8 and Kd7 from the legal options.

This method has two great strengths: it is super easy to implement, and it is very robust, as long as your pseudo-legal move generation is correct. Every weird cases that can exist on a chess board are handheld with a very simple algorithm.

The major drawback is obviously the fact that it is very inefficient, as you need to generate all pseudo-legal moves for every possibilities. To cover our example board, this method requires 5 additional generations. With a more complex board it can easily reach 40.

However is it really an issue? If you go for a chess game with two human players, this solution is working well enough, so I think it is worthy to be mentioned. It will indeed be unusable with AI players that depends on the best performances possible for the move generation.

Legal moves by analysis

The logic behind BarbChessEngine legal move generation is represented with the following sequence:


The first thing we want to do is to identify all squares attacked by opponent pieces. These are all the squares where, in any case, the king should be or should move to.


Computing the previous attacking map, we are able to determine what are the threats for the king, the pieces attacking him directly. If at least one threat exists, then we are in a check situation.

There are two type of threats: direct threats are pawns, knights and the king, and are easy to solve, because the next move won't change their respective capacities to threat the king. Pawns for example will always attack the same squares, whatever the movement done by its opponent.


Line threats, so the rooks, bishops and the queen, are another story, as a chosen move can change the squares attacked by these threats. Pseudo-legal moves of a pinned piece can expose the king to threats that yet didn't put the king in check.


Line threats have to be considered, even if they don't directly have the king in target. I won't go through the process on how to recognize if a piece is a line threat or not, but what we need to know for every line threats is:

  • The position of the threat
  • If the threat is directly attacking the king or not
  • A set of squares that are where the player could place a piece in order to block the threat. Or on the contrary, where the player should not move a piece out of.
In our previous example:
- h5 is a line threat.
- h5 is not attacking the king directly.
- Options to block the threat are b5, b5, d5, e5, f5, g5.

Generate friendly pseudo-legal moves

This is the exact same step as we've done already, so I'll pass on this one.

Prune illegal moves

Removing illegal moves from the pseudo-legals is the last step. It is easily done by using these rules:

  • Remove all king moves that ends in an attacked square.
  • Remove all moves that start from the blocking squares of a line threat which do not end on blocking squares of the same line threat (or the line threat itself). In short, a pinned piece can move only on the attacking line of the piece which pinned it, or eat it.
  • If the king is in check, then there is at least one attacker. Then a move is legal if:
    • it finishes on the threat.
    • it finishes on a blocking option if the attacker is a line threat (except if it's the king).
    • the king is moving out of attacking squares.
  • Finally, if the king is in check, but there is more than one attacker, then the only viable moves are king moves leaving attacking squares.

Plus: Castles are not available if the king is in check!

The _en-passant_ rule troubles a bit these conditions: for example you can eat a piece that was blocking the line of a line threat without blocking it, which is pretty unique. So a bit of additional work is needed on this subject.

And this is it, we have our legal moves!

Testing

To assess the robustness of the legal move generation, the Chess Programming Wiki proposes a performance test (Perft) which count all legal moves walking in the generation tree, and comparing it to predetermined values.

For example, starting from the initial board, white has 20 legal moves (8 pawns that move 1 or 2 squares forward and 4 knight moves). If we make one of these moves, we can count how many moves black has, which is 20 as well. Doing this for every white moves, the sum of possibilities from the initial board, after a exploration depth of 2, is 400. Obviously you can go way deeper:


DepthNodes
120
2400
38,902
4197,281
54,865,609
6119,060,324

These precomputed results also exist for different positions, and are very valuable to debug the move generation.

For now, BarbChessEngine don't go that far in depth for every tests, which means there is still room for improvement. There are still some very special cases that are very hard to see and catch, and I believe this engine is already capable enough to go to the next step, and develop an AI.

BarbAI

As I said earlier, programming the engine was already a task, so for now, I've settled with a simple AI: I'm pretty sure though that it could be challenging enough for beginners!

BarbAI is a MiniMax implementation with Alpha-Beta pruning. In short, it explores all possible moves, gives them a evaluation based on a evaluation function, and takes the move that minimizes the losses for the AI (supposing that the player will always take the better play). The two important points are:

  • The evaluation function that rates a board. This first version is very simple, it counts the number of pieces for each player (each type of piece having its own value - a queen is 9x better than a pawn) and returns the difference. This means this version has no understanding of positioning in the board, the only thing that matters in the end are capture moves.
  • The depth of the MiniMax algorithm is crucial as well. It needs to be able to compute many moves in advance, otherwise it is very easy to beat. The ability to go deeper in a reasonable amount of time is directly linked to the operation that will be called the most in the move exploration: the legal move generation - that's why it is that important to have it quick!

Conclusion

Programming a chess engine revealed itself to be harder than I thought, but it was a very interesting challenge. There will be a part 2 focusing on a smart AI, way harder to beat than this first version, but I'll have to leave it to the future. I hope you found it interesting though!

References

- Sebastian Lague's Chess Programming video, clearly the major reference.

- Chess Programming Wiki that contains a tremendous amount of useful info.

- Wikipedia on MiniMax.

Get BarbChess

Leave a comment

Log in with itch.io to leave a comment.