# # SPDX-License-Identifier: BSD-2-Clause # # Copyright (c) 2023 Klara, Inc. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions # are met: # 1. Redistributions of source code must retain the above copyright # notice, this list of conditions and the following disclaimer. # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE # ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS # OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) # HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF # SUCH DAMAGE. # # sysexits(3) : ${EX_USAGE:=64} : ${EX_UNAVAILABLE:=69} : ${EX_CANTCREAT:=73} : ${EX_TEMPFAIL:=75} atf_test_case badargs badargs_body() { atf_check -s exit:${EX_USAGE} -e not-empty lockf atf_check -s exit:${EX_USAGE} -e not-empty lockf "testlock" } atf_test_case basic basic_body() { # Something innocent so that it does eventually go away without our # intervention. lockf "testlock" sleep 10 & lpid=$! # Make sure that the lock exists... while ! test -e "testlock"; do sleep 0.1 done # Attempt both verbose and silent re-lock atf_check -s exit:${EX_TEMPFAIL} -e not-empty \ lockf -t 0 "testlock" sleep 0 atf_check -s exit:${EX_TEMPFAIL} -e empty \ lockf -t 0 -s "testlock" sleep 0 # Make sure it cleans up after the initial sleep 10 is over. wait "$lpid" atf_check test ! -e "testlock" } atf_test_case fdlock fdlock_body() { # First, make sure we don't get a false positive -- existing uses with # numeric filenames shouldn't switch to being fdlocks automatically. atf_check lockf -k "9" sleep 0 atf_check test -e "9" rm "9" subexit_lockfail=1 subexit_created=2 subexit_lockok=3 subexit_concurrent=4 ( lockf -s -t 0 9 if [ $? -ne 0 ]; then exit "$subexit_lockfail" fi if [ -e "9" ]; then exit "$subexit_created" fi ) 9> "testlock1" rc=$? atf_check test "$rc" -eq 0 sub_delay=5 # But is it actually locking? Child 1 will acquire the lock and then # signal that it's ok for the second child to try. The second child # will try to acquire the lock and fail immediately, signal that it # tried, then try again with an indefinite timeout. On that one, we'll # just check how long we ended up waiting -- it should be at least # $sub_delay. ( lockf -s -t 0 /dev/fd/9 if [ $? -ne 0 ]; then exit "$subexit_lockfail" fi # Signal touch ".lock_acquired" while [ ! -e ".lock_attempted" ]; do sleep 0.5 done sleep "$sub_delay" if [ -e ".lock_acquired_again" ]; then exit "$subexit_concurrent" fi ) 9> "testlock2" & lpid1=$! ( while [ ! -e ".lock_acquired" ]; do sleep 0.5 done # Got the signal, try lockf -s -t 0 9 if [ $? -ne "${EX_TEMPFAIL}" ]; then exit "$subexit_lockok" fi touch ".lock_attempted" start=$(date +"%s") lockf -s 9 touch ".lock_acquired_again" now=$(date +"%s") elapsed=$((now - start)) if [ "$elapsed" -lt "$sub_delay" ]; then exit "$subexit_concurrent" fi ) 9> "testlock2" & lpid2=$! wait "$lpid1" status1=$? wait "$lpid2" status2=$? atf_check test "$status1" -eq 0 atf_check test "$status2" -eq 0 } atf_test_case keep keep_body() { lockf -k "testlock" sleep 10 & lpid=$! # Make sure that the lock exists now... while ! test -e "testlock"; do sleep 0.5 done kill "$lpid" wait "$lpid" # And it still exits after the lock has been relinquished. atf_check test -e "testlock" } atf_test_case needfile needfile_body() { # Hopefully the clock doesn't jump. start=$(date +"%s") # Should fail if the lockfile does not yet exist. atf_check -s exit:"${EX_UNAVAILABLE}" lockf -sn "testlock" sleep 30 # It's hard to guess how quickly we should have finished that; one would # hope that it exits fast, but to be safe we specified a sleep 30 under # lock so that we have a good margin below that duration that we can # safely test to make sure we didn't actually execute the program, more # or less. now=$(date +"%s") tpass=$((now - start)) atf_check test "$tpass" -lt 10 } atf_test_case timeout timeout_body() { lockf "testlock" sleep 30 & lpid=$! while ! test -e "testlock"; do sleep 0.5 done start=$(date +"%s") timeout=2 atf_check -s exit:${EX_TEMPFAIL} lockf -st "$timeout" "testlock" sleep 0 # We should have taken no less than our timeout, at least. now=$(date +"%s") tpass=$((now - start)) atf_check test "$tpass" -ge "$timeout" kill "$lpid" wait "$lpid" || true } atf_test_case wrlock wrlock_head() { atf_set "require.user" "unprivileged" } wrlock_body() { touch "testlock" chmod -w "testlock" # Demonstrate that we can lock the file normally, but -w fails if we # can't write. atf_check lockf -kt 0 "testlock" sleep 0 atf_check -s exit:${EX_CANTCREAT} -e not-empty \ lockf -wt 0 "testlock" sleep 0 } atf_init_test_cases() { atf_add_test_case badargs atf_add_test_case basic atf_add_test_case fdlock atf_add_test_case keep atf_add_test_case needfile atf_add_test_case timeout atf_add_test_case wrlock }