dir.c 6.0 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38
/*
 * dir.c
 *
 * PURPOSE
 *  Directory handling routines for the OSTA-UDF(tm) filesystem.
 *
 * COPYRIGHT
 *	This file is distributed under the terms of the GNU General Public
 *	License (GPL). Copies of the GPL can be obtained from:
 *		ftp://prep.ai.mit.edu/pub/gnu/GPL
 *	Each contributing author retains all rights to their own work.
 *
 *  (C) 1998-2004 Ben Fennema
 *
 * HISTORY
 *
 *  10/05/98 dgb  Split directory operations into its own file
 *                Implemented directory reads via do_udf_readdir
 *  10/06/98      Made directory operations work!
 *  11/17/98      Rewrote directory to support ICBTAG_FLAG_AD_LONG
 *  11/25/98 blf  Rewrote directory handling (readdir+lookup) to support reading
 *                across blocks.
 *  12/12/98      Split out the lookup code to namei.c. bulk of directory
 *                code now in directory.c:udf_fileident_read.
 */

#include "udfdecl.h"

#include <linux/string.h>
#include <linux/errno.h>
#include <linux/mm.h>
#include <linux/slab.h>
#include <linux/smp_lock.h>
#include <linux/buffer_head.h>

#include "udf_i.h"
#include "udf_sb.h"

39 40
static int do_udf_readdir(struct inode *dir, struct file *filp,
			  filldir_t filldir, void *dirent)
L
Linus Torvalds 已提交
41 42
{
	struct udf_fileident_bh fibh;
43
	struct fileIdentDesc *fi = NULL;
L
Linus Torvalds 已提交
44 45 46 47 48 49 50 51 52
	struct fileIdentDesc cfi;
	int block, iblock;
	loff_t nf_pos = filp->f_pos - 1;
	int flen;
	char fname[UDF_NAME_LEN];
	char *nameptr;
	uint16_t liu;
	uint8_t lfi;
	loff_t size = (udf_ext0_offset(dir) + dir->i_size) >> 2;
J
Jan Kara 已提交
53 54 55
	struct buffer_head *tmp, *bha[16];
	kernel_lb_addr eloc;
	uint32_t elen;
56
	sector_t offset;
L
Linus Torvalds 已提交
57 58
	int i, num;
	unsigned int dt_type;
59
	struct extent_position epos = { NULL, 0, {0, 0} };
60
	struct udf_inode_info *iinfo;
L
Linus Torvalds 已提交
61 62 63 64 65 66 67

	if (nf_pos >= size)
		return 0;

	if (nf_pos == 0)
		nf_pos = (udf_ext0_offset(dir) >> 2);

68
	fibh.soffset = fibh.eoffset = (nf_pos & ((dir->i_sb->s_blocksize - 1) >> 2)) << 2;
69 70
	iinfo = UDF_I(dir);
	if (iinfo->i_alloc_type == ICBTAG_FLAG_AD_IN_ICB) {
L
Linus Torvalds 已提交
71
		fibh.sbh = fibh.ebh = NULL;
72 73
	} else if (inode_bmap(dir, nf_pos >> (dir->i_sb->s_blocksize_bits - 2),
			      &epos, &eloc, &elen, &offset) == (EXT_RECORDED_ALLOCATED >> 30)) {
L
Linus Torvalds 已提交
74
		block = udf_get_lb_pblock(dir->i_sb, eloc, offset);
75
		if ((++offset << dir->i_sb->s_blocksize_bits) < elen) {
76
			if (iinfo->i_alloc_type == ICBTAG_FLAG_AD_SHORT)
J
Jan Kara 已提交
77
				epos.offset -= sizeof(short_ad);
78
			else if (iinfo->i_alloc_type ==
79
					ICBTAG_FLAG_AD_LONG)
J
Jan Kara 已提交
80
				epos.offset -= sizeof(long_ad);
81
		} else {
L
Linus Torvalds 已提交
82
			offset = 0;
83
		}
L
Linus Torvalds 已提交
84

85
		if (!(fibh.sbh = fibh.ebh = udf_tread(dir->i_sb, block))) {
J
Jan Kara 已提交
86
			brelse(epos.bh);
L
Linus Torvalds 已提交
87 88
			return -EIO;
		}
89 90

		if (!(offset & ((16 >> (dir->i_sb->s_blocksize_bits - 9)) - 1))) {
L
Linus Torvalds 已提交
91
			i = 16 >> (dir->i_sb->s_blocksize_bits - 9);
92
			if (i + offset > (elen >> dir->i_sb->s_blocksize_bits))
93
				i = (elen >> dir->i_sb->s_blocksize_bits) - offset;
94
			for (num = 0; i > 0; i--) {
95
				block = udf_get_lb_pblock(dir->i_sb, eloc, offset + i);
L
Linus Torvalds 已提交
96
				tmp = udf_tgetblk(dir->i_sb, block);
97
				if (tmp && !buffer_uptodate(tmp) && !buffer_locked(tmp))
L
Linus Torvalds 已提交
98 99 100 101
					bha[num++] = tmp;
				else
					brelse(tmp);
			}
102
			if (num) {
L
Linus Torvalds 已提交
103
				ll_rw_block(READA, num, bha);
104
				for (i = 0; i < num; i++)
L
Linus Torvalds 已提交
105 106 107
					brelse(bha[i]);
			}
		}
108
	} else {
J
Jan Kara 已提交
109
		brelse(epos.bh);
L
Linus Torvalds 已提交
110 111 112
		return -ENOENT;
	}

113
	while (nf_pos < size) {
L
Linus Torvalds 已提交
114 115
		filp->f_pos = nf_pos + 1;

116 117 118
		fi = udf_fileident_read(dir, &nf_pos, &fibh, &cfi, &epos, &eloc,
					&elen, &offset);
		if (!fi) {
L
Linus Torvalds 已提交
119
			if (fibh.sbh != fibh.ebh)
J
Jan Kara 已提交
120 121 122
				brelse(fibh.ebh);
			brelse(fibh.sbh);
			brelse(epos.bh);
L
Linus Torvalds 已提交
123 124 125 126 127 128
			return 0;
		}

		liu = le16_to_cpu(cfi.lengthOfImpUse);
		lfi = cfi.lengthFileIdent;

129
		if (fibh.sbh == fibh.ebh) {
L
Linus Torvalds 已提交
130
			nameptr = fi->fileIdent + liu;
131
		} else {
L
Linus Torvalds 已提交
132 133
			int poffset;	/* Unpaded ending offset */

134
			poffset = fibh.soffset + sizeof(struct fileIdentDesc) + liu + lfi;
L
Linus Torvalds 已提交
135

136 137 138
			if (poffset >= lfi) {
				nameptr = (char *)(fibh.ebh->b_data + poffset - lfi);
			} else {
L
Linus Torvalds 已提交
139
				nameptr = fname;
140 141 142 143
				memcpy(nameptr, fi->fileIdent + liu,
				       lfi - poffset);
				memcpy(nameptr + lfi - poffset,
				       fibh.ebh->b_data, poffset);
L
Linus Torvalds 已提交
144 145 146
			}
		}

147 148
		if ((cfi.fileCharacteristics & FID_FILE_CHAR_DELETED) != 0) {
			if (!UDF_QUERY_FLAG(dir->i_sb, UDF_FLAG_UNDELETE))
L
Linus Torvalds 已提交
149 150
				continue;
		}
151 152 153

		if ((cfi.fileCharacteristics & FID_FILE_CHAR_HIDDEN) != 0) {
			if (!UDF_QUERY_FLAG(dir->i_sb, UDF_FLAG_UNHIDE))
L
Linus Torvalds 已提交
154 155 156
				continue;
		}

157
		if (cfi.fileCharacteristics & FID_FILE_CHAR_PARENT) {
J
Josef Sipek 已提交
158
			iblock = parent_ino(filp->f_path.dentry);
L
Linus Torvalds 已提交
159 160 161
			flen = 2;
			memcpy(fname, "..", flen);
			dt_type = DT_DIR;
162
		} else {
L
Linus Torvalds 已提交
163 164 165 166 167 168 169
			kernel_lb_addr tloc = lelb_to_cpu(cfi.icb.extLocation);

			iblock = udf_get_lb_pblock(dir->i_sb, tloc, 0);
			flen = udf_get_filename(dir->i_sb, nameptr, fname, lfi);
			dt_type = DT_UNKNOWN;
		}

170
		if (flen) {
171
			if (filldir(dirent, fname, flen, filp->f_pos, iblock, dt_type) < 0) {
L
Linus Torvalds 已提交
172
				if (fibh.sbh != fibh.ebh)
J
Jan Kara 已提交
173 174 175
					brelse(fibh.ebh);
				brelse(fibh.sbh);
				brelse(epos.bh);
176
	 			return 0;
L
Linus Torvalds 已提交
177 178
			}
		}
179
	} /* end while */
L
Linus Torvalds 已提交
180 181 182 183

	filp->f_pos = nf_pos + 1;

	if (fibh.sbh != fibh.ebh)
J
Jan Kara 已提交
184 185 186
		brelse(fibh.ebh);
	brelse(fibh.sbh);
	brelse(epos.bh);
L
Linus Torvalds 已提交
187 188 189

	return 0;
}
190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243

/*
 * udf_readdir
 *
 * PURPOSE
 *	Read a directory entry.
 *
 * DESCRIPTION
 *	Optional - sys_getdents() will return -ENOTDIR if this routine is not
 *	available.
 *
 *	Refer to sys_getdents() in fs/readdir.c
 *	sys_getdents() -> .
 *
 * PRE-CONDITIONS
 *	filp			Pointer to directory file.
 *	buf			Pointer to directory entry buffer.
 *	filldir			Pointer to filldir function.
 *
 * POST-CONDITIONS
 *	<return>		>=0 on success.
 *
 * HISTORY
 *	July 1, 1997 - Andrew E. Mileski
 *	Written, tested, and released.
 */

static int udf_readdir(struct file *filp, void *dirent, filldir_t filldir)
{
	struct inode *dir = filp->f_path.dentry->d_inode;
	int result;

	lock_kernel();

	if (filp->f_pos == 0) {
		if (filldir(dirent, ".", 1, filp->f_pos, dir->i_ino, DT_DIR) < 0) {
			unlock_kernel();
			return 0;
		}
		filp->f_pos++;
	}

	result = do_udf_readdir(dir, filp, filldir, dirent);
	unlock_kernel();
 	return result;
}

/* readdir and lookup functions */
const struct file_operations udf_dir_operations = {
	.read			= generic_read_dir,
	.readdir		= udf_readdir,
	.ioctl			= udf_ioctl,
	.fsync			= udf_fsync_file,
};