chess:programming:magic_bitboards
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
chess:programming:magic_bitboards [2021/10/27 22:12] – peter | chess:programming:magic_bitboards [2021/10/30 13:12] (current) – peter | ||
---|---|---|---|
Line 3: | Line 3: | ||
Magic BitBoards are an efficient way to take a position and obtain the legal moves for a sliding piece. | Magic BitBoards are an efficient way to take a position and obtain the legal moves for a sliding piece. | ||
- | ---- | + | The occupancy |
- | + | ||
- | ===== Magic BitBoard Principle ===== | + | |
- | + | ||
- | The basic principle is: | + | |
- | + | ||
- | - Isolate the squares that a given piece may reach from a given position, without taking board occupancy | + | |
- | - Perform | + | |
- | - Multiply the result by a magic value M and shift the result to the right by a magic amount S. This gives us I (for Index). | + | |
- | - Use I as an index in a lookup table to retrieve the bitmask of squares that can actually be reached with this configuration. | + | |
- | + | ||
- | To sum it up: | + | |
- | + | ||
- | < | + | |
- | I = ((T & O) * M) >> S | + | |
- | </ | + | |
- | + | ||
- | <WRAP info> | + | |
- | **NOTE: | + | |
- | + | ||
- | * Reachable_squares = lookup[I] | + | |
- | + | ||
- | * T, M, S and lookup are all pre-computed and depend on the position of the piece (P = 0 ... 63). | + | |
- | * So, a more accurate formula would be: < | + | |
- | I = ((T[P] & O) * M[P]) >> S[P] | + | |
- | </ | + | |
- | * reachable_squares = lookup[P][I] | + | |
- | + | ||
- | The purpose of step #3 is to transform the 64-bit value T & O into a much smaller one, so that a table of a reasonable size can be used. | + | |
- | + | ||
- | * What we get by computing **< | + | |
- | + | ||
- | * The ' | + | |
- | * This is a **Perfect Hash Function** problem. | + | |
- | * However, no perfect hashing has been found for magic bitboards so far, which means that the lookup tables in use typically contain many unused ' | + | |
- | * Most of the time, they are built by running an extensive brute-force search. | + | |
- | + | ||
- | </ | + | |
---- | ---- | ||
- | ===== Example ===== | + | [[Chess: |
- | The Rook can move any amount of positions horizontally but cannot jump over the King. | + | [[Chess: |
- | < | ||
- | ┌───┬───┬───┬───┬───┬───┬───┬───┐ | ||
- | │ | ||
- | └───┴───┴───┴───┴───┴───┴───┴───┘ | ||
- | 7 | ||
- | </ | ||
- | |||
- | How to verify that the following is an illegal move, because the Rook cannot jump over the King? | ||
- | |||
- | < | ||
- | ┌───┬───┬───┬───┬───┬───┬───┬───┐ | ||
- | │ | ||
- | └───┴───┴───┴───┴───┴───┴───┴───┘ | ||
- | </ | ||
- | |||
- | |||
- | <WRAP info> | ||
- | **NOTE: | ||
- | |||
- | * Bitwise operations and pre-computed lookup tables will be needed. | ||
- | |||
- | Here, the position of the piece is in [0 ... 7] and the occupancy bitmask is in [0x00 ... 0xFF] (as it is 8-bit wide). | ||
- | |||
- | * Therefore, it is entirely feasible to build a direct lookup table based on the position and the current board without applying the **magic** part. | ||
- | |||
- | * This would be: < | ||
- | reachable_squares = lookup[P][board] | ||
- | </ | ||
- | |||
- | * This will result in a lookup table containing: < | ||
- | 8 * 2^8 = 2048 entries | ||
- | </ | ||
- | |||
- | * Obviously we cannot do that for chess, as it would contain: < | ||
- | 64 * 2^64 = 1, | ||
- | </ | ||
- | |||
- | * Hence the need for the magic multiply and shift operations to store the data in a more compact manner. | ||
- | |||
- | </ | ||
---- | ---- | ||
- | ===== Build a Lookup Table ===== | + | ===== References |
- | + | ||
- | <code cpp> | + | |
- | int xPos = 5; // Position of the ' | + | |
- | UINT64 board = 1 << xPos, // Initial board. | + | |
- | UINT lookup[]; | + | |
- | + | ||
- | void buildLookup() | + | |
- | { | + | |
- | int i, pos, msk; | + | |
- | + | ||
- | // Iterate on all possible positions. | + | |
- | for(pos=0; pos<8; pos++) | + | |
- | { | + | |
- | // Iterate on all possible occupancy masks. | + | |
- | for(lookup[pos] = [], msk=0; msk< | + | |
- | { | + | |
- | // Initialize to zero. | + | |
- | lookup[pos][msk] = 0; | + | |
- | + | ||
- | // Compute valid moves to the left. | + | |
- | for(i=pos+1; | + | |
- | { | + | |
- | lookup[pos][msk] |= 1 << i; | + | |
- | } | + | |
- | + | ||
- | // Compute valid moves to the right. | + | |
- | for(i=pos-1; | + | |
- | { | + | |
- | lookup[pos][msk] |= 1 << i; | + | |
- | } | + | |
- | } | + | |
- | } | + | |
- | } | + | |
- | + | ||
- | + | ||
- | bool isReachable(int pos) | + | |
- | { | + | |
- | // TODO. | + | |
- | + | ||
- | // Get valid target squares from the lookup table. | + | |
- | int target = lookup[xPos][board]; | + | |
- | + | ||
- | // return (!target & (1 << pos)); | + | |
- | + | ||
- | int n =7-pos; | + | |
- | + | ||
- | // Check if pos is valid and reachable. | + | |
- | if (n != xPos && (board ^= 1 << n)) | + | |
- | return true | + | |
- | else | + | |
- | return false; | + | |
- | + | ||
- | // Reachable = (target & (1 << n)). | + | |
- | //if ((board & (1 << n) ? ' | + | |
- | } | + | |
- | </ | + | |
- | <WRAP info> | + | https:// |
- | **NOTE:** The table is storing all possible positions of the Rook piece and each possible board (from 00000000 to 11111111). | + | |
- | </WRAP> | + | |
+ | https:// | ||
chess/programming/magic_bitboards.1635372726.txt.gz · Last modified: 2021/10/27 22:12 by peter