pjs/js2/tests/cpp/js2_shell.cpp

263 строки
7.4 KiB
C++
Исходник Обычный вид История

2000-01-11 00:22:43 +03:00
// -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
//
// The contents of this file are subject to the Netscape Public
// License Version 1.1 (the "License"); you may not use this file
// except in compliance with the License. You may obtain a copy of
// the License at http://www.mozilla.org/NPL/
//
// Software distributed under the License is distributed on an "AS
// IS" basis, WITHOUT WARRANTY OF ANY KIND, either express oqr
// implied. See the License for the specific language governing
// rights and limitations under the License.
//
// The Original Code is the JavaScript 2 Prototype.
//
// The Initial Developer of the Original Code is Netscape
// Communications Corporation. Portions created by Netscape are
// Copyright (C) 1998 Netscape Communications Corporation. All
// Rights Reserved.
//
// JS2 shell.
//
#include <assert.h>
2000-01-11 00:22:43 +03:00
#include "world.h"
#include "interpreter.h"
#include "icodegenerator.h"
#include "debugger.h"
2000-04-07 03:47:33 +04:00
#if defined(XP_MAC) && !defined(XP_MAC_MPW)
2000-01-11 00:22:43 +03:00
#include <SIOUX.h>
#include <MacTypes.h>
static char *mac_argv[] = {"js2", 0};
static void initConsole(StringPtr consoleName,
const char* startupMessage,
int &argc, char **&argv)
{
SIOUXSettings.autocloseonquit = false;
SIOUXSettings.asktosaveonclose = false;
SIOUXSetTitle(consoleName);
// Set up a buffer for stderr (otherwise it's a pig).
static char buffer[BUFSIZ];
setvbuf(stderr, buffer, _IOLBF, BUFSIZ);
JavaScript::stdOut << startupMessage;
argc = 1;
argv = mac_argv;
}
2000-04-07 03:47:33 +04:00
#endif
2000-01-11 00:22:43 +03:00
namespace JavaScript {
namespace Shell {
using namespace ICG;
using namespace JSTypes;
using namespace Interpreter;
// Interactively read a line from the input stream in and put it into
// s. Return false if reached the end of input before reading anything.
2000-05-12 09:15:52 +04:00
static bool promptLine(LineReader &inReader, string &s, const char *prompt)
{
if (prompt) {
stdOut << prompt;
2000-05-12 09:15:52 +04:00
#ifdef XP_MAC_MPW
// Print a CR after the prompt because MPW grabs the entire
// line when entering an interactive command.
stdOut << '\n';
2000-05-12 09:15:52 +04:00
#endif
}
return inReader.readLine(s) != 0;
}
JavaScript::World world;
2000-05-05 08:45:39 +04:00
JavaScript::Debugger::Shell jsd(world, stdin, JavaScript::stdOut,
JavaScript::stdOut);
2000-05-09 23:01:00 +04:00
const bool showTokens = false;
2000-05-09 03:12:10 +04:00
2000-05-09 23:01:00 +04:00
static JSValue print(const JSValues &argv)
{
size_t n = argv.size();
if (n > 0) {
stdOut << argv[0];
for (size_t i = 1; i < n; ++i)
stdOut << ' ' << argv[i];
}
stdOut << "\n";
return kUndefinedValue;
}
static void genCode(World &world, Context &cx, StmtNode *p)
2000-05-09 03:12:10 +04:00
{
JSScope glob;
ICodeGenerator icg(&world, &glob);
icg.isScript();
TypedRegister ret(NotARegister, &None_Type);
while (p) {
ret = icg.genStmt(p);
p = p->next;
}
2000-05-27 02:34:42 +04:00
icg.returnStmt(ret);
2000-06-16 05:37:47 +04:00
// stdOut << '\n';
// stdOut << icg;
2000-05-09 03:12:10 +04:00
JSValue result = cx.interpret(icg.complete(), JSValues());
stdOut << "result = " << result << "\n";
2000-05-09 03:12:10 +04:00
}
static void readEvalPrint(FILE *in, World &world)
{
2000-05-18 04:01:33 +04:00
JSScope glob;
Context cx(world, &glob);
StringAtom& printName = world.identifiers[widenCString("print")];
glob.defineNativeFunction(printName, print);
String buffer;
string line;
String sourceLocation = widenCString("console");
LineReader inReader(in);
2000-05-12 09:15:52 +04:00
while (promptLine(inReader, line, buffer.empty() ? "js> " : "> ")) {
appendChars(buffer, line.data(), line.size());
try {
Arena a;
Parser p(world, a, buffer, sourceLocation);
if (showTokens) {
Lexer &l = p.lexer;
while (true) {
const Token &t = l.get(true);
if (t.hasKind(Token::end))
break;
stdOut << ' ';
t.print(stdOut, true);
}
stdOut << '\n';
} else {
StmtNode *parsedStatements = p.parseProgram();
ASSERT(p.lexer.peek(true).hasKind(Token::end));
{
2000-06-01 07:31:17 +04:00
PrettyPrinter f(stdOut, 30);
{
PrettyPrinter::Block b(f, 2);
f << "Program =";
f.linearBreak(1);
StmtNode::printStatements(f, parsedStatements);
}
f.end();
}
stdOut << '\n';
#if 0
// Generate code for parsedStatements, which is a linked list of zero or more statements
genCode(world, cx, parsedStatements);
2000-05-18 04:01:33 +04:00
#endif
}
clear(buffer);
} catch (Exception &e) {
/* If we got a syntax error on the end of input,
* then wait for a continuation
* of input rather than printing the error message. */
if (!(e.hasKind(Exception::syntaxError) &&
e.lineNum && e.pos == buffer.size() &&
e.sourceFile == sourceLocation)) {
stdOut << '\n' << e.fullMessage();
clear(buffer);
2000-01-11 00:22:43 +03:00
}
}
}
stdOut << '\n';
}
2000-04-28 04:11:18 +04:00
2000-04-28 17:31:39 +04:00
/**
* Poor man's instruction tracing facility.
*/
class Tracer : public Context::Listener {
typedef InstructionStream::difference_type InstructionOffset;
2000-05-12 05:20:34 +04:00
void listen(Context* context, Context::Event event)
2000-04-28 17:31:39 +04:00
{
2000-05-12 05:20:34 +04:00
if (event & Context::EV_STEP) {
ICodeModule *iCode = context->getICode();
JSValues &registers = context->getRegisters();
InstructionIterator pc = context->getPC();
InstructionOffset offset = (pc - iCode->its_iCode->begin());
printFormat(stdOut, "trace [%02u:%04u]: ",
iCode->mID, offset);
2000-05-12 05:20:34 +04:00
Instruction* i = *pc;
stdOut << *i;
if (i->op() != BRANCH && i->count() > 0) {
stdOut << " [";
i->printOperands(stdOut, registers);
stdOut << "]\n";
} else {
stdOut << '\n';
}
2000-04-28 17:31:39 +04:00
}
}
};
2000-05-09 03:12:10 +04:00
2000-06-16 05:37:47 +04:00
char * tests[] = {
"function fact(n) { if (n > 1) return n * fact(n-1); else return 1; } print(fact(6), \" should be 720\"); return;" ,
"a = { f1: 1, f2: 2}; print(a.f2++, \" should be 2\"); print(a.f2 <<= 1, \" should be 6\"); return;"
};
void testCompile()
{
JSScope glob;
Context cx(world, &glob);
StringAtom& printName = world.identifiers[widenCString("print")];
glob.defineNativeFunction(printName, print);
for (int i = 0; i < sizeof(tests) / sizeof(char *); i++) {
String testScript = widenCString(tests[i]);
Arena a;
Parser p(world, a, testScript, widenCString("testCompile"));
StmtNode *parsedStatements = p.parseProgram();
JSScope glob;
ICodeGenerator icg(&world, &glob);
2000-06-16 05:37:47 +04:00
icg.isScript();
while (parsedStatements) {
icg.genStmt(parsedStatements);
parsedStatements = parsedStatements->next;
}
cx.interpret(icg.complete(), JSValues());
}
}
2000-04-28 05:43:03 +04:00
} /* namespace Shell */
} /* namespace JavaScript */
2000-06-16 05:37:47 +04:00
int main(int argc, char **argv)
2000-01-11 00:22:43 +03:00
{
#if defined(XP_MAC) && !defined(XP_MAC_MPW)
2000-01-11 00:22:43 +03:00
initConsole("\pJavaScript Shell", "Welcome to the js2 shell.\n", argc, argv);
#endif
2000-04-29 04:29:56 +04:00
using namespace JavaScript;
using namespace Shell;
2000-05-12 09:15:52 +04:00
#if 0
2000-06-16 05:37:47 +04:00
testCompile();
2000-05-12 09:15:52 +04:00
#endif
readEvalPrint(stdin, world);
2000-04-05 01:38:25 +04:00
return 0;
// return ProcessArgs(argv + 1, argc - 1);
2000-01-11 00:22:43 +03:00
}