162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci *
462306a36Sopenharmony_ci * Copyright (C) 2019-2021 Paragon Software GmbH, All rights reserved.
562306a36Sopenharmony_ci *
662306a36Sopenharmony_ci *  Directory handling functions for NTFS-based filesystems.
762306a36Sopenharmony_ci *
862306a36Sopenharmony_ci */
962306a36Sopenharmony_ci
1062306a36Sopenharmony_ci#include <linux/fs.h>
1162306a36Sopenharmony_ci#include <linux/nls.h>
1262306a36Sopenharmony_ci
1362306a36Sopenharmony_ci#include "debug.h"
1462306a36Sopenharmony_ci#include "ntfs.h"
1562306a36Sopenharmony_ci#include "ntfs_fs.h"
1662306a36Sopenharmony_ci
1762306a36Sopenharmony_ci/* Convert little endian UTF-16 to NLS string. */
1862306a36Sopenharmony_ciint ntfs_utf16_to_nls(struct ntfs_sb_info *sbi, const __le16 *name, u32 len,
1962306a36Sopenharmony_ci		      u8 *buf, int buf_len)
2062306a36Sopenharmony_ci{
2162306a36Sopenharmony_ci	int ret, warn;
2262306a36Sopenharmony_ci	u8 *op;
2362306a36Sopenharmony_ci	struct nls_table *nls = sbi->options->nls;
2462306a36Sopenharmony_ci
2562306a36Sopenharmony_ci	static_assert(sizeof(wchar_t) == sizeof(__le16));
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_ci	if (!nls) {
2862306a36Sopenharmony_ci		/* UTF-16 -> UTF-8 */
2962306a36Sopenharmony_ci		ret = utf16s_to_utf8s((wchar_t *)name, len, UTF16_LITTLE_ENDIAN,
3062306a36Sopenharmony_ci				      buf, buf_len);
3162306a36Sopenharmony_ci		buf[ret] = '\0';
3262306a36Sopenharmony_ci		return ret;
3362306a36Sopenharmony_ci	}
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_ci	op = buf;
3662306a36Sopenharmony_ci	warn = 0;
3762306a36Sopenharmony_ci
3862306a36Sopenharmony_ci	while (len--) {
3962306a36Sopenharmony_ci		u16 ec;
4062306a36Sopenharmony_ci		int charlen;
4162306a36Sopenharmony_ci		char dump[5];
4262306a36Sopenharmony_ci
4362306a36Sopenharmony_ci		if (buf_len < NLS_MAX_CHARSET_SIZE) {
4462306a36Sopenharmony_ci			ntfs_warn(sbi->sb,
4562306a36Sopenharmony_ci				  "filename was truncated while converting.");
4662306a36Sopenharmony_ci			break;
4762306a36Sopenharmony_ci		}
4862306a36Sopenharmony_ci
4962306a36Sopenharmony_ci		ec = le16_to_cpu(*name++);
5062306a36Sopenharmony_ci		charlen = nls->uni2char(ec, op, buf_len);
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_ci		if (charlen > 0) {
5362306a36Sopenharmony_ci			op += charlen;
5462306a36Sopenharmony_ci			buf_len -= charlen;
5562306a36Sopenharmony_ci			continue;
5662306a36Sopenharmony_ci		}
5762306a36Sopenharmony_ci
5862306a36Sopenharmony_ci		*op++ = '_';
5962306a36Sopenharmony_ci		buf_len -= 1;
6062306a36Sopenharmony_ci		if (warn)
6162306a36Sopenharmony_ci			continue;
6262306a36Sopenharmony_ci
6362306a36Sopenharmony_ci		warn = 1;
6462306a36Sopenharmony_ci		hex_byte_pack(&dump[0], ec >> 8);
6562306a36Sopenharmony_ci		hex_byte_pack(&dump[2], ec);
6662306a36Sopenharmony_ci		dump[4] = 0;
6762306a36Sopenharmony_ci
6862306a36Sopenharmony_ci		ntfs_err(sbi->sb, "failed to convert \"%s\" to %s", dump,
6962306a36Sopenharmony_ci			 nls->charset);
7062306a36Sopenharmony_ci	}
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_ci	*op = '\0';
7362306a36Sopenharmony_ci	return op - buf;
7462306a36Sopenharmony_ci}
7562306a36Sopenharmony_ci
7662306a36Sopenharmony_ci// clang-format off
7762306a36Sopenharmony_ci#define PLANE_SIZE	0x00010000
7862306a36Sopenharmony_ci
7962306a36Sopenharmony_ci#define SURROGATE_PAIR	0x0000d800
8062306a36Sopenharmony_ci#define SURROGATE_LOW	0x00000400
8162306a36Sopenharmony_ci#define SURROGATE_BITS	0x000003ff
8262306a36Sopenharmony_ci// clang-format on
8362306a36Sopenharmony_ci
8462306a36Sopenharmony_ci/*
8562306a36Sopenharmony_ci * put_utf16 - Modified version of put_utf16 from fs/nls/nls_base.c
8662306a36Sopenharmony_ci *
8762306a36Sopenharmony_ci * Function is sparse warnings free.
8862306a36Sopenharmony_ci */
8962306a36Sopenharmony_cistatic inline void put_utf16(wchar_t *s, unsigned int c,
9062306a36Sopenharmony_ci			     enum utf16_endian endian)
9162306a36Sopenharmony_ci{
9262306a36Sopenharmony_ci	static_assert(sizeof(wchar_t) == sizeof(__le16));
9362306a36Sopenharmony_ci	static_assert(sizeof(wchar_t) == sizeof(__be16));
9462306a36Sopenharmony_ci
9562306a36Sopenharmony_ci	switch (endian) {
9662306a36Sopenharmony_ci	default:
9762306a36Sopenharmony_ci		*s = (wchar_t)c;
9862306a36Sopenharmony_ci		break;
9962306a36Sopenharmony_ci	case UTF16_LITTLE_ENDIAN:
10062306a36Sopenharmony_ci		*(__le16 *)s = __cpu_to_le16(c);
10162306a36Sopenharmony_ci		break;
10262306a36Sopenharmony_ci	case UTF16_BIG_ENDIAN:
10362306a36Sopenharmony_ci		*(__be16 *)s = __cpu_to_be16(c);
10462306a36Sopenharmony_ci		break;
10562306a36Sopenharmony_ci	}
10662306a36Sopenharmony_ci}
10762306a36Sopenharmony_ci
10862306a36Sopenharmony_ci/*
10962306a36Sopenharmony_ci * _utf8s_to_utf16s
11062306a36Sopenharmony_ci *
11162306a36Sopenharmony_ci * Modified version of 'utf8s_to_utf16s' allows to
11262306a36Sopenharmony_ci * detect -ENAMETOOLONG without writing out of expected maximum.
11362306a36Sopenharmony_ci */
11462306a36Sopenharmony_cistatic int _utf8s_to_utf16s(const u8 *s, int inlen, enum utf16_endian endian,
11562306a36Sopenharmony_ci			    wchar_t *pwcs, int maxout)
11662306a36Sopenharmony_ci{
11762306a36Sopenharmony_ci	u16 *op;
11862306a36Sopenharmony_ci	int size;
11962306a36Sopenharmony_ci	unicode_t u;
12062306a36Sopenharmony_ci
12162306a36Sopenharmony_ci	op = pwcs;
12262306a36Sopenharmony_ci	while (inlen > 0 && *s) {
12362306a36Sopenharmony_ci		if (*s & 0x80) {
12462306a36Sopenharmony_ci			size = utf8_to_utf32(s, inlen, &u);
12562306a36Sopenharmony_ci			if (size < 0)
12662306a36Sopenharmony_ci				return -EINVAL;
12762306a36Sopenharmony_ci			s += size;
12862306a36Sopenharmony_ci			inlen -= size;
12962306a36Sopenharmony_ci
13062306a36Sopenharmony_ci			if (u >= PLANE_SIZE) {
13162306a36Sopenharmony_ci				if (maxout < 2)
13262306a36Sopenharmony_ci					return -ENAMETOOLONG;
13362306a36Sopenharmony_ci
13462306a36Sopenharmony_ci				u -= PLANE_SIZE;
13562306a36Sopenharmony_ci				put_utf16(op++,
13662306a36Sopenharmony_ci					  SURROGATE_PAIR |
13762306a36Sopenharmony_ci						  ((u >> 10) & SURROGATE_BITS),
13862306a36Sopenharmony_ci					  endian);
13962306a36Sopenharmony_ci				put_utf16(op++,
14062306a36Sopenharmony_ci					  SURROGATE_PAIR | SURROGATE_LOW |
14162306a36Sopenharmony_ci						  (u & SURROGATE_BITS),
14262306a36Sopenharmony_ci					  endian);
14362306a36Sopenharmony_ci				maxout -= 2;
14462306a36Sopenharmony_ci			} else {
14562306a36Sopenharmony_ci				if (maxout < 1)
14662306a36Sopenharmony_ci					return -ENAMETOOLONG;
14762306a36Sopenharmony_ci
14862306a36Sopenharmony_ci				put_utf16(op++, u, endian);
14962306a36Sopenharmony_ci				maxout--;
15062306a36Sopenharmony_ci			}
15162306a36Sopenharmony_ci		} else {
15262306a36Sopenharmony_ci			if (maxout < 1)
15362306a36Sopenharmony_ci				return -ENAMETOOLONG;
15462306a36Sopenharmony_ci
15562306a36Sopenharmony_ci			put_utf16(op++, *s++, endian);
15662306a36Sopenharmony_ci			inlen--;
15762306a36Sopenharmony_ci			maxout--;
15862306a36Sopenharmony_ci		}
15962306a36Sopenharmony_ci	}
16062306a36Sopenharmony_ci	return op - pwcs;
16162306a36Sopenharmony_ci}
16262306a36Sopenharmony_ci
16362306a36Sopenharmony_ci/*
16462306a36Sopenharmony_ci * ntfs_nls_to_utf16 - Convert input string to UTF-16.
16562306a36Sopenharmony_ci * @name:	Input name.
16662306a36Sopenharmony_ci * @name_len:	Input name length.
16762306a36Sopenharmony_ci * @uni:	Destination memory.
16862306a36Sopenharmony_ci * @max_ulen:	Destination memory.
16962306a36Sopenharmony_ci * @endian:	Endian of target UTF-16 string.
17062306a36Sopenharmony_ci *
17162306a36Sopenharmony_ci * This function is called:
17262306a36Sopenharmony_ci * - to create NTFS name
17362306a36Sopenharmony_ci * - to create symlink
17462306a36Sopenharmony_ci *
17562306a36Sopenharmony_ci * Return: UTF-16 string length or error (if negative).
17662306a36Sopenharmony_ci */
17762306a36Sopenharmony_ciint ntfs_nls_to_utf16(struct ntfs_sb_info *sbi, const u8 *name, u32 name_len,
17862306a36Sopenharmony_ci		      struct cpu_str *uni, u32 max_ulen,
17962306a36Sopenharmony_ci		      enum utf16_endian endian)
18062306a36Sopenharmony_ci{
18162306a36Sopenharmony_ci	int ret, slen;
18262306a36Sopenharmony_ci	const u8 *end;
18362306a36Sopenharmony_ci	struct nls_table *nls = sbi->options->nls;
18462306a36Sopenharmony_ci	u16 *uname = uni->name;
18562306a36Sopenharmony_ci
18662306a36Sopenharmony_ci	static_assert(sizeof(wchar_t) == sizeof(u16));
18762306a36Sopenharmony_ci
18862306a36Sopenharmony_ci	if (!nls) {
18962306a36Sopenharmony_ci		/* utf8 -> utf16 */
19062306a36Sopenharmony_ci		ret = _utf8s_to_utf16s(name, name_len, endian, uname, max_ulen);
19162306a36Sopenharmony_ci		uni->len = ret;
19262306a36Sopenharmony_ci		return ret;
19362306a36Sopenharmony_ci	}
19462306a36Sopenharmony_ci
19562306a36Sopenharmony_ci	for (ret = 0, end = name + name_len; name < end; ret++, name += slen) {
19662306a36Sopenharmony_ci		if (ret >= max_ulen)
19762306a36Sopenharmony_ci			return -ENAMETOOLONG;
19862306a36Sopenharmony_ci
19962306a36Sopenharmony_ci		slen = nls->char2uni(name, end - name, uname + ret);
20062306a36Sopenharmony_ci		if (!slen)
20162306a36Sopenharmony_ci			return -EINVAL;
20262306a36Sopenharmony_ci		if (slen < 0)
20362306a36Sopenharmony_ci			return slen;
20462306a36Sopenharmony_ci	}
20562306a36Sopenharmony_ci
20662306a36Sopenharmony_ci#ifdef __BIG_ENDIAN
20762306a36Sopenharmony_ci	if (endian == UTF16_LITTLE_ENDIAN) {
20862306a36Sopenharmony_ci		int i = ret;
20962306a36Sopenharmony_ci
21062306a36Sopenharmony_ci		while (i--) {
21162306a36Sopenharmony_ci			__cpu_to_le16s(uname);
21262306a36Sopenharmony_ci			uname++;
21362306a36Sopenharmony_ci		}
21462306a36Sopenharmony_ci	}
21562306a36Sopenharmony_ci#else
21662306a36Sopenharmony_ci	if (endian == UTF16_BIG_ENDIAN) {
21762306a36Sopenharmony_ci		int i = ret;
21862306a36Sopenharmony_ci
21962306a36Sopenharmony_ci		while (i--) {
22062306a36Sopenharmony_ci			__cpu_to_be16s(uname);
22162306a36Sopenharmony_ci			uname++;
22262306a36Sopenharmony_ci		}
22362306a36Sopenharmony_ci	}
22462306a36Sopenharmony_ci#endif
22562306a36Sopenharmony_ci
22662306a36Sopenharmony_ci	uni->len = ret;
22762306a36Sopenharmony_ci	return ret;
22862306a36Sopenharmony_ci}
22962306a36Sopenharmony_ci
23062306a36Sopenharmony_ci/*
23162306a36Sopenharmony_ci * dir_search_u - Helper function.
23262306a36Sopenharmony_ci */
23362306a36Sopenharmony_cistruct inode *dir_search_u(struct inode *dir, const struct cpu_str *uni,
23462306a36Sopenharmony_ci			   struct ntfs_fnd *fnd)
23562306a36Sopenharmony_ci{
23662306a36Sopenharmony_ci	int err = 0;
23762306a36Sopenharmony_ci	struct super_block *sb = dir->i_sb;
23862306a36Sopenharmony_ci	struct ntfs_sb_info *sbi = sb->s_fs_info;
23962306a36Sopenharmony_ci	struct ntfs_inode *ni = ntfs_i(dir);
24062306a36Sopenharmony_ci	struct NTFS_DE *e;
24162306a36Sopenharmony_ci	int diff;
24262306a36Sopenharmony_ci	struct inode *inode = NULL;
24362306a36Sopenharmony_ci	struct ntfs_fnd *fnd_a = NULL;
24462306a36Sopenharmony_ci
24562306a36Sopenharmony_ci	if (!fnd) {
24662306a36Sopenharmony_ci		fnd_a = fnd_get();
24762306a36Sopenharmony_ci		if (!fnd_a) {
24862306a36Sopenharmony_ci			err = -ENOMEM;
24962306a36Sopenharmony_ci			goto out;
25062306a36Sopenharmony_ci		}
25162306a36Sopenharmony_ci		fnd = fnd_a;
25262306a36Sopenharmony_ci	}
25362306a36Sopenharmony_ci
25462306a36Sopenharmony_ci	err = indx_find(&ni->dir, ni, NULL, uni, 0, sbi, &diff, &e, fnd);
25562306a36Sopenharmony_ci
25662306a36Sopenharmony_ci	if (err)
25762306a36Sopenharmony_ci		goto out;
25862306a36Sopenharmony_ci
25962306a36Sopenharmony_ci	if (diff) {
26062306a36Sopenharmony_ci		err = -ENOENT;
26162306a36Sopenharmony_ci		goto out;
26262306a36Sopenharmony_ci	}
26362306a36Sopenharmony_ci
26462306a36Sopenharmony_ci	inode = ntfs_iget5(sb, &e->ref, uni);
26562306a36Sopenharmony_ci	if (!IS_ERR(inode) && is_bad_inode(inode)) {
26662306a36Sopenharmony_ci		iput(inode);
26762306a36Sopenharmony_ci		err = -EINVAL;
26862306a36Sopenharmony_ci	}
26962306a36Sopenharmony_ciout:
27062306a36Sopenharmony_ci	fnd_put(fnd_a);
27162306a36Sopenharmony_ci
27262306a36Sopenharmony_ci	return err == -ENOENT ? NULL : err ? ERR_PTR(err) : inode;
27362306a36Sopenharmony_ci}
27462306a36Sopenharmony_ci
27562306a36Sopenharmony_cistatic inline int ntfs_filldir(struct ntfs_sb_info *sbi, struct ntfs_inode *ni,
27662306a36Sopenharmony_ci			       const struct NTFS_DE *e, u8 *name,
27762306a36Sopenharmony_ci			       struct dir_context *ctx)
27862306a36Sopenharmony_ci{
27962306a36Sopenharmony_ci	const struct ATTR_FILE_NAME *fname;
28062306a36Sopenharmony_ci	unsigned long ino;
28162306a36Sopenharmony_ci	int name_len;
28262306a36Sopenharmony_ci	u32 dt_type;
28362306a36Sopenharmony_ci
28462306a36Sopenharmony_ci	fname = Add2Ptr(e, sizeof(struct NTFS_DE));
28562306a36Sopenharmony_ci
28662306a36Sopenharmony_ci	if (fname->type == FILE_NAME_DOS)
28762306a36Sopenharmony_ci		return 0;
28862306a36Sopenharmony_ci
28962306a36Sopenharmony_ci	if (!mi_is_ref(&ni->mi, &fname->home))
29062306a36Sopenharmony_ci		return 0;
29162306a36Sopenharmony_ci
29262306a36Sopenharmony_ci	ino = ino_get(&e->ref);
29362306a36Sopenharmony_ci
29462306a36Sopenharmony_ci	if (ino == MFT_REC_ROOT)
29562306a36Sopenharmony_ci		return 0;
29662306a36Sopenharmony_ci
29762306a36Sopenharmony_ci	/* Skip meta files. Unless option to show metafiles is set. */
29862306a36Sopenharmony_ci	if (!sbi->options->showmeta && ntfs_is_meta_file(sbi, ino))
29962306a36Sopenharmony_ci		return 0;
30062306a36Sopenharmony_ci
30162306a36Sopenharmony_ci	if (sbi->options->nohidden && (fname->dup.fa & FILE_ATTRIBUTE_HIDDEN))
30262306a36Sopenharmony_ci		return 0;
30362306a36Sopenharmony_ci
30462306a36Sopenharmony_ci	name_len = ntfs_utf16_to_nls(sbi, fname->name, fname->name_len, name,
30562306a36Sopenharmony_ci				     PATH_MAX);
30662306a36Sopenharmony_ci	if (name_len <= 0) {
30762306a36Sopenharmony_ci		ntfs_warn(sbi->sb, "failed to convert name for inode %lx.",
30862306a36Sopenharmony_ci			  ino);
30962306a36Sopenharmony_ci		return 0;
31062306a36Sopenharmony_ci	}
31162306a36Sopenharmony_ci
31262306a36Sopenharmony_ci	/*
31362306a36Sopenharmony_ci	 * NTFS: symlinks are "dir + reparse" or "file + reparse"
31462306a36Sopenharmony_ci	 * Unfortunately reparse attribute is used for many purposes (several dozens).
31562306a36Sopenharmony_ci	 * It is not possible here to know is this name symlink or not.
31662306a36Sopenharmony_ci	 * To get exactly the type of name we should to open inode (read mft).
31762306a36Sopenharmony_ci	 * getattr for opened file (fstat) correctly returns symlink.
31862306a36Sopenharmony_ci	 */
31962306a36Sopenharmony_ci	dt_type = (fname->dup.fa & FILE_ATTRIBUTE_DIRECTORY) ? DT_DIR : DT_REG;
32062306a36Sopenharmony_ci
32162306a36Sopenharmony_ci	/*
32262306a36Sopenharmony_ci	 * It is not reliable to detect the type of name using duplicated information
32362306a36Sopenharmony_ci	 * stored in parent directory.
32462306a36Sopenharmony_ci	 * The only correct way to get the type of name - read MFT record and find ATTR_STD.
32562306a36Sopenharmony_ci	 * The code below is not good idea.
32662306a36Sopenharmony_ci	 * It does additional locks/reads just to get the type of name.
32762306a36Sopenharmony_ci	 * Should we use additional mount option to enable branch below?
32862306a36Sopenharmony_ci	 */
32962306a36Sopenharmony_ci	if ((fname->dup.fa & FILE_ATTRIBUTE_REPARSE_POINT) &&
33062306a36Sopenharmony_ci	    ino != ni->mi.rno) {
33162306a36Sopenharmony_ci		struct inode *inode = ntfs_iget5(sbi->sb, &e->ref, NULL);
33262306a36Sopenharmony_ci		if (!IS_ERR_OR_NULL(inode)) {
33362306a36Sopenharmony_ci			dt_type = fs_umode_to_dtype(inode->i_mode);
33462306a36Sopenharmony_ci			iput(inode);
33562306a36Sopenharmony_ci		}
33662306a36Sopenharmony_ci	}
33762306a36Sopenharmony_ci
33862306a36Sopenharmony_ci	return !dir_emit(ctx, (s8 *)name, name_len, ino, dt_type);
33962306a36Sopenharmony_ci}
34062306a36Sopenharmony_ci
34162306a36Sopenharmony_ci/*
34262306a36Sopenharmony_ci * ntfs_read_hdr - Helper function for ntfs_readdir().
34362306a36Sopenharmony_ci */
34462306a36Sopenharmony_cistatic int ntfs_read_hdr(struct ntfs_sb_info *sbi, struct ntfs_inode *ni,
34562306a36Sopenharmony_ci			 const struct INDEX_HDR *hdr, u64 vbo, u64 pos,
34662306a36Sopenharmony_ci			 u8 *name, struct dir_context *ctx)
34762306a36Sopenharmony_ci{
34862306a36Sopenharmony_ci	int err;
34962306a36Sopenharmony_ci	const struct NTFS_DE *e;
35062306a36Sopenharmony_ci	u32 e_size;
35162306a36Sopenharmony_ci	u32 end = le32_to_cpu(hdr->used);
35262306a36Sopenharmony_ci	u32 off = le32_to_cpu(hdr->de_off);
35362306a36Sopenharmony_ci
35462306a36Sopenharmony_ci	for (;; off += e_size) {
35562306a36Sopenharmony_ci		if (off + sizeof(struct NTFS_DE) > end)
35662306a36Sopenharmony_ci			return -1;
35762306a36Sopenharmony_ci
35862306a36Sopenharmony_ci		e = Add2Ptr(hdr, off);
35962306a36Sopenharmony_ci		e_size = le16_to_cpu(e->size);
36062306a36Sopenharmony_ci		if (e_size < sizeof(struct NTFS_DE) || off + e_size > end)
36162306a36Sopenharmony_ci			return -1;
36262306a36Sopenharmony_ci
36362306a36Sopenharmony_ci		if (de_is_last(e))
36462306a36Sopenharmony_ci			return 0;
36562306a36Sopenharmony_ci
36662306a36Sopenharmony_ci		/* Skip already enumerated. */
36762306a36Sopenharmony_ci		if (vbo + off < pos)
36862306a36Sopenharmony_ci			continue;
36962306a36Sopenharmony_ci
37062306a36Sopenharmony_ci		if (le16_to_cpu(e->key_size) < SIZEOF_ATTRIBUTE_FILENAME)
37162306a36Sopenharmony_ci			return -1;
37262306a36Sopenharmony_ci
37362306a36Sopenharmony_ci		ctx->pos = vbo + off;
37462306a36Sopenharmony_ci
37562306a36Sopenharmony_ci		/* Submit the name to the filldir callback. */
37662306a36Sopenharmony_ci		err = ntfs_filldir(sbi, ni, e, name, ctx);
37762306a36Sopenharmony_ci		if (err)
37862306a36Sopenharmony_ci			return err;
37962306a36Sopenharmony_ci	}
38062306a36Sopenharmony_ci}
38162306a36Sopenharmony_ci
38262306a36Sopenharmony_ci/*
38362306a36Sopenharmony_ci * ntfs_readdir - file_operations::iterate_shared
38462306a36Sopenharmony_ci *
38562306a36Sopenharmony_ci * Use non sorted enumeration.
38662306a36Sopenharmony_ci * We have an example of broken volume where sorted enumeration
38762306a36Sopenharmony_ci * counts each name twice.
38862306a36Sopenharmony_ci */
38962306a36Sopenharmony_cistatic int ntfs_readdir(struct file *file, struct dir_context *ctx)
39062306a36Sopenharmony_ci{
39162306a36Sopenharmony_ci	const struct INDEX_ROOT *root;
39262306a36Sopenharmony_ci	u64 vbo;
39362306a36Sopenharmony_ci	size_t bit;
39462306a36Sopenharmony_ci	loff_t eod;
39562306a36Sopenharmony_ci	int err = 0;
39662306a36Sopenharmony_ci	struct inode *dir = file_inode(file);
39762306a36Sopenharmony_ci	struct ntfs_inode *ni = ntfs_i(dir);
39862306a36Sopenharmony_ci	struct super_block *sb = dir->i_sb;
39962306a36Sopenharmony_ci	struct ntfs_sb_info *sbi = sb->s_fs_info;
40062306a36Sopenharmony_ci	loff_t i_size = i_size_read(dir);
40162306a36Sopenharmony_ci	u32 pos = ctx->pos;
40262306a36Sopenharmony_ci	u8 *name = NULL;
40362306a36Sopenharmony_ci	struct indx_node *node = NULL;
40462306a36Sopenharmony_ci	u8 index_bits = ni->dir.index_bits;
40562306a36Sopenharmony_ci
40662306a36Sopenharmony_ci	/* Name is a buffer of PATH_MAX length. */
40762306a36Sopenharmony_ci	static_assert(NTFS_NAME_LEN * 4 < PATH_MAX);
40862306a36Sopenharmony_ci
40962306a36Sopenharmony_ci	eod = i_size + sbi->record_size;
41062306a36Sopenharmony_ci
41162306a36Sopenharmony_ci	if (pos >= eod)
41262306a36Sopenharmony_ci		return 0;
41362306a36Sopenharmony_ci
41462306a36Sopenharmony_ci	if (!dir_emit_dots(file, ctx))
41562306a36Sopenharmony_ci		return 0;
41662306a36Sopenharmony_ci
41762306a36Sopenharmony_ci	/* Allocate PATH_MAX bytes. */
41862306a36Sopenharmony_ci	name = __getname();
41962306a36Sopenharmony_ci	if (!name)
42062306a36Sopenharmony_ci		return -ENOMEM;
42162306a36Sopenharmony_ci
42262306a36Sopenharmony_ci	if (!ni->mi_loaded && ni->attr_list.size) {
42362306a36Sopenharmony_ci		/*
42462306a36Sopenharmony_ci		 * Directory inode is locked for read.
42562306a36Sopenharmony_ci		 * Load all subrecords to avoid 'write' access to 'ni' during
42662306a36Sopenharmony_ci		 * directory reading.
42762306a36Sopenharmony_ci		 */
42862306a36Sopenharmony_ci		ni_lock(ni);
42962306a36Sopenharmony_ci		if (!ni->mi_loaded && ni->attr_list.size) {
43062306a36Sopenharmony_ci			err = ni_load_all_mi(ni);
43162306a36Sopenharmony_ci			if (!err)
43262306a36Sopenharmony_ci				ni->mi_loaded = true;
43362306a36Sopenharmony_ci		}
43462306a36Sopenharmony_ci		ni_unlock(ni);
43562306a36Sopenharmony_ci		if (err)
43662306a36Sopenharmony_ci			goto out;
43762306a36Sopenharmony_ci	}
43862306a36Sopenharmony_ci
43962306a36Sopenharmony_ci	root = indx_get_root(&ni->dir, ni, NULL, NULL);
44062306a36Sopenharmony_ci	if (!root) {
44162306a36Sopenharmony_ci		err = -EINVAL;
44262306a36Sopenharmony_ci		goto out;
44362306a36Sopenharmony_ci	}
44462306a36Sopenharmony_ci
44562306a36Sopenharmony_ci	if (pos >= sbi->record_size) {
44662306a36Sopenharmony_ci		bit = (pos - sbi->record_size) >> index_bits;
44762306a36Sopenharmony_ci	} else {
44862306a36Sopenharmony_ci		err = ntfs_read_hdr(sbi, ni, &root->ihdr, 0, pos, name, ctx);
44962306a36Sopenharmony_ci		if (err)
45062306a36Sopenharmony_ci			goto out;
45162306a36Sopenharmony_ci		bit = 0;
45262306a36Sopenharmony_ci	}
45362306a36Sopenharmony_ci
45462306a36Sopenharmony_ci	if (!i_size) {
45562306a36Sopenharmony_ci		ctx->pos = eod;
45662306a36Sopenharmony_ci		goto out;
45762306a36Sopenharmony_ci	}
45862306a36Sopenharmony_ci
45962306a36Sopenharmony_ci	for (;;) {
46062306a36Sopenharmony_ci		vbo = (u64)bit << index_bits;
46162306a36Sopenharmony_ci		if (vbo >= i_size) {
46262306a36Sopenharmony_ci			ctx->pos = eod;
46362306a36Sopenharmony_ci			goto out;
46462306a36Sopenharmony_ci		}
46562306a36Sopenharmony_ci
46662306a36Sopenharmony_ci		err = indx_used_bit(&ni->dir, ni, &bit);
46762306a36Sopenharmony_ci		if (err)
46862306a36Sopenharmony_ci			goto out;
46962306a36Sopenharmony_ci
47062306a36Sopenharmony_ci		if (bit == MINUS_ONE_T) {
47162306a36Sopenharmony_ci			ctx->pos = eod;
47262306a36Sopenharmony_ci			goto out;
47362306a36Sopenharmony_ci		}
47462306a36Sopenharmony_ci
47562306a36Sopenharmony_ci		vbo = (u64)bit << index_bits;
47662306a36Sopenharmony_ci		if (vbo >= i_size) {
47762306a36Sopenharmony_ci			ntfs_inode_err(dir, "Looks like your dir is corrupt");
47862306a36Sopenharmony_ci			err = -EINVAL;
47962306a36Sopenharmony_ci			goto out;
48062306a36Sopenharmony_ci		}
48162306a36Sopenharmony_ci
48262306a36Sopenharmony_ci		err = indx_read(&ni->dir, ni, bit << ni->dir.idx2vbn_bits,
48362306a36Sopenharmony_ci				&node);
48462306a36Sopenharmony_ci		if (err)
48562306a36Sopenharmony_ci			goto out;
48662306a36Sopenharmony_ci
48762306a36Sopenharmony_ci		err = ntfs_read_hdr(sbi, ni, &node->index->ihdr,
48862306a36Sopenharmony_ci				    vbo + sbi->record_size, pos, name, ctx);
48962306a36Sopenharmony_ci		if (err)
49062306a36Sopenharmony_ci			goto out;
49162306a36Sopenharmony_ci
49262306a36Sopenharmony_ci		bit += 1;
49362306a36Sopenharmony_ci	}
49462306a36Sopenharmony_ci
49562306a36Sopenharmony_ciout:
49662306a36Sopenharmony_ci
49762306a36Sopenharmony_ci	__putname(name);
49862306a36Sopenharmony_ci	put_indx_node(node);
49962306a36Sopenharmony_ci
50062306a36Sopenharmony_ci	if (err == -ENOENT) {
50162306a36Sopenharmony_ci		err = 0;
50262306a36Sopenharmony_ci		ctx->pos = pos;
50362306a36Sopenharmony_ci	}
50462306a36Sopenharmony_ci
50562306a36Sopenharmony_ci	return err;
50662306a36Sopenharmony_ci}
50762306a36Sopenharmony_ci
50862306a36Sopenharmony_cistatic int ntfs_dir_count(struct inode *dir, bool *is_empty, size_t *dirs,
50962306a36Sopenharmony_ci			  size_t *files)
51062306a36Sopenharmony_ci{
51162306a36Sopenharmony_ci	int err = 0;
51262306a36Sopenharmony_ci	struct ntfs_inode *ni = ntfs_i(dir);
51362306a36Sopenharmony_ci	struct NTFS_DE *e = NULL;
51462306a36Sopenharmony_ci	struct INDEX_ROOT *root;
51562306a36Sopenharmony_ci	struct INDEX_HDR *hdr;
51662306a36Sopenharmony_ci	const struct ATTR_FILE_NAME *fname;
51762306a36Sopenharmony_ci	u32 e_size, off, end;
51862306a36Sopenharmony_ci	size_t drs = 0, fles = 0, bit = 0;
51962306a36Sopenharmony_ci	struct indx_node *node = NULL;
52062306a36Sopenharmony_ci	size_t max_indx = i_size_read(&ni->vfs_inode) >> ni->dir.index_bits;
52162306a36Sopenharmony_ci
52262306a36Sopenharmony_ci	if (is_empty)
52362306a36Sopenharmony_ci		*is_empty = true;
52462306a36Sopenharmony_ci
52562306a36Sopenharmony_ci	root = indx_get_root(&ni->dir, ni, NULL, NULL);
52662306a36Sopenharmony_ci	if (!root)
52762306a36Sopenharmony_ci		return -EINVAL;
52862306a36Sopenharmony_ci
52962306a36Sopenharmony_ci	hdr = &root->ihdr;
53062306a36Sopenharmony_ci
53162306a36Sopenharmony_ci	for (;;) {
53262306a36Sopenharmony_ci		end = le32_to_cpu(hdr->used);
53362306a36Sopenharmony_ci		off = le32_to_cpu(hdr->de_off);
53462306a36Sopenharmony_ci
53562306a36Sopenharmony_ci		for (; off + sizeof(struct NTFS_DE) <= end; off += e_size) {
53662306a36Sopenharmony_ci			e = Add2Ptr(hdr, off);
53762306a36Sopenharmony_ci			e_size = le16_to_cpu(e->size);
53862306a36Sopenharmony_ci			if (e_size < sizeof(struct NTFS_DE) ||
53962306a36Sopenharmony_ci			    off + e_size > end)
54062306a36Sopenharmony_ci				break;
54162306a36Sopenharmony_ci
54262306a36Sopenharmony_ci			if (de_is_last(e))
54362306a36Sopenharmony_ci				break;
54462306a36Sopenharmony_ci
54562306a36Sopenharmony_ci			fname = de_get_fname(e);
54662306a36Sopenharmony_ci			if (!fname)
54762306a36Sopenharmony_ci				continue;
54862306a36Sopenharmony_ci
54962306a36Sopenharmony_ci			if (fname->type == FILE_NAME_DOS)
55062306a36Sopenharmony_ci				continue;
55162306a36Sopenharmony_ci
55262306a36Sopenharmony_ci			if (is_empty) {
55362306a36Sopenharmony_ci				*is_empty = false;
55462306a36Sopenharmony_ci				if (!dirs && !files)
55562306a36Sopenharmony_ci					goto out;
55662306a36Sopenharmony_ci			}
55762306a36Sopenharmony_ci
55862306a36Sopenharmony_ci			if (fname->dup.fa & FILE_ATTRIBUTE_DIRECTORY)
55962306a36Sopenharmony_ci				drs += 1;
56062306a36Sopenharmony_ci			else
56162306a36Sopenharmony_ci				fles += 1;
56262306a36Sopenharmony_ci		}
56362306a36Sopenharmony_ci
56462306a36Sopenharmony_ci		if (bit >= max_indx)
56562306a36Sopenharmony_ci			goto out;
56662306a36Sopenharmony_ci
56762306a36Sopenharmony_ci		err = indx_used_bit(&ni->dir, ni, &bit);
56862306a36Sopenharmony_ci		if (err)
56962306a36Sopenharmony_ci			goto out;
57062306a36Sopenharmony_ci
57162306a36Sopenharmony_ci		if (bit == MINUS_ONE_T)
57262306a36Sopenharmony_ci			goto out;
57362306a36Sopenharmony_ci
57462306a36Sopenharmony_ci		if (bit >= max_indx)
57562306a36Sopenharmony_ci			goto out;
57662306a36Sopenharmony_ci
57762306a36Sopenharmony_ci		err = indx_read(&ni->dir, ni, bit << ni->dir.idx2vbn_bits,
57862306a36Sopenharmony_ci				&node);
57962306a36Sopenharmony_ci		if (err)
58062306a36Sopenharmony_ci			goto out;
58162306a36Sopenharmony_ci
58262306a36Sopenharmony_ci		hdr = &node->index->ihdr;
58362306a36Sopenharmony_ci		bit += 1;
58462306a36Sopenharmony_ci	}
58562306a36Sopenharmony_ci
58662306a36Sopenharmony_ciout:
58762306a36Sopenharmony_ci	put_indx_node(node);
58862306a36Sopenharmony_ci	if (dirs)
58962306a36Sopenharmony_ci		*dirs = drs;
59062306a36Sopenharmony_ci	if (files)
59162306a36Sopenharmony_ci		*files = fles;
59262306a36Sopenharmony_ci
59362306a36Sopenharmony_ci	return err;
59462306a36Sopenharmony_ci}
59562306a36Sopenharmony_ci
59662306a36Sopenharmony_cibool dir_is_empty(struct inode *dir)
59762306a36Sopenharmony_ci{
59862306a36Sopenharmony_ci	bool is_empty = false;
59962306a36Sopenharmony_ci
60062306a36Sopenharmony_ci	ntfs_dir_count(dir, &is_empty, NULL, NULL);
60162306a36Sopenharmony_ci
60262306a36Sopenharmony_ci	return is_empty;
60362306a36Sopenharmony_ci}
60462306a36Sopenharmony_ci
60562306a36Sopenharmony_ci// clang-format off
60662306a36Sopenharmony_ciconst struct file_operations ntfs_dir_operations = {
60762306a36Sopenharmony_ci	.llseek		= generic_file_llseek,
60862306a36Sopenharmony_ci	.read		= generic_read_dir,
60962306a36Sopenharmony_ci	.iterate_shared	= ntfs_readdir,
61062306a36Sopenharmony_ci	.fsync		= generic_file_fsync,
61162306a36Sopenharmony_ci	.open		= ntfs_file_open,
61262306a36Sopenharmony_ci};
61362306a36Sopenharmony_ci// clang-format on
614