20-logos

前言

logos语法我们其实并不陌生,在之前的文章中早已经使用过了,例如15-Hook原理(二)反Hook防护 & MokeyDev
文章末尾,我们hook了ViewController中的btnClick1方法,使用的就是logos语法。

logos语法,其实是CydiaSubstruct框架提供的一组宏定义。便于开发者使用宏进行HOOK操作。语法简单,功能强大且稳定。

一、基本使用

我们通过一个简单的案例演示一下,如何使用logos语法进行Hook。

1.1 搭建Demo App

  1. 首先,创建一个App,命名为Demo
  2. 打开ViewController.m,写入一下代码👇
#import "ViewController.h"

@interface ViewController ()

@property (weak, nonatomic) IBOutlet UITextField *uid;
@property (weak, nonatomic) IBOutlet UITextField *pwd;

@end

@implementation ViewController

- (void)viewDidLoad {
   [super viewDidLoad];
}

- (IBAction)loginBtnClick:(id)sender {
   [self postUID:self.uid.text PWD:self.pwd.text];
}

-(void)postUID:(NSString *)uid PWD:(NSString *)pwd{
   if ([uid isEqualToString:@"Zang"] && [pwd isEqualToString:@"123456"]) {
       UIAlertController * alertVC = [UIAlertController alertControllerWithTitle:@"登录成功" message:nil preferredStyle:(UIAlertControllerStyleAlert)];
       UIAlertAction * cancel = [UIAlertAction actionWithTitle:@"确定" style:(UIAlertActionStyleCancel) handler:nil];
       [alertVC addAction:cancel];
       [self showViewController:alertVC sender:nil];        
   }else{        
       UIAlertController * alertVC = [UIAlertController alertControllerWithTitle:@"登录失败" message:nil preferredStyle:(UIAlertControllerStyleAlert)];
       UIAlertAction * cancel = [UIAlertAction actionWithTitle:@"确定" style:(UIAlertActionStyleCancel) handler:nil];
       [alertVC addAction:cancel];
       [self showViewController:alertVC sender:nil];
   }
}

@end
  1. 编译项目,生成Demo.app文件。

1.2 dump Demo

我们将Demo视作要Hook的App,那么接下来就是找到要Hook的对象。

  1. 先使用class-dump工具,导出App的头文件,从中找到关键代码。

class-dump官网链接

  1. 将Demo.app的MachO文件,拷贝到class-dump的同级目录,使用class-dump指令,导出头文件👇
./class-dump -H Demo -o ./header/
  1. ViewController.h文件中,就可以找到关键代码👇
#import <UIKit/UIViewController.h>

@class UITextField;

@interface ViewController : UIViewController
{
   UITextField *_uid;
   UITextField *_pwd;
}

- (void).cxx_destruct;
@property(nonatomic) __weak UITextField *pwd; // @synthesize pwd=_pwd;
@property(nonatomic) __weak UITextField *uid; // @synthesize uid=_uid;
- (void)postUID:(id)arg1 PWD:(id)arg2;
- (void)loginBtnClick:(id)arg1;
- (void)viewDidLoad;

@end

1.3 对Demo App项目进行HOOK

接下来就是搭建MokeyDev项目了,用来对Demo App进行Hook。

  1. Demo.app拷贝到MokeyDev项目的TargetApp文件夹中
  2. 在MokeyDemoDylib中的Logos目录下,修改.xm文件的打开方式
  1. 使用logos语法,对ViewController中的loginBtnClick:方法进行hook
    直接在MokeyDemoDylib.xm文件,写入以下代码👇
#import <UIKit/UIKit.h>

@interface ViewController : UIViewController

@end

%hook ViewController

- (void)loginBtnClick:(id)arg1 {
  UIAlertController * alertVC = [UIAlertController alertControllerWithTitle:@"🍺🍺🍺🍺🍺" message:nil preferredStyle:(UIAlertControllerStyleAlert)];
  UIAlertAction * cancel = [UIAlertAction actionWithTitle:@"确定" style:(UIAlertActionStyleCancel) handler:nil];
  [alertVC addAction:cancel];
  [self showViewController:alertVC sender:nil];
}

%end

⚠️注意:@interface ViewController : UIViewController一定要声明,否则会报错👇

使用logos语法,和日常开发的逻辑很相似,并不需要加入hook相关的额外代码。最终.xm文件中的logos语法,会被转为.mm文件中的代码,然后参与编译👇

二、logos语法

logos指令分为三个等级👇

  1. Block level:块等级
  2. Top level:顶部等级
  3. Function level:函数等级

2.1 Block level

块等级,必须以%end结尾,并且不应存在于函数或方法中。

%group
%group Groupname
  • %group用于条件初始化,代码兼容性
  • %group不能在另一个%group块内,未分组的代码都在隐式_ungrouped组中
  • 如果没有其他分组,自动初始化_ungrouped
  • %group必须配合%ctor使用,如果存在多个分组,每个分组都必须使用%init指令,手动对其进行初始化

使用示例👇

%group iOS8
%hook IOS8_SPECIFIC_CLASS
  // your code here
%end // end hook
%end // end group ios8

%group iOS9
%hook IOS9_SPECIFIC_CLASS
  // your code here
%end // end hook
%end // end group ios9

%ctor {
  if (kCFCoreFoundationVersionNumber > 1200) {
      %init(iOS9);
  } else {
      %init(iOS8);
  }
}
%hook
%hook Classname

指定类定义一个HOOK块,可放在%group块内

使用示例👇

%hook ViewController
- (void)loginBtnClick:(id)arg1 {
  NSLog(@"🍺🍺🍺🍺🍺");
  %orig; // 原始方法的调用
}
%end
%new
%new
%new(signature)

HOOK的类添加新方法signature是新方法的OC类型编码,如果省略,将生成一个

⚠️注意:%new必须在%hook块内!

使用示例👇

%hook ViewController
%new
- (void)my_NSLog {
   NSLog(@"🍺🍺🍺🍺🍺");
}
%end
%end
%subclass
%subclass Classname: Superclass <Protocol list>
  • 创建子类,不支持ivars。使用%new添加新方法,实例化新类的对象,必须使用%c操作符。
  • 可以写在%group块内

使用示例👇

%subclass MyObject : NSObject

- (id)init {
  self = %orig;
  [self setSomeValue:@"value"];
  return self;
}

//the following two new methods act as `@property (nonatomic, retain) id someValue;`
%new
- (id)someValue {
  return objc_getAssociatedObject(self, @selector(someValue));
}

%new
- (void)setSomeValue:(id)value {
  objc_setAssociatedObject(self, @selector(someValue), value, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

%end

%ctor {
  MyObject *myObject = [[%c(MyObject) alloc] init];
  NSLog(@"myObject: %@", [myObject someValue]);
}
%property
%property (nonatomic|assign|retain|copy|weak|strong|getter|setter) Type name;
  • 在类中添加新的属性,可添加到%subclass创建的新类中,也可以添加到现有类
  • 必须在%subclass%hook块内
%end
%end

用于标记group/hook/subclass的块结束

2.2 Top level

顶部等级,指令不应存在于group/hook/subclass块中。

%config
%config(Key=Value);

设置Logos配置项。有以下配置选项👇

%hookf
%hookf(rtype, symbolName, args...) { … }

指定函数进行HOOK,如果函数名称作为字符串传递,则将动态查找该函数。

  • 示例1
// 原始函数的定义
FILE *fopen(const char *path, const char *mode);
// 使用以下方式HOOK
%hookf(FILE *, fopen, const char *path, const char *mode) {
  NSLog(@"Hey, we're hooking fopen to deny relative paths!");
  if (path[0] != '/') {
      return NULL;
  }
  return %orig; // 原始方法的调用
}
  • 示例2
CFBooleanRef (*orig_MGGetBoolAnswer)(CFStringRef);
CFBooleanRef fixed_MGGetBoolAnswer(CFStringRef string)
{
  if (CFEqual(string, CFSTR("StarkCapability"))) {
      return kCFBooleanTrue;
  }
  return orig_MGGetBoolAnswer(string);
}

%ctor {
  MSHookFunction(((void *)MSFindSymbol(NULL, "_MGGetBoolAnswer")), (void *)fixed_MGGetBoolAnswer, (void **)&orig_MGGetBoolAnswer);
  ...
}

%hookf(CFBooleanRef, "_MGGetBoolAnswer", CFStringRef string)
{
  if (CFEqual(string, CFSTR("StarkCapability"))) {
      return kCFBooleanTrue;
  }
  return %orig;
}
%ctor
%ctor { … }

生成匿名构造函数

%dtor
%dtor { … }

生成匿名析构函数

2.3 Function level

函数等级,仅存在于函数或方法中

%init
%init;
%init([<class>=<expr>, …]);
%init(Group[, [+|-]<class>=<expr>, …]);

初始化分组默认分组
例如👇

%hook SomeClass
-(id)init {
   return %orig;
}
%end

%ctor {
   %init(SomeClass=objc_getClass("class with spaces in the name"));
}
%c
%c([+|-]Class)

传入实例对象名类名

%orig
%orig
%orig(arg1, …)

原始方法的调用,不能用于%new

%log
%log;
%log([(<type>)<expr>, …]);

输出日志。

总结

  • Logos

    • Logos语法是CydiaSubstruct框架提供的一组宏定义
    • 详情可查看:官方文档
  • 语法

    • %hook%end勾住某个类,在一个代码块中直接写需要勾住的方法
    • %group%end用于分组,配合%ctor构造函数使用,每个组都必须%init
    • %log输出方法的相信信息,包含调用者、方法名、参数
    • %orig调用原始方法,可传递参数,可接收返回值
    • %c类似getClass函数,获取一个类对象
    • %new添加某个方法
  • xm文件

    • xm代表支持OCC/C++语法
    • xm文件不参与执行
    • 编译该文件,需要导入头文件,以便编译通过
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容