#pragma once #include "root.h" #include "ActiveDOMObject.h" #include #include #include #include #include #include #include "CachedScript.h" #include "wtf/URL.h" namespace uWS { template struct WebSocketContext; } #ifndef ZIG_GLOBAL_OBJECT_DEFINED #include "ZigGlobalObject.h" #endif struct us_socket_t; struct us_socket_context_t; struct us_loop_t; namespace WebCore { class WebSocket; class ScriptExecutionContext; class EventLoopTask { WTF_MAKE_FAST_ALLOCATED; public: enum CleanupTaskTag { CleanupTask }; template::value && std::is_convertible>::value>::type> EventLoopTask(T task) : m_task(WTFMove(task)) , m_isCleanupTask(false) { } EventLoopTask(Function&& task) : m_task([task = WTFMove(task)](ScriptExecutionContext&) { task(); }) , m_isCleanupTask(false) { } template>::value>::type> EventLoopTask(CleanupTaskTag, T task) : m_task(WTFMove(task)) , m_isCleanupTask(true) { } void performTask(ScriptExecutionContext& context) { m_task(context); delete this; } bool isCleanupTask() const { return m_isCleanupTask; } protected: Function m_task; bool m_isCleanupTask; }; class ScriptExecutionContext : public CanMakeWeakPtr { public: public: ScriptExecutionContext(JSC::VM* vm, JSC::JSGlobalObject* globalObject) : m_vm(vm) , m_globalObject(globalObject) { } JSC::JSGlobalObject* jsGlobalObject() { return m_globalObject; } template us_socket_context_t* webSocketContext() { if constexpr (isSSL) { return this->webSocketContextSSL(); } else { return this->webSocketContextNoSSL(); } } const WTF::URL& url() const { return m_url; } bool activeDOMObjectsAreSuspended() { return false; } bool activeDOMObjectsAreStopped() { return false; } bool isContextThread() { return true; } bool isDocument() { return false; } bool isWorkerGlobalScope() { return true; } bool isJSExecutionForbidden() { return false; } void reportException(const String& errorMessage, int lineNumber, int columnNumber, const String& sourceURL, JSC::Exception* exception, RefPtr&&, CachedScript* = nullptr, bool = false) { } // void reportUnhandledPromiseRejection(JSC::JSGlobalObject&, JSC::JSPromise&, RefPtr&&) // { // } void postTask(Function&& lambda) { auto* task = new EventLoopTask(WTFMove(lambda)); reinterpret_cast(m_globalObject)->queueTask(task); } // Executes the task on context's thread asynchronously. void postTask(EventLoopTask* task) { reinterpret_cast(m_globalObject)->queueTask(task); } // Executes the task on context's thread asynchronously. template void postCrossThreadTask(Arguments&&... arguments) { postTask([crossThreadTask = createCrossThreadTask(arguments...)](ScriptExecutionContext&) mutable { crossThreadTask.performTask(); }); } JSC::VM& vm() { return *m_vm; } private: JSC::VM* m_vm = nullptr; JSC::JSGlobalObject* m_globalObject = nullptr; WTF::URL m_url = WTF::URL(); us_socket_context_t* webSocketContextSSL(); us_socket_context_t* webSocketContextNoSSL(); us_socket_context_t* connectedWebSocketKindClientSSL(); us_socket_context_t* connectedWebSocketKindClient(); us_socket_context_t* m_ssl_client_websockets_ctx = nullptr; us_socket_context_t* m_client_websockets_ctx = nullptr; us_socket_context_t* m_connected_ssl_client_websockets_ctx = nullptr; us_socket_context_t* m_connected_client_websockets_ctx = nullptr; public: template us_socket_context_t* connnectedWebSocketContext() { if constexpr (isSSL) { if (!m_connected_ssl_client_websockets_ctx) { m_connected_ssl_client_websockets_ctx = connectedWebSocketKindClientSSL(); } return m_connected_ssl_client_websockets_ctx; } else { if (!m_connected_client_websockets_ctx) { m_connected_client_websockets_ctx = connectedWebSocketKindClient(); } return m_connected_client_websockets_ctx; } } }; }