]> git.saurik.com Git - apple/launchd.git/blob - launchd/src/IPC.c
c82b1f06b1c7d3a54d36773f7713e54c50cc2833
[apple/launchd.git] / launchd / src / IPC.c
1 /**
2 * IPC.c - System Starter IPC routines
3 * Wilfredo Sanchez | wsanchez@opensource.apple.com
4 * Kevin Van Vechten | kevinvv@uclink4.berkeley.edu
5 * $Apple$
6 **
7 * Copyright (c) 1999-2002 Apple Computer, Inc. All rights reserved.
8 *
9 * @APPLE_APACHE_LICENSE_HEADER_START@
10 *
11 * Licensed under the Apache License, Version 2.0 (the "License");
12 * you may not use this file except in compliance with the License.
13 * You may obtain a copy of the License at
14 *
15 * http://www.apache.org/licenses/LICENSE-2.0
16 *
17 * Unless required by applicable law or agreed to in writing, software
18 * distributed under the License is distributed on an "AS IS" BASIS,
19 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
20 * See the License for the specific language governing permissions and
21 * limitations under the License.
22 *
23 * @APPLE_APACHE_LICENSE_HEADER_END@
24 **/
25
26 #include <sys/wait.h>
27 #include <mach/mach.h>
28 #include <mach/message.h>
29 #include <mach/mach_error.h>
30 #include <CoreFoundation/CoreFoundation.h>
31 #include <syslog.h>
32
33 #include "bootstrap_public.h"
34 #include "IPC.h"
35 #include "StartupItems.h"
36 #include "SystemStarter.h"
37 #include "SystemStarterIPC.h"
38
39 /* Structure to pass StartupContext and anItem to the termination handler. */
40 typedef struct TerminationContextStorage {
41 StartupContext aStartupContext;
42 CFMutableDictionaryRef anItem;
43 } *TerminationContext;
44
45 /**
46 * A CFMachPort invalidation callback that records the termination of
47 * a startup item task. Stops the current run loop to give system_starter
48 * another go at running items.
49 **/
50 static void
51 startupItemTerminated(CFMachPortRef aMachPort, void *anInfo)
52 {
53 TerminationContext aTerminationContext = (TerminationContext) anInfo;
54
55 if (aMachPort) {
56 mach_port_deallocate(mach_task_self(), CFMachPortGetPort(aMachPort));
57 }
58 if (aTerminationContext && aTerminationContext->anItem) {
59 pid_t aPID = 0;
60 pid_t rPID = 0;
61 int aStatus = 0;
62 CFMutableDictionaryRef anItem = aTerminationContext->anItem;
63 StartupContext aStartupContext = aTerminationContext->aStartupContext;
64
65 /* Get the exit status */
66 if (anItem) {
67 aPID = StartupItemGetPID(anItem);
68 if (aPID > 0)
69 rPID = waitpid(aPID, &aStatus, 0);
70 }
71 if (aStartupContext) {
72 --aStartupContext->aRunningCount;
73
74 /* Record the item's status */
75 if (aStartupContext->aStatusDict) {
76 StartupItemExit(aStartupContext->aStatusDict, anItem, (WIFEXITED(aStatus) && WEXITSTATUS(aStatus) == 0));
77 if (aStatus) {
78 CF_syslog(LOG_WARNING, CFSTR("%@ (%d) did not complete successfully"), CFDictionaryGetValue(anItem, CFSTR("Description")), aPID);
79 } else {
80 CF_syslog(LOG_DEBUG, CFSTR("Finished %@ (%d)"), CFDictionaryGetValue(anItem, CFSTR("Description")), aPID);
81 }
82 }
83 /*
84 * If the item failed to start, then add it to the
85 * failed list
86 */
87 if (WEXITSTATUS(aStatus) || WTERMSIG(aStatus) || WCOREDUMP(aStatus)) {
88 CFDictionarySetValue(anItem, kErrorKey, kErrorReturnNonZero);
89 AddItemToFailedList(aStartupContext, anItem);
90 }
91 /*
92 * Remove the item from the waiting list regardless
93 * if it was successful or it failed.
94 */
95 RemoveItemFromWaitingList(aStartupContext, anItem);
96 }
97 }
98 if (aTerminationContext)
99 free(aTerminationContext);
100 }
101
102 void
103 MonitorStartupItem(StartupContext aStartupContext, CFMutableDictionaryRef anItem)
104 {
105 pid_t aPID = StartupItemGetPID(anItem);
106 if (anItem && aPID > 0) {
107 mach_port_t aPort;
108 kern_return_t aResult;
109 CFMachPortContext aContext;
110 CFMachPortRef aMachPort;
111 CFRunLoopSourceRef aSource;
112 TerminationContext aTerminationContext = (TerminationContext) malloc(sizeof(struct TerminationContextStorage));
113
114 aTerminationContext->aStartupContext = aStartupContext;
115 aTerminationContext->anItem = anItem;
116
117 aContext.version = 0;
118 aContext.info = aTerminationContext;
119 aContext.retain = 0;
120 aContext.release = 0;
121
122 if ((aResult = task_name_for_pid(mach_task_self(), aPID, &aPort)) != KERN_SUCCESS)
123 goto out_bad;
124
125 if (!(aMachPort = CFMachPortCreateWithPort(NULL, aPort, NULL, &aContext, NULL)))
126 goto out_bad;
127
128 if (!(aSource = CFMachPortCreateRunLoopSource(NULL, aMachPort, 0))) {
129 CFRelease(aMachPort);
130 goto out_bad;
131 }
132 CFMachPortSetInvalidationCallBack(aMachPort, startupItemTerminated);
133 CFRunLoopAddSource(CFRunLoopGetCurrent(), aSource, kCFRunLoopCommonModes);
134 CFRelease(aSource);
135 CFRelease(aMachPort);
136 return;
137 out_bad:
138 /*
139 * The assumption is something failed, the task already
140 * terminated.
141 */
142 startupItemTerminated(NULL, aTerminationContext);
143 }
144 }
145
146 /**
147 * Returns a reference to an item based on tokens passed in an IPC message.
148 * This is useful for figuring out which item the message came from.
149 *
150 * Currently two tokens are supported:
151 * kIPCProcessIDKey - the pid of the running startup script
152 * kIPCServiceNameKey - a name of a service that the item provides. This
153 * takes precedence over the pid key when both are present.
154 **/
155 static CFMutableDictionaryRef
156 itemFromIPCMessage(StartupContext aStartupContext, CFDictionaryRef anIPCMessage)
157 {
158 CFMutableDictionaryRef anItem = NULL;
159
160 if (aStartupContext && anIPCMessage) {
161 CFStringRef aServiceName = CFDictionaryGetValue(anIPCMessage, kIPCServiceNameKey);
162 CFIndex aPID = 0;
163 CFNumberRef aPIDNumber = CFDictionaryGetValue(anIPCMessage, kIPCProcessIDKey);
164
165 if (aServiceName && CFGetTypeID(aServiceName) == CFStringGetTypeID()) {
166 anItem = StartupItemListGetProvider(aStartupContext->aWaitingList, aServiceName);
167 } else if (aPIDNumber &&
168 CFGetTypeID(aPIDNumber) == CFNumberGetTypeID() &&
169 CFNumberGetValue(aPIDNumber, kCFNumberCFIndexType, &aPID)) {
170 anItem = StartupItemWithPID(aStartupContext->aWaitingList, aPID);
171 }
172 }
173 return anItem;
174 }
175
176 /**
177 * Displays a message on the console.
178 * aConsoleMessage will be localized according to the dictionary in the specified item.
179 * Running tems may be specified by their current process id. Items may also be specified
180 * by one of the service names they provide.
181 **/
182 static void
183 consoleMessage(StartupContext aStartupContext, CFDictionaryRef anIPCMessage)
184 {
185 if (aStartupContext && anIPCMessage) {
186 CFStringRef aConsoleMessage = CFDictionaryGetValue(anIPCMessage, kIPCConsoleMessageKey);
187
188 if (aConsoleMessage && CFGetTypeID(aConsoleMessage) == CFStringGetTypeID()) {
189 CF_syslog(LOG_INFO, CFSTR("%@"), aConsoleMessage);
190 }
191 }
192 }
193
194 /**
195 * Records the success or failure or a particular service.
196 * If no service name is specified, but a pid is, then all services provided
197 * by the item are flagged.
198 **/
199 static void
200 statusMessage(StartupContext aStartupContext, CFDictionaryRef anIPCMessage)
201 {
202 if (anIPCMessage && aStartupContext && aStartupContext->aStatusDict) {
203 CFMutableDictionaryRef anItem = itemFromIPCMessage(aStartupContext, anIPCMessage);
204 CFStringRef aServiceName = CFDictionaryGetValue(anIPCMessage, kIPCServiceNameKey);
205 CFBooleanRef aStatus = CFDictionaryGetValue(anIPCMessage, kIPCStatusKey);
206
207 if (anItem && aStatus &&
208 CFGetTypeID(aStatus) == CFBooleanGetTypeID() &&
209 (!aServiceName || CFGetTypeID(aServiceName) == CFStringGetTypeID())) {
210 StartupItemSetStatus(aStartupContext->aStatusDict, anItem, aServiceName, CFBooleanGetValue(aStatus), TRUE);
211 }
212 }
213 }
214
215 /**
216 * Queries one of several configuration settings.
217 */
218 static CFDataRef
219 queryConfigSetting(StartupContext aStartupContext, CFDictionaryRef anIPCMessage)
220 {
221 char *aValue = "";
222
223 if (anIPCMessage) {
224 CFStringRef aSetting = CFDictionaryGetValue(anIPCMessage, kIPCConfigSettingKey);
225
226 if (aSetting && CFGetTypeID(aSetting) == CFStringGetTypeID()) {
227 if (CFEqual(aSetting, kIPCConfigSettingVerboseFlag)) {
228 aValue = gVerboseFlag ? "-YES-" : "-NO-";
229 } else if (CFEqual(aSetting, kIPCConfigSettingNetworkUp)) {
230 Boolean aNetworkUpFlag = FALSE;
231 if (aStartupContext && aStartupContext->aStatusDict) {
232 aNetworkUpFlag = CFDictionaryContainsKey(aStartupContext->aStatusDict, CFSTR("Network"));
233 }
234 aValue = aNetworkUpFlag ? "-YES-" : "-NO-";
235 }
236 }
237 }
238 return CFDataCreate(NULL, (const UInt8 *)aValue, strlen(aValue) + 1); /* aValue + null */
239 }
240
241 static void *handleIPCMessage(void *aMsgParam, CFIndex aMessageSize __attribute__((unused)), CFAllocatorRef anAllocator __attribute__((unused)), void *aMachPort) {
242 SystemStarterIPCMessage *aMessage = (SystemStarterIPCMessage *) aMsgParam;
243 SystemStarterIPCMessage *aReplyMessage = NULL;
244
245 CFDataRef aResult = NULL;
246 CFDataRef aData = NULL;
247
248 if (aMessage->aHeader.msgh_bits & MACH_MSGH_BITS_COMPLEX) {
249 syslog(LOG_WARNING, "Ignoring out-of-line IPC message");
250 return NULL;
251 } else {
252 mach_msg_security_trailer_t *aSecurityTrailer = (mach_msg_security_trailer_t *)
253 ((uint8_t *) aMessage + round_msg(sizeof(SystemStarterIPCMessage) + aMessage->aByteLength));
254
255 /*
256 * CFRunLoop includes the format 0 message trailer with the
257 * passed message.
258 */
259 if (aSecurityTrailer->msgh_trailer_type == MACH_MSG_TRAILER_FORMAT_0 &&
260 aSecurityTrailer->msgh_sender.val[0] != 0) {
261 syslog(LOG_WARNING, "Ignoring IPC message sent from uid %d", aSecurityTrailer->msgh_sender.val[0]);
262 return NULL;
263 }
264 }
265
266 if (aMessage->aProtocol != kIPCProtocolVersion) {
267 syslog(LOG_WARNING, "Unsupported IPC protocol version number: %d. Message ignored", aMessage->aProtocol);
268 return NULL;
269 }
270 aData = CFDataCreateWithBytesNoCopy(NULL,
271 (uint8_t *) aMessage + sizeof(SystemStarterIPCMessage),
272 aMessage->aByteLength,
273 kCFAllocatorNull);
274 /*
275 * Dispatch the IPC message.
276 */
277 if (aData) {
278 StartupContext aStartupContext = NULL;
279 CFStringRef anErrorString = NULL;
280 CFDictionaryRef anIPCMessage = (CFDictionaryRef) CFPropertyListCreateFromXMLData(NULL, aData, kCFPropertyListImmutable, &anErrorString);
281
282 CF_syslog(LOG_DEBUG, CFSTR("IPC message = %@"), anIPCMessage);
283
284 if (aMachPort) {
285 CFMachPortContext aMachPortContext;
286 CFMachPortGetContext((CFMachPortRef) aMachPort, &aMachPortContext);
287 aStartupContext = (StartupContext) aMachPortContext.info;
288 }
289 if (anIPCMessage && CFGetTypeID(anIPCMessage) == CFDictionaryGetTypeID()) {
290 /* switch on the type of the IPC message */
291 CFStringRef anIPCMessageType = CFDictionaryGetValue(anIPCMessage, kIPCMessageKey);
292 if (anIPCMessageType && CFGetTypeID(anIPCMessageType) == CFStringGetTypeID()) {
293 if (CFEqual(anIPCMessageType, kIPCConsoleMessage)) {
294 consoleMessage(aStartupContext, anIPCMessage);
295 } else if (CFEqual(anIPCMessageType, kIPCStatusMessage)) {
296 statusMessage(aStartupContext, anIPCMessage);
297 } else if (CFEqual(anIPCMessageType, kIPCQueryMessage)) {
298 aResult = queryConfigSetting(aStartupContext, anIPCMessage);
299 }
300 }
301 } else {
302 CF_syslog(LOG_ERR, CFSTR("Unable to parse IPC message: %@"), anErrorString);
303 }
304 CFRelease(aData);
305 } else {
306 syslog(LOG_ERR, "Out of memory. Could not allocate space for IPC message");
307 }
308
309 /*
310 * Generate a Mach message for the result data.
311 */
312 if (!aResult)
313 aResult = CFDataCreateWithBytesNoCopy(NULL, (const UInt8 *)"", 1, kCFAllocatorNull);
314 if (aResult) {
315 CFIndex aDataSize = CFDataGetLength(aResult);
316 CFIndex aReplyMessageSize = round_msg(sizeof(SystemStarterIPCMessage) + aDataSize + 3);
317 aReplyMessage = CFAllocatorAllocate(kCFAllocatorSystemDefault, aReplyMessageSize, 0);
318 if (aReplyMessage) {
319 aReplyMessage->aHeader.msgh_id = -1 * (SInt32) aMessage->aHeader.msgh_id;
320 aReplyMessage->aHeader.msgh_size = aReplyMessageSize;
321 aReplyMessage->aHeader.msgh_remote_port = aMessage->aHeader.msgh_remote_port;
322 aReplyMessage->aHeader.msgh_local_port = MACH_PORT_NULL;
323 aReplyMessage->aHeader.msgh_reserved = 0;
324 aReplyMessage->aHeader.msgh_bits = MACH_MSGH_BITS(MACH_MSG_TYPE_MOVE_SEND_ONCE, 0);
325 aReplyMessage->aBody.msgh_descriptor_count = 0;
326 aReplyMessage->aProtocol = kIPCProtocolVersion;
327 aReplyMessage->aByteLength = CFDataGetLength(aResult);
328 memmove((uint8_t *) aReplyMessage + sizeof(SystemStarterIPCMessage),
329 CFDataGetBytePtr(aResult),
330 CFDataGetLength(aResult));
331 }
332 CFRelease(aResult);
333 }
334 if (!aReplyMessage) {
335 syslog(LOG_ERR, "Out of memory. Could not allocate IPC result");
336 }
337 return aReplyMessage;
338 }
339
340
341 static mach_port_t
342 getIPCPort(void *anInfo)
343 {
344 return anInfo ? CFMachPortGetPort((CFMachPortRef) anInfo) : MACH_PORT_NULL;
345 }
346
347 CFRunLoopSourceRef
348 CreateIPCRunLoopSource(CFStringRef aPortName, StartupContext aStartupContext)
349 {
350 CFRunLoopSourceRef aSource = NULL;
351 CFMachPortRef aMachPort = NULL;
352 CFMachPortContext aContext;
353 kern_return_t aKernReturn = KERN_FAILURE;
354
355 aContext.version = 0;
356 aContext.info = (void *) aStartupContext;
357 aContext.retain = 0;
358 aContext.release = 0;
359 aContext.copyDescription = 0;
360 aMachPort = CFMachPortCreate(NULL, NULL, &aContext, NULL);
361
362 if (aMachPort && aPortName) {
363 CFIndex aPortNameLength = CFStringGetLength(aPortName);
364 CFIndex aPortNameSize = CFStringGetMaximumSizeForEncoding(aPortNameLength, kCFStringEncodingUTF8) + 1;
365 char *aBuffer = CFAllocatorAllocate(NULL, aPortNameSize, 0);
366 if (aBuffer && CFStringGetCString(aPortName,
367 aBuffer,
368 aPortNameSize,
369 kCFStringEncodingUTF8)) {
370 aKernReturn = bootstrap_register(bootstrap_port, aBuffer, CFMachPortGetPort(aMachPort));
371 }
372 if (aBuffer)
373 CFAllocatorDeallocate(NULL, aBuffer);
374 }
375 if (aMachPort && aKernReturn == KERN_SUCCESS) {
376 CFRunLoopSourceContext1 aSourceContext;
377 aSourceContext.version = 1;
378 aSourceContext.info = aMachPort;
379 aSourceContext.retain = CFRetain;
380 aSourceContext.release = CFRelease;
381 aSourceContext.copyDescription = CFCopyDescription;
382 aSourceContext.equal = CFEqual;
383 aSourceContext.hash = CFHash;
384 aSourceContext.getPort = getIPCPort;
385 aSourceContext.perform = (void *) handleIPCMessage;
386 aSource = CFRunLoopSourceCreate(NULL, 0, (CFRunLoopSourceContext *) & aSourceContext);
387 }
388 if (aMachPort && (!aSource || aKernReturn != KERN_SUCCESS)) {
389 CFMachPortInvalidate(aMachPort);
390 CFRelease(aMachPort);
391 aMachPort = NULL;
392 }
393 return aSource;
394 }