ref: 7d02e382d314d5bdde7978ccb7a64ea9201d03db
dir: /primitives.c/
/******************************************************************* ** w o r d s . c ** Forth Inspired Command Language ** ANS Forth CORE word-set written in C ** Author: John Sadler ([email protected]) ** Created: 19 July 1997 ** $Id: primitives.c,v 1.7 2010/12/02 13:56:43 asau Exp $ *******************************************************************/ /* ** Copyright (c) 1997-2001 John Sadler ([email protected]) ** All rights reserved. ** ** Get the latest Ficl release at http://ficl.sourceforge.net ** ** I am interested in hearing from anyone who uses Ficl. If you have ** a problem, a success story, a defect, an enhancement request, or ** if you would like to contribute to the Ficl release, please ** contact me by email at the address above. ** ** L I C E N S E and D I S C L A I M E R ** ** 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. */ #include <stdlib.h> #include <stdio.h> #include <string.h> #include <ctype.h> #include "ficl.h" /* ** Control structure building words use these ** strings' addresses as markers on the stack to ** check for structure completion. */ static char doTag[] = "do"; static char colonTag[] = "colon"; static char leaveTag[] = "leave"; static char destTag[] = "target"; static char origTag[] = "origin"; static char caseTag[] = "case"; static char ofTag[] = "of"; static char fallthroughTag[] = "fallthrough"; /* ** C O N T R O L S T R U C T U R E B U I L D E R S ** ** Push current dictionary location for later branch resolution. ** The location may be either a branch target or a patch address... */ static void markBranch(ficlDictionary *dictionary, ficlVm *vm, char *tag) { ficlStackPushPointer(vm->dataStack, dictionary->here); ficlStackPushPointer(vm->dataStack, tag); return; } static void markControlTag(ficlVm *vm, char *tag) { ficlStackPushPointer(vm->dataStack, tag); return; } static void matchControlTag(ficlVm *vm, char *wantTag) { char *tag; FICL_STACK_CHECK(vm->dataStack, 1, 0); tag = (char *)ficlStackPopPointer(vm->dataStack); /* ** Changed the code below to compare the pointers first (by popular demand) */ if ( (tag != wantTag) && strcmp(tag, wantTag) ) { ficlVmThrowError(vm, "Error -- unmatched control structure \"%s\"", wantTag); } return; } /* ** Expect a branch target address on the param stack, ** FICL_VM_STATE_COMPILE a literal offset from the current dictionary location ** to the target address */ static void resolveBackBranch(ficlDictionary *dictionary, ficlVm *vm, char *tag) { ficlInteger offset; ficlCell *patchAddr; matchControlTag(vm, tag); FICL_STACK_CHECK(vm->dataStack, 1, 0); patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack); offset = patchAddr - dictionary->here; ficlDictionaryAppendCell(dictionary, FICL_LVALUE_TO_CELL(offset)); return; } /* ** Expect a branch patch address on the param stack, ** FICL_VM_STATE_COMPILE a literal offset from the patch location ** to the current dictionary location */ static void resolveForwardBranch(ficlDictionary *dictionary, ficlVm *vm, char *tag) { ficlInteger offset; ficlCell *patchAddr; matchControlTag(vm, tag); FICL_STACK_CHECK(vm->dataStack, 1, 0); patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack); offset = dictionary->here - patchAddr; *patchAddr = FICL_LVALUE_TO_CELL(offset); return; } /* ** Match the tag to the top of the stack. If success, ** sopy "here" address into the ficlCell whose address is next ** on the stack. Used by do..leave..loop. */ static void resolveAbsBranch(ficlDictionary *dictionary, ficlVm *vm, char *wantTag) { ficlCell *patchAddr; char *tag; FICL_STACK_CHECK(vm->dataStack, 2, 0); tag = (char *)ficlStackPopPointer(vm->dataStack); /* ** Changed the comparison below to compare the pointers first (by popular demand) */ if ((tag != wantTag) && strcmp(tag, wantTag)) { ficlVmTextOut(vm, "Warning -- Unmatched control word: "); ficlVmTextOut(vm, wantTag); ficlVmTextOut(vm, "\n"); } patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack); *patchAddr = FICL_LVALUE_TO_CELL(dictionary->here); return; } /************************************************************************** c o l o n d e f i n i t i o n s ** Code to begin compiling a colon definition ** This function sets the state to FICL_VM_STATE_COMPILE, then creates a ** new word whose name is the next word in the input stream ** and whose code is colonParen. **************************************************************************/ static void ficlPrimitiveColon(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlString name = ficlVmGetWord(vm); vm->state = FICL_VM_STATE_COMPILE; markControlTag(vm, colonTag); ficlDictionaryAppendWord(dictionary, name, (ficlPrimitive)ficlInstructionColonParen, FICL_WORD_DEFAULT | FICL_WORD_SMUDGED); #if FICL_WANT_LOCALS vm->callback.system->localsCount = 0; #endif return; } static void ficlPrimitiveSemicolonCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); matchControlTag(vm, colonTag); #if FICL_WANT_LOCALS if (vm->callback.system->localsCount > 0) { ficlDictionary *locals = ficlSystemGetLocals(vm->callback.system); ficlDictionaryEmpty(locals, locals->forthWordlist->size); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionUnlinkParen); } vm->callback.system->localsCount = 0; #endif ficlDictionaryAppendUnsigned(dictionary, ficlInstructionSemiParen); vm->state = FICL_VM_STATE_INTERPRET; ficlDictionaryUnsmudge(dictionary); return; } /************************************************************************** e x i t ** CORE ** This function simply pops the previous instruction ** pointer and returns to the "next" loop. Used for exiting from within ** a definition. Note that exitParen is identical to semiParen - they ** are in two different functions so that "see" can correctly identify ** the end of a colon definition, even if it uses "exit". **************************************************************************/ static void ficlPrimitiveExitCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); FICL_IGNORE(vm); #if FICL_WANT_LOCALS if (vm->callback.system->localsCount > 0) { ficlDictionaryAppendUnsigned(dictionary, ficlInstructionUnlinkParen); } #endif ficlDictionaryAppendUnsigned(dictionary, ficlInstructionExitParen); return; } /************************************************************************** c o n s t a n t ** IMMEDIATE ** Compiles a constant into the dictionary. Constants return their ** value when invoked. Expects a value on top of the parm stack. **************************************************************************/ static void ficlPrimitiveConstant(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlString name = ficlVmGetWord(vm); FICL_STACK_CHECK(vm->dataStack, 1, 0); ficlDictionaryAppendConstantInstruction(dictionary, name, ficlInstructionConstantParen, ficlStackPopInteger(vm->dataStack)); return; } static void ficlPrimitive2Constant(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlString name = ficlVmGetWord(vm); FICL_STACK_CHECK(vm->dataStack, 2, 0); ficlDictionaryAppend2ConstantInstruction(dictionary, name, ficlInstruction2ConstantParen, ficlStackPop2Integer(vm->dataStack)); return; } /************************************************************************** d i s p l a y C e l l ** Drop and print the contents of the ficlCell at the top of the param ** stack **************************************************************************/ static void ficlPrimitiveDot(ficlVm *vm) { ficlCell c; FICL_STACK_CHECK(vm->dataStack, 1, 0); c = ficlStackPop(vm->dataStack); ficlLtoa((c).i, vm->pad, vm->base); strcat(vm->pad, " "); ficlVmTextOut(vm, vm->pad); return; } static void ficlPrimitiveUDot(ficlVm *vm) { ficlUnsigned u; FICL_STACK_CHECK(vm->dataStack, 1, 0); u = ficlStackPopUnsigned(vm->dataStack); ficlUltoa(u, vm->pad, vm->base); strcat(vm->pad, " "); ficlVmTextOut(vm, vm->pad); return; } static void ficlPrimitiveHexDot(ficlVm *vm) { ficlUnsigned u; FICL_STACK_CHECK(vm->dataStack, 1, 0); u = ficlStackPopUnsigned(vm->dataStack); ficlUltoa(u, vm->pad, 16); strcat(vm->pad, " "); ficlVmTextOut(vm, vm->pad); return; } /************************************************************************** s t r l e n ** Ficl ( c-string -- length ) ** ** Returns the length of a C-style (zero-terminated) string. ** ** --lch **/ static void ficlPrimitiveStrlen(ficlVm *vm) { char *address = (char *)ficlStackPopPointer(vm->dataStack); ficlStackPushInteger(vm->dataStack, strlen(address)); } /************************************************************************** s p r i n t f ** Ficl ( i*x c-addr-fmt u-fmt c-addr-buffer u-buffer -- c-addr-buffer u-written success-flag ) ** Similar to the C sprintf() function. It formats into a buffer based on ** a "format" string. Each character in the format string is copied verbatim ** to the output buffer, until SPRINTF encounters a percent sign ("%"). ** SPRINTF then skips the percent sign, and examines the next character ** (the "format character"). Here are the valid format characters: ** s - read a C-ADDR U-LENGTH string from the stack and copy it to ** the buffer ** d - read a ficlCell from the stack, format it as a string (base-10, ** signed), and copy it to the buffer ** x - same as d, except in base-16 ** u - same as d, but unsigned ** % - output a literal percent-sign to the buffer ** SPRINTF returns the c-addr-buffer argument unchanged, the number of bytes ** written, and a flag indicating whether or not it ran out of space while ** writing to the output buffer (FICL_TRUE if it ran out of space). ** ** If SPRINTF runs out of space in the buffer to store the formatted string, ** it still continues parsing, in an effort to preserve your stack (otherwise ** it might leave uneaten arguments behind). ** ** --lch **************************************************************************/ static void ficlPrimitiveSprintf(ficlVm *vm) /* */ { int bufferLength = ficlStackPopInteger(vm->dataStack); char *buffer = (char *)ficlStackPopPointer(vm->dataStack); char *bufferStart = buffer; int formatLength = ficlStackPopInteger(vm->dataStack); char *format = (char *)ficlStackPopPointer(vm->dataStack); char *formatStop = format + formatLength; int base = 10; int unsignedInteger = 0; /* false */ int append = 1; /* true */ while (format < formatStop) { char scratch[64]; char *source; int actualLength; int desiredLength; int leadingZeroes; if (*format != '%') { source = format; actualLength = desiredLength = 1; leadingZeroes = 0; } else { format++; if (format == formatStop) break; leadingZeroes = (*format == '0'); if (leadingZeroes) { format++; if (format == formatStop) break; } desiredLength = isdigit((unsigned char)*format); if (desiredLength) { desiredLength = strtoul(format, &format, 10); if (format == formatStop) break; } else if (*format == '*') { desiredLength = ficlStackPopInteger(vm->dataStack); format++; if (format == formatStop) break; } switch (*format) { case 's': case 'S': { actualLength = ficlStackPopInteger(vm->dataStack); source = (char *)ficlStackPopPointer(vm->dataStack); break; } case 'x': case 'X': base = 16; case 'u': case 'U': unsignedInteger = 1; /* true */ case 'd': case 'D': { int integer = ficlStackPopInteger(vm->dataStack); if (unsignedInteger) ficlUltoa(integer, scratch, base); else ficlLtoa(integer, scratch, base); base = 10; unsignedInteger = 0; /* false */ source = scratch; actualLength = strlen(scratch); break; } case '%': source = format; actualLength = 1; default: continue; } } if (append) { if (!desiredLength) desiredLength = actualLength; if (desiredLength > bufferLength) { append = 0; /* false */ desiredLength = bufferLength; } while (desiredLength > actualLength) { *buffer++ = (char)((leadingZeroes) ? '0' : ' '); bufferLength--; desiredLength--; } memcpy(buffer, source, actualLength); buffer += actualLength; bufferLength -= actualLength; } format++; } ficlStackPushPointer(vm->dataStack, bufferStart); ficlStackPushInteger(vm->dataStack, buffer - bufferStart); ficlStackPushInteger(vm->dataStack, append && FICL_TRUE); } /************************************************************************** d u p & f r i e n d s ** **************************************************************************/ static void ficlPrimitiveDepth(ficlVm *vm) { int i; FICL_STACK_CHECK(vm->dataStack, 0, 1); i = ficlStackDepth(vm->dataStack); ficlStackPushInteger(vm->dataStack, i); return; } /************************************************************************** e m i t & f r i e n d s ** **************************************************************************/ static void ficlPrimitiveEmit(ficlVm *vm) { char *buffer = vm->pad; int i; FICL_STACK_CHECK(vm->dataStack, 1, 0); i = ficlStackPopInteger(vm->dataStack); buffer[0] = (char)i; buffer[1] = '\0'; ficlVmTextOut(vm, buffer); return; } static void ficlPrimitiveCR(ficlVm *vm) { ficlVmTextOut(vm, "\n"); return; } static void ficlPrimitiveBackslash(ficlVm *vm) { char *trace = ficlVmGetInBuf(vm); char *stop = ficlVmGetInBufEnd(vm); char c = *trace; while ((trace != stop) && (c != '\r') && (c != '\n')) { c = *++trace; } /* ** Cope with DOS or UNIX-style EOLs - ** Check for /r, /n, /r/n, or /n/r end-of-line sequences, ** and point trace to next char. If EOL is \0, we're done. */ if (trace != stop) { trace++; if ( (trace != stop) && (c != *trace) && ((*trace == '\r') || (*trace == '\n')) ) trace++; } ficlVmUpdateTib(vm, trace); return; } /* ** paren CORE ** Compilation: Perform the execution semantics given below. ** Execution: ( "ccc<paren>" -- ) ** Parse ccc delimited by ) (right parenthesis). ( is an immediate word. ** The number of characters in ccc may be zero to the number of characters ** in the parse area. ** */ static void ficlPrimitiveParenthesis(ficlVm *vm) { ficlVmParseStringEx(vm, ')', 0); return; } /************************************************************************** F E T C H & S T O R E ** **************************************************************************/ /************************************************************************** i f C o I m ** IMMEDIATE ** Compiles code for a conditional branch into the dictionary ** and pushes the branch patch address on the stack for later ** patching by ELSE or THEN/ENDIF. **************************************************************************/ static void ficlPrimitiveIfCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionBranch0ParenWithCheck); markBranch(dictionary, vm, origTag); ficlDictionaryAppendUnsigned(dictionary, 1); return; } /************************************************************************** e l s e C o I m ** ** IMMEDIATE -- compiles an "else"... ** 1) FICL_VM_STATE_COMPILE a branch and a patch address; the address gets patched ** by "endif" to point past the "else" code. ** 2) Pop the the "if" patch address ** 3) Patch the "if" branch to point to the current FICL_VM_STATE_COMPILE address. ** 4) Push the "else" patch address. ("endif" patches this to jump past ** the "else" code. **************************************************************************/ static void ficlPrimitiveElseCoIm(ficlVm *vm) { ficlCell *patchAddr; ficlInteger offset; ficlDictionary *dictionary = ficlVmGetDictionary(vm); /* (1) FICL_VM_STATE_COMPILE branch runtime */ ficlDictionaryAppendUnsigned(dictionary, ficlInstructionBranchParenWithCheck); matchControlTag(vm, origTag); patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack); /* (2) pop "if" patch addr */ markBranch(dictionary, vm, origTag); /* (4) push "else" patch addr */ ficlDictionaryAppendUnsigned(dictionary, 1); /* (1) FICL_VM_STATE_COMPILE patch placeholder */ offset = dictionary->here - patchAddr; *patchAddr = FICL_LVALUE_TO_CELL(offset); /* (3) Patch "if" */ return; } /************************************************************************** e n d i f C o I m ** **************************************************************************/ static void ficlPrimitiveEndifCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); resolveForwardBranch(dictionary, vm, origTag); return; } /************************************************************************** c a s e C o I m ** IMMEDIATE FICL_VM_STATE_COMPILE-ONLY ** ** ** At FICL_VM_STATE_COMPILE-time, a CASE-SYS (see DPANS94 6.2.0873) looks like this: ** i*addr i caseTag ** and an OF-SYS (see DPANS94 6.2.1950) looks like this: ** i*addr i caseTag addr ofTag ** The integer under caseTag is the count of fixup addresses that branch ** to ENDCASE. **************************************************************************/ static void ficlPrimitiveCaseCoIm(ficlVm *vm) { FICL_STACK_CHECK(vm->dataStack, 0, 2); ficlStackPushUnsigned(vm->dataStack, 0); markControlTag(vm, caseTag); return; } /************************************************************************** e n d c a s eC o I m ** IMMEDIATE FICL_VM_STATE_COMPILE-ONLY **************************************************************************/ static void ficlPrimitiveEndcaseCoIm(ficlVm *vm) { ficlUnsigned fixupCount; ficlDictionary *dictionary; ficlCell *patchAddr; ficlInteger offset; /* ** if the last OF ended with FALLTHROUGH, ** just add the FALLTHROUGH fixup to the ** ENDOF fixups */ if (ficlStackGetTop(vm->dataStack).p == fallthroughTag) { matchControlTag(vm, fallthroughTag); patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack); matchControlTag(vm, caseTag); fixupCount = ficlStackPopUnsigned(vm->dataStack); ficlStackPushPointer(vm->dataStack, patchAddr); ficlStackPushUnsigned(vm->dataStack, fixupCount + 1); markControlTag(vm, caseTag); } matchControlTag(vm, caseTag); FICL_STACK_CHECK(vm->dataStack, 1, 0); fixupCount = ficlStackPopUnsigned(vm->dataStack); FICL_STACK_CHECK(vm->dataStack, fixupCount, 0); dictionary = ficlVmGetDictionary(vm); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionDrop); while (fixupCount--) { patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack); offset = dictionary->here - patchAddr; *patchAddr = FICL_LVALUE_TO_CELL(offset); } return; } /************************************************************************** o f C o I m ** IMMEDIATE FICL_VM_STATE_COMPILE-ONLY **************************************************************************/ static void ficlPrimitiveOfCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlCell *fallthroughFixup = NULL; FICL_STACK_CHECK(vm->dataStack, 1, 3); if (ficlStackGetTop(vm->dataStack).p == fallthroughTag) { matchControlTag(vm, fallthroughTag); fallthroughFixup = (ficlCell *)ficlStackPopPointer(vm->dataStack); } matchControlTag(vm, caseTag); markControlTag(vm, caseTag); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionOfParen); markBranch(dictionary, vm, ofTag); ficlDictionaryAppendUnsigned(dictionary, 2); if (fallthroughFixup != NULL) { ficlInteger offset = dictionary->here - fallthroughFixup; *fallthroughFixup = FICL_LVALUE_TO_CELL(offset); } return; } /************************************************************************** e n d o f C o I m ** IMMEDIATE FICL_VM_STATE_COMPILE-ONLY **************************************************************************/ static void ficlPrimitiveEndofCoIm(ficlVm *vm) { ficlCell *patchAddr; ficlUnsigned fixupCount; ficlInteger offset; ficlDictionary *dictionary = ficlVmGetDictionary(vm); FICL_STACK_CHECK(vm->dataStack, 4, 3); /* ensure we're in an OF, */ matchControlTag(vm, ofTag); /* grab the address of the branch location after the OF */ patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack); /* ensure we're also in a "case" */ matchControlTag(vm, caseTag); /* grab the current number of ENDOF fixups */ fixupCount = ficlStackPopUnsigned(vm->dataStack); /* FICL_VM_STATE_COMPILE branch runtime */ ficlDictionaryAppendUnsigned(dictionary, ficlInstructionBranchParenWithCheck); /* push a new ENDOF fixup, the updated count of ENDOF fixups, and the caseTag */ ficlStackPushPointer(vm->dataStack, dictionary->here); ficlStackPushUnsigned(vm->dataStack, fixupCount + 1); markControlTag(vm, caseTag); /* reserve space for the ENDOF fixup */ ficlDictionaryAppendUnsigned(dictionary, 2); /* and patch the original OF */ offset = dictionary->here - patchAddr; *patchAddr = FICL_LVALUE_TO_CELL(offset); } /************************************************************************** f a l l t h r o u g h C o I m ** IMMEDIATE FICL_VM_STATE_COMPILE-ONLY **************************************************************************/ static void ficlPrimitiveFallthroughCoIm(ficlVm *vm) { ficlCell *patchAddr; ficlInteger offset; ficlDictionary *dictionary = ficlVmGetDictionary(vm); FICL_STACK_CHECK(vm->dataStack, 4, 3); /* ensure we're in an OF, */ matchControlTag(vm, ofTag); /* grab the address of the branch location after the OF */ patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack); /* ensure we're also in a "case" */ matchControlTag(vm, caseTag); /* okay, here we go. put the case tag back. */ markControlTag(vm, caseTag); /* FICL_VM_STATE_COMPILE branch runtime */ ficlDictionaryAppendUnsigned(dictionary, ficlInstructionBranchParenWithCheck); /* push a new FALLTHROUGH fixup and the fallthroughTag */ ficlStackPushPointer(vm->dataStack, dictionary->here); markControlTag(vm, fallthroughTag); /* reserve space for the FALLTHROUGH fixup */ ficlDictionaryAppendUnsigned(dictionary, 2); /* and patch the original OF */ offset = dictionary->here - patchAddr; *patchAddr = FICL_LVALUE_TO_CELL(offset); } /************************************************************************** h a s h ** hash ( c-addr u -- code) ** calculates hashcode of specified string and leaves it on the stack **************************************************************************/ static void ficlPrimitiveHash(ficlVm *vm) { ficlString s; FICL_STRING_SET_LENGTH(s, ficlStackPopUnsigned(vm->dataStack)); FICL_STRING_SET_POINTER(s, ficlStackPopPointer(vm->dataStack)); ficlStackPushUnsigned(vm->dataStack, ficlHashCode(s)); return; } /************************************************************************** ** Report error **************************************************************************/ static void ficlVmThrowNotFound(ficlVm *vm, ficlString s) { ficlVmThrowError(vm, "%.*s not found", FICL_STRING_GET_LENGTH(s), FICL_STRING_GET_POINTER(s)); } /************************************************************************** i n t e r p r e t ** This is the "user interface" of a Forth. It does the following: ** while there are words in the VM's Text Input Buffer ** Copy next word into the pad (ficlVmGetWord) ** Attempt to find the word in the dictionary (ficlDictionaryLookup) ** If successful, execute the word. ** Otherwise, attempt to convert the word to a number (isNumber) ** If successful, push the number onto the parameter stack. ** Otherwise, print an error message and exit loop... ** End Loop ** ** From the standard, section 3.4 ** Text interpretation (see 6.1.1360 EVALUATE and 6.1.2050 QUIT) shall ** repeat the following steps until either the parse area is empty or an ** ambiguous condition exists: ** a) Skip leading spaces and parse a name (see 3.4.1); **************************************************************************/ static void ficlPrimitiveInterpret(ficlVm *vm) { ficlString s; int i; ficlSystem *system; FICL_VM_ASSERT(vm, vm); system = vm->callback.system; s = ficlVmGetWord0(vm); /* ** Get next word...if out of text, we're done. */ if (s.length == 0) { ficlVmThrow(vm, FICL_VM_STATUS_OUT_OF_TEXT); } /* ** Run the parse chain against the incoming token until somebody eats it. ** Otherwise emit an error message and give up. */ for (i=0; i < FICL_MAX_PARSE_STEPS; i++) { ficlWord *word = system->parseList[i]; if (word == NULL) break; if (word->code == ficlPrimitiveParseStepParen) { ficlParseStep pStep; pStep = (ficlParseStep)(word->param->fn); if ((*pStep)(vm, s)) return; } else { ficlStackPushPointer(vm->dataStack, FICL_STRING_GET_POINTER(s)); ficlStackPushUnsigned(vm->dataStack, FICL_STRING_GET_LENGTH(s)); ficlVmExecuteXT(vm, word); if (ficlStackPopInteger(vm->dataStack)) return; } } ficlVmThrowNotFound(vm, s); return; /* back to inner interpreter */ } /* ** Surrogate precompiled parse step for ficlParseWord (this step is hard coded in ** FICL_VM_STATE_INTERPRET) */ static void ficlPrimitiveLookup(ficlVm *vm) { ficlString name; FICL_STRING_SET_LENGTH(name, ficlStackPopUnsigned(vm->dataStack)); FICL_STRING_SET_POINTER(name, ficlStackPopPointer(vm->dataStack)); ficlStackPushInteger(vm->dataStack, ficlVmParseWord(vm, name)); return; } /************************************************************************** p a r e n P a r s e S t e p ** (parse-step) ( c-addr u -- flag ) ** runtime for a precompiled parse step - pop a counted string off the ** stack, run the parse step against it, and push the result flag (FICL_TRUE ** if success, FICL_FALSE otherwise). **************************************************************************/ void ficlPrimitiveParseStepParen(ficlVm *vm) { ficlString s; ficlWord *word = vm->runningWord; ficlParseStep pStep = (ficlParseStep)(word->param->fn); FICL_STRING_SET_LENGTH(s, ficlStackPopInteger(vm->dataStack)); FICL_STRING_SET_POINTER(s, ficlStackPopPointer(vm->dataStack)); ficlStackPushInteger(vm->dataStack, (*pStep)(vm, s)); return; } static void ficlPrimitiveAddParseStep(ficlVm *vm) { ficlWord *pStep; ficlDictionary *dictionary = ficlVmGetDictionary(vm); FICL_STACK_CHECK(vm->dataStack, 1, 0); pStep = (ficlWord *)(ficlStackPop(vm->dataStack).p); if ((pStep != NULL) && ficlDictionaryIsAWord(dictionary, pStep)) ficlSystemAddParseStep(vm->callback.system, pStep); return; } /************************************************************************** l i t e r a l I m ** ** IMMEDIATE code for "literal". This function gets a value from the stack ** and compiles it into the dictionary preceded by the code for "(literal)". ** IMMEDIATE **************************************************************************/ void ficlPrimitiveLiteralIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlInteger value; value = ficlStackPopInteger(vm->dataStack); switch (value) { case 1: case 2: case 3: case 4: case 5: case 6: case 7: case 8: case 9: case 10: case 11: case 12: case 13: case 14: case 15: case 16: ficlDictionaryAppendUnsigned(dictionary, value); break; case 0: case -1: case -2: case -3: case -4: case -5: case -6: case -7: case -8: case -9: case -10: case -11: case -12: case -13: case -14: case -15: case -16: ficlDictionaryAppendUnsigned(dictionary, ficlInstruction0- value); break; default: ficlDictionaryAppendUnsigned(dictionary, ficlInstructionLiteralParen); ficlDictionaryAppendUnsigned(dictionary, value); break; } return; } static void ficlPrimitive2LiteralIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlDictionaryAppendUnsigned(dictionary, ficlInstruction2LiteralParen); ficlDictionaryAppendCell(dictionary, ficlStackPop(vm->dataStack)); ficlDictionaryAppendCell(dictionary, ficlStackPop(vm->dataStack)); return; } /************************************************************************** D o / L o o p ** do -- IMMEDIATE FICL_VM_STATE_COMPILE ONLY ** Compiles code to initialize a loop: FICL_VM_STATE_COMPILE (do), ** allot space to hold the "leave" address, push a branch ** target address for the loop. ** (do) -- runtime for "do" ** pops index and limit from the p stack and moves them ** to the r stack, then skips to the loop body. ** loop -- IMMEDIATE FICL_VM_STATE_COMPILE ONLY ** +loop ** Compiles code for the test part of a loop: ** FICL_VM_STATE_COMPILE (loop), resolve forward branch from "do", and ** copy "here" address to the "leave" address allotted by "do" ** i,j,k -- FICL_VM_STATE_COMPILE ONLY ** Runtime: Push loop indices on param stack (i is innermost loop...) ** Note: each loop has three values on the return stack: ** ( R: leave limit index ) ** "leave" is the absolute address of the next ficlCell after the loop ** limit and index are the loop control variables. ** leave -- FICL_VM_STATE_COMPILE ONLY ** Runtime: pop the loop control variables, then pop the ** "leave" address and jump (absolute) there. **************************************************************************/ static void ficlPrimitiveDoCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionDoParen); /* ** Allot space for a pointer to the end ** of the loop - "leave" uses this... */ markBranch(dictionary, vm, leaveTag); ficlDictionaryAppendUnsigned(dictionary, 0); /* ** Mark location of head of loop... */ markBranch(dictionary, vm, doTag); return; } static void ficlPrimitiveQDoCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionQDoParen); /* ** Allot space for a pointer to the end ** of the loop - "leave" uses this... */ markBranch(dictionary, vm, leaveTag); ficlDictionaryAppendUnsigned(dictionary, 0); /* ** Mark location of head of loop... */ markBranch(dictionary, vm, doTag); return; } static void ficlPrimitiveLoopCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionLoopParen); resolveBackBranch(dictionary, vm, doTag); resolveAbsBranch(dictionary, vm, leaveTag); return; } static void ficlPrimitivePlusLoopCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionPlusLoopParen); resolveBackBranch(dictionary, vm, doTag); resolveAbsBranch(dictionary, vm, leaveTag); return; } /************************************************************************** v a r i a b l e ** **************************************************************************/ static void ficlPrimitiveVariable(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlString name = ficlVmGetWord(vm); ficlDictionaryAppendWord(dictionary, name, (ficlPrimitive)ficlInstructionVariableParen, FICL_WORD_DEFAULT); ficlVmDictionaryAllotCells(vm, dictionary, 1); return; } static void ficlPrimitive2Variable(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlString name = ficlVmGetWord(vm); ficlDictionaryAppendWord(dictionary, name, (ficlPrimitive)ficlInstructionVariableParen, FICL_WORD_DEFAULT); ficlVmDictionaryAllotCells(vm, dictionary, 2); return; } /************************************************************************** b a s e & f r i e n d s ** **************************************************************************/ static void ficlPrimitiveBase(ficlVm *vm) { ficlCell *pBase; FICL_STACK_CHECK(vm->dataStack, 0, 1); pBase = (ficlCell *)(&vm->base); ficlStackPush(vm->dataStack, FICL_LVALUE_TO_CELL(pBase)); return; } static void ficlPrimitiveDecimal(ficlVm *vm) { vm->base = 10; return; } static void ficlPrimitiveHex(ficlVm *vm) { vm->base = 16; return; } /************************************************************************** a l l o t & f r i e n d s ** **************************************************************************/ static void ficlPrimitiveAllot(ficlVm *vm) { ficlDictionary *dictionary; ficlInteger i; FICL_STACK_CHECK(vm->dataStack, 1, 0); dictionary = ficlVmGetDictionary(vm); i = ficlStackPopInteger(vm->dataStack); FICL_VM_DICTIONARY_CHECK(vm, dictionary, i); ficlVmDictionaryAllot(vm, dictionary, i); return; } static void ficlPrimitiveHere(ficlVm *vm) { ficlDictionary *dictionary; FICL_STACK_CHECK(vm->dataStack, 0, 1); dictionary = ficlVmGetDictionary(vm); ficlStackPushPointer(vm->dataStack, dictionary->here); return; } /************************************************************************** t i c k ** tick CORE ( "<spaces>name" -- xt ) ** Skip leading space delimiters. Parse name delimited by a space. Find ** name and return xt, the execution token for name. An ambiguous condition ** exists if name is not found. **************************************************************************/ void ficlPrimitiveTick(ficlVm *vm) { ficlWord *word = NULL; ficlString name = ficlVmGetWord(vm); FICL_STACK_CHECK(vm->dataStack, 0, 1); word = ficlDictionaryLookup(ficlVmGetDictionary(vm), name); if (!word) ficlVmThrowNotFound(vm, name); ficlStackPushPointer(vm->dataStack, word); return; } static void ficlPrimitiveBracketTickCoIm(ficlVm *vm) { ficlPrimitiveTick(vm); ficlPrimitiveLiteralIm(vm); return; } /************************************************************************** p o s t p o n e ** Lookup the next word in the input stream and FICL_VM_STATE_COMPILE code to ** insert it into definitions created by the resulting word ** (defers compilation, even of immediate words) **************************************************************************/ static void ficlPrimitivePostponeCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlWord *word; ficlWord *pComma = ficlSystemLookup(vm->callback.system, ","); FICL_VM_ASSERT(vm, pComma); ficlPrimitiveTick(vm); word = (ficlWord *)ficlStackGetTop(vm->dataStack).p; if (ficlWordIsImmediate(word)) { ficlDictionaryAppendCell(dictionary, ficlStackPop(vm->dataStack)); } else { ficlPrimitiveLiteralIm(vm); ficlDictionaryAppendCell(dictionary, FICL_LVALUE_TO_CELL(pComma)); } return; } /************************************************************************** e x e c u t e ** Pop an execution token (pointer to a word) off the stack and ** run it **************************************************************************/ static void ficlPrimitiveExecute(ficlVm *vm) { ficlWord *word; FICL_STACK_CHECK(vm->dataStack, 1, 0); word = (ficlWord *)ficlStackPopPointer(vm->dataStack); ficlVmExecuteWord(vm, word); return; } /************************************************************************** i m m e d i a t e ** Make the most recently compiled word IMMEDIATE -- it executes even ** in FICL_VM_STATE_COMPILE state (most often used for control compiling words ** such as IF, THEN, etc) **************************************************************************/ static void ficlPrimitiveImmediate(ficlVm *vm) { FICL_IGNORE(vm); ficlDictionarySetImmediate(ficlVmGetDictionary(vm)); return; } static void ficlPrimitiveCompileOnly(ficlVm *vm) { FICL_IGNORE(vm); ficlDictionarySetFlags(ficlVmGetDictionary(vm), FICL_WORD_COMPILE_ONLY); return; } static void ficlPrimitiveSetObjectFlag(ficlVm *vm) { FICL_IGNORE(vm); ficlDictionarySetFlags(ficlVmGetDictionary(vm), FICL_WORD_OBJECT); return; } static void ficlPrimitiveIsObject(ficlVm *vm) { int flag; ficlWord *word = (ficlWord *)ficlStackPopPointer(vm->dataStack); flag = ((word != NULL) && (word->flags & FICL_WORD_OBJECT)) ? FICL_TRUE : FICL_FALSE; ficlStackPushInteger(vm->dataStack, flag); return; } static void ficlPrimitiveCountedStringQuoteIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); if (vm->state == FICL_VM_STATE_INTERPRET) { ficlCountedString *counted = (ficlCountedString *) dictionary->here; ficlVmGetString(vm, counted, '\"'); ficlStackPushPointer(vm->dataStack, counted); /* move HERE past string so it doesn't get overwritten. --lch */ ficlVmDictionaryAllot(vm, dictionary, counted->length + sizeof(ficlUnsigned8)); } else /* FICL_VM_STATE_COMPILE state */ { ficlDictionaryAppendUnsigned(dictionary, ficlInstructionCStringLiteralParen); dictionary->here = FICL_POINTER_TO_CELL(ficlVmGetString(vm, (ficlCountedString *)dictionary->here, '\"')); ficlDictionaryAlign(dictionary); } return; } /************************************************************************** d o t Q u o t e ** IMMEDIATE word that compiles a string literal for later display ** FICL_VM_STATE_COMPILE fiStringLiteralParen, then copy the bytes of the string from the ** TIB to the dictionary. Backpatch the count byte and align the dictionary. **************************************************************************/ static void ficlPrimitiveDotQuoteCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlWord *pType = ficlSystemLookup(vm->callback.system, "type"); FICL_VM_ASSERT(vm, pType); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionStringLiteralParen); dictionary->here = FICL_POINTER_TO_CELL(ficlVmGetString(vm, (ficlCountedString *)dictionary->here, '\"')); ficlDictionaryAlign(dictionary); ficlDictionaryAppendCell(dictionary, FICL_LVALUE_TO_CELL(pType)); return; } static void ficlPrimitiveDotParen(ficlVm *vm) { char *from = ficlVmGetInBuf(vm); char *stop = ficlVmGetInBufEnd(vm); char *to = vm->pad; char c; /* ** Note: the standard does not want leading spaces skipped. */ for (c = *from; (from != stop) && (c != ')'); c = *++from) *to++ = c; *to = '\0'; if ((from != stop) && (c == ')')) from++; ficlVmTextOut(vm, vm->pad); ficlVmUpdateTib(vm, from); return; } /************************************************************************** s l i t e r a l ** STRING ** Interpretation: Interpretation semantics for this word are undefined. ** Compilation: ( c-addr1 u -- ) ** Append the run-time semantics given below to the current definition. ** Run-time: ( -- c-addr2 u ) ** Return c-addr2 u describing a string consisting of the characters ** specified by c-addr1 u during compilation. A program shall not alter ** the returned string. **************************************************************************/ static void ficlPrimitiveSLiteralCoIm(ficlVm *vm) { ficlDictionary *dictionary; char *from; char *to; ficlUnsigned length; FICL_STACK_CHECK(vm->dataStack, 2, 0); dictionary = ficlVmGetDictionary(vm); length = ficlStackPopUnsigned(vm->dataStack); from = (char *)ficlStackPopPointer(vm->dataStack); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionStringLiteralParen); to = (char *) dictionary->here; *to++ = (char) length; for (; length > 0; --length) { *to++ = *from++; } *to++ = 0; dictionary->here = FICL_POINTER_TO_CELL(ficlAlignPointer(to)); return; } /************************************************************************** s t a t e ** Return the address of the VM's state member (must be sized the ** same as a ficlCell for this reason) **************************************************************************/ static void ficlPrimitiveState(ficlVm *vm) { FICL_STACK_CHECK(vm->dataStack, 0, 1); ficlStackPushPointer(vm->dataStack, &vm->state); return; } /************************************************************************** c r e a t e . . . d o e s > ** Make a new word in the dictionary with the run-time effect of ** a variable (push my address), but with extra space allotted ** for use by does> . **************************************************************************/ static void ficlPrimitiveCreate(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlString name = ficlVmGetWord(vm); ficlDictionaryAppendWord(dictionary, name, (ficlPrimitive)ficlInstructionCreateParen, FICL_WORD_DEFAULT); ficlVmDictionaryAllotCells(vm, dictionary, 1); return; } static void ficlPrimitiveDoesCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); #if FICL_WANT_LOCALS if (vm->callback.system->localsCount > 0) { ficlDictionary *locals = ficlSystemGetLocals(vm->callback.system); ficlDictionaryEmpty(locals, locals->forthWordlist->size); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionUnlinkParen); } vm->callback.system->localsCount = 0; #endif FICL_IGNORE(vm); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionDoesParen); return; } /************************************************************************** t o b o d y ** to-body CORE ( xt -- a-addr ) ** a-addr is the data-field address corresponding to xt. An ambiguous ** condition exists if xt is not for a word defined via CREATE. **************************************************************************/ static void ficlPrimitiveToBody(ficlVm *vm) { ficlWord *word; FICL_STACK_CHECK(vm->dataStack, 1, 1); word = (ficlWord *)ficlStackPopPointer(vm->dataStack); ficlStackPushPointer(vm->dataStack, word->param + 1); return; } /* ** from-body Ficl ( a-addr -- xt ) ** Reverse effect of >body */ static void ficlPrimitiveFromBody(ficlVm *vm) { char *ptr; FICL_STACK_CHECK(vm->dataStack, 1, 1); ptr = (char *)ficlStackPopPointer(vm->dataStack) - sizeof (ficlWord); ficlStackPushPointer(vm->dataStack, ptr); return; } /* ** >name Ficl ( xt -- c-addr u ) ** Push the address and length of a word's name given its address ** xt. */ static void ficlPrimitiveToName(ficlVm *vm) { ficlWord *word; FICL_STACK_CHECK(vm->dataStack, 1, 2); word = (ficlWord *)ficlStackPopPointer(vm->dataStack); ficlStackPushPointer(vm->dataStack, word->name); ficlStackPushUnsigned(vm->dataStack, word->length); return; } static void ficlPrimitiveLastWord(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlWord *wp = dictionary->smudge; FICL_VM_ASSERT(vm, wp); ficlVmPush(vm, FICL_LVALUE_TO_CELL(wp)); return; } /************************************************************************** l b r a c k e t e t c ** **************************************************************************/ static void ficlPrimitiveLeftBracketCoIm(ficlVm *vm) { vm->state = FICL_VM_STATE_INTERPRET; return; } static void ficlPrimitiveRightBracket(ficlVm *vm) { vm->state = FICL_VM_STATE_COMPILE; return; } /************************************************************************** p i c t u r e d n u m e r i c w o r d s ** ** less-number-sign CORE ( -- ) ** Initialize the pictured numeric output conversion process. ** (clear the pad) **************************************************************************/ static void ficlPrimitiveLessNumberSign(ficlVm *vm) { ficlCountedString *counted = FICL_POINTER_TO_COUNTED_STRING(vm->pad); counted->length = 0; return; } /* ** number-sign CORE ( ud1 -- ud2 ) ** Divide ud1 by the number in BASE giving the quotient ud2 and the remainder ** n. (n is the least-significant digit of ud1.) Convert n to external form ** and add the resulting character to the beginning of the pictured numeric ** output string. An ambiguous condition exists if # executes outside of a ** <# #> delimited number conversion. */ static void ficlPrimitiveNumberSign(ficlVm *vm) { ficlCountedString *counted; ficl2Unsigned u; ficl2UnsignedQR uqr; FICL_STACK_CHECK(vm->dataStack, 2, 2); counted = FICL_POINTER_TO_COUNTED_STRING(vm->pad); u = ficlStackPop2Unsigned(vm->dataStack); uqr = ficl2UnsignedDivide(u, (ficlUnsigned16)(vm->base)); counted->text[counted->length++] = ficlDigitToCharacter(uqr.remainder); ficlStackPush2Unsigned(vm->dataStack, uqr.quotient); return; } /* ** number-sign-greater CORE ( xd -- c-addr u ) ** Drop xd. Make the pictured numeric output string available as a character ** string. c-addr and u specify the resulting character string. A program ** may replace characters within the string. */ static void ficlPrimitiveNumberSignGreater(ficlVm *vm) { ficlCountedString *counted; FICL_STACK_CHECK(vm->dataStack, 2, 2); counted = FICL_POINTER_TO_COUNTED_STRING(vm->pad); counted->text[counted->length] = 0; ficlStringReverse(counted->text); ficlStackDrop(vm->dataStack, 2); ficlStackPushPointer(vm->dataStack, counted->text); ficlStackPushUnsigned(vm->dataStack, counted->length); return; } /* ** number-sign-s CORE ( ud1 -- ud2 ) ** Convert one digit of ud1 according to the rule for #. Continue conversion ** until the quotient is zero. ud2 is zero. An ambiguous condition exists if ** #S executes outside of a <# #> delimited number conversion. ** TO DO: presently does not use ud1 hi ficlCell - use it! */ static void ficlPrimitiveNumberSignS(ficlVm *vm) { ficlCountedString *counted; ficl2Unsigned u; ficl2UnsignedQR uqr; FICL_STACK_CHECK(vm->dataStack, 2, 2); counted = FICL_POINTER_TO_COUNTED_STRING(vm->pad); u = ficlStackPop2Unsigned(vm->dataStack); do { uqr = ficl2UnsignedDivide(u, (ficlUnsigned16)(vm->base)); counted->text[counted->length++] = ficlDigitToCharacter(uqr.remainder); u = uqr.quotient; } while (FICL_2UNSIGNED_NOT_ZERO(u)); ficlStackPush2Unsigned(vm->dataStack, u); return; } /* ** HOLD CORE ( char -- ) ** Add char to the beginning of the pictured numeric output string. An ambiguous ** condition exists if HOLD executes outside of a <# #> delimited number conversion. */ static void ficlPrimitiveHold(ficlVm *vm) { ficlCountedString *counted; int i; FICL_STACK_CHECK(vm->dataStack, 1, 0); counted = FICL_POINTER_TO_COUNTED_STRING(vm->pad); i = ficlStackPopInteger(vm->dataStack); counted->text[counted->length++] = (char) i; return; } /* ** SIGN CORE ( n -- ) ** If n is negative, add a minus sign to the beginning of the pictured ** numeric output string. An ambiguous condition exists if SIGN ** executes outside of a <# #> delimited number conversion. */ static void ficlPrimitiveSign(ficlVm *vm) { ficlCountedString *counted; int i; FICL_STACK_CHECK(vm->dataStack, 1, 0); counted = FICL_POINTER_TO_COUNTED_STRING(vm->pad); i = ficlStackPopInteger(vm->dataStack); if (i < 0) counted->text[counted->length++] = '-'; return; } /************************************************************************** t o N u m b e r ** to-number CORE ( ud1 c-addr1 u1 -- ud2 c-addr2 u2 ) ** ud2 is the unsigned result of converting the characters within the ** string specified by c-addr1 u1 into digits, using the number in BASE, ** and adding each into ud1 after multiplying ud1 by the number in BASE. ** Conversion continues left-to-right until a character that is not ** convertible, including any + or -, is encountered or the string is ** entirely converted. c-addr2 is the location of the first unconverted ** character or the first character past the end of the string if the string ** was entirely converted. u2 is the number of unconverted characters in the ** string. An ambiguous condition exists if ud2 overflows during the ** conversion. **************************************************************************/ static void ficlPrimitiveToNumber(ficlVm *vm) { ficlUnsigned length; char *trace; ficl2Unsigned accumulator; ficlUnsigned base = vm->base; ficlUnsigned c; ficlUnsigned digit; FICL_STACK_CHECK(vm->dataStack,4,4); length = ficlStackPopUnsigned(vm->dataStack); trace = (char *)ficlStackPopPointer(vm->dataStack); accumulator = ficlStackPop2Unsigned(vm->dataStack); for (c = *trace; length > 0; c = *++trace, length--) { if (c < '0') break; digit = c - '0'; if (digit > 9) digit = tolower(c) - 'a' + 10; /* ** Note: following test also catches chars between 9 and a ** because 'digit' is unsigned! */ if (digit >= base) break; accumulator = ficl2UnsignedMultiplyAccumulate(accumulator, base, digit); } ficlStackPush2Unsigned(vm->dataStack, accumulator); ficlStackPushPointer(vm->dataStack, trace); ficlStackPushUnsigned(vm->dataStack, length); return; } /************************************************************************** q u i t & a b o r t ** quit CORE ( -- ) ( R: i*x -- ) ** Empty the return stack, store zero in SOURCE-ID if it is present, make ** the user input device the input source, and enter interpretation state. ** Do not display a message. Repeat the following: ** ** Accept a line from the input source into the input buffer, set >IN to ** zero, and FICL_VM_STATE_INTERPRET. ** Display the implementation-defined system prompt if in ** interpretation state, all processing has been completed, and no ** ambiguous condition exists. **************************************************************************/ static void ficlPrimitiveQuit(ficlVm *vm) { ficlVmThrow(vm, FICL_VM_STATUS_QUIT); return; } static void ficlPrimitiveAbort(ficlVm *vm) { ficlVmThrow(vm, FICL_VM_STATUS_ABORT); return; } /************************************************************************** a c c e p t ** accept CORE ( c-addr +n1 -- +n2 ) ** Receive a string of at most +n1 characters. An ambiguous condition ** exists if +n1 is zero or greater than 32,767. Display graphic characters ** as they are received. A program that depends on the presence or absence ** of non-graphic characters in the string has an environmental dependency. ** The editing functions, if any, that the system performs in order to ** construct the string are implementation-defined. ** ** (Although the standard text doesn't say so, I assume that the intent ** of 'accept' is to store the string at the address specified on ** the stack.) ** Implementation: if there's more text in the TIB, use it. Otherwise ** throw out for more text. Copy characters up to the max count into the ** address given, and return the number of actual characters copied. ** ** Note (sobral) this may not be the behavior you'd expect if you're ** trying to get user input at load time! **************************************************************************/ static void ficlPrimitiveAccept(ficlVm *vm) { ficlUnsigned size; char *address; ficlUnsigned length; char *trace; char *end; FICL_STACK_CHECK(vm->dataStack, 2, 1); trace = ficlVmGetInBuf(vm); end = ficlVmGetInBufEnd(vm); length = end - trace; if (length == 0) ficlVmThrow(vm, FICL_VM_STATUS_RESTART); /* ** Now we have something in the text buffer - use it */ size = ficlStackPopInteger(vm->dataStack); address = (char *)ficlStackPopPointer(vm->dataStack); length = (size < length) ? size : length; strncpy(address, trace, length); trace += length; ficlVmUpdateTib(vm, trace); ficlStackPushInteger(vm->dataStack, length); return; } /************************************************************************** a l i g n ** 6.1.0705 ALIGN CORE ( -- ) ** If the data-space pointer is not aligned, reserve enough space to ** align it. **************************************************************************/ static void ficlPrimitiveAlign(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); FICL_IGNORE(vm); ficlDictionaryAlign(dictionary); return; } /************************************************************************** a l i g n e d ** **************************************************************************/ static void ficlPrimitiveAligned(ficlVm *vm) { void *addr; FICL_STACK_CHECK(vm->dataStack,1,1); addr = ficlStackPopPointer(vm->dataStack); ficlStackPushPointer(vm->dataStack, ficlAlignPointer(addr)); return; } /************************************************************************** b e g i n & f r i e n d s ** Indefinite loop control structures ** A.6.1.0760 BEGIN ** Typical use: ** : X ... BEGIN ... test UNTIL ; ** or ** : X ... BEGIN ... test WHILE ... REPEAT ; **************************************************************************/ static void ficlPrimitiveBeginCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); markBranch(dictionary, vm, destTag); return; } static void ficlPrimitiveUntilCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionBranch0ParenWithCheck); resolveBackBranch(dictionary, vm, destTag); return; } static void ficlPrimitiveWhileCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); FICL_STACK_CHECK(vm->dataStack, 2, 5); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionBranch0ParenWithCheck); markBranch(dictionary, vm, origTag); /* equivalent to 2swap */ ficlStackRoll(vm->dataStack, 3); ficlStackRoll(vm->dataStack, 3); ficlDictionaryAppendUnsigned(dictionary, 1); return; } static void ficlPrimitiveRepeatCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionBranchParenWithCheck); /* expect "begin" branch marker */ resolveBackBranch(dictionary, vm, destTag); /* expect "while" branch marker */ resolveForwardBranch(dictionary, vm, origTag); return; } static void ficlPrimitiveAgainCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlDictionaryAppendUnsigned(dictionary, ficlInstructionBranchParenWithCheck); /* expect "begin" branch marker */ resolveBackBranch(dictionary, vm, destTag); return; } /************************************************************************** c h a r & f r i e n d s ** 6.1.0895 CHAR CORE ( "<spaces>name" -- char ) ** Skip leading space delimiters. Parse name delimited by a space. ** Put the value of its first character onto the stack. ** ** bracket-char CORE ** Interpretation: Interpretation semantics for this word are undefined. ** Compilation: ( "<spaces>name" -- ) ** Skip leading space delimiters. Parse name delimited by a space. ** Append the run-time semantics given below to the current definition. ** Run-time: ( -- char ) ** Place char, the value of the first character of name, on the stack. **************************************************************************/ static void ficlPrimitiveChar(ficlVm *vm) { ficlString s; FICL_STACK_CHECK(vm->dataStack, 0, 1); s = ficlVmGetWord(vm); ficlStackPushUnsigned(vm->dataStack, (ficlUnsigned)(s.text[0])); return; } static void ficlPrimitiveCharCoIm(ficlVm *vm) { ficlPrimitiveChar(vm); ficlPrimitiveLiteralIm(vm); return; } /************************************************************************** c h a r P l u s ** char-plus CORE ( c-addr1 -- c-addr2 ) ** Add the size in address units of a character to c-addr1, giving c-addr2. **************************************************************************/ static void ficlPrimitiveCharPlus(ficlVm *vm) { char *p; FICL_STACK_CHECK(vm->dataStack,1,1); p = (char *)ficlStackPopPointer(vm->dataStack); ficlStackPushPointer(vm->dataStack, p + 1); return; } /************************************************************************** c h a r s ** chars CORE ( n1 -- n2 ) ** n2 is the size in address units of n1 characters. ** For most processors, this function can be a no-op. To guarantee ** portability, we'll multiply by sizeof (char). **************************************************************************/ #if defined (_M_IX86) #pragma warning(disable: 4127) #endif static void ficlPrimitiveChars(ficlVm *vm) { if (sizeof (char) > 1) { ficlInteger i; FICL_STACK_CHECK(vm->dataStack,1,1); i = ficlStackPopInteger(vm->dataStack); ficlStackPushInteger(vm->dataStack, i * sizeof (char)); } /* otherwise no-op! */ return; } #if defined (_M_IX86) #pragma warning(default: 4127) #endif /************************************************************************** c o u n t ** COUNT CORE ( c-addr1 -- c-addr2 u ) ** Return the character string specification for the counted string stored ** at c-addr1. c-addr2 is the address of the first character after c-addr1. ** u is the contents of the character at c-addr1, which is the length in ** characters of the string at c-addr2. **************************************************************************/ static void ficlPrimitiveCount(ficlVm *vm) { ficlCountedString *counted; FICL_STACK_CHECK(vm->dataStack,1,2); counted = (ficlCountedString *)ficlStackPopPointer(vm->dataStack); ficlStackPushPointer(vm->dataStack, counted->text); ficlStackPushUnsigned(vm->dataStack, counted->length); return; } /************************************************************************** e n v i r o n m e n t ? ** environment-query CORE ( c-addr u -- FICL_FALSE | i*x FICL_TRUE ) ** c-addr is the address of a character string and u is the string's ** character count. u may have a value in the range from zero to an ** implementation-defined maximum which shall not be less than 31. The ** character string should contain a keyword from 3.2.6 Environmental ** queries or the optional word sets to be checked for correspondence ** with an attribute of the present environment. If the system treats the ** attribute as unknown, the returned flag is FICL_FALSE; otherwise, the flag ** is FICL_TRUE and the i*x returned is of the type specified in the table for ** the attribute queried. **************************************************************************/ static void ficlPrimitiveEnvironmentQ(ficlVm *vm) { ficlDictionary *environment; ficlWord *word; ficlString name; FICL_STACK_CHECK(vm->dataStack, 2, 1); environment = vm->callback.system->environment; name.length = ficlStackPopUnsigned(vm->dataStack); name.text = (char *)ficlStackPopPointer(vm->dataStack); word = ficlDictionaryLookup(environment, name); if (word != NULL) { ficlVmExecuteWord(vm, word); ficlStackPushInteger(vm->dataStack, FICL_TRUE); } else { ficlStackPushInteger(vm->dataStack, FICL_FALSE); } return; } /************************************************************************** e v a l u a t e ** EVALUATE CORE ( i*x c-addr u -- j*x ) ** Save the current input source specification. Store minus-one (-1) in ** SOURCE-ID if it is present. Make the string described by c-addr and u ** both the input source and input buffer, set >IN to zero, and FICL_VM_STATE_INTERPRET. ** When the parse area is empty, restore the prior input source ** specification. Other stack effects are due to the words EVALUATEd. ** **************************************************************************/ static void ficlPrimitiveEvaluate(ficlVm *vm) { ficlCell id; int result; ficlString string; FICL_STACK_CHECK(vm->dataStack,2,0); FICL_STRING_SET_LENGTH(string, ficlStackPopUnsigned(vm->dataStack)); FICL_STRING_SET_POINTER(string, ficlStackPopPointer(vm->dataStack)); id = vm->sourceId; vm->sourceId.i = -1; result = ficlVmExecuteString(vm, string); vm->sourceId = id; if (result != FICL_VM_STATUS_OUT_OF_TEXT) ficlVmThrow(vm, result); return; } /************************************************************************** s t r i n g q u o t e ** Interpreting: get string delimited by a quote from the input stream, ** copy to a scratch area, and put its count and address on the stack. ** Compiling: FICL_VM_STATE_COMPILE code to push the address and count of a string ** literal, FICL_VM_STATE_COMPILE the string from the input stream, and align the dictionary ** pointer. **************************************************************************/ static void ficlPrimitiveStringQuoteIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); if (vm->state == FICL_VM_STATE_INTERPRET) { ficlCountedString *counted = (ficlCountedString *)dictionary->here; ficlVmGetString(vm, counted, '\"'); ficlStackPushPointer(vm->dataStack, counted->text); ficlStackPushUnsigned(vm->dataStack, counted->length); } else /* FICL_VM_STATE_COMPILE state */ { ficlDictionaryAppendUnsigned(dictionary, ficlInstructionStringLiteralParen); dictionary->here = FICL_POINTER_TO_CELL(ficlVmGetString(vm, (ficlCountedString *)dictionary->here, '\"')); ficlDictionaryAlign(dictionary); } return; } /************************************************************************** t y p e ** Pop count and char address from stack and print the designated string. **************************************************************************/ static void ficlPrimitiveType(ficlVm *vm) { ficlUnsigned length; char *s; FICL_STACK_CHECK(vm->dataStack, 2, 0); length = ficlStackPopUnsigned(vm->dataStack); s = (char *)ficlStackPopPointer(vm->dataStack); if ((s == NULL) || (length == 0)) return; /* ** Since we don't have an output primitive for a counted string ** (oops), make sure the string is null terminated. If not, copy ** and terminate it. */ if (s[length] != 0) { char *here = (char *)ficlVmGetDictionary(vm)->here; if (s != here) strncpy(here, s, length); here[length] = '\0'; s = here; } ficlVmTextOut(vm, s); return; } /************************************************************************** w o r d ** word CORE ( char "<chars>ccc<char>" -- c-addr ) ** Skip leading delimiters. Parse characters ccc delimited by char. An ** ambiguous condition exists if the length of the parsed string is greater ** than the implementation-defined length of a counted string. ** ** c-addr is the address of a transient region containing the parsed word ** as a counted string. If the parse area was empty or contained no ** characters other than the delimiter, the resulting string has a zero ** length. A space, not included in the length, follows the string. A ** program may replace characters within the string. ** NOTE! Ficl also NULL-terminates the dest string. **************************************************************************/ static void ficlPrimitiveWord(ficlVm *vm) { ficlCountedString *counted; char delim; ficlString name; FICL_STACK_CHECK(vm->dataStack, 1, 1); counted = (ficlCountedString *)vm->pad; delim = (char)ficlStackPopInteger(vm->dataStack); name = ficlVmParseStringEx(vm, delim, 1); if (FICL_STRING_GET_LENGTH(name) > FICL_PAD_SIZE - 1) FICL_STRING_SET_LENGTH(name, FICL_PAD_SIZE - 1); counted->length = (ficlUnsigned8)FICL_STRING_GET_LENGTH(name); strncpy(counted->text, FICL_STRING_GET_POINTER(name), FICL_STRING_GET_LENGTH(name)); /* store an extra space at the end of the primitive... why? dunno yet. Guy Carver did it. */ counted->text[counted->length] = ' '; counted->text[counted->length + 1] = 0; ficlStackPushPointer(vm->dataStack, counted); return; } /************************************************************************** p a r s e - w o r d ** Ficl PARSE-WORD ( <spaces>name -- c-addr u ) ** Skip leading spaces and parse name delimited by a space. c-addr is the ** address within the input buffer and u is the length of the selected ** string. If the parse area is empty, the resulting string has a zero length. **************************************************************************/ static void ficlPrimitiveParseNoCopy(ficlVm *vm) { ficlString s; FICL_STACK_CHECK(vm->dataStack, 0, 2); s = ficlVmGetWord0(vm); ficlStackPushPointer(vm->dataStack, FICL_STRING_GET_POINTER(s)); ficlStackPushUnsigned(vm->dataStack, FICL_STRING_GET_LENGTH(s)); return; } /************************************************************************** p a r s e ** CORE EXT ( char "ccc<char>" -- c-addr u ) ** Parse ccc delimited by the delimiter char. ** c-addr is the address (within the input buffer) and u is the length of ** the parsed string. If the parse area was empty, the resulting string has ** a zero length. ** NOTE! PARSE differs from WORD: it does not skip leading delimiters. **************************************************************************/ static void ficlPrimitiveParse(ficlVm *vm) { ficlString s; char delim; FICL_STACK_CHECK(vm->dataStack, 1, 2); delim = (char)ficlStackPopInteger(vm->dataStack); s = ficlVmParseStringEx(vm, delim, 0); ficlStackPushPointer(vm->dataStack, FICL_STRING_GET_POINTER(s)); ficlStackPushUnsigned(vm->dataStack, FICL_STRING_GET_LENGTH(s)); return; } /************************************************************************** f i n d ** FIND CORE ( c-addr -- c-addr 0 | xt 1 | xt -1 ) ** Find the definition named in the counted string at c-addr. If the ** definition is not found, return c-addr and zero. If the definition is ** found, return its execution token xt. If the definition is immediate, ** also return one (1), otherwise also return minus-one (-1). For a given ** string, the values returned by FIND while compiling may differ from ** those returned while not compiling. **************************************************************************/ static void do_find(ficlVm *vm, ficlString name, void *returnForFailure) { ficlWord *word; word = ficlDictionaryLookup(ficlVmGetDictionary(vm), name); if (word) { ficlStackPushPointer(vm->dataStack, word); ficlStackPushInteger(vm->dataStack, (ficlWordIsImmediate(word) ? 1 : -1)); } else { ficlStackPushPointer(vm->dataStack, returnForFailure); ficlStackPushUnsigned(vm->dataStack, 0); } return; } /************************************************************************** f i n d ** FIND CORE ( c-addr -- c-addr 0 | xt 1 | xt -1 ) ** Find the definition named in the counted string at c-addr. If the ** definition is not found, return c-addr and zero. If the definition is ** found, return its execution token xt. If the definition is immediate, ** also return one (1), otherwise also return minus-one (-1). For a given ** string, the values returned by FIND while compiling may differ from ** those returned while not compiling. **************************************************************************/ static void ficlPrimitiveCFind(ficlVm *vm) { ficlCountedString *counted; ficlString name; FICL_STACK_CHECK(vm->dataStack, 1, 2); counted = (ficlCountedString *)ficlStackPopPointer(vm->dataStack); FICL_STRING_SET_FROM_COUNTED_STRING(name, *counted); do_find(vm, name, counted); } /************************************************************************** s f i n d ** Ficl ( c-addr u -- 0 0 | xt 1 | xt -1 ) ** Like FIND, but takes "c-addr u" for the string. **************************************************************************/ static void ficlPrimitiveSFind(ficlVm *vm) { ficlString name; FICL_STACK_CHECK(vm->dataStack, 2, 2); name.length = ficlStackPopInteger(vm->dataStack); name.text = (char *)ficlStackPopPointer(vm->dataStack); do_find(vm, name, NULL); } /************************************************************************** r e c u r s e ** **************************************************************************/ static void ficlPrimitiveRecurseCoIm(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); FICL_IGNORE(vm); ficlDictionaryAppendCell(dictionary, FICL_LVALUE_TO_CELL(dictionary->smudge)); return; } /************************************************************************** s o u r c e ** CORE ( -- c-addr u ) ** c-addr is the address of, and u is the number of characters in, the ** input buffer. **************************************************************************/ static void ficlPrimitiveSource(ficlVm *vm) { FICL_STACK_CHECK(vm->dataStack,0,2); ficlStackPushPointer(vm->dataStack, vm->tib.text); ficlStackPushInteger(vm->dataStack, ficlVmGetInBufLen(vm)); return; } /************************************************************************** v e r s i o n ** non-standard... **************************************************************************/ static void ficlPrimitiveVersion(ficlVm *vm) { ficlVmTextOut(vm, "Ficl version " FICL_VERSION "\n"); return; } /************************************************************************** t o I n ** to-in CORE **************************************************************************/ static void ficlPrimitiveToIn(ficlVm *vm) { FICL_STACK_CHECK(vm->dataStack,0,1); ficlStackPushPointer(vm->dataStack, &vm->tib.index); return; } /************************************************************************** c o l o n N o N a m e ** CORE EXT ( C: -- colon-sys ) ( S: -- xt ) ** Create an unnamed colon definition and push its address. ** Change state to FICL_VM_STATE_COMPILE. **************************************************************************/ static void ficlPrimitiveColonNoName(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlWord *word; ficlString name; FICL_STRING_SET_LENGTH(name, 0); FICL_STRING_SET_POINTER(name, NULL); vm->state = FICL_VM_STATE_COMPILE; word = ficlDictionaryAppendWord(dictionary, name, (ficlPrimitive)ficlInstructionColonParen, FICL_WORD_DEFAULT | FICL_WORD_SMUDGED); ficlStackPushPointer(vm->dataStack, word); markControlTag(vm, colonTag); return; } /************************************************************************** u s e r V a r i a b l e ** user ( u -- ) "<spaces>name" ** Get a name from the input stream and create a user variable ** with the name and the index supplied. The run-time effect ** of a user variable is to push the address of the indexed ficlCell ** in the running vm's user array. ** ** User variables are vm local cells. Each vm has an array of ** FICL_USER_CELLS of them when FICL_WANT_USER is nonzero. ** Ficl's user facility is implemented with two primitives, ** "user" and "(user)", a variable ("nUser") (in softcore.c) that ** holds the index of the next free user ficlCell, and a redefinition ** (also in softcore) of "user" that defines a user word and increments ** nUser. **************************************************************************/ #if FICL_WANT_USER static void ficlPrimitiveUser(ficlVm *vm) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlString name = ficlVmGetWord(vm); ficlCell c; c = ficlStackPop(vm->dataStack); if (c.i >= FICL_USER_CELLS) { ficlVmThrowError(vm, "Error - out of user space"); } ficlDictionaryAppendWord(dictionary, name, (ficlPrimitive)ficlInstructionUserParen, FICL_WORD_DEFAULT); ficlDictionaryAppendCell(dictionary, c); return; } #endif #if FICL_WANT_LOCALS /* ** Each local is recorded in a private locals dictionary as a ** word that does doLocalIm at runtime. DoLocalIm compiles code ** into the client definition to fetch the value of the ** corresponding local variable from the return stack. ** The private dictionary gets initialized at the end of each block ** that uses locals (in ; and does> for example). */ void ficlLocalParenIm(ficlVm *vm, int isDouble, int isFloat) { ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlInteger nLocal = vm->runningWord->param[0].i; #if !FICL_WANT_FLOAT FICL_VM_ASSERT(vm, !isFloat); /* get rid of unused parameter warning */ isFloat = 0; #endif /* FICL_WANT_FLOAT */ if (vm->state == FICL_VM_STATE_INTERPRET) { ficlStack *stack; #if FICL_WANT_FLOAT if (isFloat) stack = vm->floatStack; else #endif /* FICL_WANT_FLOAT */ stack = vm->dataStack; ficlStackPush(stack, vm->returnStack->frame[nLocal]); if (isDouble) ficlStackPush(stack, vm->returnStack->frame[nLocal+1]); } else { ficlInstruction instruction; ficlInteger appendLocalOffset; #if FICL_WANT_FLOAT if (isFloat) { instruction = (isDouble) ? ficlInstructionGetF2LocalParen : ficlInstructionGetFLocalParen; appendLocalOffset = FICL_TRUE; } else #endif /* FICL_WANT_FLOAT */ if (nLocal == 0) { instruction = (isDouble) ? ficlInstructionGet2Local0 : ficlInstructionGetLocal0; appendLocalOffset = FICL_FALSE; } else if ((nLocal == 1) && !isDouble) { instruction = ficlInstructionGetLocal1; appendLocalOffset = FICL_FALSE; } else { instruction = (isDouble) ? ficlInstructionGet2LocalParen : ficlInstructionGetLocalParen; appendLocalOffset = FICL_TRUE; } ficlDictionaryAppendUnsigned(dictionary, instruction); if (appendLocalOffset) ficlDictionaryAppendCell(dictionary, FICL_LVALUE_TO_CELL(nLocal)); } return; } static void ficlPrimitiveDoLocalIm(ficlVm *vm) { ficlLocalParenIm(vm, 0, 0); } static void ficlPrimitiveDo2LocalIm(ficlVm *vm) { ficlLocalParenIm(vm, 1, 0); } #if FICL_WANT_FLOAT static void ficlPrimitiveDoFLocalIm(ficlVm *vm) { ficlLocalParenIm(vm, 0, 1); } static void ficlPrimitiveDoF2LocalIm(ficlVm *vm) { ficlLocalParenIm(vm, 1, 1); } #endif /* FICL_WANT_FLOAT */ /************************************************************************** l o c a l P a r e n ** paren-local-paren LOCAL ** Interpretation: Interpretation semantics for this word are undefined. ** Execution: ( c-addr u -- ) ** When executed during compilation, (LOCAL) passes a message to the ** system that has one of two meanings. If u is non-zero, ** the message identifies a new local whose definition name is given by ** the string of characters identified by c-addr u. If u is zero, ** the message is last local and c-addr has no significance. ** ** The result of executing (LOCAL) during compilation of a definition is ** to create a set of named local identifiers, each of which is ** a definition name, that only have execution semantics within the scope ** of that definition's source. ** ** local Execution: ( -- x ) ** ** Push the local's value, x, onto the stack. The local's value is ** initialized as described in 13.3.3 Processing locals and may be ** changed by preceding the local's name with TO. An ambiguous condition ** exists when local is executed while in interpretation state. **************************************************************************/ void ficlLocalParen(ficlVm *vm, int isDouble, int isFloat) { ficlDictionary *dictionary; ficlString name; FICL_STACK_CHECK(vm->dataStack,2,0); dictionary = ficlVmGetDictionary(vm); FICL_STRING_SET_LENGTH(name, ficlStackPopUnsigned(vm->dataStack)); FICL_STRING_SET_POINTER(name, (char *)ficlStackPopPointer(vm->dataStack)); if (FICL_STRING_GET_LENGTH(name) > 0) { /* add a local to the **locals** dictionary and update localsCount */ ficlPrimitive code; ficlInstruction instruction; ficlDictionary *locals = ficlSystemGetLocals(vm->callback.system); if (vm->callback.system->localsCount >= FICL_MAX_LOCALS) { ficlVmThrowError(vm, "Error: out of local space"); } #if !FICL_WANT_FLOAT FICL_VM_ASSERT(vm, !isFloat); /* get rid of unused parameter warning */ isFloat = 0; #else /* FICL_WANT_FLOAT */ if (isFloat) { if (isDouble) { code = ficlPrimitiveDoF2LocalIm; instruction = ficlInstructionToF2LocalParen; } else { code = ficlPrimitiveDoFLocalIm; instruction = ficlInstructionToFLocalParen; } } else #endif /* FICL_WANT_FLOAT */ if (isDouble) { code = ficlPrimitiveDo2LocalIm; instruction = ficlInstructionTo2LocalParen; } else { code = ficlPrimitiveDoLocalIm; instruction = ficlInstructionToLocalParen; } ficlDictionaryAppendWord(locals, name, code, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionaryAppendCell(locals, FICL_LVALUE_TO_CELL(vm->callback.system->localsCount)); if (vm->callback.system->localsCount == 0) { /* FICL_VM_STATE_COMPILE code to create a local stack frame */ ficlDictionaryAppendUnsigned(dictionary, ficlInstructionLinkParen); /* save location in dictionary for #locals */ vm->callback.system->localsFixup = dictionary->here; ficlDictionaryAppendCell(dictionary, FICL_LVALUE_TO_CELL(vm->callback.system->localsCount)); } ficlDictionaryAppendUnsigned(dictionary, instruction); ficlDictionaryAppendCell(dictionary, FICL_LVALUE_TO_CELL(vm->callback.system->localsCount)); vm->callback.system->localsCount += (isDouble) ? 2 : 1; } else if (vm->callback.system->localsCount > 0) { /* write localsCount to (link) param area in dictionary */ *(ficlInteger *)(vm->callback.system->localsFixup) = vm->callback.system->localsCount; } return; } static void ficlPrimitiveLocalParen(ficlVm *vm) { ficlLocalParen(vm, 0, 0); } static void ficlPrimitive2LocalParen(ficlVm *vm) { ficlLocalParen(vm, 1, 0); } #endif /* FICL_WANT_LOCALS */ /************************************************************************** t o V a l u e ** CORE EXT ** Interpretation: ( x "<spaces>name" -- ) ** Skip leading spaces and parse name delimited by a space. Store x in ** name. An ambiguous condition exists if name was not defined by VALUE. ** NOTE: In Ficl, VALUE is an alias of CONSTANT **************************************************************************/ static void ficlPrimitiveToValue(ficlVm *vm) { ficlString name = ficlVmGetWord(vm); ficlDictionary *dictionary = ficlVmGetDictionary(vm); ficlWord *word; ficlInstruction instruction = 0; ficlStack *stack; ficlInteger isDouble; #if FICL_WANT_LOCALS ficlInteger nLocal; ficlInteger appendLocalOffset; ficlInteger isFloat; #endif /* FICL_WANT_LOCALS */ #if FICL_WANT_LOCALS if ((vm->callback.system->localsCount > 0) && (vm->state == FICL_VM_STATE_COMPILE)) { ficlDictionary *locals; locals = ficlSystemGetLocals(vm->callback.system); word = ficlDictionaryLookup(locals, name); if (!word) goto TO_GLOBAL; if (word->code == ficlPrimitiveDoLocalIm) { instruction = ficlInstructionToLocalParen; isDouble = isFloat = FICL_FALSE; } else if (word->code == ficlPrimitiveDo2LocalIm) { instruction = ficlInstructionTo2LocalParen; isDouble = FICL_TRUE; isFloat = FICL_FALSE; } #if FICL_WANT_FLOAT else if (word->code == ficlPrimitiveDoFLocalIm) { instruction = ficlInstructionToFLocalParen; isDouble = FICL_FALSE; isFloat = FICL_TRUE; } else if (word->code == ficlPrimitiveDoF2LocalIm) { instruction = ficlInstructionToF2LocalParen; isDouble = isFloat = FICL_TRUE; } #endif /* FICL_WANT_FLOAT */ else { ficlVmThrowError(vm, "to %.*s : local is of unknown type", FICL_STRING_GET_LENGTH(name), FICL_STRING_GET_POINTER(name)); return; } nLocal = word->param[0].i; appendLocalOffset = FICL_TRUE; #if FICL_WANT_FLOAT if (!isFloat) { #endif /* FICL_WANT_FLOAT */ if (nLocal == 0) { instruction = (isDouble) ? ficlInstructionTo2Local0 : ficlInstructionToLocal0; appendLocalOffset = FICL_FALSE; } else if ((nLocal == 1) && !isDouble) { instruction = ficlInstructionToLocal1; appendLocalOffset = FICL_FALSE; } #if FICL_WANT_FLOAT } #endif /* FICL_WANT_FLOAT */ ficlDictionaryAppendUnsigned(dictionary, instruction); if (appendLocalOffset) ficlDictionaryAppendCell(dictionary, FICL_LVALUE_TO_CELL(nLocal)); return; } #endif #if FICL_WANT_LOCALS TO_GLOBAL: #endif /* FICL_WANT_LOCALS */ word = ficlDictionaryLookup(dictionary, name); if (!word) ficlVmThrowNotFound(vm, name); switch ((ficlInstruction)word->code) { case ficlInstructionConstantParen: instruction = ficlInstructionStore; stack = vm->dataStack; isDouble = FICL_FALSE; break; case ficlInstruction2ConstantParen: instruction = ficlInstruction2Store; stack = vm->dataStack; isDouble = FICL_TRUE; break; #if FICL_WANT_FLOAT case ficlInstructionFConstantParen: instruction = ficlInstructionFStore; stack = vm->floatStack; isDouble = FICL_FALSE; break; case ficlInstructionF2ConstantParen: instruction = ficlInstructionF2Store; stack = vm->floatStack; isDouble = FICL_TRUE; break; #endif /* FICL_WANT_FLOAT */ default: { ficlVmThrowError(vm, "to %.*s : value/constant is of unknown type", FICL_STRING_GET_LENGTH(name), FICL_STRING_GET_POINTER(name)); return; } } if (vm->state == FICL_VM_STATE_INTERPRET) { word->param[0] = ficlStackPop(stack); if (isDouble) word->param[1] = ficlStackPop(stack); } else /* FICL_VM_STATE_COMPILE code to store to word's param */ { ficlStackPushPointer(vm->dataStack, &word->param[0]); ficlPrimitiveLiteralIm(vm); ficlDictionaryAppendUnsigned(dictionary, instruction); } return; } /************************************************************************** f m S l a s h M o d ** f-m-slash-mod CORE ( d1 n1 -- n2 n3 ) ** Divide d1 by n1, giving the floored quotient n3 and the remainder n2. ** Input and output stack arguments are signed. An ambiguous condition ** exists if n1 is zero or if the quotient lies outside the range of a ** single-ficlCell signed integer. **************************************************************************/ static void ficlPrimitiveFMSlashMod(ficlVm *vm) { ficl2Integer d1; ficlInteger n1; ficl2IntegerQR qr; FICL_STACK_CHECK(vm->dataStack, 3, 2); n1 = ficlStackPopInteger(vm->dataStack); d1 = ficlStackPop2Integer(vm->dataStack); qr = ficl2IntegerDivideFloored(d1, n1); ficlStackPushInteger(vm->dataStack, qr.remainder); ficlStackPushInteger(vm->dataStack, FICL_2UNSIGNED_GET_LOW(qr.quotient)); return; } /************************************************************************** s m S l a s h R e m ** s-m-slash-remainder CORE ( d1 n1 -- n2 n3 ) ** Divide d1 by n1, giving the symmetric quotient n3 and the remainder n2. ** Input and output stack arguments are signed. An ambiguous condition ** exists if n1 is zero or if the quotient lies outside the range of a ** single-ficlCell signed integer. **************************************************************************/ static void ficlPrimitiveSMSlashRem(ficlVm *vm) { ficl2Integer d1; ficlInteger n1; ficl2IntegerQR qr; FICL_STACK_CHECK(vm->dataStack, 3, 2); n1 = ficlStackPopInteger(vm->dataStack); d1 = ficlStackPop2Integer(vm->dataStack); qr = ficl2IntegerDivideSymmetric(d1, n1); ficlStackPushInteger(vm->dataStack, qr.remainder); ficlStackPushInteger(vm->dataStack, FICL_2UNSIGNED_GET_LOW(qr.quotient)); return; } static void ficlPrimitiveMod(ficlVm *vm) { ficl2Integer d1; ficlInteger n1; ficlInteger i; ficl2IntegerQR qr; FICL_STACK_CHECK(vm->dataStack, 2, 1); n1 = ficlStackPopInteger(vm->dataStack); i = ficlStackPopInteger(vm->dataStack); FICL_INTEGER_TO_2INTEGER(i, d1); qr = ficl2IntegerDivideSymmetric(d1, n1); ficlStackPushInteger(vm->dataStack, qr.remainder); return; } /************************************************************************** u m S l a s h M o d ** u-m-slash-mod CORE ( ud u1 -- u2 u3 ) ** Divide ud by u1, giving the quotient u3 and the remainder u2. ** All values and arithmetic are unsigned. An ambiguous condition ** exists if u1 is zero or if the quotient lies outside the range of a ** single-ficlCell unsigned integer. *************************************************************************/ static void ficlPrimitiveUMSlashMod(ficlVm *vm) { ficl2Unsigned ud; ficlUnsigned u1; ficl2UnsignedQR uqr; u1 = ficlStackPopUnsigned(vm->dataStack); ud = ficlStackPop2Unsigned(vm->dataStack); uqr = ficl2UnsignedDivide(ud, u1); ficlStackPushUnsigned(vm->dataStack, uqr.remainder); ficlStackPushUnsigned(vm->dataStack, FICL_2UNSIGNED_GET_LOW(uqr.quotient)); return; } /************************************************************************** m S t a r ** m-star CORE ( n1 n2 -- d ) ** d is the signed product of n1 times n2. **************************************************************************/ static void ficlPrimitiveMStar(ficlVm *vm) { ficlInteger n2; ficlInteger n1; ficl2Integer d; FICL_STACK_CHECK(vm->dataStack, 2, 2); n2 = ficlStackPopInteger(vm->dataStack); n1 = ficlStackPopInteger(vm->dataStack); d = ficl2IntegerMultiply(n1, n2); ficlStackPush2Integer(vm->dataStack, d); return; } static void ficlPrimitiveUMStar(ficlVm *vm) { ficlUnsigned u2; ficlUnsigned u1; ficl2Unsigned ud; FICL_STACK_CHECK(vm->dataStack, 2, 2); u2 = ficlStackPopUnsigned(vm->dataStack); u1 = ficlStackPopUnsigned(vm->dataStack); ud = ficl2UnsignedMultiply(u1, u2); ficlStackPush2Unsigned(vm->dataStack, ud); return; } /************************************************************************** d n e g a t e ** DOUBLE ( d1 -- d2 ) ** d2 is the negation of d1. **************************************************************************/ static void ficlPrimitiveDNegate(ficlVm *vm) { ficl2Integer i = ficlStackPop2Integer(vm->dataStack); i = ficl2IntegerNegate(i); ficlStackPush2Integer(vm->dataStack, i); return; } /************************************************************************** p a d ** CORE EXT ( -- c-addr ) ** c-addr is the address of a transient region that can be used to hold ** data for intermediate processing. **************************************************************************/ static void ficlPrimitivePad(ficlVm *vm) { ficlStackPushPointer(vm->dataStack, vm->pad); } /************************************************************************** s o u r c e - i d ** CORE EXT, FILE ( -- 0 | -1 | fileid ) ** Identifies the input source as follows: ** ** SOURCE-ID Input source ** --------- ------------ ** fileid Text file fileid ** -1 String (via EVALUATE) ** 0 User input device **************************************************************************/ static void ficlPrimitiveSourceID(ficlVm *vm) { ficlStackPushInteger(vm->dataStack, vm->sourceId.i); return; } /************************************************************************** r e f i l l ** CORE EXT ( -- flag ) ** Attempt to fill the input buffer from the input source, returning a FICL_TRUE ** flag if successful. ** When the input source is the user input device, attempt to receive input ** into the terminal input buffer. If successful, make the result the input ** buffer, set >IN to zero, and return FICL_TRUE. Receipt of a line containing no ** characters is considered successful. If there is no input available from ** the current input source, return FICL_FALSE. ** When the input source is a string from EVALUATE, return FICL_FALSE and ** perform no other action. **************************************************************************/ static void ficlPrimitiveRefill(ficlVm *vm) { ficlInteger ret = (vm->sourceId.i == -1) ? FICL_FALSE : FICL_TRUE; if (ret && (vm->restart == 0)) ficlVmThrow(vm, FICL_VM_STATUS_RESTART); ficlStackPushInteger(vm->dataStack, ret); return; } /************************************************************************** freebsd exception handling words ** Catch, from ANS Forth standard. Installs a safety net, then EXECUTE ** the word in ToS. If an exception happens, restore the state to what ** it was before, and pushes the exception value on the stack. If not, ** push zero. ** ** Notice that Catch implements an inner interpreter. This is ugly, ** but given how Ficl works, it cannot be helped. The problem is that ** colon definitions will be executed *after* the function returns, ** while "code" definitions will be executed immediately. I considered ** other solutions to this problem, but all of them shared the same ** basic problem (with added disadvantages): if Ficl ever changes it's ** inner thread modus operandi, one would have to fix this word. ** ** More comments can be found throughout catch's code. ** ** Daniel C. Sobral Jan 09/1999 ** sadler may 2000 -- revised to follow ficl.c:ficlExecXT. **************************************************************************/ static void ficlPrimitiveCatch(ficlVm *vm) { int except; jmp_buf vmState; ficlVm vmCopy; ficlStack dataStackCopy; ficlStack returnStackCopy; ficlWord *word; FICL_VM_ASSERT(vm, vm); FICL_VM_ASSERT(vm, vm->callback.system->exitInnerWord); /* ** Get xt. ** We need this *before* we save the stack pointer, or ** we'll have to pop one element out of the stack after ** an exception. I prefer to get done with it up front. :-) */ FICL_STACK_CHECK(vm->dataStack, 1, 0); word = (ficlWord *)ficlStackPopPointer(vm->dataStack); /* ** Save vm's state -- a catch will not back out environmental ** changes. ** ** We are *not* saving dictionary state, since it is ** global instead of per vm, and we are not saving ** stack contents, since we are not required to (and, ** thus, it would be useless). We save vm, and vm ** "stacks" (a structure containing general information ** about it, including the current stack pointer). */ memcpy((void*)&vmCopy, (void*)vm, sizeof(ficlVm)); memcpy((void*)&dataStackCopy, (void*)vm->dataStack, sizeof(ficlStack)); memcpy((void*)&returnStackCopy, (void*)vm->returnStack, sizeof(ficlStack)); /* ** Give vm a jmp_buf */ vm->exceptionHandler = &vmState; /* ** Safety net */ except = setjmp(vmState); switch (except) { /* ** Setup condition - push poison pill so that the VM throws ** VM_INNEREXIT if the XT terminates normally, then execute ** the XT */ case 0: ficlVmPushIP(vm, &(vm->callback.system->exitInnerWord)); /* Open mouth, insert emetic */ ficlVmExecuteWord(vm, word); ficlVmInnerLoop(vm, 0); break; /* ** Normal exit from XT - lose the poison pill, ** restore old setjmp vector and push a zero. */ case FICL_VM_STATUS_INNER_EXIT: ficlVmPopIP(vm); /* Gack - hurl poison pill */ vm->exceptionHandler = vmCopy.exceptionHandler; /* Restore just the setjmp vector */ ficlStackPushInteger(vm->dataStack, 0); /* Push 0 -- everything is ok */ break; /* ** Some other exception got thrown - restore pre-existing VM state ** and push the exception code */ default: /* Restore vm's state */ memcpy((void*)vm, (void*)&vmCopy, sizeof(ficlVm)); memcpy((void*)vm->dataStack, (void*)&dataStackCopy, sizeof(ficlStack)); memcpy((void*)vm->returnStack, (void*)&returnStackCopy, sizeof(ficlStack)); ficlStackPushInteger(vm->dataStack, except);/* Push error */ break; } } /************************************************************************** ** t h r o w ** EXCEPTION ** Throw -- From ANS Forth standard. ** ** Throw takes the ToS and, if that's different from zero, ** returns to the last executed catch context. Further throws will ** unstack previously executed "catches", in LIFO mode. ** ** Daniel C. Sobral Jan 09/1999 **************************************************************************/ static void ficlPrimitiveThrow(ficlVm *vm) { int except; except = ficlStackPopInteger(vm->dataStack); if (except) ficlVmThrow(vm, except); } /************************************************************************** ** a l l o c a t e ** MEMORY **************************************************************************/ static void ficlPrimitiveAllocate(ficlVm *vm) { size_t size; void *p; size = ficlStackPopInteger(vm->dataStack); p = ficlMalloc(size); ficlStackPushPointer(vm->dataStack, p); if (p) ficlStackPushInteger(vm->dataStack, 0); else ficlStackPushInteger(vm->dataStack, 1); } /************************************************************************** ** f r e e ** MEMORY **************************************************************************/ static void ficlPrimitiveFree(ficlVm *vm) { void *p; p = ficlStackPopPointer(vm->dataStack); ficlFree(p); ficlStackPushInteger(vm->dataStack, 0); } /************************************************************************** ** r e s i z e ** MEMORY **************************************************************************/ static void ficlPrimitiveResize(ficlVm *vm) { size_t size; void *new, *old; size = ficlStackPopInteger(vm->dataStack); old = ficlStackPopPointer(vm->dataStack); new = ficlRealloc(old, size); if (new) { ficlStackPushPointer(vm->dataStack, new); ficlStackPushInteger(vm->dataStack, 0); } else { ficlStackPushPointer(vm->dataStack, old); ficlStackPushInteger(vm->dataStack, 1); } } /************************************************************************** ** e x i t - i n n e r ** Signals execXT that an inner loop has completed **************************************************************************/ static void ficlPrimitiveExitInner(ficlVm *vm) { ficlVmThrow(vm, FICL_VM_STATUS_INNER_EXIT); } #if 0 /************************************************************************** ** **************************************************************************/ static void ficlPrimitiveName(ficlVm *vm) { FICL_IGNORE(vm); return; } #endif /************************************************************************** f i c l C o m p i l e C o r e ** Builds the primitive wordset and the environment-query namespace. **************************************************************************/ void ficlSystemCompileCore(ficlSystem *system) { ficlWord *interpret; ficlDictionary *dictionary = ficlSystemGetDictionary(system); ficlDictionary *environment = ficlSystemGetEnvironment(system); FICL_SYSTEM_ASSERT(system, dictionary); FICL_SYSTEM_ASSERT(system, environment); #define FICL_TOKEN(token, description) #define FICL_INSTRUCTION_TOKEN(token, description, flags) ficlDictionarySetInstruction(dictionary, description, token, flags); #include "ficltokens.h" #undef FICL_TOKEN #undef FICL_INSTRUCTION_TOKEN /* ** The Core word set ** see softcore.c for definitions of: abs bl space spaces abort" */ ficlDictionarySetPrimitive(dictionary, "#", ficlPrimitiveNumberSign, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "#>", ficlPrimitiveNumberSignGreater,FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "#s", ficlPrimitiveNumberSignS, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "\'", ficlPrimitiveTick, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "(", ficlPrimitiveParenthesis, FICL_WORD_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "+loop", ficlPrimitivePlusLoopCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, ".", ficlPrimitiveDot, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, ".\"", ficlPrimitiveDotQuoteCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, ":", ficlPrimitiveColon, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, ";", ficlPrimitiveSemicolonCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "<#", ficlPrimitiveLessNumberSign, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, ">body", ficlPrimitiveToBody, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, ">in", ficlPrimitiveToIn, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, ">number", ficlPrimitiveToNumber, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "abort", ficlPrimitiveAbort, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "accept", ficlPrimitiveAccept, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "align", ficlPrimitiveAlign, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "aligned", ficlPrimitiveAligned, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "allot", ficlPrimitiveAllot, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "base", ficlPrimitiveBase, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "begin", ficlPrimitiveBeginCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "case", ficlPrimitiveCaseCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "char", ficlPrimitiveChar, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "char+", ficlPrimitiveCharPlus, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "chars", ficlPrimitiveChars, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "constant", ficlPrimitiveConstant, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "count", ficlPrimitiveCount, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "cr", ficlPrimitiveCR, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "create", ficlPrimitiveCreate, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "decimal", ficlPrimitiveDecimal, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "depth", ficlPrimitiveDepth, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "do", ficlPrimitiveDoCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "does>", ficlPrimitiveDoesCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "else", ficlPrimitiveElseCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "emit", ficlPrimitiveEmit, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "endcase", ficlPrimitiveEndcaseCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "endof", ficlPrimitiveEndofCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "environment?", ficlPrimitiveEnvironmentQ,FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "evaluate", ficlPrimitiveEvaluate, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "execute", ficlPrimitiveExecute, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "exit", ficlPrimitiveExitCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "fallthrough",ficlPrimitiveFallthroughCoIm,FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "find", ficlPrimitiveCFind, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "fm/mod", ficlPrimitiveFMSlashMod, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "here", ficlPrimitiveHere, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "hold", ficlPrimitiveHold, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "if", ficlPrimitiveIfCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "immediate", ficlPrimitiveImmediate, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "literal", ficlPrimitiveLiteralIm, FICL_WORD_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "loop", ficlPrimitiveLoopCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "m*", ficlPrimitiveMStar, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "mod", ficlPrimitiveMod, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "of", ficlPrimitiveOfCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "postpone", ficlPrimitivePostponeCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "quit", ficlPrimitiveQuit, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "recurse", ficlPrimitiveRecurseCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "repeat", ficlPrimitiveRepeatCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "s\"", ficlPrimitiveStringQuoteIm, FICL_WORD_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "sign", ficlPrimitiveSign, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "sm/rem", ficlPrimitiveSMSlashRem, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "source", ficlPrimitiveSource, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "state", ficlPrimitiveState, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "then", ficlPrimitiveEndifCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "type", ficlPrimitiveType, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "u.", ficlPrimitiveUDot, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "um*", ficlPrimitiveUMStar, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "um/mod", ficlPrimitiveUMSlashMod, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "until", ficlPrimitiveUntilCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "variable", ficlPrimitiveVariable, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "while", ficlPrimitiveWhileCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "word", ficlPrimitiveWord, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "[", ficlPrimitiveLeftBracketCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "[\']", ficlPrimitiveBracketTickCoIm,FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "[char]", ficlPrimitiveCharCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "]", ficlPrimitiveRightBracket, FICL_WORD_DEFAULT); /* ** The Core Extensions word set... ** see softcore.fr for other definitions */ /* "#tib" */ ficlDictionarySetPrimitive(dictionary, ".(", ficlPrimitiveDotParen, FICL_WORD_IMMEDIATE); /* ".r" */ ficlDictionarySetPrimitive(dictionary, ":noname", ficlPrimitiveColonNoName, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "?do", ficlPrimitiveQDoCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "again", ficlPrimitiveAgainCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "c\"", ficlPrimitiveCountedStringQuoteIm, FICL_WORD_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "hex", ficlPrimitiveHex, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "pad", ficlPrimitivePad, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "parse", ficlPrimitiveParse, FICL_WORD_DEFAULT); /* query restore-input save-input tib u.r u> unused [FICL_VM_STATE_COMPILE] */ ficlDictionarySetPrimitive(dictionary, "refill", ficlPrimitiveRefill, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "source-id", ficlPrimitiveSourceID, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "to", ficlPrimitiveToValue, FICL_WORD_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "value", ficlPrimitiveConstant, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "\\", ficlPrimitiveBackslash, FICL_WORD_IMMEDIATE); /* ** Environment query values for the Core word set */ ficlDictionarySetConstant(environment, "/counted-string", FICL_COUNTED_STRING_MAX); ficlDictionarySetConstant(environment, "/hold", FICL_PAD_SIZE); ficlDictionarySetConstant(environment, "/pad", FICL_PAD_SIZE); ficlDictionarySetConstant(environment, "address-unit-bits", 8); ficlDictionarySetConstant(environment, "core", FICL_TRUE); ficlDictionarySetConstant(environment, "core-ext", FICL_FALSE); ficlDictionarySetConstant(environment, "floored", FICL_FALSE); ficlDictionarySetConstant(environment, "max-char", UCHAR_MAX); ficlDictionarySetConstant(environment, "max-n", 0x7fffffff); ficlDictionarySetConstant(environment, "max-u", 0xffffffff); { ficl2Unsigned combined; FICL_2UNSIGNED_SET(INT_MAX, UINT_MAX, combined); ficlDictionarySet2Constant(environment,"max-d", FICL_2UNSIGNED_TO_2INTEGER(combined)); FICL_2UNSIGNED_SET(UINT_MAX, UINT_MAX, combined); ficlDictionarySet2Constant(environment,"max-ud", FICL_2UNSIGNED_TO_2INTEGER(combined)); } ficlDictionarySetConstant(environment, "return-stack-cells",FICL_DEFAULT_STACK_SIZE); ficlDictionarySetConstant(environment, "stack-cells", FICL_DEFAULT_STACK_SIZE); /* ** The optional Double-Number word set (partial) */ ficlDictionarySetPrimitive(dictionary, "2constant", ficlPrimitive2Constant, FICL_WORD_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "2value", ficlPrimitive2Constant, FICL_WORD_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "2literal", ficlPrimitive2LiteralIm, FICL_WORD_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "2variable", ficlPrimitive2Variable, FICL_WORD_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "dnegate", ficlPrimitiveDNegate, FICL_WORD_DEFAULT); /* ** The optional Exception and Exception Extensions word set */ ficlDictionarySetPrimitive(dictionary, "catch", ficlPrimitiveCatch, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "throw", ficlPrimitiveThrow, FICL_WORD_DEFAULT); ficlDictionarySetConstant(environment, "exception", FICL_TRUE); ficlDictionarySetConstant(environment, "exception-ext", FICL_TRUE); /* ** The optional Locals and Locals Extensions word set ** see softcore.c for implementation of locals| */ #if FICL_WANT_LOCALS ficlDictionarySetPrimitive(dictionary, "doLocal", ficlPrimitiveDoLocalIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "(local)", ficlPrimitiveLocalParen, FICL_WORD_COMPILE_ONLY); ficlDictionarySetPrimitive(dictionary, "(2local)", ficlPrimitive2LocalParen, FICL_WORD_COMPILE_ONLY); ficlDictionarySetConstant(environment, "locals", FICL_TRUE); ficlDictionarySetConstant(environment, "locals-ext", FICL_TRUE); ficlDictionarySetConstant(environment, "#locals", FICL_MAX_LOCALS); #endif /* ** The optional Memory-Allocation word set */ ficlDictionarySetPrimitive(dictionary, "allocate", ficlPrimitiveAllocate, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "free", ficlPrimitiveFree, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "resize", ficlPrimitiveResize, FICL_WORD_DEFAULT); ficlDictionarySetConstant(environment, "memory-alloc", FICL_TRUE); /* ** The optional Search-Order word set */ ficlSystemCompileSearch(system); /* ** The optional Programming-Tools and Programming-Tools Extensions word set */ ficlSystemCompileTools(system); /* ** The optional File-Access and File-Access Extensions word set */ #if FICL_WANT_FILE ficlSystemCompileFile(system); #endif /* ** Ficl extras */ ficlDictionarySetPrimitive(dictionary, ".ver", ficlPrimitiveVersion, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, ">name", ficlPrimitiveToName, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "add-parse-step", ficlPrimitiveAddParseStep, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "body>", ficlPrimitiveFromBody, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "compile-only", ficlPrimitiveCompileOnly, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "endif", ficlPrimitiveEndifCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "last-word", ficlPrimitiveLastWord, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "hash", ficlPrimitiveHash, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "objectify", ficlPrimitiveSetObjectFlag, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "?object", ficlPrimitiveIsObject, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "parse-word",ficlPrimitiveParseNoCopy, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "sfind", ficlPrimitiveSFind, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "sliteral", ficlPrimitiveSLiteralCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE); ficlDictionarySetPrimitive(dictionary, "sprintf", ficlPrimitiveSprintf, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "strlen", ficlPrimitiveStrlen, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "x.", ficlPrimitiveHexDot, FICL_WORD_DEFAULT); #if FICL_WANT_USER ficlDictionarySetPrimitive(dictionary, "user", ficlPrimitiveUser, FICL_WORD_DEFAULT); #endif /* ** internal support words */ interpret = ficlDictionarySetPrimitive(dictionary, "interpret", ficlPrimitiveInterpret, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "lookup", ficlPrimitiveLookup, FICL_WORD_DEFAULT); ficlDictionarySetPrimitive(dictionary, "(parse-step)", ficlPrimitiveParseStepParen, FICL_WORD_DEFAULT); system->exitInnerWord = ficlDictionarySetPrimitive(dictionary, "exit-inner",ficlPrimitiveExitInner, FICL_WORD_DEFAULT); /* ** Set constants representing the internal instruction words ** If you want all of 'em, turn that "#if 0" to "#if 1". ** By default you only get the numbers (fi0, fiNeg1, etc). */ #define FICL_TOKEN(token, description) ficlDictionarySetConstant(dictionary, #token, token); #if 0 #define FICL_INSTRUCTION_TOKEN(token, description, flags) ficlDictionarySetConstant(dictionary, #token, token); #else #define FICL_INSTRUCTION_TOKEN(token, description, flags) #endif /* 0 */ #include "ficltokens.h" #undef FICL_TOKEN #undef FICL_INSTRUCTION_TOKEN /* ** Set up system's outer interpreter loop - maybe this should be in initSystem? */ system->interpreterLoop[0] = (ficlInstruction)interpret; system->interpreterLoop[1] = (ficlInstruction)ficlInstructionBranchParen; system->interpreterLoop[2] = (ficlInstruction)(void *)(-2); FICL_SYSTEM_ASSERT(system, ficlDictionaryCellsAvailable(dictionary) > 0); return; }