# IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html package xCAT::PPCpower; use strict; use Getopt::Long; use xCAT::PPCcli qw(SUCCESS EXPECT_ERROR RC_ERROR NR_ERROR); ########################################################################## # Parse the command line for options and operands ########################################################################## sub parse_args { my $request = shift; my $args = $request->{arg}; my %opt = (); my @rpower = qw(on off stat state reset boot of); my @VERSION = qw( 2.0 ); ############################################# # Responds with usage statement ############################################# local *usage = sub { return( [ $_[0], "rpower -h|--help", "rpower -v|--version", "rpower [-V|--verbose] noderange " . join( '|', @rpower ), " -h writes usage information to standard output", " -v displays command version", " -V verbose output" ]); }; ############################################# # Process command-line arguments ############################################# if ( !defined( $args )) { return(usage( "No command specified" )); } ############################################# # Checks case in GetOptions, allows opts # to be grouped (e.g. -vx), and terminates # at the first unrecognized option. ############################################# @ARGV = @$args; $Getopt::Long::ignorecase = 0; Getopt::Long::Configure( "bundling" ); if ( !GetOptions( \%opt, qw(h|help V|Verbose v|version) )) { return( usage() ); } #################################### # Option -h for Help #################################### if ( exists( $opt{h} )) { return( usage() ); } #################################### # Option -v for version #################################### if ( exists( $opt{v} )) { return( \@VERSION ); } #################################### # Check for "-" with no option #################################### if ( grep(/^-$/, @ARGV )) { return(usage( "Missing option: -" )); } #################################### # Unsupported commands #################################### my ($cmd) = grep(/^$ARGV[0]$/, @rpower ); if ( !defined( $cmd )) { return(usage( "Invalid command: $ARGV[0]" )); } #################################### # Check for an extra argument #################################### shift @ARGV; if ( defined( $ARGV[0] )) { return(usage( "Invalid Argument: $ARGV[0]" )); } #################################### # Change "stat" to "state" #################################### $request->{op} = $cmd; $cmd =~ s/^stat$/state/; #################################### # Power commands special case #################################### if ( $cmd ne "state" ) { $cmd = ($cmd eq "boot") ? "powercmd_boot" : "powercmd"; } $request->{method} = $cmd; return( \%opt ); } ########################################################################## # Builds a hash of CEC/LPAR information returned from HMC/IVM ########################################################################## sub enumerate { my $exp = shift; my $node = shift; my $mtms = shift; my $filter = shift; my %outhash = (); my %cmds = (); ###################################### # Check for CEC/LPAR/BPAs in list ###################################### while (my ($name,$d) = each(%$node) ) { if ( @$d[4] =~ /^fsp|lpar|bpa$/ ) { $cmds{@$d[4]} = 1; } } ###################################### # Check for HMC/IVMs in list ###################################### my ($name) = keys %$node; my $type = @{$node->{$name}}[4]; if ( $type =~ /^hmc|ivm$/ ) { $outhash{$name} = "Running"; } foreach ( keys %cmds ) { my $values = xCAT::PPCcli::lssyscfg( $exp, $_, $mtms, $filter ); my $Rc = shift(@$values); ################################## # Return error ################################## if ( $Rc != SUCCESS ) { return( [$Rc,@$values[0]] ); } ################################## # Save LPARs by name ################################## foreach ( @$values ) { my ($name,$state) = split /,/; $outhash{ $name } = $state; } } return( [SUCCESS,\%outhash] ); } ########################################################################## # Performs boot operation (Off->On, On->Reset) ########################################################################## sub powercmd_boot { my $request = shift; my $hash = shift; my $exp = shift; my $filter = "name,state"; my @output = (); ###################################### # Power commands are grouped by CEC # not Hardware Control Point ###################################### ###################################### # Get CEC MTMS ###################################### my ($name) = keys %$hash; my $mtms = @{$hash->{$name}}[2]; ###################################### # Build CEC/LPAR information hash ###################################### my $stat = enumerate( $exp, $hash, $mtms, $filter ); my $Rc = shift(@$stat); my $data = @$stat[0]; while (my ($name,$d) = each(%$hash) ) { ################################## # Output error ################################## if ( $Rc != SUCCESS ) { push @output, [$name,$data]; next; } ################################## # Node not found ################################## if ( !exists( $data->{$name} )) { push @output, [$name,"Node not found"]; next; } ################################## # Convert state to on/off ################################## my $state = power_status($data->{$name}); my $op = ($state =~ /^Off|Not Activated$/) ? "on" : "reset"; ############################## # Send power command ############################## my $result = xCAT::PPCcli::chsysstate( $exp, $op, $d ); push @output, [$name,@$result[1]]; } return( \@output ); } ########################################################################## # Performs power control operations (on,off,reboot,etc) ########################################################################## sub powercmd { my $request = shift; my $hash = shift; my $exp = shift; my @result = (); #################################### # Power commands are grouped by CEC # not Hardware Control Point #################################### while (my ($name,$d) = each(%$hash) ) { ################################ # Send command to each LPAR ################################ my $values = xCAT::PPCcli::chsysstate( $exp, $request->{op}, $d ); my $Rc = shift(@$values); ################################ # Return result ################################ push @result, [$name,@$values[0]]; } return( \@result ); } ########################################################################## # Queries CEC/LPAR power status (On or Off) ########################################################################## sub power_status { my @states = ( "Operating", "Running", "Open Firmware" ); foreach ( @states ) { if ( /^$_[0]$/ ) { return("on"); } } return("off"); } ########################################################################## # Queries CEC/LPAR power state ########################################################################## sub state { my $request = shift; my $hash = shift; my $exp = shift; my $prefix = shift; my $convert = shift; my $filter = "name,state"; my @result = (); if ( !defined( $prefix )) { $prefix = ""; } while (my ($mtms,$h) = each(%$hash) ) { ###################################### # Build CEC/LPAR information hash ###################################### my $stat = enumerate( $exp, $h, $mtms, $filter ); my $Rc = shift(@$stat); my $data = @$stat[0]; while (my ($name,$d) = each(%$h) ) { ################################## # Output error ################################## if ( $Rc != SUCCESS ) { push @result, [$name, "$prefix$data"]; next; } ################################## # Node not found ################################## if ( !exists( $data->{$name} )) { push @result, [$name, $prefix."Node not found"]; next; } ################################## # Output value ################################## my $value = $data->{$name}; ############################## # Convert state to on/off ############################## if ( defined( $convert )) { $value = power_status( $value ); } push @result, [$name,"$prefix$value"]; } } return( \@result ); } 1;