0%

frida hook

frida启动方式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
# frida -h
usage: frida [options] target

positional arguments:
args extra arguments and/or target

optional arguments:
-h, --help show this help message and exit
-D ID, --device ID connect to device with the given ID
-U, --usb connect to USB device
-R, --remote connect to remote frida-server
-H HOST, --host HOST connect to remote frida-server on HOST
--certificate CERTIFICATE
speak TLS with HOST, expecting CERTIFICATE
--origin ORIGIN connect to remote server with “Origin” header set to ORIGIN
--token TOKEN authenticate with HOST using TOKEN
--keepalive-interval INTERVAL
set keepalive interval in seconds, or 0 to disable (defaults to -1 to auto-select based on transport)
--p2p establish a peer-to-peer connection with target
--stun-server ADDRESS
set STUN server ADDRESS to use with --p2p
--relay address,username,password,turn-{udp,tcp,tls}
add relay to use with --p2p
-f TARGET, --file TARGET
spawn FILE
-F, --attach-frontmost
attach to frontmost application
-n NAME, --attach-name NAME
attach to NAME
-p PID, --attach-pid PID
attach to PID
-W PATTERN, --await PATTERN
await spawn matching PATTERN
--stdio {inherit,pipe}
stdio behavior when spawning (defaults to “inherit”)
--aux option set aux option when spawning, such as “uid=(int)42” (supported types are: string, bool, int)
--realm {native,emulated}
realm to attach in
--runtime {qjs,v8} script runtime to use
--debug enable the Node.js compatible script debugger
--squelch-crash if enabled, will not dump crash report to console
-O FILE, --options-file FILE
text file containing additional command line options
--version show program's version number and exit
-l SCRIPT, --load SCRIPT
load SCRIPT
-P PARAMETERS_JSON, --parameters PARAMETERS_JSON
parameters as JSON, same as Gadget
-C USER_CMODULE, --cmodule USER_CMODULE
load CMODULE
--toolchain {any,internal,external}
CModule toolchain to use when compiling from source code
-c CODESHARE_URI, --codeshare CODESHARE_URI
load CODESHARE_URI
-e CODE, --eval CODE evaluate CODE
-q quiet mode (no prompt) and quit after -l and -e
-t TIMEOUT, --timeout TIMEOUT
seconds to wait before terminating in quiet mode
--no-pause automatically start main thread after startup
-o LOGFILE, --output LOGFILE
output to log file
--eternalize eternalize the script before exit
--exit-on-error exit with code 1 after encountering any exception in the SCRIPT
--auto-perform wrap entered code with Java.perform
--auto-reload Enable auto reload of provided scripts and c module (on by default, will be required in the future)
--no-auto-reload Disable auto reload of provided scripts and c module

Injection

attach hook: 这种模式建立在app已经启动的情况下,frida利用ptrace的原理注入app进而完成Hook
spawn hook: 将app启动权限交与frida 来控制。使用spawn实现hook时会由frida将app重启在进行hook
注意:由于attach hook 基于ptrace原理进行完成,因此无法在IDA正在调试的目标app以attach注入进程中,当然若先用frida
attach注入
在使用IDA进行调试则正常

Shell

attach

1
2
frida -Ul script_hook.js [-n] app_name
frida -Ul script_hook.js -p pid

spawn

1
frida -Ul script_hook.js -f Identifier(package name) --no-pause

Python script

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
import sys
import frida
from loguru import logger
from pptrint import pprint


def message_callback(message, data):
logger.info(f"[*] {message}")


device = frida.get_use_device(-1)

# attach
process = device.attach('className')
script = process.create_script('hook_script')
script.on('message', message_callback)
script.load()
sys.stdin.read()

# spawn
device = frida.get_use_device(-1)
pid = devices.spawn(['packageName'])
process = device.attach(pid)
script = process.create_script('hook_script')
script.on('message', message_callback)
script.load()
sys.stdin.read()

frida api

JavaScript-api : https://frida.re/docs/javascript-api

JavaScript-api-java: https://frida.re/docs/javascript-api/#java

JavaScript-api-module: https://frida.re/docs/javascript-api/#module

Hook

1
2
3
4
Java.perfrom(function () {
console.log('script successfully loaded, start hook...');
// hook script
});

Hook 类方法

1
2
3
4
5
6
7
8
9
Java.perfrom(function () {
console.log('script successfully loaded, start hook...');
// hook class script
let class_name = Java.use('com.xxx.xxx.class_name');
class_name.method.implementation = function () {
// do something
// this.xx
}
});

this.成员变量名.value

Hook 内部(匿名)类方法

1
2
3
4
5
6
7
8
9
10
Java.perfrom(function () {
console.log('script successfully loaded, start hook...');
// hook class script
// 类路径$内部类名 在smail找
let class_name = Java.use('com.xxx.xxx.class_name$xx');
class_name.method.implementation = function () {
// do something
// this.xx
}
});

从匿名类/内部类访问外部类的属性写法: this.this$0.value.外部类的属性名.value

Hook 重载方法

1
2
3
4
5
6
7
8
9
10
11
12
13
Java.perfrom(function () {
console.log('script successfully loaded, start hook...');
// hook class script
// 类路径$内部类名 在smail找
let class_name = Java.use('com.xxx.xxx.class_name');
class_name.method.overload(参数1, 参数2
...).
implementation = function () {
// do something
// this.xx
}
});
// overload(参数1,参数2...) 可以根据报错来确定

Hook 构造方法

1
2
3
4
5
6
7
8
9
10
Java.perfrom(function () {
console.log('script successfully loaded, start hook...');
// hook class script
// 类路径$内部类名 在smail找
let class_name = Java.use('com.xxx.xxx.class_name');
class_name.$init().implementation = function () {
// do something
// this.xx
}
});

Hook 实例

1
2
3
4
Java.perfrom(function () {
console.log('script successfully loaded, start hook...');
let variable = Java.use('com.xxx.xxx.class_name').$new(参数);
});