2 * Copyright (c) 2016 Apple Inc. All rights reserved.
5 typedef char *kobject_description_t
[512];
13 #include <sys/types.h>
14 #include <sys/sysctl.h>
19 #include <sys/kauth.h>
35 #include <mach/mach.h>
40 char *(^sysc_string
)(const char *name
) = ^(const char *name
) {
44 if (-1 == sysctlbyname(name
, NULL
, &len
, NULL
, 0)) {
45 warnc(errno
, "sysctl: %s", name
);
46 } else if (0 != len
) {
48 if (-1 == sysctlbyname(name
, p
, &len
, NULL
, 0)) {
49 warnc(errno
, "sysctl: %s", name
);
57 char *s
= sysc_string("kern.corefile");
59 s
= strdup("/cores/core.%P");
63 static const struct proc_bsdinfo
*
64 get_bsdinfo(pid_t pid
)
68 struct proc_bsdinfo
*pbi
= calloc(1, sizeof (*pbi
));
69 if (0 != proc_pidinfo(pid
, PROC_PIDTBSDINFO
, 0, pbi
, sizeof (*pbi
)))
76 format_gcore_name(const char *fmt
, pid_t pid
, uid_t uid
, const char *nm
)
78 __block
size_t resid
= MAXPATHLEN
;
79 __block
char *p
= calloc(1, resid
);
82 int (^outchar
)(int c
) = ^(int c
) {
91 ptrdiff_t (^outstr
)(const char *str
) = ^(const char *str
) {
93 while (*s
&& 0 != outchar(*s
++))
98 ptrdiff_t (^outint
)(int value
) = ^(int value
) {
100 snprintf(id
, sizeof (id
), "%u", value
);
104 ptrdiff_t (^outtstamp
)(void) = ^(void) {
110 strftime(tstamp
, sizeof (tstamp
), "%Y%m%dT%H%M%SZ", &tm
);
111 return outstr(tstamp
);
116 for (int i
= 0; resid
> 0; i
++)
117 switch (c
= fmt
[i
]) {
123 switch (c
= fmt
[i
]) {
137 outtstamp(); // ISO 8601 format
141 err(EX_DATAERR
, "unknown format char: %%%c", c
);
143 err(EX_DATAERR
, "bad format char %%\\%03o", c
);
145 err(EX_DATAERR
, "bad format specifier");
157 make_gcore_path(char **corefmtp
, pid_t pid
, uid_t uid
, const char *nm
)
159 char *corefmt
= *corefmtp
;
160 if (NULL
== corefmt
) {
161 const char defcore
[] = "%N-%P-%T";
162 if (NULL
== (corefmt
= kern_corefile()))
163 corefmt
= strdup(defcore
);
165 // use the same directory as kern.corefile
166 char *p
= strrchr(corefmt
, '/');
169 size_t len
= strlen(corefmt
) + strlen(defcore
) + 2;
170 char *buf
= malloc(len
);
171 snprintf(buf
, len
, "%s/%s", corefmt
, defcore
);
175 if (OPTIONS_DEBUG(opt
, 3))
176 printf("corefmt '%s'\n", corefmt
);
179 char *path
= format_gcore_name(corefmt
, pid
, uid
, nm
);
185 static bool proc_same_data_model(const struct proc_bsdinfo
*pbi
) {
186 #if defined(__LP64__)
187 return (pbi
->pbi_flags
& PROC_FLAG_LP64
) != 0;
189 return (pbi
->pbi_flags
& PROC_FLAG_LP64
) == 0;
193 static bool task_same_data_model(const task_flags_info_data_t
*tfid
) {
194 #if defined(__LP64__)
195 return (tfid
->flags
& TF_LP64
) != 0;
197 return (tfid
->flags
& TF_LP64
) == 0;
202 * Change credentials for writing out the file
205 change_credentials(gid_t uid
, uid_t gid
)
207 if ((getgid() != gid
&& -1 == setgid(gid
)) ||
208 (getuid() != uid
&& -1 == setuid(uid
)))
209 errc(EX_NOPERM
, errno
, "insufficient privilege");
210 if (uid
!= getuid() || gid
!= getgid())
211 err(EX_OSERR
, "wrong credentials");
215 openout(const char *corefname
, char **coretname
, struct stat
*st
)
217 const int tfd
= open(corefname
, O_WRONLY
);
219 if (ENOENT
== errno
) {
221 * Arrange for a core file to appear "atomically": write the data
222 * to the file + ".tmp" suffix, then fchmod and rename it into
223 * place once the dump completes successfully.
225 const size_t nmlen
= strlen(corefname
) + 4 + 1;
226 char *tnm
= malloc(nmlen
);
227 snprintf(tnm
, nmlen
, "%s.tmp", corefname
);
228 const int fd
= open(tnm
, O_WRONLY
| O_CREAT
| O_TRUNC
, 0600);
229 if (-1 == fd
|| -1 == fstat(fd
, st
))
230 errc(EX_CANTCREAT
, errno
, "%s", tnm
);
231 if (!S_ISREG(st
->st_mode
) || 1 != st
->st_nlink
)
232 errx(EX_CANTCREAT
, "%s: invalid attributes", tnm
);
236 errc(EX_CANTCREAT
, errno
, "%s", corefname
);
237 } else if (-1 == fstat(tfd
, st
)) {
239 errx(EX_CANTCREAT
, "%s: fstat", corefname
);
240 } else if (S_ISBLK(st
->st_mode
) || S_ISCHR(st
->st_mode
)) {
242 * Write dump to a device, no rename!
248 errc(EX_CANTCREAT
, EEXIST
, "%s", corefname
);
253 closeout(int fd
, int ecode
, char *corefname
, char *coretname
, const struct stat
*st
)
255 if (0 != ecode
&& !opt
->preserve
&& S_ISREG(st
->st_mode
))
256 ftruncate(fd
, 0); // limit large file clutter
257 if (0 == ecode
&& S_ISREG(st
->st_mode
))
258 fchmod(fd
, 0400); // protect core files
259 if (-1 == close(fd
)) {
260 warnc(errno
, "%s: close", coretname
? coretname
: corefname
);
263 if (NULL
!= coretname
) {
264 if (0 == ecode
&& -1 == rename(coretname
, corefname
)) {
265 warnc(errno
, "cannot rename %s to %s", coretname
, corefname
);
276 const struct options
*opt
;
278 static const size_t oneK
= 1024;
279 static const size_t oneM
= oneK
* oneK
;
281 #define LARGEST_CHUNKSIZE INT32_MAX
282 #define DEFAULT_COMPRESSION_CHUNKSIZE (16 * oneM)
283 #define DEFAULT_NC_THRESHOLD (17 * oneK)
285 static struct options options
= {
296 .calgorithm
= COMPRESSION_LZFSE
,
297 .ncthresh
= DEFAULT_NC_THRESHOLD
,
302 gcore_main(int argc
, char *const *argv
)
305 #define ZOPT_CHSIZE (ZOPT_ALG + 1)
307 static char *const zoptkeys
[] = {
308 [ZOPT_ALG
] = "algorithm",
309 [ZOPT_CHSIZE
] = "chunksize",
313 err_set_exit_b(^(int eval
) {
314 if (EX_USAGE
== eval
) {
316 "usage:\t%s [-s] [-v] [[-o file] | [-c pathfmt ]] [-b size] "
322 "[-Z compression-options] "
328 fprintf(stderr
, "where compression-options:\n");
329 const char zvalfmt
[] = "\t%s=%s\t\t%s\n";
330 fprintf(stderr
, zvalfmt
, zoptkeys
[ZOPT_ALG
], "alg",
331 "set compression algorithm");
332 fprintf(stderr
, zvalfmt
, zoptkeys
[ZOPT_CHSIZE
], "size",
333 "set compression chunksize, Mib");
338 char *corefmt
= NULL
;
339 char *corefname
= NULL
;
344 while ((c
= getopt(argc
, argv
, "vdsxCFZ:o:c:b:t:")) != -1) {
350 case 's': /* FreeBSD compat: stop while gathering */
353 case 'o': /* Linux (& SunOS) compat: basic name */
354 corefname
= strdup(optarg
);
356 case 'c': /* FreeBSD compat: basic name */
357 /* (also allows pattern-based naming) */
358 corefmt
= strdup(optarg
);
361 case 'b': /* bound the size of the core file */
362 if (NULL
!= optarg
) {
363 off_t bsize
= atoi(optarg
) * oneM
;
365 options
.sizebound
= bsize
;
367 errx(EX_USAGE
, "invalid bound");
369 errx(EX_USAGE
, "no bound specified");
371 case 'v': /* verbose output */
376 * dev and debugging help
379 case 'd': /* debugging */
386 * Remaining options are experimental and/or
387 * affect the content of the core file
389 case 'x': /* write extended format (small) core files */
391 options
.chunksize
= DEFAULT_COMPRESSION_CHUNKSIZE
;
393 case 'C': /* forcibly corpsify rather than suspend */
396 case 'Z': /* control compression options */
398 * Only LZFSE and LZ4 seem practical.
399 * (Default to LZ4 compression when the
400 * process is suspended, LZFSE when corpsed?)
402 if (0 == options
.extended
)
403 errx(EX_USAGE
, "illegal flag combination");
408 switch (getsubopt(&sopts
, zoptkeys
, &value
)) {
409 case ZOPT_ALG
: /* change the algorithm */
411 errx(EX_USAGE
, "missing algorithm for "
414 if (strcmp(value
, "lz4") == 0)
415 options
.calgorithm
= COMPRESSION_LZ4
;
416 else if (strcmp(value
, "zlib") == 0)
417 options
.calgorithm
= COMPRESSION_ZLIB
;
418 else if (strcmp(value
, "lzma") == 0)
419 options
.calgorithm
= COMPRESSION_LZMA
;
420 else if (strcmp(value
, "lzfse") == 0)
421 options
.calgorithm
= COMPRESSION_LZFSE
;
423 errx(EX_USAGE
, "unknown algorithm '%s'"
425 value
, zoptkeys
[ZOPT_ALG
]);
427 case ZOPT_CHSIZE
: /* set the chunksize */
429 errx(EX_USAGE
, "no value specified for "
431 zoptkeys
[ZOPT_CHSIZE
]);
432 if ((chsize
= atoi(value
)) < 1)
433 errx(EX_USAGE
, "chunksize %lu too small", chsize
);
434 if (chsize
> (LARGEST_CHUNKSIZE
/ oneM
))
435 errx(EX_USAGE
, "chunksize %lu too large", chsize
);
436 options
.chunksize
= chsize
* oneM
;
440 errx(EX_USAGE
, "illegal suboption '%s'",
443 errx(EX_USAGE
, "missing suboption");
447 case 't': /* set the F_NOCACHE threshold */
448 if (NULL
!= optarg
) {
449 size_t tsize
= atoi(optarg
) * oneK
;
451 options
.ncthresh
= tsize
;
453 errx(EX_USAGE
, "invalid nc threshold");
455 errx(EX_USAGE
, "no threshold specified");
457 case 'F': /* maximize filerefs */
458 options
.allfilerefs
++;
461 errx(EX_USAGE
, "unknown flag");
466 errx(EX_USAGE
, "no pid specified");
468 errx(EX_USAGE
, "too many arguments");
471 if (NULL
!= corefname
&& NULL
!= corefmt
)
472 errx(EX_USAGE
, "specify only one of -o and -c");
473 if (!opt
->extended
&& opt
->allfilerefs
)
474 errx(EX_USAGE
, "unknown flag");
478 if (opt
->ncthresh
< ((vm_offset_t
)1 << pageshift_host
))
479 errx(EX_USAGE
, "threshold %lu less than host pagesize", opt
->ncthresh
);
481 const pid_t apid
= atoi(argv
[optind
]);
483 mach_port_t corpse
= MACH_PORT_NULL
;
487 /* look for corpse - dead or alive */
488 mach_port_array_t parray
= NULL
;
489 mach_msg_type_number_t pcount
= 0;
490 ret
= mach_ports_lookup(mach_task_self(), &parray
, &pcount
);
491 if (KERN_SUCCESS
== ret
&& pcount
> 0) {
492 task_t tcorpse
= parray
[0];
493 mig_deallocate((vm_address_t
)parray
, pcount
* sizeof (*parray
));
495 ret
= pid_for_task(tcorpse
, &tpid
);
496 if (KERN_SUCCESS
== ret
&& tpid
!= getpid()) {
503 if (pid
< 1 || getpid() == pid
)
504 errx(EX_DATAERR
, "invalid pid: %d", pid
);
506 if (0 == apid
&& MACH_PORT_NULL
== corpse
)
507 errx(EX_DATAERR
, "missing or bad corpse from parent");
509 task_t task
= TASK_NULL
;
510 const struct proc_bsdinfo
*pbi
= NULL
;
511 const int rc
= kill(pid
, 0);
514 /* process or corpse that may respond to signals */
515 pbi
= get_bsdinfo(pid
);
518 if (rc
== 0 && pbi
!= NULL
) {
519 /* process or corpse that responds to signals */
521 /* make our data model match the data model of the target */
522 if (-1 == reexec_to_match_lp64ness(pbi
->pbi_flags
& PROC_FLAG_LP64
))
523 errc(1, errno
, "cannot match data model of %d", pid
);
525 if (!proc_same_data_model(pbi
))
526 errx(EX_OSERR
, "cannot match data model of %d", pid
);
528 if (pbi
->pbi_ruid
!= pbi
->pbi_svuid
||
529 pbi
->pbi_rgid
!= pbi
->pbi_svgid
)
530 errx(EX_NOPERM
, "pid %d - not dumping a set-id process", pid
);
532 if (NULL
== corefname
)
533 corefname
= make_gcore_path(&corefmt
, pbi
->pbi_pid
, pbi
->pbi_uid
, pbi
->pbi_name
[0] ? pbi
->pbi_name
: pbi
->pbi_comm
);
535 if (MACH_PORT_NULL
== corpse
) {
536 ret
= task_for_pid(mach_task_self(), pid
, &task
);
537 if (KERN_SUCCESS
!= ret
) {
538 if (KERN_FAILURE
== ret
)
539 errx(EX_NOPERM
, "insufficient privilege");
541 errx(EX_NOPERM
, "task_for_pid: %s", mach_error_string(ret
));
546 * Have either the corpse port or the task port so adopt the
547 * credentials of the target process, *before* opening the
548 * core file, and analyzing the address space.
550 * If we are unable to match the target credentials, bail out.
552 change_credentials(pbi
->pbi_uid
, pbi
->pbi_gid
);
554 if (MACH_PORT_NULL
== corpse
) {
556 errx(EX_OSERR
, "cannot get process info for %d", pid
);
560 errc(EX_DATAERR
, errno
, "no process with pid %d", pid
);
562 errc(EX_DATAERR
, errno
, "pid %d", pid
);
565 /* a corpse with no live process backing it */
567 assert(0 == apid
&& TASK_NULL
== task
);
569 task_flags_info_data_t tfid
;
570 mach_msg_type_number_t count
= TASK_FLAGS_INFO_COUNT
;
571 ret
= task_info(corpse
, TASK_FLAGS_INFO
, (task_info_t
)&tfid
, &count
);
572 if (KERN_SUCCESS
!= ret
)
573 err_mach(ret
, NULL
, "task_info");
574 if (!task_same_data_model(&tfid
))
575 errx(EX_OSERR
, "data model mismatch for target corpse");
577 if (opt
->suspend
|| opt
->corpsify
)
578 errx(EX_USAGE
, "cannot use -s or -C option with a corpse");
580 errx(EX_USAGE
, "cannot use -c with a corpse");
581 if (NULL
== corefname
)
582 corefname
= make_gcore_path(&corefmt
, pid
, -2, "corpse");
585 * Only have a corpse, thus no process credentials.
588 change_credentials(-2, -2);
592 char *coretname
= NULL
;
593 const int fd
= openout(corefname
, &coretname
, &cst
);
596 printf("Dumping core ");
597 if (OPTIONS_DEBUG(opt
, 1)) {
598 printf("(%s", opt
->extended
? "extended" : "vanilla");
599 if (0 != opt
->sizebound
) {
601 printf(", <= %s", str_hsize(hstr
, opt
->sizebound
));
605 printf("for pid %d to %s\n", pid
, corefname
);
610 if (MACH_PORT_NULL
== corpse
) {
611 assert(TASK_NULL
!= task
);
614 * The "traditional" way to capture a consistent core dump is to
615 * suspend the process while examining it and writing it out.
616 * Yet suspending a large process for a long time can have
617 * unpleasant side-effects. Alternatively dumping from the live
618 * process can lead to an inconsistent state in the core file.
620 * Instead we can ask xnu to create a 'corpse' - the process is transiently
621 * suspended while a COW snapshot of the address space is constructed
622 * in the kernel and dump from that. This vastly reduces the suspend
623 * time, but it is more resource hungry and thus may fail.
625 * The -s flag (opt->suspend) causes a task_suspend/task_resume
626 * The -C flag (opt->corpse) causes a corpse be taken, exiting if that fails.
627 * Both flags can be specified, in which case corpse errors are ignored.
629 * With no flags, we imitate traditional behavior though more
630 * efficiently: we try to take a corpse-based dump, in the event that
631 * fails, dump the live process.
634 int trycorpse
= 1; /* default: use corpses */
635 int badcorpse_is_fatal
= 1; /* default: failure to create is an error */
637 if (!opt
->suspend
&& !opt
->corpsify
) {
638 /* try a corpse dump, else dump the live process */
639 badcorpse_is_fatal
= 0;
640 } else if (opt
->suspend
) {
641 trycorpse
= opt
->corpsify
;
642 /* if suspended anyway, ignore corpse-creation errors */
643 badcorpse_is_fatal
= 0;
651 * Create a corpse from the image before dumping it
653 ret
= task_generate_corpse(task
, &corpse
);
656 if (OPTIONS_DEBUG(opt
, 1))
657 printf("Corpse generated on port %x, task %x\n",
659 ecode
= coredump(corpse
, fd
, pbi
);
660 mach_port_deallocate(mach_task_self(), corpse
);
663 if (badcorpse_is_fatal
|| opt
->verbose
) {
664 warnx("failed to snapshot pid %d: %s\n",
665 pid
, mach_error_string(ret
));
666 if (badcorpse_is_fatal
) {
667 ecode
= KERN_RESOURCE_SHORTAGE
== ret
? EX_TEMPFAIL
: EX_OSERR
;
671 ecode
= coredump(task
, fd
, pbi
);
676 * Examine the task directly
678 ecode
= coredump(task
, fd
, pbi
);
686 * Handed a corpse by our parent.
688 ecode
= coredump(corpse
, fd
, pbi
);
689 mach_port_deallocate(mach_task_self(), corpse
);
692 ecode
= closeout(fd
, ecode
, corefname
, coretname
, &cst
);
694 errx(ecode
, "failed to dump core for pid %d", pid
);
698 #if defined(CONFIG_GCORE_FREF) || defined(CONFIG_GCORE_MAP) || defined(GCONFIG_GCORE_CONV)
701 getcorefd(const char *infile
)
703 const int fd
= open(infile
, O_RDONLY
| O_CLOEXEC
);
705 errc(EX_DATAERR
, errno
, "cannot open %s", infile
);
707 struct mach_header mh
;
708 if (-1 == pread(fd
, &mh
, sizeof (mh
), 0))
709 errc(EX_OSERR
, errno
, "cannot read mach header from %s", infile
);
711 static const char cant_match_data_model
[] = "cannot match the data model of %s";
713 if (-1 == reexec_to_match_lp64ness(MH_MAGIC_64
== mh
.magic
))
714 errc(1, errno
, cant_match_data_model
, infile
);
716 if (NATIVE_MH_MAGIC
!= mh
.magic
)
717 errx(EX_OSERR
, cant_match_data_model
, infile
);
718 if (MH_CORE
!= mh
.filetype
)
719 errx(EX_DATAERR
, "%s is not a mach core file", infile
);
725 #ifdef CONFIG_GCORE_FREF
728 gcore_fref_main(int argc
, char *argv
[])
730 err_set_exit_b(^(int eval
) {
731 if (EX_USAGE
== eval
) {
732 fprintf(stderr
, "usage:\t%s %s corefile\n", pgm
, argv
[1]);
736 errx(EX_USAGE
, "no input corefile");
738 errx(EX_USAGE
, "too many arguments");
740 return gcore_fref(getcorefd(argv
[2]));
743 #endif /* CONFIG_GCORE_FREF */
745 #ifdef CONFIG_GCORE_MAP
748 gcore_map_main(int argc
, char *argv
[])
750 err_set_exit_b(^(int eval
) {
751 if (EX_USAGE
== eval
) {
752 fprintf(stderr
, "usage:\t%s %s corefile\n", pgm
, argv
[1]);
756 errx(EX_USAGE
, "no input corefile");
758 errx(EX_USAGE
, "too many arguments");
760 return gcore_map(getcorefd(argv
[2]));
765 #ifdef CONFIG_GCORE_CONV
768 gcore_conv_main(int argc
, char *argv
[])
770 err_set_exit_b(^(int eval
) {
771 if (EX_USAGE
== eval
)
773 "usage:\t%s %s [-v] [-L searchpath] [-z] [-s] incore outcore\n", pgm
, argv
[1]);
776 char *searchpath
= NULL
;
781 while ((c
= getopt(argc
, argv
, "dzvL:s")) != -1) {
784 * likely documented options
787 searchpath
= strdup(optarg
);
796 options
.dsymforuuid
++;
799 * dev and debugging help
809 errx(EX_USAGE
, "unknown flag");
813 errx(EX_USAGE
, "no input corefile");
814 if (optind
== argc
- 1)
815 errx(EX_USAGE
, "no output corefile");
816 if (optind
< argc
- 2)
817 errx(EX_USAGE
, "too many arguments");
819 const char *incore
= argv
[optind
];
820 char *corefname
= strdup(argv
[optind
+1]);
826 if (opt
->ncthresh
< ((vm_offset_t
)1 << pageshift_host
))
827 errx(EX_USAGE
, "threshold %lu less than host pagesize", opt
->ncthresh
);
829 const int infd
= getcorefd(incore
);
831 char *coretname
= NULL
;
832 const int fd
= openout(corefname
, &coretname
, &cst
);
833 int ecode
= gcore_conv(infd
, searchpath
, zf
, fd
);
834 ecode
= closeout(fd
, ecode
, corefname
, coretname
, &cst
);
836 errx(ecode
, "failed to convert core file successfully");
842 main(int argc
, char *argv
[])
844 if (NULL
== (pgm
= strrchr(*argv
, '/')))
848 #ifdef CONFIG_GCORE_FREF
849 if (argc
> 1 && 0 == strcmp(argv
[1], "fref")) {
850 return gcore_fref_main(argc
, argv
);
853 #ifdef CONFIG_GCORE_MAP
854 if (argc
> 1 && 0 == strcmp(argv
[1], "map")) {
855 return gcore_map_main(argc
, argv
);
858 #ifdef CONFIG_GCORE_CONV
859 if (argc
> 1 && 0 == strcmp(argv
[1], "conv")) {
860 return gcore_conv_main(argc
, argv
);
863 return gcore_main(argc
, argv
);