macutils/fileio/wrfile.c
2022-09-29 22:49:54 +02:00

846 lines
19 KiB
C

#include "wrfile.h"
#ifdef TYPES_H
#include <sys/types.h>
#endif /* TYPES_H */
#include <sys/stat.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <stdio.h>
#include <unistd.h>
#include "machdr.h"
#include "wrfileopt.h"
#include "../util/util.h"
#ifdef AUFSPLUS
#include "../util/curtime.h"
#define AUFS
#endif /* AUFSPLUS */
#ifdef AUFS
#include "aufs.h"
#define APPLESHARE
#endif /* AUFS */
#ifdef APPLEDOUBLE
#include "appledouble.h"
#include "../util/curtime.h"
#define APPLESHARE
#endif /* APPLEDOUBLE */
#define TEXT 0
#define DATA 1
#define RSRC 2
#define FULL 3
#define MACB 4
#define FORK 5
#define APSH 6
#define MACS 7
#define UNIX 8
#ifdef SCAN
#define MACI 9
#endif /* SCAN */
#ifdef UNDEF /* Do not declare sprintf; not portable (but lint will complain) */
char *sprintf();
#endif /* UNDEF */
#ifdef AUFS
static void check_aufs();
static void aufs_namings();
static void wr_aufs_info();
#endif /* AUFS */
#ifdef APPLEDOUBLE
static void check_appledouble();
static void appledouble_namings();
static void wr_appledouble_info();
#endif /* APPLEDOUBLE */
#ifdef APPLESHARE
static void mk_share_name();
#endif /* APPLESHARE */
#ifndef BSD
/* all those stupid differences! */
#define bcopy(src,dest,length) memcpy((dest),(src),(length))
#define bzero(block,length) memset((block),0,(length))
#endif /* BSD */
#define INFO_FORK 1
#define RSRC_FORK 2
#define DATA_FORK 3
static char f_info[I_NAMELEN];
static char f_data[I_NAMELEN*3];
static char f_rsrc[I_NAMELEN];
static char f_text[I_NAMELEN];
static char f_unix[I_NAMELEN];
static char f_bin[I_NAMELEN];
static char f_folder[] = ".foldername";
static char share_name[256];
#ifdef APPLESHARE
static char hex[] = "0123456789abcdef";
#endif /* APPLESHARE */
#ifdef AUFS
static char infodir[] = ".finderinfo";
static char rsrcdir[] = ".resource";
#define INFOSZ sizeof(infodir)
#define RSRCSZ sizeof(rsrcdir)
static char f_info_aufs[I_NAMELEN*3+INFOSZ];
static char f_rsrc_aufs[I_NAMELEN*3+RSRCSZ];
#endif /* AUFS */
#ifdef APPLEDOUBLE
static char infodir[] = ".AppleDouble";
#define INFOSZ sizeof(infodir)
static char f_info_appledouble[I_NAMELEN*3+INFOSZ];
#endif /* APPLEDOUBLE */
static int mode = MACB;
static int mode_restricted = 0;
static int mode_s_restricted = 0;
char *out_buffer, *out_ptr;
static char init_buffer[128];
static char *buffer = &(init_buffer[0]);
static char *rbuffer = NULL, *dbuffer = NULL;
static char *ptr;
static unsigned long rsz, dsz, totsize, maxsize;
void
define_name (char *text)
{
(void)sprintf(f_info, "%s.info", text);
(void)sprintf(f_rsrc, "%s.rsrc", text);
(void)sprintf(f_data, "%s.data", text);
(void)sprintf(f_text, "%s.text", text);
(void)sprintf(f_bin, "%s.bin", text);
(void)sprintf(f_unix, "%s", text);
#ifdef APPLESHARE
/* Do not do namestuffing here. We want to base again on the information in
the info header, so this is delayed
*/
#endif /* APPLESHARE */
}
void
start_info (char *info, unsigned long rsize, unsigned long dsize)
{
int rs, ds;
rsz = rsize;
dsz = dsize;
rs = (((rsz + 127) >> 7) << 7);
ds = (((dsz + 127) >> 7) << 7);
totsize = rs + ds + 128;
if(buffer == &(init_buffer[0])) {
buffer = (char *)malloc((unsigned)totsize);
} else if(maxsize < totsize) {
buffer = (char *)realloc(buffer, (unsigned)totsize);
}
maxsize = totsize;
if(buffer == NULL) {
(void)fprintf(stderr, "Insufficient memory, aborting\n");
exit(1);
}
dbuffer = buffer + 128;
rbuffer = dbuffer + ds;
(void)bzero(buffer, (int)totsize);
ptr = buffer;
(void)bcopy(info, ptr, 128);
#ifdef AUFS
/* Now we do filenaming etc. */
if(mode == APSH) {
aufs_namings();
}
#endif /* AUFS */
#ifdef APPLEDOUBLE
/* Now we do filenaming etc. */
if(mode == APSH) {
appledouble_namings();
}
#endif /* APPLEDOUBLE */
}
void
start_rsrc (void)
{
out_buffer = out_ptr = rbuffer;
}
void
start_data (void)
{
out_buffer = out_ptr = dbuffer;
}
void
end_file (void)
{
FILE *fp;
int i, c;
buffer[I_FLAGOFF] &= (~INITED_MASK);
switch(mode) {
case FULL:
case FORK:
fp = fopen(f_info, "w");
if(fp == NULL) {
perror(f_info);
exit(1);
}
(void)fwrite(buffer, 1, 128, fp);
(void)fclose(fp);
if(rsz != 0 || mode == FULL) {
fp = fopen(f_rsrc, "w");
if(fp == NULL) {
perror(f_rsrc);
exit(1);
}
(void)fwrite(rbuffer, 1, (int)rsz, fp);
(void)fclose(fp);
}
if(dsz != 0 || mode == FULL) {
fp = fopen(f_data, "w");
if(fp == NULL) {
perror(f_data);
exit(1);
}
(void)fwrite(dbuffer, 1, (int)dsz, fp);
(void)fclose(fp);
}
break;
case RSRC:
fp = fopen(f_rsrc, "w");
if(fp == NULL) {
perror(f_rsrc);
exit(1);
}
(void)fwrite(rbuffer, 1, (int)rsz, fp);
(void)fclose(fp);
break;
case DATA:
fp = fopen(f_data, "w");
if(fp == NULL) {
perror(f_data);
exit(1);
}
(void)fwrite(dbuffer, 1, (int)dsz, fp);
(void)fclose(fp);
break;
case TEXT:
fp = fopen(f_text, "w");
if(fp == NULL) {
perror(f_data);
exit(1);
}
for(i = 0; i < dsz; i++) {
c = dbuffer[i];
if(c == '\012' || c == '\015') {
dbuffer[i] = '\027' -c;
}
}
(void)fwrite(dbuffer, 1, (int)dsz, fp);
(void)fclose(fp);
break;
case UNIX:
fp = fopen(f_unix, "w");
if(fp == NULL) {
perror(f_data);
exit(1);
}
for(i = 0; i < dsz; i++) {
c = dbuffer[i];
if(c == '\012' || c == '\015') {
dbuffer[i] = '\027' -c;
}
}
(void)fwrite(dbuffer, 1, (int)dsz, fp);
(void)fclose(fp);
break;
case MACB:
fp = fopen(f_bin, "w");
if(fp == NULL) {
perror(f_bin);
exit(1);
}
if(buffer[I_FLAGOFF + 1] & PROTCT_MASK) {
buffer[I_LOCKOFF] = 1;
}
buffer[I_FLAGOFF + 1] = 0;
buffer[I_LOCKOFF + 1] = 0;
(void)fwrite(buffer, 1, (int)totsize, fp);
(void)fclose(fp);
break;
case MACS:
#ifdef SCAN
case MACI:
#endif /* SCAN */
if(buffer[I_FLAGOFF + 1] & PROTCT_MASK) {
buffer[I_LOCKOFF] = 1;
}
buffer[I_FLAGOFF + 1] = 0;
buffer[I_LOCKOFF + 1] = 0;
(void)fwrite(buffer, 1, (int)totsize, stdout);
break;
#ifdef AUFS
case APSH:
fp = fopen(f_info_aufs, "w");
if(fp == NULL) {
perror(f_info_aufs);
exit(1);
}
wr_aufs_info(fp);
(void) fclose(fp);
fp = fopen(f_rsrc_aufs, "w");
if(fp == NULL) {
perror(f_rsrc_aufs);
exit(1);
}
(void)fwrite(rbuffer, 1, (int)rsz, fp);
(void)fclose(fp);
fp = fopen(f_data, "w");
if(fp == NULL) {
perror(f_data);
exit(1);
}
(void)fwrite(dbuffer, 1, (int)dsz, fp);
(void)fclose(fp);
break;
#endif /* AUFS */
#ifdef APPLEDOUBLE
case APSH:
fp = fopen(f_info_appledouble, "w");
if(fp == NULL) {
perror(f_info_appledouble);
exit(1);
}
wr_appledouble_info(fp);
(void)fwrite(rbuffer, 1, (int)rsz, fp);
(void)fclose(fp);
fp = fopen(f_data, "w");
if(fp == NULL) {
perror(f_data);
exit(1);
}
(void)fwrite(dbuffer, 1, (int)dsz, fp);
(void)fclose(fp);
break;
#endif /* APPLEDOUBLE */
}
}
#ifdef SCAN
void
do_idf (char *name, int kind)
{
int n;
if(mode != MACI) {
return;
}
n = strlen(name);
(void)bzero(buffer, INFOBYTES);
buffer[I_NAMEOFF + 1] = kind;
put4(buffer + I_DLENOFF, (unsigned long)n);
(void)fwrite(buffer, 1, INFOBYTES, stdout);
if(n != 0) {
(void)fwrite(name, 1, n, stdout);
n = (((n + 127) >> 7) << 7) - n;
while(n-- > 0) {
(void)fputc(0, stdout);
}
}
}
#endif /* SCAN */
void
do_mkdir (char *name, char *header)
{
struct stat sbuf;
FILE *fp;
#ifdef NOMKDIR
char command[21]; /* Systems without mkdir system call but more than 14
char file names? Ridiculous! */
int sysreturn;
#endif /* MKDIR */
#ifdef APPLESHARE
char dirinfo[I_NAMELEN*3+INFOSZ+10];
#endif /* APPLESHARE */
#ifndef SCAN
if(mode == MACS) {
#else /* SCAN */
if(mode == MACS || mode == MACI) {
#endif /* SCAN */
header[I_NAMEOFF] |= 0x80;
(void)fwrite(header, 1, INFOBYTES, stdout);
header[I_NAMEOFF] &= 0x7f;
return;
}
#ifdef APPLESHARE
if(mode == APSH) {
(void)bcopy(header, buffer, INFOBYTES);
mk_share_name();
} else {
(void)strcpy(share_name, name);
}
#else /* APPLESHARE */
(void)strcpy(share_name, name);
#endif /* APPLESHARE */
if(stat(share_name, &sbuf) == -1) { /* directory doesn't exist */
#ifndef NOMKDIR
if(mkdir(share_name, 0777) == -1) {
(void)fprintf(stderr, "Can't create subdirectory %s\n", share_name);
exit(1);
}
#else /* NOMKDIR */
sprintf(command, "mkdir %s", share_name);
if((sysreturn = system(command)) != 0) {
(void)fprintf(stderr, "Can't create subdirectory %s\n", share_name);
exit(sysreturn);
}
#endif /* NOMKDIR */
} else { /* something exists with this name */
if((sbuf.st_mode & S_IFMT) != S_IFDIR) {
(void)fprintf(stderr, "Directory name %s already in use\n",
share_name);
exit(1);
}
}
(void)chdir(share_name);
#ifdef APPLESHARE
#ifdef AUFS
if(mode == APSH) {
if(stat(rsrcdir, &sbuf) == -1) { /* directory doesn't exist */
if(mkdir(rsrcdir, 0777) == -1) {
(void)fprintf(stderr, "Can't create subdirectory %s\n",
rsrcdir);
exit(1);
}
} else {
if((sbuf.st_mode & S_IFMT) != S_IFDIR) {
(void)fprintf(stderr, "Directory name %s already in use\n",
rsrcdir);
exit(1);
}
}
if(stat(infodir, &sbuf) == -1) { /* directory doesn't exist */
if(mkdir(infodir, 0777) == -1) {
(void)fprintf(stderr, "Can't create subdirectory %s\n",
infodir);
exit(1);
}
} else {
if((sbuf.st_mode & S_IFMT) != S_IFDIR) {
(void)fprintf(stderr, "Directory name %s already in use\n",
infodir);
exit(1);
}
}
dirinfo[0] = 0;
(void)strcat(dirinfo, "../");
(void)strcat(dirinfo, infodir);
(void)strcat(dirinfo, "/");
(void)strcat(dirinfo, share_name);
fp = fopen(dirinfo, "w");
if(fp == NULL) {
perror(dirinfo);
exit(1);
}
wr_aufs_info(fp);
(void)fclose(fp);
} else {
fp = fopen(f_folder, "w");
if(fp == NULL) {
perror(f_folder);
exit(1);
}
header[I_NAMEOFF] |= 0x80;
(void)fwrite(header, 1, INFOBYTES, fp);
header[I_NAMEOFF] &= 0x7f;
(void)fclose(fp);
}
#endif /* AUFS */
#ifdef APPLEDOUBLE
if(mode == APSH) {
if(stat(infodir, &sbuf) == -1) { /* directory doesn't exist */
if(mkdir(infodir, 0777) == -1) {
(void)fprintf(stderr, "Can't create subdirectory %s\n",
infodir);
exit(1);
}
} else {
if((sbuf.st_mode & S_IFMT) != S_IFDIR) {
(void)fprintf(stderr, "Directory name %s already in use\n",
infodir);
exit(1);
}
}
dirinfo[0] = 0;
(void)strcat(dirinfo, infodir);
(void)strcat(dirinfo, "/.Parent");
fp = fopen(dirinfo, "w");
if(fp == NULL) {
perror(dirinfo);
exit(1);
}
rsz = 0;
wr_appledouble_info(fp);
(void)fclose(fp);
} else {
fp = fopen(f_folder, "w");
if(fp == NULL) {
perror(f_folder);
exit(1);
}
header[I_NAMEOFF] |= 0x80;
(void)fwrite(header, 1, INFOBYTES, fp);
header[I_NAMEOFF] &= 0x7f;
(void)fclose(fp);
}
#endif /* APPLEDOUBLE */
#else /* APPLESHARE */
fp = fopen(f_folder, "w");
if(fp == NULL) {
perror(f_folder);
exit(1);
}
header[I_NAMEOFF] |= 0x80;
(void)fwrite(header, 1, INFOBYTES, fp);
header[I_NAMEOFF] &= 0x7f;
(void)fclose(fp);
#endif /* APPLESHARE */
}
void enddir(void)
{
char header[INFOBYTES];
int i;
#ifndef SCAN
if(mode == MACS) {
#else /* SCAN */
if(mode == MACS || mode == MACI) {
#endif /* SCAN */
for(i = 0; i < INFOBYTES; i++) {
header[i] = 0;
}
header[I_NAMEOFF] = 0x80;
(void)fwrite(header, 1, INFOBYTES, stdout);
} else {
(void)chdir("..");
}
}
#ifdef APPLESHARE
#ifdef AUFS
static void check_aufs()
{
/* check for .resource/ and .finderinfo/ */
struct stat stbuf;
int error = 0;
if(stat(rsrcdir,&stbuf) < 0) {
error ++;
} else {
if((stbuf.st_mode & S_IFMT) != S_IFDIR) {
error ++;
}
}
if(stat(infodir,&stbuf) < 0) {
error ++;
} else {
if((stbuf.st_mode & S_IFMT) != S_IFDIR) {
error++;
}
}
if(error) {
(void)fprintf(stderr, "Not in an Aufs folder.\n");
exit(1);
}
}
static void aufs_namings()
{
mk_share_name();
(void)sprintf(f_info_aufs, "%s/%s", infodir, share_name);
(void)sprintf(f_rsrc_aufs, "%s/%s", rsrcdir, share_name);
(void)sprintf(f_data, "%s", share_name);
}
static void wr_aufs_info(fp)
FILE *fp;
{
FileInfo theinfo;
int n;
bzero((char *) &theinfo, sizeof theinfo);
theinfo.fi_magic1 = FI_MAGIC1;
theinfo.fi_version = FI_VERSION;
theinfo.fi_magic = FI_MAGIC;
theinfo.fi_bitmap = FI_BM_MACINTOSHFILENAME;
/* AUFS stores Unix times. */
#ifdef AUFSPLUS
theinfo.fi_datemagic = FI_MAGIC;
theinfo.fi_datevalid = FI_CDATE | FI_MDATE;
put4(theinfo.fi_ctime, get4(buffer + I_CTIMOFF) - TIMEDIFF);
put4(theinfo.fi_mtime, get4(buffer + I_MTIMOFF) - TIMEDIFF);
put4(theinfo.fi_utime, (unsigned long)time((time_t *)0));
#endif /* AUFSPLUS */
bcopy(buffer + I_TYPEOFF, theinfo.fi_fndr, 4);
bcopy(buffer + I_AUTHOFF, theinfo.fi_fndr + 4, 4);
bcopy(buffer + I_FLAGOFF, theinfo.fi_fndr + 8, 2);
if((n = buffer[I_NAMEOFF] & 0xff) > F_NAMELEN) {
n = F_NAMELEN;
}
(void)strncpy((char *)theinfo.fi_macfilename, buffer + I_NAMEOFF + 1,n);
/* theinfo.fi_macfilename[n] = '\0'; */
(void)strcpy((char *)theinfo.fi_comnt,
"Converted by Unix utility to Aufs format");
theinfo.fi_comln = strlen((char *)theinfo.fi_comnt);
(void)fwrite((char *) &theinfo, 1, sizeof theinfo, fp);
}
#endif /* AUFS */
#ifdef APPLEDOUBLE
static void check_appledouble()
{
/* check for .AppleDouble/ */
struct stat stbuf;
int error = 0;
if(stat(infodir,&stbuf) < 0) {
error ++;
} else {
if((stbuf.st_mode & S_IFMT) != S_IFDIR) {
error++;
}
}
if(error) {
(void)fprintf(stderr, "Not in an AppleDouble folder.\n");
exit(1);
}
}
static void appledouble_namings()
{
mk_share_name();
(void)snprintf(f_info_appledouble, sizeof(f_info_appledouble), "%s/%s", infodir, share_name);
(void)snprintf(f_data, sizeof(f_data), "%s", share_name);
}
static void wr_appledouble_info(fp)
FILE *fp;
{
FileInfo theinfo;
int n;
bzero((char *) &theinfo, sizeof theinfo);
put4(theinfo.fi_magic, (unsigned long)FI_MAGIC);
put2(theinfo.fi_version, (unsigned long)FI_VERSION);
put4(theinfo.fi_fill5, (unsigned long)FI_FILL5);
put4(theinfo.fi_fill6, (unsigned long)FI_FILL6);
put4(theinfo.fi_hlen, (unsigned long)FI_HLEN);
put4(theinfo.fi_fill7, (unsigned long)FI_FILL7);
put4(theinfo.fi_namptr, (unsigned long)FI_NAMPTR);
put4(theinfo.fi_fill9, (unsigned long)FI_FILL9);
put4(theinfo.fi_commptr, (unsigned long)FI_COMMPTR);
put4(theinfo.fi_fill12, (unsigned long)FI_FILL12);
put4(theinfo.fi_timeptr, (unsigned long)FI_TIMEPTR);
put4(theinfo.fi_timesize, (unsigned long)FI_TIMESIZE);
put4(theinfo.fi_fill15, (unsigned long)FI_FILL15);
put4(theinfo.fi_infoptr, (unsigned long)FI_INFOPTR);
put4(theinfo.fi_infosize, (unsigned long)FI_INFOSIZE);
bcopy(buffer + I_TYPEOFF, theinfo.fi_type, 4);
bcopy(buffer + I_AUTHOFF, theinfo.fi_auth, 4);
bcopy(buffer + I_FLAGOFF, theinfo.fi_finfo, 2);
/* AppleDouble stores Unix times. */
put4(theinfo.fi_ctime, get4(buffer + I_CTIMOFF) - TIMEDIFF);
put4(theinfo.fi_mtime, get4(buffer + I_MTIMOFF) - TIMEDIFF);
if((n = buffer[I_NAMEOFF] & 0xff) > F_NAMELEN) {
n = F_NAMELEN;
}
put4(theinfo.fi_namlen, (unsigned long)n);
(void)strncpy((char *)theinfo.fi_name, buffer + I_NAMEOFF + 1,n);
/* theinfo.fi_macfilename[n] = '\0'; */
(void)strcpy((char *)theinfo.fi_comment,
"Converted by Unix utility to AppleDouble format");
put4(theinfo.fi_commsize, (unsigned long)strlen(theinfo.fi_comment));
put4(theinfo.fi_rsrc, (unsigned long)rsz);
/* Still TODO */
/* char fi_ctime[4]; /* Creation time (Unix time) */
/* char fi_mtime[4]; /* Modification time (Unix time) */
(void)fwrite((char *) &theinfo, 1, sizeof theinfo, fp);
}
#endif /* APPLEDOUBLE */
static void mk_share_name()
{
int ch;
char *mp, *up;
mp = buffer + 2;
up = &(share_name[0]);
while((ch = *mp++)) {
if(isascii(ch) && ! iscntrl(ch) && isprint(ch) && ch != '/') {
*up++ = ch;
} else {
*up++ = ':';
*up++ = hex[(ch >> 4) & 0xf];
*up++ = hex[ch & 0xf];
}
}
*up = 0;
}
#endif /* APPLESHARE */
int wrfileopt(c)
char c;
{
switch(c) {
case 'b':
mode = MACB;
break;
case 'r':
if(mode_restricted) {
return 0;
}
mode = RSRC;
break;
case 'd':
if(mode_restricted) {
return 0;
}
mode = DATA;
break;
case 'u':
if(mode_restricted) {
return 0;
}
mode = TEXT;
break;
case 'U':
if(mode_restricted) {
return 0;
}
mode = UNIX;
break;
case 'f':
mode = FORK;
break;
case '3':
mode = FULL;
break;
case 's':
if(mode_s_restricted) {
return 0;
}
mode = MACS;
break;
#ifdef SCAN
case 'S':
if(mode_s_restricted) {
return 0;
}
mode = MACI;
break;
#endif /* SCAN */
case 'a':
#ifdef APPLESHARE
#ifdef AUFS
check_aufs();
mode = APSH;
break;
#endif /* AUFS */
#ifdef APPLEDOUBLE
check_appledouble();
mode = APSH;
break;
#endif /* APPLEDOUBLE */
#else /* APPLESHARE */
(void)fprintf(stderr, "Sorry, Apple-Unix sharing is not supported.\n");
(void)fprintf(stderr, "Recompile or omit -a option.\n");
exit(1);
#endif /* APPLESHARE */
default:
return 0;
}
return 1;
}
void give_wrfileopt()
{
(void)fprintf(stderr, "File output options:\n");
(void)fprintf(stderr, "-b:\tMacBinary (default)\n");
if(!mode_s_restricted) {
(void)fprintf(stderr, "-s:\tMacBinary stream to standard output\n");
#ifdef SCAN
(void)fprintf(stderr,
"-S:\tas -s but with indication of orignal Unix filename\n");
#endif /* SCAN */
}
(void)fprintf(stderr, "-f:\tthree fork mode, skipping empty forks\n");
(void)fprintf(stderr, "-3:\tthe same, writing also empty forks\n");
if(!mode_restricted) {
(void)fprintf(stderr, "-r:\tresource forks only\n");
(void)fprintf(stderr, "-d:\tdata forks only\n");
(void)fprintf(stderr,
"-u:\tdata forks only with Mac -> Unix text file translation\n");
(void)fprintf(stderr,
"-U:\tas -u, but filename will not have an extension\n");
}
#ifdef APPLESHARE
#ifdef AUFS
(void)fprintf(stderr, "-a:\tAUFS format\n");
#endif /* AUFS */
#ifdef APPLEDOUBLE
(void)fprintf(stderr, "-a:\tAppleDouble format\n");
#endif /* APPLEDOUBLE */
#else /* APPLESHARE */
(void)fprintf(stderr, "-a:\tnot supported, needs recompilation\n");
#endif /* APPLESHARE */
}
void set_wrfileopt(int restricted)
{
mode_restricted = restricted;
}
void set_s_wrfileopt(int restricted)
{
mode_s_restricted = restricted;
}
char *get_wrfileopt()
{
static char options[20];
(void)strcpy(options, "b");
if(!mode_s_restricted) {
(void)strcat(options, "s");
#ifdef SCAN
(void)strcat(options, "S");
#endif /* SCAN */
}
(void)strcat(options, "f3");
if(!mode_restricted) {
(void)strcat(options, "rduU");
}
(void)strcat(options, "a");
return options;
}
char *get_mina(void)
{
#ifdef APPLESHARE
#ifdef AUFS
return ", AUFS supported";
#endif /* AUFS */
#ifdef APPLEDOUBLE
return ", AppleDouble supported";
#endif /* APPLEDOUBLE */
#else /* APPLESHARE */
return ", no Apple-Unix sharing supported";
#endif /* APPLESHARE */
}