source: trunk/dll/valid.c@ 1722

Last change on this file since 1722 was 1691, checked in by Gregg Young, 12 years ago

Fix typo

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 28.6 KB
RevLine 
[48]1
2/***********************************************************************
3
4 $Id: valid.c 1691 2013-08-11 17:00:37Z gyoung $
5
6 File name manipulation routines
7
[103]8 Copyright (c) 1993, 1998 M. Kimes
[1498]9 Copyright (c) 2002, 2010 Steven H.Levine
[48]10
[184]11 23 Nov 02 SHL RootName: rework for sanity
12 27 Nov 02 SHL MakeFullName: correct typo
13 11 Jun 03 SHL Add JFS and FAT32 support
14 15 Jun 04 SHL Implement Jim Read's removable logic
15 31 Jul 04 SHL Comments
16 01 Aug 04 SHL Rework lstrip/rstrip usage
17 03 Jun 05 SHL Drop CD_DEBUG logic
[274]18 28 Nov 05 SHL MakeValidDir: correct DosQuerySysInfo args
[328]19 22 Jul 06 SHL Use Runtime_Error
[530]20 22 Oct 06 GKY Add NDFS32 support
21 22 Oct 06 GKY Increased BUFFER_BYTES in CheckDrive to 8192 to fix NDFS32 scan failure
[552]22 07 Jan 07 GKY Move error strings etc. to string file
23 18 Feb 07 GKY Add more drive types and icons
[697]24 16 Jun 07 SHL Update for OpenWatcom
[793]25 20 Aug 07 GKY Move #pragma alloc_text to end for OpenWatcom compat
[897]26 30 Dec 07 GKY Change TestDates to TestFDates can compare by filename or FDATE/FTIME data
27 30 Dec 07 GKY Add TestCDates to compare CNRITEMs by CDATE/CTIME data
[1104]28 19 Jul 08 GKY Replace save_dir2(dir) with pFM2SaveDirectory
[1358]29 25 Dec 08 GKY Add code to allow write verify to be turned off on a per drive basis
[1375]30 03 Jan 09 GKY Check for system that is protectonly to gray out Dos/Win command lines and prevent
[1394]31 Dos/Win programs from being inserted into the execute dialog with message why.
[1387]32 03 Jan 09 GKY Modify IsExecutable to prevent some text files from being treated as executable
[1394]33 and prevent dlls from being loaded into execute dialog.
[1387]34 05 Jan 09 GKY Fix IsBinary to recognize values above \xc4 (maybe lower) as positive.
[1394]35 When these high codes were it text files they showed as hex in new view.
[1402]36 08 Mar 09 GKY Additional strings move to PCSZs
[1480]37 12 Jul 09 GKY Add xDosQueryAppType and xDosAlloc... to allow FM/2 to load in high memory
[1444]38 22 Jul 09 GKY Consolidated driveflag setting code in DriveFlagsOne
39 22 Jul 09 GKY Check if drives support EAs add driveflag for this
40 22 Jul 09 GKY Add LocalHD driveflag
41 22 Jul 09 GKY Streamline scanning code for faster Tree rescans
[1498]42 17 JAN 10 GKY Changes to get working with Watcom 1.9 Beta (1/16/10). Mostly cast CHAR CONSTANT * as CHAR *.
[1627]43 26 Aug 11 GKY Add a low mem version of xDosAlloc* wrappers; move error checking into all the
44 xDosAlloc* wrappers.
[48]45
46***********************************************************************/
47
[907]48#include <string.h>
49#include <ctype.h>
50
[2]51#define INCL_DOS
52#define INCL_WIN
[551]53#define INCL_DOSDEVIOCTL // DosDevIOCtl
[841]54#define INCL_LONGLONG
[2]55
[1187]56#include "fm3dll.h"
[1214]57#include "treecnr.h" // Data declaration(s)
58#include "info.h" // Data declaration(s)
59#include "notebook.h" // Data declaration(s)
[907]60#include "fm3str.h"
61#include "errutil.h" // Dos_Error...
62#include "strutil.h" // GetPString
[1162]63#include "valid.h"
[1187]64#include "dirs.h" // save_dir2
65#include "strips.h" // bstrip
[1214]66#include "init.h" // GetTidForWindow
[1438]67#include "wrappers.h" // xDosAllocMem
[2]68
[328]69
[1162]70//static BOOL IsDesktop(HAB hab, HWND hwnd);
71//static BOOL IsFileSame(CHAR * filename1, CHAR * filename2);
[1187]72//static char *IsVowel(char a);
[1162]73
[1214]74// Data definitions
75static PSZ pszSrcFile = __FILE__;
76
77#pragma data_seg(GLOBAL2)
[1402]78PCSZ HPFS = "HPFS";
79PCSZ JFS = "JFS";
80PCSZ CDFS = "CDFS";
81PCSZ ISOFS = "ISOFS";
82PCSZ FAT32 = "FAT32";
[1544]83PCSZ FAT = "FAT";
[1402]84PCSZ HPFS386 = "HPFS386";
85PCSZ CBSIFS = "CBSIFS";
86PCSZ NDFS32 = "NDFS32";
87PCSZ RAMFS = "RAMFS";
88PCSZ NTFS = "NTFS";
[1444]89PCSZ LAN = "LAN";
[1354]90BOOL fVerifyOffChecked[26];
[1214]91
[551]92APIRET MakeFullName(char *pszFileName)
[103]93{
[1673]94 // pszFileName must be CCHMAXPATH long minimum!
[2]95
[551]96 char szPathName[CCHMAXPATH];
[2]97 APIRET rc;
98
99 DosError(FERR_DISABLEHARDERR);
[103]100 rc = DosQueryPathInfo(pszFileName,
[551]101 FIL_QUERYFULLNAME, szPathName, sizeof(szPathName));
102 if (!rc)
[103]103 strcpy(pszFileName, szPathName); // Pass back actual name
[2]104 return rc;
105}
106
[551]107char *RootName(char *filename)
[103]108{
[551]109 char *p = NULL, *pp;
[2]110
[48]111 // Return filename, strip path parts
[689]112 // Return empty string when filename ends with backslash
[48]113
[551]114 if (filename) {
115 p = strrchr(filename, '\\');
116 pp = strrchr(filename, '/');
117 p = (p) ? (pp) ? (p > pp) ? p : pp : p : pp;
[2]118 }
[1673]119 if (!p) // name is itself a root
[2]120 p = filename;
[1673]121 else // skip past backslash
[2]122 p++;
123 return p;
124}
125
[915]126 /** TestFDate
[2]127 * return 1 (file2 newer than file1),
128 * 0 (files same)
129 * or -1 (file1 newer than file2)
[897]130 * Make the FILSTATUS pointers NULL if passing file names
131 * if the FILESTATUS information is already available it can be passed instead
132 * Make the files NULL if passing FILESTATUS buffers
[2]133 */
134
[915]135int TestFDates(char *file1, char *file2, FDATE *datevar1, FTIME *timevar1,
[1394]136 FDATE *datevar2, FTIME *timevar2)
[915]137{
[551]138 int comp = 0;
[847]139 FILESTATUS3 fs3o, fs3n;
[2]140
[897]141 if (file1){
[2]142 DosError(FERR_DISABLEHARDERR);
[897]143 DosQueryPathInfo(file1, FIL_STANDARD, &fs3o, sizeof(fs3o));
144 datevar1 = &fs3o.fdateLastWrite;
145 timevar1 = &fs3o.ftimeLastWrite;
[2]146 }
[897]147 if (file2) {
148 DosError(FERR_DISABLEHARDERR);
149 DosQueryPathInfo(file2, FIL_STANDARD, &fs3n, sizeof(fs3n));
150 datevar2 = &fs3n.fdateLastWrite;
151 timevar2 = &fs3n.ftimeLastWrite;
152 }
153 if (&datevar1 && &datevar2 && &timevar1 && &timevar2) {
154 comp = (datevar2->year >
[1394]155 datevar1->year) ? 1 :
[897]156 (datevar2->year <
157 datevar1->year) ? -1 :
158 (datevar2->month >
159 datevar1->month) ? 1 :
160 (datevar2->month <
161 datevar1->month) ? -1 :
162 (datevar2->day >
163 datevar1->day) ? 1 :
164 (datevar2->day <
165 datevar1->day) ? -1 :
166 (timevar2->hours >
167 timevar1->hours) ? 1 :
168 (timevar2->hours <
169 timevar1->hours) ? -1 :
170 (timevar2->minutes >
171 timevar1->minutes) ? 1 :
172 (timevar2->minutes <
173 timevar1->minutes) ? -1 :
174 (timevar2->twosecs >
175 timevar1->twosecs) ? 1 :
176 (timevar2->twosecs < timevar1->twosecs) ? -1 : 0;
177 }
178 return comp;
[2]179}
180
[915]181 /** TestCDate
[897]182 * return 1 (file2 newer than file1),
183 * 0 (files same)
184 * or -1 (file1 newer than file2)
185 */
186
[915]187int TestCDates(CDATE *datevar1, CTIME *timevar1,
[1394]188 CDATE *datevar2, CTIME *timevar2)
[915]189{
[897]190 int comp = 0;
191
192 if (&datevar1 && &datevar2 && &timevar1 && &timevar2) {
193 comp = (datevar2->year >
[1394]194 datevar1->year) ? 1 :
[897]195 (datevar2->year <
196 datevar1->year) ? -1 :
197 (datevar2->month >
198 datevar1->month) ? 1 :
199 (datevar2->month <
200 datevar1->month) ? -1 :
201 (datevar2->day >
202 datevar1->day) ? 1 :
203 (datevar2->day <
204 datevar1->day) ? -1 :
205 (timevar2->hours >
206 timevar1->hours) ? 1 :
207 (timevar2->hours <
208 timevar1->hours) ? -1 :
209 (timevar2->minutes >
210 timevar1->minutes) ? 1 :
211 (timevar2->minutes <
212 timevar1->minutes) ? -1 :
213 (timevar2->seconds >
214 timevar1->seconds) ? 1 :
215 (timevar2->seconds < timevar1->seconds) ? -1 : 0;
216 }
217 return comp;
218}
219
[551]220BOOL IsNewer(char *file1, char *file2)
[103]221{
[1673]222 // return TRUE if file2 is newer than file1
[2]223
[897]224 return (TestFDates(file1, file2, NULL, NULL, NULL, NULL) > 0);
[2]225}
226
[1394]227#if 0 // JBS 11 Sep 08
[551]228BOOL IsDesktop(HAB hab, HWND hwnd)
[103]229{
[2]230 HWND hwndDesktop;
231
[551]232 if (hwnd == HWND_DESKTOP)
[2]233 return TRUE;
[551]234 hwndDesktop = WinQueryDesktopWindow(hab, NULLHANDLE);
235 if (hwnd == hwndDesktop)
[2]236 return TRUE;
237 return FALSE;
238}
[1162]239#endif
[2]240
[551]241BOOL ParentIsDesktop(HWND hwnd, HWND hwndParent)
[103]242{
[2]243 HWND hwndDesktop;
244 BOOL ret = FALSE;
245
[551]246 if (!hwndParent)
247 hwndParent = WinQueryWindow(hwnd, QW_PARENT);
248 if (hwndParent == HWND_DESKTOP)
[2]249 ret = TRUE;
250 else {
[551]251 hwndDesktop = WinQueryDesktopWindow(WinQueryAnchorBlock(hwnd), (HWND) 0);
252 if (hwndDesktop == hwndParent)
[2]253 ret = TRUE;
254 }
255 return ret;
256}
257
[946]258/** CheckDrive
[109]259 * @param chDrive drive letter
260 * @param pszFileSystem pointer to buffer to return file system type or NULL
261 * @param pulType pointer to long word to return drive flags or NULL
262 * @returns removability flag, 1 = removable, 0 = not removable, -1 = error
263 */
[2]264
[551]265INT CheckDrive(CHAR chDrive, CHAR * pszFileSystem, ULONG * pulType)
[70]266{
[1444]267 CHAR szPath[4];
[551]268 VOID *pvBuffer = NULL;
269 CHAR *pfsn;
270 CHAR *pfsd;
271 ULONG clBufferSize;
272 APIRET rc;
273 ULONG ulAction;
274 ULONG clParmBytes;
275 ULONG clDataBytes;
276 HFILE hDev;
[1444]277 EASIZEBUF easize = {0};
278 ULONG ulDataLen = sizeof(EASIZEBUF);
279 ULONG ulParmLen = 0;
[551]280
[103]281# pragma pack(1)
[551]282 struct
283 {
284 BYTE Cmd;
285 BYTE Unit;
286 }
287 parmPkt =
288 {
289 0, 0};
[103]290# define BPB_REMOVABLE_MEDIA 0x08 // 3 - Media is removable
291 struct
292 {
293 BIOSPARAMETERBLOCK bpb;
[551]294 USHORT cCylinders; // Documented but not implemented
295 BYTE bDeviceType; // Documented but not implemented
296 USHORT fsDeviceAttr; // Documented but not implemented
297 }
298 dataPkt;
299
[103]300# pragma pack()
[551]301 BYTE NonRemovable;
[2]302 PFSQBUFFER2 pfsq;
303
[70]304 if (pszFileSystem)
305 *pszFileSystem = 0;
[530]306
[70]307 if (pulType)
308 *pulType = 0;
[2]309
[530]310# define BUFFER_BYTES 8192
[1628]311 if (xDosAllocMem(&pvBuffer, BUFFER_BYTES, pszSrcFile, __LINE__))
[70]312 return -1; // Say failed
[103]313 szPath[0] = chDrive;
314 szPath[1] = ':';
315 szPath[2] = 0;
316 clBufferSize = BUFFER_BYTES;
[2]317 DosError(FERR_DISABLEHARDERR);
[328]318 rc = DosQueryFSAttach(szPath, 0, FSAIL_QUERYNAME,
[551]319 (PFSQBUFFER2) pvBuffer, &clBufferSize);
320 if (rc) {
[1673]321 // can't get any info at all
[103]322 DosFreeMem(pvBuffer);
[2]323 DosError(FERR_DISABLEHARDERR);
[70]324 return -1; // Say failed
[2]325 }
326
[551]327 pfsq = (PFSQBUFFER2) pvBuffer;
[689]328 pfsn = (PCHAR)(pfsq->szName) + pfsq->cbName + 1;
[2]329 pfsd = pfsn + pfsq->cbFSDName + 1;
[1444]330 strupr(pfsn);
[2]331
[551]332 if (pszFileSystem) {
[70]333 strncpy(pszFileSystem, pfsn, CCHMAXPATH);
334 pszFileSystem[CCHMAXPATH - 1] = 0;
[2]335 }
[1444]336 szPath[2] = '\\';
337 szPath[3] = 0;
338 DosFSCtl(&easize, sizeof(EASIZEBUF), &ulDataLen, NULL, ulParmLen,
339 &ulParmLen, FSCTL_MAX_EASIZE, szPath, -1, FSCTL_PATHNAME);
340 //DbgMsg(pszSrcFile, __LINE__, "%i %i %s %s", easize.cbMaxEASize, easize.cbMaxEAListSize, szPath, pfsn);
341 szPath[2] = 0;
342 if (pulType && easize.cbMaxEASize == 0)
343 *pulType |= DRIVE_NOEASUPPORT;
[552]344 if (pulType && (!strcmp(pfsn, CDFS) || !strcmp(pfsn, ISOFS)))
[1444]345 *pulType |= DRIVE_NOTWRITEABLE | DRIVE_CDROM | DRIVE_REMOVABLE;
[555]346 if (pulType && !strcmp(pfsn, NTFS))
[1351]347 *pulType |= DRIVE_NOTWRITEABLE;
[552]348 if (pulType && !strcmp(pfsn, NDFS32)){
[1394]349 *pulType |= DRIVE_VIRTUAL;
[552]350 }
351 if (pulType && !strcmp(pfsn, RAMFS)){
[1394]352 *pulType |= DRIVE_RAMDISK;
[552]353 }
[1444]354 if (((PFSQBUFFER2) pvBuffer)->iType == FSAT_REMOTEDRV) {
[70]355 if (pulType)
356 *pulType |= DRIVE_REMOTE;
[552]357
[551]358 if (pulType && !strcmp(pfsn, CBSIFS)) {
[70]359 *pulType |= DRIVE_ZIPSTREAM;
360 *pulType &= ~DRIVE_REMOTE;
361 *pulType |= DRIVE_NOLONGNAMES;
[551]362 if (pfsq->cbFSAData) {
[103]363 ULONG FType;
[2]364
[551]365 if (CheckDrive(*pfsd, NULL, &FType) != -1) {
[103]366 if (FType & DRIVE_REMOVABLE)
367 *pulType |= DRIVE_REMOVABLE;
368 if (~FType & DRIVE_NOLONGNAMES)
369 *pulType &= ~DRIVE_NOLONGNAMES;
370 }
[552]371
[2]372 }
373 }
[70]374 if (pulType &&
[1444]375 (!strcmp(pfsn, LAN) ||
[1394]376 !strcmp(pfsn, RAMFS) ||
[1444]377 !strcmp(pfsn, NDFS32))) {
[70]378 *pulType &= ~DRIVE_NOLONGNAMES;
379 }
[552]380
[103]381 DosFreeMem(pvBuffer);
382 return 0; // Remotes are non-removable
[2]383 }
384
[70]385 // Local drive
[551]386 if (strcmp(pfsn, HPFS) &&
387 strcmp(pfsn, CDFS) &&
[552]388 strcmp(pfsn, ISOFS) &&
[1444]389 strcmp(pfsn, JFS) &&
[552]390 strcmp(pfsn, FAT32) &&
[555]391 strcmp(pfsn, NTFS) &&
[552]392 strcmp(pfsn, HPFS386)) {
[551]393 if (pulType)
[103]394 (*pulType) |= DRIVE_NOLONGNAMES; // Others can not have long names
[2]395 }
396
397 DosError(FERR_DISABLEHARDERR);
[844]398 rc = DosOpen(szPath, &hDev, &ulAction, 0, 0, FILE_OPEN,
399 OPEN_ACCESS_READONLY | OPEN_SHARE_DENYNONE |
400 OPEN_FLAGS_DASD | OPEN_FLAGS_FAIL_ON_ERROR, 0);
[551]401 if (rc) {
[2]402 DosError(FERR_DISABLEHARDERR);
[70]403 if (pulType)
404 *pulType |= DRIVE_REMOVABLE; // Assume removable if can not access
[103]405 DosFreeMem(pvBuffer);
[70]406 return 1; // Say removable
[2]407 }
[103]408
409 clParmBytes = sizeof(parmPkt.Cmd);
410 clDataBytes = sizeof(NonRemovable);
411 NonRemovable = 1; // Preset as non removable
[2]412 DosError(FERR_DISABLEHARDERR);
[1673]413 rc = DosDevIOCtl(hDev, IOCTL_DISK, DSK_BLOCKREMOVABLE, &parmPkt.Cmd, // Address of the command-specific argument list.
414 sizeof(parmPkt.Cmd), // Length, in bytes, of pParams.
415 &clParmBytes, // Pointer to the length of parameters.
416 &NonRemovable, // Address of the data area.
417 sizeof(NonRemovable),// Length, in bytes, of pData.
418 &clDataBytes); // Pointer to the length of data.
[103]419
[328]420 if (!rc && NonRemovable) {
[103]421 // Could be USB so check BPB flags
422 clParmBytes = sizeof(parmPkt.Cmd);
423 clDataBytes = sizeof(dataPkt);
424 memset(&dataPkt, 0xff, sizeof(dataPkt));
425 DosError(FERR_DISABLEHARDERR);
[1673]426 rc = DosDevIOCtl(hDev, IOCTL_DISK, DSK_GETDEVICEPARAMS, &parmPkt.Cmd,// Address of the command-specific argument list.
427 sizeof(parmPkt.Cmd), // Length, in bytes, of pParams.
428 &clParmBytes, // Pointer to the length of parameters.
429 &dataPkt, // Address of the data area.
430 sizeof(dataPkt), // Length, in bytes, of pData.
431 &clDataBytes); // Pointer to the length of data.
[103]432
[328]433 if (!rc && (dataPkt.bpb.fsDeviceAttr & BPB_REMOVABLE_MEDIA))
[103]434 NonRemovable = 0;
435 }
436
437 DosClose(hDev);
438
[70]439 if (!NonRemovable && pulType)
440 *pulType |= DRIVE_REMOVABLE;
[103]441
442 DosFreeMem(pvBuffer);
443
[70]444 return NonRemovable ? 0 : 1;
[2]445}
446
[1193]447#if 0 // JBS 11 Sep 08
[551]448BOOL IsFileSame(CHAR * filename1, CHAR * filename2)
[103]449{
[1673]450 // returns: -1 (error), 0 (is a directory), or 1 (is a file)
[2]451
[841]452 FILESTATUS3L fsa1, fsa2;
[551]453 APIRET ret;
[2]454
[551]455 if (filename1 && filename2) {
[2]456 DosError(FERR_DISABLEHARDERR);
[841]457 ret = DosQueryPathInfo(filename1, FIL_STANDARDL, &fsa1,
[551]458 (ULONG) sizeof(fsa1));
459 if (!ret) {
[2]460 DosError(FERR_DISABLEHARDERR);
[841]461 ret = DosQueryPathInfo(filename2, FIL_STANDARDL, &fsa2,
[551]462 (ULONG) sizeof(fsa2));
463 if (!ret) {
464 if (fsa1.cbFile == fsa2.cbFile &&
465 (fsa1.attrFile & (~FILE_ARCHIVED)) ==
466 (fsa2.attrFile & (~FILE_ARCHIVED)))
[103]467 return TRUE;
[2]468 }
469 }
470 }
471 return FALSE;
472}
[1162]473#endif
[2]474
[1672]475/**
476 * Check if argument is file or directory
477 * @returns: -1 (error), 0 (is a directory), or 1 (is a file)
478 */
479
[1394]480INT IsFile(PCSZ filename)
[103]481{
[2]482
[847]483 FILESTATUS3 fsa;
[551]484 APIRET ret;
[2]485
[551]486 if (filename && *filename) {
[2]487 DosError(FERR_DISABLEHARDERR);
[1498]488 ret = DosQueryPathInfo((CHAR *) filename, FIL_STANDARD, &fsa, (ULONG) sizeof(fsa));
[551]489 if (!ret)
[2]490 return ((fsa.attrFile & FILE_DIRECTORY) == 0);
[551]491 else if (IsValidDrive(*filename) && IsRoot(filename))
[2]492 return 0;
493 }
[1673]494 return -1; // error; doesn't exist or can't read or null filename
[2]495}
496
[551]497BOOL IsFullName(CHAR * filename)
[103]498{
[2]499 return (filename) ?
[551]500 (isalpha(*filename) && filename[1] == ':' && filename[2] == '\\') : 0;
[2]501}
502
[1394]503BOOL IsRoot(PCSZ filename)
[103]504{
[2]505 return (filename && isalpha(*filename) && filename[1] == ':' &&
[103]506 filename[2] == '\\' && !filename[3]);
[2]507}
508
[551]509BOOL IsValidDir(CHAR * path)
[103]510{
[551]511 CHAR fullname[CCHMAXPATH];
[847]512 FILESTATUS3 fs;
[2]513
[551]514 if (path) {
[2]515 DosError(FERR_DISABLEHARDERR);
[551]516 if (!DosQueryPathInfo(path,
517 FIL_QUERYFULLNAME, fullname, sizeof(fullname))) {
518 if (IsValidDrive(*fullname)) {
519 if (!IsRoot(fullname)) {
[103]520 DosError(FERR_DISABLEHARDERR);
[551]521 if (!DosQueryPathInfo(fullname,
[847]522 FIL_STANDARD,
[551]523 &fs,
524 sizeof(fs)) && (fs.attrFile & FILE_DIRECTORY))
[103]525 return TRUE;
526 }
527 else
528 return TRUE;
[2]529 }
530 }
531 }
532 return FALSE;
533}
534
[551]535BOOL IsValidDrive(CHAR drive)
[103]536{
[551]537 CHAR Path[] = " :", Buffer[256];
[2]538 APIRET Status;
[551]539 ULONG Size;
540 ULONG ulDriveNum, ulDriveMap;
[2]541
[551]542 if (!isalpha(drive) ||
543 (driveflags[toupper(drive) - 'A'] & (DRIVE_IGNORE | DRIVE_INVALID)))
[2]544 return FALSE;
545 DosError(FERR_DISABLEHARDERR);
[551]546 Status = DosQCurDisk(&ulDriveNum, &ulDriveMap);
547 if (!Status) {
[766]548 if (!(ulDriveMap & (1 << (ULONG) (toupper(drive) - 'A'))))
[2]549 return FALSE;
550 Path[0] = toupper(drive);
551 Size = sizeof(Buffer);
552 DosError(FERR_DISABLEHARDERR);
553 Status = DosQueryFSAttach(Path,
[103]554 0,
[551]555 FSAIL_QUERYNAME, (PFSQBUFFER2) Buffer, &Size);
[2]556 }
557 return (Status == 0);
558}
559
[274]560//=== MakeValidDir() build valid directory name ===
[2]561
[551]562CHAR *MakeValidDir(CHAR * path)
[103]563{
[551]564 ULONG ulDrv;
565 CHAR *p;
[847]566 FILESTATUS3 fs;
[551]567 APIRET rc;
[2]568
[274]569 if (!MakeFullName(path)) {
570 if (IsValidDrive(*path)) {
571 // Passed name is valid - trim to directory
572 for (;;) {
573 if (IsRoot(path))
[103]574 return path;
575 DosError(FERR_DISABLEHARDERR);
[847]576 rc = DosQueryPathInfo(path, FIL_STANDARD, &fs, sizeof(fs));
[274]577 if (!rc && (fs.attrFile & FILE_DIRECTORY))
[103]578 return path;
[551]579 p = strrchr(path, '\\');
[274]580 if (p) {
581 if (p < path + 3)
[103]582 p++;
583 *p = 0;
584 }
585 else
586 break;
[2]587 }
588 }
589 }
[274]590 // Fall back to boot drive
[2]591 DosError(FERR_DISABLEHARDERR);
[551]592 if (!DosQuerySysInfo(QSV_BOOT_DRIVE, QSV_BOOT_DRIVE, &ulDrv, sizeof(ulDrv))) {
[274]593 ulDrv += '@';
594 if (ulDrv < 'C')
595 ulDrv = 'C';
[551]596 strcpy(path, " :\\");
597 *path = (CHAR) ulDrv;
[2]598 }
599 else
[1104]600 strcpy(path, pFM2SaveDirectory); // Fall back to fm3.ini drive or current dir - should never occur
[2]601 return path;
602}
603
[1480]604BOOL IsExecutable(CHAR *filename)
[103]605{
[2]606 register CHAR *p;
[551]607 APIRET ret;
608 ULONG apptype;
[2]609
[551]610 if (filename) {
[2]611 DosError(FERR_DISABLEHARDERR);
[551]612 p = strrchr(filename, '.');
613 if (p)
[1480]614 ret = DosQueryAppType(filename, &apptype);
[2]615 else {
616
617 char fname[CCHMAXPATH + 2];
618
[551]619 strcpy(fname, filename);
620 strcat(fname, ".");
[1480]621 ret = DosQueryAppType(fname, &apptype);
[1369]622 }
[1372]623 if (apptype & (FAPPTYP_DLL |
[1394]624 FAPPTYP_PHYSDRV |
625 FAPPTYP_VIRTDRV |
626 FAPPTYP_PROTDLL))
[1372]627 return FALSE;
628 if (apptype == 0x000b && (!p ||
[1398]629 (stricmp(p, PCSZ_DOTEXE) &&
630 stricmp(p, PCSZ_DOTCOM) &&
631 stricmp(p, PCSZ_DOTCMD) &&
632 stricmp(p, PCSZ_DOTBAT) &&
633 stricmp(p, PCSZ_DOTBTM))))
[1372]634 return FALSE;
[1369]635 if (!fProtectOnly) {
636 if ((!ret && (!apptype ||
[1394]637 (apptype &
638 (FAPPTYP_NOTWINDOWCOMPAT |
639 FAPPTYP_WINDOWCOMPAT |
640 FAPPTYP_WINDOWAPI |
641 FAPPTYP_BOUND |
642 FAPPTYP_DOS |
643 FAPPTYP_WINDOWSREAL |
644 FAPPTYP_WINDOWSPROT |
645 FAPPTYP_32BIT |
646 FAPPTYP_WINDOWSPROT31)))) ||
[1398]647 (p && (!stricmp(p, PCSZ_DOTCMD) || !stricmp(p, PCSZ_DOTBAT) || !stricmp(p, PCSZ_DOTBTM))))
[1394]648 return TRUE;
[1369]649 }
650 else if ((!ret && (!apptype ||
[1394]651 (apptype &
652 (FAPPTYP_NOTWINDOWCOMPAT |
653 FAPPTYP_WINDOWCOMPAT |
654 FAPPTYP_WINDOWAPI |
655 FAPPTYP_BOUND |
656 FAPPTYP_32BIT)))) ||
[1398]657 (p && (!stricmp(p, PCSZ_DOTCMD) || !stricmp(p, PCSZ_DOTBTM))))
[2]658 return TRUE;
[1369]659 if (fProtectOnly && (apptype &
[1394]660 (FAPPTYP_DOS |
661 FAPPTYP_WINDOWSREAL |
662 FAPPTYP_WINDOWSPROT |
663 FAPPTYP_WINDOWSPROT31)) &&
[1398]664 (p && (!stricmp(p, PCSZ_DOTEXE) || !stricmp(p, PCSZ_DOTCOM) ||
665 !stricmp(p, PCSZ_DOTBAT))))
[1369]666 saymsg(MB_OK,
[1394]667 HWND_DESKTOP,
668 NullStr,
669 GetPString(IDS_NOTPROTECTONLYEXE),
670 filename);
[2]671 }
672 return FALSE;
673}
674
[551]675VOID ArgDriveFlags(INT argc, CHAR ** argv)
[103]676{
[2]677 INT x;
678
[551]679 for (x = 1; x < argc; x++) {
680 if (*argv[x] == '/' && isalpha(argv[x][1])) {
[2]681
682 CHAR *p = &argv[x][1];
683
[551]684 while (isalpha(*p)) {
[103]685 driveflags[toupper(*p) - 'A'] |= DRIVE_IGNORE;
686 p++;
[2]687 }
688 }
[551]689 else if (*argv[x] == ';' && isalpha(argv[x][1])) {
[2]690
691 CHAR *p = &argv[x][1];
692
[551]693 while (isalpha(*p)) {
[103]694 driveflags[toupper(*p) - 'A'] |= DRIVE_NOPRESCAN;
695 p++;
[2]696 }
697 }
[552]698 else if (*argv[x] == '`' && isalpha(argv[x][1])) {
699
700 CHAR *p = &argv[x][1];
701
702 while (isalpha(*p)) {
703 driveflags[toupper(*p) - 'A'] |= DRIVE_NOSTATS;
704 p++;
705 }
706 }
[551]707 else if (*argv[x] == ',' && isalpha(argv[x][1])) {
[2]708
709 CHAR *p = &argv[x][1];
710
[551]711 while (isalpha(*p)) {
[103]712 driveflags[toupper(*p) - 'A'] |= DRIVE_NOLOADICONS;
713 p++;
[2]714 }
715 }
[552]716 else if (*argv[x] == '-' && isalpha(argv[x][1])) {
[2]717
718 CHAR *p = &argv[x][1];
719
[551]720 while (isalpha(*p)) {
[103]721 driveflags[toupper(*p) - 'A'] |= DRIVE_NOLOADSUBJS;
722 p++;
[2]723 }
724 }
[551]725 else if (*argv[x] == '\'' && isalpha(argv[x][1])) {
[2]726
727 CHAR *p = &argv[x][1];
728
[551]729 while (isalpha(*p)) {
[103]730 driveflags[toupper(*p) - 'A'] |= DRIVE_NOLOADLONGS;
731 p++;
[2]732 }
733 }
734 }
735}
736
[1444]737VOID DriveFlagsOne(INT x, CHAR *FileSystem, VOID *volser)
[70]738{
[551]739 INT removable;
[1444]740 CHAR szDrive[] = " :\\";
[551]741 ULONG drvtype;
[2]742
[551]743 *szDrive = (CHAR) (x + 'A');
[2]744 *FileSystem = 0;
745 drvtype = 0;
[551]746 removable = CheckDrive(*szDrive, FileSystem, &drvtype);
[1444]747 strupr(FileSystem);
[2]748 driveserial[x] = -1;
749 driveflags[x] &= (DRIVE_IGNORE | DRIVE_NOPRESCAN | DRIVE_NOLOADICONS |
[103]750 DRIVE_NOLOADSUBJS | DRIVE_NOLOADLONGS |
[1394]751 DRIVE_INCLUDEFILES | DRIVE_SLOW | DRIVE_NOSTATS |
752 DRIVE_WRITEVERIFYOFF);
[551]753 if (removable != -1) {
[1444]754 if (!volser) {
755 struct
756 {
757 ULONG serial;
758 CHAR volumelength;
759 CHAR volumelabel[CCHMAXPATH];
760 }
761 volserl;
762
763 DosError(FERR_DISABLEHARDERR);
764 if (!DosQueryFSInfo((ULONG) x + 1, FSIL_VOLSER, &volserl, sizeof(volserl)))
765 driveserial[x] = volserl.serial;
766 else
767 DosError(FERR_DISABLEHARDERR);
[551]768 }
[1444]769 else {
[2]770 DosError(FERR_DISABLEHARDERR);
[1444]771 if (DosQueryFSInfo((ULONG) x + 1, FSIL_VOLSER, volser,
772 sizeof(ULONG) + sizeof(CHAR) + CCHMAXPATH))
773 DosError(FERR_DISABLEHARDERR);
774 }
[2]775 }
776 else
777 driveflags[x] |= DRIVE_INVALID;
[1444]778 driveflags[x] |= ((removable == -1 || removable == 1) ? DRIVE_REMOVABLE : 0);
[551]779 if (drvtype & DRIVE_REMOTE)
[2]780 driveflags[x] |= DRIVE_REMOTE;
[1444]781 if (drvtype & DRIVE_NOEASUPPORT)
782 driveflags[x] |= DRIVE_NOEASUPPORT;
783 if(!strcmp(FileSystem,NDFS32)){
[552]784 driveflags[x] |= DRIVE_VIRTUAL;
785 driveflags[x] &= (~DRIVE_REMOTE);
786 }
[1444]787 if(!strcmp(FileSystem,RAMFS)){
[552]788 driveflags[x] |= DRIVE_RAMDISK;
789 driveflags[x] &= (~DRIVE_REMOTE);
790 }
[1444]791 if(!strcmp(FileSystem,NTFS))
[555]792 driveflags[x] |= DRIVE_NOTWRITEABLE;
[551]793 if (strcmp(FileSystem, HPFS) &&
794 strcmp(FileSystem, CDFS) &&
[552]795 strcmp(FileSystem, ISOFS) &&
[1444]796 strcmp(FileSystem, JFS) &&
797 strcmp(FileSystem, LAN) &&
[552]798 strcmp(FileSystem, RAMFS) &&
799 strcmp(FileSystem, FAT32) &&
[555]800 strcmp(FileSystem, NTFS) &&
801 strcmp(FileSystem, NDFS32) &&
[552]802 strcmp(FileSystem, HPFS386)) {
[2]803 driveflags[x] |= DRIVE_NOLONGNAMES;
[70]804 }
[1444]805 if ((!strcmp(FileSystem, HPFS) ||
[1544]806 !strcmp(FileSystem, FAT) ||
[1444]807 !strcmp(FileSystem, JFS) ||
808 !strcmp(FileSystem, FAT32) ||
809 !strcmp(FileSystem, NTFS) ||
810 !strcmp(FileSystem, HPFS386)) && ~driveflags[x] & DRIVE_REMOVABLE) {
811 driveflags[x] |= DRIVE_LOCALHD;
812 }
[552]813 if (!strcmp(FileSystem, CDFS) || !strcmp(FileSystem, ISOFS)) {
[551]814 driveflags[x] |= (DRIVE_REMOVABLE | DRIVE_NOTWRITEABLE | DRIVE_CDROM);
[2]815 }
[1444]816 if (!strcmp(FileSystem, CBSIFS)) {
[2]817 driveflags[x] |= DRIVE_ZIPSTREAM;
818 driveflags[x] &= (~DRIVE_REMOTE);
[551]819 if (drvtype & DRIVE_REMOVABLE)
[2]820 driveflags[x] |= DRIVE_REMOVABLE;
[551]821 if (!(drvtype & DRIVE_NOLONGNAMES))
[2]822 driveflags[x] &= (~DRIVE_NOLONGNAMES);
823 }
824}
825
[551]826VOID FillInDriveFlags(VOID * dummy)
[103]827{
[1354]828 ULONG ulDriveNum, ulDriveMap, size;
[2]829 register INT x;
830
[551]831 for (x = 0; x < 26; x++)
[2]832 driveflags[x] &= (DRIVE_IGNORE | DRIVE_NOPRESCAN | DRIVE_NOLOADICONS |
[103]833 DRIVE_NOLOADSUBJS | DRIVE_NOLOADLONGS |
[1394]834 DRIVE_INCLUDEFILES | DRIVE_SLOW | DRIVE_NOSTATS |
835 DRIVE_WRITEVERIFYOFF);
[551]836 memset(driveserial, -1, sizeof(driveserial));
[2]837 DosError(FERR_DISABLEHARDERR);
[551]838 DosQCurDisk(&ulDriveNum, &ulDriveMap);
839 for (x = 0; x < 26; x++) {
[766]840 if (ulDriveMap & (1 << x) && !(driveflags[x] & DRIVE_IGNORE)) {
[2]841 {
[1394]842 ULONG flags = 0, size = sizeof(ULONG);
843 CHAR FlagKey[80];
[2]844
[1394]845 sprintf(FlagKey, "%c.DriveFlags", (CHAR) (x + 'A'));
846 if (PrfQueryProfileData(fmprof, appname, FlagKey, &flags, &size) &&
847 size == sizeof(ULONG))
848 driveflags[x] |= flags;
[2]849 }
850
[551]851 if (x > 1) {
[1444]852 if (!(driveflags[x] & DRIVE_NOPRESCAN)) {
853 CHAR FileSystem[CCHMAXPATH];
854 DriveFlagsOne(x, FileSystem, NULL);
855 }
[103]856 else
857 driveserial[x] = -1;
[2]858 }
859 else {
[103]860 driveflags[x] |= (DRIVE_REMOVABLE | DRIVE_NOLONGNAMES);
861 driveserial[x] = -1;
[2]862 }
863 }
[766]864 else if (!(ulDriveMap & (1 << x)))
[2]865 driveflags[x] |= DRIVE_INVALID;
866 }
867 {
[551]868 ULONG startdrive = 3L;
[2]869
870 DosError(FERR_DISABLEHARDERR);
[551]871 DosQuerySysInfo(QSV_BOOT_DRIVE, QSV_BOOT_DRIVE,
872 (PVOID) & startdrive, (ULONG) sizeof(ULONG));
873 if (startdrive)
[2]874 driveflags[startdrive - 1] |= DRIVE_BOOT;
875 }
[1354]876 {
877 INT x;
878 CHAR Key[80];
879
880 for (x = 2; x < 26; x++) {
881 sprintf(Key, "%c.VerifyOffChecked", (CHAR) (x + 'A'));
882 size = sizeof(BOOL);
883 PrfQueryProfileData(fmprof, appname, Key, &fVerifyOffChecked[x], &size);
884 if (!fVerifyOffChecked[x]) {
[1394]885 if (driveflags[x] & DRIVE_REMOVABLE)
886 driveflags[x] |= DRIVE_WRITEVERIFYOFF;
887 if (!(driveflags[x] & DRIVE_INVALID)) {
888 fVerifyOffChecked[x] = TRUE;
889 PrfWriteProfileData(fmprof, appname, Key, &fVerifyOffChecked[x], sizeof(BOOL));
890 }
[1354]891 }
892 }
893 }
[2]894}
895
[551]896CHAR *assign_ignores(CHAR * s)
[103]897{
[551]898 register INT x;
899 register CHAR *p, *pp;
[2]900
901 *s = '/';
902 s[1] = 0;
903 p = s + 1;
[551]904 if (s) {
905 for (x = 0; x < 26; x++) {
906 if ((driveflags[x] & DRIVE_IGNORE) != 0) {
907 *p = (CHAR) x + 'A';
[103]908 p++;
909 *p = 0;
[2]910 }
911 }
912 }
[551]913 if (!s[1]) {
[2]914 *s = 0;
915 pp = s;
916 }
917 else {
918 pp = &s[strlen(s)];
919 *pp = ' ';
920 pp++;
921 }
922 *pp = ';';
923 pp[1] = 0;
924 p = pp + 1;
[551]925 if (pp) {
926 for (x = 0; x < 26; x++) {
927 if ((driveflags[x] & DRIVE_NOPRESCAN) != 0) {
928 *p = (CHAR) x + 'A';
[103]929 p++;
930 *p = 0;
[2]931 }
932 }
933 }
[551]934 if (!pp[1])
[2]935 *pp = 0;
936 pp = &s[strlen(s)];
937 *pp = ' ';
938 pp++;
939 *pp = ',';
940 pp[1] = 0;
941 p = pp + 1;
[551]942 if (pp) {
943 for (x = 0; x < 26; x++) {
944 if ((driveflags[x] & DRIVE_NOLOADICONS) != 0) {
945 *p = (CHAR) x + 'A';
[103]946 p++;
947 *p = 0;
[2]948 }
949 }
950 }
[551]951 if (!pp[1])
[2]952 *pp = 0;
953 pp = &s[strlen(s)];
954 *pp = ' ';
955 pp++;
[552]956 *pp = '-';
[2]957 pp[1] = 0;
958 p = pp + 1;
[551]959 if (pp) {
960 for (x = 0; x < 26; x++) {
961 if ((driveflags[x] & DRIVE_NOLOADSUBJS) != 0) {
962 *p = (CHAR) x + 'A';
[103]963 p++;
964 *p = 0;
[2]965 }
966 }
967 }
[551]968 if (!pp[1])
[2]969 *pp = 0;
970 pp = &s[strlen(s)];
971 *pp = ' ';
972 pp++;
[552]973 *pp = '`';
974 pp[1] = 0;
975 p = pp + 1;
976 if (pp) {
977 for (x = 0; x < 26; x++) {
978 if ((driveflags[x] & DRIVE_NOSTATS) != 0) {
979 *p = (CHAR) x + 'A';
980 p++;
981 *p = 0;
982 }
983 }
984 }
985 if (!pp[1])
986 *pp = 0;
987 pp = &s[strlen(s)];
988 *pp = ' ';
989 pp++;
[2]990 *pp = '\'';
991 pp[1] = 0;
992 p = pp + 1;
[551]993 if (pp) {
994 for (x = 0; x < 26; x++) {
995 if ((driveflags[x] & DRIVE_NOLOADLONGS) != 0) {
996 *p = (CHAR) x + 'A';
[103]997 p++;
998 *p = 0;
[2]999 }
1000 }
1001 }
[551]1002 if (!pp[1])
[2]1003 *pp = 0;
[123]1004 bstrip(s);
[2]1005 return s;
1006}
1007
[1394]1008BOOL needs_quoting(PCSZ f)
[103]1009{
[2]1010 register CHAR *p = " &|<>";
1011
[551]1012 while (*p) {
1013 if (strchr(f, *p))
[2]1014 return TRUE;
1015 p++;
1016 }
1017 return FALSE;
1018}
1019
[551]1020BOOL IsBinary(register CHAR * str, ULONG len)
[103]1021{
[766]1022 register ULONG x = 0;
[2]1023
[551]1024 if (str) {
1025 while (x < len) {
[1386]1026 if ((UINT) str[x] < ' ' && str[x] != '\r' && str[x] != '\n' && str[x] != '\t'
1027 && str[x] != '\x1b' && str[x] != '\x1a' && str[x] != '\x07'
[1394]1028 && str[x] != '\x0c') {
1029 return TRUE;
[1386]1030 }
[2]1031 x++;
1032 }
1033 }
1034 return FALSE;
1035}
1036
[551]1037BOOL TestBinary(CHAR * filename)
[103]1038{
[551]1039 HFILE handle;
1040 ULONG ulAction;
1041 ULONG len;
1042 APIRET rc;
[850]1043 CHAR buff[4096]; // 06 Oct 07 SHL protect against NTFS defect
[2]1044
[551]1045 if (filename) {
[844]1046 if (!DosOpen(filename, &handle, &ulAction, 0, 0,
1047 OPEN_ACTION_FAIL_IF_NEW | OPEN_ACTION_OPEN_IF_EXISTS,
1048 OPEN_FLAGS_FAIL_ON_ERROR | OPEN_FLAGS_NOINHERIT |
1049 OPEN_FLAGS_SEQUENTIAL | OPEN_SHARE_DENYNONE |
1050 OPEN_ACCESS_READONLY, 0)) {
[2]1051 len = 512;
[551]1052 rc = DosRead(handle, buff, len, &len);
[2]1053 DosClose(handle);
[551]1054 if (!rc && len)
1055 return IsBinary(buff, len);
[2]1056 }
1057 }
1058 return FALSE;
1059}
1060
[1193]1061#if 0 // JBS 11 Sep 08
[551]1062char *IsVowel(char a)
[103]1063{
[551]1064 return (strchr("aeiouAEIOU", a) != NULL) ? "n" : NullStr;
[2]1065}
[1187]1066#endif
[2]1067
[551]1068VOID GetDesktopName(CHAR * objectpath, ULONG size)
[103]1069{
[551]1070 PFN WQDPath;
[2]1071 HMODULE hmod = 0;
[551]1072 APIRET rc;
[766]1073 ULONG startdrive = 3;
[551]1074 CHAR objerr[CCHMAXPATH];
[2]1075
[328]1076 if (!objectpath) {
1077 Runtime_Error(pszSrcFile, __LINE__, "null pointer");
[2]1078 return;
[328]1079 }
[2]1080 *objectpath = 0;
[328]1081 if (OS2ver[0] > 20 || (OS2ver[0] == 20 && OS2ver[1] >= 30)) {
[1673]1082 /**
[2]1083 * if running under warp, we can get the desktop name
1084 * this way...
1085 */
[551]1086 rc = DosLoadModule(objerr, sizeof(objerr), "PMWP", &hmod);
1087 if (!rc) {
1088 rc = DosQueryProcAddr(hmod, 262, NULL, &WQDPath);
1089 if (!rc)
1090 WQDPath(objectpath, size);
[2]1091 DosFreeModule(hmod);
1092 }
1093 }
[328]1094 if (!*objectpath) {
1095 // Fall back to INI content
1096 if (!PrfQueryProfileString(HINI_SYSTEMPROFILE,
1097 "FolderWorkareaRunningObjects",
1098 NULL,
1099 "\0",
[551]1100 (PVOID) objectpath, sizeof(objectpath))) {
1101 Win_Error(HWND_DESKTOP, HWND_DESKTOP, pszSrcFile, __LINE__,
[1398]1102 PCSZ_PRFQUERYPROFILESTRING);
[2]1103 *objectpath = 0;
[328]1104 }
1105 else if (!*objectpath || IsFile(objectpath)) {
1106 Runtime_Error(pszSrcFile, __LINE__, "bad FolderWorkareaRunningObjects");
1107 *objectpath = 0;
1108 }
1109 if (!*objectpath) {
[552]1110 // Fall back
[2]1111 DosError(FERR_DISABLEHARDERR);
[551]1112 DosQuerySysInfo(QSV_BOOT_DRIVE, QSV_BOOT_DRIVE,
1113 (PVOID) & startdrive, (ULONG) sizeof(ULONG));
[552]1114 sprintf(objectpath, GetPString(IDS_PATHTODESKTOP), ((CHAR) startdrive) + '@');
[2]1115 }
1116 }
1117}
[793]1118
[1691]1119#pragma alloc_text(VALID,CheckDrive,IsRoot,IsFile,IsFullName,needs_quoting)
[793]1120#pragma alloc_text(VALID,IsValidDir,IsValidDrive,MakeValidDir,IsVowel)
[897]1121#pragma alloc_text(VALID,IsFileSame,IsNewer,TestFDates,TestCDates,RootName,MakeFullName)
[793]1122#pragma alloc_text(VALID,IsExecutable,IsBinary,IsDesktop,ParentIsDesktop)
1123#pragma alloc_text(FILLFLAGS,FillInDriveFlags,assign_ignores)
1124#pragma alloc_text(FILLFLAGS,ArgDriveFlags,DriveFlagsOne)
1125#pragma alloc_text(FINDDESK,GetDesktopName)
Note: See TracBrowser for help on using the repository browser.