2 * Copyright (c) 2000-2009, 2011, 2015, 2016 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 * Modification History
27 * June 1, 2001 Allan Nathanson <ajn@apple.com>
28 * - public API conversion
30 * March 24, 2000 Allan Nathanson <ajn@apple.com>
35 #include "configd_server.h"
38 #include <bsm/libbsm.h>
39 #include <sys/types.h>
44 __SCDynamicStoreOpen(SCDynamicStoreRef
*store
, CFStringRef name
)
47 * allocate and initialize a new session
49 *store
= (SCDynamicStoreRef
)__SCDynamicStoreCreatePrivate(NULL
, name
, NULL
, NULL
);
52 * If necessary, initialize the store and session data dictionaries
54 if (storeData
== NULL
) {
55 sessionData
= CFDictionaryCreateMutable(NULL
,
57 &kCFTypeDictionaryKeyCallBacks
,
58 &kCFTypeDictionaryValueCallBacks
);
59 storeData
= CFDictionaryCreateMutable(NULL
,
61 &kCFTypeDictionaryKeyCallBacks
,
62 &kCFTypeDictionaryValueCallBacks
);
63 patternData
= CFDictionaryCreateMutable(NULL
,
65 &kCFTypeDictionaryKeyCallBacks
,
66 &kCFTypeDictionaryValueCallBacks
);
67 changedKeys
= CFSetCreateMutable(NULL
,
69 &kCFTypeSetCallBacks
);
70 deferredRemovals
= CFSetCreateMutable(NULL
,
72 &kCFTypeSetCallBacks
);
73 removedSessionKeys
= CFSetCreateMutable(NULL
,
75 &kCFTypeSetCallBacks
);
83 openMPCopyDescription(const void *info
)
85 return CFStringCreateWithFormat(NULL
, NULL
, CFSTR("<SCDynamicStore MP>"));
91 _configopen(mach_port_t server
,
92 xmlData_t nameRef
, /* raw XML bytes */
93 mach_msg_type_number_t nameLen
,
94 xmlData_t optionsRef
, /* raw XML bytes */
95 mach_msg_type_number_t optionsLen
,
96 mach_port_t
*newServer
,
98 audit_token_t audit_token
)
100 CFDictionaryRef info
;
101 serverSessionRef mySession
;
102 CFStringRef name
= NULL
; /* name (un-serialized) */
103 CFMutableDictionaryRef newInfo
;
104 mach_port_t oldNotify
;
105 CFDictionaryRef options
= NULL
; /* options (un-serialized) */
106 CFStringRef sessionKey
;
107 kern_return_t status
;
108 SCDynamicStorePrivateRef storePrivate
;
109 CFBooleanRef useSessionKeys
= NULL
;
111 *sc_status
= kSCStatusOK
;
113 /* un-serialize the name */
114 if (!_SCUnserializeString(&name
, NULL
, (void *)nameRef
, nameLen
)) {
115 *sc_status
= kSCStatusFailed
;
118 if ((optionsRef
!= NULL
) && (optionsLen
> 0)) {
119 /* un-serialize the [session] options */
120 if (!_SCUnserialize((CFPropertyListRef
*)&options
, NULL
, (void *)optionsRef
, optionsLen
)) {
121 *sc_status
= kSCStatusFailed
;
125 if (*sc_status
!= kSCStatusOK
) {
129 if (!isA_CFString(name
)) {
130 *sc_status
= kSCStatusInvalidArgument
;
134 if (options
!= NULL
) {
135 if (!isA_CFDictionary(options
)) {
136 *sc_status
= kSCStatusInvalidArgument
;
141 * [pre-]process any provided options
143 useSessionKeys
= CFDictionaryGetValue(options
, kSCDynamicStoreUseSessionKeys
);
144 if (useSessionKeys
!= NULL
) {
145 if (!isA_CFBoolean(useSessionKeys
)) {
146 *sc_status
= kSCStatusInvalidArgument
;
153 * establish the new session
155 mySession
= addSession(server
, openMPCopyDescription
);
156 if (mySession
== NULL
) {
157 SC_log(LOG_NOTICE
, "session is already open");
158 *sc_status
= kSCStatusFailed
; /* you can't re-open an "open" session */
162 *newServer
= mySession
->key
;
163 __MACH_PORT_DEBUG(TRUE
, "*** _configopen (after addSession)", *newServer
);
165 /* save the audit_token in case we need to check the callers credentials */
166 mySession
->auditToken
= audit_token
;
168 /* Create and add a run loop source for the port */
169 mySession
->serverRunLoopSource
= CFMachPortCreateRunLoopSource(NULL
, mySession
->serverPort
, 0);
170 CFRunLoopAddSource(CFRunLoopGetCurrent(),
171 mySession
->serverRunLoopSource
,
172 kCFRunLoopDefaultMode
);
174 SC_trace("open : %5d : %@",
178 *sc_status
= __SCDynamicStoreOpen(&mySession
->store
, name
);
179 storePrivate
= (SCDynamicStorePrivateRef
)mySession
->store
;
182 * Make the server port accessible to the framework routines.
183 * ... and be sure to clear before calling CFRelease(store)
185 storePrivate
->server
= *newServer
;
188 * Process any provided [session] options
190 if (useSessionKeys
!= NULL
) {
191 storePrivate
->useSessionKeys
= CFBooleanGetValue(useSessionKeys
);
194 /* Request a notification when/if the client dies */
195 status
= mach_port_request_notification(mach_task_self(),
197 MACH_NOTIFY_NO_SENDERS
,
200 MACH_MSG_TYPE_MAKE_SEND_ONCE
,
202 if (status
!= KERN_SUCCESS
) {
203 SC_log(LOG_NOTICE
, "mach_port_request_notification() failed: %s", mach_error_string(status
));
204 cleanupSession(*newServer
);
205 *newServer
= MACH_PORT_NULL
;
206 *sc_status
= kSCStatusFailed
;
209 __MACH_PORT_DEBUG(TRUE
, "*** _configopen (after mach_port_request_notification)", *newServer
);
211 if (oldNotify
!= MACH_PORT_NULL
) {
212 SC_log(LOG_NOTICE
, "oldNotify != MACH_PORT_NULL");
216 * Save the name of the calling application / plug-in with the session data.
218 sessionKey
= CFStringCreateWithFormat(NULL
, NULL
, CFSTR("%d"), *newServer
);
219 info
= CFDictionaryGetValue(sessionData
, sessionKey
);
221 newInfo
= CFDictionaryCreateMutableCopy(NULL
, 0, info
);
223 newInfo
= CFDictionaryCreateMutable(NULL
,
225 &kCFTypeDictionaryKeyCallBacks
,
226 &kCFTypeDictionaryValueCallBacks
);
228 CFDictionarySetValue(newInfo
, kSCDName
, name
);
229 CFDictionarySetValue(sessionData
, sessionKey
, newInfo
);
231 CFRelease(sessionKey
);
234 * Note: at this time we should be holding ONE send right and
235 * ONE receive right to the server. The send right is
236 * moved to the caller.
241 if (name
!= NULL
) CFRelease(name
);
242 if (options
!= NULL
) CFRelease(options
);