-
Notifications
You must be signed in to change notification settings - Fork 15
/
readme.txt
338 lines (254 loc) · 14.3 KB
/
readme.txt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
0. Updates
* Oct 2023 - Added a version for the Plus4.
* Jul 2020 - Added a version for the Atari.
* May 2020 - Advanced Build instructions at XI.
* May 2020 - I created a version for the Commander X16 (R37).
* May 2020 - I created a graphics version for the C64.
* May 2020 - [raxiss] created a nice Oric-1/Atmos/Telestrat port.
* Jan 2020 - Oliver Schmidt created a beautiful Apple II port. See IX below.
* There are images for all platforms in the releases tab.
I. Introduction (Feb 14, 2014)
I started playing chess about 3 months ago and this got me wondering how
difficult it would be to make a computer chess game. I decided to try and
since it's just a "for the fun of it" project, I decided to make it for the
Commodore 64, still my all-time favorite computer. Using the excellent cc65
tools I could do it all in C and thus make it portable to other systems also.
I learnt that making the game isn't hard, but getting the computer to play
something that resembles a reasonable game is hard. I don't know enough about
chess to get it right, but in this version of today, 14 Feb 2014, the AI is
not very good.
The game was developed on OS X using cc65 and the VICE emulator.
There is a video of the game here: http://youtu.be/bkA4vtwxaJg
II. Use and keys
The colors here refer to the C64 version. The terminal version has a minimal
working display but does try to somewhat match the colors of the C64.
The user controls an on-screen cursor. The cursor changes color to indicate
a state. The colors for selection are:
Green - the piece can be selected
Red - The piece cannot be selected as it doesn't have valid moves
Purple - Empty tile or piece on the other side
Blue - The currently selected piece
Cyan - A valid destination for the currently selected piece
To move the cursor, use the cursor keys. To select a piece, press the RETURN
key while the piece is selected. To deselect the piece, press RETURN on the
same piece again, or press RUN/STOP.
To bring up the menu, press the M key, or the RUN/STOP key when no piece is
selected. Pressing RUN/STOP in a menu backs out of the menu, to the previous
menu or back to the game. Press RETURN to select a menu item and use the up
and down cursor keys to change the selection.
While a side is under human control, there are a few more options. Press B to
toggle on/off a state showing on every tile how many of both black and white's
pieces can attack that tile. Pressing A will toggle a highlight of all of the
pieces on the opposing side that attack the selected tile. Pressing D will
toggle a highlight of all the pieces on the side currently playing's side that
can defend the selected tile. All three of these options basically give a
visual representation of the Attack DB. The colors are: For attackers Cyan
and for defenders Red.
Lastly, the game has an Undo/Redo stack that tracks the last 254 moves. Pressing
U will undo the last move and R will redo the last move. In an AI/Human game,
the undo will undo the last AI and human player move, so the human player can
make a different move.
III. Distribution
This version has code for a C64, using multi-colored text mode, and also code
for a terminal version using the curses library. The terminal version was
only tested on OS X but I suspect it will run under Linux and Windows.
IV. Building from source
a) For the C64 (and other cc65 supported platforms):
Using a properly installed cc65 distribution, the C64 version should build
using make in the folder with the Makefile. I suggest compiling for speed
but optimizing for size does save a bit of memory (1K at present):
make OPTIONS=optspeed
b) For a terminal version:
I built it on OS X using the following command line from the src folder:
cc -I. -lcurses -funsigned-char globals.c undo.c board.c cpu.c human.c \
frontend.c main.c term/platTerm.c -o chess
V. Porting
The code in the src folder should compile on any system (cc65 has type char as
unsigned by default - the char type is almost the only type really used in the
code). A new system will need platform specific implementations of the
functions in plat.h. When I created the terminal port, it literally worked in
under an hour as it was mostly replacing cursor positioning and printing
function names, along with initialization and color management. I had to redo
the log update and timer completely but still, the only file that needed
changes was the platTerm.c copy I made from plat64.c.
In the C64 specific folder under src is a data.c file which contains the
graphics characters to draw the pieces. The layout and chosen bit-pattern of
that data is also explained in that file.
VI. The AI & other thoughts on the code
The game is a fine 2-player chess game, but the computer is not a great chess
player. My approach for the AI is this:
For each piece, calculate a score (see later) for the tile the piece currently
occupies. Then look at all available moves for the piece, and score each
destination tile separately. If the computer is fast, I would now "effect"
each move and run the same algorithm on the opponent, getting a "retaliation
score". Effect the opponent move and run the algorithm again, getting a
subsequent score. The accumulated "score - retaliation score + this side
next score - other side next retaliation score" sum, up to as many levels
deep as desired, would be the final score for that piece and destination.
Since the C64 isn't fast enough for all that, I have it calculate the score
for the piece where it stands and for all destinations. The highest scoring
move, if valid, becomes the move for the piece. The scores for all pieces
are then stack-ranked. Some number of these are then chosen to pursue. I set
it to 16 (gWidth), making it pursues all pieces.
Pursuit of the best moves means doing the depth search for opponent moves and
back to own moves. This is set to go to a level controlled by a variable
named gMaxLevel.
There is another variable, gDeepThoughts, that affects difficulty and speed.
This variable, when set to 1, ensure the moves chosen when evaluating best
moves, are valid. It also, when set, causes the AttackDB to be updated.
Both of these are slow operations. Not doing the work makes things a lot
faster, but obviously less accurate. Especially the further away the
thinking gets from the current, accurate, state.
All three these variables are set from the difficulty selection if there's
an AI player.
Scoring a piece means this: Positive points encourage the piece to move,
negative points discourage making a move.
A) For where the piece stands the score is calculated by looking at:
If this piece is under attack increase chance to move, else decrease
If this piece is being defended, decrease chance to move, else increase
Providing support to a piece on own team, decrease
if supported piece is under attack
if only defender, decrease else increase
if supported piece is more valuable, decrease
not supporting a piece, increase
B) For every destination the piece can move to, score like this:
If this piece will be under attack there, decrease
If it will be defended there, increase else decrease
If a piece is taken at dest, increase
If providing support to a piece on same side from there, increase
if that piece is under attack, increase
if this will be the only defender of that piece, increase
if the supported piece is more valuable than this piece, increase
If attacking a piece on the other side from there, increase
if the attacked piece is more valuable, increase
if the attacked piece has no defenders, increase
The values for increase and decrease aren't always 1. Some situations I
deemed more important so the value may be 2, or the value of the piece itself
for which I use: 1 PAWN, 3 KNIGHT, 3 BISHOP, 5 ROOK, 9 QUEEN, 10 KING, but
modified to 2+(3*value). The +2 compensates for the +/-1's that encourage or
discourage a move, and the 3*value makes the value really meaningful.
There is another scoring opportunity that happens before any other. It is
meant to take a holistic view of the board. Currently, all it does is see
if the king has no moves then all of its neighboring pieces on the same side
are encouraged to move and; it encourages pawns to move so they can get to
promote. The closer they get to the opposite side, the stronger the
encouragement.
I have no real plans to keep working on this project. As stated, I wanted
to see how hard it would be, and now I know. I rushed this V1.0 release so,
sadly, I am sure there will be bugs. There's also lots of room to experiment
with scores, values and relative importance of things like being under attack
vs. supporting another piece.
The code is reasonably clean but I really didn't design this as a game. It
all evolved from the writing of the functions to build an array of valid moves
into a game. The en passant and castling is somewhat hacked in, for example
and may be hard to make sense of.
VII. Credits
The Makefile has the following notice:
###############################################################################
### Generic Makefile for cc65 projects - full version with abstract options ###
### V1.3.0(w) 2010 - 2013 Oliver Schmidt & Patryk "Silver Dream !" Łogiewa ###
###############################################################################
cl65 --version prints the following in my installation:
cl65 V2.13.9 - (C) Copyright 1998-2011 Ullrich von Bassewitz
VIII. Contact
Feel free to send me an email if you have any comments. If you do make a port
or something else, I would love to hear about it!
swessels@email.com
Feb 14, 2014
Thank you!
IX. Apple II Specific Version Information
General display uses the Apple II hires mode accessed via custom asm
functions.
Menu display uses the 4 line bottom text option of the Apple II hires mode via
cc65 CONIO functions.
All hires access is byte aligned, therefore the horizontal resolution is 40
(bytes).
Hires access is done via simple (binary) ROPs (raster operations) by using
actual 6502 (immediate) opcodes.
The C64 and Curses implementation both make heavy use of colors while Apple II
implementation mustn't depend on (but may benefit from) colors. Therefore the
user-operated cursor inverts the border of the current field. It's hard to find
a compromise between making the cursor visible well and showing the piece
"under" the cursor well. Additionally it is desirable to show different cursor
states (empty, invalid, valid). The approach chosen is to have different
thicknesses of the inverted border:
Valid: Thin
Invalid: Medium
Empty: Thick
When it comes to showing attackers/defenders (via the keys A / D) there's no
alternative to resorting to colors:
Attackers: Red
Defenders: Green
So the only field display variant left is the piece selected for moving. Instead
of introducing a third type of highlighting (beside border inversion and
coloring) it is simply colored Magenta. The reasoning:
On a monochrome display the user won't have much fun showing attackers/defenders
anyway. And without showing those the selected piece is the only colored (aka
striped) piece making it clearly visible.
On a color display a third color (beside attackers/defenders) works just fine.
As the Apple II doesn't have cursor-up and cursor-down keys the keys O and L
work as alternatives to the those cursor keys.
There's a video showing the Apple II version here: https://youtu.be/PPy-cg4ghDY
Oliver Schmidt
Jan 19, 2020
XI. Build Instructions
All of the 8-Bit versions of cc65 Chess can be built using make.
I recommend the game be built for speed on almost all targets, which also
results in smaller file and is essential for all targets. This is done by
using the OPTIONS=optspeed command line to make. For the Atari, it is
essential to use OPTIONS=optsize as the 48K Atari really needs the extra
1K of memory.
When you type make (using GNU Make) the default behaviour is to make all of the
versions. Currently, that means the following (cc65 target name in brackets):
* Commodore 64 HiRes (c64)
* Commodore 64 Multicolor Text (c64.chr)
* Apple 2 (apple2)
* Oric-1/Atmos/Telestrat (atmos)
* Commander X16 (cx16)
* Atari (atari - Needs at least 48K, tested on Atari 800 48K)
Most platforms have an additional step that can be performed, which is to make a
program (prg), disk (dsk, atr) or tape (tap) file. Do make again, but with dsk
(Apple 2 dsk), atr (Atari disk), tap (Oric tape), prg (C64 prg), cprg (c64.chr
prg) or cxprg (cX16 prg) on the command line.
The two steps can be combined into a single make command, by using "all" as the
first target, i.e:
make OPTIONS=optsize all dsk atr tap prg cprg cxprg
Making a terminal version (using curses) - See IV (b) above.
Examples:
1) Make everything, and then make the images for all platforms.
make OPTIONS=optsize
This will make the following files:
cc65-Chess.apple2
cc65-Chess.atari
cc65-Chess.atmos
cc65-Chess.c64
cc65-Chess.c64.chr
cc65-Chess.cx16
make dsk atr tap prg cprg cxprg
This will make the following files:
cc65-Chess.tap
cc65-Chess.atr
cc65-Chess.dsk
cc65-Chess-c64.prg
cc65-Chess-chr.prg
cc65-Chess-cx16.prg
Once you have used the OPTIONS=optsize (or OPTIONS=optspeed) on the
command-line, you do not have to use it again since the options are
saved in a file called Makefile.options.
2) Build just one version (let's say the Oric)
make OPTIONS=optsize TARGETS=atmos tap
This will create a ready to run TAP file named cc65-Chess.tap
3) You can also start an emulator directly from make with the test command-line.
make OPTIONS=optsize atmos test
This last command example is a good way of callimg make to build and test any of
the targets by itself, provided you have configured an emulator in the Makefile.
In this case, it will call the emulator with cc65-Chess.atmos but Oricutron
doesn't mind that the exytension isn't .tap.
Look for _EMUCMD in the Makefile. You may have to specify a full path to the
emulator, and in some cases you may need to change the test: command itself. For
example, to run AppleWin I removed the $< from $(EMUCMD) $< in the test:
section, because AppleWin did not like the extra (cc64-Chess.apple2) file being
passed, and I had to give it the full path to cc64-Chess.dsk as part of apple2_EMUCMD.
Lastly - the Atari, CX16 and C64 versions use the same piece defenitions that Oliver
Schmidt added for the Apple II, and kindly agreed to let me use for these
versions as well. See genPieces.cpp in the specific src folder for more details.