win32.c
资源名称:unzip540.zip [点击查看]
上传用户:andy_li
上传日期:2007-01-06
资源大小:1019k
文件大小:84k
源码类别:
压缩解压
开发平台:
MultiPlatform
- /*---------------------------------------------------------------------------
- win32.c
- 32-bit Windows-specific (NT/95) routines for use with Info-ZIP's UnZip 5.3
- and later.
- Contains: GetLoadPath()
- Opendir()
- Readdir()
- Closedir()
- process_defer_NT() process any deferred items
- SetSD() set security descriptor on file
- EvalExtraFields() evaluate and process and extra field NOW
- IsWinNT() indicate type of WIN32 platform
- test_NTSD() test integrity of NT security data
- utime2FileTime()
- FStampIsLocTime()
- FileTime2utime()
- VFatFileTime2utime()
- UTCtime2Localtime()
- NTtzbugWorkaround()
- getNTfiletime()
- close_outfile()
- stamp_file()
- isfloppy()
- NTQueryVolInfo()
- IsVolumeOldFAT()
- do_wild()
- mapattr()
- mapname()
- map2fat()
- checkdir()
- version()
- zstat_win32()
- getch_win32()
- ---------------------------------------------------------------------------*/
- #define UNZIP_INTERNAL
- #include "unzip.h"
- #include <windows.h> /* must be AFTER unzip.h to avoid struct G problems */
- #ifdef __RSXNT__
- # include "win32/rsxntwin.h"
- #endif
- #include "win32/nt.h"
- #ifndef FUNZIP /* most of this file is not used with fUnZip */
- #if (defined(__GO32__) || defined(__EMX__) || defined(__CYGWIN32__))
- # define MKDIR(path,mode) mkdir(path,mode)
- #else
- # define MKDIR(path,mode) mkdir(path)
- #endif
- #ifdef HAVE_WORKING_DIRENT_H
- # undef HAVE_WORKING_DIRENT_H
- #endif
- /* The emxrtl dirent support of (__GO32__ || __EMX__) converts to lowercase! */
- #if defined(__CYGWIN32__)
- # define HAVE_WORKING_DIRENT_H
- #endif
- #ifndef SFX
- # ifdef HAVE_WORKING_DIRENT_H
- # include <dirent.h> /* use readdir() */
- # define zdirent dirent
- # define zDIR DIR
- # define Opendir opendir
- # define Readdir readdir
- # define Closedir closedir
- # else /* !HAVE_WORKING_DIRENT_H */
- typedef struct zdirent {
- char reserved [21];
- char ff_attrib;
- short ff_ftime;
- short ff_fdate;
- long size;
- char d_name[MAX_PATH];
- int d_first;
- HANDLE d_hFindFile;
- } zDIR;
- static zDIR *Opendir (const char *n);
- static struct zdirent *Readdir (zDIR *d);
- static void Closedir (zDIR *d);
- # endif /* ?HAVE_WORKING_DIRENT_H */
- #endif /* !SFX */
- /* Function prototypes */
- #ifdef NTSD_EAS
- static int SetSD(__GPRO__ char *path, PVOLUMECAPS VolumeCaps,
- uch *eb_ptr, unsigned eb_len);
- static int EvalExtraFields(__GPRO__ char *path, uch *ef_ptr,
- unsigned ef_len);
- #endif
- #if (defined(USE_EF_UT_TIME) || defined(NT_TZBUG_WORKAROUND) ||
- defined(TIMESTAMP))
- static void utime2FileTime(time_t ut, FILETIME *pft);
- static int FStampIsLocTime(__GPRO__ const char *path);
- #endif /* USE_EF_UT_TIME || NT_TZBUG_WORKAROUND || TIMESTAMP */
- #ifdef NT_TZBUG_WORKAROUND
- static int FileTime2utime(const FILETIME *pft, time_t *ut);
- #ifdef W32_STAT_BANDAID
- static int VFatFileTime2utime(const FILETIME *pft, time_t *ut);
- #endif
- static time_t UTCtime2Localtime(time_t utctime);
- static void NTtzbugWorkaround(time_t ut, FILETIME *pft);
- #endif /* NT_TZBUG_WORKAROUND */
- static int getNTfiletime (__GPRO__ FILETIME *pModFT, FILETIME *pAccFT,
- FILETIME *pCreFT);
- static int isfloppy (int nDrive);
- static int NTQueryVolInfo (__GPRO__ const char *name);
- static int IsVolumeOldFAT (__GPRO__ const char *name);
- static void map2fat (char *pathcomp, char **pEndFAT);
- #ifdef __MINGW32__
- int _CRT_glob = 0; /* suppress command line globbing by C RTL */
- #endif
- /* static int created_dir; */ /* used by mapname(), checkdir() */
- /* static int renamed_fullpath; */ /* ditto */
- /* static int fnlen; */ /* ditto */
- /* static unsigned nLabelDrive; */ /* ditto */
- extern char Far TruncNTSD[]; /* in extract.c */
- #ifdef SFX
- /**************************/
- /* Function GetLoadPath() */
- /**************************/
- char *GetLoadPath(__GPRO)
- {
- #ifdef MSC
- extern char *_pgmptr;
- return _pgmptr;
- #else /* use generic API call */
- GetModuleFileName(NULL, G.filename, FILNAMSIZ-1);
- _ISO_INTERN(G.filename); /* translate to codepage of C rtl's stdio */
- return G.filename;
- #endif
- } /* end function GetLoadPath() */
- #else /* !SFX */
- #ifndef HAVE_WORKING_DIRENT_H
- /**********************/ /* Borrowed from ZIP 2.0 sources */
- /* Function Opendir() */ /* Difference: no special handling for */
- /**********************/ /* hidden or system files. */
- static zDIR *Opendir(n)
- const char *n; /* directory to open */
- {
- zDIR *d; /* malloc'd return value */
- char *p; /* malloc'd temporary string */
- WIN32_FIND_DATA fd;
- extent len = strlen(n);
- /* Start searching for files in directory n */
- if ((d = (zDIR *)malloc(sizeof(zDIR))) == NULL ||
- (p = malloc(strlen(n) + 5)) == NULL)
- {
- if (d != (zDIR *)NULL)
- free((void *)d);
- return (zDIR *)NULL;
- }
- INTERN_TO_ISO(n, p);
- if (len > 0) {
- if (p[len-1] == ':')
- p[len++] = '.'; /* x: => x:. */
- else if (p[len-1] == '/' || p[len-1] == '\')
- --len; /* foo/ => foo */
- }
- strcpy(p+len, "/*");
- if (INVALID_HANDLE_VALUE == (d->d_hFindFile = FindFirstFile(p, &fd))) {
- free((zvoid *)d);
- free((zvoid *)p);
- return NULL;
- }
- strcpy(d->d_name, fd.cFileName);
- free((zvoid *)p);
- d->d_first = 1;
- return d;
- } /* end of function Opendir() */
- /**********************/ /* Borrowed from ZIP 2.0 sources */
- /* Function Readdir() */ /* Difference: no special handling for */
- /**********************/ /* hidden or system files. */
- static struct zdirent *Readdir(d)
- zDIR *d; /* directory stream from which to read */
- {
- /* Return pointer to first or next directory entry, or NULL if end. */
- if ( d->d_first )
- d->d_first = 0;
- else
- {
- WIN32_FIND_DATA fd;
- if ( !FindNextFile(d->d_hFindFile, &fd) )
- return NULL;
- ISO_TO_INTERN(fd.cFileName, d->d_name);
- }
- return (struct zdirent *)d;
- } /* end of function Readdir() */
- /***********************/
- /* Function Closedir() */ /* Borrowed from ZIP 2.0 sources */
- /***********************/
- static void Closedir(d)
- zDIR *d; /* directory stream to close */
- {
- FindClose(d->d_hFindFile);
- free(d);
- }
- #endif /* !HAVE_WORKING_DIRENT_H */
- #endif /* ?SFX */
- #ifdef NTSD_EAS
- /*********************************/
- /* Function process_defer_NT() */
- /*********************************/
- void process_defer_NT(__G)
- __GDEF
- {
- /* process deferred items */
- DWORD dir, bytes;
- DWORD dirfail, bytesfail;
- ProcessDefer(&dir, &bytes, &dirfail, &bytesfail);
- if (!uO.tflag && (uO.qflag < 2)) {
- if (dir)
- Info(slide, 0, ((char *)slide,
- " updated: %lu directory entries with %lu bytes security",
- (ulg)dir, (ulg)bytes));
- if (dirfail)
- Info(slide, 0, ((char *)slide,
- " failed: %lu directory entries with %lu bytes security",
- (ulg)dirfail, (ulg)bytesfail));
- }
- }
- /**********************/
- /* Function SetSD() */ /* return almost-PK errors */
- /**********************/
- static int SetSD(__G__ path, VolumeCaps, eb_ptr, eb_len)
- __GDEF
- char *path;
- PVOLUMECAPS VolumeCaps;
- uch *eb_ptr;
- unsigned eb_len;
- {
- ulg ntsd_ucSize;
- uch *security_data;
- int error;
- if (eb_ptr == NULL || eb_len < EB_NTSD_L_LEN)
- return PK_OK; /* not a valid NTSD extra field: assume OK */
- /* check if we know how to handle this version */
- if (*(eb_ptr + (EB_HEADSIZE+EB_NTSD_VERSION)) > (uch)EB_NTSD_MAX_VER)
- return PK_OK;
- ntsd_ucSize = makelong(eb_ptr + (EB_HEADSIZE+EB_UCSIZE_P));
- if (ntsd_ucSize > 0L && eb_len <= (EB_NTSD_L_LEN + EB_CMPRHEADLEN))
- return IZ_EF_TRUNC; /* no compressed data! */
- /* allocate storage for uncompressed data */
- security_data = (uch *)malloc((extent)ntsd_ucSize);
- if (security_data == (uch *)NULL)
- return PK_MEM4;
- error = memextract(__G__ security_data, ntsd_ucSize,
- (eb_ptr + (EB_HEADSIZE+EB_NTSD_L_LEN)), (ulg)(eb_len - EB_NTSD_L_LEN));
- if (error == PK_OK) {
- if (SecuritySet(path, VolumeCaps, security_data)) {
- error = PK_COOL;
- if (!uO.tflag && (uO.qflag < 2) &&
- (!(VolumeCaps->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)))
- Info(slide, 0, ((char *)slide, " (%ld bytes security)",
- ntsd_ucSize));
- }
- }
- free(security_data);
- return error;
- }
- /********************************/ /* scan extra fields for something */
- /* Function EvalExtraFields() */ /* we happen to know */
- /********************************/
- static int EvalExtraFields(__G__ path, ef_ptr, ef_len)
- __GDEF
- char *path;
- uch *ef_ptr;
- unsigned ef_len;
- {
- int rc = PK_OK;
- if (!uO.X_flag)
- return PK_OK; /* user said don't process ACLs; for now, no other
- extra block types are handled here */
- while (ef_len >= EB_HEADSIZE)
- {
- unsigned eb_id = makeword(EB_ID + ef_ptr);
- unsigned eb_len = makeword(EB_LEN + ef_ptr);
- if (eb_len > (ef_len - EB_HEADSIZE)) {
- /* discovered some extra field inconsistency! */
- Trace((stderr,
- "EvalExtraFields: block length %u > rest ef_size %un", eb_len,
- ef_len - EB_HEADSIZE));
- break;
- }
- switch (eb_id)
- {
- /* process security descriptor extra data if:
- Caller is WinNT AND
- Target local/remote drive supports acls AND
- Target file is not a directory (else we defer processing
- until later)
- */
- case EF_NTSD:
- if (IsWinNT()) {
- VOLUMECAPS VolumeCaps;
- /* provide useful input */
- VolumeCaps.dwFileAttributes = G.pInfo->file_attr;
- VolumeCaps.bUsePrivileges = (uO.X_flag > 1);
- /* check target volume capabilities - just fall through
- * and try if fail */
- if (GetVolumeCaps(G.rootpath, path, &VolumeCaps) &&
- !(VolumeCaps.dwFileSystemFlags & FS_PERSISTENT_ACLS))
- {
- rc = PK_OK;
- break;
- }
- rc = SetSD(__G__ path, &VolumeCaps, ef_ptr, eb_len);
- } else
- rc = PK_OK;
- break;
- #if 0
- /* perhaps later we can add support for unzipping OS/2 EAs to NT */
- case EF_OS2:
- rc = SetEAs(__G__ path, ef_ptr);
- break;
- case EF_PKUNIX:
- case EF_IZUNIX:
- case EF_IZUNIX2:
- case EF_TIME:
- break; /* handled elsewhere */
- #else /* ! 0 */
- #ifdef DEBUG
- case EF_AV:
- case EF_OS2:
- case EF_PKVMS:
- case EF_PKW32:
- case EF_PKUNIX:
- case EF_IZVMS:
- case EF_IZUNIX:
- case EF_IZUNIX2:
- case EF_TIME:
- case EF_MAC3:
- case EF_JLMAC:
- case EF_ZIPIT:
- case EF_VMCMS:
- case EF_MVS:
- case EF_ACL:
- case EF_BEOS:
- case EF_QDOS:
- case EF_AOSVS:
- case EF_SPARK:
- case EF_MD5:
- case EF_ASIUNIX:
- break; /* shut up for other known e.f. blocks */
- #endif /* DEBUG */
- #endif /* ? 0 */
- default:
- Trace((stderr,
- "EvalExtraFields: unknown extra field block, ID=%un",
- eb_id));
- break;
- }
- ef_ptr += (eb_len + EB_HEADSIZE);
- ef_len -= (eb_len + EB_HEADSIZE);
- if (rc != PK_OK)
- break;
- }
- return rc;
- }
- #ifndef SFX
- /**************************/
- /* Function test_NTSD() */ /* returns PK_WARN when NTSD data is invalid */
- /**************************/
- #ifdef __BORLANDC__
- /* Turn off warning about not using all parameters for this function only */
- #pragma argsused
- #endif
- int test_NTSD(__G__ eb, eb_size, eb_ucptr, eb_ucsize)
- __GDEF
- uch *eb;
- unsigned eb_size;
- uch *eb_ucptr;
- ulg eb_ucsize;
- {
- int r = PK_OK;
- if (!ValidateSecurity(eb_ucptr))
- r = PK_WARN;
- return r;
- } /* end function test_NTSD() */
- #endif /* !SFX */
- #endif /* NTSD_EAS */
- /**********************/
- /* Function IsWinNT() */
- /**********************/
- int IsWinNT(void) /* returns TRUE if real NT, FALSE if Win95 or Win32s */
- {
- static DWORD g_PlatformId = 0xFFFFFFFF; /* saved platform indicator */
- if (g_PlatformId == 0xFFFFFFFF) {
- /* note: GetVersionEx() doesn't exist on WinNT 3.1 */
- if (GetVersion() < 0x80000000)
- g_PlatformId = TRUE;
- else
- g_PlatformId = FALSE;
- }
- return (int)g_PlatformId;
- }
- /* DEBUG_TIME insertion: */
- #ifdef DEBUG_TIME
- static int show_NTFileTime(FILE *hdo, char *TTmsg, int isloc, FILETIME *pft);
- static int show_NTFileTime(FILE *hdo, char *TTmsg, int isloc, FILETIME *pft)
- {
- SYSTEMTIME w32tm;
- int rval;
- rval = FileTimeToSystemTime(pft, &w32tm);
- if (!rval) {
- fprintf(hdo, "%sn %08lX,%08lX (%s) -> Conversion failed !!!n",
- TTmsg, (ulg)(pft->dwHighDateTime), (ulg)(pft->dwLowDateTime),
- (isloc ? "local" : "UTC"));
- } else {
- fprintf(hdo, "%sn %08lx,%08lx -> %04u-%02u-%02u, %02u:%02u:%02u %sn",
- TTmsg, (ulg)(pft->dwHighDateTime), (ulg)(pft->dwLowDateTime),
- w32tm.wYear, w32tm.wMonth, w32tm.wDay, w32tm.wHour,
- w32tm.wMinute, w32tm.wSecond, (isloc ? "local" : "UTC"));
- }
- return rval;
- }
- #define FTTrace(x) show_NTFileTime x
- #else
- #define FTTrace(x)
- #endif /* DEBUG_TIME */
- /* end of TIME_DEBUG insertion */
- #if (defined(USE_EF_UT_TIME) || defined(NT_TZBUG_WORKAROUND) ||
- defined(TIMESTAMP))
- #if ((defined(__GNUC__) || defined(ULONG_LONG_MAX)) && !defined(HAVE_INT64))
- typedef long long LLONG64;
- typedef unsigned long long ULLNG64;
- # define HAVE_INT64
- #endif
- #if (defined(__WATCOMC__) && (__WATCOMC__ >= 1100) && !defined(HAVE_INT64))
- typedef __int64 LLONG64;
- typedef unsigned __int64 ULLNG64;
- # define HAVE_INT64
- #endif
- #if (defined(_MSC_VER) && (_MSC_VER >= 1100) && !defined(HAVE_INT64))
- typedef __int64 LLONG64;
- typedef unsigned __int64 ULLNG64;
- # define HAVE_INT64
- #endif
- /*****************************/
- /* Function utime2FileTime() */ /* convert Unix time_t format into the */
- /*****************************/ /* form used by SetFileTime() in NT/95 */
- #define UNIX_TIME_ZERO_HI 0x019DB1DEUL
- #define UNIX_TIME_ZERO_LO 0xD53E8000UL
- #define NT_QUANTA_PER_UNIX 10000000L
- static void utime2FileTime(time_t ut, FILETIME *pft)
- {
- #ifdef HAVE_INT64
- ULLNG64 NTtime;
- /* NT_QUANTA_PER_UNIX is small enough so that "ut * NT_QUANTA_PER_UNIX"
- * cannot overflow in 64-bit signed calculation, regardless wether "ut"
- * is signed or unsigned. */
- NTtime = ((LLONG64)ut * NT_QUANTA_PER_UNIX) +
- ((ULLNG64)UNIX_TIME_ZERO_LO + ((ULLNG64)UNIX_TIME_ZERO_HI << 32));
- pft->dwLowDateTime = (DWORD)NTtime;
- pft->dwHighDateTime = (DWORD)(NTtime >> 32);
- #else /* !HAVE_INT64 (64-bit integer arithmetics may not be supported) */
- unsigned int b1, b2, carry = 0;
- unsigned long r0, r1, r2, r3;
- long r4; /* signed, to catch environments with signed time_t */
- b1 = ut & 0xFFFF;
- b2 = (ut >> 16) & 0xFFFF; /* if ut is over 32 bits, too bad */
- r1 = b1 * (NT_QUANTA_PER_UNIX & 0xFFFF);
- r2 = b1 * (NT_QUANTA_PER_UNIX >> 16);
- r3 = b2 * (NT_QUANTA_PER_UNIX & 0xFFFF);
- r4 = b2 * (NT_QUANTA_PER_UNIX >> 16);
- r0 = (r1 + (r2 << 16)) & 0xFFFFFFFFL;
- if (r0 < r1)
- carry++;
- r1 = r0;
- r0 = (r0 + (r3 << 16)) & 0xFFFFFFFFL;
- if (r0 < r1)
- carry++;
- pft->dwLowDateTime = r0 + UNIX_TIME_ZERO_LO;
- if (pft->dwLowDateTime < r0)
- carry++;
- pft->dwHighDateTime = r4 + (r2 >> 16) + (r3 >> 16)
- + UNIX_TIME_ZERO_HI + carry;
- #endif /* ?HAVE_INT64 */
- } /* end function utime2FileTime() */
- /******************************/
- /* Function FStampIsLocTime() */
- /******************************/
- static int FStampIsLocTime(__GPRO__ const char *path)
- {
- return (NTQueryVolInfo(__G__ path) ? G.lastVolLocTim : FALSE);
- }
- #endif /* USE_EF_UT_TIME || NT_TZBUG_WORKAROUND || TIMESTAMP */
- #ifndef NT_TZBUG_WORKAROUND
- # define UTIME_BOUNDCHECK_1(utimval)
- if (fs_uses_loctime) {
- utime_dosmin = dos_to_unix_time(DOSTIME_MINIMUM);
- if ((ulg)utimval < (ulg)utime_dosmin)
- utimval = utime_dosmin;
- }
- # define UTIME_BOUNDCHECK_N(utimval)
- if (fs_uses_loctime && ((ulg)utimval < (ulg)utime_dosmin))
- utimval = utime_dosmin;
- # define NT_TZBUG_PRECOMPENSATE(ut, pft)
- #else /* NT_TZBUG_WORKAROUND */
- # define UNIX_TIME_UMAX_HI 0x0236485EUL
- # define UNIX_TIME_UMAX_LO 0xD4A5E980UL
- # define UNIX_TIME_SMIN_HI 0x0151669EUL
- # define UNIX_TIME_SMIN_LO 0xD53E8000UL
- # define UNIX_TIME_SMAX_HI 0x01E9FD1EUL
- # define UNIX_TIME_SMAX_LO 0xD4A5E980UL
- # define UTIME_1980_JAN_01_00_00 315532800L
- # define UTIME_BOUNDCHECK_1(utimval)
- # define UTIME_BOUNDCHECK_N(utimval)
- # define NT_TZBUG_PRECOMPENSATE(ut, pft)
- if (fs_uses_loctime) NTtzbugWorkaround(ut, pft);
- /* nonzero if `y' is a leap year, else zero */
- # define leap(y) (((y)%4 == 0 && (y)%100 != 0) || (y)%400 == 0)
- /* number of leap years from 1970 to `y' (not including `y' itself) */
- # define nleap(y) (((y)-1969)/4 - ((y)-1901)/100 + ((y)-1601)/400)
- extern ZCONST ush ydays[]; /* defined in fileio.c */
- /*****************************/
- /* Function FileTime2utime() */
- /*****************************/
- static int FileTime2utime(const FILETIME *pft, time_t *ut)
- {
- #ifdef HAVE_INT64
- ULLNG64 NTtime;
- NTtime = ((ULLNG64)pft->dwLowDateTime +
- ((ULLNG64)pft->dwHighDateTime << 32));
- /* underflow and overflow handling */
- #ifdef CHECK_UTIME_SIGNED_UNSIGNED
- if ((time_t)0x80000000L < (time_t)0L)
- {
- if (NTtime < ((ULLNG64)UNIX_TIME_SMIN_LO +
- ((ULLNG64)UNIX_TIME_SMIN_HI << 32))) {
- *ut = (time_t)LONG_MIN;
- return FALSE;
- }
- if (NTtime > ((ULLNG64)UNIX_TIME_SMAX_LO +
- ((ULLNG64)UNIX_TIME_SMAX_HI << 32))) {
- *ut = (time_t)LONG_MAX;
- return FALSE;
- }
- }
- else
- #endif /* CHECK_UTIME_SIGNED_UNSIGNED */
- {
- if (NTtime < ((ULLNG64)UNIX_TIME_ZERO_LO +
- ((ULLNG64)UNIX_TIME_ZERO_HI << 32))) {
- *ut = (time_t)0;
- return FALSE;
- }
- if (NTtime > ((ULLNG64)UNIX_TIME_UMAX_LO +
- ((ULLNG64)UNIX_TIME_UMAX_HI << 32))) {
- *ut = (time_t)ULONG_MAX;
- return FALSE;
- }
- }
- NTtime -= ((ULLNG64)UNIX_TIME_ZERO_LO +
- ((ULLNG64)UNIX_TIME_ZERO_HI << 32));
- *ut = (time_t)(NTtime / (unsigned long)NT_QUANTA_PER_UNIX);
- return TRUE;
- #else /* !HAVE_INT64 (64-bit integer arithmetics may not be supported) */
- time_t days;
- SYSTEMTIME w32tm;
- /* underflow and overflow handling */
- #ifdef CHECK_UTIME_SIGNED_UNSIGNED
- if ((time_t)0x80000000L < (time_t)0L)
- {
- if ((pft->dwHighDateTime < UNIX_TIME_SMIN_HI) ||
- ((pft->dwHighDateTime == UNIX_TIME_SMIN_HI) &&
- (pft->dwLowDateTime < UNIX_TIME_SMIN_LO))) {
- *ut = (time_t)LONG_MIN;
- return FALSE;
- if ((pft->dwHighDateTime > UNIX_TIME_SMAX_HI) ||
- ((pft->dwHighDateTime == UNIX_TIME_SMAX_HI) &&
- (pft->dwLowDateTime > UNIX_TIME_SMAX_LO))) {
- *ut = (time_t)LONG_MAX;
- return FALSE;
- }
- }
- else
- #endif /* CHECK_UTIME_SIGNED_UNSIGNED */
- {
- if ((pft->dwHighDateTime < UNIX_TIME_ZERO_HI) ||
- ((pft->dwHighDateTime == UNIX_TIME_ZERO_HI) &&
- (pft->dwLowDateTime < UNIX_TIME_ZERO_LO))) {
- *ut = (time_t)0;
- return FALSE;
- }
- if ((pft->dwHighDateTime > UNIX_TIME_UMAX_HI) ||
- ((pft->dwHighDateTime == UNIX_TIME_UMAX_HI) &&
- (pft->dwLowDateTime > UNIX_TIME_UMAX_LO))) {
- *ut = (time_t)ULONG_MAX;
- return FALSE;
- }
- }
- FileTimeToSystemTime(pft, &w32tm);
- /* set `days' to the number of days into the year */
- days = w32tm.wDay - 1 + ydays[w32tm.wMonth-1] +
- (w32tm.wMonth > 2 && leap (w32tm.wYear));
- /* now set `days' to the number of days since 1 Jan 1970 */
- days += 365 * (time_t)(w32tm.wYear - 1970) +
- (time_t)(nleap(w32tm.wYear));
- *ut = (time_t)(86400L * days + 3600L * (time_t)w32tm.wHour +
- (time_t)(60 * w32tm.wMinute + w32tm.wSecond));
- return TRUE;
- #endif /* ?HAVE_INT64 */
- } /* end function FileTime2utime() */
- #ifdef W32_STAT_BANDAID
- /*********************************/
- /* Function VFatFileTime2utime() */
- /*********************************/
- static int VFatFileTime2utime(const FILETIME *pft, time_t *ut)
- {
- FILETIME lft;
- #ifndef HAVE_MKTIME
- WORD wDOSDate, wDOSTime;
- #else
- SYSTEMTIME w32tm;
- struct tm ltm;
- #endif
- FileTimeToLocalFileTime(pft, &lft);
- FTTrace((stdout, "VFatFT2utime, feed for mktime()", 1, &lft));
- #ifndef HAVE_MKTIME
- /* This version of the FILETIME-to-UNIXTIME conversion function
- * uses DOS-DATE-TIME format as intermediate stage. For modification
- * and access times, this is no problem. But, the extra fine resolution
- * of the VFAT-stored creation time gets lost.
- */
- FileTimeToDosDateTime(&lft, &wDOSDate, &wDOSTime);
- TTrace((stdout,"DosDateTime is %04u-%02u-%02u %02u:%02u:%02un",
- (unsigned)((wDOSDate>>9)&0x7f)+1980,(unsigned)((wDOSDate>>5)&0x0f),
- (unsigned)(wDOSDate&0x1f),(unsigned)((wDOSTime>>11)&0x1f),
- (unsigned)((wDOSTime>>5)&0x3f),(unsigned)((wDOSTime<<1)&0x3e)));
- *ut = dos_to_unix_time(((ulg)wDOSDate << 16) | (ulg)wDOSTime);
- /* a cheap error check: dos_to_unix_time() only returns an odd time
- * when clipping at maximum time_t value. DOS_DATE_TIME values have
- * a resolution of 2 seconds and are therefore even numbers.
- */
- return (((*ut)&1) == (time_t)0);
- #else /* HAVE_MKTIME */
- FileTimeToSystemTime(&lft, &w32tm);
- /* underflow and overflow handling */
- /* TODO: The range checks are not accurate, the actual limits may
- * be off by one daylight-saving-time shift (typically 1 hour),
- * depending on the current state of "is_dst".
- */
- #ifdef CHECK_UTIME_SIGNED_UNSIGNED
- if ((time_t)0x80000000L < (time_t)0L)
- {
- if ((pft->dwHighDateTime < UNIX_TIME_SMIN_HI) ||
- ((pft->dwHighDateTime == UNIX_TIME_SMIN_HI) &&
- (pft->dwLowDateTime < UNIX_TIME_SMIN_LO))) {
- *ut = (time_t)LONG_MIN;
- return FALSE;
- if ((pft->dwHighDateTime > UNIX_TIME_SMAX_HI) ||
- ((pft->dwHighDateTime == UNIX_TIME_SMAX_HI) &&
- (pft->dwLowDateTime > UNIX_TIME_SMAX_LO))) {
- *ut = (time_t)LONG_MAX;
- return FALSE;
- }
- }
- else
- #endif /* CHECK_UTIME_SIGNED_UNSIGNED */
- {
- if ((pft->dwHighDateTime < UNIX_TIME_ZERO_HI) ||
- ((pft->dwHighDateTime == UNIX_TIME_ZERO_HI) &&
- (pft->dwLowDateTime < UNIX_TIME_ZERO_LO))) {
- *ut = (time_t)0;
- return FALSE;
- }
- if ((pft->dwHighDateTime > UNIX_TIME_UMAX_HI) ||
- ((pft->dwHighDateTime == UNIX_TIME_UMAX_HI) &&
- (pft->dwLowDateTime > UNIX_TIME_UMAX_LO))) {
- *ut = (time_t)ULONG_MAX;
- return FALSE;
- }
- }
- ltm.tm_year = w32tm.wYear - 1900;
- ltm.tm_mon = w32tm.wMonth - 1;
- ltm.tm_mday = w32tm.wDay;
- ltm.tm_hour = w32tm.wHour;
- ltm.tm_min = w32tm.wMinute;
- ltm.tm_sec = w32tm.wSecond;
- ltm.tm_isdst = -1; /* let mktime determine if DST is in effect */
- *ut = mktime(<m);
- /* a cheap error check: mktime returns "(time_t)-1L" on conversion errors.
- * Normally, we would have to apply a consistency check because "-1"
- * could also be a valid time. But, it is quite unlikely to read back odd
- * time numbers from file systems that store time stamps in DOS format.
- * (The only known exception is creation time on VFAT partitions.)
- */
- return (*ut != (time_t)-1L);
- #endif /* ?HAVE_MKTIME */
- } /* end function VFatFileTime2utime() */
- #endif /* W32_STAT_BANDAID */
- /********************************/
- /* Function UTCtime2Localtime() */ /* borrowed from Zip's mkgmtime() */
- /********************************/
- static time_t UTCtime2Localtime(time_t utctime)
- {
- time_t utc = utctime;
- struct tm *tm;
- unsigned years, months, days, hours, minutes, seconds;
- #ifdef __BORLANDC__ /* Borland C++ 5.x crashes when trying to reference tm */
- if (utc < UTIME_1980_JAN_01_00_00)
- utc = UTIME_1980_JAN_01_00_00;
- #endif
- tm = localtime(&utc);
- if (tm == (struct tm *)NULL)
- /* localtime() did not accept given utc time value; as an emergency
- exit, the unconverted utctime value is returned */
- return utctime;
- years = tm->tm_year + 1900; /* year - 1900 -> year */
- months = tm->tm_mon; /* 0..11 */
- days = tm->tm_mday - 1; /* 1..31 -> 0..30 */
- hours = tm->tm_hour; /* 0..23 */
- minutes = tm->tm_min; /* 0..59 */
- seconds = tm->tm_sec; /* 0..61 in ANSI C */
- /* set `days' to the number of days into the year */
- days += ydays[months] + (months > 1 && leap(years));
- /* now set `days' to the number of days since 1 Jan 1970 */
- days += 365 * (years - 1970) + nleap(years);
- return (time_t)(86400L * (ulg)days + 3600L * (ulg)hours +
- (ulg)(60 * minutes + seconds));
- } /* end function UTCtime2Localtime() */
- /********************************/
- /* Function NTtzbugWorkaround() */
- /********************************/
- static void NTtzbugWorkaround(time_t ut, FILETIME *pft)
- {
- FILETIME C_RTL_locft, NTAPI_locft;
- time_t ux_loctime = UTCtime2Localtime(ut);
- /* This routine is only used when the target file system stores time-
- * stamps as local time in MSDOS format. Thus we make sure that the
- * resulting timestamp is within the range of MSDOS date-time values. */
- if (ux_loctime < UTIME_1980_JAN_01_00_00)
- ux_loctime = UTIME_1980_JAN_01_00_00;
- utime2FileTime(ux_loctime, &C_RTL_locft);
- if (!FileTimeToLocalFileTime(pft, &NTAPI_locft))
- return;
- else {
- long time_shift_l, time_shift_h;
- int carry = 0;
- time_shift_l = C_RTL_locft.dwLowDateTime - NTAPI_locft.dwLowDateTime;
- if (C_RTL_locft.dwLowDateTime < NTAPI_locft.dwLowDateTime)
- carry--;
- time_shift_h = C_RTL_locft.dwHighDateTime - NTAPI_locft.dwHighDateTime;
- pft->dwLowDateTime += time_shift_l;
- if (pft->dwLowDateTime < (ulg)time_shift_l)
- carry++;
- pft->dwHighDateTime += time_shift_h + carry;
- TTrace((stdout, "FileTime shift: %08lx:%08lxn",
- time_shift_h+carry,time_shift_l));
- }
- } /* end function NTtzbugWorkaround() */
- #endif /* ?NT_TZBUG_WORKAROUND */
- /****************************/ /* Get the file time in a format that */
- /* Function getNTfiletime() */ /* can be used by SetFileTime() in NT */
- /****************************/
- static int getNTfiletime(__G__ pModFT, pAccFT, pCreFT)
- __GDEF
- FILETIME *pModFT;
- FILETIME *pAccFT;
- FILETIME *pCreFT;
- {
- #ifdef NT_TZBUG_WORKAROUND
- time_t ux_modtime;
- #else /* !NT_TZBUG_WORKAROUND */
- FILETIME locft; /* 64-bit value made up of two 32-bit [low & high] */
- WORD wDOSDate; /* for converting from DOS date to Windows NT */
- WORD wDOSTime;
- #endif /* ?NT_TZBUG_WORKAROUND */
- #ifdef USE_EF_UT_TIME
- unsigned eb_izux_flg;
- iztimes z_utime; /* struct for Unix-style actime & modtime, + creatime */
- #endif
- #if (defined(USE_EF_UT_TIME) && !defined(NT_TZBUG_WORKAROUND))
- time_t utime_dosmin;
- # endif
- #if (defined(USE_EF_UT_TIME) || defined(NT_TZBUG_WORKAROUND))
- int fs_uses_loctime = FStampIsLocTime(__G__ G.filename);
- #endif
- /* Copy and/or convert time and date variables, if necessary;
- * return a flag indicating which time stamps are available. */
- #ifdef USE_EF_UT_TIME
- if (G.extra_field &&
- #ifdef IZ_CHECK_TZ
- G.tz_is_valid &&
- #endif
- ((eb_izux_flg = ef_scan_for_izux(G.extra_field,
- G.lrec.extra_field_length, 0, G.lrec.last_mod_dos_datetime,
- &z_utime, NULL)) & EB_UT_FL_MTIME))
- {
- TTrace((stderr, "getNTfiletime: Unix e.f. modif. time = %lun",
- z_utime.mtime));
- UTIME_BOUNDCHECK_1(z_utime.mtime)
- utime2FileTime(z_utime.mtime, pModFT);
- NT_TZBUG_PRECOMPENSATE(z_utime.mtime, pModFT)
- if (eb_izux_flg & EB_UT_FL_ATIME) {
- UTIME_BOUNDCHECK_N(z_utime.atime)
- utime2FileTime(z_utime.atime, pAccFT);
- NT_TZBUG_PRECOMPENSATE(z_utime.atime, pAccFT)
- }
- if (eb_izux_flg & EB_UT_FL_CTIME) {
- UTIME_BOUNDCHECK_N(z_utime.ctime)
- utime2FileTime(z_utime.ctime, pCreFT);
- NT_TZBUG_PRECOMPENSATE(z_utime.ctime, pCreFT)
- }
- return (int)eb_izux_flg;
- }
- #endif /* USE_EF_UT_TIME */
- #ifdef NT_TZBUG_WORKAROUND
- ux_modtime = dos_to_unix_time(G.lrec.last_mod_dos_datetime);
- utime2FileTime(ux_modtime, pModFT);
- NT_TZBUG_PRECOMPENSATE(ux_modtime, pModFT)
- #else /* !NT_TZBUG_WORKAROUND */
- wDOSTime = (WORD)(G.lrec.last_mod_dos_datetime);
- wDOSDate = (WORD)(G.lrec.last_mod_dos_datetime >> 16);
- /* The DosDateTimeToFileTime() function converts a DOS date/time
- * into a 64-bit Windows NT file time */
- if (!DosDateTimeToFileTime(wDOSDate, wDOSTime, &locft))
- {
- Info(slide, 0, ((char *)slide, "DosDateTime failed: %dn",
- (int)GetLastError()));
- return 0;
- }
- if (!LocalFileTimeToFileTime(&locft, pModFT))
- {
- Info(slide, 0, ((char *)slide, "LocalFileTime failed: %dn",
- (int)GetLastError()));
- *pModFT = locft;
- }
- #endif /* ?NT_TZBUG_WORKAROUND */
- *pAccFT = *pModFT;
- return (EB_UT_FL_MTIME | EB_UT_FL_ATIME);
- } /* end function getNTfiletime() */
- /****************************/
- /* Function close_outfile() */
- /****************************/
- void close_outfile(__G)
- __GDEF
- {
- FILETIME Modft; /* File time type defined in NT, `last modified' time */
- FILETIME Accft; /* NT file time type, `last access' time */
- FILETIME Creft; /* NT file time type, `file creation' time */
- HANDLE hFile; /* File handle defined in NT */
- int gotTime;
- #ifdef __RSXNT__ /* RSXNT/EMX C rtl uses OEM charset */
- char *ansi_name = (char *)alloca(strlen(G.filename) + 1);
- INTERN_TO_ISO(G.filename, ansi_name);
- # define Ansi_Fname ansi_name
- #else
- # define Ansi_Fname G.filename
- #endif
- /* Close the file and then re-open it using the Win32
- * CreateFile call, so that the file can be created
- * with GENERIC_WRITE access, otherwise the SetFileTime
- * call will fail. */
- fclose(G.outfile);
- /* don't set the time stamp and attributes on standard output */
- if (uO.cflag)
- return;
- gotTime = getNTfiletime(__G__ &Modft, &Accft, &Creft);
- /* open a handle to the file before processing extra fields;
- we do this in case new security on file prevents us from updating
- time stamps */
- hFile = CreateFile(Ansi_Fname, GENERIC_WRITE, FILE_SHARE_WRITE, NULL,
- OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
- /* sfield@microsoft.com: set attributes before time in case we decide to
- support other filetime members later. This also allows us to apply
- attributes before the security is changed, which may prevent this
- from succeeding otherwise. Also, since most files don't have
- any interesting attributes, only change them if something other than
- FILE_ATTRIBUTE_ARCHIVE appears in the attributes. This works well
- as an optimization because FILE_ATTRIBUTE_ARCHIVE gets applied to the
- file anyway, when it's created new. */
- if((G.pInfo->file_attr & 0x7F) & ~FILE_ATTRIBUTE_ARCHIVE) {
- if (!SetFileAttributes(Ansi_Fname, G.pInfo->file_attr & 0x7F))
- Info(slide, 1, ((char *)slide,
- "nwarning (%d): could not set file attributesn",
- (int)GetLastError()));
- }
- #ifdef NTSD_EAS
- /* set extra fields, both stored-in-zipfile and .LONGNAME flavors */
- if (G.extra_field) { /* zipfile extra field may have extended attribs */
- int err = EvalExtraFields(__G__ G.filename, G.extra_field,
- G.lrec.extra_field_length);
- if (err == IZ_EF_TRUNC) {
- if (uO.qflag)
- Info(slide, 1, ((char *)slide, "%-22s ",
- FnFilter1(G.filename)));
- Info(slide, 1, ((char *)slide, LoadFarString(TruncNTSD),
- makeword(G.extra_field+2)-10, uO.qflag? "n":""));
- }
- }
- #endif /* NTSD_EAS */
- if ( hFile == INVALID_HANDLE_VALUE )
- Info(slide, 1, ((char *)slide,
- "nCreateFile error %d when trying set file timen",
- (int)GetLastError()));
- else {
- if (gotTime) {
- FILETIME *pModft = (gotTime & EB_UT_FL_MTIME) ? &Modft : NULL;
- FILETIME *pAccft = (gotTime & EB_UT_FL_ATIME) ? &Accft : NULL;
- FILETIME *pCreft = (gotTime & EB_UT_FL_CTIME) ? &Creft : NULL;
- if (!SetFileTime(hFile, pCreft, pAccft, pModft))
- Info(slide, 0, ((char *)slide, "nSetFileTime failed: %dn",
- (int)GetLastError()));
- }
- CloseHandle(hFile);
- }
- return;
- #undef Ansi_Fname
- } /* end function close_outfile() */
- #ifdef TIMESTAMP
- /*************************/
- /* Function stamp_file() */
- /*************************/
- int stamp_file(__GPRO__ ZCONST char *fname, time_t modtime)
- {
- FILETIME Modft; /* File time type defined in NT, `last modified' time */
- HANDLE hFile; /* File handle defined in NT */
- int errstat = 0; /* return status: 0 == "OK", -1 == "Failure" */
- #ifndef NT_TZBUG_WORKAROUND
- time_t utime_dosmin; /* internal variable for UTIME_BOUNDCHECK_1 */
- #endif
- int fs_uses_loctime = FStampIsLocTime(__G__ fname);
- #ifdef __RSXNT__ /* RSXNT/EMX C rtl uses OEM charset */
- char *ansi_name = (char *)alloca(strlen(fname) + 1);
- INTERN_TO_ISO(fname, ansi_name);
- # define Ansi_Fname ansi_name
- #else
- # define Ansi_Fname fname
- #endif
- /* open a handle to the file to prepare setting the mod-time stamp */
- hFile = CreateFile(Ansi_Fname, GENERIC_WRITE, FILE_SHARE_WRITE, NULL,
- OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
- if ( hFile == INVALID_HANDLE_VALUE ) {
- errstat = -1;
- } else {
- /* convert time_t modtime into WIN32 native 64bit format */
- UTIME_BOUNDCHECK_1(modtime)
- utime2FileTime(modtime, &Modft);
- NT_TZBUG_PRECOMPENSATE(modtime, &Modft)
- /* set Access and Modification times of the file to modtime */
- if (!SetFileTime(hFile, NULL, &Modft, &Modft)) {
- errstat = -1;
- }
- CloseHandle(hFile);
- }
- return errstat;
- #undef Ansi_Fname
- } /* end function stamp_file() */
- #endif /* TIMESTAMP */
- /***********************/
- /* Function isfloppy() */ /* more precisely, is it removable? */
- /***********************/
- static int isfloppy(int nDrive) /* 1 == A:, 2 == B:, etc. */
- {
- char rootPathName[4];
- rootPathName[0] = (char)('A' + nDrive - 1); /* build the root path */
- rootPathName[1] = ':'; /* name, e.g. "A:/" */
- rootPathName[2] = '/';
- rootPathName[3] = ' ';
- return (GetDriveType(rootPathName) == DRIVE_REMOVABLE);
- } /* end function isfloppy() */
- /*****************************/
- /* Function NTQueryVolInfo() */
- /*****************************/
- /*
- * Note: 8.3 limits on filenames apply only to old-style FAT filesystems.
- * More recent versions of Windows (Windows NT 3.5 / Windows 4.0)
- * can support long filenames (LFN) on FAT filesystems. Check the
- * filesystem maximum component length field to detect LFN support.
- */
- static int NTQueryVolInfo(__GPRO__ const char *name)
- {
- /* static char lastRootPath[4] = ""; */
- /* static int lastVolOldFAT; */
- /* static int lastVolLocTim; */
- char *tmp0;
- char tmp1[MAX_PATH], tmp2[MAX_PATH];
- unsigned volSerNo, maxCompLen, fileSysFlags;
- #ifdef __RSXNT__ /* RSXNT/EMX C rtl uses OEM charset */
- char *ansi_name = (char *)alloca(strlen(name) + 1);
- INTERN_TO_ISO(name, ansi_name);
- name = ansi_name;
- #endif
- if ((!strncmp(name, "//", 2) || !strncmp(name,"\\", 2)) &&
- (name[2] != ' ' && name[2] != '/' && name[2] != '\')) {
- /* GetFullPathname() and GetVolumeInformation() do not work
- * on UNC names. For now, we return "error".
- * **FIXME**: check if UNC name is mapped to a drive letter
- * and use mapped drive for volume info query.
- */
- return FALSE;
- }
- if (isalpha((uch)name[0]) && (name[1] == ':'))
- tmp0 = (char *)name;
- else
- {
- if (!GetFullPathName(name, MAX_PATH, tmp1, &tmp0))
- return FALSE;
- tmp0 = &tmp1[0];
- }
- if (strncmp(G.lastRootPath, tmp0, 2) != 0) {
- /* For speed, we skip repeated queries for the same device */
- strncpy(G.lastRootPath, tmp0, 2); /* Build the root path name, */
- G.lastRootPath[2] = '/'; /* e.g. "A:/" */
- G.lastRootPath[3] = ' ';
- if (!GetVolumeInformation((LPCTSTR)G.lastRootPath,
- (LPTSTR)tmp1, (DWORD)MAX_PATH,
- (LPDWORD)&volSerNo, (LPDWORD)&maxCompLen,
- (LPDWORD)&fileSysFlags, (LPTSTR)tmp2, (DWORD)MAX_PATH)) {
- G.lastRootPath[0] = ' ';
- return FALSE;
- }
- /* LFNs are available if the component length is > 12 */
- G.lastVolOldFAT = (maxCompLen <= 12);
- /* G.lastVolOldFAT = !strncmp(strupr(tmp2), "FAT", 3); old version */
- /* Volumes in (V)FAT and (OS/2) HPFS format store file timestamps in
- * local time!
- */
- G.lastVolLocTim = !strncmp(strupr(tmp2), "VFAT", 4) ||
- !strncmp(tmp2, "HPFS", 4) ||
- !strncmp(tmp2, "FAT", 3);
- }
- return TRUE;
- } /* end function NTQueryVolInfo() */
- /*****************************/
- /* Function IsVolumeOldFAT() */
- /*****************************/
- static int IsVolumeOldFAT(__GPRO__ const char *name)
- {
- return (NTQueryVolInfo(__G__ name) ? G.lastVolOldFAT : FALSE);
- }
- #ifndef SFX
- /************************/
- /* Function do_wild() */ /* identical to OS/2 version */
- /************************/
- char *do_wild(__G__ wildspec)
- __GDEF
- char *wildspec; /* only used first time on a given dir */
- {
- /* static zDIR *wild_dir = NULL; */
- /* static char *dirname, *wildname, matchname[FILNAMSIZ]; */
- /* static int firstcall=TRUE, have_dirname, dirnamelen; */
- char *fnamestart;
- struct zdirent *file;
- /* Even when we're just returning wildspec, we *always* do so in
- * matchname[]--calling routine is allowed to append four characters
- * to the returned string, and wildspec may be a pointer to argv[].
- */
- if (!G.notfirstcall) { /* first call: must initialize everything */
- G.notfirstcall = TRUE;
- if (!iswild(wildspec)) {
- strcpy(G.matchname, wildspec);
- G.have_dirname = FALSE;
- G.wild_dir = NULL;
- return G.matchname;
- }
- /* break the wildspec into a directory part and a wildcard filename */
- if ((G.wildname = strrchr(wildspec, '/')) == NULL &&
- (G.wildname = strrchr(wildspec, ':')) == NULL) {
- G.dirname = ".";
- G.dirnamelen = 1;
- G.have_dirname = FALSE;
- G.wildname = wildspec;
- } else {
- ++G.wildname; /* point at character after '/' or ':' */
- G.dirnamelen = G.wildname - wildspec;
- if ((G.dirname = (char *)malloc(G.dirnamelen+1)) == NULL) {
- Info(slide, 1, ((char *)slide,
- "warning: cannot allocate wildcard buffersn"));
- strcpy(G.matchname, wildspec);
- return G.matchname; /* but maybe filespec was not a wildcard */
- }
- strncpy(G.dirname, wildspec, G.dirnamelen);
- G.dirname[G.dirnamelen] = ' '; /* terminate for strcpy below */
- G.have_dirname = TRUE;
- }
- Trace((stderr, "do_wild: dirname = [%s]n", G.dirname));
- if ((G.wild_dir = (zvoid *)Opendir(G.dirname)) != NULL) {
- if (G.have_dirname) {
- strcpy(G.matchname, G.dirname);
- fnamestart = G.matchname + G.dirnamelen;
- } else
- fnamestart = G.matchname;
- while ((file = Readdir((zDIR *)G.wild_dir)) != NULL) {
- Trace((stderr, "do_wild: Readdir returns %sn", file->d_name));
- strcpy(fnamestart, file->d_name);
- if (strrchr(fnamestart, '.') == (char *)NULL)
- strcat(fnamestart, ".");
- if (match(fnamestart, G.wildname, 1) && /* 1 == ignore case */
- /* skip "." and ".." directory entries */
- strcmp(fnamestart, ".") && strcmp(fnamestart, "..")) {
- Trace((stderr, "do_wild: match() succeedsn"));
- /* remove trailing dot */
- fnamestart += strlen(fnamestart) - 1;
- if (*fnamestart == '.')
- *fnamestart = ' ';
- return G.matchname;
- }
- }
- /* if we get to here directory is exhausted, so close it */
- Closedir((zDIR *)G.wild_dir);
- G.wild_dir = NULL;
- }
- Trace((stderr, "do_wild: Opendir(%s) returns NULLn", G.dirname));
- /* return the raw wildspec in case that works (e.g., directory not
- * searchable, but filespec was not wild and file is readable) */
- strcpy(G.matchname, wildspec);
- return G.matchname;
- }
- /* last time through, might have failed opendir but returned raw wildspec */
- if (G.wild_dir == NULL) {
- G.notfirstcall = FALSE; /* reset for new wildspec */
- if (G.have_dirname)
- free(G.dirname);
- return (char *)NULL;
- }
- /* If we've gotten this far, we've read and matched at least one entry
- * successfully (in a previous call), so dirname has been copied into
- * matchname already.
- */
- if (G.have_dirname) {
- /* strcpy(G.matchname, G.dirname); */
- fnamestart = G.matchname + G.dirnamelen;
- } else
- fnamestart = G.matchname;
- while ((file = Readdir((zDIR *)G.wild_dir)) != NULL) {
- Trace((stderr, "do_wild: readdir returns %sn", file->d_name));
- strcpy(fnamestart, file->d_name);
- if (strrchr(fnamestart, '.') == (char *)NULL)
- strcat(fnamestart, ".");
- if (match(fnamestart, G.wildname, 1)) { /* 1 == ignore case */
- Trace((stderr, "do_wild: match() succeedsn"));
- /* remove trailing dot */
- fnamestart += strlen(fnamestart) - 1;
- if (*fnamestart == '.')
- *fnamestart = ' ';
- return G.matchname;
- }
- }
- Closedir((zDIR *)G.wild_dir); /* at least one entry read; nothing left */
- G.wild_dir = NULL;
- G.notfirstcall = FALSE; /* reset for new wildspec */
- if (G.have_dirname)
- free(G.dirname);
- return (char *)NULL;
- } /* end function do_wild() */
- #endif /* !SFX */
- /**********************/
- /* Function mapattr() */
- /**********************/
- /* Identical to MS-DOS, OS/2 versions. However, NT has a lot of extra
- * permission stuff, so this function should probably be extended in the
- * future. */
- int mapattr(__G)
- __GDEF
- {
- /* set archive bit for file entries (file is not backed up): */
- G.pInfo->file_attr = ((unsigned)G.crec.external_file_attributes |
- (G.crec.external_file_attributes & FILE_ATTRIBUTE_DIRECTORY ?
- 0 : FILE_ATTRIBUTE_ARCHIVE)) & 0xff;
- return 0;
- } /* end function mapattr() */
- /************************/
- /* Function mapname() */
- /************************/
- /* return 0 if no error, 1 if caution (filename */
- int mapname(__G__ renamed) /* truncated), 2 if warning (skip file because */
- __GDEF /* dir doesn't exist), 3 if error (skip file), */
- int renamed; /* or 10 if out of memory (skip file) */
- { /* [also IZ_VOL_LABEL, IZ_CREATED_DIR] */
- char pathcomp[FILNAMSIZ]; /* path-component buffer */
- char *pp, *cp=NULL; /* character pointers */
- char *lastsemi = NULL; /* pointer to last semi-colon in pathcomp */
- int error;
- register unsigned workch; /* hold the character being tested */
- /*---------------------------------------------------------------------------
- Initialize various pointers and counters and stuff.
- ---------------------------------------------------------------------------*/
- /* can create path as long as not just freshening, or if user told us */
- G.create_dirs = (!uO.fflag || renamed);
- G.created_dir = FALSE; /* not yet */
- G.renamed_fullpath = FALSE;
- G.fnlen = strlen(G.filename);
- if (renamed) {
- cp = G.filename - 1; /* point to beginning of renamed name... */
- while (*++cp)
- if (*cp == '\') /* convert backslashes to forward */
- *cp = '/';
- cp = G.filename;
- /* use temporary rootpath if user gave full pathname */
- if (G.filename[0] == '/') {
- G.renamed_fullpath = TRUE;
- pathcomp[0] = '/'; /* copy the '/' and terminate */
- pathcomp[1] = ' ';
- ++cp;
- } else if (isalpha((uch)G.filename[0]) && G.filename[1] == ':') {
- G.renamed_fullpath = TRUE;
- pp = pathcomp;
- *pp++ = *cp++; /* copy the "d:" (+ '/', possibly) */
- *pp++ = *cp++;
- if (*cp == '/')
- *pp++ = *cp++; /* otherwise add "./"? */
- *pp = ' ';
- }
- }
- /* pathcomp is ignored unless renamed_fullpath is TRUE: */
- if ((error = checkdir(__G__ pathcomp, INIT)) != 0) /* init path buffer */
- return error; /* ...unless no mem or vol label on hard disk */
- *pathcomp = ' '; /* initialize translation buffer */
- pp = pathcomp; /* point to translation buffer */
- if (!renamed) { /* cp already set if renamed */
- if (uO.jflag) /* junking directories */
- cp = (char *)strrchr(G.filename, '/');
- if (cp == NULL) /* no '/' or not junking dirs */
- cp = G.filename; /* point to internal zipfile-member pathname */
- else
- ++cp; /* point to start of last component of path */
- }
- /*---------------------------------------------------------------------------
- Begin main loop through characters in filename.
- ---------------------------------------------------------------------------*/
- while ((workch = (uch)*cp++) != 0) {
- switch (workch) {
- case '/': /* can assume -j flag not given */
- *pp = ' ';
- if ((error = checkdir(__G__ pathcomp, APPEND_DIR)) > 1)
- return error;
- pp = pathcomp; /* reset conversion buffer for next piece */
- lastsemi = NULL; /* leave directory semi-colons alone */
- break;
- case ':': /* drive names not stored in zipfile, */
- case '<': /* so no colons allowed */
- case '>': /* no redirection symbols allowed either */
- case '|': /* no pipe signs allowed */
- case '"': /* no double quotes allowed */
- case '?': /* no wildcards allowed */
- case '*':
- *pp++ = '_'; /* these rules apply equally to FAT and NTFS */
- break;
- case ';': /* start of VMS version? */
- lastsemi = pp; /* remove VMS version later... */
- *pp++ = ';'; /* but keep semicolon for now */
- break;
- case ' ': /* keep spaces unless specifically */
- /* NT cannot create filenames with spaces on FAT volumes */
- if (uO.sflag || IsVolumeOldFAT(__G__ G.filename))
- *pp++ = '_';
- else
- *pp++ = ' ';
- break;
- default:
- /* allow European characters in filenames: */
- if (isprint(workch) || workch >= 127)
- *pp++ = (char)workch;
- } /* end switch */
- } /* end while loop */
- *pp = ' '; /* done with pathcomp: terminate it */
- /* if not saving them, remove VMS version numbers (appended "###") */
- if (!uO.V_flag && lastsemi) {
- pp = lastsemi + 1; /* semi-colon was kept: expect #'s after */
- while (isdigit((uch)(*pp)))
- ++pp;
- if (*pp == ' ') /* only digits between ';' and end: nuke */
- *lastsemi = ' ';
- }
- /*---------------------------------------------------------------------------
- Report if directory was created (and no file to create: filename ended
- in '/'), check name to be sure it exists, and combine path and name be-
- fore exiting.
- ---------------------------------------------------------------------------*/
- if (G.filename[G.fnlen-1] == '/') {
- checkdir(__G__ G.filename, GETPATH);
- if (G.created_dir) {
- #ifdef __RSXNT__ /* RSXNT/EMX C rtl uses OEM charset */
- char *ansi_name = (char *)alloca(strlen(G.filename) + 1);
- INTERN_TO_ISO(G.filename, ansi_name);
- # define Ansi_Fname ansi_name
- #else
- # define Ansi_Fname G.filename
- #endif
- if (QCOND2) {
- Info(slide, 0, ((char *)slide, " creating: %-22sn",
- FnFilter1(G.filename)));
- }
- /* set file attributes:
- The default for newly created directories is "DIR attribute
- flags set", so there is no need to change attributes unless
- one of the DOS style attribute flags is set. The readonly
- attribute need not be masked, since it does not prevent
- modifications in the new directory. */
- if(G.pInfo->file_attr & (0x7F & ~FILE_ATTRIBUTE_DIRECTORY)) {
- if (!SetFileAttributes(Ansi_Fname, G.pInfo->file_attr & 0x7F))
- Info(slide, 1, ((char *)slide,
- "nwarning (%d): could not set file attributes for %sn",
- (int)GetLastError(), G.filename));
- }
- #ifdef NTSD_EAS
- /* set extra fields, both stored-in-zipfile and .LONGNAME flavors */
- if (G.extra_field) { /* zipfile e.f. may have extended attribs */
- int err = EvalExtraFields(__G__ G.filename, G.extra_field,
- G.lrec.extra_field_length);
- if (err == IZ_EF_TRUNC) {
- if (uO.qflag)
- Info(slide, 1, ((char *)slide, "%-22s ",
- FnFilter1(G.filename)));
- Info(slide, 1, ((char *)slide, LoadFarString(TruncNTSD),
- makeword(G.extra_field+2)-10, uO.qflag? "n":""));
- }
- }
- #endif /* NTSD_EAS */
- return IZ_CREATED_DIR; /* set dir time (note trailing '/') */
- }
- return 2; /* dir existed already; don't look for data to extract */
- }
- if (*pathcomp == ' ') {
- Info(slide, 1, ((char *)slide, "mapname: conversion of %s failedn",
- FnFilter1(G.filename)));
- return 3;
- }
- checkdir(__G__ pathcomp, APPEND_NAME); /* returns 1 if truncated: care? */
- checkdir(__G__ G.filename, GETPATH);
- Trace((stderr, "mapname returns with filename = [%s] (error = %d)nn",
- FnFilter1(G.filename), error));
- if (G.pInfo->vollabel) { /* set the volume label now */
- char drive[4];
- #ifdef __RSXNT__ /* RSXNT/EMX C rtl uses OEM charset */
- char *ansi_name = (char *)alloca(strlen(G.filename) + 1);
- INTERN_TO_ISO(G.filename, ansi_name);
- # define Ansi_Fname ansi_name
- #else
- # define Ansi_Fname G.filename
- #endif
- /* Build a drive string, e.g. "b:" */
- drive[0] = (char)('a' + G.nLabelDrive - 1);
- strcpy(drive + 1, ":\");
- if (QCOND2)
- Info(slide, 0, ((char *)slide, "labelling %s %-22sn", drive,
- FnFilter1(G.filename)));
- if (!SetVolumeLabel(drive, Ansi_Fname)) {
- Info(slide, 1, ((char *)slide,
- "mapname: error setting volume labeln"));
- return 3;
- }
- return 2; /* success: skip the "extraction" quietly */
- #undef Ansi_Fname
- }
- return error;
- } /* end function mapname() */
- /**********************/
- /* Function map2fat() */ /* Not quite identical to OS/2 version */
- /**********************/
- static void map2fat(pathcomp, pEndFAT)
- char *pathcomp, **pEndFAT;
- {
- char *ppc = pathcomp; /* variable pointer to pathcomp */
- char *pEnd = *pEndFAT; /* variable pointer to buildpathFAT */
- char *pBegin = *pEndFAT; /* constant pointer to start of this comp. */
- char *last_dot = NULL; /* last dot not converted to underscore */
- int dotname = FALSE; /* flag: path component begins with dot */
- /* ("." and ".." don't count) */
- register unsigned workch; /* hold the character being tested */
- /* Only need check those characters which are legal in NTFS but not
- * in FAT: to get here, must already have passed through mapname.
- * Also must truncate path component to ensure 8.3 compliance.
- */
- while ((workch = (uch)*ppc++) != 0) {
- switch (workch) {
- case '[':
- case ']':
- case '+':
- case ',':
- case ';':
- case '=':
- *pEnd++ = '_'; /* convert brackets to underscores */
- break;
- case '.':
- if (pEnd == *pEndFAT) { /* nothing appended yet... */
- if (*ppc == ' ') /* don't bother appending a */
- break; /* "./" component to the path */
- else if (*ppc == '.' && ppc[1] == ' ') { /* "../" */
- *pEnd++ = '.'; /* add first dot, unchanged... */
- ++ppc; /* skip second dot, since it will */
- } else { /* be "added" at end of if-block */
- *pEnd++ = '_'; /* FAT doesn't allow null filename */
- dotname = TRUE; /* bodies, so map .exrc -> _.exrc */
- } /* (extra '_' now, "dot" below) */
- } else if (dotname) { /* found a second dot, but still */
- dotname = FALSE; /* have extra leading underscore: */
- *pEnd = ' '; /* remove it by shifting chars */
- pEnd = *pEndFAT + 1; /* left one space (e.g., .p1.p2: */
- while (pEnd[1]) { /* __p1 -> _p1_p2 -> _p1.p2 when */
- *pEnd = pEnd[1]; /* finished) [opt.: since first */
- ++pEnd; /* two chars are same, can start */
- } /* shifting at second position] */
- }
- last_dot = pEnd; /* point at last dot so far... */
- *pEnd++ = '_'; /* convert dot to underscore for now */
- break;
- default:
- *pEnd++ = (char)workch;
- } /* end switch */
- } /* end while loop */
- *pEnd = ' '; /* terminate buildpathFAT */
- /* NOTE: keep in mind that pEnd points to the end of the path
- * component, and *pEndFAT still points to the *beginning* of it...
- * Also note that the algorithm does not try to get too fancy:
- * if there are no dots already, the name either gets truncated
- * at 8 characters or the last underscore is converted to a dot
- * (only if more characters are saved that way). In no case is
- * a dot inserted between existing characters.
- */
- if (last_dot == NULL) { /* no dots: check for underscores... */
- char *plu = strrchr(pBegin, '_'); /* pointer to last underscore */
- if (plu == NULL) { /* no dots, no underscores: truncate at 8 chars */
- *pEndFAT += 8; /* (or could insert '.' and keep 11...?) */
- if (*pEndFAT > pEnd)
- *pEndFAT = pEnd; /* oops...didn't have 8 chars to truncate */
- else
- **pEndFAT = ' ';
- } else if (MIN(plu - pBegin, 8) + MIN(pEnd - plu - 1, 3) > 8) {
- last_dot = plu; /* be lazy: drop through to next if-blk */
- } else if ((pEnd - *pEndFAT) > 8) {
- *pEndFAT += 8; /* more fits into just basename than if */
- **pEndFAT = ' '; /* convert last underscore to dot */
- } else
- *pEndFAT = pEnd; /* whole thing fits into 8 chars or less */
- }
- if (last_dot != NULL) { /* one dot (or two, in the case of */
- *last_dot = '.'; /* "..") is OK: put it back in */
- if ((last_dot - pBegin) > 8) {
- char *p, *q;
- int i;
- p = last_dot;
- q = last_dot = pBegin + 8;
- for (i = 0; (i < 4) && *p; ++i) /* too many chars in basename: */
- *q++ = *p++; /* shift .ext left and trun- */
- *q = ' '; /* cate/terminate it */
- *pEndFAT = q;
- } else if ((pEnd - last_dot) > 4) { /* too many chars in extension */
- *pEndFAT = last_dot + 4;
- **pEndFAT = ' ';
- } else
- *pEndFAT = pEnd; /* filename is fine; point at terminating zero */
- if ((last_dot - pBegin) > 0 && last_dot[-1] == ' ')
- last_dot[-1] = '_'; /* NO blank in front of '.'! */
- }
- } /* end function map2fat() */
- /***********************/ /* Borrowed from os2.c for UnZip 5.1. */
- /* Function checkdir() */ /* Difference: no EA stuff */
- /***********************/ /* HPFS stuff works on NTFS too */
- int checkdir(__G__ pathcomp, flag)
- __GDEF
- char *pathcomp;
- int flag;
- /*
- * returns: 1 - (on APPEND_NAME) truncated filename
- * 2 - path doesn't exist, not allowed to create
- * 3 - path doesn't exist, tried to create and failed; or
- * path exists and is not a directory, but is supposed to be
- * 4 - path is too long
- * 10 - can't allocate memory for filename buffers
- */
- {
- /* static int rootlen = 0; */ /* length of rootpath */
- /* static char *rootpath; */ /* user's "extract-to" directory */
- /* static char *buildpathHPFS; */ /* full path (so far) to extracted file, */
- /* static char *buildpathFAT; */ /* both HPFS/EA (main) and FAT versions */
- /* static char *endHPFS; */ /* corresponding pointers to end of */
- /* static char *endFAT; */ /* buildpath (' ') */
- # define FN_MASK 7
- # define FUNCTION (flag & FN_MASK)
- /*---------------------------------------------------------------------------
- APPEND_DIR: append the path component to the path being built and check
- for its existence. If doesn't exist and we are creating directories, do
- so for this one; else signal success or error as appropriate.
- ---------------------------------------------------------------------------*/
- if (FUNCTION == APPEND_DIR) {
- char *p = pathcomp;
- int too_long=FALSE;
- Trace((stderr, "appending dir segment [%s]n", pathcomp));
- while ((*G.endHPFS = *p++) != ' ') /* copy to HPFS filename */
- ++G.endHPFS;
- if (!IsVolumeOldFAT(__G__ G.buildpathHPFS)) {
- p = pathcomp;
- while ((*G.endFAT = *p++) != ' ') /* copy to FAT filename, too */
- ++G.endFAT;
- } else
- map2fat(pathcomp, &G.endFAT); /* map into FAT fn, update endFAT */
- /* GRR: could do better check, see if overrunning buffer as we go:
- * check endHPFS-buildpathHPFS after each append, set warning variable
- * if within 20 of FILNAMSIZ; then if var set, do careful check when
- * appending. Clear variable when begin new path. */
- /* next check: need to append '/', at least one-char name, ' ' */
- if ((G.endHPFS-G.buildpathHPFS) > FILNAMSIZ-3)
- too_long = TRUE; /* check if extracting dir? */
- #ifdef FIX_STAT_BUG
- /* Borland C++ 5.0 does not handle a call to stat() well if the
- * directory does not exist (it tends to crash in strange places.)
- * This is apparently a problem only when compiling for GUI rather
- * than console. The code below attempts to work around this problem.
- */
- if (access(G.buildpathFAT, 0) != 0) {
- if (!G.create_dirs) { /* told not to create (freshening) */
- free(G.buildpathHPFS);
- free(G.buildpathFAT);
- return 2; /* path doesn't exist: nothing to do */
- }
- if (too_long) { /* GRR: should allow FAT extraction w/o EAs */
- Info(slide, 1, ((char *)slide,
- "checkdir error: path too long: %sn",
- FnFilter1(G.buildpathHPFS)));
- free(G.buildpathHPFS);
- free(G.buildpathFAT);
- return 4; /* no room for filenames: fatal */
- }
- if (MKDIR(G.buildpathFAT, 0777) == -1) { /* create the directory */
- Info(slide, 1, ((char *)slide,
- "checkdir error: cannot create %sn
- unable to process %s.n",
- FnFilter2(G.buildpathFAT), FnFilter1(G.filename)));
- free(G.buildpathHPFS);
- free(G.buildpathFAT);
- return 3; /* path didn't exist, tried to create, failed */
- }
- G.created_dir = TRUE;
- }
- #endif /* FIX_STAT_BUG */
- if (SSTAT(G.buildpathFAT, &G.statbuf)) /* path doesn't exist */
- {
- if (!G.create_dirs) { /* told not to create (freshening) */
- free(G.buildpathHPFS);
- free(G.buildpathFAT);
- return 2; /* path doesn't exist: nothing to do */
- }
- if (too_long) { /* GRR: should allow FAT extraction w/o EAs */
- Info(slide, 1, ((char *)slide,
- "checkdir error: path too long: %sn",
- FnFilter1(G.buildpathHPFS)));
- free(G.buildpathHPFS);
- free(G.buildpathFAT);
- return 4; /* no room for filenames: fatal */
- }
- if (MKDIR(G.buildpathFAT, 0777) == -1) { /* create the directory */
- Info(slide, 1, ((char *)slide,
- "checkdir error: cannot create %sn
- unable to process %s.n",
- FnFilter2(G.buildpathFAT), FnFilter1(G.filename)));
- free(G.buildpathHPFS);
- free(G.buildpathFAT);
- return 3; /* path didn't exist, tried to create, failed */
- }
- G.created_dir = TRUE;
- } else if (!S_ISDIR(G.statbuf.st_mode)) {
- Info(slide, 1, ((char *)slide,
- "checkdir error: %s exists but is not directoryn
- unable to process %s.n",
- FnFilter2(G.buildpathFAT), FnFilter1(G.filename)));
- free(G.buildpathHPFS);
- free(G.buildpathFAT);
- return 3; /* path existed but wasn't dir */
- }
- if (too_long) {
- Info(slide, 1, ((char *)slide,
- "checkdir error: path too long: %sn",
- FnFilter1(G.buildpathHPFS)));
- free(G.buildpathHPFS);
- free(G.buildpathFAT);
- return 4; /* no room for filenames: fatal */
- }
- *G.endHPFS++ = '/';
- *G.endFAT++ = '/';
- *G.endHPFS = *G.endFAT = ' ';
- Trace((stderr, "buildpathHPFS now = [%s]nbuildpathFAT now = [%s]n",
- FnFilter1(G.buildpathHPFS), FnFilter2(G.buildpathFAT)));
- return 0;
- } /* end if (FUNCTION == APPEND_DIR) */
- /*---------------------------------------------------------------------------
- GETPATH: copy full FAT path to the string pointed at by pathcomp (want
- filename to reflect name used on disk, not EAs; if full path is HPFS,
- buildpathFAT and buildpathHPFS will be identical). Also free both paths.
- ---------------------------------------------------------------------------*/
- if (FUNCTION == GETPATH) {
- Trace((stderr, "getting and freeing FAT path [%s]n",
- FnFilter1(G.buildpathFAT)));
- Trace((stderr, "freeing HPFS path [%s]n",
- FnFilter1(G.buildpathHPFS)));
- strcpy(pathcomp, G.buildpathFAT);
- free(G.buildpathFAT);
- free(G.buildpathHPFS);
- G.buildpathHPFS = G.buildpathFAT = G.endHPFS = G.endFAT = NULL;
- return 0;
- }
- /*---------------------------------------------------------------------------
- APPEND_NAME: assume the path component is the filename; append it and
- return without checking for existence.
- ---------------------------------------------------------------------------*/
- if (FUNCTION == APPEND_NAME) {
- char *p = pathcomp;
- int error = 0;
- Trace((stderr, "appending filename [%s]n", FnFilter1(pathcomp)));
- while ((*G.endHPFS = *p++) != ' ') { /* copy to HPFS filename */
- ++G.endHPFS;
- if ((G.endHPFS-G.buildpathHPFS) >= FILNAMSIZ) {
- *--G.endHPFS = ' ';
- Info(slide, 1, ((char *)slide,
- "checkdir warning: path too long; truncatingn
- %sn -> %sn",
- FnFilter1(G.filename), FnFilter2(G.buildpathHPFS)));
- error = 1; /* filename truncated */
- }
- }
- if ( G.pInfo->vollabel || !IsVolumeOldFAT(__G__ G.buildpathHPFS)) {
- p = pathcomp;
- while ((*G.endFAT = *p++) != ' ') /* copy to FAT filename, too */
- ++G.endFAT;
- } else
- map2fat(pathcomp, &G.endFAT); /* map into FAT fn, update endFAT */
- Trace((stderr, "buildpathHPFS: %snbuildpathFAT: %sn",
- FnFilter1(G.buildpathHPFS), FnFilter2(G.buildpathFAT)));
- return error; /* could check for existence, prompt for new name... */
- } /* end if (FUNCTION == APPEND_NAME) */
- /*---------------------------------------------------------------------------
- INIT: allocate and initialize buffer space for the file currently being
- extracted. If file was renamed with an absolute path, don't prepend the
- extract-to path.
- ---------------------------------------------------------------------------*/
- if (FUNCTION == INIT) {
- Trace((stderr, "initializing buildpathHPFS and buildpathFAT to "));
- if ((G.buildpathHPFS = (char *)malloc(G.fnlen+G.rootlen+1)) == NULL)
- return 10;
- if ((G.buildpathFAT = (char *)malloc(G.fnlen+G.rootlen+1)) == NULL) {
- free(G.buildpathHPFS);
- return 10;
- }
- if (G.pInfo->vollabel) { /* use root or renamed path, but don't store */
- /* GRR: for network drives, do strchr() and return IZ_VOL_LABEL if not [1] */
- if (G.renamed_fullpath && pathcomp[1] == ':')
- *G.buildpathHPFS = (char)ToLower(*pathcomp);
- else if (!G.renamed_fullpath && G.rootlen > 1 &&
- G.rootpath[1] == ':')
- *G.buildpathHPFS = (char)ToLower(*G.rootpath);
- else {
- char tmpN[MAX_PATH], *tmpP;
- if (GetFullPathName(".", MAX_PATH, tmpN, &tmpP) > MAX_PATH)
- { /* by definition of MAX_PATH we should never get here */
- Info(slide, 1, ((char *)slide,
- "checkdir warning: current dir path too longn"));
- return 1; /* can't get drive letter */
- }
- G.nLabelDrive = *tmpN - 'a' + 1;
- *G.buildpathHPFS = (char)(G.nLabelDrive - 1 + 'a');
- }
- G.nLabelDrive = *G.buildpathHPFS - 'a' + 1; /* save for mapname() */
- if (uO.volflag == 0 || *G.buildpathHPFS < 'a' /* no labels/bogus? */
- || (uO.volflag == 1 && !isfloppy(G.nLabelDrive))) { /* !fixed */
- free(G.buildpathHPFS);
- free(G.buildpathFAT);
- return IZ_VOL_LABEL; /* skipping with message */
- }
- *G.buildpathHPFS = ' ';
- } else if (G.renamed_fullpath) /* pathcomp = valid data */
- strcpy(G.buildpathHPFS, pathcomp);
- else if (G.rootlen > 0)
- strcpy(G.buildpathHPFS, G.rootpath);
- else
- *G.buildpathHPFS = ' ';
- G.endHPFS = G.buildpathHPFS;
- G.endFAT = G.buildpathFAT;
- while ((*G.endFAT = *G.endHPFS) != ' ') {
- ++G.endFAT;
- ++G.endHPFS;
- }
- Trace((stderr, "[%s]n", FnFilter1(G.buildpathHPFS)));
- return 0;
- }
- /*---------------------------------------------------------------------------
- ROOT: if appropriate, store the path in rootpath and create it if neces-
- sary; else assume it's a zipfile member and return. This path segment
- gets used in extracting all members from every zipfile specified on the
- command line. Note that under OS/2 and MS-DOS, if a candidate extract-to
- directory specification includes a drive letter (leading "x:"), it is
- treated just as if it had a trailing '/'--that is, one directory level
- will be created if the path doesn't exist, unless this is otherwise pro-
- hibited (e.g., freshening).
- ---------------------------------------------------------------------------*/
- #if (!defined(SFX) || defined(SFX_EXDIR))
- if (FUNCTION == ROOT) {
- Trace((stderr, "initializing root path to [%s]n",
- FnFilter1(pathcomp)));
- if (pathcomp == NULL) {
- G.rootlen = 0;
- return 0;
- }
- if ((G.rootlen = strlen(pathcomp)) > 0) {
- int had_trailing_pathsep=FALSE, has_drive=FALSE, xtra=2;
- if (isalpha((uch)pathcomp[0]) && pathcomp[1] == ':')
- has_drive = TRUE; /* drive designator */
- if (pathcomp[G.rootlen-1] == '/' || pathcomp[G.rootlen-1] == '\') {
- pathcomp[--G.rootlen] = ' ';
- had_trailing_pathsep = TRUE;
- }
- if (has_drive && (G.rootlen == 2)) {
- if (!had_trailing_pathsep) /* i.e., original wasn't "x:/" */
- xtra = 3; /* room for '.' + '/' + 0 at end of "x:" */
- } else if (G.rootlen > 0) { /* need not check "x:." and "x:/" */
- if (SSTAT(pathcomp, &G.statbuf) || !S_ISDIR(G.statbuf.st_mode))
- {
- /* path does not exist */
- if (!G.create_dirs /* || iswild(pathcomp) */ ) {
- G.rootlen = 0;
- return 2; /* treat as stored file */
- }
- /* create directory (could add loop here to scan pathcomp
- * and create more than one level, but really necessary?) */
- if (MKDIR(pathcomp, 0777) == -1) {
- Info(slide, 1, ((char *)slide,
- "checkdir: cannot create extraction directory: %sn",
- FnFilter1(pathcomp)));
- G.rootlen = 0; /* path didn't exist, tried to create, */
- return 3; /* failed: file exists, or need 2+ levels */
- }
- }
- }
- if ((G.rootpath = (char *)malloc(G.rootlen+xtra)) == NULL) {
- G.rootlen = 0;
- return 10;
- }
- strcpy(G.rootpath, pathcomp);
- if (xtra == 3) /* had just "x:", make "x:." */
- G.rootpath[G.rootlen++] = '.';
- G.rootpath[G.rootlen++] = '/';
- G.rootpath[G.rootlen] = ' ';
- Trace((stderr, "rootpath now = [%s]n", FnFilter1(G.rootpath)));
- }
- return 0;
- }
- #endif /* !SFX || SFX_EXDIR */
- /*---------------------------------------------------------------------------
- END: free rootpath, immediately prior to program exit.
- ---------------------------------------------------------------------------*/
- if (FUNCTION == END) {
- Trace((stderr, "freeing rootpathn"));
- if (G.rootlen > 0) {
- free(G.rootpath);
- G.rootlen = 0;
- }
- return 0;
- }
- return 99; /* should never reach */
- } /* end function checkdir() */
- #ifndef SFX
- #ifndef WINDLL
- /************************/
- /* Function version() */
- /************************/
- void version(__G)
- __GDEF
- {
- int len;
- #if (defined(_MSC_VER) || defined(__WATCOMC__) || defined(__DJGPP__))
- char buf[80];
- #if (defined(_MSC_VER) && (_MSC_VER > 900))
- char buf2[80];
- #endif
- #endif
- len = sprintf((char *)slide, CompiledWith,
- #if defined(_MSC_VER) /* MSC == VC++, but what about SDK compiler? */
- (sprintf(buf, "Microsoft C %d.%02d ", _MSC_VER/100, _MSC_VER%100), buf),
- # if (_MSC_VER == 800)
- "(Visual C++ v1.1)",
- # elif (_MSC_VER == 850)
- "(Windows NT v3.5 SDK)",
- # elif (_MSC_VER == 900)
- "(Visual C++ v2.x)",
- # elif (_MSC_VER > 900)
- (sprintf(buf2, "(Visual C++ %d.%d)", _MSC_VER/100 - 6, _MSC_VER%100/10),
- buf2),
- # else
- "(bad version)",
- # endif
- #elif defined(__WATCOMC__)
- # if (__WATCOMC__ % 10 > 0)
- (sprintf(buf, "Watcom C/C++ %d.%02d", __WATCOMC__ / 100,
- __WATCOMC__ % 100), buf), "",
- # else
- (sprintf(buf, "Watcom C/C++ %d.%d", __WATCOMC__ / 100,
- (__WATCOMC__ % 100) / 10), buf), "",
- # endif
- #elif defined(__BORLANDC__)
- "Borland C++",
- # if (__BORLANDC__ < 0x0200)
- " 1.0",
- # elif (__BORLANDC__ == 0x0200)
- " 2.0",
- # elif (__BORLANDC__ == 0x0400)
- " 3.0",
- # elif (__BORLANDC__ == 0x0410) /* __BCPLUSPLUS__ = 0x0310 */
- " 3.1",
- # elif (__BORLANDC__ == 0x0452) /* __BCPLUSPLUS__ = 0x0320 */
- " 4.0 or 4.02",
- # elif (__BORLANDC__ == 0x0460) /* __BCPLUSPLUS__ = 0x0340 */
- " 4.5",
- # elif (__BORLANDC__ == 0x0500) /* __BCPLUSPLUS__ = 0x0340 */
- " 5.0",
- # elif (__BORLANDC__ == 0x0520) /* __BCPLUSPLUS__ = 0x0520 */
- " 5.2 (C++ Builder)", /* GRR: assume this will stay sync'd? */
- # else
- " later than 5.2",
- # endif
- #elif defined(__LCC__)
- "LCC-Win32", "",
- #elif defined(__GNUC__)
- # if defined(__RSXNT__)
- # if (defined(__DJGPP__) && !defined(__EMX__))
- (sprintf(buf, "rsxnt(djgpp v%d.%02d) / gcc ",
- __DJGPP__, __DJGPP_MINOR__), buf),
- # elif defined(__DJGPP__)
- (sprintf(buf, "rsxnt(emx+djgpp v%d.%02d) / gcc ",
- __DJGPP__, __DJGPP_MINOR__), buf),
- # elif (defined(__GO32__) && !defined(__EMX__))
- "rsxnt(djgpp v1.x) / gcc ",
- # elif defined(__GO32__)
- "rsxnt(emx + djgpp v1.x) / gcc ",
- # elif defined(__EMX__)
- "rsxnt(emx)+gcc ",
- # else
- "rsxnt(unknown) / gcc ",
- # endif
- # elif defined(__CYGWIN32__)
- "cygnus win32 / gcc ",
- # elif defined(__MINGW32__)
- "mingw32 / gcc ",
- # else
- "gcc ",
- # endif
- __VERSION__,
- #else /* !_MSC_VER, !__WATCOMC__, !__BORLANDC__, !__LCC__, !__GNUC__ */
- "unknown compiler (SDK?)", "",
- #endif /* ?compilers */
- "Windows 95 / Windows NT", "n(32-bit)",
- #ifdef __DATE__
- " on ", __DATE__
- #else
- "", ""
- #endif
- );
- (*G.message)((zvoid *)&G, slide, (ulg)len, 0);
- return;
- } /* end function version() */
- #endif /* !WINDLL */
- #endif /* !SFX */
- #ifdef W32_STAT_BANDAID
- /* All currently known variants of WIN32 operating systems (Windows 95/98,
- * WinNT 3.x, 4.0, 5.0) have a nasty bug in the OS kernel concerning
- * conversions between UTC and local time: In the time conversion functions
- * of the Win32 API, the timezone offset (including seasonal daylight saving
- * shift) between UTC and local time evaluation is erratically based on the
- * current system time. The correct evaluation must determine the offset
- * value as it {was/is/will be} for the actual time to be converted.
- *
- * Some versions of MS C runtime lib's stat() returns utc time-stamps so
- * that localtime(timestamp) corresponds to the (potentially false) local
- * time shown by the OS' system programs (Explorer, command shell dir, etc.)
- * The RSXNT port follows the same strategy, but fails to recognize the
- * access-time attribute.
- *
- * For the NTFS file system (and other filesystems that store time-stamps
- * as UTC values), this results in st_mtime (, st_{c|a}time) fields which
- * are not stable but vary according to the seasonal change of "daylight
- * saving time in effect / not in effect".
- *
- * Other C runtime libs (CygWin, or the CRT DLLs supplied with Win95/NT
- * return the unix-time equivalent of the UTC FILETIME values as got back
- * from the Win32 API call. This time, return values from NTFS are correct
- * whereas utimes from files on (V)FAT volumes vary according to the DST
- * switches.
- *
- * To achieve timestamp consistency of UTC (UT extra field) values in
- * Zip archives, the Info-ZIP programs require work-around code for
- * proper time handling in stat() (and other time handling routines).
- */
- /* stat() functions under Windows95 tend to fail for root directories. *
- * Watcom and Borland, at least, are affected by this bug. Watcom made *
- * a partial fix for 11.0 but still missed some cases. This substitute *
- * detects the case and fills in reasonable values. Otherwise we get *
- * effects like failure to extract to a root dir because it's not found. */
- int zstat_win32(__W32STAT_GLOBALS__ const char *path, struct stat *buf)
- {
- if (!stat(path, buf))
- {
- #ifdef NT_TZBUG_WORKAROUND
- /* stat was successful, now redo the time-stamp fetches */
- int fs_uses_loctime = FStampIsLocTime(__G__ path);
- HANDLE h;
- FILETIME Modft, Accft, Creft;
- #ifdef __RSXNT__ /* RSXNT/EMX C rtl uses OEM charset */
- char *ansi_path = (char *)alloca(strlen(path) + 1);
- INTERN_TO_ISO(path, ansi_path);
- # define Ansi_Path ansi_path
- #else
- # define Ansi_Path path
- #endif
- TTrace((stdout, "stat(%s) finds modtime %08lxn", path, buf->st_mtime));
- h = CreateFile(Ansi_Path, GENERIC_READ,
- FILE_SHARE_READ | FILE_SHARE_WRITE, NULL,
- OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
- if (h != INVALID_HANDLE_VALUE) {
- BOOL ftOK = GetFileTime(h, &Creft, &Accft, &Modft);
- CloseHandle(h);
- if (ftOK) {
- FTTrace((stdout, "GetFileTime returned Modft", 0, &Modft));
- FTTrace((stdout, "GetFileTime returned Creft", 0, &Creft));
- if (!fs_uses_loctime) {
- /* On a filesystem that stores UTC timestamps, we refill
- * the time fields of the struct stat buffer by directly
- * using the UTC values as returned by the Win32
- * GetFileTime() API call.
- */
- FileTime2utime(&Modft, &(buf->st_mtime));
- if (Accft.dwLowDateTime != 0 || Accft.dwHighDateTime != 0)
- FileTime2utime(&Accft, &(buf->st_atime));
- else
- buf->st_atime = buf->st_mtime;
- if (Creft.dwLowDateTime != 0 || Creft.dwHighDateTime != 0)
- FileTime2utime(&Creft, &(buf->st_ctime));
- else
- buf->st_ctime = buf->st_mtime;
- TTrace((stdout,"NTFS, recalculated modtime %08lxn",
- buf->st_mtime));
- } else {
- /* On VFAT and FAT-like filesystems, the FILETIME values
- * are converted back to the stable local time before
- * converting them to UTC unix time-stamps.
- */
- VFatFileTime2utime(&Modft, &(buf->st_mtime));
- if (Accft.dwLowDateTime != 0 || Accft.dwHighDateTime != 0)
- VFatFileTime2utime(&Accft, &(buf->st_atime));
- else
- buf->st_atime = buf->st_mtime;
- if (Creft.dwLowDateTime != 0 || Creft.dwHighDateTime != 0)
- VFatFileTime2utime(&Creft, &(buf->st_ctime));
- else
- buf->st_ctime = buf->st_mtime;
- TTrace((stdout, "VFAT, recalculated modtime %08lxn",
- buf->st_mtime));
- }
- }
- }
- # undef Ansi_Path
- #endif /* NT_TZBUG_WORKAROUND */
- return 0;
- }
- #ifdef W32_STATROOT_FIX
- else
- {
- DWORD flags;
- #ifdef __RSXNT__ /* RSXNT/EMX C rtl uses OEM charset */
- char *ansi_path = (char *)alloca(strlen(path) + 1);
- INTERN_TO_ISO(path, ansi_path);
- # define Ansi_Path ansi_path
- #else
- # define Ansi_Path path
- #endif
- flags = GetFileAttributes(Ansi_Path);
- if (flags != 0xFFFFFFFF && flags & FILE_ATTRIBUTE_DIRECTORY) {
- Trace((stderr, "nstat("%s",...) failed on existing directoryn",
- path));
- memset(buf, 0, sizeof(struct stat));
- buf->st_atime = buf->st_ctime = buf->st_mtime =
- dos_to_unix_time(DOSTIME_MINIMUM); /* 1-1-80 */
- buf->st_mode = S_IFDIR | S_IREAD |
- ((flags & FILE_ATTRIBUTE_READONLY) ? 0 : S_IWRITE);
- return 0;
- } /* assumes: stat() won't fail on non-dirs without good reason */
- # undef Ansi_Path
- }
- #endif /* W32_STATROOT_FIX */
- return -1;
- }
- #endif /* W32_STAT_BANDAID */
- #endif /* !FUNZIP */
- #ifndef WINDLL
- /* This replacement getch() function was originally created for Watcom C
- * and then additionally used with CYGWIN. Since UnZip 5.4, all other Win32
- * ports apply this replacement rather that their supplied getch() (or
- * alike) function. There are problems with unabsorbed LF characters left
- * over in the keyboard buffer under Win95 (and 98) when ENTER was pressed.
- * (Under Win95, ENTER returns two(!!) characters: CR-LF.) This problem
- * does not appear when run on a WinNT console prompt!
- */
- /* Watcom 10.6's getch() does not handle Alt+<digit><digit><digit>. */
- /* Note that if PASSWD_FROM_STDIN is defined, the file containing */
- /* the password must have a carriage return after the word, not a */
- /* Unix-style newline (linefeed only). This discards linefeeds. */
- int getch_win32(void)
- {
- HANDLE stin;
- DWORD rc;
- unsigned char buf[2];
- int ret = -1;
- DWORD odemode = ~(DWORD)0;
- # ifdef PASSWD_FROM_STDIN
- stin = GetStdHandle(STD_INPUT_HANDLE);
- # else
- stin = CreateFile("CONIN$", GENERIC_READ | GENERIC_WRITE,
- FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL);
- if (stin == INVALID_HANDLE_VALUE)
- return -1;
- # endif
- if (GetConsoleMode(stin, &odemode))
- SetConsoleMode(stin, ENABLE_PROCESSED_INPUT); /* raw except ^C noticed */
- if (ReadFile(stin, &buf, 1, &rc, NULL) && rc == 1)
- ret = buf[0];
- /* when the user hits return we get CR LF. We discard the LF, not the CR,
- * because when we call this for the first time after a previous input
- * such as the one for "replace foo? [y]es, ..." the LF may still be in
- * the input stream before whatever the user types at our prompt. */
- if (ret == 'n')
- if (ReadFile(stin, &buf, 1, &rc, NULL) && rc == 1)
- ret = buf[0];
- if (odemode != ~(DWORD)0)
- SetConsoleMode(stin, odemode);
- # ifndef PASSWD_FROM_STDIN
- CloseHandle(stin);
- # endif
- return ret;
- }
- #endif /* !WINDLL */