dir.c 5.5 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
	struct fileIdentDesc cfi;
	int block, iblock;
J
Jan Kara 已提交
46
	loff_t nf_pos = (filp->f_pos - 1) << 2;
L
Linus Torvalds 已提交
47 48 49 50 51
	int flen;
	char fname[UDF_NAME_LEN];
	char *nameptr;
	uint16_t liu;
	uint8_t lfi;
J
Jan Kara 已提交
52
	loff_t size = udf_ext0_offset(dir) + dir->i_size;
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

	if (nf_pos >= size)
		return 0;

	if (nf_pos == 0)
J
Jan Kara 已提交
66
		nf_pos = udf_ext0_offset(dir);
L
Linus Torvalds 已提交
67

J
Jan Kara 已提交
68
	fibh.soffset = fibh.eoffset = nf_pos & (dir->i_sb->s_blocksize - 1);
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;
J
Jan Kara 已提交
72
	} else if (inode_bmap(dir, nf_pos >> dir->i_sb->s_blocksize_bits,
73
			      &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) {
J
Jan Kara 已提交
114
		filp->f_pos = (nf_pos >> 2) + 1;
L
Linus Torvalds 已提交
115

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

J
Jan Kara 已提交
181
	filp->f_pos = (nf_pos >> 2) + 1;
L
Linus Torvalds 已提交
182 183

	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

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,
};