summaryrefslogtreecommitdiffstats
path: root/js/src/jsexn.cpp
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--js/src/jsexn.cpp883
1 files changed, 883 insertions, 0 deletions
diff --git a/js/src/jsexn.cpp b/js/src/jsexn.cpp
new file mode 100644
index 0000000000..dd4e31c86a
--- /dev/null
+++ b/js/src/jsexn.cpp
@@ -0,0 +1,883 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
+ * vim: set ts=8 sts=2 et sw=2 tw=80:
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+/*
+ * JS standard exception implementation.
+ */
+
+#include "jsexn.h"
+
+#include "mozilla/Assertions.h"
+#include "mozilla/Maybe.h"
+#include "mozilla/ScopeExit.h"
+
+#include <new>
+#include <stdarg.h>
+#include <stdint.h>
+#include <stdio.h>
+#include <string.h>
+#include <utility>
+
+#include "jsapi.h"
+#include "jsfriendapi.h"
+#include "jstypes.h"
+
+#include "frontend/FrontendContext.h" // AutoReportFrontendContext
+#include "js/CharacterEncoding.h" // JS::UTF8Chars, JS::ConstUTF8CharsZ
+#include "js/Class.h"
+#include "js/ColumnNumber.h" // JS::ColumnNumberOneOrigin, JS::TaggedColumnNumberOneOrigin
+#include "js/Conversions.h"
+#include "js/ErrorReport.h" // JS::PrintError
+#include "js/Exception.h" // JS::ExceptionStack
+#include "js/experimental/TypedData.h" // JS_IsArrayBufferViewObject
+#include "js/friend/ErrorMessages.h" // JSErrNum, js::GetErrorMessage, JSMSG_*
+#include "js/Object.h" // JS::GetBuiltinClass
+#include "js/PropertyAndElement.h" // JS_GetProperty, JS_HasProperty
+#include "js/SavedFrameAPI.h"
+#include "js/Stack.h"
+#include "js/UniquePtr.h"
+#include "js/Value.h"
+#include "js/Warnings.h" // JS::{,Set}WarningReporter
+#include "js/Wrapper.h"
+#include "util/Memory.h"
+#include "util/StringBuffer.h"
+#include "vm/Compartment.h"
+#include "vm/ErrorObject.h"
+#include "vm/FrameIter.h" // js::NonBuiltinFrameIter
+#include "vm/JSAtomUtils.h" // ClassName
+#include "vm/JSContext.h"
+#include "vm/JSObject.h"
+#include "vm/JSScript.h"
+#include "vm/Realm.h"
+#include "vm/SavedFrame.h"
+#include "vm/SavedStacks.h"
+#include "vm/SelfHosting.h"
+#include "vm/Stack.h"
+#include "vm/StringType.h"
+#include "vm/SymbolType.h"
+#include "wasm/WasmJS.h" // WasmExceptionObject
+
+#include "vm/Compartment-inl.h"
+#include "vm/ErrorObject-inl.h"
+#include "vm/JSContext-inl.h"
+#include "vm/JSObject-inl.h"
+#include "vm/ObjectOperations-inl.h" // js::GetProperty
+#include "vm/SavedStacks-inl.h"
+
+using namespace js;
+
+using JS::SavedFrameSelfHosted;
+
+size_t ExtraMallocSize(JSErrorReport* report) {
+ if (report->linebuf()) {
+ /*
+ * Count with null terminator and alignment.
+ * See CopyExtraData for the details about alignment.
+ */
+ return (report->linebufLength() + 1) * sizeof(char16_t) + 1;
+ }
+
+ return 0;
+}
+
+size_t ExtraMallocSize(JSErrorNotes::Note* note) { return 0; }
+
+bool CopyExtraData(JSContext* cx, uint8_t** cursor, JSErrorReport* copy,
+ JSErrorReport* report) {
+ if (report->linebuf()) {
+ /*
+ * Make sure cursor is properly aligned for char16_t for platforms
+ * which need it and it's at the end of the buffer on exit.
+ */
+ size_t alignment_backlog = 0;
+ if (size_t(*cursor) % 2) {
+ (*cursor)++;
+ } else {
+ alignment_backlog = 1;
+ }
+
+ size_t linebufSize = (report->linebufLength() + 1) * sizeof(char16_t);
+ const char16_t* linebufCopy = (const char16_t*)(*cursor);
+ js_memcpy(*cursor, report->linebuf(), linebufSize);
+ *cursor += linebufSize + alignment_backlog;
+ copy->initBorrowedLinebuf(linebufCopy, report->linebufLength(),
+ report->tokenOffset());
+ }
+
+ /* Copy non-pointer members. */
+ copy->isMuted = report->isMuted;
+ copy->exnType = report->exnType;
+ copy->isWarning_ = report->isWarning_;
+
+ /* Deep copy notes. */
+ if (report->notes) {
+ auto copiedNotes = report->notes->copy(cx);
+ if (!copiedNotes) {
+ return false;
+ }
+ copy->notes = std::move(copiedNotes);
+ } else {
+ copy->notes.reset(nullptr);
+ }
+
+ return true;
+}
+
+bool CopyExtraData(JSContext* cx, uint8_t** cursor, JSErrorNotes::Note* copy,
+ JSErrorNotes::Note* report) {
+ return true;
+}
+
+template <typename T>
+static UniquePtr<T> CopyErrorHelper(JSContext* cx, T* report) {
+ /*
+ * We use a single malloc block to make a deep copy of JSErrorReport or
+ * JSErrorNotes::Note, except JSErrorNotes linked from JSErrorReport with
+ * the following layout:
+ * JSErrorReport or JSErrorNotes::Note
+ * char array with characters for message_
+ * char array with characters for filename
+ * char16_t array with characters for linebuf (only for JSErrorReport)
+ * Such layout together with the properties enforced by the following
+ * asserts does not need any extra alignment padding.
+ */
+ static_assert(sizeof(T) % sizeof(const char*) == 0);
+ static_assert(sizeof(const char*) % sizeof(char16_t) == 0);
+
+ size_t filenameSize =
+ report->filename ? strlen(report->filename.c_str()) + 1 : 0;
+ size_t messageSize = 0;
+ if (report->message()) {
+ messageSize = strlen(report->message().c_str()) + 1;
+ }
+
+ /*
+ * The mallocSize can not overflow since it represents the sum of the
+ * sizes of already allocated objects.
+ */
+ size_t mallocSize =
+ sizeof(T) + messageSize + filenameSize + ExtraMallocSize(report);
+ uint8_t* cursor = cx->pod_calloc<uint8_t>(mallocSize);
+ if (!cursor) {
+ return nullptr;
+ }
+
+ UniquePtr<T> copy(new (cursor) T());
+ cursor += sizeof(T);
+
+ if (report->message()) {
+ copy->initBorrowedMessage((const char*)cursor);
+ js_memcpy(cursor, report->message().c_str(), messageSize);
+ cursor += messageSize;
+ }
+
+ if (report->filename) {
+ copy->filename = JS::ConstUTF8CharsZ((const char*)cursor);
+ js_memcpy(cursor, report->filename.c_str(), filenameSize);
+ cursor += filenameSize;
+ }
+
+ if (!CopyExtraData(cx, &cursor, copy.get(), report)) {
+ return nullptr;
+ }
+
+ MOZ_ASSERT(cursor == (uint8_t*)copy.get() + mallocSize);
+
+ // errorMessageName should be static.
+ copy->errorMessageName = report->errorMessageName;
+
+ /* Copy non-pointer members. */
+ copy->sourceId = report->sourceId;
+ copy->lineno = report->lineno;
+ copy->column = report->column;
+ copy->errorNumber = report->errorNumber;
+
+ return copy;
+}
+
+UniquePtr<JSErrorNotes::Note> js::CopyErrorNote(JSContext* cx,
+ JSErrorNotes::Note* note) {
+ return CopyErrorHelper(cx, note);
+}
+
+UniquePtr<JSErrorReport> js::CopyErrorReport(JSContext* cx,
+ JSErrorReport* report) {
+ return CopyErrorHelper(cx, report);
+}
+
+struct SuppressErrorsGuard {
+ JSContext* cx;
+ JS::WarningReporter prevReporter;
+ JS::AutoSaveExceptionState prevState;
+
+ explicit SuppressErrorsGuard(JSContext* cx)
+ : cx(cx),
+ prevReporter(JS::SetWarningReporter(cx, nullptr)),
+ prevState(cx) {}
+
+ ~SuppressErrorsGuard() { JS::SetWarningReporter(cx, prevReporter); }
+};
+
+// Cut off the stack if it gets too deep (most commonly for infinite recursion
+// errors).
+static const size_t MAX_REPORTED_STACK_DEPTH = 1u << 7;
+
+bool js::CaptureStack(JSContext* cx, MutableHandleObject stack) {
+ return CaptureCurrentStack(
+ cx, stack, JS::StackCapture(JS::MaxFrames(MAX_REPORTED_STACK_DEPTH)));
+}
+
+JSString* js::ComputeStackString(JSContext* cx) {
+ SuppressErrorsGuard seg(cx);
+
+ RootedObject stack(cx);
+ if (!CaptureStack(cx, &stack)) {
+ return nullptr;
+ }
+
+ RootedString str(cx);
+ if (!BuildStackString(cx, cx->realm()->principals(), stack, &str)) {
+ return nullptr;
+ }
+
+ return str.get();
+}
+
+JSErrorReport* js::ErrorFromException(JSContext* cx, HandleObject objArg) {
+ // It's ok to UncheckedUnwrap here, since all we do is get the
+ // JSErrorReport, and consumers are careful with the information they get
+ // from that anyway. Anyone doing things that would expose anything in the
+ // JSErrorReport to page script either does a security check on the
+ // JSErrorReport's principal or also tries to do toString on our object and
+ // will fail if they can't unwrap it.
+ RootedObject obj(cx, UncheckedUnwrap(objArg));
+ if (!obj->is<ErrorObject>()) {
+ return nullptr;
+ }
+
+ JSErrorReport* report = obj->as<ErrorObject>().getOrCreateErrorReport(cx);
+ if (!report) {
+ MOZ_ASSERT(cx->isThrowingOutOfMemory());
+ cx->recoverFromOutOfMemory();
+ }
+
+ return report;
+}
+
+JS_PUBLIC_API JSObject* JS::ExceptionStackOrNull(HandleObject objArg) {
+ ErrorObject* errorObject = objArg->maybeUnwrapIf<ErrorObject>();
+ if (errorObject) {
+ return errorObject->stack();
+ }
+
+ WasmExceptionObject* wasmObject =
+ objArg->maybeUnwrapIf<WasmExceptionObject>();
+ if (wasmObject) {
+ return wasmObject->stack();
+ }
+
+ return nullptr;
+}
+
+JS_PUBLIC_API JSLinearString* js::GetErrorTypeName(JSContext* cx,
+ int16_t exnType) {
+ /*
+ * JSEXN_INTERNALERR returns null to prevent that "InternalError: "
+ * is prepended before "uncaught exception: "
+ */
+ if (exnType < 0 || exnType >= JSEXN_LIMIT || exnType == JSEXN_INTERNALERR ||
+ exnType == JSEXN_WARN || exnType == JSEXN_NOTE) {
+ return nullptr;
+ }
+ JSProtoKey key = GetExceptionProtoKey(JSExnType(exnType));
+ return ClassName(key, cx);
+}
+
+bool js::ErrorToException(JSContext* cx, JSErrorReport* reportp,
+ JSErrorCallback callback, void* userRef) {
+ MOZ_ASSERT(!reportp->isWarning());
+
+ // Find the exception index associated with this error.
+ JSErrNum errorNumber = static_cast<JSErrNum>(reportp->errorNumber);
+ if (!callback) {
+ callback = GetErrorMessage;
+ }
+ const JSErrorFormatString* errorString = callback(userRef, errorNumber);
+ JSExnType exnType =
+ errorString ? static_cast<JSExnType>(errorString->exnType) : JSEXN_ERR;
+ MOZ_ASSERT(exnType < JSEXN_ERROR_LIMIT);
+
+ // Prevent infinite recursion.
+ if (cx->generatingError) {
+ return false;
+ }
+
+ cx->generatingError = true;
+ auto restore = mozilla::MakeScopeExit([cx] { cx->generatingError = false; });
+
+ // Create an exception object.
+ RootedString messageStr(cx, reportp->newMessageString(cx));
+ if (!messageStr) {
+ return false;
+ }
+
+ Rooted<JSString*> fileName(cx);
+ if (const char* filename = reportp->filename.c_str()) {
+ fileName =
+ JS_NewStringCopyUTF8N(cx, JS::UTF8Chars(filename, strlen(filename)));
+ if (!fileName) {
+ return false;
+ }
+ } else {
+ fileName = cx->emptyString();
+ }
+
+ uint32_t sourceId = reportp->sourceId;
+ uint32_t lineNumber = reportp->lineno;
+ JS::ColumnNumberOneOrigin columnNumber = reportp->column;
+
+ // Error reports don't provide a |cause|, so we default to |Nothing| here.
+ auto cause = JS::NothingHandleValue;
+
+ RootedObject stack(cx);
+ if (!CaptureStack(cx, &stack)) {
+ return false;
+ }
+
+ UniquePtr<JSErrorReport> report = CopyErrorReport(cx, reportp);
+ if (!report) {
+ return false;
+ }
+
+ ErrorObject* errObject =
+ ErrorObject::create(cx, exnType, stack, fileName, sourceId, lineNumber,
+ columnNumber, std::move(report), messageStr, cause);
+ if (!errObject) {
+ return false;
+ }
+
+ // Throw it.
+ RootedValue errValue(cx, ObjectValue(*errObject));
+ Rooted<SavedFrame*> nstack(cx);
+ if (stack) {
+ nstack = &stack->as<SavedFrame>();
+ }
+ cx->setPendingException(errValue, nstack);
+ return true;
+}
+
+using SniffingBehavior = JS::ErrorReportBuilder::SniffingBehavior;
+
+static bool IsDuckTypedErrorObject(JSContext* cx, HandleObject exnObject,
+ const char** filename_strp) {
+ /*
+ * This function is called from ErrorReport::init and so should not generate
+ * any new exceptions.
+ */
+ AutoClearPendingException acpe(cx);
+
+ bool found;
+ if (!JS_HasProperty(cx, exnObject, "message", &found) || !found) {
+ return false;
+ }
+
+ // First try "filename".
+ const char* filename_str = *filename_strp;
+ if (!JS_HasProperty(cx, exnObject, filename_str, &found)) {
+ return false;
+ }
+ if (!found) {
+ // If that doesn't work, try "fileName".
+ filename_str = "fileName";
+ if (!JS_HasProperty(cx, exnObject, filename_str, &found) || !found) {
+ return false;
+ }
+ }
+
+ if (!JS_HasProperty(cx, exnObject, "lineNumber", &found) || !found) {
+ return false;
+ }
+
+ *filename_strp = filename_str;
+ return true;
+}
+
+static bool GetPropertyNoException(JSContext* cx, HandleObject obj,
+ SniffingBehavior behavior,
+ Handle<PropertyName*> name,
+ MutableHandleValue vp) {
+ // This function has no side-effects so always use it.
+ if (GetPropertyPure(cx, obj, NameToId(name), vp.address())) {
+ return true;
+ }
+
+ if (behavior == SniffingBehavior::WithSideEffects) {
+ AutoClearPendingException acpe(cx);
+ return GetProperty(cx, obj, obj, name, vp);
+ }
+
+ return false;
+}
+
+// Create a new error message similar to what Error.prototype.toString would
+// produce when called on an object with those property values for name and
+// message.
+static JSString* FormatErrorMessage(JSContext* cx, HandleString name,
+ HandleString message) {
+ if (name && message) {
+ AutoClearPendingException acpe(cx);
+ JSStringBuilder sb(cx);
+
+ // Prefix the message with the error type, if it exists.
+ if (!sb.append(name) || !sb.append(": ") || !sb.append(message)) {
+ return nullptr;
+ }
+
+ return sb.finishString();
+ }
+
+ return name ? name : message;
+}
+
+static JSString* ErrorReportToString(JSContext* cx, HandleObject exn,
+ JSErrorReport* reportp,
+ SniffingBehavior behavior) {
+ // The error object might have custom `name` overwriting the exnType in the
+ // error report. Try getting that property and use the exnType as a fallback.
+ RootedString name(cx);
+ RootedValue nameV(cx);
+ if (GetPropertyNoException(cx, exn, behavior, cx->names().name, &nameV) &&
+ nameV.isString()) {
+ name = nameV.toString();
+ }
+
+ // We do NOT want to use GetErrorTypeName() here because it will not do the
+ // "right thing" for JSEXN_INTERNALERR. That is, the caller of this API
+ // expects that "InternalError: " will be prepended but GetErrorTypeName
+ // goes out of its way to avoid this.
+ if (!name) {
+ JSExnType type = static_cast<JSExnType>(reportp->exnType);
+ if (type != JSEXN_WARN && type != JSEXN_NOTE) {
+ name = ClassName(GetExceptionProtoKey(type), cx);
+ }
+ }
+
+ RootedString message(cx);
+ RootedValue messageV(cx);
+ if (GetPropertyNoException(cx, exn, behavior, cx->names().message,
+ &messageV) &&
+ messageV.isString()) {
+ message = messageV.toString();
+ }
+
+ if (!message) {
+ message = reportp->newMessageString(cx);
+ if (!message) {
+ return nullptr;
+ }
+ }
+
+ return FormatErrorMessage(cx, name, message);
+}
+
+JS::ErrorReportBuilder::ErrorReportBuilder(JSContext* cx)
+ : reportp(nullptr), exnObject(cx) {}
+
+JS::ErrorReportBuilder::~ErrorReportBuilder() = default;
+
+bool JS::ErrorReportBuilder::init(JSContext* cx,
+ const JS::ExceptionStack& exnStack,
+ SniffingBehavior sniffingBehavior) {
+ MOZ_ASSERT(!cx->isExceptionPending());
+ MOZ_ASSERT(!reportp);
+
+ if (exnStack.exception().isObject()) {
+ // Because ToString below could error and an exception object could become
+ // unrooted, we must root our exception object, if any.
+ exnObject = &exnStack.exception().toObject();
+ reportp = ErrorFromException(cx, exnObject);
+ }
+
+ // Be careful not to invoke ToString if we've already successfully extracted
+ // an error report, since the exception might be wrapped in a security
+ // wrapper, and ToString-ing it might throw.
+ RootedString str(cx);
+ if (reportp) {
+ str = ErrorReportToString(cx, exnObject, reportp, sniffingBehavior);
+ } else if (exnStack.exception().isSymbol()) {
+ RootedValue strVal(cx);
+ if (js::SymbolDescriptiveString(cx, exnStack.exception().toSymbol(),
+ &strVal)) {
+ str = strVal.toString();
+ } else {
+ str = nullptr;
+ }
+ } else if (exnObject && sniffingBehavior == NoSideEffects) {
+ str = cx->names().Object;
+ } else {
+ str = js::ToString<CanGC>(cx, exnStack.exception());
+ }
+
+ if (!str) {
+ cx->clearPendingException();
+ }
+
+ // If ErrorFromException didn't get us a JSErrorReport, then the object
+ // was not an ErrorObject, security-wrapped or otherwise. However, it might
+ // still quack like one. Give duck-typing a chance. We start by looking for
+ // "filename" (all lowercase), since that's where DOMExceptions store their
+ // filename. Then we check "fileName", which is where Errors store it. We
+ // have to do it in that order, because DOMExceptions have Error.prototype
+ // on their proto chain, and hence also have a "fileName" property, but its
+ // value is "".
+ const char* filename_str = "filename";
+ if (!reportp && exnObject && sniffingBehavior == WithSideEffects &&
+ IsDuckTypedErrorObject(cx, exnObject, &filename_str)) {
+ // Temporary value for pulling properties off of duck-typed objects.
+ RootedValue val(cx);
+
+ RootedString name(cx);
+ if (JS_GetProperty(cx, exnObject, "name", &val) && val.isString()) {
+ name = val.toString();
+ } else {
+ cx->clearPendingException();
+ }
+
+ RootedString msg(cx);
+ if (JS_GetProperty(cx, exnObject, "message", &val) && val.isString()) {
+ msg = val.toString();
+ } else {
+ cx->clearPendingException();
+ }
+
+ // If we have the right fields, override the ToString we performed on
+ // the exception object above with something built out of its quacks
+ // (i.e. as much of |NameQuack: MessageQuack| as we can make).
+ str = FormatErrorMessage(cx, name, msg);
+
+ {
+ AutoClearPendingException acpe(cx);
+ if (JS_GetProperty(cx, exnObject, filename_str, &val)) {
+ RootedString tmp(cx, js::ToString<CanGC>(cx, val));
+ if (tmp) {
+ filename = JS_EncodeStringToUTF8(cx, tmp);
+ }
+ }
+ }
+ if (!filename) {
+ filename = DuplicateString("");
+ if (!filename) {
+ ReportOutOfMemory(cx);
+ return false;
+ }
+ }
+
+ uint32_t lineno;
+ if (!JS_GetProperty(cx, exnObject, "lineNumber", &val) ||
+ !ToUint32(cx, val, &lineno)) {
+ cx->clearPendingException();
+ lineno = 0;
+ }
+
+ uint32_t column;
+ if (!JS_GetProperty(cx, exnObject, "columnNumber", &val) ||
+ !ToUint32(cx, val, &column)) {
+ cx->clearPendingException();
+ column = 0;
+ }
+
+ reportp = &ownedReport;
+ new (reportp) JSErrorReport();
+ ownedReport.filename = JS::ConstUTF8CharsZ(filename.get());
+ ownedReport.lineno = lineno;
+ ownedReport.exnType = JSEXN_INTERNALERR;
+ ownedReport.column = JS::ColumnNumberOneOrigin(column);
+
+ if (str) {
+ // Note that using |str| for |message_| here is kind of wrong,
+ // because |str| is supposed to be of the format
+ // |ErrorName: ErrorMessage|, and |message_| is supposed to
+ // correspond to |ErrorMessage|. But this is what we've
+ // historically done for duck-typed error objects.
+ //
+ // If only this stuff could get specced one day...
+ if (auto utf8 = JS_EncodeStringToUTF8(cx, str)) {
+ ownedReport.initOwnedMessage(utf8.release());
+ } else {
+ cx->clearPendingException();
+ str = nullptr;
+ }
+ }
+ }
+
+ const char* utf8Message = nullptr;
+ if (str) {
+ toStringResultBytesStorage = JS_EncodeStringToUTF8(cx, str);
+ utf8Message = toStringResultBytesStorage.get();
+ if (!utf8Message) {
+ cx->clearPendingException();
+ }
+ }
+ if (!utf8Message) {
+ utf8Message = "unknown (can't convert to string)";
+ }
+
+ if (!reportp) {
+ // This is basically an inlined version of
+ //
+ // JS_ReportErrorNumberUTF8(cx, GetErrorMessage, nullptr,
+ // JSMSG_UNCAUGHT_EXCEPTION, utf8Message);
+ //
+ // but without the reporting bits. Instead it just puts all
+ // the stuff we care about in our ownedReport and message_.
+ if (!populateUncaughtExceptionReportUTF8(cx, exnStack.stack(),
+ utf8Message)) {
+ // Just give up. We're out of memory or something; not much we can
+ // do here.
+ return false;
+ }
+ } else {
+ toStringResult_ = JS::ConstUTF8CharsZ(utf8Message, strlen(utf8Message));
+ }
+
+ return true;
+}
+
+bool JS::ErrorReportBuilder::populateUncaughtExceptionReportUTF8(
+ JSContext* cx, HandleObject stack, ...) {
+ va_list ap;
+ va_start(ap, stack);
+ bool ok = populateUncaughtExceptionReportUTF8VA(cx, stack, ap);
+ va_end(ap);
+ return ok;
+}
+
+bool JS::ErrorReportBuilder::populateUncaughtExceptionReportUTF8VA(
+ JSContext* cx, HandleObject stack, va_list ap) {
+ new (&ownedReport) JSErrorReport();
+ ownedReport.isWarning_ = false;
+ ownedReport.errorNumber = JSMSG_UNCAUGHT_EXCEPTION;
+
+ bool skippedAsync;
+ Rooted<SavedFrame*> frame(
+ cx, UnwrapSavedFrame(cx, cx->realm()->principals(), stack,
+ SavedFrameSelfHosted::Exclude, skippedAsync));
+ if (frame) {
+ filename = StringToNewUTF8CharsZ(cx, *frame->getSource());
+ if (!filename) {
+ return false;
+ }
+
+ // |ownedReport.filename| inherits the lifetime of |ErrorReport::filename|.
+ ownedReport.filename = JS::ConstUTF8CharsZ(filename.get());
+ ownedReport.sourceId = frame->getSourceId();
+ ownedReport.lineno = frame->getLine();
+ ownedReport.column =
+ JS::ColumnNumberOneOrigin(frame->getColumn().oneOriginValue());
+ ownedReport.isMuted = frame->getMutedErrors();
+ } else {
+ // XXXbz this assumes the stack we have right now is still
+ // related to our exception object.
+ NonBuiltinFrameIter iter(cx, cx->realm()->principals());
+ if (!iter.done()) {
+ ownedReport.filename = JS::ConstUTF8CharsZ(iter.filename());
+ JS::TaggedColumnNumberOneOrigin column;
+ ownedReport.sourceId =
+ iter.hasScript() ? iter.script()->scriptSource()->id() : 0;
+ ownedReport.lineno = iter.computeLine(&column);
+ ownedReport.column = JS::ColumnNumberOneOrigin(column.oneOriginValue());
+ ownedReport.isMuted = iter.mutedErrors();
+ }
+ }
+
+ AutoReportFrontendContext fc(cx);
+ if (!ExpandErrorArgumentsVA(&fc, GetErrorMessage, nullptr,
+ JSMSG_UNCAUGHT_EXCEPTION, ArgumentsAreUTF8,
+ &ownedReport, ap)) {
+ return false;
+ }
+
+ toStringResult_ = ownedReport.message();
+ reportp = &ownedReport;
+ return true;
+}
+
+JSObject* js::CopyErrorObject(JSContext* cx, Handle<ErrorObject*> err) {
+ UniquePtr<JSErrorReport> copyReport;
+ if (JSErrorReport* errorReport = err->getErrorReport()) {
+ copyReport = CopyErrorReport(cx, errorReport);
+ if (!copyReport) {
+ return nullptr;
+ }
+ }
+
+ RootedString message(cx, err->getMessage());
+ if (message && !cx->compartment()->wrap(cx, &message)) {
+ return nullptr;
+ }
+ RootedString fileName(cx, err->fileName(cx));
+ if (!cx->compartment()->wrap(cx, &fileName)) {
+ return nullptr;
+ }
+ RootedObject stack(cx, err->stack());
+ if (!cx->compartment()->wrap(cx, &stack)) {
+ return nullptr;
+ }
+ if (stack && JS_IsDeadWrapper(stack)) {
+ // ErrorObject::create expects |stack| to be either nullptr or a (possibly
+ // wrapped) SavedFrame instance.
+ stack = nullptr;
+ }
+ Rooted<mozilla::Maybe<Value>> cause(cx, mozilla::Nothing());
+ if (auto maybeCause = err->getCause()) {
+ RootedValue errorCause(cx, maybeCause.value());
+ if (!cx->compartment()->wrap(cx, &errorCause)) {
+ return nullptr;
+ }
+ cause = mozilla::Some(errorCause.get());
+ }
+ uint32_t sourceId = err->sourceId();
+ uint32_t lineNumber = err->lineNumber();
+ JS::ColumnNumberOneOrigin columnNumber = err->columnNumber();
+ JSExnType errorType = err->type();
+
+ // Create the Error object.
+ return ErrorObject::create(cx, errorType, stack, fileName, sourceId,
+ lineNumber, columnNumber, std::move(copyReport),
+ message, cause);
+}
+
+JS_PUBLIC_API bool JS::CreateError(JSContext* cx, JSExnType type,
+ HandleObject stack, HandleString fileName,
+ uint32_t lineNumber,
+ JS::ColumnNumberOneOrigin columnNumber,
+ JSErrorReport* report, HandleString message,
+ Handle<mozilla::Maybe<Value>> cause,
+ MutableHandleValue rval) {
+ cx->check(stack, fileName, message);
+ AssertObjectIsSavedFrameOrWrapper(cx, stack);
+
+ js::UniquePtr<JSErrorReport> rep;
+ if (report) {
+ rep = CopyErrorReport(cx, report);
+ if (!rep) {
+ return false;
+ }
+ }
+
+ JSObject* obj =
+ js::ErrorObject::create(cx, type, stack, fileName, 0, lineNumber,
+ columnNumber, std::move(rep), message, cause);
+ if (!obj) {
+ return false;
+ }
+
+ rval.setObject(*obj);
+ return true;
+}
+
+const char* js::ValueToSourceForError(JSContext* cx, HandleValue val,
+ UniqueChars& bytes) {
+ if (val.isUndefined()) {
+ return "undefined";
+ }
+
+ if (val.isNull()) {
+ return "null";
+ }
+
+ AutoClearPendingException acpe(cx);
+
+ RootedString str(cx, JS_ValueToSource(cx, val));
+ if (!str) {
+ return "<<error converting value to string>>";
+ }
+
+ JSStringBuilder sb(cx);
+ if (val.hasObjectPayload()) {
+ RootedObject valObj(cx, &val.getObjectPayload());
+ ESClass cls;
+ if (!JS::GetBuiltinClass(cx, valObj, &cls)) {
+ return "<<error determining class of value>>";
+ }
+ const char* s;
+ if (cls == ESClass::Array) {
+ s = "the array ";
+ } else if (cls == ESClass::ArrayBuffer) {
+ s = "the array buffer ";
+ } else if (JS_IsArrayBufferViewObject(valObj)) {
+ s = "the typed array ";
+#ifdef ENABLE_RECORD_TUPLE
+ } else if (cls == ESClass::Record) {
+ s = "the record ";
+ } else if (cls == ESClass::Tuple) {
+ s = "the tuple ";
+#endif
+ } else {
+ s = "the object ";
+ }
+ if (!sb.append(s, strlen(s))) {
+ return "<<error converting value to string>>";
+ }
+ } else if (val.isNumber()) {
+ if (!sb.append("the number ")) {
+ return "<<error converting value to string>>";
+ }
+ } else if (val.isString()) {
+ if (!sb.append("the string ")) {
+ return "<<error converting value to string>>";
+ }
+ } else if (val.isBigInt()) {
+ if (!sb.append("the BigInt ")) {
+ return "<<error converting value to string>>";
+ }
+ } else {
+ MOZ_ASSERT(val.isBoolean() || val.isSymbol());
+ bytes = StringToNewUTF8CharsZ(cx, *str);
+ return bytes.get();
+ }
+ if (!sb.append(str)) {
+ return "<<error converting value to string>>";
+ }
+ str = sb.finishString();
+ if (!str) {
+ return "<<error converting value to string>>";
+ }
+ bytes = StringToNewUTF8CharsZ(cx, *str);
+ return bytes.get();
+}
+
+bool js::GetInternalError(JSContext* cx, unsigned errorNumber,
+ MutableHandleValue error) {
+ FixedInvokeArgs<1> args(cx);
+ args[0].set(Int32Value(errorNumber));
+ return CallSelfHostedFunction(cx, cx->names().GetInternalError,
+ NullHandleValue, args, error);
+}
+
+bool js::GetTypeError(JSContext* cx, unsigned errorNumber,
+ MutableHandleValue error) {
+ FixedInvokeArgs<1> args(cx);
+ args[0].set(Int32Value(errorNumber));
+ return CallSelfHostedFunction(cx, cx->names().GetTypeError, NullHandleValue,
+ args, error);
+}
+
+bool js::GetAggregateError(JSContext* cx, unsigned errorNumber,
+ MutableHandleValue error) {
+ FixedInvokeArgs<1> args(cx);
+ args[0].set(Int32Value(errorNumber));
+ return CallSelfHostedFunction(cx, cx->names().GetAggregateError,
+ NullHandleValue, args, error);
+}
+
+JS_PUBLIC_API mozilla::Maybe<Value> JS::GetExceptionCause(JSObject* exc) {
+ if (!exc->is<ErrorObject>()) {
+ return mozilla::Nothing();
+ }
+ auto& error = exc->as<ErrorObject>();
+ return error.getCause();
+}