[PATCH 2/2] cifs: don't add dot and dotdot entry by default at the start of cifs_readdir

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Currently, dot and dotdot are added by default at the start of cifs_readdir.
This works well for the servers which send dot and dotdot entries at the start
in readdir query response.

For servers which do not send these two entries at the beginning,
this behavior creates 2 problems.
1) adding dot and dotdot increments ctx->pos by 2 hence the valid
   first 2 entries are missed in readdir output at client side.
2) Currently when we encounter dot or dotdot, the entry is simple skipped.

In case of smb2, if some buggy server sends only 1 entry in response per
cifs_query_dir_next request, and if that entry is happen to be dot OR dotdot,
getdents call return 0 bytes to user app making it falsely believe that there
are no more directory entries left.

This patch tries to solve the above issues by processing dot and dotdot entries
when they are encountered. This will have no effect on servers which already
sends these 2 entries at the start.

Signed-off-by: Namjae Jeon <namjae.jeon@xxxxxxxxxxx>
Signed-off-by: Ashish Sangwan <a.sangwan@xxxxxxxxxxx>
---
 fs/cifs/cifsglob.h |  2 ++
 fs/cifs/cifssmb.c  |  4 ++--
 fs/cifs/readdir.c  | 43 ++++++++++++++++++++++++++++++++++++++-----
 3 files changed, 42 insertions(+), 7 deletions(-)

diff --git a/fs/cifs/cifsglob.h b/fs/cifs/cifsglob.h
index 6e13911..864ea1a 100644
--- a/fs/cifs/cifsglob.h
+++ b/fs/cifs/cifsglob.h
@@ -987,6 +987,8 @@ struct cifs_search_info {
 	bool emptyDir:1;
 	bool unicode:1;
 	bool smallBuf:1; /* so we know which buf_release function to call */
+	bool gotdot:1;
+	bool gotdotdot:1;
 };
 
 struct cifs_open_parms {
diff --git a/fs/cifs/cifssmb.c b/fs/cifs/cifssmb.c
index fa13d5e..db49633 100644
--- a/fs/cifs/cifssmb.c
+++ b/fs/cifs/cifssmb.c
@@ -4475,8 +4475,8 @@ findFirstRetry:
 
 			psrch_inf->entries_in_buffer =
 					le16_to_cpu(parms->SearchCount);
-			psrch_inf->index_of_last_entry = 2 /* skip . and .. */ +
-				psrch_inf->entries_in_buffer;
+			psrch_inf->index_of_last_entry =
+					psrch_inf->entries_in_buffer;
 			lnoff = le16_to_cpu(parms->LastNameOffset);
 			if (CIFSMaxBufSize < lnoff) {
 				cifs_dbg(VFS, "ignoring corrupt resume name\n");
diff --git a/fs/cifs/readdir.c b/fs/cifs/readdir.c
index 8eaf20a..ab0639a 100644
--- a/fs/cifs/readdir.c
+++ b/fs/cifs/readdir.c
@@ -698,9 +698,26 @@ static int cifs_filldir(char *find_entry, struct file *file,
 		return -EINVAL;
 	}
 
-	/* skip . and .. since we added them first */
-	if (cifs_entry_is_dot(&de, file_info->srch_inf.unicode))
+	if (cifs_entry_is_dot(&de, file_info->srch_inf.unicode) == 1) {
+		if (file_info->srch_inf.gotdot == true)
+			return -EINVAL;
+		if (!dir_emit_dot(file, ctx)) {
+			cifs_dbg(VFS, "Filldir for current dir failed\n");
+			return -ENOMEM;
+		}
+		file_info->srch_inf.gotdot = true;
+		return 0;
+	}
+	if (cifs_entry_is_dot(&de, file_info->srch_inf.unicode) == 2) {
+		if (file_info->srch_inf.gotdotdot == true)
+			return -EINVAL;
+		if (!dir_emit_dotdot(file, ctx)) {
+			cifs_dbg(VFS, "Filldir for parent dir failed\n");
+			return -ENOMEM;
+		}
+		file_info->srch_inf.gotdotdot = true;
 		return 0;
+	}
 
 	if (file_info->srch_inf.unicode) {
 		struct nls_table *nlt = cifs_sb->local_nls;
@@ -786,9 +803,6 @@ int cifs_readdir(struct file *file, struct dir_context *ctx)
 			goto rddir2_exit;
 	}
 
-	if (!dir_emit_dots(file, ctx))
-		goto rddir2_exit;
-
 	/* 1) If search is active,
 		is in current search buffer?
 		if it before then restart search
@@ -864,6 +878,25 @@ int cifs_readdir(struct file *file, struct dir_context *ctx)
 	kfree(tmp_buf);
 
 rddir2_exit:
+	if (cifsFile->srch_inf.endOfSearch) {
+		if (cifsFile->srch_inf.gotdot != true) {
+			if (!dir_emit_dot(file, ctx)) {
+				cifs_dbg(VFS,
+					"Filldir for current dir failed\n");
+				rc = -ENOMEM;
+			}
+			ctx->pos++;
+		}
+
+		if (cifsFile->srch_inf.gotdotdot != true) {
+			if (!dir_emit_dotdot(file, ctx)) {
+				cifs_dbg(VFS,
+					"Filldir for parent dir failed\n");
+				rc = -ENOMEM;
+			}
+			ctx->pos++;
+		}
+	}
 	free_xid(xid);
 	return rc;
 }
-- 
1.7.11-rc0

--
To unsubscribe from this list: send the line "unsubscribe linux-cifs" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html




[Linux USB Devel]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux