Jason Edmeades : cmd.exe: Add support for dir /A filtering.
Alexandre Julliard
julliard at wine.codeweavers.com
Mon Mar 19 08:17:48 CDT 2007
Module: wine
Branch: master
Commit: d22447946a379fa53201244325c47a8544445f5b
URL: http://source.winehq.org/git/wine.git/?a=commit;h=d22447946a379fa53201244325c47a8544445f5b
Author: Jason Edmeades <us at edmeades.me.uk>
Date: Sun Mar 18 21:55:52 2007 +0000
cmd.exe: Add support for dir /A filtering.
---
programs/cmd/directory.c | 53 ++++++++++++++++++++++++++++++++++++++++++++++
1 files changed, 53 insertions(+), 0 deletions(-)
diff --git a/programs/cmd/directory.c b/programs/cmd/directory.c
index 67a3276..0f514e9 100644
--- a/programs/cmd/directory.c
+++ b/programs/cmd/directory.c
@@ -64,6 +64,7 @@ static DISPLAYTIME dirTime;
static DISPLAYORDER dirOrder;
static BOOL orderReverse, orderGroupDirs, orderGroupDirsReverse, orderByCol;
static BOOL seperator;
+static ULONG showattrs, attrsbits;
/*****************************************************************************
* WCMD_directory
@@ -107,6 +108,8 @@ void WCMD_directory (void) {
orderReverse = FALSE;
orderGroupDirs = FALSE;
orderGroupDirsReverse = FALSE;
+ showattrs = 0;
+ attrsbits = 0;
/* Handle args - Loop through so right most is the effective one */
/* Note: /- appears to be a negate rather than an off, eg. dir
@@ -189,6 +192,45 @@ void WCMD_directory (void) {
}
p = p - 1; /* So when step on, move to '/' */
break;
+ case 'A': p = p + 1;
+ showattrs = 0;
+ attrsbits = 0;
+ if (*p==':') p++; /* Skip optional : */
+ while (*p && *p != '/') {
+ BOOL anegate = FALSE;
+ ULONG mask;
+
+ /* Note /A: - options are 'offs' not toggles */
+ if (*p=='-') {
+ anegate = TRUE;
+ p++;
+ }
+
+ WINE_TRACE("Processing subparm '%c' (in %s)\n", *p, quals);
+ switch (*p) {
+ case 'D': mask = FILE_ATTRIBUTE_DIRECTORY; break;
+ case 'H': mask = FILE_ATTRIBUTE_HIDDEN; break;
+ case 'S': mask = FILE_ATTRIBUTE_SYSTEM; break;
+ case 'R': mask = FILE_ATTRIBUTE_READONLY; break;
+ case 'A': mask = FILE_ATTRIBUTE_ARCHIVE; break;
+ default:
+ SetLastError(ERROR_INVALID_PARAMETER);
+ WCMD_print_error();
+ return;
+ }
+
+ /* Keep running list of bits we care about */
+ attrsbits |= mask;
+
+ /* Mask shows what MUST be in the bits we care about */
+ if (anegate) showattrs = showattrs & ~mask;
+ else showattrs |= mask;
+
+ p++;
+ }
+ p = p - 1; /* So when step on, move to '/' */
+ WINE_TRACE("Result: showattrs %x, bits %x\n", showattrs, attrsbits);
+ break;
default:
SetLastError(ERROR_INVALID_PARAMETER);
WCMD_print_error();
@@ -313,6 +355,9 @@ void WCMD_list_directory (char *search_path, int level) {
return;
}
do {
+ /* Skip any which are filtered out by attribute */
+ if (((fd+entry_count)->dwFileAttributes & attrsbits) != showattrs) continue;
+
entry_count++;
/* Keep running track of longest filename for wide output */
@@ -331,6 +376,14 @@ void WCMD_list_directory (char *search_path, int level) {
} while (FindNextFile(hff, (fd+entry_count)) != 0);
FindClose (hff);
+ /* Handle case where everything is filtered out */
+ if (entry_count == 0) {
+ SetLastError (ERROR_FILE_NOT_FOUND);
+ WCMD_print_error ();
+ HeapFree(GetProcessHeap(),0,fd);
+ return;
+ }
+
/* Sort the list of files */
qsort (fd, entry_count, sizeof(WIN32_FIND_DATA), WCMD_dir_sort);
More information about the wine-cvs
mailing list