-
Notifications
You must be signed in to change notification settings - Fork 68
/
Copy pathplugin.c
324 lines (292 loc) · 7.55 KB
/
plugin.c
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
/*
* Copyright (c) 2009 Michael Steil
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. 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 AUTHOR 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 AUTHOR 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.
*/
/*
* This plugin interface makes use of the standard plugin facility built into
* Commodore BASIC that is used by BASIC extensions such as Simons' BASIC.
* There are several vectors at 0x0300 in RAM for functions like error printing,
* tokenizing, de-tokenizing and the interpreter loop. We hook this from C.
* Since this adds code to the interpreter loop, it is disabled by default,
* and can be enabled like this:
*
* SYS 1
*
* It can be disabled with:
*
* SYS 0
*
* Please note that the current implementation does not tokenize new keywords,
* but stores them verbatim and compares strings when during execution, which
* is very bad for performance. Also, there is currently no demo code for
* added functions.
*/
#include <stdio.h>
#include <stdlib.h>
#ifdef _WIN32
#include <windows.h>
#undef ERROR_FILE_NOT_FOUND /* avoid conflict with CBM value below */
#endif
#include "plugin.h"
#include "glue.h"
#include "console.h"
static unsigned short
get_chrptr() {
return RAM[0x7A] | RAM[0x7B]<<8;
}
static void
set_chrptr(unsigned short a) {
RAM[0x7A] = a & 0xFF;
RAM[0x7B] = a >> 8;
}
int
compare(const char *s1) {
const unsigned char *s = (const unsigned char *)s1;
unsigned short chrptr = get_chrptr();
while (*s) {
CHRGET();
if (A != *s++) {
set_chrptr(chrptr);
return 0;
}
}
CHRGET();
return 1;
}
/*
* Continuation
*
* This will put a magic value onto the stack and run the main
* function again with another PC value as a start address.
* When the code returns, it will find the magic value, and
* the main function will quit, so we end up here again.
*/
static void
call(unsigned short pc) {
PC = pc;
PUSH_WORD(MAGIC_CONTINUATION-1);
main(0,0);
}
static void
check_comma() {
call(0xAEFD);
}
static unsigned short
get_word() {
call(0xAD8A);
call(0xB7F7);
return RAM[0x14] | (RAM[0x15]<<8);
}
static unsigned char
get_byte() {
call(0xB79E);
return X;
}
static void
get_string(char *s) {
int i;
call(0xAD9E);
call(0xB6A3);
for (i = 0; i < A; i++)
s[i] = RAM[(X|(Y<<8))+i];
s[A] = 0;
}
#define ERROR_TOO_MANY_FILES 0x01
#define ERROR_FILE_OPEN 0x02
#define ERROR_FILE_NOT_OPEN 0x03
#define ERROR_FILE_NOT_FOUND 0x04
#define ERROR_DEVICE_NOT_PRESENT 0x05
#define ERROR_NOT_INPUT_FILE 0x06
#define ERROR_NOT_OUTPUT_FILE 0x07
#define ERROR_MISSING_FILE_NAME 0x08
#define ERROR_ILLEGAL_DEVICE_NUMBER 0x09
#define ERROR_NEXT_WITHOUT_FOR 0x0A
#define ERROR_SYNTAX 0x0B
#define ERROR_RETURN_WITHOUT_GOSUB 0x0C
#define ERROR_OUT_OF_DATA 0x0D
#define ERROR_ILLEGAL_QUANTITY 0x0E
#define ERROR_OVERFLOW 0x0F
#define ERROR_OUT_OF_MEMORY 0x10
#define ERROR_UNDEFD_STATMENT 0x11
#define ERROR_BAD_SUBSCRIPT 0x12
#define ERROR_REDIMD_ARRAY 0x13
#define ERROR_DEVISION_BY_ZERO 0x14
#define ERROR_ILLEGAL_DIRECT 0x15
#define ERROR_TYPE_MISMATCH 0x16
#define ERROR_STRING_TOO_LONG 0x17
#define ERROR_FILE_DATA 0x18
#define ERROR_FORMULA_TOO_COMPLEX 0x19
#define ERROR_CANT_CONTINUE 0x1A
#define ERROR_UNDEFD_FUNCTION 0x1B
#define ERROR_VERIFY 0x1C
#define ERROR_LOAD 0x1D
#define ERROR_BREAK 0x1E
static unsigned short
error(unsigned char index) {
X = index;
return 0xA437; /* error handler */
}
/*
* Print BASIC Error Message
*
* We could add handling of extra error codes here, or
* print friendlier strings, or implement "ON ERROR GOTO".
*/
unsigned short
plugin_error() {
return 0;
}
/*
* BASIC Warm Start
*
* This gets called whenever we are in direct mode.
*/
unsigned short
plugin_main() {
return 0;
}
/*
* Tokenize BASIC Text
*/
unsigned short
plugin_crnch() {
return 0;
}
/*
* BASIC Text LIST
*/
unsigned short
plugin_qplop() {
return 0;
}
/*
* BASIC Char. Dispatch
*
* This is used for interpreting statements.
*/
unsigned short
plugin_gone() {
set_chrptr(get_chrptr()+1);
for (;;) {
unsigned short chrptr;
set_chrptr(get_chrptr()-1);
chrptr = get_chrptr();
/*
* this example shows:
* - how to get a 16 bit integer
* - how to get an 8 bit integer
* - how to check for a comma delimiter
* - how to do error handling
*/
if (compare("LOCATE")) {
unsigned char x,y;
y = get_byte(); /* 'line' first */
check_comma();
x = get_byte(); /* then 'column' */
/* XXX ignores terminal size */
if (x>80 || y>25 || x==0 || y==0)
return error(ERROR_ILLEGAL_QUANTITY);
move_cursor(x, y);
continue;
}
/* Implements the ANSI Set Graphics Mode
as appears on http://ascii-table.com/ansi-escape-sequences.php
Note that some attributes will be implemented differently depending on what
terminal program you are using.
*/
if (compare("ANSISGM")) {
char attr, fg, bg;
attr = get_byte(); /* Attribute */
check_comma();
fg = get_byte(); /* Foreground color */
check_comma();
bg = get_byte(); /* Background color */
if (attr < 0 || attr > 8) return error(ERROR_ILLEGAL_QUANTITY);
if (fg < 30 || fg > 37) return error(ERROR_ILLEGAL_QUANTITY);
if (bg < 40 || bg > 47) return error(ERROR_ILLEGAL_QUANTITY);
printf("\033[%d;%d;%dm",attr,bg,fg);
continue;
}
/* ANSI erase line */
if (compare("ANSIEL")) {
printf("\033[K");
}
/* ANSI save cursor pos */
if (compare("ANSISC")) {
printf("\033[s");
}
/* ANSI restore cursor pos */
if (compare("ANSIRC")) {
printf("\033[u");
}
/*
* this example shows:
* - how to override existing keywords
* - how to hand the instruction to the
* original interpreter if we don't want
* to handle it
*/
if (compare("\222")) { /* 0x92 - WAIT */
unsigned short a;
unsigned char b;
a = get_word();
check_comma();
b = get_byte();
if (a==6502) {
printf("MICROSOFT!");
continue;
} else {
set_chrptr(chrptr);
return 0;
}
}
/*
* this example shows:
* - how to deal with new keywords that contain
* existing keywords
* - how to parse a string
*/
if (compare("\236TEM")) {
char s[256];
get_string(s);
system(s);
continue;
}
if (compare("QUIT")) {
exit(0);
}
break;
}
return 0;
}
/*
* BASIC Token Evaluation
*
* This is used for expression evaluation.
* New functions and operators go here.
*/
unsigned short
plugin_eval() {
return 0;
}