From 39f1785165f45ef1eaefb88b023d981f921e8446 Mon Sep 17 00:00:00 2001 From: "Jay Freeman (saurik)" Date: Sat, 20 Dec 2008 09:15:34 +0000 Subject: [PATCH] Completely redid Mobile Safe Mode, added MobileUnions (unreleased), and performed a minor optimization to MSHookFunction (remove an extra jump between through-handlers). --- MobileSafety.mm | 115 +++++++++++++++++++++++++++++++++++++----------- make.sh | 4 +- 2 files changed, 91 insertions(+), 28 deletions(-) diff --git a/MobileSafety.mm b/MobileSafety.mm index 6684eb4..83070a7 100644 --- a/MobileSafety.mm +++ b/MobileSafety.mm @@ -38,31 +38,84 @@ #import #import #import +#import + +#import +#import +#import +#import +#import #include @protocol MobileSubstrate -- (id) sharedInstance; -- (void) activateAlertItem:(id)item; -- (id) initWithTitle:(NSString *)title body:(NSString *)body; - (id) ms$initWithSize:(CGSize)size; -- (void) ms$drawRect:(CGRect)rect; -- (id) darkGrayColor; -- (void) setBackgroundColor:(id)color; - (int) ms$maxIconColumns; @end -static void MSAlert(id self, SEL sel) { +Class $SafeModeAlertItem; +Class $SBAlertItemsController; + +void SafeModeAlertItem$alertSheet$buttonClicked$(id self, SEL sel, id sheet, int button) { + switch (button) { + case 1: + break; + + case 2: + exit(0); + break; + + case 3: + [UIApp openURL:[NSURL URLWithString:@"http://cydia.saurik.com/safemode/"]]; + break; + } + + [self dismiss]; +} + +void SafeModeAlertItem$configure$requirePasscodeForActions$(id self, SEL sel, BOOL configure, BOOL require) { + UIModalView *sheet([self alertSheet]); + [sheet setDelegate:self]; + [sheet setBodyText:@"We apologize for the inconvenience, but SpringBoard has just crashed.\n\nA recent software installation, upgrade, or removal might have been the cause of this.\n\nYour device is now running in Safe Mode. All extensions that support this safety system are disabled.\n\nReboot (or restart SpringBoard) to return to the normal mode. To return to this dialog touch the status bar."]; + [sheet addButtonWithTitle:@"OK"]; + [sheet addButtonWithTitle:@"Restart"]; + [sheet addButtonWithTitle:@"Help"]; + [sheet setNumberOfRows:1]; +} + +void SafeModeAlertItem$performUnlockAction(id self, SEL sel) { + [[$SBAlertItemsController sharedInstance] activateAlertItem:self]; +} + +static void MSAlert() { + if ($SafeModeAlertItem == nil) + $SafeModeAlertItem = objc_lookUpClass("SafeModeAlertItem"); + if ($SafeModeAlertItem == nil) { + $SafeModeAlertItem = objc_allocateClassPair(objc_getClass("SBAlertItem"), "SafeModeAlertItem", 0); + if ($SafeModeAlertItem == nil) + return; + + class_addMethod($SafeModeAlertItem, @selector(alertSheet:buttonClicked:), (IMP) &SafeModeAlertItem$alertSheet$buttonClicked$, "v@:@i"); + class_addMethod($SafeModeAlertItem, @selector(configure:requirePasscodeForActions:), (IMP) &SafeModeAlertItem$configure$requirePasscodeForActions$, "v@:cc"); + class_addMethod($SafeModeAlertItem, @selector(performUnlockAction), (IMP) SafeModeAlertItem$performUnlockAction, "v@:"); + objc_registerClassPair($SafeModeAlertItem); + } + + if ($SBAlertItemsController != nil) + [[$SBAlertItemsController sharedInstance] activateAlertItem:[[$SafeModeAlertItem alloc] init]]; +} + +MSHook(void, SBStatusBar$mouseDown$, SBStatusBar *self, SEL sel, GSEventRef event) { + MSAlert(); + _SBStatusBar$mouseDown$(self, sel, event); +} + +static void SBIconController$showInfoAlertIfNeeded(id self, SEL sel) { static bool loaded = false; - if (!loaded) - loaded = true; - else return; - - [[(id) objc_getClass("SBAlertItemsController") sharedInstance] activateAlertItem: - [[(id) objc_getClass("SBDismissOnlyAlertItem") alloc] - initWithTitle:@"Mobile Substrate Safe Mode" - body:@"We apologize for the inconvenience, but SpringBoard has just crashed.\n\nA recent software installation, upgrade, or removal might have been the cause of this.\n\nIf you are using IntelliScreen, then it probably crashed.\n\nYour device is now running in Safe Mode. All extensions that support this safety system are disabled.\n\nReboot (or restart SpringBoard) to return to the normal mode." - ]]; + if (loaded) + return; + loaded = true; + MSAlert(); } static int SBButtonBar$maxIconColumns(id self, SEL sel) { @@ -82,30 +135,38 @@ static int SBButtonBar$maxIconColumns(id self, SEL sel) { } return max; } -static id SBContentLayer$initWithSize$(id self, SEL sel, CGSize size) { +static id SBContentLayer$initWithSize$(SBContentLayer *self, SEL sel, CGSize size) { self = [self ms$initWithSize:size]; if (self == nil) return nil; - [self setBackgroundColor:[(id) objc_getClass("UIColor") darkGrayColor]]; + [self setBackgroundColor:[UIColor darkGrayColor]]; return self; } -static void SBStatusBarTimeView$drawRect$(id self, SEL sel, CGRect rect) { - id &_time(MSHookIvar(self, "_time")); +MSHook(void, SBStatusBarTimeView$tile, SBStatusBarTimeView *self, SEL sel) { + NSString *&_time(MSHookIvar(self, "_time")); + CGRect &_textRect(MSHookIvar(self, "_textRect")); if (_time != nil) - [_time autorelease]; - _time = [@"Safe Mode" retain]; - return [self ms$drawRect:rect]; + [_time release]; + _time = [@"Safe Mode, Click!" retain]; + GSFontRef font([self textFont]); + CGSize size([_time sizeWithFont:(id)font]); + CGRect frame([self frame]); + NSLog(@"%f:%f,%f:%f:%u", size.height, size.width, frame.size.height, frame.size.width, [[(id) objc_getClass("SBStatusBarController") sharedStatusBarController] statusBarOrientation]); + _textRect.size = size; + _textRect.origin.x = (frame.size.width - size.width) / 2; + _textRect.origin.y = (frame.size.height - size.height) / 2; } #define Dylib_ "/Library/MobileSubstrate/MobileSubstrate.dylib" -extern "C" void MSSafety() { +extern "C" void MSInitialize() { NSLog(@"MS:Warning: Entering Safe Mode"); MSHookMessage(objc_getClass("SBButtonBar"), @selector(maxIconColumns), (IMP) &SBButtonBar$maxIconColumns, "ms$"); MSHookMessage(objc_getClass("SBContentLayer"), @selector(initWithSize:), (IMP) &SBContentLayer$initWithSize$, "ms$"); - MSHookMessage(objc_getClass("SBStatusBarTimeView"), @selector(drawRect:), (IMP) &SBStatusBarTimeView$drawRect$, "ms$"); + _SBStatusBar$mouseDown$ = MSHookMessage(objc_getClass("SBStatusBar"), @selector(mouseDown:), &$SBStatusBar$mouseDown$); + _SBStatusBarTimeView$tile = MSHookMessage(objc_getClass("SBStatusBarTimeView"), @selector(tile), &$SBStatusBarTimeView$tile); char *dil = getenv("DYLD_INSERT_LIBRARIES"); if (dil == NULL) @@ -125,9 +186,11 @@ extern "C" void MSSafety() { } } + $SBAlertItemsController = objc_getClass("SBAlertItemsController"); + if (Class _class = objc_getClass("SBIconController")) { SEL sel(@selector(showInfoAlertIfNeeded)); if (Method method = class_getInstanceMethod(_class, sel)) - method_setImplementation(method, (IMP) &MSAlert); + method_setImplementation(method, (IMP) &SBIconController$showInfoAlertIfNeeded); } } diff --git a/make.sh b/make.sh index 79d3ffc..8560869 100755 --- a/make.sh +++ b/make.sh @@ -1,5 +1,5 @@ #!/bin/bash flags=(-O2 -g0 -fno-exceptions -fvisibility=hidden) cycc -i2.0 -oMobileSafety.dylib -- "${flags[@]}" -dynamiclib MobileSafety.mm \ - -framework CoreFoundation -framework Foundation \ - -L. -lsubstrate -lobjc -init _MSSafety + -framework CoreFoundation -framework Foundation -framework UIKit \ + -L. -lsubstrate -lobjc -init _MSInitialize -- 2.45.2