Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2006-2008 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2006-2008 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "chrome/browser/debugger/debugger_shell.h" | 5 #include "chrome/browser/debugger/debugger_shell.h" |
| 6 | 6 |
| 7 #include "build/build_config.h" | 7 #include "build/build_config.h" |
| 8 | 8 |
| 9 #include "base/file_util.h" | 9 #include "base/file_util.h" |
| 10 #include "base/path_service.h" | 10 #include "base/path_service.h" |
| (...skipping 17 matching lines...) Expand all Loading... | |
| 28 #endif | 28 #endif |
| 29 | 29 |
| 30 DebuggerShell::DebuggerShell(DebuggerInputOutput* io) : io_(io), | 30 DebuggerShell::DebuggerShell(DebuggerInputOutput* io) : io_(io), |
| 31 debugger_ready_(true) { | 31 debugger_ready_(true) { |
| 32 } | 32 } |
| 33 | 33 |
| 34 DebuggerShell::~DebuggerShell() { | 34 DebuggerShell::~DebuggerShell() { |
| 35 io_->Stop(); | 35 io_->Stop(); |
| 36 io_ = NULL; | 36 io_ = NULL; |
| 37 | 37 |
| 38 v8::Locker locked; | |
| 38 v8::HandleScope scope; | 39 v8::HandleScope scope; |
| 39 SubshellFunction("exit", 0, NULL); | 40 SubshellFunction("exit", 0, NULL); |
| 40 v8::V8::RemoveMessageListeners(&DelegateMessageListener); | 41 v8::V8::RemoveMessageListeners(&DelegateMessageListener); |
| 41 v8_this_.Dispose(); | 42 v8_this_.Dispose(); |
| 42 v8_context_.Dispose(); | 43 v8_context_.Dispose(); |
| 43 shell_.Dispose(); | 44 shell_.Dispose(); |
| 44 } | 45 } |
| 45 | 46 |
| 46 void DebuggerShell::Start() { | 47 void DebuggerShell::Start() { |
| 47 io_->Start(this); | 48 io_->Start(this); |
| 48 | 49 |
| 50 v8::Locker locked; | |
| 49 v8::HandleScope scope; | 51 v8::HandleScope scope; |
| 50 | 52 |
| 51 v8_this_ = v8::Persistent<v8::External>::New(v8::External::New(this)); | 53 v8_this_ = v8::Persistent<v8::External>::New(v8::External::New(this)); |
| 52 | 54 |
| 53 v8::V8::AddMessageListener(&DelegateMessageListener, v8_this_); | 55 v8::V8::AddMessageListener(&DelegateMessageListener, v8_this_); |
| 54 | 56 |
| 55 v8::Local<v8::ObjectTemplate> global_template = v8::ObjectTemplate::New(); | 57 v8::Local<v8::ObjectTemplate> global_template = v8::ObjectTemplate::New(); |
| 56 | 58 |
| 57 // shell function | 59 // shell function |
| 58 v8::Local<v8::FunctionTemplate> shell_template = | 60 v8::Local<v8::FunctionTemplate> shell_template = |
| (...skipping 127 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 186 } | 188 } |
| 187 | 189 |
| 188 void DebuggerShell::DelegateMessageListener(v8::Handle<v8::Message> message, | 190 void DebuggerShell::DelegateMessageListener(v8::Handle<v8::Message> message, |
| 189 v8::Handle<v8::Value> data) { | 191 v8::Handle<v8::Value> data) { |
| 190 DCHECK(!data.IsEmpty()); | 192 DCHECK(!data.IsEmpty()); |
| 191 DebuggerShell* debugger = | 193 DebuggerShell* debugger = |
| 192 static_cast<DebuggerShell*>(v8::External::Cast(*data)->Value()); | 194 static_cast<DebuggerShell*>(v8::External::Cast(*data)->Value()); |
| 193 debugger->MessageListener(message); | 195 debugger->MessageListener(message); |
| 194 } | 196 } |
| 195 | 197 |
| 196 void DebuggerShell::MessageListener(v8::Handle<v8::Message> message) { | 198 void DebuggerShell::MessageListener(v8::Handle<v8::Message> message) { |
|
Erik does not do reviews
2009/03/03 17:20:09
Since there can only be one message listener, I as
| |
| 197 v8::HandleScope scope; | 199 v8::HandleScope scope; |
| 198 v8::Local<v8::String> msg_str = message->Get(); | 200 v8::Local<v8::String> msg_str = message->Get(); |
| 199 PrintObject(msg_str); | 201 PrintObject(msg_str); |
| 200 | 202 |
| 201 v8::Handle<v8::Value> data = message->GetScriptResourceName(); | 203 v8::Handle<v8::Value> data = message->GetScriptResourceName(); |
| 202 if (!data.IsEmpty() && !data->IsUndefined()) { | 204 if (!data.IsEmpty() && !data->IsUndefined()) { |
| 203 std::wstring out; | 205 std::wstring out; |
| 204 ObjectToString(data, &out); | 206 ObjectToString(data, &out); |
| 205 int line_number = message->GetLineNumber(); | 207 int line_number = message->GetLineNumber(); |
| 206 if (line_number >= 0) | 208 if (line_number >= 0) |
| 207 out += StringPrintf(L":%d", line_number); | 209 out += StringPrintf(L":%d", line_number); |
| 208 PrintLine(out); | 210 PrintLine(out); |
| 209 data = message->GetSourceLine(); | 211 data = message->GetSourceLine(); |
| 210 if (!data->IsUndefined()) { | 212 if (!data->IsUndefined()) { |
| 211 ObjectToString(data, &out); | 213 ObjectToString(data, &out); |
| 212 PrintLine(out); | 214 PrintLine(out); |
| 213 } | 215 } |
| 214 } | 216 } |
| 215 } | 217 } |
| 216 | 218 |
| 217 void DebuggerShell::Debug(TabContents* tab) { | 219 void DebuggerShell::Debug(TabContents* tab) { |
| 220 v8::Locker locked; | |
| 218 v8::HandleScope outer; | 221 v8::HandleScope outer; |
| 219 v8::Context::Scope scope(v8_context_); | 222 v8::Context::Scope scope(v8_context_); |
| 220 | 223 |
| 221 v8::Local<v8::Object> global = v8_context_->Global(); | 224 v8::Local<v8::Object> global = v8_context_->Global(); |
| 222 v8::Local<v8::Value> function = global->Get(v8::String::New("debug")); | 225 v8::Local<v8::Value> function = global->Get(v8::String::New("debug")); |
| 223 if (function->IsFunction()) { | 226 if (function->IsFunction()) { |
| 224 TabNode* node = new TabNode(tab); | 227 TabNode* node = new TabNode(tab); |
| 225 v8::Handle<v8::Value> argv[] = {node->NewInstance()}; | 228 v8::Handle<v8::Value> argv[] = {node->NewInstance()}; |
| 226 PrintObject(v8::Function::Cast(*function)->Call(global, 1, argv)); | 229 PrintObject(v8::Function::Cast(*function)->Call(global, 1, argv)); |
| 227 } | 230 } |
| 228 } | 231 } |
| 229 | 232 |
| 230 void DebuggerShell::DebugMessage(const std::wstring& msg) { | 233 void DebuggerShell::DebugMessage(const std::wstring& msg) { |
| 234 v8::Locker locked; | |
| 231 v8::HandleScope scope; | 235 v8::HandleScope scope; |
| 232 | 236 |
| 233 if (msg.length()) { | 237 if (msg.length()) { |
| 234 if ((msg[0] == L'{' || msg[0] == L'[' || msg[0] == L'(') && (!shell_.IsEmpty ())) { | 238 if ((msg[0] == L'{' || msg[0] == L'[' || msg[0] == L'(') && (!shell_.IsEmpty ())) { |
| 235 // v8's wide String constructor requires uint16 rather than wchar | 239 // v8's wide String constructor requires uint16 rather than wchar |
| 236 const uint16* data = reinterpret_cast<const uint16* >(msg.c_str()); | 240 const uint16* data = reinterpret_cast<const uint16* >(msg.c_str()); |
| 237 v8::Handle<v8::Value> argv[] = {v8::String::New(data)}; | 241 v8::Handle<v8::Value> argv[] = {v8::String::New(data)}; |
| 238 PrintObject(SubshellFunction("response", 1, argv)); | 242 PrintObject(SubshellFunction("response", 1, argv)); |
| 239 PrintPrompt(); | 243 PrintPrompt(); |
| 240 } else { | 244 } else { |
| 241 if (msg[msg.length() - 1] == L'\n') | 245 if (msg[msg.length() - 1] == L'\n') |
| 242 PrintString(msg); | 246 PrintString(msg); |
| 243 else | 247 else |
| 244 PrintLine(msg); | 248 PrintLine(msg); |
| 245 } | 249 } |
| 246 } | 250 } |
| 247 } | 251 } |
| 248 | 252 |
| 249 void DebuggerShell::OnDebugAttach() { | 253 void DebuggerShell::OnDebugAttach() { |
| 254 v8::Locker locked; | |
| 250 v8::HandleScope scope; | 255 v8::HandleScope scope; |
| 251 SubshellFunction("on_attach", 0, NULL); | 256 SubshellFunction("on_attach", 0, NULL); |
| 252 } | 257 } |
| 253 | 258 |
| 254 void DebuggerShell::OnDebugDisconnect() { | 259 void DebuggerShell::OnDebugDisconnect() { |
| 260 v8::Locker locked; | |
| 255 v8::HandleScope scope; | 261 v8::HandleScope scope; |
| 256 SubshellFunction("on_disconnect", 0, NULL); | 262 SubshellFunction("on_disconnect", 0, NULL); |
| 257 } | 263 } |
| 258 | 264 |
| 259 void DebuggerShell::ObjectToString(v8::Handle<v8::Value> result, std::wstring* s tr) { | 265 void DebuggerShell::ObjectToString(v8::Handle<v8::Value> result, std::wstring* s tr) { |
| 260 v8::HandleScope scope; | 266 v8::HandleScope scope; |
| 261 if (!result.IsEmpty() && !result->IsUndefined()) { | 267 if (!result.IsEmpty() && !result->IsUndefined()) { |
| 262 v8::Local<v8::String> str_obj = result->ToString(); | 268 v8::Local<v8::String> str_obj = result->ToString(); |
| 263 if (!str_obj.IsEmpty()) { | 269 if (!str_obj.IsEmpty()) { |
| 264 int length = str_obj->Length(); | 270 int length = str_obj->Length(); |
| (...skipping 51 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 316 void DebuggerShell::PrintLine(const std::string& out) { | 322 void DebuggerShell::PrintLine(const std::string& out) { |
| 317 if (io_) | 323 if (io_) |
| 318 io_->OutputLine(out); | 324 io_->OutputLine(out); |
| 319 } | 325 } |
| 320 | 326 |
| 321 void DebuggerShell::PrintPrompt() { | 327 void DebuggerShell::PrintPrompt() { |
| 322 std::wstring out = L"Chrome> "; | 328 std::wstring out = L"Chrome> "; |
| 323 if (!shell_.IsEmpty()) { | 329 if (!shell_.IsEmpty()) { |
| 324 if (!debugger_ready_) | 330 if (!debugger_ready_) |
| 325 return; | 331 return; |
| 332 v8::Locker locked; | |
| 326 v8::HandleScope outer; | 333 v8::HandleScope outer; |
| 327 v8::Handle<v8::Value> result = CompileAndRun("shell_.prompt()"); | 334 v8::Handle<v8::Value> result = CompileAndRun("shell_.prompt()"); |
| 328 if (!result.IsEmpty() && !result->IsUndefined()) { | 335 if (!result.IsEmpty() && !result->IsUndefined()) { |
| 329 ObjectToString(result, &out); | 336 ObjectToString(result, &out); |
| 330 } | 337 } |
| 331 } | 338 } |
| 332 if (io_) | 339 if (io_) |
| 333 io_->OutputPrompt(out); | 340 io_->OutputPrompt(out); |
| 334 } | 341 } |
| 335 | 342 |
| 336 void DebuggerShell::ProcessCommand(const std::wstring& data) { | 343 void DebuggerShell::ProcessCommand(const std::wstring& data) { |
| 344 v8::Locker locked; | |
| 337 v8::HandleScope outer; | 345 v8::HandleScope outer; |
| 338 v8::Context::Scope scope(v8_context_); | 346 v8::Context::Scope scope(v8_context_); |
| 339 if (!shell_.IsEmpty() && data.substr(0, 7) != L"source(") { | 347 if (!shell_.IsEmpty() && data.substr(0, 7) != L"source(") { |
| 340 if (data == L"exit") { | 348 if (data == L"exit") { |
| 341 PrintObject(SubshellFunction("exit", 0, NULL)); | 349 PrintObject(SubshellFunction("exit", 0, NULL)); |
| 342 v8_context_->Global()->Delete(v8::String::New("shell_")); | 350 v8_context_->Global()->Delete(v8::String::New("shell_")); |
| 343 shell_.Dispose(); | 351 shell_.Dispose(); |
| 344 shell_.Clear(); | 352 shell_.Clear(); |
| 345 } else { | 353 } else { |
| 346 const uint16* utf16 = reinterpret_cast<const uint16*>(data.c_str()); | 354 const uint16* utf16 = reinterpret_cast<const uint16*>(data.c_str()); |
| (...skipping 21 matching lines...) Expand all Loading... | |
| 368 } | 376 } |
| 369 | 377 |
| 370 void DebuggerShell::LoadUserConfig() { | 378 void DebuggerShell::LoadUserConfig() { |
| 371 std::wstring path; | 379 std::wstring path; |
| 372 PathService::Get(chrome::DIR_USER_DATA, &path); | 380 PathService::Get(chrome::DIR_USER_DATA, &path); |
| 373 file_util::AppendToPath(&path, L"debugger_custom.js"); | 381 file_util::AppendToPath(&path, L"debugger_custom.js"); |
| 374 LoadFile(path); | 382 LoadFile(path); |
| 375 } | 383 } |
| 376 | 384 |
| 377 void DebuggerShell::DidConnect() { | 385 void DebuggerShell::DidConnect() { |
| 386 v8::Locker locked; | |
| 378 v8::HandleScope outer; | 387 v8::HandleScope outer; |
| 379 v8::Context::Scope scope(v8_context_); | 388 v8::Context::Scope scope(v8_context_); |
| 380 | 389 |
| 381 LoadUserConfig(); | 390 LoadUserConfig(); |
| 382 | 391 |
| 383 PrintPrompt(); | 392 PrintPrompt(); |
| 384 } | 393 } |
| 385 | 394 |
| 386 void DebuggerShell::DidDisconnect() { | 395 void DebuggerShell::DidDisconnect() { |
| 396 v8::Locker locked; | |
| 387 v8::HandleScope outer; | 397 v8::HandleScope outer; |
| 388 SubshellFunction("exit", 0, NULL); | 398 SubshellFunction("exit", 0, NULL); |
| 389 } | 399 } |
| 390 | 400 |
| 391 v8::Handle<v8::Value> DebuggerShell::CompileAndRun( | 401 v8::Handle<v8::Value> DebuggerShell::CompileAndRun( |
| 392 const std::string& str, | 402 const std::string& str, |
| 393 const std::string& filename) { | 403 const std::string& filename) { |
| 394 const std::wstring wstr = UTF8ToWide(str); | 404 const std::wstring wstr = UTF8ToWide(str); |
| 395 return CompileAndRun(wstr, filename); | 405 return CompileAndRun(wstr, filename); |
| 396 } | 406 } |
| 397 | 407 |
| 398 v8::Handle<v8::Value> DebuggerShell::CompileAndRun( | 408 v8::Handle<v8::Value> DebuggerShell::CompileAndRun( |
| 399 const std::wstring& wstr, | 409 const std::wstring& wstr, |
| 400 const std::string& filename) { | 410 const std::string& filename) { |
| 411 v8::Locker locked; | |
| 401 v8::Context::Scope scope(v8_context_); | 412 v8::Context::Scope scope(v8_context_); |
| 402 v8::Handle<v8::String> scriptname; | 413 v8::Handle<v8::String> scriptname; |
| 403 if (filename.length() > 0) { | 414 if (filename.length() > 0) { |
| 404 scriptname = v8::String::New(filename.c_str()); | 415 scriptname = v8::String::New(filename.c_str()); |
| 405 } else { | 416 } else { |
| 406 scriptname = v8::String::New(""); | 417 scriptname = v8::String::New(""); |
| 407 } | 418 } |
| 408 const uint16* utf16 = reinterpret_cast<const uint16*>(wstr.c_str()); | 419 const uint16* utf16 = reinterpret_cast<const uint16*>(wstr.c_str()); |
| 409 v8::ScriptOrigin origin = v8::ScriptOrigin(scriptname); | 420 v8::ScriptOrigin origin = v8::ScriptOrigin(scriptname); |
| 410 v8::Local<v8::Script> code = | 421 v8::Local<v8::Script> code = |
| 411 v8::Script::Compile(v8::String::New(utf16), &origin); | 422 v8::Script::Compile(v8::String::New(utf16), &origin); |
| 412 if (!code.IsEmpty()) { | 423 if (!code.IsEmpty()) { |
| 413 v8::Local<v8::Value> result = code->Run(); | 424 v8::Local<v8::Value> result = code->Run(); |
| 414 if (!result.IsEmpty()) { | 425 if (!result.IsEmpty()) { |
| 415 return result; | 426 return result; |
| 416 } | 427 } |
| 417 } | 428 } |
| 418 return v8::Undefined(); | 429 return v8::Undefined(); |
| 419 } | 430 } |
| OLD | NEW |