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:
parent
0d166f4a87
commit
74c176ca2f
@ -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;
|
||||||
|
|
||||||
memcpy (&fat_arch,
|
if (is_64)
|
||||||
((const char *) arch_view.data
|
memcpy (&fat_arch,
|
||||||
+ i * sizeof (struct macho_fat_arch)),
|
(const char *) arch_view.data + i * arch_size,
|
||||||
sizeof fat_arch);
|
arch_size);
|
||||||
|
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);
|
||||||
|
Loading…
Reference in New Issue
Block a user