Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

569 lines
11 KiB

/*++
Copyright (c) 1990 Microsoft Corporation
Module Name:
path.hxx
Abstract:
The PATH class provides an interface to the complete
Win32 name space. Complete means that it will correctly
handle long, drive or UNC based, blank embedded, mixed case
names. It should eliminate the need for everyone to code
statements such as "is the second char a ':'" or "search for
the first '\' from the end of the name". That is, access to and
manipulation of path and file names should be performed soley
through the PATH member functions. This will eliminate
the recoding of standard name manipulation code and ensure
complete support of Win32 functionality, such as codepage and
DBCS support.
Author:
Steve Rowe 13-Dec-90
Environment:
ULIB, user
Notes:
To clarify terminology used here, the following describes a
canonicalized path (in butchered BNF/reg exp):
{Canon} ::= {Prefix}"\"{Name}
{Prefix} ::= {Device}{Dirs}
{Dirs} ::= {"\"{Component}}*
{Device} ::= {Drive}|{Machine}
{Drive} ::= {Letter}":"
{Machine} ::= "\\"{Char}+
{Letter} ::= valid drive letter [a-zA-Z]
{Char} ::= valid filename/directory char [~:\]
{Component} ::= {Char}+
{Name} ::= {Base - excluding "."} | { {Base}"."{Ext} }
{Base} ::= {Char}+
{Ext} ::= {Char - excluding "."}+
Legend:
-------
{x}* - 0 or more x
{x}+ - 1 or more x
"x" - just x (not the quotes)
{x}|{y} - x or y (not both or none)
Examples:
---------
# Canon
--- -----
(1) x:\abc\def.ghi\jkl.mnop
(2) \\x\abc\def.ghi\jkl.mnop
(3) c:\config.sys
(1) (2) (3)
Prefix x:\abc\def.ghi \\x\abc\def.ghi c:
Device x: \\x c:
Dirs \abc \abc\def.ghi
Name jkl.mnop jkl.mnop config.sys
Base jkl jkl config
Ext mnop mnop sys
Component numbers are 0-based.
--*/
#if !defined( _PATH_)
#define _PATH_
#include "wstring.hxx"
#include "array.hxx"
//
// Forward references & declarations
//
DECLARE_CLASS( PATH );
//
// PATHSTATE maintains the number of characters within each component that
// makes up a PATH
//
struct _PATHSTATE {
//
// Prefix
//
CHNUM PrefixLen;
CHNUM DeviceLen;
CHNUM DirsLen;
CHNUM SeparatorLen;
//
// Name
//
CHNUM NameLen;
CHNUM BaseLen;
CHNUM ExtLen;
};
DEFINE_TYPE( struct _PATHSTATE, PATHSTATE );
typedef enum PATH_ANALYZE_CODE {
PATH_OK,
PATH_COULD_BE_FLOPPY,
PATH_OUT_OF_MEMORY,
PATH_INVALID_DRIVE_SPEC,
PATH_NO_MOUNT_POINT_FOR_VOLUME_NAME_PATH
};
class PATH : public OBJECT {
public:
ULIB_EXPORT
DECLARE_CONSTRUCTOR( PATH );
DECLARE_CAST_MEMBER_FUNCTION( PATH );
NONVIRTUAL
ULIB_EXPORT
BOOLEAN
Initialize (
IN PCWSTR InitialPath,
IN BOOLEAN Canonicalize DEFAULT FALSE
);
NONVIRTUAL
ULIB_EXPORT
BOOLEAN
Initialize (
IN PCWSTRING InitialPath,
IN BOOLEAN Canonicalize DEFAULT FALSE
);
NONVIRTUAL
ULIB_EXPORT
BOOLEAN
Initialize (
IN PCPATH InitialPath,
IN BOOLEAN Canonicalize DEFAULT FALSE
);
VIRTUAL
ULIB_EXPORT
~PATH (
);
NONVIRTUAL
ULIB_EXPORT
BOOLEAN
AppendBase (
IN PCWSTRING Base,
IN BOOLEAN Absolute DEFAULT FALSE
);
NONVIRTUAL
ULIB_EXPORT
BOOLEAN
EndsWithDelimiter (
) CONST;
NONVIRTUAL
PCWSTRING
GetPathString (
) CONST;
NONVIRTUAL
ULIB_EXPORT
BOOLEAN
HasWildCard (
) CONST;
NONVIRTUAL
ULIB_EXPORT
BOOLEAN
IsDrive (
) CONST;
NONVIRTUAL
BOOLEAN
IsRoot (
) CONST;
NONVIRTUAL
ULIB_EXPORT
BOOLEAN
ModifyName (
IN PCWSTRING Pattern
);
NONVIRTUAL
PWSTRING
QueryBase (
);
NONVIRTUAL
ULIB_EXPORT
PARRAY
QueryComponentArray (
OUT PARRAY Array DEFAULT NULL
) CONST;
NONVIRTUAL
PWSTRING
QueryDevice (
);
NONVIRTUAL
PWSTRING
QueryDirs (
);
NONVIRTUAL
PWSTRING
QueryDirsAndName (
);
NONVIRTUAL
PWSTRING
QueryExt (
);
NONVIRTUAL
ULIB_EXPORT
PPATH
QueryFullPath (
) CONST;
NONVIRTUAL
ULIB_EXPORT
PWSTRING
QueryFullPathString (
) CONST;
NONVIRTUAL
PWSTRING
QueryName (
) CONST;
NONVIRTUAL
ULIB_EXPORT
PPATH
QueryPath (
) CONST;
NONVIRTUAL
PWSTRING
QueryPrefix (
);
NONVIRTUAL
ULIB_EXPORT
PWSTRING
QueryRoot (
);
NONVIRTUAL
ULIB_EXPORT
PPATH
QueryWCExpansion(
IN PPATH BasePath
);
NONVIRTUAL
BOOLEAN
SetBase (
IN PCWSTRING NewBase
);
ULIB_EXPORT
BOOLEAN
SetDevice (
IN PCWSTRING NewDevice
);
NONVIRTUAL
BOOLEAN
SetExt (
IN PCWSTRING NewExt
);
NONVIRTUAL
ULIB_EXPORT
BOOLEAN
SetName (
IN PCWSTRING NewName
);
NONVIRTUAL
BOOLEAN
SetPrefix (
IN PCWSTRING NewPrefix
);
NONVIRTUAL
ULIB_EXPORT
BOOLEAN
TruncateBase (
);
NONVIRTUAL
ULIB_EXPORT
VOID
TruncateNameAtColon (
);
NONVIRTUAL
ULIB_EXPORT
VOID
TruncateDelimiter (
);
ULIB_EXPORT
BOOLEAN
AppendDelimiter (
);
ULIB_EXPORT
BOOLEAN
AppendString (
IN PCWSTRING String
);
NONVIRTUAL
ULIB_EXPORT
BOOLEAN
IsGuidVolName(
);
NONVIRTUAL
ULIB_EXPORT
PPATH
QueryMountPointPath(
);
NONVIRTUAL
ULIB_EXPORT
PWSTRING
QueryGuidString(
OUT PWSTRING LongestMountPointPath,
OUT PBOOLEAN IsLongestMountPointADriveLetter,
OUT PWSTRING DrivePath
);
NONVIRTUAL
ULIB_EXPORT
PATH_ANALYZE_CODE
AnalyzePath(
OUT PWSTRING VolumeName,
OUT PPATH VolumePath,
OUT PWSTRING RemainingPath
);
private:
NONVIRTUAL
VOID
Construct (
);
NONVIRTUAL
BOOLEAN
ExpandWildCards(
IN OUT PWSTRING pStr1,
IN OUT PWSTRING pStr2
);
BOOLEAN
Initialize (
);
NONVIRTUAL
CHNUM
QueryBaseStart (
) CONST;
CHNUM
QueryDeviceLen(
IN PWSTRING pString
) CONST;
NONVIRTUAL
CHNUM
QueryDeviceStart (
) CONST;
NONVIRTUAL
CHNUM
QueryExtStart (
) CONST;
NONVIRTUAL
CHNUM
QueryNameStart (
) CONST;
NONVIRTUAL
CHNUM
QueryPrefixStart (
) CONST;
NONVIRTUAL
VOID
SetPathState (
);
STATIC
BOOLEAN
BuildMountPoint(
IN PWSTR Name,
IN PCWSTR GuidNameToMatch,
OUT PWSTR MountPointPath
);
#if DBG==1
ULONG _Signature;
BOOLEAN _Initialized;
#endif
//
// path data
//
WCHAR _PathBuffer[MAX_PATH];
FSTRING _PathString;
PATHSTATE _PathState;
};
INLINE
CHNUM
PATH::QueryPrefixStart (
) CONST
{
return( 0 );
}
INLINE
CHNUM
PATH::QueryNameStart (
) CONST
{
//
// Increment past the '\'
//
return( QueryPrefixStart() + _PathState.PrefixLen + _PathState.SeparatorLen );
}
INLINE
CHNUM
PATH::QueryBaseStart (
) CONST
{
return( QueryNameStart() );
}
INLINE
CHNUM
PATH::QueryDeviceStart (
) CONST
{
return( 0 );
}
INLINE
CHNUM
PATH::QueryExtStart (
) CONST
{
return( QueryNameStart() + _PathState.BaseLen + 1 );
}
INLINE
PCWSTRING
PATH::GetPathString (
) CONST
{
return( &_PathString );
}
INLINE
PWSTRING
PATH::QueryBase (
)
{
return( _PathState.BaseLen ? _PathString.QueryString( QueryBaseStart(), _PathState.BaseLen ) : NULL );
}
INLINE
PWSTRING
PATH::QueryDirs (
)
{
return( _PathState.DirsLen ? _PathString.QueryString( QueryDeviceStart() + _PathState.DeviceLen, _PathState.DirsLen ) : NULL );
}
INLINE
PWSTRING
PATH::QueryDirsAndName (
)
{
return( ( _PathState.DirsLen || _PathState.NameLen ) ? _PathString.QueryString( QueryDeviceStart() + _PathState.DeviceLen ) : NULL );
}
INLINE
PWSTRING
PATH::QueryDevice (
)
{
return( _PathState.DeviceLen ? _PathString.QueryString( QueryDeviceStart(), _PathState.DeviceLen ) : NULL );
}
INLINE
PWSTRING
PATH::QueryExt (
)
{
return( _PathState.ExtLen ? _PathString.QueryString( QueryExtStart(), _PathState.ExtLen ) : NULL );
}
INLINE
PWSTRING
PATH::QueryName (
) CONST
{
return( _PathState.NameLen ? _PathString.QueryString( QueryNameStart(), _PathState.NameLen ) : NULL );
}
INLINE
PWSTRING
PATH::QueryPrefix (
)
{
return( _PathState.PrefixLen ? _PathString.QueryString( QueryPrefixStart(), _PathState.PrefixLen ) : NULL );
}
#endif // _PATH_