]> git.saurik.com Git - apple/configd.git/blob - configd.tproj/_configclose.c
configd-130.tar.gz
[apple/configd.git] / configd.tproj / _configclose.c
1 /*
2 * Copyright (c) 2000-2003 Apple Computer, Inc. All rights reserved.
3 *
4 * @APPLE_LICENSE_HEADER_START@
5 *
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
11 * file.
12 *
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.
20 *
21 * @APPLE_LICENSE_HEADER_END@
22 */
23
24 /*
25 * Modification History
26 *
27 * June 1, 2001 Allan Nathanson <ajn@apple.com>
28 * - public API conversion
29 *
30 * March 24, 2000 Allan Nathanson <ajn@apple.com>
31 * - initial revision
32 */
33
34 #include <unistd.h>
35
36 #include "configd.h"
37 #include "session.h"
38
39 #define N_QUICK 16
40
41 static Boolean
42 isMySessionKey(CFStringRef sessionKey, CFStringRef key)
43 {
44 CFDictionaryRef dict;
45 CFStringRef storeSessionKey;
46
47 dict = CFDictionaryGetValue(storeData, key);
48 if (!dict) {
49 /* if key no longer exists */
50 return FALSE;
51 }
52
53 storeSessionKey = CFDictionaryGetValue(dict, kSCDSession);
54 if (!storeSessionKey) {
55 /* if this is not a session key */
56 return FALSE;
57 }
58
59 if (!CFEqual(sessionKey, storeSessionKey)) {
60 /* if this is not "my" session key */
61 return FALSE;
62 }
63
64 return TRUE;
65 }
66
67
68 static void
69 removeAllKeys(SCDynamicStoreRef store, Boolean isRegex)
70 {
71 SCDynamicStorePrivateRef storePrivate = (SCDynamicStorePrivateRef)store;
72 CFSetRef keys;
73 CFIndex n;
74
75 keys = isRegex ? storePrivate->patterns : storePrivate->keys;
76 n = CFSetGetCount(keys);
77 if (n > 0) {
78 CFIndex i;
79 CFArrayRef keysToRemove;
80 const void * watchedKeys_q[N_QUICK];
81 const void ** watchedKeys = watchedKeys_q;
82
83 if (n > (CFIndex)(sizeof(watchedKeys_q) / sizeof(CFStringRef)))
84 watchedKeys = CFAllocatorAllocate(NULL, n * sizeof(CFStringRef), 0);
85 CFSetGetValues(keys, watchedKeys);
86 keysToRemove = CFArrayCreate(NULL, watchedKeys, n, &kCFTypeArrayCallBacks);
87 if (watchedKeys != watchedKeys_q) CFAllocatorDeallocate(NULL, watchedKeys);
88 for (i = 0; i < n; i++) {
89 (void) __SCDynamicStoreRemoveWatchedKey(store,
90 CFArrayGetValueAtIndex(keysToRemove, i),
91 isRegex,
92 TRUE);
93 }
94 CFRelease(keysToRemove);
95 }
96
97 return;
98 }
99
100
101 __private_extern__
102 int
103 __SCDynamicStoreClose(SCDynamicStoreRef *store, Boolean internal)
104 {
105 CFDictionaryRef dict;
106 CFArrayRef keys;
107 CFIndex keyCnt;
108 serverSessionRef mySession;
109 CFStringRef sessionKey;
110 SCDynamicStorePrivateRef storePrivate = (SCDynamicStorePrivateRef)*store;
111
112 if ((*store == NULL) || (storePrivate->server == MACH_PORT_NULL)) {
113 return kSCStatusNoStoreSession; /* you must have an open session to play */
114 }
115
116 if (_configd_trace) {
117 SCTrace(TRUE, _configd_trace,
118 CFSTR("%s : %5d\n"),
119 internal ? "*close " : "close ",
120 storePrivate->server);
121 }
122
123 /* Remove all notification keys and patterns */
124 removeAllKeys(*store, FALSE); // keys
125 removeAllKeys(*store, TRUE); // patterns
126
127 /* Remove/cancel any outstanding notification requests. */
128 (void) __SCDynamicStoreNotifyCancel(*store);
129
130 /* Remove any session keys */
131 sessionKey = CFStringCreateWithFormat(NULL, NULL, CFSTR("%d"), storePrivate->server);
132 dict = CFDictionaryGetValue(sessionData, sessionKey);
133 keys = CFDictionaryGetValue(dict, kSCDSessionKeys);
134 if (keys && ((keyCnt = CFArrayGetCount(keys)) > 0)) {
135 Boolean wasLocked;
136 CFIndex i;
137
138 /*
139 * if necessary, claim a lock to ensure that we inform
140 * any processes that a session key was removed.
141 */
142 wasLocked = (storeLocked > 0);
143 if (!wasLocked) {
144 (void) __SCDynamicStoreLock(*store, FALSE);
145 }
146
147 /* remove keys from "locked" store" */
148 for (i = 0; i < keyCnt; i++) {
149 if (isMySessionKey(sessionKey, CFArrayGetValueAtIndex(keys, i))) {
150 (void) __SCDynamicStoreRemoveValue(*store, CFArrayGetValueAtIndex(keys, i), TRUE);
151 }
152 }
153
154 if (wasLocked) {
155 /* remove keys from "unlocked" store" */
156 _swapLockedStoreData();
157 for (i = 0; i < keyCnt; i++) {
158 if (isMySessionKey(sessionKey, CFArrayGetValueAtIndex(keys, i)))
159 (void) __SCDynamicStoreRemoveValue(*store, CFArrayGetValueAtIndex(keys, i), TRUE);
160 }
161 _swapLockedStoreData();
162 }
163
164 /*
165 * Note: everyone who calls __SCDynamicStoreClose() ends
166 * up removing this sessions dictionary. As such,
167 * we don't need to worry about the session keys.
168 */
169 }
170 CFRelease(sessionKey);
171
172 /* release the lock */
173 if (storePrivate->locked) {
174 (void) __SCDynamicStoreUnlock(*store, FALSE);
175 }
176
177 /*
178 * invalidate and release our run loop source on the server
179 * port (for this client). Then, release the port.
180 */
181 mySession = getSession(storePrivate->server);
182 if (mySession->serverRunLoopSource) {
183 CFRunLoopSourceInvalidate(mySession->serverRunLoopSource);
184 CFRelease(mySession->serverRunLoopSource);
185 }
186 CFMachPortInvalidate(mySession->serverPort);
187 CFRelease(mySession->serverPort);
188
189 storePrivate->server = MACH_PORT_NULL;
190 CFRelease(*store);
191 *store = NULL;
192
193 return kSCStatusOK;
194 }
195
196
197 __private_extern__
198 kern_return_t
199 _configclose(mach_port_t server, int *sc_status)
200 {
201 serverSessionRef mySession = getSession(server);
202
203 if (!mySession) {
204 *sc_status = kSCStatusNoStoreSession; /* you must have an open session to play */
205 return KERN_SUCCESS;
206 }
207
208 /*
209 * Close the session.
210 */
211 *sc_status = __SCDynamicStoreClose(&mySession->store, FALSE);
212 if (*sc_status != kSCStatusOK) {
213 return KERN_SUCCESS;
214 }
215
216 /*
217 * Remove the session entry.
218 */
219 removeSession(server);
220
221 return KERN_SUCCESS;
222 }