mirror of
git://sourceware.org/git/newlib-cygwin.git
synced 2025-01-18 20:39:33 +08:00
cf157504a8
(get_long_name): Correctly prototype GetLongPathName. (get_long_paths): Implement using get_long_name to cut down on code duplication. (doit): Do various things to make path output work predictably.
768 lines
16 KiB
C++
768 lines
16 KiB
C++
/* cygpath.cc -- convert pathnames between Windows and Unix format
|
|
Copyright 1998, 1999, 2000, 2001, 2002 Red Hat, Inc.
|
|
|
|
This file is part of Cygwin.
|
|
|
|
This software is a copyrighted work licensed under the terms of the
|
|
Cygwin license. Please consult the file "CYGWIN_LICENSE" for
|
|
details. */
|
|
|
|
#define NOCOMATTRIBUTE
|
|
|
|
#include <shlobj.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <limits.h>
|
|
#include <getopt.h>
|
|
#include <windows.h>
|
|
#include <io.h>
|
|
#include <sys/fcntl.h>
|
|
#include <sys/cygwin.h>
|
|
#include <ctype.h>
|
|
|
|
static const char version[] = "$Revision$";
|
|
|
|
static char *prog_name;
|
|
static char *file_arg;
|
|
static int path_flag, unix_flag, windows_flag, absolute_flag;
|
|
static int shortname_flag, longname_flag;
|
|
static int ignore_flag, allusers_flag, output_flag;
|
|
static int mixed_flag;
|
|
static const char *format_type_arg;
|
|
|
|
static struct option long_options[] = {
|
|
{(char *) "absolute", no_argument, NULL, 'a'},
|
|
{(char *) "close", required_argument, NULL, 'c'},
|
|
{(char *) "dos", no_argument, NULL, 'd'},
|
|
{(char *) "file", required_argument, NULL, 'f'},
|
|
{(char *) "help", no_argument, NULL, 'h'},
|
|
{(char *) "ignore", no_argument, NULL, 'i'},
|
|
{(char *) "long-name", no_argument, NULL, 'l'},
|
|
{(char *) "mixed", no_argument, NULL, 'm'},
|
|
{(char *) "option", no_argument, NULL, 'o'},
|
|
{(char *) "path", no_argument, NULL, 'p'},
|
|
{(char *) "short-name", no_argument, NULL, 's'},
|
|
{(char *) "type", required_argument, NULL, 't'},
|
|
{(char *) "unix", no_argument, NULL, 'u'},
|
|
{(char *) "version", no_argument, NULL, 'v'},
|
|
{(char *) "windows", no_argument, NULL, 'w'},
|
|
{(char *) "allusers", no_argument, NULL, 'A'},
|
|
{(char *) "desktop", no_argument, NULL, 'D'},
|
|
{(char *) "homeroot", no_argument, NULL, 'H'},
|
|
{(char *) "smprograms", no_argument, NULL, 'P'},
|
|
{(char *) "sysdir", no_argument, NULL, 'S'},
|
|
{(char *) "windir", no_argument, NULL, 'W'},
|
|
{0, no_argument, 0, 0}
|
|
};
|
|
|
|
static char options[] = "ac:df:hilmopst:uvwADHPSW";
|
|
|
|
static void
|
|
usage (FILE * stream, int status)
|
|
{
|
|
if (!ignore_flag || !status)
|
|
fprintf (stream, "\
|
|
Usage: %s (-d|-m|-u|-w|-t TYPE) [-c HANDLE] [-f FILE] [options] NAME\n\
|
|
%s [-ADHPSW] \n\
|
|
Output type options:\n\
|
|
-d, --dos print DOS (short) form of NAME (C:\\PROGRA~1\\)\n\
|
|
-m, --mixed like --windows, but with regular slashes (C:/WINNT)\n\
|
|
-u, --unix (default) print Unix form of NAME (/cygdrive/c/winnt)\n\
|
|
-w, --windows print Windows form of NAME (C:\\WINNT)\n\
|
|
-t, --type TYPE print TYPE form: 'dos', 'mixed', 'unix', or 'windows'\n\
|
|
Path conversion options:\n\
|
|
-a, --absolute output absolute path\n\
|
|
-l, --long-name print Windows long form of NAME (with -w, -m only)\n\
|
|
-p, --path NAME is a PATH list (i.e., '/bin:/usr/bin')\n\
|
|
-s, --short-name print DOS (short) form of NAME (with -w, -m only)\n\
|
|
System information:\n\
|
|
-A, --allusers use `All Users' instead of current user for -D, -P\n\
|
|
-D, --desktop output `Desktop' directory and exit\n\
|
|
-H, --homeroot output `Profiles' directory (home root) and exit\n\
|
|
-P, --smprograms output Start Menu `Programs' directory and exit\n\
|
|
-S, --sysdir output system directory and exit\n\
|
|
-W, --windir output `Windows' directory and exit\n\
|
|
", prog_name, prog_name);
|
|
if (stream == stdout)
|
|
{
|
|
fprintf (stream, "\
|
|
Other options:\n\
|
|
-f, --file FILE read FILE for input; use - to read from STDIN\n\
|
|
-o, --option read options from FILE as well (for use with --file)\n\
|
|
-c, --close HANDLE close HANDLE (for use in captured process)\n\
|
|
-i, --ignore ignore missing argument\n\
|
|
-h, --help output usage information and exit\n\
|
|
-v, --version output version information and exit\n\
|
|
");
|
|
}
|
|
else
|
|
fprintf(stream, "Try `%s --help' for more information.\n", prog_name);
|
|
exit (ignore_flag ? 0 : status);
|
|
}
|
|
|
|
static char *
|
|
get_short_paths (char *path)
|
|
{
|
|
char *sbuf;
|
|
char *sptr;
|
|
char *next;
|
|
char *ptr = path;
|
|
char *end = strrchr (path, 0);
|
|
DWORD acc = 0;
|
|
DWORD len;
|
|
|
|
while (ptr != NULL)
|
|
{
|
|
next = ptr;
|
|
ptr = strchr (ptr, ';');
|
|
if (ptr)
|
|
*ptr++ = 0;
|
|
len = GetShortPathName (next, NULL, 0);
|
|
if (len == 0 && GetLastError () == ERROR_INVALID_PARAMETER)
|
|
{
|
|
fprintf (stderr, "%s: cannot create short name of %s\n", prog_name,
|
|
next);
|
|
exit (2);
|
|
}
|
|
acc += len + 1;
|
|
}
|
|
sptr = sbuf = (char *) malloc (acc + 1);
|
|
if (sbuf == NULL)
|
|
{
|
|
fprintf (stderr, "%s: out of memory\n", prog_name);
|
|
exit (1);
|
|
}
|
|
ptr = path;
|
|
for (;;)
|
|
{
|
|
len = GetShortPathName (ptr, sptr, acc);
|
|
if (len == 0 && GetLastError () == ERROR_INVALID_PARAMETER)
|
|
{
|
|
fprintf (stderr, "%s: cannot create short name of %s\n", prog_name,
|
|
ptr);
|
|
exit (2);
|
|
}
|
|
|
|
ptr = strrchr (ptr, 0);
|
|
sptr = strrchr (sptr, 0);
|
|
if (ptr == end)
|
|
break;
|
|
*sptr = ';';
|
|
++ptr, ++sptr;
|
|
acc -= len + 1;
|
|
}
|
|
return sbuf;
|
|
}
|
|
|
|
static char *
|
|
get_short_name (const char *filename)
|
|
{
|
|
char *sbuf, buf[MAX_PATH];
|
|
DWORD len = GetShortPathName (filename, buf, MAX_PATH);
|
|
if (len == 0 && GetLastError () == ERROR_INVALID_PARAMETER)
|
|
{
|
|
fprintf (stderr, "%s: cannot create short name of %s\n", prog_name,
|
|
filename);
|
|
exit (2);
|
|
}
|
|
sbuf = (char *) malloc (++len);
|
|
if (sbuf == NULL)
|
|
{
|
|
fprintf (stderr, "%s: out of memory\n", prog_name);
|
|
exit (1);
|
|
}
|
|
return strcpy (sbuf, buf);
|
|
}
|
|
|
|
static DWORD WINAPI
|
|
get_long_path_name_w32impl (LPCSTR src, LPSTR sbuf, DWORD)
|
|
{
|
|
char buf1[MAX_PATH], buf2[MAX_PATH], *ptr;
|
|
const char *pelem, *next;
|
|
WIN32_FIND_DATA w32_fd;
|
|
int len;
|
|
|
|
strcpy (buf1, src);
|
|
*buf2 = 0;
|
|
pelem = src;
|
|
ptr = buf2;
|
|
while (pelem)
|
|
{
|
|
next = pelem;
|
|
if (*next == '\\')
|
|
{
|
|
strcat (ptr++, "\\");
|
|
pelem++;
|
|
if (!*pelem)
|
|
break;
|
|
continue;
|
|
}
|
|
pelem = strchr (next, '\\');
|
|
len = pelem ? (pelem++ - next) : strlen (next);
|
|
strncpy (ptr, next, len);
|
|
ptr[len] = 0;
|
|
if (next[1] != ':' && strcmp(next, ".") && strcmp(next, ".."))
|
|
{
|
|
if (FindFirstFile (buf2, &w32_fd) != INVALID_HANDLE_VALUE)
|
|
strcpy (ptr, w32_fd.cFileName);
|
|
}
|
|
ptr += strlen (ptr);
|
|
if (pelem)
|
|
{
|
|
*ptr++ = '\\';
|
|
*ptr = 0;
|
|
}
|
|
}
|
|
if (sbuf)
|
|
strcpy (sbuf, buf2);
|
|
SetLastError (0);
|
|
return strlen (buf2) + (sbuf ? 0 : 1);
|
|
}
|
|
|
|
static char *
|
|
get_long_name (const char *filename, DWORD& len)
|
|
{
|
|
char *sbuf, buf[MAX_PATH];
|
|
static HINSTANCE k32 = LoadLibrary ("kernel32.dll");
|
|
static DWORD (WINAPI *GetLongPathName) (LPCSTR, LPSTR, DWORD) =
|
|
(DWORD (WINAPI *) (LPCSTR, LPSTR, DWORD)) GetProcAddress (k32, "GetLongPathName");
|
|
if (!GetLongPathName)
|
|
GetLongPathName = get_long_path_name_w32impl;
|
|
|
|
len = GetLongPathName (filename, buf, MAX_PATH);
|
|
if (len == 0 && GetLastError () == ERROR_INVALID_PARAMETER)
|
|
{
|
|
fprintf (stderr, "%s: cannot create long name of %s\n", prog_name,
|
|
filename);
|
|
exit (2);
|
|
}
|
|
sbuf = (char *) malloc (len + 1);
|
|
if (!sbuf)
|
|
{
|
|
fprintf (stderr, "%s: out of memory\n", prog_name);
|
|
exit (1);
|
|
}
|
|
return strcpy (sbuf, buf);
|
|
}
|
|
|
|
static char *
|
|
get_long_paths (char *path)
|
|
{
|
|
char *sbuf;
|
|
char *ptr;
|
|
int n = 1;
|
|
|
|
ptr = path;
|
|
while ((ptr = strchr (ptr, ';')))
|
|
{
|
|
ptr++;
|
|
n++;
|
|
}
|
|
|
|
char *paths[n];
|
|
DWORD acc = 0;
|
|
int i;
|
|
if (!n)
|
|
return strdup ("");
|
|
|
|
for (i = 0, ptr = path; ptr; i++)
|
|
{
|
|
DWORD len;
|
|
char *next = ptr;
|
|
ptr = strchr (ptr, ';');
|
|
if (ptr)
|
|
*ptr++ = 0;
|
|
paths[i] = get_long_name (next, len);
|
|
acc += len + 1;
|
|
}
|
|
|
|
sbuf = (char *) malloc (acc + 1);
|
|
if (sbuf == NULL)
|
|
{
|
|
fprintf (stderr, "%s: out of memory\n", prog_name);
|
|
exit (1);
|
|
}
|
|
|
|
sbuf[0] = '\0';
|
|
for (i = 0; i < n; i++)
|
|
{
|
|
strcat (strcat (sbuf, paths[i]), ";");
|
|
free (paths[i]);
|
|
}
|
|
|
|
strchr (sbuf, '\0')[-1] = '\0';
|
|
return sbuf;
|
|
}
|
|
|
|
static void
|
|
convert_slashes (char* name)
|
|
{
|
|
while ((name = strchr (name, '\\')) != NULL)
|
|
{
|
|
if (*name == '\\')
|
|
*name = '/';
|
|
name++;
|
|
}
|
|
}
|
|
|
|
static char *
|
|
get_mixed_name (const char* filename)
|
|
{
|
|
char* mixed_buf = strdup (filename);
|
|
|
|
if (mixed_buf == NULL)
|
|
{
|
|
fprintf (stderr, "%s: out of memory\n", prog_name);
|
|
exit (1);
|
|
}
|
|
|
|
convert_slashes (mixed_buf);
|
|
|
|
return mixed_buf;
|
|
}
|
|
|
|
static void
|
|
dowin (char option)
|
|
{
|
|
char *buf, buf1[MAX_PATH], buf2[MAX_PATH];
|
|
DWORD len = MAX_PATH;
|
|
WIN32_FIND_DATA w32_fd;
|
|
LPITEMIDLIST id;
|
|
HINSTANCE k32;
|
|
BOOL (*GetProfilesDirectoryAPtr) (LPSTR, LPDWORD) = 0;
|
|
|
|
buf = buf1;
|
|
switch (option)
|
|
{
|
|
case 'D':
|
|
SHGetSpecialFolderLocation (NULL, allusers_flag ?
|
|
CSIDL_COMMON_DESKTOPDIRECTORY : CSIDL_DESKTOPDIRECTORY, &id);
|
|
SHGetPathFromIDList (id, buf);
|
|
/* This if clause is a Fix for Win95 without any "All Users" */
|
|
if (strlen (buf) == 0)
|
|
{
|
|
SHGetSpecialFolderLocation (NULL, CSIDL_DESKTOPDIRECTORY, &id);
|
|
SHGetPathFromIDList (id, buf);
|
|
}
|
|
break;
|
|
|
|
case 'P':
|
|
SHGetSpecialFolderLocation (NULL, allusers_flag ?
|
|
CSIDL_COMMON_PROGRAMS : CSIDL_PROGRAMS, &id);
|
|
SHGetPathFromIDList (id, buf);
|
|
/* This if clause is a Fix for Win95 without any "All Users" */
|
|
if (strlen (buf) == 0)
|
|
{
|
|
SHGetSpecialFolderLocation (NULL, CSIDL_PROGRAMS, &id);
|
|
SHGetPathFromIDList (id, buf);
|
|
}
|
|
break;
|
|
|
|
case 'H':
|
|
k32 = LoadLibrary ("userenv");
|
|
if (k32)
|
|
GetProfilesDirectoryAPtr = (BOOL (*) (LPSTR, LPDWORD))
|
|
GetProcAddress (k32, "GetProfilesDirectoryA");
|
|
if (GetProfilesDirectoryAPtr)
|
|
(*GetProfilesDirectoryAPtr) (buf, &len);
|
|
else
|
|
{
|
|
GetWindowsDirectory (buf, MAX_PATH);
|
|
strcat (buf, "\\Profiles");
|
|
}
|
|
break;
|
|
|
|
case 'S':
|
|
GetSystemDirectory (buf, MAX_PATH);
|
|
FindFirstFile (buf, &w32_fd);
|
|
strcpy (strrchr (buf, '\\') + 1, w32_fd.cFileName);
|
|
break;
|
|
|
|
case 'W':
|
|
GetWindowsDirectory (buf, MAX_PATH);
|
|
break;
|
|
|
|
default:
|
|
usage (stderr, 1);
|
|
}
|
|
|
|
if (!windows_flag)
|
|
{
|
|
cygwin_conv_to_posix_path (buf, buf2);
|
|
buf = buf2;
|
|
}
|
|
else
|
|
{
|
|
if (shortname_flag)
|
|
buf = get_short_name (buf);
|
|
if (mixed_flag)
|
|
buf = get_mixed_name (buf);
|
|
}
|
|
printf ("%s\n", buf);
|
|
exit (0);
|
|
}
|
|
|
|
static void
|
|
doit (char *filename)
|
|
{
|
|
char *buf;
|
|
DWORD len;
|
|
int retval;
|
|
int (*conv_func) (const char *, char *);
|
|
|
|
if (!path_flag)
|
|
{
|
|
len = strlen (filename);
|
|
if (len)
|
|
len += 100;
|
|
else if (ignore_flag)
|
|
exit (0);
|
|
else
|
|
{
|
|
fprintf (stderr, "%s: can't convert empty path\n", prog_name);
|
|
exit (1);
|
|
}
|
|
}
|
|
else if (unix_flag)
|
|
len = cygwin_win32_to_posix_path_list_buf_size (filename);
|
|
else
|
|
len = cygwin_posix_to_win32_path_list_buf_size (filename);
|
|
|
|
buf = (char *) malloc (len);
|
|
if (buf == NULL)
|
|
{
|
|
fprintf (stderr, "%s: out of memory\n", prog_name);
|
|
exit (1);
|
|
}
|
|
|
|
if (path_flag)
|
|
{
|
|
if (unix_flag)
|
|
cygwin_win32_to_posix_path_list (filename, buf);
|
|
else
|
|
{
|
|
cygwin_posix_to_win32_path_list (filename, buf);
|
|
if (shortname_flag)
|
|
buf = get_short_paths (buf);
|
|
if (longname_flag)
|
|
buf = get_long_paths (buf);
|
|
if (mixed_flag)
|
|
buf = get_mixed_name (buf);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (unix_flag)
|
|
conv_func = (absolute_flag ? cygwin_conv_to_full_posix_path :
|
|
cygwin_conv_to_posix_path);
|
|
else
|
|
conv_func = (absolute_flag ? cygwin_conv_to_full_win32_path :
|
|
cygwin_conv_to_win32_path);
|
|
retval = conv_func (filename, buf);
|
|
if (mixed_flag)
|
|
buf = get_mixed_name (buf);
|
|
if (retval < 0)
|
|
{
|
|
fprintf (stderr, "%s: error converting \"%s\"\n",
|
|
prog_name, filename);
|
|
exit (1);
|
|
}
|
|
if (!unix_flag)
|
|
{
|
|
if (shortname_flag)
|
|
buf = get_short_name (buf);
|
|
if (longname_flag)
|
|
buf = get_long_name (buf, len);
|
|
}
|
|
}
|
|
|
|
puts (buf);
|
|
}
|
|
|
|
static void
|
|
print_version ()
|
|
{
|
|
const char *v = strchr (version, ':');
|
|
int len;
|
|
if (!v)
|
|
{
|
|
v = "?";
|
|
len = 1;
|
|
}
|
|
else
|
|
{
|
|
v += 2;
|
|
len = strchr (v, ' ') - v;
|
|
}
|
|
printf ("\
|
|
cygpath (cygwin) %.*s\n\
|
|
Path Conversion Utility\n\
|
|
Copyright 1998, 1999, 2000, 2001, 2002 Red Hat, Inc.\n\
|
|
Compiled on %s\n\
|
|
", len, v, __DATE__);
|
|
}
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
int c, o = 0;
|
|
int options_from_file_flag;
|
|
char *filename;
|
|
|
|
prog_name = strrchr (argv[0], '/');
|
|
if (prog_name == NULL)
|
|
prog_name = strrchr (argv[0], '\\');
|
|
if (prog_name == NULL)
|
|
prog_name = argv[0];
|
|
else
|
|
prog_name++;
|
|
|
|
path_flag = 0;
|
|
unix_flag = 1;
|
|
windows_flag = 0;
|
|
shortname_flag = 0;
|
|
longname_flag = 0;
|
|
mixed_flag = 0;
|
|
ignore_flag = 0;
|
|
options_from_file_flag = 0;
|
|
allusers_flag = 0;
|
|
output_flag = 0;
|
|
while ((c = getopt_long (argc, argv, options,
|
|
long_options, (int *) NULL)) != EOF)
|
|
{
|
|
switch (c)
|
|
{
|
|
case 'a':
|
|
absolute_flag = 1;
|
|
break;
|
|
|
|
case 'c':
|
|
CloseHandle ((HANDLE) strtoul (optarg, NULL, 16));
|
|
break;
|
|
|
|
case 'd':
|
|
if (windows_flag)
|
|
usage (stderr, 1);
|
|
unix_flag = 0;
|
|
windows_flag = 1;
|
|
shortname_flag = 1;
|
|
break;
|
|
|
|
case 'f':
|
|
file_arg = optarg;
|
|
break;
|
|
|
|
case 'o':
|
|
options_from_file_flag = 1;
|
|
break;
|
|
|
|
case 'p':
|
|
path_flag = 1;
|
|
break;
|
|
|
|
case 'u':
|
|
if (windows_flag || mixed_flag)
|
|
usage (stderr, 1);
|
|
unix_flag = 1;
|
|
break;
|
|
|
|
case 'w':
|
|
if (windows_flag || mixed_flag)
|
|
usage (stderr, 1);
|
|
unix_flag = 0;
|
|
windows_flag = 1;
|
|
break;
|
|
|
|
case 'm':
|
|
unix_flag = 0;
|
|
windows_flag = 1;
|
|
mixed_flag = 1;
|
|
break;
|
|
|
|
case 'l':
|
|
longname_flag = 1;
|
|
break;
|
|
|
|
case 's':
|
|
shortname_flag = 1;
|
|
break;
|
|
|
|
case 't':
|
|
if (optarg == NULL)
|
|
usage (stderr, 1);
|
|
|
|
format_type_arg = (*optarg == '=') ? (optarg + 1) : (optarg);
|
|
if (strcasecmp (format_type_arg, "dos") == 0)
|
|
{
|
|
if (windows_flag || longname_flag)
|
|
usage (stderr, 1);
|
|
unix_flag = 0;
|
|
windows_flag = 1;
|
|
shortname_flag = 1;
|
|
}
|
|
else if (strcasecmp (format_type_arg, "mixed") == 0)
|
|
{
|
|
unix_flag = 0;
|
|
mixed_flag = 1;
|
|
}
|
|
else if (strcasecmp (format_type_arg, "unix") == 0)
|
|
{
|
|
if (windows_flag)
|
|
usage (stderr, 1);
|
|
unix_flag = 1;
|
|
}
|
|
else if (strcasecmp (format_type_arg, "windows") == 0)
|
|
{
|
|
if (mixed_flag)
|
|
usage (stderr, 1);
|
|
unix_flag = 0;
|
|
windows_flag = 1;
|
|
}
|
|
else
|
|
usage (stderr, 1);
|
|
break;
|
|
|
|
case 'A':
|
|
allusers_flag = 1;
|
|
break;
|
|
|
|
case 'D':
|
|
case 'H':
|
|
case 'P':
|
|
case 'S':
|
|
case 'W':
|
|
if (output_flag)
|
|
usage (stderr, 1);
|
|
output_flag = 1;
|
|
o = c;
|
|
break;
|
|
|
|
case 'i':
|
|
ignore_flag = 1;
|
|
break;
|
|
|
|
case 'h':
|
|
usage (stdout, 0);
|
|
break;
|
|
|
|
case 'v':
|
|
print_version ();
|
|
exit (0);
|
|
|
|
default:
|
|
usage (stderr, 1);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (options_from_file_flag && !file_arg)
|
|
usage (stderr, 1);
|
|
|
|
if (longname_flag && !windows_flag)
|
|
usage (stderr, 1);
|
|
|
|
if (shortname_flag && !windows_flag)
|
|
usage (stderr, 1);
|
|
|
|
if (!unix_flag && !windows_flag && !mixed_flag && !options_from_file_flag)
|
|
usage (stderr, 1);
|
|
|
|
if (!file_arg)
|
|
{
|
|
if (output_flag)
|
|
dowin (o);
|
|
|
|
if (optind != argc - 1)
|
|
usage (stderr, 1);
|
|
|
|
filename = argv[optind];
|
|
doit (filename);
|
|
}
|
|
else
|
|
{
|
|
FILE *fp;
|
|
char buf[PATH_MAX * 2 + 1];
|
|
|
|
if (argv[optind])
|
|
usage (stderr, 1);
|
|
|
|
if (strcmp (file_arg, "-") != 0)
|
|
fp = fopen (file_arg, "rt");
|
|
else
|
|
{
|
|
fp = stdin;
|
|
setmode (0, O_TEXT);
|
|
}
|
|
if (fp == NULL)
|
|
{
|
|
perror ("cygpath");
|
|
exit (1);
|
|
}
|
|
|
|
setbuf (stdout, NULL);
|
|
while (fgets (buf, sizeof (buf), fp) != NULL)
|
|
{
|
|
char *s = buf;
|
|
char *p = strchr (s, '\n');
|
|
if (p)
|
|
*p = '\0';
|
|
if (options_from_file_flag && *s == '-')
|
|
{
|
|
char c;
|
|
for (c = *++s; c && !isspace (c); c = *++s)
|
|
switch (c)
|
|
{
|
|
case 'a':
|
|
absolute_flag = 1;
|
|
break;
|
|
case 'i':
|
|
ignore_flag = 1;
|
|
break;
|
|
case 's':
|
|
shortname_flag = 1;
|
|
longname_flag = 0;
|
|
break;
|
|
case 'l':
|
|
shortname_flag = 0;
|
|
longname_flag = 1;
|
|
break;
|
|
case 'm':
|
|
unix_flag = 0;
|
|
windows_flag = 1;
|
|
mixed_flag = 1;
|
|
case 'w':
|
|
unix_flag = 0;
|
|
windows_flag = 1;
|
|
break;
|
|
case 'u':
|
|
windows_flag = 0;
|
|
unix_flag = 1;
|
|
break;
|
|
case 'p':
|
|
path_flag = 1;
|
|
break;
|
|
case 'D':
|
|
case 'H':
|
|
case 'P':
|
|
case 'S':
|
|
case 'W':
|
|
output_flag = 1;
|
|
o = c;
|
|
break;
|
|
}
|
|
if (*s)
|
|
do
|
|
s++;
|
|
while (*s && isspace (*s));
|
|
}
|
|
if (*s && !output_flag)
|
|
doit (s);
|
|
if (!*s && output_flag)
|
|
dowin (o);
|
|
}
|
|
}
|
|
|
|
exit (0);
|
|
}
|