forked from opa334/libSandy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlibSandy.m
126 lines (109 loc) · 4.48 KB
/
libSandy.m
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
#import <CoreFoundation/CoreFoundation.h>
#import <Foundation/Foundation.h>
#import <xpc/xpc.h>
#import <mach-o/dyld.h>
#import <sandbox_private.h>
#import <roothide.h>
#import <sandyd.h>
#import "HBLogWeak.h"
#import "libSandy.h"
NSString *safe_getExecutablePath()
{
char executablePathC[PATH_MAX];
uint32_t executablePathCSize = sizeof(executablePathC);
_NSGetExecutablePath(&executablePathC[0], &executablePathCSize);
return [NSString stringWithUTF8String:executablePathC];
}
// calling libSandy functions from inside sandyd itself locks the system up so we need to prevent it
static BOOL isRunningInsideSandyd()
{
static BOOL isSandyd;
static dispatch_once_t onceToken;
dispatch_once (&onceToken, ^{
isSandyd = [safe_getExecutablePath().lastPathComponent isEqualToString:@"sandyd"];
});
return isSandyd;
}
static BOOL sandydCommunicationWorks(void)
{
return sandbox_check(getpid(), "mach-lookup", SANDBOX_FILTER_GLOBAL_NAME | SANDBOX_CHECK_NO_REPORT, "com.opa334.sandyd") == 0;
}
static BOOL consumeGlobalExtensions(void)
{
if (!sandydCommunicationWorks()) {
NSString *plistPath = jbroot(@"/usr/lib/sandyd_global.plist");
if (![[NSFileManager defaultManager] fileExistsAtPath:plistPath]) {
NSLog(@"[libSandy consumeGlobalExtensions] FATAL ERROR: /usr/lib/sandyd_global.plist does not exist");
return NO;
}
NSDictionary *plistDict = [NSDictionary dictionaryWithContentsOfFile:plistPath];
if (!plistDict) return NO;
NSArray *extensions = plistDict[@"extensions"];
if (!extensions) return NO;
for (NSString *extension in extensions) {
__unused int cr = sandbox_extension_consume(extension.UTF8String);
HBLogDebugWeak(@"[libSandy consumeGlobalExtensions] sandbox_extension_consume(\"%s\") => %d", extension.UTF8String, cr);
}
if (!sandydCommunicationWorks()) {
NSLog(@"[libSandy consumeGlobalExtensions] FATAL ERROR: communication still does not work, even after consuming sandbox extensions");
return NO;
}
}
return YES;
}
static xpc_object_t sandydSendMessage(xpc_object_t message)
{
if (!consumeGlobalExtensions()) return nil;
xpc_connection_t connection = xpc_connection_create_mach_service("com.opa334.sandyd", 0, XPC_CONNECTION_MACH_SERVICE_PRIVILEGED);
xpc_connection_set_event_handler(connection, ^(xpc_object_t object){});
xpc_connection_resume(connection);
return xpc_connection_send_message_with_reply_sync(connection, message);
}
int libSandy_applyProfile(const char *profileName)
{
if (isRunningInsideSandyd()) return 0;
HBLogDebugWeak(@"[libSandy libSandy_applyProfile] attempting to apply profile %s", profileName);
xpc_object_t getExtensionsMessage = xpc_dictionary_create(NULL, NULL, 0);
xpc_dictionary_set_int64(getExtensionsMessage, "id", SANDYD_MESSAGE_GET_PROFILE_EXTENSIONS);
xpc_dictionary_set_string(getExtensionsMessage, "profile", profileName);
__block int returnCode = kLibSandyErrorXPCFailure;
xpc_object_t reply = sandydSendMessage(getExtensionsMessage);
if (reply) {
xpc_type_t replyType = xpc_get_type(reply);
HBLogDebugWeak(@"[libSandy libSandy_applyProfile] got reply %s", xpc_copy_description(reply));
if (replyType == XPC_TYPE_DICTIONARY) {
xpc_object_t extensions = xpc_dictionary_get_value(reply, "extensions");
xpc_type_t extensionsType = xpc_get_type(extensions);
if (extensionsType == XPC_TYPE_ARRAY) {
HBLogDebugWeak(@"[libSandy libSandy_applyProfile] got extensions %s", xpc_copy_description(extensions));
returnCode = kLibSandyErrorRestricted;
xpc_array_apply(extensions, ^bool(size_t index, xpc_object_t value) {
if (xpc_get_type(value) == XPC_TYPE_STRING) {
returnCode = kLibSandySuccess; // if returned extensions has one or more tokens: SUCCESS
const char *ext = xpc_string_get_string_ptr(value);
__unused int64_t suc = sandbox_extension_consume(ext);
HBLogDebugWeak(@"[libSandy libSandy_applyProfile] Consumed extension (%s) -> %lld", ext, suc);
}
return true;
});
}
}
}
HBLogDebugWeak(@"[libSandy libSandy_applyProfile] applied profile %s => %d", profileName, returnCode);
return returnCode;
}
bool libSandy_works()
{
if (isRunningInsideSandyd()) return YES;
xpc_object_t testMessage = xpc_dictionary_create(NULL, NULL, 0);
xpc_dictionary_set_int64(testMessage, "id", SANDYD_MESSAGE_TEST_CONNECTION);
bool returnCode = false;
xpc_object_t reply = sandydSendMessage(testMessage);
if (reply) {
xpc_type_t replyType = xpc_get_type(reply);
if (replyType == XPC_TYPE_DICTIONARY) {
returnCode = xpc_dictionary_get_bool(reply, "works");
}
}
return returnCode;
}