DeviceSimulatorMain.m [plain text]
//
// main.m
// DeviceSimulator
//
//
#import <Foundation/Foundation.h>
#import <Foundation/NSXPCConnection_Private.h>
#import <SOSCircle/CKBridge/SOSCloudKeychainConstants.h>
#import <objc/runtime.h>
#import <utilities/debugging.h>
#import <securityd/SOSCloudCircleServer.h>
#import <Security/SecureObjectSync/SOSPeerInfo.h>
#import <Security/SecureObjectSync/SOSCloudCircleInternal.h>
#import <Security/SecureObjectSync/SOSViews.h>
#import <Security/SecureObjectSync/SOSInternal.h>
#import "DeviceSimulator.h"
#import "SOSCloudKeychainClient.h"
#import "MultiDeviceNetworkingProtocol.h"
#import "SecCFWrappers.h"
#import "spi.h"
struct SOSCloudTransport MDNTransport;
@class MDNetwork;
NSString *deviceInstance = NULL;
static NSString *deviceHomeDir = NULL;
static MDNetwork *deviceNetwork = NULL;
@interface MDNetwork : NSObject<MultiDeviceNetworkingProtocol,MultiDeviceNetworkingCallbackProtocol,NSXPCListenerDelegate>
@property NSXPCConnection *connection;
@property NSXPCListener *callbackListener;
@property dispatch_queue_t flushQueue;
@property NSMutableDictionary *pendingKeys;
@property NSSet *registeredKeys;
- (instancetype)initWithConnection:(NSXPCConnection *)connection;
@end
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wprotocol"
@implementation MDNetwork
- (instancetype)initWithConnection:(NSXPCConnection *)connection
{
self = [super init];
if (self) {
self.connection = connection;
self.callbackListener = [NSXPCListener anonymousListener];
self.callbackListener.delegate = self;
[self.callbackListener resume];
__typeof(self) weakSelf = self;
self.connection.invalidationHandler = ^{
__typeof(self) strongSelf = weakSelf;
[strongSelf.callbackListener invalidate];
strongSelf.callbackListener = nil;
exit(0);
};
self.flushQueue = dispatch_queue_create("MDNetwork.flushqueue", 0);
self.pendingKeys = [NSMutableDictionary dictionary];
self.registeredKeys = [NSSet set];
[[self.connection synchronousRemoteObjectProxyWithErrorHandler:^(NSError * _Nonnull error) {
NSLog(@"network register callback failed with: //abort();
}] MDNRegisterCallback:[self.callbackListener endpoint] complete:^void(NSDictionary *values, NSError *error) {
;
}];
}
return self;
}
- (BOOL)listener:(NSXPCListener *)listener shouldAcceptNewConnection:(NSXPCConnection *)newConnection
{
newConnection.exportedInterface = [NSXPCInterface interfaceWithProtocol:@protocol(MultiDeviceNetworkingCallbackProtocol)];
newConnection.exportedObject = self;
[newConnection resume];
return YES;
}
- (void)MDNCItemsChanged:(NSDictionary *)values complete:(MDNComplete)complete
{
NSMutableDictionary *requestedKeys = [NSMutableDictionary dictionary];
@synchronized(self.pendingKeys) {
secnotice("MDN", "items update:
[self.pendingKeys addEntriesFromDictionary:values];
for (NSString *key in self.registeredKeys) {
id data = self.pendingKeys[key];
if (data) {
requestedKeys[key] = data;
self.pendingKeys[key] = nil;
}
}
}
if (requestedKeys.count) {
dispatch_async(self.flushQueue, ^{
secnotice("MDN", "engine processing keys: NSArray *handled = CFBridgingRelease(SOSCCHandleUpdateMessage((__bridge CFDictionaryRef)requestedKeys));
/*
* Ok, our dear Engine might not have handled all messages.
* So put them back unless there are new messages around that
* have overwritten the previous message.
*/
for (NSString *key in handled) {
requestedKeys[key] = NULL;
}
if (requestedKeys.count) {
@synchronized(self.pendingKeys) {
for (NSString *key in requestedKeys) {
if (self.pendingKeys[key] == nil) {
self.pendingKeys[key] = requestedKeys[key];
}
}
}
}
});
}
complete(NULL, NULL);
}
/* Oh, ObjC, you are my friend */
- (void)forwardInvocation:(NSInvocation *)invocation
{
struct objc_method_description desc = protocol_getMethodDescription(@protocol(MultiDeviceNetworkingProtocol), [invocation selector], true, true);
if (desc.name == NULL) {
[super forwardInvocation:invocation];
} else {
__block bool gogogo = true;
id object = [self.connection remoteObjectProxyWithErrorHandler:^(NSError * _Nonnull error) {
gogogo = false;
NSLog(@"network failed with: //abort();
}];
if(gogogo) [invocation invokeWithTarget:object];
}
}
@end
#pragma clang diagnostic pop
#define HANDLE_NO_NETWORK(_replyBlock) \
if (deviceNetwork == NULL) { \
replyBlock((__bridge CFDictionaryRef)@{}, (__bridge CFErrorRef)[NSError errorWithDomain:@"MDNNetwork" code:1 userInfo:NULL]); \
return; \
}
static void
DSCloudPut(SOSCloudTransportRef transport, CFDictionaryRef valuesToPut, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
@autoreleasepool {
HANDLE_NO_NETWORK(replyBlock);
secnotice("MDN", "CloudPut:
[deviceNetwork MDNCloudPut:(__bridge NSDictionary *)valuesToPut complete:^(NSDictionary *returnedValues, NSError *error) {
dispatch_async(processQueue, ^{
replyBlock((__bridge CFDictionaryRef)returnedValues, (__bridge CFErrorRef)error);
});
}];
}
}
static NSString *nKeyAlwaysKeys = @"AlwaysKeys";
static NSString *nKeyFirstUnlockKeys = @"FirstUnlockKeys";
static NSString *nKeyUnlockedKeys = @"UnlockedKeys";
static NSString *nMessageKeyParameter = @"KeyParameter";
static NSString *nMessageCircle = @"Circle";
static NSString *nMessageMessage = @"Message";
static void
DSCloudUpdateKeys(SOSCloudTransportRef transport, CFDictionaryRef cfkeys, CFStringRef accountUUID, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
/*
* Currently doesn't deal with lock state, just smash (HULK!) them all together
*/
@autoreleasepool {
NSDictionary *keys = (__bridge NSDictionary *)cfkeys;
NSMutableSet *newSet = [NSMutableSet set];
@synchronized(deviceNetwork.pendingKeys) {
for (NSString *type in @[ nMessageKeyParameter, nMessageCircle, nMessageMessage]) {
NSDictionary *typeDict = keys[type];
for (NSString *lockType in @[ nKeyAlwaysKeys, nKeyFirstUnlockKeys, nKeyUnlockedKeys]) {
NSArray *lockArray = typeDict[lockType];
if (lockArray) {
[newSet unionSet:[NSMutableSet setWithArray:lockArray]];
}
}
}
deviceNetwork.registeredKeys = newSet;
}
/* update engine with stuff */
[deviceNetwork MDNCItemsChanged:@{} complete:^(NSDictionary *returnedValues, NSError *error) {
if (replyBlock)
replyBlock((__bridge CFDictionaryRef)returnedValues, (__bridge CFErrorRef)error);
}];
}
}
static void
DSCloudGetDeviceID(SOSCloudTransportRef transport, CloudKeychainReplyBlock replyBlock)
{
if (replyBlock)
replyBlock((__bridge CFDictionaryRef)@{}, NULL);
}
// Debug calls
static void
DSCloudGet(SOSCloudTransportRef transport, CFArrayRef keysToGet, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
if (replyBlock)
replyBlock((__bridge CFDictionaryRef)@{}, NULL);
}
static void
DSCloudGetAll(SOSCloudTransportRef transport, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
if (replyBlock)
replyBlock((__bridge CFDictionaryRef)@{}, NULL);
}
static void
DSCloudsynchronize(SOSCloudTransportRef transport, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
if (replyBlock)
replyBlock((__bridge CFDictionaryRef)@{}, NULL);
}
static void
DSCloudsynchronizeAndWait(SOSCloudTransportRef transport, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
@autoreleasepool {
HANDLE_NO_NETWORK(replyBlock);
[deviceNetwork MDNCloudsynchronizeAndWait:@{} complete:^(NSDictionary *returnedValues, NSError *error) {
dispatch_async(processQueue, ^{
replyBlock((__bridge CFDictionaryRef)returnedValues, (__bridge CFErrorRef)error);
});
}];
}
}
static void
DSCloudRemoveObjectForKey(SOSCloudTransportRef transport, CFStringRef keyToRemove, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
if (keyToRemove == NULL) {
dispatch_async(processQueue, ^{
replyBlock(NULL, NULL);
});
return;
}
@autoreleasepool {
HANDLE_NO_NETWORK(replyBlock);
[deviceNetwork MDNCloudRemoveKeys:@[(__bridge NSString *)keyToRemove] complete:^(NSDictionary *returnedValues, NSError *error) {
dispatch_async(processQueue, ^{
replyBlock((__bridge CFDictionaryRef)returnedValues, (__bridge CFErrorRef)error);
});
}];
}
}
static void DSCloudremoveKeys(SOSCloudTransportRef transport, CFArrayRef keys, CFStringRef accountUUID, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
@autoreleasepool {
HANDLE_NO_NETWORK(replyBlock);
[deviceNetwork MDNCloudRemoveKeys:(__bridge NSArray *)keys complete:^(NSDictionary *returnedValues, NSError *error) {
dispatch_async(processQueue, ^{
replyBlock((__bridge CFDictionaryRef)returnedValues, (__bridge CFErrorRef)error);
});
}];
}
}
static void
DSCloudclearAll(SOSCloudTransportRef transport, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
@autoreleasepool {
HANDLE_NO_NETWORK(replyBlock);
[deviceNetwork MDNCloudRemoveKeys:NULL complete:^(NSDictionary *returnedValues, NSError *error) {
dispatch_async(processQueue, ^{
replyBlock((__bridge CFDictionaryRef)returnedValues, (__bridge CFErrorRef)error);
});
}];
}
}
static bool
DSCloudhasPendingKey(SOSCloudTransportRef transport, CFStringRef keyName, CFErrorRef* error)
{
bool status = false;
@synchronized(deviceNetwork.pendingKeys) {
status = deviceNetwork.pendingKeys[(__bridge NSString *)keyName] != nil;
}
return status;
}
static void
DSCloudrequestSyncWithPeers(SOSCloudTransportRef transport, CFArrayRef /* CFStringRef */ peers, CFArrayRef /* CFStringRef */ backupPeers, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
CFSetRef sPeers = CFSetCreateCopyOfArrayForCFTypes(peers);
CFSetRef sBackupPeers = CFSetCreateCopyOfArrayForCFTypes(backupPeers);
if (sPeers == NULL || sBackupPeers == NULL) {
CFReleaseNull(sPeers);
CFReleaseNull(sBackupPeers);
} else {
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
CFErrorRef error = NULL;
CFSetRef result = SOSCCProcessSyncWithPeers_Server(sPeers, sBackupPeers, &error);
CFRelease(sPeers);
CFRelease(sBackupPeers);
CFReleaseNull(result);
CFReleaseNull(error);
});
}
if (replyBlock) {
dispatch_async(processQueue, ^{
replyBlock((__bridge CFDictionaryRef)@{}, NULL);
});
}
}
static bool
DSCloudhasPeerSyncPending(SOSCloudTransportRef transport, CFStringRef peerID, CFErrorRef* error)
{
return false;
}
static void
DSCloudrequestEnsurePeerRegistration(SOSCloudTransportRef transport, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
CFErrorRef eprError = NULL;
if (!SOSCCProcessEnsurePeerRegistration_Server(&eprError)) {
secnotice("coder", "SOSCCProcessEnsurePeerRegistration failed with: }
CFReleaseNull(eprError);
if (replyBlock)
replyBlock((__bridge CFDictionaryRef)@{}, NULL);
});
}
static void DSCloudrequestPerfCounters(SOSCloudTransportRef transport, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
if (replyBlock)
replyBlock((__bridge CFDictionaryRef)@{}, NULL);
}
static void DSCloudflush(SOSCloudTransportRef transport, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
@autoreleasepool {
HANDLE_NO_NETWORK(replyBlock);
[deviceNetwork MDNCloudFlush:^(NSDictionary *returnedValues, NSError *error) {
dispatch_async(deviceNetwork.flushQueue, ^{
dispatch_async(processQueue, ^{
replyBlock((__bridge CFDictionaryRef)returnedValues, (__bridge CFErrorRef)error);
});
});
}];
}
}
static void DSCloudcounters(SOSCloudTransportRef transport, dispatch_queue_t processQueue, CloudKeychainReplyBlock replyBlock)
{
if (replyBlock)
replyBlock((__bridge CFDictionaryRef)@{}, NULL);
}
@interface ServiceDelegate : NSObject <NSXPCListenerDelegate>
@end
@implementation ServiceDelegate
- (BOOL)listener:(NSXPCListener *)listener shouldAcceptNewConnection:(NSXPCConnection *)newConnection {
newConnection.exportedInterface = [NSXPCInterface interfaceWithProtocol:@protocol(DeviceSimulatorProtocol)];
DeviceSimulator *exportedObject = [DeviceSimulator new];
exportedObject.conn = newConnection;
newConnection.exportedObject = exportedObject;
[newConnection resume];
return YES;
}
@end
void
boot_securityd(NSXPCListenerEndpoint *network)
{
secLogDisable();
securityd_init((__bridge CFURLRef)[NSURL URLWithString:deviceHomeDir]);
NSXPCConnection *connection = [[NSXPCConnection alloc] initWithListenerEndpoint:network];
connection.remoteObjectInterface = [NSXPCInterface interfaceWithProtocol:@protocol(MultiDeviceNetworkingProtocol)];
[connection resume];
deviceNetwork = [[MDNetwork alloc] initWithConnection:connection];
}
/*
* Make sure each of th peers don't trample on each's others state
*/
@interface SOSCachedNotification (override)
@end
@implementation SOSCachedNotification (override)
+ (NSString *)swizzled_notificationName:(const char *)notificationName
{
return [NSString stringWithFormat:@" [SOSCachedNotification swizzled_notificationName:notificationName], deviceInstance];
}
+ (void)load {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
Method orignal = class_getClassMethod(self, @selector(notificationName:));
Method swizzled = class_getClassMethod(self, @selector(swizzled_notificationName:));
method_exchangeImplementations(orignal, swizzled);
});
}
@end
int main(int argc, const char *argv[])
{
struct sigaction action;
memset(&action, 0, sizeof(action));
deviceInstance = [[NSXPCListener _UUID] UUIDString];
NSURL *tempPath = [[NSFileManager defaultManager] temporaryDirectory];
deviceHomeDir = [[tempPath path] stringByAppendingPathComponent:deviceInstance];
[[NSFileManager defaultManager] createDirectoryAtPath:deviceHomeDir
withIntermediateDirectories:NO
attributes:NULL
error:NULL];
MDNTransport.put = DSCloudPut;
MDNTransport.updateKeys = DSCloudUpdateKeys;
MDNTransport.getDeviceID = DSCloudGetDeviceID;
MDNTransport.get = DSCloudGet;
MDNTransport.getAll = DSCloudGetAll;
MDNTransport.synchronize = DSCloudsynchronize;
MDNTransport.synchronizeAndWait = DSCloudsynchronizeAndWait;
MDNTransport.clearAll = DSCloudclearAll;
MDNTransport.removeObjectForKey = DSCloudRemoveObjectForKey;
MDNTransport.hasPendingKey = DSCloudhasPendingKey;
MDNTransport.requestSyncWithPeers = DSCloudrequestSyncWithPeers;
MDNTransport.hasPeerSyncPending = DSCloudhasPeerSyncPending;
MDNTransport.requestEnsurePeerRegistration = DSCloudrequestEnsurePeerRegistration;
MDNTransport.requestPerfCounters = DSCloudrequestPerfCounters;
MDNTransport.flush = DSCloudflush;
MDNTransport.itemsChangedBlock = CFBridgingRetain(^CFArrayRef(CFDictionaryRef values) {
// default change block doesn't handle messages, keep em
return CFBridgingRetain(@[]);
});
MDNTransport.removeKeys = DSCloudremoveKeys;
MDNTransport.counters = DSCloudcounters;
SOSCloudTransportSetDefaultTransport(&MDNTransport);
// Create the delegate for the service.
ServiceDelegate *delegate = [ServiceDelegate new];
signal(SIGPIPE, SIG_IGN);
// Set up the one NSXPCListener for this service. It will handle all incoming connections.
NSXPCListener *listener = [NSXPCListener serviceListener];
listener.delegate = delegate;
// Resuming the serviceListener starts this service. This method does not return.
[listener resume];
return 0;
}