2009-03-15 18:02:36 +00:00
|
|
|
/* Copyright (c) 2008-2009, Avian Contributors
|
2008-02-19 18:06:52 +00:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2008-01-23 16:56:25 +00:00
|
|
|
#include "stdint.h"
|
|
|
|
#include "stdio.h"
|
|
|
|
#include "string.h"
|
|
|
|
|
|
|
|
#include "sys/stat.h"
|
|
|
|
#include "sys/mman.h"
|
|
|
|
#include "fcntl.h"
|
|
|
|
#include "unistd.h"
|
|
|
|
|
|
|
|
#include "mach-o/loader.h"
|
|
|
|
#include "mach-o/nlist.h"
|
|
|
|
|
2008-06-04 23:09:09 +00:00
|
|
|
|
|
|
|
|
2008-01-23 16:56:25 +00:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
inline unsigned
|
|
|
|
pad(unsigned n)
|
|
|
|
{
|
|
|
|
return (n + (4 - 1)) & ~(4 - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2008-06-04 23:09:09 +00:00
|
|
|
writeObject(const char* architecture,
|
|
|
|
FILE* out, const uint8_t* data, unsigned size,
|
2008-12-19 02:52:03 +00:00
|
|
|
const char* segmentName, const char* sectionName,
|
2008-03-31 03:43:43 +00:00
|
|
|
const char* startName, const char* endName)
|
2008-01-23 16:56:25 +00:00
|
|
|
{
|
2008-03-31 03:43:43 +00:00
|
|
|
unsigned startNameLength = strlen(startName) + 1;
|
|
|
|
unsigned endNameLength = strlen(endName) + 1;
|
2008-01-23 16:56:25 +00:00
|
|
|
|
2008-06-04 23:09:09 +00:00
|
|
|
cpu_type_t cpuType;
|
|
|
|
cpu_subtype_t cpuSubtype;
|
|
|
|
if (strcmp(architecture, "x86") == 0) {
|
|
|
|
cpuType = CPU_TYPE_I386;
|
|
|
|
cpuSubtype = CPU_SUBTYPE_I386_ALL;
|
|
|
|
} else if (strcmp(architecture, "powerpc") == 0) {
|
|
|
|
cpuType = CPU_TYPE_POWERPC;
|
|
|
|
cpuSubtype = CPU_SUBTYPE_POWERPC_ALL;
|
|
|
|
}
|
|
|
|
|
2008-01-23 16:56:25 +00:00
|
|
|
mach_header header = {
|
|
|
|
MH_MAGIC, // magic
|
2008-06-04 23:09:09 +00:00
|
|
|
cpuType,
|
|
|
|
cpuSubtype,
|
2008-01-23 16:56:25 +00:00
|
|
|
MH_OBJECT, // filetype,
|
|
|
|
2, // ncmds
|
|
|
|
sizeof(segment_command)
|
2008-03-31 04:23:51 +00:00
|
|
|
+ sizeof(section)
|
2008-01-23 16:56:25 +00:00
|
|
|
+ sizeof(symtab_command), // sizeofcmds
|
|
|
|
0 // flags
|
|
|
|
};
|
|
|
|
|
|
|
|
segment_command segment = {
|
|
|
|
LC_SEGMENT, // cmd
|
|
|
|
sizeof(segment_command) + sizeof(section), // cmdsize
|
2008-12-19 02:52:03 +00:00
|
|
|
"", // segname
|
2008-01-23 16:56:25 +00:00
|
|
|
0, // vmaddr
|
|
|
|
pad(size), // vmsize
|
|
|
|
sizeof(mach_header)
|
|
|
|
+ sizeof(segment_command)
|
2008-03-31 04:23:51 +00:00
|
|
|
+ sizeof(section)
|
2008-01-23 16:56:25 +00:00
|
|
|
+ sizeof(symtab_command), // fileoff
|
|
|
|
pad(size), // filesize
|
|
|
|
7, // maxprot
|
|
|
|
7, // initprot
|
2008-03-31 04:23:51 +00:00
|
|
|
1, // nsects
|
2008-01-23 16:56:25 +00:00
|
|
|
0 // flags
|
|
|
|
};
|
|
|
|
|
2008-12-19 02:52:03 +00:00
|
|
|
strncpy(segment.segname, segmentName, sizeof(segment.segname));
|
|
|
|
|
2008-03-31 04:23:51 +00:00
|
|
|
section sect = {
|
2008-12-19 02:52:03 +00:00
|
|
|
"", // sectname
|
|
|
|
"", // segname
|
2008-01-23 16:56:25 +00:00
|
|
|
0, // addr
|
|
|
|
pad(size), // size
|
|
|
|
sizeof(mach_header)
|
|
|
|
+ sizeof(segment_command)
|
2008-03-31 04:23:51 +00:00
|
|
|
+ sizeof(section)
|
2008-01-23 16:56:25 +00:00
|
|
|
+ sizeof(symtab_command), // offset
|
|
|
|
0, // align
|
|
|
|
0, // reloff
|
|
|
|
0, // nreloc
|
|
|
|
S_REGULAR, // flags
|
|
|
|
0, // reserved1
|
|
|
|
0, // reserved2
|
|
|
|
};
|
|
|
|
|
2008-12-19 02:52:03 +00:00
|
|
|
strncpy(sect.segname, segmentName, sizeof(sect.segname));
|
|
|
|
strncpy(sect.sectname, sectionName, sizeof(sect.sectname));
|
|
|
|
|
2008-01-23 16:56:25 +00:00
|
|
|
symtab_command symbolTable = {
|
|
|
|
LC_SYMTAB, // cmd
|
|
|
|
sizeof(symtab_command), // cmdsize
|
|
|
|
sizeof(mach_header)
|
|
|
|
+ sizeof(segment_command)
|
2008-03-31 04:23:51 +00:00
|
|
|
+ sizeof(section)
|
2008-01-23 16:56:25 +00:00
|
|
|
+ sizeof(symtab_command)
|
|
|
|
+ pad(size), // symoff
|
|
|
|
2, // nsyms
|
|
|
|
sizeof(mach_header)
|
|
|
|
+ sizeof(segment_command)
|
|
|
|
+ sizeof(section)
|
|
|
|
+ sizeof(symtab_command)
|
|
|
|
+ pad(size)
|
|
|
|
+ (sizeof(struct nlist) * 2), // stroff
|
2008-03-31 03:43:43 +00:00
|
|
|
1 + startNameLength + endNameLength, // strsize
|
2008-01-23 16:56:25 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct nlist symbolList[] = {
|
|
|
|
{
|
|
|
|
reinterpret_cast<char*>(1), // n_un
|
|
|
|
N_SECT | N_EXT, // n_type
|
|
|
|
1, // n_sect
|
|
|
|
0, // n_desc
|
|
|
|
0 // n_value
|
|
|
|
},
|
|
|
|
{
|
2008-03-31 03:43:43 +00:00
|
|
|
reinterpret_cast<char*>(1 + startNameLength), // n_un
|
|
|
|
N_SECT | N_EXT, // n_type
|
2008-03-31 04:23:51 +00:00
|
|
|
1, // n_sect
|
2008-01-23 16:56:25 +00:00
|
|
|
0, // n_desc
|
2008-03-31 04:23:51 +00:00
|
|
|
size // n_value
|
2008-01-23 16:56:25 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
fwrite(&header, 1, sizeof(header), out);
|
|
|
|
fwrite(&segment, 1, sizeof(segment), out);
|
2008-03-31 04:23:51 +00:00
|
|
|
fwrite(§, 1, sizeof(sect), out);
|
2008-01-23 16:56:25 +00:00
|
|
|
fwrite(&symbolTable, 1, sizeof(symbolTable), out);
|
|
|
|
|
|
|
|
fwrite(data, 1, size, out);
|
|
|
|
for (unsigned i = 0; i < pad(size) - size; ++i) fputc(0, out);
|
|
|
|
|
|
|
|
fwrite(&symbolList, 1, sizeof(symbolList), out);
|
|
|
|
|
|
|
|
fputc(0, out);
|
2008-03-31 03:43:43 +00:00
|
|
|
fwrite(startName, 1, startNameLength, out);
|
|
|
|
fwrite(endName, 1, endNameLength, out);
|
2008-01-23 16:56:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
int
|
|
|
|
main(int argc, const char** argv)
|
|
|
|
{
|
2009-02-09 23:22:01 +00:00
|
|
|
if (argc != 7) {
|
2008-01-23 16:56:25 +00:00
|
|
|
fprintf(stderr,
|
2009-02-09 23:22:01 +00:00
|
|
|
"usage: %s <architecture> <input file> <segment name> "
|
|
|
|
"<section name> <start symbol name> <end symbol name>\n",
|
2008-01-23 16:56:25 +00:00
|
|
|
argv[0]);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t* data = 0;
|
|
|
|
unsigned size;
|
2008-06-04 23:09:09 +00:00
|
|
|
int fd = open(argv[2], O_RDONLY);
|
2008-01-23 16:56:25 +00:00
|
|
|
if (fd != -1) {
|
|
|
|
struct stat s;
|
|
|
|
int r = fstat(fd, &s);
|
|
|
|
if (r != -1) {
|
|
|
|
data = static_cast<uint8_t*>
|
|
|
|
(mmap(0, s.st_size, PROT_READ, MAP_PRIVATE, fd, 0));
|
|
|
|
size = s.st_size;
|
|
|
|
}
|
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data) {
|
2009-02-09 23:22:01 +00:00
|
|
|
writeObject
|
|
|
|
(argv[1], stdout, data, size, argv[3], argv[4], argv[5], argv[6]);
|
|
|
|
|
2008-01-23 16:56:25 +00:00
|
|
|
munmap(data, size);
|
2009-02-09 23:22:01 +00:00
|
|
|
|
2008-01-23 16:56:25 +00:00
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
perror(argv[0]);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|