]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
smb: smbdirect: introduce smbdirect_socket_wait_for_credits()
authorStefan Metzmacher <metze@samba.org>
Fri, 19 Sep 2025 07:48:58 +0000 (09:48 +0200)
committerSteve French <stfrench@microsoft.com>
Thu, 16 Apr 2026 02:58:19 +0000 (21:58 -0500)
This is a copy of wait_for_credits() in the server, which
will be replaced by this soon.

This will allow us to share more common code between client
and server soon.

Cc: Steve French <smfrench@gmail.com>
Cc: Tom Talpey <tom@talpey.com>
Cc: Long Li <longli@microsoft.com>
Cc: Namjae Jeon <linkinjeon@kernel.org>
Cc: linux-cifs@vger.kernel.org
Cc: samba-technical@lists.samba.org
Signed-off-by: Stefan Metzmacher <metze@samba.org>
Acked-by: Namjae Jeon <linkinjeon@kernel.org>
Signed-off-by: Steve French <stfrench@microsoft.com>
fs/smb/common/smbdirect/smbdirect_internal.h
fs/smb/common/smbdirect/smbdirect_socket.c

index 2d7c69f71ee064a8f8f8ff17a7d68b32a6cdf219..ff4db1c3f12815e6599b09c38ee68259a38f9eb1 100644 (file)
@@ -32,6 +32,13 @@ static void __smbdirect_socket_schedule_cleanup(struct smbdirect_socket *sc,
                __func__, __LINE__, __error, &__force_status); \
 } while (0)
 
+static int smbdirect_socket_wait_for_credits(struct smbdirect_socket *sc,
+                                            enum smbdirect_socket_status expected_status,
+                                            int unexpected_errno,
+                                            wait_queue_head_t *waitq,
+                                            atomic_t *total_credits,
+                                            int needed);
+
 static void smbdirect_connection_idle_timer_work(struct work_struct *work);
 
 #endif /* __FS_SMB_COMMON_SMBDIRECT_INTERNAL_H__ */
index 05a284526aa2179dfdf5ff07c966e9d6fb6451fd..bbd794fddc1ecd526aa7bf8cc36b410958038116 100644 (file)
@@ -250,3 +250,32 @@ static void smbdirect_socket_cleanup_work(struct work_struct *work)
         */
        smbdirect_socket_wake_up_all(sc);
 }
+
+__maybe_unused /* this is temporary while this file is included in others */
+static int smbdirect_socket_wait_for_credits(struct smbdirect_socket *sc,
+                                            enum smbdirect_socket_status expected_status,
+                                            int unexpected_errno,
+                                            wait_queue_head_t *waitq,
+                                            atomic_t *total_credits,
+                                            int needed)
+{
+       int ret;
+
+       if (WARN_ON_ONCE(needed < 0))
+               return -EINVAL;
+
+       do {
+               if (atomic_sub_return(needed, total_credits) >= 0)
+                       return 0;
+
+               atomic_add(needed, total_credits);
+               ret = wait_event_interruptible(*waitq,
+                                              atomic_read(total_credits) >= needed ||
+                                              sc->status != expected_status);
+
+               if (sc->status != expected_status)
+                       return unexpected_errno;
+               else if (ret < 0)
+                       return ret;
+       } while (true);
+}