|  | // Copyright 2023 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include <dlfcn.h> | 
|  | #include <mach-o/dyld.h> | 
|  | #include <stdlib.h> | 
|  | #include <string.h> | 
|  |  | 
|  | typedef int (*entry_point)(int, char**); | 
|  |  | 
|  | static char kMainRelativePath[] = "/Frameworks/HelloMain.framework/HelloMain"; | 
|  |  | 
|  | static char* GetHelloMainPath(void) { | 
|  | uint32_t buffer_size = 0; | 
|  | _NSGetExecutablePath(NULL, &buffer_size); | 
|  | if (buffer_size == 0) | 
|  | return NULL; | 
|  |  | 
|  | uint32_t suffix_len = sizeof(kMainRelativePath) + 1; | 
|  | if (buffer_size >= UINT32_MAX - suffix_len) | 
|  | return NULL; | 
|  |  | 
|  | char* buffer = malloc(suffix_len + buffer_size); | 
|  | if (!buffer) | 
|  | return NULL; | 
|  |  | 
|  | if (_NSGetExecutablePath(buffer, &buffer_size) != 0) | 
|  | return NULL; | 
|  |  | 
|  | char* last_sep = strrchr(buffer, '/'); | 
|  | if (!last_sep) | 
|  | return NULL; | 
|  |  | 
|  | memcpy(last_sep, kMainRelativePath, suffix_len); | 
|  | return buffer; | 
|  | } | 
|  |  | 
|  | int main(int argc, char** argv) { | 
|  | char* hello_main_path = GetHelloMainPath(); | 
|  | if (!hello_main_path) | 
|  | return 1; | 
|  |  | 
|  | void* hello_main_handle = dlopen(hello_main_path, RTLD_NOW); | 
|  | if (!hello_main_handle) | 
|  | return 1; | 
|  |  | 
|  | free(hello_main_path); | 
|  | hello_main_path = NULL; | 
|  |  | 
|  | entry_point hello_main_entry = dlsym(hello_main_handle, "hello_main"); | 
|  | if (!hello_main_entry) | 
|  | return 1; | 
|  |  | 
|  | int retcode = hello_main_entry(argc, argv); | 
|  | if (retcode) { | 
|  | return retcode; | 
|  | } | 
|  |  | 
|  | if (!dlclose(hello_main_handle)) { | 
|  | return 1; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } |