• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1// Copyright 2012 the V8 project authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5(function (global, utils) {
6"use strict";
7
8// ----------------------------------------------------------------------------
9// Imports
10
11var FrameMirror = global.FrameMirror;
12var GlobalArray = global.Array;
13var GlobalRegExp = global.RegExp;
14var IsNaN = global.isNaN;
15var JSONParse = global.JSON.parse;
16var JSONStringify = global.JSON.stringify;
17var LookupMirror = global.LookupMirror;
18var MakeError;
19var MakeTypeError;
20var MakeMirror = global.MakeMirror;
21var MakeMirrorSerializer = global.MakeMirrorSerializer;
22var MathMin = global.Math.min;
23var Mirror = global.Mirror;
24var MirrorType;
25var ParseInt = global.parseInt;
26var ValueMirror = global.ValueMirror;
27
28utils.Import(function(from) {
29  MakeError = from.MakeError;
30  MakeTypeError = from.MakeTypeError;
31  MirrorType = from.MirrorType;
32});
33
34//----------------------------------------------------------------------------
35
36// Default number of frames to include in the response to backtrace request.
37var kDefaultBacktraceLength = 10;
38
39var Debug = {};
40
41// Regular expression to skip "crud" at the beginning of a source line which is
42// not really code. Currently the regular expression matches whitespace and
43// comments.
44var sourceLineBeginningSkip = /^(?:\s*(?:\/\*.*?\*\/)*)*/;
45
46// Debug events which can occour in the V8 JavaScript engine. These originate
47// from the API include file debug.h.
48Debug.DebugEvent = { Break: 1,
49                     Exception: 2,
50                     NewFunction: 3,
51                     BeforeCompile: 4,
52                     AfterCompile: 5,
53                     CompileError: 6,
54                     PromiseEvent: 7,
55                     AsyncTaskEvent: 8 };
56
57// Types of exceptions that can be broken upon.
58Debug.ExceptionBreak = { Caught : 0,
59                         Uncaught: 1 };
60
61// The different types of steps.
62Debug.StepAction = { StepOut: 0,
63                     StepNext: 1,
64                     StepIn: 2,
65                     StepFrame: 3 };
66
67// The different types of scripts matching enum ScriptType in objects.h.
68Debug.ScriptType = { Native: 0,
69                     Extension: 1,
70                     Normal: 2 };
71
72// The different types of script compilations matching enum
73// Script::CompilationType in objects.h.
74Debug.ScriptCompilationType = { Host: 0,
75                                Eval: 1,
76                                JSON: 2 };
77
78// The different script break point types.
79Debug.ScriptBreakPointType = { ScriptId: 0,
80                               ScriptName: 1,
81                               ScriptRegExp: 2 };
82
83// The different types of breakpoint position alignments.
84// Must match BreakPositionAlignment in debug.h.
85Debug.BreakPositionAlignment = {
86  Statement: 0,
87  BreakPosition: 1
88};
89
90function ScriptTypeFlag(type) {
91  return (1 << type);
92}
93
94// Globals.
95var next_response_seq = 0;
96var next_break_point_number = 1;
97var break_points = [];
98var script_break_points = [];
99var debugger_flags = {
100  breakPointsActive: {
101    value: true,
102    getValue: function() { return this.value; },
103    setValue: function(value) {
104      this.value = !!value;
105      %SetBreakPointsActive(this.value);
106    }
107  },
108  breakOnCaughtException: {
109    getValue: function() { return Debug.isBreakOnException(); },
110    setValue: function(value) {
111      if (value) {
112        Debug.setBreakOnException();
113      } else {
114        Debug.clearBreakOnException();
115      }
116    }
117  },
118  breakOnUncaughtException: {
119    getValue: function() { return Debug.isBreakOnUncaughtException(); },
120    setValue: function(value) {
121      if (value) {
122        Debug.setBreakOnUncaughtException();
123      } else {
124        Debug.clearBreakOnUncaughtException();
125      }
126    }
127  },
128};
129
130
131// Create a new break point object and add it to the list of break points.
132function MakeBreakPoint(source_position, opt_script_break_point) {
133  var break_point = new BreakPoint(source_position, opt_script_break_point);
134  break_points.push(break_point);
135  return break_point;
136}
137
138
139// Object representing a break point.
140// NOTE: This object does not have a reference to the function having break
141// point as this would cause function not to be garbage collected when it is
142// not used any more. We do not want break points to keep functions alive.
143function BreakPoint(source_position, opt_script_break_point) {
144  this.source_position_ = source_position;
145  if (opt_script_break_point) {
146    this.script_break_point_ = opt_script_break_point;
147  } else {
148    this.number_ = next_break_point_number++;
149  }
150  this.hit_count_ = 0;
151  this.active_ = true;
152  this.condition_ = null;
153  this.ignoreCount_ = 0;
154}
155
156
157BreakPoint.prototype.number = function() {
158  return this.number_;
159};
160
161
162BreakPoint.prototype.func = function() {
163  return this.func_;
164};
165
166
167BreakPoint.prototype.source_position = function() {
168  return this.source_position_;
169};
170
171
172BreakPoint.prototype.hit_count = function() {
173  return this.hit_count_;
174};
175
176
177BreakPoint.prototype.active = function() {
178  if (this.script_break_point()) {
179    return this.script_break_point().active();
180  }
181  return this.active_;
182};
183
184
185BreakPoint.prototype.condition = function() {
186  if (this.script_break_point() && this.script_break_point().condition()) {
187    return this.script_break_point().condition();
188  }
189  return this.condition_;
190};
191
192
193BreakPoint.prototype.ignoreCount = function() {
194  return this.ignoreCount_;
195};
196
197
198BreakPoint.prototype.script_break_point = function() {
199  return this.script_break_point_;
200};
201
202
203BreakPoint.prototype.enable = function() {
204  this.active_ = true;
205};
206
207
208BreakPoint.prototype.disable = function() {
209  this.active_ = false;
210};
211
212
213BreakPoint.prototype.setCondition = function(condition) {
214  this.condition_ = condition;
215};
216
217
218BreakPoint.prototype.setIgnoreCount = function(ignoreCount) {
219  this.ignoreCount_ = ignoreCount;
220};
221
222
223BreakPoint.prototype.isTriggered = function(exec_state) {
224  // Break point not active - not triggered.
225  if (!this.active()) return false;
226
227  // Check for conditional break point.
228  if (this.condition()) {
229    // If break point has condition try to evaluate it in the top frame.
230    try {
231      var mirror = exec_state.frame(0).evaluate(this.condition());
232      // If no sensible mirror or non true value break point not triggered.
233      if (!(mirror instanceof ValueMirror) || !mirror.value_) {
234        return false;
235      }
236    } catch (e) {
237      // Exception evaluating condition counts as not triggered.
238      return false;
239    }
240  }
241
242  // Update the hit count.
243  this.hit_count_++;
244  if (this.script_break_point_) {
245    this.script_break_point_.hit_count_++;
246  }
247
248  // If the break point has an ignore count it is not triggered.
249  if (this.ignoreCount_ > 0) {
250    this.ignoreCount_--;
251    return false;
252  }
253
254  // Break point triggered.
255  return true;
256};
257
258
259// Function called from the runtime when a break point is hit. Returns true if
260// the break point is triggered and supposed to break execution.
261function IsBreakPointTriggered(break_id, break_point) {
262  return break_point.isTriggered(MakeExecutionState(break_id));
263}
264
265
266// Object representing a script break point. The script is referenced by its
267// script name or script id and the break point is represented as line and
268// column.
269function ScriptBreakPoint(type, script_id_or_name, opt_line, opt_column,
270                          opt_groupId, opt_position_alignment) {
271  this.type_ = type;
272  if (type == Debug.ScriptBreakPointType.ScriptId) {
273    this.script_id_ = script_id_or_name;
274  } else if (type == Debug.ScriptBreakPointType.ScriptName) {
275    this.script_name_ = script_id_or_name;
276  } else if (type == Debug.ScriptBreakPointType.ScriptRegExp) {
277    this.script_regexp_object_ = new GlobalRegExp(script_id_or_name);
278  } else {
279    throw MakeError(kDebugger, "Unexpected breakpoint type " + type);
280  }
281  this.line_ = opt_line || 0;
282  this.column_ = opt_column;
283  this.groupId_ = opt_groupId;
284  this.position_alignment_ = IS_UNDEFINED(opt_position_alignment)
285      ? Debug.BreakPositionAlignment.Statement : opt_position_alignment;
286  this.hit_count_ = 0;
287  this.active_ = true;
288  this.condition_ = null;
289  this.ignoreCount_ = 0;
290  this.break_points_ = [];
291}
292
293
294// Creates a clone of script breakpoint that is linked to another script.
295ScriptBreakPoint.prototype.cloneForOtherScript = function (other_script) {
296  var copy = new ScriptBreakPoint(Debug.ScriptBreakPointType.ScriptId,
297      other_script.id, this.line_, this.column_, this.groupId_,
298      this.position_alignment_);
299  copy.number_ = next_break_point_number++;
300  script_break_points.push(copy);
301
302  copy.hit_count_ = this.hit_count_;
303  copy.active_ = this.active_;
304  copy.condition_ = this.condition_;
305  copy.ignoreCount_ = this.ignoreCount_;
306  return copy;
307};
308
309
310ScriptBreakPoint.prototype.number = function() {
311  return this.number_;
312};
313
314
315ScriptBreakPoint.prototype.groupId = function() {
316  return this.groupId_;
317};
318
319
320ScriptBreakPoint.prototype.type = function() {
321  return this.type_;
322};
323
324
325ScriptBreakPoint.prototype.script_id = function() {
326  return this.script_id_;
327};
328
329
330ScriptBreakPoint.prototype.script_name = function() {
331  return this.script_name_;
332};
333
334
335ScriptBreakPoint.prototype.script_regexp_object = function() {
336  return this.script_regexp_object_;
337};
338
339
340ScriptBreakPoint.prototype.line = function() {
341  return this.line_;
342};
343
344
345ScriptBreakPoint.prototype.column = function() {
346  return this.column_;
347};
348
349
350ScriptBreakPoint.prototype.actual_locations = function() {
351  var locations = [];
352  for (var i = 0; i < this.break_points_.length; i++) {
353    locations.push(this.break_points_[i].actual_location);
354  }
355  return locations;
356};
357
358
359ScriptBreakPoint.prototype.update_positions = function(line, column) {
360  this.line_ = line;
361  this.column_ = column;
362};
363
364
365ScriptBreakPoint.prototype.hit_count = function() {
366  return this.hit_count_;
367};
368
369
370ScriptBreakPoint.prototype.active = function() {
371  return this.active_;
372};
373
374
375ScriptBreakPoint.prototype.condition = function() {
376  return this.condition_;
377};
378
379
380ScriptBreakPoint.prototype.ignoreCount = function() {
381  return this.ignoreCount_;
382};
383
384
385ScriptBreakPoint.prototype.enable = function() {
386  this.active_ = true;
387};
388
389
390ScriptBreakPoint.prototype.disable = function() {
391  this.active_ = false;
392};
393
394
395ScriptBreakPoint.prototype.setCondition = function(condition) {
396  this.condition_ = condition;
397};
398
399
400ScriptBreakPoint.prototype.setIgnoreCount = function(ignoreCount) {
401  this.ignoreCount_ = ignoreCount;
402
403  // Set ignore count on all break points created from this script break point.
404  for (var i = 0; i < this.break_points_.length; i++) {
405    this.break_points_[i].setIgnoreCount(ignoreCount);
406  }
407};
408
409
410// Check whether a script matches this script break point. Currently this is
411// only based on script name.
412ScriptBreakPoint.prototype.matchesScript = function(script) {
413  if (this.type_ == Debug.ScriptBreakPointType.ScriptId) {
414    return this.script_id_ == script.id;
415  } else {
416    // We might want to account columns here as well.
417    if (!(script.line_offset <= this.line_  &&
418          this.line_ < script.line_offset + script.lineCount())) {
419      return false;
420    }
421    if (this.type_ == Debug.ScriptBreakPointType.ScriptName) {
422      return this.script_name_ == script.nameOrSourceURL();
423    } else if (this.type_ == Debug.ScriptBreakPointType.ScriptRegExp) {
424      return this.script_regexp_object_.test(script.nameOrSourceURL());
425    } else {
426      throw MakeError(kDebugger, "Unexpected breakpoint type " + this.type_);
427    }
428  }
429};
430
431
432// Set the script break point in a script.
433ScriptBreakPoint.prototype.set = function (script) {
434  var column = this.column();
435  var line = this.line();
436  // If the column is undefined the break is on the line. To help locate the
437  // first piece of breakable code on the line try to find the column on the
438  // line which contains some source.
439  if (IS_UNDEFINED(column)) {
440    var source_line = script.sourceLine(this.line());
441
442    // Allocate array for caching the columns where the actual source starts.
443    if (!script.sourceColumnStart_) {
444      script.sourceColumnStart_ = new GlobalArray(script.lineCount());
445    }
446
447    // Fill cache if needed and get column where the actual source starts.
448    if (IS_UNDEFINED(script.sourceColumnStart_[line])) {
449      script.sourceColumnStart_[line] =
450          source_line.match(sourceLineBeginningSkip)[0].length;
451    }
452    column = script.sourceColumnStart_[line];
453  }
454
455  // Convert the line and column into an absolute position within the script.
456  var position = Debug.findScriptSourcePosition(script, this.line(), column);
457
458  // If the position is not found in the script (the script might be shorter
459  // than it used to be) just ignore it.
460  if (IS_NULL(position)) return;
461
462  // Create a break point object and set the break point.
463  var break_point = MakeBreakPoint(position, this);
464  break_point.setIgnoreCount(this.ignoreCount());
465  var actual_position = %SetScriptBreakPoint(script, position,
466                                             this.position_alignment_,
467                                             break_point);
468  if (IS_UNDEFINED(actual_position)) {
469    actual_position = position;
470  }
471  var actual_location = script.locationFromPosition(actual_position, true);
472  break_point.actual_location = { line: actual_location.line,
473                                  column: actual_location.column,
474                                  script_id: script.id };
475  this.break_points_.push(break_point);
476  return break_point;
477};
478
479
480// Clear all the break points created from this script break point
481ScriptBreakPoint.prototype.clear = function () {
482  var remaining_break_points = [];
483  for (var i = 0; i < break_points.length; i++) {
484    if (break_points[i].script_break_point() &&
485        break_points[i].script_break_point() === this) {
486      %ClearBreakPoint(break_points[i]);
487    } else {
488      remaining_break_points.push(break_points[i]);
489    }
490  }
491  break_points = remaining_break_points;
492  this.break_points_ = [];
493};
494
495
496// Function called from runtime when a new script is compiled to set any script
497// break points set in this script.
498function UpdateScriptBreakPoints(script) {
499  for (var i = 0; i < script_break_points.length; i++) {
500    var break_point = script_break_points[i];
501    if ((break_point.type() == Debug.ScriptBreakPointType.ScriptName ||
502         break_point.type() == Debug.ScriptBreakPointType.ScriptRegExp) &&
503        break_point.matchesScript(script)) {
504      break_point.set(script);
505    }
506  }
507}
508
509
510function GetScriptBreakPoints(script) {
511  var result = [];
512  for (var i = 0; i < script_break_points.length; i++) {
513    if (script_break_points[i].matchesScript(script)) {
514      result.push(script_break_points[i]);
515    }
516  }
517  return result;
518}
519
520
521Debug.setListener = function(listener, opt_data) {
522  if (!IS_FUNCTION(listener) && !IS_UNDEFINED(listener) && !IS_NULL(listener)) {
523    throw MakeTypeError(kDebuggerType);
524  }
525  %SetDebugEventListener(listener, opt_data);
526};
527
528
529Debug.breakLocations = function(f, opt_position_aligment) {
530  if (!IS_FUNCTION(f)) throw MakeTypeError(kDebuggerType);
531  var position_aligment = IS_UNDEFINED(opt_position_aligment)
532      ? Debug.BreakPositionAlignment.Statement : opt_position_aligment;
533  return %GetBreakLocations(f, position_aligment);
534};
535
536// Returns a Script object. If the parameter is a function the return value
537// is the script in which the function is defined. If the parameter is a string
538// the return value is the script for which the script name has that string
539// value.  If it is a regexp and there is a unique script whose name matches
540// we return that, otherwise undefined.
541Debug.findScript = function(func_or_script_name) {
542  if (IS_FUNCTION(func_or_script_name)) {
543    return %FunctionGetScript(func_or_script_name);
544  } else if (IS_REGEXP(func_or_script_name)) {
545    var scripts = Debug.scripts();
546    var last_result = null;
547    var result_count = 0;
548    for (var i in scripts) {
549      var script = scripts[i];
550      if (func_or_script_name.test(script.name)) {
551        last_result = script;
552        result_count++;
553      }
554    }
555    // Return the unique script matching the regexp.  If there are more
556    // than one we don't return a value since there is no good way to
557    // decide which one to return.  Returning a "random" one, say the
558    // first, would introduce nondeterminism (or something close to it)
559    // because the order is the heap iteration order.
560    if (result_count == 1) {
561      return last_result;
562    } else {
563      return UNDEFINED;
564    }
565  } else {
566    return %GetScript(func_or_script_name);
567  }
568};
569
570// Returns the script source. If the parameter is a function the return value
571// is the script source for the script in which the function is defined. If the
572// parameter is a string the return value is the script for which the script
573// name has that string value.
574Debug.scriptSource = function(func_or_script_name) {
575  return this.findScript(func_or_script_name).source;
576};
577
578
579Debug.source = function(f) {
580  if (!IS_FUNCTION(f)) throw MakeTypeError(kDebuggerType);
581  return %FunctionGetSourceCode(f);
582};
583
584
585Debug.sourcePosition = function(f) {
586  if (!IS_FUNCTION(f)) throw MakeTypeError(kDebuggerType);
587  return %FunctionGetScriptSourcePosition(f);
588};
589
590
591Debug.findFunctionSourceLocation = function(func, opt_line, opt_column) {
592  var script = %FunctionGetScript(func);
593  var script_offset = %FunctionGetScriptSourcePosition(func);
594  return script.locationFromLine(opt_line, opt_column, script_offset);
595};
596
597
598// Returns the character position in a script based on a line number and an
599// optional position within that line.
600Debug.findScriptSourcePosition = function(script, opt_line, opt_column) {
601  var location = script.locationFromLine(opt_line, opt_column);
602  return location ? location.position : null;
603};
604
605
606Debug.findBreakPoint = function(break_point_number, remove) {
607  var break_point;
608  for (var i = 0; i < break_points.length; i++) {
609    if (break_points[i].number() == break_point_number) {
610      break_point = break_points[i];
611      // Remove the break point from the list if requested.
612      if (remove) {
613        break_points.splice(i, 1);
614      }
615      break;
616    }
617  }
618  if (break_point) {
619    return break_point;
620  } else {
621    return this.findScriptBreakPoint(break_point_number, remove);
622  }
623};
624
625Debug.findBreakPointActualLocations = function(break_point_number) {
626  for (var i = 0; i < script_break_points.length; i++) {
627    if (script_break_points[i].number() == break_point_number) {
628      return script_break_points[i].actual_locations();
629    }
630  }
631  for (var i = 0; i < break_points.length; i++) {
632    if (break_points[i].number() == break_point_number) {
633      return [break_points[i].actual_location];
634    }
635  }
636  return [];
637};
638
639Debug.setBreakPoint = function(func, opt_line, opt_column, opt_condition) {
640  if (!IS_FUNCTION(func)) throw MakeTypeError(kDebuggerType);
641  // Break points in API functions are not supported.
642  if (%FunctionIsAPIFunction(func)) {
643    throw MakeError(kDebugger, 'Cannot set break point in native code.');
644  }
645  // Find source position relative to start of the function
646  var break_position =
647      this.findFunctionSourceLocation(func, opt_line, opt_column).position;
648  var source_position = break_position - this.sourcePosition(func);
649  // Find the script for the function.
650  var script = %FunctionGetScript(func);
651  // Break in builtin JavaScript code is not supported.
652  if (script.type == Debug.ScriptType.Native) {
653    throw MakeError(kDebugger, 'Cannot set break point in native code.');
654  }
655  // If the script for the function has a name convert this to a script break
656  // point.
657  if (script && script.id) {
658    // Adjust the source position to be script relative.
659    source_position += %FunctionGetScriptSourcePosition(func);
660    // Find line and column for the position in the script and set a script
661    // break point from that.
662    var location = script.locationFromPosition(source_position, false);
663    return this.setScriptBreakPointById(script.id,
664                                        location.line, location.column,
665                                        opt_condition);
666  } else {
667    // Set a break point directly on the function.
668    var break_point = MakeBreakPoint(source_position);
669    var actual_position =
670        %SetFunctionBreakPoint(func, source_position, break_point);
671    actual_position += this.sourcePosition(func);
672    var actual_location = script.locationFromPosition(actual_position, true);
673    break_point.actual_location = { line: actual_location.line,
674                                    column: actual_location.column,
675                                    script_id: script.id };
676    break_point.setCondition(opt_condition);
677    return break_point.number();
678  }
679};
680
681
682Debug.setBreakPointByScriptIdAndPosition = function(script_id, position,
683                                                    condition, enabled,
684                                                    opt_position_alignment)
685{
686  var break_point = MakeBreakPoint(position);
687  break_point.setCondition(condition);
688  if (!enabled) {
689    break_point.disable();
690  }
691  var scripts = this.scripts();
692  var position_alignment = IS_UNDEFINED(opt_position_alignment)
693      ? Debug.BreakPositionAlignment.Statement : opt_position_alignment;
694  for (var i = 0; i < scripts.length; i++) {
695    if (script_id == scripts[i].id) {
696      break_point.actual_position = %SetScriptBreakPoint(scripts[i], position,
697          position_alignment, break_point);
698      break;
699    }
700  }
701  return break_point;
702};
703
704
705Debug.enableBreakPoint = function(break_point_number) {
706  var break_point = this.findBreakPoint(break_point_number, false);
707  // Only enable if the breakpoint hasn't been deleted:
708  if (break_point) {
709    break_point.enable();
710  }
711};
712
713
714Debug.disableBreakPoint = function(break_point_number) {
715  var break_point = this.findBreakPoint(break_point_number, false);
716  // Only enable if the breakpoint hasn't been deleted:
717  if (break_point) {
718    break_point.disable();
719  }
720};
721
722
723Debug.changeBreakPointCondition = function(break_point_number, condition) {
724  var break_point = this.findBreakPoint(break_point_number, false);
725  break_point.setCondition(condition);
726};
727
728
729Debug.changeBreakPointIgnoreCount = function(break_point_number, ignoreCount) {
730  if (ignoreCount < 0) throw MakeError(kDebugger, 'Invalid argument');
731  var break_point = this.findBreakPoint(break_point_number, false);
732  break_point.setIgnoreCount(ignoreCount);
733};
734
735
736Debug.clearBreakPoint = function(break_point_number) {
737  var break_point = this.findBreakPoint(break_point_number, true);
738  if (break_point) {
739    return %ClearBreakPoint(break_point);
740  } else {
741    break_point = this.findScriptBreakPoint(break_point_number, true);
742    if (!break_point) throw MakeError(kDebugger, 'Invalid breakpoint');
743  }
744};
745
746
747Debug.clearAllBreakPoints = function() {
748  for (var i = 0; i < break_points.length; i++) {
749    var break_point = break_points[i];
750    %ClearBreakPoint(break_point);
751  }
752  break_points = [];
753};
754
755
756Debug.disableAllBreakPoints = function() {
757  // Disable all user defined breakpoints:
758  for (var i = 1; i < next_break_point_number; i++) {
759    Debug.disableBreakPoint(i);
760  }
761  // Disable all exception breakpoints:
762  %ChangeBreakOnException(Debug.ExceptionBreak.Caught, false);
763  %ChangeBreakOnException(Debug.ExceptionBreak.Uncaught, false);
764};
765
766
767Debug.findScriptBreakPoint = function(break_point_number, remove) {
768  var script_break_point;
769  for (var i = 0; i < script_break_points.length; i++) {
770    if (script_break_points[i].number() == break_point_number) {
771      script_break_point = script_break_points[i];
772      // Remove the break point from the list if requested.
773      if (remove) {
774        script_break_point.clear();
775        script_break_points.splice(i,1);
776      }
777      break;
778    }
779  }
780  return script_break_point;
781};
782
783
784// Sets a breakpoint in a script identified through id or name at the
785// specified source line and column within that line.
786Debug.setScriptBreakPoint = function(type, script_id_or_name,
787                                     opt_line, opt_column, opt_condition,
788                                     opt_groupId, opt_position_alignment) {
789  // Create script break point object.
790  var script_break_point =
791      new ScriptBreakPoint(type, script_id_or_name, opt_line, opt_column,
792                           opt_groupId, opt_position_alignment);
793
794  // Assign number to the new script break point and add it.
795  script_break_point.number_ = next_break_point_number++;
796  script_break_point.setCondition(opt_condition);
797  script_break_points.push(script_break_point);
798
799  // Run through all scripts to see if this script break point matches any
800  // loaded scripts.
801  var scripts = this.scripts();
802  for (var i = 0; i < scripts.length; i++) {
803    if (script_break_point.matchesScript(scripts[i])) {
804      script_break_point.set(scripts[i]);
805    }
806  }
807
808  return script_break_point.number();
809};
810
811
812Debug.setScriptBreakPointById = function(script_id,
813                                         opt_line, opt_column,
814                                         opt_condition, opt_groupId,
815                                         opt_position_alignment) {
816  return this.setScriptBreakPoint(Debug.ScriptBreakPointType.ScriptId,
817                                  script_id, opt_line, opt_column,
818                                  opt_condition, opt_groupId,
819                                  opt_position_alignment);
820};
821
822
823Debug.setScriptBreakPointByName = function(script_name,
824                                           opt_line, opt_column,
825                                           opt_condition, opt_groupId) {
826  return this.setScriptBreakPoint(Debug.ScriptBreakPointType.ScriptName,
827                                  script_name, opt_line, opt_column,
828                                  opt_condition, opt_groupId);
829};
830
831
832Debug.setScriptBreakPointByRegExp = function(script_regexp,
833                                             opt_line, opt_column,
834                                             opt_condition, opt_groupId) {
835  return this.setScriptBreakPoint(Debug.ScriptBreakPointType.ScriptRegExp,
836                                  script_regexp, opt_line, opt_column,
837                                  opt_condition, opt_groupId);
838};
839
840
841Debug.enableScriptBreakPoint = function(break_point_number) {
842  var script_break_point = this.findScriptBreakPoint(break_point_number, false);
843  script_break_point.enable();
844};
845
846
847Debug.disableScriptBreakPoint = function(break_point_number) {
848  var script_break_point = this.findScriptBreakPoint(break_point_number, false);
849  script_break_point.disable();
850};
851
852
853Debug.changeScriptBreakPointCondition = function(
854    break_point_number, condition) {
855  var script_break_point = this.findScriptBreakPoint(break_point_number, false);
856  script_break_point.setCondition(condition);
857};
858
859
860Debug.changeScriptBreakPointIgnoreCount = function(
861    break_point_number, ignoreCount) {
862  if (ignoreCount < 0) throw MakeError(kDebugger, 'Invalid argument');
863  var script_break_point = this.findScriptBreakPoint(break_point_number, false);
864  script_break_point.setIgnoreCount(ignoreCount);
865};
866
867
868Debug.scriptBreakPoints = function() {
869  return script_break_points;
870};
871
872
873Debug.clearStepping = function() {
874  %ClearStepping();
875};
876
877Debug.setBreakOnException = function() {
878  return %ChangeBreakOnException(Debug.ExceptionBreak.Caught, true);
879};
880
881Debug.clearBreakOnException = function() {
882  return %ChangeBreakOnException(Debug.ExceptionBreak.Caught, false);
883};
884
885Debug.isBreakOnException = function() {
886  return !!%IsBreakOnException(Debug.ExceptionBreak.Caught);
887};
888
889Debug.setBreakOnUncaughtException = function() {
890  return %ChangeBreakOnException(Debug.ExceptionBreak.Uncaught, true);
891};
892
893Debug.clearBreakOnUncaughtException = function() {
894  return %ChangeBreakOnException(Debug.ExceptionBreak.Uncaught, false);
895};
896
897Debug.isBreakOnUncaughtException = function() {
898  return !!%IsBreakOnException(Debug.ExceptionBreak.Uncaught);
899};
900
901Debug.showBreakPoints = function(f, full, opt_position_alignment) {
902  if (!IS_FUNCTION(f)) throw MakeError(kDebuggerType);
903  var source = full ? this.scriptSource(f) : this.source(f);
904  var offset = full ? this.sourcePosition(f) : 0;
905  var locations = this.breakLocations(f, opt_position_alignment);
906  if (!locations) return source;
907  locations.sort(function(x, y) { return x - y; });
908  var result = "";
909  var prev_pos = 0;
910  var pos;
911  for (var i = 0; i < locations.length; i++) {
912    pos = locations[i] - offset;
913    result += source.slice(prev_pos, pos);
914    result += "[B" + i + "]";
915    prev_pos = pos;
916  }
917  pos = source.length;
918  result += source.substring(prev_pos, pos);
919  return result;
920};
921
922
923// Get all the scripts currently loaded. Locating all the scripts is based on
924// scanning the heap.
925Debug.scripts = function() {
926  // Collect all scripts in the heap.
927  return %DebugGetLoadedScripts();
928};
929
930
931Debug.debuggerFlags = function() {
932  return debugger_flags;
933};
934
935Debug.MakeMirror = MakeMirror;
936
937function MakeExecutionState(break_id) {
938  return new ExecutionState(break_id);
939}
940
941function ExecutionState(break_id) {
942  this.break_id = break_id;
943  this.selected_frame = 0;
944}
945
946ExecutionState.prototype.prepareStep = function(action) {
947  if (action === Debug.StepAction.StepIn ||
948      action === Debug.StepAction.StepOut ||
949      action === Debug.StepAction.StepNext ||
950      action === Debug.StepAction.StepFrame) {
951    return %PrepareStep(this.break_id, action);
952  }
953  throw MakeTypeError(kDebuggerType);
954};
955
956ExecutionState.prototype.evaluateGlobal = function(source, disable_break,
957    opt_additional_context) {
958  return MakeMirror(%DebugEvaluateGlobal(this.break_id, source,
959                                         TO_BOOLEAN(disable_break),
960                                         opt_additional_context));
961};
962
963ExecutionState.prototype.frameCount = function() {
964  return %GetFrameCount(this.break_id);
965};
966
967ExecutionState.prototype.threadCount = function() {
968  return %GetThreadCount(this.break_id);
969};
970
971ExecutionState.prototype.frame = function(opt_index) {
972  // If no index supplied return the selected frame.
973  if (opt_index == null) opt_index = this.selected_frame;
974  if (opt_index < 0 || opt_index >= this.frameCount()) {
975    throw MakeTypeError(kDebuggerFrame);
976  }
977  return new FrameMirror(this.break_id, opt_index);
978};
979
980ExecutionState.prototype.setSelectedFrame = function(index) {
981  var i = TO_NUMBER(index);
982  if (i < 0 || i >= this.frameCount()) {
983    throw MakeTypeError(kDebuggerFrame);
984  }
985  this.selected_frame = i;
986};
987
988ExecutionState.prototype.selectedFrame = function() {
989  return this.selected_frame;
990};
991
992ExecutionState.prototype.debugCommandProcessor = function(opt_is_running) {
993  return new DebugCommandProcessor(this, opt_is_running);
994};
995
996
997function MakeBreakEvent(break_id, break_points_hit) {
998  return new BreakEvent(break_id, break_points_hit);
999}
1000
1001
1002function BreakEvent(break_id, break_points_hit) {
1003  this.frame_ = new FrameMirror(break_id, 0);
1004  this.break_points_hit_ = break_points_hit;
1005}
1006
1007
1008BreakEvent.prototype.eventType = function() {
1009  return Debug.DebugEvent.Break;
1010};
1011
1012
1013BreakEvent.prototype.func = function() {
1014  return this.frame_.func();
1015};
1016
1017
1018BreakEvent.prototype.sourceLine = function() {
1019  return this.frame_.sourceLine();
1020};
1021
1022
1023BreakEvent.prototype.sourceColumn = function() {
1024  return this.frame_.sourceColumn();
1025};
1026
1027
1028BreakEvent.prototype.sourceLineText = function() {
1029  return this.frame_.sourceLineText();
1030};
1031
1032
1033BreakEvent.prototype.breakPointsHit = function() {
1034  return this.break_points_hit_;
1035};
1036
1037
1038BreakEvent.prototype.toJSONProtocol = function() {
1039  var o = { seq: next_response_seq++,
1040            type: "event",
1041            event: "break",
1042            body: { invocationText: this.frame_.invocationText() }
1043          };
1044
1045  // Add script related information to the event if available.
1046  var script = this.func().script();
1047  if (script) {
1048    o.body.sourceLine = this.sourceLine(),
1049    o.body.sourceColumn = this.sourceColumn(),
1050    o.body.sourceLineText = this.sourceLineText(),
1051    o.body.script = MakeScriptObject_(script, false);
1052  }
1053
1054  // Add an Array of break points hit if any.
1055  if (this.breakPointsHit()) {
1056    o.body.breakpoints = [];
1057    for (var i = 0; i < this.breakPointsHit().length; i++) {
1058      // Find the break point number. For break points originating from a
1059      // script break point supply the script break point number.
1060      var breakpoint = this.breakPointsHit()[i];
1061      var script_break_point = breakpoint.script_break_point();
1062      var number;
1063      if (script_break_point) {
1064        number = script_break_point.number();
1065      } else {
1066        number = breakpoint.number();
1067      }
1068      o.body.breakpoints.push(number);
1069    }
1070  }
1071  return JSONStringify(ObjectToProtocolObject_(o));
1072};
1073
1074
1075function MakeExceptionEvent(break_id, exception, uncaught, promise) {
1076  return new ExceptionEvent(break_id, exception, uncaught, promise);
1077}
1078
1079
1080function ExceptionEvent(break_id, exception, uncaught, promise) {
1081  this.exec_state_ = new ExecutionState(break_id);
1082  this.exception_ = exception;
1083  this.uncaught_ = uncaught;
1084  this.promise_ = promise;
1085}
1086
1087
1088ExceptionEvent.prototype.eventType = function() {
1089  return Debug.DebugEvent.Exception;
1090};
1091
1092
1093ExceptionEvent.prototype.exception = function() {
1094  return this.exception_;
1095};
1096
1097
1098ExceptionEvent.prototype.uncaught = function() {
1099  return this.uncaught_;
1100};
1101
1102
1103ExceptionEvent.prototype.promise = function() {
1104  return this.promise_;
1105};
1106
1107
1108ExceptionEvent.prototype.func = function() {
1109  return this.exec_state_.frame(0).func();
1110};
1111
1112
1113ExceptionEvent.prototype.sourceLine = function() {
1114  return this.exec_state_.frame(0).sourceLine();
1115};
1116
1117
1118ExceptionEvent.prototype.sourceColumn = function() {
1119  return this.exec_state_.frame(0).sourceColumn();
1120};
1121
1122
1123ExceptionEvent.prototype.sourceLineText = function() {
1124  return this.exec_state_.frame(0).sourceLineText();
1125};
1126
1127
1128ExceptionEvent.prototype.toJSONProtocol = function() {
1129  var o = new ProtocolMessage();
1130  o.event = "exception";
1131  o.body = { uncaught: this.uncaught_,
1132             exception: MakeMirror(this.exception_)
1133           };
1134
1135  // Exceptions might happen whithout any JavaScript frames.
1136  if (this.exec_state_.frameCount() > 0) {
1137    o.body.sourceLine = this.sourceLine();
1138    o.body.sourceColumn = this.sourceColumn();
1139    o.body.sourceLineText = this.sourceLineText();
1140
1141    // Add script information to the event if available.
1142    var script = this.func().script();
1143    if (script) {
1144      o.body.script = MakeScriptObject_(script, false);
1145    }
1146  } else {
1147    o.body.sourceLine = -1;
1148  }
1149
1150  return o.toJSONProtocol();
1151};
1152
1153
1154function MakeCompileEvent(script, type) {
1155  return new CompileEvent(script, type);
1156}
1157
1158
1159function CompileEvent(script, type) {
1160  this.script_ = MakeMirror(script);
1161  this.type_ = type;
1162}
1163
1164
1165CompileEvent.prototype.eventType = function() {
1166  return this.type_;
1167};
1168
1169
1170CompileEvent.prototype.script = function() {
1171  return this.script_;
1172};
1173
1174
1175CompileEvent.prototype.toJSONProtocol = function() {
1176  var o = new ProtocolMessage();
1177  o.running = true;
1178  switch (this.type_) {
1179    case Debug.DebugEvent.BeforeCompile:
1180      o.event = "beforeCompile";
1181      break;
1182    case Debug.DebugEvent.AfterCompile:
1183      o.event = "afterCompile";
1184      break;
1185    case Debug.DebugEvent.CompileError:
1186      o.event = "compileError";
1187      break;
1188  }
1189  o.body = {};
1190  o.body.script = this.script_;
1191
1192  return o.toJSONProtocol();
1193};
1194
1195
1196function MakeScriptObject_(script, include_source) {
1197  var o = { id: script.id(),
1198            name: script.name(),
1199            lineOffset: script.lineOffset(),
1200            columnOffset: script.columnOffset(),
1201            lineCount: script.lineCount(),
1202          };
1203  if (!IS_UNDEFINED(script.data())) {
1204    o.data = script.data();
1205  }
1206  if (include_source) {
1207    o.source = script.source();
1208  }
1209  return o;
1210}
1211
1212
1213function MakePromiseEvent(event_data) {
1214  return new PromiseEvent(event_data);
1215}
1216
1217
1218function PromiseEvent(event_data) {
1219  this.promise_ = event_data.promise;
1220  this.parentPromise_ = event_data.parentPromise;
1221  this.status_ = event_data.status;
1222  this.value_ = event_data.value;
1223}
1224
1225
1226PromiseEvent.prototype.promise = function() {
1227  return MakeMirror(this.promise_);
1228}
1229
1230
1231PromiseEvent.prototype.parentPromise = function() {
1232  return MakeMirror(this.parentPromise_);
1233}
1234
1235
1236PromiseEvent.prototype.status = function() {
1237  return this.status_;
1238}
1239
1240
1241PromiseEvent.prototype.value = function() {
1242  return MakeMirror(this.value_);
1243}
1244
1245
1246function MakeAsyncTaskEvent(event_data) {
1247  return new AsyncTaskEvent(event_data);
1248}
1249
1250
1251function AsyncTaskEvent(event_data) {
1252  this.type_ = event_data.type;
1253  this.name_ = event_data.name;
1254  this.id_ = event_data.id;
1255}
1256
1257
1258AsyncTaskEvent.prototype.type = function() {
1259  return this.type_;
1260}
1261
1262
1263AsyncTaskEvent.prototype.name = function() {
1264  return this.name_;
1265}
1266
1267
1268AsyncTaskEvent.prototype.id = function() {
1269  return this.id_;
1270}
1271
1272
1273function DebugCommandProcessor(exec_state, opt_is_running) {
1274  this.exec_state_ = exec_state;
1275  this.running_ = opt_is_running || false;
1276}
1277
1278
1279DebugCommandProcessor.prototype.processDebugRequest = function (request) {
1280  return this.processDebugJSONRequest(request);
1281};
1282
1283
1284function ProtocolMessage(request) {
1285  // Update sequence number.
1286  this.seq = next_response_seq++;
1287
1288  if (request) {
1289    // If message is based on a request this is a response. Fill the initial
1290    // response from the request.
1291    this.type = 'response';
1292    this.request_seq = request.seq;
1293    this.command = request.command;
1294  } else {
1295    // If message is not based on a request it is a dabugger generated event.
1296    this.type = 'event';
1297  }
1298  this.success = true;
1299  // Handler may set this field to control debugger state.
1300  this.running = UNDEFINED;
1301}
1302
1303
1304ProtocolMessage.prototype.setOption = function(name, value) {
1305  if (!this.options_) {
1306    this.options_ = {};
1307  }
1308  this.options_[name] = value;
1309};
1310
1311
1312ProtocolMessage.prototype.failed = function(message, opt_details) {
1313  this.success = false;
1314  this.message = message;
1315  if (IS_OBJECT(opt_details)) {
1316    this.error_details = opt_details;
1317  }
1318};
1319
1320
1321ProtocolMessage.prototype.toJSONProtocol = function() {
1322  // Encode the protocol header.
1323  var json = {};
1324  json.seq= this.seq;
1325  if (this.request_seq) {
1326    json.request_seq = this.request_seq;
1327  }
1328  json.type = this.type;
1329  if (this.event) {
1330    json.event = this.event;
1331  }
1332  if (this.command) {
1333    json.command = this.command;
1334  }
1335  if (this.success) {
1336    json.success = this.success;
1337  } else {
1338    json.success = false;
1339  }
1340  if (this.body) {
1341    // Encode the body part.
1342    var bodyJson;
1343    var serializer = MakeMirrorSerializer(true, this.options_);
1344    if (this.body instanceof Mirror) {
1345      bodyJson = serializer.serializeValue(this.body);
1346    } else if (this.body instanceof GlobalArray) {
1347      bodyJson = [];
1348      for (var i = 0; i < this.body.length; i++) {
1349        if (this.body[i] instanceof Mirror) {
1350          bodyJson.push(serializer.serializeValue(this.body[i]));
1351        } else {
1352          bodyJson.push(ObjectToProtocolObject_(this.body[i], serializer));
1353        }
1354      }
1355    } else {
1356      bodyJson = ObjectToProtocolObject_(this.body, serializer);
1357    }
1358    json.body = bodyJson;
1359    json.refs = serializer.serializeReferencedObjects();
1360  }
1361  if (this.message) {
1362    json.message = this.message;
1363  }
1364  if (this.error_details) {
1365    json.error_details = this.error_details;
1366  }
1367  json.running = this.running;
1368  return JSONStringify(json);
1369};
1370
1371
1372DebugCommandProcessor.prototype.createResponse = function(request) {
1373  return new ProtocolMessage(request);
1374};
1375
1376
1377DebugCommandProcessor.prototype.processDebugJSONRequest = function(
1378    json_request) {
1379  var request;  // Current request.
1380  var response;  // Generated response.
1381  try {
1382    try {
1383      // Convert the JSON string to an object.
1384      request = JSONParse(json_request);
1385
1386      // Create an initial response.
1387      response = this.createResponse(request);
1388
1389      if (!request.type) {
1390        throw MakeError(kDebugger, 'Type not specified');
1391      }
1392
1393      if (request.type != 'request') {
1394        throw MakeError(kDebugger,
1395                        "Illegal type '" + request.type + "' in request");
1396      }
1397
1398      if (!request.command) {
1399        throw MakeError(kDebugger, 'Command not specified');
1400      }
1401
1402      if (request.arguments) {
1403        var args = request.arguments;
1404        // TODO(yurys): remove request.arguments.compactFormat check once
1405        // ChromeDevTools are switched to 'inlineRefs'
1406        if (args.inlineRefs || args.compactFormat) {
1407          response.setOption('inlineRefs', true);
1408        }
1409        if (!IS_UNDEFINED(args.maxStringLength)) {
1410          response.setOption('maxStringLength', args.maxStringLength);
1411        }
1412      }
1413
1414      var key = request.command.toLowerCase();
1415      var handler = DebugCommandProcessor.prototype.dispatch_[key];
1416      if (IS_FUNCTION(handler)) {
1417        %_Call(handler, this, request, response);
1418      } else {
1419        throw MakeError(kDebugger,
1420                        'Unknown command "' + request.command + '" in request');
1421      }
1422    } catch (e) {
1423      // If there is no response object created one (without command).
1424      if (!response) {
1425        response = this.createResponse();
1426      }
1427      response.success = false;
1428      response.message = TO_STRING(e);
1429    }
1430
1431    // Return the response as a JSON encoded string.
1432    try {
1433      if (!IS_UNDEFINED(response.running)) {
1434        // Response controls running state.
1435        this.running_ = response.running;
1436      }
1437      response.running = this.running_;
1438      return response.toJSONProtocol();
1439    } catch (e) {
1440      // Failed to generate response - return generic error.
1441      return '{"seq":' + response.seq + ',' +
1442              '"request_seq":' + request.seq + ',' +
1443              '"type":"response",' +
1444              '"success":false,' +
1445              '"message":"Internal error: ' + TO_STRING(e) + '"}';
1446    }
1447  } catch (e) {
1448    // Failed in one of the catch blocks above - most generic error.
1449    return '{"seq":0,"type":"response","success":false,"message":"Internal error"}';
1450  }
1451};
1452
1453
1454DebugCommandProcessor.prototype.continueRequest_ = function(request, response) {
1455  // Check for arguments for continue.
1456  if (request.arguments) {
1457    var action = Debug.StepAction.StepIn;
1458
1459    // Pull out arguments.
1460    var stepaction = request.arguments.stepaction;
1461
1462    // Get the stepaction argument.
1463    if (stepaction) {
1464      if (stepaction == 'in') {
1465        action = Debug.StepAction.StepIn;
1466      } else if (stepaction == 'next') {
1467        action = Debug.StepAction.StepNext;
1468      } else if (stepaction == 'out') {
1469        action = Debug.StepAction.StepOut;
1470      } else {
1471        throw MakeError(kDebugger,
1472                        'Invalid stepaction argument "' + stepaction + '".');
1473      }
1474    }
1475
1476    // Set up the VM for stepping.
1477    this.exec_state_.prepareStep(action);
1478  }
1479
1480  // VM should be running after executing this request.
1481  response.running = true;
1482};
1483
1484
1485DebugCommandProcessor.prototype.breakRequest_ = function(request, response) {
1486  // Ignore as break command does not do anything when broken.
1487};
1488
1489
1490DebugCommandProcessor.prototype.setBreakPointRequest_ =
1491    function(request, response) {
1492  // Check for legal request.
1493  if (!request.arguments) {
1494    response.failed('Missing arguments');
1495    return;
1496  }
1497
1498  // Pull out arguments.
1499  var type = request.arguments.type;
1500  var target = request.arguments.target;
1501  var line = request.arguments.line;
1502  var column = request.arguments.column;
1503  var enabled = IS_UNDEFINED(request.arguments.enabled) ?
1504      true : request.arguments.enabled;
1505  var condition = request.arguments.condition;
1506  var ignoreCount = request.arguments.ignoreCount;
1507  var groupId = request.arguments.groupId;
1508
1509  // Check for legal arguments.
1510  if (!type || IS_UNDEFINED(target)) {
1511    response.failed('Missing argument "type" or "target"');
1512    return;
1513  }
1514
1515  // Either function or script break point.
1516  var break_point_number;
1517  if (type == 'function') {
1518    // Handle function break point.
1519    if (!IS_STRING(target)) {
1520      response.failed('Argument "target" is not a string value');
1521      return;
1522    }
1523    var f;
1524    try {
1525      // Find the function through a global evaluate.
1526      f = this.exec_state_.evaluateGlobal(target).value();
1527    } catch (e) {
1528      response.failed('Error: "' + TO_STRING(e) +
1529                      '" evaluating "' + target + '"');
1530      return;
1531    }
1532    if (!IS_FUNCTION(f)) {
1533      response.failed('"' + target + '" does not evaluate to a function');
1534      return;
1535    }
1536
1537    // Set function break point.
1538    break_point_number = Debug.setBreakPoint(f, line, column, condition);
1539  } else if (type == 'handle') {
1540    // Find the object pointed by the specified handle.
1541    var handle = ParseInt(target, 10);
1542    var mirror = LookupMirror(handle);
1543    if (!mirror) {
1544      return response.failed('Object #' + handle + '# not found');
1545    }
1546    if (!mirror.isFunction()) {
1547      return response.failed('Object #' + handle + '# is not a function');
1548    }
1549
1550    // Set function break point.
1551    break_point_number = Debug.setBreakPoint(mirror.value(),
1552                                             line, column, condition);
1553  } else if (type == 'script') {
1554    // set script break point.
1555    break_point_number =
1556        Debug.setScriptBreakPointByName(target, line, column, condition,
1557                                        groupId);
1558  } else if (type == 'scriptId') {
1559    break_point_number =
1560        Debug.setScriptBreakPointById(target, line, column, condition, groupId);
1561  } else if (type == 'scriptRegExp') {
1562    break_point_number =
1563        Debug.setScriptBreakPointByRegExp(target, line, column, condition,
1564                                          groupId);
1565  } else {
1566    response.failed('Illegal type "' + type + '"');
1567    return;
1568  }
1569
1570  // Set additional break point properties.
1571  var break_point = Debug.findBreakPoint(break_point_number);
1572  if (ignoreCount) {
1573    Debug.changeBreakPointIgnoreCount(break_point_number, ignoreCount);
1574  }
1575  if (!enabled) {
1576    Debug.disableBreakPoint(break_point_number);
1577  }
1578
1579  // Add the break point number to the response.
1580  response.body = { type: type,
1581                    breakpoint: break_point_number };
1582
1583  // Add break point information to the response.
1584  if (break_point instanceof ScriptBreakPoint) {
1585    if (break_point.type() == Debug.ScriptBreakPointType.ScriptId) {
1586      response.body.type = 'scriptId';
1587      response.body.script_id = break_point.script_id();
1588    } else if (break_point.type() == Debug.ScriptBreakPointType.ScriptName) {
1589      response.body.type = 'scriptName';
1590      response.body.script_name = break_point.script_name();
1591    } else if (break_point.type() == Debug.ScriptBreakPointType.ScriptRegExp) {
1592      response.body.type = 'scriptRegExp';
1593      response.body.script_regexp = break_point.script_regexp_object().source;
1594    } else {
1595      throw MakeError(kDebugger,
1596                      "Unexpected breakpoint type: " + break_point.type());
1597    }
1598    response.body.line = break_point.line();
1599    response.body.column = break_point.column();
1600    response.body.actual_locations = break_point.actual_locations();
1601  } else {
1602    response.body.type = 'function';
1603    response.body.actual_locations = [break_point.actual_location];
1604  }
1605};
1606
1607
1608DebugCommandProcessor.prototype.changeBreakPointRequest_ = function(
1609    request, response) {
1610  // Check for legal request.
1611  if (!request.arguments) {
1612    response.failed('Missing arguments');
1613    return;
1614  }
1615
1616  // Pull out arguments.
1617  var break_point = TO_NUMBER(request.arguments.breakpoint);
1618  var enabled = request.arguments.enabled;
1619  var condition = request.arguments.condition;
1620  var ignoreCount = request.arguments.ignoreCount;
1621
1622  // Check for legal arguments.
1623  if (!break_point) {
1624    response.failed('Missing argument "breakpoint"');
1625    return;
1626  }
1627
1628  // Change enabled state if supplied.
1629  if (!IS_UNDEFINED(enabled)) {
1630    if (enabled) {
1631      Debug.enableBreakPoint(break_point);
1632    } else {
1633      Debug.disableBreakPoint(break_point);
1634    }
1635  }
1636
1637  // Change condition if supplied
1638  if (!IS_UNDEFINED(condition)) {
1639    Debug.changeBreakPointCondition(break_point, condition);
1640  }
1641
1642  // Change ignore count if supplied
1643  if (!IS_UNDEFINED(ignoreCount)) {
1644    Debug.changeBreakPointIgnoreCount(break_point, ignoreCount);
1645  }
1646};
1647
1648
1649DebugCommandProcessor.prototype.clearBreakPointGroupRequest_ = function(
1650    request, response) {
1651  // Check for legal request.
1652  if (!request.arguments) {
1653    response.failed('Missing arguments');
1654    return;
1655  }
1656
1657  // Pull out arguments.
1658  var group_id = request.arguments.groupId;
1659
1660  // Check for legal arguments.
1661  if (!group_id) {
1662    response.failed('Missing argument "groupId"');
1663    return;
1664  }
1665
1666  var cleared_break_points = [];
1667  var new_script_break_points = [];
1668  for (var i = 0; i < script_break_points.length; i++) {
1669    var next_break_point = script_break_points[i];
1670    if (next_break_point.groupId() == group_id) {
1671      cleared_break_points.push(next_break_point.number());
1672      next_break_point.clear();
1673    } else {
1674      new_script_break_points.push(next_break_point);
1675    }
1676  }
1677  script_break_points = new_script_break_points;
1678
1679  // Add the cleared break point numbers to the response.
1680  response.body = { breakpoints: cleared_break_points };
1681};
1682
1683
1684DebugCommandProcessor.prototype.clearBreakPointRequest_ = function(
1685    request, response) {
1686  // Check for legal request.
1687  if (!request.arguments) {
1688    response.failed('Missing arguments');
1689    return;
1690  }
1691
1692  // Pull out arguments.
1693  var break_point = TO_NUMBER(request.arguments.breakpoint);
1694
1695  // Check for legal arguments.
1696  if (!break_point) {
1697    response.failed('Missing argument "breakpoint"');
1698    return;
1699  }
1700
1701  // Clear break point.
1702  Debug.clearBreakPoint(break_point);
1703
1704  // Add the cleared break point number to the response.
1705  response.body = { breakpoint: break_point };
1706};
1707
1708
1709DebugCommandProcessor.prototype.listBreakpointsRequest_ = function(
1710    request, response) {
1711  var array = [];
1712  for (var i = 0; i < script_break_points.length; i++) {
1713    var break_point = script_break_points[i];
1714
1715    var description = {
1716      number: break_point.number(),
1717      line: break_point.line(),
1718      column: break_point.column(),
1719      groupId: break_point.groupId(),
1720      hit_count: break_point.hit_count(),
1721      active: break_point.active(),
1722      condition: break_point.condition(),
1723      ignoreCount: break_point.ignoreCount(),
1724      actual_locations: break_point.actual_locations()
1725    };
1726
1727    if (break_point.type() == Debug.ScriptBreakPointType.ScriptId) {
1728      description.type = 'scriptId';
1729      description.script_id = break_point.script_id();
1730    } else if (break_point.type() == Debug.ScriptBreakPointType.ScriptName) {
1731      description.type = 'scriptName';
1732      description.script_name = break_point.script_name();
1733    } else if (break_point.type() == Debug.ScriptBreakPointType.ScriptRegExp) {
1734      description.type = 'scriptRegExp';
1735      description.script_regexp = break_point.script_regexp_object().source;
1736    } else {
1737      throw MakeError(kDebugger,
1738                      "Unexpected breakpoint type: " + break_point.type());
1739    }
1740    array.push(description);
1741  }
1742
1743  response.body = {
1744    breakpoints: array,
1745    breakOnExceptions: Debug.isBreakOnException(),
1746    breakOnUncaughtExceptions: Debug.isBreakOnUncaughtException()
1747  };
1748};
1749
1750
1751DebugCommandProcessor.prototype.disconnectRequest_ =
1752    function(request, response) {
1753  Debug.disableAllBreakPoints();
1754  this.continueRequest_(request, response);
1755};
1756
1757
1758DebugCommandProcessor.prototype.setExceptionBreakRequest_ =
1759    function(request, response) {
1760  // Check for legal request.
1761  if (!request.arguments) {
1762    response.failed('Missing arguments');
1763    return;
1764  }
1765
1766  // Pull out and check the 'type' argument:
1767  var type = request.arguments.type;
1768  if (!type) {
1769    response.failed('Missing argument "type"');
1770    return;
1771  }
1772
1773  // Initialize the default value of enable:
1774  var enabled;
1775  if (type == 'all') {
1776    enabled = !Debug.isBreakOnException();
1777  } else if (type == 'uncaught') {
1778    enabled = !Debug.isBreakOnUncaughtException();
1779  }
1780
1781  // Pull out and check the 'enabled' argument if present:
1782  if (!IS_UNDEFINED(request.arguments.enabled)) {
1783    enabled = request.arguments.enabled;
1784    if ((enabled != true) && (enabled != false)) {
1785      response.failed('Illegal value for "enabled":"' + enabled + '"');
1786    }
1787  }
1788
1789  // Now set the exception break state:
1790  if (type == 'all') {
1791    %ChangeBreakOnException(Debug.ExceptionBreak.Caught, enabled);
1792  } else if (type == 'uncaught') {
1793    %ChangeBreakOnException(Debug.ExceptionBreak.Uncaught, enabled);
1794  } else {
1795    response.failed('Unknown "type":"' + type + '"');
1796  }
1797
1798  // Add the cleared break point number to the response.
1799  response.body = { 'type': type, 'enabled': enabled };
1800};
1801
1802
1803DebugCommandProcessor.prototype.backtraceRequest_ = function(
1804    request, response) {
1805  // Get the number of frames.
1806  var total_frames = this.exec_state_.frameCount();
1807
1808  // Create simple response if there are no frames.
1809  if (total_frames == 0) {
1810    response.body = {
1811      totalFrames: total_frames
1812    };
1813    return;
1814  }
1815
1816  // Default frame range to include in backtrace.
1817  var from_index = 0;
1818  var to_index = kDefaultBacktraceLength;
1819
1820  // Get the range from the arguments.
1821  if (request.arguments) {
1822    if (request.arguments.fromFrame) {
1823      from_index = request.arguments.fromFrame;
1824    }
1825    if (request.arguments.toFrame) {
1826      to_index = request.arguments.toFrame;
1827    }
1828    if (request.arguments.bottom) {
1829      var tmp_index = total_frames - from_index;
1830      from_index = total_frames - to_index;
1831      to_index = tmp_index;
1832    }
1833    if (from_index < 0 || to_index < 0) {
1834      return response.failed('Invalid frame number');
1835    }
1836  }
1837
1838  // Adjust the index.
1839  to_index = MathMin(total_frames, to_index);
1840
1841  if (to_index <= from_index) {
1842    var error = 'Invalid frame range';
1843    return response.failed(error);
1844  }
1845
1846  // Create the response body.
1847  var frames = [];
1848  for (var i = from_index; i < to_index; i++) {
1849    frames.push(this.exec_state_.frame(i));
1850  }
1851  response.body = {
1852    fromFrame: from_index,
1853    toFrame: to_index,
1854    totalFrames: total_frames,
1855    frames: frames
1856  };
1857};
1858
1859
1860DebugCommandProcessor.prototype.frameRequest_ = function(request, response) {
1861  // No frames no source.
1862  if (this.exec_state_.frameCount() == 0) {
1863    return response.failed('No frames');
1864  }
1865
1866  // With no arguments just keep the selected frame.
1867  if (request.arguments) {
1868    var index = request.arguments.number;
1869    if (index < 0 || this.exec_state_.frameCount() <= index) {
1870      return response.failed('Invalid frame number');
1871    }
1872
1873    this.exec_state_.setSelectedFrame(request.arguments.number);
1874  }
1875  response.body = this.exec_state_.frame();
1876};
1877
1878
1879DebugCommandProcessor.prototype.resolveFrameFromScopeDescription_ =
1880    function(scope_description) {
1881  // Get the frame for which the scope or scopes are requested.
1882  // With no frameNumber argument use the currently selected frame.
1883  if (scope_description && !IS_UNDEFINED(scope_description.frameNumber)) {
1884    var frame_index = scope_description.frameNumber;
1885    if (frame_index < 0 || this.exec_state_.frameCount() <= frame_index) {
1886      throw MakeTypeError(kDebuggerFrame);
1887    }
1888    return this.exec_state_.frame(frame_index);
1889  } else {
1890    return this.exec_state_.frame();
1891  }
1892};
1893
1894
1895// Gets scope host object from request. It is either a function
1896// ('functionHandle' argument must be specified) or a stack frame
1897// ('frameNumber' may be specified and the current frame is taken by default).
1898DebugCommandProcessor.prototype.resolveScopeHolder_ =
1899    function(scope_description) {
1900  if (scope_description && "functionHandle" in scope_description) {
1901    if (!IS_NUMBER(scope_description.functionHandle)) {
1902      throw MakeError(kDebugger, 'Function handle must be a number');
1903    }
1904    var function_mirror = LookupMirror(scope_description.functionHandle);
1905    if (!function_mirror) {
1906      throw MakeError(kDebugger, 'Failed to find function object by handle');
1907    }
1908    if (!function_mirror.isFunction()) {
1909      throw MakeError(kDebugger,
1910                      'Value of non-function type is found by handle');
1911    }
1912    return function_mirror;
1913  } else {
1914    // No frames no scopes.
1915    if (this.exec_state_.frameCount() == 0) {
1916      throw MakeError(kDebugger, 'No scopes');
1917    }
1918
1919    // Get the frame for which the scopes are requested.
1920    var frame = this.resolveFrameFromScopeDescription_(scope_description);
1921    return frame;
1922  }
1923}
1924
1925
1926DebugCommandProcessor.prototype.scopesRequest_ = function(request, response) {
1927  var scope_holder = this.resolveScopeHolder_(request.arguments);
1928
1929  // Fill all scopes for this frame or function.
1930  var total_scopes = scope_holder.scopeCount();
1931  var scopes = [];
1932  for (var i = 0; i < total_scopes; i++) {
1933    scopes.push(scope_holder.scope(i));
1934  }
1935  response.body = {
1936    fromScope: 0,
1937    toScope: total_scopes,
1938    totalScopes: total_scopes,
1939    scopes: scopes
1940  };
1941};
1942
1943
1944DebugCommandProcessor.prototype.scopeRequest_ = function(request, response) {
1945  // Get the frame or function for which the scope is requested.
1946  var scope_holder = this.resolveScopeHolder_(request.arguments);
1947
1948  // With no scope argument just return top scope.
1949  var scope_index = 0;
1950  if (request.arguments && !IS_UNDEFINED(request.arguments.number)) {
1951    scope_index = TO_NUMBER(request.arguments.number);
1952    if (scope_index < 0 || scope_holder.scopeCount() <= scope_index) {
1953      return response.failed('Invalid scope number');
1954    }
1955  }
1956
1957  response.body = scope_holder.scope(scope_index);
1958};
1959
1960
1961// Reads value from protocol description. Description may be in form of type
1962// (for singletons), raw value (primitive types supported in JSON),
1963// string value description plus type (for primitive values) or handle id.
1964// Returns raw value or throws exception.
1965DebugCommandProcessor.resolveValue_ = function(value_description) {
1966  if ("handle" in value_description) {
1967    var value_mirror = LookupMirror(value_description.handle);
1968    if (!value_mirror) {
1969      throw MakeError(kDebugger, "Failed to resolve value by handle, ' #" +
1970                                 value_description.handle + "# not found");
1971    }
1972    return value_mirror.value();
1973  } else if ("stringDescription" in value_description) {
1974    if (value_description.type == MirrorType.BOOLEAN_TYPE) {
1975      return TO_BOOLEAN(value_description.stringDescription);
1976    } else if (value_description.type == MirrorType.NUMBER_TYPE) {
1977      return TO_NUMBER(value_description.stringDescription);
1978    } if (value_description.type == MirrorType.STRING_TYPE) {
1979      return TO_STRING(value_description.stringDescription);
1980    } else {
1981      throw MakeError(kDebugger, "Unknown type");
1982    }
1983  } else if ("value" in value_description) {
1984    return value_description.value;
1985  } else if (value_description.type == MirrorType.UNDEFINED_TYPE) {
1986    return UNDEFINED;
1987  } else if (value_description.type == MirrorType.NULL_TYPE) {
1988    return null;
1989  } else {
1990    throw MakeError(kDebugger, "Failed to parse value description");
1991  }
1992};
1993
1994
1995DebugCommandProcessor.prototype.setVariableValueRequest_ =
1996    function(request, response) {
1997  if (!request.arguments) {
1998    response.failed('Missing arguments');
1999    return;
2000  }
2001
2002  if (IS_UNDEFINED(request.arguments.name)) {
2003    response.failed('Missing variable name');
2004  }
2005  var variable_name = request.arguments.name;
2006
2007  var scope_description = request.arguments.scope;
2008
2009  // Get the frame or function for which the scope is requested.
2010  var scope_holder = this.resolveScopeHolder_(scope_description);
2011
2012  if (IS_UNDEFINED(scope_description.number)) {
2013    response.failed('Missing scope number');
2014  }
2015  var scope_index = TO_NUMBER(scope_description.number);
2016
2017  var scope = scope_holder.scope(scope_index);
2018
2019  var new_value =
2020      DebugCommandProcessor.resolveValue_(request.arguments.newValue);
2021
2022  scope.setVariableValue(variable_name, new_value);
2023
2024  var new_value_mirror = MakeMirror(new_value);
2025
2026  response.body = {
2027    newValue: new_value_mirror
2028  };
2029};
2030
2031
2032DebugCommandProcessor.prototype.evaluateRequest_ = function(request, response) {
2033  if (!request.arguments) {
2034    return response.failed('Missing arguments');
2035  }
2036
2037  // Pull out arguments.
2038  var expression = request.arguments.expression;
2039  var frame = request.arguments.frame;
2040  var global = request.arguments.global;
2041  var disable_break = request.arguments.disable_break;
2042  var additional_context = request.arguments.additional_context;
2043
2044  // The expression argument could be an integer so we convert it to a
2045  // string.
2046  try {
2047    expression = TO_STRING(expression);
2048  } catch(e) {
2049    return response.failed('Failed to convert expression argument to string');
2050  }
2051
2052  // Check for legal arguments.
2053  if (!IS_UNDEFINED(frame) && global) {
2054    return response.failed('Arguments "frame" and "global" are exclusive');
2055  }
2056
2057  var additional_context_object;
2058  if (additional_context) {
2059    additional_context_object = {};
2060    for (var i = 0; i < additional_context.length; i++) {
2061      var mapping = additional_context[i];
2062
2063      if (!IS_STRING(mapping.name)) {
2064        return response.failed("Context element #" + i +
2065            " doesn't contain name:string property");
2066      }
2067
2068      var raw_value = DebugCommandProcessor.resolveValue_(mapping);
2069      additional_context_object[mapping.name] = raw_value;
2070    }
2071  }
2072
2073  // Global evaluate.
2074  if (global) {
2075    // Evaluate in the native context.
2076    response.body = this.exec_state_.evaluateGlobal(
2077        expression, TO_BOOLEAN(disable_break), additional_context_object);
2078    return;
2079  }
2080
2081  // Default value for disable_break is true.
2082  if (IS_UNDEFINED(disable_break)) {
2083    disable_break = true;
2084  }
2085
2086  // No frames no evaluate in frame.
2087  if (this.exec_state_.frameCount() == 0) {
2088    return response.failed('No frames');
2089  }
2090
2091  // Check whether a frame was specified.
2092  if (!IS_UNDEFINED(frame)) {
2093    var frame_number = TO_NUMBER(frame);
2094    if (frame_number < 0 || frame_number >= this.exec_state_.frameCount()) {
2095      return response.failed('Invalid frame "' + frame + '"');
2096    }
2097    // Evaluate in the specified frame.
2098    response.body = this.exec_state_.frame(frame_number).evaluate(
2099        expression, TO_BOOLEAN(disable_break), additional_context_object);
2100    return;
2101  } else {
2102    // Evaluate in the selected frame.
2103    response.body = this.exec_state_.frame().evaluate(
2104        expression, TO_BOOLEAN(disable_break), additional_context_object);
2105    return;
2106  }
2107};
2108
2109
2110DebugCommandProcessor.prototype.lookupRequest_ = function(request, response) {
2111  if (!request.arguments) {
2112    return response.failed('Missing arguments');
2113  }
2114
2115  // Pull out arguments.
2116  var handles = request.arguments.handles;
2117
2118  // Check for legal arguments.
2119  if (IS_UNDEFINED(handles)) {
2120    return response.failed('Argument "handles" missing');
2121  }
2122
2123  // Set 'includeSource' option for script lookup.
2124  if (!IS_UNDEFINED(request.arguments.includeSource)) {
2125    var includeSource = TO_BOOLEAN(request.arguments.includeSource);
2126    response.setOption('includeSource', includeSource);
2127  }
2128
2129  // Lookup handles.
2130  var mirrors = {};
2131  for (var i = 0; i < handles.length; i++) {
2132    var handle = handles[i];
2133    var mirror = LookupMirror(handle);
2134    if (!mirror) {
2135      return response.failed('Object #' + handle + '# not found');
2136    }
2137    mirrors[handle] = mirror;
2138  }
2139  response.body = mirrors;
2140};
2141
2142
2143DebugCommandProcessor.prototype.referencesRequest_ =
2144    function(request, response) {
2145  if (!request.arguments) {
2146    return response.failed('Missing arguments');
2147  }
2148
2149  // Pull out arguments.
2150  var type = request.arguments.type;
2151  var handle = request.arguments.handle;
2152
2153  // Check for legal arguments.
2154  if (IS_UNDEFINED(type)) {
2155    return response.failed('Argument "type" missing');
2156  }
2157  if (IS_UNDEFINED(handle)) {
2158    return response.failed('Argument "handle" missing');
2159  }
2160  if (type != 'referencedBy' && type != 'constructedBy') {
2161    return response.failed('Invalid type "' + type + '"');
2162  }
2163
2164  // Lookup handle and return objects with references the object.
2165  var mirror = LookupMirror(handle);
2166  if (mirror) {
2167    if (type == 'referencedBy') {
2168      response.body = mirror.referencedBy();
2169    } else {
2170      response.body = mirror.constructedBy();
2171    }
2172  } else {
2173    return response.failed('Object #' + handle + '# not found');
2174  }
2175};
2176
2177
2178DebugCommandProcessor.prototype.sourceRequest_ = function(request, response) {
2179  // No frames no source.
2180  if (this.exec_state_.frameCount() == 0) {
2181    return response.failed('No source');
2182  }
2183
2184  var from_line;
2185  var to_line;
2186  var frame = this.exec_state_.frame();
2187  if (request.arguments) {
2188    // Pull out arguments.
2189    from_line = request.arguments.fromLine;
2190    to_line = request.arguments.toLine;
2191
2192    if (!IS_UNDEFINED(request.arguments.frame)) {
2193      var frame_number = TO_NUMBER(request.arguments.frame);
2194      if (frame_number < 0 || frame_number >= this.exec_state_.frameCount()) {
2195        return response.failed('Invalid frame "' + frame + '"');
2196      }
2197      frame = this.exec_state_.frame(frame_number);
2198    }
2199  }
2200
2201  // Get the script selected.
2202  var script = frame.func().script();
2203  if (!script) {
2204    return response.failed('No source');
2205  }
2206
2207  // Get the source slice and fill it into the response.
2208  var slice = script.sourceSlice(from_line, to_line);
2209  if (!slice) {
2210    return response.failed('Invalid line interval');
2211  }
2212  response.body = {};
2213  response.body.source = slice.sourceText();
2214  response.body.fromLine = slice.from_line;
2215  response.body.toLine = slice.to_line;
2216  response.body.fromPosition = slice.from_position;
2217  response.body.toPosition = slice.to_position;
2218  response.body.totalLines = script.lineCount();
2219};
2220
2221
2222DebugCommandProcessor.prototype.scriptsRequest_ = function(request, response) {
2223  var types = ScriptTypeFlag(Debug.ScriptType.Normal);
2224  var includeSource = false;
2225  var idsToInclude = null;
2226  if (request.arguments) {
2227    // Pull out arguments.
2228    if (!IS_UNDEFINED(request.arguments.types)) {
2229      types = TO_NUMBER(request.arguments.types);
2230      if (IsNaN(types) || types < 0) {
2231        return response.failed('Invalid types "' +
2232                               request.arguments.types + '"');
2233      }
2234    }
2235
2236    if (!IS_UNDEFINED(request.arguments.includeSource)) {
2237      includeSource = TO_BOOLEAN(request.arguments.includeSource);
2238      response.setOption('includeSource', includeSource);
2239    }
2240
2241    if (IS_ARRAY(request.arguments.ids)) {
2242      idsToInclude = {};
2243      var ids = request.arguments.ids;
2244      for (var i = 0; i < ids.length; i++) {
2245        idsToInclude[ids[i]] = true;
2246      }
2247    }
2248
2249    var filterStr = null;
2250    var filterNum = null;
2251    if (!IS_UNDEFINED(request.arguments.filter)) {
2252      var num = TO_NUMBER(request.arguments.filter);
2253      if (!IsNaN(num)) {
2254        filterNum = num;
2255      }
2256      filterStr = request.arguments.filter;
2257    }
2258  }
2259
2260  // Collect all scripts in the heap.
2261  var scripts = %DebugGetLoadedScripts();
2262
2263  response.body = [];
2264
2265  for (var i = 0; i < scripts.length; i++) {
2266    if (idsToInclude && !idsToInclude[scripts[i].id]) {
2267      continue;
2268    }
2269    if (filterStr || filterNum) {
2270      var script = scripts[i];
2271      var found = false;
2272      if (filterNum && !found) {
2273        if (script.id && script.id === filterNum) {
2274          found = true;
2275        }
2276      }
2277      if (filterStr && !found) {
2278        if (script.name && script.name.indexOf(filterStr) >= 0) {
2279          found = true;
2280        }
2281      }
2282      if (!found) continue;
2283    }
2284    if (types & ScriptTypeFlag(scripts[i].type)) {
2285      response.body.push(MakeMirror(scripts[i]));
2286    }
2287  }
2288};
2289
2290
2291DebugCommandProcessor.prototype.threadsRequest_ = function(request, response) {
2292  // Get the number of threads.
2293  var total_threads = this.exec_state_.threadCount();
2294
2295  // Get information for all threads.
2296  var threads = [];
2297  for (var i = 0; i < total_threads; i++) {
2298    var details = %GetThreadDetails(this.exec_state_.break_id, i);
2299    var thread_info = { current: details[0],
2300                        id: details[1]
2301                      };
2302    threads.push(thread_info);
2303  }
2304
2305  // Create the response body.
2306  response.body = {
2307    totalThreads: total_threads,
2308    threads: threads
2309  };
2310};
2311
2312
2313DebugCommandProcessor.prototype.suspendRequest_ = function(request, response) {
2314  response.running = false;
2315};
2316
2317
2318DebugCommandProcessor.prototype.versionRequest_ = function(request, response) {
2319  response.body = {
2320    V8Version: %GetV8Version()
2321  };
2322};
2323
2324
2325DebugCommandProcessor.prototype.changeLiveRequest_ = function(
2326    request, response) {
2327  if (!request.arguments) {
2328    return response.failed('Missing arguments');
2329  }
2330  var script_id = request.arguments.script_id;
2331  var preview_only = !!request.arguments.preview_only;
2332
2333  var scripts = %DebugGetLoadedScripts();
2334
2335  var the_script = null;
2336  for (var i = 0; i < scripts.length; i++) {
2337    if (scripts[i].id == script_id) {
2338      the_script = scripts[i];
2339    }
2340  }
2341  if (!the_script) {
2342    response.failed('Script not found');
2343    return;
2344  }
2345
2346  var change_log = new GlobalArray();
2347
2348  if (!IS_STRING(request.arguments.new_source)) {
2349    throw "new_source argument expected";
2350  }
2351
2352  var new_source = request.arguments.new_source;
2353
2354  var result_description;
2355  try {
2356    result_description = Debug.LiveEdit.SetScriptSource(the_script,
2357        new_source, preview_only, change_log);
2358  } catch (e) {
2359    if (e instanceof Debug.LiveEdit.Failure && "details" in e) {
2360      response.failed(e.message, e.details);
2361      return;
2362    }
2363    throw e;
2364  }
2365  response.body = {change_log: change_log, result: result_description};
2366
2367  if (!preview_only && !this.running_ && result_description.stack_modified) {
2368    response.body.stepin_recommended = true;
2369  }
2370};
2371
2372
2373DebugCommandProcessor.prototype.restartFrameRequest_ = function(
2374    request, response) {
2375  if (!request.arguments) {
2376    return response.failed('Missing arguments');
2377  }
2378  var frame = request.arguments.frame;
2379
2380  // No frames to evaluate in frame.
2381  if (this.exec_state_.frameCount() == 0) {
2382    return response.failed('No frames');
2383  }
2384
2385  var frame_mirror;
2386  // Check whether a frame was specified.
2387  if (!IS_UNDEFINED(frame)) {
2388    var frame_number = TO_NUMBER(frame);
2389    if (frame_number < 0 || frame_number >= this.exec_state_.frameCount()) {
2390      return response.failed('Invalid frame "' + frame + '"');
2391    }
2392    // Restart specified frame.
2393    frame_mirror = this.exec_state_.frame(frame_number);
2394  } else {
2395    // Restart selected frame.
2396    frame_mirror = this.exec_state_.frame();
2397  }
2398
2399  var result_description = Debug.LiveEdit.RestartFrame(frame_mirror);
2400  response.body = {result: result_description};
2401};
2402
2403
2404DebugCommandProcessor.prototype.debuggerFlagsRequest_ = function(request,
2405                                                                 response) {
2406  // Check for legal request.
2407  if (!request.arguments) {
2408    response.failed('Missing arguments');
2409    return;
2410  }
2411
2412  // Pull out arguments.
2413  var flags = request.arguments.flags;
2414
2415  response.body = { flags: [] };
2416  if (!IS_UNDEFINED(flags)) {
2417    for (var i = 0; i < flags.length; i++) {
2418      var name = flags[i].name;
2419      var debugger_flag = debugger_flags[name];
2420      if (!debugger_flag) {
2421        continue;
2422      }
2423      if ('value' in flags[i]) {
2424        debugger_flag.setValue(flags[i].value);
2425      }
2426      response.body.flags.push({ name: name, value: debugger_flag.getValue() });
2427    }
2428  } else {
2429    for (var name in debugger_flags) {
2430      var value = debugger_flags[name].getValue();
2431      response.body.flags.push({ name: name, value: value });
2432    }
2433  }
2434};
2435
2436
2437DebugCommandProcessor.prototype.v8FlagsRequest_ = function(request, response) {
2438  var flags = request.arguments.flags;
2439  if (!flags) flags = '';
2440  %SetFlags(flags);
2441};
2442
2443
2444DebugCommandProcessor.prototype.gcRequest_ = function(request, response) {
2445  var type = request.arguments.type;
2446  if (!type) type = 'all';
2447
2448  var before = %GetHeapUsage();
2449  %CollectGarbage(type);
2450  var after = %GetHeapUsage();
2451
2452  response.body = { "before": before, "after": after };
2453};
2454
2455
2456DebugCommandProcessor.prototype.dispatch_ = (function() {
2457  var proto = DebugCommandProcessor.prototype;
2458  return {
2459    "continue":             proto.continueRequest_,
2460    "break"   :             proto.breakRequest_,
2461    "setbreakpoint" :       proto.setBreakPointRequest_,
2462    "changebreakpoint":     proto.changeBreakPointRequest_,
2463    "clearbreakpoint":      proto.clearBreakPointRequest_,
2464    "clearbreakpointgroup": proto.clearBreakPointGroupRequest_,
2465    "disconnect":           proto.disconnectRequest_,
2466    "setexceptionbreak":    proto.setExceptionBreakRequest_,
2467    "listbreakpoints":      proto.listBreakpointsRequest_,
2468    "backtrace":            proto.backtraceRequest_,
2469    "frame":                proto.frameRequest_,
2470    "scopes":               proto.scopesRequest_,
2471    "scope":                proto.scopeRequest_,
2472    "setvariablevalue":     proto.setVariableValueRequest_,
2473    "evaluate":             proto.evaluateRequest_,
2474    "lookup":               proto.lookupRequest_,
2475    "references":           proto.referencesRequest_,
2476    "source":               proto.sourceRequest_,
2477    "scripts":              proto.scriptsRequest_,
2478    "threads":              proto.threadsRequest_,
2479    "suspend":              proto.suspendRequest_,
2480    "version":              proto.versionRequest_,
2481    "changelive":           proto.changeLiveRequest_,
2482    "restartframe":         proto.restartFrameRequest_,
2483    "flags":                proto.debuggerFlagsRequest_,
2484    "v8flag":               proto.v8FlagsRequest_,
2485    "gc":                   proto.gcRequest_,
2486  };
2487})();
2488
2489
2490// Check whether the previously processed command caused the VM to become
2491// running.
2492DebugCommandProcessor.prototype.isRunning = function() {
2493  return this.running_;
2494};
2495
2496
2497DebugCommandProcessor.prototype.systemBreak = function(cmd, args) {
2498  return %SystemBreak();
2499};
2500
2501
2502/**
2503 * Convert an Object to its debugger protocol representation. The representation
2504 * may be serilized to a JSON object using JSON.stringify().
2505 * This implementation simply runs through all string property names, converts
2506 * each property value to a protocol value and adds the property to the result
2507 * object. For type "object" the function will be called recursively. Note that
2508 * circular structures will cause infinite recursion.
2509 * @param {Object} object The object to format as protocol object.
2510 * @param {MirrorSerializer} mirror_serializer The serializer to use if any
2511 *     mirror objects are encountered.
2512 * @return {Object} Protocol object value.
2513 */
2514function ObjectToProtocolObject_(object, mirror_serializer) {
2515  var content = {};
2516  for (var key in object) {
2517    // Only consider string keys.
2518    if (typeof key == 'string') {
2519      // Format the value based on its type.
2520      var property_value_json = ValueToProtocolValue_(object[key],
2521                                                      mirror_serializer);
2522      // Add the property if relevant.
2523      if (!IS_UNDEFINED(property_value_json)) {
2524        content[key] = property_value_json;
2525      }
2526    }
2527  }
2528
2529  return content;
2530}
2531
2532
2533/**
2534 * Convert an array to its debugger protocol representation. It will convert
2535 * each array element to a protocol value.
2536 * @param {Array} array The array to format as protocol array.
2537 * @param {MirrorSerializer} mirror_serializer The serializer to use if any
2538 *     mirror objects are encountered.
2539 * @return {Array} Protocol array value.
2540 */
2541function ArrayToProtocolArray_(array, mirror_serializer) {
2542  var json = [];
2543  for (var i = 0; i < array.length; i++) {
2544    json.push(ValueToProtocolValue_(array[i], mirror_serializer));
2545  }
2546  return json;
2547}
2548
2549
2550/**
2551 * Convert a value to its debugger protocol representation.
2552 * @param {*} value The value to format as protocol value.
2553 * @param {MirrorSerializer} mirror_serializer The serializer to use if any
2554 *     mirror objects are encountered.
2555 * @return {*} Protocol value.
2556 */
2557function ValueToProtocolValue_(value, mirror_serializer) {
2558  // Format the value based on its type.
2559  var json;
2560  switch (typeof value) {
2561    case 'object':
2562      if (value instanceof Mirror) {
2563        json = mirror_serializer.serializeValue(value);
2564      } else if (IS_ARRAY(value)){
2565        json = ArrayToProtocolArray_(value, mirror_serializer);
2566      } else {
2567        json = ObjectToProtocolObject_(value, mirror_serializer);
2568      }
2569      break;
2570
2571    case 'boolean':
2572    case 'string':
2573    case 'number':
2574      json = value;
2575      break;
2576
2577    default:
2578      json = null;
2579  }
2580  return json;
2581}
2582
2583
2584// -------------------------------------------------------------------
2585// Exports
2586
2587utils.InstallConstants(global, [
2588  "Debug", Debug,
2589  "DebugCommandProcessor", DebugCommandProcessor,
2590  "BreakEvent", BreakEvent,
2591  "CompileEvent", CompileEvent,
2592  "BreakPoint", BreakPoint,
2593]);
2594
2595// Functions needed by the debugger runtime.
2596utils.InstallFunctions(utils, DONT_ENUM, [
2597  "MakeExecutionState", MakeExecutionState,
2598  "MakeExceptionEvent", MakeExceptionEvent,
2599  "MakeBreakEvent", MakeBreakEvent,
2600  "MakeCompileEvent", MakeCompileEvent,
2601  "MakePromiseEvent", MakePromiseEvent,
2602  "MakeAsyncTaskEvent", MakeAsyncTaskEvent,
2603  "IsBreakPointTriggered", IsBreakPointTriggered,
2604  "UpdateScriptBreakPoints", UpdateScriptBreakPoints,
2605]);
2606
2607// Export to liveedit.js
2608utils.Export(function(to) {
2609  to.GetScriptBreakPoints = GetScriptBreakPoints;
2610});
2611
2612})
2613