From 86c2a15691e76b4b8b75c39f08e27b1bb3dbe0f7 Mon Sep 17 00:00:00 2001 From: "yangguo@chromium.org" Date: Thu, 24 May 2012 11:00:05 +0000 Subject: [PATCH] messages.js: Get better function names in stack traces. CallSite.getFunctionName() is able to retrieve names for functions better than getFunction().name. Use it in CallSite.toString(). Code by marja@chromium.org. BUG=NONE TEST=stack-traces.js: Added testClassNames. Review URL: https://chromiumcodereview.appspot.com/10384196 git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@11652 ce2b1a6d-e550-0410-aec6-3dcde31c8c00 --- src/messages.js | 47 ++++++++++++++++++++++-------------- test/mjsunit/stack-traces.js | 14 +++++++++++ 2 files changed, 43 insertions(+), 18 deletions(-) diff --git a/src/messages.js b/src/messages.js index ab7193672..2a00ba846 100644 --- a/src/messages.js +++ b/src/messages.js @@ -788,15 +788,7 @@ function CallSiteGetThis() { } function CallSiteGetTypeName() { - var constructor = this.receiver.constructor; - if (!constructor) { - return %_CallFunction(this.receiver, ObjectToString); - } - var constructorName = constructor.name; - if (!constructorName) { - return %_CallFunction(this.receiver, ObjectToString); - } - return constructorName; + return GetTypeName(this, false); } function CallSiteIsToplevel() { @@ -830,8 +822,10 @@ function CallSiteGetFunctionName() { var name = this.fun.name; if (name) { return name; - } else { - return %FunctionGetInferredName(this.fun); + } + name = %FunctionGetInferredName(this.fun); + if (name) { + return name; } // Maybe this is an evaluation? var script = %FunctionGetScript(this.fun); @@ -952,20 +946,24 @@ function CallSiteToString() { fileLocation = "unknown source"; } var line = ""; - var functionName = this.getFunction().name; - var addPrefix = true; + var functionName = this.getFunctionName(); + var addSuffix = true; var isConstructor = this.isConstructor(); var isMethodCall = !(this.isToplevel() || isConstructor); if (isMethodCall) { + var typeName = GetTypeName(this, true); var methodName = this.getMethodName(); - line += this.getTypeName() + "."; if (functionName) { + if (typeName && functionName.indexOf(typeName) != 0) { + line += typeName + "."; + } line += functionName; - if (methodName && (methodName != functionName)) { + if (methodName && functionName.lastIndexOf("." + methodName) != + functionName.length - methodName.length - 1) { line += " [as " + methodName + "]"; } } else { - line += methodName || ""; + line += typeName + "." + (methodName || ""); } } else if (isConstructor) { line += "new " + (functionName || ""); @@ -973,9 +971,9 @@ function CallSiteToString() { line += functionName; } else { line += fileLocation; - addPrefix = false; + addSuffix = false; } - if (addPrefix) { + if (addSuffix) { line += " (" + fileLocation + ")"; } return line; @@ -1085,6 +1083,19 @@ function FormatRawStackTrace(error, raw_stack) { } } +function GetTypeName(obj, requireConstructor) { + var constructor = obj.receiver.constructor; + if (!constructor) { + return requireConstructor ? null : + %_CallFunction(obj.receiver, ObjectToString); + } + var constructorName = constructor.name; + if (!constructorName) { + return requireConstructor ? null : + %_CallFunction(obj.receiver, ObjectToString); + } + return constructorName; +} function captureStackTrace(obj, cons_opt) { var stackTraceLimit = $Error.stackTraceLimit; diff --git a/test/mjsunit/stack-traces.js b/test/mjsunit/stack-traces.js index 536e71bbb..438eec979 100644 --- a/test/mjsunit/stack-traces.js +++ b/test/mjsunit/stack-traces.js @@ -111,6 +111,18 @@ function testStrippedCustomError() { throw new CustomError("hep-hey", CustomError); } +MyObj = function() { FAIL; } + +MyObjCreator = function() {} + +MyObjCreator.prototype.Create = function() { + return new MyObj(); +} + +function testClassNames() { + (new MyObjCreator).Create(); +} + // Utility function for testing that the expected strings occur // in the stack trace produced when running the given function. function testTrace(name, fun, expected, unexpected) { @@ -254,6 +266,8 @@ testTrace("testDefaultCustomError", testDefaultCustomError, ["collectStackTrace"]); testTrace("testStrippedCustomError", testStrippedCustomError, ["hep-hey"], ["new CustomError", "collectStackTrace"]); +testTrace("testClassNames", testClassNames, + ["new MyObj", "MyObjCreator.Create"], ["as Create"]); testCallerCensorship(); testUnintendedCallerCensorship(); testErrorsDuringFormatting(); -- 2.34.1