vfs_dir.c 4.5 KB
Newer Older
1 2 3 4 5 6 7 8 9
/*
 * linux/fs/9p/vfs_dir.c
 *
 * This file contains vfs directory ops for the 9P2000 protocol.
 *
 *  Copyright (C) 2004 by Eric Van Hensbergen <ericvh@gmail.com>
 *  Copyright (C) 2002 by Ron Minnich <rminnich@lanl.gov>
 *
 *  This program is free software; you can redistribute it and/or modify
10 11
 *  it under the terms of the GNU General Public License version 2
 *  as published by the Free Software Foundation.
12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with this program; if not, write to:
 *  Free Software Foundation
 *  51 Franklin Street, Fifth Floor
 *  Boston, MA  02111-1301  USA
 *
 */

#include <linux/module.h>
#include <linux/errno.h>
#include <linux/fs.h>
#include <linux/file.h>
#include <linux/stat.h>
#include <linux/string.h>
32
#include <linux/sched.h>
33 34
#include <linux/inet.h>
#include <linux/idr.h>
35
#include <linux/slab.h>
36 37
#include <net/9p/9p.h>
#include <net/9p/client.h>
38 39

#include "v9fs.h"
40
#include "v9fs_vfs.h"
41 42
#include "fid.h"

E
Eric Van Hensbergen 已提交
43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60
/**
 * struct p9_rdir - readdir accounting
 * @mutex: mutex protecting readdir
 * @head: start offset of current dirread buffer
 * @tail: end offset of current dirread buffer
 * @buf: dirread buffer
 *
 * private structure for keeping track of readdir
 * allocated on demand
 */

struct p9_rdir {
	struct mutex mutex;
	int head;
	int tail;
	uint8_t *buf;
};

61 62 63 64 65 66
/**
 * dt_type - return file type
 * @mistat: mistat structure
 *
 */

67
static inline int dt_type(struct p9_wstat *mistat)
68 69 70 71
{
	unsigned long perm = mistat->mode;
	int rettype = DT_REG;

72
	if (perm & P9_DMDIR)
73
		rettype = DT_DIR;
74
	if (perm & P9_DMSYMLINK)
75 76 77 78 79
		rettype = DT_LNK;

	return rettype;
}

80 81 82 83 84 85 86 87 88
static void p9stat_init(struct p9_wstat *stbuf)
{
	stbuf->name  = NULL;
	stbuf->uid   = NULL;
	stbuf->gid   = NULL;
	stbuf->muid  = NULL;
	stbuf->extension = NULL;
}

89 90
/**
 * v9fs_dir_readdir - read a directory
E
Eric Van Hensbergen 已提交
91
 * @filp: opened file structure
92 93 94 95 96 97 98
 * @dirent: directory structure ???
 * @filldir: function to populate directory structure ???
 *
 */

static int v9fs_dir_readdir(struct file *filp, void *dirent, filldir_t filldir)
{
99
	int over;
100
	struct p9_wstat st;
E
Eric Van Hensbergen 已提交
101
	int err = 0;
102
	struct p9_fid *fid;
E
Eric Van Hensbergen 已提交
103
	int buflen;
E
Eric Van Hensbergen 已提交
104 105
	int reclen = 0;
	struct p9_rdir *rdir;
106 107 108 109

	P9_DPRINTK(P9_DEBUG_VFS, "name %s\n", filp->f_path.dentry->d_name.name);
	fid = filp->private_data;

E
Eric Van Hensbergen 已提交
110
	buflen = fid->clnt->msize - P9_IOHDRSZ;
E
Eric Van Hensbergen 已提交
111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133

	/* allocate rdir on demand */
	if (!fid->rdir) {
		rdir = kmalloc(sizeof(struct p9_rdir) + buflen, GFP_KERNEL);

		if (rdir == NULL) {
			err = -ENOMEM;
			goto exit;
		}
		spin_lock(&filp->f_dentry->d_lock);
		if (!fid->rdir) {
			rdir->buf = (uint8_t *)rdir + sizeof(struct p9_rdir);
			mutex_init(&rdir->mutex);
			rdir->head = rdir->tail = 0;
			fid->rdir = (void *) rdir;
			rdir = NULL;
		}
		spin_unlock(&filp->f_dentry->d_lock);
		kfree(rdir);
	}
	rdir = (struct p9_rdir *) fid->rdir;

	err = mutex_lock_interruptible(&rdir->mutex);
134 135
	if (err)
		return err;
E
Eric Van Hensbergen 已提交
136 137 138 139 140 141 142 143 144 145 146
	while (err == 0) {
		if (rdir->tail == rdir->head) {
			err = v9fs_file_readn(filp, rdir->buf, NULL,
							buflen, filp->f_pos);
			if (err <= 0)
				goto unlock_and_exit;

			rdir->head = 0;
			rdir->tail = err;
		}
		while (rdir->head < rdir->tail) {
147
			p9stat_init(&st);
E
Eric Van Hensbergen 已提交
148 149
			err = p9stat_read(rdir->buf + rdir->head,
						buflen - rdir->head, &st,
150
						fid->clnt->proto_version);
151 152
			if (err) {
				P9_DPRINTK(P9_DEBUG_VFS, "returned %d\n", err);
E
Eric Van Hensbergen 已提交
153
				err = -EIO;
154
				p9stat_free(&st);
E
Eric Van Hensbergen 已提交
155
				goto unlock_and_exit;
E
Eric Van Hensbergen 已提交
156
			}
E
Eric Van Hensbergen 已提交
157
			reclen = st.size+2;
E
Eric Van Hensbergen 已提交
158

159
			over = filldir(dirent, st.name, strlen(st.name),
E
Eric Van Hensbergen 已提交
160 161
			    filp->f_pos, v9fs_qid2ino(&st.qid), dt_type(&st));

162
			p9stat_free(&st);
E
Eric Van Hensbergen 已提交
163 164 165

			if (over) {
				err = 0;
E
Eric Van Hensbergen 已提交
166
				goto unlock_and_exit;
E
Eric Van Hensbergen 已提交
167
			}
E
Eric Van Hensbergen 已提交
168 169
			rdir->head += reclen;
			filp->f_pos += reclen;
E
Eric Van Hensbergen 已提交
170
		}
171 172
	}

E
Eric Van Hensbergen 已提交
173 174 175
unlock_and_exit:
	mutex_unlock(&rdir->mutex);
exit:
E
Eric Van Hensbergen 已提交
176
	return err;
177 178
}

179

180 181 182 183 184 185 186 187 188
/**
 * v9fs_dir_release - close a directory
 * @inode: inode of the directory
 * @filp: file pointer to a directory
 *
 */

int v9fs_dir_release(struct inode *inode, struct file *filp)
{
189
	struct p9_fid *fid;
190

191 192 193
	fid = filp->private_data;
	P9_DPRINTK(P9_DEBUG_VFS,
			"inode: %p filp: %p fid: %d\n", inode, filp, fid->fid);
194
	filemap_write_and_wait(inode->i_mapping);
195
	p9_client_clunk(fid);
196 197 198
	return 0;
}

199
const struct file_operations v9fs_dir_operations = {
200
	.read = generic_read_dir,
201
	.llseek = generic_file_llseek,
202 203 204 205
	.readdir = v9fs_dir_readdir,
	.open = v9fs_file_open,
	.release = v9fs_dir_release,
};