Windows2003-3790/ds/netapi/svcdlls/upssvc/apcsmart/ulinecon.cxx
2020-09-30 16:53:55 +02:00

318 lines
10 KiB
C++

/*
*
* NOTES:
*
* REVISIONS:
* ker02DEC92: Initial breakout of sensor classes into indiv files
* ker04DEC92: Initial Filling in of Member Functions
* SjA11Dec92: Validate method now returns ErrNO_ERROR rather than VALID.
* pcy17Dec92: Now I do StateSensor::Validate before doing my own.
* pcy16Feb93: No longer need to register for UPS_STATE
* pcy16Feb93: Get transfer cause and ignore if self test
* ajr22Feb93: included utils.h
* jod05Apr93: Added changes for Deep Discharge
* cad09Sep93: Fix for lights test causing events
* pcy09Sep93: Check LINE_AVAILABILITY to make sure line is bad
* pcy10Sep93: Hopefully this works now
* pcy18Sep93: Cleaned up line condition handling
* pcy20Sep93: handle line fails during tests and sim pwr fail
* pcy12Oct93: 2 ABNORMALS to cause a line bad (fixes LF during cal)
* pcy13Apr94: Use automatic variables decrease dynamic mem allocation
* djs16Mar95: changed upsstate.h to sysstate.h
* cgm10Apr96: Destructor should unregister
*/
#define INCL_BASE
#define INCL_DOS
#define INCL_NOPM
#include "cdefine.h"
extern "C" {
#if (C_OS & C_OS2)
#include <os2.h>
#endif
#include <stdlib.h>
#include <stdio.h>
#include <malloc.h>
#include <string.h>
}
#include "ulinecon.h"
#include "comctrl.h"
#include "device.h"
#include "event.h"
#include "utils.h"
#include "sysstate.h"
#include "timerman.h"
#define QCOMMAND 0
#define NINECOMMAND 1
UtilityLineConditionSensor :: UtilityLineConditionSensor(PDevice aParent, PCommController aCommController)
: StateSensor(aParent, aCommController, UTILITY_LINE_CONDITION),
theInformationSource(0), theLineFailCount(0), theUpsState(0)
{
storeState(LINE_GOOD);
theCommController->RegisterEvent(UTILITY_LINE_CONDITION, this);
theDevice->RegisterEvent(BATTERY_CALIBRATION_CONDITION, this);
theDevice->RegisterEvent(SIMULATE_POWER_FAIL, this);
theDevice->RegisterEvent(SELF_TEST_STATE, this);
}
INT UtilityLineConditionSensor::Update(PEvent anEvent)
{
INT err = ErrNO_ERROR;
INT aCode = anEvent->GetCode();
PCHAR aValue = anEvent->GetValue();
switch(aCode)
{
case BATTERY_CALIBRATION_CONDITION:
switch(atoi(aValue))
{
case BATTERY_CALIBRATION_IN_PROGRESS:
theCommController->UnregisterEvent(UTILITY_LINE_CONDITION, this);
theCommController->RegisterEvent(LINE_CONDITION_TEST, this);
theInformationSource = NINECOMMAND;
break;
case NO_BATTERY_CALIBRATION:
case NO_BATTERY_CALIBRATION_IN_PROGRESS:
case BATTERY_CALIBRATION_CANCELLED:
if (theInformationSource == NINECOMMAND)
{
theCommController->UnregisterEvent(LINE_CONDITION_TEST, this);
theCommController->RegisterEvent(UTILITY_LINE_CONDITION, this);
theInformationSource = QCOMMAND;
}
break;
}
break;
case LINE_CONDITION_TEST:
//
// Convert the event to a UTILITY_LINE_CONDITION event and
// send it to myself
//
switch(atoi(aValue))
{
case ABNORMAL_CONDITION:
{
//
// Only after the second successive abnormal condition should
// we bother with this. This is so we ignore ABNORMAL_COND
// events at the start of deep discharge
//
if(theLineFailCount > 0)
{
CHAR lbtmp[31];
sprintf(lbtmp, "%d", LINE_BAD);
if ((err = StateSensor::Validate(UTILITY_LINE_CONDITION,
lbtmp)) == ErrNO_ERROR)
{
Event evt(UTILITY_LINE_CONDITION, LINE_BAD);
storeValue(lbtmp);
UpdateObj::Update(&evt);
}
}
else
{
theLineFailCount++;
}
}
break;
case NO_ABNORMAL_CONDITION:
{
theLineFailCount = 0;
CHAR lbtmp[31];
sprintf(lbtmp, "%d", LINE_GOOD);
if ((err = StateSensor::Validate(UTILITY_LINE_CONDITION,
lbtmp)) == ErrNO_ERROR)
{
Event evt(UTILITY_LINE_CONDITION, LINE_GOOD);
storeValue(lbtmp);
UpdateObj::Update(&evt);
}
}
break;
} // switch for case LINE_CONDITION_TEST
break;
case UTILITY_LINE_CONDITION:
if ((err = Validate(aCode, aValue)) == ErrNO_ERROR)
{
storeValue(aValue);
UpdateObj::Update(anEvent);
}
break;
case SIMULATE_POWER_FAIL:
switch(atoi(aValue))
{
case SIMULATE_POWER_FAIL:
SET_BIT(theUpsState, SIMULATE_POWER_FAIL_BIT);
break;
case SIMULATE_POWER_FAIL_OVER:
CLEAR_BIT(theUpsState, SIMULATE_POWER_FAIL_BIT);
break;
}
break;
case SELF_TEST_STATE:
switch(atoi(aValue))
{
case SELF_TEST_IN_PROGRESS:
SET_BIT(theUpsState, SELF_TEST_BIT);
break;
case NO_SELF_TEST_IN_PROGRESS:
CLEAR_BIT(theUpsState, SELF_TEST_BIT);
break;
}
break;
}
return err;
}
INT UtilityLineConditionSensor::Validate(INT aCode, const PCHAR aValue)
{
INT err = ErrNO_ERROR;
if (aCode != theSensorCode)
{
err = ErrINVALID_CODE;
}
else if(IS_STATE(UPS_STATE_IN_SELF_TEST))
{
err = ErrTEST_IN_PROGRESS;
}
else
{
switch(atoi(aValue))
{
case LINE_GOOD:
{
theLineFailCount = 0;
//
// Make sure line is good to handle UPSLinkism
// that doesnt change tranfer cause when line
// fails during self test or sim pwrfail
//
CHAR line_state[32];
theCommController->Get(LINE_CONDITION_TEST,
line_state);
if(atoi(line_state) == ABNORMAL_CONDITION)
{
err = ErrINVALID_VALUE;
}
break;
}
case LINE_BAD:
//
// Dont do this if we're a BackUPS
//
if (theDevice->IsA() != BACKUPS)
{
//
// If we're in DeepDischarge ignore LINE_BAD events
//
if(theInformationSource != NINECOMMAND)
{
//
// If we're simulating a power fail
// don't try to stop the event
//
if(!(IS_STATE(UPS_STATE_SIMULATED_POWER_FAIL)))
{
CHAR transfer_cause[32];
CHAR self_test[32];
CHAR line_state[32];
//
// If transfer is due to a self test double check
// to see if line is bad
//
theCommController->Get(TRANSFER_CAUSE, transfer_cause);
_itoa(SELF_TEST_TRANSFER, self_test, 10);
if (strcmp(transfer_cause, self_test) == 0)
{
//
// Make sure line is bad to handle UPSLinkism
// that doesnt change tranfer cause when line
// fails during self test or sim pwrfail
//
theCommController->Get(LINE_CONDITION_TEST, line_state);
if(atoi(line_state) == NO_ABNORMAL_CONDITION)
{
err = ErrINVALID_VALUE;
}
else
{
//
// Wait a while to handle UPS-Linkism for
// the 9 command that returns 00 at the start
// of a self test
//
_theTimerManager->Wait(3000L);
theCommController->Get(LINE_CONDITION_TEST, line_state);
if(atoi(line_state) == NO_ABNORMAL_CONDITION)
{
err = ErrINVALID_VALUE;
}
}
}
//
// Same for Lights Test
//
CHAR ups_status[32];
theDevice->Get(UPS_STATE, ups_status);
if (atoi(ups_status) & UPS_STATE_IN_LIGHTS_TEST)
{
err = ErrINVALID_VALUE;
}
}
}
} // if (!= BACKUPS)
break;
default:
err = ErrINVALID_VALUE;
break;
}
if (err == ErrNO_ERROR)
{
err = StateSensor::Validate(aCode, aValue);
}
}
return err;
}
UtilityLineConditionSensor :: ~UtilityLineConditionSensor()
{
if (theCommController)
theCommController->UnregisterEvent(UTILITY_LINE_CONDITION, this);
if (theDevice) {
theDevice->UnregisterEvent(BATTERY_CALIBRATION_CONDITION, this);
theDevice->UnregisterEvent(SIMULATE_POWER_FAIL, this);
theDevice->UnregisterEvent(SELF_TEST_STATE, this);
}
}