mirror of
https://github.com/darlinghq/darling-JavaScriptCore.git
synced 2025-04-05 16:31:37 +00:00
308 lines
13 KiB
Plaintext
308 lines
13 KiB
Plaintext
/*
|
|
* Copyright (C) 2019 Apple Inc. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
|
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
|
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
|
|
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#import "config.h"
|
|
#import "JSAPIGlobalObject.h"
|
|
|
|
#if JSC_OBJC_API_ENABLED
|
|
|
|
#import "APICast.h"
|
|
#import "CallFrameInlines.h"
|
|
#import "CatchScope.h"
|
|
#import "Completion.h"
|
|
#import "Error.h"
|
|
#import "Exception.h"
|
|
#import "JSContextInternal.h"
|
|
#import "JSInternalPromise.h"
|
|
#import "JSModuleLoader.h"
|
|
#import "JSNativeStdFunction.h"
|
|
#import "JSPromise.h"
|
|
#import "JSScriptInternal.h"
|
|
#import "JSSourceCode.h"
|
|
#import "JSValueInternal.h"
|
|
#import "JSVirtualMachineInternal.h"
|
|
#import "JavaScriptCore.h"
|
|
#import "ObjectConstructor.h"
|
|
#import "SourceOrigin.h"
|
|
#import <wtf/URL.h>
|
|
|
|
#ifdef DARLING_NONUNIFIED_BUILD
|
|
#include "IdentifierInlines.h"
|
|
#include "AuxiliaryBarrierInlines.h"
|
|
#include "StrongInlines.h"
|
|
#include "JSObjectInlines.h"
|
|
#include "JSGlobalObjectInlines.h"
|
|
#endif
|
|
|
|
namespace JSC {
|
|
|
|
const ClassInfo JSAPIGlobalObject::s_info = { "GlobalObject", &Base::s_info, nullptr, nullptr, CREATE_METHOD_TABLE(JSAPIGlobalObject) };
|
|
|
|
const GlobalObjectMethodTable JSAPIGlobalObject::s_globalObjectMethodTable = {
|
|
&supportsRichSourceInfo,
|
|
&shouldInterruptScript,
|
|
&javaScriptRuntimeFlags,
|
|
nullptr, // queueTaskToEventLoop
|
|
&shouldInterruptScriptBeforeTimeout,
|
|
&moduleLoaderImportModule, // moduleLoaderImportModule
|
|
&moduleLoaderResolve, // moduleLoaderResolve
|
|
&moduleLoaderFetch, // moduleLoaderFetch
|
|
&moduleLoaderCreateImportMetaProperties, // moduleLoaderCreateImportMetaProperties
|
|
&moduleLoaderEvaluate, // moduleLoaderEvaluate
|
|
nullptr, // promiseRejectionTracker
|
|
&reportUncaughtExceptionAtEventLoop,
|
|
¤tScriptExecutionOwner,
|
|
&scriptExecutionStatus,
|
|
nullptr, // defaultLanguage
|
|
nullptr, // compileStreaming
|
|
nullptr, // instantiateStreaming
|
|
};
|
|
|
|
void JSAPIGlobalObject::reportUncaughtExceptionAtEventLoop(JSGlobalObject* globalObject, Exception* exception)
|
|
{
|
|
JSContext *context = [JSContext contextWithJSGlobalContextRef:toGlobalRef(globalObject)];
|
|
[context notifyException:toRef(globalObject->vm(), exception->value())];
|
|
}
|
|
|
|
static Expected<URL, String> computeValidImportSpecifier(const URL& base, const String& specifier)
|
|
{
|
|
URL absoluteURL(URL(), specifier);
|
|
if (absoluteURL.isValid())
|
|
return absoluteURL;
|
|
|
|
if (!specifier.startsWith('/') && !specifier.startsWith("./") && !specifier.startsWith("../"))
|
|
return makeUnexpected(makeString("Module specifier: "_s, specifier, " does not start with \"/\", \"./\", or \"../\". Referenced from: "_s, base.string()));
|
|
|
|
if (specifier.startsWith('/')) {
|
|
absoluteURL = URL(URL({ }, "file://"), specifier);
|
|
if (absoluteURL.isValid())
|
|
return absoluteURL;
|
|
}
|
|
|
|
if (base == URL())
|
|
return makeUnexpected("Could not determine the base URL for loading."_s);
|
|
|
|
if (!base.isValid())
|
|
return makeUnexpected(makeString("Referrering script's url is not valid: "_s, base.string()));
|
|
|
|
absoluteURL = URL(base, specifier);
|
|
if (absoluteURL.isValid())
|
|
return absoluteURL;
|
|
return makeUnexpected(makeString("Could not form valid URL from identifier and base. Tried:"_s, absoluteURL.string()));
|
|
}
|
|
|
|
Identifier JSAPIGlobalObject::moduleLoaderResolve(JSGlobalObject* globalObject, JSModuleLoader*, JSValue key, JSValue referrer, JSValue)
|
|
{
|
|
VM& vm = globalObject->vm();
|
|
auto scope = DECLARE_THROW_SCOPE(vm);
|
|
ASSERT_UNUSED(globalObject, globalObject == globalObject);
|
|
ASSERT(key.isString() || key.isSymbol());
|
|
String name = key.toWTFString(globalObject);
|
|
RETURN_IF_EXCEPTION(scope, { });
|
|
|
|
URL base;
|
|
if (JSString* referrerString = jsDynamicCast<JSString*>(vm, referrer)) {
|
|
String value = referrerString->value(globalObject);
|
|
RETURN_IF_EXCEPTION(scope, { });
|
|
URL referrerURL({ }, value);
|
|
RELEASE_ASSERT(referrerURL.isValid());
|
|
base = WTFMove(referrerURL);
|
|
}
|
|
|
|
auto result = computeValidImportSpecifier(base, name);
|
|
if (result)
|
|
return Identifier::fromString(vm, result.value().string());
|
|
|
|
throwVMError(globalObject, scope, createError(globalObject, result.error()));
|
|
return { };
|
|
}
|
|
|
|
JSInternalPromise* JSAPIGlobalObject::moduleLoaderImportModule(JSGlobalObject* globalObject, JSModuleLoader*, JSString* specifierValue, JSValue, const SourceOrigin& sourceOrigin)
|
|
{
|
|
VM& vm = globalObject->vm();
|
|
auto scope = DECLARE_CATCH_SCOPE(vm);
|
|
auto reject = [&] (Exception* exception) -> JSInternalPromise* {
|
|
auto* promise = JSInternalPromise::create(vm, globalObject->internalPromiseStructure());
|
|
if (UNLIKELY(isTerminatedExecutionException(vm, exception)))
|
|
return promise;
|
|
JSValue error = exception->value();
|
|
scope.clearException();
|
|
// FIXME: We could have error since any JS call can throw stack-overflow errors.
|
|
// https://bugs.webkit.org/show_bug.cgi?id=203402
|
|
promise->reject(globalObject, error);
|
|
return promise;
|
|
};
|
|
|
|
auto import = [&] (URL& url) {
|
|
auto result = importModule(globalObject, Identifier::fromString(vm, url.string()), jsUndefined(), jsUndefined());
|
|
if (UNLIKELY(scope.exception()))
|
|
return reject(scope.exception());
|
|
return result;
|
|
};
|
|
|
|
auto specifier = specifierValue->value(globalObject);
|
|
if (UNLIKELY(scope.exception()))
|
|
return reject(scope.exception());
|
|
|
|
auto result = computeValidImportSpecifier(sourceOrigin.url(), specifier);
|
|
if (result)
|
|
return import(result.value());
|
|
auto* promise = JSInternalPromise::create(vm, globalObject->internalPromiseStructure());
|
|
// FIXME: We could have error since any JS call can throw stack-overflow errors.
|
|
// https://bugs.webkit.org/show_bug.cgi?id=203402
|
|
promise->reject(globalObject, createError(globalObject, result.error()));
|
|
return promise;
|
|
}
|
|
|
|
JSInternalPromise* JSAPIGlobalObject::moduleLoaderFetch(JSGlobalObject* globalObject, JSModuleLoader*, JSValue key, JSValue, JSValue)
|
|
{
|
|
VM& vm = globalObject->vm();
|
|
auto scope = DECLARE_CATCH_SCOPE(vm);
|
|
|
|
ASSERT(globalObject == globalObject);
|
|
JSContext *context = [JSContext contextWithJSGlobalContextRef:toGlobalRef(globalObject)];
|
|
|
|
JSInternalPromise* promise = JSInternalPromise::create(vm, globalObject->internalPromiseStructure());
|
|
|
|
Identifier moduleKey = key.toPropertyKey(globalObject);
|
|
if (UNLIKELY(scope.exception())) {
|
|
Exception* exception = scope.exception();
|
|
if (UNLIKELY(isTerminatedExecutionException(vm, exception)))
|
|
return promise;
|
|
scope.clearException();
|
|
promise->reject(globalObject, exception->value());
|
|
return promise;
|
|
}
|
|
|
|
if (UNLIKELY(![context moduleLoaderDelegate])) {
|
|
promise->reject(globalObject, createError(globalObject, "No module loader provided."));
|
|
return promise;
|
|
}
|
|
|
|
auto strongPromise = Strong<JSInternalPromise>(vm, promise);
|
|
auto* resolve = JSNativeStdFunction::create(vm, globalObject, 1, "resolve", [=] (JSGlobalObject* globalObject, CallFrame* callFrame) {
|
|
// This captures the globalObject but that's ok because our structure keeps it alive anyway.
|
|
VM& vm = globalObject->vm();
|
|
JSContext *context = [JSContext contextWithJSGlobalContextRef:toGlobalRef(globalObject)];
|
|
id script = valueToObject(context, toRef(globalObject, callFrame->argument(0)));
|
|
|
|
MarkedArgumentBuffer args;
|
|
|
|
auto rejectPromise = [&] (String message) {
|
|
strongPromise.get()->reject(globalObject, createTypeError(globalObject, message));
|
|
return encodedJSUndefined();
|
|
};
|
|
|
|
if (UNLIKELY(![script isKindOfClass:[JSScript class]]))
|
|
return rejectPromise("First argument of resolution callback is not a JSScript"_s);
|
|
|
|
JSScript* jsScript = static_cast<JSScript *>(script);
|
|
|
|
JSSourceCode* source = [jsScript jsSourceCode];
|
|
if (UNLIKELY([jsScript type] != kJSScriptTypeModule))
|
|
return rejectPromise("The JSScript that was provided did not have expected type of kJSScriptTypeModule."_s);
|
|
|
|
NSURL *sourceURL = [jsScript sourceURL];
|
|
String oldModuleKey { [sourceURL absoluteString] };
|
|
if (UNLIKELY(Identifier::fromString(vm, oldModuleKey) != moduleKey))
|
|
return rejectPromise(makeString("The same JSScript was provided for two different identifiers, previously: ", oldModuleKey, " and now: ", moduleKey.string()));
|
|
|
|
strongPromise.get()->resolve(globalObject, source);
|
|
return encodedJSUndefined();
|
|
});
|
|
|
|
auto* reject = JSNativeStdFunction::create(vm, globalObject, 1, "reject", [=] (JSGlobalObject*, CallFrame* callFrame) {
|
|
strongPromise.get()->reject(globalObject, callFrame->argument(0));
|
|
return encodedJSUndefined();
|
|
});
|
|
|
|
[[context moduleLoaderDelegate] context:context fetchModuleForIdentifier:[::JSValue valueWithJSValueRef:toRef(globalObject, key) inContext:context] withResolveHandler:[::JSValue valueWithJSValueRef:toRef(globalObject, resolve) inContext:context] andRejectHandler:[::JSValue valueWithJSValueRef:toRef(globalObject, reject) inContext:context]];
|
|
if (context.exception) {
|
|
promise->reject(globalObject, toJS(globalObject, [context.exception JSValueRef]));
|
|
context.exception = nil;
|
|
}
|
|
return promise;
|
|
}
|
|
|
|
JSObject* JSAPIGlobalObject::moduleLoaderCreateImportMetaProperties(JSGlobalObject* globalObject, JSModuleLoader*, JSValue key, JSModuleRecord*, JSValue)
|
|
{
|
|
VM& vm = globalObject->vm();
|
|
auto scope = DECLARE_THROW_SCOPE(vm);
|
|
|
|
JSObject* metaProperties = constructEmptyObject(vm, globalObject->nullPrototypeObjectStructure());
|
|
RETURN_IF_EXCEPTION(scope, nullptr);
|
|
|
|
metaProperties->putDirect(vm, Identifier::fromString(vm, "filename"), key);
|
|
RETURN_IF_EXCEPTION(scope, nullptr);
|
|
|
|
return metaProperties;
|
|
}
|
|
|
|
JSValue JSAPIGlobalObject::moduleLoaderEvaluate(JSGlobalObject* globalObject, JSModuleLoader* moduleLoader, JSValue key, JSValue moduleRecordValue, JSValue scriptFetcher)
|
|
{
|
|
VM& vm = globalObject->vm();
|
|
auto scope = DECLARE_THROW_SCOPE(vm);
|
|
|
|
JSContext *context = [JSContext contextWithJSGlobalContextRef:toGlobalRef(globalObject)];
|
|
id <JSModuleLoaderDelegate> moduleLoaderDelegate = [context moduleLoaderDelegate];
|
|
NSURL *url = nil;
|
|
|
|
if ([moduleLoaderDelegate respondsToSelector:@selector(willEvaluateModule:)] || [moduleLoaderDelegate respondsToSelector:@selector(didEvaluateModule:)]) {
|
|
String moduleKey = key.toWTFString(globalObject);
|
|
RETURN_IF_EXCEPTION(scope, { });
|
|
url = [NSURL URLWithString:static_cast<NSString *>(moduleKey)];
|
|
}
|
|
|
|
if ([moduleLoaderDelegate respondsToSelector:@selector(willEvaluateModule:)])
|
|
[moduleLoaderDelegate willEvaluateModule:url];
|
|
|
|
scope.release();
|
|
JSValue result = moduleLoader->evaluateNonVirtual(globalObject, key, moduleRecordValue, scriptFetcher);
|
|
|
|
if ([moduleLoaderDelegate respondsToSelector:@selector(didEvaluateModule:)])
|
|
[moduleLoaderDelegate didEvaluateModule:url];
|
|
|
|
return result;
|
|
}
|
|
|
|
JSValue JSAPIGlobalObject::loadAndEvaluateJSScriptModule(const JSLockHolder&, JSScript *script)
|
|
{
|
|
ASSERT(script.type == kJSScriptTypeModule);
|
|
VM& vm = this->vm();
|
|
auto scope = DECLARE_THROW_SCOPE(vm);
|
|
|
|
Identifier key = Identifier::fromString(vm, String { [[script sourceURL] absoluteString] });
|
|
JSInternalPromise* promise = importModule(this, key, jsUndefined(), jsUndefined());
|
|
RETURN_IF_EXCEPTION(scope, { });
|
|
auto* result = JSPromise::create(vm, this->promiseStructure());
|
|
result->resolve(this, promise);
|
|
RETURN_IF_EXCEPTION(scope, { });
|
|
return result;
|
|
}
|
|
|
|
}
|
|
|
|
#endif // JSC_OBJC_API_ENABLED
|