orangefs: rewrite readdir to fix several bugs
In the past, readdir assumed that the user buffer will be large enough that all entries from the server will fit. If this was not true, entries would be skipped. Since it works now, request 512 entries rather than 96 per server operation. Signed-off-by: Martin Brandenburg <martin@omnibond.com> Signed-off-by: Mike Marshall <hubcap@omnibond.com>
This commit is contained in:
Родитель
17930b252c
Коммит
382f4581e6
|
@ -1,7 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* (C) 2001 Clemson University and The University of Chicago
|
* Copyright 2017 Omnibond Systems, L.L.C.
|
||||||
*
|
|
||||||
* See COPYING in top-level directory.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "protocol.h"
|
#include "protocol.h"
|
||||||
|
@ -9,388 +7,244 @@
|
||||||
#include "orangefs-bufmap.h"
|
#include "orangefs-bufmap.h"
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* decode routine used by kmod to deal with the blob sent from
|
* There can be up to 512 directory entries. Each entry is encoded as
|
||||||
* userspace for readdirs. The blob contains zero or more of these
|
* follows:
|
||||||
* sub-blobs:
|
* 4 bytes: string size (n)
|
||||||
* __u32 - represents length of the character string that follows.
|
* n bytes: string
|
||||||
* string - between 1 and ORANGEFS_NAME_MAX bytes long.
|
* 1 byte: trailing zero
|
||||||
* padding - (if needed) to cause the __u32 plus the string to be
|
* padding to 8 bytes
|
||||||
* eight byte aligned.
|
* 16 bytes: khandle
|
||||||
* khandle - sizeof(khandle) bytes.
|
* padding to 8 bytes
|
||||||
*/
|
*/
|
||||||
static long decode_dirents(char *ptr, size_t size,
|
#define MAX_DIRECTORY ((4 + 257 + 3 + 16)*512)
|
||||||
struct orangefs_readdir_response_s *readdir)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
struct orangefs_readdir_response_s *rd =
|
|
||||||
(struct orangefs_readdir_response_s *) ptr;
|
|
||||||
char *buf = ptr;
|
|
||||||
int khandle_size = sizeof(struct orangefs_khandle);
|
|
||||||
size_t offset = offsetof(struct orangefs_readdir_response_s,
|
|
||||||
dirent_array);
|
|
||||||
/* 8 reflects eight byte alignment */
|
|
||||||
int smallest_blob = khandle_size + 8;
|
|
||||||
__u32 len;
|
|
||||||
int aligned_len;
|
|
||||||
int sizeof_u32 = sizeof(__u32);
|
|
||||||
long ret;
|
|
||||||
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG, "%s: size:%zu:\n", __func__, size);
|
struct orangefs_dir {
|
||||||
|
__u64 token;
|
||||||
/* size is = offset on empty dirs, > offset on non-empty dirs... */
|
void *directory;
|
||||||
if (size < offset) {
|
size_t i, len;
|
||||||
gossip_err("%s: size:%zu: offset:%zu:\n",
|
int error;
|
||||||
__func__,
|
};
|
||||||
size,
|
|
||||||
offset);
|
|
||||||
ret = -EINVAL;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((size == offset) && (readdir->orangefs_dirent_outcount != 0)) {
|
|
||||||
gossip_err("%s: size:%zu: dirent_outcount:%d:\n",
|
|
||||||
__func__,
|
|
||||||
size,
|
|
||||||
readdir->orangefs_dirent_outcount);
|
|
||||||
ret = -EINVAL;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
readdir->token = rd->token;
|
|
||||||
readdir->orangefs_dirent_outcount = rd->orangefs_dirent_outcount;
|
|
||||||
readdir->dirent_array = kcalloc(readdir->orangefs_dirent_outcount,
|
|
||||||
sizeof(*readdir->dirent_array),
|
|
||||||
GFP_KERNEL);
|
|
||||||
if (readdir->dirent_array == NULL) {
|
|
||||||
gossip_err("%s: kcalloc failed.\n", __func__);
|
|
||||||
ret = -ENOMEM;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
buf += offset;
|
|
||||||
size -= offset;
|
|
||||||
|
|
||||||
for (i = 0; i < readdir->orangefs_dirent_outcount; i++) {
|
|
||||||
if (size < smallest_blob) {
|
|
||||||
gossip_err("%s: size:%zu: smallest_blob:%d:\n",
|
|
||||||
__func__,
|
|
||||||
size,
|
|
||||||
smallest_blob);
|
|
||||||
ret = -EINVAL;
|
|
||||||
goto free;
|
|
||||||
}
|
|
||||||
|
|
||||||
len = *(__u32 *)buf;
|
|
||||||
if ((len < 1) || (len > ORANGEFS_NAME_MAX)) {
|
|
||||||
gossip_err("%s: len:%d:\n", __func__, len);
|
|
||||||
ret = -EINVAL;
|
|
||||||
goto free;
|
|
||||||
}
|
|
||||||
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
|
||||||
"%s: size:%zu: len:%d:\n",
|
|
||||||
__func__,
|
|
||||||
size,
|
|
||||||
len);
|
|
||||||
|
|
||||||
readdir->dirent_array[i].d_name = buf + sizeof_u32;
|
|
||||||
readdir->dirent_array[i].d_length = len;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Calculate "aligned" length of this string and its
|
|
||||||
* associated __u32 descriptor.
|
|
||||||
*/
|
|
||||||
aligned_len = ((sizeof_u32 + len + 1) + 7) & ~7;
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
|
||||||
"%s: aligned_len:%d:\n",
|
|
||||||
__func__,
|
|
||||||
aligned_len);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* The end of the blob should coincide with the end
|
|
||||||
* of the last sub-blob.
|
|
||||||
*/
|
|
||||||
if (size < aligned_len + khandle_size) {
|
|
||||||
gossip_err("%s: ran off the end of the blob.\n",
|
|
||||||
__func__);
|
|
||||||
ret = -EINVAL;
|
|
||||||
goto free;
|
|
||||||
}
|
|
||||||
size -= aligned_len + khandle_size;
|
|
||||||
|
|
||||||
buf += aligned_len;
|
|
||||||
|
|
||||||
readdir->dirent_array[i].khandle =
|
|
||||||
*(struct orangefs_khandle *) buf;
|
|
||||||
buf += khandle_size;
|
|
||||||
}
|
|
||||||
ret = buf - ptr;
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG, "%s: returning:%ld:\n", __func__, ret);
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
free:
|
|
||||||
kfree(readdir->dirent_array);
|
|
||||||
readdir->dirent_array = NULL;
|
|
||||||
|
|
||||||
out:
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Read directory entries from an instance of an open directory.
|
* The userspace component sends several directory entries of the
|
||||||
|
* following format. The first four bytes are the string length not
|
||||||
|
* including a trailing zero byte. This is followed by the string and a
|
||||||
|
* trailing zero padded to the next four byte boundry. This is followed
|
||||||
|
* by the sixteen byte khandle padded to the next eight byte boundry.
|
||||||
|
*
|
||||||
|
* The trailer_buf starts with a struct orangefs_readdir_response_s
|
||||||
|
* which must be skipped to get to the directory data.
|
||||||
*/
|
*/
|
||||||
static int orangefs_readdir(struct file *file, struct dir_context *ctx)
|
|
||||||
|
static int orangefs_dir_more(struct orangefs_inode_s *oi,
|
||||||
|
struct orangefs_dir *od, struct dentry *dentry)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
const size_t offset =
|
||||||
int buffer_index;
|
sizeof(struct orangefs_readdir_response_s);
|
||||||
/*
|
struct orangefs_readdir_response_s *resp;
|
||||||
* ptoken supports Orangefs' distributed directory logic, added
|
struct orangefs_kernel_op_s *op;
|
||||||
* in 2.9.2.
|
int bufi, r;
|
||||||
*/
|
|
||||||
__u64 *ptoken = file->private_data;
|
|
||||||
__u64 pos = 0;
|
|
||||||
ino_t ino = 0;
|
|
||||||
struct dentry *dentry = file->f_path.dentry;
|
|
||||||
struct orangefs_kernel_op_s *new_op = NULL;
|
|
||||||
struct orangefs_inode_s *orangefs_inode = ORANGEFS_I(dentry->d_inode);
|
|
||||||
struct orangefs_readdir_response_s readdir_response;
|
|
||||||
void *dents_buf;
|
|
||||||
int i = 0;
|
|
||||||
int len = 0;
|
|
||||||
ino_t current_ino = 0;
|
|
||||||
char *current_entry = NULL;
|
|
||||||
long bytes_decoded;
|
|
||||||
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
op = op_alloc(ORANGEFS_VFS_OP_READDIR);
|
||||||
"%s: ctx->pos:%lld, ptoken = %llu\n",
|
if (!op) {
|
||||||
__func__,
|
od->error = -ENOMEM;
|
||||||
lld(ctx->pos),
|
return -ENOMEM;
|
||||||
llu(*ptoken));
|
|
||||||
|
|
||||||
pos = (__u64) ctx->pos;
|
|
||||||
|
|
||||||
/* are we done? */
|
|
||||||
if (pos == ORANGEFS_READDIR_END) {
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
|
||||||
"Skipping to termination path\n");
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
|
||||||
"orangefs_readdir called on %pd (pos=%llu)\n",
|
|
||||||
dentry, llu(pos));
|
|
||||||
|
|
||||||
memset(&readdir_response, 0, sizeof(readdir_response));
|
|
||||||
|
|
||||||
new_op = op_alloc(ORANGEFS_VFS_OP_READDIR);
|
|
||||||
if (!new_op)
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Only the indices are shared. No memory is actually shared, but the
|
* Despite the badly named field, readdir does not use shared
|
||||||
* mechanism is used.
|
* memory. However, there are a limited number of readdir
|
||||||
|
* slots, which must be allocated here. This flag simply tells
|
||||||
|
* the op scheduler to return the op here for retry.
|
||||||
*/
|
*/
|
||||||
new_op->uses_shared_memory = 1;
|
op->uses_shared_memory = 1;
|
||||||
new_op->upcall.req.readdir.refn = orangefs_inode->refn;
|
op->upcall.req.readdir.refn = oi->refn;
|
||||||
new_op->upcall.req.readdir.max_dirent_count =
|
op->upcall.req.readdir.token = od->token;
|
||||||
|
op->upcall.req.readdir.max_dirent_count =
|
||||||
ORANGEFS_MAX_DIRENT_COUNT_READDIR;
|
ORANGEFS_MAX_DIRENT_COUNT_READDIR;
|
||||||
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
again:
|
||||||
"%s: upcall.req.readdir.refn.khandle: %pU\n",
|
bufi = orangefs_readdir_index_get();
|
||||||
__func__,
|
if (bufi < 0) {
|
||||||
&new_op->upcall.req.readdir.refn.khandle);
|
op_release(op);
|
||||||
|
od->error = bufi;
|
||||||
new_op->upcall.req.readdir.token = *ptoken;
|
return bufi;
|
||||||
|
|
||||||
get_new_buffer_index:
|
|
||||||
buffer_index = orangefs_readdir_index_get();
|
|
||||||
if (buffer_index < 0) {
|
|
||||||
ret = buffer_index;
|
|
||||||
gossip_lerr("orangefs_readdir: orangefs_readdir_index_get() failure (%d)\n",
|
|
||||||
ret);
|
|
||||||
goto out_free_op;
|
|
||||||
}
|
}
|
||||||
new_op->upcall.req.readdir.buf_index = buffer_index;
|
|
||||||
|
|
||||||
ret = service_operation(new_op,
|
op->upcall.req.readdir.buf_index = bufi;
|
||||||
"orangefs_readdir",
|
|
||||||
|
r = service_operation(op, "orangefs_readdir",
|
||||||
get_interruptible_flag(dentry->d_inode));
|
get_interruptible_flag(dentry->d_inode));
|
||||||
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
orangefs_readdir_index_put(bufi);
|
||||||
"Readdir downcall status is %d. ret:%d\n",
|
|
||||||
new_op->downcall.status,
|
|
||||||
ret);
|
|
||||||
|
|
||||||
orangefs_readdir_index_put(buffer_index);
|
if (op_state_purged(op)) {
|
||||||
|
if (r == -EAGAIN) {
|
||||||
if (ret == -EAGAIN && op_state_purged(new_op)) {
|
vfree(op->downcall.trailer_buf);
|
||||||
/* Client-core indices are invalid after it restarted. */
|
goto again;
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
} else if (r == -EIO) {
|
||||||
"%s: Getting new buffer_index for retry of readdir..\n",
|
vfree(op->downcall.trailer_buf);
|
||||||
__func__);
|
op_release(op);
|
||||||
goto get_new_buffer_index;
|
od->error = r;
|
||||||
|
return r;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret == -EIO && op_state_purged(new_op)) {
|
if (r < 0) {
|
||||||
gossip_err("%s: Client is down. Aborting readdir call.\n",
|
vfree(op->downcall.trailer_buf);
|
||||||
__func__);
|
op_release(op);
|
||||||
goto out_free_op;
|
od->error = r;
|
||||||
|
return r;
|
||||||
|
} else if (op->downcall.status) {
|
||||||
|
vfree(op->downcall.trailer_buf);
|
||||||
|
op_release(op);
|
||||||
|
od->error = op->downcall.status;
|
||||||
|
return op->downcall.status;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret < 0 || new_op->downcall.status != 0) {
|
resp = (struct orangefs_readdir_response_s *)
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
op->downcall.trailer_buf;
|
||||||
"Readdir request failed. Status:%d\n",
|
od->token = resp->token;
|
||||||
new_op->downcall.status);
|
|
||||||
if (ret >= 0)
|
if (od->len + op->downcall.trailer_size - offset <=
|
||||||
ret = new_op->downcall.status;
|
MAX_DIRECTORY) {
|
||||||
goto out_free_op;
|
memcpy(od->directory + od->len,
|
||||||
|
op->downcall.trailer_buf + offset,
|
||||||
|
op->downcall.trailer_size - offset);
|
||||||
|
od->len += op->downcall.trailer_size - offset;
|
||||||
|
} else {
|
||||||
|
/* This limit was chosen based on protocol limits. */
|
||||||
|
gossip_err("orangefs_dir_more: userspace sent too much data\n");
|
||||||
|
vfree(op->downcall.trailer_buf);
|
||||||
|
op_release(op);
|
||||||
|
od->error = -EIO;
|
||||||
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
dents_buf = new_op->downcall.trailer_buf;
|
vfree(op->downcall.trailer_buf);
|
||||||
if (dents_buf == NULL) {
|
op_release(op);
|
||||||
gossip_err("Invalid NULL buffer in readdir response\n");
|
return 0;
|
||||||
ret = -ENOMEM;
|
}
|
||||||
goto out_free_op;
|
|
||||||
}
|
|
||||||
|
|
||||||
bytes_decoded = decode_dirents(dents_buf, new_op->downcall.trailer_size,
|
|
||||||
&readdir_response);
|
|
||||||
if (bytes_decoded < 0) {
|
|
||||||
ret = bytes_decoded;
|
|
||||||
gossip_err("Could not decode readdir from buffer %d\n", ret);
|
|
||||||
goto out_vfree;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bytes_decoded != new_op->downcall.trailer_size) {
|
|
||||||
gossip_err("orangefs_readdir: # bytes decoded (%ld) "
|
|
||||||
"!= trailer size (%ld)\n",
|
|
||||||
bytes_decoded,
|
|
||||||
(long)new_op->downcall.trailer_size);
|
|
||||||
ret = -EINVAL;
|
|
||||||
goto out_destroy_handle;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
static int orangefs_dir_fill(struct orangefs_inode_s *oi,
|
||||||
|
struct orangefs_dir *od, struct dentry *dentry,
|
||||||
|
struct dir_context *ctx)
|
||||||
|
{
|
||||||
|
struct orangefs_khandle *khandle;
|
||||||
|
__u32 *len, padlen;
|
||||||
|
char *s;
|
||||||
|
while (od->i < od->len) {
|
||||||
|
if (od->len < od->i + sizeof *len)
|
||||||
|
goto eio;
|
||||||
|
len = od->directory + od->i;
|
||||||
/*
|
/*
|
||||||
* orangefs doesn't actually store dot and dot-dot, but
|
* len is the size of the string itself. padlen is the
|
||||||
* we need to have them represented.
|
* total size of the encoded string.
|
||||||
*/
|
*/
|
||||||
if (pos == 0) {
|
padlen = (sizeof *len + *len + 1) +
|
||||||
ino = get_ino_from_khandle(dentry->d_inode);
|
(4 - (sizeof *len + *len + 1)%8)%8;
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
if (od->len < od->i + padlen + sizeof *khandle)
|
||||||
"%s: calling dir_emit of \".\" with pos = %llu\n",
|
goto eio;
|
||||||
__func__,
|
s = od->directory + od->i + sizeof *len;
|
||||||
llu(pos));
|
if (s[*len] != 0)
|
||||||
ret = dir_emit(ctx, ".", 1, ino, DT_DIR);
|
goto eio;
|
||||||
pos += 1;
|
khandle = od->directory + od->i + padlen;
|
||||||
|
|
||||||
|
if (!dir_emit(ctx, s, *len,
|
||||||
|
orangefs_khandle_to_ino(khandle), DT_UNKNOWN))
|
||||||
|
return 0;
|
||||||
|
od->i += padlen + sizeof *khandle;
|
||||||
|
od->i = od->i + (8 - od->i%8)%8;
|
||||||
|
ctx->pos = 2 + od->i;
|
||||||
}
|
}
|
||||||
|
BUG_ON(od->i > od->len);
|
||||||
|
return 0;
|
||||||
|
eio:
|
||||||
|
gossip_err("orangefs_dir_fill: userspace returns corrupt data\n");
|
||||||
|
od->error = -EIO;
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
|
||||||
if (pos == 1) {
|
static int orangefs_dir_iterate(struct file *file,
|
||||||
ino = get_parent_ino_from_dentry(dentry);
|
struct dir_context *ctx)
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
{
|
||||||
"%s: calling dir_emit of \"..\" with pos = %llu\n",
|
struct orangefs_inode_s *oi;
|
||||||
__func__,
|
struct orangefs_dir *od;
|
||||||
llu(pos));
|
struct dentry *dentry;
|
||||||
ret = dir_emit(ctx, "..", 2, ino, DT_DIR);
|
int r;
|
||||||
pos += 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
dentry = file->f_path.dentry;
|
||||||
* we stored ORANGEFS_ITERATE_NEXT in ctx->pos last time around
|
oi = ORANGEFS_I(dentry->d_inode);
|
||||||
* to prevent "finding" dot and dot-dot on any iteration
|
od = file->private_data;
|
||||||
* other than the first.
|
|
||||||
*/
|
|
||||||
if (ctx->pos == ORANGEFS_ITERATE_NEXT)
|
|
||||||
ctx->pos = 0;
|
|
||||||
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
if (od->error)
|
||||||
"%s: dirent_outcount:%d:\n",
|
return od->error;
|
||||||
__func__,
|
|
||||||
readdir_response.orangefs_dirent_outcount);
|
|
||||||
for (i = ctx->pos;
|
|
||||||
i < readdir_response.orangefs_dirent_outcount;
|
|
||||||
i++) {
|
|
||||||
len = readdir_response.dirent_array[i].d_length;
|
|
||||||
current_entry = readdir_response.dirent_array[i].d_name;
|
|
||||||
current_ino = orangefs_khandle_to_ino(
|
|
||||||
&readdir_response.dirent_array[i].khandle);
|
|
||||||
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
if (ctx->pos == 0) {
|
||||||
"calling dir_emit for %s with len %d"
|
if (!dir_emit_dot(file, ctx))
|
||||||
", ctx->pos %ld\n",
|
return 0;
|
||||||
current_entry,
|
ctx->pos++;
|
||||||
len,
|
}
|
||||||
(unsigned long)ctx->pos);
|
if (ctx->pos == 1) {
|
||||||
/*
|
if (!dir_emit_dotdot(file, ctx))
|
||||||
* type is unknown. We don't return object type
|
return 0;
|
||||||
* in the dirent_array. This leaves getdents
|
|
||||||
* clueless about type.
|
|
||||||
*/
|
|
||||||
ret =
|
|
||||||
dir_emit(ctx, current_entry, len, current_ino, DT_UNKNOWN);
|
|
||||||
if (!ret)
|
|
||||||
break;
|
|
||||||
ctx->pos++;
|
ctx->pos++;
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
|
||||||
"%s: ctx->pos:%lld\n",
|
|
||||||
__func__,
|
|
||||||
lld(ctx->pos));
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
r = 0;
|
||||||
* we ran all the way through the last batch, set up for
|
|
||||||
* getting another batch...
|
if (od->i < od->len) {
|
||||||
*/
|
r = orangefs_dir_fill(oi, od, dentry, ctx);
|
||||||
if (ret) {
|
if (r)
|
||||||
*ptoken = readdir_response.token;
|
return r;
|
||||||
ctx->pos = ORANGEFS_ITERATE_NEXT;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
if (od->token != ORANGEFS_READDIR_END) {
|
||||||
* Did we hit the end of the directory?
|
r = orangefs_dir_more(oi, od, dentry);
|
||||||
*/
|
if (r)
|
||||||
if (readdir_response.token == ORANGEFS_READDIR_END) {
|
return r;
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG,
|
r = orangefs_dir_fill(oi, od, dentry, ctx);
|
||||||
"End of dir detected; setting ctx->pos to ORANGEFS_READDIR_END.\n");
|
|
||||||
ctx->pos = ORANGEFS_READDIR_END;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
out_destroy_handle:
|
return r;
|
||||||
/* kfree(NULL) is safe */
|
|
||||||
kfree(readdir_response.dirent_array);
|
|
||||||
out_vfree:
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG, "vfree %p\n", dents_buf);
|
|
||||||
vfree(dents_buf);
|
|
||||||
out_free_op:
|
|
||||||
op_release(new_op);
|
|
||||||
gossip_debug(GOSSIP_DIR_DEBUG, "orangefs_readdir returning %d\n", ret);
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int orangefs_dir_open(struct inode *inode, struct file *file)
|
static int orangefs_dir_open(struct inode *inode, struct file *file)
|
||||||
{
|
{
|
||||||
__u64 *ptoken;
|
struct orangefs_dir *od;
|
||||||
|
file->private_data = kmalloc(sizeof(struct orangefs_dir),
|
||||||
file->private_data = kmalloc(sizeof(__u64), GFP_KERNEL);
|
GFP_KERNEL);
|
||||||
if (!file->private_data)
|
if (!file->private_data)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
od = file->private_data;
|
||||||
ptoken = file->private_data;
|
od->token = ORANGEFS_READDIR_START;
|
||||||
*ptoken = ORANGEFS_READDIR_START;
|
/*
|
||||||
|
* XXX: It seems wasteful to allocate such a large buffer for
|
||||||
|
* each request. Most will be much smaller.
|
||||||
|
*/
|
||||||
|
od->directory = alloc_pages_exact(MAX_DIRECTORY, GFP_KERNEL);
|
||||||
|
if (!od->directory) {
|
||||||
|
kfree(file->private_data);
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
od->i = 0;
|
||||||
|
od->len = 0;
|
||||||
|
od->error = 0;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int orangefs_dir_release(struct inode *inode, struct file *file)
|
static int orangefs_dir_release(struct inode *inode, struct file *file)
|
||||||
{
|
{
|
||||||
|
struct orangefs_dir *od = file->private_data;
|
||||||
orangefs_flush_inode(inode);
|
orangefs_flush_inode(inode);
|
||||||
kfree(file->private_data);
|
free_pages_exact(od->directory, MAX_DIRECTORY);
|
||||||
|
kfree(od);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** ORANGEFS implementation of VFS directory operations */
|
|
||||||
const struct file_operations orangefs_dir_operations = {
|
const struct file_operations orangefs_dir_operations = {
|
||||||
.read = generic_read_dir,
|
.read = generic_read_dir,
|
||||||
.iterate = orangefs_readdir,
|
.iterate = orangefs_dir_iterate,
|
||||||
.open = orangefs_dir_open,
|
.open = orangefs_dir_open,
|
||||||
.release = orangefs_dir_release,
|
.release = orangefs_dir_release
|
||||||
};
|
};
|
||||||
|
|
|
@ -40,16 +40,6 @@ struct orangefs_mkdir_response {
|
||||||
struct orangefs_object_kref refn;
|
struct orangefs_object_kref refn;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
|
||||||
* duplication of some system interface structures so that I don't have
|
|
||||||
* to allocate extra memory
|
|
||||||
*/
|
|
||||||
struct orangefs_dirent {
|
|
||||||
char *d_name;
|
|
||||||
int d_length;
|
|
||||||
struct orangefs_khandle khandle;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct orangefs_statfs_response {
|
struct orangefs_statfs_response {
|
||||||
__s64 block_size;
|
__s64 block_size;
|
||||||
__s64 blocks_total;
|
__s64 blocks_total;
|
||||||
|
@ -131,12 +121,16 @@ struct orangefs_downcall_s {
|
||||||
} resp;
|
} resp;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The readdir response comes in the trailer. It is followed by the
|
||||||
|
* directory entries as described in dir.c.
|
||||||
|
*/
|
||||||
|
|
||||||
struct orangefs_readdir_response_s {
|
struct orangefs_readdir_response_s {
|
||||||
__u64 token;
|
__u64 token;
|
||||||
__u64 directory_version;
|
__u64 directory_version;
|
||||||
__u32 __pad2;
|
__u32 __pad2;
|
||||||
__u32 orangefs_dirent_outcount;
|
__u32 orangefs_dirent_outcount;
|
||||||
struct orangefs_dirent *dirent_array;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* __DOWNCALL_H */
|
#endif /* __DOWNCALL_H */
|
||||||
|
|
|
@ -52,12 +52,7 @@
|
||||||
*/
|
*/
|
||||||
#define ORANGEFS_MAX_DEBUG_STRING_LEN 0x00000800
|
#define ORANGEFS_MAX_DEBUG_STRING_LEN 0x00000800
|
||||||
|
|
||||||
/*
|
#define ORANGEFS_MAX_DIRENT_COUNT_READDIR 512
|
||||||
* The maximum number of directory entries in a single request is 96.
|
|
||||||
* XXX: Why can this not be higher. The client-side code can handle up to 512.
|
|
||||||
* XXX: What happens if we expect more than the client can return?
|
|
||||||
*/
|
|
||||||
#define ORANGEFS_MAX_DIRENT_COUNT_READDIR 96
|
|
||||||
|
|
||||||
#include "upcall.h"
|
#include "upcall.h"
|
||||||
#include "downcall.h"
|
#include "downcall.h"
|
||||||
|
|
Загрузка…
Ссылка в новой задаче