1
This commit is contained in:
445
linux/srcds_run
Normal file
445
linux/srcds_run
Normal file
@ -0,0 +1,445 @@
|
||||
#!/bin/sh
|
||||
#
|
||||
# Copyright (c) 2004, Valve LLC. All rights reserved.
|
||||
#
|
||||
# a wrapper script for the main Source engine dedicated server binary.
|
||||
# Performs auto-restarting of the server on crash. You can
|
||||
# extend this to log crashes and more.
|
||||
#
|
||||
|
||||
# setup the libraries, local dir first!
|
||||
export LD_LIBRARY_PATH=".:bin:$LD_LIBRARY_PATH"
|
||||
|
||||
init() {
|
||||
# Initialises the various variables
|
||||
# Set up the defaults
|
||||
GAME=""
|
||||
DEBUG=""
|
||||
RESTART="yes"
|
||||
HL=./srcds_i486
|
||||
HL_LOCATION=./
|
||||
HL_DETECT=1
|
||||
TIMEOUT=10 # time to wait after a crash (in seconds)
|
||||
CRASH_DEBUG_MSG="email debug.log to linux@valvesoftware.com"
|
||||
GDB="gdb" # the gdb binary to run
|
||||
DEBUG_LOG="debug.log"
|
||||
PID_FILE="" # only needed it DEBUG is set so init later
|
||||
STEAM=""
|
||||
PID_FILE_SET=0
|
||||
STEAMERR=""
|
||||
SIGINT_ACTION="quit 0" # exit normally on sig int
|
||||
NO_TRAP=0
|
||||
AUTO_UPDATE=""
|
||||
STEAM_USER=""
|
||||
STEAM_PASSWORD=""
|
||||
PARAMS=$*
|
||||
SCRIPT_LOCATION=$PWD
|
||||
|
||||
# Remove any old default pid files
|
||||
# Cant do this as they may be still running
|
||||
#rm -f hlds.*.pid
|
||||
|
||||
# use the $FORCE environment variable if its set
|
||||
if test -n "$FORCE" ; then
|
||||
# Note: command line -binary will override this
|
||||
HL=$FORCE
|
||||
HL_DETECT=0
|
||||
fi
|
||||
|
||||
while test $# -gt 0; do
|
||||
case "$1" in
|
||||
"-game")
|
||||
GAME="$2"
|
||||
shift ;;
|
||||
"-debug")
|
||||
DEBUG=1
|
||||
# Ensure that PID_FILE is set
|
||||
PID_FILE_SET=1
|
||||
if test -z "$PID_FILE"; then
|
||||
PID_FILE="hlds.$$.pid"
|
||||
fi ;;
|
||||
"-norestart")
|
||||
RESTART="" ;;
|
||||
"-pidfile")
|
||||
PID_FILE="$2"
|
||||
PID_FILE_SET=1
|
||||
shift ;;
|
||||
"-binary")
|
||||
HL="$2"
|
||||
HL_DETECT=0
|
||||
shift ;;
|
||||
"-timeout")
|
||||
TIMEOUT="$2"
|
||||
shift ;;
|
||||
"-gdb")
|
||||
GDB="$2"
|
||||
shift ;;
|
||||
"-debuglog")
|
||||
DEBUG_LOG="$2"
|
||||
shift ;;
|
||||
"-autoupdate")
|
||||
AUTO_UPDATE="yes"
|
||||
STEAM="./steam"
|
||||
RESTART="yes" ;;
|
||||
"-steamerr")
|
||||
STEAMERR=1 ;;
|
||||
"-ignoresigint")
|
||||
SIGINT_ACTION="" ;;
|
||||
"-notrap")
|
||||
NO_TRAP=1 ;;
|
||||
"-steamuser")
|
||||
STEAM_USER="$2";
|
||||
shift ;;
|
||||
"-steampass")
|
||||
STEAM_PASSWORD="$2";
|
||||
shift ;;
|
||||
"-help")
|
||||
# quit with syntax
|
||||
quit 2
|
||||
;;
|
||||
esac
|
||||
shift
|
||||
done
|
||||
|
||||
# Ensure we have a game specified - if none is then default to CSS
|
||||
if test -z "$GAME"; then
|
||||
GAME="cstrike"
|
||||
PARAMS="$PARAMS -game $GAME"
|
||||
fi
|
||||
|
||||
# If the game exists in the 'orangebox' directory then run that version instead
|
||||
# of the one in ./
|
||||
if test -f "orangebox/$GAME/gameinfo.txt"; then
|
||||
HL_LOCATION=./orangebox/
|
||||
else
|
||||
HL_LOCATION=./
|
||||
fi
|
||||
|
||||
# Ensure that the game directory exists at all
|
||||
if test ! -d "$HL_LOCATION$GAME"; then
|
||||
echo "Invalid game type '$GAME' sepecified."
|
||||
quit 1
|
||||
fi
|
||||
|
||||
if test 0 -eq "$NO_TRAP"; then
|
||||
# Set up the int handler
|
||||
# N.B. Dont use SIGINT symbolic value
|
||||
# as its just INT under ksh
|
||||
trap "$SIGINT_ACTION" 2
|
||||
fi
|
||||
|
||||
# Only detect the CPU if it hasnt been set with
|
||||
# either environment or command line
|
||||
if test "$HL_DETECT" -eq 1; then
|
||||
detectcpu
|
||||
fi
|
||||
|
||||
if test ! -f "$HL_LOCATION$HL"; then
|
||||
echo "Source Engine binary '$HL_LOCATION$HL' not found, exiting"
|
||||
quit 1
|
||||
elif test ! -x "$HL_LOCATION$HL"; then
|
||||
# Could try chmod but dont know what we will be
|
||||
# chmoding so just fail.
|
||||
echo "Source engine binary '$HL_LOCATION$HL' not executable, exiting"
|
||||
quit 1
|
||||
fi
|
||||
|
||||
# Setup debugging
|
||||
if test -n "$DEBUG" ; then
|
||||
#turn on core dumps :) (if possible)
|
||||
echo "Enabling debug mode"
|
||||
if test "unlimited" != `ulimit -c` && test "`ulimit -c`" -eq 0 ; then
|
||||
ulimit -c 2000
|
||||
fi
|
||||
GDB_TEST=`$GDB -v`
|
||||
if test -z "$GDB_TEST"; then
|
||||
echo "Please install gdb first."
|
||||
echo "goto http://www.gnu.org/software/gdb/ "
|
||||
DEBUG="" # turn off debugging cause gdb isn't installed
|
||||
fi
|
||||
fi
|
||||
|
||||
if test -n "$STEAM_PASSWORD" && test -z "$STEAM_USER"; then
|
||||
echo "You must set both the steam username and password."
|
||||
quit 1
|
||||
fi
|
||||
|
||||
#if test 1 -eq $PID_FILE_SET && test -n "$PID_FILE"; then
|
||||
# HL_CMD="$HL $PARAMS -pidfile $PID_FILE"
|
||||
#else
|
||||
HL_CMD="$HL $PARAMS"
|
||||
#fi
|
||||
}
|
||||
|
||||
syntax () {
|
||||
# Prints script syntax
|
||||
|
||||
echo "Syntax:"
|
||||
echo "$0 [-game <game>] [-debug] [-norestart] [-pidfile]"
|
||||
echo " [-binary [srcds_i486]"
|
||||
echo " [-timeout <number>] [-gdb <gdb>] [-autoupdate]"
|
||||
echo " [-steamerr] [-ignoresigint] [-steamuser <username>]"
|
||||
echo " [-steampass <password>] [-debuglog <logname>]"
|
||||
echo "Params:"
|
||||
echo "-game <game> Specifies the <game> to run."
|
||||
echo "-debug Run debugging on failed servers if possible."
|
||||
echo "-debuglog <logname> Log debug output to this file."
|
||||
echo "-norestart Don't attempt to restart failed servers."
|
||||
echo "-pidfile <pidfile> Use the specified <pidfile> to store the server pid."
|
||||
echo "-binary <binary> Use the specified binary ( no auto detection )."
|
||||
echo "-timeout <number> Sleep for <number> seconds before restarting"
|
||||
echo " a failed server."
|
||||
echo "-gdb <gdb> Use <dbg> as the debugger of failed servers."
|
||||
echo "-steamerr Quit on steam update failure."
|
||||
echo "-steamuser <username> Use this username for steam updates."
|
||||
echo "-steampass <password> Use this password for steam updates"
|
||||
echo " (-steamuser must be specified as well)."
|
||||
echo "-ignoresigint Ignore signal INT ( prevents CTRL+C quitting"
|
||||
echo " the script )."
|
||||
echo "-notrap Don't use trap. This prevents automatic"
|
||||
echo " removal of old lock files."
|
||||
echo ""
|
||||
echo "Note: All parameters specified as passed through to the server"
|
||||
echo "including any not listed."
|
||||
}
|
||||
|
||||
debugcore () {
|
||||
# Debugs any core file if DEBUG is set and
|
||||
# the exitcode is none 0
|
||||
|
||||
exitcode=$1
|
||||
|
||||
if test $exitcode -ne 0; then
|
||||
if test -n "$DEBUG" ; then
|
||||
echo "bt" > debug.cmds;
|
||||
echo "info locals" >> debug.cmds;
|
||||
echo "info sharedlibrary" >> debug.cmds
|
||||
echo "info frame" >> debug.cmds; # works, but gives an error... must be last
|
||||
echo "----------------------------------------------" >> $DEBUG_LOG
|
||||
echo "CRASH: `date`" >> $DEBUG_LOG
|
||||
echo "Start Line: $HL_CMD" >> $DEBUG_LOG
|
||||
|
||||
# check to see if a core was dumped
|
||||
if test -f core ; then
|
||||
CORE="core"
|
||||
elif test -f core.`cat $PID_FILE`; then
|
||||
CORE=core.`cat $PID_FILE`
|
||||
elif test -f "$HL_LOCATION$HL.core" ; then
|
||||
CORE="$HL_LOCATION$HL.core"
|
||||
fi
|
||||
|
||||
if test -n "$CORE"; then
|
||||
$GDB $HL_LOCATION$HL $CORE -x debug.cmds -batch >> $DEBUG_LOG
|
||||
fi
|
||||
|
||||
echo "End of Source crash report" >> $DEBUG_LOG
|
||||
echo "----------------------------------------------" >> $DEBUG_LOG
|
||||
echo $CRASH_DEBUG_MSG
|
||||
rm debug.cmds
|
||||
else
|
||||
echo "Add \"-debug\" to the $0 command line to generate a debug.log to help with solving this problem"
|
||||
fi
|
||||
fi
|
||||
}
|
||||
|
||||
detectcpu() {
|
||||
# Attempts to auto detect the CPU
|
||||
echo "Auto detecting CPU"
|
||||
|
||||
if test -e /proc/cpuinfo; then
|
||||
CPU_VERSION="`grep "cpu family" /proc/cpuinfo | cut -f2 -d":" | tr -d " " | uniq`";
|
||||
if test $CPU_VERSION -lt 4; then
|
||||
echo "Error: srcds REQUIRES a 486 CPU or better";
|
||||
quit 1
|
||||
elif test $CPU_VERSION -ge 6; then
|
||||
FEATURES="`grep 'flags' /proc/cpuinfo`";
|
||||
SSE2="`echo $FEATURES |grep -i SSE2`"
|
||||
AMD="`grep AMD /proc/cpuinfo`";
|
||||
if test -n "$AMD"; then
|
||||
OPTERON="`grep Opteron /proc/cpuinfo`";
|
||||
PLATFORM="`uname -m`"
|
||||
if test -z "$OPTERON"; then
|
||||
OPTERON="`grep "Athlon HX" /proc/cpuinfo`";
|
||||
if test -z "$OPTERON"; then
|
||||
OPTERON="`grep "Athlon(tm) 64" /proc/cpuinfo`";
|
||||
fi
|
||||
fi
|
||||
|
||||
if test -n "$OPTERON" && test "x86_64" = "$PLATFORM"; then
|
||||
echo "Using AMD-Opteron (64 bit) Optimised binary."
|
||||
HL=./srcds_amd
|
||||
else
|
||||
echo "Using AMD Optimised binary."
|
||||
HL=./srcds_amd
|
||||
fi
|
||||
elif test -n "$SSE2"; then
|
||||
# CPU supports SSE2 P4 +
|
||||
echo "Using SSE2 Optimised binary."
|
||||
HL=./srcds_i686
|
||||
else
|
||||
echo "Using default binary."
|
||||
fi
|
||||
else
|
||||
echo "Using default binary."
|
||||
fi
|
||||
|
||||
elif test "FreeBSD" = `uname`; then
|
||||
CPU="`grep 'CPU:' /var/run/dmesg.boot`"
|
||||
FEATURES="`grep 'Features=' /var/run/dmesg.boot`"
|
||||
AMD="`echo $CPU |grep AMD`"
|
||||
I686="`echo $CPU |grep 686`"
|
||||
SSE2="`echo $FEATURES |grep -i SSE2`"
|
||||
if test -n "$AMD"; then
|
||||
echo "Using AMD Optimised binary."
|
||||
HL=./srcds_amd
|
||||
elif test -n "$SSE2" ; then
|
||||
echo "Using SSE2 Optimised binary."
|
||||
HL=./srcds_i686
|
||||
else
|
||||
echo "Using default binary."
|
||||
fi
|
||||
else
|
||||
echo "Using default binary."
|
||||
fi
|
||||
}
|
||||
|
||||
update() {
|
||||
updatesingle
|
||||
}
|
||||
|
||||
updatesingle() {
|
||||
# Run the steam update
|
||||
# exits on failure if STEAMERR is set
|
||||
|
||||
if test -n "$AUTO_UPDATE"; then
|
||||
if test -f "$STEAM"; then
|
||||
echo "Updating server using Steam."
|
||||
|
||||
if test "$GAME" = "cstrike"; then
|
||||
GAME="Counter-Strike Source";
|
||||
fi
|
||||
if test "$GAME" = "dod"; then
|
||||
GAME="dods";
|
||||
fi
|
||||
|
||||
CMD="$STEAM -command update -dir .";
|
||||
if test -n "$STEAM_USER"; then
|
||||
CMD="$CMD -username $STEAM_USER";
|
||||
fi
|
||||
if test -n "$STEAM_PASSWORD"; then
|
||||
CMD="$CMD -password $STEAM_PASSWORD";
|
||||
fi
|
||||
|
||||
$CMD -game "$GAME"
|
||||
if test $? -ne 0; then
|
||||
if test -n "$STEAMERR"; then
|
||||
echo "`date`: Steam Update failed, exiting."
|
||||
quit 1
|
||||
else
|
||||
echo "`date`: Steam Update failed, ignoring."
|
||||
return 0
|
||||
fi
|
||||
fi
|
||||
else
|
||||
if test -n "$STEAMERR"; then
|
||||
echo "Could not locate steam binary:$STEAM, exiting.";
|
||||
quit 1
|
||||
else
|
||||
echo "Could not locate steam binary:$STEAM, ignoring.";
|
||||
return 0
|
||||
fi
|
||||
fi
|
||||
fi
|
||||
|
||||
return 1
|
||||
}
|
||||
|
||||
run() {
|
||||
# Runs the steam update and server
|
||||
# Loops if RESTART is set
|
||||
# Debugs if server failure is detected
|
||||
# Note: if RESTART is not set then
|
||||
# 1. DEBUG is set then the server is NOT exec'd
|
||||
# 2. DEBUG is not set the the server is exec'd
|
||||
|
||||
if test -n "$RESTART" ; then
|
||||
echo "Auto-restarting the server on crash"
|
||||
|
||||
#loop forever
|
||||
while true
|
||||
do
|
||||
# Update if needed
|
||||
update
|
||||
|
||||
# Run the server
|
||||
cd $HL_LOCATION
|
||||
$HL_CMD
|
||||
retval=$?
|
||||
cd $SCRIPT_LOCATION
|
||||
if test $retval -eq 0 && test -z "$AUTO_UPDATE"; then
|
||||
break; # if 0 is returned then just quit
|
||||
fi
|
||||
|
||||
debugcore $retval
|
||||
|
||||
echo "`date`: Server restart in $TIMEOUT seconds"
|
||||
|
||||
# don't thrash the hard disk if the server dies, wait a little
|
||||
sleep $TIMEOUT
|
||||
done # while true
|
||||
else
|
||||
# Update if needed
|
||||
update
|
||||
|
||||
# Run the server
|
||||
if test -z "$DEBUG"; then
|
||||
# debug not requested we can exec
|
||||
exec $HL_CMD
|
||||
else
|
||||
# debug requested we can't exec
|
||||
$HL_CMD
|
||||
debugcore $?
|
||||
fi
|
||||
fi
|
||||
}
|
||||
|
||||
quit() {
|
||||
# Exits with the give error code, 1
|
||||
# if none specified.
|
||||
# exit code 2 also prints syntax
|
||||
exitcode="$1"
|
||||
|
||||
# default to failure
|
||||
if test -z "$exitcode"; then
|
||||
exitcode=1
|
||||
fi
|
||||
|
||||
case "$exitcode" in
|
||||
0)
|
||||
echo "`date`: Server Quit" ;;
|
||||
2)
|
||||
syntax ;;
|
||||
*)
|
||||
echo "`date`: Server Failed" ;;
|
||||
esac
|
||||
|
||||
# Remove pid file
|
||||
if test -n "$PID_FILE" && test -f "$PID_FILE" ; then
|
||||
# The specified pid file
|
||||
rm -f $PID_FILE
|
||||
fi
|
||||
|
||||
# reset SIGINT and then kill ourselves properly
|
||||
trap - 2
|
||||
kill -2 $$
|
||||
}
|
||||
|
||||
# Initialise
|
||||
init $*
|
||||
|
||||
# Run
|
||||
run
|
||||
|
||||
# Quit normally
|
||||
quit 0
|
||||
|
Reference in New Issue
Block a user