blob: e4658b1cee05bd3c8e695848132801cd53e689bb [file] [log] [blame]
Steve Blocka7e24c12009-10-30 11:49:00 +00001// Copyright 2009 the V8 project authors. All rights reserved.
2// Redistribution and use in source and binary forms, with or without
3// modification, are permitted provided that the following conditions are
4// met:
5//
6// * Redistributions of source code must retain the above copyright
7// notice, this list of conditions and the following disclaimer.
8// * Redistributions in binary form must reproduce the above
9// copyright notice, this list of conditions and the following
10// disclaimer in the documentation and/or other materials provided
11// with the distribution.
12// * Neither the name of Google Inc. nor the names of its
13// contributors may be used to endorse or promote products derived
14// from this software without specific prior written permission.
15//
16// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
28
29#include <stdlib.h>
30#include <errno.h>
31
32#include "d8.h"
33#include "d8-debug.h"
34#include "debug.h"
35#include "api.h"
36#include "natives.h"
37#include "platform.h"
38
39
40namespace v8 {
41
42
43const char* Shell::kHistoryFileName = ".d8_history";
44const char* Shell::kPrompt = "d8> ";
45
46
47LineEditor *LineEditor::first_ = NULL;
48
49
50LineEditor::LineEditor(Type type, const char* name)
51 : type_(type),
52 name_(name),
53 next_(first_) {
54 first_ = this;
55}
56
57
58LineEditor* LineEditor::Get() {
59 LineEditor* current = first_;
60 LineEditor* best = current;
61 while (current != NULL) {
62 if (current->type_ > best->type_)
63 best = current;
64 current = current->next_;
65 }
66 return best;
67}
68
69
70class DumbLineEditor: public LineEditor {
71 public:
72 DumbLineEditor() : LineEditor(LineEditor::DUMB, "dumb") { }
73 virtual i::SmartPointer<char> Prompt(const char* prompt);
74};
75
76
77static DumbLineEditor dumb_line_editor;
78
79
80i::SmartPointer<char> DumbLineEditor::Prompt(const char* prompt) {
81 static const int kBufferSize = 256;
82 char buffer[kBufferSize];
83 printf("%s", prompt);
84 char* str = fgets(buffer, kBufferSize, stdin);
85 return i::SmartPointer<char>(str ? i::StrDup(str) : str);
86}
87
88
89CounterMap* Shell::counter_map_;
90i::OS::MemoryMappedFile* Shell::counters_file_ = NULL;
91CounterCollection Shell::local_counters_;
92CounterCollection* Shell::counters_ = &local_counters_;
93Persistent<Context> Shell::utility_context_;
94Persistent<Context> Shell::evaluation_context_;
95
96
97bool CounterMap::Match(void* key1, void* key2) {
98 const char* name1 = reinterpret_cast<const char*>(key1);
99 const char* name2 = reinterpret_cast<const char*>(key2);
100 return strcmp(name1, name2) == 0;
101}
102
103
104// Converts a V8 value to a C string.
105const char* ToCString(const v8::String::Utf8Value& value) {
106 return *value ? *value : "<string conversion failed>";
107}
108
109
110// Executes a string within the current v8 context.
111bool Shell::ExecuteString(Handle<String> source,
112 Handle<Value> name,
113 bool print_result,
114 bool report_exceptions) {
115 HandleScope handle_scope;
116 TryCatch try_catch;
117 if (i::FLAG_debugger) {
118 // When debugging make exceptions appear to be uncaught.
119 try_catch.SetVerbose(true);
120 }
121 Handle<Script> script = Script::Compile(source, name);
122 if (script.IsEmpty()) {
123 // Print errors that happened during compilation.
124 if (report_exceptions && !i::FLAG_debugger)
125 ReportException(&try_catch);
126 return false;
127 } else {
128 Handle<Value> result = script->Run();
129 if (result.IsEmpty()) {
130 // Print errors that happened during execution.
131 if (report_exceptions && !i::FLAG_debugger)
132 ReportException(&try_catch);
133 return false;
134 } else {
135 if (print_result && !result->IsUndefined()) {
136 // If all went well and the result wasn't undefined then print
137 // the returned value.
138 v8::String::Utf8Value str(result);
139 const char* cstr = ToCString(str);
140 printf("%s\n", cstr);
141 }
142 return true;
143 }
144 }
145}
146
147
148Handle<Value> Shell::Print(const Arguments& args) {
149 Handle<Value> val = Write(args);
150 printf("\n");
151 return val;
152}
153
154
155Handle<Value> Shell::Write(const Arguments& args) {
156 for (int i = 0; i < args.Length(); i++) {
157 HandleScope handle_scope;
158 if (i != 0) {
159 printf(" ");
160 }
161 v8::String::Utf8Value str(args[i]);
162 fwrite(*str, sizeof(**str), str.length(), stdout);
163 }
164 return Undefined();
165}
166
167
168Handle<Value> Shell::Read(const Arguments& args) {
169 String::Utf8Value file(args[0]);
170 if (*file == NULL) {
171 return ThrowException(String::New("Error loading file"));
172 }
173 Handle<String> source = ReadFile(*file);
174 if (source.IsEmpty()) {
175 return ThrowException(String::New("Error loading file"));
176 }
177 return source;
178}
179
180
181Handle<Value> Shell::ReadLine(const Arguments& args) {
182 i::SmartPointer<char> line(i::ReadLine(""));
183 if (*line == NULL) {
184 return Null();
185 }
186 size_t len = strlen(*line);
187 if (len > 0 && line[len - 1] == '\n') {
188 --len;
189 }
190 return String::New(*line, len);
191}
192
193
194Handle<Value> Shell::Load(const Arguments& args) {
195 for (int i = 0; i < args.Length(); i++) {
196 HandleScope handle_scope;
197 String::Utf8Value file(args[i]);
198 if (*file == NULL) {
199 return ThrowException(String::New("Error loading file"));
200 }
201 Handle<String> source = ReadFile(*file);
202 if (source.IsEmpty()) {
203 return ThrowException(String::New("Error loading file"));
204 }
205 if (!ExecuteString(source, String::New(*file), false, false)) {
206 return ThrowException(String::New("Error executing file"));
207 }
208 }
209 return Undefined();
210}
211
212
213Handle<Value> Shell::Yield(const Arguments& args) {
214 v8::Unlocker unlocker;
215 return Undefined();
216}
217
218
219Handle<Value> Shell::Quit(const Arguments& args) {
220 int exit_code = args[0]->Int32Value();
221 OnExit();
222 exit(exit_code);
223 return Undefined();
224}
225
226
227Handle<Value> Shell::Version(const Arguments& args) {
228 return String::New(V8::GetVersion());
229}
230
231
232void Shell::ReportException(v8::TryCatch* try_catch) {
233 HandleScope handle_scope;
234 v8::String::Utf8Value exception(try_catch->Exception());
235 const char* exception_string = ToCString(exception);
236 Handle<Message> message = try_catch->Message();
237 if (message.IsEmpty()) {
238 // V8 didn't provide any extra information about this error; just
239 // print the exception.
240 printf("%s\n", exception_string);
241 } else {
242 // Print (filename):(line number): (message).
243 v8::String::Utf8Value filename(message->GetScriptResourceName());
244 const char* filename_string = ToCString(filename);
245 int linenum = message->GetLineNumber();
246 printf("%s:%i: %s\n", filename_string, linenum, exception_string);
247 // Print line of source code.
248 v8::String::Utf8Value sourceline(message->GetSourceLine());
249 const char* sourceline_string = ToCString(sourceline);
250 printf("%s\n", sourceline_string);
251 // Print wavy underline (GetUnderline is deprecated).
252 int start = message->GetStartColumn();
253 for (int i = 0; i < start; i++) {
254 printf(" ");
255 }
256 int end = message->GetEndColumn();
257 for (int i = start; i < end; i++) {
258 printf("^");
259 }
260 printf("\n");
261 }
262}
263
264
265Handle<Array> Shell::GetCompletions(Handle<String> text, Handle<String> full) {
266 HandleScope handle_scope;
267 Context::Scope context_scope(utility_context_);
268 Handle<Object> global = utility_context_->Global();
269 Handle<Value> fun = global->Get(String::New("GetCompletions"));
270 static const int kArgc = 3;
271 Handle<Value> argv[kArgc] = { evaluation_context_->Global(), text, full };
272 Handle<Value> val = Handle<Function>::Cast(fun)->Call(global, kArgc, argv);
273 return handle_scope.Close(Handle<Array>::Cast(val));
274}
275
276
277#ifdef ENABLE_DEBUGGER_SUPPORT
278Handle<Object> Shell::DebugMessageDetails(Handle<String> message) {
279 Context::Scope context_scope(utility_context_);
280 Handle<Object> global = utility_context_->Global();
281 Handle<Value> fun = global->Get(String::New("DebugMessageDetails"));
282 static const int kArgc = 1;
283 Handle<Value> argv[kArgc] = { message };
284 Handle<Value> val = Handle<Function>::Cast(fun)->Call(global, kArgc, argv);
285 return Handle<Object>::Cast(val);
286}
287
288
289Handle<Value> Shell::DebugCommandToJSONRequest(Handle<String> command) {
290 Context::Scope context_scope(utility_context_);
291 Handle<Object> global = utility_context_->Global();
292 Handle<Value> fun = global->Get(String::New("DebugCommandToJSONRequest"));
293 static const int kArgc = 1;
294 Handle<Value> argv[kArgc] = { command };
295 Handle<Value> val = Handle<Function>::Cast(fun)->Call(global, kArgc, argv);
296 return val;
297}
298#endif
299
300
301int32_t* Counter::Bind(const char* name, bool is_histogram) {
302 int i;
303 for (i = 0; i < kMaxNameSize - 1 && name[i]; i++)
304 name_[i] = static_cast<char>(name[i]);
305 name_[i] = '\0';
306 is_histogram_ = is_histogram;
307 return ptr();
308}
309
310
311void Counter::AddSample(int32_t sample) {
312 count_++;
313 sample_total_ += sample;
314}
315
316
317CounterCollection::CounterCollection() {
318 magic_number_ = 0xDEADFACE;
319 max_counters_ = kMaxCounters;
320 max_name_size_ = Counter::kMaxNameSize;
321 counters_in_use_ = 0;
322}
323
324
325Counter* CounterCollection::GetNextCounter() {
326 if (counters_in_use_ == kMaxCounters) return NULL;
327 return &counters_[counters_in_use_++];
328}
329
330
331void Shell::MapCounters(const char* name) {
332 counters_file_ = i::OS::MemoryMappedFile::create(name,
333 sizeof(CounterCollection), &local_counters_);
334 void* memory = (counters_file_ == NULL) ?
335 NULL : counters_file_->memory();
336 if (memory == NULL) {
337 printf("Could not map counters file %s\n", name);
338 exit(1);
339 }
340 counters_ = static_cast<CounterCollection*>(memory);
341 V8::SetCounterFunction(LookupCounter);
342 V8::SetCreateHistogramFunction(CreateHistogram);
343 V8::SetAddHistogramSampleFunction(AddHistogramSample);
344}
345
346
347int CounterMap::Hash(const char* name) {
348 int h = 0;
349 int c;
350 while ((c = *name++) != 0) {
351 h += h << 5;
352 h += c;
353 }
354 return h;
355}
356
357
358Counter* Shell::GetCounter(const char* name, bool is_histogram) {
359 Counter* counter = counter_map_->Lookup(name);
360
361 if (counter == NULL) {
362 counter = counters_->GetNextCounter();
363 if (counter != NULL) {
364 counter_map_->Set(name, counter);
365 counter->Bind(name, is_histogram);
366 }
367 } else {
368 ASSERT(counter->is_histogram() == is_histogram);
369 }
370 return counter;
371}
372
373
374int* Shell::LookupCounter(const char* name) {
375 Counter* counter = GetCounter(name, false);
376
377 if (counter != NULL) {
378 return counter->ptr();
379 } else {
380 return NULL;
381 }
382}
383
384
385void* Shell::CreateHistogram(const char* name,
386 int min,
387 int max,
388 size_t buckets) {
389 return GetCounter(name, true);
390}
391
392
393void Shell::AddHistogramSample(void* histogram, int sample) {
394 Counter* counter = reinterpret_cast<Counter*>(histogram);
395 counter->AddSample(sample);
396}
397
398
399void Shell::Initialize() {
400 Shell::counter_map_ = new CounterMap();
401 // Set up counters
402 if (i::FLAG_map_counters != NULL)
403 MapCounters(i::FLAG_map_counters);
404 if (i::FLAG_dump_counters) {
405 V8::SetCounterFunction(LookupCounter);
406 V8::SetCreateHistogramFunction(CreateHistogram);
407 V8::SetAddHistogramSampleFunction(AddHistogramSample);
408 }
409
410 // Initialize the global objects
411 HandleScope scope;
412 Handle<ObjectTemplate> global_template = ObjectTemplate::New();
413 global_template->Set(String::New("print"), FunctionTemplate::New(Print));
414 global_template->Set(String::New("write"), FunctionTemplate::New(Write));
415 global_template->Set(String::New("read"), FunctionTemplate::New(Read));
416 global_template->Set(String::New("readline"),
417 FunctionTemplate::New(ReadLine));
418 global_template->Set(String::New("load"), FunctionTemplate::New(Load));
419 global_template->Set(String::New("quit"), FunctionTemplate::New(Quit));
420 global_template->Set(String::New("version"), FunctionTemplate::New(Version));
421
422 Handle<ObjectTemplate> os_templ = ObjectTemplate::New();
423 AddOSMethods(os_templ);
424 global_template->Set(String::New("os"), os_templ);
425
426 utility_context_ = Context::New(NULL, global_template);
427 utility_context_->SetSecurityToken(Undefined());
428 Context::Scope utility_scope(utility_context_);
429
430 i::JSArguments js_args = i::FLAG_js_arguments;
431 i::Handle<i::FixedArray> arguments_array =
432 i::Factory::NewFixedArray(js_args.argc());
433 for (int j = 0; j < js_args.argc(); j++) {
434 i::Handle<i::String> arg =
435 i::Factory::NewStringFromUtf8(i::CStrVector(js_args[j]));
436 arguments_array->set(j, *arg);
437 }
438 i::Handle<i::JSArray> arguments_jsarray =
439 i::Factory::NewJSArrayWithElements(arguments_array);
440 global_template->Set(String::New("arguments"),
441 Utils::ToLocal(arguments_jsarray));
442
443#ifdef ENABLE_DEBUGGER_SUPPORT
444 // Install the debugger object in the utility scope
445 i::Debug::Load();
446 i::JSObject* debug = i::Debug::debug_context()->global();
447 utility_context_->Global()->Set(String::New("$debug"),
448 Utils::ToLocal(&debug));
449#endif
450
451 // Run the d8 shell utility script in the utility context
452 int source_index = i::NativesCollection<i::D8>::GetIndex("d8");
453 i::Vector<const char> shell_source
454 = i::NativesCollection<i::D8>::GetScriptSource(source_index);
455 i::Vector<const char> shell_source_name
456 = i::NativesCollection<i::D8>::GetScriptName(source_index);
457 Handle<String> source = String::New(shell_source.start(),
458 shell_source.length());
459 Handle<String> name = String::New(shell_source_name.start(),
460 shell_source_name.length());
461 Handle<Script> script = Script::Compile(source, name);
462 script->Run();
463
464 // Mark the d8 shell script as native to avoid it showing up as normal source
465 // in the debugger.
466 i::Handle<i::JSFunction> script_fun = Utils::OpenHandle(*script);
467 i::Handle<i::Script> script_object =
468 i::Handle<i::Script>(i::Script::cast(script_fun->shared()->script()));
469 script_object->set_type(i::Smi::FromInt(i::Script::TYPE_NATIVE));
470
471 // Create the evaluation context
472 evaluation_context_ = Context::New(NULL, global_template);
473 evaluation_context_->SetSecurityToken(Undefined());
474
475#ifdef ENABLE_DEBUGGER_SUPPORT
476 // Set the security token of the debug context to allow access.
477 i::Debug::debug_context()->set_security_token(i::Heap::undefined_value());
478
479 // Start the debugger agent if requested.
480 if (i::FLAG_debugger_agent) {
481 v8::Debug::EnableAgent("d8 shell", i::FLAG_debugger_port);
482 }
483
484 // Start the in-process debugger if requested.
485 if (i::FLAG_debugger && !i::FLAG_debugger_agent) {
486 v8::Debug::SetDebugEventListener(HandleDebugEvent);
487 }
488#endif
489}
490
491
492void Shell::OnExit() {
493 if (i::FLAG_dump_counters) {
494 ::printf("+----------------------------------------+-------------+\n");
495 ::printf("| Name | Value |\n");
496 ::printf("+----------------------------------------+-------------+\n");
497 for (CounterMap::Iterator i(counter_map_); i.More(); i.Next()) {
498 Counter* counter = i.CurrentValue();
499 if (counter->is_histogram()) {
500 ::printf("| c:%-36s | %11i |\n", i.CurrentKey(), counter->count());
501 ::printf("| t:%-36s | %11i |\n",
502 i.CurrentKey(),
503 counter->sample_total());
504 } else {
505 ::printf("| %-38s | %11i |\n", i.CurrentKey(), counter->count());
506 }
507 }
508 ::printf("+----------------------------------------+-------------+\n");
509 }
510 if (counters_file_ != NULL)
511 delete counters_file_;
512}
513
514
515static char* ReadChars(const char* name, int* size_out) {
516 v8::Unlocker unlocker; // Release the V8 lock while reading files.
517 FILE* file = i::OS::FOpen(name, "rb");
518 if (file == NULL) return NULL;
519
520 fseek(file, 0, SEEK_END);
521 int size = ftell(file);
522 rewind(file);
523
524 char* chars = new char[size + 1];
525 chars[size] = '\0';
526 for (int i = 0; i < size;) {
527 int read = fread(&chars[i], 1, size - i, file);
528 i += read;
529 }
530 fclose(file);
531 *size_out = size;
532 return chars;
533}
534
535
536static char* ReadToken(char* data, char token) {
537 char* next = i::OS::StrChr(data, token);
538 if (next != NULL) {
539 *next = '\0';
540 return (next + 1);
541 }
542
543 return NULL;
544}
545
546
547static char* ReadLine(char* data) {
548 return ReadToken(data, '\n');
549}
550
551
552static char* ReadWord(char* data) {
553 return ReadToken(data, ' ');
554}
555
556
557// Reads a file into a v8 string.
558Handle<String> Shell::ReadFile(const char* name) {
559 int size = 0;
560 char* chars = ReadChars(name, &size);
561 if (chars == NULL) return Handle<String>();
562 Handle<String> result = String::New(chars);
563 delete[] chars;
564 return result;
565}
566
567
568void Shell::RunShell() {
569 LineEditor* editor = LineEditor::Get();
570 printf("V8 version %s [console: %s]\n", V8::GetVersion(), editor->name());
571 editor->Open();
572 while (true) {
573 Locker locker;
574 HandleScope handle_scope;
575 Context::Scope context_scope(evaluation_context_);
576 i::SmartPointer<char> input = editor->Prompt(Shell::kPrompt);
577 if (input.is_empty())
578 break;
579 editor->AddHistory(*input);
580 Handle<String> name = String::New("(d8)");
581 ExecuteString(String::New(*input), name, true, true);
582 }
583 editor->Close();
584 printf("\n");
585}
586
587
588class ShellThread : public i::Thread {
589 public:
590 ShellThread(int no, i::Vector<const char> files)
591 : no_(no), files_(files) { }
592 virtual void Run();
593 private:
594 int no_;
595 i::Vector<const char> files_;
596};
597
598
599void ShellThread::Run() {
600 // Prepare the context for this thread.
601 Locker locker;
602 HandleScope scope;
603 Handle<ObjectTemplate> global_template = ObjectTemplate::New();
604 global_template->Set(String::New("print"),
605 FunctionTemplate::New(Shell::Print));
606 global_template->Set(String::New("write"),
607 FunctionTemplate::New(Shell::Write));
608 global_template->Set(String::New("read"),
609 FunctionTemplate::New(Shell::Read));
610 global_template->Set(String::New("readline"),
611 FunctionTemplate::New(Shell::ReadLine));
612 global_template->Set(String::New("load"),
613 FunctionTemplate::New(Shell::Load));
614 global_template->Set(String::New("yield"),
615 FunctionTemplate::New(Shell::Yield));
616 global_template->Set(String::New("version"),
617 FunctionTemplate::New(Shell::Version));
618
619 char* ptr = const_cast<char*>(files_.start());
620 while ((ptr != NULL) && (*ptr != '\0')) {
621 // For each newline-separated line.
622 char* next_line = ReadLine(ptr);
623
624 if (*ptr == '#') {
625 // Skip comment lines.
626 ptr = next_line;
627 continue;
628 }
629
630 Persistent<Context> thread_context = Context::New(NULL, global_template);
631 thread_context->SetSecurityToken(Undefined());
632 Context::Scope context_scope(thread_context);
633
634 while ((ptr != NULL) && (*ptr != '\0')) {
635 char* filename = ptr;
636 ptr = ReadWord(ptr);
637
638 // Skip empty strings.
639 if (strlen(filename) == 0) {
640 break;
641 }
642
643 Handle<String> str = Shell::ReadFile(filename);
644 if (str.IsEmpty()) {
645 printf("WARNING: %s not found\n", filename);
646 break;
647 }
648
649 Shell::ExecuteString(str, String::New(filename), false, false);
650 }
651
652 thread_context.Dispose();
653 ptr = next_line;
654 }
655}
656
657
658int Shell::Main(int argc, char* argv[]) {
659 i::FlagList::SetFlagsFromCommandLine(&argc, argv, true);
660 if (i::FLAG_help) {
661 return 1;
662 }
663 Initialize();
664 bool run_shell = (argc == 1);
665
666 // Default use preemption if threads are created.
667 bool use_preemption = true;
668
669 // Default to use lowest possible thread preemption interval to test as many
670 // edgecases as possible.
671 int preemption_interval = 1;
672
673 i::List<i::Thread*> threads(1);
674
675 {
676 // Acquire the V8 lock once initialization has finished. Since the thread
677 // below may spawn new threads accessing V8 holding the V8 lock here is
678 // mandatory.
679 Locker locker;
680 Context::Scope context_scope(evaluation_context_);
681 for (int i = 1; i < argc; i++) {
682 char* str = argv[i];
683 if (strcmp(str, "--shell") == 0) {
684 run_shell = true;
685 } else if (strcmp(str, "--preemption") == 0) {
686 use_preemption = true;
687 } else if (strcmp(str, "--no-preemption") == 0) {
688 use_preemption = false;
689 } else if (strcmp(str, "--preemption-interval") == 0) {
690 if (i + 1 < argc) {
691 char* end = NULL;
692 preemption_interval = strtol(argv[++i], &end, 10); // NOLINT
693 if (preemption_interval <= 0 || *end != '\0' || errno == ERANGE) {
694 printf("Invalid value for --preemption-interval '%s'\n", argv[i]);
695 return 1;
696 }
697 } else {
698 printf("Missing value for --preemption-interval\n");
699 return 1;
700 }
701 } else if (strcmp(str, "-f") == 0) {
702 // Ignore any -f flags for compatibility with other stand-alone
703 // JavaScript engines.
704 continue;
705 } else if (strncmp(str, "--", 2) == 0) {
706 printf("Warning: unknown flag %s.\nTry --help for options\n", str);
707 } else if (strcmp(str, "-e") == 0 && i + 1 < argc) {
708 // Execute argument given to -e option directly.
709 v8::HandleScope handle_scope;
710 v8::Handle<v8::String> file_name = v8::String::New("unnamed");
711 v8::Handle<v8::String> source = v8::String::New(argv[i + 1]);
712 if (!ExecuteString(source, file_name, false, true)) {
713 OnExit();
714 return 1;
715 }
716 i++;
717 } else if (strcmp(str, "-p") == 0 && i + 1 < argc) {
718 int size = 0;
719 const char* files = ReadChars(argv[++i], &size);
720 if (files == NULL) return 1;
721 ShellThread* thread =
722 new ShellThread(threads.length(),
723 i::Vector<const char>(files, size));
724 thread->Start();
725 threads.Add(thread);
726 } else {
727 // Use all other arguments as names of files to load and run.
728 HandleScope handle_scope;
729 Handle<String> file_name = v8::String::New(str);
730 Handle<String> source = ReadFile(str);
731 if (source.IsEmpty()) {
732 printf("Error reading '%s'\n", str);
733 return 1;
734 }
735 if (!ExecuteString(source, file_name, false, true)) {
736 OnExit();
737 return 1;
738 }
739 }
740 }
741
742 // Start preemption if threads have been created and preemption is enabled.
743 if (threads.length() > 0 && use_preemption) {
744 Locker::StartPreemption(preemption_interval);
745 }
746
747#ifdef ENABLE_DEBUGGER_SUPPORT
748 // Run the remote debugger if requested.
749 if (i::FLAG_remote_debugger) {
750 RunRemoteDebugger(i::FLAG_debugger_port);
751 return 0;
752 }
753#endif
754 }
755 if (run_shell)
756 RunShell();
757 for (int i = 0; i < threads.length(); i++) {
758 i::Thread* thread = threads[i];
759 thread->Join();
760 delete thread;
761 }
762 OnExit();
763 return 0;
764}
765
766
767} // namespace v8
768
769
770int main(int argc, char* argv[]) {
771 return v8::Shell::Main(argc, argv);
772}