2 * Copyright (C) 1999-2000 Harri Porten (porten@kde.org)
3 * Copyright (C) 2004, 2005, 2006, 2007, 2008 Apple Inc. All rights reserved.
4 * Copyright (C) 2006 Bjoern Graf (bjoern.graf@gmail.com)
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Library General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Library General Public License for more details.
16 * You should have received a copy of the GNU Library General Public License
17 * along with this library; see the file COPYING.LIB. If not, write to
18 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19 * Boston, MA 02110-1301, USA.
25 #include "BytecodeGenerator.h"
26 #include "Completion.h"
27 #include "InitializeThreading.h"
30 #include "PrototypeFunction.h"
31 #include "SamplingTool.h"
42 #include <readline/history.h>
43 #include <readline/readline.h>
54 #if COMPILER(MSVC) && !PLATFORM(WIN_CE)
60 #include <QCoreApplication>
67 static void cleanupGlobalData(JSGlobalData
*);
68 static bool fillBufferWithContentsOfFile(const UString
& fileName
, Vector
<char>& buffer
);
70 static JSValuePtr
functionPrint(ExecState
*, JSObject
*, JSValuePtr
, const ArgList
&);
71 static JSValuePtr
functionDebug(ExecState
*, JSObject
*, JSValuePtr
, const ArgList
&);
72 static JSValuePtr
functionGC(ExecState
*, JSObject
*, JSValuePtr
, const ArgList
&);
73 static JSValuePtr
functionVersion(ExecState
*, JSObject
*, JSValuePtr
, const ArgList
&);
74 static JSValuePtr
functionRun(ExecState
*, JSObject
*, JSValuePtr
, const ArgList
&);
75 static JSValuePtr
functionLoad(ExecState
*, JSObject
*, JSValuePtr
, const ArgList
&);
76 static JSValuePtr
functionReadline(ExecState
*, JSObject
*, JSValuePtr
, const ArgList
&);
77 static NO_RETURN JSValuePtr
functionQuit(ExecState
*, JSObject
*, JSValuePtr
, const ArgList
&);
88 Vector
<UString
> fileNames
;
89 Vector
<UString
> arguments
;
92 static const char interactivePrompt
[] = "> ";
93 static const UString
interpreterName("Interpreter");
99 long getElapsedMS(); // call stop() first
105 #elif PLATFORM(WIN_OS)
109 // Windows does not have timeval, disabling this class for now (bug 7399)
115 void StopWatch::start()
118 QDateTime t
= QDateTime::currentDateTime();
119 m_startTime
= t
.toTime_t() * 1000 + t
.time().msec();
120 #elif PLATFORM(WIN_OS)
121 m_startTime
= timeGetTime();
123 gettimeofday(&m_startTime
, 0);
127 void StopWatch::stop()
130 QDateTime t
= QDateTime::currentDateTime();
131 m_stopTime
= t
.toTime_t() * 1000 + t
.time().msec();
132 #elif PLATFORM(WIN_OS)
133 m_stopTime
= timeGetTime();
135 gettimeofday(&m_stopTime
, 0);
139 long StopWatch::getElapsedMS()
141 #if PLATFORM(WIN_OS) || PLATFORM(QT)
142 return m_stopTime
- m_startTime
;
145 timersub(&m_stopTime
, &m_startTime
, &elapsedTime
);
147 return elapsedTime
.tv_sec
* 1000 + lroundf(elapsedTime
.tv_usec
/ 1000.0f
);
151 class GlobalObject
: public JSGlobalObject
{
153 GlobalObject(const Vector
<UString
>& arguments
);
154 virtual UString
className() const { return "global"; }
156 COMPILE_ASSERT(!IsInteger
<GlobalObject
>::value
, WTF_IsInteger_GlobalObject_false
);
157 ASSERT_CLASS_FITS_IN_CELL(GlobalObject
);
159 GlobalObject::GlobalObject(const Vector
<UString
>& arguments
)
162 putDirectFunction(globalExec(), new (globalExec()) PrototypeFunction(globalExec(), prototypeFunctionStructure(), 1, Identifier(globalExec(), "debug"), functionDebug
));
163 putDirectFunction(globalExec(), new (globalExec()) PrototypeFunction(globalExec(), prototypeFunctionStructure(), 1, Identifier(globalExec(), "print"), functionPrint
));
164 putDirectFunction(globalExec(), new (globalExec()) PrototypeFunction(globalExec(), prototypeFunctionStructure(), 0, Identifier(globalExec(), "quit"), functionQuit
));
165 putDirectFunction(globalExec(), new (globalExec()) PrototypeFunction(globalExec(), prototypeFunctionStructure(), 0, Identifier(globalExec(), "gc"), functionGC
));
166 putDirectFunction(globalExec(), new (globalExec()) PrototypeFunction(globalExec(), prototypeFunctionStructure(), 1, Identifier(globalExec(), "version"), functionVersion
));
167 putDirectFunction(globalExec(), new (globalExec()) PrototypeFunction(globalExec(), prototypeFunctionStructure(), 1, Identifier(globalExec(), "run"), functionRun
));
168 putDirectFunction(globalExec(), new (globalExec()) PrototypeFunction(globalExec(), prototypeFunctionStructure(), 1, Identifier(globalExec(), "load"), functionLoad
));
169 putDirectFunction(globalExec(), new (globalExec()) PrototypeFunction(globalExec(), prototypeFunctionStructure(), 0, Identifier(globalExec(), "readline"), functionReadline
));
171 JSObject
* array
= constructEmptyArray(globalExec());
172 for (size_t i
= 0; i
< arguments
.size(); ++i
)
173 array
->put(globalExec(), i
, jsString(globalExec(), arguments
[i
]));
174 putDirect(Identifier(globalExec(), "arguments"), array
);
177 JSValuePtr
functionPrint(ExecState
* exec
, JSObject
*, JSValuePtr
, const ArgList
& args
)
179 for (unsigned i
= 0; i
< args
.size(); ++i
) {
183 printf("%s", args
.at(exec
, i
).toString(exec
).UTF8String().c_str());
188 return jsUndefined();
191 JSValuePtr
functionDebug(ExecState
* exec
, JSObject
*, JSValuePtr
, const ArgList
& args
)
193 fprintf(stderr
, "--> %s\n", args
.at(exec
, 0).toString(exec
).UTF8String().c_str());
194 return jsUndefined();
197 JSValuePtr
functionGC(ExecState
* exec
, JSObject
*, JSValuePtr
, const ArgList
&)
200 exec
->heap()->collect();
201 return jsUndefined();
204 JSValuePtr
functionVersion(ExecState
*, JSObject
*, JSValuePtr
, const ArgList
&)
206 // We need this function for compatibility with the Mozilla JS tests but for now
207 // we don't actually do any version-specific handling
208 return jsUndefined();
211 JSValuePtr
functionRun(ExecState
* exec
, JSObject
*, JSValuePtr
, const ArgList
& args
)
214 UString fileName
= args
.at(exec
, 0).toString(exec
);
216 if (!fillBufferWithContentsOfFile(fileName
, script
))
217 return throwError(exec
, GeneralError
, "Could not open file.");
219 JSGlobalObject
* globalObject
= exec
->lexicalGlobalObject();
222 evaluate(globalObject
->globalExec(), globalObject
->globalScopeChain(), makeSource(script
.data(), fileName
));
225 return jsNumber(globalObject
->globalExec(), stopWatch
.getElapsedMS());
228 JSValuePtr
functionLoad(ExecState
* exec
, JSObject
*, JSValuePtr
, const ArgList
& args
)
230 UString fileName
= args
.at(exec
, 0).toString(exec
);
232 if (!fillBufferWithContentsOfFile(fileName
, script
))
233 return throwError(exec
, GeneralError
, "Could not open file.");
235 JSGlobalObject
* globalObject
= exec
->lexicalGlobalObject();
236 evaluate(globalObject
->globalExec(), globalObject
->globalScopeChain(), makeSource(script
.data(), fileName
));
238 return jsUndefined();
241 JSValuePtr
functionReadline(ExecState
* exec
, JSObject
*, JSValuePtr
, const ArgList
&)
243 Vector
<char, 256> line
;
245 while ((c
= getchar()) != EOF
) {
246 // FIXME: Should we also break on \r?
252 return jsString(exec
, line
.data());
255 JSValuePtr
functionQuit(ExecState
* exec
, JSObject
*, JSValuePtr
, const ArgList
&)
257 cleanupGlobalData(&exec
->globalData());
261 // Use SEH for Release builds only to get rid of the crash report dialog
262 // (luckily the same tests fail in Release and Debug builds so far). Need to
263 // be in a separate main function because the jscmain function requires object
266 #if COMPILER(MSVC) && !defined(_DEBUG)
268 #define EXCEPT(x) } __except (EXCEPTION_EXECUTE_HANDLER) { x; }
274 int jscmain(int argc
, char** argv
, JSGlobalData
*);
276 int main(int argc
, char** argv
)
278 #if defined(_DEBUG) && PLATFORM(WIN_OS)
279 _CrtSetReportFile(_CRT_WARN
, _CRTDBG_FILE_STDERR
);
280 _CrtSetReportMode(_CRT_WARN
, _CRTDBG_MODE_FILE
);
281 _CrtSetReportFile(_CRT_ERROR
, _CRTDBG_FILE_STDERR
);
282 _CrtSetReportMode(_CRT_ERROR
, _CRTDBG_MODE_FILE
);
283 _CrtSetReportFile(_CRT_ASSERT
, _CRTDBG_FILE_STDERR
);
284 _CrtSetReportMode(_CRT_ASSERT
, _CRTDBG_MODE_FILE
);
288 QCoreApplication
app(argc
, argv
);
291 // Initialize JSC before getting JSGlobalData.
292 JSC::initializeThreading();
294 // We can't use destructors in the following code because it uses Windows
295 // Structured Exception Handling
297 JSGlobalData
* globalData
= JSGlobalData::create().releaseRef();
299 res
= jscmain(argc
, argv
, globalData
);
302 cleanupGlobalData(globalData
);
306 static void cleanupGlobalData(JSGlobalData
* globalData
)
309 globalData
->heap
.destroy();
313 static bool runWithScripts(GlobalObject
* globalObject
, const Vector
<UString
>& fileNames
, bool dump
)
318 BytecodeGenerator::setDumpsGeneratedCode(true);
320 #if ENABLE(OPCODE_SAMPLING)
321 Interpreter
* interpreter
= globalObject
->globalData()->interpreter
;
322 interpreter
->setSampler(new SamplingTool(interpreter
));
326 for (size_t i
= 0; i
< fileNames
.size(); i
++) {
327 UString fileName
= fileNames
[i
];
329 if (!fillBufferWithContentsOfFile(fileName
, script
))
330 return false; // fail early so we can catch missing files
332 #if ENABLE(OPCODE_SAMPLING)
333 interpreter
->sampler()->start();
335 Completion completion
= evaluate(globalObject
->globalExec(), globalObject
->globalScopeChain(), makeSource(script
.data(), fileName
));
336 success
= success
&& completion
.complType() != Throw
;
338 if (completion
.complType() == Throw
)
339 printf("Exception: %s\n", completion
.value().toString(globalObject
->globalExec()).ascii());
341 printf("End: %s\n", completion
.value().toString(globalObject
->globalExec()).ascii());
344 globalObject
->globalExec()->clearException();
346 #if ENABLE(OPCODE_SAMPLING)
347 interpreter
->sampler()->stop();
351 #if ENABLE(OPCODE_SAMPLING)
352 interpreter
->sampler()->dump(globalObject
->globalExec());
353 delete interpreter
->sampler();
358 static void runInteractive(GlobalObject
* globalObject
)
362 char* line
= readline(interactivePrompt
);
367 Completion completion
= evaluate(globalObject
->globalExec(), globalObject
->globalScopeChain(), makeSource(line
, interpreterName
));
370 puts(interactivePrompt
);
371 Vector
<char, 256> line
;
373 while ((c
= getchar()) != EOF
) {
374 // FIXME: Should we also break on \r?
380 Completion completion
= evaluate(globalObject
->globalExec(), globalObject
->globalScopeChain(), makeSource(line
.data(), interpreterName
));
382 if (completion
.complType() == Throw
)
383 printf("Exception: %s\n", completion
.value().toString(globalObject
->globalExec()).ascii());
385 printf("%s\n", completion
.value().toString(globalObject
->globalExec()).UTF8String().c_str());
387 globalObject
->globalExec()->clearException();
392 static NO_RETURN
void printUsageStatement()
394 fprintf(stderr
, "Usage: jsc [options] [files] [-- arguments]\n");
395 fprintf(stderr
, " -d Dumps bytecode (debug builds only)\n");
396 fprintf(stderr
, " -f Specifies a source file (deprecated)\n");
397 fprintf(stderr
, " -h|--help Prints this help message\n");
398 fprintf(stderr
, " -i Enables interactive mode (default if no files are specified)\n");
399 fprintf(stderr
, " -s Installs signal handlers that exit on a crash (Unix platforms only)\n");
403 static void parseArguments(int argc
, char** argv
, Options
& options
)
406 for (; i
< argc
; ++i
) {
407 const char* arg
= argv
[i
];
408 if (strcmp(arg
, "-f") == 0) {
410 printUsageStatement();
411 options
.fileNames
.append(argv
[i
]);
414 if (strcmp(arg
, "-h") == 0 || strcmp(arg
, "--help") == 0) {
415 printUsageStatement();
417 if (strcmp(arg
, "-i") == 0) {
418 options
.interactive
= true;
421 if (strcmp(arg
, "-d") == 0) {
425 if (strcmp(arg
, "-s") == 0) {
427 signal(SIGILL
, _exit
);
428 signal(SIGFPE
, _exit
);
429 signal(SIGBUS
, _exit
);
430 signal(SIGSEGV
, _exit
);
434 if (strcmp(arg
, "--") == 0) {
438 options
.fileNames
.append(argv
[i
]);
441 if (options
.fileNames
.isEmpty())
442 options
.interactive
= true;
444 for (; i
< argc
; ++i
)
445 options
.arguments
.append(argv
[i
]);
448 int jscmain(int argc
, char** argv
, JSGlobalData
* globalData
)
453 parseArguments(argc
, argv
, options
);
455 GlobalObject
* globalObject
= new (globalData
) GlobalObject(options
.arguments
);
456 bool success
= runWithScripts(globalObject
, options
.fileNames
, options
.dump
);
457 if (options
.interactive
&& success
)
458 runInteractive(globalObject
);
460 return success
? 0 : 3;
463 static bool fillBufferWithContentsOfFile(const UString
& fileName
, Vector
<char>& buffer
)
465 FILE* f
= fopen(fileName
.UTF8String().c_str(), "r");
467 fprintf(stderr
, "Could not open file: %s\n", fileName
.UTF8String().c_str());
471 size_t buffer_size
= 0;
472 size_t buffer_capacity
= 1024;
474 buffer
.resize(buffer_capacity
);
476 while (!feof(f
) && !ferror(f
)) {
477 buffer_size
+= fread(buffer
.data() + buffer_size
, 1, buffer_capacity
- buffer_size
, f
);
478 if (buffer_size
== buffer_capacity
) { // guarantees space for trailing '\0'
479 buffer_capacity
*= 2;
480 buffer
.resize(buffer_capacity
);
484 buffer
[buffer_size
] = '\0';