2020-09-30 16:53:55 +02:00

1752 lines
54 KiB
C

/*******************************************************************************
*
* Copyright 1999 American Power Conversion, All Rights Reserved
*
* TITLE: UPSREG.C
*
* VERSION: 1.0
*
* AUTHOR: SteveT
*
* DATE: 07 June, 1999
*
* Revision History:
* v-stebe 23May2000 added check for NULL in ReadRegistryValue() (bug 112595)
* v-stebe 23May2000 added code to avoid rewriting the registry value if the
* value is unchanged in setDwordValue() (bug #92799)
* v-stebe 11Sep2000 Fixed PREfix error (bug #170456)
*******************************************************************************/
/*
* system includes
*/
#include <windows.h>
#include <tchar.h>
/*
* local includes
*/
//#include "upsdefines.h"
#include "upsreg.h"
#include "regdefs.h"
#ifdef __cplusplus
extern "C" {
#endif
/*
* Reg entry info structure declaration
*/
struct _reg_entry
{
HKEY hKey; /* the key */
LPTSTR lpSubKey; /* address of SubKey name */
LPTSTR lpValueName; /* address of name of value to query */
DWORD ulType; /* buffer for value type */
LPBYTE lpData; /* address of data buffer */
DWORD cbData; /* data buffer size */
BOOL changed; /* ID of dialog that changed this entry */
};
/*
* local function pre-declarations
*/
void freeBlock(struct _reg_entry *aBlock[]);
void readBlock(struct _reg_entry *aBlock[], BOOL changed);
void writeBlock(struct _reg_entry *aBlock[], BOOL forceAll);
LONG setDwordValue(struct _reg_entry *aRegEntry, DWORD aValue);
LONG setStringValue(struct _reg_entry *aRegEntry, LPCTSTR aBuffer);
static BOOL isRegistryInitialized();
static void CheckForUpgrade();
static void InitializeServiceKeys();
static void InitializeServiceProviders();
static void InitializeConfigValues();
static void InitializeStatusValues();
/*
* Reg entry value name declarations
*/
#define UPS_VENDOR _T("Vendor")
#define UPS_MODEL _T("Model")
#define UPS_SERIALNUMBER _T("SerialNumber")
#define UPS_FIRMWAREREV _T("FirmwareRev")
#define UPS_UTILITYSTATUS _T("UtilityPowerStatus")
#define UPS_RUNTIME _T("TotalUPSRuntime")
#define UPS_BATTERYSTATUS _T("BatteryStatus")
#define UPS_PORT _T("Port")
#define UPS_OPTIONS _T("Options")
#define UPS_SHUTDOWNWAIT _T("ShutdownWait")
#define UPS_FIRSTMESSAGEDELAY _T("FirstMessageDelay")
#define UPS_MESSAGEINTERVAL _T("MessageInterval")
#define UPS_SERVICEDLL _T("ServiceProviderDLL")
#define UPS_NOTIFYENABLE _T("NotifyEnable")
#define UPS_SHUTBATTENABLE _T("ShutdownOnBatteryEnable")
#define UPS_SHUTBATTWAIT _T("ShutdownOnBatteryWait")
#define UPS_RUNTASKENABLE _T("RunTaskEnable")
#define UPS_TASKNAME _T("TaskName")
#define UPS_TURNUPSOFFENABLE _T("TurnUPSOffEnable")
#define UPS_APCLINKURL _T("APCLinkURL")
#define UPS_CUSTOMOPTIONS _T("CustomOptions")
#define UPS_UPGRADE _T("Upgrade")
#define UPS_COMMSTATUS _T("CommStatus")
#define UPS_CRITICALPOWERACTION _T("CriticalPowerAction")
#define UPS_TURNUPSOFFWAIT _T("TurnUPSOffWait")
#define UPS_SHOWTAB _T("ShowUPSTab")
#define UPS_BATTERYCAPACITY _T("BatteryCapacity")
#define UPS_IMAGEPATH _T("ImagePath")
#define UPS_ERRORCONTROL _T("ErrorControl")
#define UPS_OBJECTNAME _T("ObjectName")
#define UPS_START _T("Start")
#define UPS_TYPE _T("Type")
// This specifies the key to examine to determine if the registry
// has been updated for the UPS Service.
#define UPS_SERVICE_INITIALIZED_KEY TEXT("SYSTEM\\CurrentControlSet\\Services\\UPS\\Config")
// Specifies the name of the BatteryLife key used in the NT 4.0 UPS Service
#define UPS_BATTLIFE_KEY TEXT("BatteryLife")
// This specifies the default name for the shutdown Task
#define DEFAULT_SHUTDOWN_TASK_NAME TEXT("")
// Default values for the Config settings
#define DEFAULT_CONFIG_VENDOR_OLD TEXT("\\(NONE)")
#define DEFAULT_CONFIG_VENDOR TEXT("")
#define DEFAULT_CONFIG_MODEL TEXT("")
#define DEFAULT_CONFIG_PORT TEXT("COM1")
#define DEFAULT_CONFIG_OPTIONS 0x7e
#define DEFAULT_CONFIG_FIRSTMSG_DELAY 5
#define DEFAULT_CONFIG_MESSAGE_INTERVAL 120
#define DEFAULT_CONFIG_PROVIDER_DLL TEXT("")
#define DEFAULT_CONFIG_NOTIFY_ENABLE 1
#define DEFAULT_CONFIG_SHUTDOWN_ONBATT_ENABLE FALSE
#define DEFAULT_CONFIG_SHUTDOWN_ONBATT_WAIT 2
#define DEFAULT_CONFIG_RUNTASK_ENABLE FALSE
#define DEFAULT_CONFIG_TASK_NAME DEFAULT_SHUTDOWN_TASK_NAME
#define DEFAULT_CONFIG_TURNOFF_UPS_ENABLE TRUE
#define DEFAULT_CONFIG_CUSTOM_OPTIONS UPS_DEFAULT_SIGMASK
#define DEFAULT_CONFIG_CRITICALPOWERACTION UPS_SHUTDOWN_SHUTDOWN
#define DEFAULT_CONFIG_TURNOFF_UPS_WAIT 180
#define DEFAULT_CONFIG_ERRORCONTROL 1
#define DEFAULT_CONFIG_OBJECTNAME TEXT("LocalSystem")
#define DEFAULT_CONFIG_START SERVICE_DEMAND_START
#define DEFAULT_CONFIG_TYPE 16
#define DEFAULT_CONFIG_SHOWUPSTAB FALSE
// Default values for the Status settings
#define DEFAULT_STATUS_SERIALNO TEXT("")
#define DEFAULT_STATUS_FIRMWARE_REV TEXT("")
#define DEFAULT_STATUS_UTILITY_STAT 0
#define DEFAULT_STATUS_TOTAL_RUNTIME 0
#define DEFAULT_STATUS_BATTERY_STAT 0
#define DEFAULT_STATUS_BATTERY_CAPACITY 0
// Default values for upgraded services
#define UPGRADE_CONFIG_VENDOR_OLD TEXT("\\Generic")
#define UPGRADE_CONFIG_VENDOR TEXT("")
#define UPGRADE_CONFIG_MODEL TEXT("")
/*
* Allocate the individual Configuration Reg entry records
*/
struct _reg_entry UPSConfigVendor = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_VENDOR,REG_SZ,NULL,0,FALSE};
struct _reg_entry UPSConfigModel = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_MODEL,REG_SZ,NULL,0,FALSE};
struct _reg_entry UPSConfigPort = {HKEY_LOCAL_MACHINE,UPS_DEFAULT_ROOT,UPS_PORT,REG_SZ,NULL,0,FALSE};
struct _reg_entry UPSConfigOptions = {HKEY_LOCAL_MACHINE,UPS_DEFAULT_ROOT,UPS_OPTIONS,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigServiceDLL = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_SERVICEDLL,REG_SZ,NULL,0,FALSE};
struct _reg_entry UPSConfigNotifyEnable = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_NOTIFYENABLE,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigFirstMessageDelay= {HKEY_LOCAL_MACHINE,UPS_DEFAULT_ROOT,UPS_FIRSTMESSAGEDELAY,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigMessageInterval = {HKEY_LOCAL_MACHINE,UPS_DEFAULT_ROOT,UPS_MESSAGEINTERVAL,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigShutBattEnable = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_SHUTBATTENABLE,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigShutBattWait = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_SHUTBATTWAIT,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigRunTaskEnable = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_RUNTASKENABLE,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigTaskName = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_TASKNAME,REG_SZ,NULL,0,FALSE};
struct _reg_entry UPSConfigTurnOffEnable = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_TURNUPSOFFENABLE,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigCustomOptions = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_CUSTOMOPTIONS,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigAPCLinkURL = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_APCLINKURL,REG_SZ,NULL,0,FALSE};
struct _reg_entry UPSConfigShutdownWait = {HKEY_LOCAL_MACHINE,UPS_DEFAULT_ROOT,UPS_SHUTDOWNWAIT,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigUpgrade = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_UPGRADE,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigCriticalPowerAction = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_CRITICALPOWERACTION,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigTurnOffWait = {HKEY_LOCAL_MACHINE,UPS_CONFIG_ROOT,UPS_TURNUPSOFFWAIT,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigImagePath = {HKEY_LOCAL_MACHINE,UPS_DEFAULT_ROOT,UPS_IMAGEPATH,REG_EXPAND_SZ,NULL,0,FALSE};
struct _reg_entry UPSConfigObjectName = {HKEY_LOCAL_MACHINE,UPS_DEFAULT_ROOT,UPS_OBJECTNAME,REG_EXPAND_SZ,NULL,0,FALSE};
struct _reg_entry UPSConfigErrorControl = {HKEY_LOCAL_MACHINE,UPS_DEFAULT_ROOT,UPS_ERRORCONTROL,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigStart = {HKEY_LOCAL_MACHINE,UPS_DEFAULT_ROOT,UPS_START,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigType = {HKEY_LOCAL_MACHINE,UPS_DEFAULT_ROOT,UPS_TYPE,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSConfigShowUPSTab = {HKEY_LOCAL_MACHINE,UPS_DEFAULT_ROOT,UPS_SHOWTAB,REG_DWORD,NULL,0,FALSE};
/*
* Allocate the individual Status Reg entry records
*/
struct _reg_entry UPSStatusSerialNum = {HKEY_LOCAL_MACHINE,UPS_STATUS_ROOT,UPS_SERIALNUMBER,REG_SZ,NULL,0,FALSE};
struct _reg_entry UPSStatusFirmRev = {HKEY_LOCAL_MACHINE,UPS_STATUS_ROOT,UPS_FIRMWAREREV,REG_SZ,NULL,0,FALSE};
struct _reg_entry UPSStatusUtilityStatus= {HKEY_LOCAL_MACHINE,UPS_STATUS_ROOT,UPS_UTILITYSTATUS,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSStatusRuntime = {HKEY_LOCAL_MACHINE,UPS_STATUS_ROOT,UPS_RUNTIME,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSStatusBatteryStatus= {HKEY_LOCAL_MACHINE,UPS_STATUS_ROOT,UPS_BATTERYSTATUS,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSStatusCommStatus = {HKEY_LOCAL_MACHINE,UPS_STATUS_ROOT,UPS_COMMSTATUS,REG_DWORD,NULL,0,FALSE};
struct _reg_entry UPSStatusBatteryCapacity = {HKEY_LOCAL_MACHINE,UPS_STATUS_ROOT,UPS_BATTERYCAPACITY,REG_DWORD,NULL,0,FALSE};
/*
* Allocate an array of pointers to the Configuration Reg entry records
*/
struct _reg_entry *ConfigBlock[] = {&UPSConfigVendor,
&UPSConfigModel,
&UPSConfigPort,
&UPSConfigOptions,
&UPSConfigServiceDLL,
&UPSConfigNotifyEnable,
&UPSConfigFirstMessageDelay,
&UPSConfigMessageInterval,
&UPSConfigShutBattEnable,
&UPSConfigShutBattWait,
&UPSConfigRunTaskEnable,
&UPSConfigTaskName,
&UPSConfigTurnOffEnable,
&UPSConfigCustomOptions,
&UPSConfigAPCLinkURL,
&UPSConfigShutdownWait,
&UPSConfigUpgrade,
&UPSConfigCriticalPowerAction,
&UPSConfigTurnOffWait,
&UPSConfigImagePath,
&UPSConfigObjectName,
&UPSConfigErrorControl,
&UPSConfigStart,
&UPSConfigType,
&UPSConfigShowUPSTab,
NULL};
/*
* Allocate an array of pointers to the Status Reg entry records
*/
struct _reg_entry *StatusBlock[] = {&UPSStatusSerialNum,
&UPSStatusFirmRev,
&UPSStatusUtilityStatus,
&UPSStatusRuntime,
&UPSStatusBatteryStatus,
&UPSStatusCommStatus,
&UPSStatusBatteryCapacity,
NULL};
/******************************************************************
* Public functions
*/
LONG GetUPSConfigVendor(LPTSTR aBuffer, size_t aBufferLen)
{
return getStringValue( &UPSConfigVendor, aBuffer, aBufferLen);
}
LONG GetUPSConfigModel(LPTSTR aBuffer, size_t aBufferLen)
{
return getStringValue(&UPSConfigModel, aBuffer, aBufferLen);
}
LONG GetUPSConfigPort(LPTSTR aBuffer, size_t aBufferLen)
{
return getStringValue(&UPSConfigPort, aBuffer, aBufferLen);
}
LONG GetUPSConfigOptions( LPDWORD aValue)
{
return getDwordValue( &UPSConfigOptions, aValue);
}
LONG GetUPSConfigServiceDLL(LPTSTR aBuffer, size_t aBufferLen)
{
return getStringValue(&UPSConfigServiceDLL, aBuffer, aBufferLen);
}
LONG GetUPSConfigNotifyEnable( LPDWORD aValue)
{
return getDwordValue( &UPSConfigNotifyEnable, aValue);
}
LONG GetUPSConfigFirstMessageDelay( LPDWORD aValue)
{
return getDwordValue( &UPSConfigFirstMessageDelay, aValue);
}
LONG GetUPSConfigMessageInterval( LPDWORD aValue)
{
return getDwordValue( &UPSConfigMessageInterval, aValue);
}
LONG GetUPSConfigShutdownOnBatteryEnable( LPDWORD aValue)
{
return getDwordValue( &UPSConfigShutBattEnable, aValue);
}
LONG GetUPSConfigShutdownOnBatteryWait( LPDWORD aValue)
{
return getDwordValue( &UPSConfigShutBattWait, aValue);
}
LONG GetUPSConfigRunTaskEnable( LPDWORD aValue)
{
return getDwordValue( &UPSConfigRunTaskEnable, aValue);
}
LONG GetUPSConfigTaskName( LPTSTR aBuffer, size_t aBufferLen)
{
return getStringValue(&UPSConfigTaskName, aBuffer, aBufferLen);
}
LONG GetUPSConfigTurnOffEnable( LPDWORD aValue)
{
return getDwordValue( &UPSConfigTurnOffEnable, aValue);
}
LONG GetUPSConfigCustomOptions( LPDWORD aValue)
{
return getDwordValue( &UPSConfigCustomOptions, aValue);
}
LONG GetUPSConfigAPCLinkURL(LPTSTR aBuffer, size_t aBufferLen)
{
return getStringValue(&UPSConfigAPCLinkURL, aBuffer, aBufferLen);
}
LONG GetUPSConfigShutdownWait( LPDWORD aValue)
{
return getDwordValue( &UPSConfigShutdownWait, aValue);
}
LONG GetUPSConfigUpgrade( LPDWORD aValue)
{
return getDwordValue( &UPSConfigUpgrade, aValue);
}
LONG GetUPSConfigCriticalPowerAction( LPDWORD aValue)
{
return getDwordValue( &UPSConfigCriticalPowerAction, aValue);
}
LONG GetUPSConfigTurnOffWait( LPDWORD aValue)
{
return getDwordValue( &UPSConfigTurnOffWait, aValue);
}
LONG GetUPSConfigShowUPSTab( LPDWORD aValue)
{
return getDwordValue( &UPSConfigShowUPSTab, aValue);
}
LONG GetUPSConfigImagePath(LPTSTR aBuffer, size_t aBufferLen)
{
return getStringValue(&UPSConfigImagePath, aBuffer, aBufferLen);
}
LONG GetUPSConfigObjectName(LPTSTR aBuffer, size_t aBufferLen)
{
return getStringValue(&UPSConfigObjectName, aBuffer, aBufferLen);
}
LONG GetUPSConfigErrorControl( LPDWORD aValue)
{
return getDwordValue( &UPSConfigErrorControl, aValue);
}
LONG GetUPSConfigStart( LPDWORD aValue)
{
return getDwordValue( &UPSConfigStart, aValue);
}
LONG GetUPSConfigType( LPDWORD aValue)
{
return getDwordValue( &UPSConfigType, aValue);
}
///////////////////////////////////////////
LONG SetUPSConfigVendor( LPCTSTR aBuffer)
{
return setStringValue( &UPSConfigVendor, aBuffer);
}
LONG SetUPSConfigModel( LPCTSTR aBuffer)
{
return setStringValue( &UPSConfigModel, aBuffer);
}
LONG SetUPSConfigPort( LPCTSTR aBuffer)
{
return setStringValue( &UPSConfigPort, aBuffer);
}
LONG SetUPSConfigOptions( DWORD aValue)
{
return setDwordValue( &UPSConfigOptions, aValue);
}
LONG SetUPSConfigServiceDLL( LPCTSTR aBuffer)
{
return setStringValue( &UPSConfigServiceDLL, aBuffer);
}
LONG SetUPSConfigNotifyEnable( DWORD aValue)
{
return setDwordValue( &UPSConfigNotifyEnable, aValue);
}
LONG SetUPSConfigFirstMessageDelay( DWORD aValue)
{
return setDwordValue( &UPSConfigFirstMessageDelay, aValue);
}
LONG SetUPSConfigMessageInterval( DWORD aValue)
{
return setDwordValue( &UPSConfigMessageInterval, aValue);
}
LONG SetUPSConfigShutdownOnBatteryEnable( DWORD aValue)
{
return setDwordValue( &UPSConfigShutBattEnable, aValue);
}
LONG SetUPSConfigShutdownOnBatteryWait( DWORD aValue)
{
return setDwordValue( &UPSConfigShutBattWait, aValue);
}
LONG SetUPSConfigRunTaskEnable( DWORD aValue)
{
return setDwordValue( &UPSConfigRunTaskEnable, aValue);
}
LONG SetUPSConfigTaskName( LPCTSTR aBuffer)
{
return setStringValue( &UPSConfigTaskName, aBuffer);
}
LONG SetUPSConfigTurnOffEnable( DWORD aValue)
{
return setDwordValue( &UPSConfigTurnOffEnable, aValue);
}
LONG SetUPSConfigCustomOptions( DWORD aValue)
{
return setDwordValue( &UPSConfigCustomOptions, aValue);
}
LONG SetUPSConfigAPCLinkURL( LPCTSTR aBuffer)
{
return setStringValue( &UPSConfigAPCLinkURL, aBuffer);
}
LONG SetUPSConfigShutdownWait( DWORD aValue)
{
return setDwordValue( &UPSConfigShutdownWait, aValue);
}
LONG SetUPSConfigUpgrade( DWORD aValue)
{
return setDwordValue( &UPSConfigUpgrade, aValue);
}
LONG SetUPSConfigCriticalPowerAction( DWORD aValue)
{
return setDwordValue( &UPSConfigCriticalPowerAction, aValue);
}
LONG SetUPSConfigTurnOffWait( DWORD aValue)
{
return setDwordValue( &UPSConfigTurnOffWait, aValue);
}
LONG SetUPSConfigShowUPSTab( DWORD aValue)
{
return setDwordValue( &UPSConfigShowUPSTab, aValue);
}
LONG SetUPSConfigImagePath( LPCTSTR aBuffer)
{
return setStringValue( &UPSConfigImagePath, aBuffer);
}
LONG SetUPSConfigObjectName( LPCTSTR aBuffer)
{
return setStringValue( &UPSConfigObjectName, aBuffer);
}
LONG SetUPSConfigErrorControl( DWORD aValue)
{
return setDwordValue( &UPSConfigErrorControl, aValue);
}
LONG SetUPSConfigStart( DWORD aValue)
{
return setDwordValue( &UPSConfigStart, aValue);
}
LONG SetUPSConfigType( DWORD aValue)
{
return setDwordValue( &UPSConfigType, aValue);
}
////////////////////////////////////////////////
LONG GetUPSStatusSerialNum(LPTSTR aBuffer, size_t aBufferLen)
{
return getStringValue( &UPSStatusSerialNum, aBuffer, aBufferLen);
}
LONG GetUPSStatusFirmRev(LPTSTR aBuffer, size_t aBufferLen)
{
return getStringValue(&UPSStatusFirmRev, aBuffer, aBufferLen);
}
LONG GetUPSStatusUtilityStatus( LPDWORD aValue)
{
return getDwordValue( &UPSStatusUtilityStatus, aValue);
}
LONG GetUPSStatusRuntime( LPDWORD aValue)
{
return getDwordValue( &UPSStatusRuntime, aValue);
}
LONG GetUPSStatusBatteryStatus( LPDWORD aValue)
{
return getDwordValue( &UPSStatusBatteryStatus, aValue);
}
LONG GetUPSStatusCommStatus( LPDWORD aValue)
{
return getDwordValue( &UPSStatusCommStatus, aValue);
}
LONG GetUPSBatteryCapacity( LPDWORD aValue)
{
return getDwordValue( &UPSStatusBatteryCapacity, aValue);
}
/////////////////////////////////////////
LONG SetUPSStatusSerialNum( LPCTSTR aBuffer)
{
return setStringValue( &UPSStatusSerialNum, aBuffer);
}
LONG SetUPSStatusFirmRev( LPCTSTR aBuffer)
{
return setStringValue( &UPSStatusFirmRev, aBuffer);
}
LONG SetUPSStatusUtilityStatus( DWORD aValue)
{
return setDwordValue( &UPSStatusUtilityStatus,aValue);
}
LONG SetUPSStatusRuntime( DWORD aValue)
{
return setDwordValue( &UPSStatusRuntime,aValue);
}
LONG SetUPSStatusBatteryStatus( DWORD aValue)
{
return setDwordValue( &UPSStatusBatteryStatus,aValue);
}
LONG SetUPSStatusCommStatus( DWORD aValue)
{
return setDwordValue( &UPSStatusCommStatus,aValue);
}
LONG SetUPSStatusBatteryCapacity( DWORD aValue)
{
return setDwordValue( &UPSStatusBatteryCapacity,aValue);
}
//////////////////////////////////////////////////////////////
void FreeUPSConfigBlock()
{
freeBlock(ConfigBlock);
}
void FreeUPSStatusBlock()
{
freeBlock(StatusBlock);
}
void InitUPSConfigBlock()
{
readBlock(ConfigBlock,FALSE);
}
void InitUPSStatusBlock()
{
readBlock(StatusBlock,FALSE);
}
void RestoreUPSConfigBlock()
{
readBlock(ConfigBlock, TRUE);
}
void RestoreUPSStatusBlock()
{
readBlock(StatusBlock, TRUE);
}
void SaveUPSConfigBlock(BOOL forceAll)
{
writeBlock(ConfigBlock, forceAll);
}
void SaveUPSStatusBlock(BOOL forceAll)
{
writeBlock(StatusBlock, forceAll);
}
/******************************************************************
* Local functions
*/
/*
* freeBlock()
*
* Description: Frees storage allocated when a block of registry
* entries is read
*
* Parameters: aBlock - pointer to an array of _reg_entry structures
*
* Returns:
*/
void freeBlock(struct _reg_entry *aBlock[])
{
while ((NULL != aBlock) && (NULL != *aBlock))
{
struct _reg_entry *anEntry = *aBlock;
if (NULL != anEntry->lpData)
{
LocalFree(anEntry->lpData);
}
anEntry->lpData = NULL;
anEntry->cbData = 0;
anEntry->changed = FALSE;
aBlock++;
}
}
/*
* readBlock()
*
* Description: Loads all of the items in a array of registry entries
*
* Parameters: aBlock - pointer to an array of _reg_entry structures
* changed - boolean which, when true, causes only the
* structures that are marked as changed to be loaded.
*
* Returns:
*/
void readBlock(struct _reg_entry *aBlock[], BOOL changed)
{
LONG res;
HKEY hkResult;
while ((NULL != aBlock) && (NULL != *aBlock))
{
struct _reg_entry *anEntry = *aBlock;
/*
* if changed is FALSE, we read all entries
* otherwise, only re-read the changed entries
*/
if ((FALSE == changed) || (TRUE == anEntry->changed))
{
/*
* delete current value, in case this is a reload
*/
if (NULL != anEntry->lpData)
{
LocalFree(anEntry->lpData);
}
anEntry->lpData = NULL;
anEntry->cbData = 0;
anEntry->changed = FALSE;
/*
* open key
*/
res = RegOpenKeyEx( anEntry->hKey,
anEntry->lpSubKey,
0,
KEY_QUERY_VALUE,
&hkResult);
if (ERROR_SUCCESS == res)
{
DWORD ulTmpType;
/*
* query for the data size
*/
res = RegQueryValueEx( hkResult,
anEntry->lpValueName,
NULL,
&ulTmpType,
NULL,
&anEntry->cbData);
/*
* if the data has 0 size, we don't read it
*/
if ((ERROR_SUCCESS == res) &&
(anEntry->cbData > 0) &&
(anEntry->ulType == ulTmpType) &&
(NULL != (anEntry->lpData = (LPBYTE)LocalAlloc(LMEM_FIXED, anEntry->cbData))))
// (NULL != (anEntry->lpData = (LPBYTE)malloc(anEntry->cbData))))
{
/*
* query for data
*/
res = RegQueryValueEx( hkResult,
anEntry->lpValueName,
NULL,
&ulTmpType,
anEntry->lpData,
&anEntry->cbData);
/*
* something went wrong; reset
*/
if (ERROR_SUCCESS != res)
{
LocalFree(anEntry->lpData);
anEntry->lpData = NULL;
anEntry->cbData = 0;
}
}
else
{
anEntry->cbData = 0;
}
RegCloseKey(hkResult);
}
}
aBlock++;
}
}
/*
* writeBlock()
*
* Description: Stores all of the items in a array of registry entries
*
* Parameters: aBlock - pointer to an array of _reg_entry structures
* forceAll - boolean which, when true, causes all of the
* structures to be written to the registry, otherwise only
* those entries that are marked as changed are stored.
*
* Returns:
*/
void writeBlock(struct _reg_entry *aBlock[], BOOL forceAll)
{
LONG res;
HKEY hkResult;
while ((NULL != aBlock) && (NULL != *aBlock))
{
struct _reg_entry *anEntry = *aBlock;
/*
* if forcall is true, write out everything
* otherwise only write out the changed entries
*/
if ((NULL != anEntry->lpData) &&
((TRUE == forceAll) || (TRUE == anEntry->changed)))
{
/*
* open key
*/
res = RegOpenKeyEx( anEntry->hKey,
anEntry->lpSubKey,
0,
KEY_SET_VALUE,
&hkResult);
if (ERROR_SUCCESS == res)
{
/*
* set data
*/
res = RegSetValueEx( hkResult,
anEntry->lpValueName,
0,
anEntry->ulType,
anEntry->lpData,
anEntry->cbData);
RegCloseKey(hkResult);
}
anEntry->changed = FALSE;
}
aBlock++;
}
}
/*
* setDwordValue()
*
* Description: Sets the value of a REG_DWORD entry record.
*
* Parameters: aRegEntry - pointer to a _reg_entry structure
* aValue - the value to store in the entry
*
* Returns: ERROR_SUCCESS, E_OUTOFMEMORY, ERROR_INVALID_PARAMETER
*/
LONG setDwordValue(struct _reg_entry *aRegEntry, DWORD aValue)
{
LONG res = ERROR_SUCCESS;
DWORD value_changed = TRUE;
if (NULL != aRegEntry)
{
/*
* Check to see if a value already exists
*/
if (NULL != aRegEntry->lpData)
{
/*
* If the value is the different, delete it.
*/
if (memcmp(aRegEntry->lpData, &aValue, sizeof(aValue)) != 0) {
LocalFree (aRegEntry->lpData);
aRegEntry->lpData = NULL;
aRegEntry->cbData = 0;
}
else {
/*
* The value is the same, don't change it. That would cause an
* unnecessary write to the registry (see bug #92799)
*/
value_changed = FALSE;
}
}
/*
* set value
*/
if (value_changed) {
aRegEntry->cbData = sizeof(DWORD);
if (NULL != (aRegEntry->lpData = LocalAlloc(LMEM_FIXED, aRegEntry->cbData)))
{
*((DWORD*)aRegEntry->lpData) = aValue;
aRegEntry->changed = TRUE;
}
else
{
res = E_OUTOFMEMORY;
aRegEntry->cbData = 0;
}
}
}
else
{
res = ERROR_INVALID_PARAMETER;
}
return res;
}
/*
* setStringValue()
*
* Description: Sets the value of a REG_SZ entry record.
*
* Parameters: aRegEntry - pointer to a _reg_entry structure
* aBuffer - pointer to the string to store in the entry
*
* Returns: ERROR_SUCCESS, E_OUTOFMEMORY, ERROR_INVALID_PARAMETER
*/
LONG setStringValue(struct _reg_entry *aRegEntry, LPCTSTR aBuffer)
{
LONG res = ERROR_SUCCESS;
if ((NULL != aRegEntry) && (NULL != aBuffer))
{
/*
* if value already exists, delete it
*/
if (NULL != aRegEntry->lpData)
{
LocalFree(aRegEntry->lpData);
aRegEntry->lpData = NULL;
aRegEntry->cbData = 0;
}
/*
* set value
*/
aRegEntry->cbData = (_tcslen(aBuffer)+1)*sizeof(TCHAR);
if (NULL != (aRegEntry->lpData = LocalAlloc(LMEM_FIXED, aRegEntry->cbData)))
{
_tcscpy((LPTSTR)aRegEntry->lpData,aBuffer);
aRegEntry->changed = TRUE;
}
else
{
res = E_OUTOFMEMORY;
aRegEntry->cbData = 0;
}
}
else
{
res = ERROR_INVALID_PARAMETER;
}
return res;
}
/*
* getStringValue()
*
* Description: Gets the value of a REG_SZ entry record.
*
* Parameters:
* aRegEntry pointer to a _reg_entry structure
* aBuffer pointer to the string to receive the string
* aBufferLen the length (in characters) of the buffer
*
* Returns: ERROR_SUCCESS, REGDB_E_INVALIDVALUE, ERROR_INVALID_PARAMETER
*/
LONG getStringValue(struct _reg_entry *aRegEntry, LPTSTR aBuffer, size_t aBufferLen)
{
LONG res = REGDB_E_INVALIDVALUE;
if ((NULL != aRegEntry) && (NULL != aBuffer))
{
if (NULL != aRegEntry->lpData)
{
if (_tcslen((LPCTSTR) aRegEntry->lpData) <= aBufferLen) {
_tcscpy(aBuffer, (LPCTSTR)aRegEntry->lpData);
res = ERROR_SUCCESS;
}
}
}
else
{
res = ERROR_INVALID_PARAMETER;
}
return res;
}
/*
* getDwordValue()
*
* Description: Gets the value of a REG_DWORD entry record.
*
* Parameters: aRegEntry - pointer to a _reg_entry structure
* aValue - pointer to the variable to receive the value
*
* Returns: ERROR_SUCCESS, REGDB_E_INVALIDVALUE, ERROR_INVALID_PARAMETER
*/
LONG getDwordValue(struct _reg_entry *aRegEntry, LPDWORD aValue)
{
LONG res = ERROR_SUCCESS;
if ((NULL != aRegEntry) && (NULL != aValue))
{
if (NULL != aRegEntry->lpData)
{
*aValue = *((DWORD*)aRegEntry->lpData);
}
else
{
res = REGDB_E_INVALIDVALUE;
}
}
else
{
res = ERROR_INVALID_PARAMETER;
}
return res;
}
/**
* InitializeRegistry
*
* Description:
* This function initiates the registry for the UPS service and the
* configuration application. When called, this function calls the
* function isRegistryInitialized(..) to determine if the registry
* has been initialied. If it has not, the following Keys are updated:
* Status
* Config
* ServiceProviders
*
* The values for the ServiceProviders key is supplied in the regdefs.h
* header file.
*
* Parameters:
* none
*
* Returns:
* TRUE if able to open registry keys with write access.
*/
BOOL InitializeRegistry() {
BOOL ret_val = FALSE;
HKEY key;
TCHAR szKeyName[MAX_PATH] = _T("");
// Initialize UPS Service registry keys
InitializeServiceKeys();
// Check to see if the registry is already initialized
if (isRegistryInitialized() == FALSE) {
CheckForUpgrade();
InitializeServiceProviders();
InitializeConfigValues();
InitializeStatusValues();
}
/*
* Remove "(None)" and "Generic" Service Provider keys if they exist
* This fixes a localization bug introduced in RC2
*/
_tcscpy(szKeyName, UPS_SERVICE_ROOT);
_tcscat(szKeyName, DEFAULT_CONFIG_VENDOR_OLD);
RegDeleteKey(HKEY_LOCAL_MACHINE, szKeyName);
_tcscpy(szKeyName, UPS_SERVICE_ROOT);
_tcscat(szKeyName, UPGRADE_CONFIG_VENDOR_OLD);
RegDeleteKey(HKEY_LOCAL_MACHINE, szKeyName);
// ...and check if we have write access
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE,
UPS_DEFAULT_ROOT,
0,
KEY_WRITE,
&key) == ERROR_SUCCESS )
{
RegCloseKey(key);
ret_val = TRUE;
}
return ret_val;
}
/**
* isRegistryInitialized
*
* Description:
* This function determines if the registry has been initialized for
* the UPS service. This is done by examine the registry key specified
* by the identifier UPS_SERVICE_INITIALIED_KEY. If the key is present,
* the registry is assumed to be initialized and TRUE is returned.
* Otherwise, FALSE is returned.
*
* Parameters:
* none
*
* Returns:
* TRUE - if the registry has been initialized for the UPS service
* FALSE - otherwise
*/
static BOOL isRegistryInitialized() {
BOOL ret_val = FALSE;
HKEY key;
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, UPS_SERVICE_INITIALIZED_KEY,
0, KEY_QUERY_VALUE, &key) == ERROR_SUCCESS) {
ret_val = TRUE;
RegCloseKey(key);
}
return ret_val;
}
/**
* CheckForUpgrade
*
* Description:
* This function determines if this installation is an upgrade from
* the WINNT 4.x UPS service. This is done by checking to see if the
* Config registry key is present. If it is not present and the Options
* key is set to UPS_INSTALLED, then the Upgrade registry key is set to
* TRUE. Otherwise, it is set to FALSE.
*
* Parameters:
* none
*
* Returns:
* nothing
*/
static void CheckForUpgrade() {
DWORD result;
HKEY key;
DWORD options = 0;
// Create the Config key
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, UPS_CONFIG_ROOT, 0, NULL,
REG_OPTION_NON_VOLATILE, KEY_SET_VALUE, NULL, &key, &result) == ERROR_SUCCESS) {
// close the key, we only needed to create it
RegCloseKey(key);
// Check to see if the Config key was present
if (result != REG_OPENED_EXISTING_KEY) {
// Config key was not found
InitUPSConfigBlock();
// Check the port value
if (ERROR_SUCCESS != GetUPSConfigOptions(&options)) {
// Options key is not found
SetUPSConfigUpgrade(FALSE);
}
else if (options & UPS_INSTALLED) {
// The Options key is present and UPS_INSTALLED is set
// This is an upgrade
SetUPSConfigUpgrade(TRUE);
}
else {
// The Config key is present and UPS_INSTALLED is not set
SetUPSConfigUpgrade(FALSE);
}
}
else {
// Config key does not exist
SetUPSConfigUpgrade(FALSE);
}
// Write the Config values, force a save of all values
SaveUPSConfigBlock(TRUE);
// Free the Config block
FreeUPSConfigBlock();
}
}
/**
* InitializeServiceKeys
*
* Description:
* This function initializes the UPS service registry keys to
* default values, if the values are not present.
*
* Parameters:
* none
*
* Returns:
* nothing
*/
static void InitializeServiceKeys() {
TCHAR tmpString[MAX_PATH];
DWORD tmpDword;
// Initialize the registry functions
InitUPSConfigBlock();
// Check the service keys and initialize any missing keys
if (GetUPSConfigImagePath(tmpString, MAX_PATH) != ERROR_SUCCESS) {
SetUPSConfigImagePath(DEFAULT_CONFIG_IMAGEPATH);
}
if (GetUPSConfigObjectName(tmpString, MAX_PATH) != ERROR_SUCCESS) {
SetUPSConfigObjectName(DEFAULT_CONFIG_OBJECTNAME);
}
if (GetUPSConfigErrorControl(&tmpDword) != ERROR_SUCCESS) {
SetUPSConfigErrorControl(DEFAULT_CONFIG_ERRORCONTROL);
}
if (GetUPSConfigStart(&tmpDword) != ERROR_SUCCESS) {
SetUPSConfigStart(DEFAULT_CONFIG_START);
}
if (GetUPSConfigType(&tmpDword) != ERROR_SUCCESS) {
SetUPSConfigType(DEFAULT_CONFIG_TYPE);
}
// Write the Config values, force a save of all values
SaveUPSConfigBlock(TRUE);
// Free the Status block
FreeUPSConfigBlock();
}
/**
* InitializeServiceProviders
*
* Description:
* This function initializes the ServiceProviders registry keys with the
* data provided in the global structure _theStaticProvidersTable. This
* structure is defined in the file regdefs.h and is automatically
* generated.
*
* Parameters:
* none
*
* Returns:
* nothing
*/
static void InitializeServiceProviders() {
DWORD result;
HKEY key;
int index = 0;
// Loop through the list of Service Providers
while (_theStaticProvidersTable[index].theModelName != NULL) {
// Open the vendor registry key
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, _theStaticProvidersTable[index].theVendorKey,
0, NULL, REG_OPTION_NON_VOLATILE, KEY_SET_VALUE, NULL, &key, &result) == ERROR_SUCCESS) {
// Set the model value
RegSetValueEx(key, _theStaticProvidersTable[index].theModelName, 0, REG_SZ,
(LPSTR) _theStaticProvidersTable[index].theValue,
wcslen(_theStaticProvidersTable[index].theValue)*sizeof(TCHAR));
RegCloseKey(key);
}
// Increment counter
index++;
}
}
/**
* InitializeConfigValues
*
* Description:
* This function initializes the Config registry keys with
* default values.
*
* Parameters:
* none
*
* Returns:
* nothing
*/
static void InitializeConfigValues() {
DWORD result;
HKEY key;
DWORD options_val, batt_life, type;
DWORD batt_life_size = sizeof(DWORD);
// Create the Config key
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, UPS_CONFIG_ROOT, 0, NULL,
REG_OPTION_NON_VOLATILE, KEY_SET_VALUE, NULL, &key, &result) == ERROR_SUCCESS) {
// close the key, we only needed to create it
RegCloseKey(key);
// Initialize the registry functions
InitUPSConfigBlock();
// Set default values
SetUPSConfigServiceDLL(DEFAULT_CONFIG_PROVIDER_DLL);
SetUPSConfigNotifyEnable(DEFAULT_CONFIG_NOTIFY_ENABLE);
SetUPSConfigShutdownOnBatteryEnable(DEFAULT_CONFIG_SHUTDOWN_ONBATT_ENABLE);
SetUPSConfigShutdownOnBatteryWait(DEFAULT_CONFIG_SHUTDOWN_ONBATT_WAIT);
SetUPSConfigRunTaskEnable(DEFAULT_CONFIG_RUNTASK_ENABLE);
SetUPSConfigTaskName(DEFAULT_CONFIG_TASK_NAME);
SetUPSConfigTurnOffEnable(DEFAULT_CONFIG_TURNOFF_UPS_ENABLE);
SetUPSConfigCustomOptions(DEFAULT_CONFIG_CUSTOM_OPTIONS);
SetUPSConfigCriticalPowerAction(DEFAULT_CONFIG_CRITICALPOWERACTION);
SetUPSConfigTurnOffWait(DEFAULT_CONFIG_TURNOFF_UPS_WAIT);
// If this is not an upgrade, set the appropriate values
if ((GetUPSConfigUpgrade(&result) != ERROR_SUCCESS) || (result == FALSE)) {
SetUPSConfigVendor(DEFAULT_CONFIG_VENDOR);
SetUPSConfigModel(DEFAULT_CONFIG_MODEL);
SetUPSConfigPort(DEFAULT_CONFIG_PORT);
SetUPSConfigOptions(DEFAULT_CONFIG_OPTIONS);
SetUPSConfigFirstMessageDelay(DEFAULT_CONFIG_FIRSTMSG_DELAY);
SetUPSConfigMessageInterval(DEFAULT_CONFIG_MESSAGE_INTERVAL);
}
else {
// This is an upgrade
SetUPSConfigVendor(UPGRADE_CONFIG_VENDOR);
SetUPSConfigModel(UPGRADE_CONFIG_MODEL);
// Migrate the run command file option bit to the RunTaskEnable key
if ((GetUPSConfigOptions(&options_val) == ERROR_SUCCESS) &&
(options_val & UPS_RUNCMDFILE)) {
// Run command file is enabled, set RunTaskEnable to TRUE
SetUPSConfigRunTaskEnable(TRUE);
}
else {
// Run command file is not enabled
SetUPSConfigRunTaskEnable(FALSE);
}
// Migrate the BatteryLife value to the ShutdownOnBatteryWait value
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, UPS_DEFAULT_ROOT, 0, KEY_ALL_ACCESS, &key) == ERROR_SUCCESS) {
result = RegQueryValueEx(key, UPS_BATTLIFE_KEY, NULL, &type, (LPBYTE) &batt_life, &batt_life_size);
if ((result == ERROR_SUCCESS) && (type == REG_DWORD)) {
// Migrate the value and enable shutdown on battery
SetUPSConfigShutdownOnBatteryWait(batt_life);
SetUPSConfigShutdownOnBatteryEnable(TRUE);
// Delete the value
RegDeleteValue(key, UPS_BATTLIFE_KEY);
}
// Close the key
RegCloseKey(key);
}
}
// Write the Config values, force a save of all values
SaveUPSConfigBlock(TRUE);
// Free the Config block
FreeUPSConfigBlock();
}
}
/**
* InitializeStatusValues
*
* Description:
* This function initializes the Status registry keys with
* default values.
*
* Parameters:
* none
*
* Returns:
* nothing
*/
static void InitializeStatusValues() {
DWORD result;
HKEY key;
// Create the Status key
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, UPS_STATUS_ROOT, 0, NULL,
REG_OPTION_NON_VOLATILE, KEY_SET_VALUE, NULL, &key, &result) == ERROR_SUCCESS) {
// close the key, we only needed to create it
RegCloseKey(key);
// Initialize the registry functions
InitUPSStatusBlock();
// Put in default values
SetUPSStatusSerialNum(DEFAULT_STATUS_SERIALNO);
SetUPSStatusFirmRev(DEFAULT_STATUS_FIRMWARE_REV);
SetUPSStatusUtilityStatus(DEFAULT_STATUS_UTILITY_STAT);
SetUPSStatusRuntime(DEFAULT_STATUS_TOTAL_RUNTIME);
SetUPSStatusBatteryStatus(DEFAULT_STATUS_BATTERY_STAT);
SetUPSStatusBatteryCapacity(DEFAULT_STATUS_BATTERY_CAPACITY);
// Write the Config values, force a save of all values
SaveUPSStatusBlock(TRUE);
// Free the Status block
FreeUPSStatusBlock();
}
}
///////////////////////////////////////////////////////////////////////////////
// upsdata.c
///////////////////////////////////////////////////////////////////////////////
//Note that the order of the following RegField is linked to the enum
//tUPSDataItemID.
//Do not change these value without due care and attention. It's OK to change
//them as long as the enum is updated to match.
//To access the RegField associated with Firmware, for example, use
//g_upsRegFields[(DWORD) eREG_FIRMWARE_REVISION]
static RegField g_upsRegFields[] = {
{ HKEY_LOCAL_MACHINE, CONFIG_KEY_NAME, TEXT("Vendor"), REG_SZ },
{ HKEY_LOCAL_MACHINE, CONFIG_KEY_NAME, TEXT("Model"), REG_SZ },
{ HKEY_LOCAL_MACHINE, STATUS_KEY_NAME, TEXT("SerialNumber"), REG_SZ },
{ HKEY_LOCAL_MACHINE, STATUS_KEY_NAME, TEXT("FirmwareRev"), REG_SZ },
{ HKEY_LOCAL_MACHINE, STATUS_KEY_NAME, TEXT("UtilityPowerStatus"), REG_DWORD },
{ HKEY_LOCAL_MACHINE, STATUS_KEY_NAME, TEXT("TotalUPSRuntime"), REG_DWORD },
{ HKEY_LOCAL_MACHINE, STATUS_KEY_NAME, TEXT("BatteryCapacity"), REG_DWORD },
{ HKEY_LOCAL_MACHINE, STATUS_KEY_NAME, TEXT("BatteryStatus"), REG_DWORD },
{ HKEY_LOCAL_MACHINE, UPS_KEY_NAME, TEXT("Options"), REG_DWORD },
{ HKEY_LOCAL_MACHINE, CONFIG_KEY_NAME, TEXT("ServiceProviderDLL"), REG_EXPAND_SZ },
{ HKEY_LOCAL_MACHINE, CONFIG_KEY_NAME, TEXT("ShutdownOnBatteryEnable"), REG_DWORD },
{ HKEY_LOCAL_MACHINE, CONFIG_KEY_NAME, TEXT("ShutdownOnBatteryWait"), REG_DWORD },
{ HKEY_LOCAL_MACHINE, CONFIG_KEY_NAME, TEXT("TurnUPSOffEnable"), REG_DWORD },
{ HKEY_LOCAL_MACHINE, CONFIG_KEY_NAME, TEXT("APCLinkURL"), REG_SZ },
{ HKEY_LOCAL_MACHINE, CONFIG_KEY_NAME, TEXT("Upgrade"), REG_DWORD },
{ HKEY_LOCAL_MACHINE, STATUS_KEY_NAME, TEXT("CommStatus"), REG_DWORD },
{ HKEY_LOCAL_MACHINE, UPS_KEY_NAME, TEXT("Port"), REG_SZ } };
// functions
///////////////////////////////////////////////////////////////////////////////
DWORD ReadRegistryValue (const tUPSDataItemID aDataItemID,
DWORD aAllowedTypes,
DWORD * aTypePtr,
LPBYTE aReturnBuffer,
DWORD * aBufferSizePtr);
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////_/_//
//////////////////////////////////////////////////////////////////////////_/_//
// RegField * GetRegField (DWORD aIndex);
//
// Description: This function returns a pointer to a RegField from the
// static array of RegFields named g_upsRegFields. The parameter
// aIndex is an index into this array.
//
// Additional Information:
//
// Parameters:
//
// DWORD aIndex :- A index into array of known RegFields g_upsRegFields
//
// Return Value: If aIndex is within range this function returns a point to
// the corresponding RegField, otherwise it ASSERTs and returns
// NULL.
//
RegField * GetRegField (DWORD aIndex) {
static const DWORD numRegFields = DIMENSION_OF(g_upsRegFields);
RegField * pRequiredReg = NULL;
if (aIndex < numRegFields) {
pRequiredReg = &g_upsRegFields[aIndex];
}
else {
_ASSERT(FALSE);
}
return(pRequiredReg);
}
//////////////////////////////////////////////////////////////////////////_/_//
//////////////////////////////////////////////////////////////////////////_/_//
// BOOL GetUPSDataItemDWORD (const tUPSDataItemID aDataItemID, DWORD * aReturnValuePtr);
//
// Description: This function reads the DWORD value from the registry that
// corresponds to the registry field identified by aDataItemID.
// The registry value must be one of the DWORD types (REG_DWORD,
// REG_DWORD_LITTLE_ENDIAN, REG_DWORD_BIG_ENDIAN)
//
// For example, if aDataItemID is eREG_UPS_OPTIONS (=7), the
// RegField at index 7 in g_upsRegFields identifies the required
// registry information. The RegField identifies that the registry
// key is HKLM\SYSTEM\CurrentControlSet\Services\UPS and the
// value name is Options and it's of type DWORD. Using this
// information this function gets the information from the
// registry and puts the result in aReturnValuePtr.
//
// Additional Information:
//
// Parameters:
//
// const tUPSDataItemID aDataItemID :- This parameter identifies the registry
// value being queried. The value ranges
// from eREG_VENDOR_NAME (which equals 0)
// to eREG_PORT, the values incrementing
// by 1 for each enum in the range. The
// range of values in tUPSDataItemID
// corresponds directly to the number of
// elements in the array g_upsRegFields
// because this enum is used to index the
// elements in g_upsRegFields.
//
// DWORD * aReturnValuePtr :- The DWORD value is returned through this
// pointer.
//
// Return Value:
//
BOOL GetUPSDataItemDWORD (const tUPSDataItemID aDataItemID, DWORD * aReturnValuePtr) {
BOOL bGotValue = FALSE;
DWORD nDWORDSize = sizeof(DWORD);
if (ReadRegistryValue(aDataItemID, REG_DWORD, NULL, (LPBYTE) aReturnValuePtr, &nDWORDSize) == ERROR_SUCCESS) {
bGotValue = TRUE;
}
return(bGotValue);
}
//////////////////////////////////////////////////////////////////////////_/_//
//////////////////////////////////////////////////////////////////////////_/_//
// BOOL GetUPSDataItemString (const tUPSDataItemID aDataItemID,
// LPTSTR aBufferPtr,
// DWORD * pSizeOfBufferPtr);
//
// Description: This function reads the string value from the registry that
// corresponds to the registry field identified by aDataItemID.
// The registry value must be one of the string types (REG_SZ or
// REG_EXPAND_SZ)
//
// Additional Information:
//
// Parameters:
//
// const tUPSDataItemID aDataItemID :- This parameter identifies the registry
// value being queried. The value must be
// one of the string types (REG_SZ or
// REG_EXPAND_SZ).
//
// LPTSTR aBufferPtr :- The buffer into which the data is to be placed. This
// parameter can be NULL in which case no data is
// retrieved.
//
// DWORD * pSizeOfBufferPtr :- This should point to a DWORD that contains the
// size of the buffer. This parameter cannot be
// NULL. When this function returns this value
// will contain the size actually required. This
// is useful if the user want to determine how
// big a buffer is required by calling this
// function with aBufferPtr set to NULL and
// pSizeOfBufferPtr pointing to a DWORD that
// is set to 0. When the function returns the
// DWORD pointed to by pSizeOfBufferPtr should
// contain the size of string required. This can
// then be used to dynamically allocate memory
// and call this function again with the buffer
// included this time.
//
// Return Value: The function returns TRUE if successful, FALSE otherwise.
//
BOOL GetUPSDataItemString (const tUPSDataItemID aDataItemID,
LPTSTR aBufferPtr,
DWORD * pSizeOfBufferPtr) {
BOOL bGotValue = FALSE;
DWORD nType = 0;
if (ReadRegistryValue(aDataItemID,
REG_SZ | REG_EXPAND_SZ,
&nType,
(LPBYTE) aBufferPtr,
pSizeOfBufferPtr) == ERROR_SUCCESS) {
//RegQueryValueEx stores the size of the data, in bytes, in the variable
//pointed to by lpcbData. If the data has the REG_SZ, REG_MULTI_SZ or
//REG_EXPAND_SZ type, then lpcbData will also include the size of the
//terminating null character.
//For Unicode the terminating NULL character is two-bytes.
if ((pSizeOfBufferPtr != NULL) && (*pSizeOfBufferPtr > sizeof(TCHAR))) {
if (nType == REG_EXPAND_SZ) {
TCHAR expandBuffer[MAX_MESSAGE_LENGTH] = TEXT("");
DWORD expandBufferSize = DIMENSION_OF(expandBuffer);
//ExpandEnvironmentStrings return number of bytes(ANSI) or
//number of character(UNICODE) including the NULL character
if (ExpandEnvironmentStrings(aBufferPtr, expandBuffer, expandBufferSize) > 0) {
_tcscpy(aBufferPtr, expandBuffer);
}
}
bGotValue = TRUE;
}
}
return(bGotValue);
}
//////////////////////////////////////////////////////////////////////////_/_//
//////////////////////////////////////////////////////////////////////////_/_//
// DWORD ReadRegistryValue (const tUPSDataItemID aDataItemID,
// DWORD aAllowedTypes,
// DWORD * aTypePtr,
// LPBYTE aReturnBuffer,
// DWORD * aBufferSizePtr);
//
// Description: This function reads the registry value identified by
// aDataItemID. This function can read any type of registry value
// but the value must match that identified in the RegField
// description for this field.
//
// For example, if aDataItemID is eREG_UPS_OPTIONS (=7), the
// RegField at index 7 in g_upsRegFields identifies the required
// registry information. The RegField identifies that the registry
// key is HKLM\SYSTEM\CurrentControlSet\Services\UPS and the
// value name is Options and it's of type DWORD. This function
// will succeed only if it's called with an aAllowedTypes value
// equal to REG_DWORD.
//
// Additional Information:
//
// Parameters:
//
// const tUPSDataItemID aDataItemID :- This parameter identifies the registry
// value being queried.
//
// DWORD aAllowedTypes :- This identifies the allowed type of the registry
// data. The registry value types are not bit value
// that can be |'d (the types are sequentially
// numbered 0, 1, 2, 3, 4, not 0, 1, 2, 4, 8).
// However, the parameter is still called
// aAllowedTypes because we actually call the function
// with a value of REG_SZ | REG_EXPAND_SZ (1 | 2) to
// allow the same function to work if the value is
// either of these. Except for this case assume that
// the aAllowedTypes is actually aAllowedType.
//
// DWORD * aTypePtr :- A pointer to the buffer that will receive the type.
// If the type is not required then this parameter can be
// set to NULL.
//
// LPBYTE aReturnBuffer :- The buffer into which the data is to be placed.
// This parameter can be NULL in which case no data
// is retrieved.
//
// DWORD * aBufferSizePtr :- This should point to a DWORD that contains the
// size of the buffer. This parameter cannot be
// NULL. When this function returns this value
// will contain the size actually required.
//
// Return Value: This function returns a Win32 error code, ERROS_SUCCESS on
// success.
//
DWORD ReadRegistryValue (const tUPSDataItemID aDataItemID,
DWORD aAllowedTypes,
DWORD * aTypePtr,
LPBYTE aReturnBuffer,
DWORD * aBufferSizePtr) {
DWORD ret_val = ERROR_NO_MATCH;
RegField * pRegField = GetRegField((DWORD) aDataItemID);
if (pRegField != NULL ) {
_ASSERT((pRegField->theValueType & aAllowedTypes) == pRegField->theValueType);
ret_val = ReadRegistryValueData(pRegField->theRootKey,
pRegField->theKeyName,
pRegField->theValueName,
aAllowedTypes,
aTypePtr,
(LPTSTR) aReturnBuffer,
aBufferSizePtr);
}
return ret_val;
}
//////////////////////////////////////////////////////////////////////////_/_//
//////////////////////////////////////////////////////////////////////////_/_//
// DWORD ReadRegistryValueData (HKEY aRootKey,
// LPCTSTR aKeyName,
// LPCTSTR aValueName,
// DWORD aAllowedTypes,
// DWORD * aTypePtr,
// LPTSTR aReturnBuffer,
// DWORD * aBufferSizePtr);
//
// Description:
//
// Additional Information:
//
// Parameters:
//
// HKEY aRootKey :- A handle to an open registry key.
//
// LPCTSTR aKeyName :- The name of the key relative to the open key.
//
// LPCTSTR aValueName :- The name of the value to read
//
// DWORD aAllowedTypes :- See help on ReadRegistryValue.
//
// DWORD * aTypePtr :- A pointer to the buffer that will receive the type.
// If the type is not required then this parameter can be
// set to NULL.
//
// LPBYTE aReturnBuffer :- The buffer into which the data is to be placed.
// This parameter can be NULL in which case no data
// is retrieved.
//
// DWORD * aBufferSizePtr :- This should point to a DWORD that contains the
// size of the buffer. This parameter cannot be
// NULL. When this function returns this value
// will contain the size actually required.
//
// Return Value: This function returns a Win32 error code, ERROS_SUCCESS on
// success.
//
DWORD ReadRegistryValueData (HKEY aRootKey,
LPCTSTR aKeyName,
LPCTSTR aValueName,
DWORD aAllowedTypes,
DWORD * aTypePtr,
LPTSTR aReturnBuffer,
DWORD * aBufferSizePtr) {
DWORD nType = 0;
DWORD errCode = ERROR_INVALID_PARAMETER;
HKEY hOpenKey = NULL;
if ((errCode = RegOpenKeyEx(aRootKey,
aKeyName,
0,
KEY_READ,
&hOpenKey)) == ERROR_SUCCESS) {
_ASSERT(hOpenKey != NULL);
//Key exists and is now open
if ((errCode = RegQueryValueEx(hOpenKey,
aValueName,
NULL,
&nType,
(LPBYTE) aReturnBuffer,
aBufferSizePtr)) == ERROR_SUCCESS) {
if (aTypePtr != NULL) {
*aTypePtr = nType;
}
if ((nType & aAllowedTypes) == 0) {
//The value type in the registry does not match the expected
//type for this function call.
_ASSERT(FALSE);
}
if ((aReturnBuffer != NULL) && (*aBufferSizePtr == 1)) {
//If the registry entry was in fact empty, the buffer needs to
//be 1 character, for the NULL termination.
*aReturnBuffer = TEXT('\0');
}
}
else {
//Something prevented us from reading the value.
//The value may not exist, the buffer size might
//not be large enough. May be using the function to
//read a DWORD value on a registry value that is a
//shorter string.
if (errCode == ERROR_MORE_DATA) {
//There is most likely a mismatch between the type we expect
//and the actual type of this registry value.
_ASSERT(FALSE);
}
}
RegCloseKey(hOpenKey);
}
return(errCode);
}
#ifdef __cplusplus
}
#endif