2 * Copyright (c) 2005 Apple Computer, Inc. All rights reserved.
4 * @APPLE_APACHE_LICENSE_HEADER_START@
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
10 * http://www.apache.org/licenses/LICENSE-2.0
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
18 * @APPLE_APACHE_LICENSE_HEADER_END@
21 static const char *const __rcs_file_version__
= "$Revision: 23748 $";
27 #include <Security/Authorization.h>
28 #include <Security/AuthorizationTags.h>
29 #include <Security/AuthSession.h>
31 #include <sys/types.h>
32 #include <sys/queue.h>
33 #include <sys/event.h>
35 #include <sys/ucred.h>
36 #include <sys/fcntl.h>
39 #include <sys/sysctl.h>
40 #include <sys/sockio.h>
42 #include <sys/resource.h>
43 #include <sys/ioctl.h>
44 #include <sys/mount.h>
45 #include <sys/kern_event.h>
46 #include <sys/reboot.h>
48 #include <netinet/in.h>
49 #include <netinet/in_var.h>
50 #include <netinet6/nd6.h>
70 #if TARGET_OS_EMBEDDED
74 #include "bootstrap.h"
76 #include "vproc_internal.h"
79 #include "launchd_runtime.h"
80 #include "launchd_core_logic.h"
81 #include "launchd_unix_ipc.h"
83 #define LAUNCHD_CONF ".launchd.conf"
84 #define SECURITY_LIB "/System/Library/Frameworks/Security.framework/Versions/A/Security"
85 #define SHUTDOWN_LOG_DIR "/var/log/shutdown"
88 extern char **environ
;
90 static void pfsystem_callback(void *, struct kevent
*);
92 static kq_callback kqpfsystem_callback
= pfsystem_callback
;
94 static void pid1_magic_init(void);
96 static void testfd_or_openfd(int fd
, const char *path
, int flags
);
97 static bool get_network_state(void);
98 static void monitor_networking_state(void);
99 static void fatal_signal_handler(int sig
, siginfo_t
*si
, void *uap
);
100 static void handle_pid1_crashes_separately(void);
101 static void prep_shutdown_log_dir(void);
103 #if TARGET_OS_EMBEDDED
104 static void *update_thread(void *nothing
);
107 static bool re_exec_in_single_user_mode
= false;
108 static void *crash_addr
;
109 static pid_t crash_pid
;
111 #if TARGET_OS_EMBEDDED
112 static unsigned int g_sync_frequency
= 30;
115 static bool shutdown_in_progress
= false;
116 bool debug_shutdown_hangs
= false;
117 bool network_up
= false;
120 main(int argc
, char *const *argv
)
125 testfd_or_openfd(STDIN_FILENO
, _PATH_DEVNULL
, O_RDONLY
);
126 testfd_or_openfd(STDOUT_FILENO
, _PATH_DEVNULL
, O_WRONLY
);
127 testfd_or_openfd(STDERR_FILENO
, _PATH_DEVNULL
, O_WRONLY
);
129 while ((ch
= getopt(argc
, argv
, "s")) != -1) {
131 case 's': sflag
= true; break; /* single user */
132 case '?': /* we should do something with the global optopt variable here */
134 fprintf(stderr
, "%s: ignoring unknown arguments\n", getprogname());
139 if (getpid() != 1 && getppid() != 1) {
140 fprintf(stderr
, "%s: This program is not meant to be run directly.\n", getprogname());
144 launchd_runtime_init();
146 if (NULL
== getenv("PATH")) {
147 setenv("PATH", _PATH_STDPATH
, 1);
156 #if TARGET_OS_EMBEDDED
158 pthread_attr_init(&attr
);
159 pthread_attr_setstacksize(&attr
, PTHREAD_STACK_MIN
);
161 if( getpid() == 1 ) {
162 /* Start the update thread -- rdar://problem/5039559&6153301 */
164 int err
= pthread_create(&t
, &attr
, update_thread
, NULL
);
165 launchd_assumes(err
== 0);
169 monitor_networking_state();
172 handle_pid1_crashes_separately();
177 launchd_runtime_init2();
183 handle_pid1_crashes_separately(void)
185 struct sigaction fsa
;
187 fsa
.sa_sigaction
= fatal_signal_handler
;
188 fsa
.sa_flags
= SA_SIGINFO
;
189 sigemptyset(&fsa
.sa_mask
);
191 launchd_assumes(sigaction(SIGILL
, &fsa
, NULL
) != -1);
192 launchd_assumes(sigaction(SIGFPE
, &fsa
, NULL
) != -1);
193 launchd_assumes(sigaction(SIGBUS
, &fsa
, NULL
) != -1);
194 launchd_assumes(sigaction(SIGSEGV
, &fsa
, NULL
) != -1);
197 #define PID1_CRASH_LOGFILE "/var/log/launchd-pid1.crash"
199 /* This hack forces the dynamic linker to resolve these symbols ASAP */
200 static __attribute__((unused
)) typeof(sync
) *__junk_dyld_trick1
= sync
;
201 static __attribute__((unused
)) typeof(sleep
) *__junk_dyld_trick2
= sleep
;
202 static __attribute__((unused
)) typeof(reboot
) *__junk_dyld_trick3
= reboot
;
205 fatal_signal_handler(int sig
, siginfo_t
*si
, void *uap
__attribute__((unused
)))
207 const char *doom_why
= "at instruction";
208 char *sample_args
[] = { "/usr/bin/sample", "1", "1", "-file", PID1_CRASH_LOGFILE
, NULL
};
212 crash_addr
= si
->si_addr
;
213 crash_pid
= si
->si_pid
;
215 unlink(PID1_CRASH_LOGFILE
);
217 switch ((sample_p
= vfork())) {
219 execve(sample_args
[0], sample_args
, environ
);
223 waitpid(sample_p
, &wstatus
, 0);
235 doom_why
= "trying to read/write";
238 runtime_syslog(LOG_EMERG
, "We crashed %s: %p (sent by PID %u)", doom_why
, crash_addr
, crash_pid
);
247 pid1_magic_init(void)
249 launchd_assumes(setsid() != -1);
250 launchd_assumes(chdir("/") != -1);
251 launchd_assumes(setlogin("root") != -1);
252 launchd_assumes(mount("fdesc", "/dev", MNT_UNION
, NULL
) != -1);
260 launchd_assumes(fcntl(fd
, F_SETFD
, 1) != -1);
266 prep_shutdown_log_dir(void)
268 launchd_assumes(mkdir(SHUTDOWN_LOG_DIR
, S_IRWXU
) != -1 || errno
== EEXIST
);
271 #if TARGET_OS_EMBEDDED
273 update_thread(void *nothing
__attribute__((unused
)))
275 while( g_sync_frequency
) {
277 sleep(g_sync_frequency
);
285 launchd_shutdown(void)
289 if (shutdown_in_progress
) {
293 shutdown_in_progress
= true;
295 if (getpid() == 1 && stat("/var/db/debugShutdownHangs", &sb
) != -1) {
297 * When this changes to a more sustainable API, update this:
298 * http://howto.apple.com/db.cgi?Debugging_Apps_Non-Responsive_At_Shutdown
300 runtime_setlogmask(LOG_UPTO(LOG_DEBUG
));
301 prep_shutdown_log_dir();
302 debug_shutdown_hangs
= true;
305 launchd_assert(jobmgr_shutdown(root_jobmgr
) != NULL
);
309 launchd_single_user(void)
311 runtime_syslog(LOG_NOTICE
, "Going to single-user mode");
313 re_exec_in_single_user_mode
= true;
319 runtime_kill(-1, SIGKILL
);
323 launchd_SessionCreate(void)
326 OSStatus (*sescr
)(SessionCreationFlags flags
, SessionAttributeBits attributes
);
329 if (launchd_assumes((seclib
= dlopen(SECURITY_LIB
, RTLD_LAZY
)) != NULL
)) {
330 if (launchd_assumes((sescr
= dlsym(seclib
, "SessionCreate")) != NULL
)) {
331 launchd_assumes(sescr(0, 0) == noErr
);
333 launchd_assumes(dlclose(seclib
) != -1);
339 testfd_or_openfd(int fd
, const char *path
, int flags
)
343 if (-1 != (tmpfd
= dup(fd
))) {
344 launchd_assumes(runtime_close(tmpfd
) == 0);
346 if (-1 == (tmpfd
= open(path
, flags
| O_NOCTTY
, DEFFILEMODE
))) {
347 runtime_syslog(LOG_ERR
, "open(\"%s\", ...): %m", path
);
348 } else if (tmpfd
!= fd
) {
349 launchd_assumes(dup2(tmpfd
, fd
) != -1);
350 launchd_assumes(runtime_close(tmpfd
) == 0);
356 get_network_state(void)
358 struct ifaddrs
*ifa
, *ifai
;
362 /* Workaround 4978696: getifaddrs() reports false ENOMEM */
363 while ((r
= getifaddrs(&ifa
)) == -1 && errno
== ENOMEM
) {
364 runtime_syslog(LOG_DEBUG
, "Worked around bug: 4978696");
365 launchd_assumes(sched_yield() != -1);
368 if (!launchd_assumes(r
!= -1)) {
372 for (ifai
= ifa
; ifai
; ifai
= ifai
->ifa_next
) {
373 if (!(ifai
->ifa_flags
& IFF_UP
)) {
376 if (ifai
->ifa_flags
& IFF_LOOPBACK
) {
379 if (ifai
->ifa_addr
->sa_family
!= AF_INET
&& ifai
->ifa_addr
->sa_family
!= AF_INET6
) {
392 monitor_networking_state(void)
394 int pfs
= _fd(socket(PF_SYSTEM
, SOCK_RAW
, SYSPROTO_EVENT
));
395 struct kev_request kev_req
;
397 network_up
= get_network_state();
399 if (!launchd_assumes(pfs
!= -1)) {
403 memset(&kev_req
, 0, sizeof(kev_req
));
404 kev_req
.vendor_code
= KEV_VENDOR_APPLE
;
405 kev_req
.kev_class
= KEV_NETWORK_CLASS
;
407 if (!launchd_assumes(ioctl(pfs
, SIOCSKEVFILT
, &kev_req
) != -1)) {
412 launchd_assumes(kevent_mod(pfs
, EVFILT_READ
, EV_ADD
, 0, 0, &kqpfsystem_callback
) != -1);
416 pfsystem_callback(void *obj
__attribute__((unused
)), struct kevent
*kev
)
418 bool new_networking_state
;
421 launchd_assumes(read(kev
->ident
, &buf
, sizeof(buf
)) != -1);
423 new_networking_state
= get_network_state();
425 if (new_networking_state
!= network_up
) {
426 network_up
= new_networking_state
;
427 jobmgr_dispatch_all_semaphores(root_jobmgr
);
432 _log_launchd_bug(const char *rcs_rev
, const char *path
, unsigned int line
, const char *test
)
434 int saved_errno
= errno
;
436 const char *file
= strrchr(path
, '/');
437 char *rcs_rev_tmp
= strchr(rcs_rev
, ' ');
446 strlcpy(buf
, rcs_rev
, sizeof(buf
));
448 strlcpy(buf
, rcs_rev_tmp
+ 1, sizeof(buf
));
449 rcs_rev_tmp
= strchr(buf
, ' ');
455 runtime_syslog(LOG_NOTICE
, "Bug: %s:%u (%s):%u: %s", file
, line
, buf
, saved_errno
, test
);