Sun, 07 Jun 2026 16:56:22 +0200
extracts FEN code into a separate compilation unit
relates to #842
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
1 | /* |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
2 | * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
3 | * |
|
55
54ea19938d57
updated copyright and version info
Mike Becker <universe@uap-core.de>
parents:
54
diff
changeset
|
4 | * Copyright 2016 Mike Becker. All rights reserved. |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
5 | * |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
6 | * Redistribution and use in source and binary forms, with or without |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
7 | * modification, are permitted provided that the following conditions are met: |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
8 | * |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
9 | * 1. Redistributions of source code must retain the above copyright |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
10 | * notice, this list of conditions and the following disclaimer. |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
11 | * |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
12 | * 2. Redistributions in binary form must reproduce the above copyright |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
13 | * notice, this list of conditions and the following disclaimer in the |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
14 | * documentation and/or other materials provided with the distribution. |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
15 | * |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
16 | * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
17 | * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
18 | * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
19 | * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
20 | * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
21 | * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
22 | * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
23 | * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
24 | * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
25 | * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
26 | * POSSIBILITY OF SUCH DAMAGE. |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
27 | * |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
28 | */ |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
29 | |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
30 | #include "pgn.h" |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
31 | |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
32 | #include <stdlib.h> |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
33 | #include <ctype.h> |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
34 | #include <string.h> |
|
120
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
35 | #include <sys/time.h> |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
36 | #include <time.h> |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
37 | |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
38 | enum { |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
39 | pgn_error_missing_quote = 1, |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
40 | pgn_error_missing_bracket, |
|
70
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
41 | pgn_error_missing_brace, |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
42 | pgn_error_missing_dot, |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
43 | pgn_error_move_syntax, |
|
126
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
44 | pgn_error_move_semantics, |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
45 | pgn_error_result_syntax, |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
46 | }; |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
47 | |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
48 | static const char* pgn_error_strings[] = { |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
49 | "No Error.", |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
50 | "Tag values must be enclosed in double-quotes.", |
|
70
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
51 | "Missing closing brace '}' for comment.", |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
52 | "Tags must be enclosed in square brackets: '[Key \"Value\"]'.", |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
53 | "Move numbers must be terminated with a dot (e.g. '13.' - not '13').", |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
54 | "Move is syntactically incorrect.", |
|
126
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
55 | "Move is not valid according to chess rules.", |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
56 | "Result syntax is incorrect. Expected 1-0, 0-1, 1/2-1/2, or *.", |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
57 | }; |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
58 | |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
59 | const char* pgn_error_str(int code) { |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
60 | return pgn_error_strings[code]; |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
61 | } |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
62 | |
|
130
3fc6b1d6cbe9
implement optional delay - resolves #820
Mike Becker <universe@uap-core.de>
parents:
127
diff
changeset
|
63 | int read_pgn(FILE* stream, GameState *gamestate) { |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
64 | int c, i; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
65 | |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
66 | char result[8]; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
67 | |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
68 | char tagkey[32]; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
69 | char tagvalue[128]; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
70 | |
|
64
4eda5df55f86
fixes castling not printed correctly to PGN
Mike Becker <universe@uap-core.de>
parents:
60
diff
changeset
|
71 | /* read tag pairs */ |
|
82
e75865d90111
remove unnecessary loop variable
Mike Becker <universe@uap-core.de>
parents:
81
diff
changeset
|
72 | while (true) { |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
73 | while (isspace(c = fgetc(stream))); |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
74 | if (c == '1') { |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
75 | break; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
76 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
77 | if (c != '[') { |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
78 | return pgn_error_missing_bracket; |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
79 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
80 | while (isspace(c = fgetc(stream))); |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
81 | i = 0; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
82 | do { |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
83 | tagkey[i++] = c; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
84 | } while (!isspace(c = fgetc(stream))); |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
85 | tagkey[i] = '\0'; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
86 | while (isspace(c = fgetc(stream))); |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
87 | if (c != '"') { |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
88 | return pgn_error_missing_quote; |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
89 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
90 | i = 0; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
91 | while ((c = fgetc(stream)) != '"') { |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
92 | if (c == '\n' || c == EOF) { |
|
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
93 | return pgn_error_missing_quote; |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
94 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
95 | tagvalue[i++] = c; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
96 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
97 | tagvalue[i] = '\0'; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
98 | if (fgetc(stream) != ']') { |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
99 | return pgn_error_missing_bracket; |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
100 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
101 | |
|
126
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
102 | // TODO: read clock info |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
103 | if (strcmp("Result", tagkey) == 0) { |
|
126
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
104 | // TODO: why are we parsing this? it is never used! |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
105 | memcpy(result, tagvalue, 8); |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
106 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
107 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
108 | |
|
64
4eda5df55f86
fixes castling not printed correctly to PGN
Mike Becker <universe@uap-core.de>
parents:
60
diff
changeset
|
109 | /* read moves */ |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
110 | if (fgetc(stream) != '.') { |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
111 | return pgn_error_missing_dot; |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
112 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
113 | |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
114 | char movestr[10]; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
115 | Move move; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
116 | uint8_t curcol = WHITE; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
117 | |
|
82
e75865d90111
remove unnecessary loop variable
Mike Becker <universe@uap-core.de>
parents:
81
diff
changeset
|
118 | while (true) { |
|
64
4eda5df55f86
fixes castling not printed correctly to PGN
Mike Becker <universe@uap-core.de>
parents:
60
diff
changeset
|
119 | /* move */ |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
120 | while (isspace(c = fgetc(stream))); |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
121 | i = 0; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
122 | do { |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
123 | movestr[i++] = c; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
124 | if (i >= 10) { |
|
81
82d3b044aa69
fix incorrect error value when a move string is too long in a PGN
Mike Becker <universe@uap-core.de>
parents:
80
diff
changeset
|
125 | return pgn_error_move_syntax; |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
126 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
127 | } while (!isspace(c = fgetc(stream))); |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
128 | movestr[i] = '\0'; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
129 | if (eval_move(gamestate, movestr, &move, curcol) |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
130 | != VALID_MOVE_SYNTAX) { |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
131 | return pgn_error_move_syntax; |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
132 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
133 | if (validate_move(gamestate, &move) != VALID_MOVE_SEMANTICS) { |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
134 | return pgn_error_move_semantics; |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
135 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
136 | apply_move(gamestate, &move); |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
137 | |
|
70
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
138 | /* skip spaces */ |
|
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
139 | while (isspace(c = fgetc(stream))); |
|
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
140 | |
|
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
141 | /* parse possible comment */ |
|
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
142 | if (c == '{') { |
|
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
143 | // TODO: interpret comment (may contain clock info) |
|
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
144 | do { |
|
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
145 | c = fgetc(stream); |
|
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
146 | } while (c != '}' && c != EOF); |
|
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
147 | if (c == EOF) { |
|
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
148 | return pgn_error_missing_brace; |
|
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
149 | } |
| 88 | 150 | /* skip spaces */ |
| 151 | while (isspace(c = fgetc(stream))); | |
|
70
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
152 | } |
|
5427beba96d1
pgn parser can now handle comments (although it ignores them for now)
Mike Becker <universe@uap-core.de>
parents:
65
diff
changeset
|
153 | |
|
64
4eda5df55f86
fixes castling not printed correctly to PGN
Mike Becker <universe@uap-core.de>
parents:
60
diff
changeset
|
154 | /* end of game data encountered */ |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
155 | if (c == EOF) { |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
156 | break; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
157 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
158 | if (c == '1' || c == '0') { |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
159 | c = fgetc(stream); |
|
126
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
160 | // TODO: #842 - allow games to be continued if possible |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
161 | if (c == '-') { |
|
126
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
162 | if (!gamestate->checkmate) { |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
163 | // TODO: maybe we should not parse this here; |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
164 | // there is an unused result string from the STR |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
165 | c = fgetc(stream); |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
166 | if (c == '1') { |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
167 | gamestate->wresign = true; |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
168 | } else if (c == '0') { |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
169 | gamestate->bresign = true; |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
170 | } else { |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
171 | return pgn_error_result_syntax; |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
172 | } |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
173 | } |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
174 | break; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
175 | } else if (c == '/') { |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
176 | gamestate->remis = !gamestate->stalemate; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
177 | break; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
178 | } else { |
|
64
4eda5df55f86
fixes castling not printed correctly to PGN
Mike Becker <universe@uap-core.de>
parents:
60
diff
changeset
|
179 | /* oops, it was a move number, go back! */ |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
180 | fseek(stream, -1, SEEK_CUR); |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
181 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
182 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
183 | |
|
64
4eda5df55f86
fixes castling not printed correctly to PGN
Mike Becker <universe@uap-core.de>
parents:
60
diff
changeset
|
184 | /* we have eaten the next valuable byte, so go back */ |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
185 | fseek(stream, -1, SEEK_CUR); |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
186 | |
|
64
4eda5df55f86
fixes castling not printed correctly to PGN
Mike Becker <universe@uap-core.de>
parents:
60
diff
changeset
|
187 | /* skip move number after black move */ |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
188 | if (curcol == BLACK) { |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
189 | while (isdigit(c = fgetc(stream))); |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
190 | if (c != '.') { |
|
60
0c50aac49e55
adds error messages to PGN reader
Mike Becker <universe@uap-core.de>
parents:
59
diff
changeset
|
191 | return pgn_error_missing_dot; |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
192 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
193 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
194 | curcol = opponent_color(curcol); |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
195 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
196 | |
|
59
3fa1de896666
fixes inappropriate use of EXIT_ macros + adds a sample PGN file
Mike Becker <universe@uap-core.de>
parents:
55
diff
changeset
|
197 | return 0; |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
198 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
199 | |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
200 | static void pgn_insert_newlines(char *block) { |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
201 | size_t pos = 0; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
202 | size_t last_space_pos = 0; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
203 | size_t line_len = 0; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
204 | while (block[pos] != '\0') { |
|
118
0f2ffbc408ce
prettier line breaking in exported PGNs
Mike Becker <universe@uap-core.de>
parents:
117
diff
changeset
|
205 | if (block[pos] == 0x1f) { |
|
0f2ffbc408ce
prettier line breaking in exported PGNs
Mike Becker <universe@uap-core.de>
parents:
117
diff
changeset
|
206 | block[pos++] = ' '; |
|
0f2ffbc408ce
prettier line breaking in exported PGNs
Mike Becker <universe@uap-core.de>
parents:
117
diff
changeset
|
207 | continue; |
|
0f2ffbc408ce
prettier line breaking in exported PGNs
Mike Becker <universe@uap-core.de>
parents:
117
diff
changeset
|
208 | } |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
209 | if (block[pos] == ' ') { |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
210 | last_space_pos = pos; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
211 | } |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
212 | line_len++; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
213 | if (line_len > 80) { |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
214 | block[last_space_pos] = '\n'; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
215 | line_len = pos - last_space_pos; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
216 | } |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
217 | pos++; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
218 | } |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
219 | } |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
220 | |
|
120
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
221 | static const char *pgn_date(GameState *gamestate) { |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
222 | static char date[16]; |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
223 | time_t dateseconds; |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
224 | if (gamestate->movecount == 0) { |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
225 | struct timeval curtimestamp; |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
226 | gettimeofday(&curtimestamp, NULL); |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
227 | dateseconds = curtimestamp.tv_sec; |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
228 | } else { |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
229 | dateseconds = (time_t) gamestate->moves[0].timestamp.tv_sec; |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
230 | } |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
231 | struct tm *tm = gmtime(&dateseconds); |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
232 | if (tm == NULL) { |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
233 | /* fallback - should never happen in reality */ |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
234 | strncpy(date, "????.??.??", sizeof(date)); |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
235 | } else { |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
236 | int year = tm->tm_year + 1900; |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
237 | int month = tm->tm_mon + 1; |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
238 | int day = tm->tm_mday; |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
239 | #ifdef __GNUC__ |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
240 | /* this is required to tell GCC that it does not need to warn |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
241 | * about the small buffer size of the date string. |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
242 | */ |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
243 | if (year < 1900 || year > 9999) __builtin_unreachable(); |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
244 | if (month < 1 || month > 12) __builtin_unreachable(); |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
245 | if (day < 1 || day > 31) __builtin_unreachable(); |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
246 | #endif |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
247 | snprintf(date, sizeof(date), "%04d.%02d.%02d", |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
248 | year, month, day); |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
249 | } |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
250 | return date; |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
251 | } |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
252 | |
|
127
7258a46bf5ae
ask for player names during PGN export
Mike Becker <universe@uap-core.de>
parents:
126
diff
changeset
|
253 | const char *pgn_player_name(GameState *gamestate, uint8_t color) { |
|
7258a46bf5ae
ask for player names during PGN export
Mike Becker <universe@uap-core.de>
parents:
126
diff
changeset
|
254 | const char *name = color == WHITE ? gamestate->wname : gamestate->bname; |
|
7258a46bf5ae
ask for player names during PGN export
Mike Becker <universe@uap-core.de>
parents:
126
diff
changeset
|
255 | return name[0] != '\0' ? name : "Anonymous"; |
|
7258a46bf5ae
ask for player names during PGN export
Mike Becker <universe@uap-core.de>
parents:
126
diff
changeset
|
256 | } |
|
7258a46bf5ae
ask for player names during PGN export
Mike Becker <universe@uap-core.de>
parents:
126
diff
changeset
|
257 | |
|
130
3fc6b1d6cbe9
implement optional delay - resolves #820
Mike Becker <universe@uap-core.de>
parents:
127
diff
changeset
|
258 | void write_pgn(FILE* stream, GameState *gamestate, bool export_comments) { |
|
120
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
259 | /* STR tag pairs */ |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
260 | fprintf(stream, "[Event \"%s\"]\n", "terminal-chess game"); |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
261 | fprintf(stream, "[Site \"%s\"]\n", "Somewhere on Earth"); |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
262 | fprintf(stream, "[Date \"%s\"]\n", pgn_date(gamestate)); |
|
57577c9af5d3
add remaining missing STR tags to PGN exports
Mike Becker <universe@uap-core.de>
parents:
118
diff
changeset
|
263 | fprintf(stream, "[Round \"%s\"]\n", "-"); |
|
127
7258a46bf5ae
ask for player names during PGN export
Mike Becker <universe@uap-core.de>
parents:
126
diff
changeset
|
264 | fprintf(stream, "[White \"%s\"]\n", pgn_player_name(gamestate, WHITE)); |
|
7258a46bf5ae
ask for player names during PGN export
Mike Becker <universe@uap-core.de>
parents:
126
diff
changeset
|
265 | fprintf(stream, "[Black \"%s\"]\n", pgn_player_name(gamestate, BLACK)); |
|
102
463c648e6a9b
change return type of write_pgn to void
Mike Becker <universe@uap-core.de>
parents:
101
diff
changeset
|
266 | |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
267 | char *result; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
268 | if (gamestate->stalemate || gamestate->remis) { |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
269 | result = "1/2-1/2"; |
|
126
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
270 | } else if (gamestate->wresign) { |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
271 | result = "0-1"; |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
272 | } else if (gamestate->bresign) { |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
273 | result = "1-0"; |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
274 | } else if (gamestate->checkmate) { |
|
98
9cb41383540f
change move list from linked list to array (prepares game replays)
Mike Becker <universe@uap-core.de>
parents:
88
diff
changeset
|
275 | if (gamestate->movecount > 0) { |
|
9cb41383540f
change move list from linked list to array (prepares game replays)
Mike Becker <universe@uap-core.de>
parents:
88
diff
changeset
|
276 | result = (last_move(gamestate).piece & COLOR_MASK) == WHITE ? |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
277 | "1-0" : "0-1"; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
278 | } else { |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
279 | result = "0-1"; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
280 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
281 | } else { |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
282 | result = "*"; |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
283 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
284 | fprintf(stream, "[Result \"%s\"]\n\n", result); |
|
126
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
285 | |
|
d58b2abdd330
fix wrong game state when the waiting player resigns
Mike Becker <universe@uap-core.de>
parents:
121
diff
changeset
|
286 | // TODO: add optional clock info |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
287 | |
|
64
4eda5df55f86
fixes castling not printed correctly to PGN
Mike Becker <universe@uap-core.de>
parents:
60
diff
changeset
|
288 | /* moves */ |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
289 | size_t moveblkcap = 4096; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
290 | char *moveblk = malloc(moveblkcap); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
291 | char *moveblkptr = moveblk; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
292 | if (moveblk == NULL) { |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
293 | // TODO: error handling (for the entire function actually) |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
294 | abort(); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
295 | } |
|
98
9cb41383540f
change move list from linked list to array (prepares game replays)
Mike Becker <universe@uap-core.de>
parents:
88
diff
changeset
|
296 | for (unsigned i = 0 ; i < gamestate->movecount ; i++) { |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
297 | /* reallocate move block buffer if needed */ |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
298 | { |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
299 | size_t moveblksize = moveblkptr - moveblk; |
|
121
53f714ac783d
fix critical OOM bug in PGN export
Mike Becker <universe@uap-core.de>
parents:
120
diff
changeset
|
300 | if (moveblksize + 128 > moveblkcap) { |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
301 | moveblkcap *= 2; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
302 | char *newmoveblk = realloc(moveblk, moveblkcap); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
303 | if (newmoveblk == NULL) { |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
304 | free(moveblk); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
305 | abort(); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
306 | } |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
307 | moveblk = newmoveblk; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
308 | moveblkptr = moveblk + moveblksize; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
309 | } |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
310 | } |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
311 | |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
312 | int snpr; /* return value of printf calls */ |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
313 | |
|
98
9cb41383540f
change move list from linked list to array (prepares game replays)
Mike Becker <universe@uap-core.de>
parents:
88
diff
changeset
|
314 | if (i % 2 == 0) { |
|
118
0f2ffbc408ce
prettier line breaking in exported PGNs
Mike Becker <universe@uap-core.de>
parents:
117
diff
changeset
|
315 | snpr = snprintf(moveblkptr, 16, "%d.\x1f%s", |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
316 | 1+i/2, gamestate->moves[i].string); |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
317 | } else { |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
318 | snpr = snprintf(moveblkptr, 16, "%s", |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
319 | gamestate->moves[i].string); |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
320 | } |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
321 | moveblkptr += snpr; |
|
101
d4f19182fdc2
add clock times to PGN output
Mike Becker <universe@uap-core.de>
parents:
98
diff
changeset
|
322 | |
|
117
ee539a9691f0
add prompt whether PGN should be exported with comments
Mike Becker <universe@uap-core.de>
parents:
107
diff
changeset
|
323 | if (!export_comments) { |
|
ee539a9691f0
add prompt whether PGN should be exported with comments
Mike Becker <universe@uap-core.de>
parents:
107
diff
changeset
|
324 | *(moveblkptr++) = ' '; |
|
ee539a9691f0
add prompt whether PGN should be exported with comments
Mike Becker <universe@uap-core.de>
parents:
107
diff
changeset
|
325 | continue; |
|
ee539a9691f0
add prompt whether PGN should be exported with comments
Mike Becker <universe@uap-core.de>
parents:
107
diff
changeset
|
326 | } |
|
ee539a9691f0
add prompt whether PGN should be exported with comments
Mike Becker <universe@uap-core.de>
parents:
107
diff
changeset
|
327 | |
|
101
d4f19182fdc2
add clock times to PGN output
Mike Becker <universe@uap-core.de>
parents:
98
diff
changeset
|
328 | /* add clock times when the game was under time control */ |
|
130
3fc6b1d6cbe9
implement optional delay - resolves #820
Mike Becker <universe@uap-core.de>
parents:
127
diff
changeset
|
329 | if (gamestate->info.timecontrol) { |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
330 | memcpy(moveblkptr, " {[%clk ", 8); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
331 | moveblkptr += 8; |
|
101
d4f19182fdc2
add clock times to PGN output
Mike Becker <universe@uap-core.de>
parents:
98
diff
changeset
|
332 | unsigned clkmv = i + 2; /* time for the next move! */ |
|
130
3fc6b1d6cbe9
implement optional delay - resolves #820
Mike Becker <universe@uap-core.de>
parents:
127
diff
changeset
|
333 | uint16_t clk = remaining_movetime2(gamestate, clkmv); |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
334 | snpr = print_clk(clk, moveblkptr, true); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
335 | moveblkptr += snpr; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
336 | *(moveblkptr++) = ']'; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
337 | *(moveblkptr++) = '}'; |
|
106
bdc9528d3e2b
add elapsed move time commands to PGN exports
Mike Becker <universe@uap-core.de>
parents:
102
diff
changeset
|
338 | |
|
bdc9528d3e2b
add elapsed move time commands to PGN exports
Mike Becker <universe@uap-core.de>
parents:
102
diff
changeset
|
339 | /* elapsed move time */ |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
340 | memcpy(moveblkptr, " {[%emt ", 8); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
341 | moveblkptr += 8; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
342 | uint16_t emt = gamestate->moves[i].movetime.tv_sec; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
343 | snpr = print_clk(emt, moveblkptr, true); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
344 | moveblkptr += snpr; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
345 | *(moveblkptr++) = ']'; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
346 | *(moveblkptr++) = '}'; |
|
101
d4f19182fdc2
add clock times to PGN output
Mike Becker <universe@uap-core.de>
parents:
98
diff
changeset
|
347 | } |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
348 | |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
349 | *(moveblkptr++) = ' '; |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
350 | } |
|
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
351 | |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
352 | if (result[0] != '*') { |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
353 | size_t rlen = strlen(result); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
354 | memcpy(moveblkptr, result, rlen); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
355 | moveblkptr += rlen; |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
356 | } |
|
107
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
357 | *(moveblkptr++) = '\n'; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
358 | *moveblkptr = 0; |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
359 | |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
360 | pgn_insert_newlines(moveblk); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
361 | fputs(moveblk, stream); |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
362 | |
|
36dd94278142
fix that PGN (with comments) can exceed 80 chars
Mike Becker <universe@uap-core.de>
parents:
106
diff
changeset
|
363 | free(moveblk); |
|
50
41017d0a72c5
added pgn parser and writer (without comment support yet) + minor refactorings
Mike Becker <universe@uap-core.de>
parents:
diff
changeset
|
364 | } |