2010-03-26 01:20:59 +02:00
/*
bcmxcp . c - driver for powerware UPS
Total rewrite of bcmxcp . c ( nut ver - 1.4 .3 )
* Copyright ( c ) 2002 , Martin Schroeder *
* emes - at - geomer . de *
* All rights reserved . *
2012-06-01 16:55:19 +03:00
Copyright ( C )
2004 Kjell Claesson < kjell . claesson - at - epost . tidanet . se >
2004 Tore Ø rpetveit < tore - at - orpetveit . net >
2015-04-30 16:53:36 +03:00
2011 - 2015 Arnaud Quette < ArnaudQuette @ Eaton . com >
2010-03-26 01:20:59 +02:00
Thanks to Tore Ø rpetveit < tore - at - orpetveit . net > that sent me the
manuals for bcm / xcp .
And to Fabio Di Niro < fabio . diniro @ email . it > and his metasys module .
It influenced the layout of this driver .
Modified for USB by Wolfgang Ocker < weo @ weo1 . de >
ojw0000 2007 Apr5 Oliver Wilcock - modified to control individual load segments ( outlet .2 . shutdown . return ) on Powerware PW5125 .
2011-09-29 21:14:46 +03:00
Modified to support setvar for outlet . n . delay . start by Rich Wrenn ( RFW ) 9 - 3 - 11.
2015-04-30 16:53:36 +03:00
Modified to support setvar for outlet . n . delay . shutdown by Arnaud Quette , 9 - 12 - 11
2011-09-29 21:14:46 +03:00
2010-03-26 01:20:59 +02:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
TODO List :
2015-04-30 16:53:36 +03:00
Extend the parsing of the Standard ID Block , to read :
Config Block Length : ( High priority )
Give information if config block is
present , and how long it is , if it exist .
If config block exist , read the config block and parse the
' Length of the Extended Limits Configuration Block ' for
extended configuration commands
Statistic map Size : ( Low priority )
May be used to se if there is a Statistic Map .
It holds data on the utility power quality for
the past month and since last reset . Number of
times on battery and how long . Up time and utility
frequency deviation . ( Only larger ups ' es )
Size of Alarm History Log : ( Low priority )
See if it have any alarm history block and enable
command to dump it .
Maximum Supported Command Length : ( Med . to High priority )
Give info about the ups receive buffer size .
Size of Alarm Block : ( Med . to High priority )
Make a smarter handling of the Active alarm ' s if we know the length
of the Active Alarm Block . Don ' t need the long loop to parse the
alarm ' s . Maybe use another way to set up the alarm struct in the
' init_alarm_map ' .
Parse ' Communication Capabilities Block ' ( Low priority )
Get info of the connected ports ID , number of baud rates ,
command and respnse length .
Parse ' Communication Port List Block ' : ( Low priority )
This block gives info about the communication ports . Some ups ' es
have multiple comport ' s , and use one port for eatch load segment .
In this block it is possible to get :
Number of ports . ( In this List )
This Comport id ( Which Comm Port is reporting this block . )
Comport id ( Id for eatch port listed . The first comport ID = 1 )
Baudrate of the listed port .
Serial config .
Port usage :
What this Comm Port is being used for :
0 = Unknown usage , No communication occurring .
1 = Undefined / Unknown communication occurring
2 = Waiting to communicate with a UPS
3 = Communication established with a UPS
4 = Waiting to communicate with software or adapter
5 = Communication established software ( e . g . , LanSafe )
or adapter ( e . g . , ConnectUPS )
6 = Communicating with a Display Device
7 = Multi - drop Serial channel
8 = Communicating with an Outlet Controller
Number of outlets . ( Number of Outlets " assigned to " ( controlled by ) this Comm Port )
Outlet number . ( Each assigned Outlet is listed ( 1 - 64 ) )
' Set outlet parameter command ( 0x97 ) ' to alter the delay
settings or turn the outlet on or off with a delay ( 0 - 32767 seconds )
Rewrite some parts of the driver , to minimise code duplication . ( Like the instant commands )
Implement support for Password Authorization ( XCP spec , § 4.3 .2 )
Complete support for settable variables ( upsh . setvar )
2010-03-26 01:20:59 +02:00
*/
# include "main.h"
2013-11-24 17:00:12 +02:00
# include <math.h> /* For ldexp() */
# include <float.h> /*for FLT_MAX */
2011-09-29 21:14:46 +03:00
# include "nut_stdint.h" /* for uint8_t, uint16_t, uint32_t, ... */
2010-03-26 01:20:59 +02:00
# include "bcmxcp_io.h"
# include "bcmxcp.h"
2013-11-24 17:00:12 +02:00
# define DRIVER_NAME "BCMXCP UPS driver"
2016-07-18 03:11:41 +03:00
# define DRIVER_VERSION "0.31"
2011-09-29 21:14:46 +03:00
2013-11-24 17:00:12 +02:00
# define MAX_NUT_NAME_LENGTH 128
# define NUT_OUTLET_POSITION 7
2010-03-26 01:20:59 +02:00
/* driver description structure */
upsdrv_info_t upsdrv_info = {
2015-04-30 16:53:36 +03:00
DRIVER_NAME ,
DRIVER_VERSION ,
" Martin Schroeder <emes@geomer.de> \n " \
" Kjell Claesson <kjell.claesson@epost.tidanet.se> \n " \
" Tore Ørpetveit <tore@orpetveit.net> \n " \
" Arnaud Quette <ArnaudQuette@Eaton.com> \n " \
" Wolfgang Ocker <weo@weo1.de> \n " \
" Oliver Wilcock \n " \
" Prachi Gandhi <prachisgandhi@eaton.com> \n " \
" Alf Høgemark <alf@i100> \n " \
" Gavrilov Igor " ,
DRV_STABLE ,
{ & comm_upsdrv_info , NULL }
2010-03-26 01:20:59 +02:00
} ;
static int get_word ( const unsigned char * ) ;
static long int get_long ( const unsigned char * ) ;
static float get_float ( const unsigned char * data ) ;
2013-11-24 17:00:12 +02:00
static void init_command_map ( void ) ;
2010-03-26 01:20:59 +02:00
static void init_meter_map ( void ) ;
static void init_alarm_map ( void ) ;
2013-11-24 17:00:12 +02:00
static bool_t init_command ( int size ) ;
2010-03-26 01:20:59 +02:00
static void init_config ( void ) ;
static void init_limit ( void ) ;
2015-04-30 16:53:36 +03:00
static void init_ext_vars ( void ) ;
2013-11-24 17:00:12 +02:00
static void init_topology ( void ) ;
2010-03-26 01:20:59 +02:00
static void init_ups_meter_map ( const unsigned char * map , unsigned char len ) ;
static void init_ups_alarm_map ( const unsigned char * map , unsigned char len ) ;
2013-11-24 17:00:12 +02:00
static bool_t set_alarm_support_in_alarm_map ( const unsigned char * map , const int mapIndex , const int bitmask , const int alarmMapIndex , const int alarmBlockIndex ) ;
2010-03-26 01:20:59 +02:00
static void decode_meter_map_entry ( const unsigned char * entry , const unsigned char format , char * value ) ;
static int init_outlet ( unsigned char len ) ;
2013-11-24 17:00:12 +02:00
static void init_system_test_capabilities ( void ) ;
2010-03-26 01:20:59 +02:00
static int instcmd ( const char * cmdname , const char * extra ) ;
2013-11-24 17:00:12 +02:00
static int setvar ( const char * varname , const char * val ) ;
static int decode_instcmd_exec ( const int res , const unsigned char exec_status , const char * cmdname , const char * success_msg ) ;
2015-04-30 16:53:36 +03:00
static int decode_setvar_exec ( const int res , const unsigned char exec_status , const char * cmdname , const char * success_msg ) ;
2013-11-24 17:00:12 +02:00
static float calculate_ups_load ( const unsigned char * data ) ;
2010-03-26 01:20:59 +02:00
2013-11-24 17:00:12 +02:00
static const char * nut_find_infoval ( info_lkp_t * xcp2info , const double value , const bool_t debug_output_nonexisting ) ;
2010-03-26 01:20:59 +02:00
2011-01-26 11:35:08 +02:00
const char * FreqTol [ 3 ] = { " +/-2% " , " +/-5% " , " +/-7 " } ;
2015-04-30 16:53:36 +03:00
const char * ABMStatus [ 4 ] = { " charging " , " discharging " , " floating " , " resting " } ;
const char * OutletStatus [ 9 ] = { " unknown " , " on/closed " , " off/open " , " on with pending " , " off with pending " , " unknown " , " unknown " , " failed and closed " , " failed and open " } ;
2013-11-24 17:00:12 +02:00
/* Standard Authorization Block */
2010-03-26 01:20:59 +02:00
unsigned char AUTHOR [ 4 ] = { 0xCF , 0x69 , 0xE8 , 0xD5 } ;
int nphases = 0 ;
int outlet_block_len = 0 ;
2011-01-26 11:35:08 +02:00
const char * cpu_name [ 5 ] = { " Cont: " , " Inve: " , " Rect: " , " Netw: " , " Disp: " } ;
2015-04-30 16:53:36 +03:00
const char * horn_stat [ 3 ] = { " disabled " , " enabled " , " muted " } ;
2012-06-01 16:55:19 +03:00
/* Battery test results */
info_lkp_t batt_test_info [ ] = {
{ 0 , " No test initiated " , NULL } ,
{ 1 , " In progress " , NULL } ,
{ 2 , " Done and passed " , NULL } ,
{ 3 , " Aborted " , NULL } ,
{ 4 , " Done and error " , NULL } ,
{ 5 , " Test scheduled " , NULL } ,
/* Not sure about the meaning of the below ones! */
{ 6 , NULL , NULL } , /* The string was present but it has now been removed */
{ 7 , NULL , NULL } , /* The string was not installed at the last power up */
{ 0 , NULL , NULL }
} ;
2013-11-24 17:00:12 +02:00
/* Topology map results */
info_lkp_t topology_info [ ] = {
{ BCMXCP_TOPOLOGY_OFFLINE_SWITCHER_1P , " Off-line switcher, Single Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_LINEINT_UPS_1P , " Line-Interactive UPS, Single Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_LINEINT_UPS_2P , " Line-Interactive UPS, Two Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_LINEINT_UPS_3P , " Line-Interactive UPS, Three Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_DUAL_AC_ONLINE_UPS_1P , " Dual AC Input, On-Line UPS, Single Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_DUAL_AC_ONLINE_UPS_2P , " Dual AC Input, On-Line UPS, Two Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_DUAL_AC_ONLINE_UPS_3P , " Dual AC Input, On-Line UPS, Three Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_ONLINE_UPS_1P , " On-Line UPS, Single Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_ONLINE_UPS_2P , " On-Line UPS, Two Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_ONLINE_UPS_3P , " On-Line UPS, Three Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_PARA_REDUND_ONLINE_UPS_1P , " Parallel Redundant On-Line UPS, Single Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_PARA_REDUND_ONLINE_UPS_2P , " Parallel Redundant On-Line UPS, Two Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_PARA_REDUND_ONLINE_UPS_3P , " Parallel Redundant On-Line UPS, Three Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_PARA_CAPACITY_ONLINE_UPS_1P , " Parallel for Capacity On-Line UPS, Single Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_PARA_CAPACITY_ONLINE_UPS_2P , " Parallel for Capacity On-Line UPS, Two Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_PARA_CAPACITY_ONLINE_UPS_3P , " Parallel for Capacity On-Line UPS, Three Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_SYSTEM_BYPASS_MODULE_3P , " System Bypass Module, Three Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_HOT_TIE_CABINET_3P , " Hot-Tie Cabinet, Three Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_OUTLET_CONTROLLER_1P , " Outlet Controller, Single Phase " , NULL } ,
{ BCMXCP_TOPOLOGY_DUAL_AC_STATIC_SWITCH_3P , " Dual AC Input Static Switch Module, 3 Phase " , NULL } ,
{ 0 , NULL , NULL }
} ;
/* Command map results */
info_lkp_t command_map_info [ ] = {
{ PW_INIT_BAT_TEST , " test.battery.start " , NULL } ,
{ PW_LOAD_OFF_RESTART , " shutdown.return " , NULL } ,
{ PW_UPS_OFF , " shutdown.stayoff " , NULL } ,
2015-04-30 16:53:36 +03:00
{ PW_UPS_ON , " load.on " , NULL } ,
{ PW_GO_TO_BYPASS , " bypass.start " , NULL } ,
2013-11-24 17:00:12 +02:00
{ 0 , NULL , NULL }
} ;
/* System test capabilities results */
info_lkp_t system_test_info [ ] = {
{ PW_SYS_TEST_GENERAL , " test.system.start " , NULL } ,
/* { PW_SYS_TEST_SCHEDULE_BATTERY_COMMISSION, "test.battery.start.delayed", NULL }, */
/* { PW_SYS_TEST_ALTERNATE_AC_INPUT, "test.alternate_acinput.start", NULL }, */
{ PW_SYS_TEST_FLASH_LIGHTS , " test.panel.start " , NULL } ,
{ 0 , NULL , NULL }
} ;
2012-06-01 16:55:19 +03:00
/* allocate storage for shared variables (extern in bcmxcp.h) */
2013-11-24 17:00:12 +02:00
BCMXCP_COMMAND_MAP_ENTRY_t
bcmxcp_command_map [ BCMXCP_COMMAND_MAP_MAX ] ;
2012-06-01 16:55:19 +03:00
BCMXCP_METER_MAP_ENTRY_t
bcmxcp_meter_map [ BCMXCP_METER_MAP_MAX ] ;
BCMXCP_ALARM_MAP_ENTRY_t
bcmxcp_alarm_map [ BCMXCP_ALARM_MAP_MAX ] ;
BCMXCP_STATUS_t
bcmxcp_status ;
2010-03-26 01:20:59 +02:00
/* get_word function from nut driver metasys.c */
2013-11-24 17:00:12 +02:00
int get_word ( const unsigned char * buffer ) /* return an integer reading a word in the supplied buffer */
2010-03-26 01:20:59 +02:00
{
unsigned char a , b ;
int result ;
a = buffer [ 0 ] ;
b = buffer [ 1 ] ;
result = b * 256 + a ;
return result ;
}
2013-11-24 17:00:12 +02:00
/* get_long function from nut driver metasys.c for meter readings*/
long int get_long ( const unsigned char * buffer ) /* return a long integer reading 4 bytes in the supplied buffer.*/
2010-03-26 01:20:59 +02:00
{
unsigned char a , b , c , d ;
long int result ;
a = buffer [ 0 ] ;
b = buffer [ 1 ] ;
c = buffer [ 2 ] ;
d = buffer [ 3 ] ;
result = ( 256 * 256 * 256 * d ) + ( 256 * 256 * c ) + ( 256 * b ) + a ;
return result ;
}
/* get_float funktion for convering IEEE-754 to float */
float get_float ( const unsigned char * data )
{
int s , e ;
unsigned long src ;
long f ;
src = ( ( unsigned long ) data [ 3 ] < < 24 ) |
( ( unsigned long ) data [ 2 ] < < 16 ) |
( ( unsigned long ) data [ 1 ] < < 8 ) |
( ( unsigned long ) data [ 0 ] ) ;
s = ( src & 0x80000000UL ) > > 31 ;
e = ( src & 0x7F800000UL ) > > 23 ;
f = ( src & 0x007FFFFFUL ) ;
if ( e = = 255 & & f ! = 0 )
{
/* NaN (Not a Number) */
return FLT_MAX ;
}
if ( e = = 255 & & f = = 0 & & s = = 1 )
{
/* Negative infinity */
return - FLT_MAX ;
}
if ( e = = 255 & & f = = 0 & & s = = 0 )
{
/* Positive infinity */
return FLT_MAX ;
}
if ( e > 0 & & e < 255 )
{
/* Normal number */
f + = 0x00800000UL ;
if ( s ) f = - f ;
return ldexp ( f , e - 150 ) ;
}
if ( e = = 0 & & f ! = 0 )
{
/* Denormal number */
if ( s ) f = - f ;
return ldexp ( f , - 149 ) ;
}
if ( e = = 0 & & f = = 0 & & ( s = = 1 | | s = = 0 ) )
{
/* Zero */
return 0 ;
}
/* Never happens */
upslogx ( LOG_ERR , " s = %d, e = %d, f = %lu \n " , s , e , f ) ;
return 0 ;
}
/* lightweight function to calculate the 8-bit
* two ' s complement checksum of buf , using XCP data length ( including header )
* the result must be 0 for the sequence data to be valid */
int checksum_test ( const unsigned char * buf )
{
unsigned char checksum = 0 ;
int i , length ;
/* buf[2] is the length of the XCP frame ; add 5 for the header */
length = ( int ) ( buf [ 2 ] ) + 5 ;
for ( i = 0 ; i < length ; i + + ) {
checksum + = buf [ i ] ;
}
/* Compute the 8-bit, Two's Complement checksum now and return it */
checksum = ( ( 0x100 - checksum ) & 0xFF ) ;
return ( checksum = = 0 ) ;
}
unsigned char calc_checksum ( const unsigned char * buf )
{
unsigned char c ;
int i ;
c = 0 ;
for ( i = 0 ; i < 2 + buf [ 1 ] ; i + + )
c - = buf [ i ] ;
return c ;
}
2013-11-24 17:00:12 +02:00
void init_command_map ( )
{
int i = 0 ;
/* Clean entire map */
memset ( & bcmxcp_command_map , 0 , sizeof ( BCMXCP_COMMAND_MAP_ENTRY_t ) * BCMXCP_COMMAND_MAP_MAX ) ;
/* Set all command descriptions */
bcmxcp_command_map [ PW_ID_BLOCK_REQ ] . command_desc = " PW_ID_BLOCK_REQ " ;
bcmxcp_command_map [ PW_EVENT_HISTORY_LOG_REQ ] . command_desc = " PW_EVENT_HISTORY_LOG_REQ " ;
bcmxcp_command_map [ PW_STATUS_REQ ] . command_desc = " PW_STATUS_REQ " ;
bcmxcp_command_map [ PW_METER_BLOCK_REQ ] . command_desc = " PW_METER_BLOCK_REQ " ;
bcmxcp_command_map [ PW_CUR_ALARM_REQ ] . command_desc = " PW_CUR_ALARM_REQ " ;
bcmxcp_command_map [ PW_CONFIG_BLOCK_REQ ] . command_desc = " PW_CONFIG_BLOCK_REQ " ;
bcmxcp_command_map [ PW_UTILITY_STATISTICS_BLOCK_REQ ] . command_desc = " PW_UTILITY_STATISTICS_BLOCK_REQ " ;
bcmxcp_command_map [ PW_WAVEFORM_BLOCK_REQ ] . command_desc = " PW_WAVEFORM_BLOCK_REQ " ;
bcmxcp_command_map [ PW_BATTERY_REQ ] . command_desc = " PW_BATTERY_REQ " ;
bcmxcp_command_map [ PW_LIMIT_BLOCK_REQ ] . command_desc = " PW_LIMIT_BLOCK_REQ " ;
bcmxcp_command_map [ PW_TEST_RESULT_REQ ] . command_desc = " PW_TEST_RESULT_REQ " ;
bcmxcp_command_map [ PW_COMMAND_LIST_REQ ] . command_desc = " PW_COMMAND_LIST_REQ " ;
bcmxcp_command_map [ PW_OUT_MON_BLOCK_REQ ] . command_desc = " PW_OUT_MON_BLOCK_REQ " ;
bcmxcp_command_map [ PW_COM_CAP_REQ ] . command_desc = " PW_COM_CAP_REQ " ;
bcmxcp_command_map [ PW_UPS_TOP_DATA_REQ ] . command_desc = " PW_UPS_TOP_DATA_REQ " ;
bcmxcp_command_map [ PW_COM_PORT_LIST_BLOCK_REQ ] . command_desc = " PW_COM_PORT_LIST_BLOCK_REQ " ;
bcmxcp_command_map [ PW_REQUEST_SCRATCHPAD_DATA_REQ ] . command_desc = " PW_REQUEST_SCRATCHPAD_DATA_REQ " ;
bcmxcp_command_map [ PW_GO_TO_BYPASS ] . command_desc = " PW_GO_TO_BYPASS " ;
bcmxcp_command_map [ PW_UPS_ON ] . command_desc = " PW_UPS_ON " ;
bcmxcp_command_map [ PW_LOAD_OFF_RESTART ] . command_desc = " PW_LOAD_OFF_RESTART " ;
bcmxcp_command_map [ PW_UPS_OFF ] . command_desc = " PW_UPS_OFF " ;
bcmxcp_command_map [ PW_DECREMENT_OUTPUT_VOLTAGE ] . command_desc = " PW_DECREMENT_OUTPUT_VOLTAGE " ;
bcmxcp_command_map [ PW_INCREMENT_OUTPUT_VOLTAGE ] . command_desc = " PW_INCREMENT_OUTPUT_VOLTAGE " ;
bcmxcp_command_map [ PW_SET_TIME_AND_DATE ] . command_desc = " PW_SET_TIME_AND_DATE " ;
bcmxcp_command_map [ PW_UPS_ON_TIME ] . command_desc = " PW_UPS_ON_TIME " ;
bcmxcp_command_map [ PW_UPS_ON_AT_TIME ] . command_desc = " PW_UPS_ON_AT_TIME " ;
bcmxcp_command_map [ PW_UPS_OFF_TIME ] . command_desc = " PW_UPS_OFF_TIME " ;
bcmxcp_command_map [ PW_UPS_OFF_AT_TIME ] . command_desc = " PW_UPS_OFF_AT_TIME " ;
bcmxcp_command_map [ PW_SET_CONF_COMMAND ] . command_desc = " PW_SET_CONF_COMMAND " ;
bcmxcp_command_map [ PW_SET_OUTLET_COMMAND ] . command_desc = " PW_SET_OUTLET_COMMAND " ;
bcmxcp_command_map [ PW_SET_COM_COMMAND ] . command_desc = " PW_SET_COM_COMMAND " ;
bcmxcp_command_map [ PW_SET_SCRATHPAD_SECTOR ] . command_desc = " PW_SET_SCRATHPAD_SECTOR " ;
bcmxcp_command_map [ PW_SET_POWER_STRATEGY ] . command_desc = " PW_SET_POWER_STRATEGY " ;
bcmxcp_command_map [ PW_SET_REQ_ONLY_MODE ] . command_desc = " PW_SET_REQ_ONLY_MODE " ;
bcmxcp_command_map [ PW_SET_UNREQUESTED_MODE ] . command_desc = " PW_SET_UNREQUESTED_MODE " ;
bcmxcp_command_map [ PW_INIT_BAT_TEST ] . command_desc = " PW_INIT_BAT_TEST " ;
bcmxcp_command_map [ PW_INIT_SYS_TEST ] . command_desc = " PW_INIT_SYS_TEST " ;
bcmxcp_command_map [ PW_SELECT_SUBMODULE ] . command_desc = " PW_SELECT_SUBMODULE " ;
bcmxcp_command_map [ PW_AUTHORIZATION_CODE ] . command_desc = " PW_AUTHORIZATION_CODE " ;
for ( i = 0 ; i < BCMXCP_COMMAND_MAP_MAX ; i + + ) {
bcmxcp_command_map [ i ] . command_byte = 0 ;
}
}
2010-03-26 01:20:59 +02:00
void init_meter_map ( )
{
/* Clean entire map */
memset ( & bcmxcp_meter_map , 0 , sizeof ( BCMXCP_METER_MAP_ENTRY_t ) * BCMXCP_METER_MAP_MAX ) ;
/* Set all corresponding mappings NUT <-> BCM/XCP */
2013-11-24 17:00:12 +02:00
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VOLTS_AB ] . nut_entity = " output.L1-L2.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VOLTS_BC ] . nut_entity = " output.L2-L3.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VOLTS_CA ] . nut_entity = " output.L3-L1.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_VOLTS_AB ] . nut_entity = " input.L1-L2.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_VOLTS_BC ] . nut_entity = " input.L2-L3.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_VOLTS_CA ] . nut_entity = " input.L3-L1.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_CURRENT_PHASE_B ] . nut_entity = " input.L2.current " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_CURRENT_PHASE_C ] . nut_entity = " input.L3.current " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_WATTS ] . nut_entity = " input.realpower " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VA ] . nut_entity = " ups.power " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_VA ] . nut_entity = " input.power " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_POWER_FACTOR ] . nut_entity = " output.powerfactor " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_POWER_FACTOR ] . nut_entity = " input.powerfactor " ;
if ( nphases = = 1 ) {
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_CURRENT_PHASE_A ] . nut_entity = " input.current " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_PERCENT_LOAD_PHASE_A ] . nut_entity = " ups.load " ; /* TODO: Decide on corresponding three-phase variable mapping. */
bcmxcp_meter_map [ BCMXCP_METER_MAP_BYPASS_VOLTS_PHASE_A ] . nut_entity = " input.bypass.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_VOLTS_PHASE_A ] . nut_entity = " input.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_A ] . nut_entity = " output.current " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_A_BAR_CHART ] . nut_entity = " output.current.nominal " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VOLTS_A ] . nut_entity = " output.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_WATTS ] . nut_entity = " ups.realpower " ;
} else {
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_CURRENT_PHASE_A ] . nut_entity = " input.L1.current " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_PERCENT_LOAD_PHASE_A ] . nut_entity = " output.L1.power.percent " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_PERCENT_LOAD_PHASE_B ] . nut_entity = " output.L2.power.percent " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_PERCENT_LOAD_PHASE_C ] . nut_entity = " output.L3.power.percent " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VA_PHASE_A ] . nut_entity = " output.L1.power " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VA_PHASE_B ] . nut_entity = " output.L2.power " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VA_PHASE_C ] . nut_entity = " output.L3.power " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_BYPASS_VOLTS_PHASE_A ] . nut_entity = " input.bypass.L1-N.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_BYPASS_VOLTS_PHASE_B ] . nut_entity = " input.bypass.L2-N.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_BYPASS_VOLTS_PHASE_C ] . nut_entity = " input.bypass.L3-N.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_VOLTS_PHASE_A ] . nut_entity = " input.L1-N.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_A ] . nut_entity = " output.L1.current " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_A_BAR_CHART ] . nut_entity = " output.L1.current.nominal " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VOLTS_A ] . nut_entity = " output.L1-N.voltage " ;
}
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_FREQUENCY ] . nut_entity = " output.frequency " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_FREQUENCY ] . nut_entity = " input.frequency " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_BYPASS_FREQUENCY ] . nut_entity = " input.bypass.frequency " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_BATTERY_CURRENT ] . nut_entity = " battery.current " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_BATTERY_VOLTAGE ] . nut_entity = " battery.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_PERCENT_BATTERY_LEFT ] . nut_entity = " battery.charge " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_BATTERY_TIME_REMAINING ] . nut_entity = " battery.runtime " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_BATTERY_DCUV_BAR_CHART ] . nut_entity = " battery.voltage.low " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_LOW_BATTERY_WARNING_V_BAR_CHART ] . nut_entity = " battery.charge.low " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_BATTERY_DISCHARGING_CURRENT_BAR_CHART ] . nut_entity = " battery.current.total " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_VOLTS_PHASE_B ] . nut_entity = " input.L2-N.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_INPUT_VOLTS_PHASE_C ] . nut_entity = " input.L3-N.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_AMBIENT_TEMPERATURE ] . nut_entity = " ambient.temperature " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_HEATSINK_TEMPERATURE ] . nut_entity = " ups.temperature " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_POWER_SUPPLY_TEMPERATURE ] . nut_entity = " ambient.1.temperature " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_B ] . nut_entity = " output.L2.current " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_C ] . nut_entity = " output.L3.current " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_B_BAR_CHART ] . nut_entity = " output.L2.current.nominal " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_C_BAR_CHART ] . nut_entity = " output.L3.current.nominal " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VA_BAR_CHART ] . nut_entity = " ups.power.nominal " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_DATE ] . nut_entity = " ups.date " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_TIME ] . nut_entity = " ups.time " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_BATTERY_TEMPERATURE ] . nut_entity = " battery.temperature " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VOLTS_B ] . nut_entity = " output.L2-N.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VOLTS_C ] . nut_entity = " output.L3-N.voltage " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_WATTS_PHASE_A ] . nut_entity = " ups.L1-N.realpower " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_WATTS_PHASE_B ] . nut_entity = " ups.L2-N.realpower " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_WATTS_PHASE_C ] . nut_entity = " ups.L3-N.realpower " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_WATTS_PHASE_A_B_C_BAR_CHART ] . nut_entity = " ups.realpower.nominal " ;
bcmxcp_meter_map [ BCMXCP_METER_MAP_LINE_EVENT_COUNTER ] . nut_entity = " input.quality " ;
2010-03-26 01:20:59 +02:00
}
void init_alarm_map ( )
{
/* Clean entire map */
memset ( & bcmxcp_alarm_map , 0 , sizeof ( BCMXCP_ALARM_MAP_ENTRY_t ) * BCMXCP_ALARM_MAP_MAX ) ;
2013-11-24 17:00:12 +02:00
/* Set all alarm descriptions */
2010-03-26 01:20:59 +02:00
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_AC_OVER_VOLTAGE ] . alarm_desc = " INVERTER_AC_OVER_VOLTAGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_AC_UNDER_VOLTAGE ] . alarm_desc = " INVERTER_AC_UNDER_VOLTAGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_OVER_OR_UNDER_FREQ ] . alarm_desc = " INVERTER_OVER_OR_UNDER_FREQ " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BYPASS_AC_OVER_VOLTAGE ] . alarm_desc = " BYPASS_AC_OVER_VOLTAGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BYPASS_AC_UNDER_VOLTAGE ] . alarm_desc = " BYPASS_AC_UNDER_VOLTAGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BYPASS_OVER_OR_UNDER_FREQ ] . alarm_desc = " BYPASS_OVER_OR_UNDER_FREQ " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INPUT_AC_OVER_VOLTAGE ] . alarm_desc = " INPUT_AC_OVER_VOLTAGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INPUT_AC_UNDER_VOLTAGE ] . alarm_desc = " INPUT_AC_UNDER_VOLTAGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INPUT_UNDER_OR_OVER_FREQ ] . alarm_desc = " INPUT_UNDER_OR_OVER_FREQ " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTPUT_OVER_VOLTAGE ] . alarm_desc = " OUTPUT_OVER_VOLTAGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTPUT_UNDER_VOLTAGE ] . alarm_desc = " OUTPUT_UNDER_VOLTAGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTPUT_UNDER_OR_OVER_FREQ ] . alarm_desc = " OUTPUT_UNDER_OR_OVER_FREQ " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_REMOTE_EMERGENCY_PWR_OFF ] . alarm_desc = " REMOTE_EMERGENCY_PWR_OFF " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_REMOTE_GO_TO_BYPASS ] . alarm_desc = " REMOTE_GO_TO_BYPASS " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BUILDING_ALARM_6 ] . alarm_desc = " BUILDING_ALARM_6 " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BUILDING_ALARM_5 ] . alarm_desc = " BUILDING_ALARM_5 " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BUILDING_ALARM_4 ] . alarm_desc = " BUILDING_ALARM_4 " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BUILDING_ALARM_3 ] . alarm_desc = " BUILDING_ALARM_3 " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BUILDING_ALARM_2 ] . alarm_desc = " BUILDING_ALARM_2 " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BUILDING_ALARM_1 ] . alarm_desc = " BUILDING_ALARM_1 " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_STATIC_SWITCH_OVER_TEMP ] . alarm_desc = " STATIC_SWITCH_OVER_TEMP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CHARGER_OVER_TEMP ] . alarm_desc = " CHARGER_OVER_TEMP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CHARGER_LOGIC_PWR_FAIL ] . alarm_desc = " CHARGER_LOGIC_PWR_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CHARGER_OVER_VOLTAGE_OR_CURRENT ] . alarm_desc = " CHARGER_OVER_VOLTAGE_OR_CURRENT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_OVER_TEMP ] . alarm_desc = " INVERTER_OVER_TEMP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTPUT_OVERLOAD ] . alarm_desc = " OUTPUT_OVERLOAD " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_INPUT_OVER_CURRENT ] . alarm_desc = " RECTIFIER_INPUT_OVER_CURRENT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_OUTPUT_OVER_CURRENT ] . alarm_desc = " INVERTER_OUTPUT_OVER_CURRENT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_DC_LINK_OVER_VOLTAGE ] . alarm_desc = " DC_LINK_OVER_VOLTAGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_DC_LINK_UNDER_VOLTAGE ] . alarm_desc = " DC_LINK_UNDER_VOLTAGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_FAILED ] . alarm_desc = " RECTIFIER_FAILED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_FAULT ] . alarm_desc = " INVERTER_FAULT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_CONNECTOR_FAIL ] . alarm_desc = " BATTERY_CONNECTOR_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BYPASS_BREAKER_FAIL ] . alarm_desc = " BYPASS_BREAKER_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CHARGER_FAIL ] . alarm_desc = " CHARGER_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RAMP_UP_FAILED ] . alarm_desc = " RAMP_UP_FAILED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_STATIC_SWITCH_FAILED ] . alarm_desc = " STATIC_SWITCH_FAILED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_ANALOG_AD_REF_FAIL ] . alarm_desc = " ANALOG_AD_REF_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BYPASS_UNCALIBRATED ] . alarm_desc = " BYPASS_UNCALIBRATED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_UNCALIBRATED ] . alarm_desc = " RECTIFIER_UNCALIBRATED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTPUT_UNCALIBRATED ] . alarm_desc = " OUTPUT_UNCALIBRATED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_UNCALIBRATED ] . alarm_desc = " INVERTER_UNCALIBRATED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_DC_VOLT_UNCALIBRATED ] . alarm_desc = " DC_VOLT_UNCALIBRATED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTPUT_CURRENT_UNCALIBRATED ] . alarm_desc = " OUTPUT_CURRENT_UNCALIBRATED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_CURRENT_UNCALIBRATED ] . alarm_desc = " RECTIFIER_CURRENT_UNCALIBRATED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_CURRENT_UNCALIBRATED ] . alarm_desc = " BATTERY_CURRENT_UNCALIBRATED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_ON_OFF_STAT_FAIL ] . alarm_desc = " INVERTER_ON_OFF_STAT_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_CURRENT_LIMIT ] . alarm_desc = " BATTERY_CURRENT_LIMIT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_STARTUP_FAIL ] . alarm_desc = " INVERTER_STARTUP_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_ANALOG_BOARD_AD_STAT_FAIL ] . alarm_desc = " ANALOG_BOARD_AD_STAT_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTPUT_CURRENT_OVER_100 ] . alarm_desc = " OUTPUT_CURRENT_OVER_100 " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_GROUND_FAULT ] . alarm_desc = " BATTERY_GROUND_FAULT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_WAITING_FOR_CHARGER_SYNC ] . alarm_desc = " WAITING_FOR_CHARGER_SYNC " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_NV_RAM_FAIL ] . alarm_desc = " NV_RAM_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_ANALOG_BOARD_AD_TIMEOUT ] . alarm_desc = " ANALOG_BOARD_AD_TIMEOUT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SHUTDOWN_IMMINENT ] . alarm_desc = " SHUTDOWN_IMMINENT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_LOW ] . alarm_desc = " BATTERY_LOW " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_UTILITY_FAIL ] . alarm_desc = " UTILITY_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTPUT_SHORT_CIRCUIT ] . alarm_desc = " OUTPUT_SHORT_CIRCUIT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_UTILITY_NOT_PRESENT ] . alarm_desc = " UTILITY_NOT_PRESENT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_FULL_TIME_CHARGING ] . alarm_desc = " FULL_TIME_CHARGING " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_FAST_BYPASS_COMMAND ] . alarm_desc = " FAST_BYPASS_COMMAND " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_AD_ERROR ] . alarm_desc = " AD_ERROR " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INTERNAL_COM_FAIL ] . alarm_desc = " INTERNAL_COM_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_SELFTEST_FAIL ] . alarm_desc = " RECTIFIER_SELFTEST_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_EEPROM_FAIL ] . alarm_desc = " RECTIFIER_EEPROM_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_EPROM_FAIL ] . alarm_desc = " RECTIFIER_EPROM_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INPUT_LINE_VOLTAGE_LOSS ] . alarm_desc = " INPUT_LINE_VOLTAGE_LOSS " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_DC_OVER_VOLTAGE ] . alarm_desc = " BATTERY_DC_OVER_VOLTAGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_POWER_SUPPLY_OVER_TEMP ] . alarm_desc = " POWER_SUPPLY_OVER_TEMP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_POWER_SUPPLY_FAIL ] . alarm_desc = " POWER_SUPPLY_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_POWER_SUPPLY_5V_FAIL ] . alarm_desc = " POWER_SUPPLY_5V_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_POWER_SUPPLY_12V_FAIL ] . alarm_desc = " POWER_SUPPLY_12V_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_HEATSINK_OVER_TEMP ] . alarm_desc = " HEATSINK_OVER_TEMP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_HEATSINK_TEMP_SENSOR_FAIL ] . alarm_desc = " HEATSINK_TEMP_SENSOR_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_CURRENT_OVER_125 ] . alarm_desc = " RECTIFIER_CURRENT_OVER_125 " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_FAULT_INTERRUPT_FAIL ] . alarm_desc = " RECTIFIER_FAULT_INTERRUPT_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_POWER_CAPASITOR_FAIL ] . alarm_desc = " RECTIFIER_POWER_CAPASITOR_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_PROGRAM_STACK_ERROR ] . alarm_desc = " INVERTER_PROGRAM_STACK_ERROR " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_BOARD_SELFTEST_FAIL ] . alarm_desc = " INVERTER_BOARD_SELFTEST_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_AD_SELFTEST_FAIL ] . alarm_desc = " INVERTER_AD_SELFTEST_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_RAM_SELFTEST_FAIL ] . alarm_desc = " INVERTER_RAM_SELFTEST_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_NV_MEMORY_CHECKSUM_FAIL ] . alarm_desc = " NV_MEMORY_CHECKSUM_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_PROGRAM_CHECKSUM_FAIL ] . alarm_desc = " PROGRAM_CHECKSUM_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_CPU_SELFTEST_FAIL ] . alarm_desc = " INVERTER_CPU_SELFTEST_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_NETWORK_NOT_RESPONDING ] . alarm_desc = " NETWORK_NOT_RESPONDING " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_FRONT_PANEL_SELFTEST_FAIL ] . alarm_desc = " FRONT_PANEL_SELFTEST_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_NODE_EEPROM_VERIFICATION_ERROR ] . alarm_desc = " NODE_EEPROM_VERIFICATION_ERROR " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTPUT_AC_OVER_VOLT_TEST_FAIL ] . alarm_desc = " OUTPUT_AC_OVER_VOLT_TEST_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTPUT_DC_OVER_VOLTAGE ] . alarm_desc = " OUTPUT_DC_OVER_VOLTAGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INPUT_PHASE_ROTATION_ERROR ] . alarm_desc = " INPUT_PHASE_ROTATION_ERROR " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_RAMP_UP_TEST_FAILED ] . alarm_desc = " INVERTER_RAMP_UP_TEST_FAILED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_OFF_COMMAND ] . alarm_desc = " INVERTER_OFF_COMMAND " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_ON_COMMAND ] . alarm_desc = " INVERTER_ON_COMMAND " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_TO_BYPASS_COMMAND ] . alarm_desc = " TO_BYPASS_COMMAND " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_FROM_BYPASS_COMMAND ] . alarm_desc = " FROM_BYPASS_COMMAND " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_AUTO_MODE_COMMAND ] . alarm_desc = " AUTO_MODE_COMMAND " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_EMERGENCY_SHUTDOWN_COMMAND ] . alarm_desc = " EMERGENCY_SHUTDOWN_COMMAND " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SETUP_SWITCH_OPEN ] . alarm_desc = " SETUP_SWITCH_OPEN " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_OVER_VOLT_INT ] . alarm_desc = " INVERTER_OVER_VOLT_INT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_UNDER_VOLT_INT ] . alarm_desc = " INVERTER_UNDER_VOLT_INT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_ABSOLUTE_DCOV_ACOV ] . alarm_desc = " ABSOLUTE_DCOV_ACOV " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_PHASE_A_CURRENT_LIMIT ] . alarm_desc = " PHASE_A_CURRENT_LIMIT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_PHASE_B_CURRENT_LIMIT ] . alarm_desc = " PHASE_B_CURRENT_LIMIT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_PHASE_C_CURRENT_LIMIT ] . alarm_desc = " PHASE_C_CURRENT_LIMIT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BYPASS_NOT_AVAILABLE ] . alarm_desc = " BYPASS_NOT_AVAILABLE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_BREAKER_OPEN ] . alarm_desc = " RECTIFIER_BREAKER_OPEN " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_CONTACTOR_OPEN ] . alarm_desc = " BATTERY_CONTACTOR_OPEN " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_CONTACTOR_OPEN ] . alarm_desc = " INVERTER_CONTACTOR_OPEN " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BYPASS_BREAKER_OPEN ] . alarm_desc = " BYPASS_BREAKER_OPEN " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INV_BOARD_ACOV_INT_TEST_FAIL ] . alarm_desc = " INV_BOARD_ACOV_INT_TEST_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_OVER_TEMP_TRIP ] . alarm_desc = " INVERTER_OVER_TEMP_TRIP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INV_BOARD_ACUV_INT_TEST_FAIL ] . alarm_desc = " INV_BOARD_ACUV_INT_TEST_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_VOLTAGE_FEEDBACK_ERROR ] . alarm_desc = " INVERTER_VOLTAGE_FEEDBACK_ERROR " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_DC_UNDER_VOLTAGE_TIMEOUT ] . alarm_desc = " DC_UNDER_VOLTAGE_TIMEOUT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_AC_UNDER_VOLTAGE_TIMEOUT ] . alarm_desc = " AC_UNDER_VOLTAGE_TIMEOUT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_DC_UNDER_VOLTAGE_WHILE_CHARGE ] . alarm_desc = " DC_UNDER_VOLTAGE_WHILE_CHARGE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_VOLTAGE_BIAS_ERROR ] . alarm_desc = " INVERTER_VOLTAGE_BIAS_ERROR " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_PHASE_ROTATION ] . alarm_desc = " RECTIFIER_PHASE_ROTATION " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BYPASS_PHASER_ROTATION ] . alarm_desc = " BYPASS_PHASER_ROTATION " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SYSTEM_INTERFACE_BOARD_FAIL ] . alarm_desc = " SYSTEM_INTERFACE_BOARD_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_PARALLEL_BOARD_FAIL ] . alarm_desc = " PARALLEL_BOARD_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LOST_LOAD_SHARING_PHASE_A ] . alarm_desc = " LOST_LOAD_SHARING_PHASE_A " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LOST_LOAD_SHARING_PHASE_B ] . alarm_desc = " LOST_LOAD_SHARING_PHASE_B " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LOST_LOAD_SHARING_PHASE_C ] . alarm_desc = " LOST_LOAD_SHARING_PHASE_C " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_DC_OVER_VOLTAGE_TIMEOUT ] . alarm_desc = " DC_OVER_VOLTAGE_TIMEOUT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_TOTALLY_DISCHARGED ] . alarm_desc = " BATTERY_TOTALLY_DISCHARGED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_PHASE_BIAS_ERROR ] . alarm_desc = " INVERTER_PHASE_BIAS_ERROR " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_VOLTAGE_BIAS_ERROR_2 ] . alarm_desc = " INVERTER_VOLTAGE_BIAS_ERROR_2 " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_DC_LINK_BLEED_COMPLETE ] . alarm_desc = " DC_LINK_BLEED_COMPLETE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LARGE_CHARGER_INPUT_CURRENT ] . alarm_desc = " LARGE_CHARGER_INPUT_CURRENT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INV_VOLT_TOO_LOW_FOR_RAMP_LEVEL ] . alarm_desc = " INV_VOLT_TOO_LOW_FOR_RAMP_LEVEL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LOSS_OF_REDUNDANCY ] . alarm_desc = " LOSS_OF_REDUNDANCY " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LOSS_OF_SYNC_BUS ] . alarm_desc = " LOSS_OF_SYNC_BUS " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_BREAKER_SHUNT_TRIP ] . alarm_desc = " RECTIFIER_BREAKER_SHUNT_TRIP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LOSS_OF_CHARGER_SYNC ] . alarm_desc = " LOSS_OF_CHARGER_SYNC " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_LOW_LEVEL_TEST_TIMEOUT ] . alarm_desc = " INVERTER_LOW_LEVEL_TEST_TIMEOUT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTPUT_BREAKER_OPEN ] . alarm_desc = " OUTPUT_BREAKER_OPEN " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CONTROL_POWER_ON ] . alarm_desc = " CONTROL_POWER_ON " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_ON ] . alarm_desc = " INVERTER_ON " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CHARGER_ON ] . alarm_desc = " CHARGER_ON " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BYPASS_ON ] . alarm_desc = " BYPASS_ON " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BYPASS_POWER_LOSS ] . alarm_desc = " BYPASS_POWER_LOSS " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_ON_MANUAL_BYPASS ] . alarm_desc = " ON_MANUAL_BYPASS " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BYPASS_MANUAL_TURN_OFF ] . alarm_desc = " BYPASS_MANUAL_TURN_OFF " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_BLEEDING_DC_LINK_VOLT ] . alarm_desc = " INVERTER_BLEEDING_DC_LINK_VOLT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CPU_ISR_ERROR ] . alarm_desc = " CPU_ISR_ERROR " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SYSTEM_ISR_RESTART ] . alarm_desc = " SYSTEM_ISR_RESTART " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_PARALLEL_DC ] . alarm_desc = " PARALLEL_DC " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_NEEDS_SERVICE ] . alarm_desc = " BATTERY_NEEDS_SERVICE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_CHARGING ] . alarm_desc = " BATTERY_CHARGING " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_NOT_CHARGED ] . alarm_desc = " BATTERY_NOT_CHARGED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_DISABLED_BATTERY_TIME ] . alarm_desc = " DISABLED_BATTERY_TIME " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SERIES_7000_ENABLE ] . alarm_desc = " SERIES_7000_ENABLE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OTHER_UPS_ON ] . alarm_desc = " OTHER_UPS_ON " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_PARALLEL_INVERTER ] . alarm_desc = " PARALLEL_INVERTER " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_UPS_IN_PARALLEL ] . alarm_desc = " UPS_IN_PARALLEL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTPUT_BREAKER_REALY_FAIL ] . alarm_desc = " OUTPUT_BREAKER_REALY_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CONTROL_POWER_OFF ] . alarm_desc = " CONTROL_POWER_OFF " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LEVEL_2_OVERLOAD_PHASE_A ] . alarm_desc = " LEVEL_2_OVERLOAD_PHASE_A " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LEVEL_2_OVERLOAD_PHASE_B ] . alarm_desc = " LEVEL_2_OVERLOAD_PHASE_B " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LEVEL_2_OVERLOAD_PHASE_C ] . alarm_desc = " LEVEL_2_OVERLOAD_PHASE_C " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LEVEL_3_OVERLOAD_PHASE_A ] . alarm_desc = " LEVEL_3_OVERLOAD_PHASE_A " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LEVEL_3_OVERLOAD_PHASE_B ] . alarm_desc = " LEVEL_3_OVERLOAD_PHASE_B " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LEVEL_3_OVERLOAD_PHASE_C ] . alarm_desc = " LEVEL_3_OVERLOAD_PHASE_C " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LEVEL_4_OVERLOAD_PHASE_A ] . alarm_desc = " LEVEL_4_OVERLOAD_PHASE_A " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LEVEL_4_OVERLOAD_PHASE_B ] . alarm_desc = " LEVEL_4_OVERLOAD_PHASE_B " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LEVEL_4_OVERLOAD_PHASE_C ] . alarm_desc = " LEVEL_4_OVERLOAD_PHASE_C " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_UPS_ON_BATTERY ] . alarm_desc = " UPS_ON_BATTERY " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_UPS_ON_BYPASS ] . alarm_desc = " UPS_ON_BYPASS " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LOAD_DUMPED ] . alarm_desc = " LOAD_DUMPED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LOAD_ON_INVERTER ] . alarm_desc = " LOAD_ON_INVERTER " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_UPS_ON_COMMAND ] . alarm_desc = " UPS_ON_COMMAND " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_UPS_OFF_COMMAND ] . alarm_desc = " UPS_OFF_COMMAND " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_LOW_BATTERY_SHUTDOWN ] . alarm_desc = " LOW_BATTERY_SHUTDOWN " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_AUTO_ON_ENABLED ] . alarm_desc = " AUTO_ON_ENABLED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SOFTWARE_INCOMPABILITY_DETECTED ] . alarm_desc = " SOFTWARE_INCOMPABILITY_DETECTED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_TEMP_SENSOR_FAILED ] . alarm_desc = " INVERTER_TEMP_SENSOR_FAILED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_DC_START_OCCURED ] . alarm_desc = " DC_START_OCCURED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_IN_PARALLEL_OPERATION ] . alarm_desc = " IN_PARALLEL_OPERATION " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SYNCING_TO_BYPASS ] . alarm_desc = " SYNCING_TO_BYPASS " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RAMPING_UPS_UP ] . alarm_desc = " RAMPING_UPS_UP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_ON_DELAY ] . alarm_desc = " INVERTER_ON_DELAY " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CHARGER_ON_DELAY ] . alarm_desc = " CHARGER_ON_DELAY " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_WAITING_FOR_UTIL_INPUT ] . alarm_desc = " WAITING_FOR_UTIL_INPUT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CLOSE_BYPASS_BREAKER ] . alarm_desc = " CLOSE_BYPASS_BREAKER " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_TEMPORARY_BYPASS_OPERATION ] . alarm_desc = " TEMPORARY_BYPASS_OPERATION " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SYNCING_TO_OUTPUT ] . alarm_desc = " SYNCING_TO_OUTPUT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BYPASS_FAILURE ] . alarm_desc = " BYPASS_FAILURE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_AUTO_OFF_COMMAND_EXECUTED ] . alarm_desc = " AUTO_OFF_COMMAND_EXECUTED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_AUTO_ON_COMMAND_EXECUTED ] . alarm_desc = " AUTO_ON_COMMAND_EXECUTED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_TEST_FAILED ] . alarm_desc = " BATTERY_TEST_FAILED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_FUSE_FAIL ] . alarm_desc = " FUSE_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_FAN_FAIL ] . alarm_desc = " FAN_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SITE_WIRING_FAULT ] . alarm_desc = " SITE_WIRING_FAULT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BACKFEED_CONTACTOR_FAIL ] . alarm_desc = " BACKFEED_CONTACTOR_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_ON_BUCK ] . alarm_desc = " ON_BUCK " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_ON_BOOST ] . alarm_desc = " ON_BOOST " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_ON_DOUBLE_BOOST ] . alarm_desc = " ON_DOUBLE_BOOST " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERIES_DISCONNECTED ] . alarm_desc = " BATTERIES_DISCONNECTED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_UPS_CABINET_OVER_TEMP ] . alarm_desc = " UPS_CABINET_OVER_TEMP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_TRANSFORMER_OVER_TEMP ] . alarm_desc = " TRANSFORMER_OVER_TEMP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_AMBIENT_UNDER_TEMP ] . alarm_desc = " AMBIENT_UNDER_TEMP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_AMBIENT_OVER_TEMP ] . alarm_desc = " AMBIENT_OVER_TEMP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CABINET_DOOR_OPEN ] . alarm_desc = " CABINET_DOOR_OPEN " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CABINET_DOOR_OPEN_VOLT_PRESENT ] . alarm_desc = " CABINET_DOOR_OPEN_VOLT_PRESENT " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_AUTO_SHUTDOWN_PENDING ] . alarm_desc = " AUTO_SHUTDOWN_PENDING " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_TAP_SWITCHING_REALY_PENDING ] . alarm_desc = " TAP_SWITCHING_REALY_PENDING " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_UNABLE_TO_CHARGE_BATTERIES ] . alarm_desc = " UNABLE_TO_CHARGE_BATTERIES " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_STARTUP_FAILURE_CHECK_EPO ] . alarm_desc = " STARTUP_FAILURE_CHECK_EPO " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_AUTOMATIC_STARTUP_PENDING ] . alarm_desc = " AUTOMATIC_STARTUP_PENDING " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_MODEM_FAILED ] . alarm_desc = " MODEM_FAILED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INCOMING_MODEM_CALL_STARTED ] . alarm_desc = " INCOMING_MODEM_CALL_STARTED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_OUTGOING_MODEM_CALL_STARTED ] . alarm_desc = " OUTGOING_MODEM_CALL_STARTED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_MODEM_CONNECTION_ESTABLISHED ] . alarm_desc = " MODEM_CONNECTION_ESTABLISHED " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_MODEM_CALL_COMPLETED_SUCCESS ] . alarm_desc = " MODEM_CALL_COMPLETED_SUCCESS " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_MODEM_CALL_COMPLETED_FAIL ] . alarm_desc = " MODEM_CALL_COMPLETED_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INPUT_BREAKER_FAIL ] . alarm_desc = " INPUT_BREAKER_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SYSINIT_IN_PROGRESS ] . alarm_desc = " SYSINIT_IN_PROGRESS " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_AUTOCALIBRATION_FAIL ] . alarm_desc = " AUTOCALIBRATION_FAIL " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SELECTIVE_TRIP_OF_MODULE ] . alarm_desc = " SELECTIVE_TRIP_OF_MODULE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_OUTPUT_FAILURE ] . alarm_desc = " INVERTER_OUTPUT_FAILURE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_ABNORMAL_OUTPUT_VOLT_AT_STARTUP ] . alarm_desc = " ABNORMAL_OUTPUT_VOLT_AT_STARTUP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_RECTIFIER_OVER_TEMP ] . alarm_desc = " RECTIFIER_OVER_TEMP " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CONFIG_ERROR ] . alarm_desc = " CONFIG_ERROR " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_REDUNDANCY_LOSS_DUE_TO_OVERLOAD ] . alarm_desc = " REDUNDANCY_LOSS_DUE_TO_OVERLOAD " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_ON_ALTERNATE_AC_SOURCE ] . alarm_desc = " ON_ALTERNATE_AC_SOURCE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_IN_HIGH_EFFICIENCY_MODE ] . alarm_desc = " IN_HIGH_EFFICIENCY_MODE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SYSTEM_NOTICE_ACTIVE ] . alarm_desc = " SYSTEM_NOTICE_ACTIVE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SYSTEM_ALARM_ACTIVE ] . alarm_desc = " SYSTEM_ALARM_ACTIVE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_ALTERNATE_POWER_SOURCE_NOT_AVAILABLE ] . alarm_desc = " ALTERNATE_POWER_SOURCE_NOT_AVAILABLE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CURRENT_BALANCE_FAILURE ] . alarm_desc = " CURRENT_BALANCE_FAILURE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CHECK_AIR_FILTER ] . alarm_desc = " CHECK_AIR_FILTER " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SUBSYSTEM_NOTICE_ACTIVE ] . alarm_desc = " SUBSYSTEM_NOTICE_ACTIVE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SUBSYSTEM_ALARM_ACTIVE ] . alarm_desc = " SUBSYSTEM_ALARM_ACTIVE " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CHARGER_ON_COMMAND ] . alarm_desc = " CHARGER_ON_COMMAND " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_CHARGER_OFF_COMMAND ] . alarm_desc = " CHARGER_OFF_COMMAND " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_UPS_NORMAL ] . alarm_desc = " UPS_NORMAL " ;
2012-06-01 16:55:19 +03:00
bcmxcp_alarm_map [ BCMXCP_ALARM_INVERTER_PHASE_ROTATION ] . alarm_desc = " INVERTER_PHASE_ROTATION " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_UPS_OFF ] . alarm_desc = " UPS_OFF " ;
2010-03-26 01:20:59 +02:00
bcmxcp_alarm_map [ BCMXCP_ALARM_EXTERNAL_COMMUNICATION_FAILURE ] . alarm_desc = " EXTERNAL_COMMUNICATION_FAILURE " ;
2012-06-01 16:55:19 +03:00
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_TEST_INPROGRESS ] . alarm_desc = " BATTERY_TEST_INPROGRESS " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_SYSTEM_TEST_INPROGRESS ] . alarm_desc = " SYSTEM_TEST_INPROGRESS " ;
bcmxcp_alarm_map [ BCMXCP_ALARM_BATTERY_TEST_ABORTED ] . alarm_desc = " BATTERY_TEST_ABORTED " ;
2010-03-26 01:20:59 +02:00
}
/* Get information on UPS commands */
2013-11-24 17:00:12 +02:00
bool_t init_command ( int size )
2010-03-26 01:20:59 +02:00
{
unsigned char answer [ PW_ANSWER_MAX_SIZE ] ;
2013-11-24 17:00:12 +02:00
unsigned char commandByte ;
const char * nutvalue ;
int res , iIndex = 0 , ncounter , NumComms = 0 , i ;
2010-03-26 01:20:59 +02:00
2013-11-24 17:00:12 +02:00
upsdebugx ( 1 , " entering init_command(%i) " , size ) ;
2010-03-26 01:20:59 +02:00
res = command_read_sequence ( PW_COMMAND_LIST_REQ , answer ) ;
if ( res < = 0 )
2013-11-24 17:00:12 +02:00
{
2010-03-26 01:20:59 +02:00
upsdebugx ( 2 , " No command list block. " ) ;
2013-11-24 17:00:12 +02:00
return FALSE ;
}
2010-03-26 01:20:59 +02:00
else
2013-11-24 17:00:12 +02:00
{
2010-03-26 01:20:59 +02:00
upsdebugx ( 2 , " Command list block supported. " ) ;
2013-11-24 17:00:12 +02:00
res = answer [ iIndex ] ;
NumComms = ( int ) res ; /* Number of commands implemented in this UPS */
upsdebugx ( 3 , " Number of commands implemented in ups %d " , res ) ;
iIndex + + ;
res = answer [ iIndex ] ; /* Entry length - bytes reported for each command */
iIndex + + ;
upsdebugx ( 5 , " bytes per command %d " , res ) ;
/* In case of debug - make explanation of values */
upsdebugx ( 2 , " Index \t Cmd byte \t Description " ) ;
/* Get command bytes if size of command block matches with size from standard ID block */
if ( NumComms + 2 = = size )
{
for ( ncounter = 0 ; ncounter < NumComms ; ncounter + + )
{
commandByte = answer [ iIndex ] ;
if ( commandByte > = 0 & & commandByte < BCMXCP_COMMAND_MAP_MAX ) {
upsdebugx ( 2 , " %03d \t %02x \t %s " , ncounter , commandByte , bcmxcp_command_map [ commandByte ] . command_desc ) ;
bcmxcp_command_map [ commandByte ] . command_byte = commandByte ;
}
else {
upsdebugx ( 2 , " %03d \t %02x \t %s " , ncounter , commandByte , " Unknown command, the commandByte is not mapped " ) ;
}
iIndex + + ;
}
/* Map supported commands to instcmd */
for ( i = 0 ; i < BCMXCP_COMMAND_MAP_MAX ; i + + ) {
if ( bcmxcp_command_map [ i ] . command_desc ! = NULL ) {
if ( bcmxcp_command_map [ i ] . command_byte > 0 ) {
if ( ( nutvalue = nut_find_infoval ( command_map_info , bcmxcp_command_map [ i ] . command_byte , FALSE ) ) ! = NULL ) {
dstate_addcmd ( nutvalue ) ;
upsdebugx ( 2 , " Added support for instcmd %s " , nutvalue ) ;
}
}
}
}
return TRUE ;
}
else {
upsdebugx ( 1 , " Invalid response received from Command List block " ) ;
return FALSE ;
}
}
2010-03-26 01:20:59 +02:00
}
void init_ups_meter_map ( const unsigned char * map , unsigned char len )
{
unsigned int iIndex , iOffset = 0 ;
/* In case of debug - make explanation of values */
2013-11-24 17:00:12 +02:00
upsdebugx ( 2 , " Index \t Offset \t Format \t NUT " ) ;
2010-03-26 01:20:59 +02:00
/* Loop thru map */
for ( iIndex = 0 ; iIndex < len & & iIndex < BCMXCP_METER_MAP_MAX ; iIndex + + )
{
bcmxcp_meter_map [ iIndex ] . format = map [ iIndex ] ;
if ( map [ iIndex ] ! = 0 )
{
/* Set meter map entry offset */
bcmxcp_meter_map [ iIndex ] . meter_block_index = iOffset ;
/* Debug info */
upsdebugx ( 2 , " %04d \t %04d \t %2x \t %s " , iIndex , iOffset , bcmxcp_meter_map [ iIndex ] . format ,
( bcmxcp_meter_map [ iIndex ] . nut_entity = = NULL ? " None " : bcmxcp_meter_map [ iIndex ] . nut_entity ) ) ;
iOffset + = 4 ;
}
}
upsdebugx ( 2 , " \n " ) ;
}
void decode_meter_map_entry ( const unsigned char * entry , const unsigned char format , char * value )
{
long lValue = 0 ;
char sFormat [ 32 ] ;
float fValue ;
2013-11-24 17:00:12 +02:00
unsigned char dd , mm , yy , cc , hh , ss ;
2010-03-26 01:20:59 +02:00
/* Paranoid input sanity checks */
if ( value = = NULL )
return ;
* value = ' \0 ' ;
if ( entry = = ( unsigned char * ) NULL | | format = = 0x00 )
return ;
/* Get data based on format */
if ( format = = 0xf0 ) {
/* Long integer */
lValue = get_long ( entry ) ;
snprintf ( value , 127 , " %d " , ( int ) lValue ) ;
}
else if ( ( format & 0xf0 ) = = 0xf0 ) {
/* Fixed point integer */
fValue = get_long ( entry ) / ldexp ( 1 , format & 0x0f ) ;
snprintf ( value , 127 , " %.2f " , fValue ) ;
}
else if ( format < = 0x97 ) {
/* Floating point */
fValue = get_float ( entry ) ;
2013-11-24 17:00:12 +02:00
/* Format is packed BCD */
2010-03-26 01:20:59 +02:00
snprintf ( sFormat , 31 , " %%%d.%df " , ( ( format & 0xf0 ) > > 4 ) , ( format & 0x0f ) ) ;
snprintf ( value , 127 , sFormat , fValue ) ;
}
else if ( format = = 0xe2 ) {
/* Seconds */
lValue = get_long ( entry ) ;
snprintf ( value , 127 , " %d " , ( int ) lValue ) ;
}
else if ( format = = 0xe0 ) {
/* Date */
2013-11-24 17:00:12 +02:00
/* Format is packed BCD for each byte, and cc uses most signifcant bit to signal date format */
dd = entry [ 0 ] ;
mm = entry [ 1 ] ;
yy = entry [ 2 ] ;
2010-03-26 01:20:59 +02:00
cc = entry [ 3 ] ;
/* Check format type */
if ( cc & 0x80 ) {
2013-11-24 17:00:12 +02:00
/* Month:Day format */
snprintf ( value , 127 , " %d%d/%d%d/%d%d%d%d " , ( ( dd & 0xf0 ) > > 4 ) , ( dd & 0x0f ) , ( ( mm & 0xf0 ) > > 4 ) , ( mm & 0x0f ) , ( ( ( cc & 0x7f ) & 0xf0 ) > > 4 ) , ( ( cc & 0x7f ) & 0x0f ) , ( ( yy & 0xf0 ) > > 4 ) , ( yy & 0x0f ) ) ;
2010-03-26 01:20:59 +02:00
}
else {
2013-11-24 17:00:12 +02:00
/* Julian format */
/* TODO test this, unsure if the day part is correct, i.e. how we use the two bytes mm and dd to calculate the number of julian days */
snprintf ( value , 127 , " %d%d%d%d:%d%d%d " , ( ( ( cc & 0x7f ) & 0xf0 ) > > 4 ) , ( ( cc & 0x7f ) & 0x0f ) , ( ( yy & 0xf0 ) > > 4 ) , ( yy & 0x0f ) , ( mm & 0x0f ) , ( ( dd & 0xf0 ) > > 4 ) , ( dd & 0x0f ) ) ;
2010-03-26 01:20:59 +02:00
}
}
else if ( format = = 0xe1 ) {
/* Time */
2013-11-24 17:00:12 +02:00
/* Format is packed BCD for each byte */
cc = entry [ 0 ] ;
ss = entry [ 1 ] ;
mm = entry [ 2 ] ;
2010-03-26 01:20:59 +02:00
hh = entry [ 3 ] ;
2013-11-24 17:00:12 +02:00
snprintf ( value , 127 , " %d%d:%d%d:%d%d.%d%d " , ( ( hh & 0xf0 ) > > 4 ) , ( hh & 0x0f ) , ( ( mm & 0xf0 ) > > 4 ) , ( mm & 0x0f ) , ( ( ss & 0xf0 ) > > 4 ) , ( ss & 0x0f ) , ( ( cc & 0xf0 ) > > 4 ) , ( cc & 0x0f ) ) ;
2010-03-26 01:20:59 +02:00
}
else {
/* Unknown format */
snprintf ( value , 127 , " ??? " ) ;
return ;
}
return ;
}
void init_ups_alarm_map ( const unsigned char * map , unsigned char len )
{
2013-11-24 17:00:12 +02:00
unsigned int iIndex = 0 ;
2010-03-26 01:20:59 +02:00
int alarm = 0 ;
/* In case of debug - make explanation of values */
2013-11-24 17:00:12 +02:00
upsdebugx ( 2 , " Index \t Alarm \t Supported " ) ;
2010-03-26 01:20:59 +02:00
/* Loop thru map */
for ( iIndex = 0 ; iIndex < len & & iIndex < BCMXCP_ALARM_MAP_MAX / 8 ; iIndex + + )
{
/* Bit 0 */
2013-11-24 17:00:12 +02:00
if ( set_alarm_support_in_alarm_map ( map , iIndex , 0x01 , iIndex * 8 , alarm ) = = TRUE )
2010-03-26 01:20:59 +02:00
alarm + + ;
/* Bit 1 */
2013-11-24 17:00:12 +02:00
if ( set_alarm_support_in_alarm_map ( map , iIndex , 0x02 , iIndex * 8 + 1 , alarm ) = = TRUE )
2010-03-26 01:20:59 +02:00
alarm + + ;
/* Bit 2 */
2013-11-24 17:00:12 +02:00
if ( set_alarm_support_in_alarm_map ( map , iIndex , 0x04 , iIndex * 8 + 2 , alarm ) = = TRUE )
2010-03-26 01:20:59 +02:00
alarm + + ;
/* Bit 3 */
2013-11-24 17:00:12 +02:00
if ( set_alarm_support_in_alarm_map ( map , iIndex , 0x08 , iIndex * 8 + 3 , alarm ) = = TRUE )
2010-03-26 01:20:59 +02:00
alarm + + ;
/* Bit 4 */
2013-11-24 17:00:12 +02:00
if ( set_alarm_support_in_alarm_map ( map , iIndex , 0x10 , iIndex * 8 + 4 , alarm ) = = TRUE )
2010-03-26 01:20:59 +02:00
alarm + + ;
/* Bit 5 */
2013-11-24 17:00:12 +02:00
if ( set_alarm_support_in_alarm_map ( map , iIndex , 0x20 , iIndex * 8 + 5 , alarm ) = = TRUE )
2010-03-26 01:20:59 +02:00
alarm + + ;
/* Bit 6 */
2013-11-24 17:00:12 +02:00
if ( set_alarm_support_in_alarm_map ( map , iIndex , 0x40 , iIndex * 8 + 6 , alarm ) = = TRUE )
alarm + + ;
/* Bit 7 */
if ( set_alarm_support_in_alarm_map ( map , iIndex , 0x80 , iIndex * 8 + 7 , alarm ) = = TRUE )
alarm + + ;
}
upsdebugx ( 2 , " \n " ) ;
}
bool_t set_alarm_support_in_alarm_map ( const unsigned char * map , const int mapIndex , const int bitmask , const int alarmMapIndex , const int alarmBlockIndex ) {
/* Check what the alarm block tells about the support for the alarm */
if ( map [ mapIndex ] & bitmask )
2010-03-26 01:20:59 +02:00
{
/* Set alarm active */
2013-11-24 17:00:12 +02:00
bcmxcp_alarm_map [ alarmMapIndex ] . alarm_block_index = alarmBlockIndex ;
2010-03-26 01:20:59 +02:00
}
else
{
/* Set alarm inactive */
2013-11-24 17:00:12 +02:00
bcmxcp_alarm_map [ alarmMapIndex ] . alarm_block_index = - 1 ;
2010-03-26 01:20:59 +02:00
}
2013-11-24 17:00:12 +02:00
/* Return if the alarm was supported or not */
if ( bcmxcp_alarm_map [ alarmMapIndex ] . alarm_block_index > = 0 ) {
2010-03-26 01:20:59 +02:00
/* Debug info */
2013-11-24 17:00:12 +02:00
upsdebugx ( 2 , " %04d \t %s \t Yes " , bcmxcp_alarm_map [ alarmMapIndex ] . alarm_block_index , bcmxcp_alarm_map [ alarmMapIndex ] . alarm_desc ) ;
return TRUE ;
2010-03-26 01:20:59 +02:00
}
2013-11-24 17:00:12 +02:00
else {
2010-03-26 01:20:59 +02:00
/* Debug info */
2013-11-24 17:00:12 +02:00
upsdebugx ( 3 , " %04d \t %s \t No " , bcmxcp_alarm_map [ alarmMapIndex ] . alarm_block_index , bcmxcp_alarm_map [ alarmMapIndex ] . alarm_desc ) ;
return FALSE ;
2010-03-26 01:20:59 +02:00
}
}
int init_outlet ( unsigned char len )
{
unsigned char answer [ PW_ANSWER_MAX_SIZE ] ;
int iIndex = 0 , res , num ;
int num_outlet , size_outlet ;
int outlet_num , outlet_state ;
short auto_dly_off , auto_dly_on ;
char outlet_name [ 25 ] ;
res = command_read_sequence ( PW_OUT_MON_BLOCK_REQ , answer ) ;
if ( res < = 0 )
fatal_with_errno ( EXIT_FAILURE , " Could not communicate with the ups " ) ;
2011-09-29 21:14:46 +03:00
else
upsdebugx ( 1 , " init_outlet(%i), res=%i " , len , res ) ;
2010-03-26 01:20:59 +02:00
num_outlet = answer [ iIndex + + ] ;
2013-11-24 17:00:12 +02:00
upsdebugx ( 2 , " Number of outlets: %d " , num_outlet ) ;
2010-03-26 01:20:59 +02:00
size_outlet = answer [ iIndex + + ] ;
2013-11-24 17:00:12 +02:00
upsdebugx ( 2 , " Number of bytes: %d " , size_outlet ) ;
2010-03-26 01:20:59 +02:00
for ( num = 1 ; num < = num_outlet ; num + + ) {
outlet_num = answer [ iIndex + + ] ;
2013-11-24 17:00:12 +02:00
upsdebugx ( 2 , " Outlet number: %d " , outlet_num ) ;
2010-03-26 01:20:59 +02:00
snprintf ( outlet_name , sizeof ( outlet_name ) - 1 , " outlet.%d.id " , num ) ;
dstate_setinfo ( outlet_name , " %d " , outlet_num ) ;
outlet_state = answer [ iIndex + + ] ;
2013-11-24 17:00:12 +02:00
upsdebugx ( 2 , " Outlet state: %d " , outlet_state ) ;
2010-03-26 01:20:59 +02:00
snprintf ( outlet_name , sizeof ( outlet_name ) - 1 , " outlet.%d.status " , num ) ;
2015-04-30 16:53:36 +03:00
if ( outlet_state > 0 & & outlet_state < 9 )
dstate_setinfo ( outlet_name , " %s " , OutletStatus [ outlet_state ] ) ;
2010-03-26 01:20:59 +02:00
auto_dly_off = get_word ( answer + iIndex ) ;
iIndex + = 2 ;
2013-11-24 17:00:12 +02:00
upsdebugx ( 2 , " Auto delay off: %d " , auto_dly_off ) ;
2010-03-26 01:20:59 +02:00
snprintf ( outlet_name , sizeof ( outlet_name ) - 1 , " outlet.%d.delay.shutdown " , num ) ;
dstate_setinfo ( outlet_name , " %d " , auto_dly_off ) ;
2013-11-24 17:00:12 +02:00
dstate_setflags ( outlet_name , ST_FLAG_RW | ST_FLAG_STRING ) ;
dstate_setaux ( outlet_name , 5 ) ;
2010-03-26 01:20:59 +02:00
auto_dly_on = get_word ( answer + iIndex ) ;
iIndex + = 2 ;
2013-11-24 17:00:12 +02:00
upsdebugx ( 2 , " Auto delay on: %d " , auto_dly_on ) ;
2010-03-26 01:20:59 +02:00
snprintf ( outlet_name , sizeof ( outlet_name ) - 1 , " outlet.%d.delay.start " , num ) ;
dstate_setinfo ( outlet_name , " %d " , auto_dly_on ) ;
2013-11-24 17:00:12 +02:00
dstate_setflags ( outlet_name , ST_FLAG_RW | ST_FLAG_STRING ) ;
dstate_setaux ( outlet_name , 5 ) ;
2010-03-26 01:20:59 +02:00
}
return num_outlet ;
}
2015-04-30 16:53:36 +03:00
void init_ext_vars ( void )
{
unsigned char answer [ PW_ANSWER_MAX_SIZE ] , cbuf [ 5 ] ;
int length = 0 , index = 0 ;
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
cbuf [ 0 ] = PW_SET_CONF_COMMAND ;
cbuf [ 1 ] = PW_CONF_REQ ;
cbuf [ 2 ] = 0x0 ;
cbuf [ 3 ] = 0x0 ;
length = command_write_sequence ( cbuf , 4 , answer ) ;
if ( length < = 0 )
fatal_with_errno ( EXIT_FAILURE , " Could not communicate with the ups " ) ;
if ( length < 4 ) //UPS dont have configurable vars
return ;
for ( index = 3 ; index < length ; index + + ) {
switch ( answer [ index ] ) {
case PW_CONF_LOW_DEV_LIMIT : dstate_setinfo ( " input.transfer.boost.high " , " %d " , 0 ) ;
dstate_setflags ( " input.transfer.boost.high " , ST_FLAG_RW | ST_FLAG_STRING ) ;
dstate_setaux ( " input.transfer.boost.high " , 3 ) ;
break ;
case PW_CONF_HIGH_DEV_LIMIT : dstate_setinfo ( " input.transfer.trim.low " , " %d " , 0 ) ;
dstate_setflags ( " input.transfer.trim.low " , ST_FLAG_RW | ST_FLAG_STRING ) ;
dstate_setaux ( " input.transfer.trim.low " , 3 ) ;
break ;
case PW_CONF_LOW_BATT : dstate_setinfo ( " battery.runtime.low " , " %d " , 0 ) ;
dstate_setflags ( " battery.runtime.low " , ST_FLAG_RW | ST_FLAG_STRING ) ;
dstate_setaux ( " battery.runtime.low " , 2 ) ;
break ;
case PW_CONF_BEEPER : dstate_addcmd ( " beeper.disable " ) ;
dstate_addcmd ( " beeper.enable " ) ;
dstate_addcmd ( " beeper.mute " ) ;
break ;
case PW_CONF_RETURN_DELAY : dstate_setinfo ( " input.transfer.delay " , " %d " , 0 ) ;
dstate_setflags ( " input.transfer.delay " , ST_FLAG_RW | ST_FLAG_STRING ) ;
dstate_setaux ( " input.transfer.delay " , 5 ) ;
break ;
case PW_CONF_RETURN_CAP : dstate_setinfo ( " battery.charge.restart " , " %d " , 0 ) ;
dstate_setflags ( " battery.charge.restart " , ST_FLAG_RW | ST_FLAG_STRING ) ;
dstate_setaux ( " battery.charge.restart " , 3 ) ;
break ;
case PW_CONF_MAX_TEMP : dstate_setinfo ( " ambient.temperature.high " , " %d " , 0 ) ;
dstate_setflags ( " ambient.temperature.high " , ST_FLAG_RW | ST_FLAG_STRING ) ;
dstate_setaux ( " ambient.temperature.high " , 3 ) ;
break ;
case PW_CONF_NOMINAL_OUT_VOLTAGE : dstate_setinfo ( " output.voltage.nominal " , " %d " , 0 ) ;
dstate_setflags ( " output.voltage.nominal " , ST_FLAG_RW | ST_FLAG_STRING ) ;
dstate_setaux ( " output.voltage.nominal " , 3 ) ;
break ;
case PW_CONF_SLEEP_TH_LOAD : dstate_setinfo ( " battery.energysave.load " , " %d " , 0 ) ;
dstate_setflags ( " battery.energysave.load " , ST_FLAG_RW | ST_FLAG_STRING ) ;
dstate_setaux ( " battery.energysave.load " , 3 ) ;
break ;
case PW_CONF_SLEEP_DELAY : dstate_setinfo ( " battery.energysave.delay " , " %d " , 0 ) ;
dstate_setflags ( " battery.energysave.delay " , ST_FLAG_RW | ST_FLAG_STRING ) ;
dstate_setaux ( " battery.energysave.delay " , 3 ) ;
break ;
case PW_CONF_BATT_STRINGS : dstate_setinfo ( " battery.packs " , " %d " , 0 ) ;
dstate_setflags ( " battery.packs " , ST_FLAG_RW | ST_FLAG_STRING ) ;
dstate_setaux ( " battery.packs " , 1 ) ;
break ;
}
}
}
2010-03-26 01:20:59 +02:00
void init_config ( void )
{
unsigned char answer [ PW_ANSWER_MAX_SIZE ] ;
2015-04-30 16:53:36 +03:00
int voltage = 0 , frequency = 0 , res , tmp = 0 ;
2010-03-26 01:20:59 +02:00
char sValue [ 17 ] ;
2013-11-24 17:00:12 +02:00
char sPartNumber [ 17 ] ;
2010-03-26 01:20:59 +02:00
2012-06-01 16:55:19 +03:00
res = command_read_sequence ( PW_CONFIG_BLOCK_REQ , answer ) ;
2010-03-26 01:20:59 +02:00
if ( res < = 0 )
fatal_with_errno ( EXIT_FAILURE , " Could not communicate with the ups " ) ;
/* Get validation mask for status bitmap */
bcmxcp_status . topology_mask = answer [ BCMXCP_CONFIG_BLOCK_HW_MODULES_INSTALLED_BYTE3 ] ;
/* Nominal output voltage of ups */
voltage = get_word ( ( answer + BCMXCP_CONFIG_BLOCK_NOMINAL_OUTPUT_VOLTAGE ) ) ;
if ( voltage ! = 0 )
dstate_setinfo ( " output.voltage.nominal " , " %d " , voltage ) ;
2013-11-24 17:00:12 +02:00
/* Nominal Output Frequency */
frequency = get_word ( ( answer + BCMXCP_CONFIG_BLOCK_NOMINAL_OUTPUT_FREQ ) ) ;
if ( frequency ! = 0 )
dstate_setinfo ( " output.frequency.nominal " , " %d " , frequency ) ;
2010-03-26 01:20:59 +02:00
2015-04-30 16:53:36 +03:00
/*Number of EBM*/
tmp = ( int ) * ( answer + BCMXCP_CONFIG_BLOCK_BATTERY_DATA_WORD3 ) ;
if ( tmp ! = 0 )
dstate_setinfo ( " battery.packs " , " %d " , tmp ) ;
2013-11-24 17:00:12 +02:00
/* UPS serial number */
snprintf ( sValue , sizeof ( sValue ) , " %s " , answer + BCMXCP_CONFIG_BLOCK_SERIAL_NUMBER ) ;
if ( sValue [ 0 ] ! = ' \0 ' )
dstate_setinfo ( " ups.serial " , " %s " , sValue ) ;
2015-04-30 16:53:36 +03:00
2013-11-24 17:00:12 +02:00
/* UPS Part Number*/
snprintf ( sPartNumber , sizeof ( sPartNumber ) , " %s " , answer + BCMXCP_CONFIG_BLOCK_PART_NUMBER ) ;
if ( sPartNumber [ 0 ] ! = ' \0 ' )
dstate_setinfo ( " device.part " , " %s " , sPartNumber ) ;
2010-03-26 01:20:59 +02:00
}
void init_limit ( void )
{
unsigned char answer [ PW_ANSWER_MAX_SIZE ] ;
int value , res ;
res = command_read_sequence ( PW_LIMIT_BLOCK_REQ , answer ) ;
if ( res < = 0 ) {
fatal_with_errno ( EXIT_FAILURE , " Could not communicate with the ups " ) ;
}
/* Nominal input voltage */
value = get_word ( ( answer + BCMXCP_EXT_LIMITS_BLOCK_NOMINAL_INPUT_VOLTAGE ) ) ;
if ( value ! = 0 ) {
dstate_setinfo ( " input.voltage.nominal " , " %d " , value ) ;
}
/* Nominal input frequency */
value = get_word ( ( answer + BCMXCP_EXT_LIMITS_BLOCK_NOMINAL_INPUT_FREQ ) ) ;
if ( value ! = 0 ) {
2013-11-24 17:00:12 +02:00
int fnom = value ;
2010-03-26 01:20:59 +02:00
dstate_setinfo ( " input.frequency.nominal " , " %d " , value ) ;
/* Input frequency deviation */
value = get_word ( ( answer + BCMXCP_EXT_LIMITS_BLOCK_FREQ_DEV_LIMIT ) ) ;
if ( value ! = 0 ) {
value / = 100 ;
dstate_setinfo ( " input.frequency.low " , " %d " , fnom - value ) ;
dstate_setinfo ( " input.frequency.high " , " %d " , fnom + value ) ;
}
}
/* Bypass Voltage Low Deviation Limit / Transfer to Boost Voltage */
value = get_word ( ( answer + BCMXCP_EXT_LIMITS_BLOCK_VOLTAGE_LOW_DEV_LIMIT ) ) ;
if ( value ! = 0 ) {
dstate_setinfo ( " input.transfer.boost.high " , " %d " , value ) ;
}
/* Bypass Voltage High Deviation Limit / Transfer to Buck Voltage */
value = get_word ( ( answer + BCMXCP_EXT_LIMITS_BLOCK_VOLTAGE_HIGE_DEV_LIMIT ) ) ;
if ( value ! = 0 ) {
dstate_setinfo ( " input.transfer.trim.low " , " %d " , value ) ;
}
/* Low battery warning */
bcmxcp_status . lowbatt = answer [ BCMXCP_EXT_LIMITS_BLOCK_LOW_BATT_WARNING ] * 60 ;
/* Check if we should warn the user that her shutdown delay is to long? */
if ( bcmxcp_status . shutdowndelay > bcmxcp_status . lowbatt )
upslogx ( LOG_WARNING , " Shutdown delay longer than battery capacity when Low Battery warning is given. (max %d seconds) " , bcmxcp_status . lowbatt ) ;
/* Horn Status: */
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_HORN_STATUS ] ;
if ( value > = 0 & & value < = 2 ) {
dstate_setinfo ( " ups.beeper.status " , " %s " , horn_stat [ value ] ) ;
}
/* Minimum Supported Input Voltage */
value = get_word ( ( answer + BCMXCP_EXT_LIMITS_BLOCK_MIN_INPUT_VOLTAGE ) ) ;
if ( value ! = 0 ) {
dstate_setinfo ( " input.transfer.low " , " %d " , value ) ;
}
/* Maximum Supported Input Voltage */
value = get_word ( ( answer + BCMXCP_EXT_LIMITS_BLOCK_MAX_INPUT_VOLTAGE ) ) ;
if ( value ! = 0 ) {
dstate_setinfo ( " input.transfer.high " , " %d " , value ) ;
}
/* Ambient Temperature Lower Alarm Limit */
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_AMBIENT_TEMP_LOW ] ;
if ( value ! = 0 ) {
dstate_setinfo ( " ambient.temperature.low " , " %d " , value ) ;
}
2013-11-24 17:00:12 +02:00
/* Ambient Temperature Upper Alarm Limit */
2010-03-26 01:20:59 +02:00
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_AMBIENT_TEMP_HIGE ] ;
if ( value ! = 0 ) {
dstate_setinfo ( " ambient.temperature.high " , " %d " , value ) ;
}
2015-04-30 16:53:36 +03:00
/*Sleep minimum load*/
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_SLEEP_TH_LOAD ] ;
if ( value ! = 0 ) {
dstate_setinfo ( " battery.energysave.load " , " %d " , value ) ;
}
/* Sleep delay*/
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_SLEEP_DELAY ] ;
if ( value ! = 0 ) {
dstate_setinfo ( " battery.energysave.delay " , " %d " , value ) ;
}
/* Low batt minutes warning*/
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_LOW_BATT_WARNING ] ;
if ( value ! = 0 ) {
dstate_setinfo ( " battery.runtime.low " , " %d " , value ) ;
}
/* Return to mains delay */
value = get_word ( answer + BCMXCP_EXT_LIMITS_BLOCK_RETURN_STAB_DELAY ) ;
if ( value ! = 0 ) {
dstate_setinfo ( " input.transfer.delay " , " %d " , value ) ;
}
/* Minimum return capacity*/
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_BATT_CAPACITY_RETURN ] ;
if ( value ! = 0 ) {
dstate_setinfo ( " battery.charge.restart " , " %d " , value ) ;
}
2010-03-26 01:20:59 +02:00
}
2013-11-24 17:00:12 +02:00
void init_topology ( void )
{
unsigned char answer [ PW_ANSWER_MAX_SIZE ] ;
const char * nutvalue ;
int res , value ;
res = command_read_sequence ( PW_UPS_TOP_DATA_REQ , answer ) ;
if ( res < = 0 )
fatal_with_errno ( EXIT_FAILURE , " Could not communicate with the ups " ) ;
value = get_word ( answer ) ;
if ( ( nutvalue = nut_find_infoval ( topology_info , value , TRUE ) ) ! = NULL ) {
dstate_setinfo ( " ups.description " , " %s " , nutvalue ) ;
}
}
void init_system_test_capabilities ( void )
{
unsigned char answer [ PW_ANSWER_MAX_SIZE ] , cbuf [ 5 ] ;
const char * nutvalue ;
int res , value , i ;
/* Query what system test capabilities are supported */
send_write_command ( AUTHOR , 4 ) ;
2015-04-30 16:53:36 +03:00
2013-11-24 17:00:12 +02:00
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
cbuf [ 0 ] = PW_INIT_SYS_TEST ;
cbuf [ 1 ] = PW_SYS_TEST_REPORT_CAPABILITIES ;
res = command_write_sequence ( cbuf , 2 , answer ) ;
if ( res < = 0 ) {
upslogx ( LOG_ERR , " Short read from UPS " ) ;
return ;
}
if ( ( unsigned char ) answer [ 0 ] ! = BCMXCP_RETURN_ACCEPTED ) {
upsdebugx ( 2 , " System test capabilities list not supported " ) ;
return ;
}
/* Add instcmd for system test capabilities */
for ( i = 3 ; i < res ; i + + ) {
value = answer [ i ] ;
if ( ( nutvalue = nut_find_infoval ( system_test_info , value , TRUE ) ) ! = NULL ) {
upsdebugx ( 2 , " Added support for instcmd %s " , nutvalue ) ;
dstate_addcmd ( nutvalue ) ;
}
}
}
2010-03-26 01:20:59 +02:00
void upsdrv_initinfo ( void )
{
unsigned char answer [ PW_ANSWER_MAX_SIZE ] ;
char * pTmp ;
char outlet_name [ 27 ] ;
char power_rating [ 10 ] ;
int iRating = 0 , iIndex = 0 , res , len ;
int ncpu = 0 , buf ;
2013-11-24 17:00:12 +02:00
int conf_block_len = 0 , alarm_block_len = 0 , cmd_list_len = 0 , topology_block_len = 0 ;
bool_t got_cmd_list = FALSE ;
/* Init BCM/XCP command descriptions */
init_command_map ( ) ;
2010-03-26 01:20:59 +02:00
/* Init BCM/XCP alarm descriptions */
init_alarm_map ( ) ;
/* Get vars from ups.conf */
if ( getval ( " shutdown_delay " ) ! = NULL )
bcmxcp_status . shutdowndelay = atoi ( getval ( " shutdown_delay " ) ) ;
else
bcmxcp_status . shutdowndelay = 120 ;
/* Get information on UPS from UPS ID block */
res = command_read_sequence ( PW_ID_BLOCK_REQ , answer ) ;
if ( res < = 0 )
fatal_with_errno ( EXIT_FAILURE , " Could not communicate with the ups " ) ;
/* Get number of CPU's in ID block */
len = answer [ iIndex + + ] ;
buf = len * 11 ;
pTmp = xmalloc ( buf + 1 ) ;
pTmp [ 0 ] = 0 ;
/* If there is one or more CPU number, get it */
if ( len > 0 ) {
do {
if ( ( answer [ iIndex ] ! = 0x00 ) | | ( answer [ iIndex + 1 ] ! = 0x00 ) ) {
/* Get the ups firmware. The major number is in the last byte, the minor is in the first */
snprintfcat ( pTmp , buf + 1 , " %s%02x.%02x " , cpu_name [ ncpu ] , answer [ iIndex + 1 ] , answer [ iIndex ] ) ;
}
iIndex + = 2 ;
len - - ;
ncpu + + ;
} while ( ( len > 0 ) & & ( ncpu < = 5 ) ) ;
dstate_setinfo ( " ups.firmware " , " %s " , pTmp ) ;
/* Increment index to point at end of CPU bytes. */
iIndex + = len * 2 ;
}
2012-06-01 16:55:19 +03:00
free ( pTmp ) ;
2010-03-26 01:20:59 +02:00
/* Get rating in kVA, if present */
if ( ( iRating = answer [ iIndex + + ] ) > 0 )
iRating * = 1000 ;
else
{
/* The rating is given as 2 byte VA */
iRating = get_word ( answer + iIndex ) * 50 ;
iIndex + = 2 ;
}
dstate_setinfo ( " ups.power.nominal " , " %d " , iRating ) ;
/* Get information on Phases from UPS */
nphases = ( answer [ iIndex + + ] ) ;
dstate_setinfo ( " output.phases " , " %d " , nphases ) ;
/* Init BCM/XCP <-> NUT meter map */
init_meter_map ( ) ;
2013-11-24 17:00:12 +02:00
/* Skip UPS' phase angle, as NUT do not care */
2010-03-26 01:20:59 +02:00
iIndex + = 1 ;
2013-11-24 17:00:12 +02:00
/* Set manufacturer name */
2010-03-26 01:20:59 +02:00
dstate_setinfo ( " ups.mfr " , " Eaton " ) ;
/* Get length of UPS description */
len = answer [ iIndex + + ] ;
/* Extract and reformat the model string */
pTmp = xmalloc ( len + 15 ) ;
snprintf ( pTmp , len + 1 , " %s " , answer + iIndex ) ;
pTmp [ len + 1 ] = 0 ;
iIndex + = len ;
/* power rating in the model name is in the form "<rating>i"
* ie " 1500i " , " 500i " , . . .
* some models already includes it , so check to avoid duplication */
snprintf ( power_rating , sizeof ( power_rating ) , " %ii " , iRating ) ;
if ( strstr ( pTmp , power_rating ) = = NULL ) {
snprintfcat ( pTmp , len + 10 , " %s " , power_rating ) ;
}
2016-07-18 03:11:41 +03:00
dstate_setinfo ( " ups.model " , " %s " , str_rtrim ( pTmp , ' ' ) ) ;
2010-03-26 01:20:59 +02:00
free ( pTmp ) ;
/* Get meter map info from ups, and init our map */
len = answer [ iIndex + + ] ;
upsdebugx ( 2 , " Length of meter map: %d \n " , len ) ;
init_ups_meter_map ( answer + iIndex , len ) ;
iIndex + = len ;
/* Next is alarm map */
len = answer [ iIndex + + ] ;
upsdebugx ( 2 , " Length of alarm map: %d \n " , len ) ;
init_ups_alarm_map ( answer + iIndex , len ) ;
iIndex + = len ;
/* Then the Config_block_length */
conf_block_len = get_word ( answer + iIndex ) ;
upsdebugx ( 2 , " Length of Config_block: %d \n " , conf_block_len ) ;
iIndex + = 2 ;
/* Next is statistics map */
len = answer [ iIndex + + ] ;
upsdebugx ( 2 , " Length of statistics map: %d \n " , len ) ;
/* init_statistics_map(answer+iIndex, len); */
iIndex + = len ;
/* Size of the alarm history log */
2013-11-24 17:00:12 +02:00
len = get_word ( answer + iIndex ) ;
upsdebugx ( 2 , " Length of alarm history log: %d \n " , len ) ;
2010-03-26 01:20:59 +02:00
iIndex + = 2 ;
2013-11-24 17:00:12 +02:00
/* Size of custom event log, always 0 according to spec */
2010-03-26 01:20:59 +02:00
iIndex + = 2 ;
/* Size of topology block */
2013-11-24 17:00:12 +02:00
topology_block_len = get_word ( answer + iIndex ) ;
upsdebugx ( 2 , " Length of topology block: %d \n " , topology_block_len ) ;
2010-03-26 01:20:59 +02:00
iIndex + = 2 ;
/* Maximum supported command length */
2013-11-24 17:00:12 +02:00
len = answer [ iIndex + + ] ;
upsdebugx ( 2 , " Length of max supported command length: %d \n " , len ) ;
2010-03-26 01:20:59 +02:00
/* Size of command list block */
2011-06-01 23:31:49 +03:00
if ( iIndex < res )
cmd_list_len = get_word ( answer + iIndex ) ;
2010-03-26 01:20:59 +02:00
upsdebugx ( 2 , " Length of command list: %d \n " , cmd_list_len ) ;
iIndex + = 2 ;
/* Size of outlet monitoring block */
2011-06-01 23:31:49 +03:00
if ( iIndex < res )
outlet_block_len = get_word ( answer + iIndex ) ;
2010-03-26 01:20:59 +02:00
upsdebugx ( 2 , " Length of outlet_block: %d \n " , outlet_block_len ) ;
iIndex + = 2 ;
/* Size of the alarm block */
2011-06-01 23:31:49 +03:00
if ( iIndex < res )
alarm_block_len = get_word ( answer + iIndex ) ;
2010-03-26 01:20:59 +02:00
upsdebugx ( 2 , " Length of alarm_block: %d \n " , alarm_block_len ) ;
/* End of UPS ID block request */
/* Due to a bug in PW5115 firmware, we need to use blocklength > 8.
The protocol state that outlet block is only implemented if there is
at least 2 outlet block . 5115 has only one outlet , but has outlet block ! */
if ( outlet_block_len > 8 ) {
len = init_outlet ( outlet_block_len ) ;
for ( res = 1 ; res < = len ; res + + ) {
snprintf ( outlet_name , sizeof ( outlet_name ) - 1 , " outlet.%d.shutdown.return " , res ) ;
dstate_addcmd ( outlet_name ) ;
2015-04-30 16:53:36 +03:00
snprintf ( outlet_name , sizeof ( outlet_name ) - 1 , " outlet.%d.load.on " , res ) ;
dstate_addcmd ( outlet_name ) ;
snprintf ( outlet_name , sizeof ( outlet_name ) - 1 , " outlet.%d.load.off " , res ) ;
dstate_addcmd ( outlet_name ) ;
2010-03-26 01:20:59 +02:00
}
}
/* Get information on UPS configuration */
init_config ( ) ;
/* Get information on UPS extended limits */
init_limit ( ) ;
/* Get information on UPS commands */
2011-06-01 23:31:49 +03:00
if ( cmd_list_len )
2013-11-24 17:00:12 +02:00
got_cmd_list = init_command ( cmd_list_len ) ;
/* Add default commands if we were not able to query UPS for support */
if ( got_cmd_list = = FALSE ) {
dstate_addcmd ( " shutdown.return " ) ;
dstate_addcmd ( " shutdown.stayoff " ) ;
dstate_addcmd ( " test.battery.start " ) ;
}
/* Get information on UPS topology */
if ( topology_block_len )
init_topology ( ) ;
/* Get information on system test capabilities */
if ( bcmxcp_command_map [ PW_INIT_SYS_TEST ] . command_byte > 0 ) {
init_system_test_capabilities ( ) ;
}
2015-04-30 16:53:36 +03:00
/* Get information about configurable external variables*/
init_ext_vars ( ) ;
2010-03-26 01:20:59 +02:00
upsh . instcmd = instcmd ;
2013-11-24 17:00:12 +02:00
upsh . setvar = setvar ;
2010-03-26 01:20:59 +02:00
}
void upsdrv_updateinfo ( void )
{
unsigned char answer [ PW_ANSWER_MAX_SIZE ] ;
2013-11-24 17:00:12 +02:00
unsigned char status , topology ;
2010-03-26 01:20:59 +02:00
char sValue [ 128 ] ;
2015-04-30 16:53:36 +03:00
int iIndex , res , value ;
2013-11-24 17:00:12 +02:00
bool_t has_ups_load = FALSE ;
2012-06-01 16:55:19 +03:00
int batt_status = 0 ;
2013-11-24 17:00:12 +02:00
const char * nutvalue ;
float calculated_load ;
2010-03-26 01:20:59 +02:00
/* Get info from UPS */
res = command_read_sequence ( PW_METER_BLOCK_REQ , answer ) ;
if ( res < = 0 ) {
upslogx ( LOG_ERR , " Short read from UPS " ) ;
dstate_datastale ( ) ;
return ;
}
/* Loop thru meter map, get all data UPS is willing to offer */
for ( iIndex = 0 ; iIndex < BCMXCP_METER_MAP_MAX ; iIndex + + ) {
if ( bcmxcp_meter_map [ iIndex ] . format ! = 0 & & bcmxcp_meter_map [ iIndex ] . nut_entity ! = NULL ) {
decode_meter_map_entry ( answer + bcmxcp_meter_map [ iIndex ] . meter_block_index ,
bcmxcp_meter_map [ iIndex ] . format , sValue ) ;
/* Set result */
dstate_setinfo ( bcmxcp_meter_map [ iIndex ] . nut_entity , " %s " , sValue ) ;
2013-11-24 17:00:12 +02:00
/* Check if we read ups.load */
if ( has_ups_load = = FALSE & & ! strcasecmp ( bcmxcp_meter_map [ iIndex ] . nut_entity , " ups.load " ) ) {
has_ups_load = TRUE ;
}
}
2010-03-26 01:20:59 +02:00
}
2013-11-24 17:00:12 +02:00
/* Calculate ups.load if UPS does not report it directly */
if ( has_ups_load = = FALSE ) {
calculated_load = calculate_ups_load ( answer ) ;
if ( calculated_load > = 0.0f ) {
dstate_setinfo ( " ups.load " , " %5.1f " , calculated_load ) ;
}
2010-03-26 01:20:59 +02:00
}
/* Due to a bug in PW5115 firmware, we need to use blocklength > 8.
The protocol state that outlet block is only implemented if there is
at least 2 outlet block . 5115 has only one outlet , but has outlet block . */
if ( outlet_block_len > 8 ) {
init_outlet ( outlet_block_len ) ;
}
/* Get alarm info from UPS */
res = command_read_sequence ( PW_CUR_ALARM_REQ , answer ) ;
if ( res < = 0 ) {
upslogx ( LOG_ERR , " Short read from UPS " ) ;
dstate_datastale ( ) ;
return ;
}
else
{
bcmxcp_status . alarm_on_battery = 0 ;
bcmxcp_status . alarm_low_battery = 0 ;
2013-11-24 17:00:12 +02:00
/* Set alarms */
2010-03-26 01:20:59 +02:00
alarm_init ( ) ;
/* Loop thru alarm map, get all alarms UPS is willing to offer */
for ( iIndex = 0 ; iIndex < BCMXCP_ALARM_MAP_MAX ; iIndex + + ) {
if ( bcmxcp_alarm_map [ iIndex ] . alarm_block_index > = 0 & & bcmxcp_alarm_map [ iIndex ] . alarm_desc ! = NULL ) {
2013-11-24 17:00:12 +02:00
if ( answer [ bcmxcp_alarm_map [ iIndex ] . alarm_block_index ] > 0 ) {
2010-03-26 01:20:59 +02:00
alarm_set ( bcmxcp_alarm_map [ iIndex ] . alarm_desc ) ;
if ( iIndex = = BCMXCP_ALARM_UPS_ON_BATTERY ) {
bcmxcp_status . alarm_on_battery = 1 ;
}
2013-11-24 17:00:12 +02:00
else if ( iIndex = = BCMXCP_ALARM_BATTERY_LOW ) {
2010-03-26 01:20:59 +02:00
bcmxcp_status . alarm_low_battery = 1 ;
}
2013-11-24 17:00:12 +02:00
else if ( iIndex = = BCMXCP_ALARM_BATTERY_TEST_FAILED ) {
2012-06-01 16:55:19 +03:00
bcmxcp_status . alarm_replace_battery = 1 ;
}
2013-11-24 17:00:12 +02:00
else if ( iIndex = = BCMXCP_ALARM_BATTERY_NEEDS_SERVICE ) {
2012-06-01 16:55:19 +03:00
bcmxcp_status . alarm_replace_battery = 1 ;
}
2010-03-26 01:20:59 +02:00
}
}
}
2013-11-24 17:00:12 +02:00
/* Confirm alarms */
2010-03-26 01:20:59 +02:00
alarm_commit ( ) ;
}
/* Get status info from UPS */
res = command_read_sequence ( PW_STATUS_REQ , answer ) ;
if ( res < = 0 )
{
upslogx ( LOG_ERR , " Short read from UPS " ) ;
dstate_datastale ( ) ;
return ;
}
else
{
/* Get overall status */
memcpy ( & status , answer , sizeof ( status ) ) ;
/* Get topology status bitmap, validate */
memcpy ( & topology , answer + 1 , sizeof ( topology ) ) ;
topology & = bcmxcp_status . topology_mask ;
/* Set status */
status_init ( ) ;
switch ( status ) {
2013-11-24 17:00:12 +02:00
case BCMXCP_STATUS_ONLINE : /* On line, everything is fine */
2010-03-26 01:20:59 +02:00
status_set ( " OL " ) ;
break ;
2013-11-24 17:00:12 +02:00
case BCMXCP_STATUS_ONBATTERY : /* Off line */
2010-03-26 01:20:59 +02:00
if ( bcmxcp_status . alarm_on_battery = = 0 )
status_set ( " OB " ) ;
break ;
2013-11-24 17:00:12 +02:00
case BCMXCP_STATUS_OVERLOAD : /* Overload */
2010-03-26 01:20:59 +02:00
status_set ( " OL " ) ;
status_set ( " OVER " ) ;
break ;
2013-11-24 17:00:12 +02:00
case BCMXCP_STATUS_TRIM : /* Trim */
2010-03-26 01:20:59 +02:00
status_set ( " OL " ) ;
status_set ( " TRIM " ) ;
break ;
2013-11-24 17:00:12 +02:00
case BCMXCP_STATUS_BOOST1 :
case BCMXCP_STATUS_BOOST2 : /* Boost */
2010-03-26 01:20:59 +02:00
status_set ( " OL " ) ;
status_set ( " BOOST " ) ;
break ;
2013-11-24 17:00:12 +02:00
case BCMXCP_STATUS_BYPASS : /* Bypass */
2010-03-26 01:20:59 +02:00
status_set ( " OL " ) ;
status_set ( " BYPASS " ) ;
break ;
2013-11-24 17:00:12 +02:00
case BCMXCP_STATUS_OFF : /* Mostly off */
2010-03-26 01:20:59 +02:00
status_set ( " OFF " ) ;
break ;
default : /* Unknown, assume it is OK... */
status_set ( " OL " ) ;
break ;
}
/* We might have to modify status based on topology status */
if ( ( topology & 0x20 ) & & bcmxcp_status . alarm_low_battery = = 0 )
status_set ( " LB " ) ;
/* And finally, we might need to modify status based on alarms - the most correct way */
if ( bcmxcp_status . alarm_on_battery )
status_set ( " OB " ) ;
if ( bcmxcp_status . alarm_low_battery )
status_set ( " LB " ) ;
2012-06-01 16:55:19 +03:00
if ( bcmxcp_status . alarm_replace_battery )
status_set ( " RB " ) ;
2010-03-26 01:20:59 +02:00
status_commit ( ) ;
}
2012-06-01 16:55:19 +03:00
/* Get battery info from UPS, if exist */
res = command_read_sequence ( PW_BATTERY_REQ , answer ) ;
if ( res < = 0 )
{
upsdebugx ( 1 , " Failed to read Battery Status from UPS " ) ;
}
else
{
/* Only parse the status (first byte)
* Powerware 5115 RM output :
* 02 00 78 1 d 42 00 e0 17 42 1 e 00 00 00 00 00 00 00 00 00 01 03
* Powerware 9130 output :
* 03 0 a d7 25 42 0 a d7 25 42 00 9 a 19 6 d 43 cd cc 4 c 3 e 01 00 01 03
*/
upsdebug_hex ( 2 , " Battery Status " , answer , res ) ;
2015-04-30 16:53:36 +03:00
batt_status = answer [ BCMXCP_BATTDATA_BLOCK_BATT_TEST_STATUS ] ;
2012-06-01 16:55:19 +03:00
2013-11-24 17:00:12 +02:00
if ( ( nutvalue = nut_find_infoval ( batt_test_info , batt_status , TRUE ) ) ! = NULL ) {
2012-06-01 16:55:19 +03:00
dstate_setinfo ( " ups.test.result " , " %s " , nutvalue ) ;
upsdebugx ( 2 , " Battery Status = %s (%i) " , nutvalue , batt_status ) ;
}
else {
upsdebugx ( 1 , " Failed to extract Battery Status from answer " ) ;
}
2015-04-30 16:53:36 +03:00
/*Extracting internal batteries ABM status*/
/*Placed first in ABM statuses list. For examples above - on position BCMXCP_BATTDATA_BLOCK_NUMBER_OF_STRINGS (18):
PW5115RM - 0 - no external strings , no status bytes ,
so next byte ( 19 ) - number of ABM statuses , next ( 20 ) - first ABM Status for internal batteries .
PW9130 - 1 - one external string , so one additional status byte ( # 19 - 00 - no test run ) , next ( 20 ) - number of ABM statuses ,
next ( 21 ) - ABM Status for internal batteries .
*/
value = * ( answer + BCMXCP_BATTDATA_BLOCK_NUMBER_OF_STRINGS + * ( answer + BCMXCP_BATTDATA_BLOCK_NUMBER_OF_STRINGS ) * 1 + 2 ) ;
upsdebugx ( 2 , " ABM Status = %d " , value ) ;
if ( value > 0 & & value < 5 )
dstate_setinfo ( " battery.charger.status " , " %s " , ABMStatus [ value - 1 ] ) ;
2012-06-01 16:55:19 +03:00
}
2015-04-30 16:53:36 +03:00
res = command_read_sequence ( PW_LIMIT_BLOCK_REQ , answer ) ;
if ( res < = 0 ) {
upsdebugx ( 1 , " Failed to read EXT LIMITs from UPS " ) ;
} else
{
/* Nominal input voltage */
value = get_word ( ( answer + BCMXCP_EXT_LIMITS_BLOCK_NOMINAL_INPUT_VOLTAGE ) ) ;
if ( value ! = 0 ) {
dstate_setinfo ( " input.voltage.nominal " , " %d " , value ) ;
}
/* Bypass Voltage Low Deviation Limit / Transfer to Boost Voltage */
value = get_word ( ( answer + BCMXCP_EXT_LIMITS_BLOCK_VOLTAGE_LOW_DEV_LIMIT ) ) ;
if ( value ! = 0 ) {
dstate_setinfo ( " input.transfer.boost.high " , " %d " , value ) ;
}
/* Bypass Voltage High Deviation Limit / Transfer to Buck Voltage */
value = get_word ( ( answer + BCMXCP_EXT_LIMITS_BLOCK_VOLTAGE_HIGE_DEV_LIMIT ) ) ;
if ( value ! = 0 ) {
dstate_setinfo ( " input.transfer.trim.low " , " %d " , value ) ;
}
/* Minimum Supported Input Voltage */
value = get_word ( ( answer + BCMXCP_EXT_LIMITS_BLOCK_MIN_INPUT_VOLTAGE ) ) ;
if ( value ! = 0 ) {
dstate_setinfo ( " input.transfer.low " , " %d " , value ) ;
}
/* Maximum Supported Input Voltage */
value = get_word ( ( answer + BCMXCP_EXT_LIMITS_BLOCK_MAX_INPUT_VOLTAGE ) ) ;
if ( value ! = 0 ) {
dstate_setinfo ( " input.transfer.high " , " %d " , value ) ;
}
/* Horn Status: */
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_HORN_STATUS ] ;
if ( value > = 0 & & value < = 2 ) {
dstate_setinfo ( " ups.beeper.status " , " %s " , horn_stat [ value ] ) ;
}
/* AAmbient Temperature Upper Alarm Limit */
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_AMBIENT_TEMP_HIGE ] ;
if ( value ! = 0 ) {
dstate_setinfo ( " ambient.temperature.high " , " %d " , value ) ;
}
/*Sleep minimum load*/
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_SLEEP_TH_LOAD ] ;
if ( value ! = 0 ) {
dstate_setinfo ( " battery.energysave.load " , " %d " , value ) ;
}
/* Sleep delay*/
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_SLEEP_DELAY ] ;
if ( value ! = 0 ) {
dstate_setinfo ( " battery.energysave.delay " , " %d " , value ) ;
}
/* Low batt minutes warning*/
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_LOW_BATT_WARNING ] ;
if ( value ! = 0 ) {
dstate_setinfo ( " battery.runtime.low " , " %d " , value ) ;
}
/* Return to mains delay */
value = get_word ( answer + BCMXCP_EXT_LIMITS_BLOCK_RETURN_STAB_DELAY ) ;
if ( value ! = 0 ) {
dstate_setinfo ( " input.transfer.delay " , " %d " , value ) ;
}
/* Minimum return capacity*/
value = answer [ BCMXCP_EXT_LIMITS_BLOCK_BATT_CAPACITY_RETURN ] ;
if ( value ! = 0 ) {
dstate_setinfo ( " battery.charge.restart " , " %d " , value ) ;
}
} ;
res = command_read_sequence ( PW_CONFIG_BLOCK_REQ , answer ) ;
if ( res < = 0 ) {
upsdebugx ( 1 , " Failed to read CONF BLOCK from UPS " ) ;
}
else
{
/*Nominal output voltage*/
value = get_word ( ( answer + BCMXCP_CONFIG_BLOCK_NOMINAL_OUTPUT_VOLTAGE ) ) ;
if ( value ! = 0 )
dstate_setinfo ( " output.voltage.nominal " , " %d " , value ) ;
/*Number of EBM*/
value = ( int ) * ( answer + BCMXCP_CONFIG_BLOCK_BATTERY_DATA_WORD3 ) ;
if ( value ! = 0 )
dstate_setinfo ( " battery.packs " , " %d " , value ) ;
}
2010-03-26 01:20:59 +02:00
dstate_dataok ( ) ;
}
2013-11-24 17:00:12 +02:00
float calculate_ups_load ( const unsigned char * answer )
2010-03-26 01:20:59 +02:00
{
2013-11-24 17:00:12 +02:00
char sValue [ 128 ] ;
float output = 0 , max_output = - FLT_MAX , fValue = - FLT_MAX ;
2010-03-26 01:20:59 +02:00
2013-11-24 17:00:12 +02:00
if ( bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VA ] . format ! = 0 & & /* Output VA */
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VA_BAR_CHART ] . format ! = 0 ) /* Max output VA */
{
decode_meter_map_entry ( answer + bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VA ] . meter_block_index ,
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VA ] . format , sValue ) ;
output = atof ( sValue ) ;
decode_meter_map_entry ( answer + bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VA_BAR_CHART ] . meter_block_index ,
bcmxcp_meter_map [ BCMXCP_METER_MAP_OUTPUT_VA_BAR_CHART ] . format , sValue ) ;
max_output = atof ( sValue ) ;
}
else if ( bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_A ] . format ! = 0 & & /* Output A */
bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_A_BAR_CHART ] . format ! = 0 ) /* Max output A */
{
decode_meter_map_entry ( answer + bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_A ] . meter_block_index ,
bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_A ] . format , sValue ) ;
output = atof ( sValue ) ;
decode_meter_map_entry ( answer + bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_A_BAR_CHART ] . meter_block_index ,
bcmxcp_meter_map [ BCMXCP_METER_MAP_LOAD_CURRENT_PHASE_A_BAR_CHART ] . format , sValue ) ;
max_output = atof ( sValue ) ;
}
if ( max_output > 0.0 )
fValue = 100 * ( output / max_output ) ;
2010-03-26 01:20:59 +02:00
2013-11-24 17:00:12 +02:00
return fValue ;
}
2010-03-26 01:20:59 +02:00
2013-11-24 17:00:12 +02:00
void upsdrv_shutdown ( void )
{
upsdebugx ( 1 , " upsdrv_shutdown... " ) ;
2010-03-26 01:20:59 +02:00
2013-11-24 17:00:12 +02:00
/* Try to shutdown with delay */
if ( instcmd ( " shutdown.return " , NULL ) = = STAT_INSTCMD_HANDLED ) {
/* Shutdown successful */
2010-03-26 01:20:59 +02:00
return ;
}
2013-11-24 17:00:12 +02:00
/* If the above doesn't work, try shutdown.stayoff */
if ( instcmd ( " shutdown.stayoff " , NULL ) = = STAT_INSTCMD_HANDLED ) {
/* Shutdown successful */
return ;
2010-03-26 01:20:59 +02:00
}
2013-11-24 17:00:12 +02:00
fatalx ( EXIT_FAILURE , " Shutdown failed! " ) ;
2010-03-26 01:20:59 +02:00
}
static int instcmd ( const char * cmdname , const char * extra )
{
2011-09-29 21:14:46 +03:00
unsigned char answer [ 128 ] , cbuf [ 6 ] ;
2013-11-24 17:00:12 +02:00
char success_msg [ 40 ] ;
2015-04-30 16:53:36 +03:00
char namebuf [ MAX_NUT_NAME_LENGTH ] ;
2011-09-29 21:14:46 +03:00
char varname [ 32 ] ;
const char * varvalue = NULL ;
2015-04-30 16:53:36 +03:00
int res , sec , outlet_num ;
2013-11-24 17:00:12 +02:00
int sddelay = 0x03 ; /* outlet off in 3 seconds, by default */
2010-03-26 01:20:59 +02:00
upsdebugx ( 1 , " entering instcmd(%s) " , cmdname ) ;
2015-04-30 16:53:36 +03:00
if ( ! strcasecmp ( cmdname , " shutdown.return " ) ) {
2010-03-26 01:20:59 +02:00
send_write_command ( AUTHOR , 4 ) ;
2013-11-24 17:00:12 +02:00
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
2010-03-26 01:20:59 +02:00
cbuf [ 0 ] = PW_LOAD_OFF_RESTART ;
2015-04-30 16:53:36 +03:00
cbuf [ 1 ] = ( unsigned char ) ( bcmxcp_status . shutdowndelay & 0x00ff ) ; /* "delay" sec delay for shutdown, */
cbuf [ 2 ] = ( unsigned char ) ( bcmxcp_status . shutdowndelay > > 8 ) ; /* high byte sec. From ups.conf. */
2010-03-26 01:20:59 +02:00
2015-04-30 16:53:36 +03:00
res = command_write_sequence ( cbuf , 3 , answer ) ;
2010-03-26 01:20:59 +02:00
sec = ( 256 * ( unsigned char ) answer [ 3 ] ) + ( unsigned char ) answer [ 2 ] ;
2013-11-24 17:00:12 +02:00
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " Going down in %d sec " , sec ) ;
2010-03-26 01:20:59 +02:00
2013-11-24 17:00:12 +02:00
return decode_instcmd_exec ( res , ( unsigned char ) answer [ 0 ] , cmdname , success_msg ) ;
2015-04-30 16:53:36 +03:00
}
2010-03-26 01:20:59 +02:00
2015-04-30 16:53:36 +03:00
if ( ! strcasecmp ( cmdname , " shutdown.stayoff " ) ) {
2010-03-26 01:20:59 +02:00
send_write_command ( AUTHOR , 4 ) ;
2013-11-24 17:00:12 +02:00
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
2010-03-26 01:20:59 +02:00
2015-04-30 16:53:36 +03:00
res = command_read_sequence ( PW_UPS_OFF , answer ) ;
2010-03-26 01:20:59 +02:00
2015-04-30 16:53:36 +03:00
return decode_instcmd_exec ( res , ( unsigned char ) answer [ 0 ] , cmdname , " Going down NOW " ) ;
}
2010-03-26 01:20:59 +02:00
2015-04-30 16:53:36 +03:00
if ( ! strcasecmp ( cmdname , " load.on " ) ) {
send_write_command ( AUTHOR , 4 ) ;
2010-03-26 01:20:59 +02:00
2015-04-30 16:53:36 +03:00
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
res = command_read_sequence ( PW_UPS_ON , answer ) ;
return decode_instcmd_exec ( res , ( unsigned char ) answer [ 0 ] , cmdname , " Enabling " ) ;
2010-03-26 01:20:59 +02:00
}
2015-04-30 16:53:36 +03:00
if ( ! strcasecmp ( cmdname , " bypass.start " ) ) {
2010-03-26 01:20:59 +02:00
send_write_command ( AUTHOR , 4 ) ;
2013-11-24 17:00:12 +02:00
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
2010-03-26 01:20:59 +02:00
2015-04-30 16:53:36 +03:00
res = command_read_sequence ( PW_GO_TO_BYPASS , answer ) ;
2010-03-26 01:20:59 +02:00
2015-04-30 16:53:36 +03:00
return decode_instcmd_exec ( res , ( unsigned char ) answer [ 0 ] , cmdname , " Bypass enabled " ) ;
2010-03-26 01:20:59 +02:00
}
2012-06-01 16:55:19 +03:00
/* Note: test result will be parsed from Battery status block,
* part of the update loop , and published into ups . test . result
*/
2010-03-26 01:20:59 +02:00
if ( ! strcasecmp ( cmdname , " test.battery.start " ) ) {
send_write_command ( AUTHOR , 4 ) ;
2013-11-24 17:00:12 +02:00
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
2010-03-26 01:20:59 +02:00
cbuf [ 0 ] = PW_INIT_BAT_TEST ;
2013-11-24 17:00:12 +02:00
cbuf [ 1 ] = 0x0A ; /* 10 sec start delay for test.*/
cbuf [ 2 ] = 0x1E ; /* 30 sec test duration.*/
2010-03-26 01:20:59 +02:00
res = command_write_sequence ( cbuf , 3 , answer ) ;
2013-11-24 17:00:12 +02:00
return decode_instcmd_exec ( res , ( unsigned char ) answer [ 0 ] , cmdname , " Testing battery now " ) ;
/* Get test info from UPS ?
Should we wait for 50 sec and get the
answer from the test .
Or return , as we may lose line power
and need to do a shutdown . */
2015-04-30 16:53:36 +03:00
}
2013-11-24 17:00:12 +02:00
if ( ! strcasecmp ( cmdname , " test.system.start " ) ) {
send_write_command ( AUTHOR , 4 ) ;
2015-04-30 16:53:36 +03:00
2013-11-24 17:00:12 +02:00
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
2010-03-26 01:20:59 +02:00
2013-11-24 17:00:12 +02:00
cbuf [ 0 ] = PW_INIT_SYS_TEST ;
cbuf [ 1 ] = PW_SYS_TEST_GENERAL ;
res = command_write_sequence ( cbuf , 2 , answer ) ;
return decode_instcmd_exec ( res , ( unsigned char ) answer [ 0 ] , cmdname , " Testing system now " ) ;
}
if ( ! strcasecmp ( cmdname , " test.panel.start " ) ) {
send_write_command ( AUTHOR , 4 ) ;
2015-04-30 16:53:36 +03:00
2013-11-24 17:00:12 +02:00
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
cbuf [ 0 ] = PW_INIT_SYS_TEST ;
cbuf [ 1 ] = PW_SYS_TEST_FLASH_LIGHTS ;
cbuf [ 2 ] = 0x0A ; /* Flash and beep 10 times */
res = command_write_sequence ( cbuf , 3 , answer ) ;
return decode_instcmd_exec ( res , ( unsigned char ) answer [ 0 ] , cmdname , " Testing panel now " ) ;
2015-04-30 16:53:36 +03:00
}
if ( ! strcasecmp ( cmdname , " beeper.disable " ) | | ! strcasecmp ( cmdname , " beeper.enable " ) | | ! strcasecmp ( cmdname , " beeper.mute " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
cbuf [ 0 ] = PW_SET_CONF_COMMAND ;
cbuf [ 1 ] = PW_CONF_BEEPER ;
switch ( cmdname [ 7 ] ) {
case ' d ' :
case ' D ' : {
cbuf [ 2 ] = 0x0 ; /*disable beeper*/
break ;
}
case ' e ' :
case ' E ' : {
cbuf [ 2 ] = 0x1 ; /*enable beeper*/
break ;
}
case ' m ' :
case ' M ' : {
cbuf [ 2 ] = 0x2 ;
break ; /*mute beeper*/
}
}
cbuf [ 3 ] = 0x0 ; /*padding*/
res = command_write_sequence ( cbuf , 4 , answer ) ;
return decode_instcmd_exec ( res , ( unsigned char ) answer [ 0 ] , cmdname , " Beeper status changed " ) ;
}
strncpy ( namebuf , cmdname , sizeof ( namebuf ) ) ;
namebuf [ NUT_OUTLET_POSITION ] = ' n ' ; /* Assumes a maximum of 9 outlets */
if ( ! strcasecmp ( namebuf , " outlet.n.shutdown.return " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
/* Get the shutdown delay, if any */
snprintf ( varname , sizeof ( varname ) - 1 , " outlet.%c.delay.shutdown " , cmdname [ NUT_OUTLET_POSITION ] ) ;
if ( ( varvalue = dstate_getinfo ( varname ) ) ! = NULL ) {
sddelay = atoi ( varvalue ) ;
}
/*if -1 then use global shutdown_delay from ups.conf*/
if ( sddelay = = - 1 ) sddelay = bcmxcp_status . shutdowndelay ;
outlet_num = cmdname [ NUT_OUTLET_POSITION ] - ' 0 ' ;
if ( outlet_num < 1 | | outlet_num > 9 )
return STAT_INSTCMD_FAILED ;
cbuf [ 0 ] = PW_LOAD_OFF_RESTART ;
cbuf [ 1 ] = sddelay & 0xff ;
cbuf [ 2 ] = sddelay > > 8 ; /* high byte of the 2 byte time argument */
cbuf [ 3 ] = outlet_num ; /* which outlet load segment? Assumes outlet number at position 8 of the command string. */
res = command_write_sequence ( cbuf , 4 , answer ) ;
sec = ( 256 * ( unsigned char ) answer [ 3 ] ) + ( unsigned char ) answer [ 2 ] ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " Going down in %d sec " , sec ) ;
return decode_instcmd_exec ( res , ( unsigned char ) answer [ 0 ] , cmdname , success_msg ) ;
}
if ( ! strcasecmp ( namebuf , " outlet.n.load.on " ) | | ! strcasecmp ( namebuf , " outlet.n.load.off " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
outlet_num = cmdname [ NUT_OUTLET_POSITION ] - ' 0 ' ;
if ( outlet_num < 1 | | outlet_num > 9 )
return STAT_INSTCMD_FAILED ;
cbuf [ 0 ] = ( cmdname [ NUT_OUTLET_POSITION + 8 ] = = ' n ' ) ? PW_UPS_ON : PW_UPS_OFF ; /* Cmd oN or not*/
cbuf [ 1 ] = outlet_num ; /* Outlet number */
res = command_write_sequence ( cbuf , 2 , answer ) ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " Outlet %d is %s " , outlet_num , ( cmdname [ NUT_OUTLET_POSITION + 8 ] = = ' n ' ) ? " On " : " Off " ) ;
return decode_instcmd_exec ( res , ( unsigned char ) answer [ 0 ] , cmdname , success_msg ) ;
2010-03-26 01:20:59 +02:00
}
upslogx ( LOG_NOTICE , " instcmd: unknown command [%s] " , cmdname ) ;
return STAT_INSTCMD_UNKNOWN ;
}
2013-11-24 17:00:12 +02:00
static int decode_instcmd_exec ( const int res , const unsigned char exec_status , const char * cmdname , const char * success_msg )
{
if ( res < = 0 ) {
upslogx ( LOG_ERR , " [%s] Short read from UPS " , cmdname ) ;
dstate_datastale ( ) ;
return STAT_INSTCMD_FAILED ;
}
/* Decode the status code from command execution */
switch ( exec_status ) {
case BCMXCP_RETURN_ACCEPTED : {
upslogx ( LOG_NOTICE , " [%s] %s " , cmdname , success_msg ) ;
upsdrv_comm_good ( ) ;
return STAT_INSTCMD_HANDLED ;
break ;
}
case BCMXCP_RETURN_ACCEPTED_PARAMETER_ADJUST : {
upslogx ( LOG_NOTICE , " [%s] Parameter adjusted " , cmdname ) ;
upslogx ( LOG_NOTICE , " [%s] %s " , cmdname , success_msg ) ;
upsdrv_comm_good ( ) ;
return STAT_INSTCMD_HANDLED ;
break ;
}
case BCMXCP_RETURN_BUSY : {
upslogx ( LOG_NOTICE , " [%s] Busy or disbled by front panel " , cmdname ) ;
return STAT_INSTCMD_FAILED ;
break ;
}
case BCMXCP_RETURN_UNRECOGNISED : {
upslogx ( LOG_NOTICE , " [%s] Unrecognised command byte or corrupt checksum " , cmdname ) ;
return STAT_INSTCMD_FAILED ;
break ;
}
case BCMXCP_RETURN_INVALID_PARAMETER : {
upslogx ( LOG_NOTICE , " [%s] Invalid parameter " , cmdname ) ;
return STAT_INSTCMD_INVALID ;
break ;
}
case BCMXCP_RETURN_PARAMETER_OUT_OF_RANGE : {
upslogx ( LOG_NOTICE , " [%s] Parameter out of range " , cmdname ) ;
return STAT_INSTCMD_INVALID ;
break ;
}
default : {
upslogx ( LOG_NOTICE , " [%s] Not supported " , cmdname ) ;
return STAT_INSTCMD_INVALID ;
break ;
}
}
}
2010-03-26 01:20:59 +02:00
void upsdrv_help ( void )
{
}
/* list flags and values that you want to receive via -x */
void upsdrv_makevartable ( void )
{
addvar ( VAR_VALUE , " shutdown_delay " , " Specify shutdown delay (seconds) " ) ;
addvar ( VAR_VALUE , " baud_rate " , " Specify communication speed (ex: 9600) " ) ;
}
2011-09-29 21:14:46 +03:00
int setvar ( const char * varname , const char * val )
{
unsigned char answer [ 128 ] , cbuf [ 5 ] ;
char namebuf [ MAX_NUT_NAME_LENGTH ] ;
2015-04-30 16:53:36 +03:00
char success_msg [ 50 ] ;
int res , sec , outlet_num , tmp ;
2011-09-29 21:14:46 +03:00
int onOff_setting = PW_AUTO_OFF_DELAY ;
upsdebugx ( 1 , " entering setvar(%s, %s) " , varname , val ) ;
2015-04-30 16:53:36 +03:00
if ( ! strcasecmp ( varname , " input.transfer.boost.high " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
tmp = atoi ( val ) ;
if ( tmp < 0 | | tmp > 460 ) {
return STAT_SET_INVALID ;
}
cbuf [ 0 ] = PW_SET_CONF_COMMAND ;
cbuf [ 1 ] = PW_CONF_LOW_DEV_LIMIT ;
cbuf [ 2 ] = tmp & 0xff ;
cbuf [ 3 ] = tmp > > 8 ;
res = command_write_sequence ( cbuf , 4 , answer ) ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " BOOST threshold volage set to %d V " , tmp ) ;
return decode_setvar_exec ( res , ( unsigned char ) answer [ 0 ] , varname , success_msg ) ;
}
if ( ! strcasecmp ( varname , " input.transfer.trim.low " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
tmp = atoi ( val ) ;
if ( tmp < 110 | | tmp > 540 ) {
return STAT_SET_INVALID ;
}
cbuf [ 0 ] = PW_SET_CONF_COMMAND ;
cbuf [ 1 ] = PW_CONF_HIGH_DEV_LIMIT ;
cbuf [ 2 ] = tmp & 0xff ;
cbuf [ 3 ] = tmp > > 8 ;
res = command_write_sequence ( cbuf , 4 , answer ) ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " TRIM threshold volage set to %d V " , tmp ) ;
return decode_setvar_exec ( res , ( unsigned char ) answer [ 0 ] , varname , success_msg ) ;
}
if ( ! strcasecmp ( varname , " battery.runtime.low " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
tmp = atoi ( val ) ;
if ( tmp < 0 | | tmp > 30 ) {
return STAT_SET_INVALID ;
}
cbuf [ 0 ] = PW_SET_CONF_COMMAND ;
cbuf [ 1 ] = PW_CONF_LOW_BATT ;
cbuf [ 2 ] = tmp ;
cbuf [ 3 ] = 0x0 ;
res = command_write_sequence ( cbuf , 4 , answer ) ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " Low battery warning time set to %d min " , tmp ) ;
return decode_setvar_exec ( res , ( unsigned char ) answer [ 0 ] , varname , success_msg ) ;
}
if ( ! strcasecmp ( varname , " input.transfer.delay " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
tmp = atoi ( val ) ;
if ( tmp < 1 | | tmp > 18000 ) {
return STAT_SET_INVALID ;
}
cbuf [ 0 ] = PW_SET_CONF_COMMAND ;
cbuf [ 1 ] = PW_CONF_RETURN_DELAY ;
cbuf [ 2 ] = tmp & 0xff ;
cbuf [ 3 ] = tmp > > 8 ;
res = command_write_sequence ( cbuf , 4 , answer ) ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " Mains return delay set to %d sec " , tmp ) ;
return decode_setvar_exec ( res , ( unsigned char ) answer [ 0 ] , varname , success_msg ) ;
}
if ( ! strcasecmp ( varname , " battery.charge.restart " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
tmp = atoi ( val ) ;
if ( tmp < 0 | | tmp > 100 ) {
return STAT_SET_INVALID ;
}
cbuf [ 0 ] = PW_SET_CONF_COMMAND ;
cbuf [ 1 ] = PW_CONF_RETURN_CAP ;
cbuf [ 2 ] = tmp & 0xff ;
cbuf [ 3 ] = 0x0 ;
res = command_write_sequence ( cbuf , 4 , answer ) ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " Mains return minimum battery capacity set to %d %% " , tmp ) ;
return decode_setvar_exec ( res , ( unsigned char ) answer [ 0 ] , varname , success_msg ) ;
}
if ( ! strcasecmp ( varname , " ambient.temperature.high " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
tmp = atoi ( val ) ;
if ( tmp < 0 | | tmp > 100 ) {
return STAT_SET_INVALID ;
}
cbuf [ 0 ] = PW_SET_CONF_COMMAND ;
cbuf [ 1 ] = PW_CONF_MAX_TEMP ;
cbuf [ 2 ] = tmp & 0xff ;
cbuf [ 3 ] = 0x0 ;
res = command_write_sequence ( cbuf , 4 , answer ) ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " Maximum temperature set to %d C " , tmp ) ;
return decode_setvar_exec ( res , ( unsigned char ) answer [ 0 ] , varname , success_msg ) ;
}
if ( ! strcasecmp ( varname , " output.voltage.nominal " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
tmp = atoi ( val ) ;
if ( tmp < 0 | | tmp > 460 ) {
return STAT_SET_INVALID ;
}
cbuf [ 0 ] = PW_SET_CONF_COMMAND ;
cbuf [ 1 ] = PW_CONF_NOMINAL_OUT_VOLTAGE ;
cbuf [ 2 ] = tmp & 0xff ;
cbuf [ 3 ] = tmp > > 8 ;
res = command_write_sequence ( cbuf , 4 , answer ) ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " Nominal output voltage set to %d V " , tmp ) ;
return decode_setvar_exec ( res , ( unsigned char ) answer [ 0 ] , varname , success_msg ) ;
}
if ( ! strcasecmp ( varname , " battery.energysave.load " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
tmp = atoi ( val ) ;
if ( tmp < 0 | | tmp > 100 ) {
return STAT_SET_INVALID ;
}
cbuf [ 0 ] = PW_SET_CONF_COMMAND ;
cbuf [ 1 ] = PW_CONF_SLEEP_TH_LOAD ;
cbuf [ 2 ] = tmp & 0xff ;
cbuf [ 3 ] = 0x0 ;
res = command_write_sequence ( cbuf , 4 , answer ) ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " Minimum load before sleep countdown set to %d %% " , tmp ) ;
return decode_setvar_exec ( res , ( unsigned char ) answer [ 0 ] , varname , success_msg ) ;
}
if ( ! strcasecmp ( varname , " battery.energysave.delay " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
tmp = atoi ( val ) ;
if ( tmp < 0 | | tmp > 255 ) {
return STAT_SET_INVALID ;
}
cbuf [ 0 ] = PW_SET_CONF_COMMAND ;
cbuf [ 1 ] = PW_CONF_SLEEP_DELAY ;
cbuf [ 2 ] = tmp & 0xff ;
cbuf [ 3 ] = 0x0 ;
res = command_write_sequence ( cbuf , 4 , answer ) ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " Delay before sleep shutdown set to %d min " , tmp ) ;
return decode_setvar_exec ( res , ( unsigned char ) answer [ 0 ] , varname , success_msg ) ;
}
if ( ! strcasecmp ( varname , " battery.packs " ) ) {
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
tmp = atoi ( val ) ;
if ( tmp < 0 | | tmp > 5 ) {
return STAT_SET_INVALID ;
}
cbuf [ 0 ] = PW_SET_CONF_COMMAND ;
cbuf [ 1 ] = PW_CONF_BATT_STRINGS ;
cbuf [ 2 ] = tmp ;
cbuf [ 3 ] = 0x0 ;
res = command_write_sequence ( cbuf , 4 , answer ) ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " EBM Count set to %d " , tmp ) ;
return decode_setvar_exec ( res , ( unsigned char ) answer [ 0 ] , varname , success_msg ) ;
}
2011-09-29 21:14:46 +03:00
strncpy ( namebuf , varname , sizeof ( namebuf ) ) ;
namebuf [ NUT_OUTLET_POSITION ] = ' n ' ; /* Assumes a maximum of 9 outlets */
2015-04-30 16:53:36 +03:00
if ( ( ! strcasecmp ( namebuf , " outlet.n.delay.start " ) ) | |
( ! strcasecmp ( namebuf , " outlet.n.delay.shutdown " ) ) ) {
2011-09-29 21:14:46 +03:00
2015-04-30 16:53:36 +03:00
if ( outlet_block_len < = 8 ) {
return STAT_SET_INVALID ;
}
2011-09-29 21:14:46 +03:00
2015-04-30 16:53:36 +03:00
if ( ! strcasecmp ( namebuf , " outlet.n.delay.start " ) ) {
onOff_setting = PW_AUTO_ON_DELAY ;
}
2011-09-29 21:14:46 +03:00
2015-04-30 16:53:36 +03:00
send_write_command ( AUTHOR , 4 ) ;
sleep ( PW_SLEEP ) ; /* Need to. Have to wait at least 0,25 sec max 16 sec */
outlet_num = varname [ NUT_OUTLET_POSITION ] - ' 0 ' ;
if ( outlet_num < 1 | | outlet_num > 9 ) {
return STAT_SET_INVALID ;
}
sec = atoi ( val ) ;
/* Check value:
* 0 - 32767 are valid values
* - 1 means no Automatic off or restart
* for Auto Off Delay :
* 0 - 30 are valid but ill - advised */
if ( sec < - 1 | | sec > 0x7FFF ) {
return STAT_SET_INVALID ;
}
cbuf [ 0 ] = PW_SET_OUTLET_COMMAND ; /* Cmd */
cbuf [ 1 ] = onOff_setting ; /* Set Auto Off (1) or On (2) Delay */
cbuf [ 2 ] = outlet_num ; /* Outlet number */
cbuf [ 3 ] = sec & 0xff ; /* Delay in seconds LSB */
cbuf [ 4 ] = sec > > 8 ; /* Delay in seconds MSB */
res = command_write_sequence ( cbuf , 5 , answer ) ;
snprintf ( success_msg , sizeof ( success_msg ) - 1 , " Outlet %d %s delay set to %d sec " ,
outlet_num , ( onOff_setting = = PW_AUTO_ON_DELAY ) ? " start " : " shutdown " , sec ) ;
return decode_setvar_exec ( res , ( unsigned char ) answer [ 0 ] , varname , success_msg ) ;
2011-09-29 21:14:46 +03:00
}
2015-04-30 16:53:36 +03:00
return STAT_SET_INVALID ;
}
2011-09-29 21:14:46 +03:00
2015-04-30 16:53:36 +03:00
static int decode_setvar_exec ( const int res , const unsigned char exec_status , const char * cmdname , const char * success_msg )
{
2011-09-29 21:14:46 +03:00
if ( res < = 0 ) {
2015-04-30 16:53:36 +03:00
upslogx ( LOG_ERR , " [%s] Short read from UPS " , cmdname ) ;
2011-09-29 21:14:46 +03:00
dstate_datastale ( ) ;
2015-04-30 16:53:36 +03:00
return STAT_SET_FAILED ;
2011-09-29 21:14:46 +03:00
}
2015-04-30 16:53:36 +03:00
/* Decode the status code from command execution */
switch ( exec_status ) {
2013-11-24 17:00:12 +02:00
case BCMXCP_RETURN_ACCEPTED : {
2015-04-30 16:53:36 +03:00
upslogx ( LOG_NOTICE , " [%s] %s " , cmdname , success_msg ) ;
2013-11-24 17:00:12 +02:00
upsdrv_comm_good ( ) ;
return STAT_SET_HANDLED ;
break ;
}
case BCMXCP_RETURN_ACCEPTED_PARAMETER_ADJUST : {
2015-04-30 16:53:36 +03:00
upslogx ( LOG_NOTICE , " [%s] Parameter adjusted " , cmdname ) ;
upslogx ( LOG_NOTICE , " [%s] %s " , cmdname , success_msg ) ;
2013-11-24 17:00:12 +02:00
upsdrv_comm_good ( ) ;
2011-09-29 21:14:46 +03:00
return STAT_SET_HANDLED ;
break ;
}
2013-11-24 17:00:12 +02:00
case BCMXCP_RETURN_BUSY : {
2015-04-30 16:53:36 +03:00
upslogx ( LOG_NOTICE , " [%s] Busy or disbled by front panel " , cmdname ) ;
return STAT_SET_FAILED ;
2011-09-29 21:14:46 +03:00
break ;
}
2015-04-30 16:53:36 +03:00
case BCMXCP_RETURN_UNRECOGNISED : {
upslogx ( LOG_NOTICE , " [%s] Unrecognised command byte or corrupt checksum " , cmdname ) ;
return STAT_SET_FAILED ;
2011-09-29 21:14:46 +03:00
break ;
}
2013-11-24 17:00:12 +02:00
case BCMXCP_RETURN_INVALID_PARAMETER : {
2015-04-30 16:53:36 +03:00
upslogx ( LOG_NOTICE , " [%s] Invalid parameter " , cmdname ) ;
return STAT_SET_INVALID ;
break ;
}
case BCMXCP_RETURN_PARAMETER_OUT_OF_RANGE : {
upslogx ( LOG_NOTICE , " [%s] Parameter out of range " , cmdname ) ;
return STAT_SET_INVALID ;
2011-09-29 21:14:46 +03:00
break ;
}
default : {
2015-04-30 16:53:36 +03:00
upslogx ( LOG_NOTICE , " [%s] Not supported " , cmdname ) ;
return STAT_SET_INVALID ;
2011-09-29 21:14:46 +03:00
break ;
}
}
}
2012-06-01 16:55:19 +03:00
/*******************************
* Extracted from usbhid - ups . c *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2011-09-29 21:14:46 +03:00
2012-06-01 16:55:19 +03:00
/* find the NUT value matching that XCP Item value */
2013-11-24 17:00:12 +02:00
static const char * nut_find_infoval ( info_lkp_t * xcp2info , const double value , const bool_t debug_output_nonexisting )
2012-06-01 16:55:19 +03:00
{
2013-11-24 17:00:12 +02:00
info_lkp_t * info_lkp ;
2012-06-01 16:55:19 +03:00
/* if a conversion function is defined, use 'value' as argument for it */
if ( xcp2info - > fun ! = NULL ) {
return xcp2info - > fun ( value ) ;
}
/* use 'value' as an index for a lookup in an array */
for ( info_lkp = xcp2info ; info_lkp - > nut_value ! = NULL ; info_lkp + + ) {
if ( info_lkp - > xcp_value = = ( long ) value ) {
upsdebugx ( 5 , " nut_find_infoval: found %s (value: %ld) " , info_lkp - > nut_value , ( long ) value ) ;
return info_lkp - > nut_value ;
}
}
2013-11-24 17:00:12 +02:00
if ( debug_output_nonexisting = = TRUE ) {
upsdebugx ( 3 , " nut_find_infoval: no matching INFO_* value for this XCP value (%g) " , value ) ;
}
2012-06-01 16:55:19 +03:00
return NULL ;
}