#include "ficl.h"
#include <limits.h>
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";
static void
markBranch(ficlDictionary *dictionary, ficlVm *vm, char *tag)
{
ficlStackPushPointer(vm->dataStack, dictionary->here);
ficlStackPushPointer(vm->dataStack, tag);
}
static void
markControlTag(ficlVm *vm, char *tag)
{
ficlStackPushPointer(vm->dataStack, tag);
}
static void
matchControlTag(ficlVm *vm, char *wantTag)
{
char *tag;
FICL_STACK_CHECK(vm->dataStack, 1, 0);
tag = (char *)ficlStackPopPointer(vm->dataStack);
if ((tag != wantTag) && strcmp(tag, wantTag)) {
ficlVmThrowError(vm,
"Error -- unmatched control structure \"%s\"", wantTag);
}
}
static void
resolveBackBranch(ficlDictionary *dictionary, ficlVm *vm, char *tag)
{
ficlCell *patchAddr, c;
matchControlTag(vm, tag);
FICL_STACK_CHECK(vm->dataStack, 1, 0);
patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack);
c.i = patchAddr - dictionary->here;
ficlDictionaryAppendCell(dictionary, c);
}
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).i = offset;
}
static void
resolveAbsBranch(ficlDictionary *dictionary, ficlVm *vm, char *wantTag)
{
ficlCell *patchAddr;
char *tag;
FICL_STACK_CHECK(vm->dataStack, 2, 0);
tag = ficlStackPopPointer(vm->dataStack);
if ((tag != wantTag) && strcmp(tag, wantTag)) {
ficlVmTextOut(vm, "Warning -- Unmatched control word: ");
ficlVmTextOut(vm, wantTag);
ficlVmTextOut(vm, "\n");
}
patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack);
(*patchAddr).p = dictionary->here;
}
static void
ficlPrimitiveColon(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlString name = ficlVmGetWord(vm);
vm->state = FICL_VM_STATE_COMPILE;
markControlTag(vm, colonTag);
(void) ficlDictionaryAppendWord(dictionary, name,
(ficlPrimitive)ficlInstructionColonParen,
FICL_WORD_DEFAULT | FICL_WORD_SMUDGED);
#if FICL_WANT_LOCALS
vm->callback.system->localsCount = 0;
#endif
}
static void
ficlPrimitiveSemicolonCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
matchControlTag(vm, colonTag);
#if FICL_WANT_LOCALS
if (vm->callback.system->localsCount > 0) {
ficlDictionary *locals;
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);
}
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);
}
static void
ficlPrimitiveConstant(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlString name = ficlVmGetWord(vm);
FICL_STACK_CHECK(vm->dataStack, 1, 0);
(void) ficlDictionaryAppendConstantInstruction(dictionary, name,
ficlInstructionConstantParen, ficlStackPopInteger(vm->dataStack));
}
static void
ficlPrimitive2Constant(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlString name = ficlVmGetWord(vm);
FICL_STACK_CHECK(vm->dataStack, 2, 0);
(void) ficlDictionaryAppend2ConstantInstruction(dictionary, name,
ficlInstruction2ConstantParen, ficlStackPop2Integer(vm->dataStack));
}
static void
ficlPrimitiveDot(ficlVm *vm)
{
ficlCell c;
FICL_STACK_CHECK(vm->dataStack, 1, 0);
c = ficlStackPop(vm->dataStack);
(void) ficlLtoa((c).i, vm->pad, vm->base);
(void) strcat(vm->pad, " ");
ficlVmTextOut(vm, vm->pad);
}
static void
ficlPrimitiveUDot(ficlVm *vm)
{
ficlUnsigned u;
FICL_STACK_CHECK(vm->dataStack, 1, 0);
u = ficlStackPopUnsigned(vm->dataStack);
(void) ficlUltoa(u, vm->pad, vm->base);
(void) strcat(vm->pad, " ");
ficlVmTextOut(vm, vm->pad);
}
static void
ficlPrimitiveHexDot(ficlVm *vm)
{
ficlUnsigned u;
FICL_STACK_CHECK(vm->dataStack, 1, 0);
u = ficlStackPopUnsigned(vm->dataStack);
(void) ficlUltoa(u, vm->pad, 16);
(void) strcat(vm->pad, " ");
ficlVmTextOut(vm, vm->pad);
}
static void
ficlPrimitiveStrlen(ficlVm *vm)
{
char *address = (char *)ficlStackPopPointer(vm->dataStack);
ficlStackPushInteger(vm->dataStack, strlen(address));
}
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;
int append = 1;
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;
case 'd':
case 'D': {
int integer;
integer = ficlStackPopInteger(vm->dataStack);
if (unsignedInteger)
(void) ficlUltoa(integer, scratch,
base);
else
(void) ficlLtoa(integer, scratch, base);
base = 10;
unsignedInteger = 0;
source = scratch;
actualLength = strlen(scratch);
break;
}
case '%':
source = format;
actualLength = 1;
default:
continue;
}
}
if (append) {
if (!desiredLength)
desiredLength = actualLength;
if (desiredLength > bufferLength) {
append = 0;
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, FICL_BOOL(!append));
}
static void
ficlPrimitiveDepth(ficlVm *vm)
{
int i;
FICL_STACK_CHECK(vm->dataStack, 0, 1);
i = ficlStackDepth(vm->dataStack);
ficlStackPushInteger(vm->dataStack, i);
}
static void
ficlPrimitiveEmit(ficlVm *vm)
{
char buffer[2];
int i;
FICL_STACK_CHECK(vm->dataStack, 1, 0);
i = ficlStackPopInteger(vm->dataStack);
buffer[0] = (char)i;
buffer[1] = '\0';
ficlVmTextOut(vm, buffer);
}
static void
ficlPrimitiveCR(ficlVm *vm)
{
ficlVmTextOut(vm, "\n");
}
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;
}
if (trace != stop) {
trace++;
if ((trace != stop) && (c != *trace) &&
((*trace == '\r') || (*trace == '\n')))
trace++;
}
ficlVmUpdateTib(vm, trace);
}
static void
ficlPrimitiveParenthesis(ficlVm *vm)
{
(void) ficlVmParseStringEx(vm, ')', 0);
}
static void
ficlPrimitiveIfCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlDictionaryAppendUnsigned(dictionary,
ficlInstructionBranch0ParenWithCheck);
markBranch(dictionary, vm, origTag);
ficlDictionaryAppendUnsigned(dictionary, 1);
}
static void
ficlPrimitiveElseCoIm(ficlVm *vm)
{
ficlCell *patchAddr;
ficlInteger offset;
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlDictionaryAppendUnsigned(dictionary,
ficlInstructionBranchParenWithCheck);
matchControlTag(vm, origTag);
patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack);
markBranch(dictionary, vm, origTag);
ficlDictionaryAppendUnsigned(dictionary, 1);
offset = dictionary->here - patchAddr;
(*patchAddr).i = offset;
}
static void
ficlPrimitiveEndifCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
resolveForwardBranch(dictionary, vm, origTag);
}
static void
ficlPrimitiveCaseCoIm(ficlVm *vm)
{
FICL_STACK_CHECK(vm->dataStack, 0, 2);
ficlStackPushUnsigned(vm->dataStack, 0);
markControlTag(vm, caseTag);
}
static void
ficlPrimitiveEndcaseCoIm(ficlVm *vm)
{
ficlUnsigned fixupCount;
ficlDictionary *dictionary;
ficlCell *patchAddr;
ficlInteger offset;
if (ficlStackGetTop(vm->dataStack).p == fallthroughTag) {
matchControlTag(vm, fallthroughTag);
patchAddr = 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).i = offset;
}
}
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 = 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).i = offset;
}
}
static void
ficlPrimitiveEndofCoIm(ficlVm *vm)
{
ficlCell *patchAddr;
ficlUnsigned fixupCount;
ficlInteger offset;
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
FICL_STACK_CHECK(vm->dataStack, 4, 3);
matchControlTag(vm, ofTag);
patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack);
matchControlTag(vm, caseTag);
fixupCount = ficlStackPopUnsigned(vm->dataStack);
ficlDictionaryAppendUnsigned(dictionary,
ficlInstructionBranchParenWithCheck);
ficlStackPushPointer(vm->dataStack, dictionary->here);
ficlStackPushUnsigned(vm->dataStack, fixupCount + 1);
markControlTag(vm, caseTag);
ficlDictionaryAppendUnsigned(dictionary, 2);
offset = dictionary->here - patchAddr;
(*patchAddr).i = offset;
}
static void
ficlPrimitiveFallthroughCoIm(ficlVm *vm)
{
ficlCell *patchAddr;
ficlInteger offset;
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
FICL_STACK_CHECK(vm->dataStack, 4, 3);
matchControlTag(vm, ofTag);
patchAddr = (ficlCell *)ficlStackPopPointer(vm->dataStack);
matchControlTag(vm, caseTag);
markControlTag(vm, caseTag);
ficlDictionaryAppendUnsigned(dictionary,
ficlInstructionBranchParenWithCheck);
ficlStackPushPointer(vm->dataStack, dictionary->here);
markControlTag(vm, fallthroughTag);
ficlDictionaryAppendUnsigned(dictionary, 2);
offset = dictionary->here - patchAddr;
(*patchAddr).i = offset;
}
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));
}
static void
ficlPrimitiveInterpret(ficlVm *vm)
{
ficlString s;
int i;
ficlSystem *system;
FICL_VM_ASSERT(vm, vm);
system = vm->callback.system;
s = ficlVmGetWord0(vm);
if (s.length == 0) {
ficlVmThrow(vm, FICL_VM_STATUS_OUT_OF_TEXT);
}
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));
(void) ficlVmExecuteXT(vm, word);
if (ficlStackPopInteger(vm->dataStack))
return;
}
}
ficlVmThrowError(vm, "%.*s not found", FICL_STRING_GET_LENGTH(s),
FICL_STRING_GET_POINTER(s));
}
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));
}
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));
}
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))
(void) ficlSystemAddParseStep(vm->callback.system, pStep);
}
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;
}
}
static void
ficlPrimitive2LiteralIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlDictionaryAppendUnsigned(dictionary, ficlInstruction2LiteralParen);
ficlDictionaryAppendCell(dictionary, ficlStackPop(vm->dataStack));
ficlDictionaryAppendCell(dictionary, ficlStackPop(vm->dataStack));
}
static void
ficlPrimitiveDoCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlDictionaryAppendUnsigned(dictionary, ficlInstructionDoParen);
markBranch(dictionary, vm, leaveTag);
ficlDictionaryAppendUnsigned(dictionary, 0);
markBranch(dictionary, vm, doTag);
}
static void
ficlPrimitiveQDoCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlDictionaryAppendUnsigned(dictionary, ficlInstructionQDoParen);
markBranch(dictionary, vm, leaveTag);
ficlDictionaryAppendUnsigned(dictionary, 0);
markBranch(dictionary, vm, doTag);
}
static void
ficlPrimitiveLoopCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlDictionaryAppendUnsigned(dictionary, ficlInstructionLoopParen);
resolveBackBranch(dictionary, vm, doTag);
resolveAbsBranch(dictionary, vm, leaveTag);
}
static void
ficlPrimitivePlusLoopCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlDictionaryAppendUnsigned(dictionary, ficlInstructionPlusLoopParen);
resolveBackBranch(dictionary, vm, doTag);
resolveAbsBranch(dictionary, vm, leaveTag);
}
static void
ficlPrimitiveVariable(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlString name = ficlVmGetWord(vm);
(void) ficlDictionaryAppendWord(dictionary, name,
(ficlPrimitive)ficlInstructionVariableParen, FICL_WORD_DEFAULT);
ficlVmDictionaryAllotCells(vm, dictionary, 1);
}
static void
ficlPrimitive2Variable(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlString name = ficlVmGetWord(vm);
(void) ficlDictionaryAppendWord(dictionary, name,
(ficlPrimitive)ficlInstructionVariableParen, FICL_WORD_DEFAULT);
ficlVmDictionaryAllotCells(vm, dictionary, 2);
}
static void
ficlPrimitiveBase(ficlVm *vm)
{
ficlCell *pBase, c;
FICL_STACK_CHECK(vm->dataStack, 0, 1);
pBase = (ficlCell *)(&vm->base);
c.p = pBase;
ficlStackPush(vm->dataStack, c);
}
static void
ficlPrimitiveDecimal(ficlVm *vm)
{
vm->base = 10;
}
static void
ficlPrimitiveHex(ficlVm *vm)
{
vm->base = 16;
}
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);
}
static void
ficlPrimitiveHere(ficlVm *vm)
{
ficlDictionary *dictionary;
FICL_STACK_CHECK(vm->dataStack, 0, 1);
dictionary = ficlVmGetDictionary(vm);
ficlStackPushPointer(vm->dataStack, dictionary->here);
}
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)
ficlVmThrowError(vm, "%.*s not found",
FICL_STRING_GET_LENGTH(name),
FICL_STRING_GET_POINTER(name));
ficlStackPushPointer(vm->dataStack, word);
}
static void
ficlPrimitiveBracketTickCoIm(ficlVm *vm)
{
ficlPrimitiveTick(vm);
ficlPrimitiveLiteralIm(vm);
}
static void
ficlPrimitivePostponeCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlWord *word;
ficlWord *pComma = ficlSystemLookup(vm->callback.system, ",");
ficlCell c;
FICL_VM_ASSERT(vm, pComma);
ficlPrimitiveTick(vm);
word = ficlStackGetTop(vm->dataStack).p;
if (ficlWordIsImmediate(word)) {
ficlDictionaryAppendCell(dictionary,
ficlStackPop(vm->dataStack));
} else {
ficlPrimitiveLiteralIm(vm);
c.p = pComma;
ficlDictionaryAppendCell(dictionary, c);
}
}
static void
ficlPrimitiveExecute(ficlVm *vm)
{
ficlWord *word;
FICL_STACK_CHECK(vm->dataStack, 1, 0);
word = ficlStackPopPointer(vm->dataStack);
ficlVmExecuteWord(vm, word);
}
static void
ficlPrimitiveImmediate(ficlVm *vm)
{
FICL_IGNORE(vm);
ficlDictionarySetImmediate(ficlVmGetDictionary(vm));
}
static void
ficlPrimitiveCompileOnly(ficlVm *vm)
{
FICL_IGNORE(vm);
ficlDictionarySetFlags(ficlVmGetDictionary(vm), FICL_WORD_COMPILE_ONLY);
}
static void
ficlPrimitiveSetObjectFlag(ficlVm *vm)
{
FICL_IGNORE(vm);
ficlDictionarySetFlags(ficlVmGetDictionary(vm), FICL_WORD_OBJECT);
}
static void
ficlPrimitiveIsObject(ficlVm *vm)
{
ficlInteger flag;
ficlWord *word = (ficlWord *)ficlStackPopPointer(vm->dataStack);
flag = ((word != NULL) && (word->flags & FICL_WORD_OBJECT))?
FICL_TRUE : FICL_FALSE;
ficlStackPushInteger(vm->dataStack, flag);
}
static void
ficlPrimitiveCountedStringQuoteIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
if (vm->state == FICL_VM_STATE_INTERPRET) {
ficlCountedString *counted = (ficlCountedString *)
dictionary->here;
(void) ficlVmGetString(vm, counted, '\"');
ficlStackPushPointer(vm->dataStack, counted);
ficlVmDictionaryAllot(vm, dictionary,
counted->length + sizeof (ficlUnsigned8));
} else {
ficlDictionaryAppendUnsigned(dictionary,
ficlInstructionCStringLiteralParen);
dictionary->here =
FICL_POINTER_TO_CELL(ficlVmGetString(vm,
(ficlCountedString *)dictionary->here, '\"'));
ficlDictionaryAlign(dictionary);
}
}
static void
ficlPrimitiveDotQuoteCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlWord *pType = ficlSystemLookup(vm->callback.system, "type");
ficlCell c;
FICL_VM_ASSERT(vm, pType);
ficlDictionaryAppendUnsigned(dictionary,
ficlInstructionStringLiteralParen);
dictionary->here =
FICL_POINTER_TO_CELL(ficlVmGetString(vm,
(ficlCountedString *)dictionary->here, '\"'));
ficlDictionaryAlign(dictionary);
c.p = pType;
ficlDictionaryAppendCell(dictionary, c);
}
static void
ficlPrimitiveDotParen(ficlVm *vm)
{
char *from = ficlVmGetInBuf(vm);
char *stop = ficlVmGetInBufEnd(vm);
char *to = vm->pad;
char c;
for (c = *from; (from != stop) && (c != ')'); c = *++from)
*to++ = c;
*to = '\0';
if ((from != stop) && (c == ')'))
from++;
ficlVmTextOut(vm, vm->pad);
ficlVmUpdateTib(vm, from);
}
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 = 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));
}
static void ficlPrimitiveState(ficlVm *vm)
{
FICL_STACK_CHECK(vm->dataStack, 0, 1);
ficlStackPushPointer(vm->dataStack, &vm->state);
}
static void
ficlPrimitiveCreate(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlString name = ficlVmGetWord(vm);
(void) ficlDictionaryAppendWord(dictionary, name,
(ficlPrimitive)ficlInstructionCreateParen, FICL_WORD_DEFAULT);
ficlVmDictionaryAllotCells(vm, dictionary, 1);
}
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);
}
static void
ficlPrimitiveToBody(ficlVm *vm)
{
ficlWord *word;
FICL_STACK_CHECK(vm->dataStack, 1, 1);
word = ficlStackPopPointer(vm->dataStack);
ficlStackPushPointer(vm->dataStack, word->param + 1);
}
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);
}
static void
ficlPrimitiveToName(ficlVm *vm)
{
ficlWord *word;
FICL_STACK_CHECK(vm->dataStack, 1, 2);
word = ficlStackPopPointer(vm->dataStack);
ficlStackPushPointer(vm->dataStack, word->name);
ficlStackPushUnsigned(vm->dataStack, word->length);
}
static void
ficlPrimitiveLastWord(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlWord *wp = dictionary->smudge;
ficlCell c;
FICL_VM_ASSERT(vm, wp);
c.p = wp;
ficlVmPush(vm, c);
}
static void
ficlPrimitiveLeftBracketCoIm(ficlVm *vm)
{
vm->state = FICL_VM_STATE_INTERPRET;
}
static void
ficlPrimitiveRightBracket(ficlVm *vm)
{
vm->state = FICL_VM_STATE_COMPILE;
}
static void
ficlPrimitiveLessNumberSign(ficlVm *vm)
{
ficlCountedString *counted = FICL_POINTER_TO_COUNTED_STRING(vm->pad);
counted->length = 0;
}
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);
}
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;
(void) ficlStringReverse(counted->text);
ficlStackDrop(vm->dataStack, 2);
ficlStackPushPointer(vm->dataStack, counted->text);
ficlStackPushUnsigned(vm->dataStack, counted->length);
}
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);
}
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;
}
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++] = '-';
}
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;
if (digit >= base)
break;
accumulator = ficl2UnsignedMultiplyAccumulate(accumulator,
base, digit);
}
ficlStackPush2Unsigned(vm->dataStack, accumulator);
ficlStackPushPointer(vm->dataStack, trace);
ficlStackPushUnsigned(vm->dataStack, length);
}
static void
ficlPrimitiveQuit(ficlVm *vm)
{
ficlVmThrow(vm, FICL_VM_STATUS_QUIT);
}
static void
ficlPrimitiveAbort(ficlVm *vm)
{
ficlVmThrow(vm, FICL_VM_STATUS_ABORT);
}
static void
ficlPrimitiveAccept(ficlVm *vm)
{
ficlUnsigned size;
char *address;
int c;
ficlUnsigned length = 0;
FICL_STACK_CHECK(vm->dataStack, 2, 1);
size = ficlStackPopInteger(vm->dataStack);
address = ficlStackPopPointer(vm->dataStack);
while (size != length) {
c = getchar();
if (c == '\n' || c == '\r')
break;
address[length++] = c;
}
ficlStackPushInteger(vm->dataStack, length);
}
static void
ficlPrimitiveAlign(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
FICL_IGNORE(vm);
ficlDictionaryAlign(dictionary);
}
static void
ficlPrimitiveAligned(ficlVm *vm)
{
void *addr;
FICL_STACK_CHECK(vm->dataStack, 1, 1);
addr = ficlStackPopPointer(vm->dataStack);
ficlStackPushPointer(vm->dataStack, ficlAlignPointer(addr));
}
static void
ficlPrimitiveBeginCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
markBranch(dictionary, vm, destTag);
}
static void
ficlPrimitiveUntilCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlDictionaryAppendUnsigned(dictionary,
ficlInstructionBranch0ParenWithCheck);
resolveBackBranch(dictionary, vm, destTag);
}
static void
ficlPrimitiveWhileCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
FICL_STACK_CHECK(vm->dataStack, 2, 5);
ficlDictionaryAppendUnsigned(dictionary,
ficlInstructionBranch0ParenWithCheck);
markBranch(dictionary, vm, origTag);
ficlStackRoll(vm->dataStack, 3);
ficlStackRoll(vm->dataStack, 3);
ficlDictionaryAppendUnsigned(dictionary, 1);
}
static void
ficlPrimitiveRepeatCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlDictionaryAppendUnsigned(dictionary,
ficlInstructionBranchParenWithCheck);
resolveBackBranch(dictionary, vm, destTag);
resolveForwardBranch(dictionary, vm, origTag);
}
static void
ficlPrimitiveAgainCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlDictionaryAppendUnsigned(dictionary,
ficlInstructionBranchParenWithCheck);
resolveBackBranch(dictionary, vm, destTag);
}
static void
ficlPrimitiveChar(ficlVm *vm)
{
ficlString s;
FICL_STACK_CHECK(vm->dataStack, 0, 1);
s = ficlVmGetWord(vm);
ficlStackPushUnsigned(vm->dataStack, (ficlUnsigned)(s.text[0]));
}
static void
ficlPrimitiveCharCoIm(ficlVm *vm)
{
ficlPrimitiveChar(vm);
ficlPrimitiveLiteralIm(vm);
}
static void
ficlPrimitiveCharPlus(ficlVm *vm)
{
char *p;
FICL_STACK_CHECK(vm->dataStack, 1, 1);
p = ficlStackPopPointer(vm->dataStack);
ficlStackPushPointer(vm->dataStack, p + 1);
}
#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));
}
}
#if defined(_M_IX86)
#pragma warning(default: 4127)
#endif
static void
ficlPrimitiveCount(ficlVm *vm)
{
ficlCountedString *counted;
FICL_STACK_CHECK(vm->dataStack, 1, 2);
counted = ficlStackPopPointer(vm->dataStack);
ficlStackPushPointer(vm->dataStack, counted->text);
ficlStackPushUnsigned(vm->dataStack, counted->length);
}
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 = ficlStackPopPointer(vm->dataStack);
word = ficlDictionaryLookup(environment, name);
if (word != NULL) {
ficlVmExecuteWord(vm, word);
ficlStackPushInteger(vm->dataStack, FICL_TRUE);
} else {
ficlStackPushInteger(vm->dataStack, FICL_FALSE);
}
}
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);
}
static void
ficlPrimitiveStringQuoteIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
if (vm->state == FICL_VM_STATE_INTERPRET) {
ficlCountedString *counted;
counted = (ficlCountedString *)dictionary->here;
(void) ficlVmGetString(vm, counted, '\"');
ficlStackPushPointer(vm->dataStack, counted->text);
ficlStackPushUnsigned(vm->dataStack, counted->length);
} else {
ficlDictionaryAppendUnsigned(dictionary,
ficlInstructionStringLiteralParen);
dictionary->here = FICL_POINTER_TO_CELL(
ficlVmGetString(vm, (ficlCountedString *)dictionary->here,
'\"'));
ficlDictionaryAlign(dictionary);
}
}
static void
ficlPrimitiveType(ficlVm *vm)
{
ficlUnsigned length;
char *s;
FICL_STACK_CHECK(vm->dataStack, 2, 0);
length = ficlStackPopUnsigned(vm->dataStack);
s = ficlStackPopPointer(vm->dataStack);
if ((s == NULL) || (length == 0))
return;
if (s[length] != 0) {
char *here = (char *)ficlVmGetDictionary(vm)->here;
if (s != here)
(void) strncpy(here, s, length);
here[length] = '\0';
s = here;
}
ficlVmTextOut(vm, s);
}
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);
(void) strncpy(counted->text, FICL_STRING_GET_POINTER(name),
FICL_STRING_GET_LENGTH(name));
counted->text[counted->length] = ' ';
counted->text[counted->length + 1] = 0;
ficlStackPushPointer(vm->dataStack, counted);
}
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));
}
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));
}
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);
}
}
static void
ficlPrimitiveCFind(ficlVm *vm)
{
ficlCountedString *counted;
ficlString name;
FICL_STACK_CHECK(vm->dataStack, 1, 2);
counted = ficlStackPopPointer(vm->dataStack);
FICL_STRING_SET_FROM_COUNTED_STRING(name, *counted);
do_find(vm, name, counted);
}
static void
ficlPrimitiveSFind(ficlVm *vm)
{
ficlString name;
FICL_STACK_CHECK(vm->dataStack, 2, 2);
name.length = ficlStackPopInteger(vm->dataStack);
name.text = ficlStackPopPointer(vm->dataStack);
do_find(vm, name, NULL);
}
static void
ficlPrimitiveRecurseCoIm(ficlVm *vm)
{
ficlDictionary *dictionary = ficlVmGetDictionary(vm);
ficlCell c;
FICL_IGNORE(vm);
c.p = dictionary->smudge;
ficlDictionaryAppendCell(dictionary, c);
}
static void
ficlPrimitiveSource(ficlVm *vm)
{
FICL_STACK_CHECK(vm->dataStack, 0, 2);
ficlStackPushPointer(vm->dataStack, vm->tib.text);
ficlStackPushInteger(vm->dataStack, ficlVmGetInBufLen(vm));
}
static void
ficlPrimitiveVersion(ficlVm *vm)
{
ficlVmTextOut(vm, "Ficl version " FICL_VERSION "\n");
}
static void
ficlPrimitiveToIn(ficlVm *vm)
{
FICL_STACK_CHECK(vm->dataStack, 0, 1);
ficlStackPushPointer(vm->dataStack, &vm->tib.index);
}
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);
}
#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");
}
(void) ficlDictionaryAppendWord(dictionary, name,
(ficlPrimitive)ficlInstructionUserParen, FICL_WORD_DEFAULT);
ficlDictionaryAppendCell(dictionary, c);
}
#endif
#if FICL_WANT_LOCALS
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);
isFloat = 0;
#endif
if (vm->state == FICL_VM_STATE_INTERPRET) {
ficlStack *stack;
#if FICL_WANT_FLOAT
if (isFloat)
stack = vm->floatStack;
else
#endif
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
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)
ficlDictionaryAppendUnsigned(dictionary, nLocal);
}
}
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
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) {
ficlPrimitive code;
ficlInstruction instruction;
ficlDictionary *locals;
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);
isFloat = 0;
#else
if (isFloat) {
if (isDouble) {
code = ficlPrimitiveDoF2LocalIm;
instruction = ficlInstructionToF2LocalParen;
} else {
code = ficlPrimitiveDoFLocalIm;
instruction = ficlInstructionToFLocalParen;
}
} else
#endif
if (isDouble) {
code = ficlPrimitiveDo2LocalIm;
instruction = ficlInstructionTo2LocalParen;
} else {
code = ficlPrimitiveDoLocalIm;
instruction = ficlInstructionToLocalParen;
}
(void) ficlDictionaryAppendWord(locals, name, code,
FICL_WORD_COMPILE_ONLY_IMMEDIATE);
ficlDictionaryAppendUnsigned(locals,
vm->callback.system->localsCount);
if (vm->callback.system->localsCount == 0) {
ficlDictionaryAppendUnsigned(dictionary,
ficlInstructionLinkParen);
vm->callback.system->localsFixup = dictionary->here;
ficlDictionaryAppendUnsigned(dictionary,
vm->callback.system->localsCount);
}
ficlDictionaryAppendUnsigned(dictionary, instruction);
ficlDictionaryAppendUnsigned(dictionary,
vm->callback.system->localsCount);
vm->callback.system->localsCount += (isDouble) ? 2 : 1;
} else if (vm->callback.system->localsCount > 0) {
*(ficlInteger *)(vm->callback.system->localsFixup) =
vm->callback.system->localsCount;
}
}
static void
ficlPrimitiveLocalParen(ficlVm *vm)
{
ficlLocalParen(vm, 0, 0);
}
static void
ficlPrimitive2LocalParen(ficlVm *vm)
{
ficlLocalParen(vm, 1, 0);
}
#endif
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
#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
else {
ficlVmThrowError(vm,
"to %.*s : local is of unknown type",
FICL_STRING_GET_LENGTH(name),
FICL_STRING_GET_POINTER(name));
}
nLocal = word->param[0].i;
appendLocalOffset = FICL_TRUE;
#if FICL_WANT_FLOAT
if (!isFloat) {
#endif
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
ficlDictionaryAppendUnsigned(dictionary, instruction);
if (appendLocalOffset)
ficlDictionaryAppendUnsigned(dictionary, nLocal);
return;
}
#endif
#if FICL_WANT_LOCALS
TO_GLOBAL:
#endif
word = ficlDictionaryLookup(dictionary, name);
if (!word)
ficlVmThrowError(vm, "%.*s not found",
FICL_STRING_GET_LENGTH(name),
FICL_STRING_GET_POINTER(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
default:
ficlVmThrowError(vm,
"to %.*s : value/constant is of unknown type",
FICL_STRING_GET_LENGTH(name),
FICL_STRING_GET_POINTER(name));
break;
}
if (vm->state == FICL_VM_STATE_INTERPRET) {
word->param[0] = ficlStackPop(stack);
if (isDouble)
word->param[1] = ficlStackPop(stack);
} else {
ficlStackPushPointer(vm->dataStack, &word->param[0]);
ficlPrimitiveLiteralIm(vm);
ficlDictionaryAppendUnsigned(dictionary, instruction);
}
}
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));
}
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));
}
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);
}
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));
}
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);
}
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);
}
static void
ficlPrimitive2Rot(ficlVm *vm)
{
ficl2Integer d1, d2, d3;
FICL_STACK_CHECK(vm->dataStack, 6, 6);
d3 = ficlStackPop2Integer(vm->dataStack);
d2 = ficlStackPop2Integer(vm->dataStack);
d1 = ficlStackPop2Integer(vm->dataStack);
ficlStackPush2Integer(vm->dataStack, d2);
ficlStackPush2Integer(vm->dataStack, d3);
ficlStackPush2Integer(vm->dataStack, d1);
}
static void
ficlPrimitivePad(ficlVm *vm)
{
ficlStackPushPointer(vm->dataStack, vm->pad);
}
static void
ficlPrimitiveSourceID(ficlVm *vm)
{
ficlStackPushInteger(vm->dataStack, vm->sourceId.i);
}
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);
}
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);
FICL_STACK_CHECK(vm->dataStack, 1, 0);
word = ficlStackPopPointer(vm->dataStack);
memcpy((void*)&vmCopy, (void*)vm, sizeof (ficlVm));
memcpy((void*)&dataStackCopy, (void*)vm->dataStack, sizeof (ficlStack));
memcpy((void*)&returnStackCopy, (void*)vm->returnStack,
sizeof (ficlStack));
vm->exceptionHandler = &vmState;
except = setjmp(vmState);
switch (except) {
case 0:
ficlVmPushIP(vm, &(vm->callback.system->exitInnerWord));
ficlVmExecuteWord(vm, word);
ficlVmInnerLoop(vm, 0);
break;
case FICL_VM_STATUS_INNER_EXIT:
ficlVmPopIP(vm);
vm->exceptionHandler = vmCopy.exceptionHandler;
ficlStackPushInteger(vm->dataStack, 0);
break;
default:
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);
break;
}
}
static void
ficlPrimitiveThrow(ficlVm *vm)
{
int except;
except = ficlStackPopInteger(vm->dataStack);
if (except)
ficlVmThrow(vm, except);
}
static void
ficlPrimitiveAllocate(ficlVm *vm)
{
size_t size;
void *p;
size = ficlStackPopInteger(vm->dataStack);
p = ficlMalloc(size);
ficlStackPushPointer(vm->dataStack, p);
if (p != NULL)
ficlStackPushInteger(vm->dataStack, 0);
else
ficlStackPushInteger(vm->dataStack, 1);
}
static void
ficlPrimitiveFree(ficlVm *vm)
{
void *p;
p = ficlStackPopPointer(vm->dataStack);
ficlFree(p);
ficlStackPushInteger(vm->dataStack, 0);
}
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);
}
}
static void
ficlPrimitiveExitInner(ficlVm *vm)
{
ficlVmThrow(vm, FICL_VM_STATUS_INNER_EXIT);
}
#if 0
static void
ficlPrimitiveName(ficlVm *vm)
{
FICL_IGNORE(vm);
}
#endif
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) \
(void) ficlDictionarySetInstruction(dictionary, description, token, \
flags);
#include "ficltokens.h"
#undef FICL_TOKEN
#undef FICL_INSTRUCTION_TOKEN
(void) ficlDictionarySetPrimitive(dictionary, "#",
ficlPrimitiveNumberSign, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "#>",
ficlPrimitiveNumberSignGreater, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "#s",
ficlPrimitiveNumberSignS, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "\'",
ficlPrimitiveTick, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "(",
ficlPrimitiveParenthesis, FICL_WORD_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "+loop",
ficlPrimitivePlusLoopCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, ".",
ficlPrimitiveDot, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, ".\"",
ficlPrimitiveDotQuoteCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, ":",
ficlPrimitiveColon, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, ";",
ficlPrimitiveSemicolonCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "<#",
ficlPrimitiveLessNumberSign, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, ">body",
ficlPrimitiveToBody, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, ">in",
ficlPrimitiveToIn, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, ">number",
ficlPrimitiveToNumber, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "abort",
ficlPrimitiveAbort, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "accept",
ficlPrimitiveAccept, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "align",
ficlPrimitiveAlign, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "aligned",
ficlPrimitiveAligned, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "allot",
ficlPrimitiveAllot, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "base",
ficlPrimitiveBase, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "begin",
ficlPrimitiveBeginCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "case",
ficlPrimitiveCaseCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "char",
ficlPrimitiveChar, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "char+",
ficlPrimitiveCharPlus, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "chars",
ficlPrimitiveChars, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "constant",
ficlPrimitiveConstant, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "count",
ficlPrimitiveCount, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "cr",
ficlPrimitiveCR, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "create",
ficlPrimitiveCreate, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "decimal",
ficlPrimitiveDecimal, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "depth",
ficlPrimitiveDepth, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "do",
ficlPrimitiveDoCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "does>",
ficlPrimitiveDoesCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "else",
ficlPrimitiveElseCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "emit",
ficlPrimitiveEmit, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "endcase",
ficlPrimitiveEndcaseCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "endof",
ficlPrimitiveEndofCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "environment?",
ficlPrimitiveEnvironmentQ, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "evaluate",
ficlPrimitiveEvaluate, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "execute",
ficlPrimitiveExecute, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "exit",
ficlPrimitiveExitCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "fallthrough",
ficlPrimitiveFallthroughCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "find",
ficlPrimitiveCFind, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "fm/mod",
ficlPrimitiveFMSlashMod, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "here",
ficlPrimitiveHere, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "hold",
ficlPrimitiveHold, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "if",
ficlPrimitiveIfCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "immediate",
ficlPrimitiveImmediate, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "literal",
ficlPrimitiveLiteralIm, FICL_WORD_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "loop",
ficlPrimitiveLoopCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "m*",
ficlPrimitiveMStar, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "mod",
ficlPrimitiveMod, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "of",
ficlPrimitiveOfCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "postpone",
ficlPrimitivePostponeCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "quit",
ficlPrimitiveQuit, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "recurse",
ficlPrimitiveRecurseCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "repeat",
ficlPrimitiveRepeatCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "s\"",
ficlPrimitiveStringQuoteIm, FICL_WORD_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "sign",
ficlPrimitiveSign, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "sm/rem",
ficlPrimitiveSMSlashRem, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "source",
ficlPrimitiveSource, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "state",
ficlPrimitiveState, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "then",
ficlPrimitiveEndifCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "type",
ficlPrimitiveType, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "u.",
ficlPrimitiveUDot, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "um*",
ficlPrimitiveUMStar, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "um/mod",
ficlPrimitiveUMSlashMod, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "until",
ficlPrimitiveUntilCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "variable",
ficlPrimitiveVariable, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "while",
ficlPrimitiveWhileCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "word",
ficlPrimitiveWord, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "[",
ficlPrimitiveLeftBracketCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "[\']",
ficlPrimitiveBracketTickCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "[char]",
ficlPrimitiveCharCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "]",
ficlPrimitiveRightBracket, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, ".(",
ficlPrimitiveDotParen, FICL_WORD_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, ":noname",
ficlPrimitiveColonNoName, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "?do",
ficlPrimitiveQDoCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "again",
ficlPrimitiveAgainCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "c\"",
ficlPrimitiveCountedStringQuoteIm, FICL_WORD_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "hex",
ficlPrimitiveHex, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "pad",
ficlPrimitivePad, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "parse",
ficlPrimitiveParse, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "refill",
ficlPrimitiveRefill, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "source-id",
ficlPrimitiveSourceID, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "to",
ficlPrimitiveToValue, FICL_WORD_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "value",
ficlPrimitiveConstant, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "\\",
ficlPrimitiveBackslash, FICL_WORD_IMMEDIATE);
(void) ficlDictionarySetConstant(environment, "/counted-string",
FICL_COUNTED_STRING_MAX);
(void) ficlDictionarySetConstant(environment, "/hold", FICL_PAD_SIZE);
(void) ficlDictionarySetConstant(environment, "/pad", FICL_PAD_SIZE);
(void) ficlDictionarySetConstant(environment, "address-unit-bits", 8);
(void) ficlDictionarySetConstant(environment, "core", FICL_TRUE);
(void) ficlDictionarySetConstant(environment, "core-ext", FICL_FALSE);
(void) ficlDictionarySetConstant(environment, "floored", FICL_FALSE);
(void) ficlDictionarySetConstant(environment, "max-char", UCHAR_MAX);
(void) ficlDictionarySetConstant(environment, "max-n", LONG_MAX);
(void) ficlDictionarySetConstant(environment, "max-u", ULONG_MAX);
{
ficl2Integer id;
ficlInteger low, high;
low = ULONG_MAX;
high = LONG_MAX;
FICL_2INTEGER_SET(high, low, id);
(void) ficlDictionarySet2Constant(environment, "max-d", id);
high = ULONG_MAX;
FICL_2INTEGER_SET(high, low, id);
(void) ficlDictionarySet2Constant(environment, "max-ud", id);
}
(void) ficlDictionarySetConstant(environment, "return-stack-cells",
FICL_DEFAULT_STACK_SIZE);
(void) ficlDictionarySetConstant(environment, "stack-cells",
FICL_DEFAULT_STACK_SIZE);
(void) ficlDictionarySetPrimitive(dictionary, "2constant",
ficlPrimitive2Constant, FICL_WORD_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "2literal",
ficlPrimitive2LiteralIm, FICL_WORD_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "2variable",
ficlPrimitive2Variable, FICL_WORD_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "2rot",
ficlPrimitive2Rot, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "2value",
ficlPrimitive2Constant, FICL_WORD_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "catch",
ficlPrimitiveCatch, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "throw",
ficlPrimitiveThrow, FICL_WORD_DEFAULT);
(void) ficlDictionarySetConstant(environment, "exception", FICL_TRUE);
(void) ficlDictionarySetConstant(environment, "exception-ext",
FICL_TRUE);
#if FICL_WANT_LOCALS
(void) ficlDictionarySetPrimitive(dictionary, "doLocal",
ficlPrimitiveDoLocalIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "(local)",
ficlPrimitiveLocalParen, FICL_WORD_COMPILE_ONLY);
(void) ficlDictionarySetPrimitive(dictionary, "(2local)",
ficlPrimitive2LocalParen, FICL_WORD_COMPILE_ONLY);
(void) ficlDictionarySetConstant(environment, "locals", FICL_TRUE);
(void) ficlDictionarySetConstant(environment, "locals-ext", FICL_TRUE);
(void) ficlDictionarySetConstant(environment, "#locals",
FICL_MAX_LOCALS);
#endif
(void) ficlDictionarySetPrimitive(dictionary, "allocate",
ficlPrimitiveAllocate, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "free",
ficlPrimitiveFree, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "resize",
ficlPrimitiveResize, FICL_WORD_DEFAULT);
(void) ficlDictionarySetConstant(environment, "memory-alloc",
FICL_TRUE);
ficlSystemCompileSearch(system);
ficlSystemCompileTools(system);
#if FICL_WANT_FILE
ficlSystemCompileFile(system);
#endif
(void) ficlDictionarySetPrimitive(dictionary, ".ver",
ficlPrimitiveVersion, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, ">name",
ficlPrimitiveToName, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "add-parse-step",
ficlPrimitiveAddParseStep, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "body>",
ficlPrimitiveFromBody, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "compile-only",
ficlPrimitiveCompileOnly, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "endif",
ficlPrimitiveEndifCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "last-word",
ficlPrimitiveLastWord, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "hash",
ficlPrimitiveHash, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "objectify",
ficlPrimitiveSetObjectFlag, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "?object",
ficlPrimitiveIsObject, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "parse-word",
ficlPrimitiveParseNoCopy, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "sfind",
ficlPrimitiveSFind, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "sliteral",
ficlPrimitiveSLiteralCoIm, FICL_WORD_COMPILE_ONLY_IMMEDIATE);
(void) ficlDictionarySetPrimitive(dictionary, "sprintf",
ficlPrimitiveSprintf, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "strlen",
ficlPrimitiveStrlen, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "x.",
ficlPrimitiveHexDot, FICL_WORD_DEFAULT);
#if FICL_WANT_USER
(void) ficlDictionarySetPrimitive(dictionary, "user",
ficlPrimitiveUser, FICL_WORD_DEFAULT);
#endif
interpret = ficlDictionarySetPrimitive(dictionary, "interpret",
ficlPrimitiveInterpret, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "lookup",
ficlPrimitiveLookup, FICL_WORD_DEFAULT);
(void) ficlDictionarySetPrimitive(dictionary, "(parse-step)",
ficlPrimitiveParseStepParen, FICL_WORD_DEFAULT);
system->exitInnerWord = ficlDictionarySetPrimitive(dictionary,
"exit-inner", ficlPrimitiveExitInner, FICL_WORD_DEFAULT);
#define FICL_TOKEN(token, description) \
(void) 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
#include "ficltokens.h"
#undef FICL_TOKEN
#undef FICL_INSTRUCTION_TOKEN
system->interpreterLoop[0] = interpret;
system->interpreterLoop[1] = (ficlWord *)ficlInstructionBranchParen;
system->interpreterLoop[2] = (ficlWord *)(void *)(-2);
FICL_SYSTEM_ASSERT(system,
ficlDictionaryCellsAvailable(dictionary) > 0);
}