Skip to content

Commit c0bcc79

Browse files
committed
Potentially fix random freeze during 'Initializing Jailbreak' stage
1 parent bbadac5 commit c0bcc79

File tree

1 file changed

+29
-18
lines changed

1 file changed

+29
-18
lines changed

Application/Dopamine/Jailbreak/DOJailbreaker.m

+29-18
Original file line numberDiff line numberDiff line change
@@ -303,24 +303,39 @@ - (NSError *)loadBasebinTrustcache
303303
return [NSError errorWithDomain:JBErrorDomain code:JBErrorCodeFailedBasebinTrustcache userInfo:@{NSLocalizedDescriptionKey : @"Failed to load BaseBin trustcache"}];
304304
}
305305

306-
- (NSError *)injectLaunchdHook
307-
{
308-
mach_port_t serverPort = MACH_PORT_NULL;
309-
mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &serverPort);
310-
mach_port_insert_right(mach_task_self(), serverPort, serverPort, MACH_MSG_TYPE_MAKE_SEND);
306+
struct boomerang_info {
307+
mach_port_t serverPort;
308+
dispatch_semaphore_t boomerangDone;
309+
};
311310

312-
// Host a boomerang server that will be used by launchdhook to get the jailbreak primitives from this app
313-
dispatch_semaphore_t boomerangDone = dispatch_semaphore_create(0);
314-
dispatch_source_t serverSource = dispatch_source_create(DISPATCH_SOURCE_TYPE_MACH_RECV, (uintptr_t)serverPort, 0, dispatch_get_main_queue());
315-
dispatch_source_set_event_handler(serverSource, ^{
311+
void *boomerang_server(struct boomerang_info *info)
312+
{
313+
while (true) {
316314
xpc_object_t xdict = nil;
317-
if (!xpc_pipe_receive(serverPort, &xdict)) {
315+
if (!xpc_pipe_receive(info->serverPort, &xdict)) {
318316
if (jbserver_received_boomerang_xpc_message(&gBoomerangServer, xdict) == JBS_BOOMERANG_DONE) {
319-
dispatch_semaphore_signal(boomerangDone);
317+
dispatch_semaphore_signal(info->boomerangDone);
318+
break;
320319
}
321320
}
322-
});
323-
dispatch_resume(serverSource);
321+
}
322+
return NULL;
323+
}
324+
325+
- (NSError *)injectLaunchdHook
326+
{
327+
// Host a boomerang server that will be used by launchdhook to get the jailbreak primitives from this app
328+
mach_port_t serverPort = MACH_PORT_NULL;
329+
mach_port_allocate(mach_task_self(), MACH_PORT_RIGHT_RECEIVE, &serverPort);
330+
mach_port_insert_right(mach_task_self(), serverPort, serverPort, MACH_MSG_TYPE_MAKE_SEND);
331+
332+
struct boomerang_info info;
333+
info.serverPort = serverPort;
334+
info.boomerangDone = dispatch_semaphore_create(0);
335+
336+
pthread_t boomerangThread;
337+
pthread_create(&boomerangThread, NULL, (void *(*)(void *))boomerang_server, &info);
338+
pthread_detach(boomerangThread);
324339

325340
// Stash port to server in launchd's initPorts[2]
326341
// Since we don't have the neccessary entitlements, we need to do it over jbctl
@@ -331,28 +346,24 @@ - (NSError *)injectLaunchdHook
331346
const char *jbctlPath = JBROOT_PATH("/basebin/jbctl");
332347
int spawnError = posix_spawn(&spawnedPid, jbctlPath, NULL, &attr, (char *const *)(const char *[]){ jbctlPath, "internal", "launchd_stash_port", NULL }, NULL);
333348
if (spawnError != 0) {
334-
dispatch_cancel(serverSource);
335349
return [NSError errorWithDomain:JBErrorDomain code:JBErrorCodeFailedLaunchdInjection userInfo:@{NSLocalizedDescriptionKey : [NSString stringWithFormat:@"Spawning jbctl failed with error code %d", spawnError]}];
336350
}
337351
posix_spawnattr_destroy(&attr);
338352
int status = 0;
339353
do {
340354
if (waitpid(spawnedPid, &status, 0) == -1) {
341-
dispatch_cancel(serverSource);
342355
return [NSError errorWithDomain:JBErrorDomain code:JBErrorCodeFailedLaunchdInjection userInfo:@{NSLocalizedDescriptionKey : @"Waiting for jbctl failed"}];;
343356
}
344357
} while (!WIFEXITED(status) && !WIFSIGNALED(status));
345358

346359
// Inject launchdhook.dylib into launchd via opainject
347360
int r = exec_cmd(JBROOT_PATH("/basebin/opainject"), "1", JBROOT_PATH("/basebin/launchdhook.dylib"), NULL);
348361
if (r != 0) {
349-
dispatch_cancel(serverSource);
350362
return [NSError errorWithDomain:JBErrorDomain code:JBErrorCodeFailedLaunchdInjection userInfo:@{NSLocalizedDescriptionKey : [NSString stringWithFormat:@"opainject failed with error code %d", r]}];
351363
}
352364

353365
// Wait for everything to finish
354-
dispatch_semaphore_wait(boomerangDone, DISPATCH_TIME_FOREVER);
355-
dispatch_cancel(serverSource);
366+
dispatch_semaphore_wait(info.boomerangDone, DISPATCH_TIME_FOREVER);
356367
mach_port_deallocate(mach_task_self(), serverPort);
357368

358369
return nil;

0 commit comments

Comments
 (0)