aboutsummaryrefslogtreecommitdiff
path: root/engines/glk/alan2/parse.cpp
diff options
context:
space:
mode:
authorPaul Gilbert2019-06-16 21:39:47 -0700
committerPaul Gilbert2019-06-22 14:40:22 -0700
commit84dfa6d3b83e080f099ad606dc7fbeb40ebe4b97 (patch)
tree6f90969f77df66f0eff50365688930af49f5c0b3 /engines/glk/alan2/parse.cpp
parent004df0ef90f8a4b6367b529f66aeffe65e824977 (diff)
downloadscummvm-rg350-84dfa6d3b83e080f099ad606dc7fbeb40ebe4b97.tar.gz
scummvm-rg350-84dfa6d3b83e080f099ad606dc7fbeb40ebe4b97.tar.bz2
scummvm-rg350-84dfa6d3b83e080f099ad606dc7fbeb40ebe4b97.zip
GLK: ALAN2: Remove previously added code fragments
Diffstat (limited to 'engines/glk/alan2/parse.cpp')
-rw-r--r--engines/glk/alan2/parse.cpp969
1 files changed, 0 insertions, 969 deletions
diff --git a/engines/glk/alan2/parse.cpp b/engines/glk/alan2/parse.cpp
deleted file mode 100644
index 8e6427d09c..0000000000
--- a/engines/glk/alan2/parse.cpp
+++ /dev/null
@@ -1,969 +0,0 @@
-/* ScummVM - Graphic Adventure Engine
- *
- * ScummVM is the legal property of its developers, whose names
- * are too numerous to list here. Please refer to the COPYRIGHT
- * file distributed with this source distribution.
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License
- * as published by the Free Software Foundation; either version 2
- * of the License, or (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
- *
- */
-
-#include "common/stack.h"
-#include "glk/alan2/alan2.h"
-#include "glk/alan2/execute.h"
-#include "glk/alan2/interpreter.h"
-#include "glk/alan2/parse.h"
-#include "glk/alan2/types.h"
-#include "glk/alan2/util.h"
-#include "common/debug.h"
-#include "common/file.h"
-#include "decode.h"
-
-namespace Glk {
-namespace Alan2 {
-
-uint32 litCount = 0;
-
-// All procedures for getting a command and turning it into a list of
-// dictionary entries are placed here.
-
-Parser::Parser() {
- wrds[0] = EOF;
- plural = false;
-
- // TODO
-}
-
-void Parser::unknown(char *inputStr) {
- Common::String str = Common::String::format("'%s'?", inputStr);
-
- // TODO
-#if 0
-#if ISO == 0
- fromIso(str, str);
-#endif
-#endif
-
- _vm->output(str);
- eol = true;
- _vm->printError(M_UNKNOWN_WORD);
-}
-
-int Parser::lookup(char *wrd) {
- for (int i = 0; !endOfTable(&dict[i]); i++) {
- if (strcmp(wrd, (char *)addrTo(dict[i].wrd)) == 0)
- return i;
- }
-
- unknown(wrd);
- return EOF;
-}
-
-char *Parser::gettoken(char *tokenBuffer) {
- static char *marker;
- static char oldch;
-
- if (tokenBuffer == nullptr)
- *marker = oldch;
- else
- marker = tokenBuffer;
-
- while (*marker != '\0' && Common::isSpace(*marker) && *marker != '\n') marker++;
- tokenBuffer = marker;
-
- if (Common::isAlpha(*marker))
- while (*marker && (Common::isAlpha(*marker) || Common::isDigit(*marker) || *marker == '\'')) marker++;
- else if (Common::isDigit(*marker))
- while (Common::isDigit(*marker)) marker++;
- else if (*marker == '\"') {
- marker++;
- while (*marker != '\"') marker++;
- marker++;
- } else if (*marker == '\0' || *marker == '\n')
- return nullptr;
- else
- marker++;
-
- oldch = *marker;
- *marker = '\0';
-
- return tokenBuffer;
-}
-
-void Parser::agetline() {
- // static char buf[LISTLEN + 1]; // The input buffer
- // static char isobuf[LISTLEN + 1]; // The input buffer in ISO
-
- _vm->paragraph();
-
- // TODO
-#if 0
-
- do {
-#if defined(HAVE_ANSI) || defined(GLK)
- statusline();
-#endif
- debug("> ");
-
-#if 0
- if (logflg)
- fprintf(logfil, "> ");
-#endif
-
-#ifdef USE_READLINE
- if (!readline(buf)) {
- newline();
- quit();
- }
-#else
- if (fgets(buf, LISTLEN, stdin) == nullptr) {
- newline();
- quit();
- }
-#endif
-
- getPageSize();
- anyOutput = FALSE;
-
- if (logflg)
- fprintf(logfil, "%s\n", buf);
-
-#if ISO == 0
- toIso(isobuf, buf, NATIVECHARSET);
-#else
- strcpy(isobuf, buf);
-#endif
-
- token = gettoken(isobuf);
-
- if (token != nullptr && strcmp("debug", token) == 0 && _vm->header->debug) {
- dbgflg = true;
- debug();
- token = nullptr;
- }
- } while (token == nullptr);
-
- eol = false;
- lin = 1;
-
-#endif
-}
-
-void Parser::scan() {
- uint i;
- // int w;
-
- agetline();
- wrds[0] = 0;
-
- for (i = 0; i < litCount; i++)
- if (litValues[i].type == TYPSTR && litValues[i].value != 0)
- free((char *) litValues[i].value);
-
- litCount = 0;
-
- do {
- if (Common::isAlpha(token[0])) {
- Common::String tmp = token;
- tmp.toLowercase();
- strcpy(token, tmp.c_str());
-
- // w = lookup(token);
-
- // TODO
- //if (!isNoise(w))
- // wrds[i++] = w;
- } else if (Common::isDigit(token[0])) {
- if (litCount > MAXPARAMS)
- error("Too many parameters.");
-
- wrds[i++] = dictsize + litCount; // Word outside dictionary = literal
- litValues[litCount].type = TYPNUM;
- litValues[litCount++].value = atoi(token);
- } else if (token[0] == '\"') {
- if (litCount > MAXPARAMS)
- error("Too many parameters.");
-
- wrds[i++] = dictsize + litCount; // Word outside dictionary = literal
- litValues[litCount].type = TYPSTR;
-
- // Remove the string quotes while copying
- Common::String tmp = token;
- tmp.deleteChar(0);
- tmp.deleteLastChar();
- char *str = scumm_strdup(tmp.c_str());
-
- litValues[litCount++].value = (Aptr) str;
- } else if (token[0] == ',') {
- //wrds[i++] = conjWord; // TODO
- } else
- unknown(token);
-
- wrds[i] = EOF;
- eol = (token = gettoken(nullptr)) == nullptr;
- } while (!eol);
-}
-
-void Parser::nonverb() {
- if (isDir(wrds[wrdidx])) {
- wrdidx++;
- if (wrds[wrdidx] != EOF && !isConj(wrds[wrdidx]))
- _vm->printError(M_WHAT);
-// TODO
-#if 0
- else
- go(dict[wrds[wrdidx-1]].code);
-#endif
-
- if (wrds[wrdidx] != EOF)
- wrdidx++;
- } else
- _vm->printError(M_WHAT);
-}
-
-Aword Parser::where(Aword id) {
- if (isObj(id))
- return objloc(id);
- else if (isAct(id))
- return acts[id - ACTMIN].loc;
- else
- error("Can't WHERE item (%ld).", (unsigned long)id);
-}
-
-Aword Parser::objloc(Aword obj) {
- if (isCnt(objs[obj - OBJMIN].loc)) // In something ?
- if (isObj(objs[obj - OBJMIN].loc) || isAct(objs[obj - OBJMIN].loc))
- return where(objs[obj - OBJMIN].loc);
- else // Containers not anywhere is where the hero is!
- return where(HERO);
- else
- return(objs[obj - OBJMIN].loc);
-}
-
-Abool Parser::objhere(Aword obj) {
- if (isCnt(objs[obj - OBJMIN].loc)) { // In something?
- if (isObj(objs[obj - OBJMIN].loc) || isAct(objs[obj - OBJMIN].loc))
- return isHere(objs[obj - OBJMIN].loc);
- else // If the container wasn't anywhere, assume where HERO is!
- return (int)where(HERO) == _vm->cur.loc;
- } else
- return((int)objs[obj - OBJMIN].loc == _vm->cur.loc);
-}
-
-
-Abool Parser::isHere(Aword id) {
- if (isObj(id))
- return objhere(id);
- else if (isAct(id))
- return (int)acts[id - ACTMIN].loc == _vm->cur.loc;
- else
- error("Can't HERE item (%ld).", (unsigned long)id);
-}
-
-// ----------------------------------------------------------------------------
-
-void Parser::buildall(ParamElem list[]) {
- uint o, i = 0;
- bool found = false;
-
- for (o = OBJMIN; o <= OBJMAX; o++) {
- if (isHere(o)) {
- found = true;
- list[i].code = o;
- list[i++].firstWord = (Aword)EOF;
- }
- }
-
- if (!found)
- _vm->printError(M_WHAT_ALL);
- else
- list[i].code = (Aword)EOF;
-}
-
-void Parser::listCopy(ParamElem a[], ParamElem b[]) {
- int i;
-
- for (i = 0; b[i].code != (Aword)EOF; i++)
- a[i] = b[i];
-
- a[i].code = (Aword)EOF;
-}
-
-bool Parser::listContains(ParamElem l[], Aword e) {
- int i;
-
- for (i = 0; l[i].code != (Aword)EOF && l[i].code != e; i++);
-
- return (l[i].code == e);
-}
-
-void Parser::listIntersection(ParamElem a[], ParamElem b[]) {
- int i, last = 0;
-
- for (i = 0; a[i].code != (Aword)EOF; i++)
- if (listContains(b, a[i].code))
- a[last++] = a[i];
-
- a[last].code = (Aword)EOF;
-}
-
-void Parser::listCopyFromDictionary(ParamElem p[], Aword r[]) {
- int i;
-
- for (i = 0; r[i] != (Aword)EOF; i++) {
- p[i].code = r[i];
- p[i].firstWord = (Aword)EOF;
- }
-
- p[i].code = (Aword)EOF;
-}
-
-int Parser::listLength(ParamElem a[]) {
- int i = 0;
-
- while (a[i].code != (Aword)EOF)
- i++;
-
- return (i);
-}
-
-void Parser::listCompact(ParamElem a[]) {
- int i, j;
-
- for (i = 0, j = 0; a[j].code != (Aword)EOF; j++)
- if (a[j].code != 0)
- a[i++] = a[j];
-
- a[i].code = (Aword)EOF;
-}
-
-void Parser::listMerge(ParamElem a[], ParamElem b[]) {
- int i, last;
-
- for (last = 0; a[last].code != (Aword)EOF; last++); // Find end of list
-
- for (i = 0; b[i].code != (Aword)EOF; i++) {
- if (!listContains(a, b[i].code)) {
- a[last++] = b[i];
- a[last].code = (Aword)EOF;
- }
- }
-}
-
-void Parser::listSubtract(ParamElem a[], ParamElem b[]) {
- for (int i = 0; a[i].code != (Aword)EOF; i++)
- if (listContains(b, a[i].code))
- a[i].code = 0; // Mark empty
-
- listCompact(a);
-}
-
-void Parser::unambig(ParamElem plst[]) {
- int i;
- bool found = false; // Adjective or noun found?
- static ParamElem *refs; // Entities referenced by word
- static ParamElem *savlst; // Saved list for backup at EOF
- int firstWord, lastWord; // The words the player used
-
- if (refs == nullptr)
- refs = new ParamElem[MAXENTITY + 1];
-
- if (savlst == nullptr)
- savlst = new ParamElem[MAXENTITY + 1];
-
- if (isLiteral(wrds[wrdidx])) {
- // Transform the word into a reference to the literal value
- plst[0].code = wrds[wrdidx++] - dictsize + LITMIN;
- plst[0].firstWord = (Aword)EOF; // No words used!
- plst[1].code = (Aword)EOF;
- return;
- }
-
- plst[0].code = (Aword)EOF; // Make empty
-
- if (isIt(wrds[wrdidx])) {
- wrdidx++;
-
- // Use last object in previous command!
- for (i = listLength(pparams)-1; i >= 0 && (pparams[i].code == 0 || pparams[i].code >= LITMIN); i--);
-
- if (i < 0)
- _vm->printError(M_WHAT_IT);
-
- if (!isHere(pparams[i].code)) {
- params[0].code = pparams[i].code;
- params[0].firstWord = (Aword)EOF;
- params[1].code = (Aword)EOF;
- _vm->printError(M_NO_SUCH);
- }
-
- plst[0] = pparams[i];
- plst[0].firstWord = (Aword)EOF; // No words used!
- plst[1].code = (Aword)EOF;
- return;
- }
-
- firstWord = wrdidx;
-
- while (wrds[wrdidx] != EOF && isAdj(wrds[wrdidx])) {
- // If this word can be a noun and there is no noun following break loop
- if (isNoun(wrds[wrdidx]) && (wrds[wrdidx+1] == EOF || !isNoun(wrds[wrdidx+1])))
- break;
-
- listCopyFromDictionary(refs, (Aword *)addrTo(dict[wrds[wrdidx]].adjrefs));
- listCopy(savlst, plst); // To save it for backtracking
-
- if (found)
- listIntersection(plst, refs);
- else {
- listCopy(plst, refs);
- found = true;
- }
-
- wrdidx++;
- }
-
- if (wrds[wrdidx] != EOF) {
- if (isNoun(wrds[wrdidx])) {
- listCopyFromDictionary(refs, (Aword *)addrTo(dict[wrds[wrdidx]].nounrefs));
- if (found)
- listIntersection(plst, refs);
- else {
- listCopy(plst, refs);
- found = true;
- }
-
- wrdidx++;
- } else
- _vm->printError(M_NOUN);
- } else if (found) {
- if (isNoun(wrds[wrdidx-1])) {
- // Perhaps the last word was also a noun?
- listCopy(plst, savlst); // Restore to before last adjective
- listCopyFromDictionary(refs, (Aword *)addrTo(dict[wrds[wrdidx-1]].nounrefs));
-
- if (plst[0].code == (Aword)EOF)
- listCopy(plst, refs);
- else
- listIntersection(plst, refs);
- } else
- _vm->printError(M_NOUN);
- }
-
- lastWord = wrdidx - 1;
-
- // Allow remote objects, but resolve ambiguities by presence
- if (listLength(plst) > 1) {
- for (i=0; plst[i].code != (Aword)EOF; i++)
- if (!isHere(plst[i].code))
- plst[i].code = 0;
-
- listCompact(plst);
- }
-
- if (listLength(plst) > 1 || (found && listLength(plst) == 0)) {
- params[0].code = 0; /* Just make it anything != EOF */
- params[0].firstWord = firstWord; /* Remember words for errors below */
- params[0].lastWord = lastWord;
- params[1].code = (Aword)EOF; /* But be sure to terminate */
-
- if (listLength(plst) > 1)
- _vm->printError(M_WHICH_ONE);
- else if (found && listLength(plst) == 0)
- _vm->printError(M_NO_SUCH);
- } else {
- plst[0].firstWord = firstWord;
- plst[0].lastWord = lastWord;
- }
-}
-
-void Parser::simple(ParamElem olst[]) {
- static ParamElem *tlst = nullptr;
- int savidx = wrdidx;
- bool savplur = false;
- int i;
-
- if (tlst == nullptr)
- tlst = new ParamElem[MAXENTITY + 1];
-
- tlst[0].code = (Aword)EOF;
-
- for (;;) {
- if (isThem(wrds[wrdidx])) {
- plural = true;
-
- for (i = 0; pmlst[i].code != (Aword)EOF; i++)
- if (!isHere(pmlst[i].code))
- pmlst[i].code = 0;
-
- listCompact(pmlst);
-
- if (listLength(pmlst) == 0)
- _vm->printError(M_WHAT_THEM);
-
- listCopy(olst, pmlst);
- olst[0].firstWord = (Aword)EOF; // No words used
- wrdidx++;
- } else {
- unambig(olst); // Look for unambigous noun phrase
-
- if (listLength(olst) == 0) { // Failed!
- listCopy(olst, tlst);
- wrdidx = savidx;
- plural = savplur;
- return;
- }
- }
-
- listMerge(tlst, olst);
-
- if (wrds[wrdidx] != EOF
- && (isConj(wrds[wrdidx])
- && (isAdj(wrds[wrdidx+1]) || isNoun(wrds[wrdidx+1])))) {
- // More parameters in a conjunction separated list ?
- savplur = plural;
- savidx = wrdidx;
- wrdidx++;
- plural = true;
- } else {
- listCopy(olst, tlst);
- return;
- }
- }
-}
-
-void Parser::complex(ParamElem olst[]) {
- // Above this procedure we can use the is* tests, but not below since
- // they work on words.Below all is converted to indices into the
- // entity tables.Particularly this goes for literals...
-
- static ParamElem *alst = nullptr;
-
- if (alst == nullptr)
- alst = new ParamElem[MAXENTITY + 1];
-
- if (isAll(wrds[wrdidx])) {
- plural = true;
- buildall(alst); // Build list of all objects
- wrdidx++;
- if (wrds[wrdidx] != EOF && isBut(wrds[wrdidx])) {
- wrdidx++;
- simple(olst);
-
- if (listLength(olst) == 0)
- _vm->printError(M_AFTER_BUT);
-
- listSubtract(alst, olst);
- if (listLength(alst) == 0)
- _vm->printError(M_NOT_MUCH);
- }
-
- listCopy(olst, alst);
- allLength = listLength(olst);
- } else
- simple(olst); // Look for simple noun group
-}
-
-bool Parser::claCheck(ClaElem *cla) {
- bool ok = false;
-
- if ((cla->classes & (Aword)CLA_OBJ) != 0)
- ok = ok || isObj(params[cla->code - 1].code);
- if ((cla->classes & (Aword)CLA_CNT) != 0)
- ok = ok || isCnt(params[cla->code - 1].code);
- if ((cla->classes & (Aword)CLA_ACT) != 0)
- ok = ok || isAct(params[cla->code - 1].code);
- if ((cla->classes & (Aword)CLA_NUM) != 0)
- ok = ok || isNum(params[cla->code - 1].code);
- if ((cla->classes & (Aword)CLA_STR) != 0)
- ok = ok || isStr(params[cla->code - 1].code);
- if ((cla->classes & (Aword)CLA_COBJ) != 0)
- ok = ok || (isCnt(params[cla->code - 1].code) && isObj(params[cla->code - 1].code));
- if ((cla->classes & (Aword)CLA_CACT) != 0)
- ok = ok || (isCnt(params[cla->code - 1].code) && isAct(params[cla->code - 1].code));
-
- return ok;
-}
-
-void Parser::resolve(ParamElem plst[]) {
- if (allLength > 0)
- return; // ALL has already done this
-
- // Resolve ambiguities by presence
- for (int i = 0; plst[i].code != (Aword)EOF; i++) {
- if (plst[i].code < LITMIN) { // Literals are always 'here'
- if (!isHere(plst[i].code)) {
- params[0] = plst[i]; // Copy error param as first one for message
- params[1].code = (Aword)EOF; // But be sure to terminate
- _vm->printError(M_NO_SUCH);
- }
- }
- }
-}
-
-bool Parser::endOfTable(StxElem *addr) {
- Aword *x = (Aword *)addr;
- return *x == (Aword)EOF;
-}
-
-bool Parser::endOfTable(ElmElem *addr) {
- Aword *x = (Aword *)addr;
- return *x == (Aword)EOF;
-}
-
-bool Parser::endOfTable(ClaElem *addr) {
- Aword *x = (Aword *)addr;
- return *x == (Aword)EOF;
-}
-
-bool Parser::endOfTable(VrbElem *addr) {
- Aword *x = (Aword *)addr;
- return *x == (Aword)EOF;
-}
-
-bool Parser::endOfTable(AltElem *addr) {
- Aword *x = (Aword *)addr;
- return *x == (Aword)EOF;
-}
-
-bool Parser::endOfTable(ChkElem *addr) {
- Aword *x = (Aword *)addr;
- return *x == (Aword)EOF;
-}
-
-bool Parser::endOfTable(WrdElem *addr) {
- Aword *x = (Aword *)addr;
- return *x == (Aword)EOF;
-}
-
-AltElem *Parser::findalt(Aword vrbsadr, Aword param) {
- VrbElem *vrb;
- AltElem *alt;
-
- if (vrbsadr == 0)
- return nullptr;
-
- for (vrb = (VrbElem *)addrTo(vrbsadr); !endOfTable(vrb); vrb++) {
- if ((int)vrb->code == _vm->cur.vrb) {
- for (alt = (AltElem *)addrTo(vrb->alts); !endOfTable(alt); alt++)
- if (alt->param == param || alt->param == 0)
- return alt;
- return nullptr;
- }
- }
-
- return nullptr;
-}
-
-bool Parser::trycheck(Aaddr adr, bool act) {
- ChkElem *chk = (ChkElem *)addrTo(adr);
-
- if (chk->exp == 0) {
- _vm->_interpreter->interpret(chk->stms);
- return false;
- } else {
- while (!endOfTable(chk)) {
- _vm->_interpreter->interpret(chk->exp);
- if (!(Abool)_vm->_stack->pop()) {
- if (act)
- _vm->_interpreter->interpret(chk->stms);
- return false;
- }
- chk++;
- }
- return true;
- }
-}
-
-bool Parser::possible() {
- AltElem *alt[MAXPARAMS + 2]; // List of alt-pointers, one for each param
- int i; // Parameter index
-
- _vm->fail = false;
- alt[0] = findalt(_vm->header->vrbs, 0);
-
- // Perform global checks
- if (alt[0] != 0 && alt[0]->checks != 0) {
- if (!trycheck(alt[0]->checks, false))
- return false;
- if (_vm->fail)
- return false;
- }
-
- // Now CHECKs in this location
- alt[1] = findalt(locs[_vm->cur.loc - LOCMIN].vrbs, 0);
- if (alt[1] != 0 && alt[1]->checks != 0)
- if (!trycheck(alt[1]->checks, false))
- return false;
-
- for (i = 0; params[i].code != (Aword)EOF; i++) {
- alt[i + 2] = findalt(objs[params[i].code - OBJMIN].vrbs, i + 1);
- // CHECKs in a possible parameter
- if (alt[i + 2] != 0 && alt[i + 2]->checks != 0)
- if (!trycheck(alt[i + 2]->checks, false))
- return false;
- }
-
- for (i = 0; i < 2 || params[i - 2].code != (Aword)EOF; i++)
- if (alt[i] != 0 && alt[i]->action != 0)
- break;
- if (i >= 2 && params[i - 2].code == (Aword)EOF)
- // Didn't find any code for this verb/object combination
- return false;
- else
- return true;
-}
-
-void Parser::tryMatch(ParamElem mlstArr[]) {
- ElmElem *elms; // Pointer to element list
- StxElem *stx; // Pointer to syntax list
- ClaElem *cla; // Pointer to class definitions
- bool anyPlural = false; // Any parameter that was plural?
- int i, p;
- static ParamElem *tlst = nullptr; // List of params found by complex()
- static bool *checked = nullptr; // Corresponding parameter checked?
-
- if (tlst == nullptr) {
- tlst = new ParamElem[MAXENTITY + 1];
- checked = new bool[MAXENTITY + 1];
- }
-
- for (stx = stxs; !endOfTable(stx); stx++)
- if ((int)stx->code == vrbcode)
- break;
-
- if (endOfTable(stx))
- _vm->printError(M_WHAT);
-
- elms = (ElmElem *) addrTo(stx->elms);
-
- while (true) {
- // End of input?
- if (wrds[wrdidx] == EOF || isConj(wrds[wrdidx])) {
- while (!endOfTable(elms) && elms->code != (Aword)EOS)
- elms++;
-
- if (endOfTable(elms))
- _vm->printError(M_WHAT);
- else
- break;
- } else {
- // A preposition?
- if (isPrep(wrds[wrdidx])) {
- while (!endOfTable(elms) && elms->code != dict[wrds[wrdidx]].code)
- elms++;
-
- if (endOfTable(elms))
- _vm->printError(M_WHAT);
- else
- wrdidx++;
- } else {
- // Must be a parameter!
- while (!endOfTable(elms) && elms->code != 0)
- elms++;
-
- if (endOfTable(elms))
- _vm->printError(M_WHAT);
-
- // Get it!
- plural = false;
- complex(tlst);
-
- if (listLength(tlst) == 0) // No object!?
- _vm->printError(M_WHAT);
-
- if ((elms->flags & OMNIBIT) == 0) // Omnipotent parameter?
- resolve(tlst); // If its not an omnipotent parameter, resolve by presence
-
- if (plural) {
- if ((elms->flags & MULTIPLEBIT) == 0) // Allowed multiple?
- _vm->printError(M_MULTIPLE);
- else {
- // Mark this as the multiple position in which to insert
- // actual parameter values later
- params[paramidx++].code = 0;
- listCopy(mlstArr, tlst);
- anyPlural = true;
- }
- } else
- params[paramidx++] = tlst[0];
-
- params[paramidx].code = (Aword)EOF;
- }
-
- elms = (ElmElem *) addrTo(elms->next);
- }
- }
-
- // Now perform class checks
- if (elms->next == 0) // No verb code, verb not declared!
- _vm->printError(M_CANT0);
-
- for (p = 0; params[p].code != (Aword)EOF; p++) /* Mark all parameters unchecked */
- checked[p] = false;
-
- for (cla = (ClaElem *) addrTo(elms->next); !endOfTable(cla); cla++) {
- if (params[cla->code - 1].code == 0) {
- // This was a multiple parameter, so check all and remove failing
- for (i = 0; mlstArr[i].code != (Aword)EOF; i++) {
- params[cla->code-1] = mlstArr[i];
- if (!claCheck(cla)) {
- // Multiple could be both an explicit list of params and an ALL
- if (allLength == 0) {
- char marker[80];
- // It wasn't ALL, we need to say something about it, so
- // prepare a printout with $1/2/3
- sprintf(marker, "($%ld)", (unsigned long) cla->code);
- _vm->output(marker);
- _vm->_interpreter->interpret(cla->stms);
- _vm->paragraph();
- }
-
- mlstArr[i].code = 0; // In any case remove it from the list
- }
- }
-
- params[cla->code - 1].code = 0;
- } else {
- if (!claCheck(cla)) {
- _vm->_interpreter->interpret(cla->stms);
- _vm->printError(MSGMAX); // Return to player without saying anything
- }
- }
-
- checked[cla->code - 1] = true; // Remember that it's already checked
- }
-
- // Now check the rest of the parameters, must be objects
- for (p = 0; params[p].code != (Aword)EOF; p++) {
- if (!checked[p]) {
- if (params[p].code == 0) {
- // This was a multiple parameter, check all and remove failing
- for (i = 0; mlstArr[i].code != (Aword)EOF; i++) {
- if (mlstArr[i].code != 0 && !isObj(mlstArr[i].code)) // Skip any empty slots
- mlstArr[i].code = 0;
- }
- } else if (!isObj(params[p].code))
- _vm->printError(M_CANT0);
- }
- }
-
- // Set verb code
- _vm->cur.vrb = ((Aword *) cla)[1]; // Take first word after end of table!
-
- // Finally, if ALL was used, try to find out what was applicable
- if (allLength > 0) {
- for (p = 0; params[p].code != 0; p++); // Find multiple marker
-
- for (i = 0; i < allLength; i++) {
- if (mlstArr[i].code != 0) { // Already empty?
- params[p] = mlstArr[i];
-
- if (!possible())
- mlstArr[i].code = 0; // Remove this from list
- }
- }
-
- params[p].code = 0; // Restore multiple marker
- listCompact(mlstArr);
-
- if (listLength(mlstArr) == 0) {
- params[0].code = (Aword)EOF;
- _vm->printError(M_WHAT_ALL);
- }
- } else if (anyPlural) {
- listCompact(mlstArr);
-
- // If there where multiple parameters but non left, exit without a
- // word, assuming we have already said enough
- if (listLength(mlstArr) == 0)
- _vm->printError(MSGMAX);
- }
-
- plural = anyPlural; // Remember that we found plural objects
-}
-
-void Parser::match(ParamElem *mlstArr) {
- tryMatch(mlstArr); // try to understand what the user said
-
- if (wrds[wrdidx] != EOF && !isConj(wrds[wrdidx]))
- _vm->printError(M_WHAT);
- if (wrds[wrdidx] != EOF) // More on this line?
- wrdidx++; // If so skip the AND
-}
-
-void Parser::action(ParamElem plst[]) {
- int i, mpos;
- char marker[10];
-
- if (plural) {
- // The code == 0 means this is a multiple position. We must loop
- // over this position (and replace it by each present in the plst)
- for (mpos = 0; params[mpos].code != 0; mpos++); // Find multiple position
-
- sprintf(marker, "($%d)", mpos + 1); // Prepare a printout with $1/2/3
-
- for (i = 0; plst[i].code != (Aword)EOF; i++) {
- params[mpos] = plst[i];
- _vm->output(marker);
- //do_it(); // TODO
-
- if (plst[i + 1].code != (Aword)EOF)
- _vm->paragraph();
- }
-
- params[mpos].code = 0;
- } //else // TODO
- //do_it();
-}
-
-void Parser::parse() {
- if (mlst == nullptr) { // Allocate large enough paramlists
- mlst = new ParamElem[MAXENTITY + 1];
- mlst[0].code = (Aword)EOF;
- pmlst = new ParamElem[MAXENTITY + 1];
- params = new ParamElem[MAXENTITY + 1];
- params[0].code = (Aword)EOF;
- pparams = new ParamElem[MAXENTITY + 1];
- }
-
- if (wrds[wrdidx] == EOF) {
- wrdidx = 0;
- scan();
- } else if (false/*anyOutput*/) // TODO
- _vm->paragraph();
-
- allLength = 0;
- paramidx = 0;
- listCopy(pparams, params);
- params[0].code = (Aword)EOF;
- listCopy(pmlst, mlst);
- mlst[0].code = (Aword)EOF;
-
- if (isVerb(wrds[wrdidx])) {
- vrbwrd = wrds[wrdidx];
- vrbcode = dict[vrbwrd].code;
- wrdidx++;
- match(mlst);
- action(mlst); // mlst contains possible multiple params
- } else {
- params[0].code = (Aword)EOF;
- pmlst[0].code = (Aword)EOF;
- nonverb();
- }
-}
-
-} // End of namespace Alan2
-} // End of namespace Glk