2001-09-25 22:43:09 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2008-01-30 02:11:48 +00:00
|
|
|
/* vim: set ts=2 sw=2 et tw=78: */
|
2012-05-21 11:12:37 +00:00
|
|
|
/* 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/. */
|
1998-12-03 00:33:00 +00:00
|
|
|
|
|
|
|
/**
|
2003-11-18 01:58:43 +00:00
|
|
|
* This is not a generated file. It contains common utility functions
|
1998-12-03 00:33:00 +00:00
|
|
|
* invoked from the JavaScript code generated from IDL interfaces.
|
|
|
|
* The goal of the utility functions is to cut down on the size of
|
|
|
|
* the generated code itself.
|
|
|
|
*/
|
|
|
|
|
1998-12-22 22:03:20 +00:00
|
|
|
#include "nsJSUtils.h"
|
1998-12-03 00:33:00 +00:00
|
|
|
#include "jsapi.h"
|
2013-08-27 02:05:20 +00:00
|
|
|
#include "js/OldDebugAPI.h"
|
2013-09-04 21:06:55 +00:00
|
|
|
#include "jsfriendapi.h"
|
1998-12-03 00:33:00 +00:00
|
|
|
#include "nsIScriptContext.h"
|
|
|
|
#include "nsIScriptGlobalObject.h"
|
1999-04-24 02:37:41 +00:00
|
|
|
#include "nsIXPConnect.h"
|
1999-09-08 23:14:30 +00:00
|
|
|
#include "nsCOMPtr.h"
|
2008-02-13 03:52:43 +00:00
|
|
|
#include "nsIScriptSecurityManager.h"
|
2011-07-20 19:18:54 +00:00
|
|
|
#include "nsPIDOMWindow.h"
|
2013-08-08 23:51:34 +00:00
|
|
|
#include "GeckoProfiler.h"
|
2006-06-13 03:07:47 +00:00
|
|
|
#include "nsDOMJSUtils.h" // for GetScriptContextFromJSContext
|
2013-01-17 02:50:25 +00:00
|
|
|
#include "nsJSPrincipals.h"
|
2013-08-15 18:17:48 +00:00
|
|
|
#include "xpcpublic.h"
|
|
|
|
#include "nsContentUtils.h"
|
2013-11-20 16:48:00 +00:00
|
|
|
#include "nsGlobalWindow.h"
|
2012-03-31 04:42:20 +00:00
|
|
|
|
2013-08-08 22:53:04 +00:00
|
|
|
bool
|
Landing the XPCDOM_20010329_BRANCH branch, changes mostly done by jband@netscape.com and jst@netscape.com, also some changes done by shaver@mozilla.org, peterv@netscape.com and markh@activestate.com. r= and sr= by vidur@netscape.com, jband@netscape.com, jst@netscpae.com, danm@netscape.com, hyatt@netscape.com, shaver@mozilla.org, dbradley@netscape.com, rpotts@netscape.com.
2001-05-08 17:42:36 +00:00
|
|
|
nsJSUtils::GetCallingLocation(JSContext* aContext, const char* *aFilename,
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t* aLineno)
|
1999-09-08 23:14:30 +00:00
|
|
|
{
|
2014-02-25 15:43:14 +00:00
|
|
|
JS::AutoFilename filename;
|
2012-04-16 19:30:00 +00:00
|
|
|
unsigned lineno = 0;
|
Landing the XPCDOM_20010329_BRANCH branch, changes mostly done by jband@netscape.com and jst@netscape.com, also some changes done by shaver@mozilla.org, peterv@netscape.com and markh@activestate.com. r= and sr= by vidur@netscape.com, jband@netscape.com, jst@netscpae.com, danm@netscape.com, hyatt@netscape.com, shaver@mozilla.org, dbradley@netscape.com, rpotts@netscape.com.
2001-05-08 17:42:36 +00:00
|
|
|
|
2014-02-25 15:43:14 +00:00
|
|
|
if (!JS::DescribeScriptedCaller(aContext, &filename, &lineno)) {
|
2013-08-07 06:59:54 +00:00
|
|
|
return false;
|
1999-09-08 23:14:30 +00:00
|
|
|
}
|
(13163, r=alecf, scc, waterson, others; names available on request)
- Fix most of bug 13163 (see TODO for rest). This entails adding a version-string argument to nsIScriptContext::EvaluateString and passing it around lots of places in content sinks.
- Fix leaks and confusion about mSecurityManager and mNameSpaceManager in nsJSEnvironment.cpp. These still need to move from nsJSContext to nsGlobalWindow or thereabouts, jband and vidur are looking at that.
- Added comments and expanded tabs in nsJSEnvironment.cpp, esp. to EvaluateString. Also changed various nsresult vars to be named rv. Also restored brace/style conformity to nsJSProtocolHandler.cpp.
- Factored CompileFunction from AddScriptEventListener to pave the way for brutal sharing of compiled JS event handlers via JS_CloneFunctionObject.
- Lots of nsCOMPtr uses added. I'm using one for mNameSpaceManager. Hold mSecurityManager as a service explicitly, on the other hand (awaiting scc's fix to allow comptrs for services), and release in nsJSContext's dtor (fixing a leak). These two managers should be moved to the window object -- TODO item below.
- Hold JSRuntimeService along with JSRuntime for live of nsJSEnvironment, fix for shaver.
- Fix window.setTimeout etc. so the filename and line number of the timeout expr is propagated. This meant factoring nsJSUtils.cpp code.
- Fix all content sinks to use the same, and up-to-date JavaScript version parsing (whether for script type or for old language attribute); also fix SplitMimeType clones to strip whitespace.
- With waterson, fix bug in brutal-sharing version of XUL content sink: script src= should not evaluate the inline content of its tag.
1999-10-31 00:43:30 +00:00
|
|
|
|
2014-02-25 15:43:14 +00:00
|
|
|
*aFilename = filename.get();
|
2012-04-16 19:30:00 +00:00
|
|
|
*aLineno = lineno;
|
|
|
|
|
2013-08-07 06:59:54 +00:00
|
|
|
return true;
|
(13163, r=alecf, scc, waterson, others; names available on request)
- Fix most of bug 13163 (see TODO for rest). This entails adding a version-string argument to nsIScriptContext::EvaluateString and passing it around lots of places in content sinks.
- Fix leaks and confusion about mSecurityManager and mNameSpaceManager in nsJSEnvironment.cpp. These still need to move from nsJSContext to nsGlobalWindow or thereabouts, jband and vidur are looking at that.
- Added comments and expanded tabs in nsJSEnvironment.cpp, esp. to EvaluateString. Also changed various nsresult vars to be named rv. Also restored brace/style conformity to nsJSProtocolHandler.cpp.
- Factored CompileFunction from AddScriptEventListener to pave the way for brutal sharing of compiled JS event handlers via JS_CloneFunctionObject.
- Lots of nsCOMPtr uses added. I'm using one for mNameSpaceManager. Hold mSecurityManager as a service explicitly, on the other hand (awaiting scc's fix to allow comptrs for services), and release in nsJSContext's dtor (fixing a leak). These two managers should be moved to the window object -- TODO item below.
- Hold JSRuntimeService along with JSRuntime for live of nsJSEnvironment, fix for shaver.
- Fix window.setTimeout etc. so the filename and line number of the timeout expr is propagated. This meant factoring nsJSUtils.cpp code.
- Fix all content sinks to use the same, and up-to-date JavaScript version parsing (whether for script type or for old language attribute); also fix SplitMimeType clones to strip whitespace.
- With waterson, fix bug in brutal-sharing version of XUL content sink: script src= should not evaluate the inline content of its tag.
1999-10-31 00:43:30 +00:00
|
|
|
}
|
|
|
|
|
2004-02-09 22:48:53 +00:00
|
|
|
nsIScriptGlobalObject *
|
2012-11-09 15:43:57 +00:00
|
|
|
nsJSUtils::GetStaticScriptGlobal(JSObject* aObj)
|
1999-12-18 20:29:29 +00:00
|
|
|
{
|
2013-11-20 16:48:00 +00:00
|
|
|
if (!aObj)
|
2012-07-30 14:20:58 +00:00
|
|
|
return nullptr;
|
2013-11-20 16:48:00 +00:00
|
|
|
return xpc::WindowGlobalOrNull(aObj);
|
1999-12-18 20:29:29 +00:00
|
|
|
}
|
|
|
|
|
2004-02-09 22:48:53 +00:00
|
|
|
nsIScriptContext *
|
2012-11-09 15:43:57 +00:00
|
|
|
nsJSUtils::GetStaticScriptContext(JSObject* aObj)
|
1999-12-18 20:29:29 +00:00
|
|
|
{
|
2012-11-09 15:43:57 +00:00
|
|
|
nsIScriptGlobalObject *nativeGlobal = GetStaticScriptGlobal(aObj);
|
2003-11-18 01:58:43 +00:00
|
|
|
if (!nativeGlobal)
|
2012-07-30 14:20:58 +00:00
|
|
|
return nullptr;
|
2004-02-09 22:48:53 +00:00
|
|
|
|
2012-03-23 17:13:29 +00:00
|
|
|
return nativeGlobal->GetScriptContext();
|
2003-11-18 01:58:43 +00:00
|
|
|
}
|
1999-12-18 20:29:29 +00:00
|
|
|
|
2004-02-09 22:48:53 +00:00
|
|
|
nsIScriptGlobalObject *
|
|
|
|
nsJSUtils::GetDynamicScriptGlobal(JSContext* aContext)
|
1999-12-18 20:29:29 +00:00
|
|
|
{
|
2004-02-09 22:48:53 +00:00
|
|
|
nsIScriptContext *scriptCX = GetDynamicScriptContext(aContext);
|
2001-06-20 03:27:48 +00:00
|
|
|
if (!scriptCX)
|
2012-07-30 14:20:58 +00:00
|
|
|
return nullptr;
|
2004-02-09 22:48:53 +00:00
|
|
|
return scriptCX->GetGlobalObject();
|
2003-11-18 01:58:43 +00:00
|
|
|
}
|
1999-12-18 20:29:29 +00:00
|
|
|
|
2004-02-09 22:48:53 +00:00
|
|
|
nsIScriptContext *
|
|
|
|
nsJSUtils::GetDynamicScriptContext(JSContext *aContext)
|
1999-12-18 20:29:29 +00:00
|
|
|
{
|
2004-02-09 22:48:53 +00:00
|
|
|
return GetScriptContextFromJSContext(aContext);
|
1999-12-18 20:29:29 +00:00
|
|
|
}
|
2010-12-20 16:21:58 +00:00
|
|
|
|
2012-08-22 15:56:38 +00:00
|
|
|
uint64_t
|
2011-08-24 20:44:35 +00:00
|
|
|
nsJSUtils::GetCurrentlyRunningCodeInnerWindowID(JSContext *aContext)
|
2010-12-20 16:21:58 +00:00
|
|
|
{
|
|
|
|
if (!aContext)
|
|
|
|
return 0;
|
|
|
|
|
2012-08-22 15:56:38 +00:00
|
|
|
uint64_t innerWindowID = 0;
|
2011-02-20 04:46:44 +00:00
|
|
|
|
2013-07-29 23:45:27 +00:00
|
|
|
JSObject *jsGlobal = JS::CurrentGlobalOrNull(aContext);
|
2011-02-20 04:46:44 +00:00
|
|
|
if (jsGlobal) {
|
2012-11-09 15:43:57 +00:00
|
|
|
nsIScriptGlobalObject *scriptGlobal = GetStaticScriptGlobal(jsGlobal);
|
2011-02-20 04:46:44 +00:00
|
|
|
if (scriptGlobal) {
|
|
|
|
nsCOMPtr<nsPIDOMWindow> win = do_QueryInterface(scriptGlobal);
|
|
|
|
if (win)
|
2011-08-24 20:44:35 +00:00
|
|
|
innerWindowID = win->WindowID();
|
2011-02-20 04:46:44 +00:00
|
|
|
}
|
|
|
|
}
|
2010-12-20 16:21:58 +00:00
|
|
|
|
2011-08-24 20:44:35 +00:00
|
|
|
return innerWindowID;
|
2010-12-20 16:21:58 +00:00
|
|
|
}
|
|
|
|
|
2012-11-09 15:43:57 +00:00
|
|
|
void
|
|
|
|
nsJSUtils::ReportPendingException(JSContext *aContext)
|
|
|
|
{
|
|
|
|
if (JS_IsExceptionPending(aContext)) {
|
|
|
|
bool saved = JS_SaveFrameChain(aContext);
|
2013-07-17 18:53:52 +00:00
|
|
|
{
|
2014-03-05 13:32:27 +00:00
|
|
|
// JS_SaveFrameChain set the compartment of aContext to null, so we need
|
|
|
|
// to enter a compartment. The question is, which one? We don't want to
|
|
|
|
// enter the original compartment of aContext (or the compartment of the
|
|
|
|
// current exception on aContext, for that matter) because when we
|
|
|
|
// JS_ReportPendingException the JS engine can try to duck-type the
|
|
|
|
// exception and produce a JSErrorReport. It will then pass that
|
|
|
|
// JSErrorReport to the error reporter on aContext, which might expose
|
|
|
|
// information from it to script via onerror handlers. So it's very
|
|
|
|
// important that the duck typing happen in the same compartment as the
|
|
|
|
// onerror handler. In practice, that's the compartment of the window (or
|
|
|
|
// otherwise default global) of aContext, so use that here.
|
2013-09-04 21:06:55 +00:00
|
|
|
nsIScriptContext* scx = GetScriptContextFromJSContext(aContext);
|
|
|
|
JS::Rooted<JSObject*> scope(aContext);
|
2013-09-04 21:06:57 +00:00
|
|
|
scope = scx ? scx->GetWindowProxy()
|
2013-09-04 21:06:55 +00:00
|
|
|
: js::DefaultObjectForContextOrNull(aContext);
|
2014-03-26 13:59:03 +00:00
|
|
|
if (!scope) {
|
|
|
|
// The SafeJSContext has no default object associated with it.
|
|
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
MOZ_ASSERT(aContext == nsContentUtils::GetSafeJSContext());
|
|
|
|
scope = xpc::GetSafeJSContextGlobal();
|
|
|
|
}
|
2013-09-04 21:06:55 +00:00
|
|
|
JSAutoCompartment ac(aContext, scope);
|
2013-07-17 18:53:52 +00:00
|
|
|
JS_ReportPendingException(aContext);
|
|
|
|
}
|
2012-11-09 15:43:57 +00:00
|
|
|
if (saved) {
|
|
|
|
JS_RestoreFrameChain(aContext);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-01-17 02:50:25 +00:00
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsJSUtils::CompileFunction(JSContext* aCx,
|
2013-11-11 08:04:41 +00:00
|
|
|
JS::Handle<JSObject*> aTarget,
|
2013-01-17 02:50:25 +00:00
|
|
|
JS::CompileOptions& aOptions,
|
|
|
|
const nsACString& aName,
|
|
|
|
uint32_t aArgCount,
|
|
|
|
const char** aArgArray,
|
|
|
|
const nsAString& aBody,
|
|
|
|
JSObject** aFunctionObject)
|
|
|
|
{
|
|
|
|
MOZ_ASSERT(js::GetEnterCompartmentDepth(aCx) > 0);
|
2013-01-17 02:50:27 +00:00
|
|
|
MOZ_ASSERT_IF(aTarget, js::IsObjectInContextCompartment(aTarget, aCx));
|
2013-01-17 02:50:25 +00:00
|
|
|
MOZ_ASSERT_IF(aOptions.versionSet, aOptions.version != JSVERSION_UNKNOWN);
|
2013-01-23 06:12:50 +00:00
|
|
|
mozilla::DebugOnly<nsIScriptContext*> ctx = GetScriptContextFromJSContext(aCx);
|
|
|
|
MOZ_ASSERT_IF(ctx, ctx->IsContextInitialized());
|
2013-01-17 02:50:25 +00:00
|
|
|
|
|
|
|
// Do the junk Gecko is supposed to do before calling into JSAPI.
|
2013-09-09 03:28:48 +00:00
|
|
|
if (aTarget) {
|
|
|
|
JS::ExposeObjectToActiveJS(aTarget);
|
|
|
|
}
|
2013-01-17 02:50:25 +00:00
|
|
|
|
|
|
|
// Compile.
|
|
|
|
JSFunction* fun = JS::CompileFunction(aCx, aTarget, aOptions,
|
|
|
|
PromiseFlatCString(aName).get(),
|
|
|
|
aArgCount, aArgArray,
|
|
|
|
PromiseFlatString(aBody).get(),
|
|
|
|
aBody.Length());
|
2013-01-23 06:12:50 +00:00
|
|
|
if (!fun) {
|
|
|
|
ReportPendingException(aCx);
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
2013-01-17 02:50:25 +00:00
|
|
|
|
|
|
|
*aFunctionObject = JS_GetFunctionObject(fun);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2013-08-08 23:51:34 +00:00
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsJSUtils::EvaluateString(JSContext* aCx,
|
|
|
|
const nsAString& aScript,
|
|
|
|
JS::Handle<JSObject*> aScopeObject,
|
2013-08-08 23:51:35 +00:00
|
|
|
JS::CompileOptions& aCompileOptions,
|
2014-04-01 10:34:39 +00:00
|
|
|
const EvaluateOptions& aEvaluateOptions,
|
|
|
|
JS::MutableHandle<JS::Value> aRetValue,
|
2013-09-11 23:42:09 +00:00
|
|
|
void **aOffThreadToken)
|
2014-04-25 14:11:56 +00:00
|
|
|
{
|
|
|
|
const nsPromiseFlatString& flatScript = PromiseFlatString(aScript);
|
|
|
|
JS::SourceBufferHolder srcBuf(flatScript.get(), aScript.Length(),
|
|
|
|
JS::SourceBufferHolder::NoOwnership);
|
|
|
|
return EvaluateString(aCx, srcBuf, aScopeObject, aCompileOptions,
|
|
|
|
aEvaluateOptions, aRetValue, aOffThreadToken);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsJSUtils::EvaluateString(JSContext* aCx,
|
|
|
|
JS::SourceBufferHolder& aSrcBuf,
|
|
|
|
JS::Handle<JSObject*> aScopeObject,
|
|
|
|
JS::CompileOptions& aCompileOptions,
|
|
|
|
const EvaluateOptions& aEvaluateOptions,
|
|
|
|
JS::MutableHandle<JS::Value> aRetValue,
|
|
|
|
void **aOffThreadToken)
|
2013-08-08 23:51:34 +00:00
|
|
|
{
|
|
|
|
PROFILER_LABEL("JS", "EvaluateString");
|
2013-08-08 23:51:35 +00:00
|
|
|
MOZ_ASSERT_IF(aCompileOptions.versionSet,
|
|
|
|
aCompileOptions.version != JSVERSION_UNKNOWN);
|
2014-04-01 10:34:39 +00:00
|
|
|
MOZ_ASSERT_IF(aEvaluateOptions.coerceToString, aEvaluateOptions.needResult);
|
|
|
|
MOZ_ASSERT_IF(!aEvaluateOptions.reportUncaught, aEvaluateOptions.needResult);
|
2013-08-08 23:51:34 +00:00
|
|
|
MOZ_ASSERT(aCx == nsContentUtils::GetCurrentJSContext());
|
2014-04-25 14:11:56 +00:00
|
|
|
MOZ_ASSERT(aSrcBuf.get());
|
2013-08-08 23:51:34 +00:00
|
|
|
|
|
|
|
// Unfortunately, the JS engine actually compiles scripts with a return value
|
|
|
|
// in a different, less efficient way. Furthermore, it can't JIT them in many
|
|
|
|
// cases. So we need to be explicitly told whether the caller cares about the
|
2014-04-01 10:34:39 +00:00
|
|
|
// return value. Callers can do this by calling the other overload of
|
|
|
|
// EvaluateString() which calls this function with aEvaluateOptions.needResult
|
|
|
|
// set to false.
|
|
|
|
aRetValue.setUndefined();
|
2013-08-08 23:51:34 +00:00
|
|
|
|
2013-09-09 03:28:48 +00:00
|
|
|
JS::ExposeObjectToActiveJS(aScopeObject);
|
2013-08-08 23:51:34 +00:00
|
|
|
nsAutoMicroTask mt;
|
2013-11-13 00:43:31 +00:00
|
|
|
nsresult rv = NS_OK;
|
2013-08-08 23:51:34 +00:00
|
|
|
|
|
|
|
bool ok = false;
|
2013-11-13 00:43:31 +00:00
|
|
|
nsIScriptSecurityManager* ssm = nsContentUtils::GetSecurityManager();
|
|
|
|
NS_ENSURE_TRUE(ssm->ScriptAllowed(js::GetGlobalForObjectCrossCompartment(aScopeObject)), NS_OK);
|
2013-08-08 23:51:34 +00:00
|
|
|
|
2013-08-19 09:46:24 +00:00
|
|
|
mozilla::Maybe<AutoDontReportUncaught> dontReport;
|
|
|
|
if (!aEvaluateOptions.reportUncaught) {
|
|
|
|
// We need to prevent AutoLastFrameCheck from reporting and clearing
|
|
|
|
// any pending exceptions.
|
|
|
|
dontReport.construct(aCx);
|
|
|
|
}
|
|
|
|
|
2013-08-08 23:51:34 +00:00
|
|
|
// Scope the JSAutoCompartment so that we can later wrap the return value
|
|
|
|
// into the caller's cx.
|
|
|
|
{
|
|
|
|
JSAutoCompartment ac(aCx, aScopeObject);
|
|
|
|
|
2013-11-11 08:04:41 +00:00
|
|
|
JS::Rooted<JSObject*> rootedScope(aCx, aScopeObject);
|
2013-09-11 23:42:09 +00:00
|
|
|
if (aOffThreadToken) {
|
2014-03-17 16:17:58 +00:00
|
|
|
JS::Rooted<JSScript*>
|
|
|
|
script(aCx, JS::FinishOffThreadScript(aCx, JS_GetRuntime(aCx), *aOffThreadToken));
|
2013-09-11 23:42:09 +00:00
|
|
|
*aOffThreadToken = nullptr; // Mark the token as having been finished.
|
|
|
|
if (script) {
|
2014-04-01 10:34:39 +00:00
|
|
|
if (aEvaluateOptions.needResult) {
|
|
|
|
ok = JS_ExecuteScript(aCx, rootedScope, script, aRetValue);
|
|
|
|
} else {
|
|
|
|
ok = JS_ExecuteScript(aCx, rootedScope, script);
|
|
|
|
}
|
2013-09-11 23:42:09 +00:00
|
|
|
} else {
|
|
|
|
ok = false;
|
|
|
|
}
|
|
|
|
} else {
|
2014-04-01 10:34:39 +00:00
|
|
|
if (aEvaluateOptions.needResult) {
|
|
|
|
ok = JS::Evaluate(aCx, rootedScope, aCompileOptions,
|
2014-04-25 14:11:56 +00:00
|
|
|
aSrcBuf, aRetValue);
|
2014-04-01 10:34:39 +00:00
|
|
|
} else {
|
|
|
|
ok = JS::Evaluate(aCx, rootedScope, aCompileOptions,
|
2014-04-25 14:11:56 +00:00
|
|
|
aSrcBuf);
|
2014-04-01 10:34:39 +00:00
|
|
|
}
|
2013-09-11 23:42:09 +00:00
|
|
|
}
|
|
|
|
|
2014-04-01 10:34:39 +00:00
|
|
|
if (ok && aEvaluateOptions.coerceToString && !aRetValue.isUndefined()) {
|
|
|
|
JS::Rooted<JS::Value> value(aCx, aRetValue);
|
2013-11-16 12:31:36 +00:00
|
|
|
JSString* str = JS::ToString(aCx, value);
|
2013-08-08 23:51:34 +00:00
|
|
|
ok = !!str;
|
2014-04-01 10:34:39 +00:00
|
|
|
aRetValue.set(ok ? JS::StringValue(str) : JS::UndefinedValue());
|
2013-08-08 23:51:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ok) {
|
2013-08-08 23:51:35 +00:00
|
|
|
if (aEvaluateOptions.reportUncaught) {
|
|
|
|
ReportPendingException(aCx);
|
2014-04-01 10:34:39 +00:00
|
|
|
if (aEvaluateOptions.needResult) {
|
|
|
|
aRetValue.setUndefined();
|
2013-08-08 23:51:35 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
rv = JS_IsExceptionPending(aCx) ? NS_ERROR_FAILURE
|
|
|
|
: NS_ERROR_OUT_OF_MEMORY;
|
2013-11-11 08:04:41 +00:00
|
|
|
JS::Rooted<JS::Value> exn(aCx);
|
2013-09-19 07:54:01 +00:00
|
|
|
JS_GetPendingException(aCx, &exn);
|
2014-04-01 10:34:39 +00:00
|
|
|
if (aEvaluateOptions.needResult) {
|
|
|
|
aRetValue.set(exn);
|
2013-09-19 07:54:01 +00:00
|
|
|
}
|
2013-08-08 23:51:35 +00:00
|
|
|
JS_ClearPendingException(aCx);
|
2013-08-08 23:51:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wrap the return value into whatever compartment aCx was in.
|
2014-04-01 10:34:39 +00:00
|
|
|
if (aEvaluateOptions.needResult) {
|
|
|
|
JS::Rooted<JS::Value> v(aCx, aRetValue);
|
2013-10-26 16:19:05 +00:00
|
|
|
if (!JS_WrapValue(aCx, &v)) {
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
}
|
2014-04-01 10:34:39 +00:00
|
|
|
aRetValue.set(v);
|
2013-10-26 16:19:05 +00:00
|
|
|
}
|
2013-08-08 23:51:35 +00:00
|
|
|
return rv;
|
2013-08-08 23:51:34 +00:00
|
|
|
}
|
2013-09-04 21:06:55 +00:00
|
|
|
|
2014-04-01 10:34:39 +00:00
|
|
|
nsresult
|
|
|
|
nsJSUtils::EvaluateString(JSContext* aCx,
|
|
|
|
const nsAString& aScript,
|
|
|
|
JS::Handle<JSObject*> aScopeObject,
|
|
|
|
JS::CompileOptions& aCompileOptions,
|
|
|
|
void **aOffThreadToken)
|
|
|
|
{
|
|
|
|
EvaluateOptions options;
|
|
|
|
options.setNeedResult(false);
|
|
|
|
JS::RootedValue unused(aCx);
|
|
|
|
return EvaluateString(aCx, aScript, aScopeObject, aCompileOptions,
|
|
|
|
options, &unused, aOffThreadToken);
|
|
|
|
}
|
|
|
|
|
2014-04-25 14:11:56 +00:00
|
|
|
nsresult
|
|
|
|
nsJSUtils::EvaluateString(JSContext* aCx,
|
|
|
|
JS::SourceBufferHolder& aSrcBuf,
|
|
|
|
JS::Handle<JSObject*> aScopeObject,
|
|
|
|
JS::CompileOptions& aCompileOptions,
|
|
|
|
void **aOffThreadToken)
|
|
|
|
{
|
|
|
|
EvaluateOptions options;
|
|
|
|
options.setNeedResult(false);
|
|
|
|
JS::RootedValue unused(aCx);
|
|
|
|
return EvaluateString(aCx, aSrcBuf, aScopeObject, aCompileOptions,
|
|
|
|
options, &unused, aOffThreadToken);
|
|
|
|
}
|
|
|
|
|
2013-09-04 21:06:55 +00:00
|
|
|
//
|
|
|
|
// nsDOMJSUtils.h
|
|
|
|
//
|
|
|
|
|
|
|
|
JSObject* GetDefaultScopeFromJSContext(JSContext *cx)
|
|
|
|
{
|
|
|
|
// DOM JSContexts don't store their default compartment object on
|
|
|
|
// the cx, so in those cases we need to fetch it via the scx
|
|
|
|
// instead.
|
|
|
|
nsIScriptContext *scx = GetScriptContextFromJSContext(cx);
|
|
|
|
if (scx) {
|
2013-09-04 21:06:57 +00:00
|
|
|
return scx->GetWindowProxy();
|
2013-09-04 21:06:55 +00:00
|
|
|
}
|
|
|
|
return js::DefaultObjectForContextOrNull(cx);
|
|
|
|
}
|