WorkerAgent.cpp 1.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354
  1. /*
  2. * Copyright (c) 2023, Andrew Kaster <akaster@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibWeb/Bindings/HostDefined.h>
  7. #include <LibWeb/HTML/WorkerAgent.h>
  8. #include <LibWeb/Page/Page.h>
  9. #include <LibWeb/Worker/WebWorkerClient.h>
  10. namespace Web::HTML {
  11. JS_DEFINE_ALLOCATOR(WorkerAgent);
  12. WorkerAgent::WorkerAgent(AK::URL url, WorkerOptions const& options, JS::GCPtr<MessagePort> outside_port)
  13. : m_worker_options(options)
  14. , m_url(move(url))
  15. , m_outside_port(outside_port)
  16. {
  17. }
  18. void WorkerAgent::initialize(JS::Realm& realm)
  19. {
  20. Base::initialize(realm);
  21. m_message_port = MessagePort::create(realm);
  22. m_message_port->entangle_with(*m_outside_port);
  23. TransferDataHolder data_holder;
  24. MUST(m_message_port->transfer_steps(data_holder));
  25. // NOTE: This blocking IPC call may launch another process.
  26. // If spinning the event loop for this can cause other javascript to execute, we're in trouble.
  27. auto worker_ipc_sockets = Bindings::host_defined_page(realm).client().request_worker_agent();
  28. auto worker_socket = MUST(Core::LocalSocket::adopt_fd(worker_ipc_sockets.socket.take_fd()));
  29. MUST(worker_socket->set_blocking(true));
  30. auto fd_passing_socket = MUST(Core::LocalSocket::adopt_fd(worker_ipc_sockets.fd_passing_socket.take_fd()));
  31. m_worker_ipc = make_ref_counted<WebWorkerClient>(move(worker_socket));
  32. m_worker_ipc->set_fd_passing_socket(move(fd_passing_socket));
  33. m_worker_ipc->async_start_dedicated_worker(m_url, m_worker_options.type, m_worker_options.credentials, m_worker_options.name, move(data_holder));
  34. }
  35. void WorkerAgent::visit_edges(Cell::Visitor& visitor)
  36. {
  37. Base::visit_edges(visitor);
  38. visitor.visit(m_message_port);
  39. visitor.visit(m_outside_port);
  40. }
  41. }