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.
1961 lines
52 KiB
1961 lines
52 KiB
/*
|
|
* cnrlink.c - CNRLink ADT module.
|
|
*/
|
|
|
|
|
|
/* Headers
|
|
**********/
|
|
|
|
#include "project.h"
|
|
#pragma hdrstop
|
|
|
|
#include "cnrlink.h"
|
|
#include "server.h"
|
|
|
|
|
|
/* Constants
|
|
************/
|
|
|
|
/* WNetUseConnection() flag combinations */
|
|
|
|
#define ALL_CONNECT_IN_FLAGS (CONNECT_UPDATE_PROFILE |\
|
|
CONNECT_UPDATE_RECENT |\
|
|
CONNECT_TEMPORARY |\
|
|
CONNECT_INTERACTIVE |\
|
|
CONNECT_PROMPT |\
|
|
CONNECT_REDIRECT)
|
|
|
|
#define ALL_CONNECT_OUT_FLAGS (CONNECT_REFCOUNT |\
|
|
CONNECT_LOCALDRIVE)
|
|
|
|
|
|
/* Macros
|
|
*********/
|
|
|
|
/* macros for accessing ICNRLINK data */
|
|
|
|
#define ICNRL_Remote_Name_PtrA(picnrl) \
|
|
((LPSTR)(((PBYTE)(picnrl)) + (picnrl)->ucbNetNameOffset))
|
|
|
|
#define ICNRL_Device_PtrA(picnrl) \
|
|
((LPSTR)(((PBYTE)(picnrl)) + (picnrl)->ucbDeviceOffset))
|
|
|
|
#define ICNRL_Remote_Name_PtrW(picnrl) \
|
|
((LPWSTR)(((PBYTE)(picnrl)) + (picnrl)->ucbNetNameOffsetW))
|
|
|
|
#define ICNRL_Device_PtrW(picnrl) \
|
|
((LPWSTR)(((PBYTE)(picnrl)) + (picnrl)->ucbDeviceOffsetW))
|
|
|
|
#define IS_ICNRL_ANSI(picnrl) \
|
|
((PBYTE)(picnrl) + ((PICNRLINKW)(picnrl))->ucbNetNameOffset) == (PBYTE)&(((PICNRLINKW)(picnrl))->ucbNetNameOffsetW)
|
|
|
|
#ifdef UNICODE
|
|
#define ICNRL_Remote_Name_Ptr(picnrl) ICNRL_Remote_Name_PtrW(picnrl)
|
|
#define ICNRL_Device_Ptr(picnrl) ICNRL_Device_PtrW(picnrl)
|
|
#else
|
|
#define ICNRL_Remote_Name_Ptr(picnrl) ICNRL_Remote_Name_PtrA(picnrl)
|
|
#define ICNRL_Device_Ptr(picnrl) ICNRL_Device_PtrA(picnrl)
|
|
#endif
|
|
|
|
/* Types
|
|
********/
|
|
|
|
/*
|
|
@doc INTERNAL
|
|
|
|
@enum ICNRLINKFLAGS | Internal CNRLink structure flags.
|
|
*/
|
|
|
|
typedef enum _icnrlinkflags
|
|
{
|
|
/*
|
|
@emem ICNRL_FL_VALID_DEVICE | If set, last redirected drive is valid. If
|
|
clear, last redirected drive is not valid.
|
|
*/
|
|
|
|
ICNRL_FL_VALID_DEVICE = 0x0001,
|
|
|
|
/*
|
|
@emem ICNRL_FL_VALID_NET_TYPE | If set, net type is valid. If clear, net
|
|
type is not valid.
|
|
*/
|
|
|
|
ICNRL_FL_VALID_NET_TYPE = 0x0002,
|
|
|
|
/* @emem ALL_ICNRL_FLAGS | All internal CNRLink structure flags. */
|
|
|
|
ALL_ICNRL_FLAGS = (ICNRL_FL_VALID_DEVICE |
|
|
ICNRL_FL_VALID_NET_TYPE)
|
|
}
|
|
ICNRLINKFLAGS;
|
|
|
|
/*
|
|
@doc INTERNAL
|
|
|
|
@struct ICNRLINK | Internal definition of relocatable connectable network
|
|
resource (CNR) link structure. An <t ILINKINFO> structure may contain an
|
|
ICNRLINK structure. An ICNRLINK structure consists of a header described as
|
|
below, followed by variable-length data.
|
|
*/
|
|
|
|
typedef struct _icnrlinkA
|
|
{
|
|
/*
|
|
@field UINT | ucbSize | Length of ICNRLINK structure in bytes, including
|
|
ucbSize field.
|
|
*/
|
|
|
|
UINT ucbSize;
|
|
|
|
/*
|
|
@field DWORD | dwFlags | A bit mask of flags from the <t ICNRLINKFLAGS>
|
|
enumeration.
|
|
*/
|
|
|
|
DWORD dwFlags;
|
|
|
|
/*
|
|
@field UINT | ucbNetNameOffset | Offset in bytes of CNR name string from
|
|
base of structure. The CNR name string may be passed to
|
|
WNetUseConnection() to add a connection to the CNR.<nl>
|
|
Example CNRLink name string: "\\\\fredbird\\work".
|
|
*/
|
|
|
|
UINT ucbNetNameOffset;
|
|
|
|
/*
|
|
@field UINT | ucbDeviceOffset | Offset in bytes of last redirected local
|
|
device string from base of structure. This field is only valid if
|
|
ICNRL_FL_VALID_DEVICE is set in dwFlags. The last redirected local
|
|
device string may be passed to WNetUseConnection() to add a redirected
|
|
device connection to the CNR.<nl>
|
|
Example last redirected local device string: "D:".
|
|
*/
|
|
|
|
UINT ucbDeviceOffset;
|
|
|
|
/*
|
|
@field DWORD | dwNetType | The network type as returned in a
|
|
NETINFOSTRUCT. This field is only valid if ICNRL_FL_VALID_NET_TYPE is
|
|
set in dwFlags. The net type is used to retrieve the host net resource's
|
|
host NP's name to use in calling WNetUseConnection().<nl>
|
|
Example net type: WNNC_NET_NETWARE.
|
|
*/
|
|
|
|
DWORD dwNetType;
|
|
}
|
|
ICNRLINKA;
|
|
DECLARE_STANDARD_TYPES(ICNRLINKA);
|
|
|
|
#ifdef UNICODE
|
|
typedef struct _icnrlinkW
|
|
{
|
|
/*
|
|
@field UINT | ucbSize | Length of ICNRLINK structure in bytes, including
|
|
ucbSize field.
|
|
*/
|
|
|
|
UINT ucbSize;
|
|
|
|
/*
|
|
@field DWORD | dwFlags | A bit mask of flags from the <t ICNRLINKFLAGS>
|
|
enumeration.
|
|
*/
|
|
|
|
DWORD dwFlags;
|
|
|
|
/*
|
|
@field UINT | ucbNetNameOffset | Offset in bytes of CNR name string from
|
|
base of structure. The CNR name string may be passed to
|
|
WNetUseConnection() to add a connection to the CNR.<nl>
|
|
Example CNRLink name string: "\\\\fredbird\\work".
|
|
*/
|
|
|
|
UINT ucbNetNameOffset;
|
|
|
|
/*
|
|
@field UINT | ucbDeviceOffset | Offset in bytes of last redirected local
|
|
device string from base of structure. This field is only valid if
|
|
ICNRL_FL_VALID_DEVICE is set in dwFlags. The last redirected local
|
|
device string may be passed to WNetUseConnection() to add a redirected
|
|
device connection to the CNR.<nl>
|
|
Example last redirected local device string: "D:".
|
|
*/
|
|
|
|
UINT ucbDeviceOffset;
|
|
|
|
/*
|
|
@field DWORD | dwNetType | The network type as returned in a
|
|
NETINFOSTRUCT. This field is only valid if ICNRL_FL_VALID_NET_TYPE is
|
|
set in dwFlags. The net type is used to retrieve the host net resource's
|
|
host NP's name to use in calling WNetUseConnection().<nl>
|
|
Example net type: WNNC_NET_NETWARE.
|
|
*/
|
|
|
|
DWORD dwNetType;
|
|
|
|
/*
|
|
These members are for storing the unicode version of the strings
|
|
*/
|
|
UINT ucbNetNameOffsetW;
|
|
UINT ucbDeviceOffsetW;
|
|
}
|
|
ICNRLINKW;
|
|
DECLARE_STANDARD_TYPES(ICNRLINKW);
|
|
#endif
|
|
|
|
#ifdef UNICODE
|
|
#define ICNRLINK ICNRLINKW
|
|
#define PICNRLINK PICNRLINKW
|
|
#define CICNRLINK CICNRLINKW
|
|
#define PCICNRLINK PCICNRLINKW
|
|
#else
|
|
#define ICNRLINK ICNRLINKA
|
|
#define PICNRLINK PICNRLINKA
|
|
#define CICNRLINK CICNRLINKA
|
|
#define PCICNRLINK PCICNRLINKA
|
|
#endif
|
|
|
|
/* Exported from MPR.DLL, but not in winnetwk.h
|
|
*/
|
|
#ifdef UNICODE
|
|
DWORD APIENTRY WNetGetResourceInformationW (LPNETRESOURCE lpNetResource, LPVOID lpBuffer, LPDWORD cbBuffer, LPTSTR * lplpSystem);
|
|
#define WNetGetResourceInformation WNetGetResourceInformationW
|
|
#else
|
|
DWORD APIENTRY WNetGetResourceInformationA (LPNETRESOURCE lpNetResource, LPVOID lpBuffer, LPDWORD cbBuffer, LPTSTR * lplpSystem);
|
|
#define WNetGetResourceInformation WNetGetResourceInformationA
|
|
#endif
|
|
|
|
|
|
/***************************** Private Functions *****************************/
|
|
|
|
/* Module Prototypes
|
|
********************/
|
|
|
|
PRIVATE_CODE BOOL GetNetPathFromLocalPath(LPCTSTR, LPTSTR, LPCTSTR *, PBOOL, PDWORD);
|
|
PRIVATE_CODE BOOL UnifyICNRLinkInfo(LPCTSTR, DWORD, LPCTSTR, DWORD, PICNRLINK *, PUINT);
|
|
PRIVATE_CODE BOOL GetNetType(LPCTSTR, PDWORD);
|
|
PRIVATE_CODE BOOL GetNetProviderName(PCICNRLINK, LPTSTR);
|
|
PRIVATE_CODE COMPARISONRESULT CompareNetNames(LPCTSTR, LPCTSTR);
|
|
PRIVATE_CODE BOOL SearchForRedirectedConnection(PCICNRLINK, LPTSTR, int);
|
|
|
|
#if defined(DEBUG) || defined (VSTF)
|
|
|
|
PRIVATE_CODE BOOL IsValidDevice(LPCTSTR);
|
|
PRIVATE_CODE BOOL IsValidNetType(DWORD);
|
|
PRIVATE_CODE BOOL IsValidPCICNRLINK(PCICNRLINK);
|
|
|
|
#endif
|
|
|
|
#if defined(DEBUG)
|
|
|
|
PRIVATE_CODE BOOL IsValidNetProviderName(LPCTSTR);
|
|
|
|
#endif
|
|
|
|
#if 0
|
|
|
|
DWORD APIENTRY
|
|
WNetGetNetworkInformationW(
|
|
LPCWSTR lpProvider,
|
|
LPNETINFOSTRUCT lpNetInfoStruct
|
|
)
|
|
{
|
|
if (wcsicmp(lpProvider, L"Microsoft Windows Network") == 0)
|
|
{
|
|
lpNetInfoStruct->wNetType = (WORD)WNNC_NET_LANMAN;
|
|
return ERROR_SUCCESS;
|
|
}
|
|
else if (wcsicmp(lpProvider, L"Novell Network") == 0)
|
|
{
|
|
lpNetInfoStruct->wNetType = (WORD)WNNC_NET_NETWARE;
|
|
return ERROR_SUCCESS;
|
|
}
|
|
else
|
|
{
|
|
return ERROR_NOT_SUPPORTED;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
** GetNetPathFromLocalPath()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE BOOL GetNetPathFromLocalPath(LPCTSTR pcszLocalPath,
|
|
LPTSTR pszNetNameBuf,
|
|
LPCTSTR *ppcszCommonPathSuffix,
|
|
PBOOL pbIsShared, PDWORD pdwNetType)
|
|
{
|
|
BOOL bResult = TRUE;
|
|
PCSERVERVTABLE pcsvt;
|
|
|
|
ASSERT(IsDrivePath(pcszLocalPath));
|
|
ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszNetNameBuf, STR, MAX_PATH_LEN));
|
|
ASSERT(IS_VALID_WRITE_PTR(ppcszCommonPathSuffix, LPCTSTR));
|
|
ASSERT(IS_VALID_WRITE_PTR(pbIsShared, BOOL));
|
|
ASSERT(IS_VALID_WRITE_PTR(pdwNetType, DWORD));
|
|
|
|
*pbIsShared = FALSE;
|
|
|
|
if (GetServerVTable(&pcsvt))
|
|
{
|
|
TCHAR rgchSharedPath[MAX_PATH_LEN];
|
|
|
|
ASSERT(lstrlen(pcszLocalPath) < ARRAYSIZE(rgchSharedPath));
|
|
lstrcpyn(rgchSharedPath, pcszLocalPath, ARRAYSIZE(rgchSharedPath));
|
|
|
|
FOREVER
|
|
{
|
|
if ((pcsvt->GetNetResourceFromLocalPath)(rgchSharedPath,
|
|
pszNetNameBuf, MAX_PATH_LEN,
|
|
pdwNetType))
|
|
{
|
|
ASSERT(lstrlen(pszNetNameBuf) < MAX_PATH_LEN);
|
|
|
|
/* Determine common path suffix. */
|
|
|
|
*ppcszCommonPathSuffix = pcszLocalPath + lstrlen(rgchSharedPath);
|
|
|
|
/* Skip any leading slash. */
|
|
|
|
if (IS_SLASH(**ppcszCommonPathSuffix))
|
|
*ppcszCommonPathSuffix = CharNext(*ppcszCommonPathSuffix);
|
|
|
|
ASSERT(! IS_SLASH(**ppcszCommonPathSuffix));
|
|
|
|
// if it is terminated with a $ it is a hidden share, in that
|
|
// case don't consider this shared
|
|
*pbIsShared = pszNetNameBuf[lstrlen(pszNetNameBuf) -1] != TEXT('$');
|
|
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
if (! DeleteLastDrivePathElement(rgchSharedPath))
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
ASSERT(! bResult ||
|
|
! *pbIsShared ||
|
|
(EVAL(IsUNCPath(pszNetNameBuf)) &&
|
|
IS_VALID_STRING_PTR(*ppcszCommonPathSuffix, CSTR) &&
|
|
EVAL(*ppcszCommonPathSuffix >= pcszLocalPath) &&
|
|
EVAL(IsStringContained(pcszLocalPath, *ppcszCommonPathSuffix)) &&
|
|
EVAL(IsValidNetType(*pdwNetType))));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
/*
|
|
** UnifyICNRLinkInfo()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE BOOL UnifyICNRLinkInfo(LPCTSTR pcszNetName, DWORD dwFlags,
|
|
LPCTSTR pcszDevice, DWORD dwNetType,
|
|
PICNRLINK *ppicnrl, PUINT pucbICNRLinkLen)
|
|
{
|
|
BOOL bResult;
|
|
UINT ucbDataOffset;
|
|
#ifdef UNICODE
|
|
BOOL bUnicode;
|
|
UINT cchChars;
|
|
CHAR szAnsiNetName[MAX_PATH];
|
|
CHAR szAnsiDevice[MAX_PATH];
|
|
UINT cbAnsiNetName;
|
|
UINT cbWideNetName;
|
|
UINT cbAnsiDevice;
|
|
UINT cbWideDevice;
|
|
UINT cbChars;
|
|
#endif
|
|
|
|
ASSERT(IsUNCPath(pcszNetName));
|
|
ASSERT(FLAGS_ARE_VALID(dwFlags, ALL_ICNRL_FLAGS));
|
|
ASSERT(IS_FLAG_CLEAR(dwFlags, ICNRL_FL_VALID_DEVICE) ||
|
|
IsValidDevice(pcszDevice));
|
|
ASSERT(IS_FLAG_CLEAR(dwFlags, ICNRL_FL_VALID_NET_TYPE) ||
|
|
IsValidNetType(dwNetType));
|
|
ASSERT(IS_VALID_WRITE_PTR(ppicnrl, PCNRLINK));
|
|
ASSERT(IS_VALID_WRITE_PTR(pucbICNRLinkLen, UINT));
|
|
|
|
#ifdef UNICODE
|
|
bUnicode = FALSE;
|
|
|
|
cbAnsiNetName = WideCharToMultiByte(CP_ACP, 0,
|
|
pcszNetName, -1,
|
|
szAnsiNetName, ARRAYSIZE(szAnsiNetName),
|
|
0, 0);
|
|
if ( cbAnsiNetName == 0 )
|
|
{
|
|
bUnicode = FALSE;
|
|
}
|
|
else
|
|
{
|
|
WCHAR szWideNetName[MAX_PATH];
|
|
|
|
cbChars = MultiByteToWideChar(CP_ACP, 0,
|
|
szAnsiNetName, -1,
|
|
szWideNetName, ARRAYSIZE(szWideNetName));
|
|
if ( cbChars == 0 || lstrcmp(pcszNetName,szWideNetName) != 0 )
|
|
{
|
|
bUnicode = TRUE;
|
|
}
|
|
}
|
|
|
|
if (IS_FLAG_SET(dwFlags, ICNRL_FL_VALID_DEVICE))
|
|
{
|
|
cbAnsiDevice = WideCharToMultiByte(CP_ACP, 0,
|
|
pcszDevice, -1,
|
|
szAnsiDevice, ARRAYSIZE(szAnsiDevice),
|
|
0, 0);
|
|
if ( cbAnsiDevice == 0 )
|
|
{
|
|
bUnicode = FALSE;
|
|
}
|
|
else
|
|
{
|
|
WCHAR szWideDevice[MAX_PATH];
|
|
|
|
cchChars = MultiByteToWideChar(CP_ACP, 0,
|
|
szAnsiDevice, -1,
|
|
szWideDevice, ARRAYSIZE(szWideDevice));
|
|
if ( cchChars == 0 || lstrcmp(pcszDevice,szWideDevice) != 0 )
|
|
{
|
|
bUnicode = TRUE;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
cbAnsiDevice = 0;
|
|
}
|
|
|
|
if ( bUnicode )
|
|
{
|
|
ucbDataOffset = SIZEOF(ICNRLINKW);
|
|
|
|
/* (+ 1) for null terminator. */
|
|
cbWideNetName = (lstrlen(pcszNetName) + 1) * sizeof(TCHAR);
|
|
|
|
if (IS_FLAG_SET(dwFlags, ICNRL_FL_VALID_DEVICE))
|
|
cbWideDevice = (lstrlen(pcszDevice) + 1) * sizeof(TCHAR);
|
|
else
|
|
cbWideDevice = 0;
|
|
|
|
}
|
|
else
|
|
{
|
|
ucbDataOffset = SIZEOF(ICNRLINKA);
|
|
|
|
cbWideNetName = 0;
|
|
cbWideDevice = 0;
|
|
}
|
|
|
|
*pucbICNRLinkLen = ucbDataOffset +
|
|
cbAnsiNetName +
|
|
cbAnsiDevice;
|
|
if ( bUnicode )
|
|
{
|
|
*pucbICNRLinkLen = ALIGN_WORD_CNT(*pucbICNRLinkLen) +
|
|
cbWideNetName +
|
|
cbWideDevice;
|
|
}
|
|
|
|
#else
|
|
|
|
/* Assume we won't overflow *pucbICNRLinkLen here. */
|
|
|
|
/* (+ 1) for null terminator. */
|
|
|
|
*pucbICNRLinkLen = SIZEOF(**ppicnrl) +
|
|
(lstrlen(pcszNetName) + 1) * SIZEOF(TCHAR);
|
|
|
|
if (IS_FLAG_SET(dwFlags, ICNRL_FL_VALID_DEVICE))
|
|
/* (+ 1) for null terminator. */
|
|
*pucbICNRLinkLen += (lstrlen(pcszDevice) + 1) * SIZEOF(TCHAR);
|
|
|
|
ucbDataOffset = SIZEOF(ICNRLINKA);
|
|
#endif
|
|
|
|
bResult = AllocateMemory(*pucbICNRLinkLen, ppicnrl);
|
|
|
|
if (bResult)
|
|
{
|
|
(*ppicnrl)->ucbSize = *pucbICNRLinkLen;
|
|
(*ppicnrl)->dwFlags = dwFlags;
|
|
|
|
if (IS_FLAG_SET(dwFlags, ICNRL_FL_VALID_NET_TYPE))
|
|
(*ppicnrl)->dwNetType = dwNetType;
|
|
else
|
|
(*ppicnrl)->dwNetType = 0;
|
|
|
|
/* Append remote name. */
|
|
|
|
(*ppicnrl)->ucbNetNameOffset = ucbDataOffset;
|
|
|
|
// lstrcpy: Enough memory is allocated above to hold the strings
|
|
// so no need to bound it here.
|
|
#ifdef UNICODE
|
|
lstrcpyA(ICNRL_Remote_Name_PtrA(*ppicnrl), szAnsiNetName);
|
|
ucbDataOffset += cbAnsiNetName;
|
|
|
|
if (IS_FLAG_SET(dwFlags, ICNRL_FL_VALID_DEVICE))
|
|
{
|
|
/* Append device name. */
|
|
|
|
(*ppicnrl)->ucbDeviceOffset = ucbDataOffset;
|
|
lstrcpyA(ICNRL_Device_PtrA(*ppicnrl), szAnsiDevice);
|
|
|
|
ucbDataOffset += cbAnsiDevice;
|
|
}
|
|
else
|
|
{
|
|
(*ppicnrl)->ucbDeviceOffset = 0;
|
|
}
|
|
|
|
if ( bUnicode )
|
|
{
|
|
ucbDataOffset = ALIGN_WORD_CNT(ucbDataOffset);
|
|
|
|
(*ppicnrl)->ucbNetNameOffsetW = ucbDataOffset;
|
|
|
|
lstrcpy(ICNRL_Remote_Name_PtrW(*ppicnrl), pcszNetName);
|
|
ucbDataOffset += cbWideNetName;
|
|
|
|
if (IS_FLAG_SET(dwFlags, ICNRL_FL_VALID_DEVICE))
|
|
{
|
|
/* Append device name. */
|
|
|
|
(*ppicnrl)->ucbDeviceOffsetW = ucbDataOffset;
|
|
lstrcpy(ICNRL_Device_Ptr(*ppicnrl), pcszDevice);
|
|
|
|
/* (+ 1) for null terminator. */
|
|
ucbDataOffset += cbWideDevice;
|
|
}
|
|
else
|
|
{
|
|
(*ppicnrl)->ucbDeviceOffsetW = 0;
|
|
}
|
|
|
|
}
|
|
#else
|
|
lstrcpy(ICNRL_Remote_Name_Ptr(*ppicnrl), pcszNetName);
|
|
/* (+ 1) for null terminator. */
|
|
ucbDataOffset += lstrlen(pcszNetName) + 1;
|
|
|
|
if (IS_FLAG_SET(dwFlags, ICNRL_FL_VALID_DEVICE))
|
|
{
|
|
/* Append device name. */
|
|
|
|
(*ppicnrl)->ucbDeviceOffset = ucbDataOffset;
|
|
lstrcpy(ICNRL_Device_Ptr(*ppicnrl), pcszDevice);
|
|
#ifdef DEBUG
|
|
/* (+ 1) for null terminator. */
|
|
ucbDataOffset += (lstrlen(pcszDevice) + 1) * SIZEOF(TCHAR);
|
|
#endif
|
|
}
|
|
else
|
|
(*ppicnrl)->ucbDeviceOffset = 0;
|
|
#endif
|
|
|
|
/* Do all the calculated lengths match? */
|
|
|
|
ASSERT(ucbDataOffset == (*ppicnrl)->ucbSize);
|
|
ASSERT(ucbDataOffset == *pucbICNRLinkLen);
|
|
}
|
|
|
|
ASSERT(! bResult ||
|
|
(IS_VALID_STRUCT_PTR(*ppicnrl, CICNRLINK) &&
|
|
EVAL(*pucbICNRLinkLen == GetCNRLinkLen((PCCNRLINK)*ppicnrl))));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
/*
|
|
** GetNetType()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE BOOL GetNetType(LPCTSTR pcszCNRName, PDWORD pdwNetType)
|
|
{
|
|
BOOL bResult = FALSE;
|
|
NETRESOURCE nrIn;
|
|
NETRESOURCEBUF nrbufOut;
|
|
DWORD dwcbBufLen = SIZEOF(nrbufOut);
|
|
LPTSTR pszFileSysPath;
|
|
DWORD dwNetResult;
|
|
#ifdef DEBUG
|
|
DWORD dwcmsTicks;
|
|
#endif
|
|
|
|
ASSERT(IsValidCNRName(pcszCNRName));
|
|
ASSERT(IS_VALID_WRITE_PTR(pdwNetType, DWORD));
|
|
|
|
/* RAIDRAID: (15691) We only support disk resource connections here. */
|
|
|
|
ZeroMemory(&nrIn, SIZEOF(nrIn));
|
|
nrIn.lpRemoteName = (LPTSTR)pcszCNRName;
|
|
nrIn.dwType = RESOURCETYPE_DISK;
|
|
|
|
#ifdef DEBUG
|
|
dwcmsTicks = GetTickCount();
|
|
#endif
|
|
|
|
dwNetResult = WNetGetResourceInformation(&nrIn, &(nrbufOut.rgbyte),
|
|
&dwcbBufLen, &pszFileSysPath);
|
|
|
|
#ifdef DEBUG
|
|
|
|
dwcmsTicks = GetTickCount() - dwcmsTicks;
|
|
|
|
TRACE_OUT((TEXT("GetRemotePathInfo(): WNetGetResourceInformation() on net resource %s took %lu.%03lu seconds."),
|
|
pcszCNRName,
|
|
(dwcmsTicks / 1000),
|
|
(dwcmsTicks % 1000)));
|
|
|
|
#endif
|
|
|
|
if (dwNetResult == ERROR_SUCCESS)
|
|
{
|
|
if (nrbufOut.nr.lpProvider)
|
|
{
|
|
NETINFOSTRUCT nis;
|
|
|
|
ASSERT(IS_VALID_STRING_PTR(nrbufOut.nr.lpProvider, STR));
|
|
|
|
nis.cbStructure = SIZEOF(nis);
|
|
|
|
dwNetResult = WNetGetNetworkInformation(nrbufOut.nr.lpProvider, &nis);
|
|
|
|
if (dwNetResult == ERROR_SUCCESS)
|
|
{
|
|
*pdwNetType = ((nis.wNetType) << 16);
|
|
bResult = TRUE;
|
|
|
|
TRACE_OUT((TEXT("GetNetType(): Net type for CNR %s is %#08lx."),
|
|
pcszCNRName,
|
|
*pdwNetType));
|
|
}
|
|
else
|
|
WARNING_OUT((TEXT("GetNetType(): WNetGetNetworkInformation() failed for %s NP, returning %lu."),
|
|
nrbufOut.nr.lpProvider,
|
|
dwNetResult));
|
|
}
|
|
else
|
|
WARNING_OUT((TEXT("GetNetType(): WNetGetResourceInformation() was unable to determine the NP for CNR %s."),
|
|
pcszCNRName));
|
|
}
|
|
else
|
|
WARNING_OUT((TEXT("GetNetType(): WNetGetResourceInformation() failed for CNR %s, returning %lu."),
|
|
pcszCNRName,
|
|
dwNetResult));
|
|
|
|
ASSERT(! bResult ||
|
|
IsValidNetType(*pdwNetType));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
/*
|
|
** GetNetProviderName()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE BOOL GetNetProviderName(PCICNRLINK pcicnrl, LPTSTR pszNPNameBuf)
|
|
{
|
|
BOOL bResult = FALSE;
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(pcicnrl, CICNRLINK));
|
|
ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszNPNameBuf, STR, MAX_PATH_LEN));
|
|
|
|
if (IS_FLAG_SET(pcicnrl->dwFlags, ICNRL_FL_VALID_NET_TYPE))
|
|
{
|
|
DWORD dwcbNPNameBufLen;
|
|
DWORD dwNetResult;
|
|
|
|
dwcbNPNameBufLen = MAX_PATH_LEN;
|
|
|
|
dwNetResult = WNetGetProviderName(pcicnrl->dwNetType, pszNPNameBuf,
|
|
&dwcbNPNameBufLen);
|
|
|
|
if (dwNetResult == ERROR_SUCCESS)
|
|
{
|
|
bResult = TRUE;
|
|
|
|
#ifdef UNICODE
|
|
//
|
|
// Unicode builds need to accept both ansi and unicode ICNRLINK structures.
|
|
// Note the use of '%S' (upper case). This will accept an ANSI string
|
|
// in a UNICODE build environment.
|
|
//
|
|
if (IS_ICNRL_ANSI(pcicnrl))
|
|
TRACE_OUT((TEXT("GetNetProviderName(): NP for CNR %S is %s."),
|
|
ICNRL_Remote_Name_PtrA(pcicnrl),
|
|
pszNPNameBuf));
|
|
else
|
|
#endif
|
|
TRACE_OUT((TEXT("GetNetProviderName(): NP for CNR %s is %s."),
|
|
ICNRL_Remote_Name_Ptr(pcicnrl),
|
|
pszNPNameBuf));
|
|
}
|
|
else
|
|
WARNING_OUT((TEXT("GetNetProviderName(): WNetGetProviderName() failed for CNR %s's net type %#08lx, returning %lu."),
|
|
TEXT("<Remote Name>"), // ICNRL_Remote_Name_Ptr(pcicnrl),
|
|
pcicnrl->dwNetType,
|
|
dwNetResult));
|
|
}
|
|
else
|
|
WARNING_OUT((TEXT("GetNetProviderName(): Net type for CNR %s is not known. Unable to determine NP name."),
|
|
TEXT("<Remote Name>"))); // ICNRL_Remote_Name_Ptr(pcicnrl)));
|
|
|
|
ASSERT(! bResult ||
|
|
IsValidNetProviderName(pszNPNameBuf));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
/*
|
|
** CompareNetNames()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE COMPARISONRESULT CompareNetNames(LPCTSTR pcszFirstNetName,
|
|
LPCTSTR pcszSecondNetName)
|
|
{
|
|
ASSERT(IS_VALID_STRING_PTR(pcszFirstNetName, CSTR));
|
|
ASSERT(IS_VALID_STRING_PTR(pcszSecondNetName, CSTR));
|
|
|
|
return(MapIntToComparisonResult(lstrcmp(pcszFirstNetName,
|
|
pcszSecondNetName)));
|
|
}
|
|
|
|
|
|
/*
|
|
** SearchForRedirectedConnection()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE BOOL SearchForRedirectedConnection(PCICNRLINK pcicnrl,
|
|
LPTSTR pszRootPathBuf,
|
|
int cchMax)
|
|
{
|
|
BOOL bResult = FALSE;
|
|
HANDLE henum;
|
|
DWORD dwNetResult;
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(pcicnrl, CICNRLINK));
|
|
ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszRootPathBuf, STR, MAX_PATH_LEN));
|
|
|
|
#ifdef DEBUG
|
|
#ifdef UNICODE
|
|
{
|
|
LPWSTR pszWideNetName;
|
|
WCHAR szWideNetName[MAX_PATH];
|
|
|
|
if (IS_ICNRL_ANSI(pcicnrl))
|
|
{
|
|
pszWideNetName = szWideNetName;
|
|
|
|
MultiByteToWideChar(CP_ACP, 0,
|
|
ICNRL_Remote_Name_PtrA(pcicnrl), -1,
|
|
szWideNetName, ARRAYSIZE(szWideNetName));
|
|
} else {
|
|
pszWideNetName = ICNRL_Remote_Name_PtrW(pcicnrl);
|
|
}
|
|
|
|
WARNING_OUT((TEXT("SearchForRedirectedConnection(): Enumerating local connections searching for redirected connection to CNR \"%s\"."),
|
|
pszWideNetName));
|
|
}
|
|
#else
|
|
WARNING_OUT((TEXT("SearchForRedirectedConnection(): Enumerating local connections searching for redirected connection to CNR \"%s\"."),
|
|
ICNRL_Remote_Name_Ptr(pcicnrl)));
|
|
#endif
|
|
#endif
|
|
|
|
/* RAIDRAID: (15691) We only support container resources here. */
|
|
|
|
dwNetResult = WNetOpenEnum(RESOURCE_CONNECTED, RESOURCETYPE_DISK,
|
|
RESOURCEUSAGE_CONTAINER | RESOURCEUSAGE_ATTACHED,
|
|
NULL, &henum);
|
|
|
|
if (dwNetResult == WN_SUCCESS)
|
|
{
|
|
DWORD dwc = 1;
|
|
NETRESOURCEBUF nrbuf;
|
|
DWORD dwcbBufLen = SIZEOF(nrbuf);
|
|
|
|
while ((dwNetResult = WNetEnumResource(henum, &dwc, &(nrbuf.rgbyte),
|
|
&dwcbBufLen))
|
|
== WN_SUCCESS)
|
|
{
|
|
/* Is this a redirected connection? */
|
|
|
|
if (nrbuf.nr.lpRemoteName != NULL)
|
|
{
|
|
if (nrbuf.nr.lpLocalName != NULL)
|
|
{
|
|
/* Yes. Is it a redirected connection to the desired CNR? */
|
|
|
|
#ifdef UNICODE
|
|
WCHAR szWideNetName[MAX_PATH];
|
|
LPWSTR pszWideNetName;
|
|
|
|
if (IS_ICNRL_ANSI(pcicnrl))
|
|
{
|
|
pszWideNetName = szWideNetName;
|
|
MultiByteToWideChar(CP_ACP, 0,
|
|
ICNRL_Remote_Name_PtrA(pcicnrl), -1,
|
|
szWideNetName, ARRAYSIZE(szWideNetName));
|
|
}
|
|
else
|
|
{
|
|
pszWideNetName = ICNRL_Remote_Name_Ptr(pcicnrl);
|
|
}
|
|
if (CompareNetNames(pszWideNetName,
|
|
nrbuf.nr.lpRemoteName)
|
|
== CR_EQUAL)
|
|
#else
|
|
if (CompareNetNames(ICNRL_Remote_Name_Ptr(pcicnrl),
|
|
nrbuf.nr.lpRemoteName)
|
|
== CR_EQUAL)
|
|
#endif
|
|
{
|
|
/* Yes. */
|
|
|
|
ASSERT(lstrlen(nrbuf.nr.lpLocalName) < MAX_PATH_LEN);
|
|
|
|
lstrcpyn(pszRootPathBuf, nrbuf.nr.lpLocalName, cchMax);
|
|
bResult = TRUE;
|
|
|
|
TRACE_OUT((TEXT("SearchForRedirectedConnection(): Found CNR \"%s\" connected to %s."),
|
|
nrbuf.nr.lpRemoteName,
|
|
pszRootPathBuf));
|
|
|
|
break;
|
|
}
|
|
else
|
|
/* No. */
|
|
TRACE_OUT((TEXT("SearchForRedirectedConnection(): Skipping unmatched enumerated connection to CNR \"%s\" on %s."),
|
|
nrbuf.nr.lpRemoteName,
|
|
nrbuf.nr.lpLocalName));
|
|
}
|
|
else
|
|
/* No. */
|
|
TRACE_OUT((TEXT("SearchForRedirectedConnection(): Skipping enumerated deviceless connection to CNR \"%s\"."),
|
|
nrbuf.nr.lpRemoteName));
|
|
}
|
|
else
|
|
WARNING_OUT((TEXT("SearchForRedirectedConnection(): Skipping enumerated connection with no CNR name.")));
|
|
}
|
|
|
|
if (! bResult && dwNetResult != WN_NO_MORE_ENTRIES)
|
|
WARNING_OUT((TEXT("SearchForRedirectedConnection(): WNetEnumResource() failed, returning %lu."),
|
|
dwNetResult));
|
|
|
|
dwNetResult = WNetCloseEnum(henum);
|
|
|
|
if (dwNetResult != WN_SUCCESS)
|
|
WARNING_OUT((TEXT("SearchForRedirectedConnection(): WNetCloseEnum() failed, returning %lu."),
|
|
dwNetResult));
|
|
}
|
|
else
|
|
WARNING_OUT((TEXT("SearchForRedirectedConnection(): WNetOpenEnum() failed, returning %lu."),
|
|
dwNetResult));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
#if defined(DEBUG) || defined (VSTF)
|
|
|
|
/*
|
|
** IsValidDevice()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE BOOL IsValidDevice(LPCTSTR pcszDevice)
|
|
{
|
|
/* Any valid string < MAX_PATH_LEN bytes long is a valid device name. */
|
|
|
|
return(IS_VALID_STRING_PTR(pcszDevice, CSTR) &&
|
|
EVAL(lstrlen(pcszDevice) < MAX_PATH_LEN));
|
|
}
|
|
|
|
|
|
/*
|
|
** IsValidNetType()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE BOOL IsValidNetType(DWORD dwNetType)
|
|
{
|
|
BOOL bResult;
|
|
|
|
switch (dwNetType & 0xffff0000)
|
|
{
|
|
default:
|
|
WARNING_OUT((TEXT("IsValidNetType(): Unexpected net type %#08lx is neither NetWare nor LANMan."),
|
|
dwNetType));
|
|
/* Fall through... */
|
|
|
|
case WNNC_NET_LANMAN:
|
|
case WNNC_NET_NETWARE:
|
|
bResult = TRUE;
|
|
break;
|
|
|
|
}
|
|
|
|
if (dwNetType & 0x0000ffff)
|
|
WARNING_OUT((TEXT("IsValidNetType(): Low word of net type %#08lx is non-zero."),
|
|
dwNetType));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
/*
|
|
** IsValidPCICNRLINK()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE BOOL IsValidPCICNRLINK(PCICNRLINK pcicnrl)
|
|
{
|
|
BOOL bResult;
|
|
|
|
if (IS_VALID_READ_PTR(pcicnrl, CICNRLINK) &&
|
|
IS_VALID_READ_BUFFER_PTR(pcicnrl, CICNRLINK, pcicnrl->ucbSize) &&
|
|
FLAGS_ARE_VALID(pcicnrl->dwFlags, ALL_ICNRL_FLAGS) &&
|
|
EVAL(IsValidCNRName(ICNRL_Remote_Name_Ptr(pcicnrl))) &&
|
|
EVAL(IsContained(pcicnrl, pcicnrl->ucbSize,
|
|
ICNRL_Remote_Name_PtrA(pcicnrl),
|
|
lstrlenA(ICNRL_Remote_Name_PtrA(pcicnrl)))) &&
|
|
(IS_FLAG_CLEAR(pcicnrl->dwFlags, ICNRL_FL_VALID_NET_TYPE) ||
|
|
EVAL(IsValidNetType(pcicnrl->dwNetType))))
|
|
{
|
|
if (IS_FLAG_CLEAR(pcicnrl->dwFlags, ICNRL_FL_VALID_DEVICE))
|
|
{
|
|
ASSERT(! pcicnrl->ucbDeviceOffset);
|
|
bResult = TRUE;
|
|
}
|
|
else
|
|
bResult = (EVAL(IsValidDevice(ICNRL_Device_Ptr(pcicnrl))) &&
|
|
EVAL(IsContained(pcicnrl, pcicnrl->ucbSize,
|
|
ICNRL_Device_PtrA(pcicnrl),
|
|
lstrlenA(ICNRL_Device_PtrA(pcicnrl)))));
|
|
}
|
|
else
|
|
bResult = FALSE;
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
#if defined(DEBUG)
|
|
|
|
/*
|
|
** IsValidNetProviderName()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PRIVATE_CODE BOOL IsValidNetProviderName(LPCTSTR pcszNetProvider)
|
|
{
|
|
/* Any string < MAX_PATH_LEN characters long is a valid NP name. */
|
|
|
|
return(IS_VALID_STRING_PTR(pcszNetProvider, CSTR) &&
|
|
lstrlen(pcszNetProvider) < MAX_PATH_LEN);
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
/****************************** Public Functions *****************************/
|
|
|
|
|
|
/*
|
|
** CreateLocalCNRLink()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
**
|
|
** If TRUE is returned:
|
|
** 1) *ppcnrl is only valid if *pucbCNRLinkLen > 0.
|
|
** 2) pszLocalBasePathBuf is valid.
|
|
** 3) *ppcszCommonPathSuffix is valid.
|
|
**
|
|
** If *pucbCNRLinkLen == 0, pszLocalBasePathBuf is a copy of pcszLocalPath, and
|
|
** *ppcszCommonPathSuffix points at the null terminator of pcszLocalPath.
|
|
**
|
|
** If *pucbCNRLinkLen > 0, pszLocalBasePathBuf is the closest shared local base
|
|
** path, and *ppcszCommonPathSuffix points at that path's suffix in
|
|
** pcszLocalPath.
|
|
*/
|
|
PUBLIC_CODE BOOL CreateLocalCNRLink(LPCTSTR pcszLocalPath, PCNRLINK *ppcnrl,
|
|
PUINT pucbCNRLinkLen,
|
|
LPTSTR pszLocalBasePathBuf,
|
|
int cchMax,
|
|
LPCTSTR *ppcszCommonPathSuffix)
|
|
{
|
|
BOOL bResult;
|
|
TCHAR rgchNetName[MAX_PATH_LEN];
|
|
BOOL bShared;
|
|
DWORD dwNetType;
|
|
|
|
ASSERT(IsDrivePath(pcszLocalPath));
|
|
ASSERT(IS_VALID_WRITE_PTR(ppcnrl, PCNRLINK));
|
|
ASSERT(IS_VALID_WRITE_PTR(pucbCNRLinkLen, UINT));
|
|
ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszLocalBasePathBuf, STR, MAX_PATH_LEN));
|
|
ASSERT(IS_VALID_WRITE_PTR(ppcszCommonPathSuffix, LPCTSTR));
|
|
|
|
bResult = GetNetPathFromLocalPath(pcszLocalPath, rgchNetName,
|
|
ppcszCommonPathSuffix, &bShared,
|
|
&dwNetType);
|
|
|
|
if (bResult)
|
|
{
|
|
if (bShared)
|
|
{
|
|
bResult = UnifyICNRLinkInfo(rgchNetName, ICNRL_FL_VALID_NET_TYPE,
|
|
NULL, dwNetType, (PICNRLINK *)ppcnrl,
|
|
pucbCNRLinkLen);
|
|
|
|
if (bResult)
|
|
{
|
|
UINT ucbLocalBasePathLen;
|
|
|
|
/* Copy local base path into output buffer. */
|
|
|
|
ASSERT(*ppcszCommonPathSuffix >= pcszLocalPath);
|
|
ucbLocalBasePathLen = (UINT)(*ppcszCommonPathSuffix - pcszLocalPath);
|
|
|
|
CopyMemory(pszLocalBasePathBuf, pcszLocalPath, ucbLocalBasePathLen * sizeof(TCHAR));
|
|
pszLocalBasePathBuf[ucbLocalBasePathLen] = TEXT('\0');
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Not shared. No CNRLink. */
|
|
|
|
*pucbCNRLinkLen = 0;
|
|
|
|
/* Copy entire local path into output buffer. */
|
|
|
|
lstrcpyn(pszLocalBasePathBuf, pcszLocalPath, cchMax);
|
|
|
|
/* Common path suffix is the empty string. */
|
|
|
|
*ppcszCommonPathSuffix = pcszLocalPath + lstrlen(pcszLocalPath);
|
|
}
|
|
}
|
|
|
|
ASSERT(! bResult ||
|
|
(EVAL(IsDrivePath(pszLocalBasePathBuf)) &&
|
|
IS_VALID_STRING_PTR(*ppcszCommonPathSuffix, CSTR) &&
|
|
EVAL(IsStringContained(pcszLocalPath, *ppcszCommonPathSuffix)) &&
|
|
(! *pucbCNRLinkLen ||
|
|
(IS_VALID_STRUCT_PTR((PCICNRLINK)*ppcnrl, CICNRLINK) &&
|
|
EVAL(*pucbCNRLinkLen == GetCNRLinkLen(*ppcnrl))))));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
/*
|
|
** CreateRemoteCNRLink()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL CreateRemoteCNRLink(LPCTSTR pcszRemotePath, LPCTSTR pcszCNRName,
|
|
PCNRLINK *ppcnrl, PUINT pucbCNRLinkLen)
|
|
{
|
|
BOOL bResult;
|
|
/* "D:" + null terminator. */
|
|
TCHAR rgchDrive[3];
|
|
DWORD dwNetType;
|
|
|
|
ASSERT(IsCanonicalPath(pcszRemotePath));
|
|
ASSERT(IsValidCNRName(pcszCNRName));
|
|
ASSERT(IS_VALID_WRITE_PTR(ppcnrl, PCNRLINK));
|
|
ASSERT(IS_VALID_WRITE_PTR(pucbCNRLinkLen, UINT));
|
|
|
|
/* Determine net provider. */
|
|
|
|
bResult = GetNetType(pcszCNRName, &dwNetType);
|
|
|
|
if (bResult)
|
|
{
|
|
DWORD dwFlags = ICNRL_FL_VALID_NET_TYPE;
|
|
|
|
/* Determine last redirected drive, if any. */
|
|
|
|
if (IsDrivePath(pcszRemotePath))
|
|
{
|
|
MyLStrCpyN(rgchDrive, pcszRemotePath, ARRAYSIZE(rgchDrive));
|
|
SET_FLAG(dwFlags, ICNRL_FL_VALID_DEVICE);
|
|
}
|
|
else
|
|
rgchDrive[0] = TEXT('\0');
|
|
|
|
bResult = UnifyICNRLinkInfo(pcszCNRName, dwFlags, rgchDrive, dwNetType,
|
|
(PICNRLINK *)ppcnrl, pucbCNRLinkLen);
|
|
}
|
|
|
|
ASSERT(! bResult ||
|
|
(IS_VALID_STRUCT_PTR((PCICNRLINK)*ppcnrl, CICNRLINK) &&
|
|
EVAL(*pucbCNRLinkLen == GetCNRLinkLen(*ppcnrl))));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
/*
|
|
** DestroyCNRLink()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE void DestroyCNRLink(PCNRLINK pcnrl)
|
|
{
|
|
ASSERT(IS_VALID_STRUCT_PTR(pcnrl, CCNRLINK));
|
|
|
|
FreeMemory(pcnrl);
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
/*
|
|
** CompareCNRLinks()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
**
|
|
** CNR link data is compared in the following order:
|
|
**
|
|
** 1) net name
|
|
**
|
|
** N.b., net types are ignored when comparing CNRLinks.
|
|
*/
|
|
PUBLIC_CODE COMPARISONRESULT CompareCNRLinks(PCCNRLINK pccnrlFirst,
|
|
PCCNRLINK pccnrlSecond)
|
|
{
|
|
#ifdef UNICODE
|
|
WCHAR szWideNetNameFirst[MAX_PATH];
|
|
LPWSTR pszWideNetNameFirst;
|
|
WCHAR szWideNetNameSecond[MAX_PATH];
|
|
LPWSTR pszWideNetNameSecond;
|
|
#endif
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrlFirst, CCNRLINK));
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrlSecond, CCNRLINK));
|
|
|
|
#ifdef UNICODE
|
|
if (IS_ICNRL_ANSI(pccnrlFirst))
|
|
{
|
|
pszWideNetNameFirst = szWideNetNameFirst;
|
|
MultiByteToWideChar(CP_ACP, 0,
|
|
ICNRL_Remote_Name_PtrA((PCICNRLINK)pccnrlFirst), -1,
|
|
szWideNetNameFirst, ARRAYSIZE(szWideNetNameFirst));
|
|
|
|
}
|
|
else
|
|
{
|
|
pszWideNetNameFirst = ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrlFirst);
|
|
}
|
|
|
|
if (IS_ICNRL_ANSI(pccnrlSecond))
|
|
{
|
|
pszWideNetNameSecond = szWideNetNameSecond;
|
|
MultiByteToWideChar(CP_ACP, 0,
|
|
ICNRL_Remote_Name_PtrA((PCICNRLINK)pccnrlSecond), -1,
|
|
szWideNetNameSecond, ARRAYSIZE(szWideNetNameSecond));
|
|
|
|
}
|
|
else
|
|
{
|
|
pszWideNetNameSecond = ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrlSecond);
|
|
}
|
|
|
|
return(CompareNetNames(pszWideNetNameFirst,pszWideNetNameSecond));
|
|
#else
|
|
return(CompareNetNames(ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrlFirst),
|
|
ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrlSecond)));
|
|
#endif
|
|
}
|
|
|
|
|
|
/*
|
|
** GetLocalPathFromCNRLink()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL GetLocalPathFromCNRLink(PCCNRLINK pccnrl,
|
|
LPTSTR pszLocalPathBuf,
|
|
PDWORD pdwOutFlags)
|
|
{
|
|
BOOL bResult;
|
|
PCSERVERVTABLE pcsvt;
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK));
|
|
ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszLocalPathBuf, STR, MAX_PATH_LEN));
|
|
ASSERT(IS_VALID_WRITE_PTR(pdwOutFlags, DWORD));
|
|
|
|
*pdwOutFlags = 0;
|
|
|
|
bResult = GetServerVTable(&pcsvt);
|
|
|
|
if (bResult)
|
|
{
|
|
DWORD dwNetType;
|
|
BOOL bIsLocal;
|
|
|
|
/*
|
|
* Get local path for share. N.b., the share name must be in upper case
|
|
* here for MSSHRUI.DLL.
|
|
*/
|
|
|
|
dwNetType = (IS_FLAG_SET(((PCICNRLINK)pccnrl)->dwFlags,
|
|
ICNRL_FL_VALID_NET_TYPE) ?
|
|
((PCICNRLINK)pccnrl)->dwNetType :
|
|
0);
|
|
|
|
#ifdef UNICODE
|
|
{
|
|
WCHAR szWideNetName[MAX_PATH];
|
|
LPWSTR pszWideNetName = szWideNetName;
|
|
|
|
if (IS_ICNRL_ANSI(pccnrl))
|
|
{
|
|
MultiByteToWideChar(CP_ACP, 0,
|
|
ICNRL_Remote_Name_PtrA((PCICNRLINK)pccnrl), -1,
|
|
szWideNetName, ARRAYSIZE(szWideNetName));
|
|
|
|
}
|
|
else
|
|
{
|
|
pszWideNetName = ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrl);
|
|
}
|
|
bResult = (pcsvt->GetLocalPathFromNetResource)(
|
|
pszWideNetName, dwNetType,
|
|
pszLocalPathBuf, MAX_PATH_LEN, &bIsLocal);
|
|
}
|
|
#else
|
|
bResult = (pcsvt->GetLocalPathFromNetResource)(
|
|
ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrl), dwNetType,
|
|
pszLocalPathBuf, MAX_PATH_LEN, &bIsLocal);
|
|
#endif
|
|
|
|
if (bIsLocal)
|
|
SET_FLAG(*pdwOutFlags, CNR_FL_LOCAL);
|
|
}
|
|
|
|
ASSERT(FLAGS_ARE_VALID(*pdwOutFlags, ALL_CNR_FLAGS) &&
|
|
(! bResult ||
|
|
(EVAL(IS_FLAG_SET(*pdwOutFlags, CNR_FL_LOCAL)) &&
|
|
EVAL(IsLocalDrivePath(pszLocalPathBuf)))));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
/*
|
|
** GetRemotePathFromCNRLink()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE void GetRemotePathFromCNRLink(PCCNRLINK pccnrl,
|
|
LPTSTR pszRemotePathBuf,
|
|
int cchMax)
|
|
{
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK));
|
|
ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszRemotePathBuf, STR, MAX_PATH_LEN));
|
|
|
|
/* It's ok that this is broken for non-UNC CNR names. */
|
|
|
|
/* (- 1) for trailing slash. */
|
|
|
|
#ifdef UNICODE
|
|
ASSERT(IS_ICNRL_ANSI(pccnrl) ? (lstrlenA(ICNRL_Remote_Name_PtrA((PCICNRLINK)pccnrl)) < MAX_PATH_LEN - 1) :
|
|
(lstrlenW(ICNRL_Remote_Name_PtrW((PCICNRLINK)pccnrl)) < MAX_PATH_LEN - 1));
|
|
#else
|
|
ASSERT(lstrlenA(ICNRL_Remote_Name_PtrA((PCICNRLINK)pccnrl)) < MAX_PATH_LEN - 1);
|
|
#endif
|
|
|
|
#ifdef UNICODE
|
|
{
|
|
WCHAR szWideNetName[MAX_PATH];
|
|
LPWSTR pszWideNetName;
|
|
|
|
if (IS_ICNRL_ANSI(pccnrl))
|
|
{
|
|
pszWideNetName = szWideNetName;
|
|
MultiByteToWideChar(CP_ACP, 0,
|
|
ICNRL_Remote_Name_PtrA((PCICNRLINK)pccnrl), -1,
|
|
szWideNetName, ARRAYSIZE(szWideNetName));
|
|
|
|
}
|
|
else
|
|
{
|
|
pszWideNetName = ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrl);
|
|
}
|
|
lstrcpyn(pszRemotePathBuf, pszWideNetName, cchMax);
|
|
}
|
|
#else
|
|
lstrcpyn(pszRemotePathBuf, ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrl), cchMax);
|
|
#endif
|
|
CatPath(pszRemotePathBuf, TEXT("\\"), cchMax);
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
/*
|
|
** ConnectToCNR()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL ConnectToCNR(PCCNRLINK pccnrl, DWORD dwInFlags,
|
|
HWND hwndOwner, LPTSTR pszRootPathBuf,
|
|
PDWORD pdwOutFlags)
|
|
{
|
|
BOOL bResult = FALSE;
|
|
BOOL bValidDevice;
|
|
BOOL bRedirect;
|
|
BOOL bTryLastDevice = FALSE;
|
|
DWORD dwcbRootPathBufLen;
|
|
LPTSTR pszNetName;
|
|
LPTSTR pszDevice;
|
|
#ifdef UNICODE
|
|
WCHAR szWideNetName[MAX_PATH];
|
|
WCHAR szWideDevice[MAX_PATH];
|
|
#endif
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK));
|
|
ASSERT(FLAGS_ARE_VALID(dwInFlags, ALL_CONNECT_IN_FLAGS));
|
|
ASSERT(IS_FLAG_CLEAR(dwInFlags, CONNECT_INTERACTIVE) ||
|
|
IS_VALID_HANDLE(hwndOwner, WND));
|
|
ASSERT(IS_VALID_WRITE_BUFFER_PTR(pszRootPathBuf, STR, MAX_PATH_LEN));
|
|
ASSERT(IS_VALID_WRITE_PTR(pdwOutFlags, DWORD));
|
|
|
|
*pdwOutFlags = 0;
|
|
|
|
#ifdef UNICODE
|
|
if (IS_ICNRL_ANSI(pccnrl))
|
|
{
|
|
pszNetName = szWideNetName;
|
|
MultiByteToWideChar(CP_ACP, 0,
|
|
ICNRL_Remote_Name_PtrA((PCICNRLINK)pccnrl), -1,
|
|
szWideNetName, ARRAYSIZE(szWideNetName));
|
|
|
|
}
|
|
else
|
|
{
|
|
pszNetName = ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrl);
|
|
}
|
|
#else
|
|
pszNetName = ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrl);
|
|
#endif
|
|
|
|
/* Do we have an old redirected device to try? */
|
|
|
|
bValidDevice = IS_FLAG_SET(((PCICNRLINK)pccnrl)->dwFlags,
|
|
ICNRL_FL_VALID_DEVICE);
|
|
|
|
#ifdef UNICODE
|
|
if ( bValidDevice )
|
|
{
|
|
if (IS_ICNRL_ANSI(pccnrl))
|
|
{
|
|
pszDevice = szWideDevice;
|
|
MultiByteToWideChar(CP_ACP, 0,
|
|
ICNRL_Device_PtrA((PCICNRLINK)pccnrl), -1,
|
|
szWideDevice, ARRAYSIZE(szWideNetName));
|
|
|
|
}
|
|
else
|
|
{
|
|
pszDevice = ICNRL_Device_Ptr((PCICNRLINK)pccnrl);
|
|
}
|
|
}
|
|
#else
|
|
pszDevice = ICNRL_Device_Ptr((PCICNRLINK)pccnrl);
|
|
#endif
|
|
|
|
bRedirect = (bValidDevice || IS_FLAG_SET(dwInFlags, CONNECT_REDIRECT));
|
|
|
|
if (bRedirect)
|
|
{
|
|
if (bValidDevice)
|
|
{
|
|
DWORD dwNetResult;
|
|
/* "X:" + null terminator */
|
|
TCHAR rgchDrive[2 + 1];
|
|
|
|
/* Yes. Is it already connected to the desired CNR? */
|
|
|
|
TRACE_OUT((TEXT("ConnectToCNR(): Calling WNetGetConnection() to check %s for CNR \"%s\"."),
|
|
pszDevice, pszNetName));
|
|
|
|
dwcbRootPathBufLen = MAX_PATH_LEN;
|
|
|
|
/* WNetGetConnection requires the device name to have no trailing
|
|
** backslash.
|
|
*/
|
|
MyLStrCpyN(rgchDrive, pszDevice, ARRAYSIZE(rgchDrive));
|
|
dwNetResult = WNetGetConnection(rgchDrive, pszRootPathBuf, &dwcbRootPathBufLen);
|
|
|
|
if (dwNetResult == WN_SUCCESS)
|
|
{
|
|
if (CompareNetNames(pszNetName, pszRootPathBuf)
|
|
== CR_EQUAL)
|
|
{
|
|
TRACE_OUT((TEXT("ConnectToCNR(): Found matching CNR \"%s\" on %s."),
|
|
pszRootPathBuf,
|
|
pszDevice));
|
|
|
|
ASSERT(lstrlenA(ICNRL_Device_PtrA((PCICNRLINK)pccnrl)) < MAX_PATH_LEN);
|
|
lstrcpyn(pszRootPathBuf, pszDevice, MAX_PATH_LEN);
|
|
|
|
bResult = TRUE;
|
|
}
|
|
else
|
|
TRACE_OUT((TEXT("ConnectToCNR(): Found unmatched CNR \"%s\" on %s."),
|
|
pszRootPathBuf,
|
|
pszDevice));
|
|
}
|
|
else
|
|
{
|
|
TRACE_OUT((TEXT("ConnectToCNR(): WNetGetConnection() failed on %s."),
|
|
pszDevice));
|
|
|
|
/*
|
|
* Only attempt a connection to the last redirected device if that
|
|
* device is not already in use.
|
|
*/
|
|
|
|
bTryLastDevice = (GetDriveType(pszDevice)
|
|
== DRIVE_NO_ROOT_DIR);
|
|
}
|
|
}
|
|
|
|
if (! bResult)
|
|
/* See if the desired CNR is connected to any local device. */
|
|
bResult = SearchForRedirectedConnection((PCICNRLINK)pccnrl,
|
|
pszRootPathBuf, MAX_PATH_LEN);
|
|
/*
|
|
* Assume that no reference count is maintained for redirected device
|
|
* connections, so we do not have to add a found redirected device
|
|
* connection again.
|
|
*/
|
|
}
|
|
|
|
if (! bResult)
|
|
{
|
|
NETRESOURCE nr;
|
|
TCHAR rgchNPName[MAX_PATH_LEN];
|
|
|
|
/* RAIDRAID: (15691) We only support disk resource connections here. */
|
|
|
|
ZeroMemory(&nr, SIZEOF(nr));
|
|
nr.lpRemoteName = pszNetName;
|
|
nr.dwType = RESOURCETYPE_DISK;
|
|
if (GetNetProviderName((PCICNRLINK)pccnrl, rgchNPName))
|
|
nr.lpProvider = rgchNPName;
|
|
|
|
/* Shall we try the old device? */
|
|
|
|
if (bTryLastDevice)
|
|
{
|
|
/* Yes. */
|
|
|
|
ASSERT(bValidDevice);
|
|
|
|
nr.lpLocalName = pszDevice;
|
|
|
|
WARNING_OUT((TEXT("ConnectToCNR(): Calling WNetUseConnection() to attempt to connect %s to CNR \"%s\"."),
|
|
nr.lpLocalName,
|
|
nr.lpRemoteName));
|
|
}
|
|
else
|
|
{
|
|
/* No. Shall we attempt to force a redirected connection? */
|
|
|
|
if (bValidDevice)
|
|
{
|
|
/*
|
|
* Yes. N.b., the caller may already have set CONNECT_REDIRECT in
|
|
* dwInFlags here.
|
|
*/
|
|
|
|
SET_FLAG(dwInFlags, CONNECT_REDIRECT);
|
|
|
|
WARNING_OUT((TEXT("ConnectToCNR(): Calling WNetUseConnection() to establish auto-picked redirected connection to CNR \"%s\"."),
|
|
nr.lpRemoteName));
|
|
}
|
|
else
|
|
/* No. */
|
|
WARNING_OUT((TEXT("ConnectToCNR(): Calling WNetUseConnection() to establish connection to CNR \"%s\"."),
|
|
TEXT("<nr.lpRemoteName>"))); // nr.lpRemoteName));
|
|
|
|
ASSERT(! nr.lpLocalName);
|
|
}
|
|
|
|
dwcbRootPathBufLen = MAX_PATH_LEN;
|
|
|
|
bResult = (WNetUseConnection(hwndOwner, &nr, NULL, NULL, dwInFlags,
|
|
pszRootPathBuf, &dwcbRootPathBufLen,
|
|
pdwOutFlags)
|
|
== NO_ERROR);
|
|
}
|
|
|
|
if (bResult)
|
|
CatPath(pszRootPathBuf, TEXT("\\"), MAX_PATH_LEN);
|
|
|
|
ASSERT(! bResult ||
|
|
(IS_VALID_STRING_PTR(pszRootPathBuf, STR) &&
|
|
FLAGS_ARE_VALID(*pdwOutFlags, ALL_CONNECT_OUT_FLAGS)));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
/*
|
|
** DisconnectFromCNR()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL DisconnectFromCNR(PCCNRLINK pccnrl)
|
|
{
|
|
DWORD dwNetResult;
|
|
LPTSTR pszNetName;
|
|
#ifdef UNICODE
|
|
WCHAR szWideNetName[MAX_PATH];
|
|
#endif
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK));
|
|
|
|
#ifdef UNICODE
|
|
if (IS_ICNRL_ANSI(pccnrl))
|
|
{
|
|
pszNetName = szWideNetName;
|
|
MultiByteToWideChar(CP_ACP, 0,
|
|
ICNRL_Remote_Name_PtrA((PCICNRLINK)pccnrl), -1,
|
|
szWideNetName, MAX_PATH);
|
|
|
|
}
|
|
else
|
|
{
|
|
pszNetName = ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrl);
|
|
}
|
|
#else
|
|
pszNetName = ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrl);
|
|
#endif
|
|
|
|
dwNetResult = WNetCancelConnection2(pszNetName,
|
|
CONNECT_REFCOUNT, FALSE);
|
|
|
|
if (dwNetResult == NO_ERROR)
|
|
WARNING_OUT((TEXT("DisconnectFromCNR(): Reduced connection reference count on CNR \"%s\"."),
|
|
pszNetName));
|
|
else
|
|
WARNING_OUT((TEXT("DisconnectFromCNR(): Failed to reduce connection reference count on CNR \"%s\". WNetCancelConnection2() returned %lu."),
|
|
pszNetName));
|
|
|
|
return(dwNetResult == NO_ERROR);
|
|
}
|
|
|
|
|
|
/*
|
|
** IsCNRAvailable()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL IsCNRAvailable(PCCNRLINK pccnrl)
|
|
{
|
|
TCHAR rgchCNRRoot[MAX_PATH_LEN];
|
|
LPTSTR pszNetName;
|
|
#ifdef UNICODE
|
|
WCHAR szWideNetName[MAX_PATH];
|
|
#endif
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK));
|
|
|
|
#ifdef UNICODE
|
|
if (IS_ICNRL_ANSI(pccnrl))
|
|
{
|
|
pszNetName = szWideNetName;
|
|
MultiByteToWideChar(CP_ACP, 0,
|
|
ICNRL_Remote_Name_PtrA((PCICNRLINK)pccnrl), -1,
|
|
szWideNetName, MAX_PATH);
|
|
|
|
}
|
|
else
|
|
{
|
|
pszNetName = ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrl);
|
|
}
|
|
#else
|
|
pszNetName = ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrl);
|
|
#endif
|
|
|
|
ASSERT(lstrlen(pszNetName) < ARRAYSIZE(rgchCNRRoot) - 1);
|
|
lstrcpyn(rgchCNRRoot, pszNetName, ARRAYSIZE(rgchCNRRoot));
|
|
CatPath(rgchCNRRoot, TEXT("\\"), ARRAYSIZE(rgchCNRRoot));
|
|
|
|
return(PathExists(rgchCNRRoot));
|
|
}
|
|
|
|
|
|
/*
|
|
** GetCNRLinkLen()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE UINT GetCNRLinkLen(PCCNRLINK pccnrl)
|
|
{
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK));
|
|
|
|
return(((PCICNRLINK)pccnrl)->ucbSize);
|
|
}
|
|
|
|
|
|
/*
|
|
** GetCNRNetType()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL GetCNRNetType(PCCNRLINK pccnrl, PCDWORD *ppcdwNetType)
|
|
{
|
|
BOOL bResult;
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK));
|
|
|
|
bResult = IS_FLAG_SET(((PCICNRLINK)pccnrl)->dwFlags,
|
|
ICNRL_FL_VALID_NET_TYPE);
|
|
|
|
if (bResult)
|
|
*ppcdwNetType = &(((PCICNRLINK)pccnrl)->dwNetType);
|
|
|
|
ASSERT(! bResult ||
|
|
IsValidNetType(**ppcdwNetType));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
|
|
/*
|
|
** GetCNRName()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL GetCNRName(PCCNRLINK pccnrl, LPCSTR *ppcszCNRName)
|
|
{
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK));
|
|
|
|
*ppcszCNRName = ICNRL_Remote_Name_PtrA((PCICNRLINK)pccnrl);
|
|
|
|
ASSERT(IS_VALID_STRING_PTRA(*ppcszCNRName, CSTR));
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
#ifdef UNICODE
|
|
/*
|
|
** GetCNRNameW()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL GetCNRNameW(PCCNRLINK pccnrl, LPCWSTR *ppcszCNRName)
|
|
{
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK));
|
|
|
|
if (IS_ICNRL_ANSI(pccnrl))
|
|
*ppcszCNRName = NULL;
|
|
else
|
|
{
|
|
*ppcszCNRName = ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrl);
|
|
ASSERT(IS_VALID_STRING_PTR(*ppcszCNRName, CSTR));
|
|
}
|
|
|
|
return(TRUE);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
** GetLastRedirectedDevice()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL GetLastRedirectedDevice(PCCNRLINK pccnrl, LPCSTR *ppcszDevice)
|
|
{
|
|
BOOL bResult;
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK));
|
|
|
|
bResult = IS_FLAG_SET(((PCICNRLINK)pccnrl)->dwFlags, ICNRL_FL_VALID_DEVICE);
|
|
|
|
if (bResult)
|
|
*ppcszDevice = ICNRL_Device_PtrA((PCICNRLINK)pccnrl);
|
|
|
|
ASSERT(! bResult ||
|
|
IS_VALID_STRING_PTRA(*ppcszDevice, CSTR));
|
|
|
|
return(bResult);
|
|
}
|
|
|
|
#ifdef UNICODE
|
|
/*
|
|
** GetLastRedirectedDeviceW()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL GetLastRedirectedDeviceW(PCCNRLINK pccnrl, LPCWSTR *ppcszDevice)
|
|
{
|
|
BOOL bResult;
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK));
|
|
|
|
bResult = IS_FLAG_SET(((PCICNRLINK)pccnrl)->dwFlags, ICNRL_FL_VALID_DEVICE);
|
|
|
|
if (bResult)
|
|
if (IS_ICNRL_ANSI(pccnrl))
|
|
*ppcszDevice = NULL;
|
|
else
|
|
{
|
|
*ppcszDevice = ICNRL_Device_Ptr((PCICNRLINK)pccnrl);
|
|
ASSERT(! bResult ||
|
|
IS_VALID_STRING_PTR(*ppcszDevice, CSTR));
|
|
}
|
|
|
|
return(bResult);
|
|
}
|
|
#endif
|
|
|
|
#if defined(DEBUG) || defined (VSTF)
|
|
|
|
/*
|
|
** IsValidPCCNRLINK()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE BOOL IsValidPCCNRLINK(PCCNRLINK pccnrl)
|
|
{
|
|
return(IS_VALID_STRUCT_PTR((PCICNRLINK)pccnrl, CICNRLINK));
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
/*
|
|
** DumpCNRLink()
|
|
**
|
|
**
|
|
**
|
|
** Arguments:
|
|
**
|
|
** Returns:
|
|
**
|
|
** Side Effects: none
|
|
*/
|
|
PUBLIC_CODE void DumpCNRLink(PCCNRLINK pccnrl)
|
|
{
|
|
ASSERT(IS_VALID_STRUCT_PTR(pccnrl, CCNRLINK));
|
|
|
|
PLAIN_TRACE_OUT((TEXT("%s%s[CNR link] ucbSize %#x"),
|
|
INDENT_STRING,
|
|
INDENT_STRING,
|
|
((PCICNRLINK)pccnrl)->ucbSize));
|
|
PLAIN_TRACE_OUT((TEXT("%s%s[CNR link] dwFLags = %#08lx"),
|
|
INDENT_STRING,
|
|
INDENT_STRING,
|
|
((PCICNRLINK)pccnrl)->dwFlags));
|
|
PLAIN_TRACE_OUT((TEXT("%s%s[CNR link] CNR name \"%s\""),
|
|
INDENT_STRING,
|
|
INDENT_STRING,
|
|
ICNRL_Remote_Name_Ptr((PCICNRLINK)pccnrl)));
|
|
if (IS_FLAG_SET(((PCICNRLINK)pccnrl)->dwFlags, ICNRL_FL_VALID_NET_TYPE))
|
|
PLAIN_TRACE_OUT((TEXT("%s%s[CNR link] net type %#08lx"),
|
|
INDENT_STRING,
|
|
INDENT_STRING,
|
|
((PCICNRLINK)pccnrl)->dwNetType));
|
|
else
|
|
PLAIN_TRACE_OUT((TEXT("%s%s[CNR link] net type unknown"),
|
|
INDENT_STRING,
|
|
INDENT_STRING));
|
|
if (IS_FLAG_SET(((PCICNRLINK)pccnrl)->dwFlags, ICNRL_FL_VALID_DEVICE))
|
|
PLAIN_TRACE_OUT((TEXT("%s%s[CNR link] last redirected local device \"%s\""),
|
|
INDENT_STRING,
|
|
INDENT_STRING,
|
|
ICNRL_Device_Ptr((PCICNRLINK)pccnrl)));
|
|
else
|
|
PLAIN_TRACE_OUT((TEXT("%s%s[CNR link] no last redirected local device"),
|
|
INDENT_STRING,
|
|
INDENT_STRING));
|
|
|
|
return;
|
|
}
|
|
|
|
#endif
|