/*++ Copyright (c) 1991 Microsoft Corporation Module Name: attrib.cxx Abstract: This utility allows the user to change file attributes. It is functionaly compatible with DOS 5 attrib utility. Author: Jaime F. Sasson Environment: ULIB, User Mode --*/ #include "ulib.hxx" #include "error.hxx" #include "arg.hxx" #include "array.hxx" #include "path.hxx" #include "wstring.hxx" #include "substrng.hxx" #include "dir.hxx" #include "filter.hxx" #include "system.hxx" #include "arrayit.hxx" #include "stream.hxx" #include "smsg.hxx" #include "rtmsg.h" #include "attrib.hxx" PSTREAM Get_Standard_Input_Stream(); PSTREAM Get_Standard_Output_Stream(); extern "C" { #include } ERRSTACK* perrstk; DEFINE_CONSTRUCTOR( ATTRIB, PROGRAM ); BOOLEAN ATTRIB::Initialize( ) /*++ Routine Description: Initializes an ATTRIB class. Arguments: None. Return Value: BOOLEAN - Indicates if the initialization succeeded. --*/ { PWSTRING DynSubDirectory; PWSTRING DynSubFileName; ARGUMENT_LEXEMIZER ArgLex; ARRAY LexArray; ARRAY ArgumentArray; STRING_ARGUMENT ProgramNameArgument; // STRING_ARGUMENT FileNameArgument; PCWSTRING FullFileNameString; PATH DirectoryNamePath; PCWSTRING InvalidName; PCWSTRING TempPathString; DSTRING TempPathStringDrive; PATH PathDrive; DSTRING BackSlashString; STRING_ARGUMENT InvalidSwitch; STRING_ARGUMENT InvalidSwitchPlus; STRING_ARGUMENT InvalidSwitchMinus; PWSTRING InvalidArgument; DSTRING InvalidSwitchString; BOOLEAN ActOnDirectory; _InitialDirectory = NULL; // // Initialize MESSAGE object // _OutStream = Get_Standard_Output_Stream(); _Message.Initialize( _OutStream, Get_Standard_Input_Stream() ); // // Initialize string that contains End-Of-Line characters // if( !_EndOfLineString.Initialize( (LPWSTR)L"\r\n" ) ) { DebugPrint( "_EndOfLineString.Initialize() failed" ); return( FALSE ); } // // Parse command line // if ( !LexArray.Initialize( ) ) { DebugPrint( "LexArray.Initialize() failed \n" ); return( FALSE ); } if ( !ArgLex.Initialize( &LexArray ) ) { DebugPrint( "ArgLex.Initialize() failed \n" ); return( FALSE ); } ArgLex.PutSwitches( "/" ); ArgLex.SetCaseSensitive( FALSE ); ArgLex.PutStartQuotes( "\""); ArgLex.PutEndQuotes( "\""); ArgLex.PutSeparators( " \t" ); if( !ArgLex.PrepareToParse() ) { DebugPrint( "ArgLex.PrepareToParse() failed \n" ); _Message.Set( MSG_ATTRIB_PARAMETER_NOT_CORRECT ); _Message.Display( " " ); return( FALSE ); } if ( !ArgumentArray.Initialize() ) { DebugPrint( "ArgumentArray.Initialize() failed \n" ); return( FALSE ); } if( !ProgramNameArgument.Initialize("*") || !_FlagRemoveSystemAttribute.Initialize( "-S" ) || !_FlagAddSystemAttribute.Initialize( "+S" ) || !_FlagRemoveHiddenAttribute.Initialize( "-H" ) || !_FlagAddHiddenAttribute.Initialize( "+H" ) || !_FlagRemoveReadOnlyAttribute.Initialize( "-R" ) || !_FlagAddReadOnlyAttribute.Initialize( "+R" ) || !_FlagRemoveArchiveAttribute.Initialize( "-A" ) || !_FlagAddArchiveAttribute.Initialize( "+A" ) || !_FlagRecurseDirectories.Initialize( "/S" ) || !_FlagDisplayHelp.Initialize( "/?" ) || !InvalidSwitch.Initialize( "/*" ) || !InvalidSwitchPlus.Initialize( "+*" ) || !InvalidSwitchMinus.Initialize( "-*" ) || !_FileNameArgument.Initialize( "*" ) ) { DebugPrint( "Unable to initialize flag or string arguments \n" ); return( FALSE ); } if( !ArgumentArray.Put( &ProgramNameArgument ) || !ArgumentArray.Put( &_FlagRemoveSystemAttribute ) || !ArgumentArray.Put( &_FlagAddSystemAttribute ) || !ArgumentArray.Put( &_FlagRemoveHiddenAttribute ) || !ArgumentArray.Put( &_FlagAddHiddenAttribute ) || !ArgumentArray.Put( &_FlagRemoveReadOnlyAttribute ) || !ArgumentArray.Put( &_FlagAddReadOnlyAttribute ) || !ArgumentArray.Put( &_FlagRemoveArchiveAttribute ) || !ArgumentArray.Put( &_FlagAddArchiveAttribute ) || !ArgumentArray.Put( &_FlagRecurseDirectories ) || !ArgumentArray.Put( &_FlagDisplayHelp ) || !ArgumentArray.Put( &InvalidSwitch ) || !ArgumentArray.Put( &InvalidSwitchPlus ) || !ArgumentArray.Put( &InvalidSwitchMinus ) || !ArgumentArray.Put( &_FileNameArgument ) ) { DebugPrint( "ArgumentArray.Put() failed \n" ); return( FALSE ); } if( !ArgLex.DoParsing( &ArgumentArray ) ) { _Message.Set( MSG_ATTRIB_PARAMETER_NOT_CORRECT ); _Message.Display( " " ); return( FALSE ); } // // Check the existance of an invalid switch // if( InvalidSwitch.IsValueSet() || InvalidSwitchPlus.IsValueSet() || InvalidSwitchMinus.IsValueSet() ) { if( InvalidSwitch.IsValueSet() ) { // // The invalid switch starts with '/' // if( !InvalidSwitchString.Initialize( "/" ) ) { DebugPrint( "InvalidSwitchString.Initialize( / ) failed \n" ); return( FALSE ); } InvalidArgument = InvalidSwitch.GetString(); DebugPtrAssert( InvalidArgument ); } else if ( InvalidSwitchPlus.IsValueSet() ) { // // The invalid switch starts with '+' // if( !InvalidSwitchString.Initialize( "+" ) ) { DebugPrint( "InvalidSwitchString.Initialize( + ) failed \n" ); return( FALSE ); } InvalidArgument = InvalidSwitchPlus.GetString(); DebugPtrAssert( InvalidArgument ); } else { // // The invalid switch starts with '-' // if( !InvalidSwitchString.Initialize( "-" ) ) { DebugPrint( "InvalidSwitchString.Initialize( - ) failed \n" ); return( FALSE ); } InvalidArgument = InvalidSwitchMinus.GetString(); DebugPtrAssert( InvalidArgument ); } // // Display the error message followed by the invalid switch // if( !InvalidSwitchString.Strcat( InvalidArgument ) ) { DebugPrint( "InvalidSwitchString.Strcat( InvalidArgument ) failed \n" ); return( FALSE ); } _Message.Set( MSG_ATTRIB_INVALID_SWITCH ); _Message.Display( "%W", &InvalidSwitchString ); return( FALSE ); } // // +S -S or +H -H or +R -R or +A -A are not valid // combination of arguments // if( ( _FlagRemoveSystemAttribute.QueryFlag() && _FlagAddSystemAttribute.QueryFlag() ) || ( _FlagRemoveHiddenAttribute.QueryFlag() && _FlagAddHiddenAttribute.QueryFlag() ) || ( _FlagRemoveReadOnlyAttribute.QueryFlag() && _FlagAddReadOnlyAttribute.QueryFlag() ) || ( _FlagRemoveArchiveAttribute.QueryFlag() && _FlagAddArchiveAttribute.QueryFlag() ) ) { _Message.Set( MSG_ATTRIB_PARAMETER_NOT_CORRECT ); _Message.Display( " " ); return( FALSE ); } if( _FlagRemoveSystemAttribute.QueryFlag() || _FlagAddSystemAttribute.QueryFlag() || _FlagRemoveHiddenAttribute.QueryFlag() || _FlagAddHiddenAttribute.QueryFlag() || _FlagRemoveReadOnlyAttribute.QueryFlag() || _FlagAddReadOnlyAttribute.QueryFlag() || _FlagRemoveArchiveAttribute.QueryFlag() || _FlagAddArchiveAttribute.QueryFlag() ) { _PrintAttribInfo = FALSE; _ResetMask = (FSN_ATTRIBUTE)0xffffffff; if( _FlagRemoveSystemAttribute.QueryFlag() ) { _ResetMask &= ~FSN_ATTRIBUTE_SYSTEM; } if( _FlagRemoveHiddenAttribute.QueryFlag() ) { _ResetMask &= ~FSN_ATTRIBUTE_HIDDEN; } if( _FlagRemoveReadOnlyAttribute.QueryFlag() ) { _ResetMask &= ~FSN_ATTRIBUTE_READONLY; } if( _FlagRemoveArchiveAttribute.QueryFlag() ) { _ResetMask &= ~FSN_ATTRIBUTE_ARCHIVE; } _MakeMask = 0; if( _FlagAddSystemAttribute.QueryFlag() ) { _MakeMask |= FSN_ATTRIBUTE_SYSTEM; } if( _FlagAddHiddenAttribute.QueryFlag() ) { _MakeMask |= FSN_ATTRIBUTE_HIDDEN; } if( _FlagAddReadOnlyAttribute.QueryFlag() ) { _MakeMask |= FSN_ATTRIBUTE_READONLY; } if( _FlagAddArchiveAttribute.QueryFlag() ) { _MakeMask |= FSN_ATTRIBUTE_ARCHIVE; } } else { _PrintAttribInfo = TRUE; } // // Get filename // if( !_FileNameArgument.IsValueSet() ) { // // User didn't specify file name. Use *.* as default // FullFileNameString = NULL; if( !_FullFileNamePath.Initialize( (LPWSTR)L"*.*", TRUE ) ) { DebugPrint( "_FullFileNamePath.Initialize() failed \n" ); return( FALSE ); } } else { // // Get name specified in the command line // FullFileNameString = _FileNameArgument.GetPath()->GetPathString(); DebugPtrAssert( FullFileNameString ); if( !_FullFileNamePath.Initialize( FullFileNameString, TRUE ) ) { DebugPrint( "_FullFileNamePath.Initialize() failed \n" ); return( FALSE ); } } // // Get prefix and verify that it exists // if( ( DynSubDirectory = _FullFileNamePath.QueryPrefix() ) == NULL ) { DebugPrint( "_FullFileNamePath.QueryPrefix() failed \n" ); return( FALSE ); } if( !DirectoryNamePath.Initialize( DynSubDirectory ) ) { DELETE( DynSubDirectory ); DebugPrint( "DirectoryNamePath.Initialize() failed \n" ); return( FALSE ); } DELETE( DynSubDirectory ); // // Have to test if DirectoryNamePath is a drive, and if it is // add \ to it otherwise it won't be able to find a file that is // in the root directory, if the current directory is not the root. // if( DirectoryNamePath.IsDrive() ) { if( !BackSlashString.Initialize( "\\" ) ) { DebugPrint( "BackSlashString.Initialize() failed \n" ); return( FALSE ); } TempPathString = DirectoryNamePath.GetPathString(); DebugPtrAssert( TempPathString ); if( !TempPathStringDrive.Initialize( TempPathString ) ) { DebugPrint( "TempPathStringDrive.Initialize() failed \n" ); return( FALSE ); } TempPathStringDrive.Strcat( &BackSlashString ); if( !PathDrive.Initialize( &TempPathStringDrive ) ) { DebugPrint( "PathDrive.Initialize() failed \n" ); return( FALSE ); } if( !DirectoryNamePath.Initialize( &PathDrive ) ) { DebugPrint( "DirectoryNamePath.Initialize() failed \n" ); return( FALSE ); } } if( ( _InitialDirectory = SYSTEM::QueryDirectory( &DirectoryNamePath ) ) == NULL ) { InvalidName = DirectoryNamePath.GetPathString(); DebugPtrAssert( InvalidName ); _Message.Set( MSG_ATTRIB_PATH_NOT_FOUND ); _Message.Display( "%W", InvalidName ); return( FALSE ); } // // Initialize filter for directories // if( !_FsnFilterDirectory.Initialize() ) { DELETE( _InitialDirectory ); DebugPrint( "_FsnFilterDirectory.Initialize() failed \n" ); return( FALSE ); } if( !_FsnFilterDirectory.SetFileName( "*.*" ) ) { DELETE( _InitialDirectory ); DebugPrint( "_FsnFilterDirectory.SetFilename() failed \n" ); return( FALSE ); } if( !_FsnFilterDirectory.SetAttributes( FSN_ATTRIBUTE_DIRECTORY ) ) { DELETE( _InitialDirectory ); DebugPrint( "_FsnFilterDirectory.SetAttributes() failed \n" ); return( FALSE ); } // // Get file name and initialize filter for files // if( ( DynSubFileName = _FullFileNamePath.QueryName() ) == NULL ) { if( _FileNameArgument.IsValueSet() ) { InvalidName = _FileNameArgument.GetPath()->GetPathString(); } else { InvalidName = DirectoryNamePath.GetPathString(); } DebugPtrAssert( InvalidName ); _Message.Set( MSG_ATTRIB_FILE_NOT_FOUND ); _Message.Display( "%W", InvalidName ); DELETE( _InitialDirectory ); return( FALSE ); } // // Determine whether attrib should act on a directory. // It will do so only if the user specify a directory name that does // not contain the characters '*' or '?'. // Also, attrib will not act on directories if the switch /S is specified. // if( ( _FlagRecurseDirectories.QueryFlag() ) || ( FullFileNameString == NULL ) || ( FullFileNameString->Strchr( ( WCHAR )'*' ) != INVALID_CHNUM ) || ( FullFileNameString->Strchr( ( WCHAR )'?' ) != INVALID_CHNUM ) ) { ActOnDirectory = FALSE; } else { ActOnDirectory = TRUE; } if( !_FsnFilterFile.Initialize() ) { DELETE( _InitialDirectory ); DELETE( DynSubFileName ); DebugPrint( "FsnFilter.Initialize() failed \n" ); return( FALSE ); } if( !_FsnFilterFile.SetFileName( DynSubFileName ) ) { DELETE( _InitialDirectory ); DELETE( DynSubFileName ); DebugPrint( "FsnFilter.SetFilename() failed \n" ); return( FALSE ); } if( !ActOnDirectory ) { if( !_FsnFilterFile.SetAttributes( 0, 0, FSN_ATTRIBUTE_DIRECTORY ) ) { DELETE( _InitialDirectory ); DELETE( DynSubFileName ); DebugPrint( "FsnFilter.SetAttributes() failed \n" ); return( FALSE ); } } else { if( !_FsnFilterFile.SetAttributes( 0, 0, 0 ) ) { DELETE( _InitialDirectory ); DELETE( DynSubFileName ); DebugPrint( "FsnFilter.SetAttributes() failed \n" ); return( FALSE ); } } DELETE( DynSubFileName ); if( _FlagDisplayHelp.QueryFlag() ) { _Message.Set( MSG_ATTRIB_HELP_MESSAGE ); _Message.Display( " " ); return( FALSE ); } _FoundFile = FALSE; LexArray.DeleteAllMembers(); return( TRUE ); } VOID ATTRIB::Terminate( ) /*++ Routine Description: Deletes objects created during initialization. Arguments: None. Return Value: None. --*/ { if( _InitialDirectory != NULL ) { DELETE( _InitialDirectory ); } } VOID ATTRIB::DisplayFileNotFoundMessage( ) /*++ Routine Description: Displays a message indicating that no file that meets the file filter criteria was found. Arguments: None. Return Value: None. --*/ { PCWSTRING FileName; if( !_FoundFile ) { if( _FileNameArgument.IsValueSet() ) { FileName = _FileNameArgument.GetPath()->GetPathString(); } else { FileName = _FullFileNamePath.GetPathString(); } DebugPtrAssert( FileName ); _Message.Set( MSG_ATTRIB_FILE_NOT_FOUND ); _Message.Display( "%W", FileName ); } } VOID ATTRIB::DisplayFileAttribute ( IN PCFSNODE Fsn ) /*++ Routine Description: Displays a filename and its attributes Arguments: Fsn - A pointer to an FSNODE that contains the information about the file. Return Value: None. --*/ { // PCWC_STRING pcWcString; PCWSTRING pcWcString; WCHAR Buffer[ 12 ]; DSTRING String; DebugPtrAssert( Fsn ); DebugPtrAssert( Fsn->GetPath( )); pcWcString = ( Fsn->GetPath( ))->GetPathString( ); DebugPtrAssert( pcWcString ); swprintf( Buffer, ( LPWSTR )L"%lc %lc%lc%lc ", Fsn->IsArchived() ? ( WCHAR )'A' : ( WCHAR )' ', Fsn->IsSystem() ? ( WCHAR )'S' : ( WCHAR )' ', Fsn->IsHidden() ? ( WCHAR )'H' : ( WCHAR )' ', Fsn->IsReadOnly() ? ( WCHAR )'R' : ( WCHAR )' ' ); if( !String.Initialize( Buffer ) || !String.Strcat( pcWcString ) || !String.Strcat( &_EndOfLineString ) || !_OutStream->WriteString( &String ) ) { DebugPrint( "Unable to display message" ); } } BOOLEAN ATTRIB::ChangeFileAttributes( IN PFSNODE FsnFile ) /*++ Routine Description: Changes the file attributes. The attributes will be changed depending on the argumets specified in the command line, and on the current attributes of the file. The algorithm for changing attributes is presented below: if( ( -s and -h were specified as arguments ) or ( -s and +h were specified as arguments ) or ( +s and -h were specified as arguments ) or ( +s and +h were specified as arguments ) ) { Change file attributes; } else if ( ( -h and +h were not specified as arguments ) and ( file has hidden attribute ) ) { print( "Not resetting hidden file: " ); } else if ( ( -s and +s were not specified as arguments ) and ( file has system attribute ) ) { print( "Not resetting system file: " ); } else { Change file attributes; } Arguments: FsnFile - A pointer to an FSNODE that contains the information about the file. Return Value: BOOLEAN - Returns FALSE if this function fails due to a failure in an API call. --*/ { // BOOLEAN Result; BOOLEAN Change; DWORD Win32Error; // PCWC_STRING pcWcString; PCWSTRING pcWcString; FSN_ATTRIBUTE Attributes; DebugPtrAssert( FsnFile->GetPath( )); pcWcString = ( FsnFile->GetPath( ))->GetPathString( ); DebugPtrAssert( pcWcString ); if( ( ( _FlagAddSystemAttribute.QueryFlag() || _FlagRemoveSystemAttribute.QueryFlag() ) && ( _FlagAddHiddenAttribute.QueryFlag() || _FlagRemoveHiddenAttribute.QueryFlag() ) ) ) { Change = TRUE; } else if( !_FlagAddHiddenAttribute.QueryFlag() && !_FlagRemoveHiddenAttribute.QueryFlag() && FsnFile->IsHidden() ) { // DebugPtrAssert( FsnFile->GetPath( )); // pcWcString = ( FsnFile->GetPath( ))->GetPathString( ); // DebugPtrAssert( pcWcString ); _Message.Set( MSG_ATTRIB_NOT_RESETTING_HIDDEN_FILE ); _Message.Display( "%W", pcWcString ); Change = FALSE; } else if( !_FlagAddSystemAttribute.QueryFlag() && !_FlagRemoveSystemAttribute.QueryFlag() && FsnFile->IsSystem() ) { // DebugPtrAssert( FsnFile->GetPath( )); // pcWcString = ( FsnFile->GetPath( ))->GetPathString( ); // DebugPtrAssert( pcWcString ); _Message.Set( MSG_ATTRIB_NOT_RESETTING_SYS_FILE ); _Message.Display( "%W", pcWcString ); Change = FALSE; } else { Change = TRUE; } // Result = TRUE; if( Change ) { Attributes = FsnFile->QueryAttributes(); if( !FsnFile->SetAttributes( ( Attributes & _ResetMask ) | _MakeMask, &Win32Error ) ) { if( Win32Error == ERROR_ACCESS_DENIED ) { _Message.Set( MSG_ATTRIB_ACCESS_DENIED ); } else { _Message.Set( MSG_ATTRIB_UNABLE_TO_CHANGE_ATTRIBUTE ); } _Message.Display( "%W", pcWcString ); DebugPrint( "Unable to change file attribute \n" ); return( FALSE ); } } return( TRUE ); } BOOLEAN ATTRIB::ExamineFiles( IN PFSN_DIRECTORY Directory ) /*++ Routine Description: Builds an array of files in the specified directory, and tries to change the attributes of each of these files. Does the same thing in all subdirectories, if the "recurse" flag was specified in the command line. Arguments: Directory - Pointer to an FSN_DIRECTORY that describes the directory to be examined Return Value: Boolean: TRUE if Successful. --*/ { PARRAY DirectoryArray; PFSN_DIRECTORY FsnDirectory; PARRAY_ITERATOR DirectoryArrayIterator; PARRAY FileArray; PARRAY_ITERATOR FileArrayIterator; PFSNODE FsnFile; DebugPtrAssert( Directory ); // // If /S was specified as argument in the command line, builds // an array of PFSN_DIRECTORY of all sub-directories in the current // directory and examines the files in each sub-directory // if( _FlagRecurseDirectories.QueryFlag() ) { if( ( DirectoryArray = Directory->QueryFsnodeArray( &_FsnFilterDirectory ) ) == NULL ) { DebugPrint( "Directory->QueryFsnodeArray( &_FsnFilterDirectory ) failed \n" ); return( FALSE ); } if( ( DirectoryArrayIterator = ( PARRAY_ITERATOR )( DirectoryArray->QueryIterator() ) ) == NULL ) { DebugPrint( "DirectoryArray->QueryIterator() failed \n" ); return( FALSE ); } while( ( FsnDirectory = ( PFSN_DIRECTORY )( DirectoryArrayIterator->GetNext( ) ) ) != NULL ) { ExamineFiles( FsnDirectory ); DELETE( FsnDirectory ); } DELETE( DirectoryArrayIterator ); DELETE( DirectoryArray ); } // // Builds an array of FSNODEs of the files tha meet the 'filter' // criteria, and change or display the attributes of these files // if( ( FileArray = Directory->QueryFsnodeArray( &_FsnFilterFile ) ) == NULL ) { DebugPrint( "Directory->QueryFsnodeArray( &_FsnFilterFile ) failed \n" ); return( FALSE ); } if( ( FileArrayIterator = ( PARRAY_ITERATOR )( FileArray->QueryIterator() ) ) == NULL ) { DebugPrint( "FileArray->QueryIterator() failed \n" ); return( FALSE ); } while( ( FsnFile = ( PFSNODE )( FileArrayIterator->GetNext( ) ) ) != NULL ) { if( _PrintAttribInfo ) { DisplayFileAttribute( FsnFile ); } else { ChangeFileAttributes( FsnFile ); } _FoundFile = TRUE; DELETE( FsnFile ); } DELETE( FileArrayIterator ); DELETE( FileArray ); return( TRUE ); } ULONG _CRTAPI1 main() { DEFINE_CLASS_DESCRIPTOR( ATTRIB ); { ATTRIB Attrib; perrstk = NEW ERRSTACK; if( Attrib.Initialize() ) { Attrib.ExamineFiles( Attrib.GetInitialDirectory() ); Attrib.DisplayFileNotFoundMessage(); } Attrib.Terminate(); DELETE( perrstk ); } return( 0 ); }