2023-02-02 03:00:30 -07:00
/*
* Copyright ( c ) 2023 , Andrew Kaster < akaster @ serenityos . org >
*
* SPDX - License - Identifier : BSD - 2 - Clause
*/
# include "HelperProcess.h"
2023-08-01 11:56:10 -06:00
ErrorOr < NonnullRefPtr < WebView : : WebContentClient > > launch_web_content_process ( WebView : : ViewImplementation & view ,
ReadonlySpan < String > candidate_web_content_paths ,
WebView : : EnableCallgrindProfiling enable_callgrind_profiling ,
WebView : : IsLayoutTestMode is_layout_test_mode ,
2023-08-02 11:52:59 -06:00
Ladybird : : UseLagomNetworking use_lagom_networking )
2023-02-02 03:00:30 -07:00
{
2023-08-01 11:56:10 -06:00
int socket_fds [ 2 ] { } ;
TRY ( Core : : System : : socketpair ( AF_LOCAL , SOCK_STREAM , 0 , socket_fds ) ) ;
int ui_fd = socket_fds [ 0 ] ;
int wc_fd = socket_fds [ 1 ] ;
int fd_passing_socket_fds [ 2 ] { } ;
TRY ( Core : : System : : socketpair ( AF_LOCAL , SOCK_STREAM , 0 , fd_passing_socket_fds ) ) ;
int ui_fd_passing_fd = fd_passing_socket_fds [ 0 ] ;
int wc_fd_passing_fd = fd_passing_socket_fds [ 1 ] ;
if ( auto child_pid = TRY ( Core : : System : : fork ( ) ) ; child_pid = = 0 ) {
TRY ( Core : : System : : close ( ui_fd_passing_fd ) ) ;
TRY ( Core : : System : : close ( ui_fd ) ) ;
auto takeover_string = TRY ( String : : formatted ( " WebContent:{} " , wc_fd ) ) ;
TRY ( Core : : System : : setenv ( " SOCKET_TAKEOVER " sv , takeover_string , true ) ) ;
auto webcontent_fd_passing_socket_string = TRY ( String : : number ( wc_fd_passing_fd ) ) ;
ErrorOr < void > result ;
for ( auto const & path : candidate_web_content_paths ) {
constexpr auto callgrind_prefix_length = 3 ;
if ( Core : : System : : access ( path , X_OK ) . is_error ( ) )
continue ;
auto arguments = Vector {
" valgrind " sv ,
" --tool=callgrind " sv ,
" --instr-atstart=no " sv ,
path . bytes_as_string_view ( ) ,
" --webcontent-fd-passing-socket " sv ,
webcontent_fd_passing_socket_string
} ;
if ( enable_callgrind_profiling = = WebView : : EnableCallgrindProfiling : : No )
arguments . remove ( 0 , callgrind_prefix_length ) ;
if ( is_layout_test_mode = = WebView : : IsLayoutTestMode : : Yes )
arguments . append ( " --layout-test-mode " sv ) ;
2023-08-02 11:52:59 -06:00
if ( use_lagom_networking = = Ladybird : : UseLagomNetworking : : Yes )
2023-08-01 14:39:19 -06:00
arguments . append ( " --use-lagom-networking " sv ) ;
2023-08-01 11:56:10 -06:00
result = Core : : System : : exec ( arguments [ 0 ] , arguments . span ( ) , Core : : System : : SearchInPath : : Yes ) ;
if ( ! result . is_error ( ) )
break ;
}
if ( result . is_error ( ) )
warnln ( " Could not launch any of {}: {} " , candidate_web_content_paths , result . error ( ) ) ;
VERIFY_NOT_REACHED ( ) ;
}
TRY ( Core : : System : : close ( wc_fd_passing_fd ) ) ;
TRY ( Core : : System : : close ( wc_fd ) ) ;
auto socket = TRY ( Core : : LocalSocket : : adopt_fd ( ui_fd ) ) ;
TRY ( socket - > set_blocking ( true ) ) ;
auto new_client = TRY ( adopt_nonnull_ref_or_enomem ( new ( nothrow ) WebView : : WebContentClient ( move ( socket ) , view ) ) ) ;
new_client - > set_fd_passing_socket ( TRY ( Core : : LocalSocket : : adopt_fd ( ui_fd_passing_fd ) ) ) ;
if ( enable_callgrind_profiling = = WebView : : EnableCallgrindProfiling : : Yes ) {
dbgln ( ) ;
dbgln ( " \033 [1;45mLaunched WebContent process under callgrind! \033 [0m " ) ;
dbgln ( " \033 [100mRun ` \033 [4mcallgrind_control -i on \033 [24m` to start instrumentation and ` \033 [4mcallgrind_control -i off \033 [24m` stop it again. \033 [0m " ) ;
dbgln ( ) ;
}
return new_client ;
2023-02-02 03:00:30 -07:00
}
2023-08-01 14:39:19 -06:00
2023-08-02 18:13:23 -06:00
template < typename Client >
ErrorOr < NonnullRefPtr < Client > > launch_generic_server_process ( ReadonlySpan < String > candidate_server_paths , StringView serenity_resource_root , StringView server_name )
2023-08-01 14:39:19 -06:00
{
int socket_fds [ 2 ] { } ;
TRY ( Core : : System : : socketpair ( AF_LOCAL , SOCK_STREAM , 0 , socket_fds ) ) ;
int ui_fd = socket_fds [ 0 ] ;
2023-08-02 18:13:23 -06:00
int server_fd = socket_fds [ 1 ] ;
2023-08-01 14:39:19 -06:00
int fd_passing_socket_fds [ 2 ] { } ;
TRY ( Core : : System : : socketpair ( AF_LOCAL , SOCK_STREAM , 0 , fd_passing_socket_fds ) ) ;
int ui_fd_passing_fd = fd_passing_socket_fds [ 0 ] ;
2023-08-02 18:13:23 -06:00
int server_fd_passing_fd = fd_passing_socket_fds [ 1 ] ;
2023-08-01 14:39:19 -06:00
if ( auto child_pid = TRY ( Core : : System : : fork ( ) ) ; child_pid = = 0 ) {
TRY ( Core : : System : : close ( ui_fd ) ) ;
TRY ( Core : : System : : close ( ui_fd_passing_fd ) ) ;
2023-08-02 18:13:23 -06:00
auto takeover_string = TRY ( String : : formatted ( " {}:{} " , server_name , server_fd ) ) ;
2023-08-01 14:39:19 -06:00
TRY ( Core : : System : : setenv ( " SOCKET_TAKEOVER " sv , takeover_string , true ) ) ;
2023-08-02 18:13:23 -06:00
auto fd_passing_socket_string = TRY ( String : : number ( server_fd_passing_fd ) ) ;
2023-08-01 14:39:19 -06:00
ErrorOr < void > result ;
2023-08-02 18:13:23 -06:00
for ( auto const & path : candidate_server_paths ) {
2023-08-01 14:39:19 -06:00
if ( Core : : System : : access ( path , X_OK ) . is_error ( ) )
continue ;
2023-08-02 18:13:23 -06:00
auto arguments = Vector < StringView , 5 > {
2023-08-01 14:39:19 -06:00
path . bytes_as_string_view ( ) ,
" --fd-passing-socket " sv ,
fd_passing_socket_string ,
2023-08-02 17:21:17 -06:00
" --serenity-resource-root " sv ,
serenity_resource_root ,
2023-08-01 14:39:19 -06:00
} ;
result = Core : : System : : exec ( arguments [ 0 ] , arguments . span ( ) , Core : : System : : SearchInPath : : Yes ) ;
if ( ! result . is_error ( ) )
break ;
}
if ( result . is_error ( ) )
2023-08-02 18:13:23 -06:00
warnln ( " Could not launch any of {}: {} " , candidate_server_paths , result . error ( ) ) ;
2023-08-01 14:39:19 -06:00
VERIFY_NOT_REACHED ( ) ;
}
2023-08-02 18:13:23 -06:00
TRY ( Core : : System : : close ( server_fd ) ) ;
TRY ( Core : : System : : close ( server_fd_passing_fd ) ) ;
2023-08-01 14:39:19 -06:00
auto socket = TRY ( Core : : LocalSocket : : adopt_fd ( ui_fd ) ) ;
TRY ( socket - > set_blocking ( true ) ) ;
2023-08-02 18:13:23 -06:00
auto new_client = TRY ( try_make_ref_counted < Client > ( move ( socket ) ) ) ;
2023-08-01 14:39:19 -06:00
new_client - > set_fd_passing_socket ( TRY ( Core : : LocalSocket : : adopt_fd ( ui_fd_passing_fd ) ) ) ;
return new_client ;
}
2023-08-02 18:13:23 -06:00
ErrorOr < NonnullRefPtr < Protocol : : RequestClient > > launch_request_server_process ( ReadonlySpan < String > candidate_request_server_paths , StringView serenity_resource_root )
{
return launch_generic_server_process < Protocol : : RequestClient > ( candidate_request_server_paths , serenity_resource_root , " RequestServer " sv ) ;
}
ErrorOr < NonnullRefPtr < Protocol : : WebSocketClient > > launch_web_socket_process ( ReadonlySpan < String > candidate_web_socket_paths , StringView serenity_resource_root )
{
return launch_generic_server_process < Protocol : : WebSocketClient > ( candidate_web_socket_paths , serenity_resource_root , " WebSocket " sv ) ;
}