mirror of https://github.com/tongzx/nt5src
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.
2237 lines
61 KiB
2237 lines
61 KiB
/********************************************************************/
|
|
/** Microsoft LAN Manager **/
|
|
/** Copyright(c) Microsoft Corp., 1987-1991 **/
|
|
/********************************************************************/
|
|
|
|
/***
|
|
* use.c
|
|
* Functions for displaying and manipulating network uses
|
|
* Redirected device can only be: disks a: to z:;
|
|
* comm devs com1[:] to com9[:]; and lpt1[:] to lpt9[:].
|
|
* NOTE: even though it uses the WNet*** calls, it is not
|
|
* intended to be used for networks other than LM.
|
|
* we use WNet purely to leverage off the persistent
|
|
* connections.
|
|
*
|
|
* History:
|
|
* mm/dd/yy, who, comment
|
|
* 06/09/87, andyh, new code
|
|
* 07/02/87, andyh, del with ucond = 1
|
|
* 10/31/88, erichn, uses OS2.H instead of DOSCALLS
|
|
* 01/04/89, erichn, filenames now MAX_PATH LONG
|
|
* 05/02/89, erichn, NLS conversion
|
|
* 05/09/89, erichn, local security mods
|
|
* 05/19/89, erichn, NETCMD output sorting
|
|
* 06/08/89, erichn, canonicalization sweep
|
|
* 06/23/89, erichn, replaced old NetI canon calls with new I_Net
|
|
* 03/03/90, thomaspa, INTERNAL retry with mixed case for
|
|
* password errors from down-level servers
|
|
* 03/06/90, thomaspa, integrate INTERNAL to shipped product
|
|
* 02/09/91, danhi, change to use lm 16/32 mapping layer
|
|
* 02/20/91, robdu, added profile update code
|
|
* 02/18/92, chuckc, use WNet*** to handle sticky connections (part I)
|
|
* 04/25/92, jonn, removed two cases for build fix
|
|
* 09/21/92 keithmo, use unicode versions of WNet*** API.
|
|
*/
|
|
|
|
/* Include files */
|
|
|
|
#define INCL_NOCOMMON
|
|
#define INCL_DOSFILEMGR
|
|
#define INCL_ERRORS
|
|
#define INCL_ERROR_H
|
|
#include <os2.h>
|
|
#include <search.h>
|
|
#include <lmcons.h>
|
|
#include <lmerr.h>
|
|
#include <lmapibuf.h>
|
|
#include <lmaccess.h>
|
|
#include <lmuse.h>
|
|
#include <apperr.h>
|
|
#include <apperr2.h>
|
|
#include <icanon.h>
|
|
#include <lui.h>
|
|
#include <wincred.h>
|
|
|
|
#include <dlwksta.h>
|
|
#include "mwksta.h"
|
|
#include "netcmds.h"
|
|
#include "nettext.h"
|
|
#include "msystem.h"
|
|
|
|
// pull in the win32 headers
|
|
#include <mpr.h> // for MPR_* manifests
|
|
|
|
|
|
//
|
|
// structure for combines LM and WNet info
|
|
//
|
|
typedef struct _NET_USE_INFO {
|
|
LPWSTR lpLocalName ;
|
|
LPWSTR lpRemoteName ;
|
|
LPWSTR lpProviderName ;
|
|
DWORD dwType ;
|
|
DWORD dwStatus ;
|
|
DWORD dwRefCount ;
|
|
DWORD dwUseCount ;
|
|
BOOL fIsLanman ;
|
|
} NET_USE_INFO ;
|
|
|
|
/* Static variables */
|
|
|
|
TCHAR *LanmanProviderName = NULL ;
|
|
|
|
/* Forward declarations */
|
|
|
|
VOID LanmanDisplayUse(LPUSE_INFO_1);
|
|
int NEAR is_admin_dollar(TCHAR FAR *);
|
|
int __cdecl CmpUseInfo(const VOID FAR *, const VOID FAR *);
|
|
VOID NEAR UseInit(VOID);
|
|
|
|
USHORT QueryDefaultPersistence(BOOL *pfRemember) ;
|
|
DWORD SetDefaultPersistence(BOOL fRemember) ;
|
|
BOOL CheckIfWantUpdate(TCHAR *dev, TCHAR *resource) ;
|
|
VOID WNetErrorExit(ULONG err);
|
|
WCHAR *GetLanmanProviderName(void) ;
|
|
TCHAR *MapWildCard(TCHAR *dev, TCHAR *startdev) ;
|
|
DWORD MprUseEnum(PDWORD num_read,
|
|
NET_USE_INFO **NetUseInfoBuffer,
|
|
PDWORD NetUseInfoCount);
|
|
DWORD LanmanUseAugment(DWORD num_read,
|
|
NET_USE_INFO *NetUseInfoBuffer) ;
|
|
DWORD UnavailUseAugment(PDWORD num_read,
|
|
NET_USE_INFO **NetUseInfoBuffer,
|
|
PDWORD NetUseInfoCount);
|
|
VOID MprUseDisplay(TCHAR *dev) ;
|
|
VOID use_del_all() ;
|
|
|
|
/* Externs */
|
|
|
|
extern int YorN_Switch;
|
|
|
|
/* Message related definitions */
|
|
|
|
#define USE_STATUS_OK 0
|
|
#define USE_STATUS_PAUSED ( USE_STATUS_OK + 1 )
|
|
#define USE_STATUS_SESSION_LOST ( USE_STATUS_PAUSED + 1 )
|
|
#define USE_STATUS_NET_ERROR ( USE_STATUS_SESSION_LOST + 1 )
|
|
#define USE_STATUS_CONNECTING ( USE_STATUS_NET_ERROR + 1 )
|
|
#define USE_STATUS_RECONNECTING ( USE_STATUS_CONNECTING + 1 )
|
|
#define USE_STATUS_UNAVAIL ( USE_STATUS_RECONNECTING + 1 )
|
|
#ifdef DEBUG
|
|
#define USE_STATUS_UNKNOWN ( USE_STATUS_UNAVAIL + 1 )
|
|
#endif
|
|
|
|
#define USE_REMEMBERED 0xFFFE
|
|
|
|
static MESSAGE UseStatusList[] =
|
|
{
|
|
{ APE2_USE_STATUS_OK, NULL },
|
|
{ APE2_USE_STATUS_PAUSED, NULL },
|
|
{ APE2_USE_STATUS_SESSION_LOST, NULL },
|
|
{ APE2_USE_STATUS_NET_ERROR, NULL },
|
|
{ APE2_USE_STATUS_CONNECTING, NULL },
|
|
{ APE2_USE_STATUS_RECONNECTING, NULL },
|
|
{ APE2_USE_STATUS_UNAVAIL, NULL }
|
|
#ifdef DEBUG
|
|
,
|
|
{ APE2_GEN_UNKNOWN, NULL }
|
|
#endif
|
|
};
|
|
|
|
#define NUM_STATUS_MSGS (sizeof(UseStatusList)/sizeof(UseStatusList[0]))
|
|
|
|
#define USE_MSG_LOCAL 0
|
|
#define USE_MSG_REMOTE ( USE_MSG_LOCAL + 1 )
|
|
#define USE_MSG_TYPE ( USE_MSG_REMOTE + 1 )
|
|
#define USE_TYPE_TBD ( USE_MSG_TYPE + 1 )
|
|
#define USE_MSG_STATUS ( USE_TYPE_TBD + 1 )
|
|
#define USE_STATUS_TBD ( USE_MSG_STATUS + 1 )
|
|
#define USE_MSG_OPEN_COUNT ( USE_STATUS_TBD + 1 )
|
|
#define USE_MSG_USE_COUNT ( USE_MSG_OPEN_COUNT + 1 )
|
|
|
|
static MESSAGE UseMsgList[] =
|
|
{
|
|
{ APE2_USE_MSG_LOCAL, NULL },
|
|
{ APE2_USE_MSG_REMOTE, NULL },
|
|
{ APE2_USE_MSG_TYPE, NULL },
|
|
{ APE2_GEN_UNKNOWN /* ie, TBD */, NULL },
|
|
{ APE2_USE_MSG_STATUS, NULL },
|
|
{ APE2_GEN_UNKNOWN /* ie, TBD */, NULL },
|
|
{ APE2_USE_MSG_OPEN_COUNT, NULL },
|
|
{ APE2_USE_MSG_USE_COUNT, NULL }
|
|
};
|
|
|
|
#define NUM_USE_MSGS (sizeof(UseMsgList)/sizeof(UseMsgList[0]))
|
|
|
|
/***
|
|
* use_display_all()
|
|
* Display all network uses
|
|
*
|
|
* Args:
|
|
* none
|
|
*
|
|
* Returns:
|
|
* 0 - success
|
|
* exit(2) - command failed
|
|
*/
|
|
VOID
|
|
use_display_all(
|
|
VOID
|
|
)
|
|
{
|
|
DWORD err; /* API return status */
|
|
DWORD num_read; /* num entries read by API */
|
|
DWORD maxLen; /* max message length */
|
|
DWORD i;
|
|
int msgno;
|
|
BOOL fRemember ;
|
|
DWORD NetUseInfoCount = 0 ;
|
|
NET_USE_INFO *NetUseInfoBuffer = NULL ;
|
|
NET_USE_INFO *pNetUseInfo ;
|
|
|
|
UseInit();
|
|
|
|
if (err = MprUseEnum(&num_read, &NetUseInfoBuffer, &NetUseInfoCount))
|
|
ErrorExit(err);
|
|
|
|
|
|
if (err = UnavailUseAugment(&num_read, &NetUseInfoBuffer, &NetUseInfoCount))
|
|
ErrorExit(err);
|
|
|
|
if (err = LanmanUseAugment(num_read, NetUseInfoBuffer))
|
|
ErrorExit(err);
|
|
|
|
|
|
if (QueryDefaultPersistence(&fRemember) == NERR_Success)
|
|
InfoPrint(fRemember ? APE_ConnectionsAreRemembered :
|
|
APE_ConnectionsAreNotRemembered);
|
|
else
|
|
InfoPrint(APE_ProfileReadError);
|
|
|
|
if (num_read == 0)
|
|
EmptyExit();
|
|
|
|
for (i = 0, pNetUseInfo = NetUseInfoBuffer;
|
|
i < num_read; i++, pNetUseInfo++)
|
|
{
|
|
//
|
|
// if we find at least one entry we will display, break
|
|
//
|
|
if (!(pNetUseInfo->fIsLanman)
|
|
|| (pNetUseInfo->dwStatus != USE_OK)
|
|
|| (pNetUseInfo->dwUseCount != 0)
|
|
|| (pNetUseInfo->dwRefCount != 0))
|
|
break;
|
|
}
|
|
if (i == num_read)
|
|
EmptyExit(); // loop reached limit, so no entries to display
|
|
|
|
qsort(NetUseInfoBuffer,
|
|
num_read,
|
|
sizeof(NET_USE_INFO), CmpUseInfo);
|
|
|
|
GetMessageList(NUM_STATUS_MSGS, UseStatusList, &maxLen);
|
|
|
|
PrintNL();
|
|
InfoPrint(APE2_USE_HEADER);
|
|
PrintLine();
|
|
|
|
for (i = 0, pNetUseInfo = NetUseInfoBuffer;
|
|
i < num_read;
|
|
i++, pNetUseInfo++)
|
|
{
|
|
TCHAR *status_string ;
|
|
|
|
switch(pNetUseInfo->dwStatus)
|
|
{
|
|
case USE_OK:
|
|
if ((pNetUseInfo->dwUseCount == 0)
|
|
&& (pNetUseInfo->dwRefCount == 0)
|
|
&& pNetUseInfo->fIsLanman)
|
|
continue;
|
|
else
|
|
msgno = USE_STATUS_OK;
|
|
break;
|
|
case USE_PAUSED:
|
|
msgno = USE_STATUS_PAUSED;
|
|
break;
|
|
case USE_SESSLOST:
|
|
msgno = USE_STATUS_SESSION_LOST;
|
|
break;
|
|
case USE_NETERR:
|
|
msgno = USE_STATUS_NET_ERROR;
|
|
break;
|
|
case USE_CONN:
|
|
msgno = USE_STATUS_CONNECTING;
|
|
break;
|
|
case USE_REMEMBERED:
|
|
msgno = USE_STATUS_UNAVAIL;
|
|
break;
|
|
case USE_RECONN:
|
|
msgno = USE_STATUS_RECONNECTING;
|
|
break;
|
|
default:
|
|
msgno = -1;
|
|
break;
|
|
}
|
|
|
|
if (msgno != -1)
|
|
status_string = UseStatusList[msgno].msg_text ;
|
|
else
|
|
status_string = TEXT("") ;
|
|
|
|
{
|
|
TCHAR Buffer1[13],Buffer2[10],Buffer3[MAX_PATH + 1];
|
|
|
|
if( wcslen( pNetUseInfo->lpRemoteName ) <= 25 ) {
|
|
WriteToCon(TEXT("%Fs %Fs %Fs %Fs\r\n"),
|
|
PaddedString(12,status_string,Buffer1),
|
|
PaddedString( 9,pNetUseInfo->lpLocalName,Buffer2),
|
|
PaddedString(25,pNetUseInfo->lpRemoteName,Buffer3),
|
|
pNetUseInfo->lpProviderName);
|
|
}
|
|
else
|
|
{
|
|
TCHAR Buffer4[13],Buffer5[10],Buffer6[25];
|
|
WriteToCon(TEXT("%Fs %Fs %Fs \r\n%Fs %Fs %Fs %Fs\r\n"),
|
|
PaddedString(12,status_string,Buffer1),
|
|
PaddedString( 9,pNetUseInfo->lpLocalName,Buffer2),
|
|
PaddedString(wcslen(pNetUseInfo->lpRemoteName),
|
|
pNetUseInfo->lpRemoteName,
|
|
Buffer3),
|
|
PaddedString(12,TEXT(""),Buffer4),
|
|
PaddedString(9,TEXT(""),Buffer5),
|
|
PaddedString(25,TEXT(""),Buffer6),
|
|
pNetUseInfo->lpProviderName);
|
|
}
|
|
}
|
|
}
|
|
|
|
FreeMem((LPBYTE)NetUseInfoBuffer);
|
|
InfoSuccess();
|
|
|
|
}
|
|
|
|
/***
|
|
* LanmanUseAugment()
|
|
* Enumerate uses from Lanman
|
|
*
|
|
* Args:
|
|
* none
|
|
*
|
|
* Returns:
|
|
* 0 - success
|
|
* exit(2) - command failed
|
|
*/
|
|
DWORD
|
|
LanmanUseAugment(
|
|
DWORD num_read,
|
|
NET_USE_INFO *NetUseInfoBuffer
|
|
)
|
|
{
|
|
DWORD dwErr;
|
|
DWORD cTotalAvail;
|
|
LPSTR pBuffer;
|
|
DWORD numLMread; /* num entries read by API */
|
|
DWORD j;
|
|
DWORD i;
|
|
LPUSE_INFO_1 use_entry;
|
|
NET_USE_INFO *pNetUseInfo = NetUseInfoBuffer ;
|
|
|
|
dwErr = NetUseEnum(NULL, 1, &pBuffer, MAX_PREFERRED_LENGTH,
|
|
&numLMread, &cTotalAvail, NULL);
|
|
|
|
if (dwErr != NERR_Success)
|
|
{
|
|
// consider as success (ie. there are no Lanman ones)
|
|
return NERR_Success;
|
|
}
|
|
|
|
if (numLMread == 0)
|
|
{
|
|
return NERR_Success;
|
|
}
|
|
|
|
//
|
|
// for all MPR returned entries that are Lanman uses,
|
|
// augment with extra info if we have it.
|
|
//
|
|
for (i = 0; i < num_read; i++, pNetUseInfo++)
|
|
{
|
|
//
|
|
// not LM, skip it
|
|
//
|
|
if (!(pNetUseInfo->fIsLanman))
|
|
continue ;
|
|
|
|
//
|
|
// lets find it in the NetUseEnum return data
|
|
//
|
|
for (j = 0, use_entry = (LPUSE_INFO_1) pBuffer;
|
|
j < numLMread; j++, use_entry++)
|
|
{
|
|
//
|
|
// look for match. if device names are present & match, we've found
|
|
// one. else we match only if remote names match *and* both device
|
|
// names are not present.
|
|
//
|
|
TCHAR *local = use_entry->ui1_local ;
|
|
TCHAR *remote = use_entry->ui1_remote ;
|
|
|
|
|
|
if ( (local && *local && !_tcsicmp(pNetUseInfo->lpLocalName,local))
|
|
||
|
|
( (!local || !*local) &&
|
|
!*(pNetUseInfo->lpLocalName) &&
|
|
!_tcsicmp(pNetUseInfo->lpRemoteName,remote)
|
|
)
|
|
)
|
|
{
|
|
//
|
|
// found the device in the LM list or
|
|
// found as deviceless connection
|
|
//
|
|
pNetUseInfo->dwUseCount = use_entry->ui1_usecount ;
|
|
pNetUseInfo->dwRefCount = use_entry->ui1_refcount ;
|
|
pNetUseInfo->dwStatus = use_entry->ui1_status ;
|
|
break ;
|
|
}
|
|
}
|
|
}
|
|
NetApiBufferFree(pBuffer);
|
|
|
|
return NERR_Success;
|
|
}
|
|
|
|
|
|
|
|
/***
|
|
* MprUseEnum()
|
|
* Enumerates uses returned by WNET
|
|
*
|
|
* Args:
|
|
* none
|
|
*
|
|
* Returns:
|
|
* 0 - success
|
|
* exit(2) - command failed
|
|
*/
|
|
DWORD
|
|
MprUseEnum(
|
|
LPDWORD num_read,
|
|
NET_USE_INFO **NetUseInfoBuffer,
|
|
LPDWORD NetUseInfoCount
|
|
)
|
|
{
|
|
DWORD EntriesRead = 0;
|
|
LPBYTE Buffer;
|
|
DWORD dwErr;
|
|
DWORD dwAllocErr;
|
|
HANDLE EnumHandle;
|
|
DWORD BufferSize, Count;
|
|
static TCHAR *NullString = TEXT("");
|
|
|
|
//
|
|
// initialize
|
|
//
|
|
*num_read = 0;
|
|
*NetUseInfoCount = 64; // assume 64 entries initially. realloc if need
|
|
if (dwAllocErr = AllocMem( *NetUseInfoCount * sizeof(NET_USE_INFO),
|
|
(LPBYTE *) NetUseInfoBuffer ))
|
|
{
|
|
ErrorExit(dwAllocErr);
|
|
}
|
|
|
|
//
|
|
// allocate memory and open the enumeration
|
|
//
|
|
if (dwAllocErr = AllocMem(BufferSize = 4096, &Buffer))
|
|
{
|
|
ErrorExit(dwAllocErr);
|
|
}
|
|
|
|
dwErr = WNetOpenEnum(RESOURCE_CONNECTED, 0, 0, NULL, &EnumHandle) ;
|
|
if (dwErr != WN_SUCCESS)
|
|
{
|
|
return dwErr;
|
|
}
|
|
|
|
do
|
|
{
|
|
Count = 0xFFFFFFFF ;
|
|
dwErr = WNetEnumResource(EnumHandle, &Count, Buffer, &BufferSize) ;
|
|
|
|
if (dwErr == WN_SUCCESS || dwErr == WN_NO_MORE_ENTRIES)
|
|
{
|
|
LPNETRESOURCE lpNetResource ;
|
|
NET_USE_INFO *lpNetUseInfo ;
|
|
DWORD i ;
|
|
|
|
//
|
|
// grow the buffer if need. note there are no
|
|
// pointers that point back to the buffer, so we
|
|
// should be fine with the realloc.
|
|
//
|
|
if (EntriesRead + Count >= *NetUseInfoCount)
|
|
{
|
|
//
|
|
// make sure it can hold all the new data, and add 64
|
|
// to reduce the number of reallocs
|
|
//
|
|
*NetUseInfoCount = EntriesRead + Count + 64;
|
|
dwAllocErr = ReallocMem(*NetUseInfoCount * sizeof(NET_USE_INFO),
|
|
(LPBYTE *)NetUseInfoBuffer) ;
|
|
if (dwAllocErr != NERR_Success)
|
|
return dwAllocErr;
|
|
}
|
|
lpNetResource = (LPNETRESOURCE) Buffer ;
|
|
lpNetUseInfo = *NetUseInfoBuffer + EntriesRead ;
|
|
|
|
//
|
|
// stick the entries into the NetUseInfoBuffer
|
|
//
|
|
for ( i = 0;
|
|
i < Count;
|
|
i++,EntriesRead++,lpNetUseInfo++,lpNetResource++ )
|
|
{
|
|
lpNetUseInfo->lpLocalName = lpNetResource->lpLocalName ?
|
|
lpNetResource->lpLocalName : NullString ;
|
|
lpNetUseInfo->lpRemoteName = lpNetResource->lpRemoteName ?
|
|
lpNetResource->lpRemoteName : NullString ;
|
|
lpNetUseInfo->lpProviderName = lpNetResource->lpProvider ?
|
|
lpNetResource->lpProvider : NullString ;
|
|
lpNetUseInfo->dwType = lpNetResource->dwType ;
|
|
lpNetUseInfo->fIsLanman =
|
|
(_tcscmp(lpNetResource->lpProvider,LanmanProviderName)==0) ;
|
|
lpNetUseInfo->dwStatus = 0xFFFFFFFF ;
|
|
lpNetUseInfo->dwRefCount =
|
|
lpNetUseInfo->dwUseCount = 0 ;
|
|
|
|
}
|
|
|
|
//
|
|
// allocate a new buffer for next set, since we still need
|
|
// data in the old one, we dont free it. Netcmd lets the
|
|
// system clean up when it exits.
|
|
//
|
|
if (dwErr == WN_SUCCESS)
|
|
{
|
|
if (dwAllocErr = AllocMem(BufferSize = 4096, &Buffer))
|
|
{
|
|
ErrorExit(dwAllocErr);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return dwErr;
|
|
}
|
|
}
|
|
while (dwErr == WN_SUCCESS);
|
|
|
|
dwErr = WNetCloseEnum(EnumHandle) ; // we dont report any errors here
|
|
|
|
*num_read = EntriesRead ;
|
|
return NERR_Success ;
|
|
}
|
|
|
|
|
|
/***
|
|
* UnavailUseAugment()
|
|
* Enumerate unavail uses & tags them on.
|
|
*
|
|
* Args:
|
|
* none
|
|
*
|
|
* Returns:
|
|
* 0 - success
|
|
* exit(2) - command failed
|
|
*/
|
|
DWORD
|
|
UnavailUseAugment(
|
|
LPDWORD NumRead,
|
|
NET_USE_INFO **NetUseInfoBuffer,
|
|
LPDWORD NetUseInfoCount
|
|
)
|
|
{
|
|
LPBYTE Buffer ;
|
|
DWORD dwErr ;
|
|
HANDLE EnumHandle ;
|
|
DWORD BufferSize, Count, InitialUseInfoCount ;
|
|
DWORD err ;
|
|
static TCHAR *NullString = TEXT("") ;
|
|
|
|
InitialUseInfoCount = *NumRead ;
|
|
|
|
//
|
|
// allocate memory and open the enumeration
|
|
//
|
|
if (err = AllocMem(BufferSize = 4096, &Buffer))
|
|
{
|
|
ErrorExit(err);
|
|
}
|
|
|
|
dwErr = WNetOpenEnum(RESOURCE_REMEMBERED, 0, 0, NULL, &EnumHandle) ;
|
|
|
|
if (dwErr != WN_SUCCESS)
|
|
{
|
|
return dwErr;
|
|
}
|
|
|
|
do
|
|
{
|
|
Count = 0xFFFFFFFF ;
|
|
dwErr = WNetEnumResource(EnumHandle, &Count, Buffer, &BufferSize) ;
|
|
|
|
if (dwErr == WN_SUCCESS || dwErr == WN_NO_MORE_ENTRIES)
|
|
{
|
|
LPNETRESOURCE lpNetResource ;
|
|
NET_USE_INFO *lpNetUseInfo ;
|
|
DWORD i,j ;
|
|
|
|
if (Count == 0xFFFFFFFF || Count == 0)
|
|
break ;
|
|
|
|
lpNetResource = (LPNETRESOURCE) Buffer ;
|
|
|
|
//
|
|
// for each entry, see if it is an unavail one
|
|
//
|
|
for ( i = 0;
|
|
i < Count;
|
|
i++,lpNetResource++ )
|
|
{
|
|
lpNetUseInfo = *NetUseInfoBuffer ;
|
|
|
|
//
|
|
// search thru the ones we already have
|
|
//
|
|
for (j = 0;
|
|
j < InitialUseInfoCount;
|
|
j++, ++lpNetUseInfo)
|
|
{
|
|
if (lpNetUseInfo->lpLocalName &&
|
|
lpNetResource->lpLocalName)
|
|
{
|
|
if ( *lpNetUseInfo->lpLocalName != 0 )
|
|
{
|
|
// Use _tcsnicmp because the Net api returns an LPTX
|
|
// redirection without the ':' whereas the WNet api
|
|
// includes the ':'.
|
|
if (_tcsnicmp(lpNetResource->lpLocalName,
|
|
lpNetUseInfo->lpLocalName,
|
|
_tcslen(lpNetUseInfo->lpLocalName))==0)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
else if (*lpNetResource->lpLocalName == 0)
|
|
{
|
|
break ;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// if we broke out early, this is already connected, so
|
|
// we dont bother add an 'unavailable' entry.
|
|
//
|
|
if (j < InitialUseInfoCount)
|
|
continue ;
|
|
|
|
//
|
|
// grow the buffer if need. note there are no
|
|
// pointers that point back to the buffer, so we
|
|
// should be fine with the realloc.
|
|
//
|
|
if (*NumRead >= *NetUseInfoCount)
|
|
{
|
|
//
|
|
// make sure it can hold all the new data, and add 64
|
|
// to reduce the number of reallocs
|
|
//
|
|
*NetUseInfoCount += 64 ;
|
|
err = ReallocMem(*NetUseInfoCount * sizeof(NET_USE_INFO),
|
|
(LPBYTE *) NetUseInfoBuffer);
|
|
|
|
if (err != NERR_Success)
|
|
{
|
|
return err;
|
|
}
|
|
}
|
|
|
|
lpNetUseInfo = *NetUseInfoBuffer + *NumRead ;
|
|
|
|
lpNetUseInfo->lpLocalName = lpNetResource->lpLocalName ?
|
|
lpNetResource->lpLocalName : NullString ;
|
|
lpNetUseInfo->lpRemoteName = lpNetResource->lpRemoteName ?
|
|
lpNetResource->lpRemoteName : NullString ;
|
|
lpNetUseInfo->lpProviderName = lpNetResource->lpProvider ?
|
|
lpNetResource->lpProvider : NullString ;
|
|
lpNetUseInfo->dwType = lpNetResource->dwType ;
|
|
lpNetUseInfo->fIsLanman = FALSE ; // no more info of interest
|
|
lpNetUseInfo->dwStatus = USE_REMEMBERED ;
|
|
lpNetUseInfo->dwRefCount =
|
|
lpNetUseInfo->dwUseCount = 0 ;
|
|
|
|
_tcsupr(lpNetUseInfo->lpLocalName) ;
|
|
*NumRead += 1 ;
|
|
}
|
|
|
|
//
|
|
// allocate a new buffer for next set, since we still need
|
|
// data in the old one, we dont free it. Netcmd lets the
|
|
// system clean up when it exits.
|
|
//
|
|
if (dwErr == WN_SUCCESS)
|
|
{
|
|
if (err = AllocMem(BufferSize = 4096, &Buffer))
|
|
{
|
|
ErrorExit(err);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return dwErr;
|
|
}
|
|
}
|
|
while (dwErr == WN_SUCCESS) ;
|
|
|
|
dwErr = WNetCloseEnum(EnumHandle) ; // we dont report any errors here
|
|
|
|
return NERR_Success ;
|
|
}
|
|
|
|
|
|
/***
|
|
* CmpUseInfo(use1,use2)
|
|
*
|
|
* Compares two USE_INFO_1 structures and returns a relative
|
|
* lexical value, suitable for using in qsort.
|
|
*
|
|
*/
|
|
|
|
int __cdecl CmpUseInfo(const VOID FAR * use1, const VOID FAR * use2)
|
|
{
|
|
register USHORT localDev1, localDev2;
|
|
register DWORD devType1, devType2;
|
|
|
|
/* first sort by whether use has local device name */
|
|
localDev1 = ((NET_USE_INFO *) use1)->lpLocalName[0];
|
|
localDev2 = ((NET_USE_INFO *) use2)->lpLocalName[0];
|
|
if (localDev1 && !localDev2)
|
|
return -1;
|
|
if (localDev2 && !localDev1)
|
|
return +1;
|
|
|
|
/* then sort by device type */
|
|
devType1 = ((NET_USE_INFO *) use1)->dwType;
|
|
devType2 = ((NET_USE_INFO *) use2)->dwType;
|
|
if (devType1 != devType2)
|
|
return( (devType1 < devType2) ? -1 : 1 );
|
|
|
|
|
|
/* if local device, sort by local name */
|
|
if (localDev1)
|
|
return _tcsicmp ( ((NET_USE_INFO *) use1)->lpLocalName,
|
|
((NET_USE_INFO *) use2)->lpLocalName);
|
|
else
|
|
/* sort by remote name */
|
|
return _tcsicmp ( ((NET_USE_INFO *) use1)->lpRemoteName,
|
|
((NET_USE_INFO *) use2)->lpRemoteName);
|
|
}
|
|
|
|
|
|
|
|
/***
|
|
* use_unc()
|
|
* Process "NET USE unc-name" command line (display or add)
|
|
*
|
|
* Args:
|
|
* name - the unc name
|
|
*
|
|
* Returns:
|
|
* 0 - success
|
|
* exit(2) - command failed
|
|
*/
|
|
VOID use_unc(TCHAR * name)
|
|
{
|
|
DWORD dwErr;
|
|
LPUSE_INFO_1 use_entry;
|
|
|
|
UseInit();
|
|
if (dwErr = NetUseGetInfo(NULL,
|
|
name,
|
|
1,
|
|
(LPBYTE *) &use_entry))
|
|
{
|
|
//
|
|
// hit an error, so just add it
|
|
//
|
|
NetApiBufferFree((LPBYTE) use_entry);
|
|
use_add(NULL, name, NULL, FALSE, TRUE);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// it is Lanman. treat it as we have in the past
|
|
//
|
|
if ((use_entry->ui1_usecount == 0) && (use_entry->ui1_refcount == 0))
|
|
use_add(NULL, name, NULL, FALSE, FALSE);
|
|
else
|
|
LanmanDisplayUse(use_entry);
|
|
NetApiBufferFree((CHAR FAR *) use_entry);
|
|
InfoSuccess();
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/***
|
|
* use_display_dev()
|
|
* Display status of redirected device.
|
|
*
|
|
* Args:
|
|
* dev - redirected device
|
|
*
|
|
* Returns:
|
|
* 0 - success
|
|
* exit(2) - command failed
|
|
*/
|
|
VOID use_display_dev(TCHAR * dev)
|
|
{
|
|
DWORD dwErr;
|
|
LPUSE_INFO_1 use_entry = NULL;
|
|
|
|
if (IsWildCard(dev))
|
|
help_help(0, USAGE_ONLY) ;
|
|
|
|
UseInit();
|
|
if (dwErr = NetUseGetInfo(NULL,
|
|
dev,
|
|
1,
|
|
(LPBYTE *) &use_entry))
|
|
{
|
|
//
|
|
// Lanman failed, so try MPR
|
|
//
|
|
NetApiBufferFree((LPBYTE) use_entry);
|
|
MprUseDisplay(dev) ;
|
|
InfoSuccess();
|
|
}
|
|
else
|
|
{
|
|
LanmanDisplayUse(use_entry);
|
|
NetApiBufferFree((CHAR FAR *) use_entry);
|
|
InfoSuccess();
|
|
}
|
|
}
|
|
|
|
VOID
|
|
MprUseDisplay(
|
|
TCHAR *dev
|
|
)
|
|
{
|
|
DWORD dwErr;
|
|
DWORD dwLength = 0;
|
|
LPTSTR lpRemoteName;
|
|
DWORD maxLen;
|
|
|
|
//
|
|
// Figure out how large a buffer we need
|
|
//
|
|
dwErr = WNetGetConnection(dev, NULL, &dwLength);
|
|
|
|
if (dwErr != WN_MORE_DATA)
|
|
{
|
|
ErrorExit(dwErr);
|
|
}
|
|
|
|
dwErr = AllocMem(dwLength * sizeof(TCHAR), &lpRemoteName);
|
|
|
|
if (dwErr != NERR_Success)
|
|
{
|
|
ErrorExit(dwErr);
|
|
}
|
|
|
|
dwErr = WNetGetConnection(dev, lpRemoteName, &dwLength);
|
|
|
|
if (dwErr != WN_SUCCESS && dwErr != WN_CONNECTION_CLOSED)
|
|
{
|
|
ErrorExit(dwErr);
|
|
}
|
|
|
|
dwLength = _tcslen(dev);
|
|
|
|
if (dwLength == 2 && _istalpha(dev[0]) && dev[1] == TEXT(':'))
|
|
{
|
|
UseMsgList[USE_TYPE_TBD].msg_number = APE2_USE_TYPE_DISK;
|
|
}
|
|
else if (dwLength >= 3 && _tcsnicmp(dev, TEXT("LPT"), 3) == 0)
|
|
{
|
|
UseMsgList[USE_TYPE_TBD].msg_number = APE2_USE_TYPE_PRINT;
|
|
}
|
|
else
|
|
{
|
|
UseMsgList[USE_TYPE_TBD].msg_number = APE2_GEN_UNKNOWN;
|
|
}
|
|
|
|
GetMessageList(NUM_USE_MSGS, UseMsgList, &maxLen);
|
|
maxLen += 5;
|
|
|
|
WriteToCon(fmtPSZ,0,maxLen,
|
|
PaddedString(maxLen,UseMsgList[USE_MSG_LOCAL].msg_text,NULL),
|
|
dev);
|
|
|
|
WriteToCon(fmtPSZ,0,maxLen,
|
|
PaddedString(maxLen,UseMsgList[USE_MSG_REMOTE].msg_text,NULL),
|
|
lpRemoteName);
|
|
|
|
WriteToCon(fmtNPSZ,0,maxLen,
|
|
PaddedString(maxLen,UseMsgList[USE_MSG_TYPE].msg_text,NULL),
|
|
UseMsgList[USE_TYPE_TBD].msg_text);
|
|
|
|
FreeMem(lpRemoteName);
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
/***
|
|
* use_add()
|
|
* Add a redirection
|
|
*
|
|
* Args:
|
|
* dev - local device to redirect
|
|
* name - remote name to redirect to
|
|
* pass - password to use when validating the use
|
|
* comm - TRUE --> use as a char dev
|
|
* print_ok - should a message be printed on success?
|
|
*
|
|
* Returns:
|
|
* 0 - success
|
|
* exit(2) - command failed
|
|
*/
|
|
VOID use_add(TCHAR * dev, TCHAR * name, TCHAR * pass, int comm, int print_ok)
|
|
{
|
|
static TCHAR pbuf[(PWLEN>CREDUI_MAX_PASSWORD_LENGTH?PWLEN:CREDUI_MAX_PASSWORD_LENGTH)+1];
|
|
static TCHAR UserBuffer[CREDUI_MAX_USERNAME_LENGTH+1];
|
|
static TCHAR ServerNameBuffer[MAX_PATH+1];
|
|
|
|
USHORT err; /* short return status */
|
|
ULONG ulErr ; /* long return status */
|
|
ULONG longtype; /* type field for I_NetPath */
|
|
NETRESOURCEW netresource ;
|
|
BOOL fRememberSwitch = FALSE ;
|
|
BOOL fRemember = FALSE ;
|
|
BOOL fSmartCard = FALSE;
|
|
ULONG bConnectFlags = 0L ;
|
|
LPWSTR pw_username = NULL;
|
|
LPWSTR pw_pass = NULL;
|
|
TCHAR *devicename = dev ;
|
|
BOOL fExitCodeIsDrive = FALSE ;
|
|
BOOL fSaveCred = FALSE;
|
|
|
|
// unreferenced
|
|
(void) comm ;
|
|
|
|
UseInit();
|
|
|
|
//
|
|
// Build a non-UNC version of name to connect to.
|
|
//
|
|
|
|
if ( name != NULL ) {
|
|
TCHAR *SlashPointer;
|
|
|
|
//
|
|
// Lob off the leading backslashes
|
|
//
|
|
|
|
if ( name[0] == '\\' && name[1] == '\\' ) {
|
|
_tcscpy( ServerNameBuffer, &name[2] );
|
|
} else {
|
|
_tcscpy( ServerNameBuffer, name );
|
|
}
|
|
|
|
//
|
|
// Lob off the share name
|
|
//
|
|
|
|
SlashPointer = _tcschr( ServerNameBuffer, '\\');
|
|
if ( SlashPointer != NULL ) {
|
|
*SlashPointer = '\0';
|
|
}
|
|
|
|
} else {
|
|
ServerNameBuffer[0] = '\0';
|
|
}
|
|
|
|
// make sure we clean this up
|
|
AddToMemClearList(pbuf, sizeof(pbuf), FALSE) ;
|
|
|
|
// deal with any wild card Device specification
|
|
if (devicename)
|
|
{
|
|
// If the devicname is a '?', then the exit code should be the ASCII
|
|
// value of the drive that we connect.
|
|
if (IsQuestionMark(devicename))
|
|
{
|
|
fExitCodeIsDrive = TRUE;
|
|
}
|
|
devicename = MapWildCard(devicename, NULL) ;
|
|
if (!devicename)
|
|
{
|
|
// this can omly happen if no drives left
|
|
ErrorExit(APE_UseWildCardNoneLeft) ;
|
|
}
|
|
}
|
|
|
|
/* Initialize netresource structure */
|
|
netresource.lpProvider = NULL ;
|
|
netresource.lpLocalName = NULL ;
|
|
netresource.lpRemoteName = NULL ;
|
|
netresource.dwType = 0L ;
|
|
|
|
if (devicename)
|
|
{
|
|
if (I_NetPathType(NULL, devicename, &longtype, 0L))
|
|
ErrorExit(APE_UnknDevType);
|
|
|
|
/*
|
|
* NOTE: I would haved LOVED to have used a switch statement here.
|
|
* But since types are now LONGS, and the compiler doesn't support
|
|
* long switch statements, we're stuck with multiple if's. Sorry.
|
|
*/
|
|
if (longtype == ITYPE_DEVICE_DISK)
|
|
netresource.dwType = RESOURCETYPE_DISK;
|
|
else if (longtype == ITYPE_DEVICE_LPT)
|
|
netresource.dwType = RESOURCETYPE_PRINT;
|
|
else if (longtype == ITYPE_DEVICE_COM)
|
|
netresource.dwType = RESOURCETYPE_PRINT;
|
|
else
|
|
ErrorExit(APE_UnknDevType);
|
|
|
|
netresource.lpLocalName = devicename ;
|
|
}
|
|
else
|
|
{
|
|
netresource.dwType = RESOURCETYPE_ANY;
|
|
netresource.lpLocalName = L"" ;
|
|
}
|
|
|
|
if( name != NULL )
|
|
{
|
|
netresource.lpRemoteName = name ;
|
|
}
|
|
|
|
{
|
|
USHORT i;
|
|
|
|
// Find out if the /USER or /PERSISTENT switches are used
|
|
for (i = 0; SwitchList[i]; i++)
|
|
{
|
|
//
|
|
// Handle the /PERSISTENT switch
|
|
//
|
|
if ( !(_tcsncmp(SwitchList[i],
|
|
swtxt_SW_USE_PERSISTENT,
|
|
_tcslen(swtxt_SW_USE_PERSISTENT))) )
|
|
{
|
|
LPTSTR ptr;
|
|
DWORD res;
|
|
DWORD answer;
|
|
|
|
fRememberSwitch = TRUE;
|
|
|
|
// find the colon separator
|
|
if ((ptr = FindColon(SwitchList[i])) == NULL)
|
|
{
|
|
ErrorExit(APE_InvalidSwitchArg);
|
|
}
|
|
|
|
// parse string after colon for YES or NO
|
|
if ((res = LUI_ParseYesNo(ptr,&answer)) != 0)
|
|
{
|
|
ErrorExitInsTxt(APE_CmdArgIllegal,SwitchList[i]);
|
|
}
|
|
|
|
fRemember = (answer == LUI_YES_VAL) ;
|
|
|
|
//
|
|
// Handle the /USER switch
|
|
//
|
|
}
|
|
else if ( !(_tcsncmp(SwitchList[i],
|
|
swtxt_SW_USE_USER,
|
|
_tcslen(swtxt_SW_USE_USER))) )
|
|
{
|
|
PTCHAR ptr;
|
|
// find the colon separator
|
|
if ((ptr = FindColon(SwitchList[i])) == NULL)
|
|
ErrorExit(APE_InvalidSwitchArg);
|
|
|
|
pw_username = ptr;
|
|
|
|
//
|
|
// Handle the /SMARTCARD switch
|
|
//
|
|
|
|
} else if ( !(_tcscmp(SwitchList[i], swtxt_SW_USE_SMARTCARD ))) {
|
|
|
|
fSmartCard = TRUE;
|
|
|
|
//
|
|
// Handle the /SAVECRED switch
|
|
//
|
|
|
|
} else if ( !(_tcscmp(SwitchList[i], swtxt_SW_USE_SAVECRED ))) {
|
|
|
|
fSaveCred = TRUE;
|
|
|
|
|
|
//
|
|
// Handle the /Delete switch
|
|
// (The parser really doesn't let this through.)
|
|
//
|
|
}
|
|
else if ( !(_tcscmp(SwitchList[i], swtxt_SW_DELETE)) )
|
|
{
|
|
// what the heck? adding and deleting?
|
|
ErrorExit(APE_ConflictingSwitches) ;
|
|
}
|
|
// ignore other switches
|
|
}
|
|
}
|
|
|
|
// remember switch was not specified
|
|
if (!fRememberSwitch)
|
|
{
|
|
if (QueryDefaultPersistence(&fRemember)!=NERR_Success)
|
|
InfoPrint(APE_ProfileReadError);
|
|
}
|
|
|
|
//
|
|
// /user and /savecred are mutually exclusive.
|
|
// This is because the auth packages don't call cred man if the user name
|
|
// is specified. Therefore, the auth package didn't pass the target info to cred man.
|
|
// If there is no target info, the UI will save a server specific cred.
|
|
//
|
|
|
|
if ( pw_username != NULL && fSaveCred ) {
|
|
ErrorExit(APE_ConflictingSwitches) ;
|
|
}
|
|
|
|
|
|
//
|
|
// Handle /SMARTCARD switch.
|
|
// Prompt for smart card credentials.
|
|
//
|
|
|
|
if ( fSmartCard ) {
|
|
|
|
//
|
|
// We don't know how to save smartcard creds
|
|
//
|
|
|
|
if ( fSaveCred ) {
|
|
ErrorExit(APE_ConflictingSwitches) ;
|
|
}
|
|
|
|
//
|
|
// If a user name was specified,
|
|
// use it to select which smart card to use.
|
|
//
|
|
|
|
if ( pw_username != NULL ) {
|
|
_tcscpy( UserBuffer, pw_username );
|
|
} else {
|
|
UserBuffer[0] = '\0';
|
|
}
|
|
|
|
//
|
|
// If password is specified,
|
|
// use it as the default PIN
|
|
|
|
if ( pass != NULL ) {
|
|
|
|
// Consider "*" to be the same as "not specified"
|
|
if (! _tcscmp(pass, TEXT("*"))) {
|
|
pbuf[0] = '\0';
|
|
} else {
|
|
if (err = LUI_CanonPassword(pass)) {
|
|
ErrorExit(err);
|
|
}
|
|
_tcscpy( pbuf, pass );
|
|
}
|
|
|
|
} else {
|
|
pbuf[0] = '\0';
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Call the common UI.
|
|
//
|
|
// RtlZeroMemory( &UiInfo, sizeof(UiInfo) );
|
|
// UiInfo.dwVersion = 1;
|
|
|
|
ulErr = CredUICmdLinePromptForCredentialsW(
|
|
ServerNameBuffer, // Target name
|
|
NULL, // No context
|
|
NO_ERROR, // No authentication error
|
|
UserBuffer,
|
|
CREDUI_MAX_USERNAME_LENGTH,
|
|
pbuf,
|
|
CREDUI_MAX_PASSWORD_LENGTH,
|
|
NULL, // SaveFlag not allowed unless flag is specified
|
|
CREDUI_FLAGS_REQUIRE_SMARTCARD |
|
|
CREDUI_FLAGS_DO_NOT_PERSIST );
|
|
|
|
if ( ulErr != NO_ERROR ) {
|
|
ErrorExit(ulErr);
|
|
}
|
|
|
|
pw_username = UserBuffer;
|
|
pw_pass = pbuf;
|
|
|
|
|
|
//
|
|
// Handle cases where the password is specified on the command line.
|
|
//
|
|
|
|
} else if (pass) {
|
|
|
|
//
|
|
// We don't know how to save creds we don't prompt for
|
|
//
|
|
|
|
if ( fSaveCred ) {
|
|
ErrorExit(APE_ConflictingSwitches) ;
|
|
}
|
|
|
|
|
|
|
|
if (! _tcscmp(pass, TEXT("*")))
|
|
{
|
|
|
|
|
|
pass = pbuf;
|
|
IStrings[0] = name;
|
|
ReadPass(pass, PWLEN, 0, APE_UsePassPrompt, 1, FALSE);
|
|
if (err = LUI_CanonPassword(pass))
|
|
ErrorExit(err);
|
|
}
|
|
else
|
|
{
|
|
if (err = LUI_CanonPassword(pass))
|
|
ErrorExit(err);
|
|
}
|
|
|
|
pw_pass = pass;
|
|
}
|
|
|
|
|
|
//
|
|
// Loop handling assigning to the next drive letter
|
|
//
|
|
do {
|
|
|
|
//if persistent, the check for clash with existing remembered connection
|
|
bConnectFlags = 0 ;
|
|
if (fRemember)
|
|
{
|
|
if (CheckIfWantUpdate(devicename, name))
|
|
bConnectFlags |= CONNECT_UPDATE_PROFILE ;
|
|
}
|
|
|
|
//
|
|
// Allow it to prompt for us if the credential aren't on the command line
|
|
//
|
|
if ( !pass && !fSmartCard) {
|
|
bConnectFlags |= CONNECT_INTERACTIVE | CONNECT_COMMANDLINE;
|
|
|
|
//
|
|
// If the caller wants to save both username and password,
|
|
// create an enterprise peristed cred.
|
|
//
|
|
if ( fSaveCred ) {
|
|
bConnectFlags |= CONNECT_CMD_SAVECRED;
|
|
}
|
|
}
|
|
|
|
ulErr = WNetAddConnection2(&netresource,
|
|
pw_pass,
|
|
pw_username,
|
|
bConnectFlags) ;
|
|
|
|
switch(ulErr)
|
|
{
|
|
case WN_SUCCESS:
|
|
if (fRememberSwitch)
|
|
{
|
|
DWORD dwErr = SetDefaultPersistence(fRemember);
|
|
|
|
if (dwErr != NERR_Success)
|
|
{
|
|
ErrorExit(dwErr);
|
|
}
|
|
}
|
|
if (print_ok)
|
|
{
|
|
if (IsWildCard(dev)) // if originally a wildcard
|
|
{
|
|
IStrings[0] = devicename;
|
|
IStrings[1] = name;
|
|
InfoPrintIns(APE_UseWildCardSuccess, 2) ;
|
|
}
|
|
InfoSuccess();
|
|
}
|
|
if (fExitCodeIsDrive)
|
|
{
|
|
MyExit((int)devicename[0]);
|
|
}
|
|
return;
|
|
|
|
case WN_BAD_PASSWORD:
|
|
case WN_ACCESS_DENIED:
|
|
case ERROR_LOGON_FAILURE:
|
|
|
|
WNetErrorExit(ulErr);
|
|
|
|
case ERROR_ALREADY_ASSIGNED:
|
|
if (!IsWildCard(dev))
|
|
ErrorExit(ERROR_ALREADY_ASSIGNED) ;
|
|
|
|
// Get another drive letter
|
|
(devicename[0])++;
|
|
devicename = MapWildCard(TEXT("*"), devicename) ;
|
|
if (!devicename)
|
|
{
|
|
// this can only happen if no drives left
|
|
ErrorExit(APE_UseWildCardNoneLeft) ;
|
|
}
|
|
netresource.lpLocalName = devicename ;
|
|
break;
|
|
|
|
case WN_BAD_NETNAME:
|
|
if (is_admin_dollar(name))
|
|
ErrorExit(APE_BadAdminConfig);
|
|
// else drop thru
|
|
|
|
default:
|
|
WNetErrorExit(ulErr);
|
|
}
|
|
|
|
} while ( ulErr == ERROR_ALREADY_ASSIGNED );
|
|
|
|
}
|
|
|
|
|
|
VOID
|
|
use_set_remembered(
|
|
VOID
|
|
)
|
|
{
|
|
PTCHAR ptr;
|
|
BOOL fRemember ;
|
|
|
|
// Find the /persistent switch
|
|
if ((ptr = FindColon(SwitchList[0])) == NULL)
|
|
ErrorExit(APE_InvalidSwitchArg);
|
|
|
|
if ( !(_tcscmp(SwitchList[0], swtxt_SW_USE_PERSISTENT) ) )
|
|
{
|
|
DWORD res;
|
|
DWORD answer;
|
|
|
|
if ((res = LUI_ParseYesNo(ptr,&answer)) != 0)
|
|
{
|
|
ErrorExitInsTxt(APE_CmdArgIllegal,SwitchList[0]) ;
|
|
}
|
|
|
|
fRemember = (answer == LUI_YES_VAL) ;
|
|
|
|
res = SetDefaultPersistence(fRemember);
|
|
|
|
if (res != NERR_Success)
|
|
{
|
|
ErrorExit(res);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ErrorExit(APE_InvalidSwitch);
|
|
}
|
|
|
|
|
|
InfoSuccess();
|
|
}
|
|
|
|
|
|
/***
|
|
* use_del()
|
|
* Delete a redirection
|
|
*
|
|
* Args:
|
|
* dev - device OR unc-name to delete
|
|
* print_ok - print success message?
|
|
*
|
|
* Returns:
|
|
* 0 - success
|
|
* exit(2) - command failed
|
|
*/
|
|
VOID use_del(TCHAR * dev, BOOL deviceless, int print_ok)
|
|
{
|
|
ULONG ulErr; /* API return status */
|
|
BOOL fRememberSwitch = FALSE ;
|
|
ULONG bConnectFlags = 0L ;
|
|
|
|
USHORT i;
|
|
|
|
UseInit();
|
|
|
|
// Find out if the /PERSISTENT switch is used
|
|
for (i = 0; SwitchList[i]; i++)
|
|
{
|
|
if ( !(_tcscmp(SwitchList[i], swtxt_SW_USE_PERSISTENT)) )
|
|
ErrorExit(APE_InvalidSwitch);
|
|
}
|
|
|
|
if (IsWildCard(dev))
|
|
{
|
|
use_del_all() ;
|
|
goto gone;
|
|
}
|
|
|
|
bConnectFlags = CONNECT_UPDATE_PROFILE ; // deletes always update
|
|
|
|
if ((ulErr = WNetCancelConnection2( dev,
|
|
bConnectFlags,
|
|
FALSE)) != WN_SUCCESS)
|
|
{
|
|
if (ulErr != WN_OPEN_FILES)
|
|
WNetErrorExit(ulErr);
|
|
}
|
|
else
|
|
goto gone;
|
|
|
|
InfoPrintInsTxt(APE_OpenHandles, dev);
|
|
if (!YorN(APE_UseBlowAway, 0))
|
|
NetcmdExit(2);
|
|
|
|
|
|
if ((ulErr = WNetCancelConnection2( (LPWSTR)dev,
|
|
bConnectFlags,
|
|
TRUE )) != WN_SUCCESS)
|
|
WNetErrorExit(ulErr);
|
|
|
|
gone:
|
|
|
|
if (print_ok)
|
|
if ( IsWildCard( dev ) )
|
|
InfoSuccess();
|
|
else
|
|
InfoPrintInsTxt(APE_DelSuccess, dev);
|
|
}
|
|
|
|
/***
|
|
* use_del_all()
|
|
* Delete all redirections
|
|
*
|
|
* Args:
|
|
* none
|
|
*
|
|
* Returns:
|
|
* 0 - success
|
|
* exit(2) - command failed
|
|
*/
|
|
VOID use_del_all()
|
|
{
|
|
|
|
DWORD err = 0; /* API return status */
|
|
ULONG ulErr = 0; /* WNet error */
|
|
ULONG ulFirstErr = 0; /* WNet error */
|
|
DWORD num_read = 0; /* num entries read by API */
|
|
DWORD i = 0,j = 0;
|
|
DWORD NetUseInfoCount = 0 ;
|
|
NET_USE_INFO *NetUseInfoBuffer = NULL ;
|
|
NET_USE_INFO *pNetUseInfo = NULL;
|
|
ULONG bConnectFlags = 0L ;
|
|
|
|
UseInit();
|
|
|
|
if (err = MprUseEnum(&num_read, &NetUseInfoBuffer, &NetUseInfoCount))
|
|
ErrorExit(err);
|
|
|
|
if (err = UnavailUseAugment(&num_read, &NetUseInfoBuffer, &NetUseInfoCount))
|
|
ErrorExit(err);
|
|
|
|
if (err = LanmanUseAugment(num_read, NetUseInfoBuffer))
|
|
ErrorExit(err);
|
|
|
|
if (num_read == 0)
|
|
EmptyExit();
|
|
|
|
|
|
for (i = 0, pNetUseInfo = NetUseInfoBuffer;
|
|
i < num_read; i++, pNetUseInfo++)
|
|
{
|
|
//
|
|
// if we find at least one entry we will display, break
|
|
//
|
|
if (!(pNetUseInfo->fIsLanman)
|
|
|| (pNetUseInfo->dwStatus != USE_OK)
|
|
|| (pNetUseInfo->dwUseCount != 0)
|
|
|| (pNetUseInfo->dwRefCount != 0))
|
|
break;
|
|
}
|
|
|
|
qsort(NetUseInfoBuffer,
|
|
num_read,
|
|
sizeof(NET_USE_INFO), CmpUseInfo);
|
|
|
|
if (i != num_read)
|
|
{
|
|
InfoPrint(APE_KillDevList);
|
|
|
|
for (i = 0, pNetUseInfo = NetUseInfoBuffer;
|
|
i < num_read;
|
|
i++, pNetUseInfo++)
|
|
{
|
|
if (pNetUseInfo->lpLocalName[0] != NULLC)
|
|
WriteToCon(TEXT(" %Fws %Fws\r\n"),
|
|
PaddedString(15,pNetUseInfo->lpLocalName,NULL),
|
|
pNetUseInfo->lpRemoteName);
|
|
else
|
|
WriteToCon(TEXT(" %Fws %Fws\r\n"),
|
|
PaddedString(15,pNetUseInfo->lpLocalName,NULL),
|
|
pNetUseInfo->lpRemoteName);
|
|
|
|
}
|
|
|
|
InfoPrint(APE_KillCancel);
|
|
if (!YorN(APE_ProceedWOp, 0))
|
|
NetcmdExit(2);
|
|
}
|
|
|
|
bConnectFlags = CONNECT_UPDATE_PROFILE ; // deletes always update
|
|
|
|
ulErr = NO_ERROR;
|
|
ulFirstErr = NO_ERROR;
|
|
|
|
for (i = 0, pNetUseInfo = NetUseInfoBuffer;
|
|
i < num_read;
|
|
i++, pNetUseInfo++)
|
|
{
|
|
/* delete both local and UNC uses */
|
|
if (pNetUseInfo->lpLocalName[0] != NULLC)
|
|
{
|
|
ulErr = WNetCancelConnection2( pNetUseInfo->lpLocalName,
|
|
bConnectFlags,
|
|
FALSE);
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
* Delete All UNC uses to use_entry->ui1_remote
|
|
*/
|
|
if ( pNetUseInfo->dwUseCount == 0 )
|
|
{
|
|
pNetUseInfo->dwUseCount = 1;
|
|
}
|
|
for( j = 0; j < pNetUseInfo->dwUseCount; j++ )
|
|
{
|
|
ulErr = WNetCancelConnection2( pNetUseInfo->lpRemoteName,
|
|
bConnectFlags,
|
|
FALSE );
|
|
if ( ulErr != NO_ERROR )
|
|
break;
|
|
}
|
|
}
|
|
|
|
switch(ulErr)
|
|
{
|
|
case NO_ERROR:
|
|
/* The use was returned by Enum, but is already gone */
|
|
case WN_BAD_NETNAME:
|
|
case WN_NOT_CONNECTED:
|
|
break;
|
|
|
|
case WN_OPEN_FILES:
|
|
if (pNetUseInfo->lpLocalName[0] != NULLC)
|
|
IStrings[0] = pNetUseInfo->lpLocalName;
|
|
else
|
|
IStrings[0] = pNetUseInfo->lpRemoteName;
|
|
InfoPrintIns(APE_OpenHandles, 1);
|
|
if (!YorN(APE_UseBlowAway, 0))
|
|
continue;
|
|
|
|
if (pNetUseInfo->lpLocalName[0] != NULLC)
|
|
{
|
|
ulErr = WNetCancelConnection2( pNetUseInfo->lpLocalName,
|
|
bConnectFlags,
|
|
TRUE ) ;
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
* Delete All UNC uses to use_entry->ui1_remote
|
|
*/
|
|
for( j = 0; j < pNetUseInfo->dwUseCount; j++ )
|
|
{
|
|
ulErr = WNetCancelConnection2( pNetUseInfo->lpRemoteName,
|
|
bConnectFlags,
|
|
TRUE );
|
|
if ( ulErr != NO_ERROR )
|
|
break;
|
|
}
|
|
}
|
|
if (ulErr == NO_ERROR)
|
|
break;
|
|
// fall through
|
|
|
|
default:
|
|
ulFirstErr = ulErr;
|
|
}
|
|
}
|
|
|
|
FreeMem((LPBYTE)NetUseInfoBuffer);
|
|
|
|
if (ulFirstErr != NO_ERROR)
|
|
WNetErrorExit( ulErr );
|
|
|
|
}
|
|
|
|
|
|
|
|
/***
|
|
* LanmanDisplayUse()
|
|
* Display info from a USE_INFO_1 struct
|
|
*
|
|
* Args:
|
|
* use_entry - pointer to a USE_INFO_1 struct
|
|
*
|
|
* Returns:
|
|
* 0
|
|
*/
|
|
|
|
VOID
|
|
LanmanDisplayUse(
|
|
LPUSE_INFO_1 use_entry
|
|
)
|
|
{
|
|
DWORD maxLen;
|
|
USHORT status = APE2_GEN_UNKNOWN;
|
|
USHORT type = APE2_GEN_UNKNOWN;
|
|
|
|
switch(use_entry->ui1_asg_type)
|
|
{
|
|
case USE_DISKDEV:
|
|
type = APE2_USE_TYPE_DISK;
|
|
break;
|
|
case USE_SPOOLDEV:
|
|
type = APE2_USE_TYPE_PRINT;
|
|
break;
|
|
case USE_CHARDEV:
|
|
type = APE2_USE_TYPE_COMM;
|
|
break;
|
|
case USE_IPC:
|
|
type = APE2_USE_TYPE_IPC;
|
|
break;
|
|
}
|
|
|
|
UseMsgList[USE_TYPE_TBD].msg_number = type;
|
|
|
|
switch(use_entry->ui1_status)
|
|
{
|
|
case USE_OK:
|
|
status = APE2_USE_STATUS_OK;
|
|
break;
|
|
case USE_PAUSED:
|
|
status = APE2_USE_STATUS_PAUSED;
|
|
break;
|
|
case USE_SESSLOST:
|
|
status = APE2_USE_STATUS_SESSION_LOST;
|
|
break;
|
|
case USE_NETERR:
|
|
status = APE2_USE_STATUS_NET_ERROR;
|
|
break;
|
|
case USE_CONN:
|
|
status = APE2_USE_STATUS_CONNECTING;
|
|
break;
|
|
case USE_RECONN:
|
|
status = APE2_USE_STATUS_RECONNECTING;
|
|
break;
|
|
}
|
|
|
|
UseMsgList[USE_STATUS_TBD].msg_number = status;
|
|
|
|
GetMessageList(NUM_USE_MSGS, UseMsgList, &maxLen);
|
|
maxLen += 5;
|
|
|
|
WriteToCon(fmtPSZ,0,maxLen,
|
|
PaddedString(maxLen, UseMsgList[USE_MSG_LOCAL].msg_text, NULL),
|
|
use_entry->ui1_local);
|
|
|
|
WriteToCon(fmtPSZ,0,maxLen,
|
|
PaddedString(maxLen, UseMsgList[USE_MSG_REMOTE].msg_text, NULL),
|
|
use_entry->ui1_remote);
|
|
|
|
WriteToCon(fmtNPSZ,0,maxLen,
|
|
PaddedString(maxLen, UseMsgList[USE_MSG_TYPE].msg_text, NULL),
|
|
UseMsgList[USE_TYPE_TBD].msg_text);
|
|
|
|
WriteToCon(fmtNPSZ,0,maxLen,
|
|
PaddedString(maxLen, UseMsgList[USE_MSG_STATUS].msg_text, NULL),
|
|
UseMsgList[USE_STATUS_TBD].msg_text);
|
|
|
|
WriteToCon(fmtUSHORT,0,maxLen,
|
|
PaddedString(maxLen, UseMsgList[USE_MSG_OPEN_COUNT].msg_text, NULL),
|
|
use_entry->ui1_refcount);
|
|
|
|
WriteToCon(fmtUSHORT,0,maxLen,
|
|
PaddedString(maxLen, UseMsgList[USE_MSG_USE_COUNT].msg_text, NULL),
|
|
use_entry->ui1_usecount);
|
|
}
|
|
|
|
/***
|
|
* use_add_home()
|
|
* Add a use for the user's home directory
|
|
*
|
|
* Args:
|
|
* Dev - device to be used as the home directory
|
|
* Pass - password, or NULL if none supplied
|
|
*
|
|
* Returns:
|
|
* 0 - success
|
|
* exit(2) - command failed
|
|
*/
|
|
|
|
void
|
|
use_add_home(
|
|
LPTSTR Dev,
|
|
LPTSTR Pass
|
|
)
|
|
{
|
|
DWORD dwErr;
|
|
TCHAR HomeDir[PATHLEN];
|
|
TCHAR Server[MAX_PATH + 1];
|
|
TCHAR FAR *SubPath;
|
|
ULONG Type;
|
|
TCHAR User[UNLEN + 1];
|
|
TCHAR *DeviceName = Dev ;
|
|
LPWKSTA_INFO_1 info_entry_w;
|
|
LPUSER_INFO_11 user_entry;
|
|
|
|
//
|
|
// If necessary, start the workstation and logon.
|
|
//
|
|
UseInit();
|
|
|
|
//
|
|
// Get the user name and the name of the server that logged the user on.
|
|
//
|
|
dwErr = MNetWkstaGetInfo(1, (LPBYTE*) &info_entry_w) ;
|
|
|
|
if (dwErr)
|
|
{
|
|
ErrorExit(dwErr);
|
|
}
|
|
|
|
_tcscpy(User, info_entry_w->wki1_username);
|
|
_tcscpy(Server, TEXT("\\\\")) ;
|
|
_tcscat(Server, info_entry_w->wki1_logon_server);
|
|
NetApiBufferFree((TCHAR FAR *) info_entry_w);
|
|
|
|
/* Now get user information (ie. the home directory). If you were */
|
|
/* logged on STANDALONE, and the local machine is a STANDALONE */
|
|
/* server, this will still work. Otherwise (eg., you are logged on */
|
|
/* STANDALONE on a DOS workstation), it will fail because there is */
|
|
/* no local UAS. */
|
|
|
|
dwErr = NetUserGetInfo(Server, User, 11, (LPBYTE *) &user_entry);
|
|
|
|
if (dwErr)
|
|
{
|
|
ErrorExit(APE_UseHomeDirNotDetermined);
|
|
}
|
|
|
|
_tcscpy(HomeDir, user_entry->usri11_home_dir);
|
|
NetApiBufferFree((TCHAR FAR *) user_entry);
|
|
|
|
/* If it is null string, return a "not set" error msg. */
|
|
if (HomeDir[0] == NULLC)
|
|
ErrorExit(APE_UseHomeDirNotSet);
|
|
|
|
/* Make sure the home directory is a UNC name. This does not */
|
|
/* insure that the sharename is usable under DOS, but the */
|
|
/* general policy on this issue is to make it the admin's */
|
|
/* responsibility to be aware of non-8.3 sharename implications, */
|
|
/* and "net share" does issue a warning. */
|
|
dwErr = I_NetPathType(NULL, HomeDir, &Type, 0L);
|
|
if (dwErr || Type != ITYPE_UNC)
|
|
ErrorExitInsTxt(APE_UseHomeDirNotUNC, HomeDir);
|
|
|
|
/* Split the home directory into a remote name and a subpath. */
|
|
/* After doing this, HomeDir points to the remote name, and */
|
|
/* SubPath points to a subpath, or a null string if there is */
|
|
/* no subpath. */
|
|
|
|
/* Find the backslash between the computername and the sharename. */
|
|
SubPath = _tcschr(HomeDir + 2, '\\');
|
|
|
|
/* Find the next backslash, if there is one. */
|
|
SubPath = _tcschr(SubPath + 1, '\\');
|
|
|
|
if (SubPath)
|
|
{
|
|
*SubPath = NULLC;
|
|
SubPath++;
|
|
}
|
|
else
|
|
SubPath = NULL_STRING;
|
|
|
|
/* Map the wild cards as need */
|
|
if (DeviceName)
|
|
{
|
|
DeviceName = MapWildCard(DeviceName, NULL) ;
|
|
if (!DeviceName)
|
|
{
|
|
// this can only happen if no drives left
|
|
ErrorExit(APE_UseWildCardNoneLeft) ;
|
|
}
|
|
}
|
|
|
|
|
|
/* Do the use. If we return, we succeeded. */
|
|
use_add(DeviceName, HomeDir, Pass, FALSE, FALSE);
|
|
|
|
IStrings[0] = DeviceName;
|
|
IStrings[1] = HomeDir;
|
|
IStrings[2] = DeviceName;
|
|
IStrings[3] = SubPath;
|
|
|
|
InfoPrintIns(APE_UseHomeDirSuccess, 4);
|
|
return;
|
|
}
|
|
|
|
int NEAR is_admin_dollar(TCHAR FAR * name)
|
|
{
|
|
TCHAR FAR * tfpC;
|
|
|
|
tfpC = _tcspbrk(name + 2, TEXT("\\/"));
|
|
if (tfpC == NULL)
|
|
return(0);
|
|
tfpC += 1;
|
|
return(!_tcsicmp(ADMIN_DOLLAR, tfpC));
|
|
}
|
|
|
|
|
|
/***
|
|
* UseInit()
|
|
* Common initialization processing for all the use.c module entry
|
|
* points.
|
|
*
|
|
* Args: None.
|
|
*
|
|
* Returns: None.
|
|
*/
|
|
|
|
VOID NEAR
|
|
UseInit(VOID)
|
|
{
|
|
|
|
LanmanProviderName = GetLanmanProviderName() ;
|
|
if (LanmanProviderName == NULL)
|
|
LanmanProviderName = TEXT("") ;
|
|
}
|
|
|
|
|
|
/*
|
|
* query the user profile to see if connections are currently being remembered
|
|
*/
|
|
USHORT QueryDefaultPersistence(BOOL *pfRemember)
|
|
{
|
|
// by adding the two, we are guaranteed to have enough
|
|
TCHAR szAnswer[(sizeof(MPR_YES_VALUE)+sizeof(MPR_NO_VALUE))/sizeof(TCHAR)] ;
|
|
ULONG iRes, len ;
|
|
|
|
len = DIMENSION(szAnswer) ;
|
|
iRes = GetProfileString(MPR_NETWORK_SECTION,
|
|
MPR_SAVECONNECTION_KEY,
|
|
MPR_YES_VALUE, // default is yes
|
|
szAnswer,
|
|
len) ;
|
|
|
|
if (iRes == len) // error
|
|
return(APE_ProfileReadError) ;
|
|
|
|
*pfRemember = (_tcsicmp(szAnswer,MPR_YES_VALUE)==0) ;
|
|
return (NERR_Success) ;
|
|
}
|
|
|
|
/*
|
|
* set if connections are currently being remembered
|
|
*/
|
|
DWORD
|
|
SetDefaultPersistence(
|
|
BOOL fRemember
|
|
)
|
|
{
|
|
BOOL fSuccess ;
|
|
|
|
fSuccess = (WriteProfileString(MPR_NETWORK_SECTION,
|
|
MPR_SAVECONNECTION_KEY,
|
|
fRemember ? MPR_YES_VALUE : MPR_NO_VALUE ) != 0);
|
|
|
|
return (fSuccess ? NERR_Success : APE_ProfileWriteError) ;
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
* WNetErrorExit()
|
|
* maps the Winnet error to NERR and error exits
|
|
*
|
|
* arguments: ULONG win32 error code.
|
|
* return value: none. this baby dont return
|
|
*/
|
|
VOID
|
|
WNetErrorExit(
|
|
ULONG ulWNetErr
|
|
)
|
|
{
|
|
WCHAR w_ErrorText[256];
|
|
WCHAR w_ProviderText[64];
|
|
LONG ulExtendedError ;
|
|
DWORD err ;
|
|
|
|
switch (ulWNetErr)
|
|
{
|
|
case WN_SUCCESS :
|
|
return ;
|
|
|
|
case WN_BAD_POINTER :
|
|
case WN_BAD_VALUE :
|
|
err = ERROR_INVALID_PARAMETER ;
|
|
break ;
|
|
|
|
case WN_BAD_USER :
|
|
err = APE_BadUserContext ;
|
|
break ;
|
|
|
|
case WN_NO_NET_OR_BAD_PATH :
|
|
err = ERROR_BAD_NET_NAME ;
|
|
break ;
|
|
|
|
case WN_NO_NETWORK :
|
|
err = NERR_WkstaNotStarted ;
|
|
break ;
|
|
|
|
case WN_NOT_CONNECTED :
|
|
err = NERR_UseNotFound ;
|
|
break ;
|
|
|
|
case WN_DEVICE_IN_USE :
|
|
err = NERR_DevInUse ;
|
|
break ;
|
|
|
|
case WN_BAD_PROFILE :
|
|
case WN_CANNOT_OPEN_PROFILE :
|
|
err = APE_ProfileReadError ;
|
|
break ;
|
|
|
|
/*
|
|
* these should not happen under the calls we currently make,
|
|
* but for completeness, they are there.
|
|
*/
|
|
case WN_BAD_PROVIDER :
|
|
case WN_CONNECTION_CLOSED :
|
|
case WN_NOT_CONTAINER :
|
|
case WN_FUNCTION_BUSY :
|
|
case WN_DEVICE_ERROR :
|
|
err = ERROR_UNEXP_NET_ERR ;
|
|
break ;
|
|
|
|
/*
|
|
* special case this one
|
|
*/
|
|
case WN_EXTENDED_ERROR :
|
|
// get the extended error
|
|
ulWNetErr = WNetGetLastError(&ulExtendedError,
|
|
(LPWSTR)w_ErrorText,
|
|
DIMENSION(w_ErrorText),
|
|
(LPWSTR)w_ProviderText,
|
|
DIMENSION(w_ProviderText));
|
|
|
|
// if we got it, print it out
|
|
if (ulWNetErr == WN_SUCCESS)
|
|
{
|
|
TCHAR buf[16] ;
|
|
|
|
IStrings[0] = _ultow(ulExtendedError, buf, 10);
|
|
ErrorPrint(APE_OS2Error,1) ;
|
|
WriteToCon(TEXT("%ws\r\n"), w_ErrorText) ;
|
|
MyExit(2) ;
|
|
}
|
|
|
|
// otherwise report it as unexpected error
|
|
err = ERROR_UNEXP_NET_ERR ;
|
|
break ;
|
|
|
|
default:
|
|
// the the remainder dont need to be mapped.
|
|
err = ulWNetErr ;
|
|
break ;
|
|
}
|
|
|
|
ErrorExit(err) ;
|
|
}
|
|
|
|
|
|
/*
|
|
* code to handle the situation where the user has a remembered
|
|
* connection that is currently not used, and we need to figure out
|
|
* if we need to delete it first.
|
|
*
|
|
* returns whether we need update profile.
|
|
*/
|
|
BOOL CheckIfWantUpdate(TCHAR *dev, TCHAR *resource)
|
|
{
|
|
WCHAR w_RemoteName[MAX_PATH];
|
|
ULONG ulErr, cchRemoteName = DIMENSION(w_RemoteName);
|
|
|
|
// if deviceless, no problem since never remembered anyway.
|
|
if (dev == NULL)
|
|
return FALSE ;
|
|
|
|
// check out the device
|
|
ulErr = WNetGetConnection( (LPWSTR)dev,
|
|
(LPWSTR)w_RemoteName,
|
|
&cchRemoteName );
|
|
|
|
// device is really connected, bag out
|
|
if (ulErr == WN_SUCCESS)
|
|
ErrorExit(ERROR_ALREADY_ASSIGNED) ;
|
|
|
|
// it is an unavail remembered device, so prompt as need
|
|
if (ulErr == WN_CONNECTION_CLOSED)
|
|
{
|
|
// if the new and old are the same we just return FALSE
|
|
// since the user effectively does not change his profile
|
|
if (!_tcsicmp(w_RemoteName, resource))
|
|
{
|
|
return FALSE ;
|
|
}
|
|
|
|
// check if YES/NO switch is specified.
|
|
if (YorN_Switch == 2)
|
|
{
|
|
// he specified /NO, so we tell him why we bag out
|
|
IStrings[0] = dev ;
|
|
IStrings[1] = w_RemoteName ;
|
|
ErrorExitIns(APE_DeviceIsRemembered,2) ;
|
|
}
|
|
|
|
// nothing specified, so ask user
|
|
if (YorN_Switch == 0)
|
|
{
|
|
IStrings[0] = dev ;
|
|
IStrings[1] = w_RemoteName ;
|
|
if (!LUI_YorNIns(IStrings,2,APE_OverwriteRemembered,1))
|
|
{
|
|
// he said no, so quit
|
|
NetcmdExit(2) ;
|
|
}
|
|
}
|
|
|
|
// remove the persistent connection,
|
|
// we get here if the user specifies /YES, or didnt
|
|
// specify anything but consented,
|
|
if (WNetCancelConnection2( dev,
|
|
CONNECT_UPDATE_PROFILE,
|
|
FALSE) != WN_SUCCESS)
|
|
|
|
ErrorExit(APE_ProfileWriteError) ;
|
|
|
|
|
|
}
|
|
|
|
// if we get here then all is cool, let the caller carry on.
|
|
return TRUE ;
|
|
}
|
|
|
|
|
|
|
|
#define PROVIDER_NAME_LEN 256
|
|
#define PROVIDER_NAME_VALUE L"Name"
|
|
#define PROVIDER_NAME_KEY L"System\\CurrentControlSet\\Services\\LanmanWorkstation\\NetworkProvider"
|
|
|
|
/***
|
|
* GetLanmanProviderName()
|
|
* Reads the Lanman provider name from the registry.
|
|
* This is to make sure we only use the LM provider even
|
|
* if someone else supports UNC.
|
|
*
|
|
* Args:
|
|
* none
|
|
*
|
|
* Returns:
|
|
* pointer to provider name if success
|
|
* NULL if cannot read registry
|
|
* ErrorExit() for other errors.
|
|
*/
|
|
WCHAR *GetLanmanProviderName(void)
|
|
{
|
|
LONG Nerr;
|
|
LPBYTE buffer ;
|
|
HKEY hKey ;
|
|
DWORD buffersize, datatype ;
|
|
|
|
buffersize = PROVIDER_NAME_LEN * sizeof(WCHAR) ;
|
|
datatype = REG_SZ ;
|
|
if (Nerr = AllocMem(buffersize, &buffer))
|
|
ErrorExit(Nerr) ;
|
|
|
|
Nerr = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
PROVIDER_NAME_KEY,
|
|
0L,
|
|
KEY_QUERY_VALUE,
|
|
&hKey) ;
|
|
|
|
if (Nerr != ERROR_SUCCESS)
|
|
{
|
|
// if cannot read, we use NULL. this is more generous
|
|
// than normal, but at least the command will still work if
|
|
// we cannot get to this.
|
|
return NULL ;
|
|
}
|
|
|
|
Nerr = RegQueryValueExW(hKey,
|
|
PROVIDER_NAME_VALUE,
|
|
0L,
|
|
&datatype,
|
|
(LPBYTE)buffer,
|
|
&buffersize) ;
|
|
|
|
if (Nerr == ERROR_MORE_DATA)
|
|
{
|
|
if (Nerr = AllocMem(buffersize, &buffer))
|
|
{
|
|
RegCloseKey(hKey) ; // ignore any error here. its harmless
|
|
// and NET.EXE doesn't hang around.
|
|
ErrorExit(Nerr);
|
|
}
|
|
|
|
Nerr = RegQueryValueExW(hKey,
|
|
PROVIDER_NAME_VALUE,
|
|
0L,
|
|
&datatype,
|
|
(LPBYTE)buffer,
|
|
&buffersize) ;
|
|
|
|
}
|
|
|
|
(void) RegCloseKey(hKey) ; // ignore any error here. its harmless
|
|
// and NET.EXE doesnt hang around anyway.
|
|
|
|
if (Nerr != ERROR_SUCCESS)
|
|
{
|
|
return(NULL) ; // treat as cannot read
|
|
}
|
|
|
|
|
|
return ((WCHAR *) buffer);
|
|
}
|
|
|
|
/***
|
|
* MapWildCard()
|
|
* Maps the wildcard ASTERISK to next avail drive.
|
|
*
|
|
* Args:
|
|
* dev - the input string. Must NOT be NULL.
|
|
*
|
|
* Returns:
|
|
* dev unchanged if it is not the wildcard
|
|
* pointer to next avail drive id dev is wildcard
|
|
* NULL if there are no avail drives left
|
|
*/
|
|
LPTSTR
|
|
MapWildCard(
|
|
LPTSTR dev,
|
|
LPTSTR startdev
|
|
)
|
|
{
|
|
static TCHAR new_dev[DEVLEN+1] ;
|
|
|
|
//
|
|
// if not the wold card char, just return it unchanged
|
|
//
|
|
if (!IsWildCard(dev))
|
|
{
|
|
return dev ;
|
|
}
|
|
|
|
//
|
|
// need find the next avail drive letter
|
|
// note: the char advance does not need to be DBCS safe,
|
|
// since we are only dealing with drive letters.
|
|
//
|
|
if ( startdev != NULL )
|
|
{
|
|
_tcscpy(new_dev, startdev);
|
|
}
|
|
else
|
|
{
|
|
_tcscpy(new_dev,TEXT("Z:\\")) ;
|
|
}
|
|
|
|
while ( TRUE )
|
|
{
|
|
if (GetDriveType(new_dev) == 1) // 1 means root not found
|
|
{
|
|
//
|
|
// check if it's a remembered connection
|
|
//
|
|
DWORD status;
|
|
TCHAR remote_name[40]; // length doesn't matter since we
|
|
// check for WN_MORE_DATA
|
|
DWORD length = sizeof(remote_name)/sizeof(TCHAR);
|
|
|
|
new_dev[2] = 0 ;
|
|
|
|
status = WNetGetConnection(new_dev, remote_name, &length);
|
|
if (status == WN_CONNECTION_CLOSED ||
|
|
status == WN_MORE_DATA ||
|
|
status == WN_SUCCESS)
|
|
{
|
|
//
|
|
// it's a remembered connection; try the next drive
|
|
//
|
|
new_dev[2] = TEXT('\\');
|
|
}
|
|
else
|
|
{
|
|
return (new_dev) ;
|
|
}
|
|
}
|
|
|
|
if ( new_dev[0] == 'c' || new_dev[0] == 'C' )
|
|
{
|
|
break;
|
|
}
|
|
|
|
--new_dev[0] ;
|
|
}
|
|
|
|
//
|
|
// if we got here, there were no drives left
|
|
//
|
|
return NULL ;
|
|
}
|
|
|