A combination of fixes and new features for fruitstrap.
* Ability to pass command-line arguments to the running application. * Fiddled with the device support path discovery so that it works for me with XCode 4.3 on Lion. * Fixed some bugs with write_gdb_prep_cmds (the string replace on dcp_noprivate with an incorrect length lead to a frequent crash). * Fill in address.sun_len to make a warning go away. * Handled gdb exiting slightly more gracefully. * Closed device control sessions when we don't need them. These sessions timing out lead to frequent failures the next time they were used. * Added some getopt argument parsing. * Added verbose and timeout (timeout for connecting to device) arguments.
This commit is contained in:
parent
e6d623252a
commit
91e9185dbe
8
demo.c
8
demo.c
@ -1,5 +1,9 @@
|
|||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
|
||||||
int main() {
|
int main(int argc, const char* argv[]) {
|
||||||
|
int i;
|
||||||
|
for (i = 0; i < argc; i++) {
|
||||||
|
printf("argv[%d] = %s\n", i, argv[i]);
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
258
fruitstrap.c
258
fruitstrap.c
@ -7,6 +7,7 @@
|
|||||||
#include <sys/un.h>
|
#include <sys/un.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
|
#include <getopt.h>
|
||||||
#include "MobileDevice.h"
|
#include "MobileDevice.h"
|
||||||
|
|
||||||
#define FDVENDOR_PATH "/tmp/fruitstrap-remote-debugserver"
|
#define FDVENDOR_PATH "/tmp/fruitstrap-remote-debugserver"
|
||||||
@ -31,7 +32,7 @@
|
|||||||
mem 0x1000 0x3fffffff cache\n\
|
mem 0x1000 0x3fffffff cache\n\
|
||||||
mem 0x40000000 0xffffffff none\n\
|
mem 0x40000000 0xffffffff none\n\
|
||||||
mem 0x00000000 0x0fff none\n\
|
mem 0x00000000 0x0fff none\n\
|
||||||
run\n\
|
run {args}\n\
|
||||||
set minimal-signal-handling 0\n\
|
set minimal-signal-handling 0\n\
|
||||||
set inferior-auto-start-cfm off\n\
|
set inferior-auto-start-cfm off\n\
|
||||||
set sharedLibrary load-rules dyld \".*libobjc.*\" all dyld \".*CoreFoundation.*\" all dyld \".*Foundation.*\" all dyld \".*libSystem.*\" all dyld \".*AppKit.*\" all dyld \".*PBGDBIntrospectionSupport.*\" all dyld \".*/usr/lib/dyld.*\" all dyld \".*CarbonDataFormatters.*\" all dyld \".*libauto.*\" all dyld \".*CFDataFormatters.*\" all dyld \"/System/Library/Frameworks\\\\\\\\|/System/Library/PrivateFrameworks\\\\\\\\|/usr/lib\" extern dyld \".*\" all exec \".*\" all\n\
|
set sharedLibrary load-rules dyld \".*libobjc.*\" all dyld \".*CoreFoundation.*\" all dyld \".*Foundation.*\" all dyld \".*libSystem.*\" all dyld \".*AppKit.*\" all dyld \".*PBGDBIntrospectionSupport.*\" all dyld \".*/usr/lib/dyld.*\" all dyld \".*CarbonDataFormatters.*\" all dyld \".*libauto.*\" all dyld \".*CFDataFormatters.*\" all dyld \"/System/Library/Frameworks\\\\\\\\|/System/Library/PrivateFrameworks\\\\\\\\|/usr/lib\" extern dyld \".*\" all exec \".*\" all\n\
|
||||||
@ -44,9 +45,11 @@ int AMDeviceSecureInstallApplication(int zero, AMDeviceRef device, CFURLRef url,
|
|||||||
int AMDeviceMountImage(AMDeviceRef device, CFStringRef image, CFDictionaryRef options, void *callback, int cbarg);
|
int AMDeviceMountImage(AMDeviceRef device, CFStringRef image, CFDictionaryRef options, void *callback, int cbarg);
|
||||||
int AMDeviceLookupApplications(AMDeviceRef device, int zero, CFDictionaryRef* result);
|
int AMDeviceLookupApplications(AMDeviceRef device, int zero, CFDictionaryRef* result);
|
||||||
|
|
||||||
bool found_device = false, debug = false;
|
bool found_device = false, debug = false, verbose = false;
|
||||||
char *app_path = NULL;
|
char *app_path = NULL;
|
||||||
char *device_id = NULL;
|
char *device_id = NULL;
|
||||||
|
char *args = NULL;
|
||||||
|
int timeout = 0;
|
||||||
CFStringRef last_path = NULL;
|
CFStringRef last_path = NULL;
|
||||||
service_conn_t gdbfd;
|
service_conn_t gdbfd;
|
||||||
|
|
||||||
@ -66,23 +69,83 @@ Boolean path_exists(CFTypeRef path) {
|
|||||||
CFStringRef copy_device_support_path(AMDeviceRef device) {
|
CFStringRef copy_device_support_path(AMDeviceRef device) {
|
||||||
CFStringRef version = AMDeviceCopyValue(device, 0, CFSTR("ProductVersion"));
|
CFStringRef version = AMDeviceCopyValue(device, 0, CFSTR("ProductVersion"));
|
||||||
CFStringRef build = AMDeviceCopyValue(device, 0, CFSTR("BuildVersion"));
|
CFStringRef build = AMDeviceCopyValue(device, 0, CFSTR("BuildVersion"));
|
||||||
CFStringRef path_with_build = CFStringCreateWithFormat(NULL, NULL, CFSTR("/Developer/Platforms/iPhoneOS.platform/DeviceSupport/%@ (%@)"), version, build);
|
const char* home = getenv("HOME");
|
||||||
CFStringRef path_without_build = CFStringCreateWithFormat(NULL, NULL, CFSTR("/Developer/Platforms/iPhoneOS.platform/DeviceSupport/%@"), version);
|
CFStringRef path;
|
||||||
|
bool found = false;
|
||||||
|
|
||||||
|
path = CFStringCreateWithFormat(NULL, NULL, CFSTR("%s/Library/Developer/Xcode/iOS DeviceSupport/%@ (%@)"), home, version, build);
|
||||||
|
found = path_exists(path);
|
||||||
|
|
||||||
|
if (!found)
|
||||||
|
{
|
||||||
|
path = CFStringCreateWithFormat(NULL, NULL, CFSTR("/Developer/Platforms/iPhoneOS.platform/DeviceSupport/%@ (%@)"), version, build);
|
||||||
|
found = path_exists(path);
|
||||||
|
}
|
||||||
|
if (!found)
|
||||||
|
{
|
||||||
|
path = CFStringCreateWithFormat(NULL, NULL, CFSTR("%s/Library/Developer/Xcode/iOS DeviceSupport/%@"), home, version);
|
||||||
|
found = path_exists(path);
|
||||||
|
}
|
||||||
|
if (!found)
|
||||||
|
{
|
||||||
|
path = CFStringCreateWithFormat(NULL, NULL, CFSTR("/Developer/Platforms/iPhoneOS.platform/DeviceSupport/%@"), version);
|
||||||
|
found = path_exists(path);
|
||||||
|
}
|
||||||
|
|
||||||
CFRelease(version);
|
CFRelease(version);
|
||||||
CFRelease(build);
|
CFRelease(build);
|
||||||
|
|
||||||
// they tack the build number on for beta builds
|
if (!found)
|
||||||
// there is almost certainly a better way of doing this
|
{
|
||||||
if (path_exists(path_with_build)) {
|
CFRelease(path);
|
||||||
CFRelease(path_without_build);
|
|
||||||
return path_with_build;
|
|
||||||
} else if (path_exists(path_without_build)) {
|
|
||||||
CFRelease(path_with_build);
|
|
||||||
return path_without_build;
|
|
||||||
} else {
|
|
||||||
printf("[ !! ] Unable to locate DeviceSupport directory.\n");
|
printf("[ !! ] Unable to locate DeviceSupport directory.\n");
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
|
||||||
|
CFStringRef copy_developer_disk_image_path(AMDeviceRef device) {
|
||||||
|
CFStringRef version = AMDeviceCopyValue(device, 0, CFSTR("ProductVersion"));
|
||||||
|
CFStringRef build = AMDeviceCopyValue(device, 0, CFSTR("BuildVersion"));
|
||||||
|
const char *home = getenv("HOME");
|
||||||
|
CFStringRef path;
|
||||||
|
bool found = false;
|
||||||
|
|
||||||
|
path = CFStringCreateWithFormat(NULL, NULL, CFSTR("%s/Library/Developer/Xcode/iOS DeviceSupport/%@ (%@)/DeveloperDiskImage.dmg"), home, version, build);
|
||||||
|
found = path_exists(path);
|
||||||
|
|
||||||
|
if (!found) {
|
||||||
|
path = CFStringCreateWithFormat(NULL, NULL, CFSTR("/Developer/Platforms/iPhoneOS.platform/DeviceSupport/%@ (%@/DeveloperDiskImage.dmg)"), version, build);
|
||||||
|
found = path_exists(path);
|
||||||
|
}
|
||||||
|
if (!found) {
|
||||||
|
path = CFStringCreateWithFormat(NULL, NULL, CFSTR("%s/Library/Developer/Xcode/iOS DeviceSupport/@%/DeveloperDiskImage.dmg"), home, version);
|
||||||
|
found = path_exists(path);
|
||||||
|
}
|
||||||
|
if (!found) {
|
||||||
|
path = CFStringCreateWithFormat(NULL, NULL, CFSTR("/Developer/Platforms/iPhoneOS.platform/DeviceSupport/@%/DeveloperDiskImage.dmg"), version);
|
||||||
|
found = path_exists(path);
|
||||||
|
}
|
||||||
|
if (!found) {
|
||||||
|
path = CFStringCreateWithFormat(NULL, NULL, CFSTR("%s/Library/Developer/Xcode/iOS DeviceSupport/Latest/DeveloperDiskImage.dmg"), home);
|
||||||
|
found = path_exists(path);
|
||||||
|
}
|
||||||
|
if (!found) {
|
||||||
|
path = CFStringCreateWithFormat(NULL, NULL, CFSTR("/Developer/Platforms/iPhoneOS.platform/DeviceSupport/Latest/DeveloperDiskImage.dmg"));
|
||||||
|
found = path_exists(path);
|
||||||
|
}
|
||||||
|
|
||||||
|
CFRelease(version);
|
||||||
|
CFRelease(build);
|
||||||
|
|
||||||
|
if (!found) {
|
||||||
|
CFRelease(path);
|
||||||
|
printf("[ !! ] Unable to locate DeviceSupport directory containing DeveloperDiskImage.dmg.\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return path;
|
||||||
}
|
}
|
||||||
|
|
||||||
void mount_callback(CFDictionaryRef dict, int arg) {
|
void mount_callback(CFDictionaryRef dict, int arg) {
|
||||||
@ -99,10 +162,19 @@ void mount_callback(CFDictionaryRef dict, int arg) {
|
|||||||
|
|
||||||
void mount_developer_image(AMDeviceRef device) {
|
void mount_developer_image(AMDeviceRef device) {
|
||||||
CFStringRef ds_path = copy_device_support_path(device);
|
CFStringRef ds_path = copy_device_support_path(device);
|
||||||
CFStringRef image_path = CFStringCreateWithFormat(NULL, NULL, CFSTR("%@/DeveloperDiskImage.dmg"), ds_path);
|
CFStringRef image_path = copy_developer_disk_image_path(device);
|
||||||
CFStringRef sig_path = CFStringCreateWithFormat(NULL, NULL, CFSTR("%@/DeveloperDiskImage.dmg.signature"), ds_path);
|
CFStringRef sig_path = CFStringCreateWithFormat(NULL, NULL, CFSTR("%@.signature"), image_path);
|
||||||
CFRelease(ds_path);
|
CFRelease(ds_path);
|
||||||
|
|
||||||
|
if (verbose) {
|
||||||
|
printf("Device support path: ");
|
||||||
|
fflush(stdout);
|
||||||
|
CFShow(ds_path);
|
||||||
|
printf("Developer disk image: ");
|
||||||
|
fflush(stdout);
|
||||||
|
CFShow(image_path);
|
||||||
|
}
|
||||||
|
|
||||||
FILE* sig = fopen(CFStringGetCStringPtr(sig_path, kCFStringEncodingMacRoman), "rb");
|
FILE* sig = fopen(CFStringGetCStringPtr(sig_path, kCFStringEncodingMacRoman), "rb");
|
||||||
void *sig_buf = malloc(128);
|
void *sig_buf = malloc(128);
|
||||||
assert(fread(sig_buf, 1, 128, sig) == 128);
|
assert(fread(sig_buf, 1, 128, sig) == 128);
|
||||||
@ -116,8 +188,10 @@ void mount_developer_image(AMDeviceRef device) {
|
|||||||
CFRelease(sig_data);
|
CFRelease(sig_data);
|
||||||
|
|
||||||
int result = AMDeviceMountImage(device, image_path, options, &mount_callback, 0);
|
int result = AMDeviceMountImage(device, image_path, options, &mount_callback, 0);
|
||||||
if (result == 0 || result == 0xe8000076 /* already mounted */) {
|
if (result == 0) {
|
||||||
printf("[ 95%%] Developer disk image mounted successfully\n");
|
printf("[ 95%%] Developer disk image mounted successfully\n");
|
||||||
|
} else if (result == 0xe8000076 /* already mounted */) {
|
||||||
|
printf("[ 95%%] Developer disk image already mounted\n");
|
||||||
} else {
|
} else {
|
||||||
printf("[ !! ] Unable to mount developer disk image. (%x)\n", result);
|
printf("[ !! ] Unable to mount developer disk image. (%x)\n", result);
|
||||||
exit(1);
|
exit(1);
|
||||||
@ -198,7 +272,7 @@ CFURLRef copy_device_app_url(AMDeviceRef device, CFStringRef identifier) {
|
|||||||
|
|
||||||
CFStringRef app_path = CFDictionaryGetValue(app_dict, CFSTR("Path"));
|
CFStringRef app_path = CFDictionaryGetValue(app_dict, CFSTR("Path"));
|
||||||
assert(app_path != NULL);
|
assert(app_path != NULL);
|
||||||
|
|
||||||
CFURLRef url = CFURLCreateWithFileSystemPath(NULL, app_path, kCFURLPOSIXPathStyle, true);
|
CFURLRef url = CFURLCreateWithFileSystemPath(NULL, app_path, kCFURLPOSIXPathStyle, true);
|
||||||
CFRelease(result);
|
CFRelease(result);
|
||||||
return url;
|
return url;
|
||||||
@ -225,20 +299,35 @@ void write_gdb_prep_cmds(AMDeviceRef device, CFURLRef disk_app_url) {
|
|||||||
|
|
||||||
CFStringRef ds_path = copy_device_support_path(device);
|
CFStringRef ds_path = copy_device_support_path(device);
|
||||||
CFStringFindAndReplace(cmds, CFSTR("{ds_path}"), ds_path, range, 0);
|
CFStringFindAndReplace(cmds, CFSTR("{ds_path}"), ds_path, range, 0);
|
||||||
|
range.length = CFStringGetLength(cmds);
|
||||||
|
|
||||||
|
if (args) {
|
||||||
|
CFStringRef cf_args = CFStringCreateWithCString(NULL, args, kCFStringEncodingASCII);
|
||||||
|
CFStringFindAndReplace(cmds, CFSTR("{args}"), cf_args, range, 0);
|
||||||
|
CFRelease(cf_args);
|
||||||
|
} else {
|
||||||
|
CFStringFindAndReplace(cmds, CFSTR(" {args}"), CFSTR(""), range, 0);
|
||||||
|
}
|
||||||
|
range.length = CFStringGetLength(cmds);
|
||||||
|
|
||||||
CFStringRef bundle_identifier = copy_disk_app_identifier(disk_app_url);
|
CFStringRef bundle_identifier = copy_disk_app_identifier(disk_app_url);
|
||||||
CFURLRef device_app_url = copy_device_app_url(device, bundle_identifier);
|
CFURLRef device_app_url = copy_device_app_url(device, bundle_identifier);
|
||||||
CFStringRef device_app_path = CFURLCopyFileSystemPath(device_app_url, kCFURLPOSIXPathStyle);
|
CFStringRef device_app_path = CFURLCopyFileSystemPath(device_app_url, kCFURLPOSIXPathStyle);
|
||||||
CFStringFindAndReplace(cmds, CFSTR("{device_app}"), device_app_path, range, 0);
|
CFStringFindAndReplace(cmds, CFSTR("{device_app}"), device_app_path, range, 0);
|
||||||
|
range.length = CFStringGetLength(cmds);
|
||||||
|
|
||||||
CFStringRef disk_app_path = CFURLCopyFileSystemPath(disk_app_url, kCFURLPOSIXPathStyle);
|
CFStringRef disk_app_path = CFURLCopyFileSystemPath(disk_app_url, kCFURLPOSIXPathStyle);
|
||||||
CFStringFindAndReplace(cmds, CFSTR("{disk_app}"), disk_app_path, range, 0);
|
CFStringFindAndReplace(cmds, CFSTR("{disk_app}"), disk_app_path, range, 0);
|
||||||
|
range.length = CFStringGetLength(cmds);
|
||||||
|
|
||||||
CFURLRef device_container_url = CFURLCreateCopyDeletingLastPathComponent(NULL, device_app_url);
|
CFURLRef device_container_url = CFURLCreateCopyDeletingLastPathComponent(NULL, device_app_url);
|
||||||
CFStringRef device_container_path = CFURLCopyFileSystemPath(device_container_url, kCFURLPOSIXPathStyle);
|
CFStringRef device_container_path = CFURLCopyFileSystemPath(device_container_url, kCFURLPOSIXPathStyle);
|
||||||
CFMutableStringRef dcp_noprivate = CFStringCreateMutableCopy(NULL, 0, device_container_path);
|
CFMutableStringRef dcp_noprivate = CFStringCreateMutableCopy(NULL, 0, device_container_path);
|
||||||
|
range.length = CFStringGetLength(dcp_noprivate);
|
||||||
CFStringFindAndReplace(dcp_noprivate, CFSTR("/private/var/"), CFSTR("/var/"), range, 0);
|
CFStringFindAndReplace(dcp_noprivate, CFSTR("/private/var/"), CFSTR("/var/"), range, 0);
|
||||||
|
range.length = CFStringGetLength(cmds);
|
||||||
CFStringFindAndReplace(cmds, CFSTR("{device_container}"), dcp_noprivate, range, 0);
|
CFStringFindAndReplace(cmds, CFSTR("{device_container}"), dcp_noprivate, range, 0);
|
||||||
|
range.length = CFStringGetLength(cmds);
|
||||||
|
|
||||||
CFURLRef disk_container_url = CFURLCreateCopyDeletingLastPathComponent(NULL, disk_app_url);
|
CFURLRef disk_container_url = CFURLCreateCopyDeletingLastPathComponent(NULL, disk_app_url);
|
||||||
CFStringRef disk_container_path = CFURLCopyFileSystemPath(disk_container_url, kCFURLPOSIXPathStyle);
|
CFStringRef disk_container_path = CFURLCopyFileSystemPath(disk_container_url, kCFURLPOSIXPathStyle);
|
||||||
@ -275,6 +364,7 @@ void start_remote_debug_server(AMDeviceRef device) {
|
|||||||
memset(&address, 0, sizeof(address));
|
memset(&address, 0, sizeof(address));
|
||||||
address.sun_family = AF_UNIX;
|
address.sun_family = AF_UNIX;
|
||||||
strcpy(address.sun_path, FDVENDOR_PATH);
|
strcpy(address.sun_path, FDVENDOR_PATH);
|
||||||
|
address.sun_len = SUN_LEN(&address);
|
||||||
CFDataRef address_data = CFDataCreate(NULL, (const UInt8 *)&address, sizeof(address));
|
CFDataRef address_data = CFDataCreate(NULL, (const UInt8 *)&address, sizeof(address));
|
||||||
|
|
||||||
unlink(FDVENDOR_PATH);
|
unlink(FDVENDOR_PATH);
|
||||||
@ -284,9 +374,14 @@ void start_remote_debug_server(AMDeviceRef device) {
|
|||||||
CFRunLoopAddSource(CFRunLoopGetMain(), CFSocketCreateRunLoopSource(NULL, fdvendor, 0), kCFRunLoopCommonModes);
|
CFRunLoopAddSource(CFRunLoopGetMain(), CFSocketCreateRunLoopSource(NULL, fdvendor, 0), kCFRunLoopCommonModes);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void gdb_ready_handler(int signum)
|
||||||
|
{
|
||||||
|
_exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
void handle_device(AMDeviceRef device) {
|
void handle_device(AMDeviceRef device) {
|
||||||
if (found_device) return; // handle one device only
|
if (found_device) return; // handle one device only
|
||||||
|
|
||||||
CFStringRef found_device_id = AMDeviceCopyDeviceIdentifier(device);
|
CFStringRef found_device_id = AMDeviceCopyDeviceIdentifier(device);
|
||||||
|
|
||||||
if (device_id != NULL) {
|
if (device_id != NULL) {
|
||||||
@ -312,30 +407,52 @@ void handle_device(AMDeviceRef device) {
|
|||||||
CFURLRef relative_url = CFURLCreateWithFileSystemPath(NULL, path, kCFURLPOSIXPathStyle, false);
|
CFURLRef relative_url = CFURLCreateWithFileSystemPath(NULL, path, kCFURLPOSIXPathStyle, false);
|
||||||
CFURLRef url = CFURLCopyAbsoluteURL(relative_url);
|
CFURLRef url = CFURLCopyAbsoluteURL(relative_url);
|
||||||
|
|
||||||
CFRelease(path);
|
|
||||||
CFRelease(relative_url);
|
CFRelease(relative_url);
|
||||||
|
|
||||||
|
int afcFd;
|
||||||
|
assert(AMDeviceStartService(device, CFSTR("com.apple.afc"), &afcFd, NULL) == 0);
|
||||||
|
assert(AMDeviceStopSession(device) == 0);
|
||||||
|
assert(AMDeviceDisconnect(device) == 0);
|
||||||
|
assert(AMDeviceTransferApplication(afcFd, path, NULL, transfer_callback, NULL) == 0);
|
||||||
|
|
||||||
|
close(afcFd);
|
||||||
|
|
||||||
CFStringRef keys[] = { CFSTR("PackageType") };
|
CFStringRef keys[] = { CFSTR("PackageType") };
|
||||||
CFStringRef values[] = { CFSTR("Developer") };
|
CFStringRef values[] = { CFSTR("Developer") };
|
||||||
CFDictionaryRef options = CFDictionaryCreate(NULL, (const void **)&keys, (const void **)&values, 1, &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks);
|
CFDictionaryRef options = CFDictionaryCreate(NULL, (const void **)&keys, (const void **)&values, 1, &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks);
|
||||||
|
|
||||||
mach_error_t transfer_error = AMDeviceSecureTransferPath(0, device, url, options, &transfer_callback, 0);
|
AMDeviceConnect(device);
|
||||||
if (transfer_error) {
|
assert(AMDeviceIsPaired(device));
|
||||||
printf("[ !! ] Unable to transfer package to device. (%x)\n", transfer_error);
|
assert(AMDeviceValidatePairing(device) == 0);
|
||||||
|
assert(AMDeviceStartSession(device) == 0);
|
||||||
|
|
||||||
|
int installFd;
|
||||||
|
assert(AMDeviceStartService(device, CFSTR("com.apple.mobile.installation_proxy"), &installFd, NULL) == 0);
|
||||||
|
|
||||||
|
assert(AMDeviceStopSession(device) == 0);
|
||||||
|
assert(AMDeviceDisconnect(device) == 0);
|
||||||
|
|
||||||
|
mach_error_t result = AMDeviceInstallApplication(installFd, path, options, install_callback, NULL);
|
||||||
|
if (result != 0)
|
||||||
|
{
|
||||||
|
printf("AMDeviceInstallApplication failed: %d\n", result);
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
mach_error_t install_error = AMDeviceSecureInstallApplication(0, device, url, options, &install_callback, 0);
|
close(installFd);
|
||||||
if (install_error) {
|
|
||||||
printf("[ !! ] Unable to install package. (%x)\n", install_error);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
CFRelease(path);
|
||||||
CFRelease(options);
|
CFRelease(options);
|
||||||
|
|
||||||
printf("[100%%] Installed package %s\n", app_path);
|
printf("[100%%] Installed package %s\n", app_path);
|
||||||
|
|
||||||
if (!debug) exit(0); // no debug phase
|
if (!debug) exit(0); // no debug phase
|
||||||
|
|
||||||
|
AMDeviceConnect(device);
|
||||||
|
assert(AMDeviceIsPaired(device));
|
||||||
|
assert(AMDeviceValidatePairing(device) == 0);
|
||||||
|
assert(AMDeviceStartSession(device) == 0);
|
||||||
|
|
||||||
printf("------ Debug phase ------\n");
|
printf("------ Debug phase ------\n");
|
||||||
|
|
||||||
mount_developer_image(device); // put debugserver on the device
|
mount_developer_image(device); // put debugserver on the device
|
||||||
@ -347,11 +464,13 @@ void handle_device(AMDeviceRef device) {
|
|||||||
printf("[100%%] Connecting to remote debug server\n");
|
printf("[100%%] Connecting to remote debug server\n");
|
||||||
printf("-------------------------\n");
|
printf("-------------------------\n");
|
||||||
|
|
||||||
|
signal(SIGHUP, gdb_ready_handler);
|
||||||
|
|
||||||
pid_t parent = getpid();
|
pid_t parent = getpid();
|
||||||
int pid = fork();
|
int pid = fork();
|
||||||
if (pid == 0) {
|
if (pid == 0) {
|
||||||
system(GDB_SHELL); // launch gdb
|
system(GDB_SHELL); // launch gdb
|
||||||
kill(parent, SIGTERM); // "No. I am your father."
|
kill(parent, SIGHUP); // "No. I am your father."
|
||||||
_exit(0);
|
_exit(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -365,35 +484,78 @@ void device_callback(struct am_device_notification_callback_info *info, void *ar
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int main(int argc, char *argv[]) {
|
void timeout_callback(CFRunLoopTimerRef timer, void *info) {
|
||||||
if (argc < 2 || argc > 4) {
|
if (!found_device) {
|
||||||
printf("usage: %s [-d] <app> [device_id]\n", argv[0]);
|
printf("Timed out waiting for device.\n");
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void usage(const char* app) {
|
||||||
if (strcmp(argv[1], "-d") == 0) {
|
printf("usage: %s [-d/--debug] [-i/--id device_id] -b/--bundle bundle.app [-a/--args arguments] [-t/--timeout timeout(seconds)]\n", app);
|
||||||
assert(argc == 3 || argc == 4);
|
}
|
||||||
debug = true;
|
|
||||||
app_path = argv[2];
|
int main(int argc, char *argv[]) {
|
||||||
if (argc == 4) {
|
static struct option longopts[] = {
|
||||||
device_id = argv[3];
|
{ "debug", no_argument, NULL, 'd' },
|
||||||
}
|
{ "id", required_argument, NULL, 'i' },
|
||||||
printf("------ Install phase ------\n");
|
{ "bundle", required_argument, NULL, 'b' },
|
||||||
} else {
|
{ "args", required_argument, NULL, 'a' },
|
||||||
assert(argc == 2 || argc == 3);
|
{ "verbose", no_argument, NULL, 'v' },
|
||||||
app_path = argv[1];
|
{ "timeout", required_argument, NULL, 't' },
|
||||||
if (argc == 3) {
|
{ NULL, 0, NULL, 0 },
|
||||||
device_id = argv[2];
|
};
|
||||||
|
char ch;
|
||||||
|
|
||||||
|
while ((ch = getopt_long(argc, argv, "dvi:b:a:t:", longopts, NULL)) != -1)
|
||||||
|
{
|
||||||
|
switch (ch) {
|
||||||
|
case 'd':
|
||||||
|
debug = 1;
|
||||||
|
break;
|
||||||
|
case 'i':
|
||||||
|
device_id = optarg;
|
||||||
|
break;
|
||||||
|
case 'b':
|
||||||
|
app_path = optarg;
|
||||||
|
break;
|
||||||
|
case 'a':
|
||||||
|
args = optarg;
|
||||||
|
break;
|
||||||
|
case 'v':
|
||||||
|
verbose = 1;
|
||||||
|
break;
|
||||||
|
case 't':
|
||||||
|
timeout = atoi(optarg);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
usage(argv[0]);
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!app_path) {
|
||||||
|
usage(argv[0]);
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("------ Install phase ------\n");
|
||||||
|
|
||||||
assert(access(app_path, F_OK) == 0);
|
assert(access(app_path, F_OK) == 0);
|
||||||
|
|
||||||
AMDSetLogLevel(5); // otherwise syslog gets flooded with crap
|
AMDSetLogLevel(5); // otherwise syslog gets flooded with crap
|
||||||
printf("[....] Waiting for iOS device to be connected\n");
|
if (timeout > 0)
|
||||||
|
{
|
||||||
|
CFRunLoopTimerRef timer = CFRunLoopTimerCreate(NULL, CFAbsoluteTimeGetCurrent() + timeout, 0, 0, 0, timeout_callback, NULL);
|
||||||
|
CFRunLoopAddTimer(CFRunLoopGetCurrent(), timer, kCFRunLoopCommonModes);
|
||||||
|
printf("[....] Waiting up to %d seconds for iOS device to be connected\n", timeout);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
printf("[....] Waiting for iOS device to be connected\n");
|
||||||
|
}
|
||||||
|
|
||||||
struct am_device_notification *notify;
|
struct am_device_notification *notify;
|
||||||
AMDeviceNotificationSubscribe(&device_callback, 0, 0, NULL, ¬ify);
|
AMDeviceNotificationSubscribe(&device_callback, 0, 0, NULL, ¬ify);
|
||||||
CFRunLoopRun();
|
CFRunLoopRun();
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user