/* * linkinfo.c - LinkInfo ADT module. */ /* Headers **********/ #include "project.h" #pragma hdrstop #include "volumeid.h" #include "cnrlink.h" #include // for ALIGNMENT_MACHINE /* Macros *********/ /* macros for accessing ILINKINFO data */ #define ILI_Volume_ID_Ptr(pili) \ ((PVOLUMEID)(((PBYTE)(pili)) + (pili)->ucbVolumeIDOffset)) #define ILI_Local_Base_Path_PtrA(pili) \ ((LPSTR)(((PBYTE)(pili)) + (pili)->ucbLocalBasePathOffset)) #define ILI_CNR_Link_Ptr(pili) \ ((PCNRLINK)(((PBYTE)(pili)) + (pili)->ucbCNRLinkOffset)) #define ILI_Common_Path_Suffix_PtrA(pili) \ ((LPSTR)(((PBYTE)(pili)) + (pili)->ucbCommonPathSuffixOffset)) #define ILI_Local_Base_Path_PtrW(pili) \ ((LPWSTR)(((PBYTE)(pili)) + (pili)->ucbLocalBasePathOffsetW)) #define ILI_Common_Path_Suffix_PtrW(pili) \ ((LPWSTR)(((PBYTE)(pili)) + (pili)->ucbCommonPathSuffixOffsetW)) #ifdef UNICODE #define ILI_Local_Base_Path_Ptr(pili) ILI_Local_Base_Path_PtrW(pili) #define ILI_Common_Path_Suffix_Ptr(pili) ILI_Common_Path_Suffix_PtrW(pili) #else #define ILI_Local_Base_Path_Ptr(pili) ILI_Local_Base_Path_PtrA(pili) #define ILI_Common_Path_Suffix_Ptr(pili) ILI_Common_Path_Suffix_PtrA(pili) #endif /* Types ********/ /****************************************************************************** @doc LINKINFOAPI @struct LINKINFO | External definition of LinkInfo structure. @field UINT | ucbSize | The size of the LINKINFO structure in bytes, including the ucbSize field. An ILINKINFO structure consists of a header described as below, followed by variable-length data that is opaque to the caller. ******************************************************************************/ /* @doc INTERNAL @enum ILINKINFOFLAGS | Internal LinkInfo structure flags. */ typedef enum _ilinkinfoflags { /* @emem ILI_FL_LOCAL_INFO_VALID | If set, volume ID and local path are valid. If clear, volume ID and local path are not valid. */ ILI_FL_LOCAL_INFO_VALID = 0x0001, /* @emem ILI_FL_REMOTE_INFO_VALID | If set, CNRLink and path suffix are valid. If clear, CNRLink and path suffix not valid. */ ILI_FL_REMOTE_INFO_VALID = 0x0002, /* @emem ALL_ILINKINFO_FLAGS | All internal LinkInfo structure flags. */ ALL_ILINKINFO_FLAGS = (ILI_FL_LOCAL_INFO_VALID |\ ILI_FL_REMOTE_INFO_VALID) } ILINKINFOFLAGS; /* @doc INTERNAL @struct ILINKINFO | Internal definition of relocatable, extensible, internal LinkInfo structure. An ILINKINFO structure may contain an structure and an structure. An ILINKINFO structure consists of a header described as below, followed by variable-length data. */ typedef struct _ilinkinfoA { /* @field LINKINFO | li | External sub-structure. */ LINKINFO li; /* @field UINT | ucbHeaderSize | Size of the ILINKINFO header structure in bytes. */ UINT ucbHeaderSize; /* @field DWORD | dwFlags | A bit mask of flags from the enumeration. */ DWORD dwFlags; /* @field UINT | ucbVolumeIDOffset | Offset in bytes of sub-structure from base of structure. */ UINT ucbVolumeIDOffset; /* @field UINT | ucbLocalBasePathOffset | Offset in bytes of local base path string from base of structure. The local base path is a valid file system path. The local base path string + the common path suffix string form the local path string, which is a valid file system path. The local base path string refers to the same resource as the CNRLink's CNR name string. Example local base path string: "c:\\work". E.g., if local path "c:\\work" is shared as "\\\\fredbird\\work", an ILinkInfo structure would break local path "c:\\work\\footwear\\sneakers.doc" up into local base path "c:\\work", CNRLink CNR name "\\\\fredbird\\work", and common path suffix "footwear\\sneakers.doc". */ UINT ucbLocalBasePathOffset; /* @field UINT | ucbCNRLinkOffset | Offset in bytes of sub-structure from base of structure. The file system name of the CNRLink's CNR name + the common path suffix string form the remote path string, which is a valid file system path. The CNRLink's CNR name string refers to the same resource as the local base path string. */ UINT ucbCNRLinkOffset; /* @field UINT | ucbCommonPathSuffixOffset | Offset in bytes of common path suffix string from base of structure. Example common path suffix string: "footwear\\sneakers.doc". */ UINT ucbCommonPathSuffixOffset; } ILINKINFOA; DECLARE_STANDARD_TYPES(ILINKINFOA); #ifdef UNICODE typedef struct _ilinkinfoW { /* @field LINKINFO | li | External sub-structure. */ LINKINFO li; /* @field UINT | ucbHeaderSize | Size of the ILINKINFO header structure in bytes. */ UINT ucbHeaderSize; /* @field DWORD | dwFlags | A bit mask of flags from the enumeration. */ DWORD dwFlags; /* @field UINT | ucbVolumeIDOffset | Offset in bytes of sub-structure from base of structure. */ UINT ucbVolumeIDOffset; /* @field UINT | ucbLocalBasePathOffset | Offset in bytes of local base path string from base of structure. The local base path is a valid file system path. The local base path string + the common path suffix string form the local path string, which is a valid file system path. The local base path string refers to the same resource as the CNRLink's CNR name string. Example local base path string: "c:\\work". E.g., if local path "c:\\work" is shared as "\\\\fredbird\\work", an ILinkInfo structure would break local path "c:\\work\\footwear\\sneakers.doc" up into local base path "c:\\work", CNRLink CNR name "\\\\fredbird\\work", and common path suffix "footwear\\sneakers.doc". */ UINT ucbLocalBasePathOffset; /* @field UINT | ucbCNRLinkOffset | Offset in bytes of sub-structure from base of structure. The file system name of the CNRLink's CNR name + the common path suffix string form the remote path string, which is a valid file system path. The CNRLink's CNR name string refers to the same resource as the local base path string. */ UINT ucbCNRLinkOffset; /* @field UINT | ucbCommonPathSuffixOffset | Offset in bytes of common path suffix string from base of structure. Example common path suffix string: "footwear\\sneakers.doc". */ UINT ucbCommonPathSuffixOffset; /* These fields duplicate the above ones except that they are for the unicode versions of the strings. */ UINT ucbLocalBasePathOffsetW; UINT ucbCommonPathSuffixOffsetW; } ILINKINFOW; DECLARE_STANDARD_TYPES(ILINKINFOW); #endif #ifdef UNICODE #define ILINKINFO ILINKINFOW #define PILINKINFO PILINKINFOW #define CILINKINFO CILINKINFOW #define PCILINKINFO PCILINKINFOW #else #define ILINKINFO ILINKINFOA #define PILINKINFO PILINKINFOA #define CILINKINFO CILINKINFOA #define PCILINKINFO PCILINKINFOA #endif /***************************** Private Functions *****************************/ /* Module Prototypes ********************/ PRIVATE_CODE BOOL CreateILinkInfo(LPCTSTR, PILINKINFO *); PRIVATE_CODE BOOL CreateLocalILinkInfo(LPCTSTR, PILINKINFO *); PRIVATE_CODE BOOL CreateRemoteILinkInfo(LPCTSTR, LPCTSTR, LPCTSTR, PILINKINFO *); PRIVATE_CODE BOOL UnifyILinkInfo(PCVOLUMEID, UINT, LPCTSTR, PCCNRLINK, UINT, LPCTSTR, PILINKINFO *); PRIVATE_CODE void DestroyILinkInfo(PILINKINFO); PRIVATE_CODE BOOL UpdateILinkInfo(PCILINKINFO, LPCTSTR, PDWORD, PILINKINFO *); PRIVATE_CODE BOOL UseNewILinkInfo(PCILINKINFO, PCILINKINFO); PRIVATE_CODE BOOL ResolveLocalILinkInfo(PCILINKINFO, LPTSTR, int, DWORD); PRIVATE_CODE BOOL ResolveRemoteILinkInfo(PCILINKINFO, LPTSTR, DWORD, HWND, PDWORD); PRIVATE_CODE BOOL ResolveILinkInfo(PCILINKINFO, LPTSTR, int, DWORD, HWND, PDWORD); PRIVATE_CODE BOOL ResolveLocalPathFromServer(PCILINKINFO, LPTSTR, PDWORD); PRIVATE_CODE void GetLocalPathFromILinkInfo(PCILINKINFO, LPTSTR, int); PRIVATE_CODE void GetRemotePathFromILinkInfo(PCILINKINFO, LPTSTR, int); PRIVATE_CODE COMPARISONRESULT CompareILinkInfoReferents(PCILINKINFO, PCILINKINFO); PRIVATE_CODE COMPARISONRESULT CompareILinkInfoLocalData(PCILINKINFO, PCILINKINFO); PRIVATE_CODE COMPARISONRESULT CompareLocalPaths(PCILINKINFO, PCILINKINFO); PRIVATE_CODE COMPARISONRESULT CompareILinkInfoRemoteData(PCILINKINFO, PCILINKINFO); PRIVATE_CODE COMPARISONRESULT CompareILinkInfoVolumes(PCILINKINFO, PCILINKINFO); PRIVATE_CODE BOOL CheckCombinedPathLen(LPCTSTR, LPCTSTR); PRIVATE_CODE BOOL GetILinkInfoData(PCILINKINFO, LINKINFODATATYPE, PCVOID *); PRIVATE_CODE BOOL DisconnectILinkInfo(PCILINKINFO); #if defined(DEBUG) || defined(EXPV) PRIVATE_CODE BOOL IsValidLINKINFODATATYPE(LINKINFODATATYPE); #endif #if defined(DEBUG) || defined(VSTF) PRIVATE_CODE BOOL CheckILIFlags(PCILINKINFO); PRIVATE_CODE BOOL CheckILICommonPathSuffix(PCILINKINFO); PRIVATE_CODE BOOL CheckILILocalInfo(PCILINKINFO); PRIVATE_CODE BOOL CheckILIRemoteInfo(PCILINKINFO); PRIVATE_CODE BOOL IsValidPCLINKINFO(PCLINKINFO); PRIVATE_CODE BOOL IsValidPCILINKINFO(PCILINKINFO); #endif #ifdef DEBUG PRIVATE_CODE void DumpILinkInfo(PCILINKINFO); #endif /* ** CreateILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL CreateILinkInfo(LPCTSTR pcszPath, PILINKINFO *ppili) { BOOL bResult = FALSE; TCHAR rgchCanonicalPath[MAX_PATH_LEN]; DWORD dwCanonicalPathFlags; TCHAR rgchCNRName[MAX_PATH_LEN]; LPTSTR pszRootPathSuffix; ASSERT(IS_VALID_STRING_PTR(pcszPath, CSTR)); ASSERT(IS_VALID_WRITE_PTR(ppili, PILINKINFO)); if (GetCanonicalPathInfo(pcszPath, rgchCanonicalPath, &dwCanonicalPathFlags, rgchCNRName, &pszRootPathSuffix)) { if (IS_FLAG_SET(dwCanonicalPathFlags, GCPI_OFL_REMOTE)) bResult = CreateRemoteILinkInfo(rgchCanonicalPath, rgchCNRName, pszRootPathSuffix, ppili); else bResult = CreateLocalILinkInfo(rgchCanonicalPath, ppili); } return(bResult); } /* ** CreateLocalILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL CreateLocalILinkInfo(LPCTSTR pcszLocalPath, PILINKINFO *ppili) { BOOL bResult; PVOLUMEID pvolid; UINT ucbVolumeIDLen; ASSERT(IsLocalDrivePath(pcszLocalPath)); ASSERT(IS_VALID_WRITE_PTR(ppili, PILINKINFO)); bResult = CreateVolumeID(pcszLocalPath, &pvolid, &ucbVolumeIDLen); if (bResult) { PCNRLINK pcnrl; UINT ucbCNRLinkLen; TCHAR rgchLocalBasePath[MAX_PATH_LEN]; LPCTSTR pcszCommonPathSuffix; bResult = CreateLocalCNRLink(pcszLocalPath, &pcnrl, &ucbCNRLinkLen, rgchLocalBasePath, ARRAYSIZE(rgchLocalBasePath), &pcszCommonPathSuffix); if (bResult) { /* Wrap them up. */ bResult = UnifyILinkInfo(pvolid, ucbVolumeIDLen, rgchLocalBasePath, pcnrl, ucbCNRLinkLen, pcszCommonPathSuffix, ppili); if (ucbCNRLinkLen > 0) DestroyCNRLink(pcnrl); } if (ucbVolumeIDLen > 0) DestroyVolumeID(pvolid); } ASSERT(! bResult || IS_VALID_STRUCT_PTR(*ppili, CILINKINFO)); return(bResult); } /* ** CreateRemoteILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL CreateRemoteILinkInfo(LPCTSTR pcszRemotePath, LPCTSTR pcszCNRName, LPCTSTR pcszRootPathSuffix, PILINKINFO *ppili) { BOOL bResult; PCNRLINK pcnrl; UINT ucbCNRLinkLen; ASSERT(IsCanonicalPath(pcszRemotePath)); ASSERT(IsValidCNRName(pcszCNRName)); ASSERT(IS_VALID_STRING_PTR(pcszRootPathSuffix, CSTR)); ASSERT(IS_VALID_WRITE_PTR(ppili, PILINKINFO)); bResult = CreateRemoteCNRLink(pcszRemotePath, pcszCNRName, &pcnrl, &ucbCNRLinkLen); if (bResult) { /* Wrap it up. */ bResult = UnifyILinkInfo(NULL, 0, EMPTY_STRING, pcnrl, ucbCNRLinkLen, pcszRootPathSuffix, ppili); if (EVAL(ucbCNRLinkLen > 0)) DestroyCNRLink(pcnrl); } ASSERT(! bResult || IS_VALID_STRUCT_PTR(*ppili, CILINKINFO)); return(bResult); } /* ** UnifyILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL UnifyILinkInfo(PCVOLUMEID pcvolid, UINT ucbVolumeIDLen, LPCTSTR pcszLocalBasePath, PCCNRLINK pccnrl, UINT ucbCNRLinkLen, LPCTSTR pcszCommonPathSuffix, PILINKINFO *ppili) { BOOL bResult; UINT ucbILinkInfoLen; UINT ucbDataOffset; UINT cbAnsiLocalBasePath; UINT cbAnsiCommonPathSuffix; #ifdef UNICODE BOOL bUnicode; UINT cchChars; CHAR szAnsiLocalBasePath[MAX_PATH*2]; CHAR szAnsiCommonPathSuffix[MAX_PATH*2]; UINT cbWideLocalBasePath; UINT cbWideCommonPathSuffix; UINT cbChars; #endif ASSERT(! ucbVolumeIDLen || (IS_VALID_STRUCT_PTR(pcvolid, CVOLUMEID) && IsDrivePath(pcszLocalBasePath))); ASSERT(! ucbCNRLinkLen || IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK)); ASSERT(IS_VALID_STRING_PTR(pcszCommonPathSuffix, CSTR)); ASSERT(IS_VALID_WRITE_PTR(ppili, PILINKINFO)); #ifdef UNICODE bUnicode = FALSE; /* ** Convert the common-path string from UNICODE->ansi and back again ** to determine if the string contains any non-ansi characters. If no ** characters are lost in the conversion then the string contains only ** ansi chars. */ cbAnsiCommonPathSuffix = WideCharToMultiByte(CP_ACP, 0, pcszCommonPathSuffix, -1, szAnsiCommonPathSuffix, ARRAYSIZE(szAnsiCommonPathSuffix), 0, 0); if ( cbAnsiCommonPathSuffix == 0 ) { bUnicode = FALSE; } else { WCHAR szWideCommonPathSuffix[MAX_PATH]; cbChars = MultiByteToWideChar(CP_ACP, 0, szAnsiCommonPathSuffix, -1, szWideCommonPathSuffix, ARRAYSIZE(szWideCommonPathSuffix)); if ( cbChars == 0 || lstrcmp(pcszCommonPathSuffix,szWideCommonPathSuffix) != 0 ) { bUnicode = TRUE; } } if (ucbVolumeIDLen > 0) { /* ** Convert the localbase-path string from UNICODE->ansi and back again ** to determine if the string contains any non-ansi characters. If no ** characters are lost in the conversion then the string contains only ** ansi chars. */ cbAnsiLocalBasePath = WideCharToMultiByte(CP_ACP, 0, pcszLocalBasePath, -1, szAnsiLocalBasePath, ARRAYSIZE(szAnsiLocalBasePath), 0, 0); if ( cbAnsiLocalBasePath == 0 ) { bUnicode = FALSE; } else { WCHAR szWideLocalBasePath[MAX_PATH]; cchChars = MultiByteToWideChar(CP_ACP, 0, szAnsiLocalBasePath, -1, szWideLocalBasePath, ARRAYSIZE(szWideLocalBasePath)); if ( cchChars == 0 || lstrcmp(pcszLocalBasePath,szWideLocalBasePath) != 0 ) { bUnicode = TRUE; } } } else { cbAnsiLocalBasePath = 0; } if ( bUnicode ) { ucbDataOffset = SIZEOF(ILINKINFOW); /* (+ 1) for null terminator. */ cbWideCommonPathSuffix = (lstrlen(pcszCommonPathSuffix) + 1) * sizeof(TCHAR); if (ucbVolumeIDLen > 0) cbWideLocalBasePath = (lstrlen(pcszLocalBasePath) + 1) * sizeof(TCHAR); else cbWideLocalBasePath = 0; } else { ucbDataOffset = SIZEOF(ILINKINFOA); cbWideCommonPathSuffix = 0; cbWideLocalBasePath = 0; } ucbILinkInfoLen = ucbDataOffset + ucbVolumeIDLen + cbAnsiLocalBasePath; if ( bUnicode && ucbVolumeIDLen > 0 ) { ucbILinkInfoLen = ALIGN_WORD_CNT(ucbILinkInfoLen); ucbILinkInfoLen += cbWideLocalBasePath; } if ( ucbCNRLinkLen > 0 ) { ucbILinkInfoLen = ALIGN_DWORD_CNT(ucbILinkInfoLen); ucbILinkInfoLen += ucbCNRLinkLen; } ucbILinkInfoLen += cbAnsiCommonPathSuffix; if ( bUnicode ) { ucbILinkInfoLen = ALIGN_WORD_CNT(ucbILinkInfoLen); ucbILinkInfoLen += cbWideCommonPathSuffix; } #else /* Calculate total length. */ /* Assume we don't overflow ucbILinkInfoLen here. */ /* * Base structure size plus common path suffix length. (+ 1) for null * terminator. */ cbAnsiCommonPathSuffix = lstrlen(pcszCommonPathSuffix) + 1; ucbILinkInfoLen = SIZEOF(**ppili) + cbAnsiCommonPathSuffix; /* Plus size of local information. */ if (ucbVolumeIDLen > 0) { /* (+ 1) for null terminator. */ cbAnsiLocalBasePath = lstrlen(pcszLocalBasePath) + 1; ucbILinkInfoLen += ucbVolumeIDLen + cbAnsiLocalBasePath; } /* Plus size of remote information. */ if (ucbCNRLinkLen > 0) /* (+ 1) for null terminator. */ ucbILinkInfoLen += ucbCNRLinkLen; ucbDataOffset = SIZEOF(**ppili); #endif /* Try to allocate a container. */ bResult = AllocateMemory(ucbILinkInfoLen, ppili); if (bResult) { (*ppili)->li.ucbSize = ucbILinkInfoLen; (*ppili)->ucbHeaderSize = ucbDataOffset; (*ppili)->dwFlags = 0; /* Do we have local information? */ if (ucbVolumeIDLen > 0) { /* Yes. Add it to the structure. */ ASSERT(IS_VALID_STRUCT_PTR(pcvolid, CVOLUMEID)); ASSERT(IsDrivePath(pcszLocalBasePath)); /* Append local volume ID. */ (*ppili)->ucbVolumeIDOffset = ucbDataOffset; CopyMemory(ILI_Volume_ID_Ptr(*ppili), pcvolid, ucbVolumeIDLen); ucbDataOffset += ucbVolumeIDLen; /* Append local path. */ // lstrcpy: Enough memory is allocated above so unnecessary to do a // bounded copy (*ppili)->ucbLocalBasePathOffset = ucbDataOffset; #ifdef UNICODE lstrcpyA(ILI_Local_Base_Path_PtrA(*ppili), szAnsiLocalBasePath); ucbDataOffset += cbAnsiLocalBasePath; if ( bUnicode ) { ucbDataOffset = ALIGN_WORD_CNT(ucbDataOffset); (*ppili)->ucbLocalBasePathOffsetW = ucbDataOffset; lstrcpy(ILI_Local_Base_Path_PtrW(*ppili), pcszLocalBasePath); ucbDataOffset += cbWideLocalBasePath; } #else lstrcpy(ILI_Local_Base_Path_Ptr(*ppili), pcszLocalBasePath); ucbDataOffset += cbAnsiLocalBasePath; #endif SET_FLAG((*ppili)->dwFlags, ILI_FL_LOCAL_INFO_VALID); } /* Do we have remote information? */ if (ucbCNRLinkLen > 0) { ucbDataOffset = ALIGN_DWORD_CNT(ucbDataOffset); /* Yes. Add it to the structure. */ ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK)); /* Append CNR link. */ (*ppili)->ucbCNRLinkOffset = ucbDataOffset; CopyMemory(ILI_CNR_Link_Ptr(*ppili), pccnrl, ucbCNRLinkLen); ucbDataOffset += ucbCNRLinkLen; SET_FLAG((*ppili)->dwFlags, ILI_FL_REMOTE_INFO_VALID); } /* Append common path suffix. */ ASSERT(IS_VALID_STRING_PTR(pcszCommonPathSuffix, CSTR)); (*ppili)->ucbCommonPathSuffixOffset = ucbDataOffset; #ifdef UNICODE lstrcpyA(ILI_Common_Path_Suffix_PtrA(*ppili), szAnsiCommonPathSuffix); ucbDataOffset += cbAnsiCommonPathSuffix; if ( bUnicode ) { ucbDataOffset = ALIGN_WORD_CNT(ucbDataOffset); (*ppili)->ucbCommonPathSuffixOffsetW = ucbDataOffset; lstrcpy(ILI_Common_Path_Suffix_Ptr(*ppili), pcszCommonPathSuffix); ucbDataOffset += cbWideCommonPathSuffix; } #else /* UNICODE */ lstrcpy(ILI_Common_Path_Suffix_Ptr(*ppili), pcszCommonPathSuffix); #ifdef DEBUG /* ** NOTE: This same increment was present above in the UNICODE section ** enclosed in an #ifdef DEBUG block. ** It was causing the assertion below (ucbDataOffset == ucbILinkInfoLen) ** to fail. I have left stmt instance in the ansi build untouched. ** If the assertion fails in the ansi build you should ** try removing this next statement. [brianau - 4/15/99] */ ucbDataOffset += cbAnsiCommonPathSuffix; #endif #endif /* Do all the calculated lengths match? */ // ASSERT(ucbDataOffset == (*ppili)->li.ucbSize); ASSERT(ucbDataOffset == ucbILinkInfoLen); } ASSERT(! bResult || IS_VALID_STRUCT_PTR(*ppili, CILINKINFO)); return(bResult); } /* ** DestroyILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE void DestroyILinkInfo(PILINKINFO pili) { ASSERT(IS_VALID_STRUCT_PTR(pili, CILINKINFO)); FreeMemory(pili); return; } /* ** UpdateILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none ** ** An ILinkInfo structure is updated in the following cases: ** ** local information: ** ** 1) the local path has changed ** 2) remote information is available for the local path ** ** remote information: ** ** 3) the remote information is local to this machine, and local information ** is available for the remote path */ PRIVATE_CODE BOOL UpdateILinkInfo(PCILINKINFO pcili, LPCTSTR pcszResolvedPath, PDWORD pdwOutFlags, PILINKINFO *ppiliUpdated) { BOOL bResult; ASSERT(IS_VALID_STRUCT_PTR(pcili, CILINKINFO)); ASSERT(IS_VALID_STRING_PTR(pcszResolvedPath, CSTR)); ASSERT(IS_VALID_WRITE_PTR(pdwOutFlags, DWORD)); ASSERT(IS_VALID_WRITE_PTR(ppiliUpdated, PILINKINFO)); *pdwOutFlags = 0; bResult = CreateILinkInfo(pcszResolvedPath, ppiliUpdated); if (bResult) { if (UseNewILinkInfo(pcili, *ppiliUpdated)) { SET_FLAG(*pdwOutFlags, RLI_OFL_UPDATED); WARNING_OUT((TEXT("UpdateILinkInfo(): Updating ILinkInfo for path %s."), pcszResolvedPath)); } } ASSERT(! bResult || (IS_FLAG_CLEAR(*pdwOutFlags, RLI_OFL_UPDATED) || IS_VALID_STRUCT_PTR(*ppiliUpdated, CILINKINFO))); return(bResult); } /* ** UseNewILinkInfo() ** ** ** ** Arguments: ** ** Returns: TRUE if the new ILinkInfo structure contains more or ** different information than the old ILinkInfo structure. ** ** Side Effects: none */ PRIVATE_CODE BOOL UseNewILinkInfo(PCILINKINFO pciliOld, PCILINKINFO pciliNew) { BOOL bUpdate = FALSE; ASSERT(IS_VALID_STRUCT_PTR(pciliOld, CILINKINFO)); ASSERT(IS_VALID_STRUCT_PTR(pciliNew, CILINKINFO)); /* Does the new ILinkInfo structure contain local information? */ if (IS_FLAG_SET(pciliNew->dwFlags, ILI_FL_LOCAL_INFO_VALID)) { /* Yes. Does the old ILinkInfo structure contain local information? */ if (IS_FLAG_SET(pciliOld->dwFlags, ILI_FL_LOCAL_INFO_VALID)) /* * Yes. Update the old ILinkInfo structure if local information * differs. */ bUpdate = (CompareILinkInfoLocalData(pciliOld, pciliNew) != CR_EQUAL); else /* No. Update the old ILinkInfo structure. */ bUpdate = TRUE; } else /* No. Do not update the old ILinkInfo structure. */ bUpdate = FALSE; /* * Do we already need to update the old ILinkInfo structure based on local * information comparison? */ if (! bUpdate) { /* No. Compare remote information. */ /* Does the new ILinkInfo structure contain remote information? */ if (IS_FLAG_SET(pciliNew->dwFlags, ILI_FL_REMOTE_INFO_VALID)) { /* * Yes. Does the old ILinkInfo structure contain remote information? */ if (IS_FLAG_SET(pciliOld->dwFlags, ILI_FL_REMOTE_INFO_VALID)) /* * Yes. Update the old ILinkInfo structure if remote information * differs. */ bUpdate = (CompareILinkInfoRemoteData(pciliOld, pciliNew) != CR_EQUAL); else /* No. Update the old ILinkInfo structure. */ bUpdate = TRUE; } } return(bUpdate); } /* ** ResolveLocalILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL ResolveLocalILinkInfo(PCILINKINFO pcili, LPTSTR pszResolvedPathBuf, int cchMax, DWORD dwInFlags) { BOOL bResult; DWORD dwLocalSearchFlags; TCHAR rgchLocalPath[MAX_PATH_LEN]; ASSERT(IS_VALID_STRUCT_PTR(pcili, CILINKINFO)); ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszResolvedPathBuf, STR, MAX_PATH_LEN)); ASSERT(FLAGS_ARE_VALID(dwInFlags, ALL_RLI_IFLAGS)); /* Search for local path. */ TRACE_OUT((TEXT("ResolveLocalILinkInfo(): Attempting to resolve LinkInfo locally."))); GetLocalPathFromILinkInfo(pcili, rgchLocalPath, ARRAYSIZE(rgchLocalPath)); if (IS_FLAG_SET(dwInFlags, RLI_IFL_LOCAL_SEARCH)) dwLocalSearchFlags = SFLP_IFL_LOCAL_SEARCH; else dwLocalSearchFlags = 0; bResult = SearchForLocalPath(ILI_Volume_ID_Ptr(pcili), rgchLocalPath, dwLocalSearchFlags, pszResolvedPathBuf, cchMax); ASSERT(! bResult || EVAL(IsCanonicalPath(pszResolvedPathBuf))); return(bResult); } /* ** ResolveRemoteILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL ResolveRemoteILinkInfo(PCILINKINFO pcili, LPTSTR pszResolvedPathBuf, DWORD dwInFlags, HWND hwndOwner, PDWORD pdwOutFlags) { BOOL bResult; ASSERT(IS_VALID_STRUCT_PTR(pcili, CILINKINFO)); ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszResolvedPathBuf, STR, MAX_PATH_LEN)); ASSERT(FLAGS_ARE_VALID(dwInFlags, ALL_RLI_IFLAGS)); ASSERT(IS_FLAG_CLEAR(dwInFlags, RLI_IFL_ALLOW_UI) || IS_VALID_HANDLE(hwndOwner, WND)); ASSERT(IS_VALID_WRITE_PTR(pdwOutFlags, DWORD)); ASSERT(IS_FLAG_CLEAR(dwInFlags, RLI_IFL_TEMPORARY) || IS_FLAG_SET(dwInFlags, RLI_IFL_CONNECT)); TRACE_OUT((TEXT("ResolveRemoteILinkInfo(): Attempting to resolve LinkInfo remotely."))); /* Connect if requested. */ if (IS_FLAG_SET(dwInFlags, RLI_IFL_CONNECT)) { DWORD dwConnectInFlags; DWORD dwConnectOutFlags; dwConnectInFlags = 0; if (IS_FLAG_SET(dwInFlags, RLI_IFL_TEMPORARY)) dwConnectInFlags = CONNECT_TEMPORARY; if (IS_FLAG_SET(dwInFlags, RLI_IFL_ALLOW_UI)) SET_FLAG(dwConnectInFlags, CONNECT_INTERACTIVE); if (IS_FLAG_SET(dwInFlags, RLI_IFL_REDIRECT)) SET_FLAG(dwConnectInFlags, CONNECT_REDIRECT); bResult = ConnectToCNR(ILI_CNR_Link_Ptr(pcili), dwConnectInFlags, hwndOwner, pszResolvedPathBuf, &dwConnectOutFlags); if (bResult) { #ifdef UNICODE WCHAR szWideCommonPathSuffix[MAX_PATH]; LPWSTR pszWideCommonPathSuffix; if (pcili->ucbHeaderSize == SIZEOF(ILINKINFOA)) { pszWideCommonPathSuffix = szWideCommonPathSuffix; MultiByteToWideChar(CP_ACP, 0, ILI_Common_Path_Suffix_PtrA(pcili), -1, szWideCommonPathSuffix, ARRAYSIZE(szWideCommonPathSuffix)); } else { pszWideCommonPathSuffix = ILI_Common_Path_Suffix_Ptr(pcili); } CatPath(pszResolvedPathBuf, pszWideCommonPathSuffix, MAX_PATH_LEN); #else CatPath(pszResolvedPathBuf, ILI_Common_Path_Suffix_Ptr(pcili), MAX_PATH_LEN); #endif if (IS_FLAG_SET(dwConnectOutFlags, CONNECT_REFCOUNT)) { ASSERT(IS_FLAG_CLEAR(dwConnectOutFlags, CONNECT_LOCALDRIVE)); SET_FLAG(*pdwOutFlags, RLI_OFL_DISCONNECT); } } } else { /* * It's ok that IsCNRAvailable() and GetRemotePathFromILinkInfo() are * broken for NPs whose CNR names are not valid file system root paths. * * For NPs whose CNR names are valid file system root paths, * IsCNRAvailable() will succeed or fail, and * GetRemotePathFromILinkInfo() will be called only on success. * * For NPs whose CNR names are not valid file system root paths, * IsCNRAvailable() will fail and GetRemotePathFromILinkInfo() will not * be called. */ bResult = IsCNRAvailable(ILI_CNR_Link_Ptr(pcili)); if (bResult) GetRemotePathFromILinkInfo(pcili, pszResolvedPathBuf, MAX_PATH_LEN); } ASSERT(! bResult || (EVAL(IsCanonicalPath(pszResolvedPathBuf)) && FLAGS_ARE_VALID(*pdwOutFlags, ALL_RLI_OFLAGS))); return(bResult); } /* ** ResolveILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL ResolveILinkInfo(PCILINKINFO pcili, LPTSTR pszResolvedPathBuf, int cchMax, DWORD dwInFlags, HWND hwndOwner, PDWORD pdwOutFlags) { BOOL bResult; BOOL bLocalInfoValid; BOOL bRemoteInfoValid; BOOL bLocalShare; ASSERT(IS_VALID_STRUCT_PTR(pcili, CILINKINFO)); ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszResolvedPathBuf, STR, MAX_PATH_LEN)); ASSERT(FLAGS_ARE_VALID(dwInFlags, ALL_RLI_IFLAGS)); ASSERT(IS_FLAG_CLEAR(dwInFlags, RLI_IFL_ALLOW_UI) || IS_VALID_HANDLE(hwndOwner, WND)); ASSERT(IS_VALID_WRITE_PTR(pdwOutFlags, DWORD)); *pdwOutFlags = 0; /* Describe LinkInfo contents. */ bRemoteInfoValid = IS_FLAG_SET(pcili->dwFlags, ILI_FL_REMOTE_INFO_VALID); bLocalInfoValid = IS_FLAG_SET(pcili->dwFlags, ILI_FL_LOCAL_INFO_VALID); ASSERT(bLocalInfoValid || bRemoteInfoValid); /* * RAIDRAID: (15703) We will resolve to the wrong local path for a share * that has been moved to another path here. */ bLocalShare = FALSE; if (bRemoteInfoValid) { DWORD dwLocalShareFlags; /* Ask the server for the local path. */ bResult = ResolveLocalPathFromServer(pcili, pszResolvedPathBuf, &dwLocalShareFlags); if (IS_FLAG_SET(dwLocalShareFlags, CNR_FL_LOCAL)) bLocalShare = TRUE; if (bResult) { ASSERT(IS_FLAG_SET(dwLocalShareFlags, CNR_FL_LOCAL)); TRACE_OUT((TEXT("ResolveILinkInfo(): Resolved local path from server."))); } } else /* Can't tell if the referent is local or not. */ bResult = FALSE; if (! bResult) { /* Try local path. */ if (bLocalInfoValid) bResult = ResolveLocalILinkInfo(pcili, pszResolvedPathBuf, cchMax, dwInFlags); if (! bResult) { /* Try remote path. */ if (bRemoteInfoValid && ! bLocalShare) bResult = ResolveRemoteILinkInfo(pcili, pszResolvedPathBuf, dwInFlags, hwndOwner, pdwOutFlags); } } ASSERT(! bResult || (EVAL(IsCanonicalPath(pszResolvedPathBuf)) && FLAGS_ARE_VALID(*pdwOutFlags, ALL_RLI_OFLAGS))); return(bResult); } /* ** ResolveLocalPathFromServer() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL ResolveLocalPathFromServer(PCILINKINFO pcili, LPTSTR pszResolvedPathBuf, PDWORD pdwOutFlags) { BOOL bResult; ASSERT(IS_VALID_STRUCT_PTR(pcili, CILINKINFO)); ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszResolvedPathBuf, STR, MAX_PATH_LEN)); ASSERT(IS_VALID_WRITE_PTR(pdwOutFlags, DWORD)); ASSERT(IS_FLAG_SET(pcili->dwFlags, ILI_FL_REMOTE_INFO_VALID)); /* Try to get local path from server. */ bResult = GetLocalPathFromCNRLink(ILI_CNR_Link_Ptr(pcili), pszResolvedPathBuf, pdwOutFlags); if (bResult) { #ifdef UNICODE WCHAR szWideCommonPathSuffix[MAX_PATH]; LPWSTR pszWideCommonPathSuffix; ASSERT(IS_FLAG_SET(*pdwOutFlags, CNR_FL_LOCAL)); if (pcili->ucbHeaderSize == SIZEOF(ILINKINFOA)) { pszWideCommonPathSuffix = szWideCommonPathSuffix; MultiByteToWideChar(CP_ACP, 0, ILI_Common_Path_Suffix_PtrA(pcili), -1, szWideCommonPathSuffix, MAX_PATH); } else { pszWideCommonPathSuffix = ILI_Common_Path_Suffix_Ptr(pcili); } CatPath(pszResolvedPathBuf, pszWideCommonPathSuffix, MAX_PATH_LEN); #else ASSERT(IS_FLAG_SET(*pdwOutFlags, CNR_FL_LOCAL)); CatPath(pszResolvedPathBuf, ILI_Common_Path_Suffix_Ptr(pcili), MAX_PATH_LEN); #endif } ASSERT(FLAGS_ARE_VALID(*pdwOutFlags, ALL_CNR_FLAGS) && (! bResult || (EVAL(IS_FLAG_SET(*pdwOutFlags, CNR_FL_LOCAL)) && EVAL(IsLocalDrivePath(pszResolvedPathBuf))))); return(bResult); } /* ** GetLocalPathFromILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE void GetLocalPathFromILinkInfo(PCILINKINFO pcili, LPTSTR pszResolvedPathBuf, int cchMax) { #ifdef UNICODE WCHAR szWideLocalBasePath[MAX_PATH]; LPWSTR pszWideLocalBasePath; WCHAR szWideCommonPathSuffix[MAX_PATH]; LPWSTR pszWideCommonPathSuffix; #endif ASSERT(IS_VALID_STRUCT_PTR(pcili, CILINKINFO)); ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszResolvedPathBuf, STR, cchMax)); #ifdef UNICODE if (pcili->ucbHeaderSize == SIZEOF(ILINKINFOA)) { pszWideLocalBasePath = szWideLocalBasePath; MultiByteToWideChar(CP_ACP, 0, ILI_Local_Base_Path_PtrA(pcili), -1, szWideLocalBasePath, ARRAYSIZE(szWideLocalBasePath)); pszWideCommonPathSuffix = szWideCommonPathSuffix; MultiByteToWideChar(CP_ACP, 0, ILI_Common_Path_Suffix_PtrA(pcili), -1, szWideCommonPathSuffix, ARRAYSIZE(szWideCommonPathSuffix)); } else { pszWideLocalBasePath = ILI_Local_Base_Path_Ptr(pcili); pszWideCommonPathSuffix = ILI_Common_Path_Suffix_Ptr(pcili); } lstrcpyn(pszResolvedPathBuf, pszWideLocalBasePath, cchMax); CatPath(pszResolvedPathBuf, pszWideCommonPathSuffix, cchMax); #else lstrcpyn(pszResolvedPathBuf, ILI_Local_Base_Path_Ptr(pcili), cchMax); CatPath(pszResolvedPathBuf, ILI_Common_Path_Suffix_Ptr(pcili), cchMax); #endif ASSERT(lstrlen(pszResolvedPathBuf) < cchMax); ASSERT(IsDrivePath(pszResolvedPathBuf)); return; } /* ** GetRemotePathFromILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE void GetRemotePathFromILinkInfo(PCILINKINFO pcili, LPTSTR pszResolvedPathBuf, int cchMax) { #ifdef UNICODE WCHAR szWideCommonPathSuffix[MAX_PATH]; LPWSTR pszWideCommonPathSuffix; #endif ASSERT(IS_VALID_STRUCT_PTR(pcili, CILINKINFO)); ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszResolvedPathBuf, STR, cchMax)); /* It's ok that this is broken for non-UNC CNR names. */ GetRemotePathFromCNRLink(ILI_CNR_Link_Ptr(pcili), pszResolvedPathBuf, cchMax); #ifdef UNICODE if (pcili->ucbHeaderSize == SIZEOF(ILINKINFOA)) { pszWideCommonPathSuffix = szWideCommonPathSuffix; MultiByteToWideChar(CP_ACP, 0, ILI_Common_Path_Suffix_PtrA(pcili), -1, szWideCommonPathSuffix, ARRAYSIZE(szWideCommonPathSuffix)); } else { pszWideCommonPathSuffix = ILI_Common_Path_Suffix_Ptr(pcili); } CatPath(pszResolvedPathBuf, pszWideCommonPathSuffix, cchMax); #else CatPath(pszResolvedPathBuf, ILI_Common_Path_Suffix_Ptr(pcili), cchMax); #endif return; } /* ** CompareILinkInfoReferents() ** ** Compares the referents of two ILINKINFO structures. ** ** Arguments: ** ** Returns: ** ** Side Effects: none ** ** Comparison is performed on ILINKINFO data in only one of the following ways ** in the following order: ** ** 1) local data compared with local data ** 2) remote data compared with remote data ** 3) local data only < remote data only */ PRIVATE_CODE COMPARISONRESULT CompareILinkInfoReferents(PCILINKINFO pciliFirst, PCILINKINFO pciliSecond) { COMPARISONRESULT cr; ASSERT(IS_VALID_STRUCT_PTR(pciliFirst, CILINKINFO)); ASSERT(IS_VALID_STRUCT_PTR(pciliSecond, CILINKINFO)); /* * We can't just perform a binary comparison of the two ILinkInfos here. We * may have two LinkInfos that refer to the same path, but differ in case on * a non-case-sensitive file system. */ /* Compare ILinkInfos by local or remote data. */ if (IS_FLAG_SET(pciliFirst->dwFlags, ILI_FL_LOCAL_INFO_VALID) && IS_FLAG_SET(pciliSecond->dwFlags, ILI_FL_LOCAL_INFO_VALID)) /* Compare local data. */ cr = CompareILinkInfoLocalData(pciliFirst, pciliSecond); else if (IS_FLAG_SET(pciliFirst->dwFlags, ILI_FL_REMOTE_INFO_VALID) && IS_FLAG_SET(pciliSecond->dwFlags, ILI_FL_REMOTE_INFO_VALID)) /* Compare remote data. */ cr = CompareILinkInfoRemoteData(pciliFirst, pciliSecond); else { /* * One contains only valid local information and the other contains only * valid remote information. */ ASSERT(! ((pciliFirst->dwFlags & (ILI_FL_LOCAL_INFO_VALID | ILI_FL_REMOTE_INFO_VALID)) & (pciliSecond->dwFlags & (ILI_FL_LOCAL_INFO_VALID | ILI_FL_REMOTE_INFO_VALID)))); /* By fiat, local only < remote only. */ if (IS_FLAG_SET(pciliFirst->dwFlags, ILI_FL_LOCAL_INFO_VALID)) cr = CR_FIRST_SMALLER; else cr = CR_FIRST_LARGER; } ASSERT(IsValidCOMPARISONRESULT(cr)); return(cr); } /* ** CompareILinkInfoLocalData() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none ** ** Local ILinkInfo data is compared in the following order: ** ** 1) volume ID ** 2) sub path from root */ PRIVATE_CODE COMPARISONRESULT CompareILinkInfoLocalData(PCILINKINFO pciliFirst, PCILINKINFO pciliSecond) { COMPARISONRESULT cr; ASSERT(IS_VALID_STRUCT_PTR(pciliFirst, CILINKINFO)); ASSERT(IS_VALID_STRUCT_PTR(pciliSecond, CILINKINFO)); cr = CompareVolumeIDs(ILI_Volume_ID_Ptr(pciliFirst), ILI_Volume_ID_Ptr(pciliSecond)); if (cr == CR_EQUAL) cr = CompareLocalPaths(pciliFirst, pciliSecond); ASSERT(IsValidCOMPARISONRESULT(cr)); return(cr); } /* ** CompareLocalPaths() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE COMPARISONRESULT CompareLocalPaths(PCILINKINFO pciliFirst, PCILINKINFO pciliSecond) { COMPARISONRESULT cr; TCHAR rgchFirstLocalPath[MAX_PATH_LEN]; TCHAR rgchSecondLocalPath[MAX_PATH_LEN]; ASSERT(IS_VALID_STRUCT_PTR(pciliFirst, CILINKINFO)); ASSERT(IS_VALID_STRUCT_PTR(pciliSecond, CILINKINFO)); GetLocalPathFromILinkInfo(pciliFirst, rgchFirstLocalPath, ARRAYSIZE(rgchFirstLocalPath)); GetLocalPathFromILinkInfo(pciliSecond, rgchSecondLocalPath, ARRAYSIZE(rgchSecondLocalPath)); cr = ComparePathStrings(rgchFirstLocalPath, rgchSecondLocalPath); ASSERT(IsValidCOMPARISONRESULT(cr)); return(cr); } /* ** CompareILinkInfoRemoteData() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE COMPARISONRESULT CompareILinkInfoRemoteData(PCILINKINFO pciliFirst, PCILINKINFO pciliSecond) { COMPARISONRESULT cr; #ifdef UNICODE WCHAR szWideCommonPathSuffixFirst[MAX_PATH]; WCHAR szWideCommonPathSuffixSecond[MAX_PATH]; LPWSTR pszWideCommonPathSuffixFirst; LPWSTR pszWideCommonPathSuffixSecond; #endif ASSERT(IS_VALID_STRUCT_PTR(pciliFirst, CILINKINFO)); ASSERT(IS_VALID_STRUCT_PTR(pciliSecond, CILINKINFO)); cr = CompareCNRLinks(ILI_CNR_Link_Ptr(pciliFirst), ILI_CNR_Link_Ptr(pciliSecond)); #ifdef UNICODE if (pciliFirst->ucbHeaderSize == SIZEOF(ILINKINFOA)) { pszWideCommonPathSuffixFirst = szWideCommonPathSuffixFirst; MultiByteToWideChar(CP_ACP, 0, ILI_Common_Path_Suffix_PtrA(pciliFirst), -1, szWideCommonPathSuffixFirst, MAX_PATH); } else { pszWideCommonPathSuffixFirst = ILI_Common_Path_Suffix_Ptr(pciliFirst); } if (pciliSecond->ucbHeaderSize == SIZEOF(ILINKINFOA)) { pszWideCommonPathSuffixSecond = szWideCommonPathSuffixSecond; MultiByteToWideChar(CP_ACP, 0, ILI_Common_Path_Suffix_PtrA(pciliSecond), -1, szWideCommonPathSuffixSecond, MAX_PATH); } else { pszWideCommonPathSuffixSecond = ILI_Common_Path_Suffix_Ptr(pciliSecond); } #else if (cr == CR_EQUAL) cr = ComparePathStrings(ILI_Common_Path_Suffix_Ptr(pciliFirst), ILI_Common_Path_Suffix_Ptr(pciliSecond)); #endif ASSERT(IsValidCOMPARISONRESULT(cr)); return(cr); } /* ** CompareILinkInfoVolumes() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE COMPARISONRESULT CompareILinkInfoVolumes(PCILINKINFO pciliFirst, PCILINKINFO pciliSecond) { COMPARISONRESULT cr; BOOL bFirstLocal; BOOL bFirstRemote; BOOL bSecondLocal; BOOL bSecondRemote; ASSERT(IS_VALID_STRUCT_PTR(pciliFirst, CILINKINFO)); ASSERT(IS_VALID_STRUCT_PTR(pciliSecond, CILINKINFO)); bFirstLocal = IS_FLAG_SET(((PCILINKINFO)pciliFirst)->dwFlags, ILI_FL_LOCAL_INFO_VALID); bFirstRemote = IS_FLAG_SET(((PCILINKINFO)pciliFirst)->dwFlags, ILI_FL_REMOTE_INFO_VALID); bSecondLocal = IS_FLAG_SET(((PCILINKINFO)pciliSecond)->dwFlags, ILI_FL_LOCAL_INFO_VALID); bSecondRemote = IS_FLAG_SET(((PCILINKINFO)pciliSecond)->dwFlags, ILI_FL_REMOTE_INFO_VALID); if (bFirstLocal && bSecondLocal) /* First and second have local information. */ cr = CompareVolumeIDs(ILI_Volume_ID_Ptr((PCILINKINFO)pciliFirst), ILI_Volume_ID_Ptr((PCILINKINFO)pciliSecond)); else if (bFirstRemote && bSecondRemote) /* First and second have remote information. */ cr = CompareCNRLinks(ILI_CNR_Link_Ptr((PCILINKINFO)pciliFirst), ILI_CNR_Link_Ptr((PCILINKINFO)pciliSecond)); else { /* * One contains only valid local information and the other contains only * valid remote information. */ ASSERT(! ((pciliFirst->dwFlags & (ILI_FL_LOCAL_INFO_VALID | ILI_FL_REMOTE_INFO_VALID)) & (pciliSecond->dwFlags & (ILI_FL_LOCAL_INFO_VALID | ILI_FL_REMOTE_INFO_VALID)))); /* By fiat, local only < remote only. */ if (bFirstLocal) /* * First has only local information. Second has only remote * information. */ cr = CR_FIRST_SMALLER; else /* * First has only remote information. Second has only local * information. */ cr = CR_FIRST_LARGER; } return(cr); } /* ** CheckCombinedPathLen() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL CheckCombinedPathLen(LPCTSTR pcszBase, LPCTSTR pcszSuffix) { BOOL bResult; ASSERT(IS_VALID_STRING_PTR(pcszBase, CSTR)); ASSERT(IS_VALID_STRING_PTR(pcszSuffix, CSTR)); bResult = EVAL(lstrlen(pcszBase) + lstrlen(pcszSuffix) < MAX_PATH_LEN); if (bResult) { TCHAR rgchCombinedPath[MAX_PATH_LEN + 1]; lstrcpyn(rgchCombinedPath, pcszBase, ARRAYSIZE(rgchCombinedPath)); CatPath(rgchCombinedPath, pcszSuffix, ARRAYSIZE(rgchCombinedPath)); bResult = EVAL(lstrlen(rgchCombinedPath) < MAX_PATH_LEN); } return(bResult); } /* ** GetILinkInfoData() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL GetILinkInfoData(PCILINKINFO pcili, LINKINFODATATYPE lidt, PCVOID *ppcvData) { BOOL bResult = FALSE; ASSERT(IS_VALID_STRUCT_PTR(pcili, CILINKINFO)); ASSERT(IsValidLINKINFODATATYPE(lidt)); ASSERT(IS_VALID_WRITE_PTR(ppcvData, PCVOID)); switch (lidt) { case LIDT_VOLUME_SERIAL_NUMBER: if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_LOCAL_INFO_VALID)) bResult = GetVolumeSerialNumber(ILI_Volume_ID_Ptr(pcili), (PCDWORD *)ppcvData); break; case LIDT_DRIVE_TYPE: if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_LOCAL_INFO_VALID)) bResult = GetVolumeDriveType(ILI_Volume_ID_Ptr(pcili), (PCUINT *)ppcvData); break; case LIDT_VOLUME_LABEL: if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_LOCAL_INFO_VALID)) bResult = GetVolumeLabel(ILI_Volume_ID_Ptr(pcili), (LPCSTR *)ppcvData); break; case LIDT_VOLUME_LABELW: #ifdef UNICODE if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_LOCAL_INFO_VALID)) bResult = GetVolumeLabelW(ILI_Volume_ID_Ptr(pcili), (LPCTSTR *)ppcvData); #endif break; case LIDT_LOCAL_BASE_PATH: if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_LOCAL_INFO_VALID)) { *ppcvData = ILI_Local_Base_Path_PtrA(pcili); bResult = TRUE; } break; case LIDT_LOCAL_BASE_PATHW: #ifdef UNICODE if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_LOCAL_INFO_VALID)) { if (pcili->ucbHeaderSize == SIZEOF(ILINKINFOA)) *ppcvData = NULL; else *ppcvData = ILI_Local_Base_Path_PtrW(pcili); bResult = TRUE; } #endif break; case LIDT_NET_TYPE: if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_REMOTE_INFO_VALID)) bResult = GetCNRNetType(ILI_CNR_Link_Ptr(pcili), (PCDWORD *)ppcvData); break; case LIDT_NET_RESOURCE: if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_REMOTE_INFO_VALID)) bResult = GetCNRName(ILI_CNR_Link_Ptr(pcili), (LPCSTR *)ppcvData); break; case LIDT_NET_RESOURCEW: #ifdef UNICODE if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_REMOTE_INFO_VALID)) bResult = GetCNRNameW(ILI_CNR_Link_Ptr(pcili), (LPCWSTR *)ppcvData); #endif break; case LIDT_REDIRECTED_DEVICE: if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_REMOTE_INFO_VALID)) bResult = GetLastRedirectedDevice(ILI_CNR_Link_Ptr(pcili), (LPCSTR *)ppcvData); break; case LIDT_REDIRECTED_DEVICEW: #ifdef UNICODE if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_REMOTE_INFO_VALID)) bResult = GetLastRedirectedDeviceW(ILI_CNR_Link_Ptr(pcili), (LPCWSTR *)ppcvData); #endif break; case LIDT_COMMON_PATH_SUFFIX: *ppcvData = ILI_Common_Path_Suffix_PtrA(pcili); bResult = TRUE; break; case LIDT_COMMON_PATH_SUFFIXW: #ifdef UNICODE if (pcili->ucbHeaderSize == sizeof(ILINKINFOA)) { *ppcvData = NULL; } else { *ppcvData = ILI_Common_Path_Suffix_PtrW(pcili); } bResult = TRUE; #endif break; default: bResult = FALSE; ERROR_OUT((TEXT("GetILinkInfoData(): Bad LINKINFODATATYPE %d."), lidt)); break; } return(bResult); } /* ** DisconnectILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL DisconnectILinkInfo(PCILINKINFO pcili) { ASSERT(IS_VALID_STRUCT_PTR(pcili, CILINKINFO)); return(DisconnectFromCNR(ILI_CNR_Link_Ptr(pcili))); } #if defined(DEBUG) || defined(EXPV) /* ** IsValidLINKINFODATATYPE() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL IsValidLINKINFODATATYPE(LINKINFODATATYPE lidt) { BOOL bResult; switch (lidt) { case LIDT_VOLUME_SERIAL_NUMBER: case LIDT_DRIVE_TYPE: case LIDT_VOLUME_LABEL: case LIDT_VOLUME_LABELW: case LIDT_LOCAL_BASE_PATH: case LIDT_LOCAL_BASE_PATHW: case LIDT_NET_TYPE: case LIDT_NET_RESOURCE: case LIDT_NET_RESOURCEW: case LIDT_REDIRECTED_DEVICE: case LIDT_COMMON_PATH_SUFFIX: case LIDT_COMMON_PATH_SUFFIXW: bResult = TRUE; break; default: bResult = FALSE; ERROR_OUT((TEXT("IsValidLINKINFODATATYPE(): Invalid LINKINFODATATYPE %d."), lidt)); break; } return(bResult); } #endif #if defined(DEBUG) || defined(VSTF) /* ** CheckILIFlags() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL CheckILIFlags(PCILINKINFO pcili) { return(FLAGS_ARE_VALID(pcili->dwFlags, ALL_ILINKINFO_FLAGS) && (IS_FLAG_SET(pcili->dwFlags, ILI_FL_LOCAL_INFO_VALID) || IS_FLAG_SET(pcili->dwFlags, ILI_FL_REMOTE_INFO_VALID))); } /* ** CheckILICommonPathSuffix() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL CheckILICommonPathSuffix(PCILINKINFO pcili) { return(IS_VALID_STRING_PTRA(ILI_Common_Path_Suffix_PtrA(pcili), CSTR) && EVAL(IsContained(pcili, pcili->li.ucbSize, ILI_Common_Path_Suffix_PtrA(pcili), lstrlenA(ILI_Common_Path_Suffix_PtrA(pcili)))) && EVAL(! IS_SLASH(*ILI_Common_Path_Suffix_PtrA(pcili)))); } /* ** CheckILILocalInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL CheckILILocalInfo(PCILINKINFO pcili) { #ifdef UNICODE WCHAR szWideLocalBasePath[MAX_PATH]; WCHAR szWideCommonPathSuffix[MAX_PATH]; LPWSTR pszWideLocalBasePath; LPWSTR pszWideCommonPathSuffix; if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_LOCAL_INFO_VALID)) return FALSE; if (!IS_VALID_STRUCT_PTR(ILI_Volume_ID_Ptr(pcili), CVOLUMEID)) return FALSE; if (!EVAL(IsContained(pcili, pcili->li.ucbSize,ILI_Volume_ID_Ptr(pcili), GetVolumeIDLen(ILI_Volume_ID_Ptr(pcili))))) return FALSE; if (pcili->ucbHeaderSize == SIZEOF(ILINKINFOA)) { pszWideLocalBasePath = szWideLocalBasePath; MultiByteToWideChar(CP_ACP, 0, ILI_Local_Base_Path_PtrA(pcili), -1, szWideLocalBasePath, MAX_PATH); pszWideCommonPathSuffix = szWideCommonPathSuffix; MultiByteToWideChar(CP_ACP, 0, ILI_Common_Path_Suffix_PtrA(pcili), -1, szWideCommonPathSuffix, MAX_PATH); } else { pszWideLocalBasePath = ILI_Local_Base_Path_Ptr(pcili); pszWideCommonPathSuffix = ILI_Common_Path_Suffix_Ptr(pcili); } if (!EVAL(IsDrivePath(pszWideLocalBasePath))) return FALSE; if (!EVAL(IsContained(pcili, pcili->li.ucbSize, ILI_Local_Base_Path_PtrA(pcili), lstrlenA(ILI_Local_Base_Path_PtrA(pcili))))) return FALSE; if (!EVAL(CheckCombinedPathLen(pszWideLocalBasePath, pszWideCommonPathSuffix))) return FALSE; return TRUE; #else return(IS_FLAG_CLEAR(pcili->dwFlags, ILI_FL_LOCAL_INFO_VALID) || /* Check volume ID. */ (IS_VALID_STRUCT_PTR(ILI_Volume_ID_Ptr(pcili), CVOLUMEID) && EVAL(IsContained(pcili, pcili->li.ucbSize, ILI_Volume_ID_Ptr(pcili), GetVolumeIDLen(ILI_Volume_ID_Ptr(pcili)))) && /* Check local base path. */ EVAL(IsDrivePath(ILI_Local_Base_Path_Ptr(pcili))) && EVAL(IsContained(pcili, pcili->li.ucbSize, ILI_Local_Base_Path_PtrA(pcili), lstrlen(ILI_Local_Base_Path_Ptr(pcili)))) && EVAL(CheckCombinedPathLen(ILI_Local_Base_Path_Ptr(pcili), ILI_Common_Path_Suffix_Ptr(pcili))))); #endif } /* ** CheckILIRemoteInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL CheckILIRemoteInfo(PCILINKINFO pcili) { BOOL bResult; if (IS_FLAG_CLEAR(pcili->dwFlags, ILI_FL_REMOTE_INFO_VALID)) bResult = TRUE; else { /* Check CNR link. */ if (IS_VALID_STRUCT_PTR(ILI_CNR_Link_Ptr(pcili), CCNRLINK) && EVAL(IsContained(pcili, pcili->li.ucbSize, ILI_CNR_Link_Ptr(pcili), GetCNRLinkLen(ILI_CNR_Link_Ptr(pcili))))) { TCHAR rgchRemoteBasePath[MAX_PATH_LEN]; #ifdef UNICODE WCHAR szWideCommonPathSuffix[MAX_PATH]; LPWSTR pszWideCommonPathSuffix; #endif /* RAIDRAID: (15724) This is broken for non-UNC CNR names. */ GetRemotePathFromCNRLink(ILI_CNR_Link_Ptr(pcili), rgchRemoteBasePath, ARRAYSIZE(rgchRemoteBasePath)); #ifdef UNICODE if (pcili->ucbHeaderSize == SIZEOF(ILINKINFOA)) { pszWideCommonPathSuffix = szWideCommonPathSuffix; MultiByteToWideChar(CP_ACP, 0, ILI_Common_Path_Suffix_PtrA(pcili), -1, szWideCommonPathSuffix, ARRAYSIZE(szWideCommonPathSuffix)); } else { pszWideCommonPathSuffix = ILI_Common_Path_Suffix_Ptr(pcili); } bResult = EVAL(CheckCombinedPathLen(rgchRemoteBasePath, pszWideCommonPathSuffix)); #else bResult = EVAL(CheckCombinedPathLen(rgchRemoteBasePath, ILI_Common_Path_Suffix_Ptr(pcili))); #endif } else bResult = FALSE; } return(bResult); } /* ** IsValidPCLINKINFO() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL IsValidPCLINKINFO(PCLINKINFO pcli) { return(IS_VALID_STRUCT_PTR((PCILINKINFO)pcli, CILINKINFO)); } /* ** IsValidPCILINKINFO() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE BOOL IsValidPCILINKINFO(PCILINKINFO pcili) { /* * A "valid" LinkInfo structure has the following characteristics: * * 1) entire structure is readable * 2) size of ILINKINFO header structure >= SIZEOF(CILINKINFO) * 3) flags are valid * 4) either local info or remote info or both are valid * 5) contained structures and strings are valid and are entirely contained * in LinkInfo structure * 6) lstrlen() of combined paths < MAX_PATH_LEN */ return(IS_VALID_READ_PTR(pcili, CILINKINFO) && IS_VALID_READ_BUFFER_PTR(pcili, CILINKINFO, pcili->li.ucbSize) && EVAL(pcili->ucbHeaderSize >= SIZEOF(*pcili)) && EVAL(CheckILIFlags(pcili)) && EVAL(CheckILICommonPathSuffix(pcili)) && EVAL(CheckILILocalInfo(pcili)) && EVAL(CheckILIRemoteInfo(pcili))); } #endif #ifdef DEBUG /* ** DumpILinkInfo() ** ** ** ** Arguments: ** ** Returns: ** ** Side Effects: none */ PRIVATE_CODE void DumpILinkInfo(PCILINKINFO pcili) { #ifdef UNICODE WCHAR szWideCommonPathSuffix[MAX_PATH]; LPWSTR pszWideCommonPathSuffix; #endif ASSERT(IS_VALID_STRUCT_PTR(pcili, CILINKINFO)); PLAIN_TRACE_OUT((TEXT("%s[LinkInfo] ucbSize = %#x"), INDENT_STRING, pcili->li.ucbSize)); PLAIN_TRACE_OUT((TEXT("%s%s[ILinkInfo] ucbHeaderSize = %#x"), INDENT_STRING, INDENT_STRING, pcili->ucbHeaderSize)); PLAIN_TRACE_OUT((TEXT("%s%s[ILinkInfo] dwFLags = %#08lx"), INDENT_STRING, INDENT_STRING, pcili->dwFlags)); if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_LOCAL_INFO_VALID)) { #ifdef UNICODE WCHAR szWideLocalBasePath[MAX_PATH]; LPWSTR pszWideLocalBasePath; #endif DumpVolumeID(ILI_Volume_ID_Ptr(pcili)); #ifdef UNICODE if (pcili->ucbHeaderSize == SIZEOF(ILINKINFOA)) { pszWideLocalBasePath = szWideLocalBasePath; MultiByteToWideChar(CP_ACP, 0, ILI_Local_Base_Path_PtrA(pcili), -1, szWideLocalBasePath, ARRAYSIZE(szWideLocalBasePath)); } else { pszWideLocalBasePath = ILI_Local_Base_Path_Ptr(pcili); } PLAIN_TRACE_OUT((TEXT("%s%s[ILinkInfo] local base path \"%s\""), INDENT_STRING, INDENT_STRING, pszWideLocalBasePath)); #else PLAIN_TRACE_OUT((TEXT("%s%s[ILinkInfo] local base path \"%s\""), INDENT_STRING, INDENT_STRING, ILI_Local_Base_Path_Ptr(pcili))); #endif } if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_REMOTE_INFO_VALID)) DumpCNRLink(ILI_CNR_Link_Ptr(pcili)); #ifdef UNICODE if ( pcili->ucbHeaderSize == SIZEOF(ILINKINFOA)) { pszWideCommonPathSuffix = szWideCommonPathSuffix; MultiByteToWideChar(CP_ACP, 0, ILI_Common_Path_Suffix_PtrA(pcili), -1, szWideCommonPathSuffix, ARRAYSIZE(szWideCommonPathSuffix)); } else { pszWideCommonPathSuffix = ILI_Common_Path_Suffix_Ptr(pcili); } PLAIN_TRACE_OUT((TEXT("%s%s[ILinkInfo] common path suffix \"%s\""), INDENT_STRING, INDENT_STRING, pszWideCommonPathSuffix)); #else PLAIN_TRACE_OUT((TEXT("%s%s[ILinkInfo] common path suffix \"%s\""), INDENT_STRING, INDENT_STRING, ILI_Common_Path_Suffix_Ptr(pcili))); #endif return; } #endif /***************************** Exported Functions ****************************/ /****************************************************************************** @doc LINKINFOAPI @func BOOL | CreateLinkInfo | Creates a LinkInfo structure for a path. @parm PCSTR | pcszPath | A pointer to the path string that a LinkInfo structure is to be created for. @parm PLINKINFO * | ppli | A pointer to a PLINKINFO to be filled in with a pointer to the new LinkInfo structure. *ppli is only valid if TRUE is returned. @rdesc If a LinkInfo structure was created successfully, TRUE is returned, and *ppli contains a pointer to the new LinkInfo structure. Otherwise, a LinkInfo structure was not created successfully, and *ppli is undefined. The reason for failure may be determined by calling GetLastError(). @comm Once the caller is finshed with the LinkInfo structure returned by CreateLinkInfo(), DestroyLinkInfo() should be called to free the LinkInfo structure. The contents of the LinkInfo structure returned are opaque to the caller, with the exception of the first field of the LinkInfo structure. The first field of the LinkInfo structure, ucbSize, is a UINT containing the size of the LinkInfo structure in bytes, including the ucbSize field. The LinkInfo structure is created in memory that is private to the LinkInfo APIs. The returned LinkInfo structure should be copied into the caller's memory, and the DestroyLinkInfo() should be called to free the LinkInfo structure from the LinkInfo APIs' private memory. @xref DestroyLinkInfo ******************************************************************************/ LINKINFOAPI BOOL WINAPI CreateLinkInfo(LPCTSTR pcszPath, PLINKINFO *ppli) { BOOL bResult; DebugEntry(CreateLinkInfo); #ifdef EXPV /* Verify parameters. */ if (IS_VALID_STRING_PTR(pcszPath, CSTR) && IS_VALID_WRITE_PTR(ppli, PLINKINFO)) #endif { bResult = CreateILinkInfo(pcszPath, (PILINKINFO *)ppli); #ifdef DEBUG if (bResult) { TRACE_OUT((TEXT("CreateLinkInfo(): LinkInfo created for path %s:"), pcszPath)); DumpILinkInfo(*(PILINKINFO *)ppli); } #endif } #ifdef EXPV else { SetLastError(ERROR_INVALID_PARAMETER); bResult = FALSE; } #endif ASSERT(! bResult || IS_VALID_STRUCT_PTR(*ppli, CLINKINFO)); DebugExitBOOL(CreateLinkInfo, bResult); return(bResult); } #ifdef UNICODE LINKINFOAPI BOOL WINAPI CreateLinkInfoA(LPCSTR pcszPath, PLINKINFO *ppli) { LPWSTR lpwstr; UINT cchPath; cchPath = lstrlenA(pcszPath) + 1; lpwstr = (LPWSTR)_alloca(cchPath*SIZEOF(WCHAR)); if (MultiByteToWideChar(CP_ACP, 0, pcszPath, cchPath, lpwstr, cchPath) == 0) { return FALSE; } else { return CreateLinkInfo(lpwstr,ppli); } } #endif /****************************************************************************** @doc LINKINFOAPI @func void | DestroyLinkInfo | Destroys a LinkInfo structure created by CreateLinkInfo(). @parm PLINKINFO | pli | A pointer to the LinkInfo structure to be destroyed. @xref CreateLinkInfo ******************************************************************************/ LINKINFOAPI void WINAPI DestroyLinkInfo(PLINKINFO pli) { DebugEntry(DestroyLinkInfo); #ifdef EXPV /* Verify parameters. */ if ( IS_VALID_STRUCT_PTR(pli, CLINKINFO)) #endif { DestroyILinkInfo((PILINKINFO)pli); } DebugExitVOID(DestroyLinkInfo); return; } /****************************************************************************** @doc LINKINFOAPI @func int | CompareLinkInfoReferents | Compares the referents of two LinkInfo structures. @parm PCLINKINFO | pcliFirst | A pointer to the first LinkInfo structure whose referent is to be compared. @parm PCLINKINFO | pcliSecond | A pointer to the second LinkInfo structure whose referent is to be compared. @rdesc If the referent of the first LinkInfo structure is less than the referent of the second LinkInfo structure, a negative value is returned. If the referent of the first LinkInfo structure is the same as the referent of the second LinkInfo structure, zero is returned. If the referent of the first LinkInfo structure is larger than the referent of the second LinkInfo structure, a positive value is returned. An invalid LinkInfo structure is considered to have a referent that is less than the referent of any valid LinkInfo structure. All invalid LinkInfo structures are considered to have the same referent. @comm The value returned is actually a COMPARISONRESULT, for clients that understand COMPARISONRESULTs, like SYNCENG.DLL. @xref CompareLinkInfoVolumes ******************************************************************************/ LINKINFOAPI int WINAPI CompareLinkInfoReferents(PCLINKINFO pcliFirst, PCLINKINFO pcliSecond) { COMPARISONRESULT cr; BOOL bFirstValid; BOOL bSecondValid; DebugEntry(CompareLinkInfoReferents); bFirstValid = IS_VALID_STRUCT_PTR(pcliFirst, CLINKINFO); bSecondValid = IS_VALID_STRUCT_PTR(pcliSecond, CLINKINFO); if (bFirstValid) { if (bSecondValid) cr = CompareILinkInfoReferents((PCILINKINFO)pcliFirst, (PCILINKINFO)pcliSecond); else cr = CR_FIRST_LARGER; } else { if (bSecondValid) cr = CR_FIRST_SMALLER; else cr = CR_EQUAL; } ASSERT(IsValidCOMPARISONRESULT(cr)); DebugExitCOMPARISONRESULT(CompareLinkInfoReferents, cr); return(cr); } /****************************************************************************** @doc LINKINFOAPI @func int | CompareLinkInfoVolumes | Compares the volumes of the referents of two LinkInfo structures. @parm PCLINKINFO | pcliFirst | A pointer to the first LinkInfo structure whose referent's volume is to be compared. @parm PCLINKINFO | pcliSecond | A pointer to the second LinkInfo structure referent's volume is to be compared. @rdesc If the volume of the referent of the first LinkInfo structure is less than the volume of the referent of the second LinkInfo structure, a negative value is returned. If the volume of the referent of the first LinkInfo structure is the same as the volume of the referent of the second LinkInfo structure, zero is returned. If the volume of the referent of the first LinkInfo structure is larger than the volume of the referent of the second LinkInfo structure, a positive value is returned. An invalid LinkInfo structure is considered to have a referent's volume that is less than the referent's volume of any valid LinkInfo structure. All invalid LinkInfo structures are considered to have the same referent's volume. @comm The value returned is actually a COMPARISONRESULT, for clients that understand COMPARISONRESULTs, like SYNCENG.DLL. @xref CompareLinkInfoReferents ******************************************************************************/ LINKINFOAPI int WINAPI CompareLinkInfoVolumes(PCLINKINFO pcliFirst, PCLINKINFO pcliSecond) { COMPARISONRESULT cr; BOOL bFirstValid; BOOL bSecondValid; DebugEntry(CompareLinkInfoVolumes); bFirstValid = IS_VALID_STRUCT_PTR(pcliFirst, CLINKINFO); bSecondValid = IS_VALID_STRUCT_PTR(pcliSecond, CLINKINFO); if (bFirstValid) { if (bSecondValid) cr = CompareILinkInfoVolumes((PCILINKINFO)pcliFirst, (PCILINKINFO)pcliSecond); else cr = CR_FIRST_LARGER; } else { if (bSecondValid) cr = CR_FIRST_SMALLER; else cr = CR_EQUAL; } ASSERT(IsValidCOMPARISONRESULT(cr)); DebugExitCOMPARISONRESULT(CompareLinkInfoVolumes, cr); return(cr); } /****************************************************************************** @doc LINKINFOAPI @func BOOL | ResolveLinkInfo | Resolves a LinkInfo structure into a file system path on an available volume. @parm PCLINKINFO | pcli | A pointer to the LinkInfo structure to be resolved. @parm PSTR | pszResolvedPathBuf | A pointer to a buffer to be filled in with the path resolved to the LinkInfo structure's referent. @parm DWORD | dwInFlags | A bit mask of flags. This parameter may be any combination of the following values: @flag RLI_IFL_CONNECT | If set, connect to the referent's parent connectable network resource if necessary. If clear, no connection is established. @flag RLI_IFL_ALLOW_UI | If set, interaction with the user is permitted, and the hwndOwner parameter identifies the parent window to be used for any ui required. If clear, interaction with the user is not permitted. @flag RLI_IFL_REDIRECT | If set, the resolved path is a redirected logical device path. If clear, the resolved path is only a redirected logical device path if the RLI_IFL_CONNECT flag is set, and the network requires a redirected logical device path to make a connection. @flag RLI_IFL_UPDATE | If set and the source LinkInfo structure needs updating, RLI_OFL_UPDATED will be set in *pdwOutFlags and *ppliUpdated will point to an updated LinkInfo structure. If clear, RLI_OFL_UPDATED will be clear in *pdwOutFlags and *ppliUpdated is undefined. @flag RLI_IFL_LOCAL_SEARCH | If set, first the last known logical device for the referent's volume is checked for the volume, followed by all other local logical devices that handle the referent's volume's media type. If clear, only the last known logical device for the referent's volume is checked for the volume. @parm HWND | hwndOwner | A handle to the parent window to be used to bring up any ui required. This parameter is only used if RLI_IFL_ALLOW_UI is set in dwInFlags. Otherwise, it is ignored. @parm PDWORD | pdwOutFlags | A pointer to a DWORD to be filled in with a bit mask of flags. *pdwOutFlags is only valid if TRUE is returned. *pdwOutFlags may be any combination of the following values: @flag RLI_OFL_UPDATED | Only set if RLI_IFL_UPDATE was set in dwInFlags. If set, the source LinkInfo structure needed updating, and *ppliUpdated points to an updated LinkInfo structure. If clear, either RLI_IFL_UPDATE was clear in dwInFlags or the source LinkInfo structure didn't need updating, and *ppliUpdated is undefined. @parm PLINKINFO * | ppliUpdated | If RLI_IFL_UPDATE is set in dwInFlags, ppliUpdated is a pointer to a PLINKINFO to be filled in with a pointer to an updated LinkInfo structure, if necessary. If RLI_IFL_UPDATE is clear in dwInFlags, ppliUpdated is ignored. *ppliUpdated is only valid if RLI_OFL_UPDATED is set in *pdwOutFlags @rdesc If the LinkInfo was resolved to a path on an available successfully, TRUE is returned, pszResolvedPathBuf's buffer is filled in with a file system path to the LinkInfo structure's referent, and *pdwOutFlags is filled in as described above. Otherwise, FALSE is returned, the contents of pszResolved's buffer are undefined, and the contents of *pdwOutFlags are undefined. The reason for failure may be determined by calling GetLastError(). @comm Once the caller is finshed with any new, updated LinkInfo structure returned by ResolveLinkInfo(), DestroyLinkInfo() should be called to free the LinkInfo structure. @xref DestroyLinkInfo DisconnectLinkInfo ******************************************************************************/ LINKINFOAPI BOOL WINAPI ResolveLinkInfo(PCLINKINFO pcli, LPTSTR pszResolvedPathBuf, // MUST BE MAX_PATH_LEN SIZE DWORD dwInFlags, HWND hwndOwner, PDWORD pdwOutFlags, PLINKINFO *ppliUpdated) { BOOL bResult; DebugEntry(ResolveLinkInfo); #ifdef EXPV /* Verify parameters. */ if ( IS_VALID_STRUCT_PTR(pcli, CLINKINFO) && IS_VALID_WRITE_BUFFER_PTR(pszResolvedPathBuf, STR, MAX_PATH_LEN) && FLAGS_ARE_VALID(dwInFlags, ALL_RLI_IFLAGS) && (IS_FLAG_CLEAR(dwInFlags, RLI_IFL_ALLOW_UI) || IS_VALID_HANDLE(hwndOwner, WND)) && IS_VALID_WRITE_PTR(pdwOutFlags, DWORD) && (IS_FLAG_CLEAR(dwInFlags, RLI_IFL_UPDATE) || IS_VALID_WRITE_PTR(ppliUpdated, PLINKINFO)) && EVAL(IS_FLAG_CLEAR(dwInFlags, RLI_IFL_TEMPORARY) || IS_FLAG_SET(dwInFlags, RLI_IFL_CONNECT))) #endif { DWORD dwTempFlags; *pdwOutFlags = 0; bResult = ResolveILinkInfo((PCILINKINFO)pcli, pszResolvedPathBuf, MAX_PATH_LEN, dwInFlags, hwndOwner, &dwTempFlags); if (bResult) { *pdwOutFlags |= dwTempFlags; if (IS_FLAG_SET(dwInFlags, RLI_IFL_UPDATE)) { bResult = UpdateILinkInfo((PCILINKINFO)pcli, pszResolvedPathBuf, &dwTempFlags, (PILINKINFO *)ppliUpdated); if (bResult) *pdwOutFlags |= dwTempFlags; } } #ifdef DEBUG TRACE_OUT((TEXT("ResolveLinkInfo(): flags %#08lx, given LinkInfo:"), dwInFlags)); DumpILinkInfo((PCILINKINFO)pcli); if (bResult) { TRACE_OUT((TEXT("ResolveLinkInfo(): Resolved path %s with flags %#08lx."), pszResolvedPathBuf, *pdwOutFlags)); if (IS_FLAG_SET(*pdwOutFlags, RLI_OFL_UPDATED)) { ASSERT(IS_FLAG_SET(dwInFlags, RLI_IFL_UPDATE)); TRACE_OUT((TEXT("UpdateLinkInfo(): updated LinkInfo:"))); DumpILinkInfo(*(PILINKINFO *)ppliUpdated); } else { if (IS_FLAG_SET(dwInFlags, RLI_IFL_UPDATE)) TRACE_OUT((TEXT("UpdateLinkInfo(): No update required."))); else TRACE_OUT((TEXT("UpdateLinkInfo(): No update requested."))); } } else WARNING_OUT((TEXT("ResolveLinkInfo(): Referent's volume is unavailable."))); #endif } #ifdef EXPV else { SetLastError(ERROR_INVALID_PARAMETER); bResult = FALSE; } #endif ASSERT(! bResult || (FLAGS_ARE_VALID(*pdwOutFlags, ALL_RLI_OFLAGS) && EVAL(IsCanonicalPath(pszResolvedPathBuf)) && EVAL(! (IS_FLAG_CLEAR(dwInFlags, RLI_IFL_UPDATE) && IS_FLAG_SET(*pdwOutFlags, RLI_OFL_UPDATED))) && (IS_FLAG_CLEAR(*pdwOutFlags, RLI_OFL_UPDATED) || IS_VALID_STRUCT_PTR(*ppliUpdated, CLINKINFO)))); DebugExitBOOL(ResolveLinkInfo, bResult); return(bResult); } #ifdef UNICODE LINKINFOAPI BOOL WINAPI ResolveLinkInfoA(PCLINKINFO pcli, LPSTR pszResolvedPathBuf, DWORD dwInFlags, HWND hwndOwner, PDWORD pdwOutFlags, PLINKINFO *ppliUpdated) { WCHAR szWideResolvedPathBuf[MAX_PATH]; BOOL fResolved; fResolved = ResolveLinkInfo(pcli, szWideResolvedPathBuf, dwInFlags, hwndOwner, pdwOutFlags, ppliUpdated); if ( fResolved ) { if (WideCharToMultiByte( CP_ACP, 0, szWideResolvedPathBuf, -1, pszResolvedPathBuf, MAX_PATH, NULL, NULL ) == 0) { return FALSE; } } return fResolved; } #endif /****************************************************************************** @doc LINKINFOAPI @func BOOL | DisconnectLinkInfo | Cancels a connection to a net resource established by a previous call to ResolveLinkInfo(). DisconnectLinkInfo() should only be called if RLI_OFL_DISCONNECT was set in *pdwOutFlags on return from ResolveLinkInfo() on the given LinkInfo structure, or its updated equivalent. @parm PCLINKINFO | pcli | A pointer to the LinkInfo structure whose connection is to be canceled. @rdesc If the function completed successfully, TRUE is returned. Otherwise, FALSE is returned. The reason for failure may be determined by calling GetLastError(). @xref ResolveLinkInfo ******************************************************************************/ LINKINFOAPI BOOL WINAPI DisconnectLinkInfo(PCLINKINFO pcli) { BOOL bResult; DebugEntry(DisconnectLinkInfo); #ifdef EXPV /* Verify parameters. */ if ( IS_VALID_STRUCT_PTR(pcli, CLINKINFO)) #endif { bResult = DisconnectILinkInfo((PCILINKINFO)pcli); } #ifdef EXPV else { SetLastError(ERROR_INVALID_PARAMETER); bResult = FALSE; } #endif DebugExitBOOL(DisconnectLinkInfo, bResult); return(bResult); } /****************************************************************************** @doc LINKINFOAPI @func BOOL | GetLinkInfoData | Retrieves a pointer to data in a LinkInfo structure. @parm PCLINKINFO | pcli | A pointer to the LinkInfo structure to retrieve data from. @parm LINKINFODATATYPE | lidt | The type of data to be retrieved from the LinkInfo structure. lidt may be one of the following values: @flag LIDT_VOLUME_SERIAL_NUMBER | *ppcvData is a PCDWORD that points to the LinkInfo structure's referent's volume's serial number. @flag LIDT_DRIVE_TYPE | *ppcvData is a PCUINT that points to the LinkInfo structure's referent's volume's host drive type. @flag LIDT_VOLUME_LABEL | *ppcvData is a PCSTR that points to the LinkInfo structure's referent's volume's label. @flag LIDT_LOCAL_BASE_PATH | *ppcvData is a PCSTR that points to the LinkInfo structure's referent's local base path. @flag LIDT_NET_RESOURCE | *ppcvData is a PCSTR that points to the LinkInfo structure's referent's parent network resource's name. @flag LIDT_COMMON_PATH_SUFFIX | *ppcvData is a PCSTR that points to the LinkInfo structure's referent's common path suffix. @rdesc If the function completed successfully, TRUE is returned, and *ppcvData is filled in with a pointer to the data requested from LinkInfo structure or NULL if that was a valid field, but empty. Otherwise, FALSE is returned, and the contents of *ppcvData are undefined. The reason for failure may be determined by calling GetLastError(). @comm A LinkInfo structure may only contain some of the LinkInfo data listed above. ******************************************************************************/ LINKINFOAPI BOOL WINAPI GetLinkInfoData(PCLINKINFO pcli, LINKINFODATATYPE lidt, PCVOID *ppcvData) { BOOL bResult; DebugEntry(GetLinkInfoData); #ifdef EXPV /* Verify parameters. */ if ( IS_VALID_STRUCT_PTR(pcli, CLINKINFO) && EVAL(IsValidLINKINFODATATYPE(lidt)) && IS_VALID_WRITE_PTR(ppcvData, PCVOID)) #endif { bResult = GetILinkInfoData((PCILINKINFO)pcli, lidt, ppcvData); } #ifdef EXPV else { SetLastError(ERROR_INVALID_PARAMETER); bResult = FALSE; } #endif ASSERT(!bResult || ((NULL == *ppcvData) || IS_VALID_READ_BUFFER_PTR(*ppcvData, LinkInfoData, 1))); DebugExitBOOL(GetLinkInfoData, bResult); return(bResult); } /****************************************************************************** @doc LINKINFOAPI @func BOOL | IsValidLinkInfo | Determines whether or not a LinkInfo structure is valid. @parm PCLINKINFO | pcli | A pointer to the LinkInfo structure to be checked for validity. @rdesc If the function completed successfully, TRUE is returned. Otherwise, FALSE is returned. ******************************************************************************/ // // IS_ALIGNMENT_MACHINE_ALIGNED_DWORD_CNT validates alignment only on // alignment machines. For machines that are not alignment-sensitive, // it declares that all values are aligned. // #ifdef ALIGNMENT_MACHINE #define IS_ALIGNMENT_MACHINE_ALIGNED_DWORD_CNT(x) IS_ALIGNED_DWORD_CNT(x) #else #define IS_ALIGNMENT_MACHINE_ALIGNED_DWORD_CNT(x) TRUE #endif LINKINFOAPI BOOL WINAPI IsValidLinkInfo(PCLINKINFO pcli) { BOOL bResult; PCILINKINFO pcili = (PCILINKINFO)pcli; DebugEntry(IsValidLinkInfo); // First make sure it's readable and not ridiculously small if ((pcli == NULL) || // read the ucbSize pcli->ucbSize < pcili->ucbHeaderSize || // header fits in buffer pcili->ucbHeaderSize < sizeof(ILINKINFOA)// smallest supported header ) { bResult = FALSE; goto exit; } // Make sure no field points outside our buffer if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_LOCAL_INFO_VALID)) { if (pcili->ucbVolumeIDOffset >= pcli->ucbSize || pcili->ucbLocalBasePathOffset >= pcli->ucbSize || pcili->ucbCommonPathSuffixOffset >= pcli->ucbSize || !IS_ALIGNMENT_MACHINE_ALIGNED_DWORD_CNT(pcili->ucbVolumeIDOffset)) { bResult = FALSE; goto exit; } } // The ucbCNRLinkOffset field is valid only sometimes if (IS_FLAG_SET(pcili->dwFlags, ILI_FL_REMOTE_INFO_VALID)) { if (pcili->ucbCNRLinkOffset >= pcli->ucbSize || !IS_ALIGNMENT_MACHINE_ALIGNED_DWORD_CNT(pcili->ucbCNRLinkOffset)) { bResult = FALSE; goto exit; } } // The UNICODE version has some more fields than the ANSI version. // Those UNICODE fields must be WORD-aligned. if (pcili->ucbHeaderSize >= sizeof(ILINKINFOW)) { if (pcili->ucbLocalBasePathOffsetW >= pcli->ucbSize || pcili->ucbCommonPathSuffixOffsetW >= pcli->ucbSize || !IS_ALIGNED_WORD_CNT(pcili->ucbLocalBasePathOffsetW) || !IS_ALIGNED_WORD_CNT(pcili->ucbCommonPathSuffixOffsetW)) { bResult = FALSE; goto exit; } } // Survived the validation ordeal! bResult = TRUE; exit:; DebugExitBOOL(IsValidLinkInfo, bResult); return(bResult); }