You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

440 lines
12 KiB
Bash

#!/bin/bash
#
# Copyright 2013-2023 Software Radio Systems Limited
#
# This file is part of srsRAN
#
# srsRAN is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of
# the License, or (at your option) any later version.
#
# srsRAN is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# A copy of the GNU Affero General Public License can be found in
# the LICENSE file in the top-level directory of this distribution
# and at http://www.gnu.org/licenses/.
#
epc_pid=0
enb_pid=0
ue_pid=0
print_use(){
echo "Please call script with srsRAN build path as first argument and number of PRBs as second (number of component carrier is optional)"
echo "E.g. ./run_lte.sh [build_path] [nof_prb] [num_cc]"
exit -1
}
kill_lte(){
echo "Shutting down LTE network"
# Allow to gracefully exit
kill -SIGTERM $ue_pid
sleep 4
kill -SIGTERM $enb_pid
sleep 4
kill -SIGTERM $epc_pid
sleep 4
# Force kill if they are still running
if ps -p $ue_pid > /dev/null
then
echo "Killing UE"
kill -9 $ue_pid
fi
if ps -p $enb_pid > /dev/null
then
echo "Killing eNB"
kill -9 $enb_pid
fi
if ps -p $epc_pid > /dev/null
then
echo "Killing EPC"
kill -9 $epc_pid
fi
if [ -f ./srsRAN.backtrace.crash ]; then
echo "Rename backtrace"
mv ./srsRAN.backtrace.crash srsRAN.backtrace.log
fi
# Delete netns
ip netns delete $ue_netns
# Don't exit with return code ..
}
valid_num_prb()
{
case "$1" in
"6"|"15"|"25"|"50"|"75"|"100")
return 0;;
*)
echo "Invalid number of PRBs. Possible values are [6, 15, 25, 50, 75, 100]."
return 1;;
esac
}
check_ping()
{
loss_rate=$(cat ./${nof_prb}prb_screenlog_ping_ul.log | grep -oP '\d+(?=% packet loss)')
if [ "$loss_rate" != "0" ] 2>/dev/null; then
echo "Error. Detected $loss_rate per-cent loss rate in UL ping."
exit 1
fi
loss_rate=$(cat ./${nof_prb}prb_screenlog_ping_dl.log | grep -oP '\d+(?=% packet loss)')
if [ "$loss_rate" != "0" ] 2>/dev/null; then
echo "Error. Detected $loss_rate per-cent loss rate in DL ping."
exit 1
fi
}
check_ue()
{
# check that app exited correctly
last_line=$(cat ./${nof_prb}prb_screenlog_srsue.log | tail -n1)
if [[ "$last_line" != *"--- exiting ---"* ]]; then
echo "Error. srsUE didn't seem to have exited properly."
exit 1
fi
# Check UE logs don't contain Error from any layer
num_error=$(cat ./${nof_prb}prb_ue.log | grep "\[E\]" | wc -l)
if [ "$num_error" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_error error messages in UE logs. We should finish without any error."
exit 1
fi
# Check UE logs don't contain Warning from any layer
num_warning=$(cat ./${nof_prb}prb_ue.log | grep "\[W\]" | wc -l)
if [ "$num_warning" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_warning warning messages in UE logs. We should finish without any."
exit 1
fi
# Check PDSCH
num_error=$(cat ./${nof_prb}prb_ue.log | grep "KO" | wc -l)
if [ "$num_error" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_error PDSCH errors in UE logs. We should finish without any error."
exit 1
fi
# Check PUCCH ack
num_error=$(cat ./${nof_prb}prb_ue.log | grep "PUCCH" | grep "ack=0" | wc -l)
if [ "$num_error" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_error PUCCH errors in UE logs. We should finish without any error."
exit 1
fi
# Check PHICH
num_error=$(cat ./${nof_prb}prb_ue.log | grep "hi=0" | wc -l)
if [ "$num_error" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_error PHICH errors in UE logs. We should finish without any error."
exit 1
fi
# Check CQI is 15
num_error=$(cat ./${nof_prb}prb_ue.log | grep "PUCCH" | grep "cqi=" | grep -v "cqi=15" | wc -l)
if [ "$num_error" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_error CQI != 15 in UE logs. We should finish with all cqi=15."
exit 1
fi
# Check number of PRACHs
num_prach=$(cat ./${nof_prb}prb_ue.log | grep "preamble=" | wc -l)
if [ "$num_prach" != "1" ] 2>/dev/null; then
echo "Error. Detected $num_prach PRACH(s). But should be only 1."
exit 1
fi
# Check duplicate TB at MAC
num_error=$(cat ./${nof_prb}prb_ue.log | grep "duplicate" | wc -l)
if [ "$num_error" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_error duplicate TBs in UE logs. We should finish without any duplicate."
exit 1
fi
}
check_enb()
{
# check that app exited correctly
last_line=$(cat ./${nof_prb}prb_screenlog_srsenb.log | tail -n1)
if [[ "$last_line" != *"--- exiting ---"* ]]; then
echo "Error. srsENB didn't seem to have exited properly."
exit 1
fi
# Check PRACH results
num_prach=$(cat ./${nof_prb}prb_screenlog_srsenb.log | grep RACH: | wc -l)
if [ "$num_prach" != "1" ] 2>/dev/null; then
echo "Error. Detected $num_prach PRACH(s). But should be only 1."
exit 1
fi
# check [E]
num_error=$(cat ./${nof_prb}prb_enb.log | grep "\[E\]" | wc -l)
if [ "$num_error" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_error error messages in eNB logs. We should finish without any error."
exit 1
fi
# Check [W]
num_warning=$(cat ./${nof_prb}prb_enb.log | grep "\[W\]" | wc -l)
if [ "$num_warning" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_warning warning messages in eNB logs. We should finish without any."
exit 1
fi
# Check for PUSCH errors
num_error=$(cat ./${nof_prb}prb_enb.log | grep "KO" | wc -l)
if [ "$num_error" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_error PUSCH errors in eNB logs. We should finish without any error."
exit 1
fi
# Check PUCCH ack
num_error=$(cat ./${nof_prb}prb_enb.log | grep "PUCCH" | grep "ack=0" | wc -l)
if [ "$num_error" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_error PUCCH errors in eNB logs. We should finish without any error."
exit 1
fi
# Check CQI is 15
num_error=$(cat ./${nof_prb}prb_enb.log | grep "PUCCH" | grep "cqi=" | grep -v "cqi=15" | wc -l)
if [ "$num_error" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_error CQI != 15 in eNB logs. We should finish with all qci=15."
exit 1
fi
}
check_epc()
{
# check that app exited correctly
last_line=$(cat ./${nof_prb}prb_screenlog_srsepc.log | tail -n1)
if [[ "$last_line" != *"--- exiting ---"* ]]; then
echo "Error. srsEPC didn't seem to have exited properly."
exit 1
fi
# check [E]
num_error=$(cat ./${nof_prb}prb_epc.log | grep "\[E\]" | wc -l)
if [ "$num_error" != "0" ] 2>/dev/null; then
echo "Error. Detected $num_error error messages in EPC logs. We should finish without any error."
exit 1
fi
}
# check if build path has been passed
if ([ ! $1 ])
then
print_use
exit
fi
build_path="$1"
# check number of PRBs
if ([ ! $2 ])
then
print_use
exit
fi
nof_prb="$2"
# check number of CC
num_cc="1"
if ([ $3 ])
then
num_cc="$3"
fi
echo "Using $num_cc component carrier(s) in srsENB"
base_srate="23.04e6"
if ([ "$nof_prb" == "75" ])
then
base_srate="15.36e6"
fi
if ([ "$nof_prb" == "15" ])
then
base_srate="3.84e6"
fi
# Check for LTE binaries in build path
if [ ! -x "$build_path/srsenb/src/srsenb" ]; then
echo "Error! srsENB binary not found or not executable!"
exit 1
fi
if [ ! -x "$build_path/srsepc/src/srsepc" ]; then
echo "Error! srsEPC binary not found or not executable!"
exit 1
fi
if [ ! -x "$build_path/srsue/src/srsue" ]; then
echo "Error! srsUE binary not found or not executable!"
exit 1
fi
# Check PRB parameter
if ! valid_num_prb "$nof_prb"; then
exit 1
fi
# Adding netns
ue_netns="ue1"
ip netns add $ue_netns
if [ $? -ne 0 ]
then
echo "Couldn't add netns. Missing permissions? Network namespace $ue_netns already exists?"
exit 1
fi
epc_args="$build_path/../srsepc/epc.conf.example \
--hss.db_file=$build_path/../srsepc/user_db.csv.example \
--log.filename=./${nof_prb}prb_epc.log"
enb_args="$build_path/../srsenb/enb.conf.example \
--enb_files.sib_config=$build_path/../srsenb/sib.conf.example \
--enb_files.rb_config=$build_path/../srsenb/rb.conf.example \
--rf.device_name=zmq \
--log.all_level=info \
--enb.n_prb=$nof_prb \
--log.filename=./${nof_prb}prb_enb.log"
ue_args="$build_path/../srsue/ue.conf.example \
--rf.device_name=zmq \
--rf.rx_gain 60 \
--stack.have_tti_time_stats=false \
--gw.netns=$ue_netns \
--log.all_level=info \
--log.filename=./${nof_prb}prb_ue.log \
--pcap.enable=mac \
--pcap.mac_filename=./${nof_prb}prb_ue.pcap"
if ([ "$num_cc" == "2" ])
then
enb_args="$enb_args --enb_files.rr_config=$build_path/../srsenb/rr_2ca.conf.example \
--rf.device_args=\"fail_on_disconnect=true,base_srate=${base_srate},id=enb,tx_port0=tcp://*:2000,tx_port1=tcp://*:2002,rx_port0=tcp://localhost:2001,rx_port1=tcp://localhost:2003,tx_freq0=2630e6,tx_freq1=2636e6,rx_freq0=2510e6,rx_freq1=2516e6\""
ue_args="$ue_args --rf.dl_earfcn=2850,2910 --rf.nof_carriers=2 --rrc.ue_category=7 --rrc.release=10 \
--rf.device_args=\"tx_port0=tcp://*:2001,tx_port1=tcp://*:2003,rx_port0=tcp://localhost:2000,rx_port1=tcp://localhost:2002,id=ue,base_srate=${base_srate},tx_freq0=2510e6,tx_freq1=2516e6,rx_freq0=2630e6,rx_freq1=2636e6\""
else
enb_args="$enb_args --enb_files.rr_config=$build_path/../srsenb/rr.conf.example \
--rf.device_args=\"fail_on_disconnect=true,tx_port0=tcp://*:2000,rx_port0=tcp://localhost:2001,id=enb,base_srate=${base_srate}\""
ue_args="$ue_args --rf.device_args=\"tx_port0=tcp://*:2001,rx_port0=tcp://localhost:2000,id=ue,base_srate=${base_srate}\""
fi
# Remove existing log files
log_files=$(ls -l | grep ${nof_prb}prb_)
if [ ! -z "$log_files" ]; then
echo "Deleting existing log files for ${nof_prb} PRB."
eval "rm -Rf ${nof_prb}prb_*.log"
fi
# Run srsEPC
echo "Starting srsEPC"
screen -S srsepc -dm -L -Logfile ./${nof_prb}prb_screenlog_srsepc.log $build_path/srsepc/src/srsepc $epc_args
sleep 2
epc_pid=$(pgrep srsepc)
if [ -z "$epc_pid" ]
then
echo "Couldn't start EPC."
kill_lte
exit 1
fi
echo "srsEPC's PID is $epc_pid"
# Run srsENB
echo "Starting srsENB"
screen -S srsenb -dm -L -Logfile ./${nof_prb}prb_screenlog_srsenb.log $build_path/srsenb/src/srsenb $enb_args
sleep 2
enb_pid=$(pgrep srsenb)
if [ -z "$enb_pid" ]
then
echo "Couldn't start ENB"
kill_lte
exit 1
fi
echo "srsENB's PID is $enb_pid"
# Run srsUE
echo "Starting srsUE"
screen -S srsue -dm -L -Logfile ${nof_prb}prb_screenlog_srsue.log $build_path/srsue/src/srsue $ue_args
sleep 2
ue_pid=$(pgrep srsue)
if [ -z "$ue_pid" ]
then
echo "Couldn't start UE"
kill_lte
exit 1
fi
echo "srsUE's PID is $ue_pid"
# Wait until UE is connected
timeout=20
tun_dev="tun_srsue"
echo "Waiting for $tun_dev to become available for ${timeout}s .."
wait_time=0
ip=$(ip netns exec $ue_netns ip addr show $tun_dev | grep "inet\b" | awk '{print $2}' | cut -d/ -f1)
while [ -z $ip ] && [ $wait_time -ne $timeout ]
do
# Wait 1s before next attempt
sleep 1
let wait_time++
ip=$(ip netns exec $ue_netns ip addr show $tun_dev | grep "inet\b" | awk '{print $2}' | cut -d/ -f1)
done
if [ ! -z $ip ]; then
echo "$tun_dev is up with IP $ip!"
sleep 1
else
echo "$tun_dev didn't get up after ${timeout}s"
kill_lte
exit 1
fi
# extract MME IP
baseip=`echo $ip | cut -d"." -f1-3`
mme_ip=$baseip".1"
echo "MME IP is $mme_ip"
# show UE/eNB console trace
screen -S srsue -X stuff "t$(printf \\r)"
screen -S srsenb -X stuff "t$(printf \\r)"
# Run tests now
# run UL ping test
echo "Run UL ping"
ip netns exec $ue_netns screen -dm -L -Logfile ./${nof_prb}prb_screenlog_ping_ul.log ping $mme_ip -c 3
sleep 4
# sleep 3s to get RRC connection release
echo "Run DL ping"
screen -dm -L -Logfile ./${nof_prb}prb_screenlog_ping_dl.log ping $ip -c 3
sleep 4
# run UDP DL (rate must not be more than max DL rate for 6 PRB)
echo "Run DL iperf"
screen -dm -L -Logfile ./${nof_prb}prb_screenlog_iperf_dl.log iperf -c $ip -u -t 1 -i 1 -b 1M
sleep 3
# Stop all running LTE components and remove netns
kill_lte
# check results
check_ping
check_ue
check_enb
check_epc
echo "All tests passed!"
exit 0