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.
1202 lines
26 KiB
1202 lines
26 KiB
/*
|
|
File utils.c
|
|
|
|
Contains common utilities for the ras dialup server ui.
|
|
|
|
Paul Mayfield, 9/30/97
|
|
*/
|
|
|
|
#include "rassrv.h"
|
|
|
|
// Remoteaccess parameters key
|
|
const WCHAR pszregRasParameters[]
|
|
= L"SYSTEM\\CurrentControlSet\\Services\\RemoteAccess\\Parameters";
|
|
|
|
// Registry key values
|
|
const WCHAR pszregServerFlags[] = L"ServerFlags";
|
|
const WCHAR pszregShowIcon[] = L"Rassrv_EnableIconsInTray";
|
|
const WCHAR pszregPure[] = L"UsersConfiguredWithMMC";
|
|
const WCHAR pszregLogLevel[] = L"LoggingFlags";
|
|
|
|
// Here is the instance of the global variables
|
|
RASSRVUI_GLOBALS Globals;
|
|
|
|
DWORD
|
|
gblInit(
|
|
IN HINSTANCE hInstDll,
|
|
OUT RASSRVUI_GLOBALS * pGlobs)
|
|
{
|
|
// Clear out the memory
|
|
ZeroMemory(pGlobs, sizeof(RASSRVUI_GLOBALS));
|
|
|
|
// Record the module for use in future resource fuction calls.
|
|
Globals.hInstDll = hInstDll;
|
|
|
|
// Initialize the global variable lock
|
|
InitializeCriticalSection(&(pGlobs->csLock));
|
|
|
|
// Create the global heap
|
|
pGlobs->hPrivateHeap = HeapCreate(0, 4096, 0);
|
|
|
|
// Register the context ID atom for use in the Windows XxxProp calls
|
|
// which are used to associate a context with a dialog window handle.
|
|
Globals.atmRassrvPageData =
|
|
(LPCTSTR)GlobalAddAtom(TEXT("RASSRVUI_PAGE_DATA"));
|
|
if (!Globals.atmRassrvPageData)
|
|
{
|
|
return GetLastError();
|
|
}
|
|
Globals.atmRassrvPageId =
|
|
(LPCTSTR)GlobalAddAtom(TEXT("RASSRVUI_PAGE_ID"));
|
|
if (!Globals.atmRassrvPageId)
|
|
{
|
|
return GetLastError();
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
DWORD
|
|
gblCleanup(
|
|
IN RASSRVUI_GLOBALS * Globs)
|
|
{
|
|
if (Globs->hRasServer != NULL)
|
|
{
|
|
MprAdminServerDisconnect(Globs->hRasServer);
|
|
Globs->hRasServer = NULL;
|
|
}
|
|
|
|
if (Globs->hPrivateHeap)
|
|
{
|
|
HeapDestroy(Globs->hPrivateHeap);
|
|
}
|
|
|
|
GlobalDeleteAtom(LOWORD(Globals.atmRassrvPageData));
|
|
GlobalDeleteAtom(LOWORD(Globals.atmRassrvPageId));
|
|
|
|
DeleteCriticalSection(&(Globs->csLock));
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
//
|
|
// Loads the machine flags
|
|
//
|
|
DWORD
|
|
gblLoadMachineFlags(
|
|
IN RASSRVUI_GLOBALS * pGlobs)
|
|
{
|
|
DWORD dwErr = NO_ERROR;
|
|
PDSROLE_PRIMARY_DOMAIN_INFO_BASIC pInfo = NULL;
|
|
BOOL bEnabled, bDefault;
|
|
|
|
// If we're already initialized, there's nothing to
|
|
// do
|
|
//
|
|
if (pGlobs->dwMachineFlags & RASSRVUI_MACHINE_F_Initialized)
|
|
{
|
|
return NO_ERROR;
|
|
}
|
|
|
|
do
|
|
{
|
|
// Find out what kind of machine we are
|
|
//
|
|
dwErr = DsRoleGetPrimaryDomainInformation(
|
|
NULL,
|
|
DsRolePrimaryDomainInfoBasic,
|
|
(LPBYTE *)&pInfo );
|
|
|
|
if (dwErr != NO_ERROR)
|
|
{
|
|
break;
|
|
}
|
|
|
|
if ((pInfo->MachineRole != DsRole_RoleStandaloneWorkstation) &&
|
|
(pInfo->MachineRole != DsRole_RoleMemberWorkstation))
|
|
{
|
|
pGlobs->dwMachineFlags |= RASSRVUI_MACHINE_F_Server;
|
|
}
|
|
|
|
if ((pInfo->MachineRole != DsRole_RoleStandaloneWorkstation) &&
|
|
(pInfo->MachineRole != DsRole_RoleStandaloneServer))
|
|
{
|
|
pGlobs->dwMachineFlags |= RASSRVUI_MACHINE_F_Member;
|
|
}
|
|
|
|
// Record that we've been initailized
|
|
//
|
|
pGlobs->dwMachineFlags |= RASSRVUI_MACHINE_F_Initialized;
|
|
|
|
} while (FALSE);
|
|
|
|
// Cleanup
|
|
{
|
|
if (pInfo)
|
|
{
|
|
DsRoleFreeMemory (pInfo);
|
|
}
|
|
}
|
|
|
|
return dwErr;
|
|
}
|
|
|
|
//
|
|
// Establishes communication with the ras server if
|
|
// not already established
|
|
//
|
|
DWORD
|
|
gblConnectToRasServer()
|
|
{
|
|
DWORD dwErr = NO_ERROR;;
|
|
|
|
EnterCriticalSection(&(Globals.csLock));
|
|
|
|
if (Globals.hRasServer == NULL)
|
|
{
|
|
dwErr = MprAdminServerConnect(NULL, &Globals.hRasServer);
|
|
}
|
|
|
|
LeaveCriticalSection(&(Globals.csLock));
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
/* Enhanced list view callback to report drawing information. 'HwndLv' is
|
|
** the handle of the list view control. 'DwItem' is the index of the item
|
|
** being drawn.
|
|
**
|
|
** Returns the address of the draw information.
|
|
*/
|
|
LVXDRAWINFO*
|
|
LvDrawInfoCallback(
|
|
IN HWND hwndLv,
|
|
IN DWORD dwItem )
|
|
{
|
|
/* The enhanced list view is used only to get the "wide selection bar"
|
|
** feature so our option list is not very interesting.
|
|
**
|
|
** Fields are 'nCols', 'dxIndent', 'dwFlags', 'adwFlags[]'.
|
|
*/
|
|
static LVXDRAWINFO info = { 1, 0, 0, { 0 } };
|
|
|
|
return &info;
|
|
}
|
|
|
|
//
|
|
// Allocates memory. If bZero is TRUE, it also zeros the memory.
|
|
//
|
|
PVOID
|
|
RassrvAlloc (
|
|
IN DWORD dwSize,
|
|
IN BOOL bZero)
|
|
{
|
|
PVOID pvRet = NULL;
|
|
HANDLE hHeap = NULL;
|
|
|
|
hHeap =
|
|
(Globals.hPrivateHeap) ? Globals.hPrivateHeap : GetProcessHeap();
|
|
|
|
pvRet = HeapAlloc(
|
|
hHeap,
|
|
(bZero) ? HEAP_ZERO_MEMORY: 0,
|
|
dwSize);
|
|
|
|
return pvRet;
|
|
}
|
|
|
|
//
|
|
// Frees memory allocated by RassrvAlloc
|
|
//
|
|
VOID
|
|
RassrvFree (
|
|
IN PVOID pvBuf)
|
|
{
|
|
PVOID pvRet;
|
|
HANDLE hHeap;
|
|
|
|
hHeap =
|
|
(Globals.hPrivateHeap) ? Globals.hPrivateHeap : GetProcessHeap();
|
|
|
|
if (pvBuf)
|
|
{
|
|
HeapFree(hHeap, 0, pvBuf);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Adds a user to the local machine
|
|
//
|
|
DWORD
|
|
RasSrvAddUser (
|
|
IN PWCHAR pszUserLogonName,
|
|
IN PWCHAR pszUserComment,
|
|
IN PWCHAR pszUserPassword)
|
|
{
|
|
NET_API_STATUS nStatus;
|
|
WCHAR pszDomainUser[1024];
|
|
WCHAR pszCompName[1024];
|
|
LOCALGROUP_MEMBERS_INFO_3 meminfo;
|
|
DWORD dwSize = 1024, dwErr;
|
|
USER_INFO_2 * pUser2;
|
|
RAS_USER_0 UserInfo;
|
|
|
|
// Initialize the base user information
|
|
USER_INFO_1 User =
|
|
{
|
|
pszUserLogonName,
|
|
pszUserPassword,
|
|
0,
|
|
USER_PRIV_USER,
|
|
L"",
|
|
L"",
|
|
UF_SCRIPT | UF_DONT_EXPIRE_PASSWD | UF_NORMAL_ACCOUNT,
|
|
L""
|
|
};
|
|
|
|
// Add the user
|
|
nStatus = NetUserAdd(
|
|
NULL,
|
|
1,
|
|
(LPBYTE)&User,
|
|
NULL);
|
|
|
|
// If the user wasn't added, find out why
|
|
if (nStatus != NERR_Success)
|
|
{
|
|
switch (nStatus)
|
|
{
|
|
case ERROR_ACCESS_DENIED:
|
|
return ERROR_ACCESS_DENIED;
|
|
|
|
case NERR_UserExists:
|
|
return ERROR_USER_EXISTS;
|
|
|
|
case NERR_PasswordTooShort:
|
|
return ERROR_INVALID_PASSWORDNAME;
|
|
|
|
case NERR_InvalidComputer:
|
|
case NERR_NotPrimary:
|
|
case NERR_GroupExists:
|
|
default:
|
|
return ERROR_CAN_NOT_COMPLETE;
|
|
}
|
|
}
|
|
|
|
// Now that the user is added, add the user's full name
|
|
nStatus = NetUserGetInfo(NULL, pszUserLogonName, 2, (LPBYTE*)&pUser2);
|
|
if (nStatus == NERR_Success)
|
|
{
|
|
// Modify the full name in the structure
|
|
pUser2->usri2_full_name = pszUserComment;
|
|
NetUserSetInfo(NULL, pszUserLogonName, 2, (LPBYTE)pUser2, NULL);
|
|
NetApiBufferFree((LPBYTE)pUser2);
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
//
|
|
// Deletes a user from the system local user datbase
|
|
//
|
|
DWORD
|
|
RasSrvDeleteUser(
|
|
PWCHAR pszUserLogonName)
|
|
{
|
|
NET_API_STATUS nStatus;
|
|
|
|
// Delete the user and return the status code. If the
|
|
// specified user is not in the user database, consider
|
|
// it a success
|
|
nStatus = NetUserDel(NULL, pszUserLogonName);
|
|
if (nStatus != NERR_Success)
|
|
{
|
|
switch (nStatus)
|
|
{
|
|
case ERROR_ACCESS_DENIED:
|
|
return ERROR_ACCESS_DENIED;
|
|
|
|
case NERR_UserNotFound:
|
|
return NO_ERROR;
|
|
}
|
|
return nStatus;
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
//
|
|
// Changes the full name and password of a user. If
|
|
// either of pszFullName or pszPassword is null, it is
|
|
// ignored.
|
|
//
|
|
DWORD
|
|
RasSrvEditUser (
|
|
IN PWCHAR pszLogonName,
|
|
IN OPTIONAL PWCHAR pszFullName,
|
|
IN OPTIONAL PWCHAR pszPassword)
|
|
{
|
|
NET_API_STATUS nStatus;
|
|
DWORD dwSize = 1024, dwErr = NO_ERROR, dwParamErr;
|
|
USER_INFO_2 * pUser2;
|
|
|
|
// if nothing to set, return
|
|
if (!pszFullName && !pszPassword)
|
|
{
|
|
return NO_ERROR;
|
|
}
|
|
|
|
// First, get this user's data so that we can manipulate it.
|
|
//
|
|
nStatus = NetUserGetInfo(
|
|
NULL,
|
|
pszLogonName,
|
|
2,
|
|
(LPBYTE*)(&pUser2));
|
|
if (nStatus != NERR_Success)
|
|
{
|
|
return nStatus;
|
|
}
|
|
|
|
dwErr = NO_ERROR;
|
|
do
|
|
{
|
|
// Fill in the blanks accordingly
|
|
if (pszFullName)
|
|
{
|
|
pUser2->usri2_full_name = pszFullName;
|
|
}
|
|
|
|
if (pszPassword)
|
|
{
|
|
pUser2->usri2_password = pszPassword;
|
|
}
|
|
|
|
// Add the user
|
|
nStatus = NetUserSetInfo(
|
|
NULL, // server name
|
|
pszLogonName, // user name
|
|
2, // level
|
|
(LPBYTE)pUser2, // buf
|
|
&dwParamErr); // param error
|
|
if (nStatus != NERR_Success)
|
|
{
|
|
dwErr = nStatus;
|
|
break;
|
|
}
|
|
|
|
} while (FALSE);
|
|
|
|
// Cleanup
|
|
{
|
|
NetApiBufferFree(pUser2);
|
|
}
|
|
|
|
return dwErr;
|
|
}
|
|
|
|
// Returns whether a dword registry value was set or not. If the named
|
|
// value does not exist, the value of bDefault is assigned.
|
|
DWORD
|
|
RassrvRegGetDwEx(
|
|
IN DWORD * lpdwFlag,
|
|
IN DWORD dwDefault,
|
|
IN CONST PWCHAR pszKeyName,
|
|
IN CONST PWCHAR pszValueName,
|
|
IN BOOL bCreate)
|
|
{
|
|
DWORD dwErr, dwVal, dwType = REG_DWORD, dwSize = sizeof(DWORD);
|
|
HKEY hKey = NULL;
|
|
|
|
if (!lpdwFlag)
|
|
{
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
do
|
|
{
|
|
if (bCreate)
|
|
{
|
|
DWORD dwDisposition;
|
|
|
|
dwErr = RegCreateKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
pszKeyName,
|
|
0,
|
|
NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_READ,
|
|
NULL,
|
|
&hKey,
|
|
&dwDisposition);
|
|
if (dwErr != ERROR_SUCCESS)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Open the registry key
|
|
dwErr = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
pszKeyName,
|
|
0,
|
|
KEY_READ,
|
|
&hKey);
|
|
if (dwErr != ERROR_SUCCESS)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Read the value
|
|
dwErr = RegQueryValueExW(
|
|
hKey,
|
|
pszValueName,
|
|
0,
|
|
&dwType,
|
|
(BYTE *)&dwVal,
|
|
&dwSize);
|
|
if (dwErr != ERROR_SUCCESS)
|
|
{
|
|
dwErr = NO_ERROR;
|
|
dwVal = dwDefault;
|
|
}
|
|
|
|
// Return the value read
|
|
*lpdwFlag = dwVal;
|
|
|
|
} while (FALSE);
|
|
|
|
// Cleanup
|
|
{
|
|
if (hKey)
|
|
{
|
|
RegCloseKey(hKey);
|
|
}
|
|
}
|
|
|
|
return dwErr;
|
|
}
|
|
|
|
|
|
// Returns whether a dword registry value was set or not. If the named
|
|
// value does not exist, the value of bDefault is assigned.
|
|
DWORD
|
|
RassrvRegGetDw(
|
|
IN DWORD * lpdwFlag,
|
|
IN DWORD dwDefault,
|
|
IN CONST PWCHAR pszKeyName,
|
|
IN CONST PWCHAR pszValueName)
|
|
{
|
|
return RassrvRegGetDwEx(
|
|
lpdwFlag,
|
|
dwDefault,
|
|
pszKeyName,
|
|
pszValueName,
|
|
FALSE);
|
|
}
|
|
|
|
//
|
|
// Sets a dword registry value. If the named value does not exist,
|
|
// it is automatically created.
|
|
//
|
|
DWORD
|
|
RassrvRegSetDwEx(
|
|
IN DWORD dwFlag,
|
|
IN CONST PWCHAR pszKeyName,
|
|
IN CONST PWCHAR pszValueName,
|
|
IN BOOL bCreate)
|
|
{
|
|
DWORD dwErr = NO_ERROR, dwVal, dwType = REG_DWORD;
|
|
DWORD dwSize = sizeof(DWORD);
|
|
HKEY hKey = NULL;
|
|
|
|
dwVal = dwFlag;
|
|
|
|
do
|
|
{
|
|
if (bCreate)
|
|
{
|
|
DWORD dwDisposition;
|
|
|
|
dwErr = RegCreateKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
pszKeyName,
|
|
0,
|
|
NULL,
|
|
REG_OPTION_NON_VOLATILE,
|
|
KEY_WRITE,
|
|
NULL,
|
|
&hKey,
|
|
&dwDisposition);
|
|
if (dwErr != ERROR_SUCCESS)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Open the registry key
|
|
dwErr = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
pszKeyName,
|
|
0,
|
|
KEY_WRITE,
|
|
&hKey);
|
|
if (dwErr != ERROR_SUCCESS)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Set the value
|
|
dwErr = RegSetValueExW(
|
|
hKey,
|
|
pszValueName,
|
|
0,
|
|
dwType,
|
|
(CONST BYTE *)&dwVal,
|
|
dwSize);
|
|
if (dwErr != ERROR_SUCCESS)
|
|
{
|
|
break;
|
|
}
|
|
|
|
} while (FALSE);
|
|
|
|
// Cleanup
|
|
{
|
|
if (hKey)
|
|
{
|
|
RegCloseKey(hKey);
|
|
}
|
|
}
|
|
|
|
return dwErr;
|
|
}
|
|
|
|
DWORD
|
|
RassrvRegSetDw(
|
|
IN DWORD dwFlag,
|
|
IN CONST PWCHAR pszKeyName,
|
|
IN CONST PWCHAR pszValueName)
|
|
{
|
|
return RassrvRegSetDwEx(dwFlag, pszKeyName, pszValueName, FALSE);
|
|
}
|
|
|
|
//
|
|
// Warns the user that we are about to switch to MMC returning TRUE if
|
|
// user agrees to this and FALSE otherwise
|
|
//
|
|
BOOL
|
|
RassrvWarnMMCSwitch(
|
|
IN HWND hwndDlg)
|
|
{
|
|
PWCHAR pszWarning, pszTitle;
|
|
|
|
pszWarning =
|
|
(PWCHAR) PszLoadString(Globals.hInstDll, WRN_SWITCHING_TO_MMC);
|
|
pszTitle =
|
|
(PWCHAR) PszLoadString(Globals.hInstDll, WRN_TITLE);
|
|
|
|
if (MessageBox(
|
|
hwndDlg,
|
|
pszWarning,
|
|
pszTitle,
|
|
MB_YESNO | MB_ICONEXCLAMATION) == IDYES)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// Switches to mmc based on the console identifier passed in
|
|
//
|
|
DWORD
|
|
RassrvLaunchMMC (
|
|
IN DWORD dwConsoleId)
|
|
{
|
|
STARTUPINFOA startupinfo;
|
|
PROCESS_INFORMATION procinfo;
|
|
CHAR * pszConsole;
|
|
CHAR pszBuf[1024], pszDir[1024];
|
|
|
|
// Set the command line accordingly
|
|
switch (dwConsoleId)
|
|
{
|
|
case RASSRVUI_NETWORKCONSOLE:
|
|
pszConsole = "netmgmt.msc";
|
|
break;
|
|
|
|
case RASSRVUI_USERCONSOLE:
|
|
pszConsole = NULL;
|
|
break;
|
|
|
|
case RASSRVUI_SERVICESCONSOLE:
|
|
pszConsole = "compmgmt.msc";
|
|
break;
|
|
|
|
case RASSRVUI_MPRCONSOLE:
|
|
default:
|
|
pszConsole = "rrasmgmt.msc";
|
|
break;
|
|
}
|
|
|
|
if (pszConsole)
|
|
{
|
|
GetSystemDirectoryA (pszDir, sizeof(pszDir));
|
|
sprintf (pszBuf, "mmc %s\\%s", pszDir, pszConsole);
|
|
}
|
|
else
|
|
strcpy (pszBuf, "mmc.exe");
|
|
|
|
// Launch MMC
|
|
ZeroMemory(&startupinfo, sizeof(startupinfo));
|
|
startupinfo.cb = sizeof(startupinfo);
|
|
CreateProcessA(
|
|
NULL, // name of executable module
|
|
pszBuf, // command line string
|
|
NULL, // process security attributes
|
|
NULL, // thread security attributes
|
|
FALSE, // handle inheritance flag
|
|
NORMAL_PRIORITY_CLASS, // creation flags
|
|
NULL, // new environment block
|
|
NULL, // current directory name
|
|
&startupinfo, // STARTUPINFO
|
|
&procinfo); // PROCESS_INFORMATION
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
//
|
|
// Retrieve a string from the registry
|
|
//
|
|
DWORD
|
|
RassrvRegGetStr(
|
|
OUT PWCHAR pszBuf,
|
|
IN PWCHAR pszDefault,
|
|
IN CONST PWCHAR pszKeyName,
|
|
IN CONST PWCHAR pszValueName)
|
|
{
|
|
DWORD dwErr = NO_ERROR, dwVal, dwType = REG_SZ, dwSize = 512;
|
|
HKEY hKey = NULL;
|
|
|
|
do
|
|
{
|
|
// Open the registry key
|
|
dwErr = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
pszKeyName,
|
|
0,
|
|
KEY_READ,
|
|
&hKey);
|
|
if (dwErr != ERROR_SUCCESS)
|
|
{
|
|
break;
|
|
}
|
|
|
|
// Read the value
|
|
dwErr = RegQueryValueExW(
|
|
hKey,
|
|
pszValueName,
|
|
0,
|
|
&dwType,
|
|
(BYTE *)pszBuf,
|
|
&dwSize);
|
|
if (dwErr != ERROR_SUCCESS)
|
|
{
|
|
dwErr = NO_ERROR;
|
|
wcscpy(pszBuf, pszDefault);
|
|
}
|
|
|
|
} while (FALSE);
|
|
|
|
// Cleanup
|
|
{
|
|
if (hKey)
|
|
{
|
|
RegCloseKey(hKey);
|
|
}
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
//
|
|
// Save a string to the registry
|
|
//
|
|
DWORD
|
|
RassrvRegSetStr(
|
|
IN PWCHAR pszStr,
|
|
IN CONST PWCHAR pszKeyName,
|
|
IN CONST PWCHAR pszValueName)
|
|
{
|
|
DWORD dwErr = NO_ERROR, dwVal, dwType = REG_SZ, dwSize;
|
|
HKEY hKey = NULL;
|
|
|
|
dwSize = wcslen(pszStr)*sizeof(WCHAR) + sizeof(WCHAR);
|
|
|
|
do
|
|
{
|
|
// Open the registry key
|
|
dwErr = RegOpenKeyExW(
|
|
HKEY_LOCAL_MACHINE,
|
|
pszKeyName,
|
|
0,
|
|
KEY_WRITE,
|
|
&hKey);
|
|
if (dwErr != ERROR_SUCCESS)
|
|
{
|
|
break;
|
|
}
|
|
|
|
// Set the value
|
|
dwErr = RegSetValueExW(
|
|
hKey,
|
|
pszValueName,
|
|
0,
|
|
dwType,
|
|
(CONST BYTE *)pszStr,
|
|
dwSize);
|
|
|
|
if (dwErr != ERROR_SUCCESS)
|
|
{
|
|
break;
|
|
}
|
|
|
|
} while (FALSE);
|
|
|
|
// Cleanup
|
|
{
|
|
if (hKey)
|
|
{
|
|
RegCloseKey(hKey);
|
|
}
|
|
}
|
|
|
|
return dwErr;
|
|
}
|
|
|
|
//
|
|
// Gets the machine flags
|
|
//
|
|
DWORD
|
|
RasSrvGetMachineFlags(
|
|
OUT LPDWORD lpdwFlags)
|
|
{
|
|
GBL_LOCK;
|
|
|
|
gblLoadMachineFlags(&Globals);
|
|
*lpdwFlags = Globals.dwMachineFlags;
|
|
|
|
GBL_UNLOCK;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
//
|
|
// Get multilink status
|
|
//
|
|
DWORD
|
|
RasSrvGetMultilink(
|
|
OUT BOOL * pbEnabled)
|
|
{
|
|
DWORD dwFlags = PPPCFG_NegotiateMultilink;
|
|
|
|
if (!pbEnabled)
|
|
{
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
// Read the flags
|
|
RassrvRegGetDw(
|
|
&dwFlags,
|
|
PPPCFG_NegotiateMultilink,
|
|
(const PWCHAR)pszregRasParameters,
|
|
(const PWCHAR)pszregServerFlags);
|
|
|
|
// Assign the enable state accordingly
|
|
if (dwFlags & PPPCFG_NegotiateMultilink)
|
|
{
|
|
*pbEnabled = TRUE;
|
|
}
|
|
else
|
|
{
|
|
*pbEnabled = FALSE;
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
//
|
|
// Private internal function that enables/disables multilink
|
|
//
|
|
DWORD
|
|
RasSrvSetMultilink(
|
|
IN BOOL bEnable)
|
|
{
|
|
DWORD dwFlags = PPPCFG_NegotiateMultilink;
|
|
|
|
// Read the flags
|
|
RassrvRegGetDw(
|
|
&dwFlags,
|
|
PPPCFG_NegotiateMultilink,
|
|
(const PWCHAR)pszregRasParameters,
|
|
(const PWCHAR)pszregServerFlags);
|
|
|
|
// Assign the enable state accordingly
|
|
if (bEnable)
|
|
{
|
|
dwFlags |= PPPCFG_NegotiateMultilink;
|
|
}
|
|
else
|
|
{
|
|
dwFlags &= ~PPPCFG_NegotiateMultilink;
|
|
}
|
|
|
|
// Set the flags
|
|
RassrvRegSetDw(
|
|
dwFlags,
|
|
(CONST PWCHAR)pszregRasParameters,
|
|
(CONST PWCHAR)pszregServerFlags);
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
//
|
|
// Initialize the show icon setting
|
|
//
|
|
DWORD
|
|
RasSrvGetIconShow(
|
|
OUT BOOL * pbEnabled)
|
|
{
|
|
DWORD dwErr = NO_ERROR, dwFlags = 0;
|
|
BOOL bDefault = TRUE;
|
|
|
|
// Get machine flags
|
|
//
|
|
dwErr = RasSrvGetMachineFlags(&dwFlags);
|
|
if (dwErr != NO_ERROR)
|
|
{
|
|
*pbEnabled = FALSE;
|
|
return dwErr;
|
|
}
|
|
|
|
// Always off for member server
|
|
//
|
|
if ((dwFlags & RASSRVUI_MACHINE_F_Server) &&
|
|
(dwFlags & RASSRVUI_MACHINE_F_Member))
|
|
{
|
|
*pbEnabled = FALSE;
|
|
return NO_ERROR;
|
|
}
|
|
|
|
// Set default
|
|
//
|
|
if (dwFlags & RASSRVUI_MACHINE_F_Server)
|
|
{
|
|
bDefault = FALSE;
|
|
}
|
|
else
|
|
{
|
|
bDefault = TRUE;
|
|
}
|
|
|
|
// Load the machine flags and return accordingly
|
|
//
|
|
*pbEnabled = bDefault;
|
|
dwErr = RassrvRegGetDw(
|
|
pbEnabled,
|
|
bDefault,
|
|
(CONST PWCHAR)pszregRasParameters,
|
|
(CONST PWCHAR)pszregShowIcon);
|
|
|
|
return dwErr;
|
|
}
|
|
|
|
//
|
|
// Save the show icon setting
|
|
//
|
|
DWORD
|
|
RasSrvSetIconShow(
|
|
IN BOOL bEnable)
|
|
{
|
|
return RassrvRegSetDw(
|
|
bEnable,
|
|
(CONST PWCHAR)pszregRasParameters,
|
|
(CONST PWCHAR)pszregShowIcon);
|
|
}
|
|
|
|
//
|
|
// Save the log level
|
|
//
|
|
DWORD
|
|
RasSrvSetLogLevel(
|
|
IN DWORD dwLevel)
|
|
{
|
|
return RassrvRegSetDw(
|
|
dwLevel,
|
|
(CONST PWCHAR)pszregRasParameters,
|
|
(CONST PWCHAR)pszregLogLevel);
|
|
}
|
|
|
|
// Calls WinHelp to popup context sensitive help. 'pdwMap' is an array
|
|
// of control-ID help-ID pairs terminated with a 0,0 pair. 'UnMsg' is
|
|
// WM_HELP or WM_CONTEXTMENU indicating the message received requesting
|
|
// help. 'Wparam' and 'lparam' are the parameters of the message received
|
|
// requesting help.
|
|
DWORD
|
|
RasSrvHelp (
|
|
IN HWND hwndDlg,
|
|
IN UINT unMsg,
|
|
IN WPARAM wparam,
|
|
IN LPARAM lparam,
|
|
IN const DWORD* pdwMap)
|
|
{
|
|
HWND hwnd;
|
|
UINT unType;
|
|
TCHAR pszHelpFile[] = TEXT("Netcfg.hlp");
|
|
|
|
// Validate parameters
|
|
if (! (unMsg==WM_HELP || unMsg==WM_CONTEXTMENU))
|
|
{
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
// If no map is provided, no help will show
|
|
if (!pdwMap)
|
|
{
|
|
return NO_ERROR;
|
|
}
|
|
|
|
// If an actual help topic is request...
|
|
if (unMsg == WM_HELP)
|
|
{
|
|
LPHELPINFO p = (LPHELPINFO )lparam;
|
|
|
|
TRACE4( "ContextHelp(WM_HELP,t=%d,id=%d,h=$%08x,s=$%08x)",
|
|
p->iContextType, p->iCtrlId,p->hItemHandle ,hwndDlg );
|
|
|
|
if (p->iContextType != HELPINFO_WINDOW)
|
|
{
|
|
return NO_ERROR;
|
|
}
|
|
|
|
hwnd = p->hItemHandle;
|
|
unType = HELP_WM_HELP;
|
|
}
|
|
|
|
// Standard Win95 method that produces a one-item "What's This?"
|
|
// menu that user must click to get help.
|
|
else
|
|
{
|
|
TRACE1( "ContextHelp(WM_CONTEXTMENU,h=$%08x)", wparam );
|
|
|
|
hwnd = (HWND )wparam;
|
|
unType = HELP_CONTEXTMENU;
|
|
}
|
|
|
|
WinHelp( hwnd, pszHelpFile, unType, (ULONG_PTR)pdwMap );
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
BOOL CALLBACK
|
|
WSDlgProc(
|
|
HWND hwnd,
|
|
UINT unMsg,
|
|
WPARAM wParam,
|
|
LPARAM lParam )
|
|
|
|
/* Standard Win32 dialog procedure.
|
|
*/
|
|
{
|
|
if (unMsg == WM_INITDIALOG)
|
|
{
|
|
HMENU hmenu;
|
|
RECT r1, r2;
|
|
|
|
/* Remove Close from the system menu since some people think it kills
|
|
** the app and not just the popup.
|
|
*/
|
|
hmenu = GetSystemMenu( hwnd, FALSE );
|
|
if (hmenu && DeleteMenu( hmenu, SC_CLOSE, MF_BYCOMMAND ))
|
|
{
|
|
DrawMenuBar( hwnd );
|
|
}
|
|
|
|
// Center the window
|
|
GetWindowRect(hwnd, &r1);
|
|
GetWindowRect(GetDesktopWindow(), &r2);
|
|
MoveWindow(
|
|
hwnd,
|
|
(r2.right - r2.left)/2 - (r1.right - r1.left)/2,
|
|
(r2.bottom - r2.top)/2 - (r1.bottom - r1.top)/2,
|
|
r1.right - r1.left,
|
|
r1.bottom - r1.top,
|
|
TRUE);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// Bring up the start waiting for services dialog
|
|
//
|
|
DWORD
|
|
RasSrvShowServiceWait(
|
|
IN HINSTANCE hInst,
|
|
IN HWND hwndParent,
|
|
OUT HANDLE * phData)
|
|
{
|
|
// Set the hourglass cursor
|
|
*phData = (HANDLE) SetCursor (LoadCursor (NULL, IDC_WAIT));
|
|
ShowCursor (TRUE);
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
//
|
|
// Bring down wait for services dialog
|
|
//
|
|
DWORD
|
|
RasSrvFinishServiceWait (
|
|
IN HANDLE hData)
|
|
{
|
|
HICON hIcon = (HICON)hData;
|
|
|
|
if (hIcon == NULL)
|
|
{
|
|
hIcon = LoadCursor (NULL, IDC_ARROW);
|
|
}
|
|
|
|
SetCursor (hIcon);
|
|
ShowCursor (TRUE);
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Function: EnableBackupPrivilege
|
|
//
|
|
// Enables/disables backup privilege for the current process.
|
|
//-----------------------------------------------------------------------
|
|
|
|
DWORD
|
|
EnableRebootPrivilege(
|
|
IN BOOL bEnable)
|
|
{
|
|
LUID luid;
|
|
HANDLE hToken = NULL;
|
|
TOKEN_PRIVILEGES tp;
|
|
BOOL bOk;
|
|
|
|
// We first have to try to get the token of the current
|
|
// thread since if it is impersonating, adjusting the
|
|
// privileges of the process will have no affect.
|
|
bOk = OpenThreadToken(
|
|
GetCurrentThread(),
|
|
TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY,
|
|
TRUE,
|
|
&hToken);
|
|
if (bOk == FALSE)
|
|
{
|
|
// There is no thread token -- open it up for the
|
|
// process instead.
|
|
OpenProcessToken(
|
|
GetCurrentProcess(),
|
|
TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY,
|
|
&hToken
|
|
);
|
|
}
|
|
|
|
// Get the LUID of the privilege
|
|
if (!LookupPrivilegeValue(NULL, SE_SHUTDOWN_NAME, &luid))
|
|
{
|
|
|
|
DWORD dwErr = GetLastError();
|
|
if(NULL != hToken)
|
|
{
|
|
CloseHandle(hToken);
|
|
}
|
|
return dwErr;
|
|
}
|
|
|
|
// Adjust the token privileges
|
|
tp.PrivilegeCount = 1;
|
|
tp.Privileges[0].Luid = luid;
|
|
tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
|
|
|
|
// Commit changes to the system
|
|
if (!AdjustTokenPrivileges(
|
|
hToken, !bEnable, &tp, sizeof(TOKEN_PRIVILEGES), NULL, NULL
|
|
))
|
|
{
|
|
DWORD dwErr = GetLastError();
|
|
if(NULL != hToken)
|
|
{
|
|
CloseHandle(hToken);
|
|
}
|
|
return dwErr;
|
|
}
|
|
|
|
// Even if AdjustTokenPrivileges succeeded (see MSDN) you still
|
|
// need to verify success by calling GetLastError.
|
|
if (GetLastError() == ERROR_NOT_ALL_ASSIGNED)
|
|
{
|
|
if(NULL != hToken)
|
|
{
|
|
CloseHandle(hToken);
|
|
}
|
|
return ERROR_NOT_ALL_ASSIGNED;
|
|
}
|
|
|
|
if(NULL != hToken)
|
|
{
|
|
CloseHandle(hToken);
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
// Pops up a warning with the given parent window and reboots
|
|
// windows
|
|
DWORD RasSrvReboot(HWND hwndParent)
|
|
{
|
|
DWORD dwOldState;
|
|
INT iRet;
|
|
PWCHAR pszWarn, pszTitle;
|
|
|
|
// Load the strings
|
|
pszWarn =
|
|
(PWCHAR) PszLoadString(Globals.hInstDll, WRN_REBOOT_REQUIRED);
|
|
pszTitle =
|
|
(PWCHAR) PszLoadString(Globals.hInstDll, WRN_TITLE);
|
|
|
|
// Display the warning
|
|
iRet = MessageBoxW(
|
|
hwndParent,
|
|
pszWarn,
|
|
pszTitle,
|
|
MB_YESNO | MB_APPLMODAL);
|
|
if (iRet != IDYES)
|
|
{
|
|
return ERROR_CANCELLED;
|
|
}
|
|
|
|
// Enable the reboot privelege
|
|
EnableRebootPrivilege(TRUE);
|
|
|
|
ExitWindowsEx(EWX_REBOOT, 0);
|
|
|
|
// Restore the reboot privelege
|
|
EnableRebootPrivilege(FALSE);
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|