/* 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. * * $URL$ * $Id$ * */ #include "agos/agos.h" #include "agos/vga.h" namespace AGOS { #define OPCODE(x) _OPCODE(AGOSEngine_PN, x) void AGOSEngine_PN::setupOpcodes() { static const OpcodeEntryPN opcodes[] = { /* 00 */ OPCODE(opn_opcode00), OPCODE(opn_opcode01), OPCODE(opn_opcode02), OPCODE(opn_opcode03), /* 04 */ OPCODE(opn_opcode04), OPCODE(opn_opcode05), OPCODE(opn_opcode06), OPCODE(opn_opcode07), /* 08 */ OPCODE(opn_opcode08), OPCODE(opn_opcode09), OPCODE(opn_opcode10), OPCODE(opn_opcode11), /* 12 */ OPCODE(opn_opcode12), OPCODE(opn_opcode13), OPCODE(opn_opcode14), OPCODE(opn_opcode15), /* 16 */ OPCODE(opn_opcode16), OPCODE(opn_opcode17), OPCODE(opn_opcode18), OPCODE(opn_opcode19), /* 20 */ OPCODE(opn_opcode20), OPCODE(opn_opcode21), OPCODE(opn_opcode22), OPCODE(opn_opcode23), /* 24 */ OPCODE(opn_opcode24), OPCODE(opn_opcode25), OPCODE(opn_opcode26), OPCODE(opn_opcode27), /* 28 */ OPCODE(opn_opcode28), OPCODE(opn_opcode29), OPCODE(opn_opcode30), OPCODE(opn_opcode31), /* 32 */ OPCODE(opn_opcode32), OPCODE(opn_opcode33), OPCODE(opn_opcode34), OPCODE(opn_opcode35), /* 36 */ OPCODE(opn_opcode36), OPCODE(opn_opcode37), OPCODE(opn_opcode38), OPCODE(opn_opcode39), /* 40 */ OPCODE(opn_opcode40), OPCODE(opn_opcode41), OPCODE(opn_opcode42), OPCODE(opn_opcode43), /* 44 */ OPCODE(opn_opcode44), OPCODE(opn_opcode45), OPCODE(opn_opcode46), OPCODE(opn_opcode47), /* 48 */ OPCODE(opn_opcode48), OPCODE(opn_opcode49), OPCODE(opn_opcode50), OPCODE(opn_opcode51), /* 52 */ OPCODE(opn_opcode52), OPCODE(opn_opcode53), OPCODE(opn_opcode54), OPCODE(opn_opcode55), /* 56 */ OPCODE(opn_opcode56), OPCODE(opn_opcode57), OPCODE(o_invalid), OPCODE(o_invalid), /* 60 */ OPCODE(o_invalid), OPCODE(o_invalid), OPCODE(opn_opcode62), OPCODE(opn_opcode63), }; _opcodesPN = opcodes; _numOpcodes = 64; } void AGOSEngine_PN::executeOpcode(int opcode) { OpcodeProcPN op = _opcodesPN[opcode].proc; (this->*op) (); } #define readfromline() (_linct-- ? (int)*_workptr++ : readoverr()) int readoverr() { error("Internal Error - Line Over-run"); } // ----------------------------------------------------------------------- // Personal Nightmare Opcodes // ----------------------------------------------------------------------- void AGOSEngine_PN::opn_opcode00() { uint8 *str = _workptr; varval(); writeval(str, varval()); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode01() { uint8 *str = _workptr; int32 sp = varval() + varval(); _variableArray[12] = sp % 65536; _variableArray[13] = sp / 65536; if (sp > 65535) sp=65535; writeval(str, (int)sp); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode02() { uint8 *str = _workptr; int32 sp = varval(); sp -= varval(); _variableArray[12] = sp % 65536; _variableArray[13] = sp / 65536; if(sp < 0) sp = 0; writeval(str, (int)sp); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode03() { uint8 *str = _workptr; int32 sp = varval() * varval(); _variableArray[12] = sp % 65536; _variableArray[13] = sp / 65536; if (sp > 65535) sp = 65535; writeval(str, (int)sp); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode04() { uint8 *str = _workptr; int32 sp = varval(); int32 sp2 = varval(); if (sp2 == 0) error("Division by 0"); sp = sp / sp2; _variableArray[12] = sp % 65536; _variableArray[13] = sp / 65536; writeval(str, (int)sp); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode05() { pcf((uint8)'\n'); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode06() { pmesd(varval()); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode07() { int32 sp = varval(); plocd((int)sp, varval()); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode08() { int32 sp = varval(); pobjd((int)sp, varval()); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode09() { pmesd(varval()); pcf((uint8)'\n'); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode10() { int32 sp = varval(); plocd((int)sp, varval()); pcf((uint8)'\n'); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode11() { int32 sp = varval(); pobjd((int)sp, varval()); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode12() { char bf[8]; int a = 0; sprintf(bf,"%d", varval()); while(bf[a]) pcf(bf[a++]); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode13() { char bf[8]; int a = 0; sprintf(bf,"%d", varval()); while(bf[a]) pcf(bf[a++]); pcf((uint8)'\n'); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode14() { clearWindow(_windowArray[_curWindow]); pcf((uint8)255); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode15() { int32 x = varval();; if ((x < 0) || (x > 4)) x = 0; pcf((unsigned char)254); _curWindow = x; _xofs = (8 * _windowArray[_curWindow]->textLength) / 6 + 1; setScriptReturn(true); } void AGOSEngine_PN::opn_opcode16() { int32 sp = varval(); setScriptReturn((sp >= 0 && sp <= 4)); } void AGOSEngine_PN::opn_opcode17() { int16 v1 = varval(); int16 v2 = varval(); setScriptReturn(v1 < v2); } void AGOSEngine_PN::opn_opcode18() { int16 v1 = varval(); int16 v2 = varval(); setScriptReturn(v1 > v2); } void AGOSEngine_PN::opn_opcode19() { int16 v1 = varval(); int16 v2 = varval(); setScriptReturn(v1 == v2); } void AGOSEngine_PN::opn_opcode20() { int16 v1 = varval(); int16 v2 = varval(); setScriptReturn(v1 != v2); } void AGOSEngine_PN::opn_opcode21() { setposition(_procnum, varval()); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode22() { int pf[8]; int a; a = varval(); funcentry(pf, a); funccpy(pf); setposition(a, 0); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode23() { setScriptReturn(actCallD(varval())); } void AGOSEngine_PN::opn_opcode24() { popstack(-1); longjmp(*(_stackbase->savearea), 2); setScriptReturn(false); } void AGOSEngine_PN::opn_opcode25() { popstack(-1); longjmp(*(_stackbase->savearea), 1); setScriptReturn(false); } void AGOSEngine_PN::opn_opcode26() { while ((_stackbase->classnum != -1) && (_stackbase != NULL)) junkstack(); dumpstack(); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode27() { quitGame(); // Make sure the quit event is processed immediately. delay(0); } void AGOSEngine_PN::opn_opcode28() { addstack(varval()); _stackbase->savearea = _cjmpbuff; setScriptReturn(false); } void AGOSEngine_PN::opn_opcode29() { popstack(varval()); longjmp(*(_stackbase->savearea), -1); setScriptReturn(false); } void AGOSEngine_PN::opn_opcode30() { _variableArray[1] = varval(); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode31() { int a, slot = 0; char bf[60]; if ((a = varval()) > 2) { setScriptReturn(false); return; } switch (a) { case 0: getFilename(); slot = matchSaveGame(_saveFile, countSaveGames()); strcpy(bf, genSaveName(slot)); break; case 1: strcpy(bf, "test.sav"); break; case 2: // NOTE: Is this case ever used? error("opn_opcode31: case 2"); break; } a = loadfl(bf); if (a) setScriptReturn(badload(a)); else setScriptReturn(true); } void AGOSEngine_PN::opn_opcode32() { char bf[60]; int a; if ((a = varval()) > 2) { setScriptReturn(true); return; } switch (a) { case 0: getFilename(); strcpy(bf, genSaveName(countSaveGames())); break; case 1: strcpy(bf, "test.sav"); break; case 2: // NOTE: Is this case ever used? error("opn_opcode32: case 2"); break; } a = savfl(bf); setScriptReturn(a); } void AGOSEngine_PN::opn_opcode33() { setScriptReturn((varval() < 3) ? 1 : 0); } void AGOSEngine_PN::opn_opcode34() { uint16 msgNum1, msgNum2; varval(); getResponse((int)_variableArray[166], (int)_variableArray[167], msgNum1, msgNum2); _variableArray[168]= msgNum1; _variableArray[169]= msgNum2; setScriptReturn(true); } void AGOSEngine_PN::opn_opcode35() { int a; uint8 *sav = _workptr; varval(); a = varval(); if ((a = gvwrd((uint8 *)_wordcp, a)) == -1) { setScriptReturn(false); return; } writeval(sav, a); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode36() { int ct = 0; while (ct < _dataBase[57] + 1) _wordcp[ct++] = 0; ct = 1; if (isspace(*_inpp)) while ((*_inpp) && (isspace(*_inpp))) _inpp++; if (*_inpp == 0) { setScriptReturn(false); return; } _curwrdptr = _inpp; _wordcp[0] = *_inpp++; if ((_wordcp[0] == '.') || (_wordcp[0] == ',') || (_wordcp[0] == '"')) { setScriptReturn(true); return; } while ((*_inpp != '.') && (*_inpp != ',') && (!isspace(*_inpp)) && (*_inpp != '\0') && (*_inpp!='"')) { if (ct < _dataBase[57]) _wordcp[ct++] = *_inpp; _inpp++; } setScriptReturn(true); } void AGOSEngine_PN::opn_opcode37() { _curwrdptr = NULL; _inputReady = 1; interact(_inputline, 49); if ((_inpp = strchr(_inputline,'\n')) != NULL) *_inpp = '\0'; _inpp = _inputline; setScriptReturn(true); } void AGOSEngine_PN::opn_opcode38() { _noScanFlag = 1; clearInputLine(); writeval(_workptr, _keyPressed.ascii); _keyPressed.reset(); _noScanFlag = 0; varval(); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode39() { pcf((uint8)varval()); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode40() { setScriptReturn(doaction() | doaction()); } void AGOSEngine_PN::opn_opcode41() { setScriptReturn(doaction() & doaction()); } void AGOSEngine_PN::opn_opcode42() { setScriptReturn(doaction() ^ doaction()); } void AGOSEngine_PN::opn_opcode43() { setScriptReturn(!(doaction())); } void AGOSEngine_PN::opn_opcode44() { pcf((uint8)254); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode45() { uint8 *myptr; int x; if (_havinit == 0) { _seed = (int16)getTime(); _havinit = 1; } _seed = 1 + (75 * (_seed + 1) - 1) % 65537; myptr = _workptr; varval(); x = varval(); if (x == 0) error("Illegal range specified for RANDOM"); writeval(myptr, (_seed % x)); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode46() { char *x = _curwrdptr; if (x == NULL) { setScriptReturn(true); return; } pcf(*x); if ((*x == '.') || (*x == '"') || (*x == ',')) { setScriptReturn(true); return; } x++; while ((*x != '.') && (*x != ',') && (*x != '"') && (!isspace(*x)) && (*x != '\0')) pcf(*x++); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode47() { pmesd(varval() * 256 + varval()); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode48() { pmesd(varval() * 256 + varval()); pcf((uint8)'\n'); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode49() { setScriptReturn(findentry()); } void AGOSEngine_PN::opn_opcode50() { _fnst = 0; setScriptReturn(findset()); } void AGOSEngine_PN::opn_opcode51() { _fnst = varval(); setScriptReturn(findset()); } void AGOSEngine_PN::opn_opcode52() { int32 mode = varval(); if (mode == 1) { setWindowImage(mode, varval(), true); } else { setWindowImageEx(mode, varval()); } setScriptReturn(true); } void AGOSEngine_PN::opn_opcode53() { vc27_resetSprite(); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode54() { stopAnimate(varval()); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode55() { varval(); varval(); varval(); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode56() { varval(); varval(); varval(); setScriptReturn(true); } void AGOSEngine_PN::opn_opcode57() { uint16 windowNum = varval(); uint16 vgaSpriteId = varval(); int16 x = varval(); int16 y = varval(); uint16 palette = varval(); _lockWord |= 0x40; animate(windowNum, 0, vgaSpriteId, x, y, palette); _lockWord &= ~0x40; setScriptReturn(true); } void AGOSEngine_PN::opn_opcode62() { int32 zoneNum = varval(); _lockWord |= 0x80; vc29_stopAllSounds(); _hitCalled = 0; _oneClick = 0; loadZone(zoneNum); setWindowImage(2, 2); _copyScnFlag = 0; _vgaSpriteChanged = 0; _lockWord &= ~0x80; setScriptReturn(true); } void AGOSEngine_PN::opn_opcode63() { int a = readfromline(); switch (a) { case 65: setScriptReturn(inventoryOn(varval())); break; case 64: setScriptReturn((_lockWord & 0x10) != 0); break; case 63: setScriptReturn(inventoryOff()); break; default: error("opn_opcode63: unknown code %d", a); } } int AGOSEngine_PN::inventoryOn(int val) { writeVariable(210, val); if (_lockWord & 0x10) { iconPage(); } else { _lockWord |= 0x10; _hitAreaList = _invHitAreas; _windowArray[2]->textColor = 0; windowPutChar(_windowArray[2], 13); clearVideoWindow(4, 0); drawIconHitBar(); _objects = _variableArray[211]; _objectCountS = -1; iconPage(); } return 1; } int AGOSEngine_PN::inventoryOff() { if (_lockWord & 0x10) { _windowArray[2]->textColor = 15; restoreBlock(48, 2, 272, 130); _hitAreaList = _hitAreas; _lockWord &= ~0x10; _vgaSpriteChanged++; } return 1; } // ----------------------------------------------------------------------- // Personal Nightmare Script Code // ----------------------------------------------------------------------- static int bitvalue[8] = {128, 64, 32, 16, 8, 4, 2, 1}; int AGOSEngine_PN::bitextract(uint32 ptr, int offs) { return ((bitvalue[offs % 8] & _dataBase[ptr + offs / 8]) != 0); } uint16 AGOSEngine_PN::getptr(uint32 pos) { if (pos > _dataBaseSize) error("getptr: Read beyond EOF (%d)", pos); return (int)READ_LE_UINT16(_dataBase + pos); } uint32 AGOSEngine_PN::getlong(uint32 pos) { if (pos > _dataBaseSize) error("getlong: Read beyond EOF (%d)", pos); return (uint32)READ_LE_UINT24(_dataBase + pos); } int AGOSEngine_PN::varval() { int a; int b; if ((a = readfromline()) < 247) { return a; } switch (a) { case 249: b = readfromline(); return((int)(b + 256 * readfromline())); break; case 250: return(readfromline()); case 251: return((int)_variableArray[varval()]); case 252: b = varval(); return((int)_dataBase[_quickptr[0] + b * _quickshort[0] + varval()]); case 254: b = varval(); return((int)_dataBase[_quickptr[3] + b * _quickshort[2] + varval()]); case 247: b = varval(); return((int)getptr(_quickptr[11] + (b * _quickshort[4]) + (2 * varval()))); case 248: b = varval(); return((int)getptr(_quickptr[12] + (b * _quickshort[5]) + (2 * varval()))); case 253: b = varval(); return(bitextract((int32)_quickptr[1] + b * _quickshort[1], varval())); case 255: b = varval(); return(bitextract((int32)_quickptr[4] + b * _quickshort[3], varval())); default: error("VARVAL : Illegal code encountered"); } } void AGOSEngine_PN::writeval(uint8 *ptr, int val) { uint8 *savpt = _workptr; int lsav = _linct, a, b, x; _workptr = ptr; _linct = 255; if ((a = readfromline()) < 247) error("Write to constant"); switch (a) { case 249: error("Write to constant"); break; case 250: error("Write to constant"); break; case 251: _variableArray[varval()] = val; break; case 252: b = varval(); _dataBase[_quickptr[0] + b * _quickshort[0] + varval()] = val; break; case 254: b = varval(); _dataBase[_quickptr[3] + b * _quickshort[2] + varval()] = val; break; case 247: b = varval(); x = _quickptr[11] + b * _quickshort[4] + varval() * 2; WRITE_LE_UINT16(_dataBase + x, val); break; case 248: b = varval(); x = _quickptr[12] + b * _quickshort[5] + varval() * 2; WRITE_LE_UINT16(_dataBase + x, val); break; case 253: b = varval(); setbitf((uint32)_quickptr[1] + b * _quickshort[1], varval(), val); break; case 255: b = varval(); setbitf((uint32)_quickptr[4] + b * _quickshort[3], varval(), val); break; default: error("WRITEVAL : undefined evaluation"); } _linct = lsav; _workptr = savpt; } void AGOSEngine_PN::setbitf(uint32 ptr, int offs, int val) { ptr += offs / 8; _dataBase[ptr] &= (255 - bitvalue[offs % 8]); if (val != 0) _dataBase[ptr] |= bitvalue[offs % 8]; } int AGOSEngine_PN::actCallD(int n) { int pf[8]; funcentry(pf, n); addstack(-1); funccpy(pf); setposition(n, 0); return(doline(1)); } int AGOSEngine_PN::doaction() { if (_linct == 0) return 0; _opcode = readfromline(); if (_opcode > 63) { return (actCallD(_opcode - 64)); } setScriptReturn(0); executeOpcode(_opcode); delay(0); return getScriptReturn(); } int AGOSEngine_PN::doline(int needsave) { int x; jmp_buf *ljmpbuff = NULL; jmp_buf *mybuf; mybuf = (jmp_buf *)malloc(sizeof(jmp_buf)); if (mybuf == NULL) error("Out of memory - stack overflow"); if ((x = setjmp(*mybuf)) > 0) { dumpstack(); _cjmpbuff = ljmpbuff; free((char *)mybuf); return (x - 1); } if (x == -1) { _cjmpbuff = mybuf; goto carryon; } ljmpbuff = _cjmpbuff; _cjmpbuff = mybuf; if (needsave) _stackbase->savearea = mybuf; nln: _linct = ((*_linebase) & 127) - 1; _workptr = _linebase + 1; if (*_linebase > 127) { x = varval(); if (x != (int)_variableArray[1]) goto skipln; } carryon: while((x = doaction()) && !shouldQuit()); skipln: _linebase += 127 & *_linebase; _linembr++; if (!shouldQuit()) goto nln; return 0; } int AGOSEngine_PN::findentry() { int stepmt; int curObj = 0; uint32 ofs = _quickptr[11]; int c1, c2; c1 = varval(); c2 = varval(); stepmt = _quickshort[4]; while (curObj < _quickshort[6]) { if (((c1 == 255) || (c1 == getptr(ofs))) && (c2 == getptr(ofs + 2))) { _variableArray[23] = curObj; return 1; } curObj++; ofs += stepmt; } return 0; } int AGOSEngine_PN::findset() { int curObj = _fnst; int c1, c2, c3, c4; int stepmt = _quickshort[4]; uint32 ofs = _quickptr[11] + stepmt * curObj; c1 = varval(); c2 = varval(); c3 = varval(); c4 = varval(); while (curObj < _quickshort[6]) { if (((c1 ==255) || (c1 == getptr(ofs))) && ((c2 == 255) || (c2 == getptr(ofs + 2))) && ((c3 == 255) || (c3 == getptr(ofs + 4))) && ((c4 == 255) || (c4 == getptr(ofs + 6)))) { _variableArray[23] = curObj; _fnst = curObj + 1; return 1; } curObj++; ofs += stepmt; } return 0; } void AGOSEngine_PN::funccpy(int *store) { int a = 0; int b = 24; while (a < 8) { _variableArray[b++] = *store++; a++; } } void AGOSEngine_PN::funcentry(int *store, int procn) { int ct = 0; int nprm; nprm = _dataBase[getlong(_quickptr[6] + 3L * procn)]; while (ct < nprm) { *store++ = varval(); ct++; } } int AGOSEngine_PN::gvwrd(uint8 *wptr, int mask) { int val, code, q = _dataBase[57]; uint8 *vocbase = _dataBase + getlong(15); while (*vocbase != 255) { if (*vocbase < 128) { val = vocbase[q] + 256 * vocbase[q + 1]; code = vocbase[q + 2]; } if (wrdmatch(vocbase, mask, wptr, code)) return val; vocbase += (*vocbase > 127) ? q : q + 3; } return -1; } int AGOSEngine_PN::samewrd(uint8 *w1, uint8 *w2, int ln) { int ct = 0; while (ct < ln) { if (toupper(*w1) > toupper(*w2)) return 1; if (toupper(*w1) < toupper(*w2)) return -1; ct++; w1++; w2++; } return 0; } int AGOSEngine_PN::setposition(int process, int line) { uint8 *ourptr; int np; int ct = 0; ourptr = _dataBase + getlong(_quickptr[6] + 3L * process); np = *ourptr++; while (ct < line) { ourptr += (127 & *ourptr); ct++; } x1: _linebase = ourptr; _linct = (127 & (*ourptr)) - 1; if (*ourptr++ > 127) { ct = varval(); if (ct != (int)_variableArray[1]) { ourptr += _linct - 1; line++; goto x1; } } _linembr = line; _procnum = process; _variableArray[0] = process; _workptr = ourptr; return np; } int AGOSEngine_PN::wrdmatch(uint8 *word1, int mask1, uint8 *word2, int mask2) { uint8 sv; if ((mask1 & mask2) == 0) return 0; sv = *word1; *word1 &= 127; if (samewrd(word1, word2, _dataBase[57])) { *word1 = sv; return 0; } *word1 = sv; return 1; } // ----------------------------------------------------------------------- // Personal Nightmare Stack Code // ----------------------------------------------------------------------- void AGOSEngine_PN::addstack(int type) { struct stackframe *a; int pt, ct = 0; a = (struct stackframe *)malloc(sizeof(struct stackframe)); if (a == NULL) error("Out of memory - stack overflow"); a->nextframe = _stackbase; _stackbase = a; pt = 0; while (ct < 6) a->flag[ct++] = _variableArray[pt++]; ct = 0; pt = 24; while (ct < 8) a->param[ct++] = _variableArray[pt++]; a->classnum = type; a->ll = _linct; a->linenum = _linembr; a->linpos = _workptr; a->lbase = _linebase; a->process = _procnum; } void AGOSEngine_PN::dumpstack() { struct stackframe *a; if (_stackbase == NULL) error("Stack underflow or unknown longjmp"); a = _stackbase->nextframe; free((char *)_stackbase); _stackbase = a; } void AGOSEngine_PN::junkstack() { struct stackframe *a; if (_stackbase == NULL) error("Stack underflow or unknown longjmp"); a = _stackbase->nextframe; if (_stackbase->classnum == -1) free((char *)_stackbase->savearea); free((char *)_stackbase); _stackbase = a; } void AGOSEngine_PN::popstack(int type) { int a = 0, b; while ((_stackbase != NULL) && (_stackbase->classnum != type)) junkstack(); if (_stackbase == NULL) error("Stack underflow or unknown longjmp"); _linct = _stackbase->ll; _linebase = _stackbase->lbase; _workptr = _stackbase->linpos; _procnum = _stackbase->process; _linembr = _stackbase->linenum; b = 0; while (a < 6) _variableArray[b++] = _stackbase->flag[a++]; b = 24; a = 0; while (a < 8) _variableArray[b++] = _stackbase->param[a++]; } } // End of namespace AGOS