|
|
/* |
|
|
|
|
|
LittleRookChess.ino |
|
|
|
|
|
A Simple Chess Engine (ported from U8glib) |
|
|
|
|
|
Universal 8bit Graphics Library (https://github.com/olikraus/u8g2/) |
|
|
|
|
|
Copyright (c) 2016, olikraus@gmail.com |
|
|
All rights reserved. |
|
|
|
|
|
Redistribution and use in source and binary forms, with or without modification, |
|
|
are permitted provided that the following conditions are met: |
|
|
|
|
|
* Redistributions of source code must retain the above copyright notice, this list |
|
|
of conditions and the following disclaimer. |
|
|
|
|
|
* Redistributions in binary form must reproduce the above copyright notice, this |
|
|
list of conditions and the following disclaimer in the documentation and/or other |
|
|
materials provided with the distribution. |
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND |
|
|
CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, |
|
|
INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF |
|
|
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
|
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR |
|
|
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
|
|
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT |
|
|
NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
|
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER |
|
|
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, |
|
|
STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
|
|
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF |
|
|
ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
|
|
|
|
|
Current Rule Limitation |
|
|
- no minor promotion, only "Queening" of the pawn |
|
|
- threefold repetition is not detected (same board situation appears three times) |
|
|
Note: Could be implemented, but requires tracking of the complete game |
|
|
- Fifty-move rule is not checked (no pawn move, no capture within last 50 moves) |
|
|
|
|
|
Words |
|
|
Ply a half move |
|
|
|
|
|
General Links |
|
|
http://chessprogramming.wikispaces.com/ |
|
|
|
|
|
Arduino specific |
|
|
http://www.arduino.cc/cgi-bin/yabb2/YaBB.pl?num=1260055596 |
|
|
|
|
|
Prefixes |
|
|
chess_ Generic Chess Application Interface |
|
|
ce_ Chess engine, used internally, these function should not be called directly |
|
|
cu_ Chess utility function |
|
|
stack_ Internal function for stack handling |
|
|
|
|
|
Issues |
|
|
10.01.2011 |
|
|
- castling to the right does not move the rook |
|
|
--> done |
|
|
- castling to the left: King can only move two squares |
|
|
--> done |
|
|
|
|
|
11.01.2011 |
|
|
Next Steps: |
|
|
- replace stack_NextCurrentPos with cu_NextPos, cleanup code according to the loop variable |
|
|
--> done |
|
|
- Castling: Need to check for fields under attack |
|
|
--> done |
|
|
|
|
|
- Check for WIN / LOOSE situation, perhaps call ce_Eval() once on the top-level board setup |
|
|
just after the real move |
|
|
- cleanup cu_Move |
|
|
--> almost done |
|
|
- add some heuristics to the eval procedure |
|
|
- add right side menu |
|
|
--> done |
|
|
- clean up chess_ManualMove |
|
|
--> done |
|
|
- finish menu (consider is_game_end, undo move) |
|
|
- end condition: if KING is under attack and if KING can not move to a field which is under attack... |
|
|
then the game is lost. What will be returned by the Eval procedure? is it -INF? |
|
|
--> finished |
|
|
|
|
|
- reduce the use of variable color, all should be reduced to board_orientation and ply&1 |
|
|
|
|
|
- chess_GetNextMarked shoud make use of cu_NextPos |
|
|
--> done |
|
|
- chess_ManualMove: again cleanup, solve draw issue (KING is not in check and no legal moves are available) |
|
|
--> done |
|
|
22.01.2011 |
|
|
- simplify eval_t ce_Eval(void) |
|
|
- position eval does not work, still moves side pawn :-( |
|
|
maybe because all pieces are considered |
|
|
--> done |
|
|
|
|
|
17. June 2016 |
|
|
U8g2/Arduboy Port |
|
|
*/ |
|
|
|
|
|
#include <Arduino.h> |
|
|
#include <U8g2lib.h> |
|
|
|
|
|
#ifdef U8X8_HAVE_HW_SPI |
|
|
#include <SPI.h> |
|
|
#endif |
|
|
#ifdef U8X8_HAVE_HW_I2C |
|
|
#include <Wire.h> |
|
|
#endif |
|
|
|
|
|
|
|
|
/* |
|
|
U8glib Example Overview: |
|
|
Frame Buffer Examples: clearBuffer/sendBuffer. Fast, but may not work with all Arduino boards because of RAM consumption |
|
|
Page Buffer Examples: firstPage/nextPage. Less RAM usage, should work with all Arduino boards. |
|
|
U8x8 Text Only Example: No RAM usage, direct communication with display controller. No graphics, 8x8 Text only. |
|
|
|
|
|
This is a page buffer example. |
|
|
*/ |
|
|
|
|
|
void chess_Init(u8g2_t *u8g, uint8_t body_color); |
|
|
|
|
|
#define ARDUBOY |
|
|
//#define PI_SHIELD |
|
|
|
|
|
#ifdef ARDUBOY |
|
|
/*=== ARDUBOY Production, Kickstarter Edition ===*/ |
|
|
U8G2_SSD1306_128X64_NONAME_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 12, /* dc=*/ 4, /* reset=*/ 6); // Arduboy (Production, Kickstarter Edition) |
|
|
void setup(void) { |
|
|
//u8g2.begin(/*Select=*/ A0, /*Right/Next=*/ 5, /*Left/Prev=*/ 9, /*Up=*/ 8, /*Down=*/ 10, /*Home/Cancel=*/ A1); // Arduboy DevKit |
|
|
u8g2.begin(/*Select=*/ 7, /*Right/Next=*/ A1, /*Left/Prev=*/ A2, /*Up=*/ A0, /*Down=*/ A3, /*Home/Cancel=*/ 8); // Arduboy 10 (Production) |
|
|
chess_Init(u8g2.getU8g2(), 1); /* assuming Arduboy OLED here, so make the body_color be 1 for the white OLED pixel */ |
|
|
} |
|
|
#endif |
|
|
|
|
|
#ifdef PI_SHIELD |
|
|
/*=== Pax Instruments Shield ===*/ |
|
|
U8G2_ST7567_PI_132X64_1_4W_HW_SPI u8g2(U8G2_R2, /* cs=*/ 7, /* dc=*/ 9, /* reset=*/ 8); // Pax Instruments Shield, LCD_BL=6 |
|
|
void setup(void) { |
|
|
u8g2.begin(/*Select=*/ 4, /*Right/Next=*/ 5, /*Left/Prev=*/ A1, /*Up=*/ U8X8_PIN_NONE, /*Down=*/ U8X8_PIN_NONE, /*Home/Cancel=*/ A2); // Pax Instrument Shield |
|
|
|
|
|
/* U8g2 Project: Pax Instruments Shield: Enable Backlight */ |
|
|
pinMode(6, OUTPUT); |
|
|
digitalWrite(6, 0); |
|
|
|
|
|
chess_Init(u8g2.getU8g2(), 0); /* assuming Arduboy OLED here, so make the body_color be 1 for the white OLED pixel */ |
|
|
} |
|
|
#endif |
|
|
|
|
|
/* Ignore PROGMEM for now */ |
|
|
#define CHESS_PROGMEM |
|
|
#define chess_pgm_read(p) (*(p)) |
|
|
#define CHESS_PSTR(s) (s) |
|
|
#define u8g2_DrawStrP u8g2_DrawStr |
|
|
|
|
|
/* backward compatibility */ |
|
|
#define u8g2_SetDefaultBackgroundColor(u8g2) \ |
|
|
u8g2_SetDrawColor(u8g2, 0) |
|
|
#define u8g2_SetDefaultForegroundColor(u8g2) \ |
|
|
u8g2_SetDrawColor(u8g2, 1) |
|
|
|
|
|
|
|
|
/* menu button definitions: mapped to the u8g2 menu events */ |
|
|
|
|
|
#define CHESS_KEY_NONE 0 |
|
|
#define CHESS_KEY_NEXT U8X8_MSG_GPIO_MENU_NEXT |
|
|
#define CHESS_KEY_PREV U8X8_MSG_GPIO_MENU_PREV |
|
|
#define CHESS_KEY_SELECT U8X8_MSG_GPIO_MENU_SELECT |
|
|
#define CHESS_KEY_BACK U8X8_MSG_GPIO_MENU_HOME |
|
|
|
|
|
|
|
|
/* |
|
|
SAN identifies each piece by a single upper case letter. The standard English |
|
|
values: pawn = "P", knight = "N", bishop = "B", rook = "R", queen = "Q", and |
|
|
king = "K". |
|
|
*/ |
|
|
|
|
|
/* numbers for the various pieces */ |
|
|
#define PIECE_NONE 0 |
|
|
#define PIECE_PAWN 1 |
|
|
#define PIECE_KNIGHT 2 |
|
|
#define PIECE_BISHOP 3 |
|
|
#define PIECE_ROOK 4 |
|
|
#define PIECE_QUEEN 5 |
|
|
#define PIECE_KING 6 |
|
|
|
|
|
/* color definitions */ |
|
|
#define COLOR_WHITE 0 |
|
|
#define COLOR_BLACK 1 |
|
|
|
|
|
/* a mask, which includes COLOR and PIECE number */ |
|
|
#define COLOR_PIECE_MASK 0x01f |
|
|
|
|
|
#define CP_MARK_MASK 0x20 |
|
|
|
|
|
#define ILLEGAL_POSITION 255 |
|
|
|
|
|
/* This is the build in upper limit of the search stack */ |
|
|
/* This value defines the amount of memory allocated for the search stack */ |
|
|
/* The search depth of this chess engine can never exceed this value */ |
|
|
#define STACK_MAX_SIZE 5 |
|
|
|
|
|
/* chess half move stack: twice the number of undo's, a user can do */ |
|
|
#define CHM_USER_SIZE 6 |
|
|
|
|
|
/* the CHM_LIST_SIZE must be larger than the maximum search depth */ |
|
|
/* the overall size of ste half move stack */ |
|
|
#define CHM_LIST_SIZE (STACK_MAX_SIZE+CHM_USER_SIZE+2) |
|
|
|
|
|
typedef int16_t eval_t; /* a variable type to store results from the evaluation */ |
|
|
//#define EVAL_T_LOST -32768 |
|
|
#define EVAL_T_MIN -32767 |
|
|
#define EVAL_T_MAX 32767 |
|
|
//#define EVAL_T_WIN 32767 |
|
|
|
|
|
/* for maintainance of our own stack: this is the definition of one element on the stack */ |
|
|
struct _stack_element_struct |
|
|
{ |
|
|
/* the current source position which is investigated */ |
|
|
uint8_t current_pos; |
|
|
uint8_t current_cp; |
|
|
uint8_t current_color; /* COLOR_WHITE or COLOR_BLACK: must be predefines */ |
|
|
|
|
|
/* the move which belongs to that value, both values are game positions */ |
|
|
uint8_t best_from_pos; |
|
|
uint8_t best_to_pos; |
|
|
/* the best value, which has been dicovered so far */ |
|
|
eval_t best_eval; |
|
|
}; |
|
|
typedef struct _stack_element_struct stack_element_t; |
|
|
typedef struct _stack_element_struct *stack_element_p; |
|
|
|
|
|
/* chess half move history */ |
|
|
struct _chm_struct |
|
|
{ |
|
|
uint8_t main_cp; /* the main piece, which is moved */ |
|
|
uint8_t main_src; /* the source position of the main piece */ |
|
|
uint8_t main_dest; /* the destination of the main piece */ |
|
|
|
|
|
uint8_t other_cp; /* another piece: the captured one, the ROOK in case of castling or PIECE_NONE */ |
|
|
uint8_t other_src; /* the delete position of other_cp. Often identical to main_dest except for e.p. and castling */ |
|
|
uint8_t other_dest; /* only used for castling: ROOK destination pos */ |
|
|
|
|
|
/* the position of the last pawn, which did a double move forward */ |
|
|
/* this is required to check en passant conditions */ |
|
|
/* this array can be indexed by the color of the current player */ |
|
|
/* this is the condition BEFORE the move was done */ |
|
|
uint8_t pawn_dbl_move[2]; |
|
|
|
|
|
/* flags for the movement of rook and king; required for castling */ |
|
|
/* a 1 means: castling is (still) possible */ |
|
|
/* a 0 means: castling not possible */ |
|
|
/* bit 0 left side white */ |
|
|
/* bit 1 right side white */ |
|
|
/* bit 2 left side black */ |
|
|
/* bit 3 right side black */ |
|
|
/* this is the condition BEFORE the move was done */ |
|
|
uint8_t castling_possible; |
|
|
}; |
|
|
|
|
|
typedef struct _chm_struct chm_t; |
|
|
typedef struct _chm_struct *chm_p; |
|
|
|
|
|
/* little rook chess, main structure */ |
|
|
struct _lrc_struct |
|
|
{ |
|
|
/* half-move (ply) counter: Counts the number of half-moves so far. Starts with 0 */ |
|
|
/* the lowest bit is used to derive the color of the current player */ |
|
|
/* will be set to zero in chess_SetupBoard() */ |
|
|
uint8_t ply_count; |
|
|
|
|
|
/* the half move stack position counter, counts the number of elements in chm_list */ |
|
|
uint8_t chm_pos; |
|
|
|
|
|
/* each element contains a colored piece, empty fields have value 0 */ |
|
|
/* the field with index 0 is black (lower left) */ |
|
|
uint8_t board[64]; |
|
|
/* the position of the last pawn, which did a double move forward */ |
|
|
/* this is required to check en passant conditions */ |
|
|
/* this array can be indexed by the color of the current player */ |
|
|
uint8_t pawn_dbl_move[2]; |
|
|
|
|
|
/* flags for the movement of rook and king; required for castling */ |
|
|
/* a 1 means: castling is (still) possible */ |
|
|
/* a 0 means: castling not possible */ |
|
|
/* bit 0 left side white */ |
|
|
/* bit 1 right side white */ |
|
|
/* bit 2 left side black */ |
|
|
/* bit 3 right side black */ |
|
|
uint8_t castling_possible; |
|
|
|
|
|
/* board orientation */ |
|
|
/* 0: white is below COLOR_WHITE */ |
|
|
/* 1: black is below COLOR_BLACK */ |
|
|
/* bascially, this can be used as a color */ |
|
|
uint8_t orientation; |
|
|
|
|
|
/* exchange colors of the pieces */ |
|
|
/* 0: white has an empty body, use this for bright background color */ |
|
|
/* 1: black has an empty body, use this for dark backround color */ |
|
|
uint8_t strike_out_color; |
|
|
|
|
|
/* 0, when the game is ongoing */ |
|
|
/* 1, when the game is stopped (lost or draw) */ |
|
|
uint8_t is_game_end; |
|
|
/* the color of the side which lost the game */ |
|
|
/* this value is only valid, when is_game_end is not 0 */ |
|
|
/* values 0 and 1 represent WHITE and BLACK, 2 means a draw */ |
|
|
uint8_t lost_side_color; |
|
|
|
|
|
|
|
|
|
|
|
/* checks are executed in ce_LoopRecur */ |
|
|
/* these checks will put some marks on the board */ |
|
|
/* this will be used by the interface to find out */ |
|
|
/* legal moves */ |
|
|
uint8_t check_src_pos; |
|
|
uint8_t check_mode; /* CHECK_MODE_NONE, CHECK_MODE_MOVEABLE, CHECK_MODE_TARGET_MOVE */ |
|
|
|
|
|
|
|
|
/* count of the attacking pieces, indexed by color */ |
|
|
uint8_t find_piece_cnt[2]; |
|
|
|
|
|
/* sum of the attacking pieces, indexed by color */ |
|
|
uint8_t find_piece_weight[2]; |
|
|
|
|
|
/* points to the current element of the search stack */ |
|
|
/* this stack is NEVER empty. The value 0 points to the first element of the stack */ |
|
|
/* actually "curr_depth" represent half-moves (plies) */ |
|
|
uint8_t curr_depth; |
|
|
uint8_t max_depth; |
|
|
stack_element_p curr_element; |
|
|
|
|
|
/* allocated memory for the search stack */ |
|
|
stack_element_t stack_memory[STACK_MAX_SIZE]; |
|
|
|
|
|
/* the half move stack, used for move undo and depth search, size is stored in chm_pos */ |
|
|
chm_t chm_list[CHM_LIST_SIZE]; |
|
|
}; |
|
|
typedef struct _lrc_struct lrc_t; |
|
|
|
|
|
#define CHECK_MODE_NONE 0 |
|
|
#define CHECK_MODE_MOVEABLE 1 |
|
|
#define CHECK_MODE_TARGET_MOVE 2 |
|
|
|
|
|
|
|
|
|
|
|
/*==============================================================*/ |
|
|
/* global variables */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
lrc_t lrc_obj; |
|
|
u8g2_t *lrc_u8g; /* pointer to the C object of u8g2 lib, this is used by the chess engine */ |
|
|
|
|
|
|
|
|
/*==============================================================*/ |
|
|
/* forward declarations */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
/* |
|
|
apply no inline to some of the functions: |
|
|
avr-gcc very often inlines functions, however not inline saves a lot of program memory! |
|
|
On the other hand there are some really short procedures which should be inlined (like cp_GetColor) |
|
|
These procedures are marked static to prevent the generation of the expanded procedure, which |
|
|
also saves space. |
|
|
*/ |
|
|
|
|
|
uint8_t stack_Push(uint8_t color) U8G2_NOINLINE; |
|
|
void stack_Pop(void) U8G2_NOINLINE; |
|
|
void stack_InitCurrElement(void) U8G2_NOINLINE; |
|
|
void stack_Init(uint8_t max) U8G2_NOINLINE; |
|
|
void stack_SetMove(eval_t val, uint8_t to_pos) U8G2_NOINLINE; |
|
|
uint8_t cu_NextPos(uint8_t pos) U8G2_NOINLINE; |
|
|
static uint8_t cu_gpos2bpos(uint8_t gpos); |
|
|
static uint8_t cp_Construct(uint8_t color, uint8_t piece); |
|
|
static uint8_t cp_GetPiece(uint8_t cp); |
|
|
static uint8_t cp_GetColor(uint8_t cp); |
|
|
uint8_t cp_GetFromBoard(uint8_t pos) U8G2_NOINLINE; |
|
|
void cp_SetOnBoard(uint8_t pos, uint8_t cp) U8G2_NOINLINE; |
|
|
|
|
|
void cu_ClearBoard(void) U8G2_NOINLINE; |
|
|
void chess_SetupBoard(void) U8G2_NOINLINE; |
|
|
eval_t ce_Eval(void); |
|
|
|
|
|
void cu_ClearMoveHistory(void) U8G2_NOINLINE; |
|
|
void cu_ReduceHistoryByFullMove(void) U8G2_NOINLINE; |
|
|
void cu_UndoHalfMove(void) U8G2_NOINLINE; |
|
|
chm_p cu_PushHalfMove(void) U8G2_NOINLINE; |
|
|
|
|
|
|
|
|
void ce_CalculatePositionWeight(uint8_t pos); |
|
|
uint8_t ce_GetPositionAttackWeight(uint8_t pos, uint8_t color); |
|
|
|
|
|
void chess_Thinking(void); |
|
|
void ce_LoopPieces(void); |
|
|
|
|
|
|
|
|
/*==============================================================*/ |
|
|
/* search stack */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
/* get current element from stack */ |
|
|
struct _stack_element_struct *stack_GetCurrElement(void) |
|
|
{ |
|
|
return lrc_obj.curr_element; |
|
|
} |
|
|
|
|
|
uint8_t stack_Push(uint8_t color) |
|
|
{ |
|
|
if ( lrc_obj.curr_depth == lrc_obj.max_depth ) |
|
|
return 0; |
|
|
lrc_obj.curr_depth++; |
|
|
lrc_obj.curr_element = lrc_obj.stack_memory+lrc_obj.curr_depth; |
|
|
|
|
|
/* change view for the evaluation */ |
|
|
color ^= 1; |
|
|
stack_GetCurrElement()->current_color = color; |
|
|
|
|
|
return 1; |
|
|
} |
|
|
|
|
|
void stack_Pop(void) |
|
|
{ |
|
|
lrc_obj.curr_depth--; |
|
|
lrc_obj.curr_element = lrc_obj.stack_memory+lrc_obj.curr_depth; |
|
|
} |
|
|
|
|
|
/* reset the current element on the stack */ |
|
|
void stack_InitCurrElement(void) |
|
|
{ |
|
|
stack_element_p e = stack_GetCurrElement(); |
|
|
e->best_eval = EVAL_T_MIN; |
|
|
e->best_from_pos = ILLEGAL_POSITION; |
|
|
e->best_to_pos = ILLEGAL_POSITION; |
|
|
} |
|
|
|
|
|
/* resets the search stack (and the check mode) */ |
|
|
void stack_Init(uint8_t max) |
|
|
{ |
|
|
lrc_obj.curr_depth = 0; |
|
|
lrc_obj.curr_element = lrc_obj.stack_memory; |
|
|
lrc_obj.max_depth = max; |
|
|
lrc_obj.check_mode = CHECK_MODE_NONE; |
|
|
stack_InitCurrElement(); |
|
|
stack_GetCurrElement()->current_color = lrc_obj.ply_count; |
|
|
stack_GetCurrElement()->current_color &= 1; |
|
|
} |
|
|
|
|
|
/* assign evaluation value and store the move, if this is the best move */ |
|
|
/* assumes, that current_pos contains the source position */ |
|
|
void stack_SetMove(eval_t val, uint8_t to_pos) |
|
|
{ |
|
|
stack_element_p e = stack_GetCurrElement(); |
|
|
if ( e->best_eval < val ) |
|
|
{ |
|
|
e->best_eval = val; |
|
|
e->best_from_pos = e->current_pos; |
|
|
e->best_to_pos = to_pos; |
|
|
} |
|
|
} |
|
|
|
|
|
/* |
|
|
calculate next position on a 0x88 board |
|
|
loop is constructed in this way: |
|
|
i = 0; |
|
|
do |
|
|
{ |
|
|
... |
|
|
i = cu_NextPos(i); |
|
|
} while( i != 0 ); |
|
|
|
|
|
next pos might be started with an illegal position like 255 |
|
|
*/ |
|
|
uint8_t cu_NextPos(uint8_t pos) |
|
|
{ |
|
|
/* calculate next gpos */ |
|
|
pos++; |
|
|
if ( ( pos & 0x08 ) != 0 ) |
|
|
{ |
|
|
pos+= 0x10; |
|
|
pos&= 0xf0; |
|
|
} |
|
|
if ( ( pos & 0x80 ) != 0 ) |
|
|
pos = 0; |
|
|
return pos; |
|
|
} |
|
|
|
|
|
uint8_t cu_PrevPos(uint8_t pos) |
|
|
{ |
|
|
/* calculate prev gpos */ |
|
|
pos--; |
|
|
if ( ( pos & 0x80 ) != 0 ) |
|
|
pos = 0x077; |
|
|
else if ( ( pos & 0x08 ) != 0 ) |
|
|
{ |
|
|
pos &= 0xf0; |
|
|
pos |= 0x07; |
|
|
} |
|
|
return pos; |
|
|
} |
|
|
|
|
|
|
|
|
/*==============================================================*/ |
|
|
/* position transltion */ |
|
|
/*==============================================================*/ |
|
|
/* |
|
|
there are two positions |
|
|
1. game position (gpos): BCD encoded x-y values |
|
|
2. board position (bpos): a number between 0 and 63, only used to access the board. |
|
|
*/ |
|
|
/* |
|
|
gpos: game position value |
|
|
returns: board position |
|
|
note: does not do any checks |
|
|
*/ |
|
|
static uint8_t cu_gpos2bpos(uint8_t gpos) |
|
|
{ |
|
|
uint8_t bpos = gpos; |
|
|
bpos &= 0xf0; |
|
|
bpos >>= 1; |
|
|
gpos &= 0x0f; |
|
|
bpos |= gpos; |
|
|
return bpos; |
|
|
} |
|
|
|
|
|
#define gpos_IsIllegal(gpos) ((gpos) & 0x088) |
|
|
|
|
|
|
|
|
/*==============================================================*/ |
|
|
/* colored piece handling */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
#define cp_IsMarked(cp) ((cp) & CP_MARK_MASK) |
|
|
|
|
|
|
|
|
/* |
|
|
piece: one of PIECE_xxx |
|
|
color: COLOR_WHITE or COLOR_BLACK |
|
|
|
|
|
returns: A colored piece |
|
|
*/ |
|
|
static uint8_t cp_Construct(uint8_t color, uint8_t piece) |
|
|
{ |
|
|
color <<= 4; |
|
|
color |= piece; |
|
|
return color; |
|
|
} |
|
|
|
|
|
/* inline is better than a macro */ |
|
|
static uint8_t cp_GetPiece(uint8_t cp) |
|
|
{ |
|
|
cp &= 0x0f; |
|
|
return cp; |
|
|
} |
|
|
|
|
|
/* |
|
|
we could use a macro: |
|
|
#define cp_GetColor(cp) (((cp) >> 4)&1) |
|
|
however, inlined functions are sometimes much better |
|
|
*/ |
|
|
static uint8_t cp_GetColor(uint8_t cp) |
|
|
{ |
|
|
cp >>= 4; |
|
|
cp &= 1; |
|
|
return cp; |
|
|
} |
|
|
|
|
|
/* |
|
|
pos: game position |
|
|
returns the colored piece at the given position |
|
|
*/ |
|
|
uint8_t cp_GetFromBoard(uint8_t pos) |
|
|
{ |
|
|
return lrc_obj.board[cu_gpos2bpos(pos)]; |
|
|
} |
|
|
|
|
|
/* |
|
|
pos: game position |
|
|
cp: colored piece |
|
|
*/ |
|
|
void cp_SetOnBoard(uint8_t pos, uint8_t cp) |
|
|
{ |
|
|
/*printf("cp_SetOnBoard gpos:%02x cp:%02x\n", pos, cp);*/ |
|
|
lrc_obj.board[cu_gpos2bpos(pos)] = cp; |
|
|
} |
|
|
|
|
|
/*==============================================================*/ |
|
|
/* global board access */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
void cu_ClearBoard(void) |
|
|
{ |
|
|
uint8_t i; |
|
|
/* clear the board */ |
|
|
for( i = 0; i < 64; i++ ) |
|
|
lrc_obj.board[i] = PIECE_NONE; |
|
|
|
|
|
lrc_obj.ply_count = 0; |
|
|
lrc_obj.orientation = COLOR_WHITE; |
|
|
|
|
|
lrc_obj.pawn_dbl_move[0] = ILLEGAL_POSITION; |
|
|
lrc_obj.pawn_dbl_move[1] = ILLEGAL_POSITION; |
|
|
|
|
|
lrc_obj.castling_possible = 0x0f; |
|
|
|
|
|
lrc_obj.is_game_end = 0; |
|
|
lrc_obj.lost_side_color = 0; |
|
|
|
|
|
/* clear half move history */ |
|
|
cu_ClearMoveHistory(); |
|
|
|
|
|
} |
|
|
|
|
|
/* |
|
|
test setup |
|
|
white wins in one move |
|
|
*/ |
|
|
void chess_SetupBoardTest01(void) |
|
|
{ |
|
|
cu_ClearBoard(); |
|
|
lrc_obj.board[7+7*8] = cp_Construct(COLOR_BLACK, PIECE_KING); |
|
|
lrc_obj.board[7+5*8] = cp_Construct(COLOR_WHITE, PIECE_PAWN); |
|
|
lrc_obj.board[3] = cp_Construct(COLOR_WHITE, PIECE_KING); |
|
|
lrc_obj.board[0+7*8] = cp_Construct(COLOR_BLACK, PIECE_ROOK); |
|
|
lrc_obj.board[6] = cp_Construct(COLOR_WHITE, PIECE_QUEEN); |
|
|
} |
|
|
|
|
|
/* setup the global board */ |
|
|
void chess_SetupBoard(void) |
|
|
{ |
|
|
uint8_t i; |
|
|
register uint8_t bp, wp; |
|
|
|
|
|
/* clear the board */ |
|
|
cu_ClearBoard(); |
|
|
|
|
|
/* precronstruct pawns */ |
|
|
wp = cp_Construct(COLOR_WHITE, PIECE_PAWN); |
|
|
bp = cp_Construct(COLOR_BLACK, PIECE_PAWN); |
|
|
|
|
|
/* setup pawn */ |
|
|
for( i = 0; i < 8; i++ ) |
|
|
{ |
|
|
lrc_obj.board[i+8] = wp; |
|
|
lrc_obj.board[i+6*8] = bp; |
|
|
} |
|
|
|
|
|
/* assign remaining pieces */ |
|
|
|
|
|
lrc_obj.board[0] = cp_Construct(COLOR_WHITE, PIECE_ROOK); |
|
|
lrc_obj.board[1] = cp_Construct(COLOR_WHITE, PIECE_KNIGHT); |
|
|
lrc_obj.board[2] = cp_Construct(COLOR_WHITE, PIECE_BISHOP); |
|
|
lrc_obj.board[3] = cp_Construct(COLOR_WHITE, PIECE_QUEEN); |
|
|
lrc_obj.board[4] = cp_Construct(COLOR_WHITE, PIECE_KING); |
|
|
lrc_obj.board[5] = cp_Construct(COLOR_WHITE, PIECE_BISHOP); |
|
|
lrc_obj.board[6] = cp_Construct(COLOR_WHITE, PIECE_KNIGHT); |
|
|
lrc_obj.board[7] = cp_Construct(COLOR_WHITE, PIECE_ROOK); |
|
|
|
|
|
lrc_obj.board[0+7*8] = cp_Construct(COLOR_BLACK, PIECE_ROOK); |
|
|
lrc_obj.board[1+7*8] = cp_Construct(COLOR_BLACK, PIECE_KNIGHT); |
|
|
lrc_obj.board[2+7*8] = cp_Construct(COLOR_BLACK, PIECE_BISHOP); |
|
|
lrc_obj.board[3+7*8] = cp_Construct(COLOR_BLACK, PIECE_QUEEN); |
|
|
lrc_obj.board[4+7*8] = cp_Construct(COLOR_BLACK, PIECE_KING); |
|
|
lrc_obj.board[5+7*8] = cp_Construct(COLOR_BLACK, PIECE_BISHOP); |
|
|
lrc_obj.board[6+7*8] = cp_Construct(COLOR_BLACK, PIECE_KNIGHT); |
|
|
lrc_obj.board[7+7*8] = cp_Construct(COLOR_BLACK, PIECE_ROOK); |
|
|
|
|
|
//chess_SetupBoardTest01(); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
/*==============================================================*/ |
|
|
/* checks */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
/* |
|
|
checks if the position is somehow illegal |
|
|
*/ |
|
|
uint8_t cu_IsIllegalPosition(uint8_t pos, uint8_t my_color) |
|
|
{ |
|
|
uint8_t board_cp; |
|
|
/* check, if the position is offboard */ |
|
|
if ( gpos_IsIllegal(pos) != 0 ) |
|
|
return 1; |
|
|
/* get the piece from the board */ |
|
|
board_cp = cp_GetFromBoard(pos); |
|
|
/* check if hit our own pieces */ |
|
|
if ( board_cp != 0 ) |
|
|
if ( cp_GetColor(board_cp) == my_color ) |
|
|
return 1; |
|
|
/* all ok, we could go to this position */ |
|
|
return 0; |
|
|
} |
|
|
|
|
|
/*==============================================================*/ |
|
|
/* evaluation procedure */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
/* |
|
|
basic idea is to return a value between EVAL_T_MIN and EVAL_T_MAX |
|
|
*/ |
|
|
|
|
|
/* |
|
|
the weight table uses the PIECE number as index: |
|
|
#define PIECE_NONE 0 |
|
|
#define PIECE_PAWN 1 |
|
|
#define PIECE_KNIGHT 2 |
|
|
#define PIECE_BISHOP 3 |
|
|
#define PIECE_ROOK 4 |
|
|
#define PIECE_QUEEN 5 |
|
|
#define PIECE_KING 6 |
|
|
the king itself is not counted |
|
|
*/ |
|
|
uint8_t ce_piece_weight[] = { 0, 1, 3, 3, 5, 9, 0 }; |
|
|
uint8_t ce_pos_weight[] = { 0, 1, 1, 2, 2, 1, 1, 0}; |
|
|
/* |
|
|
evaluate the current situation on the global board |
|
|
*/ |
|
|
eval_t ce_Eval(void) |
|
|
{ |
|
|
uint8_t cp; |
|
|
uint8_t is_my_king_present = 0; |
|
|
uint8_t is_opposit_king_present = 0; |
|
|
eval_t material_my_color = 0; |
|
|
eval_t material_opposit_color = 0; |
|
|
eval_t position_my_color = 0; |
|
|
eval_t position_opposit_color = 0; |
|
|
eval_t result; |
|
|
uint8_t pos; |
|
|
|
|
|
pos = 0; |
|
|
do |
|
|
{ |
|
|
/* get colored piece from the board */ |
|
|
cp = cp_GetFromBoard(pos); |
|
|
|
|
|
if ( cp_GetPiece(cp) != PIECE_NONE ) |
|
|
{ |
|
|
if ( stack_GetCurrElement()->current_color == cp_GetColor(cp) ) |
|
|
{ |
|
|
/* this is our color */ |
|
|
/* check if we found our king */ |
|
|
if ( cp_GetPiece(cp) == PIECE_KING ) |
|
|
is_my_king_present = 1; |
|
|
material_my_color += ce_piece_weight[cp_GetPiece(cp)]; |
|
|
if ( cp_GetPiece(cp) == PIECE_PAWN || cp_GetPiece(cp) == PIECE_KNIGHT ) |
|
|
{ |
|
|
position_my_color += ce_pos_weight[pos&7]*ce_pos_weight[(pos>>4)&7]; |
|
|
} |
|
|
} |
|
|
else |
|
|
{ |
|
|
/* this is the opposit color */ |
|
|
if ( cp_GetPiece(cp) == PIECE_KING ) |
|
|
is_opposit_king_present = 1; |
|
|
material_opposit_color += ce_piece_weight[cp_GetPiece(cp)]; |
|
|
if ( cp_GetPiece(cp) == PIECE_PAWN || cp_GetPiece(cp) == PIECE_KNIGHT ) |
|
|
{ |
|
|
position_opposit_color += ce_pos_weight[pos&7]*ce_pos_weight[(pos>>4)&7]; |
|
|
} |
|
|
} |
|
|
} |
|
|
pos = cu_NextPos(pos); |
|
|
} while( pos != 0 ); |
|
|
|
|
|
|
|
|
/* decide if we lost or won the game */ |
|
|
if ( is_my_king_present == 0 ) |
|
|
return EVAL_T_MIN; /*_LOST*/ |
|
|
if ( is_opposit_king_present == 0 ) |
|
|
return EVAL_T_MAX; /*_WIN*/ |
|
|
|
|
|
/* here is the evaluation function */ |
|
|
|
|
|
result = material_my_color - material_opposit_color; |
|
|
result <<= 3; |
|
|
result += position_my_color - position_opposit_color; |
|
|
return result; |
|
|
} |
|
|
|
|
|
/*==============================================================*/ |
|
|
/* move backup and restore */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
|
|
|
/* this procedure must be called to keep the size as low as possible */ |
|
|
/* if the chm_list is large enough, it could hold the complete history */ |
|
|
/* but for an embedded controler... it is deleted for every engine search */ |
|
|
void cu_ClearMoveHistory(void) |
|
|
{ |
|
|
lrc_obj.chm_pos = 0; |
|
|
} |
|
|
|
|
|
void cu_ReduceHistoryByFullMove(void) |
|
|
{ |
|
|
uint8_t i; |
|
|
while( lrc_obj.chm_pos > CHM_USER_SIZE ) |
|
|
{ |
|
|
i = 0; |
|
|
for(;;) |
|
|
{ |
|
|
if ( i+2 >= lrc_obj.chm_pos ) |
|
|
break; |
|
|
lrc_obj.chm_list[i] = lrc_obj.chm_list[i+2]; |
|
|
i++; |
|
|
} |
|
|
lrc_obj.chm_pos -= 2; |
|
|
} |
|
|
} |
|
|
|
|
|
void cu_UndoHalfMove(void) |
|
|
{ |
|
|
chm_p chm; |
|
|
|
|
|
if ( lrc_obj.chm_pos == 0 ) |
|
|
return; |
|
|
|
|
|
lrc_obj.chm_pos--; |
|
|
|
|
|
chm = lrc_obj.chm_list+lrc_obj.chm_pos; |
|
|
|
|
|
lrc_obj.pawn_dbl_move[0] = chm->pawn_dbl_move[0]; |
|
|
lrc_obj.pawn_dbl_move[1] = chm->pawn_dbl_move[1]; |
|
|
lrc_obj.castling_possible = chm->castling_possible; |
|
|
|
|
|
cp_SetOnBoard(chm->main_src, chm->main_cp); |
|
|
cp_SetOnBoard(chm->main_dest, PIECE_NONE); |
|
|
|
|
|
if ( chm->other_src != ILLEGAL_POSITION ) |
|
|
cp_SetOnBoard(chm->other_src, chm->other_cp); |
|
|
if ( chm->other_dest != ILLEGAL_POSITION ) |
|
|
cp_SetOnBoard(chm->other_dest, PIECE_NONE); |
|
|
|
|
|
} |
|
|
|
|
|
/* |
|
|
assumes, that the following members of the returned chm structure are filled |
|
|
uint8_t main_cp; the main piece, which is moved |
|
|
uint8_t main_src; the source position of the main piece |
|
|
uint8_t main_dest; the destination of the main piece |
|
|
|
|
|
uint8_t other_cp; another piece: the captured one, the ROOK in case of castling or PIECE_NONE |
|
|
uint8_t other_src; the delete position of other_cp. Often identical to main_dest except for e.p. and castling |
|
|
uint8_t other_dest; only used for castling: ROOK destination pos |
|
|
|
|
|
*/ |
|
|
chm_p cu_PushHalfMove(void) |
|
|
{ |
|
|
chm_p chm; |
|
|
|
|
|
chm = lrc_obj.chm_list+lrc_obj.chm_pos; |
|
|
if ( lrc_obj.chm_pos < CHM_LIST_SIZE-1) |
|
|
lrc_obj.chm_pos++; |
|
|
|
|
|
chm->pawn_dbl_move[0] = lrc_obj.pawn_dbl_move[0]; |
|
|
chm->pawn_dbl_move[1] = lrc_obj.pawn_dbl_move[1]; |
|
|
chm->castling_possible = lrc_obj.castling_possible; |
|
|
return chm; |
|
|
} |
|
|
|
|
|
|
|
|
char chess_piece_to_char[] = "NBRQK"; |
|
|
|
|
|
/* |
|
|
simple moves on empty field: Ka1-b2 |
|
|
capture moves: Ka1xb2 |
|
|
castling: 0-0 or 0-0-0 |
|
|
*/ |
|
|
|
|
|
static void cu_add_pos(char *s, uint8_t pos) U8G2_NOINLINE; |
|
|
|
|
|
static void cu_add_pos(char *s, uint8_t pos) |
|
|
{ |
|
|
*s = pos; |
|
|
*s &= 15; |
|
|
*s += 'a'; |
|
|
s++; |
|
|
*s = pos; |
|
|
*s >>= 4; |
|
|
*s += '1'; |
|
|
} |
|
|
|
|
|
const char *cu_GetHalfMoveStr(uint8_t idx) |
|
|
{ |
|
|
chm_p chm; |
|
|
static char buf[7]; /*Ka1-b2*/ |
|
|
char *p = buf; |
|
|
chm = lrc_obj.chm_list+idx; |
|
|
|
|
|
if ( cp_GetPiece(chm->main_cp) != PIECE_NONE ) |
|
|
{ |
|
|
if ( cp_GetPiece(chm->main_cp) > PIECE_PAWN ) |
|
|
{ |
|
|
*p++ = chess_piece_to_char[cp_GetPiece(chm->main_cp)-2]; |
|
|
} |
|
|
cu_add_pos(p, chm->main_src); |
|
|
p+=2; |
|
|
if ( cp_GetPiece(chm->other_cp) == PIECE_NONE ) |
|
|
*p++ = '-'; |
|
|
else |
|
|
*p++ = 'x'; |
|
|
cu_add_pos(p, chm->main_dest); |
|
|
p+=2; |
|
|
} |
|
|
*p = '\0'; |
|
|
return buf; |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*==============================================================*/ |
|
|
/* move */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
/* |
|
|
Move a piece from source position to a destination on the board |
|
|
This function |
|
|
- does not perform any checking |
|
|
- however it processes "en passant" and casteling |
|
|
- backup the move and allow 1x undo |
|
|
|
|
|
2011-02-05: |
|
|
- fill pawn_dbl_move[] for double pawn moves |
|
|
--> done |
|
|
- Implement casteling |
|
|
--> done |
|
|
- en passant |
|
|
--> done |
|
|
- pawn conversion/promotion |
|
|
--> done |
|
|
- half-move backup |
|
|
--> done |
|
|
- cleanup everything, minimize variables |
|
|
--> done |
|
|
*/ |
|
|
|
|
|
void cu_Move(uint8_t src, uint8_t dest) |
|
|
{ |
|
|
/* start backup structure */ |
|
|
chm_p chm = cu_PushHalfMove(); |
|
|
|
|
|
/* these are the values from the board at the positions, provided as arguments to this function */ |
|
|
uint8_t cp_src, cp_dest; |
|
|
|
|
|
/* Maybe a second position is cleared and one additional location is set */ |
|
|
uint8_t clr_pos2; |
|
|
uint8_t set_pos2; |
|
|
uint8_t set_cp2; |
|
|
|
|
|
/* get values from board */ |
|
|
cp_src = cp_GetFromBoard(src); |
|
|
cp_dest = cp_GetFromBoard(dest); |
|
|
|
|
|
/* fill backup structure */ |
|
|
|
|
|
chm->main_cp = cp_src; |
|
|
chm->main_src = src; |
|
|
chm->main_dest = dest; |
|
|
|
|
|
chm->other_cp = cp_dest; /* prepace capture backup */ |
|
|
chm->other_src = dest; |
|
|
chm->other_dest = ILLEGAL_POSITION; |
|
|
|
|
|
/* setup results as far as possible with some suitable values */ |
|
|
|
|
|
clr_pos2 = ILLEGAL_POSITION; /* for en passant and castling, two positions might be cleared */ |
|
|
set_pos2 = ILLEGAL_POSITION; /* only used for castling */ |
|
|
set_cp2 = PIECE_NONE; /* ROOK for castling */ |
|
|
|
|
|
/* check for PAWN */ |
|
|
if ( cp_GetPiece(cp_src) == PIECE_PAWN ) |
|
|
{ |
|
|
|
|
|
/* double step: is the distance 2 rows */ |
|
|
if ( (src - dest == 32) || ( dest - src == 32 ) ) |
|
|
{ |
|
|
/* remember the destination position */ |
|
|
lrc_obj.pawn_dbl_move[cp_GetColor(cp_src)] = dest; |
|
|
} |
|
|
|
|
|
/* check if the PAWN is able to promote */ |
|
|
else if ( (dest>>4) == 0 || (dest>>4) == 7 ) |
|
|
{ |
|
|
/* do simple "queening" */ |
|
|
cp_src &= ~PIECE_PAWN; |
|
|
cp_src |= PIECE_QUEEN; |
|
|
} |
|
|
|
|
|
/* is it en passant capture? */ |
|
|
/* check for side move */ |
|
|
else if ( ((src + dest) & 1) != 0 ) |
|
|
{ |
|
|
/* check, if target field is empty */ |
|
|
if ( cp_GetPiece(cp_dest) == PIECE_NONE ) |
|
|
{ |
|
|
/* this is en passant */ |
|
|
/* no further checking required, because legal moves are assumed here */ |
|
|
/* however... the captured pawn position must be valid */ |
|
|
clr_pos2 = lrc_obj.pawn_dbl_move[cp_GetColor(cp_src) ^ 1]; |
|
|
chm->other_src = clr_pos2; |
|
|
chm->other_cp = cp_GetFromBoard(clr_pos2); |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
/* check for the KING */ |
|
|
else if ( cp_GetPiece(cp_src) == PIECE_KING ) |
|
|
{ |
|
|
/* disallow castling, if the KING has moved */ |
|
|
if ( cp_GetColor(cp_src) == COLOR_WHITE ) |
|
|
{ |
|
|
/* if white KING has moved, disallow castling for white */ |
|
|
lrc_obj.castling_possible &= 0x0c; |
|
|
} |
|
|
else |
|
|
{ |
|
|
/* if black KING has moved, disallow castling for black */ |
|
|
lrc_obj.castling_possible &= 0x03; |
|
|
} |
|
|
|
|
|
/* has it been castling to the left? */ |
|
|
if ( src - dest == 2 ) |
|
|
{ |
|
|
/* let the ROOK move to pos2 */ |
|
|
set_pos2 = src-1; |
|
|
set_cp2 = cp_GetFromBoard(src-4); |
|
|
|
|
|
/* the ROOK must be cleared from the original position */ |
|
|
clr_pos2 = src-4; |
|
|
|
|
|
chm->other_cp = set_cp2; |
|
|
chm->other_src = clr_pos2; |
|
|
chm->other_dest = set_pos2; |
|
|
} |
|
|
|
|
|
/* has it been castling to the right? */ |
|
|
else if ( dest - src == 2 ) |
|
|
{ |
|
|
/* let the ROOK move to pos2 */ |
|
|
set_pos2 = src+1; |
|
|
set_cp2 = cp_GetFromBoard(src+3); |
|
|
|
|
|
/* the ROOK must be cleared from the original position */ |
|
|
clr_pos2 = src+3; |
|
|
|
|
|
chm->other_cp = set_cp2; |
|
|
chm->other_src = clr_pos2; |
|
|
chm->other_dest = set_pos2; |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
/* check for the ROOK */ |
|
|
else if ( cp_GetPiece(cp_src) == PIECE_ROOK ) |
|
|
{ |
|
|
/* disallow white left castling */ |
|
|
if ( src == 0x00 ) |
|
|
lrc_obj.castling_possible &= ~0x01; |
|
|
/* disallow white right castling */ |
|
|
if ( src == 0x07 ) |
|
|
lrc_obj.castling_possible &= ~0x02; |
|
|
/* disallow black left castling */ |
|
|
if ( src == 0x70 ) |
|
|
lrc_obj.castling_possible &= ~0x04; |
|
|
/* disallow black right castling */ |
|
|
if ( src == 0x77 ) |
|
|
lrc_obj.castling_possible &= ~0x08; |
|
|
} |
|
|
|
|
|
|
|
|
/* apply new board situation */ |
|
|
|
|
|
cp_SetOnBoard(dest, cp_src); |
|
|
|
|
|
if ( set_pos2 != ILLEGAL_POSITION ) |
|
|
cp_SetOnBoard(set_pos2, set_cp2); |
|
|
|
|
|
cp_SetOnBoard(src, PIECE_NONE); |
|
|
|
|
|
if ( clr_pos2 != ILLEGAL_POSITION ) |
|
|
cp_SetOnBoard(clr_pos2, PIECE_NONE); |
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
/* |
|
|
this subprocedure decides for evaluation of the current board situation or further (deeper) investigation |
|
|
Argument pos is the new target position if the current piece |
|
|
|
|
|
*/ |
|
|
uint8_t ce_LoopRecur(uint8_t pos) |
|
|
{ |
|
|
eval_t eval; |
|
|
|
|
|
/* 1. check if target position is occupied by the same player (my_color) */ |
|
|
/* of if pos is somehow illegal or not valid */ |
|
|
if ( cu_IsIllegalPosition(pos, stack_GetCurrElement()->current_color) != 0 ) |
|
|
return 0; |
|
|
|
|
|
/* 2. move piece to the specified position, capture opponent piece if required */ |
|
|
cu_Move(stack_GetCurrElement()->current_pos, pos); |
|
|
|
|
|
|
|
|
/* 3. */ |
|
|
/* if depth reached: evaluate */ |
|
|
/* else: go down next level */ |
|
|
/* no eval if there had been any valid half-moves, so the default value (MIN) will be returned. */ |
|
|
if ( stack_Push(stack_GetCurrElement()->current_color) == 0 ) |
|
|
{ |
|
|
eval = ce_Eval(); |
|
|
} |
|
|
else |
|
|
{ |
|
|
/* init the element, which has been pushed */ |
|
|
stack_InitCurrElement(); |
|
|
/* start over with ntext level */ |
|
|
ce_LoopPieces(); |
|
|
/* get the best move from opponents view, so invert the result */ |
|
|
eval = -stack_GetCurrElement()->best_eval; |
|
|
stack_Pop(); |
|
|
} |
|
|
|
|
|
/* 4. store result */ |
|
|
stack_SetMove(eval, pos); |
|
|
|
|
|
/* 5. undo the move */ |
|
|
cu_UndoHalfMove(); |
|
|
|
|
|
/* 6. check special modes */ |
|
|
/* the purpose of these checks is to mark special pieces and positions on the board */ |
|
|
/* these marks can be checked by the user interface to highlight special positions */ |
|
|
if ( lrc_obj.check_mode != 0 ) |
|
|
{ |
|
|
stack_element_p e = stack_GetCurrElement(); |
|
|
if ( lrc_obj.check_mode == CHECK_MODE_MOVEABLE ) |
|
|
{ |
|
|
cp_SetOnBoard(e->current_pos, e->current_cp | CP_MARK_MASK ); |
|
|
} |
|
|
else if ( lrc_obj.check_mode == CHECK_MODE_TARGET_MOVE ) |
|
|
{ |
|
|
if ( e->current_pos == lrc_obj.check_src_pos ) |
|
|
{ |
|
|
cp_SetOnBoard(pos, cp_GetFromBoard(pos) | CP_MARK_MASK ); |
|
|
} |
|
|
} |
|
|
} |
|
|
return 1; |
|
|
} |
|
|
|
|
|
/*==============================================================*/ |
|
|
/* move pieces which can move one or more steps into a direction */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
/* |
|
|
subprocedure to generate various target positions for some pieces |
|
|
special cases are handled in the piece specific sub-procedure |
|
|
|
|
|
Arguments: |
|
|
d: a list of potential directions |
|
|
is_multi_step: if the piece can only do one step (zero for KING and KNIGHT) |
|
|
*/ |
|
|
static const uint8_t ce_dir_offset_rook[] CHESS_PROGMEM = { 1, 16, (uint8_t)-16, (uint8_t)-1, 0 }; |
|
|
static const uint8_t ce_dir_offset_bishop[] CHESS_PROGMEM = { 15, 17, (uint8_t)-17, (uint8_t)-15, 0 }; |
|
|
static const uint8_t ce_dir_offset_queen[] CHESS_PROGMEM = { 1, 16, (uint8_t)-16, (uint8_t)-1, 15, 17, (uint8_t)-17, (uint8_t)-15, 0 }; |
|
|
static const uint8_t ce_dir_offset_knight[] CHESS_PROGMEM = {14, (uint8_t)-14, 18, (uint8_t)-18, 31, (uint8_t)-31, 33, (uint8_t)-33, 0}; |
|
|
|
|
|
void ce_LoopDirsSingleMultiStep(const uint8_t *d, uint8_t is_multi_step) |
|
|
{ |
|
|
uint8_t loop_pos; |
|
|
|
|
|
/* with all directions */ |
|
|
for(;;) |
|
|
{ |
|
|
if ( chess_pgm_read(d) == 0 ) |
|
|
break; |
|
|
|
|
|
/* start again from the initial position */ |
|
|
loop_pos = stack_GetCurrElement()->current_pos; |
|
|
|
|
|
/* check direction */ |
|
|
do |
|
|
{ |
|
|
/* check next position into one direction */ |
|
|
loop_pos += chess_pgm_read(d); |
|
|
|
|
|
/* |
|
|
go further to ce_LoopRecur() |
|
|
0 will be returned if the target position is illegal or a piece of the own color |
|
|
this is used to stop walking into one direction |
|
|
*/ |
|
|
if ( ce_LoopRecur(loop_pos) == 0 ) |
|
|
break; |
|
|
|
|
|
/* stop if we had hit another piece */ |
|
|
if ( cp_GetPiece(cp_GetFromBoard(loop_pos)) != PIECE_NONE ) |
|
|
break; |
|
|
} while( is_multi_step ); |
|
|
d++; |
|
|
} |
|
|
} |
|
|
|
|
|
void ce_LoopRook(void) |
|
|
{ |
|
|
ce_LoopDirsSingleMultiStep(ce_dir_offset_rook, 1); |
|
|
} |
|
|
|
|
|
void ce_LoopBishop(void) |
|
|
{ |
|
|
ce_LoopDirsSingleMultiStep(ce_dir_offset_bishop, 1); |
|
|
} |
|
|
|
|
|
void ce_LoopQueen(void) |
|
|
{ |
|
|
ce_LoopDirsSingleMultiStep(ce_dir_offset_queen, 1); |
|
|
} |
|
|
|
|
|
void ce_LoopKnight(void) |
|
|
{ |
|
|
ce_LoopDirsSingleMultiStep(ce_dir_offset_knight, 0); |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
/*==============================================================*/ |
|
|
/* move king */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
uint8_t cu_IsKingCastling(uint8_t mask, int8_t direction, uint8_t cnt) U8G2_NOINLINE; |
|
|
|
|
|
/* |
|
|
checks, if the king can do castling |
|
|
|
|
|
Arguments: |
|
|
mask: the bit-mask for the global "castling possible" flag |
|
|
direction: left castling: -1, right castling 1 |
|
|
cnt: number of fields to be checked: 3 or 2 |
|
|
*/ |
|
|
uint8_t cu_IsKingCastling(uint8_t mask, int8_t direction, uint8_t cnt) |
|
|
{ |
|
|
uint8_t pos; |
|
|
uint8_t opponent_color; |
|
|
|
|
|
/* check if the current board state allows castling */ |
|
|
if ( (lrc_obj.castling_possible & mask) == 0 ) |
|
|
return 0; /* castling not allowed */ |
|
|
|
|
|
/* get the position of the KING, could be white or black king */ |
|
|
pos = stack_GetCurrElement()->current_pos; |
|
|
|
|
|
/* calculate the color of the opponent */ |
|
|
opponent_color = 1; |
|
|
opponent_color -= stack_GetCurrElement()->current_color; |
|
|
|
|
|
/* if the KING itself is given check... */ |
|
|
if ( ce_GetPositionAttackWeight(pos, opponent_color) > 0 ) |
|
|
return 0; |
|
|
|
|
|
|
|
|
/* check if fields in the desired direction are emtpy */ |
|
|
for(;;) |
|
|
{ |
|
|
/* go to the next field */ |
|
|
pos += direction; |
|
|
/* check for a piece */ |
|
|
if ( cp_GetPiece(cp_GetFromBoard(pos)) != PIECE_NONE ) |
|
|
return 0; /* castling not allowed */ |
|
|
|
|
|
/* if some of the fields are under attack */ |
|
|
if ( ce_GetPositionAttackWeight(pos, opponent_color) > 0 ) |
|
|
return 0; |
|
|
|
|
|
cnt--; |
|
|
if ( cnt == 0 ) |
|
|
break; |
|
|
} |
|
|
return 1; /* castling allowed */ |
|
|
} |
|
|
|
|
|
void ce_LoopKing(void) |
|
|
{ |
|
|
/* |
|
|
there is an interessting timing problem in this procedure |
|
|
it must be checked for castling first and as second step the normal |
|
|
KING movement. If we would first check for normal moves, than |
|
|
any marks might be overwritten by the ROOK in the case of castling. |
|
|
*/ |
|
|
|
|
|
/* castling (this must be done before checking normal moves (see above) */ |
|
|
if ( stack_GetCurrElement()->current_color == COLOR_WHITE ) |
|
|
{ |
|
|
/* white left castling */ |
|
|
if ( cu_IsKingCastling(1, -1, 3) != 0 ) |
|
|
{ |
|
|
/* check for attacked fields */ |
|
|
ce_LoopRecur(stack_GetCurrElement()->current_pos-2); |
|
|
} |
|
|
/* white right castling */ |
|
|
if ( cu_IsKingCastling(2, 1, 2) != 0 ) |
|
|
{ |
|
|
/* check for attacked fields */ |
|
|
ce_LoopRecur(stack_GetCurrElement()->current_pos+2); |
|
|
} |
|
|
} |
|
|
else |
|
|
{ |
|
|
/* black left castling */ |
|
|
if ( cu_IsKingCastling(4, -1, 3) != 0 ) |
|
|
{ |
|
|
/* check for attacked fields */ |
|
|
ce_LoopRecur(stack_GetCurrElement()->current_pos-2); |
|
|
} |
|
|
/* black right castling */ |
|
|
if ( cu_IsKingCastling(8, 1, 2) != 0 ) |
|
|
{ |
|
|
/* check for attacked fields */ |
|
|
ce_LoopRecur(stack_GetCurrElement()->current_pos+2); |
|
|
} |
|
|
} |
|
|
|
|
|
/* reuse queen directions */ |
|
|
ce_LoopDirsSingleMultiStep(ce_dir_offset_queen, 0); |
|
|
} |
|
|
|
|
|
|
|
|
/*==============================================================*/ |
|
|
/* move pawn */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
/* |
|
|
doppelschritt: nur von der grundlinie aus, beide (!) felder vor dem bauern m<EFBFBD>ssen frei sein |
|
|
en passant: nur unmittelbar nachdem ein doppelschritt ausgef<EFBFBD>hrt wurde. |
|
|
*/ |
|
|
void ce_LoopPawnSideCapture(uint8_t loop_pos) |
|
|
{ |
|
|
if ( gpos_IsIllegal(loop_pos) == 0 ) |
|
|
{ |
|
|
/* get the piece from the board */ |
|
|
/* if the field is NOT empty */ |
|
|
if ( cp_GetPiece(cp_GetFromBoard(loop_pos)) != PIECE_NONE ) |
|
|
{ |
|
|
/* normal capture */ |
|
|
ce_LoopRecur(loop_pos); |
|
|
/* TODO: check for pawn conversion/promotion */ |
|
|
} |
|
|
else |
|
|
{ |
|
|
/* check conditions for en passant capture */ |
|
|
if ( stack_GetCurrElement()->current_color == COLOR_WHITE ) |
|
|
{ |
|
|
if ( lrc_obj.pawn_dbl_move[COLOR_BLACK]+16 == loop_pos ) |
|
|
{ |
|
|
ce_LoopRecur(loop_pos); |
|
|
/* note: pawn conversion/promotion can not occur */ |
|
|
} |
|
|
} |
|
|
else |
|
|
{ |
|
|
if ( lrc_obj.pawn_dbl_move[COLOR_WHITE] == loop_pos+16 ) |
|
|
{ |
|
|
ce_LoopRecur(loop_pos); |
|
|
/* note: pawn conversion/promotion can not occur */ |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
void ce_LoopPawn(void) |
|
|
{ |
|
|
uint8_t initial_pos = stack_GetCurrElement()->current_pos; |
|
|
uint8_t my_color = stack_GetCurrElement()->current_color; |
|
|
|
|
|
uint8_t loop_pos; |
|
|
uint8_t line; |
|
|
|
|
|
/* one step forward */ |
|
|
|
|
|
loop_pos = initial_pos; |
|
|
line = initial_pos; |
|
|
line >>= 4; |
|
|
if ( my_color == COLOR_WHITE ) |
|
|
loop_pos += 16; |
|
|
else |
|
|
loop_pos -= 16; |
|
|
if ( gpos_IsIllegal(loop_pos) == 0 ) |
|
|
{ |
|
|
/* if the field is empty */ |
|
|
if ( cp_GetPiece(cp_GetFromBoard(loop_pos)) == PIECE_NONE ) |
|
|
{ |
|
|
/* TODO: check for and loop through piece conversion/promotion */ |
|
|
ce_LoopRecur(loop_pos); |
|
|
|
|
|
/* second step forward */ |
|
|
|
|
|
/* if pawn is on his starting line */ |
|
|
if ( (my_color == COLOR_WHITE && line == 1) || (my_color == COLOR_BLACK && line == 6 ) ) |
|
|
{ |
|
|
/* the place before the pawn is not occupied, so we can do double moves, see above */ |
|
|
|
|
|
if ( my_color == COLOR_WHITE ) |
|
|
loop_pos += 16; |
|
|
else |
|
|
loop_pos -= 16; |
|
|
if ( cp_GetPiece(cp_GetFromBoard(loop_pos)) == PIECE_NONE ) |
|
|
{ |
|
|
/* this is a special case, other promotions of the pawn can not occur */ |
|
|
ce_LoopRecur(loop_pos); |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
/* capture */ |
|
|
|
|
|
loop_pos = initial_pos; |
|
|
if ( my_color == COLOR_WHITE ) |
|
|
loop_pos += 15; |
|
|
else |
|
|
loop_pos -= 15; |
|
|
ce_LoopPawnSideCapture(loop_pos); |
|
|
|
|
|
|
|
|
loop_pos = initial_pos; |
|
|
if ( my_color == COLOR_WHITE ) |
|
|
loop_pos += 17; |
|
|
else |
|
|
loop_pos -= 17; |
|
|
ce_LoopPawnSideCapture(loop_pos); |
|
|
} |
|
|
|
|
|
/*==============================================================*/ |
|
|
/* attacked */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
/* |
|
|
from a starting position, search for a piece, that might jump to that postion. |
|
|
return: |
|
|
the two global variables |
|
|
lrc_obj.find_piece_weight[0]; |
|
|
lrc_obj.find_piece_weight[1]; |
|
|
will be increased by the weight of the attacked pieces of that color. |
|
|
it is usually required to reset these global variables to zero, before using |
|
|
this function. |
|
|
*/ |
|
|
|
|
|
void ce_FindPieceByStep(uint8_t start_pos, uint8_t piece, const uint8_t *d, uint8_t is_multi_step) |
|
|
{ |
|
|
uint8_t loop_pos, cp; |
|
|
|
|
|
/* with all directions */ |
|
|
for(;;) |
|
|
{ |
|
|
if ( chess_pgm_read(d) == 0 ) |
|
|
break; |
|
|
|
|
|
/* start again from the initial position */ |
|
|
loop_pos = start_pos; |
|
|
|
|
|
/* check direction */ |
|
|
do |
|
|
{ |
|
|
/* check next position into one direction */ |
|
|
loop_pos += chess_pgm_read(d); |
|
|
|
|
|
/* check if the board boundary has been crossed */ |
|
|
if ( (loop_pos & 0x088) != 0 ) |
|
|
break; |
|
|
|
|
|
/* get the colored piece from the board */ |
|
|
cp = cp_GetFromBoard(loop_pos); |
|
|
|
|
|
/* stop if we had hit another piece */ |
|
|
if ( cp_GetPiece(cp) != PIECE_NONE ) |
|
|
{ |
|
|
/* if it is the piece we are looking for, then add the weight */ |
|
|
if ( cp_GetPiece(cp) == piece ) |
|
|
{ |
|
|
lrc_obj.find_piece_weight[cp_GetColor(cp)] += ce_piece_weight[piece]; |
|
|
lrc_obj.find_piece_cnt[cp_GetColor(cp)]++; |
|
|
} |
|
|
/* in any case, break out of the inner loop */ |
|
|
break; |
|
|
} |
|
|
} while( is_multi_step ); |
|
|
d++; |
|
|
} |
|
|
} |
|
|
|
|
|
void ce_FindPawnPiece(uint8_t dest_pos, uint8_t color) |
|
|
{ |
|
|
uint8_t cp; |
|
|
/* check if the board boundary has been crossed */ |
|
|
if ( (dest_pos & 0x088) == 0 ) |
|
|
{ |
|
|
/* get the colored piece from the board */ |
|
|
cp = cp_GetFromBoard(dest_pos); |
|
|
/* only if there is a pawn of the matching color */ |
|
|
if ( cp_GetPiece(cp) == PIECE_PAWN ) |
|
|
{ |
|
|
if ( cp_GetColor(cp) == color ) |
|
|
{ |
|
|
/* the weight of the PAWN */ |
|
|
lrc_obj.find_piece_weight[color] += 1; |
|
|
lrc_obj.find_piece_cnt[color]++; |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
/* |
|
|
find out, which pieces do attack a specified field |
|
|
used to |
|
|
- check if the KING can do castling |
|
|
- check if the KING must move |
|
|
|
|
|
may be used in the eval procedure ... once... |
|
|
|
|
|
the result is stored in the global array |
|
|
uint8_t lrc_obj.find_piece_weight[2]; |
|
|
which is indexed with the color. |
|
|
lrc_obj.find_piece_weight[COLOR_WHITE] is the sum of all white pieces |
|
|
which can directly move to this field. |
|
|
|
|
|
example: |
|
|
if the black KING is at "pos" and lrc_obj.find_piece_weight[COLOR_WHITE] is not zero |
|
|
(after executing ce_CalculatePositionWeight(pos)) then the KING must be protected or moveed, because |
|
|
the KING was given check. |
|
|
*/ |
|
|
|
|
|
void ce_CalculatePositionWeight(uint8_t pos) |
|
|
{ |
|
|
|
|
|
lrc_obj.find_piece_weight[0] = 0; |
|
|
lrc_obj.find_piece_weight[1] = 0; |
|
|
lrc_obj.find_piece_cnt[0] = 0; |
|
|
lrc_obj.find_piece_cnt[1] = 0; |
|
|
|
|
|
if ( (pos & 0x088) != 0 ) |
|
|
return; |
|
|
|
|
|
ce_FindPieceByStep(pos, PIECE_ROOK, ce_dir_offset_rook, 1); |
|
|
ce_FindPieceByStep(pos, PIECE_BISHOP, ce_dir_offset_bishop, 1); |
|
|
ce_FindPieceByStep(pos, PIECE_QUEEN, ce_dir_offset_queen, 1); |
|
|
ce_FindPieceByStep(pos, PIECE_KNIGHT, ce_dir_offset_knight, 0); |
|
|
ce_FindPieceByStep(pos, PIECE_KING, ce_dir_offset_queen, 0); |
|
|
|
|
|
ce_FindPawnPiece(pos+17, COLOR_BLACK); |
|
|
ce_FindPawnPiece(pos+15, COLOR_BLACK); |
|
|
ce_FindPawnPiece(pos-17, COLOR_WHITE); |
|
|
ce_FindPawnPiece(pos-15, COLOR_WHITE); |
|
|
} |
|
|
|
|
|
/* |
|
|
calculate the summed weight of pieces with specified color which can move to a specified position |
|
|
|
|
|
argument: |
|
|
pos: the position which should be analysed |
|
|
color: the color of those pieces which should be analysed |
|
|
e.g. if a black piece is at 'pos' and 'color' is white then this procedure returns the white atting count |
|
|
*/ |
|
|
uint8_t ce_GetPositionAttackWeight(uint8_t pos, uint8_t color) |
|
|
{ |
|
|
ce_CalculatePositionWeight(pos); |
|
|
return lrc_obj.find_piece_weight[color]; |
|
|
} |
|
|
|
|
|
uint8_t ce_GetPositionAttackCount(uint8_t pos, uint8_t color) |
|
|
{ |
|
|
ce_CalculatePositionWeight(pos); |
|
|
return lrc_obj.find_piece_cnt[color]; |
|
|
} |
|
|
|
|
|
|
|
|
/*==============================================================*/ |
|
|
/* depth search starts here: loop over all pieces of the current color on the board */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
void ce_LoopPieces(void) |
|
|
{ |
|
|
stack_element_p e = stack_GetCurrElement(); |
|
|
/* start with lower left position (A1) */ |
|
|
e->current_pos = 0; |
|
|
do |
|
|
{ |
|
|
e->current_cp = cp_GetFromBoard(e->current_pos); |
|
|
/* check if the position on the board is empty */ |
|
|
if ( e->current_cp != 0 ) |
|
|
{ |
|
|
/* only generate moves for the current color */ |
|
|
if ( e->current_color == cp_GetColor(e->current_cp) ) |
|
|
{ |
|
|
chess_Thinking(); |
|
|
|
|
|
/* find out which piece is used */ |
|
|
switch(cp_GetPiece(e->current_cp)) |
|
|
{ |
|
|
case PIECE_NONE: |
|
|
break; |
|
|
case PIECE_PAWN: |
|
|
ce_LoopPawn(); |
|
|
break; |
|
|
case PIECE_KNIGHT: |
|
|
ce_LoopKnight(); |
|
|
break; |
|
|
case PIECE_BISHOP: |
|
|
ce_LoopBishop(); |
|
|
break; |
|
|
case PIECE_ROOK: |
|
|
ce_LoopRook(); |
|
|
break; |
|
|
case PIECE_QUEEN: |
|
|
ce_LoopQueen(); |
|
|
break; |
|
|
case PIECE_KING: |
|
|
ce_LoopKing(); |
|
|
break; |
|
|
} |
|
|
} |
|
|
} |
|
|
e->current_pos = cu_NextPos(e->current_pos); |
|
|
} while( e->current_pos != 0 ); |
|
|
} |
|
|
|
|
|
/*==============================================================*/ |
|
|
/* user interface */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
/* |
|
|
eval_t chess_EvalCurrBoard(uint8_t color) |
|
|
{ |
|
|
stack_Init(0); |
|
|
stack_GetCurrElement()->current_color = color; |
|
|
ce_LoopPieces(); |
|
|
return stack_GetCurrElement()->best_eval; |
|
|
} |
|
|
*/ |
|
|
|
|
|
/* clear any marks on the board */ |
|
|
void chess_ClearMarks(void) |
|
|
{ |
|
|
uint8_t i; |
|
|
for( i = 0; i < 64; i++ ) |
|
|
lrc_obj.board[i] &= ~CP_MARK_MASK; |
|
|
} |
|
|
|
|
|
/* |
|
|
Mark all pieces which can do moves. This is done by setting flags on the global board |
|
|
*/ |
|
|
void chess_MarkMovable(void) |
|
|
{ |
|
|
stack_Init(0); |
|
|
//stack_GetCurrElement()->current_color = color; |
|
|
lrc_obj.check_mode = CHECK_MODE_MOVEABLE; |
|
|
ce_LoopPieces(); |
|
|
} |
|
|
|
|
|
/* |
|
|
Checks, if the piece can move from src_pos to dest_pos |
|
|
|
|
|
src_pos: The game position of a piece on the chess board |
|
|
*/ |
|
|
void chess_MarkTargetMoves(uint8_t src_pos) |
|
|
{ |
|
|
stack_Init(0); |
|
|
stack_GetCurrElement()->current_color = cp_GetColor(cp_GetFromBoard(src_pos)); |
|
|
lrc_obj.check_src_pos = src_pos; |
|
|
lrc_obj.check_mode = CHECK_MODE_TARGET_MOVE; |
|
|
ce_LoopPieces(); |
|
|
} |
|
|
|
|
|
/* |
|
|
first call should start with 255 |
|
|
this procedure will return 255 if |
|
|
- there are no marks at all |
|
|
- it has looped over all marks once |
|
|
*/ |
|
|
uint8_t chess_GetNextMarked(uint8_t arg, uint8_t is_prev) |
|
|
{ |
|
|
uint8_t i; |
|
|
uint8_t pos = arg; |
|
|
for(i = 0; i < 64; i++) |
|
|
{ |
|
|
if ( is_prev != 0 ) |
|
|
pos = cu_PrevPos(pos); |
|
|
else |
|
|
pos = cu_NextPos(pos); |
|
|
if ( arg != 255 && pos == 0 ) |
|
|
return 255; |
|
|
if ( cp_IsMarked(cp_GetFromBoard(pos)) ) |
|
|
return pos; |
|
|
} |
|
|
return 255; |
|
|
} |
|
|
|
|
|
|
|
|
/* make a manual move: this is a little bit more than cu_Move() */ |
|
|
void chess_ManualMove(uint8_t src, uint8_t dest) |
|
|
{ |
|
|
uint8_t cp; |
|
|
|
|
|
/* printf("chess_ManualMove %02x -> %02x\n", src, dest); */ |
|
|
|
|
|
/* if all other things fail, this is the place where the game is to be decided: */ |
|
|
/* ... if the KING is captured */ |
|
|
cp = cp_GetFromBoard(dest); |
|
|
if ( cp_GetPiece(cp) == PIECE_KING ) |
|
|
{ |
|
|
lrc_obj.is_game_end = 1; |
|
|
lrc_obj.lost_side_color = cp_GetColor(cp); |
|
|
} |
|
|
|
|
|
/* clear ply history here, to avoid memory overflow */ |
|
|
/* may be the last X moves can be kept here */ |
|
|
cu_ReduceHistoryByFullMove(); |
|
|
/* perform the move on the board */ |
|
|
cu_Move(src, dest); |
|
|
|
|
|
/* update en passant double move positions: en passant position is removed after two half moves */ |
|
|
lrc_obj.pawn_dbl_move[lrc_obj.ply_count&1] = ILLEGAL_POSITION; |
|
|
|
|
|
/* update the global half move counter */ |
|
|
lrc_obj.ply_count++; |
|
|
|
|
|
|
|
|
/* make a small check about the end of the game */ |
|
|
/* use at least depth 1, because we must know if the king can still move */ |
|
|
/* this is: King moves at level 0 and will be captured at level 1 */ |
|
|
/* so we check if the king can move and will not be captured at search level 1 */ |
|
|
|
|
|
stack_Init(1); |
|
|
ce_LoopPieces(); |
|
|
|
|
|
/* printf("chess_ManualMove/analysis best_from_pos %02x -> best_to_pos %02x\n", stack_GetCurrElement()->best_from_pos, stack_GetCurrElement()->best_to_pos); */ |
|
|
|
|
|
/* analyse the eval result */ |
|
|
|
|
|
/* check if the other player has any moves left */ |
|
|
if ( stack_GetCurrElement()->best_from_pos == ILLEGAL_POSITION ) |
|
|
{ |
|
|
uint8_t color; |
|
|
/* conditions: */ |
|
|
/* 1. no King, should never happen, opposite color has won */ |
|
|
/* this is already checked above at the beginning if this procedure */ |
|
|
/* 2. King is under attack, opposite color has won */ |
|
|
/* 3. King is not under attack, game is a draw */ |
|
|
|
|
|
uint8_t i = 0; |
|
|
color = lrc_obj.ply_count; |
|
|
color &= 1; |
|
|
do |
|
|
{ |
|
|
cp = cp_GetFromBoard(i); |
|
|
/* look for the King */ |
|
|
if ( cp_GetPiece(cp) == PIECE_KING ) |
|
|
{ |
|
|
if ( cp_GetColor(cp) == color ) |
|
|
{ |
|
|
/* check if KING is attacked */ |
|
|
if ( ce_GetPositionAttackCount(i, color^1) != 0 ) |
|
|
{ |
|
|
/* KING is under attack (check) and can not move: Game is lost */ |
|
|
lrc_obj.is_game_end = 1; |
|
|
lrc_obj.lost_side_color = color; |
|
|
} |
|
|
else |
|
|
{ |
|
|
/* KING is NOT under attack (check) but can not move: Game is a draw */ |
|
|
lrc_obj.is_game_end = 1; |
|
|
lrc_obj.lost_side_color = 2; |
|
|
} |
|
|
/* break out of the loop */ |
|
|
break; |
|
|
} |
|
|
} |
|
|
i = cu_NextPos(i); |
|
|
} while( i != 0 ); |
|
|
} |
|
|
} |
|
|
|
|
|
/* let the computer do a move */ |
|
|
void chess_ComputerMove(uint8_t depth) |
|
|
{ |
|
|
stack_Init(depth); |
|
|
|
|
|
//stack_GetCurrElement()->current_color = lrc_obj.ply_count; |
|
|
//stack_GetCurrElement()->current_color &= 1; |
|
|
|
|
|
cu_ReduceHistoryByFullMove(); |
|
|
ce_LoopPieces(); |
|
|
|
|
|
chess_ManualMove(stack_GetCurrElement()->best_from_pos, stack_GetCurrElement()->best_to_pos); |
|
|
} |
|
|
|
|
|
|
|
|
/*==============================================================*/ |
|
|
/* unix code */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
#ifdef UNIX_MAIN |
|
|
|
|
|
#else |
|
|
|
|
|
/*==============================================================*/ |
|
|
/* display menu */ |
|
|
/*==============================================================*/ |
|
|
|
|
|
extern const uint8_t chess_pieces_body_bm[] CHESS_PROGMEM; // forward decl |
|
|
extern const uint8_t chess_black_pieces_bm[] CHESS_PROGMEM; // forward decl |
|
|
|
|
|
// uint8_t chess_key_code = 0; // obsolete, u8g2 does proper debouncing |
|
|
uint8_t chess_key_cmd = 0; |
|
|
#define CHESS_STATE_MENU 0 |
|
|
#define CHESS_STATE_SELECT_START 1 |
|
|
#define CHESS_STATE_SELECT_PIECE 2 |
|
|
#define CHESS_STATE_SELECT_TARGET_POS 3 |
|
|
#define CHESS_STATE_THINKING 4 |
|
|
#define CHESS_STATE_GAME_END 5 |
|
|
uint8_t chess_state = CHESS_STATE_MENU; |
|
|
uint8_t chess_source_pos = 255; |
|
|
uint8_t chess_target_pos = 255; |
|
|
|
|
|
#define MNU_FONT u8g2_font_5x8_tr |
|
|
#define MNU_ENTRY_HEIGHT 9 |
|
|
|
|
|
const char *mnu_title = "Little Rook Chess"; |
|
|
const char *mnu_list[] = { "New Game (White)", "New Game (Black)", "Undo Move", "Return" }; |
|
|
uint8_t mnu_pos = 0; |
|
|
uint8_t mnu_max = 4; |
|
|
|
|
|
void mnu_DrawHome(uint8_t is_highlight) |
|
|
{ |
|
|
uint8_t x = lrc_u8g->width - 35; |
|
|
uint8_t y = (lrc_u8g->height-1); |
|
|
uint8_t t; |
|
|
|
|
|
u8g2_SetFont(lrc_u8g, u8g2_font_5x7_tr); |
|
|
u8g2_SetDefaultForegroundColor(lrc_u8g); |
|
|
if ( chess_state == CHESS_STATE_THINKING ) |
|
|
u8g2_DrawBitmap(lrc_u8g, x, y-8, 1, 8, chess_black_pieces_bm+24); |
|
|
else |
|
|
{ |
|
|
t = u8g2_DrawStrP(lrc_u8g, x, y -1, CHESS_PSTR("Options")); |
|
|
|
|
|
if ( is_highlight ) |
|
|
u8g2_DrawFrame(lrc_u8g, x-1, y - MNU_ENTRY_HEIGHT +1, t, MNU_ENTRY_HEIGHT); |
|
|
} |
|
|
} |
|
|
|
|
|
void mnu_DrawEntry(uint8_t y, const char *str, uint8_t is_clr_background, uint8_t is_highlight) |
|
|
{ |
|
|
uint8_t t, x; |
|
|
u8g2_SetFont(lrc_u8g, MNU_FONT); |
|
|
t = u8g2_GetStrWidth(lrc_u8g, str); |
|
|
x = u8g2_GetDisplayWidth(lrc_u8g); |
|
|
x -= t; |
|
|
x >>= 1; |
|
|
|
|
|
if ( is_clr_background ) |
|
|
{ |
|
|
u8g2_SetDefaultBackgroundColor(lrc_u8g); |
|
|
u8g2_DrawBox(lrc_u8g, x-3, (lrc_u8g->height-1) - (y+MNU_ENTRY_HEIGHT-1+2), t+5, MNU_ENTRY_HEIGHT+4); |
|
|
} |
|
|
|
|
|
u8g2_SetDefaultForegroundColor(lrc_u8g); |
|
|
u8g2_DrawStr(lrc_u8g, x, (lrc_u8g->height-1) - y, str); |
|
|
|
|
|
if ( is_highlight ) |
|
|
{ |
|
|
u8g2_DrawFrame(lrc_u8g, 0, (lrc_u8g->height-1) - y -MNU_ENTRY_HEIGHT +1, u8g2_GetDisplayWidth(lrc_u8g), MNU_ENTRY_HEIGHT); |
|
|
} |
|
|
} |
|
|
|
|
|
void mnu_Draw(void) |
|
|
{ |
|
|
uint8_t i; |
|
|
uint8_t t,y; |
|
|
/* calculate hight of the complete menu */ |
|
|
y = mnu_max; |
|
|
y++; /* consider also some space for the title */ |
|
|
y++; /* consider also some space for the title */ |
|
|
y *= MNU_ENTRY_HEIGHT; |
|
|
|
|
|
/* calculate how much space will be left */ |
|
|
t = u8g2_GetDisplayHeight(lrc_u8g); |
|
|
t -= y; |
|
|
|
|
|
/* topmost pos start half of that empty space from the top */ |
|
|
t >>= 1; |
|
|
y = u8g2_GetDisplayHeight(lrc_u8g); |
|
|
y -= t; |
|
|
|
|
|
y -= MNU_ENTRY_HEIGHT; |
|
|
mnu_DrawEntry(y, mnu_title, 0, 0); |
|
|
|
|
|
y -= MNU_ENTRY_HEIGHT; |
|
|
|
|
|
|
|
|
u8g2_DrawBitmap(lrc_u8g, 0, 1, 1, 8, chess_black_pieces_bm+24); |
|
|
u8g2_DrawBitmap(lrc_u8g, 128-8, 1, 1, 8, chess_black_pieces_bm+24); |
|
|
|
|
|
|
|
|
for( i = 0; i < mnu_max; i++ ) |
|
|
{ |
|
|
y -= MNU_ENTRY_HEIGHT; |
|
|
mnu_DrawEntry(y, mnu_list[i], 0, i == mnu_pos); |
|
|
} |
|
|
} |
|
|
|
|
|
void mnu_Step(uint8_t key_cmd) |
|
|
{ |
|
|
if ( key_cmd == CHESS_KEY_NEXT ) |
|
|
{ |
|
|
if ( mnu_pos+1 < mnu_max ) |
|
|
mnu_pos++; |
|
|
} |
|
|
else if ( key_cmd == CHESS_KEY_PREV ) |
|
|
{ |
|
|
if ( mnu_pos > 0 ) |
|
|
mnu_pos--; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const uint8_t chess_pieces_body_bm[] CHESS_PROGMEM = |
|
|
{ |
|
|
/* PAWN */ 0x00, 0x00, 0x00, 0x18, 0x18, 0x00, 0x00, 0x00, /* 0x00, 0x00, 0x00, 0x0c, 0x0c, 0x00, 0x00, 0x00, */ |
|
|
/* KNIGHT */ 0x00, 0x00, 0x1c, 0x2c, 0x04, 0x04, 0x0e, 0x00, |
|
|
/* BISHOP */ 0x00, 0x00, 0x1c, 0x1c, 0x1c, 0x08, 0x00, 0x00, /* 0x00, 0x00, 0x08, 0x1c, 0x1c, 0x08, 0x00, 0x00, */ |
|
|
/* ROOK */ 0x00, 0x00, 0x00, 0x1c, 0x1c, 0x1c, 0x1c, 0x00, |
|
|
/* QUEEN */ 0x00, 0x00, 0x14, 0x1c, 0x08, 0x1c, 0x08, 0x00, |
|
|
/* KING */ 0x00, 0x00, 0x00, 0x08, 0x3e, 0x1c, 0x08, 0x00, |
|
|
}; |
|
|
|
|
|
#ifdef NOT_REQUIRED |
|
|
/* white pieces are constructed by painting black pieces and cutting out the white area */ |
|
|
const uint8_t chess_white_pieces_bm[] CHESS_PROGMEM = |
|
|
{ |
|
|
/* PAWN */ 0x00, 0x00, 0x0c, 0x12, 0x12, 0x0c, 0x1e, 0x00, |
|
|
/* KNIGHT */ 0x00, 0x1c, 0x22, 0x52, 0x6a, 0x0a, 0x11, 0x1f, |
|
|
/* BISHOP */ 0x00, 0x08, 0x14, 0x22, 0x22, 0x14, 0x08, 0x7f, |
|
|
/* ROOK */ 0x00, 0x55, 0x7f, 0x22, 0x22, 0x22, 0x22, 0x7f, |
|
|
/* QUEEN */ 0x00, 0x55, 0x2a, 0x22, 0x14, 0x22, 0x14, 0x7f, |
|
|
/* KING */ 0x08, 0x1c, 0x49, 0x77, 0x41, 0x22, 0x14, 0x7f, |
|
|
}; |
|
|
#endif |
|
|
|
|
|
const uint8_t chess_black_pieces_bm[] CHESS_PROGMEM = |
|
|
{ |
|
|
/* PAWN */ 0x00, 0x00, 0x18, 0x3c, 0x3c, 0x18, 0x3c, 0x00, /* 0x00, 0x00, 0x0c, 0x1e, 0x1e, 0x0c, 0x1e, 0x00, */ |
|
|
/* KNIGHT */ 0x00, 0x1c, 0x3e, 0x7e, 0x6e, 0x0e, 0x1f, 0x1f, |
|
|
/* BISHOP */ 0x00, 0x1c, 0x2e, 0x3e, 0x3e, 0x1c, 0x08, 0x7f, /*0x00, 0x08, 0x1c, 0x3e, 0x3e, 0x1c, 0x08, 0x7f,*/ |
|
|
/* ROOK */ 0x00, 0x55, 0x7f, 0x3e, 0x3e, 0x3e, 0x3e, 0x7f, |
|
|
/* QUEEN */ 0x00, 0x55, 0x3e, 0x3e, 0x1c, 0x3e, 0x1c, 0x7f, |
|
|
/* KING -*/ 0x08, 0x1c, 0x49, 0x7f, 0x7f, 0x3e, 0x1c, 0x7f, |
|
|
}; |
|
|
|
|
|
|
|
|
#if defined(DOGXL160_HW_GR) |
|
|
#define BOXSIZE 13 |
|
|
#define BOXOFFSET 3 |
|
|
#else |
|
|
#define BOXSIZE 8 |
|
|
#define BOXOFFSET 1 |
|
|
#endif |
|
|
|
|
|
u8g2_uint_t chess_low_edge; |
|
|
uint8_t chess_boxsize = 8; |
|
|
uint8_t chess_boxoffset = 1; |
|
|
|
|
|
|
|
|
void chess_DrawFrame(uint8_t pos, uint8_t is_bold) |
|
|
{ |
|
|
u8g2_uint_t x0, y0; |
|
|
|
|
|
x0 = pos; |
|
|
x0 &= 15; |
|
|
if ( lrc_obj.orientation != COLOR_WHITE ) |
|
|
x0 ^= 7; |
|
|
|
|
|
y0 = pos; |
|
|
y0>>= 4; |
|
|
if ( lrc_obj.orientation != COLOR_WHITE ) |
|
|
y0 ^= 7; |
|
|
|
|
|
x0 *= chess_boxsize; |
|
|
y0 *= chess_boxsize; |
|
|
|
|
|
u8g2_SetDefaultForegroundColor(lrc_u8g); |
|
|
u8g2_DrawFrame(lrc_u8g, x0, chess_low_edge - y0 - chess_boxsize+1, chess_boxsize, chess_boxsize); |
|
|
|
|
|
|
|
|
if ( is_bold ) |
|
|
{ |
|
|
x0--; |
|
|
y0++; |
|
|
|
|
|
u8g2_DrawFrame(lrc_u8g, x0, chess_low_edge - y0 - chess_boxsize +1, chess_boxsize+2, chess_boxsize+2); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void chess_DrawBoard(void) |
|
|
{ |
|
|
uint8_t i, j, cp; |
|
|
const uint8_t *ptr; /* pointer into CHESS_PROGMEM */ |
|
|
|
|
|
{ |
|
|
uint8_t x_offset = 1; |
|
|
u8g2_SetDefaultForegroundColor(lrc_u8g); |
|
|
for( i = 0; i < 8*8; i+=8 ) |
|
|
{ |
|
|
for( j = 0; j < 8*8; j+=8 ) |
|
|
{ |
|
|
if ( ((i^j) & 8) == 0 ) |
|
|
{ |
|
|
if ( lrc_obj.orientation == COLOR_WHITE ) |
|
|
{ |
|
|
cp = lrc_obj.board[i+j/8]; |
|
|
} |
|
|
else |
|
|
{ |
|
|
cp = lrc_obj.board[(7-i/8)*8+7-j/8]; |
|
|
} |
|
|
if ( cp_GetPiece(cp) == PIECE_NONE ) |
|
|
{ |
|
|
u8g2_DrawPixel(lrc_u8g, j+0+x_offset, chess_low_edge - i-0); |
|
|
u8g2_DrawPixel(lrc_u8g, j+0+x_offset, chess_low_edge - i-2); |
|
|
u8g2_DrawPixel(lrc_u8g, j+0+x_offset, chess_low_edge - i-4); |
|
|
u8g2_DrawPixel(lrc_u8g, j+0+x_offset, chess_low_edge - i-6); |
|
|
u8g2_DrawPixel(lrc_u8g, j+2+x_offset, chess_low_edge - i-0); |
|
|
u8g2_DrawPixel(lrc_u8g, j+2+x_offset, chess_low_edge - i-6); |
|
|
u8g2_DrawPixel(lrc_u8g, j+4+x_offset, chess_low_edge - i-0); |
|
|
u8g2_DrawPixel(lrc_u8g, j+4+x_offset, chess_low_edge - i-6); |
|
|
u8g2_DrawPixel(lrc_u8g, j+6+x_offset, chess_low_edge - i-0); |
|
|
u8g2_DrawPixel(lrc_u8g, j+6+x_offset, chess_low_edge - i-2); |
|
|
u8g2_DrawPixel(lrc_u8g, j+6+x_offset, chess_low_edge - i-4); |
|
|
u8g2_DrawPixel(lrc_u8g, j+6+x_offset, chess_low_edge - i-6); |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
for ( i = 0; i < 8; i++ ) |
|
|
{ |
|
|
for ( j = 0; j < 8; j++ ) |
|
|
{ |
|
|
/* get piece from global board */ |
|
|
if ( lrc_obj.orientation == COLOR_WHITE ) |
|
|
{ |
|
|
cp = lrc_obj.board[i*8+j]; |
|
|
} |
|
|
else |
|
|
{ |
|
|
cp = lrc_obj.board[(7-i)*8+7-j]; |
|
|
} |
|
|
if ( cp_GetPiece(cp) != PIECE_NONE ) |
|
|
{ |
|
|
ptr = chess_black_pieces_bm; |
|
|
ptr += (cp_GetPiece(cp)-1)*8; |
|
|
|
|
|
u8g2_SetDefaultForegroundColor(lrc_u8g); |
|
|
u8g2_DrawBitmap(lrc_u8g, j*chess_boxsize+chess_boxoffset-1, chess_low_edge - (i*chess_boxsize+chess_boxsize-chess_boxoffset), 1, 8, ptr); |
|
|
|
|
|
if ( cp_GetColor(cp) == lrc_obj.strike_out_color ) |
|
|
{ |
|
|
ptr = chess_pieces_body_bm; |
|
|
ptr += (cp_GetPiece(cp)-1)*8; |
|
|
u8g2_SetDefaultBackgroundColor(lrc_u8g); |
|
|
u8g2_DrawBitmap(lrc_u8g, j*chess_boxsize+chess_boxoffset-1, chess_low_edge - (i*chess_boxsize+chess_boxsize-chess_boxoffset), 1, 8, ptr); |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
u8g2_SetDefaultForegroundColor(lrc_u8g); |
|
|
|
|
|
if ( (chess_source_pos & 0x88) == 0 ) |
|
|
{ |
|
|
chess_DrawFrame(chess_source_pos, 1); |
|
|
} |
|
|
|
|
|
if ( (chess_target_pos & 0x88) == 0 ) |
|
|
{ |
|
|
chess_DrawFrame(chess_target_pos, 0); |
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
void chess_Thinking(void) |
|
|
{ |
|
|
} |
|
|
|
|
|
void chess_Init(u8g2_t *u8g, uint8_t body_color) |
|
|
{ |
|
|
lrc_u8g = u8g; |
|
|
|
|
|
chess_low_edge = u8g2_GetDisplayHeight(lrc_u8g); |
|
|
chess_low_edge--; |
|
|
u8g2_SetBitmapMode(u8g, 1); // restore previous behaviour |
|
|
|
|
|
{ |
|
|
|
|
|
chess_boxsize = 8; |
|
|
chess_boxoffset = 1; |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
lrc_obj.strike_out_color = body_color; |
|
|
chess_SetupBoard(); |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void chess_Draw(void) |
|
|
{ |
|
|
if ( chess_state == CHESS_STATE_MENU ) |
|
|
{ |
|
|
if ( lrc_obj.ply_count == 0) |
|
|
mnu_max = 2; |
|
|
else |
|
|
mnu_max = 4; |
|
|
mnu_Draw(); |
|
|
} |
|
|
else |
|
|
{ |
|
|
chess_DrawBoard(); |
|
|
|
|
|
{ |
|
|
uint8_t i; |
|
|
uint8_t entries = lrc_obj.chm_pos; |
|
|
if ( entries > 4 ) |
|
|
entries = 4; |
|
|
|
|
|
u8g2_SetFont(lrc_u8g, u8g2_font_5x7_tr); |
|
|
u8g2_SetDefaultForegroundColor(lrc_u8g); |
|
|
for( i = 0; i < entries; i++ ) |
|
|
{ |
|
|
|
|
|
u8g2_DrawStr(lrc_u8g, u8g2_GetDisplayWidth(lrc_u8g)-35, 8*(i+1), cu_GetHalfMoveStr(lrc_obj.chm_pos-entries+i)); |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
if ( chess_state == CHESS_STATE_SELECT_PIECE ) |
|
|
mnu_DrawHome(chess_source_pos == 255); |
|
|
else if ( chess_state == CHESS_STATE_SELECT_TARGET_POS ) |
|
|
mnu_DrawHome(chess_target_pos == 255); |
|
|
else |
|
|
mnu_DrawHome(0); |
|
|
|
|
|
if ( chess_state == CHESS_STATE_GAME_END ) |
|
|
{ |
|
|
switch( lrc_obj.lost_side_color ) |
|
|
{ |
|
|
case COLOR_WHITE: |
|
|
mnu_DrawEntry(u8g2_GetDisplayHeight(lrc_u8g) / 2-2, "Black wins", 1, 1); |
|
|
break; |
|
|
case COLOR_BLACK: |
|
|
mnu_DrawEntry(u8g2_GetDisplayHeight(lrc_u8g) / 2-2, "White wins", 1, 1); |
|
|
break; |
|
|
default: |
|
|
mnu_DrawEntry(u8g2_GetDisplayHeight(lrc_u8g) / 2-2, "Stalemate", 1, 1); |
|
|
break; |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void chess_Step(uint8_t keycode) |
|
|
{ |
|
|
chess_key_cmd = keycode; |
|
|
|
|
|
/* |
|
|
if ( keycode == CHESS_KEY_NONE ) |
|
|
{ |
|
|
chess_key_cmd = chess_key_code; |
|
|
chess_key_code = CHESS_KEY_NONE; |
|
|
} |
|
|
else |
|
|
{ |
|
|
chess_key_cmd = CHESS_KEY_NONE; |
|
|
chess_key_code = keycode; |
|
|
} |
|
|
*/ |
|
|
//chess_ComputerMove(2); |
|
|
switch(chess_state) |
|
|
{ |
|
|
case CHESS_STATE_MENU: |
|
|
mnu_Step(chess_key_cmd); |
|
|
if ( chess_key_cmd == CHESS_KEY_SELECT ) |
|
|
{ |
|
|
if ( mnu_pos == 0 ) |
|
|
{ |
|
|
chess_SetupBoard(); |
|
|
lrc_obj.orientation = 0; |
|
|
chess_state = CHESS_STATE_SELECT_START; |
|
|
} |
|
|
else if ( mnu_pos == 1 ) |
|
|
{ |
|
|
chess_SetupBoard(); |
|
|
lrc_obj.orientation = 1; |
|
|
chess_state = CHESS_STATE_THINKING; |
|
|
} |
|
|
else if ( mnu_pos == 2 ) |
|
|
{ |
|
|
if ( lrc_obj.ply_count >= 2 ) |
|
|
{ |
|
|
cu_UndoHalfMove(); |
|
|
cu_UndoHalfMove(); |
|
|
lrc_obj.ply_count-=2; |
|
|
if ( lrc_obj.ply_count == 0 ) |
|
|
mnu_pos = 0; |
|
|
} |
|
|
chess_state = CHESS_STATE_SELECT_START; |
|
|
} |
|
|
else if ( mnu_pos == 3 ) |
|
|
{ |
|
|
chess_state = CHESS_STATE_SELECT_START; |
|
|
} |
|
|
} |
|
|
break; |
|
|
case CHESS_STATE_SELECT_START: |
|
|
chess_ClearMarks(); |
|
|
chess_MarkMovable(); |
|
|
chess_source_pos = chess_GetNextMarked(255, 0); |
|
|
chess_target_pos = ILLEGAL_POSITION; |
|
|
chess_state = CHESS_STATE_SELECT_PIECE; |
|
|
break; |
|
|
|
|
|
case CHESS_STATE_SELECT_PIECE: |
|
|
if ( chess_key_cmd == CHESS_KEY_NEXT ) |
|
|
{ |
|
|
chess_source_pos = chess_GetNextMarked(chess_source_pos, lrc_obj.orientation); |
|
|
} |
|
|
else if ( chess_key_cmd == CHESS_KEY_PREV ) |
|
|
{ |
|
|
chess_source_pos = chess_GetNextMarked(chess_source_pos, 1-lrc_obj.orientation ); |
|
|
} |
|
|
else if ( chess_key_cmd == CHESS_KEY_SELECT ) |
|
|
{ |
|
|
if ( chess_source_pos == 255 ) |
|
|
{ |
|
|
chess_state = CHESS_STATE_MENU; |
|
|
} |
|
|
else |
|
|
{ |
|
|
chess_ClearMarks(); |
|
|
chess_MarkTargetMoves(chess_source_pos); |
|
|
chess_target_pos = chess_GetNextMarked(255, 0); |
|
|
chess_state = CHESS_STATE_SELECT_TARGET_POS; |
|
|
} |
|
|
} |
|
|
break; |
|
|
case CHESS_STATE_SELECT_TARGET_POS: |
|
|
if ( chess_key_cmd == CHESS_KEY_NEXT ) |
|
|
{ |
|
|
chess_target_pos = chess_GetNextMarked(chess_target_pos, lrc_obj.orientation); |
|
|
if ( chess_target_pos == 255 ) |
|
|
chess_target_pos = chess_GetNextMarked(chess_target_pos, lrc_obj.orientation); |
|
|
} |
|
|
else if ( chess_key_cmd == CHESS_KEY_PREV ) |
|
|
{ |
|
|
chess_target_pos = chess_GetNextMarked(chess_target_pos, 1-lrc_obj.orientation); |
|
|
if ( chess_target_pos == 255 ) |
|
|
chess_target_pos = chess_GetNextMarked(chess_target_pos, 1-lrc_obj.orientation); |
|
|
} |
|
|
else if ( chess_key_cmd == CHESS_KEY_BACK ) |
|
|
{ |
|
|
chess_ClearMarks(); |
|
|
chess_MarkMovable(); |
|
|
chess_target_pos = ILLEGAL_POSITION; |
|
|
chess_state = CHESS_STATE_SELECT_PIECE; |
|
|
} |
|
|
else if ( chess_key_cmd == CHESS_KEY_SELECT ) |
|
|
{ |
|
|
chess_ManualMove(chess_source_pos, chess_target_pos); |
|
|
if ( lrc_obj.is_game_end != 0 ) |
|
|
chess_state = CHESS_STATE_GAME_END; |
|
|
else |
|
|
chess_state = CHESS_STATE_THINKING; |
|
|
/* clear marks as some kind of feedback to the user... it simply looks better */ |
|
|
chess_source_pos = ILLEGAL_POSITION; |
|
|
chess_target_pos = ILLEGAL_POSITION; |
|
|
chess_ClearMarks(); |
|
|
} |
|
|
break; |
|
|
case CHESS_STATE_THINKING: |
|
|
chess_ComputerMove(2); |
|
|
if ( lrc_obj.is_game_end != 0 ) |
|
|
chess_state = CHESS_STATE_GAME_END; |
|
|
else |
|
|
chess_state = CHESS_STATE_SELECT_START; |
|
|
//chess_state = CHESS_STATE_THINKING |
|
|
break; |
|
|
case CHESS_STATE_GAME_END: |
|
|
if ( chess_key_cmd != CHESS_KEY_NONE ) |
|
|
{ |
|
|
chess_state = CHESS_STATE_MENU; |
|
|
chess_SetupBoard(); |
|
|
} |
|
|
break; |
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
#endif /* UNIX_MAIN */ |
|
|
|
|
|
void loop(void) { |
|
|
static uint8_t keycode = 0; |
|
|
|
|
|
u8g2.firstPage(); |
|
|
do { |
|
|
chess_Draw(); |
|
|
if ( keycode == 0 ) |
|
|
keycode = u8g2.getMenuEvent(); |
|
|
} while ( u8g2.nextPage() ); |
|
|
|
|
|
if ( keycode == 0 ) |
|
|
keycode = u8g2.getMenuEvent(); |
|
|
|
|
|
if ( keycode == U8X8_MSG_GPIO_MENU_DOWN ) |
|
|
keycode = CHESS_KEY_NEXT; |
|
|
if ( keycode == U8X8_MSG_GPIO_MENU_UP ) |
|
|
keycode = CHESS_KEY_PREV; |
|
|
//if ( keycode == U8X8_MSG_GPIO_MENU_HOME ) |
|
|
// keycode = CHESS_KEY_SELECT; |
|
|
|
|
|
chess_Step(keycode); |
|
|
keycode = 0; |
|
|
delay(1); |
|
|
} |
|
|
|
|
|
|