blob: 5a1e63a763f7e9dae70528cf10bf649445fd565d [file] [log] [blame]
ager@chromium.org41826e72009-03-30 13:30:57 +00001// Copyright 2009 the V8 project authors. All rights reserved.
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +00002// 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
ager@chromium.orga74f0da2008-12-03 16:05:52 +000029#include <stdlib.h>
ager@chromium.orgbb29dc92009-03-24 13:25:23 +000030#include <errno.h>
ager@chromium.orga74f0da2008-12-03 16:05:52 +000031
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +000032#include "d8.h"
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +000033#include "d8-debug.h"
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +000034#include "debug.h"
35#include "api.h"
36#include "natives.h"
ager@chromium.orgddb913d2009-01-27 10:01:48 +000037#include "platform.h"
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +000038
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);
ager@chromium.orga74f0da2008-12-03 16:05:52 +000085 return i::SmartPointer<char>(str ? i::StrDup(str) : str);
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +000086}
87
88
kasperl@chromium.orgf5aa8372009-03-24 14:47:14 +000089CounterMap* Shell::counter_map_;
ager@chromium.orga74f0da2008-12-03 16:05:52 +000090i::OS::MemoryMappedFile* Shell::counters_file_ = NULL;
91CounterCollection Shell::local_counters_;
92CounterCollection* Shell::counters_ = &local_counters_;
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +000093Persistent<Context> Shell::utility_context_;
94Persistent<Context> Shell::evaluation_context_;
95
96
kasperl@chromium.orgf5aa8372009-03-24 14:47:14 +000097bool 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
iposva@chromium.org245aa852009-02-10 00:49:54 +0000104// Converts a V8 value to a C string.
ricow@chromium.orgc9c80822010-04-21 08:22:37 +0000105const char* Shell::ToCString(const v8::String::Utf8Value& value) {
iposva@chromium.org245aa852009-02-10 00:49:54 +0000106 return *value ? *value : "<string conversion failed>";
107}
108
109
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000110// 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;
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000117 if (i::FLAG_debugger) {
118 // When debugging make exceptions appear to be uncaught.
119 try_catch.SetVerbose(true);
120 }
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000121 Handle<Script> script = Script::Compile(source, name);
122 if (script.IsEmpty()) {
123 // Print errors that happened during compilation.
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000124 if (report_exceptions && !i::FLAG_debugger)
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000125 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.
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000131 if (report_exceptions && !i::FLAG_debugger)
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000132 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.
iposva@chromium.org245aa852009-02-10 00:49:54 +0000138 v8::String::Utf8Value str(result);
139 const char* cstr = ToCString(str);
140 printf("%s\n", cstr);
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000141 }
142 return true;
143 }
144 }
145}
146
147
148Handle<Value> Shell::Print(const Arguments& args) {
christian.plesner.hansen@gmail.com5a6af922009-08-12 14:20:51 +0000149 Handle<Value> val = Write(args);
150 printf("\n");
151 return val;
152}
153
154
155Handle<Value> Shell::Write(const Arguments& args) {
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000156 for (int i = 0; i < args.Length(); i++) {
157 HandleScope handle_scope;
christian.plesner.hansen@gmail.com5a6af922009-08-12 14:20:51 +0000158 if (i != 0) {
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000159 printf(" ");
160 }
iposva@chromium.org245aa852009-02-10 00:49:54 +0000161 v8::String::Utf8Value str(args[i]);
sgjesse@chromium.orgac6aa172009-12-04 12:29:05 +0000162 int n = fwrite(*str, sizeof(**str), str.length(), stdout);
163 if (n != str.length()) {
164 printf("Error in fwrite\n");
165 exit(1);
166 }
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000167 }
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000168 return Undefined();
169}
170
171
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000172Handle<Value> Shell::Read(const Arguments& args) {
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000173 String::Utf8Value file(args[0]);
174 if (*file == NULL) {
175 return ThrowException(String::New("Error loading file"));
176 }
177 Handle<String> source = ReadFile(*file);
178 if (source.IsEmpty()) {
179 return ThrowException(String::New("Error loading file"));
180 }
181 return source;
182}
183
184
ager@chromium.orgadd848f2009-08-13 12:44:13 +0000185Handle<Value> Shell::ReadLine(const Arguments& args) {
ager@chromium.orga1645e22009-09-09 19:27:10 +0000186 i::SmartPointer<char> line(i::ReadLine(""));
187 if (*line == NULL) {
188 return Null();
ager@chromium.orgadd848f2009-08-13 12:44:13 +0000189 }
ager@chromium.orga1645e22009-09-09 19:27:10 +0000190 size_t len = strlen(*line);
191 if (len > 0 && line[len - 1] == '\n') {
ager@chromium.orgadd848f2009-08-13 12:44:13 +0000192 --len;
193 }
ager@chromium.orga1645e22009-09-09 19:27:10 +0000194 return String::New(*line, len);
ager@chromium.orgadd848f2009-08-13 12:44:13 +0000195}
196
197
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000198Handle<Value> Shell::Load(const Arguments& args) {
199 for (int i = 0; i < args.Length(); i++) {
200 HandleScope handle_scope;
201 String::Utf8Value file(args[i]);
iposva@chromium.org245aa852009-02-10 00:49:54 +0000202 if (*file == NULL) {
203 return ThrowException(String::New("Error loading file"));
204 }
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000205 Handle<String> source = ReadFile(*file);
206 if (source.IsEmpty()) {
207 return ThrowException(String::New("Error loading file"));
208 }
209 if (!ExecuteString(source, String::New(*file), false, false)) {
sgjesse@chromium.orgac6aa172009-12-04 12:29:05 +0000210 return ThrowException(String::New("Error executing file"));
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000211 }
212 }
213 return Undefined();
214}
215
216
ager@chromium.orgbb29dc92009-03-24 13:25:23 +0000217Handle<Value> Shell::Yield(const Arguments& args) {
218 v8::Unlocker unlocker;
219 return Undefined();
220}
221
222
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000223Handle<Value> Shell::Quit(const Arguments& args) {
224 int exit_code = args[0]->Int32Value();
225 OnExit();
226 exit(exit_code);
227 return Undefined();
228}
229
230
231Handle<Value> Shell::Version(const Arguments& args) {
232 return String::New(V8::GetVersion());
233}
234
235
236void Shell::ReportException(v8::TryCatch* try_catch) {
237 HandleScope handle_scope;
iposva@chromium.org245aa852009-02-10 00:49:54 +0000238 v8::String::Utf8Value exception(try_catch->Exception());
239 const char* exception_string = ToCString(exception);
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000240 Handle<Message> message = try_catch->Message();
241 if (message.IsEmpty()) {
242 // V8 didn't provide any extra information about this error; just
243 // print the exception.
iposva@chromium.org245aa852009-02-10 00:49:54 +0000244 printf("%s\n", exception_string);
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000245 } else {
246 // Print (filename):(line number): (message).
iposva@chromium.org245aa852009-02-10 00:49:54 +0000247 v8::String::Utf8Value filename(message->GetScriptResourceName());
248 const char* filename_string = ToCString(filename);
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000249 int linenum = message->GetLineNumber();
iposva@chromium.org245aa852009-02-10 00:49:54 +0000250 printf("%s:%i: %s\n", filename_string, linenum, exception_string);
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000251 // Print line of source code.
iposva@chromium.org245aa852009-02-10 00:49:54 +0000252 v8::String::Utf8Value sourceline(message->GetSourceLine());
253 const char* sourceline_string = ToCString(sourceline);
254 printf("%s\n", sourceline_string);
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000255 // Print wavy underline (GetUnderline is deprecated).
256 int start = message->GetStartColumn();
257 for (int i = 0; i < start; i++) {
258 printf(" ");
259 }
260 int end = message->GetEndColumn();
261 for (int i = start; i < end; i++) {
262 printf("^");
263 }
264 printf("\n");
265 }
266}
267
268
269Handle<Array> Shell::GetCompletions(Handle<String> text, Handle<String> full) {
270 HandleScope handle_scope;
271 Context::Scope context_scope(utility_context_);
272 Handle<Object> global = utility_context_->Global();
273 Handle<Value> fun = global->Get(String::New("GetCompletions"));
274 static const int kArgc = 3;
275 Handle<Value> argv[kArgc] = { evaluation_context_->Global(), text, full };
276 Handle<Value> val = Handle<Function>::Cast(fun)->Call(global, kArgc, argv);
277 return handle_scope.Close(Handle<Array>::Cast(val));
278}
279
280
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000281#ifdef ENABLE_DEBUGGER_SUPPORT
kasperl@chromium.org7be3c992009-03-12 07:19:55 +0000282Handle<Object> Shell::DebugMessageDetails(Handle<String> message) {
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000283 Context::Scope context_scope(utility_context_);
284 Handle<Object> global = utility_context_->Global();
kasperl@chromium.org7be3c992009-03-12 07:19:55 +0000285 Handle<Value> fun = global->Get(String::New("DebugMessageDetails"));
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000286 static const int kArgc = 1;
kasperl@chromium.org7be3c992009-03-12 07:19:55 +0000287 Handle<Value> argv[kArgc] = { message };
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000288 Handle<Value> val = Handle<Function>::Cast(fun)->Call(global, kArgc, argv);
kasperl@chromium.org7be3c992009-03-12 07:19:55 +0000289 return Handle<Object>::Cast(val);
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000290}
291
292
293Handle<Value> Shell::DebugCommandToJSONRequest(Handle<String> command) {
294 Context::Scope context_scope(utility_context_);
295 Handle<Object> global = utility_context_->Global();
296 Handle<Value> fun = global->Get(String::New("DebugCommandToJSONRequest"));
297 static const int kArgc = 1;
298 Handle<Value> argv[kArgc] = { command };
299 Handle<Value> val = Handle<Function>::Cast(fun)->Call(global, kArgc, argv);
300 return val;
301}
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000302#endif
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000303
304
kasperl@chromium.orgd1e3e722009-04-14 13:38:25 +0000305int32_t* Counter::Bind(const char* name, bool is_histogram) {
ager@chromium.orga74f0da2008-12-03 16:05:52 +0000306 int i;
307 for (i = 0; i < kMaxNameSize - 1 && name[i]; i++)
308 name_[i] = static_cast<char>(name[i]);
309 name_[i] = '\0';
kasperl@chromium.orgd1e3e722009-04-14 13:38:25 +0000310 is_histogram_ = is_histogram;
311 return ptr();
312}
313
314
315void Counter::AddSample(int32_t sample) {
316 count_++;
317 sample_total_ += sample;
ager@chromium.orga74f0da2008-12-03 16:05:52 +0000318}
319
320
321CounterCollection::CounterCollection() {
322 magic_number_ = 0xDEADFACE;
323 max_counters_ = kMaxCounters;
324 max_name_size_ = Counter::kMaxNameSize;
325 counters_in_use_ = 0;
326}
327
328
329Counter* CounterCollection::GetNextCounter() {
330 if (counters_in_use_ == kMaxCounters) return NULL;
331 return &counters_[counters_in_use_++];
332}
333
334
335void Shell::MapCounters(const char* name) {
336 counters_file_ = i::OS::MemoryMappedFile::create(name,
337 sizeof(CounterCollection), &local_counters_);
338 void* memory = (counters_file_ == NULL) ?
339 NULL : counters_file_->memory();
340 if (memory == NULL) {
341 printf("Could not map counters file %s\n", name);
342 exit(1);
343 }
344 counters_ = static_cast<CounterCollection*>(memory);
345 V8::SetCounterFunction(LookupCounter);
kasperl@chromium.orgd1e3e722009-04-14 13:38:25 +0000346 V8::SetCreateHistogramFunction(CreateHistogram);
347 V8::SetAddHistogramSampleFunction(AddHistogramSample);
ager@chromium.orga74f0da2008-12-03 16:05:52 +0000348}
349
350
kasperl@chromium.orgf5aa8372009-03-24 14:47:14 +0000351int CounterMap::Hash(const char* name) {
352 int h = 0;
353 int c;
354 while ((c = *name++) != 0) {
355 h += h << 5;
356 h += c;
357 }
358 return h;
359}
360
361
kasperl@chromium.orgd1e3e722009-04-14 13:38:25 +0000362Counter* Shell::GetCounter(const char* name, bool is_histogram) {
kasperl@chromium.orgf5aa8372009-03-24 14:47:14 +0000363 Counter* counter = counter_map_->Lookup(name);
kasperl@chromium.orgd1e3e722009-04-14 13:38:25 +0000364
365 if (counter == NULL) {
366 counter = counters_->GetNextCounter();
367 if (counter != NULL) {
368 counter_map_->Set(name, counter);
369 counter->Bind(name, is_histogram);
370 }
371 } else {
372 ASSERT(counter->is_histogram() == is_histogram);
373 }
374 return counter;
375}
376
377
378int* Shell::LookupCounter(const char* name) {
379 Counter* counter = GetCounter(name, false);
380
kasperl@chromium.orgf5aa8372009-03-24 14:47:14 +0000381 if (counter != NULL) {
382 return counter->ptr();
kasperl@chromium.orgd1e3e722009-04-14 13:38:25 +0000383 } else {
384 return NULL;
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000385 }
kasperl@chromium.orgd1e3e722009-04-14 13:38:25 +0000386}
387
388
389void* Shell::CreateHistogram(const char* name,
390 int min,
391 int max,
392 size_t buckets) {
393 return GetCounter(name, true);
394}
395
396
397void Shell::AddHistogramSample(void* histogram, int sample) {
398 Counter* counter = reinterpret_cast<Counter*>(histogram);
399 counter->AddSample(sample);
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000400}
401
402
403void Shell::Initialize() {
kasperl@chromium.orgf5aa8372009-03-24 14:47:14 +0000404 Shell::counter_map_ = new CounterMap();
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000405 // Set up counters
ager@chromium.orga74f0da2008-12-03 16:05:52 +0000406 if (i::FLAG_map_counters != NULL)
407 MapCounters(i::FLAG_map_counters);
kasperl@chromium.orgd1e3e722009-04-14 13:38:25 +0000408 if (i::FLAG_dump_counters) {
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000409 V8::SetCounterFunction(LookupCounter);
kasperl@chromium.orgd1e3e722009-04-14 13:38:25 +0000410 V8::SetCreateHistogramFunction(CreateHistogram);
411 V8::SetAddHistogramSampleFunction(AddHistogramSample);
412 }
413
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000414 // Initialize the global objects
415 HandleScope scope;
416 Handle<ObjectTemplate> global_template = ObjectTemplate::New();
417 global_template->Set(String::New("print"), FunctionTemplate::New(Print));
christian.plesner.hansen@gmail.com5a6af922009-08-12 14:20:51 +0000418 global_template->Set(String::New("write"), FunctionTemplate::New(Write));
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000419 global_template->Set(String::New("read"), FunctionTemplate::New(Read));
ager@chromium.orgadd848f2009-08-13 12:44:13 +0000420 global_template->Set(String::New("readline"),
421 FunctionTemplate::New(ReadLine));
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000422 global_template->Set(String::New("load"), FunctionTemplate::New(Load));
423 global_template->Set(String::New("quit"), FunctionTemplate::New(Quit));
424 global_template->Set(String::New("version"), FunctionTemplate::New(Version));
ager@chromium.org71daaf62009-04-01 07:22:49 +0000425
426 Handle<ObjectTemplate> os_templ = ObjectTemplate::New();
427 AddOSMethods(os_templ);
428 global_template->Set(String::New("os"), os_templ);
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000429
430 utility_context_ = Context::New(NULL, global_template);
431 utility_context_->SetSecurityToken(Undefined());
432 Context::Scope utility_scope(utility_context_);
433
ager@chromium.org3bf7b912008-11-17 09:09:45 +0000434 i::JSArguments js_args = i::FLAG_js_arguments;
435 i::Handle<i::FixedArray> arguments_array =
436 i::Factory::NewFixedArray(js_args.argc());
437 for (int j = 0; j < js_args.argc(); j++) {
438 i::Handle<i::String> arg =
439 i::Factory::NewStringFromUtf8(i::CStrVector(js_args[j]));
440 arguments_array->set(j, *arg);
441 }
442 i::Handle<i::JSArray> arguments_jsarray =
443 i::Factory::NewJSArrayWithElements(arguments_array);
444 global_template->Set(String::New("arguments"),
445 Utils::ToLocal(arguments_jsarray));
446
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000447#ifdef ENABLE_DEBUGGER_SUPPORT
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000448 // Install the debugger object in the utility scope
449 i::Debug::Load();
ager@chromium.orgac091b72010-05-05 07:34:42 +0000450 i::Handle<i::JSObject> debug
451 = i::Handle<i::JSObject>(i::Debug::debug_context()->global());
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000452 utility_context_->Global()->Set(String::New("$debug"),
ager@chromium.orgac091b72010-05-05 07:34:42 +0000453 Utils::ToLocal(debug));
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000454#endif
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000455
456 // Run the d8 shell utility script in the utility context
457 int source_index = i::NativesCollection<i::D8>::GetIndex("d8");
458 i::Vector<const char> shell_source
459 = i::NativesCollection<i::D8>::GetScriptSource(source_index);
460 i::Vector<const char> shell_source_name
461 = i::NativesCollection<i::D8>::GetScriptName(source_index);
462 Handle<String> source = String::New(shell_source.start(),
463 shell_source.length());
464 Handle<String> name = String::New(shell_source_name.start(),
465 shell_source_name.length());
kasperl@chromium.org7be3c992009-03-12 07:19:55 +0000466 Handle<Script> script = Script::Compile(source, name);
467 script->Run();
468
469 // Mark the d8 shell script as native to avoid it showing up as normal source
470 // in the debugger.
kmillikin@chromium.org5d8f0e62010-03-24 08:21:20 +0000471 i::Handle<i::Object> compiled_script = Utils::OpenHandle(*script);
472 i::Handle<i::Script> script_object = compiled_script->IsJSFunction()
473 ? i::Handle<i::Script>(i::Script::cast(
474 i::JSFunction::cast(*compiled_script)->shared()->script()))
475 : i::Handle<i::Script>(i::Script::cast(
476 i::SharedFunctionInfo::cast(*compiled_script)->script()));
ager@chromium.orge2902be2009-06-08 12:21:35 +0000477 script_object->set_type(i::Smi::FromInt(i::Script::TYPE_NATIVE));
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000478
479 // Create the evaluation context
480 evaluation_context_ = Context::New(NULL, global_template);
481 evaluation_context_->SetSecurityToken(Undefined());
ager@chromium.orga74f0da2008-12-03 16:05:52 +0000482
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000483#ifdef ENABLE_DEBUGGER_SUPPORT
ager@chromium.orga74f0da2008-12-03 16:05:52 +0000484 // Set the security token of the debug context to allow access.
485 i::Debug::debug_context()->set_security_token(i::Heap::undefined_value());
ager@chromium.orgeadaf222009-06-16 09:43:10 +0000486
487 // Start the debugger agent if requested.
488 if (i::FLAG_debugger_agent) {
kmillikin@chromium.org3cdd9e12010-09-06 11:39:48 +0000489 v8::Debug::EnableAgent("d8 shell", i::FLAG_debugger_port, true);
ager@chromium.orgeadaf222009-06-16 09:43:10 +0000490 }
491
492 // Start the in-process debugger if requested.
493 if (i::FLAG_debugger && !i::FLAG_debugger_agent) {
494 v8::Debug::SetDebugEventListener(HandleDebugEvent);
495 }
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000496#endif
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000497}
498
499
500void Shell::OnExit() {
501 if (i::FLAG_dump_counters) {
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000502 ::printf("+----------------------------------------+-------------+\n");
503 ::printf("| Name | Value |\n");
504 ::printf("+----------------------------------------+-------------+\n");
kasperl@chromium.orgf5aa8372009-03-24 14:47:14 +0000505 for (CounterMap::Iterator i(counter_map_); i.More(); i.Next()) {
506 Counter* counter = i.CurrentValue();
kasperl@chromium.orgd1e3e722009-04-14 13:38:25 +0000507 if (counter->is_histogram()) {
508 ::printf("| c:%-36s | %11i |\n", i.CurrentKey(), counter->count());
509 ::printf("| t:%-36s | %11i |\n",
510 i.CurrentKey(),
511 counter->sample_total());
512 } else {
513 ::printf("| %-38s | %11i |\n", i.CurrentKey(), counter->count());
514 }
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000515 }
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000516 ::printf("+----------------------------------------+-------------+\n");
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000517 }
ager@chromium.orga74f0da2008-12-03 16:05:52 +0000518 if (counters_file_ != NULL)
519 delete counters_file_;
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000520}
521
522
kasperl@chromium.orgb3284ad2009-05-18 06:12:45 +0000523static char* ReadChars(const char* name, int* size_out) {
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000524 v8::Unlocker unlocker; // Release the V8 lock while reading files.
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000525 FILE* file = i::OS::FOpen(name, "rb");
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000526 if (file == NULL) return NULL;
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000527
528 fseek(file, 0, SEEK_END);
529 int size = ftell(file);
530 rewind(file);
531
532 char* chars = new char[size + 1];
533 chars[size] = '\0';
534 for (int i = 0; i < size;) {
535 int read = fread(&chars[i], 1, size - i, file);
536 i += read;
537 }
538 fclose(file);
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000539 *size_out = size;
540 return chars;
541}
542
543
ager@chromium.org381abbb2009-02-25 13:23:22 +0000544static char* ReadToken(char* data, char token) {
545 char* next = i::OS::StrChr(data, token);
546 if (next != NULL) {
547 *next = '\0';
548 return (next + 1);
549 }
550
551 return NULL;
552}
553
554
555static char* ReadLine(char* data) {
556 return ReadToken(data, '\n');
557}
558
559
560static char* ReadWord(char* data) {
561 return ReadToken(data, ' ');
562}
563
564
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000565// Reads a file into a v8 string.
566Handle<String> Shell::ReadFile(const char* name) {
567 int size = 0;
568 char* chars = ReadChars(name, &size);
569 if (chars == NULL) return Handle<String>();
570 Handle<String> result = String::New(chars);
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000571 delete[] chars;
572 return result;
573}
574
575
576void Shell::RunShell() {
577 LineEditor* editor = LineEditor::Get();
578 printf("V8 version %s [console: %s]\n", V8::GetVersion(), editor->name());
vegorov@chromium.org2356e6f2010-06-09 09:38:56 +0000579 if (i::FLAG_debugger) {
580 printf("JavaScript debugger enabled\n");
581 }
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000582 editor->Open();
583 while (true) {
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000584 Locker locker;
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000585 HandleScope handle_scope;
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000586 Context::Scope context_scope(evaluation_context_);
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000587 i::SmartPointer<char> input = editor->Prompt(Shell::kPrompt);
588 if (input.is_empty())
589 break;
590 editor->AddHistory(*input);
591 Handle<String> name = String::New("(d8)");
592 ExecuteString(String::New(*input), name, true, true);
593 }
594 editor->Close();
595 printf("\n");
596}
597
598
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000599class ShellThread : public i::Thread {
600 public:
601 ShellThread(int no, i::Vector<const char> files)
602 : no_(no), files_(files) { }
603 virtual void Run();
604 private:
605 int no_;
606 i::Vector<const char> files_;
607};
608
609
610void ShellThread::Run() {
611 // Prepare the context for this thread.
612 Locker locker;
613 HandleScope scope;
614 Handle<ObjectTemplate> global_template = ObjectTemplate::New();
615 global_template->Set(String::New("print"),
616 FunctionTemplate::New(Shell::Print));
christian.plesner.hansen@gmail.com5a6af922009-08-12 14:20:51 +0000617 global_template->Set(String::New("write"),
618 FunctionTemplate::New(Shell::Write));
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000619 global_template->Set(String::New("read"),
620 FunctionTemplate::New(Shell::Read));
ager@chromium.orgadd848f2009-08-13 12:44:13 +0000621 global_template->Set(String::New("readline"),
622 FunctionTemplate::New(Shell::ReadLine));
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000623 global_template->Set(String::New("load"),
624 FunctionTemplate::New(Shell::Load));
ager@chromium.orgbb29dc92009-03-24 13:25:23 +0000625 global_template->Set(String::New("yield"),
626 FunctionTemplate::New(Shell::Yield));
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000627 global_template->Set(String::New("version"),
628 FunctionTemplate::New(Shell::Version));
629
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000630 char* ptr = const_cast<char*>(files_.start());
631 while ((ptr != NULL) && (*ptr != '\0')) {
632 // For each newline-separated line.
ager@chromium.org381abbb2009-02-25 13:23:22 +0000633 char* next_line = ReadLine(ptr);
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000634
ager@chromium.org381abbb2009-02-25 13:23:22 +0000635 if (*ptr == '#') {
636 // Skip comment lines.
637 ptr = next_line;
638 continue;
639 }
640
641 Persistent<Context> thread_context = Context::New(NULL, global_template);
642 thread_context->SetSecurityToken(Undefined());
643 Context::Scope context_scope(thread_context);
644
645 while ((ptr != NULL) && (*ptr != '\0')) {
646 char* filename = ptr;
647 ptr = ReadWord(ptr);
648
649 // Skip empty strings.
650 if (strlen(filename) == 0) {
651 break;
652 }
653
654 Handle<String> str = Shell::ReadFile(filename);
655 if (str.IsEmpty()) {
656 printf("WARNING: %s not found\n", filename);
657 break;
658 }
659
660 Shell::ExecuteString(str, String::New(filename), false, false);
661 }
662
663 thread_context.Dispose();
664 ptr = next_line;
665 }
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000666}
667
668
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000669int Shell::Main(int argc, char* argv[]) {
670 i::FlagList::SetFlagsFromCommandLine(&argc, argv, true);
ager@chromium.org3bf7b912008-11-17 09:09:45 +0000671 if (i::FLAG_help) {
672 return 1;
673 }
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000674 Initialize();
675 bool run_shell = (argc == 1);
ager@chromium.orgbb29dc92009-03-24 13:25:23 +0000676
677 // Default use preemption if threads are created.
678 bool use_preemption = true;
679
680 // Default to use lowest possible thread preemption interval to test as many
681 // edgecases as possible.
682 int preemption_interval = 1;
683
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000684 i::List<i::Thread*> threads(1);
685
686 {
687 // Acquire the V8 lock once initialization has finished. Since the thread
688 // below may spawn new threads accessing V8 holding the V8 lock here is
689 // mandatory.
690 Locker locker;
691 Context::Scope context_scope(evaluation_context_);
692 for (int i = 1; i < argc; i++) {
693 char* str = argv[i];
694 if (strcmp(str, "--shell") == 0) {
695 run_shell = true;
ager@chromium.orgbb29dc92009-03-24 13:25:23 +0000696 } else if (strcmp(str, "--preemption") == 0) {
697 use_preemption = true;
698 } else if (strcmp(str, "--no-preemption") == 0) {
699 use_preemption = false;
700 } else if (strcmp(str, "--preemption-interval") == 0) {
701 if (i + 1 < argc) {
kasperl@chromium.orgb3284ad2009-05-18 06:12:45 +0000702 char* end = NULL;
ager@chromium.orgbb29dc92009-03-24 13:25:23 +0000703 preemption_interval = strtol(argv[++i], &end, 10); // NOLINT
704 if (preemption_interval <= 0 || *end != '\0' || errno == ERANGE) {
705 printf("Invalid value for --preemption-interval '%s'\n", argv[i]);
706 return 1;
707 }
708 } else {
709 printf("Missing value for --preemption-interval\n");
710 return 1;
711 }
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000712 } else if (strcmp(str, "-f") == 0) {
713 // Ignore any -f flags for compatibility with other stand-alone
714 // JavaScript engines.
715 continue;
716 } else if (strncmp(str, "--", 2) == 0) {
717 printf("Warning: unknown flag %s.\nTry --help for options\n", str);
718 } else if (strcmp(str, "-e") == 0 && i + 1 < argc) {
719 // Execute argument given to -e option directly.
720 v8::HandleScope handle_scope;
721 v8::Handle<v8::String> file_name = v8::String::New("unnamed");
722 v8::Handle<v8::String> source = v8::String::New(argv[i + 1]);
ager@chromium.orgbb29dc92009-03-24 13:25:23 +0000723 if (!ExecuteString(source, file_name, false, true)) {
724 OnExit();
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000725 return 1;
ager@chromium.orgbb29dc92009-03-24 13:25:23 +0000726 }
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000727 i++;
728 } else if (strcmp(str, "-p") == 0 && i + 1 < argc) {
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000729 int size = 0;
kasperl@chromium.orgb3284ad2009-05-18 06:12:45 +0000730 const char* files = ReadChars(argv[++i], &size);
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000731 if (files == NULL) return 1;
kasperl@chromium.orgb3284ad2009-05-18 06:12:45 +0000732 ShellThread* thread =
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000733 new ShellThread(threads.length(),
734 i::Vector<const char>(files, size));
735 thread->Start();
736 threads.Add(thread);
737 } else {
738 // Use all other arguments as names of files to load and run.
739 HandleScope handle_scope;
740 Handle<String> file_name = v8::String::New(str);
741 Handle<String> source = ReadFile(str);
742 if (source.IsEmpty()) {
743 printf("Error reading '%s'\n", str);
744 return 1;
745 }
ager@chromium.orgbb29dc92009-03-24 13:25:23 +0000746 if (!ExecuteString(source, file_name, false, true)) {
747 OnExit();
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000748 return 1;
ager@chromium.orgbb29dc92009-03-24 13:25:23 +0000749 }
ager@chromium.orga74f0da2008-12-03 16:05:52 +0000750 }
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000751 }
kasperl@chromium.org7be3c992009-03-12 07:19:55 +0000752
ager@chromium.orgbb29dc92009-03-24 13:25:23 +0000753 // Start preemption if threads have been created and preemption is enabled.
754 if (threads.length() > 0 && use_preemption) {
755 Locker::StartPreemption(preemption_interval);
756 }
757
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000758#ifdef ENABLE_DEBUGGER_SUPPORT
kasperl@chromium.org7be3c992009-03-12 07:19:55 +0000759 // Run the remote debugger if requested.
760 if (i::FLAG_remote_debugger) {
761 RunRemoteDebugger(i::FLAG_debugger_port);
762 return 0;
763 }
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000764#endif
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000765 }
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000766 if (run_shell)
767 RunShell();
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000768 for (int i = 0; i < threads.length(); i++) {
kasperl@chromium.orgb3284ad2009-05-18 06:12:45 +0000769 i::Thread* thread = threads[i];
ager@chromium.orgddb913d2009-01-27 10:01:48 +0000770 thread->Join();
771 delete thread;
772 }
kasperl@chromium.org5a8ca6c2008-10-23 13:57:19 +0000773 OnExit();
774 return 0;
775}
776
777
778} // namespace v8
779
780
781int main(int argc, char* argv[]) {
782 return v8::Shell::Main(argc, argv);
783}