- Consider? As the keyword debug - Allow goto to be omitted if immediately after it is a decimal integer value - Addition of machine language related functions - Add tron / troff command (display execution line number) Version 0.05 - from wonbe005.lzh
2602 lines
54 KiB
C
2602 lines
54 KiB
C
/* wonbe: WonderWitch Basic Environment */
|
|
/* First Created: Nov.3,2000 by Pie Dey Co.,Ltd. */
|
|
|
|
/* This source code is distributed under GNU General Public License (GPL) */
|
|
/* see http://www.gnu.org/ about GPL */
|
|
|
|
/* This source code is for: */
|
|
/* Visual C++ 1.5 (for WW binary) */
|
|
/* Visual C++ 6.0 (for Win32 binary) */
|
|
/* Turbo C++ 1.0 (for WW binary, by Hirotaka JOE Ohkubo) */
|
|
/* GNU Win32 gcc (for Win32 binary, by Hirotaka JOE Ohkubo) */
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#ifdef WIN32
|
|
#include <assert.h>
|
|
#endif
|
|
#ifdef WW
|
|
#include <sys/comm.h>
|
|
#include <sys/process.h>
|
|
#include <sys/fcntl.h>
|
|
#include <sys/bios.h>
|
|
#define assert(x)
|
|
#endif
|
|
#include "keywords.h"
|
|
#ifdef WIN32
|
|
#include "win32text.h"
|
|
#endif
|
|
|
|
/* cygwin でなぜか stdarg.h が読み込まれてない & __GNUC__ が効かない... */
|
|
#ifdef CYGWIN
|
|
#include <stdarg.h>
|
|
#endif
|
|
|
|
char myVersion[] = "0.05";
|
|
|
|
#ifdef WW
|
|
#define NEWLINE "\xa\xd"
|
|
#else
|
|
#define NEWLINE "\n"
|
|
#endif
|
|
|
|
#ifdef WW
|
|
#define MEMMOVE mymemmove
|
|
#define FAR far
|
|
#else
|
|
#define MEMMOVE mymemmove
|
|
#define FAR
|
|
#endif
|
|
|
|
#ifdef WIN32
|
|
typedef unsigned char BYTE;
|
|
#endif
|
|
typedef unsigned short WORD;
|
|
typedef signed short SHORT;
|
|
typedef unsigned long DWORD;
|
|
|
|
#define EOL (0x0d)
|
|
|
|
#ifdef WIN32
|
|
typedef int BOOL;
|
|
#endif
|
|
#define FALSE 0
|
|
#define TRUE 1
|
|
|
|
typedef void (*NEARPROC)();
|
|
|
|
#define WORKAREA_SIZE ((WORD)49152)
|
|
#define LINE_SIZE (256)
|
|
|
|
/* PLAY statement support */
|
|
#ifdef WW
|
|
#include <il_sound.h>
|
|
SoundIL soundIL;
|
|
extern BYTE *_heap;
|
|
BOOL soundAvailable;
|
|
#endif
|
|
|
|
/* 実行のみのモードで動いているとき (シリアル経由で端末が居ないとき) */
|
|
BOOL runtimeOnly;
|
|
|
|
#ifdef WW
|
|
void fatalExit( const char * msg )
|
|
{
|
|
text_screen_init();
|
|
text_put_string( 0, 0, (char*)msg );
|
|
text_put_string( 0, 3, "PUSH ANY BUTTON" );
|
|
key_wait();
|
|
}
|
|
#endif
|
|
|
|
BOOL commonPrint( FILE FAR * fp, const char * msg,... )
|
|
{
|
|
#ifdef WW
|
|
static
|
|
#endif
|
|
char buf[256];
|
|
vsprintf(buf, msg, (va_list)(&msg+1) );
|
|
if( fp != NULL ) {
|
|
int r;
|
|
r = fprintf( fp, buf );
|
|
return r >= 0;
|
|
} else {
|
|
#ifdef WW
|
|
if( runtimeOnly ) {
|
|
fatalExit( buf );
|
|
} else {
|
|
comm_send_string( buf );
|
|
}
|
|
#else
|
|
printf( buf );
|
|
#endif
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
void consoleInput( BYTE * buf, WORD len )
|
|
{
|
|
#ifdef WW
|
|
WORD cursor;
|
|
if( runtimeOnly ) {
|
|
fatalExit("consoleInput not allowed in runtime mode");
|
|
} else {
|
|
cursor = 0;
|
|
while( TRUE ) {
|
|
int ch;
|
|
ch = comm_receive_char();
|
|
if( ch == 0x0d ) { /* CR */
|
|
buf[cursor] = '\0';
|
|
return;
|
|
}
|
|
if( ch == 0x09 ) { /* TAB */
|
|
goto inputAsChar;
|
|
}
|
|
if( ch == 0x08 ) { /* BS */
|
|
if( cursor > 0 ) {
|
|
cursor--;
|
|
comm_send_char( (unsigned char)ch );
|
|
continue;
|
|
}
|
|
}
|
|
if( ch >= 0 && ch < ' ' ) {
|
|
/* ignore it */
|
|
continue;
|
|
}
|
|
if( ch >= ' ' && ch <= 0xfc ) {
|
|
inputAsChar:
|
|
if( cursor < len-1 ) {
|
|
buf[cursor] = ch;
|
|
cursor++;
|
|
comm_send_char( (unsigned char)ch );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#else
|
|
gets(buf);
|
|
#endif
|
|
}
|
|
|
|
typedef struct {
|
|
WORD id;
|
|
const BYTE * name;
|
|
} KEYWORDITEM;
|
|
|
|
KEYWORDITEM keywordsA[] = {
|
|
{ KEYWORD_AND,"and" },
|
|
{ KEYWORD_ABS,"abs" },
|
|
{ KEYWORD_AX,"ax"},
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsB[] = {
|
|
{ KEYWORD_BREAK,"break" },
|
|
{ KEYWORD_BX, "bx" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsC[] = {
|
|
{ KEYWORD_CHR,"chr" },
|
|
{ KEYWORD_CLS,"cls" },
|
|
{ KEYWORD_CONT,"cont" },
|
|
{ KEYWORD_CALL, "call" },
|
|
{ KEYWORD_CSEG, "cseg" },
|
|
{ KEYWORD_CX,"cx" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsD[] = {
|
|
{ KEYWORD_DEBUG,"debug" },
|
|
{ KEYWORD_DSEG, "dseg" },
|
|
{ KEYWORD_DX, "dx" },
|
|
{ KEYWORD_DI, "di" },
|
|
{ KEYWORD_DS, "ds" },
|
|
{ KEYWORD_DEFSEG, "defseg" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsE[] = {
|
|
{ KEYWORD_END,"end" },
|
|
{ KEYWORD_EXIT,"exit" },
|
|
{ KEYWORD_ES, "es" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsF[] = {
|
|
{ KEYWORD_FOR,"for" },
|
|
{ KEYWORD_FILES,"files" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsG[] = {
|
|
{ KEYWORD_GOTO,"goto" },
|
|
{ KEYWORD_GOSUB,"gosub" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsI[] = {
|
|
{ KEYWORD_IF,"if" },
|
|
{ KEYWORD_INT, "int" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsL[] = {
|
|
{ KEYWORD_LOCATE,"locate" },
|
|
{ KEYWORD_LIST,"list" },
|
|
{ KEYWORD_LOAD,"load" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsM[] = {
|
|
{ KEYWORD_MERGE,"merge" },
|
|
{ 0,NULL }
|
|
};
|
|
KEYWORDITEM keywordsN[] = {
|
|
{ KEYWORD_NEXT,"next" },
|
|
{ KEYWORD_NOT,"not" },
|
|
{ KEYWORD_NEW,"new" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsO[] = {
|
|
{ KEYWORD_OR,"or" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsP[] = {
|
|
{ KEYWORD_PRINT,"print" },
|
|
{ KEYWORD_PLAY,"play" },
|
|
{ KEYWORD_PEEK, "peek" },
|
|
{ KEYWORD_POKE, "poke" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsR[] = {
|
|
{ KEYWORD_REM,"rem" },
|
|
{ KEYWORD_RETURN,"return" },
|
|
{ KEYWORD_RND,"rnd" },
|
|
{ KEYWORD_RUN,"run" },
|
|
{ KEYWORD_RANDOMIZE,"randomize" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsS[] = {
|
|
{ KEYWORD_STEP,"step" },
|
|
{ KEYWORD_SCAN_A,"scan_a" },
|
|
{ KEYWORD_SCAN_B,"scan_b" },
|
|
{ KEYWORD_SCAN_X1,"scan_x1" },
|
|
{ KEYWORD_SCAN_X2,"scan_x2" },
|
|
{ KEYWORD_SCAN_X3,"scan_x3" },
|
|
{ KEYWORD_SCAN_X4,"scan_x4" },
|
|
{ KEYWORD_SCAN_Y1,"scan_y1" },
|
|
{ KEYWORD_SCAN_Y2,"scan_y2" },
|
|
{ KEYWORD_SCAN_Y3,"scan_y3" },
|
|
{ KEYWORD_SCAN_Y4,"scan_y4" },
|
|
{ KEYWORD_SCAN,"scan" }, /* scanはscna_Xより後になければならない */
|
|
{ KEYWORD_SAVE,"save" },
|
|
{ KEYWORD_SSEG, "sseg" },
|
|
{ KEYWORD_SI, "si" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsT[] = {
|
|
{ KEYWORD_THEN,"then" },
|
|
{ KEYWORD_TO,"to" },
|
|
{ KEYWORD_TICK,"tick" },
|
|
{ KEYWORD_TRON,"tron" },
|
|
{ KEYWORD_TROFF,"troff" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsV[] = {
|
|
{ KEYWORD_VARPTR, "varptr" },
|
|
{ KEYWORD_VARSEG, "varseg" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsW[] = {
|
|
{ KEYWORD_WAITVB,"waitvb" },
|
|
{ KEYWORD_WAIT,"wait" }, // waitはwaitvbより後になければならない
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM keywordsX[] = {
|
|
{ KEYWORD_XOR,"xor" },
|
|
{ 0,NULL }
|
|
};
|
|
|
|
KEYWORDITEM * keywordsIndex[26] = {
|
|
keywordsA, keywordsB, keywordsC, keywordsD,
|
|
keywordsE, keywordsF, keywordsG, NULL,
|
|
keywordsI, NULL, NULL, keywordsL,
|
|
keywordsM, keywordsN, keywordsO, keywordsP,
|
|
NULL, keywordsR, keywordsS, keywordsT,
|
|
NULL, keywordsV, keywordsW, keywordsX,
|
|
NULL, NULL
|
|
};
|
|
|
|
/* インタラクティブモード時の1行バッファ */
|
|
BYTE waRawLine[LINE_SIZE];
|
|
BYTE waCookedLine[LINE_SIZE];
|
|
|
|
/* コードとデータを収める領域 */
|
|
WORD codeTop;
|
|
WORD dataTop;
|
|
/*WORD savedPointer; /* breakしたときの実行中位置 */
|
|
BYTE wa[WORKAREA_SIZE];
|
|
|
|
/* グローバル変数領域 */
|
|
#define NUMBER_OF_SIMPLE_VARIABLES (26)
|
|
SHORT globalVariables[NUMBER_OF_SIMPLE_VARIABLES];
|
|
SHORT topLevelVariables[NUMBER_OF_SIMPLE_VARIABLES];
|
|
SHORT * localVariables; /* あるスコープで有効なローカル変数領域を持つ */
|
|
|
|
/* 現在実行中の位置に関する情報 */
|
|
/* waかwaCookedLineのどちらかの中を示すかNULL */
|
|
BYTE * executionPointer;
|
|
/* contコマンドで実行を再開するポイント */
|
|
BYTE * resumePointer;
|
|
|
|
/* 現在処理中の行番号 (0ならダイレクトモード) */
|
|
WORD currentLineNumber;
|
|
|
|
/* gosubとforのためのスタック */
|
|
#define STACK_TYPE_GOSUB 1
|
|
#define STACK_TYPE_FOR 2
|
|
typedef struct {
|
|
WORD type; /* 1==GOSUB, 2==FOR */
|
|
/* for GOSUB and FOR */
|
|
BYTE * returnPointer;
|
|
/* for GOSUB */
|
|
SHORT * lastLocalVariables;
|
|
SHORT simpleVariables[NUMBER_OF_SIMPLE_VARIABLES];
|
|
/* for FOR */
|
|
SHORT * pvar; /* counter variable */
|
|
SHORT limit; /* limit value */
|
|
SHORT step; /* step value */
|
|
} STACK;
|
|
#define STACK_MAX 8
|
|
STACK stacks[STACK_MAX];
|
|
int stackPointer;
|
|
|
|
/* インタラクティブモードかインタプリタモードか */
|
|
BOOL bInteractive;
|
|
|
|
/* インタプリタ脱出後の振る舞いのリクエスト */
|
|
BOOL bForceToReturnSuper;
|
|
|
|
int requestNextAction; /* 以下のいずれか */
|
|
/* 何もせず現状のまま */
|
|
#define REQUEST_NO_ACTION 0
|
|
/* 対話モードへ行け */
|
|
#define REQUEST_INTERACTIVE 1
|
|
/* runLineNumberから実行を開始せよ */
|
|
#define REQUEST_RUN_LINENUMBER 2
|
|
/* waRawLineに入っている文字列をファイル名としてそれを読み込んで実行せよ */
|
|
#define REQUEST_RUN_FILE 3
|
|
/* newせよ */
|
|
#define REQUEST_NEW 4
|
|
/* waRawLineに入っている文字列をファイル名としてそれをロード */
|
|
#define REQUEST_LOAD_FILE 5
|
|
/* WONBEを終了してOSに戻れ */
|
|
#define REQUEST_EXIT_SYSTEM 6
|
|
/* waRawLineに入っている文字列をファイル名としてそれをマージ */
|
|
#define REQUEST_MERGE_FILE 7
|
|
/* REQUEST_RUN_LINENUMBERで実行開始をリクエストする行番号 */
|
|
WORD runLineNumber;
|
|
// 編集機能が追加専用モードであることを示す
|
|
// REQUEST_RUN_FILE or REQUEST_LOAD_FILE経由で処理するときだけTRUEとなる
|
|
BOOL appendMode = FALSE;
|
|
|
|
// Read/Writeされる擬似変数の置き場所
|
|
WORD defseg;
|
|
WORD registers[KEYWORD_ES-KEYWORD_AX+1];
|
|
|
|
// トレースモード(tron/troff)
|
|
BOOL traceFlag = FALSE;
|
|
|
|
// マシン語アクセス関連
|
|
#ifdef VC15
|
|
// VC++1.5のときに使うコード
|
|
WORD getdseg()
|
|
{
|
|
WORD s;
|
|
__asm mov s, ds;
|
|
return s;
|
|
}
|
|
WORD getsseg()
|
|
{
|
|
WORD s;
|
|
__asm mov s, ss;
|
|
return s;
|
|
}
|
|
WORD getcseg()
|
|
{
|
|
WORD s;
|
|
__asm mov s, cs;
|
|
return s;
|
|
}
|
|
#else
|
|
WORD getdseg()
|
|
{
|
|
return 0; // DUMMY VALUE
|
|
}
|
|
WORD getsseg()
|
|
{
|
|
return 0; // DUMMY VALUE
|
|
}
|
|
WORD getcseg()
|
|
{
|
|
return 0; // DUMMY VALUE
|
|
}
|
|
#endif
|
|
|
|
/* エラー発生 */
|
|
void syntaxError()
|
|
{
|
|
commonPrint(NULL, "Syntax Error in %d\x7" NEWLINE, currentLineNumber );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
void divideByZero()
|
|
{
|
|
commonPrint(NULL, "Divide by 0 in %d\x7" NEWLINE, currentLineNumber );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
void outOfArraySubscription()
|
|
{
|
|
commonPrint(NULL, "Out of Array Subscription in %d\x7" NEWLINE, currentLineNumber );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
void lineNumberNotFound( WORD lineNumber )
|
|
{
|
|
commonPrint(NULL, "Line Number %d not Found in %d\x7" NEWLINE, lineNumber, currentLineNumber );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
void stackOverflow()
|
|
{
|
|
commonPrint(NULL, "Stack Overflow in %d\x7" NEWLINE, currentLineNumber );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
void stackUnderflow()
|
|
{
|
|
commonPrint(NULL, "Stack Underflow in %d\x7" NEWLINE, currentLineNumber );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
void nextWithoutFor()
|
|
{
|
|
commonPrint(NULL, "Next without For in %d\x7" NEWLINE, currentLineNumber );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
void outOfMemory()
|
|
{
|
|
commonPrint(NULL, "Out of memory in %d\x7" NEWLINE, currentLineNumber );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
void loadError()
|
|
{
|
|
commonPrint(NULL, "File Not Found or Load Error in %d\x7" NEWLINE, currentLineNumber );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
void cantContinue()
|
|
{
|
|
commonPrint(NULL, "Can't Continue in %d\x7" NEWLINE, currentLineNumber );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
void saveError()
|
|
{
|
|
commonPrint(NULL, "Save Erroor in %d\x7" NEWLINE, currentLineNumber );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
void paramError()
|
|
{
|
|
commonPrint(NULL, "Parameter Erroor in %d\x7" NEWLINE, currentLineNumber );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
void lineNumberZeroError()
|
|
{
|
|
// このエラーはエディタが出すものなのでランタイムで発生することはあり得ない
|
|
// なので現在行番号を報告する意味がない
|
|
//commonPrint(NULL, "Invalid Line Number 0 in %d\x7" NEWLINE, currentLineNumber );
|
|
commonPrint(NULL, "Invalid Line Number 0\x7" NEWLINE );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
/* ユーティリティ関数達 */
|
|
void mymemmove( BYTE * dst, const BYTE * src, WORD len )
|
|
{
|
|
WORD i;
|
|
if( dst < src ) {
|
|
for( i=0; i<len; i++ ) {
|
|
*dst++ = *src++;
|
|
}
|
|
} else {
|
|
dst += len;
|
|
src += len;
|
|
for( i=0; i<len; i++ ) {
|
|
*--dst = *--src;
|
|
}
|
|
}
|
|
}
|
|
|
|
BYTE * skipToEOL( BYTE * p )
|
|
{
|
|
while( TRUE ) {
|
|
if( *p == EOL ) return p;
|
|
switch( *p++ ) {
|
|
case 0x01: /* 次にあるのは10進2バイト整数 */
|
|
case 0x02: /* 次にあるのは16進2バイト整数 */
|
|
p += 2;
|
|
break;
|
|
case 0x03: /* 次にあるのは可変長文字列 0x00でターミネート */
|
|
{
|
|
while( TRUE ) {
|
|
if( *p++ == '\0' ) break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// 短い関数をインライン展開しないコンパイラ向けの最適化としてマクロ化
|
|
#define skipToNextLine(p) ( skipToEOL(p+4)+1 )
|
|
//BYTE * skipToNextLine( BYTE * p )
|
|
//{
|
|
// return skipToEOL( p+4 ); // +4は行番号2バイト行長さ2バイト
|
|
//}
|
|
|
|
#if 1
|
|
#define mytolower(ch) ((ch >= 'A' && ch <= 'Z')? ch - 'A' + 'a' : ch)
|
|
#else
|
|
BYTE mytolower( BYTE ch )
|
|
{
|
|
if( ch >= 'A' && ch <= 'Z' ) {
|
|
return ch - 'A' + 'a';
|
|
}
|
|
return ch;
|
|
}
|
|
#endif
|
|
|
|
/* シフトJISの1バイト目かどうか判断。マクロなので副作用に注意 */
|
|
#define IsDBCSLeadByte(ch) ((ch >= 0x81 && ch <= 0x9f) || (ch >= 0xe0 && ch <= 0xfc))
|
|
|
|
/* 実行時情報のクリア (contできなくなる) */
|
|
void clearRuntimeInfo()
|
|
{
|
|
memset( globalVariables, 0, sizeof(globalVariables) );
|
|
memset( topLevelVariables, 0, sizeof(topLevelVariables) );
|
|
memset( stacks, 0, sizeof(stacks) );
|
|
memset( registers, 0, sizeof(registers) );
|
|
executionPointer = NULL;
|
|
resumePointer = NULL;
|
|
currentLineNumber = 0;
|
|
localVariables = topLevelVariables;
|
|
stackPointer = 0;
|
|
defseg = getdseg();
|
|
srand(0);
|
|
}
|
|
|
|
/* 行頭の行番号の、実行時の定型処理 */
|
|
void processLineHeader()
|
|
{
|
|
if( !bInteractive ) {
|
|
currentLineNumber = *((WORD *)executionPointer);
|
|
if( traceFlag ) {
|
|
commonPrint( NULL, "[%d]", currentLineNumber );
|
|
}
|
|
executionPointer += 4; // +4は行番号2バイト行長さ2バイト
|
|
} else {
|
|
currentLineNumber = 0;
|
|
}
|
|
}
|
|
|
|
/* 行番号処理 */
|
|
BYTE * getLineReferenceFromLineNumber( WORD lineNumber )
|
|
{
|
|
BYTE * p;
|
|
WORD t;
|
|
p = wa;
|
|
while( TRUE ) {
|
|
t = *((WORD *)p);
|
|
if( t == 0 ) {
|
|
return NULL;
|
|
}
|
|
if( lineNumber == t ) {
|
|
return p;
|
|
}
|
|
p += *((WORD *)(p+2));
|
|
}
|
|
}
|
|
|
|
BYTE * getInsertionPointFromLineNumber( WORD lineNumber )
|
|
{
|
|
BYTE * p;
|
|
WORD t;
|
|
p = wa;
|
|
while( TRUE ) {
|
|
t = *((WORD *)p);
|
|
if( t == 0 ) {
|
|
return p;
|
|
}
|
|
if( lineNumber <= t ) {
|
|
return p;
|
|
}
|
|
p += *((WORD *)(p+2));
|
|
}
|
|
}
|
|
|
|
/* 配列管理 */
|
|
int availableArrayItems()
|
|
{
|
|
return (WORKAREA_SIZE-dataTop) >> 1;
|
|
}
|
|
|
|
SHORT * derefArrayItem( int index )
|
|
{
|
|
if( index < 0 || index >= availableArrayItems() ) {
|
|
outOfArraySubscription();
|
|
return NULL; /* そのインデックスは使えません */
|
|
}
|
|
return (SHORT*)(&wa[dataTop+index*2]);
|
|
}
|
|
|
|
/* 物理テキストの管理 */
|
|
#define SCREEN_SIZE_X (224/8)
|
|
#define SCREEN_SIZE_Y (144/8)
|
|
|
|
WORD screen[SCREEN_SIZE_X][SCREEN_SIZE_Y];
|
|
|
|
SHORT xCursor, yCursor;
|
|
|
|
#define TABSTOP 8
|
|
|
|
void ptextScrollUp()
|
|
{
|
|
int x, y;
|
|
for( y=0; y<SCREEN_SIZE_Y-1; y++ ) {
|
|
for( x=0; x<SCREEN_SIZE_X; x++ ) {
|
|
screen[x][y] = screen[x][y+1];
|
|
}
|
|
}
|
|
for( x=0; x<SCREEN_SIZE_X; x++ ) {
|
|
screen[x][SCREEN_SIZE_Y-1] = ' ';
|
|
}
|
|
for( y=0; y<SCREEN_SIZE_Y; y++ ) {
|
|
for( x=0; x<SCREEN_SIZE_X; x++ ) {
|
|
#ifdef WIN32
|
|
drawChar( x, y, screen[x][y] );
|
|
#endif
|
|
#ifdef WW
|
|
text_put_char( x, y, screen[x][y] );
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
void ptextNewline()
|
|
{
|
|
xCursor = 0;
|
|
yCursor++;
|
|
if( yCursor >= SCREEN_SIZE_Y ) {
|
|
ptextScrollUp();
|
|
yCursor = SCREEN_SIZE_Y-1;
|
|
}
|
|
}
|
|
|
|
void ptextTab()
|
|
{
|
|
if( xCursor+TABSTOP >= SCREEN_SIZE_X ) {
|
|
ptextNewline();
|
|
return;
|
|
}
|
|
xCursor = (xCursor+TABSTOP) / TABSTOP * TABSTOP;
|
|
}
|
|
|
|
void ptextString( const BYTE FAR * msg )
|
|
{
|
|
while( TRUE ) {
|
|
WORD ch;
|
|
if( *msg == '\0' ) break;
|
|
if( IsDBCSLeadByte(*msg) ) {
|
|
ch = (msg[0] << 8) + msg[1];
|
|
msg+=2;
|
|
} else {
|
|
ch = msg[0];
|
|
msg++;
|
|
}
|
|
screen[xCursor][yCursor] = ch;
|
|
#ifdef WIN32
|
|
drawChar( xCursor, yCursor, ch );
|
|
#endif
|
|
#ifdef WW
|
|
text_put_char( xCursor, yCursor, ch );
|
|
#endif
|
|
xCursor++;
|
|
if( xCursor >= SCREEN_SIZE_X ) {
|
|
xCursor = 0;
|
|
yCursor++;
|
|
if( yCursor >= SCREEN_SIZE_Y ) {
|
|
ptextScrollUp();
|
|
yCursor = SCREEN_SIZE_Y-1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void ptextLocate( int x, int y )
|
|
{
|
|
if( x < 0 ) x = 0;
|
|
if( x >= SCREEN_SIZE_X ) x = SCREEN_SIZE_X-1;
|
|
if( y < 0 ) y = 0;
|
|
if( y >= SCREEN_SIZE_Y ) y = SCREEN_SIZE_Y-1;
|
|
xCursor = x;
|
|
yCursor = y;
|
|
}
|
|
|
|
void ptextCLS()
|
|
{
|
|
int x,y;
|
|
xCursor = 0;
|
|
yCursor = 0;
|
|
#ifdef WIN32
|
|
clearText();
|
|
#endif
|
|
#ifdef WW
|
|
for( y=0; y<SCREEN_SIZE_Y; y++ ) {
|
|
text_fill_char(0, y, SCREEN_SIZE_X, ' ' );
|
|
}
|
|
#endif
|
|
for( y=0; y<SCREEN_SIZE_Y; y++ ) {
|
|
for( x=0; x<SCREEN_SIZE_X; x++ ) {
|
|
screen[x][y] = ' ';
|
|
}
|
|
}
|
|
}
|
|
|
|
/* 中間コードからソーステキストに戻しながら出力(listとsaveで使う) */
|
|
BOOL sourceDump( FILE FAR * fp, WORD from, WORD to )
|
|
{
|
|
BYTE * p;
|
|
BOOL b;
|
|
p = getInsertionPointFromLineNumber( from );
|
|
while( TRUE ) {
|
|
WORD lineNumber;
|
|
lineNumber = *((WORD*)p);
|
|
if( lineNumber == 0 ) break;
|
|
if( lineNumber > to ) break;
|
|
p += 4;
|
|
b = commonPrint(fp,"%d",lineNumber);
|
|
if( b == FALSE ) return FALSE;
|
|
while( TRUE ) {
|
|
if( *p == EOL ) {
|
|
p++;
|
|
break;
|
|
}
|
|
if( *p == 0x01 ) {
|
|
WORD n;
|
|
p++;
|
|
n = *((WORD*)p);
|
|
p += 2;
|
|
b = commonPrint(fp,"%d",n);
|
|
if( b == FALSE ) return FALSE;
|
|
} else if( *p == 0x02 ) {
|
|
WORD n;
|
|
p++;
|
|
n = *((WORD*)p);
|
|
p += 2;
|
|
b = commonPrint(fp,"0x%x",n);
|
|
if( b == FALSE ) return FALSE;
|
|
} else if( *p == 0x03 ) {
|
|
p++;
|
|
b = commonPrint(fp,"\"%s\"",p);
|
|
if( b == FALSE ) return FALSE;
|
|
p += strlen(p)+1;
|
|
} else if( *p == '\'' ) { /* コメントのあとはそのまま行末まで転送 */
|
|
while( TRUE ) {
|
|
if( *p == EOL ) break;
|
|
b = commonPrint(fp,"%c",*p);
|
|
if( b == FALSE ) return FALSE;
|
|
p++;
|
|
}
|
|
} else if( *p >= 0x80 ) {
|
|
int i;
|
|
BOOL found;
|
|
found = FALSE;
|
|
for( i=0; i<26; i++ ) {
|
|
KEYWORDITEM * k;
|
|
if( found ) break;
|
|
k = keywordsIndex[i];
|
|
if( k == NULL ) continue;
|
|
while( TRUE ) {
|
|
if( k->id == 0 ) break;
|
|
if( k->id == *p ) {
|
|
b = commonPrint(fp, k->name );
|
|
if( b == FALSE ) return FALSE;
|
|
p++;
|
|
if( k->id == KEYWORD_REM ) { /* コメントのあとはそのまま行末まで転送 */
|
|
while( TRUE ) {
|
|
if( *p == EOL ) break;
|
|
b = commonPrint(fp,"%c",*p);
|
|
if( b == FALSE ) return FALSE;
|
|
p++;
|
|
}
|
|
}
|
|
found = TRUE;
|
|
break;
|
|
}
|
|
k++;
|
|
}
|
|
}
|
|
if( !found ) {
|
|
b = commonPrint(fp,"[?%x?]", *p );
|
|
if( b == FALSE ) return FALSE;
|
|
p++;
|
|
}
|
|
} else {
|
|
b = commonPrint(fp,"%c",*p);
|
|
if( b == FALSE ) return FALSE;
|
|
p++;
|
|
}
|
|
}
|
|
b = commonPrint(fp,NEWLINE);
|
|
if( b == FALSE ) return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/* 式計算機能 */
|
|
SHORT expr(); /* forward decl */
|
|
|
|
SHORT * getArrayReference()
|
|
{
|
|
BYTE ch;
|
|
int index;
|
|
SHORT * pvar;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch != '(' && ch != '[' ) {
|
|
syntaxError();
|
|
return NULL;
|
|
}
|
|
index = expr();
|
|
if( bForceToReturnSuper ) return NULL;
|
|
pvar = derefArrayItem( index );
|
|
if( pvar == NULL ) return NULL;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch != ')' && ch != ']' ) {
|
|
syntaxError();
|
|
return NULL;
|
|
}
|
|
return pvar;
|
|
}
|
|
|
|
SHORT calcValue()
|
|
{
|
|
BYTE ch;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch >= 'A' && ch <= 'Z' ) {
|
|
return globalVariables[ch-'A'];
|
|
}
|
|
if( ch >= 'a' && ch <= 'z' ) {
|
|
return localVariables[ch-'a'];
|
|
}
|
|
if( ch == '@' ) {
|
|
SHORT * pvar;
|
|
pvar = getArrayReference();
|
|
if( pvar == NULL ) return -1;
|
|
return *pvar;
|
|
}
|
|
if( ch >= KEYWORD_AX && ch <= KEYWORD_ES ) {
|
|
return registers[ch-KEYWORD_AX];
|
|
}
|
|
switch( ch ) {
|
|
case 0x01: /* 次にあるのは10進2バイト整数 */
|
|
case 0x02: /* 次にあるのは16進2バイト整数 */
|
|
{
|
|
SHORT t;
|
|
t = *((SHORT *)executionPointer);
|
|
executionPointer += 2;
|
|
return t;
|
|
}
|
|
case '(':
|
|
{
|
|
SHORT t;
|
|
t = expr();
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch != ')' ) {
|
|
syntaxError();
|
|
return -1;
|
|
}
|
|
return t;
|
|
}
|
|
case '-':
|
|
return -calcValue();
|
|
case KEYWORD_NOT:
|
|
return ~calcValue();
|
|
case KEYWORD_SCAN:
|
|
#ifdef WW
|
|
return key_press_check();
|
|
#else
|
|
return win32_scan();
|
|
#endif
|
|
case KEYWORD_WAIT:
|
|
#ifdef WW
|
|
return key_wait();
|
|
#else
|
|
return win32_wait();
|
|
#endif
|
|
case KEYWORD_RND:
|
|
{
|
|
SHORT t;
|
|
t = calcValue();
|
|
return rand() % t;
|
|
}
|
|
case KEYWORD_ABS:
|
|
{
|
|
SHORT t;
|
|
t = calcValue();
|
|
if( t < 0 ) return -t;
|
|
return t;
|
|
}
|
|
case KEYWORD_TICK:
|
|
#ifdef WW
|
|
return sys_get_tick_count();
|
|
#else
|
|
return win32_get_tick_count();
|
|
#endif
|
|
case KEYWORD_SCAN_A:
|
|
return 4;
|
|
case KEYWORD_SCAN_B:
|
|
return 8;
|
|
case KEYWORD_SCAN_X1:
|
|
return 16;
|
|
case KEYWORD_SCAN_X2:
|
|
return 32;
|
|
case KEYWORD_SCAN_X3:
|
|
return 64;
|
|
case KEYWORD_SCAN_X4:
|
|
return 128;
|
|
case KEYWORD_SCAN_Y1:
|
|
return 256;
|
|
case KEYWORD_SCAN_Y2:
|
|
return 512;
|
|
case KEYWORD_SCAN_Y3:
|
|
return 1024;
|
|
case KEYWORD_SCAN_Y4:
|
|
return 2048;
|
|
case KEYWORD_DEFSEG:
|
|
return defseg;
|
|
case KEYWORD_DSEG:
|
|
return getdseg();
|
|
case KEYWORD_CSEG:
|
|
return getcseg();
|
|
case KEYWORD_SSEG:
|
|
return getsseg();
|
|
case KEYWORD_VARPTR:
|
|
case KEYWORD_VARSEG:
|
|
{
|
|
BOOL requestSeg;
|
|
SHORT * refer;
|
|
requestSeg = (ch == KEYWORD_VARSEG);
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch != '(' ) {
|
|
syntaxError();
|
|
return -1;
|
|
}
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch == '@' ) { /* is it l-value? */
|
|
SHORT * pvar;
|
|
pvar = getArrayReference();
|
|
if( pvar == NULL ) return -1;
|
|
refer = pvar;
|
|
} else if( ch >= 'A' && ch <= 'Z' ) {
|
|
refer = &globalVariables[ch-'A'];
|
|
} else if( ch >= 'a' && ch <= 'z' ) {
|
|
refer = &localVariables[ch-'a'];
|
|
} else if( ch >= KEYWORD_AX && ch <= KEYWORD_ES ) {
|
|
refer = ®isters[ch-KEYWORD_AX];
|
|
} else if( ch == KEYWORD_DEFSEG ) {
|
|
refer = &defseg;
|
|
} else {
|
|
syntaxError();
|
|
return -1;
|
|
}
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch != ')' ) {
|
|
syntaxError();
|
|
return -1;
|
|
}
|
|
#ifdef WW
|
|
if( requestSeg ) {
|
|
return getdseg();
|
|
} else {
|
|
return (SHORT)refer;
|
|
}
|
|
#else
|
|
printf("varseg/varptr: %08x\n", refer );
|
|
return 0;
|
|
#endif
|
|
}
|
|
case KEYWORD_PEEK:
|
|
{
|
|
SHORT t;
|
|
BYTE FAR * fulladdr;
|
|
t = calcValue();
|
|
fulladdr = (BYTE FAR *)(((DWORD)defseg << 16) + (WORD)t);
|
|
#ifdef WW
|
|
return *fulladdr;
|
|
#else
|
|
printf("peek: %08x\n", fulladdr );
|
|
return 0;
|
|
#endif
|
|
}
|
|
}
|
|
syntaxError();
|
|
return -1;
|
|
}
|
|
|
|
SHORT expr4th()
|
|
{
|
|
SHORT acc;
|
|
acc = calcValue();
|
|
if( bForceToReturnSuper ) return -1;
|
|
|
|
while( TRUE ) {
|
|
BYTE ch;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
switch( ch ) {
|
|
case '*':
|
|
acc = acc * calcValue();
|
|
break;
|
|
case '/':
|
|
{
|
|
SHORT t;
|
|
t = calcValue();
|
|
if( t == 0 ) {
|
|
divideByZero();
|
|
} else {
|
|
acc = acc / t;
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
executionPointer--; /* unget it */
|
|
return acc;
|
|
}
|
|
if( bForceToReturnSuper ) return -1;
|
|
}
|
|
}
|
|
|
|
SHORT expr3nd()
|
|
{
|
|
SHORT acc;
|
|
acc = expr4th();
|
|
if( bForceToReturnSuper ) return -1;
|
|
|
|
while( TRUE ) {
|
|
BYTE ch;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
switch( ch ) {
|
|
case '+':
|
|
acc = acc + expr4th();
|
|
break;
|
|
case '-':
|
|
acc = acc - expr4th();
|
|
break;
|
|
default:
|
|
executionPointer--; /* unget it */
|
|
return acc;
|
|
}
|
|
if( bForceToReturnSuper ) return -1;
|
|
}
|
|
}
|
|
|
|
SHORT expr2nd()
|
|
{
|
|
SHORT acc;
|
|
acc = expr3nd();
|
|
if( bForceToReturnSuper ) return -1;
|
|
|
|
while( TRUE ) {
|
|
BYTE ch;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
switch( ch ) {
|
|
case '>':
|
|
{
|
|
BYTE ch2;
|
|
while( TRUE ) {
|
|
ch2 = *executionPointer++;
|
|
if( ch2 != ' ' && ch2 != '\t' ) break;
|
|
}
|
|
if( ch2 == '=' ) {
|
|
acc = (acc >= expr3nd());
|
|
if( acc != 0 ) acc = -1;
|
|
} else {
|
|
executionPointer--;
|
|
acc = (acc > expr3nd());
|
|
if( acc != 0 ) acc = -1;
|
|
}
|
|
}
|
|
break;
|
|
case '<':
|
|
{
|
|
BYTE ch2;
|
|
while( TRUE ) {
|
|
ch2 = *executionPointer++;
|
|
if( ch2 != ' ' && ch2 != '\t' ) break;
|
|
}
|
|
if( ch2 == '=' ) {
|
|
acc = (acc <= expr3nd());
|
|
if( acc != 0 ) acc = -1;
|
|
} else if( ch2 == '>' ) {
|
|
acc = (acc != expr3nd());
|
|
if( acc != 0 ) acc = -1;
|
|
} else {
|
|
executionPointer--;
|
|
acc = (acc < expr3nd());
|
|
if( acc != 0 ) acc = -1;
|
|
}
|
|
}
|
|
break;
|
|
case '=':
|
|
acc = (acc == expr3nd());
|
|
if( acc != 0 ) acc = -1;
|
|
break;
|
|
default:
|
|
executionPointer--; /* unget it */
|
|
return acc;
|
|
}
|
|
if( bForceToReturnSuper ) return -1;
|
|
}
|
|
}
|
|
|
|
SHORT expr()
|
|
{
|
|
SHORT acc;
|
|
acc = expr2nd();
|
|
if( bForceToReturnSuper ) return -1;
|
|
|
|
while( TRUE ) {
|
|
BYTE ch;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
switch( ch ) {
|
|
case KEYWORD_AND:
|
|
acc = acc & expr2nd();
|
|
break;
|
|
case KEYWORD_OR:
|
|
acc = acc | expr2nd();
|
|
break;
|
|
case KEYWORD_XOR:
|
|
acc = acc ^ expr2nd();
|
|
break;
|
|
default:
|
|
executionPointer--; /* unget it */
|
|
return acc;
|
|
}
|
|
if( bForceToReturnSuper ) return -1;
|
|
}
|
|
}
|
|
|
|
/* 各ステートメント実行処理メイン */
|
|
|
|
void st_assignment( SHORT * pvar ) /* 代入ステートメントだけ例外的に処理する */
|
|
{
|
|
BYTE ch;
|
|
SHORT val;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch != '=' ) {
|
|
syntaxError();
|
|
return;
|
|
}
|
|
val = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
*pvar = val;
|
|
}
|
|
|
|
void printOrDebug( BOOL bPrint )
|
|
{
|
|
WORD lastChar;
|
|
BYTE ch;
|
|
lastChar = '\0';
|
|
while( TRUE ) {
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch == EOL || ch == ':' || ch == '\'' ) {
|
|
executionPointer--; /* unget it */
|
|
break;
|
|
}
|
|
lastChar = ch;
|
|
switch( ch ) {
|
|
case 0x03:
|
|
if( bPrint ) {
|
|
ptextString( executionPointer );
|
|
} else {
|
|
commonPrint( NULL, executionPointer );
|
|
}
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch == '\0' ) break;
|
|
}
|
|
break;
|
|
case ';':
|
|
break;
|
|
case ',':
|
|
if( bPrint ) {
|
|
ptextTab();
|
|
} else {
|
|
commonPrint( NULL, "\t" );
|
|
}
|
|
break;
|
|
case KEYWORD_CHR:
|
|
{
|
|
WORD val;
|
|
#ifdef WW
|
|
static
|
|
#endif
|
|
BYTE str[3];
|
|
val = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
if( val >= 0x100 ) {
|
|
str[0] = (val >> 8);
|
|
str[1] = (val & 0xff);
|
|
str[2] = '\0';
|
|
} else {
|
|
str[0] = (BYTE)val;
|
|
str[1] = '\0';
|
|
}
|
|
if( bPrint ) {
|
|
ptextString( str );
|
|
} else {
|
|
commonPrint( NULL, str );
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
{
|
|
SHORT val;
|
|
#ifdef WW
|
|
static
|
|
#endif
|
|
BYTE str[10];
|
|
executionPointer--; /* unget it */
|
|
val = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
sprintf( str, "%d", val );
|
|
if( bPrint ) {
|
|
ptextString( str );
|
|
} else {
|
|
commonPrint( NULL, str );
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
if( lastChar != ';' && lastChar != ',' ) {
|
|
if( bPrint ) {
|
|
ptextNewline();
|
|
} else {
|
|
commonPrint( NULL, NEWLINE );
|
|
}
|
|
}
|
|
}
|
|
|
|
void st_print()
|
|
{
|
|
printOrDebug( TRUE );
|
|
}
|
|
|
|
void st_debug()
|
|
{
|
|
printOrDebug( FALSE );
|
|
}
|
|
|
|
void st_locate()
|
|
{
|
|
SHORT x, y;
|
|
BYTE ch;
|
|
x = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch != ',' ) {
|
|
syntaxError();
|
|
return;
|
|
}
|
|
y = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
ptextLocate( x, y );
|
|
}
|
|
|
|
void st_cls()
|
|
{
|
|
ptextCLS();
|
|
}
|
|
|
|
void st_goto()
|
|
{
|
|
SHORT val;
|
|
BYTE * t;
|
|
val = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
t = getLineReferenceFromLineNumber( val );
|
|
if( t == NULL ) {
|
|
lineNumberNotFound( val );
|
|
return;
|
|
}
|
|
executionPointer = t;
|
|
bInteractive = FALSE;
|
|
processLineHeader();
|
|
}
|
|
|
|
void st_gosub()
|
|
{
|
|
SHORT val;
|
|
BYTE * t;
|
|
val = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
t = getLineReferenceFromLineNumber( val );
|
|
if( t == NULL ) {
|
|
lineNumberNotFound( val );
|
|
return;
|
|
}
|
|
if( stackPointer+1 >= STACK_MAX ) {
|
|
stackOverflow();
|
|
return;
|
|
}
|
|
stacks[stackPointer].type = STACK_TYPE_GOSUB;
|
|
stacks[stackPointer].returnPointer = executionPointer;
|
|
stacks[stackPointer].lastLocalVariables = localVariables;
|
|
localVariables = stacks[stackPointer].simpleVariables;
|
|
stackPointer++;
|
|
executionPointer = t;
|
|
bInteractive = FALSE;
|
|
processLineHeader();
|
|
}
|
|
|
|
void st_return()
|
|
{
|
|
while( TRUE ) {
|
|
if( stackPointer == 0 ) {
|
|
stackUnderflow();
|
|
return;
|
|
}
|
|
stackPointer--;
|
|
if( stacks[stackPointer].type == STACK_TYPE_GOSUB ) break;
|
|
}
|
|
executionPointer = stacks[stackPointer].returnPointer;
|
|
if( executionPointer >= waCookedLine && executionPointer < waCookedLine+LINE_SIZE ) {
|
|
bInteractive = TRUE;
|
|
}
|
|
localVariables = stacks[stackPointer].lastLocalVariables;
|
|
}
|
|
|
|
void st_if()
|
|
{
|
|
SHORT val;
|
|
BYTE ch;
|
|
val = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch != KEYWORD_THEN ) {
|
|
syntaxError();
|
|
return;
|
|
}
|
|
if( val != 0 ) {
|
|
while( TRUE ) {
|
|
ch = *executionPointer;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
executionPointer++;
|
|
}
|
|
if( ch == 0x01 ) {
|
|
// thenのあとに整数が直接書かれた場合は、それにgotoする。
|
|
st_goto();
|
|
return;
|
|
}
|
|
return; /* thenの次から継続実行する */
|
|
}
|
|
/* 条件不成立につき、行末まで読み飛ばす */
|
|
executionPointer = skipToEOL( executionPointer );
|
|
}
|
|
|
|
void st_for()
|
|
{
|
|
BYTE ch;
|
|
SHORT from, to, step;
|
|
SHORT * pvar;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch == '@' ) { /* is it l-value? */
|
|
pvar = getArrayReference();
|
|
if( pvar == NULL ) return;
|
|
} else if( ch >= 'A' && ch <= 'Z' ) {
|
|
pvar = &globalVariables[ch-'A'];
|
|
} else if( ch >= 'a' && ch <= 'z' ) {
|
|
pvar = &localVariables[ch-'a'];
|
|
} else {
|
|
syntaxError();
|
|
return;
|
|
}
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch != '=' ) {
|
|
syntaxError();
|
|
return;
|
|
}
|
|
from = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch != KEYWORD_TO ) {
|
|
syntaxError();
|
|
return;
|
|
}
|
|
to = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch == KEYWORD_STEP ) {
|
|
step = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
} else {
|
|
step = 1;
|
|
executionPointer--; /* unget it */
|
|
}
|
|
|
|
if( stackPointer+1 >= STACK_MAX ) {
|
|
stackOverflow();
|
|
return;
|
|
}
|
|
stacks[stackPointer].type = STACK_TYPE_FOR;
|
|
stacks[stackPointer].returnPointer = executionPointer;
|
|
*pvar = from;
|
|
stacks[stackPointer].pvar = pvar;
|
|
stacks[stackPointer].limit = to;
|
|
stacks[stackPointer].step = step;
|
|
stackPointer++;
|
|
}
|
|
|
|
void st_next()
|
|
{
|
|
if( stackPointer == 0 ) {
|
|
nextWithoutFor();
|
|
return;
|
|
}
|
|
if( stacks[stackPointer-1].type != STACK_TYPE_FOR ) {
|
|
nextWithoutFor();
|
|
return;
|
|
}
|
|
if( stacks[stackPointer-1].limit == *(stacks[stackPointer-1].pvar) ) {
|
|
/* loop done */
|
|
stackPointer--;
|
|
return;
|
|
}
|
|
/* count step and loop again */
|
|
*(stacks[stackPointer-1].pvar) += stacks[stackPointer-1].step;
|
|
/* counter overflow? */
|
|
if( stacks[stackPointer-1].step > 0 ) {
|
|
if( stacks[stackPointer-1].limit < *(stacks[stackPointer-1].pvar) ) {
|
|
/* loop done */
|
|
stackPointer--;
|
|
return;
|
|
}
|
|
} else {
|
|
if( stacks[stackPointer-1].limit > *(stacks[stackPointer-1].pvar) ) {
|
|
/* loop done */
|
|
stackPointer--;
|
|
return;
|
|
}
|
|
}
|
|
executionPointer = stacks[stackPointer-1].returnPointer;
|
|
}
|
|
|
|
/* endステートメント: 正常な終了 */
|
|
void st_end()
|
|
{
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
}
|
|
|
|
/* breakステートメント: デバッグ用の中断 */
|
|
void st_break()
|
|
{
|
|
commonPrint(NULL, "Break in %d\x7" NEWLINE, currentLineNumber);
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
resumePointer = executionPointer; /* contする場所はbreakの次。breakのみの例外処理 */
|
|
}
|
|
|
|
void st_rem()
|
|
{
|
|
executionPointer = skipToEOL( executionPointer );
|
|
}
|
|
|
|
void st_new()
|
|
{
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_NEW;
|
|
}
|
|
|
|
void st_list()
|
|
{
|
|
WORD from, to;
|
|
BYTE ch;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch == ':' || ch == EOL || ch == '\'' ) {
|
|
executionPointer--; /* unget */
|
|
from = 1;
|
|
to = 32767;
|
|
} else {
|
|
if( ch == 0x01 ) {
|
|
from = *((WORD*)executionPointer);
|
|
executionPointer += 2;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
} else {
|
|
from = 1;
|
|
}
|
|
if( ch != '-' ) {
|
|
executionPointer--; /* unget */
|
|
to = from;
|
|
} else {
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch == 0x01 ) {
|
|
to = *((WORD*)executionPointer);
|
|
executionPointer += 2;
|
|
} else {
|
|
executionPointer--; /* unget */
|
|
to = 32767;
|
|
}
|
|
}
|
|
}
|
|
sourceDump( NULL, from, to ); /* リスト出力の本体を呼ぶ */
|
|
}
|
|
|
|
void st_run()
|
|
{
|
|
BYTE ch;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch == ':' || ch == EOL || ch == '\'' ) {
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_RUN_LINENUMBER;
|
|
runLineNumber = 0;
|
|
} else if( ch == 0x03 ) {
|
|
strcpy( waRawLine, executionPointer );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_RUN_FILE;
|
|
} else {
|
|
SHORT val;
|
|
executionPointer--; /* unget it */
|
|
val = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_RUN_LINENUMBER;
|
|
runLineNumber = val;
|
|
}
|
|
}
|
|
|
|
void st_cont()
|
|
{
|
|
if( resumePointer == NULL ) {
|
|
cantContinue();
|
|
return;
|
|
}
|
|
executionPointer = resumePointer;
|
|
bInteractive = FALSE;
|
|
}
|
|
|
|
void st_save()
|
|
{
|
|
BYTE ch;
|
|
BOOL b;
|
|
FILE FAR * fp;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch != 0x03 ) {
|
|
syntaxError();
|
|
return;
|
|
}
|
|
fp = fopen( executionPointer, "w" );
|
|
if( fp == NULL ) {
|
|
saveError();
|
|
return;
|
|
}
|
|
b = sourceDump( fp, 1, 32767 );
|
|
fclose( fp );
|
|
if( b == FALSE ) {
|
|
saveError();
|
|
remove( executionPointer );
|
|
return;
|
|
}
|
|
|
|
executionPointer += strlen(executionPointer)+1;
|
|
}
|
|
|
|
void st_load()
|
|
{
|
|
BYTE ch;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch == 0x03 ) {
|
|
strcpy( waRawLine, executionPointer );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_LOAD_FILE;
|
|
} else {
|
|
syntaxError();
|
|
return;
|
|
}
|
|
}
|
|
|
|
void st_merge()
|
|
{
|
|
BYTE ch;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch == 0x03 ) {
|
|
strcpy( waRawLine, executionPointer );
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_MERGE_FILE;
|
|
} else {
|
|
syntaxError();
|
|
return;
|
|
}
|
|
}
|
|
|
|
void st_randomize()
|
|
{
|
|
SHORT val;
|
|
val = expr();
|
|
srand( val );
|
|
}
|
|
|
|
void st_exit()
|
|
{
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_EXIT_SYSTEM;
|
|
}
|
|
|
|
void st_waitvb()
|
|
{
|
|
SHORT val;
|
|
val = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
if( val < 0 || val > 750 ) {
|
|
paramError();
|
|
return;
|
|
}
|
|
#ifdef WW
|
|
sys_wait( val );
|
|
#endif
|
|
#ifdef WIN32
|
|
win32_sys_wait( val );
|
|
#endif
|
|
}
|
|
|
|
void st_files()
|
|
{
|
|
#ifdef WIN32
|
|
commonPrint( NULL, "files statement not implemented in Win32\x7" NEWLINE );
|
|
#endif
|
|
#ifdef WW
|
|
int i;
|
|
struct stat statbuf;
|
|
int nent;
|
|
|
|
/*chdir("/rom0");*/
|
|
nent = nument(NULL);
|
|
|
|
for (i = 0; i < nent; i++) {
|
|
int result = getent(NULL, i, &statbuf);
|
|
if (result == E_FS_SUCCESS) {
|
|
if (statbuf.count != -1) {
|
|
/* Freya OSのバグ? 回避コード */
|
|
int p, q;
|
|
p = 0;
|
|
q = 0;
|
|
waRawLine[p++] = '"';
|
|
while( TRUE ) {
|
|
if( statbuf.name[q] == '\0' ) break;
|
|
waRawLine[p++] = statbuf.name[q++];
|
|
}
|
|
waRawLine[p++] = '"';
|
|
waRawLine[p++] = ',';
|
|
q = 0;
|
|
while( TRUE ) {
|
|
if( statbuf.info[q] == '\0' ) break;
|
|
waRawLine[p++] = statbuf.info[q++];
|
|
}
|
|
waRawLine[p++] = '\0';
|
|
commonPrint( NULL, "%s,%ld,%d" NEWLINE, waRawLine, statbuf.len, statbuf.count );
|
|
}
|
|
} else {
|
|
commonPrint( NULL, "*File Access Error\x7" NEWLINE );
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void st_play()
|
|
{
|
|
BYTE ch;
|
|
|
|
ch = *executionPointer++;
|
|
while ( ch == ' ' || ch == '\t' ) {
|
|
ch = *executionPointer++;
|
|
}
|
|
if ( ch != 0x03 ) {
|
|
syntaxError();
|
|
return;
|
|
}
|
|
|
|
#ifdef WIN32
|
|
win32_play_mml( executionPointer );
|
|
#endif
|
|
#ifdef WW
|
|
if ( soundAvailable ) {
|
|
parse_mml( _heap, executionPointer, 0 );
|
|
bgm_play( _heap, PLAY_SINGLE );
|
|
}
|
|
#endif
|
|
|
|
executionPointer += strlen(executionPointer)+1;
|
|
}
|
|
|
|
void st_poke()
|
|
{
|
|
SHORT addr, data;
|
|
BYTE ch;
|
|
BYTE FAR * fulladdr;
|
|
addr = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
while( TRUE ) {
|
|
ch = *executionPointer++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch != ',' ) {
|
|
syntaxError();
|
|
return;
|
|
}
|
|
data = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
fulladdr = (BYTE FAR *)(((DWORD)defseg << 16) + (WORD)addr);
|
|
#ifdef WW
|
|
*fulladdr = data;
|
|
#else
|
|
printf("poke: %08x %02x\n", fulladdr, data );
|
|
#endif
|
|
}
|
|
|
|
void st_call()
|
|
{
|
|
SHORT addr;
|
|
BYTE FAR * fulladdr;
|
|
addr = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
fulladdr = (BYTE FAR *)(((DWORD)defseg << 16) + (WORD)addr);
|
|
#ifdef WW
|
|
#ifdef VC15
|
|
// VC++1.5のときに使うコード
|
|
__asm {
|
|
push si
|
|
push di
|
|
push bp
|
|
mov ax,registers[(KEYWORD_AX-KEYWORD_AX)*2]
|
|
mov bx,registers[(KEYWORD_BX-KEYWORD_AX)*2]
|
|
mov cx,registers[(KEYWORD_CX-KEYWORD_AX)*2]
|
|
mov dx,registers[(KEYWORD_DX-KEYWORD_AX)*2]
|
|
mov si,registers[(KEYWORD_SI-KEYWORD_AX)*2]
|
|
mov di,registers[(KEYWORD_DI-KEYWORD_AX)*2]
|
|
mov es,registers[(KEYWORD_ES-KEYWORD_AX)*2]
|
|
push ds
|
|
mov ds,registers[(KEYWORD_DS-KEYWORD_AX)*2]
|
|
call far ptr ss:[fulladdr]
|
|
pop ds
|
|
mov registers[(KEYWORD_AX-KEYWORD_AX)*2],ax
|
|
mov registers[(KEYWORD_BX-KEYWORD_AX)*2],bx
|
|
mov registers[(KEYWORD_CX-KEYWORD_AX)*2],cx
|
|
mov registers[(KEYWORD_DX-KEYWORD_AX)*2],dx
|
|
mov registers[(KEYWORD_SI-KEYWORD_AX)*2],si
|
|
mov registers[(KEYWORD_DI-KEYWORD_AX)*2],di
|
|
mov registers[(KEYWORD_ES-KEYWORD_AX)*2],es
|
|
pop bp
|
|
pop di
|
|
pop si
|
|
}
|
|
#else
|
|
// VC++ 1.5以外の処理系のためのコードは作っていません。欲しい人、自作しましょう
|
|
#endif
|
|
#else
|
|
printf("call: %08x\n", fulladdr );
|
|
#endif
|
|
}
|
|
|
|
void st_int()
|
|
{
|
|
SHORT number;
|
|
number = expr();
|
|
if( bForceToReturnSuper ) return;
|
|
#ifdef WW
|
|
#ifdef VC15
|
|
// VC++1.5のときに使うコード
|
|
{
|
|
static BYTE specialIntCallCode[3] = { 0xcd, 0x00, 0xcb };
|
|
DWORD specialPtr;
|
|
specialIntCallCode[1] = (BYTE)number;
|
|
specialPtr = (((DWORD)getdseg() << 16) + (WORD)specialIntCallCode);
|
|
__asm {
|
|
push si
|
|
push di
|
|
push bp
|
|
mov ax,registers[(KEYWORD_AX-KEYWORD_AX)*2]
|
|
mov bx,registers[(KEYWORD_BX-KEYWORD_AX)*2]
|
|
mov cx,registers[(KEYWORD_CX-KEYWORD_AX)*2]
|
|
mov dx,registers[(KEYWORD_DX-KEYWORD_AX)*2]
|
|
mov si,registers[(KEYWORD_SI-KEYWORD_AX)*2]
|
|
mov di,registers[(KEYWORD_DI-KEYWORD_AX)*2]
|
|
mov es,registers[(KEYWORD_ES-KEYWORD_AX)*2]
|
|
push ds
|
|
mov ds,registers[(KEYWORD_DS-KEYWORD_AX)*2]
|
|
call far ptr ss:[specialPtr]
|
|
pop ds
|
|
mov registers[(KEYWORD_AX-KEYWORD_AX)*2],ax
|
|
mov registers[(KEYWORD_BX-KEYWORD_AX)*2],bx
|
|
mov registers[(KEYWORD_CX-KEYWORD_AX)*2],cx
|
|
mov registers[(KEYWORD_DX-KEYWORD_AX)*2],dx
|
|
mov registers[(KEYWORD_SI-KEYWORD_AX)*2],si
|
|
mov registers[(KEYWORD_DI-KEYWORD_AX)*2],di
|
|
mov registers[(KEYWORD_ES-KEYWORD_AX)*2],es
|
|
pop bp
|
|
pop di
|
|
pop si
|
|
}
|
|
}
|
|
#else
|
|
// VC++ 1.5以外の処理系のためのコードは作っていません。欲しい人、自作しましょう
|
|
#endif
|
|
#else
|
|
printf("int: %04x\n", number );
|
|
#endif
|
|
}
|
|
|
|
void st_tron()
|
|
{
|
|
traceFlag = TRUE;
|
|
}
|
|
|
|
void st_troff()
|
|
{
|
|
traceFlag = FALSE;
|
|
}
|
|
|
|
/* 行エディタ */
|
|
void editLine()
|
|
{
|
|
WORD wishLineNumber;
|
|
BYTE ch, * p, * target;
|
|
WORD lineNumber;
|
|
|
|
// プログラム領域に行を格納する場合は
|
|
// 先頭に2バイトの行番号と2バイトの行長さの情報が格納される
|
|
// 一方、中間言語翻訳の結果は先頭に0x01で始まる10進整数が行番号として入っている
|
|
// このギャップを転送時に解消しなければならない
|
|
|
|
// ここに来ているということは、waCookedLine[0]が0x01つまり10進整数である //
|
|
assert( waCookedLine[0] == 0x01 );
|
|
wishLineNumber = *((WORD*)&waCookedLine[1]);
|
|
|
|
/* 行番号 0 はだめ */
|
|
if ( wishLineNumber == 0 ) {
|
|
lineNumberZeroError();
|
|
return;
|
|
}
|
|
|
|
p = waCookedLine+3;
|
|
//target = getLineReferenceFromLineNumber( wishLineNumber );
|
|
target = getInsertionPointFromLineNumber( wishLineNumber );
|
|
lineNumber = *((WORD*)target);
|
|
|
|
while( TRUE ) {
|
|
ch = *p++;
|
|
if( ch != ' ' && ch != '\t' ) break;
|
|
}
|
|
if( ch == EOL ) {
|
|
BYTE * from;
|
|
/* removing the line */
|
|
if( lineNumber == 0 || lineNumber != wishLineNumber ) {
|
|
lineNumberNotFound( wishLineNumber );
|
|
return;
|
|
}
|
|
from = skipToNextLine(target);
|
|
MEMMOVE( target, from, (WORD)(dataTop-(from-wa)) );
|
|
dataTop -= from-target;
|
|
} else {
|
|
WORD len; // プログラム領域上に収めるときのサイズ
|
|
len = skipToEOL(p)-waCookedLine-1+2+1; //-1は0x01の分。+2は長さ情報。+1はEOL
|
|
if( lineNumber == 0 || lineNumber != wishLineNumber ) {
|
|
/* insert new line */
|
|
if( dataTop + len >= WORKAREA_SIZE ) {
|
|
outOfMemory();
|
|
return;
|
|
}
|
|
MEMMOVE( target+len, target, (WORD)(dataTop-(target-wa)) );
|
|
dataTop += len;
|
|
} else {
|
|
/* replace line */
|
|
WORD lost;
|
|
BYTE * nextline;
|
|
int delta;
|
|
nextline = skipToNextLine(target);
|
|
lost = nextline-target;
|
|
delta = len-lost;
|
|
if( dataTop + delta >= WORKAREA_SIZE ) {
|
|
outOfMemory();
|
|
return;
|
|
}
|
|
MEMMOVE( nextline+delta, nextline, (WORD)(dataTop-(nextline-wa)) );
|
|
dataTop += delta;
|
|
}
|
|
// まず行番号をコピーする
|
|
*((WORD*)target) = *((WORD*)&waCookedLine[1]);
|
|
// 行の長さを収める
|
|
*((WORD*)(target+2)) = len;
|
|
// 残りをコピーする
|
|
MEMMOVE( target+4, waCookedLine+3, (WORD)(len-4) );
|
|
}
|
|
}
|
|
|
|
/* 追加専門の行エディタ */
|
|
void appendLine()
|
|
{
|
|
BYTE * target;
|
|
WORD len; // プログラム領域上に収めるときのサイズ
|
|
len = skipToEOL(waCookedLine+3)-waCookedLine-1+2+1; //-1は0x01の分。+2は長さ情報。+1はEOL
|
|
if( dataTop + len >= WORKAREA_SIZE ) {
|
|
outOfMemory();
|
|
return;
|
|
}
|
|
target = &wa[dataTop-5]; // -5 is len of last-line-marker
|
|
// まず行番号をコピーする
|
|
*((WORD*)target) = *((WORD*)&waCookedLine[1]);
|
|
// 行の長さを収める
|
|
*((WORD*)(target+2)) = len;
|
|
// 残りをコピーする
|
|
MEMMOVE( target+4, waCookedLine+3, (WORD)(len-4) );
|
|
dataTop += len;
|
|
wa[dataTop-5] = 0;
|
|
wa[dataTop-4] = 0;
|
|
wa[dataTop-3] = 0;
|
|
wa[dataTop-2] = 0;
|
|
wa[dataTop-1] = EOL;
|
|
}
|
|
|
|
/* 中間言語に翻訳する */
|
|
BOOL convertInternalCode( BYTE * waCookedLine, const BYTE * waRawLine )
|
|
{
|
|
const BYTE * src = waRawLine;
|
|
BYTE * dst = waCookedLine;
|
|
while( TRUE ) {
|
|
if( *src == '\0' ) break;
|
|
if( *src == '\n' ) break;
|
|
if( *src == '\r' ) break;
|
|
if( *src == ' ' || *src == '\t' ) {
|
|
*dst++ = *src++;
|
|
} else if( *src < 0x20 ) {
|
|
syntaxError();
|
|
return FALSE;
|
|
} else if( *src >= '0' && *src <= '9' ) {
|
|
if( *src == '0' && *(src+1) == 'x' ) {
|
|
WORD acc;
|
|
/* 16進のとき */
|
|
*dst++ = 0x02;
|
|
acc = 0;
|
|
src += 2;
|
|
while( TRUE ) {
|
|
if( *src >= '0' && *src <= '9' ) {
|
|
acc *= 16;
|
|
acc += *src - '0';
|
|
} else if( *src >= 'a' && *src <= 'f' ) {
|
|
acc *= 16;
|
|
acc += *src - 'a' + 10;
|
|
} else if( *src >= 'A' && *src <= 'F' ) {
|
|
acc *= 16;
|
|
acc += *src - 'A' + 10;
|
|
} else {
|
|
break;
|
|
}
|
|
src++;
|
|
}
|
|
*((WORD*)dst) = acc;
|
|
dst += 2;
|
|
} else {
|
|
SHORT acc;
|
|
/* 10進のとき */
|
|
*dst++ = 0x01;
|
|
acc = *src-'0';
|
|
src++;
|
|
while( TRUE ) {
|
|
if( *src < '0' || *src > '9' ) break;
|
|
acc *= 10;
|
|
acc += *src - '0';
|
|
src++;
|
|
if( acc < 0 ) { /* overflow case */
|
|
syntaxError();
|
|
return FALSE;
|
|
}
|
|
}
|
|
*((SHORT*)dst) = acc;
|
|
dst += 2;
|
|
}
|
|
#if 0
|
|
/* 以下のコードは、2項演算子の-を負数と解釈することがあるので使えない */
|
|
} else if( *src == '-' && (*(src+1) >= '0' && *(src+1) <= '9' ) ) {
|
|
WORD acc;
|
|
/* 負数の10進のとき */
|
|
*dst++ = 0x01;
|
|
src++;
|
|
acc = *src-'0';
|
|
src++;
|
|
while( TRUE ) {
|
|
if( *src < '0' || *src > '9' ) break;
|
|
acc *= 10;
|
|
acc += *src - '0';
|
|
src++;
|
|
if( acc > 32768 ) { /* overflow case */
|
|
syntaxError();
|
|
return FALSE;
|
|
}
|
|
}
|
|
*((SHORT*)dst) = -((SHORT)acc);
|
|
dst += 2;
|
|
#endif
|
|
} else if( (*src >= 'a' && *src <= 'z') || (*src >= 'A' && *src <= 'Z') ) {
|
|
BYTE next = *(src+1);
|
|
if( (next >= 'a' && next <= 'z') || (next >= 'A' && next <= 'Z') ) {
|
|
WORD id;
|
|
KEYWORDITEM * p = keywordsIndex[mytolower(*src)-'a'];
|
|
if( p == NULL ) {
|
|
syntaxError();
|
|
return FALSE;
|
|
}
|
|
id = 0;
|
|
while( TRUE ) {
|
|
const BYTE * ps, * pd;
|
|
if( p->id == 0 ) {
|
|
syntaxError();
|
|
return FALSE;
|
|
}
|
|
ps = src;
|
|
pd = p->name;
|
|
while( TRUE ) {
|
|
if( *pd == '\0' ) {
|
|
id = p->id;
|
|
src = ps;
|
|
break;
|
|
}
|
|
if( mytolower(*ps) != mytolower(*pd) ) break;
|
|
ps++;
|
|
pd++;
|
|
}
|
|
if( id != 0 ) break;
|
|
p++;
|
|
}
|
|
*dst++ = (BYTE)id;
|
|
if( id == KEYWORD_REM ) {
|
|
/* コメント文の場合は行末まで何もかもそのままコピーする */
|
|
while( TRUE ) {
|
|
if( *src == '\0' ) break;
|
|
*dst++ = *src++;
|
|
}
|
|
}
|
|
} else {
|
|
*dst++ = *src++;
|
|
}
|
|
} else if( *src == '\'' ) {
|
|
*dst++ = *src++;
|
|
/* コメント文の場合は行末まで何もかもそのままコピーする */
|
|
while( TRUE ) {
|
|
if( *src == '\0' ) break;
|
|
*dst++ = *src++;
|
|
}
|
|
} else if( *src == '"' ) {
|
|
src++;
|
|
*dst++ = 0x03;
|
|
while( TRUE ) {
|
|
if( *src == '\0' ) {
|
|
syntaxError();
|
|
return FALSE;
|
|
}
|
|
if( *src == '"' ) break;
|
|
*dst++ = *src++;
|
|
}
|
|
*dst++ = '\0';
|
|
src++;
|
|
} else {
|
|
*dst++ = *src++;
|
|
}
|
|
}
|
|
*dst = EOL;
|
|
return TRUE;
|
|
}
|
|
|
|
/* ステートメント実行処理関数のリスト */
|
|
|
|
NEARPROC statements[KEYWORDS_STATEMENT_TO-KEYWORDS_STATEMENT_FROM+1] = {
|
|
st_if,
|
|
st_print,
|
|
st_locate,
|
|
st_cls,
|
|
st_goto,
|
|
st_gosub,
|
|
st_return,
|
|
st_for,
|
|
st_next,
|
|
st_end,
|
|
st_break,
|
|
st_rem,
|
|
st_new,
|
|
st_list,
|
|
st_run,
|
|
st_cont,
|
|
st_save,
|
|
st_load,
|
|
st_merge,
|
|
st_randomize,
|
|
st_exit,
|
|
st_debug,
|
|
st_waitvb,
|
|
st_files,
|
|
st_play,
|
|
st_poke,
|
|
st_call,
|
|
st_int,
|
|
st_tron,
|
|
st_troff
|
|
};
|
|
|
|
void interpreterMain()
|
|
{
|
|
WORD keyscan;
|
|
while( TRUE ) {
|
|
/* 行の開始 */
|
|
processLineHeader();
|
|
/* 最後に達してしまった? */
|
|
if( !bInteractive && currentLineNumber == 0 ) {
|
|
bForceToReturnSuper = TRUE;
|
|
requestNextAction = REQUEST_INTERACTIVE;
|
|
return;
|
|
}
|
|
|
|
while( TRUE ) {
|
|
BYTE ch;
|
|
if( !bInteractive ) {
|
|
resumePointer = executionPointer; /* contする場所を記憶する */
|
|
}
|
|
|
|
#ifdef WW
|
|
keyscan = key_press_check();
|
|
#else
|
|
keyscan = win32_scan();
|
|
#endif
|
|
if( keyscan & 0x02 ) {
|
|
st_break();
|
|
return;
|
|
}
|
|
|
|
ch = *executionPointer++;
|
|
if( ch == EOL ) break;
|
|
if( ch == ' ' || ch == '\t' || ch == ':' ) {
|
|
/* nop */
|
|
} else if( ch == '\'' ) { /* comment */
|
|
st_rem();
|
|
} else if( ch == '@' ) { /* is it l-value? */
|
|
SHORT * pvar;
|
|
pvar = getArrayReference();
|
|
if( pvar == NULL ) return;
|
|
st_assignment( pvar );
|
|
} else if( ch >= 'A' && ch <= 'Z' ) {
|
|
st_assignment( &globalVariables[ch-'A'] );
|
|
} else if( ch >= 'a' && ch <= 'z' ) {
|
|
st_assignment( &localVariables[ch-'a'] );
|
|
} else if( ch >= KEYWORD_AX && ch <= KEYWORD_ES ) {
|
|
st_assignment( ®isters[ch-KEYWORD_AX] );
|
|
} else if( ch == KEYWORD_DEFSEG ) {
|
|
st_assignment( &defseg );
|
|
} else if( ch >= KEYWORDS_STATEMENT_FROM && ch <= KEYWORDS_STATEMENT_TO ) {
|
|
(*(statements[ch-KEYWORDS_STATEMENT_FROM]))();
|
|
} else if( ch == '?' ) {
|
|
st_debug();
|
|
} else {
|
|
syntaxError();
|
|
}
|
|
if( bForceToReturnSuper ) return;
|
|
}
|
|
if( bInteractive ) return;
|
|
}
|
|
}
|
|
|
|
void interactiveMain( FILE FAR * fp )
|
|
{
|
|
if( fp == NULL ) {
|
|
commonPrint(NULL,"OK" NEWLINE);
|
|
}
|
|
while( TRUE ) {
|
|
BOOL b;
|
|
WORD l;
|
|
if( fp == NULL ) {
|
|
consoleInput( waRawLine, LINE_SIZE );
|
|
} else {
|
|
char FAR * r;
|
|
r = fgets( waRawLine, LINE_SIZE, fp );
|
|
if( r == NULL ) break;
|
|
/* ファイル中に LF があると残ってしまうのを消す */
|
|
l = strlen(waRawLine)-1;
|
|
if ( waRawLine[l] == '\x0a' ){
|
|
waRawLine[l] = '\0';
|
|
}
|
|
}
|
|
#ifdef WW
|
|
if( fp == NULL ) {
|
|
commonPrint( NULL, NEWLINE );
|
|
}
|
|
#endif
|
|
/* 中間言語に翻訳する */
|
|
b = convertInternalCode( waCookedLine, waRawLine );
|
|
if( b == FALSE ) {
|
|
bForceToReturnSuper = FALSE;
|
|
continue;
|
|
}
|
|
/* 数値で開始されているか? */
|
|
if( waCookedLine[0] == 0x01 ) {
|
|
/* 行エディタを呼び出す */
|
|
if( appendMode ) {
|
|
appendLine();
|
|
} else {
|
|
editLine();
|
|
}
|
|
if( fp == NULL ) {
|
|
clearRuntimeInfo();
|
|
}
|
|
} else {
|
|
/* その行を実行する */
|
|
executionPointer = waCookedLine;
|
|
interpreterMain();
|
|
}
|
|
if( bForceToReturnSuper ) return;
|
|
}
|
|
}
|
|
|
|
/* プログラムの実行開始 */
|
|
void do_run( WORD runLineNumber )
|
|
{
|
|
BYTE * target;
|
|
if( runLineNumber == 0 ) {
|
|
target = &wa[codeTop];
|
|
} else {
|
|
target = getLineReferenceFromLineNumber( runLineNumber );
|
|
if( target == NULL ) {
|
|
lineNumberNotFound( runLineNumber );
|
|
return;
|
|
}
|
|
}
|
|
clearRuntimeInfo();
|
|
bInteractive = FALSE;
|
|
executionPointer = target;
|
|
}
|
|
|
|
void do_new()
|
|
{
|
|
clearRuntimeInfo();
|
|
bInteractive = TRUE;
|
|
traceFlag = FALSE;
|
|
codeTop = 0;
|
|
wa[0] = 0;
|
|
wa[1] = 0;
|
|
wa[2] = 0;
|
|
wa[3] = 0;
|
|
wa[4] = EOL;
|
|
dataTop = 5;
|
|
}
|
|
|
|
BOOL do_merge( const BYTE * filename )
|
|
{
|
|
FILE FAR * fp;
|
|
fp = fopen( filename, "r" );
|
|
if( fp == NULL ) {
|
|
loadError();
|
|
return FALSE;
|
|
}
|
|
interactiveMain( fp );
|
|
fclose( fp );
|
|
clearRuntimeInfo();
|
|
if( bForceToReturnSuper ) return FALSE;
|
|
return TRUE;
|
|
}
|
|
|
|
void superMain()
|
|
{
|
|
while( TRUE ) {
|
|
BOOL b;
|
|
bForceToReturnSuper = FALSE;
|
|
switch( requestNextAction ) {
|
|
case REQUEST_INTERACTIVE:
|
|
bInteractive = TRUE;
|
|
break;
|
|
case REQUEST_RUN_LINENUMBER:
|
|
do_run( runLineNumber );
|
|
break;
|
|
case REQUEST_RUN_FILE:
|
|
do_new();
|
|
appendMode=TRUE;
|
|
b = do_merge( waRawLine );
|
|
appendMode=FALSE;
|
|
if( b ) {
|
|
do_run( 0 );
|
|
}
|
|
break;
|
|
case REQUEST_LOAD_FILE:
|
|
do_new();
|
|
appendMode=TRUE;
|
|
do_merge( waRawLine );
|
|
appendMode=FALSE;
|
|
break;
|
|
case REQUEST_MERGE_FILE:
|
|
do_merge( waRawLine );
|
|
break;
|
|
case REQUEST_NEW:
|
|
do_new();
|
|
break;
|
|
case REQUEST_EXIT_SYSTEM:
|
|
return;
|
|
}
|
|
requestNextAction = REQUEST_NO_ACTION;
|
|
if( bInteractive ) {
|
|
if( runtimeOnly ) return; /* ランタイムモードのときは絶対にインタラクティブモードに行かない */
|
|
interactiveMain( NULL );
|
|
} else {
|
|
interpreterMain();
|
|
}
|
|
bForceToReturnSuper = FALSE;
|
|
}
|
|
}
|
|
|
|
#ifdef WW
|
|
BOOL entranceUI()
|
|
{
|
|
#define MAX_ENT 15
|
|
int i;
|
|
struct stat statbuf[MAX_ENT];
|
|
int nent;
|
|
int c, select;
|
|
|
|
ptextCLS();
|
|
ptextString( "ワンべぇ V" );
|
|
ptextString( myVersion );
|
|
ptextNewline();
|
|
ptextString( "↑↓Aでファイル選択 Y2で開発環境 STARTで中止" );
|
|
|
|
c = 0;
|
|
nent = nument(NULL);
|
|
|
|
for (i = 0; i < nent; i++) {
|
|
int result;
|
|
if( c >= MAX_ENT ) break;
|
|
result = getent(NULL, i, &statbuf[c]);
|
|
if (result == E_FS_SUCCESS) {
|
|
if (statbuf[c].count != -1) {
|
|
char FAR * p;
|
|
int l;
|
|
p = statbuf[c].name;
|
|
l = strlen(p);
|
|
if( l >= 3 ) {
|
|
if( p[l-3] == '.' && p[l-2] == 'w' && p[l-1] == 'b' ) {
|
|
ptextLocate(1,c+2);
|
|
ptextString( statbuf[c].info );
|
|
c++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
/* 選択の開始 */
|
|
select = 0;
|
|
while( TRUE ) {
|
|
int key;
|
|
for( i=0; i<c; i++ ) {
|
|
ptextLocate(0,i+2);
|
|
ptextString( " " );
|
|
}
|
|
if( c > 0 ) {
|
|
ptextLocate(0,select+2);
|
|
ptextString( "→" );
|
|
}
|
|
key = key_wait();
|
|
/* START button */
|
|
if( key & 0x02 ) return FALSE; /* 中止 */
|
|
/* B button */
|
|
if( key & 0x08 ) return FALSE; /* 中止 */
|
|
if( c > 0 ) {
|
|
/* A button */
|
|
if( key & 0x04 ) {
|
|
strcpy( waRawLine, statbuf[select].name );
|
|
requestNextAction = REQUEST_RUN_FILE;
|
|
runtimeOnly = TRUE;
|
|
return TRUE;
|
|
}
|
|
/* X1 or Y1 */
|
|
if( (key & 0x10) != 0 || (key & 0x100) != 0 ) {
|
|
select = select-1;
|
|
if( select < 0 ) {
|
|
select = c-1;
|
|
}
|
|
}
|
|
/* X3 or Y3 */
|
|
if( (key & 0x40) != 0 || (key & 0x400) != 0 ) {
|
|
select = select+1;
|
|
if( select >= c ) {
|
|
select = 0;
|
|
}
|
|
}
|
|
}
|
|
/* Y2 button */
|
|
if( (key & 0x200) != 0 ) {
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
int main( int argc, char *argv[] )
|
|
{
|
|
#ifdef WW
|
|
BOOL b;
|
|
#endif
|
|
runtimeOnly = FALSE;
|
|
requestNextAction = REQUEST_NO_ACTION;
|
|
#ifdef WIN32
|
|
createText();
|
|
#endif
|
|
#ifdef WW
|
|
#ifdef CURRENT_DIR_IS_RAM0
|
|
// シンボルCURRENT_DIR_IS_RAM0を定義してビルドすると
|
|
// カレントディレクトリは/ram0で走る
|
|
// 定義しなければ/rom0で走る
|
|
chdir("/ram0");
|
|
#endif
|
|
text_screen_init();
|
|
b = entranceUI();
|
|
if( b == FALSE ) return 0;
|
|
if( !runtimeOnly ) {
|
|
comm_set_baudrate(1);
|
|
comm_open();
|
|
}
|
|
|
|
/* PLAY statement support */
|
|
soundAvailable = FALSE;
|
|
if (open_sound_il(&soundIL) == E_FS_SUCCESS) {
|
|
soundAvailable = TRUE;
|
|
// sounddrv_init(); いらないならわざわざ呼ばない。
|
|
sound_open();
|
|
}
|
|
#endif
|
|
|
|
ptextCLS();
|
|
if( requestNextAction == REQUEST_NO_ACTION ) {
|
|
do_new();
|
|
}
|
|
if( !runtimeOnly ) {
|
|
commonPrint(NULL,"ワンべぇ WonderWitch Tiny BASIC Environment Ver %s" NEWLINE,myVersion);
|
|
commonPrint(NULL,"Authors: Pie Dey Co.,Ltd., Hirotaka JOE Ohkubo" NEWLINE);
|
|
commonPrint(NULL,"This program is distributed under GNU General Public License" NEWLINE);
|
|
}
|
|
superMain();
|
|
#ifdef WIN32
|
|
deleteText();
|
|
#endif
|
|
#ifdef WW
|
|
/* PLAY statement support */
|
|
if (soundAvailable) {
|
|
sound_close();
|
|
sounddrv_release();
|
|
}
|
|
if( !runtimeOnly ) {
|
|
comm_close();
|
|
}
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
/* end of wonbe.c */
|