2 * Copyright (c) 2006-2007 Apple Inc. All Rights Reserved.
4 * @APPLE_LICENSE_HEADER_START@
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. Please obtain a copy of the License at
10 * http://www.opensource.apple.com/apsl/ and read it before using this
13 * The Original Code and all software distributed under the License are
14 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
15 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
16 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
18 * Please see the License for the specific language governing rights and
19 * limitations under the License.
21 * @APPLE_LICENSE_HEADER_END@
25 // csgeneric - generic Code representative
27 #include "csgeneric.h"
29 #include "StaticCode.h"
30 #include <securityd_client/cshosting.h>
31 #include <sys/param.h>
34 namespace CodeSigning
{
36 using MachPlusPlus::Port
;
40 // Common call-out code for cshosting RPC service
42 #define CALL(host, name, args...) \
44 if (cshosting_client_ ## name (host, mig_get_reply_port(), &result, args)) \
45 MacOSError::throwMe(errSecCSNotAHost); \
46 MacOSError::check(result);
50 // Construct a running process representation
52 GenericCode::GenericCode(SecCode
*host
, SecGuestRef guestRef
)
53 : SecCode(host
), mGuestRef(guestRef
)
59 // Identify a guest by attribute set, and return a new GenericCode representing it.
60 // This uses cshosting RPCs to ask the host (or its proxy).
62 SecCode
*GenericCode::locateGuest(CFDictionaryRef attributes
)
64 if (Port host
= hostingPort()) {
65 CFRef
<CFDataRef
> attrData
;
66 void *attrPtr
= NULL
; size_t attrLength
= 0;
68 attrData
.take(CFPropertyListCreateXMLData(NULL
, attributes
));
69 attrPtr
= (void *)CFDataGetBytePtr(attrData
);
70 attrLength
= CFDataGetLength(attrData
);
73 mach_msg_type_number_t guestPathLength
;
75 CALL(host
, findGuest
, guestRef(), attrPtr
, attrLength
,
76 &guestPath
, &guestPathLength
, &subport
);
77 secdebug("genericcode", "%p found guest chain length=%d",
78 this, guestPathLength
);
79 SecPointer
<SecCode
> code
= this;
80 for (unsigned n
= 0; n
< guestPathLength
; n
++)
81 code
= new GenericCode(code
, guestPath
[n
]);
84 return NULL
; // not found, no error
89 // Map a guest to its StaticCode.
90 // This uses cshosting RPCs to ask the host (or its proxy).
92 SecStaticCode
*GenericCode::mapGuestToStatic(SecCode
*guest
)
94 if (Port host
= hostingPort()) {
95 char path
[MAXPATHLEN
];
96 CALL(host
, guestPath
, safe_cast
<GenericCode
*>(guest
)->guestRef(), path
);
97 return (new GenericStaticCode(DiskRep::bestGuess(path
)))->retain();
99 MacOSError::throwMe(errSecCSNotAHost
);
104 // Get the Code Signing Status Word for a Code.
105 // This uses cshosting RPCs to ask the host (or its proxy).
107 uint32_t GenericCode::getGuestStatus(SecCode
*guest
)
109 if (Port host
= hostingPort()) {
111 CALL(host
, guestStatus
, safe_cast
<GenericCode
*>(guest
)->guestRef(), &status
);
114 MacOSError::throwMe(errSecCSNotAHost
);
119 // Return the Hosting Port for this Code.
120 // May return MACH_PORT_NULL if the code is not a code host.
121 // Throws if we can't get the hosting port for some reason (and can't positively
122 // determine that there is none).
124 // Note that we do NOT cache negative outcomes. Being a host is a dynamic property,
125 // and this Code may not have commenced hosting operations yet. For non- (or not-yet-)hosts
126 // we simply return NULL.
128 Port
GenericCode::hostingPort()
131 if (staticCode()->codeDirectory()->flags
& kSecCodeSignatureHost
)
132 mHostingPort
= getHostingPort();
139 // A pure GenericHost has no idea where to get a hosting port from.
140 // This method must be overridden to get one.
141 // However, we do handle a contiguous chain of GenericCodes by deferring
142 // to our next-higher host for it.
144 mach_port_t
GenericCode::getHostingPort()
146 if (GenericCode
*genericHost
= dynamic_cast<GenericCode
*>(host()))
147 return genericHost
->getHostingPort();
149 MacOSError::throwMe(errSecCSNotAHost
);