Files
coruna/TweakLoader/TweakLoader.m
2026-03-12 05:42:32 +07:00

80 lines
2.7 KiB
Objective-C

@import Darwin;
@import MachO;
#include <mach-o/ldsyms.h> /* _mh_dylib_header */
// Function pointers
extern pthread_t pthread_main_thread_np(void);
extern void _pthread_set_self(pthread_t p);
void (*_abort)(void);
int (*_close)(int);
void* (*_dlsym)(void *, const char *);
uint8_t* (*_getsectiondata)(const struct mach_header_64 *, const char *, const char *, unsigned long *);
thread_t (*_mach_thread_self)(void);
int (*_open)(const char *, int, ...);
void (*__pthread_set_self)(pthread_t p);
pthread_t (*_pthread_main_thread_np)(void);
int (*_strncmp)(const char *s1, const char *s2, size_t n);
kern_return_t (*_thread_terminate)(mach_port_t);
int (*_write)(int, const void *, size_t);
int dyld_lv_bypass_init(void * (*_dlsym)(void* handle, const char* symbol), const char *next_stage_dylib_path);
void save_section_to_file(const char *section, const char *path) {
size_t dylib_size = 0;
const char *dylib = (const char *)_getsectiondata((struct mach_header_64 *)&_mh_dylib_header, "__TEXT", section, &dylib_size);
if (!dylib || dylib_size == 0) return;
int fd = _open(path, O_CREAT | O_WRONLY | O_TRUNC, 0644);
if (fd < 0) return;
_write(fd, dylib, dylib_size);
_close(fd);
}
const char *save_actual_dylib(void) {
const char *path = "/tmp/actual.dylib";
save_section_to_file("__SBTweak", path);
return path;
}
#if __arm64e__
__attribute__((noinline)) void *pacia(void* ptr, uint64_t ctx) {
__asm__("xpaci %[value]\n" : [value] "+r"(ptr));
__asm__("pacia %0, %1" : "+r"(ptr) : "r"(ctx));
return ptr;
}
#endif
// Entry point when loaded by Coruna
int last(void) {
#if __arm64e__
_dlsym = pacia(dlsym, 0);
__pthread_set_self = pacia(_pthread_set_self, 0);
_pthread_main_thread_np = pacia(pthread_main_thread_np, 0);
#else
_dlsym = dlsym;
__pthread_set_self = _pthread_set_self;
_pthread_main_thread_np = pthread_main_thread_np;
#endif
__pthread_set_self(_pthread_main_thread_np());
_abort = _dlsym(RTLD_DEFAULT, "abort");
_close = _dlsym(RTLD_DEFAULT, "close");
_getsectiondata = _dlsym(RTLD_DEFAULT, "getsectiondata");
_mach_thread_self = _dlsym(RTLD_DEFAULT, "mach_thread_self");
_open = _dlsym(RTLD_DEFAULT, "open");
_strncmp = _dlsym(RTLD_DEFAULT, "strncmp");
_thread_terminate = _dlsym(RTLD_DEFAULT, "thread_terminate");
_write = _dlsym(RTLD_DEFAULT, "write");
// setup dyld validation bypass
const char *path = save_actual_dylib();
dyld_lv_bypass_init(_dlsym, path);
// should not return
_thread_terminate(_mach_thread_self());
return 0;
}
int end(void) {
// should never be called
return 0;
}