SOSTransportCoder.c [plain text]
#include <Security/SecureObjectSync/SOSPeer.h>
#include <Security/SecureObjectSync/SOSTransportCoder.h>
#include <Security/SecureObjectSync/SOSTransportMessage.h>
#include <Security/SecureObjectSync/SOSCoder.h>
#include <Security/SecureObjectSync/SOSAccount.h>
#include <Security/SecureObjectSync/SOSAccountPriv.h>
#include <Security/SecureObjectSync/SOSEngine.h>
#include <utilities/debugging.h>
#include <utilities/SecCFWrappers.h>
#include <AssertMacros.h>
#include <SOSInternal.h>
static SOSCoderRef SOSTransportMessageCopyPeerCoder(SOSTransportMessageRef transport, CFStringRef peer_id){
SOSCoderRef coder = NULL;
CFDataRef coderData = SOSEngineGetCoderData(SOSTransportMessageGetEngine(transport), peer_id);
if (coderData) {
CFErrorRef localError = NULL;
coder = SOSCoderCreateFromData(coderData, &localError);
if (!coder) {
secerror("Failed to make coder from valid data for peer %@ (%@). THIS IS FATAL: WE CAN'T COMMUNICATE.", peer_id, localError);
}
CFReleaseNull(localError);
}
else
secerror("Failed to get coderData from engine for peer %@. THIS IS FATAL: WE CAN'T COMMUNICATE.", peer_id);
return coder;
}
bool SOSTransportMessageSavePeerCoderData(SOSTransportMessageRef transport, SOSCoderRef coder, CFStringRef peer_id, CFErrorRef *error) {
CFDataRef coderData = NULL;
bool ok = true;
if (coder) {
coderData = SOSCoderCopyDER(coder, error);
if (coderData == NULL) {
secerror("%@ coder data failed to export %@, zapping data", transport, error ? *error : 0);
}
}
require_action_quiet(coderData, exit, ok = SOSErrorCreate(kSOSErrorAllocationFailure, error, NULL, CFSTR("Creation of coder data failed")));
ok = SOSEngineSetCoderData(SOSTransportMessageGetEngine(transport), peer_id, coderData, error);
exit:
CFReleaseNull(coderData);
return ok;
}
bool SOSTransportCoderInitializeForPeer(SOSTransportMessageRef transport, SOSFullPeerInfoRef myPeerInfo, SOSPeerInfoRef peerInfo, CFErrorRef *error){
SOSCoderRef coder = NULL;
CFStringRef peer_id = SOSPeerInfoGetPeerID(peerInfo);
CFDataRef coderData = SOSEngineGetCoderData(SOSTransportMessageGetEngine(transport), peer_id);
if(coderData != NULL) {
CFErrorRef coderError = NULL;
coder = SOSCoderCreateFromData(coderData, &coderError);
if (!coder) {
secerror("Found data but couldn't make coder for %@: %@", peer_id, coderError);
}
CFReleaseNull(coderError);
}
bool haveGoodCoder = coder;
if (!haveGoodCoder) {
secnotice("transport", "New coder for id %@.", peer_id);
if(SOSPeerInfoShouldUseIDSTransport(SOSFullPeerInfoGetPeerInfo(myPeerInfo), peerInfo))
coder = SOSCoderCreate(peerInfo, myPeerInfo, kCFBooleanTrue, error);
else
coder = SOSCoderCreate(peerInfo, myPeerInfo, kCFBooleanFalse, error);
if (coder) {
haveGoodCoder = SOSTransportMessageSavePeerCoderData(transport, coder, peer_id, error);
} else {
secerror("Couldn't make coder for %@", peer_id);
}
}
if (coder)
SOSCoderDispose(coder);
return haveGoodCoder;
}
enum SOSCoderUnwrapStatus SOSTransportMessageHandleCoderMessage(SOSTransportMessageRef transport, CFStringRef peer_id, CFDataRef codedMessage, CFDataRef *decodedMessage, CFErrorRef *error){
enum SOSCoderUnwrapStatus result = SOSCoderUnwrapError;
CFMutableDataRef localDecodedMessage = NULL;
SOSCoderStatus coderStatus = kSOSCoderDataReturned;
SOSCoderRef coder = SOSTransportMessageCopyPeerCoder(transport, peer_id);
if(!coder){
SOSAccountEnsurePeerRegistration(SOSTransportMessageGetAccount(transport), error);
coder = SOSTransportMessageCopyPeerCoder(transport, peer_id);
secnotice("transport", "Building new coder!");
}
CFErrorRef localError = NULL;
if (coder) {
coderStatus = SOSCoderUnwrap(coder, codedMessage, &localDecodedMessage, peer_id, error);
switch(coderStatus) {
case kSOSCoderDataReturned: {
logRawMessage(localDecodedMessage, false, 0);
result = SOSCoderUnwrapDecoded;
break;
}
case kSOSCoderNegotiating: result = SOSCoderUnwrapHandled;
secnotice("transport", "%@ transport negotiating", peer_id);
break;
case kSOSCoderNegotiationCompleted:
if (SOSEnginePeerDidConnect(SOSTransportMessageGetEngine(transport), peer_id, error))
result = SOSCoderUnwrapHandled;
secnotice("transport", "%@ transport negotiation complete", peer_id);
break;
case kSOSCoderFailure: secnotice("transport", "%@ transport failed handling message %@", peer_id, error ? *error : NULL);
SOSCoderReset(coder);
if (SOSCoderStart(coder, &localError) == kSOSCoderFailure) {
secerror("Attempt to recover coder failed to restart: %@", localError);
}
break;
case kSOSCoderStaleEvent: result = SOSCoderUnwrapHandled;
secinfo("transport", "%@ transport stale event ignored", peer_id);
break;
case kSOSCoderTooNew: secnotice("transport", "%@ transport received a message too soon, time to restart", peer_id);
SOSCoderReset(coder);
if(SOSCoderStart(coder, &localError) == kSOSCoderFailure){
secerror("Attempt to recover coder failed to restart: %@", localError);
}
default:
assert(false);
break;
}
if(decodedMessage)
*decodedMessage = CFRetainSafe(localDecodedMessage);
CFReleaseNull(localDecodedMessage);
SOSTransportMessageSavePeerCoderData(transport, coder, peer_id, NULL);
SOSCoderDispose(coder);
} else {
secerror("SOSTransportMessageHandleCoderMessage: Could not make a new coder!");
}
CFReleaseNull(localError);
return result;
}
bool SOSTransportMessageSendMessageIfNeeded(SOSTransportMessageRef transport, CFStringRef circle_id, CFStringRef peer_id, CFErrorRef *error) {
SOSCoderRef coder = SOSTransportMessageCopyPeerCoder(transport, peer_id);
if(!coder){
SOSAccountEnsurePeerRegistration(SOSTransportMessageGetAccount(transport), error);
coder = SOSTransportMessageCopyPeerCoder(transport, peer_id);
}
CFDataRef message_to_send = NULL;
bool ok = false;
SOSEnginePeerMessageSentBlock sent = NULL;
require_action_quiet(coder, fail, SOSCreateError(kSOSErrorAllocationFailure, CFSTR("SOSTransportMessageCopyPeerCoder failed"), *error, error));
if (SOSCoderCanWrap(coder)) {
secinfo("transport", "%@ Coder can wrap, getting message from engine", peer_id);
CFMutableDataRef codedMessage = NULL;
CFDataRef message = SOSEngineCreateMessageToSyncToPeer(SOSTransportMessageGetEngine(transport), peer_id, &sent, error);
if (!message) {
secnotice("transport", "%@ SOSEngineCreateMessageToSyncToPeer failed: %@",peer_id, *error);
}
ok = message && (SOSCoderWrap(coder, message, &codedMessage, peer_id, error) == kSOSCoderDataReturned);
if (!ok) {
secnotice("transport", "%@ SOSCoderWrap failed: %@",peer_id, *error);
}
if (ok)
CFRetainAssign(message_to_send, codedMessage);
CFReleaseNull(codedMessage);
CFReleaseNull(message);
} else {
message_to_send = SOSCoderCopyPendingResponse(coder);
secinfo("transport", "%@ negotiating, %@", peer_id, message_to_send ? CFSTR("sending negotiation message.") : CFSTR("waiting for negotiation message."));
sent = Block_copy(^(bool wasSent){
if (wasSent)
SOSCoderConsumeResponse(coder);
});
ok = true;
}
if (message_to_send) {
CFDictionaryRef peer_dict = CFDictionaryCreateForCFTypes(kCFAllocatorDefault,
peer_id, message_to_send,
NULL);
CFDictionaryRef circle_peers = CFDictionaryCreateForCFTypes(kCFAllocatorDefault,
circle_id, peer_dict,
NULL);
ok = ok && SOSTransportMessageSendMessages(transport, circle_peers, error);
if (sent)
sent(ok);
CFReleaseSafe(peer_dict);
CFReleaseSafe(circle_peers);
}
Block_release(sent);
CFReleaseSafe(message_to_send);
SOSTransportMessageSavePeerCoderData(transport, coder, peer_id, NULL);
fail:
if (coder)
SOSCoderDispose(coder);
return ok;
}