macutils/macunpack/macbinary.c

538 lines
13 KiB
C

#include "macunpack.h"
#include "macbinary.h"
#include <stdlib.h>
#include <string.h>
#include "../fileio/kind.h"
#include "../fileio/machdr.h"
#include "../fileio/wrfile.h"
#include "../util/util.h"
#include "bin.h"
#include "dd.h"
#include "dir.h"
#include "globals.h"
#include "mcb.h"
#include "sit.h"
#include "stf.h"
#include "zmahdr.h"
#include "jdw.h"
#include "lzc.h"
#include "pit.h"
#include "dia.h"
#include "cpt.h"
#include "zma.h"
#include "lzh.h"
#ifdef ASQ
extern void asq();
#endif /* ASQ */
#ifdef ARC
extern void arc();
#endif /* ARC */
static void skip_file(int skip);
#ifdef SCAN
static void get_idf(int kind);
#endif /* SCAN */
#define Z (ZMAHDRS2 + 1)
static int info_given;
void
macbinary (void)
{
char header[INFOBYTES];
int c;
while(1) {
if((c = fgetc(infp)) == EOF) {
break;
}
(void)ungetc(c, infp);
if(fread(header, 1, Z, infp) != Z) {
(void)fprintf(stderr, "Can't read MacBinary header.\n");
#ifdef SCAN
do_error("macunpack: Can't read MacBinary header");
#endif /* SCAN */
exit(1);
}
#ifdef ZMA
if(!strncmp(header + 1, ZMAHDR, ZMAHDRS2)) {
/* Can distinguish zoom data forks only this way from macbinary */
if(verbose) {
(void)fprintf(stderr, "This is a \"Zoom\" archive.\n");
}
zma(header, (uint32_t)0);
exit(0);
}
#endif /* ZMA */
if(fread(header + Z, 1, INFOBYTES - Z, infp) != INFOBYTES - Z) {
(void)fprintf(stderr, "Can't read MacBinary header.\n");
#ifdef SCAN
do_error("macunpack: Can't read MacBinary header");
#endif /* SCAN */
exit(1);
}
if(verbose && !info_given) {
do_indent(indent);
(void)fprintf(stderr, "This is \"MacBinary\" input.\n");
info_given = 1;
}
if(header[I_NAMEOFF] & 0x80) {
dir(header);
continue;
}
in_data_size = get4(header + I_DLENOFF);
in_rsrc_size = get4(header + I_RLENOFF);
in_ds = (((in_data_size + 127) >> 7) << 7);
in_rs = (((in_rsrc_size + 127) >> 7) << 7);
ds_skip = in_ds - in_data_size;
rs_skip = in_rs - in_rsrc_size;
if(dir_skip != 0) {
skip_file(in_ds + in_rs);
continue;
}
#ifdef SCAN
if(header[I_NAMEOFF] == 0) {
get_idf((int)header[I_NAMEOFF + 1]);
skip_file(ds_skip + in_rs);
continue;
}
#endif /* SCAN */
header[I_NAMEOFF + 1 + header[I_NAMEOFF]] = 0;
#ifdef BIN
if(!strncmp(header + I_TYPEOFF, "TEXT", 4) &&
!strncmp(header + I_AUTHOFF, "BnHq", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"BinHex 5.0\" packed file.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, PACK_NAME);
#endif /* SCAN */
bin(header, in_data_size, 0);
skip_file(ds_skip + in_rs);
continue;
}
if(!strncmp(header + I_TYPEOFF, "TEXT", 4) &&
!strncmp(header + I_AUTHOFF, "GJBU", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"MacBinary 1.0\" packed file.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, PACK_NAME);
#endif /* SCAN */
bin(header, in_data_size, 0);
skip_file(ds_skip + in_rs);
continue;
}
/* Recognize only if creator is UMcp. UMCP uses ttxt as default. */
if(!strncmp(header + I_TYPEOFF, "TEXT", 4) &&
!strncmp(header + I_AUTHOFF, "UMcp", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr, "This is a \"UMCP\" packed file.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, PACK_NAME);
#endif /* SCAN */
bin(header, in_data_size, 1);
skip_file(ds_skip + in_rs);
continue;
}
#endif /* BIN */
#ifdef JDW
if(!strncmp(header + I_TYPEOFF, "Smal", 4) &&
!strncmp(header + I_AUTHOFF, "Jdw ", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"Compress It\" compressed file.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, PACK_NAME);
#endif /* SCAN */
jdw((uint32_t)in_data_size);
skip_file(ds_skip + in_rs);
continue;
}
#endif /* JDW */
#ifdef STF
if(!strncmp(header + I_TYPEOFF, "COMP", 4) &&
!strncmp(header + I_AUTHOFF, "STF ", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"ShrinkToFit\" compressed file.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, PACK_NAME);
#endif /* SCAN */
stf((uint32_t)in_data_size);
skip_file(ds_skip + in_rs);
continue;
}
#endif /* STF */
#ifdef LZC
if(!strncmp(header + I_TYPEOFF, "ZIVM", 4) &&
!strncmp(header + I_AUTHOFF, "LZIV", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"MacCompress(M)\" compressed file.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, PACK_NAME);
#endif /* SCAN */
lzc(header);
skip_file(ds_skip + in_rs);
continue;
}
if(!strncmp(header + I_TYPEOFF, "ZIVU", 4) &&
!strncmp(header + I_AUTHOFF, "LZIV", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"MacCompress(U)\" compressed file.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, PACK_NAME);
#endif /* SCAN */
lzc(header);
continue;
}
#endif /* LZC */
#ifdef ASQ
if(!strncmp(header + I_TYPEOFF, "ArCv", 4) &&
!strncmp(header + I_AUTHOFF, "TrAS", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"AutoSqueeze\" compressed file.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, PACK_NAME);
#endif /* SCAN */
lzh(0);
skip_file(ds_skip + in_rs);
continue;
}
#endif /* ASQ */
#ifdef ARC
if(!strncmp(header + I_TYPEOFF, "mArc", 4) &&
!strncmp(header + I_AUTHOFF, "arc*", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr, "This is a \"ArcMac\" archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
arc();
skip_file(ds_skip + in_rs);
continue;
}
if(!strncmp(header + I_TYPEOFF, "APPL", 4) &&
!strncmp(header + I_AUTHOFF, "arc@", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"ArcMac\" self extracting archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
arc();
skip_file(ds_skip + in_rs);
continue;
}
#endif /* ARC */
#ifdef PIT
if(!strncmp(header + I_TYPEOFF, "PIT ", 4) &&
!strncmp(header + I_AUTHOFF, "PIT ", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr, "This is a \"PackIt\" archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
pit();
skip_file(ds_skip + in_rs);
continue;
}
#endif /* PIT */
#ifdef SIT
if(!strncmp(header + I_TYPEOFF, "SIT!", 4) &&
!strncmp(header + I_AUTHOFF, "SIT!", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr, "This is a \"StuffIt\" archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
sit();
skip_file(ds_skip + in_rs);
continue;
}
if(!strncmp(header + I_TYPEOFF, "SITD", 4) &&
!strncmp(header + I_AUTHOFF, "SIT!", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"StuffIt Deluxe\" archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
sit();
skip_file(ds_skip + in_rs);
continue;
}
if(!strncmp(header + I_TYPEOFF, "APPL", 4) &&
!strncmp(header + I_AUTHOFF, "aust", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"StuffIt\" self extracting archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
sit();
skip_file(ds_skip + in_rs);
continue;
}
#endif /* SIT */
#ifdef DIA
if(!strncmp(header + I_TYPEOFF, "Pack", 4) &&
!strncmp(header + I_AUTHOFF, "Pack", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr, "This is a \"Diamond\" archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
dia((unsigned char *)header);
skip_file(ds_skip + in_rs);
continue;
}
if(!strncmp(header + I_TYPEOFF, "APPL", 4) &&
!strncmp(header + I_AUTHOFF, "Pack", 4) &&
in_data_size != 0) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"Diamond\" self extracting archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
dia((unsigned char *)header);
skip_file(ds_skip + in_rs);
continue;
}
#endif /* DIA */
#ifdef CPT
if(!strncmp(header + I_TYPEOFF, "PACT", 4) &&
!strncmp(header + I_AUTHOFF, "CPCT", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr, "This is a \"Compactor\" archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
cpt();
skip_file(ds_skip + in_rs);
continue;
}
if(!strncmp(header + I_TYPEOFF, "APPL", 4) &&
!strncmp(header + I_AUTHOFF, "EXTR", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"Compactor\" self extracting archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
cpt();
skip_file(ds_skip + in_rs);
continue;
}
#endif /* CPT */
#ifdef ZMA
if(!strncmp(header + I_TYPEOFF, "zooM", 4) &&
!strncmp(header + I_AUTHOFF, "zooM", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr, "This is a \"Zoom\" archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
zma((char *)NULL, (uint32_t)in_data_size);
skip_file(ds_skip + in_rs);
continue;
}
if(!strncmp(header + I_TYPEOFF, "APPL", 4) &&
!strncmp(header + I_AUTHOFF, "Mooz", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"Zoom\" self extracting archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
zma((char *)NULL, (uint32_t)in_data_size);
skip_file(ds_skip + in_rs);
continue;
}
#endif /* ZMA */
#ifdef LZH
if(!strncmp(header + I_TYPEOFF, "LARC", 4) &&
!strncmp(header + I_AUTHOFF, "LARC", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"MacLHa (LHARC)\" archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
lzh(0);
skip_file(ds_skip + in_rs);
continue;
}
if(!strncmp(header + I_TYPEOFF, "LHA ", 4) &&
!strncmp(header + I_AUTHOFF, "LARC", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"MacLHa (LHA)\" archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
lzh(1);
skip_file(ds_skip + in_rs);
continue;
}
#endif /* LZH */
#ifdef DD
if((!strncmp(header + I_TYPEOFF, "DD01", 4) ||
!strncmp(header + I_TYPEOFF, "DDF?", 3) ||
!strncmp(header + I_TYPEOFF, "DDf?", 3)) &&
!strncmp(header + I_AUTHOFF, "DDAP", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr,
"This is a \"DiskDoubler\" compressed file.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, PACK_NAME);
#endif /* SCAN */
dd_file((unsigned char *)header);
skip_file(ds_skip + in_rs);
continue;
}
if(!strncmp(header + I_TYPEOFF, "DDAR", 4) &&
!strncmp(header + I_AUTHOFF, "DDAP", 4)) {
if(verbose) {
do_indent(indent);
(void)fprintf(stderr, "This is a \"DiskDoubler\" archive.\n");
}
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
dd_arch((unsigned char *)header);
skip_file(ds_skip + in_rs);
continue;
}
if(!strncmp(header + I_TYPEOFF, "APPL", 4) &&
!strncmp(header + I_AUTHOFF, "DSEA", 4)) {
if(verbose) {
do_indent(indent);
}
c = getc(infp);
(void)ungetc(c, infp);
if(c == 'D') {
if(verbose) {
(void)fprintf(stderr,
"This is a \"DiskDoubler\" self extracting archive.\n");
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, ARCH_NAME);
#endif /* SCAN */
}
dd_arch((unsigned char *)header);
} else {
if(verbose) {
(void)fprintf(stderr,
"This is a \"DiskDoubler\" self decompressing file.\n");
#ifdef SCAN
do_idf(header + I_NAMEOFF + 1, PACK_NAME);
#endif /* SCAN */
}
dd_file((unsigned char *)header);
}
skip_file(ds_skip + in_rs);
continue;
}
#endif /* DD */
if(header[0] == 0 /* MORE CHECKS HERE! */) {
mcb(header, (uint32_t)in_rsrc_size,
(uint32_t)in_data_size, in_ds + in_rs);
continue;
} else {
(void)fprintf(stderr, "Unrecognized archive type.\n");
exit(1);
}
}
}
static void
skip_file (int skip)
{
char buff[1024];
int n;
while(skip > 0) {
n = (skip < 1024 ? skip : 1024);
if(fread(buff, 1, n, infp) != n) {
(void)fprintf(stderr, "Incomplete file.\n");
#ifdef SCAN
do_error("macunpack: Incomplete file");
#endif /* SCAN */
exit(1);
}
skip -= n;
}
}
#ifdef SCAN
static void
get_idf (int kind)
{
char filename[255];
if(fread(filename, 1, in_data_size, infp) != in_data_size) {
(void)fprintf(stderr, "Incomplete file.\n");
#ifdef SCAN
do_error("macunpack: Incomplete file");
#endif /* SCAN */
exit(1);
}
filename[in_data_size] = 0;
do_idf(filename, kind);
}
#endif /* SCAN */