pastebin - collaborative debugging tool
rovema.kpaste.net RSS


msnfs41client: Patches for fixing SRWLock/CRITSEC waiting before |free()|, getattr debug and misc, 2024-02-28 ...
Posted by Anonymous on Wed 28th Feb 2024 16:31
raw | new post

  1. From 418713079308fd6d9a041cc8d25d54bf3a472a8e Mon Sep 17 00:00:00 2001
  2. From: Roland Mainz <roland.mainz@nrubsig.org>
  3. Date: Wed, 28 Feb 2024 10:14:20 +0100
  4. Subject: [PATCH 1/4] daemon: Add missing init of |state->lock| in
  5.  |create_open_state()|
  6.  
  7. Add missing |InitializeSRWLock(&state->lock);| in
  8. |create_open_state()|, and add a comment who releases the initial
  9. |ref_count|.
  10.  
  11. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  12. ---
  13.  daemon/open.c | 4 +++-
  14.  1 file changed, 3 insertions(+), 1 deletion(-)
  15.  
  16. diff --git a/daemon/open.c b/daemon/open.c
  17. index f605f1e..0c502db 100644
  18. --- a/daemon/open.c
  19. +++ b/daemon/open.c
  20. @@ -3,6 +3,7 @@
  21.   *
  22.   * Olga Kornievskaia <aglo@umich.edu>
  23.   * Casey Bodley <cbodley@umich.edu>
  24. + * Roland Mainz <roland.mainz@nrubsig.org>
  25.   *
  26.   * This library is free software; you can redistribute it and/or modify it
  27.   * under the terms of the GNU Lesser General Public License as published by
  28. @@ -61,7 +62,8 @@ static int create_open_state(
  29.      StringCchPrintfA((LPSTR)state->owner.owner, NFS4_OPAQUE_LIMIT, "%u",
  30.          open_owner_id);
  31.      state->owner.owner_len = (uint32_t)strlen((const char*)state->owner.owner);
  32. -    state->ref_count = 1;
  33. +    InitializeSRWLock(&state->lock);
  34. +    state->ref_count = 1; /* will be released in |cleanup_close()| */
  35.      list_init(&state->locks.list);
  36.      list_init(&state->client_entry);
  37.      InitializeCriticalSection(&state->locks.lock);
  38. --
  39. 2.43.0
  40.  
  41. From 936afc151ba8ad90553bcd3a96818c913e5b9db8 Mon Sep 17 00:00:00 2001
  42. From: Roland Mainz <roland.mainz@nrubsig.org>
  43. Date: Wed, 28 Feb 2024 16:16:38 +0100
  44. Subject: [PATCH 2/4] daemon: Make sure noone uses SRWLocks and critical
  45.  setions before |free()|
  46.  
  47. Make sure noone uses SRWLocks and critical setions before using
  48. |free()| on the underlying memory.
  49.  
  50. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  51. ---
  52.  daemon/getattr.c      |  4 ++--
  53.  daemon/namespace.c    |  2 ++
  54.  daemon/nfs41_client.c |  5 +++++
  55.  daemon/nfs41_server.c |  3 +++
  56.  daemon/open.c         | 16 +++++++++++++++
  57.  daemon/util.c         | 47 +++++++++++++++++++++++++++++++++++++++++++
  58.  daemon/util.h         |  3 +++
  59.  7 files changed, 78 insertions(+), 2 deletions(-)
  60.  
  61. diff --git a/daemon/getattr.c b/daemon/getattr.c
  62. index 9e07df9..a3727aa 100644
  63. --- a/daemon/getattr.c
  64. +++ b/daemon/getattr.c
  65. @@ -80,11 +80,11 @@ static int parse_getattr(unsigned char *buffer, uint32_t length, nfs41_upcall *u
  66.       */
  67.      volatile int ok = 0;
  68.      __try {
  69. -        if (upcall->state_ref->session->client->server != NULL)
  70. +        if (upcall->state_ref->session->client != NULL)
  71.              ok++;
  72.      }
  73.      __except(EXCEPTION_EXECUTE_HANDLER) {
  74. -        eprintf("parse_getattr: Exception accessing upcall->state_ref->session->client->server\n");
  75. +        eprintf("parse_getattr: Exception accessing upcall->state_ref->session->client\n");
  76.      }
  77.      if (ok != 1) {
  78.          status = ERROR_INVALID_PARAMETER;
  79. diff --git a/daemon/namespace.c b/daemon/namespace.c
  80. index 87425a0..1d99e98 100644
  81. --- a/daemon/namespace.c
  82. +++ b/daemon/namespace.c
  83. @@ -81,6 +81,8 @@ static void root_free(
  84.  
  85.      DPRINTF(NSLVL, ("--> nfs41_root_free()\n"));
  86.  
  87. +    EASSERT(waitcriticalsection(&root->lock) == TRUE);
  88. +
  89.      /* free clients */
  90.      list_for_each_tmp(entry, tmp, &root->clients)
  91.          nfs41_client_free(client_entry(entry));
  92. diff --git a/daemon/nfs41_client.c b/daemon/nfs41_client.c
  93. index 1d043e0..7de1d56 100644
  94. --- a/daemon/nfs41_client.c
  95. +++ b/daemon/nfs41_client.c
  96. @@ -207,6 +207,11 @@ void nfs41_client_free(
  97.      IN nfs41_client *client)
  98.  {
  99.      DPRINTF(2, ("nfs41_client_free(%llu)\n", client->clnt_id));
  100. +
  101. +    EASSERT(waitcriticalsection(&client->recovery.lock) == TRUE);
  102. +    EASSERT(waitSRWlock(&client->exid_lock) == TRUE);
  103. +    EASSERT(waitcriticalsection(&client->state.lock) == TRUE);
  104. +
  105.      nfs41_client_delegation_free(client);
  106.      if (client->session) nfs41_session_free(client->session);
  107.      nfs41_destroy_clientid(client->rpc, client->clnt_id);
  108. diff --git a/daemon/nfs41_server.c b/daemon/nfs41_server.c
  109. index 9902954..8192fed 100644
  110. --- a/daemon/nfs41_server.c
  111. +++ b/daemon/nfs41_server.c
  112. @@ -129,6 +129,9 @@ static void server_free(
  113.      IN nfs41_server *server)
  114.  {
  115.      DPRINTF(SRVLVL, ("server_free('%s')\n", server->owner));
  116. +
  117. +    EASSERT(waitSRWlock(&server->addrs.lock) == TRUE);
  118. +
  119.      nfs41_superblock_list_free(&server->superblocks);
  120.      nfs41_name_cache_free(&server->name_cache);
  121.      free(server);
  122. diff --git a/daemon/open.c b/daemon/open.c
  123. index 0c502db..0732aa2 100644
  124. --- a/daemon/open.c
  125. +++ b/daemon/open.c
  126. @@ -85,6 +85,12 @@ static void open_state_free(
  127.      IN nfs41_open_state *state)
  128.  {
  129.      struct list_entry *entry, *tmp;
  130. +    
  131. +    EASSERT(waitcriticalsection(&state->ea.lock) == TRUE);
  132. +    EASSERT(waitcriticalsection(&state->locks.lock) == TRUE);
  133. +
  134. +    EASSERT(waitSRWlock(&state->lock) == TRUE);
  135. +    EASSERT(waitSRWlock(&state->path.lock) == TRUE);
  136.  
  137.      /* free associated lock state */
  138.      list_for_each_tmp(entry, tmp, &state->locks.list)
  139. @@ -93,6 +99,12 @@ static void open_state_free(
  140.          nfs41_delegation_deref(state->delegation.state);
  141.      if (state->ea.list != INVALID_HANDLE_VALUE)
  142.          free(state->ea.list);
  143. +
  144. +    DeleteCriticalSection(&state->ea.lock);
  145. +    DeleteCriticalSection(&state->locks.lock);
  146. +
  147. +    EASSERT(state->ref_count == 0);
  148. +
  149.      free(state);
  150.  }
  151.  
  152. @@ -119,6 +131,8 @@ void nfs41_open_state_ref(
  153.          return;
  154.  #endif /* NFS41_DRIVER_STABILITY_HACKS */
  155.  
  156. +    EASSERT(state->ref_count > 0);
  157. +
  158.      const LONG count = InterlockedIncrement(&state->ref_count);
  159.  
  160.      DPRINTF(2, ("nfs41_open_state_ref('%s') count %d\n", state->path.path, count));
  161. @@ -127,6 +141,8 @@ void nfs41_open_state_ref(
  162.  void nfs41_open_state_deref(
  163.      IN nfs41_open_state *state)
  164.  {
  165. +    EASSERT(state->ref_count > 0);
  166. +
  167.      const LONG count = InterlockedDecrement(&state->ref_count);
  168.  
  169.      DPRINTF(2, ("nfs41_open_state_deref('%s') count %d\n", state->path.path, count));
  170. diff --git a/daemon/util.c b/daemon/util.c
  171. index 4c0fd4c..e2774d0 100644
  172. --- a/daemon/util.c
  173. +++ b/daemon/util.c
  174. @@ -581,3 +581,50 @@ BOOL subcmd_readcmdoutput(subcmd_popen_context *pinfo, char *buff, size_t buff_s
  175.  {
  176.      return ReadFile(pinfo->hReadPipe, buff, (DWORD)buff_size, num_buff_read_ptr, NULL);
  177.  }
  178. +
  179. +/*
  180. + * |waitSRWlock()| - Wait for outstanding locks (usually used
  181. + * before disposing (e.g. |free()| the memory of the structure
  182. + * of the lock) a SRW lock.
  183. + * Returns |TRUE| if we didn't had to wait for another thread
  184. + * to release the lock first.
  185. + */
  186. +bool_t waitSRWlock(PSRWLOCK srwlock)
  187. +{
  188. +    bool_t srw_locked;
  189. +
  190. +    /* Check whether something is still using the lock */
  191. +    srw_locked = TryAcquireSRWLockExclusive(srwlock);
  192. +    if (srw_locked) {
  193. +        ReleaseSRWLockExclusive(srwlock);
  194. +    }
  195. +    else {
  196. +        AcquireSRWLockExclusive(srwlock);
  197. +        ReleaseSRWLockExclusive(srwlock);
  198. +    }
  199. +    return srw_locked;
  200. +}
  201. +
  202. +/*
  203. + * |waitcriticalsection()| - Wait for other threads using the
  204. + * CRITICAL_SECTION (usually used before disposing (e.g.
  205. + * |free()| the memory of the structure of the CRITICAL_SECTION)
  206. + * a CRITICAL_SECTION.
  207. + * Returns |TRUE| if we didn't had to wait for another thread
  208. + * to release the lock first.
  209. + */
  210. +bool_t waitcriticalsection(LPCRITICAL_SECTION cs)
  211. +{
  212. +    bool_t cs_locked;
  213. +
  214. +    /* Check whether something is still using the critical section */
  215. +    cs_locked = TryEnterCriticalSection(cs);
  216. +    if (cs_locked) {
  217. +        LeaveCriticalSection(cs);
  218. +    }
  219. +    else {
  220. +        EnterCriticalSection(cs);
  221. +        LeaveCriticalSection(cs);
  222. +    }
  223. +    return cs_locked;
  224. +}
  225. diff --git a/daemon/util.h b/daemon/util.h
  226. index 9e1c3e5..140a3bd 100644
  227. --- a/daemon/util.h
  228. +++ b/daemon/util.h
  229. @@ -266,4 +266,7 @@ subcmd_popen_context *subcmd_popen(const char *command);
  230.  int subcmd_pclose(subcmd_popen_context *pinfo);
  231.  BOOL subcmd_readcmdoutput(subcmd_popen_context *pinfo, char *buff, size_t buff_size, DWORD *num_buff_read_ptr);
  232.  
  233. +bool_t waitSRWlock(PSRWLOCK srwlock);
  234. +bool_t waitcriticalsection(LPCRITICAL_SECTION cs);
  235. +
  236.  #endif /* !__NFS41_DAEMON_UTIL_H__ */
  237. --
  238. 2.43.0
  239.  
  240. From 1031499a998579bf180cece8ba3fdc6e40de4de1 Mon Sep 17 00:00:00 2001
  241. From: Roland Mainz <roland.mainz@nrubsig.org>
  242. Date: Wed, 28 Feb 2024 16:34:04 +0100
  243. Subject: [PATCH 3/4] daemon: |parse_getattr()| should report address of
  244.  |state_ref| on exception
  245.  
  246. |parse_getattr()| should report address of |upcall->state_ref| on
  247. exception.
  248.  
  249. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  250. ---
  251.  daemon/getattr.c | 4 +++-
  252.  1 file changed, 3 insertions(+), 1 deletion(-)
  253.  
  254. diff --git a/daemon/getattr.c b/daemon/getattr.c
  255. index a3727aa..3c69db3 100644
  256. --- a/daemon/getattr.c
  257. +++ b/daemon/getattr.c
  258. @@ -84,7 +84,9 @@ static int parse_getattr(unsigned char *buffer, uint32_t length, nfs41_upcall *u
  259.              ok++;
  260.      }
  261.      __except(EXCEPTION_EXECUTE_HANDLER) {
  262. -        eprintf("parse_getattr: Exception accessing upcall->state_ref->session->client\n");
  263. +        eprintf("parse_getattr: Exception accessing "
  264. +            "upcall->state_ref(=0x%p)->session->client\n",
  265. +            upcall->state_ref);
  266.      }
  267.      if (ok != 1) {
  268.          status = ERROR_INVALID_PARAMETER;
  269. --
  270. 2.43.0
  271.  
  272. From b2ffbe8fdea9b0fcf6ea9e91b218b577b9ae2893 Mon Sep 17 00:00:00 2001
  273. From: Roland Mainz <roland.mainz@nrubsig.org>
  274. Date: Wed, 28 Feb 2024 16:36:46 +0100
  275. Subject: [PATCH 4/4] tests: manual testing: libisl-devel is now required for
  276.  gcc build
  277.  
  278. manual_testing.txt Document that Cygwin 3.6.x requires package
  279. "libisl-devel" to build gcc.
  280.  
  281. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  282. ---
  283.  tests/manual_testing.txt | 1 +
  284.  1 file changed, 1 insertion(+)
  285.  
  286. diff --git a/tests/manual_testing.txt b/tests/manual_testing.txt
  287. index a0c9408..c9c34a6 100644
  288. --- a/tests/manual_testing.txt
  289. +++ b/tests/manual_testing.txt
  290. @@ -19,6 +19,7 @@
  291.  #   libmpfr-devel
  292.  #   libmpc-devel
  293.  #   libintl-devel
  294. +#   libisl-devel
  295.  #   flex
  296.  #   bison
  297.  #   ---- snip ----
  298. --
  299. 2.43.0

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.

Syntax highlighting:

To highlight particular lines, prefix each line with {%HIGHLIGHT}




All content is user-submitted.
The administrators of this site (kpaste.net) are not responsible for their content.
Abuse reports should be emailed to us at