#!/bin/sh # Copyright (c) 2014-2015 Oracle and/or its affiliates. All Rights Reserved. # # 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 would 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 the Free Software Foundation, # Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA # # Author: Alexey Kodanev <alexey.kodanev@oracle.com> # [ -z "$TST_LIB_LOADED" ] && . test.sh # Run command on remote host. # Options: # -b run in background # -s safe option, if something goes wrong, will exit with TBROK # -c specify command to run tst_rhost_run() { local pre_cmd= local post_cmd=' || echo RTERR' local out= local user="root" local cmd= local safe=0 OPTIND=0 while getopts :bsc:u: opt; do case "$opt" in b) pre_cmd="nohup" post_cmd=" > /dev/null 2>&1 &" out="1> /dev/null" ;; s) safe=1 ;; c) cmd=$OPTARG ;; u) user=$OPTARG ;; *) tst_brkm TBROK "tst_rhost_run: unknown option: $OPTARG" ;; esac done OPTIND=0 if [ -z "$cmd" ]; then [ "$safe" -eq 1 ] && \ tst_brkm TBROK "tst_rhost_run: command not defined" tst_resm TWARN "tst_rhost_run: command not defined" return 1 fi local output= local ret=0 if [ -n "$TST_USE_SSH" ]; then output=`ssh -n -q $user@$RHOST "sh -c \ '$pre_cmd $cmd $post_cmd'" $out 2>&1 || echo 'RTERR'` else output=`rsh -n -l $user $RHOST "sh -c \ '$pre_cmd $cmd $post_cmd'" $out 2>&1 || echo 'RTERR'` fi echo "$output" | grep -q 'RTERR$' && ret=1 if [ $ret -eq 1 ]; then output=$(echo "$output" | sed 's/RTERR//') [ "$safe" -eq 1 ] && \ tst_brkm TBROK "'$cmd' failed on '$RHOST': '$output'" fi [ -z "$out" -a -n "$output" ] && echo "$output" return $ret } # Get test interface names for local/remote host. # tst_get_ifaces [TYPE] # TYPE: { lhost | rhost }; Default value is 'lhost'. tst_get_ifaces() { local type=${1:-"lhost"} if [ "$type" = "lhost" ]; then echo "$LHOST_IFACES" else echo "$RHOST_IFACES" fi } # Get HW addresses from defined test interface names. # tst_get_hwaddrs [TYPE] # TYPE: { lhost | rhost }; Default value is 'lhost'. tst_get_hwaddrs() { local type=${1:-"lhost"} local addr= local list= for eth in $(tst_get_ifaces $type); do local addr_path="/sys/class/net/${eth}/address" case $type in lhost) addr=$(cat $addr_path) ;; rhost) addr=$(tst_rhost_run -s -c "cat $addr_path") esac [ -z "$list" ] && list="$addr" || list="$list $addr" done echo "$list" } # Get test HW address. # tst_hwaddr [TYPE] [LINK] # TYPE: { lhost | rhost }; Default value is 'lhost'. # LINK: link number starting from 0. Default value is '0'. tst_hwaddr() { local type=${1:-"lhost"} local link_num=${2:-"0"} local hwaddrs= link_num=$(( $link_num + 1 )) [ "$type" = "lhost" ] && hwaddrs=$LHOST_HWADDRS || hwaddrs=$RHOST_HWADDRS echo "$hwaddrs" | awk '{ print $'"$link_num"' }' } # Get test interface name. # tst_iface [TYPE] [LINK] # TYPE: { lhost | rhost }; Default value is 'lhost'. # LINK: link number starting from 0. Default value is '0'. tst_iface() { local type=${1:-"lhost"} local link_num=${2:-"0"} link_num=$(( $link_num + 1 )) echo "$(tst_get_ifaces $type)" | awk '{ print $'"$link_num"' }' } # Blank for an IPV4 test; 6 for an IPV6 test. TST_IPV6= tst_read_opts() { OPTIND=0 while getopts ":6" opt; do case "$opt" in 6) TST_IPV6=6;; esac done OPTIND=0 } tst_read_opts $* # Get IP address # tst_ipaddr [TYPE] # TYPE: { lhost | rhost }; Default value is 'lhost'. tst_ipaddr() { local type=${1:-"lhost"} local ipv=${TST_IPV6:-"4"} local tst_host= if [ "$type" = "lhost" ]; then eval "tst_host=\$LHOST_IPV${ipv}_HOST" else eval "tst_host=\$RHOST_IPV${ipv}_HOST" fi if [ "$TST_IPV6" ]; then echo "${IPV6_NETWORK}:${tst_host}" else echo "${IPV4_NETWORK}.${tst_host}" fi } # tst_init_iface [TYPE] [LINK] # TYPE: { lhost | rhost }; Default value is 'lhost'. # LINK: link number starting from 0. Default value is '0'. tst_init_iface() { local type=${1:-"lhost"} local link_num=${2:-"0"} local iface=$(tst_iface $type $link_num) tst_resm TINFO "initialize '$type' '$iface' interface" if [ "$type" = "lhost" ]; then ip link set $iface down || return $? ip route flush dev $iface || return $? ip addr flush dev $iface || return $? ip link set $iface up return $? fi tst_rhost_run -c "ip link set $iface down" || return $? tst_rhost_run -c "ip route flush dev $iface" || return $? tst_rhost_run -c "ip addr flush dev $iface" || return $? tst_rhost_run -c "ip link set $iface up" } # tst_add_ipaddr [TYPE] [LINK] # TYPE: { lhost | rhost }; Default value is 'lhost'. # LINK: link number starting from 0. Default value is '0'. tst_add_ipaddr() { local type=${1:-"lhost"} local link_num=${2:-"0"} local mask=24 [ "$TST_IPV6" ] && mask=64 local iface=$(tst_iface $type $link_num) if [ $type = "lhost" ]; then tst_resm TINFO "set local addr $(tst_ipaddr)/$mask" ip addr add $(tst_ipaddr)/$mask dev $iface return $? fi tst_resm TINFO "set remote addr $(tst_ipaddr rhost)/$mask" tst_rhost_run -c "ip addr add $(tst_ipaddr rhost)/$mask dev $iface" } # tst_restore_ipaddr [TYPE] [LINK] # Restore default ip addresses defined in network.sh # TYPE: { lhost | rhost }; Default value is 'lhost'. # LINK: link number starting from 0. Default value is '0'. tst_restore_ipaddr() { local type=${1:-"lhost"} local link_num=${2:-"0"} tst_init_iface $type $link_num || return $? local ret=0 local backup_tst_ipv6=$TST_IPV6 TST_IPV6= tst_add_ipaddr $type $link_num || ret=$? TST_IPV6=6 tst_add_ipaddr $type $link_num || ret=$? TST_IPV6=$backup_tst_ipv6 return $ret } # tst_netload ADDR [FILE] [TYPE] [OPTS] # Run network load test # ADDR: IP address # FILE: file with result time # TYPE: PING or TFO (TCP traffic) # OPTS: additional options tst_netload() { local ip_addr="$1" local rfile=${2:-"netload.res"} local type=${3:-"TFO"} local addopts=${@:4} local ret=0 clients_num=${clients_num:-"2"} client_requests=${client_requests:-"500000"} max_requests=${max_requests:-"3"} case "$type" in PING) local ipv6= echo "$ip_addr" | grep ":" > /dev/null [ $? -eq 0 ] && ipv6=6 tst_resm TINFO "run ping${ipv6} test with rhost '$ip_addr'..." local res= res=$(ping${ipv6} -f -c $client_requests $ip_addr -w 600 2>&1) [ $? -ne 0 ] && return 1 echo $res | sed -nE 's/.*time ([0-9]+)ms.*/\1/p' > $rfile ;; TFO) local port= port=$(tst_rhost_run -c 'tst_get_unused_port ipv6 stream') [ $? -ne 0 ] && tst_brkm TBROK "failed to get unused port" tst_resm TINFO "run tcp_fastopen with '$ip_addr', port '$port'" tst_rhost_run -s -b -c "tcp_fastopen -R $max_requests \ -g $port $addopts" # check that tcp_fastopen on rhost in 'Listening' state local sec_waited= for sec_waited in $(seq 1 60); do tst_rhost_run -c "ss -ltn | grep -q $port" && break if [ $sec_waited -eq 60 ]; then tst_resm TINFO "rhost not in LISTEN state" return 1 fi sleep 1 done # run local tcp client tcp_fastopen -a $clients_num -r $client_requests -l -H $ip_addr\ -g $port -d $rfile $addopts > /dev/null || ret=1 if [ $ret -eq 0 -a ! -f $rfile ]; then tst_brkm TBROK "can't read $rfile" fi tst_rhost_run -c "pkill -9 tcp_fastopen\$" ;; *) tst_brkm TBROK "invalid net_load type '$type'" ;; esac return $ret } # tst_ping [IFACE] [DST ADDR] [MESSAGE SIZE ARRAY] # Check icmp connectivity # IFACE: source interface name # DST ADDR: destination IPv4 or IPv6 address # MESSAGE SIZE ARRAY: message size array tst_ping() { # The max number of ICMP echo request PING_MAX=${PING_MAX:-"10"} local src_iface=${1:-"$(tst_iface)"} local dst_addr=${2:-"$(tst_ipaddr rhost)"}; shift 2 local msg_sizes=$@ local ret=0 # ping cmd use 56 as default message size for size in ${msg_sizes:-"56"}; do ping$TST_IPV6 -I $src_iface -c $PING_MAX $dst_addr \ -s $size > /dev/null 2>&1 ret=$? if [ $ret -eq 0 ]; then tst_resm TINFO "tst_ping IPv${TST_IPV6:-4} msg_size $size pass" else tst_resm TINFO "tst_ping IPv${TST_IPV6:-4} msg_size $size fail" break fi done return $ret } # Management Link [ -z "$RHOST" ] && tst_brkm TBROK "RHOST variable not defined" export RHOST="$RHOST" export PASSWD=${PASSWD:-""} # Don't use it in new tests, use tst_rhost_run() from test_net.sh instead. export LTP_RSH=${LTP_RSH:-"rsh -n"} # Test Links # Warning: make sure to set valid interface names and IP addresses below. # Set names for test interfaces, e.g. "eth0 eth1" export LHOST_IFACES=${LHOST_IFACES:-"eth0"} export RHOST_IFACES=${RHOST_IFACES:-"eth0"} # Set corresponding HW addresses, e.g. "00:00:00:00:00:01 00:00:00:00:00:02" export LHOST_HWADDRS=${LHOST_HWADDRS:-"$(tst_get_hwaddrs lhost)"} export RHOST_HWADDRS=${RHOST_HWADDRS:-"$(tst_get_hwaddrs rhost)"} # Set first three octets of the network address, default is '10.0.0' export IPV4_NETWORK=${IPV4_NETWORK:-"10.0.0"} # Set local host last octet, default is '2' export LHOST_IPV4_HOST=${LHOST_IPV4_HOST:-"2"} # Set remote host last octet, default is '1' export RHOST_IPV4_HOST=${RHOST_IPV4_HOST:-"1"} # Set the reverse of IPV4_NETWORK export IPV4_NET_REV=${IPV4_NET_REV:-"0.0.10"} # Set first three octets of the network address, default is 'fd00:1:1:1' export IPV6_NETWORK=${IPV6_NETWORK:-"fd00:1:1:1"} # Set local host last octet, default is '2' export LHOST_IPV6_HOST=${LHOST_IPV6_HOST:-":2"} # Set remote host last octet, default is '1' export RHOST_IPV6_HOST=${RHOST_IPV6_HOST:-":1"} # Reverse network portion of the IPv6 address export IPV6_NET_REV=${IPV6_NET_REV:-"1.0.0.0.1.0.0.0.1.0.0.0.0.0.d.f"} # Reverse host portion of the IPv6 address of the local host export LHOST_IPV6_REV=${LHOST_IPV6_REV:-"2.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0"} # Reverse host portion of the IPv6 address of the remote host export RHOST_IPV6_REV=${RHOST_IPV6_REV:-"1.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0"} # Networks that aren't reachable through the test links export IPV4_NET16_UNUSED=${IPV4_NET16_UNUSED:-"10.23"} export IPV6_NET32_UNUSED=${IPV6_NET32_UNUSED:-"fd00:23"} export HTTP_DOWNLOAD_DIR=${HTTP_DOWNLOAD_DIR:-"/var/www/html"} export FTP_DOWNLOAD_DIR=${FTP_DOWNLOAD_DIR:-"/var/ftp"} export FTP_UPLOAD_DIR=${FTP_UPLOAD_DIR:-"/var/ftp/pub"} export FTP_UPLOAD_URLDIR=${FTP_UPLOAD_URLDIR:-"pub"} # network/stress tests require additional parameters export NS_DURATION=${NS_DURATION:-"3600"} export NS_TIMES=${NS_TIMES:-"10000"} export CONNECTION_TOTAL=${CONNECTION_TOTAL:-"4000"} export IP_TOTAL=${IP_TOTAL:-"10000"} export IP_TOTAL_FOR_TCPIP=${IP_TOTAL_FOR_TCPIP:-"100"} export ROUTE_TOTAL=${ROUTE_TOTAL:-"10000"} export MTU_CHANGE_TIMES=${MTU_CHANGE_TIMES:-"1000"} export IF_UPDOWN_TIMES=${IF_UPDOWN_TIMES:-"10000"} export DOWNLOAD_BIGFILESIZE=${DOWNLOAD_BIGFILESIZE:-"2147483647"} export DOWNLOAD_REGFILESIZE=${DOWNLOAD_REGFILESIZE:-"1048576"} export UPLOAD_BIGFILESIZE=${UPLOAD_BIGFILESIZE:-"2147483647"} export UPLOAD_REGFILESIZE=${UPLOAD_REGFILESIZE:-"1024"} export MCASTNUM_NORMAL=${MCASTNUM_NORMAL:-"20"} export MCASTNUM_HEAVY=${MCASTNUM_HEAVY:-"40000"} # More information about network parameters can be found # in the following document: testcases/network/stress/README