2010-11-25 02:47:38 +00:00
# IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html
2011-03-02 09:32:34 +00:00
BEGIN
{
$ ::XCATROOT = $ ENV { 'XCATROOT' } ? $ ENV { 'XCATROOT' } : '/opt/xcat' ;
}
2011-11-24 09:24:56 +00:00
2011-03-02 09:32:34 +00:00
2010-11-25 02:47:38 +00:00
package xCAT::FSPvm ;
2011-03-02 09:32:34 +00:00
use lib "$::XCATROOT/lib/perl" ;
2010-11-25 02:47:38 +00:00
use strict ;
use Getopt::Long ;
use xCAT::PPCdb ;
use xCAT::PPCcli qw( SUCCESS EXPECT_ERROR RC_ERROR NR_ERROR ) ;
use xCAT::Usage ;
use xCAT::NodeRange ;
use xCAT::FSPUtils ;
2011-04-08 05:23:16 +00:00
#use Data::Dumper;
2012-06-21 09:43:16 +00:00
use xCAT::MsgUtils qw( verbose_message ) ;
2010-11-25 02:47:38 +00:00
##############################################
# Globals
##############################################
my % method = (
mkvm = > \ & mkvm_parse_args ,
lsvm = > \ & lsvm_parse_args ,
rmvm = > \ & rmvm_parse_args ,
chvm = > \ & chvm_parse_args
) ;
##########################################################################
# Parse the command line for options and operands
##########################################################################
sub parse_args {
my $ request = shift ;
my $ cmd = $ request - > { command } ;
###############################
# Invoke correct parse_args
###############################
my $ result = $ method { $ cmd } ( $ request ) ;
return ( $ result ) ;
}
##########################################################################
# Parse the chvm command line for options and operands
##########################################################################
2011-11-24 09:24:56 +00:00
sub chvm_parse_extra_options {
my $ args = shift ;
my $ opt = shift ;
if ( ref ( $ args ) ne 'ARRAY' ) {
return "$args" ;
}
foreach ( @$ args ) {
my ( $ cmd , $ value ) = split ( /\=/ , $ _ ) ;
if ( ! defined ( $ value ) ) {
return "no value specified" ;
}
if ( $ cmd =~ /^lparname$/ ) {
2012-07-24 06:53:31 +00:00
if ( $ value ne '*' && $ value !~ /^[a-zA-Z0-9-_]+$/ ) {
2011-11-24 09:24:56 +00:00
return "'$value' invalid" ;
}
my $ len = rindex $ value . "\$" , "\$" ;
if ( $ len > '47' ) {
return "'$value' is too long, max 47 characters" ;
}
# } elsif ($cmd =~ /^huge_page$/) {
# if ($value !~ /^\d+\/\d+\/\d+$/) {
# return "'$value' invalid";
# }
} else {
return "'$cmd' not support" ;
}
$ opt - > { $ cmd } = $ value ;
}
return undef ;
2011-07-25 04:30:07 +00:00
}
2011-11-24 09:24:56 +00:00
2010-11-25 02:47:38 +00:00
sub chvm_parse_args {
my $ request = shift ;
my % opt = ( ) ;
my $ cmd = $ request - > { command } ;
my $ args = $ request - > { arg } ;
#############################################
# Responds with usage statement
#############################################
local * usage = sub {
my $ usage_string = xCAT::Usage - > getUsage ( $ cmd ) ;
return ( [ $ _ [ 0 ] , $ usage_string ] ) ;
} ;
#############################################
# Process command-line arguments
#############################################
if ( ! defined ( $ args ) ) {
$ request - > { method } = $ cmd ;
2011-03-02 09:32:34 +00:00
return ( \ % opt ) ;
2010-11-25 02:47:38 +00:00
}
#############################################
# 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" ) ;
2012-06-29 09:03:22 +00:00
if ( ! GetOptions ( \ % opt , qw( V|verbose p=s i=s m=s r=s ) ) ) {
2010-11-25 02:47:38 +00:00
return ( usage ( ) ) ;
}
####################################
# Check for "-" with no option
####################################
if ( grep ( /^-$/ , @ ARGV ) ) {
return ( usage ( "Missing option: -" ) ) ;
}
####################################
# Configuration file required
####################################
2011-04-06 06:49:42 +00:00
#if ( !exists( $opt{p}) ) {
# if ( !defined( $request->{stdin} )) {
# return(usage( "Configuration file or attributes not specified" ));
# }
#}
2011-03-02 09:32:34 +00:00
my @ cfgdata ;
if ( exists ( $ opt { p } ) ) {
2011-04-06 06:49:42 +00:00
if ( exists ( $ opt { i } ) || exists ( $ opt { r } ) || exists ( $ opt { m } ) ) {
return ( usage ( "-p should NOT be used with -i, -r or -m." ) ) ;
}
2011-03-02 09:32:34 +00:00
$ opt { p } = $ request - > { cwd } - > [ 0 ] . '/' . $ opt { p } if ( $ opt { p } !~ /^\// ) ;
return ( usage ( "Profile $opt{p} cannot be found" ) ) if ( ! - f $ opt { p } ) ;
open ( PROFFILE , "<$opt{p}" ) or return ( usage ( "Cannot open profile $opt{p}" ) ) ;
while ( <PROFFILE> ) {
chomp ;
2011-03-08 06:44:07 +00:00
if ( $ _ =~ /(\d+):(\s+)(\d+)\/([\w\.\-]+)\/(\w+)\// ) {
2011-03-02 09:32:34 +00:00
push @ cfgdata , $ _ ;
} else {
return ( usage ( "Invalid line in profile: $_" ) ) ;
}
}
$ opt { profile } = \ @ cfgdata ;
}
2011-04-06 06:49:42 +00:00
if ( defined ( $ request - > { stdin } ) ) {
$ opt { p } = 1 ;
if ( exists ( $ opt { i } ) || exists ( $ opt { r } ) || exists ( $ opt { m } ) ) {
return ( usage ( "When the profile is piped into the chvm command, the -i, -r and -m could NOT be used." ) ) ;
}
}
2011-03-07 06:18:23 +00:00
#if (defined( $request->{stdin} )) {
# my $p = $request->{stdin};
# my @io = split(/\n/, $p) ;
# foreach (@io) {
# chomp;
# if( $_ =~ /(\d+):(\s+)(\d+),([\w\.\-]+),(\w+),/) {
# push @cfgdata, $_;
# } else {
# return ( usage( "Invalid line in profile: $_"));
# }
2011-03-02 09:32:34 +00:00
2011-03-07 06:18:23 +00:00
# }
2011-03-02 09:32:34 +00:00
2011-03-07 06:18:23 +00:00
# $opt{profile} = \@cfgdata;
#}
#print "in parse args:\n";
2011-04-06 06:49:42 +00:00
#print Dumper(\%opt);
if ( exists ( $ opt { i } ) ) {
if ( ! exists ( $ opt { r } ) ) {
return ( usage ( "Option -i should be used with option -r ." ) ) ;
}
if ( $ opt { i } !~ /^([1-9]{1}|[1-9]{1}[0-9]+)$/ ) {
return ( usage ( "Invalid entry: $opt{i}" ) ) ;
}
my @ id = ( 1 , 5 , 9 , 13 , 17 , 21 , 25 , 29 ) ;
my @ found = grep ( /^$opt{i}$/ , @ id ) ;
if ( @ found != 1 ) {
2011-06-23 16:31:55 +00:00
return ( usage ( "Invalid entry: $opt{i}.\n For Power 775, starting numeric id of the newly created partitions only could be 1, 5, 9, 13, 17, 21, 25 and 29." ) ) ;
2011-04-06 06:49:42 +00:00
}
#if ( !exists($opt{o}) ) {
2011-06-23 16:31:55 +00:00
# return(usage("For Power 775, -i should be used with -o"));
2011-04-06 06:49:42 +00:00
#}
#my @value = (1, 2, 3, 4, 5);
#if ( grep(/^$opt{i}$/, @id ) != 1) {
2011-06-23 16:31:55 +00:00
# return(usage( "Invalid entry: $opt{o}.\n For Power 775, octant configuration values only could be 1, 2, 3, 4, 5. Please see the details in manpage of mkvm." ));
2011-04-06 06:49:42 +00:00
#}
}
# pending memory interleaving mode (1- interleaved, 2- non-interleaved)
# non-interleaved mode means the memory cannot be shared across the processors in an octant.
# interleaved means the memory can be shared.
if ( exists ( $ opt { m } ) ) {
if ( $ opt { m } =~ /^interleaved$/ || $ opt { m } =~ /^1$/ ) {
$ opt { m } = 1 ;
} elsif ( $ opt { m } =~ /^non-interleaved$/ || $ opt { m } =~ /^2$/ ) {
$ opt { m } = 2 ;
} else {
2011-06-23 16:31:55 +00:00
return ( usage ( "Invalid entry: $opt{m}.\n For Power 775, the pending memory interleaving mode only could be interleaved(or 1), or non-interleaved(or 2)." ) ) ;
2011-04-06 06:49:42 +00:00
}
} else {
2011-11-01 02:03:30 +00:00
$ opt { m } = 1 ; # interleaved, which is the default
2011-04-06 06:49:42 +00:00
}
my @ ratio = ( 1 , 2 , 3 , 4 , 5 ) ;
my % octant_cfg = ( ) ;
if ( exists ( $ opt { r } ) ) {
if ( ! exists ( $ opt { i } ) ) {
return ( usage ( "Option -r should be used with option -i ." ) ) ;
}
my @ elems = split ( /\,/ , $ opt { r } ) ;
my $ range = "" ;
while ( my $ elem = shift @ elems ) {
2011-06-20 09:12:38 +00:00
if ( $ elem !~ /\:/ ) {
return ( usage ( "Invalid argument $elem.\n The input format for 'r' should be like this: \"-r Octant_id:Value\"." ) )
}
2011-04-06 06:49:42 +00:00
if ( $ elem !~ /\-/ ) {
my @ subelems = split ( /\:/ , $ elem ) ;
if ( $ subelems [ 0 ] < 0 || $ subelems [ 0 ] > 7 ) {
return ( usage ( "Octant ID only could be 0 to 7 in the octant configuration value $elem" ) ) ;
}
if ( grep ( /^$subelems[1]$/ , @ ratio ) != 1 ) {
2011-06-23 16:31:55 +00:00
return ( usage ( "Invalid octant configuration value in $elem.\n For Power 775, octant configuration values only could be 1, 2, 3, 4, 5. Please see the details in manpage of chvm." ) ) ;
2011-04-06 06:49:42 +00:00
}
if ( exists ( $ octant_cfg { $ subelems [ 0 ] } ) && $ octant_cfg { $ subelems [ 0 ] } == $ subelems [ 1 ] ) {
return ( usage ( "In the octant configuration rule, same octant with different octant configuration value. Error!" ) ) ;
}
$ octant_cfg { $ subelems [ 0 ] } = $ subelems [ 1 ] ;
$ range . = "$elem," ;
} else {
my @ subelems = split ( /\:/ , $ elem ) ;
my ( $ left , $ right ) = split ( /\-/ , $ subelems [ 0 ] ) ;
if ( $ left < 0 || $ left > 7 || $ right < 0 || $ right > 7 ) {
return ( usage ( "Octant ID only could be 0 to 7 in the octant configuration rule $elem" ) ) ;
}
if ( $ left == $ right ) {
if ( grep ( /^$subelems[1]$/ , @ ratio ) != 1 ) {
2011-06-23 16:31:55 +00:00
return ( usage ( "Invalid octant configuration value in $elem.\n For Power 775, octant configuration values only could be 1, 2, 3, 4, 5. Please see the details in manpage of chvm." ) ) ;
2011-04-06 06:49:42 +00:00
}
if ( exists ( $ octant_cfg { $ left } ) || $ octant_cfg { $ left } == $ subelems [ 1 ] ) {
return ( usage ( "In the octant configuration rule, same octant with different octant configuration value. Error!" ) ) ;
}
$ octant_cfg { $ left } = $ subelems [ 1 ] ;
$ range . = "$left:$subelems[1],"
} elsif ( $ left < $ right ) {
my $ i = $ left ;
for ( $ i ; $ i <= $ right ; $ i + + ) {
if ( exists ( $ octant_cfg { $ i } ) || $ octant_cfg { $ i } == $ subelems [ 1 ] ) {
return ( usage ( "In the octant configuration rule, same octant with different octant configuration value. Error!" ) ) ;
}
$ octant_cfg { $ i } = $ subelems [ 1 ] ;
$ range . = "$i:$subelems[1]," ;
}
} else {
return ( usage ( "In the octant configuration rule $elem, the left octant ID could NOT be bigger than the right octant ID" ) ) ;
}
} # end of "if .. else.."
} # end of while
} #end of if
if ( exists ( $ opt { i } ) && exists ( $ opt { r } ) ) {
$ opt { octant_cfg } { octant_cfg_value } = ( \ % octant_cfg ) ;
$ opt { octant_cfg } { memory_interleave } = $ opt { m } ;
$ opt { target } = \ @ { $ request - > { node } } ;
my $ ppctab = xCAT::Table - > new ( 'ppc' ) ;
unless ( $ ppctab ) {
return ( usage ( "Cannot open ppc table" ) ) ;
}
my $ other_p ;
foreach my $ node ( @ { $ request - > { node } } ) {
my $ parent_hash = $ ppctab - > getNodeAttribs ( $ node , [ qw( parent ) ] ) ;
my $ p = $ parent_hash - > { parent } ;
if ( ! $ p ) {
return ( usage ( "Not found the parent of $node" ) ) ;
}
if ( ! defined ( $ other_p ) ) {
$ other_p = $ p ;
}
if ( $ other_p ne $ p ) {
2011-06-23 16:31:55 +00:00
return ( usage ( "For Power 775, please make sure the noderange are in one CEC " ) ) ;
2011-04-06 06:49:42 +00:00
}
}
$ request - > { node } = [ $ other_p ] ;
$ request - > { noderange } = $ other_p ;
}
2011-08-05 06:52:14 +00:00
####################################
# Check for an extra argument
####################################
if ( defined ( $ ARGV [ 0 ] ) ) {
2011-11-24 09:24:56 +00:00
my $ check_chvm_arg = chvm_parse_extra_options ( \ @ ARGV , \ % opt ) ;
if ( defined ( $ check_chvm_arg ) ) {
return ( usage ( "Invalid argument: $check_chvm_arg" ) ) ;
2011-08-05 06:52:14 +00:00
} elsif ( ( $ opt { lparname } ne '*' ) && ( scalar ( @ { $ request - > { node } } ) > '1' ) ) {
return ( usage ( "Invalid argument: must specify '*' for more than one node" ) ) ;
}
2011-11-24 09:24:56 +00:00
if ( ( exists ( $ opt { lparname } ) || exists ( $ opt { huge_page } ) ) &&
2011-08-16 04:08:46 +00:00
( exists ( $ opt { p } ) || exists ( $ opt { i } ) || exists ( $ opt { r } ) ) ) {
return ( usage ( "lparname should NOT be used with -p, -i or -r." ) ) ;
2011-08-05 06:52:14 +00:00
}
}
2011-04-06 06:49:42 +00:00
2010-11-25 02:47:38 +00:00
####################################
# No operands - add command name
####################################
$ request - > { method } = $ cmd ;
return ( \ % opt ) ;
}
##########################################################################
# Parse the mkvm command line for options and operands
##########################################################################
sub mkvm_parse_args {
my $ request = shift ;
my % opt = ( ) ;
my $ cmd = $ request - > { command } ;
my $ args = $ request - > { arg } ;
#############################################
# Responds with usage statement
#############################################
local * usage = sub {
my $ usage_string = xCAT::Usage - > getUsage ( $ cmd ) ;
return ( [ $ _ [ 0 ] , $ usage_string ] ) ;
} ;
#############################################
# 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(V|verbose ibautocfg ibacap=s i=s l=s c=s p=s full) )) {
# return( usage() );
# }
2013-09-04 23:27:28 -07:00
if ( ! GetOptions ( \ % opt , qw( V|verbose i=s m=s r=s full part vios ) ) ) {
2010-11-25 02:47:38 +00:00
return ( usage ( ) ) ;
}
####################################
# Check for "-" with no option
####################################
if ( grep ( /^-$/ , @ ARGV ) ) {
return ( usage ( "Missing option: -" ) ) ;
}
####################################
# Check for non-zero integer
####################################
if ( exists ( $ opt { i } ) ) {
if ( $ opt { i } !~ /^([1-9]{1}|[1-9]{1}[0-9]+)$/ ) {
return ( usage ( "Invalid entry: $opt{i}" ) ) ;
}
my @ id = ( 1 , 5 , 9 , 13 , 17 , 21 , 25 , 29 ) ;
my @ found = grep ( /^$opt{i}$/ , @ id ) ;
if ( @ found != 1 ) {
2011-06-23 16:31:55 +00:00
return ( usage ( "Invalid entry: $opt{i}.\n For Power 775, starting numeric id of the newly created partitions only could be 1, 5, 9, 13, 17, 21, 25 and 29." ) ) ;
2010-11-25 02:47:38 +00:00
}
2011-03-02 09:32:34 +00:00
#if ( !exists($opt{o}) ) {
2011-06-23 16:31:55 +00:00
# return(usage("For Power 775, -i should be used with -o"));
2011-03-02 09:32:34 +00:00
#}
2010-11-25 02:47:38 +00:00
2011-03-02 09:32:34 +00:00
#my @value = (1, 2, 3, 4, 5);
#if ( grep(/^$opt{i}$/, @id ) != 1) {
2011-06-23 16:31:55 +00:00
# return(usage( "Invalid entry: $opt{o}.\n For Power 775, octant configuration values only could be 1, 2, 3, 4, 5. Please see the details in manpage of mkvm." ));
2011-03-02 09:32:34 +00:00
#}
2010-11-25 02:47:38 +00:00
}
2011-03-02 09:32:34 +00:00
# pending memory interleaving mode (1- interleaved, 2- non-interleaved)
# non-interleaved mode means the memory cannot be shared across the processors in an octant.
# interleaved means the memory can be shared.
if ( exists ( $ opt { m } ) ) {
if ( $ opt { m } =~ /^interleaved$/ || $ opt { m } =~ /^1$/ ) {
$ opt { m } = 1 ;
} elsif ( $ opt { m } =~ /^non-interleaved$/ || $ opt { m } =~ /^2$/ ) {
$ opt { m } = 2 ;
} else {
2011-06-23 16:31:55 +00:00
return ( usage ( "Invalid entry: $opt{m}.\n For Power 775, the pending memory interleaving mode only could be interleaved(or 1), or non-interleaved(or 2)." ) ) ;
2010-11-25 02:47:38 +00:00
}
2013-09-04 23:27:28 -07:00
} elsif ( ! exists ( $ opt { full } ) && ! exists ( $ opt { part } ) && ! exists ( $ opt { vios } ) ) {
2011-03-02 09:32:34 +00:00
$ opt { m } = 2 ; # non-interleaved, which is the default
}
if ( exists ( $ opt { r } ) ) {
2013-09-04 23:27:28 -07:00
my @ ratio = ( 1 , 2 , 3 , 4 , 5 ) ;
my % octant_cfg = ( ) ;
2011-03-02 09:32:34 +00:00
my @ elems = split ( /\,/ , $ opt { r } ) ;
my $ range = "" ;
while ( my $ elem = shift @ elems ) {
if ( $ elem !~ /\-/ ) {
my @ subelems = split ( /\:/ , $ elem ) ;
if ( $ subelems [ 0 ] < 0 || $ subelems [ 0 ] > 7 ) {
return ( usage ( "Octant ID only could be 0 to 7 in the octant configuration value $elem" ) ) ;
}
if ( grep ( /^$subelems[1]$/ , @ ratio ) != 1 ) {
2011-06-23 16:31:55 +00:00
return ( usage ( "Invalid octant configuration value in $elem.\n For Power 775, octant configuration values only could be 1, 2, 3, 4, 5. Please see the details in manpage of mkvm." ) ) ;
2011-03-02 09:32:34 +00:00
}
if ( exists ( $ octant_cfg { $ subelems [ 0 ] } ) && $ octant_cfg { $ subelems [ 0 ] } == $ subelems [ 1 ] ) {
return ( usage ( "In the octant configuration rule, same octant with different octant configuration value. Error!" ) ) ;
}
$ octant_cfg { $ subelems [ 0 ] } = $ subelems [ 1 ] ;
$ range . = "$elem," ;
} else {
my @ subelems = split ( /\:/ , $ elem ) ;
my ( $ left , $ right ) = split ( /\-/ , $ subelems [ 0 ] ) ;
if ( $ left < 0 || $ left > 7 || $ right < 0 || $ right > 7 ) {
return ( usage ( "Octant ID only could be 0 to 7 in the octant configuration rule $elem" ) ) ;
}
if ( $ left == $ right ) {
if ( grep ( /^$subelems[1]$/ , @ ratio ) != 1 ) {
2011-06-23 16:31:55 +00:00
return ( usage ( "Invalid octant configuration value in $elem.\n For Power 775, octant configuration values only could be 1, 2, 3, 4, 5. Please see the details in manpage of mkvm." ) ) ;
2011-03-02 09:32:34 +00:00
}
if ( exists ( $ octant_cfg { $ left } ) || $ octant_cfg { $ left } == $ subelems [ 1 ] ) {
return ( usage ( "In the octant configuration rule, same octant with different octant configuration value. Error!" ) ) ;
}
$ octant_cfg { $ left } = $ subelems [ 1 ] ;
$ range . = "$left:$subelems[1],"
} elsif ( $ left < $ right ) {
my $ i = $ left ;
for ( $ i ; $ i <= $ right ; $ i + + ) {
if ( exists ( $ octant_cfg { $ i } ) || $ octant_cfg { $ i } == $ subelems [ 1 ] ) {
return ( usage ( "In the octant configuration rule, same octant with different octant configuration value. Error!" ) ) ;
}
$ octant_cfg { $ i } = $ subelems [ 1 ] ;
$ range . = "$i:$subelems[1]," ;
}
} else {
return ( usage ( "In the octant configuration rule $elem, the left octant ID could NOT be bigger than the right octant ID" ) ) ;
}
} # end of "if .. else.."
} # end of while
2013-09-04 23:27:28 -07:00
$ opt { octant_cfg } { octant_cfg_value } = ( \ % octant_cfg ) ;
$ opt { octant_cfg } { memory_interleave } = $ opt { m } ;
2011-03-02 09:32:34 +00:00
} #end of if
2010-11-25 02:47:38 +00:00
2013-09-04 23:27:28 -07:00
if ( ( ! exists ( $ opt { i } ) || ! exists ( $ opt { r } ) ) && ! exists ( $ opt { full } ) && ! exists ( $ opt { part } ) && ! exists ( $ opt { vios } ) ) {
2011-03-02 09:32:34 +00:00
return ( usage ( ) ) ;
2010-11-25 02:47:38 +00:00
}
2011-03-02 09:32:34 +00:00
$ opt { target } = \ @ { $ request - > { node } } ;
my $ ppctab = xCAT::Table - > new ( 'ppc' ) ;
unless ( $ ppctab ) {
return ( usage ( "Cannot open ppc table" ) ) ;
}
my $ other_p ;
foreach my $ node ( @ { $ request - > { node } } ) {
my $ parent_hash = $ ppctab - > getNodeAttribs ( $ node , [ qw( parent ) ] ) ;
my $ p = $ parent_hash - > { parent } ;
if ( ! $ p ) {
return ( usage ( "Not found the parent of $node" ) ) ;
}
2013-09-04 23:27:28 -07:00
if ( exists ( $ opt { full } ) and defined ( $ other_p ) and $ other_p eq $ p ) {
return ( usage ( "Only one full partition can be created in one CEC" ) ) ;
}
2011-03-02 09:32:34 +00:00
if ( ! defined ( $ other_p ) ) {
$ other_p = $ p ;
}
if ( $ other_p ne $ p ) {
2011-06-23 16:31:55 +00:00
return ( usage ( "For Power 775, please make sure the noderange are in one CEC " ) ) ;
2011-03-02 09:32:34 +00:00
}
}
2013-09-04 23:27:28 -07:00
if ( ! exists ( $ opt { full } ) && ! exists ( $ opt { part } ) && ! exists ( $ opt { vios } ) ) {
$ request - > { node } = [ $ other_p ] ;
$ request - > { noderange } = $ other_p ;
}
2010-11-25 02:47:38 +00:00
####################################
# No operands - add command name
####################################
$ request - > { method } = $ cmd ;
return ( \ % opt ) ;
}
##########################################################################
# Parse the rmvm command line for options and operands
##########################################################################
sub rmvm_parse_args {
my $ request = shift ;
my % opt = ( ) ;
my $ cmd = $ request - > { command } ;
my $ args = $ request - > { arg } ;
#############################################
# Responds with usage statement
#############################################
local * usage = sub {
my $ usage_string = xCAT::Usage - > getUsage ( $ cmd ) ;
return ( [ $ _ [ 0 ] , $ usage_string ] ) ;
} ;
#############################################
# Process command-line arguments
#############################################
if ( ! defined ( $ args ) ) {
$ request - > { method } = $ cmd ;
return ( \ % opt ) ;
}
#############################################
# 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" ) ;
2013-09-04 23:27:28 -07:00
if ( ! GetOptions ( \ % opt , qw( V|verbose service r p|part ) ) ) {
2010-11-25 02:47:38 +00:00
return ( usage ( ) ) ;
}
2013-09-04 23:27:28 -07:00
if ( ! exists ( $ opt { p } ) ) {
return ( usage ( "rmvm doesn't support for Power 775." ) ) ;
}
2010-11-25 02:47:38 +00:00
####################################
# Check for "-" with no option
####################################
if ( grep ( /^-$/ , @ ARGV ) ) {
return ( usage ( "Missing option: -" ) ) ;
}
####################################
# Check for an extra argument
####################################
if ( defined ( $ ARGV [ 0 ] ) ) {
return ( usage ( "Invalid Argument: $ARGV[0]" ) ) ;
}
####################################
# No operands - add command name
####################################
$ request - > { method } = $ cmd ;
return ( \ % opt ) ;
}
##########################################################################
# Parse the lsvm command line for options and operands
##########################################################################
sub lsvm_parse_args {
my $ request = shift ;
my % opt = ( ) ;
my $ cmd = $ request - > { command } ;
my $ args = $ request - > { arg } ;
#############################################
# Responds with usage statement
#############################################
local * usage = sub {
my $ usage_string = xCAT::Usage - > getUsage ( $ cmd ) ;
return ( [ $ _ [ 0 ] , $ usage_string ] ) ;
} ;
#############################################
# Process command-line arguments
#############################################
if ( ! defined ( $ args ) ) {
$ request - > { method } = $ cmd ;
return ( \ % opt ) ;
}
#############################################
# 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" ) ;
2013-09-04 23:27:28 -07:00
if ( ! GetOptions ( \ % opt , qw( V|verbose l|long p|part ) ) ) {
2010-11-25 02:47:38 +00:00
return ( usage ( ) ) ;
}
####################################
# Check for "-" with no option
####################################
if ( grep ( /^-$/ , @ ARGV ) ) {
return ( usage ( "Missing option: -" ) ) ;
}
####################################
# Check for an extra argument
####################################
if ( defined ( $ ARGV [ 0 ] ) ) {
return ( usage ( "Invalid Argument: $ARGV[0]" ) ) ;
}
####################################
# No operands - add command name
####################################
$ request - > { method } = $ cmd ;
return ( \ % opt ) ;
}
##########################################################################
# Changes the configuration of an existing partition
##########################################################################
sub modify {
my $ request = shift ;
my $ hash = shift ;
2011-06-29 09:01:14 +00:00
my $ usage_string = xCAT::Usage - > getUsage ( $ request - > { command } ) ;
2011-06-29 08:31:02 +00:00
return modify_by_prof ( $ request , $ hash ) if ( $ request - > { opt } - > { p } || $ request - > { stdin } ) ;
2011-04-06 06:49:42 +00:00
return create ( $ request , $ hash ) if ( $ request - > { opt } - > { i } ) ;
2011-11-24 09:24:56 +00:00
return op_extra_cmds ( $ request , $ hash ) if ( $ request - > { opt } - > { lparname } || $ request - > { opt } - > { huge_page } ) ;
2011-06-29 09:01:14 +00:00
return ( [ [ "Error" , "Miss argument\n" . $ usage_string , 1 ] ] ) ;
2010-11-25 02:47:38 +00:00
}
2011-11-24 09:24:56 +00:00
sub do_op_extra_cmds {
2011-07-25 04:30:07 +00:00
my $ request = shift ;
my $ hash = shift ;
my @ values = ( ) ;
2011-11-24 09:24:56 +00:00
my $ action ;
my $ param ;
if ( exists ( $ request - > { opt } - > { lparname } ) ) {
$ action = "set_lpar_name" ;
$ param = $ request - > { opt } - > { lparname } ;
} elsif ( exists ( $ request - > { opt } - > { huge_page } ) ) {
$ action = "set_huge_page" ;
$ param = $ request - > { opt } - > { huge_page } ;
}
2011-07-25 04:30:07 +00:00
my $ lparname_para = $ request - > { opt } - > { lparname } ;
while ( my ( $ mtms , $ h ) = each ( %$ hash ) ) {
while ( my ( $ name , $ d ) = each ( %$ h ) ) {
2011-11-24 09:24:56 +00:00
my $ tmp_value = ( $ param eq '*' ) ? $ name : $ param ;
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "$request->{command} $action for node:$name, parm:$tmp_value." ) ;
2012-04-28 09:26:49 +00:00
my $ value = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , $ action , 0 , $ tmp_value ) ;
2011-11-24 09:24:56 +00:00
if ( @$ value [ 1 ] && ( ( @$ value [ 1 ] =~ /Error/i ) && ( @$ value [ 2 ] ne '0' ) ) ) {
return ( [ [ $ name , @$ value [ 1 ] , '1' ] ] ) ;
2011-07-25 04:30:07 +00:00
} else {
push @ values , [ $ name , "Success" , '0' ] ;
}
}
}
return \ @ values ;
}
sub check_node_info {
my $ hash = shift ;
my $ not_lpar = undef ;
while ( my ( $ mtms , $ h ) = each ( %$ hash ) ) {
while ( my ( $ name , $ d ) = each ( %$ h ) ) {
my $ node_type = @$ d [ 4 ] ;
if ( $ node_type !~ /^lpar$/ ) {
$ not_lpar = $ name ;
last ;
}
}
}
return $ not_lpar ;
}
2011-11-24 09:24:56 +00:00
sub op_extra_cmds {
2011-07-25 04:30:07 +00:00
my $ request = shift ;
my $ hash = shift ;
my $ node = $ request - > { node } ;
my $ lpar_flag = & check_node_info ( $ hash ) ;
if ( defined ( $ lpar_flag ) ) {
return ( [ [ $ lpar_flag , "Node must be LPAR" , 1 ] ] ) ;
}
2011-11-24 09:24:56 +00:00
return & do_op_extra_cmds ( $ request , $ hash ) ;
2011-07-25 04:30:07 +00:00
}
2010-11-25 02:47:38 +00:00
2011-03-02 09:32:34 +00:00
2010-11-25 02:47:38 +00:00
##########################################################################
# Changes the configuration of an existing
2011-03-02 09:32:34 +00:00
# partition based on the profile specified
2010-11-25 02:47:38 +00:00
##########################################################################
2011-03-02 09:32:34 +00:00
sub modify_by_prof {
2010-11-25 02:47:38 +00:00
my $ request = shift ;
my $ hash = shift ;
my $ name = @ { $ request - > { node } } [ 0 ] ;
my $ opt = $ request - > { opt } ;
my @ values ;
2011-03-02 09:32:34 +00:00
my $ cfgdata = $ opt - > { profile } ;
my $ profile ;
my $ cec_name ;
my $ td ;
my % io = ( ) ;
my % lpar_state = ( ) ;
2011-03-24 02:02:15 +00:00
my @ result ;
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "$request->{command} START." ) ;
2011-03-07 06:18:23 +00:00
if ( defined ( $ request - > { stdin } ) ) {
my $ p = $ request - > { stdin } ;
my @ io = split ( /\n/ , $ p ) ;
foreach ( @ io ) {
chomp ;
2011-03-08 06:44:07 +00:00
if ( $ _ =~ /(\d+):(\s+)(\d+)\/([\w\.\-]+)\/(\w+)\// ) {
2011-03-07 06:18:23 +00:00
push @$ cfgdata , $ _ ;
} else {
return ( \ [ "Error" , "Invalid line in profile: $_" , - 1 ] ) ;
}
}
}
2011-03-24 02:02:15 +00:00
#print Dumper($cfgdata);
2011-03-02 09:32:34 +00:00
while ( my ( $ cec , $ h ) = each ( %$ hash ) ) {
while ( my ( $ lpar , $ d ) = each ( %$ h ) ) {
$ td = $ d ;
2012-07-03 06:31:58 +00:00
#@$td[4] = "fsp";
2011-03-02 09:32:34 +00:00
$ cec_name = @$ d [ 3 ] ;
}
2012-07-03 06:31:58 +00:00
$ td - > [ 4 ] = "cec" ;
2011-03-02 09:32:34 +00:00
#get the current I/O slot information
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "$request->{command} :get_io_slot_info for node:$cec_name." ) ;
2011-03-02 09:32:34 +00:00
my $ action = "get_io_slot_info" ;
2012-04-28 09:26:49 +00:00
my $ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ cec_name , $ td , $ action ) ;
2011-07-01 06:26:08 +00:00
my $ Rc = $$ values [ 2 ] ;
2011-03-02 09:32:34 +00:00
if ( $ Rc != 0 ) {
2011-07-01 06:26:08 +00:00
push @ result , [ $ cec_name , $$ values [ 1 ] , $ Rc ] ;
2011-03-24 02:02:15 +00:00
return ( \ @ result ) ;
2011-03-02 09:32:34 +00:00
}
2011-07-01 06:26:08 +00:00
my @ data = split ( /\n/ , $$ values [ 1 ] ) ;
2011-03-02 09:32:34 +00:00
foreach my $ v ( @ data ) {
my ( $ lparid , $ busid , $ location , $ drc_index , $ owner_type , $ owner , $ descr ) = split ( /,/ , $ v ) ;
$ io { $ drc_index } { lparid } = $ lparid ;
$ io { $ drc_index } { owner_type } = $ owner_type ;
$ io { $ drc_index } { owner } = $ owner ;
}
#get all the nodes state in the same cec
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "$request->{command} :get all the nodes state for CEC:$cec_name." ) ;
2011-03-02 09:32:34 +00:00
$ action = "all_lpars_state" ;
undef ( $ values ) ;
2012-04-28 09:26:49 +00:00
my $ values = xCAT::FSPUtils:: fsp_state_action ( $ request , $ cec_name , $ td , $ action ) ;
2011-03-02 09:32:34 +00:00
$ Rc = shift ( @$ values ) ;
if ( $ Rc != 0 ) {
2011-03-24 02:02:15 +00:00
push @ result , [ $ cec_name , $$ values [ 0 ] , $ Rc ] ;
return ( \ @ result ) ;
2011-03-02 09:32:34 +00:00
}
foreach ( @$ values ) {
my ( $ state , $ lparid ) = split /,/ ;
$ lpar_state { $ lparid } = $ state ;
}
}
##################################
# Check if LPAR profile exists
###################################
while ( my ( $ cec , $ h ) = each ( %$ hash ) ) {
while ( my ( $ lpar , $ d ) = each ( %$ h ) ) {
my $ id = @$ d [ 0 ] ;
2011-03-07 06:18:23 +00:00
#print Dumper($cfgdata);
2011-03-02 09:32:34 +00:00
my @ found = grep ( /^$id:/ , @$ cfgdata ) ;
2011-03-08 06:44:07 +00:00
#print Dumper(\@found);
2011-03-02 09:32:34 +00:00
my $ action = "set_io_slot_owner" ;
my $ tooltype = 0 ;
foreach my $ f ( @ found ) {
2011-03-08 06:44:07 +00:00
#'1: 514/U78A9.001.0123456-P1-C17/0x21010202/2/1'
my ( $ bus , $ location , $ drc_index , @ t ) = split ( /\// , $ f ) ;
2011-03-02 09:32:34 +00:00
my $ orig_id = $ io { $ drc_index } { lparid } ;
# the current owning lpar and the new owning lpar must be in power off state
2011-03-07 06:18:23 +00:00
if ( ( $ lpar_state { $ orig_id } ne "Not Activated" ) || ( $ lpar_state { $ id } ne "Not Activated" ) ) {
2011-03-24 02:02:15 +00:00
push @ result , [ $ lpar , "For the I/O $location, the current owning lpar(id=$orig_id) of the I/O and the new owning lpar(id=$id) must be in Not Activated state at first. And then run chvm again" , - 1 ] ;
return ( \ @ result ) ;
2011-03-02 09:32:34 +00:00
}
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "$request->{command} :set_io_slot_owner io_slot_info:$f,owner:$lpar." ) ;
2012-04-28 09:26:49 +00:00
my $ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ lpar , $ d , $ action , $ tooltype , $ drc_index ) ;
2011-03-24 02:02:15 +00:00
#my $Rc = shift(@$values);
my $ Rc = pop ( @$ values ) ;
2011-03-02 09:32:34 +00:00
if ( $ Rc != 0 ) {
2011-03-24 02:02:15 +00:00
push @ result , [ $ lpar , $$ values [ 1 ] , $ Rc ] ;
2010-11-25 02:47:38 +00:00
next ;
2011-03-02 09:32:34 +00:00
}
2010-11-25 02:47:38 +00:00
}
2011-03-02 09:32:34 +00:00
2010-11-25 02:47:38 +00:00
}
}
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "$request->{command} END." ) ;
2011-03-24 02:02:15 +00:00
return ( \ @ result ) ;
2010-11-25 02:47:38 +00:00
}
2011-03-02 09:32:34 +00:00
sub enumerate {
2012-04-28 09:26:49 +00:00
my $ request = shift ;
2011-03-02 09:32:34 +00:00
my $ h = shift ;
my $ mtms = shift ;
my % outhash = ( ) ;
my $ cec ;
my $ type ;
2011-03-04 02:51:55 +00:00
my @ td ;
2011-03-02 09:32:34 +00:00
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "lsvm :enumerate START for mtms:$mtms." ) ;
2011-03-02 09:32:34 +00:00
while ( my ( $ name , $ d ) = each ( %$ h ) ) {
$ cec = @$ d [ 3 ] ;
$ type = @$ d [ 4 ] ;
2011-03-04 02:51:55 +00:00
@ td = @$ d ;
2010-11-25 02:47:38 +00:00
}
2011-03-04 02:51:55 +00:00
2012-05-07 14:10:26 +00:00
$ td [ 4 ] = "cec" ;
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "lsvm :enumerate get_io_slot_info for node:$cec." ) ;
2011-03-02 09:32:34 +00:00
my $ action = "get_io_slot_info" ;
2012-04-28 09:26:49 +00:00
my $ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ cec , \ @ td , $ action ) ;
2011-06-13 01:31:23 +00:00
#my $Rc = shift(@$values);
2011-07-01 06:26:08 +00:00
my $ Rc = $$ values [ 2 ] ;
2011-03-02 09:32:34 +00:00
if ( $ Rc != 0 ) {
2011-06-13 01:31:23 +00:00
$ outhash { 1 } = "The LPARs' I/O slots information could NOT be listed because the cec is in power off state" ;
} else {
$ outhash { 0 } = $$ values [ 1 ] ;
}
2011-03-02 09:32:34 +00:00
#my @t;
#foreach my $value ( @$values ) {
# my ($lparid, $busid, $slot_location_code, $drc_index,@t ) = split (/,/, $value);
# push (@{$outhash{$lparid}}, $value);
#}
if ( $ type =~ /^(fsp|cec)$/ ) {
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "lsvm :enumerate query_octant_cfg for node:$cec." ) ;
2011-03-02 09:32:34 +00:00
$ action = "query_octant_cfg" ;
2012-04-28 09:26:49 +00:00
my $ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ cec , \ @ td , $ action ) ;
2011-07-01 06:26:08 +00:00
my $ Rc = pop ( @$ values ) ;
2011-03-02 09:32:34 +00:00
if ( $ Rc != 0 ) {
2011-07-01 06:26:08 +00:00
return ( [ $ Rc , $$ values [ 1 ] ] ) ;
2011-03-02 09:32:34 +00:00
}
2011-03-07 06:18:23 +00:00
#$outhash{ $cec } = @$values[0];
2011-07-01 06:26:08 +00:00
my $ data = $$ values [ 1 ] ;
2011-03-07 06:18:23 +00:00
my @ value = split ( /:/ , $ data ) ;
my $ pendingpumpmode = $ value [ 0 ] ;
my $ currentpumpMode = $ value [ 1 ] ;
my $ octantcount = $ value [ 2 ] ;
my $ j = 3 ;
2011-06-29 02:32:49 +00:00
my $ res = "PendingPumpMode=$pendingpumpmode,CurrentPumpMode=$currentpumpMode,OctantCount=$octantcount:\n" ;
2011-03-07 06:18:23 +00:00
for ( my $ i = 0 ; $ i < $ octantcount ; $ i + + ) {
2011-06-29 02:32:49 +00:00
$ res = $ res . "OctantID=" . $ value [ $ j + + ] . ",PendingOctCfg=" . $ value [ $ j + + ] . ",CurrentOctCfg=" . $ value [ $ j + + ] . ",PendingMemoryInterleaveMode=" . $ value [ $ j + + ] . ",CurrentMemoryInterleaveMode=" . $ value [ $ j + + ] . ";\n" ;
2011-03-07 06:18:23 +00:00
}
$ outhash { $ cec } = $ res ;
2011-03-02 09:32:34 +00:00
}
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "lsvm :enumerate END for mtms:$mtms." ) ;
2011-03-02 09:32:34 +00:00
return ( [ 0 , \ % outhash ] ) ;
2010-11-25 02:47:38 +00:00
}
2011-11-24 09:24:56 +00:00
sub get_cec_attr_info {
2012-04-28 09:26:49 +00:00
my $ request = shift ;
2011-11-24 09:24:56 +00:00
my $ name = shift ;
my $ attr = shift ;
my $ op = shift ;
my % op_hash = (
lpar_info = > "get_lpar_info" ,
bsr = > "get_cec_bsr" ,
huge_page = > "get_huge_page"
) ;
my $ action = $ op_hash { $ op } ;
2012-04-28 09:26:49 +00:00
my $ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ attr , $ action ) ;
2011-07-25 04:30:07 +00:00
if ( @$ values [ 1 ] && ( ( @$ values [ 1 ] =~ /Error/i ) && @$ values [ 2 ] ne '0' ) ) {
return ( [ [ $ name , @$ values [ 1 ] , '1' ] ] ) ;
}
return @$ values [ 1 ] ;
}
2011-11-24 09:24:56 +00:00
sub get_cec_lpar_hugepage {
my $ name = shift ;
my $ huge_info = shift ;
my $ lparid = shift ;
my $ lparname = shift ;
my @ value = split ( /\n/ , $ huge_info ) ;
foreach my $ v ( @ value ) {
if ( $ v =~ /\s*([^\s]+)\s*:\s*([\d|\/]+)/ ) {
my $ tmp_name = $ 1 ;
my $ tmp_num = $ 2 ;
if ( $ tmp_name =~ /^$lparname$/ ) {
return $ tmp_num ;
}
}
2011-10-20 05:43:04 +00:00
}
2011-11-24 09:24:56 +00:00
return ( [ [ $ name , "can not get huge page info for lpar id $lparid" , '1' ] ] ) ;
2011-10-20 05:43:04 +00:00
2011-11-24 09:24:56 +00:00
}
2011-10-20 05:43:04 +00:00
2011-07-25 04:30:07 +00:00
sub get_cec_lpar_name {
my $ name = shift ;
my $ lpar_info = shift ;
my $ lparid = shift ;
my @ value = split ( /\n/ , $ lpar_info ) ;
foreach my $ v ( @ value ) {
if ( $ v =~ /lparname:\s*([^\,]*),\s*lparid:\s*([\d]+),/ ) {
if ( $ 2 == $ lparid ) {
return $ 1 ;
}
}
}
return ( [ [ $ name , "can not get lparname for lpar id $lparid" , '1' ] ] ) ;
2011-10-20 05:43:04 +00:00
}
sub get_cec_lpar_bsr {
my $ name = shift ;
my $ lpar_info = shift ;
my $ lparid = shift ;
my $ lparname = shift ;
my @ value = split ( /\n/ , $ lpar_info ) ;
foreach my $ v ( @ value ) {
if ( $ v =~ /\s*([^\s]+)\s*:\s*([\d]+)/ ) {
my $ tmp_name = $ 1 ;
my $ tmp_num = $ 2 ;
if ( $ tmp_name =~ /^$lparname$/ ) {
return $ tmp_num ;
}
}
}
return ( [ [ $ name , "can not get BSR info for lpar id $lparid" , '1' ] ] ) ;
}
sub get_cec_cec_bsr {
my $ name = shift ;
my $ lpar_info = shift ;
my $ index = 0 ;
my @ value = split ( /\n/ , $ lpar_info ) ;
my $ cec_bsr = "" ;
foreach my $ v ( @ value ) {
if ( $ v =~ /(Number of BSR arrays:)\s*(\d+)/i ) {
$ cec_bsr . = "$1 $2," ;
$ index + + ;
} elsif ( $ v =~ /(Bytes per BSR array:)\s*(\d+)/i ) {
$ cec_bsr . = "$1 $2," ;
$ index + + ;
} elsif ( $ v =~ /(Available BSR array:)\s*(\d+)/i ) {
$ cec_bsr . = "$1 $2;\n" ;
$ index + + ;
}
}
if ( $ index != 3 ) {
return undef ;
} else {
return $ cec_bsr ;
}
2011-07-25 04:30:07 +00:00
}
2011-11-24 09:24:56 +00:00
sub get_cec_cec_hugepage {
my $ name = shift ;
my $ huge_info = shift ;
my $ index = 0 ;
my @ value = split ( /\n/ , $ huge_info ) ;
my $ cec_hugepage = "" ;
foreach my $ v ( @ value ) {
if ( $ v =~ /(Available huge page memory\(in pages\):)\s*(\d+)/i ) {
my $ tmp = sprintf "%-40s %s;\n" , $ 1 , $ 2 ;
$ cec_hugepage . = $ tmp ;
$ index + + ;
} elsif ( $ v =~ /(Configurable huge page memory\(in pages\):)\s*(\d+)/i ) {
my $ tmp = sprintf "%-40s %s;\n" , $ 1 , $ 2 ;
$ cec_hugepage . = $ tmp ;
$ index + + ;
} elsif ( $ v =~ /(Page Size\(in GB\):)\s*(\d+)/i ) {
my $ tmp = sprintf "%-40s %s;\n" , $ 1 , $ 2 ;
$ cec_hugepage . = $ tmp ;
$ index + + ;
} elsif ( $ v =~ /(Maximum huge page memory\(in pages\):)\s*(\d+)/i ) {
my $ tmp = sprintf "%-40s %s;\n" , $ 1 , $ 2 ;
$ cec_hugepage . = $ tmp ;
$ index + + ;
} elsif ( $ v =~ /(Requested huge page memory\(in pages\):)\s*(\d+)/i ) {
my $ tmp = sprintf "%-40s %s;\n" , $ 1 , $ 2 ;
$ cec_hugepage . = $ tmp ;
$ index + + ;
}
}
if ( $ index != 5 ) {
return undef ;
}
return $ cec_hugepage ;
2011-07-25 04:30:07 +00:00
}
2010-11-25 02:47:38 +00:00
##########################################################################
2011-03-02 09:32:34 +00:00
# Lists logical partitions
2010-11-25 02:47:38 +00:00
##########################################################################
2011-03-02 09:32:34 +00:00
sub list {
2010-11-25 02:47:38 +00:00
my $ request = shift ;
my $ hash = shift ;
2011-03-02 09:32:34 +00:00
my $ args = $ request - > { opt } ;
my $ values = ( ) ;
my @ value = ( ) ;
my $ node_name ;
my $ d ;
my @ result ;
2011-07-25 04:30:07 +00:00
my $ lpar_infos ;
2011-10-20 05:43:04 +00:00
my $ bsr_infos ;
2011-11-24 09:24:56 +00:00
my $ huge_infos ;
2011-12-15 06:55:40 +00:00
my % lpar_huges = ( ) ;
2011-11-24 09:24:56 +00:00
my $ l_string = "\n" ;
2011-03-24 02:02:15 +00:00
#print Dumper($hash);
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "lsvm START" ) ;
2011-03-02 09:32:34 +00:00
while ( my ( $ mtms , $ h ) = each ( %$ hash ) ) {
2012-04-28 09:26:49 +00:00
my $ info = enumerate ( $ request , $ h , $ mtms ) ;
2011-10-20 05:43:04 +00:00
my $ Rc = shift ( @$ info ) ;
my $ data = @$ info [ 0 ] ;
2011-03-04 02:51:55 +00:00
2011-03-02 09:32:34 +00:00
while ( ( $ node_name , $ d ) = each ( %$ h ) ) {
my $ cec = @$ d [ 3 ] ;
my $ type = @$ d [ 4 ] ;
my $ id = @$ d [ 0 ] ;
2011-10-20 05:43:04 +00:00
if ( $ Rc != SUCCESS ) {
push @ result , [ $ node_name , $ data , $ Rc ] ;
next ;
}
2011-07-25 04:30:07 +00:00
my $ values = $ data - > { 0 } ;
my $ msg = $ data - > { 1 } ;
2011-03-02 09:32:34 +00:00
# if ( !exists( $data->{$id} )) {
# push @result, [$node_name, "Node not found",1];
# next;
# }
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "lsvm :parse io info for node:$node_name." ) ;
2011-06-13 01:31:23 +00:00
if ( defined ( $ msg ) ) {
push @ result , [ $ node_name , $ msg , 0 ] ;
} else {
# get the I/O slot information
2011-11-24 09:24:56 +00:00
if ( $ request - > { opt } - > { l } ) {
if ( $ type =~ /^(fsp|cec)$/ ) {
2012-04-28 09:26:49 +00:00
$ bsr_infos = get_cec_attr_info ( $ request , $ node_name , $ d , "bsr" ) ;
2011-11-24 09:24:56 +00:00
if ( ref ( $ bsr_infos ) eq 'ARRAY' ) {
return $ bsr_infos ;
}
2012-04-28 09:26:49 +00:00
$ huge_infos = get_cec_attr_info ( $ request , $ node_name , $ d , "huge_page" ) ;
2011-12-15 06:55:40 +00:00
if ( ref ( $ huge_infos ) eq 'ARRAY' ) {
return $ huge_infos ;
}
}
2012-04-28 09:26:49 +00:00
$ lpar_infos = get_cec_attr_info ( $ request , $ node_name , $ d , "lpar_info" ) ;
2011-07-25 04:30:07 +00:00
if ( ref ( $ lpar_infos ) eq 'ARRAY' ) {
return $ lpar_infos ;
}
2011-11-24 09:24:56 +00:00
}
2011-06-13 01:31:23 +00:00
my $ v ;
my @ t ;
my @ value = split ( /\n/ , $ values ) ;
foreach my $ v ( @ value ) {
my ( $ lparid , @ t ) = split ( /,/ , $ v ) ;
2011-10-20 05:43:04 +00:00
my $ ios = join ( '/' , @ t ) ;
2011-07-25 04:30:07 +00:00
if ( $ request - > { opt } - > { l } ) {
2011-11-24 09:24:56 +00:00
my $ lparname = get_cec_lpar_name ( $ node_name , $ lpar_infos , $ lparid ) ;
2011-12-15 06:55:40 +00:00
my $ hugepage ;
2011-07-25 04:30:07 +00:00
if ( $ type =~ /^(fsp|cec)$/ ) {
2011-10-20 05:43:04 +00:00
my $ lpar_bsr = get_cec_lpar_bsr ( $ node_name , $ bsr_infos , $ lparid , $ lparname ) ;
if ( ref ( $ lpar_bsr ) eq 'ARRAY' ) {
return $ lpar_bsr ;
}
$ ios . = ": " . $ lpar_bsr ;
2011-12-15 06:55:40 +00:00
$ hugepage = get_cec_lpar_hugepage ( $ node_name , $ huge_infos , $ lparid , $ lparname ) ;
if ( ref ( $ hugepage ) eq 'ARRAY' ) {
return $ hugepage ;
}
2011-07-25 04:30:07 +00:00
} else {
2011-11-24 09:24:56 +00:00
if ( $ lparid ne $ id ) {
next ;
2011-12-15 06:55:40 +00:00
}
if ( defined ( $ lpar_huges { $ lparid } ) ) {
$ hugepage = $ lpar_huges { $ lparid } ;
} else {
2012-04-28 09:26:49 +00:00
$ hugepage = get_cec_attr_info ( $ request , $ node_name , $ d , "huge_page" ) ;
2011-12-15 06:55:40 +00:00
if ( ref ( $ hugepage ) eq 'ARRAY' ) {
return $ hugepage ;
}
$ lpar_huges { $ lparid } = $ hugepage ;
}
2011-11-24 09:24:56 +00:00
}
2011-12-15 06:55:40 +00:00
$ ios . = ": " . $ hugepage ;
2011-07-25 04:30:07 +00:00
if ( ref ( $ lparname ) eq 'ARRAY' ) {
return $ lparname ;
} else {
$ lparname = "$lparname: $lparid" ;
}
2011-10-20 05:43:04 +00:00
$ l_string . = "$lparname: " . $ ios . "\n" ;
2011-07-25 04:30:07 +00:00
} else {
2011-10-20 05:43:04 +00:00
if ( $ type =~ /^(fsp|cec)$/ ) {
push @ result , [ $ lparid , $ ios , $ Rc ] ;
} else {
if ( $ lparid eq $ id ) {
push @ result , [ $ lparid , $ ios , $ Rc ] ;
}
2011-06-13 01:31:23 +00:00
}
2010-11-25 02:47:38 +00:00
}
2011-03-02 09:32:34 +00:00
}
2010-11-25 02:47:38 +00:00
}
2011-03-02 09:32:34 +00:00
# get the octant configuration value
if ( $ type =~ /^(fsp|cec)$/ ) {
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "lsvm :parse octant info for $type:$node_name." ) ;
2011-03-02 09:32:34 +00:00
my $ value = $ data - > { $ cec } ;
2011-10-20 05:43:04 +00:00
if ( $ request - > { opt } - > { l } ) {
my $ cec_bsr = get_cec_cec_bsr ( $ node_name , $ bsr_infos ) ;
2011-11-24 09:24:56 +00:00
my $ cec_hugepage = get_cec_cec_hugepage ( $ node_name , $ huge_infos ) ;
$ l_string . = $ value . $ cec_bsr ;
$ l_string . = $ cec_hugepage ;
2011-10-20 05:43:04 +00:00
} else {
$ l_string = $ value ;
}
2011-03-02 09:32:34 +00:00
}
2012-03-05 02:03:12 +00:00
if ( $ l_string =~ /^\n$/ ) {
next ;
}
2011-10-20 05:43:04 +00:00
push @ result , [ $ node_name , $ l_string , $ Rc ] ;
2011-11-24 09:24:56 +00:00
$ l_string = "\n" ;
2011-10-20 05:43:04 +00:00
} # end of while
2011-03-02 09:32:34 +00:00
} # end of while
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "lsvm END." ) ;
2011-03-02 09:32:34 +00:00
return ( \ @ result ) ;
2010-11-25 02:47:38 +00:00
}
2011-03-02 09:32:34 +00:00
2010-11-25 02:47:38 +00:00
##########################################################################
# Lists logical partitions
##########################################################################
2011-03-02 09:32:34 +00:00
sub list_orig {
2010-11-25 02:47:38 +00:00
my $ request = shift ;
my $ hash = shift ;
my $ args = $ request - > { opt } ;
my $ values = ( ) ;
my @ value = ( ) ;
my $ node_name ;
my $ d ;
my @ result ;
while ( my ( $ mtms , $ h ) = each ( %$ hash ) ) {
while ( ( $ node_name , $ d ) = each ( %$ h ) ) {
my $ lparid = @$ d [ 0 ] ;
my $ mtms = @$ d [ 2 ] ;
my $ type = @$ d [ 4 ] ;
my $ pprofile ;
####################################
# Must be CEC or LPAR
####################################
2011-03-02 09:32:34 +00:00
if ( $ type !~ /^(lpar|fsp|cec)$/ ) {
2010-11-25 02:47:38 +00:00
#$values->{$lpar} = [$lpar,"Node must be LPAR or CEC",RC_ERROR];
return ( [ $ node_name , "Node must be LPAR or CEC" , RC_ERROR ] ) ;
#next;
}
####################################
# This is a single LPAR
####################################
if ( $ type eq "lpar" ) {
#$lpars[0] = "$lpar,$lparid";
2011-03-02 09:32:34 +00:00
2010-11-25 02:47:38 +00:00
}
####################################
# This is a CEC
####################################
else {
2012-04-28 09:26:49 +00:00
my $ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ node_name , $ d , "query_octant_cfg" ) ;
2010-11-25 02:47:38 +00:00
my $ Rc = @$ values [ 2 ] ;
my $ data = @$ values [ 1 ] ;
if ( $ Rc != SUCCESS ) {
push @ result , [ $ node_name , $ data , $ Rc ] ;
} else {
my @ value = split ( /,/ , $ data ) ;
my $ pendingpumpmode = $ value [ 0 ] ;
my $ currentpumpMode = $ value [ 1 ] ;
my $ octantcount = $ value [ 2 ] ;
my $ j = 3 ;
my $ res = "PendingPumpMode=$pendingpumpmode,CurrentPumpMode=$currentpumpMode,OctantCount=$octantcount:" ;
for ( my $ i = 0 ; $ i < $ octantcount ; $ i + + ) {
$ res = $ res . "OctantID=" . $ value [ $ j + + ] . ",PendingOctCfg=" . $ value [ $ j + + ] . ",CurrentOctCfg=" . $ value [ $ j + + ] . ",PendingMemoryInterleaveMode=" . $ value [ $ j + + ] . ",CurrentMemoryInterleaveMode" . $ value [ $ j + + ] . ";" ;
}
push @ result , [ $ node_name , $ res , $ Rc ] ;
}
}
}
}
return ( \ @ result ) ;
}
##########################################################################
# Creates/changes logical partitions
##########################################################################
sub create {
my $ request = shift ;
my $ hash = shift ;
my $ opt = $ request - > { opt } ;
my @ values = ( ) ;
my @ result ;
2011-03-04 01:55:51 +00:00
my $ cec_name ;
2010-11-25 02:47:38 +00:00
my $ d ;
my $ lparid ;
my $ mtms ;
my $ type ;
my $ profile ;
my $ starting_lpar_id = $ opt - > { i } ;
2011-03-02 09:32:34 +00:00
my $ octant_cfg = $ opt - > { octant_cfg } ;
2011-02-16 10:32:45 +00:00
my $ node_number = @ { $ opt - > { target } } ;
2011-03-04 01:55:51 +00:00
my % node_id = ( ) ;
my @ nodes = @ { $ opt - > { target } } ;
2011-03-08 06:44:07 +00:00
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "$request->{command} START." ) ;
2011-03-08 06:44:07 +00:00
#print Dumper($request);
2010-11-25 02:47:38 +00:00
#####################################
# Get source node information
#####################################
while ( my ( $ cec , $ h ) = each ( %$ hash ) ) {
while ( my ( $ name , $ data ) = each ( %$ h ) ) {
$ d = $ data ;
$ lparid = @$ d [ 0 ] ;
$ mtms = @$ d [ 2 ] ;
$ type = @$ d [ 4 ] ;
2011-03-04 01:55:51 +00:00
$ cec_name = $ name ;
2010-11-25 02:47:38 +00:00
#####################################
# Must be LPAR
#####################################
2011-03-02 09:32:34 +00:00
if ( $ type !~ /^(fsp|cec)$/ ) {
2011-03-04 01:55:51 +00:00
return ( [ [ $ cec_name , "Node's parent must be fsp or CEC" , RC_ERROR ] ] ) ;
2010-11-25 02:47:38 +00:00
}
}
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "$request->{command} :query_octant_cfg for CEC:$cec_name." ) ;
2012-04-28 09:26:49 +00:00
my $ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ cec_name , $ d , "query_octant_cfg" ) ;
2011-03-02 09:32:34 +00:00
my $ Rc = shift ( @$ values ) ;
if ( $ Rc != 0 ) {
2011-03-04 01:55:51 +00:00
return ( [ [ $ cec_name , $$ values [ 0 ] , $ Rc ] ] ) ;
2011-03-02 09:32:34 +00:00
}
2011-07-01 06:26:08 +00:00
my @ v = split ( /:/ , $$ values [ 0 ] ) ;
2011-03-02 09:32:34 +00:00
$ octant_cfg - > { pendingpumpmode } = $ v [ 0 ] ;
2011-03-04 01:55:51 +00:00
my $ number_of_lpars_per_octant ;
my $ octant_num_needed ;
my $ starting_octant_id ;
my $ octant_conf_value ;
my $ octant_cfg_value = $ octant_cfg - > { octant_cfg_value } ;
my $ new_pending_interleave_mode = $ octant_cfg - > { memory_interleave } ;
$ starting_octant_id = int ( $ starting_lpar_id / 4 ) ;
my $ lparnum_from_octant = 0 ;
my $ new_pending_pump_mode = $ octant_cfg - > { pendingpumpmode } ;
my $ parameters ;
#my $parameters = "$new_pending_pump_mode:$octant_num_needed";
my $ octant_id = $ starting_octant_id ;
my $ i = 0 ;
my $ res ;
for ( $ i = 0 ; $ i < ( keys %$ octant_cfg_value ) ; $ i + + ) {
if ( ! exists ( $ octant_cfg_value - > { $ octant_id } ) ) {
2011-03-07 06:18:23 +00:00
$ res = "starting LPAR id is $starting_lpar_id, starting octant id is $starting_octant_id. The octants should be used continuously. Octant $octant_id configuration value isn't provided. Wrong plan." ;
2011-03-04 01:55:51 +00:00
return ( [ [ $ cec_name , $ res , - 1 ] ] ) ;
}
my $ octant_conf_value = $ octant_cfg_value - > { $ octant_id } ;
#octant configuration values could be 1,2,3,4,5 ; AS following:
# 1 - 1 partition with all cpus and memory of the octant
# 2 - 2 partitions with a 50/50 split of cpus and memory
# 3 - 3 partitions with a 25/25/50 split of cpus and memory
# 4 - 4 partitions with a 25/25/25/25 split of cpus and memory
# 5 - 2 partitions with a 25/75 split of cpus and memory
if ( $ octant_conf_value == 1 ) {
$ number_of_lpars_per_octant = 1 ;
} elsif ( $ octant_conf_value == 2 ) {
$ number_of_lpars_per_octant = 2 ;
} elsif ( $ octant_conf_value == 3 ) {
$ number_of_lpars_per_octant = 3 ;
} elsif ( $ octant_conf_value == 4 ) {
$ number_of_lpars_per_octant = 4 ;
} elsif ( $ octant_conf_value == 5 ) {
$ number_of_lpars_per_octant = 2 ;
} else {
$ res = "octant $i, configuration values: $octant_conf_value. Wrong octant configuration values!\n" ;
return ( [ [ $ cec_name , $ res , - 1 ] ] ) ;
}
my $ j ;
for ( $ j = 1 ; $ j < $ number_of_lpars_per_octant + 1 ; $ j + + ) {
if ( @ nodes ) {
my $ node = shift ( @ nodes ) ;
$ node_id { $ node } = $ j + $ octant_id * 4 ;
}
}
$ lparnum_from_octant += $ number_of_lpars_per_octant ;
$ octant_num_needed + + ;
$ parameters . = ":$octant_id:$octant_conf_value:$new_pending_interleave_mode" ;
$ octant_id + + ;
}
$ parameters = "$new_pending_pump_mode:$octant_num_needed" . $ parameters ;
##if($node_number != $lparnum_from_octant ) {##
if ( $ node_number > $ lparnum_from_octant ) {
$ res = "According to the partition split rule and the starting LPAR id, $lparnum_from_octant LPARs will be gotten. But the noderange has $node_number node. Wrong plan.\n" ;
return ( [ [ $ cec_name , $ res , - 1 ] ] ) ;
}
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "$request->{command} :set_octant_cfg for CEC:$cec_name,param:$parameters." ) ;
2011-03-04 01:55:51 +00:00
#$values = xCAT::FSPUtils::fsp_api_create_parttion( $starting_lpar_id, $octant_cfg, $node_number, $d, "set_octant_cfg");
2012-04-28 09:26:49 +00:00
$ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ cec_name , $ d , "set_octant_cfg" , 0 , $ parameters ) ;
2011-07-01 06:26:08 +00:00
my $ Rc = $$ values [ 2 ] ;
my $ data = $$ values [ 1 ] ;
2010-11-25 02:47:38 +00:00
if ( $ Rc != SUCCESS ) {
2011-03-04 01:55:51 +00:00
push @ result , [ $ cec_name , $ data , $ Rc ] ;
2010-11-25 02:47:38 +00:00
} else {
foreach my $ name ( @ { $ opt - > { target } } ) {
2011-03-08 06:44:07 +00:00
push @ result , [ $ name , "Success" , $ Rc ] ;
2011-03-04 07:54:01 +00:00
xCAT::FSPvm:: xCATdB ( "mkvm" , $ name , "" , $ node_id { $ name } , $ d , "fsp" , $ name ) ;
2010-11-25 02:47:38 +00:00
}
2011-03-08 06:44:07 +00:00
push @ result , [ $ cec_name , "Please reboot the CEC $cec_name before using chvm to assign the I/O slots to the LPARs" , "mkvm" ] ;
#$request->{callback}->({info => ["Please reboot the CEC $cec_name before using chvm to assign the I/O slots to the LPARs"]});
2010-11-25 02:47:38 +00:00
}
}
2012-06-21 09:43:16 +00:00
xCAT::MsgUtils - > verbose_message ( $ request , "$request->{command} END." ) ;
2010-11-25 02:47:38 +00:00
return ( \ @ result ) ;
}
##########################################################################
# Adds/removes LPARs from the xCAT database
##########################################################################
sub xCATdB {
my $ cmd = shift ;
my $ name = shift ;
my $ profile = shift ;
my $ lparid = shift ;
my $ d = shift ;
my $ hwtype = shift ;
my $ lpar = shift ;
my $ parent = shift ;
#######################################
# Remove entry
#######################################
if ( $ cmd eq "rmvm" ) {
return ( xCAT::PPCdb:: rm_ppc ( $ name ) ) ;
}
#######################################
# Change entry
#######################################
elsif ( $ cmd eq "chvm" ) {
my $ ppctab = xCAT::Table - > new ( "ppc" , - create = > 1 , - autocommit = > 1 ) ;
###################################
# Error opening ppc database
###################################
if ( ! defined ( $ ppctab ) ) {
return ( "Error opening 'ppc' database" ) ;
}
$ ppctab - > setNodeAttribs ( $ name , { pprofile = > $ profile } ) ;
}
#######################################
# Add entry
#######################################
else {
if ( ! defined ( $ profile ) ) {
$ profile = $ name ;
}
my ( $ model , $ serial ) = split /\*/ , @$ d [ 2 ] ;
my $ server = @$ d [ 3 ] ;
my $ fsp = @$ d [ 2 ] ;
###################################
# Find FSP name in ppc database
###################################
my $ tab = xCAT::Table - > new ( "ppc" ) ;
###################################
# Error opening ppc database
###################################
if ( ! defined ( $ tab ) ) {
return ( "Error opening 'ppc' database" ) ;
}
###################################
# If there is no parent provided
# this lpar should be the cloned
# in the same cec
# Otherwise it should be cloned
# between cecs
###################################
if ( ! $ parent )
{
my ( $ ent ) = $ tab - > getNodeAttribs ( $ lpar , [ 'parent' ] ) ;
###################################
# Node not found
###################################
if ( ! defined ( $ ent ) ) {
return ( "'$lpar' not found in 'ppc' database" ) ;
}
###################################
# Attributes not found
###################################
if ( ! exists ( $ ent - > { parent } ) ) {
return ( "'parent' attribute not found in 'ppc' database" ) ;
}
$ parent = $ ent - > { parent } ;
}
my $ values = join ( "," ,
"lpar" ,
$ name ,
$ lparid ,
$ model ,
$ serial ,
"" ,
$ server ,
$ profile ,
$ parent ) ;
return ( xCAT::PPCdb:: add_ppc ( $ hwtype , [ $ values ] ) ) ;
}
return undef ;
}
2013-09-04 23:27:28 -07:00
########################
#***** partition related
########################
#my @partition_query_actions = qw(part_get_partition_cap part_get_num_of_lpar_slots part_get_hyp_config_process_and_mem part_get_hyp_avail_process_and_mem part_get_service_authority_lpar_id part_get_shared_processing_resource part_get_all_vio_info lpar_lhea_mac part_get_all_io_bus_info part_get_lpar_processing part_get_lpar_memory get_huge_page get_cec_bsr);
my @ partition_query_actions = qw( part_get_partition_cap part_get_hyp_process_and_mem part_get_all_io_bus_info get_huge_page get_cec_bsr ) ;
sub parse_part_get_info {
my $ hash = shift ;
my $ data = shift ;
my @ array = split /\n/ , $ data ;
foreach my $ line ( @ array ) {
chomp ( $ line ) ;
if ( $ line =~ /Num of lpar slots: (\d+)/i ) {
$ hash - > { num_of_lpars } = $ 1 ;
} elsif ( $ line =~ /HYP Configurable Memory[^\(]*\((\d+)\s*regions\)/i ) {
$ hash - > { hyp_config_mem } = $ 1 ;
} elsif ( $ line =~ /HYP Available Memory[^\(]*\((\d+)\s*regions\)/i ) {
$ hash - > { hyp_avail_mem } = $ 1 ;
} elsif ( $ line =~ /HYP Memory Region Size[^\(]*\((\d+)\s*MB\)/i ) {
$ hash - > { mem_region_size } = $ 1 ;
} elsif ( $ line =~ /HYP Configurable Processors: (\d+),\s*Avail Processors: (\d+)/i ) {
$ hash - > { process_units_config } = $ 1 ;
$ hash - > { process_units_avail } = $ 2 ;
} elsif ( $ line =~ /Authority Lpar id:(\w+)/i ) {
$ hash - > { service_lparid } = $ 1 ;
} elsif ( $ line =~ /(\d+),(\d+),[^,]*,(\w+),[^,]*,[^,]*,\w*\(([\w| |-|_]*)\)/ ) {
$ hash - > { bus } - > { $ 3 } - > { cur_lparid } = $ 1 ;
$ hash - > { bus } - > { $ 3 } - > { bus_slot } = $ 2 ;
$ hash - > { bus } - > { $ 3 } - > { des } = $ 4 ;
} elsif ( $ line =~ /Phy drc_index:(\w+), Port group: (\w+), Phy port id: (\w+)/ ) {
$ hash - > { phy_drc_group_port } - > { $ 1 } - > { $ 2 } - > { $ 3 } = '1' ;
} elsif ( $ line =~ /adapter_id=(\w+),lpar_id=([\d|-]+).*port_group=(\d+),phys_port_id=(\d+).*drc_index=(\w+),.*/ ) {
if ( ( $ 2 == - 1 ) && ( $ 4 == 255 ) ) {
$ hash - > { logic_drc_phydrc } - > { $ 3 } - > { $ 5 } = $ 1 ;
#$hash->{logic_drc_phydrc}->{$5}->{$1} = [$2,$3,$4];
}
#} elsif ($line =~ /lpar 0:: Curr Memory::min: 1,cur: (\d+),max:/i) {
} elsif ( $ line =~ /Curr Memory Req[^\(]*\((\d+)\s*regions\)/i ) {
$ hash - > { lpar0_used_mem } = $ 1 ;
#print "===>lpar0_used_mem:$hash->{lpar0_used_mem}.\n";
} elsif ( $ line =~ /Available huge page memory\(in pages\):\s*(\d+)/ ) {
$ hash - > { huge_page_avail } = $ 1 ;
} elsif ( $ line =~ /Available BSR array:\s*(\d+)/ ) {
$ hash - > { cec_bsr_avail } = $ 1 ;
}
}
}
sub query_cec_info_actions {
my $ request = shift ;
my $ name = shift ;
my $ td = shift ;
my $ usage = shift ;
my $ action_array = shift ;
my $ lparid = @$ td [ 0 ] ;
my $ data ;
my @ array = ( ) ;
my % hash = ( ) ;
if ( ! defined ( $ action_array ) or ref ( $ action_array ) ne "ARRAY" ) {
$ action_array = \ @ partition_query_actions ;
}
foreach my $ action ( @$ action_array ) {
#$data .= "======> ret info for $action:\n";
my $ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ td , $ action ) ;
chomp ( @$ values [ 1 ] ) ;
if ( @$ values [ 1 ] =~ /^$/ ) {
next ;
}
if ( $ usage eq 0 ) {
if ( $ lparid ) {
if ( $ action eq "lpar_lhea_mac" ) {
my @ output = split /\n/ , @$ values [ 1 ] ;
foreach my $ line ( @ output ) {
if ( $ line =~ /adapter_id=\w+,lpar_id=$lparid,type=hea/ ) {
#$data .= "$line\n";
push @ array , [ $ name , $ line , 0 ] ;
}
}
#$data .= "\n";
next ;
}
if ( $ action eq "part_get_all_io_bus_info" ) {
my @ output = split /\n/ , @$ values [ 1 ] ;
foreach my $ line ( @ output ) {
if ( $ line =~ /$lparid,/ ) {
#$data .= "$line\n";
push @ array , [ $ name , $ line , 0 ] ;
}
}
#$data .= "\n";
next ;
}
}
#$data .= "@$values[1]\n\n";
push @ array , [ $ name , @$ values [ 1 ] , @$ values [ 2 ] ] ;
} else {
& parse_part_get_info ( \ % hash , @$ values [ 1 ] ) ;
}
}
if ( $ usage eq 0 ) {
#return $data;
return \ @ array ;
} else {
return \ % hash ;
}
}
#my @partition_query_actions = qw(part_get_partition_cap part_get_num_of_lpar_slots part_get_hyp_config_process_and_mem part_get_hyp_avail_process_and_mem part_get_service_authority_lpar_id part_get_shared_processing_resource part_get_all_vio_info lpar_lhea_mac part_get_all_io_bus_info part_get_lpar_processing part_get_lpar_memory get_huge_page get_cec_bsr);
sub query_cec_info {
my $ request = shift ;
my $ hash = shift ;
my $ args = $ request - > { opt } ;
my @ td = ( ) ;
my @ result = ( ) ;
while ( my ( $ mtms , $ h ) = each ( %$ hash ) ) {
while ( my ( $ name , $ d ) = each ( %$ h ) ) {
@ td = @$ d ;
if ( @$ d [ 0 ] == 0 && @$ d [ 4 ] ne "lpar" ) {
last ;
}
#my $rethash = query_cec_info_actions($request, $name, $d, 0, ["part_get_lpar_processing","part_get_lpar_memory","part_get_all_vio_info","lpar_lhea_mac","part_get_all_io_bus_info","get_huge_page","get_cec_bsr"]);
my $ rethash = query_cec_info_actions ( $ request , $ name , $ d , 0 , [ "part_get_lpar_processing" , "part_get_lpar_memory" , "part_get_all_io_bus_info" , "get_huge_page" , "get_cec_bsr" ] ) ;
#push @result, [$name, $rethash, 0];
push @ result , @$ rethash ;
}
if ( @ td [ 0 ] == 0 ) {
my $ rethash = query_cec_info_actions ( $ request , @ td [ 3 ] , \ @ td , 0 ) ;
#push @result, [@td[3], $rethash, 0];
push @ result , @$ rethash ;
}
}
return \ @ result ;
}
########################
#***** partition related
########################
my @ partition_config_actions = qw/part_set_lpar_def_state part_set_lpar_pending_proc part_set_lpar_pending_mem part_set_pending_max_vslots part_set_lpar_shared_pool_util_auth part_set_lpar_group_id part_set_lpar_avail_priority part_set_partition_placement part_set_lhea_assign_info part_set_phea_port_info part_set_lhea_port_info part_set_veth_slot_config part_set_vscsi_slot_config part_set_vfchan_slot_config part_clear_vslot_config set_huge_page set_lpar_name/ ;
sub set_lpar_undefined {
my $ request = shift ;
my $ name = shift ;
my $ attr = shift ;
my $ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ attr , "part_set_lpar_def_state" , 0 , 0x0 ) ;
if ( ! @$ values [ 2 ] ) {
return ( [ $ name , "Done" , 0 ] ) ;
}
return $ values ;
}
sub clear_service_authority_lpar {
my $ request = shift ;
my $ name = shift ;
my $ attr = shift ;
my $ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ attr , "part_get_service_authority_lpar_id" ) ;
my @ array = split /\n/ , @$ values [ 1 ] ;
my $ service_lparid = undef ;
foreach my $ line ( @ array ) {
if ( $ line =~ /Authority Lpar id:([-|\d]+)./i ) {
$ service_lparid = $ 1 ;
}
}
if ( defined ( $ service_lparid ) and $ service_lparid == @$ attr [ 0 ] ) {
xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ attr , "part_set_service_authority_lpar_id" ) ;
}
}
sub remove {
my $ request = shift ;
my $ hash = shift ;
my @ result = ( ) ;
while ( my ( $ mtms , $ h ) = each ( %$ hash ) ) {
while ( my ( $ name , $ d ) = each ( %$ h ) ) {
& clear_service_authority_lpar ( $ request , $ name , $ d ) ;
my $ values = & set_lpar_undefined ( $ request , $ name , $ d ) ;
push @ result , $ values ;
}
}
return \ @ result ;
}
sub deal_with_avail_mem {
my $ request = shift ;
my $ name = shift ;
my $ d = shift ;
my $ lparhash = shift ;
my ( $ before , $ after , $ res ) ;
my @ td = @$ d ;
@ td [ 0 ] = 0 ;
my $ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , \ @ td , "part_get_lpar_memory" ) ;
my % tmphash ;
& parse_part_get_info ( \ % tmphash , @$ values [ 1 ] ) ;
if ( exists ( $ tmphash { lpar0_used_mem } ) ) {
$ before = $ tmphash { lpar0_used_mem } ;
} else {
return ( [ $ name , "part_get_lpar_memory failed to get used memory for hypervisor." , 1 ] ) ;
}
my $ tmp_param = "1/1/" . $ lparhash - > { hyp_config_mem } ;
$ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lpar_pending_mem" , 0 , $ tmp_param ) ;
if ( @$ values [ 2 ] ) {
return $ values ;
}
$ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , \ @ td , "part_get_lpar_memory" ) ;
& parse_part_get_info ( \ % tmphash , @$ values [ 1 ] ) ;
if ( exists ( $ tmphash { lpar0_used_mem } ) ) {
$ after = $ tmphash { lpar0_used_mem } ;
$ res = $ after - $ before ;
if ( $ res < 0 ) {
return ( [ $ name , "Parse reserverd regions failed, before $before, after $after." , 1 ] ) ;
} elsif ( $ lparhash - > { hyp_avail_mem } - $ res < 0 ) {
return ( [ $ name , "Parse reserverd regions failed, no enough memory, availe:$lparhash->{hyp_avail_mem}." , - 1 ] ) ;
}
my $ mem = $ lparhash - > { memory } ;
$ mem =~ /(\d+)\/(\d+)\/(\d+)/ ;
if ( $ 2 > $ lparhash - > { hyp_avail_mem } - $ res ) {
my $ new_avail_mem = $ lparhash - > { hyp_avail_mem } - $ res ;
$ lparhash - > { memory } = "$1/$new_avail_mem/$3" ;
}
return 0 ;
} else {
return ( [ $ name , "part_get_lpar_memory failed to get used memory for hypervisor." , 1 ] ) ;
}
}
sub create_lpar {
my $ request = shift ;
my $ name = shift ;
my $ d = shift ;
my $ lparhash = shift ;
my $ values ;
if ( exists ( $ request - > { opt } - > { vios } ) ) {
$ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lpar_def_state" , 0 , 0x03 ) ;
} else {
$ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lpar_def_state" , 0 , 0x01 ) ;
}
if ( @$ values [ 2 ] ne 0 ) {
return ( [ [ $ name , @$ values [ 1 ] , @$ values [ 0 ] ] ] ) ;
}
$ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "set_lpar_name" , 0 , $ name ) ;
if ( @$ values [ 2 ] ne 0 ) {
$ values = & set_lpar_undefined ( $ request , $ name , $ d ) ;
return ( [ $ name , @$ values [ 1 ] , @$ values [ 0 ] ] ) ;
}
xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lpar_shared_pool_util_auth" ) ;
xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lpar_group_id" ) ;
xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lpar_avail_priority" ) ;
$ values = & deal_with_avail_mem ( $ request , $ name , $ d , $ lparhash ) ;
if ( ref ( $ values ) eq "ARRAY" ) {
return ( [ @$ values ] ) ;
}
#print "======>physlots:$lparhash->{physlots}.\n";
$ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "set_io_slot_owner_uber" , 0 , $ lparhash - > { physlots } ) ;
#$values = xCAT::FSPUtils::fsp_api_action($request, $name, $d, "set_io_slot_owner", 0, join(",",@phy_io_array));
if ( @$ values [ 2 ] ne 0 ) {
$ values = & set_lpar_undefined ( $ request , $ name , $ d ) ;
return ( [ $ name , @$ values [ 1 ] , @$ values [ 2 ] ] ) ;
}
if ( exists ( $ lparhash - > { phy_hea } ) ) {
my $ phy_hash = $ lparhash - > { phy_hea } ;
foreach my $ phy_drc ( keys %$ phy_hash ) {
#print "======> set_lhea_assign_info: drc_index:$phy_drc.\n";
xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lhea_assign_info" , 0 , $ phy_drc ) ;
my $ group_hash = $ phy_hash - > { $ phy_drc } ;
foreach my $ group_id ( keys %$ group_hash ) {
my @ lhea_drc = ( keys % { $ lparhash - > { logic_drc_phydrc } - > { $ group_id } } ) ;
foreach my $ phy_port_id ( keys % { $ group_hash - > { $ group_id } } ) {
my $ tmp_param = "$phy_drc,$group_id,$phy_port_id" ;
#print "======> set_phea_port_info: $tmp_param.\n";
xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_phea_port_info" , 0 , $ tmp_param ) ;
my $ tmp_lhea_param = $ lhea_drc [ $ phy_port_id ] . ",$phy_port_id" ;
#print "======> set_lhea_port_info: $tmp_lhea_param.\n";
xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lhea_port_info" , 0 , $ tmp_lhea_param ) ;
}
delete ( $ lparhash - > { logic_drc_phydrc } - > { $ group_id } - > { $ lhea_drc [ 0 ] } ) ;
delete ( $ lparhash - > { logic_drc_phydrc } - > { $ group_id } - > { $ lhea_drc [ 1 ] } ) ;
}
}
}
#print "======>cpus:$lparhash->{cpus}.\n";
$ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lpar_pending_proc" , 0 , $ lparhash - > { cpus } ) ;
if ( @$ values [ 2 ] ne 0 ) {
$ values = & set_lpar_undefined ( $ request , $ name , $ d ) ;
return ( [ $ name , @$ values [ 1 ] , @$ values [ 2 ] ] ) ;
}
#print "======>memory:$lparhash->{memory}.\n";
$ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lpar_pending_mem" , 0 , $ lparhash - > { memory } ) ;
if ( @$ values [ 2 ] ne 0 ) {
$ values = & set_lpar_undefined ( $ request , $ name , $ d ) ;
return ( [ $ name , @$ values [ 1 ] , @$ values [ 2 ] ] ) ;
}
xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lpar_comp_modes" ) ;
#print "======>memory:$lparhash->{huge_page}.\n";
xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "set_huge_page" , 0 , $ lparhash - > { huge_page } ) ;
#print "======>bsr:$lparhash->{bsr_num}.\n";
xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "set_lpar_bsr" , 0 , $ lparhash - > { bsr_num } ) ;
xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_partition_placement" ) ;
if ( exists ( $ request - > { opt } - > { vios } ) ) {
$ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lpar_def_state" , 0 , 0x04 ) ;
} else {
$ values = xCAT::FSPUtils:: fsp_api_action ( $ request , $ name , $ d , "part_set_lpar_def_state" , 0 , 0x02 ) ;
}
if ( @$ values [ 2 ] ne 0 ) {
return ( [ $ name , @$ values [ 1 ] , @$ values [ 2 ] ] ) ;
}
return ( [ $ name , "Done" , 0 ] ) ;
}
sub mkspeclpar {
my $ request = shift ;
my $ hash = shift ;
my $ values ;
my @ result = ( ) ;
my $ vmtab = xCAT::Table - > new ( 'vm' ) ;
unless ( $ vmtab ) {
return ( [ [ "Error" , "Cannot open vm table" , 1 ] ] ) ;
}
while ( my ( $ mtms , $ h ) = each ( %$ hash ) ) {
my $ memhash ;
my @ nodes = keys ( %$ h ) ;
my $ ent = $ vmtab - > getNodesAttribs ( \ @ nodes , [ 'cpus' , 'memory' , 'physlots' , 'othersettings' ] ) ;
while ( my ( $ name , $ d ) = each ( %$ h ) ) {
if ( ! exists ( $ memhash - > { run } ) ) {
my @ td = @$ d ;
@ td [ 0 ] = 0 ;
$ memhash = & query_cec_info_actions ( $ request , $ name , \ @ td , 1 , [ "part_get_hyp_process_and_mem" , "lpar_lhea_mac" ] ) ;
$ memhash - > { run } = 1 ;
}
my $ tmp_ent = $ ent - > { $ name } - > [ 0 ] ;
if ( ! defined ( $ tmp_ent ) ) {
return ( [ [ $ name , "Not find params" , 1 ] ] ) ;
} elsif ( ! exists ( $ tmp_ent - > { cpus } ) || ! exists ( $ tmp_ent - > { memory } ) || ! exists ( $ tmp_ent - > { physlots } ) ) {
return ( [ [ $ name , "The attribute 'vmcpus', 'vmmemory' and 'vmphyslots' are all needed to be specified." , 1 ] ] ) ;
}
if ( $ tmp_ent - > { memory } =~ /(\d+)(G|M)\/(\d+)(G|M)\/(\d+)(G|M)/i ) {
my $ memsize = $ memhash - > { mem_region_size } ;
my $ min = $ 1 ;
if ( $ 2 == "G" ) {
$ min = $ min * 1024 ;
}
$ min = $ min / $ memsize ;
my $ cur = $ 3 ;
if ( $ 4 == "G" ) {
$ cur = $ cur * 1024 ;
}
$ cur = $ cur / $ memsize ;
my $ max = $ 5 ;
if ( $ 6 == "G" ) {
$ max = $ max * 1024 ;
}
$ max = $ max / $ memsize ;
$ tmp_ent - > { memory } = "$min/$cur/$max" ;
}
$ tmp_ent - > { hyp_config_mem } = $ memhash - > { hyp_config_mem } ;
$ tmp_ent - > { hyp_avail_mem } = $ memhash - > { hyp_avail_mem } ;
$ tmp_ent - > { huge_page } = "0/0/0" ;
$ tmp_ent - > { bsr_num } = "0" ;
if ( exists ( $ tmp_ent - > { othersettings } ) ) {
my $ setting = $ tmp_ent - > { othersettings } ;
if ( $ setting =~ /hugepage:(\d+)/ ) {
my $ tmp = $ 1 ;
$ tmp_ent - > { huge_page } = "1/" . $ tmp . "/" . $ tmp ;
}
if ( $ setting =~ /bsr:(\d+)/ ) {
$ tmp_ent - > { bsr_num } = $ 1 ;
}
}
$ tmp_ent - > { phy_hea } = $ memhash - > { phy_drc_group_port } ;
$ tmp_ent - > { logic_drc_phydrc } = $ memhash - > { logic_drc_phydrc } ;
$ values = & create_lpar ( $ request , $ name , $ d , $ tmp_ent ) ;
push @ result , $ values ;
$ name = undef ;
$ d = undef ;
}
}
return \ @ result ;
}
2010-11-25 02:47:38 +00:00
2013-09-04 23:27:28 -07:00
sub mkfulllpar {
my $ request = shift ;
my $ hash = shift ;
my $ values ;
my @ result = ( ) ;
while ( my ( $ mtms , $ h ) = each ( %$ hash ) ) {
my $ rethash ;
while ( my ( $ name , $ d ) = each ( %$ h ) ) {
if ( ! exists ( $ rethash - > { run } ) ) {
my @ td = @$ d ;
@ td [ 0 ] = 0 ;
$ rethash = query_cec_info_actions ( $ request , $ name , \ @ td , 1 ) ;
if ( ref ( $ rethash ) ne 'HASH' ) {
return ( [ [ $ mtms , "Cann't get hypervisor info hash" , 1 ] ] ) ;
}
$ rethash - > { run } = 1 ;
#print Dumper($rethash);
}
my % lpar_param = ( ) ;
$ lpar_param { cpus } = "1/" . $ rethash - > { process_units_avail } . "/" . $ rethash - > { process_units_config } ;
$ lpar_param { memory } = "1/" . $ rethash - > { hyp_avail_mem } . "/" . $ rethash - > { hyp_config_mem } ;
$ lpar_param { hyp_config_mem } = $ rethash - > { hyp_config_mem } ;
$ lpar_param { hyp_avail_mem } = $ rethash - > { hyp_avail_mem } ;
my @ phy_io_array = keys ( % { $ rethash - > { bus } } ) ;
$ lpar_param { physlots } = join ( "," , @ phy_io_array ) ;
$ lpar_param { huge_page } = "1/" . $ rethash - > { huge_page_avail } . "/" . $ rethash - > { huge_page_avail } ;
$ lpar_param { bsr_num } = $ rethash - > { cec_bsr_avail } ;
$ lpar_param { phy_hea } = $ rethash - > { phy_drc_group_port } ;
$ lpar_param { logic_drc_phydrc } = $ rethash - > { logic_drc_phydrc } ;
$ values = & create_lpar ( $ request , $ name , $ d , \ % lpar_param ) ;
$ rethash - > { logic_drc_phydrc } = $ lpar_param { logic_drc_phydrc } ;
push @ result , $ values ;
$ name = undef ;
$ d = undef ;
}
}
return \ @ result ;
}
2010-11-25 02:47:38 +00:00
##########################################################################
# Creates logical partitions
##########################################################################
sub mkvm {
my $ request = $ _ [ 0 ] ;
my $ opt = $ request - > { opt } ;
# decide if issuing mkvm with the option '-f'.
# if yes, mklpar will be invoked to
# create a full system partition for each CECs managed by the HMC.
if ( exists ( $ opt - > { full } ) ) {
return ( mkfulllpar ( @ _ ) ) ;
2013-09-04 23:27:28 -07:00
} elsif ( exists ( $ opt - > { part } ) ) {
return ( mkspeclpar ( @ _ ) ) ;
} elsif ( exists ( $ opt - > { vios } ) ) {
return ( mkspeclpar ( @ _ ) ) ;
2010-11-25 02:47:38 +00:00
}
else {
# if no, it will execute the original function.
return ( create ( @ _ ) ) ;
}
}
##########################################################################
# Change logical partition
##########################################################################
sub chvm {
return ( modify ( @ _ ) ) ;
}
##########################################################################
2011-06-23 16:31:55 +00:00
# No rmvm for Power 775
2010-11-25 02:47:38 +00:00
##########################################################################
2013-09-04 23:27:28 -07:00
sub rmvm {
my $ request = $ _ [ 0 ] ;
my $ opt = $ request - > { opt } ;
if ( exists ( $ opt - > { p } ) ) {
return ( remove ( @ _ ) ) ;
} else {
return ( [ [ "lpar" , "rmvm only support Power Partitioning." , 1 ] ] ) ;
}
2011-03-02 09:32:34 +00:00
# return( remove(@_) );
2013-09-04 23:27:28 -07:00
}
2010-11-25 02:47:38 +00:00
##########################################################################
# Lists logical partition profile
##########################################################################
sub lsvm {
2013-09-04 23:27:28 -07:00
my $ request = shift ;
my $ hash = shift ;
my $ args = $ request - > { opt } ;
if ( exists ( $ args - > { p } ) ) {
return ( query_cec_info ( $ request , $ hash ) ) ;
} else {
return ( list ( $ request , $ hash ) ) ;
}
2010-11-25 02:47:38 +00:00
}
1 ;