forked from ProcursusTeam/uikittools-ng
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathldrestart.c
151 lines (125 loc) · 3.96 KB
/
ldrestart.c
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
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
// Huge thanks to Morpheus for this
#import <dlfcn.h>
#include <xpc/xpc.h>
#ifndef LOCALEDIR
# define LOCALEDIR "/usr/share/locale"
#endif
#if NLS
# include <libintl.h>
# define _(a) gettext(a)
# define PACKAGE "uikittools-ng"
#else
# define _(a) a
#endif
#ifndef LOCALEDIR
# define LOCALEDIR "/usr/share/locale"
#endif
#if __has_feature(objc_arc)
extern int xpc_pipe_routine(xpc_object_t pipe, xpc_object_t message, __strong xpc_object_t *reply);
#else
extern int xpc_pipe_routine(xpc_object_t pipe, xpc_object_t message, xpc_object_t *reply);
#endif
extern char *xpc_strerror(int);
#define HANDLE_SYSTEM 0
// Some of the routine #s launchd recognizes. There are quite a few subsystems
#define ROUTINE_START 0x32d // 813
#define ROUTINE_STOP 0x32e // 814
#define ROUTINE_LIST 0x32f // 815
// XPC sets up global variables using os_alloc_once. By reverse engineering
// you can determine the values. The only one we actually need is the fourth
// one, which is used as an argument to xpc_pipe_routine
struct xpc_global_data {
uint64_t a;
uint64_t xpc_flags;
mach_port_t task_bootstrap_port; /* 0x10 */
#ifndef _64
uint32_t padding;
#endif
xpc_object_t xpc_bootstrap_pipe; /* 0x18 */
// and there's more, but you'll have to wait for MOXiI 2 for those...
// ...
};
// os_alloc_once_table:
//
// Ripped this from XNU's libsystem
#define OS_ALLOC_ONCE_KEY_MAX 100
struct _os_alloc_once_s {
long once;
void *ptr;
};
extern struct _os_alloc_once_s _os_alloc_once_table[];
static int stopService(const char *ServiceName) {
xpc_object_t dict = xpc_dictionary_create(NULL, NULL, 0);
xpc_dictionary_set_uint64(dict, "subsystem", 3); // subsystem (3)
xpc_dictionary_set_uint64(dict, "handle", HANDLE_SYSTEM);
xpc_dictionary_set_uint64(dict, "routine", ROUTINE_STOP);
xpc_dictionary_set_uint64(dict, "type", 1);
xpc_dictionary_set_string(dict, "name", ServiceName);
xpc_object_t outDict = NULL;
struct xpc_global_data *xpc_gd =
(struct xpc_global_data *)_os_alloc_once_table[1].ptr;
int rc = xpc_pipe_routine(xpc_gd->xpc_bootstrap_pipe, dict, &outDict);
if (rc == 0) {
rc = xpc_dictionary_get_int64(outDict, "error");
if (rc) {
fprintf(stderr, _("Error stopping service: %d - %s\n"), rc,
xpc_strerror(rc));
return (rc);
}
}
return rc;
}
int main() {
#if NLS
setlocale(LC_ALL, "");
bindtextdomain(PACKAGE, LOCALEDIR);
textdomain(PACKAGE);
#endif
xpc_object_t dict = xpc_dictionary_create(NULL, NULL, 0);
xpc_dictionary_set_uint64(dict, "subsystem", 3); // subsystem (3)
xpc_dictionary_set_uint64(dict, "handle", HANDLE_SYSTEM);
xpc_dictionary_set_uint64(dict, "routine", ROUTINE_LIST);
xpc_dictionary_set_uint64(dict, "type", 1); // set to 1
xpc_dictionary_set_bool(dict, "legacy", 1); // mandatory
xpc_object_t outDict = NULL;
struct xpc_global_data *xpc_gd =
(struct xpc_global_data *)_os_alloc_once_table[1].ptr;
int rc = xpc_pipe_routine(xpc_gd->xpc_bootstrap_pipe, dict, &outDict);
if (rc == 0) {
int err = xpc_dictionary_get_int64(outDict, "error");
if (!err) {
// We actually got a reply!
xpc_object_t svcs = xpc_dictionary_get_value(outDict, "services");
if (!svcs) {
fprintf(stderr, _("Error: no services returned for list\n"));
return 1;
}
xpc_type_t svcsType = xpc_get_type(svcs);
if (svcsType != XPC_TYPE_DICTIONARY) {
fprintf(
stderr,
_("Error: services returned for list aren't a dictionary!\n"));
return 2;
}
xpc_dictionary_apply(
svcs, ^bool(const char *label, xpc_object_t svc) {
// if (strstr(label, "openssh") ||
// strstr(label, "log") ||
// strstr(label, "com.apple.diagnosticd")
// ) {
// return 1;
// }
int64_t pid = xpc_dictionary_get_int64(svc, "pid");
if (pid != 0) {
//printf("restart %s : %d\n", label, pid); getchar();
stopService(label);
}
return 1;
});
} else {
fprintf(stderr, _("Error: %d - %s\n"), err, xpc_strerror(err));
}
} else {
fprintf(stderr, _("Unable to get launchd: %d\n"), rc);
}
}