ChromeProcess.cpp 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. /*
  2. * Copyright (c) 2024, Andrew Kaster <akaster@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/ByteString.h>
  7. #include <LibCore/Process.h>
  8. #include <LibCore/StandardPaths.h>
  9. #include <LibCore/System.h>
  10. #include <LibIPC/ConnectionToServer.h>
  11. #include <LibWebView/Application.h>
  12. #include <LibWebView/ChromeProcess.h>
  13. #include <LibWebView/URL.h>
  14. namespace WebView {
  15. static HashMap<int, RefPtr<UIProcessConnectionFromClient>> s_connections;
  16. class UIProcessClient final
  17. : public IPC::ConnectionToServer<UIProcessClientEndpoint, UIProcessServerEndpoint> {
  18. C_OBJECT(UIProcessClient);
  19. private:
  20. UIProcessClient(NonnullOwnPtr<Core::LocalSocket>);
  21. };
  22. ErrorOr<ChromeProcess> ChromeProcess::create()
  23. {
  24. // Increase the open file limit, as the default limits on Linux cause us to run out of file descriptors with around 15 tabs open.
  25. if (auto result = Core::System::set_resource_limits(RLIMIT_NOFILE, 8192); result.is_error())
  26. warnln("Unable to increase open file limit: {}", result.error());
  27. return ChromeProcess {};
  28. }
  29. ErrorOr<ChromeProcess::ProcessDisposition> ChromeProcess::connect(Vector<ByteString> const& raw_urls, NewWindow new_window)
  30. {
  31. static constexpr auto process_name = "Ladybird"sv;
  32. auto [socket_path, pid_path] = TRY(Core::IPCProcess::paths_for_process(process_name));
  33. if (auto pid = TRY(Core::IPCProcess::get_process_pid(process_name, pid_path)); pid.has_value()) {
  34. TRY(connect_as_client(socket_path, raw_urls, new_window));
  35. return ProcessDisposition::ExitProcess;
  36. }
  37. TRY(connect_as_server(socket_path));
  38. m_pid_path = pid_path;
  39. m_pid_file = TRY(Core::File::open(pid_path, Core::File::OpenMode::Write));
  40. TRY(m_pid_file->write_until_depleted(ByteString::number(::getpid())));
  41. return ProcessDisposition::ContinueMainProcess;
  42. }
  43. ErrorOr<void> ChromeProcess::connect_as_client(ByteString const& socket_path, Vector<ByteString> const& raw_urls, NewWindow new_window)
  44. {
  45. auto socket = TRY(Core::LocalSocket::connect(socket_path));
  46. auto client = UIProcessClient::construct(move(socket));
  47. if (new_window == NewWindow::Yes) {
  48. if (!client->send_sync_but_allow_failure<Messages::UIProcessServer::CreateNewWindow>(raw_urls))
  49. dbgln("Failed to send CreateNewWindow message to UIProcess");
  50. } else {
  51. if (!client->send_sync_but_allow_failure<Messages::UIProcessServer::CreateNewTab>(raw_urls))
  52. dbgln("Failed to send CreateNewTab message to UIProcess");
  53. }
  54. return {};
  55. }
  56. ErrorOr<void> ChromeProcess::connect_as_server(ByteString const& socket_path)
  57. {
  58. auto socket_fd = TRY(Core::IPCProcess::create_ipc_socket(socket_path));
  59. m_socket_path = socket_path;
  60. auto local_server = TRY(Core::LocalServer::try_create());
  61. TRY(local_server->take_over_fd(socket_fd));
  62. m_server_connection = TRY(IPC::MultiServer<UIProcessConnectionFromClient>::try_create(move(local_server)));
  63. m_server_connection->on_new_client = [this](auto& client) {
  64. client.on_new_tab = [this](auto raw_urls) {
  65. if (this->on_new_tab)
  66. this->on_new_tab(raw_urls);
  67. };
  68. client.on_new_window = [this](auto raw_urls) {
  69. if (this->on_new_window)
  70. this->on_new_window(raw_urls);
  71. };
  72. };
  73. return {};
  74. }
  75. ChromeProcess::~ChromeProcess()
  76. {
  77. if (m_pid_file) {
  78. MUST(m_pid_file->truncate(0));
  79. MUST(Core::System::unlink(m_pid_path));
  80. }
  81. if (!m_socket_path.is_empty())
  82. MUST(Core::System::unlink(m_socket_path));
  83. }
  84. UIProcessClient::UIProcessClient(NonnullOwnPtr<Core::LocalSocket> socket)
  85. : IPC::ConnectionToServer<UIProcessClientEndpoint, UIProcessServerEndpoint>(*this, move(socket))
  86. {
  87. }
  88. UIProcessConnectionFromClient::UIProcessConnectionFromClient(NonnullOwnPtr<Core::LocalSocket> socket, int client_id)
  89. : IPC::ConnectionFromClient<UIProcessClientEndpoint, UIProcessServerEndpoint>(*this, move(socket), client_id)
  90. {
  91. s_connections.set(client_id, *this);
  92. }
  93. void UIProcessConnectionFromClient::die()
  94. {
  95. s_connections.remove(client_id());
  96. }
  97. void UIProcessConnectionFromClient::create_new_tab(Vector<ByteString> const& urls)
  98. {
  99. if (on_new_tab)
  100. on_new_tab(sanitize_urls(urls, Application::chrome_options().new_tab_page_url));
  101. }
  102. void UIProcessConnectionFromClient::create_new_window(Vector<ByteString> const& urls)
  103. {
  104. if (on_new_window)
  105. on_new_window(sanitize_urls(urls, Application::chrome_options().new_tab_page_url));
  106. }
  107. }