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.
1874 lines
52 KiB
1874 lines
52 KiB
/*++
|
|
|
|
Copyright (c) 1996 Microsoft Corporation.
|
|
All rights reserved
|
|
|
|
Module Name:
|
|
|
|
licsetup.cpp
|
|
|
|
Abstract:
|
|
|
|
This module exports a function, LicenseSetupRequestWizardPages, which
|
|
gives NT Setup a wizard page for licensing to use in system setup (if
|
|
licensing should be installed).
|
|
|
|
This wizard page is responsible for all license system configuration,
|
|
including:
|
|
o Creating the LicenseService
|
|
o Creating the ...\CurrentControlSet\Services\LicenseService key and
|
|
its values. (This key contains all configuration information for the
|
|
LicenseService.)
|
|
o Creating the ...\CurrentControlSet\Services\LicenseInfo key and its
|
|
values. (This key contains all product-specific license info.)
|
|
o Creating the appropriate registry key to register the LicenseService
|
|
with the EventLog.
|
|
|
|
Portions of this module were extracted from SETUP (specifically from
|
|
\nt\private\windows\setup\syssetup\license.c).
|
|
|
|
Author:
|
|
|
|
Jeff Parham (jeffparh) 15-Apr-1996
|
|
|
|
Revision History:
|
|
|
|
Jeff Parham (jeffparh) 17-Jul-1997
|
|
Added KSecDD to FilePrint services table for SFM
|
|
|
|
--*/
|
|
|
|
#include <windows.h>
|
|
#include <commctrl.h>
|
|
#include <setupapi.h>
|
|
#include <syssetup.h>
|
|
#include <setupbat.h>
|
|
#include <stdlib.h>
|
|
#include <htmlhelp.h>
|
|
#include <Accctrl.h>
|
|
#include <aclapi.h>
|
|
#include "liccpa.hpp"
|
|
#include "help.hpp"
|
|
#include "clicreg.hpp"
|
|
#include "config.hpp"
|
|
#include "resource.h"
|
|
#include "pridlgs.hpp"
|
|
#include "special.hpp"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//============================================================================
|
|
//
|
|
// MACROS
|
|
//
|
|
|
|
// used by setup tests? simulates a click on the NEXT button
|
|
#define WM_SIMULATENEXT ( WM_USER + 287 )
|
|
|
|
// begin or end a wait cursor
|
|
#define WM_BEGINWAITCURSOR ( WM_USER + 300 )
|
|
#define WM_ENDWAITCURSOR ( WM_USER + 301 )
|
|
|
|
// number of license wizard pages
|
|
const DWORD NUM_LICENSE_PAGES = 1;
|
|
|
|
// limits for per server licenses entered from the edit box in the
|
|
// license mode page
|
|
const int PERSERVER_EDIT_MAX = 9999;
|
|
const int PERSERVER_EDIT_MIN = 5;
|
|
|
|
// the number of chars to represent PERSERVER_EDIT_MAX
|
|
const int PERSERVER_EDIT_WIDTH = 4;
|
|
|
|
|
|
//============================================================================
|
|
//
|
|
// LOCAL PROTOTYPES
|
|
//
|
|
|
|
// decides, based on the setup type, whether licensing is installed
|
|
static BOOL LicenseSetupDisplayLicensePagesQuery( PINTERNAL_SETUP_DATA );
|
|
|
|
// License mode page functions
|
|
static HPROPSHEETPAGE LicenseSetupModePageGet( PINTERNAL_SETUP_DATA );
|
|
static INT_PTR CALLBACK LicenseSetupModeDlgProc( HWND, UINT, WPARAM, LPARAM );
|
|
|
|
// License mode page Windows message handlers
|
|
static void LicenseSetupModeOnInitDialog( HWND, LPARAM, PINTERNAL_SETUP_DATA *, LPBOOL, LPDWORD, LPDWORD );
|
|
static void LicenseSetupModeOnSetActive( HWND, PINTERNAL_SETUP_DATA, LPBOOL, LPDWORD );
|
|
static void LicenseSetupModeOnSetLicenseMode( HWND, BOOL, DWORD );
|
|
static void LicenseSetupModeOnEditUpdate( HWND, HWND, BOOL, LPDWORD );
|
|
static void LicenseSetupModeOnWaitCursor( HWND, BOOL, LPDWORD );
|
|
static BOOL LicenseSetupModeOnSetCursor( HWND, WORD, DWORD );
|
|
static void LicenseSetupModeOnNext( HWND, PINTERNAL_SETUP_DATA, BOOL, DWORD );
|
|
static void LicenseSetupModeOnHelp( HWND );
|
|
static void LicenseSetupModeOnSimulateNext( HWND );
|
|
static void LicenseSetupModeOnKillActive( HWND );
|
|
static BOOL LicenseSetupModeDoUnattended( HWND, PINTERNAL_SETUP_DATA, LPBOOL, LPDWORD );
|
|
|
|
// License configuration save functions
|
|
static DWORD LicenseSetupWrite( BOOL, DWORD );
|
|
static DWORD LicenseSetupWriteKeyLicenseInfo( BOOL, DWORD );
|
|
static DWORD LicenseSetupWriteKeyLicenseService( BOOL fWriteParametersKey );
|
|
static DWORD LicenseSetupWriteKeyEventLog();
|
|
static DWORD LicenseSetupWriteService( BOOL * fCreated );
|
|
|
|
// utility functions
|
|
static int MessageBoxFromStringID( HWND, UINT, UINT, UINT );
|
|
|
|
|
|
void CreateDirectoryWithAccess();
|
|
|
|
void CreateFileWithAccess();
|
|
|
|
BOOL IsRestrictedSmallBusSrv( void );
|
|
#define SBS_SPECIAL_USERS 10
|
|
|
|
//============================================================================
|
|
//
|
|
// GLOBAL IMPLEMENTATION
|
|
//
|
|
|
|
BOOL
|
|
APIENTRY
|
|
LicenseSetupRequestWizardPages(
|
|
HPROPSHEETPAGE * paPropSheetPages,
|
|
UINT * pcPages,
|
|
PINTERNAL_SETUP_DATA pSetupData )
|
|
{
|
|
BOOL fSuccess = FALSE;
|
|
BOOL fDisplayLicensePages;
|
|
|
|
// validate params
|
|
if ( ( NULL != pcPages )
|
|
&& ( NULL != pSetupData )
|
|
&& ( sizeof( INTERNAL_SETUP_DATA ) == pSetupData->dwSizeOf ) )
|
|
{
|
|
fDisplayLicensePages = LicenseSetupDisplayLicensePagesQuery( pSetupData );
|
|
|
|
if ( NULL == paPropSheetPages )
|
|
{
|
|
// request for number of pages only
|
|
*pcPages = fDisplayLicensePages ? NUM_LICENSE_PAGES : 0;
|
|
fSuccess = TRUE;
|
|
}
|
|
else
|
|
{
|
|
// request for actual pages
|
|
if ( !fDisplayLicensePages )
|
|
{
|
|
// no pages needed
|
|
*pcPages = 0;
|
|
fSuccess = TRUE;
|
|
}
|
|
else if ( *pcPages >= NUM_LICENSE_PAGES )
|
|
{
|
|
// create and return pages
|
|
paPropSheetPages[ 0 ] = LicenseSetupModePageGet( pSetupData );
|
|
|
|
if ( NULL != paPropSheetPages[ 0 ] )
|
|
{
|
|
*pcPages = NUM_LICENSE_PAGES;
|
|
fSuccess = TRUE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return fSuccess;
|
|
}
|
|
|
|
|
|
//============================================================================
|
|
//
|
|
// LOCAL IMPLEMENTATIONS
|
|
//
|
|
|
|
static
|
|
BOOL
|
|
LicenseSetupDisplayLicensePagesQuery(
|
|
PINTERNAL_SETUP_DATA pSetupData )
|
|
//
|
|
// The following code was extracted and modified from
|
|
// \nt\private\windows\setup\syssetup\license.c
|
|
// in setup. It returns TRUE iff the licensing wizard pages should be
|
|
// displayed as a part of setup.
|
|
//
|
|
{
|
|
BOOL fDisplayLicensePages;
|
|
|
|
if ( PRODUCT_WORKSTATION == pSetupData->ProductType )
|
|
{
|
|
//
|
|
// If installing a workstation, then do not display the licensing page
|
|
//
|
|
fDisplayLicensePages = FALSE;
|
|
}
|
|
else
|
|
{
|
|
if ( !( pSetupData->OperationFlags & SETUPOPER_NTUPGRADE ) )
|
|
{
|
|
//
|
|
// The licensing page needs to be displayed on a clean install
|
|
// of a server
|
|
//
|
|
fDisplayLicensePages = TRUE;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// If upgrading a server, find out if it was already licensed
|
|
// (NT 3.51 and later). If it was, then do not display the
|
|
// licensing page.
|
|
// We find out whether or not the system was licensed by looking
|
|
// at a value entry in the registry.
|
|
// Note that NT 3.1 and 3.5 will never have this value in the
|
|
// registry, and in these cases the licensing page needs to be
|
|
// displayed.
|
|
//
|
|
|
|
DWORD winStatus;
|
|
CLicRegLicenseService FilePrintService( FILEPRINT_SERVICE_REG_KEY );
|
|
|
|
winStatus = FilePrintService.Open( NULL, FALSE );
|
|
|
|
if ( ERROR_SUCCESS != winStatus )
|
|
{
|
|
fDisplayLicensePages = TRUE;
|
|
}
|
|
else
|
|
{
|
|
LICENSE_MODE LicenseMode;
|
|
|
|
winStatus = FilePrintService.GetMode( LicenseMode );
|
|
|
|
if ( ( ERROR_SUCCESS != winStatus )
|
|
|| ( ( LICMODE_PERSEAT != LicenseMode )
|
|
&& ( LICMODE_PERSERVER != LicenseMode ) ) )
|
|
{
|
|
fDisplayLicensePages = TRUE;
|
|
}
|
|
else
|
|
{
|
|
// set FlipAllow value if it's not already set (a setup bug in
|
|
// the betas of NT 4.0 caused this value to be absent)
|
|
FilePrintService.CanChangeMode();
|
|
|
|
// add KSecDD to FilePrint services table if it isn't there already
|
|
HKEY hkeySFM;
|
|
DWORD dwDisposition;
|
|
|
|
winStatus = RegCreateKeyEx( HKEY_LOCAL_MACHINE,
|
|
TEXT( "System\\CurrentControlSet\\Services\\LicenseService\\FilePrint\\KSecDD" ),
|
|
0,
|
|
NULL,
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
NULL,
|
|
&hkeySFM,
|
|
&dwDisposition );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
RegCloseKey( hkeySFM );
|
|
}
|
|
|
|
// Change FilePrint License name from Windows NT to Windows.
|
|
|
|
CLicRegLicenseService FilePrintService(
|
|
FILEPRINT_SERVICE_REG_KEY );
|
|
|
|
winStatus = FilePrintService.Open( NULL, FALSE );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
winStatus = FilePrintService.SetFamilyDisplayName(
|
|
FILEPRINT_SERVICE_FAMILY_DISPLAY_NAME );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
winStatus = FilePrintService.SetDisplayName(
|
|
FILEPRINT_SERVICE_DISPLAY_NAME );
|
|
}
|
|
}
|
|
|
|
//
|
|
// makarp: setting fDisplayLicensePages to true is wrong, because in such case
|
|
// the pages will be displayed, and the original settings will be lost.
|
|
//
|
|
// fDisplayLicensePages = TRUE;
|
|
|
|
//
|
|
// instead we do the stuff we want to explicitely here.
|
|
//
|
|
BOOL bFlag = FALSE;
|
|
LicenseSetupWriteService(&bFlag);
|
|
CreateDirectoryWithAccess();
|
|
CreateFileWithAccess();
|
|
|
|
fDisplayLicensePages = FALSE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return fDisplayLicensePages;
|
|
}
|
|
|
|
|
|
static
|
|
HPROPSHEETPAGE
|
|
LicenseSetupModePageGet(
|
|
PINTERNAL_SETUP_DATA pSetupData )
|
|
//
|
|
// Returns an HPROPSHEETPAGE for the license mode wizard page, or
|
|
// NULL if error.
|
|
//
|
|
{
|
|
HPROPSHEETPAGE hpsp;
|
|
PROPSHEETPAGE psp;
|
|
TCHAR szHeader[256];
|
|
TCHAR szSubHeader[512];
|
|
|
|
psp.dwSize = sizeof( psp );
|
|
psp.dwFlags = PSP_USETITLE | PSP_HASHELP | PSP_USEHEADERTITLE | PSP_USEHEADERSUBTITLE;
|
|
psp.hInstance = g_hinst;
|
|
psp.pszTemplate = MAKEINTRESOURCE( IDD_SETUP_LICENSE_MODE_PAGE );
|
|
psp.hIcon = NULL;
|
|
psp.pfnDlgProc = LicenseSetupModeDlgProc;
|
|
psp.lParam = (LPARAM) pSetupData;
|
|
psp.pszTitle = pSetupData->WizardTitle;
|
|
|
|
szHeader[0] = L'\0';
|
|
szSubHeader[0] = L'\0';
|
|
|
|
LoadString( g_hinst,
|
|
IDS_SETUP_HEADER,
|
|
szHeader,
|
|
sizeof( szHeader ) / sizeof( *szHeader ) );
|
|
|
|
LoadString( g_hinst,
|
|
IDS_SETUP_SUBHEADER,
|
|
szSubHeader,
|
|
sizeof( szSubHeader ) / sizeof( *szSubHeader ) );
|
|
|
|
psp.pszHeaderTitle = szHeader;
|
|
psp.pszHeaderSubTitle = szSubHeader;
|
|
|
|
|
|
hpsp = CreatePropertySheetPage( &psp );
|
|
|
|
return hpsp;
|
|
}
|
|
|
|
|
|
static
|
|
INT_PTR
|
|
CALLBACK
|
|
LicenseSetupModeDlgProc(
|
|
HWND hwndPage,
|
|
UINT msg,
|
|
WPARAM wParam,
|
|
LPARAM lParam )
|
|
//
|
|
// Dialog procedure for the license mode wizard page.
|
|
//
|
|
{
|
|
// static data initialized by WM_INITDIALOG
|
|
static PINTERNAL_SETUP_DATA pSetupData = NULL;
|
|
static BOOL fLicensePerServer;
|
|
static DWORD cPerServerLicenses;
|
|
static DWORD cWaitCursor;
|
|
|
|
BOOL fReturn = TRUE;
|
|
|
|
switch ( msg )
|
|
{
|
|
case WM_INITDIALOG:
|
|
LicenseSetupModeOnInitDialog( hwndPage, lParam, &pSetupData, &fLicensePerServer, &cPerServerLicenses, &cWaitCursor );
|
|
break;
|
|
|
|
case WM_SIMULATENEXT:
|
|
LicenseSetupModeOnSimulateNext( hwndPage );
|
|
break;
|
|
|
|
case WM_BEGINWAITCURSOR:
|
|
LicenseSetupModeOnWaitCursor( hwndPage, TRUE, &cWaitCursor );
|
|
break;
|
|
|
|
case WM_ENDWAITCURSOR:
|
|
LicenseSetupModeOnWaitCursor( hwndPage, FALSE, &cWaitCursor );
|
|
break;
|
|
|
|
case WM_SETCURSOR:
|
|
LicenseSetupModeOnSetCursor( hwndPage, LOWORD( lParam ), cWaitCursor );
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
switch ( HIWORD( wParam ) )
|
|
{
|
|
case BN_CLICKED:
|
|
switch ( LOWORD( wParam ) )
|
|
{
|
|
case IDC_PERSEAT:
|
|
fLicensePerServer = FALSE;
|
|
LicenseSetupModeOnSetLicenseMode( hwndPage, fLicensePerServer, cPerServerLicenses );
|
|
break;
|
|
|
|
case IDC_PERSERVER:
|
|
fLicensePerServer = TRUE;
|
|
LicenseSetupModeOnSetLicenseMode( hwndPage, fLicensePerServer, cPerServerLicenses );
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case EN_UPDATE:
|
|
if ( IDC_USERCOUNT == LOWORD( wParam ) )
|
|
{
|
|
LicenseSetupModeOnEditUpdate( hwndPage, (HWND) lParam, fLicensePerServer, &cPerServerLicenses );
|
|
}
|
|
break;
|
|
|
|
default:
|
|
fReturn = FALSE;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case WM_NOTIFY:
|
|
{
|
|
NMHDR * pNmHdr;
|
|
|
|
pNmHdr = (NMHDR *)lParam;
|
|
|
|
switch ( pNmHdr->code )
|
|
{
|
|
case PSN_SETACTIVE:
|
|
LicenseSetupModeOnSetActive( hwndPage, pSetupData, &fLicensePerServer, &cPerServerLicenses );
|
|
break;
|
|
|
|
case PSN_KILLACTIVE:
|
|
LicenseSetupModeOnKillActive( hwndPage );
|
|
break;
|
|
|
|
case PSN_WIZNEXT:
|
|
case PSN_WIZFINISH:
|
|
LicenseSetupModeOnNext( hwndPage, pSetupData, fLicensePerServer, cPerServerLicenses );
|
|
break;
|
|
|
|
case PSN_HELP:
|
|
LicenseSetupModeOnHelp( hwndPage );
|
|
break;
|
|
|
|
default:
|
|
fReturn = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
fReturn = FALSE;
|
|
}
|
|
|
|
return fReturn;
|
|
}
|
|
|
|
|
|
static
|
|
void
|
|
LicenseSetupModeOnInitDialog(
|
|
HWND hwndPage,
|
|
LPARAM lParam,
|
|
PINTERNAL_SETUP_DATA * ppSetupData,
|
|
LPBOOL pfLicensePerServer,
|
|
LPDWORD pcPerServerLicenses,
|
|
LPDWORD pcWaitCursor )
|
|
//
|
|
// Message handler for WM_INITDIALOG
|
|
//
|
|
{
|
|
// initialize static data
|
|
*ppSetupData = (PINTERNAL_SETUP_DATA) ( (LPPROPSHEETPAGE) lParam )->lParam;
|
|
*pcPerServerLicenses = 5;
|
|
*pfLicensePerServer = TRUE;
|
|
*pcWaitCursor = 0;
|
|
|
|
// limit license count edit text length
|
|
SendMessage( GetDlgItem( hwndPage, IDC_USERCOUNT ), EM_LIMITTEXT, PERSERVER_EDIT_WIDTH, 0 );
|
|
|
|
// limit license count up-down range
|
|
LONG lRange;
|
|
|
|
lRange = (LPARAM) MAKELONG( (short) PERSERVER_EDIT_MAX, (short) PERSERVER_EDIT_MIN );
|
|
SendMessage( GetDlgItem( hwndPage, IDC_USERCOUNTARROW ), UDM_SETRANGE, 0, (LPARAM) lRange );
|
|
|
|
// initialize for default license mode
|
|
LicenseSetupModeOnSetLicenseMode( hwndPage, *pfLicensePerServer, *pcPerServerLicenses );
|
|
}
|
|
|
|
|
|
static
|
|
void
|
|
LicenseSetupModeOnSetActive(
|
|
HWND hwndPage,
|
|
PINTERNAL_SETUP_DATA pSetupData,
|
|
LPBOOL pfLicensePerServer,
|
|
LPDWORD pcPerServerLicenses )
|
|
//
|
|
// Notification handler for PSN_SETACTIVE
|
|
//
|
|
{
|
|
static BOOL fFirstTime = TRUE;
|
|
BOOL fSkipPage;
|
|
|
|
#ifdef SPECIAL_USERS
|
|
*pfLicensePerServer = TRUE;
|
|
*pcPerServerLicenses = SPECIAL_USERS;
|
|
fSkipPage = TRUE;
|
|
#else
|
|
if ( IsRestrictedSmallBusSrv() )
|
|
{
|
|
*pfLicensePerServer = TRUE;
|
|
*pcPerServerLicenses = SBS_SPECIAL_USERS;
|
|
fSkipPage = TRUE;
|
|
}
|
|
else if ( pSetupData->OperationFlags & SETUPOPER_BATCH )
|
|
{
|
|
// operating in unattended mode; attempt to get all answers
|
|
// from the unattend configuration file
|
|
fSkipPage = LicenseSetupModeDoUnattended( hwndPage,
|
|
pSetupData,
|
|
pfLicensePerServer,
|
|
pcPerServerLicenses );
|
|
if ( !fSkipPage )
|
|
{
|
|
// Set defaults from unattended file
|
|
LicenseSetupModeOnSetLicenseMode( hwndPage,
|
|
*pfLicensePerServer,
|
|
*pcPerServerLicenses );
|
|
//
|
|
// makarp: setting skippage to true is wrong, because we do not want to skip page.
|
|
// we came here because we did not find sufficent answers in answer file.
|
|
//
|
|
// fSkipPage = TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// operating in interactive mode; get answers from user
|
|
fSkipPage = FALSE;
|
|
}
|
|
#endif
|
|
|
|
HWND hwndSheet = GetParent( hwndPage );
|
|
|
|
if ( fSkipPage )
|
|
{
|
|
|
|
if (fFirstTime)
|
|
{
|
|
fFirstTime = FALSE;
|
|
// skip page
|
|
// Only the first time do we need to do the processing which happens on next
|
|
PostMessage( hwndSheet, PSM_PRESSBUTTON, (WPARAM)PSBTN_NEXT, 0 );
|
|
}
|
|
else
|
|
{
|
|
// After the first time the processing is already done and we don't have to do anything
|
|
// This also solves the problem where the page needs to be skipped when the user clicks back
|
|
// on a later page and this pages needs to be skipped.
|
|
SetWindowLongPtr( hwndPage, DWLP_MSGRESULT, (LONG_PTR)-1 );
|
|
return;
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
// display page
|
|
|
|
// hide Cancel button
|
|
HWND hwndCancel = GetDlgItem( hwndSheet, IDCANCEL );
|
|
EnableWindow( hwndCancel, FALSE);
|
|
ShowWindow( hwndCancel, SW_HIDE);
|
|
|
|
PropSheet_SetWizButtons( hwndSheet, PSWIZB_NEXT | PSWIZB_BACK );
|
|
|
|
if (pSetupData)
|
|
{
|
|
pSetupData->ShowHideWizardPage(TRUE);
|
|
}
|
|
}
|
|
|
|
// success
|
|
SetWindowLongPtr( hwndPage, DWLP_MSGRESULT, (LONG_PTR)0 );
|
|
}
|
|
|
|
|
|
static
|
|
void
|
|
LicenseSetupModeOnSetLicenseMode(
|
|
HWND hwndPage,
|
|
BOOL fToPerServer,
|
|
DWORD cPerServerLicenses )
|
|
//
|
|
// Handles changing the page to signify that the given license mode
|
|
// is selected.
|
|
//
|
|
{
|
|
HWND hwndCount = GetDlgItem( hwndPage, IDC_USERCOUNT );
|
|
HWND hwndSpin = GetDlgItem( hwndPage, IDC_USERCOUNTARROW );
|
|
|
|
// set radio button states
|
|
CheckDlgButton( hwndPage, IDC_PERSEAT, !fToPerServer );
|
|
CheckDlgButton( hwndPage, IDC_PERSERVER, fToPerServer );
|
|
|
|
// set user count edit control
|
|
if ( fToPerServer )
|
|
{
|
|
// display per server count
|
|
SetDlgItemInt( hwndPage, IDC_USERCOUNT, cPerServerLicenses, FALSE );
|
|
SetFocus( hwndCount );
|
|
SendMessage( hwndCount, EM_SETSEL, 0, -1 );
|
|
}
|
|
else
|
|
{
|
|
// remove per server count
|
|
SetDlgItemText( hwndPage, IDC_USERCOUNT, TEXT( "" ) );
|
|
}
|
|
|
|
// display count up-down and edit box iff per server mode is selected
|
|
EnableWindow( hwndCount, fToPerServer );
|
|
EnableWindow( hwndSpin, fToPerServer );
|
|
}
|
|
|
|
|
|
static
|
|
void
|
|
LicenseSetupModeOnEditUpdate(
|
|
HWND hwndPage,
|
|
HWND hwndCount,
|
|
BOOL fLicensePerServer,
|
|
LPDWORD pcPerServerLicenses )
|
|
//
|
|
// Command handler for EN_UPDATE of count edit box
|
|
//
|
|
{
|
|
if ( fLicensePerServer )
|
|
{
|
|
BOOL fTranslated;
|
|
UINT nValue;
|
|
BOOL fModified = FALSE;
|
|
|
|
nValue = GetDlgItemInt( hwndPage, IDC_USERCOUNT, &fTranslated, FALSE );
|
|
|
|
if ( fTranslated )
|
|
{
|
|
// count translated; ensure its within the valid range
|
|
if ( PERSERVER_EDIT_MAX < nValue )
|
|
{
|
|
// too big
|
|
nValue = PERSERVER_EDIT_MAX;
|
|
fModified = TRUE;
|
|
}
|
|
|
|
*pcPerServerLicenses = nValue;
|
|
}
|
|
else
|
|
{
|
|
// count couldn't be translated; reset to last value
|
|
nValue = *pcPerServerLicenses;
|
|
fModified = TRUE;
|
|
}
|
|
|
|
if ( fModified )
|
|
{
|
|
// text in edit box is invalid; change it to the proper value
|
|
SetDlgItemInt( hwndPage, IDC_USERCOUNT, nValue, FALSE );
|
|
SetFocus( hwndCount );
|
|
SendMessage( hwndCount, EM_SETSEL, 0, -1 );
|
|
MessageBeep( MB_VALUELIMIT );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static
|
|
void
|
|
LicenseSetupModeOnWaitCursor(
|
|
HWND hwndDlg,
|
|
BOOL fWait,
|
|
LPDWORD pcWaitCursor )
|
|
//
|
|
// Handler for WM_BEGINWAITCURSOR / WM_ENDWAITCURSOR
|
|
//
|
|
{
|
|
if ( fWait )
|
|
{
|
|
(*pcWaitCursor)++;
|
|
|
|
if ( 1 == (*pcWaitCursor) )
|
|
{
|
|
// display wait cursor
|
|
SetCursor( LoadCursor( NULL, MAKEINTRESOURCE( IDC_WAIT ) ) );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( 0 < *pcWaitCursor )
|
|
{
|
|
(*pcWaitCursor)--;
|
|
}
|
|
|
|
if ( 0 == *pcWaitCursor )
|
|
{
|
|
// display regular cursor
|
|
SetCursor( LoadCursor( NULL, MAKEINTRESOURCE( IDC_ARROW ) ) );
|
|
}
|
|
}
|
|
|
|
// success
|
|
SetWindowLongPtr( hwndDlg, DWLP_MSGRESULT, (LONG_PTR)*pcWaitCursor );
|
|
}
|
|
|
|
|
|
static
|
|
BOOL
|
|
LicenseSetupModeOnSetCursor(
|
|
HWND hwndDlg,
|
|
WORD nHitTest,
|
|
DWORD cWaitCursor )
|
|
//
|
|
// Handler for WM_SETCURSOR
|
|
//
|
|
{
|
|
BOOL frt = FALSE;
|
|
|
|
if ( HTCLIENT == nHitTest )
|
|
{
|
|
if ( cWaitCursor > 0 )
|
|
{
|
|
// display wait cursor instead of regular cursor
|
|
SetCursor( LoadCursor( NULL, MAKEINTRESOURCE( IDC_WAIT ) ) );
|
|
SetWindowLongPtr( hwndDlg, DWLP_MSGRESULT, (LONG_PTR)TRUE );
|
|
frt = TRUE;
|
|
}
|
|
}
|
|
|
|
return frt;
|
|
}
|
|
|
|
|
|
static
|
|
void
|
|
LicenseSetupModeOnNext(
|
|
HWND hwndPage,
|
|
PINTERNAL_SETUP_DATA pSetupData,
|
|
BOOL fLicensePerServer,
|
|
DWORD cPerServerLicenses )
|
|
//
|
|
// Notification handler for PSN_WIZNEXT
|
|
//
|
|
{
|
|
DWORD winStatus;
|
|
int nButton;
|
|
|
|
if ( ( fLicensePerServer )
|
|
&& ( PERSERVER_EDIT_MIN > cPerServerLicenses )
|
|
&& !( pSetupData->OperationFlags & SETUPOPER_BATCH ) )
|
|
{
|
|
// warn user about using per server mode with less then 5 licenses
|
|
MessageBoxFromStringID( hwndPage,
|
|
IDS_LICENSE_SETUP_NO_PER_SERVER_LICENSES,
|
|
IDS_WARNING,
|
|
MB_ICONERROR | MB_OK );
|
|
nButton = IDCANCEL;
|
|
}
|
|
else
|
|
{
|
|
// per seat mode or per server mode with positive license count
|
|
nButton = IDOK;
|
|
}
|
|
|
|
if ( IDOK == nButton )
|
|
{
|
|
do
|
|
{
|
|
// save license configuration
|
|
SendMessage( hwndPage, WM_BEGINWAITCURSOR, 0, 0 );
|
|
|
|
winStatus = LicenseSetupWrite( fLicensePerServer, cPerServerLicenses );
|
|
|
|
SendMessage( hwndPage, WM_ENDWAITCURSOR, 0, 0 );
|
|
|
|
if ( ERROR_SUCCESS != winStatus )
|
|
{
|
|
// save failed; alert user
|
|
nButton = MessageBoxFromStringID( hwndPage,
|
|
IDS_LICENSE_SETUP_SAVE_FAILED,
|
|
IDS_ERROR,
|
|
MB_ICONSTOP | MB_ABORTRETRYIGNORE | MB_DEFBUTTON2 );
|
|
|
|
if ( IDIGNORE == nButton )
|
|
{
|
|
nButton = IDOK;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// save succeeded
|
|
nButton = IDOK;
|
|
}
|
|
} while ( IDRETRY == nButton );
|
|
}
|
|
|
|
if ( IDOK != nButton )
|
|
{
|
|
// don't advance to next page
|
|
SetWindowLongPtr( hwndPage, DWLP_MSGRESULT, (LONG_PTR)-1 );
|
|
}
|
|
}
|
|
|
|
|
|
static
|
|
void
|
|
LicenseSetupModeOnHelp(
|
|
HWND hwndPage )
|
|
//
|
|
// Notification handler for PSN_HELP
|
|
//
|
|
{
|
|
::HtmlHelp( hwndPage, LICCPA_HTMLHELPFILE, HH_DISPLAY_TOPIC,0);
|
|
}
|
|
|
|
|
|
static
|
|
void
|
|
LicenseSetupModeOnSimulateNext(
|
|
HWND hwndPage )
|
|
//
|
|
// Handler for WM_SIMULATENEXT (used by setup tests?)
|
|
//
|
|
{
|
|
// simulate the next button
|
|
PropSheet_PressButton( GetParent( hwndPage ), PSBTN_NEXT );
|
|
}
|
|
|
|
|
|
static
|
|
void
|
|
LicenseSetupModeOnKillActive(
|
|
HWND hwndPage )
|
|
//
|
|
// Notification handler for PSN_KILLACTIVE
|
|
//
|
|
{
|
|
// success
|
|
SetWindowLong( hwndPage, DWLP_MSGRESULT, 0);
|
|
}
|
|
|
|
typedef enum {
|
|
UnattendFullUnattend,
|
|
UnattendGUIAttended,
|
|
UnattendDefaultHide,
|
|
UnattendProvideDefault,
|
|
UnattendReadOnly } UNATTENDMODE;
|
|
|
|
static
|
|
BOOL
|
|
LicenseSetupModeDoUnattended(
|
|
HWND hwndPage,
|
|
PINTERNAL_SETUP_DATA pSetupData,
|
|
LPBOOL pfLicensePerServer,
|
|
LPDWORD pcPerServerLicenses )
|
|
//
|
|
// Get answers to wizard page from unattend file.
|
|
//
|
|
{
|
|
int cch;
|
|
LPTSTR pszBadParam;
|
|
TCHAR szLicenseMode[ 64 ];
|
|
TCHAR szPerServerLicenses[ 64 ];
|
|
TCHAR szUnattendMode[ 64 ];
|
|
UNATTENDMODE UnattendMode = UnattendDefaultHide;
|
|
|
|
pszBadParam = NULL;
|
|
|
|
SendMessage( hwndPage, WM_BEGINWAITCURSOR, 0, 0 );
|
|
|
|
// Get Unattend Mode
|
|
cch = GetPrivateProfileString( WINNT_UNATTENDED,
|
|
WINNT_U_UNATTENDMODE,
|
|
TEXT( "" ),
|
|
szUnattendMode,
|
|
sizeof( szUnattendMode ) / sizeof( *szUnattendMode ),
|
|
pSetupData->UnattendFile );
|
|
if ( 0 < cch )
|
|
{
|
|
if ( !lstrcmpi( szUnattendMode, WINNT_A_FULLUNATTENDED ) )
|
|
{
|
|
UnattendMode = UnattendFullUnattend;
|
|
}
|
|
else if ( !lstrcmpi( szUnattendMode, WINNT_A_PROVIDEDEFAULT ) )
|
|
{
|
|
UnattendMode = UnattendProvideDefault;
|
|
}
|
|
else if ( !lstrcmpi( szUnattendMode, WINNT_A_READONLY ) )
|
|
{
|
|
UnattendMode = UnattendReadOnly;
|
|
}
|
|
else if ( !lstrcmpi( szUnattendMode, WINNT_A_GUIATTENDED ) )
|
|
{
|
|
// This should never happen
|
|
UnattendMode = UnattendGUIAttended;
|
|
}
|
|
}
|
|
|
|
|
|
// get license mode
|
|
cch = GetPrivateProfileString( WINNT_LICENSEDATA_W,
|
|
WINNT_L_AUTOMODE_W,
|
|
TEXT( "" ),
|
|
szLicenseMode,
|
|
sizeof( szLicenseMode ) / sizeof( *szLicenseMode ),
|
|
pSetupData->UnattendFile );
|
|
|
|
SendMessage( hwndPage, WM_ENDWAITCURSOR, 0, 0 );
|
|
|
|
if ( 0 < cch )
|
|
{
|
|
if ( !lstrcmpi( szLicenseMode, WINNT_A_PERSEAT_W ) )
|
|
{
|
|
*pfLicensePerServer = FALSE;
|
|
}
|
|
else if ( !lstrcmpi( szLicenseMode, WINNT_A_PERSERVER_W ) )
|
|
{
|
|
*pfLicensePerServer = TRUE;
|
|
}
|
|
else
|
|
{
|
|
cch = 0;
|
|
}
|
|
}
|
|
|
|
if ( cch <= 0 )
|
|
{
|
|
// license mode absent or invalid
|
|
pszBadParam = WINNT_L_AUTOMODE_W;
|
|
}
|
|
else if ( !*pfLicensePerServer )
|
|
{
|
|
// per seat mode; no need to read per server license count
|
|
*pcPerServerLicenses = 0;
|
|
}
|
|
else
|
|
{
|
|
// get per server license count
|
|
SendMessage( hwndPage, WM_BEGINWAITCURSOR, 0, 0 );
|
|
|
|
cch = GetPrivateProfileString( WINNT_LICENSEDATA_W,
|
|
WINNT_L_AUTOUSERS_W,
|
|
TEXT( "" ),
|
|
szPerServerLicenses,
|
|
sizeof( szPerServerLicenses ) / sizeof( *szPerServerLicenses ),
|
|
pSetupData->UnattendFile );
|
|
|
|
SendMessage( hwndPage, WM_ENDWAITCURSOR, 0, 0 );
|
|
|
|
if ( 0 < cch )
|
|
{
|
|
*pcPerServerLicenses = wcstoul( szPerServerLicenses, NULL, 10 );
|
|
|
|
if ( ( PERSERVER_EDIT_MIN > *pcPerServerLicenses )
|
|
|| ( PERSERVER_EDIT_MAX < *pcPerServerLicenses ) )
|
|
{
|
|
// Don't let things go without setting a valid server license
|
|
// count.
|
|
*pcPerServerLicenses = PERSERVER_EDIT_MIN;
|
|
cch = 0;
|
|
}
|
|
}
|
|
|
|
if ( cch <= 0 )
|
|
{
|
|
// per server license count absent or invalid
|
|
pszBadParam = WINNT_L_AUTOUSERS_W;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Do not display the error message on preinstall.
|
|
//
|
|
|
|
if ( NULL != pszBadParam &&
|
|
!(pSetupData->OperationFlags & (SETUPOPER_PREINSTALL | SETUPOPER_NTUPGRADE)) &&
|
|
UnattendMode == UnattendFullUnattend )
|
|
{
|
|
// encountered a bad unattended parameter; display error
|
|
TCHAR szCaption[ 64 ];
|
|
TCHAR szFormat[ 1024 ];
|
|
TCHAR szText[ 1024 ];
|
|
|
|
LoadString( g_hinst,
|
|
IDS_LICENSE_SETUP_BAD_UNATTEND_PARAM,
|
|
szFormat,
|
|
sizeof( szFormat ) / sizeof( *szFormat ) );
|
|
|
|
LoadString( g_hinst,
|
|
IDS_ERROR,
|
|
szCaption,
|
|
sizeof( szCaption ) / sizeof( *szCaption ) );
|
|
|
|
wsprintf( szText, szFormat, pszBadParam );
|
|
|
|
MessageBox( hwndPage,
|
|
szText,
|
|
szCaption,
|
|
MB_OK | MB_ICONSTOP );
|
|
}
|
|
|
|
// If just providing defaults, return FALSE to force the page to show
|
|
if ( UnattendMode == UnattendProvideDefault )
|
|
return ( FALSE );
|
|
return ( NULL == pszBadParam );
|
|
}
|
|
|
|
|
|
static
|
|
DWORD
|
|
LicenseSetupWrite(
|
|
BOOL fLicensePerServer,
|
|
DWORD cPerServerLicenses )
|
|
//
|
|
// Write license configuration; returns ERROR_SUCCESS or Windows error.
|
|
//
|
|
{
|
|
DWORD winStatus;
|
|
BOOL fCreated = TRUE; // TRUE if service entry is created
|
|
// Used to determine if we should create
|
|
// the parameters key or leave it alone.
|
|
|
|
winStatus = LicenseSetupWriteService( &fCreated );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
winStatus = LicenseSetupWriteKeyLicenseInfo( fLicensePerServer,
|
|
cPerServerLicenses );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
winStatus = LicenseSetupWriteKeyLicenseService( fCreated );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
winStatus = LicenseSetupWriteKeyEventLog();
|
|
}
|
|
}
|
|
}
|
|
|
|
return winStatus;
|
|
}
|
|
|
|
|
|
static
|
|
DWORD
|
|
LicenseSetupWriteKeyLicenseInfo(
|
|
BOOL fLicensePerServer,
|
|
DWORD cPerServerLicenses )
|
|
//
|
|
// Create registry values:
|
|
//
|
|
// HKEY_LOCAL_MACHINE
|
|
// \System
|
|
// \CurrentControlSet
|
|
// \Services
|
|
// \LicenseInfo
|
|
// ErrorControl : REG_DWORD : 1
|
|
// Start : REG_DWORD : 3
|
|
// Type : REG_DWORD : 4
|
|
// \FilePrint
|
|
// ConcurrentLimit : REG_DWORD : fLicensePerServer ? cPerServerLicenses : 0
|
|
// DisplayName : REG_SZ : "Windows Server"
|
|
// FamilyDisplayName : REG_SZ : "Windows Server"
|
|
// Mode : REG_DWORD : fLicensePerServer ? 1 : 0
|
|
// FlipAllow : REG_DWORD : fLicensePerServer ? 1 : 0
|
|
//
|
|
{
|
|
DWORD winStatus;
|
|
BOOL fCreatedNewServiceList;
|
|
CLicRegLicense ServiceList;
|
|
|
|
winStatus = ServiceList.Open( fCreatedNewServiceList );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
CLicRegLicenseService FilePrintService( FILEPRINT_SERVICE_REG_KEY );
|
|
|
|
winStatus = FilePrintService.Open( NULL, TRUE );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
LICENSE_MODE lm;
|
|
|
|
lm = fLicensePerServer ? LICMODE_PERSERVER : LICMODE_PERSEAT;
|
|
|
|
winStatus = FilePrintService.SetMode( lm );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
winStatus = FilePrintService.SetUserLimit( fLicensePerServer ? cPerServerLicenses : 0 );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
winStatus = FilePrintService.SetChangeFlag( fLicensePerServer );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
winStatus = FilePrintService.SetFamilyDisplayName( FILEPRINT_SERVICE_FAMILY_DISPLAY_NAME );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
winStatus = FilePrintService.SetDisplayName( FILEPRINT_SERVICE_DISPLAY_NAME );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return winStatus;
|
|
}
|
|
|
|
|
|
static
|
|
DWORD
|
|
LicenseSetupWriteKeyLicenseService( BOOL fWriteParametersKey )
|
|
//
|
|
// Create registry values:
|
|
//
|
|
// HKEY_LOCAL_MACHINE
|
|
// \System
|
|
// \CurrentControlSet
|
|
// \Services
|
|
// \LicenseService
|
|
// \FilePrint
|
|
// \KSecDD
|
|
// \MSAfpSrv
|
|
// \SMBServer
|
|
// \TCP/IP Print Server
|
|
// \Parameters
|
|
// UseEnterprise : REG_DWORD : 0
|
|
// ReplicationType : REG_DWORD : 0
|
|
// ReplicationTime : REG_DWORD : 24 * 60 * 60
|
|
// EnterpriseServer : REG_SZ : ""
|
|
//
|
|
{
|
|
DWORD winStatus;
|
|
HKEY hKeyLicenseService;
|
|
DWORD dwKeyCreateDisposition;
|
|
|
|
// create LicenseInfo key
|
|
winStatus = RegCreateKeyEx( HKEY_LOCAL_MACHINE,
|
|
LICENSE_SERVICE_REG_KEY,
|
|
0,
|
|
NULL,
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
NULL,
|
|
&hKeyLicenseService,
|
|
&dwKeyCreateDisposition );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
HKEY hKeyFilePrint;
|
|
|
|
// create FilePrint key
|
|
winStatus = RegCreateKeyEx( hKeyLicenseService,
|
|
TEXT( "FilePrint" ),
|
|
0,
|
|
NULL,
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
NULL,
|
|
&hKeyFilePrint,
|
|
&dwKeyCreateDisposition );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
const LPCTSTR apszFilePrintSubkeys[] =
|
|
{
|
|
TEXT( "KSecDD" ),
|
|
TEXT( "MSAfpSrv" ),
|
|
TEXT( "SMBServer" ),
|
|
TEXT( "TCP/IP Print Server" ),
|
|
NULL
|
|
};
|
|
|
|
HKEY hKeyFilePrintSubkey;
|
|
DWORD iSubkey;
|
|
|
|
for ( iSubkey = 0; NULL != apszFilePrintSubkeys[ iSubkey ]; iSubkey++ )
|
|
{
|
|
winStatus = RegCreateKeyEx( hKeyFilePrint,
|
|
apszFilePrintSubkeys[ iSubkey ],
|
|
0,
|
|
NULL,
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
NULL,
|
|
&hKeyFilePrintSubkey,
|
|
&dwKeyCreateDisposition );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
RegCloseKey( hKeyFilePrintSubkey );
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
RegCloseKey( hKeyFilePrint );
|
|
}
|
|
|
|
RegCloseKey( hKeyLicenseService );
|
|
}
|
|
|
|
//
|
|
// Only write the Parameters key if the service was just created. That is,
|
|
// this is not an upgrade
|
|
//
|
|
if ( fWriteParametersKey && (ERROR_SUCCESS == winStatus) )
|
|
{
|
|
HKEY hKeyParameters;
|
|
|
|
// create Parameters key
|
|
winStatus = RegCreateKeyEx( HKEY_LOCAL_MACHINE,
|
|
szLicenseKey, // const
|
|
0,
|
|
NULL,
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
NULL,
|
|
&hKeyParameters,
|
|
&dwKeyCreateDisposition );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
// create LicenseService\Parameters values
|
|
winStatus = RegSetValueEx( hKeyParameters,
|
|
szUseEnterprise, // const
|
|
0,
|
|
REG_DWORD,
|
|
(CONST BYTE *) &dwUseEnterprise, // const
|
|
sizeof( dwUseEnterprise ) );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
winStatus = RegSetValueEx( hKeyParameters,
|
|
szReplicationType, // const
|
|
0,
|
|
REG_DWORD,
|
|
(CONST BYTE *) &dwReplicationType, // const
|
|
sizeof( dwReplicationType ) );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
winStatus = RegSetValueEx( hKeyParameters,
|
|
szReplicationTime, // const
|
|
0,
|
|
REG_DWORD,
|
|
(CONST BYTE *) &dwReplicationTimeInSec, // const
|
|
sizeof( dwReplicationTimeInSec ) );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
LPCTSTR pszEnterpriseServer = TEXT( "" );
|
|
|
|
winStatus = RegSetValueEx( hKeyParameters,
|
|
szEnterpriseServer, // const
|
|
0,
|
|
REG_SZ,
|
|
(CONST BYTE *) pszEnterpriseServer,
|
|
( 1 + lstrlen( pszEnterpriseServer ) ) * sizeof( *pszEnterpriseServer ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
RegCloseKey( hKeyParameters );
|
|
}
|
|
}
|
|
|
|
return winStatus;
|
|
}
|
|
|
|
|
|
static
|
|
DWORD
|
|
LicenseSetupWriteKeyEventLog()
|
|
//
|
|
// Create registry values:
|
|
//
|
|
// HKEY_LOCAL_MACHINE
|
|
// \System
|
|
// \CurrentControlSet
|
|
// \Services
|
|
// \EventLog
|
|
// \Application
|
|
// \LicenseService
|
|
// EventMessageFile : REG_EXPAND_SZ : %SystemRoot%\System32\llsrpc.dll
|
|
// TypesSupported : REG_DWORD : 7
|
|
//
|
|
{
|
|
DWORD winStatus;
|
|
HKEY hKeyLicenseService;
|
|
DWORD dwKeyCreateDisposition;
|
|
|
|
// create LicenseService key
|
|
winStatus = RegCreateKeyEx( HKEY_LOCAL_MACHINE,
|
|
TEXT( "System\\CurrentControlSet\\Services\\EventLog\\Application\\LicenseService" ),
|
|
0,
|
|
NULL,
|
|
0,
|
|
KEY_ALL_ACCESS,
|
|
NULL,
|
|
&hKeyLicenseService,
|
|
&dwKeyCreateDisposition );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
LPCTSTR pszEventMessageFile = TEXT( "%SystemRoot%\\System32\\llsrpc.dll" );
|
|
const DWORD dwTypesSupported = ( EVENTLOG_ERROR_TYPE
|
|
| EVENTLOG_WARNING_TYPE
|
|
| EVENTLOG_INFORMATION_TYPE );
|
|
|
|
winStatus = RegSetValueEx( hKeyLicenseService,
|
|
TEXT( "TypesSupported" ),
|
|
0,
|
|
REG_DWORD,
|
|
(CONST BYTE *) &dwTypesSupported,
|
|
sizeof( dwTypesSupported ) );
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
winStatus = RegSetValueEx( hKeyLicenseService,
|
|
TEXT( "EventMessageFile" ),
|
|
0,
|
|
REG_SZ,
|
|
(CONST BYTE *) pszEventMessageFile,
|
|
( 1 + lstrlen( pszEventMessageFile ) ) * sizeof( *pszEventMessageFile ) );
|
|
}
|
|
|
|
RegCloseKey( hKeyLicenseService );
|
|
}
|
|
|
|
return winStatus;
|
|
}
|
|
|
|
|
|
static
|
|
DWORD
|
|
LicenseSetupWriteService( BOOL * fCreated )
|
|
//
|
|
// Create/modify service:
|
|
//
|
|
// lpServiceName = "LicenseService"
|
|
// lpDisplayName = "License Logging Service"
|
|
// dwServiceType = SERVICE_WIN32_OWN_PROCESS
|
|
// dwStartType = LanManServerInstalled ? SERVICE_AUTO_START : SERVICE_DISABLED
|
|
// dwErrorControl = SERVICE_ERROR_NORMAL
|
|
// lpBinaryPathName = "%SystemRoot%\\System32\\llssrv.exe"
|
|
// lpLoadOrderGroup = NULL
|
|
// lpdwTagId = NULL
|
|
// lpDependencies = NULL
|
|
// lpServiceStartName = NULL
|
|
// lpPassword = NULL
|
|
//
|
|
{
|
|
SC_HANDLE hSC;
|
|
DWORD winStatus;
|
|
|
|
*fCreated = FALSE;
|
|
|
|
hSC = OpenSCManager( NULL, NULL, SC_MANAGER_ALL_ACCESS );
|
|
|
|
if ( NULL == hSC )
|
|
{
|
|
winStatus = GetLastError();
|
|
}
|
|
else
|
|
{
|
|
HKEY hKeyLanmanServerParameters;
|
|
DWORD dwStartType ;
|
|
SC_HANDLE hLicenseService = NULL;
|
|
TCHAR szServiceDisplayName[ 128 ] = TEXT( "License Logging" );
|
|
TCHAR szServiceDescription[256] = TEXT("");
|
|
TCHAR szServiceStartName [] = TEXT("NT AUTHORITY\\NetworkService");
|
|
TCHAR szServicePassword[]=TEXT("");
|
|
SERVICE_DESCRIPTION svcDescription;
|
|
QUERY_SERVICE_CONFIG* pConfig = NULL;
|
|
DWORD cbBytesNeeded = 0;
|
|
BOOL frt;
|
|
DWORD dwDesiredAccess = SERVICE_ALL_ACCESS;
|
|
|
|
|
|
|
|
|
|
// enable service iff LanmanServer was installed
|
|
winStatus = RegOpenKeyEx( HKEY_LOCAL_MACHINE,
|
|
TEXT( "SYSTEM\\CurrentControlSet\\Services\\LanmanServer\\Parameters" ),
|
|
0,
|
|
KEY_READ,
|
|
&hKeyLanmanServerParameters );
|
|
|
|
|
|
if ( ERROR_SUCCESS == winStatus )
|
|
{
|
|
|
|
dwStartType = SERVICE_AUTO_START;
|
|
|
|
hLicenseService = OpenService( hSC, TEXT( "LicenseService"), dwDesiredAccess );
|
|
|
|
if( hLicenseService != NULL )
|
|
{
|
|
|
|
cbBytesNeeded = sizeof(QUERY_SERVICE_CONFIG) + 4096;
|
|
|
|
pConfig = (LPQUERY_SERVICE_CONFIG) LocalAlloc( LPTR, cbBytesNeeded );
|
|
|
|
if ( pConfig != NULL )
|
|
{
|
|
|
|
frt = ::QueryServiceConfig( hLicenseService,
|
|
pConfig,
|
|
cbBytesNeeded,
|
|
&cbBytesNeeded );
|
|
|
|
if ( frt )
|
|
{
|
|
dwStartType = pConfig->dwStartType;
|
|
}
|
|
|
|
LocalFree ( pConfig ) ;
|
|
}
|
|
|
|
|
|
CloseServiceHandle( hLicenseService );
|
|
|
|
}
|
|
|
|
RegCloseKey( hKeyLanmanServerParameters );
|
|
|
|
}
|
|
|
|
else
|
|
{
|
|
dwStartType = SERVICE_DISABLED;
|
|
}
|
|
|
|
LoadString( g_hinst,
|
|
IDS_SERVICE_DISPLAY_NAME,
|
|
szServiceDisplayName,
|
|
sizeof( szServiceDisplayName ) / sizeof( *szServiceDisplayName ) );
|
|
|
|
LoadString( g_hinst,
|
|
IDS_SERVICE_DESCRIPTION,
|
|
szServiceDescription,
|
|
sizeof( szServiceDescription ) / sizeof( *szServiceDescription ) );
|
|
|
|
|
|
svcDescription.lpDescription = szServiceDescription;
|
|
|
|
|
|
hLicenseService = CreateService( hSC,
|
|
TEXT( "LicenseService" ),
|
|
szServiceDisplayName,
|
|
// 14659: needed to call ChangeConfig2 later
|
|
SERVICE_CHANGE_CONFIG,
|
|
SERVICE_WIN32_OWN_PROCESS,
|
|
dwStartType,
|
|
SERVICE_ERROR_NORMAL,
|
|
TEXT( "%SystemRoot%\\System32\\llssrv.exe" ),
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
szServiceStartName,
|
|
szServicePassword );
|
|
|
|
|
|
if ( NULL != hLicenseService )
|
|
{
|
|
// service successfully created
|
|
|
|
ChangeServiceConfig2( hLicenseService,
|
|
SERVICE_CONFIG_DESCRIPTION,
|
|
&svcDescription );
|
|
|
|
CloseServiceHandle( hLicenseService );
|
|
|
|
winStatus = ERROR_SUCCESS;
|
|
*fCreated = TRUE;
|
|
}
|
|
else
|
|
{
|
|
winStatus = GetLastError();
|
|
|
|
if ( ERROR_SERVICE_EXISTS == winStatus )
|
|
{
|
|
// service already exists; change configuration of existing service
|
|
hLicenseService = OpenService( hSC,
|
|
TEXT( "LicenseService" ),
|
|
SERVICE_CHANGE_CONFIG );
|
|
|
|
if ( NULL == hLicenseService )
|
|
{
|
|
winStatus = GetLastError();
|
|
}
|
|
else
|
|
{
|
|
SC_LOCK scLock;
|
|
BOOL ok;
|
|
|
|
scLock = LockServiceDatabase( hSC );
|
|
// continue even if we can't lock the database
|
|
|
|
ok = ChangeServiceConfig( hLicenseService,
|
|
SERVICE_WIN32_OWN_PROCESS,
|
|
dwStartType,
|
|
SERVICE_ERROR_NORMAL,
|
|
TEXT( "%SystemRoot%\\System32\\llssrv.exe" ),
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
szServiceStartName,
|
|
szServicePassword,
|
|
szServiceDisplayName );
|
|
|
|
if ( !ok )
|
|
{
|
|
winStatus = GetLastError();
|
|
}
|
|
else
|
|
{
|
|
ChangeServiceConfig2( hLicenseService,
|
|
SERVICE_CONFIG_DESCRIPTION,
|
|
&svcDescription);
|
|
|
|
winStatus = ERROR_SUCCESS;
|
|
}
|
|
|
|
if ( NULL != scLock )
|
|
{
|
|
UnlockServiceDatabase( scLock );
|
|
}
|
|
|
|
CloseServiceHandle( hLicenseService );
|
|
}
|
|
}
|
|
}
|
|
|
|
CloseServiceHandle( hSC );
|
|
}
|
|
|
|
|
|
CreateDirectoryWithAccess();
|
|
|
|
CreateFileWithAccess();
|
|
|
|
return winStatus;
|
|
}
|
|
|
|
void CreateDirectoryWithAccess()
|
|
{
|
|
DWORD winStatus = 0;
|
|
TCHAR tchWinDirPath[MAX_PATH+1] = L"";
|
|
PACL pNewDacl = NULL;
|
|
PACL pOldDacl = NULL;
|
|
PSECURITY_DESCRIPTOR pSD = NULL;
|
|
TCHAR tchLLSDirPath[ MAX_PATH +1] = L"";
|
|
BOOL bFlag = FALSE;
|
|
PSID pSid = NULL;
|
|
EXPLICIT_ACCESS ExplicitEntries;
|
|
SID_IDENTIFIER_AUTHORITY ntSidAuthority = SECURITY_NT_AUTHORITY;
|
|
|
|
winStatus = GetSystemWindowsDirectory( tchWinDirPath , MAX_PATH+1);
|
|
if(winStatus == 0)
|
|
{
|
|
goto cleanup;
|
|
}
|
|
|
|
lstrcpy(tchLLSDirPath, tchWinDirPath);
|
|
lstrcat( tchLLSDirPath , L"\\system32\\lls" );
|
|
|
|
// Creating new EXPLICIT_ACCESS structure to set on the directory
|
|
|
|
ZeroMemory( &ExplicitEntries, sizeof(ExplicitEntries) );
|
|
|
|
bFlag = AllocateAndInitializeSid(
|
|
&ntSidAuthority,
|
|
1,
|
|
SECURITY_NETWORK_SERVICE_RID,0,
|
|
0, 0, 0, 0, 0, 0,
|
|
&pSid );
|
|
|
|
|
|
|
|
if ( !bFlag || (pSid == NULL) ) {
|
|
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
BuildTrusteeWithSid( &ExplicitEntries.Trustee, pSid );
|
|
|
|
ExplicitEntries.grfInheritance = SUB_CONTAINERS_AND_OBJECTS_INHERIT;
|
|
ExplicitEntries.grfAccessMode = SET_ACCESS;
|
|
ExplicitEntries.grfAccessPermissions = FILE_ALL_ACCESS;
|
|
ExplicitEntries.Trustee.TrusteeForm = TRUSTEE_IS_SID;
|
|
ExplicitEntries.Trustee.TrusteeType = TRUSTEE_IS_WELL_KNOWN_GROUP;
|
|
|
|
|
|
bFlag = CreateDirectory( tchLLSDirPath, NULL );
|
|
|
|
|
|
if ( !bFlag )
|
|
{
|
|
|
|
winStatus = GetLastError();
|
|
|
|
if (ERROR_ALREADY_EXISTS != winStatus)
|
|
{
|
|
goto cleanup;
|
|
}
|
|
}
|
|
|
|
|
|
if( GetNamedSecurityInfoW( tchLLSDirPath,
|
|
SE_FILE_OBJECT,
|
|
DACL_SECURITY_INFORMATION,
|
|
NULL, // psidOwner
|
|
NULL, // psidGroup
|
|
&pOldDacl, // pDacl
|
|
NULL, // pSacl
|
|
&pSD ) != ERROR_SUCCESS)
|
|
{
|
|
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
//
|
|
// Set the Acl with the ExplicitEntry rights
|
|
//
|
|
if( SetEntriesInAcl( 1,
|
|
&ExplicitEntries,
|
|
pOldDacl,
|
|
&pNewDacl ) != ERROR_SUCCESS)
|
|
{
|
|
goto cleanup;
|
|
}
|
|
|
|
//
|
|
// SET security on the Directory
|
|
//
|
|
|
|
winStatus = SetNamedSecurityInfo(
|
|
tchLLSDirPath, // object name
|
|
SE_FILE_OBJECT , // object type
|
|
DACL_SECURITY_INFORMATION | UNPROTECTED_DACL_SECURITY_INFORMATION , // type
|
|
NULL, // new owner SID
|
|
NULL, // new primary group SID
|
|
pNewDacl, // new DACL
|
|
NULL // new SACL
|
|
);
|
|
cleanup:
|
|
|
|
if(pSid)
|
|
{
|
|
LocalFree( pSid );
|
|
}
|
|
|
|
if(pSD)
|
|
{
|
|
LocalFree(pSD);
|
|
pSD = NULL;
|
|
}
|
|
|
|
if(pNewDacl)
|
|
{
|
|
LocalFree(pNewDacl);
|
|
pNewDacl = NULL;
|
|
}
|
|
}
|
|
|
|
void CreateFileWithAccess()
|
|
{
|
|
DWORD winStatus = 0;
|
|
TCHAR tchWinDirPath[MAX_PATH+1] = L"";
|
|
PACL pNewDacl = NULL;
|
|
PACL pOldDacl = NULL;
|
|
PSECURITY_DESCRIPTOR pSD = NULL;
|
|
TCHAR tchCPLFilePath[ MAX_PATH+1 ] = L"";
|
|
BOOL bFlag = FALSE;
|
|
PSID pSid = NULL;
|
|
EXPLICIT_ACCESS ExplicitEntries;
|
|
SID_IDENTIFIER_AUTHORITY ntSidAuthority = SECURITY_NT_AUTHORITY;
|
|
HANDLE hFile = NULL;
|
|
|
|
winStatus = GetSystemWindowsDirectory( tchWinDirPath , MAX_PATH+1);
|
|
if(winStatus == 0)
|
|
{
|
|
goto cleanup;
|
|
}
|
|
|
|
lstrcpy(tchCPLFilePath, tchWinDirPath);
|
|
lstrcat( tchCPLFilePath , L"\\system32\\cpl.cfg" );
|
|
|
|
|
|
// Creating new EXPLICIT_ACCESS structure to set on the file
|
|
|
|
ZeroMemory( &ExplicitEntries, sizeof(ExplicitEntries) );
|
|
|
|
bFlag = AllocateAndInitializeSid(
|
|
&ntSidAuthority,
|
|
1,
|
|
SECURITY_NETWORK_SERVICE_RID,0,
|
|
0, 0, 0, 0, 0, 0,
|
|
&pSid );
|
|
|
|
|
|
if ( !bFlag || (pSid == NULL) ) {
|
|
|
|
goto cleanup;
|
|
}
|
|
|
|
BuildTrusteeWithSid( &ExplicitEntries.Trustee, pSid );
|
|
|
|
ExplicitEntries.grfAccessMode = SET_ACCESS;
|
|
ExplicitEntries.grfAccessPermissions = FILE_ALL_ACCESS;
|
|
ExplicitEntries.Trustee.TrusteeForm = TRUSTEE_IS_SID;
|
|
ExplicitEntries.Trustee.TrusteeType = TRUSTEE_IS_WELL_KNOWN_GROUP;
|
|
|
|
|
|
hFile = CreateFile(tchCPLFilePath,
|
|
GENERIC_READ | GENERIC_WRITE,
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
NULL,
|
|
OPEN_ALWAYS,
|
|
0,
|
|
NULL);
|
|
|
|
|
|
if(hFile == NULL)
|
|
{
|
|
winStatus = GetLastError();
|
|
|
|
if (winStatus != ERROR_ALREADY_EXISTS) {
|
|
goto cleanup ;
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if( GetNamedSecurityInfoW( tchCPLFilePath,
|
|
SE_FILE_OBJECT,
|
|
DACL_SECURITY_INFORMATION,
|
|
NULL, // psidOwner
|
|
NULL, // psidGroup
|
|
&pOldDacl, // pDacl
|
|
NULL, // pSacl
|
|
&pSD ) != ERROR_SUCCESS)
|
|
|
|
{
|
|
goto cleanup;
|
|
}
|
|
|
|
//
|
|
// Set the Acl with the ExplicitEntry rights
|
|
//
|
|
if( SetEntriesInAcl( 1,
|
|
&ExplicitEntries,
|
|
pOldDacl,
|
|
&pNewDacl ) != ERROR_SUCCESS)
|
|
{
|
|
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
//
|
|
// SET security on the File
|
|
//
|
|
|
|
|
|
|
|
winStatus = SetNamedSecurityInfo(
|
|
tchCPLFilePath, // object name
|
|
SE_FILE_OBJECT , // object type
|
|
DACL_SECURITY_INFORMATION | UNPROTECTED_DACL_SECURITY_INFORMATION , // type
|
|
NULL, // new owner SID
|
|
NULL, // new primary group SID
|
|
pNewDacl, // new DACL
|
|
NULL // new SACL
|
|
);
|
|
|
|
|
|
cleanup:
|
|
|
|
if(hFile)
|
|
{
|
|
CloseHandle(hFile);
|
|
}
|
|
|
|
if(pSid)
|
|
{
|
|
LocalFree( pSid );
|
|
}
|
|
|
|
if(pSD)
|
|
{
|
|
LocalFree(pSD);
|
|
pSD = NULL;
|
|
}
|
|
|
|
if(pNewDacl)
|
|
{
|
|
LocalFree(pNewDacl);
|
|
pNewDacl = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
static
|
|
int
|
|
MessageBoxFromStringID(
|
|
HWND hwndParent,
|
|
UINT uTextID,
|
|
UINT uCaptionID,
|
|
UINT uType )
|
|
//
|
|
// Same as MessageBox(), except Text and Caption are string resources
|
|
// instead of string pointers.
|
|
//
|
|
{
|
|
int nButton;
|
|
TCHAR szText[ 1024 ];
|
|
TCHAR szCaption[ 64 ];
|
|
|
|
LoadString( g_hinst, uTextID, szText, sizeof( szText ) / sizeof( *szText ) );
|
|
LoadString( g_hinst, uCaptionID, szCaption, sizeof( szCaption ) / sizeof( *szCaption ) );
|
|
|
|
nButton = MessageBox( hwndParent, szText, szCaption, uType );
|
|
|
|
return nButton;
|
|
}
|
|
|
|
|
|
|