494 lines
12 KiB
C
494 lines
12 KiB
C
/* Copyright 1999 American Power Conversion, All Rights Reserved
|
|
*
|
|
* Description:
|
|
* Implements the UPS to the service - it does this by
|
|
* either loading a UPS driver or by using the default
|
|
* Generic UPS interface (simple signalling)
|
|
*
|
|
* Revision History:
|
|
* mholly 19Apr1999 initial revision.
|
|
* dsmith 29Apr1999 defaulted comm status to OK
|
|
* mholly 12May1999 DLL's UPSInit no longer takes the comm port param
|
|
* sberard 17May1999 added a delay to the UPSTurnOffFunction
|
|
*
|
|
*/
|
|
|
|
#include <windows.h>
|
|
#include <tchar.h>
|
|
|
|
#include "driver.h"
|
|
#include "upsreg.h"
|
|
#include "gnrcups.h"
|
|
|
|
|
|
//
|
|
// typedefs of function pointers to aid in
|
|
// accessing functions from driver DLLs
|
|
//
|
|
typedef DWORD (*LPFUNCGETUPSSTATE)(void);
|
|
typedef void (*LPFUNCWAITFORSTATECHANGE)(DWORD, DWORD);
|
|
typedef void (*LPFUNCCANCELWAIT)(void);
|
|
typedef DWORD (*LPFUNCINIT)(void);
|
|
typedef void (*LPFUNCSTOP)(void);
|
|
typedef void (*LPFUNCTURNUPSOFF)(DWORD);
|
|
|
|
|
|
//
|
|
// UPSDRIVERINTERFACE
|
|
//
|
|
// this struct is used to gather all the driver
|
|
// interface data together in a single place, this
|
|
// struct is used to dispatch function calls to
|
|
// either a loaded driver dll, or to the Generic
|
|
// UPS interface functions
|
|
//
|
|
struct UPSDRIVERINTERFACE
|
|
{
|
|
LPFUNCINIT Init;
|
|
LPFUNCSTOP Stop;
|
|
LPFUNCGETUPSSTATE GetUPSState;
|
|
LPFUNCWAITFORSTATECHANGE WaitForStateChange;
|
|
LPFUNCCANCELWAIT CancelWait;
|
|
LPFUNCTURNUPSOFF TurnUPSOff;
|
|
|
|
HINSTANCE hDll;
|
|
};
|
|
|
|
|
|
//
|
|
// private functions used to implement the interface
|
|
//
|
|
static DWORD initializeGenericInterface(struct UPSDRIVERINTERFACE*);
|
|
static DWORD initializeDriverInterface(struct UPSDRIVERINTERFACE*,HINSTANCE);
|
|
static DWORD loadUPSMiniDriver(struct UPSDRIVERINTERFACE *);
|
|
static void unloadUPSMiniDriver(struct UPSDRIVERINTERFACE *);
|
|
static void clearStatusRegistryEntries(void);
|
|
|
|
|
|
//
|
|
// _UpsInterface
|
|
//
|
|
// This is a file-scope variable that is used by all
|
|
// the functions to get access to the actual driver
|
|
//
|
|
static struct UPSDRIVERINTERFACE _UpsInterface;
|
|
|
|
|
|
/**
|
|
* UPSInit
|
|
*
|
|
* Description:
|
|
*
|
|
* The UPSInit function must be called before any
|
|
* other function in this file
|
|
*
|
|
* Parameters:
|
|
* None
|
|
*
|
|
* Returns:
|
|
* UPS_INITOK: Initalization was successful
|
|
* UPS_INITNOSUCHDRIVER: The configured driver DLL can't be opened
|
|
* UPS_INITBADINTERFACE: The configured driver DLL doesn't support
|
|
* the UPS driver interface
|
|
* UPS_INITREGISTRYERROR: The 'Options' registry value is corrupt
|
|
* UPS_INITCOMMOPENERROR: The comm port could not be opened
|
|
* UPS_INITCOMMSETUPERROR: The comm port could not be configured
|
|
* UPS_INITUNKNOWNERROR: Undefined error has occurred
|
|
*
|
|
*/
|
|
DWORD UPSInit(void)
|
|
{
|
|
DWORD init_err = UPS_INITOK;
|
|
|
|
//
|
|
// clear out any old status data
|
|
//
|
|
clearStatusRegistryEntries();
|
|
|
|
|
|
if (UPS_INITOK == init_err) {
|
|
//
|
|
// either load a configured driver DLL or
|
|
// use the Generic UPS interface if no driver
|
|
// is specified
|
|
//
|
|
init_err = loadUPSMiniDriver(&_UpsInterface);
|
|
}
|
|
|
|
if ((UPS_INITOK == init_err) && (_UpsInterface.Init)) {
|
|
//
|
|
// tell the UPS interface to initialize itself
|
|
//
|
|
init_err = _UpsInterface.Init();
|
|
}
|
|
return init_err;
|
|
}
|
|
|
|
|
|
/**
|
|
* UPSStop
|
|
*
|
|
* Description:
|
|
* After a call to UPSStop, only the UPSInit
|
|
* function is valid. This call will unload the
|
|
* UPS driver interface and stop monitoring of the
|
|
* UPS system
|
|
*
|
|
* Parameters:
|
|
* None
|
|
*
|
|
* Returns:
|
|
* None
|
|
*
|
|
*/
|
|
void UPSStop(void)
|
|
{
|
|
if (_UpsInterface.Stop) {
|
|
_UpsInterface.Stop();
|
|
}
|
|
unloadUPSMiniDriver(&_UpsInterface);
|
|
}
|
|
|
|
|
|
/**
|
|
* UPSWaitForStateChange
|
|
*
|
|
* Description:
|
|
* Blocks until the state of the UPS differs
|
|
* from the value passed in via aState or
|
|
* anInterval milliseconds has expired. If
|
|
* anInterval has a value of INFINITE this
|
|
* function will never timeout
|
|
*
|
|
* Parameters:
|
|
* aState: defines the state to wait for a change from,
|
|
* possible values:
|
|
* UPS_ONLINE
|
|
* UPS_ONBATTERY
|
|
* UPS_LOWBATTERY
|
|
* UPS_NOCOMM
|
|
*
|
|
* anInterval: timeout in milliseconds, or INFINITE for
|
|
* no timeout interval
|
|
*
|
|
* Returns:
|
|
* None
|
|
*
|
|
*/
|
|
void UPSWaitForStateChange(DWORD aCurrentState, DWORD anInterval)
|
|
{
|
|
if (_UpsInterface.WaitForStateChange) {
|
|
_UpsInterface.WaitForStateChange(aCurrentState, anInterval);
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* UPSGetState
|
|
*
|
|
* Description:
|
|
* returns the current state of the UPS
|
|
*
|
|
* Parameters:
|
|
* None
|
|
*
|
|
* Returns:
|
|
* possible values:
|
|
* UPS_ONLINE
|
|
* UPS_ONBATTERY
|
|
* UPS_LOWBATTERY
|
|
* UPS_NOCOMM
|
|
*
|
|
*/
|
|
DWORD UPSGetState(void)
|
|
{
|
|
DWORD err = ERROR_INVALID_ACCESS;
|
|
|
|
if (_UpsInterface.GetUPSState) {
|
|
err = _UpsInterface.GetUPSState();
|
|
}
|
|
return err;
|
|
}
|
|
|
|
|
|
/**
|
|
* UPSCancelWait
|
|
*
|
|
* Description:
|
|
* interrupts pending calls to UPSWaitForStateChange
|
|
* without regard to timout or state change
|
|
*
|
|
* Parameters:
|
|
* None
|
|
*
|
|
* Returns:
|
|
* None
|
|
*
|
|
*/
|
|
void UPSCancelWait(void)
|
|
{
|
|
if (_UpsInterface.CancelWait) {
|
|
_UpsInterface.CancelWait();
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* UPSTurnOff
|
|
*
|
|
* Description:
|
|
* Attempts to turn off the outlets on the UPS
|
|
* after the specified delay. This call must
|
|
* return immediately. Any work, such as a timer,
|
|
* must be performed on a another thread.
|
|
*
|
|
* Parameters:
|
|
* aTurnOffDelay: the minimum amount of time to wait before
|
|
* turning off the outlets on the UPS
|
|
*
|
|
* Returns:
|
|
* None
|
|
*
|
|
*/
|
|
void UPSTurnOff(DWORD aTurnOffDelay)
|
|
{
|
|
if (_UpsInterface.TurnUPSOff) {
|
|
_UpsInterface.TurnUPSOff(aTurnOffDelay);
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* initializeGenericInterface
|
|
*
|
|
* Description:
|
|
* Fills in the UPSDRIVERINTERFACE struct with the functions
|
|
* of the Generic UPS interface
|
|
*
|
|
* Parameters:
|
|
* anInterface: the UPSDRIVERINTERFACE structure to
|
|
* fill in - the struct must have been
|
|
* allocated prior to calling this function
|
|
*
|
|
* Returns:
|
|
* ERROR_SUCCESS
|
|
*
|
|
*/
|
|
DWORD initializeGenericInterface(struct UPSDRIVERINTERFACE* anInterface)
|
|
{
|
|
anInterface->hDll = NULL;
|
|
anInterface->Init = GenericUPSInit;
|
|
anInterface->Stop = GenericUPSStop;
|
|
anInterface->GetUPSState = GenericUPSGetState;
|
|
anInterface->WaitForStateChange = GenericUPSWaitForStateChange;
|
|
anInterface->CancelWait = GenericUPSCancelWait;
|
|
anInterface->TurnUPSOff = GenericUPSTurnOff;
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
|
|
/**
|
|
* initializeDriverInterface
|
|
*
|
|
* Description:
|
|
* Fills in the UPSDRIVERINTERFACE struct with the functions
|
|
* of the loaded UPS driver DLL
|
|
*
|
|
* Parameters:
|
|
* anInterface: the UPSDRIVERINTERFACE structure to
|
|
* fill in - the struct must have been
|
|
* allocated prior to calling this function
|
|
* hDll: a handle to a UPS driver DLL
|
|
*
|
|
* Returns:
|
|
* ERROR_SUCCESS: DLL handle was valid, and the DLL supports the
|
|
* UPS driver interface
|
|
*
|
|
* !ERROR_SUCCESS: either the DLL handle is invalid - or the DLL
|
|
* does not fully support the UPS driver interface
|
|
*
|
|
*/
|
|
DWORD initializeDriverInterface(struct UPSDRIVERINTERFACE * anInterface,
|
|
HINSTANCE hDll)
|
|
{
|
|
DWORD err = ERROR_SUCCESS;
|
|
|
|
anInterface->hDll = hDll;
|
|
|
|
anInterface->Init =
|
|
(LPFUNCINIT)GetProcAddress(hDll, "UPSInit");
|
|
|
|
if (!anInterface->Init) {
|
|
err = GetLastError();
|
|
goto init_driver_end;
|
|
}
|
|
anInterface->Stop =
|
|
(LPFUNCSTOP)GetProcAddress(hDll, "UPSStop");
|
|
|
|
if (!anInterface->Stop) {
|
|
err = GetLastError();
|
|
goto init_driver_end;
|
|
}
|
|
anInterface->GetUPSState =
|
|
(LPFUNCGETUPSSTATE)GetProcAddress(hDll, "UPSGetState");
|
|
|
|
if (!anInterface->GetUPSState) {
|
|
err = GetLastError();
|
|
goto init_driver_end;
|
|
}
|
|
anInterface->WaitForStateChange =
|
|
(LPFUNCWAITFORSTATECHANGE)GetProcAddress(hDll,
|
|
"UPSWaitForStateChange");
|
|
|
|
if (!anInterface->WaitForStateChange) {
|
|
err = GetLastError();
|
|
goto init_driver_end;
|
|
}
|
|
anInterface->CancelWait =
|
|
(LPFUNCCANCELWAIT)GetProcAddress(hDll, "UPSCancelWait");
|
|
|
|
if (!anInterface->CancelWait) {
|
|
err = GetLastError();
|
|
goto init_driver_end;
|
|
}
|
|
anInterface->TurnUPSOff =
|
|
(LPFUNCTURNUPSOFF)GetProcAddress(hDll, "UPSTurnOff");
|
|
|
|
if (!anInterface->TurnUPSOff) {
|
|
err = GetLastError();
|
|
goto init_driver_end;
|
|
}
|
|
|
|
init_driver_end:
|
|
return err;
|
|
}
|
|
|
|
|
|
/**
|
|
* loadUPSMiniDriver
|
|
*
|
|
* Description:
|
|
* Fills in the UPSDRIVERINTERFACE struct with the functions
|
|
* of a UPS interface, either a configured driver DLL or the
|
|
* Generic UPS interface. If the configured DLL can't be
|
|
* opened or does not support the interface then an error is
|
|
* returned and the UPSDRIVERINTERFACE will not be initialized
|
|
*
|
|
* Parameters:
|
|
* anInterface: the UPSDRIVERINTERFACE structure to
|
|
* fill in - the struct must have been
|
|
* allocated prior to calling this function
|
|
*
|
|
* Returns:
|
|
* UPS_INITOK: driver interface is intialized
|
|
*
|
|
* UPS_INITNOSUCHDRIVER: the configured driver DLL can't be opened
|
|
* UPS_INITBADINTERFACE: the configured driver DLL does not
|
|
* fully support the UPS driver interface
|
|
*
|
|
*/
|
|
DWORD loadUPSMiniDriver(struct UPSDRIVERINTERFACE * aDriverInterface)
|
|
{
|
|
DWORD load_err = UPS_INITOK;
|
|
DWORD err = ERROR_SUCCESS;
|
|
TCHAR driver_name[MAX_PATH];
|
|
HINSTANCE hDll = NULL;
|
|
|
|
err = GetUPSConfigServiceDLL(driver_name, MAX_PATH);
|
|
|
|
//
|
|
// check to see if there is a key, and that its
|
|
// value is valid (a valid key has a value that
|
|
// is greater than zero characters long)
|
|
//
|
|
if (ERROR_SUCCESS == err && _tcslen(driver_name)) {
|
|
hDll = LoadLibrary(driver_name);
|
|
}
|
|
else {
|
|
//
|
|
// NO ERROR - simply means we use the
|
|
// internal generic UPS support
|
|
//
|
|
err = initializeGenericInterface(aDriverInterface);
|
|
goto load_end;
|
|
}
|
|
|
|
if (!hDll) {
|
|
//
|
|
// the configured driver could not be opened
|
|
//
|
|
err = GetLastError();
|
|
load_err = UPS_INITNOSUCHDRIVER;
|
|
goto load_end;
|
|
}
|
|
|
|
err = initializeDriverInterface(aDriverInterface, hDll);
|
|
|
|
if (ERROR_SUCCESS != err) {
|
|
load_err = UPS_INITBADINTERFACE;
|
|
goto load_end;
|
|
}
|
|
|
|
load_end:
|
|
return load_err;
|
|
}
|
|
|
|
|
|
/**
|
|
* unloadUPSMiniDriver
|
|
*
|
|
* Description:
|
|
* unloads a driver DLL if one was opened, also clears
|
|
* out the function dispatch pointers
|
|
*
|
|
* Parameters:
|
|
* anInterface: the UPSDRIVERINTERFACE structure to
|
|
* check for DLL info, and to clear
|
|
*
|
|
* Returns:
|
|
* None
|
|
*
|
|
*/
|
|
void unloadUPSMiniDriver(struct UPSDRIVERINTERFACE * aDriverInterface)
|
|
{
|
|
if (aDriverInterface) {
|
|
|
|
if (aDriverInterface->hDll) {
|
|
FreeLibrary(aDriverInterface->hDll);
|
|
aDriverInterface->hDll = NULL;
|
|
}
|
|
aDriverInterface->CancelWait = NULL;
|
|
aDriverInterface->GetUPSState = NULL;
|
|
aDriverInterface->Init = NULL;
|
|
aDriverInterface->Stop = NULL;
|
|
aDriverInterface->TurnUPSOff = NULL;
|
|
aDriverInterface->WaitForStateChange = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* clearStatusRegistryEntries
|
|
*
|
|
* Description:
|
|
* zeros out the registry status entries
|
|
*
|
|
* Parameters:
|
|
* None
|
|
*
|
|
* Returns:
|
|
* None
|
|
*
|
|
*/
|
|
void clearStatusRegistryEntries(void)
|
|
{
|
|
InitUPSStatusBlock();
|
|
SetUPSStatusSerialNum(_TEXT(""));
|
|
SetUPSStatusFirmRev(_TEXT(""));
|
|
SetUPSStatusUtilityStatus(UPS_UTILITYPOWER_UNKNOWN);
|
|
SetUPSStatusRuntime(0);
|
|
SetUPSStatusBatteryStatus(UPS_BATTERYSTATUS_UNKNOWN);
|
|
SetUPSStatusCommStatus(UPS_COMMSTATUS_OK);
|
|
SaveUPSStatusBlock(TRUE);
|
|
}
|