| Index: src/messages.js
|
| diff --git a/src/messages.js b/src/messages.js
|
| index 761b31137171f6c40adf341db598ff3d29fd624b..92ac1bc46aac4a8e525b34accca62980848b6e92 100644
|
| --- a/src/messages.js
|
| +++ b/src/messages.js
|
| @@ -1078,7 +1078,26 @@ function GetStackFrames(raw_stack) {
|
| }
|
|
|
|
|
| -function FormatStackTrace(error_string, frames) {
|
| +// Flag to prevent recursive call of Error.prepareStackTrace.
|
| +var formatting_custom_stack_trace = false;
|
| +
|
| +
|
| +function FormatStackTrace(obj, error_string, frames) {
|
| + if (IS_FUNCTION($Error.prepareStackTrace) && !formatting_custom_stack_trace) {
|
| + var array = [];
|
| + %MoveArrayContents(frames, array);
|
| + formatting_custom_stack_trace = true;
|
| + var stack_trace = void 0;
|
| + try {
|
| + stack_trace = $Error.prepareStackTrace(obj, array);
|
| + } catch (e) {
|
| + throw e; // The custom formatting function threw. Rethrow.
|
| + } finally {
|
| + formatting_custom_stack_trace = false;
|
| + }
|
| + return stack_trace;
|
| + }
|
| +
|
| var lines = new InternalArray();
|
| lines.push(error_string);
|
| for (var i = 0; i < frames.length; i++) {
|
| @@ -1115,10 +1134,6 @@ function GetTypeName(receiver, requireConstructor) {
|
| }
|
|
|
|
|
| -// Flag to prevent recursive call of Error.prepareStackTrace.
|
| -var formatting_custom_stack_trace = false;
|
| -
|
| -
|
| function captureStackTrace(obj, cons_opt) {
|
| var stackTraceLimit = $Error.stackTraceLimit;
|
| if (!stackTraceLimit || !IS_NUMBER(stackTraceLimit)) return;
|
| @@ -1129,40 +1144,30 @@ function captureStackTrace(obj, cons_opt) {
|
| cons_opt ? cons_opt : captureStackTrace,
|
| stackTraceLimit);
|
|
|
| - // Don't be lazy if the error stack formatting is custom (observable).
|
| - if (IS_FUNCTION($Error.prepareStackTrace) && !formatting_custom_stack_trace) {
|
| - var array = [];
|
| - %MoveArrayContents(GetStackFrames(stack), array);
|
| - formatting_custom_stack_trace = true;
|
| - try {
|
| - obj.stack = $Error.prepareStackTrace(obj, array);
|
| - } catch (e) {
|
| - throw e; // The custom formatting function threw. Rethrow.
|
| - } finally {
|
| - formatting_custom_stack_trace = false;
|
| - }
|
| - return;
|
| - }
|
| -
|
| var error_string = FormatErrorString(obj);
|
| - // Note that 'obj' and 'this' maybe different when called on objects that
|
| - // have the error object on its prototype chain. The getter replaces itself
|
| - // with a data property as soon as the stack trace has been formatted.
|
| - // The getter must not change the object layout as it may be called after GC.
|
| + // The holder of this getter ('obj') may not be the receiver ('this').
|
| + // When this getter is called the first time, we use the context values to
|
| + // format a stack trace string and turn this accessor pair into a data
|
| + // property (on the holder).
|
| var getter = function() {
|
| - if (IS_STRING(stack)) return stack;
|
| // Stack is still a raw array awaiting to be formatted.
|
| - stack = FormatStackTrace(error_string, GetStackFrames(stack));
|
| - // Release context value.
|
| - error_string = void 0;
|
| - return stack;
|
| + var result = FormatStackTrace(obj, error_string, GetStackFrames(stack));
|
| + // Turn this accessor into a data property.
|
| + %DefineOrRedefineDataProperty(obj, 'stack', result, NONE);
|
| + // Release context values.
|
| + stack = error_string = void 0;
|
| + return result;
|
| };
|
| - %MarkOneShotGetter(getter);
|
|
|
| - // The 'stack' property of the receiver is set as data property. If
|
| - // the receiver is the same as holder, this accessor pair is replaced.
|
| + // Set the 'stack' property on the receiver. If the receiver is the same as
|
| + // holder of this setter, the accessor pair is turned into a data property.
|
| var setter = function(v) {
|
| + // Set data property on the receiver (not necessarily holder).
|
| %DefineOrRedefineDataProperty(this, 'stack', v, NONE);
|
| + if (this === obj) {
|
| + // Release context values if holder is the same as the receiver.
|
| + stack = error_string = void 0;
|
| + }
|
| };
|
|
|
| %DefineOrRedefineAccessorProperty(obj, 'stack', getter, setter, DONT_ENUM);
|
| @@ -1300,38 +1305,36 @@ InstallFunctions($Error.prototype, DONT_ENUM, ['toString', ErrorToString]);
|
| function SetUpStackOverflowBoilerplate() {
|
| var boilerplate = MakeRangeError('stack_overflow', []);
|
|
|
| - // The raw stack trace is stored as hidden property of the copy of this
|
| - // boilerplate error object. Note that the receiver 'this' may not be that
|
| - // error object copy, but can be found on the prototype chain of 'this'.
|
| - // When the stack trace is formatted, this accessor property is replaced by
|
| - // a data property.
|
| var error_string = boilerplate.name + ": " + boilerplate.message;
|
|
|
| - // The getter must not change the object layout as it may be called after GC.
|
| - function getter() {
|
| + // The raw stack trace is stored as a hidden property on the holder of this
|
| + // getter, which may not be the same as the receiver. Find the holder to
|
| + // retrieve the raw stack trace and then turn this accessor pair into a
|
| + // data property.
|
| + var getter = function() {
|
| var holder = this;
|
| while (!IS_ERROR(holder)) {
|
| holder = %GetPrototype(holder);
|
| - if (holder == null) return MakeSyntaxError('illegal_access', []);
|
| + if (IS_NULL(holder)) return MakeSyntaxError('illegal_access', []);
|
| }
|
| - var stack = %GetOverflowedStackTrace(holder);
|
| - if (IS_STRING(stack)) return stack;
|
| - if (IS_ARRAY(stack)) {
|
| - var result = FormatStackTrace(error_string, GetStackFrames(stack));
|
| - %SetOverflowedStackTrace(holder, result);
|
| - return result;
|
| - }
|
| - return void 0;
|
| - }
|
| - %MarkOneShotGetter(getter);
|
| + var stack = %GetAndClearOverflowedStackTrace(holder);
|
| + // We may not have captured any stack trace.
|
| + if (IS_UNDEFINED(stack)) return stack;
|
| +
|
| + var result = FormatStackTrace(holder, error_string, GetStackFrames(stack));
|
| + // Replace this accessor with a data property.
|
| + %DefineOrRedefineDataProperty(holder, 'stack', result, NONE);
|
| + return result;
|
| + };
|
|
|
| - // The 'stack' property of the receiver is set as data property. If
|
| - // the receiver is the same as holder, this accessor pair is replaced.
|
| - function setter(v) {
|
| + // Set the 'stack' property on the receiver. If the receiver is the same as
|
| + // holder of this setter, the accessor pair is turned into a data property.
|
| + var setter = function(v) {
|
| %DefineOrRedefineDataProperty(this, 'stack', v, NONE);
|
| - // Release the stack trace that is stored as hidden property, if exists.
|
| - %SetOverflowedStackTrace(this, void 0);
|
| - }
|
| + // Tentatively clear the hidden property. If the receiver is the same as
|
| + // holder, we release the raw stack trace this way.
|
| + %GetAndClearOverflowedStackTrace(this);
|
| + };
|
|
|
| %DefineOrRedefineAccessorProperty(
|
| boilerplate, 'stack', getter, setter, DONT_ENUM);
|
|
|