2 * Copyright (c) 2005 Apple Computer, 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@
23 #include <CoreFoundation/CoreFoundation.h>
24 #include <mach/mach.h>
25 #include <servers/bootstrap.h>
26 #include <sys/types.h>
29 #include <sys/socket.h>
31 #include <sys/fcntl.h>
32 #include <sys/event.h>
33 #include <sys/resource.h>
34 #include <sys/param.h>
35 #include <netinet/in.h>
46 #include <readline/readline.h>
47 #include <readline/history.h>
51 #include "launch_priv.h"
53 #define LAUNCH_SECDIR "/tmp/launch-XXXXXX"
55 #define MACHINIT_JOBKEY_ONDEMAND "OnDemand"
56 #define MACHINIT_JOBKEY_SERVICENAME "ServiceName"
57 #define MACHINIT_JOBKEY_COMMAND "Command"
58 #define MACHINIT_JOBKEY_ISKUNCSERVER "isKUNCServer"
61 static void myCFDictionaryApplyFunction(const void *key
, const void *value
, void *context
);
62 static bool launch_data_array_append(launch_data_t a
, launch_data_t o
);
63 static void distill_jobs(launch_data_t
);
64 static void distill_config_file(launch_data_t
);
65 static void sock_dict_cb(launch_data_t what
, const char *key
, void *context
);
66 static void sock_dict_edit_entry(launch_data_t tmp
, const char *key
, launch_data_t fdarray
, launch_data_t thejob
);
67 static launch_data_t
CF2launch_data(CFTypeRef
);
68 static launch_data_t
read_plist_file(const char *file
, bool editondisk
, bool load
);
69 static CFPropertyListRef
CreateMyPropertyListFromFile(const char *);
70 static void WriteMyPropertyListToFile(CFPropertyListRef
, const char *);
71 static void readpath(const char *, launch_data_t
, launch_data_t
, bool editondisk
, bool load
, bool forceload
);
72 static void readfile(const char *, launch_data_t
, launch_data_t
, bool editondisk
, bool load
, bool forceload
);
74 static int demux_cmd(int argc
, char *const argv
[]);
75 static void do_rendezvous_magic(const struct addrinfo
*res
, const char *serv
, const char *label
);
76 static void workaround_bonjour_asynchronously(void);
77 static void submit_job_pass(launch_data_t jobs
);
78 static void submit_mach_jobs(launch_data_t jobs
);
79 static void let_go_of_mach_jobs(void);
80 static void do_mgroup_join(int fd
, int family
, int socktype
, int protocol
, const char *mgroup
);
81 static void print_jobs(launch_data_t j
, const char *label
, void *context
);
82 static bool is_legacy_mach_job(launch_data_t obj
);
84 static int load_and_unload_cmd(int argc
, char *const argv
[]);
85 //static int reload_cmd(int argc, char *const argv[]);
86 static int start_and_stop_cmd(int argc
, char *const argv
[]);
87 static int list_cmd(int argc
, char *const argv
[]);
89 static int setenv_cmd(int argc
, char *const argv
[]);
90 static int unsetenv_cmd(int argc
, char *const argv
[]);
91 static int getenv_and_export_cmd(int argc
, char *const argv
[]);
93 static int limit_cmd(int argc
, char *const argv
[]);
94 static int stdio_cmd(int argc
, char *const argv
[]);
95 static int fyi_cmd(int argc
, char *const argv
[]);
96 static int logupdate_cmd(int argc
, char *const argv
[]);
97 static int umask_cmd(int argc
, char *const argv
[]);
98 static int getrusage_cmd(int argc
, char *const argv
[]);
100 static int help_cmd(int argc
, char *const argv
[]);
102 static const struct {
104 int (*func
)(int argc
, char *const argv
[]);
107 { "load", load_and_unload_cmd
, "Load configuration files and/or directories" },
108 { "unload", load_and_unload_cmd
, "Unload configuration files and/or directories" },
109 // { "reload", reload_cmd, "Reload configuration files and/or directories" },
110 { "start", start_and_stop_cmd
, "Start specified jobs" },
111 { "stop", start_and_stop_cmd
, "Stop specified jobs" },
112 { "list", list_cmd
, "List jobs and information about jobs" },
113 { "setenv", setenv_cmd
, "Set an environmental variable in launchd" },
114 { "unsetenv", unsetenv_cmd
, "Unset an environmental variable in launchd" },
115 { "getenv", getenv_and_export_cmd
, "Get an environmental variable from launchd" },
116 { "export", getenv_and_export_cmd
, "Export shell settings from launchd" },
117 { "limit", limit_cmd
, "View and adjust launchd resource limits" },
118 { "stdout", stdio_cmd
, "Redirect launchd's standard out to the given path" },
119 { "stderr", stdio_cmd
, "Redirect launchd's standard error to the given path" },
120 { "shutdown", fyi_cmd
, "Prepare for system shutdown" },
121 { "reloadttys", fyi_cmd
, "Reload /etc/ttys" },
122 { "getrusage", getrusage_cmd
, "Get resource usage statistics from launchd" },
123 { "log", logupdate_cmd
, "Adjust the logging level or mask of launchd" },
124 { "umask", umask_cmd
, "Change launchd's umask" },
125 { "help", help_cmd
, "This help output" },
128 int main(int argc
, char *const argv
[])
130 bool istty
= isatty(STDIN_FILENO
);
134 exit(demux_cmd(argc
- 1, argv
+ 1));
136 if (NULL
== readline
) {
137 fprintf(stderr
, "missing library: readline\n");
141 while ((l
= readline(istty
? "launchd% " : NULL
))) {
142 char *inputstring
= l
, *argv2
[100], **ap
= argv2
;
145 while ((*ap
= strsep(&inputstring
, " \t"))) {
164 static int demux_cmd(int argc
, char *const argv
[])
171 for (i
= 0; i
< (sizeof cmds
/ sizeof cmds
[0]); i
++) {
172 if (!strcmp(cmds
[i
].name
, argv
[0]))
173 return cmds
[i
].func(argc
, argv
);
176 fprintf(stderr
, "%s: unknown subcommand \"%s\"\n", getprogname(), argv
[0]);
180 static int unsetenv_cmd(int argc
, char *const argv
[])
182 launch_data_t resp
, tmp
, msg
;
185 fprintf(stderr
, "%s usage: unsetenv <key>\n", getprogname());
189 msg
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
191 tmp
= launch_data_new_string(argv
[1]);
192 launch_data_dict_insert(msg
, tmp
, LAUNCH_KEY_UNSETUSERENVIRONMENT
);
194 resp
= launch_msg(msg
);
196 launch_data_free(msg
);
199 launch_data_free(resp
);
201 fprintf(stderr
, "launch_msg(\"%s\"): %s\n", LAUNCH_KEY_UNSETUSERENVIRONMENT
, strerror(errno
));
207 static int setenv_cmd(int argc
, char *const argv
[])
209 launch_data_t resp
, tmp
, tmpv
, msg
;
212 fprintf(stderr
, "%s usage: setenv <key> <value>\n", getprogname());
216 msg
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
217 tmp
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
219 tmpv
= launch_data_new_string(argv
[2]);
220 launch_data_dict_insert(tmp
, tmpv
, argv
[1]);
221 launch_data_dict_insert(msg
, tmp
, LAUNCH_KEY_SETUSERENVIRONMENT
);
223 resp
= launch_msg(msg
);
224 launch_data_free(msg
);
227 launch_data_free(resp
);
229 fprintf(stderr
, "launch_msg(\"%s\"): %s\n", LAUNCH_KEY_SETUSERENVIRONMENT
, strerror(errno
));
235 static void print_launchd_env(launch_data_t obj
, const char *key
, void *context
)
237 bool *is_csh
= context
;
240 fprintf(stdout
, "setenv %s %s;\n", key
, launch_data_get_string(obj
));
242 fprintf(stdout
, "%s=%s; export %s;\n", key
, launch_data_get_string(obj
), key
);
245 static void print_key_value(launch_data_t obj
, const char *key
, void *context
)
247 const char *k
= context
;
250 fprintf(stdout
, "%s\n", launch_data_get_string(obj
));
253 static int getenv_and_export_cmd(int argc
, char *const argv
[] __attribute__((unused
)))
255 launch_data_t resp
, msg
;
259 if (!strcmp(argv
[0], "export")) {
260 char *s
= getenv("SHELL");
262 is_csh
= strstr(s
, "csh") ? true : false;
263 } else if (argc
!= 2) {
264 fprintf(stderr
, "%s usage: getenv <key>\n", getprogname());
270 msg
= launch_data_new_string(LAUNCH_KEY_GETUSERENVIRONMENT
);
272 resp
= launch_msg(msg
);
273 launch_data_free(msg
);
276 if (!strcmp(argv
[0], "export"))
277 launch_data_dict_iterate(resp
, print_launchd_env
, &is_csh
);
279 launch_data_dict_iterate(resp
, print_key_value
, k
);
280 launch_data_free(resp
);
282 fprintf(stderr
, "launch_msg(\"" LAUNCH_KEY_GETUSERENVIRONMENT
"\"): %s\n", strerror(errno
));
287 static void unloadjob(launch_data_t job
)
289 launch_data_t resp
, tmp
, tmps
, msg
;
292 msg
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
293 tmp
= launch_data_alloc(LAUNCH_DATA_STRING
);
294 tmps
= launch_data_dict_lookup(job
, LAUNCH_JOBKEY_LABEL
);
297 fprintf(stderr
, "%s: Error: Missing Key: %s\n", getprogname(), LAUNCH_JOBKEY_LABEL
);
301 launch_data_set_string(tmp
, launch_data_get_string(tmps
));
302 launch_data_dict_insert(msg
, tmp
, LAUNCH_KEY_REMOVEJOB
);
303 resp
= launch_msg(msg
);
304 launch_data_free(msg
);
306 fprintf(stderr
, "%s: Error: launch_msg(): %s\n", getprogname(), strerror(errno
));
309 if (LAUNCH_DATA_ERRNO
== launch_data_get_type(resp
)) {
310 if ((e
= launch_data_get_errno(resp
)))
311 fprintf(stderr
, "%s\n", strerror(e
));
313 launch_data_free(resp
);
317 read_plist_file(const char *file
, bool editondisk
, bool load
)
319 CFPropertyListRef plist
= CreateMyPropertyListFromFile(file
);
320 launch_data_t r
= NULL
;
323 fprintf(stderr
, "%s: no plist was returned for: %s\n", getprogname(), file
);
329 CFDictionaryRemoveValue((CFMutableDictionaryRef
)plist
, CFSTR(LAUNCH_JOBKEY_DISABLED
));
331 CFDictionarySetValue((CFMutableDictionaryRef
)plist
, CFSTR(LAUNCH_JOBKEY_DISABLED
), kCFBooleanTrue
);
332 WriteMyPropertyListToFile(plist
, file
);
335 r
= CF2launch_data(plist
);
343 readfile(const char *what
, launch_data_t pass0
, launch_data_t pass1
, bool editondisk
, bool load
, bool forceload
)
345 launch_data_t tmpd
, thejob
;
346 bool job_disabled
= false;
348 if (NULL
== (thejob
= read_plist_file(what
, editondisk
, load
))) {
349 fprintf(stderr
, "%s: no plist was returned for: %s\n", getprogname(), what
);
353 if (is_legacy_mach_job(thejob
)) {
354 launch_data_array_append(pass0
, thejob
);
358 if (NULL
== launch_data_dict_lookup(thejob
, LAUNCH_JOBKEY_LABEL
)) {
359 fprintf(stderr
, "%s: missing the Label key: %s\n", getprogname(), what
);
360 launch_data_free(thejob
);
364 if ((tmpd
= launch_data_dict_lookup(thejob
, LAUNCH_JOBKEY_DISABLED
)))
365 job_disabled
= launch_data_get_bool(tmpd
);
368 job_disabled
= false;
370 if (job_disabled
&& load
) {
371 launch_data_free(thejob
);
375 launch_data_array_append(pass1
, thejob
);
379 readpath(const char *what
, launch_data_t pass0
, launch_data_t pass1
, bool editondisk
, bool load
, bool forceload
)
381 char buf
[MAXPATHLEN
];
386 if (stat(what
, &sb
) == -1)
389 if (S_ISREG(sb
.st_mode
) && !(sb
.st_mode
& S_IWOTH
)) {
390 readfile(what
, pass0
, pass1
, editondisk
, load
, forceload
);
392 if ((d
= opendir(what
)) == NULL
) {
393 fprintf(stderr
, "%s: opendir() failed to open the directory\n", getprogname());
397 while ((de
= readdir(d
))) {
398 if ((de
->d_name
[0] == '.'))
400 snprintf(buf
, sizeof(buf
), "%s/%s", what
, de
->d_name
);
402 readfile(buf
, pass0
, pass1
, editondisk
, load
, forceload
);
408 struct distill_context
{
410 launch_data_t newsockdict
;
414 distill_jobs(launch_data_t jobs
)
416 size_t i
, c
= launch_data_array_get_count(jobs
);
418 for (i
= 0; i
< c
; i
++)
419 distill_config_file(launch_data_array_get_index(jobs
, i
));
423 distill_config_file(launch_data_t id_plist
)
425 struct distill_context dc
= { id_plist
, NULL
};
428 if ((tmp
= launch_data_dict_lookup(id_plist
, LAUNCH_JOBKEY_SOCKETS
))) {
429 dc
.newsockdict
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
430 launch_data_dict_iterate(tmp
, sock_dict_cb
, &dc
);
431 launch_data_dict_insert(dc
.base
, dc
.newsockdict
, LAUNCH_JOBKEY_SOCKETS
);
435 static void sock_dict_cb(launch_data_t what
, const char *key
, void *context
)
437 struct distill_context
*dc
= context
;
438 launch_data_t fdarray
= launch_data_alloc(LAUNCH_DATA_ARRAY
);
440 launch_data_dict_insert(dc
->newsockdict
, fdarray
, key
);
442 if (launch_data_get_type(what
) == LAUNCH_DATA_DICTIONARY
) {
443 sock_dict_edit_entry(what
, key
, fdarray
, dc
->base
);
444 } else if (launch_data_get_type(what
) == LAUNCH_DATA_ARRAY
) {
448 for (i
= 0; i
< launch_data_array_get_count(what
); i
++) {
449 tmp
= launch_data_array_get_index(what
, i
);
450 sock_dict_edit_entry(tmp
, key
, fdarray
, dc
->base
);
455 static void sock_dict_edit_entry(launch_data_t tmp
, const char *key
, launch_data_t fdarray
, launch_data_t thejob
)
457 launch_data_t a
, val
;
458 const char *joblabel
;
459 int sfd
, st
= SOCK_STREAM
;
462 assert((val
= launch_data_dict_lookup(thejob
, LAUNCH_JOBKEY_LABEL
)) != NULL
);
463 joblabel
= launch_data_get_string(val
);
465 if ((val
= launch_data_dict_lookup(tmp
, LAUNCH_JOBSOCKETKEY_TYPE
))) {
466 if (!strcasecmp(launch_data_get_string(val
), "stream")) {
468 } else if (!strcasecmp(launch_data_get_string(val
), "dgram")) {
470 } else if (!strcasecmp(launch_data_get_string(val
), "seqpacket")) {
475 if ((val
= launch_data_dict_lookup(tmp
, LAUNCH_JOBSOCKETKEY_PASSIVE
)))
476 passive
= launch_data_get_bool(val
);
478 if ((val
= launch_data_dict_lookup(tmp
, LAUNCH_JOBSOCKETKEY_SECUREWITHKEY
))) {
479 char secdir
[] = LAUNCH_SECDIR
, buf
[1024];
480 launch_data_t uenv
= launch_data_dict_lookup(thejob
, LAUNCH_JOBKEY_USERENVIRONMENTVARIABLES
);
483 uenv
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
484 launch_data_dict_insert(thejob
, uenv
, LAUNCH_JOBKEY_USERENVIRONMENTVARIABLES
);
489 sprintf(buf
, "%s/%s", secdir
, key
);
491 a
= launch_data_new_string(buf
);
492 launch_data_dict_insert(tmp
, a
, LAUNCH_JOBSOCKETKEY_PATHNAME
);
493 a
= launch_data_new_string(buf
);
494 launch_data_dict_insert(uenv
, a
, launch_data_get_string(val
));
497 if ((val
= launch_data_dict_lookup(tmp
, LAUNCH_JOBSOCKETKEY_PATHNAME
))) {
498 struct sockaddr_un sun
;
503 memset(&sun
, 0, sizeof(sun
));
505 sun
.sun_family
= AF_UNIX
;
507 strncpy(sun
.sun_path
, launch_data_get_string(val
), sizeof(sun
.sun_path
));
509 if ((sfd
= _fd(socket(AF_UNIX
, st
, 0))) == -1)
512 if ((val
= launch_data_dict_lookup(tmp
, LAUNCH_JOBSOCKETKEY_PATHMODE
))) {
513 sun_mode
= (mode_t
)launch_data_get_integer(val
);
518 if (unlink(sun
.sun_path
) == -1 && errno
!= ENOENT
) {
522 oldmask
= umask(S_IRWXG
|S_IRWXO
);
523 if (bind(sfd
, (struct sockaddr
*)&sun
, sizeof(sun
)) == -1) {
530 chmod(sun
.sun_path
, sun_mode
);
532 if ((st
== SOCK_STREAM
|| st
== SOCK_SEQPACKET
)
533 && listen(sfd
, SOMAXCONN
) == -1) {
537 } else if (connect(sfd
, (struct sockaddr
*)&sun
, sizeof(sun
)) == -1) {
542 val
= launch_data_new_fd(sfd
);
543 launch_data_array_append(fdarray
, val
);
545 launch_data_t rnames
= NULL
;
546 const char *node
= NULL
, *serv
= NULL
, *mgroup
= NULL
;
548 struct addrinfo hints
, *res0
, *res
;
549 int gerr
, sock_opt
= 1;
550 bool rendezvous
= false;
552 memset(&hints
, 0, sizeof(hints
));
554 hints
.ai_socktype
= st
;
556 hints
.ai_flags
|= AI_PASSIVE
;
558 if ((val
= launch_data_dict_lookup(tmp
, LAUNCH_JOBSOCKETKEY_NODENAME
)))
559 node
= launch_data_get_string(val
);
560 if ((val
= launch_data_dict_lookup(tmp
, LAUNCH_JOBSOCKETKEY_MULTICASTGROUP
)))
561 mgroup
= launch_data_get_string(val
);
562 if ((val
= launch_data_dict_lookup(tmp
, LAUNCH_JOBSOCKETKEY_SERVICENAME
))) {
563 if (LAUNCH_DATA_INTEGER
== launch_data_get_type(val
)) {
564 sprintf(servnbuf
, "%lld", launch_data_get_integer(val
));
567 serv
= launch_data_get_string(val
);
570 if ((val
= launch_data_dict_lookup(tmp
, LAUNCH_JOBSOCKETKEY_FAMILY
))) {
571 if (!strcasecmp("IPv4", launch_data_get_string(val
)))
572 hints
.ai_family
= AF_INET
;
573 else if (!strcasecmp("IPv6", launch_data_get_string(val
)))
574 hints
.ai_family
= AF_INET6
;
576 if ((val
= launch_data_dict_lookup(tmp
, LAUNCH_JOBSOCKETKEY_PROTOCOL
))) {
577 if (!strcasecmp("TCP", launch_data_get_string(val
)))
578 hints
.ai_protocol
= IPPROTO_TCP
;
580 if ((rnames
= launch_data_dict_lookup(tmp
, LAUNCH_JOBSOCKETKEY_BONJOUR
))) {
582 if (LAUNCH_DATA_BOOL
== launch_data_get_type(rnames
)) {
583 rendezvous
= launch_data_get_bool(rnames
);
588 if ((gerr
= getaddrinfo(node
, serv
, &hints
, &res0
)) != 0) {
589 fprintf(stderr
, "getaddrinfo(): %s\n", gai_strerror(gerr
));
593 for (res
= res0
; res
; res
= res
->ai_next
) {
594 if ((sfd
= _fd(socket(res
->ai_family
, res
->ai_socktype
, res
->ai_protocol
))) == -1) {
595 fprintf(stderr
, "socket(): %s\n", strerror(errno
));
598 if (hints
.ai_flags
& AI_PASSIVE
) {
599 if (AF_INET6
== res
->ai_family
&& -1 == setsockopt(sfd
, IPPROTO_IPV6
, IPV6_V6ONLY
,
600 (void *)&sock_opt
, sizeof(sock_opt
))) {
601 fprintf(stderr
, "setsockopt(IPV6_V6ONLY): %m");
605 if (setsockopt(sfd
, SOL_SOCKET
, SO_REUSEPORT
, (void *)&sock_opt
, sizeof(sock_opt
)) == -1) {
606 fprintf(stderr
, "setsockopt(SO_REUSEPORT): %s\n", strerror(errno
));
610 if (setsockopt(sfd
, SOL_SOCKET
, SO_REUSEADDR
, (void *)&sock_opt
, sizeof(sock_opt
)) == -1) {
611 fprintf(stderr
, "setsockopt(SO_REUSEADDR): %s\n", strerror(errno
));
615 if (bind(sfd
, res
->ai_addr
, res
->ai_addrlen
) == -1) {
616 fprintf(stderr
, "bind(): %s\n", strerror(errno
));
621 do_mgroup_join(sfd
, res
->ai_family
, res
->ai_socktype
, res
->ai_protocol
, mgroup
);
623 if ((res
->ai_socktype
== SOCK_STREAM
|| res
->ai_socktype
== SOCK_SEQPACKET
)
624 && listen(sfd
, SOMAXCONN
) == -1) {
625 fprintf(stderr
, "listen(): %s\n", strerror(errno
));
628 if (rendezvous
&& (res
->ai_family
== AF_INET
|| res
->ai_family
== AF_INET6
) &&
629 (res
->ai_socktype
== SOCK_STREAM
|| res
->ai_socktype
== SOCK_DGRAM
)) {
630 if (NULL
== rnames
) {
631 do_rendezvous_magic(res
, serv
, joblabel
);
632 } else if (LAUNCH_DATA_STRING
== launch_data_get_type(rnames
)) {
633 do_rendezvous_magic(res
, launch_data_get_string(rnames
), joblabel
);
634 } else if (LAUNCH_DATA_ARRAY
== launch_data_get_type(rnames
)) {
635 size_t rn_i
, rn_ac
= launch_data_array_get_count(rnames
);
637 for (rn_i
= 0; rn_i
< rn_ac
; rn_i
++) {
638 launch_data_t rn_tmp
= launch_data_array_get_index(rnames
, rn_i
);
640 do_rendezvous_magic(res
, launch_data_get_string(rn_tmp
), joblabel
);
643 /* <rdar://problem/3964648> Launchd should not register the same service more than once */
644 /* <rdar://problem/3965154> Switch to DNSServiceRegisterAddrInfo() */
648 if (connect(sfd
, res
->ai_addr
, res
->ai_addrlen
) == -1) {
649 fprintf(stderr
, "connect(): %s\n", strerror(errno
));
653 val
= launch_data_new_fd(sfd
);
654 launch_data_array_append(fdarray
, val
);
659 static void do_mgroup_join(int fd
, int family
, int socktype
, int protocol
, const char *mgroup
)
661 struct addrinfo hints
, *res0
, *res
;
663 struct ipv6_mreq m6req
;
666 memset(&hints
, 0, sizeof(hints
));
668 hints
.ai_flags
|= AI_PASSIVE
;
669 hints
.ai_family
= family
;
670 hints
.ai_socktype
= socktype
;
671 hints
.ai_protocol
= protocol
;
673 if ((gerr
= getaddrinfo(mgroup
, NULL
, &hints
, &res0
)) != 0) {
674 fprintf(stderr
, "getaddrinfo(): %s\n", gai_strerror(gerr
));
678 for (res
= res0
; res
; res
= res
->ai_next
) {
679 if (AF_INET
== family
) {
680 memset(&mreq
, 0, sizeof(mreq
));
681 mreq
.imr_multiaddr
= ((struct sockaddr_in
*)res
->ai_addr
)->sin_addr
;
682 if (setsockopt(fd
, IPPROTO_IP
, IP_ADD_MEMBERSHIP
, &mreq
, sizeof(mreq
)) == -1) {
683 fprintf(stderr
, "setsockopt(IP_ADD_MEMBERSHIP): %s\n", strerror(errno
));
687 } else if (AF_INET6
== family
) {
688 memset(&m6req
, 0, sizeof(m6req
));
689 m6req
.ipv6mr_multiaddr
= ((struct sockaddr_in6
*)res
->ai_addr
)->sin6_addr
;
690 if (setsockopt(fd
, IPPROTO_IPV6
, IPV6_JOIN_GROUP
, &m6req
, sizeof(m6req
)) == -1) {
691 fprintf(stderr
, "setsockopt(IPV6_JOIN_GROUP): %s\n", strerror(errno
));
696 fprintf(stderr
, "unknown family during multicast group bind!\n");
704 struct bonjour_magic
{
705 SLIST_ENTRY(bonjour_magic
) sle
;
711 static SLIST_HEAD(, bonjour_magic
) bm_later
= { NULL
};
714 do_rendezvous_magic(const struct addrinfo
*res
, const char *serv
, const char *joblabel
)
716 struct bonjour_magic
*bm
= calloc(1, sizeof(struct bonjour_magic
) + strlen(joblabel
) + 1);
717 const char *typestr
= "udp";
719 if (res
->ai_socktype
== SOCK_STREAM
)
722 strcpy(bm
->label
, joblabel
);
724 asprintf(&bm
->str
, "_%s._%s.", serv
, typestr
);
726 if (res
->ai_family
== AF_INET
) {
727 bm
->port
= ((struct sockaddr_in
*)res
->ai_addr
)->sin_port
;
729 bm
->port
= ((struct sockaddr_in6
*)res
->ai_addr
)->sin6_port
;
732 SLIST_INSERT_HEAD(&bm_later
, bm
, sle
);
737 workaround_bonjour_asynchronously(void)
739 launch_data_t resp
, msg
, msgpayload
, tmpa
;
740 struct bonjour_magic
*bm
;
741 DNSServiceErrorType error
;
742 DNSServiceRef service
;
748 signal(SIGHUP
, SIG_IGN
);
752 msg
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
753 msgpayload
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
755 SLIST_FOREACH(bm
, &bm_later
, sle
) {
757 error
= DNSServiceRegister(&service
, 0, 0, NULL
, bm
->str
, NULL
, NULL
, bm
->port
, 0, NULL
, NULL
, NULL
);
758 if (error
!= kDNSServiceErr_NoError
) {
759 fprintf(stderr
, "DNSServiceRegister(\"%s\"): %d\n", bm
->str
, error
);
762 fd
= DNSServiceRefSockFD(service
);
763 tmpa
= launch_data_dict_lookup(msgpayload
, bm
->label
);
765 tmpa
= launch_data_alloc(LAUNCH_DATA_ARRAY
);
766 launch_data_dict_insert(msgpayload
, tmpa
, bm
->label
);
768 launch_data_array_append(tmpa
, launch_data_new_fd(fd
));
771 launch_data_dict_insert(msg
, msgpayload
, LAUNCH_KEY_WORKAROUNDBONJOUR
);
773 resp
= launch_msg(msg
);
775 launch_data_free(msg
);
777 if (launch_data_get_type(resp
) == LAUNCH_DATA_ERRNO
) {
778 errno
= launch_data_get_errno(resp
);
779 fprintf(stderr
, "Workaround Bonjour: %s\n", strerror(errno
));
782 launch_data_free(resp
);
787 static CFPropertyListRef
CreateMyPropertyListFromFile(const char *posixfile
)
789 CFPropertyListRef propertyList
;
790 CFStringRef errorString
;
791 CFDataRef resourceData
;
795 fileURL
= CFURLCreateFromFileSystemRepresentation(kCFAllocatorDefault
, (const UInt8
*)posixfile
, strlen(posixfile
), false);
797 fprintf(stderr
, "%s: CFURLCreateFromFileSystemRepresentation(%s) failed\n", getprogname(), posixfile
);
798 if (!CFURLCreateDataAndPropertiesFromResource(kCFAllocatorDefault
, fileURL
, &resourceData
, NULL
, NULL
, &errorCode
))
799 fprintf(stderr
, "%s: CFURLCreateDataAndPropertiesFromResource(%s) failed: %d\n", getprogname(), posixfile
, (int)errorCode
);
800 propertyList
= CFPropertyListCreateFromXMLData(kCFAllocatorDefault
, resourceData
, kCFPropertyListMutableContainers
, &errorString
);
802 fprintf(stderr
, "%s: propertyList is NULL\n", getprogname());
807 static void WriteMyPropertyListToFile(CFPropertyListRef plist
, const char *posixfile
)
809 CFDataRef resourceData
;
813 fileURL
= CFURLCreateFromFileSystemRepresentation(kCFAllocatorDefault
, (const UInt8
*)posixfile
, strlen(posixfile
), false);
815 fprintf(stderr
, "%s: CFURLCreateFromFileSystemRepresentation(%s) failed\n", getprogname(), posixfile
);
816 resourceData
= CFPropertyListCreateXMLData(kCFAllocatorDefault
, plist
);
817 if (resourceData
== NULL
)
818 fprintf(stderr
, "%s: CFPropertyListCreateXMLData(%s) failed", getprogname(), posixfile
);
819 if (!CFURLWriteDataAndPropertiesToResource(fileURL
, resourceData
, NULL
, &errorCode
))
820 fprintf(stderr
, "%s: CFURLWriteDataAndPropertiesToResource(%s) failed: %d\n", getprogname(), posixfile
, (int)errorCode
);
823 void myCFDictionaryApplyFunction(const void *key
, const void *value
, void *context
)
825 launch_data_t ik
, iw
, where
= context
;
827 ik
= CF2launch_data(key
);
828 iw
= CF2launch_data(value
);
830 launch_data_dict_insert(where
, iw
, launch_data_get_string(ik
));
831 launch_data_free(ik
);
834 static launch_data_t
CF2launch_data(CFTypeRef cfr
)
837 CFTypeID cft
= CFGetTypeID(cfr
);
839 if (cft
== CFStringGetTypeID()) {
841 CFStringGetCString(cfr
, buf
, sizeof(buf
), kCFStringEncodingUTF8
);
842 r
= launch_data_alloc(LAUNCH_DATA_STRING
);
843 launch_data_set_string(r
, buf
);
844 } else if (cft
== CFBooleanGetTypeID()) {
845 r
= launch_data_alloc(LAUNCH_DATA_BOOL
);
846 launch_data_set_bool(r
, CFBooleanGetValue(cfr
));
847 } else if (cft
== CFArrayGetTypeID()) {
848 CFIndex i
, ac
= CFArrayGetCount(cfr
);
849 r
= launch_data_alloc(LAUNCH_DATA_ARRAY
);
850 for (i
= 0; i
< ac
; i
++) {
851 CFTypeRef v
= CFArrayGetValueAtIndex(cfr
, i
);
853 launch_data_t iv
= CF2launch_data(v
);
854 launch_data_array_set_index(r
, iv
, i
);
857 } else if (cft
== CFDictionaryGetTypeID()) {
858 r
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
859 CFDictionaryApplyFunction(cfr
, myCFDictionaryApplyFunction
, r
);
860 } else if (cft
== CFDataGetTypeID()) {
861 r
= launch_data_alloc(LAUNCH_DATA_ARRAY
);
862 launch_data_set_opaque(r
, CFDataGetBytePtr(cfr
), CFDataGetLength(cfr
));
863 } else if (cft
== CFNumberGetTypeID()) {
866 CFNumberType cfnt
= CFNumberGetType(cfr
);
868 case kCFNumberSInt8Type
:
869 case kCFNumberSInt16Type
:
870 case kCFNumberSInt32Type
:
871 case kCFNumberSInt64Type
:
872 case kCFNumberCharType
:
873 case kCFNumberShortType
:
874 case kCFNumberIntType
:
875 case kCFNumberLongType
:
876 case kCFNumberLongLongType
:
877 CFNumberGetValue(cfr
, kCFNumberLongLongType
, &n
);
878 r
= launch_data_alloc(LAUNCH_DATA_INTEGER
);
879 launch_data_set_integer(r
, n
);
881 case kCFNumberFloat32Type
:
882 case kCFNumberFloat64Type
:
883 case kCFNumberFloatType
:
884 case kCFNumberDoubleType
:
885 CFNumberGetValue(cfr
, kCFNumberDoubleType
, &d
);
886 r
= launch_data_alloc(LAUNCH_DATA_REAL
);
887 launch_data_set_real(r
, d
);
899 static int help_cmd(int argc
, char *const argv
[])
901 FILE *where
= stdout
;
905 if (argc
== 0 || argv
== NULL
)
908 fprintf(where
, "usage: %s <subcommand>\n", getprogname());
909 for (i
= 0; i
< (sizeof cmds
/ sizeof cmds
[0]); i
++) {
910 l
= strlen(cmds
[i
].name
);
914 for (i
= 0; i
< (sizeof cmds
/ sizeof cmds
[0]); i
++)
915 fprintf(where
, "\t%-*s\t%s\n", cmdwidth
, cmds
[i
].name
, cmds
[i
].desc
);
920 static int _fd(int fd
)
923 fcntl(fd
, F_SETFD
, 1);
927 static int load_and_unload_cmd(int argc
, char *const argv
[])
929 launch_data_t pass0
, pass1
;
935 if (!strcmp(argv
[0], "load"))
938 while ((ch
= getopt(argc
, argv
, "wF")) != -1) {
940 case 'w': wflag
= true; break;
941 case 'F': Fflag
= true; break;
943 fprintf(stderr
, "usage: %s load [-wF] paths...\n", getprogname());
951 fprintf(stderr
, "usage: %s load [-w] paths...\n", getprogname());
955 /* I wish I didn't need to do multiple passes, but I need to load mDNSResponder and use it too.
956 * And loading legacy mach init jobs is extra fun.
958 * In later versions of launchd, I hope to load everything in the first pass,
959 * then do the Bonjour magic on the jobs that need it, and reload them, but for now,
960 * I haven't thought through the various complexities of reloading jobs, and therefore
961 * launchd doesn't have reload support right now.
964 pass0
= launch_data_alloc(LAUNCH_DATA_ARRAY
);
965 pass1
= launch_data_alloc(LAUNCH_DATA_ARRAY
);
967 for (i
= 0; i
< argc
; i
++)
968 readpath(argv
[i
], pass0
, pass1
, wflag
, lflag
, Fflag
);
970 if (launch_data_array_get_count(pass0
) == 0 &&
971 launch_data_array_get_count(pass1
) == 0) {
972 fprintf(stderr
, "nothing found to %s\n", lflag
? "load" : "unload");
973 launch_data_free(pass0
);
974 launch_data_free(pass1
);
980 submit_mach_jobs(pass0
);
981 workaround_bonjour_asynchronously();
982 submit_job_pass(pass1
);
983 let_go_of_mach_jobs();
985 for (i
= 0; i
< (int)launch_data_array_get_count(pass1
); i
++)
986 unloadjob(launch_data_array_get_index(pass1
, i
));
992 static mach_port_t
*msrvs
= NULL
;
993 static size_t msrvs_cnt
= 0;
996 submit_mach_jobs(launch_data_t jobs
)
1000 c
= launch_data_array_get_count(jobs
);
1002 msrvs
= calloc(1, sizeof(mach_port_t
) * c
);
1005 for (i
= 0; i
< c
; i
++) {
1006 launch_data_t tmp
, oai
= launch_data_array_get_index(jobs
, i
);
1007 const char *sn
= NULL
, *cmd
= NULL
;
1008 bool d
= true, k
= false;
1009 mach_port_t msr
, msv
, mhp
;
1013 if ((tmp
= launch_data_dict_lookup(oai
, MACHINIT_JOBKEY_ONDEMAND
)))
1014 d
= launch_data_get_bool(tmp
);
1015 if ((tmp
= launch_data_dict_lookup(oai
, MACHINIT_JOBKEY_ISKUNCSERVER
)))
1016 k
= launch_data_get_bool(tmp
);
1017 if ((tmp
= launch_data_dict_lookup(oai
, MACHINIT_JOBKEY_SERVICENAME
)))
1018 sn
= launch_data_get_string(tmp
);
1019 if ((tmp
= launch_data_dict_lookup(oai
, MACHINIT_JOBKEY_COMMAND
)))
1020 cmd
= launch_data_get_string(tmp
);
1022 if ((kr
= bootstrap_create_server(bootstrap_port
, (char *)cmd
, u
, d
, &msr
)) != KERN_SUCCESS
) {
1023 fprintf(stderr
, "%s: bootstrap_create_server(): %d\n", getprogname(), kr
);
1026 if ((kr
= bootstrap_create_service(msr
, (char*)sn
, &msv
)) != KERN_SUCCESS
) {
1027 fprintf(stderr
, "%s: bootstrap_create_service(): %d\n", getprogname(), kr
);
1028 mach_port_destroy(mach_task_self(), msr
);
1032 mhp
= mach_host_self();
1033 if ((kr
= host_set_UNDServer(mhp
, msv
)) != KERN_SUCCESS
)
1034 fprintf(stderr
, "%s: host_set_UNDServer(): %s\n", getprogname(), mach_error_string(kr
));
1035 mach_port_deallocate(mach_task_self(), mhp
);
1037 mach_port_deallocate(mach_task_self(), msv
);
1043 let_go_of_mach_jobs(void)
1047 for (i
= 0; i
< msrvs_cnt
; i
++)
1048 mach_port_destroy(mach_task_self(), msrvs
[i
]);
1052 submit_job_pass(launch_data_t jobs
)
1054 launch_data_t msg
, resp
;
1058 if (launch_data_array_get_count(jobs
) == 0)
1061 msg
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
1063 launch_data_dict_insert(msg
, jobs
, LAUNCH_KEY_SUBMITJOB
);
1065 resp
= launch_msg(msg
);
1068 switch (launch_data_get_type(resp
)) {
1069 case LAUNCH_DATA_ERRNO
:
1070 if ((e
= launch_data_get_errno(resp
)))
1071 fprintf(stderr
, "%s\n", strerror(e
));
1073 case LAUNCH_DATA_ARRAY
:
1074 for (i
= 0; i
< launch_data_array_get_count(jobs
); i
++) {
1075 launch_data_t obatind
= launch_data_array_get_index(resp
, i
);
1076 launch_data_t jatind
= launch_data_array_get_index(jobs
, i
);
1077 const char *lab4job
= launch_data_get_string(launch_data_dict_lookup(jatind
, LAUNCH_JOBKEY_LABEL
));
1078 if (LAUNCH_DATA_ERRNO
== launch_data_get_type(obatind
)) {
1079 e
= launch_data_get_errno(obatind
);
1082 fprintf(stderr
, "%s: %s\n", lab4job
, "Already loaded");
1085 fprintf(stderr
, "%s: %s\n", lab4job
, "Not loaded");
1088 fprintf(stderr
, "%s: %s\n", lab4job
, strerror(e
));
1096 fprintf(stderr
, "unknown respose from launchd!\n");
1099 launch_data_free(resp
);
1101 fprintf(stderr
, "launch_msg(): %s\n", strerror(errno
));
1104 launch_data_free(msg
);
1107 static int start_and_stop_cmd(int argc
, char *const argv
[])
1109 launch_data_t resp
, msg
;
1110 const char *lmsgcmd
= LAUNCH_KEY_STOPJOB
;
1113 if (!strcmp(argv
[0], "start"))
1114 lmsgcmd
= LAUNCH_KEY_STARTJOB
;
1117 fprintf(stderr
, "usage: %s %s <job label>\n", getprogname(), argv
[0]);
1121 msg
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
1122 launch_data_dict_insert(msg
, launch_data_new_string(argv
[1]), lmsgcmd
);
1124 resp
= launch_msg(msg
);
1125 launch_data_free(msg
);
1128 fprintf(stderr
, "launch_msg(): %s\n", strerror(errno
));
1130 } else if (launch_data_get_type(resp
) == LAUNCH_DATA_ERRNO
) {
1131 if ((e
= launch_data_get_errno(resp
))) {
1132 fprintf(stderr
, "%s %s error: %s\n", getprogname(), argv
[0], strerror(e
));
1136 fprintf(stderr
, "%s %s returned unknown response\n", getprogname(), argv
[0]);
1140 launch_data_free(resp
);
1144 static void print_jobs(launch_data_t j
__attribute__((unused
)), const char *label
, void *context
__attribute__((unused
)))
1146 fprintf(stdout
, "%s\n", label
);
1149 static int list_cmd(int argc
, char *const argv
[])
1151 launch_data_t resp
, msg
;
1155 while ((ch
= getopt(argc
, argv
, "v")) != -1) {
1161 fprintf(stderr
, "usage: %s list [-v]\n", getprogname());
1167 fprintf(stderr
, "usage: %s list: \"-v\" flag not implemented yet\n", getprogname());
1171 msg
= launch_data_new_string(LAUNCH_KEY_GETJOBS
);
1172 resp
= launch_msg(msg
);
1173 launch_data_free(msg
);
1176 fprintf(stderr
, "launch_msg(): %s\n", strerror(errno
));
1178 } else if (launch_data_get_type(resp
) == LAUNCH_DATA_DICTIONARY
) {
1179 launch_data_dict_iterate(resp
, print_jobs
, NULL
);
1181 fprintf(stderr
, "%s %s returned unknown response\n", getprogname(), argv
[0]);
1185 launch_data_free(resp
);
1190 static int stdio_cmd(int argc
, char *const argv
[])
1192 launch_data_t resp
, msg
, tmp
;
1193 int e
, fd
= -1, r
= 0;
1196 fprintf(stderr
, "usage: %s %s <path>\n", getprogname(), argv
[0]);
1200 fd
= open(argv
[1], O_CREAT
|O_APPEND
|O_WRONLY
, DEFFILEMODE
);
1202 msg
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
1205 tmp
= launch_data_new_string(argv
[1]);
1207 tmp
= launch_data_new_fd(fd
);
1210 if (!strcmp(argv
[0], "stdout")) {
1211 launch_data_dict_insert(msg
, tmp
, LAUNCH_KEY_SETSTDOUT
);
1213 launch_data_dict_insert(msg
, tmp
, LAUNCH_KEY_SETSTDERR
);
1216 resp
= launch_msg(msg
);
1217 launch_data_free(msg
);
1220 fprintf(stderr
, "launch_msg(): %s\n", strerror(errno
));
1222 } else if (launch_data_get_type(resp
) == LAUNCH_DATA_ERRNO
) {
1223 if ((e
= launch_data_get_errno(resp
))) {
1224 fprintf(stderr
, "%s %s error: %s\n", getprogname(), argv
[0], strerror(e
));
1228 fprintf(stderr
, "%s %s returned unknown response\n", getprogname(), argv
[0]);
1235 launch_data_free(resp
);
1240 static int fyi_cmd(int argc
, char *const argv
[])
1242 launch_data_t resp
, msg
;
1243 const char *lmsgk
= LAUNCH_KEY_RELOADTTYS
;
1247 fprintf(stderr
, "usage: %s %s\n", getprogname(), argv
[0]);
1251 if (!strcmp(argv
[0], "shutdown"))
1252 lmsgk
= LAUNCH_KEY_SHUTDOWN
;
1254 msg
= launch_data_new_string(lmsgk
);
1255 resp
= launch_msg(msg
);
1256 launch_data_free(msg
);
1259 fprintf(stderr
, "launch_msg(): %s\n", strerror(errno
));
1261 } else if (launch_data_get_type(resp
) == LAUNCH_DATA_ERRNO
) {
1262 if ((e
= launch_data_get_errno(resp
))) {
1263 fprintf(stderr
, "%s %s error: %s\n", getprogname(), argv
[0], strerror(e
));
1267 fprintf(stderr
, "%s %s returned unknown response\n", getprogname(), argv
[0]);
1271 launch_data_free(resp
);
1276 static int logupdate_cmd(int argc
, char *const argv
[])
1278 launch_data_t resp
, msg
;
1279 int e
, i
, j
, r
= 0, m
= 0;
1280 bool badargs
= false, maskmode
= false, onlymode
= false, levelmode
= false;
1281 const char *whichcmd
= LAUNCH_KEY_SETLOGMASK
;
1282 static const struct {
1286 { "debug", LOG_DEBUG
},
1287 { "info", LOG_INFO
},
1288 { "notice", LOG_NOTICE
},
1289 { "warning", LOG_WARNING
},
1290 { "error", LOG_ERR
},
1291 { "critical", LOG_CRIT
},
1292 { "alert", LOG_ALERT
},
1293 { "emergency", LOG_EMERG
},
1295 int logtblsz
= sizeof logtbl
/ sizeof logtbl
[0];
1298 if (!strcmp(argv
[1], "mask"))
1300 else if (!strcmp(argv
[1], "only"))
1302 else if (!strcmp(argv
[1], "level"))
1309 m
= LOG_UPTO(LOG_DEBUG
);
1311 if (argc
> 2 && (maskmode
|| onlymode
)) {
1312 for (i
= 2; i
< argc
; i
++) {
1313 for (j
= 0; j
< logtblsz
; j
++) {
1314 if (!strcmp(argv
[i
], logtbl
[j
].name
)) {
1316 m
&= ~(LOG_MASK(logtbl
[j
].level
));
1318 m
|= LOG_MASK(logtbl
[j
].level
);
1322 if (j
== logtblsz
) {
1327 } else if (argc
> 2 && levelmode
) {
1328 for (j
= 0; j
< logtblsz
; j
++) {
1329 if (!strcmp(argv
[2], logtbl
[j
].name
)) {
1330 m
= LOG_UPTO(logtbl
[j
].level
);
1336 } else if (argc
== 1) {
1337 whichcmd
= LAUNCH_KEY_GETLOGMASK
;
1343 fprintf(stderr
, "usage: %s [[mask loglevels...] | [only loglevels...] [level loglevel]]\n", getprogname());
1347 if (whichcmd
== LAUNCH_KEY_SETLOGMASK
) {
1348 msg
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
1349 launch_data_dict_insert(msg
, launch_data_new_integer(m
), whichcmd
);
1351 msg
= launch_data_new_string(whichcmd
);
1354 resp
= launch_msg(msg
);
1355 launch_data_free(msg
);
1358 fprintf(stderr
, "launch_msg(): %s\n", strerror(errno
));
1360 } else if (launch_data_get_type(resp
) == LAUNCH_DATA_ERRNO
) {
1361 if ((e
= launch_data_get_errno(resp
))) {
1362 fprintf(stderr
, "%s %s error: %s\n", getprogname(), argv
[0], strerror(e
));
1365 } else if (launch_data_get_type(resp
) == LAUNCH_DATA_INTEGER
) {
1366 if (whichcmd
== LAUNCH_KEY_GETLOGMASK
) {
1367 m
= launch_data_get_integer(resp
);
1368 for (j
= 0; j
< logtblsz
; j
++) {
1369 if (m
& LOG_MASK(logtbl
[j
].level
))
1370 fprintf(stdout
, "%s ", logtbl
[j
].name
);
1372 fprintf(stdout
, "\n");
1375 fprintf(stderr
, "%s %s returned unknown response\n", getprogname(), argv
[0]);
1379 launch_data_free(resp
);
1384 static const struct {
1388 { "cpu", RLIMIT_CPU
},
1389 { "filesize", RLIMIT_FSIZE
},
1390 { "data", RLIMIT_DATA
},
1391 { "stack", RLIMIT_STACK
},
1392 { "core", RLIMIT_CORE
},
1393 { "rss", RLIMIT_RSS
},
1394 { "memlock", RLIMIT_MEMLOCK
},
1395 { "maxproc", RLIMIT_NPROC
},
1396 { "maxfiles", RLIMIT_NOFILE
}
1399 static const size_t limlookupcnt
= sizeof limlookup
/ sizeof limlookup
[0];
1401 static ssize_t
name2num(const char *n
)
1405 for (i
= 0; i
< limlookupcnt
; i
++) {
1406 if (!strcmp(limlookup
[i
].name
, n
)) {
1407 return limlookup
[i
].lim
;
1413 static const char *num2name(int n
)
1417 for (i
= 0; i
< limlookupcnt
; i
++) {
1418 if (limlookup
[i
].lim
== n
)
1419 return limlookup
[i
].name
;
1424 static const char *lim2str(rlim_t val
, char *buf
)
1426 if (val
== RLIM_INFINITY
)
1427 strcpy(buf
, "unlimited");
1429 sprintf(buf
, "%lld", val
);
1433 static bool str2lim(const char *buf
, rlim_t
*res
)
1436 *res
= strtoll(buf
, &endptr
, 10);
1437 if (!strcmp(buf
, "unlimited")) {
1438 *res
= RLIM_INFINITY
;
1440 } else if (*endptr
== '\0') {
1446 static int limit_cmd(int argc
__attribute__((unused
)), char *const argv
[])
1450 struct rlimit
*lmts
= NULL
;
1451 launch_data_t resp
, resp1
= NULL
, msg
, tmp
;
1455 rlim_t slim
= -1, hlim
= -1;
1456 bool badargs
= false;
1461 if (argc
>= 3 && str2lim(argv
[2], &slim
))
1466 if (argc
== 4 && str2lim(argv
[3], &hlim
))
1469 if (argc
>= 2 && -1 == (which
= name2num(argv
[1])))
1473 fprintf(stderr
, "usage: %s %s [", getprogname(), argv
[0]);
1474 for (i
= 0; i
< limlookupcnt
; i
++)
1475 fprintf(stderr
, "%s %s", limlookup
[i
].name
, (i
+ 1) == limlookupcnt
? "" : "| ");
1476 fprintf(stderr
, "[both | soft hard]]\n");
1480 msg
= launch_data_new_string(LAUNCH_KEY_GETRESOURCELIMITS
);
1481 resp
= launch_msg(msg
);
1482 launch_data_free(msg
);
1485 fprintf(stderr
, "launch_msg(): %s\n", strerror(errno
));
1487 } else if (launch_data_get_type(resp
) == LAUNCH_DATA_OPAQUE
) {
1488 lmts
= launch_data_get_opaque(resp
);
1489 lsz
= launch_data_get_opaque_size(resp
);
1491 for (i
= 0; i
< (lsz
/ sizeof(struct rlimit
)); i
++) {
1492 if (argc
== 2 && (size_t)which
!= i
)
1494 fprintf(stdout
, "\t%-12s%-15s%-15s\n", num2name(i
),
1495 lim2str(lmts
[i
].rlim_cur
, slimstr
),
1496 lim2str(lmts
[i
].rlim_max
, hlimstr
));
1499 } else if (launch_data_get_type(resp
) == LAUNCH_DATA_STRING
) {
1500 fprintf(stderr
, "%s %s error: %s\n", getprogname(), argv
[0], launch_data_get_string(resp
));
1503 fprintf(stderr
, "%s %s returned unknown response\n", getprogname(), argv
[0]);
1507 if (argc
<= 2 || r
!= 0) {
1508 launch_data_free(resp
);
1514 lmts
[which
].rlim_cur
= slim
;
1515 lmts
[which
].rlim_max
= hlim
;
1517 msg
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
1518 tmp
= launch_data_new_opaque(lmts
, lsz
);
1519 launch_data_dict_insert(msg
, tmp
, LAUNCH_KEY_SETRESOURCELIMITS
);
1520 resp
= launch_msg(msg
);
1521 launch_data_free(msg
);
1524 fprintf(stderr
, "launch_msg(): %s\n", strerror(errno
));
1526 } else if (launch_data_get_type(resp
) == LAUNCH_DATA_STRING
) {
1527 fprintf(stderr
, "%s %s error: %s\n", getprogname(), argv
[0], launch_data_get_string(resp
));
1529 } else if (launch_data_get_type(resp
) != LAUNCH_DATA_OPAQUE
) {
1530 fprintf(stderr
, "%s %s returned unknown response\n", getprogname(), argv
[0]);
1534 launch_data_free(resp
);
1535 launch_data_free(resp1
);
1540 static int umask_cmd(int argc
, char *const argv
[])
1542 launch_data_t resp
, msg
;
1543 bool badargs
= false;
1549 m
= strtol(argv
[1], &endptr
, 8);
1550 if (*endptr
!= '\0' || m
> 0777)
1554 if (argc
> 2 || badargs
) {
1555 fprintf(stderr
, "usage: %s %s <mask>\n", getprogname(), argv
[0]);
1561 msg
= launch_data_new_string(LAUNCH_KEY_GETUMASK
);
1563 msg
= launch_data_alloc(LAUNCH_DATA_DICTIONARY
);
1564 launch_data_dict_insert(msg
, launch_data_new_integer(m
), LAUNCH_KEY_SETUMASK
);
1566 resp
= launch_msg(msg
);
1567 launch_data_free(msg
);
1570 fprintf(stderr
, "launch_msg(): %s\n", strerror(errno
));
1572 } else if (launch_data_get_type(resp
) == LAUNCH_DATA_STRING
) {
1573 fprintf(stderr
, "%s %s error: %s\n", getprogname(), argv
[0], launch_data_get_string(resp
));
1575 } else if (launch_data_get_type(resp
) != LAUNCH_DATA_INTEGER
) {
1576 fprintf(stderr
, "%s %s returned unknown response\n", getprogname(), argv
[0]);
1578 } else if (argc
== 1) {
1579 fprintf(stdout
, "%o\n", (unsigned int)launch_data_get_integer(resp
));
1582 launch_data_free(resp
);
1587 static int getrusage_cmd(int argc
, char *const argv
[])
1589 launch_data_t resp
, msg
;
1590 bool badargs
= false;
1595 else if (strcmp(argv
[1], "self") && strcmp(argv
[1], "children"))
1599 fprintf(stderr
, "usage: %s %s self | children\n", getprogname(), argv
[0]);
1603 if (!strcmp(argv
[1], "self")) {
1604 msg
= launch_data_new_string(LAUNCH_KEY_GETRUSAGESELF
);
1606 msg
= launch_data_new_string(LAUNCH_KEY_GETRUSAGECHILDREN
);
1609 resp
= launch_msg(msg
);
1610 launch_data_free(msg
);
1613 fprintf(stderr
, "launch_msg(): %s\n", strerror(errno
));
1615 } else if (launch_data_get_type(resp
) == LAUNCH_DATA_ERRNO
) {
1616 fprintf(stderr
, "%s %s error: %s\n", getprogname(), argv
[0], strerror(launch_data_get_errno(resp
)));
1618 } else if (launch_data_get_type(resp
) == LAUNCH_DATA_OPAQUE
) {
1619 struct rusage
*rusage
= launch_data_get_opaque(resp
);
1620 fprintf(stdout
, "\t%-10f\tuser time used\n",
1621 (double)rusage
->ru_utime
.tv_sec
+ (double)rusage
->ru_utime
.tv_usec
/ (double)1000000);
1622 fprintf(stdout
, "\t%-10f\tsystem time used\n",
1623 (double)rusage
->ru_stime
.tv_sec
+ (double)rusage
->ru_stime
.tv_usec
/ (double)1000000);
1624 fprintf(stdout
, "\t%-10ld\tmax resident set size\n", rusage
->ru_maxrss
);
1625 fprintf(stdout
, "\t%-10ld\tshared text memory size\n", rusage
->ru_ixrss
);
1626 fprintf(stdout
, "\t%-10ld\tunshared data size\n", rusage
->ru_idrss
);
1627 fprintf(stdout
, "\t%-10ld\tunshared stack size\n", rusage
->ru_isrss
);
1628 fprintf(stdout
, "\t%-10ld\tpage reclaims\n", rusage
->ru_minflt
);
1629 fprintf(stdout
, "\t%-10ld\tpage faults\n", rusage
->ru_majflt
);
1630 fprintf(stdout
, "\t%-10ld\tswaps\n", rusage
->ru_nswap
);
1631 fprintf(stdout
, "\t%-10ld\tblock input operations\n", rusage
->ru_inblock
);
1632 fprintf(stdout
, "\t%-10ld\tblock output operations\n", rusage
->ru_oublock
);
1633 fprintf(stdout
, "\t%-10ld\tmessages sent\n", rusage
->ru_msgsnd
);
1634 fprintf(stdout
, "\t%-10ld\tmessages received\n", rusage
->ru_msgrcv
);
1635 fprintf(stdout
, "\t%-10ld\tsignals received\n", rusage
->ru_nsignals
);
1636 fprintf(stdout
, "\t%-10ld\tvoluntary context switches\n", rusage
->ru_nvcsw
);
1637 fprintf(stdout
, "\t%-10ld\tinvoluntary context switches\n", rusage
->ru_nivcsw
);
1639 fprintf(stderr
, "%s %s returned unknown response\n", getprogname(), argv
[0]);
1643 launch_data_free(resp
);
1648 static bool launch_data_array_append(launch_data_t a
, launch_data_t o
)
1650 size_t offt
= launch_data_array_get_count(a
);
1652 return launch_data_array_set_index(a
, o
, offt
);
1656 is_legacy_mach_job(launch_data_t obj
)
1658 bool has_servicename
= launch_data_dict_lookup(obj
, MACHINIT_JOBKEY_SERVICENAME
);
1659 bool has_command
= launch_data_dict_lookup(obj
, MACHINIT_JOBKEY_COMMAND
);
1660 bool has_label
= launch_data_dict_lookup(obj
, LAUNCH_JOBKEY_LABEL
);
1662 return has_command
&& has_servicename
&& !has_label
;