2 * Copyright (c) 2000, 2001, 2003, 2004, 2006-2010 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>
42 isMySessionKey(CFStringRef sessionKey
, CFStringRef key
)
45 CFStringRef storeSessionKey
;
47 dict
= CFDictionaryGetValue(storeData
, key
);
49 /* if key no longer exists */
53 storeSessionKey
= CFDictionaryGetValue(dict
, kSCDSession
);
54 if (!storeSessionKey
) {
55 /* if this is not a session key */
59 if (!CFEqual(sessionKey
, storeSessionKey
)) {
60 /* if this is not "my" session key */
69 removeAllKeys(SCDynamicStoreRef store
, Boolean isRegex
)
71 SCDynamicStorePrivateRef storePrivate
= (SCDynamicStorePrivateRef
)store
;
75 keys
= isRegex
? storePrivate
->patterns
: storePrivate
->keys
;
76 n
= (keys
!= NULL
) ? CFArrayGetCount(keys
) : 0;
79 CFArrayRef keysToRemove
;
81 keysToRemove
= CFArrayCreateCopy(NULL
, keys
);
82 for (i
= 0; i
< n
; i
++) {
83 (void) __SCDynamicStoreRemoveWatchedKey(store
,
84 CFArrayGetValueAtIndex(keysToRemove
, i
),
88 CFRelease(keysToRemove
);
97 __SCDynamicStoreClose(SCDynamicStoreRef
*store
, Boolean internal
)
102 serverSessionRef mySession
;
103 CFStringRef sessionKey
;
104 SCDynamicStorePrivateRef storePrivate
= (SCDynamicStorePrivateRef
)*store
;
106 if ((*store
== NULL
) || (storePrivate
->server
== MACH_PORT_NULL
)) {
107 return kSCStatusNoStoreSession
; /* you must have an open session to play */
110 if (_configd_trace
) {
111 SCTrace(TRUE
, _configd_trace
,
113 internal
? "*close " : "close ",
114 storePrivate
->server
);
117 /* Remove all notification keys and patterns */
118 removeAllKeys(*store
, FALSE
); // keys
119 removeAllKeys(*store
, TRUE
); // patterns
121 /* Remove/cancel any outstanding notification requests. */
122 __MACH_PORT_DEBUG(storePrivate
->notifyPort
!= MACH_PORT_NULL
, "*** __SCDynamicStoreClose", storePrivate
->notifyPort
);
123 (void) __SCDynamicStoreNotifyCancel(*store
);
125 /* Remove any session keys */
126 sessionKey
= CFStringCreateWithFormat(NULL
, NULL
, CFSTR("%d"), storePrivate
->server
);
127 dict
= CFDictionaryGetValue(sessionData
, sessionKey
);
128 keys
= CFDictionaryGetValue(dict
, kSCDSessionKeys
);
129 if (keys
&& ((keyCnt
= CFArrayGetCount(keys
)) > 0)) {
134 * if necessary, claim a lock to ensure that we inform
135 * any processes that a session key was removed.
137 wasLocked
= (storeLocked
> 0);
139 (void) __SCDynamicStoreLock(*store
, FALSE
);
142 /* remove keys from "locked" store" */
143 for (i
= 0; i
< keyCnt
; i
++) {
144 if (isMySessionKey(sessionKey
, CFArrayGetValueAtIndex(keys
, i
))) {
145 (void) __SCDynamicStoreRemoveValue(*store
, CFArrayGetValueAtIndex(keys
, i
), TRUE
);
150 /* remove keys from "unlocked" store" */
151 _swapLockedStoreData();
152 for (i
= 0; i
< keyCnt
; i
++) {
153 if (isMySessionKey(sessionKey
, CFArrayGetValueAtIndex(keys
, i
)))
154 (void) __SCDynamicStoreRemoveValue(*store
, CFArrayGetValueAtIndex(keys
, i
), TRUE
);
156 _swapLockedStoreData();
160 * Note: everyone who calls __SCDynamicStoreClose() ends
161 * up removing this sessions dictionary. As such,
162 * we don't need to worry about the session keys.
165 CFRelease(sessionKey
);
167 /* release the lock */
168 if (storePrivate
->locked
) {
169 (void) __SCDynamicStoreUnlock(*store
, FALSE
);
173 * invalidate and release our run loop source on the server
174 * port (for this client). Then, release the port.
176 mySession
= getSession(storePrivate
->server
);
177 if (mySession
->serverRunLoopSource
) {
178 CFRunLoopSourceInvalidate(mySession
->serverRunLoopSource
);
179 CFRelease(mySession
->serverRunLoopSource
);
180 mySession
->serverRunLoopSource
= NULL
;
182 if (mySession
->serverPort
!= NULL
) {
183 CFMachPortInvalidate(mySession
->serverPort
);
184 CFRelease(mySession
->serverPort
);
185 mySession
->serverPort
= NULL
;
188 storePrivate
->server
= MACH_PORT_NULL
;
198 _configclose(mach_port_t server
, int *sc_status
)
200 serverSessionRef mySession
= getSession(server
);
202 if (mySession
== NULL
) {
203 *sc_status
= kSCStatusNoStoreSession
; /* you must have an open session to play */
210 __MACH_PORT_DEBUG(TRUE
, "*** _configclose", server
);
211 *sc_status
= __SCDynamicStoreClose(&mySession
->store
, FALSE
);
212 if (*sc_status
!= kSCStatusOK
) {
214 CFSTR("_configclose __SCDynamicStoreClose() failed, status = %s"),
215 SCErrorString(*sc_status
));
218 __MACH_PORT_DEBUG(TRUE
, "*** _configclose (after __SCDynamicStoreClose)", server
);
221 * Remove our receive right.
223 * Note: there is no need to cancel the notification request because the
224 * kernel will have no way to deliver the notification once the
225 * receive right has been removed.
227 (void) mach_port_mod_refs(mach_task_self(), server
, MACH_PORT_RIGHT_RECEIVE
, -1);
230 * Remove the session entry.
232 removeSession(server
);