- From cd1684df588eecc508a3b34d623b6c9d7031c9f4 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Sat, 6 Jan 2024 19:48:30 +0100
- Subject: [PATCH 1/3] daemon: |TerminateThread()| while thread is in
- |nfs41_send_sequence()| causes corruption
- Calling |TerminateThread()| for the NFSv4.1 session renewal thread
- causes corruption when the thread is executing |nfs41_send_sequence()|.
- The fix is simply to inform (via |SetEvent()|) the session renewal
- thread to end itself, and let the caller wait until the thread exists.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/nfs41.h | 8 ++--
- daemon/nfs41_session.c | 102 +++++++++++++++++++++++++++++++++--------
- 2 files changed, 87 insertions(+), 23 deletions(-)
- diff --git a/daemon/nfs41.h b/daemon/nfs41.h
- index 3902393..1dc88a7 100644
- --- a/daemon/nfs41.h
- +++ b/daemon/nfs41.h
- @@ -258,9 +258,11 @@ typedef struct __nfs41_session {
- nfs41_channel_attrs fore_chan_attrs;
- nfs41_channel_attrs back_chan_attrs;
- uint32_t lease_time;
- - nfs41_slot_table table;
- - // array of slots
- - HANDLE renew_thread;
- + nfs41_slot_table table; /* array of slots */
- + struct {
- + HANDLE thread_handle;
- + HANDLE cancel_event;
- + } renew;
- bool_t isValidState;
- uint32_t flags;
- nfs41_cb_session cb_session;
- diff --git a/daemon/nfs41_session.c b/daemon/nfs41_session.c
- index 4ecf3f6..a52af26 100644
- --- a/daemon/nfs41_session.c
- +++ b/daemon/nfs41_session.c
- @@ -3,6 +3,7 @@
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- + * Roland Mainz <roland.mainz@nrubsig.org>
- *
- * This library is free software; you can redistribute it and/or modify it
- * under the terms of the GNU Lesser General Public License as published by
- @@ -224,22 +225,57 @@ void nfs41_session_sequence(
- /* session renewal */
- -static unsigned int WINAPI renew_session(void *args)
- +static unsigned int WINAPI renew_session_thread(void *args)
- {
- - int status = NO_ERROR;
- nfs41_session *session = (nfs41_session *)args;
- + int status = NO_ERROR;
- + int event_status;
- +
- + dprintf(1, "renew_session_thread(session=%p): started thread %p\n",
- + session, session->renew.thread_handle);
- +
- /* sleep for 2/3 of lease_time */
- - const uint32_t sleep_time = (2 * session->lease_time*1000)/3;
- + const uint32_t sleep_time = (2UL * session->lease_time*1000UL)/3UL;
- +
- + EASSERT(sleep_time > 100UL);
- + EASSERT(sleep_time < (60*60*1000UL));
- - dprintf(1, "Creating renew_session thread: %p\n", session->renew_thread);
- while(1) {
- - dprintf(1, "Going to sleep for %dmsecs\n", sleep_time);
- - Sleep(sleep_time);
- - status = nfs41_send_sequence(session);
- - if (status)
- - dprintf(1, "renewal thread: nfs41_send_sequence failed %d\n", status);
- + dprintf(1, "renew_session_thread(session=%p): "
- + "Going to sleep for %dmsecs\n",
- + session, (int)sleep_time);
- +
- + /*
- + * sleep for |sleep_time| milliseconds, or until someone
- + * sends an event
- + */
- + event_status = WaitForSingleObjectEx(session->renew.cancel_event,
- + sleep_time, FALSE);
- + if (event_status == WAIT_TIMEOUT) {
- + dprintf(1, "renew_session_thread(session=%p): "
- + "renewing session...\n",
- + session);
- + status = nfs41_send_sequence(session);
- + if (status) {
- + eprintf("renew_session_thread(session=%p): "
- + "nfs41_send_sequence() failed status=%d\n",
- + session, status);
- + }
- + }
- + else if (event_status == WAIT_OBJECT_0) {
- + /* event received, renew thread should exit */
- + break;
- + }
- + else {
- + eprintf("renew_session_thread(session=%p): "
- + "unexpected event_status=0x%x\n",
- + session, (int)event_status);
- + }
- }
- - return status;
- +
- + dprintf(1, "renew_session_thread(session=%p): thread %p exiting\n",
- + session, session->renew.thread_handle);
- + return 0;
- }
- /* session creation */
- @@ -256,7 +292,8 @@ static int session_alloc(
- goto out;
- }
- session->client = client;
- - session->renew_thread = INVALID_HANDLE_VALUE;
- + session->renew.thread_handle = INVALID_HANDLE_VALUE;
- + session->renew.cancel_event = INVALID_HANDLE_VALUE;
- session->isValidState = FALSE;
- InitializeCriticalSection(&session->table.lock);
- @@ -329,6 +366,24 @@ int nfs41_session_renew(
- return status;
- }
- +static
- +void cancel_renew_thread(
- + IN nfs41_session *session)
- +{
- + dprintf(1, "cancel_renew_thread(session=%p): "
- + "signal thread to exit\n", session);
- + (void)SetEvent(session->renew.cancel_event);
- +
- + dprintf(1, "cancel_renew_thread(session=%p): "
- + "waiting for thread to exit\n", session);
- + (void)WaitForSingleObjectEx(session->renew.thread_handle,
- + INFINITE, FALSE);
- +
- + dprintf(1, "cancel_renew_thread(session=%p): thread done\n",
- + session);
- + (void)CloseHandle(session->renew.cancel_event);
- +}
- +
- int nfs41_session_set_lease(
- IN nfs41_session *session,
- IN uint32_t lease_time)
- @@ -336,7 +391,7 @@ int nfs41_session_set_lease(
- int status = NO_ERROR;
- uint32_t thread_id;
- - if (valid_handle(session->renew_thread)) {
- + if (valid_handle(session->renew.thread_handle)) {
- eprintf("nfs41_session_set_lease(): session "
- "renewal thread already started!\n");
- goto out;
- @@ -349,11 +404,20 @@ int nfs41_session_set_lease(
- }
- session->lease_time = lease_time;
- - session->renew_thread = (HANDLE)_beginthreadex(NULL,
- - 0, renew_session, session, 0, &thread_id);
- - if (!valid_handle(session->renew_thread)) {
- + session->renew.cancel_event = CreateEventA(NULL, TRUE, FALSE,
- + "renew.cancel_event");
- + if (!valid_handle(session->renew.cancel_event)) {
- + status = GetLastError();
- + eprintf("nfs41_session_set_lease: CreateEventA() failed %d\n",
- + status);
- + goto out;
- + }
- + session->renew.thread_handle = (HANDLE)_beginthreadex(NULL,
- + 0, renew_session_thread, session, 0, &thread_id);
- + if (!valid_handle(session->renew.thread_handle)) {
- status = GetLastError();
- - eprintf("_beginthreadex failed %d\n", status);
- + eprintf("nfs41_session_set_lease: _beginthreadex() failed %d\n",
- + status);
- goto out;
- }
- out:
- @@ -364,11 +428,9 @@ void nfs41_session_free(
- IN nfs41_session *session)
- {
- AcquireSRWLockExclusive(&session->client->session_lock);
- - if (valid_handle(session->renew_thread)) {
- + if (valid_handle(session->renew.thread_handle)) {
- dprintf(1, "nfs41_session_free: terminating session renewal thread\n");
- - if (!TerminateThread(session->renew_thread, NO_ERROR))
- - eprintf("failed to terminate renewal thread %p\n",
- - session->renew_thread);
- + cancel_renew_thread(session);
- }
- if (session->isValidState) {
- --
- 2.42.1
- From 7de2d31956ee0e937f496ed3aea656f075fed553 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Sat, 6 Jan 2024 19:54:34 +0100
- Subject: [PATCH 2/3] tests: Update manual testing instructions
- Update manual testing instructions to match newer ksh93/bash/gcc/clang
- versions.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- tests/manual_testing.txt | 13 +++++++++++--
- 1 file changed, 11 insertions(+), 2 deletions(-)
- diff --git a/tests/manual_testing.txt b/tests/manual_testing.txt
- index c23d337..761ea98 100644
- --- a/tests/manual_testing.txt
- +++ b/tests/manual_testing.txt
- @@ -1,5 +1,5 @@
- #
- -# ms-nfs41-client manual testing sequence, 2023-11-20
- +# ms-nfs41-client manual testing sequence, 2024-01-05
- #
- # Draft version, needs to be turned into automated tests
- # if possible
- @@ -30,7 +30,7 @@ git config --global --add safe.directory "$PWD"
- sed -i -r 's/mkfifo.+?(-m [[:digit:]]+)/mkfifo /g' ./src/cmd/INIT/package.sh ./bin/package
- # repeat:
- rm -Rf arch
- -time ksh93 -c 'export SHELL=/bin/bash HOSTTYPE="cygwin.i386-64"; /bin/bash ./bin/package make CC="/usr/bin/cc -std=gnu11" CCFLAGS="-Os -g" SHELL="$SHELL" HOSTTYPE="$HOSTTYPE"' 2>&1 | tee buildlog.log
- +time ksh93 -c 'export SHELL=/bin/bash HOSTTYPE="cygwin.i386-64"; /bin/bash ./bin/package make CC="/usr/bin/cc -std=gnu17" CCFLAGS="-Os -g" SHELL="$SHELL" HOSTTYPE="$HOSTTYPE"' 2>&1 | tee buildlog.log
- #
- @@ -49,6 +49,14 @@ printf "path_len=%d\n" "${#PWD}"
- git clone https://git.savannah.gnu.org/git/bash.git
- cd bash/
- +# Cygwin: workaround for configure using cp -p where ln -s should be used
- +# (this is an automake/autoconf issue, they should trust Cygwin and not use
- +# ancient workarounds for issues which no longer exists)
- +(set -o xtrace ; sed -i "s/as_ln_s='cp -pR'/as_ln_s='ln -s'/g" $(find . -name configure) )
- +./configure
- +# workaround for $ cp -p # failing with "Function not implemented"
- +(set -o xtrace ; sed -i -r 's/(cp.*)([[:space:]]+-p[[:space:]]+)/\2--no-preserve=ownership /g' $(find . -name Makefile -o -name Makefile.in) )
- +# run configure
- ./configure --with-curses
- # repeat:
- make clean && make -j4 all
- @@ -89,6 +97,7 @@ cd gcc/
- # (this is an automake/autoconf issue, they should trust Cygwin and not use
- # ancient workarounds for issues which no longer exists)
- (set -o xtrace ; sed -i "s/as_ln_s='cp -pR'/as_ln_s='ln -s'/g" $(find . -name configure) )
- +# run configure
- ./configure
- # workaround for $ cp -p # failing with "Function not implemented"
- (set -o xtrace ; sed -i -r 's/(cp.*)([[:space:]]+-p[[:space:]]+)/\1\2--no-preserve=ownership /g' $(find . -name Makefile -o -name Makefile.in) )
- --
- 2.42.1
- From 5f8125c9ccbca76dc4ef4a87dcd38b61d29b257d Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Sat, 6 Jan 2024 20:26:19 +0100
- Subject: [PATCH 3/3] tests: Add "winfstest" test suite patches and usage
- documentation
- Add "winfstest" test suite patches and usage documentation.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- cygwin/README.txt | 3 +
- ...-VS-project-file-to-VS19-and-make-fi.patch | 85 +++++++++++++++++++
- tests/winfstest/README.txt | 33 +++++++
- 3 files changed, 121 insertions(+)
- create mode 100644 tests/winfstest/0001-winfstest-Update-VS-project-file-to-VS19-and-make-fi.patch
- create mode 100644 tests/winfstest/README.txt
- diff --git a/cygwin/README.txt b/cygwin/README.txt
- index 52332d3..39dd517 100644
- --- a/cygwin/README.txt
- +++ b/cygwin/README.txt
- @@ -101,6 +101,9 @@ mkdir testdir1
- ./runtests -a -t "$PWD/testdir1" 2>&1 | tee testrun.log
- +** "winfstest" test suite:
- +See tests/winfstest/README.txt
- +
- #### ToDo:
- - Makefile/script support for release blob generaetion, local test installation, running cthon4 etc
- diff --git a/tests/winfstest/0001-winfstest-Update-VS-project-file-to-VS19-and-make-fi.patch b/tests/winfstest/0001-winfstest-Update-VS-project-file-to-VS19-and-make-fi.patch
- new file mode 100644
- index 0000000..8d843bb
- --- /dev/null
- +++ b/tests/winfstest/0001-winfstest-Update-VS-project-file-to-VS19-and-make-fi.patch
- @@ -0,0 +1,85 @@
- +From 1774b2b23a49a1a5d672b624fe3750b6f04b818d Mon Sep 17 00:00:00 2001
- +From: Roland Mainz <roland.mainz@nrubsig.org>
- +Date: Sat, 6 Jan 2024 20:02:20 +0100
- +Subject: [PATCH] winfstest: Update VS project file to VS19 and make files
- + executable
- +
- +Update Visual Studio project file to Visual Studio 19 and make
- +script files executable
- +
- +(requires https://github.com/dimov-cz/winfstest.git commit
- +id #525f878c06c585619eadd769c8ed9dcdf175b026)
- +
- +Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- +---
- + TestSuite/run-winfstest | 3 +++
- + TestSuite/simpletap.py | 0
- + TestSuite/winfstest.py | 0
- + winfstest/winfstest.vcxproj | 10 +++++-----
- + 4 files changed, 8 insertions(+), 5 deletions(-)
- + mode change 100644 => 100755 TestSuite/run-winfstest
- + mode change 100644 => 100755 TestSuite/simpletap.py
- + mode change 100644 => 100755 TestSuite/winfstest.py
- +
- +diff --git a/TestSuite/run-winfstest b/TestSuite/run-winfstest
- +old mode 100644
- +new mode 100755
- +index c7cc19f..8783703
- +--- a/TestSuite/run-winfstest
- ++++ b/TestSuite/run-winfstest
- +@@ -1,5 +1,8 @@
- + #!/bin/bash
- +
- ++set -o xtrace
- ++set -o nounset
- ++
- + case $(uname) in
- + CYGWIN*) ;;
- + *) echo "can only be run on Cygwin" 1>&2; exit 1
- +diff --git a/TestSuite/simpletap.py b/TestSuite/simpletap.py
- +old mode 100644
- +new mode 100755
- +diff --git a/TestSuite/winfstest.py b/TestSuite/winfstest.py
- +old mode 100644
- +new mode 100755
- +diff --git a/winfstest/winfstest.vcxproj b/winfstest/winfstest.vcxproj
- +index 6c8cbce..f13facf 100644
- +--- a/winfstest/winfstest.vcxproj
- ++++ b/winfstest/winfstest.vcxproj
- +@@ -21,28 +21,28 @@
- + <PropertyGroup Label="Globals">
- + <ProjectGuid>{71483DEC-695B-4EC8-9007-6E0CA9A0010C}</ProjectGuid>
- + <Keyword>MakeFileProj</Keyword>
- +- <WindowsTargetPlatformVersion>10.0.10586.0</WindowsTargetPlatformVersion>
- ++ <WindowsTargetPlatformVersion>10.0.19041.0</WindowsTargetPlatformVersion>
- + </PropertyGroup>
- + <Import Project="$(VCTargetsPath)\Microsoft.Cpp.Default.props" />
- + <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'" Label="Configuration">
- + <ConfigurationType>Application</ConfigurationType>
- + <UseDebugLibraries>true</UseDebugLibraries>
- +- <PlatformToolset>v140</PlatformToolset>
- ++ <PlatformToolset>v142</PlatformToolset>
- + </PropertyGroup>
- + <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'" Label="Configuration">
- + <ConfigurationType>Makefile</ConfigurationType>
- + <UseDebugLibraries>false</UseDebugLibraries>
- +- <PlatformToolset>v140</PlatformToolset>
- ++ <PlatformToolset>v142</PlatformToolset>
- + </PropertyGroup>
- + <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'" Label="Configuration">
- + <ConfigurationType>Application</ConfigurationType>
- + <UseDebugLibraries>true</UseDebugLibraries>
- +- <PlatformToolset>v140</PlatformToolset>
- ++ <PlatformToolset>v142</PlatformToolset>
- + </PropertyGroup>
- + <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'" Label="Configuration">
- + <ConfigurationType>Application</ConfigurationType>
- + <UseDebugLibraries>false</UseDebugLibraries>
- +- <PlatformToolset>v140</PlatformToolset>
- ++ <PlatformToolset>v142</PlatformToolset>
- + </PropertyGroup>
- + <Import Project="$(VCTargetsPath)\Microsoft.Cpp.props" />
- + <ImportGroup Label="ExtensionSettings">
- +--
- +2.42.1
- +
- diff --git a/tests/winfstest/README.txt b/tests/winfstest/README.txt
- new file mode 100644
- index 0000000..a6b35b3
- --- /dev/null
- +++ b/tests/winfstest/README.txt
- @@ -0,0 +1,33 @@
- +#
- +# winfstest testsuite usage instructions
- +# for ms-nfs41-client
- +#
- +
- +set -o xtrace
- +set -o nounset
- +set -o errexit
- +
- +git clone https://github.com/kofemann/ms-nfs41-client.git
- +git clone https://github.com/dimov-cz/winfstest.git
- +cd winfstest/
- +
- +# switch to commit which is known to work (with our patches)
- +git checkout '525f878c06c585619eadd769c8ed9dcdf175b026'
- +git am --ignore-whitespace <'../ms-nfs41-client/tests/winfstest/0001-winfstest-Update-VS-project-file-to-VS19-and-make-fi.patch'
- +
- +# build test suite binaries
- +export PATH+=":/cygdrive/c/Program Files (x86)/Microsoft Visual Studio/2019/Community/MSBuild/Current/Bin/"
- +MSBuild.exe winfstest.sln -t:Build -p:Configuration=Debug -p:Platform=x64
- +
- +# get testsuite binary path
- +cd TestSuite
- +winfstest_testsuite_path="$(pwd)"
- +
- +# create test dir on NFSv4.1 filesystem
- +mkdir -p /cygdrive/t/winfstesttmp
- +cd /cygdrive/t/winfstesttmp
- +
- +# run testsuite
- +${winfstest_testsuite_path}/run-winfstest
- +
- +# EOF.
- --
- 2.42.1
msnfs41client: Patch for NFSv4.1 session renewal thread+winfstest test suite, 2024-01-06
Posted by Anonymous on Sat 6th Jan 2024 19:39
raw | new post
Submit a correction or amendment below (click here to make a fresh posting)
After submitting an amendment, you'll be able to view the differences between the old and new posts easily.