mirror of
https://github.com/gryf/ADFlib.git
synced 2026-02-07 08:45:55 +01:00
568 lines
17 KiB
C
568 lines
17 KiB
C
/*
|
|
* unadf 1.0
|
|
*
|
|
*
|
|
* tested under Linux and Win32
|
|
*
|
|
* This file is part of ADFLib.
|
|
*
|
|
* ADFLib is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* ADFLib is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with Foobar; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
*/
|
|
|
|
#define UNADF_VERSION "1.0"
|
|
|
|
|
|
#include<stdlib.h>
|
|
#include<errno.h>
|
|
#include<string.h>
|
|
|
|
#include "adflib.h"
|
|
|
|
/* The portable way used to create a directory is to call the MKDIR command via the
|
|
* system() function.
|
|
* It is used to create the 'dir1' directory, like the 'dir1/dir11' directory
|
|
*/
|
|
|
|
/* the portable way to check if a directory 'dir1' already exists i'm using is to
|
|
* do fopen('dir1','rb'). NULL is returned if 'dir1' doesn't exists yet, an handle instead
|
|
*/
|
|
|
|
#define MKDIR "mkdir"
|
|
|
|
#ifdef WIN32
|
|
#define DIRSEP '\\'
|
|
#else
|
|
#define DIRSEP '/'
|
|
#endif /* WIN32 */
|
|
|
|
#define EXTBUFL 1024*8
|
|
|
|
|
|
void help()
|
|
{
|
|
puts("unadf [-lrcsp -v n] dumpname.adf [files-with-path] [-d extractdir]");
|
|
puts(" -l : lists root directory contents");
|
|
puts(" -r : lists directory tree contents");
|
|
puts(" -c : use dircache data (must be used with -l)");
|
|
puts(" -s : display entries logical block pointer (must be used with -l)");
|
|
putchar('\n');
|
|
puts(" -v n : mount volume #n instead of default #0 volume");
|
|
putchar('\n');
|
|
puts(" -p : send extracted files to pipe (unadf -p dump.adf Pics/pic1.gif | xv -)");
|
|
puts(" -d dir : extract to 'dir' directory");
|
|
}
|
|
|
|
void printEnt(struct Volume *vol, struct Entry* entry, char *path, BOOL sect)
|
|
{
|
|
/* do not print the links entries, ADFlib do not support them yet properly */
|
|
if (entry->type==ST_LFILE || entry->type==ST_LDIR || entry->type==ST_LSOFT)
|
|
return;
|
|
|
|
if (entry->type==ST_DIR)
|
|
printf(" ");
|
|
else
|
|
printf("%7ld ",entry->size);
|
|
|
|
printf("%4d/%02d/%02d %2d:%02d:%02d ",entry->year, entry->month, entry->days,
|
|
entry->hour, entry->mins, entry->secs);
|
|
if (sect)
|
|
printf(" %06ld ",entry->sector);
|
|
|
|
if (strlen(path)>0)
|
|
printf(" %s/",path);
|
|
else
|
|
printf(" ");
|
|
if (entry->type==ST_DIR)
|
|
printf("%s/",entry->name);
|
|
else
|
|
printf("%s",entry->name);
|
|
if (entry->comment!=NULL && strlen(entry->comment)>0)
|
|
printf(", %s",entry->comment);
|
|
putchar('\n');
|
|
|
|
}
|
|
|
|
|
|
void extractFile(struct Volume *vol, char* name, char* path, unsigned char *extbuf,
|
|
BOOL pflag, BOOL qflag)
|
|
{
|
|
struct File *file;
|
|
FILE* out;
|
|
long n;
|
|
char *filename;
|
|
|
|
filename = NULL;
|
|
if (pflag)
|
|
out = stdout;
|
|
else {
|
|
if (strlen(path)>0) {
|
|
filename=(char*)malloc(sizeof(char)* (strlen(path)+1+strlen(name)+1) );
|
|
if (!filename) return;
|
|
sprintf(filename,"%s%c%s",path,DIRSEP,name);
|
|
out = fopen(filename, "wb");
|
|
}
|
|
else
|
|
out = fopen(name, "wb");
|
|
if (!out) return;
|
|
}
|
|
|
|
file = adfOpenFile(vol, name, "r");
|
|
if (!file) { fclose(out); return; }
|
|
|
|
n = adfReadFile(file, EXTBUFL, extbuf);
|
|
while(!adfEndOfFile(file)) {
|
|
fwrite(extbuf, sizeof(unsigned char), n, out);
|
|
n = adfReadFile(file, EXTBUFL, extbuf);
|
|
}
|
|
if (n>0)
|
|
fwrite(extbuf, sizeof(unsigned char), n, out);
|
|
|
|
if (!pflag)
|
|
fclose(out);
|
|
|
|
adfCloseFile(file);
|
|
|
|
if (!qflag) {
|
|
if (filename!=NULL)
|
|
printf("x - %s\n", filename);
|
|
else
|
|
printf("x - %s\n", name);
|
|
}
|
|
|
|
if (filename!=NULL)
|
|
free(filename);
|
|
}
|
|
|
|
|
|
void extractTree(struct Volume *vol, struct List* tree, char *path, unsigned char *extbuf,
|
|
BOOL pflag, BOOL qflag)
|
|
{
|
|
struct Entry* entry;
|
|
char *buf;
|
|
char sysbuf[200];
|
|
|
|
while(tree) {
|
|
entry = (struct Entry*)tree->content;
|
|
if (entry->type==ST_DIR) {
|
|
buf = NULL;
|
|
if (strlen(path)>0) {
|
|
buf=(char*)malloc(strlen(path)+1+strlen(entry->name)+1);
|
|
if (!buf) return;
|
|
sprintf(buf,"%s%c%s",path,DIRSEP,entry->name);
|
|
sprintf(sysbuf,"%s %s",MKDIR,buf);
|
|
if (!qflag) printf("x - %s%c\n",buf,DIRSEP);
|
|
}
|
|
else {
|
|
sprintf(sysbuf,"%s %s",MKDIR,entry->name);
|
|
if (!qflag) printf("x - %s%c\n",entry->name,DIRSEP);
|
|
}
|
|
|
|
if (!pflag) system(sysbuf);
|
|
|
|
if (tree->subdir!=NULL) {
|
|
if (adfChangeDir(vol,entry->name)==RC_OK) {
|
|
if (buf!=NULL)
|
|
extractTree(vol,tree->subdir,buf,extbuf, pflag, qflag);
|
|
else
|
|
extractTree(vol,tree->subdir,entry->name,extbuf, pflag, qflag);
|
|
adfParentDir(vol);
|
|
}
|
|
else {
|
|
if (strlen(path)>0)
|
|
fprintf(stderr,"ExtractTree : dir \"%s/%s\" not found.\n",path,entry->name);
|
|
else
|
|
fprintf(stderr,"ExtractTree : dir \"%s\" not found.\n",entry->name);
|
|
}
|
|
}
|
|
|
|
if (buf!=NULL)
|
|
free(buf);
|
|
}
|
|
else if (entry->type==ST_FILE) {
|
|
extractFile(vol,entry->name,path,extbuf, pflag, qflag);
|
|
}
|
|
tree = tree->next;
|
|
}
|
|
}
|
|
|
|
|
|
void printTree(struct Volume *vol, struct List* tree, char* path, BOOL sect)
|
|
{
|
|
char *buf;
|
|
struct Entry* entry;
|
|
|
|
while(tree) {
|
|
printEnt(vol, tree->content, path, sect);
|
|
if (tree->subdir!=NULL) {
|
|
entry = (struct Entry*)tree->content;
|
|
if (strlen(path)>0) {
|
|
buf=(char*)malloc(sizeof(char)* (strlen(path)+1+strlen(entry->name)+1) );
|
|
if (!buf) {
|
|
fprintf(stderr,"printTree : malloc error\n");
|
|
return;
|
|
}
|
|
sprintf(buf,"%s/%s", path, entry->name);
|
|
printTree(vol, tree->subdir, buf, sect);
|
|
free(buf);
|
|
}
|
|
else
|
|
printTree(vol, tree->subdir, entry->name, sect);
|
|
}
|
|
tree = tree->next;
|
|
}
|
|
}
|
|
|
|
|
|
void printDev(struct Device* dev)
|
|
{
|
|
printf("Device : ");
|
|
|
|
switch(dev->devType){
|
|
case DEVTYPE_FLOPDD:
|
|
printf("Floppy DD"); break;
|
|
case DEVTYPE_FLOPHD:
|
|
printf("Floppy HD"); break;
|
|
case DEVTYPE_HARDDISK:
|
|
printf("Harddisk"); break;
|
|
case DEVTYPE_HARDFILE:
|
|
printf("Hardfile"); break;
|
|
default:
|
|
printf("???"); break;
|
|
}
|
|
|
|
printf(". Cylinders = %ld, Heads = %ld, Sectors = %ld",dev->cylinders,dev->heads,dev->sectors);
|
|
|
|
printf(". Volumes = %d\n",dev->nVol);
|
|
}
|
|
|
|
|
|
void printVol(struct Volume* vol, int volNum)
|
|
{
|
|
printf("Volume : ");
|
|
|
|
switch(vol->dev->devType) {
|
|
case DEVTYPE_FLOPDD:
|
|
printf ("Floppy 880 KBytes,");
|
|
break;
|
|
case DEVTYPE_FLOPHD:
|
|
printf ("Floppy 1760 KBytes,");
|
|
break;
|
|
case DEVTYPE_HARDDISK:
|
|
printf ("HD partition #%d %3.1f KBytes,", volNum, (vol->lastBlock - vol->firstBlock +1) * 512.0/1024.0);
|
|
break;
|
|
case DEVTYPE_HARDFILE:
|
|
printf ("HardFile %3.1f KBytes,", (vol->lastBlock - vol->firstBlock +1) * 512.0/1024.0);
|
|
break;
|
|
default:
|
|
printf ("???,");
|
|
}
|
|
|
|
if (vol->volName!=NULL)
|
|
printf(" \"%s\"", vol->volName);
|
|
|
|
printf(" between sectors [%ld-%ld].",vol->firstBlock, vol->lastBlock);
|
|
|
|
printf(" %s ",isFFS(vol->dosType) ? "FFS" : "OFS");
|
|
if (isINTL(vol->dosType))
|
|
printf ("INTL ");
|
|
if (isDIRCACHE(vol->dosType))
|
|
printf ("DIRCACHE ");
|
|
|
|
printf(". Filled at %2.1f%%.\n", 100.0-
|
|
(adfCountFreeBlocks(vol)*100.0)/(vol->lastBlock - vol->firstBlock +1) );
|
|
|
|
}
|
|
|
|
|
|
void processFile(struct Volume *vol, char* name, char* path, unsigned char *extbuf,
|
|
BOOL pflag, BOOL qflag)
|
|
{
|
|
char *sepptr, *cdstr, *fullname, *filename;
|
|
char *bigstr;
|
|
FILE *tfile;
|
|
|
|
adfToRootDir(vol);
|
|
|
|
sepptr = strchr(name, '/');
|
|
if (sepptr==NULL) {
|
|
extractFile(vol, name, path, extbuf, pflag, qflag);
|
|
}
|
|
else {
|
|
/* the all-in-one string : to call system(), to find the filename, the convert dir sep char ... */
|
|
bigstr=(char*)malloc(strlen(MKDIR)+1+strlen(path)+1+strlen(name)+1);
|
|
if (!bigstr) { fprintf(stderr,"processFile : malloc"); return; }
|
|
|
|
/* to build to extract path */
|
|
if (strlen(path)>0) {
|
|
sprintf(bigstr,"%s %s%c%s",MKDIR,path,DIRSEP,name);
|
|
cdstr = bigstr+strlen(MKDIR)+1+strlen(path)+1;
|
|
}
|
|
else {
|
|
sprintf(bigstr,"%s %s",MKDIR,name);
|
|
cdstr = bigstr+strlen(MKDIR)+1;
|
|
}
|
|
/* the directory in which the file will be extracted */
|
|
fullname = bigstr+strlen(MKDIR)+1;
|
|
|
|
/* finds the filename, and separates it from the path */
|
|
filename = strrchr(bigstr,'/')+1;
|
|
filename[-1]='\0';
|
|
|
|
sepptr = cdstr;
|
|
/* find the end of the first dir to create */
|
|
while(sepptr[0]!='/' && sepptr[0]!='\0')
|
|
sepptr++;
|
|
|
|
while(strlen(cdstr)>0) {
|
|
if (sepptr[0]=='/') { /* not the last one */
|
|
sepptr[0]='\0';
|
|
if (adfChangeDir(vol,cdstr)!=RC_OK)
|
|
return;
|
|
tfile = fopen(fullname,"r"); /* the only portable way to test if the dir exists */
|
|
if (tfile==NULL) { /* does't exist : create it */
|
|
if (!pflag) system(bigstr);
|
|
if (!qflag) printf("x - %s%c\n",fullname,DIRSEP);
|
|
}
|
|
else
|
|
fclose(tfile);
|
|
sepptr[0] = DIRSEP; /* converts the '/' to '/' or '\' */
|
|
cdstr = sepptr+1; /* next beginning of the next dir to create */
|
|
/* to find the end of the next dir */
|
|
sepptr++;
|
|
while(sepptr[0]!='/' && sepptr[0]!='\0')
|
|
sepptr++;
|
|
}
|
|
else { /* the last one */
|
|
if (adfChangeDir(vol,cdstr)!=RC_OK)
|
|
return;
|
|
tfile = fopen(fullname,"r");
|
|
if (tfile==NULL) {
|
|
if (!pflag) system(bigstr);
|
|
if (!qflag) printf("x - %s%c\n",fullname,DIRSEP);
|
|
}
|
|
else
|
|
fclose(tfile);
|
|
cdstr = cdstr+strlen(cdstr); /* at the end, ends the while loop */
|
|
}
|
|
}
|
|
extractFile(vol, filename, fullname, extbuf, pflag, qflag);
|
|
|
|
free(bigstr);
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
int main(int argc, char* argv[])
|
|
{
|
|
int i, j;
|
|
BOOL rflag, lflag, xflag, cflag, vflag, sflag, dflag, pflag, qflag;
|
|
struct List* files, *rtfiles;
|
|
char *devname, *dirname;
|
|
char strbuf[80];
|
|
unsigned char *extbuf;
|
|
int vInd, dInd, fInd, aInd;
|
|
BOOL nextArg;
|
|
|
|
struct Device *dev;
|
|
struct Volume *vol;
|
|
struct List *list, *cell;
|
|
int volNum;
|
|
BOOL true = TRUE;
|
|
|
|
if (argc<2) {
|
|
help();
|
|
exit(0);
|
|
}
|
|
|
|
rflag = lflag = cflag = vflag = sflag = dflag = pflag = qflag = FALSE;
|
|
vInd = dInd = fInd = aInd = -1;
|
|
xflag = TRUE;
|
|
dirname = NULL;
|
|
devname = NULL;
|
|
files = rtfiles = NULL;
|
|
volNum = 0;
|
|
|
|
fprintf(stderr,"unADF v%s : a unzip like for .ADF files, powered by ADFlib (v%s - %s)\n\n",
|
|
UNADF_VERSION, adfGetVersionNumber(),adfGetVersionDate());
|
|
|
|
/* parse options */
|
|
i=1;
|
|
while(i<argc) {
|
|
if (argv[i][0]=='-') {
|
|
j=1;
|
|
nextArg = FALSE;
|
|
while(j<(int)strlen(argv[i]) && !nextArg) {
|
|
switch(argv[i][j]) {
|
|
case 'v':
|
|
vflag = TRUE;
|
|
if ((i+1)<(argc-1)) {
|
|
i++;
|
|
nextArg = TRUE;
|
|
errno = 0;
|
|
volNum = atoi(argv[i]);
|
|
if (errno!=0 || volNum<0) {
|
|
fprintf(stderr,"invalid volume number, aborting.\n");
|
|
exit(1);
|
|
}
|
|
}
|
|
else
|
|
fprintf(stderr,"no volume number, -v option ignored.\n");
|
|
break;
|
|
case 'l':
|
|
lflag = TRUE;
|
|
xflag = FALSE;
|
|
break;
|
|
case 's':
|
|
sflag = TRUE;
|
|
break;
|
|
case 'c':
|
|
cflag = TRUE;
|
|
break;
|
|
case 'r':
|
|
rflag = TRUE;
|
|
break;
|
|
case 'd':
|
|
if (devname!=NULL && xflag && (i+1)==(argc-1)) {
|
|
i++;
|
|
dirname = argv[i];
|
|
if (dirname[strlen(dirname)-1]==DIRSEP)
|
|
dirname[strlen(dirname)-1]='\0';
|
|
nextArg = TRUE;
|
|
dflag = TRUE;
|
|
}
|
|
break;
|
|
case 'p':
|
|
if (xflag) {
|
|
fprintf(stderr,"sending files to pipe.\n");
|
|
pflag = TRUE;
|
|
qflag = TRUE;
|
|
}
|
|
else
|
|
fprintf(stderr,"-p option must be used with extraction, ignored.\n");
|
|
break;
|
|
case 'h':
|
|
default:
|
|
help();
|
|
exit(0);
|
|
} /* switch */
|
|
j++;
|
|
} /* while */
|
|
} /* if */
|
|
else {
|
|
/* the last non option string is taken as a filename */
|
|
if (devname==NULL) /* if the device name has been already given */
|
|
devname = argv[i];
|
|
else {
|
|
if (xflag) {
|
|
if (rtfiles==NULL)
|
|
rtfiles = files = newCell(NULL, (void*)argv[i]);
|
|
else
|
|
files = newCell(files, (void*)argv[i]);
|
|
}
|
|
else
|
|
fprintf(stderr,"Must be used with extraction, ignored.\n");
|
|
}
|
|
}
|
|
i++;
|
|
} /* while */
|
|
|
|
extbuf =(unsigned char*)malloc(EXTBUFL*sizeof(char));
|
|
if (!extbuf) { fprintf(stderr,"malloc error\n"); exit(1); }
|
|
|
|
/* initialize the library */
|
|
adfEnvInitDefault();
|
|
|
|
dev = adfMountDev( devname,TRUE );
|
|
if (!dev) {
|
|
sprintf(strbuf,"Can't mount the dump device '%s'.\n", devname);
|
|
fprintf(stderr, strbuf);
|
|
adfEnvCleanUp(); exit(1);
|
|
}
|
|
if (!qflag)
|
|
printDev(dev);
|
|
|
|
if (volNum>=dev->nVol) {
|
|
fprintf(stderr,"This device has only %d volume(s), aborting.\n",dev->nVol);
|
|
exit(1);
|
|
}
|
|
|
|
vol = adfMount(dev, volNum, TRUE);
|
|
if (!vol) {
|
|
adfUnMountDev(dev);
|
|
fprintf(stderr, "Can't mount the volume\n");
|
|
adfEnvCleanUp(); exit(1);
|
|
}
|
|
|
|
if (!qflag) {
|
|
printVol(vol, volNum);
|
|
putchar('\n');
|
|
}
|
|
|
|
if (cflag && isDIRCACHE(vol->dosType) && lflag) {
|
|
adfChgEnvProp(PR_USEDIRC,&true);
|
|
if (!qflag)
|
|
puts("Using dir cache blocks.");
|
|
}
|
|
|
|
if (lflag) {
|
|
if (!rflag) {
|
|
cell = list = adfGetDirEnt(vol,vol->curDirPtr);
|
|
while(cell) {
|
|
printEnt(vol,cell->content,"", sflag);
|
|
cell = cell->next;
|
|
}
|
|
adfFreeDirList(list);
|
|
} else {
|
|
cell = list = adfGetRDirEnt(vol,vol->curDirPtr,TRUE);
|
|
printTree(vol,cell,"", sflag);
|
|
adfFreeDirList(list);
|
|
}
|
|
}else if (xflag) {
|
|
if (rtfiles!=NULL) {
|
|
files = rtfiles;
|
|
while(files!=NULL) {
|
|
if (dirname!=NULL)
|
|
processFile(vol, (char*)files->content, dirname, extbuf, pflag, qflag);
|
|
else
|
|
processFile(vol, (char*)files->content, "", extbuf, pflag, qflag);
|
|
files = files->next;
|
|
}
|
|
freeList(rtfiles);
|
|
}
|
|
else {
|
|
cell = list = adfGetRDirEnt(vol,vol->curDirPtr,TRUE);
|
|
if (dirname==NULL)
|
|
extractTree(vol, cell, "", extbuf, pflag, qflag);
|
|
else
|
|
extractTree(vol, cell, dirname, extbuf, pflag, qflag);
|
|
adfFreeDirList(list);
|
|
}
|
|
}
|
|
else
|
|
help();
|
|
|
|
free(extbuf);
|
|
|
|
adfUnMount(vol);
|
|
adfUnMountDev(dev);
|
|
|
|
adfEnvCleanUp();
|
|
|
|
return(0);
|
|
}
|