Revert of Remove stack overflow boilerplate (patchset #3 id:40001 of https://codereview.chromium.org/2161953003/ )
Reason for revert: Clusterfuzz failures in parent CL https://codereview.chromium.org/2142933003/ Original issue's description: > Remove stack overflow boilerplate > > We no longer need to prepare the stack overflow error in advance now that > Errors are constructed in C++. > > R=yangguo@chromium.org > BUG= > > Committed: https://crrev.com/ba95d10ccbe13e2fca427228483b045576f2dc4c > Cr-Commit-Position: refs/heads/master@{#37923} TBR=yangguo@chromium.org # Skipping CQ checks because original CL landed less than 1 days ago. NOPRESUBMIT=true NOTREECHECKS=true NOTRY=true BUG= Review-Url: https://codereview.chromium.org/2169563003 Cr-Commit-Position: refs/heads/master@{#37927}
This commit is contained in:
parent
3536db45c9
commit
c4ef8a8d6e
@ -7,7 +7,6 @@
|
|||||||
|
|
||||||
#include "src/accessors.h"
|
#include "src/accessors.h"
|
||||||
#include "src/bootstrapper.h"
|
#include "src/bootstrapper.h"
|
||||||
#include "src/messages.h"
|
|
||||||
#include "src/property-descriptor.h"
|
#include "src/property-descriptor.h"
|
||||||
#include "src/string-builder.h"
|
#include "src/string-builder.h"
|
||||||
|
|
||||||
@ -17,11 +16,53 @@ namespace internal {
|
|||||||
// ES6 section 19.5.1.1 Error ( message )
|
// ES6 section 19.5.1.1 Error ( message )
|
||||||
BUILTIN(ErrorConstructor) {
|
BUILTIN(ErrorConstructor) {
|
||||||
HandleScope scope(isolate);
|
HandleScope scope(isolate);
|
||||||
RETURN_RESULT_OR_FAILURE(
|
|
||||||
isolate,
|
// 1. If NewTarget is undefined, let newTarget be the active function object,
|
||||||
ConstructError(isolate, args.target<JSFunction>(),
|
// else let newTarget be NewTarget.
|
||||||
Handle<Object>::cast(args.new_target()),
|
|
||||||
args.atOrUndefined(isolate, 1), SKIP_FIRST, false));
|
Handle<JSFunction> target = args.target<JSFunction>();
|
||||||
|
Handle<JSReceiver> new_target;
|
||||||
|
if (args.new_target()->IsJSReceiver()) {
|
||||||
|
new_target = Handle<JSReceiver>::cast(args.new_target());
|
||||||
|
} else {
|
||||||
|
new_target = target;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2. Let O be ? OrdinaryCreateFromConstructor(newTarget, "%ErrorPrototype%",
|
||||||
|
// « [[ErrorData]] »).
|
||||||
|
Handle<JSObject> err;
|
||||||
|
ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, err,
|
||||||
|
JSObject::New(target, new_target));
|
||||||
|
|
||||||
|
// 3. If message is not undefined, then
|
||||||
|
// a. Let msg be ? ToString(message).
|
||||||
|
// b. Let msgDesc be the PropertyDescriptor{[[Value]]: msg, [[Writable]]:
|
||||||
|
// true, [[Enumerable]]: false, [[Configurable]]: true}.
|
||||||
|
// c. Perform ! DefinePropertyOrThrow(O, "message", msgDesc).
|
||||||
|
// 4. Return O.
|
||||||
|
|
||||||
|
Handle<Object> msg = args.atOrUndefined(isolate, 1);
|
||||||
|
if (!msg->IsUndefined(isolate)) {
|
||||||
|
Handle<String> msg_string;
|
||||||
|
ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, msg_string,
|
||||||
|
Object::ToString(isolate, msg));
|
||||||
|
RETURN_FAILURE_ON_EXCEPTION(
|
||||||
|
isolate,
|
||||||
|
JSObject::SetOwnPropertyIgnoreAttributes(
|
||||||
|
err, isolate->factory()->message_string(), msg_string, DONT_ENUM));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Capture the stack trace unless we're setting up.
|
||||||
|
if (!isolate->bootstrapper()->IsActive()) {
|
||||||
|
// Optionally capture a more detailed stack trace for the message.
|
||||||
|
RETURN_FAILURE_ON_EXCEPTION(isolate,
|
||||||
|
isolate->CaptureAndSetDetailedStackTrace(err));
|
||||||
|
// Capture a simple stack trace for the stack property.
|
||||||
|
RETURN_FAILURE_ON_EXCEPTION(isolate,
|
||||||
|
isolate->CaptureAndSetSimpleStackTrace(err));
|
||||||
|
}
|
||||||
|
|
||||||
|
return *err;
|
||||||
}
|
}
|
||||||
|
|
||||||
// static
|
// static
|
||||||
@ -34,7 +75,6 @@ BUILTIN(ErrorCaptureStackTrace) {
|
|||||||
}
|
}
|
||||||
Handle<JSObject> object = Handle<JSObject>::cast(object_obj);
|
Handle<JSObject> object = Handle<JSObject>::cast(object_obj);
|
||||||
Handle<Object> caller = args.atOrUndefined(isolate, 2);
|
Handle<Object> caller = args.atOrUndefined(isolate, 2);
|
||||||
FrameSkipMode mode = caller->IsJSFunction() ? SKIP_UNTIL_SEEN : SKIP_NONE;
|
|
||||||
|
|
||||||
// TODO(jgruber): Eagerly format the stack trace and remove accessors.h
|
// TODO(jgruber): Eagerly format the stack trace and remove accessors.h
|
||||||
// include.
|
// include.
|
||||||
@ -96,7 +136,7 @@ BUILTIN(ErrorCaptureStackTrace) {
|
|||||||
RETURN_FAILURE_ON_EXCEPTION(isolate,
|
RETURN_FAILURE_ON_EXCEPTION(isolate,
|
||||||
isolate->CaptureAndSetDetailedStackTrace(object));
|
isolate->CaptureAndSetDetailedStackTrace(object));
|
||||||
RETURN_FAILURE_ON_EXCEPTION(
|
RETURN_FAILURE_ON_EXCEPTION(
|
||||||
isolate, isolate->CaptureAndSetSimpleStackTrace(object, mode, caller));
|
isolate, isolate->CaptureAndSetSimpleStackTrace(object, caller));
|
||||||
|
|
||||||
return *isolate->factory()->undefined_value();
|
return *isolate->factory()->undefined_value();
|
||||||
}
|
}
|
||||||
|
@ -133,6 +133,7 @@ enum BindingFlags {
|
|||||||
V(SET_ADD_METHOD_INDEX, JSFunction, set_add) \
|
V(SET_ADD_METHOD_INDEX, JSFunction, set_add) \
|
||||||
V(SET_DELETE_METHOD_INDEX, JSFunction, set_delete) \
|
V(SET_DELETE_METHOD_INDEX, JSFunction, set_delete) \
|
||||||
V(SET_HAS_METHOD_INDEX, JSFunction, set_has) \
|
V(SET_HAS_METHOD_INDEX, JSFunction, set_has) \
|
||||||
|
V(STACK_OVERFLOW_BOILERPLATE_INDEX, JSObject, stack_overflow_boilerplate) \
|
||||||
V(SYNTAX_ERROR_FUNCTION_INDEX, JSFunction, syntax_error_function) \
|
V(SYNTAX_ERROR_FUNCTION_INDEX, JSFunction, syntax_error_function) \
|
||||||
V(TYPE_ERROR_FUNCTION_INDEX, JSFunction, type_error_function) \
|
V(TYPE_ERROR_FUNCTION_INDEX, JSFunction, type_error_function) \
|
||||||
V(URI_ERROR_FUNCTION_INDEX, JSFunction, uri_error_function)
|
V(URI_ERROR_FUNCTION_INDEX, JSFunction, uri_error_function)
|
||||||
|
@ -329,9 +329,16 @@ static Handle<FixedArray> MaybeGrow(Isolate* isolate,
|
|||||||
}
|
}
|
||||||
|
|
||||||
class StackTraceHelper {
|
class StackTraceHelper {
|
||||||
|
private:
|
||||||
|
enum FrameSkipMode {
|
||||||
|
SKIP_FIRST,
|
||||||
|
SKIP_UNTIL_SEEN,
|
||||||
|
SKIP_NONE,
|
||||||
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
StackTraceHelper(Isolate* isolate, FrameSkipMode mode, Handle<Object> caller)
|
StackTraceHelper(Isolate* isolate, Handle<Object> caller)
|
||||||
: isolate_(isolate), mode_(mode), caller_(caller) {
|
: isolate_(isolate), caller_(caller) {
|
||||||
// The caller parameter can be used to skip a specific set of frames in the
|
// The caller parameter can be used to skip a specific set of frames in the
|
||||||
// stack trace. It can be:
|
// stack trace. It can be:
|
||||||
// * null, when called from a standard error constructor. We unconditionally
|
// * null, when called from a standard error constructor. We unconditionally
|
||||||
@ -340,18 +347,15 @@ class StackTraceHelper {
|
|||||||
// * a JSFunction, when called by the user from Error.captureStackTrace().
|
// * a JSFunction, when called by the user from Error.captureStackTrace().
|
||||||
// We skip each frame until encountering the caller function.
|
// We skip each frame until encountering the caller function.
|
||||||
// * For any other value, all frames are included in the trace.
|
// * For any other value, all frames are included in the trace.
|
||||||
switch (mode_) {
|
if (caller_.is_null()) {
|
||||||
case SKIP_FIRST:
|
mode_ = SKIP_FIRST;
|
||||||
DCHECK(caller_.is_null());
|
skip_next_frame_ = true;
|
||||||
skip_next_frame_ = true;
|
} else if (caller_->IsJSFunction()) {
|
||||||
break;
|
mode_ = SKIP_UNTIL_SEEN;
|
||||||
case SKIP_UNTIL_SEEN:
|
skip_next_frame_ = true;
|
||||||
DCHECK(caller_->IsJSFunction());
|
} else {
|
||||||
skip_next_frame_ = true;
|
mode_ = SKIP_NONE;
|
||||||
break;
|
skip_next_frame_ = false;
|
||||||
case SKIP_NONE:
|
|
||||||
skip_next_frame_ = false;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
encountered_strict_function_ = false;
|
encountered_strict_function_ = false;
|
||||||
sloppy_frames_ = 0;
|
sloppy_frames_ = 0;
|
||||||
@ -421,8 +425,8 @@ class StackTraceHelper {
|
|||||||
|
|
||||||
Isolate* isolate_;
|
Isolate* isolate_;
|
||||||
|
|
||||||
const FrameSkipMode mode_;
|
FrameSkipMode mode_;
|
||||||
const Handle<Object> caller_;
|
Handle<Object> caller_;
|
||||||
bool skip_next_frame_;
|
bool skip_next_frame_;
|
||||||
|
|
||||||
int sloppy_frames_;
|
int sloppy_frames_;
|
||||||
@ -430,7 +434,6 @@ class StackTraceHelper {
|
|||||||
};
|
};
|
||||||
|
|
||||||
Handle<Object> Isolate::CaptureSimpleStackTrace(Handle<JSReceiver> error_object,
|
Handle<Object> Isolate::CaptureSimpleStackTrace(Handle<JSReceiver> error_object,
|
||||||
FrameSkipMode mode,
|
|
||||||
Handle<Object> caller) {
|
Handle<Object> caller) {
|
||||||
DisallowJavascriptExecution no_js(this);
|
DisallowJavascriptExecution no_js(this);
|
||||||
|
|
||||||
@ -449,7 +452,7 @@ Handle<Object> Isolate::CaptureSimpleStackTrace(Handle<JSReceiver> error_object,
|
|||||||
Handle<FixedArray> elements =
|
Handle<FixedArray> elements =
|
||||||
factory()->NewFixedArrayWithHoles(initial_size * 4 + 1);
|
factory()->NewFixedArrayWithHoles(initial_size * 4 + 1);
|
||||||
|
|
||||||
StackTraceHelper helper(this, mode, caller);
|
StackTraceHelper helper(this, caller);
|
||||||
|
|
||||||
// First element is reserved to store the number of sloppy frames.
|
// First element is reserved to store the number of sloppy frames.
|
||||||
int cursor = 1;
|
int cursor = 1;
|
||||||
@ -569,12 +572,10 @@ MaybeHandle<JSReceiver> Isolate::CaptureAndSetDetailedStackTrace(
|
|||||||
}
|
}
|
||||||
|
|
||||||
MaybeHandle<JSReceiver> Isolate::CaptureAndSetSimpleStackTrace(
|
MaybeHandle<JSReceiver> Isolate::CaptureAndSetSimpleStackTrace(
|
||||||
Handle<JSReceiver> error_object, FrameSkipMode mode,
|
Handle<JSReceiver> error_object, Handle<Object> caller) {
|
||||||
Handle<Object> caller) {
|
|
||||||
// Capture stack trace for simple stack trace string formatting.
|
// Capture stack trace for simple stack trace string formatting.
|
||||||
Handle<Name> key = factory()->stack_trace_symbol();
|
Handle<Name> key = factory()->stack_trace_symbol();
|
||||||
Handle<Object> stack_trace =
|
Handle<Object> stack_trace = CaptureSimpleStackTrace(error_object, caller);
|
||||||
CaptureSimpleStackTrace(error_object, mode, caller);
|
|
||||||
// TODO(jgruber): Set back to STRICT once we have eagerly formatted traces.
|
// TODO(jgruber): Set back to STRICT once we have eagerly formatted traces.
|
||||||
RETURN_ON_EXCEPTION(
|
RETURN_ON_EXCEPTION(
|
||||||
this, JSReceiver::SetProperty(error_object, key, stack_trace, SLOPPY),
|
this, JSReceiver::SetProperty(error_object, key, stack_trace, SLOPPY),
|
||||||
@ -959,14 +960,20 @@ bool Isolate::MayAccess(Handle<Context> accessing_context,
|
|||||||
Object* Isolate::StackOverflow() {
|
Object* Isolate::StackOverflow() {
|
||||||
DisallowJavascriptExecution no_js(this);
|
DisallowJavascriptExecution no_js(this);
|
||||||
HandleScope scope(this);
|
HandleScope scope(this);
|
||||||
|
// At this point we cannot create an Error object using its javascript
|
||||||
Handle<JSFunction> fun = range_error_function();
|
// constructor. Instead, we copy the pre-constructed boilerplate and
|
||||||
Handle<Object> msg = factory()->NewStringFromAsciiChecked(
|
// attach the stack trace as a hidden property.
|
||||||
MessageTemplate::TemplateString(MessageTemplate::kStackOverflow));
|
|
||||||
Handle<Object> exception;
|
Handle<Object> exception;
|
||||||
ASSIGN_RETURN_FAILURE_ON_EXCEPTION(
|
if (bootstrapper()->IsActive()) {
|
||||||
this, exception, ConstructError(this, fun, fun, msg, SKIP_NONE, true));
|
// There is no boilerplate to use during bootstrapping.
|
||||||
|
exception = factory()->NewStringFromAsciiChecked(
|
||||||
|
MessageTemplate::TemplateString(MessageTemplate::kStackOverflow));
|
||||||
|
} else {
|
||||||
|
Handle<JSObject> boilerplate = stack_overflow_boilerplate();
|
||||||
|
Handle<JSObject> copy = factory()->CopyJSObject(boilerplate);
|
||||||
|
CaptureAndSetSimpleStackTrace(copy, factory()->undefined_value());
|
||||||
|
exception = copy;
|
||||||
|
}
|
||||||
Throw(*exception, nullptr);
|
Throw(*exception, nullptr);
|
||||||
|
|
||||||
#ifdef VERIFY_HEAP
|
#ifdef VERIFY_HEAP
|
||||||
|
@ -696,13 +696,12 @@ class Isolate {
|
|||||||
int frame_limit,
|
int frame_limit,
|
||||||
StackTrace::StackTraceOptions options);
|
StackTrace::StackTraceOptions options);
|
||||||
Handle<Object> CaptureSimpleStackTrace(Handle<JSReceiver> error_object,
|
Handle<Object> CaptureSimpleStackTrace(Handle<JSReceiver> error_object,
|
||||||
FrameSkipMode mode,
|
|
||||||
Handle<Object> caller);
|
Handle<Object> caller);
|
||||||
MaybeHandle<JSReceiver> CaptureAndSetDetailedStackTrace(
|
MaybeHandle<JSReceiver> CaptureAndSetDetailedStackTrace(
|
||||||
Handle<JSReceiver> error_object);
|
Handle<JSReceiver> error_object);
|
||||||
MaybeHandle<JSReceiver> CaptureAndSetSimpleStackTrace(
|
MaybeHandle<JSReceiver> CaptureAndSetSimpleStackTrace(
|
||||||
Handle<JSReceiver> error_object, FrameSkipMode mode,
|
Handle<JSReceiver> error_object,
|
||||||
Handle<Object> caller);
|
Handle<Object> caller = Handle<Object>());
|
||||||
Handle<JSArray> GetDetailedStackTrace(Handle<JSObject> error_object);
|
Handle<JSArray> GetDetailedStackTrace(Handle<JSObject> error_object);
|
||||||
|
|
||||||
// Returns if the given context may access the given global object. If
|
// Returns if the given context may access the given global object. If
|
||||||
|
@ -637,6 +637,10 @@ function MakeURIError() {
|
|||||||
return MakeGenericError(GlobalURIError, kURIMalformed);
|
return MakeGenericError(GlobalURIError, kURIMalformed);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Boilerplate for exceptions for stack overflows. Used from
|
||||||
|
// Isolate::StackOverflow().
|
||||||
|
var StackOverflowBoilerplate = MakeRangeError(kStackOverflow);
|
||||||
|
|
||||||
%InstallToContext([
|
%InstallToContext([
|
||||||
"error_format_stack_trace", FormatStackTrace,
|
"error_format_stack_trace", FormatStackTrace,
|
||||||
"get_stack_trace_line_fun", GetStackTraceLine,
|
"get_stack_trace_line_fun", GetStackTraceLine,
|
||||||
@ -647,6 +651,7 @@ function MakeURIError() {
|
|||||||
"message_get_line_number", GetLineNumber,
|
"message_get_line_number", GetLineNumber,
|
||||||
"message_get_source_line", GetSourceLine,
|
"message_get_source_line", GetSourceLine,
|
||||||
"no_side_effects_to_string_fun", NoSideEffectsToString,
|
"no_side_effects_to_string_fun", NoSideEffectsToString,
|
||||||
|
"stack_overflow_boilerplate", StackOverflowBoilerplate,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
utils.Export(function(to) {
|
utils.Export(function(to) {
|
||||||
|
@ -5,7 +5,6 @@
|
|||||||
#include "src/messages.h"
|
#include "src/messages.h"
|
||||||
|
|
||||||
#include "src/api.h"
|
#include "src/api.h"
|
||||||
#include "src/bootstrapper.h"
|
|
||||||
#include "src/execution.h"
|
#include "src/execution.h"
|
||||||
#include "src/isolate-inl.h"
|
#include "src/isolate-inl.h"
|
||||||
#include "src/keys.h"
|
#include "src/keys.h"
|
||||||
@ -446,52 +445,6 @@ MaybeHandle<String> MessageTemplate::FormatMessage(int template_index,
|
|||||||
return builder.Finish();
|
return builder.Finish();
|
||||||
}
|
}
|
||||||
|
|
||||||
MaybeHandle<Object> ConstructError(Isolate* isolate, Handle<JSFunction> target,
|
|
||||||
Handle<Object> new_target,
|
|
||||||
Handle<Object> message, FrameSkipMode mode,
|
|
||||||
bool suppress_detailed_trace) {
|
|
||||||
// 1. If NewTarget is undefined, let newTarget be the active function object,
|
|
||||||
// else let newTarget be NewTarget.
|
|
||||||
|
|
||||||
Handle<JSReceiver> new_target_recv =
|
|
||||||
new_target->IsJSReceiver() ? Handle<JSReceiver>::cast(new_target)
|
|
||||||
: Handle<JSReceiver>::cast(target);
|
|
||||||
|
|
||||||
// 2. Let O be ? OrdinaryCreateFromConstructor(newTarget, "%ErrorPrototype%",
|
|
||||||
// « [[ErrorData]] »).
|
|
||||||
Handle<JSObject> err;
|
|
||||||
ASSIGN_RETURN_ON_EXCEPTION(isolate, err,
|
|
||||||
JSObject::New(target, new_target_recv), Object);
|
|
||||||
|
|
||||||
// 3. If message is not undefined, then
|
|
||||||
// a. Let msg be ? ToString(message).
|
|
||||||
// b. Let msgDesc be the PropertyDescriptor{[[Value]]: msg, [[Writable]]:
|
|
||||||
// true, [[Enumerable]]: false, [[Configurable]]: true}.
|
|
||||||
// c. Perform ! DefinePropertyOrThrow(O, "message", msgDesc).
|
|
||||||
// 4. Return O.
|
|
||||||
|
|
||||||
if (!message->IsUndefined(isolate)) {
|
|
||||||
Handle<String> msg_string;
|
|
||||||
ASSIGN_RETURN_ON_EXCEPTION(isolate, msg_string,
|
|
||||||
Object::ToString(isolate, message), Object);
|
|
||||||
RETURN_ON_EXCEPTION(isolate, JSObject::SetOwnPropertyIgnoreAttributes(
|
|
||||||
err, isolate->factory()->message_string(),
|
|
||||||
msg_string, DONT_ENUM),
|
|
||||||
Object);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Optionally capture a more detailed stack trace for the message.
|
|
||||||
if (!suppress_detailed_trace) {
|
|
||||||
RETURN_ON_EXCEPTION(isolate, isolate->CaptureAndSetDetailedStackTrace(err),
|
|
||||||
Object);
|
|
||||||
}
|
|
||||||
// Capture a simple stack trace for the stack property.
|
|
||||||
RETURN_ON_EXCEPTION(isolate, isolate->CaptureAndSetSimpleStackTrace(
|
|
||||||
err, mode, Handle<Object>()),
|
|
||||||
Object);
|
|
||||||
|
|
||||||
return err;
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace internal
|
} // namespace internal
|
||||||
} // namespace v8
|
} // namespace v8
|
||||||
|
@ -71,21 +71,6 @@ class CallSite {
|
|||||||
uint32_t wasm_func_index_ = static_cast<uint32_t>(-1);
|
uint32_t wasm_func_index_ = static_cast<uint32_t>(-1);
|
||||||
};
|
};
|
||||||
|
|
||||||
// Determines how stack trace collection skips frames.
|
|
||||||
enum FrameSkipMode {
|
|
||||||
// Unconditionally skips the first frame. Used e.g. when the Error constructor
|
|
||||||
// is called, in which case the first frame is always a BUILTIN_EXIT frame.
|
|
||||||
SKIP_FIRST,
|
|
||||||
// Skip all frames until a specified caller function is seen.
|
|
||||||
SKIP_UNTIL_SEEN,
|
|
||||||
SKIP_NONE,
|
|
||||||
};
|
|
||||||
|
|
||||||
MaybeHandle<Object> ConstructError(Isolate* isolate, Handle<JSFunction> target,
|
|
||||||
Handle<Object> new_target,
|
|
||||||
Handle<Object> message, FrameSkipMode mode,
|
|
||||||
bool suppress_detailed_trace);
|
|
||||||
|
|
||||||
#define MESSAGE_TEMPLATES(T) \
|
#define MESSAGE_TEMPLATES(T) \
|
||||||
/* Error */ \
|
/* Error */ \
|
||||||
T(None, "") \
|
T(None, "") \
|
||||||
|
Loading…
Reference in New Issue
Block a user