1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
|
/* -*- 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/. */
#include "nsImportModule.h"
#include "mozilla/dom/ScriptSettings.h"
#include "mozJSModuleLoader.h"
#include "nsContentUtils.h"
#include "nsExceptionHandler.h"
#include "nsPrintfCString.h"
#include "xpcpublic.h"
#include "xpcprivate.h"
#include "js/PropertyAndElement.h" // JS_GetProperty
using mozilla::dom::AutoJSAPI;
namespace mozilla {
namespace loader {
static void AnnotateCrashReportWithJSException(JSContext* aCx,
const char* aURI) {
JS::RootedValue exn(aCx);
if (JS_GetPendingException(aCx, &exn)) {
JS_ClearPendingException(aCx);
JSAutoRealm ar(aCx, xpc::PrivilegedJunkScope());
JS_WrapValue(aCx, &exn);
nsAutoCString file;
uint32_t line;
uint32_t column;
nsAutoString msg;
nsContentUtils::ExtractErrorValues(aCx, exn, file, &line, &column, msg);
nsPrintfCString errorString("Failed to load module \"%s\": %s:%u:%u: %s",
aURI, file.get(), line, column,
NS_ConvertUTF16toUTF8(msg).get());
CrashReporter::AnnotateCrashReport(
CrashReporter::Annotation::JSModuleLoadError, errorString);
}
}
nsresult ImportModule(const char* aURI, const char* aExportName,
const nsIID& aIID, void** aResult, bool aInfallible) {
AutoJSAPI jsapi;
MOZ_ALWAYS_TRUE(jsapi.Init(xpc::PrivilegedJunkScope()));
JSContext* cx = jsapi.cx();
JS::RootedObject global(cx);
JS::RootedObject exports(cx);
nsresult rv = mozJSModuleLoader::Get()->Import(cx, nsDependentCString(aURI),
&global, &exports);
if (NS_WARN_IF(NS_FAILED(rv))) {
if (aInfallible) {
AnnotateCrashReportWithJSException(cx, aURI);
MOZ_CRASH_UNSAFE_PRINTF("Failed to load critical module \"%s\"", aURI);
}
return rv;
}
if (aExportName) {
JS::RootedValue namedExport(cx);
if (!JS_GetProperty(cx, exports, aExportName, &namedExport)) {
return NS_ERROR_FAILURE;
}
if (!namedExport.isObject()) {
return NS_ERROR_XPC_BAD_CONVERT_JS;
}
exports.set(&namedExport.toObject());
}
return nsXPConnect::XPConnect()->WrapJS(cx, exports, aIID, aResult);
}
nsresult ImportESModule(const char* aURI, const char* aExportName,
const nsIID& aIID, void** aResult, bool aInfallible) {
AutoJSAPI jsapi;
MOZ_ALWAYS_TRUE(jsapi.Init(xpc::PrivilegedJunkScope()));
JSContext* cx = jsapi.cx();
JS::RootedObject moduleNamespace(cx);
nsresult rv = mozJSModuleLoader::Get()->ImportESModule(
cx, nsDependentCString(aURI), &moduleNamespace);
if (NS_WARN_IF(NS_FAILED(rv))) {
if (aInfallible) {
AnnotateCrashReportWithJSException(cx, aURI);
MOZ_CRASH_UNSAFE_PRINTF("Failed to load critical module \"%s\"", aURI);
}
return rv;
}
if (aExportName) {
JS::RootedValue namedExport(cx);
if (!JS_GetProperty(cx, moduleNamespace, aExportName, &namedExport)) {
return NS_ERROR_FAILURE;
}
if (!namedExport.isObject()) {
return NS_ERROR_XPC_BAD_CONVERT_JS;
}
moduleNamespace.set(&namedExport.toObject());
}
return nsXPConnect::XPConnect()->WrapJS(cx, moduleNamespace, aIID, aResult);
}
} // namespace loader
} // namespace mozilla
|