libbacktrace: add Mach-O 64-bit FAT support

libbacktrace/:
	* macho.c (MACH_O_MH_MAGIC_FAT_64): Define.
	(MACH_O_MH_CIGAM_FAT_64): Define.
	(struct macho_fat_arch_64): Define.
	(macho_add_fat): Add and use is_64 parameter.
	(macho_add): Recognize 64-bit fat files.
This commit is contained in:
Ian Lance Taylor 2020-08-24 13:06:52 -07:00
parent 0d166f4a87
commit 74c176ca2f

View File

@ -75,7 +75,7 @@ struct macho_header_64
struct macho_header_fat struct macho_header_fat
{ {
uint32_t magic; /* Magic number (MACH_O_MH_MAGIC_FAT) */ uint32_t magic; /* Magic number (MACH_O_MH_(MAGIC|CIGAM)_FAT(_64)?) */
uint32_t nfat_arch; /* Number of components */ uint32_t nfat_arch; /* Number of components */
}; };
@ -85,6 +85,8 @@ struct macho_header_fat
#define MACH_O_MH_MAGIC_64 0xfeedfacf #define MACH_O_MH_MAGIC_64 0xfeedfacf
#define MACH_O_MH_MAGIC_FAT 0xcafebabe #define MACH_O_MH_MAGIC_FAT 0xcafebabe
#define MACH_O_MH_CIGAM_FAT 0xbebafeca #define MACH_O_MH_CIGAM_FAT 0xbebafeca
#define MACH_O_MH_MAGIC_FAT_64 0xcafebabf
#define MACH_O_MH_CIGAM_FAT_64 0xbfbafeca
/* Value for the header filetype field. */ /* Value for the header filetype field. */
@ -105,6 +107,20 @@ struct macho_fat_arch
uint32_t align; /* Alignment of this entry */ uint32_t align; /* Alignment of this entry */
}; };
/* A component of a 64-bit fat file. This is used if the magic field
is MAGIC_FAT_64. This is only used when some file size or file
offset is too large to represent in the 32-bit format. */
struct macho_fat_arch_64
{
uint32_t cputype; /* CPU type */
uint32_t cpusubtype; /* CPU subtype */
uint64_t offset; /* File offset of this entry */
uint64_t size; /* Size of this entry */
uint32_t align; /* Alignment of this entry */
uint32_t reserved; /* Reserved */
};
/* Values for the fat_arch cputype field (and the header cputype /* Values for the fat_arch cputype field (and the header cputype
field). */ field). */
@ -740,14 +756,14 @@ static int
macho_add_fat (struct backtrace_state *state, const char *filename, macho_add_fat (struct backtrace_state *state, const char *filename,
int descriptor, int swapped, off_t offset, int descriptor, int swapped, off_t offset,
const unsigned char *match_uuid, uintptr_t base_address, const unsigned char *match_uuid, uintptr_t base_address,
int skip_symtab, uint32_t nfat_arch, int skip_symtab, uint32_t nfat_arch, int is_64,
backtrace_error_callback error_callback, void *data, backtrace_error_callback error_callback, void *data,
fileline *fileline_fn, int *found_sym) fileline *fileline_fn, int *found_sym)
{ {
int arch_view_valid; int arch_view_valid;
unsigned int cputype; unsigned int cputype;
size_t arch_size;
struct backtrace_view arch_view; struct backtrace_view arch_view;
size_t archoffset;
unsigned int i; unsigned int i;
arch_view_valid = 0; arch_view_valid = 0;
@ -765,21 +781,39 @@ macho_add_fat (struct backtrace_state *state, const char *filename,
goto fail; goto fail;
#endif #endif
if (is_64)
arch_size = sizeof (struct macho_fat_arch_64);
else
arch_size = sizeof (struct macho_fat_arch);
if (!backtrace_get_view (state, descriptor, offset, if (!backtrace_get_view (state, descriptor, offset,
nfat_arch * sizeof (struct macho_fat_arch), nfat_arch * arch_size,
error_callback, data, &arch_view)) error_callback, data, &arch_view))
goto fail; goto fail;
archoffset = 0;
for (i = 0; i < nfat_arch; ++i) for (i = 0; i < nfat_arch; ++i)
{ {
struct macho_fat_arch fat_arch; struct macho_fat_arch_64 fat_arch;
uint32_t fcputype; uint32_t fcputype;
if (is_64)
memcpy (&fat_arch, memcpy (&fat_arch,
((const char *) arch_view.data (const char *) arch_view.data + i * arch_size,
+ i * sizeof (struct macho_fat_arch)), arch_size);
sizeof fat_arch); else
{
struct macho_fat_arch fat_arch_32;
memcpy (&fat_arch_32,
(const char *) arch_view.data + i * arch_size,
arch_size);
fat_arch.cputype = fat_arch_32.cputype;
fat_arch.cpusubtype = fat_arch_32.cpusubtype;
fat_arch.offset = (uint64_t) fat_arch_32.offset;
fat_arch.size = (uint64_t) fat_arch_32.size;
fat_arch.align = fat_arch_32.align;
fat_arch.reserved = 0;
}
fcputype = fat_arch.cputype; fcputype = fat_arch.cputype;
if (swapped) if (swapped)
@ -787,19 +821,17 @@ macho_add_fat (struct backtrace_state *state, const char *filename,
if (fcputype == cputype) if (fcputype == cputype)
{ {
uint32_t foffset; uint64_t foffset;
/* FIXME: What about cpusubtype? */ /* FIXME: What about cpusubtype? */
foffset = fat_arch.offset; foffset = fat_arch.offset;
if (swapped) if (swapped)
foffset = __builtin_bswap32 (foffset); foffset = __builtin_bswap64 (foffset);
backtrace_release_view (state, &arch_view, error_callback, data); backtrace_release_view (state, &arch_view, error_callback, data);
return macho_add (state, filename, descriptor, foffset, match_uuid, return macho_add (state, filename, descriptor, foffset, match_uuid,
base_address, skip_symtab, error_callback, data, base_address, skip_symtab, error_callback, data,
fileline_fn, found_sym); fileline_fn, found_sym);
} }
archoffset += sizeof (struct macho_fat_arch);
} }
error_callback (data, "could not find executable in fat file", 0); error_callback (data, "could not find executable in fat file", 0);
@ -980,6 +1012,7 @@ macho_add (struct backtrace_state *state, const char *filename, int descriptor,
hdroffset = offset + sizeof (struct macho_header_64); hdroffset = offset + sizeof (struct macho_header_64);
break; break;
case MACH_O_MH_MAGIC_FAT: case MACH_O_MH_MAGIC_FAT:
case MACH_O_MH_MAGIC_FAT_64:
{ {
struct macho_header_fat fat_header; struct macho_header_fat fat_header;
@ -987,10 +1020,12 @@ macho_add (struct backtrace_state *state, const char *filename, int descriptor,
memcpy (&fat_header, &header, sizeof fat_header); memcpy (&fat_header, &header, sizeof fat_header);
return macho_add_fat (state, filename, descriptor, 0, hdroffset, return macho_add_fat (state, filename, descriptor, 0, hdroffset,
match_uuid, base_address, skip_symtab, match_uuid, base_address, skip_symtab,
fat_header.nfat_arch, error_callback, data, fat_header.nfat_arch,
fileline_fn, found_sym); header.magic == MACH_O_MH_MAGIC_FAT_64,
error_callback, data, fileline_fn, found_sym);
} }
case MACH_O_MH_CIGAM_FAT: case MACH_O_MH_CIGAM_FAT:
case MACH_O_MH_CIGAM_FAT_64:
{ {
struct macho_header_fat fat_header; struct macho_header_fat fat_header;
uint32_t nfat_arch; uint32_t nfat_arch;
@ -1000,8 +1035,9 @@ macho_add (struct backtrace_state *state, const char *filename, int descriptor,
nfat_arch = __builtin_bswap32 (fat_header.nfat_arch); nfat_arch = __builtin_bswap32 (fat_header.nfat_arch);
return macho_add_fat (state, filename, descriptor, 1, hdroffset, return macho_add_fat (state, filename, descriptor, 1, hdroffset,
match_uuid, base_address, skip_symtab, match_uuid, base_address, skip_symtab,
nfat_arch, error_callback, data, nfat_arch,
fileline_fn, found_sym); header.magic == MACH_O_MH_CIGAM_FAT_64,
error_callback, data, fileline_fn, found_sym);
} }
default: default:
error_callback (data, "executable file is not in Mach-O format", 0); error_callback (data, "executable file is not in Mach-O format", 0);