-
Notifications
You must be signed in to change notification settings - Fork 17
/
cheapglk.h
237 lines (190 loc) · 7.68 KB
/
cheapglk.h
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
#ifndef CHEAPGLK_H
#define CHEAPGLK_H
/* cheapglk.h: Private header file for Cheapass Implementation of the
Glk API.
CheapGlk Library: version 1.0.6.
Glk API which this implements: version 0.7.5.
Designed by Andrew Plotkin <[email protected]>
http://eblong.com/zarf/glk/index.html
*/
#define LIBRARY_VERSION "1.0.6"
#include "gi_dispa.h"
#include "gi_debug.h"
/* First, we define our own TRUE and FALSE and NULL, because ANSI
is a strange world. */
#ifndef TRUE
#define TRUE 1
#endif
#ifndef FALSE
#define FALSE 0
#endif
#ifndef NULL
#define NULL 0
#endif
/* This macro is called whenever the library code catches an error
or illegal operation from the game program. */
#define gli_strict_warning(msg) \
(printf("Glk library error: %s\n", msg))
/* The overall screen size, as set by command-line options. A
better implementation would check the real screen size
somehow. */
extern int gli_screenwidth, gli_screenheight;
/* Should we assume that the terminal (or whatever is handling our
stdin/stdout) is expecting UTF-8 encoding? Normally input and output
will be the same, but they don't have to be. */
extern int gli_utf8output, gli_utf8input;
#if GIDEBUG_LIBRARY_SUPPORT
/* Has the user requested debug support? */
extern int gli_debugger;
#else /* GIDEBUG_LIBRARY_SUPPORT */
#define gli_debugger (0)
#endif /* GIDEBUG_LIBRARY_SUPPORT */
/* Get the pathname for data chunk num (as specified in command-line arguments,
if any).
*/
extern int gli_get_dataresource_info(int num, void **ptr, glui32 *len, int *isbinary);
/* Callbacks necessary for the dispatch layer. */
extern gidispatch_rock_t (*gli_register_obj)(void *obj, glui32 objclass);
extern void (*gli_unregister_obj)(void *obj, glui32 objclass,
gidispatch_rock_t objrock);
extern gidispatch_rock_t (*gli_register_arr)(void *array, glui32 len,
char *typecode);
extern void (*gli_unregister_arr)(void *array, glui32 len, char *typecode,
gidispatch_rock_t objrock);
/* Some useful type declarations. */
typedef struct glk_window_struct window_t;
typedef struct glk_stream_struct stream_t;
typedef struct glk_fileref_struct fileref_t;
#define MAGIC_WINDOW_NUM (9876)
#define MAGIC_STREAM_NUM (8769)
#define MAGIC_FILEREF_NUM (7698)
struct glk_window_struct {
glui32 magicnum;
glui32 rock;
gidispatch_rock_t disprock;
stream_t *str; /* the window stream. */
stream_t *echostr; /* the window's echo stream, if any. */
int line_request;
int line_request_uni;
int char_request;
int char_request_uni;
void *linebuf;
glui32 linebuflen;
gidispatch_rock_t inarrayrock;
};
#define strtype_File (1)
#define strtype_Window (2)
#define strtype_Memory (3)
#define strtype_Resource (4)
struct glk_stream_struct {
glui32 magicnum;
glui32 rock;
int type; /* file, window, or memory stream */
int unicode; /* one-byte or four-byte chars? Not meaningful for windows */
glui32 readcount, writecount;
int readable, writable;
/* for strtype_Window */
window_t *win;
/* for strtype_File */
FILE *file;
glui32 lastop; /* 0, filemode_Write, or filemode_Read */
/* for strtype_Resource */
int isbinary;
/* for strtype_Memory and strtype_Resource. Separate pointers for
one-byte and four-byte streams */
unsigned char *buf;
unsigned char *bufptr;
unsigned char *bufend;
unsigned char *bufeof;
glui32 *ubuf;
glui32 *ubufptr;
glui32 *ubufend;
glui32 *ubufeof;
glui32 buflen;
gidispatch_rock_t arrayrock;
gidispatch_rock_t disprock;
stream_t *next, *prev; /* in the big linked list of streams */
};
struct glk_fileref_struct {
glui32 magicnum;
glui32 rock;
char *filename;
int filetype;
int textmode;
gidispatch_rock_t disprock;
fileref_t *next, *prev; /* in the big linked list of filerefs */
};
typedef glui32 gli_case_block_t[2]; /* upper, lower */
/* If both are 0xFFFFFFFF, you have to look at the special-case table. */
typedef glui32 gli_case_special_t[3]; /* upper, lower, title */
/* Each of these points to a subarray of the unigen_special_array
(in cgunicode.c). In that subarray, element zero is the length,
and that's followed by length unicode values. */
typedef glui32 gli_decomp_block_t[2]; /* count, position */
/* The position points to a subarray of the unigen_decomp_array.
If the count is zero, there is no decomposition. */
/* Declarations of library internal functions. */
extern void gli_initialize_misc(void);
extern window_t *gli_new_window(glui32 rock);
extern void gli_delete_window(window_t *win);
extern window_t *gli_window_get(void);
extern stream_t *gli_new_stream(int type, int readable, int writable,
glui32 rock);
extern void gli_delete_stream(stream_t *str);
extern strid_t gli_stream_open_pathname(char *pathname, int writemode,
int textmode, glui32 rock);
extern void gli_stream_set_current(stream_t *str);
extern void gli_stream_fill_result(stream_t *str,
stream_result_t *result);
extern void gli_stream_echo_line(stream_t *str, char *buf, glui32 len);
extern void gli_stream_echo_line_uni(stream_t *str, glui32 *buf, glui32 len);
extern fileref_t *gli_new_fileref(char *filename, glui32 usage,
glui32 rock);
extern void gli_delete_fileref(fileref_t *fref);
extern void gli_putchar_utf8(glui32 val, FILE *fl);
extern int gli_encode_utf8(glui32 val, char *buf, int len);
extern glui32 gli_parse_utf8(unsigned char *buf, glui32 buflen,
glui32 *out, glui32 outlen);
/* A macro that I can't think of anywhere else to put it. */
#define gli_event_clearevent(evp) \
((evp)->type = evtype_None, \
(evp)->win = NULL, \
(evp)->val1 = 0, \
(evp)->val2 = 0)
/* A macro which reads and decodes one character of UTF-8. Needs no
explanation, I'm sure.
Oh, okay. The character will be written to *chptr (so pass in "&ch",
where ch is a glui32 variable). eofcond should be a condition to
evaluate end-of-stream -- true if no more characters are readable.
nextch is a function which reads the next character; this is invoked
exactly as many times as necessary.
val0, val1, val2, val3 should be glui32 scratch variables. The macro
needs these. Just define them, you don't need to pay attention to them
otherwise.
The macro itself evaluates to true if ch was successfully set, or
false if something went wrong. (Not enough characters, or an
invalid byte sequence.)
This is not the worst macro I've ever written, but I forget what the
other one was.
*/
#define UTF8_DECODE_INLINE(chptr, eofcond, nextch, val0, val1, val2, val3) ( \
(eofcond ? 0 : ( \
(((val0=nextch) < 0x80) ? (*chptr=val0, 1) : ( \
(eofcond ? 0 : ( \
(((val1=nextch) & 0xC0) != 0x80) ? 0 : ( \
(((val0 & 0xE0) == 0xC0) ? (*chptr=((val0 & 0x1F) << 6) | (val1 & 0x3F), 1) : ( \
(eofcond ? 0 : ( \
(((val2=nextch) & 0xC0) != 0x80) ? 0 : ( \
(((val0 & 0xF0) == 0xE0) ? (*chptr=(((val0 & 0xF)<<12) & 0x0000F000) | (((val1 & 0x3F)<<6) & 0x00000FC0) | (((val2 & 0x3F)) & 0x0000003F), 1) : ( \
(((val0 & 0xF0) != 0xF0 || eofcond) ? 0 : (\
(((val3=nextch) & 0xC0) != 0x80) ? 0 : (*chptr=(((val0 & 0x7)<<18) & 0x1C0000) | (((val1 & 0x3F)<<12) & 0x03F000) | (((val2 & 0x3F)<<6) & 0x000FC0) | (((val3 & 0x3F)) & 0x00003F), 1) \
)) \
)) \
)) \
)) \
)) \
)) \
)) \
)) \
)
#endif /* CHEAPGLK_H */