Bug 969786: Implement Debugger.Source.prototype.introductionScript. r=sfink

This commit is contained in:
Jim Blandy 2014-02-26 15:20:00 -08:00
Родитель 374b46f9f3
Коммит 6a4b428b35
6 изменённых файлов: 271 добавлений и 3 удалений

Просмотреть файл

@ -0,0 +1,118 @@
// Dynamically generated sources should have their introduction script and
// offset set correctly.
var g = newGlobal();
var dbg = new Debugger;
var gDO = dbg.addDebuggee(g);
var log;
// Direct eval, while the frame is live.
dbg.onDebuggerStatement = function (frame) {
log += 'd';
var source = frame.script.source;
var introducer = frame.older;
assertEq(source.introductionScript, introducer.script);
assertEq(source.introductionOffset, introducer.offset);
};
log = '';
g.eval('\n\neval("\\n\\ndebugger;");');
assertEq(log, 'd');
// Direct eval, after the frame has been popped.
var introducer, introduced;
dbg.onDebuggerStatement = function (frame) {
log += 'de1';
introducer = frame.script;
dbg.onDebuggerStatement = function (frame) {
log += 'de2';
introduced = frame.script.source;
};
};
log = '';
g.evaluate('debugger; eval("\\n\\ndebugger;");', { lineNumber: 1812 });
assertEq(log, 'de1de2');
assertEq(introduced.introductionScript, introducer);
assertEq(introducer.getOffsetLine(introduced.introductionOffset), 1812);
// Indirect eval, while the frame is live.
dbg.onDebuggerStatement = function (frame) {
log += 'd';
var source = frame.script.source;
var introducer = frame.older;
assertEq(source.introductionScript, introducer.script);
assertEq(source.introductionOffset, introducer.offset);
};
log = '';
g.eval('\n\n(0,eval)("\\n\\ndebugger;");');
assertEq(log, 'd');
// Indirect eval, after the frame has been popped.
var introducer, introduced;
dbg.onDebuggerStatement = function (frame) {
log += 'de1';
introducer = frame.script;
dbg.onDebuggerStatement = function (frame) {
log += 'de2';
introduced = frame.script.source;
};
};
log = '';
g.evaluate('debugger; (0,eval)("\\n\\ndebugger;");', { lineNumber: 1066 });
assertEq(log, 'de1de2');
assertEq(introduced.introductionScript, introducer);
assertEq(introducer.getOffsetLine(introduced.introductionOffset), 1066);
// Function constructor.
dbg.onDebuggerStatement = function (frame) {
log += 'o';
var outerScript = frame.script;
var outerOffset = frame.offset;
dbg.onDebuggerStatement = function (frame) {
log += 'i';
var source = frame.script.source;
assertEq(source.introductionScript, outerScript);
assertEq(outerScript.getOffsetLine(source.introductionOffset),
outerScript.getOffsetLine(outerOffset));
};
};
log = '';
g.eval('\n\n\ndebugger; Function("debugger;")()');
assertEq(log, 'oi');
// Function constructor, after the the introduction call's frame has been
// popped.
var introducer;
dbg.onDebuggerStatement = function (frame) {
log += 'F2';
introducer = frame.script;
};
log = '';
var fDO = gDO.evalInGlobal('debugger; Function("origami;")', { lineNumber: 1685 }).return;
var source = fDO.script.source;
assertEq(log, 'F2');
assertEq(source.introductionScript, introducer);
assertEq(introducer.getOffsetLine(source.introductionOffset), 1685);
// If the introduction script is in a different global from the script it
// introduced, we don't record it.
dbg.onDebuggerStatement = function (frame) {
log += 'x';
var source = frame.script.source;
assertEq(source.introductionScript, undefined);
assertEq(source.introductionOffset, undefined);
};
log = '';
g.eval('debugger;'); // introduction script is this top-level script
assertEq(log, 'x');
// If the code is introduced by a function that doesn't provide
// introduction information, that shouldn't be a problem.
dbg.onDebuggerStatement = function (frame) {
log += 'x';
var source = frame.script.source;
assertEq(source.introductionScript, undefined);
assertEq(source.introductionOffset, undefined);
};
log = '';
g.eval('evaluate("debugger;", { lineNumber: 1729 });');
assertEq(log, 'x');

Просмотреть файл

@ -0,0 +1,31 @@
// Calls to 'eval', etc. by JS primitives get attributed to the point of
// the primitive's call.
var g = newGlobal();
var dbg = new Debugger;
var gDO = dbg.addDebuggee(g);
var log = '';
function outerHandler(frame) {
log += 'o';
var outerScript = frame.script;
dbg.onDebuggerStatement = function (frame) {
log += 'i';
var source = frame.script.source;
var introScript = source.introductionScript;
assertEq(introScript, outerScript);
assertEq(introScript.getOffsetLine(source.introductionOffset), 1234);
};
};
log = '';
dbg.onDebuggerStatement = outerHandler;
g.evaluate('debugger; ["debugger;"].map(eval)', { lineNumber: 1234 });
assertEq(log, 'oi');
log = '';
dbg.onDebuggerStatement = outerHandler;
g.evaluate('debugger; "debugger;".replace(/.*/, eval);',
{ lineNumber: 1234 });
assertEq(log, 'oi');

Просмотреть файл

@ -1826,7 +1826,7 @@ ScriptSource::initFromOptions(ExclusiveContext *cx, const ReadOnlyCompileOptions
JS_HoldPrincipals(originPrincipals_);
introductionType_ = options.introductionType;
introductionOffset_ = options.introductionOffset;
setIntroductionOffset(options.introductionOffset);
if (options.hasIntroductionInfo) {
JS_ASSERT(options.introductionType != nullptr);

Просмотреть файл

@ -3920,13 +3920,33 @@ DebuggerSource_getElementProperty(JSContext *cx, unsigned argc, Value *vp)
return Debugger::fromChildJSObject(obj)->wrapDebuggeeValue(cx, args.rval());
}
static bool
DebuggerSource_getIntroductionScript(JSContext *cx, unsigned argc, Value *vp)
{
THIS_DEBUGSOURCE_REFERENT(cx, argc, vp, "(get introductionScript)", args, obj, sourceObject);
RootedScript script(cx, sourceObject->introductionScript());
if (script) {
RootedObject scriptDO(cx, Debugger::fromChildJSObject(obj)->wrapScript(cx, script));
if (!scriptDO)
return false;
args.rval().setObject(*scriptDO);
} else {
args.rval().setUndefined();
}
return true;
}
static bool
DebuggerSource_getIntroductionOffset(JSContext *cx, unsigned argc, Value *vp)
{
THIS_DEBUGSOURCE_REFERENT(cx, argc, vp, "(get introductionOffset)", args, obj, sourceObject);
// Regardless of what's recorded in the ScriptSourceObject and
// ScriptSource, only hand out the introduction offset if we also have
// the script within which it applies.
ScriptSource *ss = sourceObject->source();
if (ss->hasIntroductionOffset())
if (ss->hasIntroductionOffset() && sourceObject->introductionScript())
args.rval().setInt32(ss->introductionOffset());
else
args.rval().setUndefined();
@ -3936,7 +3956,7 @@ DebuggerSource_getIntroductionOffset(JSContext *cx, unsigned argc, Value *vp)
static bool
DebuggerSource_getIntroductionType(JSContext *cx, unsigned argc, Value *vp)
{
THIS_DEBUGSOURCE_REFERENT(cx, argc, vp, "(get introductionOffset)", args, obj, sourceObject);
THIS_DEBUGSOURCE_REFERENT(cx, argc, vp, "(get introductionType)", args, obj, sourceObject);
ScriptSource *ss = sourceObject->source();
if (ss->hasIntroductionType()) {
@ -3955,6 +3975,7 @@ static const JSPropertySpec DebuggerSource_properties[] = {
JS_PSG("url", DebuggerSource_getUrl, 0),
JS_PSG("element", DebuggerSource_getElement, 0),
JS_PSG("displayURL", DebuggerSource_getDisplayURL, 0),
JS_PSG("introductionScript", DebuggerSource_getIntroductionScript, 0),
JS_PSG("introductionOffset", DebuggerSource_getIntroductionOffset, 0),
JS_PSG("introductionType", DebuggerSource_getIntroductionType, 0),
JS_PSG("elementAttributeName", DebuggerSource_getElementProperty, 0),

Просмотреть файл

@ -12,6 +12,7 @@ support-files =
Debugger.Source.prototype.element-2.js
Debugger.Source.prototype.element.html
[test_Debugger.Source.prototype.introductionScript.html]
[test_Debugger.Source.prototype.introductionType.html]
[test_Debugger.Source.prototype.element.html]
[test_Debugger.Script.prototype.global.html]

Просмотреть файл

@ -0,0 +1,97 @@
<!DOCTYPE HTML>
<html>
<!--
https://bugzilla.mozilla.org/show_bug.cgi?id=969786
Debugger.Source.prototype.introductionScript and .introductionOffset should
behave when 'eval' is called with no scripted frames active at all.
-->
<head>
<meta charset="utf-8">
<title>Debugger.Source.prototype.introductionScript with no caller</title>
<script type="application/javascript" src="chrome://mochikit/content/tests/SimpleTest/SimpleTest.js"></script>
<link rel="stylesheet" type="text/css" href="chrome://mochikit/content/tests/SimpleTest/test.css">
</head>
<body>
<pre id="test">
<script>
Components.utils.import("resource://gre/modules/jsdebugger.jsm");
addDebuggerToGlobal(this);
window.onload = function () {
SimpleTest.waitForExplicitFinish();
var dbg, iframeDO, doc, script2DO;
// Create an iframe to debug.
var iframe = document.createElement("iframe");
iframe.src = "data:text/html,<div>Hi!</div>";
iframe.onload = onLoadHandler;
document.body.appendChild(iframe);
function onLoadHandler() {
// Now that the iframe's window has been created, we can add
// it as a debuggee.
dbg = new Debugger;
iframeDO = dbg.addDebuggee(iframe.contentWindow);
doc = iframe.contentWindow.document;
var script = doc.createElement('script');
script.text = "setTimeout(eval.bind(null, 'debugger;'), 0);";
dbg.onDebuggerStatement = timerHandler;
doc.body.appendChild(script);
}
function timerHandler(frame) {
// The top stack frame's source should have an undefined
// introduction script and introduction offset.
var source = frame.script.source;
ok(source.introductionScript === undefined,
"setTimeout eval introductionScript is undefined");
ok(source.introductionOffset === undefined,
"setTimeout eval introductionOffset is undefined");
// Check that the above isn't just some quirk of iframes, or the
// browser milieu destroying information: an eval script should indeed
// have proper introduction information.
var script2 = doc.createElement('script');
script2.text = "eval('debugger;');";
script2DO = iframeDO.makeDebuggeeValue(script2);
dbg.onDebuggerStatement = evalHandler;
doc.body.appendChild(script2);
}
function evalHandler(frame) {
// The top stack frame's source should be introduced by the script that
// called eval.
var source = frame.script.source;
var frame2 = frame.older;
ok(source.introductionType === 'eval',
"top frame's source was introduced by 'eval'");
ok(source.introductionScript === frame2.script,
"eval frame's introduction script is the older frame's script");
ok(source.introductionOffset === frame2.offset,
"eval frame's introduction offset is current offset in older frame");
ok(source.introductionScript.source.element === script2DO,
"eval frame's introducer belongs to script2 element");
// The frame that called eval, in turn, should have no introduction
// information. (In the future, we certainly could point at the call
// that inserted the script element into the document; if that happens,
// we can update this test.)
ok(frame2.script.source.introductionType === undefined,
"older frame has no introduction type");
ok(frame2.script.source.introductionScript === undefined,
"older frame has no introduction script");
ok(frame2.script.source.introductionOffset === null,
"older frame has no introduction offset");
SimpleTest.finish();
}
}
</script>
</pre>
</body>
</html>