2017-03-04 14:07:30 -05:00
|
|
|
/*
|
|
|
|
* convert a file to an applesingle file.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <system_error>
|
|
|
|
#include <string>
|
2017-03-04 21:39:04 -05:00
|
|
|
#include <vector>
|
2017-08-10 11:55:25 -04:00
|
|
|
#include <memory>
|
2017-03-04 14:07:30 -05:00
|
|
|
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <fcntl.h>
|
2017-03-04 21:39:04 -05:00
|
|
|
#include <sys/stat.h>
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
#include "win.h"
|
|
|
|
#else
|
|
|
|
#include <err.h>
|
|
|
|
#include <sysexits.h>
|
|
|
|
#endif
|
2017-03-04 14:07:30 -05:00
|
|
|
|
|
|
|
#include "mapped_file.h"
|
2017-08-09 23:14:15 -04:00
|
|
|
|
|
|
|
#include <afp/finder_info.h>
|
|
|
|
#include <afp/resource_fork.h>
|
2017-03-04 14:07:30 -05:00
|
|
|
|
|
|
|
#include "applefile.h"
|
|
|
|
|
2017-07-23 15:29:28 -04:00
|
|
|
#ifndef O_BINARY
|
|
|
|
#define O_BINARY 0
|
|
|
|
#endif
|
2017-03-04 14:07:30 -05:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void usage() {
|
|
|
|
fputs("Usage: applesingle [-hv] [-o outfile] file ...\n", stderr);
|
|
|
|
exit(EX_USAGE);
|
|
|
|
}
|
|
|
|
|
|
|
|
void help() {
|
|
|
|
fputs(
|
|
|
|
"Usage: applesingle [-hv] [-o outfile] file ...\n"
|
|
|
|
"\n"
|
|
|
|
" -o Specify outfile name\n"
|
|
|
|
" -h Display help\n"
|
|
|
|
" -v Be verbose\n",
|
|
|
|
stdout);
|
|
|
|
|
|
|
|
exit(EX_OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool _v = false;
|
|
|
|
int _rv = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* check if a file is apple single or apple double format (or neither). */
|
|
|
|
uint32_t classify(const mapped_file &mf) {
|
|
|
|
if (mf.size() < sizeof(ASHeader)) return 0;
|
|
|
|
|
|
|
|
const ASHeader *header = (const ASHeader *)mf.data();
|
|
|
|
if (header->magicNum == htonl(APPLESINGLE_MAGIC)) return APPLESINGLE_MAGIC;
|
|
|
|
if (header->magicNum == htonl(APPLEDOUBLE_MAGIC)) return APPLEDOUBLE_MAGIC;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* cases to consider
|
|
|
|
* 1. single file w/ fork/finder info
|
|
|
|
* 2. single file + appledouble file
|
|
|
|
* 3. single file w/o fork/finder info
|
|
|
|
* 4. file is already apple single / apple double format.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void one_file(const std::string &infile, const std::string &outfile) {
|
|
|
|
|
|
|
|
// 1. check if apple single/apple double
|
|
|
|
mapped_file mf;
|
|
|
|
ASHeader head;
|
|
|
|
ASEntry e;
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
mf = mapped_file(infile);
|
|
|
|
} catch(std::exception &ex) {
|
|
|
|
warnx("%s: %s", infile.c_str(), ex.what());
|
|
|
|
_rv = 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t fmt = classify(mf);
|
|
|
|
if (fmt == APPLESINGLE_MAGIC) {
|
|
|
|
warnx("%s: File is apple single format.", infile.c_str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (fmt == APPLEDOUBLE_MAGIC) {
|
|
|
|
warnx("%s: File is apple double format.", infile.c_str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// flag for preferred data source?
|
|
|
|
// 1. check for native fork
|
|
|
|
|
|
|
|
// 2. check for apple double data.
|
|
|
|
|
|
|
|
int count = 2; // name + data fork
|
|
|
|
|
|
|
|
std::error_code ec;
|
|
|
|
|
2017-08-09 23:14:15 -04:00
|
|
|
afp::finder_info fi;
|
|
|
|
afp::resource_fork rf;
|
|
|
|
|
|
|
|
size_t rfork_size = 0;
|
|
|
|
std::unique_ptr<uint8_t[]> rfork_data;
|
2017-07-23 15:29:28 -04:00
|
|
|
|
|
|
|
bool fi_ok = fi.open(infile, ec);
|
2017-03-04 14:07:30 -05:00
|
|
|
|
2017-08-09 23:14:15 -04:00
|
|
|
bool rf_ok = rf.open(infile, afp::resource_fork::read_only, ec);
|
2017-03-04 14:07:30 -05:00
|
|
|
|
2017-08-09 23:14:15 -04:00
|
|
|
if (rf_ok) {
|
|
|
|
rfork_size = rf.size(ec);
|
|
|
|
if (rfork_size) {
|
2017-03-04 14:07:30 -05:00
|
|
|
|
2017-08-09 23:14:15 -04:00
|
|
|
rfork_data.reset(new uint8_t[rfork_size]);
|
|
|
|
rfork_size = rf.read(rfork_data.get(), rfork_size, ec);
|
|
|
|
|
|
|
|
if (ec) {
|
|
|
|
warnx("%s resource fork: %s", infile.c_str(), ec.message().c_str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rf.close();
|
2017-03-04 14:07:30 -05:00
|
|
|
}
|
2017-08-09 23:14:15 -04:00
|
|
|
if (!rfork_size) rf_ok = false;
|
2017-03-04 14:07:30 -05:00
|
|
|
|
2017-08-09 23:14:15 -04:00
|
|
|
|
|
|
|
if (!fi_ok && !rf_ok) {
|
2017-03-04 14:07:30 -05:00
|
|
|
warnx("%s: File is not extended.", infile.c_str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-08-09 23:14:15 -04:00
|
|
|
if (rf_ok) count++;
|
2017-03-04 14:07:30 -05:00
|
|
|
if (fi_ok) count++;
|
|
|
|
|
|
|
|
|
2017-07-23 15:29:28 -04:00
|
|
|
int fd = open(outfile.c_str(), O_WRONLY | O_CREAT | O_BINARY, 0666);
|
2017-03-04 14:07:30 -05:00
|
|
|
memset(&head, 0, sizeof(head));
|
|
|
|
head.magicNum = htonl(APPLESINGLE_MAGIC);
|
|
|
|
head.versionNum = htonl(0x00020000);
|
|
|
|
head.numEntries = htons(count);
|
|
|
|
write(fd, &head, sizeof(head));
|
|
|
|
|
|
|
|
off_t offset = sizeof(ASHeader) + sizeof(ASEntry) * count;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 1 - name
|
|
|
|
e.entryID = htonl(AS_REALNAME);
|
|
|
|
e.entryOffset = htonl(offset);
|
|
|
|
e.entryLength = htonl(infile.length()); // todo -- basename it!
|
|
|
|
write(fd, &e, sizeof(e));
|
|
|
|
|
|
|
|
offset += infile.length();
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
// 2 - dates
|
|
|
|
e.entryID = htonl(AS_FILEDATES);
|
|
|
|
e.entryOffset = htonl(offset);
|
|
|
|
e.entryLength = htonl(sizeof(ASFileDates));
|
|
|
|
write(fd, &e, sizeof(e));
|
|
|
|
|
|
|
|
offset += sizeof(ASFileDates);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// 3 - finder info
|
|
|
|
if (fi_ok) {
|
|
|
|
e.entryID = htonl(AS_FINDERINFO);
|
|
|
|
e.entryOffset = htonl(offset);
|
|
|
|
e.entryLength = htonl(32);
|
|
|
|
write(fd, &e, sizeof(e));
|
|
|
|
|
|
|
|
offset += 32;
|
|
|
|
}
|
|
|
|
// 4 - data fork.
|
|
|
|
|
|
|
|
e.entryID = htonl(AS_DATA);
|
|
|
|
e.entryOffset = htonl(offset);
|
|
|
|
e.entryLength = htonl(mf.size());
|
|
|
|
write(fd, &e, sizeof(e));
|
|
|
|
|
|
|
|
offset += mf.size();
|
|
|
|
|
2017-08-09 23:14:15 -04:00
|
|
|
// 5 - resource fork
|
|
|
|
if (rf_ok) {
|
2017-03-04 14:07:30 -05:00
|
|
|
|
|
|
|
e.entryID = htonl(AS_RESOURCE);
|
|
|
|
e.entryOffset = htonl(offset);
|
2017-08-09 23:14:15 -04:00
|
|
|
e.entryLength = htonl(rfork_size);
|
2017-03-04 14:07:30 -05:00
|
|
|
write(fd, &e, sizeof(e));
|
|
|
|
|
2017-08-09 23:14:15 -04:00
|
|
|
offset += rfork_size;
|
2017-03-04 14:07:30 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// now write it..
|
|
|
|
|
|
|
|
// 1 - name
|
|
|
|
write(fd, infile.c_str(), infile.length());
|
|
|
|
|
|
|
|
// 2 - dates...
|
|
|
|
|
|
|
|
// 3 - finder info
|
|
|
|
if (fi_ok) {
|
2017-08-09 23:14:15 -04:00
|
|
|
write(fd, fi.data(), 32);
|
2017-03-04 14:07:30 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// 4 - data fork
|
|
|
|
write(fd, mf.data(), mf.size());
|
|
|
|
|
|
|
|
// 5 - resource fork?
|
2017-08-09 23:14:15 -04:00
|
|
|
if (rf_ok) {
|
|
|
|
write(fd, rfork_data.get(), rfork_size);
|
2017-03-04 14:07:30 -05:00
|
|
|
}
|
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char **argv) {
|
|
|
|
|
|
|
|
std::string _o;
|
|
|
|
|
|
|
|
int c;
|
|
|
|
|
|
|
|
while ((c = getopt(argc, argv, "o:v")) != -1) {
|
|
|
|
|
|
|
|
switch(c) {
|
|
|
|
case 'v': _v = true; break;
|
|
|
|
case 'h': help(); break;
|
|
|
|
case 'o': _o = optarg; break;
|
|
|
|
case ':':
|
|
|
|
case '?':
|
|
|
|
default:
|
|
|
|
usage();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
argv += optind;
|
|
|
|
argc -= optind;
|
|
|
|
|
|
|
|
if (!argc) usage();
|
|
|
|
|
|
|
|
for (int i = 0; i < argc; ++i) {
|
|
|
|
std::string s(argv[i]);
|
|
|
|
if (_o.empty()) _o = s + ".applesingle";
|
|
|
|
one_file(s, _o);
|
|
|
|
_o.clear();
|
|
|
|
}
|
|
|
|
return _rv;
|
2017-08-10 11:55:25 -04:00
|
|
|
}
|
|
|
|
|