Remove mutex locking between subprocesses
Bash is not flexible and the recersive mutex lock that was implemented in the previous commit is full of undefined behaviors if piping is used. We don't actually need to lock between subprocesses so with this commit we lock only between other create_ap process.
This commit is contained in:
parent
4ea5d02193
commit
fee914c359
150
create_ap
150
create_ap
@ -83,75 +83,104 @@ usage() {
|
|||||||
echo " "$PROGNAME" --stop wlan0"
|
echo " "$PROGNAME" --stop wlan0"
|
||||||
}
|
}
|
||||||
|
|
||||||
# allocate lock for the caller bash thread
|
# on success it echos a non-zero unused FD
|
||||||
alloc_lock() {
|
# on error it echos 0
|
||||||
# lock file for creating mutex
|
get_avail_fd() {
|
||||||
local LOCK_FILE=/tmp/create_ap.lock
|
local x
|
||||||
local LOCK_FD=LOCK_FD_$BASHPID
|
|
||||||
|
|
||||||
# if lock FD is already allocated just return
|
|
||||||
eval "[[ \$$LOCK_FD -ne 0 ]]" && return 0
|
|
||||||
|
|
||||||
# use the lowest unused FD. avoid FD 0 because we use it to
|
|
||||||
# indicate that LOCK_FD_$BASHPID is not set.
|
|
||||||
for x in $(seq 1 $(ulimit -n)); do
|
for x in $(seq 1 $(ulimit -n)); do
|
||||||
if [[ ! -a "/proc/$BASHPID/fd/$x" ]]; then
|
if [[ ! -a "/proc/$BASHPID/fd/$x" ]]; then
|
||||||
eval "$LOCK_FD=$x"
|
echo $x
|
||||||
|
return
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
echo 0
|
||||||
|
}
|
||||||
|
|
||||||
|
# lock file for the mutex counter
|
||||||
|
COUNTER_LOCK_FILE=/tmp/create_ap.$$.lock
|
||||||
|
|
||||||
|
cleanup_lock() {
|
||||||
|
rm -f $COUNTER_LOCK_FILE
|
||||||
|
}
|
||||||
|
|
||||||
|
init_lock() {
|
||||||
|
local LOCK_FILE=/tmp/create_ap.all.lock
|
||||||
|
|
||||||
|
# we initialize only once
|
||||||
|
[[ $LOCK_FD -ne 0 ]] && return 0
|
||||||
|
|
||||||
|
LOCK_FD=$(get_avail_fd)
|
||||||
|
[[ $LOCK_FD -eq 0 ]] && return 1
|
||||||
|
|
||||||
# open/create lock file with write access for all users
|
# open/create lock file with write access for all users
|
||||||
# otherwise normal users will not be able to use it.
|
# otherwise normal users will not be able to use it.
|
||||||
# to avoid race conditions on creation, we need to
|
# to avoid race conditions on creation, we need to
|
||||||
# use umask to set the permissions.
|
# use umask to set the permissions.
|
||||||
umask 0555
|
umask 0555
|
||||||
eval "eval \"exec \$$LOCK_FD>$LOCK_FILE\"" > /dev/null 2>&1 || return 1
|
eval "exec $LOCK_FD>$LOCK_FILE" > /dev/null 2>&1 || return 1
|
||||||
umask $SCRIPT_UMASK
|
umask $SCRIPT_UMASK
|
||||||
|
|
||||||
# there is a case where lock file was created from a normal
|
# there is a case where lock file was created from a normal
|
||||||
# user. change the owner to root as soon as we can.
|
# user. change the owner to root as soon as we can.
|
||||||
[[ $(id -u) -eq 0 ]] && chown 0:0 $LOCK_FILE
|
[[ $(id -u) -eq 0 ]] && chown 0:0 $LOCK_FILE
|
||||||
|
|
||||||
return 0
|
# create mutex counter lock file
|
||||||
fi
|
echo 0 > $COUNTER_LOCK_FILE
|
||||||
done
|
|
||||||
|
|
||||||
return 1
|
return $?
|
||||||
}
|
}
|
||||||
|
|
||||||
# recursive mutex lock for all create_ap processes/threads.
|
# recursive mutex lock for all create_ap processes
|
||||||
#
|
|
||||||
# WARNING: if you lock in a function that their caller need their
|
|
||||||
# output (i.e. the caller use | or $()) then you can have dead-lock
|
|
||||||
# if the caller took the lock before. this happens because bash creates
|
|
||||||
# a new thread for the callie function.
|
|
||||||
mutex_lock() {
|
mutex_lock() {
|
||||||
local MUTEX_COUNTER=MUTEX_COUNTER_$BASHPID
|
local counter_mutex_fd
|
||||||
local LOCK_FD=LOCK_FD_$BASHPID
|
local counter
|
||||||
|
|
||||||
# allocate lock FD if needed
|
# lock local mutex and read counter
|
||||||
if eval "[[ \$$LOCK_FD -eq 0 ]]"; then
|
counter_mutex_fd=$(get_avail_fd)
|
||||||
alloc_lock || die "Failed to allocate lock"
|
if [[ $counter_mutex_fd -ne 0 ]]; then
|
||||||
|
eval "exec $counter_mutex_fd<>$COUNTER_LOCK_FILE"
|
||||||
|
flock $counter_mutex_fd
|
||||||
|
read -u $counter_mutex_fd counter
|
||||||
|
else
|
||||||
|
echo "Failed to lock mutex counter" >&2
|
||||||
|
return 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# lock if needed and increase the counter
|
# lock global mutex and increase counter
|
||||||
eval "[[ \$$MUTEX_COUNTER -eq 0 ]]" && eval "flock \$$LOCK_FD"
|
[[ $counter -eq 0 ]] && flock $LOCK_FD
|
||||||
eval "$MUTEX_COUNTER=\$(( \$$MUTEX_COUNTER + 1 ))"
|
counter=$(( $counter + 1 ))
|
||||||
|
|
||||||
|
# write counter and unlock local mutex
|
||||||
|
echo $counter > /proc/$BASHPID/fd/$counter_mutex_fd
|
||||||
|
eval "exec ${counter_mutex_fd}<&-"
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
# recursive mutex unlock for all create_ap processes/threads
|
# recursive mutex unlock for all create_ap processes
|
||||||
mutex_unlock() {
|
mutex_unlock() {
|
||||||
local MUTEX_COUNTER=MUTEX_COUNTER_$BASHPID
|
local counter_mutex_fd
|
||||||
local LOCK_FD=LOCK_FD_$BASHPID
|
local counter
|
||||||
|
|
||||||
# if lock FD was not allocated or we didn't lock before
|
# lock local mutex and read counter
|
||||||
# then just return
|
counter_mutex_fd=$(get_avail_fd)
|
||||||
eval "[[ \$$LOCK_FD -eq 0 || \$$MUTEX_COUNTER -eq 0 ]]" && return 0
|
if [[ $counter_mutex_fd -ne 0 ]]; then
|
||||||
|
eval "exec $counter_mutex_fd<>$COUNTER_LOCK_FILE"
|
||||||
|
flock $counter_mutex_fd
|
||||||
|
read -u $counter_mutex_fd counter
|
||||||
|
else
|
||||||
|
echo "Failed to lock mutex counter" >&2
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
# unlock if needed and decrease the counter
|
# decrease counter and unlock global mutex
|
||||||
eval "$MUTEX_COUNTER=\$(( \$$MUTEX_COUNTER - 1 ))"
|
if [[ $counter -gt 0 ]]; then
|
||||||
eval "[[ \$$MUTEX_COUNTER -eq 0 ]]" && eval "flock -u \$$LOCK_FD"
|
counter=$(( $counter - 1 ))
|
||||||
|
[[ $counter -eq 0 ]] && flock -u $LOCK_FD
|
||||||
|
fi
|
||||||
|
|
||||||
|
# write counter and unlock local mutex
|
||||||
|
echo $counter > /proc/$BASHPID/fd/$counter_mutex_fd
|
||||||
|
eval "exec ${counter_mutex_fd}<&-"
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -582,7 +611,7 @@ _cleanup() {
|
|||||||
|
|
||||||
mutex_lock
|
mutex_lock
|
||||||
|
|
||||||
trap "" SIGINT SIGUSR1 SIGUSR2
|
trap "" SIGINT SIGUSR1 SIGUSR2 EXIT
|
||||||
|
|
||||||
# kill haveged_watchdog
|
# kill haveged_watchdog
|
||||||
[[ -n "$HAVEGED_WATCHDOG_PID" ]] && kill $HAVEGED_WATCHDOG_PID
|
[[ -n "$HAVEGED_WATCHDOG_PID" ]] && kill $HAVEGED_WATCHDOG_PID
|
||||||
@ -670,6 +699,7 @@ _cleanup() {
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
mutex_unlock
|
mutex_unlock
|
||||||
|
cleanup_lock
|
||||||
}
|
}
|
||||||
|
|
||||||
cleanup() {
|
cleanup() {
|
||||||
@ -681,35 +711,22 @@ cleanup() {
|
|||||||
|
|
||||||
die() {
|
die() {
|
||||||
[[ -n "$1" ]] && echo -e "\nERROR: $1\n" >&2
|
[[ -n "$1" ]] && echo -e "\nERROR: $1\n" >&2
|
||||||
# we cleanup and exit only if we are the main thread
|
# send die signal to the main process
|
||||||
if [[ $$ -eq $BASHPID ]]; then
|
[[ $BASHPID -ne $$ ]] && kill -USR2 $$
|
||||||
cleanup
|
# we don't need to call cleanup because it's traped on EXIT
|
||||||
exit 1
|
exit 1
|
||||||
else
|
|
||||||
# send die signal to the main thread
|
|
||||||
kill -USR2 $$
|
|
||||||
# terminate our thread
|
|
||||||
kill -9 $BASHPID
|
|
||||||
fi
|
|
||||||
}
|
}
|
||||||
|
|
||||||
clean_exit() {
|
clean_exit() {
|
||||||
# we cleanup and exit only if we are the main thread
|
# send clean_exit signal to the main process
|
||||||
if [[ $$ -eq $BASHPID ]]; then
|
[[ $BASHPID -ne $$ ]] && kill -USR1 $$
|
||||||
cleanup
|
# we don't need to call cleanup because it's traped on EXIT
|
||||||
exit 0
|
exit 0
|
||||||
else
|
|
||||||
# send clean_exit signal to the main thread
|
|
||||||
kill -USR1 $$
|
|
||||||
# terminate our thread
|
|
||||||
kill -9 $BASHPID
|
|
||||||
fi
|
|
||||||
}
|
}
|
||||||
|
|
||||||
# WARNING: never call mutex_lock in this function, otherwise we
|
|
||||||
# will have dead-lock when send_stop is called
|
|
||||||
list_running() {
|
list_running() {
|
||||||
local PID IFACE x
|
local PID IFACE x
|
||||||
|
mutex_lock
|
||||||
for x in /tmp/create_ap.*; do
|
for x in /tmp/create_ap.*; do
|
||||||
if [[ -f $x/pid ]]; then
|
if [[ -f $x/pid ]]; then
|
||||||
PID=$(cat $x/pid)
|
PID=$(cat $x/pid)
|
||||||
@ -720,6 +737,7 @@ list_running() {
|
|||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
|
mutex_unlock
|
||||||
}
|
}
|
||||||
|
|
||||||
is_running_pid() {
|
is_running_pid() {
|
||||||
@ -858,9 +876,10 @@ if [[ $# -lt 1 && $FIX_UNMANAGED -eq 0 && -z "$STOP_ID" && $LIST_RUNNING -eq 0
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# allocate lock for the main thread to avoid any failures later
|
trap "cleanup_lock" EXIT
|
||||||
if ! alloc_lock; then
|
|
||||||
echo "ERROR: Failed to allocate lock" >&2
|
if ! init_lock; then
|
||||||
|
echo "ERROR: Failed to initialize lock" >&2
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@ -871,9 +890,7 @@ trap "clean_exit" SIGINT SIGUSR1
|
|||||||
trap "die" SIGUSR2
|
trap "die" SIGUSR2
|
||||||
|
|
||||||
if [[ $LIST_RUNNING -eq 1 ]]; then
|
if [[ $LIST_RUNNING -eq 1 ]]; then
|
||||||
mutex_lock
|
|
||||||
list_running
|
list_running
|
||||||
mutex_unlock
|
|
||||||
exit 0
|
exit 0
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@ -1084,6 +1101,7 @@ if [[ $NO_VIRT -eq 1 && "$WIFI_IFACE" == "$INTERNET_IFACE" ]]; then
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
mutex_lock
|
mutex_lock
|
||||||
|
trap "cleanup" EXIT
|
||||||
CONFDIR=$(mktemp -d /tmp/create_ap.${WIFI_IFACE}.conf.XXXXXXXX)
|
CONFDIR=$(mktemp -d /tmp/create_ap.${WIFI_IFACE}.conf.XXXXXXXX)
|
||||||
echo "Config dir: $CONFDIR"
|
echo "Config dir: $CONFDIR"
|
||||||
echo "PID: $$"
|
echo "PID: $$"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user